{"version":3,"sources":["viewer.js","../Client/viewer/models/type-extensions.ts","../node_modules/tslib/tslib.es6.js","../Client/viewer/models/classes/token.ts","../Client/viewer/models/classes/decorators/model-utils.ts","../Client/viewer/models/interfaces/tracked-class.ts","../Client/viewer/models/services/sync-service.ts","../node_modules/rxjs/node_modules/tslib/tslib.es6.js","../node_modules/rxjs/src/internal/util/isFunction.ts","../node_modules/rxjs/src/internal/config.ts","../node_modules/rxjs/src/internal/util/hostReportError.ts","../node_modules/rxjs/src/internal/Observer.ts","../node_modules/rxjs/src/internal/util/isObject.ts","../node_modules/rxjs/src/internal/util/UnsubscriptionError.ts","../node_modules/rxjs/src/internal/Subscription.ts","../node_modules/rxjs/src/internal/symbol/rxSubscriber.ts","../node_modules/rxjs/src/internal/Subscriber.ts","../node_modules/rxjs/src/internal/util/identity.ts","../node_modules/rxjs/src/internal/util/pipe.ts","../node_modules/rxjs/src/internal/Observable.ts","../node_modules/rxjs/src/internal/util/toSubscriber.ts","../node_modules/rxjs/src/internal/util/canReportError.ts","../node_modules/rxjs/src/internal/util/ObjectUnsubscribedError.ts","../node_modules/rxjs/src/internal/SubjectSubscription.ts","../node_modules/rxjs/src/internal/Subject.ts","../node_modules/rxjs/src/internal/operators/refCount.ts","../node_modules/rxjs/src/internal/observable/ConnectableObservable.ts","../node_modules/rxjs/src/internal/BehaviorSubject.ts","../node_modules/rxjs/src/internal/scheduler/AsyncAction.ts","../node_modules/rxjs/src/internal/scheduler/Action.ts","../node_modules/rxjs/src/internal/scheduler/QueueAction.ts","../node_modules/rxjs/src/internal/Scheduler.ts","../node_modules/rxjs/src/internal/scheduler/AsyncScheduler.ts","../node_modules/rxjs/src/internal/scheduler/QueueScheduler.ts","../node_modules/rxjs/src/internal/scheduler/queue.ts","../node_modules/rxjs/src/internal/observable/empty.ts","../node_modules/rxjs/src/internal/util/isScheduler.ts","../node_modules/rxjs/src/internal/util/subscribeToArray.ts","../node_modules/rxjs/src/internal/scheduled/scheduleArray.ts","../node_modules/rxjs/src/internal/observable/fromArray.ts","../node_modules/rxjs/src/internal/observable/of.ts","../node_modules/rxjs/src/internal/observable/throwError.ts","../node_modules/rxjs/src/internal/Notification.ts","../node_modules/rxjs/src/internal/operators/observeOn.ts","../node_modules/rxjs/src/internal/ReplaySubject.ts","../node_modules/rxjs/src/internal/scheduler/async.ts","../node_modules/rxjs/src/internal/util/ArgumentOutOfRangeError.ts","../node_modules/rxjs/src/internal/util/EmptyError.ts","../node_modules/rxjs/src/internal/operators/map.ts","../node_modules/rxjs/src/internal/OuterSubscriber.ts","../node_modules/rxjs/src/internal/InnerSubscriber.ts","../node_modules/rxjs/src/internal/symbol/iterator.ts","../node_modules/rxjs/src/internal/util/isPromise.ts","../node_modules/rxjs/src/internal/util/subscribeTo.ts","../node_modules/rxjs/src/internal/util/subscribeToObservable.ts","../node_modules/rxjs/src/internal/util/subscribeToPromise.ts","../node_modules/rxjs/src/internal/util/subscribeToIterable.ts","../node_modules/rxjs/src/internal/util/subscribeToResult.ts","../node_modules/rxjs/src/internal/observable/combineLatest.ts","../node_modules/rxjs/src/internal/scheduled/scheduled.ts","../node_modules/rxjs/src/internal/util/isInteropObservable.ts","../node_modules/rxjs/src/internal/scheduled/scheduleObservable.ts","../node_modules/rxjs/src/internal/scheduled/schedulePromise.ts","../node_modules/rxjs/src/internal/util/isIterable.ts","../node_modules/rxjs/src/internal/scheduled/scheduleIterable.ts","../node_modules/rxjs/src/internal/observable/from.ts","../node_modules/rxjs/src/internal/innerSubscribe.ts","../node_modules/rxjs/src/internal/operators/mergeMap.ts","../node_modules/rxjs/src/internal/operators/mergeAll.ts","../node_modules/rxjs/src/internal/operators/concatAll.ts","../node_modules/rxjs/src/internal/observable/concat.ts","../node_modules/rxjs/src/internal/observable/merge.ts","../node_modules/rxjs/src/internal/operators/filter.ts","../node_modules/rxjs/src/internal/operators/debounceTime.ts","../node_modules/rxjs/src/internal/operators/defaultIfEmpty.ts","../node_modules/rxjs/src/internal/operators/distinctUntilChanged.ts","../node_modules/rxjs/src/internal/operators/throwIfEmpty.ts","../node_modules/rxjs/src/internal/operators/take.ts","../node_modules/rxjs/src/internal/operators/finalize.ts","../node_modules/rxjs/src/internal/operators/multicast.ts","../node_modules/rxjs/src/internal/operators/share.ts","../node_modules/rxjs/src/internal/operators/shareReplay.ts","../node_modules/rxjs/src/internal/operators/switchMap.ts","../node_modules/rxjs/src/internal/operators/takeUntil.ts","../node_modules/rxjs/src/internal/operators/takeWhile.ts","../node_modules/rxjs/src/internal/operators/throttle.ts","../node_modules/rxjs/src/internal/operators/throttleTime.ts","../Client/viewer/models/classes/utilities.ts","../Client/viewer/models/classes/history.ts","../Client/viewer/models/classes/decorators/change-tracker.ts","../Client/viewer/models/classes/decorators/change-tracker-node.ts","../Client/viewer/models/classes/primitives/primitive-utilities.ts","../Client/viewer/models/classes/decorators/kb3d-decorator.ts","../Client/viewer/models/classes/decorators/enumerable.ts","../Client/viewer/models/classes/decorators/inject-decorator.ts","../Client/viewer/models/classes/decorators/service-decorator.ts","../Client/viewer/models/interfaces/settings.ts","../Client/viewer/models/classes/kb3d-object.ts","../Client/viewer/models/classes/features/feature.ts","../Client/viewer/models/classes/features/base-connector.ts","../Client/viewer/models/classes/default-values.ts","../Client/viewer/models/classes/materials/material-mapping.ts","../Client/viewer/models/classes/materials/material-node.ts","../node_modules/@babylonjs/core/tslib.es6.js","../dev/core/src/Misc/andOrNotEvaluator.ts","../dev/core/src/Misc/tags.ts","../dev/core/src/Maths/math.scalar.ts","../dev/core/src/Maths/math.constants.ts","../dev/core/src/Misc/arrayTools.ts","../dev/core/src/Misc/typeStore.ts","../dev/core/src/Engines/performanceConfigurator.ts","../dev/core/src/Misc/observable.ts","../dev/core/src/Engines/engineStore.ts","../dev/core/src/Maths/math.vector.ts","../dev/core/src/Misc/devTools.ts","../dev/core/src/Maths/math.color.ts","../dev/core/src/Misc/decorators.ts","../dev/core/src/node.ts","../dev/core/src/Misc/domManagement.ts","../dev/core/src/Misc/logger.ts","../dev/core/src/Misc/deepCopier.ts","../dev/core/src/Misc/precisionDate.ts","../dev/core/src/Misc/webRequest.ts","../dev/core/src/Misc/filesInputStore.ts","../dev/core/src/Misc/error.ts","../dev/core/src/Misc/stringTools.ts","../dev/core/src/Engines/Processors/shaderCodeNode.ts","../dev/core/src/Engines/Processors/shaderCodeCursor.ts","../dev/core/src/Engines/Processors/shaderCodeConditionNode.ts","../dev/core/src/Engines/Processors/shaderCodeTestNode.ts","../dev/core/src/Engines/Processors/Expressions/shaderDefineExpression.ts","../dev/core/src/Engines/Processors/Expressions/Operators/shaderDefineIsDefinedOperator.ts","../dev/core/src/Engines/Processors/Expressions/Operators/shaderDefineOrOperator.ts","../dev/core/src/Engines/Processors/Expressions/Operators/shaderDefineAndOperator.ts","../dev/core/src/Engines/Processors/Expressions/Operators/shaderDefineArithmeticOperator.ts","../dev/core/src/Materials/shaderLanguage.ts","../dev/core/src/Engines/Processors/shaderProcessor.ts","../dev/core/src/Engines/shaderStore.ts","../dev/core/src/Materials/effect.ts","../dev/core/src/States/depthCullingState.ts","../dev/core/src/States/stencilState.ts","../dev/core/src/States/alphaCullingState.ts","../dev/core/src/Materials/Textures/internalTexture.ts","../dev/core/src/Materials/Textures/textureSampler.ts","../dev/core/src/Engines/WebGL/webGLShaderProcessors.ts","../dev/core/src/Engines/WebGL/webGL2ShaderProcessors.ts","../dev/core/src/Buffers/dataBuffer.ts","../dev/core/src/Meshes/WebGL/webGLDataBuffer.ts","../dev/core/src/Engines/WebGL/webGLPipelineContext.ts","../dev/core/src/Engines/WebGL/webGLHardwareTexture.ts","../dev/core/src/Materials/drawWrapper.ts","../dev/core/src/States/stencilStateComposer.ts","../dev/core/src/Engines/thinEngine.ts","../dev/core/src/Misc/timingTools.ts","../dev/core/src/Misc/fileTools.ts","../dev/core/src/Misc/retryStrategy.ts","../dev/core/src/Misc/instantiationTools.ts","../dev/core/src/Misc/guid.ts","../dev/core/src/Misc/tools.ts","../dev/core/src/Misc/coroutine.ts","../dev/core/src/Misc/smartArray.ts","../dev/core/src/Maths/math.viewport.ts","../dev/core/src/Maths/math.plane.ts","../dev/core/src/Maths/math.frustum.ts","../dev/core/src/Cameras/camera.ts","../dev/core/src/Misc/stringDictionary.ts","../dev/core/src/abstractScene.ts","../dev/core/src/Materials/materialDefines.ts","../dev/core/src/Materials/colorCurves.ts","../dev/core/src/Materials/imageProcessingConfiguration.ts","../dev/core/src/DeviceInput/InputDevices/deviceEnums.ts","../dev/core/src/Events/deviceInputEvents.ts","../dev/core/src/Engines/Extensions/engine.uniformBuffer.ts","../dev/core/src/Materials/uniformBuffer.ts","../dev/core/src/Buffers/buffer.ts","../dev/core/src/Collisions/pickingInfo.ts","../dev/core/src/Actions/actionEvent.ts","../dev/core/src/PostProcesses/postProcessManager.ts","../dev/core/src/Rendering/renderingGroup.ts","../dev/core/src/Rendering/renderingManager.ts","../dev/core/src/sceneComponent.ts","../dev/core/src/Events/pointerEvents.ts","../dev/core/src/Actions/abstractActionManager.ts","../dev/core/src/Events/keyboardEvents.ts","../dev/core/src/DeviceInput/eventFactory.ts","../dev/core/src/DeviceInput/nativeDeviceInputSystem.ts","../dev/core/src/DeviceInput/webDeviceInputSystem.ts","../dev/core/src/DeviceInput/InputDevices/deviceSource.ts","../dev/core/src/DeviceInput/internalDeviceSourceManager.ts","../dev/core/src/DeviceInput/InputDevices/deviceSourceManager.ts","../dev/core/src/Inputs/scene.inputManager.ts","../dev/core/src/Misc/perfCounter.ts","../dev/core/src/Misc/uniqueIdGenerator.ts","../dev/core/src/Lights/lightConstants.ts","../dev/core/src/Inputs/pointerPickingConfiguration.ts","../dev/core/src/scene.ts","../dev/core/src/Collisions/intersectionInfo.ts","../dev/core/src/Culling/boundingBox.ts","../dev/core/src/Culling/boundingSphere.ts","../dev/core/src/Culling/boundingInfo.ts","../dev/core/src/Maths/math.functions.ts","../dev/core/src/Meshes/subMesh.ts","../dev/core/src/Meshes/mesh.vertexData.ts","../dev/core/src/Loading/sceneLoaderFlags.ts","../dev/core/src/Compat/compatibilityOptions.ts","../dev/core/src/Meshes/geometry.ts","../dev/core/src/Misc/performanceMonitor.ts","../dev/core/src/Engines/Extensions/engine.alpha.ts","../dev/core/src/Engines/Extensions/engine.readTexture.ts","../dev/core/src/Engines/Extensions/engine.dynamicBuffer.ts","../dev/core/src/Engines/engine.ts","../dev/core/src/Maths/math.axis.ts","../dev/core/src/Meshes/transformNode.ts","../dev/core/src/Collisions/meshCollisionData.ts","../dev/core/src/Meshes/abstractMesh.ts","../dev/core/src/Materials/clipPlaneMaterialHelper.ts","../dev/core/src/Materials/materialHelper.ts","../dev/core/src/Materials/materialStencilState.ts","../dev/core/src/Materials/materialPluginEvent.ts","../dev/core/src/Maths/math.path.ts","../dev/core/src/Animations/animationKey.ts","../dev/core/src/Materials/material.ts","../dev/core/src/Materials/multiMaterial.ts","../dev/core/src/Meshes/meshLODLevel.ts","../dev/core/src/Meshes/mesh.ts","../dev/core/src/Behaviors/Cameras/autoRotationBehavior.ts","../dev/core/src/Animations/easing.ts","../dev/core/src/Animations/animationRange.ts","../dev/core/src/Maths/math.size.ts","../dev/core/src/Animations/animation.ts","../dev/core/src/Behaviors/Cameras/bouncingBehavior.ts","../dev/core/src/Behaviors/Cameras/framingBehavior.ts","../dev/core/src/Cameras/targetCamera.ts","../dev/core/src/Cameras/cameraInputsManager.ts","../dev/core/src/Cameras/Inputs/BaseCameraPointersInput.ts","../dev/core/src/Cameras/Inputs/arcRotateCameraPointersInput.ts","../dev/core/src/Cameras/Inputs/arcRotateCameraKeyboardMoveInput.ts","../dev/core/src/Cameras/Inputs/arcRotateCameraMouseWheelInput.ts","../dev/core/src/Cameras/arcRotateCameraInputsManager.ts","../dev/core/src/Cameras/arcRotateCamera.ts","../dev/core/src/Culling/ray.ts","../dev/core/src/Maths/sphericalPolynomial.ts","../Client/viewer/view/helpers/math-helper.ts","../Client/viewer/models/classes/primitives/kb-quaternion.ts","../Client/viewer/models/classes/meshes/box-node.ts","../Client/viewer/models/classes/meshes/cone-node.ts","../Client/viewer/models/classes/meshes/cylinder-node.ts","../Client/viewer/models/classes/meshes/disc-node.ts","../Client/viewer/models/classes/meshes/mesh-utilities.ts","../Client/viewer/models/classes/meshes/plane-node.ts","../Client/viewer/models/classes/primitives/kb-vector.ts","../Client/viewer/models/classes/meshes/space-node.ts","../Client/viewer/models/classes/meshes/mesh-node.ts","../Client/viewer/models/classes/meshes/primitive-mesh-node.ts","../Client/viewer/models/classes/meshes/custom-mesh-node.ts","../Client/viewer/models/classes/meshes/mesh-group-node.ts","../Client/viewer/models/classes/primitives/kb-color.ts","../Client/viewer/models/classes/primitives/kb-geometry.ts","../Client/viewer/models/classes/enums.ts","../Client/viewer/models/classes/meshes/sphere-node.ts","../Client/viewer/models/classes/meshes/torus-node.ts","../Client/viewer/models/classes/meshes/sketch-node.ts","../Client/viewer/models/classes/meshes/sketch-node-path.ts","../Client/viewer/models/classes/meshes/text-node.ts","../Client/viewer/models/classes/features/pattern-feature.ts","../Client/viewer/models/classes/features/connector.ts","../Client/viewer/models/classes/features/circular-pattern-feature.ts","../Client/viewer/models/classes/features/delete-faces-feature.ts","../Client/viewer/models/classes/features/displacement-map-feature.ts","../Client/viewer/models/classes/features/extrude-feature.ts","../Client/viewer/models/classes/features/join-geometry-feature.ts","../Client/viewer/models/classes/features/linear-pattern-feature.ts","../Client/viewer/models/classes/features/slice-feature.ts","../Client/viewer/models/classes/features/mirror-feature.ts","../Client/viewer/models/classes/features/move-vertices-feature.ts","../Client/viewer/models/classes/features/normal-smoothing-feature.ts","../Client/viewer/models/classes/handlers.ts","../Client/viewer/models/classes/features/sketch-control-point.ts","../Client/viewer/models/classes/features/uv-map-feature.ts","../Client/viewer/models/classes/features/submesh-feature.ts","../Client/viewer/models/classes/features/sweep-feature.ts","../Client/viewer/models/classes/features/transform-feature.ts","../Client/viewer/models/classes/features/weld-vertices-feature.ts","../Client/viewer/models/classes/annotations/annotation-node.ts","../Client/viewer/models/classes/annotations/dimension-node.ts","../Client/viewer/models/classes/import-job.ts","../Client/viewer/models/services/font-service.ts","../Client/viewer/models/services/http-service.ts","../Client/viewer/common-util/texture-util.ts","../Client/viewer/models/constants.ts","../Client/viewer/models/services/loading-timer-helper.ts","../Client/viewer/models/services/material-service.ts","../Client/viewer/common-util/device-helper.ts","../Client/viewer/models/services/mesh-service.ts","../Client/viewer/models/classes/lights/light-node.ts","../Client/viewer/models/classes/scene-node.ts","../Client/viewer/models/classes/kb3d-manager.ts","../Client/viewer/models/classes/lights/shadow-light-node.ts","../Client/viewer/models/classes/materials/texture-layer.ts","../Client/viewer/models/classes/lights/directional-light-node.ts","../Client/viewer/models/classes/lights/hemispheric-light-node.ts","../Client/viewer/models/classes/lights/point-light-node.ts","../Client/viewer/models/classes/lights/spot-light-node.ts","../Client/viewer/models/classes/materials/advanced-material-node.ts","../Client/viewer/models/classes/materials/color-fill-texture-layer.ts","../Client/viewer/models/classes/materials/file-texture-layer.ts","../Client/viewer/models/classes/materials/svg-texture-layer.ts","../Client/viewer/models/classes/materials/text-texture-layer.ts","../Client/viewer/models/classes/mates/mate-node.ts","../Client/viewer/view/helpers/graph-helper.ts","../Client/viewer/models/classes/viewpoints/viewpoint-node.ts","../dev/core/src/Materials/Textures/thinTexture.ts","../dev/core/src/Materials/Textures/baseTexture.ts","../dev/core/src/Misc/copyTools.ts","../dev/core/src/Materials/Textures/texture.ts","../dev/core/src/Materials/prePassConfiguration.ts","../dev/core/src/Materials/pushMaterial.ts","../dev/core/src/Materials/materialFlags.ts","../dev/core/src/Shaders/ShadersInclude/decalFragmentDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/defaultFragmentDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/sceneUboDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/meshUboDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/defaultUboDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/prePassDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/oitDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/mainUVVaryingDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/helperFunctions.ts","../dev/core/src/Shaders/ShadersInclude/lightFragmentDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/lightUboDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/lightsFragmentFunctions.ts","../dev/core/src/Shaders/ShadersInclude/shadowsFragmentFunctions.ts","../dev/core/src/Shaders/ShadersInclude/samplerFragmentDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/fresnelFunction.ts","../dev/core/src/Shaders/ShadersInclude/reflectionFunction.ts","../dev/core/src/Shaders/ShadersInclude/imageProcessingDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/imageProcessingFunctions.ts","../dev/core/src/Shaders/ShadersInclude/bumpFragmentMainFunctions.ts","../dev/core/src/Shaders/ShadersInclude/bumpFragmentFunctions.ts","../dev/core/src/Shaders/ShadersInclude/clipPlaneFragmentDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/logDepthDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/fogFragmentDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/clipPlaneFragment.ts","../dev/core/src/Shaders/ShadersInclude/bumpFragment.ts","../dev/core/src/Shaders/ShadersInclude/decalFragment.ts","../dev/core/src/Shaders/ShadersInclude/depthPrePass.ts","../dev/core/src/Shaders/ShadersInclude/lightFragment.ts","../dev/core/src/Shaders/ShadersInclude/logDepthFragment.ts","../dev/core/src/Shaders/ShadersInclude/fogFragment.ts","../dev/core/src/Shaders/ShadersInclude/oitFragment.ts","../dev/core/src/Shaders/default.fragment.ts","../dev/core/src/Shaders/ShadersInclude/decalVertexDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/defaultVertexDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/uvAttributeDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/bonesDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/bakedVertexAnimationDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/instancesDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/prePassVertexDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/samplerVertexDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/bumpVertexDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/clipPlaneVertexDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/fogVertexDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/lightVxFragmentDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/lightVxUboDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/morphTargetsVertexGlobalDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/morphTargetsVertexDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/morphTargetsVertexGlobal.ts","../dev/core/src/Shaders/ShadersInclude/morphTargetsVertex.ts","../dev/core/src/Shaders/ShadersInclude/instancesVertex.ts","../dev/core/src/Shaders/ShadersInclude/bonesVertex.ts","../dev/core/src/Shaders/ShadersInclude/bakedVertexAnimation.ts","../dev/core/src/Shaders/ShadersInclude/prePassVertex.ts","../dev/core/src/Shaders/ShadersInclude/uvVariableDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/samplerVertexImplementation.ts","../dev/core/src/Shaders/ShadersInclude/bumpVertex.ts","../dev/core/src/Shaders/ShadersInclude/clipPlaneVertex.ts","../dev/core/src/Shaders/ShadersInclude/fogVertex.ts","../dev/core/src/Shaders/ShadersInclude/shadowsVertex.ts","../dev/core/src/Shaders/ShadersInclude/vertexColorMixing.ts","../dev/core/src/Shaders/ShadersInclude/pointCloudVertex.ts","../dev/core/src/Shaders/ShadersInclude/logDepthVertex.ts","../dev/core/src/Shaders/default.vertex.ts","../dev/core/src/Materials/effectFallbacks.ts","../dev/core/src/Materials/materialPluginManager.ts","../dev/core/src/Materials/materialPluginBase.ts","../dev/core/src/Materials/material.detailMapConfiguration.ts","../dev/core/src/Materials/standardMaterial.ts","../dev/core/src/Shaders/postprocess.vertex.ts","../dev/core/src/Engines/WebGL/webGLRenderTargetWrapper.ts","../dev/core/src/Engines/renderTargetWrapper.ts","../dev/core/src/Engines/Extensions/engine.renderTarget.ts","../dev/core/src/PostProcesses/postProcess.ts","../dev/core/src/Shaders/rgbdDecode.fragment.ts","../dev/core/src/Engines/Extensions/engine.renderTargetCube.ts","../dev/core/src/Materials/effectRenderer.ts","../dev/core/src/Shaders/pass.fragment.ts","../dev/core/src/Misc/dumpTools.ts","../dev/core/src/Materials/Textures/renderTargetTexture.ts","../dev/core/src/Shaders/passCube.fragment.ts","../dev/core/src/PostProcesses/passPostProcess.ts","../dev/core/src/Misc/textureTools.ts","../dev/core/src/Misc/rgbdTextureTools.ts","../dev/core/src/Misc/brdfTextureTools.ts","../dev/core/src/Materials/PBR/pbrBRDFConfiguration.ts","../dev/core/src/Misc/HighDynamicRange/cubemapToSphericalPolynomial.ts","../dev/core/src/Materials/Textures/baseTexture.polynomial.ts","../dev/core/src/Shaders/ShadersInclude/pbrFragmentDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/pbrUboDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/pbrFragmentExtraDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/samplerFragmentAlternateDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/pbrFragmentSamplersDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/subSurfaceScatteringFunctions.ts","../dev/core/src/Shaders/ShadersInclude/importanceSampling.ts","../dev/core/src/Shaders/ShadersInclude/pbrHelperFunctions.ts","../dev/core/src/Shaders/ShadersInclude/harmonicsFunctions.ts","../dev/core/src/Shaders/ShadersInclude/pbrDirectLightingSetupFunctions.ts","../dev/core/src/Shaders/ShadersInclude/pbrDirectLightingFalloffFunctions.ts","../dev/core/src/Shaders/ShadersInclude/pbrBRDFFunctions.ts","../dev/core/src/Shaders/ShadersInclude/hdrFilteringFunctions.ts","../dev/core/src/Shaders/ShadersInclude/pbrDirectLightingFunctions.ts","../dev/core/src/Shaders/ShadersInclude/pbrIBLFunctions.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockAlbedoOpacity.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockReflectivity.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockAmbientOcclusion.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockAlphaFresnel.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockAnisotropic.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockReflection.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockSheen.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockClearcoat.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockIridescence.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockSubSurface.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockNormalGeometric.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockNormalFinal.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockLightmapInit.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockGeometryInfo.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockReflectance0.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockReflectance.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockDirectLighting.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockFinalLitComponents.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockFinalUnlitComponents.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockFinalColorComposition.ts","../dev/core/src/Shaders/ShadersInclude/pbrBlockImageProcessing.ts","../dev/core/src/Shaders/ShadersInclude/pbrDebug.ts","../dev/core/src/Shaders/pbr.fragment.ts","../dev/core/src/Shaders/ShadersInclude/pbrVertexDeclaration.ts","../dev/core/src/Shaders/pbr.vertex.ts","../dev/core/src/Materials/PBR/pbrClearCoatConfiguration.ts","../dev/core/src/Materials/PBR/pbrIridescenceConfiguration.ts","../dev/core/src/Materials/PBR/pbrAnisotropicConfiguration.ts","../dev/core/src/Materials/PBR/pbrSheenConfiguration.ts","../dev/core/src/Materials/PBR/pbrSubSurfaceConfiguration.ts","../dev/core/src/Materials/PBR/pbrBaseMaterial.ts","../dev/core/src/Materials/PBR/pbrMaterial.ts","../dev/core/src/Lights/light.ts","../dev/core/src/Lights/hemisphericLight.ts","../dev/core/src/Shaders/ShadersInclude/kernelBlurVaryingDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/packingFunctions.ts","../dev/core/src/Shaders/ShadersInclude/kernelBlurFragment.ts","../dev/core/src/Shaders/ShadersInclude/kernelBlurFragment2.ts","../dev/core/src/Shaders/kernelBlur.fragment.ts","../dev/core/src/Shaders/ShadersInclude/kernelBlurVertex.ts","../dev/core/src/Shaders/kernelBlur.vertex.ts","../dev/core/src/PostProcesses/blurPostProcess.ts","../dev/core/src/Materials/Textures/mirrorTexture.ts","../dev/core/src/Engines/Extensions/engine.cubeTexture.ts","../dev/core/src/Materials/Textures/cubeTexture.ts","../dev/core/src/Shaders/ShadersInclude/backgroundFragmentDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/backgroundUboDeclaration.ts","../dev/core/src/Shaders/background.fragment.ts","../dev/core/src/Shaders/ShadersInclude/backgroundVertexDeclaration.ts","../dev/core/src/Shaders/background.vertex.ts","../dev/core/src/Materials/Background/backgroundMaterial.ts","../dev/core/src/Meshes/Builders/planeBuilder.ts","../dev/core/src/Meshes/groundMesh.ts","../dev/core/src/Meshes/Builders/groundBuilder.ts","../dev/core/src/Meshes/Builders/boxBuilder.ts","../dev/core/src/Helpers/environmentHelper.ts","../dev/core/src/Cameras/Inputs/freeCameraKeyboardMoveInput.ts","../dev/core/src/Cameras/Inputs/freeCameraMouseInput.ts","../dev/core/src/Cameras/Inputs/BaseCameraMouseWheelInput.ts","../dev/core/src/Cameras/Inputs/freeCameraMouseWheelInput.ts","../dev/core/src/Cameras/Inputs/freeCameraTouchInput.ts","../dev/core/src/Cameras/freeCameraInputsManager.ts","../dev/core/src/Cameras/freeCamera.ts","../dev/core/src/Cameras/Inputs/freeCameraDeviceOrientationInput.ts","../dev/core/src/Cameras/deviceOrientationCamera.ts","../dev/core/src/Cameras/VR/vrCameraMetrics.ts","../dev/core/src/Shaders/vrDistortionCorrection.fragment.ts","../dev/core/src/PostProcesses/vrDistortionCorrectionPostProcess.ts","../dev/core/src/Shaders/vrMultiviewToSingleview.fragment.ts","../dev/core/src/Materials/Textures/MultiviewRenderTarget.ts","../dev/core/src/Engines/Extensions/engine.multiview.ts","../dev/core/src/PostProcesses/vrMultiviewToSingleviewPostProcess.ts","../dev/core/src/Cameras/RigModes/vrRigMode.ts","../dev/core/src/Cameras/VR/vrDeviceOrientationFreeCamera.ts","../dev/core/src/Gamepads/gamepad.ts","../dev/core/src/Gamepads/Controllers/poseEnabledController.ts","../dev/core/src/Gamepads/xboxGamepad.ts","../dev/core/src/Cameras/RigModes/webVRRigMode.ts","../dev/core/src/Engines/Extensions/engine.webVR.ts","../dev/core/src/Cameras/VR/webVRCamera.ts","../dev/core/src/Gamepads/Controllers/webVRController.ts","../dev/core/src/Engines/Extensions/engine.dynamicTexture.ts","../dev/core/src/Materials/Textures/dynamicTexture.ts","../dev/core/src/Shaders/imageProcessing.fragment.ts","../dev/core/src/PostProcesses/imageProcessingPostProcess.ts","../dev/core/src/Gamepads/dualShockGamepad.ts","../dev/core/src/Gamepads/gamepadManager.ts","../dev/core/src/Cameras/Inputs/freeCameraGamepadInput.ts","../dev/core/src/Cameras/Inputs/arcRotateCameraGamepadInput.ts","../dev/core/src/Gamepads/gamepadSceneComponent.ts","../dev/core/src/Animations/runtimeAnimation.ts","../dev/core/src/Bones/bone.ts","../dev/core/src/Animations/animatable.ts","../dev/core/src/XR/webXRLayerWrapper.ts","../dev/core/src/XR/webXRRenderTargetTextureProvider.ts","../dev/core/src/XR/webXRWebGLLayer.ts","../dev/core/src/XR/webXRManagedOutputCanvas.ts","../dev/core/src/XR/native/nativeXRRenderTarget.ts","../dev/core/src/XR/webXRSessionManager.ts","../dev/core/src/XR/webXRTypes.ts","../dev/core/src/Meshes/Builders/cylinderBuilder.ts","../dev/core/src/Meshes/Builders/torusBuilder.ts","../dev/core/src/Cameras/VR/vrExperienceHelper.ts","../dev/core/src/Misc/dds.ts","../dev/core/src/Materials/Textures/Loaders/ddsTextureLoader.ts","../dev/core/src/Shaders/rgbdEncode.fragment.ts","../dev/core/src/Misc/environmentTextureTools.ts","../dev/core/src/Materials/Textures/Loaders/envTextureLoader.ts","../dev/core/src/Misc/khronosTextureContainer.ts","../dev/core/src/Misc/workerPool.ts","../dev/core/src/Materials/Textures/ktx2decoderTypes.ts","../dev/core/src/Loading/sceneLoader.ts","../dev/core/src/Misc/khronosTextureContainer2.ts","../dev/core/src/Materials/Textures/Loaders/ktxTextureLoader.ts","../dev/core/src/XR/webXRCamera.ts","../dev/core/src/XR/webXRFeaturesManager.ts","../dev/core/src/Cameras/touchCamera.ts","../dev/core/src/Cameras/universalCamera.ts","../dev/core/src/XR/webXRExperienceHelper.ts","../dev/core/src/XR/motionController/webXRControllerComponent.ts","../dev/core/src/XR/motionController/webXRAbstractMotionController.ts","../dev/core/src/XR/motionController/webXRGenericMotionController.ts","../dev/core/src/Meshes/Builders/sphereBuilder.ts","../dev/core/src/XR/motionController/webXRProfiledMotionController.ts","../dev/core/src/XR/motionController/webXRMotionControllerManager.ts","../dev/core/src/XR/webXRInputSource.ts","../dev/core/src/XR/webXRInput.ts","../dev/core/src/XR/features/WebXRAbstractFeature.ts","../dev/core/src/Rendering/utilityLayerRenderer.ts","../dev/core/src/XR/features/WebXRControllerPointerSelection.ts","../dev/core/src/Materials/Node/Enums/nodeMaterialBlockConnectionPointTypes.ts","../dev/core/src/Materials/Node/Enums/nodeMaterialBlockTargets.ts","../dev/core/src/Materials/Node/nodeMaterialBlockConnectionPoint.ts","../dev/core/src/Decorators/nodeDecorator.ts","../dev/core/src/Materials/Node/Enums/nodeMaterialBlockConnectionPointMode.ts","../dev/core/src/Materials/Node/Enums/nodeMaterialSystemValues.ts","../dev/core/src/Materials/Node/Blocks/Input/animatedInputBlockTypes.ts","../dev/core/src/Materials/Node/nodeMaterialBuildState.ts","../dev/core/src/Materials/Node/nodeMaterialBuildStateSharedData.ts","../dev/core/src/Materials/Node/nodeMaterialBlock.ts","../dev/core/src/Materials/Node/Blocks/transformBlock.ts","../dev/core/src/Materials/Node/Blocks/Vertex/vertexOutputBlock.ts","../dev/core/src/Materials/Node/Blocks/Fragment/fragmentOutputBlock.ts","../dev/core/src/Materials/Node/Blocks/Input/inputBlock.ts","../dev/core/src/Materials/Node/Blocks/Dual/currentScreenBlock.ts","../dev/core/src/Materials/Node/Blocks/Particle/particleTextureBlock.ts","../dev/core/src/Materials/Node/Blocks/Particle/particleRampGradientBlock.ts","../dev/core/src/Materials/Node/Blocks/Particle/particleBlendMultiplyBlock.ts","../dev/core/src/Materials/Node/Blocks/vectorMergerBlock.ts","../dev/core/src/Materials/Node/Blocks/remapBlock.ts","../dev/core/src/Materials/Node/Blocks/multiplyBlock.ts","../dev/core/src/Materials/Node/Enums/nodeMaterialModes.ts","../dev/core/src/Particles/EmitterTypes/boxParticleEmitter.ts","../dev/core/src/Particles/EmitterTypes/coneParticleEmitter.ts","../dev/core/src/Particles/EmitterTypes/cylinderParticleEmitter.ts","../dev/core/src/Particles/EmitterTypes/hemisphericParticleEmitter.ts","../dev/core/src/Particles/EmitterTypes/pointParticleEmitter.ts","../dev/core/src/Particles/EmitterTypes/sphereParticleEmitter.ts","../dev/core/src/Particles/EmitterTypes/meshParticleEmitter.ts","../dev/core/src/Particles/baseParticleSystem.ts","../dev/core/src/Materials/Node/Blocks/colorSplitterBlock.ts","../dev/core/src/Materials/Textures/Procedurals/proceduralTextureSceneComponent.ts","../dev/core/src/Shaders/procedural.vertex.ts","../dev/core/src/Materials/Textures/Procedurals/proceduralTexture.ts","../dev/core/src/Materials/Node/Blocks/trigonometryBlock.ts","../dev/core/src/Materials/Node/nodeMaterial.ts","../dev/core/src/XR/features/WebXRNearInteraction.ts","../dev/core/src/Meshes/subMesh.project.ts","../dev/core/src/XR/webXREnterExitUI.ts","../dev/core/src/Meshes/instancedMesh.ts","../dev/core/src/Materials/shaderMaterial.ts","../dev/core/src/Shaders/color.fragment.ts","../dev/core/src/Shaders/color.vertex.ts","../dev/core/src/Meshes/linesMesh.ts","../dev/core/src/Meshes/Builders/linesBuilder.ts","../dev/core/src/Misc/timer.ts","../dev/core/src/XR/features/WebXRControllerTeleportation.ts","../dev/core/src/XR/webXRDefaultExperience.ts","../dev/core/src/Helpers/sceneHelpers.ts","../dev/core/src/Shaders/glowMapGeneration.fragment.ts","../dev/core/src/Shaders/glowMapGeneration.vertex.ts","../dev/core/src/Layers/effectLayer.ts","../dev/core/src/Layers/effectLayerSceneComponent.ts","../dev/core/src/Lights/shadowLight.ts","../dev/core/src/Shaders/ShadersInclude/bayerDitherFunctions.ts","../dev/core/src/Shaders/ShadersInclude/shadowMapFragmentExtraDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/shadowMapFragment.ts","../dev/core/src/Shaders/shadowMap.fragment.ts","../dev/core/src/Shaders/ShadersInclude/sceneVertexDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/meshVertexDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/shadowMapVertexDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/shadowMapUboDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/shadowMapVertexExtraDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/shadowMapVertexNormalBias.ts","../dev/core/src/Shaders/ShadersInclude/shadowMapVertexMetric.ts","../dev/core/src/Shaders/shadowMap.vertex.ts","../dev/core/src/Shaders/depthBoxBlur.fragment.ts","../dev/core/src/Shaders/ShadersInclude/shadowMapFragmentSoftTransparentShadow.ts","../dev/core/src/Lights/Shadows/shadowGenerator.ts","../dev/core/src/Shaders/depth.fragment.ts","../dev/core/src/Shaders/depth.vertex.ts","../dev/core/src/Rendering/depthRenderer.ts","../dev/core/src/Shaders/minmaxRedux.fragment.ts","../dev/core/src/Misc/depthReducer.ts","../dev/core/src/Misc/minMaxReducer.ts","../dev/core/src/Lights/Shadows/cascadedShadowGenerator.ts","../dev/core/src/Lights/Shadows/shadowGeneratorSceneComponent.ts","../dev/core/src/Lights/directionalLight.ts","../dev/core/src/Lights/pointLight.ts","../dev/core/src/Lights/spotLight.ts","../dev/core/src/Misc/sceneOptimizer.ts","../dev/core/src/Misc/sceneSerializer.ts","../dev/core/src/Rendering/depthRendererSceneComponent.ts","../dev/core/src/Shaders/line.fragment.ts","../dev/core/src/Shaders/line.vertex.ts","../dev/core/src/Rendering/edgesRenderer.ts","../dev/core/src/Engines/Extensions/engine.multiRender.ts","../dev/core/src/Materials/Textures/multiRenderTarget.ts","../dev/core/src/Shaders/ShadersInclude/mrtFragmentDeclaration.ts","../dev/core/src/Shaders/geometry.fragment.ts","../dev/core/src/Shaders/ShadersInclude/geometryVertexDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/geometryUboDeclaration.ts","../dev/core/src/Shaders/geometry.vertex.ts","../dev/core/src/Rendering/geometryBufferRenderer.ts","../dev/core/src/Rendering/geometryBufferRendererSceneComponent.ts","../dev/core/src/Shaders/outline.fragment.ts","../dev/core/src/Shaders/outline.vertex.ts","../dev/core/src/Rendering/outlineRenderer.ts","../dev/core/src/Materials/Textures/prePassRenderTarget.ts","../dev/core/src/Rendering/prePassRenderer.ts","../dev/core/src/Rendering/prePassRendererSceneComponent.ts","../Client/viewer/models/services/settings-manager-service.ts","../dev/core/src/Meshes/Builders/ribbonBuilder.ts","../dev/core/src/Meshes/Builders/discBuilder.ts","../dev/core/src/Meshes/Builders/tiledPlaneBuilder.ts","../dev/core/src/Meshes/Builders/tiledBoxBuilder.ts","../dev/core/src/Meshes/Builders/torusKnotBuilder.ts","../dev/core/src/Meshes/polygonMesh.ts","../dev/core/src/Meshes/Builders/polygonBuilder.ts","../dev/core/src/Meshes/Builders/shapeBuilder.ts","../dev/core/src/Meshes/Builders/latheBuilder.ts","../dev/core/src/Meshes/Builders/tubeBuilder.ts","../dev/core/src/Meshes/Builders/polyhedronBuilder.ts","../dev/core/src/Meshes/Builders/icoSphereBuilder.ts","../dev/core/src/Meshes/Builders/decalBuilder.ts","../dev/core/src/Meshes/Builders/capsuleBuilder.ts","../dev/core/src/Maths/math.isovector.ts","../dev/core/src/Meshes/geodesicMesh.ts","../dev/core/src/Meshes/goldbergMesh.ts","../dev/core/src/Meshes/Builders/textBuilder.ts","../dev/core/src/Meshes/meshBuilder.ts","../dev/core/src/Meshes/Builders/geodesicBuilder.ts","../dev/core/src/Meshes/Builders/goldbergBuilder.ts","../Client/viewer/view/helpers/gui-helper.ts","../Client/viewer/view/helpers/scene-helper.ts","../Client/viewer/view/kb-worker.ts","../Client/viewer/view/helpers/imagedata-polyfill.ts","../node_modules/@tweenjs/tween.js/dist/tween.esm.js","../Client/viewer/view/modules/animation.ts","../dev/core/src/XR/features/WebXRAnchorSystem.ts","../dev/core/src/XR/features/WebXRHitTest.ts","../dev/core/src/XR/features/WebXRPlaneDetector.ts","../dev/core/src/Morph/morphTarget.ts","../dev/serializers/src/glTF/2.0/glTFUtilities.ts","../dev/serializers/src/glTF/2.0/glTFAnimation.ts","../dev/serializers/src/glTF/2.0/glTFData.ts","../dev/core/src/Engines/Extensions/engine.rawTexture.ts","../dev/core/src/Materials/Textures/rawTexture.ts","../dev/core/src/Engines/constants.ts","../dev/serializers/src/glTF/2.0/glTFMaterialExporter.ts","../dev/serializers/src/glTF/2.0/glTFExporter.ts","../dev/serializers/src/glTF/2.0/glTFSerializer.ts","../dev/serializers/src/glTF/2.0/Extensions/KHR_texture_transform.ts","../dev/serializers/src/glTF/2.0/Extensions/KHR_lights_punctual.ts","../dev/serializers/src/glTF/2.0/Extensions/KHR_materials_clearcoat.ts","../dev/serializers/src/glTF/2.0/Extensions/KHR_materials_iridescence.ts","../dev/serializers/src/glTF/2.0/Extensions/KHR_materials_anisotropy.ts","../dev/serializers/src/glTF/2.0/Extensions/KHR_materials_sheen.ts","../dev/serializers/src/glTF/2.0/Extensions/KHR_materials_unlit.ts","../dev/serializers/src/glTF/2.0/Extensions/KHR_materials_ior.ts","../dev/serializers/src/glTF/2.0/Extensions/KHR_materials_specular.ts","../dev/serializers/src/glTF/2.0/Extensions/KHR_materials_volume.ts","../dev/serializers/src/glTF/2.0/Extensions/KHR_materials_transmission.ts","../dev/core/src/Meshes/thinInstanceMesh.ts","../dev/serializers/src/glTF/2.0/Extensions/EXT_mesh_gpu_instancing.ts","../dev/serializers/src/glTF/2.0/Extensions/KHR_materials_emissive_strength.ts","../Client/viewer/view/modules/exporter-gltf.ts","../dev/core/src/Materials/Textures/colorGradingTexture.ts","../dev/core/src/Misc/HighDynamicRange/panoramaToCubemap.ts","../dev/core/src/Shaders/hdrFiltering.vertex.ts","../dev/core/src/Shaders/hdrFiltering.fragment.ts","../dev/core/src/Materials/Textures/Filtering/hdrFiltering.ts","../dev/core/src/Misc/HighDynamicRange/hdr.ts","../dev/core/src/Materials/Textures/hdrCubeTexture.ts","../dev/core/src/Engines/Extensions/engine.videoTexture.ts","../dev/core/src/Engines/Extensions/engine.externalTexture.ts","../dev/core/src/Misc/tga.ts","../dev/core/src/Materials/Textures/Loaders/tgaTextureLoader.ts","../dev/core/src/Misc/basis.ts","../dev/core/src/Materials/Textures/Loaders/hdrTextureLoader.ts","../dev/core/src/Materials/Textures/Loaders/basisTextureLoader.ts","../dev/core/src/Shaders/noise.fragment.ts","../dev/core/src/Materials/Textures/Procedurals/noiseProceduralTexture.ts","../dev/core/src/Materials/Textures/refractionTexture.ts","../node_modules/fflate/esm/browser.js","../Client/viewer/view/modules/usdz/usdz-converter.ts","../Client/viewer/view/modules/usdz/usdz-babylon-converter.ts","../Client/viewer/view/modules/xr/xr-hit-tester.ts","../Client/viewer/view/render/shadow-manager.ts","../Client/viewer/view/modules/xr/xr-light.ts","../Client/viewer/view/modules/xr/xr-marker.ts","../dev/core/src/Shaders/ShadersInclude/imageProcessingCompatibility.ts","../dev/materials/src/shadowOnly/shadowOnly.fragment.ts","../dev/materials/src/shadowOnly/shadowOnly.vertex.ts","../dev/materials/src/shadowOnly/shadowOnlyMaterial.ts","../Client/viewer/view/modules/xr/xr-shadow-plane.ts","../Client/viewer/view/modules/xr/xr-transient-hit-tester.ts","../Client/viewer/view/modules/xr/xr.ts","../Client/viewer/view/modules/ar.ts","../node_modules/rxjs/src/internal/operators/first.ts","../Client/viewer/view/modules/bounding-box-cache.ts","../Client/viewer/view/helpers/view-extensions.ts","../Client/viewer/view/helpers/bounding-box-relative-helper.ts","../Client/viewer/view/helpers/enum-maps.ts","../Client/viewer/view/helpers/light-helper.ts","../Client/viewer/view/helpers/quadtree.ts","../Client/viewer/view/helpers/earcut.ts","../Client/viewer/view/helpers/geometry-helper.ts","../Client/viewer/view/helpers/text-mesh-helper.ts","../dev/core/src/Materials/greasedLinePluginMaterial.ts","../dev/core/src/Meshes/Builders/greasedLineBuilder.ts","../dev/core/src/Meshes/Builders/hemisphereBuilder.ts","../dev/core/src/Misc/greasedLineTools.ts","../dev/core/src/Meshes/greasedLineMesh.ts","../Client/viewer/view/helpers/uvmap-helper.ts","../Client/viewer/view/helpers/vertex-group-helper.ts","../Client/viewer/view/helpers/viewpoint-helper.ts","../Client/viewer/view/render/renderer.ts","../Client/viewer/view/render/annotation.ts","../Client/viewer/view/render/dimension.ts","../Client/viewer/view/render/features/connector.ts","../Client/viewer/view/render/features/feature.ts","../Client/viewer/view/render/features/move-vertices.ts","../Client/viewer/view/render/features/normal-smoothing.ts","../dev/core/src/Particles/solidParticle.ts","../dev/core/src/Particles/solidParticleSystem.ts","../Client/viewer/view/render/features/utils.ts","../Client/viewer/view/render/features/pattern.ts","../Client/viewer/view/render/features/submesh.ts","../Client/viewer/view/render/features/uvmap.ts","../Client/viewer/view/render/light.ts","../Client/viewer/view/render/mate.ts","../dev/core/src/Probes/reflectionProbe.ts","../Client/viewer/view/render/material.ts","../Client/viewer/view/modules/sketching/util.ts","../node_modules/opentype.js/node_modules/string.prototype.codepointat/codepointat.js","../node_modules/opentype.js/node_modules/tiny-inflate/index.js","../node_modules/opentype.js/src/bbox.js","../node_modules/opentype.js/src/path.js","../node_modules/opentype.js/src/check.js","../node_modules/opentype.js/src/types.js","../node_modules/opentype.js/src/table.js","../node_modules/opentype.js/src/parse.js","../node_modules/opentype.js/src/tables/cmap.js","../node_modules/opentype.js/src/encoding.js","../node_modules/opentype.js/src/draw.js","../node_modules/opentype.js/src/glyph.js","../node_modules/opentype.js/src/glyphset.js","../node_modules/opentype.js/src/tables/cff.js","../node_modules/opentype.js/src/tables/head.js","../node_modules/opentype.js/src/tables/hhea.js","../node_modules/opentype.js/src/tables/hmtx.js","../node_modules/opentype.js/src/tables/ltag.js","../node_modules/opentype.js/src/tables/maxp.js","../node_modules/opentype.js/src/tables/name.js","../node_modules/opentype.js/src/tables/os2.js","../node_modules/opentype.js/src/tables/post.js","../node_modules/opentype.js/src/tables/gsub.js","../node_modules/opentype.js/src/tables/meta.js","../node_modules/opentype.js/src/tables/sfnt.js","../node_modules/opentype.js/src/layout.js","../node_modules/opentype.js/src/position.js","../node_modules/opentype.js/src/substitution.js","../node_modules/opentype.js/src/util.js","../node_modules/opentype.js/src/tables/glyf.js","../node_modules/opentype.js/src/hintingtt.js","../node_modules/opentype.js/src/tokenizer.js","../node_modules/opentype.js/src/char.js","../node_modules/opentype.js/src/features/featureQuery.js","../node_modules/opentype.js/src/features/arab/contextCheck/arabicWord.js","../node_modules/opentype.js/src/features/arab/contextCheck/arabicSentence.js","../node_modules/opentype.js/src/features/applySubstitution.js","../node_modules/opentype.js/src/features/arab/arabicPresentationForms.js","../node_modules/opentype.js/src/features/arab/arabicRequiredLigatures.js","../node_modules/opentype.js/src/features/latn/contextCheck/latinWord.js","../node_modules/opentype.js/src/features/latn/latinLigatures.js","../node_modules/opentype.js/src/bidi.js","../node_modules/opentype.js/src/font.js","../node_modules/opentype.js/src/tables/fvar.js","../node_modules/opentype.js/src/tables/gdef.js","../node_modules/opentype.js/src/tables/gpos.js","../node_modules/opentype.js/src/tables/kern.js","../node_modules/opentype.js/src/tables/loca.js","../node_modules/opentype.js/src/opentype.js","../Client/viewer/view/render/mesh.ts","../Client/viewer/view/helpers/mesh-builders/sphere-builder.ts","../Client/viewer/view/helpers/mesh-builders/cylinder-builder.ts","../Client/viewer/view/helpers/mesh-builders/torus-builder.ts","../Client/viewer/view/helpers/mesh-builders/plane-builder.ts","../Client/viewer/view/helpers/mesh-builders/disc-builder.ts","../Client/viewer/view/render/path.ts","../Client/viewer/view/helpers/mate-helper.ts","../Client/viewer/view/render/processors/mate-sorter.ts","../Client/viewer/view/render/processors/processor.ts","../Client/viewer/view/render/processors/mate-processor.ts","../Client/viewer/view/render/features/circular-pattern.ts","../Client/viewer/view/render/features/delete-faces.ts","../Client/viewer/view/render/features/displacement-map.ts","../Client/viewer/view/render/features/extrude.ts","../dev/core/src/Meshes/csg.ts","../Client/viewer/view/helpers/join-geometry-helper.ts","../Client/viewer/view/render/features/join-geometry.ts","../Client/viewer/view/render/features/linear-pattern.ts","../Client/viewer/view/render/features/slice.ts","../Client/viewer/view/helpers/mirror-helper.ts","../Client/viewer/view/render/features/mirror.ts","../node_modules/svg-pathdata/node_modules/tslib/tslib.es6.js","../node_modules/svg-pathdata/src/SVGPathDataEncoder.ts","../node_modules/svg-pathdata/src/mathUtils.ts","../node_modules/svg-pathdata/src/SVGPathDataTransformer.ts","../node_modules/svg-pathdata/src/TransformableSVG.ts","../node_modules/svg-pathdata/src/SVGPathDataParser.ts","../node_modules/svg-pathdata/src/SVGPathData.ts","../Client/viewer/view/modules/sketching/path-parser.ts","../Client/viewer/view/modules/sketching/path-handler.ts","../node_modules/imagetracer/dist/index.mjs","../node_modules/svg-parser/src/index.js","../node_modules/svg-parser/node_modules/locate-character/dist/locate-character.es.js","../Client/viewer/view/modules/sketching/svg-converter.ts","../Client/viewer/view/modules/miteredRibbonBuilder.ts","../Client/viewer/view/render/features/sweep.ts","../Client/viewer/view/render/features/transform.ts","../Client/viewer/view/render/features/weld-vertices.ts","../Client/viewer/view/render/feature-stack.ts","../Client/viewer/view/render/features/sketch-control-point.ts","../Client/viewer/view/render/processors/reference-processor.ts","../Client/viewer/view/render/processors/sketch-processor.ts","../dev/core/src/PostProcesses/RenderPipeline/postProcessRenderPipeline.ts","../dev/core/src/PostProcesses/RenderPipeline/postProcessRenderEffect.ts","../dev/core/src/PostProcesses/RenderPipeline/postProcessRenderPipelineManager.ts","../dev/core/src/PostProcesses/RenderPipeline/postProcessRenderPipelineManagerSceneComponent.ts","../dev/core/src/Shaders/chromaticAberration.fragment.ts","../dev/core/src/Shaders/lensHighlights.fragment.ts","../dev/core/src/Shaders/depthOfField.fragment.ts","../dev/core/src/PostProcesses/RenderPipeline/Pipelines/lensRenderingPipeline.ts","../dev/core/src/Rendering/ssao2Configuration.ts","../dev/core/src/Shaders/ssao2.fragment.ts","../dev/core/src/Shaders/ssaoCombine.fragment.ts","../dev/core/src/PostProcesses/RenderPipeline/Pipelines/ssao2RenderingPipeline.ts","../dev/core/src/Shaders/fxaa.fragment.ts","../dev/core/src/Shaders/fxaa.vertex.ts","../dev/core/src/PostProcesses/fxaaPostProcess.ts","../Client/viewer/view/render/scene.ts","../Client/viewer/view/render/textures/texture.ts","../Client/viewer/view/render/textures/color-fill-texture.ts","../Client/viewer/view/render/textures/file-texture.ts","../Client/viewer/common-util/buffer-util.ts","../Client/viewer/view/render/textures/svg-texture.ts","../Client/viewer/view/render/texture-stack.ts","../Client/viewer/view/render/viewpoint.ts","../Client/viewer/view/render/renderer-manager.ts","../Client/viewer/view/modules/cameras/kb-arc-rotate-camera-wheel-input.ts","../Client/viewer/view/modules/cameras/kb-arc-rotate-camera.ts","../Client/viewer/view/modules/camera-tools.ts","../node_modules/rxjs/src/internal/operators/startWith.ts","../dev/core/src/Culling/Octrees/octreeBlock.ts","../dev/core/src/Culling/Octrees/octree.ts","../Client/viewer/view/modules/dragger.ts","../Client/viewer/view/modules/exporter-obj.ts","../dev/serializers/src/OBJ/objSerializer.ts","../dev/core/src/Shaders/ShadersInclude/boundingBoxRendererFragmentDeclaration.ts","../dev/core/src/Shaders/ShadersInclude/boundingBoxRendererUboDeclaration.ts","../dev/core/src/Shaders/boundingBoxRenderer.fragment.ts","../dev/core/src/Shaders/ShadersInclude/boundingBoxRendererVertexDeclaration.ts","../dev/core/src/Shaders/boundingBoxRenderer.vertex.ts","../dev/core/src/Rendering/boundingBoxRenderer.ts","../dev/core/src/Misc/pivotTools.ts","../dev/core/src/Behaviors/Meshes/pointerDragBehavior.ts","../Client/viewer/view/modules/gizmos/axis-morph-gizmo.ts","../dev/core/src/Gizmos/gizmo.ts","../Client/viewer/view/modules/gizmos/gizmo.ts","../Client/viewer/view/modules/gizmos/plane-rotation-gizmo.ts","../Client/viewer/view/modules/gizmos/axis-angle-gizmo.ts","../Client/viewer/view/modules/face-engine.ts","../Client/viewer/view/modules/interaction.ts","../Client/viewer/view/modules/gizmos/facet-select-gizmo.ts","../Client/viewer/view/modules/gizmos/light-gizmo.ts","../Client/viewer/view/modules/gizmos/axis-drag-gizmo.ts","../Client/viewer/view/modules/gizmos/position-gizmo.ts","../Client/viewer/view/modules/gizmos/rotation-gizmo.ts","../Client/viewer/view/modules/gizmos/axis-scale-gizmo.ts","../Client/viewer/view/modules/gizmos/scale-gizmo.ts","../dev/core/src/Misc/gradients.ts","../dev/core/src/Particles/particle.ts","../dev/core/src/Particles/subEmitter.ts","../dev/core/src/Shaders/particles.fragment.ts","../dev/core/src/Shaders/particles.vertex.ts","../dev/core/src/Particles/particleSystem.ts","../Client/viewer/view/modules/gizmos/vertex-select-gizmo.ts","../Client/viewer/view/modules/gizmos/viewpoint-gizmo.ts","../Client/viewer/view/modules/gizmo-tools.ts","../Client/viewer/view/modules/sketch-editor.ts","../Client/viewer/view/modules/snapshot.ts","../Client/viewer/view/helpers/blob-helper.ts","../Client/viewer/view/modules/highlighter.ts","../Client/viewer/view/modules/right-click-menu.ts","../Client/viewer/view/kb-viewer.ts","../Client/viewer/index.ts","../Client/viewer/models/classes/validator.ts","../Client/viewer/view/modules/mesh-utils.ts","../Client/viewer/view/helpers/environment-helper.ts","../Client/viewer/tests/kb3d-test.ts"],"names":["kb3d","exports","String","prototype","capitalize","this","charAt","toUpperCase","slice","splitWordsRegex","__decorate","decorators","target","key","desc","d","c","arguments","length","r","Object","getOwnPropertyDescriptor","Reflect","decorate","i","defineProperty","__param","paramIndex","decorator","splitWords","match","isEqual","str","undefined","equalsAny","strings","some","s","toSentence","replace","trim","Array","enumerable","writable","value","predicate","sort","a","b","item","indexOf","splice","removed","push","selector","reduce","out","v","vres","isArray","valueDelegate","arr","every","keySelector","m","Map","set","fromIndex","toIndex","undo","itemToMove","Set","keys","k","has","SuppressedError","Token","constructor","static","token","Error","tokens","map","t","isPrimitive","PrimitiveTokenMap","isPrimitiveMesh","PrimitiveMeshTokenMap","isMesh","MeshTokenMap","isMeshOrGroup","MeshAndGroupTokenMap","isMaterial","MaterialTokenMap","isFeature","FeatureTokenMap","isTexture","TextureLayerTokenMap","isLight","LightTokenMap","isViewpoint","ViewpointNode","isMate","MateNode","isConnector","Connector","SketchControlPoint","isAnnotation","AnnotationTokenMap","isNode","NodeTokenMap","isSceneNode","SceneNode","isSketchPath","SketchPath","HttpService","MaterialService","MeshService","JobService","SceneService","FontService","Ignore","Kb3dObject","ImportInfo","LightNode","DirectionalLightNode","HemisphericLightNode","PointLightNode","SpotLightNode","BoxNode","CustomMeshNode","CylinderNode","ConeNode","DiscNode","PlaneNode","SphereNode","TorusNode","MeshNode","PrimitiveMeshNode","MeshGroupNode","SpaceNode","TextNode","SketchNode","AnnotationNode","DimensionNode","KbColor","KbGeometry","KbQuaternion","KbVector","MaterialNode","AdvancedMaterialNode","MaterialMapping","TextureLayer","FileTextureLayer","SvgTextureLayer","TextTextureLayer","ColorFillTextureLayer","Feature","BaseConnector","SubmeshFeature","MoveVerticesFeature","WeldVerticesFeature","DisplacementMapFeature","UvMapFeature","LinearPatternFeature","CircularPatternFeature","JoinGeometryFeature","DeleteFacesFeature","NormalSmoothingFeature","SliceFeature","MirrorFeature","TransformFeature","ExtrudeFeature","SweepFeature","PrimitiveTokens","MapFrom","PrimitiveMeshTokens","MeshTokens","concat","MeshAndGroupTokens","SpaceTokens","SpaceTokenMap","MaterialTokens","AnnotationTokens","PositionalLightTokens","GlobalLightTokens","LightTokens","FeatureTokens","ConnectorTokens","TextureLayerTokens","NodeTokens","BaseNodeTokens","AllTokens","AllTokenMap","Kb3dModels","instanceOf","instance","model","get","classConstructor","instanceOfAny","EChangeOperation","isTrackedClass","object","isTrackedInstance","isTrackedInstanceOfAny","changedItems","deletedItems","SyncService","tracker","modified","changes","add","delete","hasChanges","size","values","extendStatics","setPrototypeOf","__proto__","p","hasOwnProperty","__extends","__","create","isFunction","x","_enable_super_gross_mode_that_will_cause_bad_things","config","Promise","useDeprecatedSynchronousErrorHandling","stack","hostReportError","err","setTimeout","empty","closed","next","error","complete","isObject","UnsubscriptionErrorImpl","errors","call","message","toString","join","name","UnsubscriptionError","Subscription","unsubscribe","_parentOrParents","_subscriptions","_ctorUnsubscribe","_unsubscribe","_a","remove","index","e","flattenUnsubscriptionErrors","len","sub","teardown","subscription","EMPTY","tmp","subscriptions","subscriptionIndex","errs","rxSubscriber","Symbol","Math","random","Subscriber","_super","destinationOrNext","_this","syncErrorValue","syncErrorThrown","syncErrorThrowable","isStopped","destination","SafeSubscriber","tslib_1.__extends","subscriber","_next","_error","_complete","_unsubscribeAndRecycle","_parentSubscriber","observerOrNext","context","bind","_context","__tryOrSetError","__tryOrUnsub","wrappedComplete","fn","parent","observable","identity","pipeFromArray","fns","input","prev","Observable","subscribe","_isScalar","_subscribe","lift","operator","source","sink","nextOrObserver","toSubscriber","_trySubscribe","observer","closed_1","canReportError","console","warn","forEach","promiseCtor","getPromiseCtor","resolve","reject","pipe","operations","_i","toPromise","ObjectUnsubscribedError","ObjectUnsubscribedErrorImpl","SubjectSubscription","subject","observers","subscriberIndex","SubjectSubscriber","Subject","hasError","thrownError","AnonymousSubject","copy","asObservable","refCount","RefCountOperator","connectable","_refCount","refCounter","RefCountSubscriber","connection","connect","sharedConnection","_connection","ConnectableObservable","subjectFactory","_isComplete","getSubject","_subject","ConnectableSubscriber","connectableObservableDescriptor","connectableProto","BehaviorSubject","_value","getValue","configurable","AsyncAction","scheduler","work","pending","schedule","state","delay","id","recycleAsyncId","requestAsyncId","setInterval","flush","clearInterval","execute","_execute","errored","errorValue","actions","Action","QueueAction","Scheduler","SchedulerAction","now","Date","AsyncScheduler","delegate","active","scheduled","action","shift","QueueScheduler","apply","queue","empty$1","emptyScheduled","isScheduler","subscribeToArray","array","scheduleArray","fromArray","of","args","pop","dispatch","Notification","kind","hasValue","observe","do","accept","toObservable","createNext","undefinedValueNotification","createError","createComplete","completeNotification","ObserveOnSubscriber","arg","notification","scheduleMessage","ObserveOnMessage","ReplaySubject","bufferSize","windowTime","Number","POSITIVE_INFINITY","_events","_infiniteTimeWindow","_bufferSize","_windowTime","nextInfiniteTimeWindow","nextTimeWindow","ReplayEvent","_getNow","_trimBufferThenGetEvents","eventsCount","spliceCount","time","max","async","ArgumentOutOfRangeError","ArgumentOutOfRangeErrorImpl","EmptyError","EmptyErrorImpl","project","thisArg","TypeError","MapOperator","MapSubscriber","count","result","OuterSubscriber","notifyNext","outerValue","innerValue","outerIndex","innerIndex","innerSub","notifyError","notifyComplete","InnerSubscriber","getSymbolIterator","iterator","isArrayLike","isPromise","then","subscribeTo","obj","obs","promise","iterable","iterator$1","done","return","subscribeToResult","outerSubscriber","innerSubscriber","NONE","CombineLatestOperator","resultSelector","CombineLatestSubscriber","observables","toRespond","unused","_outerValue","oldVal","_tryResultSelector","isInteropObservable","scheduleObservable","schedulePromise","isIterable","Symbol_iterator","scheduleIterable","from","SimpleInnerSubscriber","SimpleOuterSubscriber","innerSubscribe","mergeMap","concurrent","ii","MergeMapOperator","MergeMapSubscriber","hasCompleted","buffer","_tryNext","_innerSub","ish","innerSubscription","mergeAll","concatAll","merge","last","filter","FilterOperator","FilterSubscriber","DebounceTimeOperator","dueTime","DebounceTimeSubscriber","debouncedSubscription","lastValue","clearDebounce","dispatchNext","debouncedNext","defaultIfEmpty","defaultValue","DefaultIfEmptyOperator","DefaultIfEmptySubscriber","isEmpty","DistinctUntilChangedOperator","compare","DistinctUntilChangedSubscriber","hasKey","y","throwIfEmpty","errorFactory","defaultErrorFactory","ThrowIfEmptyOperator","ThrowIfEmptySubscriber","take","TakeOperator","total","TakeSubscriber","FinallyOperator","callback","FinallySubscriber","MulticastOperator","shareSubjectFactory","share","subjectOrSubjectFactory","shareReplay","configOrBufferSize","_b","_c","useRefCount","isComplete","shareReplayOperator","switchMap","SwitchMapOperator","SwitchMapSubscriber","takeUntil","notifier","TakeUntilOperator","takeUntilSubscriber","TakeUntilSubscriber","notifierSubscription","seenValue","TakeWhileOperator","inclusive","TakeWhileSubscriber","nextOrComplete","predicateResult","Boolean","defaultThrottleConfig","leading","trailing","throttleTime","duration","ThrottleTimeOperator","ThrottleTimeSubscriber","_hasTrailingValue","_trailingValue","throttled","dispatchNext$1","clearThrottle","isDefined","o","changesProcessed","guidRegex","isGuid","test","absoluteUrlRegex","isAbsoluteUrl","url","appendOrIncrementNumber","lastNumber","parseInt","UUID","hexBytes","substr","n","buf","bufIdx","crypto","getRandomValues","Uint8Array","randomBytesBuffered","uuidBin","initialize","history","SceneHistory","undoneChanges","ignoreChangesFromObjects","undoneState","updateState","getMergedChanges","updateUndoneState","changePart","change","description","nodes","node","properties","flatMap","property","ignoreUntilStable","clear","start","stop","addChange","processFunction","undoFunction","newChange","openChange","redoFunction","lastChangeSet","processChangeSet","reverse","restore","previousValue","operation","CreateAndPush","Push","Unshift","Shift","unshift","isModified","Pop","Remove","Splice","newValue","restoreItem","redo","lastUndoneChangeSet","createAndPush","undoTo","redoTo","changeEvent","originalValue","lastChangeOnObject","lastChangeOnObjectProperty","find","changeEvents","removeLastChange","changeSet","process","nextTick","__arrayProtoFunctions","arrayFunc","TrackedArray","parentObj","propertyName","changesIgnored","historyIgnored","super","$isTracked","items","setPropertyIsModified","res","addedToChildArray","destroy","deleteCount","insert","resultItem","removedFromChildArray","reorder","TrackChanges","options","onCreateCallback","onCreate","isPropertyEqualOverride","isPropertyEqual","onDestroy","targetClass","TrackedClass","$changesTracked","trackingState","baseObject","originalValues","defineProperties","_trackChanges","commitChanges","inherited","originalValuesMap","oldValue","proxy","changedValuesMap","updateIsModified","nodeChangesProcessed","registerDeleted","inheritedDestroyFunc","getPrototypeOf","trackChanges","rawNode","Proxy","currentValue","isPropertyModified","isEqualToOriginal","registerNewObject","trackedNewProperty","_changesIgnoredProperties","arrayHistoryIgnored","_historyIgnoredProperties","registerChangedState","prop","NodeReference","classRef","propertyKey","parameterIndex","nodeReferences","limitKeys","IgnoreChanges","IgnoreHistory","TrackChangesNode","origAddedToChildArray","arrayPropName","_parent","calculateParents","origDestroy","children","getChildren","child","_manager","unregister","z","isKbVector","g","isKbColor","w","isKbQuaternion","equal","Kb3dModel","$type","combinedConstructor","trackChangesFactory","BaseKb3dModel","Enumerable","descriptor","injectParameters","Inject","ignore","getConstructorParamsForModel","modelDef","paramsForType","sortBy","Kb3dService","eParentChangeBehavior","eNodeError","eMeshChangeTypes","_level","_errors","_selected","_expanded","generate","_dynamicId","_token","withKey","_rootScene","__rootScene","setRootScene","scene","metadata","_metadata","val","clone","serialize","serializeToJson","prettyPrint","deletion","_parentScene","register","getChildArrayForType","getChildArray","addChildren","removeChildren","insertChildAfter","afterChild","newChild","arr1","arr2","getAncestors","ancestors","getNamePath","path","self","includeSelf","alwaysIncludeChildren","included","filterByType","filterNodes","_sketchControlPoints","getSpaceParent","ALL_GEOMETRY","VertexIndices","VertexPositions","VertexNormals","VertexUvs","Submeshes","ALL_CHANGES","Transform","isFeatureNode","dependsOn","_dependsOn","affects","_affects","affectedBy","_affectedBy","_suppressed","defaults","enabled","label","fontFamily","fontSize","lineSpacing","preloadFontStylesheet","color","targetId","billboard","isPickable","keepOnTop","lineStyle","horizontalTextAlign","verticalBoxAlign","horizontalBoxAlign","offset","target2Id","lineCapStyle","axis","lineThickness","dimensionGap","dimensionUnitPrecision","labelOffset","scaleWithCamera","positionMode","position","direction","angle","positionZeroToOne","interpolateDirection","arc","pivot","meshId","distance","addEndMesh","minDisplacement","maxDisplacement","capEnd","capStart","joinType","translation","cloneGeometry","intersect","normal","removeGeometry","rotationAxis","rotationAngle","scale","center","smooth","PatternFeature","uvMapping","materialId","projectionAxis","uTile","vTile","uOffset","vOffset","mode","end","mitredExtrusion","generatesShadows","shadowQuality","shadowResolution","shadowFilter","processTransparency","intensity","visible","useEuler","eulerRotation","scaling","fixed","groundColor","ShadowLightNode","coneAngle","workflowMode","backFaceCulling","directIntensity","emissiveIntensity","environmentIntensity","specularIntensity","metallic","roughness","microSurface","useSpecularOverAlpha","useRadianceOverAlpha","ambientColor","albedoColor","specularColor","reflectionColor","anisotropyEnabled","anisotropicDirection","anisotropicIntensity","subsurfaceThickness","subsurfaceRefractionEnabled","subsurfaceReflectionEnabled","reflectionType","probePositionTargetId","subsurfaceRefractionIndex","subsurfaceRefractionIntensity","subsurfaceRefractionLinkTransparency","subsurfaceTranslucencyEnabled","subsurfaceTranslucencyIntensity","subsurfaceUseAlbedoForTintColor","subsurfaceTintColor","clearcoatEnabled","clearcoatIntensity","clearcoatTintEnabled","clearcoatTintColor","clearcoatTintIntensity","archived","location","alpha","zOffset","doubleSided","legacyTextureStacking","viewMode","emissiveColor","hasTransparency","uScale","vScale","uAngle","vAngle","wAngle","uPivot","vPivot","wPivot","width","height","strength","tile","maintainAspect","useClippingMask","blendMode","horizontalAlign","verticalAlign","horizontalFit","verticalFit","flipNormalsRg","text","connector1","connector2","mateType","flip","offsetX","offsetY","offsetZ","offsetAngle","depth","tessellation","sideOrientation","flipSurfaceOrientation","receiveShadows","castShadows","checkCollisions","opacity","radiusTop","radiusBottom","radius","subdivisions","capMode","GroundNode","segments","direction2","addFace","closePath","tracePath","pathColor","flipPath","clonePath","boundingXOverride","boundingYOverride","boundingZOverride","pathGroupKey","holeForPathGroup","type","latitudeStart","latitudeSlice","curveRadius","charSpacing","fontFile","thickness","background","environment","environmentBlur","backgroundColor","renderMode","edgeColor","edgeWidth","edgeDisplayTolerance","viewpointId","frameViewpoint","progressiveTextureLoad","enableDeferredSceneUpdates","enableGrid","enableGridAxes","gridSize","gridStepSize","gridMajorInterval","gridNormal","environmentRotation","connectorSize","enableFxaaPostProcess","enableLensEffectsPostProcess","enableSSAOPostProcess","textureSamplingMode","fullscreenEnabled","arEnabled","xrAllowScaling","unitsPerMeter","xrAddShadow","xrLightingEstimation","enableHoverEffects","enableSceneOptimization","loadingExperience","loadingImagePath","dimensionUnitScalar","dimensionUnitSymbol","conversionMap","targetMode","projection","targetPoint","targetMeshId","rotation","fov","allowOrbit","allowPan","allowZoom","limitMinimumTargetDistance","minTargetDistance","maxTargetDistance","orbitAxis","limitLatOrbit","limitMinLatitude","limitMaxLatitude","limitLngOrbit","limitMinLongitude","limitMaxLongitude","NEVER_INHERIT","isMaterialNode","_materialService","_overriddenPropertiesSet","_initialized","materials","_materials","overriddenProperties","_overriddenProperties","textures","_textures","init","refreshParentMaterial","texture","overrideProperty","parentMaterialId","inheritProperty","_parentMaterial","getTextureByName","f","inheritFromMaterial","parentMat","manager","_key","currentTextures","tex","indexOnParent","texDef","channel","findIndex","currentTexture","textureNode","deserialize","_external","getMaterialById","parentMaterial","data","getMediaReferences","refs","AndOrNotEvaluator","query","evaluateCallback","_HandleParenthesisContent","Eval","parenthesisContent","or","split","ori","_SimplifyNegation","and","j","andj","substring","booleanString","Tags","_tags","hasTags","HasTags","addTags","tagsString","AddTagsTo","removeTags","RemoveTagsFrom","matchesTagsQuery","tagsQuery","MatchesQuery","tags","asString","tagsArray","tag","_AddTagTo","EnableFor","_RemoveTagFrom","Scalar","epsilon","abs","isNaN","min","log","LOG2E","log2","floor","NaN","Infinity","normalized","current","num","Repeat","tx","to","Clamp","maxDelta","Sign","DeltaAngle","MoveTowards","amount","value1","tangent1","value2","tangent2","squared","cubed","t2","number","percent","TwoPi","PI","HCF","ToGammaSpace","ToLinearSpace","PHI","sqrt","Epsilon","ArrayTools","itemBuilder","BuildArray","observedArrayFunctions","_ObserveArray","unObserveFunctions","functionName","oldFunction","newFunction","previousLength","returnValue","previous","_observeArrayfunction","unObserveFunction","_RegisteredTypes","RegisterClass","className","GetClass","fqdn","PerformanceConfigurator","use64bits","MatrixTrackPrecisionChange","MatrixUse64Bits","MatrixTrackedMatrices","matrix","_m","MatrixCurrentType","Float32Array","EventState","mask","skipNextObservers","currentTarget","Observer","scope","_willBeUnregistered","unregisterOnNextCall","_remove","onErrorObservable","ret","notifyObservers","catch","_observers","onObserverAdded","notifyIfTriggered","_numObserversMarkedAsDeleted","_hasNotified","_eventState","_onObserverAdded","insertFirst","unregisterOnFirstCall","_lastNotifiedValue","notifyObserver","addOnce","_deferUnregister","removeCallback","updateCounter","makeObserverTopPriority","makeObserverBottomPriority","eventData","userInfo","lastReturnValue","hasObservers","cleanLastNotifiedState","hasSpecificMask","EngineStore","LastCreatedEngine","Instances","LastCreatedScene","_LastCreatedScene","OnEnginesDisposedObservable","UseFallbackTexture","FallbackTexture","_ExtractAsInt","Vector2","getClassName","getHashCode","hash","toArray","FromArrayToRef","asArray","copyFrom","copyFromFloats","otherVector","addToRef","addInPlace","addVector3","subtract","subtractToRef","subtractInPlace","multiplyInPlace","multiply","multiplyToRef","multiplyByFloats","divide","divideToRef","divideInPlace","negate","negateInPlace","negateToRef","scaleInPlace","scaleToRef","scaleAndAddToRef","equals","equalsWithEpsilon","WithinEpsilon","fract","rotateToRef","cos","sin","lengthSquared","normalize","NormalizeToRef","RandomRange","ZeroReadOnly","_ZeroReadOnly","value3","value4","part1","part2","part3","part4","Hermite1stDerivativeToRef","left","right","vector","newVector","transformation","TransformToRef","p0","p1","p2","sign","DistanceSquared","CenterToRef","ref","segA","segB","l2","Distance","Dot","proj","Zero","Vector3","_x","_isDirty","_y","_z","toQuaternion","Quaternion","RotationYawPitchRoll","addInPlaceFromFloats","subtractFromFloatsToRef","subtractFromFloats","getNormalToRef","theta","acos","phi","atan2","applyRotationQuaternionToRef","q","vx","vy","vz","qx","qy","qz","qw","_w","ty","tz","applyRotationQuaternionInPlace","applyRotationQuaternion","projectOnPlane","plane","origin","projectOnPlaneToRef","V","MathTmp","denom","pow","setAll","scaledV","equalsToFloats","minimizeInPlace","other","minimizeInPlaceFromFloats","maximizeInPlace","maximizeInPlaceFromFloats","isNonUniformWithinEpsilon","absX","absY","absZ","isNonUniform","hasAZeroComponent","normalizeFromLength","reorderInPlace","order","toLowerCase","rotateByQuaternionToRef","quaternion","toRotationMatrix","Matrix","TransformCoordinatesToRef","rotateByQuaternionAroundPointToRef","point","cross","CrossToRef","normalizeToNew","normalizeToRef","reference","vector0","vector1","d0","v0","v1","dot","vNormal","forward","NormalizeRadians","diff","TmpVectors","PitchYawRollToMoveBetweenPointsToRef","slerp","vector0Dir","vector1Dir","vector0Length","vector1Length","scale0","scale1","omega","invSin","Lerp","goal","deltaTime","lerpTime","SlerpToRef","FromArray","UpReadOnly","_UpReadOnly","DownReadOnly","_DownReadOnly","RightReadOnly","_RightReadOnly","LeftReadOnly","_LeftReadOnly","LeftHandedForwardReadOnly","_LeftHandedForwardReadOnly","RightHandedForwardReadOnly","_RightHandedForwardReadOnly","LeftHandedBackwardReadOnly","_LeftHandedBackwardReadOnly","RightHandedBackwardReadOnly","_RightHandedBackwardReadOnly","OneReadOnly","_OneReadOnly","rightHandedSystem","TransformCoordinatesFromFloatsToRef","rx","ry","rz","rw","TransformNormalToRef","TransformNormalFromFloatsToRef","ClampToRef","LerpToRef","world","transform","viewport","ProjectToRef","cw","ch","cx","cy","viewportMatrix","FromValuesToRef","inDirection","ReflectToRef","viewportWidth","viewportHeight","Unproject","IdentityReadOnly","view","UnprojectToRef","UnprojectFloatsToRef","sourceX","sourceY","sourceZ","invert","screenSource","isNDCHalfZRange","_UnprojectFromInvertedMatrixToRef","p1p0","p2p0","p2p1","vectorp0","p1p0L","p2p0L","p2p1L","nl","l","cosA","projVector","v2","projP","s0","s1","s2","edge","e0","e1","tmp2","tmp3","e0proj","e0projL","cosG","triProj","axis1","axis2","axis3","RotationFromAxisToRef","quat","RotationQuaternionFromAxisToRef","toEulerAnglesToRef","Up","Down","Forward","Backward","Right","Left","One","Vector4","toVector3","otherQuaternion","q1","conjugateToRef","conjugateInPlace","conjugate","invertInPlace","inv","toEulerAngles","zAxisY","limit","sqw","sqz","sqx","sqy","asin","FromQuaternionToRef","fromRotationMatrix","FromRotationMatrixToRef","m11","m12","m13","m21","m22","m23","m31","m32","m33","trace","quat0","quat1","RotationAxisToRef","RotationYawPitchRollToRef","vec","vecFrom","vecTo","yaw","pitch","roll","halfRoll","halfPitch","halfYaw","sinRoll","cosRoll","sinPitch","cosPitch","sinYaw","cosYaw","beta","gamma","RotationAlphaBetaGammaToRef","halfGammaPlusAlpha","halfGammaMinusAlpha","halfBeta","rotMat","FromXYZAxesToRef","up","FromLookDirectionLHToRef","LookDirectionLHToRef","FromLookDirectionRHToRef","LookDirectionRHToRef","Identity","num2","num3","num4","flag","num5","num6","Use64Bits","markAsUpdated","updateFlag","_UpdateFlagSeed","_isIdentity","_isIdentity3x2","_isIdentityDirty","_isIdentity3x2Dirty","_updateIdentityStatus","isIdentity","isIdentityDirty","isIdentity3x2","isIdentity3x2Dirty","isIdentityAs3x2","determinant","m00","m01","m02","m03","m10","m20","m30","det_22_33","det_21_33","det_21_32","det_20_33","det_20_32","det_20_31","invertToRef","reset","resultM","otherM","addToSelf","IdentityToRef","cofact_00","cofact_01","cofact_02","cofact_03","det","detInv","det_12_33","det_11_33","det_11_32","det_10_33","det_10_32","det_10_31","det_12_23","det_11_23","det_11_22","det_10_23","det_10_22","det_10_21","cofact_10","cofact_11","cofact_12","cofact_13","cofact_20","cofact_21","cofact_22","cofact_23","cofact_30","cofact_31","cofact_32","cofact_33","addAtIndex","multiplyAtIndex","setTranslationFromFloats","addTranslationFromFloats","setTranslation","vector3","getTranslation","getTranslationToRef","removeRotationAndScaling","copyToArray","multiplyToArray","tm0","tm1","tm2","tm3","tm4","tm5","tm6","tm7","tm8","tm9","tm10","tm11","tm12","tm13","tm14","tm15","om0","om1","om2","om3","om4","om5","om6","om7","om8","om9","om10","om11","om12","om13","om14","om15","om","decomposeToTransformNode","rotationQuaternion","decompose","preserveScalingNode","signX","signY","signZ","sx","sy","sz","getRow","getRowToRef","rowVector","setRow","row","setRowFromFloats","transpose","TransposeToRef","transposeToRef","toNormalMatrix","getRotationMatrix","getRotationMatrixToRef","toggleModelMatrixHandInPlace","toggleProjectionMatrixHandInPlace","_IdentityReadOnly","initialM11","initialM12","initialM13","initialM14","initialM21","initialM22","initialM23","initialM24","initialM31","initialM32","initialM33","initialM34","initialM41","initialM42","initialM43","initialM44","ComposeToRef","x2","y2","z2","xx","xy","xz","yy","yz","zz","wx","wy","wz","FromValues","zero","RotationXToRef","RotationYToRef","RotationZToRef","c1","Cross","ScalingToRef","TranslationToRef","startValue","endValue","gradient","startM","endM","DecomposeLerpToRef","startScale","startRotation","startTranslation","endScale","endRotation","endTranslation","resultScale","resultRotation","resultTranslation","eye","LookAtLHToRef","xAxis","yAxis","zAxis","xSquareLength","ex","ey","ez","LookAtRHToRef","back","znear","zfar","halfZRange","OrthoLHToRef","mtxConvertNDCToHalfZRange","bottom","top","OrthoOffCenterLHToRef","i0","i1","OrthoOffCenterRHToRef","projectionPlaneTilt","rot","tan","aspect","reverseDepthBufferMode","PerspectiveFovLHToRef","isVerticalFovFixed","PerspectiveFovRHToRef","rightHanded","rightHandedFactor","upTan","upDegrees","downTan","downDegrees","leftTan","leftDegrees","rightTan","rightDegrees","xScale","yScale","zmin","zmax","rm","mm","ReflectionToRef","temp","temp2","temp3","xaxis","yaxis","zaxis","zw","zx","yw","xw","BuildTuple","_WarnImport","colorChannelToLinearSpace","colorChannelToLinearSpaceExact","colorChannelToGammaSpace","colorChannelToGammaSpaceExact","Color3","toColor4","Color4","toLuminance","otherColor","equalsFloats","clampToRef","toHexString","intR","round","intG","intB","ToHex","toHSV","toHSVToRef","h","dm","toLinearSpace","exact","convertedColor","toLinearSpaceToRef","toGammaSpace","toGammaSpaceToRef","hue","saturation","chroma","HSVtoRGBToRef","hex","FromInts","Black","BlackReadOnly","_BlackReadOnly","returnAsColor3","intA","color3","colors","colors4","newIndex","TmpColors","__decoratorInitialStore","__mergedStore","_copySource","creationFunction","instanciate","GetTags","classStore","getMergedStore","textureMap","propertyDescriptor","sourceProperty","propertyType","SerializationHelper","AllowLoadingUniqueId","cloneTexturesOnlyOnce","uniqueId","isRenderTarget","classKey","store","currentKey","initialStore","generateSerializableMember","sourceName","getDirectStore","expandToProperty","targetKey","setCallback","generateExpandMember","serializeAsTexture","serializeAsColor3","serializeAsFresnelParameters","serializeAsVector2","serializeAsVector3","serializeAsMeshReference","serializeAsColor4","serializeAsImageProcessingConfiguration","serializeAsMatrix","animations","animationIndex","animation","entity","serializationObject","serializedProperties","targetPropertyName","rootUrl","dest","_TextureParser","_FresnelParametersParser","getLastMeshById","_ColorCurvesParser","_ImageProcessingConfigurationParser","getCameraById","ParseProperties","nativeOverride","jsFunc","params","func","_native","nativeFunc","_InternalNodeDataInfo","_doNotSerialize","_isDisposed","_sceneRootNodesIndex","_isEnabled","_isParentEnabled","_isReady","_onEnabledStateChangedObservable","_onClonedObservable","Node","constructorFunc","_NodeConstructors","accessibilityTag","_accessibilityTag","onAccessibilityTagChangedObservable","doNotSerialize","_nodeDataStorage","_parentNode","isDisposed","previousParentNode","_children","_addToSceneRootNodes","_removeFromSceneRootNodes","_syncParentEnabledState","_serializeAsParent","parentId","_scene","rootNodes","lastIdx","animationPropertiesOverride","_animationPropertiesOverride","onDispose","_onDisposeObserver","onDisposeObservable","onEnabledStateChangedObservable","onClonedObservable","reservedDataStore","_parentContainer","_ranges","onReady","_currentRenderId","_parentUpdateId","_childUpdateId","_waitingParentId","_waitingParentInstanceIndex","_waitingParsedUniqueId","_cache","_worldMatrix","_worldMatrixDeterminant","_worldMatrixDeterminantIsDirty","_isNode","_behaviors","getUniqueId","_initCache","getScene","getEngine","addBehavior","behavior","attachImmediately","isLoading","onDataLoadedObservable","attach","removeBehavior","detach","behaviors","getBehaviorByName","getWorldMatrix","getRenderId","computeWorldMatrix","_getWorldMatrixDeterminant","worldMatrixFromCache","updateCache","force","isSynchronized","_updateCache","_getActionManagerForTrigger","trigger","_initialCall","_ignoreParentClass","_isSynchronized","_markSyncedWithParent","isSynchronizedWithParent","isReady","_completeCheck","markAsDirty","_property","MAX_VALUE","isEnabled","checkAncestors","setEnabled","isDescendantOf","ancestor","_getDescendants","results","directDescendantsOnly","getDescendants","getChildMeshes","cullingStrategy","_setReady","getAnimationByName","createAnimationRange","_AnimationRangeFactory","nAnimations","createRange","deleteAnimationRange","deleteFrames","deleteRange","getAnimationRange","newParent","doNotCloneChildren","Clone","directDescendants","getAnimationRanges","animationRanges","beginAnimation","loop","speedRatio","onAnimationEnd","range","serializeAnimationRanges","serializationRanges","localRange","_force","dispose","doNotRecurse","disposeMaterialAndTextures","parsedNode","ranges","getHierarchyBoundingVectors","includeDescendants","incrementRenderId","thisAbstractMesh","getBoundingInfo","subMeshes","boundingInfo","boundingBox","minimumWorld","maximumWorld","descendants","descendant","childMesh","getTotalVertices","minBox","maxBox","CheckExtends","IsWindowObjectExist","window","IsNavigatorAvailable","navigator","IsDocumentAvailable","document","GetDOMTextContent","element","firstChild","nodeType","textContent","nextSibling","_name","_from","_to","Logger","entry","_LogLimitOutputs","level","MessageLimitReached","_Levels","_LogCache","OnNewCacheEntry","padStr","date","getHours","getMinutes","getSeconds","_CheckLimit","formattedMessage","_FormatMessage","logFunc","_AddLogEntry","_GenerateLimitMessage","LogCache","errorsCount","LogLevels","Log","_LogDisabled","Warn","MessageLogLevel","WarningLogLevel","ErrorLogLevel","_LogEnabled","NoneLogLevel","AllLogLevel","CloneValue","destinationObject","DeepCopier","doNotCopyList","mustCopyList","props","getOwnPropertyNames","GetAllPropertyNames","endsWith","sourceValue","typeOfSourceValue","clonedValue","PrecisionDate","Now","performance","WebRequest","_xhr","XMLHttpRequest","_requestURL","_injectCustomRequestHeaders","_shouldSkipRequestModifications","CustomRequestHeaders","setRequestHeader","SkipRequestModificationForBabylonCDN","includes","onprogress","readyState","status","statusText","response","responseURL","responseText","responseType","timeout","addEventListener","listener","removeEventListener","abort","send","body","open","method","update","CustomRequestModifiers","getResponseHeader","FilesInputStore","FilesToLoad","BaseError","_setPrototypeOf","proto","ErrorCodes","RuntimeError","errorCode","innerError","EncodeArrayBufferToBase64","keyStr","chr1","chr2","chr3","enc1","enc2","enc3","enc4","output","bytes","ArrayBuffer","isView","byteOffset","byteLength","DecodeBase64ToString","base64Data","atob","ShaderCodeNode","isValid","preprocessors","_d","_e","_f","_g","line","processor","lineProcessor","isFragment","processingContext","attributeKeyword","attributeKeywordName","varyingKeyword","varyingFragmentKeywordName","varyingVertexKeywordName","attributeProcessor","startsWith","varyingProcessor","varyingCheck","uniformProcessor","uniformRegexp","lookForClosingBracketForUniformBuffer","uniformBufferProcessor","uniformBufferRegexp","textureProcessor","textureRegexp","endOfUniformBufferProcessor","additionalDefineKey","additionalDefineValue","ShaderCodeCursor","_lines","currentLine","lineIndex","canRead","lines","trimmedLine","semicolonIndex","subLine","ShaderCodeConditionNode","ShaderCodeTestNode","testExpression","isTrue","ShaderDefineExpression","postfix","_OperatorPriority","infix","cacheItem","_InfixToPostfixCache","accessTime","stackIdx","pushOperand","operand","_Stack","peek","idx","InfixToPostfixCacheLimitSize","ClearCache","sortedCache","entries","InfixToPostfixCacheCleanupSize","ShaderDefineIsDefinedOperator","define","not","condition","ShaderDefineOrOperator","leftOperand","rightOperand","ShaderDefineAndOperator","ShaderDefineArithmeticOperator","testValue","ShaderLanguage","regexSE","regexSERevert","regexShaderInclude","regexShaderDecl","regexLightX","regexX","reusableMatches","ShaderProcessor","initializeShaders","sourceCode","engine","preProcessShaderCode","_ProcessIncludes","codeWithIncludes","processCodeAfterIncludes","migratedCode","_ProcessShaderConversion","_ApplyPreProcessing","vertexCode","fragmentCode","finalizeShaders","noPrecision","shouldUseHighPrecisionShader","expression","exec","operators","indexOperator","infixToPostfix","_ExtractOperation","command","_BuildSubExpression","cursor","rootNode","ifNode","_MoveCursor","first5","elseNode","elifNode","_BuildExpression","matches","_MoveCursorRegex","newRootNode","_MoveCursorWithinIf","newNode","defines","shaderLanguage","GLSL","version","platformName","_getGlobalDefines","preparedSourceCode","_ProcessPrecision","parseGLES3","_PreparePreProcessors","preProcessor","_EvaluatePreProcessors","postProcessor","_features","needShaderCodeInlining","inlineShaderCode","parts","keepProcessing","includeFile","supportsUniformBuffers","includesShadersStore","includeShaderUrl","shadersRepository","_FileToolsLoadFile","fileContent","includeContent","splits","RegExp","indexString","indexSplits","minIndex","maxIndex","sourceIncludeContent","indexParameters","newParts","part","splitPart","onSuccess","onProgress","offlineProvider","useArrayBuffer","onError","ShaderStore","ShadersRepository","ShadersRepositoryWGSL","ShadersStore","ShadersStoreWGSL","IncludesShadersStore","IncludesShadersStoreWGSL","Effect","EngineShaderStore","repo","onBindObservable","_onBindObservable","baseName","attributesNamesOrOptions","uniformsNamesOrEngine","samplers","fallbacks","onCompiled","onBind","onCompileObservable","_wasPreviouslyReady","_forceRebindOnNextCall","_wasPreviouslyUsingInstances","_bonesComputationForcedToCPU","_uniformBuffersNames","_multiTarget","_samplers","_compilationError","_allFallbacksProcessed","_uniforms","_fallbacks","_vertexSourceCodeOverride","_fragmentSourceCodeOverride","_transformFeedbackVaryings","_pipelineContext","_vertexSourceCode","_fragmentSourceCode","_vertexSourceCodeBeforeMigration","_fragmentSourceCodeBeforeMigration","_rawVertexSourceCode","_rawFragmentSourceCode","vertexSource","fragmentSource","processFinalCode","attributes","_engine","_attributesNames","_uniformsNames","uniformsNames","_samplerList","_indexParameters","transformFeedbackVaryings","multiTarget","_shaderLanguage","uniformBuffersNames","_uniformBuffersNamesList","_attributeLocationByName","_UniqueIdSeed","hostDocument","getHostDocument","vertexElement","getElementById","vertex","fragmentElement","fragment","_processingContext","_getShaderProcessingContext","processorOptions","_shouldUseHighPrecisionShader","_getShaderProcessor","GetShadersRepository","GetIncludesShadersStore","shaderPlatformName","useReverseDepthBuffer","shaderCodes","shadersLoaded","migratedVertexCode","Process","migratedFragmentCode","codeBeforeMigration","finalShaders","Finalize","_useFinalCode","_loadShader","Initialize","spectorName","WGSL","_prepareEffect","_isReadyInternal","getPipelineContext","getAttributesNames","getAttributeLocation","_attributes","getAttributeLocationByName","getAttributesCount","getUniformIndex","uniformName","getUniform","getSamplers","getUniformNames","getUniformBuffersNames","getIndexParameters","getCompilationError","allFallbacksProcessed","executeWhenCompiled","effect","isAsync","_checkIsReady","previousPipelineContext","_processCompilationErrors","shader","optionalKey","HTMLElement","shaderStore","GetShadersStore","shaderUrl","_loadFile","vertexSourceCode","_getVertexShaderCode","fragmentSourceCode","_getFragmentShaderCode","vertexSourceCodeBeforeMigration","fragmentSourceCodeBeforeMigration","rawVertexSourceCode","rawFragmentSourceCode","_rebuildProgram","scenes","markAllMaterialsAsDirty","_handlesSpectorRebuildCallback","attributesNames","createPipelineContext","rebuildRebind","_preparePipelineContext","_executeWhenRenderingStateIsCompiled","_fillEffectInformation","bindSamplers","unBindMesh","_deletePipelineContext","_getShaderCodeAndErrorLine","code","regexp","errorLine","lineNumber","uniform","attribute","LogShaderCodeOnCompilationError","lineErrorVertex","lineErrorFragment","notifyErrors","hasMoreFallbacks","isSupported","_bindTexture","setTexture","setDepthStencilTexture","setTextureArray","exName","initialPos","currentExName","channelIndex","setTextureFromPostProcess","postProcess","setTextureFromPostProcessOutput","bindUniformBuffer","bufferName","_BaseCache","useUBOBindingCache","bindUniformBufferBase","bindUniformBlock","blockName","setInt","setInt2","setInt3","setInt4","setIntArray","setIntArray2","setIntArray3","setIntArray4","setUInt","setUInt2","setUInt3","setUInt4","setUIntArray","setUIntArray2","setUIntArray3","setUIntArray4","setFloatArray","setArray","setFloatArray2","setArray2","setFloatArray3","setArray3","setFloatArray4","setArray4","setMatrices","matrices","setMatrix","setMatrix3x3","setMatrix2x2","setFloat","setBool","bool","setVector2","vector2","setFloat2","setVector3","setFloat3","setVector4","vector4","setQuaternion","setFloat4","setColor3","setColor4","setDirectColor4","color4","_releaseEffect","pixelShader","vertexShader","DepthCullingState","_isDepthTestDirty","_isDepthMaskDirty","_isDepthFuncDirty","_isCullFaceDirty","_isCullDirty","_isZOffsetDirty","_isFrontFaceDirty","isDirty","_zOffset","zOffsetUnits","_zOffsetUnits","cullFace","_cullFace","cull","_cull","depthFunc","_depthFunc","depthMask","_depthMask","depthTest","_depthTest","frontFace","_frontFace","gl","enable","CULL_FACE","disable","DEPTH_TEST","POLYGON_OFFSET_FILL","polygonOffset","StencilState","ALWAYS","funcRef","funcMask","opStencilFail","KEEP","opDepthFail","opStencilDepthPass","REPLACE","stencilFunc","stencilFuncRef","stencilFuncMask","stencilOpStencilFail","stencilOpDepthFail","stencilOpStencilDepthPass","stencilMask","stencilTest","AlphaState","_blendFunctionParameters","_blendEquationParameters","_blendConstants","_isBlendConstantsDirty","_alphaBlend","_isAlphaBlendDirty","_isBlendFunctionParametersDirty","_isBlendEquationParametersDirty","alphaBlend","setAlphaBlendConstants","setAlphaBlendFunctionParameters","value0","setAlphaEquationParameters","rgb","BLEND","blendFuncSeparate","blendEquationSeparate","blendColor","InternalTextureSource","InternalTexture","TextureSampler","wrapU","_cachedWrapU","wrapV","_cachedWrapV","wrapR","_cachedWrapR","anisotropicFilteringLevel","_cachedAnisotropicFilteringLevel","comparisonFunction","_comparisonFunction","useMipMaps","_useMipMaps","samplingMode","setParameters","compareSampler","generateMipMaps","_uniqueId","_setUniqueId","_source","delayAllocation","isCube","is3D","is2DArray","isMultiview","samples","format","onLoadedObservable","onRebuildCallback","baseWidth","baseHeight","baseDepth","invertY","_invertVScale","_associatedChannel","Unknown","_buffer","_bufferView","_bufferViewArray","_bufferViewArrayArray","_size","_extension","_files","_workingCanvas","_workingContext","_cachedCoordinatesMode","_isDisabled","_compression","_sphericalPolynomial","_sphericalPolynomialPromise","_sphericalPolynomialComputed","_lodGenerationScale","_lodGenerationOffset","_useSRGBBuffer","_lodTextureHigh","_lodTextureMid","_lodTextureLow","_isRGBD","_linearSpecularLOD","_irradianceTexture","_hardwareTexture","_maxLodLevel","_references","_gammaSpace","_Counter","_createHardwareTexture","incrementReferences","updateSize","updateTextureDimensions","_rebuild","swapAndSetIsReady","proxyInternalTexture","_swapAndDie","Temp","Url","createTexture","_originalUrl","Raw","createRawTexture","Raw3D","createRawTexture3D","Raw2DArray","createRawTexture2DArray","Dynamic","createDynamicTexture","updateDynamicTexture","getRenderingCanvas","Cube","createCubeTexture","CubeRaw","createRawCubeTexture","CubeRawRGBD","CubePrefiltered","createPrefilteredCubeTexture","swapAll","setUsage","cache","getLoadedTexturesCache","_releaseTexture","WebGLShaderProcessor","getCaps","drawBuffersExtension","regex","varyingRegex","WebGL2ShaderProcessor","varying","_isFragment","hasDrawBuffersExtension","search","hasOutput","DataBuffer","underlyingResource","references","capacity","is32Bits","WebGLDataBuffer","resource","WebGLPipelineContext","_valueCache","vertexCompilationError","fragmentCompilationError","programLinkError","programValidationError","isParallelCompiled","program","_isRenderingStateCompiled","uniforms","samplerList","getUniforms","attr","getAttributes","_cacheMatrix","_cacheFloat2","changed","_cacheFloat3","_cacheFloat4","_getShaderSource","fragmentShader","WebGLHardwareTexture","_webGLTexture","existingTexture","_MSAARenderBuffers","hardwareTexture","addMSAARenderBuffer","releaseMSAARenderBuffers","deleteRenderbuffer","release","deleteTexture","DrawWrapper","createMaterialContext","drawContext","createDrawContext","materialContext","setEffect","resetContext","StencilStateComposer","_isStencilTestDirty","_isStencilMaskDirty","_isStencilFuncDirty","_isStencilOpDirty","_func","_funcRef","_funcMask","_opStencilFail","_opDepthFail","_opStencilDepthPass","_mask","_enabled","useStencilGlobalOnly","stencilMaterial","stencilGlobal","stencilMaterialEnabled","STENCIL_TEST","stencilOp","BufferPointer","ThinEngine","NpmPackage","Version","webGLVersion","_caps","parallelShaderCompile","_webGLVersion","_shaderProcessor","_useReverseDepthBuffer","useReverse","_depthCullingState","frameId","_frameId","disableUniformBuffers","getCreationOptions","_creationOptions","highPrecisionShaderSupported","_highPrecisionShadersAllowed","needPOTTextures","forcePOTTextures","activeRenderLoops","_activeRenderLoops","doNotHandleContextLost","_doNotHandleContextLost","_supportsHardwareTextureRescaling","framebufferDimensionsObject","dimensions","_framebufferDimensionsObject","currentViewport","_cachedViewport","emptyTexture","_emptyTexture","emptyTexture3D","_emptyTexture3D","emptyTexture2DArray","_emptyTexture2DArray","emptyCubeTexture","_emptyCubeTexture","faceData","cubeData","isWebGPU","_isWebGPU","_shaderPlatformName","snapshotRendering","activate","snapshotRenderingMode","_snapshotRenderingMode","snapshotRenderingReset","OffscreenCanvas","canvas","createElement","createCanvas","_CreateCanvas","createCanvasImage","canvasOrContext","antialias","adaptToDeviceRatio","_h","_j","_k","_l","isFullscreen","cullBackFaces","renderEvenInBackground","preventCacheWipeBetweenFrames","validateShaderPrograms","hasOriginBottomLeft","_uniformBuffers","_storageBuffers","_windowIsBackground","_badOS","_badDesktopOS","_renderingQueueLaunched","onContextLostObservable","onContextRestoredObservable","_contextWasLost","disableVertexArrayObjects","_colorWrite","_colorWriteChanged","_stencilStateComposer","_stencilState","_alphaState","_alphaMode","_alphaEquation","_internalTexturesCache","_renderTargetWrapperCache","_activeChannel","_currentTextureChannel","_boundTexturesCache","_compiledEffects","_vertexAttribArraysEnabled","_uintIndicesCurrentlySet","_currentBoundBuffer","_currentFramebuffer","_dummyFramebuffer","_currentBufferPointers","_currentInstanceLocations","_currentInstanceBuffers","_vaoRecordInProgress","_mustWipeVertexAttributes","_nextFreeTextureSlots","_maxSimultaneousTextures","_maxMSAASamplesOverride","_activeRequests","_lastDevicePixelRatio","_transformTextureUrl","hostInformation","isMobile","premultipliedAlpha","onBeforeTextureInitObservable","_viewportCached","_unpackFlipYCached","enableUnpackFlipYCached","_boundUniforms","startTime","SetMatrixPrecision","useHighPrecisionMatrix","deterministicLockstep","lockstepMaxSteps","timeStep","audioEngine","stencil","_audioContext","audioEngineOptions","audioContext","_audioDestination","audioDestination","useExactSrgbConversions","_isStencilEnable","devicePixelRatio","limitDeviceRatio","_hardwareScalingLevel","getContext","_renderingCanvas","preserveDrawingBuffer","xrCompatible","userAgent","_setupMobileChecks","ua","exception","ExceptionList","targets","capture","captureConstraint","constraint","_onContextLost","evt","preventDefault","_onContextRestored","_restoreEngineAfterContextLost","_initGLContext","powerPreference","disableWebGL2Support","_gl","deleteQuery","renderbufferStorageMultisample","getContextAttributes","pixelStorei","UNPACK_COLORSPACE_CONVERSION_WEBGL","useHighPrecisionFloats","resize","_initFeatures","maxVertexAttribs","versionToLog","setAttribute","_checkForMobile","currentUA","initEngine","wipeCaches","_rebuildEffects","_rebuildComputeEffects","_rebuildBuffers","_rebuildInternalTextures","_rebuildRenderTargetWrappers","_sharedInit","currentState","internalTexture","renderTargetWrapper","ResetCache","areAllEffectsReady","uniformBuffer","storageBuffer","maxTexturesImageUnits","getParameter","MAX_TEXTURE_IMAGE_UNITS","maxCombinedTexturesImageUnits","MAX_COMBINED_TEXTURE_IMAGE_UNITS","maxVertexTextureImageUnits","MAX_VERTEX_TEXTURE_IMAGE_UNITS","maxTextureSize","MAX_TEXTURE_SIZE","maxSamples","MAX_SAMPLES","maxCubemapTextureSize","MAX_CUBE_MAP_TEXTURE_SIZE","maxRenderTextureSize","MAX_RENDERBUFFER_SIZE","MAX_VERTEX_ATTRIBS","maxVaryingVectors","MAX_VARYING_VECTORS","maxFragmentUniformVectors","MAX_FRAGMENT_UNIFORM_VECTORS","maxVertexUniformVectors","MAX_VERTEX_UNIFORM_VECTORS","getExtension","standardDerivatives","maxAnisotropy","astc","bptc","s3tc","s3tc_srgb","pvrtc","etc1","etc2","textureAnisotropicFilterExtension","uintIndices","fragmentDepthSupported","timerQuery","supportOcclusionQuery","canUseTimestampForTimerQuery","maxMSAASamples","colorBufferFloat","colorBufferHalfFloat","textureFloat","textureHalfFloat","textureHalfFloatRender","textureFloatLinearFiltering","textureFloatRender","textureHalfFloatLinearFiltering","vertexArrayObject","instancedArrays","textureLOD","texelFetch","blendMinMax","multiview","oculusMultiview","depthTextureExtension","canUseGLInstanceID","canUseGLVertexID","supportComputeShaders","supportSRGBBuffers","supportTransformFeedbacks","textureMaxLevel","texture2DArrayMaxLayerCount","MAX_ARRAY_TEXTURE_LAYERS","disableMorphTargetTexture","_glVersion","VERSION","rendererInfo","_glRenderer","UNMASKED_RENDERER_WEBGL","_glVendor","UNMASKED_VENDOR_WEBGL","VENDOR","RENDERER","HALF_FLOAT_OES","RGBA16F","RGBA32F","DEPTH24_STENCIL8","getQuery","getQueryEXT","TIMESTAMP_EXT","QUERY_COUNTER_BITS_EXT","MAX_TEXTURE_MAX_ANISOTROPY_EXT","_canRenderToFloatFramebuffer","COMPRESSED_SRGB8_ALPHA8_ASTC_4x4_KHR","COMPRESSED_SRGB_ALPHA_BPTC_UNORM_EXT","COMPRESSED_SRGB_S3TC_DXT1_EXT","COMPRESSED_SRGB_ALPHA_S3TC_DXT1_EXT","COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT","COMPRESSED_SRGB8_ETC2","COMPRESSED_SRGB8_ALPHA8_ETC2_EAC","_canRenderToHalfFloatFramebuffer","drawBuffers","drawBuffersWEBGL","DRAW_FRAMEBUFFER","FRAMEBUFFER","UNSIGNED_INT_24_8","UNSIGNED_INT_24_8_WEBGL","vertexArrayObjectExtension","createVertexArray","createVertexArrayOES","bindVertexArray","bindVertexArrayOES","deleteVertexArray","deleteVertexArrayOES","instanceExtension","drawArraysInstanced","drawArraysInstancedANGLE","drawElementsInstanced","drawElementsInstancedANGLE","vertexAttribDivisor","vertexAttribDivisorANGLE","getShaderPrecisionFormat","vertexhighp","VERTEX_SHADER","HIGH_FLOAT","fragmenthighp","FRAGMENT_SHADER","precision","blendMinMaxExtension","MAX","MAX_EXT","MIN","MIN_EXT","_glSRGBExtensionValues","SRGB","WebGL2RenderingContext","SRGB8","SRGB8_ALPHA8","sRGBExtension","SRGB_EXT","SRGB_ALPHA_EXT","forceSRGBBufferSupportState","LEQUAL","slot","forceBitmapOverHTMLImageElement","supportRenderAndCopyToLodForFloatTextures","supportDepthStencilTexture","supportShadowSamplers","uniformBufferHardCheckMatrix","allowTexturePrefiltering","trackUbosInFrame","checkUbosContentBeforeUpload","supportCSM","basisNeedsPOT","support3DTextures","needTypeSuffixInShaderConstants","supportMSAA","supportSSAO2","supportExtendedTextureFormats","supportSwitchCaseInShader","supportSyncTextureRead","needsInvertingBitmap","needToAlwaysBindUniformBuffers","supportRenderPasses","supportSpriteInstancing","_collectUbosUpdatedInFrame","isStencilEnable","_prepareWorkingCanvas","resetTextureCache","getInfo","getGlInfo","vendor","renderer","setHardwareScalingLevel","getHardwareScalingLevel","stopRenderLoop","renderFunction","_cancelFrame","_frameHandler","cancelAnimationFrame","getHostWindow","clearTimeout","_renderLoop","shouldRender","beginFrame","endFrame","_queueNewFrame","_boundRenderFunction","getAudioContext","getAudioDestination","ownerDocument","defaultView","getRenderWidth","useScreen","_currentRenderTarget","framebufferWidth","drawingBufferWidth","getRenderHeight","framebufferHeight","drawingBufferHeight","bindedRenderFunction","requester","QueueNewFrame","runRenderLoop","backBuffer","stencilStateComposer","applyStates","setBackBufferColor","textureFormat","textureType","_TempClearColorUint32","clearBufferuiv","COLOR","_TempClearColorInt32","clearBufferiv","clearColor","COLOR_BUFFER_BIT","GEQUAL","clearDepth","DEPTH_BUFFER_BIT","clearStencil","STENCIL_BUFFER_BIT","_viewport","setViewport","requiredWidth","requiredHeight","flushFramebuffer","forceSetSize","changeRatio","boundingRect","getBoundingClientRect","clientWidth","clientHeight","innerWidth","innerHeight","setSize","bindFramebuffer","rtWrapper","faceIndex","forceFullscreenViewport","lodLevel","layer","webglRTWrapper","unBindFramebuffer","_bindUnboundFramebuffer","_MSAAFramebuffer","_framebuffer","isMulti","framebufferTextureLayer","COLOR_ATTACHMENT0","framebufferTexture2D","TEXTURE_CUBE_MAP_POSITIVE_X","depthStencilTexture","_depthStencilTexture","attachment","_depthStencilTextureWithStencil","DEPTH_STENCIL_ATTACHMENT","DEPTH_ATTACHMENT","TEXTURE_2D","setState","culling","reverseSide","BACK","FRONT","setZOffset","setZOffsetUnits","CW","CCW","getDepthBuffer","setDepthBuffer","getZOffset","getZOffsetUnits","framebuffer","_currentFrameBufferIsDefaultFrameBuffer","generateMipmaps","_bindTextureDirectly","generateMipmap","disableGenerateMipMaps","onBeforeUnbind","unBindMultiColorAttachmentFramebuffer","READ_FRAMEBUFFER","blitFramebuffer","NEAREST","restoreDefaultFramebuffer","_resetVertexBufferBinding","bindArrayBuffer","_cachedVertexBuffers","createVertexBuffer","_createVertexBuffer","STATIC_DRAW","usage","vbo","createBuffer","dataBuffer","bufferData","ARRAY_BUFFER","createDynamicVertexBuffer","DYNAMIC_DRAW","_resetIndexBufferBinding","bindIndexBuffer","_cachedIndexBuffer","createIndexBuffer","indices","updatable","_normalizeIndexData","ELEMENT_ARRAY_BUFFER","BYTES_PER_ELEMENT","Uint32Array","Uint16Array","_unbindVertexArrayObject","_bindBuffer","pipelineContext","uniformLocation","getUniformBlockIndex","uniformBlockBinding","bindBuffer","updateArrayBuffer","bufferSubData","_vertexAttribPointer","indx","stride","pointer","UNSIGNED_INT","INT","vertexAttribIPointer","vertexAttribPointer","_bindIndexBufferWithCache","indexBuffer","_bindVertexBuffersAttributes","vertexBuffers","overrideVertexBuffers","unbindAllAttributes","ai","vertexBuffer","enableVertexAttribArray","getBuffer","getSize","byteStride","getIsInstanced","getInstanceDivisor","recordVertexArrayObject","vao","bindVertexArrayObject","_cachedVertexArrayObject","bindBuffersDirectly","vertexDeclaration","vertexStrideSize","_cachedEffectForVertexBuffers","attributesCount","FLOAT","bindBuffers","unbindInstanceAttributes","boundBuffer","ul","instancesBuffer","offsetLocation","releaseVertexArrayObject","_releaseBuffer","_deleteBuffer","deleteBuffer","updateAndBindInstancesBuffer","offsetLocations","bindInstancesBuffer","attributesInfo","computeStride","attributeSize","_currentEffect","attributeName","attributeType","divisor","disableInstanceAttributeByName","attributeLocation","disableInstanceAttribute","shouldClean","disableAttributeByIndex","disableVertexAttribArray","draw","useTriangles","indexStart","indexCount","instancesCount","drawElementsType","drawPointClouds","verticesStart","verticesCount","drawArraysType","drawUnIndexed","fillMode","_reportDrawCall","drawMode","_drawMode","indexFormat","UNSIGNED_SHORT","mult","drawElements","drawArrays","TRIANGLES","POINTS","LINES","LINE_LOOP","LINE_STRIP","TRIANGLE_STRIP","TRIANGLE_FAN","webGLPipelineContext","__SPECTOR_rebuildProgram","deleteProgram","createEffect","vertexToken","fragmentToken","globalDefines","fullDefines","compiledEffect","shaderVersion","_compileShader","_compileRawShader","_ConcatenateShader","createShader","NO_ERROR","tempError","getError","isContextLost","shaderSource","compileShader","getShaderSource","createRawShaderProgram","_createShaderProgram","createShaderProgram","shaderProcessingContext","shaderProgram","createProgram","attachShader","linkProgram","_finalizePipelineContext","getProgramParameter","LINK_STATUS","getShaderParameter","COMPILE_STATUS","getShaderInfoLog","getProgramInfoLog","validateProgram","VALIDATE_STATUS","deleteShader","createAsRaw","webGLRenderingState","COMPLETION_STATUS_KHR","oldHandler","getUniformLocation","getAttribLocation","enableEffect","IsWrapper","uniform1i","uniform2i","uniform3i","uniform4i","uniform1iv","uniform2iv","uniform3iv","uniform4iv","uniform1ui","uniform2ui","uniform3ui","uniform4ui","uniform1uiv","uniform2uiv","uniform3uiv","uniform4uiv","uniform1fv","uniform2fv","uniform3fv","uniform4fv","uniformMatrix4fv","uniformMatrix3fv","uniformMatrix2fv","uniform1f","uniform2f","uniform3f","uniform4f","colorMask","setColorWrite","getColorWrite","depthCullingState","alphaState","stencilState","clearInternalTexturesCache","bruteForce","_currentProgram","UNPACK_PREMULTIPLY_ALPHA_WEBGL","_getSamplingParameters","magFilter","minFilter","LINEAR","LINEAR_MIPMAP_NEAREST","LINEAR_MIPMAP_LINEAR","NEAREST_MIPMAP_LINEAR","NEAREST_MIPMAP_NEAREST","mag","_createTexture","_createInternalTexture","delayGPUTextureCreation","useSRGBBuffer","layers","filters","TEXTURE_2D_ARRAY","sizedFormat","_getRGBABufferInternalSizedFormat","internalFormat","_getInternalFormat","_getWebGLTextureType","texImage3D","texImage2D","texParameteri","TEXTURE_MAG_FILTER","TEXTURE_MIN_FILTER","TEXTURE_WRAP_S","CLAMP_TO_EDGE","TEXTURE_WRAP_T","_getUseSRGBBuffer","noMipmap","_createTextureBase","onLoad","prepareTexture","prepareTextureProcessFunction","fallback","forcedExtension","mimeType","loaderOptions","fromData","fromBlob","isBase64","originalUrl","lastDot","lastIndexOf","extension","loader","availableLoader","_TextureLoaders","canLoad","addPendingData","onLoadObserver","onInternalError","removePendingData","loadData","loadMipmap","isCompressed","loadFailed","request","onload","img","decoding","close","_FileToolsLoadImage","imageOrientation","Blob","creationFlags","_prepareWebGLTexture","potWidth","potHeight","continuationCallback","isPot","RGBA","RGB","texelFormat","UNSIGNED_BYTE","drawImage","_rescaleTexture","imageBitmapOptions","onComplete","compression","_unpackFlipY","UNPACK_FLIP_Y_WEBGL","_getUnpackAlignement","UNPACK_ALIGNMENT","_getTextureTarget","TEXTURE_CUBE_MAP","TEXTURE_3D","updateTextureSamplingMode","_setTextureParameterInteger","updateTextureWrappingMode","_getTextureWrapMode","TEXTURE_WRAP_R","_setupDepthStencilTexture","generateStencil","bilinearFiltering","samplingParameters","TEXTURE_COMPARE_FUNC","TEXTURE_COMPARE_MODE","COMPARE_REF_TO_TEXTURE","_uploadCompressedDataToTextureDirectly","lod","compressedTexImage2D","_uploadDataToTextureDirectly","imageData","babylonInternalFormat","useTextureWidthAndHeight","lodMaxWidth","lodMaxHeight","updateTextureData","xOffset","yOffset","targetForBinding","texSubImage2D","_uploadArrayBufferViewToTexture","bindTarget","_prepareWebGLTextureContinuation","GetExponentOfTwo","_setupFramebufferDepthAttachments","generateStencilBuffer","generateDepthBuffer","_createRenderBuffer","DEPTH_STENCIL","depthFormat","DEPTH_COMPONENT16","DEPTH_COMPONENT32F","STENCIL_INDEX8","STENCIL_ATTACHMENT","msInternalFormat","unbindBuffer","renderBuffer","createRenderbuffer","_updateRenderBuffer","bindRenderbuffer","RENDERBUFFER","renderbufferStorage","framebufferRenderbuffer","_deleteTexture","unbindAllTextures","_releaseRenderTargetWrapper","_setProgram","useProgram","_activateCurrentTexture","activeTexture","TEXTURE0","forTextureDataUpdate","wasPreviouslyBound","isTextureForRendering","bindTexture","_bindSamplerUniformToChannel","_setTexture","sourceSlot","_currentState","REPEAT","MIRRORED_REPEAT","isPartOfTextureArray","video","videoInternalTexture","getInternalTexture","delayLoadState","delayLoad","needToBind","coordinatesMode","textureWrapMode","_setAnisotropicLevel","_textureUnits","Int32Array","anisotropicFilterExtension","_setTextureParameterFloat","TEXTURE_MAX_ANISOTROPY_EXT","parameter","texParameterf","releaseEffects","deleteFramebuffer","releaseComputeEffects","attachContextLostEvent","attachContextRestoredEvent","_canRenderToFramebuffer","successful","fb","createFramebuffer","checkFramebufferStatus","FRAMEBUFFER_COMPLETE","readFormat","readType","readPixels","UNSIGNED_SHORT_4_4_4_4","UNSIGNED_SHORT_5_5_5_1","UNSIGNED_SHORT_5_6_5","BYTE","SHORT","HALF_FLOAT","UNSIGNED_INT_2_10_10_10_REV","UNSIGNED_INT_10F_11F_11F_REV","UNSIGNED_INT_5_9_9_9_REV","FLOAT_32_UNSIGNED_INT_24_8_REV","ALPHA","LUMINANCE","LUMINANCE_ALPHA","RED","RG","RED_INTEGER","RG_INTEGER","RGB_INTEGER","RGBA_INTEGER","R8_SNORM","RG8_SNORM","RGB8_SNORM","R8I","RG8I","RGB8I","RGBA8I","RGBA8_SNORM","R8","RG8","RGB8","RGBA8","R8UI","RG8UI","RGB8UI","RGBA8UI","R16I","RG16I","RGB16I","RGBA16I","R16UI","RG16UI","RGB16UI","RGBA16UI","R32I","RG32I","RGB32I","RGBA32I","R32UI","RG32UI","RGB32UI","RGBA32UI","R32F","RG32F","RGB32F","R16F","RG16F","RGB16F","RGB565","R11F_G11F_B10F","RGB9_E5","RGBA4","RGB5_A1","RGB10_A2","RGB10_A2UI","_getRGBAMultiSampleBufferFormat","onCompleteObservable","hasAlpha","flushRenderer","numChannels","IsSupportedAsync","IsSupported","_HasMajorPerformanceCaveat","_IsSupported","tempcanvas","WebGLRenderingContext","HasMajorPerformanceCaveat","failIfMajorPerformanceCaveat","CeilingPOT","FloorPOT","pot","NearestPOT","requestAnimationFrame","CollisionsEpsilon","TimingTools","setImmediate","Base64DataUrlRegEx","LoadFileError","file","RequestFileError","ReadFileError","FileToolsOptions","DefaultRetryStrategy","RetryStrategy","maxRetries","baseInterval","retryIndex","ExponentialBackoff","BaseUrl","CorsBehavior","PreprocessUrl","_CleanUrl","SetCorsBehavior","crossOrigin","LoadImage","usingObjectURL","URL","createObjectURL","onErrorHandler","inputText","Image","LoadFile","createImageBitmap","assign","premultiplyAlpha","imgBmp","revokeObjectURL","reason","handlersList","unloadHandlersList","handler","src","blockedURI","cspException","effectiveDirective","originalPolicy","noOfflineSupport","_","contentType","blob","loadFromOfflineSupport","loadImage","enableTexturesOffline","textureName","decodeURIComponent","blobURL","ReadFile","reader","FileReader","fileRequest","onloadend","onerror","readAsArrayBuffer","readAsText","fileOrUrl","onOpened","fileName","TestBase64DataUrl","DecodeBase64UrlToBinary","DecodeBase64UrlToString","SetImmediate","RequestFile","loadUrl","aborted","requestFile","onReadyStateChange","retryHandle","unbindEvents","onLoadEnd","DONE","handleError","retryLoop","IsFileURL","retryStrategy","waitTime","enableSceneOffline","loadFile","event","protocol","IsBase64DataUrl","uri","decodedString","bufferLength","bufferView","charCodeAt","DecodeBase64ToBinary","FileTools","_injectLTSFileTools","InstantiationTools","RegisteredExternalClasses","internalClass","RandomGUID","Tools","strategy","classes","fallbackTexture","u","pixels","Instantiate","fround","_TmpFloatArray","returnUnchangedIfNoSlash","previousAngle","newAngle","smoothFactor","previousAngleRad","ToRadians","newAngleRad","ToDegrees","allowsNullUndefined","eventPrefix","PointerEvent","referrerPolicy","FileToolLoadImage","FileToolsLoadFile","scriptUrl","scriptId","importScripts","head","getElementsByTagName","script","appendChild","LoadScript","fileToLoad","progressCallback","readAsDataURL","FileToolsReadFile","content","fileBlob","decimals","toFixed","DeepCopy","windowElement","events","successCallback","quality","toArrayBuffer","convertToBlob","_IsOffScreenCanvas","toBlob","binStr","toDataURL","getFullYear","getMonth","getDate","Download","newWindow","base64data","base64Image","ToBlob","DownloadBlob","style","display","href","download","parentElement","removeChild","click","camera","forceDownload","antialiasing","renderSprites","enableStencilBuffer","useLayerMask","ClearLogCache","PerformanceLogLevel","PerformanceUserMarkLogLevel","StartPerformanceCounter","_StartUserMark","EndPerformanceCounter","_EndUserMark","PerformanceConsoleLogLevel","_StartPerformanceConsole","_EndPerformanceConsole","_StartPerformanceCounterDisabled","_EndPerformanceCounterDisabled","counterName","_Performance","mark","measure","timeEnd","isType","el","moduleName","classObj","UseCustomRequestHeaders","GetAbsoluteUrl","PerformanceNoneLogLevel","AsyncLoop","iterations","_done","_fn","_successCallback","executeNext","breakLoop","syncedIterations","breakFunction","Run","ceil","iteration","inlineScheduler","coroutine","onStep","step","runCoroutine","abortSignal","resume","reschedule","stepResult","runCoroutineSync","SmartArray","_id","_GlobalId","compareFn","contains","SmartArrayNoDuplicate","_duplicateId","__smartArrayFlags","pushNoDuplicate","concatWithNoDuplicate","Viewport","toGlobal","renderWidth","renderHeight","toGlobalToRef","Plane","norm","magnitude","invertedMatrix","_TmpMatrix","normalX","normalY","normalZ","finalD","dotCoordinate","copyFromPoints","point1","point2","point3","x1","y1","z1","pyth","invPyth","isFrontFacingTo","signedDistanceTo","Frustum","frustumPlanes","GetPlanesToRef","frustumPlane","GetNearPlaneToRef","GetFarPlaneToRef","GetLeftPlaneToRef","GetRightPlaneToRef","GetTopPlaneToRef","GetBottomPlaneToRef","Camera","_position","newPosition","upVector","_upVector","screenArea","PERSPECTIVE_CAMERA","fovMode","FOVMODE_VERTICAL_FIXED","minZ","getAspectRatio","halfWidth","halfHeight","orthoRight","orthoLeft","orthoTop","orthoBottom","_orthoLeft","rigCamera","_rigCameras","_orthoRight","_orthoBottom","_orthoTop","_mode","setActiveOnSceneIfNoneActive","maxZ","inertia","isIntermediate","layerMask","cameraRigMode","RIG_MODE_NONE","customRenderTargets","outputRenderTarget","onViewMatrixChangedObservable","onProjectionMatrixChangedObservable","onAfterCheckInputsObservable","onRestoreStateObservable","isRigCamera","_webvrViewMatrix","_skipRendering","_projectionMatrix","_postProcesses","_activeMeshes","_globalPosition","_computedViewMatrix","_doNotComputeProjectionMatrix","_transformMatrix","_refreshFrustumPlanes","_absoluteRotation","_isCamera","_isLeftCamera","_isRightCamera","addCamera","activeCamera","renderPassId","createRenderPassId","storeState","_stateStored","_storedFov","_restoreStateValues","restoreState","fullDetails","applyVerticalCorrection","absoluteRotation","useRightHandedSystem","globalPosition","getActiveMeshes","isActiveMesh","mesh","completeCheck","pp","aspectRatio","ignoreParentClass","_isSynchronizedViewMatrix","_isSynchronizedProjectionMatrix","check","attachControl","_ignored","_noPreventDefault","detachControl","_checkInputs","_updateRigCameras","getViewMatrix","getProjectionMatrix","rigCameras","rigPostProcess","_rigPostProcess","_getFirstPostProcess","ppIndex","_cascadePostProcessesToRigCams","firstPostProcess","markTextureDirty","cam","getEffectName","attachPostProcess","insertAt","isReusable","prePassRenderer","detachPostProcess","_getViewMatrix","_cameraRigParams","vrPreViewMatrix","freezeProjectionMatrix","unfreezeProjectionMatrix","reverseDepth","getTransformationMatrix","_updateFrustumPlanes","_frustumPlanes","GetPlanes","isInFrustum","checkRigCameras","isCompletelyInFrustum","getForwardRay","getForwardRayToRef","refRay","inputs","stopAnimation","removeCamera","cameras","releaseRenderPassId","isLeftCamera","isRightCamera","leftCamera","rightCamera","getLeftTarget","getTarget","getRightTarget","setCameraRigMode","rigParams","interaxialDistance","stereoHalfAngle","createRigCamera","_setRigMode","_getVRProjectionMatrix","vrMetrics","aspectRatioFov","vrWorkMatrix","vrHMatrix","_updateCameraRotationMatrix","_updateWebVRCameraRotationMatrix","_getWebVRProjectionMatrix","_getWebVRViewMatrix","setCameraRigParameter","cameraIndex","RIG_MODE_STEREOSCOPIC_ANAGLYPH","_setupInputs","Serialize","AppendSerializedAnimations","GetConstructorFromName","isStereoscopicSideBySide","getDirection","localAxis","getDirectionToRef","interaxial_distance","Construct","_CreateDefaultParsedCamera","parsedCamera","construct","Parse","parentInstanceIndex","parse","setPosition","setTarget","parsedAnimation","ParseAnimationRanges","autoAnimate","autoAnimateFrom","autoAnimateTo","autoAnimateLoop","autoAnimateSpeed","ORTHOGRAPHIC_CAMERA","FOVMODE_HORIZONTAL_FIXED","RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL","RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED","RIG_MODE_STEREOSCOPIC_OVERUNDER","RIG_MODE_STEREOSCOPIC_INTERLACED","RIG_MODE_VR","RIG_MODE_WEBVR","RIG_MODE_CUSTOM","ForceAttachControlToAlwaysPreventDefault","StringDictionary","_count","_data","getOrAddWithFactory","factory","getOrAdd","curVal","getAndRemove","cur","first","AbstractScene","lights","meshes","skeletons","particleSystems","animationGroups","multiMaterials","morphTargetManagers","geometries","transformNodes","actionManagers","_environmentTexture","postProcesses","parser","_BabylonFileParsers","_IndividualBabylonFileParsers","jsonData","container","parserName","environmentTexture","getNodes","skeleton","bones","MaterialDefines","externalProperties","_keys","_areLightsDirty","_areLightsDisposed","_areAttributesDirty","_areTexturesDirty","_areFresnelDirty","_areMiscDirty","_arePrePassDirty","_areImageProcessingDirty","_normals","_uvs","_needNormals","_needUVs","_externalProperties","_setDefaultValue","markAsProcessed","markAsUnprocessed","markAllAsDirty","markAsImageProcessingDirty","markAsLightDirty","disposed","markAsAttributesDirty","markAsTexturesDirty","markAsFresnelDirty","markAsMiscDirty","markAsPrePassDirty","rebuild","cloneTo","defValue","default","ColorCurves","_dirty","_tempColor","_globalCurve","_highlightsCurve","_midtonesCurve","_shadowsCurve","_positiveCurve","_negativeCurve","_globalHue","_globalDensity","_globalSaturation","_globalExposure","_highlightsHue","_highlightsDensity","_highlightsSaturation","_highlightsExposure","_midtonesHue","_midtonesDensity","_midtonesSaturation","_midtonesExposure","_shadowsHue","_shadowsDensity","_shadowsSaturation","_shadowsExposure","globalHue","globalDensity","globalSaturation","globalExposure","highlightsHue","highlightsDensity","highlightsSaturation","highlightsExposure","midtonesHue","midtonesDensity","midtonesSaturation","midtonesExposure","shadowsHue","shadowsDensity","shadowsSaturation","shadowsExposure","colorCurves","positiveUniform","neutralUniform","negativeUniform","_getColorGradingDataToRef","uniformsList","density","exposure","_Clamp","_ApplyColorGradingSliderNonlinear","_FromHSBToRef","brightness","ImageProcessingConfigurationDefines","IMAGEPROCESSING","VIGNETTE","VIGNETTEBLENDMODEMULTIPLY","VIGNETTEBLENDMODEOPAQUE","TONEMAPPING","TONEMAPPING_ACES","CONTRAST","COLORCURVES","COLORGRADING","COLORGRADING3D","SAMPLER3DGREENDEPTH","SAMPLER3DBGRMAP","DITHER","IMAGEPROCESSINGPOSTPROCESS","EXPOSURE","SKIPFINALCOLORCLAMP","ImageProcessingConfiguration","_colorCurvesEnabled","_colorGradingEnabled","_colorGradingWithGreenDepth","_colorGradingBGR","_exposure","_toneMappingEnabled","_toneMappingType","TONEMAPPING_STANDARD","_contrast","vignetteStretch","vignetteCenterX","vignetteCenterY","vignetteWeight","vignetteColor","vignetteCameraFov","_vignetteBlendMode","VIGNETTEMODE_MULTIPLY","_vignetteEnabled","_ditheringEnabled","_ditheringIntensity","_skipFinalColorClamp","_applyByPostProcess","onUpdateParameters","colorCurvesEnabled","_updateParameters","colorGradingTexture","_colorGradingTexture","colorGradingEnabled","colorGradingWithGreenDepth","colorGradingBGR","toneMappingEnabled","toneMappingType","contrast","vignetteCentreY","vignetteCentreX","vignetteBlendMode","vignetteEnabled","ditheringEnabled","ditheringIntensity","skipFinalColorClamp","applyByPostProcess","PrepareUniforms","samplersList","prepareDefines","forPostProcess","_VIGNETTEMODE_MULTIPLY","overrideAspectRatio","Bind","inverseWidth","inverseHeight","vignetteScaleY","vignetteScaleX","vignetteScaleGeometricMean","Mix","vignettePower","textureSize","parsed","VIGNETTEMODE_OPAQUE","_VIGNETTEMODE_OPAQUE","DeviceType","PointerInput","NativePointerInput","DualShockInput","DualSenseInput","XboxInput","SwitchInput","DeviceInputEventType","createUniformBuffer","elements","ubo","UNIFORM_BUFFER","createDynamicUniformBuffer","updateUniformBuffer","subarray","bindBufferBase","UniformBuffer","dynamic","forceNoUniformBuffer","_noUBO","_dynamic","_uniformLocations","_uniformSizes","_uniformArraySizes","_uniformLocationPointer","_needSync","_buffers","_bufferIndex","_createBufferOnWrite","_currentFrameId","updateMatrix3x3","_updateMatrix3x3ForEffect","updateMatrix2x2","_updateMatrix2x2ForEffect","updateFloat","_updateFloatForEffect","updateFloat2","_updateFloat2ForEffect","updateFloat3","_updateFloat3ForEffect","updateFloat4","_updateFloat4ForEffect","updateFloatArray","_updateFloatArrayForEffect","updateArray","_updateArrayForEffect","updateIntArray","_updateIntArrayForEffect","updateUIntArray","_updateUIntArrayForEffect","updateMatrix","_updateMatrixForEffect","updateMatrices","_updateMatricesForEffect","updateVector3","_updateVector3ForEffect","updateVector4","_updateVector4ForEffect","updateColor3","_updateColor3ForEffect","updateColor4","_updateColor4ForEffect","updateDirectColor4","_updateDirectColor4ForEffect","updateInt","_updateIntForEffect","updateInt2","_updateInt2ForEffect","updateInt3","_updateInt3ForEffect","updateInt4","_updateInt4ForEffect","updateUInt","_updateUIntForEffect","updateUInt2","_updateUInt2ForEffect","updateUInt3","_updateUInt3ForEffect","updateUInt4","_updateUInt4ForEffect","_updateMatrix3x3ForUniform","_updateMatrix2x2ForUniform","_updateFloatForUniform","_updateFloat2ForUniform","_updateFloat3ForUniform","_updateFloat4ForUniform","_updateFloatArrayForUniform","_updateArrayForUniform","_updateIntArrayForUniform","_updateUIntArrayForUniform","_updateMatrixForUniform","_updateMatricesForUniform","_updateVector3ForUniform","_updateVector4ForUniform","_updateColor3ForUniform","_updateColor4ForUniform","_updateDirectColor4ForUniform","_updateIntForUniform","_updateInt2ForUniform","_updateInt3ForUniform","_updateInt4ForUniform","_updateUIntForUniform","_updateUInt2ForUniform","_updateUInt3ForUniform","_updateUInt4ForUniform","useUbo","isSync","isDynamic","getData","_bufferData","_fillAlignment","alignment","oldPointer","addUniform","arraySize","strideSize","addMatrix","mat","addFloat2","addFloat3","addColor3","addColor4","addMatrix3x3","addMatrix2x2","_numBuffers","_indexBuffer","currentEffect","_buffersEqual","buf1","buf2","_copyBuffer","dst","_UpdatedUbosInFrame","_createNewBuffer","_checkNewFrame","updateUniform","FloatRound","updateUniformArray","arraySizes","countToFour","baseStride","_TempBuffer","suffix","_TempBufferInt32View","_TempBufferUInt32View","updateUniformDirectly","bindToEffect","_currentEffectName","unbindEffect","setDataBuffer","uniformBuffers","_MAX_UNIFORM_SIZE","Buffer","postponeInternalCreation","instanced","useBytes","_isAlreadyOwned","_updatable","_instanced","_divisor","VertexBuffer","isUpdatable","getStrideSize","updateDynamicVertexBuffer","updateDirectly","vertexCount","_increaseReferences","instanceDivisor","_instanceDivisor","isInstanced","_computeHashCode","takeBufferOwnership","_ownsBuffer","_kind","vertexData","Int8Array","Int16Array","typeByteLength","GetTypeByteLength","DeduceStride","hashCode","getKind","getFloatData","totalVertices","forceCopy","tightlyPackedByteStride","remainder","getOffset","sizeInBytes","ForEach","UVKind","UV2Kind","UV3Kind","UV4Kind","UV5Kind","UV6Kind","NormalKind","PositionKind","ColorKind","MatricesIndicesKind","MatricesIndicesExtraKind","MatricesWeightsKind","MatricesWeightsExtraKind","TangentKind","componentCount","componentType","componentIndex","dataView","DataView","componentByteLength","componentByteOffset","_GetFloatValue","getInt8","getUint8","getInt16","getUint16","getInt32","getUint32","getFloat32","ColorInstanceKind","PickingInfo","hit","pickedPoint","pickedMesh","bu","bv","faceId","subMeshFaceId","subMeshId","pickedSprite","thinInstanceIndex","ray","originMesh","aimTransform","gripTransform","getNormal","useWorldCoordinates","useVerticesNormals","isVerticesDataPresent","getIndices","tmp0","tmp1","normals","getVerticesData","normal0","normal1","normal2","positions","vertex1","vertex2","vertex3","p1p2","p3p2","transformNormalToWorld","wm","nonUniformScaling","normalForDirectionChecking","getTextureCoordinates","uvSet","uvs","uv0","uv1","uv2","ActionEvent","pointerX","pointerY","meshUnderPointer","sourceEvent","additionalData","prim","pointerPos","PostProcessManager","_vertexBuffers","_prepareBuffers","vertices","_buildIndexBuffer","vb","_prepareFrame","sourceTexture","postProcessesEnabled","directRender","targetTexture","doNotBindFrambuffer","_debugInsertMarker","onBeforeRenderObservable","onAfterRenderObservable","setDepthWrite","_finalizeFrame","doNotPresent","_outputTexture","setAlphaMode","RenderingGroup","opaqueSortCompareFn","_opaqueSortCompareFn","PainterSortCompare","_renderOpaque","_renderOpaqueSorted","alphaTestSortCompareFn","_alphaTestSortCompareFn","_renderAlphaTest","_renderAlphaTestSorted","transparentSortCompareFn","_transparentSortCompareFn","defaultTransparentSortCompare","_renderTransparent","_renderTransparentSorted","_opaqueSubMeshes","_transparentSubMeshes","_alphaTestSubMeshes","_depthOnlySubMeshes","_particleSystems","_spriteManagers","_empty","_edgesRenderers","render","customRenderFunction","renderParticles","activeMeshes","getStencilBuffer","setStencilBuffer","_renderSprites","_renderParticles","onBeforeTransparentRendering","useOrderIndependentTransparency","excludedMeshes","depthPeelingRenderer","edgesRendererIndex","_RenderSorted","sortCompareFn","transparent","subMesh","subIndex","cameraPosition","_ZeroVector","_alphaIndex","getMesh","alphaIndex","_distanceToCamera","boundingSphere","centerWorld","sortedArray","_activeMeshesFrozenButKeepClipping","material","getMaterial","needDepthPrePass","backToFrontSortCompare","meshA","meshB","prepare","prepareSprites","needAlphaBlendingForMesh","needAlphaTesting","_renderingGroup","_edgesRenderer","dispatchSprites","spriteManager","dispatchParticles","particleSystem","onBeforeParticlesRenderingObservable","particleIndex","emitter","_activeParticles","addCount","onAfterParticlesRenderingObservable","spritesEnabled","onBeforeSpritesRenderingObservable","onAfterSpritesRenderingObservable","RenderingGroupInfo","RenderingManager","maintainStateBetweenFrames","_maintainStateBetweenFrames","_wasDispatched","spriteManagers","_useSceneAutoClearSetup","_renderingGroups","_autoClearDepthStencil","_customOpaqueSortCompareFn","_customAlphaTestSortCompareFn","_customTransparentSortCompareFn","_renderingGroupInfo","MIN_RENDERINGGROUPS","MAX_RENDERINGGROUPS","autoClear","getRenderingGroup","renderingGroupId","_prepareRenderingGroup","_clearDepthStencilBuffer","_depthStencilBufferAlreadyCleaned","info","renderingGroup","renderingGroupMask","onBeforeRenderingGroupObservable","AUTOCLEAR","getAutoClearDepthStencilSetup","_beforeRenderingGroupDrawStage","_afterRenderingGroupDrawStage","onAfterRenderingGroupObservable","resetSprites","freeRenderingGroups","setRenderingOrder","group","setRenderingAutoClearDepthStencil","autoClearDepthStencil","SceneComponentConstants","NAME_EFFECTLAYER","NAME_LAYER","NAME_LENSFLARESYSTEM","NAME_BOUNDINGBOXRENDERER","NAME_PARTICLESYSTEM","NAME_GAMEPAD","NAME_SIMPLIFICATIONQUEUE","NAME_GEOMETRYBUFFERRENDERER","NAME_PREPASSRENDERER","NAME_DEPTHRENDERER","NAME_DEPTHPEELINGRENDERER","NAME_POSTPROCESSRENDERPIPELINEMANAGER","NAME_SPRITE","NAME_SUBSURFACE","NAME_OUTLINERENDERER","NAME_PROCEDURALTEXTURE","NAME_SHADOWGENERATOR","NAME_OCTREE","NAME_PHYSICSENGINE","NAME_AUDIO","NAME_FLUIDRENDERER","STEP_ISREADYFORMESH_EFFECTLAYER","STEP_BEFOREEVALUATEACTIVEMESH_BOUNDINGBOXRENDERER","STEP_EVALUATESUBMESH_BOUNDINGBOXRENDERER","STEP_PREACTIVEMESH_BOUNDINGBOXRENDERER","STEP_CAMERADRAWRENDERTARGET_EFFECTLAYER","STEP_BEFORECAMERADRAW_PREPASS","STEP_BEFORECAMERADRAW_EFFECTLAYER","STEP_BEFORECAMERADRAW_LAYER","STEP_BEFORERENDERTARGETDRAW_PREPASS","STEP_BEFORERENDERTARGETDRAW_LAYER","STEP_BEFORERENDERINGMESH_PREPASS","STEP_BEFORERENDERINGMESH_OUTLINE","STEP_AFTERRENDERINGMESH_PREPASS","STEP_AFTERRENDERINGMESH_OUTLINE","STEP_AFTERRENDERINGGROUPDRAW_EFFECTLAYER_DRAW","STEP_AFTERRENDERINGGROUPDRAW_BOUNDINGBOXRENDERER","STEP_BEFORECAMERAUPDATE_SIMPLIFICATIONQUEUE","STEP_BEFORECAMERAUPDATE_GAMEPAD","STEP_BEFORECLEAR_PROCEDURALTEXTURE","STEP_BEFORECLEAR_PREPASS","STEP_BEFORERENDERTARGETCLEAR_PREPASS","STEP_AFTERRENDERTARGETDRAW_PREPASS","STEP_AFTERRENDERTARGETDRAW_LAYER","STEP_AFTERCAMERADRAW_PREPASS","STEP_AFTERCAMERADRAW_EFFECTLAYER","STEP_AFTERCAMERADRAW_LENSFLARESYSTEM","STEP_AFTERCAMERADRAW_EFFECTLAYER_DRAW","STEP_AFTERCAMERADRAW_LAYER","STEP_AFTERCAMERADRAW_FLUIDRENDERER","STEP_AFTERCAMERAPOSTPROCESS_LAYER","STEP_AFTERRENDERTARGETPOSTPROCESS_LAYER","STEP_AFTERRENDER_AUDIO","STEP_GATHERRENDERTARGETS_DEPTHRENDERER","STEP_GATHERRENDERTARGETS_GEOMETRYBUFFERRENDERER","STEP_GATHERRENDERTARGETS_SHADOWGENERATOR","STEP_GATHERRENDERTARGETS_POSTPROCESSRENDERPIPELINEMANAGER","STEP_GATHERACTIVECAMERARENDERTARGETS_DEPTHRENDERER","STEP_GATHERACTIVECAMERARENDERTARGETS_FLUIDRENDERER","STEP_POINTERMOVE_SPRITE","STEP_POINTERDOWN_SPRITE","STEP_POINTERUP_SPRITE","Stage","registerStep","component","PointerEventTypes","POINTERDOWN","POINTERUP","POINTERMOVE","POINTERWHEEL","POINTERPICK","POINTERTAP","POINTERDOUBLETAP","PointerInfoBase","PointerInfoPre","localX","localY","originalPickingInfo","skipOnPointerObservable","localPosition","PointerInfo","pickInfo","_pickInfo","_generatePickInfo","inputManager","_inputManager","_pickMove","_setRayOnPointerInfo","AbstractActionManager","hoverCursor","isRecursive","HasTriggers","Triggers","HasPickTriggers","tAsInt","KeyboardEventTypes","KEYDOWN","KEYUP","KeyboardInfo","KeyboardInfoPre","skipOnKeyboardObservable","EventConstants","DOM_DELTA_PIXEL","DOM_DELTA_LINE","DOM_DELTA_PAGE","DeviceEventFactory","deviceType","deviceSlot","inputIndex","deviceInputSystem","elementToAttachTo","pointerId","Keyboard","_CreateKeyboardEvent","Mouse","MouseWheelX","MouseWheelY","MouseWheelZ","_CreateWheelEvent","Touch","_CreatePointerEvent","_CreateMouseEvent","pointerType","buttons","pollInput","LeftClick","RightClick","MiddleClick","Move","button","deltaMode","deltaX","deltaY","deltaZ","_CreateEvent","Horizontal","Vertical","movementX","movementY","DeltaHorizontal","DeltaVertical","_CheckNonCharacterKeys","clientX","clientY","fromCharCode","keyCode","isKeyboardActive","isDeviceAvailable","altKey","ctrlKey","metaKey","shiftKey","NativeDeviceInputSystem","onDeviceConnected","onDeviceDisconnected","onInputChanged","_nativeInput","DeviceInputSystem","CreateDeviceEvent","_createDummyNativeInput","MAX_POINTER_INPUTS","WebDeviceInputSystem","_inputs","_keyboardActive","_pointerActive","_usingSafari","IsSafari","_usingMacOS","platform","_keyboardDownEvent","_keyboardUpEvent","_keyboardBlurEvent","_pointerMoveEvent","_pointerDownEvent","_pointerUpEvent","_pointerCancelEvent","_pointerWheelEvent","_pointerBlurEvent","_eventsAttached","_mouseId","_isUsingFirefox","_maxTouchPoints","_pointerInputClearObserver","_gamepadConnectedEvent","_gamepadDisconnectedEvent","_eventPrefix","GetPointerPrefix","_onDeviceConnected","_onDeviceDisconnected","_onInputChanged","_enableEvents","_metaKeys","_onEngineViewChanged","device","DualShock","DualSense","_updateDevice","_elementToAttachTo","_disableEvents","inputElement","getInputElement","deviceSlotKey","tabIndex","canvasTabIndex","_handleKeyActions","_handlePointerActions","_handleGamepadActions","_checkForConnectedDevices","_wheelEventName","onEndFrameObservable","getGamepads","gamepads","gamepad","_addGamePad","matchMedia","_addPointerDevice","_getGamepadDeviceType","_gamepads","_registerDevice","axes","currentX","currentY","numberOfInputs","fill","_unregisterDevice","kbKey","deviceEvent","maxTouchPoints","_activeTouchIds","_getPointerType","previousHorizontal","previousVertical","pointerLockElement","setPointerCapture","hasPointerCapture","releasePointerCapture","BrowserForward","onmousewheel","passiveSupported","noop","wheelDelta","passive","gp","valueOf","deviceName","Xbox","Switch","Generic","touches","DeviceSource","onInputChangedObservable","_deviceInputSystem","getInput","InternalDeviceSourceManager","_registeredManagers","registerManager","_devices","_addDevice","unregisterManager","numberOfDeviceTypes","deviceSource","_removeDevice","DeviceSourceManager","getDeviceSource","_firstDevice","getDeviceSources","_deviceSourceManager","onDeviceConnectedObservable","devices","onDeviceDisconnectedObservable","_updateFirstDevices","_ClickInfo","_singleClick","_doubleClick","_hasSwiped","_ignore","singleClick","doubleClick","hasSwiped","InputManager","_alreadyAttached","_meshPickProceed","_currentPickResult","_previousPickResult","_totalPointersPressed","_doubleClickOccured","_isSwiping","_swipeButtonPressed","_skipPointerTap","_isMultiTouchGesture","_pointerX","_pointerY","_startingPointerPosition","_previousStartingPointerPosition","_startingPointerTime","_previousStartingPointerTime","_pointerCaptures","_meshUnderPointerId","_movePointerInfo","_cameraObserverCount","_delayedClicks","_pointerOverMesh","getMeshUnderPointerByPointerId","unTranslatedPointer","_unTranslatedPointerX","_unTranslatedPointerY","_updatePointerPosition","canvasRect","getInputElementClientRect","_processPointerMove","pickResult","doNotHandleCursors","defaultCursor","_setCursorAndPointerOverMesh","_pointerMoveStage","isMeshPicked","pointerInfo","onPointerMove","onPointerObservable","_pickingAvailable","createPickingRay","_addCameraPointerObserver","_removeCameraPointerObserver","_checkForPicking","onPointerPick","_checkPrePointerObservable","pi","nearInteractionPickingInfo","onPrePointerObservable","pick","pointerMovePredicate","pointerMoveFastCheck","cameraToUseForPointers","pointerMoveTrianglePredicate","setPointerOverMesh","actionManager","hasPointerTriggers","simulatePointerMove","pointerEventInit","simulatePointerDown","_processPointerDown","_pickedDownMesh","hasPickTriggers","processTrigger","CreateNew","hasSpecificTrigger","isVisible","LongPressDelay","_isPointerSwiping","_pointerDownStage","onPointerDown","simulatePointerUp","doubleTap","clickInfo","_processPointerUp","_pickedUpMesh","doubleClickActionManager","_pointerUpStage","pickedDownActionManager","onPointerUp","isPointerCaptured","attachUp","attachDown","attachMove","_alreadyAttachedTo","_initActionManager","act","skipPointerUpPicking","_registeredActions","pointerUpPredicate","pointerUpFastCheck","_delayedSimpleClick","btn","cb","DoubleClickDelay","_previousButtonPressed","_initClickEvent","obs1","obs2","checkPicking","needToIgnoreNext","checkSingleClickImmediately","ExclusiveDoubleClickMode","HasSpecificTrigger","delayedClick","timeoutId","checkDoubleClick","_onPointerMove","DragMovementThreshold","isPointerLock","_verifyPointerLock","skipPointerMovePicking","enablePointerMoveEvents","constantlyUpdateMeshUnderPointer","_onPointerDown","prevEvt","preventDefaultOnPointerDown","focus","pointerDownPredicate","skipPointerDownPicking","pointerDownFastCheck","_onPointerUp","preventDefaultOnPointerUp","_onKeyDown","onPreKeyboardObservable","onKeyboardObservable","CreateNewFromScene","_onKeyUp","BrowserBack","_internalAbstractMeshDataInfo","_pointerOverDisableMeshTesting","underPointerMesh","getPointerOverMesh","_invalidateMesh","PerfCounter","_min","_max","average","_average","lastSecAverage","_lastSecAverage","_current","_totalAccumulated","_totalValueCount","_startMonitoringTime","_lastSecAccumulated","_lastSecTime","_lastSecValueCount","fetchNewFrame","newCount","fetchResult","Enabled","_fetchResult","beginMonitoring","endMonitoring","newFrame","currentTime","UniqueIdGenerator","UniqueId","_UniqueIdCounter","LightConstants","shadowEnabled","renderPriority","FALLOFF_DEFAULT","FALLOFF_PHYSICAL","FALLOFF_GLTF","FALLOFF_STANDARD","LIGHTMAP_DEFAULT","LIGHTMAP_SPECULAR","LIGHTMAP_SHADOWSONLY","INTENSITYMODE_AUTOMATIC","INTENSITYMODE_LUMINOUSPOWER","INTENSITYMODE_LUMINOUSINTENSITY","INTENSITYMODE_ILLUMINANCE","INTENSITYMODE_LUMINANCE","LIGHTTYPEID_POINTLIGHT","LIGHTTYPEID_DIRECTIONALLIGHT","LIGHTTYPEID_SPOTLIGHT","LIGHTTYPEID_HEMISPHERICLIGHT","PointerPickingConfiguration","ScenePerformancePriority","Scene","imageProcessingConfiguration","_imageProcessingConfiguration","performancePriority","_performancePriority","BackwardCompatible","skipFrustumClipping","_renderingManager","Intermediate","Aggressive","onScenePerformancePriorityChangedObservable","forceWireframe","_forceWireframe","_skipFrustumClipping","forcePointsCloud","_forcePointsCloud","beforeRender","_onBeforeRenderObserver","afterRender","_onAfterRenderObserver","beforeCameraRender","_onBeforeCameraRenderObserver","onBeforeCameraRenderObservable","afterCameraRender","_onAfterCameraRenderObserver","onAfterCameraRenderObservable","_pointerPickingConfiguration","bindEyePosition","variableName","isVector3","eyePosition","_forcedViewPosition","_mirroredCameraPosition","devicePosition","invertNormal","finalizeSceneUbo","getSceneUniformBuffer","_useRightHandedSystem","setStepId","newStepId","_currentStepId","getStepId","getInternalStep","_currentInternalStep","fogEnabled","_fogEnabled","fogMode","_fogMode","prePass","defaultRT","shadowsEnabled","_shadowsEnabled","lightsEnabled","_lightsEnabled","activeCameras","_activeCameras","_unObserveActiveCameras","onActiveCamerasChanged","_activeCamera","onActiveCameraChanged","defaultMaterial","_defaultMaterial","DefaultMaterialFactory","texturesEnabled","_texturesEnabled","skeletonsEnabled","_skeletonsEnabled","collisionCoordinator","_collisionCoordinator","CollisionCoordinatorFactory","renderingManager","_registerTransientComponents","_transientComponents","_addComponent","_components","serializableComponent","addFromContainer","_serializableComponents","_getComponent","_isScene","_blockEntityCollection","autoClearDepthAndStencil","animationsEnabled","useConstantAnimationDeltaTime","disableOfflineSupportExceptionRules","onAfterRenderCameraObservable","onBeforeAnimationsObservable","onAfterAnimationsObservable","onBeforeDrawPhaseObservable","onAfterDrawPhaseObservable","onReadyObservable","onBeforeActiveMeshesEvaluationObservable","onAfterActiveMeshesEvaluationObservable","onNewCameraAddedObservable","onCameraRemovedObservable","onNewLightAddedObservable","onLightRemovedObservable","onNewGeometryAddedObservable","onGeometryRemovedObservable","onNewTransformNodeAddedObservable","onTransformNodeRemovedObservable","onNewMeshAddedObservable","onMeshRemovedObservable","onNewSkeletonAddedObservable","onSkeletonRemovedObservable","onNewMaterialAddedObservable","onNewMultiMaterialAddedObservable","onMaterialRemovedObservable","onMultiMaterialRemovedObservable","onNewTextureAddedObservable","onTextureRemovedObservable","onBeforeRenderTargetsRenderObservable","onAfterRenderTargetsRenderObservable","onBeforeStepObservable","onAfterStepObservable","onMeshImportedObservable","onAnimationFileImportedObservable","_registeredForLateAnimationBindings","_timeAccumulator","FOGMODE_NONE","fogColor","fogDensity","fogStart","fogEnd","needsPreviousWorldMatrices","physicsEnabled","particlesEnabled","lensFlaresEnabled","collisionsEnabled","gravity","renderTargetsEnabled","dumpNextRenderTargets","importedMeshesFiles","probesEnabled","_meshesForIntersections","proceduralTexturesEnabled","_totalVertices","_activeIndices","_activeBones","_animationTime","animationTimeScale","_renderId","_executeWhenReadyTimeoutId","_intermediateRendering","_defaultFrameBufferCleared","_viewUpdateFlag","_projectionUpdateFlag","_toBeDisposed","_pendingData","dispatchAllSubMeshesOfActiveMeshes","_processedMaterials","_renderTargets","_materialsRenderTargets","_activeParticleSystems","_activeSkeletons","_softwareSkinnedMeshes","_activeAnimatables","requireLightSorting","_beforeCameraUpdateStage","Create","_beforeClearStage","_beforeRenderTargetClearStage","_gatherRenderTargetsStage","_gatherActiveCameraRenderTargetsStage","_isReadyForMeshStage","_beforeEvaluateActiveMeshStage","_evaluateSubMeshStage","_preActiveMeshStage","_cameraDrawRenderTargetStage","_beforeCameraDrawStage","_beforeRenderTargetDrawStage","_beforeRenderingMeshStage","_afterRenderingMeshStage","_afterCameraDrawStage","_afterCameraPostProcessStage","_afterRenderTargetDrawStage","_afterRenderTargetPostProcessStage","_afterRenderStage","_geometriesByUniqueId","_defaultMeshCandidates","_defaultSubMeshCandidates","_preventFreeActiveMeshesAndRenderingGroups","_activeMeshesFrozen","_skipEvaluateActiveMeshesCompletely","_allowPostProcessClearColor","getDeterministicFrameTime","getTimeStep","_blockMaterialDirtyMechanism","_perfCollector","fullOptions","useGeometryUniqueIdsMap","useMaterialMeshMap","useClonedMeshMap","virtual","_virtualScenes","_uid","postProcessManager","_createUbo","setDefaultCandidateProviders","onNewSceneAddedObservable","_getDefaultMeshCandidates","_getDefaultSubMeshCandidates","getActiveMeshCandidates","getActiveSubMeshCandidates","getIntersectingSubMeshCandidates","getCollidingSubMeshCandidates","getCachedMaterial","_cachedMaterial","getCachedEffect","_cachedEffect","getCachedVisibility","_cachedVisibility","isCachedMaterialInvalid","visibility","totalVerticesPerfCounter","getActiveIndices","totalActiveIndicesPerfCounter","getActiveParticles","activeParticlesPerfCounter","getActiveBones","activeBonesPerfCounter","getAnimationRatio","_animationRatio","getFrameId","setSceneUniformBuffer","createSceneUniformBuffer","checkRenderTargets","currentRenderPassId","hardwareInstancedRendering","hasThinInstances","instances","_storeEffectOnSubMeshes","hasRenderTargetTextures","getRenderTargetTextures","isReadyForRendering","resetCachedMaterial","registerBeforeRender","unregisterBeforeRender","registerAfterRender","unregisterAfterRender","_executeOnceBeforeRender","execFunc","executeOnceBeforeRender","wasLoading","getWaitingItemsCount","executeWhenReady","whenReadyAsync","animatables","resetLastAnimationTimeFrame","_animationTimeLast","_viewMatrix","getTransformMatrix","setTransformMatrix","viewL","projectionL","viewR","projectionR","_multiviewSceneUbo","_updateMultiviewUbo","_sceneUbo","sceneUbo","addMesh","newMesh","recursive","_resyncLightSources","removeMesh","toRemove","addTransformNode","newTransformNode","_indexInSceneTransformNodesArray","removeTransformNode","lastNode","removeSkeleton","_executeActiveContainerCleanup","removeMorphTargetManager","removeLight","_removeLightSource","sortLightsByPriority","index2","removeParticleSystem","removeAnimation","animationName","targetMask","removeAnimationGroup","removeMultiMaterial","removeMaterial","_indexInSceneMaterialArray","lastMaterial","removeActionManager","removeTexture","addLight","newLight","lightSources","CompareLightsPriority","newCamera","addSkeleton","newSkeleton","addParticleSystem","newParticleSystem","addAnimation","newAnimation","addAnimationGroup","newAnimationGroup","addMultiMaterial","newMultiMaterial","addMaterial","newMaterial","addMorphTargetManager","newMorphTargetManager","addGeometry","newGeometry","addActionManager","newActionManager","addTexture","newTexture","switchActiveCamera","setActiveCameraById","setActiveCameraByName","getCameraByName","getAnimationGroupByName","_getMaterial","allowMultiMaterials","getMaterialByUniqueID","getMaterialByName","getLastMaterialById","getTextureByUniqueId","getCameraByUniqueId","getBoneById","skeletonIndex","boneIndex","getBoneByName","getLightByName","getLightById","getLightByUniqueId","getParticleSystemById","getGeometryById","_getGeometryByUniqueId","pushGeometry","geometry","lastGeometry","getGeometries","getMeshById","getMeshesById","getTransformNodeById","getTransformNodeByUniqueId","getTransformNodesById","getMeshByUniqueId","getLastTransformNodeById","getLastEntryById","getNodeById","transformNode","light","bone","getNodeByName","getMeshByName","getTransformNodeByName","getLastSkeletonById","getSkeletonByUniqueId","getSkeletonById","getSkeletonByName","getMorphTargetManagerById","getMorphTargetById","managerIndex","morphTargetManager","numTargets","getMorphTargetByName","getPostProcessByName","postProcessIndex","uid","RandomId","addExternalData","_externalData","getExternalData","getOrAddExternalDataWithFactory","removeExternalData","_evaluateSubMesh","initialMesh","forcePush","freeProcessedMaterials","blockfreeActiveMeshesAndRenderingGroups","freeActiveMeshes","renderList","_isInIntermediateRendering","freezeActiveMeshes","skipEvaluateActiveMeshes","freezeMeshes","keepFrustumCulling","updateTransformMatrix","_evaluateActiveMeshes","_freeze","unfreezeActiveMeshes","_isActive","_unFreeze","psLength","animate","_currentLODIsUpToDate","isBlocked","hasSpecificTriggers2","meshToRender","customLODSelector","getLOD","_currentLOD","billboardMode","_preActivate","alwaysSelectAsActiveMesh","_activate","isAnInstance","_actAsRegularMesh","_onlyForInstances","_activeMesh","_postActivate","isStarted","sourceMesh","computeBonesUsingShaders","hasInstances","_renderingMultiview","_bindFrameBuffer","_multiviewTexture","_clearFrameBuffer","rtt","onClearObservable","skipInitialClear","_cleared","_clear","_renderForCamera","rigParent","bindFrameBuffer","softwareSkinnedMeshIndex","applySkeleton","needRebind","renderIndex","renderTarget","_shouldRender","hasSpecialRenderTargetCamera","_processSubCameras","_createMultiviewUbo","_useMultiviewToSingleView","_renderMultiviewToSingleView","_checkIntersections","actionIndex","parameters","getTriggerParameter","otherMesh","areIntersecting","intersectsMesh","usePreciseIntersection","currentIntersectionInProgress","_intersectionsInProgress","_executeCurrent","parameterMesh","_advancePhysicsEngineStep","_animate","isDeterministicLockStep","MinDeltaTime","getDeltaTime","MaxDeltaTime","defaultFrameTime","defaultFPS","stepsTaken","maxSubSteps","getLockstepMaxSteps","internalSteps","_checkCameraRenderTarget","resetDrawCache","passId","updateCameras","ignoreAnimations","currentActiveCamera","customIndex","freezeMaterials","freeze","unfreezeMaterials","unfreeze","stopAllAnimations","animatable","onAnimationEndObservable","activeRequests","_disposeList","itemsCopy","clearCachedVertexData","meshIndex","clearCachedData","cleanCachedTextureBuffer","baseTexture","getWorldExtends","filterPredicate","infiniteDistance","cameraViewSpace","createPickingRayToRef","enableDistantPicking","createPickingRayInCameraSpace","createPickingRayInCameraSpaceToRef","fastCheck","trianglePredicate","pickWithBoundingInfo","pickWithRay","multiPick","multiPickWithRay","_rebuildGeometries","system","spriteMgr","_rebuildTextures","_getByTags","list","listByTags","getMeshesByTags","getCamerasByTags","getLightsByTags","getMaterialByTags","getTransformNodesByTags","blockMaterialDirtyMechanism","useOfflineSupport","_loadFileAsync","_requestFile","_requestFileAsync","_readFile","_readFileAsync","getPerfCollector","setActiveCameraByID","getMaterialByID","getLastMaterialByID","getTextureByUniqueID","getCameraByID","getCameraByUniqueID","getBoneByID","getLightByID","getLightByUniqueID","getParticleSystemByID","getGeometryByID","getMeshByID","getMeshByUniqueID","getLastMeshByID","getMeshesByID","getTransformNodeByID","getTransformNodeByUniqueID","getTransformNodesByID","getNodeByID","getLastEntryByID","getLastSkeletonByID","FOGMODE_EXP","FOGMODE_EXP2","FOGMODE_LINEAR","IntersectionInfo","BoundingBox","worldMatrix","vectors","extendSize","extendSizeWorld","directions","vectorsWorld","minimum","maximum","_drawWrapperFront","_drawWrapperBack","reConstruct","minX","minY","maxX","maxY","_update","factor","tmpVectors","_TmpVector3","newRadius","minWorld","maxWorld","IsInFrustum","IsCompletelyInFrustum","intersectsPoint","pointX","pointY","pointZ","delta","intersectsSphere","sphere","IntersectsSphere","radiusWorld","intersectsMinMax","myMin","myMax","myMinX","myMinY","myMinZ","myMaxX","myMaxY","myMaxZ","box0","box1","minPoint","maxPoint","sphereCenter","sphereRadius","boundingVectors","canReturnFalse","BoundingSphere","tempRadiusVector","tempVector","isCenterInFrustum","squareDistance","sphere0","sphere1","radiusSum","_result0","_result1","computeBoxExtents","box","axisOverlap","BoundingInfo","_isLocked","isLocked","centerOn","extend","encapsulate","Minimize","Maximize","encapsulateBoundingInfo","toEncapsulate","invw","diagonalLength","_checkCollision","collider","_canDoCollision","intersects","precise","Intersects","MathHelpers","extractMinAndMax","bias","SubMesh","materialDefines","_mainDrawWrapperOverride","_getDrawWrapper","createIfNotExisting","drawWrapper","_drawWrappers","_mesh","_removeDrawWrapper","disposeWrapper","_drawWrapper","_drawWrapperOverride","_setMainDrawWrapperOverride","wrapper","materialIndex","renderingMesh","createBoundingBox","addToMesh","_linesIndexCount","_linesIndexBuffer","_lastColliderWorldVertices","_lastColliderTransformMatrix","_currentMaterial","_renderingMesh","_trianglePlanes","refreshBoundingInfo","IsGlobal","getTotalIndices","_boundingInfo","setBoundingInfo","getRenderingMesh","getReplacementMesh","getEffectiveMesh","replacementMesh","getDefaultMaterial","rootMaterial","getMaterialForRenderPass","_isMultiMaterial","effectiveMaterial","getSubMaterial","extractMinAndMaxIndexed","boundingBias","updateBoundingInfo","enableAlphaMode","_getLinesIndexBuffer","linesIndices","canIntersects","intersectsBox","checkStopper","_intersectLines","intersectionThreshold","_intersectUnIndexedLines","_unIndexed","_intersectUnIndexedTriangles","_intersectTriangles","intersectInfo","intersectionSegment","indexA","indexB","indexC","currentIntersectInfo","intersectsTriangle","newRenderingMesh","startIndex","minVertexIndex","maxVertexIndex","vertexIndex","VertexDataMaterialInfo","VertexData","coroutineFactory","_applyTo","_applyToCoroutine","tangents","uvs2","uvs3","uvs4","uvs5","uvs6","matricesIndices","matricesWeights","matricesIndicesExtra","matricesWeightsExtra","applyToMesh","applyToGeometry","updateMesh","updateGeometry","meshOrGeometry","setVerticesData","setIndices","materialInfos","matInfo","updateExtends","makeItUnique","updateVerticesData","coordinates","coordinate","transformedCoordinate","transformedNormal","_TransformVector3Coordinates","_TransformVector3Normals","_TransformVector4Normals","_FlipFaces","splitBasedOnMaterialID","materialInfo","newMaterialInfo","others","use32BitsIndices","forceCloneIndices","mergeMaterialIds","enableCompletion","vertexDatas","_mergeCoroutine","_validate","root","indexOffset","vertexOffset","currentMaterialInfo","vertexDataList","matInfoA","matInfoB","vertexDataSource","totalIndices","indexSum","indicesOffset","positionsOffset","_MergeElement","nonNullOthers","sumLen","transformRange","ret32","getElementCount","positionsElementCount","validateElementCount","elementCount","_isExpanded","materialInfoSerializationObject","copyWhenShared","_ExtractFrom","polygon","fUV","fColors","frontUVs","backUVs","wrap","orientation","capSubdivisions","p1p2x","p1p2y","p1p2z","p3p2x","p3p2y","p3p2z","faceNormalx","faceNormaly","faceNormalz","v1x","v1y","v1z","v2x","v2y","v2z","v3x","v3y","v3z","computeFacetNormals","computeFacetPositions","computeFacetPartitioning","computeDepthSort","faceNormalSign","ratio","distanceTo","facetNormals","facetPositions","facetPartitioning","depthSort","xSubRatio","ySubRatio","zSubRatio","subSq","bbSize","subDiv","X","Y","Z","nbFaces","ox","bInfo","oy","oz","b1x","b1y","b1z","b2x","b2y","b2z","b3x","b3y","b3z","block_idx_v1","block_idx_v2","block_idx_v3","block_idx_o","dsf","depthSortedFacets","ind","sqDistance","li","ln","DEFAULTSIDE","FRONTSIDE","BACKSIDE","DOUBLESIDE","lp","lu","parsedVertexData","CheckColors4","materialInfoFromJSON","setAllVerticesData","SceneLoaderFlags","ForceFullSceneLoadingForIncremental","_ForceFullSceneLoadingForIncremental","ShowLoadingScreen","_ShowLoadingScreen","loggingLevel","_LoggingLevel","CleanBoneMatrixWeights","_CleanBoneMatrixWeights","CompatibilityOptions","UseOpenGLOrientationForUV","Geometry","_boundingBias","_updateBoundingInfo","_meshes","_indexBufferIsUpdatable","_positionsCache","useBoundingInfoFromGeometry","_indices","_vertexArrayObjects","_extend","_notifyUpdate","setVerticesBuffer","removeVerticesData","_disposeVertexArrayObjects","disposeExistingBuffer","numOfMeshes","_updateExtend","_resetPointsArrayCache","buildBoundingInfo","_createGlobalSubMesh","isUnIndexed","synchronizeInstances","updateVerticesDataDirectly","getVertexBuffer","hasBoundingInfo","_bind","indexToBind","overrideVertexArrayObjects","vbs","getVertexBuffers","vaos","isVertexBufferUpdatable","_delayInfo","getVerticesDataKinds","updateIndices","gpuMemoryOnly","needToUpdateSubMeshes","updateDynamicIndexBuffer","orig","getIndexBuffer","_releaseVertexArrayObject","releaseForMesh","shouldDispose","_invalidateInstanceVertexArrayObject","_geometry","previousGeometry","_positions","_applyToMesh","_syncGeometryWithMorphTargetManager","onGeometryUpdated","_markSubMeshesAsAttributesDirty","load","onLoaded","_queueLoad","delayLoadingFile","_delayLoadingFunction","JSON","toLeftHanded","tIndices","tTemp","tPositions","tNormals","_generatePointsArray","arrayIdx","stopChecking","_toNumberArray","vbName","serializeVerticeData","uv2s","uv3s","uv4s","uv5s","uv6s","_loadedUniqueId","parsedGeometry","geometryUniqueId","geometryId","_GetGeometryByLoadedUniqueId","binaryInfo","_binaryInfo","positionsAttrDesc","positionsData","normalsAttrDesc","normalsData","tangetsAttrDesc","tangentsData","uvsAttrDesc","uvsData","uvs2AttrDesc","uvs2Data","uvs3AttrDesc","uvs3Data","uvs4AttrDesc","uvs4Data","uvs5AttrDesc","uvs5Data","uvs6AttrDesc","uvs6Data","colorsAttrDesc","colorsData","matricesIndicesAttrDesc","matricesIndicesData","floatIndices","matricesIndicesExtraAttrDesc","matricesWeightsAttrDesc","matricesWeightsData","indicesAttrDesc","indicesData","subMeshesAttrDesc","subMeshesData","AddToMesh","matricesIndex","_CleanMatricesWeights","parsedSubMesh","_shouldGenerateFlatShading","convertToFlatShadedMesh","noInfluenceBoneIndex","skeletonId","influencers","numBoneInfluencer","weight","firstZeroWeight","mweight","boundingBoxMinimum","boundingBoxMaximum","hasUVs","hasUVs2","hasUVs3","hasUVs4","hasUVs5","hasUVs6","hasColors","hasMatricesIndices","hasMatricesWeights","ImportVertexData","PerformanceMonitor","frameSampleSize","_rollingFrameTime","RollingAverage","sampleFrame","timeMs","_lastFrameTimeMs","dt","averageFrameTime","averageFrameTimeVariance","variance","instantaneousFrameTime","averageFPS","instantaneousFPS","isSaturated","_samples","bottomValue","_pos","_sampleCount","_m2","_wrapPosition","setAlphaConstants","noDepthWriteChange","ONE","ONE_MINUS_SRC_ALPHA","SRC_ALPHA","ZERO","ONE_MINUS_SRC_COLOR","DST_COLOR","CONSTANT_COLOR","ONE_MINUS_CONSTANT_COLOR","CONSTANT_ALPHA","ONE_MINUS_CONSTANT_ALPHA","DST_ALPHA","ONE_MINUS_DST_COLOR","ONE_MINUS_DST_ALPHA","getAlphaMode","setAlphaEquation","equation","getAlphaEquation","_readTexturePixelsSync","noDataConversion","dummy","sizeOrDstBuffer","copyBuffer","allocateAndCopyTypedBuffer","_readTexturePixels","dataLength","Engine","_createImageBitmapFromSource","imageSource","image","decode","imageBitmap","resizeImageBitmap","bufferWidth","bufferHeight","getImageData","engineIndex","sceneIndex","_RescalePostProcessFactory","performanceMonitor","_performanceMonitor","compatibilityMode","_compatibilityMode","enableOfflineSupport","disableManifestCheck","disableContextMenu","onResizeObservable","onCanvasBlurObservable","onCanvasFocusObservable","onCanvasPointerOutObservable","onBeginFrameObservable","customAnimationFrameRequester","onBeforeShaderCompilationObservable","onAfterShaderCompilationObservable","_deterministicLockstep","_lockstepMaxSteps","_timeStep","_fps","_deltaTime","_drawCalls","disablePerformanceMonitorInBackground","_renderPassNames","_connectVREvents","_prepareVRComponent","autoEnableWebVR","initWebVR","_rescalePostProcess","_onCanvasFocus","_onCanvasBlur","_onCanvasContextMenu","_onBlur","_onFocus","_onCanvasPointerOut","ev","elementFromPoint","hostWindow","doNotHandleTouchAction","_disableTouchAction","AudioEngineFactory","_onFullscreenChange","fullscreenElement","_pointerLockRequested","_RequestPointerlock","_onPointerLockChange","OfflineProviderFactory","viewportOwner","getScreenAspectRatio","getRenderingCanvasClientRect","generateMipMapsForCubemap","unbind","getDepthWrite","getStencilMask","setStencilMask","getStencilFunction","getStencilFunctionReference","getStencilFunctionMask","setStencilFunction","setStencilFunctionReference","setStencilFunctionMask","getStencilOperationFail","getStencilOperationDepthFail","getStencilOperationPass","setStencilOperationFail","setStencilOperationDepthFail","setStencilOperationPass","setDitheringState","setRasterizerState","RASTERIZER_DISCARD","getDepthFunction","setDepthFunction","setDepthFunctionToGreater","setDepthFunctionToGreaterOrEqual","setDepthFunctionToLess","setDepthFunctionToLessOrEqual","cacheStencilState","_cachedStencilBuffer","_cachedStencilFunction","_cachedStencilMask","_cachedStencilOperationPass","_cachedStencilOperationFail","_cachedStencilOperationDepthFail","_cachedStencilReference","restoreStencilState","setDirectViewport","scissorClear","enableScissor","disableScissor","SCISSOR_TEST","scissor","numDrawCalls","_submitVRFrame","disableVR","isVRPresenting","_requestVRFrame","getVertexShaderSource","shaders","getAttachedShaders","getFragmentShaderSource","postProcessInput","_forcedOutputTexture","_currentRenderTextureInd","_renderFrame","requestID","_renderViews","switchFullscreen","requestPointerLock","exitFullscreen","enterFullscreen","_RequestFullscreen","_ExitFullscreen","enterPointerlock","exitPointerlock","_ExitPointerlock","_measureFps","camIndex","transformFeedback","deleteTransformFeedback","createTransformFeedback","bindTransformFeedback","setTranformFeedbackVaryings","getRenderPassNames","getCurrentRenderPassName","_RenderPassIdCounter","createRenderTargetTexture","externalTextureSamplerBinding","getEffect","onApply","hostingScene","copyTexImage2D","getFps","wrapWebGLTexture","hasMipMaps","_uploadImageToTexture","updateTextureComparisonFunction","createInstancesBuffer","deleteInstancesBuffer","_clientWaitAsync","sync","flags","intervalms","clientWaitSync","WAIT_FAILED","TIMEOUT_EXPIRED","_readPixelsAsync","outputBuffer","PIXEL_PACK_BUFFER","STREAM_READ","fenceSync","SYNC_GPU_COMMANDS_COMPLETE","deleteSync","getBufferSubData","hideLoadingUI","touchAction","webkitTapHighlightColor","displayLoadingUI","loadingScreen","_loadingScreen","DefaultLoadingScreenFactory","loadingUIText","loadingUIBackgroundColor","createVideoElement","constraints","exitPointerLock","requestFunction","requestFullscreen","webkitRequestFullscreen","anyDoc","webkitCancelFullScreen","getFontOffset","font","innerHTML","block","div","whiteSpace","fontAscent","fontHeight","ascent","descent","Space","Coordinate","ALPHA_DISABLE","ALPHA_ADD","ALPHA_COMBINE","ALPHA_SUBTRACT","ALPHA_MULTIPLY","ALPHA_MAXIMIZED","ALPHA_ONEONE","ALPHA_PREMULTIPLIED","ALPHA_PREMULTIPLIED_PORTERDUFF","ALPHA_INTERPOLATE","ALPHA_SCREENMODE","DELAYLOADSTATE_NONE","DELAYLOADSTATE_LOADED","DELAYLOADSTATE_LOADING","DELAYLOADSTATE_NOTLOADED","NEVER","LESS","EQUAL","GREATER","NOTEQUAL","INCR","DECR","INVERT","INCR_WRAP","DECR_WRAP","TEXTURE_CLAMP_ADDRESSMODE","TEXTURE_WRAP_ADDRESSMODE","TEXTURE_MIRROR_ADDRESSMODE","TEXTUREFORMAT_ALPHA","TEXTUREFORMAT_LUMINANCE","TEXTUREFORMAT_LUMINANCE_ALPHA","TEXTUREFORMAT_RGB","TEXTUREFORMAT_RGBA","TEXTUREFORMAT_RED","TEXTUREFORMAT_R","TEXTUREFORMAT_RG","TEXTUREFORMAT_RED_INTEGER","TEXTUREFORMAT_R_INTEGER","TEXTUREFORMAT_RG_INTEGER","TEXTUREFORMAT_RGB_INTEGER","TEXTUREFORMAT_RGBA_INTEGER","TEXTURETYPE_UNSIGNED_BYTE","TEXTURETYPE_UNSIGNED_INT","TEXTURETYPE_FLOAT","TEXTURETYPE_HALF_FLOAT","TEXTURETYPE_BYTE","TEXTURETYPE_SHORT","TEXTURETYPE_UNSIGNED_SHORT","TEXTURETYPE_INT","TEXTURETYPE_UNSIGNED_INTEGER","TEXTURETYPE_UNSIGNED_SHORT_4_4_4_4","TEXTURETYPE_UNSIGNED_SHORT_5_5_5_1","TEXTURETYPE_UNSIGNED_SHORT_5_6_5","TEXTURETYPE_UNSIGNED_INT_2_10_10_10_REV","TEXTURETYPE_UNSIGNED_INT_24_8","TEXTURETYPE_UNSIGNED_INT_10F_11F_11F_REV","TEXTURETYPE_UNSIGNED_INT_5_9_9_9_REV","TEXTURETYPE_FLOAT_32_UNSIGNED_INT_24_8_REV","TEXTURE_NEAREST_SAMPLINGMODE","TEXTURE_BILINEAR_SAMPLINGMODE","TEXTURE_TRILINEAR_SAMPLINGMODE","TEXTURE_NEAREST_NEAREST_MIPLINEAR","TEXTURE_LINEAR_LINEAR_MIPNEAREST","TEXTURE_LINEAR_LINEAR_MIPLINEAR","TEXTURE_NEAREST_NEAREST_MIPNEAREST","TEXTURE_NEAREST_LINEAR_MIPNEAREST","TEXTURE_NEAREST_LINEAR_MIPLINEAR","TEXTURE_NEAREST_LINEAR","TEXTURE_NEAREST_NEAREST","TEXTURE_LINEAR_NEAREST_MIPNEAREST","TEXTURE_LINEAR_NEAREST_MIPLINEAR","TEXTURE_LINEAR_LINEAR","TEXTURE_LINEAR_NEAREST","TEXTURE_EXPLICIT_MODE","TEXTURE_SPHERICAL_MODE","TEXTURE_PLANAR_MODE","TEXTURE_CUBIC_MODE","TEXTURE_PROJECTION_MODE","TEXTURE_SKYBOX_MODE","TEXTURE_INVCUBIC_MODE","TEXTURE_EQUIRECTANGULAR_MODE","TEXTURE_FIXED_EQUIRECTANGULAR_MODE","TEXTURE_FIXED_EQUIRECTANGULAR_MIRRORED_MODE","SCALEMODE_FLOOR","SCALEMODE_NEAREST","SCALEMODE_CEILING","Axis","convertRHSToLHS","Compose","FromEulerAngles","TransformNode","_billboardMode","useBillboardPosition","BILLBOARDMODE_USE_POSITION","_computeUseBillboardPath","preserveParentRotationForBillboard","_preserveParentRotationForBillboard","useBillboardPath","BILLBOARDMODE_NONE","_infiniteDistance","isPure","_forward","_up","_right","_rotation","_rotationQuaternion","_scaling","_transformToBoneReferal","_isAbsoluteSynced","scalingDeterminant","ignoreNonUniformScaling","reIntegrateRotationIntoRotationQuaternion","_poseMatrix","_localMatrix","_usePivotMatrix","_absolutePosition","_absoluteScaling","_absoluteRotationQuaternion","_pivotMatrix","_postMultiplyPivotMatrix","_isWorldMatrixFrozen","onAfterWorldMatrixUpdateObservable","_nonUniformScaling","isUsingPivotMatrix","newRotation","newScaling","updatePoseMatrix","getPoseMatrix","pivotMatrixUpdated","localMatrixUpdated","absolutePosition","getAbsolutePosition","absoluteScaling","_syncAbsoluteScalingAndRotation","absoluteRotationQuaternion","setPreTransformMatrix","setPivotMatrix","postMultiplyPivotMatrix","_pivotMatrixInverse","Invert","getPivotMatrix","instantiateHierarchy","onNewNodeCreated","getChildTransformNodes","freezeWorldMatrix","newWorldMatrix","_afterComputeWorldMatrix","unfreezeWorldMatrix","isWorldMatrixFrozen","setAbsolutePosition","absolutePositionX","absolutePositionY","absolutePositionZ","invertParentWorldMatrix","setPositionWithLocalVector","TransformNormal","getPositionExpressedInLocalSpace","invLocalWorldMatrix","locallyTranslate","TransformCoordinates","lookAt","yawCor","pitchCor","rollCor","space","LOCAL","dv","_LookAtVectorCache","pos","setDirection","WORLD","rotationMatrix","parentRotationMatrix","quaternionRotation","FromEulerVectorToRef","setPivotPoint","tmat","Translation","getPivotPoint","getPivotPointToRef","getAbsolutePivotPoint","getAbsolutePivotPointToRef","setParent","preserveScalingSign","updatePivot","quatRotation","invParentMatrix","composedMatrix","currentRotation","_TmpRotation","_updateNonUniformScalingState","attachToBone","affectedTransformNode","_currentParentWhenAttachingToBone","getSkeleton","getFinalMatrix","detachFromBone","resetToPreviousParent","rotate","_RotationAxisCache","rotateAround","tmpVector","finalScale","finalTranslation","finalRotation","translationMatrix","translationMatrixInv","finalMatrix","translate","displacementVector","tempV3","addRotation","accumulation","_getEffectiveParent","isWorldMatrixCameraDependent","currentRenderId","_TmpScaling","cameraWorldMatrix","cameraGlobalPosition","_TmpTranslation","scaleMatrix","BillboardUseParentOrientation","storedTranslation","BILLBOARDMODE_ALL","eulerAngles","BILLBOARDMODE_X","BILLBOARDMODE_Y","BILLBOARDMODE_Z","camInObjSpace","resetLocalMatrix","independentOfChildren","bakedMatrix","tmpRotationQuaternion","registerAfterWorldMatrixUpdate","unregisterAfterWorldMatrixUpdate","getPositionInCameraSpace","getDistanceToCamera","currentSerializationObject","localMatrix","parsedTransformNode","pivotMatrix","normalizeToUnitCube","ignoreRotation","storedRotation","storedRotationQuaternion","sizeVec","maxDimension","serializeAsQuaternion","_MeshCollisionData","_checkCollisions","_collisionMask","_collisionGroup","_surroundingMeshes","_collider","_oldPositionForCollisions","_diffPositionForCollisions","_collisionResponse","_FacetDataStorage","facetNb","partitioningSubdivisions","partitioningBBoxRatio","facetDataEnabled","facetParameters","facetDepthSort","facetDepthSortEnabled","_InternalAbstractMeshDataInfo","_hasVertexAlpha","_useVertexColors","_numBoneInfluencers","_applyFog","_receiveShadows","_facetData","_visibility","_skeleton","_layerMask","_computeBonesUsingShaders","_isActiveIntermediate","_onlyForInstancesIntermediate","_collisionRetryCount","_morphTargetManager","_renderingGroupId","_bakedVertexAnimationManager","_material","_meshCollisionData","_enableDistantPicking","_rawBoundingInfo","AbstractMesh","nb","mustDepthSortFacets","facetDepthSortFrom","collisionRetryCount","retryCount","isFacetDataEnabled","bakedVertexAnimationManager","_markSubMeshesAsMiscDirty","rawBoundingInfo","onCollide","_onCollideObserver","onCollideObservable","onCollisionPositionChange","_onCollisionPositionChangeObserver","onCollisionPositionChangeObservable","_markSubMeshesAsDirty","pointerOverDisableMeshTesting","meshMap","onMaterialChangedObservable","_unBindEffect","_materialForRenderPass","setMaterialForRenderPass","_markSubMeshesAsLightDirty","hasVertexAlpha","useVertexColors","numBoneInfluencers","applyFog","collisionMask","collisionResponse","collisionGroup","surroundingMeshes","_lightSources","needInitialSkinMatrix","_unregisterMeshWithPoseMatrix","_registerMeshWithPoseMatrix","_bonesTransformMatrices","_waitingMaterialId","CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY","definedFacingForward","_occlusionQuery","isNearPickable","isNearGrabbable","showSubMeshesBoundingBox","isBlocker","outlineColor","Red","outlineWidth","overlayColor","overlayAlpha","useOctreeForRenderingSelection","useOctreeForPicking","useOctreeForCollisions","doNotSyncBoundingInfo","ellipsoid","ellipsoidOffset","edgesWidth","edgesColor","_masterMesh","_boundingInfoIsDirty","_waitingData","lods","_transformMatrixTexture","onRebuildObservable","_onCollisionPositionChange","collisionId","collidedMesh","_uniformBuffer","_buildUniformLayout","transferToEffect","getMeshUniformBuffer","initialCall","canAffectMesh","_resyncLightSource","isIn","getRawBoundingInfo","useBones","_preActivateForIntermediateRendering","renderId","intermediateRendering","movePOV","amountRight","amountUp","amountForward","calcMovePOV","rotMatrix","translationDelta","defForwardMult","rotatePOV","flipBack","twirlClockwise","tiltRight","calcRotatePOV","applyMorph","_refreshBoundingInfo","_getPositionData","_getData","faceIndexCount","positionIndex","targetCount","targetMorph","influence","morphTargetPositions","getPositions","needExtras","matricesIndicesExtraData","matricesWeightsExtraData","skeletonMatrices","getTransformMatrices","tempMatrix","matWeightIdx","inf","FromFloat32ArrayToRefScaled","getNormalsData","getPositionData","_updateSubMeshesBoundingInfo","otherBoundingInfo","collisionEnabled","moveWithCollisions","displacement","coordinator","createCollider","_radius","getNewPosition","_collideForSubMesh","transformMatrix","_collide","_shouldConvertRHS","_processCollisionsForSubMeshes","collisionsScalingMatrix","collisionsTransformMatrix","onlyBoundingInfo","worldToUse","skipBoundingInfo","pickingInfo","anySubmeshSupportIntersect","worldOrigin","releaseSubMeshes","includedOnlyMeshes","generators","getShadowGenerators","shadowMap","getShadowMap","isOcclusionQueryInProgress","disableFacetData","addChild","_initFacetData","updateFacetData","depthSortedIndices","needs32bits","facetDepthSortFunction","f1","f2","depthSortedFacet","facetDepthSortOrigin","bbSizeMax","getFacetLocalNormals","getFacetLocalPositions","getFacetLocalPartitioning","ComputeNormals","sind","facetData","getFacetPosition","getFacetPositionToRef","localPos","getFacetNormal","getFacetNormalToRef","localNorm","getFacetsAtLocalCoordinates","getClosestFacetAtCoordinates","projected","checkFace","facing","invMat","invVect","closest","getClosestFacetAtLocalCoordinates","tmpx","tmpy","tmpz","t0","projx","projy","projz","facetsInBlock","fib","shortest","tmpDistance","getFacetDataParameters","createNormals","alignWithNormal","upDirection","axisX","axisZ","_checkOcclusionQuery","disableEdgesRendering","enableEdgesRendering","checkVerticesInsteadOfIndices","getConnectedParticleSystems","addClipPlaneUniforms","prepareStringDefinesForClipPlanes","primaryHolder","secondaryHolder","clipPlane","clipPlane2","clipPlane3","clipPlane4","clipPlane5","clipPlane6","bindClipPlane","setClipPlane","OCCLUSION_TYPE_NONE","OCCLUSION_TYPE_OPTIMISTIC","OCCLUSION_TYPE_STRICT","OCCLUSION_ALGORITHM_TYPE_ACCURATE","OCCLUSION_ALGORITHM_TYPE_CONSERVATIVE","CULLINGSTRATEGY_STANDARD","CULLINGSTRATEGY_OPTIMISTIC_INCLUSION","CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY","MaterialHelper","optimizeUVAllocation","getTextureMatrix","coordinatesIndex","useLogarithmicDepth","pointsCloud","alphaTest","applyDecalAfterDetail","GetFogState","wasOrtho","wasPersp","isOrtho","isPersp","useInstances","useClipPlane","useThinInstances","PrepareDefinesForCamera","prepareDefinesForClipPlanes","materialSupportsBoneTexture","isUsingTextureForMatrices","nonExcluded","excludedSkinnedMesh","supportsUVs","supportsTangents","supportsNormals","numInfluencers","isUsingTextureForTargets","useVertexColor","useMorphTargets","useVertexAlpha","useBakedVertexAnimation","hasVertexColors","PrepareDefinesForBones","PrepareDefinesForMorphTargets","PrepareDefinesForBakedVertexAnimation","previousMultiview","MULTIVIEW","getViewCount","needAlphaBlending","previousDefine","ORDER_INDEPENDENT_TRANSPARENCY","previousDefine16Bits","ORDER_INDEPENDENT_TRANSPARENCY_16BITS","canRenderToMRT","previousPrePass","PREPASS","texturesList","SCENE_MRT_COUNT","mrtCount","getIndex","lightIndex","specularSupported","needNormals","needRebuild","prepareLightSpecificDefines","falloffType","specular","specularEnabled","shadowGenerator","getShadowGenerator","lightmapMode","maxSimultaneousLights","disableLighting","PrepareDefinesForLight","caps","projectedLightTexture","uniformBuffersList","updateOnlyBuffersList","uniformsListOrOptions","PrepareUniformsAndSamplersForLight","rank","lightFallbackRank","addFallback","attribs","_TmpMorphInfluencers","NUM_MORPH_INFLUENCERS","PrepareAttributesForMorphTargets","maxAttributesCount","tangent","uv","addCPUSkinningFallback","PushAttributesForInstances","INSTANCESCOLOR","needsPreviousMatrices","useSpecular","_bindLight","BindLight","linearSpace","_TempFogColor","prePassConfiguration","boneTexture","getTransformMatrixTexture","previousBones","_CopyBonesTransformationMatrices","abstractMesh","influences","LN2","MaterialStencilState","copyTo","MaterialPluginEvent","Orientation","AnimationKeyInterpolation","Material","_alpha","MiscDirtyFlag","PrePassDirtyFlag","_backFaceCulling","TextureDirtyFlag","_cullBackFaces","blockDirtyMechanism","_blockDirtyMechanism","markDirty","atomicMaterialsUpdate","_eventInfo","_callbackPluginEventHasRenderTargetTextures","_onBindObserver","onUnBindObservable","_onUnBindObservable","onEffectCreatedObservable","_onEffectCreatedObservable","alphaMode","_needDepthPrePass","checkReadyOnEveryCall","isPrePassCapable","wireframe","_fillMode","WireFrameFillMode","LineListDrawMode","LineLoopDrawMode","LineStripDrawMode","TriangleFillMode","PointFillMode","PointListDrawMode","_setDrawWrapper","doNotAdd","shadowDepthWrapper","allowShaderHotSwapping","checkReadyOnlyOnce","disableDepthWrite","disableColorWrite","forceDepthWrite","depthFunction","separateCullingPass","pointSize","_useUBO","_cachedDepthWriteState","_cachedColorWriteState","_cachedDepthFunctionState","_uniformBufferLayoutBuilt","_callbackPluginEventGeneric","_callbackPluginEventIsReadyForSubMesh","_callbackPluginEventPrepareDefines","_callbackPluginEventPrepareDefinesBeforeAttributes","_callbackPluginEventHardBindForSubMesh","_callbackPluginEventBindForSubMesh","_callbackPluginEventFillRenderTargetTextures","_forceAlphaTest","_transparencyMode","setScene","_dirtyCallbacks","_markAllSubMeshesAsTexturesDirty","_markAllSubMeshesAsLightsDirty","_markAllSubMeshesAsFresnelDirty","_markAllSubMeshesAsAttributesDirty","_markAllSubMeshesAsMiscDirty","_markAllSubMeshesAsPrePassDirty","_markAllSubMeshesAsAllDirty","_materialContext","ClockWiseSideOrientation","CounterClockWiseSideOrientation","OnEventObservable","Created","_isMaterial","isFrozen","isReadyForSubMesh","transparencyMode","MATERIAL_ALPHATESTANDBLEND","_markAllSubMeshesAsTexturesAndMiscDirty","_disableAlphaBlending","MATERIAL_OPAQUE","MATERIAL_ALPHATEST","_shouldTurnAlphaTestOn","getAlphaTestTexture","forceMaterialDirty","AllDirtyFlag","_preBind","overrideOrientation","buildUniformLayout","PrepareUniformBuffer","bindForSubMesh","bindOnlyWorldMatrix","bindView","_needToBindSceneUbo","bindViewProjection","_afterBind","BindSceneUniformBuffer","getAnimatables","GetAnimatables","getActiveTextures","activeTextures","GetActiveTextures","hasTexture","HasTexture","_clonePlugins","targetMaterial","_serializePlugins","_parsePlugins","pluginManager","plugin","_plugins","targetPlugin","getPlugin","getBindedMeshes","forceCompilation","localOptions","currentHotSwapingState","checkReady","clipPlaneState","allDone","lastError","tempSubMesh","forceCompilationAsync","_DirtyCallbackArray","_TextureDirtyCallBack","LightDirtyFlag","_LightsDirtyCallBack","FresnelDirtyFlag","_FresnelDirtyCallBack","AttributesDirtyFlag","_AttributeDirtyCallBack","_MiscDirtyCallBack","_PrePassDirtyCallBack","_markAllSubMeshesAsDirty","_RunDirtyCallBacks","_markScenePrePassDirty","enablePrePassRenderer","_AllDirtyCallBack","_markAllSubMeshesAsImageProcessingDirty","_ImageProcessingDirtyCallBack","_markAllSubMeshesAsFresnelAndMiscDirty","_FresnelAndMiscDirtyCallBack","_TextureAndMiscDirtyCallBack","_checkScenePerformancePriority","setPrePassRenderer","forceDisposeEffect","forceDisposeTextures","notBoundToMesh","Disposed","plugins","parsedMaterial","customType","overloadedAlbedo","BABYLON","LegacyPBRMaterial","pluginClassName","pluginData","pluginClassType","TriangleStripDrawMode","TriangleFanDrawMode","MATERIAL_ALPHABLEND","MATERIAL_NORMALBLENDMETHOD_WHITEOUT","MATERIAL_NORMALBLENDMETHOD_RNM","MultiMaterial","subMaterials","_subMaterials","_hookArray","_waitingSubMaterialsUniqueIds","oldPush","oldSplice","deleted","subMaterial","cloneChildren","materialsUniqueIds","matIndex","subMat","forceDisposeChildren","parsedMultiMaterial","multiMaterial","subMatId","MeshLODLevel","distanceOrScreenCoverage","_CreationDataStorage","_InstanceDataStorage","visibleInstances","batchCache","_InstancesBatch","batchCacheReplacementModeInFrozenMode","instancesBufferSize","mustReturn","renderSelf","_ThinInstanceDataStorage","matrixBuffer","previousMatrixBuffer","matrixBufferSize","matrixData","worldMatrices","_InternalMeshDataInfo","_areNormalsFrozen","_preActivateId","_LODLevels","_useLODScreenCoverage","_effectiveMaterial","_forcedInstanceCount","_overrideRenderingFillMode","Mesh","useLODScreenCoverage","_internalMeshDataInfo","_sortLODLevels","_sourcePositions","_sourceNormals","_onBeforeRenderObservable","onBeforeBindObservable","_onBeforeBindObservable","_onAfterRenderObservable","onBetweenPassObservable","_onBetweenPassObservable","onBeforeDrawObservable","_onBeforeDrawObservable","onBeforeDraw","_onBeforeDrawObserver","_thinInstanceDataStorage","forcedInstanceCount","overrideRenderingFillMode","cloneMeshMap","worldMatrixInstancedBuffer","_instanceDataStorage","instancesData","previousWorldMatrixInstancedBuffer","instancesPreviousData","manualUpdateOfWorldMatrixInstancedBuffer","manualUpdate","manualUpdateOfPreviousWorldMatrixInstancedBuffer","previousManualUpdate","forceWorldMatrixInstancedBufferUpdate","forceMatrixUpdates","clonePhysicsImpostor","_creationDataStorage","_originalBuilderSideOrientation","overrideMaterialSideOrientation","ignoreCameraMaxZ","_onBeforeDraw","isInstance","_internalMetadata","getPhysicsEngine","physicsEngine","getPluginVersion","impostor","getImpostorForPhysicsObject","physicsImpostor","physicsBody","_onMeshReadyObserverAdded","onMeshReadyObservable","_checkReadinessObserver","doNotInstantiate","createInstance","newSourcedMesh","_isMesh","ib","hasLODLevels","getLODLevels","sortingOrderFactor","addLODLevel","getLODLevelAtDistance","internalDataInfo","removeLODLevel","bSphere","distanceToCamera","compareValue","compareSign","meshArea","onLODLevelSelection","_checkDelayState","bypassInstanceData","_userInstancedBuffersStorage","kinds","forceInstanceSupport","generator","renderPassIds","areNormalsFrozen","freezeNormals","unfreezeNormals","overridenInstanceCount","sceneRenderId","intermediateDefaultRenderId","_registerInstanceForRenderId","defaultRenderId","selfDefaultRenderId","previousRenderId","thinInstanceRefreshBoundingInfo","edgesShareWithInstances","edgesRenderer","customInstances","needToRecreate","submesh","subdivide","subdivisionSize","CreateFromIndices","markVerticesDataAsUpdatable","CreateGeometryForMesh","makeGeometryUnique","updateMeshPositions","positionFunction","computeNormals","oldGeometry","allowInstancedRendering","_getRenderingFillMode","vertexArrayObjects","_draw","_getInstancesRenderList","isReplacementMode","previousBatch","isInIntermediateRendering","onlyForInstances","_renderWithInstances","batch","visibleInstanceCount","instanceStorage","currentInstancesBufferSize","instancesPreviousBuffer","needUpdateBuffer","masterMeshPreviousWorldMatrix","INSTANCEDMESH_SORT_TRANSPARENT","instanceIndex","instanceMesh","m1","m2","_previousWorldMatrix","strides","sizes","_processInstancedBuffers","_currentDrawContext","useInstancing","_renderWithThinInstances","previousMatrixData","_thinInstanceCreateMatrixBuffer","_processRendering","instanceCount","visibleInstancesForSubMesh","effectiveMeshReplacement","_occlusionDataStorage","forceRenderingWhenOccluded","oldCameraMaxZ","oldCamera","instanceDataStorage","effectiveMesh","mainDeterminant","cleanMatrixWeights","_normalizeSkinWeightsAndExtra","_normalizeSkinFourWeights","numWeights","recip","validateSkinning","skinned","valid","report","numberNotSorted","missingWeights","maxUsedWeights","numberNotNormalized","numInfluences","usedWeightCounts","lastWeight","usedWeights","tolerance","numBones","numBadBoneIndices","getBinaryData","_syncSubMeshes","setMaterialById","bakeTransformIntoVertices","submeshes","flipFaces","bakeCurrentTransformIntoVertices","bakeIndependentlyOfChildren","_disposeInstanceSpecificData","_disposeThinInstanceSpecificData","applyDisplacementMap","minHeight","maxHeight","uvOffset","uvScale","forceUpdate","heightMapWidth","heightMapHeight","applyDisplacementMapFromBuffer","_getFlattenedNormals","normalsCount","flipNormalGeneration","p3","Normalize","localIndex","_convertToUnIndexedMesh","flattenNormals","separateVertices","newData","previousSubmeshes","targetIndex","setPositions","getNormals","setNormals","getTangents","setTangents","getUVs","setUVs","synchronize","previousOne","convertToUnIndexedMesh","flipNormals","vertex_data","ExtractFromMesh","increaseVertices","numberPerEdge","currentIndices","tempIndices","deltaPosition","deltaNormal","deltaUV","side","uvPtr","normalsPtr","positionPtr","forceSharedVertices","currentUVs","currentPositions","currentColors","currentMatrixIndices","currentMatrixWeights","currentMatrixIndicesExtra","currentMatrixWeightsExtra","matrixIndices","matrixWeights","matrixIndicesExtra","matrixWeightsExtra","pstring","indexPtr","uniquePositions","ptr","facet","physicObject","jsonObject","_instancedMeshFactory","optimizeIndices","vectorPositions","dupes","SyncAsyncForLoop","realPos","testedPosition","againstPosition","originalSubMeshes","pickable","materialUniqueId","morphTargetManagerId","getPhysicsImpostor","physicsMass","getParam","physicsFriction","physicsRestitution","serializationInstance","thinInstances","enablePicking","thinInstanceEnablePicking","_userThinInstanceBuffersStorage","userThinInstance","renderOverlay","morphTarget","getActiveTarget","parsedMesh","_LinesMeshParser","_GroundMeshParser","_GoldbergMeshParser","_GreasedLineMeshParser","_TrailMeshParser","showBoundingBox","useFlatShading","_ImportGeometry","_PhysicsImpostorParser","lodMeshIds","ids","distances","lodDistances","coverages","lodCoverages","parsedInstance","thinInstanceSetBuffer","setPositionsForCPUSkinning","setNormalsForCPUSkinning","_softwareSkinningFrameId","hasNormals","tempVector3","minVector","maxVector","meshesOrMinMaxVector","minMaxVector","MinMax","Center","disposeSource","allow32BitsIndices","meshSubclass","subdivideWithSubMeshes","multiMultiMaterials","_MergeMeshesCoroutine","yieldAfterMS","createYieldingScheduler","materialArray","materialIndexArray","indiceArray","currentOverrideMaterialSideOrientation","getVertexDataFromMesh","sourceVertexData","sourceTransform","meshVertexDatas","mergeCoroutine","mergeCoroutineStep","applyToCoroutine","applyToCoroutineStep","addInstance","_indexInSourceMeshInstanceArray","removeInstance","setMaterialByID","pathArray","closeArray","diameter","diameterTop","diameterBottom","tube","radialSegments","tubularSegments","points","dashSize","gapSize","dashNb","shape","holes","earcutInjection","cap","scaleFunction","rotationFunction","ribbonCloseArray","ribbonClosePath","xmin","xmax","alphaFilter","radiusFunction","NO_CAP","CAP_START","CAP_END","CAP_ALL","NO_FLIP","FLIP_TILE","ROTATE_TILE","FLIP_ROW","ROTATE_ROW","FLIP_N_ROTATE_TILE","FLIP_N_ROTATE_ROW","CENTER","LEFT","RIGHT","TOP","BOTTOM","AutoRotationBehavior","_zoomStopsAnimation","_idleRotationSpeed","_idleRotationWaitTime","_idleRotationSpinupTime","targetAlpha","_isPointerDown","_lastFrameTime","_lastInteractionTime","_cameraRotationSpeed","_lastFrameRadius","zoomStopsAnimation","idleRotationSpeed","speed","idleRotationWaitTime","idleRotationSpinupTime","rotationInProgress","_attachedCamera","_onPrePointerObservableObserver","pointerInfoPre","_onAfterCheckInputsObserver","_reachTargetAlpha","_applyUserInteraction","timeToRotation","resetLastInteractionTime","customTime","_userIsZooming","inertialRadiusOffset","_shouldAnimationStopForInteraction","zoomHasHitLimit","_userIsMoving","inertialAlphaOffset","inertialBetaOffset","inertialPanningX","inertialPanningY","Angle","radians","_radians","degrees","Arc2","startPoint","midPoint","endPoint","startToMid","midToEnd","centerPoint","startAngle","BetweenTwoPoints","a1","a2","a3","FromDegrees","Path2","_points","_length","addLineTo","newPoint","previousPoint","addArcTo","midX","midY","endX","endY","numberOfSegments","increment","currentAngle","addQuadraticCurveTo","controlX","controlY","val0","val1","val2","addBezierCurveTo","originTangentX","originTangentY","destinationTangentX","destinationTangentY","val3","isPointInside","isInside","edgeLow","edgeHigh","edgeDx","edgeDy","EPSILON","perpEdge","lastPoint","area","getPoints","getPointAtLengthPosition","normalizedLengthPosition","lengthPosition","previousOffset","bToA","nextOffset","dir","localOffset","Path3D","firstNormal","raw","alignTangentsWithPath","_curve","_distances","_tangents","_binormals","_pointAtData","previousPointArrayIndex","subPosition","interpolateReady","interpolationMatrix","_raw","_alignTangentsWithPath","_compute","getCurve","getBinormals","getDistances","getPointAt","_updatePointAtData","getTangentAt","interpolated","getNormalAt","getBinormalAt","getDistanceAt","getPreviousPointIndexAt","getSubPositionAt","getClosestPositionTo","smallestDistance","closestPosition","subLength","_start","curvePoints","endIndex","slicePoints","_getFirstNonNullVector","tg0","pp0","_normalVector","curTang","prevNor","prevBinor","_getLastNonNullVector","nNVector","nLVector","vt","va","tgl","interpolateTNB","_updateInterpolationMatrix","_setPointAtData","currentPoint","currentLength","targetLength","parentIndex","tangentFrom","normalFrom","binormalFrom","tangentTo","normalTo","binormalTo","quatFrom","RotationQuaternionFromAxis","quatTo","Slerp","Curve3","nbPoints","bez","v3","t1","nSeg","hermite","Hermite","catmullRom","pointsCount","CatmullRom","totalPoints","second","third","steps","fullCircle","vec1","vec2","vec3","len4","len1_sq","len2_sq","len3_sq","len4_sq","dStep","_computeLength","continue","curve","continuedPoints","EasingFunction","_easingMode","EASINGMODE_EASEIN","setEasingMode","easingMode","getEasingMode","easeInCore","ease","EASINGMODE_EASEOUT","EASINGMODE_EASEINOUT","CircleEase","QuadraticEase","SineEase","AnimationRange","Size","surface","otherSize","Animation","targetProperty","framePerSecond","totalFrame","loopMode","easingFunction","dataType","parseFloat","isFinite","ANIMATIONTYPE_FLOAT","ANIMATIONTYPE_QUATERNION","ANIMATIONTYPE_VECTOR3","ANIMATIONTYPE_VECTOR2","ANIMATIONTYPE_COLOR3","ANIMATIONTYPE_COLOR4","ANIMATIONTYPE_SIZE","frame","setKeys","setEasingFunction","animationType","ANIMATIONLOOPMODE_CONSTANT","_PrepareAnimation","beginDirectAnimation","beginDirectHierarchyAnimation","sourceAnimation","referenceFrame","cloneOriginal","clonedName","firstKey","lastKey","valueStore","referenceValue","referencePosition","referenceQuaternion","referenceScaling","keyPosition","keyQuaternion","keyScaling","referenceFound","rangeValue","getRange","fromKeyFound","toKeyFound","_getKeyValue","nextKey","animationState","repeatCount","_interpolate","ANIMATIONTYPE_MATRIX","targetValue","host","frameRate","transition","runtimeAnimations","_runtimeAnimations","hasRunningRuntimeAnimations","runtimeAnimation","enableBlending","_easingFunction","blendingSpeed","targetPropertyPath","ANIMATIONLOOPMODE_CYCLE","_UniqueIdGenerator","addEvent","removeEvents","getEvents","getKeys","getHighestFrame","nKeys","getEasingFunction","floatInterpolateFunction","floatInterpolateFunctionWithTangents","outTangent","inTangent","quaternionInterpolateFunction","quaternionInterpolateFunctionWithTangents","vector3InterpolateFunction","vector3InterpolateFunctionWithTangents","vector2InterpolateFunction","vector2InterpolateFunctionWithTangents","sizeInterpolateFunction","color3InterpolateFunction","color3InterpolateFunctionWithTangents","color4InterpolateFunction","color4InterpolateFunctionWithTangents","evaluate","currentFrame","highLimitValue","keysLength","startKey","endKey","interpolation","STEP","useTangent","frameDelta","floatValue","ANIMATIONLOOPMODE_YOYO","ANIMATIONLOOPMODE_RELATIVE","offsetValue","quatValue","vec3Value","vec2Value","color3Value","color4Value","AllowMatricesInterpolation","matrixInterpolateFunction","workValue","AllowMatrixDecomposeForInterpolation","DecomposeLerp","loopBehavior","animationKey","_inTangent","_outTangent","keyData","serializedAnimation","snippetId","snippet","jsonPayload","outputs","SnippetUrl","CreateFromSnippetAsync","ParseFromSnippetAsync","BouncingBehavior","transitionDuration","lowerRadiusTransitionRange","upperRadiusTransitionRange","_autoTransitionRange","_radiusIsAnimating","_radiusBounceTransition","_animatables","autoTransitionRange","_onMeshTargetChangedObserver","onMeshTargetChangedObservable","diagonal","_isRadiusAtLimit","lowerRadiusLimit","_applyBoundRadiusAnimation","upperRadiusLimit","radiusLimit","radiusDelta","EasingMode","CreateAnimation","_cachedWheelPrecision","wheelPrecision","TransitionTo","_clearAnimationLocks","BackEase","amplitude","FramingBehavior","onTargetFramingAnimationEndObservable","FitFrustumSidesMode","_radiusScale","_positionScale","_defaultElevation","_elevationReturnTime","_elevationReturnWaitTime","_framingTime","autoCorrectCameraLimitsAndSensibility","_betaIsAnimating","radiusScale","positionScale","defaultElevation","elevation","elevationReturnTime","elevationReturnWaitTime","framingTime","zoomOnMesh","_maintainCameraAboveGround","focusOnOriginXZ","zoomOnBoundingInfo","zoomOnMeshHierarchy","zoomOnMeshesHierarchy","zoomTarget","zoomTargetY","_vectorTransition","_calculateLowerRadiusFromModelBoundingSphere","IgnoreBoundsSizeMode","panningSensibility","_radiusTransition","useInputToRestoreState","boxVectorGlobalDiagonal","frustumSlope","_getFrustumSlope","distanceForHorizontalFrustum","distanceForVerticalFrustum","timeSinceInteraction","defaultBeta","limitBeta","_betaTransition","animatabe","frustumSlopeY","isUserIsMoving","ExponentialEase","exponent","exp","TargetCamera","_tmpUpVector","_tmpTargetVector","cameraDirection","cameraRotation","ignoreParentScaling","updateUpVectorFromRotation","_tmpQuaternion","noRotationConstraint","invertRotation","inverseRotationSpeed","lockedTarget","_currentTarget","_initialFocalDistance","_camMatrix","_cameraTransformMatrix","_cameraRotationMatrix","_referencePoint","_transformedReferencePoint","_deferredPositionUpdate","_deferredRotationQuaternionUpdate","_deferredRotationUpdate","_deferredUpdated","_deferOnly","_defaultUp","_cachedRotationZ","_cachedQuaternionRotationZ","getFrontPosition","_getLockedTargetPosition","_storedPosition","_storedRotation","_storedRotationQuaternion","lockedTargetPosition","_computeLocalCameraSpeed","atan","vDir","_decideIfNeedsToMove","_updatePosition","directionMultiplier","needToMove","needToRotate","_rotateUpVectorWithCameraRotationMatrix","_computeViewMatrix","parentWorldMatrix","camLeft","camRight","leftSign","rightSign","_getRigCamPositionAndTarget","halfSpace","_TargetFocalPoint","newFocalTarget","_TargetTransformMatrix","RotationAxis","_RigCamTransformMatrix","CameraInputTypes","CameraInputsManager","attachedToElement","attached","checkInputs","getSimpleName","_addCheckInputs","noPreventDefault","inputToRemove","rebuildInputCheck","removeByType","inputType","attachInput","attachElement","detachElement","disconnect","serializedCamera","inputsmgr","parsedInputs","parsedinput","BaseCameraPointersInput","_currentActiveButton","BackCompatCameraNoPreventDefault","previousPinchSquaredDistance","previousMultiTouchPanPosition","_pointA","_pointB","_altKey","_ctrlKey","_metaKey","_shiftKey","_buttonsPressed","_pointerInput","isTouch","isInVRExclusivePointerMode","srcElement","onTouch","onDoubleTap","ed","distX","distY","pinchSquaredDistance","multiTouchPanPosition","onMultiTouch","onButtonUp","onButtonDown","_observer","_onLostFocus","onLostFocus","_contextMenuBind","onContextMenu","RegisterTopRootEvents","UnregisterTopRootEvents","ArcRotateCameraPointersInput","angularSensibilityX","angularSensibilityY","pinchPrecision","pinchDeltaPercentage","useNaturalPinchZoom","pinchZoom","multiTouchPanning","multiTouchPanAndZoom","pinchInwards","_isPanClick","_twoFingerActivityCount","_isPinching","_computeMultiTouchPanning","moveDeltaX","moveDeltaY","_computePinchZoom","MinimumRadiusForPinch","_useCtrlForPanning","pointA","pointB","pinchToPanMaxDistance","_panningMouseButton","ArcRotateCameraKeyboardMoveInput","keysUp","keysDown","keysLeft","keysRight","keysReset","zoomingSensibility","useAltToZoom","angularSpeed","_onCanvasBlurObserver","_onKeyboardObserver","_ctrlPressed","_altPressed","ArcRotateCameraMouseWheelInput","zoomToMouseLocation","wheelDeltaPercentage","customComputeDeltaFromMouseWheel","_inertialPanning","_computeDeltaFromMouseWheelLegacyEvent","mouseWheelDelta","_wheel","platformScale","estimatedTargetRadius","targetInertia","_hitPlane","_updateHitPlane","_zoomToMouse","_zeroIfClose","FromPositionAndNormal","_getPosition","targetScreenOffset","intersectsPlane","inertiaComp","lowerLimit","upperLimit","directionToZoomLocation","ArcRotateCameraInputsManager","addMouseWheel","addPointers","addKeyboard","AddNodeConstructor","ArcRotateCamera","_target","targetHost","_targetHost","_upToYMatrix","_yToUpMatrix","setMatUp","RotationAlignToRef","pointers","keyboard","mousewheel","bouncingBehavior","_bouncingBehavior","useBouncingBehavior","framingBehavior","_framingBehavior","useFramingBehavior","autoRotationBehavior","_autoRotationBehavior","useAutoRotationBehavior","lowerAlphaLimit","upperAlphaLimit","lowerBetaLimit","upperBetaLimit","panningDistanceLimit","panningOriginTarget","panningInertia","zoomOnFactor","allowUpsideDown","panningAxis","_transformedDirection","mapPanning","collisionRadius","_previousPosition","_collisionVelocity","_newPosition","_computationVector","cosa","sina","cosb","sinb","_getTargetPosition","_collisionTriggered","_targetBoundingCenter","_storedAlpha","_storedBeta","_storedRadius","_storedTarget","_storedTargetScreenOffset","ignored","useCtrlForPanning","panningMouseButton","_reset","directionModifier","localDirection","_checkLimits","rebuildAnglesAndRadius","previousAlpha","alphaCorrectionTurns","toBoundingCenter","allowSamePosition","cloneAlphaBetaRadius","overrideCloneAlphaBetaRadius","newTarget","zoomOn","doNotUpdateMaxZ","focusOn","meshesOrMinMaxVectorAndDistance","alphaShift","rigCam","Ray","intersectsBoxMinMax","intersectionTreshold","newMinimum","newMaximum","maxValue","rr","vertex0","edge1","edge2","pvec","tvec","qvec","invdet","bw","result1","result2","intersectsAxis","tm","_tmpRay","intersectsMeshes","_comparePickingInfo","pickingInfoA","pickingInfoB","sega","segb","threshold","rsegb","_Rayl","D","sN","tN","sD","tD","_Smallnum","sc","tc","qtc","qsc","dP","_RayDistant","unprojectRayToRef","nearScreenSource","farScreenSource","nearVec3","farVec3","_internalPickForMesh","rayFunction","_internalPick","computeWorldMatrixForCamera","currentCamera","forceCompute","tmpMatrix","thinMatrices","thinInstanceGetWorldMatrices","_internalMultiPick","pickingInfos","_tempPickingRay","_pickWithRayInverseMatrix","_cachedRayForTransform","SH3ylmBasisConstants","SH3ylmBasisTrigonometricTerms","applySH3","lm","SHCosKernelConvolution","SphericalHarmonics","preScaled","l00","l1_1","l10","l11","l2_2","l2_1","l20","l21","l22","deltaSolidAngle","colorVector","convertIncidentRadianceToIrradiance","convertIrradianceToLambertianRadiance","preScaleForRendering","updateFromArray","updateFromFloatsArray","FromFloatsToRef","polynomial","SphericalPolynomial","preScaledHarmonics","_harmonics","FromPolynomial","addAmbient","updateFromHarmonics","harmonics","sp","ray1","getCameraPositionWithOffset","toRadians","roundVectorToPlaces","significance","roundToPlaces","rotationToVector","toVector","fromVector","rotationToVectorToRef","transformVectorWithQuat","applyTransformToSpace","worldTransform","inSpace","inSpaceInv","spaceTransform","FromQuat","computeLocalTransform","toVec3","pivotTransform","scalingTransform","Scaling","rotationTransform","translationTransform","_toVector","_fromVector","rotAxis","multiplyScalar","scalar","computeNormal","applyMatrix4","thisVect","getRotationBtwVectors","addArrays","arrayA","arrayB","AxisType","KbVector_1","KbQuaternion_1","MeshNode_1","PrimitiveMeshNode_1","CustomMeshNode_1","KbColor_1","KbGeometry_1","basicallyEqual","isBoxMesh","isConeMesh","isCylinderMesh","isDiscMesh","isMeshGroupNode","isCustomMeshNode","isSketchNode","isTextNode","isSketchControlPoint","isMeshNode","isPrimitiveMeshNode","isSpaceNode","isPlaneMesh","newX","newY","newZ","crossProduct","dotProduct","shiftClone","toObject","toEuler","getAngle","_getRadAngle","getAxis","rad","nAxis","dV","_showPivot","getRotationQuaternion","FromYawPitchRoll","FromRotationAxis","setRotationQuaternion","features","connectors","_connectors","copyPropertiesTo","_primeMover","preTransform","getFeatureByName","_polygonCount","_verticesCount","getSpaceChildren","uvOverrides","_loading","hasGeometry","geometryLoaded","cssString","toIntObject","MAX_INT16","floatToIntStore","float","maxInt","maxVal","trunc","SketchNode_1","eContainerItemLayout","eContainerChildAlignment","eContainerItemAlignment","eContainerItemJustify","eScroll","eRelativeSize","ePadding","eColorShade","eBackgroundColor","eElevation","eRuleType","eToolbarPosition","eValidationTiming","eViewerMode","ePosition","eIconStyle","eExportType","eHotspotSource","eHotspotAttach","eHotspotPosition","eHotspotShape","eAnimationEasing","eQueryMode","eOptionFilterSource","eOutputType","eOutputFieldType","eSingleOptionBehavior","eConfiguratorImageSource","eTextColor","eConfiguratorEntityType","eBackground","eSideOrientation","eUvMode","eMaterialViewMode","eMaterialLocation","eAlphaMode","eViewpointType","eViewpointProjection","eViewpointTargetMode","eConnectorPositionMode","eDragMode","eConnectionMode","eEasing","eCylinderCap","eJoinType","eShadowQuality","eShadowResolution","eShadowFilter","eMaterialWorkflow","eFeatureCenter","eJobStatus","eJobType","eEnvironment","eImportType","eQueueMessageType","eMateType","eValidationType","eValidationSource","eTextureChannel","eTextureHorizontalAlign","eTextureVerticalAlign","eTextureFit","eTextureBlendMode","eTextureWrapAddressMode","eAnnotationLineStyle","eDimensionLineCapStyle","eDimensionAxis","eLoadingExperience","eReflectionType","eRenderMode","eTextureSamplingMode","eAxis","eSketchPathType","eFilterControl","eFilterSource","eFilterType","eFieldType","isSphereMesh","isTorusMesh","shared","extractFromBinary","fileData","meshNode","curPos","indicesDataSize","indices16","posOffset","posDataSize","posRatio","norOffset","norDataSize","norRatio","normalsInt","uvsOffset","uvsDataSize","uvsRatio","uvsInt","toBinary","fileLen","indicesBits","indicesLength","posBits","posLength","normalsOffset","normalsBits","normalsLength","uvsBits","uvsLength","inArray","canFitInUInt16Bit","setInt32","setUint16","setUint32","setFloat32","setInt16","_boundingBoxMin","_boundingBoxMax","_boundingBoxChanges","_controlPoints","_lineSegmentStartingPoints","_sketchPoints","paths","getBoundingBox","centerBB","getSketchLength","sketchLength","lastPosition","currentPos","controlPoints","vertexGroups","_previewParticleSize","vertexIndices","_vertexIndices","eHandler","_kb3dEnums","absolute","getSketch","worldPosition","TagArrayMap","_db","deleteByKeyValue","removeWhere","deleteEntryIfEmpty","getByKey","all","ArrayMap","deleteByTag","nodeResult","handlers","deleteByKey","removeValue","flat","SetMap","HandlerMap","idNameOrNode","getNodesByIdOrName","fontFilesLoading","stylesheetsLoading","fontFamilySrcs","httpService","assetRoot","readyFonts","loadFont","stylesheetUrl","fontFamilies","fontFaceSet","fonts","loadStylesheet","loadFontFamily","loadFontByFile","absPath","getAbsoluteFontPath","loadPromise","getPublicStatic","loadFontData","getFontData","relativePath","stylesheetLoad","fontFaces","matchAll","face","fontFamilyMatch","fontFamilyName","srcMatch","fontSrc","loadFaces","loadPromises","resMap","buffers","resultData","resultFonts","newFontFace","FontFace","eContentType","ABORTED","trackedAssetCalls","deferredAssetCalls","httpCallsToAbort","deferredExecuted","addCacheParams","getUrlWithParams","getEnvironmentCacheParams","trackAssetCall","addParams","fetchWrapper","fetch","credentials","arrayBuffer","json","post","put","getBinary","appendCacheParameters","abortSubject","xhr","abortSub","oEvent","getAllResponseHeaders","onabort","defer","priority","executeDeferredCalls","trackedPromise","finally","lastDeploy","cacheBuster","urlParams","URLSearchParams","prefix","headers","Headers","requestInit","append","stringify","formData","ok","getRotatedDimensions","rAngle","scaleImagePreserveAspect","targetWidth","targetHeight","naturalWidth","naturalHeight","drawWidth","drawHeight","imageWidth","imageHeight","getBinClampedSize","rounding","KBMAXPREFIX","LOCAL_ENVIRONMENT_DOMAIN","slowWarningThrown","attachLoadingTimer","fileSize","endTime","LIBRARY_PATH","textureCache","materialCache","TEXTURE_SIZES","LIBRARY_TEXTURE_SIZES","LIBRARY_ENV_SIZES","LIBRARY_MOBILE_ENV_SIZES","forceFullSizeTextures","materialUpdateChannel","abortSubjects","textureDetailLevel","screen","getMaterialLibraryCategories","materialDefinitionSet","getMaterialLibraryOptions","category","LibraryMaterialsByCategory","guid","LibraryMaterials","updateChannel","combineLatest","updatedMaterial","getTexturePath","detailLevel","getLibraryTexturePath","getAbsoluteTexturePath","getTextureImage","initialUrl","bitmapObs","processImageBlob","opts","resizeWidth","resizeHeight","resizeQuality","bitmap","activeLinkAbort","abortTextureLoad","cacheParamUrl","blobResult","initialCacheParamUrl","loadStartTime","initialBlobResult","loadTime","blobToConvert","imageElement","blobUrl","scaled","getAbsoluteEnvironmentPath","searchMaterialByName","loadArchivedMaterials","archivedFilter","searchArgs","sortField","descending","getMaterialNameList","fields","setMaterialCache","LibraryEnvironments","LibraryEnvironmentsMap","makeLibraryMaterial","indexStr","iname","baseUrl","library","ambientOcclusion","albedo","asphalt","omit","bark","bricks","chipboard","concrete","corrugatedsteel","diamondplate","fabric","gravel","ground","ice","leather","marble","metal","metalplates","paint","paintedmetal","paintedplaster","pavingstones","planks","plastic","rock","rocks","roofingtiles","rust","scratches","terrazzo","tiles","wood","woodfloor","woodsiding","matData","categoryList","fillLibraryMaterialsMap","env","previewUrl","geometryCache","hydrateGeometry","isLightNode","_forceGrid","_lights","viewpoints","_viewpoints","_scenes","mates","_mates","annotations","_annotations","importHistory","_importHistory","exportHistory","_exportHistory","isNested","runOnAllScenes","nested","getNestedPathTo","isPath","nestedId","getAllConnectors","getByToken","getAllSketchControlPoints","SET_TYPE","dynamicDb","getByDynamicId","dynamicId","throwOnNotFound","ImportOperation","Kb3dManager","isDevEnvironment","_tokenDb","_nodeDb","_nameSearchCache","_nestedSceneDb","clickHandlers","doubleClickHandlers","draggables","hotspotClickHandlers","boundingboxChangeHandlers","services","getService","serviceToken","runOnNodes","idOrName","useCache","getById","getNodesByName","nestedScene","subId","getNodeByIdOrName","getAll","getAllNodes","getFirstNodeByName","re","getWildcardRegex","getNodesContainingName","getByTokens","escapeRegex","rule","includeChildNodes","constructorParameters","hydratedParameters","service","initializeModel","afterCreate","isRoot","omitDefaultValues","defaultValues","shallowEquals","sourceArray","sourceItem","targetItem","onSerializing","deserializeJson","objectToClone","issueNewIds","nestScene","serializedNested","nestedMan","getNodeNamePathMap","matMap","lightMap","allNodes","previewImport","imported","nMap","processNode","parentNamePath","namePath","matchingNode","childProp","childArr","cNode","pathToImportParent","import","previews","parentNode","thisNode","originalId","matchingMeshNode","originalMaterialId","parentPath","setModelDefaultValues","copyingToDifferentNode","sourceProp","aProp","bProp","isTextureLayer","_init","local","_needsImageUpdate","svgXml","getC1","getC2","isActive","validate","hasAny","assert","topologicalSort","nameSelector","visited","sorted","cycles","recurse","edges","orbit","perspective","_type","newType","firstPerson","ThinTexture","_wrapU","_wrapV","_texture","_shareDepth","_cachedSize","_cachedBaseSize","_initialSamplingMode","_IsRenderTargetWrapper","getBaseSize","updateSamplingMode","releaseInternalTexture","BaseTexture","_hasAlpha","getAlphaFromRGB","_getAlphaFromRGB","_coordinatesIndex","_coordinatesMode","_isCube","gammaSpace","isRGBD","lodGenerationOffset","lodGenerationScale","linearSpecularLOD","irradianceTexture","isBlocking","loadingError","_loadingError","errorObject","_errorObject","sceneOrEngine","DEFAULT_ANISOTROPIC_FILTERING_LEVEL","invertZ","lodLevelInAlpha","_prefiltered","_forceSerialize","_IsScene","_getEngine","checkTransformsAreIdentical","getReflectionTextureMatrix","getRefractionTextureMatrix","isReadyOrNotBlocking","canRescale","_getFromCache","sampling","correctedUseSRGBBuffer","texturesCache","texturesCacheEntry","maxWidth","_readPixelsSync","allowEmptyName","numRemaining","onLoadObservable","GenerateBase64StringFromPixelData","npixels","ctx","createImageData","putImageData","canvas2","ctx2","Texture","_noMipmap","_mimeType","_isBlocking","_invertY","noMipmapOrOptions","TRILINEAR_SAMPLINGMODE","uAng","vAng","wAng","uRotationCenter","vRotationCenter","wRotationCenter","homogeneousRotationInUVTransform","inspectableCustomProperties","_rowGenerationMatrix","_cachedTextureMatrix","_projectionModeMatrix","_t0","_t1","_t2","_cachedUOffset","_cachedVOffset","_cachedUScale","_cachedVScale","_cachedUAng","_cachedVAng","_cachedWAng","_cachedReflectionProjectionMatrixId","_cachedURotationCenter","_cachedVRotationCenter","_cachedWRotationCenter","_cachedHomogeneousRotationInUVTransform","_cachedReflectionTextureMatrix","_cachedReflectionUOffset","_cachedReflectionVOffset","_cachedReflectionUScale","_cachedReflectionVScale","_cachedReflectionCoordinatesMode","_format","_delayedOnLoad","_delayedOnError","_loaderOptions","_creationFlags","_forcedExtension","errorHandler","OnTextureLoadErrorObservable","loadObserver","useDelayedTextureLoading","updateURL","_prepareRowForTextureGeneration","uBase","PROJECTION_MODE","flagMaterialsAsTextureDirty","PLANAR_MODE","projectionMatrix","savedName","SerializeBuffers","_SerializeInternalTextureUniqueId","ForceSerializeBuffers","base64String","GenerateBase64StringFromTexture","GenerateBase64StringFromTextureAsync","internalTextureUniqueId","parsedTexture","parsedCustomTexture","_samplingMode","_CubeTextureParser","hasInternalTextureUniqueId","mirrorPlane","mirrorTexture","_CreateMirror","renderTargetSize","_waitingRenderList","renderTargetTexture","reflectionProbes","probe","cubeTexture","_CreateRenderTargetTexture","CreateFromBase64String","UseSerializedUrlIfAny","jsonTexture","NEAREST_SAMPLINGMODE","NEAREST_NEAREST_MIPLINEAR","BILINEAR_SAMPLINGMODE","LINEAR_LINEAR_MIPNEAREST","LINEAR_LINEAR_MIPLINEAR","NEAREST_NEAREST_MIPNEAREST","NEAREST_LINEAR_MIPNEAREST","NEAREST_LINEAR_MIPLINEAR","NEAREST_LINEAR","NEAREST_NEAREST","LINEAR_NEAREST_MIPNEAREST","LINEAR_NEAREST_MIPLINEAR","LINEAR_LINEAR","LINEAR_NEAREST","EXPLICIT_MODE","SPHERICAL_MODE","CUBIC_MODE","SKYBOX_MODE","INVCUBIC_MODE","EQUIRECTANGULAR_MODE","FIXED_EQUIRECTANGULAR_MODE","FIXED_EQUIRECTANGULAR_MIRRORED_MODE","CLAMP_ADDRESSMODE","WRAP_ADDRESSMODE","MIRROR_ADDRESSMODE","PrePassConfiguration","previousWorldMatrices","currentRTisSceneRT","previousViewProjection","currentViewProjection","_lastUpdateFrameId","PushMaterial","storeEffectOnSubMeshes","_normalMatrix","_activeEffect","_isReadyForSubMesh","bindOnlyNormalMatrix","normalMatrix","_mustRebind","MaterialFlags","DiffuseTextureEnabled","_DiffuseTextureEnabled","MarkAllMaterialsAsDirty","DetailTextureEnabled","_DetailTextureEnabled","DecalMapEnabled","_DecalMapEnabled","AmbientTextureEnabled","_AmbientTextureEnabled","OpacityTextureEnabled","_OpacityTextureEnabled","ReflectionTextureEnabled","_ReflectionTextureEnabled","EmissiveTextureEnabled","_EmissiveTextureEnabled","SpecularTextureEnabled","_SpecularTextureEnabled","BumpTextureEnabled","_BumpTextureEnabled","LightmapTextureEnabled","_LightmapTextureEnabled","RefractionTextureEnabled","_RefractionTextureEnabled","ColorGradingTextureEnabled","_ColorGradingTextureEnabled","FresnelEnabled","_FresnelEnabled","ClearCoatTextureEnabled","_ClearCoatTextureEnabled","ClearCoatBumpTextureEnabled","_ClearCoatBumpTextureEnabled","ClearCoatTintTextureEnabled","_ClearCoatTintTextureEnabled","SheenTextureEnabled","_SheenTextureEnabled","AnisotropicTextureEnabled","_AnisotropicTextureEnabled","ThicknessTextureEnabled","_ThicknessTextureEnabled","RefractionIntensityTextureEnabled","_RefractionIntensityTextureEnabled","TranslucencyIntensityTextureEnabled","_TranslucencyIntensityTextureEnabled","IridescenceTextureEnabled","_IridescenceTextureEnabled","EffectFallbacks","_defines","_currentRank","_maxRank","currentDefines","currentFallbacks","rxOption","MaterialPluginManager","_activePlugins","_activePluginsForExtraEvents","_addPlugin","_MaterialPluginClassToMainDefine","_MaterialPluginCounter","_handlePluginEvent","_codeInjectionPoints","defineNamesFromPlugins","collectDefines","_collectPointNames","getCustomCode","_defineNamesFromPlugins","_activatePlugin","_handlePluginEventIsReadyForSubMesh","_handlePluginEventPrepareDefinesBeforeAttributes","_handlePluginEventPrepareDefines","_handlePluginEventBindForSubMesh","registerForExtraEvents","_handlePluginEventHasRenderTargetTextures","_handlePluginEventFillRenderTargetTextures","_handlePluginEventHardBindForSubMesh","prepareDefinesBeforeAttributes","hardBindForSubMesh","fillRenderTargetTextures","renderTargets","GetDefineNames","defineNames","PrepareEffect","fallbackRank","addFallbacks","_uniformList","_uboList","customCode","_injectCustomCode","_uboDeclaration","_vertexDeclaration","_fragmentDeclaration","shaderType","pointName","existingCallback","injectedCode","resolveIncludes","regexFlags","matchOption","newCode","fullPointName","MaterialPluginBase","_enable","_pluginManager","addToPluginList","_pluginDefineNames","markAllDefinesAsDirty","currentRank","ubos","MaterialDetailMapDefines","DETAIL","DETAILDIRECTUV","DETAIL_NORMALBLENDMETHOD","DetailMapConfiguration","_internalMarkAllSubMeshesAsTexturesDirty","diffuseBlendLevel","roughnessBlendLevel","bumpLevel","_normalBlendMethod","PrepareDefinesForMergedUV","BindTextureMatrix","onCreatedEffectParameters","StandardMaterialDefines","MAINUV1","MAINUV2","MAINUV3","MAINUV4","MAINUV5","MAINUV6","DIFFUSE","DIFFUSEDIRECTUV","BAKED_VERTEX_ANIMATION_TEXTURE","AMBIENT","AMBIENTDIRECTUV","OPACITY","OPACITYDIRECTUV","OPACITYRGB","REFLECTION","EMISSIVE","EMISSIVEDIRECTUV","SPECULAR","SPECULARDIRECTUV","BUMP","BUMPDIRECTUV","PARALLAX","PARALLAXOCCLUSION","SPECULAROVERALPHA","CLIPPLANE","CLIPPLANE2","CLIPPLANE3","CLIPPLANE4","CLIPPLANE5","CLIPPLANE6","ALPHATEST","DEPTHPREPASS","ALPHAFROMDIFFUSE","POINTSIZE","FOG","SPECULARTERM","DIFFUSEFRESNEL","OPACITYFRESNEL","REFLECTIONFRESNEL","REFRACTIONFRESNEL","EMISSIVEFRESNEL","FRESNEL","NORMAL","TANGENT","UV1","UV2","UV3","UV4","UV5","UV6","VERTEXCOLOR","VERTEXALPHA","NUM_BONE_INFLUENCERS","BonesPerMesh","BONETEXTURE","BONES_VELOCITY_ENABLED","INSTANCES","THIN_INSTANCES","GLOSSINESS","ROUGHNESS","EMISSIVEASILLUMINATION","LINKEMISSIVEWITHDIFFUSE","REFLECTIONFRESNELFROMSPECULAR","LIGHTMAP","LIGHTMAPDIRECTUV","OBJECTSPACE_NORMALMAP","USELIGHTMAPASSHADOWMAP","REFLECTIONMAP_3D","REFLECTIONMAP_SPHERICAL","REFLECTIONMAP_PLANAR","REFLECTIONMAP_CUBIC","USE_LOCAL_REFLECTIONMAP_CUBIC","USE_LOCAL_REFRACTIONMAP_CUBIC","REFLECTIONMAP_PROJECTION","REFLECTIONMAP_SKYBOX","REFLECTIONMAP_EXPLICIT","REFLECTIONMAP_EQUIRECTANGULAR","REFLECTIONMAP_EQUIRECTANGULAR_FIXED","REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED","REFLECTIONMAP_OPPOSITEZ","INVERTCUBICMAP","LOGARITHMICDEPTH","REFRACTION","REFRACTIONMAP_3D","REFLECTIONOVERALPHA","TWOSIDEDLIGHTING","SHADOWFLOAT","MORPHTARGETS","MORPHTARGETS_NORMAL","MORPHTARGETS_TANGENT","MORPHTARGETS_UV","MORPHTARGETS_TEXTURE","NONUNIFORMSCALING","PREMULTIPLYALPHA","ALPHATEST_AFTERALLALPHACOMPUTATIONS","ALPHABLEND","PREPASS_IRRADIANCE","PREPASS_IRRADIANCE_INDEX","PREPASS_ALBEDO_SQRT","PREPASS_ALBEDO_SQRT_INDEX","PREPASS_DEPTH","PREPASS_DEPTH_INDEX","PREPASS_NORMAL","PREPASS_NORMAL_INDEX","PREPASS_POSITION","PREPASS_POSITION_INDEX","PREPASS_VELOCITY","PREPASS_VELOCITY_INDEX","PREPASS_REFLECTIVITY","PREPASS_REFLECTIVITY_INDEX","RGBDLIGHTMAP","RGBDREFLECTION","RGBDREFRACTION","CAMERA_ORTHOGRAPHIC","CAMERA_PERSPECTIVE","IS_REFLECTION_LINEAR","IS_REFRACTION_LINEAR","DECAL_AFTER_DETAIL","setReflectionMode","modeToEnable","modes","StandardMaterial","_attachImageProcessingConfiguration","configuration","_imageProcessingObserver","cameraColorCurvesEnabled","cameraColorGradingEnabled","cameraToneMappingEnabled","cameraExposure","cameraContrast","cameraColorGradingTexture","cameraColorCurves","_diffuseTexture","_ambientTexture","_opacityTexture","_reflectionTexture","_emissiveTexture","_specularTexture","_bumpTexture","_lightmapTexture","_refractionTexture","diffuseColor","specularPower","_useAlphaFromDiffuseTexture","_useEmissiveAsIllumination","_linkEmissiveWithDiffuse","_useSpecularOverAlpha","_useReflectionOverAlpha","_disableLighting","_useObjectSpaceNormalMap","_useParallax","_useParallaxOcclusion","parallaxScaleBias","_roughness","indexOfRefraction","invertRefractionY","alphaCutOff","_useLightmapAsShadowmap","_useReflectionFresnelFromSpecular","_useGlossinessFromSpecularMapAlpha","_maxSimultaneousLights","_invertNormalMapX","_invertNormalMapY","_twoSidedLighting","_applyDecalMapAfterDetailMap","_worldViewProjectionMatrix","_globalAmbientColor","_cacheHasRenderTargetTextures","detailMap","_useLogarithmicDepth","_shouldUseAlphaFromDiffuseTexture","_opacityFresnelParameters","_hasAlphaChannel","PrepareDefinesForLights","PrepareDefinesForMultiview","oit","PrepareDefinesForPrePass","PrepareDefinesForOIT","boundingBoxSize","reflectionTexture","refractionTexture","_diffuseFresnelParameters","_emissiveFresnelParameters","_refractionFresnelParameters","_reflectionFresnelParameters","PrepareDefinesForMisc","PrepareDefinesForFrameBoundValues","PrepareDefinesForAttributes","forceWasNotReadyPreviously","lightDisposed","HandleFallbacksForShadows","PrepareAttributesForBones","PrepareAttributesForInstances","PrepareAttributesForBakedVertexAnimation","shaderName","maxSimultaneousMorphTargets","AddUniforms","AddSamplers","PrepareSamplers","PrepareUniformsAndSamplersList","csnrOptions","customShaderNameResolve","previousEffect","mustRebind","BindBonesParameters","diffuseFresnelParameters","leftColor","power","rightColor","opacityFresnelParameters","reflectionFresnelParameters","refractionFresnelParameters","emissiveFresnelParameters","boundingBoxPosition","BindLights","BindFogParameters","BindMorphTargetParameters","BindLogDepth","WebGLRenderTargetWrapper","RenderTargetWrapper","depthStencilTextureWithStencil","_isMulti","faceIndices","_faceIndices","layerIndices","_layerIndices","setSamples","initializeBuffers","updateMultipleRenderTargetTextureSampleCount","updateRenderTargetTextureSampleCount","_attachments","_generateStencilBuffer","_generateDepthBuffer","setTextures","disposePrevious","setLayerAndFaceIndices","faces","setLayerAndFaceIndex","createDepthStencilTexture","depthTextureFormat","_cloneRenderTargetWrapper","rtw","textureArray","generateDepthTexture","textureCount","lastTextureSource","Depth","DepthStencil","samplingModes","types","formats","targetTypes","layerIndex","layerCounts","internalTexture2Index","optionsMRT","createMultipleRenderTarget","createRenderTargetCubeTexture","_swapRenderTargetWrapper","bilinear","releaseTextures","disposeOnlyFramebuffers","_depthStencilBuffer","_colorTextureArray","_depthStencilTextureArray","_disposeOnlyFramebuffers","createMultiviewRenderTargetTexture","depthbuffer","_bindTextureRenderTarget","attachmentIndex","faceIndexOrLayer","currentFB","_createHardwareRenderTargetWrapper","colorAttachment","noColorAttachment","RenderTarget","currentFrameBuffer","_createDepthStencilCubeTexture","_createDepthStencilTexture","internalOptions","hasStencil","DEPTH_COMPONENT","DEPTH_COMPONENT24","DEPTH32F_STENCIL8","glRtWrapper","colorRenderbuffer","PostProcess","postProcessName","customShaderCodeProcessing","_CustomShaderCodeProcessing","_fragmentUrl","onActivate","_onActivateObserver","onActivateObservable","onSizeChanged","_onSizeChangedObserver","onSizeChangedObservable","_onApplyObserver","onApplyObservable","onBeforeRender","onAfterRender","inputTexture","restoreDefaultInputTexture","getCamera","_camera","texelSize","_shareOutputWithPostProcess","_texelSize","fragmentUrl","reusable","vertexUrl","blockCompilation","nodeMaterialSource","forceAutoClearInAlphaMode","enablePixelPerfectMode","scaleMode","alwaysForcePOT","adaptScaleToCurrentViewport","_reusable","_textureCache","_scaleRatio","_options","renderTargetSamplingMode","_textureType","_textureFormat","_vertexUrl","_parameters","updateEffect","shareOutputWith","_disposeTextures","useOwnOutput","_GetShaderCodeProcessing","defineCustomBindings","newUniforms","newSamplers","_postProcessDefines","_createRenderTargetTexture","textureOptions","postProcessChannel","lastUsedRenderId","_flushTextureCache","currentlyUsed","_resize","needMipMaps","forceDepthStencil","firstPP","maxSize","webVRCamera","desiredWidth","desiredHeight","alphaConstants","bindCustomBindings","_disposeTextureCache","_prePassEffectConfiguration","addEffectConfiguration","cameraId","parsedPostProcess","postProcessType","_Parse","targetCamera","defaultOptions","EffectRenderer","_fullscreenViewport","_onContextRestoredObserver","applyEffectWrapper","effectWrapper","saveStates","_savedStateDepthTest","_savedStateStencilTest","restoreStates","_isRenderTargetTexture","outputTexture","EffectWrapper","creationOptions","effectCreationOptions","uniformNames","useShaderStore","attributeNames","samplerNames","passPixelShader","DumpTools","_DumpToolsEngine","DumpData","_CreateDumpRenderer","data2","fileReader","EncodeScreenshotCanvasData","DumpDataAsync","DumpFramebuffer","RenderTargetTexture","_renderList","_unObserveRenderList","_renderListHasChanged","_prePassEnabled","_prePassRenderTarget","onAfterUnbind","_onAfterUnbindObserver","onAfterUnbindObservable","onClear","_onClearObserver","_renderPassIds","currentRefreshId","_currentRefreshId","setMaterialForRendering","_renderTarget","renderTargetOptions","_renderTargetOptions","_onRatioRescale","_sizeRatio","_initialSizeParameter","_boundingBoxSize","doNotChangeAspectRatio","_functionName","newLength","forceLayerMaskCheck","ignoreCameraViewport","_refreshRate","_canRescale","_isCubeData","_processSizeParameter","_resizeObserver","_generateMipMaps","_doNotChangeAspectRatio","getRenderSize","_textureMatrix","_releaseRenderPassId","_createRenderPassId","numPasses","getRenderLayers","createRenderPassIds","_bestReflectionRenderTargetDimension","resetRefreshCounter","refreshRate","addPostProcess","_postProcessManager","clearPostProcesses","removePostProcess","disableRescaling","newSize","wasCube","useCameraPostProcess","dumpForDebug","_render","checkReadiness","useCameraPostProcesses","renderListPredicate","sceneMeshes","sceneCamera","_defaultRenderListPrepared","numLayers","currentRenderList","defaultRenderList","defaultRenderListLength","getCustomRenderList","customIsReadyFunction","_renderToTarget","renderDimension","curved","_prepareRenderingManager","currentRenderListLength","checkLayerMask","isMasked","_unbindFrameBuffer","_debugPushGroup","saveGenerateMipMaps","_debugPopGroup","disposeFramebufferObjects","REFRESHRATE_RENDER_ONCE","REFRESHRATE_RENDER_ONEVERYFRAME","REFRESHRATE_RENDER_ONEVERYTWOFRAMES","PassPostProcess","ApplyPostProcess","encodedTexture","floatView","int32View","ToHalfFloat","bits","FromHalfFloat","TextureTools","CreateResizedCopy","useBilinearMode","passPostProcess","RGBDTextureTools","expandTexture","expandRGBDTexture","rgbdPostProcess","expandedTexture","outputTextureType","_instanceNumber","GetEnvironmentBRDFTexture","environmentBRDFTexture","previousState","ExpandRGBDTexture","MaterialBRDFDefines","BRDF_V_HEIGHT_CORRELATED","MS_BRDF_ENERGY_CONSERVATION","SPHERICAL_HARMONICS","SPECULAR_GLOSSINESS_ENERGY_CONSERVATION","PBRBRDFConfiguration","_internalMarkAllSubMeshesAsMiscDirty","_useEnergyConservation","DEFAULT_USE_ENERGY_CONSERVATION","useEnergyConservation","_useSmithVisibilityHeightCorrelated","DEFAULT_USE_SMITH_VISIBILITY_HEIGHT_CORRELATED","useSmithVisibilityHeightCorrelated","_useSphericalHarmonics","DEFAULT_USE_SPHERICAL_HARMONICS","useSphericalHarmonics","_useSpecularGlossinessInputEnergyConservation","DEFAULT_USE_SPECULAR_GLOSSINESS_INPUT_ENERGY_CONSERVATION","useSpecularGlossinessInputEnergyConservation","FileFaceOrientation","worldAxisForNormal","worldAxisForFileX","worldAxisForFileY","CubeMapToSphericalPolynomialTools","rightPromise","leftPromise","upPromise","downPromise","frontPromise","backPromise","down","front","cubeInfo","ConvertCubeMapToSphericalPolynomial","sphericalHarmonics","totalSolidAngle","du","halfTexel","minUV","fileFace","_FileFaces","dataArray","worldDirection","_AreaElement","MAX_HDRI_VALUE","PRESERVE_CLAMPED_COLORS","currentMax","correctionFactor","FromHarmonics","forceSphericalPolynomialsRecompute","ConvertCubeMapTextureToSphericalPolynomial","sphericalPolynomial","MaterialClearCoatDefines","CLEARCOAT","CLEARCOAT_DEFAULTIOR","CLEARCOAT_TEXTURE","CLEARCOAT_TEXTURE_ROUGHNESS","CLEARCOAT_TEXTUREDIRECTUV","CLEARCOAT_TEXTURE_ROUGHNESSDIRECTUV","CLEARCOAT_BUMP","CLEARCOAT_BUMPDIRECTUV","CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE","CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL","CLEARCOAT_REMAP_F0","CLEARCOAT_TINT","CLEARCOAT_TINT_TEXTURE","CLEARCOAT_TINT_TEXTUREDIRECTUV","CLEARCOAT_TINT_GAMMATEXTURE","PBRClearCoatConfiguration","_indexOfRefraction","_DefaultIndexOfRefraction","_useRoughnessFromMainTexture","useRoughnessFromMainTexture","_textureRoughness","textureRoughness","_remapF0OnInterfaceChange","remapF0OnInterfaceChange","bumpTexture","_isTintEnabled","isTintEnabled","tintColor","White","tintColorAtDistance","tintThickness","_tintTexture","tintTexture","disableBumpMap","_disableBumpMap","invertNormalMapX","invertNormalMapY","identicalTextures","f0","eta","MaterialIridescenceDefines","IRIDESCENCE","IRIDESCENCE_TEXTURE","IRIDESCENCE_TEXTUREDIRECTUV","IRIDESCENCE_THICKNESS_TEXTURE","IRIDESCENCE_THICKNESS_TEXTUREDIRECTUV","IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE","PBRIridescenceConfiguration","minimumThickness","_DefaultMinimumThickness","maximumThickness","_DefaultMaximumThickness","_thicknessTexture","thicknessTexture","MaterialAnisotropicDefines","ANISOTROPIC","ANISOTROPIC_TEXTURE","ANISOTROPIC_TEXTUREDIRECTUV","ANISOTROPIC_LEGACY","PBRAnisotropicConfiguration","_legacy","legacy","MaterialSheenDefines","SHEEN","SHEEN_TEXTURE","SHEEN_GAMMATEXTURE","SHEEN_TEXTURE_ROUGHNESS","SHEEN_TEXTUREDIRECTUV","SHEEN_TEXTURE_ROUGHNESSDIRECTUV","SHEEN_LINKWITHALBEDO","SHEEN_ROUGHNESS","SHEEN_ALBEDOSCALING","SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE","SHEEN_TEXTURE_ROUGHNESS_IDENTICAL","PBRSheenConfiguration","_linkSheenWithAlbedo","linkSheenWithAlbedo","_albedoScaling","albedoScaling","MaterialSubSurfaceDefines","SUBSURFACE","SS_REFRACTION","SS_REFRACTION_USE_INTENSITY_FROM_TEXTURE","SS_TRANSLUCENCY","SS_TRANSLUCENCY_USE_INTENSITY_FROM_TEXTURE","SS_SCATTERING","SS_THICKNESSANDMASK_TEXTURE","SS_THICKNESSANDMASK_TEXTUREDIRECTUV","SS_HAS_THICKNESS","SS_REFRACTIONINTENSITY_TEXTURE","SS_REFRACTIONINTENSITY_TEXTUREDIRECTUV","SS_TRANSLUCENCYINTENSITY_TEXTURE","SS_TRANSLUCENCYINTENSITY_TEXTUREDIRECTUV","SS_REFRACTIONMAP_3D","SS_REFRACTIONMAP_OPPOSITEZ","SS_LODINREFRACTIONALPHA","SS_GAMMAREFRACTION","SS_RGBDREFRACTION","SS_LINEARSPECULARREFRACTION","SS_LINKREFRACTIONTOTRANSPARENCY","SS_ALBEDOFORREFRACTIONTINT","SS_ALBEDOFORTRANSLUCENCYTINT","SS_USE_LOCAL_REFRACTIONMAP_CUBIC","SS_USE_THICKNESS_AS_DEPTH","SS_MASK_FROM_THICKNESS_TEXTURE","SS_USE_GLTF_TEXTURES","PBRSubSurfaceConfiguration","scatteringDiffusionProfile","subSurfaceConfiguration","ssDiffusionProfileColors","_scatteringDiffusionProfileIndex","enableSubSurfaceForPrePass","addDiffusionProfile","volumeIndexOfRefraction","_volumeIndexOfRefraction","_isRefractionEnabled","_isTranslucencyEnabled","_isScatteringEnabled","_internalMarkScenePrePassDirty","isRefractionEnabled","isTranslucencyEnabled","isScatteringEnabled","refractionIntensity","translucencyIntensity","useAlbedoToTintRefraction","useAlbedoToTintTranslucency","_invertRefractionY","_linkRefractionWithTransparency","linkRefractionWithTransparency","useThicknessAsDepth","diffusionDistance","_useMaskFromThicknessTexture","useMaskFromThicknessTexture","_refractionIntensityTexture","refractionIntensityTexture","_translucencyIntensityTexture","translucencyIntensityTexture","_useGltfStyleTextures","useGltfStyleTextures","_getRefractionTexture","refractionIntensityTextureIsThicknessTexture","translucencyIntensityTextureIsThicknessTexture","useOnlyThicknessTexture","thicknessScale","realTimeFiltering","lodBasedMicrosurface","LODBASEDMICROSFURACE","refractionIor","Log2","disableAlphaBlending","PBRMaterialDefines","PBR","NUM_SAMPLES","REALTIME_FILTERING","ALBEDO","GAMMAALBEDO","ALBEDODIRECTUV","AMBIENTINGRAYSCALE","ALPHAFROMALBEDO","ALPHATESTVALUE","RADIANCEOVERALPHA","ALPHAFRESNEL","LINEARALPHAFRESNEL","GAMMAEMISSIVE","REFLECTIVITY","REFLECTIVITY_GAMMA","REFLECTIVITYDIRECTUV","MICROSURFACEFROMREFLECTIVITYMAP","MICROSURFACEAUTOMATIC","MICROSURFACEMAP","MICROSURFACEMAPDIRECTUV","METALLICWORKFLOW","ROUGHNESSSTOREINMETALMAPALPHA","ROUGHNESSSTOREINMETALMAPGREEN","METALLNESSSTOREINMETALMAPBLUE","AOSTOREINMETALMAPRED","METALLIC_REFLECTANCE","METALLIC_REFLECTANCE_GAMMA","METALLIC_REFLECTANCEDIRECTUV","METALLIC_REFLECTANCE_USE_ALPHA_ONLY","REFLECTANCE","REFLECTANCE_GAMMA","REFLECTANCEDIRECTUV","ENVIRONMENTBRDF","ENVIRONMENTBRDF_RGBD","NORMALXYSCALE","GAMMALIGHTMAP","USESPHERICALFROMREFLECTIONMAP","USEIRRADIANCEMAP","USESPHERICALINVERTEX","LODINREFLECTIONALPHA","GAMMAREFLECTION","LINEARSPECULARREFLECTION","RADIANCEOCCLUSION","HORIZONOCCLUSION","USEPHYSICALLIGHTFALLOFF","USEGLTFLIGHTFALLOFF","FORCENORMALFORWARD","SPECULARAA","UNLIT","DEBUGMODE","PBRBaseMaterial","_realTimeFiltering","realTimeFilteringQuality","_realTimeFilteringQuality","_directIntensity","_emissiveIntensity","_environmentIntensity","_specularIntensity","_lightingInfos","_albedoTexture","_ambientTextureStrength","_ambientTextureImpactOnAnalyticalLights","DEFAULT_AO_ON_ANALYTICAL_LIGHTS","_reflectivityTexture","_metallicTexture","_metallic","_metallicF0Factor","_metallicReflectanceColor","_useOnlyMetallicFromMetallicReflectanceTexture","_metallicReflectanceTexture","_reflectanceTexture","_microSurfaceTexture","_ambientColor","_albedoColor","_reflectivityColor","_reflectionColor","_emissiveColor","_microSurface","_useHorizonOcclusion","_useRadianceOcclusion","_useAlphaFromAlbedoTexture","_useMicroSurfaceFromReflectivityMapAlpha","_useRoughnessFromMetallicTextureAlpha","_useRoughnessFromMetallicTextureGreen","_useMetallnessFromMetallicTextureBlue","_useAmbientOcclusionFromMetallicTextureRed","_useAmbientInGrayScale","_useAutoMicroSurfaceFromReflectivityMap","_lightFalloff","LIGHTFALLOFF_PHYSICAL","_useRadianceOverAlpha","_parallaxScaleBias","_alphaCutOff","_useAlphaFresnel","_useLinearAlphaFresnel","_environmentBRDFTexture","_forceIrradianceInFragment","_forceNormalForward","_enableSpecularAntiAliasing","_unlit","_debugMode","debugMode","debugLimit","debugFactor","brdf","clearCoat","iridescence","anisotropy","sheen","subSurface","PBRMATERIAL_OPAQUE","PBRMATERIAL_ALPHATEST","_shouldUseAlphaFromAlbedoTexture","_getReflectionTexture","isMetallicWorkflow","_prepareDefines","LIGHTFALLOFF_STANDARD","LIGHTFALLOFF_GLTF","polynomials","ior","outsideIOR","metallicF90","PBRMATERIAL_ALPHABLEND","PBRMATERIAL_ALPHATESTANDBLEND","PBRMaterial","usePhysicalLightFalloff","useGLTFLightFalloff","ambientTextureStrength","ambientTextureImpactOnAnalyticalLights","metallicF0Factor","metallicReflectanceColor","useOnlyMetallicFromMetallicReflectanceTexture","reflectivityColor","useLightmapAsShadowmap","useAlphaFromAlbedoTexture","forceAlphaTest","useMicroSurfaceFromReflectivityMapAlpha","useRoughnessFromMetallicTextureAlpha","useRoughnessFromMetallicTextureGreen","useMetallnessFromMetallicTextureBlue","useAmbientOcclusionFromMetallicTextureRed","useAmbientInGrayScale","useAutoMicroSurfaceFromReflectivityMap","useObjectSpaceNormalMap","useParallax","useParallaxOcclusion","forceIrradianceInFragment","twoSidedLighting","useAlphaFresnel","useLinearAlphaFresnel","forceNormalForward","enableSpecularAntiAliasing","useHorizonOcclusion","useRadianceOcclusion","unlit","applyDecalMapAfterDetailMap","Light","_range","_inverseSquaredRange","intensityMode","_intensityMode","_computePhotometricScale","_shadowEnabled","_markMeshesAsLightDirty","_includedOnlyMeshes","_hookArrayForIncludedOnly","_excludedMeshes","_hookArrayForExcluded","excludeWithLayerMask","_excludeWithLayerMask","_resyncMeshes","includeOnlyWithLayerMask","_includeOnlyWithLayerMask","_lightmapMode","diffuse","_photometricScale","_shadowGenerators","_excludedMeshesIds","_includedOnlyMeshesIds","_isLight","transferTexturesToEffect","iAsString","needUpdate","_lastUseSpecular","scaledIntensity","getScaledIntensity","bindShadowLight","getTypeID","clonedLight","excludedMeshesIds","includedOnlyMeshesIds","parsedLight","_getPhotometricScale","photometricScale","lightTypeID","photometricMode","apexAngleRadians","_reorderLightsInScene","_renderPriority","HemisphericLight","setDirectionToTarget","_effect","normalizeDirection","transferToNodeMaterialEffect","lightDataUniformName","BlurPostProcess","kernel","_idealKernel","_kernel","_nearestBestKernel","_blockCompilation","packedFloat","_packedFloat","varyingCount","depCount","_staticDefines","N","centerIndex","offsets","weights","totalWeight","_gaussianWeight","linearSamplingWeights","linearSamplingOffsets","linearSamplingMap","sharedCell","weightLinear","offsetLinear","maxVaryingRows","freeVaryingVec2","_glslFloat","idealKernel","sigma","decimalFigures","MirrorTexture","blurRatio","_blurRatio","_preparePostProcesses","adaptiveBlurKernel","_adaptiveBlurKernel","_autoComputeBlurKernel","blurKernel","blurKernelX","blurKernelY","_blurKernelX","_blurKernelY","dw","dh","_updateGammaSpace","_mirrorMatrix","_imageProcessingConfigChangeObserver","saveClipPlane","_sceneUBO","_currentSceneUBO","_blurX","_blurY","_partialLoadFile","loadedFiles","onfinish","onErrorCallBack","_internalCount","_cascadeLoadFiles","files","_cascadeLoadImgs","loadedImages","_partialLoadImg","tokenPendingData","_setCubeMapTextureParams","maxLevel","TEXTURE_MAX_LEVEL","createCubeTextureBase","createPolynomials","lodScale","lodOffset","beforeLoadCubeDataCallback","imageHandler","originalRootUrl","rootUrlWithoutUriParams","onloaddata","loadCubeData","supportCascades","images","imgs","TEXTURE_CUBE_MAP_POSITIVE_Y","TEXTURE_CUBE_MAP_POSITIVE_Z","TEXTURE_CUBE_MAP_NEGATIVE_X","TEXTURE_CUBE_MAP_NEGATIVE_Y","TEXTURE_CUBE_MAP_NEGATIVE_Z","CubeTexture","rotationY","_rotationY","setReflectionTextureMatrix","RotationY","rootUrlKey","extensions","prefiltered","_lodScale","_lodOffset","_extensions","_textureMatrixRefraction","_createPolynomials","isDDS","isEnv","isBasis","_loadTexture","trans","oldTexture","onLoadProcessing","newCubeTexture","BackgroundMaterialDefines","GAMMADIFFUSE","DIFFUSEHASALPHA","REFLECTIONBLUR","REFLECTIONFALLOFF","TEXTURELODSUPPORT","USERGBCOLOR","USEHIGHLIGHTANDSHADOWCOLORS","BACKMAT_SHADOWONLY","NOISE","REFLECTIONBGR","EQUIRECTANGULAR_RELFECTION_FOV","BackgroundMaterial","_perceptualColor","__perceptualColor","_computePrimaryColorFromPerceptualColor","primaryColorShadowLevel","_primaryColorShadowLevel","_computePrimaryColors","primaryColorHighlightLevel","_primaryColorHighlightLevel","reflectionStandardFresnelWeight","reflectionWeight","reflectionReflectance0","StandardReflectance0","reflectionReflectance90","StandardReflectance90","fovMultiplier","_fovMultiplier","primaryColor","reflectionBlur","diffuseTexture","_shadowLights","shadowLights","shadowLevel","sceneCenter","opacityFresnel","reflectionFresnel","reflectionFalloffDistance","reflectionAmount","useRGBColor","enableNoise","useEquirectangularFOV","_shadowOnly","shadowOnly","switchToBGR","_reflectionControls","_white","_primaryShadowColor","_primaryHighlightColor","_opacityFresnel","_reflectionBlur","_useRGBColor","_enableNoise","_primaryColor","CreatePlaneVertexData","_ComputeSides","CreatePlane","_GetDefaultSideOrientation","sourcePlane","PlaneBuilder","GroundMesh","generateOctree","_subdivisionsX","_subdivisionsY","subdivisionsX","subdivisionsY","optimize","chunksCount","octreeBlocksSize","thisAsAny","createOrUpdateSubmeshesOctree","getHeightAtCoordinates","tmpVect","_minX","_maxX","_minZ","_maxZ","_heightQuads","_initHeightQuads","_computeHeightQuads","_getFacetAt","getNormalAtCoordinates","getNormalAtCoordinatesToRef","tmpMat","updateCoordinateHeights","col","_width","_height","quad","slope","facet1","facet2","v4","v1v2","v1v3","v1v4","norm1","norm2","cd","d1","d2","CreateGroundVertexData","CreateTiledGroundVertexData","tileRow","tileCol","tileSize","applyTile","xTileMin","zTileMin","xTileMax","zTileMax","base","rowLength","square","CreateGroundFromHeightMapVertexData","colorFilter","idx1","idx2","idx3","idx4","isVisibleIdx1","isVisibleIdx2","isVisibleIdx3","CreateGround","CreateTiledGround","tiledGround","CreateGroundFromHeightMap","GroundBuilder","CreateBoxVertexData","topBaseAt","bottomBaseAt","topIndex","bottomIndex","basePositions","topFaceBase","bottomFaceBase","topFaceOrder","bottomFaceOrder","scaleArray","accumulator","currentIndex","faceUV","faceColors","totalColors","CreateBox","BoxBuilder","EnvironmentHelper","createGround","groundSize","groundTexture","_GroundTextureCDNUrl","groundOpacity","enableGroundShadow","groundShadowLevel","enableGroundMirror","groundMirrorSizeRatio","groundMirrorBlurKernel","groundMirrorAmount","groundMirrorFresnelWeight","groundMirrorFallOffDistance","groundMirrorTextureType","groundYBias","createSkybox","skyboxSize","skyboxTexture","_SkyboxTextureCDNUrl","skyboxColor","backgroundYRotation","sizeAuto","rootPosition","setupImageProcessing","_EnvironmentTextureCDNUrl","rootMesh","_rootMesh","skybox","_skybox","_skyboxTexture","skyboxMaterial","_skyboxMaterial","_ground","_groundTexture","groundMirror","_groundMirror","groundMirrorRenderList","groundMaterial","_groundMaterial","_errorHandler","_GetDefaultOptions","_setupBackground","_setupImageProcessing","updateOptions","newOptions","setMainColor","_setupEnvironmentTexture","CreateFromPrefilteredData","sceneSize","_getSceneSize","_setupGround","_setupGroundMaterial","_setupGroundDiffuseTexture","_setupGroundMirrorTexture","_setupMirrorInGroundMaterial","_setupSkybox","_setupSkyboxMaterial","_setupSkyboxReflectionTexture","sceneExtends","sceneDiagonal","sceneDiagonalLenght","wrapping","gammaGround","FreeCameraKeyboardMoveInput","keysUpward","keysDownward","rotationSpeed","keysRotateLeft","keysRotateRight","keysRotateUp","keysRotateDown","_localDirection","_getLocalRotation","FreeCameraMouseInput","touchEnabled","angularSensibility","onPointerMovedObservable","_allowCameraRotation","_activePointerId","_onMouseMove","BaseCameraMouseWheelInput","wheelPrecisionX","wheelPrecisionY","wheelPrecisionZ","onChangedObservable","_wheelDeltaX","_wheelDeltaY","_wheelDeltaZ","_ffMultiplier","_normalize","wheelDeltaX","wheelDeltaY","wheelDeltaZ","_CameraProperty","FreeCameraMouseWheelInput","_moveRelative","_rotateRelative","_moveScene","_wheelXAction","MoveRelative","_wheelXActionCoordinate","_wheelYAction","_wheelYActionCoordinate","_wheelZAction","_wheelZActionCoordinate","wheelXMoveRelative","wheelYMoveRelative","wheelZMoveRelative","wheelXRotateRelative","RotateRelative","wheelYRotateRelative","wheelZRotateRelative","wheelXMoveScene","MoveScene","wheelYMoveScene","wheelZMoveScene","_updateCamera","cameraTransformMatrix","transformedDirection","_updateCameraProperty","cameraProperty","FreeCameraTouchInput","allowMouse","touchAngularSensibility","touchMoveSensibility","singleFingerRotate","_offsetX","_offsetY","_pointerPressed","_isSafari","previousPosition","isMouseEvent","FreeCameraInputsManager","_mouseInput","_mouseWheelInput","addMouse","removeMouse","removeMouseWheel","addTouch","FreeCamera","mouse","applyGravity","_needMoveForGravity","_oldPosition","_diffPosition","_collideWithWorld","actualDisplacement","needMoveForGravity","addDeviceOrientation","_deviceOrientationInput","FreeCameraDeviceOrientationInput","rej","gotValue","eventHandler","DeviceOrientationEvent","requestPermission","_screenOrientationAngle","_screenQuaternion","_beta","_gamma","_onDeviceOrientationChangedObservable","_orientationChanged","_deviceOrientation","SmoothAngleChange","_constantTranform","DeviceOrientationCamera","_tmpDragQuaternion","_disablePointerInputWhenUsingDeviceOrientation","_dragFactor","_quaternionCache","_initialQuaternion","FromEulerAnglesToRef","disablePointerInputWhenUsingDeviceOrientation","enableHorizontalDragging","dragFactor","resetToCurrentRotation","axisName","VRCameraMetrics","compensateDistortion","multiviewEnabled","hResolution","vResolution","postProcessScaleFactor","vScreenSize","eyeToScreenDistance","leftHMatrix","hScreenSize","lensSeparationDistance","rightHMatrix","leftPreViewMatrix","interpupillaryDistance","rightPreViewMatrix","vScreenCenter","distortionK","chromaAbCorrection","lensCenterOffset","VRDistortionCorrectionPostProcess","isRightEye","_isRightEye","_distortionFactors","_postProcessScaleFactor","_lensCenterOffset","_scaleIn","_scaleFactor","_lensCenter","MultiviewRenderTarget","bindMultiviewFramebuffer","createMultiviewUbo","colorTexture","texStorage3D","multiviewTexture","ext","framebufferTextureMultisampleMultiviewOVR","framebufferTextureMultiviewOVR","bindSpaceWarpFramebuffer","_spaceWarpTexture","spaceWarpTexture","_resizeOrCreateMultiviewTexture","currentCreateSceneUniformBuffer","_transformMatrixR","VRMultiviewToSingleviewPostProcess","scaleFactor","setVRRigMode","metrics","vrCameraMetrics","GetDefault","VRDeviceOrientationFreeCamera","Gamepad","isConnected","_isConnected","browserGamepad","leftStickX","leftStickY","rightStickX","rightStickY","_leftStick","_rightStick","_invertLeftStickY","GAMEPAD","_leftStickAxisX","_leftStickAxisY","_rightStickAxisX","_rightStickAxisY","onleftstickchanged","_onleftstickchanged","onrightstickchanged","_onrightstickchanged","leftStick","newValues","rightStick","GENERIC","XBOX","POSE_ENABLED","DUALSHOCK","GenericPad","onbuttondown","_onbuttondown","onbuttonup","_onbuttonup","onButtonDownObservable","onButtonUpObservable","_buttons","_setButtonValue","buttonIndex","PoseEnabledControllerType","Xbox360Button","Xbox360Dpad","PoseEnabledControllerHelper","vrGamepad","_ControllerFactories","canCreate","_DefaultControllerFactory","PoseEnabledController","_disableTrackPosition","fixedPosition","_trackPosition","_calculatedPosition","isXR","_deviceRoomPosition","_deviceRoomRotationQuaternion","deviceRotationQuaternion","deviceScaleFactor","_maxRotationDistFromHeadset","_draggedRoomRotation","_leftHandSystemQuaternion","_deviceToWorld","_pointingPoseNode","_workingMatrix","_meshAttachedObservable","controllerType","_calculatedRotation","_updatePoseAndMesh","pose","updateFromDevice","_computeDevicePosition","distanceAway","rotationAmount","poseData","rawPose","attachToMesh","_poseControlledCamera","parents","attachToPoseControlledCamera","forwardWorld","setWebVRRigMode","vrDisplay","leftEye","getEyeParameters","rightEye","specs","frameData","parentCamera","POINTING_POSE","_vrExclusivePointerMode","_vrSupported","onVRDisplayChangedObservable","onVRRequestPresentComplete","onVRRequestPresentStart","isVRDevicePresent","_vrDisplay","getVRDevice","initWebVRAsync","eventArgs","vrSupported","_webVRInitPromise","_onVrDisplayConnect","_onVrDisplayDisconnect","_onVrDisplayPresentChange","isPresenting","_getVRDisplaysAsync","getVRDisplays","enableVR","onResolved","_onVRFullScreenTriggered","onRejected","presentationAttributes","highRefreshRate","vrPresentationAttributes","foveationLevel","useMultiview","requestPresent","_oldSize","_oldHardwareScaleFactor","exitPresent","_onVRDisplayPointerRestricted","_onVRDisplayPointerUnrestricted","submitFrame","WebVRFreeCamera","_webVROptions","_vrDevice","_specsVersion","_attached","_descendants","_standingMatrix","_worldToDevice","controllers","onControllersAttachedObservable","onControllerMeshLoadedObservable","onPoseUpdatedFromDeviceObservable","_poseSet","rigParenting","_defaultHeight","_detachIfAttached","_workingVector","_oneVector","_tmpMatrix","defaultHeight","trackPosition","controllerMeshes","defaultLightingOnControllers","_onVREnabled","success","initControllers","_frameData","VRFrameData","isController","controller","deviceDistanceToRoomGround","useStandingMatrix","stageParameters","sittingToStandingTransform","useStandingMatrixAsync","supported","_updateCacheWhenTrackingDisabledObserver","getControllerByName","hand","leftController","_leftController","rightController","_rightController","getFrameData","gamepadManager","onGamepadConnectedObservable","_onGamepadConnectedObserver","onGamepadDisconnectedObservable","_onGamepadDisconnectedObserver","resetPose","_correctPositionIfNotTrackPosition","isViewMatrix","_updateCacheCalled","viewArray","leftViewMatrix","rightViewMatrix","depthNear","depthFar","projectionArray","leftProjectionMatrix","rightProjectionMatrix","webVrController","defaultModel","controllerIndex","initControllerMesh","loadedMesh","_lightOnControllers","activateLightOnSubMeshes","firstViveWandDetected","VIVE","WebVRController","onButtonStateChange","_onButtonStateChange","_defaultModel","onTriggerStateChangedObservable","onMainButtonStateChangedObservable","onSecondaryButtonStateChangedObservable","onPadStateChangedObservable","onPadValuesChangedObservable","pad","_changes","pressChanged","touchChanged","valueChanged","newState","pressed","touched","_checkChanges","_handleButtonChange","Xbox360Pad","xboxOne","_leftTrigger","_rightTrigger","onPadDownObservable","onPadUpObservable","_buttonA","_buttonB","_buttonX","_buttonY","_buttonBack","_buttonStart","_buttonLB","_buttonRB","_buttonLeftStick","_buttonRightStick","_dPadUp","_dPadDown","_dPadLeft","_dPadRight","_isXboxOnePad","onlefttriggerchanged","_onlefttriggerchanged","onrighttriggerchanged","_onrighttriggerchanged","leftTrigger","rightTrigger","ondpaddown","_ondpaddown","ondpadup","_ondpadup","buttonType","_setDPadValue","buttonA","A","buttonB","B","buttonX","buttonY","buttonStart","Start","buttonBack","Back","buttonLB","LB","buttonRB","RB","buttonLeftStick","LeftStick","buttonRightStick","RightStick","dPadUp","dPadDown","dPadLeft","dPadRight","premulAlpha","forceBindTexture","allowGPUOptimization","glformat","DynamicTexture","_canvas","_recreate","scaleTo","fillRect","drawText","fillStyle","textSize","measureText","fillText","_IsCanvasElement","ImageProcessingPostProcess","doNotBuild","fromLinearSpace","_fromLinearSpace","FROMLINEARSPACE","DualShockButton","DualShockDpad","DualShockPad","_buttonCross","_buttonCircle","_buttonSquare","_buttonTriangle","_buttonShare","_buttonOptions","_buttonL1","_buttonR1","buttonCross","buttonCircle","Circle","buttonSquare","Square","buttonTriangle","Triangle","buttonOptions","Options","buttonShare","Share","buttonL1","L1","buttonR1","R1","GamepadManager","_babylonGamepads","_oneGamepadConnected","_isMonitoring","_gamepadEventSupported","_gamepadSupport","_onGamepadConnectedEvent","newGamepad","_addNewGamepad","_startMonitoringGamepads","_onGamepadDisconnectedEvent","disconnectedGamepad","_updateGamepadObjects","getGamepadByType","_stopMonitoringGamepads","dualShock","InitiateController","_checkGamepadsStatus","_loggedErrors","FreeCameraGamepadInput","gamepadAngularSensibility","gamepadMoveSensibility","deadzoneDelta","_yAxisScale","_cameraTransform","_deltaTransform","_vector3","_vector2","invertYAxis","lsValues","rsValues","ArcRotateCameraGamepadInput","gamepadRotationSensibility","normalizedRX","normalizedRY","normalizedLY","_gamepadManager","GamepadSystemSceneComponent","addGamepad","_beforeCameraUpdate","_staticOffsetValueQuaternion","_staticOffsetValueVector3","_staticOffsetValueVector2","_staticOffsetValueSize","_staticOffsetValueColor3","RuntimeAnimation","_currentFrame","_weight","_currentValue","targetPath","_targetPath","_currentActiveTarget","isAdditive","_host","_originalValue","_originalBlendValue","_offsetsCache","_highLimitsCache","_stopped","_blendingFactor","_directTarget","_absoluteFrameOffset","_previousElapsedTime","_previousAbsoluteFrame","_targetIsArray","_animation","_activeTargets","_animationState","_getCorrectLoopMode","_minFrame","_maxFrame","_minValue","_maxValue","newKey","_preparePath","_getOriginalValues","_clone","_enableBlending","restoreOriginal","_setValue","isDone","setValue","getRestPose","_UniversalLerp","_registerTargetForLateAnimationBinding","goToFrame","onlyOnce","_prepareForSpeedRatioChange","newSpeedRatio","newAbsoluteFrame","elapsedTimeSinceAnimationStart","frameRange","absoluteFrame","keyOffset","fromValue","toValue","syncRoot","masterFrame","fromFrame","toFrame","_onLoop","Bone","_matrix","_compose","_needToCompose","_markAsDirtyAndDecompose","parentBone","restMatrix","bindMatrix","_index","_scalingDeterminant","_needToDecompose","_linkedTransformNode","_waitingTransformNodeId","_restMatrix","_bindMatrix","_absoluteMatrix","_absoluteBindMatrix","_absoluteInverseBindMatrix","_finalMatrix","_updateAbsoluteBindMatrices","getParent","updateAbsoluteBindMatrices","getLocalMatrix","getBindMatrix","getBaseMatrix","getRestMatrix","setRestMatrix","setRestPose","getBindPose","setBindMatrix","setBindPose","returnToRest","localScaling","localRotation","getAbsoluteInverseBindMatrix","getInvertedAbsoluteTransform","getAbsoluteMatrix","getAbsoluteTransform","linkTransformNode","_numBonesWithLinkedTransformNode","getTransformNode","_decompose","_localPosition","_markAsDirtyAndCompose","getRotation","setRotation","_localRotation","getScale","setScale","_localScaling","updateLocalMatrix","updateChildren","_markAsDirty","tNode","translationMode","computeAbsoluteMatrices","_TmpMats","_TmpVecs","scaleChildren","locMat","scaleMat","cm","getScaleToRef","setYawPitchRoll","_TmpQuat","rotMatInv","_getAbsoluteInverseMatrixUnscaledToRef","_rotateWithMatrix","rmat","setAxisAngle","setRotationMatrix","rotMat2","lmat","lx","ly","lz","parentScale","parentScaleInv","getPosition","getPositionToRef","getAbsolutePositionToRef","poseMatrix","computeAbsoluteTransforms","getRotationToRef","getRotationQuaternionToRef","amat","getAbsolutePositionFromLocal","getAbsolutePositionFromLocalToRef","getLocalPositionFromAbsolute","getLocalPositionFromAbsoluteToRef","setCurrentPoseAsRest","Animatable","_syncRoot","_speedRatio","_goToFrame","loopAnimation","onAnimationLoop","playOrder","_localDelayOffset","_pausedDelay","_manualJumpDelay","_paused","_frameToSyncFromJump","disposeOnEnd","animationStarted","onAnimationLoopObservable","appendAnimations","syncWith","getAnimations","newRuntimeAnimation","getAnimationByTargetProperty","getRuntimeAnimationByTargetProperty","disableBlending","fps","pause","restart","_raiseOnAnimationEnd","useGlobalSplice","waitAsync","running","isRunning","animationTime","_processLateAnimationBindings","sortActiveAnimatables","beginWeightedAnimation","returnedAnimatable","stopCurrent","shouldRunTargetAnimations","beginHierarchyAnimation","getAnimatableByTarget","getAllAnimatablesByTarget","_lateAnimationHolders","totalAdditiveWeight","additiveAnimations","_processLateAnimationBindingsForMatrices","holder","normalizer","finalPosition","finalScaling","finalQuaternion","originalAnimation","skipOverride","animIndex","currentPosition","currentScaling","currentQuaternion","_processLateAnimationBindingsForQuaternions","refQuaternion","cumulativeQuaternion","quaternions","cumulativeAmount","matrixDecomposeMode","finalValue","copyAnimationRange","rangeName","frameOffset","rescaleAsRequired","skelDimensionsRatio","sourceRange","sourceKeys","sourceBoneLength","sourceParent","parentScalingReqd","parentRatio","dimensionsScalingReqd","destKeys","origTranslation","WebXRLayerWrapper","isFixedFoveationSupported","layerType","fixedFoveation","getWidth","getHeight","createRenderTargetTextureProvider","WebXRLayerRenderTargetTextureProvider","layerWrapper","_renderTargetTextures","XRWebGLBinding","_destroyRenderTargetTexture","getFramebufferDimensions","_framebufferDimensions","WebXRWebGLLayerWrapper","sessionManager","WebXRWebGLLayerRenderTargetTextureProvider","_layer","trySetViewportForView","xrViewport","getViewport","getRenderTargetTextureForEye","layerWidth","layerHeight","_rtt","getRenderTargetTextureForView","WebXRManagedOutputCanvasOptions","canvasOptions","framebufferScaleFactor","newCanvasCssStyle","WebXRManagedOutputCanvas","_xrSessionManager","GetDefaults","xrLayer","_xrLayerWrapper","onXRLayerInitObservable","canvasElement","_setManagedOutputCanvas","cssText","onXRSessionInit","_addCanvas","onXRSessionEnded","_removeCanvas","xrSession","createLayer","XRWebGLLayer","canvasContext","makeXRCompatible","_setCanvasSize","_originalCanvasSize","offsetWidth","offsetHeight","NativeXRLayerWrapper","NativeXRLayerRenderTargetTextureProvider","_nativeRTTProvider","xr","getNativeRenderTargetProvider","session","_nativeLayer","getRenderTargetForEye","NativeXRRenderTarget","_nativeRenderTarget","getWebXRRenderTarget","initializeXRLayerAsync","WebXRSessionManager","currentTimestamp","defaultHeightCompensation","onXRFrameObservable","onXRReferenceSpaceChanged","inXRFrameLoop","inXRSession","_onEngineDisposedObserver","referenceSpace","_referenceSpace","newReferenceSpace","sessionMode","_sessionMode","exitXRAsync","_baseLayerRTTProvider","_xrNavigator","native","initializeAsync","initializeSessionAsync","xrSessionMode","xrSessionInit","requestSession","isNative","_baseLayerWrapper","once","isSessionSupportedAsync","IsSessionSupportedAsync","resetReferenceSpace","baseReferenceSpace","runXRRenderLoop","timestamp","xrFrame","setReferenceSpaceTypeAsync","referenceSpaceType","requestReferenceSpace","rejectionReason","heightCompensation","XRRigidTransform","getOffsetReferenceSpace","viewerReferenceSpace","updateRenderStateAsync","updateRenderState","_setBaseLayerWrapper","baseLayerWrapper","_getBaseLayerWrapper","baseLayer","functionToUse","isSessionSupported","supportsSession","currentFrameRate","supportedFrameRates","updateTargetFrameRate","rate","runInXRFrame","ignoreIfNotInSession","enabledFeatures","WebXRState","WebXRTrackingState","CreateCylinderVertexData","hasRings","enclose","surfaceNb","angleStep","ringVertex","ringNormal","ringFirstVertex","ringFirstNormal","quadNormal","ringIdx","cs","i2","i3","createCylinderCap","isTop","circleVector","vbase","textureScale","textureCoordinate","CreateCylinder","cylinder","CylinderBuilder","CreateTorusVertexData","outerAngle","innerAngle","dx","dy","nextI","nextJ","CreateTorus","torus","VRExperienceHelperGazer","gazeTrackerToClone","_pointerDownOnMeshAsked","_isActionableMesh","_teleportationRequestInitiated","_teleportationBackRequestInitiated","_rotationRightAsked","_rotationLeftAsked","_dpadPressed","_activePointer","_IdCounter","_gazeTracker","targetMat","_getForwardRay","_selectionPointerDown","_currentHit","_selectionPointerUp","_activatePointer","_deactivatePointer","_updatePointerDistance","_interactionsEnabled","_teleportationEnabled","VRExperienceHelperControllerGazer","webVRController","_laserPointer","laserPointerMaterial","preloadMesh","preloadPointerPose","_setLaserPointerParent","_meshAttachedObserver","_setLaserPointerColor","_setLaserPointerLightingDisabled","disabled","makeNotPick","meshChildren","laserParent","VRExperienceHelperCameraGazer","_getCamera","VRExperienceHelper","onEnteringVR","onEnteringVRObservable","onExitingVR","onExitingVRObservable","onControllerMeshLoaded","teleportationTarget","_teleportationTarget","_isDefaultTeleportationTarget","gazeTrackerMesh","_cameraGazer","leftControllerGazeTrackerMesh","rightControllerGazeTrackerMesh","displayGaze","_displayGaze","displayLaserPointer","_displayLaserPointer","deviceOrientationCamera","_deviceOrientationCamera","currentVRCamera","_webVRready","_webVRCamera","vrDeviceOrientationCamera","_vrDeviceOrientationCamera","vrButton","_btnVR","webVROptions","_webVRsupported","_webVRrequesting","_webVRpresenting","_fullscreenVRpresenting","enableGazeEvenWhenNoPointerLock","exitVROnDoubleTap","onAfterEnteringVRObservable","_useCustomVRButton","_teleportationRequested","_teleportActive","_floorMeshesCollection","_teleportationMode","TELEPORTATIONMODE_CONSTANTTIME","_teleportationTime","_teleportationSpeed","_rotationAllowed","_teleportBackwardsVector","_teleportationFillColor","_teleportationBorderColor","_rotationAngle","_haloCenter","_padSensibilityUp","_padSensibilityDown","_gazeColor","_laserColor","_pickedLaserColor","_pickedGazeColor","onNewMeshSelected","onMeshSelectedWithController","onNewMeshPicked","onBeforeCameraTeleport","onAfterCameraTeleport","onSelectedMeshUnselected","teleportationEnabled","_teleportationInitialized","_interactionsRequested","updateGazeTrackerScale","updateGazeTrackerColor","updateControllerLaserColor","requestPointerLockOnFullScreen","xrTestDone","_onResize","_moveButtonToBottomRight","exitVR","_inputElement","offsetTop","offsetLeft","_updateButtonVisibility","_cachedAngularSensibility","_beforeRender","_castRayAndSelectObject","_noControllerIsActive","_onNewGamepadConnected","stickValues","_checkTeleportWithRay","_checkTeleportBackwards","_checkRotate","buttonPressed","_tryEnableInteractionOnController","_enableInteractionOnController","_enableTeleportationOnController","_onNewGamepadDisconnected","_workingQuaternion","useXR","createFallbackVRDeviceOrientationFreeCamera","createDeviceOrientationCamera","laserToggle","useCustomVRButton","customVRButton","rayLength","_rayLength","_hasEnteredVR","_existingCamera","createDefaultXRExperienceAsync","floorMeshes","baseExperience","onStateChangedObservable","ENTERING_XR","pointerSelection","EXITING_XR","IN_XR","NOT_IN_XR","_completeVRInit","vrDeviceOrientationCameraMetrics","title","css","SVGSVGElement","createTextNode","isInVRMode","enterVR","_displayVRButton","_onVRDisplayChangedBind","_onVRDisplayChanged","_onVrDisplayPresentChangeBind","_onVRRequestPresentStart","_onVRRequestPresentComplete","_onDefaultMeshLoaded","_circleEase","_teleportationEasing","enableTeleportation","wasPresenting","rect","_btnVRDisplayed","enterXRAsync","FromRotationMatrix","currentYRotation","currentGlobalRotation","enableInteractions","raySelectionPredicate","_floorMeshName","meshSelectionPredicate","_raySelectionPredicate","_isTeleportationFloor","addFloorMesh","floorMesh","removeFloorMesh","vrTeleportationOptions","floorMeshName","teleportation","waitForXr","teleportationMode","teleportationTime","teleportationSpeed","_postProcessMove","_createTeleportationCircles","stateObject","gazer","teleportCamera","_rotateCamera","dynamicTexture","beginPath","lineWidth","strokeStyle","stroke","teleportationCircleMaterial","animationInnerCircle","_hideTeleportationTarget","_displayTeleportationTarget","animationRotation","animationRotationKeys","animationPP","vignetteWeightKeys","animationPP2","vignetteStretchKeys","_moveTeleportationSelectorTo","pickNormal","_convertNormalToDirectionOfRay","lastFrame","TELEPORTATIONMODE_CONSTANTSPEED","dist","animationCameraTeleportation","animationCameraTeleportationKeys","midFrame","multiplier","deltaFighting","_currentMeshSelected","_notifySelectedMeshUnselected","changeGazeColor","changeLaserColor","gazerAsControllerGazer","setLaserColor","pickedColor","setLaserLightingState","setGazeColor","DDSD_MIPMAPCOUNT","DDPF_LUMINANCE","FourCCToInt32","FOURCC_DXT1","FOURCC_DXT3","FOURCC_DXT5","FOURCC_DX10","DDSTools","header","extendedHeader","headerLengthInt","mipmapCount","fourCC","dxgiFormat","isFourCC","isRGB","isLuminance","dataOffset","destArray","srcData","srcPos","StoreLODInAlphaChannel","rOffset","gOffset","bOffset","aOffset","byteArray","_ExtractLongWordOrder","loadMipmaps","lodIndex","currentFace","destTypeMustBeFilterable","sphericalPolynomialFaces","mip","internalCompressedFormat","blockBytes","bpp","computeFormats","startFace","floatArray","_GetFloatAsUIntRGBAArrayBuffer","_GetFloatRGBAArrayBuffer","_GetHalfFloatAsUIntRGBAArrayBuffer","_GetHalfFloatAsFloatRGBAArrayBuffer","floatAvailable","halfFloatAvailable","destType","dataGetter","dataGetterPolynomial","_GetFloatAsHalfFloatRGBAArrayBuffer","_GetHalfFloatRGBAArrayBuffer","_GetRGBArrayBuffer","_GetRGBAArrayBuffer","unpackAlignment","unpaddedRowSize","_GetLuminanceArrayBuffer","minLODIndex","maxLODIndex","mipmapIndex","glTextureFromLod","UploadDDSLevels","lodTexture","_DDSTextureLoader","GetDDSInfo","DefaultEnvironmentTextureImageType","MagicBytes","GetEnvInfo","manifestString","charCode","manifest","normalizeEnvInfo","specularDataPosition","imageType","_CreateEnvTextureIrradiance","polynmials","CreateImageDataArrayBufferViews","specularInfo","mipmapsCount","mipmaps","imageInfo","UploadEnvLevelsAsync","UploadLevelsAsync","_OnImageReadyAsync","generateNonLODTextures","lodTextures","cubeRtt","tempTexture","ImageBitmap","IsExponentOfTwo","ILog2","mipSlices","promises","UploadEnvSpherical","irradianceInfo","irradiance","EnvironmentTextureTools","CreateEnvTextureAsync","sphericalPolynomialPromise","cubeWidth","specularTextures","faceWidth","faceDataFloat","floatData","EncodeTextureToRGBD","rgbdEncodedData","imageEncodedData","imageQuality","infoString","infoBuffer","infoView","strLen","totalSize","finalBuffer","finalBufferView","setUint8","_ENVTextureLoader","KhronosTextureContainer","facesExpected","isInvalid","IsValid","dataSize","headerDataView","littleEndian","glType","glTypeSize","glFormat","glInternalFormat","glBaseInternalFormat","pixelWidth","pixelHeight","pixelDepth","numberOfArrayElements","numberOfFaces","numberOfMipmapLevels","bytesOfKeyValueData","loadType","COMPRESSED_2D","uploadLevels","_upload2DCompressedLevels","HEADER_LEN","imageSize","identifier","COMPRESSED_3D","TEX_2D","TEX_3D","AutoReleaseWorkerPool","WorkerPool","workers","_pendingActions","_workerInfos","worker","workerPromise","idle","workerInfo","terminate","_executeOnIdleWorker","nextAction","maxWorkers","createWorkerAsync","DefaultOptions","_maxWorkers","_createWorkerAsync","idleTimeElapsedBeforeRelease","SourceTextureFormat","TranscodeTarget","EngineFormat","SceneLoaderAnimationGroupLoadingMode","getAbsoluteUrlOrNull","applyConfig","urls","wasmUASTCToASTC","KTX2DECODER","LiteTranscoder_UASTC_ASTC","WasmModuleURL","wasmUASTCToBC7","LiteTranscoder_UASTC_BC7","wasmUASTCToRGBA_UNORM","LiteTranscoder_UASTC_RGBA_UNORM","wasmUASTCToRGBA_SRGB","LiteTranscoder_UASTC_RGBA_SRGB","wasmUASTCToR8_UNORM","LiteTranscoder_UASTC_R8_UNORM","wasmUASTCToRG8_UNORM","LiteTranscoder_UASTC_RG8_UNORM","jsMSCTranscoder","MSCTranscoder","JSModuleURL","wasmMSCTranscoder","wasmZSTDDecoder","ZSTDDecoder","KhronosTextureContainer2","hardwareConcurrency","numWorkers","_WorkerPoolPromise","_DecoderModulePromise","jsDecoderModule","URLConfig","Worker","workerContent","workerFunc","workerBlobUrl","onMessage","postMessage","LoadScriptAsync","UseFromWorkerThread","WASMMemoryManager","LoadBinariesFromCurrentThread","KTX2Decoder","DefaultNumWorkers","_Initialize","uploadAsync","compressedTexturesCaps","workerPool","decodedData","msg","DefaultDecoderOptions","_getKTX2DecoderOptions","dataCopy","decoder","transcodedFormat","isInGammaSpace","transcoderName","isUncompressedFormat","mipmap","ktx2Decoder","onmessage","GetDefaultNumWorkers","DefaultKTX2DecoderOptions","_useRGBAIfOnlyBC1BC3AvailableWhenUASTC","_ktx2DecoderOptions","useRGBAIfASTCBC7NotAvailableWhenUASTC","_useRGBAIfASTCBC7NotAvailableWhenUASTC","useRGBAIfOnlyBC1BC3AvailableWhenUASTC","forceRGBA","_forceRGBA","forceR8","_forceR8","forceRG8","_forceRG8","bypassTranscoders","_bypassTranscoders","transcodeFormatDecisionTree","UASTC","transcodeFormat","BC1_RGB","BC3_RGBA","yes","RGBA32","engineFormat","RGBA8Format","roundToMultiple4","_KTXTextureLoader","ktx","mappedFormat","mapSRGBToLinear","WebXRCamera","_firstFrame","_referenceQuaternion","_referencedPosition","_trackingState","NOT_TRACKING","onTrackingStateChanged","compensateOnFirstFrame","_rotate180","_updateNumberOfRigCameras","_updateFromXRSession","_updateReferenceSpace","_setTrackingState","realWorldHeight","basePose","getViewerPose","_updateForDualEyeDebugging","setTransformationFromNonVRCamera","otherCamera","resetToBaseReferenceSpace","yRotation","_lastXRViewerPose","emulatedPosition","TRACKING_LOST","TRACKING","xrRenderState","views","currentRig","viewCount","removedCamera","referencedMat","poseMat","transformMat","_ScaleReadOnly","WebXRFeatureName","ANCHOR_SYSTEM","BACKGROUND_REMOVER","HIT_TEST","MESH_DETECTION","PHYSICS_CONTROLLERS","PLANE_DETECTION","POINTER_SELECTION","TELEPORTATION","FEATURE_POINTS","HAND_TRACKING","IMAGE_TRACKING","NEAR_INTERACTION","DOM_OVERLAY","MOVEMENT","LIGHT_ESTIMATION","EYE_TRACKING","WALKING_LOCOMOTION","LAYERS","DEPTH_SENSING","SPACE_WARP","WebXRFeaturesManager","getEnabledFeatures","featureName","feature","featureImplementation","disableAutoAttach","attachFeature","detachFeature","constructorFunction","stable","_AvailableFeatures","latest","xrSessionManager","disableFeature","Name","enableFeature","moduleOptions","attachIfPossible","required","versionToLoad","GetStableVersionOfFeature","GetLatestVersionOfFeature","conflictingFeature","_ConflictingFeatures","constructFunction","ConstructFeature","constructed","dependentsFound","isCompatible","getEnabledFeature","nativeName","xrNativeFeatureName","requiredFeatures","optionalFeatures","getXRSessionInitExtension","extended","TouchCamera","touch","UniversalCamera","WebXRExperienceHelper","_nonVRCamera","_attachedToElement","_spectatorCamera","_originalSceneAutoClear","_supported","_spectatorMode","_lastTimestamp","onInitialXRPoseSetObservable","featuresManager","helper","_setState","sessionCreationOptions","_extendXRSessionInitObject","_nonXRToXRCamera","enableSpectatorMode","_switchSpectatorMode","disableSpecatatorMode","preferredCameraIndex","updateSpectatorCamera","onStateChanged","WebXRControllerComponent","_buttonIndex","_axesIndices","_axes","_hasChanges","_pressed","_touched","onAxisValueChangedObservable","onButtonStateChangedObservable","isAxes","isButton","nativeController","buttonUpdated","axesUpdate","BUTTON_TYPE","SQUEEZE_TYPE","THUMBSTICK_TYPE","TOUCHPAD_TYPE","TRIGGER_TYPE","SceneLoader","_RegisteredPlugins","registeredPlugin","GetDefaultPlugin","canDirectLoad","sceneFilename","queryStringPosition","dotPosition","_GetPluginForExtension","fileInfo","errorMessage","rawData","pluginExtension","directLoad","_GetDirectLoad","_GetPluginForDirectLoad","_GetPluginForFilename","isBinary","createPlugin","OnPluginActivatedObservable","dataCallback","pluginDisposed","manifestChecked","errorCallback","canUseOfflineSupport","exceptionFound","sceneFile","filename","GetFilename","GetFolderPath","meshNames","_GetFileInfo","loadingToken","disposeHandler","_FormatErrorMessage","progressHandler","successHandler","_LoadData","rewriteRootURL","importMesh","syncedPlugin","loadingPluginName","importMeshAsync","ImportMesh","Append","Load","_ShowingLoadingScreen","loadAsync","assets","loadAssetContainer","assetContainer","populateRootNodes","loadAssetContainerAsync","LoadAssetContainer","overwriteAnimations","animationGroupLoadingMode","Clean","targetConverter","animationGroup","Stop","Sync","NoSync","startingIndexForNewAnimatables","mergeAnimationsTo","ImportAnimations","NO_LOGGING","MINIMAL_LOGGING","SUMMARY_LOGGING","DETAILED_LOGGING","WebXRAbstractMotionController","layout","gamepadObject","handedness","_doNotLoadControllerMesh","_controllerCache","_initComponent","componentDef","components","gamepadIndices","_modelReady","disableAnimation","onModelLoadedObservable","getComponentIds","getComponent","getAllComponentsOfType","getComponentOfType","getMainComponent","selectComponentId","useGeneric","_getModelLoadingConstraints","loadingParams","_getGenericFilenameAndPath","_getFilenameAndPath","meshesLoaded","_getGenericParentMesh","_setRootMesh","_processLoadedModel","found","profileId","updateFromXRFrame","updateModel","handness","pulse","hapticActuatorIndex","hapticActuators","_getChildByName","_getImmediateChildByName","_lerpTransform","axisMap","axisValue","fixValueCoordinates","minMesh","maxMesh","valueMesh","lerpValue","_updateModel","WebXRGenericTriggerMotionController","GenericTriggerLayout","ProfileId","rootNodeName","visualResponses","gamepadMapping","assetPath","none","CreateSphereVertexData","diameterX","diameterY","diameterZ","dedupTopBottomIndices","totalZRotationSteps","totalYRotationSteps","zRotationStep","normalizedZ","angleZ","yRotationStep","normalizedY","angleY","rotationZ","RotationZ","afterRotZ","firstIndex","CreateSphere","SphereBuilder","WebXRProfiledMotionController","xrInput","_profile","_repositoryUrl","controllerCache","layouts","_buttonMeshMapping","_touchDots","visResKey","glbLoaded","IsPluginForExtensionAvailable","componentInLayout","mainMesh","states","visualResponseKey","visResponse","valueNodeProperty","valueNodeName","minNodeName","maxNodeName","nameOfMesh","touchPointNodeName","_xrFrame","componentProperty","WebXRMotionControllerManager","_ProfilesList","_ProfileLoadingPromises","RegisterFallbacksForProfileId","returnArray","_Fallbacks","forceProfile","profileArray","profiles","windowsMRIdx","UseOnlineRepository","firstFunction","PrioritizeOnlineRepository","_LoadProfileFromRepository","_LoadProfilesFromAvailableControllers","secondFunction","_AvailableControllers","LoadFileAsync","BaseRepositoryUrl","UpdateProfilesList","profilesList","profileToLoad","profile","DisableControllerCache","FindFallbackWithProfileId","constructionFunction","RegisterController","DefaultFallbacks","idCount","WebXRInputSource","inputSource","_tmpVector","_disposed","onMeshLoadedObservable","onMotionControllerInitObservable","targetRayMode","gripSpace","grip","GetMotionControllerWithXRInput","forceControllerProfile","motionController","doNotLoadControllerMesh","loadModel","disableMotionControllerAnimation","getWorldPointerRayToRef","gripIfAvailable","xrCamera","getPose","targetRaySpace","_lastXRPose","WebXRInput","onControllerAddedObservable","onControllerRemovedObservable","_onInputSourcesChange","_addAndRemoveControllers","added","_sessionEndedObserver","_sessionInitObserver","_frameObserver","customControllersRepositoryURL","disableOnlineControllerRepository","addInputs","removeInputs","sources","controllerOptions","forceInputProfile","doNotLoadControllerMeshes","disableControllerAnimation","keepControllers","removedControllers","ClearControllerCache","WebXRAbstractFeature","_removeOnDetach","_addNewAttachObserver","_onXRFrame","UtilityLayerRenderer","getRenderCamera","getRigParentIfPossible","_renderCamera","activeCam","originalScene","setRenderCamera","_getSharedGizmoLight","_sharedGizmoLight","utilityLayerScene","Gray","DefaultUtilityLayer","_DefaultUtilityLayer","_CreateDefaultUtilityLayerFromScene","DefaultKeepDepthUtilityLayer","_DefaultKeepDepthUtilityLayer","handleEvents","_lastPointerEvents","pickUtilitySceneFirst","onlyCheckPointerDownEvents","processAllEvents","pickingEnabled","onPointerOutObservable","_originalPointerObserver","prePointerInfo","pointerEvent","getNearPickDataForScene","scenePick","previousActiveCamera","utilityScenePick","originalScenePick","mainSceneTrackerPredicate","_notifyObservers","_afterRenderObserver","_sceneDisposeObserver","oldScene","WebXRControllerPointerSelection","_attachController","xrController","_controllers","laserPointer","selectionMesh","_generateNewMeshPair","tmpRay","disabledByNearInteraction","_attachedController","enablePointerSelectionOnAllControllers","preferredHandedness","_attachTrackedPointerRayMode","_attachGazeMode","_attachScreenRayMode","_tmpVectorForPickCompare","disablePointerLighting","disableSelectionMeshLighting","displaySelectionMesh","laserPointerPickedColor","laserPointerDefaultColor","selectionMeshDefaultColor","selectionMeshPickedColor","_identityMatrix","_screenCoordinatesRef","_viewportRef","_detachController","gazeCamera","webXRCamera","controllerId","getMeshUnderPointer","getXRControllerByPointerId","_getPointerSelectionDisabledByPointerId","_setPointerSelectionDisabledByPointerId","controllerData","controllerGlobalPosition","maxPointerDistance","disableScenePointerVectorUpdate","screenCoordinates","_utilityLayerScene","customUtilityLayerScene","timeToSelect","sceneToRenderTo","useUtilityLayer","oldPick","discMesh","timer","downTriggered","onFrameObserver","_augmentPointerInit","_pickingMoved","disablePointerUpOnTouchOut","finalPointerUpTriggered","pointerDownTriggered","forceGazeMode","overrideButtonId","selectionComponent","onButtonChangedObserver","disableSwitchOnClick","selectStartListener","selectEndListener","eventListeners","selectend","selectstart","xrControllerUniqueId","eventName","meshParent","customLasterPointerMeshGenerator","customSelectionMeshGenerator","newPick","gazeModePointerMovedFactor","screenX","screenY","lasterPointerDefaultColor","NodeMaterialBlockConnectionPointTypes","NodeMaterialBlockTargets","NodeMaterialConnectionPointCompatibilityStates","NodeMaterialConnectionPointDirection","PropertyTypeForEdition","NodeMaterialBlockConnectionPointMode","NodeMaterialSystemValues","AnimatedInputBlockTypes","AddWebXRFeature","NodeMaterialBuildState","supportUniformBuffers","constants","functions","counters","_attributeDeclaration","_uniformDeclaration","_constantDeclaration","_samplerDeclaration","_varyingTransfer","_injectAtEnd","_repeatableContentAnchorIndex","_builtCompilationString","compilationString","finalize","emitComments","sharedData","isFragmentMode","Fragment","functionCode","varyingDeclaration","extensionName","_repeatableContentAnchor","_getFreeVariableName","variableNames","_getFreeDefineName","_excludeVariableName","_emit2DSampler","_emit2DArraySampler","_getGLType","Float","Int","_emitExtension","_emitFunction","comments","_emitCodeFromInclude","includeName","repeatKey","substitutionVars","replaceStrings","replaceString","_emitFunctionFromInclude","storeKey","removeAttributes","removeUniforms","removeVaryings","removeIfDef","_registerTempVariable","temps","_emitVaryingFromString","notDefine","varyings","_emitUniformFromString","_emitFloat","NodeMaterialBuildStateSharedData","inputBlocks","textureBlocks","bindableBlocks","forcedBindableBlocks","blocksWithFallbacks","blocksWithDefines","repeatableContentBlocks","dynamicUniformBlocks","blockingBlocks","animatedInputs","hints","needWorldViewMatrix","needWorldViewProjectionMatrix","checks","emitVertex","emitFragment","notConnectedNonOptionalInputs","allowEmptyVertexProgram","emitErrors","notConnectedInput","ownerBlock","NodeMaterialConnectionPoint","type1","type2","_direction","associatedVariableName","_ownerBlock","isInput","_enforceAssociatedVariableName","_associatedVariableName","_connectedPoint","innerType","_linkedConnectionSource","AutoDetect","BasedOnInput","_typeConnectionSource","_defaultConnectionPointType","_prioritizeVertex","VertexAndFragment","Vertex","connectedPoint","hasEndpoints","isConnectedToInputBlock","connectInputBlock","sourceBlock","connectedBlocks","_endpoints","endpoints","isDirectlyConnectedToVertexOutput","endpoint","Neutral","isConnectedInVertexShader","isConnectedInFragmentShader","createCustomInputBlock","_acceptedConnectionPointType","needDualDirectionValidation","acceptedConnectionPointTypes","excludedConnectionPointTypes","onConnectionObservable","isExposedOnFrame","exposedPortPosition","canConnectTo","connectionPoint","checkCompatibilityState","Compatible","otherBlock","TargetIncompatible","AreEquivalentTypes","TypeIncompatible","targetBlock","Input","isAnAncestorOf","HierarchyIssue","connectTo","ignoreConstraints","disconnectFrom","addExcludedConnectionPointFromAllowedTypes","bitmask","All","displayName","inputName","targetBlockId","targetConnectionName","NodeMaterialBlock","newName","validateBlockName","isUnique","_isUnique","isFinalMerger","_isFinalMerger","_isInput","isTeleportOut","_isTeleportOut","isTeleportIn","_isTeleportIn","buildId","_buildId","_outputs","getInputByName","getOutputByName","inputsAreExclusive","_codeVariableName","visibleInInspector","visibleOnFrame","_originalTargetIsNeutral","_setInitialTarget","nodeMaterial","_declareOutput","_writeVariable","_writeFloat","stringVersion","registerInput","isOptional","registerOutput","Output","getFirstAvailableInput","forOutput","getFirstAvailableOutput","forBlock","getSiblingOutput","notFound","_buildBlock","updateUniformsAndSamples","provideFallbacks","initializeDefines","autoConfigure","replaceRepeatableContent","vertexShaderState","fragmentShaderState","willBeGeneratedIntoVertexShaderFromFragmentShader","_linkConnectionTypes","inputIndex0","inputIndex1","looseCoupling","_processBuild","activeBlocks","build","localBlockIsFragment","_vertexState","otherBlockWasGeneratedInVertexShader","_buildTarget","isAttribute","_noContextSwitch","reservedNames","reservedName","_customBuildStep","verbose","_inputRename","_outputRename","_dumpPropertiesCode","_dumpCode","uniqueNames","alreadyDumped","nameAsVariableName","codeString","connectedBlock","_dumpCodeForOutputConnections","connectedOutput","blockType","_deserialize","_deserializePortDisplayNamesAndExposedOnFrame","serializedInputs","serializedOutputs","port","TransformBlock","complementW","complementZ","otherAsInput","xyz","transformName","VertexOutputBlock","_isLogarithmicDepthEnabled","nodeList","fragmentOutputNodes","editableInPropertyPage","groupName","propStore","_propStore","FragmentOutputBlock","convertToGammaSpace","convertToLinearSpace","rgba","_linearDefineName","_gammaDefineName","aValue","notifiers","remapAttributeName","position2d","particle_uv","particle_color","particle_texturemask","particle_positionw","attributeInFragmentOnly","attributeAsUniform","InputBlock","isUniform","isSystemValue","_systemValue","World","WorldView","WorldViewProjection","View","ViewProjection","Projection","CameraPosition","FogColor","DeltaTime","MaterialAlpha","CameraParameters","Undefined","_animationType","None","isBoolean","matrixMode","isConstant","groupInInspector","onValueChangedObservable","setDefaultValue","setAsAttribute","Attribute","setAsSystemValue","systemValue","_storedValue","Uniform","valueCallback","_valueCallback","isUndefined","isVarying","Varying","Time","RealTime","_emitDefine","_emitConstant","_emit","_transmitWorld","worldView","worldViewProjection","_transmit","codes","valueString","valueType","wellKnownValue","CurrentScreenBlock","_samplerName","_injectVertexCode","uvInput","_mainUVName","_writeTextureRead","_writeOutput","vertexMode","_tempTextureRead","swizzle","ParticleTextureBlock","getInputBlockByPredicate","ParticleRampGradientBlock","rampColor","ParticleBlendMultiplyBlock","alphaTexture","alphaColor","VectorMergerBlock","xSwizzle","ySwizzle","zSwizzle","wSwizzle","xyzwIn","xyzIn","xyIn","zwIn","xyzw","xyzOut","xyOut","zwOut","_buildSwizzle","xInput","yInput","zInput","wInput","xyInput","zwInput","xyzInput","xyzwInput","v4Output","v3Output","v2Output","v2CompOutput","RemapBlock","targetRange","sourceMin","sourceMax","targetMin","targetMax","MultiplyBlock","NodeMaterialModes","BoxParticleEmitter","direction1","minEmitBox","maxEmitBox","startDirectionFunction","directionToUpdate","particle","isLocal","randX","randY","randZ","startPositionFunction","positionToUpdate","newOne","applyToShader","uboOrEffect","getEffectDefines","ConeParticleEmitter","_buildHeight","_angle","directionRandomizer","radiusRange","heightRange","emitFromSpawnPointOnly","CylinderParticleEmitter","_tempVector","inverseWorldMatrix","yPos","radiusDistribution","positionRadius","xPos","zPos","CylinderDirectedParticleEmitter","HemisphericParticleEmitter","randRadius","PointParticleEmitter","SphereParticleEmitter","SphereDirectedParticleEmitter","MeshParticleEmitter","_storedNormal","useMeshNormalsForDirection","randomFaceIndex","faceIndexA","faceIndexB","faceIndexC","vertexA","vertexB","vertexC","randomVertex","BaseParticleSystem","noiseTexture","_noiseTexture","isAnimationSheetEnabled","_isAnimationSheetEnabled","_hasTargetStopDurationDependantGradient","_startSizeGradients","_emitRateGradients","_lifeTimeGradients","getDragGradients","_dragGradients","getLimitVelocityGradients","_limitVelocityGradients","getColorGradients","_colorGradients","getSizeGradients","_sizeGradients","getColorRemapGradients","_colorRemapGradients","getAlphaRemapGradients","_alphaRemapGradients","getLifeTimeGradients","getAngularSpeedGradients","_angularSpeedGradients","getVelocityGradients","_velocityGradients","getStartSizeGradients","getEmitRateGradients","particleEmitterType","isBillboardBased","_isBillboardBased","_removeGradientAndTexture","gradients","valueGradient","emitRate","manualEmitCount","updateSpeed","targetStopDuration","disposeOnStop","minEmitPower","maxEmitPower","minLifeTime","maxLifeTime","minSize","minScaleX","maxScaleX","minScaleY","maxScaleY","minInitialRotation","maxInitialRotation","minAngularSpeed","maxAngularSpeed","customShader","preventAutoStart","_rootUrl","noiseStrength","BLENDMODE_ONEONE","preWarmCycles","preWarmStepOffset","spriteCellChangeSpeed","startSpriteCellID","endSpriteCellID","spriteCellWidth","spriteCellHeight","spriteCellLoop","spriteRandomStartCell","translationPivot","beginAnimationOnStart","beginAnimationFrom","beginAnimationTo","beginAnimationLoop","worldOffset","_rampGradients","startDelay","limitVelocityDamping","color1","color2","colorDead","textureMask","_isSubEmitter","_imageProcessingConfigurationDefines","createPointEmitter","particleEmitter","createHemisphericEmitter","createSphereEmitter","createDirectedSphereEmitter","createCylinderEmitter","createDirectedCylinderEmitter","createConeEmitter","createBoxEmitter","BLENDMODE_STANDARD","BLENDMODE_ADD","BLENDMODE_MULTIPLY","BLENDMODE_MULTIPLYADD","ColorSplitterBlock","rgbIn","rgbOut","rgbOutput","rOutput","gOutput","bOutput","aOutput","ProceduralTextureSceneComponent","proceduralTextures","_beforeClear","proceduralIndex","proceduralTexture","ProceduralTexture","onGeneratedObservable","onBeforeGenerationObservable","_floats","_ints","_floatsArrays","_colors3","_colors4","_vectors2","_vectors3","_matrices","_fallbackTextureUsed","_cachedDefines","_contentUpdateId","_rtWrapper","_fullEngine","setFragment","_fallbackTexture","_createRtWrapper","_createIndexBuffer","_setEffect","getContent","_contentData","_getDefines","_fragment","_checkUniform","setFloats","viewPort","onGenerated","TrigonometryBlockOperations","TrigonometryBlock","Cos","Sin","Abs","Exp","Exp2","Round","Floor","Ceiling","Sqrt","Tan","ArcTan","ArcCos","ArcSin","Fract","Radians","Degrees","NodeMaterialDefines","VERTEXCOLOR_NME","markAsUnprocessedIfDirty","NodeMaterial","_getGlobalNodeMaterialEditor","NODEEDITOR","NodeEditor","_BuildIdGenerator","_buildWasSuccessful","_cachedWorldViewMatrix","_cachedWorldViewProjectionMatrix","_optimizers","_animationFrame","BJSNODEMATERIALEDITOR","editorData","ignoreAlpha","onBuildObservable","_vertexOutputNodes","_fragmentOutputNodes","attachedBlocks","forceAlphaBlending","getBlockByName","getBlockByPredicate","getInputBlocks","blocks","registerOptimizer","optimizer","unregisterOptimizer","addOutputNode","_addVertexOutputNode","_addFragmentOutputNode","removeOutputNode","_removeVertexOutputNode","_removeFragmentOutputNode","_sharedData","_processInitializeOnLink","nodesToProcessForOtherBuildState","_preparationId","_initializeBlock","teleport","entryPoint","_resetDualBlocks","teleportOut","removeBlock","attachedBlockIndex","updateBuildId","Particle","_vertexCompilationState","_fragmentCompilationState","vertexNodes","fragmentNodes","vertexOutputNode","fragmentOutputNode","_prepareDefinesForAttributes","oldNormal","oldTangent","oldColor","uvChanged","oldUV","createPostProcess","_createEffectForPostProcess","createEffectForPostProcess","tempName","dummyMesh","_processDefines","RegisterShader","_checkInternals","createProceduralTexture","hidden","_createEffectForParticles","particleSystemDefinesJoined","particleSystemDefines","fillDefines","createEffectForParticles","setCustomEffect","particleSystemDefinesJoinedCurrent","inputBlock","createAsShadowDepthWrapper","ShadowDepthWrapper","mergedUniforms","mergedSamplers","compiledShaders","tb","getTextureBlocks","getAllTextureBlocks","_BlockIsTextureBlock","_createNodeEditor","additionalConfig","nodeEditorConfig","Show","edit","editorUrl","editorURL","EditorURL","setToDefault","positionInput","worldInput","worldPos","viewProjectionInput","worldPosdMultipliedByViewProjection","vertexOutput","pixelColor","fragmentOutput","setToDefaultPostProcess","const1","vmerger","currentScreen","setToDefaultProceduralTexture","vectorMerger","setToDefaultParticle","rampGradient","cSplitter","blendMultiply","ParseFromFileAsync","_gatherBlocks","generateCode","vertexBlocks","outputNode","fragmentBlocks","selectedBlocks","outputNodes","_restoreConnections","outputPoint","candidate","inputPoint","parseSerializedObject","parsedBlock","_tempEntryPointUniqueId","attachToEndpoint","blockIndex","outputNodeId","locations","blockId","blockMap","comment","loadFromSerialization","shareEffect","skipBuild","CreateDefault","ControllerOrbAnimationState","WebXRNearControllerMode","IgnoreTexturesAtLoadTime","_projectOnTrianglesToRef","tmpDist","ProjectOnTriangleToRef","_projectOnUnIndexedTrianglesToRef","projectToRef","WebXRNearInteraction","touchCollisionMesh","touchCollisionMeshFunction","hydrateCollisionMeshFunction","_generateNewTouchPointMesh","_generateVisualCue","nearInteractionTargetMesh","stalePick","currentAnimationState","DEHYDRATED","grabRay","hoverInteraction","nearInteraction","grabInteraction","pickedPointVisualCue","enableNearInteractionOnAllControllers","_attachNearInteractionMode","_farInteractionFeature","_hoverRadius","_pickRadius","_controllerPickRadius","_nearGrabLengthScale","nearInteractionControllerMode","CENTERED_IN_FRONT","farInteractionFeature","setFarInteractionFeature","_nearPickPredicate","_nearGrabPredicate","_nearInteractionPredicate","_controllerAvailablePredicate","excludedControllerId","_handleTransitionAnimation","HOVER","TOUCH","_processTouchPoint","handData","xrIndexTip","indexTipPose","getJointPose","axisRHSMultiplier","DISABLED","controllerPose","CENTERED_ON_CONTROLLER","accuratePickInfo","populateNearInteractionInfo","nearInteractionInfo","nearInteractionAtOrigin","utilitySceneHoverPick","_pickWithSphere","hoverPickInfo","utilitySceneNearPick","nearPick","_isControllerReadyForNearInteraction","grabCheck","squeezeComponent","onSqueezeButtonChangedObserver","meshCreationScene","motionControllerOrbMaterial","hoverSizeVec","touchSize","touchSizeVec","hydrateTransitionSize","hydrateTransitionSizeVec","touchHoverTransitionSize","touchHoverTransitionSizeVec","hoverTouchTransitionSize","touchKeys","releaseKeys","hydrateKeys","dehydrateKeys","releaseAction","hydrateAction","dehydrateAction","isHydration","sceneToUse","CreateFromCenterAndRadius","PickMeshWithSphere","tmpVec","tmpDistanceSphereToCenter","tmpDistanceSurfaceToCenter","worldToMesh","WebXREnterExitUIButton","activeButton","WebXREnterExitUI","_activeButton","activeButtonChangedObservable","_onSessionGranted","_helper","_enterXRWithButtonIndex","overlay","classList","ignoreSessionGrantedEvent","hostname","customButtons","hmdBtn","_updateButtons","renderCanvas","supportedPromises","onclick","ui","setHelperAsync","prevTitle","InstancedMesh","instancedBuffers","_sourceMesh","_billboardWorldMatrix","tempMaster","sourceMeshLODLevels","newSourceMesh","registerInstancedBuffer","expectedSize","ShaderMaterial","shaderPath","_textureArrays","_externalTextures","_uints","_colors3Arrays","_colors4Arrays","_vectors4","_quaternions","_quaternionsArrays","_matrixArrays","_matrices3x3","_matrices2x2","_vectors2Arrays","_vectors3Arrays","_vectors4Arrays","_textureSamplers","_multiview","_materialHelperNeedsPreviousMatrices","_shaderPath","externalTextures","samplerObjects","storageBuffers","setExternalTexture","setColor3Array","setColor4Array","setQuaternionArray","float32Array","setUniformBuffer","setTextureSampler","sampler","setStorageBuffer","setDefine","defineName","trimEnd","existingDefineIdx","defineToAdd","bvaManager","previousDefines","effectOverride","useSceneUBO","propName","propValue","textureArrays","ints","uints","floats","FloatArrays","colors3","colors3Arrays","colors4Arrays","vectors2","vectors3","vectors4","matrixArray","matrices3x3","matrices2x2","vectors2Arrays","vectors3Arrays","vectors4Arrays","quaternionsArrays","floatsArrays","shaderMaterial","LinesMesh","_isShaderMaterial","_color4","_lineMaterial","_subMesh","colorEffect","doNotDisposeMaterial","InstancedLinesMesh","CreateLineSystemVertexData","vertexColors","CreateDashedLinesVertexData","curvect","lg","shft","dashshft","curshft","CreateLineSystem","vertexColor","lineColors","lineSystem","CreateLines","CreateDashedLines","nbSeg","dashedLines","TimerState","setAndStartTimer","observableParameters","contextObservable","payload","completeRate","onTick","breakCondition","onAborted","onEnded","WebXRMotionControllerTeleportation","rotationEnabled","_rotationEnabled","teleportationTargetMesh","_snappedToPoint","_cachedColor4White","skipNextTeleportation","backwardsMovementEnabled","backwardsTeleportationDistance","parabolicCheckRadius","parabolicRayEnabled","straightRayEnabled","onTargetMeshPositionUpdatedObservable","forceHandedness","teleportationState","backwards","rotating","baseRotation","blocked","initMotionController","movementController","useMainComponentOnly","mainComponent","teleportationComponent","_currentTeleportationControllerId","timeToTeleport","_teleportForward","onAxisChangedObserver","axesData","snapPointsOnly","_floorMeshes","_setTargetMeshVisibility","_createDefaultTargetMesh","_snapToPositions","snapPositions","_blockedRayColor","blockedRayColor","snapToPoints","addBlockerMesh","pickBlockerMeshes","addSnapPoint","newSnapPoint","removeBlockerMesh","removeFloorMeshByName","removeSnapPoint","snapPointToRemove","setSelectionFeature","selectionFeature","_selectionFeature","targetMesh","hitPossible","_showParabolicPath","_setTargetMeshPosition","xRotation","compensation","_disposeBezierCurve","defaultTargetMeshOptions","centerX","centerY","teleportationFillColor","teleportationBorderColor","cone","torusArrowMaterial","torusConeMaterial","_teleportationRingMaterial","_findClosestSnapPointWithRadius","realPosition","snapToPositionRadius","closestPoint","closestDistance","radiusSquared","snapPosition","_quadraticBezierCurve","quadraticBezierVectors","CreateQuadraticBezier","colorsArray","generateRayPathMesh","WebXRDefaultExperience","disableDefaultUI","uiOptions","enterExitUI","CreateAsync","xrHelper","ignoreNativeCameraTransformation","inputOptions","disablePointerSelection","pointerSelectionOptions","useStablePlugins","disableTeleportation","teleportationOptions","disableNearInteraction","nearInteractionOptions","outputCanvasOptions","createDefaultLight","createDefaultCamera","createArcRotateCamera","attachCameraControls","worldExtends","worldSize","worldCenter","arcRotateCamera","freeCamera","createDefaultCameraOrLight","createDefaultSkybox","pbr","blur","setGlobalEnvTexture","hdrSkybox","hdrSkyboxMaterial","createDefaultEnvironment","createDefaultVRExperience","EffectLayer","_effectLayerOptions","mainTexture","_mainTexture","currentMesh","_materialForRendering","getEffectIntensity","_effectIntensity","setEffectIntensity","_maxSize","_mainTextureDesiredSize","_emissiveTextureAndColor","neutralColor","disableBoundingBoxesFromEffectLayer","onBeforeRenderMainTextureObservable","onBeforeComposeObservable","onBeforeRenderMeshToEffect","onAfterRenderMeshToEffect","onAfterComposeObservable","_SceneComponentInitialization","effectLayers","_mergeDrawWrapper","_generateIndexBuffer","_generateVertexBuffer","_numInternalDraws","mainTextureRatio","alphaBlendingMode","mainTextureType","_setMainTextureSize","_createMainTexture","_createTextureAndPostProcesses","mainTextureFixedSize","preWarm","_setEmissiveTextureAndColor","opaqueSubMeshes","alphaTestSubMeshes","transparentSubMeshes","depthOnlySubMeshes","_renderSubMesh","previousAlphaMode","getBoundingBoxRenderer","boundingBoxRendererEnabled","_addCustomEffectDefines","emissiveTexture","renderingMaterial","_useMeshMaterial","needAlphaTest","needAlphaBlendFromDiffuse","useAlphaFromDiffuseTexture","opacityTexture","morphInfluencers","PrepareAttributesForMorphTargetsInfluencers","cachedDefines","numDraws","_createMergeEffect","_internalRender","_disposeTextureAndPostProcesses","hasMesh","_shouldRenderMesh","_canRenderMesh","_shouldRenderEmissiveTextureForMesh","ownerMesh","textureMatrix","parsedEffectLayer","serializeAsCameraReference","AddParser","parsedData","effectLayer","removeEffectLayer","addEffectLayer","newEffectLayer","EffectLayerSceneComponent","_renderEffects","_needStencil","_previousStencilState","_isReadyForMesh","_renderMainTexture","_setStencil","_drawRenderingGroup","_setStencilBack","_drawCamera","removeFromContainer","needStencil","ShadowLight","_needProjectionMatrixCompute","_setPosition","_setDirection","shadowMinZ","_shadowMinZ","forceProjectionMatrixCompute","shadowMaxZ","_shadowMaxZ","computeTransformedInformation","transformedPosition","getDepthScale","getShadowDirection","RotationFromAxis","needCube","needProjectionMatrixCompute","getDepthMinZ","getDepthMaxZ","setShadowProjectionMatrix","viewMatrix","customProjectionMatrixBuilder","_setDefaultShadowProjectionMatrix","ShadowGenerator","_bias","normalBias","_normalBias","blurBoxOffset","_blurBoxOffset","_disposeBlurPostProcesses","blurScale","_blurScale","_blurKernel","useKernelBlur","_useKernelBlur","depthScale","_depthScale","_light","_validateFilter","_filter","FILTER_BLUREXPONENTIALSHADOWMAP","useExponentialShadowMap","FILTER_BLURCLOSEEXPONENTIALSHADOWMAP","useCloseExponentialShadowMap","FILTER_PCF","FILTER_PCSS","usePoissonSampling","_applyFilterValues","FILTER_POISSONSAMPLING","FILTER_NONE","FILTER_EXPONENTIALSHADOWMAP","useBlurExponentialShadowMap","FILTER_CLOSEEXPONENTIALSHADOWMAP","useBlurCloseExponentialShadowMap","usePercentageCloserFiltering","filteringQuality","_filteringQuality","useContactHardeningShadow","contactHardeningLightSizeUVRatio","_contactHardeningLightSizeUVRatio","darkness","_darkness","setDarkness","getDarkness","transparencyShadow","_transparencyShadow","setTransparencyShadow","_shadowMap","getShadowMapForRendering","_shadowMap2","CLASSNAME","addShadowCaster","removeShadowCaster","getLight","mapSize","_mapSize","recreateShadowMap","usefullFloatFirst","useRedTextureType","onBeforeShadowMapRenderObservable","onAfterShadowMapRenderObservable","onBeforeShadowMapRenderMeshObservable","onAfterShadowMapRenderMeshObservable","QUALITY_HIGH","enableSoftTransparentShadow","useOpacityTextureForTransparentShadow","frustumEdgeFalloff","forceBackFacesOnly","_lightDirection","_cachedPosition","_cachedDirection","_currentFaceIndex","_currentFaceIndexCache","_defaultTextureMatrix","_useRedTextureType","shadowGenerators","_sceneUBOs","_initializeGenerator","_initializeShadowMap","_createTargetRenderTexture","noPrePassRenderer","_storedUniqueId","_renderForShadowMap","_blurPostProcesses","clearZero","clearOne","_initializeBlurRTTAndPostProcesses","targetSize","_kernelBlurXPostprocess","_kernelBlurYPostprocess","_boxBlurPostprocess","_renderSubMeshForShadowMap","_bindCustomEffectForRenderSubMeshForShadowMap","isTransparent","detNeg","reverseSideOrientation","customAllowRendering","GetEffect","standalone","baseMaterial","worldOverride","_isReadyCustomDefines","_prepareShadowDefines","subMeshEffect","DEFAULT_ALPHA_CUTOFF","customShaderOptions","attrib","QUALITY_LOW","QUALITY_MEDIUM","lightPosition","_disposeRTTandPostProcesses","_disposeSceneUBOs","lightId","parsedShadowGenerator","constr","useVarianceShadowMap","useBlurVarianceShadowMap","DepthRenderer","_depthMap","storeNonLinearDepth","storeCameraSpaceZ","forceDepthWriteTransparentMeshes","useOnlyInActiveCamera","reverseCulling","_storeNonLinearDepth","_storeCameraSpaceZ","isPacked","renderSubMesh","cameraIsOrtho","numMorphInfluencers","getDepthMap","keysToDelete","_depthRenderer","DepthReducer","MinMaxReducer","onAfterReductionPerformed","_forceFullscreenViewport","_activated","_sourceTexture","setSourceTexture","depthRedux","_reductionSteps","reductionInitial","reduction","minmax","activated","deactivate","disposeAll","depthRenderer","setDepthRenderer","_depthRendererId","UpDir","ZeroVec","tmpv1","tmpv2","CascadedShadowGenerator","numCascades","_numCascades","MIN_CASCADES_COUNT","MAX_CASCADES_COUNT","_recreateSceneUBOs","freezeShadowCastersBoundingInfo","_freezeShadowCastersBoundingInfo","_freezeShadowCastersBoundingInfoObservable","_computeShadowCastersBoundingInfo","_scbiMin","_scbiMax","MIN_VALUE","_shadowCastersBoundingInfo","shadowCastersBoundingInfo","setMinMaxDistance","_minDistance","_maxDistance","_breaksAreDirty","minDistance","maxDistance","getCascadeMinExtents","cascadeIndex","_cascadeMinExtents","getCascadeMaxExtents","_cascadeMaxExtents","debug","_debug","dbg","depthClamp","_depthClamp","cascadeBlendPercentage","_cascadeBlendPercentage","lambda","_lambda","getCascadeViewMatrix","cascadeNum","_viewMatrices","getCascadeProjectionMatrix","_projectionMatrices","getCascadeTransformMatrix","_transformMatrices","_depthReducer","autoCalcDepthBounds","_autoCalcDepthBounds","autoCalcDepthBoundsRefreshRate","splitFrustum","_splitFrustum","near","far","cameraRange","_cascades","prevBreakDistance","breakDistance","_viewSpaceFrustumsZ","_frustumLengths","_computeMatrices","_computeFrustumInWorldSpace","_computeCascadeFrustum","_frustumCenter","_shadowCameraPos","_transformMatricesAsArray","prevSplitDist","splitDist","cameraInfiniteFarPlane","saveCameraMaxZ","invViewProj","cornerIndexOffset","cornerIndex","_FrustumCornersNDCSpace","_frustumCornersWorldSpace","stabilizeCascades","lightCameraPos","usefulFloatFirst","_o","_p","_q","_r","_s","_t","_u","_v","penumbraDarkness","DEFAULT_CASCADES_COUNT","_currentLayer","_lightSizeUVCorrection","_depthCorrection","ShadowGeneratorSceneComponent","_gatherRenderTargets","DirectionalLight","shadowFrustumSize","_shadowFrustumSize","shadowOrthoScale","_shadowOrthoScale","autoUpdateExtends","autoCalcShadowZBounds","_setDefaultFixedFrustumShadowProjectionMatrix","_setDefaultAutoExtendShadowProjectionMatrix","PointLight","shadowAngle","_shadowAngle","previousNeedCube","SpotLight","_cosHalfAngle","_projectionTextureProjectionLightDirty","_computeAngleValues","_innerAngle","shadowAngleScale","_shadowAngleScale","projectionTextureMatrix","_projectionTextureMatrix","projectionTextureLightNear","_projectionTextureLightNear","projectionTextureLightFar","_projectionTextureLightFar","projectionTextureUpDirection","_projectionTextureUpDirection","projectionTexture","_projectionTexture","_projectionTextureDirty","_IsProceduralTexture","_IsTexture","projectionTextureProjectionLightMatrix","_projectionTextureProjectionLightMatrix","_projectionTextureViewLightDirty","_projectionTextureViewTargetVector","_projectionTextureViewLightMatrix","_projectionTextureScalingMatrix","_computeProjectionTextureViewLightMatrix","_computeProjectionTextureProjectionLightMatrix","lightFar","lightNear","P","Q","S","_computeProjectionTextureMatrix","_lightAngleScale","_lightAngleOffset","SceneOptimization","getDescription","TextureOptimization","maximumSize","currentSize","HardwareScalingOptimization","_currentScale","maximumScale","_directionOffset","ShadowsOptimization","isInImprovementMode","PostProcessesOptimization","LensFlaresOptimization","CustomOptimization","onGetDescription","ParticlesOptimization","RenderTargetsOptimization","MergeMeshesOptimization","_canBeMerged","UpdateSelectionTree","_UpdateSelectionTree","updateSelectionTree","globalPool","globalLength","currentPool","MergeMeshes","sceneAsAny","createOrUpdateSelectionOctree","SceneOptimizerOptions","targetFrameRate","trackerDuration","optimizations","addOptimization","optimization","addCustomOptimization","SceneOptimizer","_improvementMode","currentPriorityLevel","_currentPriorityLevel","_currentFrameRate","_targetFrameRate","_trackerDuration","autoGeneratePriorities","improvementMode","_isRunning","onSuccessObservable","onNewOptimizationAppliedObservable","onFailureObservable","optim","_checkCurrentState","noOptimizationApplied","onFailure","ModerateDegradationAllowed","serializedGeometries","SerializeGeometry","serializationGeometries","SerializeMesh","serializationScene","SceneSerializer","_Serialize","checkSyncReadSupported","isPhysicsEnabled","physicEngine","physicsGravity","getPhysicsPluginName","activeCameraID","animationGroupIndex","reflectionProbe","environmentTextureRotationY","boxes","spheres","cylinders","toruses","grounds","planes","torusKnots","_CollectPromises","toSerialize","withParents","withChildren","serializeMaterial","submaterial","FinalizeSingleNode","enableDepthRenderer","force32bitsFloat","supportFullfloat","disableDepthRenderer","DepthRendererSceneComponent","_gatherActiveCameraRenderTargets","EdgesRenderer","LineEdgesRenderer","FaceAdjacencies","edgesConnectedCount","linesPositions","_linesPositions","linesNormals","_linesNormals","_linesIndices","lineShader","_lineShader","_edgeRenderLineShader","generateEdgesLines","edgesWidthScalerForOrthographic","edgesWidthScalerForPerspective","_buffersForInstances","_checkVerticesInsteadOfIndices","_epsilon","_prepareRessources","useAlternateEdgeFinder","_generateEdgesLinesAlternate","_generateEdgesLines","_meshRebuildObserver","_meshDisposeObserver","_GetShader","_ib","_processEdgeForAdjacencies","pa","pb","_processEdgeForAdjacenciesWithVertices","eps","_checkEdge","faceNormals","needToCreateLine","createLine","_tessellateTriangle","edgePoints","indexTriangle","remapVertexIndices","makePointList","pointIndices","startEdge","mainPointIndices","otherPointIndices","numMainPoints","numOtherPoints","bucketIsMain","bucketStep","bucketLimit","bucketIdxLimit","winding","numTris","bucketIdx","nbucketIdx","bucketPoints","nbucketPoints","bucket","useFastVertexMerger","epsVertexMerge","epsilonVertexMerge","mapVertices","applyTessellation","epsVertexAligned","epsilonVertexAligned","mustTesselate","triangleToTessellate","p0Index","p1Index","p2Index","p0x","p0y","p0z","p1x","p1y","p1z","p0p1","vIndex","p0p","pp1","edgesPoints","triangle","faceNormal","removeDegeneratedTriangles","ei","_indicesCount","adjacencies","faceAdjacencies","otherIndex","otherFaceAdjacencies","otherP0","otherP1","otherP2","edgeIndex","otherEdgeIndex","currentDrawWrapper","useBuffersWithInstances","thinInstanceCount","restoreSingleAttachment","bindAttachments","restoreSingleAttachmentForRenderTarget","buildTextureLayout","textureStatus","attachments","readBuffer","useSRGBBuffers","useStencilTexture","depthStencilBuffer","layerCount","isWebGL2","MultiRenderTarget","internalSizedFormat","webGLTextureType","depthTexture","depthTextureType","glDepthTextureInternalFormat","glDepthTextureFormat","glDepthTextureType","glDepthTextureAttachment","useDepthStencil","textureNames","drawOnlyOnFirstAttachmentByDefault","_textureNames","_initTypes","_multiRenderTargetOptions","_drawOnlyOnFirstAttachmentByDefault","_createInternalTextures","_createTextures","defaultType","_createInternaTextureIndexMapping","mapMainInternalTexture2Index","mapInternalTexture2MainIndex","internalTextures","mainIndex","forceFullRebuild","releaseInternalTextures","_releaseTextures","setInternalTexture","updateCount","doNotDisposeInternalTextures","GeometryBufferRenderer","_linkPrePassRenderer","_linkedWithPrePass","_prePassRenderer","_multiRenderTarget","_unlinkPrePassRenderer","_createRenderTargets","_resetLayout","_enablePosition","_enableReflectivity","_enableVelocity","_attachmentsFromPrePass","_forceTextureType","geometryBufferType","POSITION_TEXTURE_TYPE","_positionIndex","VELOCITY_TEXTURE_TYPE","_velocityIndex","REFLECTIVITY_TEXTURE_TYPE","_reflectivityIndex","DEPTH_TEXTURE_TYPE","_depthIndex","NORMAL_TEXTURE_TYPE","_normalIndex","_setAttachments","_linkInternalTexture","getTextureIndex","enablePosition","enableVelocity","_previousTransformationMatrices","enableReflectivity","_ratio","_previousBonesTransformationMatrices","excludedSkinnedMeshesFromVelocity","renderTransparentMeshes","_clearColor","_clearDepthColor","useSpecificClearForDepthTexture","_useUbo","_depthFormat","needUv","metallicWorkflow","metallicRoughnessTexture","baseColor","specularGlossinessTexture","glossiness","metallicTexture","albedoTexture","reflectivityTexture","specularTexture","buffersCount","getGBuffer","_assignRenderTargetIndices","layoutAttachmentsAll","layoutAttachmentsAllButDepth","layoutAttachmentsDepthOnly","attachmentsAll","attachmentsAllButDepth","attachmentsDepthOnly","viewProjection","bonesTransformations","_copyBonesTransformationMatrices","_geometryBufferRenderer","enableGeometryBufferRenderer","disableGeometryBufferRenderer","GeometryBufferRendererSceneComponent","getOutlineRenderer","_outlineRenderer","OutlineRenderer","_renderOutline","_renderOverlay","_passIdForDrawWrapper","_beforeRenderingMesh","_afterRenderingMesh","useOverlay","_savedDepthWrite","renderOutline","_StencilReference","currentMode","alphaBlendState","PrePassRenderTarget","_beforeCompositionPostProcesses","_internalTextureDirty","_createCompositionEffect","imageProcessingPostProcess","_checkSize","_resetPostProcessChain","_outputPostProcess","PrePassRenderer","_textureIndices","_useSpecificClearForDepthTexture","getRenderTarget","_setRenderTarget","prePassRenderTarget","_refreshGeometryBufferRendererLink","doNotUseGeometryRendererFallback","_geometryBuffer","excludedMaterials","_mrtTypes","_mrtFormats","_mrtLayout","_mrtNames","_effectConfigurations","_needsCompositionForThisPass","disableGammaTransform","TextureFormats","_createRenderTarget","rt","bindAttachmentsForEffect","excluded","_multiRenderAttachments","_defaultAttachments","_reinitializeAttachments","multiRenderLayout","clearLayout","clearDepthLayout","defaultLayout","_clearAttachments","_clearDepthAttachments","purpose","_updateGeometryBufferLayout","texturesActivated","prePassConstant","geometryBufferConstant","restoreAttachments","_beforeDraw","_setupOutputForThisPass","_postProcessesSourceForThisPass","setCustomOutput","_renderPostProcesses","postProcessChain","_afterDraw","_setEnabled","_setRenderTargetEnabled","_unlinkInternalTexture","cfg","previousMrtCount","_enableTextures","texturesRequired","_disable","_getPostProcessesSource","secondaryCamera","cameraHasImageProcessing","_hasImageProcessing","_needsImageProcessing","firstCameraPP","firstPrePassPP","needsImageProcessing","isIPPAlreadyPresent","enablePrePass","_depthPeelingRenderer","_markAllMaterialsAsPrePassDirty","disablePrePassRenderer","PrePassRendererSceneComponent","_beforeCameraDraw","_afterCameraDraw","_beforeRenderTargetDraw","_afterRenderTargetDraw","settingDefaults","parentChangeBehavior","PRESERVE_POSITION","settingsKey","savedSettings","localStorage","getItem","settings","SettingsManagerService","setItem","CreateRibbonVertexData","invertUV","defaultOffset","customUV","customColors","us","vs","uTotalDistance","vTotalDistance","minlg","ar1","ar2","idc","closePathCorr","vectlg","path1","path2","l1","path1nb","indexFirst","indexLast","positions32","normals32","uvs32","_idx","CreateRibbon","ns","si","pathPoint","colorIndex","ribbon","CreateDiscVertexData","vertexNb","CreateDisc","disc","CreateTiledPlaneVertexData","flipTile","pattern","tileWidth","tileHeight","alignH","alignHorizontal","alignV","alignVertical","tilesX","tilesY","adjustX","adjustY","startX","startY","uvBase","partialBottomRow","partialTopRow","partialLeftCol","partialRightCol","uvPart","uvBaseBR","uvBaseTR","uvBaseLC","uvBaseRC","CreateTiledBoxVertexData","halfDepth","faceVertexData","baseAlignV","facePositions","newFaceUV","vec0","mtrx0","mtrx2","mtrx3","vec4","mtrx4","RotationX","mtrx5","CreateTorusKnotVertexData","getPos","cu","su","quOverP","tang","bitan","jNext","CreateTorusKnot","torusKnot","CreateTiledPlane","CreateTiledBox","IndexedVector2","original","PolygonPoints","originalPoints","computeBounds","lmin","lmax","PolygonMeshBuilder","_addToepoint","_epoints","contours","earcut","_outlinepoints","_holes","_eholes","bjsEarcut","addHole","hole","holepoints","smoothingThreshold","buildVertexData","bounds","positionscount","totalCount","_addSide","ulength","vc","vp","vn","vc_norm","vp_norm","vn_norm","dotp","dotn","CreatePolygonVertexData","wrp","disp","distZ","totalLen","cumulate","CreatePolygon","polygonTriangulation","hNb","hPoint","ExtrudePolygon","ExtrudeShape","closeShape","_ExtrudeShapeGeneric","adjustFrame","ExtrudeShapeCustom","rotateFunction","rbCA","rbCP","custom","updtbl","extrusionPathArray","path3D","shapePaths","binormals","scl","shapePath","scaleRatio","planed","rotated","capPath","pointCap","barycenter","storage","extrudedGeneric","CreateLathe","clip","pi2","CreateTube","tubePathArray","circlePaths","radiusFunctionFinal","circlePath","pathIndex","CreatePolyhedronVertexData","polyhedra","sizeX","sizeY","sizeZ","nbfaces","faceIdx","indexes","ang","fl","CreatePolyhedron","polyhedron","PolyhedronBuilder","CreateIcoSphereVertexData","radiusX","radiusY","radiusZ","icoVertices","ico_indices","vertices_unalias_id","ico_vertexuv","island","current_indice","face_vertex_pos","face_vertex_uv","v012","v_id","interp_vertex","c2","pos_x0","pos_x1","pos_interp","vertex_normal","centroid_x0","centroid_x1","uv_x0","uv_x1","uv_interp","CreateIcoSphere","xpAxis","xnAxis","ypAxis","ynAxis","zpAxis","znAxis","DecalVertex","vertexIdx","vertexIdxForBones","localPositionOverride","localNormalOverride","matrixIndicesOverride","matrixWeightsOverride","CreateDecal","hasSkeleton","useLocalComputation","localMode","meshHasOverridenMaterial","localPositions","localNormals","matIndices","matWeights","matIndicesExtra","matWeightsExtra","cameraWorldTarget","currentVertexDataIndex","extractDecalVector3","indexId","vertexId","captureUVS","emptyArray","clipSize","clipVertices","clipFactor","GetClipFactor","mat0Index","v0Indices","v0Weights","mat1Index","v1Indices","v1Weights","sumw","v0LocalPositionX","v0LocalPositionY","v0LocalPositionZ","v1LocalPositionX","v1LocalPositionY","v1LocalPositionZ","v0LocalNormalX","v0LocalNormalY","v0LocalNormalZ","interpNormalX","interpNormalY","interpNormalZ","clipResult","nV1","nV2","nV3","nV4","v1Out","v2Out","v3Out","sourceMeshAsMesh","numMatrices","thinInstanceMatrix","ofst","decalWorldMatrix","inverseDecalWorldMatrix","meshWorldMatrix","oneFaceVertices","faceVertices","decal","CreateCapsuleVertexData","capDetail","heightSegments","heightMinusCaps","thetaLength","capsTopSegments","topCapSubdivisions","capsBottomSegments","bottomCapSubdivisions","indexArray","cosAlpha","sinAlpha","coneLength","vl","indexRow","sinA","sinTheta","cosTheta","coneHeight","i4","vDat","CreateCapsule","capsule","_IsoVector","rotate60About","rotateNeg60About","rotate120","rotateNeg120","toCartesianOrigin","isoGridSize","_PrimaryIsoTriangle","cartesian","closestTo","innerFacets","isoVecsABOB","isoVecsOBOA","isoVecsBAOA","vertexTypes","IDATA","PolyhedronData","vecToidx","fr","O","n1","oVec","aVec","bVec","oaVec","abVec","obVec","idxR","isoId","isoIdR","verts","vDist","vertByDist","matchIdx","edgematch","calcCoeffs","thirdR3","LSQD","coau","cobu","coav","cobv","createInnerFacets","edgeVecsABOB","pointR","prevR","nextR","maxPrev","maxLeftPrev","mapABOBtoOBOA","mapABOBtoBAOA","MapToFace","faceNb","geodesicData","F","oidx","aidx","bidx","OA","OB","mapped","tempVec","distFrom","vert","primVert","distFromO","distFromA","distFromB","vertData","GeodesicData","innerToData","primTri","mapABOBtoDATA","mapOBOAtoDATA","mapBAOAtoDATA","orderData","nearTo","nearIndex","sharedNodes","poleNodes","setOrder","adjVerts","dualFaces","adjacentFaces","toGoldbergPolyhedronData","goldbergPolyhedronData","verticesNb","cz","GoldbergMesh","goldbergData","faceCenters","faceZaxis","faceXaxis","faceYaxis","nbSharedFaces","nbUnsharedFaces","nbFacesAtPole","relatedGoldbergFace","poleOrShared","fromPole","_changeGoldbergFaceColors","colorRange","newCols","setGoldbergFaceColors","updateGoldbergFaceColors","_changeGoldbergFaceUVs","uvRange","points5","points6","setGoldbergFaceUVs","newUVs","updateGoldbergFaceUVs","placeOnGoldbergFaceAt","goldberg","ShapePath","resolution","_paths","_tempPaths","_resolution","moveTo","_currentPath","lineTo","quadraticCurveTo","cpx","cpy","bezierCurveTo","cpx1","cpy1","cpx2","cpy2","extractHoles","CreateShapePath","char","fontData","glyph","glyphs","outline","ha","CreateTextShapePaths","chars","line_height","yMax","yMin","underlineThickness","MeshBuilder","CreateGeodesic","BuildGeodesicData","CreateGoldberg","ba","ca","pdata","vCoord","CreateGoldbergVertexData","CreateText","holeVectors","shapeVectors","localHolesCopy","holePoints","bbox","GuiHelper","highlighter","textLabelMap","gridSettings","sceneNode","skipOrigin","gridMesh","bgColor","bgColor3","strongLineColor","FromColor3","lightLineColor","calculateGridPoints","pointColors","linesPerSide","disableGrid","gridAxesMesh","Green","Blue","pointsAxes","gridMeshAxes","createTextLabels","disableGridAxes","planeHeight","textureHeight","tmpctx","textureWidth","planeWidth","stepSize","gridCount","gridFullSize","wV1","wV2","vStep","uVec","vVec","nodeToScreen","bNode","toScreen","toNDC","Project","getScreenBoundingBox","boundingVectorsToScreen","xs","ys","minx","maxx","miny","canvasWidth","canvasHeight","updateMatrixOfAncestorChain","eWorkerTasks","eKB3DMessageType","global","idealSize","numerator","denominator","dataURL","tagName","videoWidth","videoHeight","ImageData","Kb3dWorker","toggle","requestsInFlight","workerPath","workerReturnMessage","taskId","taskComplete","taskError","newWorker","startTask","task","taskPayload","transfer","messagePayload","taskStart","_Group","_tweens","_tweensAddedDuringUpdate","tweenId","removeAll","tween","getId","preserve","tweenIds","TWEEN","_isPlaying","Group","_nextId","nextId","hrtime","getTime","Tween","_isPaused","_pauseStart","_object","_valuesStart","_valuesEnd","_valuesStartRepeat","_duration","_repeat","_repeatDelayTime","_yoyo","_reversed","_delayTime","_startTime","Easing","Linear","_interpolationFunction","Interpolation","_chainedTweens","_onStartCallback","_onStartCallbackFired","_onUpdateCallback","_onRepeatCallback","_onCompleteCallback","_onStopCallback","_group","isPlaying","isPaused","stopChainedTweens","numChainedTweens","repeat","times","repeatDelay","yoyo","easing","interpolationFunction","chain","onStart","onUpdate","onRepeat","onStop","elapsed","Quadratic","In","Out","InOut","Cubic","Quartic","Quintic","Sinusoidal","Exponential","Circular","Elastic","Bounce","Utils","Bezier","pw","bn","Bernstein","fc","Factorial","AnimationModule","kbViewer","activeAnimations","animationsAreRunning","applyDefaults","origValue","origValueType","valueIn","toVal","getTweenEasingFn","valueOut","activeAnim","animationWarningsNeedsUpdate","anims","stopAll","linear","From","inOut","anchorIdProvider","WebXRAnchorSystem","referenceSpaceForFrameAnchors","_referenceSpaceForFrameAnchors","_lastFrameDetected","_trackedAnchors","_futureAnchors","onAnchorAddedObservable","onAnchorRemovedObservable","onAnchorUpdatedObservable","_populateTmpTransformation","hitTestResult","xrHitResult","createAnchor","nativeAnchor","resolved","submitted","xrTransformation","forceCreateInCurrentFrame","xrAnchor","_createAnchorAtTransformation","anchors","doNotRemoveAnchorsOnSessionEnded","trackedAnchors","anchor","idxTracker","_findIndexInAnchorArray","_updateAnchorWithXRFrame","attachedNode","transformationMatrix","newAnchor","futureAnchor","anchorSpace","worldParentNode","WebXRHitTest","_tmpMat","_tmpPos","_tmpQuat","_initHitTestSource","offsetRay","XRRay","hitTestOptions","useReferenceSpace","entityTypes","requestHitTestSource","hitTestSource","_xrHitTestSource","cancel","autoCloneTransformation","onHitTestResultObservable","paused","disablePermanentHitTest","enableTransientHitTest","transientOffsetRay","requestHitTestSourceForTransientInput","transientHitTestProfile","hitSource","_transientXrHitTestSource","getHitTestResults","_processWebXRHitTestResult","getHitTestResultsForTransientInput","resultsPerInputSource","hitTestResults","isTransient","planeIdProvider","WebXRPlaneDetector","_detectedPlanes","onPlaneAddedObservable","onPlaneRemovedObservable","onPlaneUpdatedObservable","doNotRemovePlanesOnSessionEnded","XRPlane","detectedPlanes","worldInformation","planeIdx","xrPlane","lastChangedTime","_findIndexInPlaneArray","_updatePlaneWithXRPlane","newPlane","polygonDefinition","internalInit","preferredDetectorOptions","trySetPreferredPlaneDetectorOptions","updateWorldTrackingState","planeDetectionState","xrPoint","planeSpace","MorphTarget","_influence","onInfluenceChanged","_onDataLayoutChanged","hasPositions","hasTangents","hadPositions","hadNormals","hadTangents","hadUVs","_GLTFUtilities","bufferIndex","bufferview","bufferviewIndex","accessor","vertexStart","accessorType","_TangentType","_GLTFAnimation","babylonNode","babylonTransformNode","animationChannelTargetPath","useQuaternion","animationSampleRate","_IsTransformable","keyFrames","minMaxKeyFrames","_CalculateMinMaxKeyFrames","interpolationOrBake","_DeduceInterpolation","interpolationType","shouldBakeAnimation","_CreateBakedAnimation","_CreateLinearOrStepAnimation","_CreateCubicSplineAnimation","samplerInterpolation","inputsMin","inputsMax","dataAccessorType","runtimeGLTFAnimation","idleGLTFAnimations","nodeMap","binaryWriter","bufferViews","accessors","shouldExportAnimation","glTFAnimation","animationInfo","_DeduceAnimationInfo","channels","_AddAnimation","combinedAnimation","combinedAnimationKeys","animationKeys","babylonScene","glTFAnimations","morphAnimations","sampleAnimations","morphAnimationMeshes","animationGroupFrameDiff","targetedAnimations","targetAnimation","babylonMorphTarget","babylonMorphTargetManager","babylonMesh","combinedAnimationGroup","sampleAnimationKeys","numAnimationKeys","animationsByMorphTarget","morphTargetAnimation","morphAnimationChannels","animationData","_CreateNodeAnimation","keyframeAccessorIndex","dataAccessorIndex","outputLength","animationSampler","animationChannel","currentInput","newInputs","nodeIndex","_CreateBufferView","getByteOffset","_CreateAccessor","_GetDataAccessorElementCount","minFrame","maxFrame","sampleRate","minMaxFrames","quaternionCache","previousTime","maxUsedFrame","currKeyFrame","nextKeyFrame","prevKeyFrame","_SetInterpolatedValue","basePositionRotationOrScale","_GetBasePositionRotationOrScale","componentName","cacheValue","_ConvertFactorToVector3OrQuaternion","keyFrame","_AddKeyframeValue","_AddSplineTangent","INTANGENT","OUTTANGENT","newPositionRotationOrScale","posRotScale","tangentType","tangentValue","GLTFData","glTFFiles","downloadFiles","link","_convertRGBtoRGBATextureData","rgbData","rgbaData","_makeCreateRawTextureFunction","updateRawTexture3D","updateRawTexture2DArray","_makeUpdateRawTextureFunction","internalType","internalSizedFomat","compressedTexImage3D","updateRawTexture","updateRawCubeTexture","needConversion","createRawCubeTextureFromUrl","mipmapGenerator","internalCallback","faceDataArrays","mipData","mipSize","mipFaceData","RawTexture","Constants","ALPHA_ONEONE_ONEONE","ALPHA_ALPHATOCOLOR","ALPHA_REVERSEONEMINUS","ALPHA_SRC_DSTONEMINUSSRCALPHA","ALPHA_ONEONE_ONEZERO","ALPHA_EXCLUSION","ALPHA_LAYER_ACCUMULATE","ALPHA_EQUATION_ADD","ALPHA_EQUATION_SUBSTRACT","ALPHA_EQUATION_REVERSE_SUBTRACT","ALPHA_EQUATION_MAX","ALPHA_EQUATION_MIN","ALPHA_EQUATION_DARKEN","TEXTURE_CREATIONFLAG_STORAGE","TEXTUREFORMAT_BGRA","TEXTUREFORMAT_DEPTH24_STENCIL8","TEXTUREFORMAT_DEPTH32_FLOAT","TEXTUREFORMAT_DEPTH16","TEXTUREFORMAT_DEPTH24","TEXTUREFORMAT_DEPTH24UNORM_STENCIL8","TEXTUREFORMAT_DEPTH32FLOAT_STENCIL8","TEXTUREFORMAT_STENCIL8","TEXTUREFORMAT_COMPRESSED_RGBA_BPTC_UNORM","TEXTUREFORMAT_COMPRESSED_SRGB_ALPHA_BPTC_UNORM","TEXTUREFORMAT_COMPRESSED_RGB_BPTC_UNSIGNED_FLOAT","TEXTUREFORMAT_COMPRESSED_RGB_BPTC_SIGNED_FLOAT","TEXTUREFORMAT_COMPRESSED_RGBA_S3TC_DXT5","TEXTUREFORMAT_COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT","TEXTUREFORMAT_COMPRESSED_RGBA_S3TC_DXT3","TEXTUREFORMAT_COMPRESSED_SRGB_ALPHA_S3TC_DXT3_EXT","TEXTUREFORMAT_COMPRESSED_RGBA_S3TC_DXT1","TEXTUREFORMAT_COMPRESSED_RGB_S3TC_DXT1","TEXTUREFORMAT_COMPRESSED_SRGB_ALPHA_S3TC_DXT1_EXT","TEXTUREFORMAT_COMPRESSED_SRGB_S3TC_DXT1_EXT","TEXTUREFORMAT_COMPRESSED_RGBA_ASTC_4x4","TEXTUREFORMAT_COMPRESSED_SRGB8_ALPHA8_ASTC_4x4_KHR","TEXTUREFORMAT_COMPRESSED_RGB_ETC1_WEBGL","TEXTUREFORMAT_COMPRESSED_RGB8_ETC2","TEXTUREFORMAT_COMPRESSED_SRGB8_ETC2","TEXTUREFORMAT_COMPRESSED_RGB8_PUNCHTHROUGH_ALPHA1_ETC2","TEXTUREFORMAT_COMPRESSED_SRGB8_PUNCHTHROUGH_ALPHA1_ETC2","TEXTUREFORMAT_COMPRESSED_RGBA8_ETC2_EAC","TEXTUREFORMAT_COMPRESSED_SRGB8_ALPHA8_ETC2_EAC","TEXTURETYPE_UNDEFINED","TEXTURE_CUBE_MAP_ARRAY","TEXTURE_FILTERING_QUALITY_OFFLINE","TEXTURE_FILTERING_QUALITY_HIGH","TEXTURE_FILTERING_QUALITY_MEDIUM","TEXTURE_FILTERING_QUALITY_LOW","MATERIAL_TextureDirtyFlag","MATERIAL_LightDirtyFlag","MATERIAL_FresnelDirtyFlag","MATERIAL_AttributesDirtyFlag","MATERIAL_MiscDirtyFlag","MATERIAL_PrePassDirtyFlag","MATERIAL_AllDirtyFlag","MATERIAL_TriangleFillMode","MATERIAL_WireFrameFillMode","MATERIAL_PointFillMode","MATERIAL_PointListDrawMode","MATERIAL_LineListDrawMode","MATERIAL_LineLoopDrawMode","MATERIAL_LineStripDrawMode","MATERIAL_TriangleStripDrawMode","MATERIAL_TriangleFanDrawMode","MATERIAL_ClockWiseSideOrientation","MATERIAL_CounterClockWiseSideOrientation","ACTION_NothingTrigger","ACTION_OnPickTrigger","ACTION_OnLeftPickTrigger","ACTION_OnRightPickTrigger","ACTION_OnCenterPickTrigger","ACTION_OnPickDownTrigger","ACTION_OnDoublePickTrigger","ACTION_OnPickUpTrigger","ACTION_OnPickOutTrigger","ACTION_OnLongPressTrigger","ACTION_OnPointerOverTrigger","ACTION_OnPointerOutTrigger","ACTION_OnEveryFrameTrigger","ACTION_OnIntersectionEnterTrigger","ACTION_OnIntersectionExitTrigger","ACTION_OnKeyDownTrigger","ACTION_OnKeyUpTrigger","PARTICLES_BILLBOARDMODE_Y","PARTICLES_BILLBOARDMODE_ALL","PARTICLES_BILLBOARDMODE_STRETCHED","PARTICLES_BILLBOARDMODE_STRETCHED_LOCAL","MESHES_CULLINGSTRATEGY_STANDARD","MESHES_CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY","MESHES_CULLINGSTRATEGY_OPTIMISTIC_INCLUSION","MESHES_CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY","SCENELOADER_NO_LOGGING","SCENELOADER_MINIMAL_LOGGING","SCENELOADER_SUMMARY_LOGGING","SCENELOADER_DETAILED_LOGGING","PREPASS_IRRADIANCE_TEXTURE_TYPE","PREPASS_POSITION_TEXTURE_TYPE","PREPASS_VELOCITY_TEXTURE_TYPE","PREPASS_REFLECTIVITY_TEXTURE_TYPE","PREPASS_COLOR_TEXTURE_TYPE","PREPASS_DEPTH_TEXTURE_TYPE","PREPASS_NORMAL_TEXTURE_TYPE","PREPASS_ALBEDO_SQRT_TEXTURE_TYPE","BUFFER_CREATIONFLAG_READ","BUFFER_CREATIONFLAG_WRITE","BUFFER_CREATIONFLAG_READWRITE","BUFFER_CREATIONFLAG_UNIFORM","BUFFER_CREATIONFLAG_VERTEX","BUFFER_CREATIONFLAG_INDEX","BUFFER_CREATIONFLAG_STORAGE","RENDERPASS_MAIN","INPUT_ALT_KEY","INPUT_CTRL_KEY","INPUT_META_KEY1","INPUT_META_KEY2","INPUT_META_KEY3","INPUT_SHIFT_KEY","SNAPSHOTRENDERING_STANDARD","SNAPSHOTRENDERING_FAST","MAX_SUPPORTED_UV_SETS","GL_ALPHA_EQUATION_ADD","GL_ALPHA_EQUATION_MIN","GL_ALPHA_EQUATION_MAX","GL_ALPHA_EQUATION_SUBTRACT","GL_ALPHA_EQUATION_REVERSE_SUBTRACT","GL_ALPHA_FUNCTION_SRC","GL_ALPHA_FUNCTION_ONE_MINUS_SRC_COLOR","GL_ALPHA_FUNCTION_SRC_ALPHA","GL_ALPHA_FUNCTION_ONE_MINUS_SRC_ALPHA","GL_ALPHA_FUNCTION_DST_ALPHA","GL_ALPHA_FUNCTION_ONE_MINUS_DST_ALPHA","GL_ALPHA_FUNCTION_DST_COLOR","GL_ALPHA_FUNCTION_ONE_MINUS_DST_COLOR","GL_ALPHA_FUNCTION_SRC_ALPHA_SATURATED","GL_ALPHA_FUNCTION_CONSTANT_COLOR","GL_ALPHA_FUNCTION_ONE_MINUS_CONSTANT_COLOR","GL_ALPHA_FUNCTION_CONSTANT_ALPHA","GL_ALPHA_FUNCTION_ONE_MINUS_CONSTANT_ALPHA","_GLTFMaterialExporter","exporter","_textureMap","_internalTextureToImage","_exporter","_convertMaterialsToGLTFAsync","exportMaterials","hasTextureCoords","_convertStandardMaterialAsync","_convertPBRMaterialAsync","_stripTexturesFromMaterial","originalMaterial","alphaCutoff","emissiveFactor","originalPBRMetallicRoughness","pbrMetallicRoughness","baseColorFactor","metallicFactor","roughnessFactor","_hasTexturesPresent","normalTexture","occlusionTexture","pbrMat","baseColorTexture","extensionObject","hasTextures","_getTextureInfo","babylonTexture","textureUid","_convertToGLTFPBRMetallicRoughness","babylonStandardMaterial","P0","P1","P2","P3","cubicBezierCurve","solveForRoughness","_MaxSpecularPower","oneMinusSpecularStrength","_DielectricSpecular","glTFMaterial","babylonMaterial","materialMap","_materialMap","_exportTextureAsync","textureInfo","ambientTexture","_FuzzyEquals","_Epsilon","_SetAlphaMode","_finishMaterial","_extensionsPostExportMaterialAdditionalTextures","tasks","extensionWork","_extensionsPostExportMaterialAsync","_babylonScene","_createWhiteTexture","CreateRGBATexture","_resizeTexturesToSameDimensions","texture1","texture2","texture1Size","texture2Size","resizedTexture1","resizedTexture2","_convertPixelArrayToFloat32","factors","resizedTextures","diffuseSize","diffuseBuffer","specularGlossinessBuffer","diffusePixels","specularPixels","metallicRoughnessBuffer","baseColorBuffer","maxBaseColor","maxMetallic","maxRoughness","specularGlossiness","metallicRoughness","_convertSpecularGlossinessToMetallicRoughness","metallicRoughnessFactors","writeOutMetallicRoughnessTexture","writeOutBaseColorTexture","destinationOffset","sRGBBaseColorPixel","metallicRoughnessPixel","_getImageDataAsync","metallicRoughnessTextureData","baseColorTextureData","diffusePerceivedBrightness","_getPerceivedBrightness","specularPerceivedBrightness","_getMaxComponent","_SolveMetallic","baseColorFromDiffuse","baseColorFromSpecular","_convertMetalRoughFactorsToMetallicRoughnessAsync","babylonPBRMaterial","glTFPbrMetallicRoughness","glTFTexture","_getTextureSampler","wrapS","_getGLTFTextureWrapMode","wrapT","wrapMode","_convertSpecGlossFactorsToMetallicRoughnessAsync","specGloss","useMicrosurfaceFromReflectivityMapAlpha","samplerIndex","_exportTextureSampler","_convertSpecularGlossinessTexturesToMetallicRoughnessAsync","imageIndex","_exportImage","_exportTextureInfo","_setMetallicRoughnessPbrMaterial","texCoord","_getPixelsFromTexture","extensionPromise","_extensionsPreExportTextureAsync","_exportTextureInfoAsync","textureMimeType","internalTextureToImage","imageIndexPromise","_extensionsPostExportTextures","_imageData","getFileExtensionFromMimeType","_images","textureIndex","convertHandednessMatrix","isNoopNode","convertNodeHandedness","IsIdentity","_Exporter","_applyExtension","actionAsync","currentPromise","_applyExtensions","_ExtensionNames","preExportTextureAsync","_extensionsPostExportMeshPrimitiveAsync","meshPrimitive","babylonSubMesh","postExportMeshPrimitiveAsync","_extensionsPostExportNodeAsync","postExportNodeAsync","postExportMaterialAsync","postExportMaterialAdditionalTextures","postExportTexture","_forEachExtensions","_extensionsOnExporting","wasUsed","_glTF","extensionsUsed","extensionsRequired","onExporting","_loadExtensions","_ExtensionFactories","asset","_bufferViews","_accessors","_cameras","_nodes","_skins","_animations","_orderedImageData","_animationSampleRate","_glTFMaterialExporter","extensionKey","UnregisterExtension","_reorderIndicesBasedOnPrimitiveMode","primitiveMode","babylonIndices","secondIndex","getUInt32","thirdIndex","setUInt32","_reorderVertexAttributeDataBasedOnPrimitiveMode","vertexBufferKind","meshAttributeArray","_reorderTriangleFillMode","_reorderTriangleStripDrawMode","_reorderTriangleFanMode","_getVertexBufferFromMesh","_writeVertexAttributeData","vertexAttributeKind","_NormalizeTangentFromRef","_writeAttributeData","attributeComponentKind","writeBinaryFunc","vertexAttributes","meshMaterial","convertToLinear","setUInt8","setUInt16","vertexAttribute","writeMorphTargetAttributeData","morphTargetAttributeArray","minMax","difference","difference4","morphData","_generateJSON","shouldUseGlb","glTFPrefix","_totalByteLength","imageName","skins","_generateGLTFAsync","_generateBinaryAsync","binaryBuffer","jsonText","bin","glTFFileName","glTFBinFile","_BinaryWriter","_createSceneAsync","_localEngine","getArrayBuffer","_getPadding","_generateGLBAsync","glbFileName","encodedJsonText","jsonLength","imageByteLength","TextEncoder","encode","jsonPadding","binPadding","imagePadding","headerLength","headerBuffer","headerBufferView","jsonChunkBuffer","jsonChunkBufferView","blankCharCode","codePointAt","jsonPaddingView","binaryChunkBuffer","binaryChunkBufferView","binPaddingBuffer","binPaddingView","imagePaddingBuffer","imagePaddingView","glbData","glbFile","_setNodeTransformation","_setCameraTransformation","babylonCamera","attributeKind","bufferMesh","_createBufferViewKind","_setMorphTargetAttributes","vertexNormals","morphNormals","bufferViewIndex","vertexPositions","morphPositions","POSITION","vertexTangents","morphTangents","_getMeshPrimitiveMode","baseMesh","_setPrimitiveMode","_setAttributeKind","COLOR_0","TEXCOORD_0","TEXCOORD_1","JOINTS_0","JOINTS_1","WEIGHTS_0","WEIGHTS_1","_setPrimitiveAttributesAsync","attributeData","accessorComponentType","indexBufferViewIndex","vertexAttributeBufferViews","exportUnusedUVs","_CalculateMinMaxPositions","primitives","glTFNodeIndex","glTFNode","directDescendents","removedRootNodes","metadataSelector","extras","gltf","removeNoopRootNodes","includeCoordinateSystemConversionNodes","cameraMap","shouldExportNode","glTFCamera","yfov","orthographic","xmag","ymag","exportNodes","_getExportNodes","_createNodeMapAndAnimationsAsync","_createSkinsAsync","skinMap","_nodeMap","skin","descendent","promiseChain","_createNodeAsync","_CreateMorphTargetAnimationFromMorphTargetAnimations","_CreateNodeAnimationFromNodeAnimations","idleGLTFAnimation","_CreateNodeAndMorphAnimationFromAnimationGroups","joints","inverseBindMatrices","boneIndexMap","maxBoneIndex","bufferViewOffset","bindMatrixAccessor","inverseBindAccessorIndex","cell","_arrayBuffer","_dataView","_byteOffset","_resizeBuffer","newBuffer","copyOldBufferSize","oldUint8Array","getVector3Float32FromRef","setVector3Float32FromRef","getVector4Float32FromRef","setVector4Float32FromRef","setByte","setInt8","GLTF2Export","filePrefix","exportWithoutWaitingForScene","glTFData","_PreExportAsync","_PostExportAsync","NAME","KHR_texture_transform","_wasUsed","textureTransform","transformIsRequired","RegisterExtension","KHR_lights_punctual","lightType","lightRotationQuaternion","babylonSpotLight","spot","outerConeAngle","innerConeAngle","lightReference","parentBabylonNode","parentTranslation","parentRotation","parentMatrix","KHR_materials_clearcoat","additionalTextures","clearCoatTextureInfo","clearCoatTextureRoughnessInfo","clearCoatNormalTextureInfo","clearCoatInfo","clearcoatFactor","clearcoatTexture","clearcoatRoughnessFactor","clearcoatRoughnessTexture","clearcoatNormalTexture","KHR_materials_iridescence","iridescenceTextureInfo","iridescenceThicknessTextureInfo","iridescenceInfo","iridescenceFactor","iridescenceIor","iridescenceThicknessMinimum","iridescenceThicknessMaximum","iridescenceTexture","iridescenceThicknessTexture","KHR_materials_anisotropy","anisotropyTextureInfo","anisotropyInfo","anisotropyStrength","anisotropyRotation","anisotropyTexture","KHR_materials_sheen","sheenInfo","sheenColorFactor","sheenRoughnessFactor","sheenColorTexture","sheenRoughnessTexture","KHR_materials_unlit","unlitMaterial","KHR_materials_ior","_isExtensionEnabled","iorInfo","KHR_materials_specular","metallicReflectanceTexture","reflectanceTexture","_hasTexturesExtension","specularFactor","specularColorFactor","specularColorTexture","KHR_materials_volume","subs","volumeInfo","thicknessFactor","attenuationDistance","attenuationColor","KHR_materials_transmission","transmissionFactor","transmissionTexture","thinInstanceAdd","refresh","_thinInstanceUpdateBufferSize","thinInstanceSetMatrixAt","thinInstanceAddSelf","thinInstanceRegisterAttribute","_thinInstanceInitializeUserStorage","thinInstanceBufferUpdated","thinInstanceSetAttributeAt","staticBuffer","thinInstancePartialBufferUpdate","forceRefreshParentInfo","NEGATIVE_INFINITY","numInstances","kindIsMatrix","EXT_mesh_gpu_instancing","noTranslation","noRotation","noScale","iwt","iwr","iws","hasAnyInstanceWorldTranslation","hasAnyInstanceWorldRotation","hasAnyInstanceWorldScale","translationBuffer","rotationBuffer","scaleBuffer","_buildAccessor","bufferOffset","accessorIndex","KHR_materials_emissive_strength","tempEmissiveStrength","emissiveStrengthInfo","emissiveStrength","newEmissiveFactor","GLTFKb3dExporter","_glbExportOptions","generateGlb","GLBAsync","createGlbBlob","exportGlb","downloadBlob","exportGltf","GLTFAsync","disposeFuncs","kb3dNode","ColorGradingTexture","_onLoad","_triggerOnLoad","_load3dlTexture","tempData","pixelIndexW","pixelIndexH","pixelIndexSlice","maxColor","_NoneEmptyLineRegex","words","pixelStorageIndex","toLocaleLowerCase","PanoramaToCubeMapTools","inputWidth","inputHeight","supersample","CreateCubemapTexture","FACE_FRONT","FACE_BACK","FACE_LEFT","FACE_RIGHT","FACE_UP","FACE_DOWN","texSize","sampleFactor","sampleFactorSqr","rotDX1","rotDX2","fy","xv1","xv2","CalcProjectionSpherical","px","py","inputY","HDRFiltering","hdrScale","createMipMaps","_prefilterInternal","_effectWrapper","_effectRenderer","intTexture","_createEffect","prefilter","onFinished","HDRTools","mantissa","float32array","red","green","blue","_Ldexp","uint8array","character","_ReadStringLine","endOfHeader","findFormat","dataPosition","hdrInfo","RGBE_ReadHeader","RGBE_ReadPixels","ConvertPanoramaToCubemap","_RGBEReadPixelsRLE","num_scanlines","scanline_width","dataIndex","scanLineArrayBuffer","scanLineArray","resultBuffer","resultArray","_RGBEReadPixelsNOTRLE","_Rgbe2float","HDRCubeTexture","generateHarmonics","prefilterOnLoad","_generateHarmonics","_onError","_prefilterOnLoad","_supersample","previousOnLoad","hdrFiltering","GetCubeMapTextureData","shortArray","dataFace","_FacesMapping","useInGammaSpace","updateVideoTexture","_videoTextureSupported","clearRect","createExternalTexture","GetTGAHeader","id_length","colormap_type","image_type","colormap_index","colormap_length","colormap_size","pixel_size","UploadContent","pixel_data","use_rle","use_pal","use_grey","pixel_total","palettes","x_start","y_start","x_step","y_step","y_end","x_end","TGATools","_getImageData8bits","colormap","_getImageData16bits","_getImageData24bits","_getImageData32bits","_getImageDataGrey8bits","_getImageDataGrey16bits","_TGATextureLoader","BASIS_FORMATS","_HDRTextureLoader","pixelsDataRGB32","pixelsDataRGBA32","BasisToolsOptions","_WorkerPromise","_Worker","_actionId","TranscodeAsync","wasmBinary","initHandler","actionId","messageHandler","dataViewCopy","ignoreSupportedFormats","BindTexture","LoadTextureFromTranscodeResult","transcodeResult","rootImage","levels","cTFRGB565","transcodedPixels","BasisTools","GetInternalFormatFromBasisFormat","basisFormat","cTFETC1","cTFBC1","cTFBC4","cTFASTC_4x4","cTFETC2","cTFBC7","_BASIS_FORMAT","transcoderModulePromise","TranscodeLevel","loadedFile","levelIndex","convertToRgb565","dstSize","getImageTranscodedSizeInBytes","transcodeImage","srcByteOffset","blockWidth","blockHeight","blockY","blockX","dstI","ConvertDxtToRgb565","getImageWidth","getImageHeight","BASIS","initializeBasis","imgData","BasisFile","basisFile","getHasAlpha","imageCount","getNumImages","levelCount","getNumLevels","levelInfo","GetFileInfo","supportedCompressionFormats","bc7","GetSupportedTranscodeFormat","needsConversion","startTranscoding","loadSingleImage","mipCount","loadMipmapLevels","_BasisTextureLoader","transcodeConfig","hasMipmap","NoiseProceduralTexture","octaves","persistence","animationSpeedFactor","_updateShaderUniforms","RefractionTexture","refractionPlane","ch2","u8","u16","u32","fleb","fdeb","clim","freb","eb","revfl","revfd","rev","hMap","mb","co","le","rvb","sv","r_1","flt","fdt","flm","fdm","slc","ec","nt","captureStackTrace","wbits","wbits16","hTree","et","maxSym","tr","mbt","lft","cst","i2_1","i2_2","i2_3","lc","cl","cli","cln","cls","clen","cf","wfblk","dat","wblk","final","syms","lf","df","bs","bl","dlt","mlb","ddt","mdb","lclt","nlc","lcdt","ndc","lcfreq","lct","mlcb","nlcc","ll","dl","flen","ftlen","dtlen","llm","lcts","it","clct","deo","dflt","lvl","plvl","pre","lst","opt","msk_1","bs1_1","bs2_1","hsh","lc_1","wi","hv","imod","pimod","rem","ch_1","dif","maxn","maxd","ml","mmd","md","ti","lin","din","crct","dopt","st","mem","mrg","wcln","fnStr","td","ks","st_1","spInd","wrkr","td_1","$e$","wk","tl","cbfs","bDflt","deflateSync","pbf","wbytes","deflate","consume","cbify","fltn","te","TextDecoder","stream","strToU8","latin1","ar_1","ar","exfl","wzh","ce","extra","exl","os","mtime","crc","attrs","exf","zip","tot","slft","term","tAll","cbd","mt","cbf","oe","cdl","badd","loc","wzf","_loop_1","cr","com","ms","cbl","queueMicrotask","USDZConverter","useWorker","convertGeometry","meshObject","buildGeometryFromBuffers","buildUSDAFile","zipUSDZFiles","buildGeometryFromData","buildGeometryFromObject","buildHeader","dataToInsert","buildXform","appendFileName","escapeGUID","buildMatrixRow","buildMatrix","stringifyIntArray","buildVector3Array","stringifyFloatArray","buildVector2Array","perStruc","toPrecision","canvasResize","USDZBabylonConverter","toCleanup","modelFileName","conversionWorker","asyncGenerators","meshMaterials","baseGeometry","splitGeometryByMaterialId","baseVertexData","ExtractFromGeometry","subMeshMaterial","newIdx","oldIdx","newVertexData","geometryFileName","geometryData","buildGeometryTransferrable","usdConvertGeometry","uuid","buildMaterial","buildMaterials","textureGenerators","FromHexString","isRGBA","processImageData","processSequentially","offsetMod64","padding","usdPackage","cbs","targetTexureSize","resizeContext","HTMLCanvasElement","imagedata","canvasToBlob","dataPromise","Float64Array","pxLength","resizedData","Uint8ClampedArray","widthRatio","heightRatio","destIndex","lastLineOffset","yOffsetTarget","offsetTarget","newV","tMatrix","t2Matrix","t3Matrix","buildTexture","mapType","parsedTextureId","angleDeg","buildColor","microSurfaceTexture","geometryBuffers","XrHitTester","sessionMgr","getHitResult","hits","FromXRHitTestResult","refSpace","xrpos","xrquat","xrResult","ShadowManager","_bbCache","generatorMap","casters","receivers","projectionMatrixCache","rootBoundingBoxChanged","bbInfo","sg","convertShadowResultion","addAllCastersToGenerator","addCaster","removeCaster","toggleCaster","addReceiver","removeReceiver","toggleReceiver","clearProjectMatrixCache","allMeshes","low","medium","high","simple","advanced","XrLight","_originalLightEnabledMap","_shadowManager","bbCache","_sg","rendererManager","sceneRenderer","get3d","DELTA_PHI","XrMarker","naturalScale","MarkerMaterial","place","_placed","hide","bb","resizeToBoundingBox","getWorld","show","animateVisibility","_lineWidth","getVertexData","xSize","zSize","addCorner","numVertices","vdata","cornerX","cornerZ","ShadowOnlyMaterialDefines","ShadowOnlyMaterial","_needAlphaBlending","shadowColor","activeLight","_activeLight","_getFirstShadowLightForMesh","csg","XrShadowPlane","shadowMaterial","newBb","XrTransientHitTester","getFingers","fingers","convertedFingers","convertedHit","eXrState","SCALE_SNAP","Xr","onSessionEnd","_state","searchingForHit","_goalScale","_goalYaw","_lastScalar","_scaleSnapHigh","_scaleSnapLow","_naturalScale","_wasJustPlaced","_lightingEstimationCount","onSelectStart","initialPlacement","placeModel","placed","_initialHitTester","_transientHitTester","_sessionMgr","_lastFrame","_selectStartScreenPos","screenNdcToPixels","_isTranslating","_dragNodeStartPos","_rootNode","dragHitTester","_dragHitTester","_isRotating","interaction","getKb3dPointerArgs","handlePointerDown","_marker","_isScaling","getFingerSeparation","onSelectEnd","_fingerStartPos","screenCoord","handlePointerUp","handleClick","startSession","refSpaceType","overlayElem","xrInitOptions","domOverlay","sessionEnded","_xrCamera","_session","_refSpace","onXRFrame","_originalSceneCamera","_originalBackground","endSession","sessionEndingPromise","_shadow","_viewerPose","placeInitialMarker","processInput","setSceneTransform","runLightingEstimation","_lightProbe","estimate","getLightEstimate","primaryLightDirection","primaryLightIntensity","dragX","dragFingers","dragFingerResult","deltaPos","_goalPosition","requestLightProbe","hardwareScale","getRigCamera","firstFinger","secondFinger","invalidateRootBoundingBox","invalidate","screenPixelsToNdc","IS_IOS","IS_ANDROID","AR","isDebug","xrActive","xrLoading","gltfExporter","onEnterFunc","_onEnterFunc","startWebXR","usdz","whenAssetsLoaded","forceAndWaitForNextRender","relList","supports","currentObjectUrl","rel","endWebXr","_xr","hasDefaultValue","startIosQuickLook","usdzSrc","canScale","modelUrl","toBoundingBoxInfo","binfo","FromVec3","BoundingBoxCache","boundingBoxInvalidated","_digestCount","_watchedNodes","_changedNodes","watchForChanges","refreshAndGet","bnode","childNodes","cbb","parentMin","parentMax","childBNode","boundingBoxesStable","userIsInteracting","watchersToUpdateAndClear","externalWatchers","hasRootWatchers","hasWatchers","oldBoundingBox","boundingBoxChanged","boundingBoxesMatch","bbWatchersSubject","watcher","BoundingBoxRelativeHelper","eSideOrientationBabylonMap","eUvMapAxisBabylonMap","eUvModeBabylonMap","eAlphaModeBabylonMap","createLightVisualization","lightPositionGizmo","Yellow","coloredMaterial","gizmoOffset","Boundary","ymin","ymax","boundaryContains","halfPoint","QuadTree","duplicatesMap","_divided","xSelector","ySelector","divided","quads","nw","ne","sw","se","findByBoundary","proximity","boundary","itemPoint","getWithProximity","searchPoint","positionKey","createDivision","holeIndices","dim","invSize","hasHoles","outerLen","outerNode","linkedList","triangles","steiner","getLeftmost","compareX","eliminateHole","eliminateHoles","earcutLinked","clockwise","signedArea","insertNode","removeNode","filterPoints","again","ear","pass","zOrder","prevZ","nextZ","tail","numMerges","pSize","qSize","inSize","sortLinked","indexCurve","isEarHashed","isEar","cureLocalIntersections","splitEarcut","ax","bx","ay","by","x0","y0","pointInTriangle","locallyInside","isValidDiagonal","splitPolygon","bridge","hx","hy","mx","my","tanMin","sectorContainsSector","findHoleBridge","bridgeReverse","leftmost","intersectsPolygon","inside","middleInside","q2","o1","o2","o3","o4","onSegment","b2","an","bp","sum","deviation","polygonArea","trianglesArea","flatten","holeIndex","SubmeshVertexData","submeshData","kbGeometry","CopyToRef","extractSubmeshData","deepClone","shallowClone","pathIdxOffset","segmentIdxOffset","vertexIdxOffset","newPaths","newHoles","newArr","writeSubmeshes","verifyGeometry","outputError","indicesCount","maxVertex","GLYPH_COORDS_SCALE","IDENTITY_Q","TextMeshPolygon","lerp","conicTo","cubicTo","TextMeshFont","glyphsParent","createGlyph","charToGlyph","advanceWidth","glyphDef","commands","polys","pathSegmentCount","holePaths","poly","coords","parentSegmentIdx","meshdata","flipIndices","TextMeshVertexData","textMeshFont","getText","generatedVertices","ch1","advance","kern","getKerningValue","centerTransform","generatedVertexData","curveTransform","GreasedLineMeshMaterialType","GreasedLineMeshColorMode","GreasedLineMeshColorDistributionType","GreasedLineMeshColorDistribution","GreasedLineMeshWidthDistribution","CreateHemisphere","halfSphere","merged","MaterialGreasedLineDefines","GREASED_LINE_HAS_COLOR","GREASED_LINE_SIZE_ATTENUATION","GREASED_LINE_COLOR_DISTRIBUTION_TYPE_LINE","GREASED_LNE_RIGHT_HANDED_COORDINATE_SYSTEM","GreasedLinePluginMaterial","DEFAULT_COLOR","sizeAttenuation","colorDistributionType","COLOR_DISTRIBUTION_TYPE_LINE","GREASED_LINE_MATERIAL_NAME","useDash","dashRatio","dashOffset","DEFAULT_WIDTH_ATTENUATED","DEFAULT_WIDTH","_sizeAttenuation","colorMode","COLOR_MODE_SET","_color","useColors","_colorsDistributionType","COLOR_DISTRIBUTION_TYPE_SEGMENT","colorsSampling","_colors","dashCount","_createColorsTexture","_PrepareEmptyColorsTexture","_colorsTexture","resolutionLineWidth","_aspect","dashOptions","_BooleanToNumber","_dashArray","colorModeVisibilityColorsWidthUseColors","_EmptyColorsTexture","CUSTOM_VERTEX_DEFINITIONS","CUSTOM_VERTEX_UPDATE_POSITION","CUSTOM_VERTEX_MAIN_END","CUSTOM_FRAGMENT_DEFINITIONS","CUSTOM_FRAGMENT_MAIN_END","COLOR_MODE_ADD","COLOR_MODE_MULTIPLY","colorTable","_Color3toRGBAUint8","setColors","lazy","origColorsCount","colorArray","updateLazy","_dashCount","setColor","doNotMarkDirty","colorsDistributionType","greasedLineMaterialOptions","GreasedLineTools","vi1","vi2","vi3","pointsFromPredicate","counts","ToVector3Array","segmentCount","dividedLinePoints","segmentVector","nextPoint","what","segmentLength","subLines","GetLineSegments","SegmentizeSegmentByCount","seg","GetLineLength","SegmentizeLineBySegmentLength","lineSegments","lineLength","visbility","localSpace","lengthVisibilityRatio","sumSegmentLengths","segmentIndex","lineSegmentsLength","segmentAngle","widthUp","widthDown","widthStartUp","widthStartDown","widths","includeInner","allPoints","points2d","p2d","GreasedLineMesh","_lazy","_vertexPositions","_colorPointers","colorPointers","_widths","_previousAndSide","_nextAndCounters","addPoints","ConvertPoints","setPoints","_updateColorPointers","_createVertexBuffers","greasedLineMaterial","isLazy","_offsets","_offsetsBuffer","_createOffsetsBuffer","_widthsBuffer","_colorPointersBuffer","pointsCopy","colorPointer","jj","_initGreasedLine","indiceOffset","totalLength","partialLine","_preprocess","_createLineOptions","lineOptions","deepCopiedLineOptions","cloned","intersections","findAllIntersections","intersection","_fastCheck","_trianglePredicate","_worldToUse","firstOnly","_boundingSphere","_V_START","_V_END","_V_OFFSET_START","_V_OFFSET_END","iFloored","positionIdx1","positionIdx2","arrayIdx1","arrayIdx2","positionIdx","_CompareV3","_CopyV3","previousAndSideBuffer","nextAndCountersBuffer","widthBuffer","colorPointersBuffer","offsetBuffer","pWorking","uv3","workingUv","workingUv2","working1","UVMapHelper","generateUVMapInPlace","indexList","newControlPointMap","transformMatrixInv","geometryPositions","geometryIndices","controlPointCount","newUvs","appendVertexData","touchedControlPoints","outputUvs","useExisting","uvIdx","newU","start2","start3","updateFaceUVMapToRef","expandControlPoints","newPositions","newNormals","newUvsTyped","newUv","newVertexIndex","computeAutoFitSize","feat","planar","planarUVMap","boxUVMap","cylindrical","cylindricalUVMap","spherical","sphericalUVMap","lon","nx","ny","nz","newVertex","lonlat","lo","la","lat","xyzToLongitudeLatitude","createUvMapVisualization","uvMap","uvMapVisualization","excludeVertices","vertexGroup","vertexGroupsIntersect","createViewpointVisualization","viewpoint","targetPositionMarker","viewpointPositionGizmo","lightPositionGizmoBox","Renderer","viewer","babylonMap","handlesNode","modelToken","update3d","update3dInternal","validator","validateNode","delete3d","nodeDeletionProcessed","set3d","babylonObject","getChanges","isNew","TEXT_RESOLUTION","AnnotationRenderer","annotationStore","billboards","utilityLayer","meshInfo","parentRotationQuat","annotationInfo","positionNeedsUpdate","textNeedsUpdate","lineNeedsUpdate","createAnnotationTextModels","parentSceneMesh","getRendererNode","needsFont","fontService","targetNode","updatePositionFromAnnotation","textContainer","textRotation","lineOffset","basePoint","drawLine","updateAnnotationText","textMesh","textMaterial","keepInStore","annotationModels","newDirection","getPositionFromConnnector","newOffset","newPivot","calculateTransform","containerMesh","planeContainer","webgl1Context","labelSize","labelText","mathExp","numberVal","mathEval","evalHandlebarString","textGap","textColor","toColor3","labelStats","fullStringTextSize","substrings","lineWidths","maximumWidth","maximumHeight","maximumDescent","string","actualBoundingBoxLeft","actualBoundingBoxRight","textWidth","actualBoundingBoxAscent","actualBoundingBoxDescent","labelArray","getMultiLineStats","textHeight","totalCanvasHeight","heightRatioChange","drawFont","textBaseline","CanvasRenderingContext2D","setTransform","relativeToScene","connector","sceneMesh","offsetWithTransform","lineOffsetWithTransform","lineInfo","linesMesh","solid","dashed","loopCount","DimensionRenderer","watchPosition","dimensionStore","dimensionInfo","maintainNodeScreenSize","updateLineFromCamera","clearMaintainedNodeScreenSize","editorMode","lineCap","updateDimensionControlPoints","updateDimensionLineColor","lineCapsMesh","distanceDefaultVal","precisionMultiplier","dimensionModels","deleteLine","updateLine","dimensionModel","annotationMesh","target1","target2","targetNode2","position1","position2","angleAdjustment","startConnector","endConnector","dimensionLine","startPostion","endPosition","free","rectifiedDimensionLine","lineDirectionCross","capSize","lineToStartConnector","endOffset","lineToEndConnector","arrow","capArrowOffset","lineCapOptions","lineColor","CONNECTOR_DIAM","ConnectorRenderer","subscriptionMap","controlPointScaleRatio","template","withArrows","forceScreenScale","realLayer","utilityMesh","realParts","utilityParts","scaledSize","handleContainer","directionHandle1","directionHandle2","_isPickable","moveAlongSketch","updatePosition","updateDirection","addSubscription","removeSubscription","_visible","getSelectedMaterial","getConnectorMaterial","updateScale","updatePositionMoveAlongSketch","convertRelativeToAbsolute","referenceProcessor","triggerChangesOnReferencers","updateDirectionMoveAlongSketch","normalVector","connectorDirection","angleRot","angleRotMatrix","parentDirection","newQuaternion","deltaBtwXaxisAndDirection","worldScale","posZeroToOne","trackedLength","targetNotFound","lastLength","percentageBetween2Points","vectorBetween2Points","calculatedPosition","controlPoint","lastPathVector","nextPathVector","nextPosition","nextNextPosition","calculatedDirection","createMeshTemplate","parent3d","ball","arrowHead","arrowShaft","twoTone","connectorMaterial","KbmaxOrange","highlightColor","selectedMaterial","FeatureRenderer","vertexDataCache","updateMeshGeometryCache","vertexDataChanges","cachedData","clearVertexDataCache","updateVertexDataCache","cachedSubmeshes","updateMeshGeometry","preserveCache","clearCache","MoveVerticesRenderer","parentMesh","newMeshVertexPositions","scaleVector","meshPivotPoint","currentMeshVertexPositions","vertexPosition","NormalSmoothingRenderer","vertexNeighbors","planeNormals","vectorA","vectorB","vectorC","totalSelectedIndices","groupA","groupB","planeNormal","submeshDataArray","planeCoordinates","planePositions","calculateNormalToRef","neighbor","newNormal","calculateSmoothedUnitVector","planeNormalArray","calculateUnitVector","visitedVertices","currentNormal","currentUv","resultPositions","resultNormals","resultUvs","updateNormalSmoothingWarnings","divider","SolidParticle","particleId","indiceIndex","shapeId","idxInShape","sps","modelBoundingInfo","velocity","translateFromPivot","alive","_ind","_stillInvisible","_rotationMatrix","_model","_sps","_modelBoundingInfo","copyToRef","_bSphereOnly","ModelShape","shapeID","shapeUV","posFunction","vtxFunction","_indicesLength","_shape","_shapeUV","_shapeColors","_positionFunction","_vertexFunction","DepthSortedParticle","indLength","SolidParticleVertex","SolidParticleSystem","particles","nbParticles","recomputeNormals","counter","vars","_bSphereRadiusFactor","_pickable","_isVisibilityBoxLocked","_alwaysVisible","_depthSort","_expandable","_shapeCounter","_copy","_computeParticleColor","_computeParticleTexture","_computeParticleRotation","_computeParticleVertex","_computeBoundingBox","_autoFixFaceOrientation","_depthSortParticles","_mustUnrotateFixedNormals","_particlesIntersect","_needs32Bits","_isNotBuilt","_lastParticleId","_idxOfId","_multimaterialEnabled","_useModelMaterial","_depthSortFunction","_materialSortFunction","_autoUpdateSubMeshes","_recomputeInvisibles","enableDepthSort","enableMultiMaterial","useModelMaterial","expandable","particleIntersection","boundingSphereOnly","bSphereRadiusFactor","computeBoundingBox","autoFixFaceOrientation","pickedBySubMesh","pickedParticles","depthSortedParticles","_multimaterial","_materialIndexesById","_tmpVertex","buildMesh","addShape","_indices32","_positions32","_uvs32","_colors32","_sortParticlesByMaterial","_normals32","_fixedNormal32","_unrotateFixedNormals","lind","pickedData","setMultiMaterial","digest","meshPos","meshInd","meshUV","meshCol","meshNor","totalFacets","facetPos","facetNor","facetInd","facetUV","facetCol","sizeO","fi","_posToShape","_uvsToShapeUV","shapeInd","shapeCol","shapeNor","_setDefaultMaterial","modelShape","currentInd","_meshBuilder","_addParticle","tmpNormal","invertedRotMatrix","pt","_resetCopy","storeApart","materialIndexesById","matIdx","tmpVertex","tmpColor","tmpUV","tmpRotated","pivotBackTranslation","scaledPivot","someVertexFunction","vertexFunction","copyUvs","current_ind","idxpos","idxind","shapeNormals","shapeColors","posfunc","vtxfunc","_insertNewParticle","_rebuildParticle","rebuildMesh","removeParticles","currentNb","firstRemaining","shiftPos","shifInd","particlesLength","modelIndices","modelNormals","modelColors","modelUVs","insertParticlesFromArray","solidParticleArray","currentShapeId","noNor","newPart","currentCopy","setParticles","beforeUpdateParticles","colors32","indices32","fixedNormal32","depthSortParticles","tempVectors","camAxisX","camAxisY","camAxisZ","camInvertedPosition","tmpVector0","colidx","uvidx","uvIndex","vpos","updateParticle","particleRotationMatrix","particlePosition","particleRotation","particleScaling","particleGlobalPosition","dsp","getParticleById","parentGlobalPosition","rotatedY","rotatedX","rotatedZ","rotMatrixValues","iu","iv","updateParticleVertex","vertexX","vertexY","vertexZ","pz","normalx","normaly","normalz","rotatedx","rotatedy","rotatedz","bBox","modelBoundingInfoVectors","tempMin","tempMax","scaledX","scaledY","scaledZ","minBbox","maxBbox","bSphereCenter","halfDiag","bSphereMinBbox","bSphereMaxBbox","vbp","dspl","sid","sortedParticle","particleInd","particleIdx","faceInd","computeSubMeshes","afterUpdateParticles","_materialIndexes","_indicesByMaterial","pickedParticle","picked","getParticlesByShapeId","getParticlesByShapeIdToRef","sortedPart","indicesByMaterial","materialIndexes","vcount","subMeshIndex","lastMatIndex","_setMaterialIndexesById","_filterUniqueMaterialId","refreshVisibleSize","setVisibilityBox","vis","isAlwaysVisible","isVisibilityBoxLocked","computeParticleRotation","computeParticleColor","computeParticleTexture","computeParticleVertex","multimaterialEnabled","multimaterial","autoUpdateSubMeshes","initParticles","recycleParticle","ensureNormalsExist","patternStore","identityRotation","PatternRenderer","setParticlePositionFunc","_pattern","meshUpdated","tempVertexData","sketchNode","getNode","positionArray","getPositionArray","sketchPosition","newPattern","featureUpdaters","sourceNode","needsUpdate","needsRebuild","needsSubmeshRebuild","updates","featureItemIndex","nextUpdateFunction","featureId","processParticleSystem","resultVertexData","resultMesh","totalIndicesSourceCount","totalVerticesSourceCount","submeshesToClone","totalIndicesResultCount","totalVerticesResultCount","baseMaterialIndex","spsPatternSize","numberOfMeshes","percentagePoints","getIntervalPosition","lastPath","lastPointVec3","uvHelper","multiMaterialUseCount","featureMappings","SubmeshRenderer","submeshFeatureMapping","multiMaterialIndex","newSubmesh","existingIndex","existingSubmesh","uvMapVertexes","existingUvs","_submeshFeatureMapping","fetchMaterialNode","materialNode","existingSubmaterialIndex","UvMapRenderer","babylonUvPreviewsMap","previewLayer","updatePreviewMesh","visualizationMesh","deletePreview3d","meshPosition","_previewMaterialVisible","previewMesh","LightRenderer","showPreviews","showAllLightPreviews","shadowManager","babylonLightPreviewsMap","gizmoLayer","pLight","lightRotation","rotationChanged","rotateDirection","shadowLightNode","shadowLight","justCreated","convertShadowFilter","selected","updateLightWarnings","getLightPosition","MateRenderer","removeReflectionProbe","addReflectionProbe","newReflectionProbe","ReflectionProbe","useFloat","_add","_invertYAxis","_renderTargetTexture","currentApplyByPostProcess","_attachedMesh","lookAtFunction","perspectiveFunction","PerspectiveFovRH","PerspectiveFovLH","isReflectionProbe","parsedReflectionProbe","rp","MaterialRenderer","textureRenderer","_globalMaterialCache","orphanedMeshes","reflectionProbeMap","reflectedMeshes","meshesUsingReflectionMaterial","_renderMode","setRenderModeOnMaterial","getAllDefinedMaterials","getGlobalMaterial","_globalMaterialManager","materialDeleted","materialWarningsNeedsUpdate","loadGlobalMaterial","managerServices","matService","imaterial","matNode","newMatNode","pbrAdvMaterial","meshesToBind","_pbrAdvMaterial","updateReflectionProbes","exceptions","sceneBackground","connectorMesh","noPlane","reflector","setTransparency","tangentTexture","anisotropyTangent","clearCoatNormal","refractionProbe","subsurfaceThicknessTexture","hiddenEdges","getNumberOfMaterialRef","nrefs","exceptionList","keyValuePair","connectorId","parseSketchGeometry","currentHole","segmentPoints","currentHolePoints","createSketchPointForEditor","insertIndex","sketch","previousControlPoint","currentPathIndex","previousPath","sketchNormal","getSketchNormalVector3","rawDirection","projectedDirection","eC","projectOnNodeAxis","axisToVector3","fillSketchPolygon","positionVectors","sketchNativeNormal","holePositions","getPolygonNormal","segmentPositions","segmentIndices","flatPositions","working","calculateFaceNormalFromFloats","calculateFaceNormal","segmentNormals","geometryNormal","totalVectorCount","segmentUvs","pathHandedness","iP","$defineProperty","Tree","table","Data","sourceIndex","bitcount","destLen","ltree","dtree","sltree","sdtree","length_bits","length_base","dist_bits","dist_base","clcidx","code_tree","lengths","tinf_build_bits_base","offs","tinf_build_tree","off","tinf_getbit","bit","tinf_read_bits","tinf_decode_symbol","tinf_decode_trees","lt","hlit","hdist","hclen","sym","tinf_inflate_block_data","tinf_inflate_uncompressed_block","tinf_build_fixed_trees","tinyInflate","bfinal","derive","Path","strokeWidth","fail","argument","addPoint","addX","addY","addBezier","const","let","b2ac","addQuad","cp1x","cp1y","cp2x","cp2y","curveTo","quadTo","pathOrCommands","prevX","prevY","cmd","toPathData","decimalPlaces","floatToString","packValues","arguments$1","toSVG","svg","toDOMElement","temporaryPath","newPath","createElementNS","LIMIT32","sizeOf","constant","CHAR","CHARARRAY","USHORT","UINT24","ULONG","LONG","FIXED","FWORD","UFWORD","LONGDATETIME","TAG","Card8","Card16","OffSize","SID","NUMBER","NUMBER16","NUMBER32","REAL","nibbles","STRING","UTF8","numBytes","codePoints","numChars","UTF16","codepoint","eightBitMacEncodings","macintosh","MACSTRING","encoding","macEncodingCacheKeys","macEncodingTableCache","WeakMap","isByteEncodable","encodeVarDeltaRunAsZeroes","deltas","runLength","numDeltas","encodeVarDeltaRunAsBytes","encodeVarDeltaRunAsWords","cacheKey","cachedTable","decodingTable","encodingTable","getMacEncodingTable","VARDELTAS","INDEX","OBJECT","encodedOffsets","offSize","offsetEncoder","encodedOffset","DICT","OPERAND","OPERATOR","OP","wmm","Table","tableName","field","optionKeys","ushortList","itemName","tableList","records","itemCallback","recordList","Coverage","coverageTable","RangeRecord","ScriptList","scriptListTable","scriptRecord","defaultLangSys","reqFeatureIndex","featureIndexes","langSysRecords","langSysRecord","langSys","FeatureList","featureListTable","featureRecord","featureParams","lookupListIndexes","LookupList","lookupListTable","subtableMakers","lookupTable","subtableCallback","lookupType","lookupFlag","subtables","CHARSTRING","ops","cachedValue","op","encodingFunction","sizeOfFunction","TABLE","subtableOffsets","RECORD","LITERAL","Record","getByte","getUShort","getULong","getFixed","typeOffsets","byte","uShort","short","uLong","longDateTime","Parser","relativeOffset","parseByte","parseChar","parseCard8","parseUShort","parseCard16","parseSID","parseOffset16","parseShort","parseF2Dot14","parseULong","parseOffset32","parseFixed","parseString","parseTag","parseLongDateTime","parseVersion","minorBase","major","minor","skip","parseULongList","parseOffset16List","parseUShortList","parseShortList","parseByteList","parseList","parseList32","parseRecordList","recordDescription","rec","fieldName","fieldType","parseRecordList32","parseStruct","struct","parseValueRecord","valueFormat","valueRecord","xPlacement","yPlacement","xAdvance","yAdvance","xPlaDevice","yPlaDevice","xAdvDevice","yAdvDevice","parseValueRecordList","valueCount","parsePointer","structOffset","parsePointer32","parseListOfLists","subOffsets","subList","parseCoverage","startOffset","parseClassDef","startGlyph","classId","list32","recordList32","pointer32","offset16","uShortList","offset32","uLongList","coverage","classDef","langSysTable","reserved","parseScriptList","parseFeatureList","parseLookupList","lookupTableParsers","useMarkFilteringSet","markFilteringSet","parseFeatureVariationsList","majorVersion","minorVersion","conditionSetOffset","featureTableSubstitutionOffset","getCard8","getCard16","getShort","getTag","getBytes","bytesToString","addSegment","glyphIndex","cmap","numTables","platformId","encodingId","groupCount","language","glyphIndexMap","startCharCode","endCharCode","startGlyphId","parseCmapTableFormat12","segCount","endCountParser","startCountParser","idDeltaParser","idRangeOffsetParser","glyphIndexOffset","endCount","startCount","idDelta","idRangeOffset","parseCmapTableFormat4","make","isPlan0Only","unicode","cmapTable","unicodes","addTerminatorSegment","segCountToRemove","endCounts","startCounts","idDeltas","idRangeOffsets","glyphIds","cmap12Groups","segment","glyphId","segCountX2","searchRange","entrySelector","rangeShift","cmap4Length","cmap12Length","cmap12Offset","cffStandardStrings","cffStandardEncoding","cffExpertEncoding","standardNames","DefaultEncoding","CmapEncoding","CffEncoding","charset","GlyphNames","names","numberOfGlyphs","glyphNameIndex","addGlyphNames","lowMemory","_IndexToUnicodeMap","tables","charCodes","addGlyphNamesToUnicodeMap","addUnicode","cffEncoding","isCIDFont","glyphNames","glyphIndexToName","addGlyphNamesAll","charToGlyphIndex","charName","nameToGlyphIndex","gid","Glyph","bindConstructorValues","defineDependentProperty","externalName","internalName","GlyphSet","unitsPerEm","_path","xMin","xMax","getPath","hPoints","hinting","getCommands","getContours","currentContour","lastPointOfContour","getMetrics","xCoords","yCoords","leftSideBearing","rightSideBearing","drawPoints","drawCircles","blueCircles","redCircles","drawMetrics","_push","unicodeObj","_hmtxTableData","glyphset","glyphLoader","ttfGlyphLoader","parseGlyph","buildPath","cffGlyphLoader","parseCFFCharstring","charstring","calcCFFSubroutineBias","subrs","parseCFFIndex","conversionFn","objectOffset","objects","offsetSize","parseOperand","b0","lookup","n2","parseFloatOperand","parseCFFDict","operands","entriesToObject","getCFFString","interpretDict","dict","meta","newDict","TOP_DICT_META","PRIVATE_DICT_META","parseCFFTopDict","parseCFFPrivateDict","gatherCFFTopDicts","cffIndex","topDictArray","iTopDict","topDict","_subrs","_subrsBias","_defaultWidthX","_nominalWidthX","privateSize","private","privateOffset","privateDict","defaultWidthX","nominalWidthX","subrIndex","_privateDict","c1x","c1y","c2x","c2y","subrsBias","nStems","haveWidth","fdIndex","cff","_fdSelect","fdDict","_fdArray","newContour","parseStems","b1","b3","b4","codeIndex","subrCode","jpx","jpy","c3x","c3y","c4x","c4y","gsubrsBias","gsubrs","encodeString","makeDict","makeTopDict","makeTopDictIndex","topDicts","glyphToOps","_13","_23","dx1","dy1","dx2","dy2","formatMajor","formatMinor","parseCFFHeader","nameIndex","topDictIndex","stringIndex","globalSubrIndex","ros","fdArrayOffset","fdArray","fdSelectOffset","fdSelect","fdArrayIndex","nGlyphs","fdArrayCount","iGid","nRanges","iRange","parseCFFFDSelect","numGlyphs","charStringsIndex","privateDictOffset","subrOffset","parseCFFIndexLowMemory","charStrings","parseCFFCharset","enc","nCodes","nLeft","parseCFFEncoding","charString","getCffIndexObject","fontScale","fullName","familyName","weightName","fontBBox","fontMatrix","fontNames","makeNameIndex","postScriptName","charsets","glyphSID","makeCharsets","makeCharStringsIndex","makePrivateDict","makeStringIndex","fontRevision","checkSumAdjustment","magicNumber","created","macStyle","lowestRecPPEM","fontDirectionHint","indexToLocFormat","glyphDataFormat","createdTimestamp","hhea","ascender","descender","lineGap","advanceWidthMax","minLeftSideBearing","minRightSideBearing","xMaxExtent","caretSlopeRise","caretSlopeRun","caretOffset","metricDataFormat","numberOfHMetrics","hmtx","numMetrics","parseHmtxTableOnLowMemory","parseHmtxTableAll","ltag","stringPool","stringPoolOffset","tableVersion","numTags","maxp","maxPoints","maxContours","maxCompositePoints","maxCompositeContours","maxZones","maxTwilightPoints","maxStorage","maxFunctionDefs","maxInstructionDefs","maxStackElements","maxSizeOfInstructions","maxComponentElements","maxComponentDepth","nameTableNames","macLanguages","macLanguageToScript","windowsLanguages","getLanguageCode","platformID","languageID","utf16","macScriptEncodings","macLanguageEncodings","getEncoding","encodingID","reverseDict","makeNameRecord","nameID","addStringToPool","pool","needle","haystack","needleLength","findSubArray","stringOffset","translations","nameIDs","namesWithNumericKeys","nameTableIds","macLanguageIds","windowsLanguageIds","nameRecords","lang","macPlatform","macLanguage","macScript","macEncoding","macName","macNameOffset","winLanguage","winName","winNameOffset","unicodeRanges","begin","os2","xAvgCharWidth","usWeightClass","usWidthClass","fsType","ySubscriptXSize","ySubscriptYSize","ySubscriptXOffset","ySubscriptYOffset","ySuperscriptXSize","ySuperscriptYSize","ySuperscriptXOffset","ySuperscriptYOffset","yStrikeoutSize","yStrikeoutPosition","sFamilyClass","panose","ulUnicodeRange1","ulUnicodeRange2","ulUnicodeRange3","ulUnicodeRange4","achVendID","fsSelection","usFirstCharIndex","usLastCharIndex","sTypoAscender","sTypoDescender","sTypoLineGap","usWinAscent","usWinDescent","ulCodePageRange1","ulCodePageRange2","sxHeight","sCapHeight","usDefaultChar","usBreakChar","usMaxContent","getUnicodeRange","italicAngle","underlinePosition","isFixedPitch","minMemType42","maxMemType42","minMemType1","maxMemType1","nameLength","subtableParsers","substFormat","deltaGlyphId","substitute","sequences","alternateSets","ligatureSets","ligGlyph","lookupRecordDesc","sequenceIndex","lookupListIndex","ruleSets","glyphCount","substCount","lookupRecords","classSets","chainRuleSets","backtrack","lookahead","backtrackClassDef","inputClassDef","lookaheadClassDef","chainClassSet","backtrackCoverage","inputCoverage","lookaheadCoverage","extensionLookupType","extensionParser","substitutes","subtable","sequenceSet","alternateSet","ligatureSet","ligature","returnTable","chainRuleSet","chainRule","tableData","record","gsub","scripts","lookups","variations","numDataMaps","computeCheckSum","makeTableRecord","checkSum","makeSfntTable","sfnt","highestPowerOf2","recordFields","tableFields","tableLength","tableRecord","r1","r2","metricsForChar","notFoundMetrics","fontToTable","firstCharIndex","xMins","yMins","xMaxs","yMaxs","advanceWidths","leftSideBearings","rightSideBearings","lastCharIndex","globals","advanceWidthAvg","maxLeftSideBearing","headTable","hheaTable","maxpTable","os2Table","hasChar","hmtxTable","englishFamilyName","getEnglishName","englishStyleName","englishFullName","uniqueID","en","preferredFamily","preferredSubfamily","fontSubfamily","languageTags","nameTable","ltagTable","postTable","cffTable","metaTable","metas","sfntTable","checkSumAdjusted","searchTag","imin","imax","imid","binSearch","Layout","Position","Substitution","arraysEqual","getSubstFormat","defaultSubtable","checkArgument","parseGlyphCoordinate","shortVectorBitMask","sameBitMask","numberOfContours","_xMin","_yMin","_xMax","_yMax","endPointIndices","instructionLength","instructions","numberOfCoordinates","onCurve","isComposite","moreComponents","scale01","scale10","matchedPoints","transformPoints","newPoints","newPt","contourIndex","contour","curr","next2","componentGlyph","transformedPoints","firstPt","secondPt","getTable","createDefaultTable","getScriptNames","getDefaultScriptName","hasLatn","getScriptTable","scr","getLangSysTable","scriptTable","getFeatureTable","featIndexes","allFeatures","getLookupTables","featureTable","allLookups","getGlyphClass","classDefTable","getCoverageIndex","expandCoverage","defaultKerningTables","getKerningTables","kerningLookups","leftIndex","rightIndex","covIndex","posFormat","pairSet","pairSets","pair","secondGlyph","class1","classDef1","class2","classDef2","classRecords","gpos","getSingle","substitutions","lookupTables","getMultiple","replacements","getAlternates","alternates","getLigatures","ligatures","ligSet","lig","addSingle","substitution","coverageGlyph","addMultiple","addAlternate","addLigature","ligComponents","ligatureTable","getFeature","instructionTable","execGlyph","execComponent","glyf","loca","parseGlyfTableOnLowMemory","parseGlyfTableAll","Hinting","_fpgmState","_prepState","_errorState","roundOff","roundToGrid","roundToDoubleGrid","roundToHalfGrid","roundUpToGrid","roundDownToGrid","roundSuper","period","srPeriod","phase","srPhase","srThreshold","xUnitVector","xo","interpolate","rp1","rp2","pv","do1","do2","doa1","doa2","dm1","dm2","setRelative","normalSlope","org","rpx","rpy","yo","rpdx","rpdy","xTouched","untouch","yUnitVector","yTouched","UnitVector","getUnitVector","HPoint","prevPointOnContour","nextPointOnContour","preventExtensions","pvns","fvs","nextTouched","prevTouched","HPZero","defaultState","cvCutIn","deltaBase","deltaShift","minDis","autoFlip","State","prog","zp0","zp1","zp2","rp0","fv","dpv","initTZone","tZone","gZone","handleElse","ins","ip","nesting","SVTCA","DEBUG","SPVTCA","SFVTCA","SPVTL","p2i","p1i","SFVTL","POP","MDAP","z0","IUP","cp","np","pLen","SHP","rpi","SHC","ci","SHZ","MSIRP","MIAP","cv","cvt","GC","MD","pi1","DELTAP123","ppem","ds","ROUND","DELTAC123","SDPVTL","PUSHB","PUSHW","MDRP_MIRP","indirect","setRp0","keepD","ro","od","cvte","rp0i","activeState","ContextRange","contextName","ContextChecker","checkStart","checkEnd","openRange","ContextParams","Event","eventId","subscribers","initializeCoreEvents","this$1","coreEvents","updateContextsRanges","Tokenizer","registeredContexts","contextCheckers","registeredModifiers","isArabicChar","isIsolatedArabicChar","isTashkeelArabicChar","isLatinChar","FeatureQuery","SubstitutionAction","lookupCoverage","singleSubstitutionFormat1","singleSubstitutionFormat2","substituteIndex","lookupCoverageList","coverageList","contextParams","lookupList","lookupIndex","chainingSubstitutionFormat3","lookupsCount","inputLookups","lookaheadOffset","lookaheadContext","lookaheadParams","lookaheadLookups","backtrackContext","backtrackParams","backtrackLookups","getLookupByIndex","getLookupMethod","getSubstitutionType","ligatureSubstitutionFormat1","ligSetIndex","decompositionSubstitutionFormat1","prepState","fpgmState","fpgm","funcs","prep","oCvt","cg","gz","cc","gLen","inhibitGridFit","pa0i","pa1i","pb0i","pb1i","pa0","pa1","pb0","pb1","x3","y3","x4","y4","cip","cprog","ipBegin","rp1i","rp2i","e2","sel","ignoreCvt","getState","stateId","inboundIndex","composeRUD","RUDs","RUD","hasFAILObject","FAIL","replaceRange","silent","isTokenType","replaced","replaceToken","removeRange","removeToken","insertToken","registerModifier","modifierId","modifier","newToken","conditionParams","modifierParams","newStateValue","subsId","setCurrentIndex","rangeToText","getRangeTokens","on","registerContextChecker","contextStartCheck","contextEndCheck","getContextRanges","resetContextsRanges","runContextCheck","setEndOffset","rangeId","contextChecker","tokenize","getDefaultScriptFeaturesIndexes","getScriptFeaturesIndexes","scriptTag","mapTagsToFeatures","getScriptFeatures","featuresIndexes","lookupFeature","getFeatureLookups","getLookupSubtables","substType","supportedScript","supportedFeature","scriptFeatures","arabicWordCheck","startCheck","prevChar","endCheck","nextChar","arabicSentenceCheck","nextIsWhitespace","isWhiteSpace","arabicCharAhead","SUBSTITUTIONS","subst","compsCount","applySubstitution","arabicPresentationForms","featuresTags","tokenizer","charContextParams","CONNECT","isolated","tashkeel","willConnectPrev","willConnectNext","getContextParams","arabicRequiredLigatures","latinWordCheck","latinLigature","Bidi","baseDir","checkId","contextChecks","tokenizeText","reverseArabicSentences","rangeTokens","checkGlyphIndexStatus","applyArabicPresentationForms","applyArabicRequireLigatures","applyLatinLigatures","Font","styleName","designer","designerURL","manufacturer","manufacturerURL","license","licenseURL","copyright","trademark","weightClass","usWeightClasses","MEDIUM","widthClass","usWidthClasses","fsSelectionValues","REGULAR","_hinting","outlinesFormat","HintingTrueType","addName","nameString","nameKey","makeFvarAxis","minValue","parseFvarAxis","makeFvarInstance","inst","axisTag","parseFvarInstance","setText","registerFeatures","supportedTags","applyFeatures","checkContextReady","contextId","applyFeaturesToContexts","processText","getBidiText","getTextGlyphs","updateFeatures","defaultRenderOptions","stringToGlyphs","bidi","notdef","nameToGlyph","leftGlyph","rightGlyph","gposKerning","kerningPairs","kerning","forEachGlyph","letterSpacing","tracking","fullPath","gX","gY","gFontSize","glyphPath","getPaths","glyphPaths","getAdvanceWidth","assertNamePresent","englishName","toTables","toBuffer","intArray","webkitURL","createEvent","initEvent","dispatchEvent","fs","require","ab","arrayBufferToNodeBuffer","writeFileSync","ITALIC","UNDERSCORE","NEGATIVE","OUTLINED","STRIKEOUT","BOLD","USER_TYPO_METRICS","WWS","OBLIQUE","ULTRA_CONDENSED","EXTRA_CONDENSED","CONDENSED","SEMI_CONDENSED","SEMI_EXPANDED","EXPANDED","EXTRA_EXPANDED","ULTRA_EXPANDED","THIN","EXTRA_LIGHT","LIGHT","SEMI_BOLD","EXTRA_BOLD","BLACK","fvar","offsetToData","axisCount","axisSize","instanceSize","instanceStart","attachList","attachPoints","caretValue","pointindex","ligCaretList","ligGlyphs","markGlyphSets","gdef","markAttachClassDef","posformat","valueFormat1","valueFormat2","class1Count","class2Count","pairs","subtableVersion","nPairs","parseWindowsKernTable","parseMacKernTable","shortVersion","parseFn","glyphOffsets","glyphOffset","parseOpenTypeTableEntries","tableEntries","checksum","uncompressTable","tableEntry","inBuffer","compressedLength","outBuffer","inflate","parseBuffer","cffTableEntry","fvarTableEntry","glyfTableEntry","gdefTableEntry","gposTableEntry","gsubTableEntry","hmtxTableEntry","kernTableEntry","locaTableEntry","nameTableEntry","metaTableEntry","signature","flavor","compLength","origLength","parseWOFFTableEntries","locaTable","locaOffsets","glyfTable","kernTable","gdefTable","gposTable","gsubTable","fvarTable","primitiveShapeProperties","capModeMap","both","textMeshFonts","MeshRenderer","pivotGizmos","sketchPreviews","sketchTracings","sketchPlanePreviews","gizmo","setRenderMode","setRenderModeOnMesh","loadNewGeometry","invalidateBoundingBox","parentsVisible","isNewMesh","createOrUpdateMesh","pivotGizmo","oldPreview","createSketchPlaneVisualization","SketchPlancePreviewMesh","updateParentMeshFromNode","rebuildBaseGeometry","preTransformChanged","meshVertexData","createPrimitiveMesh","setBabylonUvData","preTransformVertices","getMeshVertexData","buildSketchGeometry","buildTextGeometry","pathSegmentIndices","pathGroupKeys","holePathsReferencingKeys","currentPath","thisPoint","primaryNormal","holesByPathsWithIndices","parentPathIndex","reorderedPathSegmentIndices","positionPointer","holesPositions","hi","holeData","newMeshVertices","indexMapToGrouped","indexOfPath","startingPoints","calculatedBoundingBox","calculateSketchBoundingBox","updateSketchPreview","segmentColors","newPreviewMesh","updateSketchTracing","sketchColor","linePoints","vertexDataBuilder","openTypeFont","fontParse","getMeshGeometry","localMaterial","globalLibMaterial","FromGeometry","cachedGeometry","existingTransform","_hasTransform","PRESERVE_PROPERTY","updateMeshTransform","recursiveLoadChildMeshGeometry","allChanges","meshVerticesNeedReset","setBabylonVertexData","updateConnectorScaling","polygonTracker","warningTracker","allParentsVisible","polygonNumber","featureNumber","MeshWarningTracker","globalMaterial","setMaterialOnMesh","updateSketchDrawing","resetMeshGeometry","_previewVisible","meshWarningsNeedsUpdate","originOffset","scalingMatrix","transformChanged","BabylonSphereBuilder","worldPivot","sliceStart","totalYRotationSegments","nextYStep","createSphere","Cylinder","angle_step","sI","createCylinder","TorusBuilder","outerSegments","innerSegments","closedRing","closedTube","lastOuterRing","lastTubeSection","createTorus","colorSet","normalSet","DiscBuilder","createDisc","kbMesh","preserveSubmeshes","invalid","uvData","squareSize","calculatedSize","boundingBoxCenter","sketchPlaneVisualization","sketchMesh","rotationVector","resultQuaternion","FromUnitVectorsToRef","controlPointRenderer","firstPoint","PathRenderer","flipMatrix","mateConnectors","mate","followerMesh","c1Mesh","c2Mesh","mateOffset","worldDelta","angleOffsetQ","targetWorldTransformMatrix","currentWorldTransformMatrix","c2LocalTransform","grandParentContainerWorldMatrix","grandParentContainerWorldMatrixInv","calculatedTargetWorldTransform","revolute","targetQ","currentQ","targetDirection","currentDirection","targetST","decomposeSwingTwist","currentST","targetRotationNew","swing","twist","fastened","slider","followerMeshTransformMatrix","resultFollowerRotMatrix","c1PlaneNormal","c1Plane","c2Direction","c1Pos","c2Pos","c2PlaneNormal","c2Plane","c1Direction","parallel","_swap","totalOffset","newAbsPos","ConstraintType","sortMates","validMates","primes","getNodeMap","Fixed","visitedNodes","visitedMates","nodePath","matePath","overConstrained","recurseMateChain","getNodeChain","c1Parent","c2Parent","sourceEdge","isOverconstrained","cycle","Processor","afterCleanupTasks","beforeDeletions","deletions","afterDeletions","beforeUpdates","afterUpdates","afterCleanup","doAfterCleanup","MateProcessor","connectorRenderer","rebuildDependencies","mateDeletions","connectorDeletions","mateChanges","connectorChanges","meshChanges","buildDependencies","sortedMates","processMate","getConnectorMesh","followerNode","pauseRenderWhile","nodeWithTransform","controlPointMesh","targetTransformMatrix","mateControlPointToConnector","originalFollowerMeshAbsPos","originalFollowerMeshAbsQuat","originalFollowerMeshLocalPos","originalFollowerMeshLocalQuat","sortResult","getValidMates","addError","rootScene","sceneMates","clearProp","CircularPatternRenderer","featurePivot","featureAxis","featureArc","workingQuaternion","DeleteFacesRenderer","newIndices","currReadIndex","DisplacementMapRenderer","textureCanvas","willReadFrequently","textureLayers","getActiveLayers","getRenderer","textureHasContent","baseImageSize","getNativeSize","clampedSize","paintedImage","paintTextureOnCanvas","displace","uInt8Array","xztoxy","ExtrudeRenderer","extrudedMeshes","shapeHoles","rotToZ","pathTransformToZ","pathTransformToZInv","shape2d","builder","hole2d","extruded","extrudedAll","applyExtrusion","getSketchShape","maxIdx","usedVertexLength","normalizeGeometry","concatArr","pushAmount","currentCSGMeshId","vertColor","coplanarFront","coplanarBack","polygonType","tj","vi","vj","Polygon","FromPoints","polygons","_plane","_front","_back","_polygons","clipPolygons","clipTo","bsp","allPolygons","CSG","meshRotation","meshScaling","meshRotationQuaternion","invertWinding","vertColors","sm","sml","il","indexIndices","sourceNormal","sourcePosition","_FromPolygons","copyTransformAttributes","union","unionInPlace","intersectInPlace","inverse","inverseInPlace","buildMeshGeometry","keepSubMeshes","polygonIndices","vertice_dict","vertex_idx","subMeshDict","subMeshObj","indexEnd","jl","localVertex","localNormal","areUvsDifferent","areColorsDifferent","materialMaxIndex","materialIndexOffset","toMesh","createJoinGeometryVisualization","visualization","joinedMeshes","JoinGeometryRenderer","bbSubscriptions","targetMeshVisualizationMap","parentMeshVisualizationMap","thisMesh","workingVertexData","tempMesh","targetTransform","newMeshInstance","featureParentMesh","resultCsg","FromMesh","targetMeshMatchTransform","targetMeshTransform","opCsg","cut","resultMeshCsg","existingMaterials","featureInstance","deleteTargetMeshVisualization","existingSubscription","targetMeshNode","originalMesh","targetMeshCopy","LinearPatternRenderer","featureTranslation","workingVector","SliceFeatureRenderer","babylonMirrorPreviewsMap","triCount","positionsCount","planePoint","preSliceSubmeshes","newSubmeshes","edgeSplits","edgeStart","edgeDir","iA","iB","vA","vB","lnDot","preSliceSubmesh","aSplit","bSplit","baseGeometryVertexLength","newPositionIndex","originalSubmeshes","originalSubmesh","visualizationPlane","mirror","mirrorVisualization","createMirrorVisualization","parentBoundingBox","getMeshBoundingBox","MirrorFeatureRenderer","planeIntersect","combinedIndices","combinedPositions","combinedNormals","combinedUvs","combinedSubmeshes","assertNumbers","numbers","annotateArcCommand","lArcFlag","sweepFlag","rX","rY","xRot","x1_","y1_","c_ScaleTemp","c_Scale","cx_","cy_","cRot","cX","cY","phi1","phi2","intersectionUnitCircleLine","termSqr","SVGPathDataTransformer","DEG","arcAt","phiDeg","bezierRoot","EPS","x01","x12","PRECISION","discriminantX4","bezierAt","TO_ABS","INFO","relative","NORMALIZE_ST","prevCurveC2X","prevCurveC2Y","prevQuadCX","prevQuadCY","SVGPathData","SMOOTH_CURVE_TO","CURVE_TO","SMOOTH_QUAD_TO","QUAD_TO","QT_TO_C","prevQuadX1","prevQuadY1","prevXAbs","prevYAbs","pathStartXAbs","pathStartYAbs","MOVE_TO","CLOSE_PATH","MATRIX","pathStartX","origX1","origX2","comRel","sqr","HORIZ_LINE_TO","LINE_TO","VERT_LINE_TO","sinRot","cosRot","xCurve","yCurve","C","A1","B1","C1","newXRot","newSinRot","newCosRot","roundVal","rf","normalizeZ","normalizeH","normalizeV","pathStartY","ARC","x1Rel","y1Rel","LINE_COMMANDS","xRel","yRel","x2Rel","y2Rel","dX","dY","phiMin","deltaPhi","partCount","phiStart","phiEnd","deltaPhi_1","xTemp","yTemp","toAbs","qtToC","normST","fixX","fixY","DRAWING_COMMANDS","T","xDerivRoots_1","derivRoot","yDerivRoots_1","xRotRad","x90","y90","phiMin_1","phiMax","normalizeXiEta","xi","xDerivRoots_2","H","yDerivRoots_2","TransformableSVG","TO_REL","NORMALIZE_HVZ","A_TO_C","SANITIZE","TRANSLATE","SCALE","ROTATE","SKEW_X","SKEW_Y","X_AXIS_SYMMETRY","Y_AXIS_SYMMETRY","ANNOTATE_ARCS","isDigit","SVGPathDataParser","curArgs","canParseCommandOrComma","SyntaxError","finishCommand","isAArcFlag","curCommandType","curNumber","isEndingDigit","curNumberHasExp","curNumberHasExpDigits","curNumberHasDecimal","COMMAND_ARG_COUNTS","curCommandRelative","chunk","parsedCommands_1","cT","boundsTransform","CALCULATE_BOUNDS","transformFunction","newCommands","transformedCommand","encodeSVGPath","finish","PathParser","control","previousCommand","angles","isEnd","getPoint","xProp","yProp","makeAbsolute","getAsControlPoint","getAsCurrentPoint","getReflectedControlPoint","addMarker","priorTo","angleBetweenPoints","addMarkerAngle","getMarkerPoints","getMarkerAngles","PathHandler","_currentPosition","initialPosition","scaleDivider","nodeManager","stubPath","parseTrace","pathM","commonPathCommand","pathCommand","pathL","pathQ","pathZ","reprojectPaths","parsePath","pathParser","pathC","pathS","pathT","pathA","checkIfNewPathNeeded","desiredPathType","activePath","addNewPath","newControlPoint","lastControlPoint","cubicBezier","quadraticBezier","secondLastControlPoint","calculateReflection","pathType","controlPoint2","VERSION_NUMBER","PATHSCAN_COMBINED_LOOKUP","SPECPALETTE","GKS","OPTION_PRESETS","corsenabled","ltres","qtres","pathomit","rightangleenhance","colorsampling","numberofcolors","mincolorratio","colorquantcycles","layering","strokewidth","linefilter","roundcoords","viewbox","lcpr","qcpr","blurradius","blurdelta","posterized1","posterized2","curvy","sharp","detailed","smoothed","grayscale","fixedpalette","randomsampling1","randomsampling2","artistic1","artistic2","artistic3","artistic4","posterized3","pal","roundToDec","places","colorQuantization","paletteacc","pixelnum","cnt","palette","imageDataCache","newImageData","pxcnt","numberOfColors","rcnt","gcnt","bcnt","graystep","colorqnum","colorstep","rndnum","generatePalette","samplePalette","ni","nj","samplePaletteByGrid","racc","gacc","bacc","aacc","wacc","imageData2","thisgk","himageData","toRgbaStr","svgPathString","tracedata","lnum","pathnum","smp","pcnt","toSvgColorStr","hcnt","holechildren","hsmp","fitSeq","seqstart","seqend","dist2","errorpoint","errorval","curvepass","pl","fitpoint","t3","splitpoint","boundingBoxIncludes","parentbbox","childbbox","pointInPoly","isin","pathScan","pacnt","lookuprow","pathfinished","holepath","boundingbox","isholepath","parentidx","parentcnt","segtype1","segtype2","linesegment","batchTracePaths","internodepaths","btracedpaths","interNodes","palen","nextidx","nextidx2","previdx","previdx2","idx5","batchTraceLayers","binternodes","btbis","layeringStep","cnum","ah","aw","imageTracer","ImageTracer","versionnumber","optionpresets","checkOptions","presetName","preset","imageDataToSVG","getImgdata","imgd","svgstr","lcnt","getSvgString","imageDataToTracedata","colornum","tracedlayer","ls","n3","n4","n5","n6","n7","n8","layercontainerid","parentid","drawLayers","bis","bpaths","batchInterNodes","batchPathScan","appendSVGString","validNameCharacters","whitespace","quotemark","currentElement","offsetLine","offsetColumn","originalLines","lineRanges","rangeContains","getLocation","column","getLocator","locate","before","beforeLine","after","neutral","cdata","closingTag","getName","getAttribute","selfClosing","allowSpaces","escaped","getQuotedAttributeValue","getUnquotedAttributeValue","SvgConverter","decover","headerText","dataUrl","imageToTracedata","calculateScaleDivider","paletteStrengths","getPaletteColorStrength","bgIdx","primaryLayers","pathsCreated","holeLayerIdx","pathSet","svgText","parsedSVG","parseSvgRecursively","getViewBoxFromSvgElement","elem","viewBox","int","findSvgViewBox","createRectangle","createCircle","createPolyLine","nameGeneratedPaths","newChildren","resultChildren","newControlPoint2","newControlPoint3","newControlPoint4","newControlPoint5","newControlPoint6","newControlPoint7","newControlPoint8","newControlPoint9","newControlPoint10","newControlPoint11","newControlPoint12","newControlPoint13","newControlPoint14","newControlPoint15","newControlPoint16","centerRadiusCircle","closeArrayCorr","useOpenGLOrientationForUV","CreateMiteredRibbon","startCap","endCap","startCapVectors","endCapVectors","resultVectors","resultCaps","firstPath","reverseHandednessOfPositions","secondLastPoint","startCapIndices","endCapIndices","combineVertexDataWithCaps","createLineCaps","capMeshIndices","capMeshPositions","totalBasePositions","totalBaseNormals","totalBaseUvs","totalBaseIndices","startCapPositions","endCapPositions","meshIndices","generatedUvs","generateUvsForCaps","combinedPositionsArray","combinedNormalsArray","combinedUvsArray","combinedIndicesArray","positionsArray","reversedArray","operationVectors","isZeroVector","findDistinctVectors","xp1","xp2","normalizedXp3","normalizedXp2","SweepRenderer","targetPathNode","pathMesh","pathPoints","pathVertexData","pathTransform","sketchTransform","pathPrimaryVector","extrudeOptions","mitredExtrude","nextLine","axisY","nextAxisX","nextAxisY","nextAxisZ","nextStartPoint","bisector","allPaths","ribbonPath","planeParallel","closedPlaneParallel","closedPlaneNormal","closedPlane","TransformRenderer","WeldVerticesRenderer","positionAverage","normalsAverage","uvAverage","selectedIndices","selectedVertices","selectedIndicesArray","vertexVisited","indicesResult","FeatureStackRenderer","queuedGeometryToReset","queuedFeaturesToProcess","processedFeatures","featureOrder","renderers","moveVertices","weldVertices","displacementMap","uvMaps","linearPatterns","circularPatterns","joinGeometry","deleteFaces","normalSmoothing","extrude","sweep","forceRebuildGeometry","applyNewVertexData","movedIndex","getMovedFeatureIndex","updateFeatureOrder","recursivelyUpdateFeatureVisibility","update3dGeometry","dataTypes","hasFeatureToUpdate","hasChange","featureIsAffected","queueGeometryUpdate","existingVertexData","currentChanges","getFeatureIndexOnParent","renderStartTime","queuedFeatureToProcess","startFeatureIndex","vertexDataCurrent","vertexDataNew","renderTime","maxRenderTime","processThisFeature","baseSubmeshes","lastSubmesh","remainingIndexCount","calculateBaseSubmeshes","oldArray","newArrayId","affectedFeatures","disabledFeatures","newAffectedFeatures","affectedFeature","featureIndex","getFeatureIndex","affector","_disableFeatureStackChange","ExtendToRef","affected","changeType","getMeshName","SketchControlPointRenderer","updateHandles","enableHandle1","enableHandle2","arcThrough3Points","handleBar","sketchPath","getRelativeDirection","invTransform","getControlPointAbsPosition","convertRelativeDirectionToAbsolute","_handleBar","handleBall","handle1","handle2","ReferenceProcessor","referencesTo","orphanedReferences","rootManager","nodeId","referenceTo","referencedById","referencedBy","referencePropertyName","referenceId","getReferencesTo","orphanedReference","changedNodeManager","targettingProperties","_prop","referenceSettings","originalReferencedNodeId","originalReferencedNodeKey","originalReferencedNode","nestedPathIdSource","nestedPathIdTarget","isNestedSource","nestedTargetNode","changeMap","triggerChangesOnReferencersSync","transformationPropagation","SketchProcessor","updatePath","sketchPathChanges","sketchControlPointChanges","sketchesToUpdate","calculateSketchBounds","getControlPoints","buildSketchPoints","getPointPosition","CreateCatmullRomSpline","getPointDirection","CreateHermiteSpline","getPointDirection2","CreateCubicBezier","ArcThru3Points","PostProcessRenderPipeline","_renderEffectsForIsolatedPass","renderEffectName","addEffect","renderEffect","_enableEffect","renderEffects","MakeArray","_disableEffect","_attachCameras","unique","cams","indicesToDelete","_detachCameras","cameraName","_enableMSAAOnFirstPostProcess","sampleCount","effectKeys","getPostProcesses","PostProcessRenderEffect","singleInstance","_singleInstance","_getPostProcesses","_indicesForCamera","pps","cameraKey","PostProcessRenderPipelineManager","_renderPipelines","supportedPipelines","renderPipelineName","pipeline","addPipeline","renderPipeline","removePipeline","attachCamerasToRenderPipeline","detachCamerasFromRenderPipeline","enableEffectInPipeline","disableEffectInPipeline","_postProcessRenderPipelineManager","PostProcessRenderPipelineManagerSceneComponent","LensRenderingPipeline","LensChromaticAberrationEffect","HighlightsEnhancingEffect","LensDepthOfFieldEffect","_pentagonBokehIsEnabled","_depthTexture","grain_texture","_grainTexture","_createGrainTexture","_edgeBlur","edge_blur","_grainAmount","grain_amount","_chromaticAberration","chromatic_aberration","_distortion","distortion","_highlightsGain","dof_gain","_highlightsThreshold","dof_threshold","_dofDistance","dof_focus_distance","_dofAperture","dof_aperture","_dofDarken","dof_darken","_dofPentagon","dof_pentagon","_blurNoise","blur_noise","_createChromaticAberrationPostProcess","_createHighlightsPostProcess","_createDepthOfFieldPostProcess","_chromaticAberrationPostProcess","_highlightsPostProcess","_depthOfFieldPostProcess","postProcessRenderPipelineManager","edgeBlur","setEdgeBlur","grainAmount","setGrainAmount","chromaticAberration","setChromaticAberration","dofAperture","setAperture","edgeDistortion","setEdgeDistortion","dofDistortion","setFocusDistance","darkenOutOfFocus","setDarkenOutOfFocus","blurNoise","pentagonBokeh","enablePentagonBokeh","disablePentagonBokeh","highlightsGain","setHighlightsGain","highlightsThreshold","setHighlightsThreshold","disableEdgeBlur","disableGrain","disableChromaticAberration","disableEdgeDistortion","disableDepthOfField","enableNoiseBlur","disableNoiseBlur","disableHighlights","disableDepthRender","SSAO2Configuration","SSAO2RenderingPipeline","_ssaoPostProcess","_getDefinesForSSAO","_sampleSphere","_generateHemisphere","textureSamples","_textureSamples","_originalColorPostProcess","_forceGeometryBuffer","geometryBufferRenderer","bypassBlur","_getDefinesForBlur","expensiveBlur","_getSamplersForBlur","_blurHPostProcess","_blurVPostProcess","_bypassBlur","_expensiveBlur","forceGeometryBuffer","SSAOOriginalSceneColorEffect","SSAORenderEffect","SSAOBlurHRenderEffect","SSAOBlurVRenderEffect","SSAOCombineRenderEffect","totalStrength","minZAspect","bilateralSamples","bilateralSoften","bilateralTolerance","_bits","ssaoRatio","_createRandomTexture","_createSSAOPostProcess","_createBlurPostProcess","_createSSAOCombinePostProcess","_ssaoCombinePostProcess","_randomTexture","bilateral","_createBlurFilter","horizontal","blurFilter","ssaoCombineSize","originalColorSize","_radicalInverse_VdC","_hammersley","_hemisphereSample_uniform","numSamples","rand","PERSPECTIVE_DEPTH_PROJECTION","ORTHO_DEPTH_PROJECTION","randVector","FxaaPostProcess","glInfo","SKYBOX_SIZE","SamplingModeMap","nearest","trilinear","SceneRenderer","gui","renderManager","yVector","processEnvironment","tnode","forceDisablePostProcess","fxaaPostProcess","lensEffectPostProcess","ssaoPostProcess","annotationNode","gridNeedsRebuild","eventPayload","fromEntries","CustomEvent","detail","bubbles","_envTexturePath","_envTexture","skyMat","getSkyBox","blendModeMap","darken","lighten","luminosity","workingCanvases","getWorkingCanvas","disposeWorkingCanvas","TextureLayerRenderer","paintTexture","updateTextureWarnings","calculateOffsetAndDimensions","ctxWidth","ctxHeight","sourceWidth","sourceHeight","contain","stretch","minRatio","middle","drawInBlendedContext","maskingCtx","getTransform","globalCompositeOperation","tempCanvas","filterImageInPlace","globalAlpha","tempContext","dstImageData","dstData","alphaScale","colorize","tmpCtx","normalRotate","rotateAngle","drawSourceToCanvas","firstTextureLayer","drawAt","pivotX","pivotY","patternCanvas","patternAttempt","createPattern","scaleOffsetU","scaleOffsetV","rotateOffsetU","rotateOffsetV","innerContext","_visualize","strokeRect","ColorFillTextureLayerRenderer","FileTextureLayerRenderer","loadTextureBuffer","resp","_bufferSource","base64ArrayBuffer","base64","encodings","byteRemainder","mainLength","SvgTextureLayerRenderer","svgHelpers","_raster","renderedWidth","renderedHeight","fontLoadPromise","fontFacesFound","fontsToLoad","generateSvg","embedFonts","imgElement","svgString","fontStyles","generateImagePromise","loadedImageElement","deferSceneReady","textureChannelMap","emissive","ambient","reflection","maskingCanvas","TextureStackRenderer","queuedTexturesToProcess","queueSize","noMipMapFallback","updateTextureStackRequest","deferCalled","fileTexture","svgTexture","updateTextureStacksExec","textureId","getTexture3dId","disposeTexture","forceRebuildTexture","resetTextureStack","updateTextureStacksInternal","texturedNode","hasTextureInChannel","channelsToProcess","channelStrength","textureLayer","existingProcess","propName3d","getObjectPropertyAtPath","getTexture","updateTextureStacks","updateTextureProps","fileTextureLayer","applyUvProperties","clippingTexture","clippingTextureRenderer","includeSupressed","canUseSimpleTexture","activeLayers","originalTexture","_obj","propPath","ViewpointRenderer","unprocessedChangesCount","ChangeMap","RendererManager","getRenderMode","forcedRenderingMode","deferredChanges","processors","annotation","dimension","mateProcessor","sketchProcessor","isStable","getChangeMap","getDeletedNodes","runChanges","newChanges","getChangedNodes","allChangesByDid","allChangeCount","processedChangeMap","processedChanges","unchangedNodes","renderStart","changeList","includeOrphaned","isGlobalMaterialOrTexture","KbArcRotateCameraMouseWheelInput","computeDeltaFromMouseWheelLegacyEvent","mouseWheelLegacyEvent","KbArcRotateCamera","inertialOffsetPanningX","inertialOffsetPanningY","inertialOrthoZoom","clientInfo","mouseX","mouseY","screenLeft","screenTop","screenWidth","screenHeight","lockRotation","lockPanning","lockZoom","_orthoWidth","updateClient","orthoWidth","clampOrthoWidth","updateOrthographicPosition","orthoFit","startZoomLevel","zoomChange","shiftX","shiftY","clientRect","focalLength","disposeListeners","updateMousePosition","updateClientWrap","ANIMATION_FPS","eCameraType","eCameraOrientation","CameraTools","cameraMoving","_activeAnimation","_lastOrientation","objectsToScaleWithDistance","_cameraSettingsChange","copyCameraStateIntoViewpoint","viewChanged","param","allow","enableFreeLook","enableFreeMove","setPipeline","activePipeline","applyScenePostProcesses","frameCamera","pan","zoom","bounding","childrenBounding","nodeInfo","minBoundingBox","maxBoundingBox","bbCenter","bbCorners","cameraRay","cameraProjectionTargetPlane","cameraDistance","cameraPositionWorldOffset","planeProjectionX","planeProjectionY","bbCenterOffset","cameraTruePosition","cameraTrueTarget","xPlane","yPlane","bbCorner","xO","yO","desiredDistance","fovH","thV","thH","xP","xA","xR","yP","yA","yR","fireViewChangedOnNextRender","getScreenPositionOfNode","getScreenBoundingBoxOfNode","moveCameraToViewpoint","createCameraFromViewpoint","_activateCamera","animateToViewpoint","circEaseOut","newCameraTarget","ArcRotate","frameTarget","frameOptions","orbitRotation","targetRotation","targetPosition","activeAlpha","normalizeRotation","translateRotation","translatePosition","translateFov","translateTarget","translateA","crossesZero","shortestArc","translateB","translateR","translateOffset","translateOrthoWidth","storeCameraSettings","_storedViewpoint","restoreCameraSettings","watch","renderedNode","relativeSize","convertToCameraType","disposePreviousCamera","overhead","Universal","universalCamera","applyViewpointProperties","startWith","orient","_lastPerspectivePosition","newPos","normalizedPosition","rotDegrees","oneRotation","rotations","OctreeBlock","maxDepth","creationFunc","_boundingVectors","_capacity","_depth","_maxDepth","_creationFunc","_minPoint","_maxPoint","addEntry","createInnerBlocks","removeEntry","entryIndex","addEntries","select","selection","allowDuplicate","intersectsRay","_CreateBlocks","worldMin","worldMax","maxBlockCapacity","currentDepth","blockSize","localMin","localMax","Octree","dynamicContent","_maxBlockCapacity","_selectionContent","CreationFuncForMeshes","CreationFuncForSubMeshes","Dragger","_externalConnectors","_dragConnectors","_dragConnectorsByDynamicId","_dropPreviewConnectors","_dragFriends","_dragFriendMeshes","_dragFriendConnectors","_matedConnectors","startDrag","pointerArgs","draggable","lastX","lastY","totalMoved","_pointerObserver","movX","movY","startDragCommit","dragging","dropped","_draggable","_connectorRenderer","_dragNode","_dragNodeStartPosition","_dragMesh","_dragNodeInverseTransform","connectionMode","connectionType","connectRadius","dragOpacity","_onComplete","setupDragSurface","dragStart","getDragArgs","setupDragFriends","setupConnectors","setOpacity","newConnection","findConnection","_activeConnection","showDropPreview","hideDropPreview","processDragPosition","applyDragResultToNode","dragArgs","applyEuler","newRotationEuler","newKbQuat","newRotationAngle","newRotationAxis","otherConnector","withMates","dragConnector","previousEuler","toggleAllConnectors","resetOpacity","disposeDropPreview","_dragPlane","_dragSurfaces","_oct","_quadTree","_dropPreview","newMouseIntersect","intersectPointerToDragSurface","disableOffset","_startMouseIntersect","surfaceIds","surfaceNodes","sn","intersectDragSurface","surfacePick","newKbRotation","dragNode","startPos","potentialConnectors","dragConnectorMesh","dragConPos","intersectOctree","otherNode","otherConnectorMesh","canConnect","connectionsAreEqual","externalConnectorMeshes","parentsEnabled","dragConnectors","selectMany","extConnectorPositions","activeMates","getActiveMates","friend","friendNode","friendMesh","showConnectors","potentials","screenPos","dragMeshTransform","createDragPreview","clonedMeshGroup","kbnode","getConnectorLocalPoint","shouldSetOpacity","getAllMeshes","originalVisibility","ObjKb3dExporter","generateObj","shouldExportMesh","OBJExport","matlibname","globalposition","textureV","inverseTransform","trunkVerts","trunkNormals","trunkUV","trunkFaces","currentV","currentTextureV","textureIndices","blanks","faceUVs","OBJ","exportObj","objData","createObjBlob","_forceShowBoundingBoxes","_boundingBoxRenderer","BoundingBoxRenderer","_showBoundingBox","frontColor","backColor","showBackLines","onBeforeBoxRenderingObservable","onAfterBoxRenderingObservable","onResourcesReadyObservable","_fillIndexBuffer","_fillIndexData","_uniformBufferFront","_uniformBufferBack","_preActiveMesh","_tag","forceShowBoundingBoxes","_prepareResources","_colorShader","_colorShaderForOcclusionQuery","boxdata","boundingBoxIndex","_createWrappersForBoundingBox","median","drawWrapperBack","drawWrapperFront","renderOcclusionBoundingBox","_renderPassIdForOcclusionQuery","PivotTools","_PivotCached","_OldPivotPoint","_PivotPostMultiplyPivotMatrix","_PivotTranslation","_PivotTmpVector","PointerDragBehavior","currentDraggingPointerID","currentDraggingPointerId","onEnabledObservable","_useAlternatePickedPointAboveMaxDragAngleDragSpeed","_activeDragButton","maxDragAngle","dragButtons","_useAlternatePickedPointAboveMaxDragAngle","dragDeltaRatio","updateDragPlane","_moving","onDragObservable","onDragStartObservable","onDragEndObservable","moveAttached","startAndReleaseDragOnPointerEvents","detachCameraControls","useObjectOrientationForDragging","validateDrag","_alternatePickedPoint","_worldDragAxis","_targetPosition","_startDragRay","_lastPointerRay","_dragDelta","_pointC","_localAxis","_lookAt","optionCount","dragAxis","dragPlaneNormal","ownerNode","_PlaneScene","lastDragPosition","pickPredicate","_activePointerInfo","_startDrag","releaseDrag","_AnyMouseId","MouseEvent","_moveDrag","_beforeRenderObserver","needMatrixUpdate","_RemoveAndStorePivotPoint","_RestorePivotPoint","dragPlanePoint","fromRay","startPickedPoint","lastRay","_updateDragPlanePosition","_pickWithRayOnDragPlane","dragLength","dragDistance","dragPlanePosition","KbAxisMorphGizmo","gizmoPosition","dragActive","_updateQuat","snapDistance","updateObservable","wireframeMaterial","hoverMaterial","axisMesh","line2","sphereMesh","dragBehavior","isHovered","GizmoAnchorPoint","GizmoCoordinatesMode","Gizmo","_isHovered","attachedMesh","_attachedNode","_attachedNodeChanged","setCustomMesh","_customMeshSet","updateGizmoRotationToMatchAttachedMesh","_updateGizmoRotationToMatchAttachedMesh","updateGizmoPositionToMatchAttachedMesh","_updateGizmoPositionToMatchAttachedMesh","anchorPoint","_anchorPoint","Local","_updateScale","_customRotationQuaternion","Origin","_rightHandtoLeftHandMatrix","customRotationQuaternion","effectiveNode","Pivot","PreserveScaling","camForward","_handlePivot","attachedNodeTransform","_matrixChanged","worldMatrixUC","parentInv","localMat","invParent","boneLocalMatrix","nodeLocalMatrix","_setGizmoMeshMaterial","gizmoMeshes","gizmoAxisCache","pointerObserver","colliderMeshes","disableMaterial","AUTO_MESH_NAME","KbGizmo","updateGizmoPosition","attachTo","attachMeshAtNodeSpace","positionSpaceTransform","KbPlaneRotationGizmo","onSnapObservable","drag","rotationMesh","planeNormalTowardsCamera","localPlaneNormalTowardsCamera","tmpSnapEvent","currentSnapDragDistance","amountToRotate","attachedMeshParent","attachedMeshPosition","originalVector","camVec","snapped","dragSteps","quaternionCoefficient","deltaAngle","_attachedMeshChanged","KbAxisAngleGizmo","positionSpace","placementOffset","_positionSpace","_updateGizmoPosition","changeTarget","onNextFrame","targetAxis","targetAxisTransformed","axisGizmo","angleMesh","angleGizmo","disableAngle","zeroVector","tempVector2","axisChange","angleChange","deltaAxis","meshFaceInfo","MeshFaceEngine","updateNodeGeometry","getMeshFaces","expandSelection","limitAngle","faceInfo","selectedFaceIds","newSelection","recursiveExpandSelection","iterCount","neighbors","faceNeighbors","adjacentFaceId","facesByEdges","trianglesCount","triId","triOffset","edgeId","faceCount","eMarqueeSelectMode","Interaction","onClick$","onFacetClick$","onDoubleClick$","onPointerMove$","onPointerDown$","onPointerUp$","onKeyboard$","onMarqueeSelect","pointerIsDown","dragInProgress","selectionTolerance","cameraFacingDirection","meshFaceEngine","onClick","onFacetClick","onDoubleClick","onKeyboard","attachScene","handleMove","handleDoubleClick","getKeyboardArgs","handleKeyboard","setSelectionTolerance","setSelectionCulling","marqueeSelectOptions","cullSelection","enableMarqueeSelect","onlyOnShiftPress","selectedFaces","vertexList","disableMarqueeSelect","fillKb3dPickInfo","handleClickTypeEvent","handlerType","getNextNodeHandlers","propagate","getHandlerMap","selectedTriangle","facetArgs","clicks","getNextClickHandlers","getNextDraggables","hover","clearHover","stopPropagation","handleMarquee","unlockCamera","canvasX","canvasY","startRay","uiOverlay","selectBox","border","Facets","dragger","topOrigin","leftOrigin","rightOrigin","bottomOrigin","topNormal","leftNormal","rightNormal","bottomNormal","boxL","boxR","boxT","boxB","forwardDirection","sideDirection","topLeft","topRight","botLeft","botRight","fP","fN","pointPlaneSizeToRef","Vertices","meshPoint","getNextDoubleClickHandlers","dbSubmeshes","KbFacetSelectGizmo","__attachedMesh","onSelectionChange","currentSelection","selectionsEnabled","highlightOpacity","utilLayer","__attachedNode","aMesh","disposeSubject","getVertexGroups","visualizeSubmesh","selectedFacets","selectFacetGroups","startFacet","endFacet","enableSelectionChange","disableSelectionChange","submeshInfo","setBackfaceCulling","setTolerance","setVisualization","highlightMaterial","faceIds","geometryNeedsUpdate","transparantMaterial","vMesh","indexGroup","KbLightGizmo","lightNode","_attachedLight","posGizmo","rotGizmo","_helperMesh","lightVisual","KbAxisDragGizmo","_CreateArrow","localDelta","KbPositionGizmo","orientGizmo","xGizmo","yGizmo","zGizmo","_attachedSpaceNode","axisVisibility","positionChange","activeGizmos","KbRotationGizmo","corr","rotationChange","KbAxisScaleGizmo","uniformScaling","_coloredMaterial","arrowMesh","arrowTail","dragStrength","useGizmoMaterial","KbScaleGizmo","uniformScaleGizmo","scaleChange","uniformScalingMesh","octahedron","ColorGradient","getColorToRef","Color3Gradient","FactorGradient","factor1","factor2","getFactor","GradientHelper","updateFunc","gradientIndex","currentGradient","nextGradient","lastIndex","colorStep","lifeTime","age","cellIndex","_attachedSubEmitters","_currentColor1","_currentColor2","_currentSize1","_currentSize2","_currentAngularSpeed1","_currentAngularSpeed2","_currentVelocity1","_currentVelocity2","_currentLimitVelocity1","_currentLimitVelocity2","_currentDrag1","_currentDrag2","_Count","_updateCellInfoFromSystem","updateCellIndex","offsetAge","changeSpeed","_randomCellOffset","_initialEndSpriteCellID","_initialStartSpriteCellID","_initialSpriteCellLoop","_inheritParticleInfoToSubEmitter","subEmitter","emitterMesh","inheritDirection","inheritedVelocityAmount","_inheritedVelocityOffset","_inheritParticleInfoToSubEmitters","_currentColorGradient","_currentSizeGradient","_currentAngularSpeedGradient","_currentVelocityGradient","_currentLimitVelocityGradient","_currentDragGradient","_initialDirection","useRampGradients","remapData","_randomNoiseCoordinates1","_randomNoiseCoordinates2","SubEmitterType","SubEmitter","END","_disposeEmitterOnDispose","serializeTexture","doNotStart","_ParseParticleSystem","ParticleSystem","_useRampGradients","_resetEffect","_particles","getActiveCount","isStopping","isAlive","getCustomEffect","_customWrappers","_getCustomDrawWrapper","_useInstancing","onBeforeDrawParticlesObservable","_onBeforeDrawParticlesObservable","vertexShaderName","customEffect","_emitterInverseWorldMatrix","onStoppedObservable","_stockParticles","_newPartsExcess","_scaledColorStep","_colorDiff","_scaledDirection","_scaledGravity","_started","_actualFrame","_currentEmitRate1","_currentEmitRate2","_currentStartSize1","_currentStartSize2","updateInAnimate","_rawTextureWidth","isGPU","lastParticle","_createParticle","_subEmitters","subEmitters","ATTACHED","newEmitter","_emitFromParticle","templateIndex","subSystem","_rootParticleSystem","activeSubSystems","defaultProjectionMatrix","_vertexArrayObject","noiseTextureData","updateFunction","noiseTextureSize","sameParticleArray","scaledUpdateSpeed","_scaledUpdateSpeed","previousAge","GetCurrentGradient","directionScale","limitVelocity","_emitterWorldMatrix","fetchedColorR","_fetchR","fetchedColorG","fetchedColorB","scaledForce","_addFactorGradient","factorGradients","newGradient","_removeFactorGradient","factorGradient","addLifeTimeGradient","removeLifeTimeGradient","addSizeGradient","removeSizeGradient","addColorRemapGradient","removeColorRemapGradient","addAlphaRemapGradient","removeAlphaRemapGradient","addAngularSpeedGradient","removeAngularSpeedGradient","addVelocityGradient","removeVelocityGradient","addLimitVelocityGradient","removeLimitVelocityGradient","addDragGradient","removeDragGradient","addEmitRateGradient","removeEmitRateGradient","addStartSizeGradient","removeStartSizeGradient","_createRampGradientTexture","_rampGradientsTexture","getRampGradients","forceRefreshGradients","_syncRampGradientTexture","addRampGradient","removeRampGradient","addColorGradient","colorGradient","removeColorGradient","drawWrappers","_vertexBuffer","_spriteBuffer","_vertexBufferSize","BILLBOARDMODE_STRETCHED","BILLBOARDMODE_STRETCHED_LOCAL","vertexSize","_vertexData","cellIndexBuffer","directionBuffer","rampDataBuffer","spriteData","getCapacity","_alive","_prepareSubEmitterInternalArray","_currentEmitRateGradient","_currentStartSizeGradient","noiseTextureAsProcedural","stopSubEmitters","_stopSubEmitters","_appendParticleVertex","initialDirection","_removeFromRoot","newParticles","emitterPosition","factorGradient1","factorGradient2","lifeTime1","lifeTime2","emitPower","attributeNamesOrOptions","effectCreationOption","fillUniformsAttributesAndSamplerNames","_GetAttributeNamesOrOptions","_GetEffectCreationOptions","_getWrapper","customWrapper","preWarmOnly","_appendParticleVertices","particleTexture","defaultViewMatrix","baseSize","outparticles","cloneTexture","shaderOptions","serialization","emitterId","colorGradients","serializedGradient","rampGradients","colorRemapGradients","colorRemapGradient","alphaRemapGradients","alphaRemapGradient","sizeGradients","sizeGradient","angularSpeedGradients","angularSpeedGradient","velocityGradients","velocityGradient","dragGradients","dragGradient","emitRateGradients","emitRateGradient","startSizeGradients","startSizeGradient","lifeTimeGradients","lifeTimeGradient","limitVelocityGradients","limitVelocityGradient","parsedParticleSystem","emitterType","cellArray","KbVertexSelectGizmo","drawVertices","vertexTracker","particleScale","redrawVertices","vertexSelectedColor","vertexUnSelectedColor","vertexTransparentColor","verticesUtilLayer","disposeMarquee","selectionMode","redraw","emitSelectionChange","updateParticleScale","selectVertices","getBarycenter","saveAnimate","particlesToProcess","CreateRGBTexture","KbViewpointGizmo","targetGizmo","_visualMesh","previousTargetMode","_gizmoMesh","_targetVisualMesh","updateHelperMeshes","updateHelperMeshDirectionToRotation","_attachedViewpoint","targetChange","deltaTargetPos","deltaRotation","viewpointNode","eulerRot","FromEulerVector","recreate","PREVIEW_NAME","GizmoTools","activeGizmoSettings","uvMapStoredMaterials","selectLight","lightGizmo","totalDeltaPosition","subjects","updateLightGizmo","selectViewpoint","viewpointGizmo","updateTargetPosition","updateRotation","totalDeltaTargetPosition","totalDeltaRotation","totalDeltaPositionQueue","totalDeltaTargetPositionQueue","totalDeltaRotationQueue","updateViewpointGizmo","positionGizmo","totalDelta","totalDeltaQueue","clearGizmo","rotateAxisAngle","updateAxisAngle","axisAngleGizmo","updateAxis","positionTransform","rotationGizmo","axisTranslate","gizmoRotation","worldTransformMatrix","positionTransformInv","positionDelta","axisDelta","worldNodePivot","nodeTransform","scaleGizmo","vertexSelectionGizmo","updateGizmoPositions","selectFacets","facetSelectionGizmo","gizmoSettings","showUvMapMaterial","_showMaterial","previousMaterial","previewMaterial","previewTexture","storedMaterial","filterToChangedValues","DefaultSketchOptions","faceCamera","snapSize","activeHitPoint","sketchPlaneIntersect","SketchEditor","defaultHighlightColor","mouseUp","sketchPoints","updateActiveOptions","currentOptions","startOnPlane","userOptions","useSketchNormal","activeSketch","currentCanvas","previewVisibleOriginal","sketchPlane","originalCamera","sketchCamera","dragObjects","lastMousePosition","sketchParent","getPointOnPlane","deletedPoints","collisionSphere","pointRendered","cpCenter","cpHandle1","cpHandle2","handle","transformInv","dragObject","movement","rightClickMenu","showContextMenu","getRightClickButtonsForLines","getRightClickButtons","isOpen","hideContextMenu","rendererNode","deleteSketchPoint","deleteSketchPath","separateSketchNode","sketchpathType","cpIndex","pointOnePosition","pointTwoPosition","selectedControlPoints","selectedPath","nextPath","nextPathPoints","addSketchNodeOnPath","spliceIndex","sketchControlPoint","eSnapshotType","SnapshotModule","takeAsBlob","takeAsDataUrl","viewpointSet","framePadding","snapshotCanvas","renderContext","newWidth","newHeight","renderingCanvas","dataurl","mime","bstr","u8arr","dataURLtoBlob","clickHandler","Highlighter","cameraManager","colorMap","nodeColors","highlighterDb","highlightPipeline","enableHighlightDebug","highlightEffect","startPostProcess","renderNode","addHighlightToRendered","activeHover","removeHighlightFromRendered","forceRenderOnNextFrame","clearAll","colorMapKey","colorToString","addNodeToColorMap","colorKey","removeNodeFromColorMap","colorMapEntry","newShader","RightClickMenu","contextMenuOpen","xPosition","yPosition","createContextMenu","contextmenu","contextMenu","zIndex","stopImmediatePropagation","newButton","innerText","FADE_TIME","KbViewer","pauseRender","normalSmoothingWarnings","fpsTracker","forceRunRenderer","positionTrackedNodes","nextFrameCallbacks","_renderLoopRunning","allAssetsLoaded","additionalLoadBlockers","enableScreenshots","webworkerPath","iOS","createScene","gizmos","sketchEditor","snapshot","objExporter","loadScene","createLoadingExperience","startRender","deferLoadingPromise","http","hideLoadingExperience","forceDisableGraphicsOptimization","waitForNextKb3dRender","optimizeScene","updateMeshWarnings","updateMaterialWarnings","updateAnimationWarnings","getFPS","getWarnings","warningMessages","totalSpotLights","totalPointLights","shadowCastingLights","tempWarning","lightsWarningMessages","updateWarnings","bigTextures2048","bigTextures4096","textureWarningMessages","subMaterialTextures","totalPolygons","recursiveGetPolygons","meshCastAndReceiveShadows","meshWarningMessages","polygonCount","warning","localMaterialCount","getLocalMaterialsRecursive","materialWarningMessages","tempArray","extractMeshGeometry","applySvgToSketch","overwriteSketchWithSvg","draggingEvent","mousePosition","lastPickedMesh","dragOver","resizeObserver","getBabylonJson","childData","preRender","frameCount","skipped","stopRender","createOverlay","ResizeObserver","pointerEvents","loadingOverlay","backgroundSize","backgroundRepeat","backgroundPosition","backgroundImage","fade","spinner","alignItems","justifyContent","sceneOptimizerOptions","sceneOptimizer","startSceneRenderAutoStop","createDefaultEnv","waitForMaterials","materialService","somePromise","nodesIntersect","node1","node2","bb1","bb2","getReferenceNodes","getTransformSpaceDifference","originNode","destinationNode","destinationMesh","originMatrix","destinationMatrix","centerMeshToOrigin","kb3dEnums","BoundingBoxRelativeHelperKbVector","convertAbsoluteToRelative","convertAbsoluteDirectionToRelative","NEW_GEOMETRY","Quad","Validator","vmap","nodeIsValid","getErrors","clearPropError","clearPropErrorWhere","clearNode","sprop","existingErrors","applyTransformToVector","bakeMeshTransform","newTransform","convertHdrToEnv","hdrFile","hdrSize","dataUri","dummyEngine","degreeToRadian","vectorDegree","optionalResult","getDimensionLinePoints","lineStart","lineEnd","getToken","tryWithKey","isCircularPattern","isInstanceOf","isLinearPattern","isObject$1","isPattern","isSubmesh","isTrackedArray","isTrackedMeshNode","isViewpointNode","kb3dTest","btnAnimate","quadraticIn","quadraticOut","btnStop","txtJson","txtCounter","btnToJson","updateJsonLength","btnToFullJson","btnFromJson","btnBabylonJson","btnWidth","btnMaterialColor","btnSaveMesh","myArray","longInt8View","btnGetSavedMesh","oReq","makeRotationAxis","mathClamp","mergeVertices","newVertexGroup","mergeVerticesIntoAll","nextVertexGroup","mergedVertexGroup","ofToken","omitUndefinedElements","removeVerticesFromAll","toPixels","viewportUnits"],"mappings":"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CAAC4nhB,MAAO,KAAQ5nhB,IAAK,MACrB,CAAC4nhB,MAAO,KAAQ5nhB,IAAK,MACrB,CAAC4nhB,MAAO,KAAQ5nhB,IAAK,MACrB,CAAC4nhB,MAAO,KAAQ5nhB,IAAK,MACrB,CAAC4nhB,MAAO,KAAQ5nhB,IAAK,MACrB,CAAC4nhB,MAAO,KAAQ5nhB,IAAK,MACrB,CAAC4nhB,MAAO,KAAQ5nhB,IAAK,MACrB,CAAC4nhB,MAAO,KAAQ5nhB,IAAK,MACrB,CAAC4nhB,MAAO,KAAQ5nhB,IAAK,MACrB,CAAC4nhB,MAAO,KAAQ5nhB,IAAK,MACrB,CAAC4nhB,MAAO,KAAQ5nhB,IAAK,MACrB,CAAC4nhB,MAAO,KAAQ5nhB,IAAK,MACrB,CAAC4nhB,MAAO,KAAQ5nhB,IAAK,MACrB,CAAC4nhB,MAAO,KAAQ5nhB,IAAK,MACrB,CAAC4nhB,MAAO,KAAQ5nhB,IAAK,MACrB,CAAC4nhB,MAAO,KAAQ5nhB,IAAK,OACrB,CAAC4nhB,MAAO,MAAQ5nhB,IAAK,OACrB,CAAC4nhB,MAAO,MAAQ5nhB,IAAK,OACrB,CAAC4nhB,MAAO,MAAQ5nhB,IAAK,OACrB,CAAC4nhB,MAAO,MAAQ5nhB,IAAK,OACrB,CAAC4nhB,MAAO,MAAQ5nhB,IAAK,OACrB,CAAC4nhB,MAAO,MAAQ5nhB,IAAK,OACrB,CAAC4nhB,MAAO,MAAQ5nhB,IAAK,OACrB,CAAC4nhB,MAAO,MAAQ5nhB,IAAK,OACrB,CAAC4nhB,MAAO,MAAQ5nhB,IAAK,OACrB,CAAC4nhB,MAAO,MAAQ5nhB,IAAK,OACrB,CAAC4nhB,MAAO,MAAS5nhB,IAAK,OACtB,CAAC4nhB,MAAO,MAAQ5nhB,IAAK,OACrB,CAAC4nhB,MAAO,MAAQ5nhB,IAAK,OACrB,CAAC4nhB,MAAO,MAAQ5nhB,IAAK,OACrB,CAAC4nhB,MAAO,MAAQ5nhB,IAAK,OACrB,CAAC4nhB,MAAO,MAAQ5nhB,IAAK,OACrB,CAAC4nhB,MAAO,MAAQ5nhB,IAAK,OACrB,CAAC4nhB,MAAO,MAAQ5nhB,IAAK,OACrB,CAAC4nhB,MAAO,MAAQ5nhB,IAAK,OACrB,CAAC4nhB,M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EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAAW,EAAG,EAAG,EAAG,EAAG,GACtCwzjB,GAAUrjjB,UAAKnQ,EAA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","file":"viewer.min.js","sourcesContent":["var kb3d = (function (exports) {\n 'use strict';\n\n String.prototype.capitalize = function () {\n return this.charAt(0).toUpperCase() + this.slice(1);\n };\n const splitWordsRegex = /[A-Z]{2,}(?=[A-Z][a-z]+[0-9]*|\\b)|[A-Z]?[a-z]+[0-9]*|[A-Z]|[0-9]+/g;\n String.prototype.splitWords = function () {\n return this.match(splitWordsRegex) || [];\n };\n String.prototype.isEqual = function (str) {\n return (this ? this.toUpperCase() : undefined) === (str ? str.toUpperCase() : undefined);\n };\n String.prototype.equalsAny = function (...strings) {\n return strings && strings.some(s => s.isEqual(this));\n };\n String.prototype.toSentence = function () {\n return this.replace(/([A-Z])/g, ' $1') // insert a space before all caps\n .replace(/^./, function (str) {\n return str.toUpperCase();\n }) // uppercase the first character\n .trim(); //trip leading and trainling whitespace\n };\n Object.defineProperty(Array.prototype, 'sortBy', {\n enumerable: false,\n writable: true,\n value: function (predicate) {\n return this.sort((a, b) => {\n if (predicate(a) < predicate(b)) {\n return -1;\n }\n else if (predicate(a) > predicate(b)) {\n return 1;\n }\n else {\n return 0;\n }\n });\n },\n });\n Object.defineProperty(Array.prototype, 'remove', {\n enumerable: false,\n writable: true,\n value: function (item) {\n const index = this.indexOf(item);\n if (index != -1) {\n return this.splice(this.indexOf(item), 1);\n }\n },\n });\n Object.defineProperty(Array.prototype, 'removeWhere', {\n enumerable: false,\n writable: true,\n value: function (predicate) {\n const removed = new Array();\n for (let i = this.length - 1; i >= 0; i--) {\n const item = this[i];\n if (predicate(item)) {\n removed.push(...this.splice(i, 1));\n }\n }\n return removed;\n },\n });\n Object.defineProperty(Array.prototype, 'contains', {\n enumerable: false,\n writable: true,\n value: function (item) {\n const index = this.indexOf(item);\n return index != -1;\n },\n });\n Object.defineProperty(Array.prototype, 'selectMany', {\n enumerable: false,\n writable: true,\n value: function (selector) {\n return this.reduce((out, v) => {\n let vres = selector(v);\n if (vres != null) {\n if (!Array.isArray(vres))\n vres = [vres];\n out.push(...vres);\n }\n return out;\n }, []);\n },\n });\n Object.defineProperty(Array.prototype, 'first', {\n enumerable: false,\n writable: true,\n value: function () {\n let item;\n if (this.length) {\n item = this[0];\n }\n return item;\n },\n });\n Object.defineProperty(Array.prototype, 'last', {\n enumerable: false,\n writable: true,\n value: function () {\n if (this.length) {\n return this[this.length - 1];\n }\n return undefined;\n },\n });\n Object.defineProperty(Array.prototype, 'getCommonOrNull', {\n enumerable: false,\n writable: true,\n value: function (valueDelegate) {\n const arr = this;\n if (this.length && arr.every(i => valueDelegate(i) == valueDelegate(arr[0]))) {\n return valueDelegate(arr[0]);\n }\n return null;\n },\n });\n Object.defineProperty(Array.prototype, 'toMap', {\n enumerable: false,\n writable: true,\n value: function (keySelector) {\n const m = new Map();\n for (const i of this) {\n m.set(keySelector(i), i);\n }\n return m;\n },\n });\n Object.defineProperty(Array.prototype, 'clear', {\n enumerable: false,\n writable: true,\n value: function () {\n if (this.length > 0) {\n this.splice(0, this.length);\n }\n },\n });\n Object.defineProperty(Array.prototype, 'reorder', {\n enumerable: false,\n writable: true,\n value: function (fromIndex, toIndex, undo) {\n if (undo) {\n if (fromIndex > toIndex) {\n fromIndex--;\n }\n else {\n toIndex++;\n }\n }\n const itemToMove = this[fromIndex];\n for (let i = fromIndex; i < this.length - 1; i++) {\n this[i] = this[i + 1];\n }\n if (toIndex > fromIndex) {\n toIndex--;\n }\n for (let i = this.length - 1; i > toIndex; i--) {\n this[i] = this[i - 1];\n }\n this[toIndex] = itemToMove;\n },\n });\n Object.defineProperty(Set.prototype, 'find', {\n enumerable: false,\n writable: true,\n value: function (predicate) {\n for (const i of this) {\n if (predicate(i))\n return i;\n }\n return undefined;\n },\n });\n Object.defineProperty(Map.prototype, 'hasAny', {\n enumerable: false,\n writable: true,\n value: function (...keys) {\n return keys.some(k => this.has(k));\n },\n });\n\n /******************************************************************************\r\n Copyright (c) Microsoft Corporation.\r\n\r\n Permission to use, copy, modify, and/or distribute this software for any\r\n purpose with or without fee is hereby granted.\r\n\r\n THE SOFTWARE IS PROVIDED \"AS IS\" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH\r\n REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY\r\n AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,\r\n INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM\r\n LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR\r\n OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR\r\n PERFORMANCE OF THIS SOFTWARE.\r\n ***************************************************************************** */\r\n\r\n function __decorate(decorators, target, key, desc) {\r\n var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;\r\n if (typeof Reflect === \"object\" && typeof Reflect.decorate === \"function\") r = Reflect.decorate(decorators, target, key, desc);\r\n else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;\r\n return c > 3 && r && Object.defineProperty(target, key, r), r;\r\n }\r\n\r\n function __param(paramIndex, decorator) {\r\n return function (target, key) { decorator(target, key, paramIndex); }\r\n }\r\n\r\n var _SuppressedError = typeof SuppressedError === \"function\" ? SuppressedError : function (error, suppressed, message) {\r\n var e = new Error(message);\r\n return e.name = \"SuppressedError\", e.error = error, e.suppressed = suppressed, e;\r\n };\n\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n class Token {\n constructor(key) {\n this.key = key;\n }\n static withKey(key) {\n const token = this[key];\n if (!token)\n throw new Error(`Could not find token for key '${key}'`);\n return token;\n }\n static tryWithKey(key) {\n const token = this[key];\n return token;\n }\n static MapFrom(...tokens) {\n return new Map(tokens.map(t => [t, null]));\n }\n isPrimitive() {\n return Token.PrimitiveTokenMap.has(this);\n }\n isPrimitiveMesh() {\n return Token.PrimitiveMeshTokenMap.has(this);\n }\n isMesh() {\n return Token.MeshTokenMap.has(this);\n }\n isMeshOrGroup() {\n return Token.MeshAndGroupTokenMap.has(this);\n }\n isMaterial() {\n return Token.MaterialTokenMap.has(this);\n }\n isFeature() {\n return Token.FeatureTokenMap.has(this);\n }\n isTexture() {\n return Token.TextureLayerTokenMap.has(this);\n }\n isLight() {\n return Token.LightTokenMap.has(this);\n }\n isViewpoint() {\n return Token.ViewpointNode === this;\n }\n isMate() {\n return Token.MateNode === this;\n }\n isConnector() {\n return Token.Connector === this || Token.SketchControlPoint === this;\n }\n isAnnotation() {\n return Token.AnnotationTokenMap.has(this);\n }\n isNode() {\n return Token.NodeTokenMap.has(this);\n }\n isSceneNode() {\n return Token.SceneNode === this;\n }\n isSketchPath() {\n return Token.SketchPath === this;\n }\n }\n /** Services */\n Token.HttpService = new Token('HttpService');\n Token.MaterialService = new Token('MaterialService');\n Token.MeshService = new Token('MeshService');\n Token.JobService = new Token('JobService');\n Token.SceneService = new Token('SceneService');\n Token.FontService = new Token('FontService');\n Token.Ignore = new Token('Ignore');\n /** Roots */\n Token.SceneNode = new Token('SceneNode');\n Token.Kb3dObject = new Token('Kb3dObject');\n Token.ImportInfo = new Token('ImportInfo');\n /** Lights */\n Token.LightNode = new Token('LightNode');\n Token.DirectionalLightNode = new Token('DirectionalLightNode');\n Token.HemisphericLightNode = new Token('HemisphericLightNode');\n Token.PointLightNode = new Token('PointLightNode');\n Token.SpotLightNode = new Token('SpotLightNode');\n /** Meshes */\n Token.BoxNode = new Token('BoxNode');\n Token.CustomMeshNode = new Token('CustomMeshNode');\n Token.CylinderNode = new Token('CylinderNode');\n Token.ConeNode = new Token('ConeNode');\n Token.DiscNode = new Token('DiscNode');\n Token.PlaneNode = new Token('PlaneNode');\n Token.SphereNode = new Token('SphereNode');\n Token.TorusNode = new Token('TorusNode');\n Token.MeshNode = new Token('MeshNode');\n Token.PrimitiveMeshNode = new Token('PrimitiveMeshNode');\n Token.MeshGroupNode = new Token('MeshGroupNode');\n Token.SpaceNode = new Token('SpaceNode');\n Token.TextNode = new Token('TextNode');\n Token.SketchNode = new Token('SketchNode');\n Token.SketchPath = new Token('SketchPath');\n /** Dimensions */\n Token.AnnotationNode = new Token('AnnotationNode');\n Token.DimensionNode = new Token('DimensionNode');\n /** Primitives */\n Token.KbColor = new Token('KbColor');\n // public static KbColor4 = new Token(\"KbColor4\");\n Token.KbGeometry = new Token('KbGeometry');\n Token.KbQuaternion = new Token('KbQuaternion');\n Token.KbVector = new Token('KbVector');\n /** Materials */\n Token.MaterialNode = new Token('MaterialNode');\n Token.AdvancedMaterialNode = new Token('AdvancedMaterialNode');\n Token.MaterialMapping = new Token('MaterialMapping');\n /** Textures */\n Token.TextureLayer = new Token('TextureLayer');\n Token.FileTextureLayer = new Token('FileTextureLayer');\n Token.SvgTextureLayer = new Token('SvgTextureLayer');\n Token.TextTextureLayer = new Token('TextTextureLayer');\n Token.ColorFillTextureLayer = new Token('ColorFillTextureLayer');\n /** Viewpoints */\n Token.ViewpointNode = new Token('ViewpointNode');\n /** Features */\n Token.Feature = new Token('Feature');\n Token.BaseConnector = new Token('BaseConnector');\n Token.Connector = new Token('Connector');\n Token.SketchControlPoint = new Token('SketchControlPoint');\n Token.SubmeshFeature = new Token('SubmeshFeature');\n Token.MoveVerticesFeature = new Token('MoveVerticesFeature');\n Token.WeldVerticesFeature = new Token('WeldVerticesFeature');\n Token.DisplacementMapFeature = new Token('DisplacementMapFeature');\n Token.UvMapFeature = new Token('UvMapFeature');\n Token.LinearPatternFeature = new Token('LinearPatternFeature');\n Token.CircularPatternFeature = new Token('CircularPatternFeature');\n Token.JoinGeometryFeature = new Token('JoinGeometryFeature');\n Token.DeleteFacesFeature = new Token('DeleteFacesFeature');\n Token.NormalSmoothingFeature = new Token('NormalSmoothingFeature');\n Token.SliceFeature = new Token('SliceFeature');\n Token.MirrorFeature = new Token('MirrorFeature');\n Token.TransformFeature = new Token('TransformFeature');\n Token.ExtrudeFeature = new Token('ExtrudeFeature');\n Token.SweepFeature = new Token('SweepFeature');\n Token.MateNode = new Token('MateNode');\n Token.PrimitiveTokens = [Token.KbVector, Token.KbQuaternion, Token.KbColor];\n Token.PrimitiveTokenMap = Token.MapFrom(...Token.PrimitiveTokens);\n Token.PrimitiveMeshTokens = [\n Token.BoxNode,\n Token.CylinderNode,\n Token.ConeNode,\n Token.DiscNode,\n Token.PlaneNode,\n Token.SphereNode,\n Token.TorusNode,\n ];\n Token.PrimitiveMeshTokenMap = Token.MapFrom(...Token.PrimitiveMeshTokens);\n Token.MeshTokens = Token.PrimitiveMeshTokens.concat(Token.CustomMeshNode, Token.SketchNode, Token.TextNode);\n Token.MeshTokenMap = Token.MapFrom(...Token.MeshTokens);\n Token.MeshAndGroupTokens = Token.MeshTokens.concat(Token.MeshGroupNode);\n Token.MeshAndGroupTokenMap = Token.MapFrom(...Token.MeshAndGroupTokens);\n Token.SpaceTokens = Token.MeshAndGroupTokens.concat(Token.SceneNode);\n Token.SpaceTokenMap = Token.MapFrom(...Token.SpaceTokens);\n Token.MaterialTokens = [Token.AdvancedMaterialNode];\n Token.MaterialTokenMap = Token.MapFrom(...Token.MaterialTokens);\n Token.AnnotationTokens = [Token.AnnotationNode, Token.DimensionNode];\n Token.AnnotationTokenMap = Token.MapFrom(...Token.AnnotationTokens);\n Token.PositionalLightTokens = [Token.PointLightNode, Token.SpotLightNode];\n Token.GlobalLightTokens = [Token.DirectionalLightNode, Token.HemisphericLightNode];\n Token.LightTokens = [...Token.GlobalLightTokens, ...Token.PositionalLightTokens];\n Token.LightTokenMap = Token.MapFrom(...Token.LightTokens);\n Token.FeatureTokens = [\n Token.MoveVerticesFeature,\n Token.WeldVerticesFeature,\n Token.DisplacementMapFeature,\n Token.UvMapFeature,\n Token.SubmeshFeature,\n Token.LinearPatternFeature,\n Token.CircularPatternFeature,\n Token.JoinGeometryFeature,\n Token.DeleteFacesFeature,\n Token.NormalSmoothingFeature,\n Token.SliceFeature,\n Token.MirrorFeature,\n Token.TransformFeature,\n Token.ExtrudeFeature,\n Token.SweepFeature,\n ];\n Token.FeatureTokenMap = Token.MapFrom(...Token.FeatureTokens);\n Token.ConnectorTokens = [Token.Connector, Token.SketchControlPoint];\n Token.TextureLayerTokens = [\n Token.TextureLayer,\n Token.FileTextureLayer,\n Token.SvgTextureLayer,\n Token.TextTextureLayer,\n Token.ColorFillTextureLayer,\n ];\n Token.TextureLayerTokenMap = Token.MapFrom(...Token.TextureLayerTokens);\n Token.NodeTokens = [\n ...Token.SpaceTokens,\n ...Token.MaterialTokens,\n ...Token.LightTokens,\n ...Token.AnnotationTokens,\n Token.ViewpointNode,\n Token.MateNode,\n ];\n Token.NodeTokenMap = Token.MapFrom(...Token.NodeTokens);\n Token.BaseNodeTokens = [\n Token.SpaceNode,\n Token.MaterialNode,\n Token.LightNode,\n Token.ViewpointNode,\n Token.MateNode,\n Token.AnnotationNode,\n ];\n Token.AllTokens = Token.NodeTokens.concat(...Token.PrimitiveTokens, Token.FeatureTokens, Token.TextureLayerTokens, Token.ConnectorTokens, [Token.SketchPath]);\n Token.AllTokenMap = Token.MapFrom(...Token.AllTokens);\n\n const Kb3dModels = new Map();\n /**\n * Curry function to filter items in an array to items of a token type\n * @param classKey String key of the class you want to see if the instance is an instance of\n */\n function ofToken(token) {\n return (instance) => instanceOf(instance, token);\n }\n /**\n * Checks if the instance of a class given is an instance of the class represented by it's\n * injectable class key. If the instance is an instance of a class that was not\n * decorated, or if the class represented by the key was not decorated, then this will\n * return false!\n * @param instance An instance of a class that was decorated\n * @param classKey String key of the class you want to see if the instance is an instance of\n */\n function instanceOf(instance, token) {\n if (!instance)\n return false;\n const model = Kb3dModels.get(token);\n return model ? instance instanceof model.classConstructor : false;\n }\n /** Curry-able version of instanceOf */\n function isInstanceOf(token) {\n return (instance) => instanceOf(instance, token);\n }\n function getToken(object) {\n const node = object;\n if (node.$type) {\n return Token.tryWithKey(node.$type);\n }\n }\n /**\n * Checks if the instance of a class given is an instance of the class represented by it's\n * injectable class key. If the instance is an instance of a class that was not\n * decorated, or if the class represented by the key was not decorated, then this will\n * return false!\n * @param instance An instance of a class that was decorated\n * @param classKey String key of the class you want to see if the instance is an instance of\n */\n function instanceOfAny(instance, ...tokens) {\n if (!instance)\n return false;\n return tokens.some(t => instanceOf(instance, t));\n }\n\n (function (EChangeOperation) {\n // CreateAndPush and Remove correspond to functions in the custom KBArray class.\n // They use Push and Splice internally but also signify we need a\n // changeTracker destroy on undo and redo respectively and create on redo and undo respectively.\n EChangeOperation[EChangeOperation[\"Push\"] = 0] = \"Push\";\n EChangeOperation[EChangeOperation[\"Pop\"] = 1] = \"Pop\";\n EChangeOperation[EChangeOperation[\"Shift\"] = 2] = \"Shift\";\n EChangeOperation[EChangeOperation[\"Unshift\"] = 3] = \"Unshift\";\n EChangeOperation[EChangeOperation[\"Splice\"] = 4] = \"Splice\";\n EChangeOperation[EChangeOperation[\"CreateAndPush\"] = 5] = \"CreateAndPush\";\n EChangeOperation[EChangeOperation[\"Remove\"] = 6] = \"Remove\";\n })(exports.EChangeOperation || (exports.EChangeOperation = {}));\n function isTrackedClass(object) {\n return object && typeof object === 'object' && '$changesTracked' in object;\n }\n function isTrackedArray(object) {\n return !!object.$isTracked;\n }\n function isTrackedInstance(instance, token) {\n if (!instance)\n return false;\n const model = Kb3dModels.get(token);\n return model ? instance instanceof model.classConstructor && isTrackedClass(instance) : false;\n }\n function isTrackedInstanceOfAny(instance, ...tokens) {\n if (!instance)\n return false;\n return tokens.some(t => isTrackedInstance(instance, t));\n }\n\n const changedItems = new Set();\n const deletedItems = new Set();\n /** This class syncs changes to the renderer to update the scene view */\n class SyncService {\n static registerChangedState(tracker, modified) {\n if (modified) {\n // If node was deleted, ignore further changes, unless the item was moved to a different parent\n if (!deletedItems.has(tracker) || tracker.changes.has('_parent')) {\n changedItems.add(tracker);\n }\n }\n else {\n changedItems.delete(tracker);\n }\n }\n // When a new object is being created, the proxy being used to set the values will be the proxy of the parent of the class being created.\n // So, we need to make sure the changed items map is always pointing to the correct proxy (the proxy with the highest specificity)\n static registerNewObject(tracker) {\n changedItems.add(tracker);\n }\n static registerDeleted(tracker) {\n deletedItems.add(tracker);\n }\n static get hasChanges() {\n return changedItems.size > 0 || deletedItems.size > 0;\n }\n static getChangedNodes() {\n return [...changedItems.values()];\n }\n static getDeletedNodes() {\n return [...deletedItems.values()];\n }\n static nodeDeletionProcessed(tracker) {\n //when a node is deleted, we remove it from all the maps\n changedItems.delete(tracker);\n deletedItems.delete(tracker);\n }\n static nodeChangesProcessed(tracker) {\n //when node changes have been processed, we only remove it from the changedItems map\n changedItems.delete(tracker);\n }\n }\n\n /*! *****************************************************************************\r\n Copyright (c) Microsoft Corporation.\r\n\r\n Permission to use, copy, modify, and/or distribute this software for any\r\n purpose with or without fee is hereby granted.\r\n\r\n THE SOFTWARE IS PROVIDED \"AS IS\" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH\r\n REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY\r\n AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,\r\n INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM\r\n LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR\r\n OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR\r\n PERFORMANCE OF THIS SOFTWARE.\r\n ***************************************************************************** */\r\n /* global Reflect, Promise */\r\n\r\n var extendStatics = function(d, b) {\r\n extendStatics = Object.setPrototypeOf ||\r\n ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||\r\n function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; };\r\n return extendStatics(d, b);\r\n };\r\n\r\n function __extends(d, b) {\r\n extendStatics(d, b);\r\n function __() { this.constructor = d; }\r\n d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());\r\n }\n\n /** PURE_IMPORTS_START PURE_IMPORTS_END */\n function isFunction(x) {\n return typeof x === 'function';\n }\n\n /** PURE_IMPORTS_START PURE_IMPORTS_END */\n var _enable_super_gross_mode_that_will_cause_bad_things = false;\n var config = {\n Promise: undefined,\n set useDeprecatedSynchronousErrorHandling(value) {\n if (value) {\n var error = /*@__PURE__*/ new Error();\n /*@__PURE__*/ console.warn('DEPRECATED! RxJS was set to use deprecated synchronous error handling behavior by code at: \\n' + error.stack);\n }\n _enable_super_gross_mode_that_will_cause_bad_things = value;\n },\n get useDeprecatedSynchronousErrorHandling() {\n return _enable_super_gross_mode_that_will_cause_bad_things;\n },\n };\n\n /** PURE_IMPORTS_START PURE_IMPORTS_END */\n function hostReportError(err) {\n setTimeout(function () { throw err; }, 0);\n }\n\n /** PURE_IMPORTS_START _config,_util_hostReportError PURE_IMPORTS_END */\n var empty = {\n closed: true,\n next: function (value) { },\n error: function (err) {\n if (config.useDeprecatedSynchronousErrorHandling) {\n throw err;\n }\n else {\n hostReportError(err);\n }\n },\n complete: function () { }\n };\n\n /** PURE_IMPORTS_START PURE_IMPORTS_END */\n var isArray = /*@__PURE__*/ (function () { return Array.isArray || (function (x) { return x && typeof x.length === 'number'; }); })();\n\n /** PURE_IMPORTS_START PURE_IMPORTS_END */\n function isObject(x) {\n return x !== null && typeof x === 'object';\n }\n\n /** PURE_IMPORTS_START PURE_IMPORTS_END */\n var UnsubscriptionErrorImpl = /*@__PURE__*/ (function () {\n function UnsubscriptionErrorImpl(errors) {\n Error.call(this);\n this.message = errors ?\n errors.length + \" errors occurred during unsubscription:\\n\" + errors.map(function (err, i) { return i + 1 + \") \" + err.toString(); }).join('\\n ') : '';\n this.name = 'UnsubscriptionError';\n this.errors = errors;\n return this;\n }\n UnsubscriptionErrorImpl.prototype = /*@__PURE__*/ Object.create(Error.prototype);\n return UnsubscriptionErrorImpl;\n })();\n var UnsubscriptionError = UnsubscriptionErrorImpl;\n\n /** PURE_IMPORTS_START _util_isArray,_util_isObject,_util_isFunction,_util_UnsubscriptionError PURE_IMPORTS_END */\n var Subscription = /*@__PURE__*/ (function () {\n function Subscription(unsubscribe) {\n this.closed = false;\n this._parentOrParents = null;\n this._subscriptions = null;\n if (unsubscribe) {\n this._ctorUnsubscribe = true;\n this._unsubscribe = unsubscribe;\n }\n }\n Subscription.prototype.unsubscribe = function () {\n var errors;\n if (this.closed) {\n return;\n }\n var _a = this, _parentOrParents = _a._parentOrParents, _ctorUnsubscribe = _a._ctorUnsubscribe, _unsubscribe = _a._unsubscribe, _subscriptions = _a._subscriptions;\n this.closed = true;\n this._parentOrParents = null;\n this._subscriptions = null;\n if (_parentOrParents instanceof Subscription) {\n _parentOrParents.remove(this);\n }\n else if (_parentOrParents !== null) {\n for (var index = 0; index < _parentOrParents.length; ++index) {\n var parent_1 = _parentOrParents[index];\n parent_1.remove(this);\n }\n }\n if (isFunction(_unsubscribe)) {\n if (_ctorUnsubscribe) {\n this._unsubscribe = undefined;\n }\n try {\n _unsubscribe.call(this);\n }\n catch (e) {\n errors = e instanceof UnsubscriptionError ? flattenUnsubscriptionErrors(e.errors) : [e];\n }\n }\n if (isArray(_subscriptions)) {\n var index = -1;\n var len = _subscriptions.length;\n while (++index < len) {\n var sub = _subscriptions[index];\n if (isObject(sub)) {\n try {\n sub.unsubscribe();\n }\n catch (e) {\n errors = errors || [];\n if (e instanceof UnsubscriptionError) {\n errors = errors.concat(flattenUnsubscriptionErrors(e.errors));\n }\n else {\n errors.push(e);\n }\n }\n }\n }\n }\n if (errors) {\n throw new UnsubscriptionError(errors);\n }\n };\n Subscription.prototype.add = function (teardown) {\n var subscription = teardown;\n if (!teardown) {\n return Subscription.EMPTY;\n }\n switch (typeof teardown) {\n case 'function':\n subscription = new Subscription(teardown);\n case 'object':\n if (subscription === this || subscription.closed || typeof subscription.unsubscribe !== 'function') {\n return subscription;\n }\n else if (this.closed) {\n subscription.unsubscribe();\n return subscription;\n }\n else if (!(subscription instanceof Subscription)) {\n var tmp = subscription;\n subscription = new Subscription();\n subscription._subscriptions = [tmp];\n }\n break;\n default: {\n throw new Error('unrecognized teardown ' + teardown + ' added to Subscription.');\n }\n }\n var _parentOrParents = subscription._parentOrParents;\n if (_parentOrParents === null) {\n subscription._parentOrParents = this;\n }\n else if (_parentOrParents instanceof Subscription) {\n if (_parentOrParents === this) {\n return subscription;\n }\n subscription._parentOrParents = [_parentOrParents, this];\n }\n else if (_parentOrParents.indexOf(this) === -1) {\n _parentOrParents.push(this);\n }\n else {\n return subscription;\n }\n var subscriptions = this._subscriptions;\n if (subscriptions === null) {\n this._subscriptions = [subscription];\n }\n else {\n subscriptions.push(subscription);\n }\n return subscription;\n };\n Subscription.prototype.remove = function (subscription) {\n var subscriptions = this._subscriptions;\n if (subscriptions) {\n var subscriptionIndex = subscriptions.indexOf(subscription);\n if (subscriptionIndex !== -1) {\n subscriptions.splice(subscriptionIndex, 1);\n }\n }\n };\n Subscription.EMPTY = (function (empty) {\n empty.closed = true;\n return empty;\n }(new Subscription()));\n return Subscription;\n }());\n function flattenUnsubscriptionErrors(errors) {\n return errors.reduce(function (errs, err) { return errs.concat((err instanceof UnsubscriptionError) ? err.errors : err); }, []);\n }\n\n /** PURE_IMPORTS_START PURE_IMPORTS_END */\n var rxSubscriber = /*@__PURE__*/ (function () {\n return typeof Symbol === 'function'\n ? /*@__PURE__*/ Symbol('rxSubscriber')\n : '@@rxSubscriber_' + /*@__PURE__*/ Math.random();\n })();\n\n /** PURE_IMPORTS_START tslib,_util_isFunction,_Observer,_Subscription,_internal_symbol_rxSubscriber,_config,_util_hostReportError PURE_IMPORTS_END */\n var Subscriber = /*@__PURE__*/ (function (_super) {\n __extends(Subscriber, _super);\n function Subscriber(destinationOrNext, error, complete) {\n var _this = _super.call(this) || this;\n _this.syncErrorValue = null;\n _this.syncErrorThrown = false;\n _this.syncErrorThrowable = false;\n _this.isStopped = false;\n switch (arguments.length) {\n case 0:\n _this.destination = empty;\n break;\n case 1:\n if (!destinationOrNext) {\n _this.destination = empty;\n break;\n }\n if (typeof destinationOrNext === 'object') {\n if (destinationOrNext instanceof Subscriber) {\n _this.syncErrorThrowable = destinationOrNext.syncErrorThrowable;\n _this.destination = destinationOrNext;\n destinationOrNext.add(_this);\n }\n else {\n _this.syncErrorThrowable = true;\n _this.destination = new SafeSubscriber(_this, destinationOrNext);\n }\n break;\n }\n default:\n _this.syncErrorThrowable = true;\n _this.destination = new SafeSubscriber(_this, destinationOrNext, error, complete);\n break;\n }\n return _this;\n }\n Subscriber.prototype[rxSubscriber] = function () { return this; };\n Subscriber.create = function (next, error, complete) {\n var subscriber = new Subscriber(next, error, complete);\n subscriber.syncErrorThrowable = false;\n return subscriber;\n };\n Subscriber.prototype.next = function (value) {\n if (!this.isStopped) {\n this._next(value);\n }\n };\n Subscriber.prototype.error = function (err) {\n if (!this.isStopped) {\n this.isStopped = true;\n this._error(err);\n }\n };\n Subscriber.prototype.complete = function () {\n if (!this.isStopped) {\n this.isStopped = true;\n this._complete();\n }\n };\n Subscriber.prototype.unsubscribe = function () {\n if (this.closed) {\n return;\n }\n this.isStopped = true;\n _super.prototype.unsubscribe.call(this);\n };\n Subscriber.prototype._next = function (value) {\n this.destination.next(value);\n };\n Subscriber.prototype._error = function (err) {\n this.destination.error(err);\n this.unsubscribe();\n };\n Subscriber.prototype._complete = function () {\n this.destination.complete();\n this.unsubscribe();\n };\n Subscriber.prototype._unsubscribeAndRecycle = function () {\n var _parentOrParents = this._parentOrParents;\n this._parentOrParents = null;\n this.unsubscribe();\n this.closed = false;\n this.isStopped = false;\n this._parentOrParents = _parentOrParents;\n return this;\n };\n return Subscriber;\n }(Subscription));\n var SafeSubscriber = /*@__PURE__*/ (function (_super) {\n __extends(SafeSubscriber, _super);\n function SafeSubscriber(_parentSubscriber, observerOrNext, error, complete) {\n var _this = _super.call(this) || this;\n _this._parentSubscriber = _parentSubscriber;\n var next;\n var context = _this;\n if (isFunction(observerOrNext)) {\n next = observerOrNext;\n }\n else if (observerOrNext) {\n next = observerOrNext.next;\n error = observerOrNext.error;\n complete = observerOrNext.complete;\n if (observerOrNext !== empty) {\n context = Object.create(observerOrNext);\n if (isFunction(context.unsubscribe)) {\n _this.add(context.unsubscribe.bind(context));\n }\n context.unsubscribe = _this.unsubscribe.bind(_this);\n }\n }\n _this._context = context;\n _this._next = next;\n _this._error = error;\n _this._complete = complete;\n return _this;\n }\n SafeSubscriber.prototype.next = function (value) {\n if (!this.isStopped && this._next) {\n var _parentSubscriber = this._parentSubscriber;\n if (!config.useDeprecatedSynchronousErrorHandling || !_parentSubscriber.syncErrorThrowable) {\n this.__tryOrUnsub(this._next, value);\n }\n else if (this.__tryOrSetError(_parentSubscriber, this._next, value)) {\n this.unsubscribe();\n }\n }\n };\n SafeSubscriber.prototype.error = function (err) {\n if (!this.isStopped) {\n var _parentSubscriber = this._parentSubscriber;\n var useDeprecatedSynchronousErrorHandling = config.useDeprecatedSynchronousErrorHandling;\n if (this._error) {\n if (!useDeprecatedSynchronousErrorHandling || !_parentSubscriber.syncErrorThrowable) {\n this.__tryOrUnsub(this._error, err);\n this.unsubscribe();\n }\n else {\n this.__tryOrSetError(_parentSubscriber, this._error, err);\n this.unsubscribe();\n }\n }\n else if (!_parentSubscriber.syncErrorThrowable) {\n this.unsubscribe();\n if (useDeprecatedSynchronousErrorHandling) {\n throw err;\n }\n hostReportError(err);\n }\n else {\n if (useDeprecatedSynchronousErrorHandling) {\n _parentSubscriber.syncErrorValue = err;\n _parentSubscriber.syncErrorThrown = true;\n }\n else {\n hostReportError(err);\n }\n this.unsubscribe();\n }\n }\n };\n SafeSubscriber.prototype.complete = function () {\n var _this = this;\n if (!this.isStopped) {\n var _parentSubscriber = this._parentSubscriber;\n if (this._complete) {\n var wrappedComplete = function () { return _this._complete.call(_this._context); };\n if (!config.useDeprecatedSynchronousErrorHandling || !_parentSubscriber.syncErrorThrowable) {\n this.__tryOrUnsub(wrappedComplete);\n this.unsubscribe();\n }\n else {\n this.__tryOrSetError(_parentSubscriber, wrappedComplete);\n this.unsubscribe();\n }\n }\n else {\n this.unsubscribe();\n }\n }\n };\n SafeSubscriber.prototype.__tryOrUnsub = function (fn, value) {\n try {\n fn.call(this._context, value);\n }\n catch (err) {\n this.unsubscribe();\n if (config.useDeprecatedSynchronousErrorHandling) {\n throw err;\n }\n else {\n hostReportError(err);\n }\n }\n };\n SafeSubscriber.prototype.__tryOrSetError = function (parent, fn, value) {\n if (!config.useDeprecatedSynchronousErrorHandling) {\n throw new Error('bad call');\n }\n try {\n fn.call(this._context, value);\n }\n catch (err) {\n if (config.useDeprecatedSynchronousErrorHandling) {\n parent.syncErrorValue = err;\n parent.syncErrorThrown = true;\n return true;\n }\n else {\n hostReportError(err);\n return true;\n }\n }\n return false;\n };\n SafeSubscriber.prototype._unsubscribe = function () {\n var _parentSubscriber = this._parentSubscriber;\n this._context = null;\n this._parentSubscriber = null;\n _parentSubscriber.unsubscribe();\n };\n return SafeSubscriber;\n }(Subscriber));\n\n /** PURE_IMPORTS_START _Subscriber PURE_IMPORTS_END */\n function canReportError(observer) {\n while (observer) {\n var _a = observer, closed_1 = _a.closed, destination = _a.destination, isStopped = _a.isStopped;\n if (closed_1 || isStopped) {\n return false;\n }\n else if (destination && destination instanceof Subscriber) {\n observer = destination;\n }\n else {\n observer = null;\n }\n }\n return true;\n }\n\n /** PURE_IMPORTS_START _Subscriber,_symbol_rxSubscriber,_Observer PURE_IMPORTS_END */\n function toSubscriber(nextOrObserver, error, complete) {\n if (nextOrObserver) {\n if (nextOrObserver instanceof Subscriber) {\n return nextOrObserver;\n }\n if (nextOrObserver[rxSubscriber]) {\n return nextOrObserver[rxSubscriber]();\n }\n }\n if (!nextOrObserver && !error && !complete) {\n return new Subscriber(empty);\n }\n return new Subscriber(nextOrObserver, error, complete);\n }\n\n /** PURE_IMPORTS_START PURE_IMPORTS_END */\n var observable = /*@__PURE__*/ (function () { return typeof Symbol === 'function' && Symbol.observable || '@@observable'; })();\n\n /** PURE_IMPORTS_START PURE_IMPORTS_END */\n function identity(x) {\n return x;\n }\n\n /** PURE_IMPORTS_START _identity PURE_IMPORTS_END */\n function pipeFromArray(fns) {\n if (fns.length === 0) {\n return identity;\n }\n if (fns.length === 1) {\n return fns[0];\n }\n return function piped(input) {\n return fns.reduce(function (prev, fn) { return fn(prev); }, input);\n };\n }\n\n /** PURE_IMPORTS_START _util_canReportError,_util_toSubscriber,_symbol_observable,_util_pipe,_config PURE_IMPORTS_END */\n var Observable = /*@__PURE__*/ (function () {\n function Observable(subscribe) {\n this._isScalar = false;\n if (subscribe) {\n this._subscribe = subscribe;\n }\n }\n Observable.prototype.lift = function (operator) {\n var observable = new Observable();\n observable.source = this;\n observable.operator = operator;\n return observable;\n };\n Observable.prototype.subscribe = function (observerOrNext, error, complete) {\n var operator = this.operator;\n var sink = toSubscriber(observerOrNext, error, complete);\n if (operator) {\n sink.add(operator.call(sink, this.source));\n }\n else {\n sink.add(this.source || (config.useDeprecatedSynchronousErrorHandling && !sink.syncErrorThrowable) ?\n this._subscribe(sink) :\n this._trySubscribe(sink));\n }\n if (config.useDeprecatedSynchronousErrorHandling) {\n if (sink.syncErrorThrowable) {\n sink.syncErrorThrowable = false;\n if (sink.syncErrorThrown) {\n throw sink.syncErrorValue;\n }\n }\n }\n return sink;\n };\n Observable.prototype._trySubscribe = function (sink) {\n try {\n return this._subscribe(sink);\n }\n catch (err) {\n if (config.useDeprecatedSynchronousErrorHandling) {\n sink.syncErrorThrown = true;\n sink.syncErrorValue = err;\n }\n if (canReportError(sink)) {\n sink.error(err);\n }\n else {\n console.warn(err);\n }\n }\n };\n Observable.prototype.forEach = function (next, promiseCtor) {\n var _this = this;\n promiseCtor = getPromiseCtor(promiseCtor);\n return new promiseCtor(function (resolve, reject) {\n var subscription;\n subscription = _this.subscribe(function (value) {\n try {\n next(value);\n }\n catch (err) {\n reject(err);\n if (subscription) {\n subscription.unsubscribe();\n }\n }\n }, reject, resolve);\n });\n };\n Observable.prototype._subscribe = function (subscriber) {\n var source = this.source;\n return source && source.subscribe(subscriber);\n };\n Observable.prototype[observable] = function () {\n return this;\n };\n Observable.prototype.pipe = function () {\n var operations = [];\n for (var _i = 0; _i < arguments.length; _i++) {\n operations[_i] = arguments[_i];\n }\n if (operations.length === 0) {\n return this;\n }\n return pipeFromArray(operations)(this);\n };\n Observable.prototype.toPromise = function (promiseCtor) {\n var _this = this;\n promiseCtor = getPromiseCtor(promiseCtor);\n return new promiseCtor(function (resolve, reject) {\n var value;\n _this.subscribe(function (x) { return value = x; }, function (err) { return reject(err); }, function () { return resolve(value); });\n });\n };\n Observable.create = function (subscribe) {\n return new Observable(subscribe);\n };\n return Observable;\n }());\n function getPromiseCtor(promiseCtor) {\n if (!promiseCtor) {\n promiseCtor = Promise;\n }\n if (!promiseCtor) {\n throw new Error('no Promise impl found');\n }\n return promiseCtor;\n }\n\n /** PURE_IMPORTS_START PURE_IMPORTS_END */\n var ObjectUnsubscribedErrorImpl = /*@__PURE__*/ (function () {\n function ObjectUnsubscribedErrorImpl() {\n Error.call(this);\n this.message = 'object unsubscribed';\n this.name = 'ObjectUnsubscribedError';\n return this;\n }\n ObjectUnsubscribedErrorImpl.prototype = /*@__PURE__*/ Object.create(Error.prototype);\n return ObjectUnsubscribedErrorImpl;\n })();\n var ObjectUnsubscribedError = ObjectUnsubscribedErrorImpl;\n\n /** PURE_IMPORTS_START tslib,_Subscription PURE_IMPORTS_END */\n var SubjectSubscription = /*@__PURE__*/ (function (_super) {\n __extends(SubjectSubscription, _super);\n function SubjectSubscription(subject, subscriber) {\n var _this = _super.call(this) || this;\n _this.subject = subject;\n _this.subscriber = subscriber;\n _this.closed = false;\n return _this;\n }\n SubjectSubscription.prototype.unsubscribe = function () {\n if (this.closed) {\n return;\n }\n this.closed = true;\n var subject = this.subject;\n var observers = subject.observers;\n this.subject = null;\n if (!observers || observers.length === 0 || subject.isStopped || subject.closed) {\n return;\n }\n var subscriberIndex = observers.indexOf(this.subscriber);\n if (subscriberIndex !== -1) {\n observers.splice(subscriberIndex, 1);\n }\n };\n return SubjectSubscription;\n }(Subscription));\n\n /** PURE_IMPORTS_START tslib,_Observable,_Subscriber,_Subscription,_util_ObjectUnsubscribedError,_SubjectSubscription,_internal_symbol_rxSubscriber PURE_IMPORTS_END */\n var SubjectSubscriber = /*@__PURE__*/ (function (_super) {\n __extends(SubjectSubscriber, _super);\n function SubjectSubscriber(destination) {\n var _this = _super.call(this, destination) || this;\n _this.destination = destination;\n return _this;\n }\n return SubjectSubscriber;\n }(Subscriber));\n var Subject = /*@__PURE__*/ (function (_super) {\n __extends(Subject, _super);\n function Subject() {\n var _this = _super.call(this) || this;\n _this.observers = [];\n _this.closed = false;\n _this.isStopped = false;\n _this.hasError = false;\n _this.thrownError = null;\n return _this;\n }\n Subject.prototype[rxSubscriber] = function () {\n return new SubjectSubscriber(this);\n };\n Subject.prototype.lift = function (operator) {\n var subject = new AnonymousSubject(this, this);\n subject.operator = operator;\n return subject;\n };\n Subject.prototype.next = function (value) {\n if (this.closed) {\n throw new ObjectUnsubscribedError();\n }\n if (!this.isStopped) {\n var observers = this.observers;\n var len = observers.length;\n var copy = observers.slice();\n for (var i = 0; i < len; i++) {\n copy[i].next(value);\n }\n }\n };\n Subject.prototype.error = function (err) {\n if (this.closed) {\n throw new ObjectUnsubscribedError();\n }\n this.hasError = true;\n this.thrownError = err;\n this.isStopped = true;\n var observers = this.observers;\n var len = observers.length;\n var copy = observers.slice();\n for (var i = 0; i < len; i++) {\n copy[i].error(err);\n }\n this.observers.length = 0;\n };\n Subject.prototype.complete = function () {\n if (this.closed) {\n throw new ObjectUnsubscribedError();\n }\n this.isStopped = true;\n var observers = this.observers;\n var len = observers.length;\n var copy = observers.slice();\n for (var i = 0; i < len; i++) {\n copy[i].complete();\n }\n this.observers.length = 0;\n };\n Subject.prototype.unsubscribe = function () {\n this.isStopped = true;\n this.closed = true;\n this.observers = null;\n };\n Subject.prototype._trySubscribe = function (subscriber) {\n if (this.closed) {\n throw new ObjectUnsubscribedError();\n }\n else {\n return _super.prototype._trySubscribe.call(this, subscriber);\n }\n };\n Subject.prototype._subscribe = function (subscriber) {\n if (this.closed) {\n throw new ObjectUnsubscribedError();\n }\n else if (this.hasError) {\n subscriber.error(this.thrownError);\n return Subscription.EMPTY;\n }\n else if (this.isStopped) {\n subscriber.complete();\n return Subscription.EMPTY;\n }\n else {\n this.observers.push(subscriber);\n return new SubjectSubscription(this, subscriber);\n }\n };\n Subject.prototype.asObservable = function () {\n var observable = new Observable();\n observable.source = this;\n return observable;\n };\n Subject.create = function (destination, source) {\n return new AnonymousSubject(destination, source);\n };\n return Subject;\n }(Observable));\n var AnonymousSubject = /*@__PURE__*/ (function (_super) {\n __extends(AnonymousSubject, _super);\n function AnonymousSubject(destination, source) {\n var _this = _super.call(this) || this;\n _this.destination = destination;\n _this.source = source;\n return _this;\n }\n AnonymousSubject.prototype.next = function (value) {\n var destination = this.destination;\n if (destination && destination.next) {\n destination.next(value);\n }\n };\n AnonymousSubject.prototype.error = function (err) {\n var destination = this.destination;\n if (destination && destination.error) {\n this.destination.error(err);\n }\n };\n AnonymousSubject.prototype.complete = function () {\n var destination = this.destination;\n if (destination && destination.complete) {\n this.destination.complete();\n }\n };\n AnonymousSubject.prototype._subscribe = function (subscriber) {\n var source = this.source;\n if (source) {\n return this.source.subscribe(subscriber);\n }\n else {\n return Subscription.EMPTY;\n }\n };\n return AnonymousSubject;\n }(Subject));\n\n /** PURE_IMPORTS_START tslib,_Subscriber PURE_IMPORTS_END */\n function refCount() {\n return function refCountOperatorFunction(source) {\n return source.lift(new RefCountOperator(source));\n };\n }\n var RefCountOperator = /*@__PURE__*/ (function () {\n function RefCountOperator(connectable) {\n this.connectable = connectable;\n }\n RefCountOperator.prototype.call = function (subscriber, source) {\n var connectable = this.connectable;\n connectable._refCount++;\n var refCounter = new RefCountSubscriber(subscriber, connectable);\n var subscription = source.subscribe(refCounter);\n if (!refCounter.closed) {\n refCounter.connection = connectable.connect();\n }\n return subscription;\n };\n return RefCountOperator;\n }());\n var RefCountSubscriber = /*@__PURE__*/ (function (_super) {\n __extends(RefCountSubscriber, _super);\n function RefCountSubscriber(destination, connectable) {\n var _this = _super.call(this, destination) || this;\n _this.connectable = connectable;\n return _this;\n }\n RefCountSubscriber.prototype._unsubscribe = function () {\n var connectable = this.connectable;\n if (!connectable) {\n this.connection = null;\n return;\n }\n this.connectable = null;\n var refCount = connectable._refCount;\n if (refCount <= 0) {\n this.connection = null;\n return;\n }\n connectable._refCount = refCount - 1;\n if (refCount > 1) {\n this.connection = null;\n return;\n }\n var connection = this.connection;\n var sharedConnection = connectable._connection;\n this.connection = null;\n if (sharedConnection && (!connection || sharedConnection === connection)) {\n sharedConnection.unsubscribe();\n }\n };\n return RefCountSubscriber;\n }(Subscriber));\n\n /** PURE_IMPORTS_START tslib,_Subject,_Observable,_Subscriber,_Subscription,_operators_refCount PURE_IMPORTS_END */\n var ConnectableObservable = /*@__PURE__*/ (function (_super) {\n __extends(ConnectableObservable, _super);\n function ConnectableObservable(source, subjectFactory) {\n var _this = _super.call(this) || this;\n _this.source = source;\n _this.subjectFactory = subjectFactory;\n _this._refCount = 0;\n _this._isComplete = false;\n return _this;\n }\n ConnectableObservable.prototype._subscribe = function (subscriber) {\n return this.getSubject().subscribe(subscriber);\n };\n ConnectableObservable.prototype.getSubject = function () {\n var subject = this._subject;\n if (!subject || subject.isStopped) {\n this._subject = this.subjectFactory();\n }\n return this._subject;\n };\n ConnectableObservable.prototype.connect = function () {\n var connection = this._connection;\n if (!connection) {\n this._isComplete = false;\n connection = this._connection = new Subscription();\n connection.add(this.source\n .subscribe(new ConnectableSubscriber(this.getSubject(), this)));\n if (connection.closed) {\n this._connection = null;\n connection = Subscription.EMPTY;\n }\n }\n return connection;\n };\n ConnectableObservable.prototype.refCount = function () {\n return refCount()(this);\n };\n return ConnectableObservable;\n }(Observable));\n var connectableObservableDescriptor = /*@__PURE__*/ (function () {\n var connectableProto = ConnectableObservable.prototype;\n return {\n operator: { value: null },\n _refCount: { value: 0, writable: true },\n _subject: { value: null, writable: true },\n _connection: { value: null, writable: true },\n _subscribe: { value: connectableProto._subscribe },\n _isComplete: { value: connectableProto._isComplete, writable: true },\n getSubject: { value: connectableProto.getSubject },\n connect: { value: connectableProto.connect },\n refCount: { value: connectableProto.refCount }\n };\n })();\n var ConnectableSubscriber = /*@__PURE__*/ (function (_super) {\n __extends(ConnectableSubscriber, _super);\n function ConnectableSubscriber(destination, connectable) {\n var _this = _super.call(this, destination) || this;\n _this.connectable = connectable;\n return _this;\n }\n ConnectableSubscriber.prototype._error = function (err) {\n this._unsubscribe();\n _super.prototype._error.call(this, err);\n };\n ConnectableSubscriber.prototype._complete = function () {\n this.connectable._isComplete = true;\n this._unsubscribe();\n _super.prototype._complete.call(this);\n };\n ConnectableSubscriber.prototype._unsubscribe = function () {\n var connectable = this.connectable;\n if (connectable) {\n this.connectable = null;\n var connection = connectable._connection;\n connectable._refCount = 0;\n connectable._subject = null;\n connectable._connection = null;\n if (connection) {\n connection.unsubscribe();\n }\n }\n };\n return ConnectableSubscriber;\n }(SubjectSubscriber));\n\n /** PURE_IMPORTS_START tslib,_Subject,_util_ObjectUnsubscribedError PURE_IMPORTS_END */\n var BehaviorSubject = /*@__PURE__*/ (function (_super) {\n __extends(BehaviorSubject, _super);\n function BehaviorSubject(_value) {\n var _this = _super.call(this) || this;\n _this._value = _value;\n return _this;\n }\n Object.defineProperty(BehaviorSubject.prototype, \"value\", {\n get: function () {\n return this.getValue();\n },\n enumerable: true,\n configurable: true\n });\n BehaviorSubject.prototype._subscribe = function (subscriber) {\n var subscription = _super.prototype._subscribe.call(this, subscriber);\n if (subscription && !subscription.closed) {\n subscriber.next(this._value);\n }\n return subscription;\n };\n BehaviorSubject.prototype.getValue = function () {\n if (this.hasError) {\n throw this.thrownError;\n }\n else if (this.closed) {\n throw new ObjectUnsubscribedError();\n }\n else {\n return this._value;\n }\n };\n BehaviorSubject.prototype.next = function (value) {\n _super.prototype.next.call(this, this._value = value);\n };\n return BehaviorSubject;\n }(Subject));\n\n /** PURE_IMPORTS_START tslib,_Subscription PURE_IMPORTS_END */\n var Action = /*@__PURE__*/ (function (_super) {\n __extends(Action, _super);\n function Action(scheduler, work) {\n return _super.call(this) || this;\n }\n Action.prototype.schedule = function (state, delay) {\n return this;\n };\n return Action;\n }(Subscription));\n\n /** PURE_IMPORTS_START tslib,_Action PURE_IMPORTS_END */\n var AsyncAction = /*@__PURE__*/ (function (_super) {\n __extends(AsyncAction, _super);\n function AsyncAction(scheduler, work) {\n var _this = _super.call(this, scheduler, work) || this;\n _this.scheduler = scheduler;\n _this.work = work;\n _this.pending = false;\n return _this;\n }\n AsyncAction.prototype.schedule = function (state, delay) {\n if (delay === void 0) {\n delay = 0;\n }\n if (this.closed) {\n return this;\n }\n this.state = state;\n var id = this.id;\n var scheduler = this.scheduler;\n if (id != null) {\n this.id = this.recycleAsyncId(scheduler, id, delay);\n }\n this.pending = true;\n this.delay = delay;\n this.id = this.id || this.requestAsyncId(scheduler, this.id, delay);\n return this;\n };\n AsyncAction.prototype.requestAsyncId = function (scheduler, id, delay) {\n if (delay === void 0) {\n delay = 0;\n }\n return setInterval(scheduler.flush.bind(scheduler, this), delay);\n };\n AsyncAction.prototype.recycleAsyncId = function (scheduler, id, delay) {\n if (delay === void 0) {\n delay = 0;\n }\n if (delay !== null && this.delay === delay && this.pending === false) {\n return id;\n }\n clearInterval(id);\n return undefined;\n };\n AsyncAction.prototype.execute = function (state, delay) {\n if (this.closed) {\n return new Error('executing a cancelled action');\n }\n this.pending = false;\n var error = this._execute(state, delay);\n if (error) {\n return error;\n }\n else if (this.pending === false && this.id != null) {\n this.id = this.recycleAsyncId(this.scheduler, this.id, null);\n }\n };\n AsyncAction.prototype._execute = function (state, delay) {\n var errored = false;\n var errorValue = undefined;\n try {\n this.work(state);\n }\n catch (e) {\n errored = true;\n errorValue = !!e && e || new Error(e);\n }\n if (errored) {\n this.unsubscribe();\n return errorValue;\n }\n };\n AsyncAction.prototype._unsubscribe = function () {\n var id = this.id;\n var scheduler = this.scheduler;\n var actions = scheduler.actions;\n var index = actions.indexOf(this);\n this.work = null;\n this.state = null;\n this.pending = false;\n this.scheduler = null;\n if (index !== -1) {\n actions.splice(index, 1);\n }\n if (id != null) {\n this.id = this.recycleAsyncId(scheduler, id, null);\n }\n this.delay = null;\n };\n return AsyncAction;\n }(Action));\n\n /** PURE_IMPORTS_START tslib,_AsyncAction PURE_IMPORTS_END */\n var QueueAction = /*@__PURE__*/ (function (_super) {\n __extends(QueueAction, _super);\n function QueueAction(scheduler, work) {\n var _this = _super.call(this, scheduler, work) || this;\n _this.scheduler = scheduler;\n _this.work = work;\n return _this;\n }\n QueueAction.prototype.schedule = function (state, delay) {\n if (delay === void 0) {\n delay = 0;\n }\n if (delay > 0) {\n return _super.prototype.schedule.call(this, state, delay);\n }\n this.delay = delay;\n this.state = state;\n this.scheduler.flush(this);\n return this;\n };\n QueueAction.prototype.execute = function (state, delay) {\n return (delay > 0 || this.closed) ?\n _super.prototype.execute.call(this, state, delay) :\n this._execute(state, delay);\n };\n QueueAction.prototype.requestAsyncId = function (scheduler, id, delay) {\n if (delay === void 0) {\n delay = 0;\n }\n if ((delay !== null && delay > 0) || (delay === null && this.delay > 0)) {\n return _super.prototype.requestAsyncId.call(this, scheduler, id, delay);\n }\n return scheduler.flush(this);\n };\n return QueueAction;\n }(AsyncAction));\n\n var Scheduler = /*@__PURE__*/ (function () {\n function Scheduler(SchedulerAction, now) {\n if (now === void 0) {\n now = Scheduler.now;\n }\n this.SchedulerAction = SchedulerAction;\n this.now = now;\n }\n Scheduler.prototype.schedule = function (work, delay, state) {\n if (delay === void 0) {\n delay = 0;\n }\n return new this.SchedulerAction(this, work).schedule(state, delay);\n };\n Scheduler.now = function () { return Date.now(); };\n return Scheduler;\n }());\n\n /** PURE_IMPORTS_START tslib,_Scheduler PURE_IMPORTS_END */\n var AsyncScheduler = /*@__PURE__*/ (function (_super) {\n __extends(AsyncScheduler, _super);\n function AsyncScheduler(SchedulerAction, now) {\n if (now === void 0) {\n now = Scheduler.now;\n }\n var _this = _super.call(this, SchedulerAction, function () {\n if (AsyncScheduler.delegate && AsyncScheduler.delegate !== _this) {\n return AsyncScheduler.delegate.now();\n }\n else {\n return now();\n }\n }) || this;\n _this.actions = [];\n _this.active = false;\n _this.scheduled = undefined;\n return _this;\n }\n AsyncScheduler.prototype.schedule = function (work, delay, state) {\n if (delay === void 0) {\n delay = 0;\n }\n if (AsyncScheduler.delegate && AsyncScheduler.delegate !== this) {\n return AsyncScheduler.delegate.schedule(work, delay, state);\n }\n else {\n return _super.prototype.schedule.call(this, work, delay, state);\n }\n };\n AsyncScheduler.prototype.flush = function (action) {\n var actions = this.actions;\n if (this.active) {\n actions.push(action);\n return;\n }\n var error;\n this.active = true;\n do {\n if (error = action.execute(action.state, action.delay)) {\n break;\n }\n } while (action = actions.shift());\n this.active = false;\n if (error) {\n while (action = actions.shift()) {\n action.unsubscribe();\n }\n throw error;\n }\n };\n return AsyncScheduler;\n }(Scheduler));\n\n /** PURE_IMPORTS_START tslib,_AsyncScheduler PURE_IMPORTS_END */\n var QueueScheduler = /*@__PURE__*/ (function (_super) {\n __extends(QueueScheduler, _super);\n function QueueScheduler() {\n return _super !== null && _super.apply(this, arguments) || this;\n }\n return QueueScheduler;\n }(AsyncScheduler));\n\n /** PURE_IMPORTS_START _QueueAction,_QueueScheduler PURE_IMPORTS_END */\n var queueScheduler = /*@__PURE__*/ new QueueScheduler(QueueAction);\n var queue = queueScheduler;\n\n /** PURE_IMPORTS_START _Observable PURE_IMPORTS_END */\n var EMPTY = /*@__PURE__*/ new Observable(function (subscriber) { return subscriber.complete(); });\n function empty$1(scheduler) {\n return scheduler ? emptyScheduled(scheduler) : EMPTY;\n }\n function emptyScheduled(scheduler) {\n return new Observable(function (subscriber) { return scheduler.schedule(function () { return subscriber.complete(); }); });\n }\n\n /** PURE_IMPORTS_START PURE_IMPORTS_END */\n function isScheduler(value) {\n return value && typeof value.schedule === 'function';\n }\n\n /** PURE_IMPORTS_START PURE_IMPORTS_END */\n var subscribeToArray = function (array) {\n return function (subscriber) {\n for (var i = 0, len = array.length; i < len && !subscriber.closed; i++) {\n subscriber.next(array[i]);\n }\n subscriber.complete();\n };\n };\n\n /** PURE_IMPORTS_START _Observable,_Subscription PURE_IMPORTS_END */\n function scheduleArray(input, scheduler) {\n return new Observable(function (subscriber) {\n var sub = new Subscription();\n var i = 0;\n sub.add(scheduler.schedule(function () {\n if (i === input.length) {\n subscriber.complete();\n return;\n }\n subscriber.next(input[i++]);\n if (!subscriber.closed) {\n sub.add(this.schedule());\n }\n }));\n return sub;\n });\n }\n\n /** PURE_IMPORTS_START _Observable,_util_subscribeToArray,_scheduled_scheduleArray PURE_IMPORTS_END */\n function fromArray(input, scheduler) {\n if (!scheduler) {\n return new Observable(subscribeToArray(input));\n }\n else {\n return scheduleArray(input, scheduler);\n }\n }\n\n /** PURE_IMPORTS_START _util_isScheduler,_fromArray,_scheduled_scheduleArray PURE_IMPORTS_END */\n function of() {\n var args = [];\n for (var _i = 0; _i < arguments.length; _i++) {\n args[_i] = arguments[_i];\n }\n var scheduler = args[args.length - 1];\n if (isScheduler(scheduler)) {\n args.pop();\n return scheduleArray(args, scheduler);\n }\n else {\n return fromArray(args);\n }\n }\n\n /** PURE_IMPORTS_START _Observable PURE_IMPORTS_END */\n function throwError(error, scheduler) {\n if (!scheduler) {\n return new Observable(function (subscriber) { return subscriber.error(error); });\n }\n else {\n return new Observable(function (subscriber) { return scheduler.schedule(dispatch, 0, { error: error, subscriber: subscriber }); });\n }\n }\n function dispatch(_a) {\n var error = _a.error, subscriber = _a.subscriber;\n subscriber.error(error);\n }\n\n /** PURE_IMPORTS_START _observable_empty,_observable_of,_observable_throwError PURE_IMPORTS_END */\n var Notification = /*@__PURE__*/ (function () {\n function Notification(kind, value, error) {\n this.kind = kind;\n this.value = value;\n this.error = error;\n this.hasValue = kind === 'N';\n }\n Notification.prototype.observe = function (observer) {\n switch (this.kind) {\n case 'N':\n return observer.next && observer.next(this.value);\n case 'E':\n return observer.error && observer.error(this.error);\n case 'C':\n return observer.complete && observer.complete();\n }\n };\n Notification.prototype.do = function (next, error, complete) {\n var kind = this.kind;\n switch (kind) {\n case 'N':\n return next && next(this.value);\n case 'E':\n return error && error(this.error);\n case 'C':\n return complete && complete();\n }\n };\n Notification.prototype.accept = function (nextOrObserver, error, complete) {\n if (nextOrObserver && typeof nextOrObserver.next === 'function') {\n return this.observe(nextOrObserver);\n }\n else {\n return this.do(nextOrObserver, error, complete);\n }\n };\n Notification.prototype.toObservable = function () {\n var kind = this.kind;\n switch (kind) {\n case 'N':\n return of(this.value);\n case 'E':\n return throwError(this.error);\n case 'C':\n return empty$1();\n }\n throw new Error('unexpected notification kind value');\n };\n Notification.createNext = function (value) {\n if (typeof value !== 'undefined') {\n return new Notification('N', value);\n }\n return Notification.undefinedValueNotification;\n };\n Notification.createError = function (err) {\n return new Notification('E', undefined, err);\n };\n Notification.createComplete = function () {\n return Notification.completeNotification;\n };\n Notification.completeNotification = new Notification('C');\n Notification.undefinedValueNotification = new Notification('N', undefined);\n return Notification;\n }());\n\n /** PURE_IMPORTS_START tslib,_Subscriber,_Notification PURE_IMPORTS_END */\n var ObserveOnSubscriber = /*@__PURE__*/ (function (_super) {\n __extends(ObserveOnSubscriber, _super);\n function ObserveOnSubscriber(destination, scheduler, delay) {\n if (delay === void 0) {\n delay = 0;\n }\n var _this = _super.call(this, destination) || this;\n _this.scheduler = scheduler;\n _this.delay = delay;\n return _this;\n }\n ObserveOnSubscriber.dispatch = function (arg) {\n var notification = arg.notification, destination = arg.destination;\n notification.observe(destination);\n this.unsubscribe();\n };\n ObserveOnSubscriber.prototype.scheduleMessage = function (notification) {\n var destination = this.destination;\n destination.add(this.scheduler.schedule(ObserveOnSubscriber.dispatch, this.delay, new ObserveOnMessage(notification, this.destination)));\n };\n ObserveOnSubscriber.prototype._next = function (value) {\n this.scheduleMessage(Notification.createNext(value));\n };\n ObserveOnSubscriber.prototype._error = function (err) {\n this.scheduleMessage(Notification.createError(err));\n this.unsubscribe();\n };\n ObserveOnSubscriber.prototype._complete = function () {\n this.scheduleMessage(Notification.createComplete());\n this.unsubscribe();\n };\n return ObserveOnSubscriber;\n }(Subscriber));\n var ObserveOnMessage = /*@__PURE__*/ (function () {\n function ObserveOnMessage(notification, destination) {\n this.notification = notification;\n this.destination = destination;\n }\n return ObserveOnMessage;\n }());\n\n /** PURE_IMPORTS_START tslib,_Subject,_scheduler_queue,_Subscription,_operators_observeOn,_util_ObjectUnsubscribedError,_SubjectSubscription PURE_IMPORTS_END */\n var ReplaySubject = /*@__PURE__*/ (function (_super) {\n __extends(ReplaySubject, _super);\n function ReplaySubject(bufferSize, windowTime, scheduler) {\n if (bufferSize === void 0) {\n bufferSize = Number.POSITIVE_INFINITY;\n }\n if (windowTime === void 0) {\n windowTime = Number.POSITIVE_INFINITY;\n }\n var _this = _super.call(this) || this;\n _this.scheduler = scheduler;\n _this._events = [];\n _this._infiniteTimeWindow = false;\n _this._bufferSize = bufferSize < 1 ? 1 : bufferSize;\n _this._windowTime = windowTime < 1 ? 1 : windowTime;\n if (windowTime === Number.POSITIVE_INFINITY) {\n _this._infiniteTimeWindow = true;\n _this.next = _this.nextInfiniteTimeWindow;\n }\n else {\n _this.next = _this.nextTimeWindow;\n }\n return _this;\n }\n ReplaySubject.prototype.nextInfiniteTimeWindow = function (value) {\n if (!this.isStopped) {\n var _events = this._events;\n _events.push(value);\n if (_events.length > this._bufferSize) {\n _events.shift();\n }\n }\n _super.prototype.next.call(this, value);\n };\n ReplaySubject.prototype.nextTimeWindow = function (value) {\n if (!this.isStopped) {\n this._events.push(new ReplayEvent(this._getNow(), value));\n this._trimBufferThenGetEvents();\n }\n _super.prototype.next.call(this, value);\n };\n ReplaySubject.prototype._subscribe = function (subscriber) {\n var _infiniteTimeWindow = this._infiniteTimeWindow;\n var _events = _infiniteTimeWindow ? this._events : this._trimBufferThenGetEvents();\n var scheduler = this.scheduler;\n var len = _events.length;\n var subscription;\n if (this.closed) {\n throw new ObjectUnsubscribedError();\n }\n else if (this.isStopped || this.hasError) {\n subscription = Subscription.EMPTY;\n }\n else {\n this.observers.push(subscriber);\n subscription = new SubjectSubscription(this, subscriber);\n }\n if (scheduler) {\n subscriber.add(subscriber = new ObserveOnSubscriber(subscriber, scheduler));\n }\n if (_infiniteTimeWindow) {\n for (var i = 0; i < len && !subscriber.closed; i++) {\n subscriber.next(_events[i]);\n }\n }\n else {\n for (var i = 0; i < len && !subscriber.closed; i++) {\n subscriber.next(_events[i].value);\n }\n }\n if (this.hasError) {\n subscriber.error(this.thrownError);\n }\n else if (this.isStopped) {\n subscriber.complete();\n }\n return subscription;\n };\n ReplaySubject.prototype._getNow = function () {\n return (this.scheduler || queue).now();\n };\n ReplaySubject.prototype._trimBufferThenGetEvents = function () {\n var now = this._getNow();\n var _bufferSize = this._bufferSize;\n var _windowTime = this._windowTime;\n var _events = this._events;\n var eventsCount = _events.length;\n var spliceCount = 0;\n while (spliceCount < eventsCount) {\n if ((now - _events[spliceCount].time) < _windowTime) {\n break;\n }\n spliceCount++;\n }\n if (eventsCount > _bufferSize) {\n spliceCount = Math.max(spliceCount, eventsCount - _bufferSize);\n }\n if (spliceCount > 0) {\n _events.splice(0, spliceCount);\n }\n return _events;\n };\n return ReplaySubject;\n }(Subject));\n var ReplayEvent = /*@__PURE__*/ (function () {\n function ReplayEvent(time, value) {\n this.time = time;\n this.value = value;\n }\n return ReplayEvent;\n }());\n\n /** PURE_IMPORTS_START _AsyncAction,_AsyncScheduler PURE_IMPORTS_END */\n var asyncScheduler = /*@__PURE__*/ new AsyncScheduler(AsyncAction);\n var async = asyncScheduler;\n\n /** PURE_IMPORTS_START PURE_IMPORTS_END */\n var ArgumentOutOfRangeErrorImpl = /*@__PURE__*/ (function () {\n function ArgumentOutOfRangeErrorImpl() {\n Error.call(this);\n this.message = 'argument out of range';\n this.name = 'ArgumentOutOfRangeError';\n return this;\n }\n ArgumentOutOfRangeErrorImpl.prototype = /*@__PURE__*/ Object.create(Error.prototype);\n return ArgumentOutOfRangeErrorImpl;\n })();\n var ArgumentOutOfRangeError = ArgumentOutOfRangeErrorImpl;\n\n /** PURE_IMPORTS_START PURE_IMPORTS_END */\n var EmptyErrorImpl = /*@__PURE__*/ (function () {\n function EmptyErrorImpl() {\n Error.call(this);\n this.message = 'no elements in sequence';\n this.name = 'EmptyError';\n return this;\n }\n EmptyErrorImpl.prototype = /*@__PURE__*/ Object.create(Error.prototype);\n return EmptyErrorImpl;\n })();\n var EmptyError = EmptyErrorImpl;\n\n /** PURE_IMPORTS_START tslib,_Subscriber PURE_IMPORTS_END */\n function map(project, thisArg) {\n return function mapOperation(source) {\n if (typeof project !== 'function') {\n throw new TypeError('argument is not a function. Are you looking for `mapTo()`?');\n }\n return source.lift(new MapOperator(project, thisArg));\n };\n }\n var MapOperator = /*@__PURE__*/ (function () {\n function MapOperator(project, thisArg) {\n this.project = project;\n this.thisArg = thisArg;\n }\n MapOperator.prototype.call = function (subscriber, source) {\n return source.subscribe(new MapSubscriber(subscriber, this.project, this.thisArg));\n };\n return MapOperator;\n }());\n var MapSubscriber = /*@__PURE__*/ (function (_super) {\n __extends(MapSubscriber, _super);\n function MapSubscriber(destination, project, thisArg) {\n var _this = _super.call(this, destination) || this;\n _this.project = project;\n _this.count = 0;\n _this.thisArg = thisArg || _this;\n return _this;\n }\n MapSubscriber.prototype._next = function (value) {\n var result;\n try {\n result = this.project.call(this.thisArg, value, this.count++);\n }\n catch (err) {\n this.destination.error(err);\n return;\n }\n this.destination.next(result);\n };\n return MapSubscriber;\n }(Subscriber));\n\n /** PURE_IMPORTS_START tslib,_Subscriber PURE_IMPORTS_END */\n var OuterSubscriber = /*@__PURE__*/ (function (_super) {\n __extends(OuterSubscriber, _super);\n function OuterSubscriber() {\n return _super !== null && _super.apply(this, arguments) || this;\n }\n OuterSubscriber.prototype.notifyNext = function (outerValue, innerValue, outerIndex, innerIndex, innerSub) {\n this.destination.next(innerValue);\n };\n OuterSubscriber.prototype.notifyError = function (error, innerSub) {\n this.destination.error(error);\n };\n OuterSubscriber.prototype.notifyComplete = function (innerSub) {\n this.destination.complete();\n };\n return OuterSubscriber;\n }(Subscriber));\n\n /** PURE_IMPORTS_START tslib,_Subscriber PURE_IMPORTS_END */\n var InnerSubscriber = /*@__PURE__*/ (function (_super) {\n __extends(InnerSubscriber, _super);\n function InnerSubscriber(parent, outerValue, outerIndex) {\n var _this = _super.call(this) || this;\n _this.parent = parent;\n _this.outerValue = outerValue;\n _this.outerIndex = outerIndex;\n _this.index = 0;\n return _this;\n }\n InnerSubscriber.prototype._next = function (value) {\n this.parent.notifyNext(this.outerValue, value, this.outerIndex, this.index++, this);\n };\n InnerSubscriber.prototype._error = function (error) {\n this.parent.notifyError(error, this);\n this.unsubscribe();\n };\n InnerSubscriber.prototype._complete = function () {\n this.parent.notifyComplete(this);\n this.unsubscribe();\n };\n return InnerSubscriber;\n }(Subscriber));\n\n /** PURE_IMPORTS_START _hostReportError PURE_IMPORTS_END */\n var subscribeToPromise = function (promise) {\n return function (subscriber) {\n promise.then(function (value) {\n if (!subscriber.closed) {\n subscriber.next(value);\n subscriber.complete();\n }\n }, function (err) { return subscriber.error(err); })\n .then(null, hostReportError);\n return subscriber;\n };\n };\n\n /** PURE_IMPORTS_START PURE_IMPORTS_END */\n function getSymbolIterator() {\n if (typeof Symbol !== 'function' || !Symbol.iterator) {\n return '@@iterator';\n }\n return Symbol.iterator;\n }\n var iterator = /*@__PURE__*/ getSymbolIterator();\n\n /** PURE_IMPORTS_START _symbol_iterator PURE_IMPORTS_END */\n var subscribeToIterable = function (iterable) {\n return function (subscriber) {\n var iterator$1 = iterable[iterator]();\n do {\n var item = void 0;\n try {\n item = iterator$1.next();\n }\n catch (err) {\n subscriber.error(err);\n return subscriber;\n }\n if (item.done) {\n subscriber.complete();\n break;\n }\n subscriber.next(item.value);\n if (subscriber.closed) {\n break;\n }\n } while (true);\n if (typeof iterator$1.return === 'function') {\n subscriber.add(function () {\n if (iterator$1.return) {\n iterator$1.return();\n }\n });\n }\n return subscriber;\n };\n };\n\n /** PURE_IMPORTS_START _symbol_observable PURE_IMPORTS_END */\n var subscribeToObservable = function (obj) {\n return function (subscriber) {\n var obs = obj[observable]();\n if (typeof obs.subscribe !== 'function') {\n throw new TypeError('Provided object does not correctly implement Symbol.observable');\n }\n else {\n return obs.subscribe(subscriber);\n }\n };\n };\n\n /** PURE_IMPORTS_START PURE_IMPORTS_END */\n var isArrayLike = (function (x) { return x && typeof x.length === 'number' && typeof x !== 'function'; });\n\n /** PURE_IMPORTS_START PURE_IMPORTS_END */\n function isPromise(value) {\n return !!value && typeof value.subscribe !== 'function' && typeof value.then === 'function';\n }\n\n /** PURE_IMPORTS_START _subscribeToArray,_subscribeToPromise,_subscribeToIterable,_subscribeToObservable,_isArrayLike,_isPromise,_isObject,_symbol_iterator,_symbol_observable PURE_IMPORTS_END */\n var subscribeTo = function (result) {\n if (!!result && typeof result[observable] === 'function') {\n return subscribeToObservable(result);\n }\n else if (isArrayLike(result)) {\n return subscribeToArray(result);\n }\n else if (isPromise(result)) {\n return subscribeToPromise(result);\n }\n else if (!!result && typeof result[iterator] === 'function') {\n return subscribeToIterable(result);\n }\n else {\n var value = isObject(result) ? 'an invalid object' : \"'\" + result + \"'\";\n var msg = \"You provided \" + value + \" where a stream was expected.\"\n + ' You can provide an Observable, Promise, Array, or Iterable.';\n throw new TypeError(msg);\n }\n };\n\n /** PURE_IMPORTS_START _InnerSubscriber,_subscribeTo,_Observable PURE_IMPORTS_END */\n function subscribeToResult(outerSubscriber, result, outerValue, outerIndex, innerSubscriber) {\n if (innerSubscriber === void 0) {\n innerSubscriber = new InnerSubscriber(outerSubscriber, outerValue, outerIndex);\n }\n if (innerSubscriber.closed) {\n return undefined;\n }\n if (result instanceof Observable) {\n return result.subscribe(innerSubscriber);\n }\n return subscribeTo(result)(innerSubscriber);\n }\n\n /** PURE_IMPORTS_START tslib,_util_isScheduler,_util_isArray,_OuterSubscriber,_util_subscribeToResult,_fromArray PURE_IMPORTS_END */\n var NONE = {};\n function combineLatest() {\n var observables = [];\n for (var _i = 0; _i < arguments.length; _i++) {\n observables[_i] = arguments[_i];\n }\n var resultSelector = undefined;\n var scheduler = undefined;\n if (isScheduler(observables[observables.length - 1])) {\n scheduler = observables.pop();\n }\n if (typeof observables[observables.length - 1] === 'function') {\n resultSelector = observables.pop();\n }\n if (observables.length === 1 && isArray(observables[0])) {\n observables = observables[0];\n }\n return fromArray(observables, scheduler).lift(new CombineLatestOperator(resultSelector));\n }\n var CombineLatestOperator = /*@__PURE__*/ (function () {\n function CombineLatestOperator(resultSelector) {\n this.resultSelector = resultSelector;\n }\n CombineLatestOperator.prototype.call = function (subscriber, source) {\n return source.subscribe(new CombineLatestSubscriber(subscriber, this.resultSelector));\n };\n return CombineLatestOperator;\n }());\n var CombineLatestSubscriber = /*@__PURE__*/ (function (_super) {\n __extends(CombineLatestSubscriber, _super);\n function CombineLatestSubscriber(destination, resultSelector) {\n var _this = _super.call(this, destination) || this;\n _this.resultSelector = resultSelector;\n _this.active = 0;\n _this.values = [];\n _this.observables = [];\n return _this;\n }\n CombineLatestSubscriber.prototype._next = function (observable) {\n this.values.push(NONE);\n this.observables.push(observable);\n };\n CombineLatestSubscriber.prototype._complete = function () {\n var observables = this.observables;\n var len = observables.length;\n if (len === 0) {\n this.destination.complete();\n }\n else {\n this.active = len;\n this.toRespond = len;\n for (var i = 0; i < len; i++) {\n var observable = observables[i];\n this.add(subscribeToResult(this, observable, undefined, i));\n }\n }\n };\n CombineLatestSubscriber.prototype.notifyComplete = function (unused) {\n if ((this.active -= 1) === 0) {\n this.destination.complete();\n }\n };\n CombineLatestSubscriber.prototype.notifyNext = function (_outerValue, innerValue, outerIndex) {\n var values = this.values;\n var oldVal = values[outerIndex];\n var toRespond = !this.toRespond\n ? 0\n : oldVal === NONE ? --this.toRespond : this.toRespond;\n values[outerIndex] = innerValue;\n if (toRespond === 0) {\n if (this.resultSelector) {\n this._tryResultSelector(values);\n }\n else {\n this.destination.next(values.slice());\n }\n }\n };\n CombineLatestSubscriber.prototype._tryResultSelector = function (values) {\n var result;\n try {\n result = this.resultSelector.apply(this, values);\n }\n catch (err) {\n this.destination.error(err);\n return;\n }\n this.destination.next(result);\n };\n return CombineLatestSubscriber;\n }(OuterSubscriber));\n\n /** PURE_IMPORTS_START _Observable,_Subscription,_symbol_observable PURE_IMPORTS_END */\n function scheduleObservable(input, scheduler) {\n return new Observable(function (subscriber) {\n var sub = new Subscription();\n sub.add(scheduler.schedule(function () {\n var observable$1 = input[observable]();\n sub.add(observable$1.subscribe({\n next: function (value) { sub.add(scheduler.schedule(function () { return subscriber.next(value); })); },\n error: function (err) { sub.add(scheduler.schedule(function () { return subscriber.error(err); })); },\n complete: function () { sub.add(scheduler.schedule(function () { return subscriber.complete(); })); },\n }));\n }));\n return sub;\n });\n }\n\n /** PURE_IMPORTS_START _Observable,_Subscription PURE_IMPORTS_END */\n function schedulePromise(input, scheduler) {\n return new Observable(function (subscriber) {\n var sub = new Subscription();\n sub.add(scheduler.schedule(function () {\n return input.then(function (value) {\n sub.add(scheduler.schedule(function () {\n subscriber.next(value);\n sub.add(scheduler.schedule(function () { return subscriber.complete(); }));\n }));\n }, function (err) {\n sub.add(scheduler.schedule(function () { return subscriber.error(err); }));\n });\n }));\n return sub;\n });\n }\n\n /** PURE_IMPORTS_START _Observable,_Subscription,_symbol_iterator PURE_IMPORTS_END */\n function scheduleIterable(input, scheduler) {\n if (!input) {\n throw new Error('Iterable cannot be null');\n }\n return new Observable(function (subscriber) {\n var sub = new Subscription();\n var iterator$1;\n sub.add(function () {\n if (iterator$1 && typeof iterator$1.return === 'function') {\n iterator$1.return();\n }\n });\n sub.add(scheduler.schedule(function () {\n iterator$1 = input[iterator]();\n sub.add(scheduler.schedule(function () {\n if (subscriber.closed) {\n return;\n }\n var value;\n var done;\n try {\n var result = iterator$1.next();\n value = result.value;\n done = result.done;\n }\n catch (err) {\n subscriber.error(err);\n return;\n }\n if (done) {\n subscriber.complete();\n }\n else {\n subscriber.next(value);\n this.schedule();\n }\n }));\n }));\n return sub;\n });\n }\n\n /** PURE_IMPORTS_START _symbol_observable PURE_IMPORTS_END */\n function isInteropObservable(input) {\n return input && typeof input[observable] === 'function';\n }\n\n /** PURE_IMPORTS_START _symbol_iterator PURE_IMPORTS_END */\n function isIterable(input) {\n return input && typeof input[iterator] === 'function';\n }\n\n /** PURE_IMPORTS_START _scheduleObservable,_schedulePromise,_scheduleArray,_scheduleIterable,_util_isInteropObservable,_util_isPromise,_util_isArrayLike,_util_isIterable PURE_IMPORTS_END */\n function scheduled(input, scheduler) {\n if (input != null) {\n if (isInteropObservable(input)) {\n return scheduleObservable(input, scheduler);\n }\n else if (isPromise(input)) {\n return schedulePromise(input, scheduler);\n }\n else if (isArrayLike(input)) {\n return scheduleArray(input, scheduler);\n }\n else if (isIterable(input) || typeof input === 'string') {\n return scheduleIterable(input, scheduler);\n }\n }\n throw new TypeError((input !== null && typeof input || input) + ' is not observable');\n }\n\n /** PURE_IMPORTS_START _Observable,_util_subscribeTo,_scheduled_scheduled PURE_IMPORTS_END */\n function from(input, scheduler) {\n if (!scheduler) {\n if (input instanceof Observable) {\n return input;\n }\n return new Observable(subscribeTo(input));\n }\n else {\n return scheduled(input, scheduler);\n }\n }\n\n /** PURE_IMPORTS_START tslib,_Subscriber,_Observable,_util_subscribeTo PURE_IMPORTS_END */\n var SimpleInnerSubscriber = /*@__PURE__*/ (function (_super) {\n __extends(SimpleInnerSubscriber, _super);\n function SimpleInnerSubscriber(parent) {\n var _this = _super.call(this) || this;\n _this.parent = parent;\n return _this;\n }\n SimpleInnerSubscriber.prototype._next = function (value) {\n this.parent.notifyNext(value);\n };\n SimpleInnerSubscriber.prototype._error = function (error) {\n this.parent.notifyError(error);\n this.unsubscribe();\n };\n SimpleInnerSubscriber.prototype._complete = function () {\n this.parent.notifyComplete();\n this.unsubscribe();\n };\n return SimpleInnerSubscriber;\n }(Subscriber));\n var SimpleOuterSubscriber = /*@__PURE__*/ (function (_super) {\n __extends(SimpleOuterSubscriber, _super);\n function SimpleOuterSubscriber() {\n return _super !== null && _super.apply(this, arguments) || this;\n }\n SimpleOuterSubscriber.prototype.notifyNext = function (innerValue) {\n this.destination.next(innerValue);\n };\n SimpleOuterSubscriber.prototype.notifyError = function (err) {\n this.destination.error(err);\n };\n SimpleOuterSubscriber.prototype.notifyComplete = function () {\n this.destination.complete();\n };\n return SimpleOuterSubscriber;\n }(Subscriber));\n function innerSubscribe(result, innerSubscriber) {\n if (innerSubscriber.closed) {\n return undefined;\n }\n if (result instanceof Observable) {\n return result.subscribe(innerSubscriber);\n }\n var subscription;\n try {\n subscription = subscribeTo(result)(innerSubscriber);\n }\n catch (error) {\n innerSubscriber.error(error);\n }\n return subscription;\n }\n\n /** PURE_IMPORTS_START tslib,_map,_observable_from,_innerSubscribe PURE_IMPORTS_END */\n function mergeMap(project, resultSelector, concurrent) {\n if (concurrent === void 0) {\n concurrent = Number.POSITIVE_INFINITY;\n }\n if (typeof resultSelector === 'function') {\n return function (source) { return source.pipe(mergeMap(function (a, i) { return from(project(a, i)).pipe(map(function (b, ii) { return resultSelector(a, b, i, ii); })); }, concurrent)); };\n }\n else if (typeof resultSelector === 'number') {\n concurrent = resultSelector;\n }\n return function (source) { return source.lift(new MergeMapOperator(project, concurrent)); };\n }\n var MergeMapOperator = /*@__PURE__*/ (function () {\n function MergeMapOperator(project, concurrent) {\n if (concurrent === void 0) {\n concurrent = Number.POSITIVE_INFINITY;\n }\n this.project = project;\n this.concurrent = concurrent;\n }\n MergeMapOperator.prototype.call = function (observer, source) {\n return source.subscribe(new MergeMapSubscriber(observer, this.project, this.concurrent));\n };\n return MergeMapOperator;\n }());\n var MergeMapSubscriber = /*@__PURE__*/ (function (_super) {\n __extends(MergeMapSubscriber, _super);\n function MergeMapSubscriber(destination, project, concurrent) {\n if (concurrent === void 0) {\n concurrent = Number.POSITIVE_INFINITY;\n }\n var _this = _super.call(this, destination) || this;\n _this.project = project;\n _this.concurrent = concurrent;\n _this.hasCompleted = false;\n _this.buffer = [];\n _this.active = 0;\n _this.index = 0;\n return _this;\n }\n MergeMapSubscriber.prototype._next = function (value) {\n if (this.active < this.concurrent) {\n this._tryNext(value);\n }\n else {\n this.buffer.push(value);\n }\n };\n MergeMapSubscriber.prototype._tryNext = function (value) {\n var result;\n var index = this.index++;\n try {\n result = this.project(value, index);\n }\n catch (err) {\n this.destination.error(err);\n return;\n }\n this.active++;\n this._innerSub(result);\n };\n MergeMapSubscriber.prototype._innerSub = function (ish) {\n var innerSubscriber = new SimpleInnerSubscriber(this);\n var destination = this.destination;\n destination.add(innerSubscriber);\n var innerSubscription = innerSubscribe(ish, innerSubscriber);\n if (innerSubscription !== innerSubscriber) {\n destination.add(innerSubscription);\n }\n };\n MergeMapSubscriber.prototype._complete = function () {\n this.hasCompleted = true;\n if (this.active === 0 && this.buffer.length === 0) {\n this.destination.complete();\n }\n this.unsubscribe();\n };\n MergeMapSubscriber.prototype.notifyNext = function (innerValue) {\n this.destination.next(innerValue);\n };\n MergeMapSubscriber.prototype.notifyComplete = function () {\n var buffer = this.buffer;\n this.active--;\n if (buffer.length > 0) {\n this._next(buffer.shift());\n }\n else if (this.active === 0 && this.hasCompleted) {\n this.destination.complete();\n }\n };\n return MergeMapSubscriber;\n }(SimpleOuterSubscriber));\n\n /** PURE_IMPORTS_START _mergeMap,_util_identity PURE_IMPORTS_END */\n function mergeAll(concurrent) {\n if (concurrent === void 0) {\n concurrent = Number.POSITIVE_INFINITY;\n }\n return mergeMap(identity, concurrent);\n }\n\n /** PURE_IMPORTS_START _mergeAll PURE_IMPORTS_END */\n function concatAll() {\n return mergeAll(1);\n }\n\n /** PURE_IMPORTS_START _of,_operators_concatAll PURE_IMPORTS_END */\n function concat() {\n var observables = [];\n for (var _i = 0; _i < arguments.length; _i++) {\n observables[_i] = arguments[_i];\n }\n return concatAll()(of.apply(void 0, observables));\n }\n\n /** PURE_IMPORTS_START _Observable,_util_isScheduler,_operators_mergeAll,_fromArray PURE_IMPORTS_END */\n function merge() {\n var observables = [];\n for (var _i = 0; _i < arguments.length; _i++) {\n observables[_i] = arguments[_i];\n }\n var concurrent = Number.POSITIVE_INFINITY;\n var scheduler = null;\n var last = observables[observables.length - 1];\n if (isScheduler(last)) {\n scheduler = observables.pop();\n if (observables.length > 1 && typeof observables[observables.length - 1] === 'number') {\n concurrent = observables.pop();\n }\n }\n else if (typeof last === 'number') {\n concurrent = observables.pop();\n }\n if (scheduler === null && observables.length === 1 && observables[0] instanceof Observable) {\n return observables[0];\n }\n return mergeAll(concurrent)(fromArray(observables, scheduler));\n }\n\n /** PURE_IMPORTS_START tslib,_Subscriber PURE_IMPORTS_END */\n function filter(predicate, thisArg) {\n return function filterOperatorFunction(source) {\n return source.lift(new FilterOperator(predicate, thisArg));\n };\n }\n var FilterOperator = /*@__PURE__*/ (function () {\n function FilterOperator(predicate, thisArg) {\n this.predicate = predicate;\n this.thisArg = thisArg;\n }\n FilterOperator.prototype.call = function (subscriber, source) {\n return source.subscribe(new FilterSubscriber(subscriber, this.predicate, this.thisArg));\n };\n return FilterOperator;\n }());\n var FilterSubscriber = /*@__PURE__*/ (function (_super) {\n __extends(FilterSubscriber, _super);\n function FilterSubscriber(destination, predicate, thisArg) {\n var _this = _super.call(this, destination) || this;\n _this.predicate = predicate;\n _this.thisArg = thisArg;\n _this.count = 0;\n return _this;\n }\n FilterSubscriber.prototype._next = function (value) {\n var result;\n try {\n result = this.predicate.call(this.thisArg, value, this.count++);\n }\n catch (err) {\n this.destination.error(err);\n return;\n }\n if (result) {\n this.destination.next(value);\n }\n };\n return FilterSubscriber;\n }(Subscriber));\n\n /** PURE_IMPORTS_START tslib,_Subscriber,_scheduler_async PURE_IMPORTS_END */\n function debounceTime(dueTime, scheduler) {\n if (scheduler === void 0) {\n scheduler = async;\n }\n return function (source) { return source.lift(new DebounceTimeOperator(dueTime, scheduler)); };\n }\n var DebounceTimeOperator = /*@__PURE__*/ (function () {\n function DebounceTimeOperator(dueTime, scheduler) {\n this.dueTime = dueTime;\n this.scheduler = scheduler;\n }\n DebounceTimeOperator.prototype.call = function (subscriber, source) {\n return source.subscribe(new DebounceTimeSubscriber(subscriber, this.dueTime, this.scheduler));\n };\n return DebounceTimeOperator;\n }());\n var DebounceTimeSubscriber = /*@__PURE__*/ (function (_super) {\n __extends(DebounceTimeSubscriber, _super);\n function DebounceTimeSubscriber(destination, dueTime, scheduler) {\n var _this = _super.call(this, destination) || this;\n _this.dueTime = dueTime;\n _this.scheduler = scheduler;\n _this.debouncedSubscription = null;\n _this.lastValue = null;\n _this.hasValue = false;\n return _this;\n }\n DebounceTimeSubscriber.prototype._next = function (value) {\n this.clearDebounce();\n this.lastValue = value;\n this.hasValue = true;\n this.add(this.debouncedSubscription = this.scheduler.schedule(dispatchNext, this.dueTime, this));\n };\n DebounceTimeSubscriber.prototype._complete = function () {\n this.debouncedNext();\n this.destination.complete();\n };\n DebounceTimeSubscriber.prototype.debouncedNext = function () {\n this.clearDebounce();\n if (this.hasValue) {\n var lastValue = this.lastValue;\n this.lastValue = null;\n this.hasValue = false;\n this.destination.next(lastValue);\n }\n };\n DebounceTimeSubscriber.prototype.clearDebounce = function () {\n var debouncedSubscription = this.debouncedSubscription;\n if (debouncedSubscription !== null) {\n this.remove(debouncedSubscription);\n debouncedSubscription.unsubscribe();\n this.debouncedSubscription = null;\n }\n };\n return DebounceTimeSubscriber;\n }(Subscriber));\n function dispatchNext(subscriber) {\n subscriber.debouncedNext();\n }\n\n /** PURE_IMPORTS_START tslib,_Subscriber PURE_IMPORTS_END */\n function defaultIfEmpty(defaultValue) {\n if (defaultValue === void 0) {\n defaultValue = null;\n }\n return function (source) { return source.lift(new DefaultIfEmptyOperator(defaultValue)); };\n }\n var DefaultIfEmptyOperator = /*@__PURE__*/ (function () {\n function DefaultIfEmptyOperator(defaultValue) {\n this.defaultValue = defaultValue;\n }\n DefaultIfEmptyOperator.prototype.call = function (subscriber, source) {\n return source.subscribe(new DefaultIfEmptySubscriber(subscriber, this.defaultValue));\n };\n return DefaultIfEmptyOperator;\n }());\n var DefaultIfEmptySubscriber = /*@__PURE__*/ (function (_super) {\n __extends(DefaultIfEmptySubscriber, _super);\n function DefaultIfEmptySubscriber(destination, defaultValue) {\n var _this = _super.call(this, destination) || this;\n _this.defaultValue = defaultValue;\n _this.isEmpty = true;\n return _this;\n }\n DefaultIfEmptySubscriber.prototype._next = function (value) {\n this.isEmpty = false;\n this.destination.next(value);\n };\n DefaultIfEmptySubscriber.prototype._complete = function () {\n if (this.isEmpty) {\n this.destination.next(this.defaultValue);\n }\n this.destination.complete();\n };\n return DefaultIfEmptySubscriber;\n }(Subscriber));\n\n /** PURE_IMPORTS_START tslib,_Subscriber PURE_IMPORTS_END */\n function distinctUntilChanged(compare, keySelector) {\n return function (source) { return source.lift(new DistinctUntilChangedOperator(compare, keySelector)); };\n }\n var DistinctUntilChangedOperator = /*@__PURE__*/ (function () {\n function DistinctUntilChangedOperator(compare, keySelector) {\n this.compare = compare;\n this.keySelector = keySelector;\n }\n DistinctUntilChangedOperator.prototype.call = function (subscriber, source) {\n return source.subscribe(new DistinctUntilChangedSubscriber(subscriber, this.compare, this.keySelector));\n };\n return DistinctUntilChangedOperator;\n }());\n var DistinctUntilChangedSubscriber = /*@__PURE__*/ (function (_super) {\n __extends(DistinctUntilChangedSubscriber, _super);\n function DistinctUntilChangedSubscriber(destination, compare, keySelector) {\n var _this = _super.call(this, destination) || this;\n _this.keySelector = keySelector;\n _this.hasKey = false;\n if (typeof compare === 'function') {\n _this.compare = compare;\n }\n return _this;\n }\n DistinctUntilChangedSubscriber.prototype.compare = function (x, y) {\n return x === y;\n };\n DistinctUntilChangedSubscriber.prototype._next = function (value) {\n var key;\n try {\n var keySelector = this.keySelector;\n key = keySelector ? keySelector(value) : value;\n }\n catch (err) {\n return this.destination.error(err);\n }\n var result = false;\n if (this.hasKey) {\n try {\n var compare = this.compare;\n result = compare(this.key, key);\n }\n catch (err) {\n return this.destination.error(err);\n }\n }\n else {\n this.hasKey = true;\n }\n if (!result) {\n this.key = key;\n this.destination.next(value);\n }\n };\n return DistinctUntilChangedSubscriber;\n }(Subscriber));\n\n /** PURE_IMPORTS_START tslib,_util_EmptyError,_Subscriber PURE_IMPORTS_END */\n function throwIfEmpty(errorFactory) {\n if (errorFactory === void 0) {\n errorFactory = defaultErrorFactory;\n }\n return function (source) {\n return source.lift(new ThrowIfEmptyOperator(errorFactory));\n };\n }\n var ThrowIfEmptyOperator = /*@__PURE__*/ (function () {\n function ThrowIfEmptyOperator(errorFactory) {\n this.errorFactory = errorFactory;\n }\n ThrowIfEmptyOperator.prototype.call = function (subscriber, source) {\n return source.subscribe(new ThrowIfEmptySubscriber(subscriber, this.errorFactory));\n };\n return ThrowIfEmptyOperator;\n }());\n var ThrowIfEmptySubscriber = /*@__PURE__*/ (function (_super) {\n __extends(ThrowIfEmptySubscriber, _super);\n function ThrowIfEmptySubscriber(destination, errorFactory) {\n var _this = _super.call(this, destination) || this;\n _this.errorFactory = errorFactory;\n _this.hasValue = false;\n return _this;\n }\n ThrowIfEmptySubscriber.prototype._next = function (value) {\n this.hasValue = true;\n this.destination.next(value);\n };\n ThrowIfEmptySubscriber.prototype._complete = function () {\n if (!this.hasValue) {\n var err = void 0;\n try {\n err = this.errorFactory();\n }\n catch (e) {\n err = e;\n }\n this.destination.error(err);\n }\n else {\n return this.destination.complete();\n }\n };\n return ThrowIfEmptySubscriber;\n }(Subscriber));\n function defaultErrorFactory() {\n return new EmptyError();\n }\n\n /** PURE_IMPORTS_START tslib,_Subscriber,_util_ArgumentOutOfRangeError,_observable_empty PURE_IMPORTS_END */\n function take(count) {\n return function (source) {\n if (count === 0) {\n return empty$1();\n }\n else {\n return source.lift(new TakeOperator(count));\n }\n };\n }\n var TakeOperator = /*@__PURE__*/ (function () {\n function TakeOperator(total) {\n this.total = total;\n if (this.total < 0) {\n throw new ArgumentOutOfRangeError;\n }\n }\n TakeOperator.prototype.call = function (subscriber, source) {\n return source.subscribe(new TakeSubscriber(subscriber, this.total));\n };\n return TakeOperator;\n }());\n var TakeSubscriber = /*@__PURE__*/ (function (_super) {\n __extends(TakeSubscriber, _super);\n function TakeSubscriber(destination, total) {\n var _this = _super.call(this, destination) || this;\n _this.total = total;\n _this.count = 0;\n return _this;\n }\n TakeSubscriber.prototype._next = function (value) {\n var total = this.total;\n var count = ++this.count;\n if (count <= total) {\n this.destination.next(value);\n if (count === total) {\n this.destination.complete();\n this.unsubscribe();\n }\n }\n };\n return TakeSubscriber;\n }(Subscriber));\n\n /** PURE_IMPORTS_START tslib,_Subscriber,_Subscription PURE_IMPORTS_END */\n function finalize(callback) {\n return function (source) { return source.lift(new FinallyOperator(callback)); };\n }\n var FinallyOperator = /*@__PURE__*/ (function () {\n function FinallyOperator(callback) {\n this.callback = callback;\n }\n FinallyOperator.prototype.call = function (subscriber, source) {\n return source.subscribe(new FinallySubscriber(subscriber, this.callback));\n };\n return FinallyOperator;\n }());\n var FinallySubscriber = /*@__PURE__*/ (function (_super) {\n __extends(FinallySubscriber, _super);\n function FinallySubscriber(destination, callback) {\n var _this = _super.call(this, destination) || this;\n _this.add(new Subscription(callback));\n return _this;\n }\n return FinallySubscriber;\n }(Subscriber));\n\n /** PURE_IMPORTS_START _util_EmptyError,_filter,_take,_defaultIfEmpty,_throwIfEmpty,_util_identity PURE_IMPORTS_END */\n function first(predicate, defaultValue) {\n var hasDefaultValue = arguments.length >= 2;\n return function (source) { return source.pipe(predicate ? filter(function (v, i) { return predicate(v, i, source); }) : identity, take(1), hasDefaultValue ? defaultIfEmpty(defaultValue) : throwIfEmpty(function () { return new EmptyError(); })); };\n }\n\n /** PURE_IMPORTS_START _observable_ConnectableObservable PURE_IMPORTS_END */\n function multicast(subjectOrSubjectFactory, selector) {\n return function multicastOperatorFunction(source) {\n var subjectFactory;\n if (typeof subjectOrSubjectFactory === 'function') {\n subjectFactory = subjectOrSubjectFactory;\n }\n else {\n subjectFactory = function subjectFactory() {\n return subjectOrSubjectFactory;\n };\n }\n if (typeof selector === 'function') {\n return source.lift(new MulticastOperator(subjectFactory, selector));\n }\n var connectable = Object.create(source, connectableObservableDescriptor);\n connectable.source = source;\n connectable.subjectFactory = subjectFactory;\n return connectable;\n };\n }\n var MulticastOperator = /*@__PURE__*/ (function () {\n function MulticastOperator(subjectFactory, selector) {\n this.subjectFactory = subjectFactory;\n this.selector = selector;\n }\n MulticastOperator.prototype.call = function (subscriber, source) {\n var selector = this.selector;\n var subject = this.subjectFactory();\n var subscription = selector(subject).subscribe(subscriber);\n subscription.add(source.subscribe(subject));\n return subscription;\n };\n return MulticastOperator;\n }());\n\n /** PURE_IMPORTS_START _multicast,_refCount,_Subject PURE_IMPORTS_END */\n function shareSubjectFactory() {\n return new Subject();\n }\n function share() {\n return function (source) { return refCount()(multicast(shareSubjectFactory)(source)); };\n }\n\n /** PURE_IMPORTS_START _ReplaySubject PURE_IMPORTS_END */\n function shareReplay(configOrBufferSize, windowTime, scheduler) {\n var config;\n if (configOrBufferSize && typeof configOrBufferSize === 'object') {\n config = configOrBufferSize;\n }\n else {\n config = {\n bufferSize: configOrBufferSize,\n windowTime: windowTime,\n refCount: false,\n scheduler: scheduler,\n };\n }\n return function (source) { return source.lift(shareReplayOperator(config)); };\n }\n function shareReplayOperator(_a) {\n var _b = _a.bufferSize, bufferSize = _b === void 0 ? Number.POSITIVE_INFINITY : _b, _c = _a.windowTime, windowTime = _c === void 0 ? Number.POSITIVE_INFINITY : _c, useRefCount = _a.refCount, scheduler = _a.scheduler;\n var subject;\n var refCount = 0;\n var subscription;\n var hasError = false;\n var isComplete = false;\n return function shareReplayOperation(source) {\n refCount++;\n var innerSub;\n if (!subject || hasError) {\n hasError = false;\n subject = new ReplaySubject(bufferSize, windowTime, scheduler);\n innerSub = subject.subscribe(this);\n subscription = source.subscribe({\n next: function (value) {\n subject.next(value);\n },\n error: function (err) {\n hasError = true;\n subject.error(err);\n },\n complete: function () {\n isComplete = true;\n subscription = undefined;\n subject.complete();\n },\n });\n if (isComplete) {\n subscription = undefined;\n }\n }\n else {\n innerSub = subject.subscribe(this);\n }\n this.add(function () {\n refCount--;\n innerSub.unsubscribe();\n innerSub = undefined;\n if (subscription && !isComplete && useRefCount && refCount === 0) {\n subscription.unsubscribe();\n subscription = undefined;\n subject = undefined;\n }\n });\n };\n }\n\n /** PURE_IMPORTS_START _observable_concat,_util_isScheduler PURE_IMPORTS_END */\n function startWith() {\n var array = [];\n for (var _i = 0; _i < arguments.length; _i++) {\n array[_i] = arguments[_i];\n }\n var scheduler = array[array.length - 1];\n if (isScheduler(scheduler)) {\n array.pop();\n return function (source) { return concat(array, source, scheduler); };\n }\n else {\n return function (source) { return concat(array, source); };\n }\n }\n\n /** PURE_IMPORTS_START tslib,_map,_observable_from,_innerSubscribe PURE_IMPORTS_END */\n function switchMap(project, resultSelector) {\n if (typeof resultSelector === 'function') {\n return function (source) { return source.pipe(switchMap(function (a, i) { return from(project(a, i)).pipe(map(function (b, ii) { return resultSelector(a, b, i, ii); })); })); };\n }\n return function (source) { return source.lift(new SwitchMapOperator(project)); };\n }\n var SwitchMapOperator = /*@__PURE__*/ (function () {\n function SwitchMapOperator(project) {\n this.project = project;\n }\n SwitchMapOperator.prototype.call = function (subscriber, source) {\n return source.subscribe(new SwitchMapSubscriber(subscriber, this.project));\n };\n return SwitchMapOperator;\n }());\n var SwitchMapSubscriber = /*@__PURE__*/ (function (_super) {\n __extends(SwitchMapSubscriber, _super);\n function SwitchMapSubscriber(destination, project) {\n var _this = _super.call(this, destination) || this;\n _this.project = project;\n _this.index = 0;\n return _this;\n }\n SwitchMapSubscriber.prototype._next = function (value) {\n var result;\n var index = this.index++;\n try {\n result = this.project(value, index);\n }\n catch (error) {\n this.destination.error(error);\n return;\n }\n this._innerSub(result);\n };\n SwitchMapSubscriber.prototype._innerSub = function (result) {\n var innerSubscription = this.innerSubscription;\n if (innerSubscription) {\n innerSubscription.unsubscribe();\n }\n var innerSubscriber = new SimpleInnerSubscriber(this);\n var destination = this.destination;\n destination.add(innerSubscriber);\n this.innerSubscription = innerSubscribe(result, innerSubscriber);\n if (this.innerSubscription !== innerSubscriber) {\n destination.add(this.innerSubscription);\n }\n };\n SwitchMapSubscriber.prototype._complete = function () {\n var innerSubscription = this.innerSubscription;\n if (!innerSubscription || innerSubscription.closed) {\n _super.prototype._complete.call(this);\n }\n this.unsubscribe();\n };\n SwitchMapSubscriber.prototype._unsubscribe = function () {\n this.innerSubscription = undefined;\n };\n SwitchMapSubscriber.prototype.notifyComplete = function () {\n this.innerSubscription = undefined;\n if (this.isStopped) {\n _super.prototype._complete.call(this);\n }\n };\n SwitchMapSubscriber.prototype.notifyNext = function (innerValue) {\n this.destination.next(innerValue);\n };\n return SwitchMapSubscriber;\n }(SimpleOuterSubscriber));\n\n /** PURE_IMPORTS_START tslib,_innerSubscribe PURE_IMPORTS_END */\n function takeUntil(notifier) {\n return function (source) { return source.lift(new TakeUntilOperator(notifier)); };\n }\n var TakeUntilOperator = /*@__PURE__*/ (function () {\n function TakeUntilOperator(notifier) {\n this.notifier = notifier;\n }\n TakeUntilOperator.prototype.call = function (subscriber, source) {\n var takeUntilSubscriber = new TakeUntilSubscriber(subscriber);\n var notifierSubscription = innerSubscribe(this.notifier, new SimpleInnerSubscriber(takeUntilSubscriber));\n if (notifierSubscription && !takeUntilSubscriber.seenValue) {\n takeUntilSubscriber.add(notifierSubscription);\n return source.subscribe(takeUntilSubscriber);\n }\n return takeUntilSubscriber;\n };\n return TakeUntilOperator;\n }());\n var TakeUntilSubscriber = /*@__PURE__*/ (function (_super) {\n __extends(TakeUntilSubscriber, _super);\n function TakeUntilSubscriber(destination) {\n var _this = _super.call(this, destination) || this;\n _this.seenValue = false;\n return _this;\n }\n TakeUntilSubscriber.prototype.notifyNext = function () {\n this.seenValue = true;\n this.complete();\n };\n TakeUntilSubscriber.prototype.notifyComplete = function () {\n };\n return TakeUntilSubscriber;\n }(SimpleOuterSubscriber));\n\n /** PURE_IMPORTS_START tslib,_Subscriber PURE_IMPORTS_END */\n function takeWhile(predicate, inclusive) {\n if (inclusive === void 0) {\n inclusive = false;\n }\n return function (source) {\n return source.lift(new TakeWhileOperator(predicate, inclusive));\n };\n }\n var TakeWhileOperator = /*@__PURE__*/ (function () {\n function TakeWhileOperator(predicate, inclusive) {\n this.predicate = predicate;\n this.inclusive = inclusive;\n }\n TakeWhileOperator.prototype.call = function (subscriber, source) {\n return source.subscribe(new TakeWhileSubscriber(subscriber, this.predicate, this.inclusive));\n };\n return TakeWhileOperator;\n }());\n var TakeWhileSubscriber = /*@__PURE__*/ (function (_super) {\n __extends(TakeWhileSubscriber, _super);\n function TakeWhileSubscriber(destination, predicate, inclusive) {\n var _this = _super.call(this, destination) || this;\n _this.predicate = predicate;\n _this.inclusive = inclusive;\n _this.index = 0;\n return _this;\n }\n TakeWhileSubscriber.prototype._next = function (value) {\n var destination = this.destination;\n var result;\n try {\n result = this.predicate(value, this.index++);\n }\n catch (err) {\n destination.error(err);\n return;\n }\n this.nextOrComplete(value, result);\n };\n TakeWhileSubscriber.prototype.nextOrComplete = function (value, predicateResult) {\n var destination = this.destination;\n if (Boolean(predicateResult)) {\n destination.next(value);\n }\n else {\n if (this.inclusive) {\n destination.next(value);\n }\n destination.complete();\n }\n };\n return TakeWhileSubscriber;\n }(Subscriber));\n\n /** PURE_IMPORTS_START tslib,_innerSubscribe PURE_IMPORTS_END */\n var defaultThrottleConfig = {\n leading: true,\n trailing: false\n };\n\n /** PURE_IMPORTS_START tslib,_Subscriber,_scheduler_async,_throttle PURE_IMPORTS_END */\n function throttleTime(duration, scheduler, config) {\n if (scheduler === void 0) {\n scheduler = async;\n }\n if (config === void 0) {\n config = defaultThrottleConfig;\n }\n return function (source) { return source.lift(new ThrottleTimeOperator(duration, scheduler, config.leading, config.trailing)); };\n }\n var ThrottleTimeOperator = /*@__PURE__*/ (function () {\n function ThrottleTimeOperator(duration, scheduler, leading, trailing) {\n this.duration = duration;\n this.scheduler = scheduler;\n this.leading = leading;\n this.trailing = trailing;\n }\n ThrottleTimeOperator.prototype.call = function (subscriber, source) {\n return source.subscribe(new ThrottleTimeSubscriber(subscriber, this.duration, this.scheduler, this.leading, this.trailing));\n };\n return ThrottleTimeOperator;\n }());\n var ThrottleTimeSubscriber = /*@__PURE__*/ (function (_super) {\n __extends(ThrottleTimeSubscriber, _super);\n function ThrottleTimeSubscriber(destination, duration, scheduler, leading, trailing) {\n var _this = _super.call(this, destination) || this;\n _this.duration = duration;\n _this.scheduler = scheduler;\n _this.leading = leading;\n _this.trailing = trailing;\n _this._hasTrailingValue = false;\n _this._trailingValue = null;\n return _this;\n }\n ThrottleTimeSubscriber.prototype._next = function (value) {\n if (this.throttled) {\n if (this.trailing) {\n this._trailingValue = value;\n this._hasTrailingValue = true;\n }\n }\n else {\n this.add(this.throttled = this.scheduler.schedule(dispatchNext$1, this.duration, { subscriber: this }));\n if (this.leading) {\n this.destination.next(value);\n }\n else if (this.trailing) {\n this._trailingValue = value;\n this._hasTrailingValue = true;\n }\n }\n };\n ThrottleTimeSubscriber.prototype._complete = function () {\n if (this._hasTrailingValue) {\n this.destination.next(this._trailingValue);\n this.destination.complete();\n }\n else {\n this.destination.complete();\n }\n };\n ThrottleTimeSubscriber.prototype.clearThrottle = function () {\n var throttled = this.throttled;\n if (throttled) {\n if (this.trailing && this._hasTrailingValue) {\n this.destination.next(this._trailingValue);\n this._trailingValue = null;\n this._hasTrailingValue = false;\n }\n throttled.unsubscribe();\n this.remove(throttled);\n this.throttled = null;\n }\n };\n return ThrottleTimeSubscriber;\n }(Subscriber));\n function dispatchNext$1(arg) {\n var subscriber = arg.subscriber;\n subscriber.clearThrottle();\n }\n\n function isDefined(item) {\n return typeof item !== 'undefined' && item !== null;\n }\n function isObject$1(o) {\n return typeof o === 'object' && o != null;\n }\n /** Type guard for filtering out undefined */\n function omitUndefinedElements(item) {\n return item !== null && typeof item !== 'undefined';\n }\n const changesProcessed = new Subject();\n const guidRegex = /^[0-9a-f]{8}-[0-9a-f]{4}-[1-5][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$/i;\n function isGuid(str) {\n return guidRegex.test(str);\n }\n const absoluteUrlRegex = /^(?:[a-z]+:)?\\/\\//i;\n function isAbsoluteUrl(url) {\n return absoluteUrlRegex.test(url);\n }\n function appendOrIncrementNumber(name) {\n const lastNumber = name.replace(/\\D*(\\d*)\\s*$/, '$1');\n if (lastNumber === '') {\n return name + ' 2';\n }\n else {\n return name.replace(/\\d*\\s*$/, (parseInt(lastNumber, 10) + 1).toString());\n }\n }\n /** UUID generation based on: https://github.com/jchook/uuid-random/blob/master/index.js */\n class UUID {\n static initialize() {\n // Pre-calculate toString(16) for speed\n for (UUID.i = 0; UUID.i < 256; UUID.i++) {\n UUID.hexBytes[UUID.i] = (UUID.i + 0x100).toString(16).substr(1);\n }\n }\n // Buffer some random bytes for speed\n static randomBytesBuffered(n) {\n if (!UUID.buf || UUID.bufIdx + n > UUID.bufferSize) {\n UUID.bufIdx = 0;\n UUID.buf = crypto.getRandomValues(new Uint8Array(UUID.bufferSize));\n }\n return UUID.buf.slice(UUID.bufIdx, (UUID.bufIdx += n));\n }\n // uuid.bin\n static uuidBin() {\n const b = UUID.randomBytesBuffered(16);\n b[6] = (b[6] & 0x0f) | 0x40;\n b[8] = (b[8] & 0x3f) | 0x80;\n return b;\n }\n static generate() {\n const b = UUID.uuidBin();\n return (UUID.hexBytes[b[0]] +\n UUID.hexBytes[b[1]] +\n UUID.hexBytes[b[2]] +\n UUID.hexBytes[b[3]] +\n '-' +\n UUID.hexBytes[b[4]] +\n UUID.hexBytes[b[5]] +\n '-' +\n UUID.hexBytes[b[6]] +\n UUID.hexBytes[b[7]] +\n '-' +\n UUID.hexBytes[b[8]] +\n UUID.hexBytes[b[9]] +\n '-' +\n UUID.hexBytes[b[10]] +\n UUID.hexBytes[b[11]] +\n UUID.hexBytes[b[12]] +\n UUID.hexBytes[b[13]] +\n UUID.hexBytes[b[14]] +\n UUID.hexBytes[b[15]]);\n }\n }\n UUID.bufIdx = 0;\n UUID.hexBytes = new Array();\n UUID.bufferSize = 4096;\n UUID.initialize();\n\n class SceneHistory {\n constructor() {\n this.active = false;\n this.changes = [];\n this.undoneChanges = [];\n this.ignoreChangesFromObjects = new Set();\n this.state = new BehaviorSubject([]);\n this.undoneState = new BehaviorSubject([]);\n }\n updateState() {\n return this.state.next(this.getMergedChanges(this.changes));\n }\n updateUndoneState() {\n return this.undoneState.next(this.getMergedChanges(this.undoneChanges));\n }\n getMergedChanges(changes) {\n return changes.map((c, index) => {\n const changePart = Array.from(c.change.values());\n return {\n description: c.description,\n index,\n nodes: changePart.map(node => node.node),\n // Return a set of properties to drop duplicate array elements\n properties: new Set(changePart.flatMap(node => node.properties.flatMap(property => property.name))).values(),\n };\n });\n }\n ignoreUntilStable(object) {\n if (this.ignoreChangesFromObjects.size === 0) {\n changesProcessed.pipe(take(1)).subscribe(() => {\n this.ignoreChangesFromObjects.clear();\n });\n }\n this.ignoreChangesFromObjects.add(object);\n }\n start() {\n this.active = true;\n }\n stop() {\n this.active = false;\n }\n /**\n * Adds a change to the history. If you added an open-ended change\n * @param description\n * @param processFunction\n * @param undoFunction\n */\n addChange(description, processFunction, undoFunction) {\n // if (this.openChange) {\n // console.warn(\n // 'Starting a new history change with a previous change still open. You may get a strange history state.',\n // `Change still open: ${this.openChange.description}`,\n // `New change: ${description}`\n // );\n // }\n let newChange;\n if (!this.openChange) {\n newChange = this.openChange = {\n change: new Map(),\n description,\n undoFunction,\n };\n }\n processFunction(false);\n const done = (description) => {\n if (this.openChange) {\n if (this.openChange.change.size > 0 || this.openChange.undoFunction !== undefined) {\n if (description) {\n this.openChange.description = description;\n }\n this.changes.push(this.openChange);\n }\n delete this.openChange;\n this.updateState();\n if (this.undoneChanges.length > 0) {\n this.undoneChanges.clear();\n this.updateUndoneState();\n }\n }\n };\n if (undoFunction) {\n this.openChange.redoFunction = () => processFunction(true);\n done();\n }\n else if (newChange) {\n changesProcessed.pipe(take(1)).subscribe(done);\n }\n }\n undo() {\n if (this.openChange) {\n throw Error(\"Can't undo while there are open changes\");\n }\n if (this.changes.length < 1) {\n return;\n }\n const lastChangeSet = this.changes.pop();\n if (lastChangeSet) {\n this.undoneChanges.push(lastChangeSet);\n if (lastChangeSet.undoFunction) {\n lastChangeSet.undoFunction();\n }\n else {\n this.processChangeSet(lastChangeSet, change => {\n // Undo changes in reverse so they are consistent with how they were added.\n change.properties\n .slice()\n .reverse()\n .forEach(property => {\n const restore = property.previousValue;\n switch (property.operation) {\n case exports.EChangeOperation.CreateAndPush:\n change.node.pop(true);\n break;\n case exports.EChangeOperation.Push:\n change.node.pop();\n break;\n case exports.EChangeOperation.Unshift:\n change.node.shift();\n break;\n case exports.EChangeOperation.Shift:\n change.node.unshift(property.previousValue);\n if (isTrackedClass(restore)) {\n restore.isModified = true;\n }\n break;\n case exports.EChangeOperation.Pop:\n change.node.push(property.previousValue);\n if (isTrackedClass(restore)) {\n restore.isModified = true;\n }\n break;\n case exports.EChangeOperation.Remove:\n case exports.EChangeOperation.Splice:\n // Property name is the index of the splice\n // Remove all the added values, add back in the old ones\n change.node.splice(property.name, property.newValue.length, ...restore);\n if (Array.isArray(restore) && restore.length > 0) {\n restore.forEach(restoreItem => {\n if (isTrackedClass(restoreItem)) {\n restoreItem.isModified = true;\n }\n });\n }\n break;\n default:\n change.node[property.name] = property.previousValue;\n }\n });\n });\n }\n this.updateState();\n this.updateUndoneState();\n }\n }\n redo() {\n if (this.openChange) {\n throw Error(\"Can't redo while there are open changes\");\n }\n if (this.undoneChanges.length < 1) {\n return;\n }\n const lastUndoneChangeSet = this.undoneChanges.pop();\n if (lastUndoneChangeSet) {\n this.changes.push(lastUndoneChangeSet);\n if (lastUndoneChangeSet.redoFunction) {\n lastUndoneChangeSet.redoFunction();\n }\n else {\n this.processChangeSet(lastUndoneChangeSet, change => {\n // Redo changes in the initial order\n change.properties.forEach(property => {\n const restore = property.newValue;\n switch (property.operation) {\n case exports.EChangeOperation.CreateAndPush:\n change.node.createAndPush(property.newValue);\n if (isTrackedClass(restore)) {\n restore.isModified = true;\n }\n break;\n case exports.EChangeOperation.Push:\n change.node.push(property.newValue);\n break;\n case exports.EChangeOperation.Unshift:\n change.node.unshift(property.newValue);\n if (isTrackedClass(restore)) {\n restore.isModified = true;\n }\n break;\n case exports.EChangeOperation.Shift:\n change.node.shift();\n break;\n case exports.EChangeOperation.Pop:\n change.node.pop();\n break;\n case exports.EChangeOperation.Remove:\n property.previousValue.length === 1 && change.node.remove(property.previousValue[0]);\n break;\n case exports.EChangeOperation.Splice:\n change.node.splice(property.name, property.previousValue.length, ...restore);\n if (Array.isArray(restore) && restore.length > 0) {\n restore.forEach(restoreItem => {\n if (isTrackedClass(restoreItem)) {\n restoreItem.isModified = true;\n }\n });\n }\n break;\n default:\n change.node[property.name] = property.newValue;\n }\n });\n });\n }\n this.updateState();\n this.updateUndoneState();\n }\n }\n undoTo(index) {\n while (this.changes.length > index + 1) {\n this.undo();\n }\n }\n redoTo(index) {\n while (this.undoneChanges.length > index) {\n this.redo();\n }\n }\n // startGroupingAllChanges() {\n // this.groupAllIncomingChanges = true;\n // this.changes.push(new Map());\n // }\n // stopGroupingAllChanges() {\n // this.groupAllIncomingChanges = false;\n // this.removeLastChangeIfEmpty();\n // this.updateState();\n // }\n // startGroupingObjectsChanges(objects: Array) {\n // this.groupChangesInObjects = new Set(objects.filter(isTrackedClass));\n // this.changes.push(new Map());\n // }\n // stopGroupingObjectsChanges() {\n // this.groupChangesInObjects = undefined;\n // this.removeLastChangeIfEmpty();\n // }\n changeEvent(object, property, value, originalValue, operation) {\n if (this.active &&\n (isTrackedClass(object) || Array.isArray(object)) &&\n value !== originalValue &&\n !this.ignoreChangesFromObjects.has(object)) {\n if (!this.openChange) {\n return;\n }\n const openChange = this.openChange;\n let lastChangeOnObject = openChange.change.get(object);\n if (!lastChangeOnObject) {\n lastChangeOnObject = { node: object, properties: [] };\n openChange.change.set(object, lastChangeOnObject);\n }\n const lastChangeOnObjectProperty = lastChangeOnObject.properties.find(openChange => openChange.name === property);\n if (!lastChangeOnObjectProperty) {\n lastChangeOnObject.properties.push({\n name: property,\n previousValue: originalValue,\n newValue: value,\n operation,\n });\n }\n else {\n lastChangeOnObjectProperty.newValue = value;\n }\n }\n }\n changeEvents(object, changes) {\n if (this.active &&\n (isTrackedClass(object) || Array.isArray(object)) &&\n !this.ignoreChangesFromObjects.has(object)) {\n changes.forEach(change => {\n this.changeEvent(object, change.property, change.value, change.originalValue, change.operation);\n });\n }\n }\n removeLastChange(object, property) {\n // todo\n }\n processChangeSet(changeSet, process) {\n changeSet.change.forEach(change => process(change));\n }\n }\n const history = new SceneHistory();\n\n function nextTick() {\n changesProcessed.next();\n }\n const __arrayProtoFunctions = {};\n ['push', 'unshift', 'shift', 'pop', 'splice', 'reorder', 'clear'].forEach((arrayFunc) => {\n __arrayProtoFunctions[arrayFunc] = Array.prototype[arrayFunc];\n // We shouldn't need this code after the Vue update to 3.2. Was needed to fix reactivity on arrays for Vue\n // Delete this when no confirmed that there are no issues with reactivity in creator\n // Array.prototype[arrayFunc] = function(...args: any) {\n // if(isTrackedArray(this)) {\n // return TrackedArray.prototype[arrayFunc].apply(this, args);\n // } else {\n // return __arrayProtoFunctions[arrayFunc].apply(this, args);\n // }\n // }\n });\n class TrackedArray extends Array {\n constructor(parentObj, propertyName, changesIgnored, historyIgnored, source) {\n super();\n this.parentObj = parentObj;\n this.propertyName = propertyName;\n this.changesIgnored = changesIgnored;\n this.historyIgnored = historyIgnored;\n this.$isTracked = true;\n if (source) {\n for (let i = 0; i < source.length; i++) {\n __arrayProtoFunctions.push.call(this, source[i]);\n }\n }\n }\n push(...items) {\n if (!this.changesIgnored) {\n this.parentObj.setPropertyIsModified(this.propertyName, true);\n if (!this.historyIgnored) {\n history.changeEvents(this, items.map(item => ({\n property: this.length,\n operation: exports.EChangeOperation.Push,\n value: item,\n originalValue: undefined,\n })));\n }\n }\n const res = __arrayProtoFunctions.push.apply(this, items); // super.push(...items);\n for (const item of items) {\n this.parentObj.addedToChildArray(this.parentObj, item, this.propertyName);\n }\n return res;\n }\n unshift(...items) {\n if (!this.changesIgnored) {\n this.parentObj.setPropertyIsModified(this.propertyName, true);\n if (!this.historyIgnored) {\n history.changeEvents(this, items.map(item => ({\n property: 0,\n operation: exports.EChangeOperation.Push,\n value: item,\n originalValue: undefined,\n })));\n }\n }\n const res = __arrayProtoFunctions.unshift.apply(this, items); // super.unshift(...items);\n for (const item of items) {\n this.parentObj.addedToChildArray(this.parentObj, item, this.propertyName);\n }\n return res;\n }\n shift(destroy) {\n const result = __arrayProtoFunctions.shift.apply(this); // super.shift();\n if (!this.changesIgnored) {\n this.parentObj.setPropertyIsModified(this.propertyName, true);\n if (!this.historyIgnored) {\n history.changeEvent(this, 0, undefined, result, exports.EChangeOperation.Shift);\n }\n }\n if (destroy && isTrackedClass(result)) {\n result.destroy();\n }\n this.parentObj.addedToChildArray(this.parentObj, result, this.propertyName);\n return result;\n }\n pop(destroy) {\n const result = __arrayProtoFunctions.pop.apply(this); // super.pop();\n if (!this.changesIgnored) {\n this.parentObj.setPropertyIsModified(this.propertyName, true);\n if (!this.historyIgnored) {\n history.changeEvent(this, this.length, undefined, result, exports.EChangeOperation.Pop);\n }\n }\n if (destroy && isTrackedClass(result)) {\n result.destroy();\n }\n this.parentObj.addedToChildArray(this.parentObj, result, this.propertyName);\n return result;\n }\n splice(start, deleteCount, ...insert) {\n // const result = Array.prototype.splice.call(array, start, deleteCount || 0, ...insert);\n let result;\n if (insert.length === 0) {\n result = __arrayProtoFunctions.splice.apply(this, [start, deleteCount]); // super.splice(start, deleteCount);\n }\n else {\n result = __arrayProtoFunctions.splice.apply(this, [start, deleteCount || 0, ...insert]); // super.splice(start, deleteCount || 0, ...insert);\n }\n result.forEach(resultItem => {\n if (isTrackedClass(resultItem)) {\n resultItem.destroy();\n }\n });\n for (const item of result) {\n this.parentObj.removedFromChildArray(this.parentObj, item, this.propertyName);\n }\n for (const item of insert) {\n this.parentObj.addedToChildArray(this.parentObj, item, this.propertyName);\n }\n if (!this.changesIgnored) {\n this.parentObj.setPropertyIsModified(this.propertyName, true);\n if (!this.historyIgnored) {\n history.changeEvent(this, start, insert, result, exports.EChangeOperation.Splice);\n }\n }\n return result;\n }\n remove(item) {\n const index = this.indexOf(item);\n if (index > -1) {\n return this.splice(index, 1);\n }\n return [];\n }\n reorder(fromIndex, toIndex, undo) {\n __arrayProtoFunctions.reorder.apply(this, [fromIndex, toIndex, undo]);\n if (!this.changesIgnored) {\n this.parentObj.setPropertyIsModified(this.propertyName, true);\n }\n }\n }\n /** Decorator to track the properties of a class. Used in conjunction with the change tracker class.\n * Properties with this decorator will be watched by the change tracker\n * @param {Object} options decorator options object\n * @param {Function} options.onCreate function that is called when a new class instance is created. Called with the proxy object that was created.\n */\n function TrackChanges(options) {\n const onCreateCallback = options ? options.onCreate : undefined;\n const isPropertyEqualOverride = options ? options.isPropertyEqual : undefined;\n const onDestroy = options ? options.onDestroy : undefined;\n const addedToChildArray = options ? options.addedToChildArray : undefined;\n const removedFromChildArray = options ? options.removedFromChildArray : undefined;\n return (targetClass) => {\n const TrackedClass = class extends targetClass {\n get $changesTracked() {\n return true;\n }\n setPropertyIsModified() { }\n destroy() { }\n constructor(...args) {\n super(...args);\n const trackingState = {\n baseObject: this,\n isModified: false,\n };\n if (typeof this.originalValues === 'undefined') {\n Object.defineProperties(this, {\n changes: {\n value: new Map(),\n enumerable: false,\n writable: false,\n configurable: false,\n },\n originalValues: {\n value: new Map(),\n enumerable: false,\n writable: false,\n configurable: false,\n },\n _trackChanges: {\n value: false,\n enumerable: false,\n writable: true,\n configurable: false,\n },\n isModified: {\n configurable: false,\n enumerable: false,\n set: () => { },\n get: () => trackingState.isModified,\n },\n commitChanges: {\n configurable: false,\n enumerable: false,\n writable: false,\n value: (inherited = false) => commitChanges(this, inherited),\n },\n setPropertyIsModified: {\n configurable: false,\n enumerable: false,\n writable: false,\n value: setPropertyIsModified,\n },\n destroy: {\n configurable: false,\n enumerable: false,\n writable: false,\n value: () => destroy(this),\n },\n trackChanges: {\n configurable: false,\n enumerable: false,\n writable: false,\n value: (active) => (this._trackChanges = active),\n },\n rawNode: {\n configurable: false,\n enumerable: false,\n writable: false,\n value: trackingState.baseObject,\n },\n });\n }\n const changedValuesMap = this.changes;\n const originalValuesMap = this.originalValues;\n const proxy = new Proxy(this, {\n // get: (target, property) => {\n // if (property === 'isModified') {\n // return trackingState.isModified;\n // } else if (property === 'commitChanges') {\n // return (inherited: boolean = false) => commitChanges(target, inherited);\n // } else if (property === 'setPropertyIsModified') {\n // return setPropertyIsModified;\n // } else if (property === 'addedToChildArray') {\n // return addedToChildArray ? addedToChildArray : () => { }\n // } else if (property === 'removedFromChildArray') {\n // return removedFromChildArray ? removedFromChildArray : () => { }\n // } else if (property === 'destroy') {\n // return () => destroy(target);\n // } else if (property === 'trackChanges') {\n // return (active: boolean) => this._trackChanges = active;\n // } else if (property === 'rawNode') {\n // return trackingState.baseObject;\n // }\n // return target[property as string];\n // },\n set: (target, property, newValue) => {\n // Allow clearing changes by setting is modified to false.\n if (property === 'isModified') {\n if (newValue === false && newValue !== trackingState.isModified) {\n originalValuesMap.clear();\n changedValuesMap.clear();\n updateIsModified();\n }\n else if (newValue === true) {\n for (const key in target) {\n setPropertyIsModified(key, true);\n }\n }\n return true;\n // Throw type error on trying to redefine decorator functions\n }\n else if (property === 'setPropertyIsModified') {\n return false;\n }\n else {\n // If object is not being tracked, just set the value without tracking the changes\n if (!this._trackChanges) {\n return Reflect.set(trackingState.baseObject, property, newValue);\n }\n const changesIgnored = Reflect.has(target, '_changesIgnoredProperties') &&\n Reflect.get(target, '_changesIgnoredProperties').has(property);\n const historyIgnored = Reflect.has(target, '_historyIgnoredProperties') &&\n Reflect.get(target, '_historyIgnoredProperties').has(property);\n // If the value is an array, then we have to proxy the array to pick up new array elements\n // Important that this occurs even if the object is not being tracked because\n // parent node management will break without the array proxy\n if (Array.isArray(newValue)) {\n newValue = new TrackedArray(proxy, property.toString(), changesIgnored, historyIgnored, newValue);\n }\n if (!changesIgnored) {\n const originalValue = originalValuesMap.get(property.toString());\n const currentValue = Reflect.get(target, property);\n // If the new value is the same as previous, do nothing.\n if (isPropertyEqualOverride &&\n isPropertyEqualOverride(currentValue, newValue, property)) {\n return true;\n }\n else if (currentValue === newValue) {\n return true;\n }\n // Skip checks if originalValue is null. It means isModified is forced to be true until node is processed.\n if (originalValue !== null) {\n if (!originalValue) {\n // If the value is changed and was never changed before, register the change\n originalValuesMap.set(property.toString(), currentValue);\n }\n else {\n let isPropertyModified;\n // If the value was changed before, see if it's the same as the original value.\n let isEqualToOriginal;\n if (isPropertyEqualOverride) {\n isEqualToOriginal = isPropertyEqualOverride(originalValue, newValue, property);\n }\n if (typeof isEqualToOriginal !== 'undefined') {\n isPropertyModified = !isEqualToOriginal;\n }\n else {\n isPropertyModified = originalValue !== newValue;\n }\n // If the value is the same as the original, de-register the change\n if (!isPropertyModified) {\n originalValuesMap.delete(property.toString());\n }\n }\n updateIsModified();\n }\n }\n if (originalValuesMap.has(property.toString())) {\n changedValuesMap.set(property.toString(), newValue);\n }\n else {\n changedValuesMap.delete(property.toString());\n }\n return Reflect.set(trackingState.baseObject, property, newValue);\n }\n },\n });\n // Don't register changes in the history when creating new objects as they have to be registered as new objects instead.\n history.ignoreUntilStable(proxy);\n // The sync service won't have the right proxy object when a new object is created\n SyncService.registerNewObject(proxy);\n // The properties set on the object in the constructor won't be tracked because the original constructor runs before\n // the object is proxied.\n let trackedNewProperty = false;\n for (const property in this) {\n if (!this._changesIgnoredProperties || !this._changesIgnoredProperties.has(property)) {\n if (Array.isArray(this[property])) {\n const arrayHistoryIgnored = this._historyIgnoredProperties && this._historyIgnoredProperties.has(property);\n // We've already filtered out change ignored properties above\n this[property] = new TrackedArray(proxy, property.toString(), false, arrayHistoryIgnored, this[property]);\n }\n originalValuesMap.set(property, this[property]);\n changedValuesMap.set(property, this[property]);\n trackedNewProperty = true;\n }\n }\n if (trackedNewProperty) {\n updateIsModified();\n }\n if (onCreateCallback) {\n onCreateCallback(proxy);\n }\n return proxy;\n function updateIsModified() {\n // If the original value map is not empty, we have some pending changes so this object is considered modified\n if (trackingState.isModified !== originalValuesMap.size > 0) {\n trackingState.isModified = originalValuesMap.size > 0;\n SyncService.registerChangedState(proxy, trackingState.isModified);\n }\n }\n function commitChanges(target, inherited) {\n if (trackingState.isModified) {\n if (!inherited) {\n // Register the change on the history\n originalValuesMap.forEach((oldValue, property) => {\n const historyIgnored = Reflect.has(target, '_historyIgnoredProperties') &&\n Reflect.get(target, '_historyIgnoredProperties').has(property);\n if (!historyIgnored) {\n history.changeEvent(proxy, property, trackingState.baseObject[property], oldValue);\n }\n });\n originalValuesMap.clear();\n changedValuesMap.clear();\n }\n updateIsModified();\n }\n SyncService.nodeChangesProcessed(target);\n }\n function setPropertyIsModified(prop, modified) {\n if (modified) {\n // When forcing a property as modified, set the original value to null so it stays modified until processed (never matches)\n originalValuesMap.set(prop, null);\n changedValuesMap.set(prop, proxy[prop]);\n }\n else {\n originalValuesMap.delete(prop);\n changedValuesMap.delete(prop);\n }\n updateIsModified();\n }\n function destroy(target) {\n if (onDestroy) {\n onDestroy(proxy);\n }\n SyncService.registerDeleted(proxy);\n const inheritedDestroyFunc = Object.getPrototypeOf(target).destroy;\n if (typeof inheritedDestroyFunc === 'function') {\n inheritedDestroyFunc();\n }\n }\n }\n };\n Object.defineProperties(TrackedClass.prototype, {\n addedToChildArray: {\n configurable: false,\n enumerable: false,\n writable: false,\n value: addedToChildArray ? addedToChildArray : () => { },\n },\n removedFromChildArray: {\n configurable: false,\n enumerable: false,\n writable: false,\n value: removedFromChildArray ? removedFromChildArray : () => { },\n },\n });\n return TrackedClass;\n };\n }\n // eslint-disable-next-line @typescript-eslint/ban-types\n function NodeReference(classRef, ...keys) {\n return (target, propertyKey, parameterIndex) => {\n if (typeof target.nodeReferences === 'undefined') {\n Object.defineProperty(target, 'nodeReferences', {\n value: new Map(),\n enumerable: false,\n writable: false,\n configurable: false,\n });\n }\n target.nodeReferences.set(propertyKey, {\n classRef,\n limitKeys: keys.length > 0 ? new Set(keys) : undefined,\n });\n };\n }\n /** Property decorator to ignore changes on some properties. The node will not be processed if only ignored properties change. */\n function IgnoreChanges(target, propertyKey, parameterIndex) {\n if (typeof target._changesIgnoredProperties === 'undefined') {\n Object.defineProperty(target, '_changesIgnoredProperties', {\n value: new Map(),\n enumerable: false,\n writable: false,\n configurable: false,\n });\n }\n target._changesIgnoredProperties.set(propertyKey, true);\n }\n /** Property decorator to ignore history on some properties. The property changing will not trigger history additions. */\n function IgnoreHistory(target, propertyKey, parameterIndex) {\n if (typeof target._historyIgnoredProperties === 'undefined') {\n Object.defineProperty(target, '_historyIgnoredProperties', {\n value: new Map(),\n enumerable: false,\n writable: false,\n configurable: false,\n });\n }\n target._historyIgnoredProperties.set(propertyKey, true);\n }\n\n function isKbVector(object) {\n return object && object.x !== undefined && object.y !== undefined && object.z !== undefined;\n }\n function isKbColor(object) {\n return (object && object.r !== undefined && object.g !== undefined && object.b !== undefined && object.a !== undefined);\n }\n function isKbQuaternion(object) {\n return (object && object.x !== undefined && object.y !== undefined && object.z !== undefined && object.w !== undefined);\n }\n function isPrimitive(object) {\n return isKbVector(object) || isKbColor(object) || isKbQuaternion(object);\n }\n\n /** Convenience decorator that sets up default options for Kb3dObjects with the change tracker */\n function TrackChangesNode(options = {}) {\n const origAddedToChildArray = options.addedToChildArray;\n options.addedToChildArray = (parent, item, arrayPropName) => {\n if (origAddedToChildArray)\n origAddedToChildArray(parent, item, arrayPropName);\n if (instanceOf(parent, Token.Kb3dObject) && instanceOf(item, Token.Kb3dObject)) {\n item._parent = parent;\n if (isTrackedClass(parent) && parent._trackChanges) {\n item.calculateParents();\n }\n }\n };\n const origDestroy = options.onDestroy;\n options.onDestroy = (item) => {\n if (origDestroy)\n origDestroy(item);\n const children = item.getChildren();\n for (let i = 0; i < children.length; i++) {\n const child = children[i];\n if (isTrackedClass(child)) {\n child.destroy();\n }\n }\n if (item._parent && instanceOf(item._parent, Token.Kb3dObject) && instanceOf(item, Token.Kb3dObject)) {\n if (item._parent._manager) {\n item._parent._manager.unregister(item);\n }\n }\n item._parent = undefined;\n item.isModified = false;\n };\n if (!options.isPropertyEqual) {\n options.isPropertyEqual = (oldValue, newValue, propertyKey) => {\n let result;\n // Check equality on node properties by handling arrays and KB primitives. Fall back to strict equality checks for other values.\n if (Array.isArray(oldValue) && Array.isArray(newValue)) {\n if (oldValue.length === newValue.length) {\n let isEqual = true;\n for (let i = 0; i < newValue.length; i++) {\n if (newValue[i] !== oldValue[i]) {\n isEqual = false;\n break;\n }\n }\n result = isEqual;\n }\n else {\n result = false;\n }\n }\n else if (isPrimitive(newValue)) {\n result = typeof oldValue !== 'undefined' && newValue.equal(oldValue);\n }\n // console.log('checking equality', oldValue, newValue, result);\n return result;\n };\n }\n return TrackChanges(options);\n }\n\n function Kb3dModel(options) {\n if (!('trackChanges' in options))\n options.trackChanges = true; //if trackChanges was not provided, default to true\n const $type = options.token.key;\n return (constructor) => {\n //set the $type on the prototype so we can always retrieve it from an instance\n Object.defineProperty(constructor.prototype, '$type', {\n enumerable: false,\n writable: true,\n value: $type,\n });\n /** we bundle up the track changes decorator here as well, as we want our map storing the constructors of\n * our models to actually be storing the constructor of the track changes proxy... so that when a class\n * is constructed using \"create\" that it will actually create the proxy class.\n */\n if (options.trackChanges) {\n const trackChangesFactory = TrackChangesNode(options);\n const combinedConstructor = trackChangesFactory(constructor);\n options.classConstructor = combinedConstructor;\n }\n else {\n options.classConstructor = constructor;\n }\n Kb3dModels.set(options.token, options);\n };\n }\n function BaseKb3dModel(options) {\n return (constructor) => {\n options.classConstructor = constructor;\n Kb3dModels.set(options.token, options);\n };\n }\n\n /**\n * Applied to a getter or setter property, will make it enumerable\n */\n function Enumerable() {\n return function (target, propertyKey, descriptor) {\n descriptor.enumerable = true;\n };\n }\n\n const injectParameters = [];\n function Inject(token) {\n return (target, propertyKey, parameterIndex) => {\n injectParameters.push({\n token: token,\n classConstructor: target,\n parameterIndex,\n ignore: false,\n });\n };\n }\n function Ignore() {\n return (target, propertyKey, parameterIndex) => {\n injectParameters.push({\n token: Token.Ignore,\n classConstructor: target,\n parameterIndex,\n ignore: true,\n });\n };\n }\n function getConstructorParamsForModel(modelDef) {\n let constructor;\n if (modelDef.trackChanges) {\n constructor = Object.getPrototypeOf(modelDef.classConstructor);\n }\n else {\n constructor = modelDef.classConstructor;\n }\n const paramsForType = injectParameters.filter(i => i.classConstructor == constructor);\n paramsForType.sortBy(p => p.index);\n if (modelDef.classConstructor && paramsForType.length < modelDef.classConstructor.length) {\n throw new Error('You must decorate every constructor parameter with an @Inject or @Ignore decorator for type ' +\n modelDef.token.key);\n }\n return paramsForType;\n }\n\n function Kb3dService(options) {\n const $type = options.token.key;\n return (constructor) => {\n constructor.prototype.$type = $type;\n options.classConstructor = constructor;\n Kb3dModels.set(options.token, options);\n };\n }\n\n (function (eParentChangeBehavior) {\n eParentChangeBehavior[eParentChangeBehavior[\"PRESERVE_POSITION\"] = 0] = \"PRESERVE_POSITION\";\n eParentChangeBehavior[eParentChangeBehavior[\"PRESERVE_PROPERTY\"] = 1] = \"PRESERVE_PROPERTY\";\n })(exports.eParentChangeBehavior || (exports.eParentChangeBehavior = {}));\n\n (function (eNodeError) {\n eNodeError[eNodeError[\"FEATURE_REFERENCE\"] = 0] = \"FEATURE_REFERENCE\";\n eNodeError[eNodeError[\"GEOMETRY_SAVE\"] = 1] = \"GEOMETRY_SAVE\";\n })(exports.eNodeError || (exports.eNodeError = {}));\n /**\n * Base class for kb3d models. Any model that has an id and $type should derive from this class\n */\n exports.Kb3dObject = class Kb3dObject {\n constructor() {\n this._level = -1;\n this._errors = [];\n this._selected = false;\n this._expanded = false;\n this.id = UUID.generate();\n this._dynamicId = UUID.generate();\n }\n get _token() {\n return Token.withKey(this.$type);\n }\n get _rootScene() {\n return this.__rootScene;\n }\n setRootScene(scene) {\n if (this.__rootScene !== scene) {\n this.__rootScene = scene;\n if (isTrackedClass(this)) {\n this.trackChanges(!!scene);\n }\n }\n }\n get metadata() {\n if (!this._metadata)\n this._metadata = {};\n return this._metadata;\n }\n set metadata(val) {\n this._metadata = val;\n }\n clone() {\n if (this._manager) {\n return this._manager.clone(this);\n }\n }\n serialize() {\n if (this._manager) {\n return this._manager.serialize(this);\n }\n }\n serializeToJson(prettyPrint = false) {\n if (this._manager) {\n return this._manager.serializeToJson(this, { prettyPrint });\n }\n return '';\n }\n calculateParents(deletion = false, parent) {\n var _a, _b;\n // When a node is deleted and recreated, the children won't have parents because they were not re-added to the parent's arrays.\n // This solves that case by making sure the hierarchy is consistent from the top down\n if (parent && this._parent !== parent) {\n this._parent = parent;\n }\n if (this._parent) {\n this._level = this._parent._level == -1 ? -1 : this._parent._level + 1;\n this._parentScene = this._parent._parentScene;\n this.setRootScene(this._parent._rootScene);\n if (instanceOf(this, Token.SceneNode)) {\n //it's a nested scene\n this._parentScene = this;\n }\n else {\n this._manager = this._parent._manager; // a nested scene uses it's own manager, but everything else should have the same as it's parent\n }\n if (this._parentScene && !deletion) {\n (_a = this._manager) === null || _a === void 0 ? void 0 : _a.register(this);\n }\n }\n else if (instanceOf(this, Token.SceneNode) && !deletion) {\n this._level = 0;\n this._parentScene = this;\n this.setRootScene(this);\n (_b = this._manager) === null || _b === void 0 ? void 0 : _b.register(this);\n }\n else {\n this._level = -1; // -1 means this is not part of a scene node hierarchy\n // TODO add this change after release on Jan 13 2023, test to ensure it works\n // if(isTrackedClass(this)) {\n // this.trackChanges(false);\n // }\n }\n for (const c of this.getChildren()) {\n c.calculateParents(deletion, this);\n }\n }\n /**\n * Derived classes that have children should override this\n */\n getChildren() {\n return [];\n }\n /**\n * Derived classes with children should override this\n */\n getChildArrayForType(token) {\n return undefined;\n }\n getChildArray(child) {\n return this.getChildArrayForType(Token.withKey(child.$type));\n }\n addChildren(...children) {\n var _a;\n for (const c of children) {\n (_a = this.getChildArray(c)) === null || _a === void 0 ? void 0 : _a.push(c);\n }\n }\n removeChildren(...children) {\n var _a;\n for (const c of children) {\n (_a = this.getChildArray(c)) === null || _a === void 0 ? void 0 : _a.remove(c);\n }\n }\n insertChildAfter(afterChild, newChild) {\n const arr1 = this.getChildArray(afterChild);\n const arr2 = this.getChildArray(newChild);\n if (!arr1 || !arr2)\n throw 'incompatible child';\n if (arr1 !== arr2)\n throw 'children mismatch';\n const index = arr1.indexOf(afterChild);\n if (index === -1)\n throw 'could not find given child';\n arr1.splice(index + 1, 0, newChild);\n }\n getAncestors() {\n const ancestors = new Array();\n let parent = this._parent;\n while (parent != null) {\n ancestors.push(parent);\n parent = parent._parent;\n }\n return ancestors;\n }\n getNamePath() {\n let path = '';\n let self = this;\n while (self && !instanceOf(self, Token.SceneNode)) {\n path = self.name + (path ? '/' + path : '');\n self = self._parent;\n }\n return path;\n }\n /**\n * Filters children recursively based on a condition. This is an O(n) operation, so only use when performance is not paramount\n * @param predicate expression to test whether the object should be included\n * @param includeSelf whether to include this kb3dObject in the search\n * @param alwaysIncludeChildren if true, will always search children even if the parent resolves to false\n */\n filter(predicate, includeSelf = false, alwaysIncludeChildren = true) {\n const res = [];\n let included = true;\n if (includeSelf && predicate(this)) {\n res.push(this);\n }\n else {\n included = false;\n }\n if (included || alwaysIncludeChildren) {\n const children = this.getChildren();\n for (const child of children) {\n res.push(...child.filter(predicate, true, alwaysIncludeChildren));\n }\n }\n return res;\n }\n /**\n * filters children by type given a list of type tokens\n * @param tokens\n */\n filterByType(...tokens) {\n return this.filter(node => instanceOfAny(node, ...tokens));\n }\n /**\n * filters children, but only those that qualify as nodes\n * @param predicate\n * @param includeSelf\n * @param alwaysIncludeChildren\n */\n filterNodes(predicate, includeSelf = false, alwaysIncludeChildren = true) {\n const nodePredicate = (node) => node.isNode() && predicate(node);\n return this.filter(nodePredicate, includeSelf, alwaysIncludeChildren);\n }\n isNode() {\n return this._token.isNode();\n }\n };\n __decorate([\n IgnoreChanges\n ], exports.Kb3dObject.prototype, \"name\", void 0);\n __decorate([\n IgnoreChanges\n ], exports.Kb3dObject.prototype, \"$type\", void 0);\n __decorate([\n IgnoreChanges\n ], exports.Kb3dObject.prototype, \"_manager\", void 0);\n __decorate([\n IgnoreChanges\n ], exports.Kb3dObject.prototype, \"_parentScene\", void 0);\n __decorate([\n IgnoreChanges\n ], exports.Kb3dObject.prototype, \"__rootScene\", void 0);\n __decorate([\n IgnoreChanges\n ], exports.Kb3dObject.prototype, \"_level\", void 0);\n __decorate([\n IgnoreChanges\n ], exports.Kb3dObject.prototype, \"_errors\", void 0);\n __decorate([\n IgnoreChanges\n ], exports.Kb3dObject.prototype, \"_expanded\", void 0);\n __decorate([\n Enumerable()\n ], exports.Kb3dObject.prototype, \"metadata\", null);\n exports.Kb3dObject = __decorate([\n BaseKb3dModel({\n token: Token.Kb3dObject,\n })\n ], exports.Kb3dObject);\n\n exports.BaseConnector = class BaseConnector extends exports.Kb3dObject {\n constructor() {\n super(...arguments);\n this._sketchControlPoints = [];\n }\n /** Extended by child classes to return the space node parent of this connector */\n getSpaceParent() {\n return undefined;\n }\n };\n exports.BaseConnector = __decorate([\n BaseKb3dModel({\n token: Token.BaseConnector,\n })\n ], exports.BaseConnector);\n\n var Feature_1;\n (function (eMeshChangeTypes) {\n eMeshChangeTypes[eMeshChangeTypes[\"VertexIndices\"] = 0] = \"VertexIndices\";\n eMeshChangeTypes[eMeshChangeTypes[\"VertexPositions\"] = 1] = \"VertexPositions\";\n eMeshChangeTypes[eMeshChangeTypes[\"VertexNormals\"] = 2] = \"VertexNormals\";\n eMeshChangeTypes[eMeshChangeTypes[\"VertexUvs\"] = 3] = \"VertexUvs\";\n eMeshChangeTypes[eMeshChangeTypes[\"Submeshes\"] = 4] = \"Submeshes\";\n eMeshChangeTypes[eMeshChangeTypes[\"Transform\"] = 5] = \"Transform\";\n })(exports.eMeshChangeTypes || (exports.eMeshChangeTypes = {}));\n const ALL_GEOMETRY = [\n exports.eMeshChangeTypes.VertexIndices,\n exports.eMeshChangeTypes.VertexPositions,\n exports.eMeshChangeTypes.VertexNormals,\n exports.eMeshChangeTypes.VertexUvs,\n exports.eMeshChangeTypes.Submeshes,\n ];\n const ALL_CHANGES = [...ALL_GEOMETRY, exports.eMeshChangeTypes.Transform];\n exports.Feature = Feature_1 = class Feature extends exports.Kb3dObject {\n get dependsOn() {\n return this.constructor._dependsOn;\n }\n get affects() {\n return this.constructor._affects;\n }\n get affectedBy() {\n return this.constructor._affectedBy;\n }\n constructor() {\n super();\n this._suppressed = false;\n }\n getAncestors() {\n return super.getAncestors();\n }\n };\n exports.Feature = Feature_1 = __decorate([\n BaseKb3dModel({\n token: Token.Feature,\n })\n ], exports.Feature);\n function isFeatureNode(node) {\n return instanceOf(node, Token.Feature);\n }\n\n let defaults = {\n AnnotationNode: {\n enabled: true,\n label: null,\n fontFamily: null,\n fontSize: 0.1,\n lineSpacing: 0,\n preloadFontStylesheet: \"\",\n color: { \"$type\": \"KbColor\", \"r\": 1.0, \"g\": 1.0, \"b\": 1.0, \"a\": 1.0 },\n targetId: null,\n billboard: false,\n isPickable: false,\n keepOnTop: false,\n lineStyle: \"solid\",\n horizontalTextAlign: \"left\",\n verticalBoxAlign: \"middle\",\n horizontalBoxAlign: \"center\",\n offset: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n name: null,\n metadata: {},\n },\n DimensionNode: {\n target2Id: null,\n lineCapStyle: \"line\",\n axis: \"free\",\n lineThickness: 0.05,\n dimensionGap: 0.1,\n dimensionUnitPrecision: 2,\n labelOffset: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n scaleWithCamera: false,\n enabled: true,\n label: null,\n fontFamily: null,\n fontSize: 0.1,\n lineSpacing: 0,\n preloadFontStylesheet: \"\",\n color: { \"$type\": \"KbColor\", \"r\": 1.0, \"g\": 1.0, \"b\": 1.0, \"a\": 1.0 },\n targetId: null,\n billboard: false,\n isPickable: false,\n keepOnTop: false,\n lineStyle: \"solid\",\n horizontalTextAlign: \"left\",\n verticalBoxAlign: \"middle\",\n horizontalBoxAlign: \"center\",\n offset: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n name: null,\n metadata: {},\n },\n BaseConnector: {\n positionMode: \"boundingBox\",\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n direction: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 1.0, \"z\": 0.0 },\n angle: 0,\n positionZeroToOne: 0,\n interpolateDirection: true,\n name: null,\n metadata: {},\n },\n Connector: {\n enabled: true,\n positionMode: \"boundingBox\",\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n direction: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 1.0, \"z\": 0.0 },\n angle: 0,\n positionZeroToOne: 0,\n interpolateDirection: true,\n name: null,\n metadata: {},\n },\n CircularPatternFeature: {\n arc: 360,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n axis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 1.0, \"z\": 0.0 },\n count: 1,\n meshId: null,\n distance: 1,\n addEndMesh: false,\n enabled: true,\n name: null,\n metadata: {},\n },\n DeleteFacesFeature: {\n enabled: true,\n name: null,\n metadata: {},\n },\n DisplacementMapFeature: {\n minDisplacement: 0,\n maxDisplacement: 1,\n enabled: true,\n name: null,\n metadata: {},\n },\n ExtrudeFeature: {\n size: 0,\n capEnd: false,\n capStart: false,\n enabled: true,\n name: null,\n metadata: {},\n },\n Feature: {\n enabled: true,\n name: null,\n metadata: {},\n },\n JoinGeometryFeature: {\n joinType: \"cut\",\n meshId: null,\n enabled: true,\n name: null,\n metadata: {},\n },\n LinearPatternFeature: {\n translation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n count: 1,\n meshId: null,\n distance: 1,\n addEndMesh: false,\n enabled: true,\n name: null,\n metadata: {},\n },\n MirrorFeature: {\n cloneGeometry: false,\n intersect: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n normal: { \"$type\": \"KbVector\", \"x\": 1.0, \"y\": 1.0, \"z\": 1.0 },\n removeGeometry: false,\n enabled: true,\n name: null,\n metadata: {},\n },\n MoveVerticesFeature: {\n translation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scale: { \"$type\": \"KbVector\", \"x\": 1.0, \"y\": 1.0, \"z\": 1.0 },\n center: \"barycenter\",\n enabled: true,\n name: null,\n metadata: {},\n },\n NormalSmoothingFeature: {\n smooth: true,\n enabled: true,\n name: null,\n metadata: {},\n },\n PatternFeature: {\n count: 1,\n meshId: null,\n distance: 1,\n addEndMesh: false,\n enabled: true,\n name: null,\n metadata: {},\n },\n SliceFeature: {\n intersect: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n normal: { \"$type\": \"KbVector\", \"x\": 1.0, \"y\": 1.0, \"z\": 1.0 },\n removeGeometry: false,\n enabled: true,\n name: null,\n metadata: {},\n },\n SubmeshFeature: {\n uvMapping: false,\n materialId: null,\n center: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n projectionAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 1.0, \"z\": 0.0 },\n angle: 0,\n scale: { \"$type\": \"KbVector\", \"x\": 1.0, \"y\": 1.0, \"z\": 1.0 },\n uTile: 1,\n vTile: 1,\n uOffset: 0,\n vOffset: 0,\n mode: \"planar\",\n enabled: true,\n name: null,\n metadata: {},\n },\n SweepFeature: {\n start: null,\n end: null,\n path: null,\n mitredExtrusion: false,\n size: 0,\n capEnd: false,\n capStart: false,\n enabled: true,\n name: null,\n metadata: {},\n },\n TransformFeature: {\n scale: { \"$type\": \"KbVector\", \"x\": 1.0, \"y\": 1.0, \"z\": 1.0 },\n enabled: true,\n name: null,\n metadata: {},\n },\n UvMapFeature: {\n center: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n projectionAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 1.0, \"z\": 0.0 },\n angle: 0,\n scale: { \"$type\": \"KbVector\", \"x\": 1.0, \"y\": 1.0, \"z\": 1.0 },\n uTile: 1,\n vTile: 1,\n uOffset: 0,\n vOffset: 0,\n mode: \"planar\",\n enabled: true,\n name: null,\n metadata: {},\n },\n WeldVerticesFeature: {\n enabled: true,\n name: null,\n metadata: {},\n },\n Kb3dObject: {\n name: null,\n metadata: {},\n },\n DirectionalLightNode: {\n direction: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 1.0, \"z\": 0.0 },\n generatesShadows: false,\n shadowQuality: \"medium\",\n shadowResolution: \"low\",\n shadowFilter: \"simple\",\n processTransparency: false,\n intensity: 1,\n color: { \"$type\": \"KbColor\", \"r\": 0.0, \"g\": 0.0, \"b\": 0.0, \"a\": 1.0 },\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n HemisphericLightNode: {\n direction: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 1.0, \"z\": 0.0 },\n groundColor: { \"$type\": \"KbColor\", \"r\": 0.0, \"g\": 0.0, \"b\": 0.0, \"a\": 1.0 },\n intensity: 1,\n color: { \"$type\": \"KbColor\", \"r\": 0.0, \"g\": 0.0, \"b\": 0.0, \"a\": 1.0 },\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n LightNode: {\n intensity: 1,\n color: { \"$type\": \"KbColor\", \"r\": 0.0, \"g\": 0.0, \"b\": 0.0, \"a\": 1.0 },\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n PointLightNode: {\n generatesShadows: false,\n shadowQuality: \"medium\",\n shadowResolution: \"low\",\n shadowFilter: \"simple\",\n processTransparency: false,\n intensity: 1,\n color: { \"$type\": \"KbColor\", \"r\": 0.0, \"g\": 0.0, \"b\": 0.0, \"a\": 1.0 },\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n ShadowLightNode: {\n generatesShadows: false,\n shadowQuality: \"medium\",\n shadowResolution: \"low\",\n shadowFilter: \"simple\",\n processTransparency: false,\n intensity: 1,\n color: { \"$type\": \"KbColor\", \"r\": 0.0, \"g\": 0.0, \"b\": 0.0, \"a\": 1.0 },\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n SpotLightNode: {\n coneAngle: 60,\n direction: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 1.0, \"z\": 0.0 },\n generatesShadows: false,\n shadowQuality: \"medium\",\n shadowResolution: \"low\",\n shadowFilter: \"simple\",\n processTransparency: false,\n intensity: 1,\n color: { \"$type\": \"KbColor\", \"r\": 0.0, \"g\": 0.0, \"b\": 0.0, \"a\": 1.0 },\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n AdvancedMaterialNode: {\n workflowMode: \"metallicRoughness\",\n backFaceCulling: false,\n directIntensity: 1,\n emissiveIntensity: 1,\n environmentIntensity: 1,\n specularIntensity: 1,\n metallic: 0,\n roughness: 1,\n microSurface: 1,\n useSpecularOverAlpha: true,\n useRadianceOverAlpha: true,\n ambientColor: { \"$type\": \"KbColor\", \"r\": 1.0, \"g\": 1.0, \"b\": 1.0, \"a\": 1.0 },\n albedoColor: { \"$type\": \"KbColor\", \"r\": 1.0, \"g\": 1.0, \"b\": 1.0, \"a\": 1.0 },\n specularColor: { \"$type\": \"KbColor\", \"r\": 1.0, \"g\": 1.0, \"b\": 1.0, \"a\": 1.0 },\n reflectionColor: { \"$type\": \"KbColor\", \"r\": 1.0, \"g\": 1.0, \"b\": 1.0, \"a\": 1.0 },\n anisotropyEnabled: false,\n anisotropicDirection: { \"$type\": \"KbVector\", \"x\": 1.0, \"y\": 1.0, \"z\": 1.0 },\n anisotropicIntensity: 1,\n subsurfaceThickness: 1,\n subsurfaceRefractionEnabled: false,\n subsurfaceReflectionEnabled: false,\n reflectionType: \"plane\",\n probePositionTargetId: null,\n subsurfaceRefractionIndex: 1,\n subsurfaceRefractionIntensity: 1,\n subsurfaceRefractionLinkTransparency: false,\n subsurfaceTranslucencyEnabled: false,\n subsurfaceTranslucencyIntensity: 1,\n subsurfaceUseAlbedoForTintColor: true,\n subsurfaceTintColor: { \"$type\": \"KbColor\", \"r\": 1.0, \"g\": 1.0, \"b\": 1.0, \"a\": 1.0 },\n clearcoatEnabled: false,\n clearcoatIntensity: 1,\n clearcoatTintEnabled: false,\n clearcoatTintColor: { \"$type\": \"KbColor\", \"r\": 1.0, \"g\": 1.0, \"b\": 1.0, \"a\": 1.0 },\n clearcoatTintIntensity: 0.1,\n archived: false,\n location: \"local\",\n alpha: 1,\n zOffset: 0,\n doubleSided: false,\n legacyTextureStacking: true,\n viewMode: \"shaded\",\n emissiveColor: { \"$type\": \"KbColor\", \"r\": 0.0, \"g\": 0.0, \"b\": 0.0, \"a\": 1.0 },\n hasTransparency: false,\n uOffset: 0,\n vOffset: 0,\n uScale: 1,\n vScale: 1,\n uAngle: 0,\n vAngle: 0,\n wAngle: 0,\n uPivot: 0.5,\n vPivot: 0.5,\n wPivot: 0.5,\n name: null,\n metadata: {},\n },\n ColorFillTextureLayer: {\n color: { \"$type\": \"KbColor\", \"r\": 0.0, \"g\": 0.0, \"b\": 0.0, \"a\": 1.0 },\n width: 256,\n height: 256,\n enabled: true,\n strength: 1,\n tile: false,\n maintainAspect: false,\n useClippingMask: false,\n blendMode: \"normal\",\n horizontalAlign: \"center\",\n verticalAlign: \"middle\",\n horizontalFit: \"none\",\n verticalFit: \"none\",\n uOffset: 0,\n vOffset: 0,\n uScale: 1,\n vScale: 1,\n uAngle: 0,\n vAngle: 0,\n wAngle: 0,\n uPivot: 0.5,\n vPivot: 0.5,\n wPivot: 0.5,\n name: null,\n metadata: {},\n },\n FileTextureLayer: {\n width: 256,\n height: 256,\n flipNormalsRg: false,\n enabled: true,\n strength: 1,\n tile: false,\n maintainAspect: false,\n useClippingMask: false,\n blendMode: \"normal\",\n horizontalAlign: \"center\",\n verticalAlign: \"middle\",\n horizontalFit: \"none\",\n verticalFit: \"none\",\n uOffset: 0,\n vOffset: 0,\n uScale: 1,\n vScale: 1,\n uAngle: 0,\n vAngle: 0,\n wAngle: 0,\n uPivot: 0.5,\n vPivot: 0.5,\n wPivot: 0.5,\n name: null,\n metadata: {},\n },\n MaterialMapping: {\n uOffset: 0,\n vOffset: 0,\n uScale: 1,\n vScale: 1,\n uAngle: 0,\n vAngle: 0,\n wAngle: 0,\n uPivot: 0.5,\n vPivot: 0.5,\n wPivot: 0.5,\n name: null,\n metadata: {},\n },\n MaterialNode: {\n location: \"local\",\n alpha: 1,\n zOffset: 0,\n reflectionType: \"plane\",\n doubleSided: false,\n legacyTextureStacking: true,\n viewMode: \"shaded\",\n emissiveColor: { \"$type\": \"KbColor\", \"r\": 0.0, \"g\": 0.0, \"b\": 0.0, \"a\": 1.0 },\n hasTransparency: false,\n uOffset: 0,\n vOffset: 0,\n uScale: 1,\n vScale: 1,\n uAngle: 0,\n vAngle: 0,\n wAngle: 0,\n uPivot: 0.5,\n vPivot: 0.5,\n wPivot: 0.5,\n name: null,\n metadata: {},\n },\n SvgTextureLayer: {\n width: 256,\n height: 256,\n preloadFontStylesheet: \"\",\n enabled: true,\n strength: 1,\n tile: false,\n maintainAspect: false,\n useClippingMask: false,\n blendMode: \"normal\",\n horizontalAlign: \"center\",\n verticalAlign: \"middle\",\n horizontalFit: \"none\",\n verticalFit: \"none\",\n uOffset: 0,\n vOffset: 0,\n uScale: 1,\n vScale: 1,\n uAngle: 0,\n vAngle: 0,\n wAngle: 0,\n uPivot: 0.5,\n vPivot: 0.5,\n wPivot: 0.5,\n name: null,\n metadata: {},\n },\n TextTextureLayer: {\n text: \"\",\n color: { \"$type\": \"KbColor\", \"r\": 1.0, \"g\": 1.0, \"b\": 1.0, \"a\": 1.0 },\n enabled: true,\n strength: 1,\n tile: false,\n maintainAspect: false,\n useClippingMask: false,\n blendMode: \"normal\",\n horizontalAlign: \"center\",\n verticalAlign: \"middle\",\n horizontalFit: \"none\",\n verticalFit: \"none\",\n uOffset: 0,\n vOffset: 0,\n uScale: 1,\n vScale: 1,\n uAngle: 0,\n vAngle: 0,\n wAngle: 0,\n uPivot: 0.5,\n vPivot: 0.5,\n wPivot: 0.5,\n name: null,\n metadata: {},\n },\n TextureLayer: {\n enabled: true,\n strength: 1,\n tile: false,\n maintainAspect: false,\n useClippingMask: false,\n blendMode: \"normal\",\n horizontalAlign: \"center\",\n verticalAlign: \"middle\",\n horizontalFit: \"none\",\n verticalFit: \"none\",\n uOffset: 0,\n vOffset: 0,\n uScale: 1,\n vScale: 1,\n uAngle: 0,\n vAngle: 0,\n wAngle: 0,\n uPivot: 0.5,\n vPivot: 0.5,\n wPivot: 0.5,\n name: null,\n metadata: {},\n },\n MateNode: {\n enabled: true,\n connector1: null,\n connector2: null,\n mateType: \"fastened\",\n flip: false,\n offsetX: 0,\n offsetY: 0,\n offsetZ: 0,\n offsetAngle: 0,\n name: null,\n metadata: {},\n },\n BoxNode: {\n height: 1,\n width: 1,\n depth: 1,\n tessellation: 1,\n sideOrientation: \"frontSide\",\n materialId: null,\n flipSurfaceOrientation: false,\n receiveShadows: false,\n castShadows: false,\n isPickable: true,\n checkCollisions: false,\n opacity: 1,\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n ConeNode: {\n height: 1,\n radiusTop: 0,\n radiusBottom: 0.5,\n tessellation: 24,\n sideOrientation: \"frontSide\",\n materialId: null,\n flipSurfaceOrientation: false,\n receiveShadows: false,\n castShadows: false,\n isPickable: true,\n checkCollisions: false,\n opacity: 1,\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n CustomMeshNode: {\n materialId: null,\n flipSurfaceOrientation: false,\n receiveShadows: false,\n castShadows: false,\n isPickable: true,\n checkCollisions: false,\n opacity: 1,\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n CylinderNode: {\n height: 1,\n radius: 0.5,\n arc: 360,\n tessellation: 24,\n subdivisions: 1,\n capMode: \"both\",\n sideOrientation: \"frontSide\",\n materialId: null,\n flipSurfaceOrientation: false,\n receiveShadows: false,\n castShadows: false,\n isPickable: true,\n checkCollisions: false,\n opacity: 1,\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n DiscNode: {\n radius: 0.5,\n tessellation: 64,\n sideOrientation: \"frontSide\",\n materialId: null,\n flipSurfaceOrientation: false,\n receiveShadows: false,\n castShadows: false,\n isPickable: true,\n checkCollisions: false,\n opacity: 1,\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n GroundNode: {\n height: 1,\n width: 1,\n subdivisions: 1,\n sideOrientation: \"frontSide\",\n materialId: null,\n flipSurfaceOrientation: false,\n receiveShadows: false,\n castShadows: false,\n isPickable: true,\n checkCollisions: false,\n opacity: 1,\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n MeshGroupNode: {\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n MeshNode: {\n materialId: null,\n flipSurfaceOrientation: false,\n receiveShadows: false,\n castShadows: false,\n isPickable: true,\n checkCollisions: false,\n opacity: 1,\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n PlaneNode: {\n depth: 4,\n width: 4,\n segments: 1,\n sideOrientation: \"frontSide\",\n materialId: null,\n flipSurfaceOrientation: false,\n receiveShadows: false,\n castShadows: false,\n isPickable: true,\n checkCollisions: false,\n opacity: 1,\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n PrimitiveMeshNode: {\n sideOrientation: \"frontSide\",\n materialId: null,\n flipSurfaceOrientation: false,\n receiveShadows: false,\n castShadows: false,\n isPickable: true,\n checkCollisions: false,\n opacity: 1,\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n SketchControlPoint: {\n positionMode: \"absolute\",\n direction2: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": -1.0, \"z\": 0.0 },\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n direction: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 1.0, \"z\": 0.0 },\n angle: 0,\n positionZeroToOne: 0,\n interpolateDirection: true,\n name: null,\n metadata: {},\n },\n SketchNode: {\n normal: \"y\",\n addFace: false,\n closePath: false,\n tracePath: false,\n lineThickness: 1,\n pathColor: { \"$type\": \"KbColor\", \"r\": 1.0, \"g\": 1.0, \"b\": 1.0, \"a\": 1.0 },\n flipPath: false,\n clonePath: null,\n boundingXOverride: null,\n boundingYOverride: null,\n boundingZOverride: null,\n materialId: null,\n flipSurfaceOrientation: false,\n receiveShadows: false,\n castShadows: false,\n isPickable: true,\n checkCollisions: false,\n opacity: 1,\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n SketchPath: {\n pathGroupKey: \"default\",\n holeForPathGroup: null,\n type: \"points\",\n tessellation: 16,\n name: null,\n metadata: {},\n },\n SpaceNode: {\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n SphereNode: {\n tessellation: 32,\n radius: 0.5,\n latitudeStart: 0,\n latitudeSlice: 180,\n arc: 360,\n sideOrientation: \"frontSide\",\n materialId: null,\n flipSurfaceOrientation: false,\n receiveShadows: false,\n castShadows: false,\n isPickable: true,\n checkCollisions: false,\n opacity: 1,\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n TextNode: {\n curveRadius: 0,\n charSpacing: 1,\n text: \"\",\n fontFile: \"\",\n fontSize: 1,\n materialId: null,\n flipSurfaceOrientation: false,\n receiveShadows: false,\n castShadows: false,\n isPickable: true,\n checkCollisions: false,\n opacity: 1,\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n TorusNode: {\n radius: 0.5,\n thickness: 0.25,\n arc: 360,\n latitudeStart: 0,\n latitudeSlice: 360,\n tessellation: 16,\n sideOrientation: \"frontSide\",\n materialId: null,\n flipSurfaceOrientation: false,\n receiveShadows: false,\n castShadows: false,\n isPickable: true,\n checkCollisions: false,\n opacity: 1,\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n SceneNode: {\n background: \"transparent\",\n environment: \"library://adams_place_bridge\",\n environmentBlur: 0.27,\n environmentIntensity: 1,\n backgroundColor: { \"$type\": \"KbColor\", \"r\": 0.0, \"g\": 0.0, \"b\": 0.0, \"a\": 0.0 },\n ambientColor: { \"$type\": \"KbColor\", \"r\": 0.0, \"g\": 0.0, \"b\": 0.0, \"a\": 1.0 },\n renderMode: \"shaded\",\n edgeColor: { \"$type\": \"KbColor\", \"r\": 1.0, \"g\": 1.0, \"b\": 1.0, \"a\": 1.0 },\n edgeWidth: 1,\n edgeDisplayTolerance: 0.95,\n viewpointId: null,\n frameViewpoint: false,\n progressiveTextureLoad: true,\n enableDeferredSceneUpdates: true,\n enableGrid: false,\n enableGridAxes: false,\n gridSize: 10,\n gridStepSize: 1,\n gridMajorInterval: 1,\n gridNormal: \"y\",\n environmentRotation: 0,\n connectorSize: 0.1,\n enableFxaaPostProcess: false,\n enableLensEffectsPostProcess: false,\n enableSSAOPostProcess: false,\n textureSamplingMode: \"trilinear\",\n fullscreenEnabled: true,\n arEnabled: true,\n xrAllowScaling: false,\n unitsPerMeter: 1,\n xrAddShadow: true,\n xrLightingEstimation: false,\n enableHoverEffects: false,\n enableSceneOptimization: true,\n loadingExperience: \"spinner\",\n loadingImagePath: null,\n dimensionUnitScalar: 1,\n dimensionUnitSymbol: \"m\",\n conversionMap: {},\n visible: true,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n useEuler: false,\n eulerRotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n rotationAngle: 0,\n scaling: 1,\n pivot: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fixed: false,\n name: null,\n metadata: {},\n },\n ViewpointNode: {\n type: \"orbit\",\n targetMode: \"point\",\n projection: \"perspective\",\n offsetX: 0,\n offsetY: 0,\n position: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n targetPoint: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n targetMeshId: null,\n rotation: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 0.0, \"z\": 0.0 },\n fov: 50,\n allowOrbit: true,\n allowPan: true,\n allowZoom: true,\n limitMinimumTargetDistance: false,\n minTargetDistance: 0.001,\n maxTargetDistance: 10000,\n orbitAxis: { \"$type\": \"KbVector\", \"x\": 0.0, \"y\": 1.0, \"z\": 0.0 },\n limitLatOrbit: false,\n limitMinLatitude: 0,\n limitMaxLatitude: 180,\n limitLngOrbit: false,\n limitMinLongitude: -180,\n limitMaxLongitude: 180,\n name: null,\n metadata: {},\n },\n };\n\n exports.MaterialMapping = class MaterialMapping extends exports.Kb3dObject {\n constructor() {\n super(...arguments);\n this.uOffset = 0; // Define an offset on the texture to offset the u coordinates of the UVs\n this.vOffset = 0; // Define an offset on the texture to offset the v coordinates of the UVs\n this.uScale = 1; // Define an offset on the texture to scale the u coordinates of the UVs\n this.vScale = 1; // Define an offset on the texture to scale the v coordinates of the UVs\n this.uAngle = 0; // Define an offset on the texture to rotate around the u coordinates of the UVs\n this.vAngle = 0; // Define an offset on the texture to rotate around the v coordinates of the UVs\n this.wAngle = 0; // Define an offset on the texture to rotate around the w coordinates of the UVs\n this.uPivot = 0.5; // Defines the center of rotation (U)\n this.vPivot = 0.5; // Defines the center of rotation (V)\n this.wPivot = 0.5; // Defines the center of rotation (W)\n }\n };\n exports.MaterialMapping = __decorate([\n BaseKb3dModel({\n token: Token.MaterialMapping,\n })\n ], exports.MaterialMapping);\n\n const NEVER_INHERIT = new Set([\n '$type',\n 'overriddenProperties',\n '_parentMaterial',\n 'parentMaterialId',\n 'location',\n 'children',\n '_selected',\n '_expanded',\n 'id',\n 'name',\n 'visible',\n '_parent',\n '_parentScene',\n '_rootScene',\n '_level',\n ]);\n exports.MaterialNode = class MaterialNode extends exports.MaterialMapping {\n constructor(_materialService) {\n super();\n this._materialService = _materialService;\n this._overriddenPropertiesSet = new Set();\n this._initialized = false;\n }\n get materials() {\n if (!this._materials)\n this._materials = [];\n return this._materials;\n }\n set materials(val) {\n this._materials = val;\n }\n get overriddenProperties() {\n if (!this._overriddenProperties)\n this._overriddenProperties = [];\n return this._overriddenProperties;\n }\n set overriddenProperties(val) {\n this._overriddenProperties = val;\n }\n get textures() {\n if (!this._textures)\n this._textures = [];\n return this._textures;\n }\n set textures(val) {\n this._textures = val;\n }\n init() {\n this.refreshParentMaterial();\n if (!this._initialized) {\n if (isTrackedClass(this)) {\n // Always track changes on initialized materials because they may be global materials\n this.trackChanges(true);\n this.isModified = true;\n }\n this._initialized = true;\n this.textures.forEach(texture => {\n texture.init(this);\n });\n }\n }\n getChildren() {\n return [...this.textures, ...this.materials];\n }\n getChildArrayForType(token) {\n if (token.isMaterial()) {\n return this.materials;\n }\n else if (token.isTexture()) {\n return this.textures;\n }\n else {\n throw 'unexpected token';\n }\n }\n overrideProperty(property) {\n if (this.parentMaterialId) {\n this._overriddenPropertiesSet.add(property);\n }\n }\n inheritProperty(property) {\n if (this.parentMaterialId) {\n this._overriddenPropertiesSet.delete(property);\n if (this._parentMaterial) {\n this[property] =\n this._parentMaterial[property] == null\n ? defaults[this.$type][property]\n : this._parentMaterial[property];\n }\n }\n }\n getTextureByName(name) {\n return [...this.textures].find(f => f.name.isEqual(name));\n }\n inheritFromMaterial(parentMat) {\n this.parentMaterialId = parentMat.id;\n this._parentMaterial = parentMat;\n if (!this._manager) {\n return;\n }\n const manager = this._manager;\n NEVER_INHERIT.forEach(prop => {\n this._overriddenPropertiesSet.add(prop);\n });\n for (const _key in parentMat) {\n const key = _key;\n if (!this._overriddenPropertiesSet.has(key)) {\n //if this is a serialization of a texture node, then we need to deserialize it\n if (Array.isArray(parentMat[key])) {\n const currentTextures = this[key];\n parentMat[key].forEach((tex, indexOnParent) => {\n const texDef = { ...tex, name: `Base ${tex.name || tex.channel}` };\n if (!texDef.id) {\n texDef.id = `${parentMat.id}_${tex.channel}`;\n }\n const currentIndex = currentTextures.findIndex(currentTexture => currentTexture.id === texDef.id);\n if (currentIndex < 0) {\n const textureNode = manager.deserialize(texDef);\n textureNode._external = true;\n // If this material is already initialized, make sure the new texture is also initialized.\n if (this._initialized) {\n textureNode.init(this);\n }\n currentTextures.splice(indexOnParent, 0, textureNode);\n }\n });\n }\n else {\n this[key] =\n parentMat[key] instanceof exports.Kb3dObject ? parentMat[key] : manager.deserialize(parentMat[key]);\n }\n }\n }\n }\n refreshParentMaterial() {\n if (this.parentMaterialId) {\n this._materialService.getMaterialById(this.parentMaterialId).subscribe(parentMaterial => {\n if (parentMaterial && parentMaterial.data) {\n this.inheritFromMaterial(parentMaterial.data);\n }\n });\n }\n }\n getMediaReferences() {\n const refs = new Array();\n for (const texture of this.textures) {\n if (instanceOf(texture, Token.FileTextureLayer) && texture.path && !isAbsoluteUrl(texture.path)) {\n refs.push({\n type: 'Media',\n id: 'media/' + texture.path,\n });\n }\n }\n return refs;\n }\n };\n __decorate([\n Enumerable()\n ], exports.MaterialNode.prototype, \"materials\", null);\n __decorate([\n IgnoreChanges\n ], exports.MaterialNode.prototype, \"_initialized\", void 0);\n __decorate([\n IgnoreChanges\n ], exports.MaterialNode.prototype, \"location\", void 0);\n __decorate([\n IgnoreChanges\n ], exports.MaterialNode.prototype, \"parentMaterialId\", void 0);\n __decorate([\n Enumerable()\n ], exports.MaterialNode.prototype, \"overriddenProperties\", null);\n __decorate([\n Enumerable()\n ], exports.MaterialNode.prototype, \"textures\", null);\n __decorate([\n IgnoreChanges\n ], exports.MaterialNode.prototype, \"legacyTextureStacking\", void 0);\n exports.MaterialNode = __decorate([\n BaseKb3dModel({\n token: Token.MaterialNode,\n })\n ], exports.MaterialNode);\n function isMaterialNode(node) {\n return instanceOfAny(node, ...Token.MaterialTokens);\n }\n\n function __decorate$1(decorators, target, key, desc) {\n var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;\n if (typeof Reflect === \"object\" && typeof Reflect.decorate === \"function\") r = Reflect.decorate(decorators, target, key, desc);\n else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;\n return c > 3 && r && Object.defineProperty(target, key, r), r;\n }\n\n /**\n * Class used to evaluate queries containing `and` and `or` operators\n */\n class AndOrNotEvaluator {\n /**\n * Evaluate a query\n * @param query defines the query to evaluate\n * @param evaluateCallback defines the callback used to filter result\n * @returns true if the query matches\n */\n static Eval(query, evaluateCallback) {\n if (!query.match(/\\([^()]*\\)/g)) {\n query = AndOrNotEvaluator._HandleParenthesisContent(query, evaluateCallback);\n }\n else {\n query = query.replace(/\\([^()]*\\)/g, (r) => {\n // remove parenthesis\n r = r.slice(1, r.length - 1);\n return AndOrNotEvaluator._HandleParenthesisContent(r, evaluateCallback);\n });\n }\n if (query === \"true\") {\n return true;\n }\n if (query === \"false\") {\n return false;\n }\n return AndOrNotEvaluator.Eval(query, evaluateCallback);\n }\n static _HandleParenthesisContent(parenthesisContent, evaluateCallback) {\n evaluateCallback =\n evaluateCallback ||\n ((r) => {\n return r === \"true\" ? true : false;\n });\n let result;\n const or = parenthesisContent.split(\"||\");\n for (const i in or) {\n if (Object.prototype.hasOwnProperty.call(or, i)) {\n let ori = AndOrNotEvaluator._SimplifyNegation(or[i].trim());\n const and = ori.split(\"&&\");\n if (and.length > 1) {\n for (let j = 0; j < and.length; ++j) {\n const andj = AndOrNotEvaluator._SimplifyNegation(and[j].trim());\n if (andj !== \"true\" && andj !== \"false\") {\n if (andj[0] === \"!\") {\n result = !evaluateCallback(andj.substring(1));\n }\n else {\n result = evaluateCallback(andj);\n }\n }\n else {\n result = andj === \"true\" ? true : false;\n }\n if (!result) {\n // no need to continue since 'false && ... && ...' will always return false\n ori = \"false\";\n break;\n }\n }\n }\n if (result || ori === \"true\") {\n // no need to continue since 'true || ... || ...' will always return true\n result = true;\n break;\n }\n // result equals false (or undefined)\n if (ori !== \"true\" && ori !== \"false\") {\n if (ori[0] === \"!\") {\n result = !evaluateCallback(ori.substring(1));\n }\n else {\n result = evaluateCallback(ori);\n }\n }\n else {\n result = ori === \"true\" ? true : false;\n }\n }\n }\n // the whole parenthesis scope is replaced by 'true' or 'false'\n return result ? \"true\" : \"false\";\n }\n static _SimplifyNegation(booleanString) {\n booleanString = booleanString.replace(/^[\\s!]+/, (r) => {\n // remove whitespaces\n r = r.replace(/[\\s]/g, () => \"\");\n return r.length % 2 ? \"!\" : \"\";\n });\n booleanString = booleanString.trim();\n if (booleanString === \"!true\") {\n booleanString = \"false\";\n }\n else if (booleanString === \"!false\") {\n booleanString = \"true\";\n }\n return booleanString;\n }\n }\n\n /**\n * Class used to store custom tags\n */\n class Tags {\n /**\n * Adds support for tags on the given object\n * @param obj defines the object to use\n */\n static EnableFor(obj) {\n obj._tags = obj._tags || {};\n obj.hasTags = () => {\n return Tags.HasTags(obj);\n };\n obj.addTags = (tagsString) => {\n return Tags.AddTagsTo(obj, tagsString);\n };\n obj.removeTags = (tagsString) => {\n return Tags.RemoveTagsFrom(obj, tagsString);\n };\n obj.matchesTagsQuery = (tagsQuery) => {\n return Tags.MatchesQuery(obj, tagsQuery);\n };\n }\n /**\n * Removes tags support\n * @param obj defines the object to use\n */\n static DisableFor(obj) {\n delete obj._tags;\n delete obj.hasTags;\n delete obj.addTags;\n delete obj.removeTags;\n delete obj.matchesTagsQuery;\n }\n /**\n * Gets a boolean indicating if the given object has tags\n * @param obj defines the object to use\n * @returns a boolean\n */\n static HasTags(obj) {\n if (!obj._tags) {\n return false;\n }\n const tags = obj._tags;\n for (const i in tags) {\n if (Object.prototype.hasOwnProperty.call(tags, i)) {\n return true;\n }\n }\n return false;\n }\n /**\n * Gets the tags available on a given object\n * @param obj defines the object to use\n * @param asString defines if the tags must be returned as a string instead of an array of strings\n * @returns the tags\n */\n static GetTags(obj, asString = true) {\n if (!obj._tags) {\n return null;\n }\n if (asString) {\n const tagsArray = [];\n for (const tag in obj._tags) {\n if (Object.prototype.hasOwnProperty.call(obj._tags, tag) && obj._tags[tag] === true) {\n tagsArray.push(tag);\n }\n }\n return tagsArray.join(\" \");\n }\n else {\n return obj._tags;\n }\n }\n /**\n * Adds tags to an object\n * @param obj defines the object to use\n * @param tagsString defines the tag string. The tags 'true' and 'false' are reserved and cannot be used as tags.\n * A tag cannot start with '||', '&&', and '!'. It cannot contain whitespaces\n */\n static AddTagsTo(obj, tagsString) {\n if (!tagsString) {\n return;\n }\n if (typeof tagsString !== \"string\") {\n return;\n }\n const tags = tagsString.split(\" \");\n tags.forEach(function (tag) {\n Tags._AddTagTo(obj, tag);\n });\n }\n /**\n * @internal\n */\n static _AddTagTo(obj, tag) {\n tag = tag.trim();\n if (tag === \"\" || tag === \"true\" || tag === \"false\") {\n return;\n }\n if (tag.match(/[\\s]/) || tag.match(/^([!]|([|]|[&]){2})/)) {\n return;\n }\n Tags.EnableFor(obj);\n obj._tags[tag] = true;\n }\n /**\n * Removes specific tags from a specific object\n * @param obj defines the object to use\n * @param tagsString defines the tags to remove\n */\n static RemoveTagsFrom(obj, tagsString) {\n if (!Tags.HasTags(obj)) {\n return;\n }\n const tags = tagsString.split(\" \");\n for (const t in tags) {\n Tags._RemoveTagFrom(obj, tags[t]);\n }\n }\n /**\n * @internal\n */\n static _RemoveTagFrom(obj, tag) {\n delete obj._tags[tag];\n }\n /**\n * Defines if tags hosted on an object match a given query\n * @param obj defines the object to use\n * @param tagsQuery defines the tag query\n * @returns a boolean\n */\n static MatchesQuery(obj, tagsQuery) {\n if (tagsQuery === undefined) {\n return true;\n }\n if (tagsQuery === \"\") {\n return Tags.HasTags(obj);\n }\n return AndOrNotEvaluator.Eval(tagsQuery, (r) => Tags.HasTags(obj) && obj._tags[r]);\n }\n }\n\n /**\n * Scalar computation library\n */\n class Scalar {\n /**\n * Boolean : true if the absolute difference between a and b is lower than epsilon (default = 1.401298E-45)\n * @param a number\n * @param b number\n * @param epsilon (default = 1.401298E-45)\n * @returns true if the absolute difference between a and b is lower than epsilon (default = 1.401298E-45)\n */\n static WithinEpsilon(a, b, epsilon = 1.401298e-45) {\n return Math.abs(a - b) <= epsilon;\n }\n /**\n * Returns a string : the upper case translation of the number i to hexadecimal.\n * @param i number\n * @returns the upper case translation of the number i to hexadecimal.\n */\n static ToHex(i) {\n const str = i.toString(16);\n if (i <= 15) {\n return (\"0\" + str).toUpperCase();\n }\n return str.toUpperCase();\n }\n /**\n * Returns -1 if value is negative and +1 is value is positive.\n * @param value the value\n * @returns the value itself if it's equal to zero.\n */\n static Sign(value) {\n value = +value; // convert to a number\n if (value === 0 || isNaN(value)) {\n return value;\n }\n return value > 0 ? 1 : -1;\n }\n /**\n * Returns the value itself if it's between min and max.\n * Returns min if the value is lower than min.\n * Returns max if the value is greater than max.\n * @param value the value to clmap\n * @param min the min value to clamp to (default: 0)\n * @param max the max value to clamp to (default: 1)\n * @returns the clamped value\n */\n static Clamp(value, min = 0, max = 1) {\n return Math.min(max, Math.max(min, value));\n }\n /**\n * the log2 of value.\n * @param value the value to compute log2 of\n * @returns the log2 of value.\n */\n static Log2(value) {\n return Math.log(value) * Math.LOG2E;\n }\n /**\n * the floor part of a log2 value.\n * @param value the value to compute log2 of\n * @returns the log2 of value.\n */\n static ILog2(value) {\n if (Math.log2) {\n return Math.floor(Math.log2(value));\n }\n if (value < 0) {\n return NaN;\n }\n else if (value === 0) {\n return -Infinity;\n }\n let n = 0;\n if (value < 1) {\n while (value < 1) {\n n++;\n value = value * 2;\n }\n n = -n;\n }\n else if (value > 1) {\n while (value > 1) {\n n++;\n value = Math.floor(value / 2);\n }\n }\n return n;\n }\n /**\n * Loops the value, so that it is never larger than length and never smaller than 0.\n *\n * This is similar to the modulo operator but it works with floating point numbers.\n * For example, using 3.0 for t and 2.5 for length, the result would be 0.5.\n * With t = 5 and length = 2.5, the result would be 0.0.\n * Note, however, that the behaviour is not defined for negative numbers as it is for the modulo operator\n * @param value the value\n * @param length the length\n * @returns the looped value\n */\n static Repeat(value, length) {\n return value - Math.floor(value / length) * length;\n }\n /**\n * Normalize the value between 0.0 and 1.0 using min and max values\n * @param value value to normalize\n * @param min max to normalize between\n * @param max min to normalize between\n * @returns the normalized value\n */\n static Normalize(value, min, max) {\n return (value - min) / (max - min);\n }\n /**\n * Denormalize the value from 0.0 and 1.0 using min and max values\n * @param normalized value to denormalize\n * @param min max to denormalize between\n * @param max min to denormalize between\n * @returns the denormalized value\n */\n static Denormalize(normalized, min, max) {\n return normalized * (max - min) + min;\n }\n /**\n * Calculates the shortest difference between two given angles given in degrees.\n * @param current current angle in degrees\n * @param target target angle in degrees\n * @returns the delta\n */\n static DeltaAngle(current, target) {\n let num = Scalar.Repeat(target - current, 360.0);\n if (num > 180.0) {\n num -= 360.0;\n }\n return num;\n }\n /**\n * PingPongs the value t, so that it is never larger than length and never smaller than 0.\n * @param tx value\n * @param length length\n * @returns The returned value will move back and forth between 0 and length\n */\n static PingPong(tx, length) {\n const t = Scalar.Repeat(tx, length * 2.0);\n return length - Math.abs(t - length);\n }\n /**\n * Interpolates between min and max with smoothing at the limits.\n *\n * This function interpolates between min and max in a similar way to Lerp. However, the interpolation will gradually speed up\n * from the start and slow down toward the end. This is useful for creating natural-looking animation, fading and other transitions.\n * @param from from\n * @param to to\n * @param tx value\n * @returns the smooth stepped value\n */\n static SmoothStep(from, to, tx) {\n let t = Scalar.Clamp(tx);\n t = -2.0 * t * t * t + 3.0 * t * t;\n return to * t + from * (1.0 - t);\n }\n /**\n * Moves a value current towards target.\n *\n * This is essentially the same as Mathf.Lerp but instead the function will ensure that the speed never exceeds maxDelta.\n * Negative values of maxDelta pushes the value away from target.\n * @param current current value\n * @param target target value\n * @param maxDelta max distance to move\n * @returns resulting value\n */\n static MoveTowards(current, target, maxDelta) {\n let result = 0;\n if (Math.abs(target - current) <= maxDelta) {\n result = target;\n }\n else {\n result = current + Scalar.Sign(target - current) * maxDelta;\n }\n return result;\n }\n /**\n * Same as MoveTowards but makes sure the values interpolate correctly when they wrap around 360 degrees.\n *\n * Variables current and target are assumed to be in degrees. For optimization reasons, negative values of maxDelta\n * are not supported and may cause oscillation. To push current away from a target angle, add 180 to that angle instead.\n * @param current current value\n * @param target target value\n * @param maxDelta max distance to move\n * @returns resulting angle\n */\n static MoveTowardsAngle(current, target, maxDelta) {\n const num = Scalar.DeltaAngle(current, target);\n let result = 0;\n if (-maxDelta < num && num < maxDelta) {\n result = target;\n }\n else {\n target = current + num;\n result = Scalar.MoveTowards(current, target, maxDelta);\n }\n return result;\n }\n /**\n * Creates a new scalar with values linearly interpolated of \"amount\" between the start scalar and the end scalar.\n * @param start start value\n * @param end target value\n * @param amount amount to lerp between\n * @returns the lerped value\n */\n static Lerp(start, end, amount) {\n return start + (end - start) * amount;\n }\n /**\n * Same as Lerp but makes sure the values interpolate correctly when they wrap around 360 degrees.\n * The parameter t is clamped to the range [0, 1]. Variables a and b are assumed to be in degrees.\n * @param start start value\n * @param end target value\n * @param amount amount to lerp between\n * @returns the lerped value\n */\n static LerpAngle(start, end, amount) {\n let num = Scalar.Repeat(end - start, 360.0);\n if (num > 180.0) {\n num -= 360.0;\n }\n return start + num * Scalar.Clamp(amount);\n }\n /**\n * Calculates the linear parameter t that produces the interpolant value within the range [a, b].\n * @param a start value\n * @param b target value\n * @param value value between a and b\n * @returns the inverseLerp value\n */\n static InverseLerp(a, b, value) {\n let result = 0;\n if (a != b) {\n result = Scalar.Clamp((value - a) / (b - a));\n }\n else {\n result = 0.0;\n }\n return result;\n }\n /**\n * Returns a new scalar located for \"amount\" (float) on the Hermite spline defined by the scalars \"value1\", \"value3\", \"tangent1\", \"tangent2\".\n * @see http://mathworld.wolfram.com/HermitePolynomial.html\n * @param value1 defines the first control point\n * @param tangent1 defines the first tangent\n * @param value2 defines the second control point\n * @param tangent2 defines the second tangent\n * @param amount defines the amount on the interpolation spline (between 0 and 1)\n * @returns hermite result\n */\n static Hermite(value1, tangent1, value2, tangent2, amount) {\n const squared = amount * amount;\n const cubed = amount * squared;\n const part1 = 2.0 * cubed - 3.0 * squared + 1.0;\n const part2 = -2.0 * cubed + 3.0 * squared;\n const part3 = cubed - 2.0 * squared + amount;\n const part4 = cubed - squared;\n return value1 * part1 + value2 * part2 + tangent1 * part3 + tangent2 * part4;\n }\n /**\n * Returns a new scalar which is the 1st derivative of the Hermite spline defined by the scalars \"value1\", \"value2\", \"tangent1\", \"tangent2\".\n * @param value1 defines the first control point\n * @param tangent1 defines the first tangent\n * @param value2 defines the second control point\n * @param tangent2 defines the second tangent\n * @param time define where the derivative must be done\n * @returns 1st derivative\n */\n static Hermite1stDerivative(value1, tangent1, value2, tangent2, time) {\n const t2 = time * time;\n return (t2 - time) * 6 * value1 + (3 * t2 - 4 * time + 1) * tangent1 + (-t2 + time) * 6 * value2 + (3 * t2 - 2 * time) * tangent2;\n }\n /**\n * Returns a random float number between and min and max values\n * @param min min value of random\n * @param max max value of random\n * @returns random value\n */\n static RandomRange(min, max) {\n if (min === max) {\n return min;\n }\n return Math.random() * (max - min) + min;\n }\n /**\n * This function returns percentage of a number in a given range.\n *\n * RangeToPercent(40,20,60) will return 0.5 (50%)\n * RangeToPercent(34,0,100) will return 0.34 (34%)\n * @param number to convert to percentage\n * @param min min range\n * @param max max range\n * @returns the percentage\n */\n static RangeToPercent(number, min, max) {\n return (number - min) / (max - min);\n }\n /**\n * This function returns number that corresponds to the percentage in a given range.\n *\n * PercentToRange(0.34,0,100) will return 34.\n * @param percent to convert to number\n * @param min min range\n * @param max max range\n * @returns the number\n */\n static PercentToRange(percent, min, max) {\n return (max - min) * percent + min;\n }\n /**\n * Returns the angle converted to equivalent value between -Math.PI and Math.PI radians.\n * @param angle The angle to normalize in radian.\n * @returns The converted angle.\n */\n static NormalizeRadians(angle) {\n // More precise but slower version kept for reference.\n // angle = angle % Tools.TwoPi;\n // angle = (angle + Tools.TwoPi) % Tools.TwoPi;\n //if (angle > Math.PI) {\n //\tangle -= Tools.TwoPi;\n //}\n angle -= Scalar.TwoPi * Math.floor((angle + Math.PI) / Scalar.TwoPi);\n return angle;\n }\n /**\n * Returns the highest common factor of two integers.\n * @param a first parameter\n * @param b second parameter\n * @returns HCF of a and b\n */\n static HCF(a, b) {\n const r = a % b;\n if (r === 0) {\n return b;\n }\n return Scalar.HCF(b, r);\n }\n }\n /**\n * Two pi constants convenient for computation.\n */\n Scalar.TwoPi = Math.PI * 2;\n\n /**\n * Constant used to convert a value to gamma space\n * @ignorenaming\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n const ToGammaSpace = 1 / 2.2;\n /**\n * Constant used to convert a value to linear space\n * @ignorenaming\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n const ToLinearSpace = 2.2;\n /**\n * Constant Golden Ratio value in Babylon.js\n * @ignorenaming\n */\n const PHI = (1 + Math.sqrt(5)) / 2;\n /**\n * Constant used to define the minimal number value in Babylon.js\n * @ignorenaming\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n const Epsilon = 0.001;\n\n /* eslint-disable @typescript-eslint/naming-convention */\n /**\n * Class containing a set of static utilities functions for arrays.\n */\n class ArrayTools {\n /**\n * Returns an array of the given size filled with elements built from the given constructor and the parameters.\n * @param size the number of element to construct and put in the array.\n * @param itemBuilder a callback responsible for creating new instance of item. Called once per array entry.\n * @returns a new array filled with new objects.\n */\n static BuildArray(size, itemBuilder) {\n const a = [];\n for (let i = 0; i < size; ++i) {\n a.push(itemBuilder());\n }\n return a;\n }\n /**\n * Returns a tuple of the given size filled with elements built from the given constructor and the parameters.\n * @param size he number of element to construct and put in the tuple.\n * @param itemBuilder a callback responsible for creating new instance of item. Called once per tuple entry.\n * @returns a new tuple filled with new objects.\n */\n static BuildTuple(size, itemBuilder) {\n return ArrayTools.BuildArray(size, itemBuilder);\n }\n }\n /**\n * Observes a function and calls the given callback when it is called.\n * @param object Defines the object the function to observe belongs to.\n * @param functionName Defines the name of the function to observe.\n * @param callback Defines the callback to call when the function is called.\n * @returns A function to call to stop observing\n */\n function _observeArrayfunction(object, functionName, callback) {\n // Finds the function to observe\n const oldFunction = object[functionName];\n if (typeof oldFunction !== \"function\") {\n return null;\n }\n // Creates a new function that calls the callback and the old function\n const newFunction = function () {\n const previousLength = object.length;\n const returnValue = newFunction.previous.apply(object, arguments);\n callback(functionName, previousLength);\n return returnValue;\n };\n // Doublishly links the new function and the old function\n oldFunction.next = newFunction;\n newFunction.previous = oldFunction;\n // Replaces the old function with the new function\n object[functionName] = newFunction;\n // Returns a function to disable the hook\n return () => {\n // Only unhook if the function is still hooked\n const previous = newFunction.previous;\n if (!previous) {\n return;\n }\n // Finds the ref to the next function in the chain\n const next = newFunction.next;\n // If in the middle of the chain, link the previous and next functions\n if (next) {\n previous.next = next;\n next.previous = previous;\n }\n // If at the end of the chain, remove the reference to the previous function\n // and restore the previous function\n else {\n previous.next = undefined;\n object[functionName] = previous;\n }\n // Lose reference to the previous and next functions\n newFunction.next = undefined;\n newFunction.previous = undefined;\n };\n }\n /**\n * Defines the list of functions to proxy when observing an array.\n * The scope is currently reduced to the common functions used in the render target render list and the scene cameras.\n */\n const observedArrayFunctions = [\"push\", \"splice\", \"pop\", \"shift\", \"unshift\"];\n /**\n * Observes an array and notifies the given observer when the array is modified.\n * @param array Defines the array to observe\n * @param callback Defines the function to call when the array is modified (in the limit of the observed array functions)\n * @returns A function to call to stop observing the array\n * @internal\n */\n function _ObserveArray(array, callback) {\n // Observes all the required array functions and stores the unhook functions\n const unObserveFunctions = observedArrayFunctions.map((name) => {\n return _observeArrayfunction(array, name, callback);\n });\n // Returns a function that unhook all the observed functions\n return () => {\n unObserveFunctions.forEach((unObserveFunction) => {\n unObserveFunction === null || unObserveFunction === void 0 ? void 0 : unObserveFunction();\n });\n };\n }\n\n /** @internal */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n const _RegisteredTypes = {};\n /**\n * @internal\n */\n function RegisterClass(className, type) {\n _RegisteredTypes[className] = type;\n }\n /**\n * @internal\n */\n function GetClass(fqdn) {\n return _RegisteredTypes[fqdn];\n }\n\n /** @internal */\n class PerformanceConfigurator {\n /**\n * @internal\n */\n static SetMatrixPrecision(use64bits) {\n PerformanceConfigurator.MatrixTrackPrecisionChange = false;\n if (use64bits && !PerformanceConfigurator.MatrixUse64Bits) {\n if (PerformanceConfigurator.MatrixTrackedMatrices) {\n for (let m = 0; m < PerformanceConfigurator.MatrixTrackedMatrices.length; ++m) {\n const matrix = PerformanceConfigurator.MatrixTrackedMatrices[m];\n const values = matrix._m;\n matrix._m = new Array(16);\n for (let i = 0; i < 16; ++i) {\n matrix._m[i] = values[i];\n }\n }\n }\n }\n PerformanceConfigurator.MatrixUse64Bits = use64bits;\n PerformanceConfigurator.MatrixCurrentType = PerformanceConfigurator.MatrixUse64Bits ? Array : Float32Array;\n PerformanceConfigurator.MatrixTrackedMatrices = null; // reclaim some memory, as we don't need _TrackedMatrices anymore\n }\n }\n /** @internal */\n PerformanceConfigurator.MatrixUse64Bits = false;\n /** @internal */\n PerformanceConfigurator.MatrixTrackPrecisionChange = true;\n /** @internal */\n PerformanceConfigurator.MatrixCurrentType = Float32Array;\n /** @internal */\n PerformanceConfigurator.MatrixTrackedMatrices = [];\n\n /**\n * A class serves as a medium between the observable and its observers\n */\n class EventState {\n /**\n * Create a new EventState\n * @param mask defines the mask associated with this state\n * @param skipNextObservers defines a flag which will instruct the observable to skip following observers when set to true\n * @param target defines the original target of the state\n * @param currentTarget defines the current target of the state\n */\n constructor(mask, skipNextObservers = false, target, currentTarget) {\n this.initialize(mask, skipNextObservers, target, currentTarget);\n }\n /**\n * Initialize the current event state\n * @param mask defines the mask associated with this state\n * @param skipNextObservers defines a flag which will instruct the observable to skip following observers when set to true\n * @param target defines the original target of the state\n * @param currentTarget defines the current target of the state\n * @returns the current event state\n */\n initialize(mask, skipNextObservers = false, target, currentTarget) {\n this.mask = mask;\n this.skipNextObservers = skipNextObservers;\n this.target = target;\n this.currentTarget = currentTarget;\n return this;\n }\n }\n /**\n * Represent an Observer registered to a given Observable object.\n */\n class Observer {\n /**\n * Creates a new observer\n * @param callback defines the callback to call when the observer is notified\n * @param mask defines the mask of the observer (used to filter notifications)\n * @param scope defines the current scope used to restore the JS context\n */\n constructor(\n /**\n * Defines the callback to call when the observer is notified\n */\n callback, \n /**\n * Defines the mask of the observer (used to filter notifications)\n */\n mask, \n /**\n * Defines the current scope used to restore the JS context\n */\n scope = null) {\n this.callback = callback;\n this.mask = mask;\n this.scope = scope;\n /** @internal */\n this._willBeUnregistered = false;\n /**\n * Gets or sets a property defining that the observer as to be unregistered after the next notification\n */\n this.unregisterOnNextCall = false;\n /**\n * this function can be used to remove the observer from the observable.\n * It will be set by the observable that the observer belongs to.\n * @internal\n */\n this._remove = null;\n }\n /**\n * Remove the observer from its observable\n * This can be used instead of using the observable's remove function.\n */\n remove() {\n if (this._remove) {\n this._remove();\n }\n }\n }\n /**\n * The Observable class is a simple implementation of the Observable pattern.\n *\n * There's one slight particularity though: a given Observable can notify its observer using a particular mask value, only the Observers registered with this mask value will be notified.\n * This enable a more fine grained execution without having to rely on multiple different Observable objects.\n * For instance you may have a given Observable that have four different types of notifications: Move (mask = 0x01), Stop (mask = 0x02), Turn Right (mask = 0X04), Turn Left (mask = 0X08).\n * A given observer can register itself with only Move and Stop (mask = 0x03), then it will only be notified when one of these two occurs and will never be for Turn Left/Right.\n */\n class Observable$1 {\n /**\n * Create an observable from a Promise.\n * @param promise a promise to observe for fulfillment.\n * @param onErrorObservable an observable to notify if a promise was rejected.\n * @returns the new Observable\n */\n static FromPromise(promise, onErrorObservable) {\n const observable = new Observable$1();\n promise\n .then((ret) => {\n observable.notifyObservers(ret);\n })\n .catch((err) => {\n if (onErrorObservable) {\n onErrorObservable.notifyObservers(err);\n }\n else {\n throw err;\n }\n });\n return observable;\n }\n /**\n * Gets the list of observers\n * Note that observers that were recently deleted may still be present in the list because they are only really deleted on the next javascript tick!\n */\n get observers() {\n return this._observers;\n }\n /**\n * Creates a new observable\n * @param onObserverAdded defines a callback to call when a new observer is added\n * @param notifyIfTriggered If set to true the observable will notify when an observer was added if the observable was already triggered.\n */\n constructor(onObserverAdded, \n /**\n * If set to true the observable will notify when an observer was added if the observable was already triggered.\n * This is helpful to single-state observables like the scene onReady or the dispose observable.\n */\n notifyIfTriggered = false) {\n this.notifyIfTriggered = notifyIfTriggered;\n this._observers = new Array();\n this._numObserversMarkedAsDeleted = 0;\n this._hasNotified = false;\n this._eventState = new EventState(0);\n if (onObserverAdded) {\n this._onObserverAdded = onObserverAdded;\n }\n }\n /**\n * Create a new Observer with the specified callback\n * @param callback the callback that will be executed for that Observer\n * @param mask the mask used to filter observers\n * @param insertFirst if true the callback will be inserted at the first position, hence executed before the others ones. If false (default behavior) the callback will be inserted at the last position, executed after all the others already present.\n * @param scope optional scope for the callback to be called from\n * @param unregisterOnFirstCall defines if the observer as to be unregistered after the next notification\n * @returns the new observer created for the callback\n */\n add(callback, mask = -1, insertFirst = false, scope = null, unregisterOnFirstCall = false) {\n if (!callback) {\n return null;\n }\n const observer = new Observer(callback, mask, scope);\n observer.unregisterOnNextCall = unregisterOnFirstCall;\n if (insertFirst) {\n this._observers.unshift(observer);\n }\n else {\n this._observers.push(observer);\n }\n if (this._onObserverAdded) {\n this._onObserverAdded(observer);\n }\n // If the observable was already triggered and the observable is set to notify if triggered, notify the new observer\n if (this._hasNotified && this.notifyIfTriggered) {\n if (this._lastNotifiedValue !== undefined) {\n this.notifyObserver(observer, this._lastNotifiedValue);\n }\n }\n // attach the remove function to the observer\n observer._remove = () => {\n this.remove(observer);\n };\n return observer;\n }\n /**\n * Create a new Observer with the specified callback and unregisters after the next notification\n * @param callback the callback that will be executed for that Observer\n * @returns the new observer created for the callback\n */\n addOnce(callback) {\n return this.add(callback, undefined, undefined, undefined, true);\n }\n /**\n * Remove an Observer from the Observable object\n * @param observer the instance of the Observer to remove\n * @returns false if it doesn't belong to this Observable\n */\n remove(observer) {\n if (!observer) {\n return false;\n }\n observer._remove = null;\n const index = this._observers.indexOf(observer);\n if (index !== -1) {\n this._deferUnregister(observer);\n return true;\n }\n return false;\n }\n /**\n * Remove a callback from the Observable object\n * @param callback the callback to remove\n * @param scope optional scope. If used only the callbacks with this scope will be removed\n * @returns false if it doesn't belong to this Observable\n */\n removeCallback(callback, scope) {\n for (let index = 0; index < this._observers.length; index++) {\n const observer = this._observers[index];\n if (observer._willBeUnregistered) {\n continue;\n }\n if (observer.callback === callback && (!scope || scope === observer.scope)) {\n this._deferUnregister(observer);\n return true;\n }\n }\n return false;\n }\n /**\n * @internal\n */\n _deferUnregister(observer) {\n if (observer._willBeUnregistered) {\n return;\n }\n this._numObserversMarkedAsDeleted++;\n observer.unregisterOnNextCall = false;\n observer._willBeUnregistered = true;\n setTimeout(() => {\n this._remove(observer);\n }, 0);\n }\n // This should only be called when not iterating over _observers to avoid callback skipping.\n // Removes an observer from the _observer Array.\n _remove(observer, updateCounter = true) {\n if (!observer) {\n return false;\n }\n const index = this._observers.indexOf(observer);\n if (index !== -1) {\n if (updateCounter) {\n this._numObserversMarkedAsDeleted--;\n }\n this._observers.splice(index, 1);\n return true;\n }\n return false;\n }\n /**\n * Moves the observable to the top of the observer list making it get called first when notified\n * @param observer the observer to move\n */\n makeObserverTopPriority(observer) {\n this._remove(observer, false);\n this._observers.unshift(observer);\n }\n /**\n * Moves the observable to the bottom of the observer list making it get called last when notified\n * @param observer the observer to move\n */\n makeObserverBottomPriority(observer) {\n this._remove(observer, false);\n this._observers.push(observer);\n }\n /**\n * Notify all Observers by calling their respective callback with the given data\n * Will return true if all observers were executed, false if an observer set skipNextObservers to true, then prevent the subsequent ones to execute\n * @param eventData defines the data to send to all observers\n * @param mask defines the mask of the current notification (observers with incompatible mask (ie mask & observer.mask === 0) will not be notified)\n * @param target defines the original target of the state\n * @param currentTarget defines the current target of the state\n * @param userInfo defines any user info to send to observers\n * @returns false if the complete observer chain was not processed (because one observer set the skipNextObservers to true)\n */\n notifyObservers(eventData, mask = -1, target, currentTarget, userInfo) {\n // this prevents potential memory leaks - if an object is disposed but the observable doesn't get cleared.\n if (this.notifyIfTriggered) {\n this._hasNotified = true;\n this._lastNotifiedValue = eventData;\n }\n if (!this._observers.length) {\n return true;\n }\n const state = this._eventState;\n state.mask = mask;\n state.target = target;\n state.currentTarget = currentTarget;\n state.skipNextObservers = false;\n state.lastReturnValue = eventData;\n state.userInfo = userInfo;\n for (const obs of this._observers) {\n if (obs._willBeUnregistered) {\n continue;\n }\n if (obs.mask & mask) {\n if (obs.unregisterOnNextCall) {\n this._deferUnregister(obs);\n }\n if (obs.scope) {\n state.lastReturnValue = obs.callback.apply(obs.scope, [eventData, state]);\n }\n else {\n state.lastReturnValue = obs.callback(eventData, state);\n }\n }\n if (state.skipNextObservers) {\n return false;\n }\n }\n return true;\n }\n /**\n * Notify a specific observer\n * @param observer defines the observer to notify\n * @param eventData defines the data to be sent to each callback\n * @param mask is used to filter observers defaults to -1\n */\n notifyObserver(observer, eventData, mask = -1) {\n // this prevents potential memory leaks - if an object is disposed but the observable doesn't get cleared.\n if (this.notifyIfTriggered) {\n this._hasNotified = true;\n this._lastNotifiedValue = eventData;\n }\n if (observer._willBeUnregistered) {\n return;\n }\n const state = this._eventState;\n state.mask = mask;\n state.skipNextObservers = false;\n if (observer.unregisterOnNextCall) {\n this._deferUnregister(observer);\n }\n observer.callback(eventData, state);\n }\n /**\n * Gets a boolean indicating if the observable has at least one observer\n * @returns true is the Observable has at least one Observer registered\n */\n hasObservers() {\n return this._observers.length - this._numObserversMarkedAsDeleted > 0;\n }\n /**\n * Clear the list of observers\n */\n clear() {\n while (this._observers.length) {\n const o = this._observers.pop();\n if (o) {\n o._remove = null;\n }\n }\n this._onObserverAdded = null;\n this._numObserversMarkedAsDeleted = 0;\n this.cleanLastNotifiedState();\n }\n /**\n * Clean the last notified state - both the internal last value and the has-notified flag\n */\n cleanLastNotifiedState() {\n this._hasNotified = false;\n this._lastNotifiedValue = undefined;\n }\n /**\n * Clone the current observable\n * @returns a new observable\n */\n clone() {\n const result = new Observable$1();\n result._observers = this._observers.slice(0);\n return result;\n }\n /**\n * Does this observable handles observer registered with a given mask\n * @param mask defines the mask to be tested\n * @returns whether or not one observer registered with the given mask is handled\n **/\n hasSpecificMask(mask = -1) {\n for (const obs of this._observers) {\n if (obs.mask & mask || obs.mask === mask) {\n return true;\n }\n }\n return false;\n }\n }\n\n /**\n * The engine store class is responsible to hold all the instances of Engine and Scene created\n * during the life time of the application.\n */\n class EngineStore {\n /**\n * Gets the latest created engine\n */\n static get LastCreatedEngine() {\n if (this.Instances.length === 0) {\n return null;\n }\n return this.Instances[this.Instances.length - 1];\n }\n /**\n * Gets the latest created scene\n */\n static get LastCreatedScene() {\n return this._LastCreatedScene;\n }\n }\n /** Gets the list of created engines */\n EngineStore.Instances = new Array();\n /**\n * Notifies when an engine was disposed.\n * Mainly used for static/cache cleanup\n */\n EngineStore.OnEnginesDisposedObservable = new Observable$1();\n /** @internal */\n EngineStore._LastCreatedScene = null;\n /**\n * Gets or sets a global variable indicating if fallback texture must be used when a texture cannot be loaded\n * @ignorenaming\n */\n EngineStore.UseFallbackTexture = true;\n /**\n * Texture content used if a texture cannot loaded\n * @ignorenaming\n */\n EngineStore.FallbackTexture = \"\";\n\n /* eslint-disable @typescript-eslint/naming-convention */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n const _ExtractAsInt = (value) => {\n return parseInt(value.toString().replace(/\\W/g, \"\"));\n };\n /**\n * Class representing a vector containing 2 coordinates\n * Example Playground - Overview - https://playground.babylonjs.com/#QYBWV4#9\n */\n class Vector2 {\n /**\n * Creates a new Vector2 from the given x and y coordinates\n * @param x defines the first coordinate\n * @param y defines the second coordinate\n */\n constructor(\n /** defines the first coordinate */\n x = 0, \n /** defines the second coordinate */\n y = 0) {\n this.x = x;\n this.y = y;\n }\n /**\n * Gets a string with the Vector2 coordinates\n * @returns a string with the Vector2 coordinates\n */\n toString() {\n return `{X: ${this.x} Y: ${this.y}}`;\n }\n /**\n * Gets class name\n * @returns the string \"Vector2\"\n */\n getClassName() {\n return \"Vector2\";\n }\n /**\n * Gets current vector hash code\n * @returns the Vector2 hash code as a number\n */\n getHashCode() {\n const x = _ExtractAsInt(this.x);\n const y = _ExtractAsInt(this.y);\n let hash = x;\n hash = (hash * 397) ^ y;\n return hash;\n }\n // Operators\n /**\n * Sets the Vector2 coordinates in the given array or Float32Array from the given index.\n * Example Playground https://playground.babylonjs.com/#QYBWV4#15\n * @param array defines the source array\n * @param index defines the offset in source array\n * @returns the current Vector2\n */\n toArray(array, index = 0) {\n array[index] = this.x;\n array[index + 1] = this.y;\n return this;\n }\n /**\n * Update the current vector from an array\n * Example Playground https://playground.babylonjs.com/#QYBWV4#39\n * @param array defines the destination array\n * @param index defines the offset in the destination array\n * @returns the current Vector2\n */\n fromArray(array, index = 0) {\n Vector2.FromArrayToRef(array, index, this);\n return this;\n }\n /**\n * Copy the current vector to an array\n * Example Playground https://playground.babylonjs.com/#QYBWV4#40\n * @returns a new array with 2 elements: the Vector2 coordinates.\n */\n asArray() {\n const result = new Array();\n this.toArray(result, 0);\n return result;\n }\n /**\n * Sets the Vector2 coordinates with the given Vector2 coordinates\n * Example Playground https://playground.babylonjs.com/#QYBWV4#24\n * @param source defines the source Vector2\n * @returns the current updated Vector2\n */\n copyFrom(source) {\n this.x = source.x;\n this.y = source.y;\n return this;\n }\n /**\n * Sets the Vector2 coordinates with the given floats\n * Example Playground https://playground.babylonjs.com/#QYBWV4#25\n * @param x defines the first coordinate\n * @param y defines the second coordinate\n * @returns the current updated Vector2\n */\n copyFromFloats(x, y) {\n this.x = x;\n this.y = y;\n return this;\n }\n /**\n * Sets the Vector2 coordinates with the given floats\n * Example Playground https://playground.babylonjs.com/#QYBWV4#62\n * @param x defines the first coordinate\n * @param y defines the second coordinate\n * @returns the current updated Vector2\n */\n set(x, y) {\n return this.copyFromFloats(x, y);\n }\n /**\n * Add another vector with the current one\n * Example Playground https://playground.babylonjs.com/#QYBWV4#11\n * @param otherVector defines the other vector\n * @returns a new Vector2 set with the addition of the current Vector2 and the given one coordinates\n */\n add(otherVector) {\n return new this.constructor(this.x + otherVector.x, this.y + otherVector.y);\n }\n /**\n * Sets the \"result\" coordinates with the addition of the current Vector2 and the given one coordinates\n * Example Playground https://playground.babylonjs.com/#QYBWV4#12\n * @param otherVector defines the other vector\n * @param result defines the target vector\n * @returns result input\n */\n addToRef(otherVector, result) {\n result.x = this.x + otherVector.x;\n result.y = this.y + otherVector.y;\n return result;\n }\n /**\n * Set the Vector2 coordinates by adding the given Vector2 coordinates\n * Example Playground https://playground.babylonjs.com/#QYBWV4#13\n * @param otherVector defines the other vector\n * @returns the current updated Vector2\n */\n addInPlace(otherVector) {\n this.x += otherVector.x;\n this.y += otherVector.y;\n return this;\n }\n /**\n * Gets a new Vector2 by adding the current Vector2 coordinates to the given Vector3 x, y coordinates\n * Example Playground https://playground.babylonjs.com/#QYBWV4#14\n * @param otherVector defines the other vector\n * @returns a new Vector2\n */\n addVector3(otherVector) {\n return new this.constructor(this.x + otherVector.x, this.y + otherVector.y);\n }\n /**\n * Gets a new Vector2 set with the subtracted coordinates of the given one from the current Vector2\n * Example Playground https://playground.babylonjs.com/#QYBWV4#61\n * @param otherVector defines the other vector\n * @returns a new Vector2\n */\n subtract(otherVector) {\n return new this.constructor(this.x - otherVector.x, this.y - otherVector.y);\n }\n /**\n * Sets the \"result\" coordinates with the subtraction of the given one from the current Vector2 coordinates.\n * Example Playground https://playground.babylonjs.com/#QYBWV4#63\n * @param otherVector defines the other vector\n * @param result defines the target vector\n * @returns result input\n */\n subtractToRef(otherVector, result) {\n result.x = this.x - otherVector.x;\n result.y = this.y - otherVector.y;\n return result;\n }\n /**\n * Sets the current Vector2 coordinates by subtracting from it the given one coordinates\n * Example Playground https://playground.babylonjs.com/#QYBWV4#88\n * @param otherVector defines the other vector\n * @returns the current updated Vector2\n */\n subtractInPlace(otherVector) {\n this.x -= otherVector.x;\n this.y -= otherVector.y;\n return this;\n }\n /**\n * Multiplies in place the current Vector2 coordinates by the given ones\n * Example Playground https://playground.babylonjs.com/#QYBWV4#43\n * @param otherVector defines the other vector\n * @returns the current updated Vector2\n */\n multiplyInPlace(otherVector) {\n this.x *= otherVector.x;\n this.y *= otherVector.y;\n return this;\n }\n /**\n * Returns a new Vector2 set with the multiplication of the current Vector2 and the given one coordinates\n * Example Playground https://playground.babylonjs.com/#QYBWV4#42\n * @param otherVector defines the other vector\n * @returns a new Vector2\n */\n multiply(otherVector) {\n return new this.constructor(this.x * otherVector.x, this.y * otherVector.y);\n }\n /**\n * Sets \"result\" coordinates with the multiplication of the current Vector2 and the given one coordinates\n * Example Playground https://playground.babylonjs.com/#QYBWV4#44\n * @param otherVector defines the other vector\n * @param result defines the target vector\n * @returns result input\n */\n multiplyToRef(otherVector, result) {\n result.x = this.x * otherVector.x;\n result.y = this.y * otherVector.y;\n return result;\n }\n /**\n * Gets a new Vector2 set with the Vector2 coordinates multiplied by the given floats\n * Example Playground https://playground.babylonjs.com/#QYBWV4#89\n * @param x defines the first coordinate\n * @param y defines the second coordinate\n * @returns a new Vector2\n */\n multiplyByFloats(x, y) {\n return new this.constructor(this.x * x, this.y * y);\n }\n /**\n * Returns a new Vector2 set with the Vector2 coordinates divided by the given one coordinates\n * Example Playground https://playground.babylonjs.com/#QYBWV4#27\n * @param otherVector defines the other vector\n * @returns a new Vector2\n */\n divide(otherVector) {\n return new this.constructor(this.x / otherVector.x, this.y / otherVector.y);\n }\n /**\n * Sets the \"result\" coordinates with the Vector2 divided by the given one coordinates\n * Example Playground https://playground.babylonjs.com/#QYBWV4#30\n * @param otherVector defines the other vector\n * @param result defines the target vector\n * @returns result input\n */\n divideToRef(otherVector, result) {\n result.x = this.x / otherVector.x;\n result.y = this.y / otherVector.y;\n return result;\n }\n /**\n * Divides the current Vector2 coordinates by the given ones\n * Example Playground https://playground.babylonjs.com/#QYBWV4#28\n * @param otherVector defines the other vector\n * @returns the current updated Vector2\n */\n divideInPlace(otherVector) {\n return this.divideToRef(otherVector, this);\n }\n /**\n * Gets a new Vector2 with current Vector2 negated coordinates\n * Example Playground https://playground.babylonjs.com/#QYBWV4#22\n * @returns a new Vector2\n */\n negate() {\n return new this.constructor(-this.x, -this.y);\n }\n /**\n * Negate this vector in place\n * Example Playground https://playground.babylonjs.com/#QYBWV4#23\n * @returns this\n */\n negateInPlace() {\n this.x *= -1;\n this.y *= -1;\n return this;\n }\n /**\n * Negate the current Vector2 and stores the result in the given vector \"result\" coordinates\n * Example Playground https://playground.babylonjs.com/#QYBWV4#41\n * @param result defines the Vector3 object where to store the result\n * @returns the result\n */\n negateToRef(result) {\n return result.copyFromFloats(this.x * -1, this.y * -1);\n }\n /**\n * Multiply the Vector2 coordinates by\n * Example Playground https://playground.babylonjs.com/#QYBWV4#59\n * @param scale defines the scaling factor\n * @returns the current updated Vector2\n */\n scaleInPlace(scale) {\n this.x *= scale;\n this.y *= scale;\n return this;\n }\n /**\n * Returns a new Vector2 scaled by \"scale\" from the current Vector2\n * Example Playground https://playground.babylonjs.com/#QYBWV4#52\n * @param scale defines the scaling factor\n * @returns a new Vector2\n */\n scale(scale) {\n const result = new this.constructor(0, 0);\n this.scaleToRef(scale, result);\n return result;\n }\n /**\n * Scale the current Vector2 values by a factor to a given Vector2\n * Example Playground https://playground.babylonjs.com/#QYBWV4#57\n * @param scale defines the scale factor\n * @param result defines the Vector2 object where to store the result\n * @returns result input\n */\n scaleToRef(scale, result) {\n result.x = this.x * scale;\n result.y = this.y * scale;\n return result;\n }\n /**\n * Scale the current Vector2 values by a factor and add the result to a given Vector2\n * Example Playground https://playground.babylonjs.com/#QYBWV4#58\n * @param scale defines the scale factor\n * @param result defines the Vector2 object where to store the result\n * @returns result input\n */\n scaleAndAddToRef(scale, result) {\n result.x += this.x * scale;\n result.y += this.y * scale;\n return result;\n }\n /**\n * Gets a boolean if two vectors are equals\n * Example Playground https://playground.babylonjs.com/#QYBWV4#31\n * @param otherVector defines the other vector\n * @returns true if the given vector coordinates strictly equal the current Vector2 ones\n */\n equals(otherVector) {\n return otherVector && this.x === otherVector.x && this.y === otherVector.y;\n }\n /**\n * Gets a boolean if two vectors are equals (using an epsilon value)\n * Example Playground https://playground.babylonjs.com/#QYBWV4#32\n * @param otherVector defines the other vector\n * @param epsilon defines the minimal distance to consider equality\n * @returns true if the given vector coordinates are close to the current ones by a distance of epsilon.\n */\n equalsWithEpsilon(otherVector, epsilon = Epsilon) {\n return otherVector && Scalar.WithinEpsilon(this.x, otherVector.x, epsilon) && Scalar.WithinEpsilon(this.y, otherVector.y, epsilon);\n }\n /**\n * Gets a new Vector2 from current Vector2 floored values\n * Example Playground https://playground.babylonjs.com/#QYBWV4#35\n * eg (1.2, 2.31) returns (1, 2)\n * @returns a new Vector2\n */\n floor() {\n return new this.constructor(Math.floor(this.x), Math.floor(this.y));\n }\n /**\n * Gets a new Vector2 from current Vector2 fractional values\n * Example Playground https://playground.babylonjs.com/#QYBWV4#34\n * eg (1.2, 2.31) returns (0.2, 0.31)\n * @returns a new Vector2\n */\n fract() {\n return new this.constructor(this.x - Math.floor(this.x), this.y - Math.floor(this.y));\n }\n /**\n * Rotate the current vector into a given result vector\n * Example Playground https://playground.babylonjs.com/#QYBWV4#49\n * @param angle defines the rotation angle\n * @param result defines the result vector where to store the rotated vector\n * @returns result input\n */\n rotateToRef(angle, result) {\n const cos = Math.cos(angle);\n const sin = Math.sin(angle);\n const x = cos * this.x - sin * this.y;\n const y = sin * this.x + cos * this.y;\n result.x = x;\n result.y = y;\n return result;\n }\n // Properties\n /**\n * Gets the length of the vector\n * @returns the vector length (float)\n */\n length() {\n return Math.sqrt(this.x * this.x + this.y * this.y);\n }\n /**\n * Gets the vector squared length\n * @returns the vector squared length (float)\n */\n lengthSquared() {\n return this.x * this.x + this.y * this.y;\n }\n // Methods\n /**\n * Normalize the vector\n * Example Playground https://playground.babylonjs.com/#QYBWV4#48\n * @returns the current updated Vector2\n */\n normalize() {\n Vector2.NormalizeToRef(this, this);\n return this;\n }\n /**\n * Gets a new Vector2 copied from the Vector2\n * Example Playground https://playground.babylonjs.com/#QYBWV4#20\n * @returns a new Vector2\n */\n clone() {\n return new this.constructor(this.x, this.y);\n }\n // Statics\n /**\n * Gets a new Vector2(0, 0)\n * @returns a new Vector2\n */\n static Zero() {\n return new Vector2(0, 0);\n }\n /**\n * Gets a new Vector2(1, 1)\n * @returns a new Vector2\n */\n static One() {\n return new Vector2(1, 1);\n }\n /**\n * Returns a new Vector2 with random values between min and max\n * @param min the minimum random value\n * @param max the maximum random value\n * @returns a Vector2 with random values between min and max\n */\n static Random(min = 0, max = 1) {\n return new Vector2(Scalar.RandomRange(min, max), Scalar.RandomRange(min, max));\n }\n /**\n * Gets a zero Vector2 that must not be updated\n */\n static get ZeroReadOnly() {\n return Vector2._ZeroReadOnly;\n }\n /**\n * Gets a new Vector2 set from the given index element of the given array\n * Example Playground https://playground.babylonjs.com/#QYBWV4#79\n * @param array defines the data source\n * @param offset defines the offset in the data source\n * @returns a new Vector2\n */\n static FromArray(array, offset = 0) {\n return new Vector2(array[offset], array[offset + 1]);\n }\n /**\n * Sets \"result\" from the given index element of the given array\n * Example Playground https://playground.babylonjs.com/#QYBWV4#80\n * @param array defines the data source\n * @param offset defines the offset in the data source\n * @param result defines the target vector\n * @returns result input\n */\n static FromArrayToRef(array, offset, result) {\n result.x = array[offset];\n result.y = array[offset + 1];\n return result;\n }\n /**\n * Gets a new Vector2 located for \"amount\" (float) on the CatmullRom spline defined by the given four Vector2\n * Example Playground https://playground.babylonjs.com/#QYBWV4#65\n * @param value1 defines 1st point of control\n * @param value2 defines 2nd point of control\n * @param value3 defines 3rd point of control\n * @param value4 defines 4th point of control\n * @param amount defines the interpolation factor\n * @returns a new Vector2\n */\n static CatmullRom(value1, value2, value3, value4, amount) {\n const squared = amount * amount;\n const cubed = amount * squared;\n const x = 0.5 *\n (2.0 * value2.x +\n (-value1.x + value3.x) * amount +\n (2.0 * value1.x - 5.0 * value2.x + 4.0 * value3.x - value4.x) * squared +\n (-value1.x + 3.0 * value2.x - 3.0 * value3.x + value4.x) * cubed);\n const y = 0.5 *\n (2.0 * value2.y +\n (-value1.y + value3.y) * amount +\n (2.0 * value1.y - 5.0 * value2.y + 4.0 * value3.y - value4.y) * squared +\n (-value1.y + 3.0 * value2.y - 3.0 * value3.y + value4.y) * cubed);\n return new value1.constructor(x, y);\n }\n /**\n * Returns a new Vector2 set with same the coordinates than \"value\" ones if the vector \"value\" is in the square defined by \"min\" and \"max\".\n * If a coordinate of \"value\" is lower than \"min\" coordinates, the returned Vector2 is given this \"min\" coordinate.\n * If a coordinate of \"value\" is greater than \"max\" coordinates, the returned Vector2 is given this \"max\" coordinate\n * Example Playground https://playground.babylonjs.com/#QYBWV4#76\n * @param value defines the value to clamp\n * @param min defines the lower limit\n * @param max defines the upper limit\n * @returns a new Vector2\n */\n static Clamp(value, min, max) {\n let x = value.x;\n x = x > max.x ? max.x : x;\n x = x < min.x ? min.x : x;\n let y = value.y;\n y = y > max.y ? max.y : y;\n y = y < min.y ? min.y : y;\n return new value.constructor(x, y);\n }\n /**\n * Returns a new Vector2 located for \"amount\" (float) on the Hermite spline defined by the vectors \"value1\", \"value2\", \"tangent1\", \"tangent2\"\n * Example Playground https://playground.babylonjs.com/#QYBWV4#81\n * @param value1 defines the 1st control point\n * @param tangent1 defines the outgoing tangent\n * @param value2 defines the 2nd control point\n * @param tangent2 defines the incoming tangent\n * @param amount defines the interpolation factor\n * @returns a new Vector2\n */\n static Hermite(value1, tangent1, value2, tangent2, amount) {\n const squared = amount * amount;\n const cubed = amount * squared;\n const part1 = 2.0 * cubed - 3.0 * squared + 1.0;\n const part2 = -2.0 * cubed + 3.0 * squared;\n const part3 = cubed - 2.0 * squared + amount;\n const part4 = cubed - squared;\n const x = value1.x * part1 + value2.x * part2 + tangent1.x * part3 + tangent2.x * part4;\n const y = value1.y * part1 + value2.y * part2 + tangent1.y * part3 + tangent2.y * part4;\n return new value1.constructor(x, y);\n }\n /**\n * Returns a new Vector2 which is the 1st derivative of the Hermite spline defined by the vectors \"value1\", \"value2\", \"tangent1\", \"tangent2\".\n * Example Playground https://playground.babylonjs.com/#QYBWV4#82\n * @param value1 defines the first control point\n * @param tangent1 defines the first tangent\n * @param value2 defines the second control point\n * @param tangent2 defines the second tangent\n * @param time define where the derivative must be done\n * @returns 1st derivative\n */\n static Hermite1stDerivative(value1, tangent1, value2, tangent2, time) {\n const result = new value1.constructor();\n this.Hermite1stDerivativeToRef(value1, tangent1, value2, tangent2, time, result);\n return result;\n }\n /**\n * Returns a new Vector2 which is the 1st derivative of the Hermite spline defined by the vectors \"value1\", \"value2\", \"tangent1\", \"tangent2\".\n * Example Playground https://playground.babylonjs.com/#QYBWV4#83\n * @param value1 defines the first control point\n * @param tangent1 defines the first tangent\n * @param value2 defines the second control point\n * @param tangent2 defines the second tangent\n * @param time define where the derivative must be done\n * @param result define where the derivative will be stored\n * @returns result input\n */\n static Hermite1stDerivativeToRef(value1, tangent1, value2, tangent2, time, result) {\n const t2 = time * time;\n result.x = (t2 - time) * 6 * value1.x + (3 * t2 - 4 * time + 1) * tangent1.x + (-t2 + time) * 6 * value2.x + (3 * t2 - 2 * time) * tangent2.x;\n result.y = (t2 - time) * 6 * value1.y + (3 * t2 - 4 * time + 1) * tangent1.y + (-t2 + time) * 6 * value2.y + (3 * t2 - 2 * time) * tangent2.y;\n return result;\n }\n /**\n * Returns a new Vector2 located for \"amount\" (float) on the linear interpolation between the vector \"start\" adn the vector \"end\".\n * Example Playground https://playground.babylonjs.com/#QYBWV4#84\n * @param start defines the start vector\n * @param end defines the end vector\n * @param amount defines the interpolation factor\n * @returns a new Vector2\n */\n static Lerp(start, end, amount) {\n const x = start.x + (end.x - start.x) * amount;\n const y = start.y + (end.y - start.y) * amount;\n return new start.constructor(x, y);\n }\n /**\n * Gets the dot product of the vector \"left\" and the vector \"right\"\n * Example Playground https://playground.babylonjs.com/#QYBWV4#90\n * @param left defines first vector\n * @param right defines second vector\n * @returns the dot product (float)\n */\n static Dot(left, right) {\n return left.x * right.x + left.y * right.y;\n }\n /**\n * Returns a new Vector2 equal to the normalized given vector\n * Example Playground https://playground.babylonjs.com/#QYBWV4#46\n * @param vector defines the vector to normalize\n * @returns a new Vector2\n */\n static Normalize(vector) {\n const newVector = new vector.constructor();\n this.NormalizeToRef(vector, newVector);\n return newVector;\n }\n /**\n * Normalize a given vector into a second one\n * Example Playground https://playground.babylonjs.com/#QYBWV4#50\n * @param vector defines the vector to normalize\n * @param result defines the vector where to store the result\n * @returns result input\n */\n static NormalizeToRef(vector, result) {\n const len = vector.length();\n if (len === 0) {\n return result;\n }\n result.x = vector.x / len;\n result.y = vector.y / len;\n return result;\n }\n /**\n * Gets a new Vector2 set with the minimal coordinate values from the \"left\" and \"right\" vectors\n * Example Playground https://playground.babylonjs.com/#QYBWV4#86\n * @param left defines 1st vector\n * @param right defines 2nd vector\n * @returns a new Vector2\n */\n static Minimize(left, right) {\n const x = left.x < right.x ? left.x : right.x;\n const y = left.y < right.y ? left.y : right.y;\n return new left.constructor(x, y);\n }\n /**\n * Gets a new Vector2 set with the maximal coordinate values from the \"left\" and \"right\" vectors\n * Example Playground https://playground.babylonjs.com/#QYBWV4#86\n * @param left defines 1st vector\n * @param right defines 2nd vector\n * @returns a new Vector2\n */\n static Maximize(left, right) {\n const x = left.x > right.x ? left.x : right.x;\n const y = left.y > right.y ? left.y : right.y;\n return new left.constructor(x, y);\n }\n /**\n * Gets a new Vector2 set with the transformed coordinates of the given vector by the given transformation matrix\n * Example Playground https://playground.babylonjs.com/#QYBWV4#17\n * @param vector defines the vector to transform\n * @param transformation defines the matrix to apply\n * @returns a new Vector2\n */\n static Transform(vector, transformation) {\n const result = new vector.constructor();\n Vector2.TransformToRef(vector, transformation, result);\n return result;\n }\n /**\n * Transforms the given vector coordinates by the given transformation matrix and stores the result in the vector \"result\" coordinates\n * Example Playground https://playground.babylonjs.com/#QYBWV4#19\n * @param vector defines the vector to transform\n * @param transformation defines the matrix to apply\n * @param result defines the target vector\n * @returns result input\n */\n static TransformToRef(vector, transformation, result) {\n const m = transformation.m;\n const x = vector.x * m[0] + vector.y * m[4] + m[12];\n const y = vector.x * m[1] + vector.y * m[5] + m[13];\n result.x = x;\n result.y = y;\n return result;\n }\n /**\n * Determines if a given vector is included in a triangle\n * Example Playground https://playground.babylonjs.com/#QYBWV4#87\n * @param p defines the vector to test\n * @param p0 defines 1st triangle point\n * @param p1 defines 2nd triangle point\n * @param p2 defines 3rd triangle point\n * @returns true if the point \"p\" is in the triangle defined by the vectors \"p0\", \"p1\", \"p2\"\n */\n static PointInTriangle(p, p0, p1, p2) {\n const a = (1 / 2) * (-p1.y * p2.x + p0.y * (-p1.x + p2.x) + p0.x * (p1.y - p2.y) + p1.x * p2.y);\n const sign = a < 0 ? -1 : 1;\n const s = (p0.y * p2.x - p0.x * p2.y + (p2.y - p0.y) * p.x + (p0.x - p2.x) * p.y) * sign;\n const t = (p0.x * p1.y - p0.y * p1.x + (p0.y - p1.y) * p.x + (p1.x - p0.x) * p.y) * sign;\n return s > 0 && t > 0 && s + t < 2 * a * sign;\n }\n /**\n * Gets the distance between the vectors \"value1\" and \"value2\"\n * Example Playground https://playground.babylonjs.com/#QYBWV4#71\n * @param value1 defines first vector\n * @param value2 defines second vector\n * @returns the distance between vectors\n */\n static Distance(value1, value2) {\n return Math.sqrt(Vector2.DistanceSquared(value1, value2));\n }\n /**\n * Returns the squared distance between the vectors \"value1\" and \"value2\"\n * Example Playground https://playground.babylonjs.com/#QYBWV4#72\n * @param value1 defines first vector\n * @param value2 defines second vector\n * @returns the squared distance between vectors\n */\n static DistanceSquared(value1, value2) {\n const x = value1.x - value2.x;\n const y = value1.y - value2.y;\n return x * x + y * y;\n }\n /**\n * Gets a new Vector2 located at the center of the vectors \"value1\" and \"value2\"\n * Example Playground https://playground.babylonjs.com/#QYBWV4#86\n * Example Playground https://playground.babylonjs.com/#QYBWV4#66\n * @param value1 defines first vector\n * @param value2 defines second vector\n * @returns a new Vector2\n */\n static Center(value1, value2) {\n const result = new value1.constructor();\n return Vector2.CenterToRef(value1, value2, result);\n }\n /**\n * Gets the center of the vectors \"value1\" and \"value2\" and stores the result in the vector \"ref\"\n * Example Playground https://playground.babylonjs.com/#QYBWV4#66\n * @param value1 defines first vector\n * @param value2 defines second vector\n * @param ref defines third vector\n * @returns ref\n */\n static CenterToRef(value1, value2, ref) {\n return ref.copyFromFloats((value1.x + value2.x) / 2, (value1.y + value2.y) / 2);\n }\n /**\n * Gets the shortest distance (float) between the point \"p\" and the segment defined by the two points \"segA\" and \"segB\".\n * Example Playground https://playground.babylonjs.com/#QYBWV4#77\n * @param p defines the middle point\n * @param segA defines one point of the segment\n * @param segB defines the other point of the segment\n * @returns the shortest distance\n */\n static DistanceOfPointFromSegment(p, segA, segB) {\n const l2 = Vector2.DistanceSquared(segA, segB);\n if (l2 === 0.0) {\n return Vector2.Distance(p, segA);\n }\n const v = segB.subtract(segA);\n const t = Math.max(0, Math.min(1, Vector2.Dot(p.subtract(segA), v) / l2));\n const proj = segA.add(v.multiplyByFloats(t, t));\n return Vector2.Distance(p, proj);\n }\n }\n Vector2._ZeroReadOnly = Vector2.Zero();\n /**\n * Class used to store (x,y,z) vector representation\n * A Vector3 is the main object used in 3D geometry\n * It can represent either the coordinates of a point the space, either a direction\n * Reminder: js uses a left handed forward facing system\n * Example Playground - Overview - https://playground.babylonjs.com/#R1F8YU\n */\n class Vector3 {\n /** Gets or sets the x coordinate */\n get x() {\n return this._x;\n }\n set x(value) {\n this._x = value;\n this._isDirty = true;\n }\n /** Gets or sets the y coordinate */\n get y() {\n return this._y;\n }\n set y(value) {\n this._y = value;\n this._isDirty = true;\n }\n /** Gets or sets the z coordinate */\n get z() {\n return this._z;\n }\n set z(value) {\n this._z = value;\n this._isDirty = true;\n }\n /**\n * Creates a new Vector3 object from the given x, y, z (floats) coordinates.\n * @param x defines the first coordinates (on X axis)\n * @param y defines the second coordinates (on Y axis)\n * @param z defines the third coordinates (on Z axis)\n */\n constructor(x = 0, y = 0, z = 0) {\n /** @internal */\n this._isDirty = true;\n this._x = x;\n this._y = y;\n this._z = z;\n }\n /**\n * Creates a string representation of the Vector3\n * Example Playground https://playground.babylonjs.com/#R1F8YU#67\n * @returns a string with the Vector3 coordinates.\n */\n toString() {\n return `{X: ${this._x} Y: ${this._y} Z: ${this._z}}`;\n }\n /**\n * Gets the class name\n * @returns the string \"Vector3\"\n */\n getClassName() {\n return \"Vector3\";\n }\n /**\n * Creates the Vector3 hash code\n * @returns a number which tends to be unique between Vector3 instances\n */\n getHashCode() {\n const x = _ExtractAsInt(this._x);\n const y = _ExtractAsInt(this._y);\n const z = _ExtractAsInt(this._z);\n let hash = x;\n hash = (hash * 397) ^ y;\n hash = (hash * 397) ^ z;\n return hash;\n }\n // Operators\n /**\n * Creates an array containing three elements : the coordinates of the Vector3\n * Example Playground https://playground.babylonjs.com/#R1F8YU#10\n * @returns a new array of numbers\n */\n asArray() {\n const result = [];\n this.toArray(result, 0);\n return result;\n }\n /**\n * Populates the given array or Float32Array from the given index with the successive coordinates of the Vector3\n * Example Playground https://playground.babylonjs.com/#R1F8YU#65\n * @param array defines the destination array\n * @param index defines the offset in the destination array\n * @returns the current Vector3\n */\n toArray(array, index = 0) {\n array[index] = this._x;\n array[index + 1] = this._y;\n array[index + 2] = this._z;\n return this;\n }\n /**\n * Update the current vector from an array\n * Example Playground https://playground.babylonjs.com/#R1F8YU#24\n * @param array defines the destination array\n * @param index defines the offset in the destination array\n * @returns the current Vector3\n */\n fromArray(array, index = 0) {\n Vector3.FromArrayToRef(array, index, this);\n return this;\n }\n /**\n * Converts the current Vector3 into a quaternion (considering that the Vector3 contains Euler angles representation of a rotation)\n * Example Playground https://playground.babylonjs.com/#R1F8YU#66\n * @returns a new Quaternion object, computed from the Vector3 coordinates\n */\n toQuaternion() {\n return Quaternion.RotationYawPitchRoll(this._y, this._x, this._z);\n }\n /**\n * Adds the given vector to the current Vector3\n * Example Playground https://playground.babylonjs.com/#R1F8YU#4\n * @param otherVector defines the second operand\n * @returns the current updated Vector3\n */\n addInPlace(otherVector) {\n return this.addInPlaceFromFloats(otherVector._x, otherVector._y, otherVector._z);\n }\n /**\n * Adds the given coordinates to the current Vector3\n * Example Playground https://playground.babylonjs.com/#R1F8YU#5\n * @param x defines the x coordinate of the operand\n * @param y defines the y coordinate of the operand\n * @param z defines the z coordinate of the operand\n * @returns the current updated Vector3\n */\n addInPlaceFromFloats(x, y, z) {\n this._x += x;\n this._y += y;\n this._z += z;\n this._isDirty = true;\n return this;\n }\n /**\n * Gets a new Vector3, result of the addition the current Vector3 and the given vector\n * Example Playground https://playground.babylonjs.com/#R1F8YU#3\n * @param otherVector defines the second operand\n * @returns the resulting Vector3\n */\n add(otherVector) {\n return new this.constructor(this._x + otherVector._x, this._y + otherVector._y, this._z + otherVector._z);\n }\n /**\n * Adds the current Vector3 to the given one and stores the result in the vector \"result\"\n * Example Playground https://playground.babylonjs.com/#R1F8YU#6\n * @param otherVector defines the second operand\n * @param result defines the Vector3 object where to store the result\n * @returns the result\n */\n addToRef(otherVector, result) {\n return result.copyFromFloats(this._x + otherVector._x, this._y + otherVector._y, this._z + otherVector._z);\n }\n /**\n * Subtract the given vector from the current Vector3\n * Example Playground https://playground.babylonjs.com/#R1F8YU#61\n * @param otherVector defines the second operand\n * @returns the current updated Vector3\n */\n subtractInPlace(otherVector) {\n this._x -= otherVector._x;\n this._y -= otherVector._y;\n this._z -= otherVector._z;\n this._isDirty = true;\n return this;\n }\n /**\n * Returns a new Vector3, result of the subtraction of the given vector from the current Vector3\n * Example Playground https://playground.babylonjs.com/#R1F8YU#60\n * @param otherVector defines the second operand\n * @returns the resulting Vector3\n */\n subtract(otherVector) {\n return new this.constructor(this._x - otherVector._x, this._y - otherVector._y, this._z - otherVector._z);\n }\n /**\n * Subtracts the given vector from the current Vector3 and stores the result in the vector \"result\".\n * Example Playground https://playground.babylonjs.com/#R1F8YU#63\n * @param otherVector defines the second operand\n * @param result defines the Vector3 object where to store the result\n * @returns the result\n */\n subtractToRef(otherVector, result) {\n return this.subtractFromFloatsToRef(otherVector._x, otherVector._y, otherVector._z, result);\n }\n /**\n * Returns a new Vector3 set with the subtraction of the given floats from the current Vector3 coordinates\n * Example Playground https://playground.babylonjs.com/#R1F8YU#62\n * @param x defines the x coordinate of the operand\n * @param y defines the y coordinate of the operand\n * @param z defines the z coordinate of the operand\n * @returns the resulting Vector3\n */\n subtractFromFloats(x, y, z) {\n return new this.constructor(this._x - x, this._y - y, this._z - z);\n }\n /**\n * Subtracts the given floats from the current Vector3 coordinates and set the given vector \"result\" with this result\n * Example Playground https://playground.babylonjs.com/#R1F8YU#64\n * @param x defines the x coordinate of the operand\n * @param y defines the y coordinate of the operand\n * @param z defines the z coordinate of the operand\n * @param result defines the Vector3 object where to store the result\n * @returns the result\n */\n subtractFromFloatsToRef(x, y, z, result) {\n return result.copyFromFloats(this._x - x, this._y - y, this._z - z);\n }\n /**\n * Gets a new Vector3 set with the current Vector3 negated coordinates\n * Example Playground https://playground.babylonjs.com/#R1F8YU#35\n * @returns a new Vector3\n */\n negate() {\n return new this.constructor(-this._x, -this._y, -this._z);\n }\n /**\n * Negate this vector in place\n * Example Playground https://playground.babylonjs.com/#R1F8YU#36\n * @returns this\n */\n negateInPlace() {\n this._x *= -1;\n this._y *= -1;\n this._z *= -1;\n this._isDirty = true;\n return this;\n }\n /**\n * Negate the current Vector3 and stores the result in the given vector \"result\" coordinates\n * Example Playground https://playground.babylonjs.com/#R1F8YU#37\n * @param result defines the Vector3 object where to store the result\n * @returns the result\n */\n negateToRef(result) {\n return result.copyFromFloats(this._x * -1, this._y * -1, this._z * -1);\n }\n /**\n * Multiplies the Vector3 coordinates by the float \"scale\"\n * Example Playground https://playground.babylonjs.com/#R1F8YU#56\n * @param scale defines the multiplier factor\n * @returns the current updated Vector3\n */\n scaleInPlace(scale) {\n this._x *= scale;\n this._y *= scale;\n this._z *= scale;\n this._isDirty = true;\n return this;\n }\n /**\n * Returns a new Vector3 set with the current Vector3 coordinates multiplied by the float \"scale\"\n * Example Playground https://playground.babylonjs.com/#R1F8YU#53\n * @param scale defines the multiplier factor\n * @returns a new Vector3\n */\n scale(scale) {\n return new this.constructor(this._x * scale, this._y * scale, this._z * scale);\n }\n /**\n * Multiplies the current Vector3 coordinates by the float \"scale\" and stores the result in the given vector \"result\" coordinates\n * Example Playground https://playground.babylonjs.com/#R1F8YU#57\n * @param scale defines the multiplier factor\n * @param result defines the Vector3 object where to store the result\n * @returns the result\n */\n scaleToRef(scale, result) {\n return result.copyFromFloats(this._x * scale, this._y * scale, this._z * scale);\n }\n /**\n * Creates a vector normal (perpendicular) to the current Vector3 and stores the result in the given vector\n * Out of the infinite possibilities the normal chosen is the one formed by rotating the current vector\n * 90 degrees about an axis which lies perpendicular to the current vector\n * and its projection on the xz plane. In the case of a current vector in the xz plane\n * the normal is calculated to be along the y axis.\n * Example Playground https://playground.babylonjs.com/#R1F8YU#230\n * Example Playground https://playground.babylonjs.com/#R1F8YU#231\n * @param result defines the Vector3 object where to store the resultant normal\n * returns the result\n */\n getNormalToRef(result) {\n /**\n * Calculates the spherical coordinates of the current vector\n * so saves on memory rather than importing whole Spherical Class\n */\n const radius = this.length();\n let theta = Math.acos(this.y / radius);\n const phi = Math.atan2(this.z, this.x);\n //makes angle 90 degs to current vector\n if (theta > Math.PI / 2) {\n theta -= Math.PI / 2;\n }\n else {\n theta += Math.PI / 2;\n }\n //Calculates resutant normal vector from spherical coordinate of perpendicular vector\n const x = radius * Math.sin(theta) * Math.cos(phi);\n const y = radius * Math.cos(theta);\n const z = radius * Math.sin(theta) * Math.sin(phi);\n result.set(x, y, z);\n return result;\n }\n /**\n * Rotates the vector using the given unit quaternion and stores the new vector in result\n * Example Playground https://playground.babylonjs.com/#R1F8YU#9\n * @param q the unit quaternion representing the rotation\n * @param result the output vector\n * @returns the result\n */\n applyRotationQuaternionToRef(q, result) {\n // Derived from https://raw.org/proof/vector-rotation-using-quaternions/\n const vx = this._x, vy = this._y, vz = this._z;\n const qx = q._x, qy = q._y, qz = q._z, qw = q._w;\n // t = 2q x v\n const tx = 2 * (qy * vz - qz * vy);\n const ty = 2 * (qz * vx - qx * vz);\n const tz = 2 * (qx * vy - qy * vx);\n // v + w t + q x t\n result._x = vx + qw * tx + qy * tz - qz * ty;\n result._y = vy + qw * ty + qz * tx - qx * tz;\n result._z = vz + qw * tz + qx * ty - qy * tx;\n result._isDirty = true;\n return result;\n }\n /**\n * Rotates the vector in place using the given unit quaternion\n * Example Playground https://playground.babylonjs.com/#R1F8YU#8\n * @param q the unit quaternion representing the rotation\n * @returns the current updated Vector3\n */\n applyRotationQuaternionInPlace(q) {\n return this.applyRotationQuaternionToRef(q, this);\n }\n /**\n * Rotates the vector using the given unit quaternion and returns the new vector\n * Example Playground https://playground.babylonjs.com/#R1F8YU#7\n * @param q the unit quaternion representing the rotation\n * @returns a new Vector3\n */\n applyRotationQuaternion(q) {\n return this.applyRotationQuaternionToRef(q, new this.constructor());\n }\n /**\n * Scale the current Vector3 values by a factor and add the result to a given Vector3\n * Example Playground https://playground.babylonjs.com/#R1F8YU#55\n * @param scale defines the scale factor\n * @param result defines the Vector3 object where to store the result\n * @returns result input\n */\n scaleAndAddToRef(scale, result) {\n return result.addInPlaceFromFloats(this._x * scale, this._y * scale, this._z * scale);\n }\n /**\n * Projects the current point Vector3 to a plane along a ray starting from a specified origin and passing through the current point Vector3.\n * Example Playground https://playground.babylonjs.com/#R1F8YU#48\n * @param plane defines the plane to project to\n * @param origin defines the origin of the projection ray\n * @returns the projected vector3\n */\n projectOnPlane(plane, origin) {\n const result = new this.constructor();\n this.projectOnPlaneToRef(plane, origin, result);\n return result;\n }\n /**\n * Projects the current point Vector3 to a plane along a ray starting from a specified origin and passing through the current point Vector3.\n * Example Playground https://playground.babylonjs.com/#R1F8YU#49\n * @param plane defines the plane to project to\n * @param origin defines the origin of the projection ray\n * @param result defines the Vector3 where to store the result\n * @returns result input\n */\n projectOnPlaneToRef(plane, origin, result) {\n const n = plane.normal;\n const d = plane.d;\n const V = MathTmp.Vector3[0];\n // ray direction\n this.subtractToRef(origin, V);\n V.normalize();\n const denom = Vector3.Dot(V, n);\n //When the ray is close to parallel to the plane return infinity vector\n if (Math.abs(denom) < Math.pow(10, -10)) {\n result.setAll(Infinity);\n }\n else {\n const t = -(Vector3.Dot(origin, n) + d) / denom;\n // P = P0 + t*V\n const scaledV = V.scaleInPlace(t);\n origin.addToRef(scaledV, result);\n }\n return result;\n }\n /**\n * Returns true if the current Vector3 and the given vector coordinates are strictly equal\n * Example Playground https://playground.babylonjs.com/#R1F8YU#19\n * @param otherVector defines the second operand\n * @returns true if both vectors are equals\n */\n equals(otherVector) {\n return otherVector && this._x === otherVector._x && this._y === otherVector._y && this._z === otherVector._z;\n }\n /**\n * Returns true if the current Vector3 and the given vector coordinates are distant less than epsilon\n * Example Playground https://playground.babylonjs.com/#R1F8YU#21\n * @param otherVector defines the second operand\n * @param epsilon defines the minimal distance to define values as equals\n * @returns true if both vectors are distant less than epsilon\n */\n equalsWithEpsilon(otherVector, epsilon = Epsilon) {\n return (otherVector &&\n Scalar.WithinEpsilon(this._x, otherVector._x, epsilon) &&\n Scalar.WithinEpsilon(this._y, otherVector._y, epsilon) &&\n Scalar.WithinEpsilon(this._z, otherVector._z, epsilon));\n }\n /**\n * Returns true if the current Vector3 coordinates equals the given floats\n * Example Playground https://playground.babylonjs.com/#R1F8YU#20\n * @param x defines the x coordinate of the operand\n * @param y defines the y coordinate of the operand\n * @param z defines the z coordinate of the operand\n * @returns true if both vectors are equal\n */\n equalsToFloats(x, y, z) {\n return this._x === x && this._y === y && this._z === z;\n }\n /**\n * Multiplies the current Vector3 coordinates by the given ones\n * Example Playground https://playground.babylonjs.com/#R1F8YU#32\n * @param otherVector defines the second operand\n * @returns the current updated Vector3\n */\n multiplyInPlace(otherVector) {\n this._x *= otherVector._x;\n this._y *= otherVector._y;\n this._z *= otherVector._z;\n this._isDirty = true;\n return this;\n }\n /**\n * Returns a new Vector3, result of the multiplication of the current Vector3 by the given vector\n * Example Playground https://playground.babylonjs.com/#R1F8YU#31\n * @param otherVector defines the second operand\n * @returns the new Vector3\n */\n multiply(otherVector) {\n return this.multiplyByFloats(otherVector._x, otherVector._y, otherVector._z);\n }\n /**\n * Multiplies the current Vector3 by the given one and stores the result in the given vector \"result\"\n * Example Playground https://playground.babylonjs.com/#R1F8YU#33\n * @param otherVector defines the second operand\n * @param result defines the Vector3 object where to store the result\n * @returns the result\n */\n multiplyToRef(otherVector, result) {\n return result.copyFromFloats(this._x * otherVector._x, this._y * otherVector._y, this._z * otherVector._z);\n }\n /**\n * Returns a new Vector3 set with the result of the multiplication of the current Vector3 coordinates by the given floats\n * Example Playground https://playground.babylonjs.com/#R1F8YU#34\n * @param x defines the x coordinate of the operand\n * @param y defines the y coordinate of the operand\n * @param z defines the z coordinate of the operand\n * @returns the new Vector3\n */\n multiplyByFloats(x, y, z) {\n return new this.constructor(this._x * x, this._y * y, this._z * z);\n }\n /**\n * Returns a new Vector3 set with the result of the division of the current Vector3 coordinates by the given ones\n * Example Playground https://playground.babylonjs.com/#R1F8YU#16\n * @param otherVector defines the second operand\n * @returns the new Vector3\n */\n divide(otherVector) {\n return new this.constructor(this._x / otherVector._x, this._y / otherVector._y, this._z / otherVector._z);\n }\n /**\n * Divides the current Vector3 coordinates by the given ones and stores the result in the given vector \"result\"\n * Example Playground https://playground.babylonjs.com/#R1F8YU#18\n * @param otherVector defines the second operand\n * @param result defines the Vector3 object where to store the result\n * @returns the result\n */\n divideToRef(otherVector, result) {\n return result.copyFromFloats(this._x / otherVector._x, this._y / otherVector._y, this._z / otherVector._z);\n }\n /**\n * Divides the current Vector3 coordinates by the given ones.\n * Example Playground https://playground.babylonjs.com/#R1F8YU#17\n * @param otherVector defines the second operand\n * @returns the current updated Vector3\n */\n divideInPlace(otherVector) {\n return this.divideToRef(otherVector, this);\n }\n /**\n * Updates the current Vector3 with the minimal coordinate values between its and the given vector ones\n * Example Playground https://playground.babylonjs.com/#R1F8YU#29\n * @param other defines the second operand\n * @returns the current updated Vector3\n */\n minimizeInPlace(other) {\n return this.minimizeInPlaceFromFloats(other._x, other._y, other._z);\n }\n /**\n * Updates the current Vector3 with the maximal coordinate values between its and the given vector ones.\n * Example Playground https://playground.babylonjs.com/#R1F8YU#27\n * @param other defines the second operand\n * @returns the current updated Vector3\n */\n maximizeInPlace(other) {\n return this.maximizeInPlaceFromFloats(other._x, other._y, other._z);\n }\n /**\n * Updates the current Vector3 with the minimal coordinate values between its and the given coordinates\n * Example Playground https://playground.babylonjs.com/#R1F8YU#30\n * @param x defines the x coordinate of the operand\n * @param y defines the y coordinate of the operand\n * @param z defines the z coordinate of the operand\n * @returns the current updated Vector3\n */\n minimizeInPlaceFromFloats(x, y, z) {\n if (x < this._x) {\n this.x = x;\n }\n if (y < this._y) {\n this.y = y;\n }\n if (z < this._z) {\n this.z = z;\n }\n return this;\n }\n /**\n * Updates the current Vector3 with the maximal coordinate values between its and the given coordinates.\n * Example Playground https://playground.babylonjs.com/#R1F8YU#28\n * @param x defines the x coordinate of the operand\n * @param y defines the y coordinate of the operand\n * @param z defines the z coordinate of the operand\n * @returns the current updated Vector3\n */\n maximizeInPlaceFromFloats(x, y, z) {\n if (x > this._x) {\n this.x = x;\n }\n if (y > this._y) {\n this.y = y;\n }\n if (z > this._z) {\n this.z = z;\n }\n return this;\n }\n /**\n * Due to float precision, scale of a mesh could be uniform but float values are off by a small fraction\n * Check if is non uniform within a certain amount of decimal places to account for this\n * @param epsilon the amount the values can differ\n * @returns if the the vector is non uniform to a certain number of decimal places\n */\n isNonUniformWithinEpsilon(epsilon) {\n const absX = Math.abs(this._x);\n const absY = Math.abs(this._y);\n if (!Scalar.WithinEpsilon(absX, absY, epsilon)) {\n return true;\n }\n const absZ = Math.abs(this._z);\n if (!Scalar.WithinEpsilon(absX, absZ, epsilon)) {\n return true;\n }\n if (!Scalar.WithinEpsilon(absY, absZ, epsilon)) {\n return true;\n }\n return false;\n }\n /**\n * Gets a boolean indicating that the vector is non uniform meaning x, y or z are not all the same\n */\n get isNonUniform() {\n const absX = Math.abs(this._x);\n const absY = Math.abs(this._y);\n if (absX !== absY) {\n return true;\n }\n const absZ = Math.abs(this._z);\n if (absX !== absZ) {\n return true;\n }\n return false;\n }\n /**\n * Gets a new Vector3 from current Vector3 floored values\n * Example Playground https://playground.babylonjs.com/#R1F8YU#22\n * @returns a new Vector3\n */\n floor() {\n return new this.constructor(Math.floor(this._x), Math.floor(this._y), Math.floor(this._z));\n }\n /**\n * Gets a new Vector3 from current Vector3 fractional values\n * Example Playground https://playground.babylonjs.com/#R1F8YU#23\n * @returns a new Vector3\n */\n fract() {\n return new this.constructor(this._x - Math.floor(this._x), this._y - Math.floor(this._y), this._z - Math.floor(this._z));\n }\n // Properties\n /**\n * Gets the length of the Vector3\n * Example Playground https://playground.babylonjs.com/#R1F8YU#25\n * @returns the length of the Vector3\n */\n length() {\n return Math.sqrt(this._x * this._x + this._y * this._y + this._z * this._z);\n }\n /**\n * Gets the squared length of the Vector3\n * Example Playground https://playground.babylonjs.com/#R1F8YU#26\n * @returns squared length of the Vector3\n */\n lengthSquared() {\n return this._x * this._x + this._y * this._y + this._z * this._z;\n }\n /**\n * Gets a boolean indicating if the vector contains a zero in one of its components\n * Example Playground https://playground.babylonjs.com/#R1F8YU#1\n */\n get hasAZeroComponent() {\n return this._x * this._y * this._z === 0;\n }\n /**\n * Normalize the current Vector3.\n * Please note that this is an in place operation.\n * Example Playground https://playground.babylonjs.com/#R1F8YU#122\n * @returns the current updated Vector3\n */\n normalize() {\n return this.normalizeFromLength(this.length());\n }\n /**\n * Reorders the x y z properties of the vector in place\n * Example Playground https://playground.babylonjs.com/#R1F8YU#44\n * @param order new ordering of the properties (eg. for vector 1,2,3 with \"ZYX\" will produce 3,2,1)\n * @returns the current updated vector\n */\n reorderInPlace(order) {\n order = order.toLowerCase();\n if (order === \"xyz\") {\n return this;\n }\n MathTmp.Vector3[0].copyFrom(this);\n [\"x\", \"y\", \"z\"].forEach((val, i) => {\n this[val] = MathTmp.Vector3[0][order[i]];\n });\n return this;\n }\n /**\n * Rotates the vector around 0,0,0 by a quaternion\n * Example Playground https://playground.babylonjs.com/#R1F8YU#47\n * @param quaternion the rotation quaternion\n * @param result vector to store the result\n * @returns the resulting vector\n */\n rotateByQuaternionToRef(quaternion, result) {\n quaternion.toRotationMatrix(MathTmp.Matrix[0]);\n Vector3.TransformCoordinatesToRef(this, MathTmp.Matrix[0], result);\n return result;\n }\n /**\n * Rotates a vector around a given point\n * Example Playground https://playground.babylonjs.com/#R1F8YU#46\n * @param quaternion the rotation quaternion\n * @param point the point to rotate around\n * @param result vector to store the result\n * @returns the resulting vector\n */\n rotateByQuaternionAroundPointToRef(quaternion, point, result) {\n this.subtractToRef(point, MathTmp.Vector3[0]);\n MathTmp.Vector3[0].rotateByQuaternionToRef(quaternion, MathTmp.Vector3[0]);\n point.addToRef(MathTmp.Vector3[0], result);\n return result;\n }\n /**\n * Returns a new Vector3 as the cross product of the current vector and the \"other\" one\n * The cross product is then orthogonal to both current and \"other\"\n * Example Playground https://playground.babylonjs.com/#R1F8YU#14\n * @param other defines the right operand\n * @returns the cross product\n */\n cross(other) {\n const result = new this.constructor();\n return Vector3.CrossToRef(this, other, result);\n }\n /**\n * Normalize the current Vector3 with the given input length.\n * Please note that this is an in place operation.\n * Example Playground https://playground.babylonjs.com/#R1F8YU#123\n * @param len the length of the vector\n * @returns the current updated Vector3\n */\n normalizeFromLength(len) {\n if (len === 0 || len === 1.0) {\n return this;\n }\n return this.scaleInPlace(1.0 / len);\n }\n /**\n * Normalize the current Vector3 to a new vector\n * Example Playground https://playground.babylonjs.com/#R1F8YU#124\n * @returns the new Vector3\n */\n normalizeToNew() {\n const normalized = new this.constructor(0, 0, 0);\n this.normalizeToRef(normalized);\n return normalized;\n }\n /**\n * Normalize the current Vector3 to the reference\n * Example Playground https://playground.babylonjs.com/#R1F8YU#125\n * @param reference define the Vector3 to update\n * @returns the updated Vector3\n */\n normalizeToRef(reference) {\n const len = this.length();\n if (len === 0 || len === 1.0) {\n return reference.copyFromFloats(this._x, this._y, this._z);\n }\n return this.scaleToRef(1.0 / len, reference);\n }\n /**\n * Creates a new Vector3 copied from the current Vector3\n * Example Playground https://playground.babylonjs.com/#R1F8YU#11\n * @returns the new Vector3\n */\n clone() {\n return new this.constructor(this._x, this._y, this._z);\n }\n /**\n * Copies the given vector coordinates to the current Vector3 ones\n * Example Playground https://playground.babylonjs.com/#R1F8YU#12\n * @param source defines the source Vector3\n * @returns the current updated Vector3\n */\n copyFrom(source) {\n return this.copyFromFloats(source._x, source._y, source._z);\n }\n /**\n * Copies the given floats to the current Vector3 coordinates\n * Example Playground https://playground.babylonjs.com/#R1F8YU#13\n * @param x defines the x coordinate of the operand\n * @param y defines the y coordinate of the operand\n * @param z defines the z coordinate of the operand\n * @returns the current updated Vector3\n */\n copyFromFloats(x, y, z) {\n this._x = x;\n this._y = y;\n this._z = z;\n this._isDirty = true;\n return this;\n }\n /**\n * Copies the given floats to the current Vector3 coordinates\n * Example Playground https://playground.babylonjs.com/#R1F8YU#58\n * @param x defines the x coordinate of the operand\n * @param y defines the y coordinate of the operand\n * @param z defines the z coordinate of the operand\n * @returns the current updated Vector3\n */\n set(x, y, z) {\n return this.copyFromFloats(x, y, z);\n }\n /**\n * Copies the given float to the current Vector3 coordinates\n * Example Playground https://playground.babylonjs.com/#R1F8YU#59\n * @param v defines the x, y and z coordinates of the operand\n * @returns the current updated Vector3\n */\n setAll(v) {\n this._x = this._y = this._z = v;\n this._isDirty = true;\n return this;\n }\n // Statics\n /**\n * Get the clip factor between two vectors\n * Example Playground https://playground.babylonjs.com/#R1F8YU#126\n * @param vector0 defines the first operand\n * @param vector1 defines the second operand\n * @param axis defines the axis to use\n * @param size defines the size along the axis\n * @returns the clip factor\n */\n static GetClipFactor(vector0, vector1, axis, size) {\n const d0 = Vector3.Dot(vector0, axis) - size;\n const d1 = Vector3.Dot(vector1, axis) - size;\n const s = d0 / (d0 - d1);\n return s;\n }\n /**\n * Get angle between two vectors\n * Example Playground https://playground.babylonjs.com/#R1F8YU#86\n * @param vector0 the starting point\n * @param vector1 the ending point\n * @param normal direction of the normal\n * @returns the angle between vector0 and vector1\n */\n static GetAngleBetweenVectors(vector0, vector1, normal) {\n const v0 = vector0.normalizeToRef(MathTmp.Vector3[1]);\n const v1 = vector1.normalizeToRef(MathTmp.Vector3[2]);\n let dot = Vector3.Dot(v0, v1);\n // Vectors are normalized so dot will be in [-1, 1] (aside precision issues enough to break the result which explains the below clamp)\n dot = Scalar.Clamp(dot, -1, 1);\n const angle = Math.acos(dot);\n const n = MathTmp.Vector3[3];\n Vector3.CrossToRef(v0, v1, n);\n if (Vector3.Dot(n, normal) > 0) {\n return isNaN(angle) ? 0 : angle;\n }\n return isNaN(angle) ? -Math.PI : -Math.acos(dot);\n }\n /**\n * Get angle between two vectors projected on a plane\n * Example Playground https://playground.babylonjs.com/#R1F8YU#87\n * Expectation compute time: 0.01 ms (median) and 0.02 ms (percentile 95%)\n * @param vector0 angle between vector0 and vector1\n * @param vector1 angle between vector0 and vector1\n * @param normal Normal of the projection plane\n * @returns the angle in radians (float) between vector0 and vector1 projected on the plane with the specified normal\n */\n static GetAngleBetweenVectorsOnPlane(vector0, vector1, normal) {\n MathTmp.Vector3[0].copyFrom(vector0);\n const v0 = MathTmp.Vector3[0];\n MathTmp.Vector3[1].copyFrom(vector1);\n const v1 = MathTmp.Vector3[1];\n MathTmp.Vector3[2].copyFrom(normal);\n const vNormal = MathTmp.Vector3[2];\n const right = MathTmp.Vector3[3];\n const forward = MathTmp.Vector3[4];\n v0.normalize();\n v1.normalize();\n vNormal.normalize();\n Vector3.CrossToRef(vNormal, v0, right);\n Vector3.CrossToRef(right, vNormal, forward);\n const angle = Math.atan2(Vector3.Dot(v1, right), Vector3.Dot(v1, forward));\n return Scalar.NormalizeRadians(angle);\n }\n /**\n * Gets the rotation that aligns the roll axis (Y) to the line joining the start point to the target point and stores it in the ref Vector3\n * Example PG https://playground.babylonjs.com/#R1F8YU#189\n * @param start the starting point\n * @param target the target point\n * @param ref the vector3 to store the result\n * @returns ref in the form (pitch, yaw, 0)\n */\n static PitchYawRollToMoveBetweenPointsToRef(start, target, ref) {\n const diff = TmpVectors.Vector3[0];\n target.subtractToRef(start, diff);\n ref._y = Math.atan2(diff.x, diff.z) || 0;\n ref._x = Math.atan2(Math.sqrt(diff.x ** 2 + diff.z ** 2), diff.y) || 0;\n ref._z = 0;\n ref._isDirty = true;\n return ref;\n }\n /**\n * Gets the rotation that aligns the roll axis (Y) to the line joining the start point to the target point\n * Example PG https://playground.babylonjs.com/#R1F8YU#188\n * @param start the starting point\n * @param target the target point\n * @returns the rotation in the form (pitch, yaw, 0)\n */\n static PitchYawRollToMoveBetweenPoints(start, target) {\n const ref = Vector3.Zero();\n return Vector3.PitchYawRollToMoveBetweenPointsToRef(start, target, ref);\n }\n /**\n * Slerp between two vectors. See also `SmoothToRef`\n * Slerp is a spherical linear interpolation\n * giving a slow in and out effect\n * Example Playground 1 https://playground.babylonjs.com/#R1F8YU#108\n * Example Playground 2 https://playground.babylonjs.com/#R1F8YU#109\n * @param vector0 Start vector\n * @param vector1 End vector\n * @param slerp amount (will be clamped between 0 and 1)\n * @param result The slerped vector\n */\n static SlerpToRef(vector0, vector1, slerp, result) {\n slerp = Scalar.Clamp(slerp, 0, 1);\n const vector0Dir = MathTmp.Vector3[0];\n const vector1Dir = MathTmp.Vector3[1];\n vector0Dir.copyFrom(vector0);\n const vector0Length = vector0Dir.length();\n vector0Dir.normalizeFromLength(vector0Length);\n vector1Dir.copyFrom(vector1);\n const vector1Length = vector1Dir.length();\n vector1Dir.normalizeFromLength(vector1Length);\n const dot = Vector3.Dot(vector0Dir, vector1Dir);\n let scale0;\n let scale1;\n if (dot < 1 - Epsilon) {\n const omega = Math.acos(dot);\n const invSin = 1 / Math.sin(omega);\n scale0 = Math.sin((1 - slerp) * omega) * invSin;\n scale1 = Math.sin(slerp * omega) * invSin;\n }\n else {\n // Use linear interpolation\n scale0 = 1 - slerp;\n scale1 = slerp;\n }\n vector0Dir.scaleInPlace(scale0);\n vector1Dir.scaleInPlace(scale1);\n result.copyFrom(vector0Dir).addInPlace(vector1Dir);\n result.scaleInPlace(Scalar.Lerp(vector0Length, vector1Length, slerp));\n return result;\n }\n /**\n * Smooth interpolation between two vectors using Slerp\n * Example Playground https://playground.babylonjs.com/#R1F8YU#110\n * @param source source vector\n * @param goal goal vector\n * @param deltaTime current interpolation frame\n * @param lerpTime total interpolation time\n * @param result the smoothed vector\n */\n static SmoothToRef(source, goal, deltaTime, lerpTime, result) {\n Vector3.SlerpToRef(source, goal, lerpTime === 0 ? 1 : deltaTime / lerpTime, result);\n return result;\n }\n /**\n * Returns a new Vector3 set from the index \"offset\" of the given array\n * Example Playground https://playground.babylonjs.com/#R1F8YU#83\n * @param array defines the source array\n * @param offset defines the offset in the source array\n * @returns the new Vector3\n */\n static FromArray(array, offset = 0) {\n return new Vector3(array[offset], array[offset + 1], array[offset + 2]);\n }\n /**\n * Returns a new Vector3 set from the index \"offset\" of the given Float32Array\n * @param array defines the source array\n * @param offset defines the offset in the source array\n * @returns the new Vector3\n * @deprecated Please use FromArray instead.\n */\n static FromFloatArray(array, offset) {\n return Vector3.FromArray(array, offset);\n }\n /**\n * Sets the given vector \"result\" with the element values from the index \"offset\" of the given array\n * Example Playground https://playground.babylonjs.com/#R1F8YU#84\n * @param array defines the source array\n * @param offset defines the offset in the source array\n * @param result defines the Vector3 where to store the result\n * @returns result input\n */\n static FromArrayToRef(array, offset, result) {\n result._x = array[offset];\n result._y = array[offset + 1];\n result._z = array[offset + 2];\n result._isDirty = true;\n return result;\n }\n /**\n * Sets the given vector \"result\" with the element values from the index \"offset\" of the given Float32Array\n * @param array defines the source array\n * @param offset defines the offset in the source array\n * @param result defines the Vector3 where to store the result\n * @deprecated Please use FromArrayToRef instead.\n */\n static FromFloatArrayToRef(array, offset, result) {\n return Vector3.FromArrayToRef(array, offset, result);\n }\n /**\n * Sets the given vector \"result\" with the given floats.\n * Example Playground https://playground.babylonjs.com/#R1F8YU#85\n * @param x defines the x coordinate of the source\n * @param y defines the y coordinate of the source\n * @param z defines the z coordinate of the source\n * @param result defines the Vector3 where to store the result\n */\n static FromFloatsToRef(x, y, z, result) {\n result.copyFromFloats(x, y, z);\n return result;\n }\n /**\n * Returns a new Vector3 set to (0.0, 0.0, 0.0)\n * @returns a new empty Vector3\n */\n static Zero() {\n return new Vector3(0.0, 0.0, 0.0);\n }\n /**\n * Returns a new Vector3 set to (1.0, 1.0, 1.0)\n * @returns a new Vector3\n */\n static One() {\n return new Vector3(1.0, 1.0, 1.0);\n }\n /**\n * Returns a new Vector3 set to (0.0, 1.0, 0.0)\n * Example Playground https://playground.babylonjs.com/#R1F8YU#71\n * @returns a new up Vector3\n */\n static Up() {\n return new Vector3(0.0, 1.0, 0.0);\n }\n /**\n * Gets an up Vector3 that must not be updated\n */\n static get UpReadOnly() {\n return Vector3._UpReadOnly;\n }\n /**\n * Gets a down Vector3 that must not be updated\n */\n static get DownReadOnly() {\n return Vector3._DownReadOnly;\n }\n /**\n * Gets a right Vector3 that must not be updated\n */\n static get RightReadOnly() {\n return Vector3._RightReadOnly;\n }\n /**\n * Gets a left Vector3 that must not be updated\n */\n static get LeftReadOnly() {\n return Vector3._LeftReadOnly;\n }\n /**\n * Gets a forward Vector3 that must not be updated\n */\n static get LeftHandedForwardReadOnly() {\n return Vector3._LeftHandedForwardReadOnly;\n }\n /**\n * Gets a forward Vector3 that must not be updated\n */\n static get RightHandedForwardReadOnly() {\n return Vector3._RightHandedForwardReadOnly;\n }\n /**\n * Gets a backward Vector3 that must not be updated\n */\n static get LeftHandedBackwardReadOnly() {\n return Vector3._LeftHandedBackwardReadOnly;\n }\n /**\n * Gets a backward Vector3 that must not be updated\n */\n static get RightHandedBackwardReadOnly() {\n return Vector3._RightHandedBackwardReadOnly;\n }\n /**\n * Gets a zero Vector3 that must not be updated\n */\n static get ZeroReadOnly() {\n return Vector3._ZeroReadOnly;\n }\n /**\n * Gets a one Vector3 that must not be updated\n */\n static get OneReadOnly() {\n return Vector3._OneReadOnly;\n }\n /**\n * Returns a new Vector3 set to (0.0, -1.0, 0.0)\n * Example Playground https://playground.babylonjs.com/#R1F8YU#71\n * @returns a new down Vector3\n */\n static Down() {\n return new Vector3(0.0, -1.0, 0.0);\n }\n /**\n * Returns a new Vector3 set to (0.0, 0.0, 1.0)\n * Example Playground https://playground.babylonjs.com/#R1F8YU#71\n * @param rightHandedSystem is the scene right-handed (negative z)\n * @returns a new forward Vector3\n */\n static Forward(rightHandedSystem = false) {\n return new Vector3(0.0, 0.0, rightHandedSystem ? -1.0 : 1.0);\n }\n /**\n * Returns a new Vector3 set to (0.0, 0.0, -1.0)\n * Example Playground https://playground.babylonjs.com/#R1F8YU#71\n * @param rightHandedSystem is the scene right-handed (negative-z)\n * @returns a new Backward Vector3\n */\n static Backward(rightHandedSystem = false) {\n return new Vector3(0.0, 0.0, rightHandedSystem ? 1.0 : -1.0);\n }\n /**\n * Returns a new Vector3 set to (1.0, 0.0, 0.0)\n * Example Playground https://playground.babylonjs.com/#R1F8YU#71\n * @returns a new right Vector3\n */\n static Right() {\n return new Vector3(1.0, 0.0, 0.0);\n }\n /**\n * Returns a new Vector3 set to (-1.0, 0.0, 0.0)\n * Example Playground https://playground.babylonjs.com/#R1F8YU#71\n * @returns a new left Vector3\n */\n static Left() {\n return new Vector3(-1.0, 0.0, 0.0);\n }\n /**\n * Returns a new Vector3 with random values between min and max\n * @param min the minimum random value\n * @param max the maximum random value\n * @returns a Vector3 with random values between min and max\n */\n static Random(min = 0, max = 1) {\n return new Vector3(Scalar.RandomRange(min, max), Scalar.RandomRange(min, max), Scalar.RandomRange(min, max));\n }\n /**\n * Returns a new Vector3 set with the result of the transformation by the given matrix of the given vector.\n * This method computes transformed coordinates only, not transformed direction vectors (ie. it takes translation in account)\n * Example Playground https://playground.babylonjs.com/#R1F8YU#111\n * @param vector defines the Vector3 to transform\n * @param transformation defines the transformation matrix\n * @returns the transformed Vector3\n */\n static TransformCoordinates(vector, transformation) {\n const result = Vector3.Zero();\n Vector3.TransformCoordinatesToRef(vector, transformation, result);\n return result;\n }\n /**\n * Sets the given vector \"result\" coordinates with the result of the transformation by the given matrix of the given vector\n * This method computes transformed coordinates only, not transformed direction vectors (ie. it takes translation in account)\n * Example Playground https://playground.babylonjs.com/#R1F8YU#113\n * @param vector defines the Vector3 to transform\n * @param transformation defines the transformation matrix\n * @param result defines the Vector3 where to store the result\n * @returns result input\n */\n static TransformCoordinatesToRef(vector, transformation, result) {\n Vector3.TransformCoordinatesFromFloatsToRef(vector._x, vector._y, vector._z, transformation, result);\n return result;\n }\n /**\n * Sets the given vector \"result\" coordinates with the result of the transformation by the given matrix of the given floats (x, y, z)\n * This method computes transformed coordinates only, not transformed direction vectors\n * Example Playground https://playground.babylonjs.com/#R1F8YU#115\n * @param x define the x coordinate of the source vector\n * @param y define the y coordinate of the source vector\n * @param z define the z coordinate of the source vector\n * @param transformation defines the transformation matrix\n * @param result defines the Vector3 where to store the result\n * @returns result input\n */\n static TransformCoordinatesFromFloatsToRef(x, y, z, transformation, result) {\n const m = transformation.m;\n const rx = x * m[0] + y * m[4] + z * m[8] + m[12];\n const ry = x * m[1] + y * m[5] + z * m[9] + m[13];\n const rz = x * m[2] + y * m[6] + z * m[10] + m[14];\n const rw = 1 / (x * m[3] + y * m[7] + z * m[11] + m[15]);\n result._x = rx * rw;\n result._y = ry * rw;\n result._z = rz * rw;\n result._isDirty = true;\n return result;\n }\n /**\n * Returns a new Vector3 set with the result of the normal transformation by the given matrix of the given vector\n * This methods computes transformed normalized direction vectors only (ie. it does not apply translation)\n * Example Playground https://playground.babylonjs.com/#R1F8YU#112\n * @param vector defines the Vector3 to transform\n * @param transformation defines the transformation matrix\n * @returns the new Vector3\n */\n static TransformNormal(vector, transformation) {\n const result = Vector3.Zero();\n Vector3.TransformNormalToRef(vector, transformation, result);\n return result;\n }\n /**\n * Sets the given vector \"result\" with the result of the normal transformation by the given matrix of the given vector\n * This methods computes transformed normalized direction vectors only (ie. it does not apply translation)\n * Example Playground https://playground.babylonjs.com/#R1F8YU#114\n * @param vector defines the Vector3 to transform\n * @param transformation defines the transformation matrix\n * @param result defines the Vector3 where to store the result\n * @returns result input\n */\n static TransformNormalToRef(vector, transformation, result) {\n this.TransformNormalFromFloatsToRef(vector._x, vector._y, vector._z, transformation, result);\n return result;\n }\n /**\n * Sets the given vector \"result\" with the result of the normal transformation by the given matrix of the given floats (x, y, z)\n * This methods computes transformed normalized direction vectors only (ie. it does not apply translation)\n * Example Playground https://playground.babylonjs.com/#R1F8YU#116\n * @param x define the x coordinate of the source vector\n * @param y define the y coordinate of the source vector\n * @param z define the z coordinate of the source vector\n * @param transformation defines the transformation matrix\n * @param result defines the Vector3 where to store the result\n * @returns result input\n */\n static TransformNormalFromFloatsToRef(x, y, z, transformation, result) {\n const m = transformation.m;\n result._x = x * m[0] + y * m[4] + z * m[8];\n result._y = x * m[1] + y * m[5] + z * m[9];\n result._z = x * m[2] + y * m[6] + z * m[10];\n result._isDirty = true;\n return result;\n }\n /**\n * Returns a new Vector3 located for \"amount\" on the CatmullRom interpolation spline defined by the vectors \"value1\", \"value2\", \"value3\", \"value4\"\n * Example Playground https://playground.babylonjs.com/#R1F8YU#69\n * @param value1 defines the first control point\n * @param value2 defines the second control point\n * @param value3 defines the third control point\n * @param value4 defines the fourth control point\n * @param amount defines the amount on the spline to use\n * @returns the new Vector3\n */\n static CatmullRom(value1, value2, value3, value4, amount) {\n const squared = amount * amount;\n const cubed = amount * squared;\n const x = 0.5 *\n (2.0 * value2._x +\n (-value1._x + value3._x) * amount +\n (2.0 * value1._x - 5.0 * value2._x + 4.0 * value3._x - value4._x) * squared +\n (-value1._x + 3.0 * value2._x - 3.0 * value3._x + value4._x) * cubed);\n const y = 0.5 *\n (2.0 * value2._y +\n (-value1._y + value3._y) * amount +\n (2.0 * value1._y - 5.0 * value2._y + 4.0 * value3._y - value4._y) * squared +\n (-value1._y + 3.0 * value2._y - 3.0 * value3._y + value4._y) * cubed);\n const z = 0.5 *\n (2.0 * value2._z +\n (-value1._z + value3._z) * amount +\n (2.0 * value1._z - 5.0 * value2._z + 4.0 * value3._z - value4._z) * squared +\n (-value1._z + 3.0 * value2._z - 3.0 * value3._z + value4._z) * cubed);\n return new value1.constructor(x, y, z);\n }\n /**\n * Returns a new Vector3 set with the coordinates of \"value\", if the vector \"value\" is in the cube defined by the vectors \"min\" and \"max\"\n * If a coordinate value of \"value\" is lower than one of the \"min\" coordinate, then this \"value\" coordinate is set with the \"min\" one\n * If a coordinate value of \"value\" is greater than one of the \"max\" coordinate, then this \"value\" coordinate is set with the \"max\" one\n * Example Playground https://playground.babylonjs.com/#R1F8YU#76\n * @param value defines the current value\n * @param min defines the lower range value\n * @param max defines the upper range value\n * @returns the new Vector3\n */\n static Clamp(value, min, max) {\n const result = new value.constructor();\n Vector3.ClampToRef(value, min, max, result);\n return result;\n }\n /**\n * Sets the given vector \"result\" with the coordinates of \"value\", if the vector \"value\" is in the cube defined by the vectors \"min\" and \"max\"\n * If a coordinate value of \"value\" is lower than one of the \"min\" coordinate, then this \"value\" coordinate is set with the \"min\" one\n * If a coordinate value of \"value\" is greater than one of the \"max\" coordinate, then this \"value\" coordinate is set with the \"max\" one\n * Example Playground https://playground.babylonjs.com/#R1F8YU#77\n * @param value defines the current value\n * @param min defines the lower range value\n * @param max defines the upper range value\n * @param result defines the Vector3 where to store the result\n * @returns result input\n */\n static ClampToRef(value, min, max, result) {\n let x = value._x;\n x = x > max._x ? max._x : x;\n x = x < min._x ? min._x : x;\n let y = value._y;\n y = y > max._y ? max._y : y;\n y = y < min._y ? min._y : y;\n let z = value._z;\n z = z > max._z ? max._z : z;\n z = z < min._z ? min._z : z;\n result.copyFromFloats(x, y, z);\n return result;\n }\n /**\n * Checks if a given vector is inside a specific range\n * Example Playground https://playground.babylonjs.com/#R1F8YU#75\n * @param v defines the vector to test\n * @param min defines the minimum range\n * @param max defines the maximum range\n */\n static CheckExtends(v, min, max) {\n min.minimizeInPlace(v);\n max.maximizeInPlace(v);\n }\n /**\n * Returns a new Vector3 located for \"amount\" (float) on the Hermite interpolation spline defined by the vectors \"value1\", \"tangent1\", \"value2\", \"tangent2\"\n * Example Playground https://playground.babylonjs.com/#R1F8YU#89\n * @param value1 defines the first control point\n * @param tangent1 defines the first tangent vector\n * @param value2 defines the second control point\n * @param tangent2 defines the second tangent vector\n * @param amount defines the amount on the interpolation spline (between 0 and 1)\n * @returns the new Vector3\n */\n static Hermite(value1, tangent1, value2, tangent2, amount) {\n const squared = amount * amount;\n const cubed = amount * squared;\n const part1 = 2.0 * cubed - 3.0 * squared + 1.0;\n const part2 = -2.0 * cubed + 3.0 * squared;\n const part3 = cubed - 2.0 * squared + amount;\n const part4 = cubed - squared;\n const x = value1._x * part1 + value2._x * part2 + tangent1._x * part3 + tangent2._x * part4;\n const y = value1._y * part1 + value2._y * part2 + tangent1._y * part3 + tangent2._y * part4;\n const z = value1._z * part1 + value2._z * part2 + tangent1._z * part3 + tangent2._z * part4;\n return new value1.constructor(x, y, z);\n }\n /**\n * Returns a new Vector3 which is the 1st derivative of the Hermite spline defined by the vectors \"value1\", \"value2\", \"tangent1\", \"tangent2\".\n * Example Playground https://playground.babylonjs.com/#R1F8YU#90\n * @param value1 defines the first control point\n * @param tangent1 defines the first tangent\n * @param value2 defines the second control point\n * @param tangent2 defines the second tangent\n * @param time define where the derivative must be done\n * @returns 1st derivative\n */\n static Hermite1stDerivative(value1, tangent1, value2, tangent2, time) {\n const result = new value1.constructor();\n this.Hermite1stDerivativeToRef(value1, tangent1, value2, tangent2, time, result);\n return result;\n }\n /**\n * Update a Vector3 with the 1st derivative of the Hermite spline defined by the vectors \"value1\", \"value2\", \"tangent1\", \"tangent2\".\n * Example Playground https://playground.babylonjs.com/#R1F8YU#91\n * @param value1 defines the first control point\n * @param tangent1 defines the first tangent\n * @param value2 defines the second control point\n * @param tangent2 defines the second tangent\n * @param time define where the derivative must be done\n * @param result define where to store the derivative\n * @returns result input\n */\n static Hermite1stDerivativeToRef(value1, tangent1, value2, tangent2, time, result) {\n const t2 = time * time;\n result._x = (t2 - time) * 6 * value1._x + (3 * t2 - 4 * time + 1) * tangent1._x + (-t2 + time) * 6 * value2._x + (3 * t2 - 2 * time) * tangent2._x;\n result._y = (t2 - time) * 6 * value1._y + (3 * t2 - 4 * time + 1) * tangent1._y + (-t2 + time) * 6 * value2._y + (3 * t2 - 2 * time) * tangent2._y;\n result._z = (t2 - time) * 6 * value1._z + (3 * t2 - 4 * time + 1) * tangent1._z + (-t2 + time) * 6 * value2._z + (3 * t2 - 2 * time) * tangent2._z;\n result._isDirty = true;\n return result;\n }\n /**\n * Returns a new Vector3 located for \"amount\" (float) on the linear interpolation between the vectors \"start\" and \"end\"\n * Example Playground https://playground.babylonjs.com/#R1F8YU#95\n * @param start defines the start value\n * @param end defines the end value\n * @param amount max defines amount between both (between 0 and 1)\n * @returns the new Vector3\n */\n static Lerp(start, end, amount) {\n const result = new start.constructor(0, 0, 0);\n Vector3.LerpToRef(start, end, amount, result);\n return result;\n }\n /**\n * Sets the given vector \"result\" with the result of the linear interpolation from the vector \"start\" for \"amount\" to the vector \"end\"\n * Example Playground https://playground.babylonjs.com/#R1F8YU#93\n * @param start defines the start value\n * @param end defines the end value\n * @param amount max defines amount between both (between 0 and 1)\n * @param result defines the Vector3 where to store the result\n * @returns result input\n */\n static LerpToRef(start, end, amount, result) {\n result._x = start._x + (end._x - start._x) * amount;\n result._y = start._y + (end._y - start._y) * amount;\n result._z = start._z + (end._z - start._z) * amount;\n result._isDirty = true;\n return result;\n }\n /**\n * Returns the dot product (float) between the vectors \"left\" and \"right\"\n * Example Playground https://playground.babylonjs.com/#R1F8YU#82\n * @param left defines the left operand\n * @param right defines the right operand\n * @returns the dot product\n */\n static Dot(left, right) {\n return left._x * right._x + left._y * right._y + left._z * right._z;\n }\n /**\n * Returns a new Vector3 as the cross product of the vectors \"left\" and \"right\"\n * The cross product is then orthogonal to both \"left\" and \"right\"\n * Example Playground https://playground.babylonjs.com/#R1F8YU#15\n * @param left defines the left operand\n * @param right defines the right operand\n * @returns the cross product\n */\n static Cross(left, right) {\n const result = new left.constructor();\n Vector3.CrossToRef(left, right, result);\n return result;\n }\n /**\n * Sets the given vector \"result\" with the cross product of \"left\" and \"right\"\n * The cross product is then orthogonal to both \"left\" and \"right\"\n * Example Playground https://playground.babylonjs.com/#R1F8YU#78\n * @param left defines the left operand\n * @param right defines the right operand\n * @param result defines the Vector3 where to store the result\n * @returns result input\n */\n static CrossToRef(left, right, result) {\n const x = left._y * right._z - left._z * right._y;\n const y = left._z * right._x - left._x * right._z;\n const z = left._x * right._y - left._y * right._x;\n result.copyFromFloats(x, y, z);\n return result;\n }\n /**\n * Returns a new Vector3 as the normalization of the given vector\n * Example Playground https://playground.babylonjs.com/#R1F8YU#98\n * @param vector defines the Vector3 to normalize\n * @returns the new Vector3\n */\n static Normalize(vector) {\n const result = Vector3.Zero();\n Vector3.NormalizeToRef(vector, result);\n return result;\n }\n /**\n * Sets the given vector \"result\" with the normalization of the given first vector\n * Example Playground https://playground.babylonjs.com/#R1F8YU#98\n * @param vector defines the Vector3 to normalize\n * @param result defines the Vector3 where to store the result\n * @returns result input\n */\n static NormalizeToRef(vector, result) {\n vector.normalizeToRef(result);\n return result;\n }\n /**\n * Project a Vector3 onto screen space\n * Example Playground https://playground.babylonjs.com/#R1F8YU#101\n * @param vector defines the Vector3 to project\n * @param world defines the world matrix to use\n * @param transform defines the transform (view x projection) matrix to use\n * @param viewport defines the screen viewport to use\n * @returns the new Vector3\n */\n static Project(vector, world, transform, viewport) {\n const result = new vector.constructor();\n Vector3.ProjectToRef(vector, world, transform, viewport, result);\n return result;\n }\n /**\n * Project a Vector3 onto screen space to reference\n * Example Playground https://playground.babylonjs.com/#R1F8YU#102\n * @param vector defines the Vector3 to project\n * @param world defines the world matrix to use\n * @param transform defines the transform (view x projection) matrix to use\n * @param viewport defines the screen viewport to use\n * @param result the vector in which the screen space will be stored\n * @returns result input\n */\n static ProjectToRef(vector, world, transform, viewport, result) {\n const cw = viewport.width;\n const ch = viewport.height;\n const cx = viewport.x;\n const cy = viewport.y;\n const viewportMatrix = MathTmp.Matrix[1];\n Matrix.FromValuesToRef(cw / 2.0, 0, 0, 0, 0, -ch / 2.0, 0, 0, 0, 0, 0.5, 0, cx + cw / 2.0, ch / 2.0 + cy, 0.5, 1, viewportMatrix);\n const matrix = MathTmp.Matrix[0];\n world.multiplyToRef(transform, matrix);\n matrix.multiplyToRef(viewportMatrix, matrix);\n Vector3.TransformCoordinatesToRef(vector, matrix, result);\n return result;\n }\n /**\n * Reflects a vector off the plane defined by a normalized normal\n * @param inDirection defines the vector direction\n * @param normal defines the normal - Must be normalized\n * @returns the resulting vector\n */\n static Reflect(inDirection, normal) {\n return this.ReflectToRef(inDirection, normal, new Vector3());\n }\n /**\n * Reflects a vector off the plane defined by a normalized normal to reference\n * @param inDirection defines the vector direction\n * @param normal defines the normal - Must be normalized\n * @param result defines the Vector3 where to store the result\n * @returns the resulting vector\n */\n static ReflectToRef(inDirection, normal, ref) {\n const tmp = TmpVectors.Vector3[0];\n tmp.copyFrom(normal).scaleInPlace(2 * Vector3.Dot(inDirection, normal));\n return ref.copyFrom(inDirection).subtractInPlace(tmp);\n }\n /**\n * @internal\n */\n static _UnprojectFromInvertedMatrixToRef(source, matrix, result) {\n Vector3.TransformCoordinatesToRef(source, matrix, result);\n const m = matrix.m;\n const num = source._x * m[3] + source._y * m[7] + source._z * m[11] + m[15];\n if (Scalar.WithinEpsilon(num, 1.0)) {\n result.scaleInPlace(1.0 / num);\n }\n return result;\n }\n /**\n * Unproject from screen space to object space\n * Example Playground https://playground.babylonjs.com/#R1F8YU#121\n * @param source defines the screen space Vector3 to use\n * @param viewportWidth defines the current width of the viewport\n * @param viewportHeight defines the current height of the viewport\n * @param world defines the world matrix to use (can be set to Identity to go to world space)\n * @param transform defines the transform (view x projection) matrix to use\n * @returns the new Vector3\n */\n static UnprojectFromTransform(source, viewportWidth, viewportHeight, world, transform) {\n return this.Unproject(source, viewportWidth, viewportHeight, world, transform, Matrix.IdentityReadOnly);\n }\n /**\n * Unproject from screen space to object space\n * Example Playground https://playground.babylonjs.com/#R1F8YU#117\n * @param source defines the screen space Vector3 to use\n * @param viewportWidth defines the current width of the viewport\n * @param viewportHeight defines the current height of the viewport\n * @param world defines the world matrix to use (can be set to Identity to go to world space)\n * @param view defines the view matrix to use\n * @param projection defines the projection matrix to use\n * @returns the new Vector3\n */\n static Unproject(source, viewportWidth, viewportHeight, world, view, projection) {\n const result = new source.constructor();\n Vector3.UnprojectToRef(source, viewportWidth, viewportHeight, world, view, projection, result);\n return result;\n }\n /**\n * Unproject from screen space to object space\n * Example Playground https://playground.babylonjs.com/#R1F8YU#119\n * @param source defines the screen space Vector3 to use\n * @param viewportWidth defines the current width of the viewport\n * @param viewportHeight defines the current height of the viewport\n * @param world defines the world matrix to use (can be set to Identity to go to world space)\n * @param view defines the view matrix to use\n * @param projection defines the projection matrix to use\n * @param result defines the Vector3 where to store the result\n * @returns result input\n */\n static UnprojectToRef(source, viewportWidth, viewportHeight, world, view, projection, result) {\n Vector3.UnprojectFloatsToRef(source._x, source._y, source._z, viewportWidth, viewportHeight, world, view, projection, result);\n return result;\n }\n /**\n * Unproject from screen space to object space\n * Example Playground https://playground.babylonjs.com/#R1F8YU#120\n * @param sourceX defines the screen space x coordinate to use\n * @param sourceY defines the screen space y coordinate to use\n * @param sourceZ defines the screen space z coordinate to use\n * @param viewportWidth defines the current width of the viewport\n * @param viewportHeight defines the current height of the viewport\n * @param world defines the world matrix to use (can be set to Identity to go to world space)\n * @param view defines the view matrix to use\n * @param projection defines the projection matrix to use\n * @param result defines the Vector3 where to store the result\n * @returns result input\n */\n static UnprojectFloatsToRef(sourceX, sourceY, sourceZ, viewportWidth, viewportHeight, world, view, projection, result) {\n var _a;\n const matrix = MathTmp.Matrix[0];\n world.multiplyToRef(view, matrix);\n matrix.multiplyToRef(projection, matrix);\n matrix.invert();\n const screenSource = MathTmp.Vector3[0];\n screenSource.x = (sourceX / viewportWidth) * 2 - 1;\n screenSource.y = -((sourceY / viewportHeight) * 2 - 1);\n if ((_a = EngineStore.LastCreatedEngine) === null || _a === void 0 ? void 0 : _a.isNDCHalfZRange) {\n screenSource.z = sourceZ;\n }\n else {\n screenSource.z = 2 * sourceZ - 1.0;\n }\n Vector3._UnprojectFromInvertedMatrixToRef(screenSource, matrix, result);\n return result;\n }\n /**\n * Gets the minimal coordinate values between two Vector3\n * Example Playground https://playground.babylonjs.com/#R1F8YU#97\n * @param left defines the first operand\n * @param right defines the second operand\n * @returns the new Vector3\n */\n static Minimize(left, right) {\n const min = new left.constructor();\n min.copyFrom(left);\n min.minimizeInPlace(right);\n return min;\n }\n /**\n * Gets the maximal coordinate values between two Vector3\n * Example Playground https://playground.babylonjs.com/#R1F8YU#96\n * @param left defines the first operand\n * @param right defines the second operand\n * @returns the new Vector3\n */\n static Maximize(left, right) {\n const max = new left.constructor();\n max.copyFrom(left);\n max.maximizeInPlace(right);\n return max;\n }\n /**\n * Returns the distance between the vectors \"value1\" and \"value2\"\n * Example Playground https://playground.babylonjs.com/#R1F8YU#81\n * @param value1 defines the first operand\n * @param value2 defines the second operand\n * @returns the distance\n */\n static Distance(value1, value2) {\n return Math.sqrt(Vector3.DistanceSquared(value1, value2));\n }\n /**\n * Returns the squared distance between the vectors \"value1\" and \"value2\"\n * Example Playground https://playground.babylonjs.com/#R1F8YU#80\n * @param value1 defines the first operand\n * @param value2 defines the second operand\n * @returns the squared distance\n */\n static DistanceSquared(value1, value2) {\n const x = value1._x - value2._x;\n const y = value1._y - value2._y;\n const z = value1._z - value2._z;\n return x * x + y * y + z * z;\n }\n /**\n * Projects \"vector\" on the triangle determined by its extremities \"p0\", \"p1\" and \"p2\", stores the result in \"ref\"\n * and returns the distance to the projected point.\n * Example Playground https://playground.babylonjs.com/#R1F8YU#104\n * From http://citeseerx.ist.psu.edu/viewdoc/download?doi=10.1.1.104.4264&rep=rep1&type=pdf\n *\n * @param vector the vector to get distance from\n * @param p0 extremity of the triangle\n * @param p1 extremity of the triangle\n * @param p2 extremity of the triangle\n * @param ref variable to store the result to\n * @returns The distance between \"ref\" and \"vector\"\n */\n static ProjectOnTriangleToRef(vector, p0, p1, p2, ref) {\n const p1p0 = MathTmp.Vector3[0];\n const p2p0 = MathTmp.Vector3[1];\n const p2p1 = MathTmp.Vector3[2];\n const normal = MathTmp.Vector3[3];\n const vectorp0 = MathTmp.Vector3[4];\n // Triangle vectors\n p1.subtractToRef(p0, p1p0);\n p2.subtractToRef(p0, p2p0);\n p2.subtractToRef(p1, p2p1);\n const p1p0L = p1p0.length();\n const p2p0L = p2p0.length();\n const p2p1L = p2p1.length();\n if (p1p0L < Epsilon || p2p0L < Epsilon || p2p1L < Epsilon) {\n // This is a degenerate triangle. As we assume this is part of a non-degenerate mesh,\n // we will find a better intersection later.\n // Let's just return one of the extremities\n ref.copyFrom(p0);\n return Vector3.Distance(vector, p0);\n }\n // Compute normal and vector to p0\n vector.subtractToRef(p0, vectorp0);\n Vector3.CrossToRef(p1p0, p2p0, normal);\n const nl = normal.length();\n if (nl < Epsilon) {\n // Extremities are aligned, we are back on the case of a degenerate triangle\n ref.copyFrom(p0);\n return Vector3.Distance(vector, p0);\n }\n normal.normalizeFromLength(nl);\n let l = vectorp0.length();\n if (l < Epsilon) {\n // Vector is p0\n ref.copyFrom(p0);\n return 0;\n }\n vectorp0.normalizeFromLength(l);\n // Project to \"proj\" that lies on the triangle plane\n const cosA = Vector3.Dot(normal, vectorp0);\n const projVector = MathTmp.Vector3[5];\n const proj = MathTmp.Vector3[6];\n projVector.copyFrom(normal).scaleInPlace(-l * cosA);\n proj.copyFrom(vector).addInPlace(projVector);\n // Compute barycentric coordinates (v0, v1 and v2 are axis from barycenter to extremities)\n const v0 = MathTmp.Vector3[4];\n const v1 = MathTmp.Vector3[5];\n const v2 = MathTmp.Vector3[7];\n const tmp = MathTmp.Vector3[8];\n v0.copyFrom(p1p0).scaleInPlace(1 / p1p0L);\n tmp.copyFrom(p2p0).scaleInPlace(1 / p2p0L);\n v0.addInPlace(tmp).scaleInPlace(-1);\n v1.copyFrom(p1p0).scaleInPlace(-1 / p1p0L);\n tmp.copyFrom(p2p1).scaleInPlace(1 / p2p1L);\n v1.addInPlace(tmp).scaleInPlace(-1);\n v2.copyFrom(p2p1).scaleInPlace(-1 / p2p1L);\n tmp.copyFrom(p2p0).scaleInPlace(-1 / p2p0L);\n v2.addInPlace(tmp).scaleInPlace(-1);\n // Determines which edge of the triangle is closest to \"proj\"\n const projP = MathTmp.Vector3[9];\n let dot;\n projP.copyFrom(proj).subtractInPlace(p0);\n Vector3.CrossToRef(v0, projP, tmp);\n dot = Vector3.Dot(tmp, normal);\n const s0 = dot;\n projP.copyFrom(proj).subtractInPlace(p1);\n Vector3.CrossToRef(v1, projP, tmp);\n dot = Vector3.Dot(tmp, normal);\n const s1 = dot;\n projP.copyFrom(proj).subtractInPlace(p2);\n Vector3.CrossToRef(v2, projP, tmp);\n dot = Vector3.Dot(tmp, normal);\n const s2 = dot;\n const edge = MathTmp.Vector3[10];\n let e0, e1;\n if (s0 > 0 && s1 < 0) {\n edge.copyFrom(p1p0);\n e0 = p0;\n e1 = p1;\n }\n else if (s1 > 0 && s2 < 0) {\n edge.copyFrom(p2p1);\n e0 = p1;\n e1 = p2;\n }\n else {\n edge.copyFrom(p2p0).scaleInPlace(-1);\n e0 = p2;\n e1 = p0;\n }\n // Determines if \"proj\" lies inside the triangle\n const tmp2 = MathTmp.Vector3[9];\n const tmp3 = MathTmp.Vector3[4];\n e0.subtractToRef(proj, tmp);\n e1.subtractToRef(proj, tmp2);\n Vector3.CrossToRef(tmp, tmp2, tmp3);\n const isOutside = Vector3.Dot(tmp3, normal) < 0;\n // If inside, we already found the projected point, \"proj\"\n if (!isOutside) {\n ref.copyFrom(proj);\n return Math.abs(l * cosA);\n }\n // If outside, we find \"triProj\", the closest point from \"proj\" on the closest edge\n const r = MathTmp.Vector3[5];\n Vector3.CrossToRef(edge, tmp3, r);\n r.normalize();\n const e0proj = MathTmp.Vector3[9];\n e0proj.copyFrom(e0).subtractInPlace(proj);\n const e0projL = e0proj.length();\n if (e0projL < Epsilon) {\n // Proj is e0\n ref.copyFrom(e0);\n return Vector3.Distance(vector, e0);\n }\n e0proj.normalizeFromLength(e0projL);\n const cosG = Vector3.Dot(r, e0proj);\n const triProj = MathTmp.Vector3[7];\n triProj.copyFrom(proj).addInPlace(r.scaleInPlace(e0projL * cosG));\n // Now we clamp \"triProj\" so it lies between e0 and e1\n tmp.copyFrom(triProj).subtractInPlace(e0);\n l = edge.length();\n edge.normalizeFromLength(l);\n let t = Vector3.Dot(tmp, edge) / Math.max(l, Epsilon);\n t = Scalar.Clamp(t, 0, 1);\n triProj.copyFrom(e0).addInPlace(edge.scaleInPlace(t * l));\n ref.copyFrom(triProj);\n return Vector3.Distance(vector, triProj);\n }\n /**\n * Returns a new Vector3 located at the center between \"value1\" and \"value2\"\n * Example Playground https://playground.babylonjs.com/#R1F8YU#72\n * @param value1 defines the first operand\n * @param value2 defines the second operand\n * @returns the new Vector3\n */\n static Center(value1, value2) {\n return Vector3.CenterToRef(value1, value2, Vector3.Zero());\n }\n /**\n * Gets the center of the vectors \"value1\" and \"value2\" and stores the result in the vector \"ref\"\n * Example Playground https://playground.babylonjs.com/#R1F8YU#73\n * @param value1 defines first vector\n * @param value2 defines second vector\n * @param ref defines third vector\n * @returns ref\n */\n static CenterToRef(value1, value2, ref) {\n return ref.copyFromFloats((value1._x + value2._x) / 2, (value1._y + value2._y) / 2, (value1._z + value2._z) / 2);\n }\n /**\n * Given three orthogonal normalized left-handed oriented Vector3 axis in space (target system),\n * RotationFromAxis() returns the rotation Euler angles (ex : rotation.x, rotation.y, rotation.z) to apply\n * to something in order to rotate it from its local system to the given target system\n * Note: axis1, axis2 and axis3 are normalized during this operation\n * Example Playground https://playground.babylonjs.com/#R1F8YU#106\n * @param axis1 defines the first axis\n * @param axis2 defines the second axis\n * @param axis3 defines the third axis\n * @returns a new Vector3\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/center_origin/target_align\n */\n static RotationFromAxis(axis1, axis2, axis3) {\n const rotation = new axis1.constructor();\n Vector3.RotationFromAxisToRef(axis1, axis2, axis3, rotation);\n return rotation;\n }\n /**\n * The same than RotationFromAxis but updates the given ref Vector3 parameter instead of returning a new Vector3\n * Example Playground https://playground.babylonjs.com/#R1F8YU#107\n * @param axis1 defines the first axis\n * @param axis2 defines the second axis\n * @param axis3 defines the third axis\n * @param ref defines the Vector3 where to store the result\n * @returns result input\n */\n static RotationFromAxisToRef(axis1, axis2, axis3, ref) {\n const quat = MathTmp.Quaternion[0];\n Quaternion.RotationQuaternionFromAxisToRef(axis1, axis2, axis3, quat);\n quat.toEulerAnglesToRef(ref);\n return ref;\n }\n }\n Vector3._UpReadOnly = Vector3.Up();\n Vector3._DownReadOnly = Vector3.Down();\n Vector3._LeftHandedForwardReadOnly = Vector3.Forward(false);\n Vector3._RightHandedForwardReadOnly = Vector3.Forward(true);\n Vector3._LeftHandedBackwardReadOnly = Vector3.Backward(false);\n Vector3._RightHandedBackwardReadOnly = Vector3.Backward(true);\n Vector3._RightReadOnly = Vector3.Right();\n Vector3._LeftReadOnly = Vector3.Left();\n Vector3._ZeroReadOnly = Vector3.Zero();\n Vector3._OneReadOnly = Vector3.One();\n /**\n * Vector4 class created for EulerAngle class conversion to Quaternion\n */\n class Vector4 {\n /**\n * Creates a Vector4 object from the given floats.\n * @param x x value of the vector\n * @param y y value of the vector\n * @param z z value of the vector\n * @param w w value of the vector\n */\n constructor(\n /** x value of the vector */\n x = 0, \n /** y value of the vector */\n y = 0, \n /** z value of the vector */\n z = 0, \n /** w value of the vector */\n w = 0) {\n this.x = x;\n this.y = y;\n this.z = z;\n this.w = w;\n }\n /**\n * Returns the string with the Vector4 coordinates.\n * @returns a string containing all the vector values\n */\n toString() {\n return `{X: ${this.x} Y: ${this.y} Z: ${this.z} W: ${this.w}}`;\n }\n /**\n * Returns the string \"Vector4\".\n * @returns \"Vector4\"\n */\n getClassName() {\n return \"Vector4\";\n }\n /**\n * Returns the Vector4 hash code.\n * @returns a unique hash code\n */\n getHashCode() {\n const x = _ExtractAsInt(this.x);\n const y = _ExtractAsInt(this.y);\n const z = _ExtractAsInt(this.z);\n const w = _ExtractAsInt(this.w);\n let hash = x;\n hash = (hash * 397) ^ y;\n hash = (hash * 397) ^ z;\n hash = (hash * 397) ^ w;\n return hash;\n }\n // Operators\n /**\n * Returns a new array populated with 4 elements : the Vector4 coordinates.\n * @returns the resulting array\n */\n asArray() {\n const result = new Array();\n this.toArray(result, 0);\n return result;\n }\n /**\n * Populates the given array from the given index with the Vector4 coordinates.\n * @param array array to populate\n * @param index index of the array to start at (default: 0)\n * @returns the Vector4.\n */\n toArray(array, index) {\n if (index === undefined) {\n index = 0;\n }\n array[index] = this.x;\n array[index + 1] = this.y;\n array[index + 2] = this.z;\n array[index + 3] = this.w;\n return this;\n }\n /**\n * Update the current vector from an array\n * @param array defines the destination array\n * @param index defines the offset in the destination array\n * @returns the current Vector3\n */\n fromArray(array, index = 0) {\n Vector4.FromArrayToRef(array, index, this);\n return this;\n }\n /**\n * Adds the given vector to the current Vector4.\n * @param otherVector the vector to add\n * @returns the updated Vector4.\n */\n addInPlace(otherVector) {\n this.x += otherVector.x;\n this.y += otherVector.y;\n this.z += otherVector.z;\n this.w += otherVector.w;\n return this;\n }\n /**\n * Returns a new Vector4 as the result of the addition of the current Vector4 and the given one.\n * @param otherVector the vector to add\n * @returns the resulting vector\n */\n add(otherVector) {\n return new this.constructor(this.x + otherVector.x, this.y + otherVector.y, this.z + otherVector.z, this.w + otherVector.w);\n }\n /**\n * Updates the given vector \"result\" with the result of the addition of the current Vector4 and the given one.\n * @param otherVector the vector to add\n * @param result the vector to store the result\n * @returns result input\n */\n addToRef(otherVector, result) {\n result.x = this.x + otherVector.x;\n result.y = this.y + otherVector.y;\n result.z = this.z + otherVector.z;\n result.w = this.w + otherVector.w;\n return result;\n }\n /**\n * Subtract in place the given vector from the current Vector4.\n * @param otherVector the vector to subtract\n * @returns the updated Vector4.\n */\n subtractInPlace(otherVector) {\n this.x -= otherVector.x;\n this.y -= otherVector.y;\n this.z -= otherVector.z;\n this.w -= otherVector.w;\n return this;\n }\n /**\n * Returns a new Vector4 with the result of the subtraction of the given vector from the current Vector4.\n * @param otherVector the vector to add\n * @returns the new vector with the result\n */\n subtract(otherVector) {\n return new this.constructor(this.x - otherVector.x, this.y - otherVector.y, this.z - otherVector.z, this.w - otherVector.w);\n }\n /**\n * Sets the given vector \"result\" with the result of the subtraction of the given vector from the current Vector4.\n * @param otherVector the vector to subtract\n * @param result the vector to store the result\n * @returns result input\n */\n subtractToRef(otherVector, result) {\n result.x = this.x - otherVector.x;\n result.y = this.y - otherVector.y;\n result.z = this.z - otherVector.z;\n result.w = this.w - otherVector.w;\n return result;\n }\n /**\n * Returns a new Vector4 set with the result of the subtraction of the given floats from the current Vector4 coordinates.\n */\n /**\n * Returns a new Vector4 set with the result of the subtraction of the given floats from the current Vector4 coordinates.\n * @param x value to subtract\n * @param y value to subtract\n * @param z value to subtract\n * @param w value to subtract\n * @returns new vector containing the result\n */\n subtractFromFloats(x, y, z, w) {\n return new this.constructor(this.x - x, this.y - y, this.z - z, this.w - w);\n }\n /**\n * Sets the given vector \"result\" set with the result of the subtraction of the given floats from the current Vector4 coordinates.\n * @param x value to subtract\n * @param y value to subtract\n * @param z value to subtract\n * @param w value to subtract\n * @param result the vector to store the result in\n * @returns result input\n */\n subtractFromFloatsToRef(x, y, z, w, result) {\n result.x = this.x - x;\n result.y = this.y - y;\n result.z = this.z - z;\n result.w = this.w - w;\n return result;\n }\n /**\n * Returns a new Vector4 set with the current Vector4 negated coordinates.\n * @returns a new vector with the negated values\n */\n negate() {\n return new this.constructor(-this.x, -this.y, -this.z, -this.w);\n }\n /**\n * Negate this vector in place\n * @returns this\n */\n negateInPlace() {\n this.x *= -1;\n this.y *= -1;\n this.z *= -1;\n this.w *= -1;\n return this;\n }\n /**\n * Negate the current Vector4 and stores the result in the given vector \"result\" coordinates\n * @param result defines the Vector3 object where to store the result\n * @returns the result\n */\n negateToRef(result) {\n return result.copyFromFloats(this.x * -1, this.y * -1, this.z * -1, this.w * -1);\n }\n /**\n * Multiplies the current Vector4 coordinates by scale (float).\n * @param scale the number to scale with\n * @returns the updated Vector4.\n */\n scaleInPlace(scale) {\n this.x *= scale;\n this.y *= scale;\n this.z *= scale;\n this.w *= scale;\n return this;\n }\n /**\n * Returns a new Vector4 set with the current Vector4 coordinates multiplied by scale (float).\n * @param scale the number to scale with\n * @returns a new vector with the result\n */\n scale(scale) {\n return new this.constructor(this.x * scale, this.y * scale, this.z * scale, this.w * scale);\n }\n /**\n * Sets the given vector \"result\" with the current Vector4 coordinates multiplied by scale (float).\n * @param scale the number to scale with\n * @param result a vector to store the result in\n * @returns result input\n */\n scaleToRef(scale, result) {\n result.x = this.x * scale;\n result.y = this.y * scale;\n result.z = this.z * scale;\n result.w = this.w * scale;\n return result;\n }\n /**\n * Scale the current Vector4 values by a factor and add the result to a given Vector4\n * @param scale defines the scale factor\n * @param result defines the Vector4 object where to store the result\n * @returns result input\n */\n scaleAndAddToRef(scale, result) {\n result.x += this.x * scale;\n result.y += this.y * scale;\n result.z += this.z * scale;\n result.w += this.w * scale;\n return result;\n }\n /**\n * Boolean : True if the current Vector4 coordinates are stricly equal to the given ones.\n * @param otherVector the vector to compare against\n * @returns true if they are equal\n */\n equals(otherVector) {\n return otherVector && this.x === otherVector.x && this.y === otherVector.y && this.z === otherVector.z && this.w === otherVector.w;\n }\n /**\n * Boolean : True if the current Vector4 coordinates are each beneath the distance \"epsilon\" from the given vector ones.\n * @param otherVector vector to compare against\n * @param epsilon (Default: very small number)\n * @returns true if they are equal\n */\n equalsWithEpsilon(otherVector, epsilon = Epsilon) {\n return (otherVector &&\n Scalar.WithinEpsilon(this.x, otherVector.x, epsilon) &&\n Scalar.WithinEpsilon(this.y, otherVector.y, epsilon) &&\n Scalar.WithinEpsilon(this.z, otherVector.z, epsilon) &&\n Scalar.WithinEpsilon(this.w, otherVector.w, epsilon));\n }\n /**\n * Boolean : True if the given floats are strictly equal to the current Vector4 coordinates.\n * @param x x value to compare against\n * @param y y value to compare against\n * @param z z value to compare against\n * @param w w value to compare against\n * @returns true if equal\n */\n equalsToFloats(x, y, z, w) {\n return this.x === x && this.y === y && this.z === z && this.w === w;\n }\n /**\n * Multiplies in place the current Vector4 by the given one.\n * @param otherVector vector to multiple with\n * @returns the updated Vector4.\n */\n multiplyInPlace(otherVector) {\n this.x *= otherVector.x;\n this.y *= otherVector.y;\n this.z *= otherVector.z;\n this.w *= otherVector.w;\n return this;\n }\n /**\n * Returns a new Vector4 set with the multiplication result of the current Vector4 and the given one.\n * @param otherVector vector to multiple with\n * @returns resulting new vector\n */\n multiply(otherVector) {\n return new this.constructor(this.x * otherVector.x, this.y * otherVector.y, this.z * otherVector.z, this.w * otherVector.w);\n }\n /**\n * Updates the given vector \"result\" with the multiplication result of the current Vector4 and the given one.\n * @param otherVector vector to multiple with\n * @param result vector to store the result\n * @returns result input\n */\n multiplyToRef(otherVector, result) {\n result.x = this.x * otherVector.x;\n result.y = this.y * otherVector.y;\n result.z = this.z * otherVector.z;\n result.w = this.w * otherVector.w;\n return result;\n }\n /**\n * Returns a new Vector4 set with the multiplication result of the given floats and the current Vector4 coordinates.\n * @param x x value multiply with\n * @param y y value multiply with\n * @param z z value multiply with\n * @param w w value multiply with\n * @returns resulting new vector\n */\n multiplyByFloats(x, y, z, w) {\n return new this.constructor(this.x * x, this.y * y, this.z * z, this.w * w);\n }\n /**\n * Returns a new Vector4 set with the division result of the current Vector4 by the given one.\n * @param otherVector vector to devide with\n * @returns resulting new vector\n */\n divide(otherVector) {\n return new this.constructor(this.x / otherVector.x, this.y / otherVector.y, this.z / otherVector.z, this.w / otherVector.w);\n }\n /**\n * Updates the given vector \"result\" with the division result of the current Vector4 by the given one.\n * @param otherVector vector to devide with\n * @param result vector to store the result\n * @returns result input\n */\n divideToRef(otherVector, result) {\n result.x = this.x / otherVector.x;\n result.y = this.y / otherVector.y;\n result.z = this.z / otherVector.z;\n result.w = this.w / otherVector.w;\n return result;\n }\n /**\n * Divides the current Vector3 coordinates by the given ones.\n * @param otherVector vector to devide with\n * @returns the updated Vector3.\n */\n divideInPlace(otherVector) {\n return this.divideToRef(otherVector, this);\n }\n /**\n * Updates the Vector4 coordinates with the minimum values between its own and the given vector ones\n * @param other defines the second operand\n * @returns the current updated Vector4\n */\n minimizeInPlace(other) {\n if (other.x < this.x) {\n this.x = other.x;\n }\n if (other.y < this.y) {\n this.y = other.y;\n }\n if (other.z < this.z) {\n this.z = other.z;\n }\n if (other.w < this.w) {\n this.w = other.w;\n }\n return this;\n }\n /**\n * Updates the Vector4 coordinates with the maximum values between its own and the given vector ones\n * @param other defines the second operand\n * @returns the current updated Vector4\n */\n maximizeInPlace(other) {\n if (other.x > this.x) {\n this.x = other.x;\n }\n if (other.y > this.y) {\n this.y = other.y;\n }\n if (other.z > this.z) {\n this.z = other.z;\n }\n if (other.w > this.w) {\n this.w = other.w;\n }\n return this;\n }\n /**\n * Gets a new Vector4 from current Vector4 floored values\n * @returns a new Vector4\n */\n floor() {\n return new this.constructor(Math.floor(this.x), Math.floor(this.y), Math.floor(this.z), Math.floor(this.w));\n }\n /**\n * Gets a new Vector4 from current Vector4 fractional values\n * @returns a new Vector4\n */\n fract() {\n return new this.constructor(this.x - Math.floor(this.x), this.y - Math.floor(this.y), this.z - Math.floor(this.z), this.w - Math.floor(this.w));\n }\n // Properties\n /**\n * Returns the Vector4 length (float).\n * @returns the length\n */\n length() {\n return Math.sqrt(this.x * this.x + this.y * this.y + this.z * this.z + this.w * this.w);\n }\n /**\n * Returns the Vector4 squared length (float).\n * @returns the length squared\n */\n lengthSquared() {\n return this.x * this.x + this.y * this.y + this.z * this.z + this.w * this.w;\n }\n // Methods\n /**\n * Normalizes in place the Vector4.\n * @returns the updated Vector4.\n */\n normalize() {\n const len = this.length();\n if (len === 0) {\n return this;\n }\n return this.scaleInPlace(1.0 / len);\n }\n /**\n * Returns a new Vector3 from the Vector4 (x, y, z) coordinates.\n * @returns this converted to a new vector3\n */\n toVector3() {\n return new Vector3(this.x, this.y, this.z);\n }\n /**\n * Returns a new Vector4 copied from the current one.\n * @returns the new cloned vector\n */\n clone() {\n return new this.constructor(this.x, this.y, this.z, this.w);\n }\n /**\n * Updates the current Vector4 with the given one coordinates.\n * @param source the source vector to copy from\n * @returns the updated Vector4.\n */\n copyFrom(source) {\n this.x = source.x;\n this.y = source.y;\n this.z = source.z;\n this.w = source.w;\n return this;\n }\n /**\n * Updates the current Vector4 coordinates with the given floats.\n * @param x float to copy from\n * @param y float to copy from\n * @param z float to copy from\n * @param w float to copy from\n * @returns the updated Vector4.\n */\n copyFromFloats(x, y, z, w) {\n this.x = x;\n this.y = y;\n this.z = z;\n this.w = w;\n return this;\n }\n /**\n * Updates the current Vector4 coordinates with the given floats.\n * @param x float to set from\n * @param y float to set from\n * @param z float to set from\n * @param w float to set from\n * @returns the updated Vector4.\n */\n set(x, y, z, w) {\n return this.copyFromFloats(x, y, z, w);\n }\n /**\n * Copies the given float to the current Vector3 coordinates\n * @param v defines the x, y, z and w coordinates of the operand\n * @returns the current updated Vector3\n */\n setAll(v) {\n this.x = this.y = this.z = this.w = v;\n return this;\n }\n // Statics\n /**\n * Returns a new Vector4 set from the starting index of the given array.\n * @param array the array to pull values from\n * @param offset the offset into the array to start at\n * @returns the new vector\n */\n static FromArray(array, offset) {\n if (!offset) {\n offset = 0;\n }\n return new Vector4(array[offset], array[offset + 1], array[offset + 2], array[offset + 3]);\n }\n /**\n * Updates the given vector \"result\" from the starting index of the given array.\n * @param array the array to pull values from\n * @param offset the offset into the array to start at\n * @param result the vector to store the result in\n * @returns result input\n */\n static FromArrayToRef(array, offset, result) {\n result.x = array[offset];\n result.y = array[offset + 1];\n result.z = array[offset + 2];\n result.w = array[offset + 3];\n return result;\n }\n /**\n * Updates the given vector \"result\" from the starting index of the given Float32Array.\n * @param array the array to pull values from\n * @param offset the offset into the array to start at\n * @param result the vector to store the result in\n * @returns result input\n */\n static FromFloatArrayToRef(array, offset, result) {\n Vector4.FromArrayToRef(array, offset, result);\n return result;\n }\n /**\n * Updates the given vector \"result\" coordinates from the given floats.\n * @param x float to set from\n * @param y float to set from\n * @param z float to set from\n * @param w float to set from\n * @param result the vector to the floats in\n * @returns result input\n */\n static FromFloatsToRef(x, y, z, w, result) {\n result.x = x;\n result.y = y;\n result.z = z;\n result.w = w;\n return result;\n }\n /**\n * Returns a new Vector4 set to (0.0, 0.0, 0.0, 0.0)\n * @returns the new vector\n */\n static Zero() {\n return new Vector4(0.0, 0.0, 0.0, 0.0);\n }\n /**\n * Returns a new Vector4 set to (1.0, 1.0, 1.0, 1.0)\n * @returns the new vector\n */\n static One() {\n return new Vector4(1.0, 1.0, 1.0, 1.0);\n }\n /**\n * Returns a new Vector4 with random values between min and max\n * @param min the minimum random value\n * @param max the maximum random value\n * @returns a Vector4 with random values between min and max\n */\n static Random(min = 0, max = 1) {\n return new Vector4(Scalar.RandomRange(min, max), Scalar.RandomRange(min, max), Scalar.RandomRange(min, max), Scalar.RandomRange(min, max));\n }\n /**\n * Gets a zero Vector4 that must not be updated\n */\n static get ZeroReadOnly() {\n return Vector4._ZeroReadOnly;\n }\n /**\n * Returns a new normalized Vector4 from the given one.\n * @param vector the vector to normalize\n * @returns the vector\n */\n static Normalize(vector) {\n const result = Vector4.Zero();\n Vector4.NormalizeToRef(vector, result);\n return result;\n }\n /**\n * Updates the given vector \"result\" from the normalization of the given one.\n * @param vector the vector to normalize\n * @param result the vector to store the result in\n * @returns result input\n */\n static NormalizeToRef(vector, result) {\n result.copyFrom(vector);\n result.normalize();\n return result;\n }\n /**\n * Returns a vector with the minimum values from the left and right vectors\n * @param left left vector to minimize\n * @param right right vector to minimize\n * @returns a new vector with the minimum of the left and right vector values\n */\n static Minimize(left, right) {\n const min = new left.constructor();\n min.copyFrom(left);\n min.minimizeInPlace(right);\n return min;\n }\n /**\n * Returns a vector with the maximum values from the left and right vectors\n * @param left left vector to maximize\n * @param right right vector to maximize\n * @returns a new vector with the maximum of the left and right vector values\n */\n static Maximize(left, right) {\n const max = new left.constructor();\n max.copyFrom(left);\n max.maximizeInPlace(right);\n return max;\n }\n /**\n * Returns the distance (float) between the vectors \"value1\" and \"value2\".\n * @param value1 value to calulate the distance between\n * @param value2 value to calulate the distance between\n * @returns the distance between the two vectors\n */\n static Distance(value1, value2) {\n return Math.sqrt(Vector4.DistanceSquared(value1, value2));\n }\n /**\n * Returns the squared distance (float) between the vectors \"value1\" and \"value2\".\n * @param value1 value to calulate the distance between\n * @param value2 value to calulate the distance between\n * @returns the distance between the two vectors squared\n */\n static DistanceSquared(value1, value2) {\n const x = value1.x - value2.x;\n const y = value1.y - value2.y;\n const z = value1.z - value2.z;\n const w = value1.w - value2.w;\n return x * x + y * y + z * z + w * w;\n }\n /**\n * Returns a new Vector4 located at the center between the vectors \"value1\" and \"value2\".\n * @param value1 value to calulate the center between\n * @param value2 value to calulate the center between\n * @returns the center between the two vectors\n */\n static Center(value1, value2) {\n return Vector4.CenterToRef(value1, value2, Vector4.Zero());\n }\n /**\n * Gets the center of the vectors \"value1\" and \"value2\" and stores the result in the vector \"ref\"\n * @param value1 defines first vector\n * @param value2 defines second vector\n * @param ref defines third vector\n * @returns ref\n */\n static CenterToRef(value1, value2, ref) {\n return ref.copyFromFloats((value1.x + value2.x) / 2, (value1.y + value2.y) / 2, (value1.z + value2.z) / 2, (value1.w + value2.w) / 2);\n }\n /**\n * Returns a new Vector4 set with the result of the transformation by the given matrix of the given vector.\n * This method computes tranformed coordinates only, not transformed direction vectors (ie. it takes translation in account)\n * The difference with Vector3.TransformCoordinates is that the w component is not used to divide the other coordinates but is returned in the w coordinate instead\n * @param vector defines the Vector3 to transform\n * @param transformation defines the transformation matrix\n * @returns the transformed Vector4\n */\n static TransformCoordinates(vector, transformation) {\n const result = Vector4.Zero();\n Vector4.TransformCoordinatesToRef(vector, transformation, result);\n return result;\n }\n /**\n * Sets the given vector \"result\" coordinates with the result of the transformation by the given matrix of the given vector\n * This method computes tranformed coordinates only, not transformed direction vectors (ie. it takes translation in account)\n * The difference with Vector3.TransformCoordinatesToRef is that the w component is not used to divide the other coordinates but is returned in the w coordinate instead\n * @param vector defines the Vector3 to transform\n * @param transformation defines the transformation matrix\n * @param result defines the Vector4 where to store the result\n * @returns result input\n */\n static TransformCoordinatesToRef(vector, transformation, result) {\n Vector4.TransformCoordinatesFromFloatsToRef(vector._x, vector._y, vector._z, transformation, result);\n return result;\n }\n /**\n * Sets the given vector \"result\" coordinates with the result of the transformation by the given matrix of the given floats (x, y, z)\n * This method computes tranformed coordinates only, not transformed direction vectors\n * The difference with Vector3.TransformCoordinatesFromFloatsToRef is that the w component is not used to divide the other coordinates but is returned in the w coordinate instead\n * @param x define the x coordinate of the source vector\n * @param y define the y coordinate of the source vector\n * @param z define the z coordinate of the source vector\n * @param transformation defines the transformation matrix\n * @param result defines the Vector4 where to store the result\n * @returns result input\n */\n static TransformCoordinatesFromFloatsToRef(x, y, z, transformation, result) {\n const m = transformation.m;\n const rx = x * m[0] + y * m[4] + z * m[8] + m[12];\n const ry = x * m[1] + y * m[5] + z * m[9] + m[13];\n const rz = x * m[2] + y * m[6] + z * m[10] + m[14];\n const rw = x * m[3] + y * m[7] + z * m[11] + m[15];\n result.x = rx;\n result.y = ry;\n result.z = rz;\n result.w = rw;\n return result;\n }\n /**\n * Returns a new Vector4 set with the result of the normal transformation by the given matrix of the given vector.\n * This methods computes transformed normalized direction vectors only.\n * @param vector the vector to transform\n * @param transformation the transformation matrix to apply\n * @returns the new vector\n */\n static TransformNormal(vector, transformation) {\n const result = new vector.constructor();\n Vector4.TransformNormalToRef(vector, transformation, result);\n return result;\n }\n /**\n * Sets the given vector \"result\" with the result of the normal transformation by the given matrix of the given vector.\n * This methods computes transformed normalized direction vectors only.\n * @param vector the vector to transform\n * @param transformation the transformation matrix to apply\n * @param result the vector to store the result in\n * @returns result input\n */\n static TransformNormalToRef(vector, transformation, result) {\n const m = transformation.m;\n const x = vector.x * m[0] + vector.y * m[4] + vector.z * m[8];\n const y = vector.x * m[1] + vector.y * m[5] + vector.z * m[9];\n const z = vector.x * m[2] + vector.y * m[6] + vector.z * m[10];\n result.x = x;\n result.y = y;\n result.z = z;\n result.w = vector.w;\n return result;\n }\n /**\n * Sets the given vector \"result\" with the result of the normal transformation by the given matrix of the given floats (x, y, z, w).\n * This methods computes transformed normalized direction vectors only.\n * @param x value to transform\n * @param y value to transform\n * @param z value to transform\n * @param w value to transform\n * @param transformation the transformation matrix to apply\n * @param result the vector to store the results in\n * @returns result input\n */\n static TransformNormalFromFloatsToRef(x, y, z, w, transformation, result) {\n const m = transformation.m;\n result.x = x * m[0] + y * m[4] + z * m[8];\n result.y = x * m[1] + y * m[5] + z * m[9];\n result.z = x * m[2] + y * m[6] + z * m[10];\n result.w = w;\n return result;\n }\n /**\n * Creates a new Vector4 from a Vector3\n * @param source defines the source data\n * @param w defines the 4th component (default is 0)\n * @returns a new Vector4\n */\n static FromVector3(source, w = 0) {\n return new Vector4(source._x, source._y, source._z, w);\n }\n }\n Vector4._ZeroReadOnly = Vector4.Zero();\n /**\n * Class used to store quaternion data\n * Example Playground - Overview - https://playground.babylonjs.com/#L49EJ7#100\n * @see https://en.wikipedia.org/wiki/Quaternion\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms\n */\n class Quaternion {\n /** Gets or sets the x coordinate */\n get x() {\n return this._x;\n }\n set x(value) {\n this._x = value;\n this._isDirty = true;\n }\n /** Gets or sets the y coordinate */\n get y() {\n return this._y;\n }\n set y(value) {\n this._y = value;\n this._isDirty = true;\n }\n /** Gets or sets the z coordinate */\n get z() {\n return this._z;\n }\n set z(value) {\n this._z = value;\n this._isDirty = true;\n }\n /** Gets or sets the w coordinate */\n get w() {\n return this._w;\n }\n set w(value) {\n this._w = value;\n this._isDirty = true;\n }\n /**\n * Creates a new Quaternion from the given floats\n * @param x defines the first component (0 by default)\n * @param y defines the second component (0 by default)\n * @param z defines the third component (0 by default)\n * @param w defines the fourth component (1.0 by default)\n */\n constructor(x = 0.0, y = 0.0, z = 0.0, w = 1.0) {\n /** @internal */\n this._isDirty = true;\n this._x = x;\n this._y = y;\n this._z = z;\n this._w = w;\n }\n /**\n * Gets a string representation for the current quaternion\n * @returns a string with the Quaternion coordinates\n */\n toString() {\n return `{X: ${this._x} Y: ${this._y} Z: ${this._z} W: ${this._w}}`;\n }\n /**\n * Gets the class name of the quaternion\n * @returns the string \"Quaternion\"\n */\n getClassName() {\n return \"Quaternion\";\n }\n /**\n * Gets a hash code for this quaternion\n * @returns the quaternion hash code\n */\n getHashCode() {\n const x = _ExtractAsInt(this._x);\n const y = _ExtractAsInt(this._y);\n const z = _ExtractAsInt(this._z);\n const w = _ExtractAsInt(this._w);\n let hash = x;\n hash = (hash * 397) ^ y;\n hash = (hash * 397) ^ z;\n hash = (hash * 397) ^ w;\n return hash;\n }\n /**\n * Copy the quaternion to an array\n * Example Playground https://playground.babylonjs.com/#L49EJ7#13\n * @returns a new array populated with 4 elements from the quaternion coordinates\n */\n asArray() {\n return [this._x, this._y, this._z, this._w];\n }\n /**\n * Stores from the starting index in the given array the Quaternion successive values\n * Example Playground https://playground.babylonjs.com/#L49EJ7#59\n * @param array defines the array where to store the x,y,z,w components\n * @param index defines an optional index in the target array to define where to start storing values\n * @returns the current Quaternion object\n */\n toArray(array, index = 0) {\n array[index] = this._x;\n array[index + 1] = this._y;\n array[index + 2] = this._z;\n array[index + 3] = this._w;\n return this;\n }\n /**\n * Check if two quaternions are equals\n * Example Playground https://playground.babylonjs.com/#L49EJ7#38\n * @param otherQuaternion defines the second operand\n * @returns true if the current quaternion and the given one coordinates are strictly equals\n */\n equals(otherQuaternion) {\n return otherQuaternion && this._x === otherQuaternion._x && this._y === otherQuaternion._y && this._z === otherQuaternion._z && this._w === otherQuaternion._w;\n }\n /**\n * Gets a boolean if two quaternions are equals (using an epsilon value)\n * Example Playground https://playground.babylonjs.com/#L49EJ7#37\n * @param otherQuaternion defines the other quaternion\n * @param epsilon defines the minimal distance to consider equality\n * @returns true if the given quaternion coordinates are close to the current ones by a distance of epsilon.\n */\n equalsWithEpsilon(otherQuaternion, epsilon = Epsilon) {\n return (otherQuaternion &&\n Scalar.WithinEpsilon(this._x, otherQuaternion._x, epsilon) &&\n Scalar.WithinEpsilon(this._y, otherQuaternion._y, epsilon) &&\n Scalar.WithinEpsilon(this._z, otherQuaternion._z, epsilon) &&\n Scalar.WithinEpsilon(this._w, otherQuaternion._w, epsilon));\n }\n /**\n * Clone the current quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#12\n * @returns a new quaternion copied from the current one\n */\n clone() {\n return new this.constructor(this._x, this._y, this._z, this._w);\n }\n /**\n * Copy a quaternion to the current one\n * Example Playground https://playground.babylonjs.com/#L49EJ7#86\n * @param other defines the other quaternion\n * @returns the updated current quaternion\n */\n copyFrom(other) {\n this._x = other._x;\n this._y = other._y;\n this._z = other._z;\n this._w = other._w;\n this._isDirty = true;\n return this;\n }\n /**\n * Updates the current quaternion with the given float coordinates\n * Example Playground https://playground.babylonjs.com/#L49EJ7#87\n * @param x defines the x coordinate\n * @param y defines the y coordinate\n * @param z defines the z coordinate\n * @param w defines the w coordinate\n * @returns the updated current quaternion\n */\n copyFromFloats(x, y, z, w) {\n this._x = x;\n this._y = y;\n this._z = z;\n this._w = w;\n this._isDirty = true;\n return this;\n }\n /**\n * Updates the current quaternion from the given float coordinates\n * Example Playground https://playground.babylonjs.com/#L49EJ7#56\n * @param x defines the x coordinate\n * @param y defines the y coordinate\n * @param z defines the z coordinate\n * @param w defines the w coordinate\n * @returns the updated current quaternion\n */\n set(x, y, z, w) {\n return this.copyFromFloats(x, y, z, w);\n }\n /**\n * Adds two quaternions\n * Example Playground https://playground.babylonjs.com/#L49EJ7#10\n * @param other defines the second operand\n * @returns a new quaternion as the addition result of the given one and the current quaternion\n */\n add(other) {\n return new this.constructor(this._x + other._x, this._y + other._y, this._z + other._z, this._w + other._w);\n }\n /**\n * Add a quaternion to the current one\n * Example Playground https://playground.babylonjs.com/#L49EJ7#11\n * @param other defines the quaternion to add\n * @returns the current quaternion\n */\n addInPlace(other) {\n this._x += other._x;\n this._y += other._y;\n this._z += other._z;\n this._w += other._w;\n this._isDirty = true;\n return this;\n }\n /**\n * Subtract two quaternions\n * Example Playground https://playground.babylonjs.com/#L49EJ7#57\n * @param other defines the second operand\n * @returns a new quaternion as the subtraction result of the given one from the current one\n */\n subtract(other) {\n return new this.constructor(this._x - other._x, this._y - other._y, this._z - other._z, this._w - other._w);\n }\n /**\n * Subtract a quaternion to the current one\n * Example Playground https://playground.babylonjs.com/#L49EJ7#58\n * @param other defines the quaternion to subtract\n * @returns the current quaternion\n */\n subtractInPlace(other) {\n this._x -= other._x;\n this._y -= other._y;\n this._z -= other._z;\n this._w -= other._w;\n this._isDirty = true;\n return this;\n }\n /**\n * Multiplies the current quaternion by a scale factor\n * Example Playground https://playground.babylonjs.com/#L49EJ7#88\n * @param value defines the scale factor\n * @returns a new quaternion set by multiplying the current quaternion coordinates by the float \"scale\"\n */\n scale(value) {\n return new this.constructor(this._x * value, this._y * value, this._z * value, this._w * value);\n }\n /**\n * Scale the current quaternion values by a factor and stores the result to a given quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#89\n * @param scale defines the scale factor\n * @param result defines the Quaternion object where to store the result\n * @returns result input\n */\n scaleToRef(scale, result) {\n result._x = this._x * scale;\n result._y = this._y * scale;\n result._z = this._z * scale;\n result._w = this._w * scale;\n result._isDirty = true;\n return result;\n }\n /**\n * Multiplies in place the current quaternion by a scale factor\n * Example Playground https://playground.babylonjs.com/#L49EJ7#90\n * @param value defines the scale factor\n * @returns the current modified quaternion\n */\n scaleInPlace(value) {\n this._x *= value;\n this._y *= value;\n this._z *= value;\n this._w *= value;\n this._isDirty = true;\n return this;\n }\n /**\n * Scale the current quaternion values by a factor and add the result to a given quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#91\n * @param scale defines the scale factor\n * @param result defines the Quaternion object where to store the result\n * @returns result input\n */\n scaleAndAddToRef(scale, result) {\n result._x += this._x * scale;\n result._y += this._y * scale;\n result._z += this._z * scale;\n result._w += this._w * scale;\n result._isDirty = true;\n return result;\n }\n /**\n * Multiplies two quaternions\n * Example Playground https://playground.babylonjs.com/#L49EJ7#43\n * @param q1 defines the second operand\n * @returns a new quaternion set as the multiplication result of the current one with the given one \"q1\"\n */\n multiply(q1) {\n const result = new this.constructor(0, 0, 0, 1.0);\n this.multiplyToRef(q1, result);\n return result;\n }\n /**\n * Sets the given \"result\" as the the multiplication result of the current one with the given one \"q1\"\n * Example Playground https://playground.babylonjs.com/#L49EJ7#45\n * @param q1 defines the second operand\n * @param result defines the target quaternion\n * @returns the current quaternion\n */\n multiplyToRef(q1, result) {\n const x = this._x * q1._w + this._y * q1._z - this._z * q1._y + this._w * q1._x;\n const y = -this._x * q1._z + this._y * q1._w + this._z * q1._x + this._w * q1._y;\n const z = this._x * q1._y - this._y * q1._x + this._z * q1._w + this._w * q1._z;\n const w = -this._x * q1._x - this._y * q1._y - this._z * q1._z + this._w * q1._w;\n result.copyFromFloats(x, y, z, w);\n return result;\n }\n /**\n * Updates the current quaternion with the multiplication of itself with the given one \"q1\"\n * Example Playground https://playground.babylonjs.com/#L49EJ7#46\n * @param q1 defines the second operand\n * @returns the currentupdated quaternion\n */\n multiplyInPlace(q1) {\n this.multiplyToRef(q1, this);\n return this;\n }\n /**\n * Conjugates the current quaternion and stores the result in the given quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#81\n * @param ref defines the target quaternion\n * @returns result input\n */\n conjugateToRef(ref) {\n ref.copyFromFloats(-this._x, -this._y, -this._z, this._w);\n return ref;\n }\n /**\n * Conjugates in place the current quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#82\n * @returns the current updated quaternion\n */\n conjugateInPlace() {\n this._x *= -1;\n this._y *= -1;\n this._z *= -1;\n this._isDirty = true;\n return this;\n }\n /**\n * Conjugates (1-q) the current quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#83\n * @returns a new quaternion\n */\n conjugate() {\n return new this.constructor(-this._x, -this._y, -this._z, this._w);\n }\n /**\n * Returns the inverse of the current quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#84\n * @returns a new quaternion\n */\n invert() {\n const conjugate = this.conjugate();\n const lengthSquared = this.lengthSquared();\n if (lengthSquared == 0 || lengthSquared == 1) {\n return conjugate;\n }\n conjugate.scaleInPlace(1 / lengthSquared);\n return conjugate;\n }\n /**\n * Invert in place the current quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#85\n * @returns this quaternion\n */\n invertInPlace() {\n this.conjugateInPlace();\n const lengthSquared = this.lengthSquared();\n if (lengthSquared == 0 || lengthSquared == 1) {\n return this;\n }\n this.scaleInPlace(1 / lengthSquared);\n return this;\n }\n /**\n * Gets squared length of current quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#29\n * @returns the quaternion length (float)\n */\n lengthSquared() {\n return this._x * this._x + this._y * this._y + this._z * this._z + this._w * this._w;\n }\n /**\n * Gets length of current quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#28\n * @returns the quaternion length (float)\n */\n length() {\n return Math.sqrt(this.lengthSquared());\n }\n /**\n * Normalize in place the current quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#54\n * @returns the current updated quaternion\n */\n normalize() {\n const len = this.length();\n if (len === 0) {\n return this;\n }\n const inv = 1.0 / len;\n this.scaleInPlace(inv);\n return this;\n }\n /**\n * Normalize a copy of the current quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#55\n * @returns the normalized quaternion\n */\n normalizeToNew() {\n const len = this.length();\n if (len === 0) {\n return this.clone();\n }\n const inv = 1.0 / len;\n return this.scale(inv);\n }\n /**\n * Returns a new Vector3 set with the Euler angles translated from the current quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#32\n * @returns a new Vector3 containing the Euler angles\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/center_origin/rotation_conventions\n */\n toEulerAngles() {\n const result = Vector3.Zero();\n this.toEulerAnglesToRef(result);\n return result;\n }\n /**\n * Sets the given vector3 \"result\" with the Euler angles translated from the current quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#31\n * @param result defines the vector which will be filled with the Euler angles\n * @returns result input\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/center_origin/rotation_conventions\n */\n toEulerAnglesToRef(result) {\n const qz = this._z;\n const qx = this._x;\n const qy = this._y;\n const qw = this._w;\n const zAxisY = qy * qz - qx * qw;\n const limit = 0.4999999;\n if (zAxisY < -limit) {\n result._y = 2 * Math.atan2(qy, qw);\n result._x = Math.PI / 2;\n result._z = 0;\n result._isDirty = true;\n }\n else if (zAxisY > limit) {\n result._y = 2 * Math.atan2(qy, qw);\n result._x = -Math.PI / 2;\n result._z = 0;\n result._isDirty = true;\n }\n else {\n const sqw = qw * qw;\n const sqz = qz * qz;\n const sqx = qx * qx;\n const sqy = qy * qy;\n result._z = Math.atan2(2.0 * (qx * qy + qz * qw), -sqz - sqx + sqy + sqw);\n result._x = Math.asin(-2.0 * zAxisY);\n result._y = Math.atan2(2.0 * (qz * qx + qy * qw), sqz - sqx - sqy + sqw);\n result._isDirty = true;\n }\n return result;\n }\n /**\n * Updates the given rotation matrix with the current quaternion values\n * Example Playground https://playground.babylonjs.com/#L49EJ7#67\n * @param result defines the target matrix\n * @returns the current unchanged quaternion\n */\n toRotationMatrix(result) {\n Matrix.FromQuaternionToRef(this, result);\n return result;\n }\n /**\n * Updates the current quaternion from the given rotation matrix values\n * Example Playground https://playground.babylonjs.com/#L49EJ7#41\n * @param matrix defines the source matrix\n * @returns the current updated quaternion\n */\n fromRotationMatrix(matrix) {\n Quaternion.FromRotationMatrixToRef(matrix, this);\n return this;\n }\n // Statics\n /**\n * Creates a new quaternion from a rotation matrix\n * Example Playground https://playground.babylonjs.com/#L49EJ7#101\n * @param matrix defines the source matrix\n * @returns a new quaternion created from the given rotation matrix values\n */\n static FromRotationMatrix(matrix) {\n const result = new Quaternion();\n Quaternion.FromRotationMatrixToRef(matrix, result);\n return result;\n }\n /**\n * Updates the given quaternion with the given rotation matrix values\n * Example Playground https://playground.babylonjs.com/#L49EJ7#102\n * @param matrix defines the source matrix\n * @param result defines the target quaternion\n * @returns result input\n */\n static FromRotationMatrixToRef(matrix, result) {\n const data = matrix.m;\n const m11 = data[0], m12 = data[4], m13 = data[8];\n const m21 = data[1], m22 = data[5], m23 = data[9];\n const m31 = data[2], m32 = data[6], m33 = data[10];\n const trace = m11 + m22 + m33;\n let s;\n if (trace > 0) {\n s = 0.5 / Math.sqrt(trace + 1.0);\n result._w = 0.25 / s;\n result._x = (m32 - m23) * s;\n result._y = (m13 - m31) * s;\n result._z = (m21 - m12) * s;\n result._isDirty = true;\n }\n else if (m11 > m22 && m11 > m33) {\n s = 2.0 * Math.sqrt(1.0 + m11 - m22 - m33);\n result._w = (m32 - m23) / s;\n result._x = 0.25 * s;\n result._y = (m12 + m21) / s;\n result._z = (m13 + m31) / s;\n result._isDirty = true;\n }\n else if (m22 > m33) {\n s = 2.0 * Math.sqrt(1.0 + m22 - m11 - m33);\n result._w = (m13 - m31) / s;\n result._x = (m12 + m21) / s;\n result._y = 0.25 * s;\n result._z = (m23 + m32) / s;\n result._isDirty = true;\n }\n else {\n s = 2.0 * Math.sqrt(1.0 + m33 - m11 - m22);\n result._w = (m21 - m12) / s;\n result._x = (m13 + m31) / s;\n result._y = (m23 + m32) / s;\n result._z = 0.25 * s;\n result._isDirty = true;\n }\n return result;\n }\n /**\n * Returns the dot product (float) between the quaternions \"left\" and \"right\"\n * Example Playground https://playground.babylonjs.com/#L49EJ7#61\n * @param left defines the left operand\n * @param right defines the right operand\n * @returns the dot product\n */\n static Dot(left, right) {\n return left._x * right._x + left._y * right._y + left._z * right._z + left._w * right._w;\n }\n /**\n * Checks if the orientations of two rotation quaternions are close to each other\n * Example Playground https://playground.babylonjs.com/#L49EJ7#60\n * @param quat0 defines the first quaternion to check\n * @param quat1 defines the second quaternion to check\n * @param epsilon defines closeness, 0 same orientation, 1 PI apart, default 0.1\n * @returns true if the two quaternions are close to each other within epsilon\n */\n static AreClose(quat0, quat1, epsilon = 0.1) {\n const dot = Quaternion.Dot(quat0, quat1);\n return 1 - dot * dot <= epsilon;\n }\n /**\n * Smooth interpolation between two quaternions using Slerp\n * Example Playground https://playground.babylonjs.com/#L49EJ7#93\n * @param source source quaternion\n * @param goal goal quaternion\n * @param deltaTime current interpolation frame\n * @param lerpTime total interpolation time\n * @param result the smoothed quaternion\n */\n static SmoothToRef(source, goal, deltaTime, lerpTime, result) {\n let slerp = lerpTime === 0 ? 1 : deltaTime / lerpTime;\n slerp = Scalar.Clamp(slerp, 0, 1);\n Quaternion.SlerpToRef(source, goal, slerp, result);\n return result;\n }\n /**\n * Creates an empty quaternion\n * @returns a new quaternion set to (0.0, 0.0, 0.0)\n */\n static Zero() {\n return new Quaternion(0.0, 0.0, 0.0, 0.0);\n }\n /**\n * Inverse a given quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#103\n * @param q defines the source quaternion\n * @returns a new quaternion as the inverted current quaternion\n */\n static Inverse(q) {\n return new q.constructor(-q._x, -q._y, -q._z, q._w);\n }\n /**\n * Inverse a given quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#104\n * @param q defines the source quaternion\n * @param result the quaternion the result will be stored in\n * @returns the result quaternion\n */\n static InverseToRef(q, result) {\n result.set(-q._x, -q._y, -q._z, q._w);\n return result;\n }\n /**\n * Creates an identity quaternion\n * @returns the identity quaternion\n */\n static Identity() {\n return new Quaternion(0.0, 0.0, 0.0, 1.0);\n }\n /**\n * Gets a boolean indicating if the given quaternion is identity\n * @param quaternion defines the quaternion to check\n * @returns true if the quaternion is identity\n */\n static IsIdentity(quaternion) {\n return quaternion && quaternion._x === 0 && quaternion._y === 0 && quaternion._z === 0 && quaternion._w === 1;\n }\n /**\n * Creates a quaternion from a rotation around an axis\n * Example Playground https://playground.babylonjs.com/#L49EJ7#72\n * @param axis defines the axis to use\n * @param angle defines the angle to use\n * @returns a new quaternion created from the given axis (Vector3) and angle in radians (float)\n */\n static RotationAxis(axis, angle) {\n return Quaternion.RotationAxisToRef(axis, angle, new Quaternion());\n }\n /**\n * Creates a rotation around an axis and stores it into the given quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#73\n * @param axis defines the axis to use\n * @param angle defines the angle to use\n * @param result defines the target quaternion\n * @returns the target quaternion\n */\n static RotationAxisToRef(axis, angle, result) {\n const sin = Math.sin(angle / 2);\n axis.normalize();\n result._w = Math.cos(angle / 2);\n result._x = axis._x * sin;\n result._y = axis._y * sin;\n result._z = axis._z * sin;\n result._isDirty = true;\n return result;\n }\n /**\n * Creates a new quaternion from data stored into an array\n * Example Playground https://playground.babylonjs.com/#L49EJ7#63\n * @param array defines the data source\n * @param offset defines the offset in the source array where the data starts\n * @returns a new quaternion\n */\n static FromArray(array, offset) {\n if (!offset) {\n offset = 0;\n }\n return new Quaternion(array[offset], array[offset + 1], array[offset + 2], array[offset + 3]);\n }\n /**\n * Updates the given quaternion \"result\" from the starting index of the given array.\n * Example Playground https://playground.babylonjs.com/#L49EJ7#64\n * @param array the array to pull values from\n * @param offset the offset into the array to start at\n * @param result the quaternion to store the result in\n * @returns result input\n */\n static FromArrayToRef(array, offset, result) {\n result._x = array[offset];\n result._y = array[offset + 1];\n result._z = array[offset + 2];\n result._w = array[offset + 3];\n result._isDirty = true;\n return result;\n }\n /**\n * Create a quaternion from Euler rotation angles\n * Example Playground https://playground.babylonjs.com/#L49EJ7#33\n * @param x Pitch\n * @param y Yaw\n * @param z Roll\n * @returns the new Quaternion\n */\n static FromEulerAngles(x, y, z) {\n const q = new Quaternion();\n Quaternion.RotationYawPitchRollToRef(y, x, z, q);\n return q;\n }\n /**\n * Updates a quaternion from Euler rotation angles\n * Example Playground https://playground.babylonjs.com/#L49EJ7#34\n * @param x Pitch\n * @param y Yaw\n * @param z Roll\n * @param result the quaternion to store the result\n * @returns the updated quaternion\n */\n static FromEulerAnglesToRef(x, y, z, result) {\n Quaternion.RotationYawPitchRollToRef(y, x, z, result);\n return result;\n }\n /**\n * Create a quaternion from Euler rotation vector\n * Example Playground https://playground.babylonjs.com/#L49EJ7#35\n * @param vec the Euler vector (x Pitch, y Yaw, z Roll)\n * @returns the new Quaternion\n */\n static FromEulerVector(vec) {\n const q = new Quaternion();\n Quaternion.RotationYawPitchRollToRef(vec._y, vec._x, vec._z, q);\n return q;\n }\n /**\n * Updates a quaternion from Euler rotation vector\n * Example Playground https://playground.babylonjs.com/#L49EJ7#36\n * @param vec the Euler vector (x Pitch, y Yaw, z Roll)\n * @param result the quaternion to store the result\n * @returns the updated quaternion\n */\n static FromEulerVectorToRef(vec, result) {\n Quaternion.RotationYawPitchRollToRef(vec._y, vec._x, vec._z, result);\n return result;\n }\n /**\n * Updates a quaternion so that it rotates vector vecFrom to vector vecTo\n * Example Playground - https://playground.babylonjs.com/#L49EJ7#70\n * @param vecFrom defines the direction vector from which to rotate\n * @param vecTo defines the direction vector to which to rotate\n * @param result the quaternion to store the result\n * @param epsilon defines the minimal dot value to define vecs as opposite. Default: `BABYLON.Epsilon`\n * @returns the updated quaternion\n */\n static FromUnitVectorsToRef(vecFrom, vecTo, result, epsilon = Epsilon) {\n const r = Vector3.Dot(vecFrom, vecTo) + 1;\n if (r < epsilon) {\n if (Math.abs(vecFrom.x) > Math.abs(vecFrom.z)) {\n result.set(-vecFrom.y, vecFrom.x, 0, 0);\n }\n else {\n result.set(0, -vecFrom.z, vecFrom.y, 0);\n }\n }\n else {\n Vector3.CrossToRef(vecFrom, vecTo, TmpVectors.Vector3[0]);\n result.set(TmpVectors.Vector3[0].x, TmpVectors.Vector3[0].y, TmpVectors.Vector3[0].z, r);\n }\n return result.normalize();\n }\n /**\n * Creates a new quaternion from the given Euler float angles (y, x, z)\n * Example Playground https://playground.babylonjs.com/#L49EJ7#77\n * @param yaw defines the rotation around Y axis\n * @param pitch defines the rotation around X axis\n * @param roll defines the rotation around Z axis\n * @returns the new quaternion\n */\n static RotationYawPitchRoll(yaw, pitch, roll) {\n const q = new Quaternion();\n Quaternion.RotationYawPitchRollToRef(yaw, pitch, roll, q);\n return q;\n }\n /**\n * Creates a new rotation from the given Euler float angles (y, x, z) and stores it in the target quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#78\n * @param yaw defines the rotation around Y axis\n * @param pitch defines the rotation around X axis\n * @param roll defines the rotation around Z axis\n * @param result defines the target quaternion\n * @returns result input\n */\n static RotationYawPitchRollToRef(yaw, pitch, roll, result) {\n // Produces a quaternion from Euler angles in the z-y-x orientation (Tait-Bryan angles)\n const halfRoll = roll * 0.5;\n const halfPitch = pitch * 0.5;\n const halfYaw = yaw * 0.5;\n const sinRoll = Math.sin(halfRoll);\n const cosRoll = Math.cos(halfRoll);\n const sinPitch = Math.sin(halfPitch);\n const cosPitch = Math.cos(halfPitch);\n const sinYaw = Math.sin(halfYaw);\n const cosYaw = Math.cos(halfYaw);\n result._x = cosYaw * sinPitch * cosRoll + sinYaw * cosPitch * sinRoll;\n result._y = sinYaw * cosPitch * cosRoll - cosYaw * sinPitch * sinRoll;\n result._z = cosYaw * cosPitch * sinRoll - sinYaw * sinPitch * cosRoll;\n result._w = cosYaw * cosPitch * cosRoll + sinYaw * sinPitch * sinRoll;\n result._isDirty = true;\n return result;\n }\n /**\n * Creates a new quaternion from the given Euler float angles expressed in z-x-z orientation\n * Example Playground https://playground.babylonjs.com/#L49EJ7#68\n * @param alpha defines the rotation around first axis\n * @param beta defines the rotation around second axis\n * @param gamma defines the rotation around third axis\n * @returns the new quaternion\n */\n static RotationAlphaBetaGamma(alpha, beta, gamma) {\n const result = new Quaternion();\n Quaternion.RotationAlphaBetaGammaToRef(alpha, beta, gamma, result);\n return result;\n }\n /**\n * Creates a new quaternion from the given Euler float angles expressed in z-x-z orientation and stores it in the target quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#69\n * @param alpha defines the rotation around first axis\n * @param beta defines the rotation around second axis\n * @param gamma defines the rotation around third axis\n * @param result defines the target quaternion\n * @returns result input\n */\n static RotationAlphaBetaGammaToRef(alpha, beta, gamma, result) {\n // Produces a quaternion from Euler angles in the z-x-z orientation\n const halfGammaPlusAlpha = (gamma + alpha) * 0.5;\n const halfGammaMinusAlpha = (gamma - alpha) * 0.5;\n const halfBeta = beta * 0.5;\n result._x = Math.cos(halfGammaMinusAlpha) * Math.sin(halfBeta);\n result._y = Math.sin(halfGammaMinusAlpha) * Math.sin(halfBeta);\n result._z = Math.sin(halfGammaPlusAlpha) * Math.cos(halfBeta);\n result._w = Math.cos(halfGammaPlusAlpha) * Math.cos(halfBeta);\n result._isDirty = true;\n return result;\n }\n /**\n * Creates a new quaternion containing the rotation value to reach the target (axis1, axis2, axis3) orientation as a rotated XYZ system (axis1, axis2 and axis3 are normalized during this operation)\n * Example Playground https://playground.babylonjs.com/#L49EJ7#75\n * @param axis1 defines the first axis\n * @param axis2 defines the second axis\n * @param axis3 defines the third axis\n * @returns the new quaternion\n */\n static RotationQuaternionFromAxis(axis1, axis2, axis3) {\n const quat = new Quaternion(0.0, 0.0, 0.0, 0.0);\n Quaternion.RotationQuaternionFromAxisToRef(axis1, axis2, axis3, quat);\n return quat;\n }\n /**\n * Creates a rotation value to reach the target (axis1, axis2, axis3) orientation as a rotated XYZ system (axis1, axis2 and axis3 are normalized during this operation) and stores it in the target quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#76\n * @param axis1 defines the first axis\n * @param axis2 defines the second axis\n * @param axis3 defines the third axis\n * @param ref defines the target quaternion\n * @returns result input\n */\n static RotationQuaternionFromAxisToRef(axis1, axis2, axis3, ref) {\n const rotMat = MathTmp.Matrix[0];\n Matrix.FromXYZAxesToRef(axis1.normalize(), axis2.normalize(), axis3.normalize(), rotMat);\n Quaternion.FromRotationMatrixToRef(rotMat, ref);\n return ref;\n }\n /**\n * Creates a new rotation value to orient an object to look towards the given forward direction, the up direction being oriented like \"up\".\n * This function works in left handed mode\n * Example Playground https://playground.babylonjs.com/#L49EJ7#96\n * @param forward defines the forward direction - Must be normalized and orthogonal to up.\n * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward.\n * @returns A new quaternion oriented toward the specified forward and up.\n */\n static FromLookDirectionLH(forward, up) {\n const quat = new Quaternion();\n Quaternion.FromLookDirectionLHToRef(forward, up, quat);\n return quat;\n }\n /**\n * Creates a new rotation value to orient an object to look towards the given forward direction with the up direction being oriented like \"up\", and stores it in the target quaternion.\n * This function works in left handed mode\n * Example Playground https://playground.babylonjs.com/#L49EJ7#97\n * @param forward defines the forward direction - Must be normalized and orthogonal to up.\n * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward.\n * @param ref defines the target quaternion.\n * @returns result input\n */\n static FromLookDirectionLHToRef(forward, up, ref) {\n const rotMat = MathTmp.Matrix[0];\n Matrix.LookDirectionLHToRef(forward, up, rotMat);\n Quaternion.FromRotationMatrixToRef(rotMat, ref);\n return ref;\n }\n /**\n * Creates a new rotation value to orient an object to look towards the given forward direction, the up direction being oriented like \"up\".\n * This function works in right handed mode\n * Example Playground https://playground.babylonjs.com/#L49EJ7#98\n * @param forward defines the forward direction - Must be normalized and orthogonal to up.\n * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward.\n * @returns A new quaternion oriented toward the specified forward and up.\n */\n static FromLookDirectionRH(forward, up) {\n const quat = new Quaternion();\n Quaternion.FromLookDirectionRHToRef(forward, up, quat);\n return quat;\n }\n /**\n * Creates a new rotation value to orient an object to look towards the given forward direction with the up direction being oriented like \"up\", and stores it in the target quaternion.\n * This function works in right handed mode\n * Example Playground https://playground.babylonjs.com/#L49EJ7#105\n * @param forward defines the forward direction - Must be normalized and orthogonal to up.\n * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward.\n * @param ref defines the target quaternion.\n * @returns result input\n */\n static FromLookDirectionRHToRef(forward, up, ref) {\n const rotMat = MathTmp.Matrix[0];\n Matrix.LookDirectionRHToRef(forward, up, rotMat);\n return Quaternion.FromRotationMatrixToRef(rotMat, ref);\n }\n /**\n * Interpolates between two quaternions\n * Example Playground https://playground.babylonjs.com/#L49EJ7#79\n * @param left defines first quaternion\n * @param right defines second quaternion\n * @param amount defines the gradient to use\n * @returns the new interpolated quaternion\n */\n static Slerp(left, right, amount) {\n const result = Quaternion.Identity();\n Quaternion.SlerpToRef(left, right, amount, result);\n return result;\n }\n /**\n * Interpolates between two quaternions and stores it into a target quaternion\n * Example Playground https://playground.babylonjs.com/#L49EJ7#92\n * @param left defines first quaternion\n * @param right defines second quaternion\n * @param amount defines the gradient to use\n * @param result defines the target quaternion\n * @returns result input\n */\n static SlerpToRef(left, right, amount, result) {\n let num2;\n let num3;\n let num4 = left._x * right._x + left._y * right._y + left._z * right._z + left._w * right._w;\n let flag = false;\n if (num4 < 0) {\n flag = true;\n num4 = -num4;\n }\n if (num4 > 0.999999) {\n num3 = 1 - amount;\n num2 = flag ? -amount : amount;\n }\n else {\n const num5 = Math.acos(num4);\n const num6 = 1.0 / Math.sin(num5);\n num3 = Math.sin((1.0 - amount) * num5) * num6;\n num2 = flag ? -Math.sin(amount * num5) * num6 : Math.sin(amount * num5) * num6;\n }\n result._x = num3 * left._x + num2 * right._x;\n result._y = num3 * left._y + num2 * right._y;\n result._z = num3 * left._z + num2 * right._z;\n result._w = num3 * left._w + num2 * right._w;\n result._isDirty = true;\n return result;\n }\n /**\n * Interpolate between two quaternions using Hermite interpolation\n * Example Playground https://playground.babylonjs.com/#L49EJ7#47\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/drawCurves#hermite-quaternion-spline\n * @param value1 defines first quaternion\n * @param tangent1 defines the incoming tangent\n * @param value2 defines second quaternion\n * @param tangent2 defines the outgoing tangent\n * @param amount defines the target quaternion\n * @returns the new interpolated quaternion\n */\n static Hermite(value1, tangent1, value2, tangent2, amount) {\n const squared = amount * amount;\n const cubed = amount * squared;\n const part1 = 2.0 * cubed - 3.0 * squared + 1.0;\n const part2 = -2.0 * cubed + 3.0 * squared;\n const part3 = cubed - 2.0 * squared + amount;\n const part4 = cubed - squared;\n const x = value1._x * part1 + value2._x * part2 + tangent1._x * part3 + tangent2._x * part4;\n const y = value1._y * part1 + value2._y * part2 + tangent1._y * part3 + tangent2._y * part4;\n const z = value1._z * part1 + value2._z * part2 + tangent1._z * part3 + tangent2._z * part4;\n const w = value1._w * part1 + value2._w * part2 + tangent1._w * part3 + tangent2._w * part4;\n return new value1.constructor(x, y, z, w);\n }\n /**\n * Returns a new Quaternion which is the 1st derivative of the Hermite spline defined by the quaternions \"value1\", \"value2\", \"tangent1\", \"tangent2\".\n * Example Playground https://playground.babylonjs.com/#L49EJ7#48\n * @param value1 defines the first control point\n * @param tangent1 defines the first tangent\n * @param value2 defines the second control point\n * @param tangent2 defines the second tangent\n * @param time define where the derivative must be done\n * @returns 1st derivative\n */\n static Hermite1stDerivative(value1, tangent1, value2, tangent2, time) {\n const result = new value1.constructor();\n this.Hermite1stDerivativeToRef(value1, tangent1, value2, tangent2, time, result);\n return result;\n }\n /**\n * Update a Quaternion with the 1st derivative of the Hermite spline defined by the quaternions \"value1\", \"value2\", \"tangent1\", \"tangent2\".\n * Example Playground https://playground.babylonjs.com/#L49EJ7#49\n * @param value1 defines the first control point\n * @param tangent1 defines the first tangent\n * @param value2 defines the second control point\n * @param tangent2 defines the second tangent\n * @param time define where the derivative must be done\n * @param result define where to store the derivative\n * @returns result input\n */\n static Hermite1stDerivativeToRef(value1, tangent1, value2, tangent2, time, result) {\n const t2 = time * time;\n result._x = (t2 - time) * 6 * value1._x + (3 * t2 - 4 * time + 1) * tangent1._x + (-t2 + time) * 6 * value2._x + (3 * t2 - 2 * time) * tangent2._x;\n result._y = (t2 - time) * 6 * value1._y + (3 * t2 - 4 * time + 1) * tangent1._y + (-t2 + time) * 6 * value2._y + (3 * t2 - 2 * time) * tangent2._y;\n result._z = (t2 - time) * 6 * value1._z + (3 * t2 - 4 * time + 1) * tangent1._z + (-t2 + time) * 6 * value2._z + (3 * t2 - 2 * time) * tangent2._z;\n result._w = (t2 - time) * 6 * value1._w + (3 * t2 - 4 * time + 1) * tangent1._w + (-t2 + time) * 6 * value2._w + (3 * t2 - 2 * time) * tangent2._w;\n result._isDirty = true;\n return result;\n }\n }\n /**\n * Class used to store matrix data (4x4)\n * Note on matrix definitions in Babylon.js for setting values directly\n * rather than using one of the methods available.\n * Matrix size is given by rows x columns.\n * A Vector3 is a 1 X 3 matrix [x, y, z].\n *\n * In Babylon.js multiplying a 1 x 3 matrix by a 4 x 4 matrix\n * is done using BABYLON.Vector4.TransformCoordinates(Vector3, Matrix).\n * and extending the passed Vector3 to a Vector4, V = [x, y, z, 1].\n * Let M be a matrix with elements m(row, column), so that\n * m(2, 3) is the element in row 2 column 3 of M.\n *\n * Multiplication is of the form VM and has the resulting Vector4\n * VM = [xm(0, 0) + ym(1, 0) + zm(2, 0) + m(3, 0), xm(0, 1) + ym(1, 1) + zm(2, 1) + m(3, 1), xm(0, 2) + ym(1, 2) + zm(2, 2) + m(3, 2), xm(0, 3) + ym(1, 3) + zm(2, 3) + m(3, 3)].\n * On the web you will find many examples that use the opposite convention of MV,\n * in which case to make use of the examples you will need to transpose the matrix.\n *\n * Example Playground - Overview Linear Algebra - https://playground.babylonjs.com/#AV9X17\n * Example Playground - Overview Transformation - https://playground.babylonjs.com/#AV9X17#1\n * Example Playground - Overview Projection - https://playground.babylonjs.com/#AV9X17#2\n */\n class Matrix {\n /**\n * Gets the precision of matrix computations\n */\n static get Use64Bits() {\n return PerformanceConfigurator.MatrixUse64Bits;\n }\n /**\n * Gets the internal data of the matrix\n */\n get m() {\n return this._m;\n }\n /**\n * Update the updateFlag to indicate that the matrix has been updated\n */\n markAsUpdated() {\n this.updateFlag = Matrix._UpdateFlagSeed++;\n this._isIdentity = false;\n this._isIdentity3x2 = false;\n this._isIdentityDirty = true;\n this._isIdentity3x2Dirty = true;\n }\n _updateIdentityStatus(isIdentity, isIdentityDirty = false, isIdentity3x2 = false, isIdentity3x2Dirty = true) {\n this._isIdentity = isIdentity;\n this._isIdentity3x2 = isIdentity || isIdentity3x2;\n this._isIdentityDirty = this._isIdentity ? false : isIdentityDirty;\n this._isIdentity3x2Dirty = this._isIdentity3x2 ? false : isIdentity3x2Dirty;\n }\n /**\n * Creates an empty matrix (filled with zeros)\n */\n constructor() {\n this._isIdentity = false;\n this._isIdentityDirty = true;\n this._isIdentity3x2 = true;\n this._isIdentity3x2Dirty = true;\n /**\n * Gets the update flag of the matrix which is an unique number for the matrix.\n * It will be incremented every time the matrix data change.\n * You can use it to speed the comparison between two versions of the same matrix.\n */\n this.updateFlag = -1;\n if (PerformanceConfigurator.MatrixTrackPrecisionChange) {\n PerformanceConfigurator.MatrixTrackedMatrices.push(this);\n }\n this._m = new PerformanceConfigurator.MatrixCurrentType(16);\n this.markAsUpdated();\n }\n // Properties\n /**\n * Check if the current matrix is identity\n * @returns true is the matrix is the identity matrix\n */\n isIdentity() {\n if (this._isIdentityDirty) {\n this._isIdentityDirty = false;\n const m = this._m;\n this._isIdentity =\n m[0] === 1.0 &&\n m[1] === 0.0 &&\n m[2] === 0.0 &&\n m[3] === 0.0 &&\n m[4] === 0.0 &&\n m[5] === 1.0 &&\n m[6] === 0.0 &&\n m[7] === 0.0 &&\n m[8] === 0.0 &&\n m[9] === 0.0 &&\n m[10] === 1.0 &&\n m[11] === 0.0 &&\n m[12] === 0.0 &&\n m[13] === 0.0 &&\n m[14] === 0.0 &&\n m[15] === 1.0;\n }\n return this._isIdentity;\n }\n /**\n * Check if the current matrix is identity as a texture matrix (3x2 store in 4x4)\n * @returns true is the matrix is the identity matrix\n */\n isIdentityAs3x2() {\n if (this._isIdentity3x2Dirty) {\n this._isIdentity3x2Dirty = false;\n if (this._m[0] !== 1.0 || this._m[5] !== 1.0 || this._m[15] !== 1.0) {\n this._isIdentity3x2 = false;\n }\n else if (this._m[1] !== 0.0 ||\n this._m[2] !== 0.0 ||\n this._m[3] !== 0.0 ||\n this._m[4] !== 0.0 ||\n this._m[6] !== 0.0 ||\n this._m[7] !== 0.0 ||\n this._m[8] !== 0.0 ||\n this._m[9] !== 0.0 ||\n this._m[10] !== 0.0 ||\n this._m[11] !== 0.0 ||\n this._m[12] !== 0.0 ||\n this._m[13] !== 0.0 ||\n this._m[14] !== 0.0) {\n this._isIdentity3x2 = false;\n }\n else {\n this._isIdentity3x2 = true;\n }\n }\n return this._isIdentity3x2;\n }\n /**\n * Gets the determinant of the matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#34\n * @returns the matrix determinant\n */\n determinant() {\n if (this._isIdentity === true) {\n return 1;\n }\n const m = this._m;\n const m00 = m[0], m01 = m[1], m02 = m[2], m03 = m[3];\n const m10 = m[4], m11 = m[5], m12 = m[6], m13 = m[7];\n const m20 = m[8], m21 = m[9], m22 = m[10], m23 = m[11];\n const m30 = m[12], m31 = m[13], m32 = m[14], m33 = m[15];\n // https://en.wikipedia.org/wiki/Laplace_expansion\n // to compute the deterrminant of a 4x4 Matrix we compute the cofactors of any row or column,\n // then we multiply each Cofactor by its corresponding matrix value and sum them all to get the determinant\n // Cofactor(i, j) = sign(i,j) * det(Minor(i, j))\n // where\n // - sign(i,j) = (i+j) % 2 === 0 ? 1 : -1\n // - Minor(i, j) is the 3x3 matrix we get by removing row i and column j from current Matrix\n //\n // Here we do that for the 1st row.\n const det_22_33 = m22 * m33 - m32 * m23;\n const det_21_33 = m21 * m33 - m31 * m23;\n const det_21_32 = m21 * m32 - m31 * m22;\n const det_20_33 = m20 * m33 - m30 * m23;\n const det_20_32 = m20 * m32 - m22 * m30;\n const det_20_31 = m20 * m31 - m30 * m21;\n const cofact_00 = +(m11 * det_22_33 - m12 * det_21_33 + m13 * det_21_32);\n const cofact_01 = -(m10 * det_22_33 - m12 * det_20_33 + m13 * det_20_32);\n const cofact_02 = +(m10 * det_21_33 - m11 * det_20_33 + m13 * det_20_31);\n const cofact_03 = -(m10 * det_21_32 - m11 * det_20_32 + m12 * det_20_31);\n return m00 * cofact_00 + m01 * cofact_01 + m02 * cofact_02 + m03 * cofact_03;\n }\n // Methods\n /**\n * Returns the matrix as a Float32Array or Array\n * Example Playground - https://playground.babylonjs.com/#AV9X17#49\n * @returns the matrix underlying array\n */\n toArray() {\n return this._m;\n }\n /**\n * Returns the matrix as a Float32Array or Array\n * Example Playground - https://playground.babylonjs.com/#AV9X17#114\n * @returns the matrix underlying array.\n */\n asArray() {\n return this._m;\n }\n /**\n * Inverts the current matrix in place\n * Example Playground - https://playground.babylonjs.com/#AV9X17#118\n * @returns the current inverted matrix\n */\n invert() {\n this.invertToRef(this);\n return this;\n }\n /**\n * Sets all the matrix elements to zero\n * @returns the current matrix\n */\n reset() {\n Matrix.FromValuesToRef(0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, this);\n this._updateIdentityStatus(false);\n return this;\n }\n /**\n * Adds the current matrix with a second one\n * Example Playground - https://playground.babylonjs.com/#AV9X17#44\n * @param other defines the matrix to add\n * @returns a new matrix as the addition of the current matrix and the given one\n */\n add(other) {\n const result = new this.constructor();\n this.addToRef(other, result);\n return result;\n }\n /**\n * Sets the given matrix \"result\" to the addition of the current matrix and the given one\n * Example Playground - https://playground.babylonjs.com/#AV9X17#45\n * @param other defines the matrix to add\n * @param result defines the target matrix\n * @returns result input\n */\n addToRef(other, result) {\n const m = this._m;\n const resultM = result._m;\n const otherM = other.m;\n for (let index = 0; index < 16; index++) {\n resultM[index] = m[index] + otherM[index];\n }\n result.markAsUpdated();\n return result;\n }\n /**\n * Adds in place the given matrix to the current matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#46\n * @param other defines the second operand\n * @returns the current updated matrix\n */\n addToSelf(other) {\n const m = this._m;\n const otherM = other.m;\n for (let index = 0; index < 16; index++) {\n m[index] += otherM[index];\n }\n this.markAsUpdated();\n return this;\n }\n /**\n * Sets the given matrix to the current inverted Matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#119\n * @param other defines the target matrix\n * @returns result input\n */\n invertToRef(other) {\n if (this._isIdentity === true) {\n Matrix.IdentityToRef(other);\n return other;\n }\n // the inverse of a Matrix is the transpose of cofactor matrix divided by the determinant\n const m = this._m;\n const m00 = m[0], m01 = m[1], m02 = m[2], m03 = m[3];\n const m10 = m[4], m11 = m[5], m12 = m[6], m13 = m[7];\n const m20 = m[8], m21 = m[9], m22 = m[10], m23 = m[11];\n const m30 = m[12], m31 = m[13], m32 = m[14], m33 = m[15];\n const det_22_33 = m22 * m33 - m32 * m23;\n const det_21_33 = m21 * m33 - m31 * m23;\n const det_21_32 = m21 * m32 - m31 * m22;\n const det_20_33 = m20 * m33 - m30 * m23;\n const det_20_32 = m20 * m32 - m22 * m30;\n const det_20_31 = m20 * m31 - m30 * m21;\n const cofact_00 = +(m11 * det_22_33 - m12 * det_21_33 + m13 * det_21_32);\n const cofact_01 = -(m10 * det_22_33 - m12 * det_20_33 + m13 * det_20_32);\n const cofact_02 = +(m10 * det_21_33 - m11 * det_20_33 + m13 * det_20_31);\n const cofact_03 = -(m10 * det_21_32 - m11 * det_20_32 + m12 * det_20_31);\n const det = m00 * cofact_00 + m01 * cofact_01 + m02 * cofact_02 + m03 * cofact_03;\n if (det === 0) {\n // not invertible\n other.copyFrom(this);\n return other;\n }\n const detInv = 1 / det;\n const det_12_33 = m12 * m33 - m32 * m13;\n const det_11_33 = m11 * m33 - m31 * m13;\n const det_11_32 = m11 * m32 - m31 * m12;\n const det_10_33 = m10 * m33 - m30 * m13;\n const det_10_32 = m10 * m32 - m30 * m12;\n const det_10_31 = m10 * m31 - m30 * m11;\n const det_12_23 = m12 * m23 - m22 * m13;\n const det_11_23 = m11 * m23 - m21 * m13;\n const det_11_22 = m11 * m22 - m21 * m12;\n const det_10_23 = m10 * m23 - m20 * m13;\n const det_10_22 = m10 * m22 - m20 * m12;\n const det_10_21 = m10 * m21 - m20 * m11;\n const cofact_10 = -(m01 * det_22_33 - m02 * det_21_33 + m03 * det_21_32);\n const cofact_11 = +(m00 * det_22_33 - m02 * det_20_33 + m03 * det_20_32);\n const cofact_12 = -(m00 * det_21_33 - m01 * det_20_33 + m03 * det_20_31);\n const cofact_13 = +(m00 * det_21_32 - m01 * det_20_32 + m02 * det_20_31);\n const cofact_20 = +(m01 * det_12_33 - m02 * det_11_33 + m03 * det_11_32);\n const cofact_21 = -(m00 * det_12_33 - m02 * det_10_33 + m03 * det_10_32);\n const cofact_22 = +(m00 * det_11_33 - m01 * det_10_33 + m03 * det_10_31);\n const cofact_23 = -(m00 * det_11_32 - m01 * det_10_32 + m02 * det_10_31);\n const cofact_30 = -(m01 * det_12_23 - m02 * det_11_23 + m03 * det_11_22);\n const cofact_31 = +(m00 * det_12_23 - m02 * det_10_23 + m03 * det_10_22);\n const cofact_32 = -(m00 * det_11_23 - m01 * det_10_23 + m03 * det_10_21);\n const cofact_33 = +(m00 * det_11_22 - m01 * det_10_22 + m02 * det_10_21);\n Matrix.FromValuesToRef(cofact_00 * detInv, cofact_10 * detInv, cofact_20 * detInv, cofact_30 * detInv, cofact_01 * detInv, cofact_11 * detInv, cofact_21 * detInv, cofact_31 * detInv, cofact_02 * detInv, cofact_12 * detInv, cofact_22 * detInv, cofact_32 * detInv, cofact_03 * detInv, cofact_13 * detInv, cofact_23 * detInv, cofact_33 * detInv, other);\n return other;\n }\n /**\n * add a value at the specified position in the current Matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#47\n * @param index the index of the value within the matrix. between 0 and 15.\n * @param value the value to be added\n * @returns the current updated matrix\n */\n addAtIndex(index, value) {\n this._m[index] += value;\n this.markAsUpdated();\n return this;\n }\n /**\n * mutiply the specified position in the current Matrix by a value\n * @param index the index of the value within the matrix. between 0 and 15.\n * @param value the value to be added\n * @returns the current updated matrix\n */\n multiplyAtIndex(index, value) {\n this._m[index] *= value;\n this.markAsUpdated();\n return this;\n }\n /**\n * Inserts the translation vector (using 3 floats) in the current matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#120\n * @param x defines the 1st component of the translation\n * @param y defines the 2nd component of the translation\n * @param z defines the 3rd component of the translation\n * @returns the current updated matrix\n */\n setTranslationFromFloats(x, y, z) {\n this._m[12] = x;\n this._m[13] = y;\n this._m[14] = z;\n this.markAsUpdated();\n return this;\n }\n /**\n * Adds the translation vector (using 3 floats) in the current matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#20\n * Example Playground - https://playground.babylonjs.com/#AV9X17#48\n * @param x defines the 1st component of the translation\n * @param y defines the 2nd component of the translation\n * @param z defines the 3rd component of the translation\n * @returns the current updated matrix\n */\n addTranslationFromFloats(x, y, z) {\n this._m[12] += x;\n this._m[13] += y;\n this._m[14] += z;\n this.markAsUpdated();\n return this;\n }\n /**\n * Inserts the translation vector in the current matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#121\n * @param vector3 defines the translation to insert\n * @returns the current updated matrix\n */\n setTranslation(vector3) {\n return this.setTranslationFromFloats(vector3._x, vector3._y, vector3._z);\n }\n /**\n * Gets the translation value of the current matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#122\n * @returns a new Vector3 as the extracted translation from the matrix\n */\n getTranslation() {\n return new Vector3(this._m[12], this._m[13], this._m[14]);\n }\n /**\n * Fill a Vector3 with the extracted translation from the matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#123\n * @param result defines the Vector3 where to store the translation\n * @returns the current matrix\n */\n getTranslationToRef(result) {\n result.x = this._m[12];\n result.y = this._m[13];\n result.z = this._m[14];\n return result;\n }\n /**\n * Remove rotation and scaling part from the matrix\n * @returns the updated matrix\n */\n removeRotationAndScaling() {\n const m = this.m;\n Matrix.FromValuesToRef(1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, m[12], m[13], m[14], m[15], this);\n this._updateIdentityStatus(m[12] === 0 && m[13] === 0 && m[14] === 0 && m[15] === 1);\n return this;\n }\n /**\n * Multiply two matrices\n * Example Playground - https://playground.babylonjs.com/#AV9X17#15\n * A.multiply(B) means apply B to A so result is B x A\n * @param other defines the second operand\n * @returns a new matrix set with the multiplication result of the current Matrix and the given one\n */\n multiply(other) {\n const result = new this.constructor();\n this.multiplyToRef(other, result);\n return result;\n }\n /**\n * Copy the current matrix from the given one\n * Example Playground - https://playground.babylonjs.com/#AV9X17#21\n * @param other defines the source matrix\n * @returns the current updated matrix\n */\n copyFrom(other) {\n other.copyToArray(this._m);\n const o = other;\n this.updateFlag = o.updateFlag;\n this._updateIdentityStatus(o._isIdentity, o._isIdentityDirty, o._isIdentity3x2, o._isIdentity3x2Dirty);\n return this;\n }\n /**\n * Populates the given array from the starting index with the current matrix values\n * @param array defines the target array\n * @param offset defines the offset in the target array where to start storing values\n * @returns the current matrix\n */\n copyToArray(array, offset = 0) {\n const source = this._m;\n array[offset] = source[0];\n array[offset + 1] = source[1];\n array[offset + 2] = source[2];\n array[offset + 3] = source[3];\n array[offset + 4] = source[4];\n array[offset + 5] = source[5];\n array[offset + 6] = source[6];\n array[offset + 7] = source[7];\n array[offset + 8] = source[8];\n array[offset + 9] = source[9];\n array[offset + 10] = source[10];\n array[offset + 11] = source[11];\n array[offset + 12] = source[12];\n array[offset + 13] = source[13];\n array[offset + 14] = source[14];\n array[offset + 15] = source[15];\n return this;\n }\n /**\n * Sets the given matrix \"result\" with the multiplication result of the current Matrix and the given one\n * A.multiplyToRef(B, R) means apply B to A and store in R and R = B x A\n * Example Playground - https://playground.babylonjs.com/#AV9X17#16\n * @param other defines the second operand\n * @param result defines the matrix where to store the multiplication\n * @returns result input\n */\n multiplyToRef(other, result) {\n if (this._isIdentity) {\n result.copyFrom(other);\n return result;\n }\n if (other._isIdentity) {\n result.copyFrom(this);\n return result;\n }\n this.multiplyToArray(other, result._m, 0);\n result.markAsUpdated();\n return result;\n }\n /**\n * Sets the Float32Array \"result\" from the given index \"offset\" with the multiplication of the current matrix and the given one\n * @param other defines the second operand\n * @param result defines the array where to store the multiplication\n * @param offset defines the offset in the target array where to start storing values\n * @returns the current matrix\n */\n multiplyToArray(other, result, offset) {\n const m = this._m;\n const otherM = other.m;\n const tm0 = m[0], tm1 = m[1], tm2 = m[2], tm3 = m[3];\n const tm4 = m[4], tm5 = m[5], tm6 = m[6], tm7 = m[7];\n const tm8 = m[8], tm9 = m[9], tm10 = m[10], tm11 = m[11];\n const tm12 = m[12], tm13 = m[13], tm14 = m[14], tm15 = m[15];\n const om0 = otherM[0], om1 = otherM[1], om2 = otherM[2], om3 = otherM[3];\n const om4 = otherM[4], om5 = otherM[5], om6 = otherM[6], om7 = otherM[7];\n const om8 = otherM[8], om9 = otherM[9], om10 = otherM[10], om11 = otherM[11];\n const om12 = otherM[12], om13 = otherM[13], om14 = otherM[14], om15 = otherM[15];\n result[offset] = tm0 * om0 + tm1 * om4 + tm2 * om8 + tm3 * om12;\n result[offset + 1] = tm0 * om1 + tm1 * om5 + tm2 * om9 + tm3 * om13;\n result[offset + 2] = tm0 * om2 + tm1 * om6 + tm2 * om10 + tm3 * om14;\n result[offset + 3] = tm0 * om3 + tm1 * om7 + tm2 * om11 + tm3 * om15;\n result[offset + 4] = tm4 * om0 + tm5 * om4 + tm6 * om8 + tm7 * om12;\n result[offset + 5] = tm4 * om1 + tm5 * om5 + tm6 * om9 + tm7 * om13;\n result[offset + 6] = tm4 * om2 + tm5 * om6 + tm6 * om10 + tm7 * om14;\n result[offset + 7] = tm4 * om3 + tm5 * om7 + tm6 * om11 + tm7 * om15;\n result[offset + 8] = tm8 * om0 + tm9 * om4 + tm10 * om8 + tm11 * om12;\n result[offset + 9] = tm8 * om1 + tm9 * om5 + tm10 * om9 + tm11 * om13;\n result[offset + 10] = tm8 * om2 + tm9 * om6 + tm10 * om10 + tm11 * om14;\n result[offset + 11] = tm8 * om3 + tm9 * om7 + tm10 * om11 + tm11 * om15;\n result[offset + 12] = tm12 * om0 + tm13 * om4 + tm14 * om8 + tm15 * om12;\n result[offset + 13] = tm12 * om1 + tm13 * om5 + tm14 * om9 + tm15 * om13;\n result[offset + 14] = tm12 * om2 + tm13 * om6 + tm14 * om10 + tm15 * om14;\n result[offset + 15] = tm12 * om3 + tm13 * om7 + tm14 * om11 + tm15 * om15;\n return this;\n }\n /**\n * Check equality between this matrix and a second one\n * @param value defines the second matrix to compare\n * @returns true is the current matrix and the given one values are strictly equal\n */\n equals(value) {\n const other = value;\n if (!other) {\n return false;\n }\n if (this._isIdentity || other._isIdentity) {\n if (!this._isIdentityDirty && !other._isIdentityDirty) {\n return this._isIdentity && other._isIdentity;\n }\n }\n const m = this.m;\n const om = other.m;\n return (m[0] === om[0] &&\n m[1] === om[1] &&\n m[2] === om[2] &&\n m[3] === om[3] &&\n m[4] === om[4] &&\n m[5] === om[5] &&\n m[6] === om[6] &&\n m[7] === om[7] &&\n m[8] === om[8] &&\n m[9] === om[9] &&\n m[10] === om[10] &&\n m[11] === om[11] &&\n m[12] === om[12] &&\n m[13] === om[13] &&\n m[14] === om[14] &&\n m[15] === om[15]);\n }\n /**\n * Clone the current matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#18\n * @returns a new matrix from the current matrix\n */\n clone() {\n const matrix = new this.constructor();\n matrix.copyFrom(this);\n return matrix;\n }\n /**\n * Returns the name of the current matrix class\n * @returns the string \"Matrix\"\n */\n getClassName() {\n return \"Matrix\";\n }\n /**\n * Gets the hash code of the current matrix\n * @returns the hash code\n */\n getHashCode() {\n let hash = _ExtractAsInt(this._m[0]);\n for (let i = 1; i < 16; i++) {\n hash = (hash * 397) ^ _ExtractAsInt(this._m[i]);\n }\n return hash;\n }\n /**\n * Decomposes the current Matrix into a translation, rotation and scaling components of the provided node\n * Example Playground - https://playground.babylonjs.com/#AV9X17#13\n * @param node the node to decompose the matrix to\n * @returns true if operation was successful\n */\n decomposeToTransformNode(node) {\n node.rotationQuaternion = node.rotationQuaternion || new Quaternion();\n return this.decompose(node.scaling, node.rotationQuaternion, node.position);\n }\n /**\n * Decomposes the current Matrix into a translation, rotation and scaling components\n * Example Playground - https://playground.babylonjs.com/#AV9X17#12\n * @param scale defines the scale vector3 given as a reference to update\n * @param rotation defines the rotation quaternion given as a reference to update\n * @param translation defines the translation vector3 given as a reference to update\n * @param preserveScalingNode Use scaling sign coming from this node. Otherwise scaling sign might change.\n * @returns true if operation was successful\n */\n decompose(scale, rotation, translation, preserveScalingNode) {\n if (this._isIdentity) {\n if (translation) {\n translation.setAll(0);\n }\n if (scale) {\n scale.setAll(1);\n }\n if (rotation) {\n rotation.copyFromFloats(0, 0, 0, 1);\n }\n return true;\n }\n const m = this._m;\n if (translation) {\n translation.copyFromFloats(m[12], m[13], m[14]);\n }\n scale = scale || MathTmp.Vector3[0];\n scale.x = Math.sqrt(m[0] * m[0] + m[1] * m[1] + m[2] * m[2]);\n scale.y = Math.sqrt(m[4] * m[4] + m[5] * m[5] + m[6] * m[6]);\n scale.z = Math.sqrt(m[8] * m[8] + m[9] * m[9] + m[10] * m[10]);\n if (preserveScalingNode) {\n const signX = preserveScalingNode.scaling.x < 0 ? -1 : 1;\n const signY = preserveScalingNode.scaling.y < 0 ? -1 : 1;\n const signZ = preserveScalingNode.scaling.z < 0 ? -1 : 1;\n scale.x *= signX;\n scale.y *= signY;\n scale.z *= signZ;\n }\n else {\n if (this.determinant() <= 0) {\n scale.y *= -1;\n }\n }\n if (scale._x === 0 || scale._y === 0 || scale._z === 0) {\n if (rotation) {\n rotation.copyFromFloats(0.0, 0.0, 0.0, 1.0);\n }\n return false;\n }\n if (rotation) {\n const sx = 1 / scale._x, sy = 1 / scale._y, sz = 1 / scale._z;\n Matrix.FromValuesToRef(m[0] * sx, m[1] * sx, m[2] * sx, 0.0, m[4] * sy, m[5] * sy, m[6] * sy, 0.0, m[8] * sz, m[9] * sz, m[10] * sz, 0.0, 0.0, 0.0, 0.0, 1.0, MathTmp.Matrix[0]);\n Quaternion.FromRotationMatrixToRef(MathTmp.Matrix[0], rotation);\n }\n return true;\n }\n /**\n * Gets specific row of the matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#36\n * @param index defines the number of the row to get\n * @returns the index-th row of the current matrix as a new Vector4\n */\n getRow(index) {\n if (index < 0 || index > 3) {\n return null;\n }\n const i = index * 4;\n return new Vector4(this._m[i + 0], this._m[i + 1], this._m[i + 2], this._m[i + 3]);\n }\n /**\n * Gets specific row of the matrix to ref\n * Example Playground - https://playground.babylonjs.com/#AV9X17#36\n * @param index defines the number of the row to get\n * @param rowVector vector to store the index-th row of the current matrix\n * @returns result input\n */\n getRowToRef(index, rowVector) {\n if (index >= 0 && index < 3) {\n const i = index * 4;\n rowVector.x = this._m[i + 0];\n rowVector.y = this._m[i + 1];\n rowVector.z = this._m[i + 2];\n rowVector.w = this._m[i + 3];\n }\n return rowVector;\n }\n /**\n * Sets the index-th row of the current matrix to the vector4 values\n * Example Playground - https://playground.babylonjs.com/#AV9X17#36\n * @param index defines the number of the row to set\n * @param row defines the target vector4\n * @returns the updated current matrix\n */\n setRow(index, row) {\n return this.setRowFromFloats(index, row.x, row.y, row.z, row.w);\n }\n /**\n * Compute the transpose of the matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#40\n * @returns the new transposed matrix\n */\n transpose() {\n const result = new this.constructor();\n Matrix.TransposeToRef(this, result);\n return result;\n }\n /**\n * Compute the transpose of the matrix and store it in a given matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#41\n * @param result defines the target matrix\n * @returns result input\n */\n transposeToRef(result) {\n Matrix.TransposeToRef(this, result);\n return result;\n }\n /**\n * Sets the index-th row of the current matrix with the given 4 x float values\n * Example Playground - https://playground.babylonjs.com/#AV9X17#36\n * @param index defines the row index\n * @param x defines the x component to set\n * @param y defines the y component to set\n * @param z defines the z component to set\n * @param w defines the w component to set\n * @returns the updated current matrix\n */\n setRowFromFloats(index, x, y, z, w) {\n if (index < 0 || index > 3) {\n return this;\n }\n const i = index * 4;\n this._m[i + 0] = x;\n this._m[i + 1] = y;\n this._m[i + 2] = z;\n this._m[i + 3] = w;\n this.markAsUpdated();\n return this;\n }\n /**\n * Compute a new matrix set with the current matrix values multiplied by scale (float)\n * @param scale defines the scale factor\n * @returns a new matrix\n */\n scale(scale) {\n const result = new this.constructor();\n this.scaleToRef(scale, result);\n return result;\n }\n /**\n * Scale the current matrix values by a factor to a given result matrix\n * @param scale defines the scale factor\n * @param result defines the matrix to store the result\n * @returns result input\n */\n scaleToRef(scale, result) {\n for (let index = 0; index < 16; index++) {\n result._m[index] = this._m[index] * scale;\n }\n result.markAsUpdated();\n return result;\n }\n /**\n * Scale the current matrix values by a factor and add the result to a given matrix\n * @param scale defines the scale factor\n * @param result defines the Matrix to store the result\n * @returns result input\n */\n scaleAndAddToRef(scale, result) {\n for (let index = 0; index < 16; index++) {\n result._m[index] += this._m[index] * scale;\n }\n result.markAsUpdated();\n return result;\n }\n /**\n * Writes to the given matrix a normal matrix, computed from this one (using values from identity matrix for fourth row and column).\n * Example Playground - https://playground.babylonjs.com/#AV9X17#17\n * @param ref matrix to store the result\n */\n toNormalMatrix(ref) {\n const tmp = MathTmp.Matrix[0];\n this.invertToRef(tmp);\n tmp.transposeToRef(ref);\n const m = ref._m;\n Matrix.FromValuesToRef(m[0], m[1], m[2], 0.0, m[4], m[5], m[6], 0.0, m[8], m[9], m[10], 0.0, 0.0, 0.0, 0.0, 1.0, ref);\n return ref;\n }\n /**\n * Gets only rotation part of the current matrix\n * @returns a new matrix sets to the extracted rotation matrix from the current one\n */\n getRotationMatrix() {\n const result = new this.constructor();\n this.getRotationMatrixToRef(result);\n return result;\n }\n /**\n * Extracts the rotation matrix from the current one and sets it as the given \"result\"\n * @param result defines the target matrix to store data to\n * @returns result input\n */\n getRotationMatrixToRef(result) {\n const scale = MathTmp.Vector3[0];\n if (!this.decompose(scale)) {\n Matrix.IdentityToRef(result);\n return result;\n }\n const m = this._m;\n const sx = 1 / scale._x, sy = 1 / scale._y, sz = 1 / scale._z;\n Matrix.FromValuesToRef(m[0] * sx, m[1] * sx, m[2] * sx, 0.0, m[4] * sy, m[5] * sy, m[6] * sy, 0.0, m[8] * sz, m[9] * sz, m[10] * sz, 0.0, 0.0, 0.0, 0.0, 1.0, result);\n return result;\n }\n /**\n * Toggles model matrix from being right handed to left handed in place and vice versa\n */\n toggleModelMatrixHandInPlace() {\n const m = this._m;\n m[2] *= -1;\n m[6] *= -1;\n m[8] *= -1;\n m[9] *= -1;\n m[14] *= -1;\n this.markAsUpdated();\n return this;\n }\n /**\n * Toggles projection matrix from being right handed to left handed in place and vice versa\n */\n toggleProjectionMatrixHandInPlace() {\n const m = this._m;\n m[8] *= -1;\n m[9] *= -1;\n m[10] *= -1;\n m[11] *= -1;\n this.markAsUpdated();\n return this;\n }\n // Statics\n /**\n * Creates a matrix from an array\n * Example Playground - https://playground.babylonjs.com/#AV9X17#42\n * @param array defines the source array\n * @param offset defines an offset in the source array\n * @returns a new Matrix set from the starting index of the given array\n */\n static FromArray(array, offset = 0) {\n const result = new Matrix();\n Matrix.FromArrayToRef(array, offset, result);\n return result;\n }\n /**\n * Copy the content of an array into a given matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#43\n * @param array defines the source array\n * @param offset defines an offset in the source array\n * @param result defines the target matrix\n * @returns result input\n */\n static FromArrayToRef(array, offset, result) {\n for (let index = 0; index < 16; index++) {\n result._m[index] = array[index + offset];\n }\n result.markAsUpdated();\n return result;\n }\n /**\n * Stores an array into a matrix after having multiplied each component by a given factor\n * Example Playground - https://playground.babylonjs.com/#AV9X17#50\n * @param array defines the source array\n * @param offset defines the offset in the source array\n * @param scale defines the scaling factor\n * @param result defines the target matrix\n * @returns result input\n */\n static FromFloat32ArrayToRefScaled(array, offset, scale, result) {\n for (let index = 0; index < 16; index++) {\n result._m[index] = array[index + offset] * scale;\n }\n result.markAsUpdated();\n return result;\n }\n /**\n * Gets an identity matrix that must not be updated\n */\n static get IdentityReadOnly() {\n return Matrix._IdentityReadOnly;\n }\n /**\n * Stores a list of values (16) inside a given matrix\n * @param initialM11 defines 1st value of 1st row\n * @param initialM12 defines 2nd value of 1st row\n * @param initialM13 defines 3rd value of 1st row\n * @param initialM14 defines 4th value of 1st row\n * @param initialM21 defines 1st value of 2nd row\n * @param initialM22 defines 2nd value of 2nd row\n * @param initialM23 defines 3rd value of 2nd row\n * @param initialM24 defines 4th value of 2nd row\n * @param initialM31 defines 1st value of 3rd row\n * @param initialM32 defines 2nd value of 3rd row\n * @param initialM33 defines 3rd value of 3rd row\n * @param initialM34 defines 4th value of 3rd row\n * @param initialM41 defines 1st value of 4th row\n * @param initialM42 defines 2nd value of 4th row\n * @param initialM43 defines 3rd value of 4th row\n * @param initialM44 defines 4th value of 4th row\n * @param result defines the target matrix\n * @returns result input\n */\n static FromValuesToRef(initialM11, initialM12, initialM13, initialM14, initialM21, initialM22, initialM23, initialM24, initialM31, initialM32, initialM33, initialM34, initialM41, initialM42, initialM43, initialM44, result) {\n const m = result._m;\n m[0] = initialM11;\n m[1] = initialM12;\n m[2] = initialM13;\n m[3] = initialM14;\n m[4] = initialM21;\n m[5] = initialM22;\n m[6] = initialM23;\n m[7] = initialM24;\n m[8] = initialM31;\n m[9] = initialM32;\n m[10] = initialM33;\n m[11] = initialM34;\n m[12] = initialM41;\n m[13] = initialM42;\n m[14] = initialM43;\n m[15] = initialM44;\n result.markAsUpdated();\n }\n /**\n * Creates new matrix from a list of values (16)\n * @param initialM11 defines 1st value of 1st row\n * @param initialM12 defines 2nd value of 1st row\n * @param initialM13 defines 3rd value of 1st row\n * @param initialM14 defines 4th value of 1st row\n * @param initialM21 defines 1st value of 2nd row\n * @param initialM22 defines 2nd value of 2nd row\n * @param initialM23 defines 3rd value of 2nd row\n * @param initialM24 defines 4th value of 2nd row\n * @param initialM31 defines 1st value of 3rd row\n * @param initialM32 defines 2nd value of 3rd row\n * @param initialM33 defines 3rd value of 3rd row\n * @param initialM34 defines 4th value of 3rd row\n * @param initialM41 defines 1st value of 4th row\n * @param initialM42 defines 2nd value of 4th row\n * @param initialM43 defines 3rd value of 4th row\n * @param initialM44 defines 4th value of 4th row\n * @returns the new matrix\n */\n static FromValues(initialM11, initialM12, initialM13, initialM14, initialM21, initialM22, initialM23, initialM24, initialM31, initialM32, initialM33, initialM34, initialM41, initialM42, initialM43, initialM44) {\n const result = new Matrix();\n const m = result._m;\n m[0] = initialM11;\n m[1] = initialM12;\n m[2] = initialM13;\n m[3] = initialM14;\n m[4] = initialM21;\n m[5] = initialM22;\n m[6] = initialM23;\n m[7] = initialM24;\n m[8] = initialM31;\n m[9] = initialM32;\n m[10] = initialM33;\n m[11] = initialM34;\n m[12] = initialM41;\n m[13] = initialM42;\n m[14] = initialM43;\n m[15] = initialM44;\n result.markAsUpdated();\n return result;\n }\n /**\n * Creates a new matrix composed by merging scale (vector3), rotation (quaternion) and translation (vector3)\n * Example Playground - https://playground.babylonjs.com/#AV9X17#24\n * @param scale defines the scale vector3\n * @param rotation defines the rotation quaternion\n * @param translation defines the translation vector3\n * @returns a new matrix\n */\n static Compose(scale, rotation, translation) {\n const result = new Matrix();\n Matrix.ComposeToRef(scale, rotation, translation, result);\n return result;\n }\n /**\n * Sets a matrix to a value composed by merging scale (vector3), rotation (quaternion) and translation (vector3)\n * Example Playground - https://playground.babylonjs.com/#AV9X17#25\n * @param scale defines the scale vector3\n * @param rotation defines the rotation quaternion\n * @param translation defines the translation vector3\n * @param result defines the target matrix\n * @returns result input\n */\n static ComposeToRef(scale, rotation, translation, result) {\n const m = result._m;\n const x = rotation._x, y = rotation._y, z = rotation._z, w = rotation._w;\n const x2 = x + x, y2 = y + y, z2 = z + z;\n const xx = x * x2, xy = x * y2, xz = x * z2;\n const yy = y * y2, yz = y * z2, zz = z * z2;\n const wx = w * x2, wy = w * y2, wz = w * z2;\n const sx = scale._x, sy = scale._y, sz = scale._z;\n m[0] = (1 - (yy + zz)) * sx;\n m[1] = (xy + wz) * sx;\n m[2] = (xz - wy) * sx;\n m[3] = 0;\n m[4] = (xy - wz) * sy;\n m[5] = (1 - (xx + zz)) * sy;\n m[6] = (yz + wx) * sy;\n m[7] = 0;\n m[8] = (xz + wy) * sz;\n m[9] = (yz - wx) * sz;\n m[10] = (1 - (xx + yy)) * sz;\n m[11] = 0;\n m[12] = translation._x;\n m[13] = translation._y;\n m[14] = translation._z;\n m[15] = 1;\n result.markAsUpdated();\n return result;\n }\n /**\n * Creates a new identity matrix\n * @returns a new identity matrix\n */\n static Identity() {\n const identity = Matrix.FromValues(1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 0.0, 1.0);\n identity._updateIdentityStatus(true);\n return identity;\n }\n /**\n * Creates a new identity matrix and stores the result in a given matrix\n * @param result defines the target matrix\n * @returns result input\n */\n static IdentityToRef(result) {\n Matrix.FromValuesToRef(1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 0.0, 1.0, result);\n result._updateIdentityStatus(true);\n return result;\n }\n /**\n * Creates a new zero matrix\n * @returns a new zero matrix\n */\n static Zero() {\n const zero = Matrix.FromValues(0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0);\n zero._updateIdentityStatus(false);\n return zero;\n }\n /**\n * Creates a new rotation matrix for \"angle\" radians around the X axis\n * Example Playground - https://playground.babylonjs.com/#AV9X17#97\n * @param angle defines the angle (in radians) to use\n * @returns the new matrix\n */\n static RotationX(angle) {\n const result = new Matrix();\n Matrix.RotationXToRef(angle, result);\n return result;\n }\n /**\n * Creates a new matrix as the invert of a given matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#124\n * @param source defines the source matrix\n * @returns the new matrix\n */\n static Invert(source) {\n const result = new source.constructor();\n source.invertToRef(result);\n return result;\n }\n /**\n * Creates a new rotation matrix for \"angle\" radians around the X axis and stores it in a given matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#98\n * @param angle defines the angle (in radians) to use\n * @param result defines the target matrix\n * @returns result input\n */\n static RotationXToRef(angle, result) {\n const s = Math.sin(angle);\n const c = Math.cos(angle);\n Matrix.FromValuesToRef(1.0, 0.0, 0.0, 0.0, 0.0, c, s, 0.0, 0.0, -s, c, 0.0, 0.0, 0.0, 0.0, 1.0, result);\n result._updateIdentityStatus(c === 1 && s === 0);\n return result;\n }\n /**\n * Creates a new rotation matrix for \"angle\" radians around the Y axis\n * Example Playground - https://playground.babylonjs.com/#AV9X17#99\n * @param angle defines the angle (in radians) to use\n * @returns the new matrix\n */\n static RotationY(angle) {\n const result = new Matrix();\n Matrix.RotationYToRef(angle, result);\n return result;\n }\n /**\n * Creates a new rotation matrix for \"angle\" radians around the Y axis and stores it in a given matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#100\n * @param angle defines the angle (in radians) to use\n * @param result defines the target matrix\n * @returns result input\n */\n static RotationYToRef(angle, result) {\n const s = Math.sin(angle);\n const c = Math.cos(angle);\n Matrix.FromValuesToRef(c, 0.0, -s, 0.0, 0.0, 1.0, 0.0, 0.0, s, 0.0, c, 0.0, 0.0, 0.0, 0.0, 1.0, result);\n result._updateIdentityStatus(c === 1 && s === 0);\n return result;\n }\n /**\n * Creates a new rotation matrix for \"angle\" radians around the Z axis\n * Example Playground - https://playground.babylonjs.com/#AV9X17#101\n * @param angle defines the angle (in radians) to use\n * @returns the new matrix\n */\n static RotationZ(angle) {\n const result = new Matrix();\n Matrix.RotationZToRef(angle, result);\n return result;\n }\n /**\n * Creates a new rotation matrix for \"angle\" radians around the Z axis and stores it in a given matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#102\n * @param angle defines the angle (in radians) to use\n * @param result defines the target matrix\n * @returns result input\n */\n static RotationZToRef(angle, result) {\n const s = Math.sin(angle);\n const c = Math.cos(angle);\n Matrix.FromValuesToRef(c, s, 0.0, 0.0, -s, c, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 0.0, 1.0, result);\n result._updateIdentityStatus(c === 1 && s === 0);\n return result;\n }\n /**\n * Creates a new rotation matrix for \"angle\" radians around the given axis\n * Example Playground - https://playground.babylonjs.com/#AV9X17#96\n * @param axis defines the axis to use\n * @param angle defines the angle (in radians) to use\n * @returns the new matrix\n */\n static RotationAxis(axis, angle) {\n const result = new Matrix();\n Matrix.RotationAxisToRef(axis, angle, result);\n return result;\n }\n /**\n * Creates a new rotation matrix for \"angle\" radians around the given axis and stores it in a given matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#94\n * @param axis defines the axis to use\n * @param angle defines the angle (in radians) to use\n * @param result defines the target matrix\n * @returns result input\n */\n static RotationAxisToRef(axis, angle, result) {\n const s = Math.sin(-angle);\n const c = Math.cos(-angle);\n const c1 = 1 - c;\n axis.normalize();\n const m = result._m;\n m[0] = axis._x * axis._x * c1 + c;\n m[1] = axis._x * axis._y * c1 - axis._z * s;\n m[2] = axis._x * axis._z * c1 + axis._y * s;\n m[3] = 0.0;\n m[4] = axis._y * axis._x * c1 + axis._z * s;\n m[5] = axis._y * axis._y * c1 + c;\n m[6] = axis._y * axis._z * c1 - axis._x * s;\n m[7] = 0.0;\n m[8] = axis._z * axis._x * c1 - axis._y * s;\n m[9] = axis._z * axis._y * c1 + axis._x * s;\n m[10] = axis._z * axis._z * c1 + c;\n m[11] = 0.0;\n m[12] = 0.0;\n m[13] = 0.0;\n m[14] = 0.0;\n m[15] = 1.0;\n result.markAsUpdated();\n return result;\n }\n /**\n * Takes normalised vectors and returns a rotation matrix to align \"from\" with \"to\".\n * Taken from http://www.iquilezles.org/www/articles/noacos/noacos.htm\n * Example Playground - https://playground.babylonjs.com/#AV9X17#93\n * @param from defines the vector to align\n * @param to defines the vector to align to\n * @param result defines the target matrix\n * @returns result input\n */\n static RotationAlignToRef(from, to, result) {\n const c = Vector3.Dot(to, from);\n const m = result._m;\n if (c < -1 + Epsilon) {\n // from and to are colinear and opposite direction.\n // compute a PI rotation on Z axis\n m[0] = -1;\n m[1] = 0;\n m[2] = 0;\n m[3] = 0;\n m[4] = 0;\n m[5] = -1;\n m[6] = 0;\n m[7] = 0;\n m[8] = 0;\n m[9] = 0;\n m[10] = 1;\n m[11] = 0;\n }\n else {\n const v = Vector3.Cross(to, from);\n const k = 1 / (1 + c);\n m[0] = v._x * v._x * k + c;\n m[1] = v._y * v._x * k - v._z;\n m[2] = v._z * v._x * k + v._y;\n m[3] = 0;\n m[4] = v._x * v._y * k + v._z;\n m[5] = v._y * v._y * k + c;\n m[6] = v._z * v._y * k - v._x;\n m[7] = 0;\n m[8] = v._x * v._z * k - v._y;\n m[9] = v._y * v._z * k + v._x;\n m[10] = v._z * v._z * k + c;\n m[11] = 0;\n }\n m[12] = 0;\n m[13] = 0;\n m[14] = 0;\n m[15] = 1;\n result.markAsUpdated();\n return result;\n }\n /**\n * Creates a rotation matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#103\n * Example Playground - https://playground.babylonjs.com/#AV9X17#105\n * @param yaw defines the yaw angle in radians (Y axis)\n * @param pitch defines the pitch angle in radians (X axis)\n * @param roll defines the roll angle in radians (Z axis)\n * @returns the new rotation matrix\n */\n static RotationYawPitchRoll(yaw, pitch, roll) {\n const result = new Matrix();\n Matrix.RotationYawPitchRollToRef(yaw, pitch, roll, result);\n return result;\n }\n /**\n * Creates a rotation matrix and stores it in a given matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#104\n * @param yaw defines the yaw angle in radians (Y axis)\n * @param pitch defines the pitch angle in radians (X axis)\n * @param roll defines the roll angle in radians (Z axis)\n * @param result defines the target matrix\n * @returns result input\n */\n static RotationYawPitchRollToRef(yaw, pitch, roll, result) {\n Quaternion.RotationYawPitchRollToRef(yaw, pitch, roll, MathTmp.Quaternion[0]);\n MathTmp.Quaternion[0].toRotationMatrix(result);\n return result;\n }\n /**\n * Creates a scaling matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#107\n * @param x defines the scale factor on X axis\n * @param y defines the scale factor on Y axis\n * @param z defines the scale factor on Z axis\n * @returns the new matrix\n */\n static Scaling(x, y, z) {\n const result = new Matrix();\n Matrix.ScalingToRef(x, y, z, result);\n return result;\n }\n /**\n * Creates a scaling matrix and stores it in a given matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#108\n * @param x defines the scale factor on X axis\n * @param y defines the scale factor on Y axis\n * @param z defines the scale factor on Z axis\n * @param result defines the target matrix\n * @returns result input\n */\n static ScalingToRef(x, y, z, result) {\n Matrix.FromValuesToRef(x, 0.0, 0.0, 0.0, 0.0, y, 0.0, 0.0, 0.0, 0.0, z, 0.0, 0.0, 0.0, 0.0, 1.0, result);\n result._updateIdentityStatus(x === 1 && y === 1 && z === 1);\n return result;\n }\n /**\n * Creates a translation matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#109\n * @param x defines the translation on X axis\n * @param y defines the translation on Y axis\n * @param z defines the translationon Z axis\n * @returns the new matrix\n */\n static Translation(x, y, z) {\n const result = new Matrix();\n Matrix.TranslationToRef(x, y, z, result);\n return result;\n }\n /**\n * Creates a translation matrix and stores it in a given matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#110\n * @param x defines the translation on X axis\n * @param y defines the translation on Y axis\n * @param z defines the translationon Z axis\n * @param result defines the target matrix\n * @returns result input\n */\n static TranslationToRef(x, y, z, result) {\n Matrix.FromValuesToRef(1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, x, y, z, 1.0, result);\n result._updateIdentityStatus(x === 0 && y === 0 && z === 0);\n return result;\n }\n /**\n * Returns a new Matrix whose values are the interpolated values for \"gradient\" (float) between the ones of the matrices \"startValue\" and \"endValue\".\n * Example Playground - https://playground.babylonjs.com/#AV9X17#55\n * @param startValue defines the start value\n * @param endValue defines the end value\n * @param gradient defines the gradient factor\n * @returns the new matrix\n */\n static Lerp(startValue, endValue, gradient) {\n const result = new startValue.constructor();\n Matrix.LerpToRef(startValue, endValue, gradient, result);\n return result;\n }\n /**\n * Set the given matrix \"result\" as the interpolated values for \"gradient\" (float) between the ones of the matrices \"startValue\" and \"endValue\".\n * Example Playground - https://playground.babylonjs.com/#AV9X17#54\n * @param startValue defines the start value\n * @param endValue defines the end value\n * @param gradient defines the gradient factor\n * @param result defines the Matrix object where to store data\n * @returns result input\n */\n static LerpToRef(startValue, endValue, gradient, result) {\n const resultM = result._m;\n const startM = startValue.m;\n const endM = endValue.m;\n for (let index = 0; index < 16; index++) {\n resultM[index] = startM[index] * (1.0 - gradient) + endM[index] * gradient;\n }\n result.markAsUpdated();\n return result;\n }\n /**\n * Builds a new matrix whose values are computed by:\n * * decomposing the the \"startValue\" and \"endValue\" matrices into their respective scale, rotation and translation matrices\n * * interpolating for \"gradient\" (float) the values between each of these decomposed matrices between the start and the end\n * * recomposing a new matrix from these 3 interpolated scale, rotation and translation matrices\n * Example Playground - https://playground.babylonjs.com/#AV9X17#22\n * Example Playground - https://playground.babylonjs.com/#AV9X17#51\n * @param startValue defines the first matrix\n * @param endValue defines the second matrix\n * @param gradient defines the gradient between the two matrices\n * @returns the new matrix\n */\n static DecomposeLerp(startValue, endValue, gradient) {\n const result = new startValue.constructor();\n Matrix.DecomposeLerpToRef(startValue, endValue, gradient, result);\n return result;\n }\n /**\n * Update a matrix to values which are computed by:\n * * decomposing the the \"startValue\" and \"endValue\" matrices into their respective scale, rotation and translation matrices\n * * interpolating for \"gradient\" (float) the values between each of these decomposed matrices between the start and the end\n * * recomposing a new matrix from these 3 interpolated scale, rotation and translation matrices\n * Example Playground - https://playground.babylonjs.com/#AV9X17#23\n * Example Playground - https://playground.babylonjs.com/#AV9X17#53\n * @param startValue defines the first matrix\n * @param endValue defines the second matrix\n * @param gradient defines the gradient between the two matrices\n * @param result defines the target matrix\n * @returns result input\n */\n static DecomposeLerpToRef(startValue, endValue, gradient, result) {\n const startScale = MathTmp.Vector3[0];\n const startRotation = MathTmp.Quaternion[0];\n const startTranslation = MathTmp.Vector3[1];\n startValue.decompose(startScale, startRotation, startTranslation);\n const endScale = MathTmp.Vector3[2];\n const endRotation = MathTmp.Quaternion[1];\n const endTranslation = MathTmp.Vector3[3];\n endValue.decompose(endScale, endRotation, endTranslation);\n const resultScale = MathTmp.Vector3[4];\n Vector3.LerpToRef(startScale, endScale, gradient, resultScale);\n const resultRotation = MathTmp.Quaternion[2];\n Quaternion.SlerpToRef(startRotation, endRotation, gradient, resultRotation);\n const resultTranslation = MathTmp.Vector3[5];\n Vector3.LerpToRef(startTranslation, endTranslation, gradient, resultTranslation);\n Matrix.ComposeToRef(resultScale, resultRotation, resultTranslation, result);\n return result;\n }\n /**\n * Creates a new matrix that transforms vertices from world space to camera space. It takes three vectors as arguments that together describe the position and orientation of the camera.\n * This function generates a matrix suitable for a left handed coordinate system\n * Example Playground - https://playground.babylonjs.com/#AV9X17#58\n * Example Playground - https://playground.babylonjs.com/#AV9X17#59\n * @param eye defines the final position of the entity\n * @param target defines where the entity should look at\n * @param up defines the up vector for the entity\n * @returns the new matrix\n */\n static LookAtLH(eye, target, up) {\n const result = new Matrix();\n Matrix.LookAtLHToRef(eye, target, up, result);\n return result;\n }\n /**\n * Sets the given \"result\" Matrix to a matrix that transforms vertices from world space to camera space. It takes three vectors as arguments that together describe the position and orientation of the camera.\n * This function generates a matrix suitable for a left handed coordinate system\n * Example Playground - https://playground.babylonjs.com/#AV9X17#60\n * Example Playground - https://playground.babylonjs.com/#AV9X17#61\n * @param eye defines the final position of the entity\n * @param target defines where the entity should look at\n * @param up defines the up vector for the entity\n * @param result defines the target matrix\n * @returns result input\n */\n static LookAtLHToRef(eye, target, up, result) {\n const xAxis = MathTmp.Vector3[0];\n const yAxis = MathTmp.Vector3[1];\n const zAxis = MathTmp.Vector3[2];\n // Z axis\n target.subtractToRef(eye, zAxis);\n zAxis.normalize();\n // X axis\n Vector3.CrossToRef(up, zAxis, xAxis);\n const xSquareLength = xAxis.lengthSquared();\n if (xSquareLength === 0) {\n xAxis.x = 1.0;\n }\n else {\n xAxis.normalizeFromLength(Math.sqrt(xSquareLength));\n }\n // Y axis\n Vector3.CrossToRef(zAxis, xAxis, yAxis);\n yAxis.normalize();\n // Eye angles\n const ex = -Vector3.Dot(xAxis, eye);\n const ey = -Vector3.Dot(yAxis, eye);\n const ez = -Vector3.Dot(zAxis, eye);\n Matrix.FromValuesToRef(xAxis._x, yAxis._x, zAxis._x, 0.0, xAxis._y, yAxis._y, zAxis._y, 0.0, xAxis._z, yAxis._z, zAxis._z, 0.0, ex, ey, ez, 1.0, result);\n }\n /**\n * Creates a new matrix that transforms vertices from world space to camera space. It takes three vectors as arguments that together describe the position and orientation of the camera.\n * This function generates a matrix suitable for a right handed coordinate system\n * Example Playground - https://playground.babylonjs.com/#AV9X17#62\n * Example Playground - https://playground.babylonjs.com/#AV9X17#63\n * @param eye defines the final position of the entity\n * @param target defines where the entity should look at\n * @param up defines the up vector for the entity\n * @returns the new matrix\n */\n static LookAtRH(eye, target, up) {\n const result = new Matrix();\n Matrix.LookAtRHToRef(eye, target, up, result);\n return result;\n }\n /**\n * Sets the given \"result\" Matrix to a matrix that transforms vertices from world space to camera space. It takes three vectors as arguments that together describe the position and orientation of the camera.\n * This function generates a matrix suitable for a right handed coordinate system\n * Example Playground - https://playground.babylonjs.com/#AV9X17#64\n * Example Playground - https://playground.babylonjs.com/#AV9X17#65\n * @param eye defines the final position of the entity\n * @param target defines where the entity should look at\n * @param up defines the up vector for the entity\n * @param result defines the target matrix\n * @returns result input\n */\n static LookAtRHToRef(eye, target, up, result) {\n const xAxis = MathTmp.Vector3[0];\n const yAxis = MathTmp.Vector3[1];\n const zAxis = MathTmp.Vector3[2];\n // Z axis\n eye.subtractToRef(target, zAxis);\n zAxis.normalize();\n // X axis\n Vector3.CrossToRef(up, zAxis, xAxis);\n const xSquareLength = xAxis.lengthSquared();\n if (xSquareLength === 0) {\n xAxis.x = 1.0;\n }\n else {\n xAxis.normalizeFromLength(Math.sqrt(xSquareLength));\n }\n // Y axis\n Vector3.CrossToRef(zAxis, xAxis, yAxis);\n yAxis.normalize();\n // Eye angles\n const ex = -Vector3.Dot(xAxis, eye);\n const ey = -Vector3.Dot(yAxis, eye);\n const ez = -Vector3.Dot(zAxis, eye);\n Matrix.FromValuesToRef(xAxis._x, yAxis._x, zAxis._x, 0.0, xAxis._y, yAxis._y, zAxis._y, 0.0, xAxis._z, yAxis._z, zAxis._z, 0.0, ex, ey, ez, 1.0, result);\n return result;\n }\n /**\n * Creates a new matrix that transforms vertices from world space to camera space. It takes two vectors as arguments that together describe the orientation of the camera. The position is assumed to be at the origin (0,0,0)\n * This function generates a matrix suitable for a left handed coordinate system\n * Example Playground - https://playground.babylonjs.com/#AV9X17#66\n * @param forward defines the forward direction - Must be normalized and orthogonal to up.\n * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward.\n * @returns the new matrix\n */\n static LookDirectionLH(forward, up) {\n const result = new Matrix();\n Matrix.LookDirectionLHToRef(forward, up, result);\n return result;\n }\n /**\n * Sets the given \"result\" Matrix to a matrix that transforms vertices from world space to camera space. It takes two vectors as arguments that together describe the orientation of the camera. The position is assumed to be at the origin (0,0,0)\n * This function generates a matrix suitable for a left handed coordinate system\n * Example Playground - https://playground.babylonjs.com/#AV9X17#67\n * @param forward defines the forward direction - Must be normalized and orthogonal to up.\n * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward.\n * @param result defines the target matrix\n * @returns result input\n */\n static LookDirectionLHToRef(forward, up, result) {\n const back = MathTmp.Vector3[0];\n back.copyFrom(forward);\n back.scaleInPlace(-1);\n const left = MathTmp.Vector3[1];\n Vector3.CrossToRef(up, back, left);\n // Generate the rotation matrix.\n Matrix.FromValuesToRef(left._x, left._y, left._z, 0.0, up._x, up._y, up._z, 0.0, back._x, back._y, back._z, 0.0, 0, 0, 0, 1.0, result);\n return result;\n }\n /**\n * Creates a new matrix that transforms vertices from world space to camera space. It takes two vectors as arguments that together describe the orientation of the camera. The position is assumed to be at the origin (0,0,0)\n * This function generates a matrix suitable for a right handed coordinate system\n * Example Playground - https://playground.babylonjs.com/#AV9X17#68\n * @param forward defines the forward direction - Must be normalized and orthogonal to up.\n * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward.\n * @returns the new matrix\n */\n static LookDirectionRH(forward, up) {\n const result = new Matrix();\n Matrix.LookDirectionRHToRef(forward, up, result);\n return result;\n }\n /**\n * Sets the given \"result\" Matrix to a matrix that transforms vertices from world space to camera space. It takes two vectors as arguments that together describe the orientation of the camera. The position is assumed to be at the origin (0,0,0)\n * This function generates a matrix suitable for a right handed coordinate system\n * Example Playground - https://playground.babylonjs.com/#AV9X17#69\n * @param forward defines the forward direction - Must be normalized and orthogonal to up.\n * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward.\n * @param result defines the target matrix\n * @returns result input\n */\n static LookDirectionRHToRef(forward, up, result) {\n const right = MathTmp.Vector3[2];\n Vector3.CrossToRef(up, forward, right);\n // Generate the rotation matrix.\n Matrix.FromValuesToRef(right._x, right._y, right._z, 0.0, up._x, up._y, up._z, 0.0, forward._x, forward._y, forward._z, 0.0, 0, 0, 0, 1.0, result);\n return result;\n }\n /**\n * Create a left-handed orthographic projection matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#70\n * @param width defines the viewport width\n * @param height defines the viewport height\n * @param znear defines the near clip plane\n * @param zfar defines the far clip plane\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\n * @returns a new matrix as a left-handed orthographic projection matrix\n */\n static OrthoLH(width, height, znear, zfar, halfZRange) {\n const matrix = new Matrix();\n Matrix.OrthoLHToRef(width, height, znear, zfar, matrix, halfZRange);\n return matrix;\n }\n /**\n * Store a left-handed orthographic projection to a given matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#71\n * @param width defines the viewport width\n * @param height defines the viewport height\n * @param znear defines the near clip plane\n * @param zfar defines the far clip plane\n * @param result defines the target matrix\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\n * @returns result input\n */\n static OrthoLHToRef(width, height, znear, zfar, result, halfZRange) {\n const n = znear;\n const f = zfar;\n const a = 2.0 / width;\n const b = 2.0 / height;\n const c = 2.0 / (f - n);\n const d = -(f + n) / (f - n);\n Matrix.FromValuesToRef(a, 0.0, 0.0, 0.0, 0.0, b, 0.0, 0.0, 0.0, 0.0, c, 0.0, 0.0, 0.0, d, 1.0, result);\n if (halfZRange) {\n result.multiplyToRef(mtxConvertNDCToHalfZRange, result);\n }\n result._updateIdentityStatus(a === 1 && b === 1 && c === 1 && d === 0);\n return result;\n }\n /**\n * Create a left-handed orthographic projection matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#72\n * @param left defines the viewport left coordinate\n * @param right defines the viewport right coordinate\n * @param bottom defines the viewport bottom coordinate\n * @param top defines the viewport top coordinate\n * @param znear defines the near clip plane\n * @param zfar defines the far clip plane\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\n * @returns a new matrix as a left-handed orthographic projection matrix\n */\n static OrthoOffCenterLH(left, right, bottom, top, znear, zfar, halfZRange) {\n const matrix = new Matrix();\n Matrix.OrthoOffCenterLHToRef(left, right, bottom, top, znear, zfar, matrix, halfZRange);\n return matrix;\n }\n /**\n * Stores a left-handed orthographic projection into a given matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#73\n * @param left defines the viewport left coordinate\n * @param right defines the viewport right coordinate\n * @param bottom defines the viewport bottom coordinate\n * @param top defines the viewport top coordinate\n * @param znear defines the near clip plane\n * @param zfar defines the far clip plane\n * @param result defines the target matrix\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\n * @returns result input\n */\n static OrthoOffCenterLHToRef(left, right, bottom, top, znear, zfar, result, halfZRange) {\n const n = znear;\n const f = zfar;\n const a = 2.0 / (right - left);\n const b = 2.0 / (top - bottom);\n const c = 2.0 / (f - n);\n const d = -(f + n) / (f - n);\n const i0 = (left + right) / (left - right);\n const i1 = (top + bottom) / (bottom - top);\n Matrix.FromValuesToRef(a, 0.0, 0.0, 0.0, 0.0, b, 0.0, 0.0, 0.0, 0.0, c, 0.0, i0, i1, d, 1.0, result);\n if (halfZRange) {\n result.multiplyToRef(mtxConvertNDCToHalfZRange, result);\n }\n result.markAsUpdated();\n return result;\n }\n /**\n * Creates a right-handed orthographic projection matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#76\n * @param left defines the viewport left coordinate\n * @param right defines the viewport right coordinate\n * @param bottom defines the viewport bottom coordinate\n * @param top defines the viewport top coordinate\n * @param znear defines the near clip plane\n * @param zfar defines the far clip plane\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\n * @returns a new matrix as a right-handed orthographic projection matrix\n */\n static OrthoOffCenterRH(left, right, bottom, top, znear, zfar, halfZRange) {\n const matrix = new Matrix();\n Matrix.OrthoOffCenterRHToRef(left, right, bottom, top, znear, zfar, matrix, halfZRange);\n return matrix;\n }\n /**\n * Stores a right-handed orthographic projection into a given matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#77\n * @param left defines the viewport left coordinate\n * @param right defines the viewport right coordinate\n * @param bottom defines the viewport bottom coordinate\n * @param top defines the viewport top coordinate\n * @param znear defines the near clip plane\n * @param zfar defines the far clip plane\n * @param result defines the target matrix\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\n * @returns result input\n */\n static OrthoOffCenterRHToRef(left, right, bottom, top, znear, zfar, result, halfZRange) {\n Matrix.OrthoOffCenterLHToRef(left, right, bottom, top, znear, zfar, result, halfZRange);\n result._m[10] *= -1; // No need to call markAsUpdated as previous function already called it and let _isIdentityDirty to true\n return result;\n }\n /**\n * Creates a left-handed perspective projection matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#85\n * @param width defines the viewport width\n * @param height defines the viewport height\n * @param znear defines the near clip plane\n * @param zfar defines the far clip plane\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\n * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal)\n * @returns a new matrix as a left-handed perspective projection matrix\n */\n static PerspectiveLH(width, height, znear, zfar, halfZRange, projectionPlaneTilt = 0) {\n const matrix = new Matrix();\n const n = znear;\n const f = zfar;\n const a = (2.0 * n) / width;\n const b = (2.0 * n) / height;\n const c = (f + n) / (f - n);\n const d = (-2.0 * f * n) / (f - n);\n const rot = Math.tan(projectionPlaneTilt);\n Matrix.FromValuesToRef(a, 0.0, 0.0, 0.0, 0.0, b, 0.0, rot, 0.0, 0.0, c, 1.0, 0.0, 0.0, d, 0.0, matrix);\n if (halfZRange) {\n matrix.multiplyToRef(mtxConvertNDCToHalfZRange, matrix);\n }\n matrix._updateIdentityStatus(false);\n return matrix;\n }\n /**\n * Creates a left-handed perspective projection matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#78\n * @param fov defines the horizontal field of view\n * @param aspect defines the aspect ratio\n * @param znear defines the near clip plane\n * @param zfar defines the far clip plane. If 0, assume we are in \"infinite zfar\" mode\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\n * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal)\n * @param reverseDepthBufferMode true to indicate that we are in a reverse depth buffer mode (meaning znear and zfar have been inverted when calling the function)\n * @returns a new matrix as a left-handed perspective projection matrix\n */\n static PerspectiveFovLH(fov, aspect, znear, zfar, halfZRange, projectionPlaneTilt = 0, reverseDepthBufferMode = false) {\n const matrix = new Matrix();\n Matrix.PerspectiveFovLHToRef(fov, aspect, znear, zfar, matrix, true, halfZRange, projectionPlaneTilt, reverseDepthBufferMode);\n return matrix;\n }\n /**\n * Stores a left-handed perspective projection into a given matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#81\n * @param fov defines the horizontal field of view\n * @param aspect defines the aspect ratio\n * @param znear defines the near clip plane\n * @param zfar defines the far clip plane. If 0, assume we are in \"infinite zfar\" mode\n * @param result defines the target matrix\n * @param isVerticalFovFixed defines it the fov is vertically fixed (default) or horizontally\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\n * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal)\n * @param reverseDepthBufferMode true to indicate that we are in a reverse depth buffer mode (meaning znear and zfar have been inverted when calling the function)\n * @returns result input\n */\n static PerspectiveFovLHToRef(fov, aspect, znear, zfar, result, isVerticalFovFixed = true, halfZRange, projectionPlaneTilt = 0, reverseDepthBufferMode = false) {\n const n = znear;\n const f = zfar;\n const t = 1.0 / Math.tan(fov * 0.5);\n const a = isVerticalFovFixed ? t / aspect : t;\n const b = isVerticalFovFixed ? t : t * aspect;\n const c = reverseDepthBufferMode && n === 0 ? -1 : f !== 0 ? (f + n) / (f - n) : 1;\n const d = reverseDepthBufferMode && n === 0 ? 2 * f : f !== 0 ? (-2.0 * f * n) / (f - n) : -2 * n;\n const rot = Math.tan(projectionPlaneTilt);\n Matrix.FromValuesToRef(a, 0.0, 0.0, 0.0, 0.0, b, 0.0, rot, 0.0, 0.0, c, 1.0, 0.0, 0.0, d, 0.0, result);\n if (halfZRange) {\n result.multiplyToRef(mtxConvertNDCToHalfZRange, result);\n }\n result._updateIdentityStatus(false);\n return result;\n }\n /**\n * Stores a left-handed perspective projection into a given matrix with depth reversed\n * Example Playground - https://playground.babylonjs.com/#AV9X17#89\n * @param fov defines the horizontal field of view\n * @param aspect defines the aspect ratio\n * @param znear defines the near clip plane\n * @param zfar not used as infinity is used as far clip\n * @param result defines the target matrix\n * @param isVerticalFovFixed defines it the fov is vertically fixed (default) or horizontally\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\n * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal)\n * @returns result input\n */\n static PerspectiveFovReverseLHToRef(fov, aspect, znear, zfar, result, isVerticalFovFixed = true, halfZRange, projectionPlaneTilt = 0) {\n const t = 1.0 / Math.tan(fov * 0.5);\n const a = isVerticalFovFixed ? t / aspect : t;\n const b = isVerticalFovFixed ? t : t * aspect;\n const rot = Math.tan(projectionPlaneTilt);\n Matrix.FromValuesToRef(a, 0.0, 0.0, 0.0, 0.0, b, 0.0, rot, 0.0, 0.0, -znear, 1.0, 0.0, 0.0, 1.0, 0.0, result);\n if (halfZRange) {\n result.multiplyToRef(mtxConvertNDCToHalfZRange, result);\n }\n result._updateIdentityStatus(false);\n return result;\n }\n /**\n * Creates a right-handed perspective projection matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#83\n * @param fov defines the horizontal field of view\n * @param aspect defines the aspect ratio\n * @param znear defines the near clip plane\n * @param zfar defines the far clip plane. If 0, assume we are in \"infinite zfar\" mode\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\n * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal)\n * @param reverseDepthBufferMode true to indicate that we are in a reverse depth buffer mode (meaning znear and zfar have been inverted when calling the function)\n * @returns a new matrix as a right-handed perspective projection matrix\n */\n static PerspectiveFovRH(fov, aspect, znear, zfar, halfZRange, projectionPlaneTilt = 0, reverseDepthBufferMode = false) {\n const matrix = new Matrix();\n Matrix.PerspectiveFovRHToRef(fov, aspect, znear, zfar, matrix, true, halfZRange, projectionPlaneTilt, reverseDepthBufferMode);\n return matrix;\n }\n /**\n * Stores a right-handed perspective projection into a given matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#84\n * @param fov defines the horizontal field of view\n * @param aspect defines the aspect ratio\n * @param znear defines the near clip plane\n * @param zfar defines the far clip plane. If 0, assume we are in \"infinite zfar\" mode\n * @param result defines the target matrix\n * @param isVerticalFovFixed defines it the fov is vertically fixed (default) or horizontally\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\n * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal)\n * @param reverseDepthBufferMode true to indicate that we are in a reverse depth buffer mode (meaning znear and zfar have been inverted when calling the function)\n * @returns result input\n */\n static PerspectiveFovRHToRef(fov, aspect, znear, zfar, result, isVerticalFovFixed = true, halfZRange, projectionPlaneTilt = 0, reverseDepthBufferMode = false) {\n //alternatively this could be expressed as:\n // m = PerspectiveFovLHToRef\n // m[10] *= -1.0;\n // m[11] *= -1.0;\n const n = znear;\n const f = zfar;\n const t = 1.0 / Math.tan(fov * 0.5);\n const a = isVerticalFovFixed ? t / aspect : t;\n const b = isVerticalFovFixed ? t : t * aspect;\n const c = reverseDepthBufferMode && n === 0 ? 1 : f !== 0 ? -(f + n) / (f - n) : -1;\n const d = reverseDepthBufferMode && n === 0 ? 2 * f : f !== 0 ? (-2 * f * n) / (f - n) : -2 * n;\n const rot = Math.tan(projectionPlaneTilt);\n Matrix.FromValuesToRef(a, 0.0, 0.0, 0.0, 0.0, b, 0.0, rot, 0.0, 0.0, c, -1.0, 0.0, 0.0, d, 0.0, result);\n if (halfZRange) {\n result.multiplyToRef(mtxConvertNDCToHalfZRange, result);\n }\n result._updateIdentityStatus(false);\n return result;\n }\n /**\n * Stores a right-handed perspective projection into a given matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#90\n * @param fov defines the horizontal field of view\n * @param aspect defines the aspect ratio\n * @param znear defines the near clip plane\n * @param zfar not used as infinity is used as far clip\n * @param result defines the target matrix\n * @param isVerticalFovFixed defines it the fov is vertically fixed (default) or horizontally\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\n * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal)\n * @returns result input\n */\n static PerspectiveFovReverseRHToRef(fov, aspect, znear, zfar, result, isVerticalFovFixed = true, halfZRange, projectionPlaneTilt = 0) {\n const t = 1.0 / Math.tan(fov * 0.5);\n const a = isVerticalFovFixed ? t / aspect : t;\n const b = isVerticalFovFixed ? t : t * aspect;\n const rot = Math.tan(projectionPlaneTilt);\n Matrix.FromValuesToRef(a, 0.0, 0.0, 0.0, 0.0, b, 0.0, rot, 0.0, 0.0, -znear, -1.0, 0.0, 0.0, -1.0, 0.0, result);\n if (halfZRange) {\n result.multiplyToRef(mtxConvertNDCToHalfZRange, result);\n }\n result._updateIdentityStatus(false);\n return result;\n }\n /**\n * Stores a perspective projection for WebVR info a given matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#92\n * @param fov defines the field of view\n * @param fov.upDegrees\n * @param fov.downDegrees\n * @param fov.leftDegrees\n * @param fov.rightDegrees\n * @param znear defines the near clip plane\n * @param zfar defines the far clip plane\n * @param result defines the target matrix\n * @param rightHanded defines if the matrix must be in right-handed mode (false by default)\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\n * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal)\n * @returns result input\n */\n static PerspectiveFovWebVRToRef(fov, znear, zfar, result, rightHanded = false, halfZRange, projectionPlaneTilt = 0) {\n const rightHandedFactor = rightHanded ? -1 : 1;\n const upTan = Math.tan((fov.upDegrees * Math.PI) / 180.0);\n const downTan = Math.tan((fov.downDegrees * Math.PI) / 180.0);\n const leftTan = Math.tan((fov.leftDegrees * Math.PI) / 180.0);\n const rightTan = Math.tan((fov.rightDegrees * Math.PI) / 180.0);\n const xScale = 2.0 / (leftTan + rightTan);\n const yScale = 2.0 / (upTan + downTan);\n const rot = Math.tan(projectionPlaneTilt);\n const m = result._m;\n m[0] = xScale;\n m[1] = m[2] = m[3] = m[4] = 0.0;\n m[5] = yScale;\n m[6] = 0.0;\n m[7] = rot;\n m[8] = (leftTan - rightTan) * xScale * 0.5;\n m[9] = -((upTan - downTan) * yScale * 0.5);\n m[10] = -zfar / (znear - zfar);\n m[11] = 1.0 * rightHandedFactor;\n m[12] = m[13] = m[15] = 0.0;\n m[14] = -(2.0 * zfar * znear) / (zfar - znear);\n if (halfZRange) {\n result.multiplyToRef(mtxConvertNDCToHalfZRange, result);\n }\n result.markAsUpdated();\n return result;\n }\n /**\n * Computes a complete transformation matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#113\n * @param viewport defines the viewport to use\n * @param world defines the world matrix\n * @param view defines the view matrix\n * @param projection defines the projection matrix\n * @param zmin defines the near clip plane\n * @param zmax defines the far clip plane\n * @returns the transformation matrix\n */\n static GetFinalMatrix(viewport, world, view, projection, zmin, zmax) {\n const cw = viewport.width;\n const ch = viewport.height;\n const cx = viewport.x;\n const cy = viewport.y;\n const viewportMatrix = Matrix.FromValues(cw / 2.0, 0.0, 0.0, 0.0, 0.0, -ch / 2.0, 0.0, 0.0, 0.0, 0.0, zmax - zmin, 0.0, cx + cw / 2.0, ch / 2.0 + cy, zmin, 1.0);\n const matrix = new world.constructor();\n world.multiplyToRef(view, matrix);\n matrix.multiplyToRef(projection, matrix);\n return matrix.multiplyToRef(viewportMatrix, matrix);\n }\n /**\n * Extracts a 2x2 matrix from a given matrix and store the result in a Float32Array\n * @param matrix defines the matrix to use\n * @returns a new Float32Array array with 4 elements : the 2x2 matrix extracted from the given matrix\n */\n static GetAsMatrix2x2(matrix) {\n const m = matrix.m;\n const arr = [m[0], m[1], m[4], m[5]];\n return PerformanceConfigurator.MatrixUse64Bits ? arr : new Float32Array(arr);\n }\n /**\n * Extracts a 3x3 matrix from a given matrix and store the result in a Float32Array\n * @param matrix defines the matrix to use\n * @returns a new Float32Array array with 9 elements : the 3x3 matrix extracted from the given matrix\n */\n static GetAsMatrix3x3(matrix) {\n const m = matrix.m;\n const arr = [m[0], m[1], m[2], m[4], m[5], m[6], m[8], m[9], m[10]];\n return PerformanceConfigurator.MatrixUse64Bits ? arr : new Float32Array(arr);\n }\n /**\n * Compute the transpose of a given matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#111\n * @param matrix defines the matrix to transpose\n * @returns the new matrix\n */\n static Transpose(matrix) {\n const result = new matrix.constructor();\n Matrix.TransposeToRef(matrix, result);\n return result;\n }\n /**\n * Compute the transpose of a matrix and store it in a target matrix\n * Example Playground - https://playground.babylonjs.com/#AV9X17#112\n * @param matrix defines the matrix to transpose\n * @param result defines the target matrix\n * @returns result input\n */\n static TransposeToRef(matrix, result) {\n const rm = result._m;\n const mm = matrix.m;\n rm[0] = mm[0];\n rm[1] = mm[4];\n rm[2] = mm[8];\n rm[3] = mm[12];\n rm[4] = mm[1];\n rm[5] = mm[5];\n rm[6] = mm[9];\n rm[7] = mm[13];\n rm[8] = mm[2];\n rm[9] = mm[6];\n rm[10] = mm[10];\n rm[11] = mm[14];\n rm[12] = mm[3];\n rm[13] = mm[7];\n rm[14] = mm[11];\n rm[15] = mm[15];\n result.markAsUpdated();\n // identity-ness does not change when transposing\n result._updateIdentityStatus(matrix._isIdentity, matrix._isIdentityDirty);\n return result;\n }\n /**\n * Computes a reflection matrix from a plane\n * Example Playground - https://playground.babylonjs.com/#AV9X17#87\n * @param plane defines the reflection plane\n * @returns a new matrix\n */\n static Reflection(plane) {\n const matrix = new Matrix();\n Matrix.ReflectionToRef(plane, matrix);\n return matrix;\n }\n /**\n * Computes a reflection matrix from a plane\n * Example Playground - https://playground.babylonjs.com/#AV9X17#88\n * @param plane defines the reflection plane\n * @param result defines the target matrix\n * @returns result input\n */\n static ReflectionToRef(plane, result) {\n plane.normalize();\n const x = plane.normal.x;\n const y = plane.normal.y;\n const z = plane.normal.z;\n const temp = -2 * x;\n const temp2 = -2 * y;\n const temp3 = -2 * z;\n Matrix.FromValuesToRef(temp * x + 1, temp2 * x, temp3 * x, 0.0, temp * y, temp2 * y + 1, temp3 * y, 0.0, temp * z, temp2 * z, temp3 * z + 1, 0.0, temp * plane.d, temp2 * plane.d, temp3 * plane.d, 1.0, result);\n return result;\n }\n /**\n * Sets the given matrix as a rotation matrix composed from the 3 left handed axes\n * @param xaxis defines the value of the 1st axis\n * @param yaxis defines the value of the 2nd axis\n * @param zaxis defines the value of the 3rd axis\n * @param result defines the target matrix\n * @returns result input\n */\n static FromXYZAxesToRef(xaxis, yaxis, zaxis, result) {\n Matrix.FromValuesToRef(xaxis._x, xaxis._y, xaxis._z, 0.0, yaxis._x, yaxis._y, yaxis._z, 0.0, zaxis._x, zaxis._y, zaxis._z, 0.0, 0.0, 0.0, 0.0, 1.0, result);\n return result;\n }\n /**\n * Creates a rotation matrix from a quaternion and stores it in a target matrix\n * @param quat defines the quaternion to use\n * @param result defines the target matrix\n * @returns result input\n */\n static FromQuaternionToRef(quat, result) {\n const xx = quat._x * quat._x;\n const yy = quat._y * quat._y;\n const zz = quat._z * quat._z;\n const xy = quat._x * quat._y;\n const zw = quat._z * quat._w;\n const zx = quat._z * quat._x;\n const yw = quat._y * quat._w;\n const yz = quat._y * quat._z;\n const xw = quat._x * quat._w;\n result._m[0] = 1.0 - 2.0 * (yy + zz);\n result._m[1] = 2.0 * (xy + zw);\n result._m[2] = 2.0 * (zx - yw);\n result._m[3] = 0.0;\n result._m[4] = 2.0 * (xy - zw);\n result._m[5] = 1.0 - 2.0 * (zz + xx);\n result._m[6] = 2.0 * (yz + xw);\n result._m[7] = 0.0;\n result._m[8] = 2.0 * (zx + yw);\n result._m[9] = 2.0 * (yz - xw);\n result._m[10] = 1.0 - 2.0 * (yy + xx);\n result._m[11] = 0.0;\n result._m[12] = 0.0;\n result._m[13] = 0.0;\n result._m[14] = 0.0;\n result._m[15] = 1.0;\n result.markAsUpdated();\n return result;\n }\n }\n Matrix._UpdateFlagSeed = 0;\n Matrix._IdentityReadOnly = Matrix.Identity();\n /**\n * @internal\n * Same as Tmp but not exported to keep it only for math functions to avoid conflicts\n */\n class MathTmp {\n }\n MathTmp.Vector3 = ArrayTools.BuildTuple(11, Vector3.Zero);\n MathTmp.Matrix = ArrayTools.BuildTuple(2, Matrix.Identity);\n MathTmp.Quaternion = ArrayTools.BuildTuple(3, Quaternion.Zero);\n /**\n * @internal\n */\n class TmpVectors {\n }\n TmpVectors.Vector2 = ArrayTools.BuildTuple(3, Vector2.Zero); // 3 temp Vector2 at once should be enough\n TmpVectors.Vector3 = ArrayTools.BuildTuple(13, Vector3.Zero); // 13 temp Vector3 at once should be enough\n TmpVectors.Vector4 = ArrayTools.BuildTuple(3, Vector4.Zero); // 3 temp Vector4 at once should be enough\n TmpVectors.Quaternion = ArrayTools.BuildTuple(2, Quaternion.Zero); // 2 temp Quaternion at once should be enough\n TmpVectors.Matrix = ArrayTools.BuildTuple(8, Matrix.Identity); // 8 temp Matrices at once should be enough\n RegisterClass(\"BABYLON.Vector2\", Vector2);\n RegisterClass(\"BABYLON.Vector3\", Vector3);\n RegisterClass(\"BABYLON.Vector4\", Vector4);\n RegisterClass(\"BABYLON.Matrix\", Matrix);\n const mtxConvertNDCToHalfZRange = Matrix.FromValues(1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0.5, 0, 0, 0, 0.5, 1);\n\n /**\n * @internal\n */\n function _WarnImport(name) {\n return `${name} needs to be imported before as it contains a side-effect required by your code.`;\n }\n\n function colorChannelToLinearSpace(color) {\n return Math.pow(color, ToLinearSpace);\n }\n function colorChannelToLinearSpaceExact(color) {\n if (color <= 0.04045) {\n return 0.0773993808 * color;\n }\n return Math.pow(0.947867299 * (color + 0.055), 2.4);\n }\n function colorChannelToGammaSpace(color) {\n return Math.pow(color, ToGammaSpace);\n }\n function colorChannelToGammaSpaceExact(color) {\n if (color <= 0.0031308) {\n return 12.92 * color;\n }\n return 1.055 * Math.pow(color, 0.41666) - 0.055;\n }\n /**\n * Class used to hold a RGB color\n */\n class Color3 {\n /**\n * Creates a new Color3 object from red, green, blue values, all between 0 and 1\n * @param r defines the red component (between 0 and 1, default is 0)\n * @param g defines the green component (between 0 and 1, default is 0)\n * @param b defines the blue component (between 0 and 1, default is 0)\n */\n constructor(\n /**\n * Defines the red component (between 0 and 1, default is 0)\n */\n r = 0, \n /**\n * Defines the green component (between 0 and 1, default is 0)\n */\n g = 0, \n /**\n * Defines the blue component (between 0 and 1, default is 0)\n */\n b = 0) {\n this.r = r;\n this.g = g;\n this.b = b;\n }\n /**\n * Creates a string with the Color3 current values\n * @returns the string representation of the Color3 object\n */\n toString() {\n return \"{R: \" + this.r + \" G:\" + this.g + \" B:\" + this.b + \"}\";\n }\n /**\n * Returns the string \"Color3\"\n * @returns \"Color3\"\n */\n getClassName() {\n return \"Color3\";\n }\n /**\n * Compute the Color3 hash code\n * @returns an unique number that can be used to hash Color3 objects\n */\n getHashCode() {\n let hash = (this.r * 255) | 0;\n hash = (hash * 397) ^ ((this.g * 255) | 0);\n hash = (hash * 397) ^ ((this.b * 255) | 0);\n return hash;\n }\n // Operators\n /**\n * Stores in the given array from the given starting index the red, green, blue values as successive elements\n * @param array defines the array where to store the r,g,b components\n * @param index defines an optional index in the target array to define where to start storing values\n * @returns the current Color3 object\n */\n toArray(array, index = 0) {\n array[index] = this.r;\n array[index + 1] = this.g;\n array[index + 2] = this.b;\n return this;\n }\n /**\n * Update the current color with values stored in an array from the starting index of the given array\n * @param array defines the source array\n * @param offset defines an offset in the source array\n * @returns the current Color3 object\n */\n fromArray(array, offset = 0) {\n Color3.FromArrayToRef(array, offset, this);\n return this;\n }\n /**\n * Returns a new Color4 object from the current Color3 and the given alpha\n * @param alpha defines the alpha component on the new Color4 object (default is 1)\n * @returns a new Color4 object\n */\n toColor4(alpha = 1) {\n return new Color4(this.r, this.g, this.b, alpha);\n }\n /**\n * Returns a new array populated with 3 numeric elements : red, green and blue values\n * @returns the new array\n */\n asArray() {\n return [this.r, this.g, this.b];\n }\n /**\n * Returns the luminance value\n * @returns a float value\n */\n toLuminance() {\n return this.r * 0.3 + this.g * 0.59 + this.b * 0.11;\n }\n /**\n * Multiply each Color3 rgb values by the given Color3 rgb values in a new Color3 object\n * @param otherColor defines the second operand\n * @returns the new Color3 object\n */\n multiply(otherColor) {\n return new Color3(this.r * otherColor.r, this.g * otherColor.g, this.b * otherColor.b);\n }\n /**\n * Multiply the rgb values of the Color3 and the given Color3 and stores the result in the object \"result\"\n * @param otherColor defines the second operand\n * @param result defines the Color3 object where to store the result\n * @returns the current Color3\n */\n multiplyToRef(otherColor, result) {\n result.r = this.r * otherColor.r;\n result.g = this.g * otherColor.g;\n result.b = this.b * otherColor.b;\n return this;\n }\n /**\n * Determines equality between Color3 objects\n * @param otherColor defines the second operand\n * @returns true if the rgb values are equal to the given ones\n */\n equals(otherColor) {\n return otherColor && this.r === otherColor.r && this.g === otherColor.g && this.b === otherColor.b;\n }\n /**\n * Determines equality between the current Color3 object and a set of r,b,g values\n * @param r defines the red component to check\n * @param g defines the green component to check\n * @param b defines the blue component to check\n * @returns true if the rgb values are equal to the given ones\n */\n equalsFloats(r, g, b) {\n return this.r === r && this.g === g && this.b === b;\n }\n /**\n * Creates a new Color3 with the current Color3 values multiplied by scale\n * @param scale defines the scaling factor to apply\n * @returns a new Color3 object\n */\n scale(scale) {\n return new Color3(this.r * scale, this.g * scale, this.b * scale);\n }\n /**\n * Multiplies the Color3 values by the float \"scale\"\n * @param scale defines the scaling factor to apply\n * @returns the current updated Color3\n */\n scaleInPlace(scale) {\n this.r *= scale;\n this.g *= scale;\n this.b *= scale;\n return this;\n }\n /**\n * Multiplies the rgb values by scale and stores the result into \"result\"\n * @param scale defines the scaling factor\n * @param result defines the Color3 object where to store the result\n * @returns the unmodified current Color3\n */\n scaleToRef(scale, result) {\n result.r = this.r * scale;\n result.g = this.g * scale;\n result.b = this.b * scale;\n return this;\n }\n /**\n * Scale the current Color3 values by a factor and add the result to a given Color3\n * @param scale defines the scale factor\n * @param result defines color to store the result into\n * @returns the unmodified current Color3\n */\n scaleAndAddToRef(scale, result) {\n result.r += this.r * scale;\n result.g += this.g * scale;\n result.b += this.b * scale;\n return this;\n }\n /**\n * Clamps the rgb values by the min and max values and stores the result into \"result\"\n * @param min defines minimum clamping value (default is 0)\n * @param max defines maximum clamping value (default is 1)\n * @param result defines color to store the result into\n * @returns the original Color3\n */\n clampToRef(min = 0, max = 1, result) {\n result.r = Scalar.Clamp(this.r, min, max);\n result.g = Scalar.Clamp(this.g, min, max);\n result.b = Scalar.Clamp(this.b, min, max);\n return this;\n }\n /**\n * Creates a new Color3 set with the added values of the current Color3 and of the given one\n * @param otherColor defines the second operand\n * @returns the new Color3\n */\n add(otherColor) {\n return new Color3(this.r + otherColor.r, this.g + otherColor.g, this.b + otherColor.b);\n }\n /**\n * Stores the result of the addition of the current Color3 and given one rgb values into \"result\"\n * @param otherColor defines the second operand\n * @param result defines Color3 object to store the result into\n * @returns the unmodified current Color3\n */\n addToRef(otherColor, result) {\n result.r = this.r + otherColor.r;\n result.g = this.g + otherColor.g;\n result.b = this.b + otherColor.b;\n return this;\n }\n /**\n * Returns a new Color3 set with the subtracted values of the given one from the current Color3\n * @param otherColor defines the second operand\n * @returns the new Color3\n */\n subtract(otherColor) {\n return new Color3(this.r - otherColor.r, this.g - otherColor.g, this.b - otherColor.b);\n }\n /**\n * Stores the result of the subtraction of given one from the current Color3 rgb values into \"result\"\n * @param otherColor defines the second operand\n * @param result defines Color3 object to store the result into\n * @returns the unmodified current Color3\n */\n subtractToRef(otherColor, result) {\n result.r = this.r - otherColor.r;\n result.g = this.g - otherColor.g;\n result.b = this.b - otherColor.b;\n return this;\n }\n /**\n * Copy the current object\n * @returns a new Color3 copied the current one\n */\n clone() {\n return new Color3(this.r, this.g, this.b);\n }\n /**\n * Copies the rgb values from the source in the current Color3\n * @param source defines the source Color3 object\n * @returns the updated Color3 object\n */\n copyFrom(source) {\n this.r = source.r;\n this.g = source.g;\n this.b = source.b;\n return this;\n }\n /**\n * Updates the Color3 rgb values from the given floats\n * @param r defines the red component to read from\n * @param g defines the green component to read from\n * @param b defines the blue component to read from\n * @returns the current Color3 object\n */\n copyFromFloats(r, g, b) {\n this.r = r;\n this.g = g;\n this.b = b;\n return this;\n }\n /**\n * Updates the Color3 rgb values from the given floats\n * @param r defines the red component to read from\n * @param g defines the green component to read from\n * @param b defines the blue component to read from\n * @returns the current Color3 object\n */\n set(r, g, b) {\n return this.copyFromFloats(r, g, b);\n }\n /**\n * Compute the Color3 hexadecimal code as a string\n * @returns a string containing the hexadecimal representation of the Color3 object\n */\n toHexString() {\n const intR = Math.round(this.r * 255);\n const intG = Math.round(this.g * 255);\n const intB = Math.round(this.b * 255);\n return \"#\" + Scalar.ToHex(intR) + Scalar.ToHex(intG) + Scalar.ToHex(intB);\n }\n /**\n * Converts current color in rgb space to HSV values\n * @returns a new color3 representing the HSV values\n */\n toHSV() {\n const result = new Color3();\n this.toHSVToRef(result);\n return result;\n }\n /**\n * Converts current color in rgb space to HSV values\n * @param result defines the Color3 where to store the HSV values\n */\n toHSVToRef(result) {\n const r = this.r;\n const g = this.g;\n const b = this.b;\n const max = Math.max(r, g, b);\n const min = Math.min(r, g, b);\n let h = 0;\n let s = 0;\n const v = max;\n const dm = max - min;\n if (max !== 0) {\n s = dm / max;\n }\n if (max != min) {\n if (max == r) {\n h = (g - b) / dm;\n if (g < b) {\n h += 6;\n }\n }\n else if (max == g) {\n h = (b - r) / dm + 2;\n }\n else if (max == b) {\n h = (r - g) / dm + 4;\n }\n h *= 60;\n }\n result.r = h;\n result.g = s;\n result.b = v;\n }\n /**\n * Computes a new Color3 converted from the current one to linear space\n * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false)\n * @returns a new Color3 object\n */\n toLinearSpace(exact = false) {\n const convertedColor = new Color3();\n this.toLinearSpaceToRef(convertedColor, exact);\n return convertedColor;\n }\n /**\n * Converts the Color3 values to linear space and stores the result in \"convertedColor\"\n * @param convertedColor defines the Color3 object where to store the linear space version\n * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false)\n * @returns the unmodified Color3\n */\n toLinearSpaceToRef(convertedColor, exact = false) {\n if (exact) {\n convertedColor.r = colorChannelToLinearSpaceExact(this.r);\n convertedColor.g = colorChannelToLinearSpaceExact(this.g);\n convertedColor.b = colorChannelToLinearSpaceExact(this.b);\n }\n else {\n convertedColor.r = colorChannelToLinearSpace(this.r);\n convertedColor.g = colorChannelToLinearSpace(this.g);\n convertedColor.b = colorChannelToLinearSpace(this.b);\n }\n return this;\n }\n /**\n * Computes a new Color3 converted from the current one to gamma space\n * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false)\n * @returns a new Color3 object\n */\n toGammaSpace(exact = false) {\n const convertedColor = new Color3();\n this.toGammaSpaceToRef(convertedColor, exact);\n return convertedColor;\n }\n /**\n * Converts the Color3 values to gamma space and stores the result in \"convertedColor\"\n * @param convertedColor defines the Color3 object where to store the gamma space version\n * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false)\n * @returns the unmodified Color3\n */\n toGammaSpaceToRef(convertedColor, exact = false) {\n if (exact) {\n convertedColor.r = colorChannelToGammaSpaceExact(this.r);\n convertedColor.g = colorChannelToGammaSpaceExact(this.g);\n convertedColor.b = colorChannelToGammaSpaceExact(this.b);\n }\n else {\n convertedColor.r = colorChannelToGammaSpace(this.r);\n convertedColor.g = colorChannelToGammaSpace(this.g);\n convertedColor.b = colorChannelToGammaSpace(this.b);\n }\n return this;\n }\n /**\n * Converts Hue, saturation and value to a Color3 (RGB)\n * @param hue defines the hue (value between 0 and 360)\n * @param saturation defines the saturation (value between 0 and 1)\n * @param value defines the value (value between 0 and 1)\n * @param result defines the Color3 where to store the RGB values\n */\n static HSVtoRGBToRef(hue, saturation, value, result) {\n const chroma = value * saturation;\n const h = hue / 60;\n const x = chroma * (1 - Math.abs((h % 2) - 1));\n let r = 0;\n let g = 0;\n let b = 0;\n if (h >= 0 && h <= 1) {\n r = chroma;\n g = x;\n }\n else if (h >= 1 && h <= 2) {\n r = x;\n g = chroma;\n }\n else if (h >= 2 && h <= 3) {\n g = chroma;\n b = x;\n }\n else if (h >= 3 && h <= 4) {\n g = x;\n b = chroma;\n }\n else if (h >= 4 && h <= 5) {\n r = x;\n b = chroma;\n }\n else if (h >= 5 && h <= 6) {\n r = chroma;\n b = x;\n }\n const m = value - chroma;\n result.set(r + m, g + m, b + m);\n }\n /**\n * Converts Hue, saturation and value to a new Color3 (RGB)\n * @param hue defines the hue (value between 0 and 360)\n * @param saturation defines the saturation (value between 0 and 1)\n * @param value defines the value (value between 0 and 1)\n * @returns a new Color3 object\n */\n static FromHSV(hue, saturation, value) {\n const result = new Color3(0, 0, 0);\n Color3.HSVtoRGBToRef(hue, saturation, value, result);\n return result;\n }\n /**\n * Creates a new Color3 from the string containing valid hexadecimal values\n * @param hex defines a string containing valid hexadecimal values\n * @returns a new Color3 object\n */\n static FromHexString(hex) {\n if (hex.substring(0, 1) !== \"#\" || hex.length !== 7) {\n return new Color3(0, 0, 0);\n }\n const r = parseInt(hex.substring(1, 3), 16);\n const g = parseInt(hex.substring(3, 5), 16);\n const b = parseInt(hex.substring(5, 7), 16);\n return Color3.FromInts(r, g, b);\n }\n /**\n * Creates a new Color3 from the starting index of the given array\n * @param array defines the source array\n * @param offset defines an offset in the source array\n * @returns a new Color3 object\n */\n static FromArray(array, offset = 0) {\n return new Color3(array[offset], array[offset + 1], array[offset + 2]);\n }\n /**\n * Creates a new Color3 from the starting index element of the given array\n * @param array defines the source array to read from\n * @param offset defines the offset in the source array\n * @param result defines the target Color3 object\n */\n static FromArrayToRef(array, offset = 0, result) {\n result.r = array[offset];\n result.g = array[offset + 1];\n result.b = array[offset + 2];\n }\n /**\n * Creates a new Color3 from integer values (< 256)\n * @param r defines the red component to read from (value between 0 and 255)\n * @param g defines the green component to read from (value between 0 and 255)\n * @param b defines the blue component to read from (value between 0 and 255)\n * @returns a new Color3 object\n */\n static FromInts(r, g, b) {\n return new Color3(r / 255.0, g / 255.0, b / 255.0);\n }\n /**\n * Creates a new Color3 with values linearly interpolated of \"amount\" between the start Color3 and the end Color3\n * @param start defines the start Color3 value\n * @param end defines the end Color3 value\n * @param amount defines the gradient value between start and end\n * @returns a new Color3 object\n */\n static Lerp(start, end, amount) {\n const result = new Color3(0.0, 0.0, 0.0);\n Color3.LerpToRef(start, end, amount, result);\n return result;\n }\n /**\n * Creates a new Color3 with values linearly interpolated of \"amount\" between the start Color3 and the end Color3\n * @param left defines the start value\n * @param right defines the end value\n * @param amount defines the gradient factor\n * @param result defines the Color3 object where to store the result\n */\n static LerpToRef(left, right, amount, result) {\n result.r = left.r + (right.r - left.r) * amount;\n result.g = left.g + (right.g - left.g) * amount;\n result.b = left.b + (right.b - left.b) * amount;\n }\n /**\n * Returns a new Color3 located for \"amount\" (float) on the Hermite interpolation spline defined by the vectors \"value1\", \"tangent1\", \"value2\", \"tangent2\"\n * @param value1 defines the first control point\n * @param tangent1 defines the first tangent Color3\n * @param value2 defines the second control point\n * @param tangent2 defines the second tangent Color3\n * @param amount defines the amount on the interpolation spline (between 0 and 1)\n * @returns the new Color3\n */\n static Hermite(value1, tangent1, value2, tangent2, amount) {\n const squared = amount * amount;\n const cubed = amount * squared;\n const part1 = 2.0 * cubed - 3.0 * squared + 1.0;\n const part2 = -2.0 * cubed + 3.0 * squared;\n const part3 = cubed - 2.0 * squared + amount;\n const part4 = cubed - squared;\n const r = value1.r * part1 + value2.r * part2 + tangent1.r * part3 + tangent2.r * part4;\n const g = value1.g * part1 + value2.g * part2 + tangent1.g * part3 + tangent2.g * part4;\n const b = value1.b * part1 + value2.b * part2 + tangent1.b * part3 + tangent2.b * part4;\n return new Color3(r, g, b);\n }\n /**\n * Returns a new Color3 which is the 1st derivative of the Hermite spline defined by the colors \"value1\", \"value2\", \"tangent1\", \"tangent2\".\n * @param value1 defines the first control point\n * @param tangent1 defines the first tangent\n * @param value2 defines the second control point\n * @param tangent2 defines the second tangent\n * @param time define where the derivative must be done\n * @returns 1st derivative\n */\n static Hermite1stDerivative(value1, tangent1, value2, tangent2, time) {\n const result = Color3.Black();\n this.Hermite1stDerivativeToRef(value1, tangent1, value2, tangent2, time, result);\n return result;\n }\n /**\n * Returns a new Color3 which is the 1st derivative of the Hermite spline defined by the colors \"value1\", \"value2\", \"tangent1\", \"tangent2\".\n * @param value1 defines the first control point\n * @param tangent1 defines the first tangent\n * @param value2 defines the second control point\n * @param tangent2 defines the second tangent\n * @param time define where the derivative must be done\n * @param result define where to store the derivative\n */\n static Hermite1stDerivativeToRef(value1, tangent1, value2, tangent2, time, result) {\n const t2 = time * time;\n result.r = (t2 - time) * 6 * value1.r + (3 * t2 - 4 * time + 1) * tangent1.r + (-t2 + time) * 6 * value2.r + (3 * t2 - 2 * time) * tangent2.r;\n result.g = (t2 - time) * 6 * value1.g + (3 * t2 - 4 * time + 1) * tangent1.g + (-t2 + time) * 6 * value2.g + (3 * t2 - 2 * time) * tangent2.g;\n result.b = (t2 - time) * 6 * value1.b + (3 * t2 - 4 * time + 1) * tangent1.b + (-t2 + time) * 6 * value2.b + (3 * t2 - 2 * time) * tangent2.b;\n }\n /**\n * Returns a Color3 value containing a red color\n * @returns a new Color3 object\n */\n static Red() {\n return new Color3(1, 0, 0);\n }\n /**\n * Returns a Color3 value containing a green color\n * @returns a new Color3 object\n */\n static Green() {\n return new Color3(0, 1, 0);\n }\n /**\n * Returns a Color3 value containing a blue color\n * @returns a new Color3 object\n */\n static Blue() {\n return new Color3(0, 0, 1);\n }\n /**\n * Returns a Color3 value containing a black color\n * @returns a new Color3 object\n */\n static Black() {\n return new Color3(0, 0, 0);\n }\n /**\n * Gets a Color3 value containing a black color that must not be updated\n */\n static get BlackReadOnly() {\n return Color3._BlackReadOnly;\n }\n /**\n * Returns a Color3 value containing a white color\n * @returns a new Color3 object\n */\n static White() {\n return new Color3(1, 1, 1);\n }\n /**\n * Returns a Color3 value containing a purple color\n * @returns a new Color3 object\n */\n static Purple() {\n return new Color3(0.5, 0, 0.5);\n }\n /**\n * Returns a Color3 value containing a magenta color\n * @returns a new Color3 object\n */\n static Magenta() {\n return new Color3(1, 0, 1);\n }\n /**\n * Returns a Color3 value containing a yellow color\n * @returns a new Color3 object\n */\n static Yellow() {\n return new Color3(1, 1, 0);\n }\n /**\n * Returns a Color3 value containing a gray color\n * @returns a new Color3 object\n */\n static Gray() {\n return new Color3(0.5, 0.5, 0.5);\n }\n /**\n * Returns a Color3 value containing a teal color\n * @returns a new Color3 object\n */\n static Teal() {\n return new Color3(0, 1.0, 1.0);\n }\n /**\n * Returns a Color3 value containing a random color\n * @returns a new Color3 object\n */\n static Random() {\n return new Color3(Math.random(), Math.random(), Math.random());\n }\n }\n // Statics\n Color3._BlackReadOnly = Color3.Black();\n /**\n * Class used to hold a RBGA color\n */\n class Color4 {\n /**\n * Creates a new Color4 object from red, green, blue values, all between 0 and 1\n * @param r defines the red component (between 0 and 1, default is 0)\n * @param g defines the green component (between 0 and 1, default is 0)\n * @param b defines the blue component (between 0 and 1, default is 0)\n * @param a defines the alpha component (between 0 and 1, default is 1)\n */\n constructor(\n /**\n * Defines the red component (between 0 and 1, default is 0)\n */\n r = 0, \n /**\n * Defines the green component (between 0 and 1, default is 0)\n */\n g = 0, \n /**\n * Defines the blue component (between 0 and 1, default is 0)\n */\n b = 0, \n /**\n * Defines the alpha component (between 0 and 1, default is 1)\n */\n a = 1) {\n this.r = r;\n this.g = g;\n this.b = b;\n this.a = a;\n }\n // Operators\n /**\n * Adds in place the given Color4 values to the current Color4 object\n * @param right defines the second operand\n * @returns the current updated Color4 object\n */\n addInPlace(right) {\n this.r += right.r;\n this.g += right.g;\n this.b += right.b;\n this.a += right.a;\n return this;\n }\n /**\n * Creates a new array populated with 4 numeric elements : red, green, blue, alpha values\n * @returns the new array\n */\n asArray() {\n return [this.r, this.g, this.b, this.a];\n }\n /**\n * Stores from the starting index in the given array the Color4 successive values\n * @param array defines the array where to store the r,g,b components\n * @param index defines an optional index in the target array to define where to start storing values\n * @returns the current Color4 object\n */\n toArray(array, index = 0) {\n array[index] = this.r;\n array[index + 1] = this.g;\n array[index + 2] = this.b;\n array[index + 3] = this.a;\n return this;\n }\n /**\n * Update the current color with values stored in an array from the starting index of the given array\n * @param array defines the source array\n * @param offset defines an offset in the source array\n * @returns the current Color4 object\n */\n fromArray(array, offset = 0) {\n Color4.FromArrayToRef(array, offset, this);\n return this;\n }\n /**\n * Determines equality between Color4 objects\n * @param otherColor defines the second operand\n * @returns true if the rgba values are equal to the given ones\n */\n equals(otherColor) {\n return otherColor && this.r === otherColor.r && this.g === otherColor.g && this.b === otherColor.b && this.a === otherColor.a;\n }\n /**\n * Creates a new Color4 set with the added values of the current Color4 and of the given one\n * @param right defines the second operand\n * @returns a new Color4 object\n */\n add(right) {\n return new Color4(this.r + right.r, this.g + right.g, this.b + right.b, this.a + right.a);\n }\n /**\n * Creates a new Color4 set with the subtracted values of the given one from the current Color4\n * @param right defines the second operand\n * @returns a new Color4 object\n */\n subtract(right) {\n return new Color4(this.r - right.r, this.g - right.g, this.b - right.b, this.a - right.a);\n }\n /**\n * Subtracts the given ones from the current Color4 values and stores the results in \"result\"\n * @param right defines the second operand\n * @param result defines the Color4 object where to store the result\n * @returns the current Color4 object\n */\n subtractToRef(right, result) {\n result.r = this.r - right.r;\n result.g = this.g - right.g;\n result.b = this.b - right.b;\n result.a = this.a - right.a;\n return this;\n }\n /**\n * Creates a new Color4 with the current Color4 values multiplied by scale\n * @param scale defines the scaling factor to apply\n * @returns a new Color4 object\n */\n scale(scale) {\n return new Color4(this.r * scale, this.g * scale, this.b * scale, this.a * scale);\n }\n /**\n * Multiplies the Color4 values by the float \"scale\"\n * @param scale defines the scaling factor to apply\n * @returns the current updated Color4\n */\n scaleInPlace(scale) {\n this.r *= scale;\n this.g *= scale;\n this.b *= scale;\n this.a *= scale;\n return this;\n }\n /**\n * Multiplies the current Color4 values by scale and stores the result in \"result\"\n * @param scale defines the scaling factor to apply\n * @param result defines the Color4 object where to store the result\n * @returns the current unmodified Color4\n */\n scaleToRef(scale, result) {\n result.r = this.r * scale;\n result.g = this.g * scale;\n result.b = this.b * scale;\n result.a = this.a * scale;\n return this;\n }\n /**\n * Scale the current Color4 values by a factor and add the result to a given Color4\n * @param scale defines the scale factor\n * @param result defines the Color4 object where to store the result\n * @returns the unmodified current Color4\n */\n scaleAndAddToRef(scale, result) {\n result.r += this.r * scale;\n result.g += this.g * scale;\n result.b += this.b * scale;\n result.a += this.a * scale;\n return this;\n }\n /**\n * Clamps the rgb values by the min and max values and stores the result into \"result\"\n * @param min defines minimum clamping value (default is 0)\n * @param max defines maximum clamping value (default is 1)\n * @param result defines color to store the result into.\n * @returns the current Color4\n */\n clampToRef(min = 0, max = 1, result) {\n result.r = Scalar.Clamp(this.r, min, max);\n result.g = Scalar.Clamp(this.g, min, max);\n result.b = Scalar.Clamp(this.b, min, max);\n result.a = Scalar.Clamp(this.a, min, max);\n return this;\n }\n /**\n * Multiply an Color4 value by another and return a new Color4 object\n * @param color defines the Color4 value to multiply by\n * @returns a new Color4 object\n */\n multiply(color) {\n return new Color4(this.r * color.r, this.g * color.g, this.b * color.b, this.a * color.a);\n }\n /**\n * Multiply a Color4 value by another and push the result in a reference value\n * @param color defines the Color4 value to multiply by\n * @param result defines the Color4 to fill the result in\n * @returns the result Color4\n */\n multiplyToRef(color, result) {\n result.r = this.r * color.r;\n result.g = this.g * color.g;\n result.b = this.b * color.b;\n result.a = this.a * color.a;\n return result;\n }\n /**\n * Creates a string with the Color4 current values\n * @returns the string representation of the Color4 object\n */\n toString() {\n return \"{R: \" + this.r + \" G:\" + this.g + \" B:\" + this.b + \" A:\" + this.a + \"}\";\n }\n /**\n * Returns the string \"Color4\"\n * @returns \"Color4\"\n */\n getClassName() {\n return \"Color4\";\n }\n /**\n * Compute the Color4 hash code\n * @returns an unique number that can be used to hash Color4 objects\n */\n getHashCode() {\n let hash = (this.r * 255) | 0;\n hash = (hash * 397) ^ ((this.g * 255) | 0);\n hash = (hash * 397) ^ ((this.b * 255) | 0);\n hash = (hash * 397) ^ ((this.a * 255) | 0);\n return hash;\n }\n /**\n * Creates a new Color4 copied from the current one\n * @returns a new Color4 object\n */\n clone() {\n return new Color4(this.r, this.g, this.b, this.a);\n }\n /**\n * Copies the given Color4 values into the current one\n * @param source defines the source Color4 object\n * @returns the current updated Color4 object\n */\n copyFrom(source) {\n this.r = source.r;\n this.g = source.g;\n this.b = source.b;\n this.a = source.a;\n return this;\n }\n /**\n * Copies the given float values into the current one\n * @param r defines the red component to read from\n * @param g defines the green component to read from\n * @param b defines the blue component to read from\n * @param a defines the alpha component to read from\n * @returns the current updated Color4 object\n */\n copyFromFloats(r, g, b, a) {\n this.r = r;\n this.g = g;\n this.b = b;\n this.a = a;\n return this;\n }\n /**\n * Copies the given float values into the current one\n * @param r defines the red component to read from\n * @param g defines the green component to read from\n * @param b defines the blue component to read from\n * @param a defines the alpha component to read from\n * @returns the current updated Color4 object\n */\n set(r, g, b, a) {\n return this.copyFromFloats(r, g, b, a);\n }\n /**\n * Compute the Color4 hexadecimal code as a string\n * @param returnAsColor3 defines if the string should only contains RGB values (off by default)\n * @returns a string containing the hexadecimal representation of the Color4 object\n */\n toHexString(returnAsColor3 = false) {\n const intR = Math.round(this.r * 255);\n const intG = Math.round(this.g * 255);\n const intB = Math.round(this.b * 255);\n if (returnAsColor3) {\n return \"#\" + Scalar.ToHex(intR) + Scalar.ToHex(intG) + Scalar.ToHex(intB);\n }\n const intA = Math.round(this.a * 255);\n return \"#\" + Scalar.ToHex(intR) + Scalar.ToHex(intG) + Scalar.ToHex(intB) + Scalar.ToHex(intA);\n }\n /**\n * Computes a new Color4 converted from the current one to linear space\n * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false)\n * @returns a new Color4 object\n */\n toLinearSpace(exact = false) {\n const convertedColor = new Color4();\n this.toLinearSpaceToRef(convertedColor, exact);\n return convertedColor;\n }\n /**\n * Converts the Color4 values to linear space and stores the result in \"convertedColor\"\n * @param convertedColor defines the Color4 object where to store the linear space version\n * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false)\n * @returns the unmodified Color4\n */\n toLinearSpaceToRef(convertedColor, exact = false) {\n if (exact) {\n convertedColor.r = colorChannelToLinearSpaceExact(this.r);\n convertedColor.g = colorChannelToLinearSpaceExact(this.g);\n convertedColor.b = colorChannelToLinearSpaceExact(this.b);\n }\n else {\n convertedColor.r = colorChannelToLinearSpace(this.r);\n convertedColor.g = colorChannelToLinearSpace(this.g);\n convertedColor.b = colorChannelToLinearSpace(this.b);\n }\n convertedColor.a = this.a;\n return this;\n }\n /**\n * Computes a new Color4 converted from the current one to gamma space\n * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false)\n * @returns a new Color4 object\n */\n toGammaSpace(exact = false) {\n const convertedColor = new Color4();\n this.toGammaSpaceToRef(convertedColor, exact);\n return convertedColor;\n }\n /**\n * Converts the Color4 values to gamma space and stores the result in \"convertedColor\"\n * @param convertedColor defines the Color4 object where to store the gamma space version\n * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false)\n * @returns the unmodified Color4\n */\n toGammaSpaceToRef(convertedColor, exact = false) {\n if (exact) {\n convertedColor.r = colorChannelToGammaSpaceExact(this.r);\n convertedColor.g = colorChannelToGammaSpaceExact(this.g);\n convertedColor.b = colorChannelToGammaSpaceExact(this.b);\n }\n else {\n convertedColor.r = colorChannelToGammaSpace(this.r);\n convertedColor.g = colorChannelToGammaSpace(this.g);\n convertedColor.b = colorChannelToGammaSpace(this.b);\n }\n convertedColor.a = this.a;\n return this;\n }\n // Statics\n /**\n * Creates a new Color4 from the string containing valid hexadecimal values.\n *\n * A valid hex string is either in the format #RRGGBB or #RRGGBBAA.\n *\n * When a hex string without alpha is passed, the resulting Color4 has\n * its alpha value set to 1.0.\n *\n * An invalid string results in a Color with all its channels set to 0.0,\n * i.e. \"transparent black\".\n *\n * @param hex defines a string containing valid hexadecimal values\n * @returns a new Color4 object\n */\n static FromHexString(hex) {\n if (hex.substring(0, 1) !== \"#\" || (hex.length !== 9 && hex.length !== 7)) {\n return new Color4(0.0, 0.0, 0.0, 0.0);\n }\n const r = parseInt(hex.substring(1, 3), 16);\n const g = parseInt(hex.substring(3, 5), 16);\n const b = parseInt(hex.substring(5, 7), 16);\n const a = hex.length === 9 ? parseInt(hex.substring(7, 9), 16) : 255;\n return Color4.FromInts(r, g, b, a);\n }\n /**\n * Creates a new Color4 object set with the linearly interpolated values of \"amount\" between the left Color4 object and the right Color4 object\n * @param left defines the start value\n * @param right defines the end value\n * @param amount defines the gradient factor\n * @returns a new Color4 object\n */\n static Lerp(left, right, amount) {\n const result = new Color4(0.0, 0.0, 0.0, 0.0);\n Color4.LerpToRef(left, right, amount, result);\n return result;\n }\n /**\n * Set the given \"result\" with the linearly interpolated values of \"amount\" between the left Color4 object and the right Color4 object\n * @param left defines the start value\n * @param right defines the end value\n * @param amount defines the gradient factor\n * @param result defines the Color4 object where to store data\n */\n static LerpToRef(left, right, amount, result) {\n result.r = left.r + (right.r - left.r) * amount;\n result.g = left.g + (right.g - left.g) * amount;\n result.b = left.b + (right.b - left.b) * amount;\n result.a = left.a + (right.a - left.a) * amount;\n }\n /**\n * Interpolate between two Color4 using Hermite interpolation\n * @param value1 defines first Color4\n * @param tangent1 defines the incoming tangent\n * @param value2 defines second Color4\n * @param tangent2 defines the outgoing tangent\n * @param amount defines the target Color4\n * @returns the new interpolated Color4\n */\n static Hermite(value1, tangent1, value2, tangent2, amount) {\n const squared = amount * amount;\n const cubed = amount * squared;\n const part1 = 2.0 * cubed - 3.0 * squared + 1.0;\n const part2 = -2.0 * cubed + 3.0 * squared;\n const part3 = cubed - 2.0 * squared + amount;\n const part4 = cubed - squared;\n const r = value1.r * part1 + value2.r * part2 + tangent1.r * part3 + tangent2.r * part4;\n const g = value1.g * part1 + value2.g * part2 + tangent1.g * part3 + tangent2.g * part4;\n const b = value1.b * part1 + value2.b * part2 + tangent1.b * part3 + tangent2.b * part4;\n const a = value1.a * part1 + value2.a * part2 + tangent1.a * part3 + tangent2.a * part4;\n return new Color4(r, g, b, a);\n }\n /**\n * Returns a new Color4 which is the 1st derivative of the Hermite spline defined by the colors \"value1\", \"value2\", \"tangent1\", \"tangent2\".\n * @param value1 defines the first control point\n * @param tangent1 defines the first tangent\n * @param value2 defines the second control point\n * @param tangent2 defines the second tangent\n * @param time define where the derivative must be done\n * @returns 1st derivative\n */\n static Hermite1stDerivative(value1, tangent1, value2, tangent2, time) {\n const result = new Color4();\n this.Hermite1stDerivativeToRef(value1, tangent1, value2, tangent2, time, result);\n return result;\n }\n /**\n * Update a Color4 with the 1st derivative of the Hermite spline defined by the colors \"value1\", \"value2\", \"tangent1\", \"tangent2\".\n * @param value1 defines the first control point\n * @param tangent1 defines the first tangent\n * @param value2 defines the second control point\n * @param tangent2 defines the second tangent\n * @param time define where the derivative must be done\n * @param result define where to store the derivative\n */\n static Hermite1stDerivativeToRef(value1, tangent1, value2, tangent2, time, result) {\n const t2 = time * time;\n result.r = (t2 - time) * 6 * value1.r + (3 * t2 - 4 * time + 1) * tangent1.r + (-t2 + time) * 6 * value2.r + (3 * t2 - 2 * time) * tangent2.r;\n result.g = (t2 - time) * 6 * value1.g + (3 * t2 - 4 * time + 1) * tangent1.g + (-t2 + time) * 6 * value2.g + (3 * t2 - 2 * time) * tangent2.g;\n result.b = (t2 - time) * 6 * value1.b + (3 * t2 - 4 * time + 1) * tangent1.b + (-t2 + time) * 6 * value2.b + (3 * t2 - 2 * time) * tangent2.b;\n result.a = (t2 - time) * 6 * value1.a + (3 * t2 - 4 * time + 1) * tangent1.a + (-t2 + time) * 6 * value2.a + (3 * t2 - 2 * time) * tangent2.a;\n }\n /**\n * Creates a new Color4 from a Color3 and an alpha value\n * @param color3 defines the source Color3 to read from\n * @param alpha defines the alpha component (1.0 by default)\n * @returns a new Color4 object\n */\n static FromColor3(color3, alpha = 1.0) {\n return new Color4(color3.r, color3.g, color3.b, alpha);\n }\n /**\n * Creates a new Color4 from the starting index element of the given array\n * @param array defines the source array to read from\n * @param offset defines the offset in the source array\n * @returns a new Color4 object\n */\n static FromArray(array, offset = 0) {\n return new Color4(array[offset], array[offset + 1], array[offset + 2], array[offset + 3]);\n }\n /**\n * Creates a new Color4 from the starting index element of the given array\n * @param array defines the source array to read from\n * @param offset defines the offset in the source array\n * @param result defines the target Color4 object\n */\n static FromArrayToRef(array, offset = 0, result) {\n result.r = array[offset];\n result.g = array[offset + 1];\n result.b = array[offset + 2];\n result.a = array[offset + 3];\n }\n /**\n * Creates a new Color3 from integer values (< 256)\n * @param r defines the red component to read from (value between 0 and 255)\n * @param g defines the green component to read from (value between 0 and 255)\n * @param b defines the blue component to read from (value between 0 and 255)\n * @param a defines the alpha component to read from (value between 0 and 255)\n * @returns a new Color3 object\n */\n static FromInts(r, g, b, a) {\n return new Color4(r / 255.0, g / 255.0, b / 255.0, a / 255.0);\n }\n /**\n * Check the content of a given array and convert it to an array containing RGBA data\n * If the original array was already containing count * 4 values then it is returned directly\n * @param colors defines the array to check\n * @param count defines the number of RGBA data to expect\n * @returns an array containing count * 4 values (RGBA)\n */\n static CheckColors4(colors, count) {\n // Check if color3 was used\n if (colors.length === count * 3) {\n const colors4 = [];\n for (let index = 0; index < colors.length; index += 3) {\n const newIndex = (index / 3) * 4;\n colors4[newIndex] = colors[index];\n colors4[newIndex + 1] = colors[index + 1];\n colors4[newIndex + 2] = colors[index + 2];\n colors4[newIndex + 3] = 1.0;\n }\n return colors4;\n }\n return colors;\n }\n }\n /**\n * @internal\n */\n class TmpColors {\n }\n TmpColors.Color3 = ArrayTools.BuildArray(3, Color3.Black);\n TmpColors.Color4 = ArrayTools.BuildArray(3, () => new Color4(0, 0, 0, 0));\n RegisterClass(\"BABYLON.Color3\", Color3);\n RegisterClass(\"BABYLON.Color4\", Color4);\n\n /* eslint-disable @typescript-eslint/no-unused-vars */\n const __decoratorInitialStore = {};\n const __mergedStore = {};\n const _copySource = function (creationFunction, source, instanciate, options = {}) {\n const destination = creationFunction();\n // Tags\n if (Tags && Tags.HasTags(source)) {\n Tags.AddTagsTo(destination, Tags.GetTags(source, true));\n }\n const classStore = getMergedStore(destination);\n // Map from source texture uniqueId to destination texture\n const textureMap = {};\n // Properties\n for (const property in classStore) {\n const propertyDescriptor = classStore[property];\n const sourceProperty = source[property];\n const propertyType = propertyDescriptor.type;\n if (sourceProperty !== undefined && sourceProperty !== null && (property !== \"uniqueId\" || SerializationHelper.AllowLoadingUniqueId)) {\n switch (propertyType) {\n case 0: // Value\n case 6: // Mesh reference\n case 11: // Camera reference\n destination[property] = sourceProperty;\n break;\n case 1: // Texture\n if (options.cloneTexturesOnlyOnce && textureMap[sourceProperty.uniqueId]) {\n destination[property] = textureMap[sourceProperty.uniqueId];\n }\n else {\n destination[property] = instanciate || sourceProperty.isRenderTarget ? sourceProperty : sourceProperty.clone();\n textureMap[sourceProperty.uniqueId] = destination[property];\n }\n break;\n case 2: // Color3\n case 3: // FresnelParameters\n case 4: // Vector2\n case 5: // Vector3\n case 7: // Color Curves\n case 10: // Quaternion\n case 12: // Matrix\n destination[property] = instanciate ? sourceProperty : sourceProperty.clone();\n break;\n }\n }\n }\n return destination;\n };\n function getDirectStore(target) {\n const classKey = target.getClassName();\n if (!__decoratorInitialStore[classKey]) {\n __decoratorInitialStore[classKey] = {};\n }\n return __decoratorInitialStore[classKey];\n }\n /**\n * Return the list of properties flagged as serializable\n * @param target host object\n */\n function getMergedStore(target) {\n const classKey = target.getClassName();\n if (__mergedStore[classKey]) {\n return __mergedStore[classKey];\n }\n __mergedStore[classKey] = {};\n const store = __mergedStore[classKey];\n let currentTarget = target;\n let currentKey = classKey;\n while (currentKey) {\n const initialStore = __decoratorInitialStore[currentKey];\n for (const property in initialStore) {\n store[property] = initialStore[property];\n }\n let parent;\n let done = false;\n do {\n parent = Object.getPrototypeOf(currentTarget);\n if (!parent.getClassName) {\n done = true;\n break;\n }\n if (parent.getClassName() !== currentKey) {\n break;\n }\n currentTarget = parent;\n } while (parent);\n if (done) {\n break;\n }\n currentKey = parent.getClassName();\n currentTarget = parent;\n }\n return store;\n }\n function generateSerializableMember(type, sourceName) {\n return (target, propertyKey) => {\n const classStore = getDirectStore(target);\n if (!classStore[propertyKey]) {\n classStore[propertyKey] = { type: type, sourceName: sourceName };\n }\n };\n }\n function generateExpandMember(setCallback, targetKey = null) {\n return (target, propertyKey) => {\n const key = targetKey || \"_\" + propertyKey;\n Object.defineProperty(target, propertyKey, {\n get: function () {\n return this[key];\n },\n set: function (value) {\n // does this object (i.e. vector3) has an equals function? use it!\n // Note - not using \"with epsilon\" here, it is expected te behave like the internal cache does.\n if (typeof this.equals === \"function\") {\n if (this.equals(value)) {\n return;\n }\n }\n if (this[key] === value) {\n return;\n }\n this[key] = value;\n target[setCallback].apply(this);\n },\n enumerable: true,\n configurable: true,\n });\n };\n }\n function expandToProperty(callback, targetKey = null) {\n return generateExpandMember(callback, targetKey);\n }\n function serialize(sourceName) {\n return generateSerializableMember(0, sourceName); // value member\n }\n function serializeAsTexture(sourceName) {\n return generateSerializableMember(1, sourceName); // texture member\n }\n function serializeAsColor3(sourceName) {\n return generateSerializableMember(2, sourceName); // color3 member\n }\n function serializeAsFresnelParameters(sourceName) {\n return generateSerializableMember(3, sourceName); // fresnel parameters member\n }\n function serializeAsVector2(sourceName) {\n return generateSerializableMember(4, sourceName); // vector2 member\n }\n function serializeAsVector3(sourceName) {\n return generateSerializableMember(5, sourceName); // vector3 member\n }\n function serializeAsMeshReference(sourceName) {\n return generateSerializableMember(6, sourceName); // mesh reference member\n }\n function serializeAsColorCurves(sourceName) {\n return generateSerializableMember(7, sourceName); // color curves\n }\n function serializeAsColor4(sourceName) {\n return generateSerializableMember(8, sourceName); // color 4\n }\n function serializeAsImageProcessingConfiguration(sourceName) {\n return generateSerializableMember(9, sourceName); // image processing\n }\n function serializeAsQuaternion(sourceName) {\n return generateSerializableMember(10, sourceName); // quaternion member\n }\n function serializeAsMatrix(sourceName) {\n return generateSerializableMember(12, sourceName); // matrix member\n }\n /**\n * Decorator used to define property that can be serialized as reference to a camera\n * @param sourceName defines the name of the property to decorate\n */\n function serializeAsCameraReference(sourceName) {\n return generateSerializableMember(11, sourceName); // camera reference member\n }\n /**\n * Class used to help serialization objects\n */\n class SerializationHelper {\n /**\n * Appends the serialized animations from the source animations\n * @param source Source containing the animations\n * @param destination Target to store the animations\n */\n static AppendSerializedAnimations(source, destination) {\n if (source.animations) {\n destination.animations = [];\n for (let animationIndex = 0; animationIndex < source.animations.length; animationIndex++) {\n const animation = source.animations[animationIndex];\n destination.animations.push(animation.serialize());\n }\n }\n }\n /**\n * Static function used to serialized a specific entity\n * @param entity defines the entity to serialize\n * @param serializationObject defines the optional target object where serialization data will be stored\n * @returns a JSON compatible object representing the serialization of the entity\n */\n static Serialize(entity, serializationObject) {\n if (!serializationObject) {\n serializationObject = {};\n }\n // Tags\n if (Tags) {\n serializationObject.tags = Tags.GetTags(entity);\n }\n const serializedProperties = getMergedStore(entity);\n // Properties\n for (const property in serializedProperties) {\n const propertyDescriptor = serializedProperties[property];\n const targetPropertyName = propertyDescriptor.sourceName || property;\n const propertyType = propertyDescriptor.type;\n const sourceProperty = entity[property];\n if (sourceProperty !== undefined && sourceProperty !== null && (property !== \"uniqueId\" || SerializationHelper.AllowLoadingUniqueId)) {\n switch (propertyType) {\n case 0: // Value\n serializationObject[targetPropertyName] = sourceProperty;\n break;\n case 1: // Texture\n serializationObject[targetPropertyName] = sourceProperty.serialize();\n break;\n case 2: // Color3\n serializationObject[targetPropertyName] = sourceProperty.asArray();\n break;\n case 3: // FresnelParameters\n serializationObject[targetPropertyName] = sourceProperty.serialize();\n break;\n case 4: // Vector2\n serializationObject[targetPropertyName] = sourceProperty.asArray();\n break;\n case 5: // Vector3\n serializationObject[targetPropertyName] = sourceProperty.asArray();\n break;\n case 6: // Mesh reference\n serializationObject[targetPropertyName] = sourceProperty.id;\n break;\n case 7: // Color Curves\n serializationObject[targetPropertyName] = sourceProperty.serialize();\n break;\n case 8: // Color 4\n serializationObject[targetPropertyName] = sourceProperty.asArray();\n break;\n case 9: // Image Processing\n serializationObject[targetPropertyName] = sourceProperty.serialize();\n break;\n case 10: // Quaternion\n serializationObject[targetPropertyName] = sourceProperty.asArray();\n break;\n case 11: // Camera reference\n serializationObject[targetPropertyName] = sourceProperty.id;\n break;\n case 12: // Matrix\n serializationObject[targetPropertyName] = sourceProperty.asArray();\n break;\n }\n }\n }\n return serializationObject;\n }\n /**\n * Given a source json and a destination object in a scene, this function will parse the source and will try to apply its content to the destination object\n * @param source the source json data\n * @param destination the destination object\n * @param scene the scene where the object is\n * @param rootUrl root url to use to load assets\n */\n static ParseProperties(source, destination, scene, rootUrl) {\n if (!rootUrl) {\n rootUrl = \"\";\n }\n const classStore = getMergedStore(destination);\n // Properties\n for (const property in classStore) {\n const propertyDescriptor = classStore[property];\n const sourceProperty = source[propertyDescriptor.sourceName || property];\n const propertyType = propertyDescriptor.type;\n if (sourceProperty !== undefined && sourceProperty !== null && (property !== \"uniqueId\" || SerializationHelper.AllowLoadingUniqueId)) {\n const dest = destination;\n switch (propertyType) {\n case 0: // Value\n dest[property] = sourceProperty;\n break;\n case 1: // Texture\n if (scene) {\n dest[property] = SerializationHelper._TextureParser(sourceProperty, scene, rootUrl);\n }\n break;\n case 2: // Color3\n dest[property] = Color3.FromArray(sourceProperty);\n break;\n case 3: // FresnelParameters\n dest[property] = SerializationHelper._FresnelParametersParser(sourceProperty);\n break;\n case 4: // Vector2\n dest[property] = Vector2.FromArray(sourceProperty);\n break;\n case 5: // Vector3\n dest[property] = Vector3.FromArray(sourceProperty);\n break;\n case 6: // Mesh reference\n if (scene) {\n dest[property] = scene.getLastMeshById(sourceProperty);\n }\n break;\n case 7: // Color Curves\n dest[property] = SerializationHelper._ColorCurvesParser(sourceProperty);\n break;\n case 8: // Color 4\n dest[property] = Color4.FromArray(sourceProperty);\n break;\n case 9: // Image Processing\n dest[property] = SerializationHelper._ImageProcessingConfigurationParser(sourceProperty);\n break;\n case 10: // Quaternion\n dest[property] = Quaternion.FromArray(sourceProperty);\n break;\n case 11: // Camera reference\n if (scene) {\n dest[property] = scene.getCameraById(sourceProperty);\n }\n break;\n case 12: // Matrix\n dest[property] = Matrix.FromArray(sourceProperty);\n break;\n }\n }\n }\n }\n /**\n * Creates a new entity from a serialization data object\n * @param creationFunction defines a function used to instanciated the new entity\n * @param source defines the source serialization data\n * @param scene defines the hosting scene\n * @param rootUrl defines the root url for resources\n * @returns a new entity\n */\n static Parse(creationFunction, source, scene, rootUrl = null) {\n const destination = creationFunction();\n // Tags\n if (Tags) {\n Tags.AddTagsTo(destination, source.tags);\n }\n SerializationHelper.ParseProperties(source, destination, scene, rootUrl);\n return destination;\n }\n /**\n * Clones an object\n * @param creationFunction defines the function used to instanciate the new object\n * @param source defines the source object\n * @returns the cloned object\n */\n static Clone(creationFunction, source, options = {}) {\n return _copySource(creationFunction, source, false, options);\n }\n /**\n * Instanciates a new object based on a source one (some data will be shared between both object)\n * @param creationFunction defines the function used to instanciate the new object\n * @param source defines the source object\n * @returns the new object\n */\n static Instanciate(creationFunction, source) {\n return _copySource(creationFunction, source, true);\n }\n }\n /**\n * Gets or sets a boolean to indicate if the UniqueId property should be serialized\n */\n SerializationHelper.AllowLoadingUniqueId = false;\n /**\n * @internal\n */\n SerializationHelper._ImageProcessingConfigurationParser = (sourceProperty) => {\n throw _WarnImport(\"ImageProcessingConfiguration\");\n };\n /**\n * @internal\n */\n SerializationHelper._FresnelParametersParser = (sourceProperty) => {\n throw _WarnImport(\"FresnelParameters\");\n };\n /**\n * @internal\n */\n SerializationHelper._ColorCurvesParser = (sourceProperty) => {\n throw _WarnImport(\"ColorCurves\");\n };\n /**\n * @internal\n */\n SerializationHelper._TextureParser = (sourceProperty, scene, rootUrl) => {\n throw _WarnImport(\"Texture\");\n };\n /**\n * Decorator used to redirect a function to a native implementation if available.\n * @internal\n */\n function nativeOverride(target, propertyKey, descriptor, predicate) {\n // Cache the original JS function for later.\n const jsFunc = descriptor.value;\n // Override the JS function to check for a native override on first invocation. Setting descriptor.value overrides the function at the early stage of code being loaded/imported.\n descriptor.value = (...params) => {\n // Assume the resolved function will be the original JS function, then we will check for the Babylon Native context.\n let func = jsFunc;\n // Check if we are executing in a Babylon Native context (e.g. check the presence of the _native global property) and if so also check if a function override is available.\n if (typeof _native !== \"undefined\" && _native[propertyKey]) {\n const nativeFunc = _native[propertyKey];\n // If a predicate was provided, then we'll need to invoke the predicate on each invocation of the underlying function to determine whether to call the native function or the JS function.\n if (predicate) {\n // The resolved function will execute the predicate and then either execute the native function or the JS function.\n func = (...params) => (predicate(...params) ? nativeFunc(...params) : jsFunc(...params));\n }\n else {\n // The resolved function will directly execute the native function.\n func = nativeFunc;\n }\n }\n // Override the JS function again with the final resolved target function.\n target[propertyKey] = func;\n // The JS function has now been overridden based on whether we're executing in the context of Babylon Native, but we still need to invoke that function.\n // Future invocations of the function will just directly invoke the final overridden function, not any of the decorator setup logic above.\n return func(...params);\n };\n }\n /**\n * Decorator factory that applies the nativeOverride decorator, but determines whether to redirect to the native implementation based on a filter function that evaluates the function arguments.\n * @param predicate\n * @example @nativeOverride.filter((...[arg1]: Parameters) => arg1.length > 20)\n * public someMethod(arg1: string, arg2: number): string {\n * @internal\n */\n nativeOverride.filter = function (predicate) {\n return (target, propertyKey, descriptor) => nativeOverride(target, propertyKey, descriptor, predicate);\n };\n\n /** @internal */\n class _InternalNodeDataInfo {\n constructor() {\n this._doNotSerialize = false;\n this._isDisposed = false;\n this._sceneRootNodesIndex = -1;\n this._isEnabled = true;\n this._isParentEnabled = true;\n this._isReady = true;\n this._onEnabledStateChangedObservable = new Observable$1();\n this._onClonedObservable = new Observable$1();\n }\n }\n /**\n * Node is the basic class for all scene objects (Mesh, Light, Camera.)\n */\n class Node {\n /**\n * Add a new node constructor\n * @param type defines the type name of the node to construct\n * @param constructorFunc defines the constructor function\n */\n static AddNodeConstructor(type, constructorFunc) {\n this._NodeConstructors[type] = constructorFunc;\n }\n /**\n * Returns a node constructor based on type name\n * @param type defines the type name\n * @param name defines the new node name\n * @param scene defines the hosting scene\n * @param options defines optional options to transmit to constructors\n * @returns the new constructor or null\n */\n static Construct(type, name, scene, options) {\n const constructorFunc = this._NodeConstructors[type];\n if (!constructorFunc) {\n return null;\n }\n return constructorFunc(name, scene, options);\n }\n /**\n * Gets or sets the accessibility tag to describe the node for accessibility purpose.\n */\n set accessibilityTag(value) {\n this._accessibilityTag = value;\n this.onAccessibilityTagChangedObservable.notifyObservers(value);\n }\n get accessibilityTag() {\n return this._accessibilityTag;\n }\n /**\n * Gets or sets a boolean used to define if the node must be serialized\n */\n get doNotSerialize() {\n if (this._nodeDataStorage._doNotSerialize) {\n return true;\n }\n if (this._parentNode) {\n return this._parentNode.doNotSerialize;\n }\n return false;\n }\n set doNotSerialize(value) {\n this._nodeDataStorage._doNotSerialize = value;\n }\n /**\n * Gets a boolean indicating if the node has been disposed\n * @returns true if the node was disposed\n */\n isDisposed() {\n return this._nodeDataStorage._isDisposed;\n }\n /**\n * Gets or sets the parent of the node (without keeping the current position in the scene)\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/parent_pivot/parent\n */\n set parent(parent) {\n if (this._parentNode === parent) {\n return;\n }\n const previousParentNode = this._parentNode;\n // Remove self from list of children of parent\n if (this._parentNode && this._parentNode._children !== undefined && this._parentNode._children !== null) {\n const index = this._parentNode._children.indexOf(this);\n if (index !== -1) {\n this._parentNode._children.splice(index, 1);\n }\n if (!parent && !this._nodeDataStorage._isDisposed) {\n this._addToSceneRootNodes();\n }\n }\n // Store new parent\n this._parentNode = parent;\n // Add as child to new parent\n if (this._parentNode) {\n if (this._parentNode._children === undefined || this._parentNode._children === null) {\n this._parentNode._children = new Array();\n }\n this._parentNode._children.push(this);\n if (!previousParentNode) {\n this._removeFromSceneRootNodes();\n }\n }\n // Enabled state\n this._syncParentEnabledState();\n }\n get parent() {\n return this._parentNode;\n }\n /**\n * @internal\n */\n _serializeAsParent(serializationObject) {\n serializationObject.parentId = this.uniqueId;\n }\n /** @internal */\n _addToSceneRootNodes() {\n if (this._nodeDataStorage._sceneRootNodesIndex === -1) {\n this._nodeDataStorage._sceneRootNodesIndex = this._scene.rootNodes.length;\n this._scene.rootNodes.push(this);\n }\n }\n /** @internal */\n _removeFromSceneRootNodes() {\n if (this._nodeDataStorage._sceneRootNodesIndex !== -1) {\n const rootNodes = this._scene.rootNodes;\n const lastIdx = rootNodes.length - 1;\n rootNodes[this._nodeDataStorage._sceneRootNodesIndex] = rootNodes[lastIdx];\n rootNodes[this._nodeDataStorage._sceneRootNodesIndex]._nodeDataStorage._sceneRootNodesIndex = this._nodeDataStorage._sceneRootNodesIndex;\n this._scene.rootNodes.pop();\n this._nodeDataStorage._sceneRootNodesIndex = -1;\n }\n }\n /**\n * Gets or sets the animation properties override\n */\n get animationPropertiesOverride() {\n if (!this._animationPropertiesOverride) {\n return this._scene.animationPropertiesOverride;\n }\n return this._animationPropertiesOverride;\n }\n set animationPropertiesOverride(value) {\n this._animationPropertiesOverride = value;\n }\n /**\n * Gets a string identifying the name of the class\n * @returns \"Node\" string\n */\n getClassName() {\n return \"Node\";\n }\n /**\n * Sets a callback that will be raised when the node will be disposed\n */\n set onDispose(callback) {\n if (this._onDisposeObserver) {\n this.onDisposeObservable.remove(this._onDisposeObserver);\n }\n this._onDisposeObserver = this.onDisposeObservable.add(callback);\n }\n /**\n * An event triggered when the enabled state of the node changes\n */\n get onEnabledStateChangedObservable() {\n return this._nodeDataStorage._onEnabledStateChangedObservable;\n }\n /**\n * An event triggered when the node is cloned\n */\n get onClonedObservable() {\n return this._nodeDataStorage._onClonedObservable;\n }\n /**\n * Creates a new Node\n * @param name the name and id to be given to this node\n * @param scene the scene this node will be added to\n */\n constructor(name, scene = null) {\n this._isDirty = false;\n this._nodeDataStorage = new _InternalNodeDataInfo();\n /**\n * Gets or sets a string used to store user defined state for the node\n */\n this.state = \"\";\n /**\n * Gets or sets an object used to store user defined information for the node\n */\n this.metadata = null;\n /**\n * For internal use only. Please do not use.\n */\n this.reservedDataStore = null;\n this._accessibilityTag = null;\n this.onAccessibilityTagChangedObservable = new Observable$1();\n /** @internal */\n this._parentContainer = null;\n /**\n * Gets a list of Animations associated with the node\n */\n this.animations = new Array();\n this._ranges = {};\n /**\n * Callback raised when the node is ready to be used\n */\n this.onReady = null;\n /** @internal */\n this._currentRenderId = -1;\n this._parentUpdateId = -1;\n /** @internal */\n this._childUpdateId = -1;\n /** @internal */\n this._waitingParentId = null;\n /** @internal */\n this._waitingParentInstanceIndex = null;\n /** @internal */\n this._waitingParsedUniqueId = null;\n /** @internal */\n this._cache = {};\n this._parentNode = null;\n /** @internal */\n this._children = null;\n /** @internal */\n this._worldMatrix = Matrix.Identity();\n /** @internal */\n this._worldMatrixDeterminant = 0;\n /** @internal */\n this._worldMatrixDeterminantIsDirty = true;\n this._animationPropertiesOverride = null;\n /** @internal */\n this._isNode = true;\n /**\n * An event triggered when the mesh is disposed\n */\n this.onDisposeObservable = new Observable$1();\n this._onDisposeObserver = null;\n // Behaviors\n this._behaviors = new Array();\n this.name = name;\n this.id = name;\n this._scene = (scene || EngineStore.LastCreatedScene);\n this.uniqueId = this._scene.getUniqueId();\n this._initCache();\n }\n /**\n * Gets the scene of the node\n * @returns a scene\n */\n getScene() {\n return this._scene;\n }\n /**\n * Gets the engine of the node\n * @returns a Engine\n */\n getEngine() {\n return this._scene.getEngine();\n }\n /**\n * Attach a behavior to the node\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors\n * @param behavior defines the behavior to attach\n * @param attachImmediately defines that the behavior must be attached even if the scene is still loading\n * @returns the current Node\n */\n addBehavior(behavior, attachImmediately = false) {\n const index = this._behaviors.indexOf(behavior);\n if (index !== -1) {\n return this;\n }\n behavior.init();\n if (this._scene.isLoading && !attachImmediately) {\n // We defer the attach when the scene will be loaded\n this._scene.onDataLoadedObservable.addOnce(() => {\n behavior.attach(this);\n });\n }\n else {\n behavior.attach(this);\n }\n this._behaviors.push(behavior);\n return this;\n }\n /**\n * Remove an attached behavior\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors\n * @param behavior defines the behavior to attach\n * @returns the current Node\n */\n removeBehavior(behavior) {\n const index = this._behaviors.indexOf(behavior);\n if (index === -1) {\n return this;\n }\n this._behaviors[index].detach();\n this._behaviors.splice(index, 1);\n return this;\n }\n /**\n * Gets the list of attached behaviors\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors\n */\n get behaviors() {\n return this._behaviors;\n }\n /**\n * Gets an attached behavior by name\n * @param name defines the name of the behavior to look for\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors\n * @returns null if behavior was not found else the requested behavior\n */\n getBehaviorByName(name) {\n for (const behavior of this._behaviors) {\n if (behavior.name === name) {\n return behavior;\n }\n }\n return null;\n }\n /**\n * Returns the latest update of the World matrix\n * @returns a Matrix\n */\n getWorldMatrix() {\n if (this._currentRenderId !== this._scene.getRenderId()) {\n this.computeWorldMatrix();\n }\n return this._worldMatrix;\n }\n /** @internal */\n _getWorldMatrixDeterminant() {\n if (this._worldMatrixDeterminantIsDirty) {\n this._worldMatrixDeterminantIsDirty = false;\n this._worldMatrixDeterminant = this._worldMatrix.determinant();\n }\n return this._worldMatrixDeterminant;\n }\n /**\n * Returns directly the latest state of the mesh World matrix.\n * A Matrix is returned.\n */\n get worldMatrixFromCache() {\n return this._worldMatrix;\n }\n // override it in derived class if you add new variables to the cache\n // and call the parent class method\n /** @internal */\n _initCache() {\n this._cache = {};\n this._cache.parent = undefined;\n }\n /**\n * @internal\n */\n updateCache(force) {\n if (!force && this.isSynchronized()) {\n return;\n }\n this._cache.parent = this.parent;\n this._updateCache();\n }\n /**\n * @internal\n */\n _getActionManagerForTrigger(trigger, _initialCall = true) {\n if (!this.parent) {\n return null;\n }\n return this.parent._getActionManagerForTrigger(trigger, false);\n }\n // override it in derived class if you add new variables to the cache\n // and call the parent class method if !ignoreParentClass\n /**\n * @internal\n */\n _updateCache(_ignoreParentClass) { }\n // override it in derived class if you add new variables to the cache\n /** @internal */\n _isSynchronized() {\n return true;\n }\n /** @internal */\n _markSyncedWithParent() {\n if (this._parentNode) {\n this._parentUpdateId = this._parentNode._childUpdateId;\n }\n }\n /** @internal */\n isSynchronizedWithParent() {\n if (!this._parentNode) {\n return true;\n }\n if (this._parentNode._isDirty || this._parentUpdateId !== this._parentNode._childUpdateId) {\n return false;\n }\n return this._parentNode.isSynchronized();\n }\n /** @internal */\n isSynchronized() {\n if (this._cache.parent !== this._parentNode) {\n this._cache.parent = this._parentNode;\n return false;\n }\n if (this._parentNode && !this.isSynchronizedWithParent()) {\n return false;\n }\n return this._isSynchronized();\n }\n /**\n * Is this node ready to be used/rendered\n * @param _completeCheck defines if a complete check (including materials and lights) has to be done (false by default)\n * @returns true if the node is ready\n */\n isReady(_completeCheck = false) {\n return this._nodeDataStorage._isReady;\n }\n /**\n * Flag the node as dirty (Forcing it to update everything)\n * @param _property helps children apply precise \"dirtyfication\"\n * @returns this node\n */\n markAsDirty(_property) {\n this._currentRenderId = Number.MAX_VALUE;\n this._isDirty = true;\n return this;\n }\n /**\n * Is this node enabled?\n * If the node has a parent, all ancestors will be checked and false will be returned if any are false (not enabled), otherwise will return true\n * @param checkAncestors indicates if this method should check the ancestors. The default is to check the ancestors. If set to false, the method will return the value of this node without checking ancestors\n * @returns whether this node (and its parent) is enabled\n */\n isEnabled(checkAncestors = true) {\n if (checkAncestors === false) {\n return this._nodeDataStorage._isEnabled;\n }\n if (!this._nodeDataStorage._isEnabled) {\n return false;\n }\n return this._nodeDataStorage._isParentEnabled;\n }\n /** @internal */\n _syncParentEnabledState() {\n this._nodeDataStorage._isParentEnabled = this._parentNode ? this._parentNode.isEnabled() : true;\n if (this._children) {\n this._children.forEach((c) => {\n c._syncParentEnabledState(); // Force children to update accordingly\n });\n }\n }\n /**\n * Set the enabled state of this node\n * @param value defines the new enabled state\n */\n setEnabled(value) {\n if (this._nodeDataStorage._isEnabled === value) {\n return;\n }\n this._nodeDataStorage._isEnabled = value;\n this._syncParentEnabledState();\n this._nodeDataStorage._onEnabledStateChangedObservable.notifyObservers(value);\n }\n /**\n * Is this node a descendant of the given node?\n * The function will iterate up the hierarchy until the ancestor was found or no more parents defined\n * @param ancestor defines the parent node to inspect\n * @returns a boolean indicating if this node is a descendant of the given node\n */\n isDescendantOf(ancestor) {\n if (this.parent) {\n if (this.parent === ancestor) {\n return true;\n }\n return this.parent.isDescendantOf(ancestor);\n }\n return false;\n }\n /**\n * @internal\n */\n _getDescendants(results, directDescendantsOnly = false, predicate) {\n if (!this._children) {\n return;\n }\n for (let index = 0; index < this._children.length; index++) {\n const item = this._children[index];\n if (!predicate || predicate(item)) {\n results.push(item);\n }\n if (!directDescendantsOnly) {\n item._getDescendants(results, false, predicate);\n }\n }\n }\n /**\n * Will return all nodes that have this node as ascendant\n * @param directDescendantsOnly defines if true only direct descendants of 'this' will be considered, if false direct and also indirect (children of children, an so on in a recursive manner) descendants of 'this' will be considered\n * @param predicate defines an optional predicate that will be called on every evaluated child, the predicate must return true for a given child to be part of the result, otherwise it will be ignored\n * @returns all children nodes of all types\n */\n getDescendants(directDescendantsOnly, predicate) {\n const results = new Array();\n this._getDescendants(results, directDescendantsOnly, predicate);\n return results;\n }\n /**\n * Get all child-meshes of this node\n * @param directDescendantsOnly defines if true only direct descendants of 'this' will be considered, if false direct and also indirect (children of children, an so on in a recursive manner) descendants of 'this' will be considered (Default: false)\n * @param predicate defines an optional predicate that will be called on every evaluated child, the predicate must return true for a given child to be part of the result, otherwise it will be ignored\n * @returns an array of AbstractMesh\n */\n getChildMeshes(directDescendantsOnly, predicate) {\n const results = [];\n this._getDescendants(results, directDescendantsOnly, (node) => {\n return (!predicate || predicate(node)) && node.cullingStrategy !== undefined;\n });\n return results;\n }\n /**\n * Get all direct children of this node\n * @param predicate defines an optional predicate that will be called on every evaluated child, the predicate must return true for a given child to be part of the result, otherwise it will be ignored\n * @param directDescendantsOnly defines if true only direct descendants of 'this' will be considered, if false direct and also indirect (children of children, an so on in a recursive manner) descendants of 'this' will be considered (Default: true)\n * @returns an array of Node\n */\n getChildren(predicate, directDescendantsOnly = true) {\n return this.getDescendants(directDescendantsOnly, predicate);\n }\n /**\n * @internal\n */\n _setReady(state) {\n if (state === this._nodeDataStorage._isReady) {\n return;\n }\n if (!state) {\n this._nodeDataStorage._isReady = false;\n return;\n }\n if (this.onReady) {\n this.onReady(this);\n }\n this._nodeDataStorage._isReady = true;\n }\n /**\n * Get an animation by name\n * @param name defines the name of the animation to look for\n * @returns null if not found else the requested animation\n */\n getAnimationByName(name) {\n for (let i = 0; i < this.animations.length; i++) {\n const animation = this.animations[i];\n if (animation.name === name) {\n return animation;\n }\n }\n return null;\n }\n /**\n * Creates an animation range for this node\n * @param name defines the name of the range\n * @param from defines the starting key\n * @param to defines the end key\n */\n createAnimationRange(name, from, to) {\n // check name not already in use\n if (!this._ranges[name]) {\n this._ranges[name] = Node._AnimationRangeFactory(name, from, to);\n for (let i = 0, nAnimations = this.animations.length; i < nAnimations; i++) {\n if (this.animations[i]) {\n this.animations[i].createRange(name, from, to);\n }\n }\n }\n }\n /**\n * Delete a specific animation range\n * @param name defines the name of the range to delete\n * @param deleteFrames defines if animation frames from the range must be deleted as well\n */\n deleteAnimationRange(name, deleteFrames = true) {\n for (let i = 0, nAnimations = this.animations.length; i < nAnimations; i++) {\n if (this.animations[i]) {\n this.animations[i].deleteRange(name, deleteFrames);\n }\n }\n this._ranges[name] = null; // said much faster than 'delete this._range[name]'\n }\n /**\n * Get an animation range by name\n * @param name defines the name of the animation range to look for\n * @returns null if not found else the requested animation range\n */\n getAnimationRange(name) {\n return this._ranges[name] || null;\n }\n /**\n * Clone the current node\n * @param name Name of the new clone\n * @param newParent New parent for the clone\n * @param doNotCloneChildren Do not clone children hierarchy\n * @returns the new transform node\n */\n clone(name, newParent, doNotCloneChildren) {\n const result = SerializationHelper.Clone(() => new Node(name, this.getScene()), this);\n if (newParent) {\n result.parent = newParent;\n }\n if (!doNotCloneChildren) {\n // Children\n const directDescendants = this.getDescendants(true);\n for (let index = 0; index < directDescendants.length; index++) {\n const child = directDescendants[index];\n child.clone(name + \".\" + child.name, result);\n }\n }\n return result;\n }\n /**\n * Gets the list of all animation ranges defined on this node\n * @returns an array\n */\n getAnimationRanges() {\n const animationRanges = [];\n let name;\n for (name in this._ranges) {\n animationRanges.push(this._ranges[name]);\n }\n return animationRanges;\n }\n /**\n * Will start the animation sequence\n * @param name defines the range frames for animation sequence\n * @param loop defines if the animation should loop (false by default)\n * @param speedRatio defines the speed factor in which to run the animation (1 by default)\n * @param onAnimationEnd defines a function to be executed when the animation ended (undefined by default)\n * @returns the object created for this animation. If range does not exist, it will return null\n */\n beginAnimation(name, loop, speedRatio, onAnimationEnd) {\n const range = this.getAnimationRange(name);\n if (!range) {\n return null;\n }\n return this._scene.beginAnimation(this, range.from, range.to, loop, speedRatio, onAnimationEnd);\n }\n /**\n * Serialize animation ranges into a JSON compatible object\n * @returns serialization object\n */\n serializeAnimationRanges() {\n const serializationRanges = [];\n for (const name in this._ranges) {\n const localRange = this._ranges[name];\n if (!localRange) {\n continue;\n }\n const range = {};\n range.name = name;\n range.from = localRange.from;\n range.to = localRange.to;\n serializationRanges.push(range);\n }\n return serializationRanges;\n }\n /**\n * Computes the world matrix of the node\n * @param _force defines if the cache version should be invalidated forcing the world matrix to be created from scratch\n * @returns the world matrix\n */\n computeWorldMatrix(_force) {\n if (!this._worldMatrix) {\n this._worldMatrix = Matrix.Identity();\n }\n return this._worldMatrix;\n }\n /**\n * Releases resources associated with this node.\n * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default)\n * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default)\n */\n dispose(doNotRecurse, disposeMaterialAndTextures = false) {\n this._nodeDataStorage._isDisposed = true;\n if (!doNotRecurse) {\n const nodes = this.getDescendants(true);\n for (const node of nodes) {\n node.dispose(doNotRecurse, disposeMaterialAndTextures);\n }\n }\n if (!this.parent) {\n this._removeFromSceneRootNodes();\n }\n else {\n this.parent = null;\n }\n // Callback\n this.onDisposeObservable.notifyObservers(this);\n this.onDisposeObservable.clear();\n this.onEnabledStateChangedObservable.clear();\n this.onClonedObservable.clear();\n // Behaviors\n for (const behavior of this._behaviors) {\n behavior.detach();\n }\n this._behaviors.length = 0;\n this.metadata = null;\n }\n /**\n * Parse animation range data from a serialization object and store them into a given node\n * @param node defines where to store the animation ranges\n * @param parsedNode defines the serialization object to read data from\n * @param _scene defines the hosting scene\n */\n static ParseAnimationRanges(node, parsedNode, _scene) {\n if (parsedNode.ranges) {\n for (let index = 0; index < parsedNode.ranges.length; index++) {\n const data = parsedNode.ranges[index];\n node.createAnimationRange(data.name, data.from, data.to);\n }\n }\n }\n /**\n * Return the minimum and maximum world vectors of the entire hierarchy under current node\n * @param includeDescendants Include bounding info from descendants as well (true by default)\n * @param predicate defines a callback function that can be customize to filter what meshes should be included in the list used to compute the bounding vectors\n * @returns the new bounding vectors\n */\n getHierarchyBoundingVectors(includeDescendants = true, predicate = null) {\n // Ensures that all world matrix will be recomputed.\n this.getScene().incrementRenderId();\n this.computeWorldMatrix(true);\n let min;\n let max;\n const thisAbstractMesh = this;\n if (thisAbstractMesh.getBoundingInfo && thisAbstractMesh.subMeshes) {\n // If this is an abstract mesh get its bounding info\n const boundingInfo = thisAbstractMesh.getBoundingInfo();\n min = boundingInfo.boundingBox.minimumWorld.clone();\n max = boundingInfo.boundingBox.maximumWorld.clone();\n }\n else {\n min = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\n max = new Vector3(-Number.MAX_VALUE, -Number.MAX_VALUE, -Number.MAX_VALUE);\n }\n if (includeDescendants) {\n const descendants = this.getDescendants(false);\n for (const descendant of descendants) {\n const childMesh = descendant;\n childMesh.computeWorldMatrix(true);\n // Filters meshes based on custom predicate function.\n if (predicate && !predicate(childMesh)) {\n continue;\n }\n //make sure we have the needed params to get mix and max\n if (!childMesh.getBoundingInfo || childMesh.getTotalVertices() === 0) {\n continue;\n }\n const childBoundingInfo = childMesh.getBoundingInfo();\n const boundingBox = childBoundingInfo.boundingBox;\n const minBox = boundingBox.minimumWorld;\n const maxBox = boundingBox.maximumWorld;\n Vector3.CheckExtends(minBox, min, max);\n Vector3.CheckExtends(maxBox, min, max);\n }\n }\n return {\n min: min,\n max: max,\n };\n }\n }\n /**\n * @internal\n */\n Node._AnimationRangeFactory = (_name, _from, _to) => {\n throw _WarnImport(\"AnimationRange\");\n };\n Node._NodeConstructors = {};\n __decorate$1([\n serialize()\n ], Node.prototype, \"name\", void 0);\n __decorate$1([\n serialize()\n ], Node.prototype, \"id\", void 0);\n __decorate$1([\n serialize()\n ], Node.prototype, \"uniqueId\", void 0);\n __decorate$1([\n serialize()\n ], Node.prototype, \"state\", void 0);\n __decorate$1([\n serialize()\n ], Node.prototype, \"metadata\", void 0);\n\n /* eslint-disable @typescript-eslint/naming-convention */\n /**\n * Checks if the window object exists\n * @returns true if the window object exists\n */\n function IsWindowObjectExist() {\n return typeof window !== \"undefined\";\n }\n /**\n * Checks if the navigator object exists\n * @returns true if the navigator object exists\n */\n function IsNavigatorAvailable() {\n return typeof navigator !== \"undefined\";\n }\n /**\n * Check if the document object exists\n * @returns true if the document object exists\n */\n function IsDocumentAvailable() {\n return typeof document !== \"undefined\";\n }\n /**\n * Extracts text content from a DOM element hierarchy\n * @param element defines the root element\n * @returns a string\n */\n function GetDOMTextContent(element) {\n let result = \"\";\n let child = element.firstChild;\n while (child) {\n if (child.nodeType === 3) {\n result += child.textContent;\n }\n child = child.nextSibling;\n }\n return result;\n }\n\n /**\n * Logger used throughout the application to allow configuration of\n * the log level required for the messages.\n */\n class Logger {\n static _CheckLimit(message, limit) {\n let entry = Logger._LogLimitOutputs[message];\n if (!entry) {\n entry = { limit, current: 1 };\n Logger._LogLimitOutputs[message] = entry;\n }\n else {\n entry.current++;\n }\n return entry.current <= entry.limit;\n }\n static _GenerateLimitMessage(message, level = 1) {\n var _a;\n const entry = Logger._LogLimitOutputs[message];\n if (!entry || !Logger.MessageLimitReached) {\n return;\n }\n const type = this._Levels[level];\n if (entry.current === entry.limit) {\n Logger[type.name](Logger.MessageLimitReached.replace(/%LIMIT%/g, \"\" + entry.limit).replace(/%TYPE%/g, (_a = type.name) !== null && _a !== void 0 ? _a : \"\"));\n }\n }\n static _AddLogEntry(entry) {\n Logger._LogCache = entry + Logger._LogCache;\n if (Logger.OnNewCacheEntry) {\n Logger.OnNewCacheEntry(entry);\n }\n }\n static _FormatMessage(message) {\n const padStr = (i) => (i < 10 ? \"0\" + i : \"\" + i);\n const date = new Date();\n return \"[\" + padStr(date.getHours()) + \":\" + padStr(date.getMinutes()) + \":\" + padStr(date.getSeconds()) + \"]: \" + message;\n }\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n static _LogDisabled(message, limit) {\n // nothing to do\n }\n static _LogEnabled(level = 1, message, limit) {\n if (limit !== undefined && !Logger._CheckLimit(message, limit)) {\n return;\n }\n const formattedMessage = Logger._FormatMessage(message);\n const type = this._Levels[level];\n type.logFunc && type.logFunc(\"BJS - \" + formattedMessage);\n const entry = `
${formattedMessage}

`;\n Logger._AddLogEntry(entry);\n Logger._GenerateLimitMessage(message, level);\n }\n /**\n * Gets current log cache (list of logs)\n */\n static get LogCache() {\n return Logger._LogCache;\n }\n /**\n * Clears the log cache\n */\n static ClearLogCache() {\n Logger._LogCache = \"\";\n Logger._LogLimitOutputs = {};\n Logger.errorsCount = 0;\n }\n /**\n * Sets the current log level (MessageLogLevel / WarningLogLevel / ErrorLogLevel)\n */\n static set LogLevels(level) {\n Logger.Log = Logger._LogDisabled;\n Logger.Warn = Logger._LogDisabled;\n Logger.Error = Logger._LogDisabled;\n [Logger.MessageLogLevel, Logger.WarningLogLevel, Logger.ErrorLogLevel].forEach((l) => {\n if ((level & l) === l) {\n const type = this._Levels[l];\n Logger[type.name] = Logger._LogEnabled.bind(Logger, l);\n }\n });\n }\n }\n /**\n * No log\n */\n Logger.NoneLogLevel = 0;\n /**\n * Only message logs\n */\n Logger.MessageLogLevel = 1;\n /**\n * Only warning logs\n */\n Logger.WarningLogLevel = 2;\n /**\n * Only error logs\n */\n Logger.ErrorLogLevel = 4;\n /**\n * All logs\n */\n Logger.AllLogLevel = 7;\n /**\n * Message to display when a message has been logged too many times\n */\n Logger.MessageLimitReached = \"Too many %TYPE%s (%LIMIT%), no more %TYPE%s will be reported for this message.\";\n Logger._LogCache = \"\";\n Logger._LogLimitOutputs = {};\n // levels according to the (binary) numbering.\n Logger._Levels = [\n {},\n { color: \"white\", logFunc: console.log, name: \"Log\" },\n { color: \"orange\", logFunc: console.warn, name: \"Warn\" },\n {},\n { color: \"red\", logFunc: console.error, name: \"Error\" },\n ];\n /**\n * Gets a value indicating the number of loading errors\n * @ignorenaming\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n Logger.errorsCount = 0;\n /**\n * Log a message to the console\n */\n Logger.Log = Logger._LogEnabled.bind(Logger, Logger.MessageLogLevel);\n /**\n * Write a warning message to the console\n */\n Logger.Warn = Logger._LogEnabled.bind(Logger, Logger.WarningLogLevel);\n /**\n * Write an error message to the console\n */\n Logger.Error = Logger._LogEnabled.bind(Logger, Logger.ErrorLogLevel);\n\n const CloneValue = (source, destinationObject) => {\n if (!source) {\n return null;\n }\n if (source.getClassName && source.getClassName() === \"Mesh\") {\n return null;\n }\n if (source.getClassName && (source.getClassName() === \"SubMesh\" || source.getClassName() === \"PhysicsBody\")) {\n return source.clone(destinationObject);\n }\n else if (source.clone) {\n return source.clone();\n }\n else if (Array.isArray(source)) {\n return source.slice();\n }\n return null;\n };\n function GetAllPropertyNames(obj) {\n const props = [];\n do {\n Object.getOwnPropertyNames(obj).forEach(function (prop) {\n if (props.indexOf(prop) === -1) {\n props.push(prop);\n }\n });\n } while ((obj = Object.getPrototypeOf(obj)));\n return props;\n }\n /**\n * Class containing a set of static utilities functions for deep copy.\n */\n class DeepCopier {\n /**\n * Tries to copy an object by duplicating every property\n * @param source defines the source object\n * @param destination defines the target object\n * @param doNotCopyList defines a list of properties to avoid\n * @param mustCopyList defines a list of properties to copy (even if they start with _)\n */\n static DeepCopy(source, destination, doNotCopyList, mustCopyList) {\n const properties = GetAllPropertyNames(source);\n for (const prop of properties) {\n if (prop[0] === \"_\" && (!mustCopyList || mustCopyList.indexOf(prop) === -1)) {\n continue;\n }\n if (prop.endsWith(\"Observable\")) {\n continue;\n }\n if (doNotCopyList && doNotCopyList.indexOf(prop) !== -1) {\n continue;\n }\n const sourceValue = source[prop];\n const typeOfSourceValue = typeof sourceValue;\n if (typeOfSourceValue === \"function\") {\n continue;\n }\n try {\n if (typeOfSourceValue === \"object\") {\n if (sourceValue instanceof Uint8Array) {\n destination[prop] = Uint8Array.from(sourceValue);\n }\n else if (sourceValue instanceof Array) {\n destination[prop] = [];\n if (sourceValue.length > 0) {\n if (typeof sourceValue[0] == \"object\") {\n for (let index = 0; index < sourceValue.length; index++) {\n const clonedValue = CloneValue(sourceValue[index], destination);\n if (destination[prop].indexOf(clonedValue) === -1) {\n // Test if auto inject was not done\n destination[prop].push(clonedValue);\n }\n }\n }\n else {\n destination[prop] = sourceValue.slice(0);\n }\n }\n }\n else {\n destination[prop] = CloneValue(sourceValue, destination);\n }\n }\n else {\n destination[prop] = sourceValue;\n }\n }\n catch (e) {\n // Log a warning (it could be because of a read-only property)\n Logger.Warn(e.message);\n }\n }\n }\n }\n\n /**\n * Class containing a set of static utilities functions for precision date\n */\n class PrecisionDate {\n /**\n * Gets either window.performance.now() if supported or Date.now() else\n */\n static get Now() {\n if (IsWindowObjectExist() && window.performance && window.performance.now) {\n return window.performance.now();\n }\n return Date.now();\n }\n }\n\n /** @internal */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n function createXMLHttpRequest() {\n // If running in Babylon Native, then defer to the native XMLHttpRequest, which has the same public contract\n if (typeof _native !== \"undefined\" && _native.XMLHttpRequest) {\n return new _native.XMLHttpRequest();\n }\n else {\n return new XMLHttpRequest();\n }\n }\n /**\n * Extended version of XMLHttpRequest with support for customizations (headers, ...)\n */\n class WebRequest {\n constructor() {\n this._xhr = createXMLHttpRequest();\n this._requestURL = \"\";\n }\n _injectCustomRequestHeaders() {\n if (this._shouldSkipRequestModifications(this._requestURL)) {\n return;\n }\n for (const key in WebRequest.CustomRequestHeaders) {\n const val = WebRequest.CustomRequestHeaders[key];\n if (val) {\n this._xhr.setRequestHeader(key, val);\n }\n }\n }\n _shouldSkipRequestModifications(url) {\n return WebRequest.SkipRequestModificationForBabylonCDN && (url.includes(\"preview.babylonjs.com\") || url.includes(\"cdn.babylonjs.com\"));\n }\n /**\n * Gets or sets a function to be called when loading progress changes\n */\n get onprogress() {\n return this._xhr.onprogress;\n }\n set onprogress(value) {\n this._xhr.onprogress = value;\n }\n /**\n * Returns client's state\n */\n get readyState() {\n return this._xhr.readyState;\n }\n /**\n * Returns client's status\n */\n get status() {\n return this._xhr.status;\n }\n /**\n * Returns client's status as a text\n */\n get statusText() {\n return this._xhr.statusText;\n }\n /**\n * Returns client's response\n */\n get response() {\n return this._xhr.response;\n }\n /**\n * Returns client's response url\n */\n get responseURL() {\n return this._xhr.responseURL;\n }\n /**\n * Returns client's response as text\n */\n get responseText() {\n return this._xhr.responseText;\n }\n /**\n * Gets or sets the expected response type\n */\n get responseType() {\n return this._xhr.responseType;\n }\n set responseType(value) {\n this._xhr.responseType = value;\n }\n /**\n * Gets or sets the timeout value in milliseconds\n */\n get timeout() {\n return this._xhr.timeout;\n }\n set timeout(value) {\n this._xhr.timeout = value;\n }\n addEventListener(type, listener, options) {\n this._xhr.addEventListener(type, listener, options);\n }\n removeEventListener(type, listener, options) {\n this._xhr.removeEventListener(type, listener, options);\n }\n /**\n * Cancels any network activity\n */\n abort() {\n this._xhr.abort();\n }\n /**\n * Initiates the request. The optional argument provides the request body. The argument is ignored if request method is GET or HEAD\n * @param body defines an optional request body\n */\n send(body) {\n if (WebRequest.CustomRequestHeaders) {\n this._injectCustomRequestHeaders();\n }\n this._xhr.send(body);\n }\n /**\n * Sets the request method, request URL\n * @param method defines the method to use (GET, POST, etc..)\n * @param url defines the url to connect with\n */\n open(method, url) {\n for (const update of WebRequest.CustomRequestModifiers) {\n if (this._shouldSkipRequestModifications(url)) {\n return;\n }\n update(this._xhr, url);\n }\n // Clean url\n url = url.replace(\"file:http:\", \"http:\");\n url = url.replace(\"file:https:\", \"https:\");\n this._requestURL = url;\n return this._xhr.open(method, url, true);\n }\n /**\n * Sets the value of a request header.\n * @param name The name of the header whose value is to be set\n * @param value The value to set as the body of the header\n */\n setRequestHeader(name, value) {\n this._xhr.setRequestHeader(name, value);\n }\n /**\n * Get the string containing the text of a particular header's value.\n * @param name The name of the header\n * @returns The string containing the text of the given header name\n */\n getResponseHeader(name) {\n return this._xhr.getResponseHeader(name);\n }\n }\n /**\n * Custom HTTP Request Headers to be sent with XMLHttpRequests\n * i.e. when loading files, where the server/service expects an Authorization header\n */\n WebRequest.CustomRequestHeaders = {};\n /**\n * Add callback functions in this array to update all the requests before they get sent to the network\n */\n WebRequest.CustomRequestModifiers = new Array();\n WebRequest.SkipRequestModificationForBabylonCDN = true;\n\n /**\n * Class used to help managing file picking and drag'n'drop\n * File Storage\n */\n class FilesInputStore {\n }\n /**\n * List of files ready to be loaded\n */\n FilesInputStore.FilesToLoad = {};\n\n /**\n * Class used to define a retry strategy when error happens while loading assets\n */\n class RetryStrategy {\n /**\n * Function used to defines an exponential back off strategy\n * @param maxRetries defines the maximum number of retries (3 by default)\n * @param baseInterval defines the interval between retries\n * @returns the strategy function to use\n */\n static ExponentialBackoff(maxRetries = 3, baseInterval = 500) {\n return (url, request, retryIndex) => {\n if (request.status !== 0 || retryIndex >= maxRetries || url.indexOf(\"file:\") !== -1) {\n return -1;\n }\n return Math.pow(2, retryIndex) * baseInterval;\n };\n }\n }\n\n /* eslint-disable @typescript-eslint/naming-convention */\n /**\n * Base error. Due to limitations of typedoc-check and missing documentation\n * in lib.es5.d.ts, cannot extend Error directly for RuntimeError.\n * @ignore\n */\n class BaseError extends Error {\n }\n // See https://stackoverflow.com/questions/12915412/how-do-i-extend-a-host-object-e-g-error-in-typescript\n // and https://github.com/Microsoft/TypeScript/wiki/Breaking-Changes#extending-built-ins-like-error-array-and-map-may-no-longer-work\n // Polyfill for Object.setPrototypeOf if necessary.\n BaseError._setPrototypeOf = Object.setPrototypeOf ||\n ((o, proto) => {\n o.__proto__ = proto;\n return o;\n });\n /* IMP! DO NOT CHANGE THE NUMBERING OF EXISTING ERROR CODES */\n /**\n * Error codes for BaseError\n */\n const ErrorCodes = {\n // Mesh errors 0-999\n /** Invalid or empty mesh vertex positions. */\n MeshInvalidPositionsError: 0,\n // Texture errors 1000-1999\n /** Unsupported texture found. */\n UnsupportedTextureError: 1000,\n // GLTFLoader errors 2000-2999\n /** Unexpected magic number found in GLTF file header. */\n GLTFLoaderUnexpectedMagicError: 2000,\n // SceneLoader errors 3000-3999\n /** SceneLoader generic error code. Ideally wraps the inner exception. */\n SceneLoaderError: 3000,\n // File related errors 4000-4999\n /** Load file error */\n LoadFileError: 4000,\n /** Request file error */\n RequestFileError: 4001,\n /** Read file error */\n ReadFileError: 4002,\n };\n /**\n * Application runtime error\n */\n class RuntimeError extends BaseError {\n /**\n * Creates a new RuntimeError\n * @param message defines the message of the error\n * @param errorCode the error code\n * @param innerError the error that caused the outer error\n */\n constructor(message, errorCode, innerError) {\n super(message);\n this.errorCode = errorCode;\n this.innerError = innerError;\n this.name = \"RuntimeError\";\n BaseError._setPrototypeOf(this, RuntimeError.prototype);\n }\n }\n\n /* eslint-disable @typescript-eslint/naming-convention */\n /**\n * Encode a buffer to a base64 string\n * @param buffer defines the buffer to encode\n * @returns the encoded string\n */\n const EncodeArrayBufferToBase64 = (buffer) => {\n const keyStr = \"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=\";\n let output = \"\";\n let chr1, chr2, chr3, enc1, enc2, enc3, enc4;\n let i = 0;\n const bytes = ArrayBuffer.isView(buffer) ? new Uint8Array(buffer.buffer, buffer.byteOffset, buffer.byteLength) : new Uint8Array(buffer);\n while (i < bytes.length) {\n chr1 = bytes[i++];\n chr2 = i < bytes.length ? bytes[i++] : Number.NaN;\n chr3 = i < bytes.length ? bytes[i++] : Number.NaN;\n enc1 = chr1 >> 2;\n enc2 = ((chr1 & 3) << 4) | (chr2 >> 4);\n enc3 = ((chr2 & 15) << 2) | (chr3 >> 6);\n enc4 = chr3 & 63;\n if (isNaN(chr2)) {\n enc3 = enc4 = 64;\n }\n else if (isNaN(chr3)) {\n enc4 = 64;\n }\n output += keyStr.charAt(enc1) + keyStr.charAt(enc2) + keyStr.charAt(enc3) + keyStr.charAt(enc4);\n }\n return output;\n };\n /**\n * Converts a given base64 string as an ASCII encoded stream of data\n * @param base64Data The base64 encoded string to decode\n * @returns Decoded ASCII string\n */\n const DecodeBase64ToString = (base64Data) => {\n return atob(base64Data);\n };\n /**\n * Converts a given base64 string into an ArrayBuffer of raw byte data\n * @param base64Data The base64 encoded string to decode\n * @returns ArrayBuffer of byte data\n */\n const DecodeBase64ToBinary = (base64Data) => {\n const decodedString = DecodeBase64ToString(base64Data);\n const bufferLength = decodedString.length;\n const bufferView = new Uint8Array(new ArrayBuffer(bufferLength));\n for (let i = 0; i < bufferLength; i++) {\n bufferView[i] = decodedString.charCodeAt(i);\n }\n return bufferView.buffer;\n };\n\n const defaultAttributeKeywordName = \"attribute\";\n const defaultVaryingKeywordName = \"varying\";\n /** @internal */\n class ShaderCodeNode {\n constructor() {\n this.children = [];\n }\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n isValid(preprocessors) {\n return true;\n }\n process(preprocessors, options) {\n var _a, _b, _c, _d, _e, _f, _g;\n let result = \"\";\n if (this.line) {\n let value = this.line;\n const processor = options.processor;\n if (processor) {\n // This must be done before other replacements to avoid mistakenly changing something that was already changed.\n if (processor.lineProcessor) {\n value = processor.lineProcessor(value, options.isFragment, options.processingContext);\n }\n const attributeKeyword = (_b = (_a = options.processor) === null || _a === void 0 ? void 0 : _a.attributeKeywordName) !== null && _b !== void 0 ? _b : defaultAttributeKeywordName;\n const varyingKeyword = options.isFragment && ((_c = options.processor) === null || _c === void 0 ? void 0 : _c.varyingFragmentKeywordName)\n ? (_d = options.processor) === null || _d === void 0 ? void 0 : _d.varyingFragmentKeywordName\n : !options.isFragment && ((_e = options.processor) === null || _e === void 0 ? void 0 : _e.varyingVertexKeywordName)\n ? (_f = options.processor) === null || _f === void 0 ? void 0 : _f.varyingVertexKeywordName\n : defaultVaryingKeywordName;\n if (!options.isFragment && processor.attributeProcessor && this.line.startsWith(attributeKeyword)) {\n value = processor.attributeProcessor(this.line, preprocessors, options.processingContext);\n }\n else if (processor.varyingProcessor &&\n (((_g = processor.varyingCheck) === null || _g === void 0 ? void 0 : _g.call(processor, this.line, options.isFragment)) || (!processor.varyingCheck && this.line.startsWith(varyingKeyword)))) {\n value = processor.varyingProcessor(this.line, options.isFragment, preprocessors, options.processingContext);\n }\n else if (processor.uniformProcessor && processor.uniformRegexp && processor.uniformRegexp.test(this.line)) {\n if (!options.lookForClosingBracketForUniformBuffer) {\n value = processor.uniformProcessor(this.line, options.isFragment, preprocessors, options.processingContext);\n }\n }\n else if (processor.uniformBufferProcessor && processor.uniformBufferRegexp && processor.uniformBufferRegexp.test(this.line)) {\n if (!options.lookForClosingBracketForUniformBuffer) {\n value = processor.uniformBufferProcessor(this.line, options.isFragment, options.processingContext);\n options.lookForClosingBracketForUniformBuffer = true;\n }\n }\n else if (processor.textureProcessor && processor.textureRegexp && processor.textureRegexp.test(this.line)) {\n value = processor.textureProcessor(this.line, options.isFragment, preprocessors, options.processingContext);\n }\n else if ((processor.uniformProcessor || processor.uniformBufferProcessor) && this.line.startsWith(\"uniform\") && !options.lookForClosingBracketForUniformBuffer) {\n const regex = /uniform\\s+(?:(?:highp)?|(?:lowp)?)\\s*(\\S+)\\s+(\\S+)\\s*;/;\n if (regex.test(this.line)) {\n // uniform\n if (processor.uniformProcessor) {\n value = processor.uniformProcessor(this.line, options.isFragment, preprocessors, options.processingContext);\n }\n }\n else {\n // Uniform buffer\n if (processor.uniformBufferProcessor) {\n value = processor.uniformBufferProcessor(this.line, options.isFragment, options.processingContext);\n options.lookForClosingBracketForUniformBuffer = true;\n }\n }\n }\n if (options.lookForClosingBracketForUniformBuffer && this.line.indexOf(\"}\") !== -1) {\n options.lookForClosingBracketForUniformBuffer = false;\n if (processor.endOfUniformBufferProcessor) {\n value = processor.endOfUniformBufferProcessor(this.line, options.isFragment, options.processingContext);\n }\n }\n }\n result += value + \"\\n\";\n }\n this.children.forEach((child) => {\n result += child.process(preprocessors, options);\n });\n if (this.additionalDefineKey) {\n preprocessors[this.additionalDefineKey] = this.additionalDefineValue || \"true\";\n }\n return result;\n }\n }\n\n /** @internal */\n class ShaderCodeCursor {\n constructor() {\n this._lines = [];\n }\n get currentLine() {\n return this._lines[this.lineIndex];\n }\n get canRead() {\n return this.lineIndex < this._lines.length - 1;\n }\n set lines(value) {\n this._lines.length = 0;\n for (const line of value) {\n // Skip empty lines\n if (!line || line === \"\\r\") {\n continue;\n }\n // Prevent removing line break in macros.\n if (line[0] === \"#\") {\n this._lines.push(line);\n continue;\n }\n // Do not split single line comments\n const trimmedLine = line.trim();\n if (!trimmedLine) {\n continue;\n }\n if (trimmedLine.startsWith(\"//\")) {\n this._lines.push(line);\n continue;\n }\n // Work with semicolon in the line\n const semicolonIndex = trimmedLine.indexOf(\";\");\n if (semicolonIndex === -1) {\n // No semicolon in the line\n this._lines.push(trimmedLine);\n }\n else if (semicolonIndex === trimmedLine.length - 1) {\n // Single semicolon at the end of the line\n // If trimmedLine == \";\", we must not push, to be backward compatible with the old code!\n if (trimmedLine.length > 1) {\n this._lines.push(trimmedLine);\n }\n }\n else {\n // Semicolon in the middle of the line\n const split = line.split(\";\");\n for (let index = 0; index < split.length; index++) {\n let subLine = split[index];\n if (!subLine) {\n continue;\n }\n subLine = subLine.trim();\n if (!subLine) {\n continue;\n }\n this._lines.push(subLine + (index !== split.length - 1 ? \";\" : \"\"));\n }\n }\n }\n }\n }\n\n /** @internal */\n class ShaderCodeConditionNode extends ShaderCodeNode {\n process(preprocessors, options) {\n for (let index = 0; index < this.children.length; index++) {\n const node = this.children[index];\n if (node.isValid(preprocessors)) {\n return node.process(preprocessors, options);\n }\n }\n return \"\";\n }\n }\n\n /** @internal */\n class ShaderCodeTestNode extends ShaderCodeNode {\n isValid(preprocessors) {\n return this.testExpression.isTrue(preprocessors);\n }\n }\n\n /* eslint-disable @typescript-eslint/naming-convention */\n /** @internal */\n class ShaderDefineExpression {\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n isTrue(preprocessors) {\n return true;\n }\n static postfixToInfix(postfix) {\n const stack = [];\n for (const c of postfix) {\n if (ShaderDefineExpression._OperatorPriority[c] === undefined) {\n stack.push(c);\n }\n else {\n const v1 = stack[stack.length - 1], v2 = stack[stack.length - 2];\n stack.length -= 2;\n stack.push(`(${v2}${c}${v1})`);\n }\n }\n return stack[stack.length - 1];\n }\n /**\n * Converts an infix expression to a postfix expression.\n *\n * This method is used to transform infix expressions, which are more human-readable,\n * into postfix expressions, also known as Reverse Polish Notation (RPN), that can be\n * evaluated more efficiently by a computer. The conversion is based on the operator\n * priority defined in _OperatorPriority.\n *\n * The function employs a stack-based algorithm for the conversion and caches the result\n * to improve performance. The cache keeps track of each converted expression's access time\n * to manage the cache size and optimize memory usage. When the cache size exceeds a specified\n * limit, the least recently accessed items in the cache are deleted.\n *\n * The cache mechanism is particularly helpful for shader compilation, where the same infix\n * expressions might be encountered repeatedly, hence the caching can speed up the process.\n *\n * @param infix - The infix expression to be converted.\n * @returns The postfix expression as an array of strings.\n */\n static infixToPostfix(infix) {\n // Is infix already in cache\n const cacheItem = ShaderDefineExpression._InfixToPostfixCache.get(infix);\n if (cacheItem) {\n cacheItem.accessTime = Date.now();\n return cacheItem.result;\n }\n // Is infix contain any operator\n if (!infix.includes(\"&&\") && !infix.includes(\"||\") && !infix.includes(\")\") && !infix.includes(\"(\")) {\n return [infix];\n }\n const result = [];\n let stackIdx = -1;\n const pushOperand = () => {\n operand = operand.trim();\n if (operand !== \"\") {\n result.push(operand);\n operand = \"\";\n }\n };\n const push = (s) => {\n if (stackIdx < ShaderDefineExpression._Stack.length - 1) {\n ShaderDefineExpression._Stack[++stackIdx] = s;\n }\n };\n const peek = () => ShaderDefineExpression._Stack[stackIdx];\n const pop = () => (stackIdx === -1 ? \"!!INVALID EXPRESSION!!\" : ShaderDefineExpression._Stack[stackIdx--]);\n let idx = 0, operand = \"\";\n while (idx < infix.length) {\n const c = infix.charAt(idx), token = idx < infix.length - 1 ? infix.substr(idx, 2) : \"\";\n if (c === \"(\") {\n operand = \"\";\n push(c);\n }\n else if (c === \")\") {\n pushOperand();\n while (stackIdx !== -1 && peek() !== \"(\") {\n result.push(pop());\n }\n pop();\n }\n else if (ShaderDefineExpression._OperatorPriority[token] > 1) {\n pushOperand();\n while (stackIdx !== -1 && ShaderDefineExpression._OperatorPriority[peek()] >= ShaderDefineExpression._OperatorPriority[token]) {\n result.push(pop());\n }\n push(token);\n idx++;\n }\n else {\n operand += c;\n }\n idx++;\n }\n pushOperand();\n while (stackIdx !== -1) {\n if (peek() === \"(\") {\n pop();\n }\n else {\n result.push(pop());\n }\n }\n // If the cache is at capacity, clear it before adding a new item\n if (ShaderDefineExpression._InfixToPostfixCache.size >= ShaderDefineExpression.InfixToPostfixCacheLimitSize) {\n ShaderDefineExpression.ClearCache();\n }\n // Add the new item to the cache, including the current time as the last access time\n ShaderDefineExpression._InfixToPostfixCache.set(infix, { result, accessTime: Date.now() });\n return result;\n }\n static ClearCache() {\n // Convert the cache to an array and sort by last access time\n const sortedCache = Array.from(ShaderDefineExpression._InfixToPostfixCache.entries()).sort((a, b) => a[1].accessTime - b[1].accessTime);\n // Remove the least recently accessed half of the cache\n for (let i = 0; i < ShaderDefineExpression.InfixToPostfixCacheCleanupSize; i++) {\n ShaderDefineExpression._InfixToPostfixCache.delete(sortedCache[i][0]);\n }\n }\n }\n /**\n * Cache items count limit for the InfixToPostfix cache.\n * It uses to improve the performance of the shader compilation.\n * For details see PR: https://github.com/BabylonJS/Babylon.js/pull/13936\n */\n ShaderDefineExpression.InfixToPostfixCacheLimitSize = 50000;\n /**\n * When the cache size is exceeded, a cache cleanup will be triggered\n * and the cache will be reduced by the size specified\n * in the InfixToPostfixCacheCleanupSize variable, removing entries\n * that have not been accessed the longest.\n */\n ShaderDefineExpression.InfixToPostfixCacheCleanupSize = 25000;\n ShaderDefineExpression._InfixToPostfixCache = new Map();\n ShaderDefineExpression._OperatorPriority = {\n \")\": 0,\n \"(\": 1,\n \"||\": 2,\n \"&&\": 3,\n };\n ShaderDefineExpression._Stack = [\"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\"];\n\n /** @internal */\n class ShaderDefineIsDefinedOperator extends ShaderDefineExpression {\n constructor(define, not = false) {\n super();\n this.define = define;\n this.not = not;\n }\n isTrue(preprocessors) {\n let condition = preprocessors[this.define] !== undefined;\n if (this.not) {\n condition = !condition;\n }\n return condition;\n }\n }\n\n /** @internal */\n class ShaderDefineOrOperator extends ShaderDefineExpression {\n isTrue(preprocessors) {\n return this.leftOperand.isTrue(preprocessors) || this.rightOperand.isTrue(preprocessors);\n }\n }\n\n /** @internal */\n class ShaderDefineAndOperator extends ShaderDefineExpression {\n isTrue(preprocessors) {\n return this.leftOperand.isTrue(preprocessors) && this.rightOperand.isTrue(preprocessors);\n }\n }\n\n /** @internal */\n class ShaderDefineArithmeticOperator extends ShaderDefineExpression {\n constructor(define, operand, testValue) {\n super();\n this.define = define;\n this.operand = operand;\n this.testValue = testValue;\n }\n isTrue(preprocessors) {\n let value = preprocessors[this.define];\n if (value === undefined) {\n value = this.define;\n }\n let condition = false;\n const left = parseInt(value);\n const right = parseInt(this.testValue);\n switch (this.operand) {\n case \">\":\n condition = left > right;\n break;\n case \"<\":\n condition = left < right;\n break;\n case \"<=\":\n condition = left <= right;\n break;\n case \">=\":\n condition = left >= right;\n break;\n case \"==\":\n condition = left === right;\n break;\n case \"!=\":\n condition = left !== right;\n break;\n }\n return condition;\n }\n }\n\n /**\n * Language of the shader code\n */\n var ShaderLanguage;\n (function (ShaderLanguage) {\n /** language is GLSL (used by WebGL) */\n ShaderLanguage[ShaderLanguage[\"GLSL\"] = 0] = \"GLSL\";\n /** language is WGSL (used by WebGPU) */\n ShaderLanguage[ShaderLanguage[\"WGSL\"] = 1] = \"WGSL\";\n })(ShaderLanguage || (ShaderLanguage = {}));\n\n /* eslint-disable @typescript-eslint/no-unused-vars */\n const regexSE = /defined\\s*?\\((.+?)\\)/g;\n const regexSERevert = /defined\\s*?\\[(.+?)\\]/g;\n const regexShaderInclude = /#include\\s?<(.+)>(\\((.*)\\))*(\\[(.*)\\])*/g;\n const regexShaderDecl = /__decl__/;\n const regexLightX = /light\\{X\\}.(\\w*)/g;\n const regexX = /\\{X\\}/g;\n const reusableMatches = [];\n /** @internal */\n class ShaderProcessor {\n static Initialize(options) {\n if (options.processor && options.processor.initializeShaders) {\n options.processor.initializeShaders(options.processingContext);\n }\n }\n static Process(sourceCode, options, callback, engine) {\n var _a;\n if ((_a = options.processor) === null || _a === void 0 ? void 0 : _a.preProcessShaderCode) {\n sourceCode = options.processor.preProcessShaderCode(sourceCode, options.isFragment);\n }\n this._ProcessIncludes(sourceCode, options, (codeWithIncludes) => {\n if (options.processCodeAfterIncludes) {\n codeWithIncludes = options.processCodeAfterIncludes(options.isFragment ? \"fragment\" : \"vertex\", codeWithIncludes);\n }\n const migratedCode = this._ProcessShaderConversion(codeWithIncludes, options, engine);\n callback(migratedCode, codeWithIncludes);\n });\n }\n static PreProcess(sourceCode, options, callback, engine) {\n var _a;\n if ((_a = options.processor) === null || _a === void 0 ? void 0 : _a.preProcessShaderCode) {\n sourceCode = options.processor.preProcessShaderCode(sourceCode, options.isFragment);\n }\n this._ProcessIncludes(sourceCode, options, (codeWithIncludes) => {\n if (options.processCodeAfterIncludes) {\n codeWithIncludes = options.processCodeAfterIncludes(options.isFragment ? \"fragment\" : \"vertex\", codeWithIncludes);\n }\n const migratedCode = this._ApplyPreProcessing(codeWithIncludes, options, engine);\n callback(migratedCode, codeWithIncludes);\n });\n }\n static Finalize(vertexCode, fragmentCode, options) {\n if (!options.processor || !options.processor.finalizeShaders) {\n return { vertexCode, fragmentCode };\n }\n return options.processor.finalizeShaders(vertexCode, fragmentCode, options.processingContext);\n }\n static _ProcessPrecision(source, options) {\n var _a;\n if ((_a = options.processor) === null || _a === void 0 ? void 0 : _a.noPrecision) {\n return source;\n }\n const shouldUseHighPrecisionShader = options.shouldUseHighPrecisionShader;\n if (source.indexOf(\"precision highp float\") === -1) {\n if (!shouldUseHighPrecisionShader) {\n source = \"precision mediump float;\\n\" + source;\n }\n else {\n source = \"precision highp float;\\n\" + source;\n }\n }\n else {\n if (!shouldUseHighPrecisionShader) {\n // Moving highp to mediump\n source = source.replace(\"precision highp float\", \"precision mediump float\");\n }\n }\n return source;\n }\n static _ExtractOperation(expression) {\n const regex = /defined\\((.+)\\)/;\n const match = regex.exec(expression);\n if (match && match.length) {\n return new ShaderDefineIsDefinedOperator(match[1].trim(), expression[0] === \"!\");\n }\n const operators = [\"==\", \"!=\", \">=\", \"<=\", \"<\", \">\"];\n let operator = \"\";\n let indexOperator = 0;\n for (operator of operators) {\n indexOperator = expression.indexOf(operator);\n if (indexOperator > -1) {\n break;\n }\n }\n if (indexOperator === -1) {\n return new ShaderDefineIsDefinedOperator(expression);\n }\n const define = expression.substring(0, indexOperator).trim();\n const value = expression.substring(indexOperator + operator.length).trim();\n return new ShaderDefineArithmeticOperator(define, operator, value);\n }\n static _BuildSubExpression(expression) {\n expression = expression.replace(regexSE, \"defined[$1]\");\n const postfix = ShaderDefineExpression.infixToPostfix(expression);\n const stack = [];\n for (const c of postfix) {\n if (c !== \"||\" && c !== \"&&\") {\n stack.push(c);\n }\n else if (stack.length >= 2) {\n let v1 = stack[stack.length - 1], v2 = stack[stack.length - 2];\n stack.length -= 2;\n const operator = c == \"&&\" ? new ShaderDefineAndOperator() : new ShaderDefineOrOperator();\n if (typeof v1 === \"string\") {\n v1 = v1.replace(regexSERevert, \"defined($1)\");\n }\n if (typeof v2 === \"string\") {\n v2 = v2.replace(regexSERevert, \"defined($1)\");\n }\n operator.leftOperand = typeof v2 === \"string\" ? this._ExtractOperation(v2) : v2;\n operator.rightOperand = typeof v1 === \"string\" ? this._ExtractOperation(v1) : v1;\n stack.push(operator);\n }\n }\n let result = stack[stack.length - 1];\n if (typeof result === \"string\") {\n result = result.replace(regexSERevert, \"defined($1)\");\n }\n // note: stack.length !== 1 if there was an error in the parsing\n return typeof result === \"string\" ? this._ExtractOperation(result) : result;\n }\n static _BuildExpression(line, start) {\n const node = new ShaderCodeTestNode();\n const command = line.substring(0, start);\n let expression = line.substring(start);\n expression = expression.substring(0, (expression.indexOf(\"//\") + 1 || expression.length + 1) - 1).trim();\n if (command === \"#ifdef\") {\n node.testExpression = new ShaderDefineIsDefinedOperator(expression);\n }\n else if (command === \"#ifndef\") {\n node.testExpression = new ShaderDefineIsDefinedOperator(expression, true);\n }\n else {\n node.testExpression = this._BuildSubExpression(expression);\n }\n return node;\n }\n static _MoveCursorWithinIf(cursor, rootNode, ifNode) {\n let line = cursor.currentLine;\n while (this._MoveCursor(cursor, ifNode)) {\n line = cursor.currentLine;\n const first5 = line.substring(0, 5).toLowerCase();\n if (first5 === \"#else\") {\n const elseNode = new ShaderCodeNode();\n rootNode.children.push(elseNode);\n this._MoveCursor(cursor, elseNode);\n return;\n }\n else if (first5 === \"#elif\") {\n const elifNode = this._BuildExpression(line, 5);\n rootNode.children.push(elifNode);\n ifNode = elifNode;\n }\n }\n }\n static _MoveCursor(cursor, rootNode) {\n while (cursor.canRead) {\n cursor.lineIndex++;\n const line = cursor.currentLine;\n if (line.indexOf(\"#\") >= 0) {\n const matches = ShaderProcessor._MoveCursorRegex.exec(line);\n if (matches && matches.length) {\n const keyword = matches[0];\n switch (keyword) {\n case \"#ifdef\": {\n const newRootNode = new ShaderCodeConditionNode();\n rootNode.children.push(newRootNode);\n const ifNode = this._BuildExpression(line, 6);\n newRootNode.children.push(ifNode);\n this._MoveCursorWithinIf(cursor, newRootNode, ifNode);\n break;\n }\n case \"#else\":\n case \"#elif\":\n return true;\n case \"#endif\":\n return false;\n case \"#ifndef\": {\n const newRootNode = new ShaderCodeConditionNode();\n rootNode.children.push(newRootNode);\n const ifNode = this._BuildExpression(line, 7);\n newRootNode.children.push(ifNode);\n this._MoveCursorWithinIf(cursor, newRootNode, ifNode);\n break;\n }\n case \"#if\": {\n const newRootNode = new ShaderCodeConditionNode();\n const ifNode = this._BuildExpression(line, 3);\n rootNode.children.push(newRootNode);\n newRootNode.children.push(ifNode);\n this._MoveCursorWithinIf(cursor, newRootNode, ifNode);\n break;\n }\n }\n continue;\n }\n }\n const newNode = new ShaderCodeNode();\n newNode.line = line;\n rootNode.children.push(newNode);\n // Detect additional defines\n if (line[0] === \"#\" && line[1] === \"d\") {\n const split = line.replace(\";\", \"\").split(\" \");\n newNode.additionalDefineKey = split[1];\n if (split.length === 3) {\n newNode.additionalDefineValue = split[2];\n }\n }\n }\n return false;\n }\n static _EvaluatePreProcessors(sourceCode, preprocessors, options) {\n const rootNode = new ShaderCodeNode();\n const cursor = new ShaderCodeCursor();\n cursor.lineIndex = -1;\n cursor.lines = sourceCode.split(\"\\n\");\n // Decompose (We keep it in 2 steps so it is easier to maintain and perf hit is insignificant)\n this._MoveCursor(cursor, rootNode);\n // Recompose\n return rootNode.process(preprocessors, options);\n }\n static _PreparePreProcessors(options, engine) {\n var _a;\n const defines = options.defines;\n const preprocessors = {};\n for (const define of defines) {\n const keyValue = define.replace(\"#define\", \"\").replace(\";\", \"\").trim();\n const split = keyValue.split(\" \");\n preprocessors[split[0]] = split.length > 1 ? split[1] : \"\";\n }\n if (((_a = options.processor) === null || _a === void 0 ? void 0 : _a.shaderLanguage) === ShaderLanguage.GLSL) {\n preprocessors[\"GL_ES\"] = \"true\";\n }\n preprocessors[\"__VERSION__\"] = options.version;\n preprocessors[options.platformName] = \"true\";\n engine._getGlobalDefines(preprocessors);\n return preprocessors;\n }\n static _ProcessShaderConversion(sourceCode, options, engine) {\n let preparedSourceCode = this._ProcessPrecision(sourceCode, options);\n if (!options.processor) {\n return preparedSourceCode;\n }\n // Already converted\n if (options.processor.shaderLanguage === ShaderLanguage.GLSL && preparedSourceCode.indexOf(\"#version 3\") !== -1) {\n preparedSourceCode = preparedSourceCode.replace(\"#version 300 es\", \"\");\n if (!options.processor.parseGLES3) {\n return preparedSourceCode;\n }\n }\n const defines = options.defines;\n const preprocessors = this._PreparePreProcessors(options, engine);\n // General pre processing\n if (options.processor.preProcessor) {\n preparedSourceCode = options.processor.preProcessor(preparedSourceCode, defines, options.isFragment, options.processingContext);\n }\n preparedSourceCode = this._EvaluatePreProcessors(preparedSourceCode, preprocessors, options);\n // Post processing\n if (options.processor.postProcessor) {\n preparedSourceCode = options.processor.postProcessor(preparedSourceCode, defines, options.isFragment, options.processingContext, engine);\n }\n // Inline functions tagged with #define inline\n if (engine._features.needShaderCodeInlining) {\n preparedSourceCode = engine.inlineShaderCode(preparedSourceCode);\n }\n return preparedSourceCode;\n }\n static _ApplyPreProcessing(sourceCode, options, engine) {\n var _a, _b;\n let preparedSourceCode = sourceCode;\n const defines = options.defines;\n const preprocessors = this._PreparePreProcessors(options, engine);\n // General pre processing\n if ((_a = options.processor) === null || _a === void 0 ? void 0 : _a.preProcessor) {\n preparedSourceCode = options.processor.preProcessor(preparedSourceCode, defines, options.isFragment, options.processingContext);\n }\n preparedSourceCode = this._EvaluatePreProcessors(preparedSourceCode, preprocessors, options);\n // Post processing\n if ((_b = options.processor) === null || _b === void 0 ? void 0 : _b.postProcessor) {\n preparedSourceCode = options.processor.postProcessor(preparedSourceCode, defines, options.isFragment, options.processingContext, engine);\n }\n // Inline functions tagged with #define inline\n if (engine._features.needShaderCodeInlining) {\n preparedSourceCode = engine.inlineShaderCode(preparedSourceCode);\n }\n return preparedSourceCode;\n }\n /** @internal */\n static _ProcessIncludes(sourceCode, options, callback) {\n reusableMatches.length = 0;\n let match;\n // stay back-compat to the old matchAll syntax\n while ((match = regexShaderInclude.exec(sourceCode)) !== null) {\n reusableMatches.push(match);\n }\n let returnValue = String(sourceCode);\n let parts = [sourceCode];\n let keepProcessing = false;\n for (const match of reusableMatches) {\n let includeFile = match[1];\n // Uniform declaration\n if (includeFile.indexOf(\"__decl__\") !== -1) {\n includeFile = includeFile.replace(regexShaderDecl, \"\");\n if (options.supportsUniformBuffers) {\n includeFile = includeFile.replace(\"Vertex\", \"Ubo\").replace(\"Fragment\", \"Ubo\");\n }\n includeFile = includeFile + \"Declaration\";\n }\n if (options.includesShadersStore[includeFile]) {\n // Substitution\n let includeContent = options.includesShadersStore[includeFile];\n if (match[2]) {\n const splits = match[3].split(\",\");\n for (let index = 0; index < splits.length; index += 2) {\n const source = new RegExp(splits[index], \"g\");\n const dest = splits[index + 1];\n includeContent = includeContent.replace(source, dest);\n }\n }\n if (match[4]) {\n const indexString = match[5];\n if (indexString.indexOf(\"..\") !== -1) {\n const indexSplits = indexString.split(\"..\");\n const minIndex = parseInt(indexSplits[0]);\n let maxIndex = parseInt(indexSplits[1]);\n let sourceIncludeContent = includeContent.slice(0);\n includeContent = \"\";\n if (isNaN(maxIndex)) {\n maxIndex = options.indexParameters[indexSplits[1]];\n }\n for (let i = minIndex; i < maxIndex; i++) {\n if (!options.supportsUniformBuffers) {\n // Ubo replacement\n sourceIncludeContent = sourceIncludeContent.replace(regexLightX, (str, p1) => {\n return p1 + \"{X}\";\n });\n }\n includeContent += sourceIncludeContent.replace(regexX, i.toString()) + \"\\n\";\n }\n }\n else {\n if (!options.supportsUniformBuffers) {\n // Ubo replacement\n includeContent = includeContent.replace(regexLightX, (str, p1) => {\n return p1 + \"{X}\";\n });\n }\n includeContent = includeContent.replace(regexX, indexString);\n }\n }\n // Replace\n // Split all parts on match[0] and intersperse the parts with the include content\n const newParts = [];\n for (const part of parts) {\n const splitPart = part.split(match[0]);\n for (let i = 0; i < splitPart.length - 1; i++) {\n newParts.push(splitPart[i]);\n newParts.push(includeContent);\n }\n newParts.push(splitPart[splitPart.length - 1]);\n }\n parts = newParts;\n keepProcessing = keepProcessing || includeContent.indexOf(\"#include<\") >= 0 || includeContent.indexOf(\"#include <\") >= 0;\n }\n else {\n const includeShaderUrl = options.shadersRepository + \"ShadersInclude/\" + includeFile + \".fx\";\n ShaderProcessor._FileToolsLoadFile(includeShaderUrl, (fileContent) => {\n options.includesShadersStore[includeFile] = fileContent;\n this._ProcessIncludes(parts.join(\"\"), options, callback);\n });\n return;\n }\n }\n reusableMatches.length = 0;\n returnValue = parts.join(\"\");\n if (keepProcessing) {\n this._ProcessIncludes(returnValue.toString(), options, callback);\n }\n else {\n callback(returnValue);\n }\n }\n /**\n * Loads a file from a url\n * @param url url to load\n * @param onSuccess callback called when the file successfully loads\n * @param onProgress callback called while file is loading (if the server supports this mode)\n * @param offlineProvider defines the offline provider for caching\n * @param useArrayBuffer defines a boolean indicating that date must be returned as ArrayBuffer\n * @param onError callback called when the file fails to load\n * @returns a file request object\n * @internal\n */\n static _FileToolsLoadFile(url, onSuccess, onProgress, offlineProvider, useArrayBuffer, onError) {\n throw _WarnImport(\"FileTools\");\n }\n }\n ShaderProcessor._MoveCursorRegex = /(#ifdef)|(#else)|(#elif)|(#endif)|(#ifndef)|(#if)/;\n\n /**\n * Defines the shader related stores and directory\n */\n class ShaderStore {\n /**\n * Gets the shaders repository path for a given shader language\n * @param shaderLanguage the shader language\n * @returns the path to the shaders repository\n */\n static GetShadersRepository(shaderLanguage = ShaderLanguage.GLSL) {\n return shaderLanguage === ShaderLanguage.GLSL ? ShaderStore.ShadersRepository : ShaderStore.ShadersRepositoryWGSL;\n }\n /**\n * Gets the shaders store of a given shader language\n * @param shaderLanguage the shader language\n * @returns the shaders store\n */\n static GetShadersStore(shaderLanguage = ShaderLanguage.GLSL) {\n return shaderLanguage === ShaderLanguage.GLSL ? ShaderStore.ShadersStore : ShaderStore.ShadersStoreWGSL;\n }\n /**\n * Gets the include shaders store of a given shader language\n * @param shaderLanguage the shader language\n * @returns the include shaders store\n */\n static GetIncludesShadersStore(shaderLanguage = ShaderLanguage.GLSL) {\n return shaderLanguage === ShaderLanguage.GLSL ? ShaderStore.IncludesShadersStore : ShaderStore.IncludesShadersStoreWGSL;\n }\n }\n /**\n * Gets or sets the relative url used to load shaders if using the engine in non-minified mode\n */\n ShaderStore.ShadersRepository = \"src/Shaders/\";\n /**\n * Store of each shader (The can be looked up using effect.key)\n */\n ShaderStore.ShadersStore = {};\n /**\n * Store of each included file for a shader (The can be looked up using effect.key)\n */\n ShaderStore.IncludesShadersStore = {};\n /**\n * Gets or sets the relative url used to load shaders (WGSL) if using the engine in non-minified mode\n */\n ShaderStore.ShadersRepositoryWGSL = \"src/ShadersWGSL/\";\n /**\n * Store of each shader (WGSL)\n */\n ShaderStore.ShadersStoreWGSL = {};\n /**\n * Store of each included file for a shader (WGSL)\n */\n ShaderStore.IncludesShadersStoreWGSL = {};\n\n /**\n * Effect containing vertex and fragment shader that can be executed on an object.\n */\n class Effect {\n /**\n * Gets or sets the relative url used to load shaders if using the engine in non-minified mode\n */\n static get ShadersRepository() {\n return ShaderStore.ShadersRepository;\n }\n static set ShadersRepository(repo) {\n ShaderStore.ShadersRepository = repo;\n }\n /**\n * Observable that will be called when effect is bound.\n */\n get onBindObservable() {\n if (!this._onBindObservable) {\n this._onBindObservable = new Observable$1();\n }\n return this._onBindObservable;\n }\n /**\n * Instantiates an effect.\n * An effect can be used to create/manage/execute vertex and fragment shaders.\n * @param baseName Name of the effect.\n * @param attributesNamesOrOptions List of attribute names that will be passed to the shader or set of all options to create the effect.\n * @param uniformsNamesOrEngine List of uniform variable names that will be passed to the shader or the engine that will be used to render effect.\n * @param samplers List of sampler variables that will be passed to the shader.\n * @param engine Engine to be used to render the effect\n * @param defines Define statements to be added to the shader.\n * @param fallbacks Possible fallbacks for this effect to improve performance when needed.\n * @param onCompiled Callback that will be called when the shader is compiled.\n * @param onError Callback that will be called if an error occurs during shader compilation.\n * @param indexParameters Parameters to be used with Babylons include syntax to iterate over an array (eg. {lights: 10})\n * @param key Effect Key identifying uniquely compiled shader variants\n * @param shaderLanguage the language the shader is written in (default: GLSL)\n */\n constructor(baseName, attributesNamesOrOptions, uniformsNamesOrEngine, samplers = null, engine, defines = null, fallbacks = null, onCompiled = null, onError = null, indexParameters, key = \"\", shaderLanguage = ShaderLanguage.GLSL) {\n var _a, _b, _c;\n /**\n * Name of the effect.\n */\n this.name = null;\n /**\n * String container all the define statements that should be set on the shader.\n */\n this.defines = \"\";\n /**\n * Callback that will be called when the shader is compiled.\n */\n this.onCompiled = null;\n /**\n * Callback that will be called if an error occurs during shader compilation.\n */\n this.onError = null;\n /**\n * Callback that will be called when effect is bound.\n */\n this.onBind = null;\n /**\n * Unique ID of the effect.\n */\n this.uniqueId = 0;\n /**\n * Observable that will be called when the shader is compiled.\n * It is recommended to use executeWhenCompile() or to make sure that scene.isReady() is called to get this observable raised.\n */\n this.onCompileObservable = new Observable$1();\n /**\n * Observable that will be called if an error occurs during shader compilation.\n */\n this.onErrorObservable = new Observable$1();\n /** @internal */\n this._onBindObservable = null;\n /**\n * @internal\n * Specifies if the effect was previously ready\n */\n this._wasPreviouslyReady = false;\n /**\n * @internal\n * Forces the code from bindForSubMesh to be fully run the next time it is called\n * It is used in frozen mode to make sure the effect is properly rebound when a new effect is created\n */\n this._forceRebindOnNextCall = false;\n /**\n * @internal\n * Specifies if the effect was previously using instances\n */\n this._wasPreviouslyUsingInstances = null;\n this._isDisposed = false;\n /** @internal */\n this._bonesComputationForcedToCPU = false;\n /** @internal */\n this._uniformBuffersNames = {};\n /** @internal */\n this._multiTarget = false;\n this._samplers = {};\n this._isReady = false;\n this._compilationError = \"\";\n this._allFallbacksProcessed = false;\n this._uniforms = {};\n /**\n * Key for the effect.\n * @internal\n */\n this._key = \"\";\n this._fallbacks = null;\n this._vertexSourceCodeOverride = \"\";\n this._fragmentSourceCodeOverride = \"\";\n this._transformFeedbackVaryings = null;\n /**\n * Compiled shader to webGL program.\n * @internal\n */\n this._pipelineContext = null;\n /** @internal */\n this._vertexSourceCode = \"\";\n /** @internal */\n this._fragmentSourceCode = \"\";\n /** @internal */\n this._vertexSourceCodeBeforeMigration = \"\";\n /** @internal */\n this._fragmentSourceCodeBeforeMigration = \"\";\n /** @internal */\n this._rawVertexSourceCode = \"\";\n /** @internal */\n this._rawFragmentSourceCode = \"\";\n this.name = baseName;\n this._key = key;\n let processCodeAfterIncludes = undefined;\n let processFinalCode = null;\n if (attributesNamesOrOptions.attributes) {\n const options = attributesNamesOrOptions;\n this._engine = uniformsNamesOrEngine;\n this._attributesNames = options.attributes;\n this._uniformsNames = options.uniformsNames.concat(options.samplers);\n this._samplerList = options.samplers.slice();\n this.defines = options.defines;\n this.onError = options.onError;\n this.onCompiled = options.onCompiled;\n this._fallbacks = options.fallbacks;\n this._indexParameters = options.indexParameters;\n this._transformFeedbackVaryings = options.transformFeedbackVaryings || null;\n this._multiTarget = !!options.multiTarget;\n this._shaderLanguage = (_a = options.shaderLanguage) !== null && _a !== void 0 ? _a : ShaderLanguage.GLSL;\n if (options.uniformBuffersNames) {\n this._uniformBuffersNamesList = options.uniformBuffersNames.slice();\n for (let i = 0; i < options.uniformBuffersNames.length; i++) {\n this._uniformBuffersNames[options.uniformBuffersNames[i]] = i;\n }\n }\n processFinalCode = (_b = options.processFinalCode) !== null && _b !== void 0 ? _b : null;\n processCodeAfterIncludes = (_c = options.processCodeAfterIncludes) !== null && _c !== void 0 ? _c : undefined;\n }\n else {\n this._engine = engine;\n this.defines = defines == null ? \"\" : defines;\n this._uniformsNames = uniformsNamesOrEngine.concat(samplers);\n this._samplerList = samplers ? samplers.slice() : [];\n this._attributesNames = attributesNamesOrOptions;\n this._uniformBuffersNamesList = [];\n this._shaderLanguage = shaderLanguage;\n this.onError = onError;\n this.onCompiled = onCompiled;\n this._indexParameters = indexParameters;\n this._fallbacks = fallbacks;\n }\n this._attributeLocationByName = {};\n this.uniqueId = Effect._UniqueIdSeed++;\n let vertexSource;\n let fragmentSource;\n const hostDocument = IsWindowObjectExist() ? this._engine.getHostDocument() : null;\n if (baseName.vertexSource) {\n vertexSource = \"source:\" + baseName.vertexSource;\n }\n else if (baseName.vertexElement) {\n vertexSource = hostDocument ? hostDocument.getElementById(baseName.vertexElement) : null;\n if (!vertexSource) {\n vertexSource = baseName.vertexElement;\n }\n }\n else {\n vertexSource = baseName.vertex || baseName;\n }\n if (baseName.fragmentSource) {\n fragmentSource = \"source:\" + baseName.fragmentSource;\n }\n else if (baseName.fragmentElement) {\n fragmentSource = hostDocument ? hostDocument.getElementById(baseName.fragmentElement) : null;\n if (!fragmentSource) {\n fragmentSource = baseName.fragmentElement;\n }\n }\n else {\n fragmentSource = baseName.fragment || baseName;\n }\n this._processingContext = this._engine._getShaderProcessingContext(this._shaderLanguage);\n let processorOptions = {\n defines: this.defines.split(\"\\n\"),\n indexParameters: this._indexParameters,\n isFragment: false,\n shouldUseHighPrecisionShader: this._engine._shouldUseHighPrecisionShader,\n processor: this._engine._getShaderProcessor(this._shaderLanguage),\n supportsUniformBuffers: this._engine.supportsUniformBuffers,\n shadersRepository: ShaderStore.GetShadersRepository(this._shaderLanguage),\n includesShadersStore: ShaderStore.GetIncludesShadersStore(this._shaderLanguage),\n version: (this._engine.version * 100).toString(),\n platformName: this._engine.shaderPlatformName,\n processingContext: this._processingContext,\n isNDCHalfZRange: this._engine.isNDCHalfZRange,\n useReverseDepthBuffer: this._engine.useReverseDepthBuffer,\n processCodeAfterIncludes,\n };\n const shaderCodes = [undefined, undefined];\n const shadersLoaded = () => {\n if (shaderCodes[0] && shaderCodes[1]) {\n processorOptions.isFragment = true;\n const [migratedVertexCode, fragmentCode] = shaderCodes;\n ShaderProcessor.Process(fragmentCode, processorOptions, (migratedFragmentCode, codeBeforeMigration) => {\n this._fragmentSourceCodeBeforeMigration = codeBeforeMigration;\n if (processFinalCode) {\n migratedFragmentCode = processFinalCode(\"fragment\", migratedFragmentCode);\n }\n const finalShaders = ShaderProcessor.Finalize(migratedVertexCode, migratedFragmentCode, processorOptions);\n processorOptions = null;\n this._useFinalCode(finalShaders.vertexCode, finalShaders.fragmentCode, baseName);\n }, this._engine);\n }\n };\n this._loadShader(vertexSource, \"Vertex\", \"\", (vertexCode) => {\n ShaderProcessor.Initialize(processorOptions);\n ShaderProcessor.Process(vertexCode, processorOptions, (migratedVertexCode, codeBeforeMigration) => {\n this._rawVertexSourceCode = vertexCode;\n this._vertexSourceCodeBeforeMigration = codeBeforeMigration;\n if (processFinalCode) {\n migratedVertexCode = processFinalCode(\"vertex\", migratedVertexCode);\n }\n shaderCodes[0] = migratedVertexCode;\n shadersLoaded();\n }, this._engine);\n });\n this._loadShader(fragmentSource, \"Fragment\", \"Pixel\", (fragmentCode) => {\n this._rawFragmentSourceCode = fragmentCode;\n shaderCodes[1] = fragmentCode;\n shadersLoaded();\n });\n }\n _useFinalCode(migratedVertexCode, migratedFragmentCode, baseName) {\n if (baseName) {\n const vertex = baseName.vertexElement || baseName.vertex || baseName.spectorName || baseName;\n const fragment = baseName.fragmentElement || baseName.fragment || baseName.spectorName || baseName;\n this._vertexSourceCode = (this._shaderLanguage === ShaderLanguage.WGSL ? \"//\" : \"\") + \"#define SHADER_NAME vertex:\" + vertex + \"\\n\" + migratedVertexCode;\n this._fragmentSourceCode = (this._shaderLanguage === ShaderLanguage.WGSL ? \"//\" : \"\") + \"#define SHADER_NAME fragment:\" + fragment + \"\\n\" + migratedFragmentCode;\n }\n else {\n this._vertexSourceCode = migratedVertexCode;\n this._fragmentSourceCode = migratedFragmentCode;\n }\n this._prepareEffect();\n }\n /**\n * Unique key for this effect\n */\n get key() {\n return this._key;\n }\n /**\n * If the effect has been compiled and prepared.\n * @returns if the effect is compiled and prepared.\n */\n isReady() {\n try {\n return this._isReadyInternal();\n }\n catch (_a) {\n return false;\n }\n }\n _isReadyInternal() {\n if (this._isReady) {\n return true;\n }\n if (this._pipelineContext) {\n return this._pipelineContext.isReady;\n }\n return false;\n }\n /**\n * The engine the effect was initialized with.\n * @returns the engine.\n */\n getEngine() {\n return this._engine;\n }\n /**\n * The pipeline context for this effect\n * @returns the associated pipeline context\n */\n getPipelineContext() {\n return this._pipelineContext;\n }\n /**\n * The set of names of attribute variables for the shader.\n * @returns An array of attribute names.\n */\n getAttributesNames() {\n return this._attributesNames;\n }\n /**\n * Returns the attribute at the given index.\n * @param index The index of the attribute.\n * @returns The location of the attribute.\n */\n getAttributeLocation(index) {\n return this._attributes[index];\n }\n /**\n * Returns the attribute based on the name of the variable.\n * @param name of the attribute to look up.\n * @returns the attribute location.\n */\n getAttributeLocationByName(name) {\n return this._attributeLocationByName[name];\n }\n /**\n * The number of attributes.\n * @returns the number of attributes.\n */\n getAttributesCount() {\n return this._attributes.length;\n }\n /**\n * Gets the index of a uniform variable.\n * @param uniformName of the uniform to look up.\n * @returns the index.\n */\n getUniformIndex(uniformName) {\n return this._uniformsNames.indexOf(uniformName);\n }\n /**\n * Returns the attribute based on the name of the variable.\n * @param uniformName of the uniform to look up.\n * @returns the location of the uniform.\n */\n getUniform(uniformName) {\n return this._uniforms[uniformName];\n }\n /**\n * Returns an array of sampler variable names\n * @returns The array of sampler variable names.\n */\n getSamplers() {\n return this._samplerList;\n }\n /**\n * Returns an array of uniform variable names\n * @returns The array of uniform variable names.\n */\n getUniformNames() {\n return this._uniformsNames;\n }\n /**\n * Returns an array of uniform buffer variable names\n * @returns The array of uniform buffer variable names.\n */\n getUniformBuffersNames() {\n return this._uniformBuffersNamesList;\n }\n /**\n * Returns the index parameters used to create the effect\n * @returns The index parameters object\n */\n getIndexParameters() {\n return this._indexParameters;\n }\n /**\n * The error from the last compilation.\n * @returns the error string.\n */\n getCompilationError() {\n return this._compilationError;\n }\n /**\n * Gets a boolean indicating that all fallbacks were used during compilation\n * @returns true if all fallbacks were used\n */\n allFallbacksProcessed() {\n return this._allFallbacksProcessed;\n }\n /**\n * Adds a callback to the onCompiled observable and call the callback immediately if already ready.\n * @param func The callback to be used.\n */\n executeWhenCompiled(func) {\n if (this.isReady()) {\n func(this);\n return;\n }\n this.onCompileObservable.add((effect) => {\n func(effect);\n });\n if (!this._pipelineContext || this._pipelineContext.isAsync) {\n setTimeout(() => {\n this._checkIsReady(null);\n }, 16);\n }\n }\n _checkIsReady(previousPipelineContext) {\n try {\n if (this._isReadyInternal()) {\n return;\n }\n }\n catch (e) {\n this._processCompilationErrors(e, previousPipelineContext);\n return;\n }\n if (this._isDisposed) {\n return;\n }\n setTimeout(() => {\n this._checkIsReady(previousPipelineContext);\n }, 16);\n }\n _loadShader(shader, key, optionalKey, callback) {\n if (typeof HTMLElement !== \"undefined\") {\n // DOM element ?\n if (shader instanceof HTMLElement) {\n const shaderCode = GetDOMTextContent(shader);\n callback(shaderCode);\n return;\n }\n }\n // Direct source ?\n if (shader.substr(0, 7) === \"source:\") {\n callback(shader.substr(7));\n return;\n }\n // Base64 encoded ?\n if (shader.substr(0, 7) === \"base64:\") {\n const shaderBinary = window.atob(shader.substr(7));\n callback(shaderBinary);\n return;\n }\n const shaderStore = ShaderStore.GetShadersStore(this._shaderLanguage);\n // Is in local store ?\n if (shaderStore[shader + key + \"Shader\"]) {\n callback(shaderStore[shader + key + \"Shader\"]);\n return;\n }\n if (optionalKey && shaderStore[shader + optionalKey + \"Shader\"]) {\n callback(shaderStore[shader + optionalKey + \"Shader\"]);\n return;\n }\n let shaderUrl;\n if (shader[0] === \".\" || shader[0] === \"/\" || shader.indexOf(\"http\") > -1) {\n shaderUrl = shader;\n }\n else {\n shaderUrl = ShaderStore.GetShadersRepository(this._shaderLanguage) + shader;\n }\n // Vertex shader\n this._engine._loadFile(shaderUrl + \".\" + key.toLowerCase() + \".fx\", callback);\n }\n /**\n * Gets the vertex shader source code of this effect\n * This is the final source code that will be compiled, after all the processing has been done (pre-processing applied, code injection/replacement, etc)\n */\n get vertexSourceCode() {\n var _a, _b;\n return this._vertexSourceCodeOverride && this._fragmentSourceCodeOverride\n ? this._vertexSourceCodeOverride\n : (_b = (_a = this._pipelineContext) === null || _a === void 0 ? void 0 : _a._getVertexShaderCode()) !== null && _b !== void 0 ? _b : this._vertexSourceCode;\n }\n /**\n * Gets the fragment shader source code of this effect\n * This is the final source code that will be compiled, after all the processing has been done (pre-processing applied, code injection/replacement, etc)\n */\n get fragmentSourceCode() {\n var _a, _b;\n return this._vertexSourceCodeOverride && this._fragmentSourceCodeOverride\n ? this._fragmentSourceCodeOverride\n : (_b = (_a = this._pipelineContext) === null || _a === void 0 ? void 0 : _a._getFragmentShaderCode()) !== null && _b !== void 0 ? _b : this._fragmentSourceCode;\n }\n /**\n * Gets the vertex shader source code before migration.\n * This is the source code after the include directives have been replaced by their contents but before the code is migrated, i.e. before ShaderProcess._ProcessShaderConversion is executed.\n * This method is, among other things, responsible for parsing #if/#define directives as well as converting GLES2 syntax to GLES3 (in the case of WebGL).\n */\n get vertexSourceCodeBeforeMigration() {\n return this._vertexSourceCodeBeforeMigration;\n }\n /**\n * Gets the fragment shader source code before migration.\n * This is the source code after the include directives have been replaced by their contents but before the code is migrated, i.e. before ShaderProcess._ProcessShaderConversion is executed.\n * This method is, among other things, responsible for parsing #if/#define directives as well as converting GLES2 syntax to GLES3 (in the case of WebGL).\n */\n get fragmentSourceCodeBeforeMigration() {\n return this._fragmentSourceCodeBeforeMigration;\n }\n /**\n * Gets the vertex shader source code before it has been modified by any processing\n */\n get rawVertexSourceCode() {\n return this._rawVertexSourceCode;\n }\n /**\n * Gets the fragment shader source code before it has been modified by any processing\n */\n get rawFragmentSourceCode() {\n return this._rawFragmentSourceCode;\n }\n /**\n * Recompiles the webGL program\n * @param vertexSourceCode The source code for the vertex shader.\n * @param fragmentSourceCode The source code for the fragment shader.\n * @param onCompiled Callback called when completed.\n * @param onError Callback called on error.\n * @internal\n */\n _rebuildProgram(vertexSourceCode, fragmentSourceCode, onCompiled, onError) {\n this._isReady = false;\n this._vertexSourceCodeOverride = vertexSourceCode;\n this._fragmentSourceCodeOverride = fragmentSourceCode;\n this.onError = (effect, error) => {\n if (onError) {\n onError(error);\n }\n };\n this.onCompiled = () => {\n const scenes = this.getEngine().scenes;\n if (scenes) {\n for (let i = 0; i < scenes.length; i++) {\n scenes[i].markAllMaterialsAsDirty(63);\n }\n }\n this._pipelineContext._handlesSpectorRebuildCallback(onCompiled);\n };\n this._fallbacks = null;\n this._prepareEffect();\n }\n /**\n * Prepares the effect\n * @internal\n */\n _prepareEffect() {\n const attributesNames = this._attributesNames;\n const defines = this.defines;\n const previousPipelineContext = this._pipelineContext;\n this._isReady = false;\n try {\n const engine = this._engine;\n this._pipelineContext = engine.createPipelineContext(this._processingContext);\n this._pipelineContext._name = this._key;\n const rebuildRebind = this._rebuildProgram.bind(this);\n if (this._vertexSourceCodeOverride && this._fragmentSourceCodeOverride) {\n engine._preparePipelineContext(this._pipelineContext, this._vertexSourceCodeOverride, this._fragmentSourceCodeOverride, true, this._rawVertexSourceCode, this._rawFragmentSourceCode, rebuildRebind, null, this._transformFeedbackVaryings, this._key);\n }\n else {\n engine._preparePipelineContext(this._pipelineContext, this._vertexSourceCode, this._fragmentSourceCode, false, this._rawVertexSourceCode, this._rawFragmentSourceCode, rebuildRebind, defines, this._transformFeedbackVaryings, this._key);\n }\n engine._executeWhenRenderingStateIsCompiled(this._pipelineContext, () => {\n this._attributes = [];\n this._pipelineContext._fillEffectInformation(this, this._uniformBuffersNames, this._uniformsNames, this._uniforms, this._samplerList, this._samplers, attributesNames, this._attributes);\n // Caches attribute locations.\n if (attributesNames) {\n for (let i = 0; i < attributesNames.length; i++) {\n const name = attributesNames[i];\n this._attributeLocationByName[name] = this._attributes[i];\n }\n }\n engine.bindSamplers(this);\n this._compilationError = \"\";\n this._isReady = true;\n if (this.onCompiled) {\n this.onCompiled(this);\n }\n this.onCompileObservable.notifyObservers(this);\n this.onCompileObservable.clear();\n // Unbind mesh reference in fallbacks\n if (this._fallbacks) {\n this._fallbacks.unBindMesh();\n }\n if (previousPipelineContext) {\n this.getEngine()._deletePipelineContext(previousPipelineContext);\n }\n });\n if (this._pipelineContext.isAsync) {\n this._checkIsReady(previousPipelineContext);\n }\n }\n catch (e) {\n this._processCompilationErrors(e, previousPipelineContext);\n }\n }\n _getShaderCodeAndErrorLine(code, error, isFragment) {\n const regexp = isFragment ? /FRAGMENT SHADER ERROR: 0:(\\d+?):/ : /VERTEX SHADER ERROR: 0:(\\d+?):/;\n let errorLine = null;\n if (error && code) {\n const res = error.match(regexp);\n if (res && res.length === 2) {\n const lineNumber = parseInt(res[1]);\n const lines = code.split(\"\\n\", -1);\n if (lines.length >= lineNumber) {\n errorLine = `Offending line [${lineNumber}] in ${isFragment ? \"fragment\" : \"vertex\"} code: ${lines[lineNumber - 1]}`;\n }\n }\n }\n return [code, errorLine];\n }\n _processCompilationErrors(e, previousPipelineContext = null) {\n var _a, _b, _c;\n this._compilationError = e.message;\n const attributesNames = this._attributesNames;\n const fallbacks = this._fallbacks;\n // Let's go through fallbacks then\n Logger.Error(\"Unable to compile effect:\");\n Logger.Error(\"Uniforms: \" +\n this._uniformsNames.map(function (uniform) {\n return \" \" + uniform;\n }));\n Logger.Error(\"Attributes: \" +\n attributesNames.map(function (attribute) {\n return \" \" + attribute;\n }));\n Logger.Error(\"Defines:\\n\" + this.defines);\n if (Effect.LogShaderCodeOnCompilationError) {\n let lineErrorVertex = null, lineErrorFragment = null, code = null;\n if ((_a = this._pipelineContext) === null || _a === void 0 ? void 0 : _a._getVertexShaderCode()) {\n [code, lineErrorVertex] = this._getShaderCodeAndErrorLine(this._pipelineContext._getVertexShaderCode(), this._compilationError, false);\n if (code) {\n Logger.Error(\"Vertex code:\");\n Logger.Error(code);\n }\n }\n if ((_b = this._pipelineContext) === null || _b === void 0 ? void 0 : _b._getFragmentShaderCode()) {\n [code, lineErrorFragment] = this._getShaderCodeAndErrorLine((_c = this._pipelineContext) === null || _c === void 0 ? void 0 : _c._getFragmentShaderCode(), this._compilationError, true);\n if (code) {\n Logger.Error(\"Fragment code:\");\n Logger.Error(code);\n }\n }\n if (lineErrorVertex) {\n Logger.Error(lineErrorVertex);\n }\n if (lineErrorFragment) {\n Logger.Error(lineErrorFragment);\n }\n }\n Logger.Error(\"Error: \" + this._compilationError);\n const notifyErrors = () => {\n if (this.onError) {\n this.onError(this, this._compilationError);\n }\n this.onErrorObservable.notifyObservers(this);\n };\n // In case a previous compilation was successful, we need to restore the previous pipeline context\n if (previousPipelineContext) {\n this._pipelineContext = previousPipelineContext;\n this._isReady = true;\n notifyErrors();\n }\n // Lets try to compile fallbacks as long as we have some.\n if (fallbacks) {\n this._pipelineContext = null;\n if (fallbacks.hasMoreFallbacks) {\n this._allFallbacksProcessed = false;\n Logger.Error(\"Trying next fallback.\");\n this.defines = fallbacks.reduce(this.defines, this);\n this._prepareEffect();\n }\n else {\n // Sorry we did everything we can\n this._allFallbacksProcessed = true;\n notifyErrors();\n this.onErrorObservable.clear();\n // Unbind mesh reference in fallbacks\n if (this._fallbacks) {\n this._fallbacks.unBindMesh();\n }\n }\n }\n else {\n this._allFallbacksProcessed = true;\n // In case of error, without any prior successful compilation, let s notify observers\n if (!previousPipelineContext) {\n notifyErrors();\n }\n }\n }\n /**\n * Checks if the effect is supported. (Must be called after compilation)\n */\n get isSupported() {\n return this._compilationError === \"\";\n }\n /**\n * Binds a texture to the engine to be used as output of the shader.\n * @param channel Name of the output variable.\n * @param texture Texture to bind.\n * @internal\n */\n _bindTexture(channel, texture) {\n this._engine._bindTexture(this._samplers[channel], texture, channel);\n }\n /**\n * Sets a texture on the engine to be used in the shader.\n * @param channel Name of the sampler variable.\n * @param texture Texture to set.\n */\n setTexture(channel, texture) {\n this._engine.setTexture(this._samplers[channel], this._uniforms[channel], texture, channel);\n }\n /**\n * Sets a depth stencil texture from a render target on the engine to be used in the shader.\n * @param channel Name of the sampler variable.\n * @param texture Texture to set.\n */\n setDepthStencilTexture(channel, texture) {\n this._engine.setDepthStencilTexture(this._samplers[channel], this._uniforms[channel], texture, channel);\n }\n /**\n * Sets an array of textures on the engine to be used in the shader.\n * @param channel Name of the variable.\n * @param textures Textures to set.\n */\n setTextureArray(channel, textures) {\n const exName = channel + \"Ex\";\n if (this._samplerList.indexOf(exName + \"0\") === -1) {\n const initialPos = this._samplerList.indexOf(channel);\n for (let index = 1; index < textures.length; index++) {\n const currentExName = exName + (index - 1).toString();\n this._samplerList.splice(initialPos + index, 0, currentExName);\n }\n // Reset every channels\n let channelIndex = 0;\n for (const key of this._samplerList) {\n this._samplers[key] = channelIndex;\n channelIndex += 1;\n }\n }\n this._engine.setTextureArray(this._samplers[channel], this._uniforms[channel], textures, channel);\n }\n /**\n * Sets a texture to be the input of the specified post process. (To use the output, pass in the next post process in the pipeline)\n * @param channel Name of the sampler variable.\n * @param postProcess Post process to get the input texture from.\n */\n setTextureFromPostProcess(channel, postProcess) {\n this._engine.setTextureFromPostProcess(this._samplers[channel], postProcess, channel);\n }\n /**\n * (Warning! setTextureFromPostProcessOutput may be desired instead)\n * Sets the input texture of the passed in post process to be input of this effect. (To use the output of the passed in post process use setTextureFromPostProcessOutput)\n * @param channel Name of the sampler variable.\n * @param postProcess Post process to get the output texture from.\n */\n setTextureFromPostProcessOutput(channel, postProcess) {\n this._engine.setTextureFromPostProcessOutput(this._samplers[channel], postProcess, channel);\n }\n /**\n * Binds a buffer to a uniform.\n * @param buffer Buffer to bind.\n * @param name Name of the uniform variable to bind to.\n */\n bindUniformBuffer(buffer, name) {\n const bufferName = this._uniformBuffersNames[name];\n if (bufferName === undefined || (Effect._BaseCache[bufferName] === buffer && this._engine._features.useUBOBindingCache)) {\n return;\n }\n Effect._BaseCache[bufferName] = buffer;\n this._engine.bindUniformBufferBase(buffer, bufferName, name);\n }\n /**\n * Binds block to a uniform.\n * @param blockName Name of the block to bind.\n * @param index Index to bind.\n */\n bindUniformBlock(blockName, index) {\n this._engine.bindUniformBlock(this._pipelineContext, blockName, index);\n }\n /**\n * Sets an integer value on a uniform variable.\n * @param uniformName Name of the variable.\n * @param value Value to be set.\n * @returns this effect.\n */\n setInt(uniformName, value) {\n this._pipelineContext.setInt(uniformName, value);\n return this;\n }\n /**\n * Sets an int2 value on a uniform variable.\n * @param uniformName Name of the variable.\n * @param x First int in int2.\n * @param y Second int in int2.\n * @returns this effect.\n */\n setInt2(uniformName, x, y) {\n this._pipelineContext.setInt2(uniformName, x, y);\n return this;\n }\n /**\n * Sets an int3 value on a uniform variable.\n * @param uniformName Name of the variable.\n * @param x First int in int3.\n * @param y Second int in int3.\n * @param z Third int in int3.\n * @returns this effect.\n */\n setInt3(uniformName, x, y, z) {\n this._pipelineContext.setInt3(uniformName, x, y, z);\n return this;\n }\n /**\n * Sets an int4 value on a uniform variable.\n * @param uniformName Name of the variable.\n * @param x First int in int4.\n * @param y Second int in int4.\n * @param z Third int in int4.\n * @param w Fourth int in int4.\n * @returns this effect.\n */\n setInt4(uniformName, x, y, z, w) {\n this._pipelineContext.setInt4(uniformName, x, y, z, w);\n return this;\n }\n /**\n * Sets an int array on a uniform variable.\n * @param uniformName Name of the variable.\n * @param array array to be set.\n * @returns this effect.\n */\n setIntArray(uniformName, array) {\n this._pipelineContext.setIntArray(uniformName, array);\n return this;\n }\n /**\n * Sets an int array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n * @returns this effect.\n */\n setIntArray2(uniformName, array) {\n this._pipelineContext.setIntArray2(uniformName, array);\n return this;\n }\n /**\n * Sets an int array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n * @returns this effect.\n */\n setIntArray3(uniformName, array) {\n this._pipelineContext.setIntArray3(uniformName, array);\n return this;\n }\n /**\n * Sets an int array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n * @returns this effect.\n */\n setIntArray4(uniformName, array) {\n this._pipelineContext.setIntArray4(uniformName, array);\n return this;\n }\n /**\n * Sets an unsigned integer value on a uniform variable.\n * @param uniformName Name of the variable.\n * @param value Value to be set.\n * @returns this effect.\n */\n setUInt(uniformName, value) {\n this._pipelineContext.setInt(uniformName, value);\n return this;\n }\n /**\n * Sets an unsigned int2 value on a uniform variable.\n * @param uniformName Name of the variable.\n * @param x First unsigned int in uint2.\n * @param y Second unsigned int in uint2.\n * @returns this effect.\n */\n setUInt2(uniformName, x, y) {\n this._pipelineContext.setInt2(uniformName, x, y);\n return this;\n }\n /**\n * Sets an unsigned int3 value on a uniform variable.\n * @param uniformName Name of the variable.\n * @param x First unsigned int in uint3.\n * @param y Second unsigned int in uint3.\n * @param z Third unsigned int in uint3.\n * @returns this effect.\n */\n setUInt3(uniformName, x, y, z) {\n this._pipelineContext.setInt3(uniformName, x, y, z);\n return this;\n }\n /**\n * Sets an unsigned int4 value on a uniform variable.\n * @param uniformName Name of the variable.\n * @param x First unsigned int in uint4.\n * @param y Second unsigned int in uint4.\n * @param z Third unsigned int in uint4.\n * @param w Fourth unsigned int in uint4.\n * @returns this effect.\n */\n setUInt4(uniformName, x, y, z, w) {\n this._pipelineContext.setInt4(uniformName, x, y, z, w);\n return this;\n }\n /**\n * Sets an unsigned int array on a uniform variable.\n * @param uniformName Name of the variable.\n * @param array array to be set.\n * @returns this effect.\n */\n setUIntArray(uniformName, array) {\n this._pipelineContext.setUIntArray(uniformName, array);\n return this;\n }\n /**\n * Sets an unsigned int array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n * @returns this effect.\n */\n setUIntArray2(uniformName, array) {\n this._pipelineContext.setUIntArray2(uniformName, array);\n return this;\n }\n /**\n * Sets an unsigned int array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n * @returns this effect.\n */\n setUIntArray3(uniformName, array) {\n this._pipelineContext.setUIntArray3(uniformName, array);\n return this;\n }\n /**\n * Sets an unsigned int array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n * @returns this effect.\n */\n setUIntArray4(uniformName, array) {\n this._pipelineContext.setUIntArray4(uniformName, array);\n return this;\n }\n /**\n * Sets an float array on a uniform variable.\n * @param uniformName Name of the variable.\n * @param array array to be set.\n * @returns this effect.\n */\n setFloatArray(uniformName, array) {\n this._pipelineContext.setArray(uniformName, array);\n return this;\n }\n /**\n * Sets an float array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n * @returns this effect.\n */\n setFloatArray2(uniformName, array) {\n this._pipelineContext.setArray2(uniformName, array);\n return this;\n }\n /**\n * Sets an float array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n * @returns this effect.\n */\n setFloatArray3(uniformName, array) {\n this._pipelineContext.setArray3(uniformName, array);\n return this;\n }\n /**\n * Sets an float array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n * @returns this effect.\n */\n setFloatArray4(uniformName, array) {\n this._pipelineContext.setArray4(uniformName, array);\n return this;\n }\n /**\n * Sets an array on a uniform variable.\n * @param uniformName Name of the variable.\n * @param array array to be set.\n * @returns this effect.\n */\n setArray(uniformName, array) {\n this._pipelineContext.setArray(uniformName, array);\n return this;\n }\n /**\n * Sets an array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n * @returns this effect.\n */\n setArray2(uniformName, array) {\n this._pipelineContext.setArray2(uniformName, array);\n return this;\n }\n /**\n * Sets an array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n * @returns this effect.\n */\n setArray3(uniformName, array) {\n this._pipelineContext.setArray3(uniformName, array);\n return this;\n }\n /**\n * Sets an array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n * @returns this effect.\n */\n setArray4(uniformName, array) {\n this._pipelineContext.setArray4(uniformName, array);\n return this;\n }\n /**\n * Sets matrices on a uniform variable.\n * @param uniformName Name of the variable.\n * @param matrices matrices to be set.\n * @returns this effect.\n */\n setMatrices(uniformName, matrices) {\n this._pipelineContext.setMatrices(uniformName, matrices);\n return this;\n }\n /**\n * Sets matrix on a uniform variable.\n * @param uniformName Name of the variable.\n * @param matrix matrix to be set.\n * @returns this effect.\n */\n setMatrix(uniformName, matrix) {\n this._pipelineContext.setMatrix(uniformName, matrix);\n return this;\n }\n /**\n * Sets a 3x3 matrix on a uniform variable. (Specified as [1,2,3,4,5,6,7,8,9] will result in [1,2,3][4,5,6][7,8,9] matrix)\n * @param uniformName Name of the variable.\n * @param matrix matrix to be set.\n * @returns this effect.\n */\n setMatrix3x3(uniformName, matrix) {\n // the cast is ok because it is gl.uniformMatrix3fv() which is called at the end, and this function accepts Float32Array and Array\n this._pipelineContext.setMatrix3x3(uniformName, matrix);\n return this;\n }\n /**\n * Sets a 2x2 matrix on a uniform variable. (Specified as [1,2,3,4] will result in [1,2][3,4] matrix)\n * @param uniformName Name of the variable.\n * @param matrix matrix to be set.\n * @returns this effect.\n */\n setMatrix2x2(uniformName, matrix) {\n // the cast is ok because it is gl.uniformMatrix3fv() which is called at the end, and this function accepts Float32Array and Array\n this._pipelineContext.setMatrix2x2(uniformName, matrix);\n return this;\n }\n /**\n * Sets a float on a uniform variable.\n * @param uniformName Name of the variable.\n * @param value value to be set.\n * @returns this effect.\n */\n setFloat(uniformName, value) {\n this._pipelineContext.setFloat(uniformName, value);\n return this;\n }\n /**\n * Sets a boolean on a uniform variable.\n * @param uniformName Name of the variable.\n * @param bool value to be set.\n * @returns this effect.\n */\n setBool(uniformName, bool) {\n this._pipelineContext.setInt(uniformName, bool ? 1 : 0);\n return this;\n }\n /**\n * Sets a Vector2 on a uniform variable.\n * @param uniformName Name of the variable.\n * @param vector2 vector2 to be set.\n * @returns this effect.\n */\n setVector2(uniformName, vector2) {\n this._pipelineContext.setVector2(uniformName, vector2);\n return this;\n }\n /**\n * Sets a float2 on a uniform variable.\n * @param uniformName Name of the variable.\n * @param x First float in float2.\n * @param y Second float in float2.\n * @returns this effect.\n */\n setFloat2(uniformName, x, y) {\n this._pipelineContext.setFloat2(uniformName, x, y);\n return this;\n }\n /**\n * Sets a Vector3 on a uniform variable.\n * @param uniformName Name of the variable.\n * @param vector3 Value to be set.\n * @returns this effect.\n */\n setVector3(uniformName, vector3) {\n this._pipelineContext.setVector3(uniformName, vector3);\n return this;\n }\n /**\n * Sets a float3 on a uniform variable.\n * @param uniformName Name of the variable.\n * @param x First float in float3.\n * @param y Second float in float3.\n * @param z Third float in float3.\n * @returns this effect.\n */\n setFloat3(uniformName, x, y, z) {\n this._pipelineContext.setFloat3(uniformName, x, y, z);\n return this;\n }\n /**\n * Sets a Vector4 on a uniform variable.\n * @param uniformName Name of the variable.\n * @param vector4 Value to be set.\n * @returns this effect.\n */\n setVector4(uniformName, vector4) {\n this._pipelineContext.setVector4(uniformName, vector4);\n return this;\n }\n /**\n * Sets a Quaternion on a uniform variable.\n * @param uniformName Name of the variable.\n * @param quaternion Value to be set.\n * @returns this effect.\n */\n setQuaternion(uniformName, quaternion) {\n this._pipelineContext.setQuaternion(uniformName, quaternion);\n return this;\n }\n /**\n * Sets a float4 on a uniform variable.\n * @param uniformName Name of the variable.\n * @param x First float in float4.\n * @param y Second float in float4.\n * @param z Third float in float4.\n * @param w Fourth float in float4.\n * @returns this effect.\n */\n setFloat4(uniformName, x, y, z, w) {\n this._pipelineContext.setFloat4(uniformName, x, y, z, w);\n return this;\n }\n /**\n * Sets a Color3 on a uniform variable.\n * @param uniformName Name of the variable.\n * @param color3 Value to be set.\n * @returns this effect.\n */\n setColor3(uniformName, color3) {\n this._pipelineContext.setColor3(uniformName, color3);\n return this;\n }\n /**\n * Sets a Color4 on a uniform variable.\n * @param uniformName Name of the variable.\n * @param color3 Value to be set.\n * @param alpha Alpha value to be set.\n * @returns this effect.\n */\n setColor4(uniformName, color3, alpha) {\n this._pipelineContext.setColor4(uniformName, color3, alpha);\n return this;\n }\n /**\n * Sets a Color4 on a uniform variable\n * @param uniformName defines the name of the variable\n * @param color4 defines the value to be set\n * @returns this effect.\n */\n setDirectColor4(uniformName, color4) {\n this._pipelineContext.setDirectColor4(uniformName, color4);\n return this;\n }\n /**\n * Release all associated resources.\n **/\n dispose() {\n if (this._pipelineContext) {\n this._pipelineContext.dispose();\n }\n this._engine._releaseEffect(this);\n this._isDisposed = true;\n }\n /**\n * This function will add a new shader to the shader store\n * @param name the name of the shader\n * @param pixelShader optional pixel shader content\n * @param vertexShader optional vertex shader content\n * @param shaderLanguage the language the shader is written in (default: GLSL)\n */\n static RegisterShader(name, pixelShader, vertexShader, shaderLanguage = ShaderLanguage.GLSL) {\n if (pixelShader) {\n ShaderStore.GetShadersStore(shaderLanguage)[`${name}PixelShader`] = pixelShader;\n }\n if (vertexShader) {\n ShaderStore.GetShadersStore(shaderLanguage)[`${name}VertexShader`] = vertexShader;\n }\n }\n /**\n * Resets the cache of effects.\n */\n static ResetCache() {\n Effect._BaseCache = {};\n }\n }\n /**\n * Enable logging of the shader code when a compilation error occurs\n */\n Effect.LogShaderCodeOnCompilationError = true;\n Effect._UniqueIdSeed = 0;\n Effect._BaseCache = {};\n /**\n * Store of each shader (The can be looked up using effect.key)\n */\n Effect.ShadersStore = ShaderStore.ShadersStore;\n /**\n * Store of each included file for a shader (The can be looked up using effect.key)\n */\n Effect.IncludesShadersStore = ShaderStore.IncludesShadersStore;\n\n /**\n * @internal\n **/\n class DepthCullingState {\n /**\n * Initializes the state.\n * @param reset\n */\n constructor(reset = true) {\n this._isDepthTestDirty = false;\n this._isDepthMaskDirty = false;\n this._isDepthFuncDirty = false;\n this._isCullFaceDirty = false;\n this._isCullDirty = false;\n this._isZOffsetDirty = false;\n this._isFrontFaceDirty = false;\n if (reset) {\n this.reset();\n }\n }\n get isDirty() {\n return (this._isDepthFuncDirty ||\n this._isDepthTestDirty ||\n this._isDepthMaskDirty ||\n this._isCullFaceDirty ||\n this._isCullDirty ||\n this._isZOffsetDirty ||\n this._isFrontFaceDirty);\n }\n get zOffset() {\n return this._zOffset;\n }\n set zOffset(value) {\n if (this._zOffset === value) {\n return;\n }\n this._zOffset = value;\n this._isZOffsetDirty = true;\n }\n get zOffsetUnits() {\n return this._zOffsetUnits;\n }\n set zOffsetUnits(value) {\n if (this._zOffsetUnits === value) {\n return;\n }\n this._zOffsetUnits = value;\n this._isZOffsetDirty = true;\n }\n get cullFace() {\n return this._cullFace;\n }\n set cullFace(value) {\n if (this._cullFace === value) {\n return;\n }\n this._cullFace = value;\n this._isCullFaceDirty = true;\n }\n get cull() {\n return this._cull;\n }\n set cull(value) {\n if (this._cull === value) {\n return;\n }\n this._cull = value;\n this._isCullDirty = true;\n }\n get depthFunc() {\n return this._depthFunc;\n }\n set depthFunc(value) {\n if (this._depthFunc === value) {\n return;\n }\n this._depthFunc = value;\n this._isDepthFuncDirty = true;\n }\n get depthMask() {\n return this._depthMask;\n }\n set depthMask(value) {\n if (this._depthMask === value) {\n return;\n }\n this._depthMask = value;\n this._isDepthMaskDirty = true;\n }\n get depthTest() {\n return this._depthTest;\n }\n set depthTest(value) {\n if (this._depthTest === value) {\n return;\n }\n this._depthTest = value;\n this._isDepthTestDirty = true;\n }\n get frontFace() {\n return this._frontFace;\n }\n set frontFace(value) {\n if (this._frontFace === value) {\n return;\n }\n this._frontFace = value;\n this._isFrontFaceDirty = true;\n }\n reset() {\n this._depthMask = true;\n this._depthTest = true;\n this._depthFunc = null;\n this._cullFace = null;\n this._cull = null;\n this._zOffset = 0;\n this._zOffsetUnits = 0;\n this._frontFace = null;\n this._isDepthTestDirty = true;\n this._isDepthMaskDirty = true;\n this._isDepthFuncDirty = false;\n this._isCullFaceDirty = false;\n this._isCullDirty = false;\n this._isZOffsetDirty = true;\n this._isFrontFaceDirty = false;\n }\n apply(gl) {\n if (!this.isDirty) {\n return;\n }\n // Cull\n if (this._isCullDirty) {\n if (this.cull) {\n gl.enable(gl.CULL_FACE);\n }\n else {\n gl.disable(gl.CULL_FACE);\n }\n this._isCullDirty = false;\n }\n // Cull face\n if (this._isCullFaceDirty) {\n gl.cullFace(this.cullFace);\n this._isCullFaceDirty = false;\n }\n // Depth mask\n if (this._isDepthMaskDirty) {\n gl.depthMask(this.depthMask);\n this._isDepthMaskDirty = false;\n }\n // Depth test\n if (this._isDepthTestDirty) {\n if (this.depthTest) {\n gl.enable(gl.DEPTH_TEST);\n }\n else {\n gl.disable(gl.DEPTH_TEST);\n }\n this._isDepthTestDirty = false;\n }\n // Depth func\n if (this._isDepthFuncDirty) {\n gl.depthFunc(this.depthFunc);\n this._isDepthFuncDirty = false;\n }\n // zOffset\n if (this._isZOffsetDirty) {\n if (this.zOffset || this.zOffsetUnits) {\n gl.enable(gl.POLYGON_OFFSET_FILL);\n gl.polygonOffset(this.zOffset, this.zOffsetUnits);\n }\n else {\n gl.disable(gl.POLYGON_OFFSET_FILL);\n }\n this._isZOffsetDirty = false;\n }\n // Front face\n if (this._isFrontFaceDirty) {\n gl.frontFace(this.frontFace);\n this._isFrontFaceDirty = false;\n }\n }\n }\n\n /**\n * @internal\n **/\n class StencilState {\n constructor() {\n this.reset();\n }\n reset() {\n this.enabled = false;\n this.mask = 0xff;\n this.func = StencilState.ALWAYS;\n this.funcRef = 1;\n this.funcMask = 0xff;\n this.opStencilFail = StencilState.KEEP;\n this.opDepthFail = StencilState.KEEP;\n this.opStencilDepthPass = StencilState.REPLACE;\n }\n get stencilFunc() {\n return this.func;\n }\n set stencilFunc(value) {\n this.func = value;\n }\n get stencilFuncRef() {\n return this.funcRef;\n }\n set stencilFuncRef(value) {\n this.funcRef = value;\n }\n get stencilFuncMask() {\n return this.funcMask;\n }\n set stencilFuncMask(value) {\n this.funcMask = value;\n }\n get stencilOpStencilFail() {\n return this.opStencilFail;\n }\n set stencilOpStencilFail(value) {\n this.opStencilFail = value;\n }\n get stencilOpDepthFail() {\n return this.opDepthFail;\n }\n set stencilOpDepthFail(value) {\n this.opDepthFail = value;\n }\n get stencilOpStencilDepthPass() {\n return this.opStencilDepthPass;\n }\n set stencilOpStencilDepthPass(value) {\n this.opStencilDepthPass = value;\n }\n get stencilMask() {\n return this.mask;\n }\n set stencilMask(value) {\n this.mask = value;\n }\n get stencilTest() {\n return this.enabled;\n }\n set stencilTest(value) {\n this.enabled = value;\n }\n }\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will always pass. i.e. Pixels will be drawn in the order they are drawn */\n StencilState.ALWAYS = 519;\n /** Passed to stencilOperation to specify that stencil value must be kept */\n StencilState.KEEP = 7680;\n /** Passed to stencilOperation to specify that stencil value must be replaced */\n StencilState.REPLACE = 7681;\n\n /**\n * @internal\n **/\n class AlphaState {\n /**\n * Initializes the state.\n */\n constructor() {\n this._blendFunctionParameters = new Array(4);\n this._blendEquationParameters = new Array(2);\n this._blendConstants = new Array(4);\n this._isBlendConstantsDirty = false;\n this._alphaBlend = false;\n this._isAlphaBlendDirty = false;\n this._isBlendFunctionParametersDirty = false;\n this._isBlendEquationParametersDirty = false;\n this.reset();\n }\n get isDirty() {\n return this._isAlphaBlendDirty || this._isBlendFunctionParametersDirty || this._isBlendEquationParametersDirty;\n }\n get alphaBlend() {\n return this._alphaBlend;\n }\n set alphaBlend(value) {\n if (this._alphaBlend === value) {\n return;\n }\n this._alphaBlend = value;\n this._isAlphaBlendDirty = true;\n }\n setAlphaBlendConstants(r, g, b, a) {\n if (this._blendConstants[0] === r && this._blendConstants[1] === g && this._blendConstants[2] === b && this._blendConstants[3] === a) {\n return;\n }\n this._blendConstants[0] = r;\n this._blendConstants[1] = g;\n this._blendConstants[2] = b;\n this._blendConstants[3] = a;\n this._isBlendConstantsDirty = true;\n }\n setAlphaBlendFunctionParameters(value0, value1, value2, value3) {\n if (this._blendFunctionParameters[0] === value0 &&\n this._blendFunctionParameters[1] === value1 &&\n this._blendFunctionParameters[2] === value2 &&\n this._blendFunctionParameters[3] === value3) {\n return;\n }\n this._blendFunctionParameters[0] = value0;\n this._blendFunctionParameters[1] = value1;\n this._blendFunctionParameters[2] = value2;\n this._blendFunctionParameters[3] = value3;\n this._isBlendFunctionParametersDirty = true;\n }\n setAlphaEquationParameters(rgb, alpha) {\n if (this._blendEquationParameters[0] === rgb && this._blendEquationParameters[1] === alpha) {\n return;\n }\n this._blendEquationParameters[0] = rgb;\n this._blendEquationParameters[1] = alpha;\n this._isBlendEquationParametersDirty = true;\n }\n reset() {\n this._alphaBlend = false;\n this._blendFunctionParameters[0] = null;\n this._blendFunctionParameters[1] = null;\n this._blendFunctionParameters[2] = null;\n this._blendFunctionParameters[3] = null;\n this._blendEquationParameters[0] = null;\n this._blendEquationParameters[1] = null;\n this._blendConstants[0] = null;\n this._blendConstants[1] = null;\n this._blendConstants[2] = null;\n this._blendConstants[3] = null;\n this._isAlphaBlendDirty = true;\n this._isBlendFunctionParametersDirty = false;\n this._isBlendEquationParametersDirty = false;\n this._isBlendConstantsDirty = false;\n }\n apply(gl) {\n if (!this.isDirty) {\n return;\n }\n // Alpha blend\n if (this._isAlphaBlendDirty) {\n if (this._alphaBlend) {\n gl.enable(gl.BLEND);\n }\n else {\n gl.disable(gl.BLEND);\n }\n this._isAlphaBlendDirty = false;\n }\n // Alpha function\n if (this._isBlendFunctionParametersDirty) {\n gl.blendFuncSeparate(this._blendFunctionParameters[0], this._blendFunctionParameters[1], this._blendFunctionParameters[2], this._blendFunctionParameters[3]);\n this._isBlendFunctionParametersDirty = false;\n }\n // Alpha equation\n if (this._isBlendEquationParametersDirty) {\n gl.blendEquationSeparate(this._blendEquationParameters[0], this._blendEquationParameters[1]);\n this._isBlendEquationParametersDirty = false;\n }\n // Constants\n if (this._isBlendConstantsDirty) {\n gl.blendColor(this._blendConstants[0], this._blendConstants[1], this._blendConstants[2], this._blendConstants[3]);\n this._isBlendConstantsDirty = false;\n }\n }\n }\n\n /**\n * Class used to store a texture sampler data\n */\n class TextureSampler {\n /**\n * | Value | Type | Description |\n * | ----- | ------------------ | ----------- |\n * | 0 | CLAMP_ADDRESSMODE | |\n * | 1 | WRAP_ADDRESSMODE | |\n * | 2 | MIRROR_ADDRESSMODE | |\n */\n get wrapU() {\n return this._cachedWrapU;\n }\n set wrapU(value) {\n this._cachedWrapU = value;\n }\n /**\n * | Value | Type | Description |\n * | ----- | ------------------ | ----------- |\n * | 0 | CLAMP_ADDRESSMODE | |\n * | 1 | WRAP_ADDRESSMODE | |\n * | 2 | MIRROR_ADDRESSMODE | |\n */\n get wrapV() {\n return this._cachedWrapV;\n }\n set wrapV(value) {\n this._cachedWrapV = value;\n }\n /**\n * | Value | Type | Description |\n * | ----- | ------------------ | ----------- |\n * | 0 | CLAMP_ADDRESSMODE | |\n * | 1 | WRAP_ADDRESSMODE | |\n * | 2 | MIRROR_ADDRESSMODE | |\n */\n get wrapR() {\n return this._cachedWrapR;\n }\n set wrapR(value) {\n this._cachedWrapR = value;\n }\n /**\n * With compliant hardware and browser (supporting anisotropic filtering)\n * this defines the level of anisotropic filtering in the texture.\n * The higher the better but the slower.\n */\n get anisotropicFilteringLevel() {\n return this._cachedAnisotropicFilteringLevel;\n }\n set anisotropicFilteringLevel(value) {\n this._cachedAnisotropicFilteringLevel = value;\n }\n /**\n * Gets or sets the comparison function (513, 514, etc). Set 0 to not use a comparison function\n */\n get comparisonFunction() {\n return this._comparisonFunction;\n }\n set comparisonFunction(value) {\n this._comparisonFunction = value;\n }\n /**\n * Indicates to use the mip maps (if available on the texture).\n * Thanks to this flag, you can instruct the sampler to not sample the mipmaps even if they exist (and if the sampling mode is set to a value that normally samples the mipmaps!)\n */\n get useMipMaps() {\n return this._useMipMaps;\n }\n set useMipMaps(value) {\n this._useMipMaps = value;\n }\n /**\n * Creates a Sampler instance\n */\n constructor() {\n /**\n * Gets the sampling mode of the texture\n */\n this.samplingMode = -1;\n this._useMipMaps = true;\n /** @internal */\n this._cachedWrapU = null;\n /** @internal */\n this._cachedWrapV = null;\n /** @internal */\n this._cachedWrapR = null;\n /** @internal */\n this._cachedAnisotropicFilteringLevel = null;\n /** @internal */\n this._comparisonFunction = 0;\n }\n /**\n * Sets all the parameters of the sampler\n * @param wrapU u address mode (default: TEXTURE_WRAP_ADDRESSMODE)\n * @param wrapV v address mode (default: TEXTURE_WRAP_ADDRESSMODE)\n * @param wrapR r address mode (default: TEXTURE_WRAP_ADDRESSMODE)\n * @param anisotropicFilteringLevel anisotropic level (default: 1)\n * @param samplingMode sampling mode (default: 2)\n * @param comparisonFunction comparison function (default: 0 - no comparison function)\n * @returns the current sampler instance\n */\n setParameters(wrapU = 1, wrapV = 1, wrapR = 1, anisotropicFilteringLevel = 1, samplingMode = 2, comparisonFunction = 0) {\n this._cachedWrapU = wrapU;\n this._cachedWrapV = wrapV;\n this._cachedWrapR = wrapR;\n this._cachedAnisotropicFilteringLevel = anisotropicFilteringLevel;\n this.samplingMode = samplingMode;\n this._comparisonFunction = comparisonFunction;\n return this;\n }\n /**\n * Compares this sampler with another one\n * @param other sampler to compare with\n * @returns true if the samplers have the same parametres, else false\n */\n compareSampler(other) {\n return (this._cachedWrapU === other._cachedWrapU &&\n this._cachedWrapV === other._cachedWrapV &&\n this._cachedWrapR === other._cachedWrapR &&\n this._cachedAnisotropicFilteringLevel === other._cachedAnisotropicFilteringLevel &&\n this.samplingMode === other.samplingMode &&\n this._comparisonFunction === other._comparisonFunction &&\n this._useMipMaps === other._useMipMaps);\n }\n }\n\n /**\n * Defines the source of the internal texture\n */\n var InternalTextureSource;\n (function (InternalTextureSource) {\n /**\n * The source of the texture data is unknown\n */\n InternalTextureSource[InternalTextureSource[\"Unknown\"] = 0] = \"Unknown\";\n /**\n * Texture data comes from an URL\n */\n InternalTextureSource[InternalTextureSource[\"Url\"] = 1] = \"Url\";\n /**\n * Texture data is only used for temporary storage\n */\n InternalTextureSource[InternalTextureSource[\"Temp\"] = 2] = \"Temp\";\n /**\n * Texture data comes from raw data (ArrayBuffer)\n */\n InternalTextureSource[InternalTextureSource[\"Raw\"] = 3] = \"Raw\";\n /**\n * Texture content is dynamic (video or dynamic texture)\n */\n InternalTextureSource[InternalTextureSource[\"Dynamic\"] = 4] = \"Dynamic\";\n /**\n * Texture content is generated by rendering to it\n */\n InternalTextureSource[InternalTextureSource[\"RenderTarget\"] = 5] = \"RenderTarget\";\n /**\n * Texture content is part of a multi render target process\n */\n InternalTextureSource[InternalTextureSource[\"MultiRenderTarget\"] = 6] = \"MultiRenderTarget\";\n /**\n * Texture data comes from a cube data file\n */\n InternalTextureSource[InternalTextureSource[\"Cube\"] = 7] = \"Cube\";\n /**\n * Texture data comes from a raw cube data\n */\n InternalTextureSource[InternalTextureSource[\"CubeRaw\"] = 8] = \"CubeRaw\";\n /**\n * Texture data come from a prefiltered cube data file\n */\n InternalTextureSource[InternalTextureSource[\"CubePrefiltered\"] = 9] = \"CubePrefiltered\";\n /**\n * Texture content is raw 3D data\n */\n InternalTextureSource[InternalTextureSource[\"Raw3D\"] = 10] = \"Raw3D\";\n /**\n * Texture content is raw 2D array data\n */\n InternalTextureSource[InternalTextureSource[\"Raw2DArray\"] = 11] = \"Raw2DArray\";\n /**\n * Texture content is a depth/stencil texture\n */\n InternalTextureSource[InternalTextureSource[\"DepthStencil\"] = 12] = \"DepthStencil\";\n /**\n * Texture data comes from a raw cube data encoded with RGBD\n */\n InternalTextureSource[InternalTextureSource[\"CubeRawRGBD\"] = 13] = \"CubeRawRGBD\";\n /**\n * Texture content is a depth texture\n */\n InternalTextureSource[InternalTextureSource[\"Depth\"] = 14] = \"Depth\";\n })(InternalTextureSource || (InternalTextureSource = {}));\n /**\n * Class used to store data associated with WebGL texture data for the engine\n * This class should not be used directly\n */\n class InternalTexture extends TextureSampler {\n /**\n * Gets a boolean indicating if the texture uses mipmaps\n * TODO implements useMipMaps as a separate setting from generateMipMaps\n */\n get useMipMaps() {\n return this.generateMipMaps;\n }\n set useMipMaps(value) {\n this.generateMipMaps = value;\n }\n /** Gets the unique id of the internal texture */\n get uniqueId() {\n return this._uniqueId;\n }\n /** @internal */\n _setUniqueId(id) {\n this._uniqueId = id;\n }\n /**\n * Gets the Engine the texture belongs to.\n * @returns The babylon engine\n */\n getEngine() {\n return this._engine;\n }\n /**\n * Gets the data source type of the texture\n */\n get source() {\n return this._source;\n }\n /**\n * Creates a new InternalTexture\n * @param engine defines the engine to use\n * @param source defines the type of data that will be used\n * @param delayAllocation if the texture allocation should be delayed (default: false)\n */\n constructor(engine, source, delayAllocation = false) {\n super();\n /**\n * Defines if the texture is ready\n */\n this.isReady = false;\n /**\n * Defines if the texture is a cube texture\n */\n this.isCube = false;\n /**\n * Defines if the texture contains 3D data\n */\n this.is3D = false;\n /**\n * Defines if the texture contains 2D array data\n */\n this.is2DArray = false;\n /**\n * Defines if the texture contains multiview data\n */\n this.isMultiview = false;\n /**\n * Gets the URL used to load this texture\n */\n this.url = \"\";\n /**\n * Gets a boolean indicating if the texture needs mipmaps generation\n */\n this.generateMipMaps = false;\n /**\n * Gets the number of samples used by the texture (WebGL2+ only)\n */\n this.samples = 0;\n /**\n * Gets the type of the texture (int, float...)\n */\n this.type = -1;\n /**\n * Gets the format of the texture (RGB, RGBA...)\n */\n this.format = -1;\n /**\n * Observable called when the texture is loaded\n */\n this.onLoadedObservable = new Observable$1();\n /**\n * Observable called when the texture load is raising an error\n */\n this.onErrorObservable = new Observable$1();\n /**\n * If this callback is defined it will be called instead of the default _rebuild function\n */\n this.onRebuildCallback = null;\n /**\n * Gets the width of the texture\n */\n this.width = 0;\n /**\n * Gets the height of the texture\n */\n this.height = 0;\n /**\n * Gets the depth of the texture\n */\n this.depth = 0;\n /**\n * Gets the initial width of the texture (It could be rescaled if the current system does not support non power of two textures)\n */\n this.baseWidth = 0;\n /**\n * Gets the initial height of the texture (It could be rescaled if the current system does not support non power of two textures)\n */\n this.baseHeight = 0;\n /**\n * Gets the initial depth of the texture (It could be rescaled if the current system does not support non power of two textures)\n */\n this.baseDepth = 0;\n /**\n * Gets a boolean indicating if the texture is inverted on Y axis\n */\n this.invertY = false;\n // Private\n /** @internal */\n this._invertVScale = false;\n /** @internal */\n this._associatedChannel = -1;\n /** @internal */\n this._source = InternalTextureSource.Unknown;\n /** @internal */\n this._buffer = null;\n /** @internal */\n this._bufferView = null;\n /** @internal */\n this._bufferViewArray = null;\n /** @internal */\n this._bufferViewArrayArray = null;\n /** @internal */\n this._size = 0;\n /** @internal */\n this._extension = \"\";\n /** @internal */\n this._files = null;\n /** @internal */\n this._workingCanvas = null;\n /** @internal */\n this._workingContext = null;\n /** @internal */\n this._cachedCoordinatesMode = null;\n /** @internal */\n this._isDisabled = false;\n /** @internal */\n this._compression = null;\n /** @internal */\n this._sphericalPolynomial = null;\n /** @internal */\n this._sphericalPolynomialPromise = null;\n /** @internal */\n this._sphericalPolynomialComputed = false;\n /** @internal */\n this._lodGenerationScale = 0;\n /** @internal */\n this._lodGenerationOffset = 0;\n /** @internal */\n this._useSRGBBuffer = false;\n // The following three fields helps sharing generated fixed LODs for texture filtering\n // In environment not supporting the textureLOD extension like EDGE. They are for internal use only.\n // They are at the level of the gl texture to benefit from the cache.\n /** @internal */\n this._lodTextureHigh = null;\n /** @internal */\n this._lodTextureMid = null;\n /** @internal */\n this._lodTextureLow = null;\n /** @internal */\n this._isRGBD = false;\n /** @internal */\n this._linearSpecularLOD = false;\n /** @internal */\n this._irradianceTexture = null;\n /** @internal */\n this._hardwareTexture = null;\n /** @internal */\n this._maxLodLevel = null;\n /** @internal */\n this._references = 1;\n /** @internal */\n this._gammaSpace = null;\n this._engine = engine;\n this._source = source;\n this._uniqueId = InternalTexture._Counter++;\n if (!delayAllocation) {\n this._hardwareTexture = engine._createHardwareTexture();\n }\n }\n /**\n * Increments the number of references (ie. the number of Texture that point to it)\n */\n incrementReferences() {\n this._references++;\n }\n /**\n * Change the size of the texture (not the size of the content)\n * @param width defines the new width\n * @param height defines the new height\n * @param depth defines the new depth (1 by default)\n */\n updateSize(width, height, depth = 1) {\n this._engine.updateTextureDimensions(this, width, height, depth);\n this.width = width;\n this.height = height;\n this.depth = depth;\n this.baseWidth = width;\n this.baseHeight = height;\n this.baseDepth = depth;\n this._size = width * height * depth;\n }\n /** @internal */\n _rebuild() {\n var _a;\n this.isReady = false;\n this._cachedCoordinatesMode = null;\n this._cachedWrapU = null;\n this._cachedWrapV = null;\n this._cachedWrapR = null;\n this._cachedAnisotropicFilteringLevel = null;\n if (this.onRebuildCallback) {\n const data = this.onRebuildCallback(this);\n const swapAndSetIsReady = (proxyInternalTexture) => {\n proxyInternalTexture._swapAndDie(this, false);\n this.isReady = data.isReady;\n };\n if (data.isAsync) {\n data.proxy.then(swapAndSetIsReady);\n }\n else {\n swapAndSetIsReady(data.proxy);\n }\n return;\n }\n let proxy;\n switch (this.source) {\n case InternalTextureSource.Temp:\n break;\n case InternalTextureSource.Url:\n proxy = this._engine.createTexture((_a = this._originalUrl) !== null && _a !== void 0 ? _a : this.url, !this.generateMipMaps, this.invertY, null, this.samplingMode, \n // Do not use Proxy here as it could be fully synchronous\n // and proxy would be undefined.\n (temp) => {\n temp._swapAndDie(this, false);\n this.isReady = true;\n }, null, this._buffer, undefined, this.format, this._extension, undefined, undefined, undefined, this._useSRGBBuffer);\n return;\n case InternalTextureSource.Raw:\n proxy = this._engine.createRawTexture(this._bufferView, this.baseWidth, this.baseHeight, this.format, this.generateMipMaps, this.invertY, this.samplingMode, this._compression, this.type, undefined, this._useSRGBBuffer);\n proxy._swapAndDie(this, false);\n this.isReady = true;\n break;\n case InternalTextureSource.Raw3D:\n proxy = this._engine.createRawTexture3D(this._bufferView, this.baseWidth, this.baseHeight, this.baseDepth, this.format, this.generateMipMaps, this.invertY, this.samplingMode, this._compression, this.type);\n proxy._swapAndDie(this, false);\n this.isReady = true;\n break;\n case InternalTextureSource.Raw2DArray:\n proxy = this._engine.createRawTexture2DArray(this._bufferView, this.baseWidth, this.baseHeight, this.baseDepth, this.format, this.generateMipMaps, this.invertY, this.samplingMode, this._compression, this.type);\n proxy._swapAndDie(this, false);\n this.isReady = true;\n break;\n case InternalTextureSource.Dynamic:\n proxy = this._engine.createDynamicTexture(this.baseWidth, this.baseHeight, this.generateMipMaps, this.samplingMode);\n proxy._swapAndDie(this, false);\n this._engine.updateDynamicTexture(this, this._engine.getRenderingCanvas(), this.invertY, undefined, undefined, true);\n // The engine will make sure to update content so no need to flag it as isReady = true\n break;\n case InternalTextureSource.Cube:\n proxy = this._engine.createCubeTexture(this.url, null, this._files, !this.generateMipMaps, () => {\n proxy._swapAndDie(this, false);\n this.isReady = true;\n }, null, this.format, this._extension, false, 0, 0, null, undefined, this._useSRGBBuffer);\n return;\n case InternalTextureSource.CubeRaw:\n proxy = this._engine.createRawCubeTexture(this._bufferViewArray, this.width, this.format, this.type, this.generateMipMaps, this.invertY, this.samplingMode, this._compression);\n proxy._swapAndDie(this, false);\n this.isReady = true;\n break;\n case InternalTextureSource.CubeRawRGBD:\n // This case is being handeled by the environment texture tools and is not a part of the rebuild process.\n // To use CubeRawRGBD use updateRGBDAsync on the cube texture.\n return;\n case InternalTextureSource.CubePrefiltered:\n proxy = this._engine.createPrefilteredCubeTexture(this.url, null, this._lodGenerationScale, this._lodGenerationOffset, (proxy) => {\n if (proxy) {\n proxy._swapAndDie(this, false);\n }\n this.isReady = true;\n }, null, this.format, this._extension);\n proxy._sphericalPolynomial = this._sphericalPolynomial;\n return;\n }\n }\n /**\n * @internal\n */\n _swapAndDie(target, swapAll = true) {\n // TODO what about refcount on target?\n var _a;\n (_a = this._hardwareTexture) === null || _a === void 0 ? void 0 : _a.setUsage(target._source, this.generateMipMaps, this.isCube, this.width, this.height);\n target._hardwareTexture = this._hardwareTexture;\n if (swapAll) {\n target._isRGBD = this._isRGBD;\n }\n if (this._lodTextureHigh) {\n if (target._lodTextureHigh) {\n target._lodTextureHigh.dispose();\n }\n target._lodTextureHigh = this._lodTextureHigh;\n }\n if (this._lodTextureMid) {\n if (target._lodTextureMid) {\n target._lodTextureMid.dispose();\n }\n target._lodTextureMid = this._lodTextureMid;\n }\n if (this._lodTextureLow) {\n if (target._lodTextureLow) {\n target._lodTextureLow.dispose();\n }\n target._lodTextureLow = this._lodTextureLow;\n }\n if (this._irradianceTexture) {\n if (target._irradianceTexture) {\n target._irradianceTexture.dispose();\n }\n target._irradianceTexture = this._irradianceTexture;\n }\n const cache = this._engine.getLoadedTexturesCache();\n let index = cache.indexOf(this);\n if (index !== -1) {\n cache.splice(index, 1);\n }\n index = cache.indexOf(target);\n if (index === -1) {\n cache.push(target);\n }\n }\n /**\n * Dispose the current allocated resources\n */\n dispose() {\n this._references--;\n this.onLoadedObservable.clear();\n this.onErrorObservable.clear();\n if (this._references === 0) {\n this._engine._releaseTexture(this);\n this._hardwareTexture = null;\n }\n }\n }\n /** @internal */\n InternalTexture._Counter = 0;\n\n /** @internal */\n class WebGLShaderProcessor {\n constructor() {\n this.shaderLanguage = ShaderLanguage.GLSL;\n }\n postProcessor(code, defines, isFragment, processingContext, engine) {\n // Remove extensions\n if (!engine.getCaps().drawBuffersExtension) {\n // even if enclosed in #if/#endif, IE11 does parse the #extension declaration, so we need to remove it altogether\n const regex = /#extension.+GL_EXT_draw_buffers.+(enable|require)/g;\n code = code.replace(regex, \"\");\n }\n return code;\n }\n }\n\n const varyingRegex = /(flat\\s)?\\s*varying\\s*.*/;\n /** @internal */\n class WebGL2ShaderProcessor {\n constructor() {\n this.shaderLanguage = ShaderLanguage.GLSL;\n }\n attributeProcessor(attribute) {\n return attribute.replace(\"attribute\", \"in\");\n }\n varyingCheck(varying, _isFragment) {\n return varyingRegex.test(varying);\n }\n varyingProcessor(varying, isFragment) {\n return varying.replace(\"varying\", isFragment ? \"in\" : \"out\");\n }\n postProcessor(code, defines, isFragment) {\n const hasDrawBuffersExtension = code.search(/#extension.+GL_EXT_draw_buffers.+require/) !== -1;\n // Remove extensions\n const regex = /#extension.+(GL_OVR_multiview2|GL_OES_standard_derivatives|GL_EXT_shader_texture_lod|GL_EXT_frag_depth|GL_EXT_draw_buffers).+(enable|require)/g;\n code = code.replace(regex, \"\");\n // Replace instructions\n code = code.replace(/texture2D\\s*\\(/g, \"texture(\");\n if (isFragment) {\n const hasOutput = code.search(/layout *\\(location *= *0\\) *out/g) !== -1;\n code = code.replace(/texture2DLodEXT\\s*\\(/g, \"textureLod(\");\n code = code.replace(/textureCubeLodEXT\\s*\\(/g, \"textureLod(\");\n code = code.replace(/textureCube\\s*\\(/g, \"texture(\");\n code = code.replace(/gl_FragDepthEXT/g, \"gl_FragDepth\");\n code = code.replace(/gl_FragColor/g, \"glFragColor\");\n code = code.replace(/gl_FragData/g, \"glFragData\");\n code = code.replace(/void\\s+?main\\s*\\(/g, (hasDrawBuffersExtension || hasOutput ? \"\" : \"layout(location = 0) out vec4 glFragColor;\\n\") + \"void main(\");\n }\n else {\n const hasMultiviewExtension = defines.indexOf(\"#define MULTIVIEW\") !== -1;\n if (hasMultiviewExtension) {\n return \"#extension GL_OVR_multiview2 : require\\nlayout (num_views = 2) in;\\n\" + code;\n }\n }\n return code;\n }\n }\n\n /**\n * Class used to store gfx data (like WebGLBuffer)\n */\n class DataBuffer {\n /**\n * Gets the underlying buffer\n */\n get underlyingResource() {\n return null;\n }\n /**\n * Constructs the buffer\n */\n constructor() {\n /**\n * Gets or sets the number of objects referencing this buffer\n */\n this.references = 0;\n /** Gets or sets the size of the underlying buffer */\n this.capacity = 0;\n /**\n * Gets or sets a boolean indicating if the buffer contains 32bits indices\n */\n this.is32Bits = false;\n this.uniqueId = DataBuffer._Counter++;\n }\n }\n DataBuffer._Counter = 0;\n\n /** @internal */\n class WebGLDataBuffer extends DataBuffer {\n constructor(resource) {\n super();\n this._buffer = resource;\n }\n get underlyingResource() {\n return this._buffer;\n }\n }\n\n /** @internal */\n class WebGLPipelineContext {\n constructor() {\n this._valueCache = {};\n this.vertexCompilationError = null;\n this.fragmentCompilationError = null;\n this.programLinkError = null;\n this.programValidationError = null;\n /** @internal */\n this._isDisposed = false;\n }\n get isAsync() {\n return this.isParallelCompiled;\n }\n get isReady() {\n if (this.program) {\n if (this.isParallelCompiled) {\n return this.engine._isRenderingStateCompiled(this);\n }\n return true;\n }\n return false;\n }\n _handlesSpectorRebuildCallback(onCompiled) {\n if (onCompiled && this.program) {\n onCompiled(this.program);\n }\n }\n _fillEffectInformation(effect, uniformBuffersNames, uniformsNames, uniforms, samplerList, samplers, attributesNames, attributes) {\n const engine = this.engine;\n if (engine.supportsUniformBuffers) {\n for (const name in uniformBuffersNames) {\n effect.bindUniformBlock(name, uniformBuffersNames[name]);\n }\n }\n const effectAvailableUniforms = this.engine.getUniforms(this, uniformsNames);\n effectAvailableUniforms.forEach((uniform, index) => {\n uniforms[uniformsNames[index]] = uniform;\n });\n this._uniforms = uniforms;\n let index;\n for (index = 0; index < samplerList.length; index++) {\n const sampler = effect.getUniform(samplerList[index]);\n if (sampler == null) {\n samplerList.splice(index, 1);\n index--;\n }\n }\n samplerList.forEach((name, index) => {\n samplers[name] = index;\n });\n for (const attr of engine.getAttributes(this, attributesNames)) {\n attributes.push(attr);\n }\n }\n /**\n * Release all associated resources.\n **/\n dispose() {\n this._uniforms = {};\n this._isDisposed = true;\n }\n /**\n * @internal\n */\n _cacheMatrix(uniformName, matrix) {\n const cache = this._valueCache[uniformName];\n const flag = matrix.updateFlag;\n if (cache !== undefined && cache === flag) {\n return false;\n }\n this._valueCache[uniformName] = flag;\n return true;\n }\n /**\n * @internal\n */\n _cacheFloat2(uniformName, x, y) {\n let cache = this._valueCache[uniformName];\n if (!cache || cache.length !== 2) {\n cache = [x, y];\n this._valueCache[uniformName] = cache;\n return true;\n }\n let changed = false;\n if (cache[0] !== x) {\n cache[0] = x;\n changed = true;\n }\n if (cache[1] !== y) {\n cache[1] = y;\n changed = true;\n }\n return changed;\n }\n /**\n * @internal\n */\n _cacheFloat3(uniformName, x, y, z) {\n let cache = this._valueCache[uniformName];\n if (!cache || cache.length !== 3) {\n cache = [x, y, z];\n this._valueCache[uniformName] = cache;\n return true;\n }\n let changed = false;\n if (cache[0] !== x) {\n cache[0] = x;\n changed = true;\n }\n if (cache[1] !== y) {\n cache[1] = y;\n changed = true;\n }\n if (cache[2] !== z) {\n cache[2] = z;\n changed = true;\n }\n return changed;\n }\n /**\n * @internal\n */\n _cacheFloat4(uniformName, x, y, z, w) {\n let cache = this._valueCache[uniformName];\n if (!cache || cache.length !== 4) {\n cache = [x, y, z, w];\n this._valueCache[uniformName] = cache;\n return true;\n }\n let changed = false;\n if (cache[0] !== x) {\n cache[0] = x;\n changed = true;\n }\n if (cache[1] !== y) {\n cache[1] = y;\n changed = true;\n }\n if (cache[2] !== z) {\n cache[2] = z;\n changed = true;\n }\n if (cache[3] !== w) {\n cache[3] = w;\n changed = true;\n }\n return changed;\n }\n /**\n * Sets an integer value on a uniform variable.\n * @param uniformName Name of the variable.\n * @param value Value to be set.\n */\n setInt(uniformName, value) {\n const cache = this._valueCache[uniformName];\n if (cache !== undefined && cache === value) {\n return;\n }\n if (this.engine.setInt(this._uniforms[uniformName], value)) {\n this._valueCache[uniformName] = value;\n }\n }\n /**\n * Sets a int2 on a uniform variable.\n * @param uniformName Name of the variable.\n * @param x First int in int2.\n * @param y Second int in int2.\n */\n setInt2(uniformName, x, y) {\n if (this._cacheFloat2(uniformName, x, y)) {\n if (!this.engine.setInt2(this._uniforms[uniformName], x, y)) {\n this._valueCache[uniformName] = null;\n }\n }\n }\n /**\n * Sets a int3 on a uniform variable.\n * @param uniformName Name of the variable.\n * @param x First int in int3.\n * @param y Second int in int3.\n * @param z Third int in int3.\n */\n setInt3(uniformName, x, y, z) {\n if (this._cacheFloat3(uniformName, x, y, z)) {\n if (!this.engine.setInt3(this._uniforms[uniformName], x, y, z)) {\n this._valueCache[uniformName] = null;\n }\n }\n }\n /**\n * Sets a int4 on a uniform variable.\n * @param uniformName Name of the variable.\n * @param x First int in int4.\n * @param y Second int in int4.\n * @param z Third int in int4.\n * @param w Fourth int in int4.\n */\n setInt4(uniformName, x, y, z, w) {\n if (this._cacheFloat4(uniformName, x, y, z, w)) {\n if (!this.engine.setInt4(this._uniforms[uniformName], x, y, z, w)) {\n this._valueCache[uniformName] = null;\n }\n }\n }\n /**\n * Sets an int array on a uniform variable.\n * @param uniformName Name of the variable.\n * @param array array to be set.\n */\n setIntArray(uniformName, array) {\n this._valueCache[uniformName] = null;\n this.engine.setIntArray(this._uniforms[uniformName], array);\n }\n /**\n * Sets an int array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n */\n setIntArray2(uniformName, array) {\n this._valueCache[uniformName] = null;\n this.engine.setIntArray2(this._uniforms[uniformName], array);\n }\n /**\n * Sets an int array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n */\n setIntArray3(uniformName, array) {\n this._valueCache[uniformName] = null;\n this.engine.setIntArray3(this._uniforms[uniformName], array);\n }\n /**\n * Sets an int array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n */\n setIntArray4(uniformName, array) {\n this._valueCache[uniformName] = null;\n this.engine.setIntArray4(this._uniforms[uniformName], array);\n }\n /**\n * Sets an unsigned integer value on a uniform variable.\n * @param uniformName Name of the variable.\n * @param value Value to be set.\n */\n setUInt(uniformName, value) {\n const cache = this._valueCache[uniformName];\n if (cache !== undefined && cache === value) {\n return;\n }\n if (this.engine.setUInt(this._uniforms[uniformName], value)) {\n this._valueCache[uniformName] = value;\n }\n }\n /**\n * Sets an unsigned int2 value on a uniform variable.\n * @param uniformName Name of the variable.\n * @param x First unsigned int in uint2.\n * @param y Second unsigned int in uint2.\n */\n setUInt2(uniformName, x, y) {\n if (this._cacheFloat2(uniformName, x, y)) {\n if (!this.engine.setUInt2(this._uniforms[uniformName], x, y)) {\n this._valueCache[uniformName] = null;\n }\n }\n }\n /**\n * Sets an unsigned int3 value on a uniform variable.\n * @param uniformName Name of the variable.\n * @param x First unsigned int in uint3.\n * @param y Second unsigned int in uint3.\n * @param z Third unsigned int in uint3.\n */\n setUInt3(uniformName, x, y, z) {\n if (this._cacheFloat3(uniformName, x, y, z)) {\n if (!this.engine.setUInt3(this._uniforms[uniformName], x, y, z)) {\n this._valueCache[uniformName] = null;\n }\n }\n }\n /**\n * Sets an unsigned int4 value on a uniform variable.\n * @param uniformName Name of the variable.\n * @param x First unsigned int in uint4.\n * @param y Second unsigned int in uint4.\n * @param z Third unsigned int in uint4.\n * @param w Fourth unsigned int in uint4.\n */\n setUInt4(uniformName, x, y, z, w) {\n if (this._cacheFloat4(uniformName, x, y, z, w)) {\n if (!this.engine.setUInt4(this._uniforms[uniformName], x, y, z, w)) {\n this._valueCache[uniformName] = null;\n }\n }\n }\n /**\n * Sets an unsigned int array on a uniform variable.\n * @param uniformName Name of the variable.\n * @param array array to be set.\n */\n setUIntArray(uniformName, array) {\n this._valueCache[uniformName] = null;\n this.engine.setUIntArray(this._uniforms[uniformName], array);\n }\n /**\n * Sets an unsigned int array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n */\n setUIntArray2(uniformName, array) {\n this._valueCache[uniformName] = null;\n this.engine.setUIntArray2(this._uniforms[uniformName], array);\n }\n /**\n * Sets an unsigned int array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n */\n setUIntArray3(uniformName, array) {\n this._valueCache[uniformName] = null;\n this.engine.setUIntArray3(this._uniforms[uniformName], array);\n }\n /**\n * Sets an unsigned int array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n */\n setUIntArray4(uniformName, array) {\n this._valueCache[uniformName] = null;\n this.engine.setUIntArray4(this._uniforms[uniformName], array);\n }\n /**\n * Sets an array on a uniform variable.\n * @param uniformName Name of the variable.\n * @param array array to be set.\n */\n setArray(uniformName, array) {\n this._valueCache[uniformName] = null;\n this.engine.setArray(this._uniforms[uniformName], array);\n }\n /**\n * Sets an array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n */\n setArray2(uniformName, array) {\n this._valueCache[uniformName] = null;\n this.engine.setArray2(this._uniforms[uniformName], array);\n }\n /**\n * Sets an array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n * @returns this effect.\n */\n setArray3(uniformName, array) {\n this._valueCache[uniformName] = null;\n this.engine.setArray3(this._uniforms[uniformName], array);\n }\n /**\n * Sets an array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader)\n * @param uniformName Name of the variable.\n * @param array array to be set.\n */\n setArray4(uniformName, array) {\n this._valueCache[uniformName] = null;\n this.engine.setArray4(this._uniforms[uniformName], array);\n }\n /**\n * Sets matrices on a uniform variable.\n * @param uniformName Name of the variable.\n * @param matrices matrices to be set.\n */\n setMatrices(uniformName, matrices) {\n if (!matrices) {\n return;\n }\n this._valueCache[uniformName] = null;\n this.engine.setMatrices(this._uniforms[uniformName], matrices);\n }\n /**\n * Sets matrix on a uniform variable.\n * @param uniformName Name of the variable.\n * @param matrix matrix to be set.\n */\n setMatrix(uniformName, matrix) {\n if (this._cacheMatrix(uniformName, matrix)) {\n if (!this.engine.setMatrices(this._uniforms[uniformName], matrix.toArray())) {\n this._valueCache[uniformName] = null;\n }\n }\n }\n /**\n * Sets a 3x3 matrix on a uniform variable. (Specified as [1,2,3,4,5,6,7,8,9] will result in [1,2,3][4,5,6][7,8,9] matrix)\n * @param uniformName Name of the variable.\n * @param matrix matrix to be set.\n */\n setMatrix3x3(uniformName, matrix) {\n this._valueCache[uniformName] = null;\n this.engine.setMatrix3x3(this._uniforms[uniformName], matrix);\n }\n /**\n * Sets a 2x2 matrix on a uniform variable. (Specified as [1,2,3,4] will result in [1,2][3,4] matrix)\n * @param uniformName Name of the variable.\n * @param matrix matrix to be set.\n */\n setMatrix2x2(uniformName, matrix) {\n this._valueCache[uniformName] = null;\n this.engine.setMatrix2x2(this._uniforms[uniformName], matrix);\n }\n /**\n * Sets a float on a uniform variable.\n * @param uniformName Name of the variable.\n * @param value value to be set.\n * @returns this effect.\n */\n setFloat(uniformName, value) {\n const cache = this._valueCache[uniformName];\n if (cache !== undefined && cache === value) {\n return;\n }\n if (this.engine.setFloat(this._uniforms[uniformName], value)) {\n this._valueCache[uniformName] = value;\n }\n }\n /**\n * Sets a Vector2 on a uniform variable.\n * @param uniformName Name of the variable.\n * @param vector2 vector2 to be set.\n */\n setVector2(uniformName, vector2) {\n if (this._cacheFloat2(uniformName, vector2.x, vector2.y)) {\n if (!this.engine.setFloat2(this._uniforms[uniformName], vector2.x, vector2.y)) {\n this._valueCache[uniformName] = null;\n }\n }\n }\n /**\n * Sets a float2 on a uniform variable.\n * @param uniformName Name of the variable.\n * @param x First float in float2.\n * @param y Second float in float2.\n */\n setFloat2(uniformName, x, y) {\n if (this._cacheFloat2(uniformName, x, y)) {\n if (!this.engine.setFloat2(this._uniforms[uniformName], x, y)) {\n this._valueCache[uniformName] = null;\n }\n }\n }\n /**\n * Sets a Vector3 on a uniform variable.\n * @param uniformName Name of the variable.\n * @param vector3 Value to be set.\n */\n setVector3(uniformName, vector3) {\n if (this._cacheFloat3(uniformName, vector3.x, vector3.y, vector3.z)) {\n if (!this.engine.setFloat3(this._uniforms[uniformName], vector3.x, vector3.y, vector3.z)) {\n this._valueCache[uniformName] = null;\n }\n }\n }\n /**\n * Sets a float3 on a uniform variable.\n * @param uniformName Name of the variable.\n * @param x First float in float3.\n * @param y Second float in float3.\n * @param z Third float in float3.\n */\n setFloat3(uniformName, x, y, z) {\n if (this._cacheFloat3(uniformName, x, y, z)) {\n if (!this.engine.setFloat3(this._uniforms[uniformName], x, y, z)) {\n this._valueCache[uniformName] = null;\n }\n }\n }\n /**\n * Sets a Vector4 on a uniform variable.\n * @param uniformName Name of the variable.\n * @param vector4 Value to be set.\n */\n setVector4(uniformName, vector4) {\n if (this._cacheFloat4(uniformName, vector4.x, vector4.y, vector4.z, vector4.w)) {\n if (!this.engine.setFloat4(this._uniforms[uniformName], vector4.x, vector4.y, vector4.z, vector4.w)) {\n this._valueCache[uniformName] = null;\n }\n }\n }\n /**\n * Sets a Quaternion on a uniform variable.\n * @param uniformName Name of the variable.\n * @param quaternion Value to be set.\n */\n setQuaternion(uniformName, quaternion) {\n if (this._cacheFloat4(uniformName, quaternion.x, quaternion.y, quaternion.z, quaternion.w)) {\n if (!this.engine.setFloat4(this._uniforms[uniformName], quaternion.x, quaternion.y, quaternion.z, quaternion.w)) {\n this._valueCache[uniformName] = null;\n }\n }\n }\n /**\n * Sets a float4 on a uniform variable.\n * @param uniformName Name of the variable.\n * @param x First float in float4.\n * @param y Second float in float4.\n * @param z Third float in float4.\n * @param w Fourth float in float4.\n * @returns this effect.\n */\n setFloat4(uniformName, x, y, z, w) {\n if (this._cacheFloat4(uniformName, x, y, z, w)) {\n if (!this.engine.setFloat4(this._uniforms[uniformName], x, y, z, w)) {\n this._valueCache[uniformName] = null;\n }\n }\n }\n /**\n * Sets a Color3 on a uniform variable.\n * @param uniformName Name of the variable.\n * @param color3 Value to be set.\n */\n setColor3(uniformName, color3) {\n if (this._cacheFloat3(uniformName, color3.r, color3.g, color3.b)) {\n if (!this.engine.setFloat3(this._uniforms[uniformName], color3.r, color3.g, color3.b)) {\n this._valueCache[uniformName] = null;\n }\n }\n }\n /**\n * Sets a Color4 on a uniform variable.\n * @param uniformName Name of the variable.\n * @param color3 Value to be set.\n * @param alpha Alpha value to be set.\n */\n setColor4(uniformName, color3, alpha) {\n if (this._cacheFloat4(uniformName, color3.r, color3.g, color3.b, alpha)) {\n if (!this.engine.setFloat4(this._uniforms[uniformName], color3.r, color3.g, color3.b, alpha)) {\n this._valueCache[uniformName] = null;\n }\n }\n }\n /**\n * Sets a Color4 on a uniform variable\n * @param uniformName defines the name of the variable\n * @param color4 defines the value to be set\n */\n setDirectColor4(uniformName, color4) {\n if (this._cacheFloat4(uniformName, color4.r, color4.g, color4.b, color4.a)) {\n if (!this.engine.setFloat4(this._uniforms[uniformName], color4.r, color4.g, color4.b, color4.a)) {\n this._valueCache[uniformName] = null;\n }\n }\n }\n _getVertexShaderCode() {\n return this.vertexShader ? this.engine._getShaderSource(this.vertexShader) : null;\n }\n _getFragmentShaderCode() {\n return this.fragmentShader ? this.engine._getShaderSource(this.fragmentShader) : null;\n }\n }\n\n /** @internal */\n class WebGLHardwareTexture {\n get underlyingResource() {\n return this._webGLTexture;\n }\n constructor(existingTexture = null, context) {\n // There can be multiple buffers for a single WebGL texture because different layers of a 2DArrayTexture / 3DTexture\n // or different faces of a cube texture can be bound to different render targets at the same time.\n this._MSAARenderBuffers = null;\n this._context = context;\n if (!existingTexture) {\n existingTexture = context.createTexture();\n if (!existingTexture) {\n throw new Error(\"Unable to create webGL texture\");\n }\n }\n this.set(existingTexture);\n }\n setUsage() { }\n set(hardwareTexture) {\n this._webGLTexture = hardwareTexture;\n }\n reset() {\n this._webGLTexture = null;\n this._MSAARenderBuffers = null;\n }\n addMSAARenderBuffer(buffer) {\n if (!this._MSAARenderBuffers) {\n this._MSAARenderBuffers = [];\n }\n this._MSAARenderBuffers.push(buffer);\n }\n releaseMSAARenderBuffers() {\n if (this._MSAARenderBuffers) {\n for (const buffer of this._MSAARenderBuffers) {\n this._context.deleteRenderbuffer(buffer);\n }\n this._MSAARenderBuffers = null;\n }\n }\n release() {\n this.releaseMSAARenderBuffers();\n if (this._webGLTexture) {\n this._context.deleteTexture(this._webGLTexture);\n }\n this.reset();\n }\n }\n\n /** @internal */\n class DrawWrapper {\n static IsWrapper(effect) {\n return effect.getPipelineContext === undefined;\n }\n static GetEffect(effect) {\n return effect.getPipelineContext === undefined ? effect.effect : effect;\n }\n constructor(engine, createMaterialContext = true) {\n this.effect = null;\n this.defines = null;\n this.drawContext = engine.createDrawContext();\n if (createMaterialContext) {\n this.materialContext = engine.createMaterialContext();\n }\n }\n setEffect(effect, defines, resetContext = true) {\n var _a;\n this.effect = effect;\n if (defines !== undefined) {\n this.defines = defines;\n }\n if (resetContext) {\n (_a = this.drawContext) === null || _a === void 0 ? void 0 : _a.reset();\n }\n }\n dispose() {\n var _a;\n (_a = this.drawContext) === null || _a === void 0 ? void 0 : _a.dispose();\n }\n }\n\n /**\n * @internal\n **/\n class StencilStateComposer {\n get isDirty() {\n return this._isStencilTestDirty || this._isStencilMaskDirty || this._isStencilFuncDirty || this._isStencilOpDirty;\n }\n get func() {\n return this._func;\n }\n set func(value) {\n if (this._func === value) {\n return;\n }\n this._func = value;\n this._isStencilFuncDirty = true;\n }\n get funcRef() {\n return this._funcRef;\n }\n set funcRef(value) {\n if (this._funcRef === value) {\n return;\n }\n this._funcRef = value;\n this._isStencilFuncDirty = true;\n }\n get funcMask() {\n return this._funcMask;\n }\n set funcMask(value) {\n if (this._funcMask === value) {\n return;\n }\n this._funcMask = value;\n this._isStencilFuncDirty = true;\n }\n get opStencilFail() {\n return this._opStencilFail;\n }\n set opStencilFail(value) {\n if (this._opStencilFail === value) {\n return;\n }\n this._opStencilFail = value;\n this._isStencilOpDirty = true;\n }\n get opDepthFail() {\n return this._opDepthFail;\n }\n set opDepthFail(value) {\n if (this._opDepthFail === value) {\n return;\n }\n this._opDepthFail = value;\n this._isStencilOpDirty = true;\n }\n get opStencilDepthPass() {\n return this._opStencilDepthPass;\n }\n set opStencilDepthPass(value) {\n if (this._opStencilDepthPass === value) {\n return;\n }\n this._opStencilDepthPass = value;\n this._isStencilOpDirty = true;\n }\n get mask() {\n return this._mask;\n }\n set mask(value) {\n if (this._mask === value) {\n return;\n }\n this._mask = value;\n this._isStencilMaskDirty = true;\n }\n get enabled() {\n return this._enabled;\n }\n set enabled(value) {\n if (this._enabled === value) {\n return;\n }\n this._enabled = value;\n this._isStencilTestDirty = true;\n }\n constructor(reset = true) {\n this._isStencilTestDirty = false;\n this._isStencilMaskDirty = false;\n this._isStencilFuncDirty = false;\n this._isStencilOpDirty = false;\n this.useStencilGlobalOnly = false;\n if (reset) {\n this.reset();\n }\n }\n reset() {\n var _a;\n this.stencilMaterial = undefined;\n (_a = this.stencilGlobal) === null || _a === void 0 ? void 0 : _a.reset();\n this._isStencilTestDirty = true;\n this._isStencilMaskDirty = true;\n this._isStencilFuncDirty = true;\n this._isStencilOpDirty = true;\n }\n apply(gl) {\n var _a;\n if (!gl) {\n return;\n }\n const stencilMaterialEnabled = !this.useStencilGlobalOnly && !!((_a = this.stencilMaterial) === null || _a === void 0 ? void 0 : _a.enabled);\n this.enabled = stencilMaterialEnabled ? this.stencilMaterial.enabled : this.stencilGlobal.enabled;\n this.func = stencilMaterialEnabled ? this.stencilMaterial.func : this.stencilGlobal.func;\n this.funcRef = stencilMaterialEnabled ? this.stencilMaterial.funcRef : this.stencilGlobal.funcRef;\n this.funcMask = stencilMaterialEnabled ? this.stencilMaterial.funcMask : this.stencilGlobal.funcMask;\n this.opStencilFail = stencilMaterialEnabled ? this.stencilMaterial.opStencilFail : this.stencilGlobal.opStencilFail;\n this.opDepthFail = stencilMaterialEnabled ? this.stencilMaterial.opDepthFail : this.stencilGlobal.opDepthFail;\n this.opStencilDepthPass = stencilMaterialEnabled ? this.stencilMaterial.opStencilDepthPass : this.stencilGlobal.opStencilDepthPass;\n this.mask = stencilMaterialEnabled ? this.stencilMaterial.mask : this.stencilGlobal.mask;\n if (!this.isDirty) {\n return;\n }\n // Stencil test\n if (this._isStencilTestDirty) {\n if (this.enabled) {\n gl.enable(gl.STENCIL_TEST);\n }\n else {\n gl.disable(gl.STENCIL_TEST);\n }\n this._isStencilTestDirty = false;\n }\n // Stencil mask\n if (this._isStencilMaskDirty) {\n gl.stencilMask(this.mask);\n this._isStencilMaskDirty = false;\n }\n // Stencil func\n if (this._isStencilFuncDirty) {\n gl.stencilFunc(this.func, this.funcRef, this.funcMask);\n this._isStencilFuncDirty = false;\n }\n // Stencil op\n if (this._isStencilOpDirty) {\n gl.stencilOp(this.opStencilFail, this.opDepthFail, this.opStencilDepthPass);\n this._isStencilOpDirty = false;\n }\n }\n }\n\n /* eslint-disable @typescript-eslint/no-unused-vars */\n /**\n * Keeps track of all the buffer info used in engine.\n */\n class BufferPointer {\n }\n /**\n * The base engine class (root of all engines)\n */\n class ThinEngine {\n /**\n * Returns the current npm package of the sdk\n */\n // Not mixed with Version for tooling purpose.\n static get NpmPackage() {\n return \"babylonjs@6.15.0\";\n }\n /**\n * Returns the current version of the framework\n */\n static get Version() {\n return \"6.15.0\";\n }\n /**\n * Returns a string describing the current engine\n */\n get description() {\n let description = this.name + this.webGLVersion;\n if (this._caps.parallelShaderCompile) {\n description += \" - Parallel shader compilation\";\n }\n return description;\n }\n /**\n * Gets or sets the name of the engine\n */\n get name() {\n return this._name;\n }\n set name(value) {\n this._name = value;\n }\n /**\n * Returns the version of the engine\n */\n get version() {\n return this._webGLVersion;\n }\n get isDisposed() {\n return this._isDisposed;\n }\n /**\n * Gets or sets the relative url used to load shaders if using the engine in non-minified mode\n */\n static get ShadersRepository() {\n return Effect.ShadersRepository;\n }\n static set ShadersRepository(value) {\n Effect.ShadersRepository = value;\n }\n /**\n * @internal\n */\n _getShaderProcessor(shaderLanguage) {\n return this._shaderProcessor;\n }\n /**\n * Gets or sets a boolean indicating if depth buffer should be reverse, going from far to near.\n * This can provide greater z depth for distant objects.\n */\n get useReverseDepthBuffer() {\n return this._useReverseDepthBuffer;\n }\n set useReverseDepthBuffer(useReverse) {\n if (useReverse === this._useReverseDepthBuffer) {\n return;\n }\n this._useReverseDepthBuffer = useReverse;\n if (useReverse) {\n this._depthCullingState.depthFunc = 518;\n }\n else {\n this._depthCullingState.depthFunc = 515;\n }\n }\n /**\n * Gets the current frame id\n */\n get frameId() {\n return this._frameId;\n }\n /**\n * Gets a boolean indicating that the engine supports uniform buffers\n * @see https://doc.babylonjs.com/setup/support/webGL2#uniform-buffer-objets\n */\n get supportsUniformBuffers() {\n return this.webGLVersion > 1 && !this.disableUniformBuffers;\n }\n /**\n * Gets the options used for engine creation\n * @returns EngineOptions object\n */\n getCreationOptions() {\n return this._creationOptions;\n }\n /** @internal */\n get _shouldUseHighPrecisionShader() {\n return !!(this._caps.highPrecisionShaderSupported && this._highPrecisionShadersAllowed);\n }\n /**\n * Gets a boolean indicating that only power of 2 textures are supported\n * Please note that you can still use non power of 2 textures but in this case the engine will forcefully convert them\n */\n get needPOTTextures() {\n return this._webGLVersion < 2 || this.forcePOTTextures;\n }\n /**\n * Gets the list of current active render loop functions\n * @returns an array with the current render loop functions\n */\n get activeRenderLoops() {\n return this._activeRenderLoops;\n }\n /**\n * Gets or sets a boolean indicating if resources should be retained to be able to handle context lost events\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimize_your_scene#handling-webgl-context-lost\n */\n get doNotHandleContextLost() {\n return this._doNotHandleContextLost;\n }\n set doNotHandleContextLost(value) {\n this._doNotHandleContextLost = value;\n }\n get _supportsHardwareTextureRescaling() {\n return false;\n }\n /**\n * sets the object from which width and height will be taken from when getting render width and height\n * Will fallback to the gl object\n * @param dimensions the framebuffer width and height that will be used.\n */\n set framebufferDimensionsObject(dimensions) {\n this._framebufferDimensionsObject = dimensions;\n }\n /**\n * Gets the current viewport\n */\n get currentViewport() {\n return this._cachedViewport;\n }\n /**\n * Gets the default empty texture\n */\n get emptyTexture() {\n if (!this._emptyTexture) {\n this._emptyTexture = this.createRawTexture(new Uint8Array(4), 1, 1, 5, false, false, 1);\n }\n return this._emptyTexture;\n }\n /**\n * Gets the default empty 3D texture\n */\n get emptyTexture3D() {\n if (!this._emptyTexture3D) {\n this._emptyTexture3D = this.createRawTexture3D(new Uint8Array(4), 1, 1, 1, 5, false, false, 1);\n }\n return this._emptyTexture3D;\n }\n /**\n * Gets the default empty 2D array texture\n */\n get emptyTexture2DArray() {\n if (!this._emptyTexture2DArray) {\n this._emptyTexture2DArray = this.createRawTexture2DArray(new Uint8Array(4), 1, 1, 1, 5, false, false, 1);\n }\n return this._emptyTexture2DArray;\n }\n /**\n * Gets the default empty cube texture\n */\n get emptyCubeTexture() {\n if (!this._emptyCubeTexture) {\n const faceData = new Uint8Array(4);\n const cubeData = [faceData, faceData, faceData, faceData, faceData, faceData];\n this._emptyCubeTexture = this.createRawCubeTexture(cubeData, 1, 5, 0, false, false, 1);\n }\n return this._emptyCubeTexture;\n }\n /**\n * Gets a boolean indicating if the engine runs in WebGPU or not.\n */\n get isWebGPU() {\n return this._isWebGPU;\n }\n /**\n * Gets the shader platform name used by the effects.\n */\n get shaderPlatformName() {\n return this._shaderPlatformName;\n }\n /**\n * Enables or disables the snapshot rendering mode\n * Note that the WebGL engine does not support snapshot rendering so setting the value won't have any effect for this engine\n */\n get snapshotRendering() {\n return false;\n }\n set snapshotRendering(activate) {\n // WebGL engine does not support snapshot rendering\n }\n /**\n * Gets or sets the snapshot rendering mode\n */\n get snapshotRenderingMode() {\n return this._snapshotRenderingMode;\n }\n set snapshotRenderingMode(mode) {\n this._snapshotRenderingMode = mode;\n }\n /**\n * Creates a new snapshot at the next frame using the current snapshotRenderingMode\n */\n snapshotRenderingReset() {\n this.snapshotRendering = false;\n }\n static _CreateCanvas(width, height) {\n if (typeof document === \"undefined\") {\n return new OffscreenCanvas(width, height);\n }\n const canvas = document.createElement(\"canvas\");\n canvas.width = width;\n canvas.height = height;\n return canvas;\n }\n /**\n * Create a canvas. This method is overridden by other engines\n * @param width width\n * @param height height\n * @returns ICanvas interface\n */\n createCanvas(width, height) {\n return ThinEngine._CreateCanvas(width, height);\n }\n /**\n * Create an image to use with canvas\n * @returns IImage interface\n */\n createCanvasImage() {\n return document.createElement(\"img\");\n }\n /**\n * Creates a new engine\n * @param canvasOrContext defines the canvas or WebGL context to use for rendering. If you provide a WebGL context, Babylon.js will not hook events on the canvas (like pointers, keyboards, etc...) so no event observables will be available. This is mostly used when Babylon.js is used as a plugin on a system which already used the WebGL context\n * @param antialias defines enable antialiasing (default: false)\n * @param options defines further options to be sent to the getContext() function\n * @param adaptToDeviceRatio defines whether to adapt to the device's viewport characteristics (default: false)\n */\n constructor(canvasOrContext, antialias, options, adaptToDeviceRatio) {\n var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l;\n /** @internal */\n this._name = \"WebGL\";\n this._isDisposed = false;\n /**\n * Gets or sets a boolean that indicates if textures must be forced to power of 2 size even if not required\n */\n this.forcePOTTextures = false;\n /**\n * Gets a boolean indicating if the engine is currently rendering in fullscreen mode\n */\n this.isFullscreen = false;\n /**\n * Gets or sets a boolean indicating if back faces must be culled. If false, front faces are culled instead (true by default)\n * If non null, this takes precedence over the value from the material\n */\n this.cullBackFaces = null;\n /**\n * Gets or sets a boolean indicating if the engine must keep rendering even if the window is not in foreground\n */\n this.renderEvenInBackground = true;\n /**\n * Gets or sets a boolean indicating that cache can be kept between frames\n */\n this.preventCacheWipeBetweenFrames = false;\n /** Gets or sets a boolean indicating if the engine should validate programs after compilation */\n this.validateShaderPrograms = false;\n this._useReverseDepthBuffer = false;\n /**\n * Indicates if the z range in NDC space is 0..1 (value: true) or -1..1 (value: false)\n */\n this.isNDCHalfZRange = false;\n /**\n * Indicates that the origin of the texture/framebuffer space is the bottom left corner. If false, the origin is top left\n */\n this.hasOriginBottomLeft = true;\n /**\n * Gets or sets a boolean indicating that uniform buffers must be disabled even if they are supported\n */\n this.disableUniformBuffers = false;\n /**\n * An event triggered when the engine is disposed.\n */\n this.onDisposeObservable = new Observable$1();\n this._frameId = 0;\n /** @internal */\n this._uniformBuffers = new Array();\n /** @internal */\n this._storageBuffers = new Array();\n /** @internal */\n this._webGLVersion = 1.0;\n this._windowIsBackground = false;\n this._highPrecisionShadersAllowed = true;\n /** @internal */\n this._badOS = false;\n /** @internal */\n this._badDesktopOS = false;\n this._renderingQueueLaunched = false;\n this._activeRenderLoops = new Array();\n // Lost context\n /**\n * Observable signaled when a context lost event is raised\n */\n this.onContextLostObservable = new Observable$1();\n /**\n * Observable signaled when a context restored event is raised\n */\n this.onContextRestoredObservable = new Observable$1();\n this._contextWasLost = false;\n /** @internal */\n this._doNotHandleContextLost = false;\n /**\n * Gets or sets a boolean indicating that vertex array object must be disabled even if they are supported\n */\n this.disableVertexArrayObjects = false;\n // States\n /** @internal */\n this._colorWrite = true;\n /** @internal */\n this._colorWriteChanged = true;\n /** @internal */\n this._depthCullingState = new DepthCullingState();\n /** @internal */\n this._stencilStateComposer = new StencilStateComposer();\n /** @internal */\n this._stencilState = new StencilState();\n /** @internal */\n this._alphaState = new AlphaState();\n /** @internal */\n this._alphaMode = 1;\n /** @internal */\n this._alphaEquation = 0;\n // Cache\n /** @internal */\n this._internalTexturesCache = new Array();\n /** @internal */\n this._renderTargetWrapperCache = new Array();\n /** @internal */\n this._activeChannel = 0;\n this._currentTextureChannel = -1;\n /** @internal */\n this._boundTexturesCache = {};\n this._compiledEffects = {};\n this._vertexAttribArraysEnabled = [];\n this._uintIndicesCurrentlySet = false;\n this._currentBoundBuffer = new Array();\n /** @internal */\n this._currentFramebuffer = null;\n /** @internal */\n this._dummyFramebuffer = null;\n this._currentBufferPointers = new Array();\n this._currentInstanceLocations = new Array();\n this._currentInstanceBuffers = new Array();\n this._vaoRecordInProgress = false;\n this._mustWipeVertexAttributes = false;\n this._nextFreeTextureSlots = new Array();\n this._maxSimultaneousTextures = 0;\n this._maxMSAASamplesOverride = null;\n this._activeRequests = new Array();\n /**\n * If set to true zooming in and out in the browser will rescale the hardware-scaling correctly.\n */\n this.adaptToDeviceRatio = false;\n /** @internal */\n this._lastDevicePixelRatio = 1.0;\n /** @internal */\n this._transformTextureUrl = null;\n /**\n * Gets information about the current host\n */\n this.hostInformation = {\n isMobile: false,\n };\n /**\n * Defines whether the engine has been created with the premultipliedAlpha option on or not.\n */\n this.premultipliedAlpha = true;\n /**\n * Observable event triggered before each texture is initialized\n */\n this.onBeforeTextureInitObservable = new Observable$1();\n /** @internal */\n this._isWebGPU = false;\n this._snapshotRenderingMode = 0;\n this._viewportCached = { x: 0, y: 0, z: 0, w: 0 };\n this._unpackFlipYCached = null;\n /**\n * In case you are sharing the context with other applications, it might\n * be interested to not cache the unpack flip y state to ensure a consistent\n * value would be set.\n */\n this.enableUnpackFlipYCached = true;\n this._boundUniforms = {};\n this.startTime = PrecisionDate.Now;\n let canvas = null;\n options = options || {};\n this._creationOptions = options;\n // Save this off for use in resize().\n this.adaptToDeviceRatio = adaptToDeviceRatio !== null && adaptToDeviceRatio !== void 0 ? adaptToDeviceRatio : false;\n this._stencilStateComposer.stencilGlobal = this._stencilState;\n PerformanceConfigurator.SetMatrixPrecision(!!options.useHighPrecisionMatrix);\n options.antialias = antialias !== null && antialias !== void 0 ? antialias : options.antialias;\n options.deterministicLockstep = (_a = options.deterministicLockstep) !== null && _a !== void 0 ? _a : false;\n options.lockstepMaxSteps = (_b = options.lockstepMaxSteps) !== null && _b !== void 0 ? _b : 4;\n options.timeStep = (_c = options.timeStep) !== null && _c !== void 0 ? _c : 1 / 60;\n options.audioEngine = (_d = options.audioEngine) !== null && _d !== void 0 ? _d : true;\n options.stencil = (_e = options.stencil) !== null && _e !== void 0 ? _e : true;\n this._audioContext = (_g = (_f = options.audioEngineOptions) === null || _f === void 0 ? void 0 : _f.audioContext) !== null && _g !== void 0 ? _g : null;\n this._audioDestination = (_j = (_h = options.audioEngineOptions) === null || _h === void 0 ? void 0 : _h.audioDestination) !== null && _j !== void 0 ? _j : null;\n this.premultipliedAlpha = (_k = options.premultipliedAlpha) !== null && _k !== void 0 ? _k : true;\n this.useExactSrgbConversions = (_l = options.useExactSrgbConversions) !== null && _l !== void 0 ? _l : false;\n this._doNotHandleContextLost = !!options.doNotHandleContextLost;\n this._isStencilEnable = options.stencil ? true : false;\n // Viewport\n adaptToDeviceRatio = adaptToDeviceRatio || options.adaptToDeviceRatio || false;\n const devicePixelRatio = IsWindowObjectExist() ? window.devicePixelRatio || 1.0 : 1.0;\n const limitDeviceRatio = options.limitDeviceRatio || devicePixelRatio;\n this._hardwareScalingLevel = adaptToDeviceRatio ? 1.0 / Math.min(limitDeviceRatio, devicePixelRatio) : 1.0;\n this._lastDevicePixelRatio = devicePixelRatio;\n if (!canvasOrContext) {\n return;\n }\n if (canvasOrContext.getContext) {\n canvas = canvasOrContext;\n this._renderingCanvas = canvas;\n if (options.preserveDrawingBuffer === undefined) {\n options.preserveDrawingBuffer = false;\n }\n if (options.xrCompatible === undefined) {\n options.xrCompatible = true;\n }\n // Exceptions\n if (navigator && navigator.userAgent) {\n this._setupMobileChecks();\n const ua = navigator.userAgent;\n for (const exception of ThinEngine.ExceptionList) {\n const key = exception.key;\n const targets = exception.targets;\n const check = new RegExp(key);\n if (check.test(ua)) {\n if (exception.capture && exception.captureConstraint) {\n const capture = exception.capture;\n const constraint = exception.captureConstraint;\n const regex = new RegExp(capture);\n const matches = regex.exec(ua);\n if (matches && matches.length > 0) {\n const capturedValue = parseInt(matches[matches.length - 1]);\n if (capturedValue >= constraint) {\n continue;\n }\n }\n }\n for (const target of targets) {\n switch (target) {\n case \"uniformBuffer\":\n this.disableUniformBuffers = true;\n break;\n case \"vao\":\n this.disableVertexArrayObjects = true;\n break;\n case \"antialias\":\n options.antialias = false;\n break;\n case \"maxMSAASamples\":\n this._maxMSAASamplesOverride = 1;\n break;\n }\n }\n }\n }\n }\n // Context lost\n if (!this._doNotHandleContextLost) {\n this._onContextLost = (evt) => {\n evt.preventDefault();\n this._contextWasLost = true;\n Logger.Warn(\"WebGL context lost.\");\n this.onContextLostObservable.notifyObservers(this);\n };\n this._onContextRestored = () => {\n this._restoreEngineAfterContextLost(this._initGLContext.bind(this));\n };\n canvas.addEventListener(\"webglcontextlost\", this._onContextLost, false);\n canvas.addEventListener(\"webglcontextrestored\", this._onContextRestored, false);\n options.powerPreference = options.powerPreference || \"high-performance\";\n }\n // Detect if we are running on a faulty buggy desktop OS.\n this._badDesktopOS = /^((?!chrome|android).)*safari/i.test(navigator.userAgent);\n if (this._badDesktopOS) {\n options.xrCompatible = false;\n }\n // GL\n if (!options.disableWebGL2Support) {\n try {\n this._gl = (canvas.getContext(\"webgl2\", options) || canvas.getContext(\"experimental-webgl2\", options));\n if (this._gl) {\n this._webGLVersion = 2.0;\n this._shaderPlatformName = \"WEBGL2\";\n // Prevent weird browsers to lie (yeah that happens!)\n if (!this._gl.deleteQuery) {\n this._webGLVersion = 1.0;\n this._shaderPlatformName = \"WEBGL1\";\n }\n }\n }\n catch (e) {\n // Do nothing\n }\n }\n if (!this._gl) {\n if (!canvas) {\n throw new Error(\"The provided canvas is null or undefined.\");\n }\n try {\n this._gl = (canvas.getContext(\"webgl\", options) || canvas.getContext(\"experimental-webgl\", options));\n }\n catch (e) {\n throw new Error(\"WebGL not supported\");\n }\n }\n if (!this._gl) {\n throw new Error(\"WebGL not supported\");\n }\n }\n else {\n this._gl = canvasOrContext;\n this._renderingCanvas = this._gl.canvas;\n if (this._gl.renderbufferStorageMultisample) {\n this._webGLVersion = 2.0;\n this._shaderPlatformName = \"WEBGL2\";\n }\n else {\n this._shaderPlatformName = \"WEBGL1\";\n }\n const attributes = this._gl.getContextAttributes();\n if (attributes) {\n options.stencil = attributes.stencil;\n }\n }\n // Ensures a consistent color space unpacking of textures cross browser.\n this._gl.pixelStorei(this._gl.UNPACK_COLORSPACE_CONVERSION_WEBGL, this._gl.NONE);\n if (options.useHighPrecisionFloats !== undefined) {\n this._highPrecisionShadersAllowed = options.useHighPrecisionFloats;\n }\n this.resize();\n this._initGLContext();\n this._initFeatures();\n // Prepare buffer pointers\n for (let i = 0; i < this._caps.maxVertexAttribs; i++) {\n this._currentBufferPointers[i] = new BufferPointer();\n }\n // Shader processor\n this._shaderProcessor = this.webGLVersion > 1 ? new WebGL2ShaderProcessor() : new WebGLShaderProcessor();\n // Detect if we are running on a faulty buggy OS.\n this._badOS = /iPad/i.test(navigator.userAgent) || /iPhone/i.test(navigator.userAgent);\n // Starting with iOS 14, we can trust the browser\n // let matches = navigator.userAgent.match(/Version\\/(\\d+)/);\n // if (matches && matches.length === 2) {\n // if (parseInt(matches[1]) >= 14) {\n // this._badOS = false;\n // }\n // }\n const versionToLog = `Babylon.js v${ThinEngine.Version}`;\n console.log(versionToLog + ` - ${this.description}`);\n // Check setAttribute in case of workers\n if (this._renderingCanvas && this._renderingCanvas.setAttribute) {\n this._renderingCanvas.setAttribute(\"data-engine\", versionToLog);\n }\n }\n _setupMobileChecks() {\n if (!(navigator && navigator.userAgent)) {\n return;\n }\n // Function to check if running on mobile device\n this._checkForMobile = () => {\n const currentUA = navigator.userAgent;\n this.hostInformation.isMobile =\n currentUA.indexOf(\"Mobile\") !== -1 ||\n // Needed for iOS 13+ detection on iPad (inspired by solution from https://stackoverflow.com/questions/9038625/detect-if-device-is-ios)\n (currentUA.indexOf(\"Mac\") !== -1 && IsDocumentAvailable() && \"ontouchend\" in document);\n };\n // Set initial isMobile value\n this._checkForMobile();\n // Set up event listener to check when window is resized (used to get emulator activation to work properly)\n if (IsWindowObjectExist()) {\n window.addEventListener(\"resize\", this._checkForMobile);\n }\n }\n _restoreEngineAfterContextLost(initEngine) {\n // Adding a timeout to avoid race condition at browser level\n setTimeout(async () => {\n var _a;\n this._dummyFramebuffer = null;\n const depthTest = this._depthCullingState.depthTest; // backup those values because the call to initEngine / wipeCaches will reset them\n const depthFunc = this._depthCullingState.depthFunc;\n const depthMask = this._depthCullingState.depthMask;\n const stencilTest = this._stencilState.stencilTest;\n // Rebuild context\n await initEngine();\n // Ensure webgl and engine states are matching\n this.wipeCaches(true);\n // Rebuild effects\n this._rebuildEffects();\n (_a = this._rebuildComputeEffects) === null || _a === void 0 ? void 0 : _a.call(this);\n // Note:\n // The call to _rebuildBuffers must be made before the call to _rebuildInternalTextures because in the process of _rebuildBuffers the buffers used by the post process managers will be rebuilt\n // and we may need to use the post process manager of the scene during _rebuildInternalTextures (in WebGL1, non-POT textures are rescaled using a post process + post process manager of the scene)\n // Rebuild buffers\n this._rebuildBuffers();\n // Rebuild textures\n this._rebuildInternalTextures();\n // Rebuild textures\n this._rebuildRenderTargetWrappers();\n // Reset engine states after all the buffer/textures/... have been rebuilt\n this.wipeCaches(true);\n this._depthCullingState.depthTest = depthTest;\n this._depthCullingState.depthFunc = depthFunc;\n this._depthCullingState.depthMask = depthMask;\n this._stencilState.stencilTest = stencilTest;\n Logger.Warn(this.name + \" context successfully restored.\");\n this.onContextRestoredObservable.notifyObservers(this);\n this._contextWasLost = false;\n }, 0);\n }\n /**\n * Shared initialization across engines types.\n * @param canvas The canvas associated with this instance of the engine.\n */\n _sharedInit(canvas) {\n this._renderingCanvas = canvas;\n }\n /**\n * @internal\n */\n _getShaderProcessingContext(shaderLanguage) {\n return null;\n }\n _rebuildInternalTextures() {\n const currentState = this._internalTexturesCache.slice(); // Do a copy because the rebuild will add proxies\n for (const internalTexture of currentState) {\n internalTexture._rebuild();\n }\n }\n _rebuildRenderTargetWrappers() {\n const currentState = this._renderTargetWrapperCache.slice(); // Do a copy because the rebuild will add proxies\n for (const renderTargetWrapper of currentState) {\n renderTargetWrapper._rebuild();\n }\n }\n _rebuildEffects() {\n for (const key in this._compiledEffects) {\n const effect = this._compiledEffects[key];\n effect._pipelineContext = null; // because _prepareEffect will try to dispose this pipeline before recreating it and that would lead to webgl errors\n effect._wasPreviouslyReady = false;\n effect._prepareEffect();\n }\n Effect.ResetCache();\n }\n /**\n * Gets a boolean indicating if all created effects are ready\n * @returns true if all effects are ready\n */\n areAllEffectsReady() {\n for (const key in this._compiledEffects) {\n const effect = this._compiledEffects[key];\n if (!effect.isReady()) {\n return false;\n }\n }\n return true;\n }\n _rebuildBuffers() {\n // Uniforms\n for (const uniformBuffer of this._uniformBuffers) {\n uniformBuffer._rebuild();\n }\n // Storage buffers\n for (const storageBuffer of this._storageBuffers) {\n storageBuffer._rebuild();\n }\n }\n _initGLContext() {\n var _a;\n // Caps\n this._caps = {\n maxTexturesImageUnits: this._gl.getParameter(this._gl.MAX_TEXTURE_IMAGE_UNITS),\n maxCombinedTexturesImageUnits: this._gl.getParameter(this._gl.MAX_COMBINED_TEXTURE_IMAGE_UNITS),\n maxVertexTextureImageUnits: this._gl.getParameter(this._gl.MAX_VERTEX_TEXTURE_IMAGE_UNITS),\n maxTextureSize: this._gl.getParameter(this._gl.MAX_TEXTURE_SIZE),\n maxSamples: this._webGLVersion > 1 ? this._gl.getParameter(this._gl.MAX_SAMPLES) : 1,\n maxCubemapTextureSize: this._gl.getParameter(this._gl.MAX_CUBE_MAP_TEXTURE_SIZE),\n maxRenderTextureSize: this._gl.getParameter(this._gl.MAX_RENDERBUFFER_SIZE),\n maxVertexAttribs: this._gl.getParameter(this._gl.MAX_VERTEX_ATTRIBS),\n maxVaryingVectors: this._gl.getParameter(this._gl.MAX_VARYING_VECTORS),\n maxFragmentUniformVectors: this._gl.getParameter(this._gl.MAX_FRAGMENT_UNIFORM_VECTORS),\n maxVertexUniformVectors: this._gl.getParameter(this._gl.MAX_VERTEX_UNIFORM_VECTORS),\n parallelShaderCompile: this._gl.getExtension(\"KHR_parallel_shader_compile\") || undefined,\n standardDerivatives: this._webGLVersion > 1 || this._gl.getExtension(\"OES_standard_derivatives\") !== null,\n maxAnisotropy: 1,\n astc: this._gl.getExtension(\"WEBGL_compressed_texture_astc\") || this._gl.getExtension(\"WEBKIT_WEBGL_compressed_texture_astc\"),\n bptc: this._gl.getExtension(\"EXT_texture_compression_bptc\") || this._gl.getExtension(\"WEBKIT_EXT_texture_compression_bptc\"),\n s3tc: this._gl.getExtension(\"WEBGL_compressed_texture_s3tc\") || this._gl.getExtension(\"WEBKIT_WEBGL_compressed_texture_s3tc\"),\n // eslint-disable-next-line @typescript-eslint/naming-convention\n s3tc_srgb: this._gl.getExtension(\"WEBGL_compressed_texture_s3tc_srgb\") || this._gl.getExtension(\"WEBKIT_WEBGL_compressed_texture_s3tc_srgb\"),\n pvrtc: this._gl.getExtension(\"WEBGL_compressed_texture_pvrtc\") || this._gl.getExtension(\"WEBKIT_WEBGL_compressed_texture_pvrtc\"),\n etc1: this._gl.getExtension(\"WEBGL_compressed_texture_etc1\") || this._gl.getExtension(\"WEBKIT_WEBGL_compressed_texture_etc1\"),\n etc2: this._gl.getExtension(\"WEBGL_compressed_texture_etc\") ||\n this._gl.getExtension(\"WEBKIT_WEBGL_compressed_texture_etc\") ||\n this._gl.getExtension(\"WEBGL_compressed_texture_es3_0\"),\n textureAnisotropicFilterExtension: this._gl.getExtension(\"EXT_texture_filter_anisotropic\") ||\n this._gl.getExtension(\"WEBKIT_EXT_texture_filter_anisotropic\") ||\n this._gl.getExtension(\"MOZ_EXT_texture_filter_anisotropic\"),\n uintIndices: this._webGLVersion > 1 || this._gl.getExtension(\"OES_element_index_uint\") !== null,\n fragmentDepthSupported: this._webGLVersion > 1 || this._gl.getExtension(\"EXT_frag_depth\") !== null,\n highPrecisionShaderSupported: false,\n timerQuery: this._gl.getExtension(\"EXT_disjoint_timer_query_webgl2\") || this._gl.getExtension(\"EXT_disjoint_timer_query\"),\n supportOcclusionQuery: this._webGLVersion > 1,\n canUseTimestampForTimerQuery: false,\n drawBuffersExtension: false,\n maxMSAASamples: 1,\n colorBufferFloat: !!(this._webGLVersion > 1 && this._gl.getExtension(\"EXT_color_buffer_float\")),\n colorBufferHalfFloat: !!(this._webGLVersion > 1 && this._gl.getExtension(\"EXT_color_buffer_half_float\")),\n textureFloat: this._webGLVersion > 1 || this._gl.getExtension(\"OES_texture_float\") ? true : false,\n textureHalfFloat: this._webGLVersion > 1 || this._gl.getExtension(\"OES_texture_half_float\") ? true : false,\n textureHalfFloatRender: false,\n textureFloatLinearFiltering: false,\n textureFloatRender: false,\n textureHalfFloatLinearFiltering: false,\n vertexArrayObject: false,\n instancedArrays: false,\n textureLOD: this._webGLVersion > 1 || this._gl.getExtension(\"EXT_shader_texture_lod\") ? true : false,\n texelFetch: this._webGLVersion !== 1,\n blendMinMax: false,\n multiview: this._gl.getExtension(\"OVR_multiview2\"),\n oculusMultiview: this._gl.getExtension(\"OCULUS_multiview\"),\n depthTextureExtension: false,\n canUseGLInstanceID: this._webGLVersion > 1,\n canUseGLVertexID: this._webGLVersion > 1,\n supportComputeShaders: false,\n supportSRGBBuffers: false,\n supportTransformFeedbacks: this._webGLVersion > 1,\n textureMaxLevel: this._webGLVersion > 1,\n texture2DArrayMaxLayerCount: this._webGLVersion > 1 ? this._gl.getParameter(this._gl.MAX_ARRAY_TEXTURE_LAYERS) : 128,\n disableMorphTargetTexture: false,\n };\n // Infos\n this._glVersion = this._gl.getParameter(this._gl.VERSION);\n const rendererInfo = this._gl.getExtension(\"WEBGL_debug_renderer_info\");\n if (rendererInfo != null) {\n this._glRenderer = this._gl.getParameter(rendererInfo.UNMASKED_RENDERER_WEBGL);\n this._glVendor = this._gl.getParameter(rendererInfo.UNMASKED_VENDOR_WEBGL);\n }\n if (!this._glVendor) {\n this._glVendor = this._gl.getParameter(this._gl.VENDOR) || \"Unknown vendor\";\n }\n if (!this._glRenderer) {\n this._glRenderer = this._gl.getParameter(this._gl.RENDERER) || \"Unknown renderer\";\n }\n // Constants\n if (this._gl.HALF_FLOAT_OES !== 0x8d61) {\n this._gl.HALF_FLOAT_OES = 0x8d61; // Half floating-point type (16-bit).\n }\n if (this._gl.RGBA16F !== 0x881a) {\n this._gl.RGBA16F = 0x881a; // RGBA 16-bit floating-point color-renderable internal sized format.\n }\n if (this._gl.RGBA32F !== 0x8814) {\n this._gl.RGBA32F = 0x8814; // RGBA 32-bit floating-point color-renderable internal sized format.\n }\n if (this._gl.DEPTH24_STENCIL8 !== 35056) {\n this._gl.DEPTH24_STENCIL8 = 35056;\n }\n // Extensions\n if (this._caps.timerQuery) {\n if (this._webGLVersion === 1) {\n this._gl.getQuery = this._caps.timerQuery.getQueryEXT.bind(this._caps.timerQuery);\n }\n // WebGLQuery casted to number to avoid TS error\n this._caps.canUseTimestampForTimerQuery = ((_a = this._gl.getQuery(this._caps.timerQuery.TIMESTAMP_EXT, this._caps.timerQuery.QUERY_COUNTER_BITS_EXT)) !== null && _a !== void 0 ? _a : 0) > 0;\n }\n this._caps.maxAnisotropy = this._caps.textureAnisotropicFilterExtension\n ? this._gl.getParameter(this._caps.textureAnisotropicFilterExtension.MAX_TEXTURE_MAX_ANISOTROPY_EXT)\n : 0;\n this._caps.textureFloatLinearFiltering = this._caps.textureFloat && this._gl.getExtension(\"OES_texture_float_linear\") ? true : false;\n this._caps.textureFloatRender = this._caps.textureFloat && this._canRenderToFloatFramebuffer() ? true : false;\n this._caps.textureHalfFloatLinearFiltering =\n this._webGLVersion > 1 || (this._caps.textureHalfFloat && this._gl.getExtension(\"OES_texture_half_float_linear\")) ? true : false;\n // Compressed formats\n if (this._caps.astc) {\n this._gl.COMPRESSED_SRGB8_ALPHA8_ASTC_4x4_KHR = this._caps.astc.COMPRESSED_SRGB8_ALPHA8_ASTC_4x4_KHR;\n }\n if (this._caps.bptc) {\n this._gl.COMPRESSED_SRGB_ALPHA_BPTC_UNORM_EXT = this._caps.bptc.COMPRESSED_SRGB_ALPHA_BPTC_UNORM_EXT;\n }\n if (this._caps.s3tc_srgb) {\n this._gl.COMPRESSED_SRGB_S3TC_DXT1_EXT = this._caps.s3tc_srgb.COMPRESSED_SRGB_S3TC_DXT1_EXT;\n this._gl.COMPRESSED_SRGB_ALPHA_S3TC_DXT1_EXT = this._caps.s3tc_srgb.COMPRESSED_SRGB_ALPHA_S3TC_DXT1_EXT;\n this._gl.COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT = this._caps.s3tc_srgb.COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT;\n }\n if (this._caps.etc2) {\n this._gl.COMPRESSED_SRGB8_ETC2 = this._caps.etc2.COMPRESSED_SRGB8_ETC2;\n this._gl.COMPRESSED_SRGB8_ALPHA8_ETC2_EAC = this._caps.etc2.COMPRESSED_SRGB8_ALPHA8_ETC2_EAC;\n }\n // Checks if some of the format renders first to allow the use of webgl inspector.\n if (this._webGLVersion > 1) {\n if (this._gl.HALF_FLOAT_OES !== 0x140b) {\n this._gl.HALF_FLOAT_OES = 0x140b;\n }\n }\n this._caps.textureHalfFloatRender = this._caps.textureHalfFloat && this._canRenderToHalfFloatFramebuffer();\n // Draw buffers\n if (this._webGLVersion > 1) {\n this._caps.drawBuffersExtension = true;\n this._caps.maxMSAASamples = this._maxMSAASamplesOverride !== null ? this._maxMSAASamplesOverride : this._gl.getParameter(this._gl.MAX_SAMPLES);\n }\n else {\n const drawBuffersExtension = this._gl.getExtension(\"WEBGL_draw_buffers\");\n if (drawBuffersExtension !== null) {\n this._caps.drawBuffersExtension = true;\n this._gl.drawBuffers = drawBuffersExtension.drawBuffersWEBGL.bind(drawBuffersExtension);\n this._gl.DRAW_FRAMEBUFFER = this._gl.FRAMEBUFFER;\n for (let i = 0; i < 16; i++) {\n this._gl[\"COLOR_ATTACHMENT\" + i + \"_WEBGL\"] = drawBuffersExtension[\"COLOR_ATTACHMENT\" + i + \"_WEBGL\"];\n }\n }\n }\n // Depth Texture\n if (this._webGLVersion > 1) {\n this._caps.depthTextureExtension = true;\n }\n else {\n const depthTextureExtension = this._gl.getExtension(\"WEBGL_depth_texture\");\n if (depthTextureExtension != null) {\n this._caps.depthTextureExtension = true;\n this._gl.UNSIGNED_INT_24_8 = depthTextureExtension.UNSIGNED_INT_24_8_WEBGL;\n }\n }\n // Vertex array object\n if (this.disableVertexArrayObjects) {\n this._caps.vertexArrayObject = false;\n }\n else if (this._webGLVersion > 1) {\n this._caps.vertexArrayObject = true;\n }\n else {\n const vertexArrayObjectExtension = this._gl.getExtension(\"OES_vertex_array_object\");\n if (vertexArrayObjectExtension != null) {\n this._caps.vertexArrayObject = true;\n this._gl.createVertexArray = vertexArrayObjectExtension.createVertexArrayOES.bind(vertexArrayObjectExtension);\n this._gl.bindVertexArray = vertexArrayObjectExtension.bindVertexArrayOES.bind(vertexArrayObjectExtension);\n this._gl.deleteVertexArray = vertexArrayObjectExtension.deleteVertexArrayOES.bind(vertexArrayObjectExtension);\n }\n }\n // Instances count\n if (this._webGLVersion > 1) {\n this._caps.instancedArrays = true;\n }\n else {\n const instanceExtension = this._gl.getExtension(\"ANGLE_instanced_arrays\");\n if (instanceExtension != null) {\n this._caps.instancedArrays = true;\n this._gl.drawArraysInstanced = instanceExtension.drawArraysInstancedANGLE.bind(instanceExtension);\n this._gl.drawElementsInstanced = instanceExtension.drawElementsInstancedANGLE.bind(instanceExtension);\n this._gl.vertexAttribDivisor = instanceExtension.vertexAttribDivisorANGLE.bind(instanceExtension);\n }\n else {\n this._caps.instancedArrays = false;\n }\n }\n if (this._gl.getShaderPrecisionFormat) {\n const vertexhighp = this._gl.getShaderPrecisionFormat(this._gl.VERTEX_SHADER, this._gl.HIGH_FLOAT);\n const fragmenthighp = this._gl.getShaderPrecisionFormat(this._gl.FRAGMENT_SHADER, this._gl.HIGH_FLOAT);\n if (vertexhighp && fragmenthighp) {\n this._caps.highPrecisionShaderSupported = vertexhighp.precision !== 0 && fragmenthighp.precision !== 0;\n }\n }\n if (this._webGLVersion > 1) {\n this._caps.blendMinMax = true;\n }\n else {\n const blendMinMaxExtension = this._gl.getExtension(\"EXT_blend_minmax\");\n if (blendMinMaxExtension != null) {\n this._caps.blendMinMax = true;\n this._gl.MAX = blendMinMaxExtension.MAX_EXT;\n this._gl.MIN = blendMinMaxExtension.MIN_EXT;\n }\n }\n // sRGB buffers\n // only run this if not already set to true (in the constructor, for example)\n if (!this._caps.supportSRGBBuffers) {\n if (this._webGLVersion > 1) {\n this._caps.supportSRGBBuffers = true;\n this._glSRGBExtensionValues = {\n SRGB: WebGL2RenderingContext.SRGB,\n SRGB8: WebGL2RenderingContext.SRGB8,\n SRGB8_ALPHA8: WebGL2RenderingContext.SRGB8_ALPHA8,\n };\n }\n else {\n const sRGBExtension = this._gl.getExtension(\"EXT_sRGB\");\n if (sRGBExtension != null) {\n this._caps.supportSRGBBuffers = true;\n this._glSRGBExtensionValues = {\n SRGB: sRGBExtension.SRGB_EXT,\n SRGB8: sRGBExtension.SRGB_ALPHA_EXT,\n SRGB8_ALPHA8: sRGBExtension.SRGB_ALPHA_EXT,\n };\n }\n }\n // take into account the forced state that was provided in options\n // When the issue in angle/chrome is fixed the flag should be taken into account only when it is explicitly defined\n this._caps.supportSRGBBuffers = this._caps.supportSRGBBuffers && !!(this._creationOptions && this._creationOptions.forceSRGBBufferSupportState);\n }\n // Depth buffer\n this._depthCullingState.depthTest = true;\n this._depthCullingState.depthFunc = this._gl.LEQUAL;\n this._depthCullingState.depthMask = true;\n // Texture maps\n this._maxSimultaneousTextures = this._caps.maxCombinedTexturesImageUnits;\n for (let slot = 0; slot < this._maxSimultaneousTextures; slot++) {\n this._nextFreeTextureSlots.push(slot);\n }\n if (this._glRenderer === \"Mali-G72\") {\n // Overcome a bug when using a texture to store morph targets on Mali-G72\n this._caps.disableMorphTargetTexture = true;\n }\n }\n _initFeatures() {\n this._features = {\n forceBitmapOverHTMLImageElement: false,\n supportRenderAndCopyToLodForFloatTextures: this._webGLVersion !== 1,\n supportDepthStencilTexture: this._webGLVersion !== 1,\n supportShadowSamplers: this._webGLVersion !== 1,\n uniformBufferHardCheckMatrix: false,\n allowTexturePrefiltering: this._webGLVersion !== 1,\n trackUbosInFrame: false,\n checkUbosContentBeforeUpload: false,\n supportCSM: this._webGLVersion !== 1,\n basisNeedsPOT: this._webGLVersion === 1,\n support3DTextures: this._webGLVersion !== 1,\n needTypeSuffixInShaderConstants: this._webGLVersion !== 1,\n supportMSAA: this._webGLVersion !== 1,\n supportSSAO2: this._webGLVersion !== 1,\n supportExtendedTextureFormats: this._webGLVersion !== 1,\n supportSwitchCaseInShader: this._webGLVersion !== 1,\n supportSyncTextureRead: true,\n needsInvertingBitmap: true,\n useUBOBindingCache: true,\n needShaderCodeInlining: false,\n needToAlwaysBindUniformBuffers: false,\n supportRenderPasses: false,\n supportSpriteInstancing: true,\n _collectUbosUpdatedInFrame: false,\n };\n }\n /**\n * Gets version of the current webGL context\n * Keep it for back compat - use version instead\n */\n get webGLVersion() {\n return this._webGLVersion;\n }\n /**\n * Gets a string identifying the name of the class\n * @returns \"Engine\" string\n */\n getClassName() {\n return \"ThinEngine\";\n }\n /**\n * Returns true if the stencil buffer has been enabled through the creation option of the context.\n */\n get isStencilEnable() {\n return this._isStencilEnable;\n }\n /** @internal */\n _prepareWorkingCanvas() {\n if (this._workingCanvas) {\n return;\n }\n this._workingCanvas = this.createCanvas(1, 1);\n const context = this._workingCanvas.getContext(\"2d\");\n if (context) {\n this._workingContext = context;\n }\n }\n /**\n * Reset the texture cache to empty state\n */\n resetTextureCache() {\n for (const key in this._boundTexturesCache) {\n if (!Object.prototype.hasOwnProperty.call(this._boundTexturesCache, key)) {\n continue;\n }\n this._boundTexturesCache[key] = null;\n }\n this._currentTextureChannel = -1;\n }\n /**\n * Gets an object containing information about the current engine context\n * @returns an object containing the vendor, the renderer and the version of the current engine context\n */\n getInfo() {\n return this.getGlInfo();\n }\n /**\n * Gets an object containing information about the current webGL context\n * @returns an object containing the vendor, the renderer and the version of the current webGL context\n */\n getGlInfo() {\n return {\n vendor: this._glVendor,\n renderer: this._glRenderer,\n version: this._glVersion,\n };\n }\n /**\n * Defines the hardware scaling level.\n * By default the hardware scaling level is computed from the window device ratio.\n * if level = 1 then the engine will render at the exact resolution of the canvas. If level = 0.5 then the engine will render at twice the size of the canvas.\n * @param level defines the level to use\n */\n setHardwareScalingLevel(level) {\n this._hardwareScalingLevel = level;\n this.resize();\n }\n /**\n * Gets the current hardware scaling level.\n * By default the hardware scaling level is computed from the window device ratio.\n * if level = 1 then the engine will render at the exact resolution of the canvas. If level = 0.5 then the engine will render at twice the size of the canvas.\n * @returns a number indicating the current hardware scaling level\n */\n getHardwareScalingLevel() {\n return this._hardwareScalingLevel;\n }\n /**\n * Gets the list of loaded textures\n * @returns an array containing all loaded textures\n */\n getLoadedTexturesCache() {\n return this._internalTexturesCache;\n }\n /**\n * Gets the object containing all engine capabilities\n * @returns the EngineCapabilities object\n */\n getCaps() {\n return this._caps;\n }\n /**\n * stop executing a render loop function and remove it from the execution array\n * @param renderFunction defines the function to be removed. If not provided all functions will be removed.\n */\n stopRenderLoop(renderFunction) {\n if (!renderFunction) {\n this._activeRenderLoops.length = 0;\n this._cancelFrame();\n return;\n }\n const index = this._activeRenderLoops.indexOf(renderFunction);\n if (index >= 0) {\n this._activeRenderLoops.splice(index, 1);\n if (this._activeRenderLoops.length == 0) {\n this._cancelFrame();\n }\n }\n }\n _cancelFrame() {\n if (this._renderingQueueLaunched && this._frameHandler) {\n this._renderingQueueLaunched = false;\n if (!IsWindowObjectExist()) {\n if (typeof cancelAnimationFrame === \"function\") {\n return cancelAnimationFrame(this._frameHandler);\n }\n }\n else {\n const { cancelAnimationFrame } = this.getHostWindow() || window;\n if (typeof cancelAnimationFrame === \"function\") {\n return cancelAnimationFrame(this._frameHandler);\n }\n }\n return clearTimeout(this._frameHandler);\n }\n }\n /** @internal */\n _renderLoop() {\n if (!this._contextWasLost) {\n let shouldRender = true;\n if (this._isDisposed || (!this.renderEvenInBackground && this._windowIsBackground)) {\n shouldRender = false;\n }\n if (shouldRender) {\n // Start new frame\n this.beginFrame();\n for (let index = 0; index < this._activeRenderLoops.length; index++) {\n const renderFunction = this._activeRenderLoops[index];\n renderFunction();\n }\n // Present\n this.endFrame();\n }\n }\n if (this._activeRenderLoops.length > 0) {\n this._frameHandler = this._queueNewFrame(this._boundRenderFunction, this.getHostWindow());\n }\n else {\n this._renderingQueueLaunched = false;\n }\n }\n /**\n * Gets the HTML canvas attached with the current webGL context\n * @returns a HTML canvas\n */\n getRenderingCanvas() {\n return this._renderingCanvas;\n }\n /**\n * Gets the audio context specified in engine initialization options\n * @returns an Audio Context\n */\n getAudioContext() {\n return this._audioContext;\n }\n /**\n * Gets the audio destination specified in engine initialization options\n * @returns an audio destination node\n */\n getAudioDestination() {\n return this._audioDestination;\n }\n /**\n * Gets host window\n * @returns the host window object\n */\n getHostWindow() {\n if (!IsWindowObjectExist()) {\n return null;\n }\n if (this._renderingCanvas && this._renderingCanvas.ownerDocument && this._renderingCanvas.ownerDocument.defaultView) {\n return this._renderingCanvas.ownerDocument.defaultView;\n }\n return window;\n }\n /**\n * Gets the current render width\n * @param useScreen defines if screen size must be used (or the current render target if any)\n * @returns a number defining the current render width\n */\n getRenderWidth(useScreen = false) {\n if (!useScreen && this._currentRenderTarget) {\n return this._currentRenderTarget.width;\n }\n return this._framebufferDimensionsObject ? this._framebufferDimensionsObject.framebufferWidth : this._gl.drawingBufferWidth;\n }\n /**\n * Gets the current render height\n * @param useScreen defines if screen size must be used (or the current render target if any)\n * @returns a number defining the current render height\n */\n getRenderHeight(useScreen = false) {\n if (!useScreen && this._currentRenderTarget) {\n return this._currentRenderTarget.height;\n }\n return this._framebufferDimensionsObject ? this._framebufferDimensionsObject.framebufferHeight : this._gl.drawingBufferHeight;\n }\n /**\n * Can be used to override the current requestAnimationFrame requester.\n * @internal\n */\n _queueNewFrame(bindedRenderFunction, requester) {\n return ThinEngine.QueueNewFrame(bindedRenderFunction, requester);\n }\n /**\n * Register and execute a render loop. The engine can have more than one render function\n * @param renderFunction defines the function to continuously execute\n */\n runRenderLoop(renderFunction) {\n if (this._activeRenderLoops.indexOf(renderFunction) !== -1) {\n return;\n }\n this._activeRenderLoops.push(renderFunction);\n if (!this._renderingQueueLaunched) {\n this._renderingQueueLaunched = true;\n this._boundRenderFunction = this._renderLoop.bind(this);\n this._frameHandler = this._queueNewFrame(this._boundRenderFunction, this.getHostWindow());\n }\n }\n /**\n * Clear the current render buffer or the current render target (if any is set up)\n * @param color defines the color to use\n * @param backBuffer defines if the back buffer must be cleared\n * @param depth defines if the depth buffer must be cleared\n * @param stencil defines if the stencil buffer must be cleared\n */\n clear(color, backBuffer, depth, stencil = false) {\n var _a, _b;\n const useStencilGlobalOnly = this.stencilStateComposer.useStencilGlobalOnly;\n this.stencilStateComposer.useStencilGlobalOnly = true; // make sure the stencil mask is coming from the global stencil and not from a material (effect) which would currently be in effect\n this.applyStates();\n this.stencilStateComposer.useStencilGlobalOnly = useStencilGlobalOnly;\n let mode = 0;\n if (backBuffer && color) {\n let setBackBufferColor = true;\n if (this._currentRenderTarget) {\n const textureFormat = (_a = this._currentRenderTarget.texture) === null || _a === void 0 ? void 0 : _a.format;\n if (textureFormat === 8 ||\n textureFormat === 9 ||\n textureFormat === 10 ||\n textureFormat === 11) {\n const textureType = (_b = this._currentRenderTarget.texture) === null || _b === void 0 ? void 0 : _b.type;\n if (textureType === 7 || textureType === 5) {\n ThinEngine._TempClearColorUint32[0] = color.r * 255;\n ThinEngine._TempClearColorUint32[1] = color.g * 255;\n ThinEngine._TempClearColorUint32[2] = color.b * 255;\n ThinEngine._TempClearColorUint32[3] = color.a * 255;\n this._gl.clearBufferuiv(this._gl.COLOR, 0, ThinEngine._TempClearColorUint32);\n setBackBufferColor = false;\n }\n else {\n ThinEngine._TempClearColorInt32[0] = color.r * 255;\n ThinEngine._TempClearColorInt32[1] = color.g * 255;\n ThinEngine._TempClearColorInt32[2] = color.b * 255;\n ThinEngine._TempClearColorInt32[3] = color.a * 255;\n this._gl.clearBufferiv(this._gl.COLOR, 0, ThinEngine._TempClearColorInt32);\n setBackBufferColor = false;\n }\n }\n }\n if (setBackBufferColor) {\n this._gl.clearColor(color.r, color.g, color.b, color.a !== undefined ? color.a : 1.0);\n mode |= this._gl.COLOR_BUFFER_BIT;\n }\n }\n if (depth) {\n if (this.useReverseDepthBuffer) {\n this._depthCullingState.depthFunc = this._gl.GEQUAL;\n this._gl.clearDepth(0.0);\n }\n else {\n this._gl.clearDepth(1.0);\n }\n mode |= this._gl.DEPTH_BUFFER_BIT;\n }\n if (stencil) {\n this._gl.clearStencil(0);\n mode |= this._gl.STENCIL_BUFFER_BIT;\n }\n this._gl.clear(mode);\n }\n /**\n * @internal\n */\n _viewport(x, y, width, height) {\n if (x !== this._viewportCached.x || y !== this._viewportCached.y || width !== this._viewportCached.z || height !== this._viewportCached.w) {\n this._viewportCached.x = x;\n this._viewportCached.y = y;\n this._viewportCached.z = width;\n this._viewportCached.w = height;\n this._gl.viewport(x, y, width, height);\n }\n }\n /**\n * Set the WebGL's viewport\n * @param viewport defines the viewport element to be used\n * @param requiredWidth defines the width required for rendering. If not provided the rendering canvas' width is used\n * @param requiredHeight defines the height required for rendering. If not provided the rendering canvas' height is used\n */\n setViewport(viewport, requiredWidth, requiredHeight) {\n const width = requiredWidth || this.getRenderWidth();\n const height = requiredHeight || this.getRenderHeight();\n const x = viewport.x || 0;\n const y = viewport.y || 0;\n this._cachedViewport = viewport;\n this._viewport(x * width, y * height, width * viewport.width, height * viewport.height);\n }\n /**\n * Begin a new frame\n */\n beginFrame() { }\n /**\n * Enf the current frame\n */\n endFrame() {\n // Force a flush in case we are using a bad OS.\n if (this._badOS) {\n this.flushFramebuffer();\n }\n this._frameId++;\n }\n /**\n * Resize the view according to the canvas' size\n * @param forceSetSize true to force setting the sizes of the underlying canvas\n */\n resize(forceSetSize = false) {\n let width;\n let height;\n // Re-query hardware scaling level to handle zoomed-in resizing.\n if (this.adaptToDeviceRatio) {\n const devicePixelRatio = IsWindowObjectExist() ? window.devicePixelRatio || 1.0 : 1.0;\n const changeRatio = this._lastDevicePixelRatio / devicePixelRatio;\n this._lastDevicePixelRatio = devicePixelRatio;\n this._hardwareScalingLevel *= changeRatio;\n }\n if (IsWindowObjectExist() && IsDocumentAvailable()) {\n // make sure it is a Node object, and is a part of the document.\n if (this._renderingCanvas) {\n const boundingRect = this._renderingCanvas.getBoundingClientRect\n ? this._renderingCanvas.getBoundingClientRect()\n : {\n // fallback to last solution in case the function doesn't exist\n width: this._renderingCanvas.width * this._hardwareScalingLevel,\n height: this._renderingCanvas.height * this._hardwareScalingLevel,\n };\n width = this._renderingCanvas.clientWidth || boundingRect.width || this._renderingCanvas.width || 100;\n height = this._renderingCanvas.clientHeight || boundingRect.height || this._renderingCanvas.height || 100;\n }\n else {\n width = window.innerWidth;\n height = window.innerHeight;\n }\n }\n else {\n width = this._renderingCanvas ? this._renderingCanvas.width : 100;\n height = this._renderingCanvas ? this._renderingCanvas.height : 100;\n }\n this.setSize(width / this._hardwareScalingLevel, height / this._hardwareScalingLevel, forceSetSize);\n }\n /**\n * Force a specific size of the canvas\n * @param width defines the new canvas' width\n * @param height defines the new canvas' height\n * @param forceSetSize true to force setting the sizes of the underlying canvas\n * @returns true if the size was changed\n */\n setSize(width, height, forceSetSize = false) {\n if (!this._renderingCanvas) {\n return false;\n }\n width = width | 0;\n height = height | 0;\n if (!forceSetSize && this._renderingCanvas.width === width && this._renderingCanvas.height === height) {\n return false;\n }\n this._renderingCanvas.width = width;\n this._renderingCanvas.height = height;\n return true;\n }\n /**\n * Binds the frame buffer to the specified texture.\n * @param rtWrapper The render target wrapper to render to\n * @param faceIndex The face of the texture to render to in case of cube texture and if the render target wrapper is not a multi render target\n * @param requiredWidth The width of the target to render to\n * @param requiredHeight The height of the target to render to\n * @param forceFullscreenViewport Forces the viewport to be the entire texture/screen if true\n * @param lodLevel Defines the lod level to bind to the frame buffer\n * @param layer Defines the 2d array index to bind to the frame buffer if the render target wrapper is not a multi render target\n */\n bindFramebuffer(rtWrapper, faceIndex = 0, requiredWidth, requiredHeight, forceFullscreenViewport, lodLevel = 0, layer = 0) {\n var _a, _b, _c, _d, _e;\n const webglRTWrapper = rtWrapper;\n if (this._currentRenderTarget) {\n this.unBindFramebuffer(this._currentRenderTarget);\n }\n this._currentRenderTarget = rtWrapper;\n this._bindUnboundFramebuffer(webglRTWrapper._MSAAFramebuffer ? webglRTWrapper._MSAAFramebuffer : webglRTWrapper._framebuffer);\n const gl = this._gl;\n if (!rtWrapper.isMulti) {\n if (rtWrapper.is2DArray) {\n gl.framebufferTextureLayer(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, (_a = rtWrapper.texture._hardwareTexture) === null || _a === void 0 ? void 0 : _a.underlyingResource, lodLevel, layer);\n }\n else if (rtWrapper.isCube) {\n gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex, (_b = rtWrapper.texture._hardwareTexture) === null || _b === void 0 ? void 0 : _b.underlyingResource, lodLevel);\n }\n }\n const depthStencilTexture = rtWrapper._depthStencilTexture;\n if (depthStencilTexture) {\n const attachment = rtWrapper._depthStencilTextureWithStencil ? gl.DEPTH_STENCIL_ATTACHMENT : gl.DEPTH_ATTACHMENT;\n if (rtWrapper.is2DArray) {\n gl.framebufferTextureLayer(gl.FRAMEBUFFER, attachment, (_c = depthStencilTexture._hardwareTexture) === null || _c === void 0 ? void 0 : _c.underlyingResource, lodLevel, layer);\n }\n else if (rtWrapper.isCube) {\n gl.framebufferTexture2D(gl.FRAMEBUFFER, attachment, gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex, (_d = depthStencilTexture._hardwareTexture) === null || _d === void 0 ? void 0 : _d.underlyingResource, lodLevel);\n }\n else {\n gl.framebufferTexture2D(gl.FRAMEBUFFER, attachment, gl.TEXTURE_2D, (_e = depthStencilTexture._hardwareTexture) === null || _e === void 0 ? void 0 : _e.underlyingResource, lodLevel);\n }\n }\n if (this._cachedViewport && !forceFullscreenViewport) {\n this.setViewport(this._cachedViewport, requiredWidth, requiredHeight);\n }\n else {\n if (!requiredWidth) {\n requiredWidth = rtWrapper.width;\n if (lodLevel) {\n requiredWidth = requiredWidth / Math.pow(2, lodLevel);\n }\n }\n if (!requiredHeight) {\n requiredHeight = rtWrapper.height;\n if (lodLevel) {\n requiredHeight = requiredHeight / Math.pow(2, lodLevel);\n }\n }\n this._viewport(0, 0, requiredWidth, requiredHeight);\n }\n this.wipeCaches();\n }\n /**\n * Set various states to the webGL context\n * @param culling defines culling state: true to enable culling, false to disable it\n * @param zOffset defines the value to apply to zOffset (0 by default)\n * @param force defines if states must be applied even if cache is up to date\n * @param reverseSide defines if culling must be reversed (CCW if false, CW if true)\n * @param cullBackFaces true to cull back faces, false to cull front faces (if culling is enabled)\n * @param stencil stencil states to set\n * @param zOffsetUnits defines the value to apply to zOffsetUnits (0 by default)\n */\n setState(culling, zOffset = 0, force, reverseSide = false, cullBackFaces, stencil, zOffsetUnits = 0) {\n var _a, _b;\n // Culling\n if (this._depthCullingState.cull !== culling || force) {\n this._depthCullingState.cull = culling;\n }\n // Cull face\n const cullFace = ((_b = (_a = this.cullBackFaces) !== null && _a !== void 0 ? _a : cullBackFaces) !== null && _b !== void 0 ? _b : true) ? this._gl.BACK : this._gl.FRONT;\n if (this._depthCullingState.cullFace !== cullFace || force) {\n this._depthCullingState.cullFace = cullFace;\n }\n // Z offset\n this.setZOffset(zOffset);\n this.setZOffsetUnits(zOffsetUnits);\n // Front face\n const frontFace = reverseSide ? this._gl.CW : this._gl.CCW;\n if (this._depthCullingState.frontFace !== frontFace || force) {\n this._depthCullingState.frontFace = frontFace;\n }\n this._stencilStateComposer.stencilMaterial = stencil;\n }\n /**\n * Gets a boolean indicating if depth testing is enabled\n * @returns the current state\n */\n getDepthBuffer() {\n return this._depthCullingState.depthTest;\n }\n /**\n * Enable or disable depth buffering\n * @param enable defines the state to set\n */\n setDepthBuffer(enable) {\n this._depthCullingState.depthTest = enable;\n }\n /**\n * Set the z offset Factor to apply to current rendering\n * @param value defines the offset to apply\n */\n setZOffset(value) {\n this._depthCullingState.zOffset = this.useReverseDepthBuffer ? -value : value;\n }\n /**\n * Gets the current value of the zOffset Factor\n * @returns the current zOffset Factor state\n */\n getZOffset() {\n const zOffset = this._depthCullingState.zOffset;\n return this.useReverseDepthBuffer ? -zOffset : zOffset;\n }\n /**\n * Set the z offset Units to apply to current rendering\n * @param value defines the offset to apply\n */\n setZOffsetUnits(value) {\n this._depthCullingState.zOffsetUnits = this.useReverseDepthBuffer ? -value : value;\n }\n /**\n * Gets the current value of the zOffset Units\n * @returns the current zOffset Units state\n */\n getZOffsetUnits() {\n const zOffsetUnits = this._depthCullingState.zOffsetUnits;\n return this.useReverseDepthBuffer ? -zOffsetUnits : zOffsetUnits;\n }\n /**\n * @internal\n */\n _bindUnboundFramebuffer(framebuffer) {\n if (this._currentFramebuffer !== framebuffer) {\n this._gl.bindFramebuffer(this._gl.FRAMEBUFFER, framebuffer);\n this._currentFramebuffer = framebuffer;\n }\n }\n /** @internal */\n _currentFrameBufferIsDefaultFrameBuffer() {\n return this._currentFramebuffer === null;\n }\n /**\n * Generates the mipmaps for a texture\n * @param texture texture to generate the mipmaps for\n */\n generateMipmaps(texture) {\n this._bindTextureDirectly(this._gl.TEXTURE_2D, texture, true);\n this._gl.generateMipmap(this._gl.TEXTURE_2D);\n this._bindTextureDirectly(this._gl.TEXTURE_2D, null);\n }\n /**\n * Unbind the current render target texture from the webGL context\n * @param texture defines the render target wrapper to unbind\n * @param disableGenerateMipMaps defines a boolean indicating that mipmaps must not be generated\n * @param onBeforeUnbind defines a function which will be called before the effective unbind\n */\n unBindFramebuffer(texture, disableGenerateMipMaps = false, onBeforeUnbind) {\n var _a;\n const webglRTWrapper = texture;\n this._currentRenderTarget = null;\n // If MSAA, we need to bitblt back to main texture\n const gl = this._gl;\n if (webglRTWrapper._MSAAFramebuffer) {\n if (texture.isMulti) {\n // This texture is part of a MRT texture, we need to treat all attachments\n this.unBindMultiColorAttachmentFramebuffer(texture, disableGenerateMipMaps, onBeforeUnbind);\n return;\n }\n gl.bindFramebuffer(gl.READ_FRAMEBUFFER, webglRTWrapper._MSAAFramebuffer);\n gl.bindFramebuffer(gl.DRAW_FRAMEBUFFER, webglRTWrapper._framebuffer);\n gl.blitFramebuffer(0, 0, texture.width, texture.height, 0, 0, texture.width, texture.height, gl.COLOR_BUFFER_BIT, gl.NEAREST);\n }\n if (((_a = texture.texture) === null || _a === void 0 ? void 0 : _a.generateMipMaps) && !disableGenerateMipMaps && !texture.isCube) {\n this.generateMipmaps(texture.texture);\n }\n if (onBeforeUnbind) {\n if (webglRTWrapper._MSAAFramebuffer) {\n // Bind the correct framebuffer\n this._bindUnboundFramebuffer(webglRTWrapper._framebuffer);\n }\n onBeforeUnbind();\n }\n this._bindUnboundFramebuffer(null);\n }\n /**\n * Force a webGL flush (ie. a flush of all waiting webGL commands)\n */\n flushFramebuffer() {\n this._gl.flush();\n }\n /**\n * Unbind the current render target and bind the default framebuffer\n */\n restoreDefaultFramebuffer() {\n if (this._currentRenderTarget) {\n this.unBindFramebuffer(this._currentRenderTarget);\n }\n else {\n this._bindUnboundFramebuffer(null);\n }\n if (this._cachedViewport) {\n this.setViewport(this._cachedViewport);\n }\n this.wipeCaches();\n }\n // VBOs\n /** @internal */\n _resetVertexBufferBinding() {\n this.bindArrayBuffer(null);\n this._cachedVertexBuffers = null;\n }\n /**\n * Creates a vertex buffer\n * @param data the data for the vertex buffer\n * @returns the new WebGL static buffer\n */\n createVertexBuffer(data) {\n return this._createVertexBuffer(data, this._gl.STATIC_DRAW);\n }\n _createVertexBuffer(data, usage) {\n const vbo = this._gl.createBuffer();\n if (!vbo) {\n throw new Error(\"Unable to create vertex buffer\");\n }\n const dataBuffer = new WebGLDataBuffer(vbo);\n this.bindArrayBuffer(dataBuffer);\n if (data instanceof Array) {\n this._gl.bufferData(this._gl.ARRAY_BUFFER, new Float32Array(data), usage);\n }\n else {\n this._gl.bufferData(this._gl.ARRAY_BUFFER, data, usage);\n }\n this._resetVertexBufferBinding();\n dataBuffer.references = 1;\n return dataBuffer;\n }\n /**\n * Creates a dynamic vertex buffer\n * @param data the data for the dynamic vertex buffer\n * @returns the new WebGL dynamic buffer\n */\n createDynamicVertexBuffer(data) {\n return this._createVertexBuffer(data, this._gl.DYNAMIC_DRAW);\n }\n _resetIndexBufferBinding() {\n this.bindIndexBuffer(null);\n this._cachedIndexBuffer = null;\n }\n /**\n * Creates a new index buffer\n * @param indices defines the content of the index buffer\n * @param updatable defines if the index buffer must be updatable\n * @returns a new webGL buffer\n */\n createIndexBuffer(indices, updatable) {\n const vbo = this._gl.createBuffer();\n const dataBuffer = new WebGLDataBuffer(vbo);\n if (!vbo) {\n throw new Error(\"Unable to create index buffer\");\n }\n this.bindIndexBuffer(dataBuffer);\n const data = this._normalizeIndexData(indices);\n this._gl.bufferData(this._gl.ELEMENT_ARRAY_BUFFER, data, updatable ? this._gl.DYNAMIC_DRAW : this._gl.STATIC_DRAW);\n this._resetIndexBufferBinding();\n dataBuffer.references = 1;\n dataBuffer.is32Bits = data.BYTES_PER_ELEMENT === 4;\n return dataBuffer;\n }\n _normalizeIndexData(indices) {\n const bytesPerElement = indices.BYTES_PER_ELEMENT;\n if (bytesPerElement === 2) {\n return indices;\n }\n // Check 32 bit support\n if (this._caps.uintIndices) {\n if (indices instanceof Uint32Array) {\n return indices;\n }\n else {\n // number[] or Int32Array, check if 32 bit is necessary\n for (let index = 0; index < indices.length; index++) {\n if (indices[index] >= 65535) {\n return new Uint32Array(indices);\n }\n }\n return new Uint16Array(indices);\n }\n }\n // No 32 bit support, force conversion to 16 bit (values greater 16 bit are lost)\n return new Uint16Array(indices);\n }\n /**\n * Bind a webGL buffer to the webGL context\n * @param buffer defines the buffer to bind\n */\n bindArrayBuffer(buffer) {\n if (!this._vaoRecordInProgress) {\n this._unbindVertexArrayObject();\n }\n this._bindBuffer(buffer, this._gl.ARRAY_BUFFER);\n }\n /**\n * Bind a specific block at a given index in a specific shader program\n * @param pipelineContext defines the pipeline context to use\n * @param blockName defines the block name\n * @param index defines the index where to bind the block\n */\n bindUniformBlock(pipelineContext, blockName, index) {\n const program = pipelineContext.program;\n const uniformLocation = this._gl.getUniformBlockIndex(program, blockName);\n this._gl.uniformBlockBinding(program, uniformLocation, index);\n }\n // eslint-disable-next-line @typescript-eslint/naming-convention\n bindIndexBuffer(buffer) {\n if (!this._vaoRecordInProgress) {\n this._unbindVertexArrayObject();\n }\n this._bindBuffer(buffer, this._gl.ELEMENT_ARRAY_BUFFER);\n }\n _bindBuffer(buffer, target) {\n if (this._vaoRecordInProgress || this._currentBoundBuffer[target] !== buffer) {\n this._gl.bindBuffer(target, buffer ? buffer.underlyingResource : null);\n this._currentBoundBuffer[target] = buffer;\n }\n }\n /**\n * update the bound buffer with the given data\n * @param data defines the data to update\n */\n updateArrayBuffer(data) {\n this._gl.bufferSubData(this._gl.ARRAY_BUFFER, 0, data);\n }\n _vertexAttribPointer(buffer, indx, size, type, normalized, stride, offset) {\n const pointer = this._currentBufferPointers[indx];\n if (!pointer) {\n return;\n }\n let changed = false;\n if (!pointer.active) {\n changed = true;\n pointer.active = true;\n pointer.index = indx;\n pointer.size = size;\n pointer.type = type;\n pointer.normalized = normalized;\n pointer.stride = stride;\n pointer.offset = offset;\n pointer.buffer = buffer;\n }\n else {\n if (pointer.buffer !== buffer) {\n pointer.buffer = buffer;\n changed = true;\n }\n if (pointer.size !== size) {\n pointer.size = size;\n changed = true;\n }\n if (pointer.type !== type) {\n pointer.type = type;\n changed = true;\n }\n if (pointer.normalized !== normalized) {\n pointer.normalized = normalized;\n changed = true;\n }\n if (pointer.stride !== stride) {\n pointer.stride = stride;\n changed = true;\n }\n if (pointer.offset !== offset) {\n pointer.offset = offset;\n changed = true;\n }\n }\n if (changed || this._vaoRecordInProgress) {\n this.bindArrayBuffer(buffer);\n if (type === this._gl.UNSIGNED_INT || type === this._gl.INT) {\n this._gl.vertexAttribIPointer(indx, size, type, stride, offset);\n }\n else {\n this._gl.vertexAttribPointer(indx, size, type, normalized, stride, offset);\n }\n }\n }\n /**\n * @internal\n */\n _bindIndexBufferWithCache(indexBuffer) {\n if (indexBuffer == null) {\n return;\n }\n if (this._cachedIndexBuffer !== indexBuffer) {\n this._cachedIndexBuffer = indexBuffer;\n this.bindIndexBuffer(indexBuffer);\n this._uintIndicesCurrentlySet = indexBuffer.is32Bits;\n }\n }\n _bindVertexBuffersAttributes(vertexBuffers, effect, overrideVertexBuffers) {\n const attributes = effect.getAttributesNames();\n if (!this._vaoRecordInProgress) {\n this._unbindVertexArrayObject();\n }\n this.unbindAllAttributes();\n for (let index = 0; index < attributes.length; index++) {\n const order = effect.getAttributeLocation(index);\n if (order >= 0) {\n const ai = attributes[index];\n let vertexBuffer = null;\n if (overrideVertexBuffers) {\n vertexBuffer = overrideVertexBuffers[ai];\n }\n if (!vertexBuffer) {\n vertexBuffer = vertexBuffers[ai];\n }\n if (!vertexBuffer) {\n continue;\n }\n this._gl.enableVertexAttribArray(order);\n if (!this._vaoRecordInProgress) {\n this._vertexAttribArraysEnabled[order] = true;\n }\n const buffer = vertexBuffer.getBuffer();\n if (buffer) {\n this._vertexAttribPointer(buffer, order, vertexBuffer.getSize(), vertexBuffer.type, vertexBuffer.normalized, vertexBuffer.byteStride, vertexBuffer.byteOffset);\n if (vertexBuffer.getIsInstanced()) {\n this._gl.vertexAttribDivisor(order, vertexBuffer.getInstanceDivisor());\n if (!this._vaoRecordInProgress) {\n this._currentInstanceLocations.push(order);\n this._currentInstanceBuffers.push(buffer);\n }\n }\n }\n }\n }\n }\n /**\n * Records a vertex array object\n * @see https://doc.babylonjs.com/setup/support/webGL2#vertex-array-objects\n * @param vertexBuffers defines the list of vertex buffers to store\n * @param indexBuffer defines the index buffer to store\n * @param effect defines the effect to store\n * @param overrideVertexBuffers defines optional list of avertex buffers that overrides the entries in vertexBuffers\n * @returns the new vertex array object\n */\n recordVertexArrayObject(vertexBuffers, indexBuffer, effect, overrideVertexBuffers) {\n const vao = this._gl.createVertexArray();\n if (!vao) {\n throw new Error(\"Unable to create VAO\");\n }\n this._vaoRecordInProgress = true;\n this._gl.bindVertexArray(vao);\n this._mustWipeVertexAttributes = true;\n this._bindVertexBuffersAttributes(vertexBuffers, effect, overrideVertexBuffers);\n this.bindIndexBuffer(indexBuffer);\n this._vaoRecordInProgress = false;\n this._gl.bindVertexArray(null);\n return vao;\n }\n /**\n * Bind a specific vertex array object\n * @see https://doc.babylonjs.com/setup/support/webGL2#vertex-array-objects\n * @param vertexArrayObject defines the vertex array object to bind\n * @param indexBuffer defines the index buffer to bind\n */\n bindVertexArrayObject(vertexArrayObject, indexBuffer) {\n if (this._cachedVertexArrayObject !== vertexArrayObject) {\n this._cachedVertexArrayObject = vertexArrayObject;\n this._gl.bindVertexArray(vertexArrayObject);\n this._cachedVertexBuffers = null;\n this._cachedIndexBuffer = null;\n this._uintIndicesCurrentlySet = indexBuffer != null && indexBuffer.is32Bits;\n this._mustWipeVertexAttributes = true;\n }\n }\n /**\n * Bind webGl buffers directly to the webGL context\n * @param vertexBuffer defines the vertex buffer to bind\n * @param indexBuffer defines the index buffer to bind\n * @param vertexDeclaration defines the vertex declaration to use with the vertex buffer\n * @param vertexStrideSize defines the vertex stride of the vertex buffer\n * @param effect defines the effect associated with the vertex buffer\n */\n bindBuffersDirectly(vertexBuffer, indexBuffer, vertexDeclaration, vertexStrideSize, effect) {\n if (this._cachedVertexBuffers !== vertexBuffer || this._cachedEffectForVertexBuffers !== effect) {\n this._cachedVertexBuffers = vertexBuffer;\n this._cachedEffectForVertexBuffers = effect;\n const attributesCount = effect.getAttributesCount();\n this._unbindVertexArrayObject();\n this.unbindAllAttributes();\n let offset = 0;\n for (let index = 0; index < attributesCount; index++) {\n if (index < vertexDeclaration.length) {\n const order = effect.getAttributeLocation(index);\n if (order >= 0) {\n this._gl.enableVertexAttribArray(order);\n this._vertexAttribArraysEnabled[order] = true;\n this._vertexAttribPointer(vertexBuffer, order, vertexDeclaration[index], this._gl.FLOAT, false, vertexStrideSize, offset);\n }\n offset += vertexDeclaration[index] * 4;\n }\n }\n }\n this._bindIndexBufferWithCache(indexBuffer);\n }\n _unbindVertexArrayObject() {\n if (!this._cachedVertexArrayObject) {\n return;\n }\n this._cachedVertexArrayObject = null;\n this._gl.bindVertexArray(null);\n }\n /**\n * Bind a list of vertex buffers to the webGL context\n * @param vertexBuffers defines the list of vertex buffers to bind\n * @param indexBuffer defines the index buffer to bind\n * @param effect defines the effect associated with the vertex buffers\n * @param overrideVertexBuffers defines optional list of avertex buffers that overrides the entries in vertexBuffers\n */\n bindBuffers(vertexBuffers, indexBuffer, effect, overrideVertexBuffers) {\n if (this._cachedVertexBuffers !== vertexBuffers || this._cachedEffectForVertexBuffers !== effect) {\n this._cachedVertexBuffers = vertexBuffers;\n this._cachedEffectForVertexBuffers = effect;\n this._bindVertexBuffersAttributes(vertexBuffers, effect, overrideVertexBuffers);\n }\n this._bindIndexBufferWithCache(indexBuffer);\n }\n /**\n * Unbind all instance attributes\n */\n unbindInstanceAttributes() {\n let boundBuffer;\n for (let i = 0, ul = this._currentInstanceLocations.length; i < ul; i++) {\n const instancesBuffer = this._currentInstanceBuffers[i];\n if (boundBuffer != instancesBuffer && instancesBuffer.references) {\n boundBuffer = instancesBuffer;\n this.bindArrayBuffer(instancesBuffer);\n }\n const offsetLocation = this._currentInstanceLocations[i];\n this._gl.vertexAttribDivisor(offsetLocation, 0);\n }\n this._currentInstanceBuffers.length = 0;\n this._currentInstanceLocations.length = 0;\n }\n /**\n * Release and free the memory of a vertex array object\n * @param vao defines the vertex array object to delete\n */\n releaseVertexArrayObject(vao) {\n this._gl.deleteVertexArray(vao);\n }\n /**\n * @internal\n */\n _releaseBuffer(buffer) {\n buffer.references--;\n if (buffer.references === 0) {\n this._deleteBuffer(buffer);\n return true;\n }\n return false;\n }\n _deleteBuffer(buffer) {\n this._gl.deleteBuffer(buffer.underlyingResource);\n }\n /**\n * Update the content of a webGL buffer used with instantiation and bind it to the webGL context\n * @param instancesBuffer defines the webGL buffer to update and bind\n * @param data defines the data to store in the buffer\n * @param offsetLocations defines the offsets or attributes information used to determine where data must be stored in the buffer\n */\n updateAndBindInstancesBuffer(instancesBuffer, data, offsetLocations) {\n this.bindArrayBuffer(instancesBuffer);\n if (data) {\n this._gl.bufferSubData(this._gl.ARRAY_BUFFER, 0, data);\n }\n if (offsetLocations[0].index !== undefined) {\n this.bindInstancesBuffer(instancesBuffer, offsetLocations, true);\n }\n else {\n for (let index = 0; index < 4; index++) {\n const offsetLocation = offsetLocations[index];\n if (!this._vertexAttribArraysEnabled[offsetLocation]) {\n this._gl.enableVertexAttribArray(offsetLocation);\n this._vertexAttribArraysEnabled[offsetLocation] = true;\n }\n this._vertexAttribPointer(instancesBuffer, offsetLocation, 4, this._gl.FLOAT, false, 64, index * 16);\n this._gl.vertexAttribDivisor(offsetLocation, 1);\n this._currentInstanceLocations.push(offsetLocation);\n this._currentInstanceBuffers.push(instancesBuffer);\n }\n }\n }\n /**\n * Bind the content of a webGL buffer used with instantiation\n * @param instancesBuffer defines the webGL buffer to bind\n * @param attributesInfo defines the offsets or attributes information used to determine where data must be stored in the buffer\n * @param computeStride defines Whether to compute the strides from the info or use the default 0\n */\n bindInstancesBuffer(instancesBuffer, attributesInfo, computeStride = true) {\n this.bindArrayBuffer(instancesBuffer);\n let stride = 0;\n if (computeStride) {\n for (let i = 0; i < attributesInfo.length; i++) {\n const ai = attributesInfo[i];\n stride += ai.attributeSize * 4;\n }\n }\n for (let i = 0; i < attributesInfo.length; i++) {\n const ai = attributesInfo[i];\n if (ai.index === undefined) {\n ai.index = this._currentEffect.getAttributeLocationByName(ai.attributeName);\n }\n if (ai.index < 0) {\n continue;\n }\n if (!this._vertexAttribArraysEnabled[ai.index]) {\n this._gl.enableVertexAttribArray(ai.index);\n this._vertexAttribArraysEnabled[ai.index] = true;\n }\n this._vertexAttribPointer(instancesBuffer, ai.index, ai.attributeSize, ai.attributeType || this._gl.FLOAT, ai.normalized || false, stride, ai.offset);\n this._gl.vertexAttribDivisor(ai.index, ai.divisor === undefined ? 1 : ai.divisor);\n this._currentInstanceLocations.push(ai.index);\n this._currentInstanceBuffers.push(instancesBuffer);\n }\n }\n /**\n * Disable the instance attribute corresponding to the name in parameter\n * @param name defines the name of the attribute to disable\n */\n disableInstanceAttributeByName(name) {\n if (!this._currentEffect) {\n return;\n }\n const attributeLocation = this._currentEffect.getAttributeLocationByName(name);\n this.disableInstanceAttribute(attributeLocation);\n }\n /**\n * Disable the instance attribute corresponding to the location in parameter\n * @param attributeLocation defines the attribute location of the attribute to disable\n */\n disableInstanceAttribute(attributeLocation) {\n let shouldClean = false;\n let index;\n while ((index = this._currentInstanceLocations.indexOf(attributeLocation)) !== -1) {\n this._currentInstanceLocations.splice(index, 1);\n this._currentInstanceBuffers.splice(index, 1);\n shouldClean = true;\n index = this._currentInstanceLocations.indexOf(attributeLocation);\n }\n if (shouldClean) {\n this._gl.vertexAttribDivisor(attributeLocation, 0);\n this.disableAttributeByIndex(attributeLocation);\n }\n }\n /**\n * Disable the attribute corresponding to the location in parameter\n * @param attributeLocation defines the attribute location of the attribute to disable\n */\n disableAttributeByIndex(attributeLocation) {\n this._gl.disableVertexAttribArray(attributeLocation);\n this._vertexAttribArraysEnabled[attributeLocation] = false;\n this._currentBufferPointers[attributeLocation].active = false;\n }\n /**\n * Send a draw order\n * @param useTriangles defines if triangles must be used to draw (else wireframe will be used)\n * @param indexStart defines the starting index\n * @param indexCount defines the number of index to draw\n * @param instancesCount defines the number of instances to draw (if instantiation is enabled)\n */\n draw(useTriangles, indexStart, indexCount, instancesCount) {\n this.drawElementsType(useTriangles ? 0 : 1, indexStart, indexCount, instancesCount);\n }\n /**\n * Draw a list of points\n * @param verticesStart defines the index of first vertex to draw\n * @param verticesCount defines the count of vertices to draw\n * @param instancesCount defines the number of instances to draw (if instantiation is enabled)\n */\n drawPointClouds(verticesStart, verticesCount, instancesCount) {\n this.drawArraysType(2, verticesStart, verticesCount, instancesCount);\n }\n /**\n * Draw a list of unindexed primitives\n * @param useTriangles defines if triangles must be used to draw (else wireframe will be used)\n * @param verticesStart defines the index of first vertex to draw\n * @param verticesCount defines the count of vertices to draw\n * @param instancesCount defines the number of instances to draw (if instantiation is enabled)\n */\n drawUnIndexed(useTriangles, verticesStart, verticesCount, instancesCount) {\n this.drawArraysType(useTriangles ? 0 : 1, verticesStart, verticesCount, instancesCount);\n }\n /**\n * Draw a list of indexed primitives\n * @param fillMode defines the primitive to use\n * @param indexStart defines the starting index\n * @param indexCount defines the number of index to draw\n * @param instancesCount defines the number of instances to draw (if instantiation is enabled)\n */\n drawElementsType(fillMode, indexStart, indexCount, instancesCount) {\n // Apply states\n this.applyStates();\n this._reportDrawCall();\n // Render\n const drawMode = this._drawMode(fillMode);\n const indexFormat = this._uintIndicesCurrentlySet ? this._gl.UNSIGNED_INT : this._gl.UNSIGNED_SHORT;\n const mult = this._uintIndicesCurrentlySet ? 4 : 2;\n if (instancesCount) {\n this._gl.drawElementsInstanced(drawMode, indexCount, indexFormat, indexStart * mult, instancesCount);\n }\n else {\n this._gl.drawElements(drawMode, indexCount, indexFormat, indexStart * mult);\n }\n }\n /**\n * Draw a list of unindexed primitives\n * @param fillMode defines the primitive to use\n * @param verticesStart defines the index of first vertex to draw\n * @param verticesCount defines the count of vertices to draw\n * @param instancesCount defines the number of instances to draw (if instantiation is enabled)\n */\n drawArraysType(fillMode, verticesStart, verticesCount, instancesCount) {\n // Apply states\n this.applyStates();\n this._reportDrawCall();\n const drawMode = this._drawMode(fillMode);\n if (instancesCount) {\n this._gl.drawArraysInstanced(drawMode, verticesStart, verticesCount, instancesCount);\n }\n else {\n this._gl.drawArrays(drawMode, verticesStart, verticesCount);\n }\n }\n _drawMode(fillMode) {\n switch (fillMode) {\n // Triangle views\n case 0:\n return this._gl.TRIANGLES;\n case 2:\n return this._gl.POINTS;\n case 1:\n return this._gl.LINES;\n // Draw modes\n case 3:\n return this._gl.POINTS;\n case 4:\n return this._gl.LINES;\n case 5:\n return this._gl.LINE_LOOP;\n case 6:\n return this._gl.LINE_STRIP;\n case 7:\n return this._gl.TRIANGLE_STRIP;\n case 8:\n return this._gl.TRIANGLE_FAN;\n default:\n return this._gl.TRIANGLES;\n }\n }\n /** @internal */\n _reportDrawCall() {\n // Will be implemented by children\n }\n // Shaders\n /**\n * @internal\n */\n _releaseEffect(effect) {\n if (this._compiledEffects[effect._key]) {\n delete this._compiledEffects[effect._key];\n }\n const pipelineContext = effect.getPipelineContext();\n if (pipelineContext) {\n this._deletePipelineContext(pipelineContext);\n }\n }\n /**\n * @internal\n */\n _deletePipelineContext(pipelineContext) {\n const webGLPipelineContext = pipelineContext;\n if (webGLPipelineContext && webGLPipelineContext.program) {\n webGLPipelineContext.program.__SPECTOR_rebuildProgram = null;\n this._gl.deleteProgram(webGLPipelineContext.program);\n }\n }\n /** @internal */\n _getGlobalDefines(defines) {\n if (defines) {\n if (this.isNDCHalfZRange) {\n defines[\"IS_NDC_HALF_ZRANGE\"] = \"\";\n }\n else {\n delete defines[\"IS_NDC_HALF_ZRANGE\"];\n }\n if (this.useReverseDepthBuffer) {\n defines[\"USE_REVERSE_DEPTHBUFFER\"] = \"\";\n }\n else {\n delete defines[\"USE_REVERSE_DEPTHBUFFER\"];\n }\n if (this.useExactSrgbConversions) {\n defines[\"USE_EXACT_SRGB_CONVERSIONS\"] = \"\";\n }\n else {\n delete defines[\"USE_EXACT_SRGB_CONVERSIONS\"];\n }\n return;\n }\n else {\n let s = \"\";\n if (this.isNDCHalfZRange) {\n s += \"#define IS_NDC_HALF_ZRANGE\";\n }\n if (this.useReverseDepthBuffer) {\n if (s) {\n s += \"\\n\";\n }\n s += \"#define USE_REVERSE_DEPTHBUFFER\";\n }\n if (this.useExactSrgbConversions) {\n if (s) {\n s += \"\\n\";\n }\n s += \"#define USE_EXACT_SRGB_CONVERSIONS\";\n }\n return s;\n }\n }\n /**\n * Create a new effect (used to store vertex/fragment shaders)\n * @param baseName defines the base name of the effect (The name of file without .fragment.fx or .vertex.fx)\n * @param attributesNamesOrOptions defines either a list of attribute names or an IEffectCreationOptions object\n * @param uniformsNamesOrEngine defines either a list of uniform names or the engine to use\n * @param samplers defines an array of string used to represent textures\n * @param defines defines the string containing the defines to use to compile the shaders\n * @param fallbacks defines the list of potential fallbacks to use if shader compilation fails\n * @param onCompiled defines a function to call when the effect creation is successful\n * @param onError defines a function to call when the effect creation has failed\n * @param indexParameters defines an object containing the index values to use to compile shaders (like the maximum number of simultaneous lights)\n * @param shaderLanguage the language the shader is written in (default: GLSL)\n * @returns the new Effect\n */\n createEffect(baseName, attributesNamesOrOptions, uniformsNamesOrEngine, samplers, defines, fallbacks, onCompiled, onError, indexParameters, shaderLanguage = ShaderLanguage.GLSL) {\n var _a;\n const vertex = baseName.vertexElement || baseName.vertex || baseName.vertexToken || baseName.vertexSource || baseName;\n const fragment = baseName.fragmentElement || baseName.fragment || baseName.fragmentToken || baseName.fragmentSource || baseName;\n const globalDefines = this._getGlobalDefines();\n let fullDefines = (_a = defines !== null && defines !== void 0 ? defines : attributesNamesOrOptions.defines) !== null && _a !== void 0 ? _a : \"\";\n if (globalDefines) {\n fullDefines += globalDefines;\n }\n const name = vertex + \"+\" + fragment + \"@\" + fullDefines;\n if (this._compiledEffects[name]) {\n const compiledEffect = this._compiledEffects[name];\n if (onCompiled && compiledEffect.isReady()) {\n onCompiled(compiledEffect);\n }\n return compiledEffect;\n }\n const effect = new Effect(baseName, attributesNamesOrOptions, uniformsNamesOrEngine, samplers, this, defines, fallbacks, onCompiled, onError, indexParameters, name, shaderLanguage);\n this._compiledEffects[name] = effect;\n return effect;\n }\n // eslint-disable-next-line @typescript-eslint/naming-convention\n static _ConcatenateShader(source, defines, shaderVersion = \"\") {\n return shaderVersion + (defines ? defines + \"\\n\" : \"\") + source;\n }\n _compileShader(source, type, defines, shaderVersion) {\n return this._compileRawShader(ThinEngine._ConcatenateShader(source, defines, shaderVersion), type);\n }\n _compileRawShader(source, type) {\n const gl = this._gl;\n const shader = gl.createShader(type === \"vertex\" ? gl.VERTEX_SHADER : gl.FRAGMENT_SHADER);\n if (!shader) {\n let error = gl.NO_ERROR;\n let tempError = gl.NO_ERROR;\n while ((tempError = gl.getError()) !== gl.NO_ERROR) {\n error = tempError;\n }\n throw new Error(`Something went wrong while creating a gl ${type} shader object. gl error=${error}, gl isContextLost=${gl.isContextLost()}, _contextWasLost=${this._contextWasLost}`);\n }\n gl.shaderSource(shader, source);\n gl.compileShader(shader);\n return shader;\n }\n /**\n * @internal\n */\n _getShaderSource(shader) {\n return this._gl.getShaderSource(shader);\n }\n /**\n * Directly creates a webGL program\n * @param pipelineContext defines the pipeline context to attach to\n * @param vertexCode defines the vertex shader code to use\n * @param fragmentCode defines the fragment shader code to use\n * @param context defines the webGL context to use (if not set, the current one will be used)\n * @param transformFeedbackVaryings defines the list of transform feedback varyings to use\n * @returns the new webGL program\n */\n createRawShaderProgram(pipelineContext, vertexCode, fragmentCode, context, transformFeedbackVaryings = null) {\n context = context || this._gl;\n const vertexShader = this._compileRawShader(vertexCode, \"vertex\");\n const fragmentShader = this._compileRawShader(fragmentCode, \"fragment\");\n return this._createShaderProgram(pipelineContext, vertexShader, fragmentShader, context, transformFeedbackVaryings);\n }\n /**\n * Creates a webGL program\n * @param pipelineContext defines the pipeline context to attach to\n * @param vertexCode defines the vertex shader code to use\n * @param fragmentCode defines the fragment shader code to use\n * @param defines defines the string containing the defines to use to compile the shaders\n * @param context defines the webGL context to use (if not set, the current one will be used)\n * @param transformFeedbackVaryings defines the list of transform feedback varyings to use\n * @returns the new webGL program\n */\n createShaderProgram(pipelineContext, vertexCode, fragmentCode, defines, context, transformFeedbackVaryings = null) {\n context = context || this._gl;\n const shaderVersion = this._webGLVersion > 1 ? \"#version 300 es\\n#define WEBGL2 \\n\" : \"\";\n const vertexShader = this._compileShader(vertexCode, \"vertex\", defines, shaderVersion);\n const fragmentShader = this._compileShader(fragmentCode, \"fragment\", defines, shaderVersion);\n return this._createShaderProgram(pipelineContext, vertexShader, fragmentShader, context, transformFeedbackVaryings);\n }\n /**\n * Inline functions in shader code that are marked to be inlined\n * @param code code to inline\n * @returns inlined code\n */\n inlineShaderCode(code) {\n // no inlining needed in the WebGL engine\n return code;\n }\n /**\n * Creates a new pipeline context\n * @param shaderProcessingContext defines the shader processing context used during the processing if available\n * @returns the new pipeline\n */\n createPipelineContext(shaderProcessingContext) {\n const pipelineContext = new WebGLPipelineContext();\n pipelineContext.engine = this;\n if (this._caps.parallelShaderCompile) {\n pipelineContext.isParallelCompiled = true;\n }\n return pipelineContext;\n }\n /**\n * Creates a new material context\n * @returns the new context\n */\n createMaterialContext() {\n return undefined;\n }\n /**\n * Creates a new draw context\n * @returns the new context\n */\n createDrawContext() {\n return undefined;\n }\n _createShaderProgram(pipelineContext, vertexShader, fragmentShader, context, transformFeedbackVaryings = null) {\n const shaderProgram = context.createProgram();\n pipelineContext.program = shaderProgram;\n if (!shaderProgram) {\n throw new Error(\"Unable to create program\");\n }\n context.attachShader(shaderProgram, vertexShader);\n context.attachShader(shaderProgram, fragmentShader);\n context.linkProgram(shaderProgram);\n pipelineContext.context = context;\n pipelineContext.vertexShader = vertexShader;\n pipelineContext.fragmentShader = fragmentShader;\n if (!pipelineContext.isParallelCompiled) {\n this._finalizePipelineContext(pipelineContext);\n }\n return shaderProgram;\n }\n _finalizePipelineContext(pipelineContext) {\n const context = pipelineContext.context;\n const vertexShader = pipelineContext.vertexShader;\n const fragmentShader = pipelineContext.fragmentShader;\n const program = pipelineContext.program;\n const linked = context.getProgramParameter(program, context.LINK_STATUS);\n if (!linked) {\n // Get more info\n // Vertex\n if (!this._gl.getShaderParameter(vertexShader, this._gl.COMPILE_STATUS)) {\n const log = this._gl.getShaderInfoLog(vertexShader);\n if (log) {\n pipelineContext.vertexCompilationError = log;\n throw new Error(\"VERTEX SHADER \" + log);\n }\n }\n // Fragment\n if (!this._gl.getShaderParameter(fragmentShader, this._gl.COMPILE_STATUS)) {\n const log = this._gl.getShaderInfoLog(fragmentShader);\n if (log) {\n pipelineContext.fragmentCompilationError = log;\n throw new Error(\"FRAGMENT SHADER \" + log);\n }\n }\n const error = context.getProgramInfoLog(program);\n if (error) {\n pipelineContext.programLinkError = error;\n throw new Error(error);\n }\n }\n if (this.validateShaderPrograms) {\n context.validateProgram(program);\n const validated = context.getProgramParameter(program, context.VALIDATE_STATUS);\n if (!validated) {\n const error = context.getProgramInfoLog(program);\n if (error) {\n pipelineContext.programValidationError = error;\n throw new Error(error);\n }\n }\n }\n context.deleteShader(vertexShader);\n context.deleteShader(fragmentShader);\n pipelineContext.vertexShader = undefined;\n pipelineContext.fragmentShader = undefined;\n if (pipelineContext.onCompiled) {\n pipelineContext.onCompiled();\n pipelineContext.onCompiled = undefined;\n }\n }\n /**\n * @internal\n */\n _preparePipelineContext(pipelineContext, vertexSourceCode, fragmentSourceCode, createAsRaw, rawVertexSourceCode, rawFragmentSourceCode, rebuildRebind, defines, transformFeedbackVaryings, key) {\n const webGLRenderingState = pipelineContext;\n if (createAsRaw) {\n webGLRenderingState.program = this.createRawShaderProgram(webGLRenderingState, vertexSourceCode, fragmentSourceCode, undefined, transformFeedbackVaryings);\n }\n else {\n webGLRenderingState.program = this.createShaderProgram(webGLRenderingState, vertexSourceCode, fragmentSourceCode, defines, undefined, transformFeedbackVaryings);\n }\n webGLRenderingState.program.__SPECTOR_rebuildProgram = rebuildRebind;\n }\n /**\n * @internal\n */\n _isRenderingStateCompiled(pipelineContext) {\n const webGLPipelineContext = pipelineContext;\n if (this._isDisposed || webGLPipelineContext._isDisposed) {\n return false;\n }\n if (this._gl.getProgramParameter(webGLPipelineContext.program, this._caps.parallelShaderCompile.COMPLETION_STATUS_KHR)) {\n this._finalizePipelineContext(webGLPipelineContext);\n return true;\n }\n return false;\n }\n /**\n * @internal\n */\n _executeWhenRenderingStateIsCompiled(pipelineContext, action) {\n const webGLPipelineContext = pipelineContext;\n if (!webGLPipelineContext.isParallelCompiled) {\n action();\n return;\n }\n const oldHandler = webGLPipelineContext.onCompiled;\n if (oldHandler) {\n webGLPipelineContext.onCompiled = () => {\n oldHandler();\n action();\n };\n }\n else {\n webGLPipelineContext.onCompiled = action;\n }\n }\n /**\n * Gets the list of webGL uniform locations associated with a specific program based on a list of uniform names\n * @param pipelineContext defines the pipeline context to use\n * @param uniformsNames defines the list of uniform names\n * @returns an array of webGL uniform locations\n */\n getUniforms(pipelineContext, uniformsNames) {\n const results = new Array();\n const webGLPipelineContext = pipelineContext;\n for (let index = 0; index < uniformsNames.length; index++) {\n results.push(this._gl.getUniformLocation(webGLPipelineContext.program, uniformsNames[index]));\n }\n return results;\n }\n /**\n * Gets the list of active attributes for a given webGL program\n * @param pipelineContext defines the pipeline context to use\n * @param attributesNames defines the list of attribute names to get\n * @returns an array of indices indicating the offset of each attribute\n */\n getAttributes(pipelineContext, attributesNames) {\n const results = [];\n const webGLPipelineContext = pipelineContext;\n for (let index = 0; index < attributesNames.length; index++) {\n try {\n results.push(this._gl.getAttribLocation(webGLPipelineContext.program, attributesNames[index]));\n }\n catch (e) {\n results.push(-1);\n }\n }\n return results;\n }\n /**\n * Activates an effect, making it the current one (ie. the one used for rendering)\n * @param effect defines the effect to activate\n */\n enableEffect(effect) {\n effect = effect !== null && DrawWrapper.IsWrapper(effect) ? effect.effect : effect; // get only the effect, we don't need a Wrapper in the WebGL engine\n if (!effect || effect === this._currentEffect) {\n return;\n }\n this._stencilStateComposer.stencilMaterial = undefined;\n effect = effect;\n // Use program\n this.bindSamplers(effect);\n this._currentEffect = effect;\n if (effect.onBind) {\n effect.onBind(effect);\n }\n if (effect._onBindObservable) {\n effect._onBindObservable.notifyObservers(effect);\n }\n }\n /**\n * Set the value of an uniform to a number (int)\n * @param uniform defines the webGL uniform location where to store the value\n * @param value defines the int number to store\n * @returns true if the value was set\n */\n setInt(uniform, value) {\n if (!uniform) {\n return false;\n }\n this._gl.uniform1i(uniform, value);\n return true;\n }\n /**\n * Set the value of an uniform to a int2\n * @param uniform defines the webGL uniform location where to store the value\n * @param x defines the 1st component of the value\n * @param y defines the 2nd component of the value\n * @returns true if the value was set\n */\n setInt2(uniform, x, y) {\n if (!uniform) {\n return false;\n }\n this._gl.uniform2i(uniform, x, y);\n return true;\n }\n /**\n * Set the value of an uniform to a int3\n * @param uniform defines the webGL uniform location where to store the value\n * @param x defines the 1st component of the value\n * @param y defines the 2nd component of the value\n * @param z defines the 3rd component of the value\n * @returns true if the value was set\n */\n setInt3(uniform, x, y, z) {\n if (!uniform) {\n return false;\n }\n this._gl.uniform3i(uniform, x, y, z);\n return true;\n }\n /**\n * Set the value of an uniform to a int4\n * @param uniform defines the webGL uniform location where to store the value\n * @param x defines the 1st component of the value\n * @param y defines the 2nd component of the value\n * @param z defines the 3rd component of the value\n * @param w defines the 4th component of the value\n * @returns true if the value was set\n */\n setInt4(uniform, x, y, z, w) {\n if (!uniform) {\n return false;\n }\n this._gl.uniform4i(uniform, x, y, z, w);\n return true;\n }\n /**\n * Set the value of an uniform to an array of int32\n * @param uniform defines the webGL uniform location where to store the value\n * @param array defines the array of int32 to store\n * @returns true if the value was set\n */\n setIntArray(uniform, array) {\n if (!uniform) {\n return false;\n }\n this._gl.uniform1iv(uniform, array);\n return true;\n }\n /**\n * Set the value of an uniform to an array of int32 (stored as vec2)\n * @param uniform defines the webGL uniform location where to store the value\n * @param array defines the array of int32 to store\n * @returns true if the value was set\n */\n setIntArray2(uniform, array) {\n if (!uniform || array.length % 2 !== 0) {\n return false;\n }\n this._gl.uniform2iv(uniform, array);\n return true;\n }\n /**\n * Set the value of an uniform to an array of int32 (stored as vec3)\n * @param uniform defines the webGL uniform location where to store the value\n * @param array defines the array of int32 to store\n * @returns true if the value was set\n */\n setIntArray3(uniform, array) {\n if (!uniform || array.length % 3 !== 0) {\n return false;\n }\n this._gl.uniform3iv(uniform, array);\n return true;\n }\n /**\n * Set the value of an uniform to an array of int32 (stored as vec4)\n * @param uniform defines the webGL uniform location where to store the value\n * @param array defines the array of int32 to store\n * @returns true if the value was set\n */\n setIntArray4(uniform, array) {\n if (!uniform || array.length % 4 !== 0) {\n return false;\n }\n this._gl.uniform4iv(uniform, array);\n return true;\n }\n /**\n * Set the value of an uniform to a number (unsigned int)\n * @param uniform defines the webGL uniform location where to store the value\n * @param value defines the unsigned int number to store\n * @returns true if the value was set\n */\n setUInt(uniform, value) {\n if (!uniform) {\n return false;\n }\n this._gl.uniform1ui(uniform, value);\n return true;\n }\n /**\n * Set the value of an uniform to a unsigned int2\n * @param uniform defines the webGL uniform location where to store the value\n * @param x defines the 1st component of the value\n * @param y defines the 2nd component of the value\n * @returns true if the value was set\n */\n setUInt2(uniform, x, y) {\n if (!uniform) {\n return false;\n }\n this._gl.uniform2ui(uniform, x, y);\n return true;\n }\n /**\n * Set the value of an uniform to a unsigned int3\n * @param uniform defines the webGL uniform location where to store the value\n * @param x defines the 1st component of the value\n * @param y defines the 2nd component of the value\n * @param z defines the 3rd component of the value\n * @returns true if the value was set\n */\n setUInt3(uniform, x, y, z) {\n if (!uniform) {\n return false;\n }\n this._gl.uniform3ui(uniform, x, y, z);\n return true;\n }\n /**\n * Set the value of an uniform to a unsigned int4\n * @param uniform defines the webGL uniform location where to store the value\n * @param x defines the 1st component of the value\n * @param y defines the 2nd component of the value\n * @param z defines the 3rd component of the value\n * @param w defines the 4th component of the value\n * @returns true if the value was set\n */\n setUInt4(uniform, x, y, z, w) {\n if (!uniform) {\n return false;\n }\n this._gl.uniform4ui(uniform, x, y, z, w);\n return true;\n }\n /**\n * Set the value of an uniform to an array of unsigned int32\n * @param uniform defines the webGL uniform location where to store the value\n * @param array defines the array of unsigned int32 to store\n * @returns true if the value was set\n */\n setUIntArray(uniform, array) {\n if (!uniform) {\n return false;\n }\n this._gl.uniform1uiv(uniform, array);\n return true;\n }\n /**\n * Set the value of an uniform to an array of unsigned int32 (stored as vec2)\n * @param uniform defines the webGL uniform location where to store the value\n * @param array defines the array of unsigned int32 to store\n * @returns true if the value was set\n */\n setUIntArray2(uniform, array) {\n if (!uniform || array.length % 2 !== 0) {\n return false;\n }\n this._gl.uniform2uiv(uniform, array);\n return true;\n }\n /**\n * Set the value of an uniform to an array of unsigned int32 (stored as vec3)\n * @param uniform defines the webGL uniform location where to store the value\n * @param array defines the array of unsigned int32 to store\n * @returns true if the value was set\n */\n setUIntArray3(uniform, array) {\n if (!uniform || array.length % 3 !== 0) {\n return false;\n }\n this._gl.uniform3uiv(uniform, array);\n return true;\n }\n /**\n * Set the value of an uniform to an array of unsigned int32 (stored as vec4)\n * @param uniform defines the webGL uniform location where to store the value\n * @param array defines the array of unsigned int32 to store\n * @returns true if the value was set\n */\n setUIntArray4(uniform, array) {\n if (!uniform || array.length % 4 !== 0) {\n return false;\n }\n this._gl.uniform4uiv(uniform, array);\n return true;\n }\n /**\n * Set the value of an uniform to an array of number\n * @param uniform defines the webGL uniform location where to store the value\n * @param array defines the array of number to store\n * @returns true if the value was set\n */\n setArray(uniform, array) {\n if (!uniform) {\n return false;\n }\n if (array.length < 1) {\n return false;\n }\n this._gl.uniform1fv(uniform, array);\n return true;\n }\n /**\n * Set the value of an uniform to an array of number (stored as vec2)\n * @param uniform defines the webGL uniform location where to store the value\n * @param array defines the array of number to store\n * @returns true if the value was set\n */\n setArray2(uniform, array) {\n if (!uniform || array.length % 2 !== 0) {\n return false;\n }\n this._gl.uniform2fv(uniform, array);\n return true;\n }\n /**\n * Set the value of an uniform to an array of number (stored as vec3)\n * @param uniform defines the webGL uniform location where to store the value\n * @param array defines the array of number to store\n * @returns true if the value was set\n */\n setArray3(uniform, array) {\n if (!uniform || array.length % 3 !== 0) {\n return false;\n }\n this._gl.uniform3fv(uniform, array);\n return true;\n }\n /**\n * Set the value of an uniform to an array of number (stored as vec4)\n * @param uniform defines the webGL uniform location where to store the value\n * @param array defines the array of number to store\n * @returns true if the value was set\n */\n setArray4(uniform, array) {\n if (!uniform || array.length % 4 !== 0) {\n return false;\n }\n this._gl.uniform4fv(uniform, array);\n return true;\n }\n /**\n * Set the value of an uniform to an array of float32 (stored as matrices)\n * @param uniform defines the webGL uniform location where to store the value\n * @param matrices defines the array of float32 to store\n * @returns true if the value was set\n */\n setMatrices(uniform, matrices) {\n if (!uniform) {\n return false;\n }\n this._gl.uniformMatrix4fv(uniform, false, matrices);\n return true;\n }\n /**\n * Set the value of an uniform to a matrix (3x3)\n * @param uniform defines the webGL uniform location where to store the value\n * @param matrix defines the Float32Array representing the 3x3 matrix to store\n * @returns true if the value was set\n */\n setMatrix3x3(uniform, matrix) {\n if (!uniform) {\n return false;\n }\n this._gl.uniformMatrix3fv(uniform, false, matrix);\n return true;\n }\n /**\n * Set the value of an uniform to a matrix (2x2)\n * @param uniform defines the webGL uniform location where to store the value\n * @param matrix defines the Float32Array representing the 2x2 matrix to store\n * @returns true if the value was set\n */\n setMatrix2x2(uniform, matrix) {\n if (!uniform) {\n return false;\n }\n this._gl.uniformMatrix2fv(uniform, false, matrix);\n return true;\n }\n /**\n * Set the value of an uniform to a number (float)\n * @param uniform defines the webGL uniform location where to store the value\n * @param value defines the float number to store\n * @returns true if the value was transferred\n */\n setFloat(uniform, value) {\n if (!uniform) {\n return false;\n }\n this._gl.uniform1f(uniform, value);\n return true;\n }\n /**\n * Set the value of an uniform to a vec2\n * @param uniform defines the webGL uniform location where to store the value\n * @param x defines the 1st component of the value\n * @param y defines the 2nd component of the value\n * @returns true if the value was set\n */\n setFloat2(uniform, x, y) {\n if (!uniform) {\n return false;\n }\n this._gl.uniform2f(uniform, x, y);\n return true;\n }\n /**\n * Set the value of an uniform to a vec3\n * @param uniform defines the webGL uniform location where to store the value\n * @param x defines the 1st component of the value\n * @param y defines the 2nd component of the value\n * @param z defines the 3rd component of the value\n * @returns true if the value was set\n */\n setFloat3(uniform, x, y, z) {\n if (!uniform) {\n return false;\n }\n this._gl.uniform3f(uniform, x, y, z);\n return true;\n }\n /**\n * Set the value of an uniform to a vec4\n * @param uniform defines the webGL uniform location where to store the value\n * @param x defines the 1st component of the value\n * @param y defines the 2nd component of the value\n * @param z defines the 3rd component of the value\n * @param w defines the 4th component of the value\n * @returns true if the value was set\n */\n setFloat4(uniform, x, y, z, w) {\n if (!uniform) {\n return false;\n }\n this._gl.uniform4f(uniform, x, y, z, w);\n return true;\n }\n // States\n /**\n * Apply all cached states (depth, culling, stencil and alpha)\n */\n applyStates() {\n this._depthCullingState.apply(this._gl);\n this._stencilStateComposer.apply(this._gl);\n this._alphaState.apply(this._gl);\n if (this._colorWriteChanged) {\n this._colorWriteChanged = false;\n const enable = this._colorWrite;\n this._gl.colorMask(enable, enable, enable, enable);\n }\n }\n /**\n * Enable or disable color writing\n * @param enable defines the state to set\n */\n setColorWrite(enable) {\n if (enable !== this._colorWrite) {\n this._colorWriteChanged = true;\n this._colorWrite = enable;\n }\n }\n /**\n * Gets a boolean indicating if color writing is enabled\n * @returns the current color writing state\n */\n getColorWrite() {\n return this._colorWrite;\n }\n /**\n * Gets the depth culling state manager\n */\n get depthCullingState() {\n return this._depthCullingState;\n }\n /**\n * Gets the alpha state manager\n */\n get alphaState() {\n return this._alphaState;\n }\n /**\n * Gets the stencil state manager\n */\n get stencilState() {\n return this._stencilState;\n }\n /**\n * Gets the stencil state composer\n */\n get stencilStateComposer() {\n return this._stencilStateComposer;\n }\n // Textures\n /**\n * Clears the list of texture accessible through engine.\n * This can help preventing texture load conflict due to name collision.\n */\n clearInternalTexturesCache() {\n this._internalTexturesCache.length = 0;\n }\n /**\n * Force the entire cache to be cleared\n * You should not have to use this function unless your engine needs to share the webGL context with another engine\n * @param bruteForce defines a boolean to force clearing ALL caches (including stencil, detoh and alpha states)\n */\n wipeCaches(bruteForce) {\n if (this.preventCacheWipeBetweenFrames && !bruteForce) {\n return;\n }\n this._currentEffect = null;\n this._viewportCached.x = 0;\n this._viewportCached.y = 0;\n this._viewportCached.z = 0;\n this._viewportCached.w = 0;\n // Done before in case we clean the attributes\n this._unbindVertexArrayObject();\n if (bruteForce) {\n this._currentProgram = null;\n this.resetTextureCache();\n this._stencilStateComposer.reset();\n this._depthCullingState.reset();\n this._depthCullingState.depthFunc = this._gl.LEQUAL;\n this._alphaState.reset();\n this._alphaMode = 1;\n this._alphaEquation = 0;\n this._colorWrite = true;\n this._colorWriteChanged = true;\n this._unpackFlipYCached = null;\n this._gl.pixelStorei(this._gl.UNPACK_COLORSPACE_CONVERSION_WEBGL, this._gl.NONE);\n this._gl.pixelStorei(this._gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, 0);\n this._mustWipeVertexAttributes = true;\n this.unbindAllAttributes();\n }\n this._resetVertexBufferBinding();\n this._cachedIndexBuffer = null;\n this._cachedEffectForVertexBuffers = null;\n this.bindIndexBuffer(null);\n }\n /**\n * @internal\n */\n _getSamplingParameters(samplingMode, generateMipMaps) {\n const gl = this._gl;\n let magFilter = gl.NEAREST;\n let minFilter = gl.NEAREST;\n switch (samplingMode) {\n case 11:\n magFilter = gl.LINEAR;\n if (generateMipMaps) {\n minFilter = gl.LINEAR_MIPMAP_NEAREST;\n }\n else {\n minFilter = gl.LINEAR;\n }\n break;\n case 3:\n magFilter = gl.LINEAR;\n if (generateMipMaps) {\n minFilter = gl.LINEAR_MIPMAP_LINEAR;\n }\n else {\n minFilter = gl.LINEAR;\n }\n break;\n case 8:\n magFilter = gl.NEAREST;\n if (generateMipMaps) {\n minFilter = gl.NEAREST_MIPMAP_LINEAR;\n }\n else {\n minFilter = gl.NEAREST;\n }\n break;\n case 4:\n magFilter = gl.NEAREST;\n if (generateMipMaps) {\n minFilter = gl.NEAREST_MIPMAP_NEAREST;\n }\n else {\n minFilter = gl.NEAREST;\n }\n break;\n case 5:\n magFilter = gl.NEAREST;\n if (generateMipMaps) {\n minFilter = gl.LINEAR_MIPMAP_NEAREST;\n }\n else {\n minFilter = gl.LINEAR;\n }\n break;\n case 6:\n magFilter = gl.NEAREST;\n if (generateMipMaps) {\n minFilter = gl.LINEAR_MIPMAP_LINEAR;\n }\n else {\n minFilter = gl.LINEAR;\n }\n break;\n case 7:\n magFilter = gl.NEAREST;\n minFilter = gl.LINEAR;\n break;\n case 1:\n magFilter = gl.NEAREST;\n minFilter = gl.NEAREST;\n break;\n case 9:\n magFilter = gl.LINEAR;\n if (generateMipMaps) {\n minFilter = gl.NEAREST_MIPMAP_NEAREST;\n }\n else {\n minFilter = gl.NEAREST;\n }\n break;\n case 10:\n magFilter = gl.LINEAR;\n if (generateMipMaps) {\n minFilter = gl.NEAREST_MIPMAP_LINEAR;\n }\n else {\n minFilter = gl.NEAREST;\n }\n break;\n case 2:\n magFilter = gl.LINEAR;\n minFilter = gl.LINEAR;\n break;\n case 12:\n magFilter = gl.LINEAR;\n minFilter = gl.NEAREST;\n break;\n }\n return {\n min: minFilter,\n mag: magFilter,\n };\n }\n /** @internal */\n _createTexture() {\n const texture = this._gl.createTexture();\n if (!texture) {\n throw new Error(\"Unable to create texture\");\n }\n return texture;\n }\n /** @internal */\n _createHardwareTexture() {\n return new WebGLHardwareTexture(this._createTexture(), this._gl);\n }\n /**\n * Creates an internal texture without binding it to a framebuffer\n * @internal\n * @param size defines the size of the texture\n * @param options defines the options used to create the texture\n * @param delayGPUTextureCreation true to delay the texture creation the first time it is really needed. false to create it right away\n * @param source source type of the texture\n * @returns a new internal texture\n */\n _createInternalTexture(size, options, delayGPUTextureCreation = true, source = InternalTextureSource.Unknown) {\n var _a;\n let generateMipMaps = false;\n let type = 0;\n let samplingMode = 3;\n let format = 5;\n let useSRGBBuffer = false;\n let samples = 1;\n let label;\n if (options !== undefined && typeof options === \"object\") {\n generateMipMaps = !!options.generateMipMaps;\n type = options.type === undefined ? 0 : options.type;\n samplingMode = options.samplingMode === undefined ? 3 : options.samplingMode;\n format = options.format === undefined ? 5 : options.format;\n useSRGBBuffer = options.useSRGBBuffer === undefined ? false : options.useSRGBBuffer;\n samples = (_a = options.samples) !== null && _a !== void 0 ? _a : 1;\n label = options.label;\n }\n else {\n generateMipMaps = !!options;\n }\n useSRGBBuffer && (useSRGBBuffer = this._caps.supportSRGBBuffers && (this.webGLVersion > 1 || this.isWebGPU));\n if (type === 1 && !this._caps.textureFloatLinearFiltering) {\n // if floating point linear (gl.FLOAT) then force to NEAREST_SAMPLINGMODE\n samplingMode = 1;\n }\n else if (type === 2 && !this._caps.textureHalfFloatLinearFiltering) {\n // if floating point linear (HALF_FLOAT) then force to NEAREST_SAMPLINGMODE\n samplingMode = 1;\n }\n if (type === 1 && !this._caps.textureFloat) {\n type = 0;\n Logger.Warn(\"Float textures are not supported. Type forced to TEXTURETYPE_UNSIGNED_BYTE\");\n }\n const gl = this._gl;\n const texture = new InternalTexture(this, source);\n const width = size.width || size;\n const height = size.height || size;\n const layers = size.layers || 0;\n const filters = this._getSamplingParameters(samplingMode, generateMipMaps);\n const target = layers !== 0 ? gl.TEXTURE_2D_ARRAY : gl.TEXTURE_2D;\n const sizedFormat = this._getRGBABufferInternalSizedFormat(type, format, useSRGBBuffer);\n const internalFormat = this._getInternalFormat(format);\n const textureType = this._getWebGLTextureType(type);\n // Bind\n this._bindTextureDirectly(target, texture);\n if (layers !== 0) {\n texture.is2DArray = true;\n gl.texImage3D(target, 0, sizedFormat, width, height, layers, 0, internalFormat, textureType, null);\n }\n else {\n gl.texImage2D(target, 0, sizedFormat, width, height, 0, internalFormat, textureType, null);\n }\n gl.texParameteri(target, gl.TEXTURE_MAG_FILTER, filters.mag);\n gl.texParameteri(target, gl.TEXTURE_MIN_FILTER, filters.min);\n gl.texParameteri(target, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n gl.texParameteri(target, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n // MipMaps\n if (generateMipMaps) {\n this._gl.generateMipmap(target);\n }\n this._bindTextureDirectly(target, null);\n texture._useSRGBBuffer = useSRGBBuffer;\n texture.baseWidth = width;\n texture.baseHeight = height;\n texture.width = width;\n texture.height = height;\n texture.depth = layers;\n texture.isReady = true;\n texture.samples = samples;\n texture.generateMipMaps = generateMipMaps;\n texture.samplingMode = samplingMode;\n texture.type = type;\n texture.format = format;\n texture.label = label;\n this._internalTexturesCache.push(texture);\n return texture;\n }\n /**\n * @internal\n */\n _getUseSRGBBuffer(useSRGBBuffer, noMipmap) {\n // Generating mipmaps for sRGB textures is not supported in WebGL1 so we must disable the support if mipmaps is enabled\n return useSRGBBuffer && this._caps.supportSRGBBuffers && (this.webGLVersion > 1 || this.isWebGPU || noMipmap);\n }\n _createTextureBase(url, noMipmap, invertY, scene, samplingMode = 3, onLoad = null, onError = null, prepareTexture, prepareTextureProcessFunction, buffer = null, fallback = null, format = null, forcedExtension = null, mimeType, loaderOptions, useSRGBBuffer) {\n url = url || \"\";\n const fromData = url.substr(0, 5) === \"data:\";\n const fromBlob = url.substr(0, 5) === \"blob:\";\n const isBase64 = fromData && url.indexOf(\";base64,\") !== -1;\n const texture = fallback ? fallback : new InternalTexture(this, InternalTextureSource.Url);\n if (texture !== fallback) {\n texture.label = url.substring(0, 60); // default label, can be overriden by the caller\n }\n const originalUrl = url;\n if (this._transformTextureUrl && !isBase64 && !fallback && !buffer) {\n url = this._transformTextureUrl(url);\n }\n if (originalUrl !== url) {\n texture._originalUrl = originalUrl;\n }\n // establish the file extension, if possible\n const lastDot = url.lastIndexOf(\".\");\n let extension = forcedExtension ? forcedExtension : lastDot > -1 ? url.substring(lastDot).toLowerCase() : \"\";\n let loader = null;\n // Remove query string\n const queryStringIndex = extension.indexOf(\"?\");\n if (queryStringIndex > -1) {\n extension = extension.split(\"?\")[0];\n }\n for (const availableLoader of ThinEngine._TextureLoaders) {\n if (availableLoader.canLoad(extension, mimeType)) {\n loader = availableLoader;\n break;\n }\n }\n if (scene) {\n scene.addPendingData(texture);\n }\n texture.url = url;\n texture.generateMipMaps = !noMipmap;\n texture.samplingMode = samplingMode;\n texture.invertY = invertY;\n texture._useSRGBBuffer = this._getUseSRGBBuffer(!!useSRGBBuffer, noMipmap);\n if (!this._doNotHandleContextLost) {\n // Keep a link to the buffer only if we plan to handle context lost\n texture._buffer = buffer;\n }\n let onLoadObserver = null;\n if (onLoad && !fallback) {\n onLoadObserver = texture.onLoadedObservable.add(onLoad);\n }\n if (!fallback) {\n this._internalTexturesCache.push(texture);\n }\n const onInternalError = (message, exception) => {\n if (scene) {\n scene.removePendingData(texture);\n }\n if (url === originalUrl) {\n if (onLoadObserver) {\n texture.onLoadedObservable.remove(onLoadObserver);\n }\n if (EngineStore.UseFallbackTexture) {\n this._createTextureBase(EngineStore.FallbackTexture, noMipmap, texture.invertY, scene, samplingMode, null, onError, prepareTexture, prepareTextureProcessFunction, buffer, texture);\n }\n message = (message || \"Unknown error\") + (EngineStore.UseFallbackTexture ? \" - Fallback texture was used\" : \"\");\n texture.onErrorObservable.notifyObservers({ message, exception });\n if (onError) {\n onError(message, exception);\n }\n }\n else {\n // fall back to the original url if the transformed url fails to load\n Logger.Warn(`Failed to load ${url}, falling back to ${originalUrl}`);\n this._createTextureBase(originalUrl, noMipmap, texture.invertY, scene, samplingMode, onLoad, onError, prepareTexture, prepareTextureProcessFunction, buffer, texture, format, forcedExtension, mimeType, loaderOptions, useSRGBBuffer);\n }\n };\n // processing for non-image formats\n if (loader) {\n const callback = (data) => {\n loader.loadData(data, texture, (width, height, loadMipmap, isCompressed, done, loadFailed) => {\n if (loadFailed) {\n onInternalError(\"TextureLoader failed to load data\");\n }\n else {\n prepareTexture(texture, extension, scene, { width, height }, texture.invertY, !loadMipmap, isCompressed, () => {\n done();\n return false;\n }, samplingMode);\n }\n }, loaderOptions);\n };\n if (!buffer) {\n this._loadFile(url, (data) => callback(new Uint8Array(data)), undefined, scene ? scene.offlineProvider : undefined, true, (request, exception) => {\n onInternalError(\"Unable to load \" + (request ? request.responseURL : url, exception));\n });\n }\n else {\n if (buffer instanceof ArrayBuffer) {\n callback(new Uint8Array(buffer));\n }\n else if (ArrayBuffer.isView(buffer)) {\n callback(buffer);\n }\n else {\n if (onError) {\n onError(\"Unable to load: only ArrayBuffer or ArrayBufferView is supported\", null);\n }\n }\n }\n }\n else {\n const onload = (img) => {\n if (fromBlob && !this._doNotHandleContextLost) {\n // We need to store the image if we need to rebuild the texture\n // in case of a webgl context lost\n texture._buffer = img;\n }\n prepareTexture(texture, extension, scene, img, texture.invertY, noMipmap, false, prepareTextureProcessFunction, samplingMode);\n };\n // According to the WebGL spec section 6.10, ImageBitmaps must be inverted on creation.\n // So, we pass imageOrientation to _FileToolsLoadImage() as it may create an ImageBitmap.\n if (!fromData || isBase64) {\n if (buffer && (typeof buffer.decoding === \"string\" || buffer.close)) {\n onload(buffer);\n }\n else {\n ThinEngine._FileToolsLoadImage(url, onload, onInternalError, scene ? scene.offlineProvider : null, mimeType, texture.invertY && this._features.needsInvertingBitmap ? { imageOrientation: \"flipY\" } : undefined);\n }\n }\n else if (typeof buffer === \"string\" || buffer instanceof ArrayBuffer || ArrayBuffer.isView(buffer) || buffer instanceof Blob) {\n ThinEngine._FileToolsLoadImage(buffer, onload, onInternalError, scene ? scene.offlineProvider : null, mimeType, texture.invertY && this._features.needsInvertingBitmap ? { imageOrientation: \"flipY\" } : undefined);\n }\n else if (buffer) {\n onload(buffer);\n }\n }\n return texture;\n }\n /**\n * Usually called from Texture.ts.\n * Passed information to create a WebGLTexture\n * @param url defines a value which contains one of the following:\n * * A conventional http URL, e.g. 'http://...' or 'file://...'\n * * A base64 string of in-line texture data, e.g. '...'\n * * An indicator that data being passed using the buffer parameter, e.g. 'data:mytexture.jpg'\n * @param noMipmap defines a boolean indicating that no mipmaps shall be generated. Ignored for compressed textures. They must be in the file\n * @param invertY when true, image is flipped when loaded. You probably want true. Certain compressed textures may invert this if their default is inverted (eg. ktx)\n * @param scene needed for loading to the correct scene\n * @param samplingMode mode with should be used sample / access the texture (Default: Texture.TRILINEAR_SAMPLINGMODE)\n * @param onLoad optional callback to be called upon successful completion\n * @param onError optional callback to be called upon failure\n * @param buffer a source of a file previously fetched as either a base64 string, an ArrayBuffer (compressed or image format), HTMLImageElement (image format), or a Blob\n * @param fallback an internal argument in case the function must be called again, due to etc1 not having alpha capabilities\n * @param format internal format. Default: RGB when extension is '.jpg' else RGBA. Ignored for compressed textures\n * @param forcedExtension defines the extension to use to pick the right loader\n * @param mimeType defines an optional mime type\n * @param loaderOptions options to be passed to the loader\n * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg)\n * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU).\n * @returns a InternalTexture for assignment back into BABYLON.Texture\n */\n createTexture(url, noMipmap, invertY, scene, samplingMode = 3, onLoad = null, onError = null, buffer = null, fallback = null, format = null, forcedExtension = null, mimeType, loaderOptions, creationFlags, useSRGBBuffer) {\n return this._createTextureBase(url, noMipmap, invertY, scene, samplingMode, onLoad, onError, this._prepareWebGLTexture.bind(this), (potWidth, potHeight, img, extension, texture, continuationCallback) => {\n const gl = this._gl;\n const isPot = img.width === potWidth && img.height === potHeight;\n const internalFormat = format\n ? this._getInternalFormat(format, texture._useSRGBBuffer)\n : extension === \".jpg\" && !texture._useSRGBBuffer\n ? gl.RGB\n : texture._useSRGBBuffer\n ? this._glSRGBExtensionValues.SRGB8_ALPHA8\n : gl.RGBA;\n let texelFormat = format ? this._getInternalFormat(format) : extension === \".jpg\" && !texture._useSRGBBuffer ? gl.RGB : gl.RGBA;\n if (texture._useSRGBBuffer && this.webGLVersion === 1) {\n texelFormat = internalFormat;\n }\n if (isPot) {\n gl.texImage2D(gl.TEXTURE_2D, 0, internalFormat, texelFormat, gl.UNSIGNED_BYTE, img);\n return false;\n }\n const maxTextureSize = this._caps.maxTextureSize;\n if (img.width > maxTextureSize || img.height > maxTextureSize || !this._supportsHardwareTextureRescaling) {\n this._prepareWorkingCanvas();\n if (!this._workingCanvas || !this._workingContext) {\n return false;\n }\n this._workingCanvas.width = potWidth;\n this._workingCanvas.height = potHeight;\n this._workingContext.drawImage(img, 0, 0, img.width, img.height, 0, 0, potWidth, potHeight);\n gl.texImage2D(gl.TEXTURE_2D, 0, internalFormat, texelFormat, gl.UNSIGNED_BYTE, this._workingCanvas);\n texture.width = potWidth;\n texture.height = potHeight;\n return false;\n }\n else {\n // Using shaders when possible to rescale because canvas.drawImage is lossy\n const source = new InternalTexture(this, InternalTextureSource.Temp);\n this._bindTextureDirectly(gl.TEXTURE_2D, source, true);\n gl.texImage2D(gl.TEXTURE_2D, 0, internalFormat, texelFormat, gl.UNSIGNED_BYTE, img);\n this._rescaleTexture(source, texture, scene, internalFormat, () => {\n this._releaseTexture(source);\n this._bindTextureDirectly(gl.TEXTURE_2D, texture, true);\n continuationCallback();\n });\n }\n return true;\n }, buffer, fallback, format, forcedExtension, mimeType, loaderOptions, useSRGBBuffer);\n }\n /**\n * Loads an image as an HTMLImageElement.\n * @param input url string, ArrayBuffer, or Blob to load\n * @param onLoad callback called when the image successfully loads\n * @param onError callback called when the image fails to load\n * @param offlineProvider offline provider for caching\n * @param mimeType optional mime type\n * @param imageBitmapOptions optional the options to use when creating an ImageBitmap\n * @returns the HTMLImageElement of the loaded image\n * @internal\n */\n static _FileToolsLoadImage(input, onLoad, onError, offlineProvider, mimeType, imageBitmapOptions) {\n throw _WarnImport(\"FileTools\");\n }\n /**\n * @internal\n */\n _rescaleTexture(source, destination, scene, internalFormat, onComplete) { }\n /**\n * Creates a raw texture\n * @param data defines the data to store in the texture\n * @param width defines the width of the texture\n * @param height defines the height of the texture\n * @param format defines the format of the data\n * @param generateMipMaps defines if the engine should generate the mip levels\n * @param invertY defines if data must be stored with Y axis inverted\n * @param samplingMode defines the required sampling mode (Texture.NEAREST_SAMPLINGMODE by default)\n * @param compression defines the compression used (null by default)\n * @param type defines the type fo the data (Engine.TEXTURETYPE_UNSIGNED_INT by default)\n * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg)\n * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU).\n * @returns the raw texture inside an InternalTexture\n */\n createRawTexture(data, width, height, format, generateMipMaps, invertY, samplingMode, compression = null, type = 0, creationFlags = 0, useSRGBBuffer = false) {\n throw _WarnImport(\"Engine.RawTexture\");\n }\n /**\n * Creates a new raw cube texture\n * @param data defines the array of data to use to create each face\n * @param size defines the size of the textures\n * @param format defines the format of the data\n * @param type defines the type of the data (like Engine.TEXTURETYPE_UNSIGNED_INT)\n * @param generateMipMaps defines if the engine should generate the mip levels\n * @param invertY defines if data must be stored with Y axis inverted\n * @param samplingMode defines the required sampling mode (like Texture.NEAREST_SAMPLINGMODE)\n * @param compression defines the compression used (null by default)\n * @returns the cube texture as an InternalTexture\n */\n createRawCubeTexture(data, size, format, type, generateMipMaps, invertY, samplingMode, compression = null) {\n throw _WarnImport(\"Engine.RawTexture\");\n }\n /**\n * Creates a new raw 3D texture\n * @param data defines the data used to create the texture\n * @param width defines the width of the texture\n * @param height defines the height of the texture\n * @param depth defines the depth of the texture\n * @param format defines the format of the texture\n * @param generateMipMaps defines if the engine must generate mip levels\n * @param invertY defines if data must be stored with Y axis inverted\n * @param samplingMode defines the required sampling mode (like Texture.NEAREST_SAMPLINGMODE)\n * @param compression defines the compressed used (can be null)\n * @param textureType defines the compressed used (can be null)\n * @returns a new raw 3D texture (stored in an InternalTexture)\n */\n createRawTexture3D(data, width, height, depth, format, generateMipMaps, invertY, samplingMode, compression = null, textureType = 0) {\n throw _WarnImport(\"Engine.RawTexture\");\n }\n /**\n * Creates a new raw 2D array texture\n * @param data defines the data used to create the texture\n * @param width defines the width of the texture\n * @param height defines the height of the texture\n * @param depth defines the number of layers of the texture\n * @param format defines the format of the texture\n * @param generateMipMaps defines if the engine must generate mip levels\n * @param invertY defines if data must be stored with Y axis inverted\n * @param samplingMode defines the required sampling mode (like Texture.NEAREST_SAMPLINGMODE)\n * @param compression defines the compressed used (can be null)\n * @param textureType defines the compressed used (can be null)\n * @returns a new raw 2D array texture (stored in an InternalTexture)\n */\n createRawTexture2DArray(data, width, height, depth, format, generateMipMaps, invertY, samplingMode, compression = null, textureType = 0) {\n throw _WarnImport(\"Engine.RawTexture\");\n }\n /**\n * @internal\n */\n _unpackFlipY(value) {\n if (this._unpackFlipYCached !== value) {\n this._gl.pixelStorei(this._gl.UNPACK_FLIP_Y_WEBGL, value ? 1 : 0);\n if (this.enableUnpackFlipYCached) {\n this._unpackFlipYCached = value;\n }\n }\n }\n /** @internal */\n _getUnpackAlignement() {\n return this._gl.getParameter(this._gl.UNPACK_ALIGNMENT);\n }\n _getTextureTarget(texture) {\n if (texture.isCube) {\n return this._gl.TEXTURE_CUBE_MAP;\n }\n else if (texture.is3D) {\n return this._gl.TEXTURE_3D;\n }\n else if (texture.is2DArray || texture.isMultiview) {\n return this._gl.TEXTURE_2D_ARRAY;\n }\n return this._gl.TEXTURE_2D;\n }\n /**\n * Update the sampling mode of a given texture\n * @param samplingMode defines the required sampling mode\n * @param texture defines the texture to update\n * @param generateMipMaps defines whether to generate mipmaps for the texture\n */\n updateTextureSamplingMode(samplingMode, texture, generateMipMaps = false) {\n const target = this._getTextureTarget(texture);\n const filters = this._getSamplingParameters(samplingMode, texture.useMipMaps || generateMipMaps);\n this._setTextureParameterInteger(target, this._gl.TEXTURE_MAG_FILTER, filters.mag, texture);\n this._setTextureParameterInteger(target, this._gl.TEXTURE_MIN_FILTER, filters.min);\n if (generateMipMaps) {\n texture.generateMipMaps = true;\n this._gl.generateMipmap(target);\n }\n this._bindTextureDirectly(target, null);\n texture.samplingMode = samplingMode;\n }\n /**\n * Update the dimensions of a texture\n * @param texture texture to update\n * @param width new width of the texture\n * @param height new height of the texture\n * @param depth new depth of the texture\n */\n updateTextureDimensions(texture, width, height, depth = 1) { }\n /**\n * Update the sampling mode of a given texture\n * @param texture defines the texture to update\n * @param wrapU defines the texture wrap mode of the u coordinates\n * @param wrapV defines the texture wrap mode of the v coordinates\n * @param wrapR defines the texture wrap mode of the r coordinates\n */\n updateTextureWrappingMode(texture, wrapU, wrapV = null, wrapR = null) {\n const target = this._getTextureTarget(texture);\n if (wrapU !== null) {\n this._setTextureParameterInteger(target, this._gl.TEXTURE_WRAP_S, this._getTextureWrapMode(wrapU), texture);\n texture._cachedWrapU = wrapU;\n }\n if (wrapV !== null) {\n this._setTextureParameterInteger(target, this._gl.TEXTURE_WRAP_T, this._getTextureWrapMode(wrapV), texture);\n texture._cachedWrapV = wrapV;\n }\n if ((texture.is2DArray || texture.is3D) && wrapR !== null) {\n this._setTextureParameterInteger(target, this._gl.TEXTURE_WRAP_R, this._getTextureWrapMode(wrapR), texture);\n texture._cachedWrapR = wrapR;\n }\n this._bindTextureDirectly(target, null);\n }\n /**\n * @internal\n */\n _setupDepthStencilTexture(internalTexture, size, generateStencil, bilinearFiltering, comparisonFunction, samples = 1) {\n const width = size.width || size;\n const height = size.height || size;\n const layers = size.layers || 0;\n internalTexture.baseWidth = width;\n internalTexture.baseHeight = height;\n internalTexture.width = width;\n internalTexture.height = height;\n internalTexture.is2DArray = layers > 0;\n internalTexture.depth = layers;\n internalTexture.isReady = true;\n internalTexture.samples = samples;\n internalTexture.generateMipMaps = false;\n internalTexture.samplingMode = bilinearFiltering ? 2 : 1;\n internalTexture.type = 0;\n internalTexture._comparisonFunction = comparisonFunction;\n const gl = this._gl;\n const target = this._getTextureTarget(internalTexture);\n const samplingParameters = this._getSamplingParameters(internalTexture.samplingMode, false);\n gl.texParameteri(target, gl.TEXTURE_MAG_FILTER, samplingParameters.mag);\n gl.texParameteri(target, gl.TEXTURE_MIN_FILTER, samplingParameters.min);\n gl.texParameteri(target, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n gl.texParameteri(target, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n // TEXTURE_COMPARE_FUNC/MODE are only availble in WebGL2.\n if (this.webGLVersion > 1) {\n if (comparisonFunction === 0) {\n gl.texParameteri(target, gl.TEXTURE_COMPARE_FUNC, 515);\n gl.texParameteri(target, gl.TEXTURE_COMPARE_MODE, gl.NONE);\n }\n else {\n gl.texParameteri(target, gl.TEXTURE_COMPARE_FUNC, comparisonFunction);\n gl.texParameteri(target, gl.TEXTURE_COMPARE_MODE, gl.COMPARE_REF_TO_TEXTURE);\n }\n }\n }\n /**\n * @internal\n */\n _uploadCompressedDataToTextureDirectly(texture, internalFormat, width, height, data, faceIndex = 0, lod = 0) {\n const gl = this._gl;\n let target = gl.TEXTURE_2D;\n if (texture.isCube) {\n target = gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex;\n }\n if (texture._useSRGBBuffer) {\n switch (internalFormat) {\n case 37492:\n case 36196:\n // Note, if using ETC1 and sRGB is requested, this will use ETC2 if available.\n if (this._caps.etc2) {\n internalFormat = gl.COMPRESSED_SRGB8_ETC2;\n }\n else {\n texture._useSRGBBuffer = false;\n }\n break;\n case 37496:\n if (this._caps.etc2) {\n internalFormat = gl.COMPRESSED_SRGB8_ALPHA8_ETC2_EAC;\n }\n else {\n texture._useSRGBBuffer = false;\n }\n break;\n case 36492:\n internalFormat = gl.COMPRESSED_SRGB_ALPHA_BPTC_UNORM_EXT;\n break;\n case 37808:\n internalFormat = gl.COMPRESSED_SRGB8_ALPHA8_ASTC_4x4_KHR;\n break;\n case 33776:\n if (this._caps.s3tc_srgb) {\n internalFormat = gl.COMPRESSED_SRGB_S3TC_DXT1_EXT;\n }\n else {\n // S3TC sRGB extension not supported\n texture._useSRGBBuffer = false;\n }\n break;\n case 33777:\n if (this._caps.s3tc_srgb) {\n internalFormat = gl.COMPRESSED_SRGB_ALPHA_S3TC_DXT1_EXT;\n }\n else {\n // S3TC sRGB extension not supported\n texture._useSRGBBuffer = false;\n }\n break;\n case 33779:\n if (this._caps.s3tc_srgb) {\n internalFormat = gl.COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT;\n }\n else {\n // S3TC sRGB extension not supported\n texture._useSRGBBuffer = false;\n }\n break;\n default:\n // We don't support a sRGB format corresponding to internalFormat, so revert to non sRGB format\n texture._useSRGBBuffer = false;\n break;\n }\n }\n this._gl.compressedTexImage2D(target, lod, internalFormat, width, height, 0, data);\n }\n /**\n * @internal\n */\n _uploadDataToTextureDirectly(texture, imageData, faceIndex = 0, lod = 0, babylonInternalFormat, useTextureWidthAndHeight = false) {\n const gl = this._gl;\n const textureType = this._getWebGLTextureType(texture.type);\n const format = this._getInternalFormat(texture.format);\n const internalFormat = babylonInternalFormat === undefined\n ? this._getRGBABufferInternalSizedFormat(texture.type, texture.format, texture._useSRGBBuffer)\n : this._getInternalFormat(babylonInternalFormat, texture._useSRGBBuffer);\n this._unpackFlipY(texture.invertY);\n let target = gl.TEXTURE_2D;\n if (texture.isCube) {\n target = gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex;\n }\n const lodMaxWidth = Math.round(Math.log(texture.width) * Math.LOG2E);\n const lodMaxHeight = Math.round(Math.log(texture.height) * Math.LOG2E);\n const width = useTextureWidthAndHeight ? texture.width : Math.pow(2, Math.max(lodMaxWidth - lod, 0));\n const height = useTextureWidthAndHeight ? texture.height : Math.pow(2, Math.max(lodMaxHeight - lod, 0));\n gl.texImage2D(target, lod, internalFormat, width, height, 0, format, textureType, imageData);\n }\n /**\n * Update a portion of an internal texture\n * @param texture defines the texture to update\n * @param imageData defines the data to store into the texture\n * @param xOffset defines the x coordinates of the update rectangle\n * @param yOffset defines the y coordinates of the update rectangle\n * @param width defines the width of the update rectangle\n * @param height defines the height of the update rectangle\n * @param faceIndex defines the face index if texture is a cube (0 by default)\n * @param lod defines the lod level to update (0 by default)\n * @param generateMipMaps defines whether to generate mipmaps or not\n */\n updateTextureData(texture, imageData, xOffset, yOffset, width, height, faceIndex = 0, lod = 0, generateMipMaps = false) {\n const gl = this._gl;\n const textureType = this._getWebGLTextureType(texture.type);\n const format = this._getInternalFormat(texture.format);\n this._unpackFlipY(texture.invertY);\n let targetForBinding = gl.TEXTURE_2D;\n let target = gl.TEXTURE_2D;\n if (texture.isCube) {\n target = gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex;\n targetForBinding = gl.TEXTURE_CUBE_MAP;\n }\n this._bindTextureDirectly(targetForBinding, texture, true);\n gl.texSubImage2D(target, lod, xOffset, yOffset, width, height, format, textureType, imageData);\n if (generateMipMaps) {\n this._gl.generateMipmap(target);\n }\n this._bindTextureDirectly(targetForBinding, null);\n }\n /**\n * @internal\n */\n _uploadArrayBufferViewToTexture(texture, imageData, faceIndex = 0, lod = 0) {\n const gl = this._gl;\n const bindTarget = texture.isCube ? gl.TEXTURE_CUBE_MAP : gl.TEXTURE_2D;\n this._bindTextureDirectly(bindTarget, texture, true);\n this._uploadDataToTextureDirectly(texture, imageData, faceIndex, lod);\n this._bindTextureDirectly(bindTarget, null, true);\n }\n _prepareWebGLTextureContinuation(texture, scene, noMipmap, isCompressed, samplingMode) {\n const gl = this._gl;\n if (!gl) {\n return;\n }\n const filters = this._getSamplingParameters(samplingMode, !noMipmap);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, filters.mag);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, filters.min);\n if (!noMipmap && !isCompressed) {\n gl.generateMipmap(gl.TEXTURE_2D);\n }\n this._bindTextureDirectly(gl.TEXTURE_2D, null);\n // this.resetTextureCache();\n if (scene) {\n scene.removePendingData(texture);\n }\n texture.onLoadedObservable.notifyObservers(texture);\n texture.onLoadedObservable.clear();\n }\n _prepareWebGLTexture(texture, extension, scene, img, invertY, noMipmap, isCompressed, processFunction, samplingMode = 3) {\n const maxTextureSize = this.getCaps().maxTextureSize;\n const potWidth = Math.min(maxTextureSize, this.needPOTTextures ? ThinEngine.GetExponentOfTwo(img.width, maxTextureSize) : img.width);\n const potHeight = Math.min(maxTextureSize, this.needPOTTextures ? ThinEngine.GetExponentOfTwo(img.height, maxTextureSize) : img.height);\n const gl = this._gl;\n if (!gl) {\n return;\n }\n if (!texture._hardwareTexture) {\n // this.resetTextureCache();\n if (scene) {\n scene.removePendingData(texture);\n }\n return;\n }\n this._bindTextureDirectly(gl.TEXTURE_2D, texture, true);\n this._unpackFlipY(invertY === undefined ? true : invertY ? true : false);\n texture.baseWidth = img.width;\n texture.baseHeight = img.height;\n texture.width = potWidth;\n texture.height = potHeight;\n texture.isReady = true;\n if (processFunction(potWidth, potHeight, img, extension, texture, () => {\n this._prepareWebGLTextureContinuation(texture, scene, noMipmap, isCompressed, samplingMode);\n })) {\n // Returning as texture needs extra async steps\n return;\n }\n this._prepareWebGLTextureContinuation(texture, scene, noMipmap, isCompressed, samplingMode);\n }\n /**\n * @internal\n */\n _setupFramebufferDepthAttachments(generateStencilBuffer, generateDepthBuffer, width, height, samples = 1) {\n const gl = this._gl;\n // Create the depth/stencil buffer\n if (generateStencilBuffer && generateDepthBuffer) {\n return this._createRenderBuffer(width, height, samples, gl.DEPTH_STENCIL, gl.DEPTH24_STENCIL8, gl.DEPTH_STENCIL_ATTACHMENT);\n }\n if (generateDepthBuffer) {\n let depthFormat = gl.DEPTH_COMPONENT16;\n if (this._webGLVersion > 1) {\n depthFormat = gl.DEPTH_COMPONENT32F;\n }\n return this._createRenderBuffer(width, height, samples, depthFormat, depthFormat, gl.DEPTH_ATTACHMENT);\n }\n if (generateStencilBuffer) {\n return this._createRenderBuffer(width, height, samples, gl.STENCIL_INDEX8, gl.STENCIL_INDEX8, gl.STENCIL_ATTACHMENT);\n }\n return null;\n }\n /**\n * @internal\n */\n _createRenderBuffer(width, height, samples, internalFormat, msInternalFormat, attachment, unbindBuffer = true) {\n const gl = this._gl;\n const renderBuffer = gl.createRenderbuffer();\n return this._updateRenderBuffer(renderBuffer, width, height, samples, internalFormat, msInternalFormat, attachment, unbindBuffer);\n }\n _updateRenderBuffer(renderBuffer, width, height, samples, internalFormat, msInternalFormat, attachment, unbindBuffer = true) {\n const gl = this._gl;\n gl.bindRenderbuffer(gl.RENDERBUFFER, renderBuffer);\n if (samples > 1 && gl.renderbufferStorageMultisample) {\n gl.renderbufferStorageMultisample(gl.RENDERBUFFER, samples, msInternalFormat, width, height);\n }\n else {\n gl.renderbufferStorage(gl.RENDERBUFFER, internalFormat, width, height);\n }\n gl.framebufferRenderbuffer(gl.FRAMEBUFFER, attachment, gl.RENDERBUFFER, renderBuffer);\n if (unbindBuffer) {\n gl.bindRenderbuffer(gl.RENDERBUFFER, null);\n }\n return renderBuffer;\n }\n /**\n * @internal\n */\n _releaseTexture(texture) {\n var _a;\n this._deleteTexture((_a = texture._hardwareTexture) === null || _a === void 0 ? void 0 : _a.underlyingResource);\n // Unbind channels\n this.unbindAllTextures();\n const index = this._internalTexturesCache.indexOf(texture);\n if (index !== -1) {\n this._internalTexturesCache.splice(index, 1);\n }\n // Integrated fixed lod samplers.\n if (texture._lodTextureHigh) {\n texture._lodTextureHigh.dispose();\n }\n if (texture._lodTextureMid) {\n texture._lodTextureMid.dispose();\n }\n if (texture._lodTextureLow) {\n texture._lodTextureLow.dispose();\n }\n // Integrated irradiance map.\n if (texture._irradianceTexture) {\n texture._irradianceTexture.dispose();\n }\n }\n /**\n * @internal\n */\n _releaseRenderTargetWrapper(rtWrapper) {\n const index = this._renderTargetWrapperCache.indexOf(rtWrapper);\n if (index !== -1) {\n this._renderTargetWrapperCache.splice(index, 1);\n }\n }\n _deleteTexture(texture) {\n if (texture) {\n this._gl.deleteTexture(texture);\n }\n }\n _setProgram(program) {\n if (this._currentProgram !== program) {\n this._gl.useProgram(program);\n this._currentProgram = program;\n }\n }\n /**\n * Binds an effect to the webGL context\n * @param effect defines the effect to bind\n */\n bindSamplers(effect) {\n const webGLPipelineContext = effect.getPipelineContext();\n this._setProgram(webGLPipelineContext.program);\n const samplers = effect.getSamplers();\n for (let index = 0; index < samplers.length; index++) {\n const uniform = effect.getUniform(samplers[index]);\n if (uniform) {\n this._boundUniforms[index] = uniform;\n }\n }\n this._currentEffect = null;\n }\n _activateCurrentTexture() {\n if (this._currentTextureChannel !== this._activeChannel) {\n this._gl.activeTexture(this._gl.TEXTURE0 + this._activeChannel);\n this._currentTextureChannel = this._activeChannel;\n }\n }\n /**\n * @internal\n */\n _bindTextureDirectly(target, texture, forTextureDataUpdate = false, force = false) {\n var _a, _b;\n let wasPreviouslyBound = false;\n const isTextureForRendering = texture && texture._associatedChannel > -1;\n if (forTextureDataUpdate && isTextureForRendering) {\n this._activeChannel = texture._associatedChannel;\n }\n const currentTextureBound = this._boundTexturesCache[this._activeChannel];\n if (currentTextureBound !== texture || force) {\n this._activateCurrentTexture();\n if (texture && texture.isMultiview) {\n //this._gl.bindTexture(target, texture ? texture._colorTextureArray : null);\n console.error(target, texture);\n throw \"_bindTextureDirectly called with a multiview texture!\";\n }\n else {\n this._gl.bindTexture(target, (_b = (_a = texture === null || texture === void 0 ? void 0 : texture._hardwareTexture) === null || _a === void 0 ? void 0 : _a.underlyingResource) !== null && _b !== void 0 ? _b : null);\n }\n this._boundTexturesCache[this._activeChannel] = texture;\n if (texture) {\n texture._associatedChannel = this._activeChannel;\n }\n }\n else if (forTextureDataUpdate) {\n wasPreviouslyBound = true;\n this._activateCurrentTexture();\n }\n if (isTextureForRendering && !forTextureDataUpdate) {\n this._bindSamplerUniformToChannel(texture._associatedChannel, this._activeChannel);\n }\n return wasPreviouslyBound;\n }\n /**\n * @internal\n */\n _bindTexture(channel, texture, name) {\n if (channel === undefined) {\n return;\n }\n if (texture) {\n texture._associatedChannel = channel;\n }\n this._activeChannel = channel;\n const target = texture ? this._getTextureTarget(texture) : this._gl.TEXTURE_2D;\n this._bindTextureDirectly(target, texture);\n }\n /**\n * Unbind all textures from the webGL context\n */\n unbindAllTextures() {\n for (let channel = 0; channel < this._maxSimultaneousTextures; channel++) {\n this._activeChannel = channel;\n this._bindTextureDirectly(this._gl.TEXTURE_2D, null);\n this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, null);\n if (this.webGLVersion > 1) {\n this._bindTextureDirectly(this._gl.TEXTURE_3D, null);\n this._bindTextureDirectly(this._gl.TEXTURE_2D_ARRAY, null);\n }\n }\n }\n /**\n * Sets a texture to the according uniform.\n * @param channel The texture channel\n * @param uniform The uniform to set\n * @param texture The texture to apply\n * @param name The name of the uniform in the effect\n */\n setTexture(channel, uniform, texture, name) {\n if (channel === undefined) {\n return;\n }\n if (uniform) {\n this._boundUniforms[channel] = uniform;\n }\n this._setTexture(channel, texture);\n }\n _bindSamplerUniformToChannel(sourceSlot, destination) {\n const uniform = this._boundUniforms[sourceSlot];\n if (!uniform || uniform._currentState === destination) {\n return;\n }\n this._gl.uniform1i(uniform, destination);\n uniform._currentState = destination;\n }\n _getTextureWrapMode(mode) {\n switch (mode) {\n case 1:\n return this._gl.REPEAT;\n case 0:\n return this._gl.CLAMP_TO_EDGE;\n case 2:\n return this._gl.MIRRORED_REPEAT;\n }\n return this._gl.REPEAT;\n }\n _setTexture(channel, texture, isPartOfTextureArray = false, depthStencilTexture = false, name = \"\") {\n // Not ready?\n if (!texture) {\n if (this._boundTexturesCache[channel] != null) {\n this._activeChannel = channel;\n this._bindTextureDirectly(this._gl.TEXTURE_2D, null);\n this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, null);\n if (this.webGLVersion > 1) {\n this._bindTextureDirectly(this._gl.TEXTURE_3D, null);\n this._bindTextureDirectly(this._gl.TEXTURE_2D_ARRAY, null);\n }\n }\n return false;\n }\n // Video\n if (texture.video) {\n this._activeChannel = channel;\n const videoInternalTexture = texture.getInternalTexture();\n if (videoInternalTexture) {\n videoInternalTexture._associatedChannel = channel;\n }\n texture.update();\n }\n else if (texture.delayLoadState === 4) {\n // Delay loading\n texture.delayLoad();\n return false;\n }\n let internalTexture;\n if (depthStencilTexture) {\n internalTexture = texture.depthStencilTexture;\n }\n else if (texture.isReady()) {\n internalTexture = texture.getInternalTexture();\n }\n else if (texture.isCube) {\n internalTexture = this.emptyCubeTexture;\n }\n else if (texture.is3D) {\n internalTexture = this.emptyTexture3D;\n }\n else if (texture.is2DArray) {\n internalTexture = this.emptyTexture2DArray;\n }\n else {\n internalTexture = this.emptyTexture;\n }\n if (!isPartOfTextureArray && internalTexture) {\n internalTexture._associatedChannel = channel;\n }\n let needToBind = true;\n if (this._boundTexturesCache[channel] === internalTexture) {\n if (!isPartOfTextureArray) {\n this._bindSamplerUniformToChannel(internalTexture._associatedChannel, channel);\n }\n needToBind = false;\n }\n this._activeChannel = channel;\n const target = this._getTextureTarget(internalTexture);\n if (needToBind) {\n this._bindTextureDirectly(target, internalTexture, isPartOfTextureArray);\n }\n if (internalTexture && !internalTexture.isMultiview) {\n // CUBIC_MODE and SKYBOX_MODE both require CLAMP_TO_EDGE. All other modes use REPEAT.\n if (internalTexture.isCube && internalTexture._cachedCoordinatesMode !== texture.coordinatesMode) {\n internalTexture._cachedCoordinatesMode = texture.coordinatesMode;\n const textureWrapMode = texture.coordinatesMode !== 3 && texture.coordinatesMode !== 5\n ? 1\n : 0;\n texture.wrapU = textureWrapMode;\n texture.wrapV = textureWrapMode;\n }\n if (internalTexture._cachedWrapU !== texture.wrapU) {\n internalTexture._cachedWrapU = texture.wrapU;\n this._setTextureParameterInteger(target, this._gl.TEXTURE_WRAP_S, this._getTextureWrapMode(texture.wrapU), internalTexture);\n }\n if (internalTexture._cachedWrapV !== texture.wrapV) {\n internalTexture._cachedWrapV = texture.wrapV;\n this._setTextureParameterInteger(target, this._gl.TEXTURE_WRAP_T, this._getTextureWrapMode(texture.wrapV), internalTexture);\n }\n if (internalTexture.is3D && internalTexture._cachedWrapR !== texture.wrapR) {\n internalTexture._cachedWrapR = texture.wrapR;\n this._setTextureParameterInteger(target, this._gl.TEXTURE_WRAP_R, this._getTextureWrapMode(texture.wrapR), internalTexture);\n }\n this._setAnisotropicLevel(target, internalTexture, texture.anisotropicFilteringLevel);\n }\n return true;\n }\n /**\n * Sets an array of texture to the webGL context\n * @param channel defines the channel where the texture array must be set\n * @param uniform defines the associated uniform location\n * @param textures defines the array of textures to bind\n * @param name name of the channel\n */\n setTextureArray(channel, uniform, textures, name) {\n if (channel === undefined || !uniform) {\n return;\n }\n if (!this._textureUnits || this._textureUnits.length !== textures.length) {\n this._textureUnits = new Int32Array(textures.length);\n }\n for (let i = 0; i < textures.length; i++) {\n const texture = textures[i].getInternalTexture();\n if (texture) {\n this._textureUnits[i] = channel + i;\n texture._associatedChannel = channel + i;\n }\n else {\n this._textureUnits[i] = -1;\n }\n }\n this._gl.uniform1iv(uniform, this._textureUnits);\n for (let index = 0; index < textures.length; index++) {\n this._setTexture(this._textureUnits[index], textures[index], true);\n }\n }\n /**\n * @internal\n */\n _setAnisotropicLevel(target, internalTexture, anisotropicFilteringLevel) {\n const anisotropicFilterExtension = this._caps.textureAnisotropicFilterExtension;\n if (internalTexture.samplingMode !== 11 &&\n internalTexture.samplingMode !== 3 &&\n internalTexture.samplingMode !== 2) {\n anisotropicFilteringLevel = 1; // Forcing the anisotropic to 1 because else webgl will force filters to linear\n }\n if (anisotropicFilterExtension && internalTexture._cachedAnisotropicFilteringLevel !== anisotropicFilteringLevel) {\n this._setTextureParameterFloat(target, anisotropicFilterExtension.TEXTURE_MAX_ANISOTROPY_EXT, Math.min(anisotropicFilteringLevel, this._caps.maxAnisotropy), internalTexture);\n internalTexture._cachedAnisotropicFilteringLevel = anisotropicFilteringLevel;\n }\n }\n _setTextureParameterFloat(target, parameter, value, texture) {\n this._bindTextureDirectly(target, texture, true, true);\n this._gl.texParameterf(target, parameter, value);\n }\n _setTextureParameterInteger(target, parameter, value, texture) {\n if (texture) {\n this._bindTextureDirectly(target, texture, true, true);\n }\n this._gl.texParameteri(target, parameter, value);\n }\n /**\n * Unbind all vertex attributes from the webGL context\n */\n unbindAllAttributes() {\n if (this._mustWipeVertexAttributes) {\n this._mustWipeVertexAttributes = false;\n for (let i = 0; i < this._caps.maxVertexAttribs; i++) {\n this.disableAttributeByIndex(i);\n }\n return;\n }\n for (let i = 0, ul = this._vertexAttribArraysEnabled.length; i < ul; i++) {\n if (i >= this._caps.maxVertexAttribs || !this._vertexAttribArraysEnabled[i]) {\n continue;\n }\n this.disableAttributeByIndex(i);\n }\n }\n /**\n * Force the engine to release all cached effects. This means that next effect compilation will have to be done completely even if a similar effect was already compiled\n */\n releaseEffects() {\n for (const name in this._compiledEffects) {\n const webGLPipelineContext = this._compiledEffects[name].getPipelineContext();\n this._deletePipelineContext(webGLPipelineContext);\n }\n this._compiledEffects = {};\n }\n /**\n * Dispose and release all associated resources\n */\n dispose() {\n var _a;\n this._isDisposed = true;\n this.stopRenderLoop();\n // Clear observables\n if (this.onBeforeTextureInitObservable) {\n this.onBeforeTextureInitObservable.clear();\n }\n // Empty texture\n if (this._emptyTexture) {\n this._releaseTexture(this._emptyTexture);\n this._emptyTexture = null;\n }\n if (this._emptyCubeTexture) {\n this._releaseTexture(this._emptyCubeTexture);\n this._emptyCubeTexture = null;\n }\n if (this._dummyFramebuffer) {\n this._gl.deleteFramebuffer(this._dummyFramebuffer);\n }\n // Release effects\n this.releaseEffects();\n (_a = this.releaseComputeEffects) === null || _a === void 0 ? void 0 : _a.call(this);\n // Unbind\n this.unbindAllAttributes();\n this._boundUniforms = {};\n // Events\n if (IsWindowObjectExist()) {\n if (this._renderingCanvas) {\n if (!this._doNotHandleContextLost) {\n this._renderingCanvas.removeEventListener(\"webglcontextlost\", this._onContextLost);\n this._renderingCanvas.removeEventListener(\"webglcontextrestored\", this._onContextRestored);\n }\n window.removeEventListener(\"resize\", this._checkForMobile);\n }\n }\n this._workingCanvas = null;\n this._workingContext = null;\n this._currentBufferPointers.length = 0;\n this._renderingCanvas = null;\n this._currentProgram = null;\n this._boundRenderFunction = null;\n Effect.ResetCache();\n // Abort active requests\n for (const request of this._activeRequests) {\n request.abort();\n }\n this.onDisposeObservable.notifyObservers(this);\n this.onDisposeObservable.clear();\n }\n /**\n * Attach a new callback raised when context lost event is fired\n * @param callback defines the callback to call\n */\n attachContextLostEvent(callback) {\n if (this._renderingCanvas) {\n this._renderingCanvas.addEventListener(\"webglcontextlost\", callback, false);\n }\n }\n /**\n * Attach a new callback raised when context restored event is fired\n * @param callback defines the callback to call\n */\n attachContextRestoredEvent(callback) {\n if (this._renderingCanvas) {\n this._renderingCanvas.addEventListener(\"webglcontextrestored\", callback, false);\n }\n }\n /**\n * Get the current error code of the webGL context\n * @returns the error code\n * @see https://developer.mozilla.org/en-US/docs/Web/API/WebGLRenderingContext/getError\n */\n getError() {\n return this._gl.getError();\n }\n _canRenderToFloatFramebuffer() {\n if (this._webGLVersion > 1) {\n return this._caps.colorBufferFloat;\n }\n return this._canRenderToFramebuffer(1);\n }\n _canRenderToHalfFloatFramebuffer() {\n if (this._webGLVersion > 1) {\n return this._caps.colorBufferFloat;\n }\n return this._canRenderToFramebuffer(2);\n }\n // Thank you : http://stackoverflow.com/questions/28827511/webgl-ios-render-to-floating-point-texture\n _canRenderToFramebuffer(type) {\n const gl = this._gl;\n //clear existing errors\n // eslint-disable-next-line no-empty\n while (gl.getError() !== gl.NO_ERROR) { }\n let successful = true;\n const texture = gl.createTexture();\n gl.bindTexture(gl.TEXTURE_2D, texture);\n gl.texImage2D(gl.TEXTURE_2D, 0, this._getRGBABufferInternalSizedFormat(type), 1, 1, 0, gl.RGBA, this._getWebGLTextureType(type), null);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST);\n const fb = gl.createFramebuffer();\n gl.bindFramebuffer(gl.FRAMEBUFFER, fb);\n gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, texture, 0);\n const status = gl.checkFramebufferStatus(gl.FRAMEBUFFER);\n successful = successful && status === gl.FRAMEBUFFER_COMPLETE;\n successful = successful && gl.getError() === gl.NO_ERROR;\n //try render by clearing frame buffer's color buffer\n if (successful) {\n gl.clear(gl.COLOR_BUFFER_BIT);\n successful = successful && gl.getError() === gl.NO_ERROR;\n }\n //try reading from frame to ensure render occurs (just creating the FBO is not sufficient to determine if rendering is supported)\n if (successful) {\n //in practice it's sufficient to just read from the backbuffer rather than handle potentially issues reading from the texture\n gl.bindFramebuffer(gl.FRAMEBUFFER, null);\n const readFormat = gl.RGBA;\n const readType = gl.UNSIGNED_BYTE;\n const buffer = new Uint8Array(4);\n gl.readPixels(0, 0, 1, 1, readFormat, readType, buffer);\n successful = successful && gl.getError() === gl.NO_ERROR;\n }\n //clean up\n gl.deleteTexture(texture);\n gl.deleteFramebuffer(fb);\n gl.bindFramebuffer(gl.FRAMEBUFFER, null);\n //clear accumulated errors\n // eslint-disable-next-line no-empty\n while (!successful && gl.getError() !== gl.NO_ERROR) { }\n return successful;\n }\n /**\n * @internal\n */\n _getWebGLTextureType(type) {\n if (this._webGLVersion === 1) {\n switch (type) {\n case 1:\n return this._gl.FLOAT;\n case 2:\n return this._gl.HALF_FLOAT_OES;\n case 0:\n return this._gl.UNSIGNED_BYTE;\n case 8:\n return this._gl.UNSIGNED_SHORT_4_4_4_4;\n case 9:\n return this._gl.UNSIGNED_SHORT_5_5_5_1;\n case 10:\n return this._gl.UNSIGNED_SHORT_5_6_5;\n }\n return this._gl.UNSIGNED_BYTE;\n }\n switch (type) {\n case 3:\n return this._gl.BYTE;\n case 0:\n return this._gl.UNSIGNED_BYTE;\n case 4:\n return this._gl.SHORT;\n case 5:\n return this._gl.UNSIGNED_SHORT;\n case 6:\n return this._gl.INT;\n case 7: // Refers to UNSIGNED_INT\n return this._gl.UNSIGNED_INT;\n case 1:\n return this._gl.FLOAT;\n case 2:\n return this._gl.HALF_FLOAT;\n case 8:\n return this._gl.UNSIGNED_SHORT_4_4_4_4;\n case 9:\n return this._gl.UNSIGNED_SHORT_5_5_5_1;\n case 10:\n return this._gl.UNSIGNED_SHORT_5_6_5;\n case 11:\n return this._gl.UNSIGNED_INT_2_10_10_10_REV;\n case 12:\n return this._gl.UNSIGNED_INT_24_8;\n case 13:\n return this._gl.UNSIGNED_INT_10F_11F_11F_REV;\n case 14:\n return this._gl.UNSIGNED_INT_5_9_9_9_REV;\n case 15:\n return this._gl.FLOAT_32_UNSIGNED_INT_24_8_REV;\n }\n return this._gl.UNSIGNED_BYTE;\n }\n /**\n * @internal\n */\n _getInternalFormat(format, useSRGBBuffer = false) {\n let internalFormat = useSRGBBuffer ? this._glSRGBExtensionValues.SRGB8_ALPHA8 : this._gl.RGBA;\n switch (format) {\n case 0:\n internalFormat = this._gl.ALPHA;\n break;\n case 1:\n internalFormat = this._gl.LUMINANCE;\n break;\n case 2:\n internalFormat = this._gl.LUMINANCE_ALPHA;\n break;\n case 6:\n internalFormat = this._gl.RED;\n break;\n case 7:\n internalFormat = this._gl.RG;\n break;\n case 4:\n internalFormat = useSRGBBuffer ? this._glSRGBExtensionValues.SRGB : this._gl.RGB;\n break;\n case 5:\n internalFormat = useSRGBBuffer ? this._glSRGBExtensionValues.SRGB8_ALPHA8 : this._gl.RGBA;\n break;\n }\n if (this._webGLVersion > 1) {\n switch (format) {\n case 8:\n internalFormat = this._gl.RED_INTEGER;\n break;\n case 9:\n internalFormat = this._gl.RG_INTEGER;\n break;\n case 10:\n internalFormat = this._gl.RGB_INTEGER;\n break;\n case 11:\n internalFormat = this._gl.RGBA_INTEGER;\n break;\n }\n }\n return internalFormat;\n }\n /**\n * @internal\n */\n _getRGBABufferInternalSizedFormat(type, format, useSRGBBuffer = false) {\n if (this._webGLVersion === 1) {\n if (format !== undefined) {\n switch (format) {\n case 0:\n return this._gl.ALPHA;\n case 1:\n return this._gl.LUMINANCE;\n case 2:\n return this._gl.LUMINANCE_ALPHA;\n case 4:\n return useSRGBBuffer ? this._glSRGBExtensionValues.SRGB : this._gl.RGB;\n }\n }\n return this._gl.RGBA;\n }\n switch (type) {\n case 3:\n switch (format) {\n case 6:\n return this._gl.R8_SNORM;\n case 7:\n return this._gl.RG8_SNORM;\n case 4:\n return this._gl.RGB8_SNORM;\n case 8:\n return this._gl.R8I;\n case 9:\n return this._gl.RG8I;\n case 10:\n return this._gl.RGB8I;\n case 11:\n return this._gl.RGBA8I;\n default:\n return this._gl.RGBA8_SNORM;\n }\n case 0:\n switch (format) {\n case 6:\n return this._gl.R8;\n case 7:\n return this._gl.RG8;\n case 4:\n return useSRGBBuffer ? this._glSRGBExtensionValues.SRGB8 : this._gl.RGB8; // By default. Other possibilities are RGB565, SRGB8.\n case 5:\n return useSRGBBuffer ? this._glSRGBExtensionValues.SRGB8_ALPHA8 : this._gl.RGBA8; // By default. Other possibilities are RGB5_A1, RGBA4, SRGB8_ALPHA8.\n case 8:\n return this._gl.R8UI;\n case 9:\n return this._gl.RG8UI;\n case 10:\n return this._gl.RGB8UI;\n case 11:\n return this._gl.RGBA8UI;\n case 0:\n return this._gl.ALPHA;\n case 1:\n return this._gl.LUMINANCE;\n case 2:\n return this._gl.LUMINANCE_ALPHA;\n default:\n return this._gl.RGBA8;\n }\n case 4:\n switch (format) {\n case 8:\n return this._gl.R16I;\n case 9:\n return this._gl.RG16I;\n case 10:\n return this._gl.RGB16I;\n case 11:\n return this._gl.RGBA16I;\n default:\n return this._gl.RGBA16I;\n }\n case 5:\n switch (format) {\n case 8:\n return this._gl.R16UI;\n case 9:\n return this._gl.RG16UI;\n case 10:\n return this._gl.RGB16UI;\n case 11:\n return this._gl.RGBA16UI;\n default:\n return this._gl.RGBA16UI;\n }\n case 6:\n switch (format) {\n case 8:\n return this._gl.R32I;\n case 9:\n return this._gl.RG32I;\n case 10:\n return this._gl.RGB32I;\n case 11:\n return this._gl.RGBA32I;\n default:\n return this._gl.RGBA32I;\n }\n case 7: // Refers to UNSIGNED_INT\n switch (format) {\n case 8:\n return this._gl.R32UI;\n case 9:\n return this._gl.RG32UI;\n case 10:\n return this._gl.RGB32UI;\n case 11:\n return this._gl.RGBA32UI;\n default:\n return this._gl.RGBA32UI;\n }\n case 1:\n switch (format) {\n case 6:\n return this._gl.R32F; // By default. Other possibility is R16F.\n case 7:\n return this._gl.RG32F; // By default. Other possibility is RG16F.\n case 4:\n return this._gl.RGB32F; // By default. Other possibilities are RGB16F, R11F_G11F_B10F, RGB9_E5.\n case 5:\n return this._gl.RGBA32F; // By default. Other possibility is RGBA16F.\n default:\n return this._gl.RGBA32F;\n }\n case 2:\n switch (format) {\n case 6:\n return this._gl.R16F;\n case 7:\n return this._gl.RG16F;\n case 4:\n return this._gl.RGB16F; // By default. Other possibilities are R11F_G11F_B10F, RGB9_E5.\n case 5:\n return this._gl.RGBA16F;\n default:\n return this._gl.RGBA16F;\n }\n case 10:\n return this._gl.RGB565;\n case 13:\n return this._gl.R11F_G11F_B10F;\n case 14:\n return this._gl.RGB9_E5;\n case 8:\n return this._gl.RGBA4;\n case 9:\n return this._gl.RGB5_A1;\n case 11:\n switch (format) {\n case 5:\n return this._gl.RGB10_A2; // By default. Other possibility is RGB5_A1.\n case 11:\n return this._gl.RGB10_A2UI;\n default:\n return this._gl.RGB10_A2;\n }\n }\n return useSRGBBuffer ? this._glSRGBExtensionValues.SRGB8_ALPHA8 : this._gl.RGBA8;\n }\n /**\n * @internal\n */\n _getRGBAMultiSampleBufferFormat(type, format = 5) {\n switch (type) {\n case 1:\n switch (format) {\n case 6:\n return this._gl.R32F;\n default:\n return this._gl.RGBA32F;\n }\n case 2:\n switch (format) {\n case 6:\n return this._gl.R16F;\n default:\n return this._gl.RGBA16F;\n }\n }\n return this._gl.RGBA8;\n }\n /**\n * @internal\n */\n _loadFile(url, onSuccess, onProgress, offlineProvider, useArrayBuffer, onError) {\n const request = ThinEngine._FileToolsLoadFile(url, onSuccess, onProgress, offlineProvider, useArrayBuffer, onError);\n this._activeRequests.push(request);\n request.onCompleteObservable.add((request) => {\n this._activeRequests.splice(this._activeRequests.indexOf(request), 1);\n });\n return request;\n }\n /**\n * Loads a file from a url\n * @param url url to load\n * @param onSuccess callback called when the file successfully loads\n * @param onProgress callback called while file is loading (if the server supports this mode)\n * @param offlineProvider defines the offline provider for caching\n * @param useArrayBuffer defines a boolean indicating that date must be returned as ArrayBuffer\n * @param onError callback called when the file fails to load\n * @returns a file request object\n * @internal\n */\n static _FileToolsLoadFile(url, onSuccess, onProgress, offlineProvider, useArrayBuffer, onError) {\n throw _WarnImport(\"FileTools\");\n }\n /**\n * Reads pixels from the current frame buffer. Please note that this function can be slow\n * @param x defines the x coordinate of the rectangle where pixels must be read\n * @param y defines the y coordinate of the rectangle where pixels must be read\n * @param width defines the width of the rectangle where pixels must be read\n * @param height defines the height of the rectangle where pixels must be read\n * @param hasAlpha defines whether the output should have alpha or not (defaults to true)\n * @param flushRenderer true to flush the renderer from the pending commands before reading the pixels\n * @returns a ArrayBufferView promise (Uint8Array) containing RGBA colors\n */\n readPixels(x, y, width, height, hasAlpha = true, flushRenderer = true) {\n const numChannels = hasAlpha ? 4 : 3;\n const format = hasAlpha ? this._gl.RGBA : this._gl.RGB;\n const data = new Uint8Array(height * width * numChannels);\n if (flushRenderer) {\n this.flushFramebuffer();\n }\n this._gl.readPixels(x, y, width, height, format, this._gl.UNSIGNED_BYTE, data);\n return Promise.resolve(data);\n }\n /**\n * Gets a Promise indicating if the engine can be instantiated (ie. if a webGL context can be found)\n */\n static get IsSupportedAsync() {\n return Promise.resolve(this.isSupported());\n }\n /**\n * Gets a boolean indicating if the engine can be instantiated (ie. if a webGL context can be found)\n */\n static get IsSupported() {\n return this.isSupported(); // Backward compat\n }\n /**\n * Gets a boolean indicating if the engine can be instantiated (ie. if a webGL context can be found)\n * @returns true if the engine can be created\n * @ignorenaming\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n static isSupported() {\n if (this._HasMajorPerformanceCaveat !== null) {\n return !this._HasMajorPerformanceCaveat; // We know it is performant so WebGL is supported\n }\n if (this._IsSupported === null) {\n try {\n const tempcanvas = this._CreateCanvas(1, 1);\n const gl = tempcanvas.getContext(\"webgl\") || tempcanvas.getContext(\"experimental-webgl\");\n this._IsSupported = gl != null && !!window.WebGLRenderingContext;\n }\n catch (e) {\n this._IsSupported = false;\n }\n }\n return this._IsSupported;\n }\n /**\n * Gets a boolean indicating if the engine can be instantiated on a performant device (ie. if a webGL context can be found and it does not use a slow implementation)\n */\n static get HasMajorPerformanceCaveat() {\n if (this._HasMajorPerformanceCaveat === null) {\n try {\n const tempcanvas = this._CreateCanvas(1, 1);\n const gl = tempcanvas.getContext(\"webgl\", { failIfMajorPerformanceCaveat: true }) ||\n tempcanvas.getContext(\"experimental-webgl\", { failIfMajorPerformanceCaveat: true });\n this._HasMajorPerformanceCaveat = !gl;\n }\n catch (e) {\n this._HasMajorPerformanceCaveat = false;\n }\n }\n return this._HasMajorPerformanceCaveat;\n }\n /**\n * Find the next highest power of two.\n * @param x Number to start search from.\n * @returns Next highest power of two.\n */\n static CeilingPOT(x) {\n x--;\n x |= x >> 1;\n x |= x >> 2;\n x |= x >> 4;\n x |= x >> 8;\n x |= x >> 16;\n x++;\n return x;\n }\n /**\n * Find the next lowest power of two.\n * @param x Number to start search from.\n * @returns Next lowest power of two.\n */\n static FloorPOT(x) {\n x = x | (x >> 1);\n x = x | (x >> 2);\n x = x | (x >> 4);\n x = x | (x >> 8);\n x = x | (x >> 16);\n return x - (x >> 1);\n }\n /**\n * Find the nearest power of two.\n * @param x Number to start search from.\n * @returns Next nearest power of two.\n */\n static NearestPOT(x) {\n const c = ThinEngine.CeilingPOT(x);\n const f = ThinEngine.FloorPOT(x);\n return c - x > x - f ? f : c;\n }\n /**\n * Get the closest exponent of two\n * @param value defines the value to approximate\n * @param max defines the maximum value to return\n * @param mode defines how to define the closest value\n * @returns closest exponent of two of the given value\n */\n static GetExponentOfTwo(value, max, mode = 2) {\n let pot;\n switch (mode) {\n case 1:\n pot = ThinEngine.FloorPOT(value);\n break;\n case 2:\n pot = ThinEngine.NearestPOT(value);\n break;\n case 3:\n default:\n pot = ThinEngine.CeilingPOT(value);\n break;\n }\n return Math.min(pot, max);\n }\n /**\n * Queue a new function into the requested animation frame pool (ie. this function will be executed by the browser (or the javascript engine) for the next frame)\n * @param func - the function to be called\n * @param requester - the object that will request the next frame. Falls back to window.\n * @returns frame number\n */\n static QueueNewFrame(func, requester) {\n // Note that there is kind of a typing issue here, as `setTimeout` might return something else than a number (NodeJs returns a NodeJS.Timeout object).\n // Also if the global `requestAnimationFrame`'s returnType is number, `requester.requestPostAnimationFrame` and `requester.requestAnimationFrame` types\n // are `any`.\n if (!IsWindowObjectExist()) {\n if (typeof requestAnimationFrame === \"function\") {\n return requestAnimationFrame(func);\n }\n }\n else {\n const { requestAnimationFrame } = requester || window;\n if (typeof requestAnimationFrame === \"function\") {\n return requestAnimationFrame(func);\n }\n }\n // fallback to the global `setTimeout`.\n // In most cases (aka in the browser), `window` is the global object, so instead of calling `window.setTimeout` we could call the global `setTimeout`.\n return setTimeout(func, 16);\n }\n /**\n * Gets host document\n * @returns the host document object\n */\n getHostDocument() {\n if (this._renderingCanvas && this._renderingCanvas.ownerDocument) {\n return this._renderingCanvas.ownerDocument;\n }\n return IsDocumentAvailable() ? document : null;\n }\n }\n ThinEngine._TempClearColorUint32 = new Uint32Array(4);\n ThinEngine._TempClearColorInt32 = new Int32Array(4);\n /** Use this array to turn off some WebGL2 features on known buggy browsers version */\n ThinEngine.ExceptionList = [\n { key: \"Chrome/63.0\", capture: \"63\\\\.0\\\\.3239\\\\.(\\\\d+)\", captureConstraint: 108, targets: [\"uniformBuffer\"] },\n { key: \"Firefox/58\", capture: null, captureConstraint: null, targets: [\"uniformBuffer\"] },\n { key: \"Firefox/59\", capture: null, captureConstraint: null, targets: [\"uniformBuffer\"] },\n { key: \"Chrome/72.+?Mobile\", capture: null, captureConstraint: null, targets: [\"vao\"] },\n { key: \"Chrome/73.+?Mobile\", capture: null, captureConstraint: null, targets: [\"vao\"] },\n { key: \"Chrome/74.+?Mobile\", capture: null, captureConstraint: null, targets: [\"vao\"] },\n { key: \"Mac OS.+Chrome/71\", capture: null, captureConstraint: null, targets: [\"vao\"] },\n { key: \"Mac OS.+Chrome/72\", capture: null, captureConstraint: null, targets: [\"vao\"] },\n { key: \"Mac OS.+Chrome\", capture: null, captureConstraint: null, targets: [\"uniformBuffer\"] },\n // desktop osx safari 15.4\n { key: \".*AppleWebKit.*(15.4).*Safari\", capture: null, captureConstraint: null, targets: [\"antialias\", \"maxMSAASamples\"] },\n // mobile browsers using safari 15.4 on ios\n { key: \".*(15.4).*AppleWebKit.*Safari\", capture: null, captureConstraint: null, targets: [\"antialias\", \"maxMSAASamples\"] },\n ];\n /** @internal */\n ThinEngine._TextureLoaders = [];\n // Updatable statics so stick with vars here\n /**\n * Gets or sets the epsilon value used by collision engine\n */\n ThinEngine.CollisionsEpsilon = 0.001;\n // Statics\n ThinEngine._IsSupported = null;\n ThinEngine._HasMajorPerformanceCaveat = null;\n\n /**\n * Class used to provide helper for timing\n */\n class TimingTools {\n /**\n * Polyfill for setImmediate\n * @param action defines the action to execute after the current execution block\n */\n static SetImmediate(action) {\n if (IsWindowObjectExist() && window.setImmediate) {\n window.setImmediate(action);\n }\n else {\n setTimeout(action, 1);\n }\n }\n }\n\n /* eslint-disable @typescript-eslint/naming-convention */\n const Base64DataUrlRegEx = new RegExp(/^data:([^,]+\\/[^,]+)?;base64,/i);\n /** @ignore */\n class LoadFileError extends RuntimeError {\n /**\n * Creates a new LoadFileError\n * @param message defines the message of the error\n * @param object defines the optional web request\n */\n constructor(message, object) {\n super(message, ErrorCodes.LoadFileError);\n this.name = \"LoadFileError\";\n BaseError._setPrototypeOf(this, LoadFileError.prototype);\n if (object instanceof WebRequest) {\n this.request = object;\n }\n else {\n this.file = object;\n }\n }\n }\n /** @ignore */\n class RequestFileError extends RuntimeError {\n /**\n * Creates a new LoadFileError\n * @param message defines the message of the error\n * @param request defines the optional web request\n */\n constructor(message, request) {\n super(message, ErrorCodes.RequestFileError);\n this.request = request;\n this.name = \"RequestFileError\";\n BaseError._setPrototypeOf(this, RequestFileError.prototype);\n }\n }\n /** @ignore */\n class ReadFileError extends RuntimeError {\n /**\n * Creates a new ReadFileError\n * @param message defines the message of the error\n * @param file defines the optional file\n */\n constructor(message, file) {\n super(message, ErrorCodes.ReadFileError);\n this.file = file;\n this.name = \"ReadFileError\";\n BaseError._setPrototypeOf(this, ReadFileError.prototype);\n }\n }\n /**\n * @internal\n */\n const FileToolsOptions = {\n /**\n * Gets or sets the retry strategy to apply when an error happens while loading an asset.\n * When defining this function, return the wait time before trying again or return -1 to\n * stop retrying and error out.\n */\n DefaultRetryStrategy: RetryStrategy.ExponentialBackoff(),\n /**\n * Gets or sets the base URL to use to load assets\n */\n BaseUrl: \"\",\n /**\n * Default behaviour for cors in the application.\n * It can be a string if the expected behavior is identical in the entire app.\n * Or a callback to be able to set it per url or on a group of them (in case of Video source for instance)\n */\n CorsBehavior: \"anonymous\",\n /**\n * Gets or sets a function used to pre-process url before using them to load assets\n * @param url\n */\n PreprocessUrl: (url) => {\n return url;\n },\n };\n /**\n * Removes unwanted characters from an url\n * @param url defines the url to clean\n * @returns the cleaned url\n */\n const _CleanUrl = (url) => {\n url = url.replace(/#/gm, \"%23\");\n return url;\n };\n /**\n * Sets the cors behavior on a dom element. This will add the required Tools.CorsBehavior to the element.\n * @param url define the url we are trying\n * @param element define the dom element where to configure the cors policy\n * @internal\n */\n const SetCorsBehavior = (url, element) => {\n if (url && url.indexOf(\"data:\") === 0) {\n return;\n }\n if (FileToolsOptions.CorsBehavior) {\n if (typeof FileToolsOptions.CorsBehavior === \"string\" || FileToolsOptions.CorsBehavior instanceof String) {\n element.crossOrigin = FileToolsOptions.CorsBehavior;\n }\n else {\n const result = FileToolsOptions.CorsBehavior(url);\n if (result) {\n element.crossOrigin = result;\n }\n }\n }\n };\n /**\n * Loads an image as an HTMLImageElement.\n * @param input url string, ArrayBuffer, or Blob to load\n * @param onLoad callback called when the image successfully loads\n * @param onError callback called when the image fails to load\n * @param offlineProvider offline provider for caching\n * @param mimeType optional mime type\n * @param imageBitmapOptions\n * @returns the HTMLImageElement of the loaded image\n * @internal\n */\n const LoadImage = (input, onLoad, onError, offlineProvider, mimeType = \"\", imageBitmapOptions) => {\n var _a;\n let url;\n let usingObjectURL = false;\n if (input instanceof ArrayBuffer || ArrayBuffer.isView(input)) {\n if (typeof Blob !== \"undefined\" && typeof URL !== \"undefined\") {\n url = URL.createObjectURL(new Blob([input], { type: mimeType }));\n usingObjectURL = true;\n }\n else {\n url = `data:${mimeType};base64,` + EncodeArrayBufferToBase64(input);\n }\n }\n else if (input instanceof Blob) {\n url = URL.createObjectURL(input);\n usingObjectURL = true;\n }\n else {\n url = _CleanUrl(input);\n url = FileToolsOptions.PreprocessUrl(input);\n }\n const engine = EngineStore.LastCreatedEngine;\n const onErrorHandler = (exception) => {\n if (onError) {\n const inputText = url || input.toString();\n onError(`Error while trying to load image: ${inputText.indexOf(\"http\") === 0 || inputText.length <= 128 ? inputText : inputText.slice(0, 128) + \"...\"}`, exception);\n }\n };\n if (typeof Image === \"undefined\" || ((_a = engine === null || engine === void 0 ? void 0 : engine._features.forceBitmapOverHTMLImageElement) !== null && _a !== void 0 ? _a : false)) {\n LoadFile(url, (data) => {\n engine\n .createImageBitmap(new Blob([data], { type: mimeType }), Object.assign({ premultiplyAlpha: \"none\" }, imageBitmapOptions))\n .then((imgBmp) => {\n onLoad(imgBmp);\n if (usingObjectURL) {\n URL.revokeObjectURL(url);\n }\n })\n .catch((reason) => {\n if (onError) {\n onError(\"Error while trying to load image: \" + input, reason);\n }\n });\n }, undefined, offlineProvider || undefined, true, (request, exception) => {\n onErrorHandler(exception);\n });\n return null;\n }\n const img = new Image();\n SetCorsBehavior(url, img);\n const handlersList = [];\n const loadHandlersList = () => {\n handlersList.forEach((handler) => {\n handler.target.addEventListener(handler.name, handler.handler);\n });\n };\n const unloadHandlersList = () => {\n handlersList.forEach((handler) => {\n handler.target.removeEventListener(handler.name, handler.handler);\n });\n handlersList.length = 0;\n };\n const loadHandler = () => {\n unloadHandlersList();\n onLoad(img);\n // Must revoke the URL after calling onLoad to avoid security exceptions in\n // certain scenarios (e.g. when hosted in vscode).\n if (usingObjectURL && img.src) {\n URL.revokeObjectURL(img.src);\n }\n };\n const errorHandler = (err) => {\n unloadHandlersList();\n onErrorHandler(err);\n if (usingObjectURL && img.src) {\n URL.revokeObjectURL(img.src);\n }\n };\n const cspHandler = (err) => {\n if (err.blockedURI !== img.src) {\n return;\n }\n unloadHandlersList();\n const cspException = new Error(`CSP violation of policy ${err.effectiveDirective} ${err.blockedURI}. Current policy is ${err.originalPolicy}`);\n EngineStore.UseFallbackTexture = false;\n onErrorHandler(cspException);\n if (usingObjectURL && img.src) {\n URL.revokeObjectURL(img.src);\n }\n img.src = \"\";\n };\n handlersList.push({ target: img, name: \"load\", handler: loadHandler });\n handlersList.push({ target: img, name: \"error\", handler: errorHandler });\n handlersList.push({ target: document, name: \"securitypolicyviolation\", handler: cspHandler });\n loadHandlersList();\n const fromBlob = url.substring(0, 5) === \"blob:\";\n const fromData = url.substring(0, 5) === \"data:\";\n const noOfflineSupport = () => {\n if (fromBlob || fromData) {\n img.src = url;\n }\n else {\n LoadFile(url, (data, _, contentType) => {\n const type = !mimeType && contentType ? contentType : mimeType;\n const blob = new Blob([data], { type });\n const url = URL.createObjectURL(blob);\n usingObjectURL = true;\n img.src = url;\n }, undefined, offlineProvider || undefined, true, (request, exception) => {\n onErrorHandler(exception);\n });\n }\n };\n const loadFromOfflineSupport = () => {\n if (offlineProvider) {\n offlineProvider.loadImage(url, img);\n }\n };\n if (!fromBlob && !fromData && offlineProvider && offlineProvider.enableTexturesOffline) {\n offlineProvider.open(loadFromOfflineSupport, noOfflineSupport);\n }\n else {\n if (url.indexOf(\"file:\") !== -1) {\n const textureName = decodeURIComponent(url.substring(5).toLowerCase());\n if (FilesInputStore.FilesToLoad[textureName] && typeof URL !== \"undefined\") {\n try {\n let blobURL;\n try {\n blobURL = URL.createObjectURL(FilesInputStore.FilesToLoad[textureName]);\n }\n catch (ex) {\n // Chrome doesn't support oneTimeOnly parameter\n blobURL = URL.createObjectURL(FilesInputStore.FilesToLoad[textureName]);\n }\n img.src = blobURL;\n usingObjectURL = true;\n }\n catch (e) {\n img.src = \"\";\n }\n return img;\n }\n }\n noOfflineSupport();\n }\n return img;\n };\n /**\n * Reads a file from a File object\n * @param file defines the file to load\n * @param onSuccess defines the callback to call when data is loaded\n * @param onProgress defines the callback to call during loading process\n * @param useArrayBuffer defines a boolean indicating that data must be returned as an ArrayBuffer\n * @param onError defines the callback to call when an error occurs\n * @returns a file request object\n * @internal\n */\n const ReadFile = (file, onSuccess, onProgress, useArrayBuffer, onError) => {\n const reader = new FileReader();\n const fileRequest = {\n onCompleteObservable: new Observable$1(),\n abort: () => reader.abort(),\n };\n reader.onloadend = () => fileRequest.onCompleteObservable.notifyObservers(fileRequest);\n if (onError) {\n reader.onerror = () => {\n onError(new ReadFileError(`Unable to read ${file.name}`, file));\n };\n }\n reader.onload = (e) => {\n //target doesn't have result from ts 1.3\n onSuccess(e.target[\"result\"]);\n };\n if (onProgress) {\n reader.onprogress = onProgress;\n }\n if (!useArrayBuffer) {\n // Asynchronous read\n reader.readAsText(file);\n }\n else {\n reader.readAsArrayBuffer(file);\n }\n return fileRequest;\n };\n /**\n * Loads a file from a url, a data url, or a file url\n * @param fileOrUrl file, url, data url, or file url to load\n * @param onSuccess callback called when the file successfully loads\n * @param onProgress callback called while file is loading (if the server supports this mode)\n * @param offlineProvider defines the offline provider for caching\n * @param useArrayBuffer defines a boolean indicating that date must be returned as ArrayBuffer\n * @param onError callback called when the file fails to load\n * @param onOpened\n * @returns a file request object\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n const LoadFile = (fileOrUrl, onSuccess, onProgress, offlineProvider, useArrayBuffer, onError, onOpened) => {\n if (fileOrUrl.name) {\n return ReadFile(fileOrUrl, onSuccess, onProgress, useArrayBuffer, onError\n ? (error) => {\n onError(undefined, error);\n }\n : undefined);\n }\n const url = fileOrUrl;\n // If file and file input are set\n if (url.indexOf(\"file:\") !== -1) {\n let fileName = decodeURIComponent(url.substring(5).toLowerCase());\n if (fileName.indexOf(\"./\") === 0) {\n fileName = fileName.substring(2);\n }\n const file = FilesInputStore.FilesToLoad[fileName];\n if (file) {\n return ReadFile(file, onSuccess, onProgress, useArrayBuffer, onError ? (error) => onError(undefined, new LoadFileError(error.message, error.file)) : undefined);\n }\n }\n // For a Base64 Data URL\n const { match, type } = TestBase64DataUrl(url);\n if (match) {\n const fileRequest = {\n onCompleteObservable: new Observable$1(),\n abort: () => () => { },\n };\n try {\n const data = useArrayBuffer ? DecodeBase64UrlToBinary(url) : DecodeBase64UrlToString(url);\n onSuccess(data, undefined, type);\n }\n catch (error) {\n if (onError) {\n onError(undefined, error);\n }\n else {\n Logger.Error(error.message || \"Failed to parse the Data URL\");\n }\n }\n TimingTools.SetImmediate(() => {\n fileRequest.onCompleteObservable.notifyObservers(fileRequest);\n });\n return fileRequest;\n }\n return RequestFile(url, (data, request) => {\n onSuccess(data, request === null || request === void 0 ? void 0 : request.responseURL, request === null || request === void 0 ? void 0 : request.getResponseHeader(\"content-type\"));\n }, onProgress, offlineProvider, useArrayBuffer, onError\n ? (error) => {\n onError(error.request, new LoadFileError(error.message, error.request));\n }\n : undefined, onOpened);\n };\n /**\n * Loads a file from a url\n * @param url url to load\n * @param onSuccess callback called when the file successfully loads\n * @param onProgress callback called while file is loading (if the server supports this mode)\n * @param offlineProvider defines the offline provider for caching\n * @param useArrayBuffer defines a boolean indicating that date must be returned as ArrayBuffer\n * @param onError callback called when the file fails to load\n * @param onOpened callback called when the web request is opened\n * @returns a file request object\n * @internal\n */\n const RequestFile = (url, onSuccess, onProgress, offlineProvider, useArrayBuffer, onError, onOpened) => {\n url = _CleanUrl(url);\n url = FileToolsOptions.PreprocessUrl(url);\n const loadUrl = FileToolsOptions.BaseUrl + url;\n let aborted = false;\n const fileRequest = {\n onCompleteObservable: new Observable$1(),\n abort: () => (aborted = true),\n };\n const requestFile = () => {\n let request = new WebRequest();\n let retryHandle = null;\n let onReadyStateChange;\n const unbindEvents = () => {\n if (!request) {\n return;\n }\n if (onProgress) {\n request.removeEventListener(\"progress\", onProgress);\n }\n if (onReadyStateChange) {\n request.removeEventListener(\"readystatechange\", onReadyStateChange);\n }\n request.removeEventListener(\"loadend\", onLoadEnd);\n };\n let onLoadEnd = () => {\n unbindEvents();\n fileRequest.onCompleteObservable.notifyObservers(fileRequest);\n fileRequest.onCompleteObservable.clear();\n onProgress = undefined;\n onReadyStateChange = null;\n onLoadEnd = null;\n onError = undefined;\n onOpened = undefined;\n onSuccess = undefined;\n };\n fileRequest.abort = () => {\n aborted = true;\n if (onLoadEnd) {\n onLoadEnd();\n }\n if (request && request.readyState !== (XMLHttpRequest.DONE || 4)) {\n request.abort();\n }\n if (retryHandle !== null) {\n clearTimeout(retryHandle);\n retryHandle = null;\n }\n request = null;\n };\n const handleError = (error) => {\n const message = error.message || \"Unknown error\";\n if (onError && request) {\n onError(new RequestFileError(message, request));\n }\n else {\n Logger.Error(message);\n }\n };\n const retryLoop = (retryIndex) => {\n if (!request) {\n return;\n }\n request.open(\"GET\", loadUrl);\n if (onOpened) {\n try {\n onOpened(request);\n }\n catch (e) {\n handleError(e);\n return;\n }\n }\n if (useArrayBuffer) {\n request.responseType = \"arraybuffer\";\n }\n if (onProgress) {\n request.addEventListener(\"progress\", onProgress);\n }\n if (onLoadEnd) {\n request.addEventListener(\"loadend\", onLoadEnd);\n }\n onReadyStateChange = () => {\n if (aborted || !request) {\n return;\n }\n // In case of undefined state in some browsers.\n if (request.readyState === (XMLHttpRequest.DONE || 4)) {\n // Some browsers have issues where onreadystatechange can be called multiple times with the same value.\n if (onReadyStateChange) {\n request.removeEventListener(\"readystatechange\", onReadyStateChange);\n }\n if ((request.status >= 200 && request.status < 300) || (request.status === 0 && (!IsWindowObjectExist() || IsFileURL()))) {\n try {\n if (onSuccess) {\n onSuccess(useArrayBuffer ? request.response : request.responseText, request);\n }\n }\n catch (e) {\n handleError(e);\n }\n return;\n }\n const retryStrategy = FileToolsOptions.DefaultRetryStrategy;\n if (retryStrategy) {\n const waitTime = retryStrategy(loadUrl, request, retryIndex);\n if (waitTime !== -1) {\n // Prevent the request from completing for retry.\n unbindEvents();\n request = new WebRequest();\n retryHandle = setTimeout(() => retryLoop(retryIndex + 1), waitTime);\n return;\n }\n }\n const error = new RequestFileError(\"Error status: \" + request.status + \" \" + request.statusText + \" - Unable to load \" + loadUrl, request);\n if (onError) {\n onError(error);\n }\n }\n };\n request.addEventListener(\"readystatechange\", onReadyStateChange);\n request.send();\n };\n retryLoop(0);\n };\n // Caching all files\n if (offlineProvider && offlineProvider.enableSceneOffline) {\n const noOfflineSupport = (request) => {\n if (request && request.status > 400) {\n if (onError) {\n onError(request);\n }\n }\n else {\n requestFile();\n }\n };\n const loadFromOfflineSupport = () => {\n // TODO: database needs to support aborting and should return a IFileRequest\n if (offlineProvider) {\n offlineProvider.loadFile(FileToolsOptions.BaseUrl + url, (data) => {\n if (!aborted && onSuccess) {\n onSuccess(data);\n }\n fileRequest.onCompleteObservable.notifyObservers(fileRequest);\n }, onProgress\n ? (event) => {\n if (!aborted && onProgress) {\n onProgress(event);\n }\n }\n : undefined, noOfflineSupport, useArrayBuffer);\n }\n };\n offlineProvider.open(loadFromOfflineSupport, noOfflineSupport);\n }\n else {\n requestFile();\n }\n return fileRequest;\n };\n /**\n * Checks if the loaded document was accessed via `file:`-Protocol.\n * @returns boolean\n * @internal\n */\n const IsFileURL = () => {\n return typeof location !== \"undefined\" && location.protocol === \"file:\";\n };\n /**\n * Test if the given uri is a valid base64 data url\n * @param uri The uri to test\n * @returns True if the uri is a base64 data url or false otherwise\n * @internal\n */\n const IsBase64DataUrl = (uri) => {\n return Base64DataUrlRegEx.test(uri);\n };\n const TestBase64DataUrl = (uri) => {\n const results = Base64DataUrlRegEx.exec(uri);\n if (results === null || results.length === 0) {\n return { match: false, type: \"\" };\n }\n else {\n const type = results[0].replace(\"data:\", \"\").replace(\"base64,\", \"\");\n return { match: true, type };\n }\n };\n /**\n * Decode the given base64 uri.\n * @param uri The uri to decode\n * @returns The decoded base64 data.\n * @internal\n */\n function DecodeBase64UrlToBinary(uri) {\n return DecodeBase64ToBinary(uri.split(\",\")[1]);\n }\n /**\n * Decode the given base64 uri into a UTF-8 encoded string.\n * @param uri The uri to decode\n * @returns The decoded base64 data.\n * @internal\n */\n const DecodeBase64UrlToString = (uri) => {\n return DecodeBase64ToString(uri.split(\",\")[1]);\n };\n /**\n * This will be executed automatically for UMD and es5.\n * If esm dev wants the side effects to execute they will have to run it manually\n * Once we build native modules those need to be exported.\n * @internal\n */\n const initSideEffects = () => {\n ThinEngine._FileToolsLoadImage = LoadImage;\n ThinEngine._FileToolsLoadFile = LoadFile;\n ShaderProcessor._FileToolsLoadFile = LoadFile;\n };\n initSideEffects();\n // deprecated\n /**\n * FileTools defined as any.\n * This should not be imported or used in future releases or in any module in the framework\n * @internal\n * @deprecated import the needed function from fileTools.ts\n */\n let FileTools;\n /**\n * @param DecodeBase64UrlToBinary\n * @param DecodeBase64UrlToString\n * @param FileToolsOptions\n * @internal\n */\n const _injectLTSFileTools = (DecodeBase64UrlToBinary, DecodeBase64UrlToString, FileToolsOptions, IsBase64DataUrl, IsFileURL, LoadFile, LoadImage, ReadFile, RequestFile, SetCorsBehavior) => {\n /**\n * Backwards compatibility.\n * @internal\n * @deprecated\n */\n FileTools = {\n DecodeBase64UrlToBinary,\n DecodeBase64UrlToString,\n DefaultRetryStrategy: FileToolsOptions.DefaultRetryStrategy,\n BaseUrl: FileToolsOptions.BaseUrl,\n CorsBehavior: FileToolsOptions.CorsBehavior,\n PreprocessUrl: FileToolsOptions.PreprocessUrl,\n IsBase64DataUrl,\n IsFileURL,\n LoadFile,\n LoadImage,\n ReadFile,\n RequestFile,\n SetCorsBehavior,\n };\n Object.defineProperty(FileTools, \"DefaultRetryStrategy\", {\n get: function () {\n return FileToolsOptions.DefaultRetryStrategy;\n },\n set: function (value) {\n FileToolsOptions.DefaultRetryStrategy = value;\n },\n });\n Object.defineProperty(FileTools, \"BaseUrl\", {\n get: function () {\n return FileToolsOptions.BaseUrl;\n },\n set: function (value) {\n FileToolsOptions.BaseUrl = value;\n },\n });\n Object.defineProperty(FileTools, \"PreprocessUrl\", {\n get: function () {\n return FileToolsOptions.PreprocessUrl;\n },\n set: function (value) {\n FileToolsOptions.PreprocessUrl = value;\n },\n });\n Object.defineProperty(FileTools, \"CorsBehavior\", {\n get: function () {\n return FileToolsOptions.CorsBehavior;\n },\n set: function (value) {\n FileToolsOptions.CorsBehavior = value;\n },\n });\n };\n _injectLTSFileTools(DecodeBase64UrlToBinary, DecodeBase64UrlToString, FileToolsOptions, IsBase64DataUrl, IsFileURL, LoadFile, LoadImage, ReadFile, RequestFile, SetCorsBehavior);\n\n /**\n * Class used to enable instantiation of objects by class name\n */\n class InstantiationTools {\n /**\n * Tries to instantiate a new object from a given class name\n * @param className defines the class name to instantiate\n * @returns the new object or null if the system was not able to do the instantiation\n */\n static Instantiate(className) {\n if (this.RegisteredExternalClasses && this.RegisteredExternalClasses[className]) {\n return this.RegisteredExternalClasses[className];\n }\n const internalClass = GetClass(className);\n if (internalClass) {\n return internalClass;\n }\n Logger.Warn(className + \" not found, you may have missed an import.\");\n const arr = className.split(\".\");\n let fn = window || this;\n for (let i = 0, len = arr.length; i < len; i++) {\n fn = fn[arr[i]];\n }\n if (typeof fn !== \"function\") {\n return null;\n }\n return fn;\n }\n }\n /**\n * Use this object to register external classes like custom textures or material\n * to allow the loaders to instantiate them\n */\n InstantiationTools.RegisteredExternalClasses = {};\n\n /**\n * Implementation from http://stackoverflow.com/questions/105034/how-to-create-a-guid-uuid-in-javascript/2117523#answer-2117523\n * Be aware Math.random() could cause collisions, but:\n * \"All but 6 of the 128 bits of the ID are randomly generated, which means that for any two ids, there's a 1 in 2^^122 (or 5.3x10^^36) chance they'll collide\"\n * @returns a pseudo random id\n */\n function RandomGUID() {\n return \"xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx\".replace(/[xy]/g, (c) => {\n const r = (Math.random() * 16) | 0, v = c === \"x\" ? r : (r & 0x3) | 0x8;\n return v.toString(16);\n });\n }\n\n /**\n * Class containing a set of static utilities functions\n */\n class Tools {\n /**\n * Gets or sets the base URL to use to load assets\n */\n static get BaseUrl() {\n return FileToolsOptions.BaseUrl;\n }\n static set BaseUrl(value) {\n FileToolsOptions.BaseUrl = value;\n }\n /**\n * Gets or sets the retry strategy to apply when an error happens while loading an asset\n */\n static get DefaultRetryStrategy() {\n return FileToolsOptions.DefaultRetryStrategy;\n }\n static set DefaultRetryStrategy(strategy) {\n FileToolsOptions.DefaultRetryStrategy = strategy;\n }\n /**\n * Default behaviour for cors in the application.\n * It can be a string if the expected behavior is identical in the entire app.\n * Or a callback to be able to set it per url or on a group of them (in case of Video source for instance)\n */\n static get CorsBehavior() {\n return FileToolsOptions.CorsBehavior;\n }\n static set CorsBehavior(value) {\n FileToolsOptions.CorsBehavior = value;\n }\n /**\n * Gets or sets a global variable indicating if fallback texture must be used when a texture cannot be loaded\n * @ignorenaming\n */\n static get UseFallbackTexture() {\n return EngineStore.UseFallbackTexture;\n }\n static set UseFallbackTexture(value) {\n EngineStore.UseFallbackTexture = value;\n }\n /**\n * Use this object to register external classes like custom textures or material\n * to allow the loaders to instantiate them\n */\n static get RegisteredExternalClasses() {\n return InstantiationTools.RegisteredExternalClasses;\n }\n static set RegisteredExternalClasses(classes) {\n InstantiationTools.RegisteredExternalClasses = classes;\n }\n /**\n * Texture content used if a texture cannot loaded\n * @ignorenaming\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n static get fallbackTexture() {\n return EngineStore.FallbackTexture;\n }\n // eslint-disable-next-line @typescript-eslint/naming-convention\n static set fallbackTexture(value) {\n EngineStore.FallbackTexture = value;\n }\n /**\n * Read the content of a byte array at a specified coordinates (taking in account wrapping)\n * @param u defines the coordinate on X axis\n * @param v defines the coordinate on Y axis\n * @param width defines the width of the source data\n * @param height defines the height of the source data\n * @param pixels defines the source byte array\n * @param color defines the output color\n */\n static FetchToRef(u, v, width, height, pixels, color) {\n const wrappedU = (Math.abs(u) * width) % width | 0;\n const wrappedV = (Math.abs(v) * height) % height | 0;\n const position = (wrappedU + wrappedV * width) * 4;\n color.r = pixels[position] / 255;\n color.g = pixels[position + 1] / 255;\n color.b = pixels[position + 2] / 255;\n color.a = pixels[position + 3] / 255;\n }\n /**\n * Interpolates between a and b via alpha\n * @param a The lower value (returned when alpha = 0)\n * @param b The upper value (returned when alpha = 1)\n * @param alpha The interpolation-factor\n * @returns The mixed value\n */\n static Mix(a, b, alpha) {\n return a * (1 - alpha) + b * alpha;\n }\n /**\n * Tries to instantiate a new object from a given class name\n * @param className defines the class name to instantiate\n * @returns the new object or null if the system was not able to do the instantiation\n */\n static Instantiate(className) {\n return InstantiationTools.Instantiate(className);\n }\n /**\n * Polyfill for setImmediate\n * @param action defines the action to execute after the current execution block\n */\n static SetImmediate(action) {\n TimingTools.SetImmediate(action);\n }\n /**\n * Function indicating if a number is an exponent of 2\n * @param value defines the value to test\n * @returns true if the value is an exponent of 2\n */\n static IsExponentOfTwo(value) {\n let count = 1;\n do {\n count *= 2;\n } while (count < value);\n return count === value;\n }\n /**\n * Returns the nearest 32-bit single precision float representation of a Number\n * @param value A Number. If the parameter is of a different type, it will get converted\n * to a number or to NaN if it cannot be converted\n * @returns number\n */\n static FloatRound(value) {\n if (Math.fround) {\n return Math.fround(value);\n }\n return (Tools._TmpFloatArray[0] = value), Tools._TmpFloatArray[0];\n }\n /**\n * Extracts the filename from a path\n * @param path defines the path to use\n * @returns the filename\n */\n static GetFilename(path) {\n const index = path.lastIndexOf(\"/\");\n if (index < 0) {\n return path;\n }\n return path.substring(index + 1);\n }\n /**\n * Extracts the \"folder\" part of a path (everything before the filename).\n * @param uri The URI to extract the info from\n * @param returnUnchangedIfNoSlash Do not touch the URI if no slashes are present\n * @returns The \"folder\" part of the path\n */\n static GetFolderPath(uri, returnUnchangedIfNoSlash = false) {\n const index = uri.lastIndexOf(\"/\");\n if (index < 0) {\n if (returnUnchangedIfNoSlash) {\n return uri;\n }\n return \"\";\n }\n return uri.substring(0, index + 1);\n }\n /**\n * Convert an angle in radians to degrees\n * @param angle defines the angle to convert\n * @returns the angle in degrees\n */\n static ToDegrees(angle) {\n return (angle * 180) / Math.PI;\n }\n /**\n * Convert an angle in degrees to radians\n * @param angle defines the angle to convert\n * @returns the angle in radians\n */\n static ToRadians(angle) {\n return (angle * Math.PI) / 180;\n }\n /**\n * Smooth angle changes (kind of low-pass filter), in particular for device orientation \"shaking\"\n * Use trigonometric functions to avoid discontinuity (0/360, -180/180)\n * @param previousAngle defines last angle value, in degrees\n * @param newAngle defines new angle value, in degrees\n * @param smoothFactor defines smoothing sensitivity; min 0: no smoothing, max 1: new data ignored\n * @returns the angle in degrees\n */\n static SmoothAngleChange(previousAngle, newAngle, smoothFactor = 0.9) {\n const previousAngleRad = this.ToRadians(previousAngle);\n const newAngleRad = this.ToRadians(newAngle);\n return this.ToDegrees(Math.atan2((1 - smoothFactor) * Math.sin(newAngleRad) + smoothFactor * Math.sin(previousAngleRad), (1 - smoothFactor) * Math.cos(newAngleRad) + smoothFactor * Math.cos(previousAngleRad)));\n }\n /**\n * Returns an array if obj is not an array\n * @param obj defines the object to evaluate as an array\n * @param allowsNullUndefined defines a boolean indicating if obj is allowed to be null or undefined\n * @returns either obj directly if obj is an array or a new array containing obj\n */\n static MakeArray(obj, allowsNullUndefined) {\n if (allowsNullUndefined !== true && (obj === undefined || obj == null)) {\n return null;\n }\n return Array.isArray(obj) ? obj : [obj];\n }\n /**\n * Gets the pointer prefix to use\n * @param engine defines the engine we are finding the prefix for\n * @returns \"pointer\" if touch is enabled. Else returns \"mouse\"\n */\n static GetPointerPrefix(engine) {\n let eventPrefix = \"pointer\";\n // Check if pointer events are supported\n if (IsWindowObjectExist() && !window.PointerEvent) {\n eventPrefix = \"mouse\";\n }\n // Special Fallback MacOS Safari...\n if (engine._badDesktopOS &&\n !engine._badOS &&\n // And not ipad pros who claim to be macs...\n !(document && \"ontouchend\" in document)) {\n eventPrefix = \"mouse\";\n }\n return eventPrefix;\n }\n /**\n * Sets the cors behavior on a dom element. This will add the required Tools.CorsBehavior to the element.\n * @param url define the url we are trying\n * @param element define the dom element where to configure the cors policy\n * @param element.crossOrigin\n */\n static SetCorsBehavior(url, element) {\n SetCorsBehavior(url, element);\n }\n /**\n * Sets the referrerPolicy behavior on a dom element.\n * @param referrerPolicy define the referrer policy to use\n * @param element define the dom element where to configure the referrer policy\n * @param element.referrerPolicy\n */\n static SetReferrerPolicyBehavior(referrerPolicy, element) {\n element.referrerPolicy = referrerPolicy;\n }\n // External files\n /**\n * Removes unwanted characters from an url\n * @param url defines the url to clean\n * @returns the cleaned url\n */\n static CleanUrl(url) {\n url = url.replace(/#/gm, \"%23\");\n return url;\n }\n /**\n * Gets or sets a function used to pre-process url before using them to load assets\n */\n static get PreprocessUrl() {\n return FileToolsOptions.PreprocessUrl;\n }\n static set PreprocessUrl(processor) {\n FileToolsOptions.PreprocessUrl = processor;\n }\n /**\n * Loads an image as an HTMLImageElement.\n * @param input url string, ArrayBuffer, or Blob to load\n * @param onLoad callback called when the image successfully loads\n * @param onError callback called when the image fails to load\n * @param offlineProvider offline provider for caching\n * @param mimeType optional mime type\n * @param imageBitmapOptions optional the options to use when creating an ImageBitmap\n * @returns the HTMLImageElement of the loaded image\n */\n static LoadImage(input, onLoad, onError, offlineProvider, mimeType, imageBitmapOptions) {\n return LoadImage(input, onLoad, onError, offlineProvider, mimeType, imageBitmapOptions);\n }\n /**\n * Loads a file from a url\n * @param url url string, ArrayBuffer, or Blob to load\n * @param onSuccess callback called when the file successfully loads\n * @param onProgress callback called while file is loading (if the server supports this mode)\n * @param offlineProvider defines the offline provider for caching\n * @param useArrayBuffer defines a boolean indicating that date must be returned as ArrayBuffer\n * @param onError callback called when the file fails to load\n * @returns a file request object\n */\n static LoadFile(url, onSuccess, onProgress, offlineProvider, useArrayBuffer, onError) {\n return LoadFile(url, onSuccess, onProgress, offlineProvider, useArrayBuffer, onError);\n }\n /**\n * Loads a file from a url\n * @param url the file url to load\n * @param useArrayBuffer defines a boolean indicating that date must be returned as ArrayBuffer\n * @returns a promise containing an ArrayBuffer corresponding to the loaded file\n */\n static LoadFileAsync(url, useArrayBuffer = true) {\n return new Promise((resolve, reject) => {\n LoadFile(url, (data) => {\n resolve(data);\n }, undefined, undefined, useArrayBuffer, (request, exception) => {\n reject(exception);\n });\n });\n }\n /**\n * Load a script (identified by an url). When the url returns, the\n * content of this file is added into a new script element, attached to the DOM (body element)\n * @param scriptUrl defines the url of the script to laod\n * @param onSuccess defines the callback called when the script is loaded\n * @param onError defines the callback to call if an error occurs\n * @param scriptId defines the id of the script element\n */\n static LoadScript(scriptUrl, onSuccess, onError, scriptId) {\n if (typeof importScripts === \"function\") {\n try {\n importScripts(scriptUrl);\n onSuccess();\n }\n catch (e) {\n onError === null || onError === void 0 ? void 0 : onError(`Unable to load script '${scriptUrl}' in worker`, e);\n }\n return;\n }\n else if (!IsWindowObjectExist()) {\n onError === null || onError === void 0 ? void 0 : onError(`Cannot load script '${scriptUrl}' outside of a window or a worker`);\n return;\n }\n const head = document.getElementsByTagName(\"head\")[0];\n const script = document.createElement(\"script\");\n script.setAttribute(\"type\", \"text/javascript\");\n script.setAttribute(\"src\", scriptUrl);\n if (scriptId) {\n script.id = scriptId;\n }\n script.onload = () => {\n if (onSuccess) {\n onSuccess();\n }\n };\n script.onerror = (e) => {\n if (onError) {\n onError(`Unable to load script '${scriptUrl}'`, e);\n }\n };\n head.appendChild(script);\n }\n /**\n * Load an asynchronous script (identified by an url). When the url returns, the\n * content of this file is added into a new script element, attached to the DOM (body element)\n * @param scriptUrl defines the url of the script to laod\n * @returns a promise request object\n */\n static LoadScriptAsync(scriptUrl) {\n return new Promise((resolve, reject) => {\n this.LoadScript(scriptUrl, () => {\n resolve();\n }, (message, exception) => {\n reject(exception || new Error(message));\n });\n });\n }\n /**\n * Loads a file from a blob\n * @param fileToLoad defines the blob to use\n * @param callback defines the callback to call when data is loaded\n * @param progressCallback defines the callback to call during loading process\n * @returns a file request object\n */\n static ReadFileAsDataURL(fileToLoad, callback, progressCallback) {\n const reader = new FileReader();\n const request = {\n onCompleteObservable: new Observable$1(),\n abort: () => reader.abort(),\n };\n reader.onloadend = () => {\n request.onCompleteObservable.notifyObservers(request);\n };\n reader.onload = (e) => {\n //target doesn't have result from ts 1.3\n callback(e.target[\"result\"]);\n };\n reader.onprogress = progressCallback;\n reader.readAsDataURL(fileToLoad);\n return request;\n }\n /**\n * Reads a file from a File object\n * @param file defines the file to load\n * @param onSuccess defines the callback to call when data is loaded\n * @param onProgress defines the callback to call during loading process\n * @param useArrayBuffer defines a boolean indicating that data must be returned as an ArrayBuffer\n * @param onError defines the callback to call when an error occurs\n * @returns a file request object\n */\n static ReadFile(file, onSuccess, onProgress, useArrayBuffer, onError) {\n return ReadFile(file, onSuccess, onProgress, useArrayBuffer, onError);\n }\n /**\n * Creates a data url from a given string content\n * @param content defines the content to convert\n * @returns the new data url link\n */\n static FileAsURL(content) {\n const fileBlob = new Blob([content]);\n const url = window.URL;\n const link = url.createObjectURL(fileBlob);\n return link;\n }\n /**\n * Format the given number to a specific decimal format\n * @param value defines the number to format\n * @param decimals defines the number of decimals to use\n * @returns the formatted string\n */\n static Format(value, decimals = 2) {\n return value.toFixed(decimals);\n }\n /**\n * Tries to copy an object by duplicating every property\n * @param source defines the source object\n * @param destination defines the target object\n * @param doNotCopyList defines a list of properties to avoid\n * @param mustCopyList defines a list of properties to copy (even if they start with _)\n */\n static DeepCopy(source, destination, doNotCopyList, mustCopyList) {\n DeepCopier.DeepCopy(source, destination, doNotCopyList, mustCopyList);\n }\n /**\n * Gets a boolean indicating if the given object has no own property\n * @param obj defines the object to test\n * @returns true if object has no own property\n */\n static IsEmpty(obj) {\n for (const i in obj) {\n if (Object.prototype.hasOwnProperty.call(obj, i)) {\n return false;\n }\n }\n return true;\n }\n /**\n * Function used to register events at window level\n * @param windowElement defines the Window object to use\n * @param events defines the events to register\n */\n static RegisterTopRootEvents(windowElement, events) {\n for (let index = 0; index < events.length; index++) {\n const event = events[index];\n windowElement.addEventListener(event.name, event.handler, false);\n try {\n if (window.parent) {\n window.parent.addEventListener(event.name, event.handler, false);\n }\n }\n catch (e) {\n // Silently fails...\n }\n }\n }\n /**\n * Function used to unregister events from window level\n * @param windowElement defines the Window object to use\n * @param events defines the events to unregister\n */\n static UnregisterTopRootEvents(windowElement, events) {\n for (let index = 0; index < events.length; index++) {\n const event = events[index];\n windowElement.removeEventListener(event.name, event.handler);\n try {\n if (windowElement.parent) {\n windowElement.parent.removeEventListener(event.name, event.handler);\n }\n }\n catch (e) {\n // Silently fails...\n }\n }\n }\n /**\n * Dumps the current bound framebuffer\n * @param width defines the rendering width\n * @param height defines the rendering height\n * @param engine defines the hosting engine\n * @param successCallback defines the callback triggered once the data are available\n * @param mimeType defines the mime type of the result\n * @param fileName defines the filename to download. If present, the result will automatically be downloaded\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\n * @returns a void promise\n */\n static async DumpFramebuffer(width, height, engine, successCallback, mimeType = \"image/png\", fileName, quality) {\n throw _WarnImport(\"DumpTools\");\n }\n /**\n * Dumps an array buffer\n * @param width defines the rendering width\n * @param height defines the rendering height\n * @param data the data array\n * @param successCallback defines the callback triggered once the data are available\n * @param mimeType defines the mime type of the result\n * @param fileName defines the filename to download. If present, the result will automatically be downloaded\n * @param invertY true to invert the picture in the Y dimension\n * @param toArrayBuffer true to convert the data to an ArrayBuffer (encoded as `mimeType`) instead of a base64 string\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\n */\n static DumpData(width, height, data, successCallback, mimeType = \"image/png\", fileName, invertY = false, toArrayBuffer = false, quality) {\n throw _WarnImport(\"DumpTools\");\n }\n /**\n * Dumps an array buffer\n * @param width defines the rendering width\n * @param height defines the rendering height\n * @param data the data array\n * @param mimeType defines the mime type of the result\n * @param fileName defines the filename to download. If present, the result will automatically be downloaded\n * @param invertY true to invert the picture in the Y dimension\n * @param toArrayBuffer true to convert the data to an ArrayBuffer (encoded as `mimeType`) instead of a base64 string\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\n * @returns a promise that resolve to the final data\n */\n static DumpDataAsync(width, height, data, mimeType = \"image/png\", fileName, invertY = false, toArrayBuffer = false, quality) {\n throw _WarnImport(\"DumpTools\");\n }\n static _IsOffScreenCanvas(canvas) {\n return canvas.convertToBlob !== undefined;\n }\n /**\n * Converts the canvas data to blob.\n * This acts as a polyfill for browsers not supporting the to blob function.\n * @param canvas Defines the canvas to extract the data from (can be an offscreen canvas)\n * @param successCallback Defines the callback triggered once the data are available\n * @param mimeType Defines the mime type of the result\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\n */\n static ToBlob(canvas, successCallback, mimeType = \"image/png\", quality) {\n // We need HTMLCanvasElement.toBlob for HD screenshots\n if (!Tools._IsOffScreenCanvas(canvas) && !canvas.toBlob) {\n // low performance polyfill based on toDataURL (https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob)\n canvas.toBlob = function (callback, type, quality) {\n setTimeout(() => {\n const binStr = atob(this.toDataURL(type, quality).split(\",\")[1]), len = binStr.length, arr = new Uint8Array(len);\n for (let i = 0; i < len; i++) {\n arr[i] = binStr.charCodeAt(i);\n }\n callback(new Blob([arr]));\n });\n };\n }\n if (Tools._IsOffScreenCanvas(canvas)) {\n canvas\n .convertToBlob({\n type: mimeType,\n quality,\n })\n .then((blob) => successCallback(blob));\n }\n else {\n canvas.toBlob(function (blob) {\n successCallback(blob);\n }, mimeType, quality);\n }\n }\n /**\n * Download a Blob object\n * @param blob the Blob object\n * @param fileName the file name to download\n * @returns\n */\n static DownloadBlob(blob, fileName) {\n //Creating a link if the browser have the download attribute on the a tag, to automatically start download generated image.\n if (\"download\" in document.createElement(\"a\")) {\n if (!fileName) {\n const date = new Date();\n const stringDate = (date.getFullYear() + \"-\" + (date.getMonth() + 1)).slice(2) + \"-\" + date.getDate() + \"_\" + date.getHours() + \"-\" + (\"0\" + date.getMinutes()).slice(-2);\n fileName = \"screenshot_\" + stringDate + \".png\";\n }\n Tools.Download(blob, fileName);\n }\n else {\n if (blob && typeof URL !== \"undefined\") {\n const url = URL.createObjectURL(blob);\n const newWindow = window.open(\"\");\n if (!newWindow) {\n return;\n }\n const img = newWindow.document.createElement(\"img\");\n img.onload = function () {\n // no longer need to read the blob so it's revoked\n URL.revokeObjectURL(url);\n };\n img.src = url;\n newWindow.document.body.appendChild(img);\n }\n }\n }\n /**\n * Encodes the canvas data to base 64, or automatically downloads the result if `fileName` is defined.\n * @param canvas The canvas to get the data from, which can be an offscreen canvas.\n * @param successCallback The callback which is triggered once the data is available. If `fileName` is defined, the callback will be invoked after the download occurs, and the `data` argument will be an empty string.\n * @param mimeType The mime type of the result.\n * @param fileName The name of the file to download. If defined, the result will automatically be downloaded. If not defined, and `successCallback` is also not defined, the result will automatically be downloaded with an auto-generated file name.\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\n */\n static EncodeScreenshotCanvasData(canvas, successCallback, mimeType = \"image/png\", fileName, quality) {\n if (typeof fileName === \"string\" || !successCallback) {\n this.ToBlob(canvas, function (blob) {\n if (blob) {\n Tools.DownloadBlob(blob, fileName);\n }\n if (successCallback) {\n successCallback(\"\");\n }\n }, mimeType, quality);\n }\n else if (successCallback) {\n if (Tools._IsOffScreenCanvas(canvas)) {\n canvas\n .convertToBlob({\n type: mimeType,\n quality,\n })\n .then((blob) => {\n const reader = new FileReader();\n reader.readAsDataURL(blob);\n reader.onloadend = () => {\n const base64data = reader.result;\n successCallback(base64data);\n };\n });\n return;\n }\n const base64Image = canvas.toDataURL(mimeType, quality);\n successCallback(base64Image);\n }\n }\n /**\n * Downloads a blob in the browser\n * @param blob defines the blob to download\n * @param fileName defines the name of the downloaded file\n */\n static Download(blob, fileName) {\n if (typeof URL === \"undefined\") {\n return;\n }\n const url = window.URL.createObjectURL(blob);\n const a = document.createElement(\"a\");\n document.body.appendChild(a);\n a.style.display = \"none\";\n a.href = url;\n a.download = fileName;\n a.addEventListener(\"click\", () => {\n if (a.parentElement) {\n a.parentElement.removeChild(a);\n }\n });\n a.click();\n window.URL.revokeObjectURL(url);\n }\n /**\n * Will return the right value of the noPreventDefault variable\n * Needed to keep backwards compatibility to the old API.\n *\n * @param args arguments passed to the attachControl function\n * @returns the correct value for noPreventDefault\n */\n static BackCompatCameraNoPreventDefault(args) {\n // is it used correctly?\n if (typeof args[0] === \"boolean\") {\n return args[0];\n }\n else if (typeof args[1] === \"boolean\") {\n return args[1];\n }\n return false;\n }\n /**\n * Captures a screenshot of the current rendering\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/renderToPNG\n * @param engine defines the rendering engine\n * @param camera defines the source camera\n * @param size This parameter can be set to a single number or to an object with the\n * following (optional) properties: precision, width, height. If a single number is passed,\n * it will be used for both width and height. If an object is passed, the screenshot size\n * will be derived from the parameters. The precision property is a multiplier allowing\n * rendering at a higher or lower resolution\n * @param successCallback defines the callback receives a single parameter which contains the\n * screenshot as a string of base64-encoded characters. This string can be assigned to the\n * src parameter of an to display it\n * @param mimeType defines the MIME type of the screenshot image (default: image/png).\n * Check your browser for supported MIME types\n * @param forceDownload force the system to download the image even if a successCallback is provided\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n static CreateScreenshot(engine, camera, size, successCallback, mimeType = \"image/png\", forceDownload = false, quality) {\n throw _WarnImport(\"ScreenshotTools\");\n }\n /**\n * Captures a screenshot of the current rendering\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/renderToPNG\n * @param engine defines the rendering engine\n * @param camera defines the source camera\n * @param size This parameter can be set to a single number or to an object with the\n * following (optional) properties: precision, width, height. If a single number is passed,\n * it will be used for both width and height. If an object is passed, the screenshot size\n * will be derived from the parameters. The precision property is a multiplier allowing\n * rendering at a higher or lower resolution\n * @param mimeType defines the MIME type of the screenshot image (default: image/png).\n * Check your browser for supported MIME types\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\n * @returns screenshot as a string of base64-encoded characters. This string can be assigned\n * to the src parameter of an to display it\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n static CreateScreenshotAsync(engine, camera, size, mimeType = \"image/png\", quality) {\n throw _WarnImport(\"ScreenshotTools\");\n }\n /**\n * Generates an image screenshot from the specified camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/renderToPNG\n * @param engine The engine to use for rendering\n * @param camera The camera to use for rendering\n * @param size This parameter can be set to a single number or to an object with the\n * following (optional) properties: precision, width, height. If a single number is passed,\n * it will be used for both width and height. If an object is passed, the screenshot size\n * will be derived from the parameters. The precision property is a multiplier allowing\n * rendering at a higher or lower resolution\n * @param successCallback The callback receives a single parameter which contains the\n * screenshot as a string of base64-encoded characters. This string can be assigned to the\n * src parameter of an to display it\n * @param mimeType The MIME type of the screenshot image (default: image/png).\n * Check your browser for supported MIME types\n * @param samples Texture samples (default: 1)\n * @param antialiasing Whether antialiasing should be turned on or not (default: false)\n * @param fileName A name for for the downloaded file.\n * @param renderSprites Whether the sprites should be rendered or not (default: false)\n * @param enableStencilBuffer Whether the stencil buffer should be enabled or not (default: false)\n * @param useLayerMask if the camera's layer mask should be used to filter what should be rendered (default: true)\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n static CreateScreenshotUsingRenderTarget(engine, camera, size, successCallback, mimeType = \"image/png\", samples = 1, antialiasing = false, fileName, renderSprites = false, enableStencilBuffer = false, useLayerMask = true, quality) {\n throw _WarnImport(\"ScreenshotTools\");\n }\n /**\n * Generates an image screenshot from the specified camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/renderToPNG\n * @param engine The engine to use for rendering\n * @param camera The camera to use for rendering\n * @param size This parameter can be set to a single number or to an object with the\n * following (optional) properties: precision, width, height. If a single number is passed,\n * it will be used for both width and height. If an object is passed, the screenshot size\n * will be derived from the parameters. The precision property is a multiplier allowing\n * rendering at a higher or lower resolution\n * @param mimeType The MIME type of the screenshot image (default: image/png).\n * Check your browser for supported MIME types\n * @param samples Texture samples (default: 1)\n * @param antialiasing Whether antialiasing should be turned on or not (default: false)\n * @param fileName A name for for the downloaded file.\n * @returns screenshot as a string of base64-encoded characters. This string can be assigned\n * @param renderSprites Whether the sprites should be rendered or not (default: false)\n * @param enableStencilBuffer Whether the stencil buffer should be enabled or not (default: false)\n * @param useLayerMask if the camera's layer mask should be used to filter what should be rendered (default: true)\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\n * to the src parameter of an to display it\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n static CreateScreenshotUsingRenderTargetAsync(engine, camera, size, mimeType = \"image/png\", samples = 1, antialiasing = false, fileName, renderSprites = false, enableStencilBuffer = false, useLayerMask = true, quality) {\n throw _WarnImport(\"ScreenshotTools\");\n }\n /**\n * Implementation from http://stackoverflow.com/questions/105034/how-to-create-a-guid-uuid-in-javascript/2117523#answer-2117523\n * Be aware Math.random() could cause collisions, but:\n * \"All but 6 of the 128 bits of the ID are randomly generated, which means that for any two ids, there's a 1 in 2^^122 (or 5.3x10^^36) chance they'll collide\"\n * @returns a pseudo random id\n */\n static RandomId() {\n return RandomGUID();\n }\n /**\n * Test if the given uri is a base64 string\n * @deprecated Please use FileTools.IsBase64DataUrl instead.\n * @param uri The uri to test\n * @returns True if the uri is a base64 string or false otherwise\n */\n static IsBase64(uri) {\n return IsBase64DataUrl(uri);\n }\n /**\n * Decode the given base64 uri.\n * @deprecated Please use FileTools.DecodeBase64UrlToBinary instead.\n * @param uri The uri to decode\n * @returns The decoded base64 data.\n */\n static DecodeBase64(uri) {\n return DecodeBase64UrlToBinary(uri);\n }\n /**\n * Gets a value indicating the number of loading errors\n * @ignorenaming\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n static get errorsCount() {\n return Logger.errorsCount;\n }\n /**\n * Log a message to the console\n * @param message defines the message to log\n */\n static Log(message) {\n Logger.Log(message);\n }\n /**\n * Write a warning message to the console\n * @param message defines the message to log\n */\n static Warn(message) {\n Logger.Warn(message);\n }\n /**\n * Write an error message to the console\n * @param message defines the message to log\n */\n static Error(message) {\n Logger.Error(message);\n }\n /**\n * Gets current log cache (list of logs)\n */\n static get LogCache() {\n return Logger.LogCache;\n }\n /**\n * Clears the log cache\n */\n static ClearLogCache() {\n Logger.ClearLogCache();\n }\n /**\n * Sets the current log level (MessageLogLevel / WarningLogLevel / ErrorLogLevel)\n */\n static set LogLevels(level) {\n Logger.LogLevels = level;\n }\n /**\n * Sets the current performance log level\n */\n static set PerformanceLogLevel(level) {\n if ((level & Tools.PerformanceUserMarkLogLevel) === Tools.PerformanceUserMarkLogLevel) {\n Tools.StartPerformanceCounter = Tools._StartUserMark;\n Tools.EndPerformanceCounter = Tools._EndUserMark;\n return;\n }\n if ((level & Tools.PerformanceConsoleLogLevel) === Tools.PerformanceConsoleLogLevel) {\n Tools.StartPerformanceCounter = Tools._StartPerformanceConsole;\n Tools.EndPerformanceCounter = Tools._EndPerformanceConsole;\n return;\n }\n Tools.StartPerformanceCounter = Tools._StartPerformanceCounterDisabled;\n Tools.EndPerformanceCounter = Tools._EndPerformanceCounterDisabled;\n }\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n static _StartPerformanceCounterDisabled(counterName, condition) { }\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n static _EndPerformanceCounterDisabled(counterName, condition) { }\n static _StartUserMark(counterName, condition = true) {\n if (!Tools._Performance) {\n if (!IsWindowObjectExist()) {\n return;\n }\n Tools._Performance = window.performance;\n }\n if (!condition || !Tools._Performance.mark) {\n return;\n }\n Tools._Performance.mark(counterName + \"-Begin\");\n }\n static _EndUserMark(counterName, condition = true) {\n if (!condition || !Tools._Performance.mark) {\n return;\n }\n Tools._Performance.mark(counterName + \"-End\");\n Tools._Performance.measure(counterName, counterName + \"-Begin\", counterName + \"-End\");\n }\n static _StartPerformanceConsole(counterName, condition = true) {\n if (!condition) {\n return;\n }\n Tools._StartUserMark(counterName, condition);\n if (console.time) {\n console.time(counterName);\n }\n }\n static _EndPerformanceConsole(counterName, condition = true) {\n if (!condition) {\n return;\n }\n Tools._EndUserMark(counterName, condition);\n console.timeEnd(counterName);\n }\n /**\n * Gets either window.performance.now() if supported or Date.now() else\n */\n static get Now() {\n return PrecisionDate.Now;\n }\n /**\n * This method will return the name of the class used to create the instance of the given object.\n * It will works only on Javascript basic data types (number, string, ...) and instance of class declared with the @className decorator.\n * @param object the object to get the class name from\n * @param isType defines if the object is actually a type\n * @returns the name of the class, will be \"object\" for a custom data type not using the @className decorator\n */\n static GetClassName(object, isType = false) {\n let name = null;\n if (!isType && object.getClassName) {\n name = object.getClassName();\n }\n else {\n if (object instanceof Object) {\n const classObj = isType ? object : Object.getPrototypeOf(object);\n name = classObj.constructor[\"__bjsclassName__\"];\n }\n if (!name) {\n name = typeof object;\n }\n }\n return name;\n }\n /**\n * Gets the first element of an array satisfying a given predicate\n * @param array defines the array to browse\n * @param predicate defines the predicate to use\n * @returns null if not found or the element\n */\n static First(array, predicate) {\n for (const el of array) {\n if (predicate(el)) {\n return el;\n }\n }\n return null;\n }\n /**\n * This method will return the name of the full name of the class, including its owning module (if any).\n * It will works only on Javascript basic data types (number, string, ...) and instance of class declared with the @className decorator or implementing a method getClassName():string (in which case the module won't be specified).\n * @param object the object to get the class name from\n * @param isType defines if the object is actually a type\n * @returns a string that can have two forms: \"moduleName.className\" if module was specified when the class' Name was registered or \"className\" if there was not module specified.\n * @ignorenaming\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n static getFullClassName(object, isType = false) {\n let className = null;\n let moduleName = null;\n if (!isType && object.getClassName) {\n className = object.getClassName();\n }\n else {\n if (object instanceof Object) {\n const classObj = isType ? object : Object.getPrototypeOf(object);\n className = classObj.constructor[\"__bjsclassName__\"];\n moduleName = classObj.constructor[\"__bjsmoduleName__\"];\n }\n if (!className) {\n className = typeof object;\n }\n }\n if (!className) {\n return null;\n }\n return (moduleName != null ? moduleName + \".\" : \"\") + className;\n }\n /**\n * Returns a promise that resolves after the given amount of time.\n * @param delay Number of milliseconds to delay\n * @returns Promise that resolves after the given amount of time\n */\n static DelayAsync(delay) {\n return new Promise((resolve) => {\n setTimeout(() => {\n resolve();\n }, delay);\n });\n }\n /**\n * Utility function to detect if the current user agent is Safari\n * @returns whether or not the current user agent is safari\n */\n static IsSafari() {\n if (!IsNavigatorAvailable()) {\n return false;\n }\n return /^((?!chrome|android).)*safari/i.test(navigator.userAgent);\n }\n }\n /**\n * Enable/Disable Custom HTTP Request Headers globally.\n * default = false\n * @see CustomRequestHeaders\n */\n Tools.UseCustomRequestHeaders = false;\n /**\n * Custom HTTP Request Headers to be sent with XMLHttpRequests\n * i.e. when loading files, where the server/service expects an Authorization header\n */\n Tools.CustomRequestHeaders = WebRequest.CustomRequestHeaders;\n Tools._TmpFloatArray = new Float32Array(1);\n /**\n * Extracts text content from a DOM element hierarchy\n * Back Compat only, please use GetDOMTextContent instead.\n */\n Tools.GetDOMTextContent = GetDOMTextContent;\n Tools.GetAbsoluteUrl = typeof document === \"object\"\n ? (url) => {\n const a = document.createElement(\"a\");\n a.href = url;\n return a.href;\n }\n : typeof URL === \"function\" && typeof location === \"object\"\n ? (url) => new URL(url, location.origin).href\n : () => {\n throw new Error(\"Unable to get absolute URL. Override BABYLON.Tools.GetAbsoluteUrl to a custom implementation for the current context.\");\n };\n // Logs\n /**\n * No log\n */\n Tools.NoneLogLevel = Logger.NoneLogLevel;\n /**\n * Only message logs\n */\n Tools.MessageLogLevel = Logger.MessageLogLevel;\n /**\n * Only warning logs\n */\n Tools.WarningLogLevel = Logger.WarningLogLevel;\n /**\n * Only error logs\n */\n Tools.ErrorLogLevel = Logger.ErrorLogLevel;\n /**\n * All logs\n */\n Tools.AllLogLevel = Logger.AllLogLevel;\n /**\n * Checks if the window object exists\n * Back Compat only, please use IsWindowObjectExist instead.\n */\n Tools.IsWindowObjectExist = IsWindowObjectExist;\n // Performances\n /**\n * No performance log\n */\n Tools.PerformanceNoneLogLevel = 0;\n /**\n * Use user marks to log performance\n */\n Tools.PerformanceUserMarkLogLevel = 1;\n /**\n * Log performance to the console\n */\n Tools.PerformanceConsoleLogLevel = 2;\n /**\n * Starts a performance counter\n */\n Tools.StartPerformanceCounter = Tools._StartPerformanceCounterDisabled;\n /**\n * Ends a specific performance counter\n */\n Tools.EndPerformanceCounter = Tools._EndPerformanceCounterDisabled;\n /**\n * An implementation of a loop for asynchronous functions.\n */\n class AsyncLoop {\n /**\n * Constructor.\n * @param iterations the number of iterations.\n * @param func the function to run each iteration\n * @param successCallback the callback that will be called upon successful execution\n * @param offset starting offset.\n */\n constructor(\n /**\n * Defines the number of iterations for the loop\n */\n iterations, func, successCallback, offset = 0) {\n this.iterations = iterations;\n this.index = offset - 1;\n this._done = false;\n this._fn = func;\n this._successCallback = successCallback;\n }\n /**\n * Execute the next iteration. Must be called after the last iteration was finished.\n */\n executeNext() {\n if (!this._done) {\n if (this.index + 1 < this.iterations) {\n ++this.index;\n this._fn(this);\n }\n else {\n this.breakLoop();\n }\n }\n }\n /**\n * Break the loop and run the success callback.\n */\n breakLoop() {\n this._done = true;\n this._successCallback();\n }\n /**\n * Create and run an async loop.\n * @param iterations the number of iterations.\n * @param fn the function to run each iteration\n * @param successCallback the callback that will be called upon successful execution\n * @param offset starting offset.\n * @returns the created async loop object\n */\n static Run(iterations, fn, successCallback, offset = 0) {\n const loop = new AsyncLoop(iterations, fn, successCallback, offset);\n loop.executeNext();\n return loop;\n }\n /**\n * A for-loop that will run a given number of iterations synchronous and the rest async.\n * @param iterations total number of iterations\n * @param syncedIterations number of synchronous iterations in each async iteration.\n * @param fn the function to call each iteration.\n * @param callback a success call back that will be called when iterating stops.\n * @param breakFunction a break condition (optional)\n * @param timeout timeout settings for the setTimeout function. default - 0.\n * @returns the created async loop object\n */\n static SyncAsyncForLoop(iterations, syncedIterations, fn, callback, breakFunction, timeout = 0) {\n return AsyncLoop.Run(Math.ceil(iterations / syncedIterations), (loop) => {\n if (breakFunction && breakFunction()) {\n loop.breakLoop();\n }\n else {\n setTimeout(() => {\n for (let i = 0; i < syncedIterations; ++i) {\n const iteration = loop.index * syncedIterations + i;\n if (iteration >= iterations) {\n break;\n }\n fn(iteration);\n if (breakFunction && breakFunction()) {\n loop.breakLoop();\n break;\n }\n }\n loop.executeNext();\n }, timeout);\n }\n }, callback);\n }\n }\n // Will only be define if Tools is imported freeing up some space when only engine is required\n EngineStore.FallbackTexture =\n \"\";\n\n /* eslint-disable @typescript-eslint/naming-convention */\n // \"Coroutines are computer program components that generalize subroutines for non-preemptive multitasking, by allowing execution to be suspended and resumed.\"\n // https://en.wikipedia.org/wiki/Coroutine\n // The inline scheduler simply steps the coroutine synchronously. This is useful for running a coroutine synchronously, and also as a helper function for other schedulers.\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n function inlineScheduler(coroutine, onStep, onError) {\n try {\n const step = coroutine.next();\n if (step.done) {\n onStep(step);\n }\n else if (!step.value) {\n // NOTE: The properties of step have been narrowed, but the type of step itself is not narrowed, so the cast below is the most type safe way to deal with this without instantiating a new object to hold the values.\n onStep(step);\n }\n else {\n step.value.then(() => {\n step.value = undefined;\n onStep(step);\n }, onError);\n }\n }\n catch (error) {\n onError(error);\n }\n }\n // The yielding scheduler steps the coroutine synchronously until the specified time interval has elapsed, then yields control so other operations can be performed.\n // A single instance of a yielding scheduler could be shared across multiple coroutines to yield when their collective work exceeds the threshold.\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n function createYieldingScheduler(yieldAfterMS = 25) {\n let startTime;\n return (coroutine, onStep, onError) => {\n const currentTime = performance.now();\n if (startTime === undefined || currentTime - startTime > yieldAfterMS) {\n // If this is the first coroutine step, or if the time interval has elapsed, record a new start time, and schedule the coroutine step to happen later, effectively yielding control of the execution context.\n startTime = currentTime;\n setTimeout(() => {\n inlineScheduler(coroutine, onStep, onError);\n }, 0);\n }\n else {\n // Otherwise it is not time to yield yet, so step the coroutine synchronously.\n inlineScheduler(coroutine, onStep, onError);\n }\n };\n }\n // Runs the specified coroutine with the specified scheduler. The success or error callback will be invoked when the coroutine finishes.\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n function runCoroutine(coroutine, scheduler, onSuccess, onError, abortSignal) {\n const resume = () => {\n let reschedule;\n const onStep = (stepResult) => {\n if (stepResult.done) {\n // If the coroutine is done, report success.\n onSuccess(stepResult.value);\n }\n else {\n // If the coroutine is not done, resume the coroutine (via the scheduler).\n if (reschedule === undefined) {\n // If reschedule is undefined at this point, then the coroutine must have stepped synchronously, so just flag another loop iteration.\n reschedule = true;\n }\n else {\n // If reschedule is defined at this point, then the coroutine must have stepped asynchronously, so call resume to restart the step loop.\n resume();\n }\n }\n };\n do {\n reschedule = undefined;\n if (!abortSignal || !abortSignal.aborted) {\n scheduler(coroutine, onStep, onError);\n }\n else {\n onError(new Error(\"Aborted\"));\n }\n if (reschedule === undefined) {\n // If reschedule is undefined at this point, then the coroutine must have stepped asynchronously, so stop looping and let the coroutine be resumed later.\n reschedule = false;\n }\n } while (reschedule);\n };\n resume();\n }\n // Runs the specified coroutine synchronously.\n /**\n * @internal\n */\n function runCoroutineSync(coroutine, abortSignal) {\n // Run the coroutine with the inline scheduler, storing the returned value, or re-throwing the error (since the error callback will be called synchronously by the inline scheduler).\n let result;\n runCoroutine(coroutine, inlineScheduler, (r) => (result = r), (e) => {\n throw e;\n }, abortSignal);\n // Synchronously return the result of the coroutine.\n return result;\n }\n // Runs the specified coroutine asynchronously with the specified scheduler.\n /**\n * @internal\n */\n function runCoroutineAsync(coroutine, scheduler, abortSignal) {\n // Run the coroutine with a yielding scheduler, resolving or rejecting the result promise when the coroutine finishes.\n return new Promise((resolve, reject) => {\n runCoroutine(coroutine, scheduler, resolve, reject, abortSignal);\n });\n }\n /**\n * Given a function that returns a Coroutine, produce a function with the same parameters that returns a T.\n * The returned function runs the coroutine synchronously.\n * @param coroutineFactory A function that returns a Coroutine.\n * @param abortSignal\n * @returns A function that runs the coroutine synchronously.\n * @internal\n */\n function makeSyncFunction(coroutineFactory, abortSignal) {\n return (...params) => {\n // Run the coroutine synchronously.\n return runCoroutineSync(coroutineFactory(...params), abortSignal);\n };\n }\n\n /**\n * Defines an GC Friendly array where the backfield array do not shrink to prevent over allocations.\n */\n class SmartArray {\n /**\n * Instantiates a Smart Array.\n * @param capacity defines the default capacity of the array.\n */\n constructor(capacity) {\n /**\n * The active length of the array.\n */\n this.length = 0;\n this.data = new Array(capacity);\n this._id = SmartArray._GlobalId++;\n }\n /**\n * Pushes a value at the end of the active data.\n * @param value defines the object to push in the array.\n */\n push(value) {\n this.data[this.length++] = value;\n if (this.length > this.data.length) {\n this.data.length *= 2;\n }\n }\n /**\n * Iterates over the active data and apply the lambda to them.\n * @param func defines the action to apply on each value.\n */\n forEach(func) {\n for (let index = 0; index < this.length; index++) {\n func(this.data[index]);\n }\n }\n /**\n * Sorts the full sets of data.\n * @param compareFn defines the comparison function to apply.\n */\n sort(compareFn) {\n this.data.sort(compareFn);\n }\n /**\n * Resets the active data to an empty array.\n */\n reset() {\n this.length = 0;\n }\n /**\n * Releases all the data from the array as well as the array.\n */\n dispose() {\n this.reset();\n if (this.data) {\n this.data.length = 0;\n }\n }\n /**\n * Concats the active data with a given array.\n * @param array defines the data to concatenate with.\n */\n concat(array) {\n if (array.length === 0) {\n return;\n }\n if (this.length + array.length > this.data.length) {\n this.data.length = (this.length + array.length) * 2;\n }\n for (let index = 0; index < array.length; index++) {\n this.data[this.length++] = (array.data || array)[index];\n }\n }\n /**\n * Returns the position of a value in the active data.\n * @param value defines the value to find the index for\n * @returns the index if found in the active data otherwise -1\n */\n indexOf(value) {\n const position = this.data.indexOf(value);\n if (position >= this.length) {\n return -1;\n }\n return position;\n }\n /**\n * Returns whether an element is part of the active data.\n * @param value defines the value to look for\n * @returns true if found in the active data otherwise false\n */\n contains(value) {\n return this.indexOf(value) !== -1;\n }\n }\n // Statics\n SmartArray._GlobalId = 0;\n /**\n * Defines an GC Friendly array where the backfield array do not shrink to prevent over allocations.\n * The data in this array can only be present once\n */\n class SmartArrayNoDuplicate extends SmartArray {\n constructor() {\n super(...arguments);\n this._duplicateId = 0;\n }\n /**\n * Pushes a value at the end of the active data.\n * THIS DOES NOT PREVENT DUPPLICATE DATA\n * @param value defines the object to push in the array.\n */\n push(value) {\n super.push(value);\n if (!value.__smartArrayFlags) {\n value.__smartArrayFlags = {};\n }\n value.__smartArrayFlags[this._id] = this._duplicateId;\n }\n /**\n * Pushes a value at the end of the active data.\n * If the data is already present, it won t be added again\n * @param value defines the object to push in the array.\n * @returns true if added false if it was already present\n */\n pushNoDuplicate(value) {\n if (value.__smartArrayFlags && value.__smartArrayFlags[this._id] === this._duplicateId) {\n return false;\n }\n this.push(value);\n return true;\n }\n /**\n * Resets the active data to an empty array.\n */\n reset() {\n super.reset();\n this._duplicateId++;\n }\n /**\n * Concats the active data with a given array.\n * This ensures no duplicate will be present in the result.\n * @param array defines the data to concatenate with.\n */\n concatWithNoDuplicate(array) {\n if (array.length === 0) {\n return;\n }\n if (this.length + array.length > this.data.length) {\n this.data.length = (this.length + array.length) * 2;\n }\n for (let index = 0; index < array.length; index++) {\n const item = (array.data || array)[index];\n this.pushNoDuplicate(item);\n }\n }\n }\n\n /**\n * Class used to represent a viewport on screen\n */\n class Viewport {\n /**\n * Creates a Viewport object located at (x, y) and sized (width, height)\n * @param x defines viewport left coordinate\n * @param y defines viewport top coordinate\n * @param width defines the viewport width\n * @param height defines the viewport height\n */\n constructor(\n /** viewport left coordinate */\n x, \n /** viewport top coordinate */\n y, \n /**viewport width */\n width, \n /** viewport height */\n height) {\n this.x = x;\n this.y = y;\n this.width = width;\n this.height = height;\n }\n /**\n * Creates a new viewport using absolute sizing (from 0-> width, 0-> height instead of 0->1)\n * @param renderWidth defines the rendering width\n * @param renderHeight defines the rendering height\n * @returns a new Viewport\n */\n toGlobal(renderWidth, renderHeight) {\n return new Viewport(this.x * renderWidth, this.y * renderHeight, this.width * renderWidth, this.height * renderHeight);\n }\n /**\n * Stores absolute viewport value into a target viewport (from 0-> width, 0-> height instead of 0->1)\n * @param renderWidth defines the rendering width\n * @param renderHeight defines the rendering height\n * @param ref defines the target viewport\n * @returns the current viewport\n */\n toGlobalToRef(renderWidth, renderHeight, ref) {\n ref.x = this.x * renderWidth;\n ref.y = this.y * renderHeight;\n ref.width = this.width * renderWidth;\n ref.height = this.height * renderHeight;\n return this;\n }\n /**\n * Returns a new Viewport copied from the current one\n * @returns a new Viewport\n */\n clone() {\n return new Viewport(this.x, this.y, this.width, this.height);\n }\n }\n\n /**\n * Represents a plane by the equation ax + by + cz + d = 0\n */\n class Plane {\n /**\n * Creates a Plane object according to the given floats a, b, c, d and the plane equation : ax + by + cz + d = 0\n * @param a a component of the plane\n * @param b b component of the plane\n * @param c c component of the plane\n * @param d d component of the plane\n */\n constructor(a, b, c, d) {\n this.normal = new Vector3(a, b, c);\n this.d = d;\n }\n /**\n * @returns the plane coordinates as a new array of 4 elements [a, b, c, d].\n */\n asArray() {\n return [this.normal.x, this.normal.y, this.normal.z, this.d];\n }\n // Methods\n /**\n * @returns a new plane copied from the current Plane.\n */\n clone() {\n return new Plane(this.normal.x, this.normal.y, this.normal.z, this.d);\n }\n /**\n * @returns the string \"Plane\".\n */\n getClassName() {\n return \"Plane\";\n }\n /**\n * @returns the Plane hash code.\n */\n getHashCode() {\n let hash = this.normal.getHashCode();\n hash = (hash * 397) ^ (this.d | 0);\n return hash;\n }\n /**\n * Normalize the current Plane in place.\n * @returns the updated Plane.\n */\n normalize() {\n const norm = Math.sqrt(this.normal.x * this.normal.x + this.normal.y * this.normal.y + this.normal.z * this.normal.z);\n let magnitude = 0.0;\n if (norm !== 0) {\n magnitude = 1.0 / norm;\n }\n this.normal.x *= magnitude;\n this.normal.y *= magnitude;\n this.normal.z *= magnitude;\n this.d *= magnitude;\n return this;\n }\n /**\n * Applies a transformation the plane and returns the result\n * @param transformation the transformation matrix to be applied to the plane\n * @returns a new Plane as the result of the transformation of the current Plane by the given matrix.\n */\n transform(transformation) {\n const invertedMatrix = Plane._TmpMatrix;\n transformation.invertToRef(invertedMatrix);\n const m = invertedMatrix.m;\n const x = this.normal.x;\n const y = this.normal.y;\n const z = this.normal.z;\n const d = this.d;\n const normalX = x * m[0] + y * m[1] + z * m[2] + d * m[3];\n const normalY = x * m[4] + y * m[5] + z * m[6] + d * m[7];\n const normalZ = x * m[8] + y * m[9] + z * m[10] + d * m[11];\n const finalD = x * m[12] + y * m[13] + z * m[14] + d * m[15];\n return new Plane(normalX, normalY, normalZ, finalD);\n }\n /**\n * Compute the dot product between the point and the plane normal\n * @param point point to calculate the dot product with\n * @returns the dot product (float) of the point coordinates and the plane normal.\n */\n dotCoordinate(point) {\n return this.normal.x * point.x + this.normal.y * point.y + this.normal.z * point.z + this.d;\n }\n /**\n * Updates the current Plane from the plane defined by the three given points.\n * @param point1 one of the points used to construct the plane\n * @param point2 one of the points used to construct the plane\n * @param point3 one of the points used to construct the plane\n * @returns the updated Plane.\n */\n copyFromPoints(point1, point2, point3) {\n const x1 = point2.x - point1.x;\n const y1 = point2.y - point1.y;\n const z1 = point2.z - point1.z;\n const x2 = point3.x - point1.x;\n const y2 = point3.y - point1.y;\n const z2 = point3.z - point1.z;\n const yz = y1 * z2 - z1 * y2;\n const xz = z1 * x2 - x1 * z2;\n const xy = x1 * y2 - y1 * x2;\n const pyth = Math.sqrt(yz * yz + xz * xz + xy * xy);\n let invPyth;\n if (pyth !== 0) {\n invPyth = 1.0 / pyth;\n }\n else {\n invPyth = 0.0;\n }\n this.normal.x = yz * invPyth;\n this.normal.y = xz * invPyth;\n this.normal.z = xy * invPyth;\n this.d = -(this.normal.x * point1.x + this.normal.y * point1.y + this.normal.z * point1.z);\n return this;\n }\n /**\n * Checks if the plane is facing a given direction (meaning if the plane's normal is pointing in the opposite direction of the given vector).\n * Note that for this function to work as expected you should make sure that:\n * - direction and the plane normal are normalized\n * - epsilon is a number just bigger than -1, something like -0.99 for eg\n * @param direction the direction to check if the plane is facing\n * @param epsilon value the dot product is compared against (returns true if dot <= epsilon)\n * @returns True if the plane is facing the given direction\n */\n isFrontFacingTo(direction, epsilon) {\n const dot = Vector3.Dot(this.normal, direction);\n return dot <= epsilon;\n }\n /**\n * Calculates the distance to a point\n * @param point point to calculate distance to\n * @returns the signed distance (float) from the given point to the Plane.\n */\n signedDistanceTo(point) {\n return Vector3.Dot(point, this.normal) + this.d;\n }\n // Statics\n /**\n * Creates a plane from an array\n * @param array the array to create a plane from\n * @returns a new Plane from the given array.\n */\n static FromArray(array) {\n return new Plane(array[0], array[1], array[2], array[3]);\n }\n /**\n * Creates a plane from three points\n * @param point1 point used to create the plane\n * @param point2 point used to create the plane\n * @param point3 point used to create the plane\n * @returns a new Plane defined by the three given points.\n */\n static FromPoints(point1, point2, point3) {\n const result = new Plane(0.0, 0.0, 0.0, 0.0);\n result.copyFromPoints(point1, point2, point3);\n return result;\n }\n /**\n * Creates a plane from an origin point and a normal\n * @param origin origin of the plane to be constructed\n * @param normal normal of the plane to be constructed\n * @returns a new Plane the normal vector to this plane at the given origin point.\n * Note : the vector \"normal\" is updated because normalized.\n */\n static FromPositionAndNormal(origin, normal) {\n const result = new Plane(0.0, 0.0, 0.0, 0.0);\n normal.normalize();\n result.normal = normal;\n result.d = -(normal.x * origin.x + normal.y * origin.y + normal.z * origin.z);\n return result;\n }\n /**\n * Calculates the distance from a plane and a point\n * @param origin origin of the plane to be constructed\n * @param normal normal of the plane to be constructed\n * @param point point to calculate distance to\n * @returns the signed distance between the plane defined by the normal vector at the \"origin\"\" point and the given other point.\n */\n static SignedDistanceToPlaneFromPositionAndNormal(origin, normal, point) {\n const d = -(normal.x * origin.x + normal.y * origin.y + normal.z * origin.z);\n return Vector3.Dot(point, normal) + d;\n }\n }\n Plane._TmpMatrix = Matrix.Identity();\n\n /**\n * Represents a camera frustum\n */\n class Frustum {\n /**\n * Gets the planes representing the frustum\n * @param transform matrix to be applied to the returned planes\n * @returns a new array of 6 Frustum planes computed by the given transformation matrix.\n */\n static GetPlanes(transform) {\n const frustumPlanes = [];\n for (let index = 0; index < 6; index++) {\n frustumPlanes.push(new Plane(0.0, 0.0, 0.0, 0.0));\n }\n Frustum.GetPlanesToRef(transform, frustumPlanes);\n return frustumPlanes;\n }\n /**\n * Gets the near frustum plane transformed by the transform matrix\n * @param transform transformation matrix to be applied to the resulting frustum plane\n * @param frustumPlane the resulting frustum plane\n */\n static GetNearPlaneToRef(transform, frustumPlane) {\n const m = transform.m;\n frustumPlane.normal.x = m[3] + m[2];\n frustumPlane.normal.y = m[7] + m[6];\n frustumPlane.normal.z = m[11] + m[10];\n frustumPlane.d = m[15] + m[14];\n frustumPlane.normalize();\n }\n /**\n * Gets the far frustum plane transformed by the transform matrix\n * @param transform transformation matrix to be applied to the resulting frustum plane\n * @param frustumPlane the resulting frustum plane\n */\n static GetFarPlaneToRef(transform, frustumPlane) {\n const m = transform.m;\n frustumPlane.normal.x = m[3] - m[2];\n frustumPlane.normal.y = m[7] - m[6];\n frustumPlane.normal.z = m[11] - m[10];\n frustumPlane.d = m[15] - m[14];\n frustumPlane.normalize();\n }\n /**\n * Gets the left frustum plane transformed by the transform matrix\n * @param transform transformation matrix to be applied to the resulting frustum plane\n * @param frustumPlane the resulting frustum plane\n */\n static GetLeftPlaneToRef(transform, frustumPlane) {\n const m = transform.m;\n frustumPlane.normal.x = m[3] + m[0];\n frustumPlane.normal.y = m[7] + m[4];\n frustumPlane.normal.z = m[11] + m[8];\n frustumPlane.d = m[15] + m[12];\n frustumPlane.normalize();\n }\n /**\n * Gets the right frustum plane transformed by the transform matrix\n * @param transform transformation matrix to be applied to the resulting frustum plane\n * @param frustumPlane the resulting frustum plane\n */\n static GetRightPlaneToRef(transform, frustumPlane) {\n const m = transform.m;\n frustumPlane.normal.x = m[3] - m[0];\n frustumPlane.normal.y = m[7] - m[4];\n frustumPlane.normal.z = m[11] - m[8];\n frustumPlane.d = m[15] - m[12];\n frustumPlane.normalize();\n }\n /**\n * Gets the top frustum plane transformed by the transform matrix\n * @param transform transformation matrix to be applied to the resulting frustum plane\n * @param frustumPlane the resulting frustum plane\n */\n static GetTopPlaneToRef(transform, frustumPlane) {\n const m = transform.m;\n frustumPlane.normal.x = m[3] - m[1];\n frustumPlane.normal.y = m[7] - m[5];\n frustumPlane.normal.z = m[11] - m[9];\n frustumPlane.d = m[15] - m[13];\n frustumPlane.normalize();\n }\n /**\n * Gets the bottom frustum plane transformed by the transform matrix\n * @param transform transformation matrix to be applied to the resulting frustum plane\n * @param frustumPlane the resulting frustum plane\n */\n static GetBottomPlaneToRef(transform, frustumPlane) {\n const m = transform.m;\n frustumPlane.normal.x = m[3] + m[1];\n frustumPlane.normal.y = m[7] + m[5];\n frustumPlane.normal.z = m[11] + m[9];\n frustumPlane.d = m[15] + m[13];\n frustumPlane.normalize();\n }\n /**\n * Sets the given array \"frustumPlanes\" with the 6 Frustum planes computed by the given transformation matrix.\n * @param transform transformation matrix to be applied to the resulting frustum planes\n * @param frustumPlanes the resulting frustum planes\n */\n static GetPlanesToRef(transform, frustumPlanes) {\n // Near\n Frustum.GetNearPlaneToRef(transform, frustumPlanes[0]);\n // Far\n Frustum.GetFarPlaneToRef(transform, frustumPlanes[1]);\n // Left\n Frustum.GetLeftPlaneToRef(transform, frustumPlanes[2]);\n // Right\n Frustum.GetRightPlaneToRef(transform, frustumPlanes[3]);\n // Top\n Frustum.GetTopPlaneToRef(transform, frustumPlanes[4]);\n // Bottom\n Frustum.GetBottomPlaneToRef(transform, frustumPlanes[5]);\n }\n /**\n * Tests if a point is located between the frustum planes.\n * @param point defines the point to test\n * @param frustumPlanes defines the frustum planes to test\n * @returns true if the point is located between the frustum planes\n */\n static IsPointInFrustum(point, frustumPlanes) {\n for (let i = 0; i < 6; i++) {\n if (frustumPlanes[i].dotCoordinate(point) < 0) {\n return false;\n }\n }\n return true;\n }\n }\n\n /**\n * This is the base class of all the camera used in the application.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras\n */\n class Camera extends Node {\n /**\n * Define the current local position of the camera in the scene\n */\n get position() {\n return this._position;\n }\n set position(newPosition) {\n this._position = newPosition;\n }\n /**\n * The vector the camera should consider as up.\n * (default is Vector3(0, 1, 0) aka Vector3.Up())\n */\n set upVector(vec) {\n this._upVector = vec;\n }\n get upVector() {\n return this._upVector;\n }\n /**\n * The screen area in scene units squared\n */\n get screenArea() {\n var _a, _b, _c, _d;\n let x = 0;\n let y = 0;\n if (this.mode === Camera.PERSPECTIVE_CAMERA) {\n if (this.fovMode === Camera.FOVMODE_VERTICAL_FIXED) {\n y = this.minZ * 2 * Math.tan(this.fov / 2);\n x = this.getEngine().getAspectRatio(this) * y;\n }\n else {\n x = this.minZ * 2 * Math.tan(this.fov / 2);\n y = x / this.getEngine().getAspectRatio(this);\n }\n }\n else {\n const halfWidth = this.getEngine().getRenderWidth() / 2.0;\n const halfHeight = this.getEngine().getRenderHeight() / 2.0;\n x = ((_a = this.orthoRight) !== null && _a !== void 0 ? _a : halfWidth) - ((_b = this.orthoLeft) !== null && _b !== void 0 ? _b : -halfWidth);\n y = ((_c = this.orthoTop) !== null && _c !== void 0 ? _c : halfHeight) - ((_d = this.orthoBottom) !== null && _d !== void 0 ? _d : -halfHeight);\n }\n return x * y;\n }\n set orthoLeft(value) {\n this._orthoLeft = value;\n for (const rigCamera of this._rigCameras) {\n rigCamera.orthoLeft = value;\n }\n }\n get orthoLeft() {\n return this._orthoLeft;\n }\n set orthoRight(value) {\n this._orthoRight = value;\n for (const rigCamera of this._rigCameras) {\n rigCamera.orthoRight = value;\n }\n }\n get orthoRight() {\n return this._orthoRight;\n }\n set orthoBottom(value) {\n this._orthoBottom = value;\n for (const rigCamera of this._rigCameras) {\n rigCamera.orthoBottom = value;\n }\n }\n get orthoBottom() {\n return this._orthoBottom;\n }\n set orthoTop(value) {\n this._orthoTop = value;\n for (const rigCamera of this._rigCameras) {\n rigCamera.orthoTop = value;\n }\n }\n get orthoTop() {\n return this._orthoTop;\n }\n set mode(mode) {\n this._mode = mode;\n // Pass the mode down to the rig cameras\n for (const rigCamera of this._rigCameras) {\n rigCamera.mode = mode;\n }\n }\n get mode() {\n return this._mode;\n }\n /**\n * Instantiates a new camera object.\n * This should not be used directly but through the inherited cameras: ArcRotate, Free...\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras\n * @param name Defines the name of the camera in the scene\n * @param position Defines the position of the camera\n * @param scene Defines the scene the camera belongs too\n * @param setActiveOnSceneIfNoneActive Defines if the camera should be set as active after creation if no other camera have been defined in the scene\n */\n constructor(name, position, scene, setActiveOnSceneIfNoneActive = true) {\n super(name, scene);\n /** @internal */\n this._position = Vector3.Zero();\n this._upVector = Vector3.Up();\n /**\n * Define the current limit on the left side for an orthographic camera\n * In scene unit\n */\n this._orthoLeft = null;\n /**\n * Define the current limit on the right side for an orthographic camera\n * In scene unit\n */\n this._orthoRight = null;\n /**\n * Define the current limit on the bottom side for an orthographic camera\n * In scene unit\n */\n this._orthoBottom = null;\n /**\n * Define the current limit on the top side for an orthographic camera\n * In scene unit\n */\n this._orthoTop = null;\n /**\n * Field Of View is set in Radians. (default is 0.8)\n */\n this.fov = 0.8;\n /**\n * Projection plane tilt around the X axis (horizontal), set in Radians. (default is 0)\n * Can be used to make vertical lines in world space actually vertical on the screen.\n * See https://forum.babylonjs.com/t/add-vertical-shift-to-3ds-max-exporter-babylon-cameras/17480\n */\n this.projectionPlaneTilt = 0;\n /**\n * Define the minimum distance the camera can see from.\n * This is important to note that the depth buffer are not infinite and the closer it starts\n * the more your scene might encounter depth fighting issue.\n */\n this.minZ = 1;\n /**\n * Define the maximum distance the camera can see to.\n * This is important to note that the depth buffer are not infinite and the further it end\n * the more your scene might encounter depth fighting issue.\n */\n this.maxZ = 10000.0;\n /**\n * Define the default inertia of the camera.\n * This helps giving a smooth feeling to the camera movement.\n */\n this.inertia = 0.9;\n /**\n * Define the mode of the camera (Camera.PERSPECTIVE_CAMERA or Camera.ORTHOGRAPHIC_CAMERA)\n */\n this._mode = Camera.PERSPECTIVE_CAMERA;\n /**\n * Define whether the camera is intermediate.\n * This is useful to not present the output directly to the screen in case of rig without post process for instance\n */\n this.isIntermediate = false;\n /**\n * Define the viewport of the camera.\n * This correspond to the portion of the screen the camera will render to in normalized 0 to 1 unit.\n */\n this.viewport = new Viewport(0, 0, 1.0, 1.0);\n /**\n * Restricts the camera to viewing objects with the same layerMask.\n * A camera with a layerMask of 1 will render mesh.layerMask & camera.layerMask!== 0\n */\n this.layerMask = 0x0fffffff;\n /**\n * fovMode sets the camera frustum bounds to the viewport bounds. (default is FOVMODE_VERTICAL_FIXED)\n */\n this.fovMode = Camera.FOVMODE_VERTICAL_FIXED;\n /**\n * Rig mode of the camera.\n * This is useful to create the camera with two \"eyes\" instead of one to create VR or stereoscopic scenes.\n * This is normally controlled byt the camera themselves as internal use.\n */\n this.cameraRigMode = Camera.RIG_MODE_NONE;\n /**\n * Defines the list of custom render target which are rendered to and then used as the input to this camera's render. Eg. display another camera view on a TV in the main scene\n * This is pretty helpful if you wish to make a camera render to a texture you could reuse somewhere\n * else in the scene. (Eg. security camera)\n *\n * To change the final output target of the camera, camera.outputRenderTarget should be used instead (eg. webXR renders to a render target corresponding to an HMD)\n */\n this.customRenderTargets = new Array();\n /**\n * When set, the camera will render to this render target instead of the default canvas\n *\n * If the desire is to use the output of a camera as a texture in the scene consider using camera.customRenderTargets instead\n */\n this.outputRenderTarget = null;\n /**\n * Observable triggered when the camera view matrix has changed.\n */\n this.onViewMatrixChangedObservable = new Observable$1();\n /**\n * Observable triggered when the camera Projection matrix has changed.\n */\n this.onProjectionMatrixChangedObservable = new Observable$1();\n /**\n * Observable triggered when the inputs have been processed.\n */\n this.onAfterCheckInputsObservable = new Observable$1();\n /**\n * Observable triggered when reset has been called and applied to the camera.\n */\n this.onRestoreStateObservable = new Observable$1();\n /**\n * Is this camera a part of a rig system?\n */\n this.isRigCamera = false;\n /** @internal */\n this._rigCameras = new Array();\n this._webvrViewMatrix = Matrix.Identity();\n /** @internal */\n this._skipRendering = false;\n /** @internal */\n this._projectionMatrix = new Matrix();\n /** @internal */\n this._postProcesses = new Array();\n /** @internal */\n this._activeMeshes = new SmartArray(256);\n this._globalPosition = Vector3.Zero();\n /** @internal */\n this._computedViewMatrix = Matrix.Identity();\n this._doNotComputeProjectionMatrix = false;\n this._transformMatrix = Matrix.Zero();\n this._refreshFrustumPlanes = true;\n this._absoluteRotation = Quaternion.Identity();\n /** @internal */\n this._isCamera = true;\n /** @internal */\n this._isLeftCamera = false;\n /** @internal */\n this._isRightCamera = false;\n this.getScene().addCamera(this);\n if (setActiveOnSceneIfNoneActive && !this.getScene().activeCamera) {\n this.getScene().activeCamera = this;\n }\n this.position = position;\n this.renderPassId = this.getScene().getEngine().createRenderPassId(`Camera ${name}`);\n }\n /**\n * Store current camera state (fov, position, etc..)\n * @returns the camera\n */\n storeState() {\n this._stateStored = true;\n this._storedFov = this.fov;\n return this;\n }\n /**\n * Restores the camera state values if it has been stored. You must call storeState() first\n */\n _restoreStateValues() {\n if (!this._stateStored) {\n return false;\n }\n this.fov = this._storedFov;\n return true;\n }\n /**\n * Restored camera state. You must call storeState() first.\n * @returns true if restored and false otherwise\n */\n restoreState() {\n if (this._restoreStateValues()) {\n this.onRestoreStateObservable.notifyObservers(this);\n return true;\n }\n return false;\n }\n /**\n * Gets the class name of the camera.\n * @returns the class name\n */\n getClassName() {\n return \"Camera\";\n }\n /**\n * Gets a string representation of the camera useful for debug purpose.\n * @param fullDetails Defines that a more verbose level of logging is required\n * @returns the string representation\n */\n toString(fullDetails) {\n let ret = \"Name: \" + this.name;\n ret += \", type: \" + this.getClassName();\n if (this.animations) {\n for (let i = 0; i < this.animations.length; i++) {\n ret += \", animation[0]: \" + this.animations[i].toString(fullDetails);\n }\n }\n return ret;\n }\n /**\n * Automatically tilts the projection plane, using `projectionPlaneTilt`, to correct the perspective effect on vertical lines.\n */\n applyVerticalCorrection() {\n const rot = this.absoluteRotation.toEulerAngles();\n this.projectionPlaneTilt = this._scene.useRightHandedSystem ? -rot.x : rot.x;\n }\n /**\n * Gets the current world space position of the camera.\n */\n get globalPosition() {\n return this._globalPosition;\n }\n /**\n * Gets the list of active meshes this frame (meshes no culled or excluded by lod s in the frame)\n * @returns the active meshe list\n */\n getActiveMeshes() {\n return this._activeMeshes;\n }\n /**\n * Check whether a mesh is part of the current active mesh list of the camera\n * @param mesh Defines the mesh to check\n * @returns true if active, false otherwise\n */\n isActiveMesh(mesh) {\n return this._activeMeshes.indexOf(mesh) !== -1;\n }\n /**\n * Is this camera ready to be used/rendered\n * @param completeCheck defines if a complete check (including post processes) has to be done (false by default)\n * @returns true if the camera is ready\n */\n isReady(completeCheck = false) {\n if (completeCheck) {\n for (const pp of this._postProcesses) {\n if (pp && !pp.isReady()) {\n return false;\n }\n }\n }\n return super.isReady(completeCheck);\n }\n /** @internal */\n _initCache() {\n super._initCache();\n this._cache.position = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\n this._cache.upVector = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\n this._cache.mode = undefined;\n this._cache.minZ = undefined;\n this._cache.maxZ = undefined;\n this._cache.fov = undefined;\n this._cache.fovMode = undefined;\n this._cache.aspectRatio = undefined;\n this._cache.orthoLeft = undefined;\n this._cache.orthoRight = undefined;\n this._cache.orthoBottom = undefined;\n this._cache.orthoTop = undefined;\n this._cache.renderWidth = undefined;\n this._cache.renderHeight = undefined;\n }\n /**\n * @internal\n */\n _updateCache(ignoreParentClass) {\n if (!ignoreParentClass) {\n super._updateCache();\n }\n this._cache.position.copyFrom(this.position);\n this._cache.upVector.copyFrom(this.upVector);\n }\n /** @internal */\n _isSynchronized() {\n return this._isSynchronizedViewMatrix() && this._isSynchronizedProjectionMatrix();\n }\n /** @internal */\n _isSynchronizedViewMatrix() {\n if (!super._isSynchronized()) {\n return false;\n }\n return this._cache.position.equals(this.position) && this._cache.upVector.equals(this.upVector) && this.isSynchronizedWithParent();\n }\n /** @internal */\n _isSynchronizedProjectionMatrix() {\n let check = this._cache.mode === this.mode && this._cache.minZ === this.minZ && this._cache.maxZ === this.maxZ;\n if (!check) {\n return false;\n }\n const engine = this.getEngine();\n if (this.mode === Camera.PERSPECTIVE_CAMERA) {\n check =\n this._cache.fov === this.fov &&\n this._cache.fovMode === this.fovMode &&\n this._cache.aspectRatio === engine.getAspectRatio(this) &&\n this._cache.projectionPlaneTilt === this.projectionPlaneTilt;\n }\n else {\n check =\n this._cache.orthoLeft === this.orthoLeft &&\n this._cache.orthoRight === this.orthoRight &&\n this._cache.orthoBottom === this.orthoBottom &&\n this._cache.orthoTop === this.orthoTop &&\n this._cache.renderWidth === engine.getRenderWidth() &&\n this._cache.renderHeight === engine.getRenderHeight();\n }\n return check;\n }\n /**\n * Attach the input controls to a specific dom element to get the input from.\n * This function is here because typescript removes the typing of the last function.\n * @param _ignored defines an ignored parameter kept for backward compatibility.\n * @param _noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\n */\n attachControl(_ignored, _noPreventDefault) { }\n /**\n * Detach the current controls from the specified dom element.\n * This function is here because typescript removes the typing of the last function.\n * @param _ignored defines an ignored parameter kept for backward compatibility.\n */\n detachControl(_ignored) { }\n /**\n * Update the camera state according to the different inputs gathered during the frame.\n */\n update() {\n this._checkInputs();\n if (this.cameraRigMode !== Camera.RIG_MODE_NONE) {\n this._updateRigCameras();\n }\n // Attempt to update the camera's view and projection matrices.\n // This call is being made because these matrices are no longer being updated\n // as a part of the picking ray process (in addition to scene.render).\n this.getViewMatrix();\n this.getProjectionMatrix();\n }\n /** @internal */\n _checkInputs() {\n this.onAfterCheckInputsObservable.notifyObservers(this);\n }\n /** @internal */\n get rigCameras() {\n return this._rigCameras;\n }\n /**\n * Gets the post process used by the rig cameras\n */\n get rigPostProcess() {\n return this._rigPostProcess;\n }\n /**\n * Internal, gets the first post process.\n * @returns the first post process to be run on this camera.\n */\n _getFirstPostProcess() {\n for (let ppIndex = 0; ppIndex < this._postProcesses.length; ppIndex++) {\n if (this._postProcesses[ppIndex] !== null) {\n return this._postProcesses[ppIndex];\n }\n }\n return null;\n }\n _cascadePostProcessesToRigCams() {\n // invalidate framebuffer\n const firstPostProcess = this._getFirstPostProcess();\n if (firstPostProcess) {\n firstPostProcess.markTextureDirty();\n }\n // glue the rigPostProcess to the end of the user postprocesses & assign to each sub-camera\n for (let i = 0, len = this._rigCameras.length; i < len; i++) {\n const cam = this._rigCameras[i];\n const rigPostProcess = cam._rigPostProcess;\n // for VR rig, there does not have to be a post process\n if (rigPostProcess) {\n const isPass = rigPostProcess.getEffectName() === \"pass\";\n if (isPass) {\n // any rig which has a PassPostProcess for rig[0], cannot be isIntermediate when there are also user postProcesses\n cam.isIntermediate = this._postProcesses.length === 0;\n }\n cam._postProcesses = this._postProcesses.slice(0).concat(rigPostProcess);\n rigPostProcess.markTextureDirty();\n }\n else {\n cam._postProcesses = this._postProcesses.slice(0);\n }\n }\n }\n /**\n * Attach a post process to the camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/usePostProcesses#attach-postprocess\n * @param postProcess The post process to attach to the camera\n * @param insertAt The position of the post process in case several of them are in use in the scene\n * @returns the position the post process has been inserted at\n */\n attachPostProcess(postProcess, insertAt = null) {\n if (!postProcess.isReusable() && this._postProcesses.indexOf(postProcess) > -1) {\n Logger.Error(\"You're trying to reuse a post process not defined as reusable.\");\n return 0;\n }\n if (insertAt == null || insertAt < 0) {\n this._postProcesses.push(postProcess);\n }\n else if (this._postProcesses[insertAt] === null) {\n this._postProcesses[insertAt] = postProcess;\n }\n else {\n this._postProcesses.splice(insertAt, 0, postProcess);\n }\n this._cascadePostProcessesToRigCams(); // also ensures framebuffer invalidated\n // Update prePass\n if (this._scene.prePassRenderer) {\n this._scene.prePassRenderer.markAsDirty();\n }\n return this._postProcesses.indexOf(postProcess);\n }\n /**\n * Detach a post process to the camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/usePostProcesses#attach-postprocess\n * @param postProcess The post process to detach from the camera\n */\n detachPostProcess(postProcess) {\n const idx = this._postProcesses.indexOf(postProcess);\n if (idx !== -1) {\n this._postProcesses[idx] = null;\n }\n // Update prePass\n if (this._scene.prePassRenderer) {\n this._scene.prePassRenderer.markAsDirty();\n }\n this._cascadePostProcessesToRigCams(); // also ensures framebuffer invalidated\n }\n /**\n * Gets the current world matrix of the camera\n */\n getWorldMatrix() {\n if (this._isSynchronizedViewMatrix()) {\n return this._worldMatrix;\n }\n // Getting the the view matrix will also compute the world matrix.\n this.getViewMatrix();\n return this._worldMatrix;\n }\n /** @internal */\n _getViewMatrix() {\n return Matrix.Identity();\n }\n /**\n * Gets the current view matrix of the camera.\n * @param force forces the camera to recompute the matrix without looking at the cached state\n * @returns the view matrix\n */\n getViewMatrix(force) {\n if (!force && this._isSynchronizedViewMatrix()) {\n return this._computedViewMatrix;\n }\n this.updateCache();\n this._computedViewMatrix = this._getViewMatrix();\n this._currentRenderId = this.getScene().getRenderId();\n this._childUpdateId++;\n this._refreshFrustumPlanes = true;\n if (this._cameraRigParams && this._cameraRigParams.vrPreViewMatrix) {\n this._computedViewMatrix.multiplyToRef(this._cameraRigParams.vrPreViewMatrix, this._computedViewMatrix);\n }\n // Notify parent camera if rig camera is changed\n if (this.parent && this.parent.onViewMatrixChangedObservable) {\n this.parent.onViewMatrixChangedObservable.notifyObservers(this.parent);\n }\n this.onViewMatrixChangedObservable.notifyObservers(this);\n this._computedViewMatrix.invertToRef(this._worldMatrix);\n return this._computedViewMatrix;\n }\n /**\n * Freeze the projection matrix.\n * It will prevent the cache check of the camera projection compute and can speed up perf\n * if no parameter of the camera are meant to change\n * @param projection Defines manually a projection if necessary\n */\n freezeProjectionMatrix(projection) {\n this._doNotComputeProjectionMatrix = true;\n if (projection !== undefined) {\n this._projectionMatrix = projection;\n }\n }\n /**\n * Unfreeze the projection matrix if it has previously been freezed by freezeProjectionMatrix.\n */\n unfreezeProjectionMatrix() {\n this._doNotComputeProjectionMatrix = false;\n }\n /**\n * Gets the current projection matrix of the camera.\n * @param force forces the camera to recompute the matrix without looking at the cached state\n * @returns the projection matrix\n */\n getProjectionMatrix(force) {\n var _a, _b, _c, _d, _e, _f, _g, _h;\n if (this._doNotComputeProjectionMatrix || (!force && this._isSynchronizedProjectionMatrix())) {\n return this._projectionMatrix;\n }\n // Cache\n this._cache.mode = this.mode;\n this._cache.minZ = this.minZ;\n this._cache.maxZ = this.maxZ;\n // Matrix\n this._refreshFrustumPlanes = true;\n const engine = this.getEngine();\n const scene = this.getScene();\n const reverseDepth = engine.useReverseDepthBuffer;\n if (this.mode === Camera.PERSPECTIVE_CAMERA) {\n this._cache.fov = this.fov;\n this._cache.fovMode = this.fovMode;\n this._cache.aspectRatio = engine.getAspectRatio(this);\n this._cache.projectionPlaneTilt = this.projectionPlaneTilt;\n if (this.minZ <= 0) {\n this.minZ = 0.1;\n }\n let getProjectionMatrix;\n if (scene.useRightHandedSystem) {\n getProjectionMatrix = Matrix.PerspectiveFovRHToRef;\n }\n else {\n getProjectionMatrix = Matrix.PerspectiveFovLHToRef;\n }\n getProjectionMatrix(this.fov, engine.getAspectRatio(this), reverseDepth ? this.maxZ : this.minZ, reverseDepth ? this.minZ : this.maxZ, this._projectionMatrix, this.fovMode === Camera.FOVMODE_VERTICAL_FIXED, engine.isNDCHalfZRange, this.projectionPlaneTilt, reverseDepth);\n }\n else {\n const halfWidth = engine.getRenderWidth() / 2.0;\n const halfHeight = engine.getRenderHeight() / 2.0;\n if (scene.useRightHandedSystem) {\n Matrix.OrthoOffCenterRHToRef((_a = this.orthoLeft) !== null && _a !== void 0 ? _a : -halfWidth, (_b = this.orthoRight) !== null && _b !== void 0 ? _b : halfWidth, (_c = this.orthoBottom) !== null && _c !== void 0 ? _c : -halfHeight, (_d = this.orthoTop) !== null && _d !== void 0 ? _d : halfHeight, reverseDepth ? this.maxZ : this.minZ, reverseDepth ? this.minZ : this.maxZ, this._projectionMatrix, engine.isNDCHalfZRange);\n }\n else {\n Matrix.OrthoOffCenterLHToRef((_e = this.orthoLeft) !== null && _e !== void 0 ? _e : -halfWidth, (_f = this.orthoRight) !== null && _f !== void 0 ? _f : halfWidth, (_g = this.orthoBottom) !== null && _g !== void 0 ? _g : -halfHeight, (_h = this.orthoTop) !== null && _h !== void 0 ? _h : halfHeight, reverseDepth ? this.maxZ : this.minZ, reverseDepth ? this.minZ : this.maxZ, this._projectionMatrix, engine.isNDCHalfZRange);\n }\n this._cache.orthoLeft = this.orthoLeft;\n this._cache.orthoRight = this.orthoRight;\n this._cache.orthoBottom = this.orthoBottom;\n this._cache.orthoTop = this.orthoTop;\n this._cache.renderWidth = engine.getRenderWidth();\n this._cache.renderHeight = engine.getRenderHeight();\n }\n this.onProjectionMatrixChangedObservable.notifyObservers(this);\n return this._projectionMatrix;\n }\n /**\n * Gets the transformation matrix (ie. the multiplication of view by projection matrices)\n * @returns a Matrix\n */\n getTransformationMatrix() {\n this._computedViewMatrix.multiplyToRef(this._projectionMatrix, this._transformMatrix);\n return this._transformMatrix;\n }\n _updateFrustumPlanes() {\n if (!this._refreshFrustumPlanes) {\n return;\n }\n this.getTransformationMatrix();\n if (!this._frustumPlanes) {\n this._frustumPlanes = Frustum.GetPlanes(this._transformMatrix);\n }\n else {\n Frustum.GetPlanesToRef(this._transformMatrix, this._frustumPlanes);\n }\n this._refreshFrustumPlanes = false;\n }\n /**\n * Checks if a cullable object (mesh...) is in the camera frustum\n * This checks the bounding box center. See isCompletelyInFrustum for a full bounding check\n * @param target The object to check\n * @param checkRigCameras If the rig cameras should be checked (eg. with webVR camera both eyes should be checked) (Default: false)\n * @returns true if the object is in frustum otherwise false\n */\n isInFrustum(target, checkRigCameras = false) {\n this._updateFrustumPlanes();\n if (checkRigCameras && this.rigCameras.length > 0) {\n let result = false;\n this.rigCameras.forEach((cam) => {\n cam._updateFrustumPlanes();\n result = result || target.isInFrustum(cam._frustumPlanes);\n });\n return result;\n }\n else {\n return target.isInFrustum(this._frustumPlanes);\n }\n }\n /**\n * Checks if a cullable object (mesh...) is in the camera frustum\n * Unlike isInFrustum this checks the full bounding box\n * @param target The object to check\n * @returns true if the object is in frustum otherwise false\n */\n isCompletelyInFrustum(target) {\n this._updateFrustumPlanes();\n return target.isCompletelyInFrustum(this._frustumPlanes);\n }\n /**\n * Gets a ray in the forward direction from the camera.\n * @param length Defines the length of the ray to create\n * @param transform Defines the transform to apply to the ray, by default the world matrix is used to create a workd space ray\n * @param origin Defines the start point of the ray which defaults to the camera position\n * @returns the forward ray\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n getForwardRay(length = 100, transform, origin) {\n throw _WarnImport(\"Ray\");\n }\n /**\n * Gets a ray in the forward direction from the camera.\n * @param refRay the ray to (re)use when setting the values\n * @param length Defines the length of the ray to create\n * @param transform Defines the transform to apply to the ray, by default the world matrx is used to create a workd space ray\n * @param origin Defines the start point of the ray which defaults to the camera position\n * @returns the forward ray\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n getForwardRayToRef(refRay, length = 100, transform, origin) {\n throw _WarnImport(\"Ray\");\n }\n /**\n * Releases resources associated with this node.\n * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default)\n * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default)\n */\n dispose(doNotRecurse, disposeMaterialAndTextures = false) {\n // Observables\n this.onViewMatrixChangedObservable.clear();\n this.onProjectionMatrixChangedObservable.clear();\n this.onAfterCheckInputsObservable.clear();\n this.onRestoreStateObservable.clear();\n // Inputs\n if (this.inputs) {\n this.inputs.clear();\n }\n // Animations\n this.getScene().stopAnimation(this);\n // Remove from scene\n this.getScene().removeCamera(this);\n while (this._rigCameras.length > 0) {\n const camera = this._rigCameras.pop();\n if (camera) {\n camera.dispose();\n }\n }\n if (this._parentContainer) {\n const index = this._parentContainer.cameras.indexOf(this);\n if (index > -1) {\n this._parentContainer.cameras.splice(index, 1);\n }\n this._parentContainer = null;\n }\n // Postprocesses\n if (this._rigPostProcess) {\n this._rigPostProcess.dispose(this);\n this._rigPostProcess = null;\n this._postProcesses.length = 0;\n }\n else if (this.cameraRigMode !== Camera.RIG_MODE_NONE) {\n this._rigPostProcess = null;\n this._postProcesses.length = 0;\n }\n else {\n let i = this._postProcesses.length;\n while (--i >= 0) {\n const postProcess = this._postProcesses[i];\n if (postProcess) {\n postProcess.dispose(this);\n }\n }\n }\n // Render targets\n let i = this.customRenderTargets.length;\n while (--i >= 0) {\n this.customRenderTargets[i].dispose();\n }\n this.customRenderTargets.length = 0;\n // Active Meshes\n this._activeMeshes.dispose();\n this.getScene().getEngine().releaseRenderPassId(this.renderPassId);\n super.dispose(doNotRecurse, disposeMaterialAndTextures);\n }\n /**\n * Gets the left camera of a rig setup in case of Rigged Camera\n */\n get isLeftCamera() {\n return this._isLeftCamera;\n }\n /**\n * Gets the right camera of a rig setup in case of Rigged Camera\n */\n get isRightCamera() {\n return this._isRightCamera;\n }\n /**\n * Gets the left camera of a rig setup in case of Rigged Camera\n */\n get leftCamera() {\n if (this._rigCameras.length < 1) {\n return null;\n }\n return this._rigCameras[0];\n }\n /**\n * Gets the right camera of a rig setup in case of Rigged Camera\n */\n get rightCamera() {\n if (this._rigCameras.length < 2) {\n return null;\n }\n return this._rigCameras[1];\n }\n /**\n * Gets the left camera target of a rig setup in case of Rigged Camera\n * @returns the target position\n */\n getLeftTarget() {\n if (this._rigCameras.length < 1) {\n return null;\n }\n return this._rigCameras[0].getTarget();\n }\n /**\n * Gets the right camera target of a rig setup in case of Rigged Camera\n * @returns the target position\n */\n getRightTarget() {\n if (this._rigCameras.length < 2) {\n return null;\n }\n return this._rigCameras[1].getTarget();\n }\n /**\n * @internal\n */\n setCameraRigMode(mode, rigParams) {\n if (this.cameraRigMode === mode) {\n return;\n }\n while (this._rigCameras.length > 0) {\n const camera = this._rigCameras.pop();\n if (camera) {\n camera.dispose();\n }\n }\n this.cameraRigMode = mode;\n this._cameraRigParams = {};\n //we have to implement stereo camera calcultating left and right viewpoints from interaxialDistance and target,\n //not from a given angle as it is now, but until that complete code rewriting provisional stereoHalfAngle value is introduced\n this._cameraRigParams.interaxialDistance = rigParams.interaxialDistance || 0.0637;\n this._cameraRigParams.stereoHalfAngle = Tools.ToRadians(this._cameraRigParams.interaxialDistance / 0.0637);\n // create the rig cameras, unless none\n if (this.cameraRigMode !== Camera.RIG_MODE_NONE) {\n const leftCamera = this.createRigCamera(this.name + \"_L\", 0);\n if (leftCamera) {\n leftCamera._isLeftCamera = true;\n }\n const rightCamera = this.createRigCamera(this.name + \"_R\", 1);\n if (rightCamera) {\n rightCamera._isRightCamera = true;\n }\n if (leftCamera && rightCamera) {\n this._rigCameras.push(leftCamera);\n this._rigCameras.push(rightCamera);\n }\n }\n this._setRigMode(rigParams);\n this._cascadePostProcessesToRigCams();\n this.update();\n }\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _setRigMode(rigParams) {\n // no-op\n }\n /** @internal */\n _getVRProjectionMatrix() {\n Matrix.PerspectiveFovLHToRef(this._cameraRigParams.vrMetrics.aspectRatioFov, this._cameraRigParams.vrMetrics.aspectRatio, this.minZ, this.maxZ, this._cameraRigParams.vrWorkMatrix, true, this.getEngine().isNDCHalfZRange);\n this._cameraRigParams.vrWorkMatrix.multiplyToRef(this._cameraRigParams.vrHMatrix, this._projectionMatrix);\n return this._projectionMatrix;\n }\n _updateCameraRotationMatrix() {\n //Here for WebVR\n }\n _updateWebVRCameraRotationMatrix() {\n //Here for WebVR\n }\n /**\n * This function MUST be overwritten by the different WebVR cameras available.\n * The context in which it is running is the RIG camera. So 'this' is the TargetCamera, left or right.\n * @internal\n */\n _getWebVRProjectionMatrix() {\n return Matrix.Identity();\n }\n /**\n * This function MUST be overwritten by the different WebVR cameras available.\n * The context in which it is running is the RIG camera. So 'this' is the TargetCamera, left or right.\n * @internal\n */\n _getWebVRViewMatrix() {\n return Matrix.Identity();\n }\n /**\n * @internal\n */\n setCameraRigParameter(name, value) {\n if (!this._cameraRigParams) {\n this._cameraRigParams = {};\n }\n this._cameraRigParams[name] = value;\n //provisionnally:\n if (name === \"interaxialDistance\") {\n this._cameraRigParams.stereoHalfAngle = Tools.ToRadians(value / 0.0637);\n }\n }\n /**\n * needs to be overridden by children so sub has required properties to be copied\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n createRigCamera(name, cameraIndex) {\n return null;\n }\n /**\n * May need to be overridden by children\n * @internal\n */\n _updateRigCameras() {\n for (let i = 0; i < this._rigCameras.length; i++) {\n this._rigCameras[i].minZ = this.minZ;\n this._rigCameras[i].maxZ = this.maxZ;\n this._rigCameras[i].fov = this.fov;\n this._rigCameras[i].upVector.copyFrom(this.upVector);\n }\n // only update viewport when ANAGLYPH\n if (this.cameraRigMode === Camera.RIG_MODE_STEREOSCOPIC_ANAGLYPH) {\n this._rigCameras[0].viewport = this._rigCameras[1].viewport = this.viewport;\n }\n }\n /** @internal */\n _setupInputs() { }\n /**\n * Serialiaze the camera setup to a json representation\n * @returns the JSON representation\n */\n serialize() {\n const serializationObject = SerializationHelper.Serialize(this);\n serializationObject.uniqueId = this.uniqueId;\n // Type\n serializationObject.type = this.getClassName();\n // Parent\n if (this.parent) {\n this.parent._serializeAsParent(serializationObject);\n }\n if (this.inputs) {\n this.inputs.serialize(serializationObject);\n }\n // Animations\n SerializationHelper.AppendSerializedAnimations(this, serializationObject);\n serializationObject.ranges = this.serializeAnimationRanges();\n serializationObject.isEnabled = this.isEnabled();\n return serializationObject;\n }\n /**\n * Clones the current camera.\n * @param name The cloned camera name\n * @param newParent The cloned camera's new parent (none by default)\n * @returns the cloned camera\n */\n clone(name, newParent = null) {\n const camera = SerializationHelper.Clone(Camera.GetConstructorFromName(this.getClassName(), name, this.getScene(), this.interaxialDistance, this.isStereoscopicSideBySide), this);\n camera.name = name;\n camera.parent = newParent;\n this.onClonedObservable.notifyObservers(camera);\n return camera;\n }\n /**\n * Gets the direction of the camera relative to a given local axis.\n * @param localAxis Defines the reference axis to provide a relative direction.\n * @returns the direction\n */\n getDirection(localAxis) {\n const result = Vector3.Zero();\n this.getDirectionToRef(localAxis, result);\n return result;\n }\n /**\n * Returns the current camera absolute rotation\n */\n get absoluteRotation() {\n this.getWorldMatrix().decompose(undefined, this._absoluteRotation);\n return this._absoluteRotation;\n }\n /**\n * Gets the direction of the camera relative to a given local axis into a passed vector.\n * @param localAxis Defines the reference axis to provide a relative direction.\n * @param result Defines the vector to store the result in\n */\n getDirectionToRef(localAxis, result) {\n Vector3.TransformNormalToRef(localAxis, this.getWorldMatrix(), result);\n }\n /**\n * Gets a camera constructor for a given camera type\n * @param type The type of the camera to construct (should be equal to one of the camera class name)\n * @param name The name of the camera the result will be able to instantiate\n * @param scene The scene the result will construct the camera in\n * @param interaxial_distance In case of stereoscopic setup, the distance between both eyes\n * @param isStereoscopicSideBySide In case of stereoscopic setup, should the sereo be side b side\n * @returns a factory method to construct the camera\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n static GetConstructorFromName(type, name, scene, interaxial_distance = 0, isStereoscopicSideBySide = true) {\n const constructorFunc = Node.Construct(type, name, scene, {\n // eslint-disable-next-line @typescript-eslint/naming-convention\n interaxial_distance: interaxial_distance,\n isStereoscopicSideBySide: isStereoscopicSideBySide,\n });\n if (constructorFunc) {\n return constructorFunc;\n }\n // Default to universal camera\n return () => Camera._CreateDefaultParsedCamera(name, scene);\n }\n /**\n * Compute the world matrix of the camera.\n * @returns the camera world matrix\n */\n computeWorldMatrix() {\n return this.getWorldMatrix();\n }\n /**\n * Parse a JSON and creates the camera from the parsed information\n * @param parsedCamera The JSON to parse\n * @param scene The scene to instantiate the camera in\n * @returns the newly constructed camera\n */\n static Parse(parsedCamera, scene) {\n const type = parsedCamera.type;\n const construct = Camera.GetConstructorFromName(type, parsedCamera.name, scene, parsedCamera.interaxial_distance, parsedCamera.isStereoscopicSideBySide);\n const camera = SerializationHelper.Parse(construct, parsedCamera, scene);\n // Parent\n if (parsedCamera.parentId !== undefined) {\n camera._waitingParentId = parsedCamera.parentId;\n }\n // Parent instance index\n if (parsedCamera.parentInstanceIndex !== undefined) {\n camera._waitingParentInstanceIndex = parsedCamera.parentInstanceIndex;\n }\n //If camera has an input manager, let it parse inputs settings\n if (camera.inputs) {\n camera.inputs.parse(parsedCamera);\n camera._setupInputs();\n }\n if (parsedCamera.upVector) {\n camera.upVector = Vector3.FromArray(parsedCamera.upVector); // need to force the upVector\n }\n if (camera.setPosition) {\n // need to force position\n camera.position.copyFromFloats(0, 0, 0);\n camera.setPosition(Vector3.FromArray(parsedCamera.position));\n }\n // Target\n if (parsedCamera.target) {\n if (camera.setTarget) {\n camera.setTarget(Vector3.FromArray(parsedCamera.target));\n }\n }\n // Apply 3d rig, when found\n if (parsedCamera.cameraRigMode) {\n const rigParams = parsedCamera.interaxial_distance ? { interaxialDistance: parsedCamera.interaxial_distance } : {};\n camera.setCameraRigMode(parsedCamera.cameraRigMode, rigParams);\n }\n // Animations\n if (parsedCamera.animations) {\n for (let animationIndex = 0; animationIndex < parsedCamera.animations.length; animationIndex++) {\n const parsedAnimation = parsedCamera.animations[animationIndex];\n const internalClass = GetClass(\"BABYLON.Animation\");\n if (internalClass) {\n camera.animations.push(internalClass.Parse(parsedAnimation));\n }\n }\n Node.ParseAnimationRanges(camera, parsedCamera, scene);\n }\n if (parsedCamera.autoAnimate) {\n scene.beginAnimation(camera, parsedCamera.autoAnimateFrom, parsedCamera.autoAnimateTo, parsedCamera.autoAnimateLoop, parsedCamera.autoAnimateSpeed || 1.0);\n }\n // Check if isEnabled is defined to be back compatible with prior serialized versions.\n if (parsedCamera.isEnabled !== undefined) {\n camera.setEnabled(parsedCamera.isEnabled);\n }\n return camera;\n }\n }\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n Camera._CreateDefaultParsedCamera = (name, scene) => {\n throw _WarnImport(\"UniversalCamera\");\n };\n /**\n * This is the default projection mode used by the cameras.\n * It helps recreating a feeling of perspective and better appreciate depth.\n * This is the best way to simulate real life cameras.\n */\n Camera.PERSPECTIVE_CAMERA = 0;\n /**\n * This helps creating camera with an orthographic mode.\n * Orthographic is commonly used in engineering as a means to produce object specifications that communicate dimensions unambiguously, each line of 1 unit length (cm, meter..whatever) will appear to have the same length everywhere on the drawing. This allows the drafter to dimension only a subset of lines and let the reader know that other lines of that length on the drawing are also that length in reality. Every parallel line in the drawing is also parallel in the object.\n */\n Camera.ORTHOGRAPHIC_CAMERA = 1;\n /**\n * This is the default FOV mode for perspective cameras.\n * This setting aligns the upper and lower bounds of the viewport to the upper and lower bounds of the camera frustum.\n */\n Camera.FOVMODE_VERTICAL_FIXED = 0;\n /**\n * This setting aligns the left and right bounds of the viewport to the left and right bounds of the camera frustum.\n */\n Camera.FOVMODE_HORIZONTAL_FIXED = 1;\n /**\n * This specifies there is no need for a camera rig.\n * Basically only one eye is rendered corresponding to the camera.\n */\n Camera.RIG_MODE_NONE = 0;\n /**\n * Simulates a camera Rig with one blue eye and one red eye.\n * This can be use with 3d blue and red glasses.\n */\n Camera.RIG_MODE_STEREOSCOPIC_ANAGLYPH = 10;\n /**\n * Defines that both eyes of the camera will be rendered side by side with a parallel target.\n */\n Camera.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL = 11;\n /**\n * Defines that both eyes of the camera will be rendered side by side with a none parallel target.\n */\n Camera.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED = 12;\n /**\n * Defines that both eyes of the camera will be rendered over under each other.\n */\n Camera.RIG_MODE_STEREOSCOPIC_OVERUNDER = 13;\n /**\n * Defines that both eyes of the camera will be rendered on successive lines interlaced for passive 3d monitors.\n */\n Camera.RIG_MODE_STEREOSCOPIC_INTERLACED = 14;\n /**\n * Defines that both eyes of the camera should be renderered in a VR mode (carbox).\n */\n Camera.RIG_MODE_VR = 20;\n /**\n * Defines that both eyes of the camera should be renderered in a VR mode (webVR).\n */\n Camera.RIG_MODE_WEBVR = 21;\n /**\n * Custom rig mode allowing rig cameras to be populated manually with any number of cameras\n */\n Camera.RIG_MODE_CUSTOM = 22;\n /**\n * Defines if by default attaching controls should prevent the default javascript event to continue.\n */\n Camera.ForceAttachControlToAlwaysPreventDefault = false;\n __decorate$1([\n serializeAsVector3(\"position\")\n ], Camera.prototype, \"_position\", void 0);\n __decorate$1([\n serializeAsVector3(\"upVector\")\n ], Camera.prototype, \"_upVector\", void 0);\n __decorate$1([\n serialize()\n ], Camera.prototype, \"orthoLeft\", null);\n __decorate$1([\n serialize()\n ], Camera.prototype, \"orthoRight\", null);\n __decorate$1([\n serialize()\n ], Camera.prototype, \"orthoBottom\", null);\n __decorate$1([\n serialize()\n ], Camera.prototype, \"orthoTop\", null);\n __decorate$1([\n serialize()\n ], Camera.prototype, \"fov\", void 0);\n __decorate$1([\n serialize()\n ], Camera.prototype, \"projectionPlaneTilt\", void 0);\n __decorate$1([\n serialize()\n ], Camera.prototype, \"minZ\", void 0);\n __decorate$1([\n serialize()\n ], Camera.prototype, \"maxZ\", void 0);\n __decorate$1([\n serialize()\n ], Camera.prototype, \"inertia\", void 0);\n __decorate$1([\n serialize()\n ], Camera.prototype, \"mode\", null);\n __decorate$1([\n serialize()\n ], Camera.prototype, \"layerMask\", void 0);\n __decorate$1([\n serialize()\n ], Camera.prototype, \"fovMode\", void 0);\n __decorate$1([\n serialize()\n ], Camera.prototype, \"cameraRigMode\", void 0);\n __decorate$1([\n serialize()\n ], Camera.prototype, \"interaxialDistance\", void 0);\n __decorate$1([\n serialize()\n ], Camera.prototype, \"isStereoscopicSideBySide\", void 0);\n\n /**\n * This class implement a typical dictionary using a string as key and the generic type T as value.\n * The underlying implementation relies on an associative array to ensure the best performances.\n * The value can be anything including 'null' but except 'undefined'\n */\n class StringDictionary {\n constructor() {\n this._count = 0;\n this._data = {};\n }\n /**\n * This will clear this dictionary and copy the content from the 'source' one.\n * If the T value is a custom object, it won't be copied/cloned, the same object will be used\n * @param source the dictionary to take the content from and copy to this dictionary\n */\n copyFrom(source) {\n this.clear();\n source.forEach((t, v) => this.add(t, v));\n }\n /**\n * Get a value based from its key\n * @param key the given key to get the matching value from\n * @returns the value if found, otherwise undefined is returned\n */\n get(key) {\n const val = this._data[key];\n if (val !== undefined) {\n return val;\n }\n return undefined;\n }\n /**\n * Get a value from its key or add it if it doesn't exist.\n * This method will ensure you that a given key/data will be present in the dictionary.\n * @param key the given key to get the matching value from\n * @param factory the factory that will create the value if the key is not present in the dictionary.\n * The factory will only be invoked if there's no data for the given key.\n * @returns the value corresponding to the key.\n */\n getOrAddWithFactory(key, factory) {\n let val = this.get(key);\n if (val !== undefined) {\n return val;\n }\n val = factory(key);\n if (val) {\n this.add(key, val);\n }\n return val;\n }\n /**\n * Get a value from its key if present in the dictionary otherwise add it\n * @param key the key to get the value from\n * @param val if there's no such key/value pair in the dictionary add it with this value\n * @returns the value corresponding to the key\n */\n getOrAdd(key, val) {\n const curVal = this.get(key);\n if (curVal !== undefined) {\n return curVal;\n }\n this.add(key, val);\n return val;\n }\n /**\n * Check if there's a given key in the dictionary\n * @param key the key to check for\n * @returns true if the key is present, false otherwise\n */\n contains(key) {\n return this._data[key] !== undefined;\n }\n /**\n * Add a new key and its corresponding value\n * @param key the key to add\n * @param value the value corresponding to the key\n * @returns true if the operation completed successfully, false if we couldn't insert the key/value because there was already this key in the dictionary\n */\n add(key, value) {\n if (this._data[key] !== undefined) {\n return false;\n }\n this._data[key] = value;\n ++this._count;\n return true;\n }\n /**\n * Update a specific value associated to a key\n * @param key defines the key to use\n * @param value defines the value to store\n * @returns true if the value was updated (or false if the key was not found)\n */\n set(key, value) {\n if (this._data[key] === undefined) {\n return false;\n }\n this._data[key] = value;\n return true;\n }\n /**\n * Get the element of the given key and remove it from the dictionary\n * @param key defines the key to search\n * @returns the value associated with the key or null if not found\n */\n getAndRemove(key) {\n const val = this.get(key);\n if (val !== undefined) {\n delete this._data[key];\n --this._count;\n return val;\n }\n return null;\n }\n /**\n * Remove a key/value from the dictionary.\n * @param key the key to remove\n * @returns true if the item was successfully deleted, false if no item with such key exist in the dictionary\n */\n remove(key) {\n if (this.contains(key)) {\n delete this._data[key];\n --this._count;\n return true;\n }\n return false;\n }\n /**\n * Clear the whole content of the dictionary\n */\n clear() {\n this._data = {};\n this._count = 0;\n }\n /**\n * Gets the current count\n */\n get count() {\n return this._count;\n }\n /**\n * Execute a callback on each key/val of the dictionary.\n * Note that you can remove any element in this dictionary in the callback implementation\n * @param callback the callback to execute on a given key/value pair\n */\n forEach(callback) {\n for (const cur in this._data) {\n const val = this._data[cur];\n callback(cur, val);\n }\n }\n /**\n * Execute a callback on every occurrence of the dictionary until it returns a valid TRes object.\n * If the callback returns null or undefined the method will iterate to the next key/value pair\n * Note that you can remove any element in this dictionary in the callback implementation\n * @param callback the callback to execute, if it return a valid T instanced object the enumeration will stop and the object will be returned\n * @returns the first item\n */\n first(callback) {\n for (const cur in this._data) {\n const val = this._data[cur];\n const res = callback(cur, val);\n if (res) {\n return res;\n }\n }\n return null;\n }\n }\n\n /**\n * Base class of the scene acting as a container for the different elements composing a scene.\n * This class is dynamically extended by the different components of the scene increasing\n * flexibility and reducing coupling\n */\n class AbstractScene {\n constructor() {\n /**\n * Gets the list of root nodes (ie. nodes with no parent)\n */\n this.rootNodes = new Array();\n /** All of the cameras added to this scene\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras\n */\n this.cameras = new Array();\n /**\n * All of the lights added to this scene\n * @see https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction\n */\n this.lights = new Array();\n /**\n * All of the (abstract) meshes added to this scene\n */\n this.meshes = new Array();\n /**\n * The list of skeletons added to the scene\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/bonesSkeletons\n */\n this.skeletons = new Array();\n /**\n * All of the particle systems added to this scene\n * @see https://doc.babylonjs.com/features/featuresDeepDive/particles/particle_system/particle_system_intro\n */\n this.particleSystems = new Array();\n /**\n * Gets a list of Animations associated with the scene\n */\n this.animations = [];\n /**\n * All of the animation groups added to this scene\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/groupAnimations\n */\n this.animationGroups = new Array();\n /**\n * All of the multi-materials added to this scene\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/multiMaterials\n */\n this.multiMaterials = new Array();\n /**\n * All of the materials added to this scene\n * In the context of a Scene, it is not supposed to be modified manually.\n * Any addition or removal should be done using the addMaterial and removeMaterial Scene methods.\n * Note also that the order of the Material within the array is not significant and might change.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/materials_introduction\n */\n this.materials = new Array();\n /**\n * The list of morph target managers added to the scene\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph\n */\n this.morphTargetManagers = new Array();\n /**\n * The list of geometries used in the scene.\n */\n this.geometries = new Array();\n /**\n * All of the transform nodes added to this scene\n * In the context of a Scene, it is not supposed to be modified manually.\n * Any addition or removal should be done using the addTransformNode and removeTransformNode Scene methods.\n * Note also that the order of the TransformNode within the array is not significant and might change.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/parent_pivot/transform_node\n */\n this.transformNodes = new Array();\n /**\n * ActionManagers available on the scene.\n * @deprecated\n */\n this.actionManagers = new Array();\n /**\n * Textures to keep.\n */\n this.textures = new Array();\n /** @internal */\n this._environmentTexture = null;\n /**\n * The list of postprocesses added to the scene\n */\n this.postProcesses = new Array();\n }\n /**\n * Adds a parser in the list of available ones\n * @param name Defines the name of the parser\n * @param parser Defines the parser to add\n */\n static AddParser(name, parser) {\n this._BabylonFileParsers[name] = parser;\n }\n /**\n * Gets a general parser from the list of available ones\n * @param name Defines the name of the parser\n * @returns the requested parser or null\n */\n static GetParser(name) {\n if (this._BabylonFileParsers[name]) {\n return this._BabylonFileParsers[name];\n }\n return null;\n }\n /**\n * Adds n individual parser in the list of available ones\n * @param name Defines the name of the parser\n * @param parser Defines the parser to add\n */\n static AddIndividualParser(name, parser) {\n this._IndividualBabylonFileParsers[name] = parser;\n }\n /**\n * Gets an individual parser from the list of available ones\n * @param name Defines the name of the parser\n * @returns the requested parser or null\n */\n static GetIndividualParser(name) {\n if (this._IndividualBabylonFileParsers[name]) {\n return this._IndividualBabylonFileParsers[name];\n }\n return null;\n }\n /**\n * Parser json data and populate both a scene and its associated container object\n * @param jsonData Defines the data to parse\n * @param scene Defines the scene to parse the data for\n * @param container Defines the container attached to the parsing sequence\n * @param rootUrl Defines the root url of the data\n */\n static Parse(jsonData, scene, container, rootUrl) {\n for (const parserName in this._BabylonFileParsers) {\n if (Object.prototype.hasOwnProperty.call(this._BabylonFileParsers, parserName)) {\n this._BabylonFileParsers[parserName](jsonData, scene, container, rootUrl);\n }\n }\n }\n /**\n * Texture used in all pbr material as the reflection texture.\n * As in the majority of the scene they are the same (exception for multi room and so on),\n * this is easier to reference from here than from all the materials.\n */\n get environmentTexture() {\n return this._environmentTexture;\n }\n set environmentTexture(value) {\n this._environmentTexture = value;\n }\n /**\n * @returns all meshes, lights, cameras, transformNodes and bones\n */\n getNodes() {\n let nodes = new Array();\n nodes = nodes.concat(this.meshes);\n nodes = nodes.concat(this.lights);\n nodes = nodes.concat(this.cameras);\n nodes = nodes.concat(this.transformNodes); // dummies\n this.skeletons.forEach((skeleton) => (nodes = nodes.concat(skeleton.bones)));\n return nodes;\n }\n }\n /**\n * Stores the list of available parsers in the application.\n */\n AbstractScene._BabylonFileParsers = {};\n /**\n * Stores the list of available individual parsers in the application.\n */\n AbstractScene._IndividualBabylonFileParsers = {};\n\n /**\n * Manages the defines for the Material\n */\n class MaterialDefines {\n /**\n * Creates a new instance\n * @param externalProperties list of external properties to inject into the object\n */\n constructor(externalProperties) {\n /** @internal */\n this._keys = [];\n this._isDirty = true;\n /** @internal */\n this._areLightsDirty = true;\n /** @internal */\n this._areLightsDisposed = false;\n /** @internal */\n this._areAttributesDirty = true;\n /** @internal */\n this._areTexturesDirty = true;\n /** @internal */\n this._areFresnelDirty = true;\n /** @internal */\n this._areMiscDirty = true;\n /** @internal */\n this._arePrePassDirty = true;\n /** @internal */\n this._areImageProcessingDirty = true;\n /** @internal */\n this._normals = false;\n /** @internal */\n this._uvs = false;\n /** @internal */\n this._needNormals = false;\n /** @internal */\n this._needUVs = false;\n this._externalProperties = externalProperties;\n // Initialize External Properties\n if (externalProperties) {\n for (const prop in externalProperties) {\n if (Object.prototype.hasOwnProperty.call(externalProperties, prop)) {\n this._setDefaultValue(prop);\n }\n }\n }\n }\n /**\n * Specifies if the material needs to be re-calculated\n */\n get isDirty() {\n return this._isDirty;\n }\n /**\n * Marks the material to indicate that it has been re-calculated\n */\n markAsProcessed() {\n this._isDirty = false;\n this._areAttributesDirty = false;\n this._areTexturesDirty = false;\n this._areFresnelDirty = false;\n this._areLightsDirty = false;\n this._areLightsDisposed = false;\n this._areMiscDirty = false;\n this._arePrePassDirty = false;\n this._areImageProcessingDirty = false;\n }\n /**\n * Marks the material to indicate that it needs to be re-calculated\n */\n markAsUnprocessed() {\n this._isDirty = true;\n }\n /**\n * Marks the material to indicate all of its defines need to be re-calculated\n */\n markAllAsDirty() {\n this._areTexturesDirty = true;\n this._areAttributesDirty = true;\n this._areLightsDirty = true;\n this._areFresnelDirty = true;\n this._areMiscDirty = true;\n this._arePrePassDirty = false;\n this._areImageProcessingDirty = true;\n this._isDirty = true;\n }\n /**\n * Marks the material to indicate that image processing needs to be re-calculated\n */\n markAsImageProcessingDirty() {\n this._areImageProcessingDirty = true;\n this._isDirty = true;\n }\n /**\n * Marks the material to indicate the lights need to be re-calculated\n * @param disposed Defines whether the light is dirty due to dispose or not\n */\n markAsLightDirty(disposed = false) {\n this._areLightsDirty = true;\n this._areLightsDisposed = this._areLightsDisposed || disposed;\n this._isDirty = true;\n }\n /**\n * Marks the attribute state as changed\n */\n markAsAttributesDirty() {\n this._areAttributesDirty = true;\n this._isDirty = true;\n }\n /**\n * Marks the texture state as changed\n */\n markAsTexturesDirty() {\n this._areTexturesDirty = true;\n this._isDirty = true;\n }\n /**\n * Marks the fresnel state as changed\n */\n markAsFresnelDirty() {\n this._areFresnelDirty = true;\n this._isDirty = true;\n }\n /**\n * Marks the misc state as changed\n */\n markAsMiscDirty() {\n this._areMiscDirty = true;\n this._isDirty = true;\n }\n /**\n * Marks the prepass state as changed\n */\n markAsPrePassDirty() {\n this._arePrePassDirty = true;\n this._isDirty = true;\n }\n /**\n * Rebuilds the material defines\n */\n rebuild() {\n this._keys.length = 0;\n for (const key of Object.keys(this)) {\n if (key[0] === \"_\") {\n continue;\n }\n this._keys.push(key);\n }\n if (this._externalProperties) {\n for (const name in this._externalProperties) {\n if (this._keys.indexOf(name) === -1) {\n this._keys.push(name);\n }\n }\n }\n }\n /**\n * Specifies if two material defines are equal\n * @param other - A material define instance to compare to\n * @returns - Boolean indicating if the material defines are equal (true) or not (false)\n */\n isEqual(other) {\n if (this._keys.length !== other._keys.length) {\n return false;\n }\n for (let index = 0; index < this._keys.length; index++) {\n const prop = this._keys[index];\n if (this[prop] !== other[prop]) {\n return false;\n }\n }\n return true;\n }\n /**\n * Clones this instance's defines to another instance\n * @param other - material defines to clone values to\n */\n cloneTo(other) {\n if (this._keys.length !== other._keys.length) {\n other._keys = this._keys.slice(0);\n }\n for (let index = 0; index < this._keys.length; index++) {\n const prop = this._keys[index];\n other[prop] = this[prop];\n }\n }\n /**\n * Resets the material define values\n */\n reset() {\n this._keys.forEach((prop) => this._setDefaultValue(prop));\n }\n _setDefaultValue(prop) {\n var _a, _b, _c, _d, _e;\n const type = (_c = (_b = (_a = this._externalProperties) === null || _a === void 0 ? void 0 : _a[prop]) === null || _b === void 0 ? void 0 : _b.type) !== null && _c !== void 0 ? _c : typeof this[prop];\n const defValue = (_e = (_d = this._externalProperties) === null || _d === void 0 ? void 0 : _d[prop]) === null || _e === void 0 ? void 0 : _e.default;\n switch (type) {\n case \"number\":\n this[prop] = defValue !== null && defValue !== void 0 ? defValue : 0;\n break;\n case \"string\":\n this[prop] = defValue !== null && defValue !== void 0 ? defValue : \"\";\n break;\n default:\n this[prop] = defValue !== null && defValue !== void 0 ? defValue : false;\n break;\n }\n }\n /**\n * Converts the material define values to a string\n * @returns - String of material define information\n */\n toString() {\n let result = \"\";\n for (let index = 0; index < this._keys.length; index++) {\n const prop = this._keys[index];\n const value = this[prop];\n const type = typeof value;\n switch (type) {\n case \"number\":\n case \"string\":\n result += \"#define \" + prop + \" \" + value + \"\\n\";\n break;\n default:\n if (value) {\n result += \"#define \" + prop + \"\\n\";\n }\n break;\n }\n }\n return result;\n }\n }\n\n /**\n * The color grading curves provide additional color adjustment that is applied after any color grading transform (3D LUT).\n * They allow basic adjustment of saturation and small exposure adjustments, along with color filter tinting to provide white balance adjustment or more stylistic effects.\n * These are similar to controls found in many professional imaging or colorist software. The global controls are applied to the entire image. For advanced tuning, extra controls are provided to adjust the shadow, midtone and highlight areas of the image;\n * corresponding to low luminance, medium luminance, and high luminance areas respectively.\n */\n class ColorCurves {\n constructor() {\n this._dirty = true;\n this._tempColor = new Color4(0, 0, 0, 0);\n this._globalCurve = new Color4(0, 0, 0, 0);\n this._highlightsCurve = new Color4(0, 0, 0, 0);\n this._midtonesCurve = new Color4(0, 0, 0, 0);\n this._shadowsCurve = new Color4(0, 0, 0, 0);\n this._positiveCurve = new Color4(0, 0, 0, 0);\n this._negativeCurve = new Color4(0, 0, 0, 0);\n this._globalHue = 30;\n this._globalDensity = 0;\n this._globalSaturation = 0;\n this._globalExposure = 0;\n this._highlightsHue = 30;\n this._highlightsDensity = 0;\n this._highlightsSaturation = 0;\n this._highlightsExposure = 0;\n this._midtonesHue = 30;\n this._midtonesDensity = 0;\n this._midtonesSaturation = 0;\n this._midtonesExposure = 0;\n this._shadowsHue = 30;\n this._shadowsDensity = 0;\n this._shadowsSaturation = 0;\n this._shadowsExposure = 0;\n }\n /**\n * Gets the global Hue value.\n * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange).\n */\n get globalHue() {\n return this._globalHue;\n }\n /**\n * Sets the global Hue value.\n * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange).\n */\n set globalHue(value) {\n this._globalHue = value;\n this._dirty = true;\n }\n /**\n * Gets the global Density value.\n * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect.\n * Values less than zero provide a filter of opposite hue.\n */\n get globalDensity() {\n return this._globalDensity;\n }\n /**\n * Sets the global Density value.\n * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect.\n * Values less than zero provide a filter of opposite hue.\n */\n set globalDensity(value) {\n this._globalDensity = value;\n this._dirty = true;\n }\n /**\n * Gets the global Saturation value.\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation.\n */\n get globalSaturation() {\n return this._globalSaturation;\n }\n /**\n * Sets the global Saturation value.\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation.\n */\n set globalSaturation(value) {\n this._globalSaturation = value;\n this._dirty = true;\n }\n /**\n * Gets the global Exposure value.\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure.\n */\n get globalExposure() {\n return this._globalExposure;\n }\n /**\n * Sets the global Exposure value.\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure.\n */\n set globalExposure(value) {\n this._globalExposure = value;\n this._dirty = true;\n }\n /**\n * Gets the highlights Hue value.\n * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange).\n */\n get highlightsHue() {\n return this._highlightsHue;\n }\n /**\n * Sets the highlights Hue value.\n * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange).\n */\n set highlightsHue(value) {\n this._highlightsHue = value;\n this._dirty = true;\n }\n /**\n * Gets the highlights Density value.\n * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect.\n * Values less than zero provide a filter of opposite hue.\n */\n get highlightsDensity() {\n return this._highlightsDensity;\n }\n /**\n * Sets the highlights Density value.\n * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect.\n * Values less than zero provide a filter of opposite hue.\n */\n set highlightsDensity(value) {\n this._highlightsDensity = value;\n this._dirty = true;\n }\n /**\n * Gets the highlights Saturation value.\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation.\n */\n get highlightsSaturation() {\n return this._highlightsSaturation;\n }\n /**\n * Sets the highlights Saturation value.\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation.\n */\n set highlightsSaturation(value) {\n this._highlightsSaturation = value;\n this._dirty = true;\n }\n /**\n * Gets the highlights Exposure value.\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure.\n */\n get highlightsExposure() {\n return this._highlightsExposure;\n }\n /**\n * Sets the highlights Exposure value.\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure.\n */\n set highlightsExposure(value) {\n this._highlightsExposure = value;\n this._dirty = true;\n }\n /**\n * Gets the midtones Hue value.\n * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange).\n */\n get midtonesHue() {\n return this._midtonesHue;\n }\n /**\n * Sets the midtones Hue value.\n * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange).\n */\n set midtonesHue(value) {\n this._midtonesHue = value;\n this._dirty = true;\n }\n /**\n * Gets the midtones Density value.\n * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect.\n * Values less than zero provide a filter of opposite hue.\n */\n get midtonesDensity() {\n return this._midtonesDensity;\n }\n /**\n * Sets the midtones Density value.\n * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect.\n * Values less than zero provide a filter of opposite hue.\n */\n set midtonesDensity(value) {\n this._midtonesDensity = value;\n this._dirty = true;\n }\n /**\n * Gets the midtones Saturation value.\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation.\n */\n get midtonesSaturation() {\n return this._midtonesSaturation;\n }\n /**\n * Sets the midtones Saturation value.\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation.\n */\n set midtonesSaturation(value) {\n this._midtonesSaturation = value;\n this._dirty = true;\n }\n /**\n * Gets the midtones Exposure value.\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure.\n */\n get midtonesExposure() {\n return this._midtonesExposure;\n }\n /**\n * Sets the midtones Exposure value.\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure.\n */\n set midtonesExposure(value) {\n this._midtonesExposure = value;\n this._dirty = true;\n }\n /**\n * Gets the shadows Hue value.\n * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange).\n */\n get shadowsHue() {\n return this._shadowsHue;\n }\n /**\n * Sets the shadows Hue value.\n * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange).\n */\n set shadowsHue(value) {\n this._shadowsHue = value;\n this._dirty = true;\n }\n /**\n * Gets the shadows Density value.\n * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect.\n * Values less than zero provide a filter of opposite hue.\n */\n get shadowsDensity() {\n return this._shadowsDensity;\n }\n /**\n * Sets the shadows Density value.\n * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect.\n * Values less than zero provide a filter of opposite hue.\n */\n set shadowsDensity(value) {\n this._shadowsDensity = value;\n this._dirty = true;\n }\n /**\n * Gets the shadows Saturation value.\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation.\n */\n get shadowsSaturation() {\n return this._shadowsSaturation;\n }\n /**\n * Sets the shadows Saturation value.\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation.\n */\n set shadowsSaturation(value) {\n this._shadowsSaturation = value;\n this._dirty = true;\n }\n /**\n * Gets the shadows Exposure value.\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure.\n */\n get shadowsExposure() {\n return this._shadowsExposure;\n }\n /**\n * Sets the shadows Exposure value.\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure.\n */\n set shadowsExposure(value) {\n this._shadowsExposure = value;\n this._dirty = true;\n }\n /**\n * Returns the class name\n * @returns The class name\n */\n getClassName() {\n return \"ColorCurves\";\n }\n /**\n * Binds the color curves to the shader.\n * @param colorCurves The color curve to bind\n * @param effect The effect to bind to\n * @param positiveUniform The positive uniform shader parameter\n * @param neutralUniform The neutral uniform shader parameter\n * @param negativeUniform The negative uniform shader parameter\n */\n static Bind(colorCurves, effect, positiveUniform = \"vCameraColorCurvePositive\", neutralUniform = \"vCameraColorCurveNeutral\", negativeUniform = \"vCameraColorCurveNegative\") {\n if (colorCurves._dirty) {\n colorCurves._dirty = false;\n // Fill in global info.\n colorCurves._getColorGradingDataToRef(colorCurves._globalHue, colorCurves._globalDensity, colorCurves._globalSaturation, colorCurves._globalExposure, colorCurves._globalCurve);\n // Compute highlights info.\n colorCurves._getColorGradingDataToRef(colorCurves._highlightsHue, colorCurves._highlightsDensity, colorCurves._highlightsSaturation, colorCurves._highlightsExposure, colorCurves._tempColor);\n colorCurves._tempColor.multiplyToRef(colorCurves._globalCurve, colorCurves._highlightsCurve);\n // Compute midtones info.\n colorCurves._getColorGradingDataToRef(colorCurves._midtonesHue, colorCurves._midtonesDensity, colorCurves._midtonesSaturation, colorCurves._midtonesExposure, colorCurves._tempColor);\n colorCurves._tempColor.multiplyToRef(colorCurves._globalCurve, colorCurves._midtonesCurve);\n // Compute shadows info.\n colorCurves._getColorGradingDataToRef(colorCurves._shadowsHue, colorCurves._shadowsDensity, colorCurves._shadowsSaturation, colorCurves._shadowsExposure, colorCurves._tempColor);\n colorCurves._tempColor.multiplyToRef(colorCurves._globalCurve, colorCurves._shadowsCurve);\n // Compute deltas (neutral is midtones).\n colorCurves._highlightsCurve.subtractToRef(colorCurves._midtonesCurve, colorCurves._positiveCurve);\n colorCurves._midtonesCurve.subtractToRef(colorCurves._shadowsCurve, colorCurves._negativeCurve);\n }\n if (effect) {\n effect.setFloat4(positiveUniform, colorCurves._positiveCurve.r, colorCurves._positiveCurve.g, colorCurves._positiveCurve.b, colorCurves._positiveCurve.a);\n effect.setFloat4(neutralUniform, colorCurves._midtonesCurve.r, colorCurves._midtonesCurve.g, colorCurves._midtonesCurve.b, colorCurves._midtonesCurve.a);\n effect.setFloat4(negativeUniform, colorCurves._negativeCurve.r, colorCurves._negativeCurve.g, colorCurves._negativeCurve.b, colorCurves._negativeCurve.a);\n }\n }\n /**\n * Prepare the list of uniforms associated with the ColorCurves effects.\n * @param uniformsList The list of uniforms used in the effect\n */\n static PrepareUniforms(uniformsList) {\n uniformsList.push(\"vCameraColorCurveNeutral\", \"vCameraColorCurvePositive\", \"vCameraColorCurveNegative\");\n }\n /**\n * Returns color grading data based on a hue, density, saturation and exposure value.\n * @param hue\n * @param density\n * @param saturation The saturation.\n * @param exposure The exposure.\n * @param result The result data container.\n */\n _getColorGradingDataToRef(hue, density, saturation, exposure, result) {\n if (hue == null) {\n return;\n }\n hue = ColorCurves._Clamp(hue, 0, 360);\n density = ColorCurves._Clamp(density, -100, 100);\n saturation = ColorCurves._Clamp(saturation, -100, 100);\n exposure = ColorCurves._Clamp(exposure, -100, 100);\n // Remap the slider/config filter density with non-linear mapping and also scale by half\n // so that the maximum filter density is only 50% control. This provides fine control\n // for small values and reasonable range.\n density = ColorCurves._ApplyColorGradingSliderNonlinear(density);\n density *= 0.5;\n exposure = ColorCurves._ApplyColorGradingSliderNonlinear(exposure);\n if (density < 0) {\n density *= -1;\n hue = (hue + 180) % 360;\n }\n ColorCurves._FromHSBToRef(hue, density, 50 + 0.25 * exposure, result);\n result.scaleToRef(2, result);\n result.a = 1 + 0.01 * saturation;\n }\n /**\n * Takes an input slider value and returns an adjusted value that provides extra control near the centre.\n * @param value The input slider value in range [-100,100].\n * @returns Adjusted value.\n */\n static _ApplyColorGradingSliderNonlinear(value) {\n value /= 100;\n let x = Math.abs(value);\n x = Math.pow(x, 2);\n if (value < 0) {\n x *= -1;\n }\n x *= 100;\n return x;\n }\n /**\n * Returns an RGBA Color4 based on Hue, Saturation and Brightness (also referred to as value, HSV).\n * @param hue The hue (H) input.\n * @param saturation The saturation (S) input.\n * @param brightness The brightness (B) input.\n * @param result\n * @result An RGBA color represented as Vector4.\n */\n static _FromHSBToRef(hue, saturation, brightness, result) {\n let h = ColorCurves._Clamp(hue, 0, 360);\n const s = ColorCurves._Clamp(saturation / 100, 0, 1);\n const v = ColorCurves._Clamp(brightness / 100, 0, 1);\n if (s === 0) {\n result.r = v;\n result.g = v;\n result.b = v;\n }\n else {\n // sector 0 to 5\n h /= 60;\n const i = Math.floor(h);\n // fractional part of h\n const f = h - i;\n const p = v * (1 - s);\n const q = v * (1 - s * f);\n const t = v * (1 - s * (1 - f));\n switch (i) {\n case 0:\n result.r = v;\n result.g = t;\n result.b = p;\n break;\n case 1:\n result.r = q;\n result.g = v;\n result.b = p;\n break;\n case 2:\n result.r = p;\n result.g = v;\n result.b = t;\n break;\n case 3:\n result.r = p;\n result.g = q;\n result.b = v;\n break;\n case 4:\n result.r = t;\n result.g = p;\n result.b = v;\n break;\n default:\n // case 5:\n result.r = v;\n result.g = p;\n result.b = q;\n break;\n }\n }\n result.a = 1;\n }\n /**\n * Returns a value clamped between min and max\n * @param value The value to clamp\n * @param min The minimum of value\n * @param max The maximum of value\n * @returns The clamped value.\n */\n static _Clamp(value, min, max) {\n return Math.min(Math.max(value, min), max);\n }\n /**\n * Clones the current color curve instance.\n * @returns The cloned curves\n */\n clone() {\n return SerializationHelper.Clone(() => new ColorCurves(), this);\n }\n /**\n * Serializes the current color curve instance to a json representation.\n * @returns a JSON representation\n */\n serialize() {\n return SerializationHelper.Serialize(this);\n }\n /**\n * Parses the color curve from a json representation.\n * @param source the JSON source to parse\n * @returns The parsed curves\n */\n static Parse(source) {\n return SerializationHelper.Parse(() => new ColorCurves(), source, null, null);\n }\n }\n __decorate$1([\n serialize()\n ], ColorCurves.prototype, \"_globalHue\", void 0);\n __decorate$1([\n serialize()\n ], ColorCurves.prototype, \"_globalDensity\", void 0);\n __decorate$1([\n serialize()\n ], ColorCurves.prototype, \"_globalSaturation\", void 0);\n __decorate$1([\n serialize()\n ], ColorCurves.prototype, \"_globalExposure\", void 0);\n __decorate$1([\n serialize()\n ], ColorCurves.prototype, \"_highlightsHue\", void 0);\n __decorate$1([\n serialize()\n ], ColorCurves.prototype, \"_highlightsDensity\", void 0);\n __decorate$1([\n serialize()\n ], ColorCurves.prototype, \"_highlightsSaturation\", void 0);\n __decorate$1([\n serialize()\n ], ColorCurves.prototype, \"_highlightsExposure\", void 0);\n __decorate$1([\n serialize()\n ], ColorCurves.prototype, \"_midtonesHue\", void 0);\n __decorate$1([\n serialize()\n ], ColorCurves.prototype, \"_midtonesDensity\", void 0);\n __decorate$1([\n serialize()\n ], ColorCurves.prototype, \"_midtonesSaturation\", void 0);\n __decorate$1([\n serialize()\n ], ColorCurves.prototype, \"_midtonesExposure\", void 0);\n // References the dependencies.\n SerializationHelper._ColorCurvesParser = ColorCurves.Parse;\n\n /**\n * @internal\n */\n class ImageProcessingConfigurationDefines extends MaterialDefines {\n constructor() {\n super();\n this.IMAGEPROCESSING = false;\n this.VIGNETTE = false;\n this.VIGNETTEBLENDMODEMULTIPLY = false;\n this.VIGNETTEBLENDMODEOPAQUE = false;\n this.TONEMAPPING = false;\n this.TONEMAPPING_ACES = false;\n this.CONTRAST = false;\n this.COLORCURVES = false;\n this.COLORGRADING = false;\n this.COLORGRADING3D = false;\n this.SAMPLER3DGREENDEPTH = false;\n this.SAMPLER3DBGRMAP = false;\n this.DITHER = false;\n this.IMAGEPROCESSINGPOSTPROCESS = false;\n this.EXPOSURE = false;\n this.SKIPFINALCOLORCLAMP = false;\n this.rebuild();\n }\n }\n /**\n * This groups together the common properties used for image processing either in direct forward pass\n * or through post processing effect depending on the use of the image processing pipeline in your scene\n * or not.\n */\n class ImageProcessingConfiguration {\n constructor() {\n /**\n * Color curves setup used in the effect if colorCurvesEnabled is set to true\n */\n this.colorCurves = new ColorCurves();\n this._colorCurvesEnabled = false;\n this._colorGradingEnabled = false;\n this._colorGradingWithGreenDepth = true;\n this._colorGradingBGR = true;\n /** @internal */\n this._exposure = 1.0;\n this._toneMappingEnabled = false;\n this._toneMappingType = ImageProcessingConfiguration.TONEMAPPING_STANDARD;\n this._contrast = 1.0;\n /**\n * Vignette stretch size.\n */\n this.vignetteStretch = 0;\n /**\n * Vignette center X Offset.\n */\n this.vignetteCenterX = 0;\n /**\n * Vignette center Y Offset.\n */\n this.vignetteCenterY = 0;\n /**\n * Vignette weight or intensity of the vignette effect.\n */\n this.vignetteWeight = 1.5;\n /**\n * Color of the vignette applied on the screen through the chosen blend mode (vignetteBlendMode)\n * if vignetteEnabled is set to true.\n */\n this.vignetteColor = new Color4(0, 0, 0, 0);\n /**\n * Camera field of view used by the Vignette effect.\n */\n this.vignetteCameraFov = 0.5;\n this._vignetteBlendMode = ImageProcessingConfiguration.VIGNETTEMODE_MULTIPLY;\n this._vignetteEnabled = false;\n this._ditheringEnabled = false;\n this._ditheringIntensity = 1.0 / 255.0;\n /** @internal */\n this._skipFinalColorClamp = false;\n /** @internal */\n this._applyByPostProcess = false;\n this._isEnabled = true;\n /**\n * An event triggered when the configuration changes and requires Shader to Update some parameters.\n */\n this.onUpdateParameters = new Observable$1();\n }\n /**\n * Gets whether the color curves effect is enabled.\n */\n get colorCurvesEnabled() {\n return this._colorCurvesEnabled;\n }\n /**\n * Sets whether the color curves effect is enabled.\n */\n set colorCurvesEnabled(value) {\n if (this._colorCurvesEnabled === value) {\n return;\n }\n this._colorCurvesEnabled = value;\n this._updateParameters();\n }\n /**\n * Color grading LUT texture used in the effect if colorGradingEnabled is set to true\n */\n get colorGradingTexture() {\n return this._colorGradingTexture;\n }\n /**\n * Color grading LUT texture used in the effect if colorGradingEnabled is set to true\n */\n set colorGradingTexture(value) {\n if (this._colorGradingTexture === value) {\n return;\n }\n this._colorGradingTexture = value;\n this._updateParameters();\n }\n /**\n * Gets whether the color grading effect is enabled.\n */\n get colorGradingEnabled() {\n return this._colorGradingEnabled;\n }\n /**\n * Sets whether the color grading effect is enabled.\n */\n set colorGradingEnabled(value) {\n if (this._colorGradingEnabled === value) {\n return;\n }\n this._colorGradingEnabled = value;\n this._updateParameters();\n }\n /**\n * Gets whether the color grading effect is using a green depth for the 3d Texture.\n */\n get colorGradingWithGreenDepth() {\n return this._colorGradingWithGreenDepth;\n }\n /**\n * Sets whether the color grading effect is using a green depth for the 3d Texture.\n */\n set colorGradingWithGreenDepth(value) {\n if (this._colorGradingWithGreenDepth === value) {\n return;\n }\n this._colorGradingWithGreenDepth = value;\n this._updateParameters();\n }\n /**\n * Gets whether the color grading texture contains BGR values.\n */\n get colorGradingBGR() {\n return this._colorGradingBGR;\n }\n /**\n * Sets whether the color grading texture contains BGR values.\n */\n set colorGradingBGR(value) {\n if (this._colorGradingBGR === value) {\n return;\n }\n this._colorGradingBGR = value;\n this._updateParameters();\n }\n /**\n * Gets the Exposure used in the effect.\n */\n get exposure() {\n return this._exposure;\n }\n /**\n * Sets the Exposure used in the effect.\n */\n set exposure(value) {\n if (this._exposure === value) {\n return;\n }\n this._exposure = value;\n this._updateParameters();\n }\n /**\n * Gets whether the tone mapping effect is enabled.\n */\n get toneMappingEnabled() {\n return this._toneMappingEnabled;\n }\n /**\n * Sets whether the tone mapping effect is enabled.\n */\n set toneMappingEnabled(value) {\n if (this._toneMappingEnabled === value) {\n return;\n }\n this._toneMappingEnabled = value;\n this._updateParameters();\n }\n /**\n * Gets the type of tone mapping effect.\n */\n get toneMappingType() {\n return this._toneMappingType;\n }\n /**\n * Sets the type of tone mapping effect used in BabylonJS.\n */\n set toneMappingType(value) {\n if (this._toneMappingType === value) {\n return;\n }\n this._toneMappingType = value;\n this._updateParameters();\n }\n /**\n * Gets the contrast used in the effect.\n */\n get contrast() {\n return this._contrast;\n }\n /**\n * Sets the contrast used in the effect.\n */\n set contrast(value) {\n if (this._contrast === value) {\n return;\n }\n this._contrast = value;\n this._updateParameters();\n }\n /**\n * Back Compat: Vignette center Y Offset.\n * @deprecated use vignetteCenterY instead\n */\n get vignetteCentreY() {\n return this.vignetteCenterY;\n }\n set vignetteCentreY(value) {\n this.vignetteCenterY = value;\n }\n /**\n * Back Compat: Vignette center X Offset.\n * @deprecated use vignetteCenterX instead\n */\n get vignetteCentreX() {\n return this.vignetteCenterX;\n }\n set vignetteCentreX(value) {\n this.vignetteCenterX = value;\n }\n /**\n * Gets the vignette blend mode allowing different kind of effect.\n */\n get vignetteBlendMode() {\n return this._vignetteBlendMode;\n }\n /**\n * Sets the vignette blend mode allowing different kind of effect.\n */\n set vignetteBlendMode(value) {\n if (this._vignetteBlendMode === value) {\n return;\n }\n this._vignetteBlendMode = value;\n this._updateParameters();\n }\n /**\n * Gets whether the vignette effect is enabled.\n */\n get vignetteEnabled() {\n return this._vignetteEnabled;\n }\n /**\n * Sets whether the vignette effect is enabled.\n */\n set vignetteEnabled(value) {\n if (this._vignetteEnabled === value) {\n return;\n }\n this._vignetteEnabled = value;\n this._updateParameters();\n }\n /**\n * Gets whether the dithering effect is enabled.\n * The dithering effect can be used to reduce banding.\n */\n get ditheringEnabled() {\n return this._ditheringEnabled;\n }\n /**\n * Sets whether the dithering effect is enabled.\n * The dithering effect can be used to reduce banding.\n */\n set ditheringEnabled(value) {\n if (this._ditheringEnabled === value) {\n return;\n }\n this._ditheringEnabled = value;\n this._updateParameters();\n }\n /**\n * Gets the dithering intensity. 0 is no dithering. Default is 1.0 / 255.0.\n */\n get ditheringIntensity() {\n return this._ditheringIntensity;\n }\n /**\n * Sets the dithering intensity. 0 is no dithering. Default is 1.0 / 255.0.\n */\n set ditheringIntensity(value) {\n if (this._ditheringIntensity === value) {\n return;\n }\n this._ditheringIntensity = value;\n this._updateParameters();\n }\n /**\n * If apply by post process is set to true, setting this to true will skip the the final color clamp step in the fragment shader\n * Applies to PBR materials.\n */\n get skipFinalColorClamp() {\n return this._skipFinalColorClamp;\n }\n /**\n * If apply by post process is set to true, setting this to true will skip the the final color clamp step in the fragment shader\n * Applies to PBR materials.\n */\n set skipFinalColorClamp(value) {\n if (this._skipFinalColorClamp === value) {\n return;\n }\n this._skipFinalColorClamp = value;\n this._updateParameters();\n }\n /**\n * Gets whether the image processing is applied through a post process or not.\n */\n get applyByPostProcess() {\n return this._applyByPostProcess;\n }\n /**\n * Sets whether the image processing is applied through a post process or not.\n */\n set applyByPostProcess(value) {\n if (this._applyByPostProcess === value) {\n return;\n }\n this._applyByPostProcess = value;\n this._updateParameters();\n }\n /**\n * Gets whether the image processing is enabled or not.\n */\n get isEnabled() {\n return this._isEnabled;\n }\n /**\n * Sets whether the image processing is enabled or not.\n */\n set isEnabled(value) {\n if (this._isEnabled === value) {\n return;\n }\n this._isEnabled = value;\n this._updateParameters();\n }\n /**\n * Method called each time the image processing information changes requires to recompile the effect.\n */\n _updateParameters() {\n this.onUpdateParameters.notifyObservers(this);\n }\n /**\n * Gets the current class name.\n * @returns \"ImageProcessingConfiguration\"\n */\n getClassName() {\n return \"ImageProcessingConfiguration\";\n }\n /**\n * Prepare the list of uniforms associated with the Image Processing effects.\n * @param uniforms The list of uniforms used in the effect\n * @param defines the list of defines currently in use\n */\n static PrepareUniforms(uniforms, defines) {\n if (defines.EXPOSURE) {\n uniforms.push(\"exposureLinear\");\n }\n if (defines.CONTRAST) {\n uniforms.push(\"contrast\");\n }\n if (defines.COLORGRADING) {\n uniforms.push(\"colorTransformSettings\");\n }\n if (defines.VIGNETTE || defines.DITHER) {\n uniforms.push(\"vInverseScreenSize\");\n }\n if (defines.VIGNETTE) {\n uniforms.push(\"vignetteSettings1\");\n uniforms.push(\"vignetteSettings2\");\n }\n if (defines.COLORCURVES) {\n ColorCurves.PrepareUniforms(uniforms);\n }\n if (defines.DITHER) {\n uniforms.push(\"ditherIntensity\");\n }\n }\n /**\n * Prepare the list of samplers associated with the Image Processing effects.\n * @param samplersList The list of uniforms used in the effect\n * @param defines the list of defines currently in use\n */\n static PrepareSamplers(samplersList, defines) {\n if (defines.COLORGRADING) {\n samplersList.push(\"txColorTransform\");\n }\n }\n /**\n * Prepare the list of defines associated to the shader.\n * @param defines the list of defines to complete\n * @param forPostProcess Define if we are currently in post process mode or not\n */\n prepareDefines(defines, forPostProcess = false) {\n if (forPostProcess !== this.applyByPostProcess || !this._isEnabled) {\n defines.VIGNETTE = false;\n defines.TONEMAPPING = false;\n defines.TONEMAPPING_ACES = false;\n defines.CONTRAST = false;\n defines.EXPOSURE = false;\n defines.COLORCURVES = false;\n defines.COLORGRADING = false;\n defines.COLORGRADING3D = false;\n defines.DITHER = false;\n defines.IMAGEPROCESSING = false;\n defines.SKIPFINALCOLORCLAMP = this.skipFinalColorClamp;\n defines.IMAGEPROCESSINGPOSTPROCESS = this.applyByPostProcess && this._isEnabled;\n return;\n }\n defines.VIGNETTE = this.vignetteEnabled;\n defines.VIGNETTEBLENDMODEMULTIPLY = this.vignetteBlendMode === ImageProcessingConfiguration._VIGNETTEMODE_MULTIPLY;\n defines.VIGNETTEBLENDMODEOPAQUE = !defines.VIGNETTEBLENDMODEMULTIPLY;\n defines.TONEMAPPING = this.toneMappingEnabled;\n switch (this._toneMappingType) {\n case ImageProcessingConfiguration.TONEMAPPING_ACES:\n defines.TONEMAPPING_ACES = true;\n break;\n default:\n defines.TONEMAPPING_ACES = false;\n break;\n }\n defines.CONTRAST = this.contrast !== 1.0;\n defines.EXPOSURE = this.exposure !== 1.0;\n defines.COLORCURVES = this.colorCurvesEnabled && !!this.colorCurves;\n defines.COLORGRADING = this.colorGradingEnabled && !!this.colorGradingTexture;\n if (defines.COLORGRADING) {\n defines.COLORGRADING3D = this.colorGradingTexture.is3D;\n }\n else {\n defines.COLORGRADING3D = false;\n }\n defines.SAMPLER3DGREENDEPTH = this.colorGradingWithGreenDepth;\n defines.SAMPLER3DBGRMAP = this.colorGradingBGR;\n defines.DITHER = this._ditheringEnabled;\n defines.IMAGEPROCESSINGPOSTPROCESS = this.applyByPostProcess;\n defines.SKIPFINALCOLORCLAMP = this.skipFinalColorClamp;\n defines.IMAGEPROCESSING = defines.VIGNETTE || defines.TONEMAPPING || defines.CONTRAST || defines.EXPOSURE || defines.COLORCURVES || defines.COLORGRADING || defines.DITHER;\n }\n /**\n * Returns true if all the image processing information are ready.\n * @returns True if ready, otherwise, false\n */\n isReady() {\n // Color Grading texture can not be none blocking.\n return !this.colorGradingEnabled || !this.colorGradingTexture || this.colorGradingTexture.isReady();\n }\n /**\n * Binds the image processing to the shader.\n * @param effect The effect to bind to\n * @param overrideAspectRatio Override the aspect ratio of the effect\n */\n bind(effect, overrideAspectRatio) {\n // Color Curves\n if (this._colorCurvesEnabled && this.colorCurves) {\n ColorCurves.Bind(this.colorCurves, effect);\n }\n // Vignette and dither handled together due to common uniform.\n if (this._vignetteEnabled || this._ditheringEnabled) {\n const inverseWidth = 1 / effect.getEngine().getRenderWidth();\n const inverseHeight = 1 / effect.getEngine().getRenderHeight();\n effect.setFloat2(\"vInverseScreenSize\", inverseWidth, inverseHeight);\n if (this._ditheringEnabled) {\n effect.setFloat(\"ditherIntensity\", 0.5 * this._ditheringIntensity);\n }\n if (this._vignetteEnabled) {\n const aspectRatio = overrideAspectRatio != null ? overrideAspectRatio : inverseHeight / inverseWidth;\n let vignetteScaleY = Math.tan(this.vignetteCameraFov * 0.5);\n let vignetteScaleX = vignetteScaleY * aspectRatio;\n const vignetteScaleGeometricMean = Math.sqrt(vignetteScaleX * vignetteScaleY);\n vignetteScaleX = Tools.Mix(vignetteScaleX, vignetteScaleGeometricMean, this.vignetteStretch);\n vignetteScaleY = Tools.Mix(vignetteScaleY, vignetteScaleGeometricMean, this.vignetteStretch);\n effect.setFloat4(\"vignetteSettings1\", vignetteScaleX, vignetteScaleY, -vignetteScaleX * this.vignetteCenterX, -vignetteScaleY * this.vignetteCenterY);\n const vignettePower = -2.0 * this.vignetteWeight;\n effect.setFloat4(\"vignetteSettings2\", this.vignetteColor.r, this.vignetteColor.g, this.vignetteColor.b, vignettePower);\n }\n }\n // Exposure\n effect.setFloat(\"exposureLinear\", this.exposure);\n // Contrast\n effect.setFloat(\"contrast\", this.contrast);\n // Color transform settings\n if (this.colorGradingTexture) {\n effect.setTexture(\"txColorTransform\", this.colorGradingTexture);\n const textureSize = this.colorGradingTexture.getSize().height;\n effect.setFloat4(\"colorTransformSettings\", (textureSize - 1) / textureSize, // textureScale\n 0.5 / textureSize, // textureOffset\n textureSize, // textureSize\n this.colorGradingTexture.level // weight\n );\n }\n }\n /**\n * Clones the current image processing instance.\n * @returns The cloned image processing\n */\n clone() {\n return SerializationHelper.Clone(() => new ImageProcessingConfiguration(), this);\n }\n /**\n * Serializes the current image processing instance to a json representation.\n * @returns a JSON representation\n */\n serialize() {\n return SerializationHelper.Serialize(this);\n }\n /**\n * Parses the image processing from a json representation.\n * @param source the JSON source to parse\n * @returns The parsed image processing\n */\n static Parse(source) {\n const parsed = SerializationHelper.Parse(() => new ImageProcessingConfiguration(), source, null, null);\n // Backward compatibility\n if (source.vignetteCentreX !== undefined) {\n parsed.vignetteCenterX = source.vignetteCentreX;\n }\n if (source.vignetteCentreY !== undefined) {\n parsed.vignetteCenterY = source.vignetteCentreY;\n }\n return parsed;\n }\n /**\n * Used to apply the vignette as a mix with the pixel color.\n */\n static get VIGNETTEMODE_MULTIPLY() {\n return this._VIGNETTEMODE_MULTIPLY;\n }\n /**\n * Used to apply the vignette as a replacement of the pixel color.\n */\n static get VIGNETTEMODE_OPAQUE() {\n return this._VIGNETTEMODE_OPAQUE;\n }\n }\n /**\n * Default tone mapping applied in BabylonJS.\n */\n ImageProcessingConfiguration.TONEMAPPING_STANDARD = 0;\n /**\n * ACES Tone mapping (used by default in unreal and unity). This can help getting closer\n * to other engines rendering to increase portability.\n */\n ImageProcessingConfiguration.TONEMAPPING_ACES = 1;\n // Static constants associated to the image processing.\n ImageProcessingConfiguration._VIGNETTEMODE_MULTIPLY = 0;\n ImageProcessingConfiguration._VIGNETTEMODE_OPAQUE = 1;\n __decorate$1([\n serializeAsColorCurves()\n ], ImageProcessingConfiguration.prototype, \"colorCurves\", void 0);\n __decorate$1([\n serialize()\n ], ImageProcessingConfiguration.prototype, \"_colorCurvesEnabled\", void 0);\n __decorate$1([\n serializeAsTexture(\"colorGradingTexture\")\n ], ImageProcessingConfiguration.prototype, \"_colorGradingTexture\", void 0);\n __decorate$1([\n serialize()\n ], ImageProcessingConfiguration.prototype, \"_colorGradingEnabled\", void 0);\n __decorate$1([\n serialize()\n ], ImageProcessingConfiguration.prototype, \"_colorGradingWithGreenDepth\", void 0);\n __decorate$1([\n serialize()\n ], ImageProcessingConfiguration.prototype, \"_colorGradingBGR\", void 0);\n __decorate$1([\n serialize()\n ], ImageProcessingConfiguration.prototype, \"_exposure\", void 0);\n __decorate$1([\n serialize()\n ], ImageProcessingConfiguration.prototype, \"_toneMappingEnabled\", void 0);\n __decorate$1([\n serialize()\n ], ImageProcessingConfiguration.prototype, \"_toneMappingType\", void 0);\n __decorate$1([\n serialize()\n ], ImageProcessingConfiguration.prototype, \"_contrast\", void 0);\n __decorate$1([\n serialize()\n ], ImageProcessingConfiguration.prototype, \"vignetteStretch\", void 0);\n __decorate$1([\n serialize()\n ], ImageProcessingConfiguration.prototype, \"vignetteCenterX\", void 0);\n __decorate$1([\n serialize()\n ], ImageProcessingConfiguration.prototype, \"vignetteCenterY\", void 0);\n __decorate$1([\n serialize()\n ], ImageProcessingConfiguration.prototype, \"vignetteWeight\", void 0);\n __decorate$1([\n serializeAsColor4()\n ], ImageProcessingConfiguration.prototype, \"vignetteColor\", void 0);\n __decorate$1([\n serialize()\n ], ImageProcessingConfiguration.prototype, \"vignetteCameraFov\", void 0);\n __decorate$1([\n serialize()\n ], ImageProcessingConfiguration.prototype, \"_vignetteBlendMode\", void 0);\n __decorate$1([\n serialize()\n ], ImageProcessingConfiguration.prototype, \"_vignetteEnabled\", void 0);\n __decorate$1([\n serialize()\n ], ImageProcessingConfiguration.prototype, \"_ditheringEnabled\", void 0);\n __decorate$1([\n serialize()\n ], ImageProcessingConfiguration.prototype, \"_ditheringIntensity\", void 0);\n __decorate$1([\n serialize()\n ], ImageProcessingConfiguration.prototype, \"_skipFinalColorClamp\", void 0);\n __decorate$1([\n serialize()\n ], ImageProcessingConfiguration.prototype, \"_applyByPostProcess\", void 0);\n __decorate$1([\n serialize()\n ], ImageProcessingConfiguration.prototype, \"_isEnabled\", void 0);\n // References the dependencies.\n SerializationHelper._ImageProcessingConfigurationParser = ImageProcessingConfiguration.Parse;\n\n ThinEngine.prototype.createUniformBuffer = function (elements) {\n const ubo = this._gl.createBuffer();\n if (!ubo) {\n throw new Error(\"Unable to create uniform buffer\");\n }\n const result = new WebGLDataBuffer(ubo);\n this.bindUniformBuffer(result);\n if (elements instanceof Float32Array) {\n this._gl.bufferData(this._gl.UNIFORM_BUFFER, elements, this._gl.STATIC_DRAW);\n }\n else {\n this._gl.bufferData(this._gl.UNIFORM_BUFFER, new Float32Array(elements), this._gl.STATIC_DRAW);\n }\n this.bindUniformBuffer(null);\n result.references = 1;\n return result;\n };\n ThinEngine.prototype.createDynamicUniformBuffer = function (elements) {\n const ubo = this._gl.createBuffer();\n if (!ubo) {\n throw new Error(\"Unable to create dynamic uniform buffer\");\n }\n const result = new WebGLDataBuffer(ubo);\n this.bindUniformBuffer(result);\n if (elements instanceof Float32Array) {\n this._gl.bufferData(this._gl.UNIFORM_BUFFER, elements, this._gl.DYNAMIC_DRAW);\n }\n else {\n this._gl.bufferData(this._gl.UNIFORM_BUFFER, new Float32Array(elements), this._gl.DYNAMIC_DRAW);\n }\n this.bindUniformBuffer(null);\n result.references = 1;\n return result;\n };\n ThinEngine.prototype.updateUniformBuffer = function (uniformBuffer, elements, offset, count) {\n this.bindUniformBuffer(uniformBuffer);\n if (offset === undefined) {\n offset = 0;\n }\n if (count === undefined) {\n if (elements instanceof Float32Array) {\n this._gl.bufferSubData(this._gl.UNIFORM_BUFFER, offset, elements);\n }\n else {\n this._gl.bufferSubData(this._gl.UNIFORM_BUFFER, offset, new Float32Array(elements));\n }\n }\n else {\n if (elements instanceof Float32Array) {\n this._gl.bufferSubData(this._gl.UNIFORM_BUFFER, 0, elements.subarray(offset, offset + count));\n }\n else {\n this._gl.bufferSubData(this._gl.UNIFORM_BUFFER, 0, new Float32Array(elements).subarray(offset, offset + count));\n }\n }\n this.bindUniformBuffer(null);\n };\n ThinEngine.prototype.bindUniformBuffer = function (buffer) {\n this._gl.bindBuffer(this._gl.UNIFORM_BUFFER, buffer ? buffer.underlyingResource : null);\n };\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n ThinEngine.prototype.bindUniformBufferBase = function (buffer, location, name) {\n this._gl.bindBufferBase(this._gl.UNIFORM_BUFFER, location, buffer ? buffer.underlyingResource : null);\n };\n ThinEngine.prototype.bindUniformBlock = function (pipelineContext, blockName, index) {\n const program = pipelineContext.program;\n const uniformLocation = this._gl.getUniformBlockIndex(program, blockName);\n if (uniformLocation !== 0xffffffff) {\n this._gl.uniformBlockBinding(program, uniformLocation, index);\n }\n };\n\n /**\n * Uniform buffer objects.\n *\n * Handles blocks of uniform on the GPU.\n *\n * If WebGL 2 is not available, this class falls back on traditional setUniformXXX calls.\n *\n * For more information, please refer to :\n * https://www.khronos.org/opengl/wiki/Uniform_Buffer_Object\n */\n class UniformBuffer {\n /**\n * Instantiates a new Uniform buffer objects.\n *\n * Handles blocks of uniform on the GPU.\n *\n * If WebGL 2 is not available, this class falls back on traditional setUniformXXX calls.\n *\n * For more information, please refer to :\n * @see https://www.khronos.org/opengl/wiki/Uniform_Buffer_Object\n * @param engine Define the engine the buffer is associated with\n * @param data Define the data contained in the buffer\n * @param dynamic Define if the buffer is updatable\n * @param name to assign to the buffer (debugging purpose)\n * @param forceNoUniformBuffer define that this object must not rely on UBO objects\n */\n constructor(engine, data, dynamic, name, forceNoUniformBuffer = false) {\n // Matrix cache\n this._valueCache = {};\n this._engine = engine;\n this._noUBO = !engine.supportsUniformBuffers || forceNoUniformBuffer;\n this._dynamic = dynamic;\n this._name = name !== null && name !== void 0 ? name : \"no-name\";\n this._data = data || [];\n this._uniformLocations = {};\n this._uniformSizes = {};\n this._uniformArraySizes = {};\n this._uniformLocationPointer = 0;\n this._needSync = false;\n if (this._engine._features.trackUbosInFrame) {\n this._buffers = [];\n this._bufferIndex = -1;\n this._createBufferOnWrite = false;\n this._currentFrameId = 0;\n }\n if (this._noUBO) {\n this.updateMatrix3x3 = this._updateMatrix3x3ForEffect;\n this.updateMatrix2x2 = this._updateMatrix2x2ForEffect;\n this.updateFloat = this._updateFloatForEffect;\n this.updateFloat2 = this._updateFloat2ForEffect;\n this.updateFloat3 = this._updateFloat3ForEffect;\n this.updateFloat4 = this._updateFloat4ForEffect;\n this.updateFloatArray = this._updateFloatArrayForEffect;\n this.updateArray = this._updateArrayForEffect;\n this.updateIntArray = this._updateIntArrayForEffect;\n this.updateUIntArray = this._updateUIntArrayForEffect;\n this.updateMatrix = this._updateMatrixForEffect;\n this.updateMatrices = this._updateMatricesForEffect;\n this.updateVector3 = this._updateVector3ForEffect;\n this.updateVector4 = this._updateVector4ForEffect;\n this.updateColor3 = this._updateColor3ForEffect;\n this.updateColor4 = this._updateColor4ForEffect;\n this.updateDirectColor4 = this._updateDirectColor4ForEffect;\n this.updateInt = this._updateIntForEffect;\n this.updateInt2 = this._updateInt2ForEffect;\n this.updateInt3 = this._updateInt3ForEffect;\n this.updateInt4 = this._updateInt4ForEffect;\n this.updateUInt = this._updateUIntForEffect;\n this.updateUInt2 = this._updateUInt2ForEffect;\n this.updateUInt3 = this._updateUInt3ForEffect;\n this.updateUInt4 = this._updateUInt4ForEffect;\n }\n else {\n this._engine._uniformBuffers.push(this);\n this.updateMatrix3x3 = this._updateMatrix3x3ForUniform;\n this.updateMatrix2x2 = this._updateMatrix2x2ForUniform;\n this.updateFloat = this._updateFloatForUniform;\n this.updateFloat2 = this._updateFloat2ForUniform;\n this.updateFloat3 = this._updateFloat3ForUniform;\n this.updateFloat4 = this._updateFloat4ForUniform;\n this.updateFloatArray = this._updateFloatArrayForUniform;\n this.updateArray = this._updateArrayForUniform;\n this.updateIntArray = this._updateIntArrayForUniform;\n this.updateUIntArray = this._updateUIntArrayForUniform;\n this.updateMatrix = this._updateMatrixForUniform;\n this.updateMatrices = this._updateMatricesForUniform;\n this.updateVector3 = this._updateVector3ForUniform;\n this.updateVector4 = this._updateVector4ForUniform;\n this.updateColor3 = this._updateColor3ForUniform;\n this.updateColor4 = this._updateColor4ForUniform;\n this.updateDirectColor4 = this._updateDirectColor4ForUniform;\n this.updateInt = this._updateIntForUniform;\n this.updateInt2 = this._updateInt2ForUniform;\n this.updateInt3 = this._updateInt3ForUniform;\n this.updateInt4 = this._updateInt4ForUniform;\n this.updateUInt = this._updateUIntForUniform;\n this.updateUInt2 = this._updateUInt2ForUniform;\n this.updateUInt3 = this._updateUInt3ForUniform;\n this.updateUInt4 = this._updateUInt4ForUniform;\n }\n }\n /**\n * Indicates if the buffer is using the WebGL2 UBO implementation,\n * or just falling back on setUniformXXX calls.\n */\n get useUbo() {\n return !this._noUBO;\n }\n /**\n * Indicates if the WebGL underlying uniform buffer is in sync\n * with the javascript cache data.\n */\n get isSync() {\n return !this._needSync;\n }\n /**\n * Indicates if the WebGL underlying uniform buffer is dynamic.\n * Also, a dynamic UniformBuffer will disable cache verification and always\n * update the underlying WebGL uniform buffer to the GPU.\n * @returns if Dynamic, otherwise false\n */\n isDynamic() {\n return this._dynamic !== undefined;\n }\n /**\n * The data cache on JS side.\n * @returns the underlying data as a float array\n */\n getData() {\n return this._bufferData;\n }\n /**\n * The underlying WebGL Uniform buffer.\n * @returns the webgl buffer\n */\n getBuffer() {\n return this._buffer;\n }\n /**\n * std140 layout specifies how to align data within an UBO structure.\n * See https://khronos.org/registry/OpenGL/specs/gl/glspec45.core.pdf#page=159\n * for specs.\n * @param size\n */\n _fillAlignment(size) {\n // This code has been simplified because we only use floats, vectors of 1, 2, 3, 4 components\n // and 4x4 matrices\n // TODO : change if other types are used\n let alignment;\n if (size <= 2) {\n alignment = size;\n }\n else {\n alignment = 4;\n }\n if (this._uniformLocationPointer % alignment !== 0) {\n const oldPointer = this._uniformLocationPointer;\n this._uniformLocationPointer += alignment - (this._uniformLocationPointer % alignment);\n const diff = this._uniformLocationPointer - oldPointer;\n for (let i = 0; i < diff; i++) {\n this._data.push(0);\n }\n }\n }\n /**\n * Adds an uniform in the buffer.\n * Warning : the subsequents calls of this function must be in the same order as declared in the shader\n * for the layout to be correct ! The addUniform function only handles types like float, vec2, vec3, vec4, mat4,\n * meaning size=1,2,3,4 or 16. It does not handle struct types.\n * @param name Name of the uniform, as used in the uniform block in the shader.\n * @param size Data size, or data directly.\n * @param arraySize The number of elements in the array, 0 if not an array.\n */\n addUniform(name, size, arraySize = 0) {\n if (this._noUBO) {\n return;\n }\n if (this._uniformLocations[name] !== undefined) {\n // Already existing uniform\n return;\n }\n // This function must be called in the order of the shader layout !\n // size can be the size of the uniform, or data directly\n let data;\n // std140 FTW...\n if (arraySize > 0) {\n if (size instanceof Array) {\n throw \"addUniform should not be use with Array in UBO: \" + name;\n }\n this._fillAlignment(4);\n this._uniformArraySizes[name] = { strideSize: size, arraySize };\n if (size == 16) {\n size = size * arraySize;\n }\n else {\n const perElementPadding = 4 - size;\n const totalPadding = perElementPadding * arraySize;\n size = size * arraySize + totalPadding;\n }\n data = [];\n // Fill with zeros\n for (let i = 0; i < size; i++) {\n data.push(0);\n }\n }\n else {\n if (size instanceof Array) {\n data = size;\n size = data.length;\n }\n else {\n size = size;\n data = [];\n // Fill with zeros\n for (let i = 0; i < size; i++) {\n data.push(0);\n }\n }\n this._fillAlignment(size);\n }\n this._uniformSizes[name] = size;\n this._uniformLocations[name] = this._uniformLocationPointer;\n this._uniformLocationPointer += size;\n for (let i = 0; i < size; i++) {\n this._data.push(data[i]);\n }\n this._needSync = true;\n }\n /**\n * Adds a Matrix 4x4 to the uniform buffer.\n * @param name Name of the uniform, as used in the uniform block in the shader.\n * @param mat A 4x4 matrix.\n */\n addMatrix(name, mat) {\n this.addUniform(name, Array.prototype.slice.call(mat.toArray()));\n }\n /**\n * Adds a vec2 to the uniform buffer.\n * @param name Name of the uniform, as used in the uniform block in the shader.\n * @param x Define the x component value of the vec2\n * @param y Define the y component value of the vec2\n */\n addFloat2(name, x, y) {\n const temp = [x, y];\n this.addUniform(name, temp);\n }\n /**\n * Adds a vec3 to the uniform buffer.\n * @param name Name of the uniform, as used in the uniform block in the shader.\n * @param x Define the x component value of the vec3\n * @param y Define the y component value of the vec3\n * @param z Define the z component value of the vec3\n */\n addFloat3(name, x, y, z) {\n const temp = [x, y, z];\n this.addUniform(name, temp);\n }\n /**\n * Adds a vec3 to the uniform buffer.\n * @param name Name of the uniform, as used in the uniform block in the shader.\n * @param color Define the vec3 from a Color\n */\n addColor3(name, color) {\n const temp = [color.r, color.g, color.b];\n this.addUniform(name, temp);\n }\n /**\n * Adds a vec4 to the uniform buffer.\n * @param name Name of the uniform, as used in the uniform block in the shader.\n * @param color Define the rgb components from a Color\n * @param alpha Define the a component of the vec4\n */\n addColor4(name, color, alpha) {\n const temp = [color.r, color.g, color.b, alpha];\n this.addUniform(name, temp);\n }\n /**\n * Adds a vec3 to the uniform buffer.\n * @param name Name of the uniform, as used in the uniform block in the shader.\n * @param vector Define the vec3 components from a Vector\n */\n addVector3(name, vector) {\n const temp = [vector.x, vector.y, vector.z];\n this.addUniform(name, temp);\n }\n /**\n * Adds a Matrix 3x3 to the uniform buffer.\n * @param name Name of the uniform, as used in the uniform block in the shader.\n */\n addMatrix3x3(name) {\n this.addUniform(name, 12);\n }\n /**\n * Adds a Matrix 2x2 to the uniform buffer.\n * @param name Name of the uniform, as used in the uniform block in the shader.\n */\n addMatrix2x2(name) {\n this.addUniform(name, 8);\n }\n /**\n * Effectively creates the WebGL Uniform Buffer, once layout is completed with `addUniform`.\n */\n create() {\n if (this._noUBO) {\n return;\n }\n if (this._buffer) {\n return; // nothing to do\n }\n // See spec, alignment must be filled as a vec4\n this._fillAlignment(4);\n this._bufferData = new Float32Array(this._data);\n this._rebuild();\n this._needSync = true;\n }\n /** @internal */\n _rebuild() {\n if (this._noUBO || !this._bufferData) {\n return;\n }\n if (this._dynamic) {\n this._buffer = this._engine.createDynamicUniformBuffer(this._bufferData);\n }\n else {\n this._buffer = this._engine.createUniformBuffer(this._bufferData);\n }\n if (this._engine._features.trackUbosInFrame) {\n this._buffers.push([this._buffer, this._engine._features.checkUbosContentBeforeUpload ? this._bufferData.slice() : undefined]);\n this._bufferIndex = this._buffers.length - 1;\n this._createBufferOnWrite = false;\n }\n }\n /** @internal */\n get _numBuffers() {\n return this._buffers.length;\n }\n /** @internal */\n get _indexBuffer() {\n return this._bufferIndex;\n }\n /** Gets the name of this buffer */\n get name() {\n return this._name;\n }\n /** Gets the current effect */\n get currentEffect() {\n return this._currentEffect;\n }\n _buffersEqual(buf1, buf2) {\n for (let i = 0; i < buf1.length; ++i) {\n if (buf1[i] !== buf2[i]) {\n return false;\n }\n }\n return true;\n }\n _copyBuffer(src, dst) {\n for (let i = 0; i < src.length; ++i) {\n dst[i] = src[i];\n }\n }\n /**\n * Updates the WebGL Uniform Buffer on the GPU.\n * If the `dynamic` flag is set to true, no cache comparison is done.\n * Otherwise, the buffer will be updated only if the cache differs.\n */\n update() {\n if (this._noUBO) {\n return;\n }\n this.bindUniformBuffer();\n if (!this._buffer) {\n this.create();\n return;\n }\n if (!this._dynamic && !this._needSync) {\n this._createBufferOnWrite = this._engine._features.trackUbosInFrame;\n return;\n }\n if (this._buffers && this._buffers.length > 1 && this._buffers[this._bufferIndex][1]) {\n if (this._buffersEqual(this._bufferData, this._buffers[this._bufferIndex][1])) {\n this._needSync = false;\n this._createBufferOnWrite = this._engine._features.trackUbosInFrame;\n return;\n }\n else {\n this._copyBuffer(this._bufferData, this._buffers[this._bufferIndex][1]);\n }\n }\n this._engine.updateUniformBuffer(this._buffer, this._bufferData);\n if (this._engine._features._collectUbosUpdatedInFrame) {\n if (!UniformBuffer._UpdatedUbosInFrame[this._name]) {\n UniformBuffer._UpdatedUbosInFrame[this._name] = 0;\n }\n UniformBuffer._UpdatedUbosInFrame[this._name]++;\n }\n this._needSync = false;\n this._createBufferOnWrite = this._engine._features.trackUbosInFrame;\n }\n _createNewBuffer() {\n if (this._bufferIndex + 1 < this._buffers.length) {\n this._bufferIndex++;\n this._buffer = this._buffers[this._bufferIndex][0];\n this._createBufferOnWrite = false;\n this._needSync = true;\n }\n else {\n this._rebuild();\n }\n }\n _checkNewFrame() {\n if (this._engine._features.trackUbosInFrame && this._currentFrameId !== this._engine.frameId) {\n this._currentFrameId = this._engine.frameId;\n this._createBufferOnWrite = false;\n if (this._buffers && this._buffers.length > 0) {\n this._needSync = this._bufferIndex !== 0;\n this._bufferIndex = 0;\n this._buffer = this._buffers[this._bufferIndex][0];\n }\n else {\n this._bufferIndex = -1;\n }\n }\n }\n /**\n * Updates the value of an uniform. The `update` method must be called afterwards to make it effective in the GPU.\n * @param uniformName Define the name of the uniform, as used in the uniform block in the shader.\n * @param data Define the flattened data\n * @param size Define the size of the data.\n */\n updateUniform(uniformName, data, size) {\n this._checkNewFrame();\n let location = this._uniformLocations[uniformName];\n if (location === undefined) {\n if (this._buffer) {\n // Cannot add an uniform if the buffer is already created\n Logger.Error(\"Cannot add an uniform after UBO has been created.\");\n return;\n }\n this.addUniform(uniformName, size);\n location = this._uniformLocations[uniformName];\n }\n if (!this._buffer) {\n this.create();\n }\n if (!this._dynamic) {\n // Cache for static uniform buffers\n let changed = false;\n for (let i = 0; i < size; i++) {\n // We are checking the matrix cache before calling updateUniform so we do not need to check it here\n // Hence the test for size === 16 to simply commit the matrix values\n if ((size === 16 && !this._engine._features.uniformBufferHardCheckMatrix) || this._bufferData[location + i] !== Tools.FloatRound(data[i])) {\n changed = true;\n if (this._createBufferOnWrite) {\n this._createNewBuffer();\n }\n this._bufferData[location + i] = data[i];\n }\n }\n this._needSync = this._needSync || changed;\n }\n else {\n // No cache for dynamic\n for (let i = 0; i < size; i++) {\n this._bufferData[location + i] = data[i];\n }\n }\n }\n /**\n * Updates the value of an uniform. The `update` method must be called afterwards to make it effective in the GPU.\n * @param uniformName Define the name of the uniform, as used in the uniform block in the shader.\n * @param data Define the flattened data\n * @param size Define the size of the data.\n */\n updateUniformArray(uniformName, data, size) {\n this._checkNewFrame();\n const location = this._uniformLocations[uniformName];\n if (location === undefined) {\n Logger.Error(\"Cannot add an uniform Array dynamically. Please, add it using addUniform and make sure that uniform buffers are supported by the current engine.\");\n return;\n }\n if (!this._buffer) {\n this.create();\n }\n const arraySizes = this._uniformArraySizes[uniformName];\n if (!this._dynamic) {\n // Cache for static uniform buffers\n let changed = false;\n let countToFour = 0;\n let baseStride = 0;\n for (let i = 0; i < size; i++) {\n if (this._bufferData[location + baseStride * 4 + countToFour] !== Tools.FloatRound(data[i])) {\n changed = true;\n if (this._createBufferOnWrite) {\n this._createNewBuffer();\n }\n this._bufferData[location + baseStride * 4 + countToFour] = data[i];\n }\n countToFour++;\n if (countToFour === arraySizes.strideSize) {\n for (; countToFour < 4; countToFour++) {\n this._bufferData[location + baseStride * 4 + countToFour] = 0;\n }\n countToFour = 0;\n baseStride++;\n }\n }\n this._needSync = this._needSync || changed;\n }\n else {\n // No cache for dynamic\n for (let i = 0; i < size; i++) {\n this._bufferData[location + i] = data[i];\n }\n }\n }\n _cacheMatrix(name, matrix) {\n this._checkNewFrame();\n const cache = this._valueCache[name];\n const flag = matrix.updateFlag;\n if (cache !== undefined && cache === flag) {\n return false;\n }\n this._valueCache[name] = flag;\n return true;\n }\n // Update methods\n _updateMatrix3x3ForUniform(name, matrix) {\n // To match std140, matrix must be realigned\n for (let i = 0; i < 3; i++) {\n UniformBuffer._TempBuffer[i * 4] = matrix[i * 3];\n UniformBuffer._TempBuffer[i * 4 + 1] = matrix[i * 3 + 1];\n UniformBuffer._TempBuffer[i * 4 + 2] = matrix[i * 3 + 2];\n UniformBuffer._TempBuffer[i * 4 + 3] = 0.0;\n }\n this.updateUniform(name, UniformBuffer._TempBuffer, 12);\n }\n _updateMatrix3x3ForEffect(name, matrix) {\n this._currentEffect.setMatrix3x3(name, matrix);\n }\n _updateMatrix2x2ForEffect(name, matrix) {\n this._currentEffect.setMatrix2x2(name, matrix);\n }\n _updateMatrix2x2ForUniform(name, matrix) {\n // To match std140, matrix must be realigned\n for (let i = 0; i < 2; i++) {\n UniformBuffer._TempBuffer[i * 4] = matrix[i * 2];\n UniformBuffer._TempBuffer[i * 4 + 1] = matrix[i * 2 + 1];\n UniformBuffer._TempBuffer[i * 4 + 2] = 0.0;\n UniformBuffer._TempBuffer[i * 4 + 3] = 0.0;\n }\n this.updateUniform(name, UniformBuffer._TempBuffer, 8);\n }\n _updateFloatForEffect(name, x) {\n this._currentEffect.setFloat(name, x);\n }\n _updateFloatForUniform(name, x) {\n UniformBuffer._TempBuffer[0] = x;\n this.updateUniform(name, UniformBuffer._TempBuffer, 1);\n }\n _updateFloat2ForEffect(name, x, y, suffix = \"\") {\n this._currentEffect.setFloat2(name + suffix, x, y);\n }\n _updateFloat2ForUniform(name, x, y) {\n UniformBuffer._TempBuffer[0] = x;\n UniformBuffer._TempBuffer[1] = y;\n this.updateUniform(name, UniformBuffer._TempBuffer, 2);\n }\n _updateFloat3ForEffect(name, x, y, z, suffix = \"\") {\n this._currentEffect.setFloat3(name + suffix, x, y, z);\n }\n _updateFloat3ForUniform(name, x, y, z) {\n UniformBuffer._TempBuffer[0] = x;\n UniformBuffer._TempBuffer[1] = y;\n UniformBuffer._TempBuffer[2] = z;\n this.updateUniform(name, UniformBuffer._TempBuffer, 3);\n }\n _updateFloat4ForEffect(name, x, y, z, w, suffix = \"\") {\n this._currentEffect.setFloat4(name + suffix, x, y, z, w);\n }\n _updateFloat4ForUniform(name, x, y, z, w) {\n UniformBuffer._TempBuffer[0] = x;\n UniformBuffer._TempBuffer[1] = y;\n UniformBuffer._TempBuffer[2] = z;\n UniformBuffer._TempBuffer[3] = w;\n this.updateUniform(name, UniformBuffer._TempBuffer, 4);\n }\n _updateFloatArrayForEffect(name, array) {\n this._currentEffect.setFloatArray(name, array);\n }\n _updateFloatArrayForUniform(name, array) {\n this.updateUniformArray(name, array, array.length);\n }\n _updateArrayForEffect(name, array) {\n this._currentEffect.setArray(name, array);\n }\n _updateArrayForUniform(name, array) {\n this.updateUniformArray(name, array, array.length);\n }\n _updateIntArrayForEffect(name, array) {\n this._currentEffect.setIntArray(name, array);\n }\n _updateIntArrayForUniform(name, array) {\n UniformBuffer._TempBufferInt32View.set(array);\n this.updateUniformArray(name, UniformBuffer._TempBuffer, array.length);\n }\n _updateUIntArrayForEffect(name, array) {\n this._currentEffect.setUIntArray(name, array);\n }\n _updateUIntArrayForUniform(name, array) {\n UniformBuffer._TempBufferUInt32View.set(array);\n this.updateUniformArray(name, UniformBuffer._TempBuffer, array.length);\n }\n _updateMatrixForEffect(name, mat) {\n this._currentEffect.setMatrix(name, mat);\n }\n _updateMatrixForUniform(name, mat) {\n if (this._cacheMatrix(name, mat)) {\n this.updateUniform(name, mat.toArray(), 16);\n }\n }\n _updateMatricesForEffect(name, mat) {\n this._currentEffect.setMatrices(name, mat);\n }\n _updateMatricesForUniform(name, mat) {\n this.updateUniform(name, mat, mat.length);\n }\n _updateVector3ForEffect(name, vector) {\n this._currentEffect.setVector3(name, vector);\n }\n _updateVector3ForUniform(name, vector) {\n UniformBuffer._TempBuffer[0] = vector.x;\n UniformBuffer._TempBuffer[1] = vector.y;\n UniformBuffer._TempBuffer[2] = vector.z;\n this.updateUniform(name, UniformBuffer._TempBuffer, 3);\n }\n _updateVector4ForEffect(name, vector) {\n this._currentEffect.setVector4(name, vector);\n }\n _updateVector4ForUniform(name, vector) {\n UniformBuffer._TempBuffer[0] = vector.x;\n UniformBuffer._TempBuffer[1] = vector.y;\n UniformBuffer._TempBuffer[2] = vector.z;\n UniformBuffer._TempBuffer[3] = vector.w;\n this.updateUniform(name, UniformBuffer._TempBuffer, 4);\n }\n _updateColor3ForEffect(name, color, suffix = \"\") {\n this._currentEffect.setColor3(name + suffix, color);\n }\n _updateColor3ForUniform(name, color) {\n UniformBuffer._TempBuffer[0] = color.r;\n UniformBuffer._TempBuffer[1] = color.g;\n UniformBuffer._TempBuffer[2] = color.b;\n this.updateUniform(name, UniformBuffer._TempBuffer, 3);\n }\n _updateColor4ForEffect(name, color, alpha, suffix = \"\") {\n this._currentEffect.setColor4(name + suffix, color, alpha);\n }\n _updateDirectColor4ForEffect(name, color, suffix = \"\") {\n this._currentEffect.setDirectColor4(name + suffix, color);\n }\n _updateColor4ForUniform(name, color, alpha) {\n UniformBuffer._TempBuffer[0] = color.r;\n UniformBuffer._TempBuffer[1] = color.g;\n UniformBuffer._TempBuffer[2] = color.b;\n UniformBuffer._TempBuffer[3] = alpha;\n this.updateUniform(name, UniformBuffer._TempBuffer, 4);\n }\n _updateDirectColor4ForUniform(name, color) {\n UniformBuffer._TempBuffer[0] = color.r;\n UniformBuffer._TempBuffer[1] = color.g;\n UniformBuffer._TempBuffer[2] = color.b;\n UniformBuffer._TempBuffer[3] = color.a;\n this.updateUniform(name, UniformBuffer._TempBuffer, 4);\n }\n _updateIntForEffect(name, x, suffix = \"\") {\n this._currentEffect.setInt(name + suffix, x);\n }\n _updateIntForUniform(name, x) {\n UniformBuffer._TempBufferInt32View[0] = x;\n this.updateUniform(name, UniformBuffer._TempBuffer, 1);\n }\n _updateInt2ForEffect(name, x, y, suffix = \"\") {\n this._currentEffect.setInt2(name + suffix, x, y);\n }\n _updateInt2ForUniform(name, x, y) {\n UniformBuffer._TempBufferInt32View[0] = x;\n UniformBuffer._TempBufferInt32View[1] = y;\n this.updateUniform(name, UniformBuffer._TempBuffer, 2);\n }\n _updateInt3ForEffect(name, x, y, z, suffix = \"\") {\n this._currentEffect.setInt3(name + suffix, x, y, z);\n }\n _updateInt3ForUniform(name, x, y, z) {\n UniformBuffer._TempBufferInt32View[0] = x;\n UniformBuffer._TempBufferInt32View[1] = y;\n UniformBuffer._TempBufferInt32View[2] = z;\n this.updateUniform(name, UniformBuffer._TempBuffer, 3);\n }\n _updateInt4ForEffect(name, x, y, z, w, suffix = \"\") {\n this._currentEffect.setInt4(name + suffix, x, y, z, w);\n }\n _updateInt4ForUniform(name, x, y, z, w) {\n UniformBuffer._TempBufferInt32View[0] = x;\n UniformBuffer._TempBufferInt32View[1] = y;\n UniformBuffer._TempBufferInt32View[2] = z;\n UniformBuffer._TempBufferInt32View[3] = w;\n this.updateUniform(name, UniformBuffer._TempBuffer, 4);\n }\n _updateUIntForEffect(name, x, suffix = \"\") {\n this._currentEffect.setUInt(name + suffix, x);\n }\n _updateUIntForUniform(name, x) {\n UniformBuffer._TempBufferUInt32View[0] = x;\n this.updateUniform(name, UniformBuffer._TempBuffer, 1);\n }\n _updateUInt2ForEffect(name, x, y, suffix = \"\") {\n this._currentEffect.setUInt2(name + suffix, x, y);\n }\n _updateUInt2ForUniform(name, x, y) {\n UniformBuffer._TempBufferUInt32View[0] = x;\n UniformBuffer._TempBufferUInt32View[1] = y;\n this.updateUniform(name, UniformBuffer._TempBuffer, 2);\n }\n _updateUInt3ForEffect(name, x, y, z, suffix = \"\") {\n this._currentEffect.setUInt3(name + suffix, x, y, z);\n }\n _updateUInt3ForUniform(name, x, y, z) {\n UniformBuffer._TempBufferUInt32View[0] = x;\n UniformBuffer._TempBufferUInt32View[1] = y;\n UniformBuffer._TempBufferUInt32View[2] = z;\n this.updateUniform(name, UniformBuffer._TempBuffer, 3);\n }\n _updateUInt4ForEffect(name, x, y, z, w, suffix = \"\") {\n this._currentEffect.setUInt4(name + suffix, x, y, z, w);\n }\n _updateUInt4ForUniform(name, x, y, z, w) {\n UniformBuffer._TempBufferUInt32View[0] = x;\n UniformBuffer._TempBufferUInt32View[1] = y;\n UniformBuffer._TempBufferUInt32View[2] = z;\n UniformBuffer._TempBufferUInt32View[3] = w;\n this.updateUniform(name, UniformBuffer._TempBuffer, 4);\n }\n /**\n * Sets a sampler uniform on the effect.\n * @param name Define the name of the sampler.\n * @param texture Define the texture to set in the sampler\n */\n setTexture(name, texture) {\n this._currentEffect.setTexture(name, texture);\n }\n /**\n * Directly updates the value of the uniform in the cache AND on the GPU.\n * @param uniformName Define the name of the uniform, as used in the uniform block in the shader.\n * @param data Define the flattened data\n */\n updateUniformDirectly(uniformName, data) {\n this.updateUniform(uniformName, data, data.length);\n this.update();\n }\n /**\n * Associates an effect to this uniform buffer\n * @param effect Define the effect to associate the buffer to\n * @param name Name of the uniform block in the shader.\n */\n bindToEffect(effect, name) {\n this._currentEffect = effect;\n this._currentEffectName = name;\n }\n /**\n * Binds the current (GPU) buffer to the effect\n */\n bindUniformBuffer() {\n if (!this._noUBO && this._buffer && this._currentEffect) {\n this._currentEffect.bindUniformBuffer(this._buffer, this._currentEffectName);\n }\n }\n /**\n * Dissociates the current effect from this uniform buffer\n */\n unbindEffect() {\n this._currentEffect = undefined;\n this._currentEffectName = undefined;\n }\n /**\n * Sets the current state of the class (_bufferIndex, _buffer) to point to the data buffer passed in parameter if this buffer is one of the buffers handled by the class (meaning if it can be found in the _buffers array)\n * This method is meant to be able to update a buffer at any time: just call setDataBuffer to set the class in the right state, call some updateXXX methods and then call udpate() => that will update the GPU buffer on the graphic card\n * @param dataBuffer buffer to look for\n * @returns true if the buffer has been found and the class internal state points to it, else false\n */\n setDataBuffer(dataBuffer) {\n if (!this._buffers) {\n return this._buffer === dataBuffer;\n }\n for (let b = 0; b < this._buffers.length; ++b) {\n const buffer = this._buffers[b];\n if (buffer[0] === dataBuffer) {\n this._bufferIndex = b;\n this._buffer = dataBuffer;\n this._createBufferOnWrite = false;\n this._currentEffect = undefined;\n return true;\n }\n }\n return false;\n }\n /**\n * Disposes the uniform buffer.\n */\n dispose() {\n if (this._noUBO) {\n return;\n }\n const uniformBuffers = this._engine._uniformBuffers;\n const index = uniformBuffers.indexOf(this);\n if (index !== -1) {\n uniformBuffers[index] = uniformBuffers[uniformBuffers.length - 1];\n uniformBuffers.pop();\n }\n if (this._engine._features.trackUbosInFrame && this._buffers) {\n for (let i = 0; i < this._buffers.length; ++i) {\n const buffer = this._buffers[i][0];\n this._engine._releaseBuffer(buffer);\n }\n }\n else if (this._buffer && this._engine._releaseBuffer(this._buffer)) {\n this._buffer = null;\n }\n }\n }\n /** @internal */\n UniformBuffer._UpdatedUbosInFrame = {};\n // Pool for avoiding memory leaks\n UniformBuffer._MAX_UNIFORM_SIZE = 256;\n UniformBuffer._TempBuffer = new Float32Array(UniformBuffer._MAX_UNIFORM_SIZE);\n UniformBuffer._TempBufferInt32View = new Int32Array(UniformBuffer._TempBuffer.buffer);\n UniformBuffer._TempBufferUInt32View = new Uint32Array(UniformBuffer._TempBuffer.buffer);\n\n /**\n * Class used to store data that will be store in GPU memory\n */\n class Buffer$1 {\n /**\n * Constructor\n * @param engine the engine\n * @param data the data to use for this buffer\n * @param updatable whether the data is updatable\n * @param stride the stride (optional)\n * @param postponeInternalCreation whether to postpone creating the internal WebGL buffer (optional)\n * @param instanced whether the buffer is instanced (optional)\n * @param useBytes set to true if the stride in in bytes (optional)\n * @param divisor sets an optional divisor for instances (1 by default)\n */\n constructor(engine, data, updatable, stride = 0, postponeInternalCreation = false, instanced = false, useBytes = false, divisor) {\n this._isAlreadyOwned = false;\n if (engine.getScene) {\n // old versions of VertexBuffer accepted 'mesh' instead of 'engine'\n this._engine = engine.getScene().getEngine();\n }\n else {\n this._engine = engine;\n }\n this._updatable = updatable;\n this._instanced = instanced;\n this._divisor = divisor || 1;\n if (data instanceof DataBuffer) {\n this._data = null;\n this._buffer = data;\n }\n else {\n this._data = data;\n this._buffer = null;\n }\n this.byteStride = useBytes ? stride : stride * Float32Array.BYTES_PER_ELEMENT;\n if (!postponeInternalCreation) {\n // by default\n this.create();\n }\n }\n /**\n * Create a new VertexBuffer based on the current buffer\n * @param kind defines the vertex buffer kind (position, normal, etc.)\n * @param offset defines offset in the buffer (0 by default)\n * @param size defines the size in floats of attributes (position is 3 for instance)\n * @param stride defines the stride size in floats in the buffer (the offset to apply to reach next value when data is interleaved)\n * @param instanced defines if the vertex buffer contains indexed data\n * @param useBytes defines if the offset and stride are in bytes *\n * @param divisor sets an optional divisor for instances (1 by default)\n * @returns the new vertex buffer\n */\n createVertexBuffer(kind, offset, size, stride, instanced, useBytes = false, divisor) {\n const byteOffset = useBytes ? offset : offset * Float32Array.BYTES_PER_ELEMENT;\n const byteStride = stride ? (useBytes ? stride : stride * Float32Array.BYTES_PER_ELEMENT) : this.byteStride;\n // a lot of these parameters are ignored as they are overridden by the buffer\n return new VertexBuffer(this._engine, this, kind, this._updatable, true, byteStride, instanced === undefined ? this._instanced : instanced, byteOffset, size, undefined, undefined, true, this._divisor || divisor);\n }\n // Properties\n /**\n * Gets a boolean indicating if the Buffer is updatable?\n * @returns true if the buffer is updatable\n */\n isUpdatable() {\n return this._updatable;\n }\n /**\n * Gets current buffer's data\n * @returns a DataArray or null\n */\n getData() {\n return this._data;\n }\n /**\n * Gets underlying native buffer\n * @returns underlying native buffer\n */\n getBuffer() {\n return this._buffer;\n }\n /**\n * Gets the stride in float32 units (i.e. byte stride / 4).\n * May not be an integer if the byte stride is not divisible by 4.\n * @returns the stride in float32 units\n * @deprecated Please use byteStride instead.\n */\n getStrideSize() {\n return this.byteStride / Float32Array.BYTES_PER_ELEMENT;\n }\n // Methods\n /**\n * Store data into the buffer. Creates the buffer if not used already.\n * If the buffer was already used, it will be updated only if it is updatable, otherwise it will do nothing.\n * @param data defines the data to store\n */\n create(data = null) {\n if (!data && this._buffer) {\n return; // nothing to do\n }\n data = data || this._data;\n if (!data) {\n return;\n }\n if (!this._buffer) {\n // create buffer\n if (this._updatable) {\n this._buffer = this._engine.createDynamicVertexBuffer(data);\n this._data = data;\n }\n else {\n this._buffer = this._engine.createVertexBuffer(data);\n }\n }\n else if (this._updatable) {\n // update buffer\n this._engine.updateDynamicVertexBuffer(this._buffer, data);\n this._data = data;\n }\n }\n /** @internal */\n _rebuild() {\n this._buffer = null;\n this.create(this._data);\n }\n /**\n * Update current buffer data\n * @param data defines the data to store\n */\n update(data) {\n this.create(data);\n }\n /**\n * Updates the data directly.\n * @param data the new data\n * @param offset the new offset\n * @param vertexCount the vertex count (optional)\n * @param useBytes set to true if the offset is in bytes\n */\n updateDirectly(data, offset, vertexCount, useBytes = false) {\n if (!this._buffer) {\n return;\n }\n if (this._updatable) {\n // update buffer\n this._engine.updateDynamicVertexBuffer(this._buffer, data, useBytes ? offset : offset * Float32Array.BYTES_PER_ELEMENT, vertexCount ? vertexCount * this.byteStride : undefined);\n if (offset === 0 && vertexCount === undefined) {\n // Keep the data if we easily can\n this._data = data;\n }\n else {\n this._data = null;\n }\n }\n }\n /** @internal */\n _increaseReferences() {\n if (!this._buffer) {\n return;\n }\n if (!this._isAlreadyOwned) {\n this._isAlreadyOwned = true;\n return;\n }\n this._buffer.references++;\n }\n /**\n * Release all resources\n */\n dispose() {\n if (!this._buffer) {\n return;\n }\n if (this._engine._releaseBuffer(this._buffer)) {\n this._buffer = null;\n this._data = null;\n }\n }\n }\n /**\n * Specialized buffer used to store vertex data\n */\n class VertexBuffer {\n /**\n * Gets or sets the instance divisor when in instanced mode\n */\n get instanceDivisor() {\n return this._instanceDivisor;\n }\n set instanceDivisor(value) {\n const isInstanced = value != 0;\n this._instanceDivisor = value;\n if (isInstanced !== this._instanced) {\n this._instanced = isInstanced;\n this._computeHashCode();\n }\n }\n /**\n * Constructor\n * @param engine the engine\n * @param data the data to use for this vertex buffer\n * @param kind the vertex buffer kind\n * @param updatable whether the data is updatable\n * @param postponeInternalCreation whether to postpone creating the internal WebGL buffer (optional)\n * @param stride the stride (optional)\n * @param instanced whether the buffer is instanced (optional)\n * @param offset the offset of the data (optional)\n * @param size the number of components (optional)\n * @param type the type of the component (optional)\n * @param normalized whether the data contains normalized data (optional)\n * @param useBytes set to true if stride and offset are in bytes (optional)\n * @param divisor defines the instance divisor to use (1 by default)\n * @param takeBufferOwnership defines if the buffer should be released when the vertex buffer is disposed\n */\n constructor(engine, data, kind, updatable, postponeInternalCreation, stride, instanced, offset, size, type, normalized = false, useBytes = false, divisor = 1, takeBufferOwnership = false) {\n if (data instanceof Buffer$1) {\n this._buffer = data;\n this._ownsBuffer = takeBufferOwnership;\n }\n else {\n this._buffer = new Buffer$1(engine, data, updatable, stride, postponeInternalCreation, instanced, useBytes);\n this._ownsBuffer = true;\n }\n this.uniqueId = VertexBuffer._Counter++;\n this._kind = kind;\n if (type == undefined) {\n const vertexData = this.getData();\n this.type = VertexBuffer.FLOAT;\n if (vertexData instanceof Int8Array) {\n this.type = VertexBuffer.BYTE;\n }\n else if (vertexData instanceof Uint8Array) {\n this.type = VertexBuffer.UNSIGNED_BYTE;\n }\n else if (vertexData instanceof Int16Array) {\n this.type = VertexBuffer.SHORT;\n }\n else if (vertexData instanceof Uint16Array) {\n this.type = VertexBuffer.UNSIGNED_SHORT;\n }\n else if (vertexData instanceof Int32Array) {\n this.type = VertexBuffer.INT;\n }\n else if (vertexData instanceof Uint32Array) {\n this.type = VertexBuffer.UNSIGNED_INT;\n }\n }\n else {\n this.type = type;\n }\n const typeByteLength = VertexBuffer.GetTypeByteLength(this.type);\n if (useBytes) {\n this._size = size || (stride ? stride / typeByteLength : VertexBuffer.DeduceStride(kind));\n this.byteStride = stride || this._buffer.byteStride || this._size * typeByteLength;\n this.byteOffset = offset || 0;\n }\n else {\n this._size = size || stride || VertexBuffer.DeduceStride(kind);\n this.byteStride = stride ? stride * typeByteLength : this._buffer.byteStride || this._size * typeByteLength;\n this.byteOffset = (offset || 0) * typeByteLength;\n }\n this.normalized = normalized;\n this._instanced = instanced !== undefined ? instanced : false;\n this._instanceDivisor = instanced ? divisor : 0;\n this._computeHashCode();\n }\n _computeHashCode() {\n // note: cast to any because the property is declared readonly\n this.hashCode =\n ((this.type - 5120) << 0) +\n ((this.normalized ? 1 : 0) << 3) +\n (this._size << 4) +\n ((this._instanced ? 1 : 0) << 6) +\n /* keep 5 bits free */\n (this.byteStride << 12);\n }\n /** @internal */\n _rebuild() {\n if (!this._buffer) {\n return;\n }\n this._buffer._rebuild();\n }\n /**\n * Returns the kind of the VertexBuffer (string)\n * @returns a string\n */\n getKind() {\n return this._kind;\n }\n // Properties\n /**\n * Gets a boolean indicating if the VertexBuffer is updatable?\n * @returns true if the buffer is updatable\n */\n isUpdatable() {\n return this._buffer.isUpdatable();\n }\n /**\n * Gets current buffer's data\n * @returns a DataArray or null\n */\n getData() {\n return this._buffer.getData();\n }\n /**\n * Gets current buffer's data as a float array. Float data is constructed if the vertex buffer data cannot be returned directly.\n * @param totalVertices number of vertices in the buffer to take into account\n * @param forceCopy defines a boolean indicating that the returned array must be cloned upon returning it\n * @returns a float array containing vertex data\n */\n getFloatData(totalVertices, forceCopy) {\n const data = this.getData();\n if (!data) {\n return null;\n }\n const tightlyPackedByteStride = this.getSize() * VertexBuffer.GetTypeByteLength(this.type);\n const count = totalVertices * this.getSize();\n if (this.type !== VertexBuffer.FLOAT || this.byteStride !== tightlyPackedByteStride) {\n const copy = new Float32Array(count);\n this.forEach(count, (value, index) => (copy[index] = value));\n return copy;\n }\n if (!(data instanceof Array || data instanceof Float32Array) || this.byteOffset !== 0 || data.length !== count) {\n if (data instanceof Array) {\n const offset = this.byteOffset / 4;\n return data.slice(offset, offset + count);\n }\n else if (data instanceof ArrayBuffer) {\n return new Float32Array(data, this.byteOffset, count);\n }\n else {\n let offset = data.byteOffset + this.byteOffset;\n if (forceCopy) {\n const result = new Float32Array(count);\n const source = new Float32Array(data.buffer, offset, count);\n result.set(source);\n return result;\n }\n // Protect against bad data\n const remainder = offset % 4;\n if (remainder) {\n offset = Math.max(0, offset - remainder);\n }\n return new Float32Array(data.buffer, offset, count);\n }\n }\n if (forceCopy) {\n return data.slice();\n }\n return data;\n }\n /**\n * Gets underlying native buffer\n * @returns underlying native buffer\n */\n getBuffer() {\n return this._buffer.getBuffer();\n }\n /**\n * Gets the stride in float32 units (i.e. byte stride / 4).\n * May not be an integer if the byte stride is not divisible by 4.\n * @returns the stride in float32 units\n * @deprecated Please use byteStride instead.\n */\n getStrideSize() {\n return this.byteStride / VertexBuffer.GetTypeByteLength(this.type);\n }\n /**\n * Returns the offset as a multiple of the type byte length.\n * @returns the offset in bytes\n * @deprecated Please use byteOffset instead.\n */\n getOffset() {\n return this.byteOffset / VertexBuffer.GetTypeByteLength(this.type);\n }\n /**\n * Returns the number of components or the byte size per vertex attribute\n * @param sizeInBytes If true, returns the size in bytes or else the size in number of components of the vertex attribute (default: false)\n * @returns the number of components\n */\n getSize(sizeInBytes = false) {\n return sizeInBytes ? this._size * VertexBuffer.GetTypeByteLength(this.type) : this._size;\n }\n /**\n * Gets a boolean indicating is the internal buffer of the VertexBuffer is instanced\n * @returns true if this buffer is instanced\n */\n getIsInstanced() {\n return this._instanced;\n }\n /**\n * Returns the instancing divisor, zero for non-instanced (integer).\n * @returns a number\n */\n getInstanceDivisor() {\n return this._instanceDivisor;\n }\n // Methods\n /**\n * Store data into the buffer. If the buffer was already used it will be either recreated or updated depending on isUpdatable property\n * @param data defines the data to store\n */\n create(data) {\n this._buffer.create(data);\n }\n /**\n * Updates the underlying buffer according to the passed numeric array or Float32Array.\n * This function will create a new buffer if the current one is not updatable\n * @param data defines the data to store\n */\n update(data) {\n this._buffer.update(data);\n }\n /**\n * Updates directly the underlying WebGLBuffer according to the passed numeric array or Float32Array.\n * Returns the directly updated WebGLBuffer.\n * @param data the new data\n * @param offset the new offset\n * @param useBytes set to true if the offset is in bytes\n */\n updateDirectly(data, offset, useBytes = false) {\n this._buffer.updateDirectly(data, offset, undefined, useBytes);\n }\n /**\n * Disposes the VertexBuffer and the underlying WebGLBuffer.\n */\n dispose() {\n if (this._ownsBuffer) {\n this._buffer.dispose();\n }\n }\n /**\n * Enumerates each value of this vertex buffer as numbers.\n * @param count the number of values to enumerate\n * @param callback the callback function called for each value\n */\n forEach(count, callback) {\n VertexBuffer.ForEach(this._buffer.getData(), this.byteOffset, this.byteStride, this._size, this.type, count, this.normalized, callback);\n }\n /**\n * Deduces the stride given a kind.\n * @param kind The kind string to deduce\n * @returns The deduced stride\n */\n static DeduceStride(kind) {\n switch (kind) {\n case VertexBuffer.UVKind:\n case VertexBuffer.UV2Kind:\n case VertexBuffer.UV3Kind:\n case VertexBuffer.UV4Kind:\n case VertexBuffer.UV5Kind:\n case VertexBuffer.UV6Kind:\n return 2;\n case VertexBuffer.NormalKind:\n case VertexBuffer.PositionKind:\n return 3;\n case VertexBuffer.ColorKind:\n case VertexBuffer.MatricesIndicesKind:\n case VertexBuffer.MatricesIndicesExtraKind:\n case VertexBuffer.MatricesWeightsKind:\n case VertexBuffer.MatricesWeightsExtraKind:\n case VertexBuffer.TangentKind:\n return 4;\n default:\n throw new Error(\"Invalid kind '\" + kind + \"'\");\n }\n }\n /**\n * Gets the byte length of the given type.\n * @param type the type\n * @returns the number of bytes\n */\n static GetTypeByteLength(type) {\n switch (type) {\n case VertexBuffer.BYTE:\n case VertexBuffer.UNSIGNED_BYTE:\n return 1;\n case VertexBuffer.SHORT:\n case VertexBuffer.UNSIGNED_SHORT:\n return 2;\n case VertexBuffer.INT:\n case VertexBuffer.UNSIGNED_INT:\n case VertexBuffer.FLOAT:\n return 4;\n default:\n throw new Error(`Invalid type '${type}'`);\n }\n }\n /**\n * Enumerates each value of the given parameters as numbers.\n * @param data the data to enumerate\n * @param byteOffset the byte offset of the data\n * @param byteStride the byte stride of the data\n * @param componentCount the number of components per element\n * @param componentType the type of the component\n * @param count the number of values to enumerate\n * @param normalized whether the data is normalized\n * @param callback the callback function called for each value\n */\n static ForEach(data, byteOffset, byteStride, componentCount, componentType, count, normalized, callback) {\n if (data instanceof Array) {\n let offset = byteOffset / 4;\n const stride = byteStride / 4;\n for (let index = 0; index < count; index += componentCount) {\n for (let componentIndex = 0; componentIndex < componentCount; componentIndex++) {\n callback(data[offset + componentIndex], index + componentIndex);\n }\n offset += stride;\n }\n }\n else {\n const dataView = data instanceof ArrayBuffer ? new DataView(data) : new DataView(data.buffer, data.byteOffset, data.byteLength);\n const componentByteLength = VertexBuffer.GetTypeByteLength(componentType);\n for (let index = 0; index < count; index += componentCount) {\n let componentByteOffset = byteOffset;\n for (let componentIndex = 0; componentIndex < componentCount; componentIndex++) {\n const value = VertexBuffer._GetFloatValue(dataView, componentType, componentByteOffset, normalized);\n callback(value, index + componentIndex);\n componentByteOffset += componentByteLength;\n }\n byteOffset += byteStride;\n }\n }\n }\n static _GetFloatValue(dataView, type, byteOffset, normalized) {\n switch (type) {\n case VertexBuffer.BYTE: {\n let value = dataView.getInt8(byteOffset);\n if (normalized) {\n value = Math.max(value / 127, -1);\n }\n return value;\n }\n case VertexBuffer.UNSIGNED_BYTE: {\n let value = dataView.getUint8(byteOffset);\n if (normalized) {\n value = value / 255;\n }\n return value;\n }\n case VertexBuffer.SHORT: {\n let value = dataView.getInt16(byteOffset, true);\n if (normalized) {\n value = Math.max(value / 32767, -1);\n }\n return value;\n }\n case VertexBuffer.UNSIGNED_SHORT: {\n let value = dataView.getUint16(byteOffset, true);\n if (normalized) {\n value = value / 65535;\n }\n return value;\n }\n case VertexBuffer.INT: {\n return dataView.getInt32(byteOffset, true);\n }\n case VertexBuffer.UNSIGNED_INT: {\n return dataView.getUint32(byteOffset, true);\n }\n case VertexBuffer.FLOAT: {\n return dataView.getFloat32(byteOffset, true);\n }\n default: {\n throw new Error(`Invalid component type ${type}`);\n }\n }\n }\n }\n VertexBuffer._Counter = 0;\n /**\n * The byte type.\n */\n VertexBuffer.BYTE = 5120;\n /**\n * The unsigned byte type.\n */\n VertexBuffer.UNSIGNED_BYTE = 5121;\n /**\n * The short type.\n */\n VertexBuffer.SHORT = 5122;\n /**\n * The unsigned short type.\n */\n VertexBuffer.UNSIGNED_SHORT = 5123;\n /**\n * The integer type.\n */\n VertexBuffer.INT = 5124;\n /**\n * The unsigned integer type.\n */\n VertexBuffer.UNSIGNED_INT = 5125;\n /**\n * The float type.\n */\n VertexBuffer.FLOAT = 5126;\n // Enums\n /**\n * Positions\n */\n VertexBuffer.PositionKind = \"position\";\n /**\n * Normals\n */\n VertexBuffer.NormalKind = \"normal\";\n /**\n * Tangents\n */\n VertexBuffer.TangentKind = \"tangent\";\n /**\n * Texture coordinates\n */\n VertexBuffer.UVKind = \"uv\";\n /**\n * Texture coordinates 2\n */\n VertexBuffer.UV2Kind = \"uv2\";\n /**\n * Texture coordinates 3\n */\n VertexBuffer.UV3Kind = \"uv3\";\n /**\n * Texture coordinates 4\n */\n VertexBuffer.UV4Kind = \"uv4\";\n /**\n * Texture coordinates 5\n */\n VertexBuffer.UV5Kind = \"uv5\";\n /**\n * Texture coordinates 6\n */\n VertexBuffer.UV6Kind = \"uv6\";\n /**\n * Colors\n */\n VertexBuffer.ColorKind = \"color\";\n /**\n * Instance Colors\n */\n VertexBuffer.ColorInstanceKind = \"instanceColor\";\n /**\n * Matrix indices (for bones)\n */\n VertexBuffer.MatricesIndicesKind = \"matricesIndices\";\n /**\n * Matrix weights (for bones)\n */\n VertexBuffer.MatricesWeightsKind = \"matricesWeights\";\n /**\n * Additional matrix indices (for bones)\n */\n VertexBuffer.MatricesIndicesExtraKind = \"matricesIndicesExtra\";\n /**\n * Additional matrix weights (for bones)\n */\n VertexBuffer.MatricesWeightsExtraKind = \"matricesWeightsExtra\";\n\n /**\n * Information about the result of picking within a scene\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/interactions/picking_collisions\n */\n class PickingInfo {\n constructor() {\n /**\n * If the pick collided with an object\n */\n this.hit = false;\n /**\n * Distance away where the pick collided\n */\n this.distance = 0;\n /**\n * The location of pick collision\n */\n this.pickedPoint = null;\n /**\n * The mesh corresponding the pick collision\n */\n this.pickedMesh = null;\n /** (See getTextureCoordinates) The barycentric U coordinate that is used when calculating the texture coordinates of the collision.*/\n this.bu = 0;\n /** (See getTextureCoordinates) The barycentric V coordinate that is used when calculating the texture coordinates of the collision.*/\n this.bv = 0;\n /** The index of the face on the mesh that was picked, or the index of the Line if the picked Mesh is a LinesMesh */\n this.faceId = -1;\n /** The index of the face on the subMesh that was picked, or the index of the Line if the picked Mesh is a LinesMesh */\n this.subMeshFaceId = -1;\n /** Id of the submesh that was picked */\n this.subMeshId = 0;\n /** If a sprite was picked, this will be the sprite the pick collided with */\n this.pickedSprite = null;\n /** If we are picking a mesh with thin instance, this will give you the picked thin instance */\n this.thinInstanceIndex = -1;\n /**\n * The ray that was used to perform the picking.\n */\n this.ray = null;\n /**\n * If a mesh was used to do the picking (eg. 6dof controller) as a \"near interaction\", this will be populated.\n */\n this.originMesh = null;\n /**\n * The aim-space transform of the input used for picking, if it is an XR input source.\n */\n this.aimTransform = null;\n /**\n * The grip-space transform of the input used for picking, if it is an XR input source.\n * Some XR sources, such as input coming from head mounted displays, do not have this.\n */\n this.gripTransform = null;\n }\n /**\n * Gets the normal corresponding to the face the pick collided with\n * @param useWorldCoordinates If the resulting normal should be relative to the world (default: false)\n * @param useVerticesNormals If the vertices normals should be used to calculate the normal instead of the normal map (default: true)\n * @returns The normal corresponding to the face the pick collided with\n * @remarks Note that the returned normal will always point towards the picking ray.\n */\n getNormal(useWorldCoordinates = false, useVerticesNormals = true) {\n if (!this.pickedMesh || (useVerticesNormals && !this.pickedMesh.isVerticesDataPresent(VertexBuffer.NormalKind))) {\n return null;\n }\n let indices = this.pickedMesh.getIndices();\n if ((indices === null || indices === void 0 ? void 0 : indices.length) === 0) {\n indices = null;\n }\n let result;\n const tmp0 = TmpVectors.Vector3[0];\n const tmp1 = TmpVectors.Vector3[1];\n const tmp2 = TmpVectors.Vector3[2];\n if (useVerticesNormals) {\n const normals = this.pickedMesh.getVerticesData(VertexBuffer.NormalKind);\n let normal0 = indices\n ? Vector3.FromArrayToRef(normals, indices[this.faceId * 3] * 3, tmp0)\n : tmp0.copyFromFloats(normals[this.faceId * 3 * 3], normals[this.faceId * 3 * 3 + 1], normals[this.faceId * 3 * 3 + 2]);\n let normal1 = indices\n ? Vector3.FromArrayToRef(normals, indices[this.faceId * 3 + 1] * 3, tmp1)\n : tmp1.copyFromFloats(normals[(this.faceId * 3 + 1) * 3], normals[(this.faceId * 3 + 1) * 3 + 1], normals[(this.faceId * 3 + 1) * 3 + 2]);\n let normal2 = indices\n ? Vector3.FromArrayToRef(normals, indices[this.faceId * 3 + 2] * 3, tmp2)\n : tmp2.copyFromFloats(normals[(this.faceId * 3 + 2) * 3], normals[(this.faceId * 3 + 2) * 3 + 1], normals[(this.faceId * 3 + 2) * 3 + 2]);\n normal0 = normal0.scale(this.bu);\n normal1 = normal1.scale(this.bv);\n normal2 = normal2.scale(1.0 - this.bu - this.bv);\n result = new Vector3(normal0.x + normal1.x + normal2.x, normal0.y + normal1.y + normal2.y, normal0.z + normal1.z + normal2.z);\n }\n else {\n const positions = this.pickedMesh.getVerticesData(VertexBuffer.PositionKind);\n const vertex1 = indices\n ? Vector3.FromArrayToRef(positions, indices[this.faceId * 3] * 3, tmp0)\n : tmp0.copyFromFloats(positions[this.faceId * 3 * 3], positions[this.faceId * 3 * 3 + 1], positions[this.faceId * 3 * 3 + 2]);\n const vertex2 = indices\n ? Vector3.FromArrayToRef(positions, indices[this.faceId * 3 + 1] * 3, tmp1)\n : tmp1.copyFromFloats(positions[(this.faceId * 3 + 1) * 3], positions[(this.faceId * 3 + 1) * 3 + 1], positions[(this.faceId * 3 + 1) * 3 + 2]);\n const vertex3 = indices\n ? Vector3.FromArrayToRef(positions, indices[this.faceId * 3 + 2] * 3, tmp2)\n : tmp2.copyFromFloats(positions[(this.faceId * 3 + 2) * 3], positions[(this.faceId * 3 + 2) * 3 + 1], positions[(this.faceId * 3 + 2) * 3 + 2]);\n const p1p2 = vertex1.subtract(vertex2);\n const p3p2 = vertex3.subtract(vertex2);\n result = Vector3.Cross(p1p2, p3p2);\n }\n const transformNormalToWorld = (pickedMesh, n) => {\n let wm = pickedMesh.getWorldMatrix();\n if (pickedMesh.nonUniformScaling) {\n TmpVectors.Matrix[0].copyFrom(wm);\n wm = TmpVectors.Matrix[0];\n wm.setTranslationFromFloats(0, 0, 0);\n wm.invert();\n wm.transposeToRef(TmpVectors.Matrix[1]);\n wm = TmpVectors.Matrix[1];\n }\n Vector3.TransformNormalToRef(n, wm, n);\n };\n if (useWorldCoordinates) {\n transformNormalToWorld(this.pickedMesh, result);\n }\n if (this.ray) {\n const normalForDirectionChecking = TmpVectors.Vector3[0].copyFrom(result);\n if (!useWorldCoordinates) {\n // the normal has not been transformed to world space as part as the normal processing, so we must do it now\n transformNormalToWorld(this.pickedMesh, normalForDirectionChecking);\n }\n // Flip the normal if the picking ray is in the same direction.\n if (Vector3.Dot(normalForDirectionChecking, this.ray.direction) > 0) {\n result.negateInPlace();\n }\n }\n result.normalize();\n return result;\n }\n /**\n * Gets the texture coordinates of where the pick occurred\n * @param uvSet The UV set to use to calculate the texture coordinates (default: VertexBuffer.UVKind)\n * @returns The vector containing the coordinates of the texture\n */\n getTextureCoordinates(uvSet = VertexBuffer.UVKind) {\n if (!this.pickedMesh || !this.pickedMesh.isVerticesDataPresent(uvSet)) {\n return null;\n }\n const indices = this.pickedMesh.getIndices();\n if (!indices) {\n return null;\n }\n const uvs = this.pickedMesh.getVerticesData(uvSet);\n if (!uvs) {\n return null;\n }\n let uv0 = Vector2.FromArray(uvs, indices[this.faceId * 3] * 2);\n let uv1 = Vector2.FromArray(uvs, indices[this.faceId * 3 + 1] * 2);\n let uv2 = Vector2.FromArray(uvs, indices[this.faceId * 3 + 2] * 2);\n uv0 = uv0.scale(this.bu);\n uv1 = uv1.scale(this.bv);\n uv2 = uv2.scale(1.0 - this.bu - this.bv);\n return new Vector2(uv0.x + uv1.x + uv2.x, uv0.y + uv1.y + uv2.y);\n }\n }\n\n /**\n * ActionEvent is the event being sent when an action is triggered.\n */\n class ActionEvent {\n /**\n * Creates a new ActionEvent\n * @param source The mesh or sprite that triggered the action\n * @param pointerX The X mouse cursor position at the time of the event\n * @param pointerY The Y mouse cursor position at the time of the event\n * @param meshUnderPointer The mesh that is currently pointed at (can be null)\n * @param sourceEvent the original (browser) event that triggered the ActionEvent\n * @param additionalData additional data for the event\n */\n constructor(\n /** The mesh or sprite that triggered the action */\n source, \n /** The X mouse cursor position at the time of the event */\n pointerX, \n /** The Y mouse cursor position at the time of the event */\n pointerY, \n /** The mesh that is currently pointed at (can be null) */\n meshUnderPointer, \n /** the original (browser) event that triggered the ActionEvent */\n sourceEvent, \n /** additional data for the event */\n additionalData) {\n this.source = source;\n this.pointerX = pointerX;\n this.pointerY = pointerY;\n this.meshUnderPointer = meshUnderPointer;\n this.sourceEvent = sourceEvent;\n this.additionalData = additionalData;\n }\n /**\n * Helper function to auto-create an ActionEvent from a source mesh.\n * @param source The source mesh that triggered the event\n * @param evt The original (browser) event\n * @param additionalData additional data for the event\n * @returns the new ActionEvent\n */\n static CreateNew(source, evt, additionalData) {\n const scene = source.getScene();\n return new ActionEvent(source, scene.pointerX, scene.pointerY, scene.meshUnderPointer || source, evt, additionalData);\n }\n /**\n * Helper function to auto-create an ActionEvent from a source sprite\n * @param source The source sprite that triggered the event\n * @param scene Scene associated with the sprite\n * @param evt The original (browser) event\n * @param additionalData additional data for the event\n * @returns the new ActionEvent\n */\n static CreateNewFromSprite(source, scene, evt, additionalData) {\n return new ActionEvent(source, scene.pointerX, scene.pointerY, scene.meshUnderPointer, evt, additionalData);\n }\n /**\n * Helper function to auto-create an ActionEvent from a scene. If triggered by a mesh use ActionEvent.CreateNew\n * @param scene the scene where the event occurred\n * @param evt The original (browser) event\n * @returns the new ActionEvent\n */\n static CreateNewFromScene(scene, evt) {\n return new ActionEvent(null, scene.pointerX, scene.pointerY, scene.meshUnderPointer, evt);\n }\n /**\n * Helper function to auto-create an ActionEvent from a primitive\n * @param prim defines the target primitive\n * @param pointerPos defines the pointer position\n * @param evt The original (browser) event\n * @param additionalData additional data for the event\n * @returns the new ActionEvent\n */\n static CreateNewFromPrimitive(prim, pointerPos, evt, additionalData) {\n return new ActionEvent(prim, pointerPos.x, pointerPos.y, null, evt, additionalData);\n }\n }\n\n /**\n * PostProcessManager is used to manage one or more post processes or post process pipelines\n * See https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/usePostProcesses\n */\n class PostProcessManager {\n /**\n * Creates a new instance PostProcess\n * @param scene The scene that the post process is associated with.\n */\n constructor(scene) {\n this._vertexBuffers = {};\n this._scene = scene;\n }\n _prepareBuffers() {\n if (this._vertexBuffers[VertexBuffer.PositionKind]) {\n return;\n }\n // VBO\n const vertices = [];\n vertices.push(1, 1);\n vertices.push(-1, 1);\n vertices.push(-1, -1);\n vertices.push(1, -1);\n this._vertexBuffers[VertexBuffer.PositionKind] = new VertexBuffer(this._scene.getEngine(), vertices, VertexBuffer.PositionKind, false, false, 2);\n this._buildIndexBuffer();\n }\n _buildIndexBuffer() {\n // Indices\n const indices = [];\n indices.push(0);\n indices.push(1);\n indices.push(2);\n indices.push(0);\n indices.push(2);\n indices.push(3);\n this._indexBuffer = this._scene.getEngine().createIndexBuffer(indices);\n }\n /**\n * Rebuilds the vertex buffers of the manager.\n * @internal\n */\n _rebuild() {\n const vb = this._vertexBuffers[VertexBuffer.PositionKind];\n if (!vb) {\n return;\n }\n vb._rebuild();\n this._buildIndexBuffer();\n }\n // Methods\n /**\n * Prepares a frame to be run through a post process.\n * @param sourceTexture The input texture to the post processes. (default: null)\n * @param postProcesses An array of post processes to be run. (default: null)\n * @returns True if the post processes were able to be run.\n * @internal\n */\n _prepareFrame(sourceTexture = null, postProcesses = null) {\n const camera = this._scene.activeCamera;\n if (!camera) {\n return false;\n }\n postProcesses = postProcesses || camera._postProcesses.filter((pp) => {\n return pp != null;\n });\n if (!postProcesses || postProcesses.length === 0 || !this._scene.postProcessesEnabled) {\n return false;\n }\n postProcesses[0].activate(camera, sourceTexture, postProcesses !== null && postProcesses !== undefined);\n return true;\n }\n /**\n * Manually render a set of post processes to a texture.\n * Please note, the frame buffer won't be unbound after the call in case you have more render to do.\n * @param postProcesses An array of post processes to be run.\n * @param targetTexture The render target wrapper to render to.\n * @param forceFullscreenViewport force gl.viewport to be full screen eg. 0,0,textureWidth,textureHeight\n * @param faceIndex defines the face to render to if a cubemap is defined as the target\n * @param lodLevel defines which lod of the texture to render to\n * @param doNotBindFrambuffer If set to true, assumes that the framebuffer has been bound previously\n */\n directRender(postProcesses, targetTexture = null, forceFullscreenViewport = false, faceIndex = 0, lodLevel = 0, doNotBindFrambuffer = false) {\n var _a;\n const engine = this._scene.getEngine();\n for (let index = 0; index < postProcesses.length; index++) {\n if (index < postProcesses.length - 1) {\n postProcesses[index + 1].activate(this._scene.activeCamera, targetTexture === null || targetTexture === void 0 ? void 0 : targetTexture.texture);\n }\n else {\n if (targetTexture) {\n engine.bindFramebuffer(targetTexture, faceIndex, undefined, undefined, forceFullscreenViewport, lodLevel);\n }\n else if (!doNotBindFrambuffer) {\n engine.restoreDefaultFramebuffer();\n }\n (_a = engine._debugInsertMarker) === null || _a === void 0 ? void 0 : _a.call(engine, `post process ${postProcesses[index].name} output`);\n }\n const pp = postProcesses[index];\n const effect = pp.apply();\n if (effect) {\n pp.onBeforeRenderObservable.notifyObservers(effect);\n // VBOs\n this._prepareBuffers();\n engine.bindBuffers(this._vertexBuffers, this._indexBuffer, effect);\n // Draw order\n engine.drawElementsType(0, 0, 6);\n pp.onAfterRenderObservable.notifyObservers(effect);\n }\n }\n // Restore depth buffer\n engine.setDepthBuffer(true);\n engine.setDepthWrite(true);\n }\n /**\n * Finalize the result of the output of the postprocesses.\n * @param doNotPresent If true the result will not be displayed to the screen.\n * @param targetTexture The render target wrapper to render to.\n * @param faceIndex The index of the face to bind the target texture to.\n * @param postProcesses The array of post processes to render.\n * @param forceFullscreenViewport force gl.viewport to be full screen eg. 0,0,textureWidth,textureHeight (default: false)\n * @internal\n */\n _finalizeFrame(doNotPresent, targetTexture, faceIndex, postProcesses, forceFullscreenViewport = false) {\n var _a;\n const camera = this._scene.activeCamera;\n if (!camera) {\n return;\n }\n postProcesses = postProcesses || camera._postProcesses.filter((pp) => {\n return pp != null;\n });\n if (postProcesses.length === 0 || !this._scene.postProcessesEnabled) {\n return;\n }\n const engine = this._scene.getEngine();\n for (let index = 0, len = postProcesses.length; index < len; index++) {\n const pp = postProcesses[index];\n if (index < len - 1) {\n pp._outputTexture = postProcesses[index + 1].activate(camera, targetTexture === null || targetTexture === void 0 ? void 0 : targetTexture.texture);\n }\n else {\n if (targetTexture) {\n engine.bindFramebuffer(targetTexture, faceIndex, undefined, undefined, forceFullscreenViewport);\n pp._outputTexture = targetTexture;\n }\n else {\n engine.restoreDefaultFramebuffer();\n pp._outputTexture = null;\n }\n (_a = engine._debugInsertMarker) === null || _a === void 0 ? void 0 : _a.call(engine, `post process ${postProcesses[index].name} output`);\n }\n if (doNotPresent) {\n break;\n }\n const effect = pp.apply();\n if (effect) {\n pp.onBeforeRenderObservable.notifyObservers(effect);\n // VBOs\n this._prepareBuffers();\n engine.bindBuffers(this._vertexBuffers, this._indexBuffer, effect);\n // Draw order\n engine.drawElementsType(0, 0, 6);\n pp.onAfterRenderObservable.notifyObservers(effect);\n }\n }\n // Restore states\n engine.setDepthBuffer(true);\n engine.setDepthWrite(true);\n engine.setAlphaMode(0);\n }\n /**\n * Disposes of the post process manager.\n */\n dispose() {\n const buffer = this._vertexBuffers[VertexBuffer.PositionKind];\n if (buffer) {\n buffer.dispose();\n this._vertexBuffers[VertexBuffer.PositionKind] = null;\n }\n if (this._indexBuffer) {\n this._scene.getEngine()._releaseBuffer(this._indexBuffer);\n this._indexBuffer = null;\n }\n }\n }\n\n /**\n * This represents the object necessary to create a rendering group.\n * This is exclusively used and created by the rendering manager.\n * To modify the behavior, you use the available helpers in your scene or meshes.\n * @internal\n */\n class RenderingGroup {\n /**\n * Set the opaque sort comparison function.\n * If null the sub meshes will be render in the order they were created\n */\n set opaqueSortCompareFn(value) {\n if (value) {\n this._opaqueSortCompareFn = value;\n }\n else {\n this._opaqueSortCompareFn = RenderingGroup.PainterSortCompare;\n }\n this._renderOpaque = this._renderOpaqueSorted;\n }\n /**\n * Set the alpha test sort comparison function.\n * If null the sub meshes will be render in the order they were created\n */\n set alphaTestSortCompareFn(value) {\n if (value) {\n this._alphaTestSortCompareFn = value;\n }\n else {\n this._alphaTestSortCompareFn = RenderingGroup.PainterSortCompare;\n }\n this._renderAlphaTest = this._renderAlphaTestSorted;\n }\n /**\n * Set the transparent sort comparison function.\n * If null the sub meshes will be render in the order they were created\n */\n set transparentSortCompareFn(value) {\n if (value) {\n this._transparentSortCompareFn = value;\n }\n else {\n this._transparentSortCompareFn = RenderingGroup.defaultTransparentSortCompare;\n }\n this._renderTransparent = this._renderTransparentSorted;\n }\n /**\n * Creates a new rendering group.\n * @param index The rendering group index\n * @param scene\n * @param opaqueSortCompareFn The opaque sort comparison function. If null no order is applied\n * @param alphaTestSortCompareFn The alpha test sort comparison function. If null no order is applied\n * @param transparentSortCompareFn The transparent sort comparison function. If null back to front + alpha index sort is applied\n */\n constructor(index, scene, opaqueSortCompareFn = null, alphaTestSortCompareFn = null, transparentSortCompareFn = null) {\n this.index = index;\n this._opaqueSubMeshes = new SmartArray(256);\n this._transparentSubMeshes = new SmartArray(256);\n this._alphaTestSubMeshes = new SmartArray(256);\n this._depthOnlySubMeshes = new SmartArray(256);\n this._particleSystems = new SmartArray(256);\n this._spriteManagers = new SmartArray(256);\n /** @internal */\n this._empty = true;\n /** @internal */\n this._edgesRenderers = new SmartArrayNoDuplicate(16);\n this._scene = scene;\n this.opaqueSortCompareFn = opaqueSortCompareFn;\n this.alphaTestSortCompareFn = alphaTestSortCompareFn;\n this.transparentSortCompareFn = transparentSortCompareFn;\n }\n /**\n * Render all the sub meshes contained in the group.\n * @param customRenderFunction Used to override the default render behaviour of the group.\n * @param renderSprites\n * @param renderParticles\n * @param activeMeshes\n * @returns true if rendered some submeshes.\n */\n render(customRenderFunction, renderSprites, renderParticles, activeMeshes) {\n if (customRenderFunction) {\n customRenderFunction(this._opaqueSubMeshes, this._alphaTestSubMeshes, this._transparentSubMeshes, this._depthOnlySubMeshes);\n return;\n }\n const engine = this._scene.getEngine();\n // Depth only\n if (this._depthOnlySubMeshes.length !== 0) {\n engine.setColorWrite(false);\n this._renderAlphaTest(this._depthOnlySubMeshes);\n engine.setColorWrite(true);\n }\n // Opaque\n if (this._opaqueSubMeshes.length !== 0) {\n this._renderOpaque(this._opaqueSubMeshes);\n }\n // Alpha test\n if (this._alphaTestSubMeshes.length !== 0) {\n this._renderAlphaTest(this._alphaTestSubMeshes);\n }\n const stencilState = engine.getStencilBuffer();\n engine.setStencilBuffer(false);\n // Sprites\n if (renderSprites) {\n this._renderSprites();\n }\n // Particles\n if (renderParticles) {\n this._renderParticles(activeMeshes);\n }\n if (this.onBeforeTransparentRendering) {\n this.onBeforeTransparentRendering();\n }\n // Transparent\n if (this._transparentSubMeshes.length !== 0 || this._scene.useOrderIndependentTransparency) {\n engine.setStencilBuffer(stencilState);\n if (this._scene.useOrderIndependentTransparency) {\n const excludedMeshes = this._scene.depthPeelingRenderer.render(this._transparentSubMeshes);\n if (excludedMeshes.length) {\n // Render leftover meshes that could not be processed by depth peeling\n this._renderTransparent(excludedMeshes);\n }\n }\n else {\n this._renderTransparent(this._transparentSubMeshes);\n }\n engine.setAlphaMode(0);\n }\n // Set back stencil to false in case it changes before the edge renderer.\n engine.setStencilBuffer(false);\n // Edges\n if (this._edgesRenderers.length) {\n for (let edgesRendererIndex = 0; edgesRendererIndex < this._edgesRenderers.length; edgesRendererIndex++) {\n this._edgesRenderers.data[edgesRendererIndex].render();\n }\n engine.setAlphaMode(0);\n }\n // Restore Stencil state.\n engine.setStencilBuffer(stencilState);\n }\n /**\n * Renders the opaque submeshes in the order from the opaqueSortCompareFn.\n * @param subMeshes The submeshes to render\n */\n _renderOpaqueSorted(subMeshes) {\n return RenderingGroup._RenderSorted(subMeshes, this._opaqueSortCompareFn, this._scene.activeCamera, false);\n }\n /**\n * Renders the opaque submeshes in the order from the alphatestSortCompareFn.\n * @param subMeshes The submeshes to render\n */\n _renderAlphaTestSorted(subMeshes) {\n return RenderingGroup._RenderSorted(subMeshes, this._alphaTestSortCompareFn, this._scene.activeCamera, false);\n }\n /**\n * Renders the opaque submeshes in the order from the transparentSortCompareFn.\n * @param subMeshes The submeshes to render\n */\n _renderTransparentSorted(subMeshes) {\n return RenderingGroup._RenderSorted(subMeshes, this._transparentSortCompareFn, this._scene.activeCamera, true);\n }\n /**\n * Renders the submeshes in a specified order.\n * @param subMeshes The submeshes to sort before render\n * @param sortCompareFn The comparison function use to sort\n * @param camera The camera position use to preprocess the submeshes to help sorting\n * @param transparent Specifies to activate blending if true\n */\n static _RenderSorted(subMeshes, sortCompareFn, camera, transparent) {\n let subIndex = 0;\n let subMesh;\n const cameraPosition = camera ? camera.globalPosition : RenderingGroup._ZeroVector;\n if (transparent) {\n for (; subIndex < subMeshes.length; subIndex++) {\n subMesh = subMeshes.data[subIndex];\n subMesh._alphaIndex = subMesh.getMesh().alphaIndex;\n subMesh._distanceToCamera = Vector3.Distance(subMesh.getBoundingInfo().boundingSphere.centerWorld, cameraPosition);\n }\n }\n const sortedArray = subMeshes.length === subMeshes.data.length ? subMeshes.data : subMeshes.data.slice(0, subMeshes.length);\n if (sortCompareFn) {\n sortedArray.sort(sortCompareFn);\n }\n const scene = sortedArray[0].getMesh().getScene();\n for (subIndex = 0; subIndex < sortedArray.length; subIndex++) {\n subMesh = sortedArray[subIndex];\n if (scene._activeMeshesFrozenButKeepClipping && !subMesh.isInFrustum(scene._frustumPlanes)) {\n continue;\n }\n if (transparent) {\n const material = subMesh.getMaterial();\n if (material && material.needDepthPrePass) {\n const engine = material.getScene().getEngine();\n engine.setColorWrite(false);\n engine.setAlphaMode(0);\n subMesh.render(false);\n engine.setColorWrite(true);\n }\n }\n subMesh.render(transparent);\n }\n }\n /**\n * Build in function which can be applied to ensure meshes of a special queue (opaque, alpha test, transparent)\n * are rendered back to front if in the same alpha index.\n *\n * @param a The first submesh\n * @param b The second submesh\n * @returns The result of the comparison\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n static defaultTransparentSortCompare(a, b) {\n // Alpha index first\n if (a._alphaIndex > b._alphaIndex) {\n return 1;\n }\n if (a._alphaIndex < b._alphaIndex) {\n return -1;\n }\n // Then distance to camera\n return RenderingGroup.backToFrontSortCompare(a, b);\n }\n /**\n * Build in function which can be applied to ensure meshes of a special queue (opaque, alpha test, transparent)\n * are rendered back to front.\n *\n * @param a The first submesh\n * @param b The second submesh\n * @returns The result of the comparison\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n static backToFrontSortCompare(a, b) {\n // Then distance to camera\n if (a._distanceToCamera < b._distanceToCamera) {\n return 1;\n }\n if (a._distanceToCamera > b._distanceToCamera) {\n return -1;\n }\n return 0;\n }\n /**\n * Build in function which can be applied to ensure meshes of a special queue (opaque, alpha test, transparent)\n * are rendered front to back (prevent overdraw).\n *\n * @param a The first submesh\n * @param b The second submesh\n * @returns The result of the comparison\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n static frontToBackSortCompare(a, b) {\n // Then distance to camera\n if (a._distanceToCamera < b._distanceToCamera) {\n return -1;\n }\n if (a._distanceToCamera > b._distanceToCamera) {\n return 1;\n }\n return 0;\n }\n /**\n * Build in function which can be applied to ensure meshes of a special queue (opaque, alpha test, transparent)\n * are grouped by material then geometry.\n *\n * @param a The first submesh\n * @param b The second submesh\n * @returns The result of the comparison\n */\n static PainterSortCompare(a, b) {\n const meshA = a.getMesh();\n const meshB = b.getMesh();\n if (meshA.material && meshB.material) {\n return meshA.material.uniqueId - meshB.material.uniqueId;\n }\n return meshA.uniqueId - meshB.uniqueId;\n }\n /**\n * Resets the different lists of submeshes to prepare a new frame.\n */\n prepare() {\n this._opaqueSubMeshes.reset();\n this._transparentSubMeshes.reset();\n this._alphaTestSubMeshes.reset();\n this._depthOnlySubMeshes.reset();\n this._particleSystems.reset();\n this.prepareSprites();\n this._edgesRenderers.reset();\n this._empty = true;\n }\n /**\n * Resets the different lists of sprites to prepare a new frame.\n */\n prepareSprites() {\n this._spriteManagers.reset();\n }\n dispose() {\n this._opaqueSubMeshes.dispose();\n this._transparentSubMeshes.dispose();\n this._alphaTestSubMeshes.dispose();\n this._depthOnlySubMeshes.dispose();\n this._particleSystems.dispose();\n this._spriteManagers.dispose();\n this._edgesRenderers.dispose();\n }\n /**\n * Inserts the submesh in its correct queue depending on its material.\n * @param subMesh The submesh to dispatch\n * @param [mesh] Optional reference to the submeshes's mesh. Provide if you have an exiting reference to improve performance.\n * @param [material] Optional reference to the submeshes's material. Provide if you have an exiting reference to improve performance.\n */\n dispatch(subMesh, mesh, material) {\n // Get mesh and materials if not provided\n if (mesh === undefined) {\n mesh = subMesh.getMesh();\n }\n if (material === undefined) {\n material = subMesh.getMaterial();\n }\n if (material === null || material === undefined) {\n return;\n }\n if (material.needAlphaBlendingForMesh(mesh)) {\n // Transparent\n this._transparentSubMeshes.push(subMesh);\n }\n else if (material.needAlphaTesting()) {\n // Alpha test\n if (material.needDepthPrePass) {\n this._depthOnlySubMeshes.push(subMesh);\n }\n this._alphaTestSubMeshes.push(subMesh);\n }\n else {\n if (material.needDepthPrePass) {\n this._depthOnlySubMeshes.push(subMesh);\n }\n this._opaqueSubMeshes.push(subMesh); // Opaque\n }\n mesh._renderingGroup = this;\n if (mesh._edgesRenderer && mesh._edgesRenderer.isEnabled) {\n this._edgesRenderers.pushNoDuplicate(mesh._edgesRenderer);\n }\n this._empty = false;\n }\n dispatchSprites(spriteManager) {\n this._spriteManagers.push(spriteManager);\n this._empty = false;\n }\n dispatchParticles(particleSystem) {\n this._particleSystems.push(particleSystem);\n this._empty = false;\n }\n _renderParticles(activeMeshes) {\n if (this._particleSystems.length === 0) {\n return;\n }\n // Particles\n const activeCamera = this._scene.activeCamera;\n this._scene.onBeforeParticlesRenderingObservable.notifyObservers(this._scene);\n for (let particleIndex = 0; particleIndex < this._particleSystems.length; particleIndex++) {\n const particleSystem = this._particleSystems.data[particleIndex];\n if ((activeCamera && activeCamera.layerMask & particleSystem.layerMask) === 0) {\n continue;\n }\n const emitter = particleSystem.emitter;\n if (!emitter.position || !activeMeshes || activeMeshes.indexOf(emitter) !== -1) {\n this._scene._activeParticles.addCount(particleSystem.render(), false);\n }\n }\n this._scene.onAfterParticlesRenderingObservable.notifyObservers(this._scene);\n }\n _renderSprites() {\n if (!this._scene.spritesEnabled || this._spriteManagers.length === 0) {\n return;\n }\n // Sprites\n const activeCamera = this._scene.activeCamera;\n this._scene.onBeforeSpritesRenderingObservable.notifyObservers(this._scene);\n for (let id = 0; id < this._spriteManagers.length; id++) {\n const spriteManager = this._spriteManagers.data[id];\n if ((activeCamera && activeCamera.layerMask & spriteManager.layerMask) !== 0) {\n spriteManager.render();\n }\n }\n this._scene.onAfterSpritesRenderingObservable.notifyObservers(this._scene);\n }\n }\n RenderingGroup._ZeroVector = Vector3.Zero();\n\n /**\n * This class is used by the onRenderingGroupObservable\n */\n class RenderingGroupInfo {\n }\n /**\n * This is the manager responsible of all the rendering for meshes sprites and particles.\n * It is enable to manage the different groups as well as the different necessary sort functions.\n * This should not be used directly aside of the few static configurations\n */\n class RenderingManager {\n /**\n * Gets or sets a boolean indicating that the manager will not reset between frames.\n * This means that if a mesh becomes invisible or transparent it will not be visible until this boolean is set to false again.\n * By default, the rendering manager will dispatch all active meshes per frame (moving them to the transparent, opaque or alpha testing lists).\n * By turning this property on, you will accelerate the rendering by keeping all these lists unchanged between frames.\n */\n get maintainStateBetweenFrames() {\n return this._maintainStateBetweenFrames;\n }\n set maintainStateBetweenFrames(value) {\n if (value === this._maintainStateBetweenFrames) {\n return;\n }\n this._maintainStateBetweenFrames = value;\n // Restore wasDispatched flags when switching to maintainStateBetweenFrames to false\n if (!this._maintainStateBetweenFrames) {\n for (const mesh of this._scene.meshes) {\n if (mesh.subMeshes) {\n for (const subMesh of mesh.subMeshes) {\n subMesh._wasDispatched = false;\n }\n }\n }\n if (this._scene.spriteManagers) {\n for (const spriteManager of this._scene.spriteManagers) {\n spriteManager._wasDispatched = false;\n }\n }\n for (const particleSystem of this._scene.particleSystems) {\n particleSystem._wasDispatched = false;\n }\n }\n }\n /**\n * Instantiates a new rendering group for a particular scene\n * @param scene Defines the scene the groups belongs to\n */\n constructor(scene) {\n /**\n * @internal\n */\n this._useSceneAutoClearSetup = false;\n this._renderingGroups = new Array();\n this._autoClearDepthStencil = {};\n this._customOpaqueSortCompareFn = {};\n this._customAlphaTestSortCompareFn = {};\n this._customTransparentSortCompareFn = {};\n this._renderingGroupInfo = new RenderingGroupInfo();\n this._maintainStateBetweenFrames = false;\n this._scene = scene;\n for (let i = RenderingManager.MIN_RENDERINGGROUPS; i < RenderingManager.MAX_RENDERINGGROUPS; i++) {\n this._autoClearDepthStencil[i] = { autoClear: true, depth: true, stencil: true };\n }\n }\n /**\n * Gets the rendering group with the specified id.\n */\n getRenderingGroup(id) {\n const renderingGroupId = id || 0;\n this._prepareRenderingGroup(renderingGroupId);\n return this._renderingGroups[renderingGroupId];\n }\n _clearDepthStencilBuffer(depth = true, stencil = true) {\n if (this._depthStencilBufferAlreadyCleaned) {\n return;\n }\n this._scene.getEngine().clear(null, false, depth, stencil);\n this._depthStencilBufferAlreadyCleaned = true;\n }\n /**\n * Renders the entire managed groups. This is used by the scene or the different render targets.\n * @internal\n */\n render(customRenderFunction, activeMeshes, renderParticles, renderSprites) {\n // Update the observable context (not null as it only goes away on dispose)\n const info = this._renderingGroupInfo;\n info.scene = this._scene;\n info.camera = this._scene.activeCamera;\n // Dispatch sprites\n if (this._scene.spriteManagers && renderSprites) {\n for (let index = 0; index < this._scene.spriteManagers.length; index++) {\n const manager = this._scene.spriteManagers[index];\n this.dispatchSprites(manager);\n }\n }\n // Render\n for (let index = RenderingManager.MIN_RENDERINGGROUPS; index < RenderingManager.MAX_RENDERINGGROUPS; index++) {\n this._depthStencilBufferAlreadyCleaned = index === RenderingManager.MIN_RENDERINGGROUPS;\n const renderingGroup = this._renderingGroups[index];\n if (!renderingGroup || renderingGroup._empty) {\n continue;\n }\n const renderingGroupMask = Math.pow(2, index);\n info.renderingGroupId = index;\n // Before Observable\n this._scene.onBeforeRenderingGroupObservable.notifyObservers(info, renderingGroupMask);\n // Clear depth/stencil if needed\n if (RenderingManager.AUTOCLEAR) {\n const autoClear = this._useSceneAutoClearSetup ? this._scene.getAutoClearDepthStencilSetup(index) : this._autoClearDepthStencil[index];\n if (autoClear && autoClear.autoClear) {\n this._clearDepthStencilBuffer(autoClear.depth, autoClear.stencil);\n }\n }\n // Render\n for (const step of this._scene._beforeRenderingGroupDrawStage) {\n step.action(index);\n }\n renderingGroup.render(customRenderFunction, renderSprites, renderParticles, activeMeshes);\n for (const step of this._scene._afterRenderingGroupDrawStage) {\n step.action(index);\n }\n // After Observable\n this._scene.onAfterRenderingGroupObservable.notifyObservers(info, renderingGroupMask);\n }\n }\n /**\n * Resets the different information of the group to prepare a new frame\n * @internal\n */\n reset() {\n if (this.maintainStateBetweenFrames) {\n return;\n }\n for (let index = RenderingManager.MIN_RENDERINGGROUPS; index < RenderingManager.MAX_RENDERINGGROUPS; index++) {\n const renderingGroup = this._renderingGroups[index];\n if (renderingGroup) {\n renderingGroup.prepare();\n }\n }\n }\n /**\n * Resets the sprites information of the group to prepare a new frame\n * @internal\n */\n resetSprites() {\n if (this.maintainStateBetweenFrames) {\n return;\n }\n for (let index = RenderingManager.MIN_RENDERINGGROUPS; index < RenderingManager.MAX_RENDERINGGROUPS; index++) {\n const renderingGroup = this._renderingGroups[index];\n if (renderingGroup) {\n renderingGroup.prepareSprites();\n }\n }\n }\n /**\n * Dispose and release the group and its associated resources.\n * @internal\n */\n dispose() {\n this.freeRenderingGroups();\n this._renderingGroups.length = 0;\n this._renderingGroupInfo = null;\n }\n /**\n * Clear the info related to rendering groups preventing retention points during dispose.\n */\n freeRenderingGroups() {\n for (let index = RenderingManager.MIN_RENDERINGGROUPS; index < RenderingManager.MAX_RENDERINGGROUPS; index++) {\n const renderingGroup = this._renderingGroups[index];\n if (renderingGroup) {\n renderingGroup.dispose();\n }\n }\n }\n _prepareRenderingGroup(renderingGroupId) {\n if (this._renderingGroups[renderingGroupId] === undefined) {\n this._renderingGroups[renderingGroupId] = new RenderingGroup(renderingGroupId, this._scene, this._customOpaqueSortCompareFn[renderingGroupId], this._customAlphaTestSortCompareFn[renderingGroupId], this._customTransparentSortCompareFn[renderingGroupId]);\n }\n }\n /**\n * Add a sprite manager to the rendering manager in order to render it this frame.\n * @param spriteManager Define the sprite manager to render\n */\n dispatchSprites(spriteManager) {\n if (this.maintainStateBetweenFrames && spriteManager._wasDispatched) {\n return;\n }\n spriteManager._wasDispatched = true;\n this.getRenderingGroup(spriteManager.renderingGroupId).dispatchSprites(spriteManager);\n }\n /**\n * Add a particle system to the rendering manager in order to render it this frame.\n * @param particleSystem Define the particle system to render\n */\n dispatchParticles(particleSystem) {\n if (this.maintainStateBetweenFrames && particleSystem._wasDispatched) {\n return;\n }\n particleSystem._wasDispatched = true;\n this.getRenderingGroup(particleSystem.renderingGroupId).dispatchParticles(particleSystem);\n }\n /**\n * Add a submesh to the manager in order to render it this frame\n * @param subMesh The submesh to dispatch\n * @param mesh Optional reference to the submeshes's mesh. Provide if you have an exiting reference to improve performance.\n * @param material Optional reference to the submeshes's material. Provide if you have an exiting reference to improve performance.\n */\n dispatch(subMesh, mesh, material) {\n if (mesh === undefined) {\n mesh = subMesh.getMesh();\n }\n if (this.maintainStateBetweenFrames && subMesh._wasDispatched) {\n return;\n }\n subMesh._wasDispatched = true;\n this.getRenderingGroup(mesh.renderingGroupId).dispatch(subMesh, mesh, material);\n }\n /**\n * Overrides the default sort function applied in the rendering group to prepare the meshes.\n * This allowed control for front to back rendering or reversely depending of the special needs.\n *\n * @param renderingGroupId The rendering group id corresponding to its index\n * @param opaqueSortCompareFn The opaque queue comparison function use to sort.\n * @param alphaTestSortCompareFn The alpha test queue comparison function use to sort.\n * @param transparentSortCompareFn The transparent queue comparison function use to sort.\n */\n setRenderingOrder(renderingGroupId, opaqueSortCompareFn = null, alphaTestSortCompareFn = null, transparentSortCompareFn = null) {\n this._customOpaqueSortCompareFn[renderingGroupId] = opaqueSortCompareFn;\n this._customAlphaTestSortCompareFn[renderingGroupId] = alphaTestSortCompareFn;\n this._customTransparentSortCompareFn[renderingGroupId] = transparentSortCompareFn;\n if (this._renderingGroups[renderingGroupId]) {\n const group = this._renderingGroups[renderingGroupId];\n group.opaqueSortCompareFn = this._customOpaqueSortCompareFn[renderingGroupId];\n group.alphaTestSortCompareFn = this._customAlphaTestSortCompareFn[renderingGroupId];\n group.transparentSortCompareFn = this._customTransparentSortCompareFn[renderingGroupId];\n }\n }\n /**\n * Specifies whether or not the stencil and depth buffer are cleared between two rendering groups.\n *\n * @param renderingGroupId The rendering group id corresponding to its index\n * @param autoClearDepthStencil Automatically clears depth and stencil between groups if true.\n * @param depth Automatically clears depth between groups if true and autoClear is true.\n * @param stencil Automatically clears stencil between groups if true and autoClear is true.\n */\n setRenderingAutoClearDepthStencil(renderingGroupId, autoClearDepthStencil, depth = true, stencil = true) {\n this._autoClearDepthStencil[renderingGroupId] = {\n autoClear: autoClearDepthStencil,\n depth: depth,\n stencil: stencil,\n };\n }\n /**\n * Gets the current auto clear configuration for one rendering group of the rendering\n * manager.\n * @param index the rendering group index to get the information for\n * @returns The auto clear setup for the requested rendering group\n */\n getAutoClearDepthStencilSetup(index) {\n return this._autoClearDepthStencil[index];\n }\n }\n /**\n * The max id used for rendering groups (not included)\n */\n RenderingManager.MAX_RENDERINGGROUPS = 4;\n /**\n * The min id used for rendering groups (included)\n */\n RenderingManager.MIN_RENDERINGGROUPS = 0;\n /**\n * Used to globally prevent autoclearing scenes.\n */\n RenderingManager.AUTOCLEAR = true;\n\n /**\n * Groups all the scene component constants in one place to ease maintenance.\n * @internal\n */\n class SceneComponentConstants {\n }\n SceneComponentConstants.NAME_EFFECTLAYER = \"EffectLayer\";\n SceneComponentConstants.NAME_LAYER = \"Layer\";\n SceneComponentConstants.NAME_LENSFLARESYSTEM = \"LensFlareSystem\";\n SceneComponentConstants.NAME_BOUNDINGBOXRENDERER = \"BoundingBoxRenderer\";\n SceneComponentConstants.NAME_PARTICLESYSTEM = \"ParticleSystem\";\n SceneComponentConstants.NAME_GAMEPAD = \"Gamepad\";\n SceneComponentConstants.NAME_SIMPLIFICATIONQUEUE = \"SimplificationQueue\";\n SceneComponentConstants.NAME_GEOMETRYBUFFERRENDERER = \"GeometryBufferRenderer\";\n SceneComponentConstants.NAME_PREPASSRENDERER = \"PrePassRenderer\";\n SceneComponentConstants.NAME_DEPTHRENDERER = \"DepthRenderer\";\n SceneComponentConstants.NAME_DEPTHPEELINGRENDERER = \"DepthPeelingRenderer\";\n SceneComponentConstants.NAME_POSTPROCESSRENDERPIPELINEMANAGER = \"PostProcessRenderPipelineManager\";\n SceneComponentConstants.NAME_SPRITE = \"Sprite\";\n SceneComponentConstants.NAME_SUBSURFACE = \"SubSurface\";\n SceneComponentConstants.NAME_OUTLINERENDERER = \"Outline\";\n SceneComponentConstants.NAME_PROCEDURALTEXTURE = \"ProceduralTexture\";\n SceneComponentConstants.NAME_SHADOWGENERATOR = \"ShadowGenerator\";\n SceneComponentConstants.NAME_OCTREE = \"Octree\";\n SceneComponentConstants.NAME_PHYSICSENGINE = \"PhysicsEngine\";\n SceneComponentConstants.NAME_AUDIO = \"Audio\";\n SceneComponentConstants.NAME_FLUIDRENDERER = \"FluidRenderer\";\n SceneComponentConstants.STEP_ISREADYFORMESH_EFFECTLAYER = 0;\n SceneComponentConstants.STEP_BEFOREEVALUATEACTIVEMESH_BOUNDINGBOXRENDERER = 0;\n SceneComponentConstants.STEP_EVALUATESUBMESH_BOUNDINGBOXRENDERER = 0;\n SceneComponentConstants.STEP_PREACTIVEMESH_BOUNDINGBOXRENDERER = 0;\n SceneComponentConstants.STEP_CAMERADRAWRENDERTARGET_EFFECTLAYER = 1;\n SceneComponentConstants.STEP_BEFORECAMERADRAW_PREPASS = 0;\n SceneComponentConstants.STEP_BEFORECAMERADRAW_EFFECTLAYER = 1;\n SceneComponentConstants.STEP_BEFORECAMERADRAW_LAYER = 2;\n SceneComponentConstants.STEP_BEFORERENDERTARGETDRAW_PREPASS = 0;\n SceneComponentConstants.STEP_BEFORERENDERTARGETDRAW_LAYER = 1;\n SceneComponentConstants.STEP_BEFORERENDERINGMESH_PREPASS = 0;\n SceneComponentConstants.STEP_BEFORERENDERINGMESH_OUTLINE = 1;\n SceneComponentConstants.STEP_AFTERRENDERINGMESH_PREPASS = 0;\n SceneComponentConstants.STEP_AFTERRENDERINGMESH_OUTLINE = 1;\n SceneComponentConstants.STEP_AFTERRENDERINGGROUPDRAW_EFFECTLAYER_DRAW = 0;\n SceneComponentConstants.STEP_AFTERRENDERINGGROUPDRAW_BOUNDINGBOXRENDERER = 1;\n SceneComponentConstants.STEP_BEFORECAMERAUPDATE_SIMPLIFICATIONQUEUE = 0;\n SceneComponentConstants.STEP_BEFORECAMERAUPDATE_GAMEPAD = 1;\n SceneComponentConstants.STEP_BEFORECLEAR_PROCEDURALTEXTURE = 0;\n SceneComponentConstants.STEP_BEFORECLEAR_PREPASS = 1;\n SceneComponentConstants.STEP_BEFORERENDERTARGETCLEAR_PREPASS = 0;\n SceneComponentConstants.STEP_AFTERRENDERTARGETDRAW_PREPASS = 0;\n SceneComponentConstants.STEP_AFTERRENDERTARGETDRAW_LAYER = 1;\n SceneComponentConstants.STEP_AFTERCAMERADRAW_PREPASS = 0;\n SceneComponentConstants.STEP_AFTERCAMERADRAW_EFFECTLAYER = 1;\n SceneComponentConstants.STEP_AFTERCAMERADRAW_LENSFLARESYSTEM = 2;\n SceneComponentConstants.STEP_AFTERCAMERADRAW_EFFECTLAYER_DRAW = 3;\n SceneComponentConstants.STEP_AFTERCAMERADRAW_LAYER = 4;\n SceneComponentConstants.STEP_AFTERCAMERADRAW_FLUIDRENDERER = 5;\n SceneComponentConstants.STEP_AFTERCAMERAPOSTPROCESS_LAYER = 0;\n SceneComponentConstants.STEP_AFTERRENDERTARGETPOSTPROCESS_LAYER = 0;\n SceneComponentConstants.STEP_AFTERRENDER_AUDIO = 0;\n SceneComponentConstants.STEP_GATHERRENDERTARGETS_DEPTHRENDERER = 0;\n SceneComponentConstants.STEP_GATHERRENDERTARGETS_GEOMETRYBUFFERRENDERER = 1;\n SceneComponentConstants.STEP_GATHERRENDERTARGETS_SHADOWGENERATOR = 2;\n SceneComponentConstants.STEP_GATHERRENDERTARGETS_POSTPROCESSRENDERPIPELINEMANAGER = 3;\n SceneComponentConstants.STEP_GATHERACTIVECAMERARENDERTARGETS_DEPTHRENDERER = 0;\n SceneComponentConstants.STEP_GATHERACTIVECAMERARENDERTARGETS_FLUIDRENDERER = 1;\n SceneComponentConstants.STEP_POINTERMOVE_SPRITE = 0;\n SceneComponentConstants.STEP_POINTERDOWN_SPRITE = 0;\n SceneComponentConstants.STEP_POINTERUP_SPRITE = 0;\n /**\n * Representation of a stage in the scene (Basically a list of ordered steps)\n * @internal\n */\n class Stage extends Array {\n /**\n * Hide ctor from the rest of the world.\n * @param items The items to add.\n */\n constructor(items) {\n super(...items);\n }\n /**\n * Creates a new Stage.\n * @returns A new instance of a Stage\n */\n static Create() {\n return Object.create(Stage.prototype);\n }\n /**\n * Registers a step in an ordered way in the targeted stage.\n * @param index Defines the position to register the step in\n * @param component Defines the component attached to the step\n * @param action Defines the action to launch during the step\n */\n registerStep(index, component, action) {\n let i = 0;\n let maxIndex = Number.MAX_VALUE;\n for (; i < this.length; i++) {\n const step = this[i];\n maxIndex = step.index;\n if (index < maxIndex) {\n break;\n }\n }\n this.splice(i, 0, { index, component, action: action.bind(component) });\n }\n /**\n * Clears all the steps from the stage.\n */\n clear() {\n this.length = 0;\n }\n }\n\n /**\n * Gather the list of pointer event types as constants.\n */\n class PointerEventTypes {\n }\n /**\n * The pointerdown event is fired when a pointer becomes active. For mouse, it is fired when the device transitions from no buttons depressed to at least one button depressed. For touch, it is fired when physical contact is made with the digitizer. For pen, it is fired when the stylus makes physical contact with the digitizer.\n */\n PointerEventTypes.POINTERDOWN = 0x01;\n /**\n * The pointerup event is fired when a pointer is no longer active.\n */\n PointerEventTypes.POINTERUP = 0x02;\n /**\n * The pointermove event is fired when a pointer changes coordinates.\n */\n PointerEventTypes.POINTERMOVE = 0x04;\n /**\n * The pointerwheel event is fired when a mouse wheel has been rotated.\n */\n PointerEventTypes.POINTERWHEEL = 0x08;\n /**\n * The pointerpick event is fired when a mesh or sprite has been picked by the pointer.\n */\n PointerEventTypes.POINTERPICK = 0x10;\n /**\n * The pointertap event is fired when a the object has been touched and released without drag.\n */\n PointerEventTypes.POINTERTAP = 0x20;\n /**\n * The pointerdoubletap event is fired when a the object has been touched and released twice without drag.\n */\n PointerEventTypes.POINTERDOUBLETAP = 0x40;\n /**\n * Base class of pointer info types.\n */\n class PointerInfoBase {\n /**\n * Instantiates the base class of pointers info.\n * @param type Defines the type of event (PointerEventTypes)\n * @param event Defines the related dom event\n */\n constructor(\n /**\n * Defines the type of event (PointerEventTypes)\n */\n type, \n /**\n * Defines the related dom event\n */\n event) {\n this.type = type;\n this.event = event;\n }\n }\n /**\n * This class is used to store pointer related info for the onPrePointerObservable event.\n * Set the skipOnPointerObservable property to true if you want the engine to stop any process after this event is triggered, even not calling onPointerObservable\n */\n class PointerInfoPre extends PointerInfoBase {\n /**\n * Instantiates a PointerInfoPre to store pointer related info to the onPrePointerObservable event.\n * @param type Defines the type of event (PointerEventTypes)\n * @param event Defines the related dom event\n * @param localX Defines the local x coordinates of the pointer when the event occured\n * @param localY Defines the local y coordinates of the pointer when the event occured\n */\n constructor(type, event, localX, localY) {\n super(type, event);\n /**\n * Ray from a pointer if available (eg. 6dof controller)\n */\n this.ray = null;\n /**\n * The original picking info that was used to trigger the pointer event\n */\n this.originalPickingInfo = null;\n this.skipOnPointerObservable = false;\n this.localPosition = new Vector2(localX, localY);\n }\n }\n /**\n * This type contains all the data related to a pointer event in Babylon.js.\n * The event member is an instance of PointerEvent for all types except PointerWheel and is of type MouseWheelEvent when type equals PointerWheel. The different event types can be found in the PointerEventTypes class.\n */\n class PointerInfo extends PointerInfoBase {\n /**\n * Defines the picking info associated with this PointerInfo object (if applicable)\n */\n get pickInfo() {\n if (!this._pickInfo) {\n this._generatePickInfo();\n }\n return this._pickInfo;\n }\n /**\n * Instantiates a PointerInfo to store pointer related info to the onPointerObservable event.\n * @param type Defines the type of event (PointerEventTypes)\n * @param event Defines the related dom event\n * @param pickInfo Defines the picking info associated to the info (if any)\n * @param inputManager Defines the InputManager to use if there is no pickInfo\n */\n constructor(type, event, pickInfo, inputManager = null) {\n super(type, event);\n this._pickInfo = pickInfo;\n this._inputManager = inputManager;\n }\n /**\n * Generates the picking info if needed\n */\n /** @internal */\n _generatePickInfo() {\n if (this._inputManager) {\n this._pickInfo = this._inputManager._pickMove(this.event);\n this._inputManager._setRayOnPointerInfo(this._pickInfo, this.event);\n this._inputManager = null;\n }\n }\n }\n\n /**\n * Abstract class used to decouple action Manager from scene and meshes.\n * Do not instantiate.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions\n */\n class AbstractActionManager {\n constructor() {\n /** Gets the cursor to use when hovering items */\n this.hoverCursor = \"\";\n /** Gets the list of actions */\n this.actions = new Array();\n /**\n * Gets or sets a boolean indicating that the manager is recursive meaning that it can trigger action from children\n */\n this.isRecursive = false;\n }\n /**\n * Does exist one action manager with at least one trigger\n **/\n static get HasTriggers() {\n for (const t in AbstractActionManager.Triggers) {\n if (Object.prototype.hasOwnProperty.call(AbstractActionManager.Triggers, t)) {\n return true;\n }\n }\n return false;\n }\n /**\n * Does exist one action manager with at least one pick trigger\n **/\n static get HasPickTriggers() {\n for (const t in AbstractActionManager.Triggers) {\n if (Object.prototype.hasOwnProperty.call(AbstractActionManager.Triggers, t)) {\n const tAsInt = parseInt(t);\n if (tAsInt >= 1 && tAsInt <= 7) {\n return true;\n }\n }\n }\n return false;\n }\n /**\n * Does exist one action manager that handles actions of a given trigger\n * @param trigger defines the trigger to be tested\n * @returns a boolean indicating whether the trigger is handled by at least one action manager\n **/\n static HasSpecificTrigger(trigger) {\n for (const t in AbstractActionManager.Triggers) {\n if (Object.prototype.hasOwnProperty.call(AbstractActionManager.Triggers, t)) {\n const tAsInt = parseInt(t);\n if (tAsInt === trigger) {\n return true;\n }\n }\n }\n return false;\n }\n }\n /** Gets the list of active triggers */\n AbstractActionManager.Triggers = {};\n\n /**\n * Gather the list of keyboard event types as constants.\n */\n class KeyboardEventTypes {\n }\n /**\n * The keydown event is fired when a key becomes active (pressed).\n */\n KeyboardEventTypes.KEYDOWN = 0x01;\n /**\n * The keyup event is fired when a key has been released.\n */\n KeyboardEventTypes.KEYUP = 0x02;\n /**\n * This class is used to store keyboard related info for the onKeyboardObservable event.\n */\n class KeyboardInfo {\n /**\n * Instantiates a new keyboard info.\n * This class is used to store keyboard related info for the onKeyboardObservable event.\n * @param type Defines the type of event (KeyboardEventTypes)\n * @param event Defines the related dom event\n */\n constructor(\n /**\n * Defines the type of event (KeyboardEventTypes)\n */\n type, \n /**\n * Defines the related dom event\n */\n event) {\n this.type = type;\n this.event = event;\n }\n }\n /**\n * This class is used to store keyboard related info for the onPreKeyboardObservable event.\n * Set the skipOnKeyboardObservable property to true if you want the engine to stop any process after this event is triggered, even not calling onKeyboardObservable\n */\n class KeyboardInfoPre extends KeyboardInfo {\n /**\n * Defines whether the engine should skip the next onKeyboardObservable associated to this pre.\n * @deprecated use skipOnKeyboardObservable property instead\n */\n get skipOnPointerObservable() {\n return this.skipOnKeyboardObservable;\n }\n set skipOnPointerObservable(value) {\n this.skipOnKeyboardObservable = value;\n }\n /**\n * Instantiates a new keyboard pre info.\n * This class is used to store keyboard related info for the onPreKeyboardObservable event.\n * @param type Defines the type of event (KeyboardEventTypes)\n * @param event Defines the related dom event\n */\n constructor(\n /**\n * Defines the type of event (KeyboardEventTypes)\n */\n type, \n /**\n * Defines the related dom event\n */\n event) {\n super(type, event);\n this.type = type;\n this.event = event;\n this.skipOnKeyboardObservable = false;\n }\n }\n\n /**\n * Enum for Device Types\n */\n var DeviceType;\n (function (DeviceType) {\n /** Generic */\n DeviceType[DeviceType[\"Generic\"] = 0] = \"Generic\";\n /** Keyboard */\n DeviceType[DeviceType[\"Keyboard\"] = 1] = \"Keyboard\";\n /** Mouse */\n DeviceType[DeviceType[\"Mouse\"] = 2] = \"Mouse\";\n /** Touch Pointers */\n DeviceType[DeviceType[\"Touch\"] = 3] = \"Touch\";\n /** PS4 Dual Shock */\n DeviceType[DeviceType[\"DualShock\"] = 4] = \"DualShock\";\n /** Xbox */\n DeviceType[DeviceType[\"Xbox\"] = 5] = \"Xbox\";\n /** Switch Controller */\n DeviceType[DeviceType[\"Switch\"] = 6] = \"Switch\";\n /** PS5 DualSense */\n DeviceType[DeviceType[\"DualSense\"] = 7] = \"DualSense\";\n })(DeviceType || (DeviceType = {}));\n // Device Enums\n /**\n * Enum for All Pointers (Touch/Mouse)\n */\n var PointerInput;\n (function (PointerInput) {\n /** Horizontal Axis (Not used in events/observables; only in polling) */\n PointerInput[PointerInput[\"Horizontal\"] = 0] = \"Horizontal\";\n /** Vertical Axis (Not used in events/observables; only in polling) */\n PointerInput[PointerInput[\"Vertical\"] = 1] = \"Vertical\";\n /** Left Click or Touch */\n PointerInput[PointerInput[\"LeftClick\"] = 2] = \"LeftClick\";\n /** Middle Click */\n PointerInput[PointerInput[\"MiddleClick\"] = 3] = \"MiddleClick\";\n /** Right Click */\n PointerInput[PointerInput[\"RightClick\"] = 4] = \"RightClick\";\n /** Browser Back */\n PointerInput[PointerInput[\"BrowserBack\"] = 5] = \"BrowserBack\";\n /** Browser Forward */\n PointerInput[PointerInput[\"BrowserForward\"] = 6] = \"BrowserForward\";\n /** Mouse Wheel X */\n PointerInput[PointerInput[\"MouseWheelX\"] = 7] = \"MouseWheelX\";\n /** Mouse Wheel Y */\n PointerInput[PointerInput[\"MouseWheelY\"] = 8] = \"MouseWheelY\";\n /** Mouse Wheel Z */\n PointerInput[PointerInput[\"MouseWheelZ\"] = 9] = \"MouseWheelZ\";\n /** Used in events/observables to identify if x/y changes occurred */\n PointerInput[PointerInput[\"Move\"] = 12] = \"Move\";\n })(PointerInput || (PointerInput = {}));\n /** @internal */\n var NativePointerInput;\n (function (NativePointerInput) {\n /** Horizontal Axis */\n NativePointerInput[NativePointerInput[\"Horizontal\"] = 0] = \"Horizontal\";\n /** Vertical Axis */\n NativePointerInput[NativePointerInput[\"Vertical\"] = 1] = \"Vertical\";\n /** Left Click or Touch */\n NativePointerInput[NativePointerInput[\"LeftClick\"] = 2] = \"LeftClick\";\n /** Middle Click */\n NativePointerInput[NativePointerInput[\"MiddleClick\"] = 3] = \"MiddleClick\";\n /** Right Click */\n NativePointerInput[NativePointerInput[\"RightClick\"] = 4] = \"RightClick\";\n /** Browser Back */\n NativePointerInput[NativePointerInput[\"BrowserBack\"] = 5] = \"BrowserBack\";\n /** Browser Forward */\n NativePointerInput[NativePointerInput[\"BrowserForward\"] = 6] = \"BrowserForward\";\n /** Mouse Wheel X */\n NativePointerInput[NativePointerInput[\"MouseWheelX\"] = 7] = \"MouseWheelX\";\n /** Mouse Wheel Y */\n NativePointerInput[NativePointerInput[\"MouseWheelY\"] = 8] = \"MouseWheelY\";\n /** Mouse Wheel Z */\n NativePointerInput[NativePointerInput[\"MouseWheelZ\"] = 9] = \"MouseWheelZ\";\n /** Delta X */\n NativePointerInput[NativePointerInput[\"DeltaHorizontal\"] = 10] = \"DeltaHorizontal\";\n /** Delta Y */\n NativePointerInput[NativePointerInput[\"DeltaVertical\"] = 11] = \"DeltaVertical\";\n })(NativePointerInput || (NativePointerInput = {}));\n /**\n * Enum for Dual Shock Gamepad\n */\n var DualShockInput;\n (function (DualShockInput) {\n /** Cross */\n DualShockInput[DualShockInput[\"Cross\"] = 0] = \"Cross\";\n /** Circle */\n DualShockInput[DualShockInput[\"Circle\"] = 1] = \"Circle\";\n /** Square */\n DualShockInput[DualShockInput[\"Square\"] = 2] = \"Square\";\n /** Triangle */\n DualShockInput[DualShockInput[\"Triangle\"] = 3] = \"Triangle\";\n /** L1 */\n DualShockInput[DualShockInput[\"L1\"] = 4] = \"L1\";\n /** R1 */\n DualShockInput[DualShockInput[\"R1\"] = 5] = \"R1\";\n /** L2 */\n DualShockInput[DualShockInput[\"L2\"] = 6] = \"L2\";\n /** R2 */\n DualShockInput[DualShockInput[\"R2\"] = 7] = \"R2\";\n /** Share */\n DualShockInput[DualShockInput[\"Share\"] = 8] = \"Share\";\n /** Options */\n DualShockInput[DualShockInput[\"Options\"] = 9] = \"Options\";\n /** L3 */\n DualShockInput[DualShockInput[\"L3\"] = 10] = \"L3\";\n /** R3 */\n DualShockInput[DualShockInput[\"R3\"] = 11] = \"R3\";\n /** DPadUp */\n DualShockInput[DualShockInput[\"DPadUp\"] = 12] = \"DPadUp\";\n /** DPadDown */\n DualShockInput[DualShockInput[\"DPadDown\"] = 13] = \"DPadDown\";\n /** DPadLeft */\n DualShockInput[DualShockInput[\"DPadLeft\"] = 14] = \"DPadLeft\";\n /** DRight */\n DualShockInput[DualShockInput[\"DPadRight\"] = 15] = \"DPadRight\";\n /** Home */\n DualShockInput[DualShockInput[\"Home\"] = 16] = \"Home\";\n /** TouchPad */\n DualShockInput[DualShockInput[\"TouchPad\"] = 17] = \"TouchPad\";\n /** LStickXAxis */\n DualShockInput[DualShockInput[\"LStickXAxis\"] = 18] = \"LStickXAxis\";\n /** LStickYAxis */\n DualShockInput[DualShockInput[\"LStickYAxis\"] = 19] = \"LStickYAxis\";\n /** RStickXAxis */\n DualShockInput[DualShockInput[\"RStickXAxis\"] = 20] = \"RStickXAxis\";\n /** RStickYAxis */\n DualShockInput[DualShockInput[\"RStickYAxis\"] = 21] = \"RStickYAxis\";\n })(DualShockInput || (DualShockInput = {}));\n /**\n * Enum for Dual Sense Gamepad\n */\n var DualSenseInput;\n (function (DualSenseInput) {\n /** Cross */\n DualSenseInput[DualSenseInput[\"Cross\"] = 0] = \"Cross\";\n /** Circle */\n DualSenseInput[DualSenseInput[\"Circle\"] = 1] = \"Circle\";\n /** Square */\n DualSenseInput[DualSenseInput[\"Square\"] = 2] = \"Square\";\n /** Triangle */\n DualSenseInput[DualSenseInput[\"Triangle\"] = 3] = \"Triangle\";\n /** L1 */\n DualSenseInput[DualSenseInput[\"L1\"] = 4] = \"L1\";\n /** R1 */\n DualSenseInput[DualSenseInput[\"R1\"] = 5] = \"R1\";\n /** L2 */\n DualSenseInput[DualSenseInput[\"L2\"] = 6] = \"L2\";\n /** R2 */\n DualSenseInput[DualSenseInput[\"R2\"] = 7] = \"R2\";\n /** Create */\n DualSenseInput[DualSenseInput[\"Create\"] = 8] = \"Create\";\n /** Options */\n DualSenseInput[DualSenseInput[\"Options\"] = 9] = \"Options\";\n /** L3 */\n DualSenseInput[DualSenseInput[\"L3\"] = 10] = \"L3\";\n /** R3 */\n DualSenseInput[DualSenseInput[\"R3\"] = 11] = \"R3\";\n /** DPadUp */\n DualSenseInput[DualSenseInput[\"DPadUp\"] = 12] = \"DPadUp\";\n /** DPadDown */\n DualSenseInput[DualSenseInput[\"DPadDown\"] = 13] = \"DPadDown\";\n /** DPadLeft */\n DualSenseInput[DualSenseInput[\"DPadLeft\"] = 14] = \"DPadLeft\";\n /** DRight */\n DualSenseInput[DualSenseInput[\"DPadRight\"] = 15] = \"DPadRight\";\n /** Home */\n DualSenseInput[DualSenseInput[\"Home\"] = 16] = \"Home\";\n /** TouchPad */\n DualSenseInput[DualSenseInput[\"TouchPad\"] = 17] = \"TouchPad\";\n /** LStickXAxis */\n DualSenseInput[DualSenseInput[\"LStickXAxis\"] = 18] = \"LStickXAxis\";\n /** LStickYAxis */\n DualSenseInput[DualSenseInput[\"LStickYAxis\"] = 19] = \"LStickYAxis\";\n /** RStickXAxis */\n DualSenseInput[DualSenseInput[\"RStickXAxis\"] = 20] = \"RStickXAxis\";\n /** RStickYAxis */\n DualSenseInput[DualSenseInput[\"RStickYAxis\"] = 21] = \"RStickYAxis\";\n })(DualSenseInput || (DualSenseInput = {}));\n /**\n * Enum for Xbox Gamepad\n */\n var XboxInput;\n (function (XboxInput) {\n /** A */\n XboxInput[XboxInput[\"A\"] = 0] = \"A\";\n /** B */\n XboxInput[XboxInput[\"B\"] = 1] = \"B\";\n /** X */\n XboxInput[XboxInput[\"X\"] = 2] = \"X\";\n /** Y */\n XboxInput[XboxInput[\"Y\"] = 3] = \"Y\";\n /** LB */\n XboxInput[XboxInput[\"LB\"] = 4] = \"LB\";\n /** RB */\n XboxInput[XboxInput[\"RB\"] = 5] = \"RB\";\n /** LT */\n XboxInput[XboxInput[\"LT\"] = 6] = \"LT\";\n /** RT */\n XboxInput[XboxInput[\"RT\"] = 7] = \"RT\";\n /** Back */\n XboxInput[XboxInput[\"Back\"] = 8] = \"Back\";\n /** Start */\n XboxInput[XboxInput[\"Start\"] = 9] = \"Start\";\n /** LS */\n XboxInput[XboxInput[\"LS\"] = 10] = \"LS\";\n /** RS */\n XboxInput[XboxInput[\"RS\"] = 11] = \"RS\";\n /** DPadUp */\n XboxInput[XboxInput[\"DPadUp\"] = 12] = \"DPadUp\";\n /** DPadDown */\n XboxInput[XboxInput[\"DPadDown\"] = 13] = \"DPadDown\";\n /** DPadLeft */\n XboxInput[XboxInput[\"DPadLeft\"] = 14] = \"DPadLeft\";\n /** DRight */\n XboxInput[XboxInput[\"DPadRight\"] = 15] = \"DPadRight\";\n /** Home */\n XboxInput[XboxInput[\"Home\"] = 16] = \"Home\";\n /** LStickXAxis */\n XboxInput[XboxInput[\"LStickXAxis\"] = 17] = \"LStickXAxis\";\n /** LStickYAxis */\n XboxInput[XboxInput[\"LStickYAxis\"] = 18] = \"LStickYAxis\";\n /** RStickXAxis */\n XboxInput[XboxInput[\"RStickXAxis\"] = 19] = \"RStickXAxis\";\n /** RStickYAxis */\n XboxInput[XboxInput[\"RStickYAxis\"] = 20] = \"RStickYAxis\";\n })(XboxInput || (XboxInput = {}));\n /**\n * Enum for Switch (Pro/JoyCon L+R) Gamepad\n */\n var SwitchInput;\n (function (SwitchInput) {\n /** B */\n SwitchInput[SwitchInput[\"B\"] = 0] = \"B\";\n /** A */\n SwitchInput[SwitchInput[\"A\"] = 1] = \"A\";\n /** Y */\n SwitchInput[SwitchInput[\"Y\"] = 2] = \"Y\";\n /** X */\n SwitchInput[SwitchInput[\"X\"] = 3] = \"X\";\n /** L */\n SwitchInput[SwitchInput[\"L\"] = 4] = \"L\";\n /** R */\n SwitchInput[SwitchInput[\"R\"] = 5] = \"R\";\n /** ZL */\n SwitchInput[SwitchInput[\"ZL\"] = 6] = \"ZL\";\n /** ZR */\n SwitchInput[SwitchInput[\"ZR\"] = 7] = \"ZR\";\n /** Minus */\n SwitchInput[SwitchInput[\"Minus\"] = 8] = \"Minus\";\n /** Plus */\n SwitchInput[SwitchInput[\"Plus\"] = 9] = \"Plus\";\n /** LS */\n SwitchInput[SwitchInput[\"LS\"] = 10] = \"LS\";\n /** RS */\n SwitchInput[SwitchInput[\"RS\"] = 11] = \"RS\";\n /** DPadUp */\n SwitchInput[SwitchInput[\"DPadUp\"] = 12] = \"DPadUp\";\n /** DPadDown */\n SwitchInput[SwitchInput[\"DPadDown\"] = 13] = \"DPadDown\";\n /** DPadLeft */\n SwitchInput[SwitchInput[\"DPadLeft\"] = 14] = \"DPadLeft\";\n /** DRight */\n SwitchInput[SwitchInput[\"DPadRight\"] = 15] = \"DPadRight\";\n /** Home */\n SwitchInput[SwitchInput[\"Home\"] = 16] = \"Home\";\n /** Capture */\n SwitchInput[SwitchInput[\"Capture\"] = 17] = \"Capture\";\n /** LStickXAxis */\n SwitchInput[SwitchInput[\"LStickXAxis\"] = 18] = \"LStickXAxis\";\n /** LStickYAxis */\n SwitchInput[SwitchInput[\"LStickYAxis\"] = 19] = \"LStickYAxis\";\n /** RStickXAxis */\n SwitchInput[SwitchInput[\"RStickXAxis\"] = 20] = \"RStickXAxis\";\n /** RStickYAxis */\n SwitchInput[SwitchInput[\"RStickYAxis\"] = 21] = \"RStickYAxis\";\n })(SwitchInput || (SwitchInput = {}));\n\n /**\n * Event Types\n */\n var DeviceInputEventType;\n (function (DeviceInputEventType) {\n // Pointers\n /** PointerMove */\n DeviceInputEventType[DeviceInputEventType[\"PointerMove\"] = 0] = \"PointerMove\";\n /** PointerDown */\n DeviceInputEventType[DeviceInputEventType[\"PointerDown\"] = 1] = \"PointerDown\";\n /** PointerUp */\n DeviceInputEventType[DeviceInputEventType[\"PointerUp\"] = 2] = \"PointerUp\";\n })(DeviceInputEventType || (DeviceInputEventType = {}));\n /**\n * Constants used for Events\n */\n class EventConstants {\n }\n /**\n * Pixel delta for Wheel Events (Default)\n */\n EventConstants.DOM_DELTA_PIXEL = 0x00;\n /**\n * Line delta for Wheel Events\n */\n EventConstants.DOM_DELTA_LINE = 0x01;\n /**\n * Page delta for Wheel Events\n */\n EventConstants.DOM_DELTA_PAGE = 0x02;\n\n /**\n * Class to wrap DeviceInputSystem data into an event object\n */\n class DeviceEventFactory {\n /**\n * Create device input events based on provided type and slot\n *\n * @param deviceType Type of device\n * @param deviceSlot \"Slot\" or index that device is referenced in\n * @param inputIndex Id of input to be checked\n * @param currentState Current value for given input\n * @param deviceInputSystem Reference to DeviceInputSystem\n * @param elementToAttachTo HTMLElement to reference as target for inputs\n * @returns IUIEvent object\n */\n static CreateDeviceEvent(deviceType, deviceSlot, inputIndex, currentState, deviceInputSystem, elementToAttachTo, pointerId) {\n switch (deviceType) {\n case DeviceType.Keyboard:\n return this._CreateKeyboardEvent(inputIndex, currentState, deviceInputSystem, elementToAttachTo);\n case DeviceType.Mouse:\n if (inputIndex === PointerInput.MouseWheelX || inputIndex === PointerInput.MouseWheelY || inputIndex === PointerInput.MouseWheelZ) {\n return this._CreateWheelEvent(deviceType, deviceSlot, inputIndex, currentState, deviceInputSystem, elementToAttachTo);\n }\n // eslint-disable-next-line no-fallthrough\n case DeviceType.Touch:\n return this._CreatePointerEvent(deviceType, deviceSlot, inputIndex, currentState, deviceInputSystem, elementToAttachTo, pointerId);\n default:\n throw `Unable to generate event for device ${DeviceType[deviceType]}`;\n }\n }\n /**\n * Creates pointer event\n *\n * @param deviceType Type of device\n * @param deviceSlot \"Slot\" or index that device is referenced in\n * @param inputIndex Id of input to be checked\n * @param currentState Current value for given input\n * @param deviceInputSystem Reference to DeviceInputSystem\n * @param elementToAttachTo HTMLElement to reference as target for inputs\n * @returns IUIEvent object (Pointer)\n */\n static _CreatePointerEvent(deviceType, deviceSlot, inputIndex, currentState, deviceInputSystem, elementToAttachTo, pointerId) {\n const evt = this._CreateMouseEvent(deviceType, deviceSlot, inputIndex, currentState, deviceInputSystem, elementToAttachTo);\n if (deviceType === DeviceType.Mouse) {\n evt.deviceType = DeviceType.Mouse;\n evt.pointerId = 1;\n evt.pointerType = \"mouse\";\n }\n else {\n evt.deviceType = DeviceType.Touch;\n evt.pointerId = pointerId !== null && pointerId !== void 0 ? pointerId : deviceSlot;\n evt.pointerType = \"touch\";\n }\n let buttons = 0;\n // Populate buttons property with current state of all mouse buttons\n // Uses values found on: https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent/buttons\n buttons += deviceInputSystem.pollInput(deviceType, deviceSlot, PointerInput.LeftClick);\n buttons += deviceInputSystem.pollInput(deviceType, deviceSlot, PointerInput.RightClick) * 2;\n buttons += deviceInputSystem.pollInput(deviceType, deviceSlot, PointerInput.MiddleClick) * 4;\n evt.buttons = buttons;\n if (inputIndex === PointerInput.Move) {\n evt.type = \"pointermove\";\n }\n else if (inputIndex >= PointerInput.LeftClick && inputIndex <= PointerInput.RightClick) {\n evt.type = currentState === 1 ? \"pointerdown\" : \"pointerup\";\n evt.button = inputIndex - 2;\n }\n return evt;\n }\n /**\n * Create Mouse Wheel Event\n * @param deviceType Type of device\n * @param deviceSlot \"Slot\" or index that device is referenced in\n * @param inputIndex Id of input to be checked\n * @param currentState Current value for given input\n * @param deviceInputSystem Reference to DeviceInputSystem\n * @param elementToAttachTo HTMLElement to reference as target for inputs\n * @returns IUIEvent object (Wheel)\n */\n static _CreateWheelEvent(deviceType, deviceSlot, inputIndex, currentState, deviceInputSystem, elementToAttachTo) {\n const evt = this._CreateMouseEvent(deviceType, deviceSlot, inputIndex, currentState, deviceInputSystem, elementToAttachTo);\n // While WheelEvents don't generally have a pointerId, we used to add one in the InputManager\n // This line has been added to make the InputManager more platform-agnostic\n // Similar code exists in the WebDeviceInputSystem to handle browser created events\n evt.pointerId = 1;\n evt.type = \"wheel\";\n evt.deltaMode = EventConstants.DOM_DELTA_PIXEL;\n evt.deltaX = 0;\n evt.deltaY = 0;\n evt.deltaZ = 0;\n switch (inputIndex) {\n case PointerInput.MouseWheelX:\n evt.deltaX = currentState;\n break;\n case PointerInput.MouseWheelY:\n evt.deltaY = currentState;\n break;\n case PointerInput.MouseWheelZ:\n evt.deltaZ = currentState;\n break;\n }\n return evt;\n }\n /**\n * Create Mouse Event\n * @param deviceType Type of device\n * @param deviceSlot \"Slot\" or index that device is referenced in\n * @param inputIndex Id of input to be checked\n * @param currentState Current value for given input\n * @param deviceInputSystem Reference to DeviceInputSystem\n * @param elementToAttachTo HTMLElement to reference as target for inputs\n * @returns IUIEvent object (Mouse)\n */\n static _CreateMouseEvent(deviceType, deviceSlot, inputIndex, currentState, deviceInputSystem, elementToAttachTo) {\n const evt = this._CreateEvent(elementToAttachTo);\n const pointerX = deviceInputSystem.pollInput(deviceType, deviceSlot, PointerInput.Horizontal);\n const pointerY = deviceInputSystem.pollInput(deviceType, deviceSlot, PointerInput.Vertical);\n // Handle offsets/deltas based on existence of HTMLElement\n if (elementToAttachTo) {\n evt.movementX = 0;\n evt.movementY = 0;\n evt.offsetX = evt.movementX - elementToAttachTo.getBoundingClientRect().x;\n evt.offsetY = evt.movementY - elementToAttachTo.getBoundingClientRect().y;\n }\n else {\n evt.movementX = deviceInputSystem.pollInput(deviceType, deviceSlot, NativePointerInput.DeltaHorizontal); // DeltaHorizontal\n evt.movementY = deviceInputSystem.pollInput(deviceType, deviceSlot, NativePointerInput.DeltaVertical); // DeltaVertical\n evt.offsetX = 0;\n evt.offsetY = 0;\n }\n this._CheckNonCharacterKeys(evt, deviceInputSystem);\n evt.clientX = pointerX;\n evt.clientY = pointerY;\n evt.x = pointerX;\n evt.y = pointerY;\n evt.deviceType = deviceType;\n evt.deviceSlot = deviceSlot;\n evt.inputIndex = inputIndex;\n return evt;\n }\n /**\n * Create Keyboard Event\n * @param inputIndex Id of input to be checked\n * @param currentState Current value for given input\n * @param deviceInputSystem Reference to DeviceInputSystem\n * @param elementToAttachTo HTMLElement to reference as target for inputs\n * @returns IEvent object (Keyboard)\n */\n static _CreateKeyboardEvent(inputIndex, currentState, deviceInputSystem, elementToAttachTo) {\n const evt = this._CreateEvent(elementToAttachTo);\n this._CheckNonCharacterKeys(evt, deviceInputSystem);\n evt.deviceType = DeviceType.Keyboard;\n evt.deviceSlot = 0;\n evt.inputIndex = inputIndex;\n evt.type = currentState === 1 ? \"keydown\" : \"keyup\";\n evt.key = String.fromCharCode(inputIndex);\n evt.keyCode = inputIndex;\n return evt;\n }\n /**\n * Add parameters for non-character keys (Ctrl, Alt, Meta, Shift)\n * @param evt Event object to add parameters to\n * @param deviceInputSystem DeviceInputSystem to pull values from\n */\n static _CheckNonCharacterKeys(evt, deviceInputSystem) {\n const isKeyboardActive = deviceInputSystem.isDeviceAvailable(DeviceType.Keyboard);\n const altKey = isKeyboardActive && deviceInputSystem.pollInput(DeviceType.Keyboard, 0, 18) === 1;\n const ctrlKey = isKeyboardActive && deviceInputSystem.pollInput(DeviceType.Keyboard, 0, 17) === 1;\n const metaKey = isKeyboardActive &&\n (deviceInputSystem.pollInput(DeviceType.Keyboard, 0, 91) === 1 ||\n deviceInputSystem.pollInput(DeviceType.Keyboard, 0, 92) === 1 ||\n deviceInputSystem.pollInput(DeviceType.Keyboard, 0, 93) === 1);\n const shiftKey = isKeyboardActive && deviceInputSystem.pollInput(DeviceType.Keyboard, 0, 16) === 1;\n evt.altKey = altKey;\n evt.ctrlKey = ctrlKey;\n evt.metaKey = metaKey;\n evt.shiftKey = shiftKey;\n }\n /**\n * Create base event object\n * @param elementToAttachTo Value to use as event target\n * @returns\n */\n static _CreateEvent(elementToAttachTo) {\n const evt = {};\n evt.preventDefault = () => { };\n evt.target = elementToAttachTo;\n return evt;\n }\n }\n\n /** @internal */\n class NativeDeviceInputSystem {\n constructor(onDeviceConnected, onDeviceDisconnected, onInputChanged) {\n this._nativeInput = _native.DeviceInputSystem\n ? new _native.DeviceInputSystem(onDeviceConnected, onDeviceDisconnected, (deviceType, deviceSlot, inputIndex, currentState) => {\n const evt = DeviceEventFactory.CreateDeviceEvent(deviceType, deviceSlot, inputIndex, currentState, this);\n onInputChanged(deviceType, deviceSlot, evt);\n })\n : this._createDummyNativeInput();\n }\n // Public functions\n /**\n * Checks for current device input value, given an id and input index. Throws exception if requested device not initialized.\n * @param deviceType Enum specifying device type\n * @param deviceSlot \"Slot\" or index that device is referenced in\n * @param inputIndex Id of input to be checked\n * @returns Current value of input\n */\n pollInput(deviceType, deviceSlot, inputIndex) {\n return this._nativeInput.pollInput(deviceType, deviceSlot, inputIndex);\n }\n /**\n * Check for a specific device in the DeviceInputSystem\n * @param deviceType Type of device to check for\n * @returns bool with status of device's existence\n */\n isDeviceAvailable(deviceType) {\n //TODO: FIx native side first\n return deviceType === DeviceType.Mouse || deviceType === DeviceType.Touch;\n }\n /**\n * Dispose of all the observables\n */\n dispose() {\n this._nativeInput.dispose();\n }\n /**\n * For versions of BabylonNative that don't have the NativeInput plugin initialized, create a dummy version\n * @returns Object with dummy functions\n */\n _createDummyNativeInput() {\n const nativeInput = {\n pollInput: () => {\n return 0;\n },\n isDeviceAvailable: () => {\n return false;\n },\n dispose: () => { },\n };\n return nativeInput;\n }\n }\n\n // eslint-disable-next-line @typescript-eslint/naming-convention\n const MAX_KEYCODES = 255;\n // eslint-disable-next-line @typescript-eslint/naming-convention\n const MAX_POINTER_INPUTS = Object.keys(PointerInput).length / 2;\n /** @internal */\n class WebDeviceInputSystem {\n constructor(engine, onDeviceConnected, onDeviceDisconnected, onInputChanged) {\n // Private Members\n this._inputs = [];\n this._keyboardActive = false;\n this._pointerActive = false;\n this._usingSafari = Tools.IsSafari();\n // Found solution for determining if MacOS is being used here:\n // https://stackoverflow.com/questions/10527983/best-way-to-detect-mac-os-x-or-windows-computers-with-javascript-or-jquery\n this._usingMacOS = IsNavigatorAvailable() && /(Mac|iPhone|iPod|iPad)/i.test(navigator.platform);\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n this._keyboardDownEvent = (evt) => { };\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n this._keyboardUpEvent = (evt) => { };\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n this._keyboardBlurEvent = (evt) => { };\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n this._pointerMoveEvent = (evt) => { };\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n this._pointerDownEvent = (evt) => { };\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n this._pointerUpEvent = (evt) => { };\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n this._pointerCancelEvent = (evt) => { };\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n this._pointerWheelEvent = (evt) => { };\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n this._pointerBlurEvent = (evt) => { };\n this._eventsAttached = false;\n this._mouseId = -1;\n this._isUsingFirefox = IsNavigatorAvailable() && navigator.userAgent && navigator.userAgent.indexOf(\"Firefox\") !== -1;\n this._maxTouchPoints = 0;\n this._pointerInputClearObserver = null;\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n this._gamepadConnectedEvent = (evt) => { };\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n this._gamepadDisconnectedEvent = (evt) => { };\n this._eventPrefix = Tools.GetPointerPrefix(engine);\n this._engine = engine;\n this._onDeviceConnected = onDeviceConnected;\n this._onDeviceDisconnected = onDeviceDisconnected;\n this._onInputChanged = onInputChanged;\n // If we need a pointerId, set one for future use\n this._mouseId = this._isUsingFirefox ? 0 : 1;\n this._enableEvents();\n if (this._usingMacOS) {\n this._metaKeys = [];\n }\n // Set callback to enable event handler switching when inputElement changes\n if (!this._engine._onEngineViewChanged) {\n this._engine._onEngineViewChanged = () => {\n this._enableEvents();\n };\n }\n }\n // Public functions\n /**\n * Checks for current device input value, given an id and input index. Throws exception if requested device not initialized.\n * @param deviceType Enum specifying device type\n * @param deviceSlot \"Slot\" or index that device is referenced in\n * @param inputIndex Id of input to be checked\n * @returns Current value of input\n */\n pollInput(deviceType, deviceSlot, inputIndex) {\n const device = this._inputs[deviceType][deviceSlot];\n if (!device) {\n throw `Unable to find device ${DeviceType[deviceType]}`;\n }\n if (deviceType >= DeviceType.DualShock && deviceType <= DeviceType.DualSense) {\n this._updateDevice(deviceType, deviceSlot, inputIndex);\n }\n const currentValue = device[inputIndex];\n if (currentValue === undefined) {\n throw `Unable to find input ${inputIndex} for device ${DeviceType[deviceType]} in slot ${deviceSlot}`;\n }\n if (inputIndex === PointerInput.Move) {\n Tools.Warn(`Unable to provide information for PointerInput.Move. Try using PointerInput.Horizontal or PointerInput.Vertical for move data.`);\n }\n return currentValue;\n }\n /**\n * Check for a specific device in the DeviceInputSystem\n * @param deviceType Type of device to check for\n * @returns bool with status of device's existence\n */\n isDeviceAvailable(deviceType) {\n return this._inputs[deviceType] !== undefined;\n }\n /**\n * Dispose of all the eventlisteners\n */\n dispose() {\n // Callbacks\n this._onDeviceConnected = () => { };\n this._onDeviceDisconnected = () => { };\n this._onInputChanged = () => { };\n delete this._engine._onEngineViewChanged;\n if (this._elementToAttachTo) {\n this._disableEvents();\n }\n }\n /**\n * Enable listening for user input events\n */\n _enableEvents() {\n const inputElement = this === null || this === void 0 ? void 0 : this._engine.getInputElement();\n if (inputElement && (!this._eventsAttached || this._elementToAttachTo !== inputElement)) {\n // Remove events before adding to avoid double events or simultaneous events on multiple canvases\n this._disableEvents();\n // If the inputs array has already been created, zero it out to before setting up events\n if (this._inputs) {\n for (const inputs of this._inputs) {\n if (inputs) {\n for (const deviceSlotKey in inputs) {\n const deviceSlot = +deviceSlotKey;\n const device = inputs[deviceSlot];\n if (device) {\n for (let inputIndex = 0; inputIndex < device.length; inputIndex++) {\n device[inputIndex] = 0;\n }\n }\n }\n }\n }\n }\n this._elementToAttachTo = inputElement;\n // Set tab index for the inputElement to the engine's canvasTabIndex, if and only if the element's tab index is -1\n this._elementToAttachTo.tabIndex = this._elementToAttachTo.tabIndex !== -1 ? this._elementToAttachTo.tabIndex : this._engine.canvasTabIndex;\n this._handleKeyActions();\n this._handlePointerActions();\n this._handleGamepadActions();\n this._eventsAttached = true;\n // Check for devices that are already connected but aren't registered. Currently, only checks for gamepads and mouse\n this._checkForConnectedDevices();\n }\n }\n /**\n * Disable listening for user input events\n */\n _disableEvents() {\n if (this._elementToAttachTo) {\n // Blur Events\n this._elementToAttachTo.removeEventListener(\"blur\", this._keyboardBlurEvent);\n this._elementToAttachTo.removeEventListener(\"blur\", this._pointerBlurEvent);\n // Keyboard Events\n this._elementToAttachTo.removeEventListener(\"keydown\", this._keyboardDownEvent);\n this._elementToAttachTo.removeEventListener(\"keyup\", this._keyboardUpEvent);\n // Pointer Events\n this._elementToAttachTo.removeEventListener(this._eventPrefix + \"move\", this._pointerMoveEvent);\n this._elementToAttachTo.removeEventListener(this._eventPrefix + \"down\", this._pointerDownEvent);\n this._elementToAttachTo.removeEventListener(this._eventPrefix + \"up\", this._pointerUpEvent);\n this._elementToAttachTo.removeEventListener(this._eventPrefix + \"cancel\", this._pointerCancelEvent);\n this._elementToAttachTo.removeEventListener(this._wheelEventName, this._pointerWheelEvent);\n // Gamepad Events\n window.removeEventListener(\"gamepadconnected\", this._gamepadConnectedEvent);\n window.removeEventListener(\"gamepaddisconnected\", this._gamepadDisconnectedEvent);\n }\n if (this._pointerInputClearObserver) {\n this._engine.onEndFrameObservable.remove(this._pointerInputClearObserver);\n }\n this._eventsAttached = false;\n }\n /**\n * Checks for existing connections to devices and register them, if necessary\n * Currently handles gamepads and mouse\n */\n _checkForConnectedDevices() {\n if (navigator.getGamepads) {\n const gamepads = navigator.getGamepads();\n for (const gamepad of gamepads) {\n if (gamepad) {\n this._addGamePad(gamepad);\n }\n }\n }\n // If the device in use has mouse capabilities, pre-register mouse\n if (typeof matchMedia === \"function\" && matchMedia(\"(pointer:fine)\").matches) {\n // This will provide a dummy value for the cursor position and is expected to be overridden when the first mouse event happens.\n // There isn't any good way to get the current position outside of a pointer event so that's why this was done.\n this._addPointerDevice(DeviceType.Mouse, 0, 0, 0);\n }\n }\n // Private functions\n /**\n * Add a gamepad to the DeviceInputSystem\n * @param gamepad A single DOM Gamepad object\n */\n _addGamePad(gamepad) {\n const deviceType = this._getGamepadDeviceType(gamepad.id);\n const deviceSlot = gamepad.index;\n this._gamepads = this._gamepads || new Array(gamepad.index + 1);\n this._registerDevice(deviceType, deviceSlot, gamepad.buttons.length + gamepad.axes.length);\n this._gamepads[deviceSlot] = deviceType;\n }\n /**\n * Add pointer device to DeviceInputSystem\n * @param deviceType Type of Pointer to add\n * @param deviceSlot Pointer ID (0 for mouse, pointerId for Touch)\n * @param currentX Current X at point of adding\n * @param currentY Current Y at point of adding\n */\n _addPointerDevice(deviceType, deviceSlot, currentX, currentY) {\n if (!this._pointerActive) {\n this._pointerActive = true;\n }\n this._registerDevice(deviceType, deviceSlot, MAX_POINTER_INPUTS);\n const pointer = this._inputs[deviceType][deviceSlot]; /* initialize our pointer position immediately after registration */\n pointer[0] = currentX;\n pointer[1] = currentY;\n }\n /**\n * Add device and inputs to device array\n * @param deviceType Enum specifying device type\n * @param deviceSlot \"Slot\" or index that device is referenced in\n * @param numberOfInputs Number of input entries to create for given device\n */\n _registerDevice(deviceType, deviceSlot, numberOfInputs) {\n if (deviceSlot === undefined) {\n throw `Unable to register device ${DeviceType[deviceType]} to undefined slot.`;\n }\n if (!this._inputs[deviceType]) {\n this._inputs[deviceType] = {};\n }\n if (!this._inputs[deviceType][deviceSlot]) {\n const device = new Array(numberOfInputs);\n device.fill(0);\n this._inputs[deviceType][deviceSlot] = device;\n this._onDeviceConnected(deviceType, deviceSlot);\n }\n }\n /**\n * Given a specific device name, remove that device from the device map\n * @param deviceType Enum specifying device type\n * @param deviceSlot \"Slot\" or index that device is referenced in\n */\n _unregisterDevice(deviceType, deviceSlot) {\n if (this._inputs[deviceType][deviceSlot]) {\n delete this._inputs[deviceType][deviceSlot];\n this._onDeviceDisconnected(deviceType, deviceSlot);\n }\n }\n /**\n * Handle all actions that come from keyboard interaction\n */\n _handleKeyActions() {\n this._keyboardDownEvent = (evt) => {\n if (!this._keyboardActive) {\n this._keyboardActive = true;\n this._registerDevice(DeviceType.Keyboard, 0, MAX_KEYCODES);\n }\n const kbKey = this._inputs[DeviceType.Keyboard][0];\n if (kbKey) {\n kbKey[evt.keyCode] = 1;\n const deviceEvent = evt;\n deviceEvent.inputIndex = evt.keyCode;\n if (this._usingMacOS && evt.metaKey && evt.key !== \"Meta\") {\n if (!this._metaKeys.includes(evt.keyCode)) {\n this._metaKeys.push(evt.keyCode);\n }\n }\n this._onInputChanged(DeviceType.Keyboard, 0, deviceEvent);\n }\n };\n this._keyboardUpEvent = (evt) => {\n if (!this._keyboardActive) {\n this._keyboardActive = true;\n this._registerDevice(DeviceType.Keyboard, 0, MAX_KEYCODES);\n }\n const kbKey = this._inputs[DeviceType.Keyboard][0];\n if (kbKey) {\n kbKey[evt.keyCode] = 0;\n const deviceEvent = evt;\n deviceEvent.inputIndex = evt.keyCode;\n if (this._usingMacOS && evt.key === \"Meta\" && this._metaKeys.length > 0) {\n for (const keyCode of this._metaKeys) {\n const deviceEvent = DeviceEventFactory.CreateDeviceEvent(DeviceType.Keyboard, 0, keyCode, 0, this, this._elementToAttachTo);\n kbKey[keyCode] = 0;\n this._onInputChanged(DeviceType.Keyboard, 0, deviceEvent);\n }\n this._metaKeys.splice(0, this._metaKeys.length);\n }\n this._onInputChanged(DeviceType.Keyboard, 0, deviceEvent);\n }\n };\n this._keyboardBlurEvent = () => {\n if (this._keyboardActive) {\n const kbKey = this._inputs[DeviceType.Keyboard][0];\n for (let i = 0; i < kbKey.length; i++) {\n if (kbKey[i] !== 0) {\n kbKey[i] = 0;\n const deviceEvent = DeviceEventFactory.CreateDeviceEvent(DeviceType.Keyboard, 0, i, 0, this, this._elementToAttachTo);\n this._onInputChanged(DeviceType.Keyboard, 0, deviceEvent);\n }\n }\n if (this._usingMacOS) {\n this._metaKeys.splice(0, this._metaKeys.length);\n }\n }\n };\n this._elementToAttachTo.addEventListener(\"keydown\", this._keyboardDownEvent);\n this._elementToAttachTo.addEventListener(\"keyup\", this._keyboardUpEvent);\n this._elementToAttachTo.addEventListener(\"blur\", this._keyboardBlurEvent);\n }\n /**\n * Handle all actions that come from pointer interaction\n */\n _handlePointerActions() {\n // If maxTouchPoints is defined, use that value. Otherwise, allow for a minimum for supported gestures like pinch\n this._maxTouchPoints = (IsNavigatorAvailable() && navigator.maxTouchPoints) || 2;\n if (!this._activeTouchIds) {\n this._activeTouchIds = new Array(this._maxTouchPoints);\n }\n for (let i = 0; i < this._maxTouchPoints; i++) {\n this._activeTouchIds[i] = -1;\n }\n this._pointerMoveEvent = (evt) => {\n const deviceType = this._getPointerType(evt);\n const deviceSlot = deviceType === DeviceType.Mouse ? 0 : this._activeTouchIds.indexOf(evt.pointerId);\n if (!this._inputs[deviceType]) {\n this._inputs[deviceType] = {};\n }\n if (!this._inputs[deviceType][deviceSlot]) {\n this._addPointerDevice(deviceType, deviceSlot, evt.clientX, evt.clientY);\n }\n const pointer = this._inputs[deviceType][deviceSlot];\n if (pointer) {\n const deviceEvent = evt;\n deviceEvent.inputIndex = PointerInput.Move;\n pointer[PointerInput.Horizontal] = evt.clientX;\n pointer[PointerInput.Vertical] = evt.clientY;\n if (evt.pointerId === undefined) {\n evt.pointerId = this._mouseId;\n }\n this._onInputChanged(deviceType, deviceSlot, deviceEvent);\n // Lets Propagate the event for move with same position.\n if (!this._usingSafari && evt.button !== -1) {\n deviceEvent.inputIndex = evt.button + 2;\n pointer[evt.button + 2] = pointer[evt.button + 2] ? 0 : 1; // Reverse state of button if evt.button has value\n this._onInputChanged(deviceType, deviceSlot, deviceEvent);\n }\n }\n };\n this._pointerDownEvent = (evt) => {\n const deviceType = this._getPointerType(evt);\n let deviceSlot = deviceType === DeviceType.Mouse ? 0 : evt.pointerId;\n if (deviceType === DeviceType.Touch) {\n const idx = this._activeTouchIds.indexOf(-1);\n if (idx >= 0) {\n deviceSlot = idx;\n this._activeTouchIds[idx] = evt.pointerId;\n }\n else {\n // We can't find an open slot to store new pointer so just return (can only support max number of touches)\n Tools.Warn(`Max number of touches exceeded. Ignoring touches in excess of ${this._maxTouchPoints}`);\n return;\n }\n }\n if (!this._inputs[deviceType]) {\n this._inputs[deviceType] = {};\n }\n if (!this._inputs[deviceType][deviceSlot]) {\n this._addPointerDevice(deviceType, deviceSlot, evt.clientX, evt.clientY);\n }\n else if (deviceType === DeviceType.Touch) {\n this._onDeviceConnected(deviceType, deviceSlot);\n }\n const pointer = this._inputs[deviceType][deviceSlot];\n if (pointer) {\n const previousHorizontal = pointer[PointerInput.Horizontal];\n const previousVertical = pointer[PointerInput.Vertical];\n if (deviceType === DeviceType.Mouse) {\n // Mouse; Set pointerId if undefined\n if (evt.pointerId === undefined) {\n evt.pointerId = this._mouseId;\n }\n if (!document.pointerLockElement) {\n try {\n this._elementToAttachTo.setPointerCapture(this._mouseId);\n }\n catch (e) {\n // DO NOTHING\n }\n }\n }\n else {\n // Touch; Since touches are dynamically assigned, only set capture if we have an id\n if (evt.pointerId && !document.pointerLockElement) {\n try {\n this._elementToAttachTo.setPointerCapture(evt.pointerId);\n }\n catch (e) {\n // DO NOTHING\n }\n }\n }\n pointer[PointerInput.Horizontal] = evt.clientX;\n pointer[PointerInput.Vertical] = evt.clientY;\n pointer[evt.button + 2] = 1;\n const deviceEvent = evt;\n // NOTE: The +2 used here to is because PointerInput has the same value progression for its mouse buttons as PointerEvent.button\n // However, we have our X and Y values front-loaded to group together the touch inputs but not break this progression\n // EG. ([X, Y, Left-click], Middle-click, etc...)\n deviceEvent.inputIndex = evt.button + 2;\n this._onInputChanged(deviceType, deviceSlot, deviceEvent);\n if (previousHorizontal !== evt.clientX || previousVertical !== evt.clientY) {\n deviceEvent.inputIndex = PointerInput.Move;\n this._onInputChanged(deviceType, deviceSlot, deviceEvent);\n }\n }\n };\n this._pointerUpEvent = (evt) => {\n var _a, _b, _c, _d, _e;\n const deviceType = this._getPointerType(evt);\n const deviceSlot = deviceType === DeviceType.Mouse ? 0 : this._activeTouchIds.indexOf(evt.pointerId);\n if (deviceType === DeviceType.Touch) {\n if (deviceSlot === -1) {\n return;\n }\n else {\n this._activeTouchIds[deviceSlot] = -1;\n }\n }\n const pointer = (_a = this._inputs[deviceType]) === null || _a === void 0 ? void 0 : _a[deviceSlot];\n if (pointer && pointer[evt.button + 2] !== 0) {\n const previousHorizontal = pointer[PointerInput.Horizontal];\n const previousVertical = pointer[PointerInput.Vertical];\n pointer[PointerInput.Horizontal] = evt.clientX;\n pointer[PointerInput.Vertical] = evt.clientY;\n pointer[evt.button + 2] = 0;\n const deviceEvent = evt;\n if (evt.pointerId === undefined) {\n evt.pointerId = this._mouseId;\n }\n if (previousHorizontal !== evt.clientX || previousVertical !== evt.clientY) {\n deviceEvent.inputIndex = PointerInput.Move;\n this._onInputChanged(deviceType, deviceSlot, deviceEvent);\n }\n // NOTE: The +2 used here to is because PointerInput has the same value progression for its mouse buttons as PointerEvent.button\n // However, we have our X and Y values front-loaded to group together the touch inputs but not break this progression\n // EG. ([X, Y, Left-click], Middle-click, etc...)\n deviceEvent.inputIndex = evt.button + 2;\n if (deviceType === DeviceType.Mouse && this._mouseId >= 0 && ((_c = (_b = this._elementToAttachTo).hasPointerCapture) === null || _c === void 0 ? void 0 : _c.call(_b, this._mouseId))) {\n this._elementToAttachTo.releasePointerCapture(this._mouseId);\n }\n else if (evt.pointerId && ((_e = (_d = this._elementToAttachTo).hasPointerCapture) === null || _e === void 0 ? void 0 : _e.call(_d, evt.pointerId))) {\n this._elementToAttachTo.releasePointerCapture(evt.pointerId);\n }\n this._onInputChanged(deviceType, deviceSlot, deviceEvent);\n if (deviceType === DeviceType.Touch) {\n this._onDeviceDisconnected(deviceType, deviceSlot);\n }\n }\n };\n this._pointerCancelEvent = (evt) => {\n var _a, _b, _c, _d;\n if (evt.pointerType === \"mouse\") {\n const pointer = this._inputs[DeviceType.Mouse][0];\n if (this._mouseId >= 0 && ((_b = (_a = this._elementToAttachTo).hasPointerCapture) === null || _b === void 0 ? void 0 : _b.call(_a, this._mouseId))) {\n this._elementToAttachTo.releasePointerCapture(this._mouseId);\n }\n for (let inputIndex = PointerInput.LeftClick; inputIndex <= PointerInput.BrowserForward; inputIndex++) {\n if (pointer[inputIndex] === 1) {\n pointer[inputIndex] = 0;\n const deviceEvent = DeviceEventFactory.CreateDeviceEvent(DeviceType.Mouse, 0, inputIndex, 0, this, this._elementToAttachTo);\n this._onInputChanged(DeviceType.Mouse, 0, deviceEvent);\n }\n }\n }\n else {\n const deviceSlot = this._activeTouchIds.indexOf(evt.pointerId);\n if ((_d = (_c = this._elementToAttachTo).hasPointerCapture) === null || _d === void 0 ? void 0 : _d.call(_c, evt.pointerId)) {\n this._elementToAttachTo.releasePointerCapture(evt.pointerId);\n }\n this._inputs[DeviceType.Touch][deviceSlot][PointerInput.LeftClick] = 0;\n const deviceEvent = DeviceEventFactory.CreateDeviceEvent(DeviceType.Touch, deviceSlot, PointerInput.LeftClick, 0, this, this._elementToAttachTo, evt.pointerId);\n this._onInputChanged(DeviceType.Touch, deviceSlot, deviceEvent);\n this._activeTouchIds[deviceSlot] = -1;\n this._onDeviceDisconnected(DeviceType.Touch, deviceSlot);\n }\n };\n // Set Wheel Event Name, code originally from scene.inputManager\n this._wheelEventName =\n \"onwheel\" in document.createElement(\"div\")\n ? \"wheel\" // Modern browsers support \"wheel\"\n : document.onmousewheel !== undefined\n ? \"mousewheel\" // Webkit and IE support at least \"mousewheel\"\n : \"DOMMouseScroll\"; // let's assume that remaining browsers are older Firefox\n // Code originally in scene.inputManager.ts\n // Chrome reports warning in console if wheel listener doesn't set an explicit passive option.\n // IE11 only supports captureEvent:boolean, not options:object, and it defaults to false.\n // Feature detection technique copied from: https://github.com/github/eventlistener-polyfill (MIT license)\n let passiveSupported = false;\n const noop = function () { };\n try {\n const options = Object.defineProperty({}, \"passive\", {\n get: function () {\n passiveSupported = true;\n },\n });\n this._elementToAttachTo.addEventListener(\"test\", noop, options);\n this._elementToAttachTo.removeEventListener(\"test\", noop, options);\n }\n catch (e) {\n /* */\n }\n this._pointerBlurEvent = () => {\n var _a, _b, _c, _d, _e;\n // Handle mouse buttons\n if (this.isDeviceAvailable(DeviceType.Mouse)) {\n const pointer = this._inputs[DeviceType.Mouse][0];\n if (this._mouseId >= 0 && ((_b = (_a = this._elementToAttachTo).hasPointerCapture) === null || _b === void 0 ? void 0 : _b.call(_a, this._mouseId))) {\n this._elementToAttachTo.releasePointerCapture(this._mouseId);\n }\n for (let inputIndex = PointerInput.LeftClick; inputIndex <= PointerInput.BrowserForward; inputIndex++) {\n if (pointer[inputIndex] === 1) {\n pointer[inputIndex] = 0;\n const deviceEvent = DeviceEventFactory.CreateDeviceEvent(DeviceType.Mouse, 0, inputIndex, 0, this, this._elementToAttachTo);\n this._onInputChanged(DeviceType.Mouse, 0, deviceEvent);\n }\n }\n }\n // Handle Active Touches\n if (this.isDeviceAvailable(DeviceType.Touch)) {\n const pointer = this._inputs[DeviceType.Touch];\n for (let deviceSlot = 0; deviceSlot < this._activeTouchIds.length; deviceSlot++) {\n const pointerId = this._activeTouchIds[deviceSlot];\n if ((_d = (_c = this._elementToAttachTo).hasPointerCapture) === null || _d === void 0 ? void 0 : _d.call(_c, pointerId)) {\n this._elementToAttachTo.releasePointerCapture(pointerId);\n }\n if (pointerId !== -1 && ((_e = pointer[deviceSlot]) === null || _e === void 0 ? void 0 : _e[PointerInput.LeftClick]) === 1) {\n pointer[deviceSlot][PointerInput.LeftClick] = 0;\n const deviceEvent = DeviceEventFactory.CreateDeviceEvent(DeviceType.Touch, deviceSlot, PointerInput.LeftClick, 0, this, this._elementToAttachTo, pointerId);\n this._onInputChanged(DeviceType.Touch, deviceSlot, deviceEvent);\n this._activeTouchIds[deviceSlot] = -1;\n this._onDeviceDisconnected(DeviceType.Touch, deviceSlot);\n }\n }\n }\n };\n this._pointerWheelEvent = (evt) => {\n const deviceType = DeviceType.Mouse;\n const deviceSlot = 0;\n if (!this._inputs[deviceType]) {\n this._inputs[deviceType] = [];\n }\n if (!this._inputs[deviceType][deviceSlot]) {\n this._pointerActive = true;\n this._registerDevice(deviceType, deviceSlot, MAX_POINTER_INPUTS);\n }\n const pointer = this._inputs[deviceType][deviceSlot];\n if (pointer) {\n pointer[PointerInput.MouseWheelX] = evt.deltaX || 0;\n pointer[PointerInput.MouseWheelY] = evt.deltaY || evt.wheelDelta || 0;\n pointer[PointerInput.MouseWheelZ] = evt.deltaZ || 0;\n const deviceEvent = evt;\n // By default, there is no pointerId for mouse wheel events so we'll add one here\n // This logic was originally in the InputManager but was added here to make the\n // InputManager more platform-agnostic\n if (evt.pointerId === undefined) {\n evt.pointerId = this._mouseId;\n }\n if (pointer[PointerInput.MouseWheelX] !== 0) {\n deviceEvent.inputIndex = PointerInput.MouseWheelX;\n this._onInputChanged(deviceType, deviceSlot, deviceEvent);\n }\n if (pointer[PointerInput.MouseWheelY] !== 0) {\n deviceEvent.inputIndex = PointerInput.MouseWheelY;\n this._onInputChanged(deviceType, deviceSlot, deviceEvent);\n }\n if (pointer[PointerInput.MouseWheelZ] !== 0) {\n deviceEvent.inputIndex = PointerInput.MouseWheelZ;\n this._onInputChanged(deviceType, deviceSlot, deviceEvent);\n }\n }\n };\n this._elementToAttachTo.addEventListener(this._eventPrefix + \"move\", this._pointerMoveEvent);\n this._elementToAttachTo.addEventListener(this._eventPrefix + \"down\", this._pointerDownEvent);\n this._elementToAttachTo.addEventListener(this._eventPrefix + \"up\", this._pointerUpEvent);\n this._elementToAttachTo.addEventListener(this._eventPrefix + \"cancel\", this._pointerCancelEvent);\n this._elementToAttachTo.addEventListener(\"blur\", this._pointerBlurEvent);\n this._elementToAttachTo.addEventListener(this._wheelEventName, this._pointerWheelEvent, passiveSupported ? { passive: false } : false);\n // Since there's no up or down event for mouse wheel or delta x/y, clear mouse values at end of frame\n this._pointerInputClearObserver = this._engine.onEndFrameObservable.add(() => {\n if (this.isDeviceAvailable(DeviceType.Mouse)) {\n const pointer = this._inputs[DeviceType.Mouse][0];\n pointer[PointerInput.MouseWheelX] = 0;\n pointer[PointerInput.MouseWheelY] = 0;\n pointer[PointerInput.MouseWheelZ] = 0;\n }\n });\n }\n /**\n * Handle all actions that come from gamepad interaction\n */\n _handleGamepadActions() {\n this._gamepadConnectedEvent = (evt) => {\n this._addGamePad(evt.gamepad);\n };\n this._gamepadDisconnectedEvent = (evt) => {\n if (this._gamepads) {\n const deviceType = this._getGamepadDeviceType(evt.gamepad.id);\n const deviceSlot = evt.gamepad.index;\n this._unregisterDevice(deviceType, deviceSlot);\n delete this._gamepads[deviceSlot];\n }\n };\n window.addEventListener(\"gamepadconnected\", this._gamepadConnectedEvent);\n window.addEventListener(\"gamepaddisconnected\", this._gamepadDisconnectedEvent);\n }\n /**\n * Update all non-event based devices with each frame\n * @param deviceType Enum specifying device type\n * @param deviceSlot \"Slot\" or index that device is referenced in\n * @param inputIndex Id of input to be checked\n */\n _updateDevice(deviceType, deviceSlot, inputIndex) {\n // Gamepads\n const gp = navigator.getGamepads()[deviceSlot];\n if (gp && deviceType === this._gamepads[deviceSlot]) {\n const device = this._inputs[deviceType][deviceSlot];\n if (inputIndex >= gp.buttons.length) {\n device[inputIndex] = gp.axes[inputIndex - gp.buttons.length].valueOf();\n }\n else {\n device[inputIndex] = gp.buttons[inputIndex].value;\n }\n }\n }\n /**\n * Gets DeviceType from the device name\n * @param deviceName Name of Device from DeviceInputSystem\n * @returns DeviceType enum value\n */\n _getGamepadDeviceType(deviceName) {\n if (deviceName.indexOf(\"054c\") !== -1) {\n // DualShock 4 Gamepad\n return deviceName.indexOf(\"0ce6\") !== -1 ? DeviceType.DualSense : DeviceType.DualShock;\n }\n else if (deviceName.indexOf(\"Xbox One\") !== -1 || deviceName.search(\"Xbox 360\") !== -1 || deviceName.search(\"xinput\") !== -1) {\n // Xbox Gamepad\n return DeviceType.Xbox;\n }\n else if (deviceName.indexOf(\"057e\") !== -1) {\n // Switch Gamepad\n return DeviceType.Switch;\n }\n return DeviceType.Generic;\n }\n /**\n * Get DeviceType from a given pointer/mouse/touch event.\n * @param evt PointerEvent to evaluate\n * @returns DeviceType interpreted from event\n */\n _getPointerType(evt) {\n let deviceType = DeviceType.Mouse;\n if (evt.pointerType === \"touch\" || evt.pointerType === \"pen\" || evt.touches) {\n deviceType = DeviceType.Touch;\n }\n return deviceType;\n }\n }\n\n /**\n * Class that handles all input for a specific device\n */\n class DeviceSource {\n /**\n * Default Constructor\n * @param deviceInputSystem - Reference to DeviceInputSystem\n * @param deviceType - Type of device\n * @param deviceSlot - \"Slot\" or index that device is referenced in\n */\n constructor(deviceInputSystem, \n /** Type of device */\n deviceType, \n /** \"Slot\" or index that device is referenced in */\n deviceSlot = 0) {\n this.deviceType = deviceType;\n this.deviceSlot = deviceSlot;\n // Public Members\n /**\n * Observable to handle device input changes per device\n */\n this.onInputChangedObservable = new Observable$1();\n this._deviceInputSystem = deviceInputSystem;\n }\n /**\n * Get input for specific input\n * @param inputIndex - index of specific input on device\n * @returns Input value from DeviceInputSystem\n */\n getInput(inputIndex) {\n return this._deviceInputSystem.pollInput(this.deviceType, this.deviceSlot, inputIndex);\n }\n }\n\n /** @internal */\n class InternalDeviceSourceManager {\n constructor(engine) {\n this._registeredManagers = new Array();\n this._refCount = 0;\n // Public Functions\n this.registerManager = (manager) => {\n for (let deviceType = 0; deviceType < this._devices.length; deviceType++) {\n const device = this._devices[deviceType];\n for (const deviceSlotKey in device) {\n const deviceSlot = +deviceSlotKey;\n manager._addDevice(new DeviceSource(this._deviceInputSystem, deviceType, deviceSlot));\n }\n }\n this._registeredManagers.push(manager);\n };\n this.unregisterManager = (manager) => {\n const idx = this._registeredManagers.indexOf(manager);\n if (idx > -1) {\n this._registeredManagers.splice(idx, 1);\n }\n };\n const numberOfDeviceTypes = Object.keys(DeviceType).length / 2;\n this._devices = new Array(numberOfDeviceTypes);\n const onDeviceConnected = (deviceType, deviceSlot) => {\n if (!this._devices[deviceType]) {\n this._devices[deviceType] = new Array();\n }\n if (!this._devices[deviceType][deviceSlot]) {\n this._devices[deviceType][deviceSlot] = deviceSlot;\n }\n for (const manager of this._registeredManagers) {\n const deviceSource = new DeviceSource(this._deviceInputSystem, deviceType, deviceSlot);\n manager._addDevice(deviceSource);\n }\n };\n const onDeviceDisconnected = (deviceType, deviceSlot) => {\n var _a;\n if ((_a = this._devices[deviceType]) === null || _a === void 0 ? void 0 : _a[deviceSlot]) {\n delete this._devices[deviceType][deviceSlot];\n }\n for (const manager of this._registeredManagers) {\n manager._removeDevice(deviceType, deviceSlot);\n }\n };\n const onInputChanged = (deviceType, deviceSlot, eventData) => {\n if (eventData) {\n for (const manager of this._registeredManagers) {\n manager._onInputChanged(deviceType, deviceSlot, eventData);\n }\n }\n };\n if (typeof _native !== \"undefined\") {\n this._deviceInputSystem = new NativeDeviceInputSystem(onDeviceConnected, onDeviceDisconnected, onInputChanged);\n }\n else {\n this._deviceInputSystem = new WebDeviceInputSystem(engine, onDeviceConnected, onDeviceDisconnected, onInputChanged);\n }\n }\n dispose() {\n this._deviceInputSystem.dispose();\n }\n }\n\n /**\n * Class to keep track of devices\n */\n class DeviceSourceManager {\n // Public Functions\n /**\n * Gets a DeviceSource, given a type and slot\n * @param deviceType - Type of Device\n * @param deviceSlot - Slot or ID of device\n * @returns DeviceSource\n */\n getDeviceSource(deviceType, deviceSlot) {\n if (deviceSlot === undefined) {\n if (this._firstDevice[deviceType] === undefined) {\n return null;\n }\n deviceSlot = this._firstDevice[deviceType];\n }\n if (!this._devices[deviceType] || this._devices[deviceType][deviceSlot] === undefined) {\n return null;\n }\n return this._devices[deviceType][deviceSlot];\n }\n /**\n * Gets an array of DeviceSource objects for a given device type\n * @param deviceType - Type of Device\n * @returns All available DeviceSources of a given type\n */\n getDeviceSources(deviceType) {\n // If device type hasn't had any devices connected yet, return empty array.\n if (!this._devices[deviceType]) {\n return [];\n }\n return this._devices[deviceType].filter((source) => {\n return !!source;\n });\n }\n /**\n * Default constructor\n * @param engine - Used to get canvas (if applicable)\n */\n constructor(engine) {\n const numberOfDeviceTypes = Object.keys(DeviceType).length / 2;\n this._devices = new Array(numberOfDeviceTypes);\n this._firstDevice = new Array(numberOfDeviceTypes);\n this._engine = engine;\n if (!this._engine._deviceSourceManager) {\n this._engine._deviceSourceManager = new InternalDeviceSourceManager(engine);\n }\n this._engine._deviceSourceManager._refCount++;\n // Observables\n this.onDeviceConnectedObservable = new Observable$1((observer) => {\n for (const devices of this._devices) {\n if (devices) {\n for (const device of devices) {\n if (device) {\n this.onDeviceConnectedObservable.notifyObserver(observer, device);\n }\n }\n }\n }\n });\n this.onDeviceDisconnectedObservable = new Observable$1();\n this._engine._deviceSourceManager.registerManager(this);\n this._onDisposeObserver = engine.onDisposeObservable.add(() => {\n this.dispose();\n });\n }\n /**\n * Dispose of DeviceSourceManager\n */\n dispose() {\n // Null out observable refs\n this.onDeviceConnectedObservable.clear();\n this.onDeviceDisconnectedObservable.clear();\n if (this._engine._deviceSourceManager) {\n this._engine._deviceSourceManager.unregisterManager(this);\n if (--this._engine._deviceSourceManager._refCount < 1) {\n this._engine._deviceSourceManager.dispose();\n delete this._engine._deviceSourceManager;\n }\n }\n this._engine.onDisposeObservable.remove(this._onDisposeObserver);\n }\n // Hidden Functions\n /**\n * @param deviceSource - Source to add\n * @internal\n */\n _addDevice(deviceSource) {\n if (!this._devices[deviceSource.deviceType]) {\n this._devices[deviceSource.deviceType] = new Array();\n }\n if (!this._devices[deviceSource.deviceType][deviceSource.deviceSlot]) {\n this._devices[deviceSource.deviceType][deviceSource.deviceSlot] = deviceSource;\n this._updateFirstDevices(deviceSource.deviceType);\n }\n this.onDeviceConnectedObservable.notifyObservers(deviceSource);\n }\n /**\n * @param deviceType - DeviceType\n * @param deviceSlot - DeviceSlot\n * @internal\n */\n _removeDevice(deviceType, deviceSlot) {\n var _a, _b;\n const deviceSource = (_a = this._devices[deviceType]) === null || _a === void 0 ? void 0 : _a[deviceSlot]; // Grab local reference to use before removing from devices\n this.onDeviceDisconnectedObservable.notifyObservers(deviceSource);\n if ((_b = this._devices[deviceType]) === null || _b === void 0 ? void 0 : _b[deviceSlot]) {\n delete this._devices[deviceType][deviceSlot];\n }\n // Even if we don't delete a device, we should still check for the first device as things may have gotten out of sync.\n this._updateFirstDevices(deviceType);\n }\n /**\n * @param deviceType - DeviceType\n * @param deviceSlot - DeviceSlot\n * @param eventData - Event\n * @internal\n */\n _onInputChanged(deviceType, deviceSlot, eventData) {\n var _a, _b;\n (_b = (_a = this._devices[deviceType]) === null || _a === void 0 ? void 0 : _a[deviceSlot]) === null || _b === void 0 ? void 0 : _b.onInputChangedObservable.notifyObservers(eventData);\n }\n // Private Functions\n _updateFirstDevices(type) {\n switch (type) {\n case DeviceType.Keyboard:\n case DeviceType.Mouse:\n this._firstDevice[type] = 0;\n break;\n case DeviceType.Touch:\n case DeviceType.DualSense:\n case DeviceType.DualShock:\n case DeviceType.Xbox:\n case DeviceType.Switch:\n case DeviceType.Generic: {\n delete this._firstDevice[type];\n // eslint-disable-next-line no-case-declarations\n const devices = this._devices[type];\n if (devices) {\n for (let i = 0; i < devices.length; i++) {\n if (devices[i]) {\n this._firstDevice[type] = i;\n break;\n }\n }\n }\n break;\n }\n }\n }\n }\n\n /** @internal */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class _ClickInfo {\n constructor() {\n this._singleClick = false;\n this._doubleClick = false;\n this._hasSwiped = false;\n this._ignore = false;\n }\n get singleClick() {\n return this._singleClick;\n }\n get doubleClick() {\n return this._doubleClick;\n }\n get hasSwiped() {\n return this._hasSwiped;\n }\n get ignore() {\n return this._ignore;\n }\n set singleClick(b) {\n this._singleClick = b;\n }\n set doubleClick(b) {\n this._doubleClick = b;\n }\n set hasSwiped(b) {\n this._hasSwiped = b;\n }\n set ignore(b) {\n this._ignore = b;\n }\n }\n /**\n * Class used to manage all inputs for the scene.\n */\n class InputManager {\n /**\n * Creates a new InputManager\n * @param scene - defines the hosting scene\n */\n constructor(scene) {\n /** This is a defensive check to not allow control attachment prior to an already active one. If already attached, previous control is unattached before attaching the new one. */\n this._alreadyAttached = false;\n this._meshPickProceed = false;\n this._currentPickResult = null;\n this._previousPickResult = null;\n this._totalPointersPressed = 0;\n this._doubleClickOccured = false;\n this._isSwiping = false;\n this._swipeButtonPressed = -1;\n this._skipPointerTap = false;\n this._isMultiTouchGesture = false;\n this._pointerX = 0;\n this._pointerY = 0;\n this._startingPointerPosition = new Vector2(0, 0);\n this._previousStartingPointerPosition = new Vector2(0, 0);\n this._startingPointerTime = 0;\n this._previousStartingPointerTime = 0;\n this._pointerCaptures = {};\n this._meshUnderPointerId = {};\n this._movePointerInfo = null;\n this._cameraObserverCount = 0;\n this._delayedClicks = [null, null, null, null, null];\n this._deviceSourceManager = null;\n this._scene = scene || EngineStore.LastCreatedScene;\n if (!this._scene) {\n return;\n }\n }\n /**\n * Gets the mesh that is currently under the pointer\n * @returns Mesh that the pointer is pointer is hovering over\n */\n get meshUnderPointer() {\n if (this._movePointerInfo) {\n // Because _pointerOverMesh is populated as part of _pickMove, we need to force a pick to update it.\n // Calling _pickMove calls _setCursorAndPointerOverMesh which calls setPointerOverMesh\n this._movePointerInfo._generatePickInfo();\n // Once we have what we need, we can clear _movePointerInfo because we don't need it anymore\n this._movePointerInfo = null;\n }\n return this._pointerOverMesh;\n }\n /**\n * When using more than one pointer (for example in XR) you can get the mesh under the specific pointer\n * @param pointerId - the pointer id to use\n * @returns The mesh under this pointer id or null if not found\n */\n getMeshUnderPointerByPointerId(pointerId) {\n return this._meshUnderPointerId[pointerId] || null;\n }\n /**\n * Gets the pointer coordinates in 2D without any translation (ie. straight out of the pointer event)\n * @returns Vector with X/Y values directly from pointer event\n */\n get unTranslatedPointer() {\n return new Vector2(this._unTranslatedPointerX, this._unTranslatedPointerY);\n }\n /**\n * Gets or sets the current on-screen X position of the pointer\n * @returns Translated X with respect to screen\n */\n get pointerX() {\n return this._pointerX;\n }\n set pointerX(value) {\n this._pointerX = value;\n }\n /**\n * Gets or sets the current on-screen Y position of the pointer\n * @returns Translated Y with respect to screen\n */\n get pointerY() {\n return this._pointerY;\n }\n set pointerY(value) {\n this._pointerY = value;\n }\n _updatePointerPosition(evt) {\n const canvasRect = this._scene.getEngine().getInputElementClientRect();\n if (!canvasRect) {\n return;\n }\n this._pointerX = evt.clientX - canvasRect.left;\n this._pointerY = evt.clientY - canvasRect.top;\n this._unTranslatedPointerX = this._pointerX;\n this._unTranslatedPointerY = this._pointerY;\n }\n _processPointerMove(pickResult, evt) {\n const scene = this._scene;\n const engine = scene.getEngine();\n const canvas = engine.getInputElement();\n if (canvas) {\n canvas.tabIndex = engine.canvasTabIndex;\n // Restore pointer\n if (!scene.doNotHandleCursors) {\n canvas.style.cursor = scene.defaultCursor;\n }\n }\n this._setCursorAndPointerOverMesh(pickResult, evt, scene);\n for (const step of scene._pointerMoveStage) {\n // If _pointerMoveState is defined, we have an active spriteManager and can't use Lazy Picking\n // Therefore, we need to force a pick to update the pickResult\n pickResult = pickResult || this._pickMove(evt);\n const isMeshPicked = (pickResult === null || pickResult === void 0 ? void 0 : pickResult.pickedMesh) ? true : false;\n pickResult = step.action(this._unTranslatedPointerX, this._unTranslatedPointerY, pickResult, isMeshPicked, canvas);\n }\n const type = evt.inputIndex >= PointerInput.MouseWheelX && evt.inputIndex <= PointerInput.MouseWheelZ ? PointerEventTypes.POINTERWHEEL : PointerEventTypes.POINTERMOVE;\n if (scene.onPointerMove) {\n // Because of lazy picking, we need to force a pick to update the pickResult\n pickResult = pickResult || this._pickMove(evt);\n scene.onPointerMove(evt, pickResult, type);\n }\n let pointerInfo;\n if (pickResult) {\n pointerInfo = new PointerInfo(type, evt, pickResult);\n this._setRayOnPointerInfo(pickResult, evt);\n }\n else {\n pointerInfo = new PointerInfo(type, evt, null, this);\n this._movePointerInfo = pointerInfo;\n }\n if (scene.onPointerObservable.hasObservers()) {\n scene.onPointerObservable.notifyObservers(pointerInfo, type);\n }\n }\n // Pointers handling\n /** @internal */\n _setRayOnPointerInfo(pickInfo, event) {\n const scene = this._scene;\n if (pickInfo && scene._pickingAvailable) {\n if (!pickInfo.ray) {\n pickInfo.ray = scene.createPickingRay(event.offsetX, event.offsetY, Matrix.Identity(), scene.activeCamera);\n }\n }\n }\n /** @internal */\n _addCameraPointerObserver(observer, mask) {\n this._cameraObserverCount++;\n return this._scene.onPointerObservable.add(observer, mask);\n }\n /** @internal */\n _removeCameraPointerObserver(observer) {\n this._cameraObserverCount--;\n return this._scene.onPointerObservable.remove(observer);\n }\n _checkForPicking() {\n return !!(this._scene.onPointerObservable.observers.length > this._cameraObserverCount || this._scene.onPointerPick);\n }\n _checkPrePointerObservable(pickResult, evt, type) {\n const scene = this._scene;\n const pi = new PointerInfoPre(type, evt, this._unTranslatedPointerX, this._unTranslatedPointerY);\n if (pickResult) {\n pi.originalPickingInfo = pickResult;\n pi.ray = pickResult.ray;\n if (pickResult.originMesh) {\n pi.nearInteractionPickingInfo = pickResult;\n }\n }\n scene.onPrePointerObservable.notifyObservers(pi, type);\n if (pi.skipOnPointerObservable) {\n return true;\n }\n else {\n return false;\n }\n }\n /** @internal */\n _pickMove(evt) {\n const scene = this._scene;\n const pickResult = scene.pick(this._unTranslatedPointerX, this._unTranslatedPointerY, scene.pointerMovePredicate, scene.pointerMoveFastCheck, scene.cameraToUseForPointers, scene.pointerMoveTrianglePredicate);\n this._setCursorAndPointerOverMesh(pickResult, evt, scene);\n return pickResult;\n }\n _setCursorAndPointerOverMesh(pickResult, evt, scene) {\n const engine = scene.getEngine();\n const canvas = engine.getInputElement();\n if (pickResult === null || pickResult === void 0 ? void 0 : pickResult.pickedMesh) {\n this.setPointerOverMesh(pickResult.pickedMesh, evt.pointerId, pickResult, evt);\n if (!scene.doNotHandleCursors && canvas && this._pointerOverMesh) {\n const actionManager = this._pointerOverMesh._getActionManagerForTrigger();\n if (actionManager && actionManager.hasPointerTriggers) {\n canvas.style.cursor = actionManager.hoverCursor || scene.hoverCursor;\n }\n }\n }\n else {\n this.setPointerOverMesh(null, evt.pointerId, pickResult, evt);\n }\n }\n /**\n * Use this method to simulate a pointer move on a mesh\n * The pickResult parameter can be obtained from a scene.pick or scene.pickWithRay\n * @param pickResult - pickingInfo of the object wished to simulate pointer event on\n * @param pointerEventInit - pointer event state to be used when simulating the pointer event (eg. pointer id for multitouch)\n */\n simulatePointerMove(pickResult, pointerEventInit) {\n const evt = new PointerEvent(\"pointermove\", pointerEventInit);\n evt.inputIndex = PointerInput.Move;\n if (this._checkPrePointerObservable(pickResult, evt, PointerEventTypes.POINTERMOVE)) {\n return;\n }\n this._processPointerMove(pickResult, evt);\n }\n /**\n * Use this method to simulate a pointer down on a mesh\n * The pickResult parameter can be obtained from a scene.pick or scene.pickWithRay\n * @param pickResult - pickingInfo of the object wished to simulate pointer event on\n * @param pointerEventInit - pointer event state to be used when simulating the pointer event (eg. pointer id for multitouch)\n */\n simulatePointerDown(pickResult, pointerEventInit) {\n const evt = new PointerEvent(\"pointerdown\", pointerEventInit);\n evt.inputIndex = evt.button + 2;\n if (this._checkPrePointerObservable(pickResult, evt, PointerEventTypes.POINTERDOWN)) {\n return;\n }\n this._processPointerDown(pickResult, evt);\n }\n _processPointerDown(pickResult, evt) {\n const scene = this._scene;\n if (pickResult === null || pickResult === void 0 ? void 0 : pickResult.pickedMesh) {\n this._pickedDownMesh = pickResult.pickedMesh;\n const actionManager = pickResult.pickedMesh._getActionManagerForTrigger();\n if (actionManager) {\n if (actionManager.hasPickTriggers) {\n actionManager.processTrigger(5, ActionEvent.CreateNew(pickResult.pickedMesh, evt, pickResult));\n switch (evt.button) {\n case 0:\n actionManager.processTrigger(2, ActionEvent.CreateNew(pickResult.pickedMesh, evt, pickResult));\n break;\n case 1:\n actionManager.processTrigger(4, ActionEvent.CreateNew(pickResult.pickedMesh, evt, pickResult));\n break;\n case 2:\n actionManager.processTrigger(3, ActionEvent.CreateNew(pickResult.pickedMesh, evt, pickResult));\n break;\n }\n }\n if (actionManager.hasSpecificTrigger(8)) {\n window.setTimeout(() => {\n const pickResult = scene.pick(this._unTranslatedPointerX, this._unTranslatedPointerY, (mesh) => ((mesh.isPickable &&\n mesh.isVisible &&\n mesh.isReady() &&\n mesh.actionManager &&\n mesh.actionManager.hasSpecificTrigger(8) &&\n mesh === this._pickedDownMesh)), false, scene.cameraToUseForPointers);\n if ((pickResult === null || pickResult === void 0 ? void 0 : pickResult.pickedMesh) && actionManager) {\n if (this._totalPointersPressed !== 0 && Date.now() - this._startingPointerTime > InputManager.LongPressDelay && !this._isPointerSwiping()) {\n this._startingPointerTime = 0;\n actionManager.processTrigger(8, ActionEvent.CreateNew(pickResult.pickedMesh, evt));\n }\n }\n }, InputManager.LongPressDelay);\n }\n }\n }\n else {\n for (const step of scene._pointerDownStage) {\n pickResult = step.action(this._unTranslatedPointerX, this._unTranslatedPointerY, pickResult, evt, false);\n }\n }\n let pointerInfo;\n const type = PointerEventTypes.POINTERDOWN;\n if (pickResult) {\n if (scene.onPointerDown) {\n scene.onPointerDown(evt, pickResult, type);\n }\n pointerInfo = new PointerInfo(type, evt, pickResult);\n this._setRayOnPointerInfo(pickResult, evt);\n }\n else {\n pointerInfo = new PointerInfo(type, evt, null, this);\n }\n if (scene.onPointerObservable.hasObservers()) {\n scene.onPointerObservable.notifyObservers(pointerInfo, type);\n }\n }\n /**\n * @internal\n * @internals Boolean if delta for pointer exceeds drag movement threshold\n */\n _isPointerSwiping() {\n return this._isSwiping;\n }\n /**\n * Use this method to simulate a pointer up on a mesh\n * The pickResult parameter can be obtained from a scene.pick or scene.pickWithRay\n * @param pickResult - pickingInfo of the object wished to simulate pointer event on\n * @param pointerEventInit - pointer event state to be used when simulating the pointer event (eg. pointer id for multitouch)\n * @param doubleTap - indicates that the pointer up event should be considered as part of a double click (false by default)\n */\n simulatePointerUp(pickResult, pointerEventInit, doubleTap) {\n const evt = new PointerEvent(\"pointerup\", pointerEventInit);\n evt.inputIndex = PointerInput.Move;\n const clickInfo = new _ClickInfo();\n if (doubleTap) {\n clickInfo.doubleClick = true;\n }\n else {\n clickInfo.singleClick = true;\n }\n if (this._checkPrePointerObservable(pickResult, evt, PointerEventTypes.POINTERUP)) {\n return;\n }\n this._processPointerUp(pickResult, evt, clickInfo);\n }\n _processPointerUp(pickResult, evt, clickInfo) {\n const scene = this._scene;\n if (pickResult === null || pickResult === void 0 ? void 0 : pickResult.pickedMesh) {\n this._pickedUpMesh = pickResult.pickedMesh;\n if (this._pickedDownMesh === this._pickedUpMesh) {\n if (scene.onPointerPick) {\n scene.onPointerPick(evt, pickResult);\n }\n if (clickInfo.singleClick && !clickInfo.ignore && scene.onPointerObservable.observers.length > this._cameraObserverCount) {\n const type = PointerEventTypes.POINTERPICK;\n const pi = new PointerInfo(type, evt, pickResult);\n this._setRayOnPointerInfo(pickResult, evt);\n scene.onPointerObservable.notifyObservers(pi, type);\n }\n }\n const actionManager = pickResult.pickedMesh._getActionManagerForTrigger();\n if (actionManager && !clickInfo.ignore) {\n actionManager.processTrigger(7, ActionEvent.CreateNew(pickResult.pickedMesh, evt, pickResult));\n if (!clickInfo.hasSwiped && clickInfo.singleClick) {\n actionManager.processTrigger(1, ActionEvent.CreateNew(pickResult.pickedMesh, evt, pickResult));\n }\n const doubleClickActionManager = pickResult.pickedMesh._getActionManagerForTrigger(6);\n if (clickInfo.doubleClick && doubleClickActionManager) {\n doubleClickActionManager.processTrigger(6, ActionEvent.CreateNew(pickResult.pickedMesh, evt, pickResult));\n }\n }\n }\n else {\n if (!clickInfo.ignore) {\n for (const step of scene._pointerUpStage) {\n pickResult = step.action(this._unTranslatedPointerX, this._unTranslatedPointerY, pickResult, evt, clickInfo.doubleClick);\n }\n }\n }\n if (this._pickedDownMesh && this._pickedDownMesh !== this._pickedUpMesh) {\n const pickedDownActionManager = this._pickedDownMesh._getActionManagerForTrigger(16);\n if (pickedDownActionManager) {\n pickedDownActionManager.processTrigger(16, ActionEvent.CreateNew(this._pickedDownMesh, evt));\n }\n }\n if (!clickInfo.ignore) {\n const pi = new PointerInfo(PointerEventTypes.POINTERUP, evt, pickResult);\n // Set ray on picking info. Note that this info will also be reused for the tap notification.\n this._setRayOnPointerInfo(pickResult, evt);\n scene.onPointerObservable.notifyObservers(pi, PointerEventTypes.POINTERUP);\n if (scene.onPointerUp) {\n scene.onPointerUp(evt, pickResult, PointerEventTypes.POINTERUP);\n }\n if (!clickInfo.hasSwiped && !this._skipPointerTap && !this._isMultiTouchGesture) {\n let type = 0;\n if (clickInfo.singleClick) {\n type = PointerEventTypes.POINTERTAP;\n }\n else if (clickInfo.doubleClick) {\n type = PointerEventTypes.POINTERDOUBLETAP;\n }\n if (type) {\n const pi = new PointerInfo(type, evt, pickResult);\n if (scene.onPointerObservable.hasObservers() && scene.onPointerObservable.hasSpecificMask(type)) {\n scene.onPointerObservable.notifyObservers(pi, type);\n }\n }\n }\n }\n }\n /**\n * Gets a boolean indicating if the current pointer event is captured (meaning that the scene has already handled the pointer down)\n * @param pointerId - defines the pointer id to use in a multi-touch scenario (0 by default)\n * @returns true if the pointer was captured\n */\n isPointerCaptured(pointerId = 0) {\n return this._pointerCaptures[pointerId];\n }\n /**\n * Attach events to the canvas (To handle actionManagers triggers and raise onPointerMove, onPointerDown and onPointerUp\n * @param attachUp - defines if you want to attach events to pointerup\n * @param attachDown - defines if you want to attach events to pointerdown\n * @param attachMove - defines if you want to attach events to pointermove\n * @param elementToAttachTo - defines the target DOM element to attach to (will use the canvas by default)\n */\n attachControl(attachUp = true, attachDown = true, attachMove = true, elementToAttachTo = null) {\n const scene = this._scene;\n const engine = scene.getEngine();\n if (!elementToAttachTo) {\n elementToAttachTo = engine.getInputElement();\n }\n if (this._alreadyAttached) {\n this.detachControl();\n }\n if (elementToAttachTo) {\n this._alreadyAttachedTo = elementToAttachTo;\n }\n this._deviceSourceManager = new DeviceSourceManager(engine);\n // Because this is only called from _initClickEvent, which is called in _onPointerUp, we'll use the pointerUpPredicate for the pick call\n this._initActionManager = (act) => {\n if (!this._meshPickProceed) {\n const pickResult = scene.skipPointerUpPicking || (scene._registeredActions === 0 && !this._checkForPicking() && !scene.onPointerUp)\n ? null\n : scene.pick(this._unTranslatedPointerX, this._unTranslatedPointerY, scene.pointerUpPredicate, scene.pointerUpFastCheck, scene.cameraToUseForPointers);\n this._currentPickResult = pickResult;\n if (pickResult) {\n act = pickResult.hit && pickResult.pickedMesh ? pickResult.pickedMesh._getActionManagerForTrigger() : null;\n }\n this._meshPickProceed = true;\n }\n return act;\n };\n this._delayedSimpleClick = (btn, clickInfo, cb) => {\n // double click delay is over and that no double click has been raised since, or the 2 consecutive keys pressed are different\n if ((Date.now() - this._previousStartingPointerTime > InputManager.DoubleClickDelay && !this._doubleClickOccured) || btn !== this._previousButtonPressed) {\n this._doubleClickOccured = false;\n clickInfo.singleClick = true;\n clickInfo.ignore = false;\n // If we have a delayed click, we need to resolve the TAP event\n if (this._delayedClicks[btn]) {\n const evt = this._delayedClicks[btn].evt;\n const type = PointerEventTypes.POINTERTAP;\n const pi = new PointerInfo(type, evt, this._currentPickResult);\n if (scene.onPointerObservable.hasObservers() && scene.onPointerObservable.hasSpecificMask(type)) {\n scene.onPointerObservable.notifyObservers(pi, type);\n }\n // Clear the delayed click\n this._delayedClicks[btn] = null;\n }\n }\n };\n this._initClickEvent = (obs1, obs2, evt, cb) => {\n var _a, _b;\n const clickInfo = new _ClickInfo();\n this._currentPickResult = null;\n let act = null;\n let checkPicking = obs1.hasSpecificMask(PointerEventTypes.POINTERPICK) ||\n obs2.hasSpecificMask(PointerEventTypes.POINTERPICK) ||\n obs1.hasSpecificMask(PointerEventTypes.POINTERTAP) ||\n obs2.hasSpecificMask(PointerEventTypes.POINTERTAP) ||\n obs1.hasSpecificMask(PointerEventTypes.POINTERDOUBLETAP) ||\n obs2.hasSpecificMask(PointerEventTypes.POINTERDOUBLETAP);\n if (!checkPicking && AbstractActionManager) {\n act = this._initActionManager(act, clickInfo);\n if (act) {\n checkPicking = act.hasPickTriggers;\n }\n }\n let needToIgnoreNext = false;\n if (checkPicking) {\n const btn = evt.button;\n clickInfo.hasSwiped = this._isPointerSwiping();\n if (!clickInfo.hasSwiped) {\n let checkSingleClickImmediately = !InputManager.ExclusiveDoubleClickMode;\n if (!checkSingleClickImmediately) {\n checkSingleClickImmediately = !obs1.hasSpecificMask(PointerEventTypes.POINTERDOUBLETAP) && !obs2.hasSpecificMask(PointerEventTypes.POINTERDOUBLETAP);\n if (checkSingleClickImmediately && !AbstractActionManager.HasSpecificTrigger(6)) {\n act = this._initActionManager(act, clickInfo);\n if (act) {\n checkSingleClickImmediately = !act.hasSpecificTrigger(6);\n }\n }\n }\n if (checkSingleClickImmediately) {\n // single click detected if double click delay is over or two different successive keys pressed without exclusive double click or no double click required\n if (Date.now() - this._previousStartingPointerTime > InputManager.DoubleClickDelay || btn !== this._previousButtonPressed) {\n clickInfo.singleClick = true;\n cb(clickInfo, this._currentPickResult);\n needToIgnoreNext = true;\n }\n }\n // at least one double click is required to be check and exclusive double click is enabled\n else {\n // Queue up a delayed click, just in case this isn't a double click\n // It should be noted that while this delayed event happens\n // because of user input, it shouldn't be considered as a direct,\n // timing-dependent result of that input. It's meant to just fire the TAP event\n const delayedClick = {\n evt: evt,\n clickInfo: clickInfo,\n timeoutId: window.setTimeout(this._delayedSimpleClick.bind(this, btn, clickInfo, cb), InputManager.DoubleClickDelay),\n };\n this._delayedClicks[btn] = delayedClick;\n }\n let checkDoubleClick = obs1.hasSpecificMask(PointerEventTypes.POINTERDOUBLETAP) || obs2.hasSpecificMask(PointerEventTypes.POINTERDOUBLETAP);\n if (!checkDoubleClick && AbstractActionManager.HasSpecificTrigger(6)) {\n act = this._initActionManager(act, clickInfo);\n if (act) {\n checkDoubleClick = act.hasSpecificTrigger(6);\n }\n }\n if (checkDoubleClick) {\n // two successive keys pressed are equal, double click delay is not over and double click has not just occurred\n if (btn === this._previousButtonPressed && Date.now() - this._previousStartingPointerTime < InputManager.DoubleClickDelay && !this._doubleClickOccured) {\n // pointer has not moved for 2 clicks, it's a double click\n if (!clickInfo.hasSwiped && !this._isPointerSwiping()) {\n this._previousStartingPointerTime = 0;\n this._doubleClickOccured = true;\n clickInfo.doubleClick = true;\n clickInfo.ignore = false;\n // If we have a pending click, we need to cancel it\n if (InputManager.ExclusiveDoubleClickMode && this._delayedClicks[btn]) {\n clearTimeout((_a = this._delayedClicks[btn]) === null || _a === void 0 ? void 0 : _a.timeoutId);\n this._delayedClicks[btn] = null;\n }\n cb(clickInfo, this._currentPickResult);\n }\n // if the two successive clicks are too far, it's just two simple clicks\n else {\n this._doubleClickOccured = false;\n this._previousStartingPointerTime = this._startingPointerTime;\n this._previousStartingPointerPosition.x = this._startingPointerPosition.x;\n this._previousStartingPointerPosition.y = this._startingPointerPosition.y;\n this._previousButtonPressed = btn;\n if (InputManager.ExclusiveDoubleClickMode) {\n // If we have a delayed click, we need to cancel it\n if (this._delayedClicks[btn]) {\n clearTimeout((_b = this._delayedClicks[btn]) === null || _b === void 0 ? void 0 : _b.timeoutId);\n this._delayedClicks[btn] = null;\n }\n cb(clickInfo, this._previousPickResult);\n }\n else {\n cb(clickInfo, this._currentPickResult);\n }\n }\n needToIgnoreNext = true;\n }\n // just the first click of the double has been raised\n else {\n this._doubleClickOccured = false;\n this._previousStartingPointerTime = this._startingPointerTime;\n this._previousStartingPointerPosition.x = this._startingPointerPosition.x;\n this._previousStartingPointerPosition.y = this._startingPointerPosition.y;\n this._previousButtonPressed = btn;\n }\n }\n }\n }\n // Even if ExclusiveDoubleClickMode is true, we need to always handle\n // up events at time of execution, unless we're explicitly ignoring them.\n if (!needToIgnoreNext) {\n cb(clickInfo, this._currentPickResult);\n }\n };\n this._onPointerMove = (evt) => {\n this._updatePointerPosition(evt);\n // Check if pointer leaves DragMovementThreshold range to determine if swipe is occurring\n if (!this._isSwiping && this._swipeButtonPressed !== -1) {\n this._isSwiping =\n Math.abs(this._startingPointerPosition.x - this._pointerX) > InputManager.DragMovementThreshold ||\n Math.abs(this._startingPointerPosition.y - this._pointerY) > InputManager.DragMovementThreshold;\n }\n // Because there's a race condition between pointermove and pointerlockchange events, we need to\n // verify that the pointer is still locked after each pointermove event.\n if (engine.isPointerLock) {\n engine._verifyPointerLock();\n }\n // PreObservable support\n if (this._checkPrePointerObservable(null, evt, evt.inputIndex >= PointerInput.MouseWheelX && evt.inputIndex <= PointerInput.MouseWheelZ ? PointerEventTypes.POINTERWHEEL : PointerEventTypes.POINTERMOVE)) {\n return;\n }\n if (!scene.cameraToUseForPointers && !scene.activeCamera) {\n return;\n }\n if (scene.skipPointerMovePicking) {\n this._processPointerMove(new PickingInfo(), evt);\n return;\n }\n if (!scene.pointerMovePredicate) {\n scene.pointerMovePredicate = (mesh) => mesh.isPickable &&\n mesh.isVisible &&\n mesh.isReady() &&\n mesh.isEnabled() &&\n (mesh.enablePointerMoveEvents || scene.constantlyUpdateMeshUnderPointer || mesh._getActionManagerForTrigger() !== null) &&\n (!scene.cameraToUseForPointers || (scene.cameraToUseForPointers.layerMask & mesh.layerMask) !== 0);\n }\n const pickResult = scene._registeredActions > 0 || scene.constantlyUpdateMeshUnderPointer ? this._pickMove(evt) : null;\n this._processPointerMove(pickResult, evt);\n };\n this._onPointerDown = (evt) => {\n var _a;\n this._totalPointersPressed++;\n this._pickedDownMesh = null;\n this._meshPickProceed = false;\n // If ExclusiveDoubleClickMode is true, we need to resolve any pending delayed clicks\n if (InputManager.ExclusiveDoubleClickMode) {\n for (let i = 0; i < this._delayedClicks.length; i++) {\n if (this._delayedClicks[i]) {\n // If the button that was pressed is the same as the one that was released,\n // just clear the timer. This will be resolved in the up event.\n if (evt.button === i) {\n clearTimeout((_a = this._delayedClicks[i]) === null || _a === void 0 ? void 0 : _a.timeoutId);\n }\n else {\n // Otherwise, we need to resolve the click\n const clickInfo = this._delayedClicks[i].clickInfo;\n this._doubleClickOccured = false;\n clickInfo.singleClick = true;\n clickInfo.ignore = false;\n const prevEvt = this._delayedClicks[i].evt;\n const type = PointerEventTypes.POINTERTAP;\n const pi = new PointerInfo(type, prevEvt, this._currentPickResult);\n if (scene.onPointerObservable.hasObservers() && scene.onPointerObservable.hasSpecificMask(type)) {\n scene.onPointerObservable.notifyObservers(pi, type);\n }\n // Clear the delayed click\n this._delayedClicks[i] = null;\n }\n }\n }\n }\n this._updatePointerPosition(evt);\n if (this._swipeButtonPressed === -1) {\n this._swipeButtonPressed = evt.button;\n }\n if (scene.preventDefaultOnPointerDown && elementToAttachTo) {\n evt.preventDefault();\n elementToAttachTo.focus();\n }\n this._startingPointerPosition.x = this._pointerX;\n this._startingPointerPosition.y = this._pointerY;\n this._startingPointerTime = Date.now();\n // PreObservable support\n if (this._checkPrePointerObservable(null, evt, PointerEventTypes.POINTERDOWN)) {\n return;\n }\n if (!scene.cameraToUseForPointers && !scene.activeCamera) {\n return;\n }\n this._pointerCaptures[evt.pointerId] = true;\n if (!scene.pointerDownPredicate) {\n scene.pointerDownPredicate = (mesh) => {\n return (mesh.isPickable &&\n mesh.isVisible &&\n mesh.isReady() &&\n mesh.isEnabled() &&\n (!scene.cameraToUseForPointers || (scene.cameraToUseForPointers.layerMask & mesh.layerMask) !== 0));\n };\n }\n // Meshes\n this._pickedDownMesh = null;\n let pickResult;\n if (scene.skipPointerDownPicking || (scene._registeredActions === 0 && !this._checkForPicking() && !scene.onPointerDown)) {\n pickResult = new PickingInfo();\n }\n else {\n pickResult = scene.pick(this._unTranslatedPointerX, this._unTranslatedPointerY, scene.pointerDownPredicate, scene.pointerDownFastCheck, scene.cameraToUseForPointers);\n }\n this._processPointerDown(pickResult, evt);\n };\n this._onPointerUp = (evt) => {\n if (this._totalPointersPressed === 0) {\n // We are attaching the pointer up to windows because of a bug in FF\n return; // So we need to test it the pointer down was pressed before.\n }\n this._totalPointersPressed--;\n this._pickedUpMesh = null;\n this._meshPickProceed = false;\n this._updatePointerPosition(evt);\n if (scene.preventDefaultOnPointerUp && elementToAttachTo) {\n evt.preventDefault();\n elementToAttachTo.focus();\n }\n this._initClickEvent(scene.onPrePointerObservable, scene.onPointerObservable, evt, (clickInfo, pickResult) => {\n // PreObservable support\n if (scene.onPrePointerObservable.hasObservers()) {\n this._skipPointerTap = false;\n if (!clickInfo.ignore) {\n if (this._checkPrePointerObservable(null, evt, PointerEventTypes.POINTERUP)) {\n // If we're skipping the next observable, we need to reset the swipe state before returning\n if (this._swipeButtonPressed === evt.button) {\n this._isSwiping = false;\n this._swipeButtonPressed = -1;\n }\n // If we're going to skip the POINTERUP, we need to reset the pointer capture\n if (evt.buttons === 0) {\n this._pointerCaptures[evt.pointerId] = false;\n }\n return;\n }\n if (!clickInfo.hasSwiped) {\n if (clickInfo.singleClick && scene.onPrePointerObservable.hasSpecificMask(PointerEventTypes.POINTERTAP)) {\n if (this._checkPrePointerObservable(null, evt, PointerEventTypes.POINTERTAP)) {\n this._skipPointerTap = true;\n }\n }\n if (clickInfo.doubleClick && scene.onPrePointerObservable.hasSpecificMask(PointerEventTypes.POINTERDOUBLETAP)) {\n if (this._checkPrePointerObservable(null, evt, PointerEventTypes.POINTERDOUBLETAP)) {\n this._skipPointerTap = true;\n }\n }\n }\n }\n }\n // There should be a pointer captured at this point so if there isn't we should reset and return\n if (!this._pointerCaptures[evt.pointerId]) {\n if (this._swipeButtonPressed === evt.button) {\n this._isSwiping = false;\n this._swipeButtonPressed = -1;\n }\n return;\n }\n // Only release capture if all buttons are released\n if (evt.buttons === 0) {\n this._pointerCaptures[evt.pointerId] = false;\n }\n if (!scene.cameraToUseForPointers && !scene.activeCamera) {\n return;\n }\n if (!scene.pointerUpPredicate) {\n scene.pointerUpPredicate = (mesh) => {\n return (mesh.isPickable &&\n mesh.isVisible &&\n mesh.isReady() &&\n mesh.isEnabled() &&\n (!scene.cameraToUseForPointers || (scene.cameraToUseForPointers.layerMask & mesh.layerMask) !== 0));\n };\n }\n // Meshes\n if (!this._meshPickProceed && ((AbstractActionManager && AbstractActionManager.HasTriggers) || this._checkForPicking() || scene.onPointerUp)) {\n this._initActionManager(null, clickInfo);\n }\n if (!pickResult) {\n pickResult = this._currentPickResult;\n }\n this._processPointerUp(pickResult, evt, clickInfo);\n this._previousPickResult = this._currentPickResult;\n if (this._swipeButtonPressed === evt.button) {\n this._isSwiping = false;\n this._swipeButtonPressed = -1;\n }\n });\n };\n this._onKeyDown = (evt) => {\n const type = KeyboardEventTypes.KEYDOWN;\n if (scene.onPreKeyboardObservable.hasObservers()) {\n const pi = new KeyboardInfoPre(type, evt);\n scene.onPreKeyboardObservable.notifyObservers(pi, type);\n if (pi.skipOnKeyboardObservable) {\n return;\n }\n }\n if (scene.onKeyboardObservable.hasObservers()) {\n const pi = new KeyboardInfo(type, evt);\n scene.onKeyboardObservable.notifyObservers(pi, type);\n }\n if (scene.actionManager) {\n scene.actionManager.processTrigger(14, ActionEvent.CreateNewFromScene(scene, evt));\n }\n };\n this._onKeyUp = (evt) => {\n const type = KeyboardEventTypes.KEYUP;\n if (scene.onPreKeyboardObservable.hasObservers()) {\n const pi = new KeyboardInfoPre(type, evt);\n scene.onPreKeyboardObservable.notifyObservers(pi, type);\n if (pi.skipOnKeyboardObservable) {\n return;\n }\n }\n if (scene.onKeyboardObservable.hasObservers()) {\n const pi = new KeyboardInfo(type, evt);\n scene.onKeyboardObservable.notifyObservers(pi, type);\n }\n if (scene.actionManager) {\n scene.actionManager.processTrigger(15, ActionEvent.CreateNewFromScene(scene, evt));\n }\n };\n // If a device connects that we can handle, wire up the observable\n this._deviceSourceManager.onDeviceConnectedObservable.add((deviceSource) => {\n if (deviceSource.deviceType === DeviceType.Mouse) {\n deviceSource.onInputChangedObservable.add((eventData) => {\n if (eventData.inputIndex === PointerInput.LeftClick ||\n eventData.inputIndex === PointerInput.MiddleClick ||\n eventData.inputIndex === PointerInput.RightClick ||\n eventData.inputIndex === PointerInput.BrowserBack ||\n eventData.inputIndex === PointerInput.BrowserForward) {\n if (attachDown && deviceSource.getInput(eventData.inputIndex) === 1) {\n this._onPointerDown(eventData);\n }\n else if (attachUp && deviceSource.getInput(eventData.inputIndex) === 0) {\n this._onPointerUp(eventData);\n }\n }\n else if (attachMove) {\n if (eventData.inputIndex === PointerInput.Move) {\n this._onPointerMove(eventData);\n }\n else if (eventData.inputIndex === PointerInput.MouseWheelX ||\n eventData.inputIndex === PointerInput.MouseWheelY ||\n eventData.inputIndex === PointerInput.MouseWheelZ) {\n this._onPointerMove(eventData);\n }\n }\n });\n }\n else if (deviceSource.deviceType === DeviceType.Touch) {\n deviceSource.onInputChangedObservable.add((eventData) => {\n if (eventData.inputIndex === PointerInput.LeftClick) {\n if (attachDown && deviceSource.getInput(eventData.inputIndex) === 1) {\n this._onPointerDown(eventData);\n if (this._totalPointersPressed > 1) {\n this._isMultiTouchGesture = true;\n }\n }\n else if (attachUp && deviceSource.getInput(eventData.inputIndex) === 0) {\n this._onPointerUp(eventData);\n if (this._totalPointersPressed === 0) {\n this._isMultiTouchGesture = false;\n }\n }\n }\n if (attachMove && eventData.inputIndex === PointerInput.Move) {\n this._onPointerMove(eventData);\n }\n });\n }\n else if (deviceSource.deviceType === DeviceType.Keyboard) {\n deviceSource.onInputChangedObservable.add((eventData) => {\n if (eventData.type === \"keydown\") {\n this._onKeyDown(eventData);\n }\n else if (eventData.type === \"keyup\") {\n this._onKeyUp(eventData);\n }\n });\n }\n });\n this._alreadyAttached = true;\n }\n /**\n * Detaches all event handlers\n */\n detachControl() {\n if (this._alreadyAttached) {\n this._deviceSourceManager.dispose();\n this._deviceSourceManager = null;\n // Cursor\n if (this._alreadyAttachedTo && !this._scene.doNotHandleCursors) {\n this._alreadyAttachedTo.style.cursor = this._scene.defaultCursor;\n }\n this._alreadyAttached = false;\n this._alreadyAttachedTo = null;\n }\n }\n /**\n * Force the value of meshUnderPointer\n * @param mesh - defines the mesh to use\n * @param pointerId - optional pointer id when using more than one pointer. Defaults to 0\n * @param pickResult - optional pickingInfo data used to find mesh\n * @param evt - optional pointer event\n */\n setPointerOverMesh(mesh, pointerId = 0, pickResult, evt) {\n if (this._meshUnderPointerId[pointerId] === mesh && (!mesh || !mesh._internalAbstractMeshDataInfo._pointerOverDisableMeshTesting)) {\n return;\n }\n const underPointerMesh = this._meshUnderPointerId[pointerId];\n let actionManager;\n if (underPointerMesh) {\n actionManager = underPointerMesh._getActionManagerForTrigger(10);\n if (actionManager) {\n actionManager.processTrigger(10, ActionEvent.CreateNew(underPointerMesh, evt, { pointerId }));\n }\n }\n if (mesh) {\n this._meshUnderPointerId[pointerId] = mesh;\n this._pointerOverMesh = mesh;\n actionManager = mesh._getActionManagerForTrigger(9);\n if (actionManager) {\n actionManager.processTrigger(9, ActionEvent.CreateNew(mesh, evt, { pointerId, pickResult }));\n }\n }\n else {\n delete this._meshUnderPointerId[pointerId];\n this._pointerOverMesh = null;\n }\n }\n /**\n * Gets the mesh under the pointer\n * @returns a Mesh or null if no mesh is under the pointer\n */\n getPointerOverMesh() {\n return this.meshUnderPointer;\n }\n /**\n * @param mesh - Mesh to invalidate\n * @internal\n */\n _invalidateMesh(mesh) {\n if (this._pointerOverMesh === mesh) {\n this._pointerOverMesh = null;\n }\n if (this._pickedDownMesh === mesh) {\n this._pickedDownMesh = null;\n }\n if (this._pickedUpMesh === mesh) {\n this._pickedUpMesh = null;\n }\n for (const pointerId in this._meshUnderPointerId) {\n if (this._meshUnderPointerId[pointerId] === mesh) {\n delete this._meshUnderPointerId[pointerId];\n }\n }\n }\n }\n /** The distance in pixel that you have to move to prevent some events */\n InputManager.DragMovementThreshold = 10; // in pixels\n /** Time in milliseconds to wait to raise long press events if button is still pressed */\n InputManager.LongPressDelay = 500; // in milliseconds\n /** Time in milliseconds with two consecutive clicks will be considered as a double click */\n InputManager.DoubleClickDelay = 300; // in milliseconds\n /**\n * This flag will modify the behavior so that, when true, a click will happen if and only if\n * another click DOES NOT happen within the DoubleClickDelay time frame. If another click does\n * happen within that time frame, the first click will not fire an event and and a double click will occur.\n */\n InputManager.ExclusiveDoubleClickMode = false;\n\n /**\n * This class is used to track a performance counter which is number based.\n * The user has access to many properties which give statistics of different nature.\n *\n * The implementer can track two kinds of Performance Counter: time and count.\n * For time you can optionally call fetchNewFrame() to notify the start of a new frame to monitor, then call beginMonitoring() to start and endMonitoring() to record the lapsed time. endMonitoring takes a newFrame parameter for you to specify if the monitored time should be set for a new frame or accumulated to the current frame being monitored.\n * For count you first have to call fetchNewFrame() to notify the start of a new frame to monitor, then call addCount() how many time required to increment the count value you monitor.\n */\n class PerfCounter {\n /**\n * Returns the smallest value ever\n */\n get min() {\n return this._min;\n }\n /**\n * Returns the biggest value ever\n */\n get max() {\n return this._max;\n }\n /**\n * Returns the average value since the performance counter is running\n */\n get average() {\n return this._average;\n }\n /**\n * Returns the average value of the last second the counter was monitored\n */\n get lastSecAverage() {\n return this._lastSecAverage;\n }\n /**\n * Returns the current value\n */\n get current() {\n return this._current;\n }\n /**\n * Gets the accumulated total\n */\n get total() {\n return this._totalAccumulated;\n }\n /**\n * Gets the total value count\n */\n get count() {\n return this._totalValueCount;\n }\n /**\n * Creates a new counter\n */\n constructor() {\n this._startMonitoringTime = 0;\n this._min = 0;\n this._max = 0;\n this._average = 0;\n this._lastSecAverage = 0;\n this._current = 0;\n this._totalValueCount = 0;\n this._totalAccumulated = 0;\n this._lastSecAccumulated = 0;\n this._lastSecTime = 0;\n this._lastSecValueCount = 0;\n }\n /**\n * Call this method to start monitoring a new frame.\n * This scenario is typically used when you accumulate monitoring time many times for a single frame, you call this method at the start of the frame, then beginMonitoring to start recording and endMonitoring(false) to accumulated the recorded time to the PerfCounter or addCount() to accumulate a monitored count.\n */\n fetchNewFrame() {\n this._totalValueCount++;\n this._current = 0;\n this._lastSecValueCount++;\n }\n /**\n * Call this method to monitor a count of something (e.g. mesh drawn in viewport count)\n * @param newCount the count value to add to the monitored count\n * @param fetchResult true when it's the last time in the frame you add to the counter and you wish to update the statistics properties (min/max/average), false if you only want to update statistics.\n */\n addCount(newCount, fetchResult) {\n if (!PerfCounter.Enabled) {\n return;\n }\n this._current += newCount;\n if (fetchResult) {\n this._fetchResult();\n }\n }\n /**\n * Start monitoring this performance counter\n */\n beginMonitoring() {\n if (!PerfCounter.Enabled) {\n return;\n }\n this._startMonitoringTime = PrecisionDate.Now;\n }\n /**\n * Compute the time lapsed since the previous beginMonitoring() call.\n * @param newFrame true by default to fetch the result and monitor a new frame, if false the time monitored will be added to the current frame counter\n */\n endMonitoring(newFrame = true) {\n if (!PerfCounter.Enabled) {\n return;\n }\n if (newFrame) {\n this.fetchNewFrame();\n }\n const currentTime = PrecisionDate.Now;\n this._current = currentTime - this._startMonitoringTime;\n if (newFrame) {\n this._fetchResult();\n }\n }\n /**\n * Call this method to end the monitoring of a frame.\n * This scenario is typically used when you accumulate monitoring time many times for a single frame, you call this method at the end of the frame, after beginMonitoring to start recording and endMonitoring(false) to accumulated the recorded time to the PerfCounter or addCount() to accumulate a monitored count.\n */\n endFrame() {\n this._fetchResult();\n }\n _fetchResult() {\n this._totalAccumulated += this._current;\n this._lastSecAccumulated += this._current;\n // Min/Max update\n this._min = Math.min(this._min, this._current);\n this._max = Math.max(this._max, this._current);\n this._average = this._totalAccumulated / this._totalValueCount;\n // Reset last sec?\n const now = PrecisionDate.Now;\n if (now - this._lastSecTime > 1000) {\n this._lastSecAverage = this._lastSecAccumulated / this._lastSecValueCount;\n this._lastSecTime = now;\n this._lastSecAccumulated = 0;\n this._lastSecValueCount = 0;\n }\n }\n }\n /**\n * Gets or sets a global boolean to turn on and off all the counters\n */\n PerfCounter.Enabled = true;\n\n /**\n * Helper class used to generate session unique ID\n */\n class UniqueIdGenerator {\n /**\n * Gets an unique (relatively to the current scene) Id\n */\n static get UniqueId() {\n const result = this._UniqueIdCounter;\n this._UniqueIdCounter++;\n return result;\n }\n }\n // Statics\n UniqueIdGenerator._UniqueIdCounter = 1;\n\n /** Defines the cross module constantsused by lights to avoid circular dependencies */\n class LightConstants {\n /**\n * Sort function to order lights for rendering.\n * @param a First Light object to compare to second.\n * @param b Second Light object to compare first.\n * @returns -1 to reduce's a's index relative to be, 0 for no change, 1 to increase a's index relative to b.\n */\n static CompareLightsPriority(a, b) {\n //shadow-casting lights have priority over non-shadow-casting lights\n //the renderPriority is a secondary sort criterion\n if (a.shadowEnabled !== b.shadowEnabled) {\n return (b.shadowEnabled ? 1 : 0) - (a.shadowEnabled ? 1 : 0);\n }\n return b.renderPriority - a.renderPriority;\n }\n }\n /**\n * Falloff Default: light is falling off following the material specification:\n * standard material is using standard falloff whereas pbr material can request special falloff per materials.\n */\n LightConstants.FALLOFF_DEFAULT = 0;\n /**\n * Falloff Physical: light is falling off following the inverse squared distance law.\n */\n LightConstants.FALLOFF_PHYSICAL = 1;\n /**\n * Falloff gltf: light is falling off as described in the gltf moving to PBR document\n * to enhance interoperability with other engines.\n */\n LightConstants.FALLOFF_GLTF = 2;\n /**\n * Falloff Standard: light is falling off like in the standard material\n * to enhance interoperability with other materials.\n */\n LightConstants.FALLOFF_STANDARD = 3;\n //lightmapMode Consts\n /**\n * If every light affecting the material is in this lightmapMode,\n * material.lightmapTexture adds or multiplies\n * (depends on material.useLightmapAsShadowmap)\n * after every other light calculations.\n */\n LightConstants.LIGHTMAP_DEFAULT = 0;\n /**\n * material.lightmapTexture as only diffuse lighting from this light\n * adds only specular lighting from this light\n * adds dynamic shadows\n */\n LightConstants.LIGHTMAP_SPECULAR = 1;\n /**\n * material.lightmapTexture as only lighting\n * no light calculation from this light\n * only adds dynamic shadows from this light\n */\n LightConstants.LIGHTMAP_SHADOWSONLY = 2;\n // Intensity Mode Consts\n /**\n * Each light type uses the default quantity according to its type:\n * point/spot lights use luminous intensity\n * directional lights use illuminance\n */\n LightConstants.INTENSITYMODE_AUTOMATIC = 0;\n /**\n * lumen (lm)\n */\n LightConstants.INTENSITYMODE_LUMINOUSPOWER = 1;\n /**\n * candela (lm/sr)\n */\n LightConstants.INTENSITYMODE_LUMINOUSINTENSITY = 2;\n /**\n * lux (lm/m^2)\n */\n LightConstants.INTENSITYMODE_ILLUMINANCE = 3;\n /**\n * nit (cd/m^2)\n */\n LightConstants.INTENSITYMODE_LUMINANCE = 4;\n // Light types ids const.\n /**\n * Light type const id of the point light.\n */\n LightConstants.LIGHTTYPEID_POINTLIGHT = 0;\n /**\n * Light type const id of the directional light.\n */\n LightConstants.LIGHTTYPEID_DIRECTIONALLIGHT = 1;\n /**\n * Light type const id of the spot light.\n */\n LightConstants.LIGHTTYPEID_SPOTLIGHT = 2;\n /**\n * Light type const id of the hemispheric light.\n */\n LightConstants.LIGHTTYPEID_HEMISPHERICLIGHT = 3;\n\n /**\n * Class used to store configuration data associated with pointer picking\n */\n class PointerPickingConfiguration {\n constructor() {\n /**\n * Gets or sets a predicate used to select candidate meshes for a pointer down event\n */\n this.pointerDownFastCheck = false;\n /**\n * Gets or sets a predicate used to select candidate meshes for a pointer up event\n */\n this.pointerUpFastCheck = false;\n /**\n * Gets or sets a predicate used to select candidate meshes for a pointer move event\n */\n this.pointerMoveFastCheck = false;\n /**\n * Gets or sets a boolean indicating if the user want to entirely skip the picking phase when a pointer move event occurs.\n */\n this.skipPointerMovePicking = false;\n /**\n * Gets or sets a boolean indicating if the user want to entirely skip the picking phase when a pointer down event occurs.\n */\n this.skipPointerDownPicking = false;\n /**\n * Gets or sets a boolean indicating if the user want to entirely skip the picking phase when a pointer up event occurs. Off by default.\n */\n this.skipPointerUpPicking = false;\n }\n }\n\n /**\n * Define how the scene should favor performance over ease of use\n */\n var ScenePerformancePriority;\n (function (ScenePerformancePriority) {\n /** Default mode. No change. Performance will be treated as less important than backward compatibility */\n ScenePerformancePriority[ScenePerformancePriority[\"BackwardCompatible\"] = 0] = \"BackwardCompatible\";\n /** Some performance options will be turned on trying to strike a balance between perf and ease of use */\n ScenePerformancePriority[ScenePerformancePriority[\"Intermediate\"] = 1] = \"Intermediate\";\n /** Performance will be top priority */\n ScenePerformancePriority[ScenePerformancePriority[\"Aggressive\"] = 2] = \"Aggressive\";\n })(ScenePerformancePriority || (ScenePerformancePriority = {}));\n /**\n * Represents a scene to be rendered by the engine.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene\n */\n class Scene extends AbstractScene {\n /**\n * Factory used to create the default material.\n * @param scene The scene to create the material for\n * @returns The default material\n */\n static DefaultMaterialFactory(scene) {\n throw _WarnImport(\"StandardMaterial\");\n }\n /**\n * Factory used to create the a collision coordinator.\n * @returns The collision coordinator\n */\n static CollisionCoordinatorFactory() {\n throw _WarnImport(\"DefaultCollisionCoordinator\");\n }\n /**\n * Texture used in all pbr material as the reflection texture.\n * As in the majority of the scene they are the same (exception for multi room and so on),\n * this is easier to reference from here than from all the materials.\n */\n get environmentTexture() {\n return this._environmentTexture;\n }\n /**\n * Texture used in all pbr material as the reflection texture.\n * As in the majority of the scene they are the same (exception for multi room and so on),\n * this is easier to set here than in all the materials.\n */\n set environmentTexture(value) {\n if (this._environmentTexture === value) {\n return;\n }\n this._environmentTexture = value;\n this.markAllMaterialsAsDirty(1);\n }\n /**\n * Default image processing configuration used either in the rendering\n * Forward main pass or through the imageProcessingPostProcess if present.\n * As in the majority of the scene they are the same (exception for multi camera),\n * this is easier to reference from here than from all the materials and post process.\n *\n * No setter as we it is a shared configuration, you can set the values instead.\n */\n get imageProcessingConfiguration() {\n return this._imageProcessingConfiguration;\n }\n /**\n * Gets or sets a value indicating how to treat performance relatively to ease of use and backward compatibility\n */\n get performancePriority() {\n return this._performancePriority;\n }\n set performancePriority(value) {\n if (value === this._performancePriority) {\n return;\n }\n this._performancePriority = value;\n switch (value) {\n case ScenePerformancePriority.BackwardCompatible:\n this.skipFrustumClipping = false;\n this._renderingManager.maintainStateBetweenFrames = false;\n this.skipPointerMovePicking = false;\n this.autoClear = true;\n break;\n case ScenePerformancePriority.Intermediate:\n this.skipFrustumClipping = false;\n this._renderingManager.maintainStateBetweenFrames = false;\n this.skipPointerMovePicking = true;\n this.autoClear = false;\n break;\n case ScenePerformancePriority.Aggressive:\n this.skipFrustumClipping = true;\n this._renderingManager.maintainStateBetweenFrames = true;\n this.skipPointerMovePicking = true;\n this.autoClear = false;\n break;\n }\n this.onScenePerformancePriorityChangedObservable.notifyObservers(value);\n }\n /**\n * Gets or sets a boolean indicating if all rendering must be done in wireframe\n */\n set forceWireframe(value) {\n if (this._forceWireframe === value) {\n return;\n }\n this._forceWireframe = value;\n this.markAllMaterialsAsDirty(16);\n }\n get forceWireframe() {\n return this._forceWireframe;\n }\n /**\n * Gets or sets a boolean indicating if we should skip the frustum clipping part of the active meshes selection\n */\n set skipFrustumClipping(value) {\n if (this._skipFrustumClipping === value) {\n return;\n }\n this._skipFrustumClipping = value;\n }\n get skipFrustumClipping() {\n return this._skipFrustumClipping;\n }\n /**\n * Gets or sets a boolean indicating if all rendering must be done in point cloud\n */\n set forcePointsCloud(value) {\n if (this._forcePointsCloud === value) {\n return;\n }\n this._forcePointsCloud = value;\n this.markAllMaterialsAsDirty(16);\n }\n get forcePointsCloud() {\n return this._forcePointsCloud;\n }\n /**\n * Gets or sets the animation properties override\n */\n get animationPropertiesOverride() {\n return this._animationPropertiesOverride;\n }\n set animationPropertiesOverride(value) {\n this._animationPropertiesOverride = value;\n }\n /** Sets a function to be executed when this scene is disposed. */\n set onDispose(callback) {\n if (this._onDisposeObserver) {\n this.onDisposeObservable.remove(this._onDisposeObserver);\n }\n this._onDisposeObserver = this.onDisposeObservable.add(callback);\n }\n /** Sets a function to be executed before rendering this scene */\n set beforeRender(callback) {\n if (this._onBeforeRenderObserver) {\n this.onBeforeRenderObservable.remove(this._onBeforeRenderObserver);\n }\n if (callback) {\n this._onBeforeRenderObserver = this.onBeforeRenderObservable.add(callback);\n }\n }\n /** Sets a function to be executed after rendering this scene */\n set afterRender(callback) {\n if (this._onAfterRenderObserver) {\n this.onAfterRenderObservable.remove(this._onAfterRenderObserver);\n }\n if (callback) {\n this._onAfterRenderObserver = this.onAfterRenderObservable.add(callback);\n }\n }\n /** Sets a function to be executed before rendering a camera*/\n set beforeCameraRender(callback) {\n if (this._onBeforeCameraRenderObserver) {\n this.onBeforeCameraRenderObservable.remove(this._onBeforeCameraRenderObserver);\n }\n this._onBeforeCameraRenderObserver = this.onBeforeCameraRenderObservable.add(callback);\n }\n /** Sets a function to be executed after rendering a camera*/\n set afterCameraRender(callback) {\n if (this._onAfterCameraRenderObserver) {\n this.onAfterCameraRenderObservable.remove(this._onAfterCameraRenderObserver);\n }\n this._onAfterCameraRenderObserver = this.onAfterCameraRenderObservable.add(callback);\n }\n /**\n * Gets or sets a predicate used to select candidate meshes for a pointer down event\n */\n get pointerDownPredicate() {\n return this._pointerPickingConfiguration.pointerDownPredicate;\n }\n set pointerDownPredicate(value) {\n this._pointerPickingConfiguration.pointerDownPredicate = value;\n }\n /**\n * Gets or sets a predicate used to select candidate meshes for a pointer up event\n */\n get pointerUpPredicate() {\n return this._pointerPickingConfiguration.pointerUpPredicate;\n }\n set pointerUpPredicate(value) {\n this._pointerPickingConfiguration.pointerUpPredicate = value;\n }\n /**\n * Gets or sets a predicate used to select candidate meshes for a pointer move event\n */\n get pointerMovePredicate() {\n return this._pointerPickingConfiguration.pointerMovePredicate;\n }\n set pointerMovePredicate(value) {\n this._pointerPickingConfiguration.pointerMovePredicate = value;\n }\n /**\n * Gets or sets a predicate used to select candidate meshes for a pointer down event\n */\n get pointerDownFastCheck() {\n return this._pointerPickingConfiguration.pointerDownFastCheck;\n }\n set pointerDownFastCheck(value) {\n this._pointerPickingConfiguration.pointerDownFastCheck = value;\n }\n /**\n * Gets or sets a predicate used to select candidate meshes for a pointer up event\n */\n get pointerUpFastCheck() {\n return this._pointerPickingConfiguration.pointerUpFastCheck;\n }\n set pointerUpFastCheck(value) {\n this._pointerPickingConfiguration.pointerUpFastCheck = value;\n }\n /**\n * Gets or sets a predicate used to select candidate meshes for a pointer move event\n */\n get pointerMoveFastCheck() {\n return this._pointerPickingConfiguration.pointerMoveFastCheck;\n }\n set pointerMoveFastCheck(value) {\n this._pointerPickingConfiguration.pointerMoveFastCheck = value;\n }\n /**\n * Gets or sets a boolean indicating if the user want to entirely skip the picking phase when a pointer move event occurs.\n */\n get skipPointerMovePicking() {\n return this._pointerPickingConfiguration.skipPointerMovePicking;\n }\n set skipPointerMovePicking(value) {\n this._pointerPickingConfiguration.skipPointerMovePicking = value;\n }\n /**\n * Gets or sets a boolean indicating if the user want to entirely skip the picking phase when a pointer down event occurs.\n */\n get skipPointerDownPicking() {\n return this._pointerPickingConfiguration.skipPointerDownPicking;\n }\n set skipPointerDownPicking(value) {\n this._pointerPickingConfiguration.skipPointerDownPicking = value;\n }\n /**\n * Gets or sets a boolean indicating if the user want to entirely skip the picking phase when a pointer up event occurs. Off by default.\n */\n get skipPointerUpPicking() {\n return this._pointerPickingConfiguration.skipPointerUpPicking;\n }\n set skipPointerUpPicking(value) {\n this._pointerPickingConfiguration.skipPointerUpPicking = value;\n }\n /**\n * Gets the pointer coordinates without any translation (ie. straight out of the pointer event)\n */\n get unTranslatedPointer() {\n return this._inputManager.unTranslatedPointer;\n }\n /**\n * Gets or sets the distance in pixel that you have to move to prevent some events. Default is 10 pixels\n */\n static get DragMovementThreshold() {\n return InputManager.DragMovementThreshold;\n }\n static set DragMovementThreshold(value) {\n InputManager.DragMovementThreshold = value;\n }\n /**\n * Time in milliseconds to wait to raise long press events if button is still pressed. Default is 500 ms\n */\n static get LongPressDelay() {\n return InputManager.LongPressDelay;\n }\n static set LongPressDelay(value) {\n InputManager.LongPressDelay = value;\n }\n /**\n * Time in milliseconds to wait to raise long press events if button is still pressed. Default is 300 ms\n */\n static get DoubleClickDelay() {\n return InputManager.DoubleClickDelay;\n }\n static set DoubleClickDelay(value) {\n InputManager.DoubleClickDelay = value;\n }\n /** If you need to check double click without raising a single click at first click, enable this flag */\n static get ExclusiveDoubleClickMode() {\n return InputManager.ExclusiveDoubleClickMode;\n }\n static set ExclusiveDoubleClickMode(value) {\n InputManager.ExclusiveDoubleClickMode = value;\n }\n /**\n * Bind the current view position to an effect.\n * @param effect The effect to be bound\n * @param variableName name of the shader variable that will hold the eye position\n * @param isVector3 true to indicates that variableName is a Vector3 and not a Vector4\n * @returns the computed eye position\n */\n bindEyePosition(effect, variableName = \"vEyePosition\", isVector3 = false) {\n var _a;\n const eyePosition = this._forcedViewPosition\n ? this._forcedViewPosition\n : this._mirroredCameraPosition\n ? this._mirroredCameraPosition\n : (_a = this.activeCamera.globalPosition) !== null && _a !== void 0 ? _a : this.activeCamera.devicePosition;\n const invertNormal = this.useRightHandedSystem === (this._mirroredCameraPosition != null);\n TmpVectors.Vector4[0].set(eyePosition.x, eyePosition.y, eyePosition.z, invertNormal ? -1 : 1);\n if (effect) {\n if (isVector3) {\n effect.setFloat3(variableName, TmpVectors.Vector4[0].x, TmpVectors.Vector4[0].y, TmpVectors.Vector4[0].z);\n }\n else {\n effect.setVector4(variableName, TmpVectors.Vector4[0]);\n }\n }\n return TmpVectors.Vector4[0];\n }\n /**\n * Update the scene ubo before it can be used in rendering processing\n * @returns the scene UniformBuffer\n */\n finalizeSceneUbo() {\n const ubo = this.getSceneUniformBuffer();\n const eyePosition = this.bindEyePosition(null);\n ubo.updateFloat4(\"vEyePosition\", eyePosition.x, eyePosition.y, eyePosition.z, eyePosition.w);\n ubo.update();\n return ubo;\n }\n /**\n * Gets or sets a boolean indicating if the scene must use right-handed coordinates system\n */\n set useRightHandedSystem(value) {\n if (this._useRightHandedSystem === value) {\n return;\n }\n this._useRightHandedSystem = value;\n this.markAllMaterialsAsDirty(16);\n }\n get useRightHandedSystem() {\n return this._useRightHandedSystem;\n }\n /**\n * Sets the step Id used by deterministic lock step\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep\n * @param newStepId defines the step Id\n */\n setStepId(newStepId) {\n this._currentStepId = newStepId;\n }\n /**\n * Gets the step Id used by deterministic lock step\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep\n * @returns the step Id\n */\n getStepId() {\n return this._currentStepId;\n }\n /**\n * Gets the internal step used by deterministic lock step\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep\n * @returns the internal step\n */\n getInternalStep() {\n return this._currentInternalStep;\n }\n /**\n * Gets or sets a boolean indicating if fog is enabled on this scene\n * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/environment_introduction#fog\n * (Default is true)\n */\n set fogEnabled(value) {\n if (this._fogEnabled === value) {\n return;\n }\n this._fogEnabled = value;\n this.markAllMaterialsAsDirty(16);\n }\n get fogEnabled() {\n return this._fogEnabled;\n }\n /**\n * Gets or sets the fog mode to use\n * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/environment_introduction#fog\n * | mode | value |\n * | --- | --- |\n * | FOGMODE_NONE | 0 |\n * | FOGMODE_EXP | 1 |\n * | FOGMODE_EXP2 | 2 |\n * | FOGMODE_LINEAR | 3 |\n */\n set fogMode(value) {\n if (this._fogMode === value) {\n return;\n }\n this._fogMode = value;\n this.markAllMaterialsAsDirty(16);\n }\n get fogMode() {\n return this._fogMode;\n }\n /**\n * Flag indicating that the frame buffer binding is handled by another component\n */\n get prePass() {\n return !!this.prePassRenderer && this.prePassRenderer.defaultRT.enabled;\n }\n /**\n * Gets or sets a boolean indicating if shadows are enabled on this scene\n */\n set shadowsEnabled(value) {\n if (this._shadowsEnabled === value) {\n return;\n }\n this._shadowsEnabled = value;\n this.markAllMaterialsAsDirty(2);\n }\n get shadowsEnabled() {\n return this._shadowsEnabled;\n }\n /**\n * Gets or sets a boolean indicating if lights are enabled on this scene\n */\n set lightsEnabled(value) {\n if (this._lightsEnabled === value) {\n return;\n }\n this._lightsEnabled = value;\n this.markAllMaterialsAsDirty(2);\n }\n get lightsEnabled() {\n return this._lightsEnabled;\n }\n /** All of the active cameras added to this scene. */\n get activeCameras() {\n return this._activeCameras;\n }\n set activeCameras(cameras) {\n if (this._unObserveActiveCameras) {\n this._unObserveActiveCameras();\n this._unObserveActiveCameras = null;\n }\n if (cameras) {\n this._unObserveActiveCameras = _ObserveArray(cameras, () => {\n this.onActiveCamerasChanged.notifyObservers(this);\n });\n }\n this._activeCameras = cameras;\n }\n /** Gets or sets the current active camera */\n get activeCamera() {\n return this._activeCamera;\n }\n set activeCamera(value) {\n if (value === this._activeCamera) {\n return;\n }\n this._activeCamera = value;\n this.onActiveCameraChanged.notifyObservers(this);\n }\n /** The default material used on meshes when no material is affected */\n get defaultMaterial() {\n if (!this._defaultMaterial) {\n this._defaultMaterial = Scene.DefaultMaterialFactory(this);\n }\n return this._defaultMaterial;\n }\n /** The default material used on meshes when no material is affected */\n set defaultMaterial(value) {\n this._defaultMaterial = value;\n }\n /**\n * Gets or sets a boolean indicating if textures are enabled on this scene\n */\n set texturesEnabled(value) {\n if (this._texturesEnabled === value) {\n return;\n }\n this._texturesEnabled = value;\n this.markAllMaterialsAsDirty(1);\n }\n get texturesEnabled() {\n return this._texturesEnabled;\n }\n /**\n * Gets or sets a boolean indicating if skeletons are enabled on this scene\n */\n set skeletonsEnabled(value) {\n if (this._skeletonsEnabled === value) {\n return;\n }\n this._skeletonsEnabled = value;\n this.markAllMaterialsAsDirty(8);\n }\n get skeletonsEnabled() {\n return this._skeletonsEnabled;\n }\n /** @internal */\n get collisionCoordinator() {\n if (!this._collisionCoordinator) {\n this._collisionCoordinator = Scene.CollisionCoordinatorFactory();\n this._collisionCoordinator.init(this);\n }\n return this._collisionCoordinator;\n }\n /**\n * Gets the scene's rendering manager\n */\n get renderingManager() {\n return this._renderingManager;\n }\n /**\n * Gets the list of frustum planes (built from the active camera)\n */\n get frustumPlanes() {\n return this._frustumPlanes;\n }\n /**\n * Registers the transient components if needed.\n */\n _registerTransientComponents() {\n // Register components that have been associated lately to the scene.\n if (this._transientComponents.length > 0) {\n for (const component of this._transientComponents) {\n component.register();\n }\n this._transientComponents.length = 0;\n }\n }\n /**\n * @internal\n * Add a component to the scene.\n * Note that the ccomponent could be registered on th next frame if this is called after\n * the register component stage.\n * @param component Defines the component to add to the scene\n */\n _addComponent(component) {\n this._components.push(component);\n this._transientComponents.push(component);\n const serializableComponent = component;\n if (serializableComponent.addFromContainer && serializableComponent.serialize) {\n this._serializableComponents.push(serializableComponent);\n }\n }\n /**\n * @internal\n * Gets a component from the scene.\n * @param name defines the name of the component to retrieve\n * @returns the component or null if not present\n */\n _getComponent(name) {\n for (const component of this._components) {\n if (component.name === name) {\n return component;\n }\n }\n return null;\n }\n /**\n * Creates a new Scene\n * @param engine defines the engine to use to render this scene\n * @param options defines the scene options\n */\n constructor(engine, options) {\n super();\n // Members\n /** @internal */\n this._inputManager = new InputManager(this);\n /** Define this parameter if you are using multiple cameras and you want to specify which one should be used for pointer position */\n this.cameraToUseForPointers = null;\n /** @internal */\n this._isScene = true;\n /** @internal */\n this._blockEntityCollection = false;\n /**\n * Gets or sets a boolean that indicates if the scene must clear the render buffer before rendering a frame\n */\n this.autoClear = true;\n /**\n * Gets or sets a boolean that indicates if the scene must clear the depth and stencil buffers before rendering a frame\n */\n this.autoClearDepthAndStencil = true;\n /**\n * Defines the color used to clear the render buffer (Default is (0.2, 0.2, 0.3, 1.0))\n */\n this.clearColor = new Color4(0.2, 0.2, 0.3, 1.0);\n /**\n * Defines the color used to simulate the ambient color (Default is (0, 0, 0))\n */\n this.ambientColor = new Color3(0, 0, 0);\n /**\n * Intensity of the environment in all pbr material.\n * This dims or reinforces the IBL lighting overall (reflection and diffuse).\n * As in the majority of the scene they are the same (exception for multi room and so on),\n * this is easier to reference from here than from all the materials.\n */\n this.environmentIntensity = 1;\n this._performancePriority = ScenePerformancePriority.BackwardCompatible;\n /**\n * Observable triggered when the performance priority is changed\n */\n this.onScenePerformancePriorityChangedObservable = new Observable$1();\n this._forceWireframe = false;\n this._skipFrustumClipping = false;\n this._forcePointsCloud = false;\n /**\n * Gets or sets a boolean indicating if animations are enabled\n */\n this.animationsEnabled = true;\n this._animationPropertiesOverride = null;\n /**\n * Gets or sets a boolean indicating if a constant deltatime has to be used\n * This is mostly useful for testing purposes when you do not want the animations to scale with the framerate\n */\n this.useConstantAnimationDeltaTime = false;\n /**\n * Gets or sets a boolean indicating if the scene must keep the meshUnderPointer property updated\n * Please note that it requires to run a ray cast through the scene on every frame\n */\n this.constantlyUpdateMeshUnderPointer = false;\n /**\n * Defines the HTML cursor to use when hovering over interactive elements\n */\n this.hoverCursor = \"pointer\";\n /**\n * Defines the HTML default cursor to use (empty by default)\n */\n this.defaultCursor = \"\";\n /**\n * Defines whether cursors are handled by the scene.\n */\n this.doNotHandleCursors = false;\n /**\n * This is used to call preventDefault() on pointer down\n * in order to block unwanted artifacts like system double clicks\n */\n this.preventDefaultOnPointerDown = true;\n /**\n * This is used to call preventDefault() on pointer up\n * in order to block unwanted artifacts like system double clicks\n */\n this.preventDefaultOnPointerUp = true;\n // Metadata\n /**\n * Gets or sets user defined metadata\n */\n this.metadata = null;\n /**\n * For internal use only. Please do not use.\n */\n this.reservedDataStore = null;\n /**\n * Use this array to add regular expressions used to disable offline support for specific urls\n */\n this.disableOfflineSupportExceptionRules = new Array();\n /**\n * An event triggered when the scene is disposed.\n */\n this.onDisposeObservable = new Observable$1();\n this._onDisposeObserver = null;\n /**\n * An event triggered before rendering the scene (right after animations and physics)\n */\n this.onBeforeRenderObservable = new Observable$1();\n this._onBeforeRenderObserver = null;\n /**\n * An event triggered after rendering the scene\n */\n this.onAfterRenderObservable = new Observable$1();\n /**\n * An event triggered after rendering the scene for an active camera (When scene.render is called this will be called after each camera)\n * This is triggered for each \"sub\" camera in a Camera Rig unlike onAfterCameraRenderObservable\n */\n this.onAfterRenderCameraObservable = new Observable$1();\n this._onAfterRenderObserver = null;\n /**\n * An event triggered before animating the scene\n */\n this.onBeforeAnimationsObservable = new Observable$1();\n /**\n * An event triggered after animations processing\n */\n this.onAfterAnimationsObservable = new Observable$1();\n /**\n * An event triggered before draw calls are ready to be sent\n */\n this.onBeforeDrawPhaseObservable = new Observable$1();\n /**\n * An event triggered after draw calls have been sent\n */\n this.onAfterDrawPhaseObservable = new Observable$1();\n /**\n * An event triggered when the scene is ready\n */\n this.onReadyObservable = new Observable$1();\n /**\n * An event triggered before rendering a camera\n */\n this.onBeforeCameraRenderObservable = new Observable$1();\n this._onBeforeCameraRenderObserver = null;\n /**\n * An event triggered after rendering a camera\n * This is triggered for the full rig Camera only unlike onAfterRenderCameraObservable\n */\n this.onAfterCameraRenderObservable = new Observable$1();\n this._onAfterCameraRenderObserver = null;\n /**\n * An event triggered when active meshes evaluation is about to start\n */\n this.onBeforeActiveMeshesEvaluationObservable = new Observable$1();\n /**\n * An event triggered when active meshes evaluation is done\n */\n this.onAfterActiveMeshesEvaluationObservable = new Observable$1();\n /**\n * An event triggered when particles rendering is about to start\n * Note: This event can be trigger more than once per frame (because particles can be rendered by render target textures as well)\n */\n this.onBeforeParticlesRenderingObservable = new Observable$1();\n /**\n * An event triggered when particles rendering is done\n * Note: This event can be trigger more than once per frame (because particles can be rendered by render target textures as well)\n */\n this.onAfterParticlesRenderingObservable = new Observable$1();\n /**\n * An event triggered when SceneLoader.Append or SceneLoader.Load or SceneLoader.ImportMesh were successfully executed\n */\n this.onDataLoadedObservable = new Observable$1();\n /**\n * An event triggered when a camera is created\n */\n this.onNewCameraAddedObservable = new Observable$1();\n /**\n * An event triggered when a camera is removed\n */\n this.onCameraRemovedObservable = new Observable$1();\n /**\n * An event triggered when a light is created\n */\n this.onNewLightAddedObservable = new Observable$1();\n /**\n * An event triggered when a light is removed\n */\n this.onLightRemovedObservable = new Observable$1();\n /**\n * An event triggered when a geometry is created\n */\n this.onNewGeometryAddedObservable = new Observable$1();\n /**\n * An event triggered when a geometry is removed\n */\n this.onGeometryRemovedObservable = new Observable$1();\n /**\n * An event triggered when a transform node is created\n */\n this.onNewTransformNodeAddedObservable = new Observable$1();\n /**\n * An event triggered when a transform node is removed\n */\n this.onTransformNodeRemovedObservable = new Observable$1();\n /**\n * An event triggered when a mesh is created\n */\n this.onNewMeshAddedObservable = new Observable$1();\n /**\n * An event triggered when a mesh is removed\n */\n this.onMeshRemovedObservable = new Observable$1();\n /**\n * An event triggered when a skeleton is created\n */\n this.onNewSkeletonAddedObservable = new Observable$1();\n /**\n * An event triggered when a skeleton is removed\n */\n this.onSkeletonRemovedObservable = new Observable$1();\n /**\n * An event triggered when a material is created\n */\n this.onNewMaterialAddedObservable = new Observable$1();\n /**\n * An event triggered when a multi material is created\n */\n this.onNewMultiMaterialAddedObservable = new Observable$1();\n /**\n * An event triggered when a material is removed\n */\n this.onMaterialRemovedObservable = new Observable$1();\n /**\n * An event triggered when a multi material is removed\n */\n this.onMultiMaterialRemovedObservable = new Observable$1();\n /**\n * An event triggered when a texture is created\n */\n this.onNewTextureAddedObservable = new Observable$1();\n /**\n * An event triggered when a texture is removed\n */\n this.onTextureRemovedObservable = new Observable$1();\n /**\n * An event triggered when render targets are about to be rendered\n * Can happen multiple times per frame.\n */\n this.onBeforeRenderTargetsRenderObservable = new Observable$1();\n /**\n * An event triggered when render targets were rendered.\n * Can happen multiple times per frame.\n */\n this.onAfterRenderTargetsRenderObservable = new Observable$1();\n /**\n * An event triggered before calculating deterministic simulation step\n */\n this.onBeforeStepObservable = new Observable$1();\n /**\n * An event triggered after calculating deterministic simulation step\n */\n this.onAfterStepObservable = new Observable$1();\n /**\n * An event triggered when the activeCamera property is updated\n */\n this.onActiveCameraChanged = new Observable$1();\n /**\n * An event triggered when the activeCameras property is updated\n */\n this.onActiveCamerasChanged = new Observable$1();\n /**\n * This Observable will be triggered before rendering each renderingGroup of each rendered camera.\n * The RenderingGroupInfo class contains all the information about the context in which the observable is called\n * If you wish to register an Observer only for a given set of renderingGroup, use the mask with a combination of the renderingGroup index elevated to the power of two (1 for renderingGroup 0, 2 for renderingrOup1, 4 for 2 and 8 for 3)\n */\n this.onBeforeRenderingGroupObservable = new Observable$1();\n /**\n * This Observable will be triggered after rendering each renderingGroup of each rendered camera.\n * The RenderingGroupInfo class contains all the information about the context in which the observable is called\n * If you wish to register an Observer only for a given set of renderingGroup, use the mask with a combination of the renderingGroup index elevated to the power of two (1 for renderingGroup 0, 2 for renderingrOup1, 4 for 2 and 8 for 3)\n */\n this.onAfterRenderingGroupObservable = new Observable$1();\n /**\n * This Observable will when a mesh has been imported into the scene.\n */\n this.onMeshImportedObservable = new Observable$1();\n /**\n * This Observable will when an animation file has been imported into the scene.\n */\n this.onAnimationFileImportedObservable = new Observable$1();\n // Animations\n /** @internal */\n this._registeredForLateAnimationBindings = new SmartArrayNoDuplicate(256);\n // Pointers\n this._pointerPickingConfiguration = new PointerPickingConfiguration();\n /**\n * This observable event is triggered when any ponter event is triggered. It is registered during Scene.attachControl() and it is called BEFORE the 3D engine process anything (mesh/sprite picking for instance).\n * You have the possibility to skip the process and the call to onPointerObservable by setting PointerInfoPre.skipOnPointerObservable to true\n */\n this.onPrePointerObservable = new Observable$1();\n /**\n * Observable event triggered each time an input event is received from the rendering canvas\n */\n this.onPointerObservable = new Observable$1();\n // Keyboard\n /**\n * This observable event is triggered when any keyboard event si raised and registered during Scene.attachControl()\n * You have the possibility to skip the process and the call to onKeyboardObservable by setting KeyboardInfoPre.skipOnPointerObservable to true\n */\n this.onPreKeyboardObservable = new Observable$1();\n /**\n * Observable event triggered each time an keyboard event is received from the hosting window\n */\n this.onKeyboardObservable = new Observable$1();\n // Coordinates system\n this._useRightHandedSystem = false;\n // Deterministic lockstep\n this._timeAccumulator = 0;\n this._currentStepId = 0;\n this._currentInternalStep = 0;\n // Fog\n this._fogEnabled = true;\n this._fogMode = Scene.FOGMODE_NONE;\n /**\n * Gets or sets the fog color to use\n * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/environment_introduction#fog\n * (Default is Color3(0.2, 0.2, 0.3))\n */\n this.fogColor = new Color3(0.2, 0.2, 0.3);\n /**\n * Gets or sets the fog density to use\n * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/environment_introduction#fog\n * (Default is 0.1)\n */\n this.fogDensity = 0.1;\n /**\n * Gets or sets the fog start distance to use\n * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/environment_introduction#fog\n * (Default is 0)\n */\n this.fogStart = 0;\n /**\n * Gets or sets the fog end distance to use\n * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/environment_introduction#fog\n * (Default is 1000)\n */\n this.fogEnd = 1000.0;\n /**\n * Flag indicating if we need to store previous matrices when rendering\n */\n this.needsPreviousWorldMatrices = false;\n // Lights\n this._shadowsEnabled = true;\n this._lightsEnabled = true;\n this._unObserveActiveCameras = null;\n // Textures\n this._texturesEnabled = true;\n // Physics\n /**\n * Gets or sets a boolean indicating if physic engines are enabled on this scene\n */\n this.physicsEnabled = true;\n // Particles\n /**\n * Gets or sets a boolean indicating if particles are enabled on this scene\n */\n this.particlesEnabled = true;\n // Sprites\n /**\n * Gets or sets a boolean indicating if sprites are enabled on this scene\n */\n this.spritesEnabled = true;\n // Skeletons\n this._skeletonsEnabled = true;\n // Lens flares\n /**\n * Gets or sets a boolean indicating if lens flares are enabled on this scene\n */\n this.lensFlaresEnabled = true;\n // Collisions\n /**\n * Gets or sets a boolean indicating if collisions are enabled on this scene\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions\n */\n this.collisionsEnabled = true;\n /**\n * Defines the gravity applied to this scene (used only for collisions)\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions\n */\n this.gravity = new Vector3(0, -9.807, 0);\n // Postprocesses\n /**\n * Gets or sets a boolean indicating if postprocesses are enabled on this scene\n */\n this.postProcessesEnabled = true;\n // Customs render targets\n /**\n * Gets or sets a boolean indicating if render targets are enabled on this scene\n */\n this.renderTargetsEnabled = true;\n /**\n * Gets or sets a boolean indicating if next render targets must be dumped as image for debugging purposes\n * We recommend not using it and instead rely on Spector.js: http://spector.babylonjs.com\n */\n this.dumpNextRenderTargets = false;\n /**\n * The list of user defined render targets added to the scene\n */\n this.customRenderTargets = new Array();\n /**\n * Gets the list of meshes imported to the scene through SceneLoader\n */\n this.importedMeshesFiles = new Array();\n // Probes\n /**\n * Gets or sets a boolean indicating if probes are enabled on this scene\n */\n this.probesEnabled = true;\n this._meshesForIntersections = new SmartArrayNoDuplicate(256);\n // Procedural textures\n /**\n * Gets or sets a boolean indicating if procedural textures are enabled on this scene\n */\n this.proceduralTexturesEnabled = true;\n // Performance counters\n this._totalVertices = new PerfCounter();\n /** @internal */\n this._activeIndices = new PerfCounter();\n /** @internal */\n this._activeParticles = new PerfCounter();\n /** @internal */\n this._activeBones = new PerfCounter();\n /** @internal */\n this._animationTime = 0;\n /**\n * Gets or sets a general scale for animation speed\n * @see https://www.babylonjs-playground.com/#IBU2W7#3\n */\n this.animationTimeScale = 1;\n this._renderId = 0;\n this._frameId = 0;\n this._executeWhenReadyTimeoutId = null;\n this._intermediateRendering = false;\n this._defaultFrameBufferCleared = false;\n this._viewUpdateFlag = -1;\n this._projectionUpdateFlag = -1;\n /** @internal */\n this._toBeDisposed = new Array(256);\n this._activeRequests = new Array();\n /** @internal */\n this._pendingData = new Array();\n this._isDisposed = false;\n /**\n * Gets or sets a boolean indicating that all submeshes of active meshes must be rendered\n * Use this boolean to avoid computing frustum clipping on submeshes (This could help when you are CPU bound)\n */\n this.dispatchAllSubMeshesOfActiveMeshes = false;\n this._activeMeshes = new SmartArray(256);\n this._processedMaterials = new SmartArray(256);\n this._renderTargets = new SmartArrayNoDuplicate(256);\n this._materialsRenderTargets = new SmartArrayNoDuplicate(256);\n /** @internal */\n this._activeParticleSystems = new SmartArray(256);\n this._activeSkeletons = new SmartArrayNoDuplicate(32);\n this._softwareSkinnedMeshes = new SmartArrayNoDuplicate(32);\n /** @internal */\n this._activeAnimatables = new Array();\n this._transformMatrix = Matrix.Zero();\n /**\n * Gets or sets a boolean indicating if lights must be sorted by priority (off by default)\n * This is useful if there are more lights that the maximum simulteanous authorized\n */\n this.requireLightSorting = false;\n /**\n * @internal\n * Backing store of defined scene components.\n */\n this._components = [];\n /**\n * @internal\n * Backing store of defined scene components.\n */\n this._serializableComponents = [];\n /**\n * List of components to register on the next registration step.\n */\n this._transientComponents = [];\n /**\n * @internal\n * Defines the actions happening before camera updates.\n */\n this._beforeCameraUpdateStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening before clear the canvas.\n */\n this._beforeClearStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening before clear the canvas.\n */\n this._beforeRenderTargetClearStage = Stage.Create();\n /**\n * @internal\n * Defines the actions when collecting render targets for the frame.\n */\n this._gatherRenderTargetsStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening for one camera in the frame.\n */\n this._gatherActiveCameraRenderTargetsStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening during the per mesh ready checks.\n */\n this._isReadyForMeshStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening before evaluate active mesh checks.\n */\n this._beforeEvaluateActiveMeshStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening during the evaluate sub mesh checks.\n */\n this._evaluateSubMeshStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening during the active mesh stage.\n */\n this._preActiveMeshStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening during the per camera render target step.\n */\n this._cameraDrawRenderTargetStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening just before the active camera is drawing.\n */\n this._beforeCameraDrawStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening just before a render target is drawing.\n */\n this._beforeRenderTargetDrawStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening just before a rendering group is drawing.\n */\n this._beforeRenderingGroupDrawStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening just before a mesh is drawing.\n */\n this._beforeRenderingMeshStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening just after a mesh has been drawn.\n */\n this._afterRenderingMeshStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening just after a rendering group has been drawn.\n */\n this._afterRenderingGroupDrawStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening just after the active camera has been drawn.\n */\n this._afterCameraDrawStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening just after the post processing\n */\n this._afterCameraPostProcessStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening just after a render target has been drawn.\n */\n this._afterRenderTargetDrawStage = Stage.Create();\n /**\n * Defines the actions happening just after the post processing on a render target\n */\n this._afterRenderTargetPostProcessStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening just after rendering all cameras and computing intersections.\n */\n this._afterRenderStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening when a pointer move event happens.\n */\n this._pointerMoveStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening when a pointer down event happens.\n */\n this._pointerDownStage = Stage.Create();\n /**\n * @internal\n * Defines the actions happening when a pointer up event happens.\n */\n this._pointerUpStage = Stage.Create();\n /**\n * an optional map from Geometry Id to Geometry index in the 'geometries' array\n */\n this._geometriesByUniqueId = null;\n this._defaultMeshCandidates = {\n data: [],\n length: 0,\n };\n this._defaultSubMeshCandidates = {\n data: [],\n length: 0,\n };\n this._preventFreeActiveMeshesAndRenderingGroups = false;\n /** @internal */\n this._activeMeshesFrozen = false;\n /** @internal */\n this._activeMeshesFrozenButKeepClipping = false;\n this._skipEvaluateActiveMeshesCompletely = false;\n /** @internal */\n this._allowPostProcessClearColor = true;\n /**\n * User updatable function that will return a deterministic frame time when engine is in deterministic lock step mode\n */\n this.getDeterministicFrameTime = () => {\n return this._engine.getTimeStep();\n };\n /** @internal */\n this._registeredActions = 0;\n this._blockMaterialDirtyMechanism = false;\n /**\n * Internal perfCollector instance used for sharing between inspector and playground.\n * Marked as protected to allow sharing between prototype extensions, but disallow access at toplevel.\n */\n this._perfCollector = null;\n this.activeCameras = new Array();\n const fullOptions = Object.assign({ useGeometryUniqueIdsMap: true, useMaterialMeshMap: true, useClonedMeshMap: true, virtual: false }, options);\n this._engine = engine || EngineStore.LastCreatedEngine;\n if (!fullOptions.virtual) {\n EngineStore._LastCreatedScene = this;\n this._engine.scenes.push(this);\n }\n else {\n this._engine._virtualScenes.push(this);\n }\n this._uid = null;\n this._renderingManager = new RenderingManager(this);\n if (PostProcessManager) {\n this.postProcessManager = new PostProcessManager(this);\n }\n if (IsWindowObjectExist()) {\n this.attachControl();\n }\n // Uniform Buffer\n this._createUbo();\n // Default Image processing definition\n if (ImageProcessingConfiguration) {\n this._imageProcessingConfiguration = new ImageProcessingConfiguration();\n }\n this.setDefaultCandidateProviders();\n if (fullOptions.useGeometryUniqueIdsMap) {\n this._geometriesByUniqueId = {};\n }\n this.useMaterialMeshMap = fullOptions.useMaterialMeshMap;\n this.useClonedMeshMap = fullOptions.useClonedMeshMap;\n if (!options || !options.virtual) {\n this._engine.onNewSceneAddedObservable.notifyObservers(this);\n }\n }\n /**\n * Gets a string identifying the name of the class\n * @returns \"Scene\" string\n */\n getClassName() {\n return \"Scene\";\n }\n /**\n * @internal\n */\n _getDefaultMeshCandidates() {\n this._defaultMeshCandidates.data = this.meshes;\n this._defaultMeshCandidates.length = this.meshes.length;\n return this._defaultMeshCandidates;\n }\n /**\n * @internal\n */\n _getDefaultSubMeshCandidates(mesh) {\n this._defaultSubMeshCandidates.data = mesh.subMeshes;\n this._defaultSubMeshCandidates.length = mesh.subMeshes.length;\n return this._defaultSubMeshCandidates;\n }\n /**\n * Sets the default candidate providers for the scene.\n * This sets the getActiveMeshCandidates, getActiveSubMeshCandidates, getIntersectingSubMeshCandidates\n * and getCollidingSubMeshCandidates to their default function\n */\n setDefaultCandidateProviders() {\n this.getActiveMeshCandidates = this._getDefaultMeshCandidates.bind(this);\n this.getActiveSubMeshCandidates = this._getDefaultSubMeshCandidates.bind(this);\n this.getIntersectingSubMeshCandidates = this._getDefaultSubMeshCandidates.bind(this);\n this.getCollidingSubMeshCandidates = this._getDefaultSubMeshCandidates.bind(this);\n }\n /**\n * Gets the mesh that is currently under the pointer\n */\n get meshUnderPointer() {\n return this._inputManager.meshUnderPointer;\n }\n /**\n * Gets or sets the current on-screen X position of the pointer\n */\n get pointerX() {\n return this._inputManager.pointerX;\n }\n set pointerX(value) {\n this._inputManager.pointerX = value;\n }\n /**\n * Gets or sets the current on-screen Y position of the pointer\n */\n get pointerY() {\n return this._inputManager.pointerY;\n }\n set pointerY(value) {\n this._inputManager.pointerY = value;\n }\n /**\n * Gets the cached material (ie. the latest rendered one)\n * @returns the cached material\n */\n getCachedMaterial() {\n return this._cachedMaterial;\n }\n /**\n * Gets the cached effect (ie. the latest rendered one)\n * @returns the cached effect\n */\n getCachedEffect() {\n return this._cachedEffect;\n }\n /**\n * Gets the cached visibility state (ie. the latest rendered one)\n * @returns the cached visibility state\n */\n getCachedVisibility() {\n return this._cachedVisibility;\n }\n /**\n * Gets a boolean indicating if the current material / effect / visibility must be bind again\n * @param material defines the current material\n * @param effect defines the current effect\n * @param visibility defines the current visibility state\n * @returns true if one parameter is not cached\n */\n isCachedMaterialInvalid(material, effect, visibility = 1) {\n return this._cachedEffect !== effect || this._cachedMaterial !== material || this._cachedVisibility !== visibility;\n }\n /**\n * Gets the engine associated with the scene\n * @returns an Engine\n */\n getEngine() {\n return this._engine;\n }\n /**\n * Gets the total number of vertices rendered per frame\n * @returns the total number of vertices rendered per frame\n */\n getTotalVertices() {\n return this._totalVertices.current;\n }\n /**\n * Gets the performance counter for total vertices\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimize_your_scene#instrumentation\n */\n get totalVerticesPerfCounter() {\n return this._totalVertices;\n }\n /**\n * Gets the total number of active indices rendered per frame (You can deduce the number of rendered triangles by dividing this number by 3)\n * @returns the total number of active indices rendered per frame\n */\n getActiveIndices() {\n return this._activeIndices.current;\n }\n /**\n * Gets the performance counter for active indices\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimize_your_scene#instrumentation\n */\n get totalActiveIndicesPerfCounter() {\n return this._activeIndices;\n }\n /**\n * Gets the total number of active particles rendered per frame\n * @returns the total number of active particles rendered per frame\n */\n getActiveParticles() {\n return this._activeParticles.current;\n }\n /**\n * Gets the performance counter for active particles\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimize_your_scene#instrumentation\n */\n get activeParticlesPerfCounter() {\n return this._activeParticles;\n }\n /**\n * Gets the total number of active bones rendered per frame\n * @returns the total number of active bones rendered per frame\n */\n getActiveBones() {\n return this._activeBones.current;\n }\n /**\n * Gets the performance counter for active bones\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimize_your_scene#instrumentation\n */\n get activeBonesPerfCounter() {\n return this._activeBones;\n }\n /**\n * Gets the array of active meshes\n * @returns an array of AbstractMesh\n */\n getActiveMeshes() {\n return this._activeMeshes;\n }\n /**\n * Gets the animation ratio (which is 1.0 is the scene renders at 60fps and 2 if the scene renders at 30fps, etc.)\n * @returns a number\n */\n getAnimationRatio() {\n return this._animationRatio !== undefined ? this._animationRatio : 1;\n }\n /**\n * Gets an unique Id for the current render phase\n * @returns a number\n */\n getRenderId() {\n return this._renderId;\n }\n /**\n * Gets an unique Id for the current frame\n * @returns a number\n */\n getFrameId() {\n return this._frameId;\n }\n /** Call this function if you want to manually increment the render Id*/\n incrementRenderId() {\n this._renderId++;\n }\n _createUbo() {\n this.setSceneUniformBuffer(this.createSceneUniformBuffer());\n }\n /**\n * Use this method to simulate a pointer move on a mesh\n * The pickResult parameter can be obtained from a scene.pick or scene.pickWithRay\n * @param pickResult pickingInfo of the object wished to simulate pointer event on\n * @param pointerEventInit pointer event state to be used when simulating the pointer event (eg. pointer id for multitouch)\n * @returns the current scene\n */\n simulatePointerMove(pickResult, pointerEventInit) {\n this._inputManager.simulatePointerMove(pickResult, pointerEventInit);\n return this;\n }\n /**\n * Use this method to simulate a pointer down on a mesh\n * The pickResult parameter can be obtained from a scene.pick or scene.pickWithRay\n * @param pickResult pickingInfo of the object wished to simulate pointer event on\n * @param pointerEventInit pointer event state to be used when simulating the pointer event (eg. pointer id for multitouch)\n * @returns the current scene\n */\n simulatePointerDown(pickResult, pointerEventInit) {\n this._inputManager.simulatePointerDown(pickResult, pointerEventInit);\n return this;\n }\n /**\n * Use this method to simulate a pointer up on a mesh\n * The pickResult parameter can be obtained from a scene.pick or scene.pickWithRay\n * @param pickResult pickingInfo of the object wished to simulate pointer event on\n * @param pointerEventInit pointer event state to be used when simulating the pointer event (eg. pointer id for multitouch)\n * @param doubleTap indicates that the pointer up event should be considered as part of a double click (false by default)\n * @returns the current scene\n */\n simulatePointerUp(pickResult, pointerEventInit, doubleTap) {\n this._inputManager.simulatePointerUp(pickResult, pointerEventInit, doubleTap);\n return this;\n }\n /**\n * Gets a boolean indicating if the current pointer event is captured (meaning that the scene has already handled the pointer down)\n * @param pointerId defines the pointer id to use in a multi-touch scenario (0 by default)\n * @returns true if the pointer was captured\n */\n isPointerCaptured(pointerId = 0) {\n return this._inputManager.isPointerCaptured(pointerId);\n }\n /**\n * Attach events to the canvas (To handle actionManagers triggers and raise onPointerMove, onPointerDown and onPointerUp\n * @param attachUp defines if you want to attach events to pointerup\n * @param attachDown defines if you want to attach events to pointerdown\n * @param attachMove defines if you want to attach events to pointermove\n */\n attachControl(attachUp = true, attachDown = true, attachMove = true) {\n this._inputManager.attachControl(attachUp, attachDown, attachMove);\n }\n /** Detaches all event handlers*/\n detachControl() {\n this._inputManager.detachControl();\n }\n /**\n * This function will check if the scene can be rendered (textures are loaded, shaders are compiled)\n * Delay loaded resources are not taking in account\n * @param checkRenderTargets true to also check that the meshes rendered as part of a render target are ready (default: true)\n * @returns true if all required resources are ready\n */\n isReady(checkRenderTargets = true) {\n var _a, _b, _c;\n if (this._isDisposed) {\n return false;\n }\n let index;\n const engine = this.getEngine();\n const currentRenderPassId = engine.currentRenderPassId;\n engine.currentRenderPassId = (_b = (_a = this.activeCamera) === null || _a === void 0 ? void 0 : _a.renderPassId) !== null && _b !== void 0 ? _b : currentRenderPassId;\n let isReady = true;\n // Pending data\n if (this._pendingData.length > 0) {\n isReady = false;\n }\n // Ensures that the pre-pass renderer is enabled if it is to be enabled.\n (_c = this.prePassRenderer) === null || _c === void 0 ? void 0 : _c.update();\n // Meshes\n if (checkRenderTargets) {\n this._processedMaterials.reset();\n this._materialsRenderTargets.reset();\n }\n for (index = 0; index < this.meshes.length; index++) {\n const mesh = this.meshes[index];\n if (!mesh.subMeshes || mesh.subMeshes.length === 0) {\n continue;\n }\n // Do not stop at the first encountered \"unready\" object as we want to ensure\n // all materials are starting off their compilation in parallel.\n if (!mesh.isReady(true)) {\n isReady = false;\n continue;\n }\n const hardwareInstancedRendering = mesh.hasThinInstances ||\n mesh.getClassName() === \"InstancedMesh\" ||\n mesh.getClassName() === \"InstancedLinesMesh\" ||\n (engine.getCaps().instancedArrays && mesh.instances.length > 0);\n // Is Ready For Mesh\n for (const step of this._isReadyForMeshStage) {\n if (!step.action(mesh, hardwareInstancedRendering)) {\n isReady = false;\n }\n }\n if (!checkRenderTargets) {\n continue;\n }\n const mat = mesh.material || this.defaultMaterial;\n if (mat) {\n if (mat._storeEffectOnSubMeshes) {\n for (const subMesh of mesh.subMeshes) {\n const material = subMesh.getMaterial();\n if (material && material.hasRenderTargetTextures && material.getRenderTargetTextures != null) {\n if (this._processedMaterials.indexOf(material) === -1) {\n this._processedMaterials.push(material);\n this._materialsRenderTargets.concatWithNoDuplicate(material.getRenderTargetTextures());\n }\n }\n }\n }\n else {\n if (mat.hasRenderTargetTextures && mat.getRenderTargetTextures != null) {\n if (this._processedMaterials.indexOf(mat) === -1) {\n this._processedMaterials.push(mat);\n this._materialsRenderTargets.concatWithNoDuplicate(mat.getRenderTargetTextures());\n }\n }\n }\n }\n }\n // Render targets\n if (checkRenderTargets) {\n for (index = 0; index < this._materialsRenderTargets.length; ++index) {\n const rtt = this._materialsRenderTargets.data[index];\n if (!rtt.isReadyForRendering()) {\n isReady = false;\n }\n }\n }\n // Geometries\n for (index = 0; index < this.geometries.length; index++) {\n const geometry = this.geometries[index];\n if (geometry.delayLoadState === 2) {\n isReady = false;\n }\n }\n // Post-processes\n if (this.activeCameras && this.activeCameras.length > 0) {\n for (const camera of this.activeCameras) {\n if (!camera.isReady(true)) {\n isReady = false;\n }\n }\n }\n else if (this.activeCamera) {\n if (!this.activeCamera.isReady(true)) {\n isReady = false;\n }\n }\n // Particles\n for (const particleSystem of this.particleSystems) {\n if (!particleSystem.isReady()) {\n isReady = false;\n }\n }\n // Layers\n if (this.layers) {\n for (const layer of this.layers) {\n if (!layer.isReady()) {\n isReady = false;\n }\n }\n }\n // Effects\n if (!engine.areAllEffectsReady()) {\n isReady = false;\n }\n engine.currentRenderPassId = currentRenderPassId;\n return isReady;\n }\n /** Resets all cached information relative to material (including effect and visibility) */\n resetCachedMaterial() {\n this._cachedMaterial = null;\n this._cachedEffect = null;\n this._cachedVisibility = null;\n }\n /**\n * Registers a function to be called before every frame render\n * @param func defines the function to register\n */\n registerBeforeRender(func) {\n this.onBeforeRenderObservable.add(func);\n }\n /**\n * Unregisters a function called before every frame render\n * @param func defines the function to unregister\n */\n unregisterBeforeRender(func) {\n this.onBeforeRenderObservable.removeCallback(func);\n }\n /**\n * Registers a function to be called after every frame render\n * @param func defines the function to register\n */\n registerAfterRender(func) {\n this.onAfterRenderObservable.add(func);\n }\n /**\n * Unregisters a function called after every frame render\n * @param func defines the function to unregister\n */\n unregisterAfterRender(func) {\n this.onAfterRenderObservable.removeCallback(func);\n }\n _executeOnceBeforeRender(func) {\n const execFunc = () => {\n func();\n setTimeout(() => {\n this.unregisterBeforeRender(execFunc);\n });\n };\n this.registerBeforeRender(execFunc);\n }\n /**\n * The provided function will run before render once and will be disposed afterwards.\n * A timeout delay can be provided so that the function will be executed in N ms.\n * The timeout is using the browser's native setTimeout so time percision cannot be guaranteed.\n * @param func The function to be executed.\n * @param timeout optional delay in ms\n */\n executeOnceBeforeRender(func, timeout) {\n if (timeout !== undefined) {\n setTimeout(() => {\n this._executeOnceBeforeRender(func);\n }, timeout);\n }\n else {\n this._executeOnceBeforeRender(func);\n }\n }\n /**\n * This function can help adding any object to the list of data awaited to be ready in order to check for a complete scene loading.\n * @param data defines the object to wait for\n */\n addPendingData(data) {\n this._pendingData.push(data);\n }\n /**\n * Remove a pending data from the loading list which has previously been added with addPendingData.\n * @param data defines the object to remove from the pending list\n */\n removePendingData(data) {\n const wasLoading = this.isLoading;\n const index = this._pendingData.indexOf(data);\n if (index !== -1) {\n this._pendingData.splice(index, 1);\n }\n if (wasLoading && !this.isLoading) {\n this.onDataLoadedObservable.notifyObservers(this);\n }\n }\n /**\n * Returns the number of items waiting to be loaded\n * @returns the number of items waiting to be loaded\n */\n getWaitingItemsCount() {\n return this._pendingData.length;\n }\n /**\n * Returns a boolean indicating if the scene is still loading data\n */\n get isLoading() {\n return this._pendingData.length > 0;\n }\n /**\n * Registers a function to be executed when the scene is ready\n * @param func - the function to be executed\n * @param checkRenderTargets true to also check that the meshes rendered as part of a render target are ready (default: false)\n */\n executeWhenReady(func, checkRenderTargets = false) {\n this.onReadyObservable.addOnce(func);\n if (this._executeWhenReadyTimeoutId !== null) {\n return;\n }\n this._checkIsReady(checkRenderTargets);\n }\n /**\n * Returns a promise that resolves when the scene is ready\n * @param checkRenderTargets true to also check that the meshes rendered as part of a render target are ready (default: false)\n * @returns A promise that resolves when the scene is ready\n */\n whenReadyAsync(checkRenderTargets = false) {\n return new Promise((resolve) => {\n this.executeWhenReady(() => {\n resolve();\n }, checkRenderTargets);\n });\n }\n /**\n * @internal\n */\n _checkIsReady(checkRenderTargets = false) {\n this._registerTransientComponents();\n if (this.isReady(checkRenderTargets)) {\n this.onReadyObservable.notifyObservers(this);\n this.onReadyObservable.clear();\n this._executeWhenReadyTimeoutId = null;\n return;\n }\n if (this._isDisposed) {\n this.onReadyObservable.clear();\n this._executeWhenReadyTimeoutId = null;\n return;\n }\n this._executeWhenReadyTimeoutId = setTimeout(() => {\n // Ensure materials effects are checked outside render loops\n this.incrementRenderId();\n this._checkIsReady(checkRenderTargets);\n }, 100);\n }\n /**\n * Gets all animatable attached to the scene\n */\n get animatables() {\n return this._activeAnimatables;\n }\n /**\n * Resets the last animation time frame.\n * Useful to override when animations start running when loading a scene for the first time.\n */\n resetLastAnimationTimeFrame() {\n this._animationTimeLast = PrecisionDate.Now;\n }\n // Matrix\n /**\n * Gets the current view matrix\n * @returns a Matrix\n */\n getViewMatrix() {\n return this._viewMatrix;\n }\n /**\n * Gets the current projection matrix\n * @returns a Matrix\n */\n getProjectionMatrix() {\n return this._projectionMatrix;\n }\n /**\n * Gets the current transform matrix\n * @returns a Matrix made of View * Projection\n */\n getTransformMatrix() {\n return this._transformMatrix;\n }\n /**\n * Sets the current transform matrix\n * @param viewL defines the View matrix to use\n * @param projectionL defines the Projection matrix to use\n * @param viewR defines the right View matrix to use (if provided)\n * @param projectionR defines the right Projection matrix to use (if provided)\n */\n setTransformMatrix(viewL, projectionL, viewR, projectionR) {\n // clear the multiviewSceneUbo if no viewR and projectionR are defined\n if (!viewR && !projectionR && this._multiviewSceneUbo) {\n this._multiviewSceneUbo.dispose();\n this._multiviewSceneUbo = null;\n }\n if (this._viewUpdateFlag === viewL.updateFlag && this._projectionUpdateFlag === projectionL.updateFlag) {\n return;\n }\n this._viewUpdateFlag = viewL.updateFlag;\n this._projectionUpdateFlag = projectionL.updateFlag;\n this._viewMatrix = viewL;\n this._projectionMatrix = projectionL;\n this._viewMatrix.multiplyToRef(this._projectionMatrix, this._transformMatrix);\n // Update frustum\n if (!this._frustumPlanes) {\n this._frustumPlanes = Frustum.GetPlanes(this._transformMatrix);\n }\n else {\n Frustum.GetPlanesToRef(this._transformMatrix, this._frustumPlanes);\n }\n if (this._multiviewSceneUbo && this._multiviewSceneUbo.useUbo) {\n this._updateMultiviewUbo(viewR, projectionR);\n }\n else if (this._sceneUbo.useUbo) {\n this._sceneUbo.updateMatrix(\"viewProjection\", this._transformMatrix);\n this._sceneUbo.updateMatrix(\"view\", this._viewMatrix);\n this._sceneUbo.updateMatrix(\"projection\", this._projectionMatrix);\n }\n }\n /**\n * Gets the uniform buffer used to store scene data\n * @returns a UniformBuffer\n */\n getSceneUniformBuffer() {\n return this._multiviewSceneUbo ? this._multiviewSceneUbo : this._sceneUbo;\n }\n /**\n * Creates a scene UBO\n * @param name name of the uniform buffer (optional, for debugging purpose only)\n * @returns a new ubo\n */\n createSceneUniformBuffer(name) {\n const sceneUbo = new UniformBuffer(this._engine, undefined, false, name !== null && name !== void 0 ? name : \"scene\");\n sceneUbo.addUniform(\"viewProjection\", 16);\n sceneUbo.addUniform(\"view\", 16);\n sceneUbo.addUniform(\"projection\", 16);\n sceneUbo.addUniform(\"vEyePosition\", 4);\n return sceneUbo;\n }\n /**\n * Sets the scene ubo\n * @param ubo the ubo to set for the scene\n */\n setSceneUniformBuffer(ubo) {\n this._sceneUbo = ubo;\n this._viewUpdateFlag = -1;\n this._projectionUpdateFlag = -1;\n }\n /**\n * Gets an unique (relatively to the current scene) Id\n * @returns an unique number for the scene\n */\n getUniqueId() {\n return UniqueIdGenerator.UniqueId;\n }\n /**\n * Add a mesh to the list of scene's meshes\n * @param newMesh defines the mesh to add\n * @param recursive if all child meshes should also be added to the scene\n */\n addMesh(newMesh, recursive = false) {\n if (this._blockEntityCollection) {\n return;\n }\n this.meshes.push(newMesh);\n newMesh._resyncLightSources();\n if (!newMesh.parent) {\n newMesh._addToSceneRootNodes();\n }\n this.onNewMeshAddedObservable.notifyObservers(newMesh);\n if (recursive) {\n newMesh.getChildMeshes().forEach((m) => {\n this.addMesh(m);\n });\n }\n }\n /**\n * Remove a mesh for the list of scene's meshes\n * @param toRemove defines the mesh to remove\n * @param recursive if all child meshes should also be removed from the scene\n * @returns the index where the mesh was in the mesh list\n */\n removeMesh(toRemove, recursive = false) {\n const index = this.meshes.indexOf(toRemove);\n if (index !== -1) {\n // Remove from the scene if mesh found\n this.meshes[index] = this.meshes[this.meshes.length - 1];\n this.meshes.pop();\n if (!toRemove.parent) {\n toRemove._removeFromSceneRootNodes();\n }\n }\n this._inputManager._invalidateMesh(toRemove);\n this.onMeshRemovedObservable.notifyObservers(toRemove);\n if (recursive) {\n toRemove.getChildMeshes().forEach((m) => {\n this.removeMesh(m);\n });\n }\n return index;\n }\n /**\n * Add a transform node to the list of scene's transform nodes\n * @param newTransformNode defines the transform node to add\n */\n addTransformNode(newTransformNode) {\n if (this._blockEntityCollection) {\n return;\n }\n if (newTransformNode.getScene() === this && newTransformNode._indexInSceneTransformNodesArray !== -1) {\n // Already there?\n return;\n }\n newTransformNode._indexInSceneTransformNodesArray = this.transformNodes.length;\n this.transformNodes.push(newTransformNode);\n if (!newTransformNode.parent) {\n newTransformNode._addToSceneRootNodes();\n }\n this.onNewTransformNodeAddedObservable.notifyObservers(newTransformNode);\n }\n /**\n * Remove a transform node for the list of scene's transform nodes\n * @param toRemove defines the transform node to remove\n * @returns the index where the transform node was in the transform node list\n */\n removeTransformNode(toRemove) {\n const index = toRemove._indexInSceneTransformNodesArray;\n if (index !== -1) {\n if (index !== this.transformNodes.length - 1) {\n const lastNode = this.transformNodes[this.transformNodes.length - 1];\n this.transformNodes[index] = lastNode;\n lastNode._indexInSceneTransformNodesArray = index;\n }\n toRemove._indexInSceneTransformNodesArray = -1;\n this.transformNodes.pop();\n if (!toRemove.parent) {\n toRemove._removeFromSceneRootNodes();\n }\n }\n this.onTransformNodeRemovedObservable.notifyObservers(toRemove);\n return index;\n }\n /**\n * Remove a skeleton for the list of scene's skeletons\n * @param toRemove defines the skeleton to remove\n * @returns the index where the skeleton was in the skeleton list\n */\n removeSkeleton(toRemove) {\n const index = this.skeletons.indexOf(toRemove);\n if (index !== -1) {\n // Remove from the scene if found\n this.skeletons.splice(index, 1);\n this.onSkeletonRemovedObservable.notifyObservers(toRemove);\n // Clean active container\n this._executeActiveContainerCleanup(this._activeSkeletons);\n }\n return index;\n }\n /**\n * Remove a morph target for the list of scene's morph targets\n * @param toRemove defines the morph target to remove\n * @returns the index where the morph target was in the morph target list\n */\n removeMorphTargetManager(toRemove) {\n const index = this.morphTargetManagers.indexOf(toRemove);\n if (index !== -1) {\n // Remove from the scene if found\n this.morphTargetManagers.splice(index, 1);\n }\n return index;\n }\n /**\n * Remove a light for the list of scene's lights\n * @param toRemove defines the light to remove\n * @returns the index where the light was in the light list\n */\n removeLight(toRemove) {\n const index = this.lights.indexOf(toRemove);\n if (index !== -1) {\n // Remove from meshes\n for (const mesh of this.meshes) {\n mesh._removeLightSource(toRemove, false);\n }\n // Remove from the scene if mesh found\n this.lights.splice(index, 1);\n this.sortLightsByPriority();\n if (!toRemove.parent) {\n toRemove._removeFromSceneRootNodes();\n }\n }\n this.onLightRemovedObservable.notifyObservers(toRemove);\n return index;\n }\n /**\n * Remove a camera for the list of scene's cameras\n * @param toRemove defines the camera to remove\n * @returns the index where the camera was in the camera list\n */\n removeCamera(toRemove) {\n const index = this.cameras.indexOf(toRemove);\n if (index !== -1) {\n // Remove from the scene if mesh found\n this.cameras.splice(index, 1);\n if (!toRemove.parent) {\n toRemove._removeFromSceneRootNodes();\n }\n }\n // Remove from activeCameras\n if (this.activeCameras) {\n const index2 = this.activeCameras.indexOf(toRemove);\n if (index2 !== -1) {\n // Remove from the scene if mesh found\n this.activeCameras.splice(index2, 1);\n }\n }\n // Reset the activeCamera\n if (this.activeCamera === toRemove) {\n if (this.cameras.length > 0) {\n this.activeCamera = this.cameras[0];\n }\n else {\n this.activeCamera = null;\n }\n }\n this.onCameraRemovedObservable.notifyObservers(toRemove);\n return index;\n }\n /**\n * Remove a particle system for the list of scene's particle systems\n * @param toRemove defines the particle system to remove\n * @returns the index where the particle system was in the particle system list\n */\n removeParticleSystem(toRemove) {\n const index = this.particleSystems.indexOf(toRemove);\n if (index !== -1) {\n this.particleSystems.splice(index, 1);\n // Clean active container\n this._executeActiveContainerCleanup(this._activeParticleSystems);\n }\n return index;\n }\n /**\n * Remove a animation for the list of scene's animations\n * @param toRemove defines the animation to remove\n * @returns the index where the animation was in the animation list\n */\n removeAnimation(toRemove) {\n const index = this.animations.indexOf(toRemove);\n if (index !== -1) {\n this.animations.splice(index, 1);\n }\n return index;\n }\n /**\n * Will stop the animation of the given target\n * @param target - the target\n * @param animationName - the name of the animation to stop (all animations will be stopped if both this and targetMask are empty)\n * @param targetMask - a function that determines if the animation should be stopped based on its target (all animations will be stopped if both this and animationName are empty)\n */\n stopAnimation(target, animationName, targetMask) {\n // Do nothing as code will be provided by animation component\n }\n /**\n * Removes the given animation group from this scene.\n * @param toRemove The animation group to remove\n * @returns The index of the removed animation group\n */\n removeAnimationGroup(toRemove) {\n const index = this.animationGroups.indexOf(toRemove);\n if (index !== -1) {\n this.animationGroups.splice(index, 1);\n }\n return index;\n }\n /**\n * Removes the given multi-material from this scene.\n * @param toRemove The multi-material to remove\n * @returns The index of the removed multi-material\n */\n removeMultiMaterial(toRemove) {\n const index = this.multiMaterials.indexOf(toRemove);\n if (index !== -1) {\n this.multiMaterials.splice(index, 1);\n }\n this.onMultiMaterialRemovedObservable.notifyObservers(toRemove);\n return index;\n }\n /**\n * Removes the given material from this scene.\n * @param toRemove The material to remove\n * @returns The index of the removed material\n */\n removeMaterial(toRemove) {\n const index = toRemove._indexInSceneMaterialArray;\n if (index !== -1 && index < this.materials.length) {\n if (index !== this.materials.length - 1) {\n const lastMaterial = this.materials[this.materials.length - 1];\n this.materials[index] = lastMaterial;\n lastMaterial._indexInSceneMaterialArray = index;\n }\n toRemove._indexInSceneMaterialArray = -1;\n this.materials.pop();\n }\n this.onMaterialRemovedObservable.notifyObservers(toRemove);\n return index;\n }\n /**\n * Removes the given action manager from this scene.\n * @deprecated\n * @param toRemove The action manager to remove\n * @returns The index of the removed action manager\n */\n removeActionManager(toRemove) {\n const index = this.actionManagers.indexOf(toRemove);\n if (index !== -1) {\n this.actionManagers.splice(index, 1);\n }\n return index;\n }\n /**\n * Removes the given texture from this scene.\n * @param toRemove The texture to remove\n * @returns The index of the removed texture\n */\n removeTexture(toRemove) {\n const index = this.textures.indexOf(toRemove);\n if (index !== -1) {\n this.textures.splice(index, 1);\n }\n this.onTextureRemovedObservable.notifyObservers(toRemove);\n return index;\n }\n /**\n * Adds the given light to this scene\n * @param newLight The light to add\n */\n addLight(newLight) {\n if (this._blockEntityCollection) {\n return;\n }\n this.lights.push(newLight);\n this.sortLightsByPriority();\n if (!newLight.parent) {\n newLight._addToSceneRootNodes();\n }\n // Add light to all meshes (To support if the light is removed and then re-added)\n for (const mesh of this.meshes) {\n if (mesh.lightSources.indexOf(newLight) === -1) {\n mesh.lightSources.push(newLight);\n mesh._resyncLightSources();\n }\n }\n this.onNewLightAddedObservable.notifyObservers(newLight);\n }\n /**\n * Sorts the list list based on light priorities\n */\n sortLightsByPriority() {\n if (this.requireLightSorting) {\n this.lights.sort(LightConstants.CompareLightsPriority);\n }\n }\n /**\n * Adds the given camera to this scene\n * @param newCamera The camera to add\n */\n addCamera(newCamera) {\n if (this._blockEntityCollection) {\n return;\n }\n this.cameras.push(newCamera);\n this.onNewCameraAddedObservable.notifyObservers(newCamera);\n if (!newCamera.parent) {\n newCamera._addToSceneRootNodes();\n }\n }\n /**\n * Adds the given skeleton to this scene\n * @param newSkeleton The skeleton to add\n */\n addSkeleton(newSkeleton) {\n if (this._blockEntityCollection) {\n return;\n }\n this.skeletons.push(newSkeleton);\n this.onNewSkeletonAddedObservable.notifyObservers(newSkeleton);\n }\n /**\n * Adds the given particle system to this scene\n * @param newParticleSystem The particle system to add\n */\n addParticleSystem(newParticleSystem) {\n if (this._blockEntityCollection) {\n return;\n }\n this.particleSystems.push(newParticleSystem);\n }\n /**\n * Adds the given animation to this scene\n * @param newAnimation The animation to add\n */\n addAnimation(newAnimation) {\n if (this._blockEntityCollection) {\n return;\n }\n this.animations.push(newAnimation);\n }\n /**\n * Adds the given animation group to this scene.\n * @param newAnimationGroup The animation group to add\n */\n addAnimationGroup(newAnimationGroup) {\n if (this._blockEntityCollection) {\n return;\n }\n this.animationGroups.push(newAnimationGroup);\n }\n /**\n * Adds the given multi-material to this scene\n * @param newMultiMaterial The multi-material to add\n */\n addMultiMaterial(newMultiMaterial) {\n if (this._blockEntityCollection) {\n return;\n }\n this.multiMaterials.push(newMultiMaterial);\n this.onNewMultiMaterialAddedObservable.notifyObservers(newMultiMaterial);\n }\n /**\n * Adds the given material to this scene\n * @param newMaterial The material to add\n */\n addMaterial(newMaterial) {\n if (this._blockEntityCollection) {\n return;\n }\n if (newMaterial.getScene() === this && newMaterial._indexInSceneMaterialArray !== -1) {\n // Already there??\n return;\n }\n newMaterial._indexInSceneMaterialArray = this.materials.length;\n this.materials.push(newMaterial);\n this.onNewMaterialAddedObservable.notifyObservers(newMaterial);\n }\n /**\n * Adds the given morph target to this scene\n * @param newMorphTargetManager The morph target to add\n */\n addMorphTargetManager(newMorphTargetManager) {\n if (this._blockEntityCollection) {\n return;\n }\n this.morphTargetManagers.push(newMorphTargetManager);\n }\n /**\n * Adds the given geometry to this scene\n * @param newGeometry The geometry to add\n */\n addGeometry(newGeometry) {\n if (this._blockEntityCollection) {\n return;\n }\n if (this._geometriesByUniqueId) {\n this._geometriesByUniqueId[newGeometry.uniqueId] = this.geometries.length;\n }\n this.geometries.push(newGeometry);\n }\n /**\n * Adds the given action manager to this scene\n * @deprecated\n * @param newActionManager The action manager to add\n */\n addActionManager(newActionManager) {\n this.actionManagers.push(newActionManager);\n }\n /**\n * Adds the given texture to this scene.\n * @param newTexture The texture to add\n */\n addTexture(newTexture) {\n if (this._blockEntityCollection) {\n return;\n }\n this.textures.push(newTexture);\n this.onNewTextureAddedObservable.notifyObservers(newTexture);\n }\n /**\n * Switch active camera\n * @param newCamera defines the new active camera\n * @param attachControl defines if attachControl must be called for the new active camera (default: true)\n */\n switchActiveCamera(newCamera, attachControl = true) {\n const canvas = this._engine.getInputElement();\n if (!canvas) {\n return;\n }\n if (this.activeCamera) {\n this.activeCamera.detachControl();\n }\n this.activeCamera = newCamera;\n if (attachControl) {\n newCamera.attachControl();\n }\n }\n /**\n * sets the active camera of the scene using its Id\n * @param id defines the camera's Id\n * @returns the new active camera or null if none found.\n */\n setActiveCameraById(id) {\n const camera = this.getCameraById(id);\n if (camera) {\n this.activeCamera = camera;\n return camera;\n }\n return null;\n }\n /**\n * sets the active camera of the scene using its name\n * @param name defines the camera's name\n * @returns the new active camera or null if none found.\n */\n setActiveCameraByName(name) {\n const camera = this.getCameraByName(name);\n if (camera) {\n this.activeCamera = camera;\n return camera;\n }\n return null;\n }\n /**\n * get an animation group using its name\n * @param name defines the material's name\n * @returns the animation group or null if none found.\n */\n getAnimationGroupByName(name) {\n for (let index = 0; index < this.animationGroups.length; index++) {\n if (this.animationGroups[index].name === name) {\n return this.animationGroups[index];\n }\n }\n return null;\n }\n _getMaterial(allowMultiMaterials, predicate) {\n for (let index = 0; index < this.materials.length; index++) {\n const material = this.materials[index];\n if (predicate(material)) {\n return material;\n }\n }\n if (allowMultiMaterials) {\n for (let index = 0; index < this.multiMaterials.length; index++) {\n const material = this.multiMaterials[index];\n if (predicate(material)) {\n return material;\n }\n }\n }\n return null;\n }\n /**\n * Get a material using its unique id\n * @param uniqueId defines the material's unique id\n * @param allowMultiMaterials determines whether multimaterials should be considered\n * @returns the material or null if none found.\n */\n getMaterialByUniqueID(uniqueId, allowMultiMaterials = false) {\n return this._getMaterial(allowMultiMaterials, (m) => m.uniqueId === uniqueId);\n }\n /**\n * get a material using its id\n * @param id defines the material's Id\n * @param allowMultiMaterials determines whether multimaterials should be considered\n * @returns the material or null if none found.\n */\n getMaterialById(id, allowMultiMaterials = false) {\n return this._getMaterial(allowMultiMaterials, (m) => m.id === id);\n }\n /**\n * Gets a material using its name\n * @param name defines the material's name\n * @param allowMultiMaterials determines whether multimaterials should be considered\n * @returns the material or null if none found.\n */\n getMaterialByName(name, allowMultiMaterials = false) {\n return this._getMaterial(allowMultiMaterials, (m) => m.name === name);\n }\n /**\n * Gets a last added material using a given id\n * @param id defines the material's id\n * @param allowMultiMaterials determines whether multimaterials should be considered\n * @returns the last material with the given id or null if none found.\n */\n getLastMaterialById(id, allowMultiMaterials = false) {\n for (let index = this.materials.length - 1; index >= 0; index--) {\n if (this.materials[index].id === id) {\n return this.materials[index];\n }\n }\n if (allowMultiMaterials) {\n for (let index = this.multiMaterials.length - 1; index >= 0; index--) {\n if (this.multiMaterials[index].id === id) {\n return this.multiMaterials[index];\n }\n }\n }\n return null;\n }\n /**\n * Get a texture using its unique id\n * @param uniqueId defines the texture's unique id\n * @returns the texture or null if none found.\n */\n getTextureByUniqueId(uniqueId) {\n for (let index = 0; index < this.textures.length; index++) {\n if (this.textures[index].uniqueId === uniqueId) {\n return this.textures[index];\n }\n }\n return null;\n }\n /**\n * Gets a texture using its name\n * @param name defines the texture's name\n * @returns the texture or null if none found.\n */\n getTextureByName(name) {\n for (let index = 0; index < this.textures.length; index++) {\n if (this.textures[index].name === name) {\n return this.textures[index];\n }\n }\n return null;\n }\n /**\n * Gets a camera using its Id\n * @param id defines the Id to look for\n * @returns the camera or null if not found\n */\n getCameraById(id) {\n for (let index = 0; index < this.cameras.length; index++) {\n if (this.cameras[index].id === id) {\n return this.cameras[index];\n }\n }\n return null;\n }\n /**\n * Gets a camera using its unique Id\n * @param uniqueId defines the unique Id to look for\n * @returns the camera or null if not found\n */\n getCameraByUniqueId(uniqueId) {\n for (let index = 0; index < this.cameras.length; index++) {\n if (this.cameras[index].uniqueId === uniqueId) {\n return this.cameras[index];\n }\n }\n return null;\n }\n /**\n * Gets a camera using its name\n * @param name defines the camera's name\n * @returns the camera or null if none found.\n */\n getCameraByName(name) {\n for (let index = 0; index < this.cameras.length; index++) {\n if (this.cameras[index].name === name) {\n return this.cameras[index];\n }\n }\n return null;\n }\n /**\n * Gets a bone using its Id\n * @param id defines the bone's Id\n * @returns the bone or null if not found\n */\n getBoneById(id) {\n for (let skeletonIndex = 0; skeletonIndex < this.skeletons.length; skeletonIndex++) {\n const skeleton = this.skeletons[skeletonIndex];\n for (let boneIndex = 0; boneIndex < skeleton.bones.length; boneIndex++) {\n if (skeleton.bones[boneIndex].id === id) {\n return skeleton.bones[boneIndex];\n }\n }\n }\n return null;\n }\n /**\n * Gets a bone using its id\n * @param name defines the bone's name\n * @returns the bone or null if not found\n */\n getBoneByName(name) {\n for (let skeletonIndex = 0; skeletonIndex < this.skeletons.length; skeletonIndex++) {\n const skeleton = this.skeletons[skeletonIndex];\n for (let boneIndex = 0; boneIndex < skeleton.bones.length; boneIndex++) {\n if (skeleton.bones[boneIndex].name === name) {\n return skeleton.bones[boneIndex];\n }\n }\n }\n return null;\n }\n /**\n * Gets a light node using its name\n * @param name defines the the light's name\n * @returns the light or null if none found.\n */\n getLightByName(name) {\n for (let index = 0; index < this.lights.length; index++) {\n if (this.lights[index].name === name) {\n return this.lights[index];\n }\n }\n return null;\n }\n /**\n * Gets a light node using its Id\n * @param id defines the light's Id\n * @returns the light or null if none found.\n */\n getLightById(id) {\n for (let index = 0; index < this.lights.length; index++) {\n if (this.lights[index].id === id) {\n return this.lights[index];\n }\n }\n return null;\n }\n /**\n * Gets a light node using its scene-generated unique Id\n * @param uniqueId defines the light's unique Id\n * @returns the light or null if none found.\n */\n getLightByUniqueId(uniqueId) {\n for (let index = 0; index < this.lights.length; index++) {\n if (this.lights[index].uniqueId === uniqueId) {\n return this.lights[index];\n }\n }\n return null;\n }\n /**\n * Gets a particle system by Id\n * @param id defines the particle system Id\n * @returns the corresponding system or null if none found\n */\n getParticleSystemById(id) {\n for (let index = 0; index < this.particleSystems.length; index++) {\n if (this.particleSystems[index].id === id) {\n return this.particleSystems[index];\n }\n }\n return null;\n }\n /**\n * Gets a geometry using its Id\n * @param id defines the geometry's Id\n * @returns the geometry or null if none found.\n */\n getGeometryById(id) {\n for (let index = 0; index < this.geometries.length; index++) {\n if (this.geometries[index].id === id) {\n return this.geometries[index];\n }\n }\n return null;\n }\n _getGeometryByUniqueId(uniqueId) {\n if (this._geometriesByUniqueId) {\n const index = this._geometriesByUniqueId[uniqueId];\n if (index !== undefined) {\n return this.geometries[index];\n }\n }\n else {\n for (let index = 0; index < this.geometries.length; index++) {\n if (this.geometries[index].uniqueId === uniqueId) {\n return this.geometries[index];\n }\n }\n }\n return null;\n }\n /**\n * Add a new geometry to this scene\n * @param geometry defines the geometry to be added to the scene.\n * @param force defines if the geometry must be pushed even if a geometry with this id already exists\n * @returns a boolean defining if the geometry was added or not\n */\n pushGeometry(geometry, force) {\n if (!force && this._getGeometryByUniqueId(geometry.uniqueId)) {\n return false;\n }\n this.addGeometry(geometry);\n this.onNewGeometryAddedObservable.notifyObservers(geometry);\n return true;\n }\n /**\n * Removes an existing geometry\n * @param geometry defines the geometry to be removed from the scene\n * @returns a boolean defining if the geometry was removed or not\n */\n removeGeometry(geometry) {\n let index;\n if (this._geometriesByUniqueId) {\n index = this._geometriesByUniqueId[geometry.uniqueId];\n if (index === undefined) {\n return false;\n }\n }\n else {\n index = this.geometries.indexOf(geometry);\n if (index < 0) {\n return false;\n }\n }\n if (index !== this.geometries.length - 1) {\n const lastGeometry = this.geometries[this.geometries.length - 1];\n if (lastGeometry) {\n this.geometries[index] = lastGeometry;\n if (this._geometriesByUniqueId) {\n this._geometriesByUniqueId[lastGeometry.uniqueId] = index;\n }\n }\n }\n if (this._geometriesByUniqueId) {\n this._geometriesByUniqueId[geometry.uniqueId] = undefined;\n }\n this.geometries.pop();\n this.onGeometryRemovedObservable.notifyObservers(geometry);\n return true;\n }\n /**\n * Gets the list of geometries attached to the scene\n * @returns an array of Geometry\n */\n getGeometries() {\n return this.geometries;\n }\n /**\n * Gets the first added mesh found of a given Id\n * @param id defines the Id to search for\n * @returns the mesh found or null if not found at all\n */\n getMeshById(id) {\n for (let index = 0; index < this.meshes.length; index++) {\n if (this.meshes[index].id === id) {\n return this.meshes[index];\n }\n }\n return null;\n }\n /**\n * Gets a list of meshes using their Id\n * @param id defines the Id to search for\n * @returns a list of meshes\n */\n getMeshesById(id) {\n return this.meshes.filter(function (m) {\n return m.id === id;\n });\n }\n /**\n * Gets the first added transform node found of a given Id\n * @param id defines the Id to search for\n * @returns the found transform node or null if not found at all.\n */\n getTransformNodeById(id) {\n for (let index = 0; index < this.transformNodes.length; index++) {\n if (this.transformNodes[index].id === id) {\n return this.transformNodes[index];\n }\n }\n return null;\n }\n /**\n * Gets a transform node with its auto-generated unique Id\n * @param uniqueId defines the unique Id to search for\n * @returns the found transform node or null if not found at all.\n */\n getTransformNodeByUniqueId(uniqueId) {\n for (let index = 0; index < this.transformNodes.length; index++) {\n if (this.transformNodes[index].uniqueId === uniqueId) {\n return this.transformNodes[index];\n }\n }\n return null;\n }\n /**\n * Gets a list of transform nodes using their Id\n * @param id defines the Id to search for\n * @returns a list of transform nodes\n */\n getTransformNodesById(id) {\n return this.transformNodes.filter(function (m) {\n return m.id === id;\n });\n }\n /**\n * Gets a mesh with its auto-generated unique Id\n * @param uniqueId defines the unique Id to search for\n * @returns the found mesh or null if not found at all.\n */\n getMeshByUniqueId(uniqueId) {\n for (let index = 0; index < this.meshes.length; index++) {\n if (this.meshes[index].uniqueId === uniqueId) {\n return this.meshes[index];\n }\n }\n return null;\n }\n /**\n * Gets a the last added mesh using a given Id\n * @param id defines the Id to search for\n * @returns the found mesh or null if not found at all.\n */\n getLastMeshById(id) {\n for (let index = this.meshes.length - 1; index >= 0; index--) {\n if (this.meshes[index].id === id) {\n return this.meshes[index];\n }\n }\n return null;\n }\n /**\n * Gets a the last transform node using a given Id\n * @param id defines the Id to search for\n * @returns the found mesh or null if not found at all.\n */\n getLastTransformNodeById(id) {\n for (let index = this.transformNodes.length - 1; index >= 0; index--) {\n if (this.transformNodes[index].id === id) {\n return this.transformNodes[index];\n }\n }\n return null;\n }\n /**\n * Gets a the last added node (Mesh, Camera, Light) using a given Id\n * @param id defines the Id to search for\n * @returns the found node or null if not found at all\n */\n getLastEntryById(id) {\n let index;\n for (index = this.meshes.length - 1; index >= 0; index--) {\n if (this.meshes[index].id === id) {\n return this.meshes[index];\n }\n }\n for (index = this.transformNodes.length - 1; index >= 0; index--) {\n if (this.transformNodes[index].id === id) {\n return this.transformNodes[index];\n }\n }\n for (index = this.cameras.length - 1; index >= 0; index--) {\n if (this.cameras[index].id === id) {\n return this.cameras[index];\n }\n }\n for (index = this.lights.length - 1; index >= 0; index--) {\n if (this.lights[index].id === id) {\n return this.lights[index];\n }\n }\n return null;\n }\n /**\n * Gets a node (Mesh, Camera, Light) using a given Id\n * @param id defines the Id to search for\n * @returns the found node or null if not found at all\n */\n getNodeById(id) {\n const mesh = this.getMeshById(id);\n if (mesh) {\n return mesh;\n }\n const transformNode = this.getTransformNodeById(id);\n if (transformNode) {\n return transformNode;\n }\n const light = this.getLightById(id);\n if (light) {\n return light;\n }\n const camera = this.getCameraById(id);\n if (camera) {\n return camera;\n }\n const bone = this.getBoneById(id);\n if (bone) {\n return bone;\n }\n return null;\n }\n /**\n * Gets a node (Mesh, Camera, Light) using a given name\n * @param name defines the name to search for\n * @returns the found node or null if not found at all.\n */\n getNodeByName(name) {\n const mesh = this.getMeshByName(name);\n if (mesh) {\n return mesh;\n }\n const transformNode = this.getTransformNodeByName(name);\n if (transformNode) {\n return transformNode;\n }\n const light = this.getLightByName(name);\n if (light) {\n return light;\n }\n const camera = this.getCameraByName(name);\n if (camera) {\n return camera;\n }\n const bone = this.getBoneByName(name);\n if (bone) {\n return bone;\n }\n return null;\n }\n /**\n * Gets a mesh using a given name\n * @param name defines the name to search for\n * @returns the found mesh or null if not found at all.\n */\n getMeshByName(name) {\n for (let index = 0; index < this.meshes.length; index++) {\n if (this.meshes[index].name === name) {\n return this.meshes[index];\n }\n }\n return null;\n }\n /**\n * Gets a transform node using a given name\n * @param name defines the name to search for\n * @returns the found transform node or null if not found at all.\n */\n getTransformNodeByName(name) {\n for (let index = 0; index < this.transformNodes.length; index++) {\n if (this.transformNodes[index].name === name) {\n return this.transformNodes[index];\n }\n }\n return null;\n }\n /**\n * Gets a skeleton using a given Id (if many are found, this function will pick the last one)\n * @param id defines the Id to search for\n * @returns the found skeleton or null if not found at all.\n */\n getLastSkeletonById(id) {\n for (let index = this.skeletons.length - 1; index >= 0; index--) {\n if (this.skeletons[index].id === id) {\n return this.skeletons[index];\n }\n }\n return null;\n }\n /**\n * Gets a skeleton using a given auto generated unique id\n * @param uniqueId defines the unique id to search for\n * @returns the found skeleton or null if not found at all.\n */\n getSkeletonByUniqueId(uniqueId) {\n for (let index = 0; index < this.skeletons.length; index++) {\n if (this.skeletons[index].uniqueId === uniqueId) {\n return this.skeletons[index];\n }\n }\n return null;\n }\n /**\n * Gets a skeleton using a given id (if many are found, this function will pick the first one)\n * @param id defines the id to search for\n * @returns the found skeleton or null if not found at all.\n */\n getSkeletonById(id) {\n for (let index = 0; index < this.skeletons.length; index++) {\n if (this.skeletons[index].id === id) {\n return this.skeletons[index];\n }\n }\n return null;\n }\n /**\n * Gets a skeleton using a given name\n * @param name defines the name to search for\n * @returns the found skeleton or null if not found at all.\n */\n getSkeletonByName(name) {\n for (let index = 0; index < this.skeletons.length; index++) {\n if (this.skeletons[index].name === name) {\n return this.skeletons[index];\n }\n }\n return null;\n }\n /**\n * Gets a morph target manager using a given id (if many are found, this function will pick the last one)\n * @param id defines the id to search for\n * @returns the found morph target manager or null if not found at all.\n */\n getMorphTargetManagerById(id) {\n for (let index = 0; index < this.morphTargetManagers.length; index++) {\n if (this.morphTargetManagers[index].uniqueId === id) {\n return this.morphTargetManagers[index];\n }\n }\n return null;\n }\n /**\n * Gets a morph target using a given id (if many are found, this function will pick the first one)\n * @param id defines the id to search for\n * @returns the found morph target or null if not found at all.\n */\n getMorphTargetById(id) {\n for (let managerIndex = 0; managerIndex < this.morphTargetManagers.length; ++managerIndex) {\n const morphTargetManager = this.morphTargetManagers[managerIndex];\n for (let index = 0; index < morphTargetManager.numTargets; ++index) {\n const target = morphTargetManager.getTarget(index);\n if (target.id === id) {\n return target;\n }\n }\n }\n return null;\n }\n /**\n * Gets a morph target using a given name (if many are found, this function will pick the first one)\n * @param name defines the name to search for\n * @returns the found morph target or null if not found at all.\n */\n getMorphTargetByName(name) {\n for (let managerIndex = 0; managerIndex < this.morphTargetManagers.length; ++managerIndex) {\n const morphTargetManager = this.morphTargetManagers[managerIndex];\n for (let index = 0; index < morphTargetManager.numTargets; ++index) {\n const target = morphTargetManager.getTarget(index);\n if (target.name === name) {\n return target;\n }\n }\n }\n return null;\n }\n /**\n * Gets a post process using a given name (if many are found, this function will pick the first one)\n * @param name defines the name to search for\n * @returns the found post process or null if not found at all.\n */\n getPostProcessByName(name) {\n for (let postProcessIndex = 0; postProcessIndex < this.postProcesses.length; ++postProcessIndex) {\n const postProcess = this.postProcesses[postProcessIndex];\n if (postProcess.name === name) {\n return postProcess;\n }\n }\n return null;\n }\n /**\n * Gets a boolean indicating if the given mesh is active\n * @param mesh defines the mesh to look for\n * @returns true if the mesh is in the active list\n */\n isActiveMesh(mesh) {\n return this._activeMeshes.indexOf(mesh) !== -1;\n }\n /**\n * Return a unique id as a string which can serve as an identifier for the scene\n */\n get uid() {\n if (!this._uid) {\n this._uid = Tools.RandomId();\n }\n return this._uid;\n }\n /**\n * Add an externally attached data from its key.\n * This method call will fail and return false, if such key already exists.\n * If you don't care and just want to get the data no matter what, use the more convenient getOrAddExternalDataWithFactory() method.\n * @param key the unique key that identifies the data\n * @param data the data object to associate to the key for this Engine instance\n * @returns true if no such key were already present and the data was added successfully, false otherwise\n */\n addExternalData(key, data) {\n if (!this._externalData) {\n this._externalData = new StringDictionary();\n }\n return this._externalData.add(key, data);\n }\n /**\n * Get an externally attached data from its key\n * @param key the unique key that identifies the data\n * @returns the associated data, if present (can be null), or undefined if not present\n */\n getExternalData(key) {\n if (!this._externalData) {\n return null;\n }\n return this._externalData.get(key);\n }\n /**\n * Get an externally attached data from its key, create it using a factory if it's not already present\n * @param key the unique key that identifies the data\n * @param factory the factory that will be called to create the instance if and only if it doesn't exists\n * @returns the associated data, can be null if the factory returned null.\n */\n getOrAddExternalDataWithFactory(key, factory) {\n if (!this._externalData) {\n this._externalData = new StringDictionary();\n }\n return this._externalData.getOrAddWithFactory(key, factory);\n }\n /**\n * Remove an externally attached data from the Engine instance\n * @param key the unique key that identifies the data\n * @returns true if the data was successfully removed, false if it doesn't exist\n */\n removeExternalData(key) {\n return this._externalData.remove(key);\n }\n _evaluateSubMesh(subMesh, mesh, initialMesh, forcePush) {\n if (forcePush || subMesh.isInFrustum(this._frustumPlanes)) {\n for (const step of this._evaluateSubMeshStage) {\n step.action(mesh, subMesh);\n }\n const material = subMesh.getMaterial();\n if (material !== null && material !== undefined) {\n // Render targets\n if (material.hasRenderTargetTextures && material.getRenderTargetTextures != null) {\n if (this._processedMaterials.indexOf(material) === -1) {\n this._processedMaterials.push(material);\n this._materialsRenderTargets.concatWithNoDuplicate(material.getRenderTargetTextures());\n }\n }\n // Dispatch\n this._renderingManager.dispatch(subMesh, mesh, material);\n }\n }\n }\n /**\n * Clear the processed materials smart array preventing retention point in material dispose.\n */\n freeProcessedMaterials() {\n this._processedMaterials.dispose();\n }\n /** Gets or sets a boolean blocking all the calls to freeActiveMeshes and freeRenderingGroups\n * It can be used in order to prevent going through methods freeRenderingGroups and freeActiveMeshes several times to improve performance\n * when disposing several meshes in a row or a hierarchy of meshes.\n * When used, it is the responsibility of the user to blockfreeActiveMeshesAndRenderingGroups back to false.\n */\n get blockfreeActiveMeshesAndRenderingGroups() {\n return this._preventFreeActiveMeshesAndRenderingGroups;\n }\n set blockfreeActiveMeshesAndRenderingGroups(value) {\n if (this._preventFreeActiveMeshesAndRenderingGroups === value) {\n return;\n }\n if (value) {\n this.freeActiveMeshes();\n this.freeRenderingGroups();\n }\n this._preventFreeActiveMeshesAndRenderingGroups = value;\n }\n /**\n * Clear the active meshes smart array preventing retention point in mesh dispose.\n */\n freeActiveMeshes() {\n if (this.blockfreeActiveMeshesAndRenderingGroups) {\n return;\n }\n this._activeMeshes.dispose();\n if (this.activeCamera && this.activeCamera._activeMeshes) {\n this.activeCamera._activeMeshes.dispose();\n }\n if (this.activeCameras) {\n for (let i = 0; i < this.activeCameras.length; i++) {\n const activeCamera = this.activeCameras[i];\n if (activeCamera && activeCamera._activeMeshes) {\n activeCamera._activeMeshes.dispose();\n }\n }\n }\n }\n /**\n * Clear the info related to rendering groups preventing retention points during dispose.\n */\n freeRenderingGroups() {\n if (this.blockfreeActiveMeshesAndRenderingGroups) {\n return;\n }\n if (this._renderingManager) {\n this._renderingManager.freeRenderingGroups();\n }\n if (this.textures) {\n for (let i = 0; i < this.textures.length; i++) {\n const texture = this.textures[i];\n if (texture && texture.renderList) {\n texture.freeRenderingGroups();\n }\n }\n }\n }\n /** @internal */\n _isInIntermediateRendering() {\n return this._intermediateRendering;\n }\n /**\n * Use this function to stop evaluating active meshes. The current list will be keep alive between frames\n * @param skipEvaluateActiveMeshes defines an optional boolean indicating that the evaluate active meshes step must be completely skipped\n * @param onSuccess optional success callback\n * @param onError optional error callback\n * @param freezeMeshes defines if meshes should be frozen (true by default)\n * @param keepFrustumCulling defines if you want to keep running the frustum clipping (false by default)\n * @returns the current scene\n */\n freezeActiveMeshes(skipEvaluateActiveMeshes = false, onSuccess, onError, freezeMeshes = true, keepFrustumCulling = false) {\n this.executeWhenReady(() => {\n if (!this.activeCamera) {\n onError && onError(\"No active camera found\");\n return;\n }\n if (!this._frustumPlanes) {\n this.updateTransformMatrix();\n }\n this._evaluateActiveMeshes();\n this._activeMeshesFrozen = true;\n this._activeMeshesFrozenButKeepClipping = keepFrustumCulling;\n this._skipEvaluateActiveMeshesCompletely = skipEvaluateActiveMeshes;\n if (freezeMeshes) {\n for (let index = 0; index < this._activeMeshes.length; index++) {\n this._activeMeshes.data[index]._freeze();\n }\n }\n onSuccess && onSuccess();\n });\n return this;\n }\n /**\n * Use this function to restart evaluating active meshes on every frame\n * @returns the current scene\n */\n unfreezeActiveMeshes() {\n for (let index = 0; index < this.meshes.length; index++) {\n const mesh = this.meshes[index];\n if (mesh._internalAbstractMeshDataInfo) {\n mesh._internalAbstractMeshDataInfo._isActive = false;\n }\n }\n for (let index = 0; index < this._activeMeshes.length; index++) {\n this._activeMeshes.data[index]._unFreeze();\n }\n this._activeMeshesFrozen = false;\n return this;\n }\n _executeActiveContainerCleanup(container) {\n const isInFastMode = this._engine.snapshotRendering && this._engine.snapshotRenderingMode === 1;\n if (!isInFastMode && this._activeMeshesFrozen && this._activeMeshes.length) {\n return; // Do not execute in frozen mode\n }\n // We need to ensure we are not in the rendering loop\n this.onBeforeRenderObservable.addOnce(() => container.dispose());\n }\n _evaluateActiveMeshes() {\n var _a;\n if (this._engine.snapshotRendering && this._engine.snapshotRenderingMode === 1) {\n if (this._activeMeshes.length > 0) {\n (_a = this.activeCamera) === null || _a === void 0 ? void 0 : _a._activeMeshes.reset();\n this._activeMeshes.reset();\n this._renderingManager.reset();\n this._processedMaterials.reset();\n this._activeParticleSystems.reset();\n this._activeSkeletons.reset();\n this._softwareSkinnedMeshes.reset();\n }\n return;\n }\n if (this._activeMeshesFrozen && this._activeMeshes.length) {\n if (!this._skipEvaluateActiveMeshesCompletely) {\n const len = this._activeMeshes.length;\n for (let i = 0; i < len; i++) {\n const mesh = this._activeMeshes.data[i];\n mesh.computeWorldMatrix();\n }\n }\n if (this._activeParticleSystems) {\n const psLength = this._activeParticleSystems.length;\n for (let i = 0; i < psLength; i++) {\n this._activeParticleSystems.data[i].animate();\n }\n }\n this._renderingManager.resetSprites();\n return;\n }\n if (!this.activeCamera) {\n return;\n }\n this.onBeforeActiveMeshesEvaluationObservable.notifyObservers(this);\n this.activeCamera._activeMeshes.reset();\n this._activeMeshes.reset();\n this._renderingManager.reset();\n this._processedMaterials.reset();\n this._activeParticleSystems.reset();\n this._activeSkeletons.reset();\n this._softwareSkinnedMeshes.reset();\n this._materialsRenderTargets.reset();\n for (const step of this._beforeEvaluateActiveMeshStage) {\n step.action();\n }\n // Determine mesh candidates\n const meshes = this.getActiveMeshCandidates();\n // Check each mesh\n const len = meshes.length;\n for (let i = 0; i < len; i++) {\n const mesh = meshes.data[i];\n mesh._internalAbstractMeshDataInfo._currentLODIsUpToDate = false;\n if (mesh.isBlocked) {\n continue;\n }\n this._totalVertices.addCount(mesh.getTotalVertices(), false);\n if (!mesh.isReady() || !mesh.isEnabled() || mesh.scaling.hasAZeroComponent) {\n continue;\n }\n mesh.computeWorldMatrix();\n // Intersections\n if (mesh.actionManager && mesh.actionManager.hasSpecificTriggers2(12, 13)) {\n this._meshesForIntersections.pushNoDuplicate(mesh);\n }\n // Switch to current LOD\n let meshToRender = this.customLODSelector ? this.customLODSelector(mesh, this.activeCamera) : mesh.getLOD(this.activeCamera);\n mesh._internalAbstractMeshDataInfo._currentLOD = meshToRender;\n mesh._internalAbstractMeshDataInfo._currentLODIsUpToDate = true;\n if (meshToRender === undefined || meshToRender === null) {\n continue;\n }\n // Compute world matrix if LOD is billboard\n if (meshToRender !== mesh && meshToRender.billboardMode !== 0) {\n meshToRender.computeWorldMatrix();\n }\n mesh._preActivate();\n if (mesh.isVisible &&\n mesh.visibility > 0 &&\n (mesh.layerMask & this.activeCamera.layerMask) !== 0 &&\n (this._skipFrustumClipping || mesh.alwaysSelectAsActiveMesh || mesh.isInFrustum(this._frustumPlanes))) {\n this._activeMeshes.push(mesh);\n this.activeCamera._activeMeshes.push(mesh);\n if (meshToRender !== mesh) {\n meshToRender._activate(this._renderId, false);\n }\n for (const step of this._preActiveMeshStage) {\n step.action(mesh);\n }\n if (mesh._activate(this._renderId, false)) {\n if (!mesh.isAnInstance) {\n meshToRender._internalAbstractMeshDataInfo._onlyForInstances = false;\n }\n else {\n if (mesh._internalAbstractMeshDataInfo._actAsRegularMesh) {\n meshToRender = mesh;\n }\n }\n meshToRender._internalAbstractMeshDataInfo._isActive = true;\n this._activeMesh(mesh, meshToRender);\n }\n mesh._postActivate();\n }\n }\n this.onAfterActiveMeshesEvaluationObservable.notifyObservers(this);\n // Particle systems\n if (this.particlesEnabled) {\n this.onBeforeParticlesRenderingObservable.notifyObservers(this);\n for (let particleIndex = 0; particleIndex < this.particleSystems.length; particleIndex++) {\n const particleSystem = this.particleSystems[particleIndex];\n if (!particleSystem.isStarted() || !particleSystem.emitter) {\n continue;\n }\n const emitter = particleSystem.emitter;\n if (!emitter.position || emitter.isEnabled()) {\n this._activeParticleSystems.push(particleSystem);\n particleSystem.animate();\n this._renderingManager.dispatchParticles(particleSystem);\n }\n }\n this.onAfterParticlesRenderingObservable.notifyObservers(this);\n }\n }\n _activeMesh(sourceMesh, mesh) {\n if (this._skeletonsEnabled && mesh.skeleton !== null && mesh.skeleton !== undefined) {\n if (this._activeSkeletons.pushNoDuplicate(mesh.skeleton)) {\n mesh.skeleton.prepare();\n this._activeBones.addCount(mesh.skeleton.bones.length, false);\n }\n if (!mesh.computeBonesUsingShaders) {\n this._softwareSkinnedMeshes.pushNoDuplicate(mesh);\n }\n }\n let forcePush = sourceMesh.hasInstances || sourceMesh.isAnInstance || this.dispatchAllSubMeshesOfActiveMeshes || this._skipFrustumClipping || mesh.alwaysSelectAsActiveMesh;\n if (mesh && mesh.subMeshes && mesh.subMeshes.length > 0) {\n const subMeshes = this.getActiveSubMeshCandidates(mesh);\n const len = subMeshes.length;\n forcePush = forcePush || len === 1;\n for (let i = 0; i < len; i++) {\n const subMesh = subMeshes.data[i];\n this._evaluateSubMesh(subMesh, mesh, sourceMesh, forcePush);\n }\n }\n }\n /**\n * Update the transform matrix to update from the current active camera\n * @param force defines a boolean used to force the update even if cache is up to date\n */\n updateTransformMatrix(force) {\n if (!this.activeCamera) {\n return;\n }\n if (this.activeCamera._renderingMultiview) {\n const leftCamera = this.activeCamera._rigCameras[0];\n const rightCamera = this.activeCamera._rigCameras[1];\n this.setTransformMatrix(leftCamera.getViewMatrix(), leftCamera.getProjectionMatrix(force), rightCamera.getViewMatrix(), rightCamera.getProjectionMatrix(force));\n }\n else {\n this.setTransformMatrix(this.activeCamera.getViewMatrix(), this.activeCamera.getProjectionMatrix(force));\n }\n }\n _bindFrameBuffer(camera, clear = true) {\n if (camera && camera._multiviewTexture) {\n camera._multiviewTexture._bindFrameBuffer();\n }\n else if (camera && camera.outputRenderTarget) {\n camera.outputRenderTarget._bindFrameBuffer();\n }\n else {\n if (!this._engine._currentFrameBufferIsDefaultFrameBuffer()) {\n this._engine.restoreDefaultFramebuffer();\n }\n }\n if (clear) {\n this._clearFrameBuffer(camera);\n }\n }\n _clearFrameBuffer(camera) {\n // we assume the framebuffer currently bound is the right one\n if (camera && camera._multiviewTexture) ;\n else if (camera && camera.outputRenderTarget && !camera._renderingMultiview) {\n const rtt = camera.outputRenderTarget;\n if (rtt.onClearObservable.hasObservers()) {\n rtt.onClearObservable.notifyObservers(this._engine);\n }\n else if (!rtt.skipInitialClear) {\n if (this.autoClear) {\n this._engine.clear(rtt.clearColor || this.clearColor, !rtt._cleared, true, true);\n }\n rtt._cleared = true;\n }\n }\n else {\n if (!this._defaultFrameBufferCleared) {\n this._defaultFrameBufferCleared = true;\n this._clear();\n }\n else {\n this._engine.clear(null, false, true, true);\n }\n }\n }\n /**\n * @internal\n */\n _renderForCamera(camera, rigParent, bindFrameBuffer = true) {\n var _a, _b, _c;\n if (camera && camera._skipRendering) {\n return;\n }\n const engine = this._engine;\n // Use _activeCamera instead of activeCamera to avoid onActiveCameraChanged\n this._activeCamera = camera;\n if (!this.activeCamera) {\n throw new Error(\"Active camera not set\");\n }\n // Viewport\n engine.setViewport(this.activeCamera.viewport);\n // Camera\n this.resetCachedMaterial();\n this._renderId++;\n if (!this.prePass && bindFrameBuffer) {\n let skipInitialClear = true;\n if (camera._renderingMultiview && camera.outputRenderTarget) {\n skipInitialClear = camera.outputRenderTarget.skipInitialClear;\n if (this.autoClear) {\n this._defaultFrameBufferCleared = false;\n camera.outputRenderTarget.skipInitialClear = false;\n }\n }\n this._bindFrameBuffer(this._activeCamera);\n if (camera._renderingMultiview && camera.outputRenderTarget) {\n camera.outputRenderTarget.skipInitialClear = skipInitialClear;\n }\n }\n this.updateTransformMatrix();\n this.onBeforeCameraRenderObservable.notifyObservers(this.activeCamera);\n // Meshes\n this._evaluateActiveMeshes();\n // Software skinning\n for (let softwareSkinnedMeshIndex = 0; softwareSkinnedMeshIndex < this._softwareSkinnedMeshes.length; softwareSkinnedMeshIndex++) {\n const mesh = this._softwareSkinnedMeshes.data[softwareSkinnedMeshIndex];\n mesh.applySkeleton(mesh.skeleton);\n }\n // Render targets\n this.onBeforeRenderTargetsRenderObservable.notifyObservers(this);\n this._renderTargets.concatWithNoDuplicate(this._materialsRenderTargets);\n if (camera.customRenderTargets && camera.customRenderTargets.length > 0) {\n this._renderTargets.concatWithNoDuplicate(camera.customRenderTargets);\n }\n if (rigParent && rigParent.customRenderTargets && rigParent.customRenderTargets.length > 0) {\n this._renderTargets.concatWithNoDuplicate(rigParent.customRenderTargets);\n }\n if (this.environmentTexture && this.environmentTexture.isRenderTarget) {\n this._renderTargets.pushNoDuplicate(this.environmentTexture);\n }\n // Collects render targets from external components.\n for (const step of this._gatherActiveCameraRenderTargetsStage) {\n step.action(this._renderTargets);\n }\n let needRebind = false;\n if (this.renderTargetsEnabled) {\n this._intermediateRendering = true;\n if (this._renderTargets.length > 0) {\n Tools.StartPerformanceCounter(\"Render targets\", this._renderTargets.length > 0);\n for (let renderIndex = 0; renderIndex < this._renderTargets.length; renderIndex++) {\n const renderTarget = this._renderTargets.data[renderIndex];\n if (renderTarget._shouldRender()) {\n this._renderId++;\n const hasSpecialRenderTargetCamera = renderTarget.activeCamera && renderTarget.activeCamera !== this.activeCamera;\n renderTarget.render(hasSpecialRenderTargetCamera, this.dumpNextRenderTargets);\n needRebind = true;\n }\n }\n Tools.EndPerformanceCounter(\"Render targets\", this._renderTargets.length > 0);\n this._renderId++;\n }\n for (const step of this._cameraDrawRenderTargetStage) {\n needRebind = step.action(this.activeCamera) || needRebind;\n }\n this._intermediateRendering = false;\n }\n this._engine.currentRenderPassId = (_c = (_b = (_a = camera.outputRenderTarget) === null || _a === void 0 ? void 0 : _a.renderPassId) !== null && _b !== void 0 ? _b : camera.renderPassId) !== null && _c !== void 0 ? _c : 0;\n // Restore framebuffer after rendering to targets\n if (needRebind && !this.prePass) {\n this._bindFrameBuffer(this._activeCamera, false);\n }\n this.onAfterRenderTargetsRenderObservable.notifyObservers(this);\n // Prepare Frame\n if (this.postProcessManager && !camera._multiviewTexture && !this.prePass) {\n this.postProcessManager._prepareFrame();\n }\n // Before Camera Draw\n for (const step of this._beforeCameraDrawStage) {\n step.action(this.activeCamera);\n }\n // Render\n this.onBeforeDrawPhaseObservable.notifyObservers(this);\n if (engine.snapshotRendering && engine.snapshotRenderingMode === 1) {\n this.finalizeSceneUbo();\n }\n this._renderingManager.render(null, null, true, true);\n this.onAfterDrawPhaseObservable.notifyObservers(this);\n // After Camera Draw\n for (const step of this._afterCameraDrawStage) {\n step.action(this.activeCamera);\n }\n // Finalize frame\n if (this.postProcessManager && !camera._multiviewTexture) {\n // if the camera has an output render target, render the post process to the render target\n const texture = camera.outputRenderTarget ? camera.outputRenderTarget.renderTarget : undefined;\n this.postProcessManager._finalizeFrame(camera.isIntermediate, texture);\n }\n // After post process\n for (const step of this._afterCameraPostProcessStage) {\n step.action(this.activeCamera);\n }\n // Reset some special arrays\n this._renderTargets.reset();\n this.onAfterCameraRenderObservable.notifyObservers(this.activeCamera);\n }\n _processSubCameras(camera, bindFrameBuffer = true) {\n if (camera.cameraRigMode === 0 || camera._renderingMultiview) {\n if (camera._renderingMultiview && !this._multiviewSceneUbo) {\n this._createMultiviewUbo();\n }\n this._renderForCamera(camera, undefined, bindFrameBuffer);\n this.onAfterRenderCameraObservable.notifyObservers(camera);\n return;\n }\n if (camera._useMultiviewToSingleView) {\n this._renderMultiviewToSingleView(camera);\n }\n else {\n // rig cameras\n this.onBeforeCameraRenderObservable.notifyObservers(camera);\n for (let index = 0; index < camera._rigCameras.length; index++) {\n this._renderForCamera(camera._rigCameras[index], camera);\n }\n }\n // Use _activeCamera instead of activeCamera to avoid onActiveCameraChanged\n this._activeCamera = camera;\n this.updateTransformMatrix();\n this.onAfterRenderCameraObservable.notifyObservers(camera);\n }\n _checkIntersections() {\n for (let index = 0; index < this._meshesForIntersections.length; index++) {\n const sourceMesh = this._meshesForIntersections.data[index];\n if (!sourceMesh.actionManager) {\n continue;\n }\n for (let actionIndex = 0; sourceMesh.actionManager && actionIndex < sourceMesh.actionManager.actions.length; actionIndex++) {\n const action = sourceMesh.actionManager.actions[actionIndex];\n if (action.trigger === 12 || action.trigger === 13) {\n const parameters = action.getTriggerParameter();\n const otherMesh = parameters.mesh ? parameters.mesh : parameters;\n const areIntersecting = otherMesh.intersectsMesh(sourceMesh, parameters.usePreciseIntersection);\n const currentIntersectionInProgress = sourceMesh._intersectionsInProgress.indexOf(otherMesh);\n if (areIntersecting && currentIntersectionInProgress === -1) {\n if (action.trigger === 12) {\n action._executeCurrent(ActionEvent.CreateNew(sourceMesh, undefined, otherMesh));\n sourceMesh._intersectionsInProgress.push(otherMesh);\n }\n else if (action.trigger === 13) {\n sourceMesh._intersectionsInProgress.push(otherMesh);\n }\n }\n else if (!areIntersecting && currentIntersectionInProgress > -1) {\n //They intersected, and now they don't.\n //is this trigger an exit trigger? execute an event.\n if (action.trigger === 13) {\n action._executeCurrent(ActionEvent.CreateNew(sourceMesh, undefined, otherMesh));\n }\n //if this is an exit trigger, or no exit trigger exists, remove the id from the intersection in progress array.\n if (!sourceMesh.actionManager.hasSpecificTrigger(13, (parameter) => {\n const parameterMesh = parameter.mesh ? parameter.mesh : parameter;\n return otherMesh === parameterMesh;\n }) ||\n action.trigger === 13) {\n sourceMesh._intersectionsInProgress.splice(currentIntersectionInProgress, 1);\n }\n }\n }\n }\n }\n }\n /**\n * @internal\n */\n _advancePhysicsEngineStep(step) {\n // Do nothing. Code will be replaced if physics engine component is referenced\n }\n /** @internal */\n _animate() {\n // Nothing to do as long as Animatable have not been imported.\n }\n /** Execute all animations (for a frame) */\n animate() {\n if (this._engine.isDeterministicLockStep()) {\n let deltaTime = Math.max(Scene.MinDeltaTime, Math.min(this._engine.getDeltaTime(), Scene.MaxDeltaTime)) + this._timeAccumulator;\n const defaultFrameTime = this._engine.getTimeStep();\n const defaultFPS = 1000.0 / defaultFrameTime / 1000.0;\n let stepsTaken = 0;\n const maxSubSteps = this._engine.getLockstepMaxSteps();\n let internalSteps = Math.floor(deltaTime / defaultFrameTime);\n internalSteps = Math.min(internalSteps, maxSubSteps);\n while (deltaTime > 0 && stepsTaken < internalSteps) {\n this.onBeforeStepObservable.notifyObservers(this);\n // Animations\n this._animationRatio = defaultFrameTime * defaultFPS;\n this._animate();\n this.onAfterAnimationsObservable.notifyObservers(this);\n // Physics\n if (this.physicsEnabled) {\n this._advancePhysicsEngineStep(defaultFrameTime);\n }\n this.onAfterStepObservable.notifyObservers(this);\n this._currentStepId++;\n stepsTaken++;\n deltaTime -= defaultFrameTime;\n }\n this._timeAccumulator = deltaTime < 0 ? 0 : deltaTime;\n }\n else {\n // Animations\n const deltaTime = this.useConstantAnimationDeltaTime ? 16 : Math.max(Scene.MinDeltaTime, Math.min(this._engine.getDeltaTime(), Scene.MaxDeltaTime));\n this._animationRatio = deltaTime * (60.0 / 1000.0);\n this._animate();\n this.onAfterAnimationsObservable.notifyObservers(this);\n // Physics\n if (this.physicsEnabled) {\n this._advancePhysicsEngineStep(deltaTime);\n }\n }\n }\n _clear() {\n if (this.autoClearDepthAndStencil || this.autoClear) {\n this._engine.clear(this.clearColor, this.autoClear || this.forceWireframe || this.forcePointsCloud, this.autoClearDepthAndStencil, this.autoClearDepthAndStencil);\n }\n }\n _checkCameraRenderTarget(camera) {\n var _a;\n if ((camera === null || camera === void 0 ? void 0 : camera.outputRenderTarget) && !(camera === null || camera === void 0 ? void 0 : camera.isRigCamera)) {\n camera.outputRenderTarget._cleared = false;\n }\n if ((_a = camera === null || camera === void 0 ? void 0 : camera.rigCameras) === null || _a === void 0 ? void 0 : _a.length) {\n for (let i = 0; i < camera.rigCameras.length; ++i) {\n const rtt = camera.rigCameras[i].outputRenderTarget;\n if (rtt) {\n rtt._cleared = false;\n }\n }\n }\n }\n /**\n * Resets the draw wrappers cache of all meshes\n * @param passId If provided, releases only the draw wrapper corresponding to this render pass id\n */\n resetDrawCache(passId) {\n if (!this.meshes) {\n return;\n }\n for (const mesh of this.meshes) {\n mesh.resetDrawCache(passId);\n }\n }\n /**\n * Render the scene\n * @param updateCameras defines a boolean indicating if cameras must update according to their inputs (true by default)\n * @param ignoreAnimations defines a boolean indicating if animations should not be executed (false by default)\n */\n render(updateCameras = true, ignoreAnimations = false) {\n var _a, _b, _c;\n if (this.isDisposed) {\n return;\n }\n if (this.onReadyObservable.hasObservers() && this._executeWhenReadyTimeoutId === null) {\n this._checkIsReady();\n }\n this._frameId++;\n this._defaultFrameBufferCleared = false;\n this._checkCameraRenderTarget(this.activeCamera);\n if ((_a = this.activeCameras) === null || _a === void 0 ? void 0 : _a.length) {\n this.activeCameras.forEach(this._checkCameraRenderTarget);\n }\n // Register components that have been associated lately to the scene.\n this._registerTransientComponents();\n this._activeParticles.fetchNewFrame();\n this._totalVertices.fetchNewFrame();\n this._activeIndices.fetchNewFrame();\n this._activeBones.fetchNewFrame();\n this._meshesForIntersections.reset();\n this.resetCachedMaterial();\n this.onBeforeAnimationsObservable.notifyObservers(this);\n // Actions\n if (this.actionManager) {\n this.actionManager.processTrigger(11);\n }\n // Animations\n if (!ignoreAnimations) {\n this.animate();\n }\n // Before camera update steps\n for (const step of this._beforeCameraUpdateStage) {\n step.action();\n }\n // Update Cameras\n if (updateCameras) {\n if (this.activeCameras && this.activeCameras.length > 0) {\n for (let cameraIndex = 0; cameraIndex < this.activeCameras.length; cameraIndex++) {\n const camera = this.activeCameras[cameraIndex];\n camera.update();\n if (camera.cameraRigMode !== 0) {\n // rig cameras\n for (let index = 0; index < camera._rigCameras.length; index++) {\n camera._rigCameras[index].update();\n }\n }\n }\n }\n else if (this.activeCamera) {\n this.activeCamera.update();\n if (this.activeCamera.cameraRigMode !== 0) {\n // rig cameras\n for (let index = 0; index < this.activeCamera._rigCameras.length; index++) {\n this.activeCamera._rigCameras[index].update();\n }\n }\n }\n }\n // Before render\n this.onBeforeRenderObservable.notifyObservers(this);\n const engine = this.getEngine();\n // Customs render targets\n this.onBeforeRenderTargetsRenderObservable.notifyObservers(this);\n const currentActiveCamera = ((_b = this.activeCameras) === null || _b === void 0 ? void 0 : _b.length) ? this.activeCameras[0] : this.activeCamera;\n if (this.renderTargetsEnabled) {\n Tools.StartPerformanceCounter(\"Custom render targets\", this.customRenderTargets.length > 0);\n this._intermediateRendering = true;\n for (let customIndex = 0; customIndex < this.customRenderTargets.length; customIndex++) {\n const renderTarget = this.customRenderTargets[customIndex];\n if (renderTarget._shouldRender()) {\n this._renderId++;\n this.activeCamera = renderTarget.activeCamera || this.activeCamera;\n if (!this.activeCamera) {\n throw new Error(\"Active camera not set\");\n }\n // Viewport\n engine.setViewport(this.activeCamera.viewport);\n // Camera\n this.updateTransformMatrix();\n renderTarget.render(currentActiveCamera !== this.activeCamera, this.dumpNextRenderTargets);\n }\n }\n Tools.EndPerformanceCounter(\"Custom render targets\", this.customRenderTargets.length > 0);\n this._intermediateRendering = false;\n this._renderId++;\n }\n this._engine.currentRenderPassId = (_c = currentActiveCamera === null || currentActiveCamera === void 0 ? void 0 : currentActiveCamera.renderPassId) !== null && _c !== void 0 ? _c : 0;\n // Restore back buffer\n this.activeCamera = currentActiveCamera;\n if (this._activeCamera && this._activeCamera.cameraRigMode !== 22 && !this.prePass) {\n this._bindFrameBuffer(this._activeCamera, false);\n }\n this.onAfterRenderTargetsRenderObservable.notifyObservers(this);\n for (const step of this._beforeClearStage) {\n step.action();\n }\n // Clear\n this._clearFrameBuffer(this.activeCamera);\n // Collects render targets from external components.\n for (const step of this._gatherRenderTargetsStage) {\n step.action(this._renderTargets);\n }\n // Multi-cameras?\n if (this.activeCameras && this.activeCameras.length > 0) {\n for (let cameraIndex = 0; cameraIndex < this.activeCameras.length; cameraIndex++) {\n this._processSubCameras(this.activeCameras[cameraIndex], cameraIndex > 0);\n }\n }\n else {\n if (!this.activeCamera) {\n throw new Error(\"No camera defined\");\n }\n this._processSubCameras(this.activeCamera, !!this.activeCamera.outputRenderTarget);\n }\n // Intersection checks\n this._checkIntersections();\n // Executes the after render stage actions.\n for (const step of this._afterRenderStage) {\n step.action();\n }\n // After render\n if (this.afterRender) {\n this.afterRender();\n }\n this.onAfterRenderObservable.notifyObservers(this);\n // Cleaning\n if (this._toBeDisposed.length) {\n for (let index = 0; index < this._toBeDisposed.length; index++) {\n const data = this._toBeDisposed[index];\n if (data) {\n data.dispose();\n }\n }\n this._toBeDisposed.length = 0;\n }\n if (this.dumpNextRenderTargets) {\n this.dumpNextRenderTargets = false;\n }\n this._activeBones.addCount(0, true);\n this._activeIndices.addCount(0, true);\n this._activeParticles.addCount(0, true);\n this._engine.restoreDefaultFramebuffer();\n }\n /**\n * Freeze all materials\n * A frozen material will not be updatable but should be faster to render\n * Note: multimaterials will not be frozen, but their submaterials will\n */\n freezeMaterials() {\n for (let i = 0; i < this.materials.length; i++) {\n this.materials[i].freeze();\n }\n }\n /**\n * Unfreeze all materials\n * A frozen material will not be updatable but should be faster to render\n */\n unfreezeMaterials() {\n for (let i = 0; i < this.materials.length; i++) {\n this.materials[i].unfreeze();\n }\n }\n /**\n * Releases all held resources\n */\n dispose() {\n if (this.isDisposed) {\n return;\n }\n this.beforeRender = null;\n this.afterRender = null;\n this.metadata = null;\n this.skeletons.length = 0;\n this.morphTargetManagers.length = 0;\n this._transientComponents.length = 0;\n this._isReadyForMeshStage.clear();\n this._beforeEvaluateActiveMeshStage.clear();\n this._evaluateSubMeshStage.clear();\n this._preActiveMeshStage.clear();\n this._cameraDrawRenderTargetStage.clear();\n this._beforeCameraDrawStage.clear();\n this._beforeRenderTargetDrawStage.clear();\n this._beforeRenderingGroupDrawStage.clear();\n this._beforeRenderingMeshStage.clear();\n this._afterRenderingMeshStage.clear();\n this._afterRenderingGroupDrawStage.clear();\n this._afterCameraDrawStage.clear();\n this._afterRenderTargetDrawStage.clear();\n this._afterRenderStage.clear();\n this._beforeCameraUpdateStage.clear();\n this._beforeClearStage.clear();\n this._gatherRenderTargetsStage.clear();\n this._gatherActiveCameraRenderTargetsStage.clear();\n this._pointerMoveStage.clear();\n this._pointerDownStage.clear();\n this._pointerUpStage.clear();\n this.importedMeshesFiles = new Array();\n if (this.stopAllAnimations) {\n // Ensures that no animatable notifies a callback that could start a new animation group, constantly adding new animatables to the active list...\n this._activeAnimatables.forEach((animatable) => {\n animatable.onAnimationEndObservable.clear();\n animatable.onAnimationEnd = null;\n });\n this.stopAllAnimations();\n }\n this.resetCachedMaterial();\n // Smart arrays\n if (this.activeCamera) {\n this.activeCamera._activeMeshes.dispose();\n this.activeCamera = null;\n }\n this.activeCameras = null;\n this._activeMeshes.dispose();\n this._renderingManager.dispose();\n this._processedMaterials.dispose();\n this._activeParticleSystems.dispose();\n this._activeSkeletons.dispose();\n this._softwareSkinnedMeshes.dispose();\n this._renderTargets.dispose();\n this._materialsRenderTargets.dispose();\n this._registeredForLateAnimationBindings.dispose();\n this._meshesForIntersections.dispose();\n this._toBeDisposed.length = 0;\n // Abort active requests\n const activeRequests = this._activeRequests.slice();\n for (const request of activeRequests) {\n request.abort();\n }\n this._activeRequests.length = 0;\n // Events\n try {\n this.onDisposeObservable.notifyObservers(this);\n }\n catch (e) {\n console.error(\"An error occurred while calling onDisposeObservable!\", e);\n }\n this.detachControl();\n // Detach cameras\n const canvas = this._engine.getInputElement();\n if (canvas) {\n for (let index = 0; index < this.cameras.length; index++) {\n this.cameras[index].detachControl();\n }\n }\n // Release animation groups\n this._disposeList(this.animationGroups);\n // Release lights\n this._disposeList(this.lights);\n // Release meshes\n this._disposeList(this.meshes, (item) => item.dispose(true));\n this._disposeList(this.transformNodes, (item) => item.dispose(true));\n // Release cameras\n const cameras = this.cameras;\n this._disposeList(cameras);\n // Release materials\n if (this._defaultMaterial) {\n this._defaultMaterial.dispose();\n }\n this._disposeList(this.multiMaterials);\n this._disposeList(this.materials);\n // Release particles\n this._disposeList(this.particleSystems);\n // Release postProcesses\n this._disposeList(this.postProcesses);\n // Release textures\n this._disposeList(this.textures);\n // Release morph targets\n this._disposeList(this.morphTargetManagers);\n // Release UBO\n this._sceneUbo.dispose();\n if (this._multiviewSceneUbo) {\n this._multiviewSceneUbo.dispose();\n }\n // Post-processes\n this.postProcessManager.dispose();\n // Components\n this._disposeList(this._components);\n // Remove from engine\n let index = this._engine.scenes.indexOf(this);\n if (index > -1) {\n this._engine.scenes.splice(index, 1);\n }\n if (EngineStore._LastCreatedScene === this) {\n if (this._engine.scenes.length > 0) {\n EngineStore._LastCreatedScene = this._engine.scenes[this._engine.scenes.length - 1];\n }\n else {\n EngineStore._LastCreatedScene = null;\n }\n }\n index = this._engine._virtualScenes.indexOf(this);\n if (index > -1) {\n this._engine._virtualScenes.splice(index, 1);\n }\n this._engine.wipeCaches(true);\n this.onDisposeObservable.clear();\n this.onBeforeRenderObservable.clear();\n this.onAfterRenderObservable.clear();\n this.onBeforeRenderTargetsRenderObservable.clear();\n this.onAfterRenderTargetsRenderObservable.clear();\n this.onAfterStepObservable.clear();\n this.onBeforeStepObservable.clear();\n this.onBeforeActiveMeshesEvaluationObservable.clear();\n this.onAfterActiveMeshesEvaluationObservable.clear();\n this.onBeforeParticlesRenderingObservable.clear();\n this.onAfterParticlesRenderingObservable.clear();\n this.onBeforeDrawPhaseObservable.clear();\n this.onAfterDrawPhaseObservable.clear();\n this.onBeforeAnimationsObservable.clear();\n this.onAfterAnimationsObservable.clear();\n this.onDataLoadedObservable.clear();\n this.onBeforeRenderingGroupObservable.clear();\n this.onAfterRenderingGroupObservable.clear();\n this.onMeshImportedObservable.clear();\n this.onBeforeCameraRenderObservable.clear();\n this.onAfterCameraRenderObservable.clear();\n this.onAfterRenderCameraObservable.clear();\n this.onReadyObservable.clear();\n this.onNewCameraAddedObservable.clear();\n this.onCameraRemovedObservable.clear();\n this.onNewLightAddedObservable.clear();\n this.onLightRemovedObservable.clear();\n this.onNewGeometryAddedObservable.clear();\n this.onGeometryRemovedObservable.clear();\n this.onNewTransformNodeAddedObservable.clear();\n this.onTransformNodeRemovedObservable.clear();\n this.onNewMeshAddedObservable.clear();\n this.onMeshRemovedObservable.clear();\n this.onNewSkeletonAddedObservable.clear();\n this.onSkeletonRemovedObservable.clear();\n this.onNewMaterialAddedObservable.clear();\n this.onNewMultiMaterialAddedObservable.clear();\n this.onMaterialRemovedObservable.clear();\n this.onMultiMaterialRemovedObservable.clear();\n this.onNewTextureAddedObservable.clear();\n this.onTextureRemovedObservable.clear();\n this.onPrePointerObservable.clear();\n this.onPointerObservable.clear();\n this.onPreKeyboardObservable.clear();\n this.onKeyboardObservable.clear();\n this.onActiveCameraChanged.clear();\n this.onScenePerformancePriorityChangedObservable.clear();\n this._isDisposed = true;\n }\n _disposeList(items, callback) {\n const itemsCopy = items.slice(0);\n callback = callback !== null && callback !== void 0 ? callback : ((item) => item.dispose());\n for (const item of itemsCopy) {\n callback(item);\n }\n items.length = 0;\n }\n /**\n * Gets if the scene is already disposed\n */\n get isDisposed() {\n return this._isDisposed;\n }\n /**\n * Call this function to reduce memory footprint of the scene.\n * Vertex buffers will not store CPU data anymore (this will prevent picking, collisions or physics to work correctly)\n */\n clearCachedVertexData() {\n for (let meshIndex = 0; meshIndex < this.meshes.length; meshIndex++) {\n const mesh = this.meshes[meshIndex];\n const geometry = mesh.geometry;\n if (geometry) {\n geometry.clearCachedData();\n }\n }\n }\n /**\n * This function will remove the local cached buffer data from texture.\n * It will save memory but will prevent the texture from being rebuilt\n */\n cleanCachedTextureBuffer() {\n for (const baseTexture of this.textures) {\n const buffer = baseTexture._buffer;\n if (buffer) {\n baseTexture._buffer = null;\n }\n }\n }\n /**\n * Get the world extend vectors with an optional filter\n *\n * @param filterPredicate the predicate - which meshes should be included when calculating the world size\n * @returns {{ min: Vector3; max: Vector3 }} min and max vectors\n */\n getWorldExtends(filterPredicate) {\n const min = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\n const max = new Vector3(-Number.MAX_VALUE, -Number.MAX_VALUE, -Number.MAX_VALUE);\n filterPredicate = filterPredicate || (() => true);\n this.meshes.filter(filterPredicate).forEach((mesh) => {\n mesh.computeWorldMatrix(true);\n if (!mesh.subMeshes || mesh.subMeshes.length === 0 || mesh.infiniteDistance) {\n return;\n }\n const boundingInfo = mesh.getBoundingInfo();\n const minBox = boundingInfo.boundingBox.minimumWorld;\n const maxBox = boundingInfo.boundingBox.maximumWorld;\n Vector3.CheckExtends(minBox, min, max);\n Vector3.CheckExtends(maxBox, min, max);\n });\n return {\n min: min,\n max: max,\n };\n }\n // Picking\n /**\n * Creates a ray that can be used to pick in the scene\n * @param x defines the x coordinate of the origin (on-screen)\n * @param y defines the y coordinate of the origin (on-screen)\n * @param world defines the world matrix to use if you want to pick in object space (instead of world space)\n * @param camera defines the camera to use for the picking\n * @param cameraViewSpace defines if picking will be done in view space (false by default)\n * @returns a Ray\n */\n createPickingRay(x, y, world, camera, cameraViewSpace = false) {\n throw _WarnImport(\"Ray\");\n }\n /**\n * Creates a ray that can be used to pick in the scene\n * @param x defines the x coordinate of the origin (on-screen)\n * @param y defines the y coordinate of the origin (on-screen)\n * @param world defines the world matrix to use if you want to pick in object space (instead of world space)\n * @param result defines the ray where to store the picking ray\n * @param camera defines the camera to use for the picking\n * @param cameraViewSpace defines if picking will be done in view space (false by default)\n * @param enableDistantPicking defines if picking should handle large values for mesh position/scaling (false by default)\n * @returns the current scene\n */\n createPickingRayToRef(x, y, world, result, camera, cameraViewSpace = false, enableDistantPicking = false) {\n throw _WarnImport(\"Ray\");\n }\n /**\n * Creates a ray that can be used to pick in the scene\n * @param x defines the x coordinate of the origin (on-screen)\n * @param y defines the y coordinate of the origin (on-screen)\n * @param camera defines the camera to use for the picking\n * @returns a Ray\n */\n createPickingRayInCameraSpace(x, y, camera) {\n throw _WarnImport(\"Ray\");\n }\n /**\n * Creates a ray that can be used to pick in the scene\n * @param x defines the x coordinate of the origin (on-screen)\n * @param y defines the y coordinate of the origin (on-screen)\n * @param result defines the ray where to store the picking ray\n * @param camera defines the camera to use for the picking\n * @returns the current scene\n */\n createPickingRayInCameraSpaceToRef(x, y, result, camera) {\n throw _WarnImport(\"Ray\");\n }\n /** @internal */\n get _pickingAvailable() {\n return false;\n }\n /** Launch a ray to try to pick a mesh in the scene\n * @param x position on screen\n * @param y position on screen\n * @param predicate Predicate function used to determine eligible meshes. Can be set to null. In this case, a mesh must be enabled, visible and with isPickable set to true\n * @param fastCheck defines if the first intersection will be used (and not the closest)\n * @param camera to use for computing the picking ray. Can be set to null. In this case, the scene.activeCamera will be used\n * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected\n * @returns a PickingInfo\n */\n pick(x, y, predicate, fastCheck, camera, trianglePredicate) {\n // Dummy info if picking as not been imported\n return new PickingInfo();\n }\n /** Launch a ray to try to pick a mesh in the scene using only bounding information of the main mesh (not using submeshes)\n * @param x position on screen\n * @param y position on screen\n * @param predicate Predicate function used to determine eligible meshes. Can be set to null. In this case, a mesh must be enabled, visible and with isPickable set to true\n * @param fastCheck defines if the first intersection will be used (and not the closest)\n * @param camera to use for computing the picking ray. Can be set to null. In this case, the scene.activeCamera will be used\n * @returns a PickingInfo (Please note that some info will not be set like distance, bv, bu and everything that cannot be capture by only using bounding infos)\n */\n pickWithBoundingInfo(x, y, predicate, fastCheck, camera) {\n // Dummy info if picking as not been imported\n return new PickingInfo();\n }\n /**\n * Use the given ray to pick a mesh in the scene. A mesh triangle can be picked both from its front and back sides,\n * irrespective of orientation.\n * @param ray The ray to use to pick meshes\n * @param predicate Predicate function used to determine eligible meshes. Can be set to null. In this case, a mesh must have isPickable set to true\n * @param fastCheck defines if the first intersection will be used (and not the closest)\n * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected\n * @returns a PickingInfo\n */\n pickWithRay(ray, predicate, fastCheck, trianglePredicate) {\n throw _WarnImport(\"Ray\");\n }\n /**\n * Launch a ray to try to pick a mesh in the scene. A mesh triangle can be picked both from its front and back sides,\n * irrespective of orientation.\n * @param x X position on screen\n * @param y Y position on screen\n * @param predicate Predicate function used to determine eligible meshes. Can be set to null. In this case, a mesh must be enabled, visible and with isPickable set to true\n * @param camera camera to use for computing the picking ray. Can be set to null. In this case, the scene.activeCamera will be used\n * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected\n * @returns an array of PickingInfo\n */\n multiPick(x, y, predicate, camera, trianglePredicate) {\n throw _WarnImport(\"Ray\");\n }\n /**\n * Launch a ray to try to pick a mesh in the scene\n * @param ray Ray to use\n * @param predicate Predicate function used to determine eligible meshes. Can be set to null. In this case, a mesh must be enabled, visible and with isPickable set to true\n * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected\n * @returns an array of PickingInfo\n */\n multiPickWithRay(ray, predicate, trianglePredicate) {\n throw _WarnImport(\"Ray\");\n }\n /**\n * Force the value of meshUnderPointer\n * @param mesh defines the mesh to use\n * @param pointerId optional pointer id when using more than one pointer\n * @param pickResult optional pickingInfo data used to find mesh\n */\n setPointerOverMesh(mesh, pointerId, pickResult) {\n this._inputManager.setPointerOverMesh(mesh, pointerId, pickResult);\n }\n /**\n * Gets the mesh under the pointer\n * @returns a Mesh or null if no mesh is under the pointer\n */\n getPointerOverMesh() {\n return this._inputManager.getPointerOverMesh();\n }\n // Misc.\n /** @internal */\n _rebuildGeometries() {\n for (const geometry of this.geometries) {\n geometry._rebuild();\n }\n for (const mesh of this.meshes) {\n mesh._rebuild();\n }\n if (this.postProcessManager) {\n this.postProcessManager._rebuild();\n }\n for (const component of this._components) {\n component.rebuild();\n }\n for (const system of this.particleSystems) {\n system.rebuild();\n }\n if (this.spriteManagers) {\n for (const spriteMgr of this.spriteManagers) {\n spriteMgr.rebuild();\n }\n }\n }\n /** @internal */\n _rebuildTextures() {\n for (const texture of this.textures) {\n texture._rebuild();\n }\n this.markAllMaterialsAsDirty(1);\n }\n // Tags\n _getByTags(list, tagsQuery, forEach) {\n if (tagsQuery === undefined) {\n // returns the complete list (could be done with Tags.MatchesQuery but no need to have a for-loop here)\n return list;\n }\n const listByTags = [];\n forEach =\n forEach ||\n ((item) => {\n return;\n });\n for (const i in list) {\n const item = list[i];\n if (Tags && Tags.MatchesQuery(item, tagsQuery)) {\n listByTags.push(item);\n forEach(item);\n }\n }\n return listByTags;\n }\n /**\n * Get a list of meshes by tags\n * @param tagsQuery defines the tags query to use\n * @param forEach defines a predicate used to filter results\n * @returns an array of Mesh\n */\n getMeshesByTags(tagsQuery, forEach) {\n return this._getByTags(this.meshes, tagsQuery, forEach);\n }\n /**\n * Get a list of cameras by tags\n * @param tagsQuery defines the tags query to use\n * @param forEach defines a predicate used to filter results\n * @returns an array of Camera\n */\n getCamerasByTags(tagsQuery, forEach) {\n return this._getByTags(this.cameras, tagsQuery, forEach);\n }\n /**\n * Get a list of lights by tags\n * @param tagsQuery defines the tags query to use\n * @param forEach defines a predicate used to filter results\n * @returns an array of Light\n */\n getLightsByTags(tagsQuery, forEach) {\n return this._getByTags(this.lights, tagsQuery, forEach);\n }\n /**\n * Get a list of materials by tags\n * @param tagsQuery defines the tags query to use\n * @param forEach defines a predicate used to filter results\n * @returns an array of Material\n */\n getMaterialByTags(tagsQuery, forEach) {\n return this._getByTags(this.materials, tagsQuery, forEach).concat(this._getByTags(this.multiMaterials, tagsQuery, forEach));\n }\n /**\n * Get a list of transform nodes by tags\n * @param tagsQuery defines the tags query to use\n * @param forEach defines a predicate used to filter results\n * @returns an array of TransformNode\n */\n getTransformNodesByTags(tagsQuery, forEach) {\n return this._getByTags(this.transformNodes, tagsQuery, forEach);\n }\n /**\n * Overrides the default sort function applied in the rendering group to prepare the meshes.\n * This allowed control for front to back rendering or reversly depending of the special needs.\n *\n * @param renderingGroupId The rendering group id corresponding to its index\n * @param opaqueSortCompareFn The opaque queue comparison function use to sort.\n * @param alphaTestSortCompareFn The alpha test queue comparison function use to sort.\n * @param transparentSortCompareFn The transparent queue comparison function use to sort.\n */\n setRenderingOrder(renderingGroupId, opaqueSortCompareFn = null, alphaTestSortCompareFn = null, transparentSortCompareFn = null) {\n this._renderingManager.setRenderingOrder(renderingGroupId, opaqueSortCompareFn, alphaTestSortCompareFn, transparentSortCompareFn);\n }\n /**\n * Specifies whether or not the stencil and depth buffer are cleared between two rendering groups.\n *\n * @param renderingGroupId The rendering group id corresponding to its index\n * @param autoClearDepthStencil Automatically clears depth and stencil between groups if true.\n * @param depth Automatically clears depth between groups if true and autoClear is true.\n * @param stencil Automatically clears stencil between groups if true and autoClear is true.\n */\n setRenderingAutoClearDepthStencil(renderingGroupId, autoClearDepthStencil, depth = true, stencil = true) {\n this._renderingManager.setRenderingAutoClearDepthStencil(renderingGroupId, autoClearDepthStencil, depth, stencil);\n }\n /**\n * Gets the current auto clear configuration for one rendering group of the rendering\n * manager.\n * @param index the rendering group index to get the information for\n * @returns The auto clear setup for the requested rendering group\n */\n getAutoClearDepthStencilSetup(index) {\n return this._renderingManager.getAutoClearDepthStencilSetup(index);\n }\n /** Gets or sets a boolean blocking all the calls to markAllMaterialsAsDirty (ie. the materials won't be updated if they are out of sync) */\n get blockMaterialDirtyMechanism() {\n return this._blockMaterialDirtyMechanism;\n }\n set blockMaterialDirtyMechanism(value) {\n if (this._blockMaterialDirtyMechanism === value) {\n return;\n }\n this._blockMaterialDirtyMechanism = value;\n if (!value) {\n // Do a complete update\n this.markAllMaterialsAsDirty(63);\n }\n }\n /**\n * Will flag all materials as dirty to trigger new shader compilation\n * @param flag defines the flag used to specify which material part must be marked as dirty\n * @param predicate If not null, it will be used to specify if a material has to be marked as dirty\n */\n markAllMaterialsAsDirty(flag, predicate) {\n if (this._blockMaterialDirtyMechanism) {\n return;\n }\n for (const material of this.materials) {\n if (predicate && !predicate(material)) {\n continue;\n }\n material.markAsDirty(flag);\n }\n }\n /**\n * @internal\n */\n _loadFile(fileOrUrl, onSuccess, onProgress, useOfflineSupport, useArrayBuffer, onError, onOpened) {\n const request = LoadFile(fileOrUrl, onSuccess, onProgress, useOfflineSupport ? this.offlineProvider : undefined, useArrayBuffer, onError, onOpened);\n this._activeRequests.push(request);\n request.onCompleteObservable.add((request) => {\n this._activeRequests.splice(this._activeRequests.indexOf(request), 1);\n });\n return request;\n }\n /**\n * @internal\n */\n _loadFileAsync(fileOrUrl, onProgress, useOfflineSupport, useArrayBuffer, onOpened) {\n return new Promise((resolve, reject) => {\n this._loadFile(fileOrUrl, (data) => {\n resolve(data);\n }, onProgress, useOfflineSupport, useArrayBuffer, (request, exception) => {\n reject(exception);\n }, onOpened);\n });\n }\n /**\n * @internal\n */\n _requestFile(url, onSuccess, onProgress, useOfflineSupport, useArrayBuffer, onError, onOpened) {\n const request = RequestFile(url, onSuccess, onProgress, useOfflineSupport ? this.offlineProvider : undefined, useArrayBuffer, onError, onOpened);\n this._activeRequests.push(request);\n request.onCompleteObservable.add((request) => {\n this._activeRequests.splice(this._activeRequests.indexOf(request), 1);\n });\n return request;\n }\n /**\n * @internal\n */\n _requestFileAsync(url, onProgress, useOfflineSupport, useArrayBuffer, onOpened) {\n return new Promise((resolve, reject) => {\n this._requestFile(url, (data) => {\n resolve(data);\n }, onProgress, useOfflineSupport, useArrayBuffer, (error) => {\n reject(error);\n }, onOpened);\n });\n }\n /**\n * @internal\n */\n _readFile(file, onSuccess, onProgress, useArrayBuffer, onError) {\n const request = ReadFile(file, onSuccess, onProgress, useArrayBuffer, onError);\n this._activeRequests.push(request);\n request.onCompleteObservable.add((request) => {\n this._activeRequests.splice(this._activeRequests.indexOf(request), 1);\n });\n return request;\n }\n /**\n * @internal\n */\n _readFileAsync(file, onProgress, useArrayBuffer) {\n return new Promise((resolve, reject) => {\n this._readFile(file, (data) => {\n resolve(data);\n }, onProgress, useArrayBuffer, (error) => {\n reject(error);\n });\n });\n }\n /**\n * This method gets the performance collector belonging to the scene, which is generally shared with the inspector.\n * @returns the perf collector belonging to the scene.\n */\n getPerfCollector() {\n throw _WarnImport(\"performanceViewerSceneExtension\");\n }\n // deprecated\n /**\n * Sets the active camera of the scene using its Id\n * @param id defines the camera's Id\n * @returns the new active camera or null if none found.\n * @deprecated Please use setActiveCameraById instead\n */\n setActiveCameraByID(id) {\n return this.setActiveCameraById(id);\n }\n /**\n * Get a material using its id\n * @param id defines the material's Id\n * @returns the material or null if none found.\n * @deprecated Please use getMaterialById instead\n */\n getMaterialByID(id) {\n return this.getMaterialById(id);\n }\n /**\n * Gets a the last added material using a given id\n * @param id defines the material's Id\n * @returns the last material with the given id or null if none found.\n * @deprecated Please use getLastMaterialById instead\n */\n getLastMaterialByID(id) {\n return this.getLastMaterialById(id);\n }\n /**\n * Get a texture using its unique id\n * @param uniqueId defines the texture's unique id\n * @returns the texture or null if none found.\n * @deprecated Please use getTextureByUniqueId instead\n */\n getTextureByUniqueID(uniqueId) {\n return this.getTextureByUniqueId(uniqueId);\n }\n /**\n * Gets a camera using its Id\n * @param id defines the Id to look for\n * @returns the camera or null if not found\n * @deprecated Please use getCameraById instead\n */\n getCameraByID(id) {\n return this.getCameraById(id);\n }\n /**\n * Gets a camera using its unique Id\n * @param uniqueId defines the unique Id to look for\n * @returns the camera or null if not found\n * @deprecated Please use getCameraByUniqueId instead\n */\n getCameraByUniqueID(uniqueId) {\n return this.getCameraByUniqueId(uniqueId);\n }\n /**\n * Gets a bone using its Id\n * @param id defines the bone's Id\n * @returns the bone or null if not found\n * @deprecated Please use getBoneById instead\n */\n getBoneByID(id) {\n return this.getBoneById(id);\n }\n /**\n * Gets a light node using its Id\n * @param id defines the light's Id\n * @returns the light or null if none found.\n * @deprecated Please use getLightById instead\n */\n getLightByID(id) {\n return this.getLightById(id);\n }\n /**\n * Gets a light node using its scene-generated unique Id\n * @param uniqueId defines the light's unique Id\n * @returns the light or null if none found.\n * @deprecated Please use getLightByUniqueId instead\n */\n getLightByUniqueID(uniqueId) {\n return this.getLightByUniqueId(uniqueId);\n }\n /**\n * Gets a particle system by Id\n * @param id defines the particle system Id\n * @returns the corresponding system or null if none found\n * @deprecated Please use getParticleSystemById instead\n */\n getParticleSystemByID(id) {\n return this.getParticleSystemById(id);\n }\n /**\n * Gets a geometry using its Id\n * @param id defines the geometry's Id\n * @returns the geometry or null if none found.\n * @deprecated Please use getGeometryById instead\n */\n getGeometryByID(id) {\n return this.getGeometryById(id);\n }\n /**\n * Gets the first added mesh found of a given Id\n * @param id defines the Id to search for\n * @returns the mesh found or null if not found at all\n * @deprecated Please use getMeshById instead\n */\n getMeshByID(id) {\n return this.getMeshById(id);\n }\n /**\n * Gets a mesh with its auto-generated unique Id\n * @param uniqueId defines the unique Id to search for\n * @returns the found mesh or null if not found at all.\n * @deprecated Please use getMeshByUniqueId instead\n */\n getMeshByUniqueID(uniqueId) {\n return this.getMeshByUniqueId(uniqueId);\n }\n /**\n * Gets a the last added mesh using a given Id\n * @param id defines the Id to search for\n * @returns the found mesh or null if not found at all.\n * @deprecated Please use getLastMeshById instead\n */\n getLastMeshByID(id) {\n return this.getLastMeshById(id);\n }\n /**\n * Gets a list of meshes using their Id\n * @param id defines the Id to search for\n * @returns a list of meshes\n * @deprecated Please use getMeshesById instead\n */\n getMeshesByID(id) {\n return this.getMeshesById(id);\n }\n /**\n * Gets the first added transform node found of a given Id\n * @param id defines the Id to search for\n * @returns the found transform node or null if not found at all.\n * @deprecated Please use getTransformNodeById instead\n */\n getTransformNodeByID(id) {\n return this.getTransformNodeById(id);\n }\n /**\n * Gets a transform node with its auto-generated unique Id\n * @param uniqueId defines the unique Id to search for\n * @returns the found transform node or null if not found at all.\n * @deprecated Please use getTransformNodeByUniqueId instead\n */\n getTransformNodeByUniqueID(uniqueId) {\n return this.getTransformNodeByUniqueId(uniqueId);\n }\n /**\n * Gets a list of transform nodes using their Id\n * @param id defines the Id to search for\n * @returns a list of transform nodes\n * @deprecated Please use getTransformNodesById instead\n */\n getTransformNodesByID(id) {\n return this.getTransformNodesById(id);\n }\n /**\n * Gets a node (Mesh, Camera, Light) using a given Id\n * @param id defines the Id to search for\n * @returns the found node or null if not found at all\n * @deprecated Please use getNodeById instead\n */\n getNodeByID(id) {\n return this.getNodeById(id);\n }\n /**\n * Gets a the last added node (Mesh, Camera, Light) using a given Id\n * @param id defines the Id to search for\n * @returns the found node or null if not found at all\n * @deprecated Please use getLastEntryById instead\n */\n getLastEntryByID(id) {\n return this.getLastEntryById(id);\n }\n /**\n * Gets a skeleton using a given Id (if many are found, this function will pick the last one)\n * @param id defines the Id to search for\n * @returns the found skeleton or null if not found at all.\n * @deprecated Please use getLastSkeletonById instead\n */\n getLastSkeletonByID(id) {\n return this.getLastSkeletonById(id);\n }\n }\n /** The fog is deactivated */\n Scene.FOGMODE_NONE = 0;\n /** The fog density is following an exponential function */\n Scene.FOGMODE_EXP = 1;\n /** The fog density is following an exponential function faster than FOGMODE_EXP */\n Scene.FOGMODE_EXP2 = 2;\n /** The fog density is following a linear function. */\n Scene.FOGMODE_LINEAR = 3;\n /**\n * Gets or sets the minimum deltatime when deterministic lock step is enabled\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep\n */\n Scene.MinDeltaTime = 1.0;\n /**\n * Gets or sets the maximum deltatime when deterministic lock step is enabled\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep\n */\n Scene.MaxDeltaTime = 1000.0;\n\n /**\n * @internal\n */\n class IntersectionInfo {\n constructor(bu, bv, distance) {\n this.bu = bu;\n this.bv = bv;\n this.distance = distance;\n this.faceId = 0;\n this.subMeshId = 0;\n }\n }\n\n /**\n * Class used to store bounding box information\n */\n class BoundingBox {\n /**\n * Creates a new bounding box\n * @param min defines the minimum vector (in local space)\n * @param max defines the maximum vector (in local space)\n * @param worldMatrix defines the new world matrix\n */\n constructor(min, max, worldMatrix) {\n /**\n * Gets the 8 vectors representing the bounding box in local space\n */\n this.vectors = ArrayTools.BuildArray(8, Vector3.Zero);\n /**\n * Gets the center of the bounding box in local space\n */\n this.center = Vector3.Zero();\n /**\n * Gets the center of the bounding box in world space\n */\n this.centerWorld = Vector3.Zero();\n /**\n * Gets the extend size in local space\n */\n this.extendSize = Vector3.Zero();\n /**\n * Gets the extend size in world space\n */\n this.extendSizeWorld = Vector3.Zero();\n /**\n * Gets the OBB (object bounding box) directions\n */\n this.directions = ArrayTools.BuildArray(3, Vector3.Zero);\n /**\n * Gets the 8 vectors representing the bounding box in world space\n */\n this.vectorsWorld = ArrayTools.BuildArray(8, Vector3.Zero);\n /**\n * Gets the minimum vector in world space\n */\n this.minimumWorld = Vector3.Zero();\n /**\n * Gets the maximum vector in world space\n */\n this.maximumWorld = Vector3.Zero();\n /**\n * Gets the minimum vector in local space\n */\n this.minimum = Vector3.Zero();\n /**\n * Gets the maximum vector in local space\n */\n this.maximum = Vector3.Zero();\n /** @internal */\n this._drawWrapperFront = null;\n /** @internal */\n this._drawWrapperBack = null;\n this.reConstruct(min, max, worldMatrix);\n }\n // Methods\n /**\n * Recreates the entire bounding box from scratch as if we call the constructor in place\n * @param min defines the new minimum vector (in local space)\n * @param max defines the new maximum vector (in local space)\n * @param worldMatrix defines the new world matrix\n */\n reConstruct(min, max, worldMatrix) {\n const minX = min.x, minY = min.y, minZ = min.z, maxX = max.x, maxY = max.y, maxZ = max.z;\n const vectors = this.vectors;\n this.minimum.copyFromFloats(minX, minY, minZ);\n this.maximum.copyFromFloats(maxX, maxY, maxZ);\n vectors[0].copyFromFloats(minX, minY, minZ);\n vectors[1].copyFromFloats(maxX, maxY, maxZ);\n vectors[2].copyFromFloats(maxX, minY, minZ);\n vectors[3].copyFromFloats(minX, maxY, minZ);\n vectors[4].copyFromFloats(minX, minY, maxZ);\n vectors[5].copyFromFloats(maxX, maxY, minZ);\n vectors[6].copyFromFloats(minX, maxY, maxZ);\n vectors[7].copyFromFloats(maxX, minY, maxZ);\n // OBB\n max.addToRef(min, this.center).scaleInPlace(0.5);\n max.subtractToRef(min, this.extendSize).scaleInPlace(0.5);\n this._worldMatrix = worldMatrix || Matrix.IdentityReadOnly;\n this._update(this._worldMatrix);\n }\n /**\n * Scale the current bounding box by applying a scale factor\n * @param factor defines the scale factor to apply\n * @returns the current bounding box\n */\n scale(factor) {\n const tmpVectors = BoundingBox._TmpVector3;\n const diff = this.maximum.subtractToRef(this.minimum, tmpVectors[0]);\n const len = diff.length();\n diff.normalizeFromLength(len);\n const distance = len * factor;\n const newRadius = diff.scaleInPlace(distance * 0.5);\n const min = this.center.subtractToRef(newRadius, tmpVectors[1]);\n const max = this.center.addToRef(newRadius, tmpVectors[2]);\n this.reConstruct(min, max, this._worldMatrix);\n return this;\n }\n /**\n * Gets the world matrix of the bounding box\n * @returns a matrix\n */\n getWorldMatrix() {\n return this._worldMatrix;\n }\n /**\n * @internal\n */\n _update(world) {\n const minWorld = this.minimumWorld;\n const maxWorld = this.maximumWorld;\n const directions = this.directions;\n const vectorsWorld = this.vectorsWorld;\n const vectors = this.vectors;\n if (!world.isIdentity()) {\n minWorld.setAll(Number.MAX_VALUE);\n maxWorld.setAll(-Number.MAX_VALUE);\n for (let index = 0; index < 8; ++index) {\n const v = vectorsWorld[index];\n Vector3.TransformCoordinatesToRef(vectors[index], world, v);\n minWorld.minimizeInPlace(v);\n maxWorld.maximizeInPlace(v);\n }\n // Extend\n maxWorld.subtractToRef(minWorld, this.extendSizeWorld).scaleInPlace(0.5);\n maxWorld.addToRef(minWorld, this.centerWorld).scaleInPlace(0.5);\n }\n else {\n minWorld.copyFrom(this.minimum);\n maxWorld.copyFrom(this.maximum);\n for (let index = 0; index < 8; ++index) {\n vectorsWorld[index].copyFrom(vectors[index]);\n }\n // Extend\n this.extendSizeWorld.copyFrom(this.extendSize);\n this.centerWorld.copyFrom(this.center);\n }\n Vector3.FromArrayToRef(world.m, 0, directions[0]);\n Vector3.FromArrayToRef(world.m, 4, directions[1]);\n Vector3.FromArrayToRef(world.m, 8, directions[2]);\n this._worldMatrix = world;\n }\n /**\n * Tests if the bounding box is intersecting the frustum planes\n * @param frustumPlanes defines the frustum planes to test\n * @returns true if there is an intersection\n */\n isInFrustum(frustumPlanes) {\n return BoundingBox.IsInFrustum(this.vectorsWorld, frustumPlanes);\n }\n /**\n * Tests if the bounding box is entirely inside the frustum planes\n * @param frustumPlanes defines the frustum planes to test\n * @returns true if there is an inclusion\n */\n isCompletelyInFrustum(frustumPlanes) {\n return BoundingBox.IsCompletelyInFrustum(this.vectorsWorld, frustumPlanes);\n }\n /**\n * Tests if a point is inside the bounding box\n * @param point defines the point to test\n * @returns true if the point is inside the bounding box\n */\n intersectsPoint(point) {\n const min = this.minimumWorld;\n const max = this.maximumWorld;\n const minX = min.x, minY = min.y, minZ = min.z, maxX = max.x, maxY = max.y, maxZ = max.z;\n const pointX = point.x, pointY = point.y, pointZ = point.z;\n const delta = -Epsilon;\n if (maxX - pointX < delta || delta > pointX - minX) {\n return false;\n }\n if (maxY - pointY < delta || delta > pointY - minY) {\n return false;\n }\n if (maxZ - pointZ < delta || delta > pointZ - minZ) {\n return false;\n }\n return true;\n }\n /**\n * Tests if the bounding box intersects with a bounding sphere\n * @param sphere defines the sphere to test\n * @returns true if there is an intersection\n */\n intersectsSphere(sphere) {\n return BoundingBox.IntersectsSphere(this.minimumWorld, this.maximumWorld, sphere.centerWorld, sphere.radiusWorld);\n }\n /**\n * Tests if the bounding box intersects with a box defined by a min and max vectors\n * @param min defines the min vector to use\n * @param max defines the max vector to use\n * @returns true if there is an intersection\n */\n intersectsMinMax(min, max) {\n const myMin = this.minimumWorld;\n const myMax = this.maximumWorld;\n const myMinX = myMin.x, myMinY = myMin.y, myMinZ = myMin.z, myMaxX = myMax.x, myMaxY = myMax.y, myMaxZ = myMax.z;\n const minX = min.x, minY = min.y, minZ = min.z, maxX = max.x, maxY = max.y, maxZ = max.z;\n if (myMaxX < minX || myMinX > maxX) {\n return false;\n }\n if (myMaxY < minY || myMinY > maxY) {\n return false;\n }\n if (myMaxZ < minZ || myMinZ > maxZ) {\n return false;\n }\n return true;\n }\n /**\n * Disposes the resources of the class\n */\n dispose() {\n var _a, _b;\n (_a = this._drawWrapperFront) === null || _a === void 0 ? void 0 : _a.dispose();\n (_b = this._drawWrapperBack) === null || _b === void 0 ? void 0 : _b.dispose();\n }\n // Statics\n /**\n * Tests if two bounding boxes are intersections\n * @param box0 defines the first box to test\n * @param box1 defines the second box to test\n * @returns true if there is an intersection\n */\n static Intersects(box0, box1) {\n return box0.intersectsMinMax(box1.minimumWorld, box1.maximumWorld);\n }\n /**\n * Tests if a bounding box defines by a min/max vectors intersects a sphere\n * @param minPoint defines the minimum vector of the bounding box\n * @param maxPoint defines the maximum vector of the bounding box\n * @param sphereCenter defines the sphere center\n * @param sphereRadius defines the sphere radius\n * @returns true if there is an intersection\n */\n static IntersectsSphere(minPoint, maxPoint, sphereCenter, sphereRadius) {\n const vector = BoundingBox._TmpVector3[0];\n Vector3.ClampToRef(sphereCenter, minPoint, maxPoint, vector);\n const num = Vector3.DistanceSquared(sphereCenter, vector);\n return num <= sphereRadius * sphereRadius;\n }\n /**\n * Tests if a bounding box defined with 8 vectors is entirely inside frustum planes\n * @param boundingVectors defines an array of 8 vectors representing a bounding box\n * @param frustumPlanes defines the frustum planes to test\n * @returns true if there is an inclusion\n */\n static IsCompletelyInFrustum(boundingVectors, frustumPlanes) {\n for (let p = 0; p < 6; ++p) {\n const frustumPlane = frustumPlanes[p];\n for (let i = 0; i < 8; ++i) {\n if (frustumPlane.dotCoordinate(boundingVectors[i]) < 0) {\n return false;\n }\n }\n }\n return true;\n }\n /**\n * Tests if a bounding box defined with 8 vectors intersects frustum planes\n * @param boundingVectors defines an array of 8 vectors representing a bounding box\n * @param frustumPlanes defines the frustum planes to test\n * @returns true if there is an intersection\n */\n static IsInFrustum(boundingVectors, frustumPlanes) {\n for (let p = 0; p < 6; ++p) {\n let canReturnFalse = true;\n const frustumPlane = frustumPlanes[p];\n for (let i = 0; i < 8; ++i) {\n if (frustumPlane.dotCoordinate(boundingVectors[i]) >= 0) {\n canReturnFalse = false;\n break;\n }\n }\n if (canReturnFalse) {\n return false;\n }\n }\n return true;\n }\n }\n BoundingBox._TmpVector3 = ArrayTools.BuildArray(3, Vector3.Zero);\n\n /**\n * Class used to store bounding sphere information\n */\n class BoundingSphere {\n /**\n * Creates a new bounding sphere\n * @param min defines the minimum vector (in local space)\n * @param max defines the maximum vector (in local space)\n * @param worldMatrix defines the new world matrix\n */\n constructor(min, max, worldMatrix) {\n /**\n * Gets the center of the bounding sphere in local space\n */\n this.center = Vector3.Zero();\n /**\n * Gets the center of the bounding sphere in world space\n */\n this.centerWorld = Vector3.Zero();\n /**\n * Gets the minimum vector in local space\n */\n this.minimum = Vector3.Zero();\n /**\n * Gets the maximum vector in local space\n */\n this.maximum = Vector3.Zero();\n this.reConstruct(min, max, worldMatrix);\n }\n /**\n * Recreates the entire bounding sphere from scratch as if we call the constructor in place\n * @param min defines the new minimum vector (in local space)\n * @param max defines the new maximum vector (in local space)\n * @param worldMatrix defines the new world matrix\n */\n reConstruct(min, max, worldMatrix) {\n this.minimum.copyFrom(min);\n this.maximum.copyFrom(max);\n const distance = Vector3.Distance(min, max);\n max.addToRef(min, this.center).scaleInPlace(0.5);\n this.radius = distance * 0.5;\n this._update(worldMatrix || Matrix.IdentityReadOnly);\n }\n /**\n * Scale the current bounding sphere by applying a scale factor\n * @param factor defines the scale factor to apply\n * @returns the current bounding box\n */\n scale(factor) {\n const newRadius = this.radius * factor;\n const tmpVectors = BoundingSphere._TmpVector3;\n const tempRadiusVector = tmpVectors[0].setAll(newRadius);\n const min = this.center.subtractToRef(tempRadiusVector, tmpVectors[1]);\n const max = this.center.addToRef(tempRadiusVector, tmpVectors[2]);\n this.reConstruct(min, max, this._worldMatrix);\n return this;\n }\n /**\n * Gets the world matrix of the bounding box\n * @returns a matrix\n */\n getWorldMatrix() {\n return this._worldMatrix;\n }\n // Methods\n /**\n * @internal\n */\n _update(worldMatrix) {\n if (!worldMatrix.isIdentity()) {\n Vector3.TransformCoordinatesToRef(this.center, worldMatrix, this.centerWorld);\n const tempVector = BoundingSphere._TmpVector3[0];\n Vector3.TransformNormalFromFloatsToRef(1.0, 1.0, 1.0, worldMatrix, tempVector);\n this.radiusWorld = Math.max(Math.abs(tempVector.x), Math.abs(tempVector.y), Math.abs(tempVector.z)) * this.radius;\n }\n else {\n this.centerWorld.copyFrom(this.center);\n this.radiusWorld = this.radius;\n }\n }\n /**\n * Tests if the bounding sphere is intersecting the frustum planes\n * @param frustumPlanes defines the frustum planes to test\n * @returns true if there is an intersection\n */\n isInFrustum(frustumPlanes) {\n const center = this.centerWorld;\n const radius = this.radiusWorld;\n for (let i = 0; i < 6; i++) {\n if (frustumPlanes[i].dotCoordinate(center) <= -radius) {\n return false;\n }\n }\n return true;\n }\n /**\n * Tests if the bounding sphere center is in between the frustum planes.\n * Used for optimistic fast inclusion.\n * @param frustumPlanes defines the frustum planes to test\n * @returns true if the sphere center is in between the frustum planes\n */\n isCenterInFrustum(frustumPlanes) {\n const center = this.centerWorld;\n for (let i = 0; i < 6; i++) {\n if (frustumPlanes[i].dotCoordinate(center) < 0) {\n return false;\n }\n }\n return true;\n }\n /**\n * Tests if a point is inside the bounding sphere\n * @param point defines the point to test\n * @returns true if the point is inside the bounding sphere\n */\n intersectsPoint(point) {\n const squareDistance = Vector3.DistanceSquared(this.centerWorld, point);\n if (this.radiusWorld * this.radiusWorld < squareDistance) {\n return false;\n }\n return true;\n }\n // Statics\n /**\n * Checks if two sphere intersect\n * @param sphere0 sphere 0\n * @param sphere1 sphere 1\n * @returns true if the spheres intersect\n */\n static Intersects(sphere0, sphere1) {\n const squareDistance = Vector3.DistanceSquared(sphere0.centerWorld, sphere1.centerWorld);\n const radiusSum = sphere0.radiusWorld + sphere1.radiusWorld;\n if (radiusSum * radiusSum < squareDistance) {\n return false;\n }\n return true;\n }\n /**\n * Creates a sphere from a center and a radius\n * @param center The center\n * @param radius radius\n * @param matrix Optional worldMatrix\n * @returns The sphere\n */\n static CreateFromCenterAndRadius(center, radius, matrix) {\n this._TmpVector3[0].copyFrom(center);\n this._TmpVector3[1].copyFromFloats(0, 0, radius);\n this._TmpVector3[2].copyFrom(center);\n this._TmpVector3[0].addInPlace(this._TmpVector3[1]);\n this._TmpVector3[2].subtractInPlace(this._TmpVector3[1]);\n const sphere = new BoundingSphere(this._TmpVector3[0], this._TmpVector3[2]);\n if (matrix) {\n sphere._worldMatrix = matrix;\n }\n else {\n sphere._worldMatrix = Matrix.Identity();\n }\n return sphere;\n }\n }\n BoundingSphere._TmpVector3 = ArrayTools.BuildArray(3, Vector3.Zero);\n\n const _result0 = { min: 0, max: 0 };\n const _result1 = { min: 0, max: 0 };\n const computeBoxExtents = (axis, box, result) => {\n const p = Vector3.Dot(box.centerWorld, axis);\n const r0 = Math.abs(Vector3.Dot(box.directions[0], axis)) * box.extendSize.x;\n const r1 = Math.abs(Vector3.Dot(box.directions[1], axis)) * box.extendSize.y;\n const r2 = Math.abs(Vector3.Dot(box.directions[2], axis)) * box.extendSize.z;\n const r = r0 + r1 + r2;\n result.min = p - r;\n result.max = p + r;\n };\n const axisOverlap = (axis, box0, box1) => {\n computeBoxExtents(axis, box0, _result0);\n computeBoxExtents(axis, box1, _result1);\n return !(_result0.min > _result1.max || _result1.min > _result0.max);\n };\n /**\n * Info for a bounding data of a mesh\n */\n class BoundingInfo {\n /**\n * Constructs bounding info\n * @param minimum min vector of the bounding box/sphere\n * @param maximum max vector of the bounding box/sphere\n * @param worldMatrix defines the new world matrix\n */\n constructor(minimum, maximum, worldMatrix) {\n this._isLocked = false;\n this.boundingBox = new BoundingBox(minimum, maximum, worldMatrix);\n this.boundingSphere = new BoundingSphere(minimum, maximum, worldMatrix);\n }\n /**\n * Recreates the entire bounding info from scratch as if we call the constructor in place\n * @param min defines the new minimum vector (in local space)\n * @param max defines the new maximum vector (in local space)\n * @param worldMatrix defines the new world matrix\n */\n reConstruct(min, max, worldMatrix) {\n this.boundingBox.reConstruct(min, max, worldMatrix);\n this.boundingSphere.reConstruct(min, max, worldMatrix);\n }\n /**\n * min vector of the bounding box/sphere\n */\n get minimum() {\n return this.boundingBox.minimum;\n }\n /**\n * max vector of the bounding box/sphere\n */\n get maximum() {\n return this.boundingBox.maximum;\n }\n /**\n * If the info is locked and won't be updated to avoid perf overhead\n */\n get isLocked() {\n return this._isLocked;\n }\n set isLocked(value) {\n this._isLocked = value;\n }\n // Methods\n /**\n * Updates the bounding sphere and box\n * @param world world matrix to be used to update\n */\n update(world) {\n if (this._isLocked) {\n return;\n }\n this.boundingBox._update(world);\n this.boundingSphere._update(world);\n }\n /**\n * Recreate the bounding info to be centered around a specific point given a specific extend.\n * @param center New center of the bounding info\n * @param extend New extend of the bounding info\n * @returns the current bounding info\n */\n centerOn(center, extend) {\n const minimum = BoundingInfo._TmpVector3[0].copyFrom(center).subtractInPlace(extend);\n const maximum = BoundingInfo._TmpVector3[1].copyFrom(center).addInPlace(extend);\n this.boundingBox.reConstruct(minimum, maximum, this.boundingBox.getWorldMatrix());\n this.boundingSphere.reConstruct(minimum, maximum, this.boundingBox.getWorldMatrix());\n return this;\n }\n /**\n * Grows the bounding info to include the given point.\n * @param point The point that will be included in the current bounding info (in local space)\n * @returns the current bounding info\n */\n encapsulate(point) {\n const minimum = Vector3.Minimize(this.minimum, point);\n const maximum = Vector3.Maximize(this.maximum, point);\n this.reConstruct(minimum, maximum, this.boundingBox.getWorldMatrix());\n return this;\n }\n /**\n * Grows the bounding info to encapsulate the given bounding info.\n * @param toEncapsulate The bounding info that will be encapsulated in the current bounding info\n * @returns the current bounding info\n */\n encapsulateBoundingInfo(toEncapsulate) {\n const invw = TmpVectors.Matrix[0];\n this.boundingBox.getWorldMatrix().invertToRef(invw);\n const v = TmpVectors.Vector3[0];\n Vector3.TransformCoordinatesToRef(toEncapsulate.boundingBox.minimumWorld, invw, v);\n this.encapsulate(v);\n Vector3.TransformCoordinatesToRef(toEncapsulate.boundingBox.maximumWorld, invw, v);\n this.encapsulate(v);\n return this;\n }\n /**\n * Scale the current bounding info by applying a scale factor\n * @param factor defines the scale factor to apply\n * @returns the current bounding info\n */\n scale(factor) {\n this.boundingBox.scale(factor);\n this.boundingSphere.scale(factor);\n return this;\n }\n /**\n * Returns `true` if the bounding info is within the frustum defined by the passed array of planes.\n * @param frustumPlanes defines the frustum to test\n * @param strategy defines the strategy to use for the culling (default is BABYLON.AbstractMesh.CULLINGSTRATEGY_STANDARD)\n * The different strategies available are:\n * * BABYLON.AbstractMesh.CULLINGSTRATEGY_STANDARD most accurate but slower @see https://doc.babylonjs.com/typedoc/classes/BABYLON.AbstractMesh#CULLINGSTRATEGY_STANDARD\n * * BABYLON.AbstractMesh.CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY faster but less accurate @see https://doc.babylonjs.com/typedoc/classes/BABYLON.AbstractMesh#CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY\n * * BABYLON.AbstractMesh.CULLINGSTRATEGY_OPTIMISTIC_INCLUSION can be faster if always visible @see https://doc.babylonjs.com/typedoc/classes/BABYLON.AbstractMesh#CULLINGSTRATEGY_OPTIMISTIC_INCLUSION\n * * BABYLON.AbstractMesh.CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY can be faster if always visible @see https://doc.babylonjs.com/typedoc/classes/BABYLON.AbstractMesh#CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY\n * @returns true if the bounding info is in the frustum planes\n */\n isInFrustum(frustumPlanes, strategy = 0) {\n const inclusionTest = strategy === 2 || strategy === 3;\n if (inclusionTest) {\n if (this.boundingSphere.isCenterInFrustum(frustumPlanes)) {\n return true;\n }\n }\n if (!this.boundingSphere.isInFrustum(frustumPlanes)) {\n return false;\n }\n const bSphereOnlyTest = strategy === 1 || strategy === 3;\n if (bSphereOnlyTest) {\n return true;\n }\n return this.boundingBox.isInFrustum(frustumPlanes);\n }\n /**\n * Gets the world distance between the min and max points of the bounding box\n */\n get diagonalLength() {\n const boundingBox = this.boundingBox;\n const diag = boundingBox.maximumWorld.subtractToRef(boundingBox.minimumWorld, BoundingInfo._TmpVector3[0]);\n return diag.length();\n }\n /**\n * Checks if a cullable object (mesh...) is in the camera frustum\n * Unlike isInFrustum this checks the full bounding box\n * @param frustumPlanes Camera near/planes\n * @returns true if the object is in frustum otherwise false\n */\n isCompletelyInFrustum(frustumPlanes) {\n return this.boundingBox.isCompletelyInFrustum(frustumPlanes);\n }\n /**\n * @internal\n */\n _checkCollision(collider) {\n return collider._canDoCollision(this.boundingSphere.centerWorld, this.boundingSphere.radiusWorld, this.boundingBox.minimumWorld, this.boundingBox.maximumWorld);\n }\n /**\n * Checks if a point is inside the bounding box and bounding sphere or the mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/interactions/mesh_intersect\n * @param point the point to check intersection with\n * @returns if the point intersects\n */\n intersectsPoint(point) {\n if (!this.boundingSphere.centerWorld) {\n return false;\n }\n if (!this.boundingSphere.intersectsPoint(point)) {\n return false;\n }\n if (!this.boundingBox.intersectsPoint(point)) {\n return false;\n }\n return true;\n }\n /**\n * Checks if another bounding info intersects the bounding box and bounding sphere or the mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/interactions/mesh_intersect\n * @param boundingInfo the bounding info to check intersection with\n * @param precise if the intersection should be done using OBB\n * @returns if the bounding info intersects\n */\n intersects(boundingInfo, precise) {\n if (!BoundingSphere.Intersects(this.boundingSphere, boundingInfo.boundingSphere)) {\n return false;\n }\n if (!BoundingBox.Intersects(this.boundingBox, boundingInfo.boundingBox)) {\n return false;\n }\n if (!precise) {\n return true;\n }\n const box0 = this.boundingBox;\n const box1 = boundingInfo.boundingBox;\n if (!axisOverlap(box0.directions[0], box0, box1)) {\n return false;\n }\n if (!axisOverlap(box0.directions[1], box0, box1)) {\n return false;\n }\n if (!axisOverlap(box0.directions[2], box0, box1)) {\n return false;\n }\n if (!axisOverlap(box1.directions[0], box0, box1)) {\n return false;\n }\n if (!axisOverlap(box1.directions[1], box0, box1)) {\n return false;\n }\n if (!axisOverlap(box1.directions[2], box0, box1)) {\n return false;\n }\n if (!axisOverlap(Vector3.Cross(box0.directions[0], box1.directions[0]), box0, box1)) {\n return false;\n }\n if (!axisOverlap(Vector3.Cross(box0.directions[0], box1.directions[1]), box0, box1)) {\n return false;\n }\n if (!axisOverlap(Vector3.Cross(box0.directions[0], box1.directions[2]), box0, box1)) {\n return false;\n }\n if (!axisOverlap(Vector3.Cross(box0.directions[1], box1.directions[0]), box0, box1)) {\n return false;\n }\n if (!axisOverlap(Vector3.Cross(box0.directions[1], box1.directions[1]), box0, box1)) {\n return false;\n }\n if (!axisOverlap(Vector3.Cross(box0.directions[1], box1.directions[2]), box0, box1)) {\n return false;\n }\n if (!axisOverlap(Vector3.Cross(box0.directions[2], box1.directions[0]), box0, box1)) {\n return false;\n }\n if (!axisOverlap(Vector3.Cross(box0.directions[2], box1.directions[1]), box0, box1)) {\n return false;\n }\n if (!axisOverlap(Vector3.Cross(box0.directions[2], box1.directions[2]), box0, box1)) {\n return false;\n }\n return true;\n }\n }\n BoundingInfo._TmpVector3 = ArrayTools.BuildArray(2, Vector3.Zero);\n\n // This helper class is only here so we can apply the nativeOverride decorator to functions.\n class MathHelpers {\n static extractMinAndMaxIndexed(positions, indices, indexStart, indexCount, minimum, maximum) {\n for (let index = indexStart; index < indexStart + indexCount; index++) {\n const offset = indices[index] * 3;\n const x = positions[offset];\n const y = positions[offset + 1];\n const z = positions[offset + 2];\n minimum.minimizeInPlaceFromFloats(x, y, z);\n maximum.maximizeInPlaceFromFloats(x, y, z);\n }\n }\n static extractMinAndMax(positions, start, count, stride, minimum, maximum) {\n for (let index = start, offset = start * stride; index < start + count; index++, offset += stride) {\n const x = positions[offset];\n const y = positions[offset + 1];\n const z = positions[offset + 2];\n minimum.minimizeInPlaceFromFloats(x, y, z);\n maximum.maximizeInPlaceFromFloats(x, y, z);\n }\n }\n }\n __decorate$1([\n nativeOverride.filter((...[positions, indices]) => !Array.isArray(positions) && !Array.isArray(indices))\n // eslint-disable-next-line @typescript-eslint/naming-convention\n ], MathHelpers, \"extractMinAndMaxIndexed\", null);\n __decorate$1([\n nativeOverride.filter((...[positions]) => !Array.isArray(positions))\n // eslint-disable-next-line @typescript-eslint/naming-convention\n ], MathHelpers, \"extractMinAndMax\", null);\n /**\n * Extracts minimum and maximum values from a list of indexed positions\n * @param positions defines the positions to use\n * @param indices defines the indices to the positions\n * @param indexStart defines the start index\n * @param indexCount defines the end index\n * @param bias defines bias value to add to the result\n * @returns minimum and maximum values\n */\n function extractMinAndMaxIndexed(positions, indices, indexStart, indexCount, bias = null) {\n const minimum = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\n const maximum = new Vector3(-Number.MAX_VALUE, -Number.MAX_VALUE, -Number.MAX_VALUE);\n MathHelpers.extractMinAndMaxIndexed(positions, indices, indexStart, indexCount, minimum, maximum);\n if (bias) {\n minimum.x -= minimum.x * bias.x + bias.y;\n minimum.y -= minimum.y * bias.x + bias.y;\n minimum.z -= minimum.z * bias.x + bias.y;\n maximum.x += maximum.x * bias.x + bias.y;\n maximum.y += maximum.y * bias.x + bias.y;\n maximum.z += maximum.z * bias.x + bias.y;\n }\n return {\n minimum: minimum,\n maximum: maximum,\n };\n }\n /**\n * Extracts minimum and maximum values from a list of positions\n * @param positions defines the positions to use\n * @param start defines the start index in the positions array\n * @param count defines the number of positions to handle\n * @param bias defines bias value to add to the result\n * @param stride defines the stride size to use (distance between two positions in the positions array)\n * @returns minimum and maximum values\n */\n function extractMinAndMax(positions, start, count, bias = null, stride) {\n const minimum = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\n const maximum = new Vector3(-Number.MAX_VALUE, -Number.MAX_VALUE, -Number.MAX_VALUE);\n if (!stride) {\n stride = 3;\n }\n MathHelpers.extractMinAndMax(positions, start, count, stride, minimum, maximum);\n if (bias) {\n minimum.x -= minimum.x * bias.x + bias.y;\n minimum.y -= minimum.y * bias.x + bias.y;\n minimum.z -= minimum.z * bias.x + bias.y;\n maximum.x += maximum.x * bias.x + bias.y;\n maximum.y += maximum.y * bias.x + bias.y;\n maximum.z += maximum.z * bias.x + bias.y;\n }\n return {\n minimum: minimum,\n maximum: maximum,\n };\n }\n\n /**\n * Defines a subdivision inside a mesh\n */\n class SubMesh {\n /**\n * Gets material defines used by the effect associated to the sub mesh\n */\n get materialDefines() {\n var _a;\n return this._mainDrawWrapperOverride ? this._mainDrawWrapperOverride.defines : (_a = this._getDrawWrapper()) === null || _a === void 0 ? void 0 : _a.defines;\n }\n /**\n * Sets material defines used by the effect associated to the sub mesh\n */\n set materialDefines(defines) {\n var _a;\n const drawWrapper = (_a = this._mainDrawWrapperOverride) !== null && _a !== void 0 ? _a : this._getDrawWrapper(undefined, true);\n drawWrapper.defines = defines;\n }\n /**\n * @internal\n */\n _getDrawWrapper(passId, createIfNotExisting = false) {\n passId = passId !== null && passId !== void 0 ? passId : this._engine.currentRenderPassId;\n let drawWrapper = this._drawWrappers[passId];\n if (!drawWrapper && createIfNotExisting) {\n this._drawWrappers[passId] = drawWrapper = new DrawWrapper(this._mesh.getScene().getEngine());\n }\n return drawWrapper;\n }\n /**\n * @internal\n */\n _removeDrawWrapper(passId, disposeWrapper = true) {\n var _a;\n if (disposeWrapper) {\n (_a = this._drawWrappers[passId]) === null || _a === void 0 ? void 0 : _a.dispose();\n }\n this._drawWrappers[passId] = undefined;\n }\n /**\n * Gets associated (main) effect (possibly the effect override if defined)\n */\n get effect() {\n var _a, _b;\n return this._mainDrawWrapperOverride ? this._mainDrawWrapperOverride.effect : (_b = (_a = this._getDrawWrapper()) === null || _a === void 0 ? void 0 : _a.effect) !== null && _b !== void 0 ? _b : null;\n }\n /** @internal */\n get _drawWrapper() {\n var _a;\n return (_a = this._mainDrawWrapperOverride) !== null && _a !== void 0 ? _a : this._getDrawWrapper(undefined, true);\n }\n /** @internal */\n get _drawWrapperOverride() {\n return this._mainDrawWrapperOverride;\n }\n /**\n * @internal\n */\n _setMainDrawWrapperOverride(wrapper) {\n this._mainDrawWrapperOverride = wrapper;\n }\n /**\n * Sets associated effect (effect used to render this submesh)\n * @param effect defines the effect to associate with\n * @param defines defines the set of defines used to compile this effect\n * @param materialContext material context associated to the effect\n * @param resetContext true to reset the draw context\n */\n setEffect(effect, defines = null, materialContext, resetContext = true) {\n const drawWrapper = this._drawWrapper;\n drawWrapper.setEffect(effect, defines, resetContext);\n if (materialContext !== undefined) {\n drawWrapper.materialContext = materialContext;\n }\n if (!effect) {\n drawWrapper.defines = null;\n drawWrapper.materialContext = undefined;\n }\n }\n /**\n * Resets the draw wrappers cache\n * @param passId If provided, releases only the draw wrapper corresponding to this render pass id\n */\n resetDrawCache(passId) {\n if (this._drawWrappers) {\n if (passId !== undefined) {\n this._removeDrawWrapper(passId);\n return;\n }\n else {\n for (const drawWrapper of this._drawWrappers) {\n drawWrapper === null || drawWrapper === void 0 ? void 0 : drawWrapper.dispose();\n }\n }\n }\n this._drawWrappers = [];\n }\n /**\n * Add a new submesh to a mesh\n * @param materialIndex defines the material index to use\n * @param verticesStart defines vertex index start\n * @param verticesCount defines vertices count\n * @param indexStart defines index start\n * @param indexCount defines indices count\n * @param mesh defines the parent mesh\n * @param renderingMesh defines an optional rendering mesh\n * @param createBoundingBox defines if bounding box should be created for this submesh\n * @returns the new submesh\n */\n static AddToMesh(materialIndex, verticesStart, verticesCount, indexStart, indexCount, mesh, renderingMesh, createBoundingBox = true) {\n return new SubMesh(materialIndex, verticesStart, verticesCount, indexStart, indexCount, mesh, renderingMesh, createBoundingBox);\n }\n /**\n * Creates a new submesh\n * @param materialIndex defines the material index to use\n * @param verticesStart defines vertex index start\n * @param verticesCount defines vertices count\n * @param indexStart defines index start\n * @param indexCount defines indices count\n * @param mesh defines the parent mesh\n * @param renderingMesh defines an optional rendering mesh\n * @param createBoundingBox defines if bounding box should be created for this submesh\n * @param addToMesh defines a boolean indicating that the submesh must be added to the mesh.subMeshes array (true by default)\n */\n constructor(\n /** the material index to use */\n materialIndex, \n /** vertex index start */\n verticesStart, \n /** vertices count */\n verticesCount, \n /** index start */\n indexStart, \n /** indices count */\n indexCount, mesh, renderingMesh, createBoundingBox = true, addToMesh = true) {\n this.materialIndex = materialIndex;\n this.verticesStart = verticesStart;\n this.verticesCount = verticesCount;\n this.indexStart = indexStart;\n this.indexCount = indexCount;\n this._mainDrawWrapperOverride = null;\n /** @internal */\n this._linesIndexCount = 0;\n this._linesIndexBuffer = null;\n /** @internal */\n this._lastColliderWorldVertices = null;\n /** @internal */\n this._lastColliderTransformMatrix = null;\n /** @internal */\n this._wasDispatched = false;\n /** @internal */\n this._renderId = 0;\n /** @internal */\n this._alphaIndex = 0;\n /** @internal */\n this._distanceToCamera = 0;\n this._currentMaterial = null;\n this._mesh = mesh;\n this._renderingMesh = renderingMesh || mesh;\n if (addToMesh) {\n mesh.subMeshes.push(this);\n }\n this._engine = this._mesh.getScene().getEngine();\n this.resetDrawCache();\n this._trianglePlanes = [];\n this._id = mesh.subMeshes.length - 1;\n if (createBoundingBox) {\n this.refreshBoundingInfo();\n mesh.computeWorldMatrix(true);\n }\n }\n /**\n * Returns true if this submesh covers the entire parent mesh\n * @ignorenaming\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n get IsGlobal() {\n return this.verticesStart === 0 && this.verticesCount === this._mesh.getTotalVertices() && this.indexStart === 0 && this.indexCount === this._mesh.getTotalIndices();\n }\n /**\n * Returns the submesh BoundingInfo object\n * @returns current bounding info (or mesh's one if the submesh is global)\n */\n getBoundingInfo() {\n if (this.IsGlobal) {\n return this._mesh.getBoundingInfo();\n }\n return this._boundingInfo;\n }\n /**\n * Sets the submesh BoundingInfo\n * @param boundingInfo defines the new bounding info to use\n * @returns the SubMesh\n */\n setBoundingInfo(boundingInfo) {\n this._boundingInfo = boundingInfo;\n return this;\n }\n /**\n * Returns the mesh of the current submesh\n * @returns the parent mesh\n */\n getMesh() {\n return this._mesh;\n }\n /**\n * Returns the rendering mesh of the submesh\n * @returns the rendering mesh (could be different from parent mesh)\n */\n getRenderingMesh() {\n return this._renderingMesh;\n }\n /**\n * Returns the replacement mesh of the submesh\n * @returns the replacement mesh (could be different from parent mesh)\n */\n getReplacementMesh() {\n return this._mesh._internalAbstractMeshDataInfo._actAsRegularMesh ? this._mesh : null;\n }\n /**\n * Returns the effective mesh of the submesh\n * @returns the effective mesh (could be different from parent mesh)\n */\n getEffectiveMesh() {\n const replacementMesh = this._mesh._internalAbstractMeshDataInfo._actAsRegularMesh ? this._mesh : null;\n return replacementMesh ? replacementMesh : this._renderingMesh;\n }\n /**\n * Returns the submesh material\n * @param getDefaultMaterial Defines whether or not to get the default material if nothing has been defined.\n * @returns null or the current material\n */\n getMaterial(getDefaultMaterial = true) {\n var _a;\n const rootMaterial = (_a = this._renderingMesh.getMaterialForRenderPass(this._engine.currentRenderPassId)) !== null && _a !== void 0 ? _a : this._renderingMesh.material;\n if (!rootMaterial) {\n return getDefaultMaterial ? this._mesh.getScene().defaultMaterial : null;\n }\n else if (this._isMultiMaterial(rootMaterial)) {\n const effectiveMaterial = rootMaterial.getSubMaterial(this.materialIndex);\n if (this._currentMaterial !== effectiveMaterial) {\n this._currentMaterial = effectiveMaterial;\n this.resetDrawCache();\n }\n return effectiveMaterial;\n }\n return rootMaterial;\n }\n _isMultiMaterial(material) {\n return material.getSubMaterial !== undefined;\n }\n // Methods\n /**\n * Sets a new updated BoundingInfo object to the submesh\n * @param data defines an optional position array to use to determine the bounding info\n * @returns the SubMesh\n */\n refreshBoundingInfo(data = null) {\n this._lastColliderWorldVertices = null;\n if (this.IsGlobal || !this._renderingMesh || !this._renderingMesh.geometry) {\n return this;\n }\n if (!data) {\n data = this._renderingMesh.getVerticesData(VertexBuffer.PositionKind);\n }\n if (!data) {\n this._boundingInfo = this._mesh.getBoundingInfo();\n return this;\n }\n const indices = this._renderingMesh.getIndices();\n let extend;\n //is this the only submesh?\n if (this.indexStart === 0 && this.indexCount === indices.length) {\n const boundingInfo = this._renderingMesh.getBoundingInfo();\n //the rendering mesh's bounding info can be used, it is the standard submesh for all indices.\n extend = { minimum: boundingInfo.minimum.clone(), maximum: boundingInfo.maximum.clone() };\n }\n else {\n extend = extractMinAndMaxIndexed(data, indices, this.indexStart, this.indexCount, this._renderingMesh.geometry.boundingBias);\n }\n if (this._boundingInfo) {\n this._boundingInfo.reConstruct(extend.minimum, extend.maximum);\n }\n else {\n this._boundingInfo = new BoundingInfo(extend.minimum, extend.maximum);\n }\n return this;\n }\n /**\n * @internal\n */\n _checkCollision(collider) {\n const boundingInfo = this.getBoundingInfo();\n return boundingInfo._checkCollision(collider);\n }\n /**\n * Updates the submesh BoundingInfo\n * @param world defines the world matrix to use to update the bounding info\n * @returns the submesh\n */\n updateBoundingInfo(world) {\n let boundingInfo = this.getBoundingInfo();\n if (!boundingInfo) {\n this.refreshBoundingInfo();\n boundingInfo = this.getBoundingInfo();\n }\n if (boundingInfo) {\n boundingInfo.update(world);\n }\n return this;\n }\n /**\n * True is the submesh bounding box intersects the frustum defined by the passed array of planes.\n * @param frustumPlanes defines the frustum planes\n * @returns true if the submesh is intersecting with the frustum\n */\n isInFrustum(frustumPlanes) {\n const boundingInfo = this.getBoundingInfo();\n if (!boundingInfo) {\n return false;\n }\n return boundingInfo.isInFrustum(frustumPlanes, this._mesh.cullingStrategy);\n }\n /**\n * True is the submesh bounding box is completely inside the frustum defined by the passed array of planes\n * @param frustumPlanes defines the frustum planes\n * @returns true if the submesh is inside the frustum\n */\n isCompletelyInFrustum(frustumPlanes) {\n const boundingInfo = this.getBoundingInfo();\n if (!boundingInfo) {\n return false;\n }\n return boundingInfo.isCompletelyInFrustum(frustumPlanes);\n }\n /**\n * Renders the submesh\n * @param enableAlphaMode defines if alpha needs to be used\n * @returns the submesh\n */\n render(enableAlphaMode) {\n this._renderingMesh.render(this, enableAlphaMode, this._mesh._internalAbstractMeshDataInfo._actAsRegularMesh ? this._mesh : undefined);\n return this;\n }\n /**\n * @internal\n */\n _getLinesIndexBuffer(indices, engine) {\n if (!this._linesIndexBuffer) {\n const linesIndices = [];\n for (let index = this.indexStart; index < this.indexStart + this.indexCount; index += 3) {\n linesIndices.push(indices[index], indices[index + 1], indices[index + 1], indices[index + 2], indices[index + 2], indices[index]);\n }\n this._linesIndexBuffer = engine.createIndexBuffer(linesIndices);\n this._linesIndexCount = linesIndices.length;\n }\n return this._linesIndexBuffer;\n }\n /**\n * Checks if the submesh intersects with a ray\n * @param ray defines the ray to test\n * @returns true is the passed ray intersects the submesh bounding box\n */\n canIntersects(ray) {\n const boundingInfo = this.getBoundingInfo();\n if (!boundingInfo) {\n return false;\n }\n return ray.intersectsBox(boundingInfo.boundingBox);\n }\n /**\n * Intersects current submesh with a ray\n * @param ray defines the ray to test\n * @param positions defines mesh's positions array\n * @param indices defines mesh's indices array\n * @param fastCheck defines if the first intersection will be used (and not the closest)\n * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected\n * @returns intersection info or null if no intersection\n */\n intersects(ray, positions, indices, fastCheck, trianglePredicate) {\n const material = this.getMaterial();\n if (!material) {\n return null;\n }\n let step = 3;\n let checkStopper = false;\n switch (material.fillMode) {\n case 3:\n case 5:\n case 6:\n case 8:\n return null;\n case 7:\n step = 1;\n checkStopper = true;\n break;\n }\n // LineMesh first as it's also a Mesh...\n if (material.fillMode === 4) {\n // Check if mesh is unindexed\n if (!indices.length) {\n return this._intersectUnIndexedLines(ray, positions, indices, this._mesh.intersectionThreshold, fastCheck);\n }\n return this._intersectLines(ray, positions, indices, this._mesh.intersectionThreshold, fastCheck);\n }\n else {\n // Check if mesh is unindexed\n if (!indices.length && this._mesh._unIndexed) {\n return this._intersectUnIndexedTriangles(ray, positions, indices, fastCheck, trianglePredicate);\n }\n return this._intersectTriangles(ray, positions, indices, step, checkStopper, fastCheck, trianglePredicate);\n }\n }\n /**\n * @internal\n */\n _intersectLines(ray, positions, indices, intersectionThreshold, fastCheck) {\n let intersectInfo = null;\n // Line test\n for (let index = this.indexStart; index < this.indexStart + this.indexCount; index += 2) {\n const p0 = positions[indices[index]];\n const p1 = positions[indices[index + 1]];\n const length = ray.intersectionSegment(p0, p1, intersectionThreshold);\n if (length < 0) {\n continue;\n }\n if (fastCheck || !intersectInfo || length < intersectInfo.distance) {\n intersectInfo = new IntersectionInfo(null, null, length);\n intersectInfo.faceId = index / 2;\n if (fastCheck) {\n break;\n }\n }\n }\n return intersectInfo;\n }\n /**\n * @internal\n */\n _intersectUnIndexedLines(ray, positions, indices, intersectionThreshold, fastCheck) {\n let intersectInfo = null;\n // Line test\n for (let index = this.verticesStart; index < this.verticesStart + this.verticesCount; index += 2) {\n const p0 = positions[index];\n const p1 = positions[index + 1];\n const length = ray.intersectionSegment(p0, p1, intersectionThreshold);\n if (length < 0) {\n continue;\n }\n if (fastCheck || !intersectInfo || length < intersectInfo.distance) {\n intersectInfo = new IntersectionInfo(null, null, length);\n intersectInfo.faceId = index / 2;\n if (fastCheck) {\n break;\n }\n }\n }\n return intersectInfo;\n }\n /**\n * @internal\n */\n _intersectTriangles(ray, positions, indices, step, checkStopper, fastCheck, trianglePredicate) {\n let intersectInfo = null;\n // Triangles test\n let faceId = -1;\n for (let index = this.indexStart; index < this.indexStart + this.indexCount - (3 - step); index += step) {\n faceId++;\n const indexA = indices[index];\n const indexB = indices[index + 1];\n const indexC = indices[index + 2];\n if (checkStopper && indexC === 0xffffffff) {\n index += 2;\n continue;\n }\n const p0 = positions[indexA];\n const p1 = positions[indexB];\n const p2 = positions[indexC];\n // stay defensive and don't check against undefined positions.\n if (!p0 || !p1 || !p2) {\n continue;\n }\n if (trianglePredicate && !trianglePredicate(p0, p1, p2, ray, indexA, indexB, indexC)) {\n continue;\n }\n const currentIntersectInfo = ray.intersectsTriangle(p0, p1, p2);\n if (currentIntersectInfo) {\n if (currentIntersectInfo.distance < 0) {\n continue;\n }\n if (fastCheck || !intersectInfo || currentIntersectInfo.distance < intersectInfo.distance) {\n intersectInfo = currentIntersectInfo;\n intersectInfo.faceId = faceId;\n if (fastCheck) {\n break;\n }\n }\n }\n }\n return intersectInfo;\n }\n /**\n * @internal\n */\n _intersectUnIndexedTriangles(ray, positions, indices, fastCheck, trianglePredicate) {\n let intersectInfo = null;\n // Triangles test\n for (let index = this.verticesStart; index < this.verticesStart + this.verticesCount; index += 3) {\n const p0 = positions[index];\n const p1 = positions[index + 1];\n const p2 = positions[index + 2];\n if (trianglePredicate && !trianglePredicate(p0, p1, p2, ray, -1, -1, -1)) {\n continue;\n }\n const currentIntersectInfo = ray.intersectsTriangle(p0, p1, p2);\n if (currentIntersectInfo) {\n if (currentIntersectInfo.distance < 0) {\n continue;\n }\n if (fastCheck || !intersectInfo || currentIntersectInfo.distance < intersectInfo.distance) {\n intersectInfo = currentIntersectInfo;\n intersectInfo.faceId = index / 3;\n if (fastCheck) {\n break;\n }\n }\n }\n }\n return intersectInfo;\n }\n /** @internal */\n _rebuild() {\n if (this._linesIndexBuffer) {\n this._linesIndexBuffer = null;\n }\n }\n // Clone\n /**\n * Creates a new submesh from the passed mesh\n * @param newMesh defines the new hosting mesh\n * @param newRenderingMesh defines an optional rendering mesh\n * @returns the new submesh\n */\n clone(newMesh, newRenderingMesh) {\n const result = new SubMesh(this.materialIndex, this.verticesStart, this.verticesCount, this.indexStart, this.indexCount, newMesh, newRenderingMesh, false);\n if (!this.IsGlobal) {\n const boundingInfo = this.getBoundingInfo();\n if (!boundingInfo) {\n return result;\n }\n result._boundingInfo = new BoundingInfo(boundingInfo.minimum, boundingInfo.maximum);\n }\n return result;\n }\n // Dispose\n /**\n * Release associated resources\n */\n dispose() {\n if (this._linesIndexBuffer) {\n this._mesh.getScene().getEngine()._releaseBuffer(this._linesIndexBuffer);\n this._linesIndexBuffer = null;\n }\n // Remove from mesh\n const index = this._mesh.subMeshes.indexOf(this);\n this._mesh.subMeshes.splice(index, 1);\n this.resetDrawCache();\n }\n /**\n * Gets the class name\n * @returns the string \"SubMesh\".\n */\n getClassName() {\n return \"SubMesh\";\n }\n // Statics\n /**\n * Creates a new submesh from indices data\n * @param materialIndex the index of the main mesh material\n * @param startIndex the index where to start the copy in the mesh indices array\n * @param indexCount the number of indices to copy then from the startIndex\n * @param mesh the main mesh to create the submesh from\n * @param renderingMesh the optional rendering mesh\n * @param createBoundingBox defines if bounding box should be created for this submesh\n * @returns a new submesh\n */\n static CreateFromIndices(materialIndex, startIndex, indexCount, mesh, renderingMesh, createBoundingBox = true) {\n let minVertexIndex = Number.MAX_VALUE;\n let maxVertexIndex = -Number.MAX_VALUE;\n const whatWillRender = renderingMesh || mesh;\n const indices = whatWillRender.getIndices();\n for (let index = startIndex; index < startIndex + indexCount; index++) {\n const vertexIndex = indices[index];\n if (vertexIndex < minVertexIndex) {\n minVertexIndex = vertexIndex;\n }\n if (vertexIndex > maxVertexIndex) {\n maxVertexIndex = vertexIndex;\n }\n }\n return new SubMesh(materialIndex, minVertexIndex, maxVertexIndex - minVertexIndex + 1, startIndex, indexCount, mesh, renderingMesh, createBoundingBox);\n }\n }\n\n /** Class used to attach material info to sub section of a vertex data class */\n class VertexDataMaterialInfo {\n }\n /**\n * This class contains the various kinds of data on every vertex of a mesh used in determining its shape and appearance\n */\n class VertexData {\n constructor() {\n this._applyTo = makeSyncFunction(this._applyToCoroutine.bind(this));\n }\n /**\n * Uses the passed data array to set the set the values for the specified kind of data\n * @param data a linear array of floating numbers\n * @param kind the type of data that is being set, eg positions, colors etc\n */\n set(data, kind) {\n if (!data.length) {\n Logger.Warn(`Setting vertex data kind '${kind}' with an empty array`);\n }\n switch (kind) {\n case VertexBuffer.PositionKind:\n this.positions = data;\n break;\n case VertexBuffer.NormalKind:\n this.normals = data;\n break;\n case VertexBuffer.TangentKind:\n this.tangents = data;\n break;\n case VertexBuffer.UVKind:\n this.uvs = data;\n break;\n case VertexBuffer.UV2Kind:\n this.uvs2 = data;\n break;\n case VertexBuffer.UV3Kind:\n this.uvs3 = data;\n break;\n case VertexBuffer.UV4Kind:\n this.uvs4 = data;\n break;\n case VertexBuffer.UV5Kind:\n this.uvs5 = data;\n break;\n case VertexBuffer.UV6Kind:\n this.uvs6 = data;\n break;\n case VertexBuffer.ColorKind:\n this.colors = data;\n break;\n case VertexBuffer.MatricesIndicesKind:\n this.matricesIndices = data;\n break;\n case VertexBuffer.MatricesWeightsKind:\n this.matricesWeights = data;\n break;\n case VertexBuffer.MatricesIndicesExtraKind:\n this.matricesIndicesExtra = data;\n break;\n case VertexBuffer.MatricesWeightsExtraKind:\n this.matricesWeightsExtra = data;\n break;\n }\n }\n /**\n * Associates the vertexData to the passed Mesh.\n * Sets it as updatable or not (default `false`)\n * @param mesh the mesh the vertexData is applied to\n * @param updatable when used and having the value true allows new data to update the vertexData\n * @returns the VertexData\n */\n applyToMesh(mesh, updatable) {\n this._applyTo(mesh, updatable, false);\n return this;\n }\n /**\n * Associates the vertexData to the passed Geometry.\n * Sets it as updatable or not (default `false`)\n * @param geometry the geometry the vertexData is applied to\n * @param updatable when used and having the value true allows new data to update the vertexData\n * @returns VertexData\n */\n applyToGeometry(geometry, updatable) {\n this._applyTo(geometry, updatable, false);\n return this;\n }\n /**\n * Updates the associated mesh\n * @param mesh the mesh to be updated\n * @returns VertexData\n */\n updateMesh(mesh) {\n this._update(mesh);\n return this;\n }\n /**\n * Updates the associated geometry\n * @param geometry the geometry to be updated\n * @returns VertexData.\n */\n updateGeometry(geometry) {\n this._update(geometry);\n return this;\n }\n /**\n * @internal\n */\n *_applyToCoroutine(meshOrGeometry, updatable = false, isAsync) {\n if (this.positions) {\n meshOrGeometry.setVerticesData(VertexBuffer.PositionKind, this.positions, updatable);\n if (isAsync) {\n yield;\n }\n }\n if (this.normals) {\n meshOrGeometry.setVerticesData(VertexBuffer.NormalKind, this.normals, updatable);\n if (isAsync) {\n yield;\n }\n }\n if (this.tangents) {\n meshOrGeometry.setVerticesData(VertexBuffer.TangentKind, this.tangents, updatable);\n if (isAsync) {\n yield;\n }\n }\n if (this.uvs) {\n meshOrGeometry.setVerticesData(VertexBuffer.UVKind, this.uvs, updatable);\n if (isAsync) {\n yield;\n }\n }\n if (this.uvs2) {\n meshOrGeometry.setVerticesData(VertexBuffer.UV2Kind, this.uvs2, updatable);\n if (isAsync) {\n yield;\n }\n }\n if (this.uvs3) {\n meshOrGeometry.setVerticesData(VertexBuffer.UV3Kind, this.uvs3, updatable);\n if (isAsync) {\n yield;\n }\n }\n if (this.uvs4) {\n meshOrGeometry.setVerticesData(VertexBuffer.UV4Kind, this.uvs4, updatable);\n if (isAsync) {\n yield;\n }\n }\n if (this.uvs5) {\n meshOrGeometry.setVerticesData(VertexBuffer.UV5Kind, this.uvs5, updatable);\n if (isAsync) {\n yield;\n }\n }\n if (this.uvs6) {\n meshOrGeometry.setVerticesData(VertexBuffer.UV6Kind, this.uvs6, updatable);\n if (isAsync) {\n yield;\n }\n }\n if (this.colors) {\n meshOrGeometry.setVerticesData(VertexBuffer.ColorKind, this.colors, updatable);\n if (isAsync) {\n yield;\n }\n }\n if (this.matricesIndices) {\n meshOrGeometry.setVerticesData(VertexBuffer.MatricesIndicesKind, this.matricesIndices, updatable);\n if (isAsync) {\n yield;\n }\n }\n if (this.matricesWeights) {\n meshOrGeometry.setVerticesData(VertexBuffer.MatricesWeightsKind, this.matricesWeights, updatable);\n if (isAsync) {\n yield;\n }\n }\n if (this.matricesIndicesExtra) {\n meshOrGeometry.setVerticesData(VertexBuffer.MatricesIndicesExtraKind, this.matricesIndicesExtra, updatable);\n if (isAsync) {\n yield;\n }\n }\n if (this.matricesWeightsExtra) {\n meshOrGeometry.setVerticesData(VertexBuffer.MatricesWeightsExtraKind, this.matricesWeightsExtra, updatable);\n if (isAsync) {\n yield;\n }\n }\n if (this.indices) {\n meshOrGeometry.setIndices(this.indices, null, updatable);\n if (isAsync) {\n yield;\n }\n }\n else {\n meshOrGeometry.setIndices([], null);\n }\n if (meshOrGeometry.subMeshes && this.materialInfos && this.materialInfos.length > 1) {\n const mesh = meshOrGeometry;\n mesh.subMeshes = [];\n for (const matInfo of this.materialInfos) {\n new SubMesh(matInfo.materialIndex, matInfo.verticesStart, matInfo.verticesCount, matInfo.indexStart, matInfo.indexCount, mesh);\n }\n }\n return this;\n }\n _update(meshOrGeometry, updateExtends, makeItUnique) {\n if (this.positions) {\n meshOrGeometry.updateVerticesData(VertexBuffer.PositionKind, this.positions, updateExtends, makeItUnique);\n }\n if (this.normals) {\n meshOrGeometry.updateVerticesData(VertexBuffer.NormalKind, this.normals, updateExtends, makeItUnique);\n }\n if (this.tangents) {\n meshOrGeometry.updateVerticesData(VertexBuffer.TangentKind, this.tangents, updateExtends, makeItUnique);\n }\n if (this.uvs) {\n meshOrGeometry.updateVerticesData(VertexBuffer.UVKind, this.uvs, updateExtends, makeItUnique);\n }\n if (this.uvs2) {\n meshOrGeometry.updateVerticesData(VertexBuffer.UV2Kind, this.uvs2, updateExtends, makeItUnique);\n }\n if (this.uvs3) {\n meshOrGeometry.updateVerticesData(VertexBuffer.UV3Kind, this.uvs3, updateExtends, makeItUnique);\n }\n if (this.uvs4) {\n meshOrGeometry.updateVerticesData(VertexBuffer.UV4Kind, this.uvs4, updateExtends, makeItUnique);\n }\n if (this.uvs5) {\n meshOrGeometry.updateVerticesData(VertexBuffer.UV5Kind, this.uvs5, updateExtends, makeItUnique);\n }\n if (this.uvs6) {\n meshOrGeometry.updateVerticesData(VertexBuffer.UV6Kind, this.uvs6, updateExtends, makeItUnique);\n }\n if (this.colors) {\n meshOrGeometry.updateVerticesData(VertexBuffer.ColorKind, this.colors, updateExtends, makeItUnique);\n }\n if (this.matricesIndices) {\n meshOrGeometry.updateVerticesData(VertexBuffer.MatricesIndicesKind, this.matricesIndices, updateExtends, makeItUnique);\n }\n if (this.matricesWeights) {\n meshOrGeometry.updateVerticesData(VertexBuffer.MatricesWeightsKind, this.matricesWeights, updateExtends, makeItUnique);\n }\n if (this.matricesIndicesExtra) {\n meshOrGeometry.updateVerticesData(VertexBuffer.MatricesIndicesExtraKind, this.matricesIndicesExtra, updateExtends, makeItUnique);\n }\n if (this.matricesWeightsExtra) {\n meshOrGeometry.updateVerticesData(VertexBuffer.MatricesWeightsExtraKind, this.matricesWeightsExtra, updateExtends, makeItUnique);\n }\n if (this.indices) {\n meshOrGeometry.setIndices(this.indices, null);\n }\n return this;\n }\n static _TransformVector3Coordinates(coordinates, transformation, offset = 0, length = coordinates.length) {\n const coordinate = TmpVectors.Vector3[0];\n const transformedCoordinate = TmpVectors.Vector3[1];\n for (let index = offset; index < offset + length; index += 3) {\n Vector3.FromArrayToRef(coordinates, index, coordinate);\n Vector3.TransformCoordinatesToRef(coordinate, transformation, transformedCoordinate);\n coordinates[index] = transformedCoordinate.x;\n coordinates[index + 1] = transformedCoordinate.y;\n coordinates[index + 2] = transformedCoordinate.z;\n }\n }\n static _TransformVector3Normals(normals, transformation, offset = 0, length = normals.length) {\n const normal = TmpVectors.Vector3[0];\n const transformedNormal = TmpVectors.Vector3[1];\n for (let index = offset; index < offset + length; index += 3) {\n Vector3.FromArrayToRef(normals, index, normal);\n Vector3.TransformNormalToRef(normal, transformation, transformedNormal);\n normals[index] = transformedNormal.x;\n normals[index + 1] = transformedNormal.y;\n normals[index + 2] = transformedNormal.z;\n }\n }\n static _TransformVector4Normals(normals, transformation, offset = 0, length = normals.length) {\n const normal = TmpVectors.Vector4[0];\n const transformedNormal = TmpVectors.Vector4[1];\n for (let index = offset; index < offset + length; index += 4) {\n Vector4.FromArrayToRef(normals, index, normal);\n Vector4.TransformNormalToRef(normal, transformation, transformedNormal);\n normals[index] = transformedNormal.x;\n normals[index + 1] = transformedNormal.y;\n normals[index + 2] = transformedNormal.z;\n normals[index + 3] = transformedNormal.w;\n }\n }\n static _FlipFaces(indices, offset = 0, length = indices.length) {\n for (let index = offset; index < offset + length; index += 3) {\n const tmp = indices[index + 1];\n indices[index + 1] = indices[index + 2];\n indices[index + 2] = tmp;\n }\n }\n /**\n * Transforms each position and each normal of the vertexData according to the passed Matrix\n * @param matrix the transforming matrix\n * @returns the VertexData\n */\n transform(matrix) {\n const flip = matrix.determinant() < 0;\n if (this.positions) {\n VertexData._TransformVector3Coordinates(this.positions, matrix);\n }\n if (this.normals) {\n VertexData._TransformVector3Normals(this.normals, matrix);\n }\n if (this.tangents) {\n VertexData._TransformVector4Normals(this.tangents, matrix);\n }\n if (flip && this.indices) {\n VertexData._FlipFaces(this.indices);\n }\n return this;\n }\n /**\n * Generates an array of vertex data where each vertex data only has one material info\n * @returns An array of VertexData\n */\n splitBasedOnMaterialID() {\n if (!this.materialInfos || this.materialInfos.length < 2) {\n return [this];\n }\n const result = new Array();\n for (const materialInfo of this.materialInfos) {\n const vertexData = new VertexData();\n if (this.positions) {\n vertexData.positions = this.positions.slice(materialInfo.verticesStart * 3, (materialInfo.verticesCount + materialInfo.verticesStart) * 3);\n }\n if (this.normals) {\n vertexData.normals = this.normals.slice(materialInfo.verticesStart * 3, (materialInfo.verticesCount + materialInfo.verticesStart) * 3);\n }\n if (this.tangents) {\n vertexData.tangents = this.tangents.slice(materialInfo.verticesStart * 4, (materialInfo.verticesCount + materialInfo.verticesStart) * 4);\n }\n if (this.colors) {\n vertexData.colors = this.colors.slice(materialInfo.verticesStart * 4, (materialInfo.verticesCount + materialInfo.verticesStart) * 4);\n }\n if (this.uvs) {\n vertexData.uvs = this.uvs.slice(materialInfo.verticesStart * 2, (materialInfo.verticesCount + materialInfo.verticesStart) * 2);\n }\n if (this.uvs2) {\n vertexData.uvs2 = this.uvs2.slice(materialInfo.verticesStart * 2, (materialInfo.verticesCount + materialInfo.verticesStart) * 2);\n }\n if (this.uvs3) {\n vertexData.uvs3 = this.uvs3.slice(materialInfo.verticesStart * 2, (materialInfo.verticesCount + materialInfo.verticesStart) * 2);\n }\n if (this.uvs4) {\n vertexData.uvs4 = this.uvs4.slice(materialInfo.verticesStart * 2, (materialInfo.verticesCount + materialInfo.verticesStart) * 2);\n }\n if (this.uvs5) {\n vertexData.uvs5 = this.uvs5.slice(materialInfo.verticesStart * 2, (materialInfo.verticesCount + materialInfo.verticesStart) * 2);\n }\n if (this.uvs6) {\n vertexData.uvs6 = this.uvs6.slice(materialInfo.verticesStart * 2, (materialInfo.verticesCount + materialInfo.verticesStart) * 2);\n }\n if (this.matricesIndices) {\n vertexData.matricesIndices = this.matricesIndices.slice(materialInfo.verticesStart * 4, (materialInfo.verticesCount + materialInfo.verticesStart) * 4);\n }\n if (this.matricesIndicesExtra) {\n vertexData.matricesIndicesExtra = this.matricesIndicesExtra.slice(materialInfo.verticesStart * 4, (materialInfo.verticesCount + materialInfo.verticesStart) * 4);\n }\n if (this.matricesWeights) {\n vertexData.matricesWeights = this.matricesWeights.slice(materialInfo.verticesStart * 4, (materialInfo.verticesCount + materialInfo.verticesStart) * 4);\n }\n if (this.matricesWeightsExtra) {\n vertexData.matricesWeightsExtra = this.matricesWeightsExtra.slice(materialInfo.verticesStart * 4, (materialInfo.verticesCount + materialInfo.verticesStart) * 4);\n }\n if (this.indices) {\n vertexData.indices = [];\n for (let index = materialInfo.indexStart; index < materialInfo.indexStart + materialInfo.indexCount; index++) {\n vertexData.indices.push(this.indices[index] - materialInfo.verticesStart);\n }\n }\n const newMaterialInfo = new VertexDataMaterialInfo();\n newMaterialInfo.indexStart = 0;\n newMaterialInfo.indexCount = vertexData.indices ? vertexData.indices.length : 0;\n newMaterialInfo.materialIndex = materialInfo.materialIndex;\n newMaterialInfo.verticesStart = 0;\n newMaterialInfo.verticesCount = (vertexData.positions ? vertexData.positions.length : 0) / 3;\n vertexData.materialInfos = [newMaterialInfo];\n result.push(vertexData);\n }\n return result;\n }\n /**\n * Merges the passed VertexData into the current one\n * @param others the VertexData to be merged into the current one\n * @param use32BitsIndices defines a boolean indicating if indices must be store in a 32 bits array\n * @param forceCloneIndices defines a boolean indicating if indices are forced to be cloned\n * @param mergeMaterialIds defines a boolean indicating if we need to merge the material infos\n * @param enableCompletion defines a boolean indicating if the vertex data should be completed to be compatible\n * @returns the modified VertexData\n */\n merge(others, use32BitsIndices = false, forceCloneIndices = false, mergeMaterialIds = false, enableCompletion = false) {\n const vertexDatas = Array.isArray(others)\n ? others.map((other) => {\n return { vertexData: other };\n })\n : [{ vertexData: others }];\n return runCoroutineSync(this._mergeCoroutine(undefined, vertexDatas, use32BitsIndices, false, forceCloneIndices, mergeMaterialIds, enableCompletion));\n }\n /**\n * @internal\n */\n *_mergeCoroutine(transform, vertexDatas, use32BitsIndices = false, isAsync, forceCloneIndices, mergeMaterialIds = false, enableCompletion = false) {\n var _a, _b, _c, _d;\n this._validate();\n let others = vertexDatas.map((vertexData) => vertexData.vertexData);\n // eslint-disable-next-line @typescript-eslint/no-this-alias\n let root = this;\n for (const other of others) {\n if (!other) {\n continue;\n }\n other._validate();\n if (!enableCompletion) {\n if (!this.normals !== !other.normals ||\n !this.tangents !== !other.tangents ||\n !this.uvs !== !other.uvs ||\n !this.uvs2 !== !other.uvs2 ||\n !this.uvs3 !== !other.uvs3 ||\n !this.uvs4 !== !other.uvs4 ||\n !this.uvs5 !== !other.uvs5 ||\n !this.uvs6 !== !other.uvs6 ||\n !this.colors !== !other.colors ||\n !this.matricesIndices !== !other.matricesIndices ||\n !this.matricesWeights !== !other.matricesWeights ||\n !this.matricesIndicesExtra !== !other.matricesIndicesExtra ||\n !this.matricesWeightsExtra !== !other.matricesWeightsExtra) {\n throw new Error(\"Cannot merge vertex data that do not have the same set of attributes\");\n }\n }\n else {\n if (!this.normals !== !other.normals) {\n if (!this.normals) {\n this.normals = Array.from(other.normals);\n }\n else {\n other.normals = Array.from(this.normals);\n }\n }\n if (!this.tangents !== !other.tangents) {\n if (!this.tangents) {\n this.tangents = Array.from(other.tangents);\n }\n else {\n other.tangents = Array.from(this.tangents);\n }\n }\n if (!this.uvs !== !other.uvs) {\n if (!this.uvs) {\n this.uvs = Array.from(other.uvs);\n }\n else {\n other.uvs = Array.from(this.uvs);\n }\n }\n if (!this.uvs2 !== !other.uvs2) {\n if (!this.uvs2) {\n this.uvs2 = Array.from(other.uvs2);\n }\n else {\n other.uvs2 = Array.from(this.uvs2);\n }\n }\n if (!this.uvs3 !== !other.uvs3) {\n if (!this.uvs3) {\n this.uvs3 = Array.from(other.uvs3);\n }\n else {\n other.uvs3 = Array.from(this.uvs3);\n }\n }\n if (!this.uvs4 !== !other.uvs4) {\n if (!this.uvs4) {\n this.uvs4 = Array.from(other.uvs4);\n }\n else {\n other.uvs4 = Array.from(this.uvs4);\n }\n }\n if (!this.uvs5 !== !other.uvs5) {\n if (!this.uvs5) {\n this.uvs5 = Array.from(other.uvs5);\n }\n else {\n other.uvs5 = Array.from(this.uvs5);\n }\n }\n if (!this.uvs6 !== !other.uvs6) {\n if (!this.uvs6) {\n this.uvs6 = Array.from(other.uvs6);\n }\n else {\n other.uvs6 = Array.from(this.uvs6);\n }\n }\n if (!this.colors !== !other.colors) {\n if (!this.colors) {\n this.colors = Array.from(other.colors);\n }\n else {\n other.colors = Array.from(this.colors);\n }\n }\n if (!this.matricesIndices !== !other.matricesIndices) {\n if (!this.matricesIndices) {\n this.matricesIndices = Array.from(other.matricesIndices);\n }\n else {\n other.matricesIndices = Array.from(this.matricesIndices);\n }\n }\n if (!this.matricesWeights !== !other.matricesWeights) {\n if (!this.matricesWeights) {\n this.matricesWeights = Array.from(other.matricesWeights);\n }\n else {\n other.matricesWeights = Array.from(this.matricesWeights);\n }\n }\n if (!this.matricesIndicesExtra !== !other.matricesIndicesExtra) {\n if (!this.matricesIndicesExtra) {\n this.matricesIndicesExtra = Array.from(other.matricesIndicesExtra);\n }\n else {\n other.matricesIndicesExtra = Array.from(this.matricesIndicesExtra);\n }\n }\n if (!this.matricesWeightsExtra !== !other.matricesWeightsExtra) {\n if (!this.matricesWeightsExtra) {\n this.matricesWeightsExtra = Array.from(other.matricesWeightsExtra);\n }\n else {\n other.matricesWeightsExtra = Array.from(this.matricesWeightsExtra);\n }\n }\n }\n }\n if (mergeMaterialIds) {\n // Merge material infos\n let materialIndex = 0;\n let indexOffset = 0;\n let vertexOffset = 0;\n const materialInfos = [];\n let currentMaterialInfo = null;\n const vertexDataList = [];\n // We need to split vertexData with more than one materialInfo\n for (const split of this.splitBasedOnMaterialID()) {\n vertexDataList.push({ vertexData: split, transform: transform });\n }\n for (const vertexData of vertexDatas) {\n for (const split of vertexData.vertexData.splitBasedOnMaterialID()) {\n vertexDataList.push({ vertexData: split, transform: vertexData.transform });\n }\n }\n // Sort by material IDs\n vertexDataList.sort((a, b) => {\n const matInfoA = a.vertexData.materialInfos ? a.vertexData.materialInfos[0].materialIndex : 0;\n const matInfoB = b.vertexData.materialInfos ? b.vertexData.materialInfos[0].materialIndex : 0;\n if (matInfoA > matInfoB) {\n return 1;\n }\n if (matInfoA === matInfoB) {\n return 0;\n }\n return -1;\n });\n // Build the new material info\n for (const vertexDataSource of vertexDataList) {\n const vertexData = vertexDataSource.vertexData;\n if (vertexData.materialInfos) {\n materialIndex = vertexData.materialInfos[0].materialIndex;\n }\n else {\n materialIndex = 0;\n }\n if (currentMaterialInfo && currentMaterialInfo.materialIndex === materialIndex) {\n currentMaterialInfo.indexCount += vertexData.indices.length;\n currentMaterialInfo.verticesCount += vertexData.positions.length / 3;\n }\n else {\n const materialInfo = new VertexDataMaterialInfo();\n materialInfo.materialIndex = materialIndex;\n materialInfo.indexStart = indexOffset;\n materialInfo.indexCount = vertexData.indices.length;\n materialInfo.verticesStart = vertexOffset;\n materialInfo.verticesCount = vertexData.positions.length / 3;\n materialInfos.push(materialInfo);\n currentMaterialInfo = materialInfo;\n }\n indexOffset += vertexData.indices.length;\n vertexOffset += vertexData.positions.length / 3;\n }\n // Extract sorted values\n const first = vertexDataList.splice(0, 1)[0];\n root = first.vertexData;\n transform = first.transform;\n others = vertexDataList.map((v) => v.vertexData);\n vertexDatas = vertexDataList;\n this.materialInfos = materialInfos;\n }\n // Merge geometries\n const totalIndices = others.reduce((indexSum, vertexData) => { var _a, _b; return indexSum + ((_b = (_a = vertexData.indices) === null || _a === void 0 ? void 0 : _a.length) !== null && _b !== void 0 ? _b : 0); }, (_b = (_a = root.indices) === null || _a === void 0 ? void 0 : _a.length) !== null && _b !== void 0 ? _b : 0);\n const sliceIndices = forceCloneIndices || others.some((vertexData) => vertexData.indices === root.indices);\n let indices = sliceIndices ? (_c = root.indices) === null || _c === void 0 ? void 0 : _c.slice() : root.indices;\n if (totalIndices > 0) {\n let indicesOffset = (_d = indices === null || indices === void 0 ? void 0 : indices.length) !== null && _d !== void 0 ? _d : 0;\n if (!indices) {\n indices = new Array(totalIndices);\n }\n if (indices.length !== totalIndices) {\n if (Array.isArray(indices)) {\n indices.length = totalIndices;\n }\n else {\n const temp = use32BitsIndices || indices instanceof Uint32Array ? new Uint32Array(totalIndices) : new Uint16Array(totalIndices);\n temp.set(indices);\n indices = temp;\n }\n if (transform && transform.determinant() < 0) {\n VertexData._FlipFaces(indices, 0, indicesOffset);\n }\n }\n let positionsOffset = root.positions ? root.positions.length / 3 : 0;\n for (const { vertexData: other, transform } of vertexDatas) {\n if (other.indices) {\n for (let index = 0; index < other.indices.length; index++) {\n indices[indicesOffset + index] = other.indices[index] + positionsOffset;\n }\n if (transform && transform.determinant() < 0) {\n VertexData._FlipFaces(indices, indicesOffset, other.indices.length);\n }\n // The call to _validate already checked for positions\n positionsOffset += other.positions.length / 3;\n indicesOffset += other.indices.length;\n if (isAsync) {\n yield;\n }\n }\n }\n }\n this.indices = indices;\n this.positions = VertexData._MergeElement(VertexBuffer.PositionKind, root.positions, transform, vertexDatas.map((other) => [other.vertexData.positions, other.transform]));\n if (isAsync) {\n yield;\n }\n if (root.normals) {\n this.normals = VertexData._MergeElement(VertexBuffer.NormalKind, root.normals, transform, vertexDatas.map((other) => [other.vertexData.normals, other.transform]));\n if (isAsync) {\n yield;\n }\n }\n if (root.tangents) {\n this.tangents = VertexData._MergeElement(VertexBuffer.TangentKind, root.tangents, transform, vertexDatas.map((other) => [other.vertexData.tangents, other.transform]));\n if (isAsync) {\n yield;\n }\n }\n if (root.uvs) {\n this.uvs = VertexData._MergeElement(VertexBuffer.UVKind, root.uvs, transform, vertexDatas.map((other) => [other.vertexData.uvs, other.transform]));\n if (isAsync) {\n yield;\n }\n }\n if (root.uvs2) {\n this.uvs2 = VertexData._MergeElement(VertexBuffer.UV2Kind, root.uvs2, transform, vertexDatas.map((other) => [other.vertexData.uvs2, other.transform]));\n if (isAsync) {\n yield;\n }\n }\n if (root.uvs3) {\n this.uvs3 = VertexData._MergeElement(VertexBuffer.UV3Kind, root.uvs3, transform, vertexDatas.map((other) => [other.vertexData.uvs3, other.transform]));\n if (isAsync) {\n yield;\n }\n }\n if (root.uvs4) {\n this.uvs4 = VertexData._MergeElement(VertexBuffer.UV4Kind, root.uvs4, transform, vertexDatas.map((other) => [other.vertexData.uvs4, other.transform]));\n if (isAsync) {\n yield;\n }\n }\n if (root.uvs5) {\n this.uvs5 = VertexData._MergeElement(VertexBuffer.UV5Kind, root.uvs5, transform, vertexDatas.map((other) => [other.vertexData.uvs5, other.transform]));\n if (isAsync) {\n yield;\n }\n }\n if (root.uvs6) {\n this.uvs6 = VertexData._MergeElement(VertexBuffer.UV6Kind, root.uvs6, transform, vertexDatas.map((other) => [other.vertexData.uvs6, other.transform]));\n if (isAsync) {\n yield;\n }\n }\n if (root.colors) {\n this.colors = VertexData._MergeElement(VertexBuffer.ColorKind, root.colors, transform, vertexDatas.map((other) => [other.vertexData.colors, other.transform]));\n if (isAsync) {\n yield;\n }\n }\n if (root.matricesIndices) {\n this.matricesIndices = VertexData._MergeElement(VertexBuffer.MatricesIndicesKind, root.matricesIndices, transform, vertexDatas.map((other) => [other.vertexData.matricesIndices, other.transform]));\n if (isAsync) {\n yield;\n }\n }\n if (root.matricesWeights) {\n this.matricesWeights = VertexData._MergeElement(VertexBuffer.MatricesWeightsKind, root.matricesWeights, transform, vertexDatas.map((other) => [other.vertexData.matricesWeights, other.transform]));\n if (isAsync) {\n yield;\n }\n }\n if (root.matricesIndicesExtra) {\n this.matricesIndicesExtra = VertexData._MergeElement(VertexBuffer.MatricesIndicesExtraKind, root.matricesIndicesExtra, transform, vertexDatas.map((other) => [other.vertexData.matricesIndicesExtra, other.transform]));\n if (isAsync) {\n yield;\n }\n }\n if (root.matricesWeightsExtra) {\n this.matricesWeightsExtra = VertexData._MergeElement(VertexBuffer.MatricesWeightsExtraKind, root.matricesWeightsExtra, transform, vertexDatas.map((other) => [other.vertexData.matricesWeightsExtra, other.transform]));\n }\n return this;\n }\n static _MergeElement(kind, source, transform, others) {\n const nonNullOthers = others.filter((other) => other[0] !== null && other[0] !== undefined);\n // If there is no source to copy and no other non-null sources then skip this element.\n if (!source && nonNullOthers.length == 0) {\n return source;\n }\n if (!source) {\n return this._MergeElement(kind, nonNullOthers[0][0], nonNullOthers[0][1], nonNullOthers.slice(1));\n }\n const len = nonNullOthers.reduce((sumLen, elements) => sumLen + elements[0].length, source.length);\n const transformRange = kind === VertexBuffer.PositionKind\n ? VertexData._TransformVector3Coordinates\n : kind === VertexBuffer.NormalKind\n ? VertexData._TransformVector3Normals\n : kind === VertexBuffer.TangentKind\n ? VertexData._TransformVector4Normals\n : () => { };\n if (source instanceof Float32Array) {\n // use non-loop method when the source is Float32Array\n const ret32 = new Float32Array(len);\n ret32.set(source);\n transform && transformRange(ret32, transform, 0, source.length);\n let offset = source.length;\n for (const [vertexData, transform] of nonNullOthers) {\n ret32.set(vertexData, offset);\n transform && transformRange(ret32, transform, offset, vertexData.length);\n offset += vertexData.length;\n }\n return ret32;\n }\n else {\n // don't use concat as it is super slow, just loop for other cases\n const ret = new Array(len);\n for (let i = 0; i < source.length; i++) {\n ret[i] = source[i];\n }\n transform && transformRange(ret, transform, 0, source.length);\n let offset = source.length;\n for (const [vertexData, transform] of nonNullOthers) {\n for (let i = 0; i < vertexData.length; i++) {\n ret[offset + i] = vertexData[i];\n }\n transform && transformRange(ret, transform, offset, vertexData.length);\n offset += vertexData.length;\n }\n return ret;\n }\n }\n _validate() {\n if (!this.positions) {\n throw new RuntimeError(\"Positions are required\", ErrorCodes.MeshInvalidPositionsError);\n }\n const getElementCount = (kind, values) => {\n const stride = VertexBuffer.DeduceStride(kind);\n if (values.length % stride !== 0) {\n throw new Error(\"The \" + kind + \"s array count must be a multiple of \" + stride);\n }\n return values.length / stride;\n };\n const positionsElementCount = getElementCount(VertexBuffer.PositionKind, this.positions);\n const validateElementCount = (kind, values) => {\n const elementCount = getElementCount(kind, values);\n if (elementCount !== positionsElementCount) {\n throw new Error(\"The \" + kind + \"s element count (\" + elementCount + \") does not match the positions count (\" + positionsElementCount + \")\");\n }\n };\n if (this.normals) {\n validateElementCount(VertexBuffer.NormalKind, this.normals);\n }\n if (this.tangents) {\n validateElementCount(VertexBuffer.TangentKind, this.tangents);\n }\n if (this.uvs) {\n validateElementCount(VertexBuffer.UVKind, this.uvs);\n }\n if (this.uvs2) {\n validateElementCount(VertexBuffer.UV2Kind, this.uvs2);\n }\n if (this.uvs3) {\n validateElementCount(VertexBuffer.UV3Kind, this.uvs3);\n }\n if (this.uvs4) {\n validateElementCount(VertexBuffer.UV4Kind, this.uvs4);\n }\n if (this.uvs5) {\n validateElementCount(VertexBuffer.UV5Kind, this.uvs5);\n }\n if (this.uvs6) {\n validateElementCount(VertexBuffer.UV6Kind, this.uvs6);\n }\n if (this.colors) {\n validateElementCount(VertexBuffer.ColorKind, this.colors);\n }\n if (this.matricesIndices) {\n validateElementCount(VertexBuffer.MatricesIndicesKind, this.matricesIndices);\n }\n if (this.matricesWeights) {\n validateElementCount(VertexBuffer.MatricesWeightsKind, this.matricesWeights);\n }\n if (this.matricesIndicesExtra) {\n validateElementCount(VertexBuffer.MatricesIndicesExtraKind, this.matricesIndicesExtra);\n }\n if (this.matricesWeightsExtra) {\n validateElementCount(VertexBuffer.MatricesWeightsExtraKind, this.matricesWeightsExtra);\n }\n }\n /**\n * Clone the current vertex data\n * @returns a copy of the current data\n */\n clone() {\n const serializationObject = this.serialize();\n return VertexData.Parse(serializationObject);\n }\n /**\n * Serializes the VertexData\n * @returns a serialized object\n */\n serialize() {\n const serializationObject = {};\n if (this.positions) {\n serializationObject.positions = Array.from(this.positions);\n }\n if (this.normals) {\n serializationObject.normals = Array.from(this.normals);\n }\n if (this.tangents) {\n serializationObject.tangents = Array.from(this.tangents);\n }\n if (this.uvs) {\n serializationObject.uvs = Array.from(this.uvs);\n }\n if (this.uvs2) {\n serializationObject.uvs2 = Array.from(this.uvs2);\n }\n if (this.uvs3) {\n serializationObject.uvs3 = Array.from(this.uvs3);\n }\n if (this.uvs4) {\n serializationObject.uvs4 = Array.from(this.uvs4);\n }\n if (this.uvs5) {\n serializationObject.uvs5 = Array.from(this.uvs5);\n }\n if (this.uvs6) {\n serializationObject.uvs6 = Array.from(this.uvs6);\n }\n if (this.colors) {\n serializationObject.colors = Array.from(this.colors);\n }\n if (this.matricesIndices) {\n serializationObject.matricesIndices = Array.from(this.matricesIndices);\n serializationObject.matricesIndices._isExpanded = true;\n }\n if (this.matricesWeights) {\n serializationObject.matricesWeights = Array.from(this.matricesWeights);\n }\n if (this.matricesIndicesExtra) {\n serializationObject.matricesIndicesExtra = Array.from(this.matricesIndicesExtra);\n serializationObject.matricesIndicesExtra._isExpanded = true;\n }\n if (this.matricesWeightsExtra) {\n serializationObject.matricesWeightsExtra = Array.from(this.matricesWeightsExtra);\n }\n serializationObject.indices = Array.from(this.indices);\n if (this.materialInfos) {\n serializationObject.materialInfos = [];\n for (const materialInfo of this.materialInfos) {\n const materialInfoSerializationObject = {\n indexStart: materialInfo.indexStart,\n indexCount: materialInfo.indexCount,\n materialIndex: materialInfo.materialIndex,\n verticesStart: materialInfo.verticesStart,\n verticesCount: materialInfo.verticesCount,\n };\n serializationObject.materialInfos.push(materialInfoSerializationObject);\n }\n }\n return serializationObject;\n }\n // Statics\n /**\n * Extracts the vertexData from a mesh\n * @param mesh the mesh from which to extract the VertexData\n * @param copyWhenShared defines if the VertexData must be cloned when shared between multiple meshes, optional, default false\n * @param forceCopy indicating that the VertexData must be cloned, optional, default false\n * @returns the object VertexData associated to the passed mesh\n */\n static ExtractFromMesh(mesh, copyWhenShared, forceCopy) {\n return VertexData._ExtractFrom(mesh, copyWhenShared, forceCopy);\n }\n /**\n * Extracts the vertexData from the geometry\n * @param geometry the geometry from which to extract the VertexData\n * @param copyWhenShared defines if the VertexData must be cloned when the geometry is shared between multiple meshes, optional, default false\n * @param forceCopy indicating that the VertexData must be cloned, optional, default false\n * @returns the object VertexData associated to the passed mesh\n */\n static ExtractFromGeometry(geometry, copyWhenShared, forceCopy) {\n return VertexData._ExtractFrom(geometry, copyWhenShared, forceCopy);\n }\n static _ExtractFrom(meshOrGeometry, copyWhenShared, forceCopy) {\n const result = new VertexData();\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.PositionKind)) {\n result.positions = meshOrGeometry.getVerticesData(VertexBuffer.PositionKind, copyWhenShared, forceCopy);\n }\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.NormalKind)) {\n result.normals = meshOrGeometry.getVerticesData(VertexBuffer.NormalKind, copyWhenShared, forceCopy);\n }\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.TangentKind)) {\n result.tangents = meshOrGeometry.getVerticesData(VertexBuffer.TangentKind, copyWhenShared, forceCopy);\n }\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.UVKind)) {\n result.uvs = meshOrGeometry.getVerticesData(VertexBuffer.UVKind, copyWhenShared, forceCopy);\n }\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.UV2Kind)) {\n result.uvs2 = meshOrGeometry.getVerticesData(VertexBuffer.UV2Kind, copyWhenShared, forceCopy);\n }\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.UV3Kind)) {\n result.uvs3 = meshOrGeometry.getVerticesData(VertexBuffer.UV3Kind, copyWhenShared, forceCopy);\n }\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.UV4Kind)) {\n result.uvs4 = meshOrGeometry.getVerticesData(VertexBuffer.UV4Kind, copyWhenShared, forceCopy);\n }\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.UV5Kind)) {\n result.uvs5 = meshOrGeometry.getVerticesData(VertexBuffer.UV5Kind, copyWhenShared, forceCopy);\n }\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.UV6Kind)) {\n result.uvs6 = meshOrGeometry.getVerticesData(VertexBuffer.UV6Kind, copyWhenShared, forceCopy);\n }\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.ColorKind)) {\n result.colors = meshOrGeometry.getVerticesData(VertexBuffer.ColorKind, copyWhenShared, forceCopy);\n }\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.MatricesIndicesKind)) {\n result.matricesIndices = meshOrGeometry.getVerticesData(VertexBuffer.MatricesIndicesKind, copyWhenShared, forceCopy);\n }\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.MatricesWeightsKind)) {\n result.matricesWeights = meshOrGeometry.getVerticesData(VertexBuffer.MatricesWeightsKind, copyWhenShared, forceCopy);\n }\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.MatricesIndicesExtraKind)) {\n result.matricesIndicesExtra = meshOrGeometry.getVerticesData(VertexBuffer.MatricesIndicesExtraKind, copyWhenShared, forceCopy);\n }\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.MatricesWeightsExtraKind)) {\n result.matricesWeightsExtra = meshOrGeometry.getVerticesData(VertexBuffer.MatricesWeightsExtraKind, copyWhenShared, forceCopy);\n }\n result.indices = meshOrGeometry.getIndices(copyWhenShared, forceCopy);\n return result;\n }\n /**\n * Creates the VertexData for a Ribbon\n * @param options an object used to set the following optional parameters for the ribbon, required but can be empty\n * * pathArray array of paths, each of which an array of successive Vector3\n * * closeArray creates a seam between the first and the last paths of the pathArray, optional, default false\n * * closePath creates a seam between the first and the last points of each path of the path array, optional, default false\n * * offset a positive integer, only used when pathArray contains a single path (offset = 10 means the point 1 is joined to the point 11), default rounded half size of the pathArray length\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * * invertUV swaps in the U and V coordinates when applying a texture, optional, default false\n * * uvs a linear array, of length 2 * number of vertices, of custom UV values, optional\n * * colors a linear array, of length 4 * number of vertices, of custom color values, optional\n * @param options.pathArray\n * @param options.closeArray\n * @param options.closePath\n * @param options.offset\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param options.invertUV\n * @param options.uvs\n * @param options.colors\n * @returns the VertexData of the ribbon\n * @deprecated use CreateRibbonVertexData instead\n */\n static CreateRibbon(options) {\n throw _WarnImport(\"ribbonBuilder\");\n }\n /**\n * Creates the VertexData for a box\n * @param options an object used to set the following optional parameters for the box, required but can be empty\n * * size sets the width, height and depth of the box to the value of size, optional default 1\n * * width sets the width (x direction) of the box, overwrites the width set by size, optional, default size\n * * height sets the height (y direction) of the box, overwrites the height set by size, optional, default size\n * * depth sets the depth (z direction) of the box, overwrites the depth set by size, optional, default size\n * * faceUV an array of 6 Vector4 elements used to set different images to each box side\n * * faceColors an array of 6 Color3 elements used to set different colors to each box side\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.size\n * @param options.width\n * @param options.height\n * @param options.depth\n * @param options.faceUV\n * @param options.faceColors\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @returns the VertexData of the box\n * @deprecated Please use CreateBoxVertexData from the BoxBuilder file instead\n */\n static CreateBox(options) {\n throw _WarnImport(\"boxBuilder\");\n }\n /**\n * Creates the VertexData for a tiled box\n * @param options an object used to set the following optional parameters for the box, required but can be empty\n * * faceTiles sets the pattern, tile size and number of tiles for a face\n * * faceUV an array of 6 Vector4 elements used to set different images to each box side\n * * faceColors an array of 6 Color3 elements used to set different colors to each box side\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * @param options.pattern\n * @param options.width\n * @param options.height\n * @param options.depth\n * @param options.tileSize\n * @param options.tileWidth\n * @param options.tileHeight\n * @param options.alignHorizontal\n * @param options.alignVertical\n * @param options.faceUV\n * @param options.faceColors\n * @param options.sideOrientation\n * @returns the VertexData of the box\n * @deprecated Please use CreateTiledBoxVertexData instead\n */\n static CreateTiledBox(options) {\n throw _WarnImport(\"tiledBoxBuilder\");\n }\n /**\n * Creates the VertexData for a tiled plane\n * @param options an object used to set the following optional parameters for the box, required but can be empty\n * * pattern a limited pattern arrangement depending on the number\n * * tileSize sets the width, height and depth of the tile to the value of size, optional default 1\n * * tileWidth sets the width (x direction) of the tile, overwrites the width set by size, optional, default size\n * * tileHeight sets the height (y direction) of the tile, overwrites the height set by size, optional, default size\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.pattern\n * @param options.tileSize\n * @param options.tileWidth\n * @param options.tileHeight\n * @param options.size\n * @param options.width\n * @param options.height\n * @param options.alignHorizontal\n * @param options.alignVertical\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @returns the VertexData of the tiled plane\n * @deprecated use CreateTiledPlaneVertexData instead\n */\n static CreateTiledPlane(options) {\n throw _WarnImport(\"tiledPlaneBuilder\");\n }\n /**\n * Creates the VertexData for an ellipsoid, defaults to a sphere\n * @param options an object used to set the following optional parameters for the box, required but can be empty\n * * segments sets the number of horizontal strips optional, default 32\n * * diameter sets the axes dimensions, diameterX, diameterY and diameterZ to the value of diameter, optional default 1\n * * diameterX sets the diameterX (x direction) of the ellipsoid, overwrites the diameterX set by diameter, optional, default diameter\n * * diameterY sets the diameterY (y direction) of the ellipsoid, overwrites the diameterY set by diameter, optional, default diameter\n * * diameterZ sets the diameterZ (z direction) of the ellipsoid, overwrites the diameterZ set by diameter, optional, default diameter\n * * arc a number from 0 to 1, to create an unclosed ellipsoid based on the fraction of the circumference (latitude) given by the arc value, optional, default 1\n * * slice a number from 0 to 1, to create an unclosed ellipsoid based on the fraction of the height (latitude) given by the arc value, optional, default 1\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.segments\n * @param options.diameter\n * @param options.diameterX\n * @param options.diameterY\n * @param options.diameterZ\n * @param options.arc\n * @param options.slice\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @returns the VertexData of the ellipsoid\n * @deprecated use CreateSphereVertexData instead\n */\n static CreateSphere(options) {\n throw _WarnImport(\"sphereBuilder\");\n }\n /**\n * Creates the VertexData for a cylinder, cone or prism\n * @param options an object used to set the following optional parameters for the box, required but can be empty\n * * height sets the height (y direction) of the cylinder, optional, default 2\n * * diameterTop sets the diameter of the top of the cone, overwrites diameter, optional, default diameter\n * * diameterBottom sets the diameter of the bottom of the cone, overwrites diameter, optional, default diameter\n * * diameter sets the diameter of the top and bottom of the cone, optional default 1\n * * tessellation the number of prism sides, 3 for a triangular prism, optional, default 24\n * * subdivisions` the number of rings along the cylinder height, optional, default 1\n * * arc a number from 0 to 1, to create an unclosed cylinder based on the fraction of the circumference given by the arc value, optional, default 1\n * * faceColors an array of Color3 elements used to set different colors to the top, rings and bottom respectively\n * * faceUV an array of Vector4 elements used to set different images to the top, rings and bottom respectively\n * * hasRings when true makes each subdivision independently treated as a face for faceUV and faceColors, optional, default false\n * * enclose when true closes an open cylinder by adding extra flat faces between the height axis and vertical edges, think cut cake\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.height\n * @param options.diameterTop\n * @param options.diameterBottom\n * @param options.diameter\n * @param options.tessellation\n * @param options.subdivisions\n * @param options.arc\n * @param options.faceColors\n * @param options.faceUV\n * @param options.hasRings\n * @param options.enclose\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @returns the VertexData of the cylinder, cone or prism\n * @deprecated please use CreateCylinderVertexData instead\n */\n static CreateCylinder(options) {\n throw _WarnImport(\"cylinderBuilder\");\n }\n /**\n * Creates the VertexData for a torus\n * @param options an object used to set the following optional parameters for the box, required but can be empty\n * * diameter the diameter of the torus, optional default 1\n * * thickness the diameter of the tube forming the torus, optional default 0.5\n * * tessellation the number of prism sides, 3 for a triangular prism, optional, default 24\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.diameter\n * @param options.thickness\n * @param options.tessellation\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @returns the VertexData of the torus\n * @deprecated use CreateTorusVertexData instead\n */\n static CreateTorus(options) {\n throw _WarnImport(\"torusBuilder\");\n }\n /**\n * Creates the VertexData of the LineSystem\n * @param options an object used to set the following optional parameters for the LineSystem, required but can be empty\n * - lines an array of lines, each line being an array of successive Vector3\n * - colors an array of line colors, each of the line colors being an array of successive Color4, one per line point\n * @param options.lines\n * @param options.colors\n * @returns the VertexData of the LineSystem\n * @deprecated use CreateLineSystemVertexData instead\n */\n static CreateLineSystem(options) {\n throw _WarnImport(\"linesBuilder\");\n }\n /**\n * Create the VertexData for a DashedLines\n * @param options an object used to set the following optional parameters for the DashedLines, required but can be empty\n * - points an array successive Vector3\n * - dashSize the size of the dashes relative to the dash number, optional, default 3\n * - gapSize the size of the gap between two successive dashes relative to the dash number, optional, default 1\n * - dashNb the intended total number of dashes, optional, default 200\n * @param options.points\n * @param options.dashSize\n * @param options.gapSize\n * @param options.dashNb\n * @returns the VertexData for the DashedLines\n * @deprecated use CreateDashedLinesVertexData instead\n */\n static CreateDashedLines(options) {\n throw _WarnImport(\"linesBuilder\");\n }\n /**\n * Creates the VertexData for a Ground\n * @param options an object used to set the following optional parameters for the Ground, required but can be empty\n * - width the width (x direction) of the ground, optional, default 1\n * - height the height (z direction) of the ground, optional, default 1\n * - subdivisions the number of subdivisions per side, optional, default 1\n * @param options.width\n * @param options.height\n * @param options.subdivisions\n * @param options.subdivisionsX\n * @param options.subdivisionsY\n * @returns the VertexData of the Ground\n * @deprecated Please use CreateGroundVertexData instead\n */\n static CreateGround(options) {\n throw _WarnImport(\"groundBuilder\");\n }\n /**\n * Creates the VertexData for a TiledGround by subdividing the ground into tiles\n * @param options an object used to set the following optional parameters for the Ground, required but can be empty\n * * xmin the ground minimum X coordinate, optional, default -1\n * * zmin the ground minimum Z coordinate, optional, default -1\n * * xmax the ground maximum X coordinate, optional, default 1\n * * zmax the ground maximum Z coordinate, optional, default 1\n * * subdivisions a javascript object {w: positive integer, h: positive integer}, `w` and `h` are the numbers of subdivisions on the ground width and height creating 'tiles', default {w: 6, h: 6}\n * * precision a javascript object {w: positive integer, h: positive integer}, `w` and `h` are the numbers of subdivisions on the tile width and height, default {w: 2, h: 2}\n * @param options.xmin\n * @param options.zmin\n * @param options.xmax\n * @param options.zmax\n * @param options.subdivisions\n * @param options.subdivisions.w\n * @param options.subdivisions.h\n * @param options.precision\n * @param options.precision.w\n * @param options.precision.h\n * @returns the VertexData of the TiledGround\n * @deprecated use CreateTiledGroundVertexData instead\n */\n static CreateTiledGround(options) {\n throw _WarnImport(\"groundBuilder\");\n }\n /**\n * Creates the VertexData of the Ground designed from a heightmap\n * @param options an object used to set the following parameters for the Ground, required and provided by CreateGroundFromHeightMap\n * * width the width (x direction) of the ground\n * * height the height (z direction) of the ground\n * * subdivisions the number of subdivisions per side\n * * minHeight the minimum altitude on the ground, optional, default 0\n * * maxHeight the maximum altitude on the ground, optional default 1\n * * colorFilter the filter to apply to the image pixel colors to compute the height, optional Color3, default (0.3, 0.59, 0.11)\n * * buffer the array holding the image color data\n * * bufferWidth the width of image\n * * bufferHeight the height of image\n * * alphaFilter Remove any data where the alpha channel is below this value, defaults 0 (all data visible)\n * @param options.width\n * @param options.height\n * @param options.subdivisions\n * @param options.minHeight\n * @param options.maxHeight\n * @param options.colorFilter\n * @param options.buffer\n * @param options.bufferWidth\n * @param options.bufferHeight\n * @param options.alphaFilter\n * @returns the VertexData of the Ground designed from a heightmap\n * @deprecated use CreateGroundFromHeightMapVertexData instead\n */\n static CreateGroundFromHeightMap(options) {\n throw _WarnImport(\"groundBuilder\");\n }\n /**\n * Creates the VertexData for a Plane\n * @param options an object used to set the following optional parameters for the plane, required but can be empty\n * * size sets the width and height of the plane to the value of size, optional default 1\n * * width sets the width (x direction) of the plane, overwrites the width set by size, optional, default size\n * * height sets the height (y direction) of the plane, overwrites the height set by size, optional, default size\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.size\n * @param options.width\n * @param options.height\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @returns the VertexData of the box\n * @deprecated use CreatePlaneVertexData instead\n */\n static CreatePlane(options) {\n throw _WarnImport(\"planeBuilder\");\n }\n /**\n * Creates the VertexData of the Disc or regular Polygon\n * @param options an object used to set the following optional parameters for the disc, required but can be empty\n * * radius the radius of the disc, optional default 0.5\n * * tessellation the number of polygon sides, optional, default 64\n * * arc a number from 0 to 1, to create an unclosed polygon based on the fraction of the circumference given by the arc value, optional, default 1\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.radius\n * @param options.tessellation\n * @param options.arc\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @returns the VertexData of the box\n * @deprecated use CreateDiscVertexData instead\n */\n static CreateDisc(options) {\n throw _WarnImport(\"discBuilder\");\n }\n /**\n * Creates the VertexData for an irregular Polygon in the XoZ plane using a mesh built by polygonTriangulation.build()\n * All parameters are provided by CreatePolygon as needed\n * @param polygon a mesh built from polygonTriangulation.build()\n * @param sideOrientation takes the values Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * @param fUV an array of Vector4 elements used to set different images to the top, rings and bottom respectively\n * @param fColors an array of Color3 elements used to set different colors to the top, rings and bottom respectively\n * @param frontUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * @param backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param wrap a boolean, default false, when true and fUVs used texture is wrapped around all sides, when false texture is applied side\n * @returns the VertexData of the Polygon\n * @deprecated use CreatePolygonVertexData instead\n */\n static CreatePolygon(polygon, sideOrientation, fUV, fColors, frontUVs, backUVs, wrap) {\n throw _WarnImport(\"polygonBuilder\");\n }\n /**\n * Creates the VertexData of the IcoSphere\n * @param options an object used to set the following optional parameters for the IcoSphere, required but can be empty\n * * radius the radius of the IcoSphere, optional default 1\n * * radiusX allows stretching in the x direction, optional, default radius\n * * radiusY allows stretching in the y direction, optional, default radius\n * * radiusZ allows stretching in the z direction, optional, default radius\n * * flat when true creates a flat shaded mesh, optional, default true\n * * subdivisions increasing the subdivisions increases the number of faces, optional, default 4\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.radius\n * @param options.radiusX\n * @param options.radiusY\n * @param options.radiusZ\n * @param options.flat\n * @param options.subdivisions\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @returns the VertexData of the IcoSphere\n * @deprecated use CreateIcoSphereVertexData instead\n */\n static CreateIcoSphere(options) {\n throw _WarnImport(\"icoSphereBuilder\");\n }\n // inspired from // http://stemkoski.github.io/Three.js/Polyhedra.html\n /**\n * Creates the VertexData for a Polyhedron\n * @param options an object used to set the following optional parameters for the polyhedron, required but can be empty\n * * type provided types are:\n * * 0 : Tetrahedron, 1 : Octahedron, 2 : Dodecahedron, 3 : Icosahedron, 4 : Rhombicuboctahedron, 5 : Triangular Prism, 6 : Pentagonal Prism, 7 : Hexagonal Prism, 8 : Square Pyramid (J1)\n * * 9 : Pentagonal Pyramid (J2), 10 : Triangular Dipyramid (J12), 11 : Pentagonal Dipyramid (J13), 12 : Elongated Square Dipyramid (J15), 13 : Elongated Pentagonal Dipyramid (J16), 14 : Elongated Pentagonal Cupola (J20)\n * * size the size of the IcoSphere, optional default 1\n * * sizeX allows stretching in the x direction, optional, default size\n * * sizeY allows stretching in the y direction, optional, default size\n * * sizeZ allows stretching in the z direction, optional, default size\n * * custom a number that overwrites the type to create from an extended set of polyhedron from https://www.babylonjs-playground.com/#21QRSK#15 with minimised editor\n * * faceUV an array of Vector4 elements used to set different images to the top, rings and bottom respectively\n * * faceColors an array of Color3 elements used to set different colors to the top, rings and bottom respectively\n * * flat when true creates a flat shaded mesh, optional, default true\n * * subdivisions increasing the subdivisions increases the number of faces, optional, default 4\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.type\n * @param options.size\n * @param options.sizeX\n * @param options.sizeY\n * @param options.sizeZ\n * @param options.custom\n * @param options.faceUV\n * @param options.faceColors\n * @param options.flat\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @returns the VertexData of the Polyhedron\n * @deprecated use CreatePolyhedronVertexData instead\n */\n static CreatePolyhedron(options) {\n throw _WarnImport(\"polyhedronBuilder\");\n }\n /**\n * Creates the VertexData for a Capsule, inspired from https://github.com/maximeq/three-js-capsule-geometry/blob/master/src/CapsuleBufferGeometry.js\n * @param options an object used to set the following optional parameters for the capsule, required but can be empty\n * @returns the VertexData of the Capsule\n * @deprecated Please use CreateCapsuleVertexData from the capsuleBuilder file instead\n */\n static CreateCapsule(options = {\n orientation: Vector3.Up(),\n subdivisions: 2,\n tessellation: 16,\n height: 1,\n radius: 0.25,\n capSubdivisions: 6,\n }) {\n throw _WarnImport(\"capsuleBuilder\");\n }\n // based on http://code.google.com/p/away3d/source/browse/trunk/fp10/Away3D/src/away3d/primitives/TorusKnot.as?spec=svn2473&r=2473\n /**\n * Creates the VertexData for a TorusKnot\n * @param options an object used to set the following optional parameters for the TorusKnot, required but can be empty\n * * radius the radius of the torus knot, optional, default 2\n * * tube the thickness of the tube, optional, default 0.5\n * * radialSegments the number of sides on each tube segments, optional, default 32\n * * tubularSegments the number of tubes to decompose the knot into, optional, default 32\n * * p the number of windings around the z axis, optional, default 2\n * * q the number of windings around the x axis, optional, default 3\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.radius\n * @param options.tube\n * @param options.radialSegments\n * @param options.tubularSegments\n * @param options.p\n * @param options.q\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @returns the VertexData of the Torus Knot\n * @deprecated use CreateTorusKnotVertexData instead\n */\n static CreateTorusKnot(options) {\n throw _WarnImport(\"torusKnotBuilder\");\n }\n // Tools\n /**\n * Compute normals for given positions and indices\n * @param positions an array of vertex positions, [...., x, y, z, ......]\n * @param indices an array of indices in groups of three for each triangular facet, [...., i, j, k, ......]\n * @param normals an array of vertex normals, [...., x, y, z, ......]\n * @param options an object used to set the following optional parameters for the TorusKnot, optional\n * * facetNormals : optional array of facet normals (vector3)\n * * facetPositions : optional array of facet positions (vector3)\n * * facetPartitioning : optional partitioning array. facetPositions is required for facetPartitioning computation\n * * ratio : optional partitioning ratio / bounding box, required for facetPartitioning computation\n * * bInfo : optional bounding info, required for facetPartitioning computation\n * * bbSize : optional bounding box size data, required for facetPartitioning computation\n * * subDiv : optional partitioning data about subdivisions on each axis (int), required for facetPartitioning computation\n * * useRightHandedSystem: optional boolean to for right handed system computation\n * * depthSort : optional boolean to enable the facet depth sort computation\n * * distanceTo : optional Vector3 to compute the facet depth from this location\n * * depthSortedFacets : optional array of depthSortedFacets to store the facet distances from the reference location\n * @param options.facetNormals\n * @param options.facetPositions\n * @param options.facetPartitioning\n * @param options.ratio\n * @param options.bInfo\n * @param options.bbSize\n * @param options.subDiv\n * @param options.useRightHandedSystem\n * @param options.depthSort\n * @param options.distanceTo\n * @param options.depthSortedFacets\n */\n static ComputeNormals(positions, indices, normals, options) {\n // temporary scalar variables\n let index = 0; // facet index\n let p1p2x = 0.0; // p1p2 vector x coordinate\n let p1p2y = 0.0; // p1p2 vector y coordinate\n let p1p2z = 0.0; // p1p2 vector z coordinate\n let p3p2x = 0.0; // p3p2 vector x coordinate\n let p3p2y = 0.0; // p3p2 vector y coordinate\n let p3p2z = 0.0; // p3p2 vector z coordinate\n let faceNormalx = 0.0; // facet normal x coordinate\n let faceNormaly = 0.0; // facet normal y coordinate\n let faceNormalz = 0.0; // facet normal z coordinate\n let length = 0.0; // facet normal length before normalization\n let v1x = 0; // vector1 x index in the positions array\n let v1y = 0; // vector1 y index in the positions array\n let v1z = 0; // vector1 z index in the positions array\n let v2x = 0; // vector2 x index in the positions array\n let v2y = 0; // vector2 y index in the positions array\n let v2z = 0; // vector2 z index in the positions array\n let v3x = 0; // vector3 x index in the positions array\n let v3y = 0; // vector3 y index in the positions array\n let v3z = 0; // vector3 z index in the positions array\n let computeFacetNormals = false;\n let computeFacetPositions = false;\n let computeFacetPartitioning = false;\n let computeDepthSort = false;\n let faceNormalSign = 1;\n let ratio = 0;\n let distanceTo = null;\n if (options) {\n computeFacetNormals = options.facetNormals ? true : false;\n computeFacetPositions = options.facetPositions ? true : false;\n computeFacetPartitioning = options.facetPartitioning ? true : false;\n faceNormalSign = options.useRightHandedSystem === true ? -1 : 1;\n ratio = options.ratio || 0;\n computeDepthSort = options.depthSort ? true : false;\n distanceTo = options.distanceTo;\n if (computeDepthSort) {\n if (distanceTo === undefined) {\n distanceTo = Vector3.Zero();\n }\n }\n }\n // facetPartitioning reinit if needed\n let xSubRatio = 0;\n let ySubRatio = 0;\n let zSubRatio = 0;\n let subSq = 0;\n if (computeFacetPartitioning && options && options.bbSize) {\n //let bbSizeMax = options.bbSize.x > options.bbSize.y ? options.bbSize.x : options.bbSize.y;\n //bbSizeMax = bbSizeMax > options.bbSize.z ? bbSizeMax : options.bbSize.z;\n xSubRatio = (options.subDiv.X * ratio) / options.bbSize.x;\n ySubRatio = (options.subDiv.Y * ratio) / options.bbSize.y;\n zSubRatio = (options.subDiv.Z * ratio) / options.bbSize.z;\n subSq = options.subDiv.max * options.subDiv.max;\n options.facetPartitioning.length = 0;\n }\n // reset the normals\n for (index = 0; index < positions.length; index++) {\n normals[index] = 0.0;\n }\n // Loop : 1 indice triplet = 1 facet\n const nbFaces = (indices.length / 3) | 0;\n for (index = 0; index < nbFaces; index++) {\n // get the indexes of the coordinates of each vertex of the facet\n v1x = indices[index * 3] * 3;\n v1y = v1x + 1;\n v1z = v1x + 2;\n v2x = indices[index * 3 + 1] * 3;\n v2y = v2x + 1;\n v2z = v2x + 2;\n v3x = indices[index * 3 + 2] * 3;\n v3y = v3x + 1;\n v3z = v3x + 2;\n p1p2x = positions[v1x] - positions[v2x]; // compute two vectors per facet : p1p2 and p3p2\n p1p2y = positions[v1y] - positions[v2y];\n p1p2z = positions[v1z] - positions[v2z];\n p3p2x = positions[v3x] - positions[v2x];\n p3p2y = positions[v3y] - positions[v2y];\n p3p2z = positions[v3z] - positions[v2z];\n // compute the face normal with the cross product\n faceNormalx = faceNormalSign * (p1p2y * p3p2z - p1p2z * p3p2y);\n faceNormaly = faceNormalSign * (p1p2z * p3p2x - p1p2x * p3p2z);\n faceNormalz = faceNormalSign * (p1p2x * p3p2y - p1p2y * p3p2x);\n // normalize this normal and store it in the array facetData\n length = Math.sqrt(faceNormalx * faceNormalx + faceNormaly * faceNormaly + faceNormalz * faceNormalz);\n length = length === 0 ? 1.0 : length;\n faceNormalx /= length;\n faceNormaly /= length;\n faceNormalz /= length;\n if (computeFacetNormals && options) {\n options.facetNormals[index].x = faceNormalx;\n options.facetNormals[index].y = faceNormaly;\n options.facetNormals[index].z = faceNormalz;\n }\n if (computeFacetPositions && options) {\n // compute and the facet barycenter coordinates in the array facetPositions\n options.facetPositions[index].x = (positions[v1x] + positions[v2x] + positions[v3x]) / 3.0;\n options.facetPositions[index].y = (positions[v1y] + positions[v2y] + positions[v3y]) / 3.0;\n options.facetPositions[index].z = (positions[v1z] + positions[v2z] + positions[v3z]) / 3.0;\n }\n if (computeFacetPartitioning && options) {\n // store the facet indexes in arrays in the main facetPartitioning array :\n // compute each facet vertex (+ facet barycenter) index in the partiniong array\n const ox = Math.floor((options.facetPositions[index].x - options.bInfo.minimum.x * ratio) * xSubRatio);\n const oy = Math.floor((options.facetPositions[index].y - options.bInfo.minimum.y * ratio) * ySubRatio);\n const oz = Math.floor((options.facetPositions[index].z - options.bInfo.minimum.z * ratio) * zSubRatio);\n const b1x = Math.floor((positions[v1x] - options.bInfo.minimum.x * ratio) * xSubRatio);\n const b1y = Math.floor((positions[v1y] - options.bInfo.minimum.y * ratio) * ySubRatio);\n const b1z = Math.floor((positions[v1z] - options.bInfo.minimum.z * ratio) * zSubRatio);\n const b2x = Math.floor((positions[v2x] - options.bInfo.minimum.x * ratio) * xSubRatio);\n const b2y = Math.floor((positions[v2y] - options.bInfo.minimum.y * ratio) * ySubRatio);\n const b2z = Math.floor((positions[v2z] - options.bInfo.minimum.z * ratio) * zSubRatio);\n const b3x = Math.floor((positions[v3x] - options.bInfo.minimum.x * ratio) * xSubRatio);\n const b3y = Math.floor((positions[v3y] - options.bInfo.minimum.y * ratio) * ySubRatio);\n const b3z = Math.floor((positions[v3z] - options.bInfo.minimum.z * ratio) * zSubRatio);\n const block_idx_v1 = b1x + options.subDiv.max * b1y + subSq * b1z;\n const block_idx_v2 = b2x + options.subDiv.max * b2y + subSq * b2z;\n const block_idx_v3 = b3x + options.subDiv.max * b3y + subSq * b3z;\n const block_idx_o = ox + options.subDiv.max * oy + subSq * oz;\n options.facetPartitioning[block_idx_o] = options.facetPartitioning[block_idx_o] ? options.facetPartitioning[block_idx_o] : new Array();\n options.facetPartitioning[block_idx_v1] = options.facetPartitioning[block_idx_v1] ? options.facetPartitioning[block_idx_v1] : new Array();\n options.facetPartitioning[block_idx_v2] = options.facetPartitioning[block_idx_v2] ? options.facetPartitioning[block_idx_v2] : new Array();\n options.facetPartitioning[block_idx_v3] = options.facetPartitioning[block_idx_v3] ? options.facetPartitioning[block_idx_v3] : new Array();\n // push each facet index in each block containing the vertex\n options.facetPartitioning[block_idx_v1].push(index);\n if (block_idx_v2 != block_idx_v1) {\n options.facetPartitioning[block_idx_v2].push(index);\n }\n if (!(block_idx_v3 == block_idx_v2 || block_idx_v3 == block_idx_v1)) {\n options.facetPartitioning[block_idx_v3].push(index);\n }\n if (!(block_idx_o == block_idx_v1 || block_idx_o == block_idx_v2 || block_idx_o == block_idx_v3)) {\n options.facetPartitioning[block_idx_o].push(index);\n }\n }\n if (computeDepthSort && options && options.facetPositions) {\n const dsf = options.depthSortedFacets[index];\n dsf.ind = index * 3;\n dsf.sqDistance = Vector3.DistanceSquared(options.facetPositions[index], distanceTo);\n }\n // compute the normals anyway\n normals[v1x] += faceNormalx; // accumulate all the normals per face\n normals[v1y] += faceNormaly;\n normals[v1z] += faceNormalz;\n normals[v2x] += faceNormalx;\n normals[v2y] += faceNormaly;\n normals[v2z] += faceNormalz;\n normals[v3x] += faceNormalx;\n normals[v3y] += faceNormaly;\n normals[v3z] += faceNormalz;\n }\n // last normalization of each normal\n for (index = 0; index < normals.length / 3; index++) {\n faceNormalx = normals[index * 3];\n faceNormaly = normals[index * 3 + 1];\n faceNormalz = normals[index * 3 + 2];\n length = Math.sqrt(faceNormalx * faceNormalx + faceNormaly * faceNormaly + faceNormalz * faceNormalz);\n length = length === 0 ? 1.0 : length;\n faceNormalx /= length;\n faceNormaly /= length;\n faceNormalz /= length;\n normals[index * 3] = faceNormalx;\n normals[index * 3 + 1] = faceNormaly;\n normals[index * 3 + 2] = faceNormalz;\n }\n }\n /**\n * @internal\n */\n static _ComputeSides(sideOrientation, positions, indices, normals, uvs, frontUVs, backUVs) {\n const li = indices.length;\n const ln = normals.length;\n let i;\n let n;\n sideOrientation = sideOrientation || VertexData.DEFAULTSIDE;\n switch (sideOrientation) {\n case VertexData.FRONTSIDE:\n // nothing changed\n break;\n case VertexData.BACKSIDE:\n // indices\n for (i = 0; i < li; i += 3) {\n const tmp = indices[i];\n indices[i] = indices[i + 2];\n indices[i + 2] = tmp;\n }\n // normals\n for (n = 0; n < ln; n++) {\n normals[n] = -normals[n];\n }\n break;\n case VertexData.DOUBLESIDE: {\n // positions\n const lp = positions.length;\n const l = lp / 3;\n for (let p = 0; p < lp; p++) {\n positions[lp + p] = positions[p];\n }\n // indices\n for (i = 0; i < li; i += 3) {\n indices[i + li] = indices[i + 2] + l;\n indices[i + 1 + li] = indices[i + 1] + l;\n indices[i + 2 + li] = indices[i] + l;\n }\n // normals\n for (n = 0; n < ln; n++) {\n normals[ln + n] = -normals[n];\n }\n // uvs\n const lu = uvs.length;\n let u = 0;\n for (u = 0; u < lu; u++) {\n uvs[u + lu] = uvs[u];\n }\n frontUVs = frontUVs ? frontUVs : new Vector4(0.0, 0.0, 1.0, 1.0);\n backUVs = backUVs ? backUVs : new Vector4(0.0, 0.0, 1.0, 1.0);\n u = 0;\n for (i = 0; i < lu / 2; i++) {\n uvs[u] = frontUVs.x + (frontUVs.z - frontUVs.x) * uvs[u];\n uvs[u + 1] = frontUVs.y + (frontUVs.w - frontUVs.y) * uvs[u + 1];\n uvs[u + lu] = backUVs.x + (backUVs.z - backUVs.x) * uvs[u + lu];\n uvs[u + lu + 1] = backUVs.y + (backUVs.w - backUVs.y) * uvs[u + lu + 1];\n u += 2;\n }\n break;\n }\n }\n }\n /**\n * Creates a VertexData from serialized data\n * @param parsedVertexData the parsed data from an imported file\n * @returns a VertexData\n */\n static Parse(parsedVertexData) {\n const vertexData = new VertexData();\n // positions\n const positions = parsedVertexData.positions;\n if (positions) {\n vertexData.set(positions, VertexBuffer.PositionKind);\n }\n // normals\n const normals = parsedVertexData.normals;\n if (normals) {\n vertexData.set(normals, VertexBuffer.NormalKind);\n }\n // tangents\n const tangents = parsedVertexData.tangents;\n if (tangents) {\n vertexData.set(tangents, VertexBuffer.TangentKind);\n }\n // uvs\n const uvs = parsedVertexData.uvs;\n if (uvs) {\n vertexData.set(uvs, VertexBuffer.UVKind);\n }\n // uv2s\n const uvs2 = parsedVertexData.uvs2;\n if (uvs2) {\n vertexData.set(uvs2, VertexBuffer.UV2Kind);\n }\n // uv3s\n const uvs3 = parsedVertexData.uvs3;\n if (uvs3) {\n vertexData.set(uvs3, VertexBuffer.UV3Kind);\n }\n // uv4s\n const uvs4 = parsedVertexData.uvs4;\n if (uvs4) {\n vertexData.set(uvs4, VertexBuffer.UV4Kind);\n }\n // uv5s\n const uvs5 = parsedVertexData.uvs5;\n if (uvs5) {\n vertexData.set(uvs5, VertexBuffer.UV5Kind);\n }\n // uv6s\n const uvs6 = parsedVertexData.uvs6;\n if (uvs6) {\n vertexData.set(uvs6, VertexBuffer.UV6Kind);\n }\n // colors\n const colors = parsedVertexData.colors;\n if (colors) {\n vertexData.set(Color4.CheckColors4(colors, positions.length / 3), VertexBuffer.ColorKind);\n }\n // matricesIndices\n const matricesIndices = parsedVertexData.matricesIndices;\n if (matricesIndices) {\n vertexData.set(matricesIndices, VertexBuffer.MatricesIndicesKind);\n }\n // matricesWeights\n const matricesWeights = parsedVertexData.matricesWeights;\n if (matricesWeights) {\n vertexData.set(matricesWeights, VertexBuffer.MatricesWeightsKind);\n }\n // indices\n const indices = parsedVertexData.indices;\n if (indices) {\n vertexData.indices = indices;\n }\n // MaterialInfos\n const materialInfos = parsedVertexData.materialInfos;\n if (materialInfos) {\n vertexData.materialInfos = [];\n for (const materialInfoFromJSON of materialInfos) {\n const materialInfo = new VertexDataMaterialInfo();\n materialInfo.indexCount = materialInfoFromJSON.indexCount;\n materialInfo.indexStart = materialInfoFromJSON.indexStart;\n materialInfo.verticesCount = materialInfoFromJSON.verticesCount;\n materialInfo.verticesStart = materialInfoFromJSON.verticesStart;\n materialInfo.materialIndex = materialInfoFromJSON.materialIndex;\n vertexData.materialInfos.push(materialInfo);\n }\n }\n return vertexData;\n }\n /**\n * Applies VertexData created from the imported parameters to the geometry\n * @param parsedVertexData the parsed data from an imported file\n * @param geometry the geometry to apply the VertexData to\n */\n static ImportVertexData(parsedVertexData, geometry) {\n const vertexData = VertexData.Parse(parsedVertexData);\n geometry.setAllVerticesData(vertexData, parsedVertexData.updatable);\n }\n }\n /**\n * Mesh side orientation : usually the external or front surface\n */\n VertexData.FRONTSIDE = 0;\n /**\n * Mesh side orientation : usually the internal or back surface\n */\n VertexData.BACKSIDE = 1;\n /**\n * Mesh side orientation : both internal and external or front and back surfaces\n */\n VertexData.DOUBLESIDE = 2;\n /**\n * Mesh side orientation : by default, `FRONTSIDE`\n */\n VertexData.DEFAULTSIDE = 0;\n __decorate$1([\n nativeOverride.filter((...[coordinates]) => !Array.isArray(coordinates))\n ], VertexData, \"_TransformVector3Coordinates\", null);\n __decorate$1([\n nativeOverride.filter((...[normals]) => !Array.isArray(normals))\n ], VertexData, \"_TransformVector3Normals\", null);\n __decorate$1([\n nativeOverride.filter((...[normals]) => !Array.isArray(normals))\n ], VertexData, \"_TransformVector4Normals\", null);\n __decorate$1([\n nativeOverride.filter((...[indices]) => !Array.isArray(indices))\n ], VertexData, \"_FlipFaces\", null);\n\n /**\n * Class used to represent data loading progression\n */\n class SceneLoaderFlags {\n /**\n * Gets or sets a boolean indicating if entire scene must be loaded even if scene contains incremental data\n */\n static get ForceFullSceneLoadingForIncremental() {\n return SceneLoaderFlags._ForceFullSceneLoadingForIncremental;\n }\n static set ForceFullSceneLoadingForIncremental(value) {\n SceneLoaderFlags._ForceFullSceneLoadingForIncremental = value;\n }\n /**\n * Gets or sets a boolean indicating if loading screen must be displayed while loading a scene\n */\n static get ShowLoadingScreen() {\n return SceneLoaderFlags._ShowLoadingScreen;\n }\n static set ShowLoadingScreen(value) {\n SceneLoaderFlags._ShowLoadingScreen = value;\n }\n /**\n * Defines the current logging level (while loading the scene)\n * @ignorenaming\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n static get loggingLevel() {\n return SceneLoaderFlags._LoggingLevel;\n }\n // eslint-disable-next-line @typescript-eslint/naming-convention\n static set loggingLevel(value) {\n SceneLoaderFlags._LoggingLevel = value;\n }\n /**\n * Gets or set a boolean indicating if matrix weights must be cleaned upon loading\n */\n static get CleanBoneMatrixWeights() {\n return SceneLoaderFlags._CleanBoneMatrixWeights;\n }\n static set CleanBoneMatrixWeights(value) {\n SceneLoaderFlags._CleanBoneMatrixWeights = value;\n }\n }\n // Flags\n SceneLoaderFlags._ForceFullSceneLoadingForIncremental = false;\n SceneLoaderFlags._ShowLoadingScreen = true;\n SceneLoaderFlags._CleanBoneMatrixWeights = false;\n SceneLoaderFlags._LoggingLevel = 0;\n\n /**\n * Options used to control default behaviors regarding compatibility support\n */\n class CompatibilityOptions {\n }\n /**\n * Defines if the system should use OpenGL convention for UVs when creating geometry or loading .babylon files (false by default)\n */\n CompatibilityOptions.UseOpenGLOrientationForUV = false;\n\n /**\n * Class used to store geometry data (vertex buffers + index buffer)\n */\n class Geometry {\n /**\n * Gets or sets the Bias Vector to apply on the bounding elements (box/sphere), the max extend is computed as v += v * bias.x + bias.y, the min is computed as v -= v * bias.x + bias.y\n */\n get boundingBias() {\n return this._boundingBias;\n }\n /**\n * Gets or sets the Bias Vector to apply on the bounding elements (box/sphere), the max extend is computed as v += v * bias.x + bias.y, the min is computed as v -= v * bias.x + bias.y\n */\n set boundingBias(value) {\n if (this._boundingBias) {\n this._boundingBias.copyFrom(value);\n }\n else {\n this._boundingBias = value.clone();\n }\n this._updateBoundingInfo(true, null);\n }\n /**\n * Static function used to attach a new empty geometry to a mesh\n * @param mesh defines the mesh to attach the geometry to\n * @returns the new Geometry\n */\n static CreateGeometryForMesh(mesh) {\n const geometry = new Geometry(Geometry.RandomId(), mesh.getScene());\n geometry.applyToMesh(mesh);\n return geometry;\n }\n /** Get the list of meshes using this geometry */\n get meshes() {\n return this._meshes;\n }\n /**\n * Creates a new geometry\n * @param id defines the unique ID\n * @param scene defines the hosting scene\n * @param vertexData defines the VertexData used to get geometry data\n * @param updatable defines if geometry must be updatable (false by default)\n * @param mesh defines the mesh that will be associated with the geometry\n */\n constructor(id, scene, vertexData, updatable = false, mesh = null) {\n /**\n * Gets the delay loading state of the geometry (none by default which means not delayed)\n */\n this.delayLoadState = 0;\n this._totalVertices = 0;\n this._isDisposed = false;\n this._indexBufferIsUpdatable = false;\n this._positionsCache = [];\n /** @internal */\n this._parentContainer = null;\n /**\n * If set to true (false by default), the bounding info applied to the meshes sharing this geometry will be the bounding info defined at the class level\n * and won't be computed based on the vertex positions (which is what we get when useBoundingInfoFromGeometry = false)\n */\n this.useBoundingInfoFromGeometry = false;\n this._scene = scene || EngineStore.LastCreatedScene;\n if (!this._scene) {\n return;\n }\n this.id = id;\n this.uniqueId = this._scene.getUniqueId();\n this._engine = this._scene.getEngine();\n this._meshes = [];\n //Init vertex buffer cache\n this._vertexBuffers = {};\n this._indices = [];\n this._updatable = updatable;\n // vertexData\n if (vertexData) {\n this.setAllVerticesData(vertexData, updatable);\n }\n else {\n this._totalVertices = 0;\n }\n if (this._engine.getCaps().vertexArrayObject) {\n this._vertexArrayObjects = {};\n }\n // applyToMesh\n if (mesh) {\n this.applyToMesh(mesh);\n mesh.computeWorldMatrix(true);\n }\n }\n /**\n * Gets the current extend of the geometry\n */\n get extend() {\n return this._extend;\n }\n /**\n * Gets the hosting scene\n * @returns the hosting Scene\n */\n getScene() {\n return this._scene;\n }\n /**\n * Gets the hosting engine\n * @returns the hosting Engine\n */\n getEngine() {\n return this._engine;\n }\n /**\n * Defines if the geometry is ready to use\n * @returns true if the geometry is ready to be used\n */\n isReady() {\n return this.delayLoadState === 1 || this.delayLoadState === 0;\n }\n /**\n * Gets a value indicating that the geometry should not be serialized\n */\n get doNotSerialize() {\n for (let index = 0; index < this._meshes.length; index++) {\n if (!this._meshes[index].doNotSerialize) {\n return false;\n }\n }\n return true;\n }\n /** @internal */\n _rebuild() {\n if (this._vertexArrayObjects) {\n this._vertexArrayObjects = {};\n }\n // Index buffer\n if (this._meshes.length !== 0 && this._indices) {\n this._indexBuffer = this._engine.createIndexBuffer(this._indices, this._updatable);\n }\n // Vertex buffers\n for (const key in this._vertexBuffers) {\n const vertexBuffer = this._vertexBuffers[key];\n vertexBuffer._rebuild();\n }\n }\n /**\n * Affects all geometry data in one call\n * @param vertexData defines the geometry data\n * @param updatable defines if the geometry must be flagged as updatable (false as default)\n */\n setAllVerticesData(vertexData, updatable) {\n vertexData.applyToGeometry(this, updatable);\n this._notifyUpdate();\n }\n /**\n * Set specific vertex data\n * @param kind defines the data kind (Position, normal, etc...)\n * @param data defines the vertex data to use\n * @param updatable defines if the vertex must be flagged as updatable (false as default)\n * @param stride defines the stride to use (0 by default). This value is deduced from the kind value if not specified\n */\n setVerticesData(kind, data, updatable = false, stride) {\n if (updatable && Array.isArray(data)) {\n // to avoid converting to Float32Array at each draw call in engine.updateDynamicVertexBuffer, we make the conversion a single time here\n data = new Float32Array(data);\n }\n const buffer = new VertexBuffer(this._engine, data, kind, updatable, this._meshes.length === 0, stride);\n this.setVerticesBuffer(buffer);\n }\n /**\n * Removes a specific vertex data\n * @param kind defines the data kind (Position, normal, etc...)\n */\n removeVerticesData(kind) {\n if (this._vertexBuffers[kind]) {\n this._vertexBuffers[kind].dispose();\n delete this._vertexBuffers[kind];\n }\n if (this._vertexArrayObjects) {\n this._disposeVertexArrayObjects();\n }\n }\n /**\n * Affect a vertex buffer to the geometry. the vertexBuffer.getKind() function is used to determine where to store the data\n * @param buffer defines the vertex buffer to use\n * @param totalVertices defines the total number of vertices for position kind (could be null)\n * @param disposeExistingBuffer disposes the existing buffer, if any (default: true)\n */\n setVerticesBuffer(buffer, totalVertices = null, disposeExistingBuffer = true) {\n const kind = buffer.getKind();\n if (this._vertexBuffers[kind] && disposeExistingBuffer) {\n this._vertexBuffers[kind].dispose();\n }\n if (buffer._buffer) {\n buffer._buffer._increaseReferences();\n }\n this._vertexBuffers[kind] = buffer;\n const meshes = this._meshes;\n const numOfMeshes = meshes.length;\n if (kind === VertexBuffer.PositionKind) {\n const data = buffer.getData();\n if (totalVertices != null) {\n this._totalVertices = totalVertices;\n }\n else {\n if (data != null) {\n this._totalVertices = data.length / (buffer.type === VertexBuffer.BYTE ? buffer.byteStride : buffer.byteStride / 4);\n }\n }\n this._updateExtend(data);\n this._resetPointsArrayCache();\n for (let index = 0; index < numOfMeshes; index++) {\n const mesh = meshes[index];\n mesh.buildBoundingInfo(this._extend.minimum, this._extend.maximum);\n mesh._createGlobalSubMesh(mesh.isUnIndexed);\n mesh.computeWorldMatrix(true);\n mesh.synchronizeInstances();\n }\n }\n this._notifyUpdate(kind);\n }\n /**\n * Update a specific vertex buffer\n * This function will directly update the underlying DataBuffer according to the passed numeric array or Float32Array\n * It will do nothing if the buffer is not updatable\n * @param kind defines the data kind (Position, normal, etc...)\n * @param data defines the data to use\n * @param offset defines the offset in the target buffer where to store the data\n * @param useBytes set to true if the offset is in bytes\n */\n updateVerticesDataDirectly(kind, data, offset, useBytes = false) {\n const vertexBuffer = this.getVertexBuffer(kind);\n if (!vertexBuffer) {\n return;\n }\n vertexBuffer.updateDirectly(data, offset, useBytes);\n this._notifyUpdate(kind);\n }\n /**\n * Update a specific vertex buffer\n * This function will create a new buffer if the current one is not updatable\n * @param kind defines the data kind (Position, normal, etc...)\n * @param data defines the data to use\n * @param updateExtends defines if the geometry extends must be recomputed (false by default)\n */\n updateVerticesData(kind, data, updateExtends = false) {\n const vertexBuffer = this.getVertexBuffer(kind);\n if (!vertexBuffer) {\n return;\n }\n vertexBuffer.update(data);\n if (kind === VertexBuffer.PositionKind) {\n this._updateBoundingInfo(updateExtends, data);\n }\n this._notifyUpdate(kind);\n }\n _updateBoundingInfo(updateExtends, data) {\n if (updateExtends) {\n this._updateExtend(data);\n }\n this._resetPointsArrayCache();\n if (updateExtends) {\n const meshes = this._meshes;\n for (const mesh of meshes) {\n if (mesh.hasBoundingInfo) {\n mesh.getBoundingInfo().reConstruct(this._extend.minimum, this._extend.maximum);\n }\n else {\n mesh.buildBoundingInfo(this._extend.minimum, this._extend.maximum);\n }\n const subMeshes = mesh.subMeshes;\n for (const subMesh of subMeshes) {\n subMesh.refreshBoundingInfo();\n }\n }\n }\n }\n /**\n * @internal\n */\n _bind(effect, indexToBind, overrideVertexBuffers, overrideVertexArrayObjects) {\n if (!effect) {\n return;\n }\n if (indexToBind === undefined) {\n indexToBind = this._indexBuffer;\n }\n const vbs = this.getVertexBuffers();\n if (!vbs) {\n return;\n }\n if (indexToBind != this._indexBuffer || (!this._vertexArrayObjects && !overrideVertexArrayObjects)) {\n this._engine.bindBuffers(vbs, indexToBind, effect, overrideVertexBuffers);\n return;\n }\n const vaos = overrideVertexArrayObjects ? overrideVertexArrayObjects : this._vertexArrayObjects;\n // Using VAO\n if (!vaos[effect.key]) {\n vaos[effect.key] = this._engine.recordVertexArrayObject(vbs, indexToBind, effect, overrideVertexBuffers);\n }\n this._engine.bindVertexArrayObject(vaos[effect.key], indexToBind);\n }\n /**\n * Gets total number of vertices\n * @returns the total number of vertices\n */\n getTotalVertices() {\n if (!this.isReady()) {\n return 0;\n }\n return this._totalVertices;\n }\n /**\n * Gets a specific vertex data attached to this geometry. Float data is constructed if the vertex buffer data cannot be returned directly.\n * @param kind defines the data kind (Position, normal, etc...)\n * @param copyWhenShared defines if the returned array must be cloned upon returning it if the current geometry is shared between multiple meshes\n * @param forceCopy defines a boolean indicating that the returned array must be cloned upon returning it\n * @returns a float array containing vertex data\n */\n getVerticesData(kind, copyWhenShared, forceCopy) {\n const vertexBuffer = this.getVertexBuffer(kind);\n if (!vertexBuffer) {\n return null;\n }\n return vertexBuffer.getFloatData(this._totalVertices, forceCopy || (copyWhenShared && this._meshes.length !== 1));\n }\n /**\n * Returns a boolean defining if the vertex data for the requested `kind` is updatable\n * @param kind defines the data kind (Position, normal, etc...)\n * @returns true if the vertex buffer with the specified kind is updatable\n */\n isVertexBufferUpdatable(kind) {\n const vb = this._vertexBuffers[kind];\n if (!vb) {\n return false;\n }\n return vb.isUpdatable();\n }\n /**\n * Gets a specific vertex buffer\n * @param kind defines the data kind (Position, normal, etc...)\n * @returns a VertexBuffer\n */\n getVertexBuffer(kind) {\n if (!this.isReady()) {\n return null;\n }\n return this._vertexBuffers[kind];\n }\n /**\n * Returns all vertex buffers\n * @returns an object holding all vertex buffers indexed by kind\n */\n getVertexBuffers() {\n if (!this.isReady()) {\n return null;\n }\n return this._vertexBuffers;\n }\n /**\n * Gets a boolean indicating if specific vertex buffer is present\n * @param kind defines the data kind (Position, normal, etc...)\n * @returns true if data is present\n */\n isVerticesDataPresent(kind) {\n if (!this._vertexBuffers) {\n if (this._delayInfo) {\n return this._delayInfo.indexOf(kind) !== -1;\n }\n return false;\n }\n return this._vertexBuffers[kind] !== undefined;\n }\n /**\n * Gets a list of all attached data kinds (Position, normal, etc...)\n * @returns a list of string containing all kinds\n */\n getVerticesDataKinds() {\n const result = [];\n let kind;\n if (!this._vertexBuffers && this._delayInfo) {\n for (kind in this._delayInfo) {\n result.push(kind);\n }\n }\n else {\n for (kind in this._vertexBuffers) {\n result.push(kind);\n }\n }\n return result;\n }\n /**\n * Update index buffer\n * @param indices defines the indices to store in the index buffer\n * @param offset defines the offset in the target buffer where to store the data\n * @param gpuMemoryOnly defines a boolean indicating that only the GPU memory must be updated leaving the CPU version of the indices unchanged (false by default)\n */\n updateIndices(indices, offset, gpuMemoryOnly = false) {\n if (!this._indexBuffer) {\n return;\n }\n if (!this._indexBufferIsUpdatable) {\n this.setIndices(indices, null, true);\n }\n else {\n const needToUpdateSubMeshes = indices.length !== this._indices.length;\n if (!gpuMemoryOnly) {\n this._indices = indices.slice();\n }\n this._engine.updateDynamicIndexBuffer(this._indexBuffer, indices, offset);\n if (needToUpdateSubMeshes) {\n for (const mesh of this._meshes) {\n mesh._createGlobalSubMesh(true);\n }\n }\n }\n }\n /**\n * Creates a new index buffer\n * @param indices defines the indices to store in the index buffer\n * @param totalVertices defines the total number of vertices (could be null)\n * @param updatable defines if the index buffer must be flagged as updatable (false by default)\n */\n setIndices(indices, totalVertices = null, updatable = false) {\n if (this._indexBuffer) {\n this._engine._releaseBuffer(this._indexBuffer);\n }\n this._indices = indices;\n this._indexBufferIsUpdatable = updatable;\n if (this._meshes.length !== 0 && this._indices) {\n this._indexBuffer = this._engine.createIndexBuffer(this._indices, updatable);\n }\n if (totalVertices != undefined) {\n // including null and undefined\n this._totalVertices = totalVertices;\n }\n for (const mesh of this._meshes) {\n mesh._createGlobalSubMesh(true);\n mesh.synchronizeInstances();\n }\n this._notifyUpdate();\n }\n /**\n * Return the total number of indices\n * @returns the total number of indices\n */\n getTotalIndices() {\n if (!this.isReady()) {\n return 0;\n }\n return this._indices.length;\n }\n /**\n * Gets the index buffer array\n * @param copyWhenShared defines if the returned array must be cloned upon returning it if the current geometry is shared between multiple meshes\n * @param forceCopy defines a boolean indicating that the returned array must be cloned upon returning it\n * @returns the index buffer array\n */\n getIndices(copyWhenShared, forceCopy) {\n if (!this.isReady()) {\n return null;\n }\n const orig = this._indices;\n if (!forceCopy && (!copyWhenShared || this._meshes.length === 1)) {\n return orig;\n }\n else {\n return orig.slice();\n }\n }\n /**\n * Gets the index buffer\n * @returns the index buffer\n */\n getIndexBuffer() {\n if (!this.isReady()) {\n return null;\n }\n return this._indexBuffer;\n }\n /**\n * @internal\n */\n _releaseVertexArrayObject(effect = null) {\n if (!effect || !this._vertexArrayObjects) {\n return;\n }\n if (this._vertexArrayObjects[effect.key]) {\n this._engine.releaseVertexArrayObject(this._vertexArrayObjects[effect.key]);\n delete this._vertexArrayObjects[effect.key];\n }\n }\n /**\n * Release the associated resources for a specific mesh\n * @param mesh defines the source mesh\n * @param shouldDispose defines if the geometry must be disposed if there is no more mesh pointing to it\n */\n releaseForMesh(mesh, shouldDispose) {\n const meshes = this._meshes;\n const index = meshes.indexOf(mesh);\n if (index === -1) {\n return;\n }\n meshes.splice(index, 1);\n if (this._vertexArrayObjects) {\n mesh._invalidateInstanceVertexArrayObject();\n }\n mesh._geometry = null;\n if (meshes.length === 0 && shouldDispose) {\n this.dispose();\n }\n }\n /**\n * Apply current geometry to a given mesh\n * @param mesh defines the mesh to apply geometry to\n */\n applyToMesh(mesh) {\n if (mesh._geometry === this) {\n return;\n }\n const previousGeometry = mesh._geometry;\n if (previousGeometry) {\n previousGeometry.releaseForMesh(mesh);\n }\n if (this._vertexArrayObjects) {\n mesh._invalidateInstanceVertexArrayObject();\n }\n const meshes = this._meshes;\n // must be done before setting vertexBuffers because of mesh._createGlobalSubMesh()\n mesh._geometry = this;\n mesh._internalAbstractMeshDataInfo._positions = null;\n this._scene.pushGeometry(this);\n meshes.push(mesh);\n if (this.isReady()) {\n this._applyToMesh(mesh);\n }\n else if (this._boundingInfo) {\n mesh.setBoundingInfo(this._boundingInfo);\n }\n }\n _updateExtend(data = null) {\n if (this.useBoundingInfoFromGeometry && this._boundingInfo) {\n this._extend = {\n minimum: this._boundingInfo.minimum.clone(),\n maximum: this._boundingInfo.maximum.clone(),\n };\n }\n else {\n if (!data) {\n data = this.getVerticesData(VertexBuffer.PositionKind);\n // This can happen if the buffer comes from a Hardware Buffer where\n // The data have not been uploaded by Babylon. (ex: Compute Shaders and Storage Buffers)\n if (!data) {\n return;\n }\n }\n this._extend = extractMinAndMax(data, 0, this._totalVertices, this.boundingBias, 3);\n }\n }\n _applyToMesh(mesh) {\n const numOfMeshes = this._meshes.length;\n // vertexBuffers\n for (const kind in this._vertexBuffers) {\n if (numOfMeshes === 1) {\n this._vertexBuffers[kind].create();\n }\n if (kind === VertexBuffer.PositionKind) {\n if (!this._extend) {\n this._updateExtend();\n }\n mesh.buildBoundingInfo(this._extend.minimum, this._extend.maximum);\n mesh._createGlobalSubMesh(mesh.isUnIndexed);\n //bounding info was just created again, world matrix should be applied again.\n mesh._updateBoundingInfo();\n }\n }\n // indexBuffer\n if (numOfMeshes === 1 && this._indices && this._indices.length > 0) {\n this._indexBuffer = this._engine.createIndexBuffer(this._indices, this._updatable);\n }\n // morphTargets\n mesh._syncGeometryWithMorphTargetManager();\n // instances\n mesh.synchronizeInstances();\n }\n _notifyUpdate(kind) {\n if (this.onGeometryUpdated) {\n this.onGeometryUpdated(this, kind);\n }\n if (this._vertexArrayObjects) {\n this._disposeVertexArrayObjects();\n }\n for (const mesh of this._meshes) {\n mesh._markSubMeshesAsAttributesDirty();\n }\n }\n /**\n * Load the geometry if it was flagged as delay loaded\n * @param scene defines the hosting scene\n * @param onLoaded defines a callback called when the geometry is loaded\n */\n load(scene, onLoaded) {\n if (this.delayLoadState === 2) {\n return;\n }\n if (this.isReady()) {\n if (onLoaded) {\n onLoaded();\n }\n return;\n }\n this.delayLoadState = 2;\n this._queueLoad(scene, onLoaded);\n }\n _queueLoad(scene, onLoaded) {\n if (!this.delayLoadingFile) {\n return;\n }\n scene.addPendingData(this);\n scene._loadFile(this.delayLoadingFile, (data) => {\n if (!this._delayLoadingFunction) {\n return;\n }\n this._delayLoadingFunction(JSON.parse(data), this);\n this.delayLoadState = 1;\n this._delayInfo = [];\n scene.removePendingData(this);\n const meshes = this._meshes;\n const numOfMeshes = meshes.length;\n for (let index = 0; index < numOfMeshes; index++) {\n this._applyToMesh(meshes[index]);\n }\n if (onLoaded) {\n onLoaded();\n }\n }, undefined, true);\n }\n /**\n * Invert the geometry to move from a right handed system to a left handed one.\n */\n toLeftHanded() {\n // Flip faces\n const tIndices = this.getIndices(false);\n if (tIndices != null && tIndices.length > 0) {\n for (let i = 0; i < tIndices.length; i += 3) {\n const tTemp = tIndices[i + 0];\n tIndices[i + 0] = tIndices[i + 2];\n tIndices[i + 2] = tTemp;\n }\n this.setIndices(tIndices);\n }\n // Negate position.z\n const tPositions = this.getVerticesData(VertexBuffer.PositionKind, false);\n if (tPositions != null && tPositions.length > 0) {\n for (let i = 0; i < tPositions.length; i += 3) {\n tPositions[i + 2] = -tPositions[i + 2];\n }\n this.setVerticesData(VertexBuffer.PositionKind, tPositions, false);\n }\n // Negate normal.z\n const tNormals = this.getVerticesData(VertexBuffer.NormalKind, false);\n if (tNormals != null && tNormals.length > 0) {\n for (let i = 0; i < tNormals.length; i += 3) {\n tNormals[i + 2] = -tNormals[i + 2];\n }\n this.setVerticesData(VertexBuffer.NormalKind, tNormals, false);\n }\n }\n // Cache\n /** @internal */\n _resetPointsArrayCache() {\n this._positions = null;\n }\n /** @internal */\n _generatePointsArray() {\n if (this._positions) {\n return true;\n }\n const data = this.getVerticesData(VertexBuffer.PositionKind);\n if (!data || data.length === 0) {\n return false;\n }\n for (let index = this._positionsCache.length * 3, arrayIdx = this._positionsCache.length; index < data.length; index += 3, ++arrayIdx) {\n this._positionsCache[arrayIdx] = Vector3.FromArray(data, index);\n }\n for (let index = 0, arrayIdx = 0; index < data.length; index += 3, ++arrayIdx) {\n this._positionsCache[arrayIdx].set(data[0 + index], data[1 + index], data[2 + index]);\n }\n // just in case the number of positions was reduced, splice the array\n this._positionsCache.length = data.length / 3;\n this._positions = this._positionsCache;\n return true;\n }\n /**\n * Gets a value indicating if the geometry is disposed\n * @returns true if the geometry was disposed\n */\n isDisposed() {\n return this._isDisposed;\n }\n _disposeVertexArrayObjects() {\n if (this._vertexArrayObjects) {\n for (const kind in this._vertexArrayObjects) {\n this._engine.releaseVertexArrayObject(this._vertexArrayObjects[kind]);\n }\n this._vertexArrayObjects = {}; // Will trigger a rebuild of the VAO if supported\n const meshes = this._meshes;\n const numOfMeshes = meshes.length;\n for (let index = 0; index < numOfMeshes; index++) {\n meshes[index]._invalidateInstanceVertexArrayObject();\n }\n }\n }\n /**\n * Free all associated resources\n */\n dispose() {\n const meshes = this._meshes;\n const numOfMeshes = meshes.length;\n let index;\n for (index = 0; index < numOfMeshes; index++) {\n this.releaseForMesh(meshes[index]);\n }\n this._meshes.length = 0;\n this._disposeVertexArrayObjects();\n for (const kind in this._vertexBuffers) {\n this._vertexBuffers[kind].dispose();\n }\n this._vertexBuffers = {};\n this._totalVertices = 0;\n if (this._indexBuffer) {\n this._engine._releaseBuffer(this._indexBuffer);\n }\n this._indexBuffer = null;\n this._indices = [];\n this.delayLoadState = 0;\n this.delayLoadingFile = null;\n this._delayLoadingFunction = null;\n this._delayInfo = [];\n this._boundingInfo = null;\n this._scene.removeGeometry(this);\n if (this._parentContainer) {\n const index = this._parentContainer.geometries.indexOf(this);\n if (index > -1) {\n this._parentContainer.geometries.splice(index, 1);\n }\n this._parentContainer = null;\n }\n this._isDisposed = true;\n }\n /**\n * Clone the current geometry into a new geometry\n * @param id defines the unique ID of the new geometry\n * @returns a new geometry object\n */\n copy(id) {\n const vertexData = new VertexData();\n vertexData.indices = [];\n const indices = this.getIndices();\n if (indices) {\n for (let index = 0; index < indices.length; index++) {\n vertexData.indices.push(indices[index]);\n }\n }\n let updatable = false;\n let stopChecking = false;\n let kind;\n for (kind in this._vertexBuffers) {\n // using slice() to make a copy of the array and not just reference it\n const data = this.getVerticesData(kind);\n if (data) {\n if (data instanceof Float32Array) {\n vertexData.set(new Float32Array(data), kind);\n }\n else {\n vertexData.set(data.slice(0), kind);\n }\n if (!stopChecking) {\n const vb = this.getVertexBuffer(kind);\n if (vb) {\n updatable = vb.isUpdatable();\n stopChecking = !updatable;\n }\n }\n }\n }\n const geometry = new Geometry(id, this._scene, vertexData, updatable);\n geometry.delayLoadState = this.delayLoadState;\n geometry.delayLoadingFile = this.delayLoadingFile;\n geometry._delayLoadingFunction = this._delayLoadingFunction;\n for (kind in this._delayInfo) {\n geometry._delayInfo = geometry._delayInfo || [];\n geometry._delayInfo.push(kind);\n }\n // Bounding info\n geometry._boundingInfo = new BoundingInfo(this._extend.minimum, this._extend.maximum);\n return geometry;\n }\n /**\n * Serialize the current geometry info (and not the vertices data) into a JSON object\n * @returns a JSON representation of the current geometry data (without the vertices data)\n */\n serialize() {\n const serializationObject = {};\n serializationObject.id = this.id;\n serializationObject.uniqueId = this.uniqueId;\n serializationObject.updatable = this._updatable;\n if (Tags && Tags.HasTags(this)) {\n serializationObject.tags = Tags.GetTags(this);\n }\n return serializationObject;\n }\n _toNumberArray(origin) {\n if (Array.isArray(origin)) {\n return origin;\n }\n else {\n return Array.prototype.slice.call(origin);\n }\n }\n /**\n * Release any memory retained by the cached data on the Geometry.\n *\n * Call this function to reduce memory footprint of the mesh.\n * Vertex buffers will not store CPU data anymore (this will prevent picking, collisions or physics to work correctly)\n */\n clearCachedData() {\n this._indices = [];\n this._resetPointsArrayCache();\n for (const vbName in this._vertexBuffers) {\n if (!Object.prototype.hasOwnProperty.call(this._vertexBuffers, vbName)) {\n continue;\n }\n this._vertexBuffers[vbName]._buffer._data = null;\n }\n }\n /**\n * Serialize all vertices data into a JSON object\n * @returns a JSON representation of the current geometry data\n */\n serializeVerticeData() {\n const serializationObject = this.serialize();\n if (this.isVerticesDataPresent(VertexBuffer.PositionKind)) {\n serializationObject.positions = this._toNumberArray(this.getVerticesData(VertexBuffer.PositionKind));\n if (this.isVertexBufferUpdatable(VertexBuffer.PositionKind)) {\n serializationObject.positions._updatable = true;\n }\n }\n if (this.isVerticesDataPresent(VertexBuffer.NormalKind)) {\n serializationObject.normals = this._toNumberArray(this.getVerticesData(VertexBuffer.NormalKind));\n if (this.isVertexBufferUpdatable(VertexBuffer.NormalKind)) {\n serializationObject.normals._updatable = true;\n }\n }\n if (this.isVerticesDataPresent(VertexBuffer.TangentKind)) {\n serializationObject.tangents = this._toNumberArray(this.getVerticesData(VertexBuffer.TangentKind));\n if (this.isVertexBufferUpdatable(VertexBuffer.TangentKind)) {\n serializationObject.tangents._updatable = true;\n }\n }\n if (this.isVerticesDataPresent(VertexBuffer.UVKind)) {\n serializationObject.uvs = this._toNumberArray(this.getVerticesData(VertexBuffer.UVKind));\n if (this.isVertexBufferUpdatable(VertexBuffer.UVKind)) {\n serializationObject.uvs._updatable = true;\n }\n }\n if (this.isVerticesDataPresent(VertexBuffer.UV2Kind)) {\n serializationObject.uv2s = this._toNumberArray(this.getVerticesData(VertexBuffer.UV2Kind));\n if (this.isVertexBufferUpdatable(VertexBuffer.UV2Kind)) {\n serializationObject.uv2s._updatable = true;\n }\n }\n if (this.isVerticesDataPresent(VertexBuffer.UV3Kind)) {\n serializationObject.uv3s = this._toNumberArray(this.getVerticesData(VertexBuffer.UV3Kind));\n if (this.isVertexBufferUpdatable(VertexBuffer.UV3Kind)) {\n serializationObject.uv3s._updatable = true;\n }\n }\n if (this.isVerticesDataPresent(VertexBuffer.UV4Kind)) {\n serializationObject.uv4s = this._toNumberArray(this.getVerticesData(VertexBuffer.UV4Kind));\n if (this.isVertexBufferUpdatable(VertexBuffer.UV4Kind)) {\n serializationObject.uv4s._updatable = true;\n }\n }\n if (this.isVerticesDataPresent(VertexBuffer.UV5Kind)) {\n serializationObject.uv5s = this._toNumberArray(this.getVerticesData(VertexBuffer.UV5Kind));\n if (this.isVertexBufferUpdatable(VertexBuffer.UV5Kind)) {\n serializationObject.uv5s._updatable = true;\n }\n }\n if (this.isVerticesDataPresent(VertexBuffer.UV6Kind)) {\n serializationObject.uv6s = this._toNumberArray(this.getVerticesData(VertexBuffer.UV6Kind));\n if (this.isVertexBufferUpdatable(VertexBuffer.UV6Kind)) {\n serializationObject.uv6s._updatable = true;\n }\n }\n if (this.isVerticesDataPresent(VertexBuffer.ColorKind)) {\n serializationObject.colors = this._toNumberArray(this.getVerticesData(VertexBuffer.ColorKind));\n if (this.isVertexBufferUpdatable(VertexBuffer.ColorKind)) {\n serializationObject.colors._updatable = true;\n }\n }\n if (this.isVerticesDataPresent(VertexBuffer.MatricesIndicesKind)) {\n serializationObject.matricesIndices = this._toNumberArray(this.getVerticesData(VertexBuffer.MatricesIndicesKind));\n serializationObject.matricesIndices._isExpanded = true;\n if (this.isVertexBufferUpdatable(VertexBuffer.MatricesIndicesKind)) {\n serializationObject.matricesIndices._updatable = true;\n }\n }\n if (this.isVerticesDataPresent(VertexBuffer.MatricesWeightsKind)) {\n serializationObject.matricesWeights = this._toNumberArray(this.getVerticesData(VertexBuffer.MatricesWeightsKind));\n if (this.isVertexBufferUpdatable(VertexBuffer.MatricesWeightsKind)) {\n serializationObject.matricesWeights._updatable = true;\n }\n }\n serializationObject.indices = this._toNumberArray(this.getIndices());\n return serializationObject;\n }\n // Statics\n /**\n * Extracts a clone of a mesh geometry\n * @param mesh defines the source mesh\n * @param id defines the unique ID of the new geometry object\n * @returns the new geometry object\n */\n static ExtractFromMesh(mesh, id) {\n const geometry = mesh._geometry;\n if (!geometry) {\n return null;\n }\n return geometry.copy(id);\n }\n /**\n * You should now use Tools.RandomId(), this method is still here for legacy reasons.\n * Implementation from http://stackoverflow.com/questions/105034/how-to-create-a-guid-uuid-in-javascript/2117523#answer-2117523\n * Be aware Math.random() could cause collisions, but:\n * \"All but 6 of the 128 bits of the ID are randomly generated, which means that for any two ids, there's a 1 in 2^^122 (or 5.3x10^^36) chance they'll collide\"\n * @returns a string containing a new GUID\n */\n static RandomId() {\n return Tools.RandomId();\n }\n static _GetGeometryByLoadedUniqueId(uniqueId, scene) {\n for (let index = 0; index < scene.geometries.length; index++) {\n if (scene.geometries[index]._loadedUniqueId === uniqueId) {\n return scene.geometries[index];\n }\n }\n return null;\n }\n /**\n * @internal\n */\n static _ImportGeometry(parsedGeometry, mesh) {\n const scene = mesh.getScene();\n // Geometry\n const geometryUniqueId = parsedGeometry.geometryUniqueId;\n const geometryId = parsedGeometry.geometryId;\n if (geometryUniqueId || geometryId) {\n const geometry = geometryUniqueId ? this._GetGeometryByLoadedUniqueId(geometryUniqueId, scene) : scene.getGeometryById(geometryId);\n if (geometry) {\n geometry.applyToMesh(mesh);\n }\n }\n else if (parsedGeometry instanceof ArrayBuffer) {\n const binaryInfo = mesh._binaryInfo;\n if (binaryInfo.positionsAttrDesc && binaryInfo.positionsAttrDesc.count > 0) {\n const positionsData = new Float32Array(parsedGeometry, binaryInfo.positionsAttrDesc.offset, binaryInfo.positionsAttrDesc.count);\n mesh.setVerticesData(VertexBuffer.PositionKind, positionsData, false);\n }\n if (binaryInfo.normalsAttrDesc && binaryInfo.normalsAttrDesc.count > 0) {\n const normalsData = new Float32Array(parsedGeometry, binaryInfo.normalsAttrDesc.offset, binaryInfo.normalsAttrDesc.count);\n mesh.setVerticesData(VertexBuffer.NormalKind, normalsData, false);\n }\n if (binaryInfo.tangetsAttrDesc && binaryInfo.tangetsAttrDesc.count > 0) {\n const tangentsData = new Float32Array(parsedGeometry, binaryInfo.tangetsAttrDesc.offset, binaryInfo.tangetsAttrDesc.count);\n mesh.setVerticesData(VertexBuffer.TangentKind, tangentsData, false);\n }\n if (binaryInfo.uvsAttrDesc && binaryInfo.uvsAttrDesc.count > 0) {\n const uvsData = new Float32Array(parsedGeometry, binaryInfo.uvsAttrDesc.offset, binaryInfo.uvsAttrDesc.count);\n if (CompatibilityOptions.UseOpenGLOrientationForUV) {\n for (let index = 1; index < uvsData.length; index += 2) {\n uvsData[index] = 1 - uvsData[index];\n }\n }\n mesh.setVerticesData(VertexBuffer.UVKind, uvsData, false);\n }\n if (binaryInfo.uvs2AttrDesc && binaryInfo.uvs2AttrDesc.count > 0) {\n const uvs2Data = new Float32Array(parsedGeometry, binaryInfo.uvs2AttrDesc.offset, binaryInfo.uvs2AttrDesc.count);\n if (CompatibilityOptions.UseOpenGLOrientationForUV) {\n for (let index = 1; index < uvs2Data.length; index += 2) {\n uvs2Data[index] = 1 - uvs2Data[index];\n }\n }\n mesh.setVerticesData(VertexBuffer.UV2Kind, uvs2Data, false);\n }\n if (binaryInfo.uvs3AttrDesc && binaryInfo.uvs3AttrDesc.count > 0) {\n const uvs3Data = new Float32Array(parsedGeometry, binaryInfo.uvs3AttrDesc.offset, binaryInfo.uvs3AttrDesc.count);\n if (CompatibilityOptions.UseOpenGLOrientationForUV) {\n for (let index = 1; index < uvs3Data.length; index += 2) {\n uvs3Data[index] = 1 - uvs3Data[index];\n }\n }\n mesh.setVerticesData(VertexBuffer.UV3Kind, uvs3Data, false);\n }\n if (binaryInfo.uvs4AttrDesc && binaryInfo.uvs4AttrDesc.count > 0) {\n const uvs4Data = new Float32Array(parsedGeometry, binaryInfo.uvs4AttrDesc.offset, binaryInfo.uvs4AttrDesc.count);\n if (CompatibilityOptions.UseOpenGLOrientationForUV) {\n for (let index = 1; index < uvs4Data.length; index += 2) {\n uvs4Data[index] = 1 - uvs4Data[index];\n }\n }\n mesh.setVerticesData(VertexBuffer.UV4Kind, uvs4Data, false);\n }\n if (binaryInfo.uvs5AttrDesc && binaryInfo.uvs5AttrDesc.count > 0) {\n const uvs5Data = new Float32Array(parsedGeometry, binaryInfo.uvs5AttrDesc.offset, binaryInfo.uvs5AttrDesc.count);\n if (CompatibilityOptions.UseOpenGLOrientationForUV) {\n for (let index = 1; index < uvs5Data.length; index += 2) {\n uvs5Data[index] = 1 - uvs5Data[index];\n }\n }\n mesh.setVerticesData(VertexBuffer.UV5Kind, uvs5Data, false);\n }\n if (binaryInfo.uvs6AttrDesc && binaryInfo.uvs6AttrDesc.count > 0) {\n const uvs6Data = new Float32Array(parsedGeometry, binaryInfo.uvs6AttrDesc.offset, binaryInfo.uvs6AttrDesc.count);\n if (CompatibilityOptions.UseOpenGLOrientationForUV) {\n for (let index = 1; index < uvs6Data.length; index += 2) {\n uvs6Data[index] = 1 - uvs6Data[index];\n }\n }\n mesh.setVerticesData(VertexBuffer.UV6Kind, uvs6Data, false);\n }\n if (binaryInfo.colorsAttrDesc && binaryInfo.colorsAttrDesc.count > 0) {\n const colorsData = new Float32Array(parsedGeometry, binaryInfo.colorsAttrDesc.offset, binaryInfo.colorsAttrDesc.count);\n mesh.setVerticesData(VertexBuffer.ColorKind, colorsData, false, binaryInfo.colorsAttrDesc.stride);\n }\n if (binaryInfo.matricesIndicesAttrDesc && binaryInfo.matricesIndicesAttrDesc.count > 0) {\n const matricesIndicesData = new Int32Array(parsedGeometry, binaryInfo.matricesIndicesAttrDesc.offset, binaryInfo.matricesIndicesAttrDesc.count);\n const floatIndices = [];\n for (let i = 0; i < matricesIndicesData.length; i++) {\n const index = matricesIndicesData[i];\n floatIndices.push(index & 0x000000ff);\n floatIndices.push((index & 0x0000ff00) >> 8);\n floatIndices.push((index & 0x00ff0000) >> 16);\n floatIndices.push((index >> 24) & 0xff); // & 0xFF to convert to v + 256 if v < 0\n }\n mesh.setVerticesData(VertexBuffer.MatricesIndicesKind, floatIndices, false);\n }\n if (binaryInfo.matricesIndicesExtraAttrDesc && binaryInfo.matricesIndicesExtraAttrDesc.count > 0) {\n const matricesIndicesData = new Int32Array(parsedGeometry, binaryInfo.matricesIndicesExtraAttrDesc.offset, binaryInfo.matricesIndicesExtraAttrDesc.count);\n const floatIndices = [];\n for (let i = 0; i < matricesIndicesData.length; i++) {\n const index = matricesIndicesData[i];\n floatIndices.push(index & 0x000000ff);\n floatIndices.push((index & 0x0000ff00) >> 8);\n floatIndices.push((index & 0x00ff0000) >> 16);\n floatIndices.push((index >> 24) & 0xff); // & 0xFF to convert to v + 256 if v < 0\n }\n mesh.setVerticesData(VertexBuffer.MatricesIndicesExtraKind, floatIndices, false);\n }\n if (binaryInfo.matricesWeightsAttrDesc && binaryInfo.matricesWeightsAttrDesc.count > 0) {\n const matricesWeightsData = new Float32Array(parsedGeometry, binaryInfo.matricesWeightsAttrDesc.offset, binaryInfo.matricesWeightsAttrDesc.count);\n mesh.setVerticesData(VertexBuffer.MatricesWeightsKind, matricesWeightsData, false);\n }\n if (binaryInfo.indicesAttrDesc && binaryInfo.indicesAttrDesc.count > 0) {\n const indicesData = new Int32Array(parsedGeometry, binaryInfo.indicesAttrDesc.offset, binaryInfo.indicesAttrDesc.count);\n mesh.setIndices(indicesData, null);\n }\n if (binaryInfo.subMeshesAttrDesc && binaryInfo.subMeshesAttrDesc.count > 0) {\n const subMeshesData = new Int32Array(parsedGeometry, binaryInfo.subMeshesAttrDesc.offset, binaryInfo.subMeshesAttrDesc.count * 5);\n mesh.subMeshes = [];\n for (let i = 0; i < binaryInfo.subMeshesAttrDesc.count; i++) {\n const materialIndex = subMeshesData[i * 5 + 0];\n const verticesStart = subMeshesData[i * 5 + 1];\n const verticesCount = subMeshesData[i * 5 + 2];\n const indexStart = subMeshesData[i * 5 + 3];\n const indexCount = subMeshesData[i * 5 + 4];\n SubMesh.AddToMesh(materialIndex, verticesStart, verticesCount, indexStart, indexCount, mesh);\n }\n }\n }\n else if (parsedGeometry.positions && parsedGeometry.normals && parsedGeometry.indices) {\n mesh.setVerticesData(VertexBuffer.PositionKind, parsedGeometry.positions, parsedGeometry.positions._updatable);\n mesh.setVerticesData(VertexBuffer.NormalKind, parsedGeometry.normals, parsedGeometry.normals._updatable);\n if (parsedGeometry.tangents) {\n mesh.setVerticesData(VertexBuffer.TangentKind, parsedGeometry.tangents, parsedGeometry.tangents._updatable);\n }\n if (parsedGeometry.uvs) {\n mesh.setVerticesData(VertexBuffer.UVKind, parsedGeometry.uvs, parsedGeometry.uvs._updatable);\n }\n if (parsedGeometry.uvs2) {\n mesh.setVerticesData(VertexBuffer.UV2Kind, parsedGeometry.uvs2, parsedGeometry.uvs2._updatable);\n }\n if (parsedGeometry.uvs3) {\n mesh.setVerticesData(VertexBuffer.UV3Kind, parsedGeometry.uvs3, parsedGeometry.uvs3._updatable);\n }\n if (parsedGeometry.uvs4) {\n mesh.setVerticesData(VertexBuffer.UV4Kind, parsedGeometry.uvs4, parsedGeometry.uvs4._updatable);\n }\n if (parsedGeometry.uvs5) {\n mesh.setVerticesData(VertexBuffer.UV5Kind, parsedGeometry.uvs5, parsedGeometry.uvs5._updatable);\n }\n if (parsedGeometry.uvs6) {\n mesh.setVerticesData(VertexBuffer.UV6Kind, parsedGeometry.uvs6, parsedGeometry.uvs6._updatable);\n }\n if (parsedGeometry.colors) {\n mesh.setVerticesData(VertexBuffer.ColorKind, Color4.CheckColors4(parsedGeometry.colors, parsedGeometry.positions.length / 3), parsedGeometry.colors._updatable);\n }\n if (parsedGeometry.matricesIndices) {\n if (!parsedGeometry.matricesIndices._isExpanded) {\n const floatIndices = [];\n for (let i = 0; i < parsedGeometry.matricesIndices.length; i++) {\n const matricesIndex = parsedGeometry.matricesIndices[i];\n floatIndices.push(matricesIndex & 0x000000ff);\n floatIndices.push((matricesIndex & 0x0000ff00) >> 8);\n floatIndices.push((matricesIndex & 0x00ff0000) >> 16);\n floatIndices.push((matricesIndex >> 24) & 0xff); // & 0xFF to convert to v + 256 if v < 0\n }\n mesh.setVerticesData(VertexBuffer.MatricesIndicesKind, floatIndices, parsedGeometry.matricesIndices._updatable);\n }\n else {\n delete parsedGeometry.matricesIndices._isExpanded;\n mesh.setVerticesData(VertexBuffer.MatricesIndicesKind, parsedGeometry.matricesIndices, parsedGeometry.matricesIndices._updatable);\n }\n }\n if (parsedGeometry.matricesIndicesExtra) {\n if (!parsedGeometry.matricesIndicesExtra._isExpanded) {\n const floatIndices = [];\n for (let i = 0; i < parsedGeometry.matricesIndicesExtra.length; i++) {\n const matricesIndex = parsedGeometry.matricesIndicesExtra[i];\n floatIndices.push(matricesIndex & 0x000000ff);\n floatIndices.push((matricesIndex & 0x0000ff00) >> 8);\n floatIndices.push((matricesIndex & 0x00ff0000) >> 16);\n floatIndices.push((matricesIndex >> 24) & 0xff); // & 0xFF to convert to v + 256 if v < 0\n }\n mesh.setVerticesData(VertexBuffer.MatricesIndicesExtraKind, floatIndices, parsedGeometry.matricesIndicesExtra._updatable);\n }\n else {\n delete parsedGeometry.matricesIndices._isExpanded;\n mesh.setVerticesData(VertexBuffer.MatricesIndicesExtraKind, parsedGeometry.matricesIndicesExtra, parsedGeometry.matricesIndicesExtra._updatable);\n }\n }\n if (parsedGeometry.matricesWeights) {\n Geometry._CleanMatricesWeights(parsedGeometry, mesh);\n mesh.setVerticesData(VertexBuffer.MatricesWeightsKind, parsedGeometry.matricesWeights, parsedGeometry.matricesWeights._updatable);\n }\n if (parsedGeometry.matricesWeightsExtra) {\n mesh.setVerticesData(VertexBuffer.MatricesWeightsExtraKind, parsedGeometry.matricesWeightsExtra, parsedGeometry.matricesWeights._updatable);\n }\n mesh.setIndices(parsedGeometry.indices, null);\n }\n // SubMeshes\n if (parsedGeometry.subMeshes) {\n mesh.subMeshes = [];\n for (let subIndex = 0; subIndex < parsedGeometry.subMeshes.length; subIndex++) {\n const parsedSubMesh = parsedGeometry.subMeshes[subIndex];\n SubMesh.AddToMesh(parsedSubMesh.materialIndex, parsedSubMesh.verticesStart, parsedSubMesh.verticesCount, parsedSubMesh.indexStart, parsedSubMesh.indexCount, mesh);\n }\n }\n // Flat shading\n if (mesh._shouldGenerateFlatShading) {\n mesh.convertToFlatShadedMesh();\n mesh._shouldGenerateFlatShading = false;\n }\n // Update\n mesh.computeWorldMatrix(true);\n scene.onMeshImportedObservable.notifyObservers(mesh);\n }\n static _CleanMatricesWeights(parsedGeometry, mesh) {\n const epsilon = 1e-3;\n if (!SceneLoaderFlags.CleanBoneMatrixWeights) {\n return;\n }\n let noInfluenceBoneIndex = 0.0;\n if (parsedGeometry.skeletonId > -1) {\n const skeleton = mesh.getScene().getLastSkeletonById(parsedGeometry.skeletonId);\n if (!skeleton) {\n return;\n }\n noInfluenceBoneIndex = skeleton.bones.length;\n }\n else {\n return;\n }\n const matricesIndices = mesh.getVerticesData(VertexBuffer.MatricesIndicesKind);\n const matricesIndicesExtra = mesh.getVerticesData(VertexBuffer.MatricesIndicesExtraKind);\n const matricesWeights = parsedGeometry.matricesWeights;\n const matricesWeightsExtra = parsedGeometry.matricesWeightsExtra;\n const influencers = parsedGeometry.numBoneInfluencer;\n const size = matricesWeights.length;\n for (let i = 0; i < size; i += 4) {\n let weight = 0.0;\n let firstZeroWeight = -1;\n for (let j = 0; j < 4; j++) {\n const w = matricesWeights[i + j];\n weight += w;\n if (w < epsilon && firstZeroWeight < 0) {\n firstZeroWeight = j;\n }\n }\n if (matricesWeightsExtra) {\n for (let j = 0; j < 4; j++) {\n const w = matricesWeightsExtra[i + j];\n weight += w;\n if (w < epsilon && firstZeroWeight < 0) {\n firstZeroWeight = j + 4;\n }\n }\n }\n if (firstZeroWeight < 0 || firstZeroWeight > influencers - 1) {\n firstZeroWeight = influencers - 1;\n }\n if (weight > epsilon) {\n const mweight = 1.0 / weight;\n for (let j = 0; j < 4; j++) {\n matricesWeights[i + j] *= mweight;\n }\n if (matricesWeightsExtra) {\n for (let j = 0; j < 4; j++) {\n matricesWeightsExtra[i + j] *= mweight;\n }\n }\n }\n else {\n if (firstZeroWeight >= 4) {\n matricesWeightsExtra[i + firstZeroWeight - 4] = 1.0 - weight;\n matricesIndicesExtra[i + firstZeroWeight - 4] = noInfluenceBoneIndex;\n }\n else {\n matricesWeights[i + firstZeroWeight] = 1.0 - weight;\n matricesIndices[i + firstZeroWeight] = noInfluenceBoneIndex;\n }\n }\n }\n mesh.setVerticesData(VertexBuffer.MatricesIndicesKind, matricesIndices);\n if (parsedGeometry.matricesWeightsExtra) {\n mesh.setVerticesData(VertexBuffer.MatricesIndicesExtraKind, matricesIndicesExtra);\n }\n }\n /**\n * Create a new geometry from persisted data (Using .babylon file format)\n * @param parsedVertexData defines the persisted data\n * @param scene defines the hosting scene\n * @param rootUrl defines the root url to use to load assets (like delayed data)\n * @returns the new geometry object\n */\n static Parse(parsedVertexData, scene, rootUrl) {\n const geometry = new Geometry(parsedVertexData.id, scene, undefined, parsedVertexData.updatable);\n geometry._loadedUniqueId = parsedVertexData.uniqueId;\n if (Tags) {\n Tags.AddTagsTo(geometry, parsedVertexData.tags);\n }\n if (parsedVertexData.delayLoadingFile) {\n geometry.delayLoadState = 4;\n geometry.delayLoadingFile = rootUrl + parsedVertexData.delayLoadingFile;\n geometry._boundingInfo = new BoundingInfo(Vector3.FromArray(parsedVertexData.boundingBoxMinimum), Vector3.FromArray(parsedVertexData.boundingBoxMaximum));\n geometry._delayInfo = [];\n if (parsedVertexData.hasUVs) {\n geometry._delayInfo.push(VertexBuffer.UVKind);\n }\n if (parsedVertexData.hasUVs2) {\n geometry._delayInfo.push(VertexBuffer.UV2Kind);\n }\n if (parsedVertexData.hasUVs3) {\n geometry._delayInfo.push(VertexBuffer.UV3Kind);\n }\n if (parsedVertexData.hasUVs4) {\n geometry._delayInfo.push(VertexBuffer.UV4Kind);\n }\n if (parsedVertexData.hasUVs5) {\n geometry._delayInfo.push(VertexBuffer.UV5Kind);\n }\n if (parsedVertexData.hasUVs6) {\n geometry._delayInfo.push(VertexBuffer.UV6Kind);\n }\n if (parsedVertexData.hasColors) {\n geometry._delayInfo.push(VertexBuffer.ColorKind);\n }\n if (parsedVertexData.hasMatricesIndices) {\n geometry._delayInfo.push(VertexBuffer.MatricesIndicesKind);\n }\n if (parsedVertexData.hasMatricesWeights) {\n geometry._delayInfo.push(VertexBuffer.MatricesWeightsKind);\n }\n geometry._delayLoadingFunction = VertexData.ImportVertexData;\n }\n else {\n VertexData.ImportVertexData(parsedVertexData, geometry);\n }\n scene.pushGeometry(geometry, true);\n return geometry;\n }\n }\n\n /**\n * Performance monitor tracks rolling average frame-time and frame-time variance over a user defined sliding-window\n */\n class PerformanceMonitor {\n /**\n * constructor\n * @param frameSampleSize The number of samples required to saturate the sliding window\n */\n constructor(frameSampleSize = 30) {\n this._enabled = true;\n this._rollingFrameTime = new RollingAverage(frameSampleSize);\n }\n /**\n * Samples current frame\n * @param timeMs A timestamp in milliseconds of the current frame to compare with other frames\n */\n sampleFrame(timeMs = PrecisionDate.Now) {\n if (!this._enabled) {\n return;\n }\n if (this._lastFrameTimeMs != null) {\n const dt = timeMs - this._lastFrameTimeMs;\n this._rollingFrameTime.add(dt);\n }\n this._lastFrameTimeMs = timeMs;\n }\n /**\n * Returns the average frame time in milliseconds over the sliding window (or the subset of frames sampled so far)\n */\n get averageFrameTime() {\n return this._rollingFrameTime.average;\n }\n /**\n * Returns the variance frame time in milliseconds over the sliding window (or the subset of frames sampled so far)\n */\n get averageFrameTimeVariance() {\n return this._rollingFrameTime.variance;\n }\n /**\n * Returns the frame time of the most recent frame\n */\n get instantaneousFrameTime() {\n return this._rollingFrameTime.history(0);\n }\n /**\n * Returns the average framerate in frames per second over the sliding window (or the subset of frames sampled so far)\n */\n get averageFPS() {\n return 1000.0 / this._rollingFrameTime.average;\n }\n /**\n * Returns the average framerate in frames per second using the most recent frame time\n */\n get instantaneousFPS() {\n const history = this._rollingFrameTime.history(0);\n if (history === 0) {\n return 0;\n }\n return 1000.0 / history;\n }\n /**\n * Returns true if enough samples have been taken to completely fill the sliding window\n */\n get isSaturated() {\n return this._rollingFrameTime.isSaturated();\n }\n /**\n * Enables contributions to the sliding window sample set\n */\n enable() {\n this._enabled = true;\n }\n /**\n * Disables contributions to the sliding window sample set\n * Samples will not be interpolated over the disabled period\n */\n disable() {\n this._enabled = false;\n //clear last sample to avoid interpolating over the disabled period when next enabled\n this._lastFrameTimeMs = null;\n }\n /**\n * Returns true if sampling is enabled\n */\n get isEnabled() {\n return this._enabled;\n }\n /**\n * Resets performance monitor\n */\n reset() {\n //clear last sample to avoid interpolating over the disabled period when next enabled\n this._lastFrameTimeMs = null;\n //wipe record\n this._rollingFrameTime.reset();\n }\n }\n /**\n * RollingAverage\n *\n * Utility to efficiently compute the rolling average and variance over a sliding window of samples\n */\n class RollingAverage {\n /**\n * constructor\n * @param length The number of samples required to saturate the sliding window\n */\n constructor(length) {\n this._samples = new Array(length);\n this.reset();\n }\n /**\n * Adds a sample to the sample set\n * @param v The sample value\n */\n add(v) {\n //http://en.wikipedia.org/wiki/Algorithms_for_calculating_variance\n let delta;\n //we need to check if we've already wrapped round\n if (this.isSaturated()) {\n //remove bottom of stack from mean\n const bottomValue = this._samples[this._pos];\n delta = bottomValue - this.average;\n this.average -= delta / (this._sampleCount - 1);\n this._m2 -= delta * (bottomValue - this.average);\n }\n else {\n this._sampleCount++;\n }\n //add new value to mean\n delta = v - this.average;\n this.average += delta / this._sampleCount;\n this._m2 += delta * (v - this.average);\n //set the new variance\n this.variance = this._m2 / (this._sampleCount - 1);\n this._samples[this._pos] = v;\n this._pos++;\n this._pos %= this._samples.length; //positive wrap around\n }\n /**\n * Returns previously added values or null if outside of history or outside the sliding window domain\n * @param i Index in history. For example, pass 0 for the most recent value and 1 for the value before that\n * @returns Value previously recorded with add() or null if outside of range\n */\n history(i) {\n if (i >= this._sampleCount || i >= this._samples.length) {\n return 0;\n }\n const i0 = this._wrapPosition(this._pos - 1.0);\n return this._samples[this._wrapPosition(i0 - i)];\n }\n /**\n * Returns true if enough samples have been taken to completely fill the sliding window\n * @returns true if sample-set saturated\n */\n isSaturated() {\n return this._sampleCount >= this._samples.length;\n }\n /**\n * Resets the rolling average (equivalent to 0 samples taken so far)\n */\n reset() {\n this.average = 0;\n this.variance = 0;\n this._sampleCount = 0;\n this._pos = 0;\n this._m2 = 0;\n }\n /**\n * Wraps a value around the sample range boundaries\n * @param i Position in sample range, for example if the sample length is 5, and i is -3, then 2 will be returned.\n * @returns Wrapped position in sample range\n */\n _wrapPosition(i) {\n const max = this._samples.length;\n return ((i % max) + max) % max;\n }\n }\n\n ThinEngine.prototype.setAlphaConstants = function (r, g, b, a) {\n this._alphaState.setAlphaBlendConstants(r, g, b, a);\n };\n ThinEngine.prototype.setAlphaMode = function (mode, noDepthWriteChange = false) {\n if (this._alphaMode === mode) {\n if (!noDepthWriteChange) {\n // Make sure we still have the correct depth mask according to the alpha mode (a transparent material could have forced writting to the depth buffer, for instance)\n const depthMask = mode === 0;\n if (this.depthCullingState.depthMask !== depthMask) {\n this.depthCullingState.depthMask = depthMask;\n }\n }\n return;\n }\n switch (mode) {\n case 0:\n this._alphaState.alphaBlend = false;\n break;\n case 7:\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA, this._gl.ONE, this._gl.ONE);\n this._alphaState.alphaBlend = true;\n break;\n case 8:\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA, this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA);\n this._alphaState.alphaBlend = true;\n break;\n case 2:\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.SRC_ALPHA, this._gl.ONE_MINUS_SRC_ALPHA, this._gl.ONE, this._gl.ONE);\n this._alphaState.alphaBlend = true;\n break;\n case 6:\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE, this._gl.ZERO, this._gl.ONE);\n this._alphaState.alphaBlend = true;\n break;\n case 1:\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.SRC_ALPHA, this._gl.ONE, this._gl.ZERO, this._gl.ONE);\n this._alphaState.alphaBlend = true;\n break;\n case 3:\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.ZERO, this._gl.ONE_MINUS_SRC_COLOR, this._gl.ONE, this._gl.ONE);\n this._alphaState.alphaBlend = true;\n break;\n case 4:\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.DST_COLOR, this._gl.ZERO, this._gl.ONE, this._gl.ONE);\n this._alphaState.alphaBlend = true;\n break;\n case 5:\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.SRC_ALPHA, this._gl.ONE_MINUS_SRC_COLOR, this._gl.ONE, this._gl.ONE);\n this._alphaState.alphaBlend = true;\n break;\n case 9:\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.CONSTANT_COLOR, this._gl.ONE_MINUS_CONSTANT_COLOR, this._gl.CONSTANT_ALPHA, this._gl.ONE_MINUS_CONSTANT_ALPHA);\n this._alphaState.alphaBlend = true;\n break;\n case 10:\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE_MINUS_SRC_COLOR, this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA);\n this._alphaState.alphaBlend = true;\n break;\n case 11:\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE, this._gl.ONE, this._gl.ONE);\n this._alphaState.alphaBlend = true;\n break;\n case 12:\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.DST_ALPHA, this._gl.ONE, this._gl.ZERO, this._gl.ZERO);\n this._alphaState.alphaBlend = true;\n break;\n case 13:\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE_MINUS_DST_COLOR, this._gl.ONE_MINUS_SRC_COLOR, this._gl.ONE_MINUS_DST_ALPHA, this._gl.ONE_MINUS_SRC_ALPHA);\n this._alphaState.alphaBlend = true;\n break;\n case 14:\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA, this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA);\n this._alphaState.alphaBlend = true;\n break;\n case 15:\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE, this._gl.ONE, this._gl.ZERO);\n this._alphaState.alphaBlend = true;\n break;\n case 16:\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE_MINUS_DST_COLOR, this._gl.ONE_MINUS_SRC_COLOR, this._gl.ZERO, this._gl.ONE);\n this._alphaState.alphaBlend = true;\n break;\n case 17:\n // Same as ALPHA_COMBINE but accumulates (1 - alpha) values in the alpha channel for a later readout in order independant transparency\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.SRC_ALPHA, this._gl.ONE_MINUS_SRC_ALPHA, this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA);\n this._alphaState.alphaBlend = true;\n break;\n }\n if (!noDepthWriteChange) {\n this.depthCullingState.depthMask = mode === 0;\n }\n this._alphaMode = mode;\n };\n ThinEngine.prototype.getAlphaMode = function () {\n return this._alphaMode;\n };\n ThinEngine.prototype.setAlphaEquation = function (equation) {\n if (this._alphaEquation === equation) {\n return;\n }\n switch (equation) {\n case 0:\n this._alphaState.setAlphaEquationParameters(32774, 32774);\n break;\n case 1:\n this._alphaState.setAlphaEquationParameters(32778, 32778);\n break;\n case 2:\n this._alphaState.setAlphaEquationParameters(32779, 32779);\n break;\n case 3:\n this._alphaState.setAlphaEquationParameters(32776, 32776);\n break;\n case 4:\n this._alphaState.setAlphaEquationParameters(32775, 32775);\n break;\n case 5:\n this._alphaState.setAlphaEquationParameters(32775, 32774);\n break;\n }\n this._alphaEquation = equation;\n };\n ThinEngine.prototype.getAlphaEquation = function () {\n return this._alphaEquation;\n };\n\n /**\n * Allocate a typed array depending on a texture type. Optionally can copy existing data in the buffer.\n * @param type type of the texture\n * @param sizeOrDstBuffer size of the array OR an existing buffer that will be used as the destination of the copy (if copyBuffer is provided)\n * @param sizeInBytes true if the size of the array is given in bytes, false if it is the number of elements of the array\n * @param copyBuffer if provided, buffer to copy into the destination buffer (either a newly allocated buffer if sizeOrDstBuffer is a number or use sizeOrDstBuffer as the destination buffer otherwise)\n * @returns the allocated buffer or sizeOrDstBuffer if the latter is an ArrayBuffer\n */\n function allocateAndCopyTypedBuffer(type, sizeOrDstBuffer, sizeInBytes = false, copyBuffer) {\n switch (type) {\n case 3: {\n const buffer = sizeOrDstBuffer instanceof ArrayBuffer ? new Int8Array(sizeOrDstBuffer) : new Int8Array(sizeOrDstBuffer);\n if (copyBuffer) {\n buffer.set(new Int8Array(copyBuffer));\n }\n return buffer;\n }\n case 0: {\n const buffer = sizeOrDstBuffer instanceof ArrayBuffer ? new Uint8Array(sizeOrDstBuffer) : new Uint8Array(sizeOrDstBuffer);\n if (copyBuffer) {\n buffer.set(new Uint8Array(copyBuffer));\n }\n return buffer;\n }\n case 4: {\n const buffer = sizeOrDstBuffer instanceof ArrayBuffer ? new Int16Array(sizeOrDstBuffer) : new Int16Array(sizeInBytes ? sizeOrDstBuffer / 2 : sizeOrDstBuffer);\n if (copyBuffer) {\n buffer.set(new Int16Array(copyBuffer));\n }\n return buffer;\n }\n case 5:\n case 8:\n case 9:\n case 10:\n case 2: {\n const buffer = sizeOrDstBuffer instanceof ArrayBuffer ? new Uint16Array(sizeOrDstBuffer) : new Uint16Array(sizeInBytes ? sizeOrDstBuffer / 2 : sizeOrDstBuffer);\n if (copyBuffer) {\n buffer.set(new Uint16Array(copyBuffer));\n }\n return buffer;\n }\n case 6: {\n const buffer = sizeOrDstBuffer instanceof ArrayBuffer ? new Int32Array(sizeOrDstBuffer) : new Int32Array(sizeInBytes ? sizeOrDstBuffer / 4 : sizeOrDstBuffer);\n if (copyBuffer) {\n buffer.set(new Int32Array(copyBuffer));\n }\n return buffer;\n }\n case 7:\n case 11:\n case 12:\n case 13:\n case 14:\n case 15: {\n const buffer = sizeOrDstBuffer instanceof ArrayBuffer ? new Uint32Array(sizeOrDstBuffer) : new Uint32Array(sizeInBytes ? sizeOrDstBuffer / 4 : sizeOrDstBuffer);\n if (copyBuffer) {\n buffer.set(new Uint32Array(copyBuffer));\n }\n return buffer;\n }\n case 1: {\n const buffer = sizeOrDstBuffer instanceof ArrayBuffer ? new Float32Array(sizeOrDstBuffer) : new Float32Array(sizeInBytes ? sizeOrDstBuffer / 4 : sizeOrDstBuffer);\n if (copyBuffer) {\n buffer.set(new Float32Array(copyBuffer));\n }\n return buffer;\n }\n }\n const buffer = sizeOrDstBuffer instanceof ArrayBuffer ? new Uint8Array(sizeOrDstBuffer) : new Uint8Array(sizeOrDstBuffer);\n if (copyBuffer) {\n buffer.set(new Uint8Array(copyBuffer));\n }\n return buffer;\n }\n ThinEngine.prototype._readTexturePixelsSync = function (texture, width, height, faceIndex = -1, level = 0, buffer = null, flushRenderer = true, noDataConversion = false, x = 0, y = 0) {\n var _a, _b;\n const gl = this._gl;\n if (!gl) {\n throw new Error(\"Engine does not have gl rendering context.\");\n }\n if (!this._dummyFramebuffer) {\n const dummy = gl.createFramebuffer();\n if (!dummy) {\n throw new Error(\"Unable to create dummy framebuffer\");\n }\n this._dummyFramebuffer = dummy;\n }\n gl.bindFramebuffer(gl.FRAMEBUFFER, this._dummyFramebuffer);\n if (faceIndex > -1) {\n gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex, (_a = texture._hardwareTexture) === null || _a === void 0 ? void 0 : _a.underlyingResource, level);\n }\n else {\n gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, (_b = texture._hardwareTexture) === null || _b === void 0 ? void 0 : _b.underlyingResource, level);\n }\n let readType = texture.type !== undefined ? this._getWebGLTextureType(texture.type) : gl.UNSIGNED_BYTE;\n if (!noDataConversion) {\n switch (readType) {\n case gl.UNSIGNED_BYTE:\n if (!buffer) {\n buffer = new Uint8Array(4 * width * height);\n }\n readType = gl.UNSIGNED_BYTE;\n break;\n default:\n if (!buffer) {\n buffer = new Float32Array(4 * width * height);\n }\n readType = gl.FLOAT;\n break;\n }\n }\n else if (!buffer) {\n buffer = allocateAndCopyTypedBuffer(texture.type, 4 * width * height);\n }\n if (flushRenderer) {\n this.flushFramebuffer();\n }\n gl.readPixels(x, y, width, height, gl.RGBA, readType, buffer);\n gl.bindFramebuffer(gl.FRAMEBUFFER, this._currentFramebuffer);\n return buffer;\n };\n ThinEngine.prototype._readTexturePixels = function (texture, width, height, faceIndex = -1, level = 0, buffer = null, flushRenderer = true, noDataConversion = false, x = 0, y = 0) {\n return Promise.resolve(this._readTexturePixelsSync(texture, width, height, faceIndex, level, buffer, flushRenderer, noDataConversion, x, y));\n };\n\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n ThinEngine.prototype.updateDynamicIndexBuffer = function (indexBuffer, indices, offset = 0) {\n // Force cache update\n this._currentBoundBuffer[this._gl.ELEMENT_ARRAY_BUFFER] = null;\n this.bindIndexBuffer(indexBuffer);\n let view;\n if (indexBuffer.is32Bits) {\n // anything else than Uint32Array needs to be converted to Uint32Array\n view = indices instanceof Uint32Array ? indices : new Uint32Array(indices);\n }\n else {\n // anything else than Uint16Array needs to be converted to Uint16Array\n view = indices instanceof Uint16Array ? indices : new Uint16Array(indices);\n }\n this._gl.bufferData(this._gl.ELEMENT_ARRAY_BUFFER, view, this._gl.DYNAMIC_DRAW);\n this._resetIndexBufferBinding();\n };\n ThinEngine.prototype.updateDynamicVertexBuffer = function (vertexBuffer, data, byteOffset, byteLength) {\n this.bindArrayBuffer(vertexBuffer);\n if (byteOffset === undefined) {\n byteOffset = 0;\n }\n const dataLength = data.byteLength || data.length;\n if (byteLength === undefined || (byteLength >= dataLength && byteOffset === 0)) {\n if (data instanceof Array) {\n this._gl.bufferSubData(this._gl.ARRAY_BUFFER, byteOffset, new Float32Array(data));\n }\n else {\n this._gl.bufferSubData(this._gl.ARRAY_BUFFER, byteOffset, data);\n }\n }\n else {\n if (data instanceof Array) {\n this._gl.bufferSubData(this._gl.ARRAY_BUFFER, 0, new Float32Array(data).subarray(byteOffset, byteOffset + byteLength));\n }\n else {\n if (data instanceof ArrayBuffer) {\n data = new Uint8Array(data, byteOffset, byteLength);\n }\n else {\n data = new Uint8Array(data.buffer, data.byteOffset + byteOffset, byteLength);\n }\n this._gl.bufferSubData(this._gl.ARRAY_BUFFER, 0, data);\n }\n }\n this._resetVertexBufferBinding();\n };\n\n /**\n * The engine class is responsible for interfacing with all lower-level APIs such as WebGL and Audio\n */\n class Engine extends ThinEngine {\n /**\n * Returns the current npm package of the sdk\n */\n // Not mixed with Version for tooling purpose.\n static get NpmPackage() {\n return ThinEngine.NpmPackage;\n }\n /**\n * Returns the current version of the framework\n */\n static get Version() {\n return ThinEngine.Version;\n }\n /** Gets the list of created engines */\n static get Instances() {\n return EngineStore.Instances;\n }\n /**\n * Gets the latest created engine\n */\n static get LastCreatedEngine() {\n return EngineStore.LastCreatedEngine;\n }\n /**\n * Gets the latest created scene\n */\n static get LastCreatedScene() {\n return EngineStore.LastCreatedScene;\n }\n /** @internal */\n /**\n * Engine abstraction for loading and creating an image bitmap from a given source string.\n * @param imageSource source to load the image from.\n * @param options An object that sets options for the image's extraction.\n * @returns ImageBitmap.\n */\n _createImageBitmapFromSource(imageSource, options) {\n const promise = new Promise((resolve, reject) => {\n const image = new Image();\n image.onload = () => {\n image.decode().then(() => {\n this.createImageBitmap(image, options).then((imageBitmap) => {\n resolve(imageBitmap);\n });\n });\n };\n image.onerror = () => {\n reject(`Error loading image ${image.src}`);\n };\n image.src = imageSource;\n });\n return promise;\n }\n /**\n * Engine abstraction for createImageBitmap\n * @param image source for image\n * @param options An object that sets options for the image's extraction.\n * @returns ImageBitmap\n */\n createImageBitmap(image, options) {\n return createImageBitmap(image, options);\n }\n /**\n * Resize an image and returns the image data as an uint8array\n * @param image image to resize\n * @param bufferWidth destination buffer width\n * @param bufferHeight destination buffer height\n * @returns an uint8array containing RGBA values of bufferWidth * bufferHeight size\n */\n resizeImageBitmap(image, bufferWidth, bufferHeight) {\n const canvas = this.createCanvas(bufferWidth, bufferHeight);\n const context = canvas.getContext(\"2d\");\n if (!context) {\n throw new Error(\"Unable to get 2d context for resizeImageBitmap\");\n }\n context.drawImage(image, 0, 0);\n // Create VertexData from map data\n // Cast is due to wrong definition in lib.d.ts from ts 1.3 - https://github.com/Microsoft/TypeScript/issues/949\n const buffer = context.getImageData(0, 0, bufferWidth, bufferHeight).data;\n return buffer;\n }\n /**\n * Will flag all materials in all scenes in all engines as dirty to trigger new shader compilation\n * @param flag defines which part of the materials must be marked as dirty\n * @param predicate defines a predicate used to filter which materials should be affected\n */\n static MarkAllMaterialsAsDirty(flag, predicate) {\n for (let engineIndex = 0; engineIndex < Engine.Instances.length; engineIndex++) {\n const engine = Engine.Instances[engineIndex];\n for (let sceneIndex = 0; sceneIndex < engine.scenes.length; sceneIndex++) {\n engine.scenes[sceneIndex].markAllMaterialsAsDirty(flag, predicate);\n }\n }\n }\n /**\n * Method called to create the default loading screen.\n * This can be overridden in your own app.\n * @param canvas The rendering canvas element\n * @returns The loading screen\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n static DefaultLoadingScreenFactory(canvas) {\n throw _WarnImport(\"LoadingScreen\");\n }\n get _supportsHardwareTextureRescaling() {\n return !!Engine._RescalePostProcessFactory;\n }\n /**\n * Gets the performance monitor attached to this engine\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimize_your_scene#engineinstrumentation\n */\n get performanceMonitor() {\n return this._performanceMonitor;\n }\n /**\n * (WebGPU only) True (default) to be in compatibility mode, meaning rendering all existing scenes without artifacts (same rendering than WebGL).\n * Setting the property to false will improve performances but may not work in some scenes if some precautions are not taken.\n * See https://doc.babylonjs.com/setup/support/webGPU/webGPUOptimization/webGPUNonCompatibilityMode for more details\n */\n get compatibilityMode() {\n return this._compatibilityMode;\n }\n set compatibilityMode(mode) {\n // not supported in WebGL\n this._compatibilityMode = true;\n }\n // Events\n /**\n * Gets the HTML element used to attach event listeners\n * @returns a HTML element\n */\n getInputElement() {\n return this._renderingCanvas;\n }\n /**\n * Creates a new engine\n * @param canvasOrContext defines the canvas or WebGL context to use for rendering. If you provide a WebGL context, Babylon.js will not hook events on the canvas (like pointers, keyboards, etc...) so no event observables will be available. This is mostly used when Babylon.js is used as a plugin on a system which already used the WebGL context\n * @param antialias defines enable antialiasing (default: false)\n * @param options defines further options to be sent to the getContext() function\n * @param adaptToDeviceRatio defines whether to adapt to the device's viewport characteristics (default: false)\n */\n constructor(canvasOrContext, antialias, options, adaptToDeviceRatio = false) {\n super(canvasOrContext, antialias, options, adaptToDeviceRatio);\n // Members\n /**\n * Gets or sets a boolean to enable/disable IndexedDB support and avoid XHR on .manifest\n **/\n this.enableOfflineSupport = false;\n /**\n * Gets or sets a boolean to enable/disable checking manifest if IndexedDB support is enabled (js will always consider the database is up to date)\n **/\n this.disableManifestCheck = false;\n /**\n * Gets or sets a boolean to enable/disable the context menu (right-click) from appearing on the main canvas\n */\n this.disableContextMenu = true;\n /**\n * Gets the list of created scenes\n */\n this.scenes = new Array();\n /** @internal */\n this._virtualScenes = new Array();\n /**\n * Event raised when a new scene is created\n */\n this.onNewSceneAddedObservable = new Observable$1();\n /**\n * Gets the list of created postprocesses\n */\n this.postProcesses = new Array();\n /**\n * Gets a boolean indicating if the pointer is currently locked\n */\n this.isPointerLock = false;\n // Observables\n /**\n * Observable event triggered each time the rendering canvas is resized\n */\n this.onResizeObservable = new Observable$1();\n /**\n * Observable event triggered each time the canvas loses focus\n */\n this.onCanvasBlurObservable = new Observable$1();\n /**\n * Observable event triggered each time the canvas gains focus\n */\n this.onCanvasFocusObservable = new Observable$1();\n /**\n * Observable event triggered each time the canvas receives pointerout event\n */\n this.onCanvasPointerOutObservable = new Observable$1();\n /**\n * Observable raised when the engine begins a new frame\n */\n this.onBeginFrameObservable = new Observable$1();\n /**\n * If set, will be used to request the next animation frame for the render loop\n */\n this.customAnimationFrameRequester = null;\n /**\n * Observable raised when the engine ends the current frame\n */\n this.onEndFrameObservable = new Observable$1();\n /**\n * Observable raised when the engine is about to compile a shader\n */\n this.onBeforeShaderCompilationObservable = new Observable$1();\n /**\n * Observable raised when the engine has just compiled a shader\n */\n this.onAfterShaderCompilationObservable = new Observable$1();\n // Deterministic lockstepMaxSteps\n this._deterministicLockstep = false;\n this._lockstepMaxSteps = 4;\n this._timeStep = 1 / 60;\n // FPS\n this._fps = 60;\n this._deltaTime = 0;\n /** @internal */\n this._drawCalls = new PerfCounter();\n /** Gets or sets the tab index to set to the rendering canvas. 1 is the minimum value to set to be able to capture keyboard events */\n this.canvasTabIndex = 1;\n /**\n * Turn this value on if you want to pause FPS computation when in background\n */\n this.disablePerformanceMonitorInBackground = false;\n this._performanceMonitor = new PerformanceMonitor();\n this._compatibilityMode = true;\n /**\n * Gets or sets the current render pass id\n */\n this.currentRenderPassId = 0;\n this._renderPassNames = [\"main\"];\n Engine.Instances.push(this);\n if (!canvasOrContext) {\n return;\n }\n this._features.supportRenderPasses = true;\n options = this._creationOptions;\n if (canvasOrContext.getContext) {\n const canvas = canvasOrContext;\n this._sharedInit(canvas);\n this._connectVREvents();\n }\n // Load WebVR Devices\n this._prepareVRComponent();\n if (options.autoEnableWebVR) {\n this.initWebVR();\n }\n }\n _initGLContext() {\n super._initGLContext();\n this._rescalePostProcess = null;\n }\n /**\n * Shared initialization across engines types.\n * @param canvas The canvas associated with this instance of the engine.\n */\n _sharedInit(canvas) {\n super._sharedInit(canvas);\n this._onCanvasFocus = () => {\n this.onCanvasFocusObservable.notifyObservers(this);\n };\n this._onCanvasBlur = () => {\n this.onCanvasBlurObservable.notifyObservers(this);\n };\n this._onCanvasContextMenu = (evt) => {\n if (this.disableContextMenu) {\n evt.preventDefault();\n }\n };\n canvas.addEventListener(\"focus\", this._onCanvasFocus);\n canvas.addEventListener(\"blur\", this._onCanvasBlur);\n canvas.addEventListener(\"contextmenu\", this._onCanvasContextMenu);\n this._onBlur = () => {\n if (this.disablePerformanceMonitorInBackground) {\n this._performanceMonitor.disable();\n }\n this._windowIsBackground = true;\n };\n this._onFocus = () => {\n if (this.disablePerformanceMonitorInBackground) {\n this._performanceMonitor.enable();\n }\n this._windowIsBackground = false;\n };\n this._onCanvasPointerOut = (ev) => {\n // Check that the element at the point of the pointer out isn't the canvas and if it isn't, notify observers\n // Note: This is a workaround for a bug with Safari\n if (document.elementFromPoint(ev.clientX, ev.clientY) !== canvas) {\n this.onCanvasPointerOutObservable.notifyObservers(ev);\n }\n };\n const hostWindow = this.getHostWindow(); // it calls IsWindowObjectExist()\n if (hostWindow && typeof hostWindow.addEventListener === \"function\") {\n hostWindow.addEventListener(\"blur\", this._onBlur);\n hostWindow.addEventListener(\"focus\", this._onFocus);\n }\n canvas.addEventListener(\"pointerout\", this._onCanvasPointerOut);\n if (!this._creationOptions.doNotHandleTouchAction) {\n this._disableTouchAction();\n }\n // Create Audio Engine if needed.\n if (!Engine.audioEngine && this._creationOptions.audioEngine && Engine.AudioEngineFactory) {\n Engine.audioEngine = Engine.AudioEngineFactory(this.getRenderingCanvas(), this.getAudioContext(), this.getAudioDestination());\n }\n if (IsDocumentAvailable()) {\n // Fullscreen\n this._onFullscreenChange = () => {\n this.isFullscreen = !!document.fullscreenElement;\n // Pointer lock\n if (this.isFullscreen && this._pointerLockRequested && canvas) {\n Engine._RequestPointerlock(canvas);\n }\n };\n document.addEventListener(\"fullscreenchange\", this._onFullscreenChange, false);\n document.addEventListener(\"webkitfullscreenchange\", this._onFullscreenChange, false);\n // Pointer lock\n this._onPointerLockChange = () => {\n this.isPointerLock = document.pointerLockElement === canvas;\n };\n document.addEventListener(\"pointerlockchange\", this._onPointerLockChange, false);\n document.addEventListener(\"webkitpointerlockchange\", this._onPointerLockChange, false);\n }\n this.enableOfflineSupport = Engine.OfflineProviderFactory !== undefined;\n this._deterministicLockstep = !!this._creationOptions.deterministicLockstep;\n this._lockstepMaxSteps = this._creationOptions.lockstepMaxSteps || 0;\n this._timeStep = this._creationOptions.timeStep || 1 / 60;\n }\n /** @internal */\n _verifyPointerLock() {\n var _a;\n (_a = this._onPointerLockChange) === null || _a === void 0 ? void 0 : _a.call(this);\n }\n /**\n * Gets current aspect ratio\n * @param viewportOwner defines the camera to use to get the aspect ratio\n * @param useScreen defines if screen size must be used (or the current render target if any)\n * @returns a number defining the aspect ratio\n */\n getAspectRatio(viewportOwner, useScreen = false) {\n const viewport = viewportOwner.viewport;\n return (this.getRenderWidth(useScreen) * viewport.width) / (this.getRenderHeight(useScreen) * viewport.height);\n }\n /**\n * Gets current screen aspect ratio\n * @returns a number defining the aspect ratio\n */\n getScreenAspectRatio() {\n return this.getRenderWidth(true) / this.getRenderHeight(true);\n }\n /**\n * Gets the client rect of the HTML canvas attached with the current webGL context\n * @returns a client rectangle\n */\n getRenderingCanvasClientRect() {\n if (!this._renderingCanvas) {\n return null;\n }\n return this._renderingCanvas.getBoundingClientRect();\n }\n /**\n * Gets the client rect of the HTML element used for events\n * @returns a client rectangle\n */\n getInputElementClientRect() {\n if (!this._renderingCanvas) {\n return null;\n }\n return this.getInputElement().getBoundingClientRect();\n }\n /**\n * Gets a boolean indicating that the engine is running in deterministic lock step mode\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep\n * @returns true if engine is in deterministic lock step mode\n */\n isDeterministicLockStep() {\n return this._deterministicLockstep;\n }\n /**\n * Gets the max steps when engine is running in deterministic lock step\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep\n * @returns the max steps\n */\n getLockstepMaxSteps() {\n return this._lockstepMaxSteps;\n }\n /**\n * Returns the time in ms between steps when using deterministic lock step.\n * @returns time step in (ms)\n */\n getTimeStep() {\n return this._timeStep * 1000;\n }\n /**\n * Force the mipmap generation for the given render target texture\n * @param texture defines the render target texture to use\n * @param unbind defines whether or not to unbind the texture after generation. Defaults to true.\n */\n generateMipMapsForCubemap(texture, unbind = true) {\n if (texture.generateMipMaps) {\n const gl = this._gl;\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, texture, true);\n gl.generateMipmap(gl.TEXTURE_CUBE_MAP);\n if (unbind) {\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, null);\n }\n }\n }\n /** States */\n /**\n * Gets a boolean indicating if depth writing is enabled\n * @returns the current depth writing state\n */\n getDepthWrite() {\n return this._depthCullingState.depthMask;\n }\n /**\n * Enable or disable depth writing\n * @param enable defines the state to set\n */\n setDepthWrite(enable) {\n this._depthCullingState.depthMask = enable;\n }\n /**\n * Gets a boolean indicating if stencil buffer is enabled\n * @returns the current stencil buffer state\n */\n getStencilBuffer() {\n return this._stencilState.stencilTest;\n }\n /**\n * Enable or disable the stencil buffer\n * @param enable defines if the stencil buffer must be enabled or disabled\n */\n setStencilBuffer(enable) {\n this._stencilState.stencilTest = enable;\n }\n /**\n * Gets the current stencil mask\n * @returns a number defining the new stencil mask to use\n */\n getStencilMask() {\n return this._stencilState.stencilMask;\n }\n /**\n * Sets the current stencil mask\n * @param mask defines the new stencil mask to use\n */\n setStencilMask(mask) {\n this._stencilState.stencilMask = mask;\n }\n /**\n * Gets the current stencil function\n * @returns a number defining the stencil function to use\n */\n getStencilFunction() {\n return this._stencilState.stencilFunc;\n }\n /**\n * Gets the current stencil reference value\n * @returns a number defining the stencil reference value to use\n */\n getStencilFunctionReference() {\n return this._stencilState.stencilFuncRef;\n }\n /**\n * Gets the current stencil mask\n * @returns a number defining the stencil mask to use\n */\n getStencilFunctionMask() {\n return this._stencilState.stencilFuncMask;\n }\n /**\n * Sets the current stencil function\n * @param stencilFunc defines the new stencil function to use\n */\n setStencilFunction(stencilFunc) {\n this._stencilState.stencilFunc = stencilFunc;\n }\n /**\n * Sets the current stencil reference\n * @param reference defines the new stencil reference to use\n */\n setStencilFunctionReference(reference) {\n this._stencilState.stencilFuncRef = reference;\n }\n /**\n * Sets the current stencil mask\n * @param mask defines the new stencil mask to use\n */\n setStencilFunctionMask(mask) {\n this._stencilState.stencilFuncMask = mask;\n }\n /**\n * Gets the current stencil operation when stencil fails\n * @returns a number defining stencil operation to use when stencil fails\n */\n getStencilOperationFail() {\n return this._stencilState.stencilOpStencilFail;\n }\n /**\n * Gets the current stencil operation when depth fails\n * @returns a number defining stencil operation to use when depth fails\n */\n getStencilOperationDepthFail() {\n return this._stencilState.stencilOpDepthFail;\n }\n /**\n * Gets the current stencil operation when stencil passes\n * @returns a number defining stencil operation to use when stencil passes\n */\n getStencilOperationPass() {\n return this._stencilState.stencilOpStencilDepthPass;\n }\n /**\n * Sets the stencil operation to use when stencil fails\n * @param operation defines the stencil operation to use when stencil fails\n */\n setStencilOperationFail(operation) {\n this._stencilState.stencilOpStencilFail = operation;\n }\n /**\n * Sets the stencil operation to use when depth fails\n * @param operation defines the stencil operation to use when depth fails\n */\n setStencilOperationDepthFail(operation) {\n this._stencilState.stencilOpDepthFail = operation;\n }\n /**\n * Sets the stencil operation to use when stencil passes\n * @param operation defines the stencil operation to use when stencil passes\n */\n setStencilOperationPass(operation) {\n this._stencilState.stencilOpStencilDepthPass = operation;\n }\n /**\n * Sets a boolean indicating if the dithering state is enabled or disabled\n * @param value defines the dithering state\n */\n setDitheringState(value) {\n if (value) {\n this._gl.enable(this._gl.DITHER);\n }\n else {\n this._gl.disable(this._gl.DITHER);\n }\n }\n /**\n * Sets a boolean indicating if the rasterizer state is enabled or disabled\n * @param value defines the rasterizer state\n */\n setRasterizerState(value) {\n if (value) {\n this._gl.disable(this._gl.RASTERIZER_DISCARD);\n }\n else {\n this._gl.enable(this._gl.RASTERIZER_DISCARD);\n }\n }\n /**\n * Gets the current depth function\n * @returns a number defining the depth function\n */\n getDepthFunction() {\n return this._depthCullingState.depthFunc;\n }\n /**\n * Sets the current depth function\n * @param depthFunc defines the function to use\n */\n setDepthFunction(depthFunc) {\n this._depthCullingState.depthFunc = depthFunc;\n }\n /**\n * Sets the current depth function to GREATER\n */\n setDepthFunctionToGreater() {\n this.setDepthFunction(516);\n }\n /**\n * Sets the current depth function to GEQUAL\n */\n setDepthFunctionToGreaterOrEqual() {\n this.setDepthFunction(518);\n }\n /**\n * Sets the current depth function to LESS\n */\n setDepthFunctionToLess() {\n this.setDepthFunction(513);\n }\n /**\n * Sets the current depth function to LEQUAL\n */\n setDepthFunctionToLessOrEqual() {\n this.setDepthFunction(515);\n }\n /**\n * Caches the the state of the stencil buffer\n */\n cacheStencilState() {\n this._cachedStencilBuffer = this.getStencilBuffer();\n this._cachedStencilFunction = this.getStencilFunction();\n this._cachedStencilMask = this.getStencilMask();\n this._cachedStencilOperationPass = this.getStencilOperationPass();\n this._cachedStencilOperationFail = this.getStencilOperationFail();\n this._cachedStencilOperationDepthFail = this.getStencilOperationDepthFail();\n this._cachedStencilReference = this.getStencilFunctionReference();\n }\n /**\n * Restores the state of the stencil buffer\n */\n restoreStencilState() {\n this.setStencilFunction(this._cachedStencilFunction);\n this.setStencilMask(this._cachedStencilMask);\n this.setStencilBuffer(this._cachedStencilBuffer);\n this.setStencilOperationPass(this._cachedStencilOperationPass);\n this.setStencilOperationFail(this._cachedStencilOperationFail);\n this.setStencilOperationDepthFail(this._cachedStencilOperationDepthFail);\n this.setStencilFunctionReference(this._cachedStencilReference);\n }\n /**\n * Directly set the WebGL Viewport\n * @param x defines the x coordinate of the viewport (in screen space)\n * @param y defines the y coordinate of the viewport (in screen space)\n * @param width defines the width of the viewport (in screen space)\n * @param height defines the height of the viewport (in screen space)\n * @returns the current viewport Object (if any) that is being replaced by this call. You can restore this viewport later on to go back to the original state\n */\n setDirectViewport(x, y, width, height) {\n const currentViewport = this._cachedViewport;\n this._cachedViewport = null;\n this._viewport(x, y, width, height);\n return currentViewport;\n }\n /**\n * Executes a scissor clear (ie. a clear on a specific portion of the screen)\n * @param x defines the x-coordinate of the bottom left corner of the clear rectangle\n * @param y defines the y-coordinate of the corner of the clear rectangle\n * @param width defines the width of the clear rectangle\n * @param height defines the height of the clear rectangle\n * @param clearColor defines the clear color\n */\n scissorClear(x, y, width, height, clearColor) {\n this.enableScissor(x, y, width, height);\n this.clear(clearColor, true, true, true);\n this.disableScissor();\n }\n /**\n * Enable scissor test on a specific rectangle (ie. render will only be executed on a specific portion of the screen)\n * @param x defines the x-coordinate of the bottom left corner of the clear rectangle\n * @param y defines the y-coordinate of the corner of the clear rectangle\n * @param width defines the width of the clear rectangle\n * @param height defines the height of the clear rectangle\n */\n enableScissor(x, y, width, height) {\n const gl = this._gl;\n // Change state\n gl.enable(gl.SCISSOR_TEST);\n gl.scissor(x, y, width, height);\n }\n /**\n * Disable previously set scissor test rectangle\n */\n disableScissor() {\n const gl = this._gl;\n gl.disable(gl.SCISSOR_TEST);\n }\n /**\n * @internal\n */\n _reportDrawCall(numDrawCalls = 1) {\n this._drawCalls.addCount(numDrawCalls, false);\n }\n /**\n * Initializes a webVR display and starts listening to display change events\n * The onVRDisplayChangedObservable will be notified upon these changes\n * @returns The onVRDisplayChangedObservable\n */\n initWebVR() {\n throw _WarnImport(\"WebVRCamera\");\n }\n /** @internal */\n _prepareVRComponent() {\n // Do nothing as the engine side effect will overload it\n }\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _connectVREvents(canvas, document) {\n // Do nothing as the engine side effect will overload it\n }\n /** @internal */\n _submitVRFrame() {\n // Do nothing as the engine side effect will overload it\n }\n /**\n * Call this function to leave webVR mode\n * Will do nothing if webVR is not supported or if there is no webVR device\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/webVRCamera\n */\n disableVR() {\n // Do nothing as the engine side effect will overload it\n }\n /**\n * Gets a boolean indicating that the system is in VR mode and is presenting\n * @returns true if VR mode is engaged\n */\n isVRPresenting() {\n return false;\n }\n /** @internal */\n _requestVRFrame() {\n // Do nothing as the engine side effect will overload it\n }\n /**\n * @internal\n */\n _loadFileAsync(url, offlineProvider, useArrayBuffer) {\n return new Promise((resolve, reject) => {\n this._loadFile(url, (data) => {\n resolve(data);\n }, undefined, offlineProvider, useArrayBuffer, (request, exception) => {\n reject(exception);\n });\n });\n }\n /**\n * Gets the source code of the vertex shader associated with a specific webGL program\n * @param program defines the program to use\n * @returns a string containing the source code of the vertex shader associated with the program\n */\n getVertexShaderSource(program) {\n const shaders = this._gl.getAttachedShaders(program);\n if (!shaders) {\n return null;\n }\n return this._gl.getShaderSource(shaders[0]);\n }\n /**\n * Gets the source code of the fragment shader associated with a specific webGL program\n * @param program defines the program to use\n * @returns a string containing the source code of the fragment shader associated with the program\n */\n getFragmentShaderSource(program) {\n const shaders = this._gl.getAttachedShaders(program);\n if (!shaders) {\n return null;\n }\n return this._gl.getShaderSource(shaders[1]);\n }\n /**\n * Sets a depth stencil texture from a render target to the according uniform.\n * @param channel The texture channel\n * @param uniform The uniform to set\n * @param texture The render target texture containing the depth stencil texture to apply\n * @param name The texture name\n */\n setDepthStencilTexture(channel, uniform, texture, name) {\n if (channel === undefined) {\n return;\n }\n if (uniform) {\n this._boundUniforms[channel] = uniform;\n }\n if (!texture || !texture.depthStencilTexture) {\n this._setTexture(channel, null, undefined, undefined, name);\n }\n else {\n this._setTexture(channel, texture, false, true, name);\n }\n }\n /**\n * Sets a texture to the webGL context from a postprocess\n * @param channel defines the channel to use\n * @param postProcess defines the source postprocess\n * @param name name of the channel\n */\n setTextureFromPostProcess(channel, postProcess, name) {\n var _a;\n let postProcessInput = null;\n if (postProcess) {\n if (postProcess._forcedOutputTexture) {\n postProcessInput = postProcess._forcedOutputTexture;\n }\n else if (postProcess._textures.data[postProcess._currentRenderTextureInd]) {\n postProcessInput = postProcess._textures.data[postProcess._currentRenderTextureInd];\n }\n }\n this._bindTexture(channel, (_a = postProcessInput === null || postProcessInput === void 0 ? void 0 : postProcessInput.texture) !== null && _a !== void 0 ? _a : null, name);\n }\n /**\n * Binds the output of the passed in post process to the texture channel specified\n * @param channel The channel the texture should be bound to\n * @param postProcess The post process which's output should be bound\n * @param name name of the channel\n */\n setTextureFromPostProcessOutput(channel, postProcess, name) {\n var _a, _b;\n this._bindTexture(channel, (_b = (_a = postProcess === null || postProcess === void 0 ? void 0 : postProcess._outputTexture) === null || _a === void 0 ? void 0 : _a.texture) !== null && _b !== void 0 ? _b : null, name);\n }\n _rebuildBuffers() {\n // Index / Vertex\n for (const scene of this.scenes) {\n scene.resetCachedMaterial();\n scene._rebuildGeometries();\n scene._rebuildTextures();\n }\n for (const scene of this._virtualScenes) {\n scene.resetCachedMaterial();\n scene._rebuildGeometries();\n scene._rebuildTextures();\n }\n super._rebuildBuffers();\n }\n /** @internal */\n _renderFrame() {\n for (let index = 0; index < this._activeRenderLoops.length; index++) {\n const renderFunction = this._activeRenderLoops[index];\n renderFunction();\n }\n }\n _cancelFrame() {\n if (this._renderingQueueLaunched && this.customAnimationFrameRequester) {\n this._renderingQueueLaunched = false;\n const { cancelAnimationFrame } = this.customAnimationFrameRequester;\n if (cancelAnimationFrame) {\n cancelAnimationFrame(this.customAnimationFrameRequester.requestID);\n }\n }\n else {\n super._cancelFrame();\n }\n }\n _renderLoop() {\n if (!this._contextWasLost) {\n let shouldRender = true;\n if (this.isDisposed || (!this.renderEvenInBackground && this._windowIsBackground)) {\n shouldRender = false;\n }\n if (shouldRender) {\n // Start new frame\n this.beginFrame();\n // Child canvases\n if (!this._renderViews()) {\n // Main frame\n this._renderFrame();\n }\n // Present\n this.endFrame();\n }\n }\n if (this._activeRenderLoops.length > 0) {\n // Register new frame\n if (this.customAnimationFrameRequester) {\n this.customAnimationFrameRequester.requestID = this._queueNewFrame(this.customAnimationFrameRequester.renderFunction || this._boundRenderFunction, this.customAnimationFrameRequester);\n this._frameHandler = this.customAnimationFrameRequester.requestID;\n }\n else if (this.isVRPresenting()) {\n this._requestVRFrame();\n }\n else {\n this._frameHandler = this._queueNewFrame(this._boundRenderFunction, this.getHostWindow());\n }\n }\n else {\n this._renderingQueueLaunched = false;\n }\n }\n /** @internal */\n _renderViews() {\n return false;\n }\n /**\n * Toggle full screen mode\n * @param requestPointerLock defines if a pointer lock should be requested from the user\n */\n switchFullscreen(requestPointerLock) {\n if (this.isFullscreen) {\n this.exitFullscreen();\n }\n else {\n this.enterFullscreen(requestPointerLock);\n }\n }\n /**\n * Enters full screen mode\n * @param requestPointerLock defines if a pointer lock should be requested from the user\n */\n enterFullscreen(requestPointerLock) {\n if (!this.isFullscreen) {\n this._pointerLockRequested = requestPointerLock;\n if (this._renderingCanvas) {\n Engine._RequestFullscreen(this._renderingCanvas);\n }\n }\n }\n /**\n * Exits full screen mode\n */\n exitFullscreen() {\n if (this.isFullscreen) {\n Engine._ExitFullscreen();\n }\n }\n /**\n * Enters Pointerlock mode\n */\n enterPointerlock() {\n if (this._renderingCanvas) {\n Engine._RequestPointerlock(this._renderingCanvas);\n }\n }\n /**\n * Exits Pointerlock mode\n */\n exitPointerlock() {\n Engine._ExitPointerlock();\n }\n /**\n * Begin a new frame\n */\n beginFrame() {\n this._measureFps();\n this.onBeginFrameObservable.notifyObservers(this);\n super.beginFrame();\n }\n /**\n * End the current frame\n */\n endFrame() {\n super.endFrame();\n this._submitVRFrame();\n this.onEndFrameObservable.notifyObservers(this);\n }\n /**\n * Resize the view according to the canvas' size\n * @param forceSetSize true to force setting the sizes of the underlying canvas\n */\n resize(forceSetSize = false) {\n // We're not resizing the size of the canvas while in VR mode & presenting\n if (this.isVRPresenting()) {\n return;\n }\n super.resize(forceSetSize);\n }\n /**\n * Force a specific size of the canvas\n * @param width defines the new canvas' width\n * @param height defines the new canvas' height\n * @param forceSetSize true to force setting the sizes of the underlying canvas\n * @returns true if the size was changed\n */\n setSize(width, height, forceSetSize = false) {\n if (!this._renderingCanvas) {\n return false;\n }\n if (!super.setSize(width, height, forceSetSize)) {\n return false;\n }\n if (this.scenes) {\n for (let index = 0; index < this.scenes.length; index++) {\n const scene = this.scenes[index];\n for (let camIndex = 0; camIndex < scene.cameras.length; camIndex++) {\n const cam = scene.cameras[camIndex];\n cam._currentRenderId = 0;\n }\n }\n if (this.onResizeObservable.hasObservers()) {\n this.onResizeObservable.notifyObservers(this);\n }\n }\n return true;\n }\n _deletePipelineContext(pipelineContext) {\n const webGLPipelineContext = pipelineContext;\n if (webGLPipelineContext && webGLPipelineContext.program) {\n if (webGLPipelineContext.transformFeedback) {\n this.deleteTransformFeedback(webGLPipelineContext.transformFeedback);\n webGLPipelineContext.transformFeedback = null;\n }\n }\n super._deletePipelineContext(pipelineContext);\n }\n createShaderProgram(pipelineContext, vertexCode, fragmentCode, defines, context, transformFeedbackVaryings = null) {\n context = context || this._gl;\n this.onBeforeShaderCompilationObservable.notifyObservers(this);\n const program = super.createShaderProgram(pipelineContext, vertexCode, fragmentCode, defines, context, transformFeedbackVaryings);\n this.onAfterShaderCompilationObservable.notifyObservers(this);\n return program;\n }\n _createShaderProgram(pipelineContext, vertexShader, fragmentShader, context, transformFeedbackVaryings = null) {\n const shaderProgram = context.createProgram();\n pipelineContext.program = shaderProgram;\n if (!shaderProgram) {\n throw new Error(\"Unable to create program\");\n }\n context.attachShader(shaderProgram, vertexShader);\n context.attachShader(shaderProgram, fragmentShader);\n if (this.webGLVersion > 1 && transformFeedbackVaryings) {\n const transformFeedback = this.createTransformFeedback();\n this.bindTransformFeedback(transformFeedback);\n this.setTranformFeedbackVaryings(shaderProgram, transformFeedbackVaryings);\n pipelineContext.transformFeedback = transformFeedback;\n }\n context.linkProgram(shaderProgram);\n if (this.webGLVersion > 1 && transformFeedbackVaryings) {\n this.bindTransformFeedback(null);\n }\n pipelineContext.context = context;\n pipelineContext.vertexShader = vertexShader;\n pipelineContext.fragmentShader = fragmentShader;\n if (!pipelineContext.isParallelCompiled) {\n this._finalizePipelineContext(pipelineContext);\n }\n return shaderProgram;\n }\n /**\n * @internal\n */\n _releaseTexture(texture) {\n super._releaseTexture(texture);\n }\n /**\n * @internal\n */\n _releaseRenderTargetWrapper(rtWrapper) {\n super._releaseRenderTargetWrapper(rtWrapper);\n // Set output texture of post process to null if the framebuffer has been released/disposed\n this.scenes.forEach((scene) => {\n scene.postProcesses.forEach((postProcess) => {\n if (postProcess._outputTexture === rtWrapper) {\n postProcess._outputTexture = null;\n }\n });\n scene.cameras.forEach((camera) => {\n camera._postProcesses.forEach((postProcess) => {\n if (postProcess) {\n if (postProcess._outputTexture === rtWrapper) {\n postProcess._outputTexture = null;\n }\n }\n });\n });\n });\n }\n /**\n * Gets the names of the render passes that are currently created\n * @returns list of the render pass names\n */\n getRenderPassNames() {\n return this._renderPassNames;\n }\n /**\n * Gets the name of the current render pass\n * @returns name of the current render pass\n */\n getCurrentRenderPassName() {\n return this._renderPassNames[this.currentRenderPassId];\n }\n /**\n * Creates a render pass id\n * @param name Name of the render pass (for debug purpose only)\n * @returns the id of the new render pass\n */\n createRenderPassId(name) {\n // Note: render pass id == 0 is always for the main render pass\n const id = ++Engine._RenderPassIdCounter;\n this._renderPassNames[id] = name !== null && name !== void 0 ? name : \"NONAME\";\n return id;\n }\n /**\n * Releases a render pass id\n * @param id id of the render pass to release\n */\n releaseRenderPassId(id) {\n this._renderPassNames[id] = undefined;\n for (let s = 0; s < this.scenes.length; ++s) {\n const scene = this.scenes[s];\n for (let m = 0; m < scene.meshes.length; ++m) {\n const mesh = scene.meshes[m];\n if (mesh.subMeshes) {\n for (let b = 0; b < mesh.subMeshes.length; ++b) {\n const subMesh = mesh.subMeshes[b];\n subMesh._removeDrawWrapper(id);\n }\n }\n }\n }\n }\n /**\n * @internal\n * Rescales a texture\n * @param source input texture\n * @param destination destination texture\n * @param scene scene to use to render the resize\n * @param internalFormat format to use when resizing\n * @param onComplete callback to be called when resize has completed\n */\n _rescaleTexture(source, destination, scene, internalFormat, onComplete) {\n this._gl.texParameteri(this._gl.TEXTURE_2D, this._gl.TEXTURE_MAG_FILTER, this._gl.LINEAR);\n this._gl.texParameteri(this._gl.TEXTURE_2D, this._gl.TEXTURE_MIN_FILTER, this._gl.LINEAR);\n this._gl.texParameteri(this._gl.TEXTURE_2D, this._gl.TEXTURE_WRAP_S, this._gl.CLAMP_TO_EDGE);\n this._gl.texParameteri(this._gl.TEXTURE_2D, this._gl.TEXTURE_WRAP_T, this._gl.CLAMP_TO_EDGE);\n const rtt = this.createRenderTargetTexture({\n width: destination.width,\n height: destination.height,\n }, {\n generateMipMaps: false,\n type: 0,\n samplingMode: 2,\n generateDepthBuffer: false,\n generateStencilBuffer: false,\n });\n if (!this._rescalePostProcess && Engine._RescalePostProcessFactory) {\n this._rescalePostProcess = Engine._RescalePostProcessFactory(this);\n }\n if (this._rescalePostProcess) {\n this._rescalePostProcess.externalTextureSamplerBinding = true;\n this._rescalePostProcess.getEffect().executeWhenCompiled(() => {\n this._rescalePostProcess.onApply = function (effect) {\n effect._bindTexture(\"textureSampler\", source);\n };\n let hostingScene = scene;\n if (!hostingScene) {\n hostingScene = this.scenes[this.scenes.length - 1];\n }\n hostingScene.postProcessManager.directRender([this._rescalePostProcess], rtt, true);\n this._bindTextureDirectly(this._gl.TEXTURE_2D, destination, true);\n this._gl.copyTexImage2D(this._gl.TEXTURE_2D, 0, internalFormat, 0, 0, destination.width, destination.height, 0);\n this.unBindFramebuffer(rtt);\n rtt.dispose();\n if (onComplete) {\n onComplete();\n }\n });\n }\n }\n // FPS\n /**\n * Gets the current framerate\n * @returns a number representing the framerate\n */\n getFps() {\n return this._fps;\n }\n /**\n * Gets the time spent between current and previous frame\n * @returns a number representing the delta time in ms\n */\n getDeltaTime() {\n return this._deltaTime;\n }\n _measureFps() {\n this._performanceMonitor.sampleFrame();\n this._fps = this._performanceMonitor.averageFPS;\n this._deltaTime = this._performanceMonitor.instantaneousFrameTime || 0;\n }\n /**\n * Wraps an external web gl texture in a Babylon texture.\n * @param texture defines the external texture\n * @param hasMipMaps defines whether the external texture has mip maps (default: false)\n * @param samplingMode defines the sampling mode for the external texture (default: 3)\n * @returns the babylon internal texture\n */\n wrapWebGLTexture(texture, hasMipMaps = false, samplingMode = 3) {\n const hardwareTexture = new WebGLHardwareTexture(texture, this._gl);\n const internalTexture = new InternalTexture(this, InternalTextureSource.Unknown, true);\n internalTexture._hardwareTexture = hardwareTexture;\n internalTexture.isReady = true;\n internalTexture.useMipMaps = hasMipMaps;\n this.updateTextureSamplingMode(samplingMode, internalTexture);\n return internalTexture;\n }\n /**\n * @internal\n */\n _uploadImageToTexture(texture, image, faceIndex = 0, lod = 0) {\n const gl = this._gl;\n const textureType = this._getWebGLTextureType(texture.type);\n const format = this._getInternalFormat(texture.format);\n const internalFormat = this._getRGBABufferInternalSizedFormat(texture.type, format);\n const bindTarget = texture.isCube ? gl.TEXTURE_CUBE_MAP : gl.TEXTURE_2D;\n this._bindTextureDirectly(bindTarget, texture, true);\n this._unpackFlipY(texture.invertY);\n let target = gl.TEXTURE_2D;\n if (texture.isCube) {\n target = gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex;\n }\n gl.texImage2D(target, lod, internalFormat, format, textureType, image);\n this._bindTextureDirectly(bindTarget, null, true);\n }\n /**\n * Updates a depth texture Comparison Mode and Function.\n * If the comparison Function is equal to 0, the mode will be set to none.\n * Otherwise, this only works in webgl 2 and requires a shadow sampler in the shader.\n * @param texture The texture to set the comparison function for\n * @param comparisonFunction The comparison function to set, 0 if no comparison required\n */\n updateTextureComparisonFunction(texture, comparisonFunction) {\n if (this.webGLVersion === 1) {\n Logger.Error(\"WebGL 1 does not support texture comparison.\");\n return;\n }\n const gl = this._gl;\n if (texture.isCube) {\n this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, texture, true);\n if (comparisonFunction === 0) {\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_COMPARE_FUNC, 515);\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_COMPARE_MODE, gl.NONE);\n }\n else {\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_COMPARE_FUNC, comparisonFunction);\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_COMPARE_MODE, gl.COMPARE_REF_TO_TEXTURE);\n }\n this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, null);\n }\n else {\n this._bindTextureDirectly(this._gl.TEXTURE_2D, texture, true);\n if (comparisonFunction === 0) {\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_COMPARE_FUNC, 515);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_COMPARE_MODE, gl.NONE);\n }\n else {\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_COMPARE_FUNC, comparisonFunction);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_COMPARE_MODE, gl.COMPARE_REF_TO_TEXTURE);\n }\n this._bindTextureDirectly(this._gl.TEXTURE_2D, null);\n }\n texture._comparisonFunction = comparisonFunction;\n }\n /**\n * Creates a webGL buffer to use with instantiation\n * @param capacity defines the size of the buffer\n * @returns the webGL buffer\n */\n createInstancesBuffer(capacity) {\n const buffer = this._gl.createBuffer();\n if (!buffer) {\n throw new Error(\"Unable to create instance buffer\");\n }\n const result = new WebGLDataBuffer(buffer);\n result.capacity = capacity;\n this.bindArrayBuffer(result);\n this._gl.bufferData(this._gl.ARRAY_BUFFER, capacity, this._gl.DYNAMIC_DRAW);\n result.references = 1;\n return result;\n }\n /**\n * Delete a webGL buffer used with instantiation\n * @param buffer defines the webGL buffer to delete\n */\n deleteInstancesBuffer(buffer) {\n this._gl.deleteBuffer(buffer);\n }\n _clientWaitAsync(sync, flags = 0, intervalms = 10) {\n const gl = this._gl;\n return new Promise((resolve, reject) => {\n const check = () => {\n const res = gl.clientWaitSync(sync, flags, 0);\n if (res == gl.WAIT_FAILED) {\n reject();\n return;\n }\n if (res == gl.TIMEOUT_EXPIRED) {\n setTimeout(check, intervalms);\n return;\n }\n resolve();\n };\n check();\n });\n }\n /**\n * @internal\n */\n _readPixelsAsync(x, y, w, h, format, type, outputBuffer) {\n if (this._webGLVersion < 2) {\n throw new Error(\"_readPixelsAsync only work on WebGL2+\");\n }\n const gl = this._gl;\n const buf = gl.createBuffer();\n gl.bindBuffer(gl.PIXEL_PACK_BUFFER, buf);\n gl.bufferData(gl.PIXEL_PACK_BUFFER, outputBuffer.byteLength, gl.STREAM_READ);\n gl.readPixels(x, y, w, h, format, type, 0);\n gl.bindBuffer(gl.PIXEL_PACK_BUFFER, null);\n const sync = gl.fenceSync(gl.SYNC_GPU_COMMANDS_COMPLETE, 0);\n if (!sync) {\n return null;\n }\n gl.flush();\n return this._clientWaitAsync(sync, 0, 10).then(() => {\n gl.deleteSync(sync);\n gl.bindBuffer(gl.PIXEL_PACK_BUFFER, buf);\n gl.getBufferSubData(gl.PIXEL_PACK_BUFFER, 0, outputBuffer);\n gl.bindBuffer(gl.PIXEL_PACK_BUFFER, null);\n gl.deleteBuffer(buf);\n return outputBuffer;\n });\n }\n dispose() {\n this.hideLoadingUI();\n this.onNewSceneAddedObservable.clear();\n // Release postProcesses\n while (this.postProcesses.length) {\n this.postProcesses[0].dispose();\n }\n // Rescale PP\n if (this._rescalePostProcess) {\n this._rescalePostProcess.dispose();\n }\n // Release scenes\n while (this.scenes.length) {\n this.scenes[0].dispose();\n }\n while (this._virtualScenes.length) {\n this._virtualScenes[0].dispose();\n }\n // Release audio engine\n if (EngineStore.Instances.length === 1 && Engine.audioEngine) {\n Engine.audioEngine.dispose();\n Engine.audioEngine = null;\n }\n //WebVR\n this.disableVR();\n // Events\n const hostWindow = this.getHostWindow(); // it calls IsWindowObjectExist()\n if (hostWindow && typeof hostWindow.removeEventListener === \"function\") {\n hostWindow.removeEventListener(\"blur\", this._onBlur);\n hostWindow.removeEventListener(\"focus\", this._onFocus);\n }\n if (this._renderingCanvas) {\n this._renderingCanvas.removeEventListener(\"focus\", this._onCanvasFocus);\n this._renderingCanvas.removeEventListener(\"blur\", this._onCanvasBlur);\n this._renderingCanvas.removeEventListener(\"pointerout\", this._onCanvasPointerOut);\n this._renderingCanvas.removeEventListener(\"contextmenu\", this._onCanvasContextMenu);\n }\n if (IsDocumentAvailable()) {\n document.removeEventListener(\"fullscreenchange\", this._onFullscreenChange);\n document.removeEventListener(\"mozfullscreenchange\", this._onFullscreenChange);\n document.removeEventListener(\"webkitfullscreenchange\", this._onFullscreenChange);\n document.removeEventListener(\"msfullscreenchange\", this._onFullscreenChange);\n document.removeEventListener(\"pointerlockchange\", this._onPointerLockChange);\n document.removeEventListener(\"mspointerlockchange\", this._onPointerLockChange);\n document.removeEventListener(\"mozpointerlockchange\", this._onPointerLockChange);\n document.removeEventListener(\"webkitpointerlockchange\", this._onPointerLockChange);\n }\n super.dispose();\n // Remove from Instances\n const index = EngineStore.Instances.indexOf(this);\n if (index >= 0) {\n EngineStore.Instances.splice(index, 1);\n }\n // no more engines left in the engine store? Notify!\n if (!Engine.Instances.length) {\n EngineStore.OnEnginesDisposedObservable.notifyObservers(this);\n }\n // Observables\n this.onResizeObservable.clear();\n this.onCanvasBlurObservable.clear();\n this.onCanvasFocusObservable.clear();\n this.onCanvasPointerOutObservable.clear();\n this.onBeginFrameObservable.clear();\n this.onEndFrameObservable.clear();\n }\n _disableTouchAction() {\n if (!this._renderingCanvas || !this._renderingCanvas.setAttribute) {\n return;\n }\n this._renderingCanvas.setAttribute(\"touch-action\", \"none\");\n this._renderingCanvas.style.touchAction = \"none\";\n this._renderingCanvas.style.webkitTapHighlightColor = \"transparent\";\n }\n // Loading screen\n /**\n * Display the loading screen\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/customLoadingScreen\n */\n displayLoadingUI() {\n if (!IsWindowObjectExist()) {\n return;\n }\n const loadingScreen = this.loadingScreen;\n if (loadingScreen) {\n loadingScreen.displayLoadingUI();\n }\n }\n /**\n * Hide the loading screen\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/customLoadingScreen\n */\n hideLoadingUI() {\n if (!IsWindowObjectExist()) {\n return;\n }\n const loadingScreen = this._loadingScreen;\n if (loadingScreen) {\n loadingScreen.hideLoadingUI();\n }\n }\n /**\n * Gets the current loading screen object\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/customLoadingScreen\n */\n get loadingScreen() {\n if (!this._loadingScreen && this._renderingCanvas) {\n this._loadingScreen = Engine.DefaultLoadingScreenFactory(this._renderingCanvas);\n }\n return this._loadingScreen;\n }\n /**\n * Sets the current loading screen object\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/customLoadingScreen\n */\n set loadingScreen(loadingScreen) {\n this._loadingScreen = loadingScreen;\n }\n /**\n * Sets the current loading screen text\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/customLoadingScreen\n */\n set loadingUIText(text) {\n this.loadingScreen.loadingUIText = text;\n }\n /**\n * Sets the current loading screen background color\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/customLoadingScreen\n */\n set loadingUIBackgroundColor(color) {\n this.loadingScreen.loadingUIBackgroundColor = color;\n }\n /**\n * creates and returns a new video element\n * @param constraints video constraints\n * @returns video element\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n createVideoElement(constraints) {\n return document.createElement(\"video\");\n }\n /** Pointerlock and fullscreen */\n /**\n * Ask the browser to promote the current element to pointerlock mode\n * @param element defines the DOM element to promote\n */\n static _RequestPointerlock(element) {\n if (element.requestPointerLock) {\n // In some browsers, requestPointerLock returns a promise.\n // Handle possible rejections to avoid an unhandled top-level exception.\n const promise = element.requestPointerLock();\n if (promise instanceof Promise)\n promise\n .then(() => {\n element.focus();\n })\n .catch(() => { });\n else\n element.focus();\n }\n }\n /**\n * Asks the browser to exit pointerlock mode\n */\n static _ExitPointerlock() {\n if (document.exitPointerLock) {\n document.exitPointerLock();\n }\n }\n /**\n * Ask the browser to promote the current element to fullscreen rendering mode\n * @param element defines the DOM element to promote\n */\n static _RequestFullscreen(element) {\n const requestFunction = element.requestFullscreen || element.webkitRequestFullscreen;\n if (!requestFunction) {\n return;\n }\n requestFunction.call(element);\n }\n /**\n * Asks the browser to exit fullscreen mode\n */\n static _ExitFullscreen() {\n const anyDoc = document;\n if (document.exitFullscreen) {\n document.exitFullscreen();\n }\n else if (anyDoc.webkitCancelFullScreen) {\n anyDoc.webkitCancelFullScreen();\n }\n }\n /**\n * Get Font size information\n * @param font font name\n * @returns an object containing ascent, height and descent\n */\n getFontOffset(font) {\n const text = document.createElement(\"span\");\n text.innerHTML = \"Hg\";\n text.setAttribute(\"style\", `font: ${font} !important`);\n const block = document.createElement(\"div\");\n block.style.display = \"inline-block\";\n block.style.width = \"1px\";\n block.style.height = \"0px\";\n block.style.verticalAlign = \"bottom\";\n const div = document.createElement(\"div\");\n div.style.whiteSpace = \"nowrap\";\n div.appendChild(text);\n div.appendChild(block);\n document.body.appendChild(div);\n let fontAscent = 0;\n let fontHeight = 0;\n try {\n fontHeight = block.getBoundingClientRect().top - text.getBoundingClientRect().top;\n block.style.verticalAlign = \"baseline\";\n fontAscent = block.getBoundingClientRect().top - text.getBoundingClientRect().top;\n }\n finally {\n document.body.removeChild(div);\n }\n return { ascent: fontAscent, height: fontHeight, descent: fontHeight - fontAscent };\n }\n }\n // Const statics\n /** Defines that alpha blending is disabled */\n Engine.ALPHA_DISABLE = 0;\n /** Defines that alpha blending to SRC ALPHA * SRC + DEST */\n Engine.ALPHA_ADD = 1;\n /** Defines that alpha blending to SRC ALPHA * SRC + (1 - SRC ALPHA) * DEST */\n Engine.ALPHA_COMBINE = 2;\n /** Defines that alpha blending to DEST - SRC * DEST */\n Engine.ALPHA_SUBTRACT = 3;\n /** Defines that alpha blending to SRC * DEST */\n Engine.ALPHA_MULTIPLY = 4;\n /** Defines that alpha blending to SRC ALPHA * SRC + (1 - SRC) * DEST */\n Engine.ALPHA_MAXIMIZED = 5;\n /** Defines that alpha blending to SRC + DEST */\n Engine.ALPHA_ONEONE = 6;\n /** Defines that alpha blending to SRC + (1 - SRC ALPHA) * DEST */\n Engine.ALPHA_PREMULTIPLIED = 7;\n /**\n * Defines that alpha blending to SRC + (1 - SRC ALPHA) * DEST\n * Alpha will be set to (1 - SRC ALPHA) * DEST ALPHA\n */\n Engine.ALPHA_PREMULTIPLIED_PORTERDUFF = 8;\n /** Defines that alpha blending to CST * SRC + (1 - CST) * DEST */\n Engine.ALPHA_INTERPOLATE = 9;\n /**\n * Defines that alpha blending to SRC + (1 - SRC) * DEST\n * Alpha will be set to SRC ALPHA + (1 - SRC ALPHA) * DEST ALPHA\n */\n Engine.ALPHA_SCREENMODE = 10;\n /** Defines that the resource is not delayed*/\n Engine.DELAYLOADSTATE_NONE = 0;\n /** Defines that the resource was successfully delay loaded */\n Engine.DELAYLOADSTATE_LOADED = 1;\n /** Defines that the resource is currently delay loading */\n Engine.DELAYLOADSTATE_LOADING = 2;\n /** Defines that the resource is delayed and has not started loading */\n Engine.DELAYLOADSTATE_NOTLOADED = 4;\n // Depht or Stencil test Constants.\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will never pass. i.e. Nothing will be drawn */\n Engine.NEVER = 512;\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will always pass. i.e. Pixels will be drawn in the order they are drawn */\n Engine.ALWAYS = 519;\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is less than the stored value */\n Engine.LESS = 513;\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is equals to the stored value */\n Engine.EQUAL = 514;\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is less than or equal to the stored value */\n Engine.LEQUAL = 515;\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is greater than the stored value */\n Engine.GREATER = 516;\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is greater than or equal to the stored value */\n Engine.GEQUAL = 518;\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is not equal to the stored value */\n Engine.NOTEQUAL = 517;\n // Stencil Actions Constants.\n /** Passed to stencilOperation to specify that stencil value must be kept */\n Engine.KEEP = 7680;\n /** Passed to stencilOperation to specify that stencil value must be replaced */\n Engine.REPLACE = 7681;\n /** Passed to stencilOperation to specify that stencil value must be incremented */\n Engine.INCR = 7682;\n /** Passed to stencilOperation to specify that stencil value must be decremented */\n Engine.DECR = 7683;\n /** Passed to stencilOperation to specify that stencil value must be inverted */\n Engine.INVERT = 5386;\n /** Passed to stencilOperation to specify that stencil value must be incremented with wrapping */\n Engine.INCR_WRAP = 34055;\n /** Passed to stencilOperation to specify that stencil value must be decremented with wrapping */\n Engine.DECR_WRAP = 34056;\n /** Texture is not repeating outside of 0..1 UVs */\n Engine.TEXTURE_CLAMP_ADDRESSMODE = 0;\n /** Texture is repeating outside of 0..1 UVs */\n Engine.TEXTURE_WRAP_ADDRESSMODE = 1;\n /** Texture is repeating and mirrored */\n Engine.TEXTURE_MIRROR_ADDRESSMODE = 2;\n /** ALPHA */\n Engine.TEXTUREFORMAT_ALPHA = 0;\n /** LUMINANCE */\n Engine.TEXTUREFORMAT_LUMINANCE = 1;\n /** LUMINANCE_ALPHA */\n Engine.TEXTUREFORMAT_LUMINANCE_ALPHA = 2;\n /** RGB */\n Engine.TEXTUREFORMAT_RGB = 4;\n /** RGBA */\n Engine.TEXTUREFORMAT_RGBA = 5;\n /** RED */\n Engine.TEXTUREFORMAT_RED = 6;\n /** RED (2nd reference) */\n Engine.TEXTUREFORMAT_R = 6;\n /** RG */\n Engine.TEXTUREFORMAT_RG = 7;\n /** RED_INTEGER */\n Engine.TEXTUREFORMAT_RED_INTEGER = 8;\n /** RED_INTEGER (2nd reference) */\n Engine.TEXTUREFORMAT_R_INTEGER = 8;\n /** RG_INTEGER */\n Engine.TEXTUREFORMAT_RG_INTEGER = 9;\n /** RGB_INTEGER */\n Engine.TEXTUREFORMAT_RGB_INTEGER = 10;\n /** RGBA_INTEGER */\n Engine.TEXTUREFORMAT_RGBA_INTEGER = 11;\n /** UNSIGNED_BYTE */\n Engine.TEXTURETYPE_UNSIGNED_BYTE = 0;\n /** UNSIGNED_BYTE (2nd reference) */\n Engine.TEXTURETYPE_UNSIGNED_INT = 0;\n /** FLOAT */\n Engine.TEXTURETYPE_FLOAT = 1;\n /** HALF_FLOAT */\n Engine.TEXTURETYPE_HALF_FLOAT = 2;\n /** BYTE */\n Engine.TEXTURETYPE_BYTE = 3;\n /** SHORT */\n Engine.TEXTURETYPE_SHORT = 4;\n /** UNSIGNED_SHORT */\n Engine.TEXTURETYPE_UNSIGNED_SHORT = 5;\n /** INT */\n Engine.TEXTURETYPE_INT = 6;\n /** UNSIGNED_INT */\n Engine.TEXTURETYPE_UNSIGNED_INTEGER = 7;\n /** UNSIGNED_SHORT_4_4_4_4 */\n Engine.TEXTURETYPE_UNSIGNED_SHORT_4_4_4_4 = 8;\n /** UNSIGNED_SHORT_5_5_5_1 */\n Engine.TEXTURETYPE_UNSIGNED_SHORT_5_5_5_1 = 9;\n /** UNSIGNED_SHORT_5_6_5 */\n Engine.TEXTURETYPE_UNSIGNED_SHORT_5_6_5 = 10;\n /** UNSIGNED_INT_2_10_10_10_REV */\n Engine.TEXTURETYPE_UNSIGNED_INT_2_10_10_10_REV = 11;\n /** UNSIGNED_INT_24_8 */\n Engine.TEXTURETYPE_UNSIGNED_INT_24_8 = 12;\n /** UNSIGNED_INT_10F_11F_11F_REV */\n Engine.TEXTURETYPE_UNSIGNED_INT_10F_11F_11F_REV = 13;\n /** UNSIGNED_INT_5_9_9_9_REV */\n Engine.TEXTURETYPE_UNSIGNED_INT_5_9_9_9_REV = 14;\n /** FLOAT_32_UNSIGNED_INT_24_8_REV */\n Engine.TEXTURETYPE_FLOAT_32_UNSIGNED_INT_24_8_REV = 15;\n /** nearest is mag = nearest and min = nearest and mip = none */\n Engine.TEXTURE_NEAREST_SAMPLINGMODE = 1;\n /** Bilinear is mag = linear and min = linear and mip = nearest */\n Engine.TEXTURE_BILINEAR_SAMPLINGMODE = 2;\n /** Trilinear is mag = linear and min = linear and mip = linear */\n Engine.TEXTURE_TRILINEAR_SAMPLINGMODE = 3;\n /** nearest is mag = nearest and min = nearest and mip = linear */\n Engine.TEXTURE_NEAREST_NEAREST_MIPLINEAR = 8;\n /** Bilinear is mag = linear and min = linear and mip = nearest */\n Engine.TEXTURE_LINEAR_LINEAR_MIPNEAREST = 11;\n /** Trilinear is mag = linear and min = linear and mip = linear */\n Engine.TEXTURE_LINEAR_LINEAR_MIPLINEAR = 3;\n /** mag = nearest and min = nearest and mip = nearest */\n Engine.TEXTURE_NEAREST_NEAREST_MIPNEAREST = 4;\n /** mag = nearest and min = linear and mip = nearest */\n Engine.TEXTURE_NEAREST_LINEAR_MIPNEAREST = 5;\n /** mag = nearest and min = linear and mip = linear */\n Engine.TEXTURE_NEAREST_LINEAR_MIPLINEAR = 6;\n /** mag = nearest and min = linear and mip = none */\n Engine.TEXTURE_NEAREST_LINEAR = 7;\n /** mag = nearest and min = nearest and mip = none */\n Engine.TEXTURE_NEAREST_NEAREST = 1;\n /** mag = linear and min = nearest and mip = nearest */\n Engine.TEXTURE_LINEAR_NEAREST_MIPNEAREST = 9;\n /** mag = linear and min = nearest and mip = linear */\n Engine.TEXTURE_LINEAR_NEAREST_MIPLINEAR = 10;\n /** mag = linear and min = linear and mip = none */\n Engine.TEXTURE_LINEAR_LINEAR = 2;\n /** mag = linear and min = nearest and mip = none */\n Engine.TEXTURE_LINEAR_NEAREST = 12;\n /** Explicit coordinates mode */\n Engine.TEXTURE_EXPLICIT_MODE = 0;\n /** Spherical coordinates mode */\n Engine.TEXTURE_SPHERICAL_MODE = 1;\n /** Planar coordinates mode */\n Engine.TEXTURE_PLANAR_MODE = 2;\n /** Cubic coordinates mode */\n Engine.TEXTURE_CUBIC_MODE = 3;\n /** Projection coordinates mode */\n Engine.TEXTURE_PROJECTION_MODE = 4;\n /** Skybox coordinates mode */\n Engine.TEXTURE_SKYBOX_MODE = 5;\n /** Inverse Cubic coordinates mode */\n Engine.TEXTURE_INVCUBIC_MODE = 6;\n /** Equirectangular coordinates mode */\n Engine.TEXTURE_EQUIRECTANGULAR_MODE = 7;\n /** Equirectangular Fixed coordinates mode */\n Engine.TEXTURE_FIXED_EQUIRECTANGULAR_MODE = 8;\n /** Equirectangular Fixed Mirrored coordinates mode */\n Engine.TEXTURE_FIXED_EQUIRECTANGULAR_MIRRORED_MODE = 9;\n // Texture rescaling mode\n /** Defines that texture rescaling will use a floor to find the closer power of 2 size */\n Engine.SCALEMODE_FLOOR = 1;\n /** Defines that texture rescaling will look for the nearest power of 2 size */\n Engine.SCALEMODE_NEAREST = 2;\n /** Defines that texture rescaling will use a ceil to find the closer power of 2 size */\n Engine.SCALEMODE_CEILING = 3;\n /**\n * Method called to create the default rescale post process on each engine.\n */\n Engine._RescalePostProcessFactory = null;\n // eslint-disable-next-line @typescript-eslint/naming-convention\n Engine._RenderPassIdCounter = 0;\n\n /** Defines supported spaces */\n var Space;\n (function (Space) {\n /** Local (object) space */\n Space[Space[\"LOCAL\"] = 0] = \"LOCAL\";\n /** World space */\n Space[Space[\"WORLD\"] = 1] = \"WORLD\";\n /** Bone space */\n Space[Space[\"BONE\"] = 2] = \"BONE\";\n })(Space || (Space = {}));\n /** Defines the 3 main axes */\n class Axis {\n }\n /** X axis */\n Axis.X = new Vector3(1.0, 0.0, 0.0);\n /** Y axis */\n Axis.Y = new Vector3(0.0, 1.0, 0.0);\n /** Z axis */\n Axis.Z = new Vector3(0.0, 0.0, 1.0);\n /**\n * Defines cartesian components.\n */\n var Coordinate;\n (function (Coordinate) {\n /** X axis */\n Coordinate[Coordinate[\"X\"] = 0] = \"X\";\n /** Y axis */\n Coordinate[Coordinate[\"Y\"] = 1] = \"Y\";\n /** Z axis */\n Coordinate[Coordinate[\"Z\"] = 2] = \"Z\";\n })(Coordinate || (Coordinate = {}));\n\n const convertRHSToLHS = Matrix.Compose(Vector3.One(), Quaternion.FromEulerAngles(0, Math.PI, 0), Vector3.Zero());\n /**\n * A TransformNode is an object that is not rendered but can be used as a center of transformation. This can decrease memory usage and increase rendering speed compared to using an empty mesh as a parent and is less complicated than using a pivot matrix.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/parent_pivot/transform_node\n */\n class TransformNode extends Node {\n /**\n * Gets or sets the billboard mode. Default is 0.\n *\n * | Value | Type | Description |\n * | --- | --- | --- |\n * | 0 | BILLBOARDMODE_NONE | |\n * | 1 | BILLBOARDMODE_X | |\n * | 2 | BILLBOARDMODE_Y | |\n * | 4 | BILLBOARDMODE_Z | |\n * | 7 | BILLBOARDMODE_ALL | |\n *\n */\n get billboardMode() {\n return this._billboardMode;\n }\n set billboardMode(value) {\n if (this._billboardMode === value) {\n return;\n }\n this._billboardMode = value;\n this._cache.useBillboardPosition = (this._billboardMode & TransformNode.BILLBOARDMODE_USE_POSITION) !== 0;\n this._computeUseBillboardPath();\n }\n /**\n * Gets or sets a boolean indicating that parent rotation should be preserved when using billboards.\n * This could be useful for glTF objects where parent rotation helps converting from right handed to left handed\n */\n get preserveParentRotationForBillboard() {\n return this._preserveParentRotationForBillboard;\n }\n set preserveParentRotationForBillboard(value) {\n if (value === this._preserveParentRotationForBillboard) {\n return;\n }\n this._preserveParentRotationForBillboard = value;\n this._computeUseBillboardPath();\n }\n _computeUseBillboardPath() {\n this._cache.useBillboardPath = this._billboardMode !== TransformNode.BILLBOARDMODE_NONE && !this.preserveParentRotationForBillboard;\n }\n /**\n * Gets or sets the distance of the object to max, often used by skybox\n */\n get infiniteDistance() {\n return this._infiniteDistance;\n }\n set infiniteDistance(value) {\n if (this._infiniteDistance === value) {\n return;\n }\n this._infiniteDistance = value;\n }\n constructor(name, scene = null, isPure = true) {\n super(name, scene);\n this._forward = new Vector3(0, 0, 1);\n this._up = new Vector3(0, 1, 0);\n this._right = new Vector3(1, 0, 0);\n // Properties\n this._position = Vector3.Zero();\n this._rotation = Vector3.Zero();\n this._rotationQuaternion = null;\n this._scaling = Vector3.One();\n this._transformToBoneReferal = null;\n this._isAbsoluteSynced = false;\n this._billboardMode = TransformNode.BILLBOARDMODE_NONE;\n this._preserveParentRotationForBillboard = false;\n /**\n * Multiplication factor on scale x/y/z when computing the world matrix. Eg. for a 1x1x1 cube setting this to 2 will make it a 2x2x2 cube\n */\n this.scalingDeterminant = 1;\n this._infiniteDistance = false;\n /**\n * Gets or sets a boolean indicating that non uniform scaling (when at least one component is different from others) should be ignored.\n * By default the system will update normals to compensate\n */\n this.ignoreNonUniformScaling = false;\n /**\n * Gets or sets a boolean indicating that even if rotationQuaternion is defined, you can keep updating rotation property and Babylon.js will just mix both\n */\n this.reIntegrateRotationIntoRotationQuaternion = false;\n // Cache\n /** @internal */\n this._poseMatrix = null;\n /** @internal */\n this._localMatrix = Matrix.Zero();\n this._usePivotMatrix = false;\n this._absolutePosition = Vector3.Zero();\n this._absoluteScaling = Vector3.Zero();\n this._absoluteRotationQuaternion = Quaternion.Identity();\n this._pivotMatrix = Matrix.Identity();\n /** @internal */\n this._postMultiplyPivotMatrix = false;\n this._isWorldMatrixFrozen = false;\n /** @internal */\n this._indexInSceneTransformNodesArray = -1;\n /**\n * An event triggered after the world matrix is updated\n */\n this.onAfterWorldMatrixUpdateObservable = new Observable$1();\n this._nonUniformScaling = false;\n if (isPure) {\n this.getScene().addTransformNode(this);\n }\n }\n /**\n * Gets a string identifying the name of the class\n * @returns \"TransformNode\" string\n */\n getClassName() {\n return \"TransformNode\";\n }\n /**\n * Gets or set the node position (default is (0.0, 0.0, 0.0))\n */\n get position() {\n return this._position;\n }\n set position(newPosition) {\n this._position = newPosition;\n this._isDirty = true;\n }\n /**\n * return true if a pivot has been set\n * @returns true if a pivot matrix is used\n */\n isUsingPivotMatrix() {\n return this._usePivotMatrix;\n }\n /**\n * Gets or sets the rotation property : a Vector3 defining the rotation value in radians around each local axis X, Y, Z (default is (0.0, 0.0, 0.0)).\n * If rotation quaternion is set, this Vector3 will be ignored and copy from the quaternion\n */\n get rotation() {\n return this._rotation;\n }\n set rotation(newRotation) {\n this._rotation = newRotation;\n this._rotationQuaternion = null;\n this._isDirty = true;\n }\n /**\n * Gets or sets the scaling property : a Vector3 defining the node scaling along each local axis X, Y, Z (default is (1.0, 1.0, 1.0)).\n */\n get scaling() {\n return this._scaling;\n }\n set scaling(newScaling) {\n this._scaling = newScaling;\n this._isDirty = true;\n }\n /**\n * Gets or sets the rotation Quaternion property : this a Quaternion object defining the node rotation by using a unit quaternion (undefined by default, but can be null).\n * If set, only the rotationQuaternion is then used to compute the node rotation (ie. node.rotation will be ignored)\n */\n get rotationQuaternion() {\n return this._rotationQuaternion;\n }\n set rotationQuaternion(quaternion) {\n this._rotationQuaternion = quaternion;\n //reset the rotation vector.\n if (quaternion) {\n this._rotation.setAll(0.0);\n }\n this._isDirty = true;\n }\n /**\n * The forward direction of that transform in world space.\n */\n get forward() {\n Vector3.TransformNormalFromFloatsToRef(0, 0, this.getScene().useRightHandedSystem ? -1.0 : 1.0, this.getWorldMatrix(), this._forward);\n return this._forward.normalize();\n }\n /**\n * The up direction of that transform in world space.\n */\n get up() {\n Vector3.TransformNormalFromFloatsToRef(0, 1, 0, this.getWorldMatrix(), this._up);\n return this._up.normalize();\n }\n /**\n * The right direction of that transform in world space.\n */\n get right() {\n Vector3.TransformNormalFromFloatsToRef(this.getScene().useRightHandedSystem ? -1.0 : 1.0, 0, 0, this.getWorldMatrix(), this._right);\n return this._right.normalize();\n }\n /**\n * Copies the parameter passed Matrix into the mesh Pose matrix.\n * @param matrix the matrix to copy the pose from\n * @returns this TransformNode.\n */\n updatePoseMatrix(matrix) {\n if (!this._poseMatrix) {\n this._poseMatrix = matrix.clone();\n return this;\n }\n this._poseMatrix.copyFrom(matrix);\n return this;\n }\n /**\n * Returns the mesh Pose matrix.\n * @returns the pose matrix\n */\n getPoseMatrix() {\n if (!this._poseMatrix) {\n this._poseMatrix = Matrix.Identity();\n }\n return this._poseMatrix;\n }\n /** @internal */\n _isSynchronized() {\n const cache = this._cache;\n if (this._billboardMode !== cache.billboardMode || this._billboardMode !== TransformNode.BILLBOARDMODE_NONE) {\n return false;\n }\n if (cache.pivotMatrixUpdated) {\n return false;\n }\n if (this._infiniteDistance) {\n return false;\n }\n if (this._position._isDirty) {\n return false;\n }\n if (this._scaling._isDirty) {\n return false;\n }\n if ((this._rotationQuaternion && this._rotationQuaternion._isDirty) || this._rotation._isDirty) {\n return false;\n }\n return true;\n }\n /** @internal */\n _initCache() {\n super._initCache();\n const cache = this._cache;\n cache.localMatrixUpdated = false;\n cache.billboardMode = -1;\n cache.infiniteDistance = false;\n cache.useBillboardPosition = false;\n cache.useBillboardPath = false;\n }\n /**\n * Returns the current mesh absolute position.\n * Returns a Vector3.\n */\n get absolutePosition() {\n return this.getAbsolutePosition();\n }\n /**\n * Returns the current mesh absolute scaling.\n * Returns a Vector3.\n */\n get absoluteScaling() {\n this._syncAbsoluteScalingAndRotation();\n return this._absoluteScaling;\n }\n /**\n * Returns the current mesh absolute rotation.\n * Returns a Quaternion.\n */\n get absoluteRotationQuaternion() {\n this._syncAbsoluteScalingAndRotation();\n return this._absoluteRotationQuaternion;\n }\n /**\n * Sets a new matrix to apply before all other transformation\n * @param matrix defines the transform matrix\n * @returns the current TransformNode\n */\n setPreTransformMatrix(matrix) {\n return this.setPivotMatrix(matrix, false);\n }\n /**\n * Sets a new pivot matrix to the current node\n * @param matrix defines the new pivot matrix to use\n * @param postMultiplyPivotMatrix defines if the pivot matrix must be cancelled in the world matrix. When this parameter is set to true (default), the inverse of the pivot matrix is also applied at the end to cancel the transformation effect\n * @returns the current TransformNode\n */\n setPivotMatrix(matrix, postMultiplyPivotMatrix = true) {\n this._pivotMatrix.copyFrom(matrix);\n this._usePivotMatrix = !this._pivotMatrix.isIdentity();\n this._cache.pivotMatrixUpdated = true;\n this._postMultiplyPivotMatrix = postMultiplyPivotMatrix;\n if (this._postMultiplyPivotMatrix) {\n if (!this._pivotMatrixInverse) {\n this._pivotMatrixInverse = Matrix.Invert(this._pivotMatrix);\n }\n else {\n this._pivotMatrix.invertToRef(this._pivotMatrixInverse);\n }\n }\n return this;\n }\n /**\n * Returns the mesh pivot matrix.\n * Default : Identity.\n * @returns the matrix\n */\n getPivotMatrix() {\n return this._pivotMatrix;\n }\n /**\n * Instantiate (when possible) or clone that node with its hierarchy\n * @param newParent defines the new parent to use for the instance (or clone)\n * @param options defines options to configure how copy is done\n * @param options.doNotInstantiate defines if the model must be instantiated or just cloned\n * @param onNewNodeCreated defines an option callback to call when a clone or an instance is created\n * @returns an instance (or a clone) of the current node with its hierarchy\n */\n instantiateHierarchy(newParent = null, options, onNewNodeCreated) {\n const clone = this.clone(\"Clone of \" + (this.name || this.id), newParent || this.parent, true);\n if (clone) {\n if (onNewNodeCreated) {\n onNewNodeCreated(this, clone);\n }\n }\n for (const child of this.getChildTransformNodes(true)) {\n child.instantiateHierarchy(clone, options, onNewNodeCreated);\n }\n return clone;\n }\n /**\n * Prevents the World matrix to be computed any longer\n * @param newWorldMatrix defines an optional matrix to use as world matrix\n * @param decompose defines whether to decompose the given newWorldMatrix or directly assign\n * @returns the TransformNode.\n */\n freezeWorldMatrix(newWorldMatrix = null, decompose = false) {\n if (newWorldMatrix) {\n if (decompose) {\n this._rotation.setAll(0);\n this._rotationQuaternion = this._rotationQuaternion || Quaternion.Identity();\n newWorldMatrix.decompose(this._scaling, this._rotationQuaternion, this._position);\n this.computeWorldMatrix(true);\n }\n else {\n this._worldMatrix = newWorldMatrix;\n this._absolutePosition.copyFromFloats(this._worldMatrix.m[12], this._worldMatrix.m[13], this._worldMatrix.m[14]);\n this._afterComputeWorldMatrix();\n }\n }\n else {\n this._isWorldMatrixFrozen = false; // no guarantee world is not already frozen, switch off temporarily\n this.computeWorldMatrix(true);\n }\n this._isDirty = false;\n this._isWorldMatrixFrozen = true;\n return this;\n }\n /**\n * Allows back the World matrix computation.\n * @returns the TransformNode.\n */\n unfreezeWorldMatrix() {\n this._isWorldMatrixFrozen = false;\n this.computeWorldMatrix(true);\n return this;\n }\n /**\n * True if the World matrix has been frozen.\n */\n get isWorldMatrixFrozen() {\n return this._isWorldMatrixFrozen;\n }\n /**\n * Returns the mesh absolute position in the World.\n * @returns a Vector3.\n */\n getAbsolutePosition() {\n this.computeWorldMatrix();\n return this._absolutePosition;\n }\n /**\n * Sets the mesh absolute position in the World from a Vector3 or an Array(3).\n * @param absolutePosition the absolute position to set\n * @returns the TransformNode.\n */\n setAbsolutePosition(absolutePosition) {\n if (!absolutePosition) {\n return this;\n }\n let absolutePositionX;\n let absolutePositionY;\n let absolutePositionZ;\n if (absolutePosition.x === undefined) {\n if (arguments.length < 3) {\n return this;\n }\n absolutePositionX = arguments[0];\n absolutePositionY = arguments[1];\n absolutePositionZ = arguments[2];\n }\n else {\n absolutePositionX = absolutePosition.x;\n absolutePositionY = absolutePosition.y;\n absolutePositionZ = absolutePosition.z;\n }\n if (this.parent) {\n const invertParentWorldMatrix = TmpVectors.Matrix[0];\n this.parent.getWorldMatrix().invertToRef(invertParentWorldMatrix);\n Vector3.TransformCoordinatesFromFloatsToRef(absolutePositionX, absolutePositionY, absolutePositionZ, invertParentWorldMatrix, this.position);\n }\n else {\n this.position.x = absolutePositionX;\n this.position.y = absolutePositionY;\n this.position.z = absolutePositionZ;\n }\n this._absolutePosition.copyFrom(absolutePosition);\n return this;\n }\n /**\n * Sets the mesh position in its local space.\n * @param vector3 the position to set in localspace\n * @returns the TransformNode.\n */\n setPositionWithLocalVector(vector3) {\n this.computeWorldMatrix();\n this.position = Vector3.TransformNormal(vector3, this._localMatrix);\n return this;\n }\n /**\n * Returns the mesh position in the local space from the current World matrix values.\n * @returns a new Vector3.\n */\n getPositionExpressedInLocalSpace() {\n this.computeWorldMatrix();\n const invLocalWorldMatrix = TmpVectors.Matrix[0];\n this._localMatrix.invertToRef(invLocalWorldMatrix);\n return Vector3.TransformNormal(this.position, invLocalWorldMatrix);\n }\n /**\n * Translates the mesh along the passed Vector3 in its local space.\n * @param vector3 the distance to translate in localspace\n * @returns the TransformNode.\n */\n locallyTranslate(vector3) {\n this.computeWorldMatrix(true);\n this.position = Vector3.TransformCoordinates(vector3, this._localMatrix);\n return this;\n }\n /**\n * Orients a mesh towards a target point. Mesh must be drawn facing user.\n * @param targetPoint the position (must be in same space as current mesh) to look at\n * @param yawCor optional yaw (y-axis) correction in radians\n * @param pitchCor optional pitch (x-axis) correction in radians\n * @param rollCor optional roll (z-axis) correction in radians\n * @param space the chosen space of the target\n * @returns the TransformNode.\n */\n lookAt(targetPoint, yawCor = 0, pitchCor = 0, rollCor = 0, space = Space.LOCAL) {\n const dv = TransformNode._LookAtVectorCache;\n const pos = space === Space.LOCAL ? this.position : this.getAbsolutePosition();\n targetPoint.subtractToRef(pos, dv);\n this.setDirection(dv, yawCor, pitchCor, rollCor);\n // Correct for parent's rotation offset\n if (space === Space.WORLD && this.parent) {\n if (this.rotationQuaternion) {\n // Get local rotation matrix of the looking object\n const rotationMatrix = TmpVectors.Matrix[0];\n this.rotationQuaternion.toRotationMatrix(rotationMatrix);\n // Offset rotation by parent's inverted rotation matrix to correct in world space\n const parentRotationMatrix = TmpVectors.Matrix[1];\n this.parent.getWorldMatrix().getRotationMatrixToRef(parentRotationMatrix);\n parentRotationMatrix.invert();\n rotationMatrix.multiplyToRef(parentRotationMatrix, rotationMatrix);\n this.rotationQuaternion.fromRotationMatrix(rotationMatrix);\n }\n else {\n // Get local rotation matrix of the looking object\n const quaternionRotation = TmpVectors.Quaternion[0];\n Quaternion.FromEulerVectorToRef(this.rotation, quaternionRotation);\n const rotationMatrix = TmpVectors.Matrix[0];\n quaternionRotation.toRotationMatrix(rotationMatrix);\n // Offset rotation by parent's inverted rotation matrix to correct in world space\n const parentRotationMatrix = TmpVectors.Matrix[1];\n this.parent.getWorldMatrix().getRotationMatrixToRef(parentRotationMatrix);\n parentRotationMatrix.invert();\n rotationMatrix.multiplyToRef(parentRotationMatrix, rotationMatrix);\n quaternionRotation.fromRotationMatrix(rotationMatrix);\n quaternionRotation.toEulerAnglesToRef(this.rotation);\n }\n }\n return this;\n }\n /**\n * Returns a new Vector3 that is the localAxis, expressed in the mesh local space, rotated like the mesh.\n * This Vector3 is expressed in the World space.\n * @param localAxis axis to rotate\n * @returns a new Vector3 that is the localAxis, expressed in the mesh local space, rotated like the mesh.\n */\n getDirection(localAxis) {\n const result = Vector3.Zero();\n this.getDirectionToRef(localAxis, result);\n return result;\n }\n /**\n * Sets the Vector3 \"result\" as the rotated Vector3 \"localAxis\" in the same rotation than the mesh.\n * localAxis is expressed in the mesh local space.\n * result is computed in the World space from the mesh World matrix.\n * @param localAxis axis to rotate\n * @param result the resulting transformnode\n * @returns this TransformNode.\n */\n getDirectionToRef(localAxis, result) {\n Vector3.TransformNormalToRef(localAxis, this.getWorldMatrix(), result);\n return this;\n }\n /**\n * Sets this transform node rotation to the given local axis.\n * @param localAxis the axis in local space\n * @param yawCor optional yaw (y-axis) correction in radians\n * @param pitchCor optional pitch (x-axis) correction in radians\n * @param rollCor optional roll (z-axis) correction in radians\n * @returns this TransformNode\n */\n setDirection(localAxis, yawCor = 0, pitchCor = 0, rollCor = 0) {\n const yaw = -Math.atan2(localAxis.z, localAxis.x) + Math.PI / 2;\n const len = Math.sqrt(localAxis.x * localAxis.x + localAxis.z * localAxis.z);\n const pitch = -Math.atan2(localAxis.y, len);\n if (this.rotationQuaternion) {\n Quaternion.RotationYawPitchRollToRef(yaw + yawCor, pitch + pitchCor, rollCor, this.rotationQuaternion);\n }\n else {\n this.rotation.x = pitch + pitchCor;\n this.rotation.y = yaw + yawCor;\n this.rotation.z = rollCor;\n }\n return this;\n }\n /**\n * Sets a new pivot point to the current node\n * @param point defines the new pivot point to use\n * @param space defines if the point is in world or local space (local by default)\n * @returns the current TransformNode\n */\n setPivotPoint(point, space = Space.LOCAL) {\n if (this.getScene().getRenderId() == 0) {\n this.computeWorldMatrix(true);\n }\n const wm = this.getWorldMatrix();\n if (space == Space.WORLD) {\n const tmat = TmpVectors.Matrix[0];\n wm.invertToRef(tmat);\n point = Vector3.TransformCoordinates(point, tmat);\n }\n return this.setPivotMatrix(Matrix.Translation(-point.x, -point.y, -point.z), true);\n }\n /**\n * Returns a new Vector3 set with the mesh pivot point coordinates in the local space.\n * @returns the pivot point\n */\n getPivotPoint() {\n const point = Vector3.Zero();\n this.getPivotPointToRef(point);\n return point;\n }\n /**\n * Sets the passed Vector3 \"result\" with the coordinates of the mesh pivot point in the local space.\n * @param result the vector3 to store the result\n * @returns this TransformNode.\n */\n getPivotPointToRef(result) {\n result.x = -this._pivotMatrix.m[12];\n result.y = -this._pivotMatrix.m[13];\n result.z = -this._pivotMatrix.m[14];\n return this;\n }\n /**\n * Returns a new Vector3 set with the mesh pivot point World coordinates.\n * @returns a new Vector3 set with the mesh pivot point World coordinates.\n */\n getAbsolutePivotPoint() {\n const point = Vector3.Zero();\n this.getAbsolutePivotPointToRef(point);\n return point;\n }\n /**\n * Sets the Vector3 \"result\" coordinates with the mesh pivot point World coordinates.\n * @param result vector3 to store the result\n * @returns this TransformNode.\n */\n getAbsolutePivotPointToRef(result) {\n this.getPivotPointToRef(result);\n Vector3.TransformCoordinatesToRef(result, this.getWorldMatrix(), result);\n return this;\n }\n /**\n * Flag the transform node as dirty (Forcing it to update everything)\n * @param property if set to \"rotation\" the objects rotationQuaternion will be set to null\n * @returns this node\n */\n markAsDirty(property) {\n if (this._isDirty) {\n return this;\n }\n // We need to explicitly update the children\n // as the scene.evaluateActiveMeshes will not poll the transform nodes\n if (this._children) {\n for (const child of this._children) {\n child.markAsDirty(property);\n }\n }\n return super.markAsDirty(property);\n }\n /**\n * Defines the passed node as the parent of the current node.\n * The node will remain exactly where it is and its position / rotation will be updated accordingly.\n * Note that if the mesh has a pivot matrix / point defined it will be applied after the parent was updated.\n * In that case the node will not remain in the same space as it is, as the pivot will be applied.\n * To avoid this, you can set updatePivot to true and the pivot will be updated to identity\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/parent_pivot/parent\n * @param node the node ot set as the parent\n * @param preserveScalingSign if true, keep scaling sign of child. Otherwise, scaling sign might change.\n * @param updatePivot if true, update the pivot matrix to keep the node in the same space as before\n * @returns this TransformNode.\n */\n setParent(node, preserveScalingSign = false, updatePivot = false) {\n if (!node && !this.parent) {\n return this;\n }\n const quatRotation = TmpVectors.Quaternion[0];\n const position = TmpVectors.Vector3[0];\n const scale = TmpVectors.Vector3[1];\n const invParentMatrix = TmpVectors.Matrix[1];\n Matrix.IdentityToRef(invParentMatrix);\n const composedMatrix = TmpVectors.Matrix[0];\n this.computeWorldMatrix(true);\n let currentRotation = this.rotationQuaternion;\n if (!currentRotation) {\n currentRotation = TransformNode._TmpRotation;\n Quaternion.RotationYawPitchRollToRef(this._rotation.y, this._rotation.x, this._rotation.z, currentRotation);\n }\n // current global transformation without pivot\n Matrix.ComposeToRef(this.scaling, currentRotation, this.position, composedMatrix);\n if (this.parent) {\n composedMatrix.multiplyToRef(this.parent.computeWorldMatrix(true), composedMatrix);\n }\n // is a node was set, calculate the difference between this and the node\n if (node) {\n node.computeWorldMatrix(true).invertToRef(invParentMatrix);\n composedMatrix.multiplyToRef(invParentMatrix, composedMatrix);\n }\n composedMatrix.decompose(scale, quatRotation, position, preserveScalingSign ? this : undefined);\n if (this.rotationQuaternion) {\n this.rotationQuaternion.copyFrom(quatRotation);\n }\n else {\n quatRotation.toEulerAnglesToRef(this.rotation);\n }\n this.scaling.copyFrom(scale);\n this.position.copyFrom(position);\n this.parent = node;\n if (updatePivot) {\n this.setPivotMatrix(Matrix.Identity());\n }\n return this;\n }\n /**\n * True if the scaling property of this object is non uniform eg. (1,2,1)\n */\n get nonUniformScaling() {\n return this._nonUniformScaling;\n }\n /**\n * @internal\n */\n _updateNonUniformScalingState(value) {\n if (this._nonUniformScaling === value) {\n return false;\n }\n this._nonUniformScaling = value;\n return true;\n }\n /**\n * Attach the current TransformNode to another TransformNode associated with a bone\n * @param bone Bone affecting the TransformNode\n * @param affectedTransformNode TransformNode associated with the bone\n * @returns this object\n */\n attachToBone(bone, affectedTransformNode) {\n this._currentParentWhenAttachingToBone = this.parent;\n this._transformToBoneReferal = affectedTransformNode;\n this.parent = bone;\n bone.getSkeleton().prepare(); // make sure bone.getFinalMatrix() is up to date\n if (bone.getFinalMatrix().determinant() < 0) {\n this.scalingDeterminant *= -1;\n }\n return this;\n }\n /**\n * Detach the transform node if its associated with a bone\n * @param resetToPreviousParent Indicates if the parent that was in effect when attachToBone was called should be set back or if we should set parent to null instead (defaults to the latter)\n * @returns this object\n */\n detachFromBone(resetToPreviousParent = false) {\n if (!this.parent) {\n if (resetToPreviousParent) {\n this.parent = this._currentParentWhenAttachingToBone;\n }\n return this;\n }\n if (this.parent.getWorldMatrix().determinant() < 0) {\n this.scalingDeterminant *= -1;\n }\n this._transformToBoneReferal = null;\n if (resetToPreviousParent) {\n this.parent = this._currentParentWhenAttachingToBone;\n }\n else {\n this.parent = null;\n }\n return this;\n }\n /**\n * Rotates the mesh around the axis vector for the passed angle (amount) expressed in radians, in the given space.\n * space (default LOCAL) can be either Space.LOCAL, either Space.WORLD.\n * Note that the property `rotationQuaternion` is then automatically updated and the property `rotation` is set to (0,0,0) and no longer used.\n * The passed axis is also normalized.\n * @param axis the axis to rotate around\n * @param amount the amount to rotate in radians\n * @param space Space to rotate in (Default: local)\n * @returns the TransformNode.\n */\n rotate(axis, amount, space) {\n axis.normalize();\n if (!this.rotationQuaternion) {\n this.rotationQuaternion = this.rotation.toQuaternion();\n this.rotation.setAll(0);\n }\n let rotationQuaternion;\n if (!space || space === Space.LOCAL) {\n rotationQuaternion = Quaternion.RotationAxisToRef(axis, amount, TransformNode._RotationAxisCache);\n this.rotationQuaternion.multiplyToRef(rotationQuaternion, this.rotationQuaternion);\n }\n else {\n if (this.parent) {\n const invertParentWorldMatrix = TmpVectors.Matrix[0];\n this.parent.getWorldMatrix().invertToRef(invertParentWorldMatrix);\n axis = Vector3.TransformNormal(axis, invertParentWorldMatrix);\n }\n rotationQuaternion = Quaternion.RotationAxisToRef(axis, amount, TransformNode._RotationAxisCache);\n rotationQuaternion.multiplyToRef(this.rotationQuaternion, this.rotationQuaternion);\n }\n return this;\n }\n /**\n * Rotates the mesh around the axis vector for the passed angle (amount) expressed in radians, in world space.\n * Note that the property `rotationQuaternion` is then automatically updated and the property `rotation` is set to (0,0,0) and no longer used.\n * The passed axis is also normalized. .\n * Method is based on http://www.euclideanspace.com/maths/geometry/affine/aroundPoint/index.htm\n * @param point the point to rotate around\n * @param axis the axis to rotate around\n * @param amount the amount to rotate in radians\n * @returns the TransformNode\n */\n rotateAround(point, axis, amount) {\n axis.normalize();\n if (!this.rotationQuaternion) {\n this.rotationQuaternion = Quaternion.RotationYawPitchRoll(this.rotation.y, this.rotation.x, this.rotation.z);\n this.rotation.setAll(0);\n }\n const tmpVector = TmpVectors.Vector3[0];\n const finalScale = TmpVectors.Vector3[1];\n const finalTranslation = TmpVectors.Vector3[2];\n const finalRotation = TmpVectors.Quaternion[0];\n const translationMatrix = TmpVectors.Matrix[0]; // T\n const translationMatrixInv = TmpVectors.Matrix[1]; // T'\n const rotationMatrix = TmpVectors.Matrix[2]; // R\n const finalMatrix = TmpVectors.Matrix[3]; // T' x R x T\n point.subtractToRef(this.position, tmpVector);\n Matrix.TranslationToRef(tmpVector.x, tmpVector.y, tmpVector.z, translationMatrix); // T\n Matrix.TranslationToRef(-tmpVector.x, -tmpVector.y, -tmpVector.z, translationMatrixInv); // T'\n Matrix.RotationAxisToRef(axis, amount, rotationMatrix); // R\n translationMatrixInv.multiplyToRef(rotationMatrix, finalMatrix); // T' x R\n finalMatrix.multiplyToRef(translationMatrix, finalMatrix); // T' x R x T\n finalMatrix.decompose(finalScale, finalRotation, finalTranslation);\n this.position.addInPlace(finalTranslation);\n finalRotation.multiplyToRef(this.rotationQuaternion, this.rotationQuaternion);\n return this;\n }\n /**\n * Translates the mesh along the axis vector for the passed distance in the given space.\n * space (default LOCAL) can be either Space.LOCAL, either Space.WORLD.\n * @param axis the axis to translate in\n * @param distance the distance to translate\n * @param space Space to rotate in (Default: local)\n * @returns the TransformNode.\n */\n translate(axis, distance, space) {\n const displacementVector = axis.scale(distance);\n if (!space || space === Space.LOCAL) {\n const tempV3 = this.getPositionExpressedInLocalSpace().add(displacementVector);\n this.setPositionWithLocalVector(tempV3);\n }\n else {\n this.setAbsolutePosition(this.getAbsolutePosition().add(displacementVector));\n }\n return this;\n }\n /**\n * Adds a rotation step to the mesh current rotation.\n * x, y, z are Euler angles expressed in radians.\n * This methods updates the current mesh rotation, either mesh.rotation, either mesh.rotationQuaternion if it's set.\n * This means this rotation is made in the mesh local space only.\n * It's useful to set a custom rotation order different from the BJS standard one YXZ.\n * Example : this rotates the mesh first around its local X axis, then around its local Z axis, finally around its local Y axis.\n * ```javascript\n * mesh.addRotation(x1, 0, 0).addRotation(0, 0, z2).addRotation(0, 0, y3);\n * ```\n * Note that `addRotation()` accumulates the passed rotation values to the current ones and computes the .rotation or .rotationQuaternion updated values.\n * Under the hood, only quaternions are used. So it's a little faster is you use .rotationQuaternion because it doesn't need to translate them back to Euler angles.\n * @param x Rotation to add\n * @param y Rotation to add\n * @param z Rotation to add\n * @returns the TransformNode.\n */\n addRotation(x, y, z) {\n let rotationQuaternion;\n if (this.rotationQuaternion) {\n rotationQuaternion = this.rotationQuaternion;\n }\n else {\n rotationQuaternion = TmpVectors.Quaternion[1];\n Quaternion.RotationYawPitchRollToRef(this.rotation.y, this.rotation.x, this.rotation.z, rotationQuaternion);\n }\n const accumulation = TmpVectors.Quaternion[0];\n Quaternion.RotationYawPitchRollToRef(y, x, z, accumulation);\n rotationQuaternion.multiplyInPlace(accumulation);\n if (!this.rotationQuaternion) {\n rotationQuaternion.toEulerAnglesToRef(this.rotation);\n }\n return this;\n }\n /**\n * @internal\n */\n _getEffectiveParent() {\n return this.parent;\n }\n /**\n * Returns whether the transform node world matrix computation needs the camera information to be computed.\n * This is the case when the node is a billboard or has an infinite distance for instance.\n * @returns true if the world matrix computation needs the camera information to be computed\n */\n isWorldMatrixCameraDependent() {\n return (this._infiniteDistance && !this.parent) || (this._billboardMode !== TransformNode.BILLBOARDMODE_NONE && !this.preserveParentRotationForBillboard);\n }\n /**\n * Computes the world matrix of the node\n * @param force defines if the cache version should be invalidated forcing the world matrix to be created from scratch\n * @param camera defines the camera used if different from the scene active camera (This is used with modes like Billboard or infinite distance)\n * @returns the world matrix\n */\n computeWorldMatrix(force = false, camera = null) {\n if (this._isWorldMatrixFrozen && !this._isDirty) {\n return this._worldMatrix;\n }\n const currentRenderId = this.getScene().getRenderId();\n if (!this._isDirty && !force && (this._currentRenderId === currentRenderId || this.isSynchronized())) {\n this._currentRenderId = currentRenderId;\n return this._worldMatrix;\n }\n camera = camera || this.getScene().activeCamera;\n this._updateCache();\n const cache = this._cache;\n cache.pivotMatrixUpdated = false;\n cache.billboardMode = this.billboardMode;\n cache.infiniteDistance = this.infiniteDistance;\n cache.parent = this._parentNode;\n this._currentRenderId = currentRenderId;\n this._childUpdateId += 1;\n this._isDirty = false;\n this._position._isDirty = false;\n this._rotation._isDirty = false;\n this._scaling._isDirty = false;\n const parent = this._getEffectiveParent();\n // Scaling\n const scaling = TransformNode._TmpScaling;\n let translation = this._position;\n // Translation\n if (this._infiniteDistance) {\n if (!this.parent && camera) {\n const cameraWorldMatrix = camera.getWorldMatrix();\n const cameraGlobalPosition = new Vector3(cameraWorldMatrix.m[12], cameraWorldMatrix.m[13], cameraWorldMatrix.m[14]);\n translation = TransformNode._TmpTranslation;\n translation.copyFromFloats(this._position.x + cameraGlobalPosition.x, this._position.y + cameraGlobalPosition.y, this._position.z + cameraGlobalPosition.z);\n }\n }\n // Scaling\n scaling.copyFromFloats(this._scaling.x * this.scalingDeterminant, this._scaling.y * this.scalingDeterminant, this._scaling.z * this.scalingDeterminant);\n // Rotation\n let rotation;\n if (this._rotationQuaternion) {\n this._rotationQuaternion._isDirty = false;\n rotation = this._rotationQuaternion;\n if (this.reIntegrateRotationIntoRotationQuaternion) {\n const len = this.rotation.lengthSquared();\n if (len) {\n this._rotationQuaternion.multiplyInPlace(Quaternion.RotationYawPitchRoll(this._rotation.y, this._rotation.x, this._rotation.z));\n this._rotation.copyFromFloats(0, 0, 0);\n }\n }\n }\n else {\n rotation = TransformNode._TmpRotation;\n Quaternion.RotationYawPitchRollToRef(this._rotation.y, this._rotation.x, this._rotation.z, rotation);\n }\n // Compose\n if (this._usePivotMatrix) {\n const scaleMatrix = TmpVectors.Matrix[1];\n Matrix.ScalingToRef(scaling.x, scaling.y, scaling.z, scaleMatrix);\n // Rotation\n const rotationMatrix = TmpVectors.Matrix[0];\n rotation.toRotationMatrix(rotationMatrix);\n // Composing transformations\n this._pivotMatrix.multiplyToRef(scaleMatrix, TmpVectors.Matrix[4]);\n TmpVectors.Matrix[4].multiplyToRef(rotationMatrix, this._localMatrix);\n // Post multiply inverse of pivotMatrix\n if (this._postMultiplyPivotMatrix) {\n this._localMatrix.multiplyToRef(this._pivotMatrixInverse, this._localMatrix);\n }\n this._localMatrix.addTranslationFromFloats(translation.x, translation.y, translation.z);\n }\n else {\n Matrix.ComposeToRef(scaling, rotation, translation, this._localMatrix);\n }\n // Parent\n if (parent && parent.getWorldMatrix) {\n if (force) {\n parent.computeWorldMatrix(force);\n }\n if (cache.useBillboardPath) {\n if (this._transformToBoneReferal) {\n parent.getWorldMatrix().multiplyToRef(this._transformToBoneReferal.getWorldMatrix(), TmpVectors.Matrix[7]);\n }\n else {\n TmpVectors.Matrix[7].copyFrom(parent.getWorldMatrix());\n }\n // Extract scaling and translation from parent\n const translation = TmpVectors.Vector3[5];\n const scale = TmpVectors.Vector3[6];\n const orientation = TmpVectors.Quaternion[0];\n TmpVectors.Matrix[7].decompose(scale, orientation, translation);\n Matrix.ScalingToRef(scale.x, scale.y, scale.z, TmpVectors.Matrix[7]);\n TmpVectors.Matrix[7].setTranslation(translation);\n if (TransformNode.BillboardUseParentOrientation) {\n // set localMatrix translation to be transformed against parent's orientation.\n this._position.applyRotationQuaternionToRef(orientation, translation);\n this._localMatrix.setTranslation(translation);\n }\n this._localMatrix.multiplyToRef(TmpVectors.Matrix[7], this._worldMatrix);\n }\n else {\n if (this._transformToBoneReferal) {\n this._localMatrix.multiplyToRef(parent.getWorldMatrix(), TmpVectors.Matrix[6]);\n TmpVectors.Matrix[6].multiplyToRef(this._transformToBoneReferal.getWorldMatrix(), this._worldMatrix);\n }\n else {\n this._localMatrix.multiplyToRef(parent.getWorldMatrix(), this._worldMatrix);\n }\n }\n this._markSyncedWithParent();\n }\n else {\n this._worldMatrix.copyFrom(this._localMatrix);\n }\n // Billboarding based on camera orientation (testing PG:http://www.babylonjs-playground.com/#UJEIL#13)\n if (cache.useBillboardPath && camera && this.billboardMode && !cache.useBillboardPosition) {\n const storedTranslation = TmpVectors.Vector3[0];\n this._worldMatrix.getTranslationToRef(storedTranslation); // Save translation\n // Cancel camera rotation\n TmpVectors.Matrix[1].copyFrom(camera.getViewMatrix());\n if (this._scene.useRightHandedSystem) {\n TmpVectors.Matrix[1].multiplyToRef(convertRHSToLHS, TmpVectors.Matrix[1]);\n }\n TmpVectors.Matrix[1].setTranslationFromFloats(0, 0, 0);\n TmpVectors.Matrix[1].invertToRef(TmpVectors.Matrix[0]);\n if ((this.billboardMode & TransformNode.BILLBOARDMODE_ALL) !== TransformNode.BILLBOARDMODE_ALL) {\n TmpVectors.Matrix[0].decompose(undefined, TmpVectors.Quaternion[0], undefined);\n const eulerAngles = TmpVectors.Vector3[1];\n TmpVectors.Quaternion[0].toEulerAnglesToRef(eulerAngles);\n if ((this.billboardMode & TransformNode.BILLBOARDMODE_X) !== TransformNode.BILLBOARDMODE_X) {\n eulerAngles.x = 0;\n }\n if ((this.billboardMode & TransformNode.BILLBOARDMODE_Y) !== TransformNode.BILLBOARDMODE_Y) {\n eulerAngles.y = 0;\n }\n if ((this.billboardMode & TransformNode.BILLBOARDMODE_Z) !== TransformNode.BILLBOARDMODE_Z) {\n eulerAngles.z = 0;\n }\n Matrix.RotationYawPitchRollToRef(eulerAngles.y, eulerAngles.x, eulerAngles.z, TmpVectors.Matrix[0]);\n }\n this._worldMatrix.setTranslationFromFloats(0, 0, 0);\n this._worldMatrix.multiplyToRef(TmpVectors.Matrix[0], this._worldMatrix);\n // Restore translation\n this._worldMatrix.setTranslation(TmpVectors.Vector3[0]);\n }\n // Billboarding based on camera position\n else if (cache.useBillboardPath && camera && cache.useBillboardPosition) {\n const storedTranslation = TmpVectors.Vector3[0];\n // Save translation\n this._worldMatrix.getTranslationToRef(storedTranslation);\n // Compute camera position in local space\n const cameraPosition = camera.globalPosition;\n this._worldMatrix.invertToRef(TmpVectors.Matrix[1]);\n const camInObjSpace = TmpVectors.Vector3[1];\n Vector3.TransformCoordinatesToRef(cameraPosition, TmpVectors.Matrix[1], camInObjSpace);\n camInObjSpace.normalize();\n // Find the lookAt info in local space\n const yaw = -Math.atan2(camInObjSpace.z, camInObjSpace.x) + Math.PI / 2;\n const len = Math.sqrt(camInObjSpace.x * camInObjSpace.x + camInObjSpace.z * camInObjSpace.z);\n const pitch = -Math.atan2(camInObjSpace.y, len);\n Quaternion.RotationYawPitchRollToRef(yaw, pitch, 0, TmpVectors.Quaternion[0]);\n if ((this.billboardMode & TransformNode.BILLBOARDMODE_ALL) !== TransformNode.BILLBOARDMODE_ALL) {\n const eulerAngles = TmpVectors.Vector3[1];\n TmpVectors.Quaternion[0].toEulerAnglesToRef(eulerAngles);\n if ((this.billboardMode & TransformNode.BILLBOARDMODE_X) !== TransformNode.BILLBOARDMODE_X) {\n eulerAngles.x = 0;\n }\n if ((this.billboardMode & TransformNode.BILLBOARDMODE_Y) !== TransformNode.BILLBOARDMODE_Y) {\n eulerAngles.y = 0;\n }\n if ((this.billboardMode & TransformNode.BILLBOARDMODE_Z) !== TransformNode.BILLBOARDMODE_Z) {\n eulerAngles.z = 0;\n }\n Matrix.RotationYawPitchRollToRef(eulerAngles.y, eulerAngles.x, eulerAngles.z, TmpVectors.Matrix[0]);\n }\n else {\n Matrix.FromQuaternionToRef(TmpVectors.Quaternion[0], TmpVectors.Matrix[0]);\n }\n // Cancel translation\n this._worldMatrix.setTranslationFromFloats(0, 0, 0);\n // Rotate according to lookat (diff from local to lookat)\n this._worldMatrix.multiplyToRef(TmpVectors.Matrix[0], this._worldMatrix);\n // Restore translation\n this._worldMatrix.setTranslation(TmpVectors.Vector3[0]);\n }\n // Normal matrix\n if (!this.ignoreNonUniformScaling) {\n if (this._scaling.isNonUniformWithinEpsilon(0.000001)) {\n this._updateNonUniformScalingState(true);\n }\n else if (parent && parent._nonUniformScaling) {\n this._updateNonUniformScalingState(parent._nonUniformScaling);\n }\n else {\n this._updateNonUniformScalingState(false);\n }\n }\n else {\n this._updateNonUniformScalingState(false);\n }\n this._afterComputeWorldMatrix();\n // Absolute position\n this._absolutePosition.copyFromFloats(this._worldMatrix.m[12], this._worldMatrix.m[13], this._worldMatrix.m[14]);\n this._isAbsoluteSynced = false;\n // Callbacks\n this.onAfterWorldMatrixUpdateObservable.notifyObservers(this);\n if (!this._poseMatrix) {\n this._poseMatrix = Matrix.Invert(this._worldMatrix);\n }\n // Cache the determinant\n this._worldMatrixDeterminantIsDirty = true;\n return this._worldMatrix;\n }\n /**\n * Resets this nodeTransform's local matrix to Matrix.Identity().\n * @param independentOfChildren indicates if all child nodeTransform's world-space transform should be preserved.\n */\n resetLocalMatrix(independentOfChildren = true) {\n this.computeWorldMatrix();\n if (independentOfChildren) {\n const children = this.getChildren();\n for (let i = 0; i < children.length; ++i) {\n const child = children[i];\n if (child) {\n child.computeWorldMatrix();\n const bakedMatrix = TmpVectors.Matrix[0];\n child._localMatrix.multiplyToRef(this._localMatrix, bakedMatrix);\n const tmpRotationQuaternion = TmpVectors.Quaternion[0];\n bakedMatrix.decompose(child.scaling, tmpRotationQuaternion, child.position);\n if (child.rotationQuaternion) {\n child.rotationQuaternion.copyFrom(tmpRotationQuaternion);\n }\n else {\n tmpRotationQuaternion.toEulerAnglesToRef(child.rotation);\n }\n }\n }\n }\n this.scaling.copyFromFloats(1, 1, 1);\n this.position.copyFromFloats(0, 0, 0);\n this.rotation.copyFromFloats(0, 0, 0);\n //only if quaternion is already set\n if (this.rotationQuaternion) {\n this.rotationQuaternion = Quaternion.Identity();\n }\n this._worldMatrix = Matrix.Identity();\n }\n _afterComputeWorldMatrix() { }\n /**\n * If you'd like to be called back after the mesh position, rotation or scaling has been updated.\n * @param func callback function to add\n *\n * @returns the TransformNode.\n */\n registerAfterWorldMatrixUpdate(func) {\n this.onAfterWorldMatrixUpdateObservable.add(func);\n return this;\n }\n /**\n * Removes a registered callback function.\n * @param func callback function to remove\n * @returns the TransformNode.\n */\n unregisterAfterWorldMatrixUpdate(func) {\n this.onAfterWorldMatrixUpdateObservable.removeCallback(func);\n return this;\n }\n /**\n * Gets the position of the current mesh in camera space\n * @param camera defines the camera to use\n * @returns a position\n */\n getPositionInCameraSpace(camera = null) {\n if (!camera) {\n camera = this.getScene().activeCamera;\n }\n return Vector3.TransformCoordinates(this.getAbsolutePosition(), camera.getViewMatrix());\n }\n /**\n * Returns the distance from the mesh to the active camera\n * @param camera defines the camera to use\n * @returns the distance\n */\n getDistanceToCamera(camera = null) {\n if (!camera) {\n camera = this.getScene().activeCamera;\n }\n return this.getAbsolutePosition().subtract(camera.globalPosition).length();\n }\n /**\n * Clone the current transform node\n * @param name Name of the new clone\n * @param newParent New parent for the clone\n * @param doNotCloneChildren Do not clone children hierarchy\n * @returns the new transform node\n */\n clone(name, newParent, doNotCloneChildren) {\n const result = SerializationHelper.Clone(() => new TransformNode(name, this.getScene()), this);\n result.name = name;\n result.id = name;\n if (newParent) {\n result.parent = newParent;\n }\n if (!doNotCloneChildren) {\n // Children\n const directDescendants = this.getDescendants(true);\n for (let index = 0; index < directDescendants.length; index++) {\n const child = directDescendants[index];\n if (child.clone) {\n child.clone(name + \".\" + child.name, result);\n }\n }\n }\n return result;\n }\n /**\n * Serializes the objects information.\n * @param currentSerializationObject defines the object to serialize in\n * @returns the serialized object\n */\n serialize(currentSerializationObject) {\n const serializationObject = SerializationHelper.Serialize(this, currentSerializationObject);\n serializationObject.type = this.getClassName();\n serializationObject.uniqueId = this.uniqueId;\n // Parent\n if (this.parent) {\n this.parent._serializeAsParent(serializationObject);\n }\n serializationObject.localMatrix = this.getPivotMatrix().asArray();\n serializationObject.isEnabled = this.isEnabled();\n return serializationObject;\n }\n // Statics\n /**\n * Returns a new TransformNode object parsed from the source provided.\n * @param parsedTransformNode is the source.\n * @param scene the scene the object belongs to\n * @param rootUrl is a string, it's the root URL to prefix the `delayLoadingFile` property with\n * @returns a new TransformNode object parsed from the source provided.\n */\n static Parse(parsedTransformNode, scene, rootUrl) {\n const transformNode = SerializationHelper.Parse(() => new TransformNode(parsedTransformNode.name, scene), parsedTransformNode, scene, rootUrl);\n if (parsedTransformNode.localMatrix) {\n transformNode.setPreTransformMatrix(Matrix.FromArray(parsedTransformNode.localMatrix));\n }\n else if (parsedTransformNode.pivotMatrix) {\n transformNode.setPivotMatrix(Matrix.FromArray(parsedTransformNode.pivotMatrix));\n }\n transformNode.setEnabled(parsedTransformNode.isEnabled);\n transformNode._waitingParsedUniqueId = parsedTransformNode.uniqueId;\n // Parent\n if (parsedTransformNode.parentId !== undefined) {\n transformNode._waitingParentId = parsedTransformNode.parentId;\n }\n if (parsedTransformNode.parentInstanceIndex !== undefined) {\n transformNode._waitingParentInstanceIndex = parsedTransformNode.parentInstanceIndex;\n }\n return transformNode;\n }\n /**\n * Get all child-transformNodes of this node\n * @param directDescendantsOnly defines if true only direct descendants of 'this' will be considered, if false direct and also indirect (children of children, an so on in a recursive manner) descendants of 'this' will be considered\n * @param predicate defines an optional predicate that will be called on every evaluated child, the predicate must return true for a given child to be part of the result, otherwise it will be ignored\n * @returns an array of TransformNode\n */\n getChildTransformNodes(directDescendantsOnly, predicate) {\n const results = [];\n this._getDescendants(results, directDescendantsOnly, (node) => {\n return (!predicate || predicate(node)) && node instanceof TransformNode;\n });\n return results;\n }\n /**\n * Releases resources associated with this transform node.\n * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default)\n * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default)\n */\n dispose(doNotRecurse, disposeMaterialAndTextures = false) {\n // Animations\n this.getScene().stopAnimation(this);\n // Remove from scene\n this.getScene().removeTransformNode(this);\n if (this._parentContainer) {\n const index = this._parentContainer.transformNodes.indexOf(this);\n if (index > -1) {\n this._parentContainer.transformNodes.splice(index, 1);\n }\n this._parentContainer = null;\n }\n this.onAfterWorldMatrixUpdateObservable.clear();\n if (doNotRecurse) {\n const transformNodes = this.getChildTransformNodes(true);\n for (const transformNode of transformNodes) {\n transformNode.parent = null;\n transformNode.computeWorldMatrix(true);\n }\n }\n super.dispose(doNotRecurse, disposeMaterialAndTextures);\n }\n /**\n * Uniformly scales the mesh to fit inside of a unit cube (1 X 1 X 1 units)\n * @param includeDescendants Use the hierarchy's bounding box instead of the mesh's bounding box. Default is false\n * @param ignoreRotation ignore rotation when computing the scale (ie. object will be axis aligned). Default is false\n * @param predicate predicate that is passed in to getHierarchyBoundingVectors when selecting which object should be included when scaling\n * @returns the current mesh\n */\n normalizeToUnitCube(includeDescendants = true, ignoreRotation = false, predicate) {\n let storedRotation = null;\n let storedRotationQuaternion = null;\n if (ignoreRotation) {\n if (this.rotationQuaternion) {\n storedRotationQuaternion = this.rotationQuaternion.clone();\n this.rotationQuaternion.copyFromFloats(0, 0, 0, 1);\n }\n else if (this.rotation) {\n storedRotation = this.rotation.clone();\n this.rotation.copyFromFloats(0, 0, 0);\n }\n }\n const boundingVectors = this.getHierarchyBoundingVectors(includeDescendants, predicate);\n const sizeVec = boundingVectors.max.subtract(boundingVectors.min);\n const maxDimension = Math.max(sizeVec.x, sizeVec.y, sizeVec.z);\n if (maxDimension === 0) {\n return this;\n }\n const scale = 1 / maxDimension;\n this.scaling.scaleInPlace(scale);\n if (ignoreRotation) {\n if (this.rotationQuaternion && storedRotationQuaternion) {\n this.rotationQuaternion.copyFrom(storedRotationQuaternion);\n }\n else if (this.rotation && storedRotation) {\n this.rotation.copyFrom(storedRotation);\n }\n }\n return this;\n }\n _syncAbsoluteScalingAndRotation() {\n if (!this._isAbsoluteSynced) {\n this._worldMatrix.decompose(this._absoluteScaling, this._absoluteRotationQuaternion);\n this._isAbsoluteSynced = true;\n }\n }\n }\n // Statics\n /**\n * Object will not rotate to face the camera\n */\n TransformNode.BILLBOARDMODE_NONE = 0;\n /**\n * Object will rotate to face the camera but only on the x axis\n */\n TransformNode.BILLBOARDMODE_X = 1;\n /**\n * Object will rotate to face the camera but only on the y axis\n */\n TransformNode.BILLBOARDMODE_Y = 2;\n /**\n * Object will rotate to face the camera but only on the z axis\n */\n TransformNode.BILLBOARDMODE_Z = 4;\n /**\n * Object will rotate to face the camera\n */\n TransformNode.BILLBOARDMODE_ALL = 7;\n /**\n * Object will rotate to face the camera's position instead of orientation\n */\n TransformNode.BILLBOARDMODE_USE_POSITION = 128;\n /**\n * Child transform with Billboard flags should or should not apply parent rotation (default if off)\n */\n TransformNode.BillboardUseParentOrientation = false;\n TransformNode._TmpRotation = Quaternion.Zero();\n TransformNode._TmpScaling = Vector3.Zero();\n TransformNode._TmpTranslation = Vector3.Zero();\n TransformNode._LookAtVectorCache = new Vector3(0, 0, 0);\n TransformNode._RotationAxisCache = new Quaternion();\n __decorate$1([\n serializeAsVector3(\"position\")\n ], TransformNode.prototype, \"_position\", void 0);\n __decorate$1([\n serializeAsVector3(\"rotation\")\n ], TransformNode.prototype, \"_rotation\", void 0);\n __decorate$1([\n serializeAsQuaternion(\"rotationQuaternion\")\n ], TransformNode.prototype, \"_rotationQuaternion\", void 0);\n __decorate$1([\n serializeAsVector3(\"scaling\")\n ], TransformNode.prototype, \"_scaling\", void 0);\n __decorate$1([\n serialize(\"billboardMode\")\n ], TransformNode.prototype, \"_billboardMode\", void 0);\n __decorate$1([\n serialize()\n ], TransformNode.prototype, \"scalingDeterminant\", void 0);\n __decorate$1([\n serialize(\"infiniteDistance\")\n ], TransformNode.prototype, \"_infiniteDistance\", void 0);\n __decorate$1([\n serialize()\n ], TransformNode.prototype, \"ignoreNonUniformScaling\", void 0);\n __decorate$1([\n serialize()\n ], TransformNode.prototype, \"reIntegrateRotationIntoRotationQuaternion\", void 0);\n\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class _MeshCollisionData {\n constructor() {\n this._checkCollisions = false;\n this._collisionMask = -1;\n this._collisionGroup = -1;\n this._surroundingMeshes = null;\n this._collider = null;\n this._oldPositionForCollisions = new Vector3(0, 0, 0);\n this._diffPositionForCollisions = new Vector3(0, 0, 0);\n this._collisionResponse = true;\n }\n }\n\n /** @internal */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class _FacetDataStorage {\n constructor() {\n this.facetNb = 0; // facet number\n this.partitioningSubdivisions = 10; // number of subdivisions per axis in the partitioning space\n this.partitioningBBoxRatio = 1.01; // the partitioning array space is by default 1% bigger than the bounding box\n this.facetDataEnabled = false; // is the facet data feature enabled on this mesh ?\n this.facetParameters = {}; // keep a reference to the object parameters to avoid memory re-allocation\n this.bbSize = Vector3.Zero(); // bbox size approximated for facet data\n this.subDiv = {\n // actual number of subdivisions per axis for ComputeNormals()\n max: 1,\n // eslint-disable-next-line @typescript-eslint/naming-convention\n X: 1,\n // eslint-disable-next-line @typescript-eslint/naming-convention\n Y: 1,\n // eslint-disable-next-line @typescript-eslint/naming-convention\n Z: 1,\n };\n this.facetDepthSort = false; // is the facet depth sort to be computed\n this.facetDepthSortEnabled = false; // is the facet depth sort initialized\n }\n }\n /**\n * @internal\n **/\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class _InternalAbstractMeshDataInfo {\n constructor() {\n this._hasVertexAlpha = false;\n this._useVertexColors = true;\n this._numBoneInfluencers = 4;\n this._applyFog = true;\n this._receiveShadows = false;\n this._facetData = new _FacetDataStorage();\n this._visibility = 1.0;\n this._skeleton = null;\n this._layerMask = 0x0fffffff;\n this._computeBonesUsingShaders = true;\n this._isActive = false;\n this._onlyForInstances = false;\n this._isActiveIntermediate = false;\n this._onlyForInstancesIntermediate = false;\n this._actAsRegularMesh = false;\n this._currentLOD = null;\n this._currentLODIsUpToDate = false;\n this._collisionRetryCount = 3;\n this._morphTargetManager = null;\n this._renderingGroupId = 0;\n this._bakedVertexAnimationManager = null;\n this._material = null;\n this._positions = null;\n this._pointerOverDisableMeshTesting = false;\n // Collisions\n this._meshCollisionData = new _MeshCollisionData();\n this._enableDistantPicking = false;\n /** @internal\n * Bounding info that is unnafected by the addition of thin instances\n */\n this._rawBoundingInfo = null;\n }\n }\n /**\n * Class used to store all common mesh properties\n */\n class AbstractMesh extends TransformNode {\n /**\n * No billboard\n */\n static get BILLBOARDMODE_NONE() {\n return TransformNode.BILLBOARDMODE_NONE;\n }\n /** Billboard on X axis */\n static get BILLBOARDMODE_X() {\n return TransformNode.BILLBOARDMODE_X;\n }\n /** Billboard on Y axis */\n static get BILLBOARDMODE_Y() {\n return TransformNode.BILLBOARDMODE_Y;\n }\n /** Billboard on Z axis */\n static get BILLBOARDMODE_Z() {\n return TransformNode.BILLBOARDMODE_Z;\n }\n /** Billboard on all axes */\n static get BILLBOARDMODE_ALL() {\n return TransformNode.BILLBOARDMODE_ALL;\n }\n /** Billboard on using position instead of orientation */\n static get BILLBOARDMODE_USE_POSITION() {\n return TransformNode.BILLBOARDMODE_USE_POSITION;\n }\n /**\n * Gets the number of facets in the mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData#what-is-a-mesh-facet\n */\n get facetNb() {\n return this._internalAbstractMeshDataInfo._facetData.facetNb;\n }\n /**\n * Gets or set the number (integer) of subdivisions per axis in the partitioning space\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData#tweaking-the-partitioning\n */\n get partitioningSubdivisions() {\n return this._internalAbstractMeshDataInfo._facetData.partitioningSubdivisions;\n }\n set partitioningSubdivisions(nb) {\n this._internalAbstractMeshDataInfo._facetData.partitioningSubdivisions = nb;\n }\n /**\n * The ratio (float) to apply to the bounding box size to set to the partitioning space.\n * Ex : 1.01 (default) the partitioning space is 1% bigger than the bounding box\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData#tweaking-the-partitioning\n */\n get partitioningBBoxRatio() {\n return this._internalAbstractMeshDataInfo._facetData.partitioningBBoxRatio;\n }\n set partitioningBBoxRatio(ratio) {\n this._internalAbstractMeshDataInfo._facetData.partitioningBBoxRatio = ratio;\n }\n /**\n * Gets or sets a boolean indicating that the facets must be depth sorted on next call to `updateFacetData()`.\n * Works only for updatable meshes.\n * Doesn't work with multi-materials\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData#facet-depth-sort\n */\n get mustDepthSortFacets() {\n return this._internalAbstractMeshDataInfo._facetData.facetDepthSort;\n }\n set mustDepthSortFacets(sort) {\n this._internalAbstractMeshDataInfo._facetData.facetDepthSort = sort;\n }\n /**\n * The location (Vector3) where the facet depth sort must be computed from.\n * By default, the active camera position.\n * Used only when facet depth sort is enabled\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData#facet-depth-sort\n */\n get facetDepthSortFrom() {\n return this._internalAbstractMeshDataInfo._facetData.facetDepthSortFrom;\n }\n set facetDepthSortFrom(location) {\n this._internalAbstractMeshDataInfo._facetData.facetDepthSortFrom = location;\n }\n /** number of collision detection tries. Change this value if not all collisions are detected and handled properly */\n get collisionRetryCount() {\n return this._internalAbstractMeshDataInfo._collisionRetryCount;\n }\n set collisionRetryCount(retryCount) {\n this._internalAbstractMeshDataInfo._collisionRetryCount = retryCount;\n }\n /**\n * gets a boolean indicating if facetData is enabled\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData#what-is-a-mesh-facet\n */\n get isFacetDataEnabled() {\n return this._internalAbstractMeshDataInfo._facetData.facetDataEnabled;\n }\n /**\n * Gets or sets the morph target manager\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/morphTargets\n */\n get morphTargetManager() {\n return this._internalAbstractMeshDataInfo._morphTargetManager;\n }\n set morphTargetManager(value) {\n if (this._internalAbstractMeshDataInfo._morphTargetManager === value) {\n return;\n }\n this._internalAbstractMeshDataInfo._morphTargetManager = value;\n this._syncGeometryWithMorphTargetManager();\n }\n /**\n * Gets or sets the baked vertex animation manager\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/baked_texture_animations\n */\n get bakedVertexAnimationManager() {\n return this._internalAbstractMeshDataInfo._bakedVertexAnimationManager;\n }\n set bakedVertexAnimationManager(value) {\n if (this._internalAbstractMeshDataInfo._bakedVertexAnimationManager === value) {\n return;\n }\n this._internalAbstractMeshDataInfo._bakedVertexAnimationManager = value;\n this._markSubMeshesAsAttributesDirty();\n }\n /** @internal */\n _syncGeometryWithMorphTargetManager() { }\n /**\n * @internal\n */\n _updateNonUniformScalingState(value) {\n if (!super._updateNonUniformScalingState(value)) {\n return false;\n }\n this._markSubMeshesAsMiscDirty();\n return true;\n }\n /** @internal */\n get rawBoundingInfo() {\n return this._internalAbstractMeshDataInfo._rawBoundingInfo;\n }\n set rawBoundingInfo(boundingInfo) {\n this._internalAbstractMeshDataInfo._rawBoundingInfo = boundingInfo;\n }\n /** Set a function to call when this mesh collides with another one */\n set onCollide(callback) {\n if (this._internalAbstractMeshDataInfo._meshCollisionData._onCollideObserver) {\n this.onCollideObservable.remove(this._internalAbstractMeshDataInfo._meshCollisionData._onCollideObserver);\n }\n this._internalAbstractMeshDataInfo._meshCollisionData._onCollideObserver = this.onCollideObservable.add(callback);\n }\n /** Set a function to call when the collision's position changes */\n set onCollisionPositionChange(callback) {\n if (this._internalAbstractMeshDataInfo._meshCollisionData._onCollisionPositionChangeObserver) {\n this.onCollisionPositionChangeObservable.remove(this._internalAbstractMeshDataInfo._meshCollisionData._onCollisionPositionChangeObserver);\n }\n this._internalAbstractMeshDataInfo._meshCollisionData._onCollisionPositionChangeObserver = this.onCollisionPositionChangeObservable.add(callback);\n }\n /**\n * Gets or sets mesh visibility between 0 and 1 (default is 1)\n */\n get visibility() {\n return this._internalAbstractMeshDataInfo._visibility;\n }\n /**\n * Gets or sets mesh visibility between 0 and 1 (default is 1)\n */\n set visibility(value) {\n if (this._internalAbstractMeshDataInfo._visibility === value) {\n return;\n }\n const oldValue = this._internalAbstractMeshDataInfo._visibility;\n this._internalAbstractMeshDataInfo._visibility = value;\n if ((oldValue === 1 && value !== 1) || (oldValue !== 1 && value === 1)) {\n this._markSubMeshesAsDirty((defines) => {\n defines.markAsMiscDirty();\n defines.markAsPrePassDirty();\n });\n }\n }\n /**\n * Gets or sets the property which disables the test that is checking that the mesh under the pointer is the same than the previous time we tested for it (default: false).\n * Set this property to true if you want thin instances picking to be reported accurately when moving over the mesh.\n * Note that setting this property to true will incur some performance penalties when dealing with pointer events for this mesh so use it sparingly.\n */\n get pointerOverDisableMeshTesting() {\n return this._internalAbstractMeshDataInfo._pointerOverDisableMeshTesting;\n }\n set pointerOverDisableMeshTesting(disable) {\n this._internalAbstractMeshDataInfo._pointerOverDisableMeshTesting = disable;\n }\n /**\n * Specifies the rendering group id for this mesh (0 by default)\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/advanced/transparent_rendering#rendering-groups\n */\n get renderingGroupId() {\n return this._internalAbstractMeshDataInfo._renderingGroupId;\n }\n set renderingGroupId(value) {\n this._internalAbstractMeshDataInfo._renderingGroupId = value;\n }\n /** Gets or sets current material */\n get material() {\n return this._internalAbstractMeshDataInfo._material;\n }\n set material(value) {\n if (this._internalAbstractMeshDataInfo._material === value) {\n return;\n }\n // remove from material mesh map id needed\n if (this._internalAbstractMeshDataInfo._material && this._internalAbstractMeshDataInfo._material.meshMap) {\n this._internalAbstractMeshDataInfo._material.meshMap[this.uniqueId] = undefined;\n }\n this._internalAbstractMeshDataInfo._material = value;\n if (value && value.meshMap) {\n value.meshMap[this.uniqueId] = this;\n }\n if (this.onMaterialChangedObservable.hasObservers()) {\n this.onMaterialChangedObservable.notifyObservers(this);\n }\n if (!this.subMeshes) {\n return;\n }\n this.resetDrawCache();\n this._unBindEffect();\n }\n /**\n * Gets the material used to render the mesh in a specific render pass\n * @param renderPassId render pass id\n * @returns material used for the render pass. If no specific material is used for this render pass, undefined is returned (meaning mesh.material is used for this pass)\n */\n getMaterialForRenderPass(renderPassId) {\n var _a;\n return (_a = this._internalAbstractMeshDataInfo._materialForRenderPass) === null || _a === void 0 ? void 0 : _a[renderPassId];\n }\n /**\n * Sets the material to be used to render the mesh in a specific render pass\n * @param renderPassId render pass id\n * @param material material to use for this render pass. If undefined is passed, no specific material will be used for this render pass but the regular material will be used instead (mesh.material)\n */\n setMaterialForRenderPass(renderPassId, material) {\n this.resetDrawCache(renderPassId);\n if (!this._internalAbstractMeshDataInfo._materialForRenderPass) {\n this._internalAbstractMeshDataInfo._materialForRenderPass = [];\n }\n this._internalAbstractMeshDataInfo._materialForRenderPass[renderPassId] = material;\n }\n /**\n * Gets or sets a boolean indicating that this mesh can receive realtime shadows\n * @see https://doc.babylonjs.com/features/featuresDeepDive/lights/shadows\n */\n get receiveShadows() {\n return this._internalAbstractMeshDataInfo._receiveShadows;\n }\n set receiveShadows(value) {\n if (this._internalAbstractMeshDataInfo._receiveShadows === value) {\n return;\n }\n this._internalAbstractMeshDataInfo._receiveShadows = value;\n this._markSubMeshesAsLightDirty();\n }\n /** Gets or sets a boolean indicating that this mesh contains vertex color data with alpha values */\n get hasVertexAlpha() {\n return this._internalAbstractMeshDataInfo._hasVertexAlpha;\n }\n set hasVertexAlpha(value) {\n if (this._internalAbstractMeshDataInfo._hasVertexAlpha === value) {\n return;\n }\n this._internalAbstractMeshDataInfo._hasVertexAlpha = value;\n this._markSubMeshesAsAttributesDirty();\n this._markSubMeshesAsMiscDirty();\n }\n /** Gets or sets a boolean indicating that this mesh needs to use vertex color data to render (if this kind of vertex data is available in the geometry) */\n get useVertexColors() {\n return this._internalAbstractMeshDataInfo._useVertexColors;\n }\n set useVertexColors(value) {\n if (this._internalAbstractMeshDataInfo._useVertexColors === value) {\n return;\n }\n this._internalAbstractMeshDataInfo._useVertexColors = value;\n this._markSubMeshesAsAttributesDirty();\n }\n /**\n * Gets or sets a boolean indicating that bone animations must be computed by the GPU (true by default)\n */\n get computeBonesUsingShaders() {\n return this._internalAbstractMeshDataInfo._computeBonesUsingShaders;\n }\n set computeBonesUsingShaders(value) {\n if (this._internalAbstractMeshDataInfo._computeBonesUsingShaders === value) {\n return;\n }\n this._internalAbstractMeshDataInfo._computeBonesUsingShaders = value;\n this._markSubMeshesAsAttributesDirty();\n }\n /** Gets or sets the number of allowed bone influences per vertex (4 by default) */\n get numBoneInfluencers() {\n return this._internalAbstractMeshDataInfo._numBoneInfluencers;\n }\n set numBoneInfluencers(value) {\n if (this._internalAbstractMeshDataInfo._numBoneInfluencers === value) {\n return;\n }\n this._internalAbstractMeshDataInfo._numBoneInfluencers = value;\n this._markSubMeshesAsAttributesDirty();\n }\n /** Gets or sets a boolean indicating that this mesh will allow fog to be rendered on it (true by default) */\n get applyFog() {\n return this._internalAbstractMeshDataInfo._applyFog;\n }\n set applyFog(value) {\n if (this._internalAbstractMeshDataInfo._applyFog === value) {\n return;\n }\n this._internalAbstractMeshDataInfo._applyFog = value;\n this._markSubMeshesAsMiscDirty();\n }\n /** When enabled, decompose picking matrices for better precision with large values for mesh position and scling */\n get enableDistantPicking() {\n return this._internalAbstractMeshDataInfo._enableDistantPicking;\n }\n set enableDistantPicking(value) {\n this._internalAbstractMeshDataInfo._enableDistantPicking = value;\n }\n /**\n * Gets or sets the current layer mask (default is 0x0FFFFFFF)\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/layerMasksAndMultiCam\n */\n get layerMask() {\n return this._internalAbstractMeshDataInfo._layerMask;\n }\n set layerMask(value) {\n if (value === this._internalAbstractMeshDataInfo._layerMask) {\n return;\n }\n this._internalAbstractMeshDataInfo._layerMask = value;\n this._resyncLightSources();\n }\n /**\n * Gets or sets a collision mask used to mask collisions (default is -1).\n * A collision between A and B will happen if A.collisionGroup & b.collisionMask !== 0\n */\n get collisionMask() {\n return this._internalAbstractMeshDataInfo._meshCollisionData._collisionMask;\n }\n set collisionMask(mask) {\n this._internalAbstractMeshDataInfo._meshCollisionData._collisionMask = !isNaN(mask) ? mask : -1;\n }\n /**\n * Gets or sets a collision response flag (default is true).\n * when collisionResponse is false, events are still triggered but colliding entity has no response\n * This helps creating trigger volume when user wants collision feedback events but not position/velocity\n * to respond to the collision.\n */\n get collisionResponse() {\n return this._internalAbstractMeshDataInfo._meshCollisionData._collisionResponse;\n }\n set collisionResponse(response) {\n this._internalAbstractMeshDataInfo._meshCollisionData._collisionResponse = response;\n }\n /**\n * Gets or sets the current collision group mask (-1 by default).\n * A collision between A and B will happen if A.collisionGroup & b.collisionMask !== 0\n */\n get collisionGroup() {\n return this._internalAbstractMeshDataInfo._meshCollisionData._collisionGroup;\n }\n set collisionGroup(mask) {\n this._internalAbstractMeshDataInfo._meshCollisionData._collisionGroup = !isNaN(mask) ? mask : -1;\n }\n /**\n * Gets or sets current surrounding meshes (null by default).\n *\n * By default collision detection is tested against every mesh in the scene.\n * It is possible to set surroundingMeshes to a defined list of meshes and then only these specified\n * meshes will be tested for the collision.\n *\n * Note: if set to an empty array no collision will happen when this mesh is moved.\n */\n get surroundingMeshes() {\n return this._internalAbstractMeshDataInfo._meshCollisionData._surroundingMeshes;\n }\n set surroundingMeshes(meshes) {\n this._internalAbstractMeshDataInfo._meshCollisionData._surroundingMeshes = meshes;\n }\n /** Gets the list of lights affecting that mesh */\n get lightSources() {\n return this._lightSources;\n }\n /** @internal */\n get _positions() {\n return null;\n }\n /**\n * Gets or sets a skeleton to apply skinning transformations\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/bonesSkeletons\n */\n set skeleton(value) {\n const skeleton = this._internalAbstractMeshDataInfo._skeleton;\n if (skeleton && skeleton.needInitialSkinMatrix) {\n skeleton._unregisterMeshWithPoseMatrix(this);\n }\n if (value && value.needInitialSkinMatrix) {\n value._registerMeshWithPoseMatrix(this);\n }\n this._internalAbstractMeshDataInfo._skeleton = value;\n if (!this._internalAbstractMeshDataInfo._skeleton) {\n this._bonesTransformMatrices = null;\n }\n this._markSubMeshesAsAttributesDirty();\n }\n get skeleton() {\n return this._internalAbstractMeshDataInfo._skeleton;\n }\n // Constructor\n /**\n * Creates a new AbstractMesh\n * @param name defines the name of the mesh\n * @param scene defines the hosting scene\n */\n constructor(name, scene = null) {\n super(name, scene, false);\n // Internal data\n /** @internal */\n this._internalAbstractMeshDataInfo = new _InternalAbstractMeshDataInfo();\n /** @internal */\n this._waitingMaterialId = null;\n /**\n * The culling strategy to use to check whether the mesh must be rendered or not.\n * This value can be changed at any time and will be used on the next render mesh selection.\n * The possible values are :\n * - AbstractMesh.CULLINGSTRATEGY_STANDARD\n * - AbstractMesh.CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY\n * - AbstractMesh.CULLINGSTRATEGY_OPTIMISTIC_INCLUSION\n * - AbstractMesh.CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY\n * Please read each static variable documentation to get details about the culling process.\n * */\n this.cullingStrategy = AbstractMesh.CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY;\n // Events\n /**\n * An event triggered when this mesh collides with another one\n */\n this.onCollideObservable = new Observable$1();\n /**\n * An event triggered when the collision's position changes\n */\n this.onCollisionPositionChangeObservable = new Observable$1();\n /**\n * An event triggered when material is changed\n */\n this.onMaterialChangedObservable = new Observable$1();\n // Properties\n /**\n * Gets or sets the orientation for POV movement & rotation\n */\n this.definedFacingForward = true;\n /** @internal */\n this._occlusionQuery = null;\n /** @internal */\n this._renderingGroup = null;\n /** Gets or sets the alpha index used to sort transparent meshes\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/advanced/transparent_rendering#alpha-index\n */\n this.alphaIndex = Number.MAX_VALUE;\n /**\n * Gets or sets a boolean indicating if the mesh is visible (renderable). Default is true\n */\n this.isVisible = true;\n /**\n * Gets or sets a boolean indicating if the mesh can be picked (by scene.pick for instance or through actions). Default is true\n */\n this.isPickable = true;\n /**\n * Gets or sets a boolean indicating if the mesh can be near picked. Default is false\n */\n this.isNearPickable = false;\n /**\n * Gets or sets a boolean indicating if the mesh can be near grabbed. Default is false\n */\n this.isNearGrabbable = false;\n /** Gets or sets a boolean indicating that bounding boxes of subMeshes must be rendered as well (false by default) */\n this.showSubMeshesBoundingBox = false;\n /** Gets or sets a boolean indicating if the mesh must be considered as a ray blocker for lens flares (false by default)\n * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/lenseFlare\n */\n this.isBlocker = false;\n /**\n * Gets or sets a boolean indicating that pointer move events must be supported on this mesh (false by default)\n */\n this.enablePointerMoveEvents = false;\n /** Defines color to use when rendering outline */\n this.outlineColor = Color3.Red();\n /** Define width to use when rendering outline */\n this.outlineWidth = 0.02;\n /** Defines color to use when rendering overlay */\n this.overlayColor = Color3.Red();\n /** Defines alpha to use when rendering overlay */\n this.overlayAlpha = 0.5;\n /** Gets or sets a boolean indicating that internal octree (if available) can be used to boost submeshes selection (true by default) */\n this.useOctreeForRenderingSelection = true;\n /** Gets or sets a boolean indicating that internal octree (if available) can be used to boost submeshes picking (true by default) */\n this.useOctreeForPicking = true;\n /** Gets or sets a boolean indicating that internal octree (if available) can be used to boost submeshes collision (true by default) */\n this.useOctreeForCollisions = true;\n /**\n * True if the mesh must be rendered in any case (this will shortcut the frustum clipping phase)\n */\n this.alwaysSelectAsActiveMesh = false;\n /**\n * Gets or sets a boolean indicating that the bounding info does not need to be kept in sync (for performance reason)\n */\n this.doNotSyncBoundingInfo = false;\n /**\n * Gets or sets the current action manager\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions\n */\n this.actionManager = null;\n /**\n * Gets or sets the ellipsoid used to impersonate this mesh when using collision engine (default is (0.5, 1, 0.5))\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions\n */\n this.ellipsoid = new Vector3(0.5, 1, 0.5);\n /**\n * Gets or sets the ellipsoid offset used to impersonate this mesh when using collision engine (default is (0, 0, 0))\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions\n */\n this.ellipsoidOffset = new Vector3(0, 0, 0);\n // Edges\n /**\n * Defines edge width used when edgesRenderer is enabled\n * @see https://www.babylonjs-playground.com/#10OJSG#13\n */\n this.edgesWidth = 1;\n /**\n * Defines edge color used when edgesRenderer is enabled\n * @see https://www.babylonjs-playground.com/#10OJSG#13\n */\n this.edgesColor = new Color4(1, 0, 0, 1);\n /** @internal */\n this._edgesRenderer = null;\n /** @internal */\n this._masterMesh = null;\n this._boundingInfo = null;\n this._boundingInfoIsDirty = true;\n /** @internal */\n this._renderId = 0;\n /** @internal */\n this._intersectionsInProgress = new Array();\n /** @internal */\n this._unIndexed = false;\n /** @internal */\n this._lightSources = new Array();\n // Loading properties\n /** @internal */\n this._waitingData = {\n lods: null,\n actions: null,\n freezeWorldMatrix: null,\n };\n /** @internal */\n this._bonesTransformMatrices = null;\n /** @internal */\n this._transformMatrixTexture = null;\n /**\n * An event triggered when the mesh is rebuilt.\n */\n this.onRebuildObservable = new Observable$1();\n this._onCollisionPositionChange = (collisionId, newPosition, collidedMesh = null) => {\n newPosition.subtractToRef(this._internalAbstractMeshDataInfo._meshCollisionData._oldPositionForCollisions, this._internalAbstractMeshDataInfo._meshCollisionData._diffPositionForCollisions);\n if (this._internalAbstractMeshDataInfo._meshCollisionData._diffPositionForCollisions.length() > Engine.CollisionsEpsilon) {\n this.position.addInPlace(this._internalAbstractMeshDataInfo._meshCollisionData._diffPositionForCollisions);\n }\n if (collidedMesh) {\n this.onCollideObservable.notifyObservers(collidedMesh);\n }\n this.onCollisionPositionChangeObservable.notifyObservers(this.position);\n };\n scene = this.getScene();\n scene.addMesh(this);\n this._resyncLightSources();\n // Mesh Uniform Buffer.\n this._uniformBuffer = new UniformBuffer(this.getScene().getEngine(), undefined, undefined, name, !this.getScene().getEngine().isWebGPU);\n this._buildUniformLayout();\n switch (scene.performancePriority) {\n case ScenePerformancePriority.Aggressive:\n this.doNotSyncBoundingInfo = true;\n // eslint-disable-next-line no-fallthrough\n case ScenePerformancePriority.Intermediate:\n this.alwaysSelectAsActiveMesh = true;\n this.isPickable = false;\n break;\n }\n }\n _buildUniformLayout() {\n this._uniformBuffer.addUniform(\"world\", 16);\n this._uniformBuffer.addUniform(\"visibility\", 1);\n this._uniformBuffer.create();\n }\n /**\n * Transfer the mesh values to its UBO.\n * @param world The world matrix associated with the mesh\n */\n transferToEffect(world) {\n const ubo = this._uniformBuffer;\n ubo.updateMatrix(\"world\", world);\n ubo.updateFloat(\"visibility\", this._internalAbstractMeshDataInfo._visibility);\n ubo.update();\n }\n /**\n * Gets the mesh uniform buffer.\n * @returns the uniform buffer of the mesh.\n */\n getMeshUniformBuffer() {\n return this._uniformBuffer;\n }\n /**\n * Returns the string \"AbstractMesh\"\n * @returns \"AbstractMesh\"\n */\n getClassName() {\n return \"AbstractMesh\";\n }\n /**\n * Gets a string representation of the current mesh\n * @param fullDetails defines a boolean indicating if full details must be included\n * @returns a string representation of the current mesh\n */\n toString(fullDetails) {\n let ret = \"Name: \" + this.name + \", isInstance: \" + (this.getClassName() !== \"InstancedMesh\" ? \"YES\" : \"NO\");\n ret += \", # of submeshes: \" + (this.subMeshes ? this.subMeshes.length : 0);\n const skeleton = this._internalAbstractMeshDataInfo._skeleton;\n if (skeleton) {\n ret += \", skeleton: \" + skeleton.name;\n }\n if (fullDetails) {\n ret += \", billboard mode: \" + [\"NONE\", \"X\", \"Y\", null, \"Z\", null, null, \"ALL\"][this.billboardMode];\n ret += \", freeze wrld mat: \" + (this._isWorldMatrixFrozen || this._waitingData.freezeWorldMatrix ? \"YES\" : \"NO\");\n }\n return ret;\n }\n /**\n * @internal\n */\n _getEffectiveParent() {\n if (this._masterMesh && this.billboardMode !== TransformNode.BILLBOARDMODE_NONE) {\n return this._masterMesh;\n }\n return super._getEffectiveParent();\n }\n /**\n * @internal\n */\n _getActionManagerForTrigger(trigger, initialCall = true) {\n if (this.actionManager && (initialCall || this.actionManager.isRecursive)) {\n if (trigger) {\n if (this.actionManager.hasSpecificTrigger(trigger)) {\n return this.actionManager;\n }\n }\n else {\n return this.actionManager;\n }\n }\n if (!this.parent) {\n return null;\n }\n return this.parent._getActionManagerForTrigger(trigger, false);\n }\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _rebuild(dispose = false) {\n this.onRebuildObservable.notifyObservers(this);\n if (this._occlusionQuery !== null) {\n this._occlusionQuery = null;\n }\n if (!this.subMeshes) {\n return;\n }\n for (const subMesh of this.subMeshes) {\n subMesh._rebuild();\n }\n }\n /** @internal */\n _resyncLightSources() {\n this._lightSources.length = 0;\n for (const light of this.getScene().lights) {\n if (!light.isEnabled()) {\n continue;\n }\n if (light.canAffectMesh(this)) {\n this._lightSources.push(light);\n }\n }\n this._markSubMeshesAsLightDirty();\n }\n /**\n * @internal\n */\n _resyncLightSource(light) {\n const isIn = light.isEnabled() && light.canAffectMesh(this);\n const index = this._lightSources.indexOf(light);\n let removed = false;\n if (index === -1) {\n if (!isIn) {\n return;\n }\n this._lightSources.push(light);\n }\n else {\n if (isIn) {\n return;\n }\n removed = true;\n this._lightSources.splice(index, 1);\n }\n this._markSubMeshesAsLightDirty(removed);\n }\n /** @internal */\n _unBindEffect() {\n for (const subMesh of this.subMeshes) {\n subMesh.setEffect(null);\n }\n }\n /**\n * @internal\n */\n _removeLightSource(light, dispose) {\n const index = this._lightSources.indexOf(light);\n if (index === -1) {\n return;\n }\n this._lightSources.splice(index, 1);\n this._markSubMeshesAsLightDirty(dispose);\n }\n _markSubMeshesAsDirty(func) {\n if (!this.subMeshes) {\n return;\n }\n for (const subMesh of this.subMeshes) {\n for (let i = 0; i < subMesh._drawWrappers.length; ++i) {\n const drawWrapper = subMesh._drawWrappers[i];\n if (!drawWrapper || !drawWrapper.defines || !drawWrapper.defines.markAllAsDirty) {\n continue;\n }\n func(drawWrapper.defines);\n }\n }\n }\n /**\n * @internal\n */\n _markSubMeshesAsLightDirty(dispose = false) {\n this._markSubMeshesAsDirty((defines) => defines.markAsLightDirty(dispose));\n }\n /** @internal */\n _markSubMeshesAsAttributesDirty() {\n this._markSubMeshesAsDirty((defines) => defines.markAsAttributesDirty());\n }\n /** @internal */\n _markSubMeshesAsMiscDirty() {\n this._markSubMeshesAsDirty((defines) => defines.markAsMiscDirty());\n }\n /**\n * Flag the AbstractMesh as dirty (Forcing it to update everything)\n * @param property if set to \"rotation\" the objects rotationQuaternion will be set to null\n * @returns this AbstractMesh\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n markAsDirty(property) {\n this._currentRenderId = Number.MAX_VALUE;\n this._isDirty = true;\n return this;\n }\n /**\n * Resets the draw wrappers cache for all submeshes of this abstract mesh\n * @param passId If provided, releases only the draw wrapper corresponding to this render pass id\n */\n resetDrawCache(passId) {\n if (!this.subMeshes) {\n return;\n }\n for (const subMesh of this.subMeshes) {\n subMesh.resetDrawCache(passId);\n }\n }\n // Methods\n /**\n * Returns true if the mesh is blocked. Implemented by child classes\n */\n get isBlocked() {\n return false;\n }\n /**\n * Returns the mesh itself by default. Implemented by child classes\n * @param camera defines the camera to use to pick the right LOD level\n * @returns the currentAbstractMesh\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n getLOD(camera) {\n return this;\n }\n /**\n * Returns 0 by default. Implemented by child classes\n * @returns an integer\n */\n getTotalVertices() {\n return 0;\n }\n /**\n * Returns a positive integer : the total number of indices in this mesh geometry.\n * @returns the number of indices or zero if the mesh has no geometry.\n */\n getTotalIndices() {\n return 0;\n }\n /**\n * Returns null by default. Implemented by child classes\n * @returns null\n */\n getIndices() {\n return null;\n }\n /**\n * Returns the array of the requested vertex data kind. Implemented by child classes\n * @param kind defines the vertex data kind to use\n * @returns null\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n getVerticesData(kind) {\n return null;\n }\n /**\n * Sets the vertex data of the mesh geometry for the requested `kind`.\n * If the mesh has no geometry, a new Geometry object is set to the mesh and then passed this vertex data.\n * Note that a new underlying VertexBuffer object is created each call.\n * If the `kind` is the `PositionKind`, the mesh BoundingInfo is renewed, so the bounding box and sphere, and the mesh World Matrix is recomputed.\n * @param kind defines vertex data kind:\n * * VertexBuffer.PositionKind\n * * VertexBuffer.UVKind\n * * VertexBuffer.UV2Kind\n * * VertexBuffer.UV3Kind\n * * VertexBuffer.UV4Kind\n * * VertexBuffer.UV5Kind\n * * VertexBuffer.UV6Kind\n * * VertexBuffer.ColorKind\n * * VertexBuffer.MatricesIndicesKind\n * * VertexBuffer.MatricesIndicesExtraKind\n * * VertexBuffer.MatricesWeightsKind\n * * VertexBuffer.MatricesWeightsExtraKind\n * @param data defines the data source\n * @param updatable defines if the data must be flagged as updatable (or static)\n * @param stride defines the vertex stride (size of an entire vertex). Can be null and in this case will be deduced from vertex data kind\n * @returns the current mesh\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n setVerticesData(kind, data, updatable, stride) {\n return this;\n }\n /**\n * Updates the existing vertex data of the mesh geometry for the requested `kind`.\n * If the mesh has no geometry, it is simply returned as it is.\n * @param kind defines vertex data kind:\n * * VertexBuffer.PositionKind\n * * VertexBuffer.UVKind\n * * VertexBuffer.UV2Kind\n * * VertexBuffer.UV3Kind\n * * VertexBuffer.UV4Kind\n * * VertexBuffer.UV5Kind\n * * VertexBuffer.UV6Kind\n * * VertexBuffer.ColorKind\n * * VertexBuffer.MatricesIndicesKind\n * * VertexBuffer.MatricesIndicesExtraKind\n * * VertexBuffer.MatricesWeightsKind\n * * VertexBuffer.MatricesWeightsExtraKind\n * @param data defines the data source\n * @param updateExtends If `kind` is `PositionKind` and if `updateExtends` is true, the mesh BoundingInfo is renewed, so the bounding box and sphere, and the mesh World Matrix is recomputed\n * @param makeItUnique If true, a new global geometry is created from this data and is set to the mesh\n * @returns the current mesh\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n updateVerticesData(kind, data, updateExtends, makeItUnique) {\n return this;\n }\n /**\n * Sets the mesh indices,\n * If the mesh has no geometry, a new Geometry object is created and set to the mesh.\n * @param indices Expects an array populated with integers or a typed array (Int32Array, Uint32Array, Uint16Array)\n * @param totalVertices Defines the total number of vertices\n * @returns the current mesh\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n setIndices(indices, totalVertices) {\n return this;\n }\n /**\n * Gets a boolean indicating if specific vertex data is present\n * @param kind defines the vertex data kind to use\n * @returns true is data kind is present\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n isVerticesDataPresent(kind) {\n return false;\n }\n /**\n * Returns the mesh BoundingInfo object or creates a new one and returns if it was undefined.\n * Note that it returns a shallow bounding of the mesh (i.e. it does not include children).\n * However, if the mesh contains thin instances, it will be expanded to include them. If you want the \"raw\" bounding data instead, then use `getRawBoundingInfo()`.\n * To get the full bounding of all children, call `getHierarchyBoundingVectors` instead.\n * @returns a BoundingInfo\n */\n getBoundingInfo() {\n if (this._masterMesh) {\n return this._masterMesh.getBoundingInfo();\n }\n if (this._boundingInfoIsDirty) {\n this._boundingInfoIsDirty = false;\n // this._boundingInfo is being created if undefined\n this._updateBoundingInfo();\n }\n // cannot be null.\n return this._boundingInfo;\n }\n /**\n * Returns the bounding info unnafected by instance data.\n * @returns the bounding info of the mesh unaffected by instance data.\n */\n getRawBoundingInfo() {\n var _a;\n return (_a = this.rawBoundingInfo) !== null && _a !== void 0 ? _a : this.getBoundingInfo();\n }\n /**\n * Overwrite the current bounding info\n * @param boundingInfo defines the new bounding info\n * @returns the current mesh\n */\n setBoundingInfo(boundingInfo) {\n this._boundingInfo = boundingInfo;\n return this;\n }\n /**\n * Returns true if there is already a bounding info\n */\n get hasBoundingInfo() {\n return this._boundingInfo !== null;\n }\n /**\n * Creates a new bounding info for the mesh\n * @param minimum min vector of the bounding box/sphere\n * @param maximum max vector of the bounding box/sphere\n * @param worldMatrix defines the new world matrix\n * @returns the new bounding info\n */\n buildBoundingInfo(minimum, maximum, worldMatrix) {\n this._boundingInfo = new BoundingInfo(minimum, maximum, worldMatrix);\n return this._boundingInfo;\n }\n /**\n * Uniformly scales the mesh to fit inside of a unit cube (1 X 1 X 1 units)\n * @param includeDescendants Use the hierarchy's bounding box instead of the mesh's bounding box. Default is false\n * @param ignoreRotation ignore rotation when computing the scale (ie. object will be axis aligned). Default is false\n * @param predicate predicate that is passed in to getHierarchyBoundingVectors when selecting which object should be included when scaling\n * @returns the current mesh\n */\n normalizeToUnitCube(includeDescendants = true, ignoreRotation = false, predicate) {\n return super.normalizeToUnitCube(includeDescendants, ignoreRotation, predicate);\n }\n /** Gets a boolean indicating if this mesh has skinning data and an attached skeleton */\n get useBones() {\n return ((this.skeleton &&\n this.getScene().skeletonsEnabled &&\n this.isVerticesDataPresent(VertexBuffer.MatricesIndicesKind) &&\n this.isVerticesDataPresent(VertexBuffer.MatricesWeightsKind)));\n }\n /** @internal */\n _preActivate() { }\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _preActivateForIntermediateRendering(renderId) { }\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _activate(renderId, intermediateRendering) {\n this._renderId = renderId;\n return true;\n }\n /** @internal */\n _postActivate() {\n // Do nothing\n }\n /** @internal */\n _freeze() {\n // Do nothing\n }\n /** @internal */\n _unFreeze() {\n // Do nothing\n }\n /**\n * Gets the current world matrix\n * @returns a Matrix\n */\n getWorldMatrix() {\n if (this._masterMesh && this.billboardMode === TransformNode.BILLBOARDMODE_NONE) {\n return this._masterMesh.getWorldMatrix();\n }\n return super.getWorldMatrix();\n }\n /** @internal */\n _getWorldMatrixDeterminant() {\n if (this._masterMesh) {\n return this._masterMesh._getWorldMatrixDeterminant();\n }\n return super._getWorldMatrixDeterminant();\n }\n /**\n * Gets a boolean indicating if this mesh is an instance or a regular mesh\n */\n get isAnInstance() {\n return false;\n }\n /**\n * Gets a boolean indicating if this mesh has instances\n */\n get hasInstances() {\n return false;\n }\n /**\n * Gets a boolean indicating if this mesh has thin instances\n */\n get hasThinInstances() {\n return false;\n }\n // ================================== Point of View Movement =================================\n /**\n * Perform relative position change from the point of view of behind the front of the mesh.\n * This is performed taking into account the meshes current rotation, so you do not have to care.\n * Supports definition of mesh facing forward or backward {@link definedFacingForwardSearch | See definedFacingForwardSearch }.\n * @param amountRight defines the distance on the right axis\n * @param amountUp defines the distance on the up axis\n * @param amountForward defines the distance on the forward axis\n * @returns the current mesh\n */\n movePOV(amountRight, amountUp, amountForward) {\n this.position.addInPlace(this.calcMovePOV(amountRight, amountUp, amountForward));\n return this;\n }\n /**\n * Calculate relative position change from the point of view of behind the front of the mesh.\n * This is performed taking into account the meshes current rotation, so you do not have to care.\n * Supports definition of mesh facing forward or backward {@link definedFacingForwardSearch | See definedFacingForwardSearch }.\n * @param amountRight defines the distance on the right axis\n * @param amountUp defines the distance on the up axis\n * @param amountForward defines the distance on the forward axis\n * @returns the new displacement vector\n */\n calcMovePOV(amountRight, amountUp, amountForward) {\n const rotMatrix = new Matrix();\n const rotQuaternion = this.rotationQuaternion ? this.rotationQuaternion : Quaternion.RotationYawPitchRoll(this.rotation.y, this.rotation.x, this.rotation.z);\n rotQuaternion.toRotationMatrix(rotMatrix);\n const translationDelta = Vector3.Zero();\n const defForwardMult = this.definedFacingForward ? -1 : 1;\n Vector3.TransformCoordinatesFromFloatsToRef(amountRight * defForwardMult, amountUp, amountForward * defForwardMult, rotMatrix, translationDelta);\n return translationDelta;\n }\n // ================================== Point of View Rotation =================================\n /**\n * Perform relative rotation change from the point of view of behind the front of the mesh.\n * Supports definition of mesh facing forward or backward {@link definedFacingForwardSearch | See definedFacingForwardSearch }.\n * @param flipBack defines the flip\n * @param twirlClockwise defines the twirl\n * @param tiltRight defines the tilt\n * @returns the current mesh\n */\n rotatePOV(flipBack, twirlClockwise, tiltRight) {\n this.rotation.addInPlace(this.calcRotatePOV(flipBack, twirlClockwise, tiltRight));\n return this;\n }\n /**\n * Calculate relative rotation change from the point of view of behind the front of the mesh.\n * Supports definition of mesh facing forward or backward {@link definedFacingForwardSearch | See definedFacingForwardSearch }.\n * @param flipBack defines the flip\n * @param twirlClockwise defines the twirl\n * @param tiltRight defines the tilt\n * @returns the new rotation vector\n */\n calcRotatePOV(flipBack, twirlClockwise, tiltRight) {\n const defForwardMult = this.definedFacingForward ? 1 : -1;\n return new Vector3(flipBack * defForwardMult, twirlClockwise, tiltRight * defForwardMult);\n }\n /**\n * This method recomputes and sets a new BoundingInfo to the mesh unless it is locked.\n * This means the mesh underlying bounding box and sphere are recomputed.\n * @param applySkeleton defines whether to apply the skeleton before computing the bounding info\n * @param applyMorph defines whether to apply the morph target before computing the bounding info\n * @returns the current mesh\n */\n refreshBoundingInfo(applySkeleton = false, applyMorph = false) {\n if (this._boundingInfo && this._boundingInfo.isLocked) {\n return this;\n }\n this._refreshBoundingInfo(this._getPositionData(applySkeleton, applyMorph), null);\n return this;\n }\n /**\n * @internal\n */\n _refreshBoundingInfo(data, bias) {\n if (data) {\n const extend = extractMinAndMax(data, 0, this.getTotalVertices(), bias);\n if (this._boundingInfo) {\n this._boundingInfo.reConstruct(extend.minimum, extend.maximum);\n }\n else {\n this._boundingInfo = new BoundingInfo(extend.minimum, extend.maximum);\n }\n }\n if (this.subMeshes) {\n for (let index = 0; index < this.subMeshes.length; index++) {\n this.subMeshes[index].refreshBoundingInfo(data);\n }\n }\n this._updateBoundingInfo();\n }\n /**\n * Internal function to get buffer data and possibly apply morphs and normals\n * @param applySkeleton\n * @param applyMorph\n * @param data\n * @param kind the kind of data you want. Can be Normal or Position\n */\n _getData(applySkeleton = false, applyMorph = false, data, kind = VertexBuffer.PositionKind) {\n data = data !== null && data !== void 0 ? data : this.getVerticesData(kind).slice();\n if (data && applyMorph && this.morphTargetManager) {\n let faceIndexCount = 0;\n let positionIndex = 0;\n for (let vertexCount = 0; vertexCount < data.length; vertexCount++) {\n for (let targetCount = 0; targetCount < this.morphTargetManager.numTargets; targetCount++) {\n const targetMorph = this.morphTargetManager.getTarget(targetCount);\n const influence = targetMorph.influence;\n if (influence > 0.0) {\n const morphTargetPositions = targetMorph.getPositions();\n if (morphTargetPositions) {\n data[vertexCount] += (morphTargetPositions[vertexCount] - data[vertexCount]) * influence;\n }\n }\n }\n faceIndexCount++;\n if (kind === VertexBuffer.PositionKind) {\n if (this._positions && faceIndexCount === 3) {\n // We want to merge into positions every 3 indices starting (but not 0)\n faceIndexCount = 0;\n const index = positionIndex * 3;\n this._positions[positionIndex++].copyFromFloats(data[index], data[index + 1], data[index + 2]);\n }\n }\n }\n }\n if (data && applySkeleton && this.skeleton) {\n const matricesIndicesData = this.getVerticesData(VertexBuffer.MatricesIndicesKind);\n const matricesWeightsData = this.getVerticesData(VertexBuffer.MatricesWeightsKind);\n if (matricesWeightsData && matricesIndicesData) {\n const needExtras = this.numBoneInfluencers > 4;\n const matricesIndicesExtraData = needExtras ? this.getVerticesData(VertexBuffer.MatricesIndicesExtraKind) : null;\n const matricesWeightsExtraData = needExtras ? this.getVerticesData(VertexBuffer.MatricesWeightsExtraKind) : null;\n const skeletonMatrices = this.skeleton.getTransformMatrices(this);\n const tempVector = TmpVectors.Vector3[0];\n const finalMatrix = TmpVectors.Matrix[0];\n const tempMatrix = TmpVectors.Matrix[1];\n let matWeightIdx = 0;\n for (let index = 0; index < data.length; index += 3, matWeightIdx += 4) {\n finalMatrix.reset();\n let inf;\n let weight;\n for (inf = 0; inf < 4; inf++) {\n weight = matricesWeightsData[matWeightIdx + inf];\n if (weight > 0) {\n Matrix.FromFloat32ArrayToRefScaled(skeletonMatrices, Math.floor(matricesIndicesData[matWeightIdx + inf] * 16), weight, tempMatrix);\n finalMatrix.addToSelf(tempMatrix);\n }\n }\n if (needExtras) {\n for (inf = 0; inf < 4; inf++) {\n weight = matricesWeightsExtraData[matWeightIdx + inf];\n if (weight > 0) {\n Matrix.FromFloat32ArrayToRefScaled(skeletonMatrices, Math.floor(matricesIndicesExtraData[matWeightIdx + inf] * 16), weight, tempMatrix);\n finalMatrix.addToSelf(tempMatrix);\n }\n }\n }\n if (kind === VertexBuffer.NormalKind) {\n Vector3.TransformNormalFromFloatsToRef(data[index], data[index + 1], data[index + 2], finalMatrix, tempVector);\n }\n else {\n Vector3.TransformCoordinatesFromFloatsToRef(data[index], data[index + 1], data[index + 2], finalMatrix, tempVector);\n }\n tempVector.toArray(data, index);\n if (kind === VertexBuffer.PositionKind && this._positions) {\n this._positions[index / 3].copyFrom(tempVector);\n }\n }\n }\n }\n return data;\n }\n /**\n * Get the normals vertex data and optionally apply skeleton and morphing.\n * @param applySkeleton defines whether to apply the skeleton\n * @param applyMorph defines whether to apply the morph target\n * @returns the normals data\n */\n getNormalsData(applySkeleton = false, applyMorph = false) {\n return this._getData(applySkeleton, applyMorph, null, VertexBuffer.NormalKind);\n }\n /**\n * Get the position vertex data and optionally apply skeleton and morphing.\n * @param applySkeleton defines whether to apply the skeleton\n * @param applyMorph defines whether to apply the morph target\n * @param data defines the position data to apply the skeleton and morph to\n * @returns the position data\n */\n getPositionData(applySkeleton = false, applyMorph = false, data) {\n return this._getData(applySkeleton, applyMorph, data, VertexBuffer.PositionKind);\n }\n /**\n * @internal\n */\n _getPositionData(applySkeleton, applyMorph) {\n var _a;\n let data = this.getVerticesData(VertexBuffer.PositionKind);\n if (this._internalAbstractMeshDataInfo._positions) {\n this._internalAbstractMeshDataInfo._positions = null;\n }\n if (data && ((applySkeleton && this.skeleton) || (applyMorph && this.morphTargetManager))) {\n data = data.slice();\n this._generatePointsArray();\n if (this._positions) {\n const pos = this._positions;\n this._internalAbstractMeshDataInfo._positions = new Array(pos.length);\n for (let i = 0; i < pos.length; i++) {\n this._internalAbstractMeshDataInfo._positions[i] = ((_a = pos[i]) === null || _a === void 0 ? void 0 : _a.clone()) || new Vector3();\n }\n }\n return this.getPositionData(applySkeleton, applyMorph, data);\n }\n return data;\n }\n /** @internal */\n _updateBoundingInfo() {\n if (this._boundingInfo) {\n this._boundingInfo.update(this.worldMatrixFromCache);\n }\n else {\n this._boundingInfo = new BoundingInfo(Vector3.Zero(), Vector3.Zero(), this.worldMatrixFromCache);\n }\n this._updateSubMeshesBoundingInfo(this.worldMatrixFromCache);\n return this;\n }\n /**\n * @internal\n */\n _updateSubMeshesBoundingInfo(matrix) {\n if (!this.subMeshes) {\n return this;\n }\n const count = this.subMeshes.length;\n for (let subIndex = 0; subIndex < count; subIndex++) {\n const subMesh = this.subMeshes[subIndex];\n if (count > 1 || !subMesh.IsGlobal) {\n subMesh.updateBoundingInfo(matrix);\n }\n }\n return this;\n }\n /** @internal */\n _afterComputeWorldMatrix() {\n if (this.doNotSyncBoundingInfo) {\n return;\n }\n // Bounding info\n this._boundingInfoIsDirty = true;\n }\n /**\n * Returns `true` if the mesh is within the frustum defined by the passed array of planes.\n * A mesh is in the frustum if its bounding box intersects the frustum\n * @param frustumPlanes defines the frustum to test\n * @returns true if the mesh is in the frustum planes\n */\n isInFrustum(frustumPlanes) {\n return this.getBoundingInfo().isInFrustum(frustumPlanes, this.cullingStrategy);\n }\n /**\n * Returns `true` if the mesh is completely in the frustum defined be the passed array of planes.\n * A mesh is completely in the frustum if its bounding box it completely inside the frustum.\n * @param frustumPlanes defines the frustum to test\n * @returns true if the mesh is completely in the frustum planes\n */\n isCompletelyInFrustum(frustumPlanes) {\n return this.getBoundingInfo().isCompletelyInFrustum(frustumPlanes);\n }\n /**\n * True if the mesh intersects another mesh or a SolidParticle object\n * @param mesh defines a target mesh or SolidParticle to test\n * @param precise Unless the parameter `precise` is set to `true` the intersection is computed according to Axis Aligned Bounding Boxes (AABB), else according to OBB (Oriented BBoxes)\n * @param includeDescendants Can be set to true to test if the mesh defined in parameters intersects with the current mesh or any child meshes\n * @returns true if there is an intersection\n */\n intersectsMesh(mesh, precise = false, includeDescendants) {\n const boundingInfo = this.getBoundingInfo();\n const otherBoundingInfo = mesh.getBoundingInfo();\n if (boundingInfo.intersects(otherBoundingInfo, precise)) {\n return true;\n }\n if (includeDescendants) {\n for (const child of this.getChildMeshes()) {\n if (child.intersectsMesh(mesh, precise, true)) {\n return true;\n }\n }\n }\n return false;\n }\n /**\n * Returns true if the passed point (Vector3) is inside the mesh bounding box\n * @param point defines the point to test\n * @returns true if there is an intersection\n */\n intersectsPoint(point) {\n return this.getBoundingInfo().intersectsPoint(point);\n }\n // Collisions\n /**\n * Gets or sets a boolean indicating that this mesh can be used in the collision engine\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions\n */\n get checkCollisions() {\n return this._internalAbstractMeshDataInfo._meshCollisionData._checkCollisions;\n }\n set checkCollisions(collisionEnabled) {\n this._internalAbstractMeshDataInfo._meshCollisionData._checkCollisions = collisionEnabled;\n }\n /**\n * Gets Collider object used to compute collisions (not physics)\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions\n */\n get collider() {\n return this._internalAbstractMeshDataInfo._meshCollisionData._collider;\n }\n /**\n * Move the mesh using collision engine\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions\n * @param displacement defines the requested displacement vector\n * @returns the current mesh\n */\n moveWithCollisions(displacement) {\n const globalPosition = this.getAbsolutePosition();\n globalPosition.addToRef(this.ellipsoidOffset, this._internalAbstractMeshDataInfo._meshCollisionData._oldPositionForCollisions);\n const coordinator = this.getScene().collisionCoordinator;\n if (!this._internalAbstractMeshDataInfo._meshCollisionData._collider) {\n this._internalAbstractMeshDataInfo._meshCollisionData._collider = coordinator.createCollider();\n }\n this._internalAbstractMeshDataInfo._meshCollisionData._collider._radius = this.ellipsoid;\n coordinator.getNewPosition(this._internalAbstractMeshDataInfo._meshCollisionData._oldPositionForCollisions, displacement, this._internalAbstractMeshDataInfo._meshCollisionData._collider, this.collisionRetryCount, this, this._onCollisionPositionChange, this.uniqueId);\n return this;\n }\n // Collisions\n /**\n * @internal\n */\n _collideForSubMesh(subMesh, transformMatrix, collider) {\n var _a;\n this._generatePointsArray();\n if (!this._positions) {\n return this;\n }\n // Transformation\n if (!subMesh._lastColliderWorldVertices || !subMesh._lastColliderTransformMatrix.equals(transformMatrix)) {\n subMesh._lastColliderTransformMatrix = transformMatrix.clone();\n subMesh._lastColliderWorldVertices = [];\n subMesh._trianglePlanes = [];\n const start = subMesh.verticesStart;\n const end = subMesh.verticesStart + subMesh.verticesCount;\n for (let i = start; i < end; i++) {\n subMesh._lastColliderWorldVertices.push(Vector3.TransformCoordinates(this._positions[i], transformMatrix));\n }\n }\n // Collide\n collider._collide(subMesh._trianglePlanes, subMesh._lastColliderWorldVertices, this.getIndices(), subMesh.indexStart, subMesh.indexStart + subMesh.indexCount, subMesh.verticesStart, !!subMesh.getMaterial(), this, this._shouldConvertRHS(), ((_a = subMesh.getMaterial()) === null || _a === void 0 ? void 0 : _a.fillMode) === 7);\n return this;\n }\n /**\n * @internal\n */\n _processCollisionsForSubMeshes(collider, transformMatrix) {\n const subMeshes = this._scene.getCollidingSubMeshCandidates(this, collider);\n const len = subMeshes.length;\n for (let index = 0; index < len; index++) {\n const subMesh = subMeshes.data[index];\n // Bounding test\n if (len > 1 && !subMesh._checkCollision(collider)) {\n continue;\n }\n this._collideForSubMesh(subMesh, transformMatrix, collider);\n }\n return this;\n }\n /** @internal */\n _shouldConvertRHS() {\n return false;\n }\n /**\n * @internal\n */\n _checkCollision(collider) {\n // Bounding box test\n if (!this.getBoundingInfo()._checkCollision(collider)) {\n return this;\n }\n // Transformation matrix\n const collisionsScalingMatrix = TmpVectors.Matrix[0];\n const collisionsTransformMatrix = TmpVectors.Matrix[1];\n Matrix.ScalingToRef(1.0 / collider._radius.x, 1.0 / collider._radius.y, 1.0 / collider._radius.z, collisionsScalingMatrix);\n this.worldMatrixFromCache.multiplyToRef(collisionsScalingMatrix, collisionsTransformMatrix);\n this._processCollisionsForSubMeshes(collider, collisionsTransformMatrix);\n return this;\n }\n // Picking\n /** @internal */\n _generatePointsArray() {\n return false;\n }\n /**\n * Checks if the passed Ray intersects with the mesh. A mesh triangle can be picked both from its front and back sides,\n * irrespective of orientation.\n * @param ray defines the ray to use. It should be in the mesh's LOCAL coordinate space.\n * @param fastCheck defines if fast mode (but less precise) must be used (false by default)\n * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected\n * @param onlyBoundingInfo defines a boolean indicating if picking should only happen using bounding info (false by default)\n * @param worldToUse defines the world matrix to use to get the world coordinate of the intersection point\n * @param skipBoundingInfo a boolean indicating if we should skip the bounding info check\n * @returns the picking info\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/interactions/mesh_intersect\n */\n intersects(ray, fastCheck, trianglePredicate, onlyBoundingInfo = false, worldToUse, skipBoundingInfo = false) {\n const pickingInfo = new PickingInfo();\n const className = this.getClassName();\n const intersectionThreshold = className === \"InstancedLinesMesh\" || className === \"LinesMesh\" || className === \"GreasedLineMesh\" ? this.intersectionThreshold : 0;\n const boundingInfo = this.getBoundingInfo();\n if (!this.subMeshes) {\n return pickingInfo;\n }\n if (!skipBoundingInfo &&\n (!ray.intersectsSphere(boundingInfo.boundingSphere, intersectionThreshold) || !ray.intersectsBox(boundingInfo.boundingBox, intersectionThreshold))) {\n return pickingInfo;\n }\n if (onlyBoundingInfo) {\n pickingInfo.hit = skipBoundingInfo ? false : true;\n pickingInfo.pickedMesh = skipBoundingInfo ? null : this;\n pickingInfo.distance = skipBoundingInfo ? 0 : Vector3.Distance(ray.origin, boundingInfo.boundingSphere.center);\n pickingInfo.subMeshId = 0;\n return pickingInfo;\n }\n if (!this._generatePointsArray()) {\n return pickingInfo;\n }\n let intersectInfo = null;\n const subMeshes = this._scene.getIntersectingSubMeshCandidates(this, ray);\n const len = subMeshes.length;\n // Check if all submeshes are using a material that don't allow picking (point/lines rendering)\n // if no submesh can be picked that way, then fallback to BBox picking\n let anySubmeshSupportIntersect = false;\n for (let index = 0; index < len; index++) {\n const subMesh = subMeshes.data[index];\n const material = subMesh.getMaterial();\n if (!material) {\n continue;\n }\n if (material.fillMode == 7 ||\n material.fillMode == 0 ||\n material.fillMode == 1 ||\n material.fillMode == 2 ||\n material.fillMode == 4) {\n anySubmeshSupportIntersect = true;\n break;\n }\n }\n // no sub mesh support intersection, fallback to BBox that has already be done\n if (!anySubmeshSupportIntersect) {\n pickingInfo.hit = true;\n pickingInfo.pickedMesh = this;\n pickingInfo.distance = Vector3.Distance(ray.origin, boundingInfo.boundingSphere.center);\n pickingInfo.subMeshId = -1;\n return pickingInfo;\n }\n // at least 1 submesh supports intersection, keep going\n for (let index = 0; index < len; index++) {\n const subMesh = subMeshes.data[index];\n // Bounding test\n if (len > 1 && !subMesh.canIntersects(ray)) {\n continue;\n }\n const currentIntersectInfo = subMesh.intersects(ray, this._positions, this.getIndices(), fastCheck, trianglePredicate);\n if (currentIntersectInfo) {\n if (fastCheck || !intersectInfo || currentIntersectInfo.distance < intersectInfo.distance) {\n intersectInfo = currentIntersectInfo;\n intersectInfo.subMeshId = index;\n if (fastCheck) {\n break;\n }\n }\n }\n }\n if (intersectInfo) {\n // Get picked point\n const world = worldToUse !== null && worldToUse !== void 0 ? worldToUse : this.getWorldMatrix();\n const worldOrigin = TmpVectors.Vector3[0];\n const direction = TmpVectors.Vector3[1];\n Vector3.TransformCoordinatesToRef(ray.origin, world, worldOrigin);\n ray.direction.scaleToRef(intersectInfo.distance, direction);\n const worldDirection = Vector3.TransformNormal(direction, world);\n const pickedPoint = worldDirection.addInPlace(worldOrigin);\n // Return result\n pickingInfo.hit = true;\n pickingInfo.distance = Vector3.Distance(worldOrigin, pickedPoint);\n pickingInfo.pickedPoint = pickedPoint;\n pickingInfo.pickedMesh = this;\n pickingInfo.bu = intersectInfo.bu || 0;\n pickingInfo.bv = intersectInfo.bv || 0;\n pickingInfo.subMeshFaceId = intersectInfo.faceId;\n pickingInfo.faceId = intersectInfo.faceId + subMeshes.data[intersectInfo.subMeshId].indexStart / (this.getClassName().indexOf(\"LinesMesh\") !== -1 ? 2 : 3);\n pickingInfo.subMeshId = intersectInfo.subMeshId;\n return pickingInfo;\n }\n return pickingInfo;\n }\n /**\n * Clones the current mesh\n * @param name defines the mesh name\n * @param newParent defines the new mesh parent\n * @param doNotCloneChildren defines a boolean indicating that children must not be cloned (false by default)\n * @returns the new mesh\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n clone(name, newParent, doNotCloneChildren) {\n return null;\n }\n /**\n * Disposes all the submeshes of the current meshnp\n * @returns the current mesh\n */\n releaseSubMeshes() {\n if (this.subMeshes) {\n while (this.subMeshes.length) {\n this.subMeshes[0].dispose();\n }\n }\n else {\n this.subMeshes = new Array();\n }\n return this;\n }\n /**\n * Releases resources associated with this abstract mesh.\n * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default)\n * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default)\n */\n dispose(doNotRecurse, disposeMaterialAndTextures = false) {\n let index;\n // mesh map release.\n if (this._scene.useMaterialMeshMap) {\n // remove from material mesh map id needed\n if (this._internalAbstractMeshDataInfo._material && this._internalAbstractMeshDataInfo._material.meshMap) {\n this._internalAbstractMeshDataInfo._material.meshMap[this.uniqueId] = undefined;\n }\n }\n // Smart Array Retainers.\n this.getScene().freeActiveMeshes();\n this.getScene().freeRenderingGroups();\n // Action manager\n if (this.actionManager !== undefined && this.actionManager !== null) {\n this.actionManager.dispose();\n this.actionManager = null;\n }\n // Skeleton\n this._internalAbstractMeshDataInfo._skeleton = null;\n if (this._transformMatrixTexture) {\n this._transformMatrixTexture.dispose();\n this._transformMatrixTexture = null;\n }\n // Intersections in progress\n for (index = 0; index < this._intersectionsInProgress.length; index++) {\n const other = this._intersectionsInProgress[index];\n const pos = other._intersectionsInProgress.indexOf(this);\n other._intersectionsInProgress.splice(pos, 1);\n }\n this._intersectionsInProgress.length = 0;\n // Lights\n const lights = this.getScene().lights;\n lights.forEach((light) => {\n let meshIndex = light.includedOnlyMeshes.indexOf(this);\n if (meshIndex !== -1) {\n light.includedOnlyMeshes.splice(meshIndex, 1);\n }\n meshIndex = light.excludedMeshes.indexOf(this);\n if (meshIndex !== -1) {\n light.excludedMeshes.splice(meshIndex, 1);\n }\n // Shadow generators\n const generators = light.getShadowGenerators();\n if (generators) {\n const iterator = generators.values();\n for (let key = iterator.next(); key.done !== true; key = iterator.next()) {\n const generator = key.value;\n const shadowMap = generator.getShadowMap();\n if (shadowMap && shadowMap.renderList) {\n meshIndex = shadowMap.renderList.indexOf(this);\n if (meshIndex !== -1) {\n shadowMap.renderList.splice(meshIndex, 1);\n }\n }\n }\n }\n });\n // SubMeshes\n if (this.getClassName() !== \"InstancedMesh\" || this.getClassName() !== \"InstancedLinesMesh\") {\n this.releaseSubMeshes();\n }\n // Query\n const engine = this.getScene().getEngine();\n if (this._occlusionQuery !== null) {\n this.isOcclusionQueryInProgress = false;\n engine.deleteQuery(this._occlusionQuery);\n this._occlusionQuery = null;\n }\n // Engine\n engine.wipeCaches();\n // Remove from scene\n this.getScene().removeMesh(this);\n if (this._parentContainer) {\n const index = this._parentContainer.meshes.indexOf(this);\n if (index > -1) {\n this._parentContainer.meshes.splice(index, 1);\n }\n this._parentContainer = null;\n }\n if (disposeMaterialAndTextures) {\n if (this.material) {\n if (this.material.getClassName() === \"MultiMaterial\") {\n this.material.dispose(false, true, true);\n }\n else {\n this.material.dispose(false, true);\n }\n }\n }\n if (!doNotRecurse) {\n // Particles\n for (index = 0; index < this.getScene().particleSystems.length; index++) {\n if (this.getScene().particleSystems[index].emitter === this) {\n this.getScene().particleSystems[index].dispose();\n index--;\n }\n }\n }\n // facet data\n if (this._internalAbstractMeshDataInfo._facetData.facetDataEnabled) {\n this.disableFacetData();\n }\n this._uniformBuffer.dispose();\n this.onAfterWorldMatrixUpdateObservable.clear();\n this.onCollideObservable.clear();\n this.onCollisionPositionChangeObservable.clear();\n this.onRebuildObservable.clear();\n super.dispose(doNotRecurse, disposeMaterialAndTextures);\n }\n /**\n * Adds the passed mesh as a child to the current mesh\n * @param mesh defines the child mesh\n * @param preserveScalingSign if true, keep scaling sign of child. Otherwise, scaling sign might change.\n * @returns the current mesh\n */\n addChild(mesh, preserveScalingSign = false) {\n mesh.setParent(this, preserveScalingSign);\n return this;\n }\n /**\n * Removes the passed mesh from the current mesh children list\n * @param mesh defines the child mesh\n * @param preserveScalingSign if true, keep scaling sign of child. Otherwise, scaling sign might change.\n * @returns the current mesh\n */\n removeChild(mesh, preserveScalingSign = false) {\n mesh.setParent(null, preserveScalingSign);\n return this;\n }\n // Facet data\n /** @internal */\n _initFacetData() {\n const data = this._internalAbstractMeshDataInfo._facetData;\n if (!data.facetNormals) {\n data.facetNormals = new Array();\n }\n if (!data.facetPositions) {\n data.facetPositions = new Array();\n }\n if (!data.facetPartitioning) {\n data.facetPartitioning = new Array();\n }\n data.facetNb = (this.getIndices().length / 3) | 0;\n data.partitioningSubdivisions = data.partitioningSubdivisions ? data.partitioningSubdivisions : 10; // default nb of partitioning subdivisions = 10\n data.partitioningBBoxRatio = data.partitioningBBoxRatio ? data.partitioningBBoxRatio : 1.01; // default ratio 1.01 = the partitioning is 1% bigger than the bounding box\n for (let f = 0; f < data.facetNb; f++) {\n data.facetNormals[f] = Vector3.Zero();\n data.facetPositions[f] = Vector3.Zero();\n }\n data.facetDataEnabled = true;\n return this;\n }\n /**\n * Updates the mesh facetData arrays and the internal partitioning when the mesh is morphed or updated.\n * This method can be called within the render loop.\n * You don't need to call this method by yourself in the render loop when you update/morph a mesh with the methods CreateXXX() as they automatically manage this computation\n * @returns the current mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\n */\n updateFacetData() {\n const data = this._internalAbstractMeshDataInfo._facetData;\n if (!data.facetDataEnabled) {\n this._initFacetData();\n }\n const positions = this.getVerticesData(VertexBuffer.PositionKind);\n const indices = this.getIndices();\n const normals = this.getVerticesData(VertexBuffer.NormalKind);\n const bInfo = this.getBoundingInfo();\n if (data.facetDepthSort && !data.facetDepthSortEnabled) {\n // init arrays, matrix and sort function on first call\n data.facetDepthSortEnabled = true;\n if (indices instanceof Uint16Array) {\n data.depthSortedIndices = new Uint16Array(indices);\n }\n else if (indices instanceof Uint32Array) {\n data.depthSortedIndices = new Uint32Array(indices);\n }\n else {\n let needs32bits = false;\n for (let i = 0; i < indices.length; i++) {\n if (indices[i] > 65535) {\n needs32bits = true;\n break;\n }\n }\n if (needs32bits) {\n data.depthSortedIndices = new Uint32Array(indices);\n }\n else {\n data.depthSortedIndices = new Uint16Array(indices);\n }\n }\n data.facetDepthSortFunction = function (f1, f2) {\n return f2.sqDistance - f1.sqDistance;\n };\n if (!data.facetDepthSortFrom) {\n const camera = this.getScene().activeCamera;\n data.facetDepthSortFrom = camera ? camera.position : Vector3.Zero();\n }\n data.depthSortedFacets = [];\n for (let f = 0; f < data.facetNb; f++) {\n const depthSortedFacet = { ind: f * 3, sqDistance: 0.0 };\n data.depthSortedFacets.push(depthSortedFacet);\n }\n data.invertedMatrix = Matrix.Identity();\n data.facetDepthSortOrigin = Vector3.Zero();\n }\n data.bbSize.x = bInfo.maximum.x - bInfo.minimum.x > Epsilon ? bInfo.maximum.x - bInfo.minimum.x : Epsilon;\n data.bbSize.y = bInfo.maximum.y - bInfo.minimum.y > Epsilon ? bInfo.maximum.y - bInfo.minimum.y : Epsilon;\n data.bbSize.z = bInfo.maximum.z - bInfo.minimum.z > Epsilon ? bInfo.maximum.z - bInfo.minimum.z : Epsilon;\n let bbSizeMax = data.bbSize.x > data.bbSize.y ? data.bbSize.x : data.bbSize.y;\n bbSizeMax = bbSizeMax > data.bbSize.z ? bbSizeMax : data.bbSize.z;\n data.subDiv.max = data.partitioningSubdivisions;\n data.subDiv.X = Math.floor((data.subDiv.max * data.bbSize.x) / bbSizeMax); // adjust the number of subdivisions per axis\n data.subDiv.Y = Math.floor((data.subDiv.max * data.bbSize.y) / bbSizeMax); // according to each bbox size per axis\n data.subDiv.Z = Math.floor((data.subDiv.max * data.bbSize.z) / bbSizeMax);\n data.subDiv.X = data.subDiv.X < 1 ? 1 : data.subDiv.X; // at least one subdivision\n data.subDiv.Y = data.subDiv.Y < 1 ? 1 : data.subDiv.Y;\n data.subDiv.Z = data.subDiv.Z < 1 ? 1 : data.subDiv.Z;\n // set the parameters for ComputeNormals()\n data.facetParameters.facetNormals = this.getFacetLocalNormals();\n data.facetParameters.facetPositions = this.getFacetLocalPositions();\n data.facetParameters.facetPartitioning = this.getFacetLocalPartitioning();\n data.facetParameters.bInfo = bInfo;\n data.facetParameters.bbSize = data.bbSize;\n data.facetParameters.subDiv = data.subDiv;\n data.facetParameters.ratio = this.partitioningBBoxRatio;\n data.facetParameters.depthSort = data.facetDepthSort;\n if (data.facetDepthSort && data.facetDepthSortEnabled) {\n this.computeWorldMatrix(true);\n this._worldMatrix.invertToRef(data.invertedMatrix);\n Vector3.TransformCoordinatesToRef(data.facetDepthSortFrom, data.invertedMatrix, data.facetDepthSortOrigin);\n data.facetParameters.distanceTo = data.facetDepthSortOrigin;\n }\n data.facetParameters.depthSortedFacets = data.depthSortedFacets;\n if (normals) {\n VertexData.ComputeNormals(positions, indices, normals, data.facetParameters);\n }\n if (data.facetDepthSort && data.facetDepthSortEnabled) {\n data.depthSortedFacets.sort(data.facetDepthSortFunction);\n const l = (data.depthSortedIndices.length / 3) | 0;\n for (let f = 0; f < l; f++) {\n const sind = data.depthSortedFacets[f].ind;\n data.depthSortedIndices[f * 3] = indices[sind];\n data.depthSortedIndices[f * 3 + 1] = indices[sind + 1];\n data.depthSortedIndices[f * 3 + 2] = indices[sind + 2];\n }\n this.updateIndices(data.depthSortedIndices, undefined, true);\n }\n return this;\n }\n /**\n * Returns the facetLocalNormals array.\n * The normals are expressed in the mesh local spac\n * @returns an array of Vector3\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\n */\n getFacetLocalNormals() {\n const facetData = this._internalAbstractMeshDataInfo._facetData;\n if (!facetData.facetNormals) {\n this.updateFacetData();\n }\n return facetData.facetNormals;\n }\n /**\n * Returns the facetLocalPositions array.\n * The facet positions are expressed in the mesh local space\n * @returns an array of Vector3\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\n */\n getFacetLocalPositions() {\n const facetData = this._internalAbstractMeshDataInfo._facetData;\n if (!facetData.facetPositions) {\n this.updateFacetData();\n }\n return facetData.facetPositions;\n }\n /**\n * Returns the facetLocalPartitioning array\n * @returns an array of array of numbers\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\n */\n getFacetLocalPartitioning() {\n const facetData = this._internalAbstractMeshDataInfo._facetData;\n if (!facetData.facetPartitioning) {\n this.updateFacetData();\n }\n return facetData.facetPartitioning;\n }\n /**\n * Returns the i-th facet position in the world system.\n * This method allocates a new Vector3 per call\n * @param i defines the facet index\n * @returns a new Vector3\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\n */\n getFacetPosition(i) {\n const pos = Vector3.Zero();\n this.getFacetPositionToRef(i, pos);\n return pos;\n }\n /**\n * Sets the reference Vector3 with the i-th facet position in the world system\n * @param i defines the facet index\n * @param ref defines the target vector\n * @returns the current mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\n */\n getFacetPositionToRef(i, ref) {\n const localPos = this.getFacetLocalPositions()[i];\n const world = this.getWorldMatrix();\n Vector3.TransformCoordinatesToRef(localPos, world, ref);\n return this;\n }\n /**\n * Returns the i-th facet normal in the world system.\n * This method allocates a new Vector3 per call\n * @param i defines the facet index\n * @returns a new Vector3\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\n */\n getFacetNormal(i) {\n const norm = Vector3.Zero();\n this.getFacetNormalToRef(i, norm);\n return norm;\n }\n /**\n * Sets the reference Vector3 with the i-th facet normal in the world system\n * @param i defines the facet index\n * @param ref defines the target vector\n * @returns the current mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\n */\n getFacetNormalToRef(i, ref) {\n const localNorm = this.getFacetLocalNormals()[i];\n Vector3.TransformNormalToRef(localNorm, this.getWorldMatrix(), ref);\n return this;\n }\n /**\n * Returns the facets (in an array) in the same partitioning block than the one the passed coordinates are located (expressed in the mesh local system)\n * @param x defines x coordinate\n * @param y defines y coordinate\n * @param z defines z coordinate\n * @returns the array of facet indexes\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\n */\n getFacetsAtLocalCoordinates(x, y, z) {\n const bInfo = this.getBoundingInfo();\n const data = this._internalAbstractMeshDataInfo._facetData;\n const ox = Math.floor(((x - bInfo.minimum.x * data.partitioningBBoxRatio) * data.subDiv.X * data.partitioningBBoxRatio) / data.bbSize.x);\n const oy = Math.floor(((y - bInfo.minimum.y * data.partitioningBBoxRatio) * data.subDiv.Y * data.partitioningBBoxRatio) / data.bbSize.y);\n const oz = Math.floor(((z - bInfo.minimum.z * data.partitioningBBoxRatio) * data.subDiv.Z * data.partitioningBBoxRatio) / data.bbSize.z);\n if (ox < 0 || ox > data.subDiv.max || oy < 0 || oy > data.subDiv.max || oz < 0 || oz > data.subDiv.max) {\n return null;\n }\n return data.facetPartitioning[ox + data.subDiv.max * oy + data.subDiv.max * data.subDiv.max * oz];\n }\n /**\n * Returns the closest mesh facet index at (x,y,z) World coordinates, null if not found\n * @param x defines x coordinate\n * @param y defines y coordinate\n * @param z defines z coordinate\n * @param projected sets as the (x,y,z) world projection on the facet\n * @param checkFace if true (default false), only the facet \"facing\" to (x,y,z) or only the ones \"turning their backs\", according to the parameter \"facing\" are returned\n * @param facing if facing and checkFace are true, only the facet \"facing\" to (x, y, z) are returned : positive dot (x, y, z) * facet position. If facing si false and checkFace is true, only the facet \"turning their backs\" to (x, y, z) are returned : negative dot (x, y, z) * facet position\n * @returns the face index if found (or null instead)\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\n */\n getClosestFacetAtCoordinates(x, y, z, projected, checkFace = false, facing = true) {\n const world = this.getWorldMatrix();\n const invMat = TmpVectors.Matrix[5];\n world.invertToRef(invMat);\n const invVect = TmpVectors.Vector3[8];\n Vector3.TransformCoordinatesFromFloatsToRef(x, y, z, invMat, invVect); // transform (x,y,z) to coordinates in the mesh local space\n const closest = this.getClosestFacetAtLocalCoordinates(invVect.x, invVect.y, invVect.z, projected, checkFace, facing);\n if (projected) {\n // transform the local computed projected vector to world coordinates\n Vector3.TransformCoordinatesFromFloatsToRef(projected.x, projected.y, projected.z, world, projected);\n }\n return closest;\n }\n /**\n * Returns the closest mesh facet index at (x,y,z) local coordinates, null if not found\n * @param x defines x coordinate\n * @param y defines y coordinate\n * @param z defines z coordinate\n * @param projected sets as the (x,y,z) local projection on the facet\n * @param checkFace if true (default false), only the facet \"facing\" to (x,y,z) or only the ones \"turning their backs\", according to the parameter \"facing\" are returned\n * @param facing if facing and checkFace are true, only the facet \"facing\" to (x, y, z) are returned : positive dot (x, y, z) * facet position. If facing si false and checkFace is true, only the facet \"turning their backs\" to (x, y, z) are returned : negative dot (x, y, z) * facet position\n * @returns the face index if found (or null instead)\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\n */\n getClosestFacetAtLocalCoordinates(x, y, z, projected, checkFace = false, facing = true) {\n let closest = null;\n let tmpx = 0.0;\n let tmpy = 0.0;\n let tmpz = 0.0;\n let d = 0.0; // tmp dot facet normal * facet position\n let t0 = 0.0;\n let projx = 0.0;\n let projy = 0.0;\n let projz = 0.0;\n // Get all the facets in the same partitioning block than (x, y, z)\n const facetPositions = this.getFacetLocalPositions();\n const facetNormals = this.getFacetLocalNormals();\n const facetsInBlock = this.getFacetsAtLocalCoordinates(x, y, z);\n if (!facetsInBlock) {\n return null;\n }\n // Get the closest facet to (x, y, z)\n let shortest = Number.MAX_VALUE; // init distance vars\n let tmpDistance = shortest;\n let fib; // current facet in the block\n let norm; // current facet normal\n let p0; // current facet barycenter position\n // loop on all the facets in the current partitioning block\n for (let idx = 0; idx < facetsInBlock.length; idx++) {\n fib = facetsInBlock[idx];\n norm = facetNormals[fib];\n p0 = facetPositions[fib];\n d = (x - p0.x) * norm.x + (y - p0.y) * norm.y + (z - p0.z) * norm.z;\n if (!checkFace || (checkFace && facing && d >= 0.0) || (checkFace && !facing && d <= 0.0)) {\n // compute (x,y,z) projection on the facet = (projx, projy, projz)\n d = norm.x * p0.x + norm.y * p0.y + norm.z * p0.z;\n t0 = -(norm.x * x + norm.y * y + norm.z * z - d) / (norm.x * norm.x + norm.y * norm.y + norm.z * norm.z);\n projx = x + norm.x * t0;\n projy = y + norm.y * t0;\n projz = z + norm.z * t0;\n tmpx = projx - x;\n tmpy = projy - y;\n tmpz = projz - z;\n tmpDistance = tmpx * tmpx + tmpy * tmpy + tmpz * tmpz; // compute length between (x, y, z) and its projection on the facet\n if (tmpDistance < shortest) {\n // just keep the closest facet to (x, y, z)\n shortest = tmpDistance;\n closest = fib;\n if (projected) {\n projected.x = projx;\n projected.y = projy;\n projected.z = projz;\n }\n }\n }\n }\n return closest;\n }\n /**\n * Returns the object \"parameter\" set with all the expected parameters for facetData computation by ComputeNormals()\n * @returns the parameters\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\n */\n getFacetDataParameters() {\n return this._internalAbstractMeshDataInfo._facetData.facetParameters;\n }\n /**\n * Disables the feature FacetData and frees the related memory\n * @returns the current mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\n */\n disableFacetData() {\n const facetData = this._internalAbstractMeshDataInfo._facetData;\n if (facetData.facetDataEnabled) {\n facetData.facetDataEnabled = false;\n facetData.facetPositions = new Array();\n facetData.facetNormals = new Array();\n facetData.facetPartitioning = new Array();\n facetData.facetParameters = null;\n facetData.depthSortedIndices = new Uint32Array(0);\n }\n return this;\n }\n /**\n * Updates the AbstractMesh indices array\n * @param indices defines the data source\n * @param offset defines the offset in the index buffer where to store the new data (can be null)\n * @param gpuMemoryOnly defines a boolean indicating that only the GPU memory must be updated leaving the CPU version of the indices unchanged (false by default)\n * @returns the current mesh\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n updateIndices(indices, offset, gpuMemoryOnly = false) {\n return this;\n }\n /**\n * Creates new normals data for the mesh\n * @param updatable defines if the normal vertex buffer must be flagged as updatable\n * @returns the current mesh\n */\n createNormals(updatable) {\n const positions = this.getVerticesData(VertexBuffer.PositionKind);\n const indices = this.getIndices();\n let normals;\n if (this.isVerticesDataPresent(VertexBuffer.NormalKind)) {\n normals = this.getVerticesData(VertexBuffer.NormalKind);\n }\n else {\n normals = [];\n }\n VertexData.ComputeNormals(positions, indices, normals, { useRightHandedSystem: this.getScene().useRightHandedSystem });\n this.setVerticesData(VertexBuffer.NormalKind, normals, updatable);\n return this;\n }\n /**\n * Align the mesh with a normal\n * @param normal defines the normal to use\n * @param upDirection can be used to redefined the up vector to use (will use the (0, 1, 0) by default)\n * @returns the current mesh\n */\n alignWithNormal(normal, upDirection) {\n if (!upDirection) {\n upDirection = Axis.Y;\n }\n const axisX = TmpVectors.Vector3[0];\n const axisZ = TmpVectors.Vector3[1];\n Vector3.CrossToRef(upDirection, normal, axisZ);\n Vector3.CrossToRef(normal, axisZ, axisX);\n if (this.rotationQuaternion) {\n Quaternion.RotationQuaternionFromAxisToRef(axisX, normal, axisZ, this.rotationQuaternion);\n }\n else {\n Vector3.RotationFromAxisToRef(axisX, normal, axisZ, this.rotation);\n }\n return this;\n }\n /** @internal */\n _checkOcclusionQuery() {\n // Will be replaced by correct code if Occlusion queries are referenced\n return false;\n }\n /**\n * Disables the mesh edge rendering mode\n * @returns the currentAbstractMesh\n */\n disableEdgesRendering() {\n throw _WarnImport(\"EdgesRenderer\");\n }\n /**\n * Enables the edge rendering mode on the mesh.\n * This mode makes the mesh edges visible\n * @param epsilon defines the maximal distance between two angles to detect a face\n * @param checkVerticesInsteadOfIndices indicates that we should check vertex list directly instead of faces\n * @param options options to the edge renderer\n * @returns the currentAbstractMesh\n * @see https://www.babylonjs-playground.com/#19O9TU#0\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n enableEdgesRendering(epsilon, checkVerticesInsteadOfIndices, options) {\n throw _WarnImport(\"EdgesRenderer\");\n }\n /**\n * This function returns all of the particle systems in the scene that use the mesh as an emitter.\n * @returns an array of particle systems in the scene that use the mesh as an emitter\n */\n getConnectedParticleSystems() {\n return this._scene.particleSystems.filter((particleSystem) => particleSystem.emitter === this);\n }\n }\n /** No occlusion */\n AbstractMesh.OCCLUSION_TYPE_NONE = 0;\n /** Occlusion set to optimistic */\n AbstractMesh.OCCLUSION_TYPE_OPTIMISTIC = 1;\n /** Occlusion set to strict */\n AbstractMesh.OCCLUSION_TYPE_STRICT = 2;\n /** Use an accurate occlusion algorithm */\n AbstractMesh.OCCLUSION_ALGORITHM_TYPE_ACCURATE = 0;\n /** Use a conservative occlusion algorithm */\n AbstractMesh.OCCLUSION_ALGORITHM_TYPE_CONSERVATIVE = 1;\n /** Default culling strategy : this is an exclusion test and it's the more accurate.\n * Test order :\n * Is the bounding sphere outside the frustum ?\n * If not, are the bounding box vertices outside the frustum ?\n * It not, then the cullable object is in the frustum.\n */\n AbstractMesh.CULLINGSTRATEGY_STANDARD = 0;\n /** Culling strategy : Bounding Sphere Only.\n * This is an exclusion test. It's faster than the standard strategy because the bounding box is not tested.\n * It's also less accurate than the standard because some not visible objects can still be selected.\n * Test : is the bounding sphere outside the frustum ?\n * If not, then the cullable object is in the frustum.\n */\n AbstractMesh.CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY = 1;\n /** Culling strategy : Optimistic Inclusion.\n * This in an inclusion test first, then the standard exclusion test.\n * This can be faster when a cullable object is expected to be almost always in the camera frustum.\n * This could also be a little slower than the standard test when the tested object center is not the frustum but one of its bounding box vertex is still inside.\n * Anyway, it's as accurate as the standard strategy.\n * Test :\n * Is the cullable object bounding sphere center in the frustum ?\n * If not, apply the default culling strategy.\n */\n AbstractMesh.CULLINGSTRATEGY_OPTIMISTIC_INCLUSION = 2;\n /** Culling strategy : Optimistic Inclusion then Bounding Sphere Only.\n * This in an inclusion test first, then the bounding sphere only exclusion test.\n * This can be the fastest test when a cullable object is expected to be almost always in the camera frustum.\n * This could also be a little slower than the BoundingSphereOnly strategy when the tested object center is not in the frustum but its bounding sphere still intersects it.\n * It's less accurate than the standard strategy and as accurate as the BoundingSphereOnly strategy.\n * Test :\n * Is the cullable object bounding sphere center in the frustum ?\n * If not, apply the Bounding Sphere Only strategy. No Bounding Box is tested here.\n */\n AbstractMesh.CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY = 3;\n RegisterClass(\"BABYLON.AbstractMesh\", AbstractMesh);\n\n /** @internal */\n function addClipPlaneUniforms(uniforms) {\n if (uniforms.indexOf(\"vClipPlane\") === -1) {\n uniforms.push(\"vClipPlane\");\n }\n if (uniforms.indexOf(\"vClipPlane2\") === -1) {\n uniforms.push(\"vClipPlane2\");\n }\n if (uniforms.indexOf(\"vClipPlane3\") === -1) {\n uniforms.push(\"vClipPlane3\");\n }\n if (uniforms.indexOf(\"vClipPlane4\") === -1) {\n uniforms.push(\"vClipPlane4\");\n }\n if (uniforms.indexOf(\"vClipPlane5\") === -1) {\n uniforms.push(\"vClipPlane5\");\n }\n if (uniforms.indexOf(\"vClipPlane6\") === -1) {\n uniforms.push(\"vClipPlane6\");\n }\n }\n /** @internal */\n function prepareStringDefinesForClipPlanes(primaryHolder, secondaryHolder, defines) {\n var _a, _b, _c, _d, _e, _f;\n const clipPlane = !!((_a = primaryHolder.clipPlane) !== null && _a !== void 0 ? _a : secondaryHolder.clipPlane);\n const clipPlane2 = !!((_b = primaryHolder.clipPlane2) !== null && _b !== void 0 ? _b : secondaryHolder.clipPlane2);\n const clipPlane3 = !!((_c = primaryHolder.clipPlane3) !== null && _c !== void 0 ? _c : secondaryHolder.clipPlane3);\n const clipPlane4 = !!((_d = primaryHolder.clipPlane4) !== null && _d !== void 0 ? _d : secondaryHolder.clipPlane4);\n const clipPlane5 = !!((_e = primaryHolder.clipPlane5) !== null && _e !== void 0 ? _e : secondaryHolder.clipPlane5);\n const clipPlane6 = !!((_f = primaryHolder.clipPlane6) !== null && _f !== void 0 ? _f : secondaryHolder.clipPlane6);\n if (clipPlane)\n defines.push(\"#define CLIPPLANE\");\n if (clipPlane2)\n defines.push(\"#define CLIPPLANE2\");\n if (clipPlane3)\n defines.push(\"#define CLIPPLANE3\");\n if (clipPlane4)\n defines.push(\"#define CLIPPLANE4\");\n if (clipPlane5)\n defines.push(\"#define CLIPPLANE5\");\n if (clipPlane6)\n defines.push(\"#define CLIPPLANE6\");\n }\n /** @internal */\n function prepareDefinesForClipPlanes(primaryHolder, secondaryHolder, defines) {\n var _a, _b, _c, _d, _e, _f;\n let changed = false;\n const clipPlane = !!((_a = primaryHolder.clipPlane) !== null && _a !== void 0 ? _a : secondaryHolder.clipPlane);\n const clipPlane2 = !!((_b = primaryHolder.clipPlane2) !== null && _b !== void 0 ? _b : secondaryHolder.clipPlane2);\n const clipPlane3 = !!((_c = primaryHolder.clipPlane3) !== null && _c !== void 0 ? _c : secondaryHolder.clipPlane3);\n const clipPlane4 = !!((_d = primaryHolder.clipPlane4) !== null && _d !== void 0 ? _d : secondaryHolder.clipPlane4);\n const clipPlane5 = !!((_e = primaryHolder.clipPlane5) !== null && _e !== void 0 ? _e : secondaryHolder.clipPlane5);\n const clipPlane6 = !!((_f = primaryHolder.clipPlane6) !== null && _f !== void 0 ? _f : secondaryHolder.clipPlane6);\n // Do not factorize this code, it breaks browsers optimizations.\n if (defines[\"CLIPPLANE\"] !== clipPlane) {\n defines[\"CLIPPLANE\"] = clipPlane;\n changed = true;\n }\n if (defines[\"CLIPPLANE2\"] !== clipPlane2) {\n defines[\"CLIPPLANE2\"] = clipPlane2;\n changed = true;\n }\n if (defines[\"CLIPPLANE3\"] !== clipPlane3) {\n defines[\"CLIPPLANE3\"] = clipPlane3;\n changed = true;\n }\n if (defines[\"CLIPPLANE4\"] !== clipPlane4) {\n defines[\"CLIPPLANE4\"] = clipPlane4;\n changed = true;\n }\n if (defines[\"CLIPPLANE5\"] !== clipPlane5) {\n defines[\"CLIPPLANE5\"] = clipPlane5;\n changed = true;\n }\n if (defines[\"CLIPPLANE6\"] !== clipPlane6) {\n defines[\"CLIPPLANE6\"] = clipPlane6;\n changed = true;\n }\n return changed;\n }\n /** @internal */\n function bindClipPlane(effect, primaryHolder, secondaryHolder) {\n var _a, _b, _c, _d, _e, _f;\n let clipPlane = (_a = primaryHolder.clipPlane) !== null && _a !== void 0 ? _a : secondaryHolder.clipPlane;\n setClipPlane(effect, \"vClipPlane\", clipPlane);\n clipPlane = (_b = primaryHolder.clipPlane2) !== null && _b !== void 0 ? _b : secondaryHolder.clipPlane2;\n setClipPlane(effect, \"vClipPlane2\", clipPlane);\n clipPlane = (_c = primaryHolder.clipPlane3) !== null && _c !== void 0 ? _c : secondaryHolder.clipPlane3;\n setClipPlane(effect, \"vClipPlane3\", clipPlane);\n clipPlane = (_d = primaryHolder.clipPlane4) !== null && _d !== void 0 ? _d : secondaryHolder.clipPlane4;\n setClipPlane(effect, \"vClipPlane4\", clipPlane);\n clipPlane = (_e = primaryHolder.clipPlane5) !== null && _e !== void 0 ? _e : secondaryHolder.clipPlane5;\n setClipPlane(effect, \"vClipPlane5\", clipPlane);\n clipPlane = (_f = primaryHolder.clipPlane6) !== null && _f !== void 0 ? _f : secondaryHolder.clipPlane6;\n setClipPlane(effect, \"vClipPlane6\", clipPlane);\n }\n function setClipPlane(effect, uniformName, clipPlane) {\n if (clipPlane) {\n effect.setFloat4(uniformName, clipPlane.normal.x, clipPlane.normal.y, clipPlane.normal.z, clipPlane.d);\n }\n }\n\n /**\n * \"Static Class\" containing the most commonly used helper while dealing with material for rendering purpose.\n *\n * It contains the basic tools to help defining defines, binding uniform for the common part of the materials.\n *\n * This works by convention in BabylonJS but is meant to be use only with shader following the in place naming rules and conventions.\n */\n class MaterialHelper {\n /**\n * Binds the scene's uniform buffer to the effect.\n * @param effect defines the effect to bind to the scene uniform buffer\n * @param sceneUbo defines the uniform buffer storing scene data\n */\n static BindSceneUniformBuffer(effect, sceneUbo) {\n sceneUbo.bindToEffect(effect, \"Scene\");\n }\n /**\n * Helps preparing the defines values about the UVs in used in the effect.\n * UVs are shared as much as we can across channels in the shaders.\n * @param texture The texture we are preparing the UVs for\n * @param defines The defines to update\n * @param key The channel key \"diffuse\", \"specular\"... used in the shader\n */\n static PrepareDefinesForMergedUV(texture, defines, key) {\n defines._needUVs = true;\n defines[key] = true;\n if (texture.optimizeUVAllocation && texture.getTextureMatrix().isIdentityAs3x2()) {\n defines[key + \"DIRECTUV\"] = texture.coordinatesIndex + 1;\n defines[\"MAINUV\" + (texture.coordinatesIndex + 1)] = true;\n }\n else {\n defines[key + \"DIRECTUV\"] = 0;\n }\n }\n /**\n * Binds a texture matrix value to its corresponding uniform\n * @param texture The texture to bind the matrix for\n * @param uniformBuffer The uniform buffer receiving the data\n * @param key The channel key \"diffuse\", \"specular\"... used in the shader\n */\n static BindTextureMatrix(texture, uniformBuffer, key) {\n const matrix = texture.getTextureMatrix();\n uniformBuffer.updateMatrix(key + \"Matrix\", matrix);\n }\n /**\n * Gets the current status of the fog (should it be enabled?)\n * @param mesh defines the mesh to evaluate for fog support\n * @param scene defines the hosting scene\n * @returns true if fog must be enabled\n */\n static GetFogState(mesh, scene) {\n return scene.fogEnabled && mesh.applyFog && scene.fogMode !== Scene.FOGMODE_NONE;\n }\n /**\n * Helper used to prepare the list of defines associated with misc. values for shader compilation\n * @param mesh defines the current mesh\n * @param scene defines the current scene\n * @param useLogarithmicDepth defines if logarithmic depth has to be turned on\n * @param pointsCloud defines if point cloud rendering has to be turned on\n * @param fogEnabled defines if fog has to be turned on\n * @param alphaTest defines if alpha testing has to be turned on\n * @param defines defines the current list of defines\n * @param applyDecalAfterDetail Defines if the decal is applied after or before the detail\n */\n static PrepareDefinesForMisc(mesh, scene, useLogarithmicDepth, pointsCloud, fogEnabled, alphaTest, defines, applyDecalAfterDetail = false) {\n if (defines._areMiscDirty) {\n defines[\"LOGARITHMICDEPTH\"] = useLogarithmicDepth;\n defines[\"POINTSIZE\"] = pointsCloud;\n defines[\"FOG\"] = fogEnabled && this.GetFogState(mesh, scene);\n defines[\"NONUNIFORMSCALING\"] = mesh.nonUniformScaling;\n defines[\"ALPHATEST\"] = alphaTest;\n defines[\"DECAL_AFTER_DETAIL\"] = applyDecalAfterDetail;\n }\n }\n /**\n * Helper used to prepare the defines relative to the active camera\n * @param scene defines the current scene\n * @param defines specifies the list of active defines\n * @returns true if the defines have been updated, else false\n */\n static PrepareDefinesForCamera(scene, defines) {\n let changed = false;\n if (scene.activeCamera) {\n const wasOrtho = defines[\"CAMERA_ORTHOGRAPHIC\"] ? 1 : 0;\n const wasPersp = defines[\"CAMERA_PERSPECTIVE\"] ? 1 : 0;\n const isOrtho = scene.activeCamera.mode === Camera.ORTHOGRAPHIC_CAMERA ? 1 : 0;\n const isPersp = scene.activeCamera.mode === Camera.PERSPECTIVE_CAMERA ? 1 : 0;\n if (wasOrtho ^ isOrtho || wasPersp ^ isPersp) {\n defines[\"CAMERA_ORTHOGRAPHIC\"] = isOrtho === 1;\n defines[\"CAMERA_PERSPECTIVE\"] = isPersp === 1;\n changed = true;\n }\n }\n return changed;\n }\n /**\n * Helper used to prepare the list of defines associated with frame values for shader compilation\n * @param scene defines the current scene\n * @param engine defines the current engine\n * @param material defines the material we are compiling the shader for\n * @param defines specifies the list of active defines\n * @param useInstances defines if instances have to be turned on\n * @param useClipPlane defines if clip plane have to be turned on\n * @param useThinInstances defines if thin instances have to be turned on\n */\n static PrepareDefinesForFrameBoundValues(scene, engine, material, defines, useInstances, useClipPlane = null, useThinInstances = false) {\n let changed = MaterialHelper.PrepareDefinesForCamera(scene, defines);\n if (useClipPlane !== false) {\n changed = prepareDefinesForClipPlanes(material, scene, defines);\n }\n if (defines[\"DEPTHPREPASS\"] !== !engine.getColorWrite()) {\n defines[\"DEPTHPREPASS\"] = !defines[\"DEPTHPREPASS\"];\n changed = true;\n }\n if (defines[\"INSTANCES\"] !== useInstances) {\n defines[\"INSTANCES\"] = useInstances;\n changed = true;\n }\n if (defines[\"THIN_INSTANCES\"] !== useThinInstances) {\n defines[\"THIN_INSTANCES\"] = useThinInstances;\n changed = true;\n }\n if (changed) {\n defines.markAsUnprocessed();\n }\n }\n /**\n * Prepares the defines for bones\n * @param mesh The mesh containing the geometry data we will draw\n * @param defines The defines to update\n */\n static PrepareDefinesForBones(mesh, defines) {\n if (mesh.useBones && mesh.computeBonesUsingShaders && mesh.skeleton) {\n defines[\"NUM_BONE_INFLUENCERS\"] = mesh.numBoneInfluencers;\n const materialSupportsBoneTexture = defines[\"BONETEXTURE\"] !== undefined;\n if (mesh.skeleton.isUsingTextureForMatrices && materialSupportsBoneTexture) {\n defines[\"BONETEXTURE\"] = true;\n }\n else {\n defines[\"BonesPerMesh\"] = mesh.skeleton.bones.length + 1;\n defines[\"BONETEXTURE\"] = materialSupportsBoneTexture ? false : undefined;\n const prePassRenderer = mesh.getScene().prePassRenderer;\n if (prePassRenderer && prePassRenderer.enabled) {\n const nonExcluded = prePassRenderer.excludedSkinnedMesh.indexOf(mesh) === -1;\n defines[\"BONES_VELOCITY_ENABLED\"] = nonExcluded;\n }\n }\n }\n else {\n defines[\"NUM_BONE_INFLUENCERS\"] = 0;\n defines[\"BonesPerMesh\"] = 0;\n if (defines[\"BONETEXTURE\"] !== undefined) {\n defines[\"BONETEXTURE\"] = false;\n }\n }\n }\n /**\n * Prepares the defines for morph targets\n * @param mesh The mesh containing the geometry data we will draw\n * @param defines The defines to update\n */\n static PrepareDefinesForMorphTargets(mesh, defines) {\n const manager = mesh.morphTargetManager;\n if (manager) {\n defines[\"MORPHTARGETS_UV\"] = manager.supportsUVs && defines[\"UV1\"];\n defines[\"MORPHTARGETS_TANGENT\"] = manager.supportsTangents && defines[\"TANGENT\"];\n defines[\"MORPHTARGETS_NORMAL\"] = manager.supportsNormals && defines[\"NORMAL\"];\n defines[\"MORPHTARGETS\"] = manager.numInfluencers > 0;\n defines[\"NUM_MORPH_INFLUENCERS\"] = manager.numInfluencers;\n defines[\"MORPHTARGETS_TEXTURE\"] = manager.isUsingTextureForTargets;\n }\n else {\n defines[\"MORPHTARGETS_UV\"] = false;\n defines[\"MORPHTARGETS_TANGENT\"] = false;\n defines[\"MORPHTARGETS_NORMAL\"] = false;\n defines[\"MORPHTARGETS\"] = false;\n defines[\"NUM_MORPH_INFLUENCERS\"] = 0;\n }\n }\n /**\n * Prepares the defines for baked vertex animation\n * @param mesh The mesh containing the geometry data we will draw\n * @param defines The defines to update\n */\n static PrepareDefinesForBakedVertexAnimation(mesh, defines) {\n const manager = mesh.bakedVertexAnimationManager;\n defines[\"BAKED_VERTEX_ANIMATION_TEXTURE\"] = manager && manager.isEnabled ? true : false;\n }\n /**\n * Prepares the defines used in the shader depending on the attributes data available in the mesh\n * @param mesh The mesh containing the geometry data we will draw\n * @param defines The defines to update\n * @param useVertexColor Precise whether vertex colors should be used or not (override mesh info)\n * @param useBones Precise whether bones should be used or not (override mesh info)\n * @param useMorphTargets Precise whether morph targets should be used or not (override mesh info)\n * @param useVertexAlpha Precise whether vertex alpha should be used or not (override mesh info)\n * @param useBakedVertexAnimation Precise whether baked vertex animation should be used or not (override mesh info)\n * @returns false if defines are considered not dirty and have not been checked\n */\n static PrepareDefinesForAttributes(mesh, defines, useVertexColor, useBones, useMorphTargets = false, useVertexAlpha = true, useBakedVertexAnimation = true) {\n if (!defines._areAttributesDirty && defines._needNormals === defines._normals && defines._needUVs === defines._uvs) {\n return false;\n }\n defines._normals = defines._needNormals;\n defines._uvs = defines._needUVs;\n defines[\"NORMAL\"] = defines._needNormals && mesh.isVerticesDataPresent(VertexBuffer.NormalKind);\n if (defines._needNormals && mesh.isVerticesDataPresent(VertexBuffer.TangentKind)) {\n defines[\"TANGENT\"] = true;\n }\n for (let i = 1; i <= 6; ++i) {\n defines[\"UV\" + i] = defines._needUVs ? mesh.isVerticesDataPresent(`uv${i === 1 ? \"\" : i}`) : false;\n }\n if (useVertexColor) {\n const hasVertexColors = mesh.useVertexColors && mesh.isVerticesDataPresent(VertexBuffer.ColorKind);\n defines[\"VERTEXCOLOR\"] = hasVertexColors;\n defines[\"VERTEXALPHA\"] = mesh.hasVertexAlpha && hasVertexColors && useVertexAlpha;\n }\n if (mesh.isVerticesDataPresent(VertexBuffer.ColorInstanceKind) && (mesh.hasInstances || mesh.hasThinInstances)) {\n defines[\"INSTANCESCOLOR\"] = true;\n }\n if (useBones) {\n this.PrepareDefinesForBones(mesh, defines);\n }\n if (useMorphTargets) {\n this.PrepareDefinesForMorphTargets(mesh, defines);\n }\n if (useBakedVertexAnimation) {\n this.PrepareDefinesForBakedVertexAnimation(mesh, defines);\n }\n return true;\n }\n /**\n * Prepares the defines related to multiview\n * @param scene The scene we are intending to draw\n * @param defines The defines to update\n */\n static PrepareDefinesForMultiview(scene, defines) {\n if (scene.activeCamera) {\n const previousMultiview = defines.MULTIVIEW;\n defines.MULTIVIEW = scene.activeCamera.outputRenderTarget !== null && scene.activeCamera.outputRenderTarget.getViewCount() > 1;\n if (defines.MULTIVIEW != previousMultiview) {\n defines.markAsUnprocessed();\n }\n }\n }\n /**\n * Prepares the defines related to order independant transparency\n * @param scene The scene we are intending to draw\n * @param defines The defines to update\n * @param needAlphaBlending Determines if the material needs alpha blending\n */\n static PrepareDefinesForOIT(scene, defines, needAlphaBlending) {\n const previousDefine = defines.ORDER_INDEPENDENT_TRANSPARENCY;\n const previousDefine16Bits = defines.ORDER_INDEPENDENT_TRANSPARENCY_16BITS;\n defines.ORDER_INDEPENDENT_TRANSPARENCY = scene.useOrderIndependentTransparency && needAlphaBlending;\n defines.ORDER_INDEPENDENT_TRANSPARENCY_16BITS = !scene.getEngine().getCaps().textureFloatLinearFiltering;\n if (previousDefine !== defines.ORDER_INDEPENDENT_TRANSPARENCY || previousDefine16Bits !== defines.ORDER_INDEPENDENT_TRANSPARENCY_16BITS) {\n defines.markAsUnprocessed();\n }\n }\n /**\n * Prepares the defines related to the prepass\n * @param scene The scene we are intending to draw\n * @param defines The defines to update\n * @param canRenderToMRT Indicates if this material renders to several textures in the prepass\n */\n static PrepareDefinesForPrePass(scene, defines, canRenderToMRT) {\n const previousPrePass = defines.PREPASS;\n if (!defines._arePrePassDirty) {\n return;\n }\n const texturesList = [\n {\n type: 1,\n define: \"PREPASS_POSITION\",\n index: \"PREPASS_POSITION_INDEX\",\n },\n {\n type: 2,\n define: \"PREPASS_VELOCITY\",\n index: \"PREPASS_VELOCITY_INDEX\",\n },\n {\n type: 3,\n define: \"PREPASS_REFLECTIVITY\",\n index: \"PREPASS_REFLECTIVITY_INDEX\",\n },\n {\n type: 0,\n define: \"PREPASS_IRRADIANCE\",\n index: \"PREPASS_IRRADIANCE_INDEX\",\n },\n {\n type: 7,\n define: \"PREPASS_ALBEDO_SQRT\",\n index: \"PREPASS_ALBEDO_SQRT_INDEX\",\n },\n {\n type: 5,\n define: \"PREPASS_DEPTH\",\n index: \"PREPASS_DEPTH_INDEX\",\n },\n {\n type: 6,\n define: \"PREPASS_NORMAL\",\n index: \"PREPASS_NORMAL_INDEX\",\n },\n ];\n if (scene.prePassRenderer && scene.prePassRenderer.enabled && canRenderToMRT) {\n defines.PREPASS = true;\n defines.SCENE_MRT_COUNT = scene.prePassRenderer.mrtCount;\n for (let i = 0; i < texturesList.length; i++) {\n const index = scene.prePassRenderer.getIndex(texturesList[i].type);\n if (index !== -1) {\n defines[texturesList[i].define] = true;\n defines[texturesList[i].index] = index;\n }\n else {\n defines[texturesList[i].define] = false;\n }\n }\n }\n else {\n defines.PREPASS = false;\n for (let i = 0; i < texturesList.length; i++) {\n defines[texturesList[i].define] = false;\n }\n }\n if (defines.PREPASS != previousPrePass) {\n defines.markAsUnprocessed();\n defines.markAsImageProcessingDirty();\n }\n }\n /**\n * Prepares the defines related to the light information passed in parameter\n * @param scene The scene we are intending to draw\n * @param mesh The mesh the effect is compiling for\n * @param light The light the effect is compiling for\n * @param lightIndex The index of the light\n * @param defines The defines to update\n * @param specularSupported Specifies whether specular is supported or not (override lights data)\n * @param state Defines the current state regarding what is needed (normals, etc...)\n * @param state.needNormals\n * @param state.needRebuild\n * @param state.shadowEnabled\n * @param state.specularEnabled\n * @param state.lightmapMode\n */\n static PrepareDefinesForLight(scene, mesh, light, lightIndex, defines, specularSupported, state) {\n var _a;\n state.needNormals = true;\n if (defines[\"LIGHT\" + lightIndex] === undefined) {\n state.needRebuild = true;\n }\n defines[\"LIGHT\" + lightIndex] = true;\n defines[\"SPOTLIGHT\" + lightIndex] = false;\n defines[\"HEMILIGHT\" + lightIndex] = false;\n defines[\"POINTLIGHT\" + lightIndex] = false;\n defines[\"DIRLIGHT\" + lightIndex] = false;\n light.prepareLightSpecificDefines(defines, lightIndex);\n // FallOff.\n defines[\"LIGHT_FALLOFF_PHYSICAL\" + lightIndex] = false;\n defines[\"LIGHT_FALLOFF_GLTF\" + lightIndex] = false;\n defines[\"LIGHT_FALLOFF_STANDARD\" + lightIndex] = false;\n switch (light.falloffType) {\n case LightConstants.FALLOFF_GLTF:\n defines[\"LIGHT_FALLOFF_GLTF\" + lightIndex] = true;\n break;\n case LightConstants.FALLOFF_PHYSICAL:\n defines[\"LIGHT_FALLOFF_PHYSICAL\" + lightIndex] = true;\n break;\n case LightConstants.FALLOFF_STANDARD:\n defines[\"LIGHT_FALLOFF_STANDARD\" + lightIndex] = true;\n break;\n }\n // Specular\n if (specularSupported && !light.specular.equalsFloats(0, 0, 0)) {\n state.specularEnabled = true;\n }\n // Shadows\n defines[\"SHADOW\" + lightIndex] = false;\n defines[\"SHADOWCSM\" + lightIndex] = false;\n defines[\"SHADOWCSMDEBUG\" + lightIndex] = false;\n defines[\"SHADOWCSMNUM_CASCADES\" + lightIndex] = false;\n defines[\"SHADOWCSMUSESHADOWMAXZ\" + lightIndex] = false;\n defines[\"SHADOWCSMNOBLEND\" + lightIndex] = false;\n defines[\"SHADOWCSM_RIGHTHANDED\" + lightIndex] = false;\n defines[\"SHADOWPCF\" + lightIndex] = false;\n defines[\"SHADOWPCSS\" + lightIndex] = false;\n defines[\"SHADOWPOISSON\" + lightIndex] = false;\n defines[\"SHADOWESM\" + lightIndex] = false;\n defines[\"SHADOWCLOSEESM\" + lightIndex] = false;\n defines[\"SHADOWCUBE\" + lightIndex] = false;\n defines[\"SHADOWLOWQUALITY\" + lightIndex] = false;\n defines[\"SHADOWMEDIUMQUALITY\" + lightIndex] = false;\n if (mesh && mesh.receiveShadows && scene.shadowsEnabled && light.shadowEnabled) {\n const shadowGenerator = (_a = light.getShadowGenerator(scene.activeCamera)) !== null && _a !== void 0 ? _a : light.getShadowGenerator();\n if (shadowGenerator) {\n const shadowMap = shadowGenerator.getShadowMap();\n if (shadowMap) {\n if (shadowMap.renderList && shadowMap.renderList.length > 0) {\n state.shadowEnabled = true;\n shadowGenerator.prepareDefines(defines, lightIndex);\n }\n }\n }\n }\n if (light.lightmapMode != LightConstants.LIGHTMAP_DEFAULT) {\n state.lightmapMode = true;\n defines[\"LIGHTMAPEXCLUDED\" + lightIndex] = true;\n defines[\"LIGHTMAPNOSPECULAR\" + lightIndex] = light.lightmapMode == LightConstants.LIGHTMAP_SHADOWSONLY;\n }\n else {\n defines[\"LIGHTMAPEXCLUDED\" + lightIndex] = false;\n defines[\"LIGHTMAPNOSPECULAR\" + lightIndex] = false;\n }\n }\n /**\n * Prepares the defines related to the light information passed in parameter\n * @param scene The scene we are intending to draw\n * @param mesh The mesh the effect is compiling for\n * @param defines The defines to update\n * @param specularSupported Specifies whether specular is supported or not (override lights data)\n * @param maxSimultaneousLights Specifies how manuy lights can be added to the effect at max\n * @param disableLighting Specifies whether the lighting is disabled (override scene and light)\n * @returns true if normals will be required for the rest of the effect\n */\n static PrepareDefinesForLights(scene, mesh, defines, specularSupported, maxSimultaneousLights = 4, disableLighting = false) {\n if (!defines._areLightsDirty) {\n return defines._needNormals;\n }\n let lightIndex = 0;\n const state = {\n needNormals: defines._needNormals,\n needRebuild: false,\n lightmapMode: false,\n shadowEnabled: false,\n specularEnabled: false,\n };\n if (scene.lightsEnabled && !disableLighting) {\n for (const light of mesh.lightSources) {\n this.PrepareDefinesForLight(scene, mesh, light, lightIndex, defines, specularSupported, state);\n lightIndex++;\n if (lightIndex === maxSimultaneousLights) {\n break;\n }\n }\n }\n defines[\"SPECULARTERM\"] = state.specularEnabled;\n defines[\"SHADOWS\"] = state.shadowEnabled;\n // Resetting all other lights if any\n for (let index = lightIndex; index < maxSimultaneousLights; index++) {\n if (defines[\"LIGHT\" + index] !== undefined) {\n defines[\"LIGHT\" + index] = false;\n defines[\"HEMILIGHT\" + index] = false;\n defines[\"POINTLIGHT\" + index] = false;\n defines[\"DIRLIGHT\" + index] = false;\n defines[\"SPOTLIGHT\" + index] = false;\n defines[\"SHADOW\" + index] = false;\n defines[\"SHADOWCSM\" + index] = false;\n defines[\"SHADOWCSMDEBUG\" + index] = false;\n defines[\"SHADOWCSMNUM_CASCADES\" + index] = false;\n defines[\"SHADOWCSMUSESHADOWMAXZ\" + index] = false;\n defines[\"SHADOWCSMNOBLEND\" + index] = false;\n defines[\"SHADOWCSM_RIGHTHANDED\" + index] = false;\n defines[\"SHADOWPCF\" + index] = false;\n defines[\"SHADOWPCSS\" + index] = false;\n defines[\"SHADOWPOISSON\" + index] = false;\n defines[\"SHADOWESM\" + index] = false;\n defines[\"SHADOWCLOSEESM\" + index] = false;\n defines[\"SHADOWCUBE\" + index] = false;\n defines[\"SHADOWLOWQUALITY\" + index] = false;\n defines[\"SHADOWMEDIUMQUALITY\" + index] = false;\n }\n }\n const caps = scene.getEngine().getCaps();\n if (defines[\"SHADOWFLOAT\"] === undefined) {\n state.needRebuild = true;\n }\n defines[\"SHADOWFLOAT\"] =\n state.shadowEnabled && ((caps.textureFloatRender && caps.textureFloatLinearFiltering) || (caps.textureHalfFloatRender && caps.textureHalfFloatLinearFiltering));\n defines[\"LIGHTMAPEXCLUDED\"] = state.lightmapMode;\n if (state.needRebuild) {\n defines.rebuild();\n }\n return state.needNormals;\n }\n /**\n * Prepares the uniforms and samplers list to be used in the effect (for a specific light)\n * @param lightIndex defines the light index\n * @param uniformsList The uniform list\n * @param samplersList The sampler list\n * @param projectedLightTexture defines if projected texture must be used\n * @param uniformBuffersList defines an optional list of uniform buffers\n * @param updateOnlyBuffersList True to only update the uniformBuffersList array\n */\n static PrepareUniformsAndSamplersForLight(lightIndex, uniformsList, samplersList, projectedLightTexture, uniformBuffersList = null, updateOnlyBuffersList = false) {\n if (uniformBuffersList) {\n uniformBuffersList.push(\"Light\" + lightIndex);\n }\n if (updateOnlyBuffersList) {\n return;\n }\n uniformsList.push(\"vLightData\" + lightIndex, \"vLightDiffuse\" + lightIndex, \"vLightSpecular\" + lightIndex, \"vLightDirection\" + lightIndex, \"vLightFalloff\" + lightIndex, \"vLightGround\" + lightIndex, \"lightMatrix\" + lightIndex, \"shadowsInfo\" + lightIndex, \"depthValues\" + lightIndex);\n samplersList.push(\"shadowSampler\" + lightIndex);\n samplersList.push(\"depthSampler\" + lightIndex);\n uniformsList.push(\"viewFrustumZ\" + lightIndex, \"cascadeBlendFactor\" + lightIndex, \"lightSizeUVCorrection\" + lightIndex, \"depthCorrection\" + lightIndex, \"penumbraDarkness\" + lightIndex, \"frustumLengths\" + lightIndex);\n if (projectedLightTexture) {\n samplersList.push(\"projectionLightSampler\" + lightIndex);\n uniformsList.push(\"textureProjectionMatrix\" + lightIndex);\n }\n }\n /**\n * Prepares the uniforms and samplers list to be used in the effect\n * @param uniformsListOrOptions The uniform names to prepare or an EffectCreationOptions containing the list and extra information\n * @param samplersList The sampler list\n * @param defines The defines helping in the list generation\n * @param maxSimultaneousLights The maximum number of simultaneous light allowed in the effect\n */\n static PrepareUniformsAndSamplersList(uniformsListOrOptions, samplersList, defines, maxSimultaneousLights = 4) {\n let uniformsList;\n let uniformBuffersList = null;\n if (uniformsListOrOptions.uniformsNames) {\n const options = uniformsListOrOptions;\n uniformsList = options.uniformsNames;\n uniformBuffersList = options.uniformBuffersNames;\n samplersList = options.samplers;\n defines = options.defines;\n maxSimultaneousLights = options.maxSimultaneousLights || 0;\n }\n else {\n uniformsList = uniformsListOrOptions;\n if (!samplersList) {\n samplersList = [];\n }\n }\n for (let lightIndex = 0; lightIndex < maxSimultaneousLights; lightIndex++) {\n if (!defines[\"LIGHT\" + lightIndex]) {\n break;\n }\n this.PrepareUniformsAndSamplersForLight(lightIndex, uniformsList, samplersList, defines[\"PROJECTEDLIGHTTEXTURE\" + lightIndex], uniformBuffersList);\n }\n if (defines[\"NUM_MORPH_INFLUENCERS\"]) {\n uniformsList.push(\"morphTargetInfluences\");\n }\n if (defines[\"BAKED_VERTEX_ANIMATION_TEXTURE\"]) {\n uniformsList.push(\"bakedVertexAnimationSettings\");\n uniformsList.push(\"bakedVertexAnimationTextureSizeInverted\");\n uniformsList.push(\"bakedVertexAnimationTime\");\n samplersList.push(\"bakedVertexAnimationTexture\");\n }\n }\n /**\n * This helps decreasing rank by rank the shadow quality (0 being the highest rank and quality)\n * @param defines The defines to update while falling back\n * @param fallbacks The authorized effect fallbacks\n * @param maxSimultaneousLights The maximum number of lights allowed\n * @param rank the current rank of the Effect\n * @returns The newly affected rank\n */\n static HandleFallbacksForShadows(defines, fallbacks, maxSimultaneousLights = 4, rank = 0) {\n let lightFallbackRank = 0;\n for (let lightIndex = 0; lightIndex < maxSimultaneousLights; lightIndex++) {\n if (!defines[\"LIGHT\" + lightIndex]) {\n break;\n }\n if (lightIndex > 0) {\n lightFallbackRank = rank + lightIndex;\n fallbacks.addFallback(lightFallbackRank, \"LIGHT\" + lightIndex);\n }\n if (!defines[\"SHADOWS\"]) {\n if (defines[\"SHADOW\" + lightIndex]) {\n fallbacks.addFallback(rank, \"SHADOW\" + lightIndex);\n }\n if (defines[\"SHADOWPCF\" + lightIndex]) {\n fallbacks.addFallback(rank, \"SHADOWPCF\" + lightIndex);\n }\n if (defines[\"SHADOWPCSS\" + lightIndex]) {\n fallbacks.addFallback(rank, \"SHADOWPCSS\" + lightIndex);\n }\n if (defines[\"SHADOWPOISSON\" + lightIndex]) {\n fallbacks.addFallback(rank, \"SHADOWPOISSON\" + lightIndex);\n }\n if (defines[\"SHADOWESM\" + lightIndex]) {\n fallbacks.addFallback(rank, \"SHADOWESM\" + lightIndex);\n }\n if (defines[\"SHADOWCLOSEESM\" + lightIndex]) {\n fallbacks.addFallback(rank, \"SHADOWCLOSEESM\" + lightIndex);\n }\n }\n }\n return lightFallbackRank++;\n }\n /**\n * Prepares the list of attributes required for morph targets according to the effect defines.\n * @param attribs The current list of supported attribs\n * @param mesh The mesh to prepare the morph targets attributes for\n * @param influencers The number of influencers\n */\n static PrepareAttributesForMorphTargetsInfluencers(attribs, mesh, influencers) {\n this._TmpMorphInfluencers.NUM_MORPH_INFLUENCERS = influencers;\n this.PrepareAttributesForMorphTargets(attribs, mesh, this._TmpMorphInfluencers);\n }\n /**\n * Prepares the list of attributes required for morph targets according to the effect defines.\n * @param attribs The current list of supported attribs\n * @param mesh The mesh to prepare the morph targets attributes for\n * @param defines The current Defines of the effect\n */\n static PrepareAttributesForMorphTargets(attribs, mesh, defines) {\n const influencers = defines[\"NUM_MORPH_INFLUENCERS\"];\n if (influencers > 0 && EngineStore.LastCreatedEngine) {\n const maxAttributesCount = EngineStore.LastCreatedEngine.getCaps().maxVertexAttribs;\n const manager = mesh.morphTargetManager;\n if (manager === null || manager === void 0 ? void 0 : manager.isUsingTextureForTargets) {\n return;\n }\n const normal = manager && manager.supportsNormals && defines[\"NORMAL\"];\n const tangent = manager && manager.supportsTangents && defines[\"TANGENT\"];\n const uv = manager && manager.supportsUVs && defines[\"UV1\"];\n for (let index = 0; index < influencers; index++) {\n attribs.push(VertexBuffer.PositionKind + index);\n if (normal) {\n attribs.push(VertexBuffer.NormalKind + index);\n }\n if (tangent) {\n attribs.push(VertexBuffer.TangentKind + index);\n }\n if (uv) {\n attribs.push(VertexBuffer.UVKind + \"_\" + index);\n }\n if (attribs.length > maxAttributesCount) {\n Logger.Error(\"Cannot add more vertex attributes for mesh \" + mesh.name);\n }\n }\n }\n }\n /**\n * Prepares the list of attributes required for baked vertex animations according to the effect defines.\n * @param attribs The current list of supported attribs\n * @param mesh The mesh to prepare the morph targets attributes for\n * @param defines The current Defines of the effect\n */\n static PrepareAttributesForBakedVertexAnimation(attribs, mesh, defines) {\n const enabled = defines[\"BAKED_VERTEX_ANIMATION_TEXTURE\"] && defines[\"INSTANCES\"];\n if (enabled) {\n attribs.push(\"bakedVertexAnimationSettingsInstanced\");\n }\n }\n /**\n * Prepares the list of attributes required for bones according to the effect defines.\n * @param attribs The current list of supported attribs\n * @param mesh The mesh to prepare the bones attributes for\n * @param defines The current Defines of the effect\n * @param fallbacks The current effect fallback strategy\n */\n static PrepareAttributesForBones(attribs, mesh, defines, fallbacks) {\n if (defines[\"NUM_BONE_INFLUENCERS\"] > 0) {\n fallbacks.addCPUSkinningFallback(0, mesh);\n attribs.push(VertexBuffer.MatricesIndicesKind);\n attribs.push(VertexBuffer.MatricesWeightsKind);\n if (defines[\"NUM_BONE_INFLUENCERS\"] > 4) {\n attribs.push(VertexBuffer.MatricesIndicesExtraKind);\n attribs.push(VertexBuffer.MatricesWeightsExtraKind);\n }\n }\n }\n /**\n * Check and prepare the list of attributes required for instances according to the effect defines.\n * @param attribs The current list of supported attribs\n * @param defines The current MaterialDefines of the effect\n */\n static PrepareAttributesForInstances(attribs, defines) {\n if (defines[\"INSTANCES\"] || defines[\"THIN_INSTANCES\"]) {\n this.PushAttributesForInstances(attribs, !!defines[\"PREPASS_VELOCITY\"]);\n }\n if (defines.INSTANCESCOLOR) {\n attribs.push(VertexBuffer.ColorInstanceKind);\n }\n }\n /**\n * Add the list of attributes required for instances to the attribs array.\n * @param attribs The current list of supported attribs\n * @param needsPreviousMatrices If the shader needs previous matrices\n */\n static PushAttributesForInstances(attribs, needsPreviousMatrices = false) {\n attribs.push(\"world0\");\n attribs.push(\"world1\");\n attribs.push(\"world2\");\n attribs.push(\"world3\");\n if (needsPreviousMatrices) {\n attribs.push(\"previousWorld0\");\n attribs.push(\"previousWorld1\");\n attribs.push(\"previousWorld2\");\n attribs.push(\"previousWorld3\");\n }\n }\n /**\n * Binds the light information to the effect.\n * @param light The light containing the generator\n * @param effect The effect we are binding the data to\n * @param lightIndex The light index in the effect used to render\n */\n static BindLightProperties(light, effect, lightIndex) {\n light.transferToEffect(effect, lightIndex + \"\");\n }\n /**\n * Binds the lights information from the scene to the effect for the given mesh.\n * @param light Light to bind\n * @param lightIndex Light index\n * @param scene The scene where the light belongs to\n * @param effect The effect we are binding the data to\n * @param useSpecular Defines if specular is supported\n * @param receiveShadows Defines if the effect (mesh) we bind the light for receives shadows\n */\n static BindLight(light, lightIndex, scene, effect, useSpecular, receiveShadows = true) {\n light._bindLight(lightIndex, scene, effect, useSpecular, receiveShadows);\n }\n /**\n * Binds the lights information from the scene to the effect for the given mesh.\n * @param scene The scene the lights belongs to\n * @param mesh The mesh we are binding the information to render\n * @param effect The effect we are binding the data to\n * @param defines The generated defines for the effect\n * @param maxSimultaneousLights The maximum number of light that can be bound to the effect\n */\n static BindLights(scene, mesh, effect, defines, maxSimultaneousLights = 4) {\n const len = Math.min(mesh.lightSources.length, maxSimultaneousLights);\n for (let i = 0; i < len; i++) {\n const light = mesh.lightSources[i];\n this.BindLight(light, i, scene, effect, typeof defines === \"boolean\" ? defines : defines[\"SPECULARTERM\"], mesh.receiveShadows);\n }\n }\n /**\n * Binds the fog information from the scene to the effect for the given mesh.\n * @param scene The scene the lights belongs to\n * @param mesh The mesh we are binding the information to render\n * @param effect The effect we are binding the data to\n * @param linearSpace Defines if the fog effect is applied in linear space\n */\n static BindFogParameters(scene, mesh, effect, linearSpace = false) {\n if (scene.fogEnabled && mesh.applyFog && scene.fogMode !== Scene.FOGMODE_NONE) {\n effect.setFloat4(\"vFogInfos\", scene.fogMode, scene.fogStart, scene.fogEnd, scene.fogDensity);\n // Convert fog color to linear space if used in a linear space computed shader.\n if (linearSpace) {\n scene.fogColor.toLinearSpaceToRef(this._TempFogColor, scene.getEngine().useExactSrgbConversions);\n effect.setColor3(\"vFogColor\", this._TempFogColor);\n }\n else {\n effect.setColor3(\"vFogColor\", scene.fogColor);\n }\n }\n }\n /**\n * Binds the bones information from the mesh to the effect.\n * @param mesh The mesh we are binding the information to render\n * @param effect The effect we are binding the data to\n * @param prePassConfiguration Configuration for the prepass, in case prepass is activated\n */\n static BindBonesParameters(mesh, effect, prePassConfiguration) {\n if (!effect || !mesh) {\n return;\n }\n if (mesh.computeBonesUsingShaders && effect._bonesComputationForcedToCPU) {\n mesh.computeBonesUsingShaders = false;\n }\n if (mesh.useBones && mesh.computeBonesUsingShaders && mesh.skeleton) {\n const skeleton = mesh.skeleton;\n if (skeleton.isUsingTextureForMatrices && effect.getUniformIndex(\"boneTextureWidth\") > -1) {\n const boneTexture = skeleton.getTransformMatrixTexture(mesh);\n effect.setTexture(\"boneSampler\", boneTexture);\n effect.setFloat(\"boneTextureWidth\", 4.0 * (skeleton.bones.length + 1));\n }\n else {\n const matrices = skeleton.getTransformMatrices(mesh);\n if (matrices) {\n effect.setMatrices(\"mBones\", matrices);\n if (prePassConfiguration && mesh.getScene().prePassRenderer && mesh.getScene().prePassRenderer.getIndex(2)) {\n if (!prePassConfiguration.previousBones[mesh.uniqueId]) {\n prePassConfiguration.previousBones[mesh.uniqueId] = matrices.slice();\n }\n effect.setMatrices(\"mPreviousBones\", prePassConfiguration.previousBones[mesh.uniqueId]);\n MaterialHelper._CopyBonesTransformationMatrices(matrices, prePassConfiguration.previousBones[mesh.uniqueId]);\n }\n }\n }\n }\n }\n // Copies the bones transformation matrices into the target array and returns the target's reference\n static _CopyBonesTransformationMatrices(source, target) {\n target.set(source);\n return target;\n }\n /**\n * Binds the morph targets information from the mesh to the effect.\n * @param abstractMesh The mesh we are binding the information to render\n * @param effect The effect we are binding the data to\n */\n static BindMorphTargetParameters(abstractMesh, effect) {\n const manager = abstractMesh.morphTargetManager;\n if (!abstractMesh || !manager) {\n return;\n }\n effect.setFloatArray(\"morphTargetInfluences\", manager.influences);\n }\n /**\n * Binds the logarithmic depth information from the scene to the effect for the given defines.\n * @param defines The generated defines used in the effect\n * @param effect The effect we are binding the data to\n * @param scene The scene we are willing to render with logarithmic scale for\n */\n static BindLogDepth(defines, effect, scene) {\n if (!defines || defines[\"LOGARITHMICDEPTH\"] || (defines.indexOf && defines.indexOf(\"LOGARITHMICDEPTH\") >= 0)) {\n const camera = scene.activeCamera;\n if (camera.mode === Camera.ORTHOGRAPHIC_CAMERA) {\n Logger.Error(\"Logarithmic depth is not compatible with orthographic cameras!\", 20);\n }\n effect.setFloat(\"logarithmicDepthConstant\", 2.0 / (Math.log(camera.maxZ + 1.0) / Math.LN2));\n }\n }\n }\n // eslint-disable-next-line @typescript-eslint/naming-convention\n MaterialHelper._TmpMorphInfluencers = { NUM_MORPH_INFLUENCERS: 0 };\n MaterialHelper._TempFogColor = Color3.Black();\n\n /**\n * Class that holds the different stencil states of a material\n * Usage example: https://playground.babylonjs.com/#CW5PRI#10\n */\n class MaterialStencilState {\n /**\n * Creates a material stencil state instance\n */\n constructor() {\n this.reset();\n }\n /**\n * Resets all the stencil states to default values\n */\n reset() {\n this.enabled = false;\n this.mask = 0xff;\n this.func = 519;\n this.funcRef = 1;\n this.funcMask = 0xff;\n this.opStencilFail = 7680;\n this.opDepthFail = 7680;\n this.opStencilDepthPass = 7681;\n }\n /**\n * Gets or sets the stencil function\n */\n get func() {\n return this._func;\n }\n set func(value) {\n this._func = value;\n }\n /**\n * Gets or sets the stencil function reference\n */\n get funcRef() {\n return this._funcRef;\n }\n set funcRef(value) {\n this._funcRef = value;\n }\n /**\n * Gets or sets the stencil function mask\n */\n get funcMask() {\n return this._funcMask;\n }\n set funcMask(value) {\n this._funcMask = value;\n }\n /**\n * Gets or sets the operation when the stencil test fails\n */\n get opStencilFail() {\n return this._opStencilFail;\n }\n set opStencilFail(value) {\n this._opStencilFail = value;\n }\n /**\n * Gets or sets the operation when the depth test fails\n */\n get opDepthFail() {\n return this._opDepthFail;\n }\n set opDepthFail(value) {\n this._opDepthFail = value;\n }\n /**\n * Gets or sets the operation when the stencil+depth test succeeds\n */\n get opStencilDepthPass() {\n return this._opStencilDepthPass;\n }\n set opStencilDepthPass(value) {\n this._opStencilDepthPass = value;\n }\n /**\n * Gets or sets the stencil mask\n */\n get mask() {\n return this._mask;\n }\n set mask(value) {\n this._mask = value;\n }\n /**\n * Enables or disables the stencil test\n */\n get enabled() {\n return this._enabled;\n }\n set enabled(value) {\n this._enabled = value;\n }\n /**\n * Get the current class name, useful for serialization or dynamic coding.\n * @returns \"MaterialStencilState\"\n */\n getClassName() {\n return \"MaterialStencilState\";\n }\n /**\n * Makes a duplicate of the current configuration into another one.\n * @param stencilState defines stencil state where to copy the info\n */\n copyTo(stencilState) {\n SerializationHelper.Clone(() => stencilState, this);\n }\n /**\n * Serializes this stencil configuration.\n * @returns - An object with the serialized config.\n */\n serialize() {\n return SerializationHelper.Serialize(this);\n }\n /**\n * Parses a stencil state configuration from a serialized object.\n * @param source - Serialized object.\n * @param scene Defines the scene we are parsing for\n * @param rootUrl Defines the rootUrl to load from\n */\n parse(source, scene, rootUrl) {\n SerializationHelper.Parse(() => this, source, scene, rootUrl);\n }\n }\n __decorate$1([\n serialize()\n ], MaterialStencilState.prototype, \"func\", null);\n __decorate$1([\n serialize()\n ], MaterialStencilState.prototype, \"funcRef\", null);\n __decorate$1([\n serialize()\n ], MaterialStencilState.prototype, \"funcMask\", null);\n __decorate$1([\n serialize()\n ], MaterialStencilState.prototype, \"opStencilFail\", null);\n __decorate$1([\n serialize()\n ], MaterialStencilState.prototype, \"opDepthFail\", null);\n __decorate$1([\n serialize()\n ], MaterialStencilState.prototype, \"opStencilDepthPass\", null);\n __decorate$1([\n serialize()\n ], MaterialStencilState.prototype, \"mask\", null);\n __decorate$1([\n serialize()\n ], MaterialStencilState.prototype, \"enabled\", null);\n\n /**\n * @internal\n */\n var MaterialPluginEvent;\n (function (MaterialPluginEvent) {\n MaterialPluginEvent[MaterialPluginEvent[\"Created\"] = 1] = \"Created\";\n MaterialPluginEvent[MaterialPluginEvent[\"Disposed\"] = 2] = \"Disposed\";\n MaterialPluginEvent[MaterialPluginEvent[\"GetDefineNames\"] = 4] = \"GetDefineNames\";\n MaterialPluginEvent[MaterialPluginEvent[\"PrepareUniformBuffer\"] = 8] = \"PrepareUniformBuffer\";\n MaterialPluginEvent[MaterialPluginEvent[\"IsReadyForSubMesh\"] = 16] = \"IsReadyForSubMesh\";\n MaterialPluginEvent[MaterialPluginEvent[\"PrepareDefines\"] = 32] = \"PrepareDefines\";\n MaterialPluginEvent[MaterialPluginEvent[\"BindForSubMesh\"] = 64] = \"BindForSubMesh\";\n MaterialPluginEvent[MaterialPluginEvent[\"PrepareEffect\"] = 128] = \"PrepareEffect\";\n MaterialPluginEvent[MaterialPluginEvent[\"GetAnimatables\"] = 256] = \"GetAnimatables\";\n MaterialPluginEvent[MaterialPluginEvent[\"GetActiveTextures\"] = 512] = \"GetActiveTextures\";\n MaterialPluginEvent[MaterialPluginEvent[\"HasTexture\"] = 1024] = \"HasTexture\";\n MaterialPluginEvent[MaterialPluginEvent[\"FillRenderTargetTextures\"] = 2048] = \"FillRenderTargetTextures\";\n MaterialPluginEvent[MaterialPluginEvent[\"HasRenderTargetTextures\"] = 4096] = \"HasRenderTargetTextures\";\n MaterialPluginEvent[MaterialPluginEvent[\"HardBindForSubMesh\"] = 8192] = \"HardBindForSubMesh\";\n })(MaterialPluginEvent || (MaterialPluginEvent = {}));\n\n /**\n * Base class for the main features of a material in Babylon.js\n */\n class Material {\n /**\n * If the material can be rendered to several textures with MRT extension\n */\n get canRenderToMRT() {\n // By default, shaders are not compatible with MRTs\n // Base classes should override that if their shader supports MRT\n return false;\n }\n /**\n * Sets the alpha value of the material\n */\n set alpha(value) {\n if (this._alpha === value) {\n return;\n }\n const oldValue = this._alpha;\n this._alpha = value;\n // Only call dirty when there is a state change (no alpha / alpha)\n if (oldValue === 1 || value === 1) {\n this.markAsDirty(Material.MiscDirtyFlag + Material.PrePassDirtyFlag);\n }\n }\n /**\n * Gets the alpha value of the material\n */\n get alpha() {\n return this._alpha;\n }\n /**\n * Sets the culling state (true to enable culling, false to disable)\n */\n set backFaceCulling(value) {\n if (this._backFaceCulling === value) {\n return;\n }\n this._backFaceCulling = value;\n this.markAsDirty(Material.TextureDirtyFlag);\n }\n /**\n * Gets the culling state\n */\n get backFaceCulling() {\n return this._backFaceCulling;\n }\n /**\n * Sets the type of faces that should be culled (true for back faces, false for front faces)\n */\n set cullBackFaces(value) {\n if (this._cullBackFaces === value) {\n return;\n }\n this._cullBackFaces = value;\n this.markAsDirty(Material.TextureDirtyFlag);\n }\n /**\n * Gets the type of faces that should be culled\n */\n get cullBackFaces() {\n return this._cullBackFaces;\n }\n /**\n * Block the dirty-mechanism for this specific material\n * When set to false after being true the material will be marked as dirty.\n */\n get blockDirtyMechanism() {\n return this._blockDirtyMechanism;\n }\n set blockDirtyMechanism(value) {\n if (this._blockDirtyMechanism === value) {\n return;\n }\n this._blockDirtyMechanism = value;\n if (!value) {\n this.markDirty();\n }\n }\n /**\n * This allows you to modify the material without marking it as dirty after every change.\n * This function should be used if you need to make more than one dirty-enabling change to the material - adding a texture, setting a new fill mode and so on.\n * The callback will pass the material as an argument, so you can make your changes to it.\n * @param callback the callback to be executed that will update the material\n */\n atomicMaterialsUpdate(callback) {\n this.blockDirtyMechanism = true;\n try {\n callback(this);\n }\n finally {\n this.blockDirtyMechanism = false;\n }\n }\n /**\n * Gets a boolean indicating that current material needs to register RTT\n */\n get hasRenderTargetTextures() {\n this._eventInfo.hasRenderTargetTextures = false;\n this._callbackPluginEventHasRenderTargetTextures(this._eventInfo);\n return this._eventInfo.hasRenderTargetTextures;\n }\n /**\n * Called during a dispose event\n */\n set onDispose(callback) {\n if (this._onDisposeObserver) {\n this.onDisposeObservable.remove(this._onDisposeObserver);\n }\n this._onDisposeObserver = this.onDisposeObservable.add(callback);\n }\n /**\n * An event triggered when the material is bound\n */\n get onBindObservable() {\n if (!this._onBindObservable) {\n this._onBindObservable = new Observable$1();\n }\n return this._onBindObservable;\n }\n /**\n * Called during a bind event\n */\n set onBind(callback) {\n if (this._onBindObserver) {\n this.onBindObservable.remove(this._onBindObserver);\n }\n this._onBindObserver = this.onBindObservable.add(callback);\n }\n /**\n * An event triggered when the material is unbound\n */\n get onUnBindObservable() {\n if (!this._onUnBindObservable) {\n this._onUnBindObservable = new Observable$1();\n }\n return this._onUnBindObservable;\n }\n /**\n * An event triggered when the effect is (re)created\n */\n get onEffectCreatedObservable() {\n if (!this._onEffectCreatedObservable) {\n this._onEffectCreatedObservable = new Observable$1();\n }\n return this._onEffectCreatedObservable;\n }\n /**\n * Sets the value of the alpha mode.\n *\n * | Value | Type | Description |\n * | --- | --- | --- |\n * | 0 | ALPHA_DISABLE | |\n * | 1 | ALPHA_ADD | |\n * | 2 | ALPHA_COMBINE | |\n * | 3 | ALPHA_SUBTRACT | |\n * | 4 | ALPHA_MULTIPLY | |\n * | 5 | ALPHA_MAXIMIZED | |\n * | 6 | ALPHA_ONEONE | |\n * | 7 | ALPHA_PREMULTIPLIED | |\n * | 8 | ALPHA_PREMULTIPLIED_PORTERDUFF | |\n * | 9 | ALPHA_INTERPOLATE | |\n * | 10 | ALPHA_SCREENMODE | |\n *\n */\n set alphaMode(value) {\n if (this._alphaMode === value) {\n return;\n }\n this._alphaMode = value;\n this.markAsDirty(Material.TextureDirtyFlag);\n }\n /**\n * Gets the value of the alpha mode\n */\n get alphaMode() {\n return this._alphaMode;\n }\n /**\n * Sets the need depth pre-pass value\n */\n set needDepthPrePass(value) {\n if (this._needDepthPrePass === value) {\n return;\n }\n this._needDepthPrePass = value;\n if (this._needDepthPrePass) {\n this.checkReadyOnEveryCall = true;\n }\n }\n /**\n * Gets the depth pre-pass value\n */\n get needDepthPrePass() {\n return this._needDepthPrePass;\n }\n /**\n * Can this material render to prepass\n */\n get isPrePassCapable() {\n return false;\n }\n /**\n * Sets the state for enabling fog\n */\n set fogEnabled(value) {\n if (this._fogEnabled === value) {\n return;\n }\n this._fogEnabled = value;\n this.markAsDirty(Material.MiscDirtyFlag);\n }\n /**\n * Gets the value of the fog enabled state\n */\n get fogEnabled() {\n return this._fogEnabled;\n }\n get wireframe() {\n switch (this._fillMode) {\n case Material.WireFrameFillMode:\n case Material.LineListDrawMode:\n case Material.LineLoopDrawMode:\n case Material.LineStripDrawMode:\n return true;\n }\n return this._scene.forceWireframe;\n }\n /**\n * Sets the state of wireframe mode\n */\n set wireframe(value) {\n this.fillMode = value ? Material.WireFrameFillMode : Material.TriangleFillMode;\n }\n /**\n * Gets the value specifying if point clouds are enabled\n */\n get pointsCloud() {\n switch (this._fillMode) {\n case Material.PointFillMode:\n case Material.PointListDrawMode:\n return true;\n }\n return this._scene.forcePointsCloud;\n }\n /**\n * Sets the state of point cloud mode\n */\n set pointsCloud(value) {\n this.fillMode = value ? Material.PointFillMode : Material.TriangleFillMode;\n }\n /**\n * Gets the material fill mode\n */\n get fillMode() {\n return this._fillMode;\n }\n /**\n * Sets the material fill mode\n */\n set fillMode(value) {\n if (this._fillMode === value) {\n return;\n }\n this._fillMode = value;\n this.markAsDirty(Material.MiscDirtyFlag);\n }\n /** @internal */\n _getDrawWrapper() {\n return this._drawWrapper;\n }\n /**\n * @internal\n */\n _setDrawWrapper(drawWrapper) {\n this._drawWrapper = drawWrapper;\n }\n /**\n * Creates a material instance\n * @param name defines the name of the material\n * @param scene defines the scene to reference\n * @param doNotAdd specifies if the material should be added to the scene\n */\n constructor(name, scene, doNotAdd) {\n /**\n * Custom shadow depth material to use for shadow rendering instead of the in-built one\n */\n this.shadowDepthWrapper = null;\n /**\n * Gets or sets a boolean indicating that the material is allowed (if supported) to do shader hot swapping.\n * This means that the material can keep using a previous shader while a new one is being compiled.\n * This is mostly used when shader parallel compilation is supported (true by default)\n */\n this.allowShaderHotSwapping = true;\n /**\n * Gets or sets user defined metadata\n */\n this.metadata = null;\n /**\n * For internal use only. Please do not use.\n */\n this.reservedDataStore = null;\n /**\n * Specifies if the ready state should be checked on each call\n */\n this.checkReadyOnEveryCall = false;\n /**\n * Specifies if the ready state should be checked once\n */\n this.checkReadyOnlyOnce = false;\n /**\n * The state of the material\n */\n this.state = \"\";\n /**\n * The alpha value of the material\n */\n this._alpha = 1.0;\n /**\n * Specifies if back face culling is enabled\n */\n this._backFaceCulling = true;\n /**\n * Specifies if back or front faces should be culled (when culling is enabled)\n */\n this._cullBackFaces = true;\n this._blockDirtyMechanism = false;\n /**\n * Callback triggered when the material is compiled\n */\n this.onCompiled = null;\n /**\n * Callback triggered when an error occurs\n */\n this.onError = null;\n /**\n * Callback triggered to get the render target textures\n */\n this.getRenderTargetTextures = null;\n /**\n * Specifies if the material should be serialized\n */\n this.doNotSerialize = false;\n /**\n * @internal\n */\n this._storeEffectOnSubMeshes = false;\n /**\n * Stores the animations for the material\n */\n this.animations = null;\n /**\n * An event triggered when the material is disposed\n */\n this.onDisposeObservable = new Observable$1();\n /**\n * An observer which watches for dispose events\n */\n this._onDisposeObserver = null;\n this._onUnBindObservable = null;\n /**\n * An observer which watches for bind events\n */\n this._onBindObserver = null;\n /**\n * Stores the value of the alpha mode\n */\n this._alphaMode = 2;\n /**\n * Stores the state of the need depth pre-pass value\n */\n this._needDepthPrePass = false;\n /**\n * Specifies if depth writing should be disabled\n */\n this.disableDepthWrite = false;\n /**\n * Specifies if color writing should be disabled\n */\n this.disableColorWrite = false;\n /**\n * Specifies if depth writing should be forced\n */\n this.forceDepthWrite = false;\n /**\n * Specifies the depth function that should be used. 0 means the default engine function\n */\n this.depthFunction = 0;\n /**\n * Specifies if there should be a separate pass for culling\n */\n this.separateCullingPass = false;\n /**\n * Stores the state specifying if fog should be enabled\n */\n this._fogEnabled = true;\n /**\n * Stores the size of points\n */\n this.pointSize = 1.0;\n /**\n * Stores the z offset Factor value\n */\n this.zOffset = 0;\n /**\n * Stores the z offset Units value\n */\n this.zOffsetUnits = 0;\n /**\n * Gives access to the stencil properties of the material\n */\n this.stencil = new MaterialStencilState();\n /**\n * Specifies if uniform buffers should be used\n */\n this._useUBO = false;\n /**\n * Stores the fill mode state\n */\n this._fillMode = Material.TriangleFillMode;\n /**\n * Specifies if the depth write state should be cached\n */\n this._cachedDepthWriteState = false;\n /**\n * Specifies if the color write state should be cached\n */\n this._cachedColorWriteState = false;\n /**\n * Specifies if the depth function state should be cached\n */\n this._cachedDepthFunctionState = 0;\n /** @internal */\n this._indexInSceneMaterialArray = -1;\n /** @internal */\n this.meshMap = null;\n /** @internal */\n this._parentContainer = null;\n /** @internal */\n this._uniformBufferLayoutBuilt = false;\n this._eventInfo = {}; // will be initialized before each event notification\n /** @internal */\n this._callbackPluginEventGeneric = () => void 0;\n /** @internal */\n this._callbackPluginEventIsReadyForSubMesh = () => void 0;\n /** @internal */\n this._callbackPluginEventPrepareDefines = () => void 0;\n /** @internal */\n this._callbackPluginEventPrepareDefinesBeforeAttributes = () => void 0;\n /** @internal */\n this._callbackPluginEventHardBindForSubMesh = () => void 0;\n /** @internal */\n this._callbackPluginEventBindForSubMesh = () => void 0;\n /** @internal */\n this._callbackPluginEventHasRenderTargetTextures = () => void 0;\n /** @internal */\n this._callbackPluginEventFillRenderTargetTextures = () => void 0;\n /**\n * Enforces alpha test in opaque or blend mode in order to improve the performances of some situations.\n */\n this._forceAlphaTest = false;\n /**\n * The transparency mode of the material.\n */\n this._transparencyMode = null;\n this.name = name;\n const setScene = scene || EngineStore.LastCreatedScene;\n if (!setScene) {\n return;\n }\n this._scene = setScene;\n this._dirtyCallbacks = {};\n this._dirtyCallbacks[1] = this._markAllSubMeshesAsTexturesDirty.bind(this);\n this._dirtyCallbacks[2] = this._markAllSubMeshesAsLightsDirty.bind(this);\n this._dirtyCallbacks[4] = this._markAllSubMeshesAsFresnelDirty.bind(this);\n this._dirtyCallbacks[8] = this._markAllSubMeshesAsAttributesDirty.bind(this);\n this._dirtyCallbacks[16] = this._markAllSubMeshesAsMiscDirty.bind(this);\n this._dirtyCallbacks[32] = this._markAllSubMeshesAsPrePassDirty.bind(this);\n this._dirtyCallbacks[63] = this._markAllSubMeshesAsAllDirty.bind(this);\n this.id = name || Tools.RandomId();\n this.uniqueId = this._scene.getUniqueId();\n this._materialContext = this._scene.getEngine().createMaterialContext();\n this._drawWrapper = new DrawWrapper(this._scene.getEngine(), false);\n this._drawWrapper.materialContext = this._materialContext;\n if (this._scene.useRightHandedSystem) {\n this.sideOrientation = Material.ClockWiseSideOrientation;\n }\n else {\n this.sideOrientation = Material.CounterClockWiseSideOrientation;\n }\n this._uniformBuffer = new UniformBuffer(this._scene.getEngine(), undefined, undefined, name);\n this._useUBO = this.getScene().getEngine().supportsUniformBuffers;\n if (!doNotAdd) {\n this._scene.addMaterial(this);\n }\n if (this._scene.useMaterialMeshMap) {\n this.meshMap = {};\n }\n Material.OnEventObservable.notifyObservers(this, MaterialPluginEvent.Created);\n }\n /**\n * Returns a string representation of the current material\n * @param fullDetails defines a boolean indicating which levels of logging is desired\n * @returns a string with material information\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n toString(fullDetails) {\n const ret = \"Name: \" + this.name;\n return ret;\n }\n /**\n * Gets the class name of the material\n * @returns a string with the class name of the material\n */\n getClassName() {\n return \"Material\";\n }\n /** @internal */\n get _isMaterial() {\n return true;\n }\n /**\n * Specifies if updates for the material been locked\n */\n get isFrozen() {\n return this.checkReadyOnlyOnce;\n }\n /**\n * Locks updates for the material\n */\n freeze() {\n this.markDirty();\n this.checkReadyOnlyOnce = true;\n }\n /**\n * Unlocks updates for the material\n */\n unfreeze() {\n this.markDirty();\n this.checkReadyOnlyOnce = false;\n }\n /**\n * Specifies if the material is ready to be used\n * @param mesh defines the mesh to check\n * @param useInstances specifies if instances should be used\n * @returns a boolean indicating if the material is ready to be used\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n isReady(mesh, useInstances) {\n return true;\n }\n /**\n * Specifies that the submesh is ready to be used\n * @param mesh defines the mesh to check\n * @param subMesh defines which submesh to check\n * @param useInstances specifies that instances should be used\n * @returns a boolean indicating that the submesh is ready or not\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n isReadyForSubMesh(mesh, subMesh, useInstances) {\n const defines = subMesh.materialDefines;\n if (!defines) {\n return false;\n }\n this._eventInfo.isReadyForSubMesh = true;\n this._eventInfo.defines = defines;\n this._callbackPluginEventIsReadyForSubMesh(this._eventInfo);\n return this._eventInfo.isReadyForSubMesh;\n }\n /**\n * Returns the material effect\n * @returns the effect associated with the material\n */\n getEffect() {\n return this._drawWrapper.effect;\n }\n /**\n * Returns the current scene\n * @returns a Scene\n */\n getScene() {\n return this._scene;\n }\n /**\n * Gets the current transparency mode.\n */\n get transparencyMode() {\n return this._transparencyMode;\n }\n /**\n * Sets the transparency mode of the material.\n *\n * | Value | Type | Description |\n * | ----- | ----------------------------------- | ----------- |\n * | 0 | OPAQUE | |\n * | 1 | ALPHATEST | |\n * | 2 | ALPHABLEND | |\n * | 3 | ALPHATESTANDBLEND | |\n *\n */\n set transparencyMode(value) {\n if (this._transparencyMode === value) {\n return;\n }\n this._transparencyMode = value;\n this._forceAlphaTest = value === Material.MATERIAL_ALPHATESTANDBLEND;\n this._markAllSubMeshesAsTexturesAndMiscDirty();\n }\n /**\n * Returns true if alpha blending should be disabled.\n */\n get _disableAlphaBlending() {\n return this._transparencyMode === Material.MATERIAL_OPAQUE || this._transparencyMode === Material.MATERIAL_ALPHATEST;\n }\n /**\n * Specifies whether or not this material should be rendered in alpha blend mode.\n * @returns a boolean specifying if alpha blending is needed\n */\n needAlphaBlending() {\n if (this._disableAlphaBlending) {\n return false;\n }\n return this.alpha < 1.0;\n }\n /**\n * Specifies if the mesh will require alpha blending\n * @param mesh defines the mesh to check\n * @returns a boolean specifying if alpha blending is needed for the mesh\n */\n needAlphaBlendingForMesh(mesh) {\n if (mesh.visibility < 1.0) {\n return true;\n }\n if (this._disableAlphaBlending) {\n return false;\n }\n return mesh.hasVertexAlpha || this.needAlphaBlending();\n }\n /**\n * Specifies whether or not this material should be rendered in alpha test mode.\n * @returns a boolean specifying if an alpha test is needed.\n */\n needAlphaTesting() {\n if (this._forceAlphaTest) {\n return true;\n }\n return false;\n }\n /**\n * Specifies if material alpha testing should be turned on for the mesh\n * @param mesh defines the mesh to check\n */\n _shouldTurnAlphaTestOn(mesh) {\n return !this.needAlphaBlendingForMesh(mesh) && this.needAlphaTesting();\n }\n /**\n * Gets the texture used for the alpha test\n * @returns the texture to use for alpha testing\n */\n getAlphaTestTexture() {\n return null;\n }\n /**\n * Marks the material to indicate that it needs to be re-calculated\n * @param forceMaterialDirty - Forces the material to be marked as dirty for all components (same as this.markAsDirty(Material.AllDirtyFlag)). You should use this flag if the material is frozen and you want to force a recompilation.\n */\n markDirty(forceMaterialDirty = false) {\n const meshes = this.getScene().meshes;\n for (const mesh of meshes) {\n if (!mesh.subMeshes) {\n continue;\n }\n for (const subMesh of mesh.subMeshes) {\n if (subMesh.getMaterial() !== this) {\n continue;\n }\n if (!subMesh.effect) {\n continue;\n }\n subMesh.effect._wasPreviouslyReady = false;\n subMesh.effect._wasPreviouslyUsingInstances = null;\n subMesh.effect._forceRebindOnNextCall = forceMaterialDirty;\n }\n }\n if (forceMaterialDirty) {\n this.markAsDirty(Material.AllDirtyFlag);\n }\n }\n /**\n * @internal\n */\n _preBind(effect, overrideOrientation = null) {\n const engine = this._scene.getEngine();\n const orientation = overrideOrientation == null ? this.sideOrientation : overrideOrientation;\n const reverse = orientation === Material.ClockWiseSideOrientation;\n engine.enableEffect(effect ? effect : this._getDrawWrapper());\n engine.setState(this.backFaceCulling, this.zOffset, false, reverse, this._scene._mirroredCameraPosition ? !this.cullBackFaces : this.cullBackFaces, this.stencil, this.zOffsetUnits);\n return reverse;\n }\n /**\n * Binds the material to the mesh\n * @param world defines the world transformation matrix\n * @param mesh defines the mesh to bind the material to\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n bind(world, mesh) { }\n /**\n * Initializes the uniform buffer layout for the shader.\n */\n buildUniformLayout() {\n const ubo = this._uniformBuffer;\n this._eventInfo.ubo = ubo;\n this._callbackPluginEventGeneric(MaterialPluginEvent.PrepareUniformBuffer, this._eventInfo);\n ubo.create();\n this._uniformBufferLayoutBuilt = true;\n }\n /**\n * Binds the submesh to the material\n * @param world defines the world transformation matrix\n * @param mesh defines the mesh containing the submesh\n * @param subMesh defines the submesh to bind the material to\n */\n bindForSubMesh(world, mesh, subMesh) {\n const effect = subMesh.effect;\n if (!effect) {\n return;\n }\n this._eventInfo.subMesh = subMesh;\n this._callbackPluginEventBindForSubMesh(this._eventInfo);\n effect._forceRebindOnNextCall = false;\n }\n /**\n * Binds the world matrix to the material\n * @param world defines the world transformation matrix\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n bindOnlyWorldMatrix(world) { }\n /**\n * Binds the view matrix to the effect\n * @param effect defines the effect to bind the view matrix to\n */\n bindView(effect) {\n if (!this._useUBO) {\n effect.setMatrix(\"view\", this.getScene().getViewMatrix());\n }\n else {\n this._needToBindSceneUbo = true;\n }\n }\n /**\n * Binds the view projection and projection matrices to the effect\n * @param effect defines the effect to bind the view projection and projection matrices to\n */\n bindViewProjection(effect) {\n if (!this._useUBO) {\n effect.setMatrix(\"viewProjection\", this.getScene().getTransformMatrix());\n effect.setMatrix(\"projection\", this.getScene().getProjectionMatrix());\n }\n else {\n this._needToBindSceneUbo = true;\n }\n }\n /**\n * Binds the view matrix to the effect\n * @param effect defines the effect to bind the view matrix to\n * @param variableName name of the shader variable that will hold the eye position\n */\n bindEyePosition(effect, variableName) {\n if (!this._useUBO) {\n this._scene.bindEyePosition(effect, variableName);\n }\n else {\n this._needToBindSceneUbo = true;\n }\n }\n /**\n * Processes to execute after binding the material to a mesh\n * @param mesh defines the rendered mesh\n * @param effect\n */\n _afterBind(mesh, effect = null) {\n this._scene._cachedMaterial = this;\n if (this._needToBindSceneUbo) {\n if (effect) {\n this._needToBindSceneUbo = false;\n MaterialHelper.BindSceneUniformBuffer(effect, this.getScene().getSceneUniformBuffer());\n this._scene.finalizeSceneUbo();\n }\n }\n if (mesh) {\n this._scene._cachedVisibility = mesh.visibility;\n }\n else {\n this._scene._cachedVisibility = 1;\n }\n if (this._onBindObservable && mesh) {\n this._onBindObservable.notifyObservers(mesh);\n }\n if (this.disableDepthWrite) {\n const engine = this._scene.getEngine();\n this._cachedDepthWriteState = engine.getDepthWrite();\n engine.setDepthWrite(false);\n }\n if (this.disableColorWrite) {\n const engine = this._scene.getEngine();\n this._cachedColorWriteState = engine.getColorWrite();\n engine.setColorWrite(false);\n }\n if (this.depthFunction !== 0) {\n const engine = this._scene.getEngine();\n this._cachedDepthFunctionState = engine.getDepthFunction() || 0;\n engine.setDepthFunction(this.depthFunction);\n }\n }\n /**\n * Unbinds the material from the mesh\n */\n unbind() {\n if (this._onUnBindObservable) {\n this._onUnBindObservable.notifyObservers(this);\n }\n if (this.depthFunction !== 0) {\n const engine = this._scene.getEngine();\n engine.setDepthFunction(this._cachedDepthFunctionState);\n }\n if (this.disableDepthWrite) {\n const engine = this._scene.getEngine();\n engine.setDepthWrite(this._cachedDepthWriteState);\n }\n if (this.disableColorWrite) {\n const engine = this._scene.getEngine();\n engine.setColorWrite(this._cachedColorWriteState);\n }\n }\n /**\n * Returns the animatable textures.\n * @returns - Array of animatable textures.\n */\n getAnimatables() {\n this._eventInfo.animatables = [];\n this._callbackPluginEventGeneric(MaterialPluginEvent.GetAnimatables, this._eventInfo);\n return this._eventInfo.animatables;\n }\n /**\n * Gets the active textures from the material\n * @returns an array of textures\n */\n getActiveTextures() {\n this._eventInfo.activeTextures = [];\n this._callbackPluginEventGeneric(MaterialPluginEvent.GetActiveTextures, this._eventInfo);\n return this._eventInfo.activeTextures;\n }\n /**\n * Specifies if the material uses a texture\n * @param texture defines the texture to check against the material\n * @returns a boolean specifying if the material uses the texture\n */\n hasTexture(texture) {\n this._eventInfo.hasTexture = false;\n this._eventInfo.texture = texture;\n this._callbackPluginEventGeneric(MaterialPluginEvent.HasTexture, this._eventInfo);\n return this._eventInfo.hasTexture;\n }\n /**\n * Makes a duplicate of the material, and gives it a new name\n * @param name defines the new name for the duplicated material\n * @returns the cloned material\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n clone(name) {\n return null;\n }\n _clonePlugins(targetMaterial, rootUrl) {\n const serializationObject = {};\n // Create plugins in targetMaterial in case they don't exist\n this._serializePlugins(serializationObject);\n Material._parsePlugins(serializationObject, targetMaterial, this._scene, rootUrl);\n // Copy the properties of the current plugins to the cloned material's plugins\n if (this.pluginManager) {\n for (const plugin of this.pluginManager._plugins) {\n const targetPlugin = targetMaterial.pluginManager.getPlugin(plugin.name);\n plugin.copyTo(targetPlugin);\n }\n }\n }\n /**\n * Gets the meshes bound to the material\n * @returns an array of meshes bound to the material\n */\n getBindedMeshes() {\n if (this.meshMap) {\n const result = new Array();\n for (const meshId in this.meshMap) {\n const mesh = this.meshMap[meshId];\n if (mesh) {\n result.push(mesh);\n }\n }\n return result;\n }\n else {\n const meshes = this._scene.meshes;\n return meshes.filter((mesh) => mesh.material === this);\n }\n }\n /**\n * Force shader compilation\n * @param mesh defines the mesh associated with this material\n * @param onCompiled defines a function to execute once the material is compiled\n * @param options defines the options to configure the compilation\n * @param onError defines a function to execute if the material fails compiling\n */\n forceCompilation(mesh, onCompiled, options, onError) {\n const localOptions = Object.assign({ clipPlane: false, useInstances: false }, options);\n const scene = this.getScene();\n const currentHotSwapingState = this.allowShaderHotSwapping;\n this.allowShaderHotSwapping = false; // Turned off to let us evaluate the real compilation state\n const checkReady = () => {\n if (!this._scene || !this._scene.getEngine()) {\n return;\n }\n const clipPlaneState = scene.clipPlane;\n if (localOptions.clipPlane) {\n scene.clipPlane = new Plane(0, 0, 0, 1);\n }\n if (this._storeEffectOnSubMeshes) {\n let allDone = true, lastError = null;\n if (mesh.subMeshes) {\n const tempSubMesh = new SubMesh(0, 0, 0, 0, 0, mesh, undefined, false, false);\n if (tempSubMesh.materialDefines) {\n tempSubMesh.materialDefines._renderId = -1;\n }\n if (!this.isReadyForSubMesh(mesh, tempSubMesh, localOptions.useInstances)) {\n if (tempSubMesh.effect && tempSubMesh.effect.getCompilationError() && tempSubMesh.effect.allFallbacksProcessed()) {\n lastError = tempSubMesh.effect.getCompilationError();\n }\n else {\n allDone = false;\n setTimeout(checkReady, 16);\n }\n }\n }\n if (allDone) {\n this.allowShaderHotSwapping = currentHotSwapingState;\n if (lastError) {\n if (onError) {\n onError(lastError);\n }\n }\n if (onCompiled) {\n onCompiled(this);\n }\n }\n }\n else {\n if (this.isReady()) {\n this.allowShaderHotSwapping = currentHotSwapingState;\n if (onCompiled) {\n onCompiled(this);\n }\n }\n else {\n setTimeout(checkReady, 16);\n }\n }\n if (localOptions.clipPlane) {\n scene.clipPlane = clipPlaneState;\n }\n };\n checkReady();\n }\n /**\n * Force shader compilation\n * @param mesh defines the mesh that will use this material\n * @param options defines additional options for compiling the shaders\n * @returns a promise that resolves when the compilation completes\n */\n forceCompilationAsync(mesh, options) {\n return new Promise((resolve, reject) => {\n this.forceCompilation(mesh, () => {\n resolve();\n }, options, (reason) => {\n reject(reason);\n });\n });\n }\n /**\n * Marks a define in the material to indicate that it needs to be re-computed\n * @param flag defines a flag used to determine which parts of the material have to be marked as dirty\n */\n markAsDirty(flag) {\n if (this.getScene().blockMaterialDirtyMechanism || this._blockDirtyMechanism) {\n return;\n }\n Material._DirtyCallbackArray.length = 0;\n if (flag & Material.TextureDirtyFlag) {\n Material._DirtyCallbackArray.push(Material._TextureDirtyCallBack);\n }\n if (flag & Material.LightDirtyFlag) {\n Material._DirtyCallbackArray.push(Material._LightsDirtyCallBack);\n }\n if (flag & Material.FresnelDirtyFlag) {\n Material._DirtyCallbackArray.push(Material._FresnelDirtyCallBack);\n }\n if (flag & Material.AttributesDirtyFlag) {\n Material._DirtyCallbackArray.push(Material._AttributeDirtyCallBack);\n }\n if (flag & Material.MiscDirtyFlag) {\n Material._DirtyCallbackArray.push(Material._MiscDirtyCallBack);\n }\n if (flag & Material.PrePassDirtyFlag) {\n Material._DirtyCallbackArray.push(Material._PrePassDirtyCallBack);\n }\n if (Material._DirtyCallbackArray.length) {\n this._markAllSubMeshesAsDirty(Material._RunDirtyCallBacks);\n }\n this.getScene().resetCachedMaterial();\n }\n /**\n * Resets the draw wrappers cache for all submeshes that are using this material\n */\n resetDrawCache() {\n const meshes = this.getScene().meshes;\n for (const mesh of meshes) {\n if (!mesh.subMeshes) {\n continue;\n }\n for (const subMesh of mesh.subMeshes) {\n if (subMesh.getMaterial() !== this) {\n continue;\n }\n subMesh.resetDrawCache();\n }\n }\n }\n /**\n * Marks all submeshes of a material to indicate that their material defines need to be re-calculated\n * @param func defines a function which checks material defines against the submeshes\n */\n _markAllSubMeshesAsDirty(func) {\n if (this.getScene().blockMaterialDirtyMechanism || this._blockDirtyMechanism) {\n return;\n }\n const meshes = this.getScene().meshes;\n for (const mesh of meshes) {\n if (!mesh.subMeshes) {\n continue;\n }\n for (const subMesh of mesh.subMeshes) {\n // We want to skip the submeshes which are not using this material or which have not yet rendered at least once\n if (subMesh.getMaterial(false) !== this) {\n continue;\n }\n for (const drawWrapper of subMesh._drawWrappers) {\n if (!drawWrapper || !drawWrapper.defines || !drawWrapper.defines.markAllAsDirty) {\n continue;\n }\n if (this._materialContext === drawWrapper.materialContext) {\n func(drawWrapper.defines);\n }\n }\n }\n }\n }\n /**\n * Indicates that the scene should check if the rendering now needs a prepass\n */\n _markScenePrePassDirty() {\n if (this.getScene().blockMaterialDirtyMechanism || this._blockDirtyMechanism) {\n return;\n }\n const prePassRenderer = this.getScene().enablePrePassRenderer();\n if (prePassRenderer) {\n prePassRenderer.markAsDirty();\n }\n }\n /**\n * Indicates that we need to re-calculated for all submeshes\n */\n _markAllSubMeshesAsAllDirty() {\n this._markAllSubMeshesAsDirty(Material._AllDirtyCallBack);\n }\n /**\n * Indicates that image processing needs to be re-calculated for all submeshes\n */\n _markAllSubMeshesAsImageProcessingDirty() {\n this._markAllSubMeshesAsDirty(Material._ImageProcessingDirtyCallBack);\n }\n /**\n * Indicates that textures need to be re-calculated for all submeshes\n */\n _markAllSubMeshesAsTexturesDirty() {\n this._markAllSubMeshesAsDirty(Material._TextureDirtyCallBack);\n }\n /**\n * Indicates that fresnel needs to be re-calculated for all submeshes\n */\n _markAllSubMeshesAsFresnelDirty() {\n this._markAllSubMeshesAsDirty(Material._FresnelDirtyCallBack);\n }\n /**\n * Indicates that fresnel and misc need to be re-calculated for all submeshes\n */\n _markAllSubMeshesAsFresnelAndMiscDirty() {\n this._markAllSubMeshesAsDirty(Material._FresnelAndMiscDirtyCallBack);\n }\n /**\n * Indicates that lights need to be re-calculated for all submeshes\n */\n _markAllSubMeshesAsLightsDirty() {\n this._markAllSubMeshesAsDirty(Material._LightsDirtyCallBack);\n }\n /**\n * Indicates that attributes need to be re-calculated for all submeshes\n */\n _markAllSubMeshesAsAttributesDirty() {\n this._markAllSubMeshesAsDirty(Material._AttributeDirtyCallBack);\n }\n /**\n * Indicates that misc needs to be re-calculated for all submeshes\n */\n _markAllSubMeshesAsMiscDirty() {\n this._markAllSubMeshesAsDirty(Material._MiscDirtyCallBack);\n }\n /**\n * Indicates that prepass needs to be re-calculated for all submeshes\n */\n _markAllSubMeshesAsPrePassDirty() {\n this._markAllSubMeshesAsDirty(Material._MiscDirtyCallBack);\n }\n /**\n * Indicates that textures and misc need to be re-calculated for all submeshes\n */\n _markAllSubMeshesAsTexturesAndMiscDirty() {\n this._markAllSubMeshesAsDirty(Material._TextureAndMiscDirtyCallBack);\n }\n _checkScenePerformancePriority() {\n if (this._scene.performancePriority !== ScenePerformancePriority.BackwardCompatible) {\n this.checkReadyOnlyOnce = true;\n // re-set the flag when the perf priority changes\n const observer = this._scene.onScenePerformancePriorityChangedObservable.addOnce(() => {\n this.checkReadyOnlyOnce = false;\n });\n // if this material is disposed before the scene is disposed, cleanup the observer\n this.onDisposeObservable.add(() => {\n this._scene.onScenePerformancePriorityChangedObservable.remove(observer);\n });\n }\n }\n /**\n * Sets the required values to the prepass renderer.\n * @param prePassRenderer defines the prepass renderer to setup.\n * @returns true if the pre pass is needed.\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n setPrePassRenderer(prePassRenderer) {\n // Do Nothing by default\n return false;\n }\n /**\n * Disposes the material\n * @param forceDisposeEffect specifies if effects should be forcefully disposed\n * @param forceDisposeTextures specifies if textures should be forcefully disposed\n * @param notBoundToMesh specifies if the material that is being disposed is known to be not bound to any mesh\n */\n dispose(forceDisposeEffect, forceDisposeTextures, notBoundToMesh) {\n const scene = this.getScene();\n // Animations\n scene.stopAnimation(this);\n scene.freeProcessedMaterials();\n // Remove from scene\n scene.removeMaterial(this);\n this._eventInfo.forceDisposeTextures = forceDisposeTextures;\n this._callbackPluginEventGeneric(MaterialPluginEvent.Disposed, this._eventInfo);\n if (this._parentContainer) {\n const index = this._parentContainer.materials.indexOf(this);\n if (index > -1) {\n this._parentContainer.materials.splice(index, 1);\n }\n this._parentContainer = null;\n }\n if (notBoundToMesh !== true) {\n // Remove from meshes\n if (this.meshMap) {\n for (const meshId in this.meshMap) {\n const mesh = this.meshMap[meshId];\n if (mesh) {\n mesh.material = null; // will set the entry in the map to undefined\n this.releaseVertexArrayObject(mesh, forceDisposeEffect);\n }\n }\n }\n else {\n const meshes = scene.meshes;\n for (const mesh of meshes) {\n if (mesh.material === this && !mesh.sourceMesh) {\n mesh.material = null;\n this.releaseVertexArrayObject(mesh, forceDisposeEffect);\n }\n }\n }\n }\n this._uniformBuffer.dispose();\n // Shader are kept in cache for further use but we can get rid of this by using forceDisposeEffect\n if (forceDisposeEffect && this._drawWrapper.effect) {\n if (!this._storeEffectOnSubMeshes) {\n this._drawWrapper.effect.dispose();\n }\n this._drawWrapper.effect = null;\n }\n this.metadata = null;\n // Callback\n this.onDisposeObservable.notifyObservers(this);\n this.onDisposeObservable.clear();\n if (this._onBindObservable) {\n this._onBindObservable.clear();\n }\n if (this._onUnBindObservable) {\n this._onUnBindObservable.clear();\n }\n if (this._onEffectCreatedObservable) {\n this._onEffectCreatedObservable.clear();\n }\n if (this._eventInfo) {\n this._eventInfo = {};\n }\n }\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n releaseVertexArrayObject(mesh, forceDisposeEffect) {\n if (mesh.geometry) {\n const geometry = mesh.geometry;\n if (this._storeEffectOnSubMeshes) {\n for (const subMesh of mesh.subMeshes) {\n geometry._releaseVertexArrayObject(subMesh.effect);\n if (forceDisposeEffect && subMesh.effect) {\n subMesh.effect.dispose();\n }\n }\n }\n else {\n geometry._releaseVertexArrayObject(this._drawWrapper.effect);\n }\n }\n }\n /**\n * Serializes this material\n * @returns the serialized material object\n */\n serialize() {\n const serializationObject = SerializationHelper.Serialize(this);\n serializationObject.stencil = this.stencil.serialize();\n serializationObject.uniqueId = this.uniqueId;\n this._serializePlugins(serializationObject);\n return serializationObject;\n }\n _serializePlugins(serializationObject) {\n serializationObject.plugins = {};\n if (this.pluginManager) {\n for (const plugin of this.pluginManager._plugins) {\n serializationObject.plugins[plugin.getClassName()] = plugin.serialize();\n }\n }\n }\n /**\n * Creates a material from parsed material data\n * @param parsedMaterial defines parsed material data\n * @param scene defines the hosting scene\n * @param rootUrl defines the root URL to use to load textures\n * @returns a new material\n */\n static Parse(parsedMaterial, scene, rootUrl) {\n if (!parsedMaterial.customType) {\n parsedMaterial.customType = \"BABYLON.StandardMaterial\";\n }\n else if (parsedMaterial.customType === \"BABYLON.PBRMaterial\" && parsedMaterial.overloadedAlbedo) {\n parsedMaterial.customType = \"BABYLON.LegacyPBRMaterial\";\n if (!BABYLON.LegacyPBRMaterial) {\n Logger.Error(\"Your scene is trying to load a legacy version of the PBRMaterial, please, include it from the materials library.\");\n return null;\n }\n }\n const materialType = Tools.Instantiate(parsedMaterial.customType);\n const material = materialType.Parse(parsedMaterial, scene, rootUrl);\n material._loadedUniqueId = parsedMaterial.uniqueId;\n return material;\n }\n static _parsePlugins(serializationObject, material, scene, rootUrl) {\n var _a;\n if (!serializationObject.plugins) {\n return;\n }\n for (const pluginClassName in serializationObject.plugins) {\n const pluginData = serializationObject.plugins[pluginClassName];\n let plugin = (_a = material.pluginManager) === null || _a === void 0 ? void 0 : _a.getPlugin(pluginData.name);\n if (!plugin) {\n const pluginClassType = Tools.Instantiate(\"BABYLON.\" + pluginClassName);\n if (pluginClassType) {\n plugin = new pluginClassType(material);\n }\n }\n plugin === null || plugin === void 0 ? void 0 : plugin.parse(pluginData, scene, rootUrl);\n }\n }\n }\n /**\n * Returns the triangle fill mode\n */\n Material.TriangleFillMode = 0;\n /**\n * Returns the wireframe mode\n */\n Material.WireFrameFillMode = 1;\n /**\n * Returns the point fill mode\n */\n Material.PointFillMode = 2;\n /**\n * Returns the point list draw mode\n */\n Material.PointListDrawMode = 3;\n /**\n * Returns the line list draw mode\n */\n Material.LineListDrawMode = 4;\n /**\n * Returns the line loop draw mode\n */\n Material.LineLoopDrawMode = 5;\n /**\n * Returns the line strip draw mode\n */\n Material.LineStripDrawMode = 6;\n /**\n * Returns the triangle strip draw mode\n */\n Material.TriangleStripDrawMode = 7;\n /**\n * Returns the triangle fan draw mode\n */\n Material.TriangleFanDrawMode = 8;\n /**\n * Stores the clock-wise side orientation\n */\n Material.ClockWiseSideOrientation = 0;\n /**\n * Stores the counter clock-wise side orientation\n */\n Material.CounterClockWiseSideOrientation = 1;\n /**\n * The dirty texture flag value\n */\n Material.TextureDirtyFlag = 1;\n /**\n * The dirty light flag value\n */\n Material.LightDirtyFlag = 2;\n /**\n * The dirty fresnel flag value\n */\n Material.FresnelDirtyFlag = 4;\n /**\n * The dirty attribute flag value\n */\n Material.AttributesDirtyFlag = 8;\n /**\n * The dirty misc flag value\n */\n Material.MiscDirtyFlag = 16;\n /**\n * The dirty prepass flag value\n */\n Material.PrePassDirtyFlag = 32;\n /**\n * The all dirty flag value\n */\n Material.AllDirtyFlag = 63;\n /**\n * MaterialTransparencyMode: No transparency mode, Alpha channel is not use.\n */\n Material.MATERIAL_OPAQUE = 0;\n /**\n * MaterialTransparencyMode: Alpha Test mode, pixel are discarded below a certain threshold defined by the alpha cutoff value.\n */\n Material.MATERIAL_ALPHATEST = 1;\n /**\n * MaterialTransparencyMode: Pixels are blended (according to the alpha mode) with the already drawn pixels in the current frame buffer.\n */\n Material.MATERIAL_ALPHABLEND = 2;\n /**\n * MaterialTransparencyMode: Pixels are blended (according to the alpha mode) with the already drawn pixels in the current frame buffer.\n * They are also discarded below the alpha cutoff threshold to improve performances.\n */\n Material.MATERIAL_ALPHATESTANDBLEND = 3;\n /**\n * The Whiteout method is used to blend normals.\n * Details of the algorithm can be found here: https://blog.selfshadow.com/publications/blending-in-detail/\n */\n Material.MATERIAL_NORMALBLENDMETHOD_WHITEOUT = 0;\n /**\n * The Reoriented Normal Mapping method is used to blend normals.\n * Details of the algorithm can be found here: https://blog.selfshadow.com/publications/blending-in-detail/\n */\n Material.MATERIAL_NORMALBLENDMETHOD_RNM = 1;\n /**\n * Event observable which raises global events common to all materials (like MaterialPluginEvent.Created)\n */\n Material.OnEventObservable = new Observable$1();\n Material._AllDirtyCallBack = (defines) => defines.markAllAsDirty();\n Material._ImageProcessingDirtyCallBack = (defines) => defines.markAsImageProcessingDirty();\n Material._TextureDirtyCallBack = (defines) => defines.markAsTexturesDirty();\n Material._FresnelDirtyCallBack = (defines) => defines.markAsFresnelDirty();\n Material._MiscDirtyCallBack = (defines) => defines.markAsMiscDirty();\n Material._PrePassDirtyCallBack = (defines) => defines.markAsPrePassDirty();\n Material._LightsDirtyCallBack = (defines) => defines.markAsLightDirty();\n Material._AttributeDirtyCallBack = (defines) => defines.markAsAttributesDirty();\n Material._FresnelAndMiscDirtyCallBack = (defines) => {\n Material._FresnelDirtyCallBack(defines);\n Material._MiscDirtyCallBack(defines);\n };\n Material._TextureAndMiscDirtyCallBack = (defines) => {\n Material._TextureDirtyCallBack(defines);\n Material._MiscDirtyCallBack(defines);\n };\n Material._DirtyCallbackArray = [];\n Material._RunDirtyCallBacks = (defines) => {\n for (const cb of Material._DirtyCallbackArray) {\n cb(defines);\n }\n };\n __decorate$1([\n serialize()\n ], Material.prototype, \"id\", void 0);\n __decorate$1([\n serialize()\n ], Material.prototype, \"uniqueId\", void 0);\n __decorate$1([\n serialize()\n ], Material.prototype, \"name\", void 0);\n __decorate$1([\n serialize()\n ], Material.prototype, \"metadata\", void 0);\n __decorate$1([\n serialize()\n ], Material.prototype, \"checkReadyOnEveryCall\", void 0);\n __decorate$1([\n serialize()\n ], Material.prototype, \"checkReadyOnlyOnce\", void 0);\n __decorate$1([\n serialize()\n ], Material.prototype, \"state\", void 0);\n __decorate$1([\n serialize(\"alpha\")\n ], Material.prototype, \"_alpha\", void 0);\n __decorate$1([\n serialize(\"backFaceCulling\")\n ], Material.prototype, \"_backFaceCulling\", void 0);\n __decorate$1([\n serialize(\"cullBackFaces\")\n ], Material.prototype, \"_cullBackFaces\", void 0);\n __decorate$1([\n serialize()\n ], Material.prototype, \"sideOrientation\", void 0);\n __decorate$1([\n serialize(\"alphaMode\")\n ], Material.prototype, \"_alphaMode\", void 0);\n __decorate$1([\n serialize()\n ], Material.prototype, \"_needDepthPrePass\", void 0);\n __decorate$1([\n serialize()\n ], Material.prototype, \"disableDepthWrite\", void 0);\n __decorate$1([\n serialize()\n ], Material.prototype, \"disableColorWrite\", void 0);\n __decorate$1([\n serialize()\n ], Material.prototype, \"forceDepthWrite\", void 0);\n __decorate$1([\n serialize()\n ], Material.prototype, \"depthFunction\", void 0);\n __decorate$1([\n serialize()\n ], Material.prototype, \"separateCullingPass\", void 0);\n __decorate$1([\n serialize(\"fogEnabled\")\n ], Material.prototype, \"_fogEnabled\", void 0);\n __decorate$1([\n serialize()\n ], Material.prototype, \"pointSize\", void 0);\n __decorate$1([\n serialize()\n ], Material.prototype, \"zOffset\", void 0);\n __decorate$1([\n serialize()\n ], Material.prototype, \"zOffsetUnits\", void 0);\n __decorate$1([\n serialize()\n ], Material.prototype, \"pointsCloud\", null);\n __decorate$1([\n serialize()\n ], Material.prototype, \"fillMode\", null);\n __decorate$1([\n serialize()\n ], Material.prototype, \"transparencyMode\", null);\n\n /**\n * A multi-material is used to apply different materials to different parts of the same object without the need of\n * separate meshes. This can be use to improve performances.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/multiMaterials\n */\n class MultiMaterial extends Material {\n /**\n * Gets or Sets the list of Materials used within the multi material.\n * They need to be ordered according to the submeshes order in the associated mesh\n */\n get subMaterials() {\n return this._subMaterials;\n }\n set subMaterials(value) {\n this._subMaterials = value;\n this._hookArray(value);\n }\n /**\n * Function used to align with Node.getChildren()\n * @returns the list of Materials used within the multi material\n */\n getChildren() {\n return this.subMaterials;\n }\n /**\n * Instantiates a new Multi Material\n * A multi-material is used to apply different materials to different parts of the same object without the need of\n * separate meshes. This can be use to improve performances.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/multiMaterials\n * @param name Define the name in the scene\n * @param scene Define the scene the material belongs to\n */\n constructor(name, scene) {\n super(name, scene, true);\n /** @internal */\n this._waitingSubMaterialsUniqueIds = [];\n this.getScene().addMultiMaterial(this);\n this.subMaterials = new Array();\n this._storeEffectOnSubMeshes = true; // multimaterial is considered like a push material\n }\n _hookArray(array) {\n const oldPush = array.push;\n array.push = (...items) => {\n const result = oldPush.apply(array, items);\n this._markAllSubMeshesAsTexturesDirty();\n return result;\n };\n const oldSplice = array.splice;\n array.splice = (index, deleteCount) => {\n const deleted = oldSplice.apply(array, [index, deleteCount]);\n this._markAllSubMeshesAsTexturesDirty();\n return deleted;\n };\n }\n /**\n * Get one of the submaterial by its index in the submaterials array\n * @param index The index to look the sub material at\n * @returns The Material if the index has been defined\n */\n getSubMaterial(index) {\n if (index < 0 || index >= this.subMaterials.length) {\n return this.getScene().defaultMaterial;\n }\n return this.subMaterials[index];\n }\n /**\n * Get the list of active textures for the whole sub materials list.\n * @returns All the textures that will be used during the rendering\n */\n getActiveTextures() {\n return super.getActiveTextures().concat(...this.subMaterials.map((subMaterial) => {\n if (subMaterial) {\n return subMaterial.getActiveTextures();\n }\n else {\n return [];\n }\n }));\n }\n /**\n * Specifies if any sub-materials of this multi-material use a given texture.\n * @param texture Defines the texture to check against this multi-material's sub-materials.\n * @returns A boolean specifying if any sub-material of this multi-material uses the texture.\n */\n hasTexture(texture) {\n var _a;\n if (super.hasTexture(texture)) {\n return true;\n }\n for (let i = 0; i < this.subMaterials.length; i++) {\n if ((_a = this.subMaterials[i]) === null || _a === void 0 ? void 0 : _a.hasTexture(texture)) {\n return true;\n }\n }\n return false;\n }\n /**\n * Gets the current class name of the material e.g. \"MultiMaterial\"\n * Mainly use in serialization.\n * @returns the class name\n */\n getClassName() {\n return \"MultiMaterial\";\n }\n /**\n * Checks if the material is ready to render the requested sub mesh\n * @param mesh Define the mesh the submesh belongs to\n * @param subMesh Define the sub mesh to look readiness for\n * @param useInstances Define whether or not the material is used with instances\n * @returns true if ready, otherwise false\n */\n isReadyForSubMesh(mesh, subMesh, useInstances) {\n for (let index = 0; index < this.subMaterials.length; index++) {\n const subMaterial = this.subMaterials[index];\n if (subMaterial) {\n if (subMaterial._storeEffectOnSubMeshes) {\n if (!subMaterial.isReadyForSubMesh(mesh, subMesh, useInstances)) {\n return false;\n }\n continue;\n }\n if (!subMaterial.isReady(mesh)) {\n return false;\n }\n }\n }\n return true;\n }\n /**\n * Clones the current material and its related sub materials\n * @param name Define the name of the newly cloned material\n * @param cloneChildren Define if submaterial will be cloned or shared with the parent instance\n * @returns the cloned material\n */\n clone(name, cloneChildren) {\n const newMultiMaterial = new MultiMaterial(name, this.getScene());\n for (let index = 0; index < this.subMaterials.length; index++) {\n let subMaterial = null;\n const current = this.subMaterials[index];\n if (cloneChildren && current) {\n subMaterial = current.clone(name + \"-\" + current.name);\n }\n else {\n subMaterial = this.subMaterials[index];\n }\n newMultiMaterial.subMaterials.push(subMaterial);\n }\n return newMultiMaterial;\n }\n /**\n * Serializes the materials into a JSON representation.\n * @returns the JSON representation\n */\n serialize() {\n const serializationObject = {};\n serializationObject.name = this.name;\n serializationObject.id = this.id;\n serializationObject.uniqueId = this.uniqueId;\n if (Tags) {\n serializationObject.tags = Tags.GetTags(this);\n }\n serializationObject.materialsUniqueIds = [];\n serializationObject.materials = [];\n for (let matIndex = 0; matIndex < this.subMaterials.length; matIndex++) {\n const subMat = this.subMaterials[matIndex];\n if (subMat) {\n serializationObject.materialsUniqueIds.push(subMat.uniqueId);\n serializationObject.materials.push(subMat.id);\n }\n else {\n serializationObject.materialsUniqueIds.push(null);\n serializationObject.materials.push(null);\n }\n }\n return serializationObject;\n }\n /**\n * Dispose the material and release its associated resources\n * @param forceDisposeEffect Define if we want to force disposing the associated effect (if false the shader is not released and could be reuse later on)\n * @param forceDisposeTextures Define if we want to force disposing the associated textures (if false, they will not be disposed and can still be use elsewhere in the app)\n * @param forceDisposeChildren Define if we want to force disposing the associated submaterials (if false, they will not be disposed and can still be use elsewhere in the app)\n */\n dispose(forceDisposeEffect, forceDisposeTextures, forceDisposeChildren) {\n const scene = this.getScene();\n if (!scene) {\n return;\n }\n if (forceDisposeChildren) {\n for (let index = 0; index < this.subMaterials.length; index++) {\n const subMaterial = this.subMaterials[index];\n if (subMaterial) {\n subMaterial.dispose(forceDisposeEffect, forceDisposeTextures);\n }\n }\n }\n const index = scene.multiMaterials.indexOf(this);\n if (index >= 0) {\n scene.multiMaterials.splice(index, 1);\n }\n super.dispose(forceDisposeEffect, forceDisposeTextures);\n }\n /**\n * Creates a MultiMaterial from parsed MultiMaterial data.\n * @param parsedMultiMaterial defines parsed MultiMaterial data.\n * @param scene defines the hosting scene\n * @returns a new MultiMaterial\n */\n static ParseMultiMaterial(parsedMultiMaterial, scene) {\n const multiMaterial = new MultiMaterial(parsedMultiMaterial.name, scene);\n multiMaterial.id = parsedMultiMaterial.id;\n multiMaterial._loadedUniqueId = parsedMultiMaterial.uniqueId;\n if (Tags) {\n Tags.AddTagsTo(multiMaterial, parsedMultiMaterial.tags);\n }\n if (parsedMultiMaterial.materialsUniqueIds) {\n multiMaterial._waitingSubMaterialsUniqueIds = parsedMultiMaterial.materialsUniqueIds;\n }\n else {\n parsedMultiMaterial.materials.forEach((subMatId) => multiMaterial.subMaterials.push(scene.getLastMaterialById(subMatId)));\n }\n return multiMaterial;\n }\n }\n RegisterClass(\"BABYLON.MultiMaterial\", MultiMaterial);\n\n /**\n * Class used to represent a specific level of detail of a mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/LOD\n */\n class MeshLODLevel {\n /**\n * Creates a new LOD level\n * @param distanceOrScreenCoverage defines either the distance or the screen coverage where this level should start being displayed\n * @param mesh defines the mesh to use to render this level\n */\n constructor(\n /** Either distance from the center of the object to show this level or the screen coverage if `useLODScreenCoverage` is set to `true` on the mesh*/\n distanceOrScreenCoverage, \n /** Defines the mesh to use to render this level */\n mesh) {\n this.distanceOrScreenCoverage = distanceOrScreenCoverage;\n this.mesh = mesh;\n }\n }\n\n /**\n * @internal\n **/\n class _CreationDataStorage {\n }\n /**\n * @internal\n **/\n class _InstanceDataStorage {\n constructor() {\n this.visibleInstances = {};\n this.batchCache = new _InstancesBatch();\n this.batchCacheReplacementModeInFrozenMode = new _InstancesBatch();\n this.instancesBufferSize = 32 * 16 * 4; // let's start with a maximum of 32 instances\n }\n }\n /**\n * @internal\n **/\n class _InstancesBatch {\n constructor() {\n this.mustReturn = false;\n this.visibleInstances = new Array();\n this.renderSelf = new Array();\n this.hardwareInstancedRendering = new Array();\n }\n }\n /**\n * @internal\n **/\n class _ThinInstanceDataStorage {\n constructor() {\n this.instancesCount = 0;\n this.matrixBuffer = null;\n this.previousMatrixBuffer = null;\n this.matrixBufferSize = 32 * 16; // let's start with a maximum of 32 thin instances\n this.matrixData = null;\n this.boundingVectors = [];\n this.worldMatrices = null;\n }\n }\n /**\n * @internal\n **/\n class _InternalMeshDataInfo {\n constructor() {\n this._areNormalsFrozen = false; // Will be used by ribbons mainly\n // Will be used to save a source mesh reference, If any\n this._source = null;\n // Will be used to for fast cloned mesh lookup\n this.meshMap = null;\n this._preActivateId = -1;\n // eslint-disable-next-line @typescript-eslint/naming-convention\n this._LODLevels = new Array();\n /** Alternative definition of LOD level, using screen coverage instead of distance */\n this._useLODScreenCoverage = false;\n this._effectiveMaterial = null;\n this._forcedInstanceCount = 0;\n this._overrideRenderingFillMode = null;\n }\n }\n /**\n * Class used to represent renderable models\n */\n class Mesh extends AbstractMesh {\n /**\n * Gets the default side orientation.\n * @param orientation the orientation to value to attempt to get\n * @returns the default orientation\n * @internal\n */\n static _GetDefaultSideOrientation(orientation) {\n return orientation || Mesh.FRONTSIDE; // works as Mesh.FRONTSIDE is 0\n }\n /**\n * Determines if the LOD levels are intended to be calculated using screen coverage (surface area ratio) instead of distance.\n */\n get useLODScreenCoverage() {\n return this._internalMeshDataInfo._useLODScreenCoverage;\n }\n set useLODScreenCoverage(value) {\n this._internalMeshDataInfo._useLODScreenCoverage = value;\n this._sortLODLevels();\n }\n get computeBonesUsingShaders() {\n return this._internalAbstractMeshDataInfo._computeBonesUsingShaders;\n }\n set computeBonesUsingShaders(value) {\n if (this._internalAbstractMeshDataInfo._computeBonesUsingShaders === value) {\n return;\n }\n if (value && this._internalMeshDataInfo._sourcePositions) {\n // switch from software to GPU computation: we need to reset the vertex and normal buffers that have been updated by the software process\n this.setVerticesData(VertexBuffer.PositionKind, this._internalMeshDataInfo._sourcePositions, true);\n if (this._internalMeshDataInfo._sourceNormals) {\n this.setVerticesData(VertexBuffer.NormalKind, this._internalMeshDataInfo._sourceNormals, true);\n }\n this._internalMeshDataInfo._sourcePositions = null;\n this._internalMeshDataInfo._sourceNormals = null;\n }\n this._internalAbstractMeshDataInfo._computeBonesUsingShaders = value;\n this._markSubMeshesAsAttributesDirty();\n }\n /**\n * An event triggered before rendering the mesh\n */\n get onBeforeRenderObservable() {\n if (!this._internalMeshDataInfo._onBeforeRenderObservable) {\n this._internalMeshDataInfo._onBeforeRenderObservable = new Observable$1();\n }\n return this._internalMeshDataInfo._onBeforeRenderObservable;\n }\n /**\n * An event triggered before binding the mesh\n */\n get onBeforeBindObservable() {\n if (!this._internalMeshDataInfo._onBeforeBindObservable) {\n this._internalMeshDataInfo._onBeforeBindObservable = new Observable$1();\n }\n return this._internalMeshDataInfo._onBeforeBindObservable;\n }\n /**\n * An event triggered after rendering the mesh\n */\n get onAfterRenderObservable() {\n if (!this._internalMeshDataInfo._onAfterRenderObservable) {\n this._internalMeshDataInfo._onAfterRenderObservable = new Observable$1();\n }\n return this._internalMeshDataInfo._onAfterRenderObservable;\n }\n /**\n * An event triggeredbetween rendering pass when using separateCullingPass = true\n */\n get onBetweenPassObservable() {\n if (!this._internalMeshDataInfo._onBetweenPassObservable) {\n this._internalMeshDataInfo._onBetweenPassObservable = new Observable$1();\n }\n return this._internalMeshDataInfo._onBetweenPassObservable;\n }\n /**\n * An event triggered before drawing the mesh\n */\n get onBeforeDrawObservable() {\n if (!this._internalMeshDataInfo._onBeforeDrawObservable) {\n this._internalMeshDataInfo._onBeforeDrawObservable = new Observable$1();\n }\n return this._internalMeshDataInfo._onBeforeDrawObservable;\n }\n /**\n * Sets a callback to call before drawing the mesh. It is recommended to use onBeforeDrawObservable instead\n */\n set onBeforeDraw(callback) {\n if (this._onBeforeDrawObserver) {\n this.onBeforeDrawObservable.remove(this._onBeforeDrawObserver);\n }\n this._onBeforeDrawObserver = this.onBeforeDrawObservable.add(callback);\n }\n get hasInstances() {\n return this.instances.length > 0;\n }\n get hasThinInstances() {\n var _a;\n return ((_a = this._thinInstanceDataStorage.instancesCount) !== null && _a !== void 0 ? _a : 0) > 0;\n }\n /**\n * Gets or sets the forced number of instances to display.\n * If 0 (default value), the number of instances is not forced and depends on the draw type\n * (regular / instance / thin instances mesh)\n */\n get forcedInstanceCount() {\n return this._internalMeshDataInfo._forcedInstanceCount;\n }\n set forcedInstanceCount(count) {\n this._internalMeshDataInfo._forcedInstanceCount = count;\n }\n /**\n * Use this property to override the Material's fillMode value\n */\n get overrideRenderingFillMode() {\n return this._internalMeshDataInfo._overrideRenderingFillMode;\n }\n set overrideRenderingFillMode(fillMode) {\n this._internalMeshDataInfo._overrideRenderingFillMode = fillMode;\n }\n /**\n * Gets the source mesh (the one used to clone this one from)\n */\n get source() {\n return this._internalMeshDataInfo._source;\n }\n /**\n * Gets the list of clones of this mesh\n * The scene must have been constructed with useClonedMeshMap=true for this to work!\n * Note that useClonedMeshMap=true is the default setting\n */\n get cloneMeshMap() {\n return this._internalMeshDataInfo.meshMap;\n }\n /**\n * Gets or sets a boolean indicating that this mesh does not use index buffer\n */\n get isUnIndexed() {\n return this._unIndexed;\n }\n set isUnIndexed(value) {\n if (this._unIndexed !== value) {\n this._unIndexed = value;\n this._markSubMeshesAsAttributesDirty();\n }\n }\n /** Gets the array buffer used to store the instanced buffer used for instances' world matrices */\n get worldMatrixInstancedBuffer() {\n return this._instanceDataStorage.instancesData;\n }\n /** Gets the array buffer used to store the instanced buffer used for instances' previous world matrices */\n get previousWorldMatrixInstancedBuffer() {\n return this._instanceDataStorage.instancesPreviousData;\n }\n /** Gets or sets a boolean indicating that the update of the instance buffer of the world matrices is manual */\n get manualUpdateOfWorldMatrixInstancedBuffer() {\n return this._instanceDataStorage.manualUpdate;\n }\n set manualUpdateOfWorldMatrixInstancedBuffer(value) {\n this._instanceDataStorage.manualUpdate = value;\n }\n /** Gets or sets a boolean indicating that the update of the instance buffer of the world matrices is manual */\n get manualUpdateOfPreviousWorldMatrixInstancedBuffer() {\n return this._instanceDataStorage.previousManualUpdate;\n }\n set manualUpdateOfPreviousWorldMatrixInstancedBuffer(value) {\n this._instanceDataStorage.previousManualUpdate = value;\n }\n /** Gets or sets a boolean indicating that the update of the instance buffer of the world matrices must be performed in all cases (and notably even in frozen mode) */\n get forceWorldMatrixInstancedBufferUpdate() {\n return this._instanceDataStorage.forceMatrixUpdates;\n }\n set forceWorldMatrixInstancedBufferUpdate(value) {\n this._instanceDataStorage.forceMatrixUpdates = value;\n }\n /**\n * @constructor\n * @param name The value used by scene.getMeshByName() to do a lookup.\n * @param scene The scene to add this mesh to.\n * @param parent The parent of this mesh, if it has one\n * @param source An optional Mesh from which geometry is shared, cloned.\n * @param doNotCloneChildren When cloning, skip cloning child meshes of source, default False.\n * When false, achieved by calling a clone(), also passing False.\n * This will make creation of children, recursive.\n * @param clonePhysicsImpostor When cloning, include cloning mesh physics impostor, default True.\n */\n constructor(name, scene = null, parent = null, source = null, doNotCloneChildren, clonePhysicsImpostor = true) {\n super(name, scene);\n // Internal data\n this._internalMeshDataInfo = new _InternalMeshDataInfo();\n // Members\n /**\n * Gets the delay loading state of the mesh (when delay loading is turned on)\n * @see https://doc.babylonjs.com/features/featuresDeepDive/importers/incrementalLoading\n */\n this.delayLoadState = 0;\n /**\n * Gets the list of instances created from this mesh\n * it is not supposed to be modified manually.\n * Note also that the order of the InstancedMesh wihin the array is not significant and might change.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/copies/instances\n */\n this.instances = new Array();\n // Private\n /** @internal */\n this._creationDataStorage = null;\n /** @internal */\n this._geometry = null;\n /** @internal */\n this._instanceDataStorage = new _InstanceDataStorage();\n /** @internal */\n this._thinInstanceDataStorage = new _ThinInstanceDataStorage();\n /** @internal */\n this._shouldGenerateFlatShading = false;\n // Use by builder only to know what orientation were the mesh build in.\n /** @internal */\n this._originalBuilderSideOrientation = Mesh.DEFAULTSIDE;\n /**\n * Use this property to change the original side orientation defined at construction time\n */\n this.overrideMaterialSideOrientation = null;\n /**\n * Gets or sets a boolean indicating whether to render ignoring the active camera's max z setting. (false by default)\n * You should not mix meshes that have this property set to true with meshes that have it set to false if they all write\n * to the depth buffer, because the z-values are not comparable in the two cases and you will get rendering artifacts if you do.\n * You can set the property to true for meshes that do not write to the depth buffer, or set the same value (either false or true) otherwise.\n * Note this will reduce performance when set to true.\n */\n this.ignoreCameraMaxZ = false;\n scene = this.getScene();\n this._onBeforeDraw = (isInstance, world, effectiveMaterial) => {\n if (isInstance && effectiveMaterial) {\n if (this._uniformBuffer) {\n this.transferToEffect(world);\n }\n else {\n effectiveMaterial.bindOnlyWorldMatrix(world);\n }\n }\n };\n if (source) {\n // Geometry\n if (source._geometry) {\n source._geometry.applyToMesh(this);\n }\n // Deep copy\n DeepCopier.DeepCopy(source, this, [\n \"name\",\n \"material\",\n \"skeleton\",\n \"instances\",\n \"parent\",\n \"uniqueId\",\n \"source\",\n \"metadata\",\n \"morphTargetManager\",\n \"hasInstances\",\n \"worldMatrixInstancedBuffer\",\n \"previousWorldMatrixInstancedBuffer\",\n \"hasLODLevels\",\n \"geometry\",\n \"isBlocked\",\n \"areNormalsFrozen\",\n \"facetNb\",\n \"isFacetDataEnabled\",\n \"lightSources\",\n \"useBones\",\n \"isAnInstance\",\n \"collider\",\n \"edgesRenderer\",\n \"forward\",\n \"up\",\n \"right\",\n \"absolutePosition\",\n \"absoluteScaling\",\n \"absoluteRotationQuaternion\",\n \"isWorldMatrixFrozen\",\n \"nonUniformScaling\",\n \"behaviors\",\n \"worldMatrixFromCache\",\n \"hasThinInstances\",\n \"cloneMeshMap\",\n \"hasBoundingInfo\",\n \"physicsBody\",\n \"physicsImpostor\",\n ], [\"_poseMatrix\"]);\n // Source mesh\n this._internalMeshDataInfo._source = source;\n if (scene.useClonedMeshMap) {\n if (!source._internalMeshDataInfo.meshMap) {\n source._internalMeshDataInfo.meshMap = {};\n }\n source._internalMeshDataInfo.meshMap[this.uniqueId] = this;\n }\n // Construction Params\n // Clone parameters allowing mesh to be updated in case of parametric shapes.\n this._originalBuilderSideOrientation = source._originalBuilderSideOrientation;\n this._creationDataStorage = source._creationDataStorage;\n // Animation ranges\n if (source._ranges) {\n const ranges = source._ranges;\n for (const name in ranges) {\n if (!Object.prototype.hasOwnProperty.call(ranges, name)) {\n continue;\n }\n if (!ranges[name]) {\n continue;\n }\n this.createAnimationRange(name, ranges[name].from, ranges[name].to);\n }\n }\n // Metadata\n if (source.metadata && source.metadata.clone) {\n this.metadata = source.metadata.clone();\n }\n else {\n this.metadata = source.metadata;\n }\n this._internalMetadata = source._internalMetadata;\n // Tags\n if (Tags && Tags.HasTags(source)) {\n Tags.AddTagsTo(this, Tags.GetTags(source, true));\n }\n // Enabled. We shouldn't need to check the source's ancestors, as this mesh\n // will have the same ones.\n this.setEnabled(source.isEnabled(false));\n // Parent\n this.parent = source.parent;\n // Pivot\n this.setPivotMatrix(source.getPivotMatrix());\n this.id = name + \".\" + source.id;\n // Material\n this.material = source.material;\n if (!doNotCloneChildren) {\n // Children\n const directDescendants = source.getDescendants(true);\n for (let index = 0; index < directDescendants.length; index++) {\n const child = directDescendants[index];\n if (child.clone) {\n child.clone(name + \".\" + child.name, this);\n }\n }\n }\n // Morphs\n if (source.morphTargetManager) {\n this.morphTargetManager = source.morphTargetManager;\n }\n // Physics clone\n if (scene.getPhysicsEngine) {\n const physicsEngine = scene.getPhysicsEngine();\n if (clonePhysicsImpostor && physicsEngine) {\n if (physicsEngine.getPluginVersion() === 1) {\n const impostor = physicsEngine.getImpostorForPhysicsObject(source);\n if (impostor) {\n this.physicsImpostor = impostor.clone(this);\n }\n }\n else if (physicsEngine.getPluginVersion() === 2) {\n if (source.physicsBody) {\n source.physicsBody.clone(this);\n }\n }\n }\n }\n // Particles\n for (let index = 0; index < scene.particleSystems.length; index++) {\n const system = scene.particleSystems[index];\n if (system.emitter === source) {\n system.clone(system.name, this);\n }\n }\n // Skeleton\n this.skeleton = source.skeleton;\n this.refreshBoundingInfo(true, true);\n this.computeWorldMatrix(true);\n }\n // Parent\n if (parent !== null) {\n this.parent = parent;\n }\n this._instanceDataStorage.hardwareInstancedRendering = this.getEngine().getCaps().instancedArrays;\n this._internalMeshDataInfo._onMeshReadyObserverAdded = (observer) => {\n // only notify once! then unregister the observer\n observer.unregisterOnNextCall = true;\n if (this.isReady(true)) {\n this.onMeshReadyObservable.notifyObservers(this);\n }\n else {\n if (!this._internalMeshDataInfo._checkReadinessObserver) {\n this._internalMeshDataInfo._checkReadinessObserver = this._scene.onBeforeRenderObservable.add(() => {\n // check for complete readiness\n if (this.isReady(true)) {\n this._scene.onBeforeRenderObservable.remove(this._internalMeshDataInfo._checkReadinessObserver);\n this._internalMeshDataInfo._checkReadinessObserver = null;\n this.onMeshReadyObservable.notifyObservers(this);\n }\n });\n }\n }\n };\n this.onMeshReadyObservable = new Observable$1(this._internalMeshDataInfo._onMeshReadyObserverAdded);\n if (source) {\n source.onClonedObservable.notifyObservers(this);\n }\n }\n instantiateHierarchy(newParent = null, options, onNewNodeCreated) {\n const instance = this.getTotalVertices() === 0 || (options && options.doNotInstantiate && (options.doNotInstantiate === true || options.doNotInstantiate(this)))\n ? this.clone(\"Clone of \" + (this.name || this.id), newParent || this.parent, true)\n : this.createInstance(\"instance of \" + (this.name || this.id));\n instance.parent = newParent || this.parent;\n instance.position = this.position.clone();\n instance.scaling = this.scaling.clone();\n if (this.rotationQuaternion) {\n instance.rotationQuaternion = this.rotationQuaternion.clone();\n }\n else {\n instance.rotation = this.rotation.clone();\n }\n if (onNewNodeCreated) {\n onNewNodeCreated(this, instance);\n }\n for (const child of this.getChildTransformNodes(true)) {\n // instancedMesh should have a different sourced mesh\n if (child.getClassName() === \"InstancedMesh\" && instance.getClassName() === \"Mesh\" && child.sourceMesh === this) {\n child.instantiateHierarchy(instance, {\n doNotInstantiate: (options && options.doNotInstantiate) || false,\n newSourcedMesh: instance,\n }, onNewNodeCreated);\n }\n else {\n child.instantiateHierarchy(instance, options, onNewNodeCreated);\n }\n }\n return instance;\n }\n /**\n * Gets the class name\n * @returns the string \"Mesh\".\n */\n getClassName() {\n return \"Mesh\";\n }\n /** @internal */\n get _isMesh() {\n return true;\n }\n /**\n * Returns a description of this mesh\n * @param fullDetails define if full details about this mesh must be used\n * @returns a descriptive string representing this mesh\n */\n toString(fullDetails) {\n let ret = super.toString(fullDetails);\n ret += \", n vertices: \" + this.getTotalVertices();\n ret += \", parent: \" + (this._waitingParentId ? this._waitingParentId : this.parent ? this.parent.name : \"NONE\");\n if (this.animations) {\n for (let i = 0; i < this.animations.length; i++) {\n ret += \", animation[0]: \" + this.animations[i].toString(fullDetails);\n }\n }\n if (fullDetails) {\n if (this._geometry) {\n const ib = this.getIndices();\n const vb = this.getVerticesData(VertexBuffer.PositionKind);\n if (vb && ib) {\n ret += \", flat shading: \" + (vb.length / 3 === ib.length ? \"YES\" : \"NO\");\n }\n }\n else {\n ret += \", flat shading: UNKNOWN\";\n }\n }\n return ret;\n }\n /** @internal */\n _unBindEffect() {\n super._unBindEffect();\n for (const instance of this.instances) {\n instance._unBindEffect();\n }\n }\n /**\n * Gets a boolean indicating if this mesh has LOD\n */\n get hasLODLevels() {\n return this._internalMeshDataInfo._LODLevels.length > 0;\n }\n /**\n * Gets the list of MeshLODLevel associated with the current mesh\n * @returns an array of MeshLODLevel\n */\n getLODLevels() {\n return this._internalMeshDataInfo._LODLevels;\n }\n _sortLODLevels() {\n const sortingOrderFactor = this._internalMeshDataInfo._useLODScreenCoverage ? -1 : 1;\n this._internalMeshDataInfo._LODLevels.sort((a, b) => {\n if (a.distanceOrScreenCoverage < b.distanceOrScreenCoverage) {\n return sortingOrderFactor;\n }\n if (a.distanceOrScreenCoverage > b.distanceOrScreenCoverage) {\n return -sortingOrderFactor;\n }\n return 0;\n });\n }\n /**\n * Add a mesh as LOD level triggered at the given distance.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/LOD\n * @param distanceOrScreenCoverage Either distance from the center of the object to show this level or the screen coverage if `useScreenCoverage` is set to `true`.\n * If screen coverage, value is a fraction of the screen's total surface, between 0 and 1.\n * Example Playground for distance https://playground.babylonjs.com/#QE7KM#197\n * Example Playground for screen coverage https://playground.babylonjs.com/#QE7KM#196\n * @param mesh The mesh to be added as LOD level (can be null)\n * @returns This mesh (for chaining)\n */\n addLODLevel(distanceOrScreenCoverage, mesh) {\n if (mesh && mesh._masterMesh) {\n Logger.Warn(\"You cannot use a mesh as LOD level twice\");\n return this;\n }\n const level = new MeshLODLevel(distanceOrScreenCoverage, mesh);\n this._internalMeshDataInfo._LODLevels.push(level);\n if (mesh) {\n mesh._masterMesh = this;\n }\n this._sortLODLevels();\n return this;\n }\n /**\n * Returns the LOD level mesh at the passed distance or null if not found.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/LOD\n * @param distance The distance from the center of the object to show this level\n * @returns a Mesh or `null`\n */\n getLODLevelAtDistance(distance) {\n const internalDataInfo = this._internalMeshDataInfo;\n for (let index = 0; index < internalDataInfo._LODLevels.length; index++) {\n const level = internalDataInfo._LODLevels[index];\n if (level.distanceOrScreenCoverage === distance) {\n return level.mesh;\n }\n }\n return null;\n }\n /**\n * Remove a mesh from the LOD array\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/LOD\n * @param mesh defines the mesh to be removed\n * @returns This mesh (for chaining)\n */\n removeLODLevel(mesh) {\n const internalDataInfo = this._internalMeshDataInfo;\n for (let index = 0; index < internalDataInfo._LODLevels.length; index++) {\n if (internalDataInfo._LODLevels[index].mesh === mesh) {\n internalDataInfo._LODLevels.splice(index, 1);\n if (mesh) {\n mesh._masterMesh = null;\n }\n }\n }\n this._sortLODLevels();\n return this;\n }\n /**\n * Returns the registered LOD mesh distant from the parameter `camera` position if any, else returns the current mesh.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/LOD\n * @param camera defines the camera to use to compute distance\n * @param boundingSphere defines a custom bounding sphere to use instead of the one from this mesh\n * @returns This mesh (for chaining)\n */\n getLOD(camera, boundingSphere) {\n const internalDataInfo = this._internalMeshDataInfo;\n if (!internalDataInfo._LODLevels || internalDataInfo._LODLevels.length === 0) {\n return this;\n }\n const bSphere = boundingSphere || this.getBoundingInfo().boundingSphere;\n const distanceToCamera = camera.mode === Camera.ORTHOGRAPHIC_CAMERA ? camera.minZ : bSphere.centerWorld.subtract(camera.globalPosition).length();\n let compareValue = distanceToCamera;\n let compareSign = 1;\n if (internalDataInfo._useLODScreenCoverage) {\n const screenArea = camera.screenArea;\n let meshArea = (bSphere.radiusWorld * camera.minZ) / distanceToCamera;\n meshArea = meshArea * meshArea * Math.PI;\n compareValue = meshArea / screenArea;\n compareSign = -1;\n }\n if (compareSign * internalDataInfo._LODLevels[internalDataInfo._LODLevels.length - 1].distanceOrScreenCoverage > compareSign * compareValue) {\n if (this.onLODLevelSelection) {\n this.onLODLevelSelection(compareValue, this, this);\n }\n return this;\n }\n for (let index = 0; index < internalDataInfo._LODLevels.length; index++) {\n const level = internalDataInfo._LODLevels[index];\n if (compareSign * level.distanceOrScreenCoverage < compareSign * compareValue) {\n if (level.mesh) {\n if (level.mesh.delayLoadState === 4) {\n level.mesh._checkDelayState();\n return this;\n }\n if (level.mesh.delayLoadState === 2) {\n return this;\n }\n level.mesh._preActivate();\n level.mesh._updateSubMeshesBoundingInfo(this.worldMatrixFromCache);\n }\n if (this.onLODLevelSelection) {\n this.onLODLevelSelection(compareValue, this, level.mesh);\n }\n return level.mesh;\n }\n }\n if (this.onLODLevelSelection) {\n this.onLODLevelSelection(compareValue, this, this);\n }\n return this;\n }\n /**\n * Gets the mesh internal Geometry object\n */\n get geometry() {\n return this._geometry;\n }\n /**\n * Returns the total number of vertices within the mesh geometry or zero if the mesh has no geometry.\n * @returns the total number of vertices\n */\n getTotalVertices() {\n if (this._geometry === null || this._geometry === undefined) {\n return 0;\n }\n return this._geometry.getTotalVertices();\n }\n /**\n * Returns the content of an associated vertex buffer\n * @param kind defines which buffer to read from (positions, indices, normals, etc). Possible `kind` values :\n * - VertexBuffer.PositionKind\n * - VertexBuffer.UVKind\n * - VertexBuffer.UV2Kind\n * - VertexBuffer.UV3Kind\n * - VertexBuffer.UV4Kind\n * - VertexBuffer.UV5Kind\n * - VertexBuffer.UV6Kind\n * - VertexBuffer.ColorKind\n * - VertexBuffer.MatricesIndicesKind\n * - VertexBuffer.MatricesIndicesExtraKind\n * - VertexBuffer.MatricesWeightsKind\n * - VertexBuffer.MatricesWeightsExtraKind\n * @param copyWhenShared defines a boolean indicating that if the mesh geometry is shared among some other meshes, the returned array is a copy of the internal one\n * @param forceCopy defines a boolean forcing the copy of the buffer no matter what the value of copyWhenShared is\n * @param bypassInstanceData defines a boolean indicating that the function should not take into account the instance data (applies only if the mesh has instances). Default: false\n * @returns a FloatArray or null if the mesh has no geometry or no vertex buffer for this kind.\n */\n getVerticesData(kind, copyWhenShared, forceCopy, bypassInstanceData) {\n var _a, _b;\n if (!this._geometry) {\n return null;\n }\n let data = bypassInstanceData\n ? undefined\n : (_b = (_a = this._userInstancedBuffersStorage) === null || _a === void 0 ? void 0 : _a.vertexBuffers[kind]) === null || _b === void 0 ? void 0 : _b.getFloatData(this.instances.length + 1, // +1 because the master mesh is not included in the instances array\n forceCopy || (copyWhenShared && this._geometry.meshes.length !== 1));\n if (!data) {\n data = this._geometry.getVerticesData(kind, copyWhenShared, forceCopy);\n }\n return data;\n }\n /**\n * Returns the mesh VertexBuffer object from the requested `kind`\n * @param kind defines which buffer to read from (positions, indices, normals, etc). Possible `kind` values :\n * - VertexBuffer.PositionKind\n * - VertexBuffer.NormalKind\n * - VertexBuffer.UVKind\n * - VertexBuffer.UV2Kind\n * - VertexBuffer.UV3Kind\n * - VertexBuffer.UV4Kind\n * - VertexBuffer.UV5Kind\n * - VertexBuffer.UV6Kind\n * - VertexBuffer.ColorKind\n * - VertexBuffer.MatricesIndicesKind\n * - VertexBuffer.MatricesIndicesExtraKind\n * - VertexBuffer.MatricesWeightsKind\n * - VertexBuffer.MatricesWeightsExtraKind\n * @param bypassInstanceData defines a boolean indicating that the function should not take into account the instance data (applies only if the mesh has instances). Default: false\n * @returns a FloatArray or null if the mesh has no vertex buffer for this kind.\n */\n getVertexBuffer(kind, bypassInstanceData) {\n var _a, _b;\n if (!this._geometry) {\n return null;\n }\n return (_b = (bypassInstanceData ? undefined : (_a = this._userInstancedBuffersStorage) === null || _a === void 0 ? void 0 : _a.vertexBuffers[kind])) !== null && _b !== void 0 ? _b : this._geometry.getVertexBuffer(kind);\n }\n /**\n * Tests if a specific vertex buffer is associated with this mesh\n * @param kind defines which buffer to check (positions, indices, normals, etc). Possible `kind` values :\n * - VertexBuffer.PositionKind\n * - VertexBuffer.NormalKind\n * - VertexBuffer.UVKind\n * - VertexBuffer.UV2Kind\n * - VertexBuffer.UV3Kind\n * - VertexBuffer.UV4Kind\n * - VertexBuffer.UV5Kind\n * - VertexBuffer.UV6Kind\n * - VertexBuffer.ColorKind\n * - VertexBuffer.MatricesIndicesKind\n * - VertexBuffer.MatricesIndicesExtraKind\n * - VertexBuffer.MatricesWeightsKind\n * - VertexBuffer.MatricesWeightsExtraKind\n * @param bypassInstanceData defines a boolean indicating that the function should not take into account the instance data (applies only if the mesh has instances). Default: false\n * @returns a boolean\n */\n isVerticesDataPresent(kind, bypassInstanceData) {\n var _a;\n if (!this._geometry) {\n if (this._delayInfo) {\n return this._delayInfo.indexOf(kind) !== -1;\n }\n return false;\n }\n return (!bypassInstanceData && ((_a = this._userInstancedBuffersStorage) === null || _a === void 0 ? void 0 : _a.vertexBuffers[kind]) !== undefined) || this._geometry.isVerticesDataPresent(kind);\n }\n /**\n * Returns a boolean defining if the vertex data for the requested `kind` is updatable.\n * @param kind defines which buffer to check (positions, indices, normals, etc). Possible `kind` values :\n * - VertexBuffer.PositionKind\n * - VertexBuffer.UVKind\n * - VertexBuffer.UV2Kind\n * - VertexBuffer.UV3Kind\n * - VertexBuffer.UV4Kind\n * - VertexBuffer.UV5Kind\n * - VertexBuffer.UV6Kind\n * - VertexBuffer.ColorKind\n * - VertexBuffer.MatricesIndicesKind\n * - VertexBuffer.MatricesIndicesExtraKind\n * - VertexBuffer.MatricesWeightsKind\n * - VertexBuffer.MatricesWeightsExtraKind\n * @param bypassInstanceData defines a boolean indicating that the function should not take into account the instance data (applies only if the mesh has instances). Default: false\n * @returns a boolean\n */\n isVertexBufferUpdatable(kind, bypassInstanceData) {\n var _a;\n if (!this._geometry) {\n if (this._delayInfo) {\n return this._delayInfo.indexOf(kind) !== -1;\n }\n return false;\n }\n if (!bypassInstanceData) {\n const buffer = (_a = this._userInstancedBuffersStorage) === null || _a === void 0 ? void 0 : _a.vertexBuffers[kind];\n if (buffer) {\n return buffer.isUpdatable();\n }\n }\n return this._geometry.isVertexBufferUpdatable(kind);\n }\n /**\n * Returns a string which contains the list of existing `kinds` of Vertex Data associated with this mesh.\n * @param bypassInstanceData defines a boolean indicating that the function should not take into account the instance data (applies only if the mesh has instances). Default: false\n * @returns an array of strings\n */\n getVerticesDataKinds(bypassInstanceData) {\n if (!this._geometry) {\n const result = new Array();\n if (this._delayInfo) {\n this._delayInfo.forEach(function (kind) {\n result.push(kind);\n });\n }\n return result;\n }\n const kinds = this._geometry.getVerticesDataKinds();\n if (!bypassInstanceData && this._userInstancedBuffersStorage) {\n for (const kind in this._userInstancedBuffersStorage.vertexBuffers) {\n if (kinds.indexOf(kind) === -1) {\n kinds.push(kind);\n }\n }\n }\n return kinds;\n }\n /**\n * Returns a positive integer : the total number of indices in this mesh geometry.\n * @returns the numner of indices or zero if the mesh has no geometry.\n */\n getTotalIndices() {\n if (!this._geometry) {\n return 0;\n }\n return this._geometry.getTotalIndices();\n }\n /**\n * Returns an array of integers or a typed array (Int32Array, Uint32Array, Uint16Array) populated with the mesh indices.\n * @param copyWhenShared If true (default false) and and if the mesh geometry is shared among some other meshes, the returned array is a copy of the internal one.\n * @param forceCopy defines a boolean indicating that the returned array must be cloned upon returning it\n * @returns the indices array or an empty array if the mesh has no geometry\n */\n getIndices(copyWhenShared, forceCopy) {\n if (!this._geometry) {\n return [];\n }\n return this._geometry.getIndices(copyWhenShared, forceCopy);\n }\n get isBlocked() {\n return this._masterMesh !== null && this._masterMesh !== undefined;\n }\n /**\n * Determine if the current mesh is ready to be rendered\n * @param completeCheck defines if a complete check (including materials and lights) has to be done (false by default)\n * @param forceInstanceSupport will check if the mesh will be ready when used with instances (false by default)\n * @returns true if all associated assets are ready (material, textures, shaders)\n */\n isReady(completeCheck = false, forceInstanceSupport = false) {\n var _a, _b, _c, _d, _e, _f, _g;\n if (this.delayLoadState === 2) {\n return false;\n }\n if (!super.isReady(completeCheck)) {\n return false;\n }\n if (!this.subMeshes || this.subMeshes.length === 0) {\n return true;\n }\n if (!completeCheck) {\n return true;\n }\n const engine = this.getEngine();\n const scene = this.getScene();\n const hardwareInstancedRendering = forceInstanceSupport || (engine.getCaps().instancedArrays && (this.instances.length > 0 || this.hasThinInstances));\n this.computeWorldMatrix();\n const mat = this.material || scene.defaultMaterial;\n if (mat) {\n if (mat._storeEffectOnSubMeshes) {\n for (const subMesh of this.subMeshes) {\n const effectiveMaterial = subMesh.getMaterial();\n if (effectiveMaterial) {\n if (effectiveMaterial._storeEffectOnSubMeshes) {\n if (!effectiveMaterial.isReadyForSubMesh(this, subMesh, hardwareInstancedRendering)) {\n return false;\n }\n }\n else {\n if (!effectiveMaterial.isReady(this, hardwareInstancedRendering)) {\n return false;\n }\n }\n }\n }\n }\n else {\n if (!mat.isReady(this, hardwareInstancedRendering)) {\n return false;\n }\n }\n }\n // Shadows\n const currentRenderPassId = engine.currentRenderPassId;\n for (const light of this.lightSources) {\n const generators = light.getShadowGenerators();\n if (!generators) {\n continue;\n }\n const iterator = generators.values();\n for (let key = iterator.next(); key.done !== true; key = iterator.next()) {\n const generator = key.value;\n if (generator && (!((_a = generator.getShadowMap()) === null || _a === void 0 ? void 0 : _a.renderList) || (((_b = generator.getShadowMap()) === null || _b === void 0 ? void 0 : _b.renderList) && ((_d = (_c = generator.getShadowMap()) === null || _c === void 0 ? void 0 : _c.renderList) === null || _d === void 0 ? void 0 : _d.indexOf(this)) !== -1))) {\n const shadowMap = generator.getShadowMap();\n const renderPassIds = (_e = shadowMap.renderPassIds) !== null && _e !== void 0 ? _e : [engine.currentRenderPassId];\n for (let p = 0; p < renderPassIds.length; ++p) {\n engine.currentRenderPassId = renderPassIds[p];\n for (const subMesh of this.subMeshes) {\n if (!generator.isReady(subMesh, hardwareInstancedRendering, (_g = (_f = subMesh.getMaterial()) === null || _f === void 0 ? void 0 : _f.needAlphaBlendingForMesh(this)) !== null && _g !== void 0 ? _g : false)) {\n engine.currentRenderPassId = currentRenderPassId;\n return false;\n }\n }\n }\n engine.currentRenderPassId = currentRenderPassId;\n }\n }\n }\n // LOD\n for (const lod of this._internalMeshDataInfo._LODLevels) {\n if (lod.mesh && !lod.mesh.isReady(hardwareInstancedRendering)) {\n return false;\n }\n }\n return true;\n }\n /**\n * Gets a boolean indicating if the normals aren't to be recomputed on next mesh `positions` array update. This property is pertinent only for updatable parametric shapes.\n */\n get areNormalsFrozen() {\n return this._internalMeshDataInfo._areNormalsFrozen;\n }\n /**\n * This function affects parametric shapes on vertex position update only : ribbons, tubes, etc. It has no effect at all on other shapes. It prevents the mesh normals from being recomputed on next `positions` array update.\n * @returns the current mesh\n */\n freezeNormals() {\n this._internalMeshDataInfo._areNormalsFrozen = true;\n return this;\n }\n /**\n * This function affects parametric shapes on vertex position update only : ribbons, tubes, etc. It has no effect at all on other shapes. It reactivates the mesh normals computation if it was previously frozen\n * @returns the current mesh\n */\n unfreezeNormals() {\n this._internalMeshDataInfo._areNormalsFrozen = false;\n return this;\n }\n /**\n * Sets a value overriding the instance count. Only applicable when custom instanced InterleavedVertexBuffer are used rather than InstancedMeshs\n */\n set overridenInstanceCount(count) {\n this._instanceDataStorage.overridenInstanceCount = count;\n }\n // Methods\n /** @internal */\n _preActivate() {\n const internalDataInfo = this._internalMeshDataInfo;\n const sceneRenderId = this.getScene().getRenderId();\n if (internalDataInfo._preActivateId === sceneRenderId) {\n return this;\n }\n internalDataInfo._preActivateId = sceneRenderId;\n this._instanceDataStorage.visibleInstances = null;\n return this;\n }\n /**\n * @internal\n */\n _preActivateForIntermediateRendering(renderId) {\n if (this._instanceDataStorage.visibleInstances) {\n this._instanceDataStorage.visibleInstances.intermediateDefaultRenderId = renderId;\n }\n return this;\n }\n /**\n * @internal\n */\n _registerInstanceForRenderId(instance, renderId) {\n if (!this._instanceDataStorage.visibleInstances) {\n this._instanceDataStorage.visibleInstances = {\n defaultRenderId: renderId,\n selfDefaultRenderId: this._renderId,\n };\n }\n if (!this._instanceDataStorage.visibleInstances[renderId]) {\n if (this._instanceDataStorage.previousRenderId !== undefined && this._instanceDataStorage.isFrozen) {\n this._instanceDataStorage.visibleInstances[this._instanceDataStorage.previousRenderId] = null;\n }\n this._instanceDataStorage.previousRenderId = renderId;\n this._instanceDataStorage.visibleInstances[renderId] = new Array();\n }\n this._instanceDataStorage.visibleInstances[renderId].push(instance);\n return this;\n }\n _afterComputeWorldMatrix() {\n super._afterComputeWorldMatrix();\n if (!this.hasThinInstances) {\n return;\n }\n if (!this.doNotSyncBoundingInfo) {\n this.thinInstanceRefreshBoundingInfo(false);\n }\n }\n /** @internal */\n _postActivate() {\n if (this.edgesShareWithInstances && this.edgesRenderer && this.edgesRenderer.isEnabled && this._renderingGroup) {\n this._renderingGroup._edgesRenderers.pushNoDuplicate(this.edgesRenderer);\n this.edgesRenderer.customInstances.push(this.getWorldMatrix());\n }\n }\n /**\n * This method recomputes and sets a new BoundingInfo to the mesh unless it is locked.\n * This means the mesh underlying bounding box and sphere are recomputed.\n * @param applySkeleton defines whether to apply the skeleton before computing the bounding info\n * @param applyMorph defines whether to apply the morph target before computing the bounding info\n * @returns the current mesh\n */\n refreshBoundingInfo(applySkeleton = false, applyMorph = false) {\n if (this.hasBoundingInfo && this.getBoundingInfo().isLocked) {\n return this;\n }\n const bias = this.geometry ? this.geometry.boundingBias : null;\n this._refreshBoundingInfo(this._getPositionData(applySkeleton, applyMorph), bias);\n return this;\n }\n /**\n * @internal\n */\n _createGlobalSubMesh(force) {\n const totalVertices = this.getTotalVertices();\n if (!totalVertices || !this.getIndices()) {\n return null;\n }\n // Check if we need to recreate the submeshes\n if (this.subMeshes && this.subMeshes.length > 0) {\n const ib = this.getIndices();\n if (!ib) {\n return null;\n }\n const totalIndices = ib.length;\n let needToRecreate = false;\n if (force) {\n needToRecreate = true;\n }\n else {\n for (const submesh of this.subMeshes) {\n if (submesh.indexStart + submesh.indexCount > totalIndices) {\n needToRecreate = true;\n break;\n }\n if (submesh.verticesStart + submesh.verticesCount > totalVertices) {\n needToRecreate = true;\n break;\n }\n }\n }\n if (!needToRecreate) {\n return this.subMeshes[0];\n }\n }\n this.releaseSubMeshes();\n return new SubMesh(0, 0, totalVertices, 0, this.getTotalIndices(), this);\n }\n /**\n * This function will subdivide the mesh into multiple submeshes\n * @param count defines the expected number of submeshes\n */\n subdivide(count) {\n if (count < 1) {\n return;\n }\n const totalIndices = this.getTotalIndices();\n let subdivisionSize = (totalIndices / count) | 0;\n let offset = 0;\n // Ensure that subdivisionSize is a multiple of 3\n while (subdivisionSize % 3 !== 0) {\n subdivisionSize++;\n }\n this.releaseSubMeshes();\n for (let index = 0; index < count; index++) {\n if (offset >= totalIndices) {\n break;\n }\n SubMesh.CreateFromIndices(0, offset, index === count - 1 ? totalIndices - offset : subdivisionSize, this, undefined, false);\n offset += subdivisionSize;\n }\n this.refreshBoundingInfo();\n this.synchronizeInstances();\n }\n /**\n * Copy a FloatArray into a specific associated vertex buffer\n * @param kind defines which buffer to write to (positions, indices, normals, etc). Possible `kind` values :\n * - VertexBuffer.PositionKind\n * - VertexBuffer.UVKind\n * - VertexBuffer.UV2Kind\n * - VertexBuffer.UV3Kind\n * - VertexBuffer.UV4Kind\n * - VertexBuffer.UV5Kind\n * - VertexBuffer.UV6Kind\n * - VertexBuffer.ColorKind\n * - VertexBuffer.MatricesIndicesKind\n * - VertexBuffer.MatricesIndicesExtraKind\n * - VertexBuffer.MatricesWeightsKind\n * - VertexBuffer.MatricesWeightsExtraKind\n * @param data defines the data source\n * @param updatable defines if the updated vertex buffer must be flagged as updatable\n * @param stride defines the data stride size (can be null)\n * @returns the current mesh\n */\n setVerticesData(kind, data, updatable = false, stride) {\n if (!this._geometry) {\n const vertexData = new VertexData();\n vertexData.set(data, kind);\n const scene = this.getScene();\n new Geometry(Geometry.RandomId(), scene, vertexData, updatable, this);\n }\n else {\n this._geometry.setVerticesData(kind, data, updatable, stride);\n }\n return this;\n }\n /**\n * Delete a vertex buffer associated with this mesh\n * @param kind defines which buffer to delete (positions, indices, normals, etc). Possible `kind` values :\n * - VertexBuffer.PositionKind\n * - VertexBuffer.UVKind\n * - VertexBuffer.UV2Kind\n * - VertexBuffer.UV3Kind\n * - VertexBuffer.UV4Kind\n * - VertexBuffer.UV5Kind\n * - VertexBuffer.UV6Kind\n * - VertexBuffer.ColorKind\n * - VertexBuffer.MatricesIndicesKind\n * - VertexBuffer.MatricesIndicesExtraKind\n * - VertexBuffer.MatricesWeightsKind\n * - VertexBuffer.MatricesWeightsExtraKind\n */\n removeVerticesData(kind) {\n if (!this._geometry) {\n return;\n }\n this._geometry.removeVerticesData(kind);\n }\n /**\n * Flags an associated vertex buffer as updatable\n * @param kind defines which buffer to use (positions, indices, normals, etc). Possible `kind` values :\n * - VertexBuffer.PositionKind\n * - VertexBuffer.UVKind\n * - VertexBuffer.UV2Kind\n * - VertexBuffer.UV3Kind\n * - VertexBuffer.UV4Kind\n * - VertexBuffer.UV5Kind\n * - VertexBuffer.UV6Kind\n * - VertexBuffer.ColorKind\n * - VertexBuffer.MatricesIndicesKind\n * - VertexBuffer.MatricesIndicesExtraKind\n * - VertexBuffer.MatricesWeightsKind\n * - VertexBuffer.MatricesWeightsExtraKind\n * @param updatable defines if the updated vertex buffer must be flagged as updatable\n */\n markVerticesDataAsUpdatable(kind, updatable = true) {\n const vb = this.getVertexBuffer(kind);\n if (!vb || vb.isUpdatable() === updatable) {\n return;\n }\n this.setVerticesData(kind, this.getVerticesData(kind), updatable);\n }\n /**\n * Sets the mesh global Vertex Buffer\n * @param buffer defines the buffer to use\n * @param disposeExistingBuffer disposes the existing buffer, if any (default: true)\n * @returns the current mesh\n */\n setVerticesBuffer(buffer, disposeExistingBuffer = true) {\n if (!this._geometry) {\n this._geometry = Geometry.CreateGeometryForMesh(this);\n }\n this._geometry.setVerticesBuffer(buffer, null, disposeExistingBuffer);\n return this;\n }\n /**\n * Update a specific associated vertex buffer\n * @param kind defines which buffer to write to (positions, indices, normals, etc). Possible `kind` values :\n * - VertexBuffer.PositionKind\n * - VertexBuffer.UVKind\n * - VertexBuffer.UV2Kind\n * - VertexBuffer.UV3Kind\n * - VertexBuffer.UV4Kind\n * - VertexBuffer.UV5Kind\n * - VertexBuffer.UV6Kind\n * - VertexBuffer.ColorKind\n * - VertexBuffer.MatricesIndicesKind\n * - VertexBuffer.MatricesIndicesExtraKind\n * - VertexBuffer.MatricesWeightsKind\n * - VertexBuffer.MatricesWeightsExtraKind\n * @param data defines the data source\n * @param updateExtends defines if extends info of the mesh must be updated (can be null). This is mostly useful for \"position\" kind\n * @param makeItUnique defines if the geometry associated with the mesh must be cloned to make the change only for this mesh (and not all meshes associated with the same geometry)\n * @returns the current mesh\n */\n updateVerticesData(kind, data, updateExtends, makeItUnique) {\n if (!this._geometry) {\n return this;\n }\n if (!makeItUnique) {\n this._geometry.updateVerticesData(kind, data, updateExtends);\n }\n else {\n this.makeGeometryUnique();\n this.updateVerticesData(kind, data, updateExtends, false);\n }\n return this;\n }\n /**\n * This method updates the vertex positions of an updatable mesh according to the `positionFunction` returned values.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#other-shapes-updatemeshpositions\n * @param positionFunction is a simple JS function what is passed the mesh `positions` array. It doesn't need to return anything\n * @param computeNormals is a boolean (default true) to enable/disable the mesh normal recomputation after the vertex position update\n * @returns the current mesh\n */\n updateMeshPositions(positionFunction, computeNormals = true) {\n const positions = this.getVerticesData(VertexBuffer.PositionKind);\n if (!positions) {\n return this;\n }\n positionFunction(positions);\n this.updateVerticesData(VertexBuffer.PositionKind, positions, false, false);\n if (computeNormals) {\n const indices = this.getIndices();\n const normals = this.getVerticesData(VertexBuffer.NormalKind);\n if (!normals) {\n return this;\n }\n VertexData.ComputeNormals(positions, indices, normals);\n this.updateVerticesData(VertexBuffer.NormalKind, normals, false, false);\n }\n return this;\n }\n /**\n * Creates a un-shared specific occurence of the geometry for the mesh.\n * @returns the current mesh\n */\n makeGeometryUnique() {\n if (!this._geometry) {\n return this;\n }\n if (this._geometry.meshes.length === 1) {\n return this;\n }\n const oldGeometry = this._geometry;\n const geometry = this._geometry.copy(Geometry.RandomId());\n oldGeometry.releaseForMesh(this, true);\n geometry.applyToMesh(this);\n return this;\n }\n /**\n * Set the index buffer of this mesh\n * @param indices defines the source data\n * @param totalVertices defines the total number of vertices referenced by this index data (can be null)\n * @param updatable defines if the updated index buffer must be flagged as updatable (default is false)\n * @returns the current mesh\n */\n setIndices(indices, totalVertices = null, updatable = false) {\n if (!this._geometry) {\n const vertexData = new VertexData();\n vertexData.indices = indices;\n const scene = this.getScene();\n new Geometry(Geometry.RandomId(), scene, vertexData, updatable, this);\n }\n else {\n this._geometry.setIndices(indices, totalVertices, updatable);\n }\n return this;\n }\n /**\n * Update the current index buffer\n * @param indices defines the source data\n * @param offset defines the offset in the index buffer where to store the new data (can be null)\n * @param gpuMemoryOnly defines a boolean indicating that only the GPU memory must be updated leaving the CPU version of the indices unchanged (false by default)\n * @returns the current mesh\n */\n updateIndices(indices, offset, gpuMemoryOnly = false) {\n if (!this._geometry) {\n return this;\n }\n this._geometry.updateIndices(indices, offset, gpuMemoryOnly);\n return this;\n }\n /**\n * Invert the geometry to move from a right handed system to a left handed one.\n * @returns the current mesh\n */\n toLeftHanded() {\n if (!this._geometry) {\n return this;\n }\n this._geometry.toLeftHanded();\n return this;\n }\n /**\n * @internal\n */\n _bind(subMesh, effect, fillMode, allowInstancedRendering = true) {\n if (!this._geometry) {\n return this;\n }\n const engine = this.getScene().getEngine();\n // Morph targets\n if (this.morphTargetManager && this.morphTargetManager.isUsingTextureForTargets) {\n this.morphTargetManager._bind(effect);\n }\n // Wireframe\n let indexToBind;\n if (this._unIndexed) {\n indexToBind = null;\n }\n else {\n switch (this._getRenderingFillMode(fillMode)) {\n case Material.PointFillMode:\n indexToBind = null;\n break;\n case Material.WireFrameFillMode:\n indexToBind = subMesh._getLinesIndexBuffer(this.getIndices(), engine);\n break;\n default:\n case Material.TriangleFillMode:\n indexToBind = this._geometry.getIndexBuffer();\n break;\n }\n }\n // VBOs\n if (!allowInstancedRendering || !this._userInstancedBuffersStorage || this.hasThinInstances) {\n this._geometry._bind(effect, indexToBind);\n }\n else {\n this._geometry._bind(effect, indexToBind, this._userInstancedBuffersStorage.vertexBuffers, this._userInstancedBuffersStorage.vertexArrayObjects);\n }\n return this;\n }\n /**\n * @internal\n */\n _draw(subMesh, fillMode, instancesCount) {\n if (!this._geometry || !this._geometry.getVertexBuffers() || (!this._unIndexed && !this._geometry.getIndexBuffer())) {\n return this;\n }\n if (this._internalMeshDataInfo._onBeforeDrawObservable) {\n this._internalMeshDataInfo._onBeforeDrawObservable.notifyObservers(this);\n }\n const scene = this.getScene();\n const engine = scene.getEngine();\n if (this._unIndexed || fillMode == Material.PointFillMode) {\n // or triangles as points\n engine.drawArraysType(fillMode, subMesh.verticesStart, subMesh.verticesCount, this.forcedInstanceCount || instancesCount);\n }\n else if (fillMode == Material.WireFrameFillMode) {\n // Triangles as wireframe\n engine.drawElementsType(fillMode, 0, subMesh._linesIndexCount, this.forcedInstanceCount || instancesCount);\n }\n else {\n engine.drawElementsType(fillMode, subMesh.indexStart, subMesh.indexCount, this.forcedInstanceCount || instancesCount);\n }\n return this;\n }\n /**\n * Registers for this mesh a javascript function called just before the rendering process\n * @param func defines the function to call before rendering this mesh\n * @returns the current mesh\n */\n registerBeforeRender(func) {\n this.onBeforeRenderObservable.add(func);\n return this;\n }\n /**\n * Disposes a previously registered javascript function called before the rendering\n * @param func defines the function to remove\n * @returns the current mesh\n */\n unregisterBeforeRender(func) {\n this.onBeforeRenderObservable.removeCallback(func);\n return this;\n }\n /**\n * Registers for this mesh a javascript function called just after the rendering is complete\n * @param func defines the function to call after rendering this mesh\n * @returns the current mesh\n */\n registerAfterRender(func) {\n this.onAfterRenderObservable.add(func);\n return this;\n }\n /**\n * Disposes a previously registered javascript function called after the rendering.\n * @param func defines the function to remove\n * @returns the current mesh\n */\n unregisterAfterRender(func) {\n this.onAfterRenderObservable.removeCallback(func);\n return this;\n }\n /**\n * @internal\n */\n _getInstancesRenderList(subMeshId, isReplacementMode = false) {\n if (this._instanceDataStorage.isFrozen) {\n if (isReplacementMode) {\n this._instanceDataStorage.batchCacheReplacementModeInFrozenMode.hardwareInstancedRendering[subMeshId] = false;\n this._instanceDataStorage.batchCacheReplacementModeInFrozenMode.renderSelf[subMeshId] = true;\n return this._instanceDataStorage.batchCacheReplacementModeInFrozenMode;\n }\n if (this._instanceDataStorage.previousBatch) {\n return this._instanceDataStorage.previousBatch;\n }\n }\n const scene = this.getScene();\n const isInIntermediateRendering = scene._isInIntermediateRendering();\n const onlyForInstances = isInIntermediateRendering\n ? this._internalAbstractMeshDataInfo._onlyForInstancesIntermediate\n : this._internalAbstractMeshDataInfo._onlyForInstances;\n const batchCache = this._instanceDataStorage.batchCache;\n batchCache.mustReturn = false;\n batchCache.renderSelf[subMeshId] = isReplacementMode || (!onlyForInstances && this.isEnabled() && this.isVisible);\n batchCache.visibleInstances[subMeshId] = null;\n if (this._instanceDataStorage.visibleInstances && !isReplacementMode) {\n const visibleInstances = this._instanceDataStorage.visibleInstances;\n const currentRenderId = scene.getRenderId();\n const defaultRenderId = isInIntermediateRendering ? visibleInstances.intermediateDefaultRenderId : visibleInstances.defaultRenderId;\n batchCache.visibleInstances[subMeshId] = visibleInstances[currentRenderId];\n if (!batchCache.visibleInstances[subMeshId] && defaultRenderId) {\n batchCache.visibleInstances[subMeshId] = visibleInstances[defaultRenderId];\n }\n }\n batchCache.hardwareInstancedRendering[subMeshId] =\n !isReplacementMode &&\n this._instanceDataStorage.hardwareInstancedRendering &&\n batchCache.visibleInstances[subMeshId] !== null &&\n batchCache.visibleInstances[subMeshId] !== undefined;\n this._instanceDataStorage.previousBatch = batchCache;\n return batchCache;\n }\n /**\n * @internal\n */\n _renderWithInstances(subMesh, fillMode, batch, effect, engine) {\n var _a;\n const visibleInstances = batch.visibleInstances[subMesh._id];\n const visibleInstanceCount = visibleInstances ? visibleInstances.length : 0;\n const instanceStorage = this._instanceDataStorage;\n const currentInstancesBufferSize = instanceStorage.instancesBufferSize;\n let instancesBuffer = instanceStorage.instancesBuffer;\n let instancesPreviousBuffer = instanceStorage.instancesPreviousBuffer;\n const matricesCount = visibleInstanceCount + 1;\n const bufferSize = matricesCount * 16 * 4;\n while (instanceStorage.instancesBufferSize < bufferSize) {\n instanceStorage.instancesBufferSize *= 2;\n }\n if (!instanceStorage.instancesData || currentInstancesBufferSize != instanceStorage.instancesBufferSize) {\n instanceStorage.instancesData = new Float32Array(instanceStorage.instancesBufferSize / 4);\n }\n if ((this._scene.needsPreviousWorldMatrices && !instanceStorage.instancesPreviousData) || currentInstancesBufferSize != instanceStorage.instancesBufferSize) {\n instanceStorage.instancesPreviousData = new Float32Array(instanceStorage.instancesBufferSize / 4);\n }\n let offset = 0;\n let instancesCount = 0;\n const renderSelf = batch.renderSelf[subMesh._id];\n const needUpdateBuffer = !instancesBuffer ||\n currentInstancesBufferSize !== instanceStorage.instancesBufferSize ||\n (this._scene.needsPreviousWorldMatrices && !instanceStorage.instancesPreviousBuffer);\n if (!this._instanceDataStorage.manualUpdate && (!instanceStorage.isFrozen || needUpdateBuffer)) {\n const world = this.getWorldMatrix();\n if (renderSelf) {\n if (this._scene.needsPreviousWorldMatrices) {\n if (!instanceStorage.masterMeshPreviousWorldMatrix) {\n instanceStorage.masterMeshPreviousWorldMatrix = world.clone();\n instanceStorage.masterMeshPreviousWorldMatrix.copyToArray(instanceStorage.instancesPreviousData, offset);\n }\n else {\n instanceStorage.masterMeshPreviousWorldMatrix.copyToArray(instanceStorage.instancesPreviousData, offset);\n instanceStorage.masterMeshPreviousWorldMatrix.copyFrom(world);\n }\n }\n world.copyToArray(instanceStorage.instancesData, offset);\n offset += 16;\n instancesCount++;\n }\n if (visibleInstances) {\n if (Mesh.INSTANCEDMESH_SORT_TRANSPARENT && this._scene.activeCamera && ((_a = subMesh.getMaterial()) === null || _a === void 0 ? void 0 : _a.needAlphaBlendingForMesh(subMesh.getRenderingMesh()))) {\n const cameraPosition = this._scene.activeCamera.globalPosition;\n for (let instanceIndex = 0; instanceIndex < visibleInstances.length; instanceIndex++) {\n const instanceMesh = visibleInstances[instanceIndex];\n instanceMesh._distanceToCamera = Vector3.Distance(instanceMesh.getBoundingInfo().boundingSphere.centerWorld, cameraPosition);\n }\n visibleInstances.sort((m1, m2) => {\n return m1._distanceToCamera > m2._distanceToCamera ? -1 : m1._distanceToCamera < m2._distanceToCamera ? 1 : 0;\n });\n }\n for (let instanceIndex = 0; instanceIndex < visibleInstances.length; instanceIndex++) {\n const instance = visibleInstances[instanceIndex];\n const matrix = instance.getWorldMatrix();\n matrix.copyToArray(instanceStorage.instancesData, offset);\n if (this._scene.needsPreviousWorldMatrices) {\n if (!instance._previousWorldMatrix) {\n instance._previousWorldMatrix = matrix.clone();\n instance._previousWorldMatrix.copyToArray(instanceStorage.instancesPreviousData, offset);\n }\n else {\n instance._previousWorldMatrix.copyToArray(instanceStorage.instancesPreviousData, offset);\n instance._previousWorldMatrix.copyFrom(matrix);\n }\n }\n offset += 16;\n instancesCount++;\n }\n }\n }\n else {\n instancesCount = (renderSelf ? 1 : 0) + visibleInstanceCount;\n }\n if (needUpdateBuffer) {\n if (instancesBuffer) {\n instancesBuffer.dispose();\n }\n if (instancesPreviousBuffer) {\n instancesPreviousBuffer.dispose();\n }\n instancesBuffer = new Buffer$1(engine, instanceStorage.instancesData, true, 16, false, true);\n instanceStorage.instancesBuffer = instancesBuffer;\n if (!this._userInstancedBuffersStorage) {\n this._userInstancedBuffersStorage = {\n data: {},\n vertexBuffers: {},\n strides: {},\n sizes: {},\n vertexArrayObjects: this.getEngine().getCaps().vertexArrayObject ? {} : undefined,\n };\n }\n this._userInstancedBuffersStorage.vertexBuffers[\"world0\"] = instancesBuffer.createVertexBuffer(\"world0\", 0, 4);\n this._userInstancedBuffersStorage.vertexBuffers[\"world1\"] = instancesBuffer.createVertexBuffer(\"world1\", 4, 4);\n this._userInstancedBuffersStorage.vertexBuffers[\"world2\"] = instancesBuffer.createVertexBuffer(\"world2\", 8, 4);\n this._userInstancedBuffersStorage.vertexBuffers[\"world3\"] = instancesBuffer.createVertexBuffer(\"world3\", 12, 4);\n if (this._scene.needsPreviousWorldMatrices) {\n instancesPreviousBuffer = new Buffer$1(engine, instanceStorage.instancesPreviousData, true, 16, false, true);\n instanceStorage.instancesPreviousBuffer = instancesPreviousBuffer;\n this._userInstancedBuffersStorage.vertexBuffers[\"previousWorld0\"] = instancesPreviousBuffer.createVertexBuffer(\"previousWorld0\", 0, 4);\n this._userInstancedBuffersStorage.vertexBuffers[\"previousWorld1\"] = instancesPreviousBuffer.createVertexBuffer(\"previousWorld1\", 4, 4);\n this._userInstancedBuffersStorage.vertexBuffers[\"previousWorld2\"] = instancesPreviousBuffer.createVertexBuffer(\"previousWorld2\", 8, 4);\n this._userInstancedBuffersStorage.vertexBuffers[\"previousWorld3\"] = instancesPreviousBuffer.createVertexBuffer(\"previousWorld3\", 12, 4);\n }\n this._invalidateInstanceVertexArrayObject();\n }\n else {\n if (!this._instanceDataStorage.isFrozen || this._instanceDataStorage.forceMatrixUpdates) {\n instancesBuffer.updateDirectly(instanceStorage.instancesData, 0, instancesCount);\n if (this._scene.needsPreviousWorldMatrices && (!this._instanceDataStorage.manualUpdate || this._instanceDataStorage.previousManualUpdate)) {\n instancesPreviousBuffer.updateDirectly(instanceStorage.instancesPreviousData, 0, instancesCount);\n }\n }\n }\n this._processInstancedBuffers(visibleInstances, renderSelf);\n // Stats\n this.getScene()._activeIndices.addCount(subMesh.indexCount * instancesCount, false);\n // Draw\n if (engine._currentDrawContext) {\n engine._currentDrawContext.useInstancing = true;\n }\n this._bind(subMesh, effect, fillMode);\n this._draw(subMesh, fillMode, instancesCount);\n // Write current matrices as previous matrices in case of manual update\n // Default behaviour when previous matrices are not specified explicitly\n // Will break if instances number/order changes\n if (this._scene.needsPreviousWorldMatrices &&\n !needUpdateBuffer &&\n this._instanceDataStorage.manualUpdate &&\n (!this._instanceDataStorage.isFrozen || this._instanceDataStorage.forceMatrixUpdates) &&\n !this._instanceDataStorage.previousManualUpdate) {\n instancesPreviousBuffer.updateDirectly(instanceStorage.instancesData, 0, instancesCount);\n }\n engine.unbindInstanceAttributes();\n return this;\n }\n /**\n * @internal\n */\n _renderWithThinInstances(subMesh, fillMode, effect, engine) {\n var _a, _b;\n // Stats\n const instancesCount = (_b = (_a = this._thinInstanceDataStorage) === null || _a === void 0 ? void 0 : _a.instancesCount) !== null && _b !== void 0 ? _b : 0;\n this.getScene()._activeIndices.addCount(subMesh.indexCount * instancesCount, false);\n // Draw\n if (engine._currentDrawContext) {\n engine._currentDrawContext.useInstancing = true;\n }\n this._bind(subMesh, effect, fillMode);\n this._draw(subMesh, fillMode, instancesCount);\n // Write current matrices as previous matrices\n // Default behaviour when previous matrices are not specified explicitly\n // Will break if instances number/order changes\n if (this._scene.needsPreviousWorldMatrices && !this._thinInstanceDataStorage.previousMatrixData && this._thinInstanceDataStorage.matrixData) {\n if (!this._thinInstanceDataStorage.previousMatrixBuffer) {\n this._thinInstanceDataStorage.previousMatrixBuffer = this._thinInstanceCreateMatrixBuffer(\"previousWorld\", this._thinInstanceDataStorage.matrixData, false);\n }\n else {\n this._thinInstanceDataStorage.previousMatrixBuffer.updateDirectly(this._thinInstanceDataStorage.matrixData, 0, instancesCount);\n }\n }\n engine.unbindInstanceAttributes();\n }\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _processInstancedBuffers(visibleInstances, renderSelf) {\n // Do nothing\n }\n /**\n * @internal\n */\n _processRendering(renderingMesh, subMesh, effect, fillMode, batch, hardwareInstancedRendering, onBeforeDraw, effectiveMaterial) {\n const scene = this.getScene();\n const engine = scene.getEngine();\n fillMode = this._getRenderingFillMode(fillMode);\n if (hardwareInstancedRendering && subMesh.getRenderingMesh().hasThinInstances) {\n this._renderWithThinInstances(subMesh, fillMode, effect, engine);\n return this;\n }\n if (hardwareInstancedRendering) {\n this._renderWithInstances(subMesh, fillMode, batch, effect, engine);\n }\n else {\n if (engine._currentDrawContext) {\n engine._currentDrawContext.useInstancing = false;\n }\n let instanceCount = 0;\n if (batch.renderSelf[subMesh._id]) {\n // Draw\n if (onBeforeDraw) {\n onBeforeDraw(false, renderingMesh.getWorldMatrix(), effectiveMaterial);\n }\n instanceCount++;\n this._draw(subMesh, fillMode, this._instanceDataStorage.overridenInstanceCount);\n }\n const visibleInstancesForSubMesh = batch.visibleInstances[subMesh._id];\n if (visibleInstancesForSubMesh) {\n const visibleInstanceCount = visibleInstancesForSubMesh.length;\n instanceCount += visibleInstanceCount;\n // Stats\n for (let instanceIndex = 0; instanceIndex < visibleInstanceCount; instanceIndex++) {\n const instance = visibleInstancesForSubMesh[instanceIndex];\n // World\n const world = instance.getWorldMatrix();\n if (onBeforeDraw) {\n onBeforeDraw(true, world, effectiveMaterial);\n }\n // Draw\n this._draw(subMesh, fillMode);\n }\n }\n // Stats\n scene._activeIndices.addCount(subMesh.indexCount * instanceCount, false);\n }\n return this;\n }\n /**\n * @internal\n */\n _rebuild(dispose = false) {\n if (this._instanceDataStorage.instancesBuffer) {\n // Dispose instance buffer to be recreated in _renderWithInstances when rendered\n if (dispose) {\n this._instanceDataStorage.instancesBuffer.dispose();\n }\n this._instanceDataStorage.instancesBuffer = null;\n }\n if (this._userInstancedBuffersStorage) {\n for (const kind in this._userInstancedBuffersStorage.vertexBuffers) {\n const buffer = this._userInstancedBuffersStorage.vertexBuffers[kind];\n if (buffer) {\n // Dispose instance buffer to be recreated in _renderWithInstances when rendered\n if (dispose) {\n buffer.dispose();\n }\n this._userInstancedBuffersStorage.vertexBuffers[kind] = null;\n }\n }\n if (this._userInstancedBuffersStorage.vertexArrayObjects) {\n this._userInstancedBuffersStorage.vertexArrayObjects = {};\n }\n }\n this._internalMeshDataInfo._effectiveMaterial = null;\n super._rebuild(dispose);\n }\n /** @internal */\n _freeze() {\n if (!this.subMeshes) {\n return;\n }\n // Prepare batches\n for (let index = 0; index < this.subMeshes.length; index++) {\n this._getInstancesRenderList(index);\n }\n this._internalMeshDataInfo._effectiveMaterial = null;\n this._instanceDataStorage.isFrozen = true;\n }\n /** @internal */\n _unFreeze() {\n this._instanceDataStorage.isFrozen = false;\n this._instanceDataStorage.previousBatch = null;\n }\n /**\n * Triggers the draw call for the mesh. Usually, you don't need to call this method by your own because the mesh rendering is handled by the scene rendering manager\n * @param subMesh defines the subMesh to render\n * @param enableAlphaMode defines if alpha mode can be changed\n * @param effectiveMeshReplacement defines an optional mesh used to provide info for the rendering\n * @returns the current mesh\n */\n render(subMesh, enableAlphaMode, effectiveMeshReplacement) {\n var _a, _b, _c;\n const scene = this.getScene();\n if (this._internalAbstractMeshDataInfo._isActiveIntermediate) {\n this._internalAbstractMeshDataInfo._isActiveIntermediate = false;\n }\n else {\n this._internalAbstractMeshDataInfo._isActive = false;\n }\n if (this._checkOcclusionQuery() && !this._occlusionDataStorage.forceRenderingWhenOccluded) {\n return this;\n }\n // Managing instances\n const batch = this._getInstancesRenderList(subMesh._id, !!effectiveMeshReplacement);\n if (batch.mustReturn) {\n return this;\n }\n // Checking geometry state\n if (!this._geometry || !this._geometry.getVertexBuffers() || (!this._unIndexed && !this._geometry.getIndexBuffer())) {\n return this;\n }\n const engine = scene.getEngine();\n let oldCameraMaxZ = 0;\n let oldCamera = null;\n if (this.ignoreCameraMaxZ && scene.activeCamera && !scene._isInIntermediateRendering()) {\n oldCameraMaxZ = scene.activeCamera.maxZ;\n oldCamera = scene.activeCamera;\n scene.activeCamera.maxZ = 0;\n scene.updateTransformMatrix(true);\n }\n if (this._internalMeshDataInfo._onBeforeRenderObservable) {\n this._internalMeshDataInfo._onBeforeRenderObservable.notifyObservers(this);\n }\n const renderingMesh = subMesh.getRenderingMesh();\n const hardwareInstancedRendering = batch.hardwareInstancedRendering[subMesh._id] ||\n renderingMesh.hasThinInstances ||\n (!!this._userInstancedBuffersStorage && !subMesh.getMesh()._internalAbstractMeshDataInfo._actAsRegularMesh);\n const instanceDataStorage = this._instanceDataStorage;\n const material = subMesh.getMaterial();\n if (!material) {\n if (oldCamera) {\n oldCamera.maxZ = oldCameraMaxZ;\n scene.updateTransformMatrix(true);\n }\n return this;\n }\n // Material\n if (!instanceDataStorage.isFrozen || !this._internalMeshDataInfo._effectiveMaterial || this._internalMeshDataInfo._effectiveMaterial !== material) {\n if (material._storeEffectOnSubMeshes) {\n if (!material.isReadyForSubMesh(this, subMesh, hardwareInstancedRendering)) {\n if (oldCamera) {\n oldCamera.maxZ = oldCameraMaxZ;\n scene.updateTransformMatrix(true);\n }\n return this;\n }\n }\n else if (!material.isReady(this, hardwareInstancedRendering)) {\n if (oldCamera) {\n oldCamera.maxZ = oldCameraMaxZ;\n scene.updateTransformMatrix(true);\n }\n return this;\n }\n this._internalMeshDataInfo._effectiveMaterial = material;\n }\n else if ((material._storeEffectOnSubMeshes && !((_a = subMesh.effect) === null || _a === void 0 ? void 0 : _a._wasPreviouslyReady)) ||\n (!material._storeEffectOnSubMeshes && !((_b = material.getEffect()) === null || _b === void 0 ? void 0 : _b._wasPreviouslyReady))) {\n if (oldCamera) {\n oldCamera.maxZ = oldCameraMaxZ;\n scene.updateTransformMatrix(true);\n }\n return this;\n }\n // Alpha mode\n if (enableAlphaMode) {\n engine.setAlphaMode(this._internalMeshDataInfo._effectiveMaterial.alphaMode);\n }\n let drawWrapper;\n if (this._internalMeshDataInfo._effectiveMaterial._storeEffectOnSubMeshes) {\n drawWrapper = subMesh._drawWrapper;\n }\n else {\n drawWrapper = this._internalMeshDataInfo._effectiveMaterial._getDrawWrapper();\n }\n const effect = (_c = drawWrapper === null || drawWrapper === void 0 ? void 0 : drawWrapper.effect) !== null && _c !== void 0 ? _c : null;\n for (const step of scene._beforeRenderingMeshStage) {\n step.action(this, subMesh, batch, effect);\n }\n if (!drawWrapper || !effect) {\n if (oldCamera) {\n oldCamera.maxZ = oldCameraMaxZ;\n scene.updateTransformMatrix(true);\n }\n return this;\n }\n const effectiveMesh = effectiveMeshReplacement || this;\n let sideOrientation;\n if (!instanceDataStorage.isFrozen && (this._internalMeshDataInfo._effectiveMaterial.backFaceCulling || this.overrideMaterialSideOrientation !== null)) {\n const mainDeterminant = effectiveMesh._getWorldMatrixDeterminant();\n sideOrientation = this.overrideMaterialSideOrientation;\n if (sideOrientation == null) {\n sideOrientation = this._internalMeshDataInfo._effectiveMaterial.sideOrientation;\n }\n if (mainDeterminant < 0) {\n sideOrientation = sideOrientation === Material.ClockWiseSideOrientation ? Material.CounterClockWiseSideOrientation : Material.ClockWiseSideOrientation;\n }\n instanceDataStorage.sideOrientation = sideOrientation;\n }\n else {\n sideOrientation = instanceDataStorage.sideOrientation;\n }\n const reverse = this._internalMeshDataInfo._effectiveMaterial._preBind(drawWrapper, sideOrientation);\n if (this._internalMeshDataInfo._effectiveMaterial.forceDepthWrite) {\n engine.setDepthWrite(true);\n }\n // Bind\n const effectiveMaterial = this._internalMeshDataInfo._effectiveMaterial;\n const fillMode = effectiveMaterial.fillMode;\n if (this._internalMeshDataInfo._onBeforeBindObservable) {\n this._internalMeshDataInfo._onBeforeBindObservable.notifyObservers(this);\n }\n if (!hardwareInstancedRendering) {\n // Binding will be done later because we need to add more info to the VB\n this._bind(subMesh, effect, fillMode, false);\n }\n const world = effectiveMesh.getWorldMatrix();\n if (effectiveMaterial._storeEffectOnSubMeshes) {\n effectiveMaterial.bindForSubMesh(world, this, subMesh);\n }\n else {\n effectiveMaterial.bind(world, this);\n }\n if (!effectiveMaterial.backFaceCulling && effectiveMaterial.separateCullingPass) {\n engine.setState(true, effectiveMaterial.zOffset, false, !reverse, effectiveMaterial.cullBackFaces, effectiveMaterial.stencil, effectiveMaterial.zOffsetUnits);\n this._processRendering(this, subMesh, effect, fillMode, batch, hardwareInstancedRendering, this._onBeforeDraw, this._internalMeshDataInfo._effectiveMaterial);\n engine.setState(true, effectiveMaterial.zOffset, false, reverse, effectiveMaterial.cullBackFaces, effectiveMaterial.stencil, effectiveMaterial.zOffsetUnits);\n if (this._internalMeshDataInfo._onBetweenPassObservable) {\n this._internalMeshDataInfo._onBetweenPassObservable.notifyObservers(subMesh);\n }\n }\n // Draw\n this._processRendering(this, subMesh, effect, fillMode, batch, hardwareInstancedRendering, this._onBeforeDraw, this._internalMeshDataInfo._effectiveMaterial);\n // Unbind\n this._internalMeshDataInfo._effectiveMaterial.unbind();\n for (const step of scene._afterRenderingMeshStage) {\n step.action(this, subMesh, batch, effect);\n }\n if (this._internalMeshDataInfo._onAfterRenderObservable) {\n this._internalMeshDataInfo._onAfterRenderObservable.notifyObservers(this);\n }\n if (oldCamera) {\n oldCamera.maxZ = oldCameraMaxZ;\n scene.updateTransformMatrix(true);\n }\n if (scene.performancePriority === ScenePerformancePriority.Aggressive && !instanceDataStorage.isFrozen) {\n this._freeze();\n }\n return this;\n }\n /**\n * Renormalize the mesh and patch it up if there are no weights\n * Similar to normalization by adding the weights compute the reciprocal and multiply all elements, this wil ensure that everything adds to 1.\n * However in the case of zero weights then we set just a single influence to 1.\n * We check in the function for extra's present and if so we use the normalizeSkinWeightsWithExtras rather than the FourWeights version.\n */\n cleanMatrixWeights() {\n if (this.isVerticesDataPresent(VertexBuffer.MatricesWeightsKind)) {\n if (this.isVerticesDataPresent(VertexBuffer.MatricesWeightsExtraKind)) {\n this._normalizeSkinWeightsAndExtra();\n }\n else {\n this._normalizeSkinFourWeights();\n }\n }\n }\n // faster 4 weight version.\n _normalizeSkinFourWeights() {\n const matricesWeights = this.getVerticesData(VertexBuffer.MatricesWeightsKind);\n const numWeights = matricesWeights.length;\n for (let a = 0; a < numWeights; a += 4) {\n // accumulate weights\n const t = matricesWeights[a] + matricesWeights[a + 1] + matricesWeights[a + 2] + matricesWeights[a + 3];\n // check for invalid weight and just set it to 1.\n if (t === 0) {\n matricesWeights[a] = 1;\n }\n else {\n // renormalize so everything adds to 1 use reciprocal\n const recip = 1 / t;\n matricesWeights[a] *= recip;\n matricesWeights[a + 1] *= recip;\n matricesWeights[a + 2] *= recip;\n matricesWeights[a + 3] *= recip;\n }\n }\n this.setVerticesData(VertexBuffer.MatricesWeightsKind, matricesWeights);\n }\n // handle special case of extra verts. (in theory gltf can handle 12 influences)\n _normalizeSkinWeightsAndExtra() {\n const matricesWeightsExtra = this.getVerticesData(VertexBuffer.MatricesWeightsExtraKind);\n const matricesWeights = this.getVerticesData(VertexBuffer.MatricesWeightsKind);\n const numWeights = matricesWeights.length;\n for (let a = 0; a < numWeights; a += 4) {\n // accumulate weights\n let t = matricesWeights[a] + matricesWeights[a + 1] + matricesWeights[a + 2] + matricesWeights[a + 3];\n t += matricesWeightsExtra[a] + matricesWeightsExtra[a + 1] + matricesWeightsExtra[a + 2] + matricesWeightsExtra[a + 3];\n // check for invalid weight and just set it to 1.\n if (t === 0) {\n matricesWeights[a] = 1;\n }\n else {\n // renormalize so everything adds to 1 use reciprocal\n const recip = 1 / t;\n matricesWeights[a] *= recip;\n matricesWeights[a + 1] *= recip;\n matricesWeights[a + 2] *= recip;\n matricesWeights[a + 3] *= recip;\n // same goes for extras\n matricesWeightsExtra[a] *= recip;\n matricesWeightsExtra[a + 1] *= recip;\n matricesWeightsExtra[a + 2] *= recip;\n matricesWeightsExtra[a + 3] *= recip;\n }\n }\n this.setVerticesData(VertexBuffer.MatricesWeightsKind, matricesWeights);\n this.setVerticesData(VertexBuffer.MatricesWeightsKind, matricesWeightsExtra);\n }\n /**\n * ValidateSkinning is used to determine that a mesh has valid skinning data along with skin metrics, if missing weights,\n * or not normalized it is returned as invalid mesh the string can be used for console logs, or on screen messages to let\n * the user know there was an issue with importing the mesh\n * @returns a validation object with skinned, valid and report string\n */\n validateSkinning() {\n const matricesWeightsExtra = this.getVerticesData(VertexBuffer.MatricesWeightsExtraKind);\n const matricesWeights = this.getVerticesData(VertexBuffer.MatricesWeightsKind);\n if (matricesWeights === null || this.skeleton == null) {\n return { skinned: false, valid: true, report: \"not skinned\" };\n }\n const numWeights = matricesWeights.length;\n let numberNotSorted = 0;\n let missingWeights = 0;\n let maxUsedWeights = 0;\n let numberNotNormalized = 0;\n const numInfluences = matricesWeightsExtra === null ? 4 : 8;\n const usedWeightCounts = new Array();\n for (let a = 0; a <= numInfluences; a++) {\n usedWeightCounts[a] = 0;\n }\n const toleranceEpsilon = 0.001;\n for (let a = 0; a < numWeights; a += 4) {\n let lastWeight = matricesWeights[a];\n let t = lastWeight;\n let usedWeights = t === 0 ? 0 : 1;\n for (let b = 1; b < numInfluences; b++) {\n const d = b < 4 ? matricesWeights[a + b] : matricesWeightsExtra[a + b - 4];\n if (d > lastWeight) {\n numberNotSorted++;\n }\n if (d !== 0) {\n usedWeights++;\n }\n t += d;\n lastWeight = d;\n }\n // count the buffer weights usage\n usedWeightCounts[usedWeights]++;\n // max influences\n if (usedWeights > maxUsedWeights) {\n maxUsedWeights = usedWeights;\n }\n // check for invalid weight and just set it to 1.\n if (t === 0) {\n missingWeights++;\n }\n else {\n // renormalize so everything adds to 1 use reciprocal\n const recip = 1 / t;\n let tolerance = 0;\n for (let b = 0; b < numInfluences; b++) {\n if (b < 4) {\n tolerance += Math.abs(matricesWeights[a + b] - matricesWeights[a + b] * recip);\n }\n else {\n tolerance += Math.abs(matricesWeightsExtra[a + b - 4] - matricesWeightsExtra[a + b - 4] * recip);\n }\n }\n // arbitrary epsilon value for dictating not normalized\n if (tolerance > toleranceEpsilon) {\n numberNotNormalized++;\n }\n }\n }\n // validate bone indices are in range of the skeleton\n const numBones = this.skeleton.bones.length;\n const matricesIndices = this.getVerticesData(VertexBuffer.MatricesIndicesKind);\n const matricesIndicesExtra = this.getVerticesData(VertexBuffer.MatricesIndicesExtraKind);\n let numBadBoneIndices = 0;\n for (let a = 0; a < numWeights; a += 4) {\n for (let b = 0; b < numInfluences; b++) {\n const index = b < 4 ? matricesIndices[a + b] : matricesIndicesExtra[a + b - 4];\n if (index >= numBones || index < 0) {\n numBadBoneIndices++;\n }\n }\n }\n // log mesh stats\n const output = \"Number of Weights = \" +\n numWeights / 4 +\n \"\\nMaximum influences = \" +\n maxUsedWeights +\n \"\\nMissing Weights = \" +\n missingWeights +\n \"\\nNot Sorted = \" +\n numberNotSorted +\n \"\\nNot Normalized = \" +\n numberNotNormalized +\n \"\\nWeightCounts = [\" +\n usedWeightCounts +\n \"]\" +\n \"\\nNumber of bones = \" +\n numBones +\n \"\\nBad Bone Indices = \" +\n numBadBoneIndices;\n return { skinned: true, valid: missingWeights === 0 && numberNotNormalized === 0 && numBadBoneIndices === 0, report: output };\n }\n /** @internal */\n _checkDelayState() {\n const scene = this.getScene();\n if (this._geometry) {\n this._geometry.load(scene);\n }\n else if (this.delayLoadState === 4) {\n this.delayLoadState = 2;\n this._queueLoad(scene);\n }\n return this;\n }\n _queueLoad(scene) {\n scene.addPendingData(this);\n const getBinaryData = this.delayLoadingFile.indexOf(\".babylonbinarymeshdata\") !== -1;\n Tools.LoadFile(this.delayLoadingFile, (data) => {\n if (data instanceof ArrayBuffer) {\n this._delayLoadingFunction(data, this);\n }\n else {\n this._delayLoadingFunction(JSON.parse(data), this);\n }\n this.instances.forEach((instance) => {\n instance.refreshBoundingInfo();\n instance._syncSubMeshes();\n });\n this.delayLoadState = 1;\n scene.removePendingData(this);\n }, () => { }, scene.offlineProvider, getBinaryData);\n return this;\n }\n /**\n * Returns `true` if the mesh is within the frustum defined by the passed array of planes.\n * A mesh is in the frustum if its bounding box intersects the frustum\n * @param frustumPlanes defines the frustum to test\n * @returns true if the mesh is in the frustum planes\n */\n isInFrustum(frustumPlanes) {\n if (this.delayLoadState === 2) {\n return false;\n }\n if (!super.isInFrustum(frustumPlanes)) {\n return false;\n }\n this._checkDelayState();\n return true;\n }\n /**\n * Sets the mesh material by the material or multiMaterial `id` property\n * @param id is a string identifying the material or the multiMaterial\n * @returns the current mesh\n */\n setMaterialById(id) {\n const materials = this.getScene().materials;\n let index;\n for (index = materials.length - 1; index > -1; index--) {\n if (materials[index].id === id) {\n this.material = materials[index];\n return this;\n }\n }\n // Multi\n const multiMaterials = this.getScene().multiMaterials;\n for (index = multiMaterials.length - 1; index > -1; index--) {\n if (multiMaterials[index].id === id) {\n this.material = multiMaterials[index];\n return this;\n }\n }\n return this;\n }\n /**\n * Returns as a new array populated with the mesh material and/or skeleton, if any.\n * @returns an array of IAnimatable\n */\n getAnimatables() {\n const results = new Array();\n if (this.material) {\n results.push(this.material);\n }\n if (this.skeleton) {\n results.push(this.skeleton);\n }\n return results;\n }\n /**\n * Modifies the mesh geometry according to the passed transformation matrix.\n * This method returns nothing, but it really modifies the mesh even if it's originally not set as updatable.\n * The mesh normals are modified using the same transformation.\n * Note that, under the hood, this method sets a new VertexBuffer each call.\n * @param transform defines the transform matrix to use\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/center_origin/bakingTransforms\n * @returns the current mesh\n */\n bakeTransformIntoVertices(transform) {\n // Position\n if (!this.isVerticesDataPresent(VertexBuffer.PositionKind)) {\n return this;\n }\n const submeshes = this.subMeshes.splice(0);\n this._resetPointsArrayCache();\n let data = this.getVerticesData(VertexBuffer.PositionKind);\n const temp = Vector3.Zero();\n let index;\n for (index = 0; index < data.length; index += 3) {\n Vector3.TransformCoordinatesFromFloatsToRef(data[index], data[index + 1], data[index + 2], transform, temp).toArray(data, index);\n }\n this.setVerticesData(VertexBuffer.PositionKind, data, this.getVertexBuffer(VertexBuffer.PositionKind).isUpdatable());\n // Normals\n if (this.isVerticesDataPresent(VertexBuffer.NormalKind)) {\n data = this.getVerticesData(VertexBuffer.NormalKind);\n for (index = 0; index < data.length; index += 3) {\n Vector3.TransformNormalFromFloatsToRef(data[index], data[index + 1], data[index + 2], transform, temp)\n .normalize()\n .toArray(data, index);\n }\n this.setVerticesData(VertexBuffer.NormalKind, data, this.getVertexBuffer(VertexBuffer.NormalKind).isUpdatable());\n }\n // flip faces?\n if (transform.determinant() < 0) {\n this.flipFaces();\n }\n // Restore submeshes\n this.releaseSubMeshes();\n this.subMeshes = submeshes;\n return this;\n }\n /**\n * Modifies the mesh geometry according to its own current World Matrix.\n * The mesh World Matrix is then reset.\n * This method returns nothing but really modifies the mesh even if it's originally not set as updatable.\n * Note that, under the hood, this method sets a new VertexBuffer each call.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/center_origin/bakingTransforms\n * @param bakeIndependentlyOfChildren indicates whether to preserve all child nodes' World Matrix during baking\n * @returns the current mesh\n */\n bakeCurrentTransformIntoVertices(bakeIndependentlyOfChildren = true) {\n this.bakeTransformIntoVertices(this.computeWorldMatrix(true));\n this.resetLocalMatrix(bakeIndependentlyOfChildren);\n return this;\n }\n // Cache\n /** @internal */\n get _positions() {\n if (this._internalAbstractMeshDataInfo._positions) {\n return this._internalAbstractMeshDataInfo._positions;\n }\n if (this._geometry) {\n return this._geometry._positions;\n }\n return null;\n }\n /** @internal */\n _resetPointsArrayCache() {\n if (this._geometry) {\n this._geometry._resetPointsArrayCache();\n }\n return this;\n }\n /** @internal */\n _generatePointsArray() {\n if (this._geometry) {\n return this._geometry._generatePointsArray();\n }\n return false;\n }\n /**\n * Returns a new Mesh object generated from the current mesh properties.\n * This method must not get confused with createInstance()\n * @param name is a string, the name given to the new mesh\n * @param newParent can be any Node object (default `null`)\n * @param doNotCloneChildren allows/denies the recursive cloning of the original mesh children if any (default `false`)\n * @param clonePhysicsImpostor allows/denies the cloning in the same time of the original mesh `body` used by the physics engine, if any (default `true`)\n * @returns a new mesh\n */\n clone(name = \"\", newParent = null, doNotCloneChildren, clonePhysicsImpostor = true) {\n return new Mesh(name, this.getScene(), newParent, this, doNotCloneChildren, clonePhysicsImpostor);\n }\n /**\n * Releases resources associated with this mesh.\n * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default)\n * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default)\n */\n dispose(doNotRecurse, disposeMaterialAndTextures = false) {\n this.morphTargetManager = null;\n if (this._geometry) {\n this._geometry.releaseForMesh(this, true);\n }\n const internalDataInfo = this._internalMeshDataInfo;\n if (internalDataInfo._onBeforeDrawObservable) {\n internalDataInfo._onBeforeDrawObservable.clear();\n }\n if (internalDataInfo._onBeforeBindObservable) {\n internalDataInfo._onBeforeBindObservable.clear();\n }\n if (internalDataInfo._onBeforeRenderObservable) {\n internalDataInfo._onBeforeRenderObservable.clear();\n }\n if (internalDataInfo._onAfterRenderObservable) {\n internalDataInfo._onAfterRenderObservable.clear();\n }\n if (internalDataInfo._onBetweenPassObservable) {\n internalDataInfo._onBetweenPassObservable.clear();\n }\n // Sources\n if (this._scene.useClonedMeshMap) {\n if (internalDataInfo.meshMap) {\n for (const uniqueId in internalDataInfo.meshMap) {\n const mesh = internalDataInfo.meshMap[uniqueId];\n if (mesh) {\n mesh._internalMeshDataInfo._source = null;\n internalDataInfo.meshMap[uniqueId] = undefined;\n }\n }\n }\n if (internalDataInfo._source && internalDataInfo._source._internalMeshDataInfo.meshMap) {\n internalDataInfo._source._internalMeshDataInfo.meshMap[this.uniqueId] = undefined;\n }\n }\n else {\n const meshes = this.getScene().meshes;\n for (const abstractMesh of meshes) {\n const mesh = abstractMesh;\n if (mesh._internalMeshDataInfo && mesh._internalMeshDataInfo._source && mesh._internalMeshDataInfo._source === this) {\n mesh._internalMeshDataInfo._source = null;\n }\n }\n }\n internalDataInfo._source = null;\n this._instanceDataStorage.visibleInstances = {};\n // Instances\n this._disposeInstanceSpecificData();\n // Thin instances\n this._disposeThinInstanceSpecificData();\n if (this._internalMeshDataInfo._checkReadinessObserver) {\n this._scene.onBeforeRenderObservable.remove(this._internalMeshDataInfo._checkReadinessObserver);\n }\n super.dispose(doNotRecurse, disposeMaterialAndTextures);\n }\n /** @internal */\n _disposeInstanceSpecificData() {\n // Do nothing\n }\n /** @internal */\n _disposeThinInstanceSpecificData() {\n // Do nothing\n }\n /** @internal */\n _invalidateInstanceVertexArrayObject() {\n // Do nothing\n }\n /**\n * Modifies the mesh geometry according to a displacement map.\n * A displacement map is a colored image. Each pixel color value (actually a gradient computed from red, green, blue values) will give the displacement to apply to each mesh vertex.\n * The mesh must be set as updatable. Its internal geometry is directly modified, no new buffer are allocated.\n * @param url is a string, the URL from the image file is to be downloaded.\n * @param minHeight is the lower limit of the displacement.\n * @param maxHeight is the upper limit of the displacement.\n * @param onSuccess is an optional Javascript function to be called just after the mesh is modified. It is passed the modified mesh and must return nothing.\n * @param uvOffset is an optional vector2 used to offset UV.\n * @param uvScale is an optional vector2 used to scale UV.\n * @param forceUpdate defines whether or not to force an update of the generated buffers. This is useful to apply on a deserialized model for instance.\n * @returns the Mesh.\n */\n applyDisplacementMap(url, minHeight, maxHeight, onSuccess, uvOffset, uvScale, forceUpdate = false) {\n const scene = this.getScene();\n const onload = (img) => {\n // Getting height map data\n const heightMapWidth = img.width;\n const heightMapHeight = img.height;\n const canvas = this.getEngine().createCanvas(heightMapWidth, heightMapHeight);\n const context = canvas.getContext(\"2d\");\n context.drawImage(img, 0, 0);\n // Create VertexData from map data\n //Cast is due to wrong definition in lib.d.ts from ts 1.3 - https://github.com/Microsoft/TypeScript/issues/949\n const buffer = context.getImageData(0, 0, heightMapWidth, heightMapHeight).data;\n this.applyDisplacementMapFromBuffer(buffer, heightMapWidth, heightMapHeight, minHeight, maxHeight, uvOffset, uvScale, forceUpdate);\n //execute success callback, if set\n if (onSuccess) {\n onSuccess(this);\n }\n };\n Tools.LoadImage(url, onload, () => { }, scene.offlineProvider);\n return this;\n }\n /**\n * Modifies the mesh geometry according to a displacementMap buffer.\n * A displacement map is a colored image. Each pixel color value (actually a gradient computed from red, green, blue values) will give the displacement to apply to each mesh vertex.\n * The mesh must be set as updatable. Its internal geometry is directly modified, no new buffer are allocated.\n * @param buffer is a `Uint8Array` buffer containing series of `Uint8` lower than 255, the red, green, blue and alpha values of each successive pixel.\n * @param heightMapWidth is the width of the buffer image.\n * @param heightMapHeight is the height of the buffer image.\n * @param minHeight is the lower limit of the displacement.\n * @param maxHeight is the upper limit of the displacement.\n * @param uvOffset is an optional vector2 used to offset UV.\n * @param uvScale is an optional vector2 used to scale UV.\n * @param forceUpdate defines whether or not to force an update of the generated buffers. This is useful to apply on a deserialized model for instance.\n * @returns the Mesh.\n */\n applyDisplacementMapFromBuffer(buffer, heightMapWidth, heightMapHeight, minHeight, maxHeight, uvOffset, uvScale, forceUpdate = false) {\n if (!this.isVerticesDataPresent(VertexBuffer.PositionKind) || !this.isVerticesDataPresent(VertexBuffer.NormalKind) || !this.isVerticesDataPresent(VertexBuffer.UVKind)) {\n Logger.Warn(\"Cannot call applyDisplacementMap: Given mesh is not complete. Position, Normal or UV are missing\");\n return this;\n }\n const positions = this.getVerticesData(VertexBuffer.PositionKind, true, true);\n const normals = this.getVerticesData(VertexBuffer.NormalKind);\n const uvs = this.getVerticesData(VertexBuffer.UVKind);\n let position = Vector3.Zero();\n const normal = Vector3.Zero();\n const uv = Vector2.Zero();\n uvOffset = uvOffset || Vector2.Zero();\n uvScale = uvScale || new Vector2(1, 1);\n for (let index = 0; index < positions.length; index += 3) {\n Vector3.FromArrayToRef(positions, index, position);\n Vector3.FromArrayToRef(normals, index, normal);\n Vector2.FromArrayToRef(uvs, (index / 3) * 2, uv);\n // Compute height\n const u = (Math.abs(uv.x * uvScale.x + (uvOffset.x % 1)) * (heightMapWidth - 1)) % heightMapWidth | 0;\n const v = (Math.abs(uv.y * uvScale.y + (uvOffset.y % 1)) * (heightMapHeight - 1)) % heightMapHeight | 0;\n const pos = (u + v * heightMapWidth) * 4;\n const r = buffer[pos] / 255.0;\n const g = buffer[pos + 1] / 255.0;\n const b = buffer[pos + 2] / 255.0;\n const gradient = r * 0.3 + g * 0.59 + b * 0.11;\n normal.normalize();\n normal.scaleInPlace(minHeight + (maxHeight - minHeight) * gradient);\n position = position.add(normal);\n position.toArray(positions, index);\n }\n VertexData.ComputeNormals(positions, this.getIndices(), normals);\n if (forceUpdate) {\n this.setVerticesData(VertexBuffer.PositionKind, positions);\n this.setVerticesData(VertexBuffer.NormalKind, normals);\n this.setVerticesData(VertexBuffer.UVKind, uvs);\n }\n else {\n this.updateVerticesData(VertexBuffer.PositionKind, positions);\n this.updateVerticesData(VertexBuffer.NormalKind, normals);\n }\n return this;\n }\n _getFlattenedNormals(indices, positions) {\n const normals = new Float32Array(indices.length * 3);\n let normalsCount = 0;\n // Decide if normals should be flipped\n const flipNormalGeneration = this.overrideMaterialSideOrientation ===\n (this._scene.useRightHandedSystem ? 1 : 0);\n // Generate new normals\n for (let index = 0; index < indices.length; index += 3) {\n const p1 = Vector3.FromArray(positions, indices[index] * 3);\n const p2 = Vector3.FromArray(positions, indices[index + 1] * 3);\n const p3 = Vector3.FromArray(positions, indices[index + 2] * 3);\n const p1p2 = p1.subtract(p2);\n const p3p2 = p3.subtract(p2);\n const normal = Vector3.Normalize(Vector3.Cross(p1p2, p3p2));\n if (flipNormalGeneration) {\n normal.scaleInPlace(-1);\n }\n // Store same normals for every vertex\n for (let localIndex = 0; localIndex < 3; localIndex++) {\n normals[normalsCount++] = normal.x;\n normals[normalsCount++] = normal.y;\n normals[normalsCount++] = normal.z;\n }\n }\n return normals;\n }\n _convertToUnIndexedMesh(flattenNormals = false) {\n const kinds = this.getVerticesDataKinds();\n const indices = this.getIndices();\n const data = {};\n const separateVertices = (data, stride) => {\n const newData = new Float32Array(indices.length * stride);\n let count = 0;\n for (let index = 0; index < indices.length; index++) {\n for (let offset = 0; offset < stride; offset++) {\n newData[count++] = data[indices[index] * stride + offset];\n }\n }\n return newData;\n };\n // Save previous submeshes\n const previousSubmeshes = this.geometry ? this.subMeshes.slice(0) : [];\n // Cache vertex data\n for (const kind of kinds) {\n data[kind] = this.getVerticesData(kind);\n }\n // Update vertex data\n for (const kind of kinds) {\n const vertexBuffer = this.getVertexBuffer(kind);\n const stride = vertexBuffer.getStrideSize();\n if (flattenNormals && kind === VertexBuffer.NormalKind) {\n const normals = this._getFlattenedNormals(indices, data[VertexBuffer.PositionKind]);\n this.setVerticesData(VertexBuffer.NormalKind, normals, vertexBuffer.isUpdatable(), stride);\n }\n else {\n this.setVerticesData(kind, separateVertices(data[kind], stride), vertexBuffer.isUpdatable(), stride);\n }\n }\n // Update morph targets\n if (this.morphTargetManager) {\n for (let targetIndex = 0; targetIndex < this.morphTargetManager.numTargets; targetIndex++) {\n const target = this.morphTargetManager.getTarget(targetIndex);\n const positions = target.getPositions();\n target.setPositions(separateVertices(positions, 3));\n const normals = target.getNormals();\n if (normals) {\n target.setNormals(flattenNormals ? this._getFlattenedNormals(indices, positions) : separateVertices(normals, 3));\n }\n const tangents = target.getTangents();\n if (tangents) {\n target.setTangents(separateVertices(tangents, 3));\n }\n const uvs = target.getUVs();\n if (uvs) {\n target.setUVs(separateVertices(uvs, 2));\n }\n }\n this.morphTargetManager.synchronize();\n }\n // Update indices\n for (let index = 0; index < indices.length; index++) {\n indices[index] = index;\n }\n this.setIndices(indices);\n this._unIndexed = true;\n // Update submeshes\n this.releaseSubMeshes();\n for (const previousOne of previousSubmeshes) {\n SubMesh.AddToMesh(previousOne.materialIndex, previousOne.indexStart, previousOne.indexCount, previousOne.indexStart, previousOne.indexCount, this);\n }\n this.synchronizeInstances();\n return this;\n }\n /**\n * Modify the mesh to get a flat shading rendering.\n * This means each mesh facet will then have its own normals. Usually new vertices are added in the mesh geometry to get this result.\n * Warning : the mesh is really modified even if not set originally as updatable and, under the hood, a new VertexBuffer is allocated.\n * @returns current mesh\n */\n convertToFlatShadedMesh() {\n return this._convertToUnIndexedMesh(true);\n }\n /**\n * This method removes all the mesh indices and add new vertices (duplication) in order to unfold facets into buffers.\n * In other words, more vertices, no more indices and a single bigger VBO.\n * The mesh is really modified even if not set originally as updatable. Under the hood, a new VertexBuffer is allocated.\n * @returns current mesh\n */\n convertToUnIndexedMesh() {\n return this._convertToUnIndexedMesh();\n }\n /**\n * Inverses facet orientations.\n * Warning : the mesh is really modified even if not set originally as updatable. A new VertexBuffer is created under the hood each call.\n * @param flipNormals will also inverts the normals\n * @returns current mesh\n */\n flipFaces(flipNormals = false) {\n const vertex_data = VertexData.ExtractFromMesh(this);\n let i;\n if (flipNormals && this.isVerticesDataPresent(VertexBuffer.NormalKind) && vertex_data.normals) {\n for (i = 0; i < vertex_data.normals.length; i++) {\n vertex_data.normals[i] *= -1;\n }\n }\n if (vertex_data.indices) {\n let temp;\n for (i = 0; i < vertex_data.indices.length; i += 3) {\n // reassign indices\n temp = vertex_data.indices[i + 1];\n vertex_data.indices[i + 1] = vertex_data.indices[i + 2];\n vertex_data.indices[i + 2] = temp;\n }\n }\n vertex_data.applyToMesh(this, this.isVertexBufferUpdatable(VertexBuffer.PositionKind));\n return this;\n }\n /**\n * Increase the number of facets and hence vertices in a mesh\n * Vertex normals are interpolated from existing vertex normals\n * Warning : the mesh is really modified even if not set originally as updatable. A new VertexBuffer is created under the hood each call.\n * @param numberPerEdge the number of new vertices to add to each edge of a facet, optional default 1\n */\n increaseVertices(numberPerEdge = 1) {\n const vertex_data = VertexData.ExtractFromMesh(this);\n const currentIndices = vertex_data.indices && !Array.isArray(vertex_data.indices) && Array.from ? Array.from(vertex_data.indices) : vertex_data.indices;\n const positions = vertex_data.positions && !Array.isArray(vertex_data.positions) && Array.from ? Array.from(vertex_data.positions) : vertex_data.positions;\n const uvs = vertex_data.uvs && !Array.isArray(vertex_data.uvs) && Array.from ? Array.from(vertex_data.uvs) : vertex_data.uvs;\n const normals = vertex_data.normals && !Array.isArray(vertex_data.normals) && Array.from ? Array.from(vertex_data.normals) : vertex_data.normals;\n if (!currentIndices || !positions) {\n Logger.Warn(\"Couldn't increase number of vertices : VertexData must contain at least indices and positions\");\n }\n else {\n vertex_data.indices = currentIndices;\n vertex_data.positions = positions;\n if (uvs) {\n vertex_data.uvs = uvs;\n }\n if (normals) {\n vertex_data.normals = normals;\n }\n const segments = numberPerEdge + 1; //segments per current facet edge, become sides of new facets\n const tempIndices = new Array();\n for (let i = 0; i < segments + 1; i++) {\n tempIndices[i] = new Array();\n }\n let a; //vertex index of one end of a side\n let b; //vertex index of other end of the side\n const deltaPosition = new Vector3(0, 0, 0);\n const deltaNormal = new Vector3(0, 0, 0);\n const deltaUV = new Vector2(0, 0);\n const indices = new Array();\n const vertexIndex = new Array();\n const side = new Array();\n let len;\n let positionPtr = positions.length;\n let uvPtr;\n if (uvs) {\n uvPtr = uvs.length;\n }\n let normalsPtr;\n if (normals) {\n normalsPtr = normals.length;\n }\n for (let i = 0; i < currentIndices.length; i += 3) {\n vertexIndex[0] = currentIndices[i];\n vertexIndex[1] = currentIndices[i + 1];\n vertexIndex[2] = currentIndices[i + 2];\n for (let j = 0; j < 3; j++) {\n a = vertexIndex[j];\n b = vertexIndex[(j + 1) % 3];\n if (side[a] === undefined && side[b] === undefined) {\n side[a] = new Array();\n side[b] = new Array();\n }\n else {\n if (side[a] === undefined) {\n side[a] = new Array();\n }\n if (side[b] === undefined) {\n side[b] = new Array();\n }\n }\n if (side[a][b] === undefined && side[b][a] === undefined) {\n side[a][b] = [];\n deltaPosition.x = (positions[3 * b] - positions[3 * a]) / segments;\n deltaPosition.y = (positions[3 * b + 1] - positions[3 * a + 1]) / segments;\n deltaPosition.z = (positions[3 * b + 2] - positions[3 * a + 2]) / segments;\n if (normals) {\n deltaNormal.x = (normals[3 * b] - normals[3 * a]) / segments;\n deltaNormal.y = (normals[3 * b + 1] - normals[3 * a + 1]) / segments;\n deltaNormal.z = (normals[3 * b + 2] - normals[3 * a + 2]) / segments;\n }\n if (uvs) {\n deltaUV.x = (uvs[2 * b] - uvs[2 * a]) / segments;\n deltaUV.y = (uvs[2 * b + 1] - uvs[2 * a + 1]) / segments;\n }\n side[a][b].push(a);\n for (let k = 1; k < segments; k++) {\n side[a][b].push(positions.length / 3);\n positions[positionPtr++] = positions[3 * a] + k * deltaPosition.x;\n positions[positionPtr++] = positions[3 * a + 1] + k * deltaPosition.y;\n positions[positionPtr++] = positions[3 * a + 2] + k * deltaPosition.z;\n if (normals) {\n normals[normalsPtr++] = normals[3 * a] + k * deltaNormal.x;\n normals[normalsPtr++] = normals[3 * a + 1] + k * deltaNormal.y;\n normals[normalsPtr++] = normals[3 * a + 2] + k * deltaNormal.z;\n }\n if (uvs) {\n uvs[uvPtr++] = uvs[2 * a] + k * deltaUV.x;\n uvs[uvPtr++] = uvs[2 * a + 1] + k * deltaUV.y;\n }\n }\n side[a][b].push(b);\n side[b][a] = new Array();\n len = side[a][b].length;\n for (let idx = 0; idx < len; idx++) {\n side[b][a][idx] = side[a][b][len - 1 - idx];\n }\n }\n }\n //Calculate positions, normals and uvs of new internal vertices\n tempIndices[0][0] = currentIndices[i];\n tempIndices[1][0] = side[currentIndices[i]][currentIndices[i + 1]][1];\n tempIndices[1][1] = side[currentIndices[i]][currentIndices[i + 2]][1];\n for (let k = 2; k < segments; k++) {\n tempIndices[k][0] = side[currentIndices[i]][currentIndices[i + 1]][k];\n tempIndices[k][k] = side[currentIndices[i]][currentIndices[i + 2]][k];\n deltaPosition.x = (positions[3 * tempIndices[k][k]] - positions[3 * tempIndices[k][0]]) / k;\n deltaPosition.y = (positions[3 * tempIndices[k][k] + 1] - positions[3 * tempIndices[k][0] + 1]) / k;\n deltaPosition.z = (positions[3 * tempIndices[k][k] + 2] - positions[3 * tempIndices[k][0] + 2]) / k;\n if (normals) {\n deltaNormal.x = (normals[3 * tempIndices[k][k]] - normals[3 * tempIndices[k][0]]) / k;\n deltaNormal.y = (normals[3 * tempIndices[k][k] + 1] - normals[3 * tempIndices[k][0] + 1]) / k;\n deltaNormal.z = (normals[3 * tempIndices[k][k] + 2] - normals[3 * tempIndices[k][0] + 2]) / k;\n }\n if (uvs) {\n deltaUV.x = (uvs[2 * tempIndices[k][k]] - uvs[2 * tempIndices[k][0]]) / k;\n deltaUV.y = (uvs[2 * tempIndices[k][k] + 1] - uvs[2 * tempIndices[k][0] + 1]) / k;\n }\n for (let j = 1; j < k; j++) {\n tempIndices[k][j] = positions.length / 3;\n positions[positionPtr++] = positions[3 * tempIndices[k][0]] + j * deltaPosition.x;\n positions[positionPtr++] = positions[3 * tempIndices[k][0] + 1] + j * deltaPosition.y;\n positions[positionPtr++] = positions[3 * tempIndices[k][0] + 2] + j * deltaPosition.z;\n if (normals) {\n normals[normalsPtr++] = normals[3 * tempIndices[k][0]] + j * deltaNormal.x;\n normals[normalsPtr++] = normals[3 * tempIndices[k][0] + 1] + j * deltaNormal.y;\n normals[normalsPtr++] = normals[3 * tempIndices[k][0] + 2] + j * deltaNormal.z;\n }\n if (uvs) {\n uvs[uvPtr++] = uvs[2 * tempIndices[k][0]] + j * deltaUV.x;\n uvs[uvPtr++] = uvs[2 * tempIndices[k][0] + 1] + j * deltaUV.y;\n }\n }\n }\n tempIndices[segments] = side[currentIndices[i + 1]][currentIndices[i + 2]];\n // reform indices\n indices.push(tempIndices[0][0], tempIndices[1][0], tempIndices[1][1]);\n for (let k = 1; k < segments; k++) {\n let j;\n for (j = 0; j < k; j++) {\n indices.push(tempIndices[k][j], tempIndices[k + 1][j], tempIndices[k + 1][j + 1]);\n indices.push(tempIndices[k][j], tempIndices[k + 1][j + 1], tempIndices[k][j + 1]);\n }\n indices.push(tempIndices[k][j], tempIndices[k + 1][j], tempIndices[k + 1][j + 1]);\n }\n }\n vertex_data.indices = indices;\n vertex_data.applyToMesh(this, this.isVertexBufferUpdatable(VertexBuffer.PositionKind));\n }\n }\n /**\n * Force adjacent facets to share vertices and remove any facets that have all vertices in a line\n * This will undo any application of covertToFlatShadedMesh\n * Warning : the mesh is really modified even if not set originally as updatable. A new VertexBuffer is created under the hood each call.\n */\n forceSharedVertices() {\n const vertex_data = VertexData.ExtractFromMesh(this);\n const currentUVs = vertex_data.uvs;\n const currentIndices = vertex_data.indices;\n const currentPositions = vertex_data.positions;\n const currentColors = vertex_data.colors;\n const currentMatrixIndices = vertex_data.matricesIndices;\n const currentMatrixWeights = vertex_data.matricesWeights;\n const currentMatrixIndicesExtra = vertex_data.matricesIndicesExtra;\n const currentMatrixWeightsExtra = vertex_data.matricesWeightsExtra;\n if (currentIndices === void 0 || currentPositions === void 0 || currentIndices === null || currentPositions === null) {\n Logger.Warn(\"VertexData contains empty entries\");\n }\n else {\n const positions = new Array();\n const indices = new Array();\n const uvs = new Array();\n const colors = new Array();\n const matrixIndices = new Array();\n const matrixWeights = new Array();\n const matrixIndicesExtra = new Array();\n const matrixWeightsExtra = new Array();\n let pstring = new Array(); //lists facet vertex positions (a,b,c) as string \"a|b|c\"\n let indexPtr = 0; // pointer to next available index value\n const uniquePositions = {}; // unique vertex positions\n let ptr; // pointer to element in uniquePositions\n let facet;\n for (let i = 0; i < currentIndices.length; i += 3) {\n facet = [currentIndices[i], currentIndices[i + 1], currentIndices[i + 2]]; //facet vertex indices\n pstring = new Array();\n for (let j = 0; j < 3; j++) {\n pstring[j] = \"\";\n for (let k = 0; k < 3; k++) {\n //small values make 0\n if (Math.abs(currentPositions[3 * facet[j] + k]) < 0.00000001) {\n currentPositions[3 * facet[j] + k] = 0;\n }\n pstring[j] += currentPositions[3 * facet[j] + k] + \"|\";\n }\n }\n //check facet vertices to see that none are repeated\n // do not process any facet that has a repeated vertex, ie is a line\n if (!(pstring[0] == pstring[1] || pstring[0] == pstring[2] || pstring[1] == pstring[2])) {\n //for each facet position check if already listed in uniquePositions\n // if not listed add to uniquePositions and set index pointer\n // if listed use its index in uniquePositions and new index pointer\n for (let j = 0; j < 3; j++) {\n ptr = uniquePositions[pstring[j]];\n if (ptr === undefined) {\n uniquePositions[pstring[j]] = indexPtr;\n ptr = indexPtr++;\n //not listed so add individual x, y, z coordinates to positions\n for (let k = 0; k < 3; k++) {\n positions.push(currentPositions[3 * facet[j] + k]);\n }\n if (currentColors !== null && currentColors !== void 0) {\n for (let k = 0; k < 4; k++) {\n colors.push(currentColors[4 * facet[j] + k]);\n }\n }\n if (currentUVs !== null && currentUVs !== void 0) {\n for (let k = 0; k < 2; k++) {\n uvs.push(currentUVs[2 * facet[j] + k]);\n }\n }\n if (currentMatrixIndices !== null && currentMatrixIndices !== void 0) {\n for (let k = 0; k < 4; k++) {\n matrixIndices.push(currentMatrixIndices[4 * facet[j] + k]);\n }\n }\n if (currentMatrixWeights !== null && currentMatrixWeights !== void 0) {\n for (let k = 0; k < 4; k++) {\n matrixWeights.push(currentMatrixWeights[4 * facet[j] + k]);\n }\n }\n if (currentMatrixIndicesExtra !== null && currentMatrixIndicesExtra !== void 0) {\n for (let k = 0; k < 4; k++) {\n matrixIndicesExtra.push(currentMatrixIndicesExtra[4 * facet[j] + k]);\n }\n }\n if (currentMatrixWeightsExtra !== null && currentMatrixWeightsExtra !== void 0) {\n for (let k = 0; k < 4; k++) {\n matrixWeightsExtra.push(currentMatrixWeightsExtra[4 * facet[j] + k]);\n }\n }\n }\n // add new index pointer to indices array\n indices.push(ptr);\n }\n }\n }\n const normals = new Array();\n VertexData.ComputeNormals(positions, indices, normals);\n //create new vertex data object and update\n vertex_data.positions = positions;\n vertex_data.indices = indices;\n vertex_data.normals = normals;\n if (currentUVs !== null && currentUVs !== void 0) {\n vertex_data.uvs = uvs;\n }\n if (currentColors !== null && currentColors !== void 0) {\n vertex_data.colors = colors;\n }\n if (currentMatrixIndices !== null && currentMatrixIndices !== void 0) {\n vertex_data.matricesIndices = matrixIndices;\n }\n if (currentMatrixWeights !== null && currentMatrixWeights !== void 0) {\n vertex_data.matricesWeights = matrixWeights;\n }\n if (currentMatrixIndicesExtra !== null && currentMatrixIndicesExtra !== void 0) {\n vertex_data.matricesIndicesExtra = matrixIndicesExtra;\n }\n if (currentMatrixWeights !== null && currentMatrixWeights !== void 0) {\n vertex_data.matricesWeightsExtra = matrixWeightsExtra;\n }\n vertex_data.applyToMesh(this, this.isVertexBufferUpdatable(VertexBuffer.PositionKind));\n }\n }\n // Instances\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/naming-convention\n static _instancedMeshFactory(name, mesh) {\n throw _WarnImport(\"InstancedMesh\");\n }\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n static _PhysicsImpostorParser(scene, physicObject, jsonObject) {\n throw _WarnImport(\"PhysicsImpostor\");\n }\n /**\n * Creates a new InstancedMesh object from the mesh model.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/copies/instances\n * @param name defines the name of the new instance\n * @returns a new InstancedMesh\n */\n createInstance(name) {\n return Mesh._instancedMeshFactory(name, this);\n }\n /**\n * Synchronises all the mesh instance submeshes to the current mesh submeshes, if any.\n * After this call, all the mesh instances have the same submeshes than the current mesh.\n * @returns the current mesh\n */\n synchronizeInstances() {\n for (let instanceIndex = 0; instanceIndex < this.instances.length; instanceIndex++) {\n const instance = this.instances[instanceIndex];\n instance._syncSubMeshes();\n }\n return this;\n }\n /**\n * Optimization of the mesh's indices, in case a mesh has duplicated vertices.\n * The function will only reorder the indices and will not remove unused vertices to avoid problems with submeshes.\n * This should be used together with the simplification to avoid disappearing triangles.\n * @param successCallback an optional success callback to be called after the optimization finished.\n * @returns the current mesh\n */\n optimizeIndices(successCallback) {\n const indices = this.getIndices();\n const positions = this.getVerticesData(VertexBuffer.PositionKind);\n if (!positions || !indices) {\n return this;\n }\n const vectorPositions = new Array();\n for (let pos = 0; pos < positions.length; pos = pos + 3) {\n vectorPositions.push(Vector3.FromArray(positions, pos));\n }\n const dupes = new Array();\n AsyncLoop.SyncAsyncForLoop(vectorPositions.length, 40, (iteration) => {\n const realPos = vectorPositions.length - 1 - iteration;\n const testedPosition = vectorPositions[realPos];\n for (let j = 0; j < realPos; ++j) {\n const againstPosition = vectorPositions[j];\n if (testedPosition.equals(againstPosition)) {\n dupes[realPos] = j;\n break;\n }\n }\n }, () => {\n for (let i = 0; i < indices.length; ++i) {\n indices[i] = dupes[indices[i]] || indices[i];\n }\n //indices are now reordered\n const originalSubMeshes = this.subMeshes.slice(0);\n this.setIndices(indices);\n this.subMeshes = originalSubMeshes;\n if (successCallback) {\n successCallback(this);\n }\n });\n return this;\n }\n /**\n * Serialize current mesh\n * @param serializationObject defines the object which will receive the serialization data\n */\n serialize(serializationObject = {}) {\n serializationObject.name = this.name;\n serializationObject.id = this.id;\n serializationObject.uniqueId = this.uniqueId;\n serializationObject.type = this.getClassName();\n if (Tags && Tags.HasTags(this)) {\n serializationObject.tags = Tags.GetTags(this);\n }\n serializationObject.position = this.position.asArray();\n if (this.rotationQuaternion) {\n serializationObject.rotationQuaternion = this.rotationQuaternion.asArray();\n }\n else if (this.rotation) {\n serializationObject.rotation = this.rotation.asArray();\n }\n serializationObject.scaling = this.scaling.asArray();\n if (this._postMultiplyPivotMatrix) {\n serializationObject.pivotMatrix = this.getPivotMatrix().asArray();\n }\n else {\n serializationObject.localMatrix = this.getPivotMatrix().asArray();\n }\n serializationObject.isEnabled = this.isEnabled(false);\n serializationObject.isVisible = this.isVisible;\n serializationObject.infiniteDistance = this.infiniteDistance;\n serializationObject.pickable = this.isPickable;\n serializationObject.receiveShadows = this.receiveShadows;\n serializationObject.billboardMode = this.billboardMode;\n serializationObject.visibility = this.visibility;\n serializationObject.checkCollisions = this.checkCollisions;\n serializationObject.isBlocker = this.isBlocker;\n serializationObject.overrideMaterialSideOrientation = this.overrideMaterialSideOrientation;\n // Parent\n if (this.parent) {\n this.parent._serializeAsParent(serializationObject);\n }\n // Geometry\n serializationObject.isUnIndexed = this.isUnIndexed;\n const geometry = this._geometry;\n if (geometry && this.subMeshes) {\n serializationObject.geometryUniqueId = geometry.uniqueId;\n serializationObject.geometryId = geometry.id;\n // SubMeshes\n serializationObject.subMeshes = [];\n for (let subIndex = 0; subIndex < this.subMeshes.length; subIndex++) {\n const subMesh = this.subMeshes[subIndex];\n serializationObject.subMeshes.push({\n materialIndex: subMesh.materialIndex,\n verticesStart: subMesh.verticesStart,\n verticesCount: subMesh.verticesCount,\n indexStart: subMesh.indexStart,\n indexCount: subMesh.indexCount,\n });\n }\n }\n // Material\n if (this.material) {\n if (!this.material.doNotSerialize) {\n serializationObject.materialUniqueId = this.material.uniqueId;\n serializationObject.materialId = this.material.id; // back compat\n }\n }\n else {\n this.material = null;\n serializationObject.materialUniqueId = this._scene.defaultMaterial.uniqueId;\n serializationObject.materialId = this._scene.defaultMaterial.id; // back compat\n }\n // Morph targets\n if (this.morphTargetManager) {\n serializationObject.morphTargetManagerId = this.morphTargetManager.uniqueId;\n }\n // Skeleton\n if (this.skeleton) {\n serializationObject.skeletonId = this.skeleton.id;\n serializationObject.numBoneInfluencers = this.numBoneInfluencers;\n }\n // Physics\n //TODO implement correct serialization for physics impostors.\n if (this.getScene()._getComponent(SceneComponentConstants.NAME_PHYSICSENGINE)) {\n const impostor = this.getPhysicsImpostor();\n if (impostor) {\n serializationObject.physicsMass = impostor.getParam(\"mass\");\n serializationObject.physicsFriction = impostor.getParam(\"friction\");\n serializationObject.physicsRestitution = impostor.getParam(\"mass\");\n serializationObject.physicsImpostor = impostor.type;\n }\n }\n // Metadata\n if (this.metadata) {\n serializationObject.metadata = this.metadata;\n }\n // Instances\n serializationObject.instances = [];\n for (let index = 0; index < this.instances.length; index++) {\n const instance = this.instances[index];\n if (instance.doNotSerialize) {\n continue;\n }\n const serializationInstance = {\n name: instance.name,\n id: instance.id,\n isEnabled: instance.isEnabled(false),\n isVisible: instance.isVisible,\n isPickable: instance.isPickable,\n checkCollisions: instance.checkCollisions,\n position: instance.position.asArray(),\n scaling: instance.scaling.asArray(),\n };\n if (instance.parent) {\n instance.parent._serializeAsParent(serializationInstance);\n }\n if (instance.rotationQuaternion) {\n serializationInstance.rotationQuaternion = instance.rotationQuaternion.asArray();\n }\n else if (instance.rotation) {\n serializationInstance.rotation = instance.rotation.asArray();\n }\n // Physics\n //TODO implement correct serialization for physics impostors.\n if (this.getScene()._getComponent(SceneComponentConstants.NAME_PHYSICSENGINE)) {\n const impostor = instance.getPhysicsImpostor();\n if (impostor) {\n serializationInstance.physicsMass = impostor.getParam(\"mass\");\n serializationInstance.physicsFriction = impostor.getParam(\"friction\");\n serializationInstance.physicsRestitution = impostor.getParam(\"mass\");\n serializationInstance.physicsImpostor = impostor.type;\n }\n }\n // Metadata\n if (instance.metadata) {\n serializationInstance.metadata = instance.metadata;\n }\n // Action Manager\n if (instance.actionManager) {\n serializationInstance.actions = instance.actionManager.serialize(instance.name);\n }\n serializationObject.instances.push(serializationInstance);\n // Animations\n SerializationHelper.AppendSerializedAnimations(instance, serializationInstance);\n serializationInstance.ranges = instance.serializeAnimationRanges();\n }\n // Thin instances\n if (this._thinInstanceDataStorage.instancesCount && this._thinInstanceDataStorage.matrixData) {\n serializationObject.thinInstances = {\n instancesCount: this._thinInstanceDataStorage.instancesCount,\n matrixData: Array.from(this._thinInstanceDataStorage.matrixData),\n matrixBufferSize: this._thinInstanceDataStorage.matrixBufferSize,\n enablePicking: this.thinInstanceEnablePicking,\n };\n if (this._userThinInstanceBuffersStorage) {\n const userThinInstance = {\n data: {},\n sizes: {},\n strides: {},\n };\n for (const kind in this._userThinInstanceBuffersStorage.data) {\n userThinInstance.data[kind] = Array.from(this._userThinInstanceBuffersStorage.data[kind]);\n userThinInstance.sizes[kind] = this._userThinInstanceBuffersStorage.sizes[kind];\n userThinInstance.strides[kind] = this._userThinInstanceBuffersStorage.strides[kind];\n }\n serializationObject.thinInstances.userThinInstance = userThinInstance;\n }\n }\n // Animations\n SerializationHelper.AppendSerializedAnimations(this, serializationObject);\n serializationObject.ranges = this.serializeAnimationRanges();\n // Layer mask\n serializationObject.layerMask = this.layerMask;\n // Alpha\n serializationObject.alphaIndex = this.alphaIndex;\n serializationObject.hasVertexAlpha = this.hasVertexAlpha;\n // Overlay\n serializationObject.overlayAlpha = this.overlayAlpha;\n serializationObject.overlayColor = this.overlayColor.asArray();\n serializationObject.renderOverlay = this.renderOverlay;\n // Fog\n serializationObject.applyFog = this.applyFog;\n // Action Manager\n if (this.actionManager) {\n serializationObject.actions = this.actionManager.serialize(this.name);\n }\n return serializationObject;\n }\n /** @internal */\n _syncGeometryWithMorphTargetManager() {\n if (!this.geometry) {\n return;\n }\n this._markSubMeshesAsAttributesDirty();\n const morphTargetManager = this._internalAbstractMeshDataInfo._morphTargetManager;\n if (morphTargetManager && morphTargetManager.vertexCount) {\n if (morphTargetManager.vertexCount !== this.getTotalVertices()) {\n Logger.Error(\"Mesh is incompatible with morph targets. Targets and mesh must all have the same vertices count.\");\n this.morphTargetManager = null;\n return;\n }\n if (morphTargetManager.isUsingTextureForTargets) {\n return;\n }\n for (let index = 0; index < morphTargetManager.numInfluencers; index++) {\n const morphTarget = morphTargetManager.getActiveTarget(index);\n const positions = morphTarget.getPositions();\n if (!positions) {\n Logger.Error(\"Invalid morph target. Target must have positions.\");\n return;\n }\n this.geometry.setVerticesData(VertexBuffer.PositionKind + index, positions, false, 3);\n const normals = morphTarget.getNormals();\n if (normals) {\n this.geometry.setVerticesData(VertexBuffer.NormalKind + index, normals, false, 3);\n }\n const tangents = morphTarget.getTangents();\n if (tangents) {\n this.geometry.setVerticesData(VertexBuffer.TangentKind + index, tangents, false, 3);\n }\n const uvs = morphTarget.getUVs();\n if (uvs) {\n this.geometry.setVerticesData(VertexBuffer.UVKind + \"_\" + index, uvs, false, 2);\n }\n }\n }\n else {\n let index = 0;\n // Positions\n while (this.geometry.isVerticesDataPresent(VertexBuffer.PositionKind + index)) {\n this.geometry.removeVerticesData(VertexBuffer.PositionKind + index);\n if (this.geometry.isVerticesDataPresent(VertexBuffer.NormalKind + index)) {\n this.geometry.removeVerticesData(VertexBuffer.NormalKind + index);\n }\n if (this.geometry.isVerticesDataPresent(VertexBuffer.TangentKind + index)) {\n this.geometry.removeVerticesData(VertexBuffer.TangentKind + index);\n }\n if (this.geometry.isVerticesDataPresent(VertexBuffer.UVKind + index)) {\n this.geometry.removeVerticesData(VertexBuffer.UVKind + \"_\" + index);\n }\n index++;\n }\n }\n }\n /**\n * Returns a new Mesh object parsed from the source provided.\n * @param parsedMesh is the source\n * @param scene defines the hosting scene\n * @param rootUrl is the root URL to prefix the `delayLoadingFile` property with\n * @returns a new Mesh\n */\n static Parse(parsedMesh, scene, rootUrl) {\n let mesh;\n if (parsedMesh.type && parsedMesh.type === \"LinesMesh\") {\n mesh = Mesh._LinesMeshParser(parsedMesh, scene);\n }\n else if (parsedMesh.type && parsedMesh.type === \"GroundMesh\") {\n mesh = Mesh._GroundMeshParser(parsedMesh, scene);\n }\n else if (parsedMesh.type && parsedMesh.type === \"GoldbergMesh\") {\n mesh = Mesh._GoldbergMeshParser(parsedMesh, scene);\n }\n else if (parsedMesh.type && parsedMesh.type === \"GreasedLineMesh\") {\n mesh = Mesh._GreasedLineMeshParser(parsedMesh, scene);\n }\n else if (parsedMesh.type && parsedMesh.type === \"TrailMesh\") {\n mesh = Mesh._TrailMeshParser(parsedMesh, scene);\n }\n else {\n mesh = new Mesh(parsedMesh.name, scene);\n }\n mesh.id = parsedMesh.id;\n mesh._waitingParsedUniqueId = parsedMesh.uniqueId;\n if (Tags) {\n Tags.AddTagsTo(mesh, parsedMesh.tags);\n }\n mesh.position = Vector3.FromArray(parsedMesh.position);\n if (parsedMesh.metadata !== undefined) {\n mesh.metadata = parsedMesh.metadata;\n }\n if (parsedMesh.rotationQuaternion) {\n mesh.rotationQuaternion = Quaternion.FromArray(parsedMesh.rotationQuaternion);\n }\n else if (parsedMesh.rotation) {\n mesh.rotation = Vector3.FromArray(parsedMesh.rotation);\n }\n mesh.scaling = Vector3.FromArray(parsedMesh.scaling);\n if (parsedMesh.localMatrix) {\n mesh.setPreTransformMatrix(Matrix.FromArray(parsedMesh.localMatrix));\n }\n else if (parsedMesh.pivotMatrix) {\n mesh.setPivotMatrix(Matrix.FromArray(parsedMesh.pivotMatrix));\n }\n mesh.setEnabled(parsedMesh.isEnabled);\n mesh.isVisible = parsedMesh.isVisible;\n mesh.infiniteDistance = parsedMesh.infiniteDistance;\n mesh.showBoundingBox = parsedMesh.showBoundingBox;\n mesh.showSubMeshesBoundingBox = parsedMesh.showSubMeshesBoundingBox;\n if (parsedMesh.applyFog !== undefined) {\n mesh.applyFog = parsedMesh.applyFog;\n }\n if (parsedMesh.pickable !== undefined) {\n mesh.isPickable = parsedMesh.pickable;\n }\n if (parsedMesh.alphaIndex !== undefined) {\n mesh.alphaIndex = parsedMesh.alphaIndex;\n }\n mesh.receiveShadows = parsedMesh.receiveShadows;\n if (parsedMesh.billboardMode !== undefined) {\n mesh.billboardMode = parsedMesh.billboardMode;\n }\n if (parsedMesh.visibility !== undefined) {\n mesh.visibility = parsedMesh.visibility;\n }\n mesh.checkCollisions = parsedMesh.checkCollisions;\n mesh.overrideMaterialSideOrientation = parsedMesh.overrideMaterialSideOrientation;\n if (parsedMesh.isBlocker !== undefined) {\n mesh.isBlocker = parsedMesh.isBlocker;\n }\n mesh._shouldGenerateFlatShading = parsedMesh.useFlatShading;\n // freezeWorldMatrix\n if (parsedMesh.freezeWorldMatrix) {\n mesh._waitingData.freezeWorldMatrix = parsedMesh.freezeWorldMatrix;\n }\n // Parent\n if (parsedMesh.parentId !== undefined) {\n mesh._waitingParentId = parsedMesh.parentId;\n }\n if (parsedMesh.parentInstanceIndex !== undefined) {\n mesh._waitingParentInstanceIndex = parsedMesh.parentInstanceIndex;\n }\n // Actions\n if (parsedMesh.actions !== undefined) {\n mesh._waitingData.actions = parsedMesh.actions;\n }\n // Overlay\n if (parsedMesh.overlayAlpha !== undefined) {\n mesh.overlayAlpha = parsedMesh.overlayAlpha;\n }\n if (parsedMesh.overlayColor !== undefined) {\n mesh.overlayColor = Color3.FromArray(parsedMesh.overlayColor);\n }\n if (parsedMesh.renderOverlay !== undefined) {\n mesh.renderOverlay = parsedMesh.renderOverlay;\n }\n // Geometry\n mesh.isUnIndexed = !!parsedMesh.isUnIndexed;\n mesh.hasVertexAlpha = parsedMesh.hasVertexAlpha;\n if (parsedMesh.delayLoadingFile) {\n mesh.delayLoadState = 4;\n mesh.delayLoadingFile = rootUrl + parsedMesh.delayLoadingFile;\n mesh.buildBoundingInfo(Vector3.FromArray(parsedMesh.boundingBoxMinimum), Vector3.FromArray(parsedMesh.boundingBoxMaximum));\n if (parsedMesh._binaryInfo) {\n mesh._binaryInfo = parsedMesh._binaryInfo;\n }\n mesh._delayInfo = [];\n if (parsedMesh.hasUVs) {\n mesh._delayInfo.push(VertexBuffer.UVKind);\n }\n if (parsedMesh.hasUVs2) {\n mesh._delayInfo.push(VertexBuffer.UV2Kind);\n }\n if (parsedMesh.hasUVs3) {\n mesh._delayInfo.push(VertexBuffer.UV3Kind);\n }\n if (parsedMesh.hasUVs4) {\n mesh._delayInfo.push(VertexBuffer.UV4Kind);\n }\n if (parsedMesh.hasUVs5) {\n mesh._delayInfo.push(VertexBuffer.UV5Kind);\n }\n if (parsedMesh.hasUVs6) {\n mesh._delayInfo.push(VertexBuffer.UV6Kind);\n }\n if (parsedMesh.hasColors) {\n mesh._delayInfo.push(VertexBuffer.ColorKind);\n }\n if (parsedMesh.hasMatricesIndices) {\n mesh._delayInfo.push(VertexBuffer.MatricesIndicesKind);\n }\n if (parsedMesh.hasMatricesWeights) {\n mesh._delayInfo.push(VertexBuffer.MatricesWeightsKind);\n }\n mesh._delayLoadingFunction = Geometry._ImportGeometry;\n if (SceneLoaderFlags.ForceFullSceneLoadingForIncremental) {\n mesh._checkDelayState();\n }\n }\n else {\n Geometry._ImportGeometry(parsedMesh, mesh);\n }\n // Material\n if (parsedMesh.materialUniqueId) {\n mesh._waitingMaterialId = parsedMesh.materialUniqueId;\n }\n else if (parsedMesh.materialId) {\n mesh._waitingMaterialId = parsedMesh.materialId;\n }\n // Morph targets\n if (parsedMesh.morphTargetManagerId > -1) {\n mesh.morphTargetManager = scene.getMorphTargetManagerById(parsedMesh.morphTargetManagerId);\n }\n // Skeleton\n if (parsedMesh.skeletonId !== undefined && parsedMesh.skeletonId !== null) {\n mesh.skeleton = scene.getLastSkeletonById(parsedMesh.skeletonId);\n if (parsedMesh.numBoneInfluencers) {\n mesh.numBoneInfluencers = parsedMesh.numBoneInfluencers;\n }\n }\n // Animations\n if (parsedMesh.animations) {\n for (let animationIndex = 0; animationIndex < parsedMesh.animations.length; animationIndex++) {\n const parsedAnimation = parsedMesh.animations[animationIndex];\n const internalClass = GetClass(\"BABYLON.Animation\");\n if (internalClass) {\n mesh.animations.push(internalClass.Parse(parsedAnimation));\n }\n }\n Node.ParseAnimationRanges(mesh, parsedMesh, scene);\n }\n if (parsedMesh.autoAnimate) {\n scene.beginAnimation(mesh, parsedMesh.autoAnimateFrom, parsedMesh.autoAnimateTo, parsedMesh.autoAnimateLoop, parsedMesh.autoAnimateSpeed || 1.0);\n }\n // Layer Mask\n if (parsedMesh.layerMask && !isNaN(parsedMesh.layerMask)) {\n mesh.layerMask = Math.abs(parseInt(parsedMesh.layerMask));\n }\n else {\n mesh.layerMask = 0x0fffffff;\n }\n // Physics\n if (parsedMesh.physicsImpostor) {\n Mesh._PhysicsImpostorParser(scene, mesh, parsedMesh);\n }\n // Levels\n if (parsedMesh.lodMeshIds) {\n mesh._waitingData.lods = {\n ids: parsedMesh.lodMeshIds,\n distances: parsedMesh.lodDistances ? parsedMesh.lodDistances : null,\n coverages: parsedMesh.lodCoverages ? parsedMesh.lodCoverages : null,\n };\n }\n // Instances\n if (parsedMesh.instances) {\n for (let index = 0; index < parsedMesh.instances.length; index++) {\n const parsedInstance = parsedMesh.instances[index];\n const instance = mesh.createInstance(parsedInstance.name);\n if (parsedInstance.id) {\n instance.id = parsedInstance.id;\n }\n if (Tags) {\n if (parsedInstance.tags) {\n Tags.AddTagsTo(instance, parsedInstance.tags);\n }\n else {\n Tags.AddTagsTo(instance, parsedMesh.tags);\n }\n }\n instance.position = Vector3.FromArray(parsedInstance.position);\n if (parsedInstance.metadata !== undefined) {\n instance.metadata = parsedInstance.metadata;\n }\n if (parsedInstance.parentId !== undefined) {\n instance._waitingParentId = parsedInstance.parentId;\n }\n if (parsedInstance.parentInstanceIndex !== undefined) {\n instance._waitingParentInstanceIndex = parsedInstance.parentInstanceIndex;\n }\n if (parsedInstance.isEnabled !== undefined && parsedInstance.isEnabled !== null) {\n instance.setEnabled(parsedInstance.isEnabled);\n }\n if (parsedInstance.isVisible !== undefined && parsedInstance.isVisible !== null) {\n instance.isVisible = parsedInstance.isVisible;\n }\n if (parsedInstance.isPickable !== undefined && parsedInstance.isPickable !== null) {\n instance.isPickable = parsedInstance.isPickable;\n }\n if (parsedInstance.rotationQuaternion) {\n instance.rotationQuaternion = Quaternion.FromArray(parsedInstance.rotationQuaternion);\n }\n else if (parsedInstance.rotation) {\n instance.rotation = Vector3.FromArray(parsedInstance.rotation);\n }\n instance.scaling = Vector3.FromArray(parsedInstance.scaling);\n if (parsedInstance.checkCollisions != undefined && parsedInstance.checkCollisions != null) {\n instance.checkCollisions = parsedInstance.checkCollisions;\n }\n if (parsedInstance.pickable != undefined && parsedInstance.pickable != null) {\n instance.isPickable = parsedInstance.pickable;\n }\n if (parsedInstance.showBoundingBox != undefined && parsedInstance.showBoundingBox != null) {\n instance.showBoundingBox = parsedInstance.showBoundingBox;\n }\n if (parsedInstance.showSubMeshesBoundingBox != undefined && parsedInstance.showSubMeshesBoundingBox != null) {\n instance.showSubMeshesBoundingBox = parsedInstance.showSubMeshesBoundingBox;\n }\n if (parsedInstance.alphaIndex != undefined && parsedInstance.showSubMeshesBoundingBox != null) {\n instance.alphaIndex = parsedInstance.alphaIndex;\n }\n // Physics\n if (parsedInstance.physicsImpostor) {\n Mesh._PhysicsImpostorParser(scene, instance, parsedInstance);\n }\n // Actions\n if (parsedInstance.actions !== undefined) {\n instance._waitingData.actions = parsedInstance.actions;\n }\n // Animation\n if (parsedInstance.animations) {\n for (let animationIndex = 0; animationIndex < parsedInstance.animations.length; animationIndex++) {\n const parsedAnimation = parsedInstance.animations[animationIndex];\n const internalClass = GetClass(\"BABYLON.Animation\");\n if (internalClass) {\n instance.animations.push(internalClass.Parse(parsedAnimation));\n }\n }\n Node.ParseAnimationRanges(instance, parsedInstance, scene);\n if (parsedInstance.autoAnimate) {\n scene.beginAnimation(instance, parsedInstance.autoAnimateFrom, parsedInstance.autoAnimateTo, parsedInstance.autoAnimateLoop, parsedInstance.autoAnimateSpeed || 1.0);\n }\n }\n }\n }\n // Thin instances\n if (parsedMesh.thinInstances) {\n const thinInstances = parsedMesh.thinInstances;\n mesh.thinInstanceEnablePicking = !!thinInstances.enablePicking;\n if (thinInstances.matrixData) {\n mesh.thinInstanceSetBuffer(\"matrix\", new Float32Array(thinInstances.matrixData), 16, false);\n mesh._thinInstanceDataStorage.matrixBufferSize = thinInstances.matrixBufferSize;\n mesh._thinInstanceDataStorage.instancesCount = thinInstances.instancesCount;\n }\n else {\n mesh._thinInstanceDataStorage.matrixBufferSize = thinInstances.matrixBufferSize;\n }\n if (parsedMesh.thinInstances.userThinInstance) {\n const userThinInstance = parsedMesh.thinInstances.userThinInstance;\n for (const kind in userThinInstance.data) {\n mesh.thinInstanceSetBuffer(kind, new Float32Array(userThinInstance.data[kind]), userThinInstance.strides[kind], false);\n mesh._userThinInstanceBuffersStorage.sizes[kind] = userThinInstance.sizes[kind];\n }\n }\n }\n return mesh;\n }\n // Skeletons\n /**\n * Prepare internal position array for software CPU skinning\n * @returns original positions used for CPU skinning. Useful for integrating Morphing with skeletons in same mesh\n */\n setPositionsForCPUSkinning() {\n const internalDataInfo = this._internalMeshDataInfo;\n if (!internalDataInfo._sourcePositions) {\n const source = this.getVerticesData(VertexBuffer.PositionKind);\n if (!source) {\n return internalDataInfo._sourcePositions;\n }\n internalDataInfo._sourcePositions = new Float32Array(source);\n if (!this.isVertexBufferUpdatable(VertexBuffer.PositionKind)) {\n this.setVerticesData(VertexBuffer.PositionKind, source, true);\n }\n }\n return internalDataInfo._sourcePositions;\n }\n /**\n * Prepare internal normal array for software CPU skinning\n * @returns original normals used for CPU skinning. Useful for integrating Morphing with skeletons in same mesh.\n */\n setNormalsForCPUSkinning() {\n const internalDataInfo = this._internalMeshDataInfo;\n if (!internalDataInfo._sourceNormals) {\n const source = this.getVerticesData(VertexBuffer.NormalKind);\n if (!source) {\n return internalDataInfo._sourceNormals;\n }\n internalDataInfo._sourceNormals = new Float32Array(source);\n if (!this.isVertexBufferUpdatable(VertexBuffer.NormalKind)) {\n this.setVerticesData(VertexBuffer.NormalKind, source, true);\n }\n }\n return internalDataInfo._sourceNormals;\n }\n /**\n * Updates the vertex buffer by applying transformation from the bones\n * @param skeleton defines the skeleton to apply to current mesh\n * @returns the current mesh\n */\n applySkeleton(skeleton) {\n if (!this.geometry) {\n return this;\n }\n if (this.geometry._softwareSkinningFrameId == this.getScene().getFrameId()) {\n return this;\n }\n this.geometry._softwareSkinningFrameId = this.getScene().getFrameId();\n if (!this.isVerticesDataPresent(VertexBuffer.PositionKind)) {\n return this;\n }\n if (!this.isVerticesDataPresent(VertexBuffer.MatricesIndicesKind)) {\n return this;\n }\n if (!this.isVerticesDataPresent(VertexBuffer.MatricesWeightsKind)) {\n return this;\n }\n const hasNormals = this.isVerticesDataPresent(VertexBuffer.NormalKind);\n const internalDataInfo = this._internalMeshDataInfo;\n if (!internalDataInfo._sourcePositions) {\n const submeshes = this.subMeshes.slice();\n this.setPositionsForCPUSkinning();\n this.subMeshes = submeshes;\n }\n if (hasNormals && !internalDataInfo._sourceNormals) {\n this.setNormalsForCPUSkinning();\n }\n // positionsData checks for not being Float32Array will only pass at most once\n let positionsData = this.getVerticesData(VertexBuffer.PositionKind);\n if (!positionsData) {\n return this;\n }\n if (!(positionsData instanceof Float32Array)) {\n positionsData = new Float32Array(positionsData);\n }\n // normalsData checks for not being Float32Array will only pass at most once\n let normalsData = this.getVerticesData(VertexBuffer.NormalKind);\n if (hasNormals) {\n if (!normalsData) {\n return this;\n }\n if (!(normalsData instanceof Float32Array)) {\n normalsData = new Float32Array(normalsData);\n }\n }\n const matricesIndicesData = this.getVerticesData(VertexBuffer.MatricesIndicesKind);\n const matricesWeightsData = this.getVerticesData(VertexBuffer.MatricesWeightsKind);\n if (!matricesWeightsData || !matricesIndicesData) {\n return this;\n }\n const needExtras = this.numBoneInfluencers > 4;\n const matricesIndicesExtraData = needExtras ? this.getVerticesData(VertexBuffer.MatricesIndicesExtraKind) : null;\n const matricesWeightsExtraData = needExtras ? this.getVerticesData(VertexBuffer.MatricesWeightsExtraKind) : null;\n const skeletonMatrices = skeleton.getTransformMatrices(this);\n const tempVector3 = Vector3.Zero();\n const finalMatrix = new Matrix();\n const tempMatrix = new Matrix();\n let matWeightIdx = 0;\n let inf;\n for (let index = 0; index < positionsData.length; index += 3, matWeightIdx += 4) {\n let weight;\n for (inf = 0; inf < 4; inf++) {\n weight = matricesWeightsData[matWeightIdx + inf];\n if (weight > 0) {\n Matrix.FromFloat32ArrayToRefScaled(skeletonMatrices, Math.floor(matricesIndicesData[matWeightIdx + inf] * 16), weight, tempMatrix);\n finalMatrix.addToSelf(tempMatrix);\n }\n }\n if (needExtras) {\n for (inf = 0; inf < 4; inf++) {\n weight = matricesWeightsExtraData[matWeightIdx + inf];\n if (weight > 0) {\n Matrix.FromFloat32ArrayToRefScaled(skeletonMatrices, Math.floor(matricesIndicesExtraData[matWeightIdx + inf] * 16), weight, tempMatrix);\n finalMatrix.addToSelf(tempMatrix);\n }\n }\n }\n Vector3.TransformCoordinatesFromFloatsToRef(internalDataInfo._sourcePositions[index], internalDataInfo._sourcePositions[index + 1], internalDataInfo._sourcePositions[index + 2], finalMatrix, tempVector3);\n tempVector3.toArray(positionsData, index);\n if (hasNormals) {\n Vector3.TransformNormalFromFloatsToRef(internalDataInfo._sourceNormals[index], internalDataInfo._sourceNormals[index + 1], internalDataInfo._sourceNormals[index + 2], finalMatrix, tempVector3);\n tempVector3.toArray(normalsData, index);\n }\n finalMatrix.reset();\n }\n this.updateVerticesData(VertexBuffer.PositionKind, positionsData);\n if (hasNormals) {\n this.updateVerticesData(VertexBuffer.NormalKind, normalsData);\n }\n return this;\n }\n // Tools\n /**\n * Returns an object containing a min and max Vector3 which are the minimum and maximum vectors of each mesh bounding box from the passed array, in the world coordinates\n * @param meshes defines the list of meshes to scan\n * @returns an object `{min:` Vector3`, max:` Vector3`}`\n */\n static MinMax(meshes) {\n let minVector = null;\n let maxVector = null;\n meshes.forEach(function (mesh) {\n const boundingInfo = mesh.getBoundingInfo();\n const boundingBox = boundingInfo.boundingBox;\n if (!minVector || !maxVector) {\n minVector = boundingBox.minimumWorld;\n maxVector = boundingBox.maximumWorld;\n }\n else {\n minVector.minimizeInPlace(boundingBox.minimumWorld);\n maxVector.maximizeInPlace(boundingBox.maximumWorld);\n }\n });\n if (!minVector || !maxVector) {\n return {\n min: Vector3.Zero(),\n max: Vector3.Zero(),\n };\n }\n return {\n min: minVector,\n max: maxVector,\n };\n }\n /**\n * Returns the center of the `{min:` Vector3`, max:` Vector3`}` or the center of MinMax vector3 computed from a mesh array\n * @param meshesOrMinMaxVector could be an array of meshes or a `{min:` Vector3`, max:` Vector3`}` object\n * @returns a vector3\n */\n static Center(meshesOrMinMaxVector) {\n const minMaxVector = meshesOrMinMaxVector instanceof Array ? Mesh.MinMax(meshesOrMinMaxVector) : meshesOrMinMaxVector;\n return Vector3.Center(minMaxVector.min, minMaxVector.max);\n }\n /**\n * Merge the array of meshes into a single mesh for performance reasons.\n * @param meshes array of meshes with the vertices to merge. Entries cannot be empty meshes.\n * @param disposeSource when true (default), dispose of the vertices from the source meshes.\n * @param allow32BitsIndices when the sum of the vertices > 64k, this must be set to true.\n * @param meshSubclass (optional) can be set to a Mesh where the merged vertices will be inserted.\n * @param subdivideWithSubMeshes when true (false default), subdivide mesh into subMeshes.\n * @param multiMultiMaterials when true (false default), subdivide mesh into subMeshes with multiple materials, ignores subdivideWithSubMeshes.\n * @returns a new mesh\n */\n static MergeMeshes(meshes, disposeSource = true, allow32BitsIndices, meshSubclass, subdivideWithSubMeshes, multiMultiMaterials) {\n return runCoroutineSync(Mesh._MergeMeshesCoroutine(meshes, disposeSource, allow32BitsIndices, meshSubclass, subdivideWithSubMeshes, multiMultiMaterials, false));\n }\n /**\n * Merge the array of meshes into a single mesh for performance reasons.\n * @param meshes array of meshes with the vertices to merge. Entries cannot be empty meshes.\n * @param disposeSource when true (default), dispose of the vertices from the source meshes.\n * @param allow32BitsIndices when the sum of the vertices > 64k, this must be set to true.\n * @param meshSubclass (optional) can be set to a Mesh where the merged vertices will be inserted.\n * @param subdivideWithSubMeshes when true (false default), subdivide mesh into subMeshes.\n * @param multiMultiMaterials when true (false default), subdivide mesh into subMeshes with multiple materials, ignores subdivideWithSubMeshes.\n * @returns a new mesh\n */\n static MergeMeshesAsync(meshes, disposeSource = true, allow32BitsIndices, meshSubclass, subdivideWithSubMeshes, multiMultiMaterials) {\n return runCoroutineAsync(Mesh._MergeMeshesCoroutine(meshes, disposeSource, allow32BitsIndices, meshSubclass, subdivideWithSubMeshes, multiMultiMaterials, true), createYieldingScheduler());\n }\n static *_MergeMeshesCoroutine(meshes, disposeSource = true, allow32BitsIndices, meshSubclass, subdivideWithSubMeshes, multiMultiMaterials, isAsync) {\n // Remove any null/undefined entries from the mesh array\n meshes = meshes.filter(Boolean);\n if (meshes.length === 0) {\n return null;\n }\n let index;\n if (!allow32BitsIndices) {\n let totalVertices = 0;\n // Counting vertices\n for (index = 0; index < meshes.length; index++) {\n totalVertices += meshes[index].getTotalVertices();\n if (totalVertices >= 65536) {\n Logger.Warn(\"Cannot merge meshes because resulting mesh will have more than 65536 vertices. Please use allow32BitsIndices = true to use 32 bits indices\");\n return null;\n }\n }\n }\n if (multiMultiMaterials) {\n subdivideWithSubMeshes = false;\n }\n const materialArray = new Array();\n const materialIndexArray = new Array();\n // Merge\n const indiceArray = new Array();\n const currentOverrideMaterialSideOrientation = meshes[0].overrideMaterialSideOrientation;\n for (index = 0; index < meshes.length; index++) {\n const mesh = meshes[index];\n if (mesh.isAnInstance) {\n Logger.Warn(\"Cannot merge instance meshes.\");\n return null;\n }\n if (currentOverrideMaterialSideOrientation !== mesh.overrideMaterialSideOrientation) {\n Logger.Warn(\"Cannot merge meshes with different overrideMaterialSideOrientation values.\");\n return null;\n }\n if (subdivideWithSubMeshes) {\n indiceArray.push(mesh.getTotalIndices());\n }\n if (multiMultiMaterials) {\n if (mesh.material) {\n const material = mesh.material;\n if (material instanceof MultiMaterial) {\n for (let matIndex = 0; matIndex < material.subMaterials.length; matIndex++) {\n if (materialArray.indexOf(material.subMaterials[matIndex]) < 0) {\n materialArray.push(material.subMaterials[matIndex]);\n }\n }\n for (let subIndex = 0; subIndex < mesh.subMeshes.length; subIndex++) {\n materialIndexArray.push(materialArray.indexOf(material.subMaterials[mesh.subMeshes[subIndex].materialIndex]));\n indiceArray.push(mesh.subMeshes[subIndex].indexCount);\n }\n }\n else {\n if (materialArray.indexOf(material) < 0) {\n materialArray.push(material);\n }\n for (let subIndex = 0; subIndex < mesh.subMeshes.length; subIndex++) {\n materialIndexArray.push(materialArray.indexOf(material));\n indiceArray.push(mesh.subMeshes[subIndex].indexCount);\n }\n }\n }\n else {\n for (let subIndex = 0; subIndex < mesh.subMeshes.length; subIndex++) {\n materialIndexArray.push(0);\n indiceArray.push(mesh.subMeshes[subIndex].indexCount);\n }\n }\n }\n }\n const source = meshes[0];\n const getVertexDataFromMesh = (mesh) => {\n const wm = mesh.computeWorldMatrix(true);\n const vertexData = VertexData.ExtractFromMesh(mesh, false, false);\n return { vertexData, transform: wm };\n };\n const { vertexData: sourceVertexData, transform: sourceTransform } = getVertexDataFromMesh(source);\n if (isAsync) {\n yield;\n }\n const meshVertexDatas = new Array(meshes.length - 1);\n for (let i = 1; i < meshes.length; i++) {\n meshVertexDatas[i - 1] = getVertexDataFromMesh(meshes[i]);\n if (isAsync) {\n yield;\n }\n }\n const mergeCoroutine = sourceVertexData._mergeCoroutine(sourceTransform, meshVertexDatas, allow32BitsIndices, isAsync, !disposeSource);\n let mergeCoroutineStep = mergeCoroutine.next();\n while (!mergeCoroutineStep.done) {\n if (isAsync) {\n yield;\n }\n mergeCoroutineStep = mergeCoroutine.next();\n }\n const vertexData = mergeCoroutineStep.value;\n if (!meshSubclass) {\n meshSubclass = new Mesh(source.name + \"_merged\", source.getScene());\n }\n const applyToCoroutine = vertexData._applyToCoroutine(meshSubclass, undefined, isAsync);\n let applyToCoroutineStep = applyToCoroutine.next();\n while (!applyToCoroutineStep.done) {\n if (isAsync) {\n yield;\n }\n applyToCoroutineStep = applyToCoroutine.next();\n }\n // Setting properties\n meshSubclass.checkCollisions = source.checkCollisions;\n meshSubclass.overrideMaterialSideOrientation = source.overrideMaterialSideOrientation;\n // Cleaning\n if (disposeSource) {\n for (index = 0; index < meshes.length; index++) {\n meshes[index].dispose();\n }\n }\n // Subdivide\n if (subdivideWithSubMeshes || multiMultiMaterials) {\n //-- removal of global submesh\n meshSubclass.releaseSubMeshes();\n index = 0;\n let offset = 0;\n //-- apply subdivision according to index table\n while (index < indiceArray.length) {\n SubMesh.CreateFromIndices(0, offset, indiceArray[index], meshSubclass, undefined, false);\n offset += indiceArray[index];\n index++;\n }\n for (const subMesh of meshSubclass.subMeshes) {\n subMesh.refreshBoundingInfo();\n }\n meshSubclass.computeWorldMatrix(true);\n }\n if (multiMultiMaterials) {\n const newMultiMaterial = new MultiMaterial(source.name + \"_merged\", source.getScene());\n newMultiMaterial.subMaterials = materialArray;\n for (let subIndex = 0; subIndex < meshSubclass.subMeshes.length; subIndex++) {\n meshSubclass.subMeshes[subIndex].materialIndex = materialIndexArray[subIndex];\n }\n meshSubclass.material = newMultiMaterial;\n }\n else {\n meshSubclass.material = source.material;\n }\n return meshSubclass;\n }\n /**\n * @internal\n */\n addInstance(instance) {\n instance._indexInSourceMeshInstanceArray = this.instances.length;\n this.instances.push(instance);\n }\n /**\n * @internal\n */\n removeInstance(instance) {\n // Remove from mesh\n const index = instance._indexInSourceMeshInstanceArray;\n if (index != -1) {\n if (index !== this.instances.length - 1) {\n const last = this.instances[this.instances.length - 1];\n this.instances[index] = last;\n last._indexInSourceMeshInstanceArray = index;\n }\n instance._indexInSourceMeshInstanceArray = -1;\n this.instances.pop();\n }\n }\n /** @internal */\n _shouldConvertRHS() {\n return this.overrideMaterialSideOrientation === Material.CounterClockWiseSideOrientation;\n }\n /** @internal */\n _getRenderingFillMode(fillMode) {\n var _a;\n const scene = this.getScene();\n if (scene.forcePointsCloud)\n return Material.PointFillMode;\n if (scene.forceWireframe)\n return Material.WireFrameFillMode;\n return (_a = this.overrideRenderingFillMode) !== null && _a !== void 0 ? _a : fillMode;\n }\n // deprecated methods\n /**\n * Sets the mesh material by the material or multiMaterial `id` property\n * @param id is a string identifying the material or the multiMaterial\n * @returns the current mesh\n * @deprecated Please use MeshBuilder instead Please use setMaterialById instead\n */\n setMaterialByID(id) {\n return this.setMaterialById(id);\n }\n /**\n * Creates a ribbon mesh.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param\n * @param name defines the name of the mesh to create\n * @param pathArray is a required array of paths, what are each an array of successive Vector3. The pathArray parameter depicts the ribbon geometry.\n * @param closeArray creates a seam between the first and the last paths of the path array (default is false)\n * @param closePath creates a seam between the first and the last points of each path of the path array\n * @param offset is taken in account only if the `pathArray` is containing a single path\n * @param scene defines the hosting scene\n * @param updatable defines if the mesh must be flagged as updatable\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\n * @param instance defines an instance of an existing Ribbon object to be updated with the passed `pathArray` parameter (https://doc.babylonjs.com/how_to/How_to_dynamically_morph_a_mesh#ribbon)\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static CreateRibbon(name, pathArray, closeArray, closePath, offset, scene, updatable, sideOrientation, instance) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates a plane polygonal mesh. By default, this is a disc.\n * @param name defines the name of the mesh to create\n * @param radius sets the radius size (float) of the polygon (default 0.5)\n * @param tessellation sets the number of polygon sides (positive integer, default 64). So a tessellation valued to 3 will build a triangle, to 4 a square, etc\n * @param scene defines the hosting scene\n * @param updatable defines if the mesh must be flagged as updatable\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static CreateDisc(name, radius, tessellation, scene, updatable, sideOrientation) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates a box mesh.\n * @param name defines the name of the mesh to create\n * @param size sets the size (float) of each box side (default 1)\n * @param scene defines the hosting scene\n * @param updatable defines if the mesh must be flagged as updatable\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static CreateBox(name, size, scene, updatable, sideOrientation) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates a sphere mesh.\n * @param name defines the name of the mesh to create\n * @param segments sets the sphere number of horizontal stripes (positive integer, default 32)\n * @param diameter sets the diameter size (float) of the sphere (default 1)\n * @param scene defines the hosting scene\n * @param updatable defines if the mesh must be flagged as updatable\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static CreateSphere(name, segments, diameter, scene, updatable, sideOrientation) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates a hemisphere mesh.\n * @param name defines the name of the mesh to create\n * @param segments sets the sphere number of horizontal stripes (positive integer, default 32)\n * @param diameter sets the diameter size (float) of the sphere (default 1)\n * @param scene defines the hosting scene\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static CreateHemisphere(name, segments, diameter, scene) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates a cylinder or a cone mesh.\n * @param name defines the name of the mesh to create\n * @param height sets the height size (float) of the cylinder/cone (float, default 2)\n * @param diameterTop set the top cap diameter (floats, default 1)\n * @param diameterBottom set the bottom cap diameter (floats, default 1). This value can't be zero\n * @param tessellation sets the number of cylinder sides (positive integer, default 24). Set it to 3 to get a prism for instance\n * @param subdivisions sets the number of rings along the cylinder height (positive integer, default 1)\n * @param scene defines the hosting scene\n * @param updatable defines if the mesh must be flagged as updatable\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static CreateCylinder(name, height, diameterTop, diameterBottom, tessellation, subdivisions, scene, updatable, sideOrientation) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n // Torus (Code from SharpDX.org)\n /**\n * Creates a torus mesh.\n * @param name defines the name of the mesh to create\n * @param diameter sets the diameter size (float) of the torus (default 1)\n * @param thickness sets the diameter size of the tube of the torus (float, default 0.5)\n * @param tessellation sets the number of torus sides (positive integer, default 16)\n * @param scene defines the hosting scene\n * @param updatable defines if the mesh must be flagged as updatable\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static CreateTorus(name, diameter, thickness, tessellation, scene, updatable, sideOrientation) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates a torus knot mesh.\n * @param name defines the name of the mesh to create\n * @param radius sets the global radius size (float) of the torus knot (default 2)\n * @param tube sets the diameter size of the tube of the torus (float, default 0.5)\n * @param radialSegments sets the number of sides on each tube segments (positive integer, default 32)\n * @param tubularSegments sets the number of tubes to decompose the knot into (positive integer, default 32)\n * @param p the number of windings on X axis (positive integers, default 2)\n * @param q the number of windings on Y axis (positive integers, default 3)\n * @param scene defines the hosting scene\n * @param updatable defines if the mesh must be flagged as updatable\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static CreateTorusKnot(name, radius, tube, radialSegments, tubularSegments, p, q, scene, updatable, sideOrientation) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates a line mesh..\n * @param name defines the name of the mesh to create\n * @param points is an array successive Vector3\n * @param scene defines the hosting scene\n * @param updatable defines if the mesh must be flagged as updatable\n * @param instance is an instance of an existing LineMesh object to be updated with the passed `points` parameter (https://doc.babylonjs.com/how_to/How_to_dynamically_morph_a_mesh#lines-and-dashedlines).\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static CreateLines(name, points, scene, updatable, instance) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates a dashed line mesh.\n * @param name defines the name of the mesh to create\n * @param points is an array successive Vector3\n * @param dashSize is the size of the dashes relatively the dash number (positive float, default 3)\n * @param gapSize is the size of the gap between two successive dashes relatively the dash number (positive float, default 1)\n * @param dashNb is the intended total number of dashes (positive integer, default 200)\n * @param scene defines the hosting scene\n * @param updatable defines if the mesh must be flagged as updatable\n * @param instance is an instance of an existing LineMesh object to be updated with the passed `points` parameter (https://doc.babylonjs.com/how_to/How_to_dynamically_morph_a_mesh#lines-and-dashedlines)\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static CreateDashedLines(name, points, dashSize, gapSize, dashNb, scene, updatable, instance) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates a polygon mesh.Please consider using the same method from the MeshBuilder class instead\n * The polygon's shape will depend on the input parameters and is constructed parallel to a ground mesh.\n * The parameter `shape` is a required array of successive Vector3 representing the corners of the polygon in th XoZ plane, that is y = 0 for all vectors.\n * You can set the mesh side orientation with the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\n * Remember you can only change the shape positions, not their number when updating a polygon.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#non-regular-polygon\n * @param name defines the name of the mesh to create\n * @param shape is a required array of successive Vector3 representing the corners of the polygon in th XoZ plane, that is y = 0 for all vectors\n * @param scene defines the hosting scene\n * @param holes is a required array of arrays of successive Vector3 used to defines holes in the polygon\n * @param updatable defines if the mesh must be flagged as updatable\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\n * @param earcutInjection can be used to inject your own earcut reference\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static CreatePolygon(name, shape, scene, holes, updatable, sideOrientation, earcutInjection) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates an extruded polygon mesh, with depth in the Y direction..\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#extruded-non-regular-polygon\n * @param name defines the name of the mesh to create\n * @param shape is a required array of successive Vector3 representing the corners of the polygon in th XoZ plane, that is y = 0 for all vectors\n * @param depth defines the height of extrusion\n * @param scene defines the hosting scene\n * @param holes is a required array of arrays of successive Vector3 used to defines holes in the polygon\n * @param updatable defines if the mesh must be flagged as updatable\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\n * @param earcutInjection can be used to inject your own earcut reference\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static ExtrudePolygon(name, shape, depth, scene, holes, updatable, sideOrientation, earcutInjection) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates an extruded shape mesh.\n * The extrusion is a parametric shape. It has no predefined shape. Its final shape will depend on the input parameters.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#extruded-shapes\n * @param name defines the name of the mesh to create\n * @param shape is a required array of successive Vector3. This array depicts the shape to be extruded in its local space : the shape must be designed in the xOy plane and will be extruded along the Z axis\n * @param path is a required array of successive Vector3. This is the axis curve the shape is extruded along\n * @param scale is the value to scale the shape\n * @param rotation is the angle value to rotate the shape each step (each path point), from the former step (so rotation added each step) along the curve\n * @param cap sets the way the extruded shape is capped. Possible values : Mesh.NO_CAP (default), Mesh.CAP_START, Mesh.CAP_END, Mesh.CAP_ALL\n * @param scene defines the hosting scene\n * @param updatable defines if the mesh must be flagged as updatable\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\n * @param instance is an instance of an existing ExtrudedShape object to be updated with the passed `shape`, `path`, `scale` or `rotation` parameters (https://doc.babylonjs.com/how_to/How_to_dynamically_morph_a_mesh#extruded-shape)\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static ExtrudeShape(name, shape, path, scale, rotation, cap, scene, updatable, sideOrientation, instance) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates an custom extruded shape mesh.\n * The custom extrusion is a parametric shape.\n * It has no predefined shape. Its final shape will depend on the input parameters.\n *\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#extruded-shapes\n * @param name defines the name of the mesh to create\n * @param shape is a required array of successive Vector3. This array depicts the shape to be extruded in its local space : the shape must be designed in the xOy plane and will be extruded along the Z axis\n * @param path is a required array of successive Vector3. This is the axis curve the shape is extruded along\n * @param scaleFunction is a custom Javascript function called on each path point\n * @param rotationFunction is a custom Javascript function called on each path point\n * @param ribbonCloseArray forces the extrusion underlying ribbon to close all the paths in its `pathArray`\n * @param ribbonClosePath forces the extrusion underlying ribbon to close its `pathArray`\n * @param cap sets the way the extruded shape is capped. Possible values : Mesh.NO_CAP (default), Mesh.CAP_START, Mesh.CAP_END, Mesh.CAP_ALL\n * @param scene defines the hosting scene\n * @param updatable defines if the mesh must be flagged as updatable\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\n * @param instance is an instance of an existing ExtrudedShape object to be updated with the passed `shape`, `path`, `scale` or `rotation` parameters (https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#extruded-shape)\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static ExtrudeShapeCustom(name, shape, path, scaleFunction, rotationFunction, ribbonCloseArray, ribbonClosePath, cap, scene, updatable, sideOrientation, instance) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates lathe mesh.\n * The lathe is a shape with a symmetry axis : a 2D model shape is rotated around this axis to design the lathe.\n * @param name defines the name of the mesh to create\n * @param shape is a required array of successive Vector3. This array depicts the shape to be rotated in its local space : the shape must be designed in the xOy plane and will be rotated around the Y axis. It's usually a 2D shape, so the Vector3 z coordinates are often set to zero\n * @param radius is the radius value of the lathe\n * @param tessellation is the side number of the lathe.\n * @param scene defines the hosting scene\n * @param updatable defines if the mesh must be flagged as updatable\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static CreateLathe(name, shape, radius, tessellation, scene, updatable, sideOrientation) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates a plane mesh.\n * @param name defines the name of the mesh to create\n * @param size sets the size (float) of both sides of the plane at once (default 1)\n * @param scene defines the hosting scene\n * @param updatable defines if the mesh must be flagged as updatable\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static CreatePlane(name, size, scene, updatable, sideOrientation) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates a ground mesh.\n * @param name defines the name of the mesh to create\n * @param width set the width of the ground\n * @param height set the height of the ground\n * @param subdivisions sets the number of subdivisions per side\n * @param scene defines the hosting scene\n * @param updatable defines if the mesh must be flagged as updatable\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static CreateGround(name, width, height, subdivisions, scene, updatable) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates a tiled ground mesh.\n * @param name defines the name of the mesh to create\n * @param xmin set the ground minimum X coordinate\n * @param zmin set the ground minimum Y coordinate\n * @param xmax set the ground maximum X coordinate\n * @param zmax set the ground maximum Z coordinate\n * @param subdivisions is an object `{w: positive integer, h: positive integer}` (default `{w: 6, h: 6}`). `w` and `h` are the numbers of subdivisions on the ground width and height. Each subdivision is called a tile\n * @param precision is an object `{w: positive integer, h: positive integer}` (default `{w: 2, h: 2}`). `w` and `h` are the numbers of subdivisions on the ground width and height of each tile\n * @param scene defines the hosting scene\n * @param updatable defines if the mesh must be flagged as updatable\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static CreateTiledGround(name, xmin, zmin, xmax, zmax, subdivisions, precision, scene, updatable) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates a ground mesh from a height map.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set/height_map\n * @param name defines the name of the mesh to create\n * @param url sets the URL of the height map image resource\n * @param width set the ground width size\n * @param height set the ground height size\n * @param subdivisions sets the number of subdivision per side\n * @param minHeight is the minimum altitude on the ground\n * @param maxHeight is the maximum altitude on the ground\n * @param scene defines the hosting scene\n * @param updatable defines if the mesh must be flagged as updatable\n * @param onReady is a callback function that will be called once the mesh is built (the height map download can last some time)\n * @param alphaFilter will filter any data where the alpha channel is below this value, defaults 0 (all data visible)\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static CreateGroundFromHeightMap(name, url, width, height, subdivisions, minHeight, maxHeight, scene, updatable, onReady, alphaFilter) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates a tube mesh.\n * The tube is a parametric shape.\n * It has no predefined shape. Its final shape will depend on the input parameters.\n *\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param\n * @param name defines the name of the mesh to create\n * @param path is a required array of successive Vector3. It is the curve used as the axis of the tube\n * @param radius sets the tube radius size\n * @param tessellation is the number of sides on the tubular surface\n * @param radiusFunction is a custom function. If it is not null, it overrides the parameter `radius`. This function is called on each point of the tube path and is passed the index `i` of the i-th point and the distance of this point from the first point of the path\n * @param cap sets the way the extruded shape is capped. Possible values : Mesh.NO_CAP (default), Mesh.CAP_START, Mesh.CAP_END, Mesh.CAP_ALL\n * @param scene defines the hosting scene\n * @param updatable defines if the mesh must be flagged as updatable\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\n * @param instance is an instance of an existing Tube object to be updated with the passed `pathArray` parameter (https://doc.babylonjs.com/how_to/How_to_dynamically_morph_a_mesh#tube)\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static CreateTube(name, path, radius, tessellation, radiusFunction, cap, scene, updatable, sideOrientation, instance) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates a polyhedron mesh.\n *.\n * * The parameter `type` (positive integer, max 14, default 0) sets the polyhedron type to build among the 15 embedded types. Please refer to the type sheet in the tutorial to choose the wanted type\n * * The parameter `size` (positive float, default 1) sets the polygon size\n * * You can overwrite the `size` on each dimension bu using the parameters `sizeX`, `sizeY` or `sizeZ` (positive floats, default to `size` value)\n * * You can build other polyhedron types than the 15 embbeded ones by setting the parameter `custom` (`polyhedronObject`, default null). If you set the parameter `custom`, this overwrittes the parameter `type`\n * * A `polyhedronObject` is a formatted javascript object. You'll find a full file with pre-set polyhedra here : https://github.com/BabylonJS/Extensions/tree/master/Polyhedron\n * * You can set the color and the UV of each side of the polyhedron with the parameters `faceColors` (Color4, default `(1, 1, 1, 1)`) and faceUV (Vector4, default `(0, 0, 1, 1)`)\n * * To understand how to set `faceUV` or `faceColors`, please read this by considering the right number of faces of your polyhedron, instead of only 6 for the box : https://doc.babylonjs.com/features/featuresDeepDive/materials/using/texturePerBoxFace\n * * The parameter `flat` (boolean, default true). If set to false, it gives the polyhedron a single global face, so less vertices and shared normals. In this case, `faceColors` and `faceUV` are ignored\n * * You can also set the mesh side orientation with the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\n * @param name defines the name of the mesh to create\n * @param options defines the options used to create the mesh\n * @param scene defines the hosting scene\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static CreatePolyhedron(name, options, scene) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates a sphere based upon an icosahedron with 20 triangular faces which can be subdivided\n * * The parameter `radius` sets the radius size (float) of the icosphere (default 1)\n * * You can set some different icosphere dimensions, for instance to build an ellipsoid, by using the parameters `radiusX`, `radiusY` and `radiusZ` (all by default have the same value than `radius`)\n * * The parameter `subdivisions` sets the number of subdivisions (positive integer, default 4). The more subdivisions, the more faces on the icosphere whatever its size\n * * The parameter `flat` (boolean, default true) gives each side its own normals. Set it to false to get a smooth continuous light reflection on the surface\n * * You can also set the mesh side orientation with the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/polyhedra#icosphere\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param scene defines the hosting scene\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static CreateIcoSphere(name, options, scene) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Creates a decal mesh.\n *.\n * A decal is a mesh usually applied as a model onto the surface of another mesh\n * @param name defines the name of the mesh\n * @param sourceMesh defines the mesh receiving the decal\n * @param position sets the position of the decal in world coordinates\n * @param normal sets the normal of the mesh where the decal is applied onto in world coordinates\n * @param size sets the decal scaling\n * @param angle sets the angle to rotate the decal\n * @returns a new Mesh\n * @deprecated Please use MeshBuilder instead\n */\n static CreateDecal(name, sourceMesh, position, normal, size, angle) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /** Creates a Capsule Mesh\n * @param name defines the name of the mesh.\n * @param options the constructors options used to shape the mesh.\n * @param scene defines the scene the mesh is scoped to.\n * @returns the capsule mesh\n * @see https://doc.babylonjs.com/how_to/capsule_shape\n * @deprecated Please use MeshBuilder instead\n */\n static CreateCapsule(name, options, scene) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n /**\n * Extends a mesh to a Goldberg mesh\n * Warning the mesh to convert MUST be an import of a perviously exported Goldberg mesh\n * @param mesh the mesh to convert\n * @returns the extended mesh\n * @deprecated Please use ExtendMeshToGoldberg instead\n */\n static ExtendToGoldberg(mesh) {\n throw new Error(\"Import MeshBuilder to populate this function\");\n }\n }\n // Consts\n /**\n * Mesh side orientation : usually the external or front surface\n */\n Mesh.FRONTSIDE = VertexData.FRONTSIDE;\n /**\n * Mesh side orientation : usually the internal or back surface\n */\n Mesh.BACKSIDE = VertexData.BACKSIDE;\n /**\n * Mesh side orientation : both internal and external or front and back surfaces\n */\n Mesh.DOUBLESIDE = VertexData.DOUBLESIDE;\n /**\n * Mesh side orientation : by default, `FRONTSIDE`\n */\n Mesh.DEFAULTSIDE = VertexData.DEFAULTSIDE;\n /**\n * Mesh cap setting : no cap\n */\n Mesh.NO_CAP = 0;\n /**\n * Mesh cap setting : one cap at the beginning of the mesh\n */\n Mesh.CAP_START = 1;\n /**\n * Mesh cap setting : one cap at the end of the mesh\n */\n Mesh.CAP_END = 2;\n /**\n * Mesh cap setting : two caps, one at the beginning and one at the end of the mesh\n */\n Mesh.CAP_ALL = 3;\n /**\n * Mesh pattern setting : no flip or rotate\n */\n Mesh.NO_FLIP = 0;\n /**\n * Mesh pattern setting : flip (reflect in y axis) alternate tiles on each row or column\n */\n Mesh.FLIP_TILE = 1;\n /**\n * Mesh pattern setting : rotate (180degs) alternate tiles on each row or column\n */\n Mesh.ROTATE_TILE = 2;\n /**\n * Mesh pattern setting : flip (reflect in y axis) all tiles on alternate rows\n */\n Mesh.FLIP_ROW = 3;\n /**\n * Mesh pattern setting : rotate (180degs) all tiles on alternate rows\n */\n Mesh.ROTATE_ROW = 4;\n /**\n * Mesh pattern setting : flip and rotate alternate tiles on each row or column\n */\n Mesh.FLIP_N_ROTATE_TILE = 5;\n /**\n * Mesh pattern setting : rotate pattern and rotate\n */\n Mesh.FLIP_N_ROTATE_ROW = 6;\n /**\n * Mesh tile positioning : part tiles same on left/right or top/bottom\n */\n Mesh.CENTER = 0;\n /**\n * Mesh tile positioning : part tiles on left\n */\n Mesh.LEFT = 1;\n /**\n * Mesh tile positioning : part tiles on right\n */\n Mesh.RIGHT = 2;\n /**\n * Mesh tile positioning : part tiles on top\n */\n Mesh.TOP = 3;\n /**\n * Mesh tile positioning : part tiles on bottom\n */\n Mesh.BOTTOM = 4;\n /**\n * Indicates that the instanced meshes should be sorted from back to front before rendering if their material is transparent\n */\n Mesh.INSTANCEDMESH_SORT_TRANSPARENT = false;\n // Statics\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n Mesh._GroundMeshParser = (parsedMesh, scene) => {\n throw _WarnImport(\"GroundMesh\");\n };\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n Mesh._GoldbergMeshParser = (parsedMesh, scene) => {\n throw _WarnImport(\"GoldbergMesh\");\n };\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n Mesh._LinesMeshParser = (parsedMesh, scene) => {\n throw _WarnImport(\"LinesMesh\");\n };\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n Mesh._GreasedLineMeshParser = (parsedMesh, scene) => {\n throw _WarnImport(\"GreasedLineMesh\");\n };\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n Mesh._TrailMeshParser = (parsedMesh, scene) => {\n throw _WarnImport(\"TrailMesh\");\n };\n RegisterClass(\"BABYLON.Mesh\", Mesh);\n\n /**\n * The autoRotation behavior (AutoRotationBehavior) is designed to create a smooth rotation of an ArcRotateCamera when there is no user interaction.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#autorotation-behavior\n */\n class AutoRotationBehavior {\n constructor() {\n this._zoomStopsAnimation = false;\n this._idleRotationSpeed = 0.05;\n this._idleRotationWaitTime = 2000;\n this._idleRotationSpinupTime = 2000;\n this.targetAlpha = null;\n this._isPointerDown = false;\n this._lastFrameTime = null;\n this._lastInteractionTime = -Infinity;\n this._cameraRotationSpeed = 0;\n this._lastFrameRadius = 0;\n }\n /**\n * Gets the name of the behavior.\n */\n get name() {\n return \"AutoRotation\";\n }\n /**\n * Sets the flag that indicates if user zooming should stop animation.\n */\n set zoomStopsAnimation(flag) {\n this._zoomStopsAnimation = flag;\n }\n /**\n * Gets the flag that indicates if user zooming should stop animation.\n */\n get zoomStopsAnimation() {\n return this._zoomStopsAnimation;\n }\n /**\n * Sets the default speed at which the camera rotates around the model.\n */\n set idleRotationSpeed(speed) {\n this._idleRotationSpeed = speed;\n }\n /**\n * Gets the default speed at which the camera rotates around the model.\n */\n get idleRotationSpeed() {\n return this._idleRotationSpeed;\n }\n /**\n * Sets the time (in milliseconds) to wait after user interaction before the camera starts rotating.\n */\n set idleRotationWaitTime(time) {\n this._idleRotationWaitTime = time;\n }\n /**\n * Gets the time (milliseconds) to wait after user interaction before the camera starts rotating.\n */\n get idleRotationWaitTime() {\n return this._idleRotationWaitTime;\n }\n /**\n * Sets the time (milliseconds) to take to spin up to the full idle rotation speed.\n */\n set idleRotationSpinupTime(time) {\n this._idleRotationSpinupTime = time;\n }\n /**\n * Gets the time (milliseconds) to take to spin up to the full idle rotation speed.\n */\n get idleRotationSpinupTime() {\n return this._idleRotationSpinupTime;\n }\n /**\n * Gets a value indicating if the camera is currently rotating because of this behavior\n */\n get rotationInProgress() {\n return Math.abs(this._cameraRotationSpeed) > 0;\n }\n /**\n * Initializes the behavior.\n */\n init() {\n // Do nothing\n }\n /**\n * Attaches the behavior to its arc rotate camera.\n * @param camera Defines the camera to attach the behavior to\n */\n attach(camera) {\n this._attachedCamera = camera;\n const scene = this._attachedCamera.getScene();\n this._onPrePointerObservableObserver = scene.onPrePointerObservable.add((pointerInfoPre) => {\n if (pointerInfoPre.type === PointerEventTypes.POINTERDOWN) {\n this._isPointerDown = true;\n return;\n }\n if (pointerInfoPre.type === PointerEventTypes.POINTERUP) {\n this._isPointerDown = false;\n }\n });\n this._onAfterCheckInputsObserver = camera.onAfterCheckInputsObservable.add(() => {\n if (this._reachTargetAlpha()) {\n return;\n }\n const now = PrecisionDate.Now;\n let dt = 0;\n if (this._lastFrameTime != null) {\n dt = now - this._lastFrameTime;\n }\n this._lastFrameTime = now;\n // Stop the animation if there is user interaction and the animation should stop for this interaction\n this._applyUserInteraction();\n const timeToRotation = now - this._lastInteractionTime - this._idleRotationWaitTime;\n const scale = Math.max(Math.min(timeToRotation / this._idleRotationSpinupTime, 1), 0);\n this._cameraRotationSpeed = this._idleRotationSpeed * scale;\n // Step camera rotation by rotation speed\n if (this._attachedCamera) {\n this._attachedCamera.alpha -= this._cameraRotationSpeed * (dt / 1000);\n }\n });\n }\n /**\n * Detaches the behavior from its current arc rotate camera.\n */\n detach() {\n if (!this._attachedCamera) {\n return;\n }\n const scene = this._attachedCamera.getScene();\n if (this._onPrePointerObservableObserver) {\n scene.onPrePointerObservable.remove(this._onPrePointerObservableObserver);\n }\n this._attachedCamera.onAfterCheckInputsObservable.remove(this._onAfterCheckInputsObserver);\n this._attachedCamera = null;\n }\n /**\n * Force-reset the last interaction time\n * @param customTime an optional time that will be used instead of the current last interaction time. For example `Date.now()`\n */\n resetLastInteractionTime(customTime) {\n this._lastInteractionTime = customTime !== null && customTime !== void 0 ? customTime : PrecisionDate.Now;\n }\n /**\n * Returns true if camera alpha reaches the target alpha\n * @returns true if camera alpha reaches the target alpha\n */\n _reachTargetAlpha() {\n if (this._attachedCamera && this.targetAlpha) {\n return Math.abs(this._attachedCamera.alpha - this.targetAlpha) < Epsilon;\n }\n return false;\n }\n /**\n * Returns true if user is scrolling.\n * @returns true if user is scrolling.\n */\n _userIsZooming() {\n if (!this._attachedCamera) {\n return false;\n }\n return this._attachedCamera.inertialRadiusOffset !== 0;\n }\n _shouldAnimationStopForInteraction() {\n if (!this._attachedCamera) {\n return false;\n }\n let zoomHasHitLimit = false;\n if (this._lastFrameRadius === this._attachedCamera.radius && this._attachedCamera.inertialRadiusOffset !== 0) {\n zoomHasHitLimit = true;\n }\n // Update the record of previous radius - works as an approx. indicator of hitting radius limits\n this._lastFrameRadius = this._attachedCamera.radius;\n return this._zoomStopsAnimation ? zoomHasHitLimit : this._userIsZooming();\n }\n /**\n * Applies any current user interaction to the camera. Takes into account maximum alpha rotation.\n */\n _applyUserInteraction() {\n if (this._userIsMoving() && !this._shouldAnimationStopForInteraction()) {\n this._lastInteractionTime = PrecisionDate.Now;\n }\n }\n // Tools\n _userIsMoving() {\n if (!this._attachedCamera) {\n return false;\n }\n return (this._attachedCamera.inertialAlphaOffset !== 0 ||\n this._attachedCamera.inertialBetaOffset !== 0 ||\n this._attachedCamera.inertialRadiusOffset !== 0 ||\n this._attachedCamera.inertialPanningX !== 0 ||\n this._attachedCamera.inertialPanningY !== 0 ||\n this._isPointerDown);\n }\n }\n\n /**\n * Defines potential orientation for back face culling\n */\n var Orientation;\n (function (Orientation) {\n /**\n * Clockwise\n */\n Orientation[Orientation[\"CW\"] = 0] = \"CW\";\n /** Counter clockwise */\n Orientation[Orientation[\"CCW\"] = 1] = \"CCW\";\n })(Orientation || (Orientation = {}));\n /**\n * Defines angle representation\n */\n class Angle {\n /**\n * Creates an Angle object of \"radians\" radians (float).\n * @param radians the angle in radians\n */\n constructor(radians) {\n this._radians = radians;\n if (this._radians < 0.0) {\n this._radians += 2.0 * Math.PI;\n }\n }\n /**\n * Get value in degrees\n * @returns the Angle value in degrees (float)\n */\n degrees() {\n return (this._radians * 180.0) / Math.PI;\n }\n /**\n * Get value in radians\n * @returns the Angle value in radians (float)\n */\n radians() {\n return this._radians;\n }\n /**\n * Gets a new Angle object valued with the gradient angle, in radians, of the line joining two points\n * @param a defines first point as the origin\n * @param b defines point\n * @returns a new Angle\n */\n static BetweenTwoPoints(a, b) {\n const delta = b.subtract(a);\n const theta = Math.atan2(delta.y, delta.x);\n return new Angle(theta);\n }\n /**\n * Gets a new Angle object from the given float in radians\n * @param radians defines the angle value in radians\n * @returns a new Angle\n */\n static FromRadians(radians) {\n return new Angle(radians);\n }\n /**\n * Gets a new Angle object from the given float in degrees\n * @param degrees defines the angle value in degrees\n * @returns a new Angle\n */\n static FromDegrees(degrees) {\n return new Angle((degrees * Math.PI) / 180.0);\n }\n }\n /**\n * This represents an arc in a 2d space.\n */\n class Arc2 {\n /**\n * Creates an Arc object from the three given points : start, middle and end.\n * @param startPoint Defines the start point of the arc\n * @param midPoint Defines the middle point of the arc\n * @param endPoint Defines the end point of the arc\n */\n constructor(\n /** Defines the start point of the arc */\n startPoint, \n /** Defines the mid point of the arc */\n midPoint, \n /** Defines the end point of the arc */\n endPoint) {\n this.startPoint = startPoint;\n this.midPoint = midPoint;\n this.endPoint = endPoint;\n const temp = Math.pow(midPoint.x, 2) + Math.pow(midPoint.y, 2);\n const startToMid = (Math.pow(startPoint.x, 2) + Math.pow(startPoint.y, 2) - temp) / 2;\n const midToEnd = (temp - Math.pow(endPoint.x, 2) - Math.pow(endPoint.y, 2)) / 2;\n const det = (startPoint.x - midPoint.x) * (midPoint.y - endPoint.y) - (midPoint.x - endPoint.x) * (startPoint.y - midPoint.y);\n this.centerPoint = new Vector2((startToMid * (midPoint.y - endPoint.y) - midToEnd * (startPoint.y - midPoint.y)) / det, ((startPoint.x - midPoint.x) * midToEnd - (midPoint.x - endPoint.x) * startToMid) / det);\n this.radius = this.centerPoint.subtract(this.startPoint).length();\n this.startAngle = Angle.BetweenTwoPoints(this.centerPoint, this.startPoint);\n const a1 = this.startAngle.degrees();\n let a2 = Angle.BetweenTwoPoints(this.centerPoint, this.midPoint).degrees();\n let a3 = Angle.BetweenTwoPoints(this.centerPoint, this.endPoint).degrees();\n // angles correction\n if (a2 - a1 > +180.0) {\n a2 -= 360.0;\n }\n if (a2 - a1 < -180.0) {\n a2 += 360.0;\n }\n if (a3 - a2 > +180.0) {\n a3 -= 360.0;\n }\n if (a3 - a2 < -180.0) {\n a3 += 360.0;\n }\n this.orientation = a2 - a1 < 0 ? Orientation.CW : Orientation.CCW;\n this.angle = Angle.FromDegrees(this.orientation === Orientation.CW ? a1 - a3 : a3 - a1);\n }\n }\n /**\n * Represents a 2D path made up of multiple 2D points\n */\n class Path2 {\n /**\n * Creates a Path2 object from the starting 2D coordinates x and y.\n * @param x the starting points x value\n * @param y the starting points y value\n */\n constructor(x, y) {\n this._points = new Array();\n this._length = 0.0;\n /**\n * If the path start and end point are the same\n */\n this.closed = false;\n this._points.push(new Vector2(x, y));\n }\n /**\n * Adds a new segment until the given coordinates (x, y) to the current Path2.\n * @param x the added points x value\n * @param y the added points y value\n * @returns the updated Path2.\n */\n addLineTo(x, y) {\n if (this.closed) {\n return this;\n }\n const newPoint = new Vector2(x, y);\n const previousPoint = this._points[this._points.length - 1];\n this._points.push(newPoint);\n this._length += newPoint.subtract(previousPoint).length();\n return this;\n }\n /**\n * Adds _numberOfSegments_ segments according to the arc definition (middle point coordinates, end point coordinates, the arc start point being the current Path2 last point) to the current Path2.\n * @param midX middle point x value\n * @param midY middle point y value\n * @param endX end point x value\n * @param endY end point y value\n * @param numberOfSegments (default: 36)\n * @returns the updated Path2.\n */\n addArcTo(midX, midY, endX, endY, numberOfSegments = 36) {\n if (this.closed) {\n return this;\n }\n const startPoint = this._points[this._points.length - 1];\n const midPoint = new Vector2(midX, midY);\n const endPoint = new Vector2(endX, endY);\n const arc = new Arc2(startPoint, midPoint, endPoint);\n let increment = arc.angle.radians() / numberOfSegments;\n if (arc.orientation === Orientation.CW) {\n increment *= -1;\n }\n let currentAngle = arc.startAngle.radians() + increment;\n for (let i = 0; i < numberOfSegments; i++) {\n const x = Math.cos(currentAngle) * arc.radius + arc.centerPoint.x;\n const y = Math.sin(currentAngle) * arc.radius + arc.centerPoint.y;\n this.addLineTo(x, y);\n currentAngle += increment;\n }\n return this;\n }\n /**\n * Adds _numberOfSegments_ segments according to the quadratic curve definition to the current Path2.\n * @param controlX control point x value\n * @param controlY control point y value\n * @param endX end point x value\n * @param endY end point y value\n * @param numberOfSegments (default: 36)\n * @returns the updated Path2.\n */\n addQuadraticCurveTo(controlX, controlY, endX, endY, numberOfSegments = 36) {\n if (this.closed) {\n return this;\n }\n const equation = (t, val0, val1, val2) => {\n const res = (1.0 - t) * (1.0 - t) * val0 + 2.0 * t * (1.0 - t) * val1 + t * t * val2;\n return res;\n };\n const startPoint = this._points[this._points.length - 1];\n for (let i = 0; i <= numberOfSegments; i++) {\n const step = i / numberOfSegments;\n const x = equation(step, startPoint.x, controlX, endX);\n const y = equation(step, startPoint.y, controlY, endY);\n this.addLineTo(x, y);\n }\n return this;\n }\n /**\n * Adds _numberOfSegments_ segments according to the bezier curve definition to the current Path2.\n * @param originTangentX tangent vector at the origin point x value\n * @param originTangentY tangent vector at the origin point y value\n * @param destinationTangentX tangent vector at the destination point x value\n * @param destinationTangentY tangent vector at the destination point y value\n * @param endX end point x value\n * @param endY end point y value\n * @param numberOfSegments (default: 36)\n * @returns the updated Path2.\n */\n addBezierCurveTo(originTangentX, originTangentY, destinationTangentX, destinationTangentY, endX, endY, numberOfSegments = 36) {\n if (this.closed) {\n return this;\n }\n const equation = (t, val0, val1, val2, val3) => {\n const res = (1.0 - t) * (1.0 - t) * (1.0 - t) * val0 + 3.0 * t * (1.0 - t) * (1.0 - t) * val1 + 3.0 * t * t * (1.0 - t) * val2 + t * t * t * val3;\n return res;\n };\n const startPoint = this._points[this._points.length - 1];\n for (let i = 0; i <= numberOfSegments; i++) {\n const step = i / numberOfSegments;\n const x = equation(step, startPoint.x, originTangentX, destinationTangentX, endX);\n const y = equation(step, startPoint.y, originTangentY, destinationTangentY, endY);\n this.addLineTo(x, y);\n }\n return this;\n }\n /**\n * Defines if a given point is inside the polygon defines by the path\n * @param point defines the point to test\n * @returns true if the point is inside\n */\n isPointInside(point) {\n let isInside = false;\n const count = this._points.length;\n for (let p = count - 1, q = 0; q < count; p = q++) {\n let edgeLow = this._points[p];\n let edgeHigh = this._points[q];\n let edgeDx = edgeHigh.x - edgeLow.x;\n let edgeDy = edgeHigh.y - edgeLow.y;\n if (Math.abs(edgeDy) > Number.EPSILON) {\n // Not parallel\n if (edgeDy < 0) {\n edgeLow = this._points[q];\n edgeDx = -edgeDx;\n edgeHigh = this._points[p];\n edgeDy = -edgeDy;\n }\n if (point.y < edgeLow.y || point.y > edgeHigh.y) {\n continue;\n }\n if (point.y === edgeLow.y && point.x === edgeLow.x) {\n return true;\n }\n else {\n const perpEdge = edgeDy * (point.x - edgeLow.x) - edgeDx * (point.y - edgeLow.y);\n if (perpEdge === 0) {\n return true;\n }\n if (perpEdge < 0) {\n continue;\n }\n isInside = !isInside;\n }\n }\n else {\n // parallel or collinear\n if (point.y !== edgeLow.y) {\n continue;\n }\n if ((edgeHigh.x <= point.x && point.x <= edgeLow.x) || (edgeLow.x <= point.x && point.x <= edgeHigh.x)) {\n return true;\n }\n }\n }\n return isInside;\n }\n /**\n * Closes the Path2.\n * @returns the Path2.\n */\n close() {\n this.closed = true;\n return this;\n }\n /**\n * Gets the sum of the distance between each sequential point in the path\n * @returns the Path2 total length (float).\n */\n length() {\n let result = this._length;\n if (this.closed) {\n const lastPoint = this._points[this._points.length - 1];\n const firstPoint = this._points[0];\n result += firstPoint.subtract(lastPoint).length();\n }\n return result;\n }\n /**\n * Gets the area of the polygon defined by the path\n * @returns area value\n */\n area() {\n const n = this._points.length;\n let value = 0.0;\n for (let p = n - 1, q = 0; q < n; p = q++) {\n value += this._points[p].x * this._points[q].y - this._points[q].x * this._points[p].y;\n }\n return value * 0.5;\n }\n /**\n * Gets the points which construct the path\n * @returns the Path2 internal array of points.\n */\n getPoints() {\n return this._points;\n }\n /**\n * Retreives the point at the distance aways from the starting point\n * @param normalizedLengthPosition the length along the path to retrieve the point from\n * @returns a new Vector2 located at a percentage of the Path2 total length on this path.\n */\n getPointAtLengthPosition(normalizedLengthPosition) {\n if (normalizedLengthPosition < 0 || normalizedLengthPosition > 1) {\n return Vector2.Zero();\n }\n const lengthPosition = normalizedLengthPosition * this.length();\n let previousOffset = 0;\n for (let i = 0; i < this._points.length; i++) {\n const j = (i + 1) % this._points.length;\n const a = this._points[i];\n const b = this._points[j];\n const bToA = b.subtract(a);\n const nextOffset = bToA.length() + previousOffset;\n if (lengthPosition >= previousOffset && lengthPosition <= nextOffset) {\n const dir = bToA.normalize();\n const localOffset = lengthPosition - previousOffset;\n return new Vector2(a.x + dir.x * localOffset, a.y + dir.y * localOffset);\n }\n previousOffset = nextOffset;\n }\n return Vector2.Zero();\n }\n /**\n * Creates a new path starting from an x and y position\n * @param x starting x value\n * @param y starting y value\n * @returns a new Path2 starting at the coordinates (x, y).\n */\n static StartingAt(x, y) {\n return new Path2(x, y);\n }\n }\n /**\n * Represents a 3D path made up of multiple 3D points\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/path3D\n */\n class Path3D {\n /**\n * new Path3D(path, normal, raw)\n * Creates a Path3D. A Path3D is a logical math object, so not a mesh.\n * please read the description in the tutorial : https://doc.babylonjs.com/features/featuresDeepDive/mesh/path3D\n * @param path an array of Vector3, the curve axis of the Path3D\n * @param firstNormal (options) Vector3, the first wanted normal to the curve. Ex (0, 1, 0) for a vertical normal.\n * @param raw (optional, default false) : boolean, if true the returned Path3D isn't normalized. Useful to depict path acceleration or speed.\n * @param alignTangentsWithPath (optional, default false) : boolean, if true the tangents will be aligned with the path.\n */\n constructor(\n /**\n * an array of Vector3, the curve axis of the Path3D\n */\n path, firstNormal = null, raw, alignTangentsWithPath = false) {\n this.path = path;\n this._curve = new Array();\n this._distances = new Array();\n this._tangents = new Array();\n this._normals = new Array();\n this._binormals = new Array();\n // holds interpolated point data\n this._pointAtData = {\n id: 0,\n point: Vector3.Zero(),\n previousPointArrayIndex: 0,\n position: 0,\n subPosition: 0,\n interpolateReady: false,\n interpolationMatrix: Matrix.Identity(),\n };\n for (let p = 0; p < path.length; p++) {\n this._curve[p] = path[p].clone(); // hard copy\n }\n this._raw = raw || false;\n this._alignTangentsWithPath = alignTangentsWithPath;\n this._compute(firstNormal, alignTangentsWithPath);\n }\n /**\n * Returns the Path3D array of successive Vector3 designing its curve.\n * @returns the Path3D array of successive Vector3 designing its curve.\n */\n getCurve() {\n return this._curve;\n }\n /**\n * Returns the Path3D array of successive Vector3 designing its curve.\n * @returns the Path3D array of successive Vector3 designing its curve.\n */\n getPoints() {\n return this._curve;\n }\n /**\n * @returns the computed length (float) of the path.\n */\n length() {\n return this._distances[this._distances.length - 1];\n }\n /**\n * Returns an array populated with tangent vectors on each Path3D curve point.\n * @returns an array populated with tangent vectors on each Path3D curve point.\n */\n getTangents() {\n return this._tangents;\n }\n /**\n * Returns an array populated with normal vectors on each Path3D curve point.\n * @returns an array populated with normal vectors on each Path3D curve point.\n */\n getNormals() {\n return this._normals;\n }\n /**\n * Returns an array populated with binormal vectors on each Path3D curve point.\n * @returns an array populated with binormal vectors on each Path3D curve point.\n */\n getBinormals() {\n return this._binormals;\n }\n /**\n * Returns an array populated with distances (float) of the i-th point from the first curve point.\n * @returns an array populated with distances (float) of the i-th point from the first curve point.\n */\n getDistances() {\n return this._distances;\n }\n /**\n * Returns an interpolated point along this path\n * @param position the position of the point along this path, from 0.0 to 1.0\n * @returns a new Vector3 as the point\n */\n getPointAt(position) {\n return this._updatePointAtData(position).point;\n }\n /**\n * Returns the tangent vector of an interpolated Path3D curve point at the specified position along this path.\n * @param position the position of the point along this path, from 0.0 to 1.0\n * @param interpolated (optional, default false) : boolean, if true returns an interpolated tangent instead of the tangent of the previous path point.\n * @returns a tangent vector corresponding to the interpolated Path3D curve point, if not interpolated, the tangent is taken from the precomputed tangents array.\n */\n getTangentAt(position, interpolated = false) {\n this._updatePointAtData(position, interpolated);\n return interpolated ? Vector3.TransformCoordinates(Vector3.Forward(), this._pointAtData.interpolationMatrix) : this._tangents[this._pointAtData.previousPointArrayIndex];\n }\n /**\n * Returns the tangent vector of an interpolated Path3D curve point at the specified position along this path.\n * @param position the position of the point along this path, from 0.0 to 1.0\n * @param interpolated (optional, default false) : boolean, if true returns an interpolated normal instead of the normal of the previous path point.\n * @returns a normal vector corresponding to the interpolated Path3D curve point, if not interpolated, the normal is taken from the precomputed normals array.\n */\n getNormalAt(position, interpolated = false) {\n this._updatePointAtData(position, interpolated);\n return interpolated ? Vector3.TransformCoordinates(Vector3.Right(), this._pointAtData.interpolationMatrix) : this._normals[this._pointAtData.previousPointArrayIndex];\n }\n /**\n * Returns the binormal vector of an interpolated Path3D curve point at the specified position along this path.\n * @param position the position of the point along this path, from 0.0 to 1.0\n * @param interpolated (optional, default false) : boolean, if true returns an interpolated binormal instead of the binormal of the previous path point.\n * @returns a binormal vector corresponding to the interpolated Path3D curve point, if not interpolated, the binormal is taken from the precomputed binormals array.\n */\n getBinormalAt(position, interpolated = false) {\n this._updatePointAtData(position, interpolated);\n return interpolated ? Vector3.TransformCoordinates(Vector3.UpReadOnly, this._pointAtData.interpolationMatrix) : this._binormals[this._pointAtData.previousPointArrayIndex];\n }\n /**\n * Returns the distance (float) of an interpolated Path3D curve point at the specified position along this path.\n * @param position the position of the point along this path, from 0.0 to 1.0\n * @returns the distance of the interpolated Path3D curve point at the specified position along this path.\n */\n getDistanceAt(position) {\n return this.length() * position;\n }\n /**\n * Returns the array index of the previous point of an interpolated point along this path\n * @param position the position of the point to interpolate along this path, from 0.0 to 1.0\n * @returns the array index\n */\n getPreviousPointIndexAt(position) {\n this._updatePointAtData(position);\n return this._pointAtData.previousPointArrayIndex;\n }\n /**\n * Returns the position of an interpolated point relative to the two path points it lies between, from 0.0 (point A) to 1.0 (point B)\n * @param position the position of the point to interpolate along this path, from 0.0 to 1.0\n * @returns the sub position\n */\n getSubPositionAt(position) {\n this._updatePointAtData(position);\n return this._pointAtData.subPosition;\n }\n /**\n * Returns the position of the closest virtual point on this path to an arbitrary Vector3, from 0.0 to 1.0\n * @param target the vector of which to get the closest position to\n * @returns the position of the closest virtual point on this path to the target vector\n */\n getClosestPositionTo(target) {\n let smallestDistance = Number.MAX_VALUE;\n let closestPosition = 0.0;\n for (let i = 0; i < this._curve.length - 1; i++) {\n const point = this._curve[i + 0];\n const tangent = this._curve[i + 1].subtract(point).normalize();\n const subLength = this._distances[i + 1] - this._distances[i + 0];\n const subPosition = Math.min((Math.max(Vector3.Dot(tangent, target.subtract(point).normalize()), 0.0) * Vector3.Distance(point, target)) / subLength, 1.0);\n const distance = Vector3.Distance(point.add(tangent.scale(subPosition * subLength)), target);\n if (distance < smallestDistance) {\n smallestDistance = distance;\n closestPosition = (this._distances[i + 0] + subLength * subPosition) / this.length();\n }\n }\n return closestPosition;\n }\n /**\n * Returns a sub path (slice) of this path\n * @param start the position of the fist path point, from 0.0 to 1.0, or a negative value, which will get wrapped around from the end of the path to 0.0 to 1.0 values\n * @param end the position of the last path point, from 0.0 to 1.0, or a negative value, which will get wrapped around from the end of the path to 0.0 to 1.0 values\n * @returns a sub path (slice) of this path\n */\n slice(start = 0.0, end = 1.0) {\n if (start < 0.0) {\n start = 1 - ((start * -1.0) % 1.0);\n }\n if (end < 0.0) {\n end = 1 - ((end * -1.0) % 1.0);\n }\n if (start > end) {\n const _start = start;\n start = end;\n end = _start;\n }\n const curvePoints = this.getCurve();\n const startPoint = this.getPointAt(start);\n let startIndex = this.getPreviousPointIndexAt(start);\n const endPoint = this.getPointAt(end);\n const endIndex = this.getPreviousPointIndexAt(end) + 1;\n const slicePoints = [];\n if (start !== 0.0) {\n startIndex++;\n slicePoints.push(startPoint);\n }\n slicePoints.push(...curvePoints.slice(startIndex, endIndex));\n if (end !== 1.0 || start === 1.0) {\n slicePoints.push(endPoint);\n }\n return new Path3D(slicePoints, this.getNormalAt(start), this._raw, this._alignTangentsWithPath);\n }\n /**\n * Forces the Path3D tangent, normal, binormal and distance recomputation.\n * @param path path which all values are copied into the curves points\n * @param firstNormal which should be projected onto the curve\n * @param alignTangentsWithPath (optional, default false) : boolean, if true the tangents will be aligned with the path\n * @returns the same object updated.\n */\n update(path, firstNormal = null, alignTangentsWithPath = false) {\n for (let p = 0; p < path.length; p++) {\n this._curve[p].x = path[p].x;\n this._curve[p].y = path[p].y;\n this._curve[p].z = path[p].z;\n }\n this._compute(firstNormal, alignTangentsWithPath);\n return this;\n }\n // private function compute() : computes tangents, normals and binormals\n _compute(firstNormal, alignTangentsWithPath = false) {\n const l = this._curve.length;\n if (l < 2) {\n return;\n }\n // first and last tangents\n this._tangents[0] = this._getFirstNonNullVector(0);\n if (!this._raw) {\n this._tangents[0].normalize();\n }\n this._tangents[l - 1] = this._curve[l - 1].subtract(this._curve[l - 2]);\n if (!this._raw) {\n this._tangents[l - 1].normalize();\n }\n // normals and binormals at first point : arbitrary vector with _normalVector()\n const tg0 = this._tangents[0];\n const pp0 = this._normalVector(tg0, firstNormal);\n this._normals[0] = pp0;\n if (!this._raw) {\n this._normals[0].normalize();\n }\n this._binormals[0] = Vector3.Cross(tg0, this._normals[0]);\n if (!this._raw) {\n this._binormals[0].normalize();\n }\n this._distances[0] = 0.0;\n // normals and binormals : next points\n let prev; // previous vector (segment)\n let cur; // current vector (segment)\n let curTang; // current tangent\n // previous normal\n let prevNor; // previous normal\n let prevBinor; // previous binormal\n for (let i = 1; i < l; i++) {\n // tangents\n prev = this._getLastNonNullVector(i);\n if (i < l - 1) {\n cur = this._getFirstNonNullVector(i);\n this._tangents[i] = alignTangentsWithPath ? cur : prev.add(cur);\n this._tangents[i].normalize();\n }\n this._distances[i] = this._distances[i - 1] + this._curve[i].subtract(this._curve[i - 1]).length();\n // normals and binormals\n // http://www.cs.cmu.edu/afs/andrew/scs/cs/15-462/web/old/asst2camera.html\n curTang = this._tangents[i];\n prevBinor = this._binormals[i - 1];\n this._normals[i] = Vector3.Cross(prevBinor, curTang);\n if (!this._raw) {\n if (this._normals[i].length() === 0) {\n prevNor = this._normals[i - 1];\n this._normals[i] = prevNor.clone();\n }\n else {\n this._normals[i].normalize();\n }\n }\n this._binormals[i] = Vector3.Cross(curTang, this._normals[i]);\n if (!this._raw) {\n this._binormals[i].normalize();\n }\n }\n this._pointAtData.id = NaN;\n }\n // private function getFirstNonNullVector(index)\n // returns the first non null vector from index : curve[index + N].subtract(curve[index])\n _getFirstNonNullVector(index) {\n let i = 1;\n let nNVector = this._curve[index + i].subtract(this._curve[index]);\n while (nNVector.length() === 0 && index + i + 1 < this._curve.length) {\n i++;\n nNVector = this._curve[index + i].subtract(this._curve[index]);\n }\n return nNVector;\n }\n // private function getLastNonNullVector(index)\n // returns the last non null vector from index : curve[index].subtract(curve[index - N])\n _getLastNonNullVector(index) {\n let i = 1;\n let nLVector = this._curve[index].subtract(this._curve[index - i]);\n while (nLVector.length() === 0 && index > i + 1) {\n i++;\n nLVector = this._curve[index].subtract(this._curve[index - i]);\n }\n return nLVector;\n }\n // private function normalVector(v0, vt, va) :\n // returns an arbitrary point in the plane defined by the point v0 and the vector vt orthogonal to this plane\n // if va is passed, it returns the va projection on the plane orthogonal to vt at the point v0\n _normalVector(vt, va) {\n let normal0;\n let tgl = vt.length();\n if (tgl === 0.0) {\n tgl = 1.0;\n }\n if (va === undefined || va === null) {\n let point;\n if (!Scalar.WithinEpsilon(Math.abs(vt.y) / tgl, 1.0, Epsilon)) {\n // search for a point in the plane\n point = new Vector3(0.0, -1.0, 0.0);\n }\n else if (!Scalar.WithinEpsilon(Math.abs(vt.x) / tgl, 1.0, Epsilon)) {\n point = new Vector3(1.0, 0.0, 0.0);\n }\n else if (!Scalar.WithinEpsilon(Math.abs(vt.z) / tgl, 1.0, Epsilon)) {\n point = new Vector3(0.0, 0.0, 1.0);\n }\n else {\n point = Vector3.Zero();\n }\n normal0 = Vector3.Cross(vt, point);\n }\n else {\n normal0 = Vector3.Cross(vt, va);\n Vector3.CrossToRef(normal0, vt, normal0);\n }\n normal0.normalize();\n return normal0;\n }\n /**\n * Updates the point at data for an interpolated point along this curve\n * @param position the position of the point along this curve, from 0.0 to 1.0\n * @param interpolateTNB\n * @interpolateTNB whether to compute the interpolated tangent, normal and binormal\n * @returns the (updated) point at data\n */\n _updatePointAtData(position, interpolateTNB = false) {\n // set an id for caching the result\n if (this._pointAtData.id === position) {\n if (!this._pointAtData.interpolateReady) {\n this._updateInterpolationMatrix();\n }\n return this._pointAtData;\n }\n else {\n this._pointAtData.id = position;\n }\n const curvePoints = this.getPoints();\n // clamp position between 0.0 and 1.0\n if (position <= 0.0) {\n return this._setPointAtData(0.0, 0.0, curvePoints[0], 0, interpolateTNB);\n }\n else if (position >= 1.0) {\n return this._setPointAtData(1.0, 1.0, curvePoints[curvePoints.length - 1], curvePoints.length - 1, interpolateTNB);\n }\n let previousPoint = curvePoints[0];\n let currentPoint;\n let currentLength = 0.0;\n const targetLength = position * this.length();\n for (let i = 1; i < curvePoints.length; i++) {\n currentPoint = curvePoints[i];\n const distance = Vector3.Distance(previousPoint, currentPoint);\n currentLength += distance;\n if (currentLength === targetLength) {\n return this._setPointAtData(position, 1.0, currentPoint, i, interpolateTNB);\n }\n else if (currentLength > targetLength) {\n const toLength = currentLength - targetLength;\n const diff = toLength / distance;\n const dir = previousPoint.subtract(currentPoint);\n const point = currentPoint.add(dir.scaleInPlace(diff));\n return this._setPointAtData(position, 1 - diff, point, i - 1, interpolateTNB);\n }\n previousPoint = currentPoint;\n }\n return this._pointAtData;\n }\n /**\n * Updates the point at data from the specified parameters\n * @param position where along the path the interpolated point is, from 0.0 to 1.0\n * @param subPosition\n * @param point the interpolated point\n * @param parentIndex the index of an existing curve point that is on, or else positionally the first behind, the interpolated point\n * @param interpolateTNB\n */\n _setPointAtData(position, subPosition, point, parentIndex, interpolateTNB) {\n this._pointAtData.point = point;\n this._pointAtData.position = position;\n this._pointAtData.subPosition = subPosition;\n this._pointAtData.previousPointArrayIndex = parentIndex;\n this._pointAtData.interpolateReady = interpolateTNB;\n if (interpolateTNB) {\n this._updateInterpolationMatrix();\n }\n return this._pointAtData;\n }\n /**\n * Updates the point at interpolation matrix for the tangents, normals and binormals\n */\n _updateInterpolationMatrix() {\n this._pointAtData.interpolationMatrix = Matrix.Identity();\n const parentIndex = this._pointAtData.previousPointArrayIndex;\n if (parentIndex !== this._tangents.length - 1) {\n const index = parentIndex + 1;\n const tangentFrom = this._tangents[parentIndex].clone();\n const normalFrom = this._normals[parentIndex].clone();\n const binormalFrom = this._binormals[parentIndex].clone();\n const tangentTo = this._tangents[index].clone();\n const normalTo = this._normals[index].clone();\n const binormalTo = this._binormals[index].clone();\n const quatFrom = Quaternion.RotationQuaternionFromAxis(normalFrom, binormalFrom, tangentFrom);\n const quatTo = Quaternion.RotationQuaternionFromAxis(normalTo, binormalTo, tangentTo);\n const quatAt = Quaternion.Slerp(quatFrom, quatTo, this._pointAtData.subPosition);\n quatAt.toRotationMatrix(this._pointAtData.interpolationMatrix);\n }\n }\n }\n /**\n * A Curve3 object is a logical object, so not a mesh, to handle curves in the 3D geometric space.\n * A Curve3 is designed from a series of successive Vector3.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/drawCurves\n */\n class Curve3 {\n /**\n * Returns a Curve3 object along a Quadratic Bezier curve : https://doc.babylonjs.com/features/featuresDeepDive/mesh/drawCurves#quadratic-bezier-curve\n * @param v0 (Vector3) the origin point of the Quadratic Bezier\n * @param v1 (Vector3) the control point\n * @param v2 (Vector3) the end point of the Quadratic Bezier\n * @param nbPoints (integer) the wanted number of points in the curve\n * @returns the created Curve3\n */\n static CreateQuadraticBezier(v0, v1, v2, nbPoints) {\n nbPoints = nbPoints > 2 ? nbPoints : 3;\n const bez = new Array();\n const equation = (t, val0, val1, val2) => {\n const res = (1.0 - t) * (1.0 - t) * val0 + 2.0 * t * (1.0 - t) * val1 + t * t * val2;\n return res;\n };\n for (let i = 0; i <= nbPoints; i++) {\n bez.push(new Vector3(equation(i / nbPoints, v0.x, v1.x, v2.x), equation(i / nbPoints, v0.y, v1.y, v2.y), equation(i / nbPoints, v0.z, v1.z, v2.z)));\n }\n return new Curve3(bez);\n }\n /**\n * Returns a Curve3 object along a Cubic Bezier curve : https://doc.babylonjs.com/features/featuresDeepDive/mesh/drawCurves#cubic-bezier-curve\n * @param v0 (Vector3) the origin point of the Cubic Bezier\n * @param v1 (Vector3) the first control point\n * @param v2 (Vector3) the second control point\n * @param v3 (Vector3) the end point of the Cubic Bezier\n * @param nbPoints (integer) the wanted number of points in the curve\n * @returns the created Curve3\n */\n static CreateCubicBezier(v0, v1, v2, v3, nbPoints) {\n nbPoints = nbPoints > 3 ? nbPoints : 4;\n const bez = new Array();\n const equation = (t, val0, val1, val2, val3) => {\n const res = (1.0 - t) * (1.0 - t) * (1.0 - t) * val0 + 3.0 * t * (1.0 - t) * (1.0 - t) * val1 + 3.0 * t * t * (1.0 - t) * val2 + t * t * t * val3;\n return res;\n };\n for (let i = 0; i <= nbPoints; i++) {\n bez.push(new Vector3(equation(i / nbPoints, v0.x, v1.x, v2.x, v3.x), equation(i / nbPoints, v0.y, v1.y, v2.y, v3.y), equation(i / nbPoints, v0.z, v1.z, v2.z, v3.z)));\n }\n return new Curve3(bez);\n }\n /**\n * Returns a Curve3 object along a Hermite Spline curve : https://doc.babylonjs.com/features/featuresDeepDive/mesh/drawCurves#hermite-spline\n * @param p1 (Vector3) the origin point of the Hermite Spline\n * @param t1 (Vector3) the tangent vector at the origin point\n * @param p2 (Vector3) the end point of the Hermite Spline\n * @param t2 (Vector3) the tangent vector at the end point\n * @param nSeg (integer) the number of curve segments or nSeg + 1 points in the array\n * @returns the created Curve3\n */\n static CreateHermiteSpline(p1, t1, p2, t2, nSeg) {\n const hermite = new Array();\n const step = 1.0 / nSeg;\n for (let i = 0; i <= nSeg; i++) {\n hermite.push(Vector3.Hermite(p1, t1, p2, t2, i * step));\n }\n return new Curve3(hermite);\n }\n /**\n * Returns a Curve3 object along a CatmullRom Spline curve :\n * @param points (array of Vector3) the points the spline must pass through. At least, four points required\n * @param nbPoints (integer) the wanted number of points between each curve control points\n * @param closed (boolean) optional with default false, when true forms a closed loop from the points\n * @returns the created Curve3\n */\n static CreateCatmullRomSpline(points, nbPoints, closed) {\n const catmullRom = new Array();\n const step = 1.0 / nbPoints;\n let amount = 0.0;\n if (closed) {\n const pointsCount = points.length;\n for (let i = 0; i < pointsCount; i++) {\n amount = 0;\n for (let c = 0; c < nbPoints; c++) {\n catmullRom.push(Vector3.CatmullRom(points[i % pointsCount], points[(i + 1) % pointsCount], points[(i + 2) % pointsCount], points[(i + 3) % pointsCount], amount));\n amount += step;\n }\n }\n catmullRom.push(catmullRom[0]);\n }\n else {\n const totalPoints = new Array();\n totalPoints.push(points[0].clone());\n Array.prototype.push.apply(totalPoints, points);\n totalPoints.push(points[points.length - 1].clone());\n let i = 0;\n for (; i < totalPoints.length - 3; i++) {\n amount = 0;\n for (let c = 0; c < nbPoints; c++) {\n catmullRom.push(Vector3.CatmullRom(totalPoints[i], totalPoints[i + 1], totalPoints[i + 2], totalPoints[i + 3], amount));\n amount += step;\n }\n }\n i--;\n catmullRom.push(Vector3.CatmullRom(totalPoints[i], totalPoints[i + 1], totalPoints[i + 2], totalPoints[i + 3], amount));\n }\n return new Curve3(catmullRom);\n }\n /**\n * Returns a Curve3 object along an arc through three vector3 points:\n * The three points should not be colinear. When they are the Curve3 is empty.\n * @param first (Vector3) the first point the arc must pass through.\n * @param second (Vector3) the second point the arc must pass through.\n * @param third (Vector3) the third point the arc must pass through.\n * @param steps (number) the larger the number of steps the more detailed the arc.\n * @param closed (boolean) optional with default false, when true forms the chord from the first and third point\n * @param fullCircle Circle (boolean) optional with default false, when true forms the complete circle through the three points\n * @returns the created Curve3\n */\n static ArcThru3Points(first, second, third, steps = 32, closed = false, fullCircle = false) {\n const arc = new Array();\n const vec1 = second.subtract(first);\n const vec2 = third.subtract(second);\n const vec3 = first.subtract(third);\n const zAxis = Vector3.Cross(vec1, vec2);\n const len4 = zAxis.length();\n if (len4 < Math.pow(10, -8)) {\n return new Curve3(arc); // colinear points arc is empty\n }\n const len1_sq = vec1.lengthSquared();\n const len2_sq = vec2.lengthSquared();\n const len3_sq = vec3.lengthSquared();\n const len4_sq = zAxis.lengthSquared();\n const len1 = vec1.length();\n const len2 = vec2.length();\n const len3 = vec3.length();\n const radius = (0.5 * len1 * len2 * len3) / len4;\n const dot1 = Vector3.Dot(vec1, vec3);\n const dot2 = Vector3.Dot(vec1, vec2);\n const dot3 = Vector3.Dot(vec2, vec3);\n const a = (-0.5 * len2_sq * dot1) / len4_sq;\n const b = (-0.5 * len3_sq * dot2) / len4_sq;\n const c = (-0.5 * len1_sq * dot3) / len4_sq;\n const center = first.scale(a).add(second.scale(b)).add(third.scale(c));\n const radiusVec = first.subtract(center);\n const xAxis = radiusVec.normalize();\n const yAxis = Vector3.Cross(zAxis, xAxis).normalize();\n if (fullCircle) {\n const dStep = (2 * Math.PI) / steps;\n for (let theta = 0; theta <= 2 * Math.PI; theta += dStep) {\n arc.push(center.add(xAxis.scale(radius * Math.cos(theta)).add(yAxis.scale(radius * Math.sin(theta)))));\n }\n arc.push(first);\n }\n else {\n const dStep = 1 / steps;\n let theta = 0;\n let point = Vector3.Zero();\n do {\n point = center.add(xAxis.scale(radius * Math.cos(theta)).add(yAxis.scale(radius * Math.sin(theta))));\n arc.push(point);\n theta += dStep;\n } while (!point.equalsWithEpsilon(third, radius * dStep * 1.1));\n arc.push(third);\n if (closed) {\n arc.push(first);\n }\n }\n return new Curve3(arc);\n }\n /**\n * A Curve3 object is a logical object, so not a mesh, to handle curves in the 3D geometric space.\n * A Curve3 is designed from a series of successive Vector3.\n * Tuto : https://doc.babylonjs.com/features/featuresDeepDive/mesh/drawCurves#curve3-object\n * @param points points which make up the curve\n */\n constructor(points) {\n this._length = 0.0;\n this._points = points;\n this._length = this._computeLength(points);\n }\n /**\n * @returns the Curve3 stored array of successive Vector3\n */\n getPoints() {\n return this._points;\n }\n /**\n * @returns the computed length (float) of the curve.\n */\n length() {\n return this._length;\n }\n /**\n * Returns a new instance of Curve3 object : var curve = curveA.continue(curveB);\n * This new Curve3 is built by translating and sticking the curveB at the end of the curveA.\n * curveA and curveB keep unchanged.\n * @param curve the curve to continue from this curve\n * @returns the newly constructed curve\n */\n continue(curve) {\n const lastPoint = this._points[this._points.length - 1];\n const continuedPoints = this._points.slice();\n const curvePoints = curve.getPoints();\n for (let i = 1; i < curvePoints.length; i++) {\n continuedPoints.push(curvePoints[i].subtract(curvePoints[0]).add(lastPoint));\n }\n const continuedCurve = new Curve3(continuedPoints);\n return continuedCurve;\n }\n _computeLength(path) {\n let l = 0;\n for (let i = 1; i < path.length; i++) {\n l += path[i].subtract(path[i - 1]).length();\n }\n return l;\n }\n }\n\n /**\n * Base class used for every default easing function.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#easing-functions\n */\n class EasingFunction {\n constructor() {\n this._easingMode = EasingFunction.EASINGMODE_EASEIN;\n }\n /**\n * Sets the easing mode of the current function.\n * @param easingMode Defines the willing mode (EASINGMODE_EASEIN, EASINGMODE_EASEOUT or EASINGMODE_EASEINOUT)\n */\n setEasingMode(easingMode) {\n const n = Math.min(Math.max(easingMode, 0), 2);\n this._easingMode = n;\n }\n /**\n * Gets the current easing mode.\n * @returns the easing mode\n */\n getEasingMode() {\n return this._easingMode;\n }\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n easeInCore(gradient) {\n throw new Error(\"You must implement this method\");\n }\n /**\n * Given an input gradient between 0 and 1, this returns the corresponding value\n * of the easing function.\n * @param gradient Defines the value between 0 and 1 we want the easing value for\n * @returns the corresponding value on the curve defined by the easing function\n */\n ease(gradient) {\n switch (this._easingMode) {\n case EasingFunction.EASINGMODE_EASEIN:\n return this.easeInCore(gradient);\n case EasingFunction.EASINGMODE_EASEOUT:\n return 1 - this.easeInCore(1 - gradient);\n }\n if (gradient >= 0.5) {\n return (1 - this.easeInCore((1 - gradient) * 2)) * 0.5 + 0.5;\n }\n return this.easeInCore(gradient * 2) * 0.5;\n }\n }\n /**\n * Interpolation follows the mathematical formula associated with the easing function.\n */\n EasingFunction.EASINGMODE_EASEIN = 0;\n /**\n * Interpolation follows 100% interpolation minus the output of the formula associated with the easing function.\n */\n EasingFunction.EASINGMODE_EASEOUT = 1;\n /**\n * Interpolation uses EaseIn for the first half of the animation and EaseOut for the second half.\n */\n EasingFunction.EASINGMODE_EASEINOUT = 2;\n /**\n * Easing function with a circle shape (see link below).\n * @see https://easings.net/#easeInCirc\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#easing-functions\n */\n class CircleEase extends EasingFunction {\n /**\n * @internal\n */\n easeInCore(gradient) {\n gradient = Math.max(0, Math.min(1, gradient));\n return 1.0 - Math.sqrt(1.0 - gradient * gradient);\n }\n }\n /**\n * Easing function with a ease back shape (see link below).\n * @see https://easings.net/#easeInBack\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#easing-functions\n */\n class BackEase extends EasingFunction {\n /**\n * Instantiates a back ease easing\n * @see https://easings.net/#easeInBack\n * @param amplitude Defines the amplitude of the function\n */\n constructor(\n /** Defines the amplitude of the function */\n amplitude = 1) {\n super();\n this.amplitude = amplitude;\n }\n /**\n * @internal\n */\n easeInCore(gradient) {\n const num = Math.max(0, this.amplitude);\n return Math.pow(gradient, 3.0) - gradient * num * Math.sin(3.1415926535897931 * gradient);\n }\n }\n /**\n * Easing function with an exponential shape (see link below).\n * @see https://easings.net/#easeInExpo\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#easing-functions\n */\n class ExponentialEase extends EasingFunction {\n /**\n * Instantiates an exponential easing function\n * @see https://easings.net/#easeInExpo\n * @param exponent Defines the exponent of the function\n */\n constructor(\n /** Defines the exponent of the function */\n exponent = 2) {\n super();\n this.exponent = exponent;\n }\n /**\n * @internal\n */\n easeInCore(gradient) {\n if (this.exponent <= 0) {\n return gradient;\n }\n return (Math.exp(this.exponent * gradient) - 1.0) / (Math.exp(this.exponent) - 1.0);\n }\n }\n /**\n * Easing function with a power of 2 shape (see link below).\n * @see https://easings.net/#easeInQuad\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#easing-functions\n */\n class QuadraticEase extends EasingFunction {\n /**\n * @internal\n */\n easeInCore(gradient) {\n return gradient * gradient;\n }\n }\n /**\n * Easing function with a sin shape (see link below).\n * @see https://easings.net/#easeInSine\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#easing-functions\n */\n class SineEase extends EasingFunction {\n /**\n * @internal\n */\n easeInCore(gradient) {\n return 1.0 - Math.sin(1.5707963267948966 * (1.0 - gradient));\n }\n }\n\n /**\n * Enum for the animation key frame interpolation type\n */\n var AnimationKeyInterpolation;\n (function (AnimationKeyInterpolation) {\n /**\n * Use tangents to interpolate between start and end values.\n */\n AnimationKeyInterpolation[AnimationKeyInterpolation[\"NONE\"] = 0] = \"NONE\";\n /**\n * Do not interpolate between keys and use the start key value only. Tangents are ignored\n */\n AnimationKeyInterpolation[AnimationKeyInterpolation[\"STEP\"] = 1] = \"STEP\";\n })(AnimationKeyInterpolation || (AnimationKeyInterpolation = {}));\n\n /**\n * Represents the range of an animation\n */\n class AnimationRange {\n /**\n * Initializes the range of an animation\n * @param name The name of the animation range\n * @param from The starting frame of the animation\n * @param to The ending frame of the animation\n */\n constructor(\n /**The name of the animation range**/\n name, \n /**The starting frame of the animation */\n from, \n /**The ending frame of the animation*/\n to) {\n this.name = name;\n this.from = from;\n this.to = to;\n }\n /**\n * Makes a copy of the animation range\n * @returns A copy of the animation range\n */\n clone() {\n return new AnimationRange(this.name, this.from, this.to);\n }\n }\n\n /**\n * Size containing width and height\n */\n class Size {\n /**\n * Creates a Size object from the given width and height (floats).\n * @param width width of the new size\n * @param height height of the new size\n */\n constructor(width, height) {\n this.width = width;\n this.height = height;\n }\n /**\n * Returns a string with the Size width and height\n * @returns a string with the Size width and height\n */\n toString() {\n return `{W: ${this.width}, H: ${this.height}}`;\n }\n /**\n * \"Size\"\n * @returns the string \"Size\"\n */\n getClassName() {\n return \"Size\";\n }\n /**\n * Returns the Size hash code.\n * @returns a hash code for a unique width and height\n */\n getHashCode() {\n let hash = this.width | 0;\n hash = (hash * 397) ^ (this.height | 0);\n return hash;\n }\n /**\n * Updates the current size from the given one.\n * @param src the given size\n */\n copyFrom(src) {\n this.width = src.width;\n this.height = src.height;\n }\n /**\n * Updates in place the current Size from the given floats.\n * @param width width of the new size\n * @param height height of the new size\n * @returns the updated Size.\n */\n copyFromFloats(width, height) {\n this.width = width;\n this.height = height;\n return this;\n }\n /**\n * Updates in place the current Size from the given floats.\n * @param width width to set\n * @param height height to set\n * @returns the updated Size.\n */\n set(width, height) {\n return this.copyFromFloats(width, height);\n }\n /**\n * Multiplies the width and height by numbers\n * @param w factor to multiple the width by\n * @param h factor to multiple the height by\n * @returns a new Size set with the multiplication result of the current Size and the given floats.\n */\n multiplyByFloats(w, h) {\n return new Size(this.width * w, this.height * h);\n }\n /**\n * Clones the size\n * @returns a new Size copied from the given one.\n */\n clone() {\n return new Size(this.width, this.height);\n }\n /**\n * True if the current Size and the given one width and height are strictly equal.\n * @param other the other size to compare against\n * @returns True if the current Size and the given one width and height are strictly equal.\n */\n equals(other) {\n if (!other) {\n return false;\n }\n return this.width === other.width && this.height === other.height;\n }\n /**\n * The surface of the Size : width * height (float).\n */\n get surface() {\n return this.width * this.height;\n }\n /**\n * Create a new size of zero\n * @returns a new Size set to (0.0, 0.0)\n */\n static Zero() {\n return new Size(0.0, 0.0);\n }\n /**\n * Sums the width and height of two sizes\n * @param otherSize size to add to this size\n * @returns a new Size set as the addition result of the current Size and the given one.\n */\n add(otherSize) {\n const r = new Size(this.width + otherSize.width, this.height + otherSize.height);\n return r;\n }\n /**\n * Subtracts the width and height of two\n * @param otherSize size to subtract to this size\n * @returns a new Size set as the subtraction result of the given one from the current Size.\n */\n subtract(otherSize) {\n const r = new Size(this.width - otherSize.width, this.height - otherSize.height);\n return r;\n }\n /**\n * Creates a new Size set at the linear interpolation \"amount\" between \"start\" and \"end\"\n * @param start starting size to lerp between\n * @param end end size to lerp between\n * @param amount amount to lerp between the start and end values\n * @returns a new Size set at the linear interpolation \"amount\" between \"start\" and \"end\"\n */\n static Lerp(start, end, amount) {\n const w = start.width + (end.width - start.width) * amount;\n const h = start.height + (end.height - start.height) * amount;\n return new Size(w, h);\n }\n }\n\n /**\n * Class used to store any kind of animation\n */\n class Animation {\n /**\n * @internal Internal use\n */\n static _PrepareAnimation(name, targetProperty, framePerSecond, totalFrame, from, to, loopMode, easingFunction) {\n let dataType = undefined;\n if (!isNaN(parseFloat(from)) && isFinite(from)) {\n dataType = Animation.ANIMATIONTYPE_FLOAT;\n }\n else if (from instanceof Quaternion) {\n dataType = Animation.ANIMATIONTYPE_QUATERNION;\n }\n else if (from instanceof Vector3) {\n dataType = Animation.ANIMATIONTYPE_VECTOR3;\n }\n else if (from instanceof Vector2) {\n dataType = Animation.ANIMATIONTYPE_VECTOR2;\n }\n else if (from instanceof Color3) {\n dataType = Animation.ANIMATIONTYPE_COLOR3;\n }\n else if (from instanceof Color4) {\n dataType = Animation.ANIMATIONTYPE_COLOR4;\n }\n else if (from instanceof Size) {\n dataType = Animation.ANIMATIONTYPE_SIZE;\n }\n if (dataType == undefined) {\n return null;\n }\n const animation = new Animation(name, targetProperty, framePerSecond, dataType, loopMode);\n const keys = [\n { frame: 0, value: from },\n { frame: totalFrame, value: to },\n ];\n animation.setKeys(keys);\n if (easingFunction !== undefined) {\n animation.setEasingFunction(easingFunction);\n }\n return animation;\n }\n /**\n * Sets up an animation\n * @param property The property to animate\n * @param animationType The animation type to apply\n * @param framePerSecond The frames per second of the animation\n * @param easingFunction The easing function used in the animation\n * @returns The created animation\n */\n static CreateAnimation(property, animationType, framePerSecond, easingFunction) {\n const animation = new Animation(property + \"Animation\", property, framePerSecond, animationType, Animation.ANIMATIONLOOPMODE_CONSTANT);\n animation.setEasingFunction(easingFunction);\n return animation;\n }\n /**\n * Create and start an animation on a node\n * @param name defines the name of the global animation that will be run on all nodes\n * @param target defines the target where the animation will take place\n * @param targetProperty defines property to animate\n * @param framePerSecond defines the number of frame per second yo use\n * @param totalFrame defines the number of frames in total\n * @param from defines the initial value\n * @param to defines the final value\n * @param loopMode defines which loop mode you want to use (off by default)\n * @param easingFunction defines the easing function to use (linear by default)\n * @param onAnimationEnd defines the callback to call when animation end\n * @param scene defines the hosting scene\n * @returns the animatable created for this animation\n */\n static CreateAndStartAnimation(name, target, targetProperty, framePerSecond, totalFrame, from, to, loopMode, easingFunction, onAnimationEnd, scene) {\n const animation = Animation._PrepareAnimation(name, targetProperty, framePerSecond, totalFrame, from, to, loopMode, easingFunction);\n if (!animation) {\n return null;\n }\n if (target.getScene) {\n scene = target.getScene();\n }\n if (!scene) {\n return null;\n }\n return scene.beginDirectAnimation(target, [animation], 0, totalFrame, animation.loopMode === 1, 1.0, onAnimationEnd);\n }\n /**\n * Create and start an animation on a node and its descendants\n * @param name defines the name of the global animation that will be run on all nodes\n * @param node defines the root node where the animation will take place\n * @param directDescendantsOnly if true only direct descendants will be used, if false direct and also indirect (children of children, an so on in a recursive manner) descendants will be used\n * @param targetProperty defines property to animate\n * @param framePerSecond defines the number of frame per second to use\n * @param totalFrame defines the number of frames in total\n * @param from defines the initial value\n * @param to defines the final value\n * @param loopMode defines which loop mode you want to use (off by default)\n * @param easingFunction defines the easing function to use (linear by default)\n * @param onAnimationEnd defines the callback to call when an animation ends (will be called once per node)\n * @returns the list of animatables created for all nodes\n * @example https://www.babylonjs-playground.com/#MH0VLI\n */\n static CreateAndStartHierarchyAnimation(name, node, directDescendantsOnly, targetProperty, framePerSecond, totalFrame, from, to, loopMode, easingFunction, onAnimationEnd) {\n const animation = Animation._PrepareAnimation(name, targetProperty, framePerSecond, totalFrame, from, to, loopMode, easingFunction);\n if (!animation) {\n return null;\n }\n const scene = node.getScene();\n return scene.beginDirectHierarchyAnimation(node, directDescendantsOnly, [animation], 0, totalFrame, animation.loopMode === 1, 1.0, onAnimationEnd);\n }\n /**\n * Creates a new animation, merges it with the existing animations and starts it\n * @param name Name of the animation\n * @param node Node which contains the scene that begins the animations\n * @param targetProperty Specifies which property to animate\n * @param framePerSecond The frames per second of the animation\n * @param totalFrame The total number of frames\n * @param from The frame at the beginning of the animation\n * @param to The frame at the end of the animation\n * @param loopMode Specifies the loop mode of the animation\n * @param easingFunction (Optional) The easing function of the animation, which allow custom mathematical formulas for animations\n * @param onAnimationEnd Callback to run once the animation is complete\n * @returns Nullable animation\n */\n static CreateMergeAndStartAnimation(name, node, targetProperty, framePerSecond, totalFrame, from, to, loopMode, easingFunction, onAnimationEnd) {\n const animation = Animation._PrepareAnimation(name, targetProperty, framePerSecond, totalFrame, from, to, loopMode, easingFunction);\n if (!animation) {\n return null;\n }\n node.animations.push(animation);\n return node.getScene().beginAnimation(node, 0, totalFrame, animation.loopMode === 1, 1.0, onAnimationEnd);\n }\n /**\n * Convert the keyframes for all animations belonging to the group to be relative to a given reference frame.\n * @param sourceAnimation defines the Animation containing keyframes to convert\n * @param referenceFrame defines the frame that keyframes in the range will be relative to\n * @param range defines the name of the AnimationRange belonging to the Animation to convert\n * @param cloneOriginal defines whether or not to clone the animation and convert the clone or convert the original animation (default is false)\n * @param clonedName defines the name of the resulting cloned Animation if cloneOriginal is true\n * @returns a new Animation if cloneOriginal is true or the original Animation if cloneOriginal is false\n */\n static MakeAnimationAdditive(sourceAnimation, referenceFrame = 0, range, cloneOriginal = false, clonedName) {\n let animation = sourceAnimation;\n if (cloneOriginal) {\n animation = sourceAnimation.clone();\n animation.name = clonedName || animation.name;\n }\n if (!animation._keys.length) {\n return animation;\n }\n referenceFrame = referenceFrame >= 0 ? referenceFrame : 0;\n let startIndex = 0;\n const firstKey = animation._keys[0];\n let endIndex = animation._keys.length - 1;\n const lastKey = animation._keys[endIndex];\n const valueStore = {\n referenceValue: firstKey.value,\n referencePosition: TmpVectors.Vector3[0],\n referenceQuaternion: TmpVectors.Quaternion[0],\n referenceScaling: TmpVectors.Vector3[1],\n keyPosition: TmpVectors.Vector3[2],\n keyQuaternion: TmpVectors.Quaternion[1],\n keyScaling: TmpVectors.Vector3[3],\n };\n let referenceFound = false;\n let from = firstKey.frame;\n let to = lastKey.frame;\n if (range) {\n const rangeValue = animation.getRange(range);\n if (rangeValue) {\n from = rangeValue.from;\n to = rangeValue.to;\n }\n }\n let fromKeyFound = firstKey.frame === from;\n let toKeyFound = lastKey.frame === to;\n // There's only one key, so use it\n if (animation._keys.length === 1) {\n const value = animation._getKeyValue(animation._keys[0]);\n valueStore.referenceValue = value.clone ? value.clone() : value;\n referenceFound = true;\n }\n // Reference frame is before the first frame, so just use the first frame\n else if (referenceFrame <= firstKey.frame) {\n const value = animation._getKeyValue(firstKey.value);\n valueStore.referenceValue = value.clone ? value.clone() : value;\n referenceFound = true;\n }\n // Reference frame is after the last frame, so just use the last frame\n else if (referenceFrame >= lastKey.frame) {\n const value = animation._getKeyValue(lastKey.value);\n valueStore.referenceValue = value.clone ? value.clone() : value;\n referenceFound = true;\n }\n // Find key bookends, create them if they don't exist\n let index = 0;\n while (!referenceFound || !fromKeyFound || (!toKeyFound && index < animation._keys.length - 1)) {\n const currentKey = animation._keys[index];\n const nextKey = animation._keys[index + 1];\n // If reference frame wasn't found yet, check if we can interpolate to it\n if (!referenceFound && referenceFrame >= currentKey.frame && referenceFrame <= nextKey.frame) {\n let value;\n if (referenceFrame === currentKey.frame) {\n value = animation._getKeyValue(currentKey.value);\n }\n else if (referenceFrame === nextKey.frame) {\n value = animation._getKeyValue(nextKey.value);\n }\n else {\n const animationState = {\n key: index,\n repeatCount: 0,\n loopMode: this.ANIMATIONLOOPMODE_CONSTANT,\n };\n value = animation._interpolate(referenceFrame, animationState);\n }\n valueStore.referenceValue = value.clone ? value.clone() : value;\n referenceFound = true;\n }\n // If from key wasn't found yet, check if we can interpolate to it\n if (!fromKeyFound && from >= currentKey.frame && from <= nextKey.frame) {\n if (from === currentKey.frame) {\n startIndex = index;\n }\n else if (from === nextKey.frame) {\n startIndex = index + 1;\n }\n else {\n const animationState = {\n key: index,\n repeatCount: 0,\n loopMode: this.ANIMATIONLOOPMODE_CONSTANT,\n };\n const value = animation._interpolate(from, animationState);\n const key = {\n frame: from,\n value: value.clone ? value.clone() : value,\n };\n animation._keys.splice(index + 1, 0, key);\n startIndex = index + 1;\n }\n fromKeyFound = true;\n }\n // If to key wasn't found yet, check if we can interpolate to it\n if (!toKeyFound && to >= currentKey.frame && to <= nextKey.frame) {\n if (to === currentKey.frame) {\n endIndex = index;\n }\n else if (to === nextKey.frame) {\n endIndex = index + 1;\n }\n else {\n const animationState = {\n key: index,\n repeatCount: 0,\n loopMode: this.ANIMATIONLOOPMODE_CONSTANT,\n };\n const value = animation._interpolate(to, animationState);\n const key = {\n frame: to,\n value: value.clone ? value.clone() : value,\n };\n animation._keys.splice(index + 1, 0, key);\n endIndex = index + 1;\n }\n toKeyFound = true;\n }\n index++;\n }\n // Conjugate the quaternion\n if (animation.dataType === Animation.ANIMATIONTYPE_QUATERNION) {\n valueStore.referenceValue.normalize().conjugateInPlace();\n }\n // Decompose matrix and conjugate the quaternion\n else if (animation.dataType === Animation.ANIMATIONTYPE_MATRIX) {\n valueStore.referenceValue.decompose(valueStore.referenceScaling, valueStore.referenceQuaternion, valueStore.referencePosition);\n valueStore.referenceQuaternion.normalize().conjugateInPlace();\n }\n // Subtract the reference value from all of the key values\n for (index = startIndex; index <= endIndex; index++) {\n const key = animation._keys[index];\n // If this key was duplicated to create a frame 0 key, skip it because its value has already been updated\n if (index && animation.dataType !== Animation.ANIMATIONTYPE_FLOAT && key.value === firstKey.value) {\n continue;\n }\n switch (animation.dataType) {\n case Animation.ANIMATIONTYPE_MATRIX:\n key.value.decompose(valueStore.keyScaling, valueStore.keyQuaternion, valueStore.keyPosition);\n valueStore.keyPosition.subtractInPlace(valueStore.referencePosition);\n valueStore.keyScaling.divideInPlace(valueStore.referenceScaling);\n valueStore.referenceQuaternion.multiplyToRef(valueStore.keyQuaternion, valueStore.keyQuaternion);\n Matrix.ComposeToRef(valueStore.keyScaling, valueStore.keyQuaternion, valueStore.keyPosition, key.value);\n break;\n case Animation.ANIMATIONTYPE_QUATERNION:\n valueStore.referenceValue.multiplyToRef(key.value, key.value);\n break;\n case Animation.ANIMATIONTYPE_VECTOR2:\n case Animation.ANIMATIONTYPE_VECTOR3:\n case Animation.ANIMATIONTYPE_COLOR3:\n case Animation.ANIMATIONTYPE_COLOR4:\n key.value.subtractToRef(valueStore.referenceValue, key.value);\n break;\n case Animation.ANIMATIONTYPE_SIZE:\n key.value.width -= valueStore.referenceValue.width;\n key.value.height -= valueStore.referenceValue.height;\n break;\n default:\n key.value -= valueStore.referenceValue;\n }\n }\n return animation;\n }\n /**\n * Transition property of an host to the target Value\n * @param property The property to transition\n * @param targetValue The target Value of the property\n * @param host The object where the property to animate belongs\n * @param scene Scene used to run the animation\n * @param frameRate Framerate (in frame/s) to use\n * @param transition The transition type we want to use\n * @param duration The duration of the animation, in milliseconds\n * @param onAnimationEnd Callback trigger at the end of the animation\n * @returns Nullable animation\n */\n static TransitionTo(property, targetValue, host, scene, frameRate, transition, duration, onAnimationEnd = null) {\n if (duration <= 0) {\n host[property] = targetValue;\n if (onAnimationEnd) {\n onAnimationEnd();\n }\n return null;\n }\n const endFrame = frameRate * (duration / 1000);\n transition.setKeys([\n {\n frame: 0,\n value: host[property].clone ? host[property].clone() : host[property],\n },\n {\n frame: endFrame,\n value: targetValue,\n },\n ]);\n if (!host.animations) {\n host.animations = [];\n }\n host.animations.push(transition);\n const animation = scene.beginAnimation(host, 0, endFrame, false);\n animation.onAnimationEnd = onAnimationEnd;\n return animation;\n }\n /**\n * Return the array of runtime animations currently using this animation\n */\n get runtimeAnimations() {\n return this._runtimeAnimations;\n }\n /**\n * Specifies if any of the runtime animations are currently running\n */\n get hasRunningRuntimeAnimations() {\n for (const runtimeAnimation of this._runtimeAnimations) {\n if (!runtimeAnimation.isStopped()) {\n return true;\n }\n }\n return false;\n }\n /**\n * Initializes the animation\n * @param name Name of the animation\n * @param targetProperty Property to animate\n * @param framePerSecond The frames per second of the animation\n * @param dataType The data type of the animation\n * @param loopMode The loop mode of the animation\n * @param enableBlending Specifies if blending should be enabled\n */\n constructor(\n /**Name of the animation */\n name, \n /**Property to animate */\n targetProperty, \n /**The frames per second of the animation */\n framePerSecond, \n /**The data type of the animation */\n dataType, \n /**The loop mode of the animation */\n loopMode, \n /**Specifies if blending should be enabled */\n enableBlending) {\n this.name = name;\n this.targetProperty = targetProperty;\n this.framePerSecond = framePerSecond;\n this.dataType = dataType;\n this.loopMode = loopMode;\n this.enableBlending = enableBlending;\n /**\n * Stores the easing function of the animation\n */\n this._easingFunction = null;\n /**\n * @internal Internal use only\n */\n this._runtimeAnimations = new Array();\n /**\n * The set of event that will be linked to this animation\n */\n this._events = new Array();\n /**\n * Stores the blending speed of the animation\n */\n this.blendingSpeed = 0.01;\n /**\n * Stores the animation ranges for the animation\n */\n this._ranges = {};\n this.targetPropertyPath = targetProperty.split(\".\");\n this.dataType = dataType;\n this.loopMode = loopMode === undefined ? Animation.ANIMATIONLOOPMODE_CYCLE : loopMode;\n this.uniqueId = Animation._UniqueIdGenerator++;\n }\n // Methods\n /**\n * Converts the animation to a string\n * @param fullDetails support for multiple levels of logging within scene loading\n * @returns String form of the animation\n */\n toString(fullDetails) {\n let ret = \"Name: \" + this.name + \", property: \" + this.targetProperty;\n ret += \", datatype: \" + [\"Float\", \"Vector3\", \"Quaternion\", \"Matrix\", \"Color3\", \"Vector2\"][this.dataType];\n ret += \", nKeys: \" + (this._keys ? this._keys.length : \"none\");\n ret += \", nRanges: \" + (this._ranges ? Object.keys(this._ranges).length : \"none\");\n if (fullDetails) {\n ret += \", Ranges: {\";\n let first = true;\n for (const name in this._ranges) {\n if (first) {\n ret += \", \";\n first = false;\n }\n ret += name;\n }\n ret += \"}\";\n }\n return ret;\n }\n /**\n * Add an event to this animation\n * @param event Event to add\n */\n addEvent(event) {\n this._events.push(event);\n this._events.sort((a, b) => a.frame - b.frame);\n }\n /**\n * Remove all events found at the given frame\n * @param frame The frame to remove events from\n */\n removeEvents(frame) {\n for (let index = 0; index < this._events.length; index++) {\n if (this._events[index].frame === frame) {\n this._events.splice(index, 1);\n index--;\n }\n }\n }\n /**\n * Retrieves all the events from the animation\n * @returns Events from the animation\n */\n getEvents() {\n return this._events;\n }\n /**\n * Creates an animation range\n * @param name Name of the animation range\n * @param from Starting frame of the animation range\n * @param to Ending frame of the animation\n */\n createRange(name, from, to) {\n // check name not already in use; could happen for bones after serialized\n if (!this._ranges[name]) {\n this._ranges[name] = new AnimationRange(name, from, to);\n }\n }\n /**\n * Deletes an animation range by name\n * @param name Name of the animation range to delete\n * @param deleteFrames Specifies if the key frames for the range should also be deleted (true) or not (false)\n */\n deleteRange(name, deleteFrames = true) {\n const range = this._ranges[name];\n if (!range) {\n return;\n }\n if (deleteFrames) {\n const from = range.from;\n const to = range.to;\n // this loop MUST go high to low for multiple splices to work\n for (let key = this._keys.length - 1; key >= 0; key--) {\n if (this._keys[key].frame >= from && this._keys[key].frame <= to) {\n this._keys.splice(key, 1);\n }\n }\n }\n this._ranges[name] = null; // said much faster than 'delete this._range[name]'\n }\n /**\n * Gets the animation range by name, or null if not defined\n * @param name Name of the animation range\n * @returns Nullable animation range\n */\n getRange(name) {\n return this._ranges[name];\n }\n /**\n * Gets the key frames from the animation\n * @returns The key frames of the animation\n */\n getKeys() {\n return this._keys;\n }\n /**\n * Gets the highest frame rate of the animation\n * @returns Highest frame rate of the animation\n */\n getHighestFrame() {\n let ret = 0;\n for (let key = 0, nKeys = this._keys.length; key < nKeys; key++) {\n if (ret < this._keys[key].frame) {\n ret = this._keys[key].frame;\n }\n }\n return ret;\n }\n /**\n * Gets the easing function of the animation\n * @returns Easing function of the animation\n */\n getEasingFunction() {\n return this._easingFunction;\n }\n /**\n * Sets the easing function of the animation\n * @param easingFunction A custom mathematical formula for animation\n */\n setEasingFunction(easingFunction) {\n this._easingFunction = easingFunction;\n }\n /**\n * Interpolates a scalar linearly\n * @param startValue Start value of the animation curve\n * @param endValue End value of the animation curve\n * @param gradient Scalar amount to interpolate\n * @returns Interpolated scalar value\n */\n floatInterpolateFunction(startValue, endValue, gradient) {\n return Scalar.Lerp(startValue, endValue, gradient);\n }\n /**\n * Interpolates a scalar cubically\n * @param startValue Start value of the animation curve\n * @param outTangent End tangent of the animation\n * @param endValue End value of the animation curve\n * @param inTangent Start tangent of the animation curve\n * @param gradient Scalar amount to interpolate\n * @returns Interpolated scalar value\n */\n floatInterpolateFunctionWithTangents(startValue, outTangent, endValue, inTangent, gradient) {\n return Scalar.Hermite(startValue, outTangent, endValue, inTangent, gradient);\n }\n /**\n * Interpolates a quaternion using a spherical linear interpolation\n * @param startValue Start value of the animation curve\n * @param endValue End value of the animation curve\n * @param gradient Scalar amount to interpolate\n * @returns Interpolated quaternion value\n */\n quaternionInterpolateFunction(startValue, endValue, gradient) {\n return Quaternion.Slerp(startValue, endValue, gradient);\n }\n /**\n * Interpolates a quaternion cubically\n * @param startValue Start value of the animation curve\n * @param outTangent End tangent of the animation curve\n * @param endValue End value of the animation curve\n * @param inTangent Start tangent of the animation curve\n * @param gradient Scalar amount to interpolate\n * @returns Interpolated quaternion value\n */\n quaternionInterpolateFunctionWithTangents(startValue, outTangent, endValue, inTangent, gradient) {\n return Quaternion.Hermite(startValue, outTangent, endValue, inTangent, gradient).normalize();\n }\n /**\n * Interpolates a Vector3 linearly\n * @param startValue Start value of the animation curve\n * @param endValue End value of the animation curve\n * @param gradient Scalar amount to interpolate (value between 0 and 1)\n * @returns Interpolated scalar value\n */\n vector3InterpolateFunction(startValue, endValue, gradient) {\n return Vector3.Lerp(startValue, endValue, gradient);\n }\n /**\n * Interpolates a Vector3 cubically\n * @param startValue Start value of the animation curve\n * @param outTangent End tangent of the animation\n * @param endValue End value of the animation curve\n * @param inTangent Start tangent of the animation curve\n * @param gradient Scalar amount to interpolate (value between 0 and 1)\n * @returns InterpolatedVector3 value\n */\n vector3InterpolateFunctionWithTangents(startValue, outTangent, endValue, inTangent, gradient) {\n return Vector3.Hermite(startValue, outTangent, endValue, inTangent, gradient);\n }\n /**\n * Interpolates a Vector2 linearly\n * @param startValue Start value of the animation curve\n * @param endValue End value of the animation curve\n * @param gradient Scalar amount to interpolate (value between 0 and 1)\n * @returns Interpolated Vector2 value\n */\n vector2InterpolateFunction(startValue, endValue, gradient) {\n return Vector2.Lerp(startValue, endValue, gradient);\n }\n /**\n * Interpolates a Vector2 cubically\n * @param startValue Start value of the animation curve\n * @param outTangent End tangent of the animation\n * @param endValue End value of the animation curve\n * @param inTangent Start tangent of the animation curve\n * @param gradient Scalar amount to interpolate (value between 0 and 1)\n * @returns Interpolated Vector2 value\n */\n vector2InterpolateFunctionWithTangents(startValue, outTangent, endValue, inTangent, gradient) {\n return Vector2.Hermite(startValue, outTangent, endValue, inTangent, gradient);\n }\n /**\n * Interpolates a size linearly\n * @param startValue Start value of the animation curve\n * @param endValue End value of the animation curve\n * @param gradient Scalar amount to interpolate\n * @returns Interpolated Size value\n */\n sizeInterpolateFunction(startValue, endValue, gradient) {\n return Size.Lerp(startValue, endValue, gradient);\n }\n /**\n * Interpolates a Color3 linearly\n * @param startValue Start value of the animation curve\n * @param endValue End value of the animation curve\n * @param gradient Scalar amount to interpolate\n * @returns Interpolated Color3 value\n */\n color3InterpolateFunction(startValue, endValue, gradient) {\n return Color3.Lerp(startValue, endValue, gradient);\n }\n /**\n * Interpolates a Color3 cubically\n * @param startValue Start value of the animation curve\n * @param outTangent End tangent of the animation\n * @param endValue End value of the animation curve\n * @param inTangent Start tangent of the animation curve\n * @param gradient Scalar amount to interpolate\n * @returns interpolated value\n */\n color3InterpolateFunctionWithTangents(startValue, outTangent, endValue, inTangent, gradient) {\n return Color3.Hermite(startValue, outTangent, endValue, inTangent, gradient);\n }\n /**\n * Interpolates a Color4 linearly\n * @param startValue Start value of the animation curve\n * @param endValue End value of the animation curve\n * @param gradient Scalar amount to interpolate\n * @returns Interpolated Color3 value\n */\n color4InterpolateFunction(startValue, endValue, gradient) {\n return Color4.Lerp(startValue, endValue, gradient);\n }\n /**\n * Interpolates a Color4 cubically\n * @param startValue Start value of the animation curve\n * @param outTangent End tangent of the animation\n * @param endValue End value of the animation curve\n * @param inTangent Start tangent of the animation curve\n * @param gradient Scalar amount to interpolate\n * @returns interpolated value\n */\n color4InterpolateFunctionWithTangents(startValue, outTangent, endValue, inTangent, gradient) {\n return Color4.Hermite(startValue, outTangent, endValue, inTangent, gradient);\n }\n /**\n * @internal Internal use only\n */\n _getKeyValue(value) {\n if (typeof value === \"function\") {\n return value();\n }\n return value;\n }\n /**\n * Evaluate the animation value at a given frame\n * @param currentFrame defines the frame where we want to evaluate the animation\n * @returns the animation value\n */\n evaluate(currentFrame) {\n return this._interpolate(currentFrame, {\n key: 0,\n repeatCount: 0,\n loopMode: Animation.ANIMATIONLOOPMODE_CONSTANT,\n });\n }\n /**\n * @internal Internal use only\n */\n _interpolate(currentFrame, state) {\n if (state.loopMode === Animation.ANIMATIONLOOPMODE_CONSTANT && state.repeatCount > 0) {\n return state.highLimitValue.clone ? state.highLimitValue.clone() : state.highLimitValue;\n }\n const keys = this._keys;\n const keysLength = keys.length;\n let key = state.key;\n while (key >= 0 && currentFrame < keys[key].frame) {\n --key;\n }\n while (key + 1 <= keysLength - 1 && currentFrame >= keys[key + 1].frame) {\n ++key;\n }\n state.key = key;\n if (key < 0) {\n return this._getKeyValue(keys[0].value);\n }\n else if (key + 1 > keysLength - 1) {\n return this._getKeyValue(keys[keysLength - 1].value);\n }\n const startKey = keys[key];\n const endKey = keys[key + 1];\n const startValue = this._getKeyValue(startKey.value);\n const endValue = this._getKeyValue(endKey.value);\n if (startKey.interpolation === AnimationKeyInterpolation.STEP) {\n if (endKey.frame > currentFrame) {\n return startValue;\n }\n else {\n return endValue;\n }\n }\n const useTangent = startKey.outTangent !== undefined && endKey.inTangent !== undefined;\n const frameDelta = endKey.frame - startKey.frame;\n // gradient : percent of currentFrame between the frame inf and the frame sup\n let gradient = (currentFrame - startKey.frame) / frameDelta;\n // check for easingFunction and correction of gradient\n const easingFunction = this.getEasingFunction();\n if (easingFunction !== null) {\n gradient = easingFunction.ease(gradient);\n }\n switch (this.dataType) {\n // Float\n case Animation.ANIMATIONTYPE_FLOAT: {\n const floatValue = useTangent\n ? this.floatInterpolateFunctionWithTangents(startValue, startKey.outTangent * frameDelta, endValue, endKey.inTangent * frameDelta, gradient)\n : this.floatInterpolateFunction(startValue, endValue, gradient);\n switch (state.loopMode) {\n case Animation.ANIMATIONLOOPMODE_CYCLE:\n case Animation.ANIMATIONLOOPMODE_CONSTANT:\n case Animation.ANIMATIONLOOPMODE_YOYO:\n return floatValue;\n case Animation.ANIMATIONLOOPMODE_RELATIVE:\n return state.offsetValue * state.repeatCount + floatValue;\n }\n break;\n }\n // Quaternion\n case Animation.ANIMATIONTYPE_QUATERNION: {\n const quatValue = useTangent\n ? this.quaternionInterpolateFunctionWithTangents(startValue, startKey.outTangent.scale(frameDelta), endValue, endKey.inTangent.scale(frameDelta), gradient)\n : this.quaternionInterpolateFunction(startValue, endValue, gradient);\n switch (state.loopMode) {\n case Animation.ANIMATIONLOOPMODE_CYCLE:\n case Animation.ANIMATIONLOOPMODE_CONSTANT:\n case Animation.ANIMATIONLOOPMODE_YOYO:\n return quatValue;\n case Animation.ANIMATIONLOOPMODE_RELATIVE:\n return quatValue.addInPlace(state.offsetValue.scale(state.repeatCount));\n }\n return quatValue;\n }\n // Vector3\n case Animation.ANIMATIONTYPE_VECTOR3: {\n const vec3Value = useTangent\n ? this.vector3InterpolateFunctionWithTangents(startValue, startKey.outTangent.scale(frameDelta), endValue, endKey.inTangent.scale(frameDelta), gradient)\n : this.vector3InterpolateFunction(startValue, endValue, gradient);\n switch (state.loopMode) {\n case Animation.ANIMATIONLOOPMODE_CYCLE:\n case Animation.ANIMATIONLOOPMODE_CONSTANT:\n case Animation.ANIMATIONLOOPMODE_YOYO:\n return vec3Value;\n case Animation.ANIMATIONLOOPMODE_RELATIVE:\n return vec3Value.add(state.offsetValue.scale(state.repeatCount));\n }\n break;\n }\n // Vector2\n case Animation.ANIMATIONTYPE_VECTOR2: {\n const vec2Value = useTangent\n ? this.vector2InterpolateFunctionWithTangents(startValue, startKey.outTangent.scale(frameDelta), endValue, endKey.inTangent.scale(frameDelta), gradient)\n : this.vector2InterpolateFunction(startValue, endValue, gradient);\n switch (state.loopMode) {\n case Animation.ANIMATIONLOOPMODE_CYCLE:\n case Animation.ANIMATIONLOOPMODE_CONSTANT:\n case Animation.ANIMATIONLOOPMODE_YOYO:\n return vec2Value;\n case Animation.ANIMATIONLOOPMODE_RELATIVE:\n return vec2Value.add(state.offsetValue.scale(state.repeatCount));\n }\n break;\n }\n // Size\n case Animation.ANIMATIONTYPE_SIZE: {\n switch (state.loopMode) {\n case Animation.ANIMATIONLOOPMODE_CYCLE:\n case Animation.ANIMATIONLOOPMODE_CONSTANT:\n case Animation.ANIMATIONLOOPMODE_YOYO:\n return this.sizeInterpolateFunction(startValue, endValue, gradient);\n case Animation.ANIMATIONLOOPMODE_RELATIVE:\n return this.sizeInterpolateFunction(startValue, endValue, gradient).add(state.offsetValue.scale(state.repeatCount));\n }\n break;\n }\n // Color3\n case Animation.ANIMATIONTYPE_COLOR3: {\n const color3Value = useTangent\n ? this.color3InterpolateFunctionWithTangents(startValue, startKey.outTangent.scale(frameDelta), endValue, endKey.inTangent.scale(frameDelta), gradient)\n : this.color3InterpolateFunction(startValue, endValue, gradient);\n switch (state.loopMode) {\n case Animation.ANIMATIONLOOPMODE_CYCLE:\n case Animation.ANIMATIONLOOPMODE_CONSTANT:\n case Animation.ANIMATIONLOOPMODE_YOYO:\n return color3Value;\n case Animation.ANIMATIONLOOPMODE_RELATIVE:\n return color3Value.add(state.offsetValue.scale(state.repeatCount));\n }\n break;\n }\n // Color4\n case Animation.ANIMATIONTYPE_COLOR4: {\n const color4Value = useTangent\n ? this.color4InterpolateFunctionWithTangents(startValue, startKey.outTangent.scale(frameDelta), endValue, endKey.inTangent.scale(frameDelta), gradient)\n : this.color4InterpolateFunction(startValue, endValue, gradient);\n switch (state.loopMode) {\n case Animation.ANIMATIONLOOPMODE_CYCLE:\n case Animation.ANIMATIONLOOPMODE_CONSTANT:\n case Animation.ANIMATIONLOOPMODE_YOYO:\n return color4Value;\n case Animation.ANIMATIONLOOPMODE_RELATIVE:\n return color4Value.add(state.offsetValue.scale(state.repeatCount));\n }\n break;\n }\n // Matrix\n case Animation.ANIMATIONTYPE_MATRIX: {\n switch (state.loopMode) {\n case Animation.ANIMATIONLOOPMODE_CYCLE:\n case Animation.ANIMATIONLOOPMODE_CONSTANT:\n case Animation.ANIMATIONLOOPMODE_YOYO: {\n if (Animation.AllowMatricesInterpolation) {\n return this.matrixInterpolateFunction(startValue, endValue, gradient, state.workValue);\n }\n return startValue;\n }\n case Animation.ANIMATIONLOOPMODE_RELATIVE: {\n return startValue;\n }\n }\n break;\n }\n }\n return 0;\n }\n /**\n * Defines the function to use to interpolate matrices\n * @param startValue defines the start matrix\n * @param endValue defines the end matrix\n * @param gradient defines the gradient between both matrices\n * @param result defines an optional target matrix where to store the interpolation\n * @returns the interpolated matrix\n */\n matrixInterpolateFunction(startValue, endValue, gradient, result) {\n if (Animation.AllowMatrixDecomposeForInterpolation) {\n if (result) {\n Matrix.DecomposeLerpToRef(startValue, endValue, gradient, result);\n return result;\n }\n return Matrix.DecomposeLerp(startValue, endValue, gradient);\n }\n if (result) {\n Matrix.LerpToRef(startValue, endValue, gradient, result);\n return result;\n }\n return Matrix.Lerp(startValue, endValue, gradient);\n }\n /**\n * Makes a copy of the animation\n * @returns Cloned animation\n */\n clone() {\n const clone = new Animation(this.name, this.targetPropertyPath.join(\".\"), this.framePerSecond, this.dataType, this.loopMode);\n clone.enableBlending = this.enableBlending;\n clone.blendingSpeed = this.blendingSpeed;\n if (this._keys) {\n clone.setKeys(this._keys);\n }\n if (this._ranges) {\n clone._ranges = {};\n for (const name in this._ranges) {\n const range = this._ranges[name];\n if (!range) {\n continue;\n }\n clone._ranges[name] = range.clone();\n }\n }\n return clone;\n }\n /**\n * Sets the key frames of the animation\n * @param values The animation key frames to set\n */\n setKeys(values) {\n this._keys = values.slice(0);\n }\n /**\n * Serializes the animation to an object\n * @returns Serialized object\n */\n serialize() {\n const serializationObject = {};\n serializationObject.name = this.name;\n serializationObject.property = this.targetProperty;\n serializationObject.framePerSecond = this.framePerSecond;\n serializationObject.dataType = this.dataType;\n serializationObject.loopBehavior = this.loopMode;\n serializationObject.enableBlending = this.enableBlending;\n serializationObject.blendingSpeed = this.blendingSpeed;\n const dataType = this.dataType;\n serializationObject.keys = [];\n const keys = this.getKeys();\n for (let index = 0; index < keys.length; index++) {\n const animationKey = keys[index];\n const key = {};\n key.frame = animationKey.frame;\n switch (dataType) {\n case Animation.ANIMATIONTYPE_FLOAT:\n key.values = [animationKey.value];\n if (animationKey.inTangent !== undefined) {\n key.values.push(animationKey.inTangent);\n }\n if (animationKey.outTangent !== undefined) {\n if (animationKey.inTangent === undefined) {\n key.values.push(undefined);\n }\n key.values.push(animationKey.outTangent);\n }\n if (animationKey.interpolation !== undefined) {\n if (animationKey.inTangent === undefined) {\n key.values.push(undefined);\n }\n if (animationKey.outTangent === undefined) {\n key.values.push(undefined);\n }\n key.values.push(animationKey.interpolation);\n }\n break;\n case Animation.ANIMATIONTYPE_QUATERNION:\n case Animation.ANIMATIONTYPE_MATRIX:\n case Animation.ANIMATIONTYPE_VECTOR3:\n case Animation.ANIMATIONTYPE_COLOR3:\n case Animation.ANIMATIONTYPE_COLOR4:\n key.values = animationKey.value.asArray();\n if (animationKey.inTangent != undefined) {\n key.values.push(animationKey.inTangent.asArray());\n }\n if (animationKey.outTangent != undefined) {\n if (animationKey.inTangent === undefined) {\n key.values.push(undefined);\n }\n key.values.push(animationKey.outTangent.asArray());\n }\n if (animationKey.interpolation !== undefined) {\n if (animationKey.inTangent === undefined) {\n key.values.push(undefined);\n }\n if (animationKey.outTangent === undefined) {\n key.values.push(undefined);\n }\n key.values.push(animationKey.interpolation);\n }\n break;\n }\n serializationObject.keys.push(key);\n }\n serializationObject.ranges = [];\n for (const name in this._ranges) {\n const source = this._ranges[name];\n if (!source) {\n continue;\n }\n const range = {};\n range.name = name;\n range.from = source.from;\n range.to = source.to;\n serializationObject.ranges.push(range);\n }\n return serializationObject;\n }\n /**\n * @internal\n */\n static _UniversalLerp(left, right, amount) {\n const constructor = left.constructor;\n if (constructor.Lerp) {\n // Lerp supported\n return constructor.Lerp(left, right, amount);\n }\n else if (constructor.Slerp) {\n // Slerp supported\n return constructor.Slerp(left, right, amount);\n }\n else if (left.toFixed) {\n // Number\n return left * (1.0 - amount) + amount * right;\n }\n else {\n // Blending not supported\n return right;\n }\n }\n /**\n * Parses an animation object and creates an animation\n * @param parsedAnimation Parsed animation object\n * @returns Animation object\n */\n static Parse(parsedAnimation) {\n const animation = new Animation(parsedAnimation.name, parsedAnimation.property, parsedAnimation.framePerSecond, parsedAnimation.dataType, parsedAnimation.loopBehavior);\n const dataType = parsedAnimation.dataType;\n const keys = [];\n let data;\n let index;\n if (parsedAnimation.enableBlending) {\n animation.enableBlending = parsedAnimation.enableBlending;\n }\n if (parsedAnimation.blendingSpeed) {\n animation.blendingSpeed = parsedAnimation.blendingSpeed;\n }\n for (index = 0; index < parsedAnimation.keys.length; index++) {\n const key = parsedAnimation.keys[index];\n let inTangent = undefined;\n let outTangent = undefined;\n let interpolation = undefined;\n switch (dataType) {\n case Animation.ANIMATIONTYPE_FLOAT:\n data = key.values[0];\n if (key.values.length >= 2) {\n inTangent = key.values[1];\n }\n if (key.values.length >= 3) {\n outTangent = key.values[2];\n }\n if (key.values.length >= 4) {\n interpolation = key.values[3];\n }\n break;\n case Animation.ANIMATIONTYPE_QUATERNION:\n data = Quaternion.FromArray(key.values);\n if (key.values.length >= 8) {\n const _inTangent = Quaternion.FromArray(key.values.slice(4, 8));\n if (!_inTangent.equals(Quaternion.Zero())) {\n inTangent = _inTangent;\n }\n }\n if (key.values.length >= 12) {\n const _outTangent = Quaternion.FromArray(key.values.slice(8, 12));\n if (!_outTangent.equals(Quaternion.Zero())) {\n outTangent = _outTangent;\n }\n }\n if (key.values.length >= 13) {\n interpolation = key.values[12];\n }\n break;\n case Animation.ANIMATIONTYPE_MATRIX:\n data = Matrix.FromArray(key.values);\n if (key.values.length >= 17) {\n interpolation = key.values[16];\n }\n break;\n case Animation.ANIMATIONTYPE_COLOR3:\n data = Color3.FromArray(key.values);\n if (key.values[3]) {\n inTangent = Color3.FromArray(key.values[3]);\n }\n if (key.values[4]) {\n outTangent = Color3.FromArray(key.values[4]);\n }\n if (key.values[5]) {\n interpolation = key.values[5];\n }\n break;\n case Animation.ANIMATIONTYPE_COLOR4:\n data = Color4.FromArray(key.values);\n if (key.values[4]) {\n inTangent = Color4.FromArray(key.values[4]);\n }\n if (key.values[5]) {\n outTangent = Color4.FromArray(key.values[5]);\n }\n if (key.values[6]) {\n interpolation = Color4.FromArray(key.values[6]);\n }\n break;\n case Animation.ANIMATIONTYPE_VECTOR3:\n default:\n data = Vector3.FromArray(key.values);\n if (key.values[3]) {\n inTangent = Vector3.FromArray(key.values[3]);\n }\n if (key.values[4]) {\n outTangent = Vector3.FromArray(key.values[4]);\n }\n if (key.values[5]) {\n interpolation = key.values[5];\n }\n break;\n }\n const keyData = {};\n keyData.frame = key.frame;\n keyData.value = data;\n if (inTangent != undefined) {\n keyData.inTangent = inTangent;\n }\n if (outTangent != undefined) {\n keyData.outTangent = outTangent;\n }\n if (interpolation != undefined) {\n keyData.interpolation = interpolation;\n }\n keys.push(keyData);\n }\n animation.setKeys(keys);\n if (parsedAnimation.ranges) {\n for (index = 0; index < parsedAnimation.ranges.length; index++) {\n data = parsedAnimation.ranges[index];\n animation.createRange(data.name, data.from, data.to);\n }\n }\n return animation;\n }\n /**\n * Appends the serialized animations from the source animations\n * @param source Source containing the animations\n * @param destination Target to store the animations\n */\n static AppendSerializedAnimations(source, destination) {\n SerializationHelper.AppendSerializedAnimations(source, destination);\n }\n /**\n * Creates a new animation or an array of animations from a snippet saved in a remote file\n * @param name defines the name of the animation to create (can be null or empty to use the one from the json data)\n * @param url defines the url to load from\n * @returns a promise that will resolve to the new animation or an array of animations\n */\n static ParseFromFileAsync(name, url) {\n return new Promise((resolve, reject) => {\n const request = new WebRequest();\n request.addEventListener(\"readystatechange\", () => {\n if (request.readyState == 4) {\n if (request.status == 200) {\n let serializationObject = JSON.parse(request.responseText);\n if (serializationObject.animations) {\n serializationObject = serializationObject.animations;\n }\n if (serializationObject.length) {\n const output = new Array();\n for (const serializedAnimation of serializationObject) {\n output.push(this.Parse(serializedAnimation));\n }\n resolve(output);\n }\n else {\n const output = this.Parse(serializationObject);\n if (name) {\n output.name = name;\n }\n resolve(output);\n }\n }\n else {\n reject(\"Unable to load the animation\");\n }\n }\n });\n request.open(\"GET\", url);\n request.send();\n });\n }\n /**\n * Creates an animation or an array of animations from a snippet saved by the Inspector\n * @param snippetId defines the snippet to load\n * @returns a promise that will resolve to the new animation or a new array of animations\n */\n static ParseFromSnippetAsync(snippetId) {\n return new Promise((resolve, reject) => {\n const request = new WebRequest();\n request.addEventListener(\"readystatechange\", () => {\n if (request.readyState == 4) {\n if (request.status == 200) {\n const snippet = JSON.parse(JSON.parse(request.responseText).jsonPayload);\n if (snippet.animations) {\n const serializationObject = JSON.parse(snippet.animations);\n const outputs = new Array();\n for (const serializedAnimation of serializationObject.animations) {\n const output = this.Parse(serializedAnimation);\n output.snippetId = snippetId;\n outputs.push(output);\n }\n resolve(outputs);\n }\n else {\n const serializationObject = JSON.parse(snippet.animation);\n const output = this.Parse(serializationObject);\n output.snippetId = snippetId;\n resolve(output);\n }\n }\n else {\n reject(\"Unable to load the snippet \" + snippetId);\n }\n }\n });\n request.open(\"GET\", this.SnippetUrl + \"/\" + snippetId.replace(/#/g, \"/\"));\n request.send();\n });\n }\n }\n Animation._UniqueIdGenerator = 0;\n /**\n * Use matrix interpolation instead of using direct key value when animating matrices\n */\n Animation.AllowMatricesInterpolation = false;\n /**\n * When matrix interpolation is enabled, this boolean forces the system to use Matrix.DecomposeLerp instead of Matrix.Lerp. Interpolation is more precise but slower\n */\n Animation.AllowMatrixDecomposeForInterpolation = true;\n /** Define the Url to load snippets */\n Animation.SnippetUrl = `https://snippet.babylonjs.com`;\n // Statics\n /**\n * Float animation type\n */\n Animation.ANIMATIONTYPE_FLOAT = 0;\n /**\n * Vector3 animation type\n */\n Animation.ANIMATIONTYPE_VECTOR3 = 1;\n /**\n * Quaternion animation type\n */\n Animation.ANIMATIONTYPE_QUATERNION = 2;\n /**\n * Matrix animation type\n */\n Animation.ANIMATIONTYPE_MATRIX = 3;\n /**\n * Color3 animation type\n */\n Animation.ANIMATIONTYPE_COLOR3 = 4;\n /**\n * Color3 animation type\n */\n Animation.ANIMATIONTYPE_COLOR4 = 7;\n /**\n * Vector2 animation type\n */\n Animation.ANIMATIONTYPE_VECTOR2 = 5;\n /**\n * Size animation type\n */\n Animation.ANIMATIONTYPE_SIZE = 6;\n /**\n * Relative Loop Mode\n */\n Animation.ANIMATIONLOOPMODE_RELATIVE = 0;\n /**\n * Cycle Loop Mode\n */\n Animation.ANIMATIONLOOPMODE_CYCLE = 1;\n /**\n * Constant Loop Mode\n */\n Animation.ANIMATIONLOOPMODE_CONSTANT = 2;\n /**\n * Yoyo Loop Mode\n */\n Animation.ANIMATIONLOOPMODE_YOYO = 4;\n /**\n * Creates an animation or an array of animations from a snippet saved by the Inspector\n * @deprecated Please use ParseFromSnippetAsync instead\n * @param snippetId defines the snippet to load\n * @returns a promise that will resolve to the new animation or a new array of animations\n */\n Animation.CreateFromSnippetAsync = Animation.ParseFromSnippetAsync;\n RegisterClass(\"BABYLON.Animation\", Animation);\n Node._AnimationRangeFactory = (name, from, to) => new AnimationRange(name, from, to);\n\n /**\n * Add a bouncing effect to an ArcRotateCamera when reaching a specified minimum and maximum radius\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#bouncing-behavior\n */\n class BouncingBehavior {\n constructor() {\n /**\n * The duration of the animation, in milliseconds\n */\n this.transitionDuration = 450;\n /**\n * Length of the distance animated by the transition when lower radius is reached\n */\n this.lowerRadiusTransitionRange = 2;\n /**\n * Length of the distance animated by the transition when upper radius is reached\n */\n this.upperRadiusTransitionRange = -2;\n this._autoTransitionRange = false;\n // Animations\n this._radiusIsAnimating = false;\n this._radiusBounceTransition = null;\n this._animatables = new Array();\n }\n /**\n * Gets the name of the behavior.\n */\n get name() {\n return \"Bouncing\";\n }\n /**\n * Gets a value indicating if the lowerRadiusTransitionRange and upperRadiusTransitionRange are defined automatically\n */\n get autoTransitionRange() {\n return this._autoTransitionRange;\n }\n /**\n * Sets a value indicating if the lowerRadiusTransitionRange and upperRadiusTransitionRange are defined automatically\n * Transition ranges will be set to 5% of the bounding box diagonal in world space\n */\n set autoTransitionRange(value) {\n if (this._autoTransitionRange === value) {\n return;\n }\n this._autoTransitionRange = value;\n const camera = this._attachedCamera;\n if (!camera) {\n return;\n }\n if (value) {\n this._onMeshTargetChangedObserver = camera.onMeshTargetChangedObservable.add((mesh) => {\n if (!mesh) {\n return;\n }\n mesh.computeWorldMatrix(true);\n const diagonal = mesh.getBoundingInfo().diagonalLength;\n this.lowerRadiusTransitionRange = diagonal * 0.05;\n this.upperRadiusTransitionRange = diagonal * 0.05;\n });\n }\n else if (this._onMeshTargetChangedObserver) {\n camera.onMeshTargetChangedObservable.remove(this._onMeshTargetChangedObserver);\n }\n }\n /**\n * Initializes the behavior.\n */\n init() {\n // Do nothing\n }\n /**\n * Attaches the behavior to its arc rotate camera.\n * @param camera Defines the camera to attach the behavior to\n */\n attach(camera) {\n this._attachedCamera = camera;\n this._onAfterCheckInputsObserver = camera.onAfterCheckInputsObservable.add(() => {\n if (!this._attachedCamera) {\n return;\n }\n // Add the bounce animation to the lower radius limit\n if (this._isRadiusAtLimit(this._attachedCamera.lowerRadiusLimit)) {\n this._applyBoundRadiusAnimation(this.lowerRadiusTransitionRange);\n }\n // Add the bounce animation to the upper radius limit\n if (this._isRadiusAtLimit(this._attachedCamera.upperRadiusLimit)) {\n this._applyBoundRadiusAnimation(this.upperRadiusTransitionRange);\n }\n });\n }\n /**\n * Detaches the behavior from its current arc rotate camera.\n */\n detach() {\n if (!this._attachedCamera) {\n return;\n }\n if (this._onAfterCheckInputsObserver) {\n this._attachedCamera.onAfterCheckInputsObservable.remove(this._onAfterCheckInputsObserver);\n }\n if (this._onMeshTargetChangedObserver) {\n this._attachedCamera.onMeshTargetChangedObservable.remove(this._onMeshTargetChangedObserver);\n }\n this._attachedCamera = null;\n }\n /**\n * Checks if the camera radius is at the specified limit. Takes into account animation locks.\n * @param radiusLimit The limit to check against.\n * @returns Bool to indicate if at limit.\n */\n _isRadiusAtLimit(radiusLimit) {\n if (!this._attachedCamera) {\n return false;\n }\n if (this._attachedCamera.radius === radiusLimit && !this._radiusIsAnimating) {\n return true;\n }\n return false;\n }\n /**\n * Applies an animation to the radius of the camera, extending by the radiusDelta.\n * @param radiusDelta The delta by which to animate to. Can be negative.\n */\n _applyBoundRadiusAnimation(radiusDelta) {\n if (!this._attachedCamera) {\n return;\n }\n if (!this._radiusBounceTransition) {\n BouncingBehavior.EasingFunction.setEasingMode(BouncingBehavior.EasingMode);\n this._radiusBounceTransition = Animation.CreateAnimation(\"radius\", Animation.ANIMATIONTYPE_FLOAT, 60, BouncingBehavior.EasingFunction);\n }\n // Prevent zoom until bounce has completed\n this._cachedWheelPrecision = this._attachedCamera.wheelPrecision;\n this._attachedCamera.wheelPrecision = Infinity;\n this._attachedCamera.inertialRadiusOffset = 0;\n // Animate to the radius limit\n this.stopAllAnimations();\n this._radiusIsAnimating = true;\n const animatable = Animation.TransitionTo(\"radius\", this._attachedCamera.radius + radiusDelta, this._attachedCamera, this._attachedCamera.getScene(), 60, this._radiusBounceTransition, this.transitionDuration, () => this._clearAnimationLocks());\n if (animatable) {\n this._animatables.push(animatable);\n }\n }\n /**\n * Removes all animation locks. Allows new animations to be added to any of the camera properties.\n */\n _clearAnimationLocks() {\n this._radiusIsAnimating = false;\n if (this._attachedCamera) {\n this._attachedCamera.wheelPrecision = this._cachedWheelPrecision;\n }\n }\n /**\n * Stops and removes all animations that have been applied to the camera\n */\n stopAllAnimations() {\n if (this._attachedCamera) {\n this._attachedCamera.animations = [];\n }\n while (this._animatables.length) {\n this._animatables[0].onAnimationEnd = null;\n this._animatables[0].stop();\n this._animatables.shift();\n }\n }\n }\n /**\n * The easing function used by animations\n */\n BouncingBehavior.EasingFunction = new BackEase(0.3);\n /**\n * The easing mode used by animations\n */\n BouncingBehavior.EasingMode = EasingFunction.EASINGMODE_EASEOUT;\n\n /**\n * The framing behavior (FramingBehavior) is designed to automatically position an ArcRotateCamera when its target is set to a mesh. It is also useful if you want to prevent the camera to go under a virtual horizontal plane.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#framing-behavior\n */\n class FramingBehavior {\n constructor() {\n /**\n * An event triggered when the animation to zoom on target mesh has ended\n */\n this.onTargetFramingAnimationEndObservable = new Observable$1();\n this._mode = FramingBehavior.FitFrustumSidesMode;\n this._radiusScale = 1.0;\n this._positionScale = 0.5;\n this._defaultElevation = 0.3;\n this._elevationReturnTime = 1500;\n this._elevationReturnWaitTime = 1000;\n this._zoomStopsAnimation = false;\n this._framingTime = 1500;\n /**\n * Define if the behavior should automatically change the configured\n * camera limits and sensibilities.\n */\n this.autoCorrectCameraLimitsAndSensibility = true;\n this._isPointerDown = false;\n this._lastInteractionTime = -Infinity;\n // Framing control\n this._animatables = new Array();\n this._betaIsAnimating = false;\n }\n /**\n * Gets the name of the behavior.\n */\n get name() {\n return \"Framing\";\n }\n /**\n * Sets the current mode used by the behavior\n */\n set mode(mode) {\n this._mode = mode;\n }\n /**\n * Gets current mode used by the behavior.\n */\n get mode() {\n return this._mode;\n }\n /**\n * Sets the scale applied to the radius (1 by default)\n */\n set radiusScale(radius) {\n this._radiusScale = radius;\n }\n /**\n * Gets the scale applied to the radius\n */\n get radiusScale() {\n return this._radiusScale;\n }\n /**\n * Sets the scale to apply on Y axis to position camera focus. 0.5 by default which means the center of the bounding box.\n */\n set positionScale(scale) {\n this._positionScale = scale;\n }\n /**\n * Gets the scale to apply on Y axis to position camera focus. 0.5 by default which means the center of the bounding box.\n */\n get positionScale() {\n return this._positionScale;\n }\n /**\n * Sets the angle above/below the horizontal plane to return to when the return to default elevation idle\n * behaviour is triggered, in radians.\n */\n set defaultElevation(elevation) {\n this._defaultElevation = elevation;\n }\n /**\n * Gets the angle above/below the horizontal plane to return to when the return to default elevation idle\n * behaviour is triggered, in radians.\n */\n get defaultElevation() {\n return this._defaultElevation;\n }\n /**\n * Sets the time (in milliseconds) taken to return to the default beta position.\n * Negative value indicates camera should not return to default.\n */\n set elevationReturnTime(speed) {\n this._elevationReturnTime = speed;\n }\n /**\n * Gets the time (in milliseconds) taken to return to the default beta position.\n * Negative value indicates camera should not return to default.\n */\n get elevationReturnTime() {\n return this._elevationReturnTime;\n }\n /**\n * Sets the delay (in milliseconds) taken before the camera returns to the default beta position.\n */\n set elevationReturnWaitTime(time) {\n this._elevationReturnWaitTime = time;\n }\n /**\n * Gets the delay (in milliseconds) taken before the camera returns to the default beta position.\n */\n get elevationReturnWaitTime() {\n return this._elevationReturnWaitTime;\n }\n /**\n * Sets the flag that indicates if user zooming should stop animation.\n */\n set zoomStopsAnimation(flag) {\n this._zoomStopsAnimation = flag;\n }\n /**\n * Gets the flag that indicates if user zooming should stop animation.\n */\n get zoomStopsAnimation() {\n return this._zoomStopsAnimation;\n }\n /**\n * Sets the transition time when framing the mesh, in milliseconds\n */\n set framingTime(time) {\n this._framingTime = time;\n }\n /**\n * Gets the transition time when framing the mesh, in milliseconds\n */\n get framingTime() {\n return this._framingTime;\n }\n /**\n * Initializes the behavior.\n */\n init() {\n // Do nothing\n }\n /**\n * Attaches the behavior to its arc rotate camera.\n * @param camera Defines the camera to attach the behavior to\n */\n attach(camera) {\n this._attachedCamera = camera;\n const scene = this._attachedCamera.getScene();\n FramingBehavior.EasingFunction.setEasingMode(FramingBehavior.EasingMode);\n this._onPrePointerObservableObserver = scene.onPrePointerObservable.add((pointerInfoPre) => {\n if (pointerInfoPre.type === PointerEventTypes.POINTERDOWN) {\n this._isPointerDown = true;\n return;\n }\n if (pointerInfoPre.type === PointerEventTypes.POINTERUP) {\n this._isPointerDown = false;\n }\n });\n this._onMeshTargetChangedObserver = camera.onMeshTargetChangedObservable.add((mesh) => {\n if (mesh) {\n this.zoomOnMesh(mesh, undefined, () => {\n this.onTargetFramingAnimationEndObservable.notifyObservers();\n });\n }\n });\n this._onAfterCheckInputsObserver = camera.onAfterCheckInputsObservable.add(() => {\n // Stop the animation if there is user interaction and the animation should stop for this interaction\n this._applyUserInteraction();\n // Maintain the camera above the ground. If the user pulls the camera beneath the ground plane, lift it\n // back to the default position after a given timeout\n this._maintainCameraAboveGround();\n });\n }\n /**\n * Detaches the behavior from its current arc rotate camera.\n */\n detach() {\n if (!this._attachedCamera) {\n return;\n }\n const scene = this._attachedCamera.getScene();\n if (this._onPrePointerObservableObserver) {\n scene.onPrePointerObservable.remove(this._onPrePointerObservableObserver);\n }\n if (this._onAfterCheckInputsObserver) {\n this._attachedCamera.onAfterCheckInputsObservable.remove(this._onAfterCheckInputsObserver);\n }\n if (this._onMeshTargetChangedObserver) {\n this._attachedCamera.onMeshTargetChangedObservable.remove(this._onMeshTargetChangedObserver);\n }\n this._attachedCamera = null;\n }\n /**\n * Targets the given mesh and updates zoom level accordingly.\n * @param mesh The mesh to target.\n * @param focusOnOriginXZ Determines if the camera should focus on 0 in the X and Z axis instead of the mesh\n * @param onAnimationEnd Callback triggered at the end of the framing animation\n */\n zoomOnMesh(mesh, focusOnOriginXZ = false, onAnimationEnd = null) {\n mesh.computeWorldMatrix(true);\n const boundingBox = mesh.getBoundingInfo().boundingBox;\n this.zoomOnBoundingInfo(boundingBox.minimumWorld, boundingBox.maximumWorld, focusOnOriginXZ, onAnimationEnd);\n }\n /**\n * Targets the given mesh with its children and updates zoom level accordingly.\n * @param mesh The mesh to target.\n * @param focusOnOriginXZ Determines if the camera should focus on 0 in the X and Z axis instead of the mesh\n * @param onAnimationEnd Callback triggered at the end of the framing animation\n */\n zoomOnMeshHierarchy(mesh, focusOnOriginXZ = false, onAnimationEnd = null) {\n mesh.computeWorldMatrix(true);\n const boundingBox = mesh.getHierarchyBoundingVectors(true);\n this.zoomOnBoundingInfo(boundingBox.min, boundingBox.max, focusOnOriginXZ, onAnimationEnd);\n }\n /**\n * Targets the given meshes with their children and updates zoom level accordingly.\n * @param meshes The mesh to target.\n * @param focusOnOriginXZ Determines if the camera should focus on 0 in the X and Z axis instead of the mesh\n * @param onAnimationEnd Callback triggered at the end of the framing animation\n */\n zoomOnMeshesHierarchy(meshes, focusOnOriginXZ = false, onAnimationEnd = null) {\n const min = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\n const max = new Vector3(-Number.MAX_VALUE, -Number.MAX_VALUE, -Number.MAX_VALUE);\n for (let i = 0; i < meshes.length; i++) {\n const boundingInfo = meshes[i].getHierarchyBoundingVectors(true);\n Vector3.CheckExtends(boundingInfo.min, min, max);\n Vector3.CheckExtends(boundingInfo.max, min, max);\n }\n this.zoomOnBoundingInfo(min, max, focusOnOriginXZ, onAnimationEnd);\n }\n /**\n * Targets the bounding box info defined by its extends and updates zoom level accordingly.\n * @param minimumWorld Determines the smaller position of the bounding box extend\n * @param maximumWorld Determines the bigger position of the bounding box extend\n * @param focusOnOriginXZ Determines if the camera should focus on 0 in the X and Z axis instead of the mesh\n * @param onAnimationEnd Callback triggered at the end of the framing animation\n * @returns true if the zoom was done\n */\n zoomOnBoundingInfo(minimumWorld, maximumWorld, focusOnOriginXZ = false, onAnimationEnd = null) {\n let zoomTarget;\n if (!this._attachedCamera) {\n return false;\n }\n // Find target by interpolating from bottom of bounding box in world-space to top via framingPositionY\n const bottom = minimumWorld.y;\n const top = maximumWorld.y;\n const zoomTargetY = bottom + (top - bottom) * this._positionScale;\n const radiusWorld = maximumWorld.subtract(minimumWorld).scale(0.5);\n if (focusOnOriginXZ) {\n zoomTarget = new Vector3(0, zoomTargetY, 0);\n }\n else {\n const centerWorld = minimumWorld.add(radiusWorld);\n zoomTarget = new Vector3(centerWorld.x, zoomTargetY, centerWorld.z);\n }\n if (!this._vectorTransition) {\n this._vectorTransition = Animation.CreateAnimation(\"target\", Animation.ANIMATIONTYPE_VECTOR3, 60, FramingBehavior.EasingFunction);\n }\n this._betaIsAnimating = true;\n let animatable = Animation.TransitionTo(\"target\", zoomTarget, this._attachedCamera, this._attachedCamera.getScene(), 60, this._vectorTransition, this._framingTime);\n if (animatable) {\n this._animatables.push(animatable);\n }\n // sets the radius and lower radius bounds\n // Small delta ensures camera is not always at lower zoom limit.\n let radius = 0;\n if (this._mode === FramingBehavior.FitFrustumSidesMode) {\n const position = this._calculateLowerRadiusFromModelBoundingSphere(minimumWorld, maximumWorld);\n if (this.autoCorrectCameraLimitsAndSensibility) {\n this._attachedCamera.lowerRadiusLimit = radiusWorld.length() + this._attachedCamera.minZ;\n }\n radius = position;\n }\n else if (this._mode === FramingBehavior.IgnoreBoundsSizeMode) {\n radius = this._calculateLowerRadiusFromModelBoundingSphere(minimumWorld, maximumWorld);\n if (this.autoCorrectCameraLimitsAndSensibility && this._attachedCamera.lowerRadiusLimit === null) {\n this._attachedCamera.lowerRadiusLimit = this._attachedCamera.minZ;\n }\n }\n // Set sensibilities\n if (this.autoCorrectCameraLimitsAndSensibility) {\n const extend = maximumWorld.subtract(minimumWorld).length();\n this._attachedCamera.panningSensibility = 5000 / extend;\n this._attachedCamera.wheelPrecision = 100 / radius;\n }\n // transition to new radius\n if (!this._radiusTransition) {\n this._radiusTransition = Animation.CreateAnimation(\"radius\", Animation.ANIMATIONTYPE_FLOAT, 60, FramingBehavior.EasingFunction);\n }\n animatable = Animation.TransitionTo(\"radius\", radius, this._attachedCamera, this._attachedCamera.getScene(), 60, this._radiusTransition, this._framingTime, () => {\n this.stopAllAnimations();\n if (onAnimationEnd) {\n onAnimationEnd();\n }\n if (this._attachedCamera && this._attachedCamera.useInputToRestoreState) {\n this._attachedCamera.storeState();\n }\n });\n if (animatable) {\n this._animatables.push(animatable);\n }\n return true;\n }\n /**\n * Calculates the lowest radius for the camera based on the bounding box of the mesh.\n * @param minimumWorld\n * @param maximumWorld\n * @returns The minimum distance from the primary mesh's center point at which the camera must be kept in order\n *\t\t to fully enclose the mesh in the viewing frustum.\n */\n _calculateLowerRadiusFromModelBoundingSphere(minimumWorld, maximumWorld) {\n const size = maximumWorld.subtract(minimumWorld);\n const boxVectorGlobalDiagonal = size.length();\n const frustumSlope = this._getFrustumSlope();\n // Formula for setting distance\n // (Good explanation: http://stackoverflow.com/questions/2866350/move-camera-to-fit-3d-scene)\n const radiusWithoutFraming = boxVectorGlobalDiagonal * 0.5;\n // Horizon distance\n const radius = radiusWithoutFraming * this._radiusScale;\n const distanceForHorizontalFrustum = radius * Math.sqrt(1.0 + 1.0 / (frustumSlope.x * frustumSlope.x));\n const distanceForVerticalFrustum = radius * Math.sqrt(1.0 + 1.0 / (frustumSlope.y * frustumSlope.y));\n let distance = Math.max(distanceForHorizontalFrustum, distanceForVerticalFrustum);\n const camera = this._attachedCamera;\n if (!camera) {\n return 0;\n }\n if (camera.lowerRadiusLimit && this._mode === FramingBehavior.IgnoreBoundsSizeMode) {\n // Don't exceed the requested limit\n distance = distance < camera.lowerRadiusLimit ? camera.lowerRadiusLimit : distance;\n }\n // Don't exceed the upper radius limit\n if (camera.upperRadiusLimit) {\n distance = distance > camera.upperRadiusLimit ? camera.upperRadiusLimit : distance;\n }\n return distance;\n }\n /**\n * Keeps the camera above the ground plane. If the user pulls the camera below the ground plane, the camera\n * is automatically returned to its default position (expected to be above ground plane).\n */\n _maintainCameraAboveGround() {\n if (this._elevationReturnTime < 0) {\n return;\n }\n const timeSinceInteraction = PrecisionDate.Now - this._lastInteractionTime;\n const defaultBeta = Math.PI * 0.5 - this._defaultElevation;\n const limitBeta = Math.PI * 0.5;\n // Bring the camera back up if below the ground plane\n if (this._attachedCamera && !this._betaIsAnimating && this._attachedCamera.beta > limitBeta && timeSinceInteraction >= this._elevationReturnWaitTime) {\n this._betaIsAnimating = true;\n //Transition to new position\n this.stopAllAnimations();\n if (!this._betaTransition) {\n this._betaTransition = Animation.CreateAnimation(\"beta\", Animation.ANIMATIONTYPE_FLOAT, 60, FramingBehavior.EasingFunction);\n }\n const animatabe = Animation.TransitionTo(\"beta\", defaultBeta, this._attachedCamera, this._attachedCamera.getScene(), 60, this._betaTransition, this._elevationReturnTime, () => {\n this._clearAnimationLocks();\n this.stopAllAnimations();\n });\n if (animatabe) {\n this._animatables.push(animatabe);\n }\n }\n }\n /**\n * Returns the frustum slope based on the canvas ratio and camera FOV\n * @returns The frustum slope represented as a Vector2 with X and Y slopes\n */\n _getFrustumSlope() {\n // Calculate the viewport ratio\n // Aspect Ratio is Height/Width.\n const camera = this._attachedCamera;\n if (!camera) {\n return Vector2.Zero();\n }\n const engine = camera.getScene().getEngine();\n const aspectRatio = engine.getAspectRatio(camera);\n // Camera FOV is the vertical field of view (top-bottom) in radians.\n // Slope of the frustum top/bottom planes in view space, relative to the forward vector.\n const frustumSlopeY = Math.tan(camera.fov / 2);\n // Slope of the frustum left/right planes in view space, relative to the forward vector.\n // Provides the amount that one side (e.g. left) of the frustum gets wider for every unit\n // along the forward vector.\n const frustumSlopeX = frustumSlopeY * aspectRatio;\n return new Vector2(frustumSlopeX, frustumSlopeY);\n }\n /**\n * Removes all animation locks. Allows new animations to be added to any of the arcCamera properties.\n */\n _clearAnimationLocks() {\n this._betaIsAnimating = false;\n }\n /**\n * Applies any current user interaction to the camera. Takes into account maximum alpha rotation.\n */\n _applyUserInteraction() {\n if (this.isUserIsMoving) {\n this._lastInteractionTime = PrecisionDate.Now;\n this.stopAllAnimations();\n this._clearAnimationLocks();\n }\n }\n /**\n * Stops and removes all animations that have been applied to the camera\n */\n stopAllAnimations() {\n if (this._attachedCamera) {\n this._attachedCamera.animations = [];\n }\n while (this._animatables.length) {\n if (this._animatables[0]) {\n this._animatables[0].onAnimationEnd = null;\n this._animatables[0].stop();\n }\n this._animatables.shift();\n }\n }\n /**\n * Gets a value indicating if the user is moving the camera\n */\n get isUserIsMoving() {\n if (!this._attachedCamera) {\n return false;\n }\n return (this._attachedCamera.inertialAlphaOffset !== 0 ||\n this._attachedCamera.inertialBetaOffset !== 0 ||\n this._attachedCamera.inertialRadiusOffset !== 0 ||\n this._attachedCamera.inertialPanningX !== 0 ||\n this._attachedCamera.inertialPanningY !== 0 ||\n this._isPointerDown);\n }\n }\n /**\n * The easing function used by animations\n */\n FramingBehavior.EasingFunction = new ExponentialEase();\n /**\n * The easing mode used by animations\n */\n FramingBehavior.EasingMode = EasingFunction.EASINGMODE_EASEINOUT;\n // Statics\n /**\n * The camera can move all the way towards the mesh.\n */\n FramingBehavior.IgnoreBoundsSizeMode = 0;\n /**\n * The camera is not allowed to zoom closer to the mesh than the point at which the adjusted bounding sphere touches the frustum sides\n */\n FramingBehavior.FitFrustumSidesMode = 1;\n\n /**\n * A target camera takes a mesh or position as a target and continues to look at it while it moves.\n * This is the base of the follow, arc rotate cameras and Free camera\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras\n */\n class TargetCamera extends Camera {\n /**\n * Instantiates a target camera that takes a mesh or position as a target and continues to look at it while it moves.\n * This is the base of the follow, arc rotate cameras and Free camera\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras\n * @param name Defines the name of the camera in the scene\n * @param position Defines the start position of the camera in the scene\n * @param scene Defines the scene the camera belongs to\n * @param setActiveOnSceneIfNoneActive Defines whether the camera should be marked as active if not other active cameras have been defined\n */\n constructor(name, position, scene, setActiveOnSceneIfNoneActive = true) {\n super(name, position, scene, setActiveOnSceneIfNoneActive);\n this._tmpUpVector = Vector3.Zero();\n this._tmpTargetVector = Vector3.Zero();\n /**\n * Define the current direction the camera is moving to\n */\n this.cameraDirection = new Vector3(0, 0, 0);\n /**\n * Define the current rotation the camera is rotating to\n */\n this.cameraRotation = new Vector2(0, 0);\n /** Gets or sets a boolean indicating that the scaling of the parent hierarchy will not be taken in account by the camera */\n this.ignoreParentScaling = false;\n /**\n * When set, the up vector of the camera will be updated by the rotation of the camera\n */\n this.updateUpVectorFromRotation = false;\n this._tmpQuaternion = new Quaternion();\n /**\n * Define the current rotation of the camera\n */\n this.rotation = new Vector3(0, 0, 0);\n /**\n * Define the current speed of the camera\n */\n this.speed = 2.0;\n /**\n * Add constraint to the camera to prevent it to move freely in all directions and\n * around all axis.\n */\n this.noRotationConstraint = false;\n /**\n * Reverses mouselook direction to 'natural' panning as opposed to traditional direct\n * panning\n */\n this.invertRotation = false;\n /**\n * Speed multiplier for inverse camera panning\n */\n this.inverseRotationSpeed = 0.2;\n /**\n * Define the current target of the camera as an object or a position.\n * Please note that locking a target will disable panning.\n */\n this.lockedTarget = null;\n /** @internal */\n this._currentTarget = Vector3.Zero();\n /** @internal */\n this._initialFocalDistance = 1;\n /** @internal */\n this._viewMatrix = Matrix.Zero();\n /** @internal */\n this._camMatrix = Matrix.Zero();\n /** @internal */\n this._cameraTransformMatrix = Matrix.Zero();\n /** @internal */\n this._cameraRotationMatrix = Matrix.Zero();\n /** @internal */\n this._referencePoint = new Vector3(0, 0, 1);\n /** @internal */\n this._transformedReferencePoint = Vector3.Zero();\n this._deferredPositionUpdate = new Vector3();\n this._deferredRotationQuaternionUpdate = new Quaternion();\n this._deferredRotationUpdate = new Vector3();\n this._deferredUpdated = false;\n this._deferOnly = false;\n this._defaultUp = Vector3.Up();\n this._cachedRotationZ = 0;\n this._cachedQuaternionRotationZ = 0;\n }\n /**\n * Gets the position in front of the camera at a given distance.\n * @param distance The distance from the camera we want the position to be\n * @returns the position\n */\n getFrontPosition(distance) {\n this.getWorldMatrix();\n const direction = this.getTarget().subtract(this.position);\n direction.normalize();\n direction.scaleInPlace(distance);\n return this.globalPosition.add(direction);\n }\n /** @internal */\n _getLockedTargetPosition() {\n if (!this.lockedTarget) {\n return null;\n }\n if (this.lockedTarget.absolutePosition) {\n const lockedTarget = this.lockedTarget;\n const m = lockedTarget.computeWorldMatrix();\n // in some cases the absolute position resets externally, but doesn't update since the matrix is cached.\n m.getTranslationToRef(lockedTarget.absolutePosition);\n }\n return this.lockedTarget.absolutePosition || this.lockedTarget;\n }\n /**\n * Store current camera state of the camera (fov, position, rotation, etc..)\n * @returns the camera\n */\n storeState() {\n this._storedPosition = this.position.clone();\n this._storedRotation = this.rotation.clone();\n if (this.rotationQuaternion) {\n this._storedRotationQuaternion = this.rotationQuaternion.clone();\n }\n return super.storeState();\n }\n /**\n * Restored camera state. You must call storeState() first\n * @returns whether it was successful or not\n * @internal\n */\n _restoreStateValues() {\n if (!super._restoreStateValues()) {\n return false;\n }\n this.position = this._storedPosition.clone();\n this.rotation = this._storedRotation.clone();\n if (this.rotationQuaternion) {\n this.rotationQuaternion = this._storedRotationQuaternion.clone();\n }\n this.cameraDirection.copyFromFloats(0, 0, 0);\n this.cameraRotation.copyFromFloats(0, 0);\n return true;\n }\n /** @internal */\n _initCache() {\n super._initCache();\n this._cache.lockedTarget = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\n this._cache.rotation = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\n this._cache.rotationQuaternion = new Quaternion(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\n }\n /**\n * @internal\n */\n _updateCache(ignoreParentClass) {\n if (!ignoreParentClass) {\n super._updateCache();\n }\n const lockedTargetPosition = this._getLockedTargetPosition();\n if (!lockedTargetPosition) {\n this._cache.lockedTarget = null;\n }\n else {\n if (!this._cache.lockedTarget) {\n this._cache.lockedTarget = lockedTargetPosition.clone();\n }\n else {\n this._cache.lockedTarget.copyFrom(lockedTargetPosition);\n }\n }\n this._cache.rotation.copyFrom(this.rotation);\n if (this.rotationQuaternion) {\n this._cache.rotationQuaternion.copyFrom(this.rotationQuaternion);\n }\n }\n // Synchronized\n /** @internal */\n _isSynchronizedViewMatrix() {\n if (!super._isSynchronizedViewMatrix()) {\n return false;\n }\n const lockedTargetPosition = this._getLockedTargetPosition();\n return ((this._cache.lockedTarget ? this._cache.lockedTarget.equals(lockedTargetPosition) : !lockedTargetPosition) &&\n (this.rotationQuaternion ? this.rotationQuaternion.equals(this._cache.rotationQuaternion) : this._cache.rotation.equals(this.rotation)));\n }\n // Methods\n /** @internal */\n _computeLocalCameraSpeed() {\n const engine = this.getEngine();\n return this.speed * Math.sqrt(engine.getDeltaTime() / (engine.getFps() * 100.0));\n }\n // Target\n /**\n * Defines the target the camera should look at.\n * @param target Defines the new target as a Vector\n */\n setTarget(target) {\n this.upVector.normalize();\n this._initialFocalDistance = target.subtract(this.position).length();\n if (this.position.z === target.z) {\n this.position.z += Epsilon;\n }\n this._referencePoint.normalize().scaleInPlace(this._initialFocalDistance);\n Matrix.LookAtLHToRef(this.position, target, this._defaultUp, this._camMatrix);\n this._camMatrix.invert();\n this.rotation.x = Math.atan(this._camMatrix.m[6] / this._camMatrix.m[10]);\n const vDir = target.subtract(this.position);\n if (vDir.x >= 0.0) {\n this.rotation.y = -Math.atan(vDir.z / vDir.x) + Math.PI / 2.0;\n }\n else {\n this.rotation.y = -Math.atan(vDir.z / vDir.x) - Math.PI / 2.0;\n }\n this.rotation.z = 0;\n if (isNaN(this.rotation.x)) {\n this.rotation.x = 0;\n }\n if (isNaN(this.rotation.y)) {\n this.rotation.y = 0;\n }\n if (isNaN(this.rotation.z)) {\n this.rotation.z = 0;\n }\n if (this.rotationQuaternion) {\n Quaternion.RotationYawPitchRollToRef(this.rotation.y, this.rotation.x, this.rotation.z, this.rotationQuaternion);\n }\n }\n /**\n * Defines the target point of the camera.\n * The camera looks towards it form the radius distance.\n */\n get target() {\n return this.getTarget();\n }\n set target(value) {\n this.setTarget(value);\n }\n /**\n * Return the current target position of the camera. This value is expressed in local space.\n * @returns the target position\n */\n getTarget() {\n return this._currentTarget;\n }\n /** @internal */\n _decideIfNeedsToMove() {\n return Math.abs(this.cameraDirection.x) > 0 || Math.abs(this.cameraDirection.y) > 0 || Math.abs(this.cameraDirection.z) > 0;\n }\n /** @internal */\n _updatePosition() {\n if (this.parent) {\n this.parent.getWorldMatrix().invertToRef(TmpVectors.Matrix[0]);\n Vector3.TransformNormalToRef(this.cameraDirection, TmpVectors.Matrix[0], TmpVectors.Vector3[0]);\n this._deferredPositionUpdate.addInPlace(TmpVectors.Vector3[0]);\n if (!this._deferOnly) {\n this.position.copyFrom(this._deferredPositionUpdate);\n }\n else {\n this._deferredUpdated = true;\n }\n return;\n }\n this._deferredPositionUpdate.addInPlace(this.cameraDirection);\n if (!this._deferOnly) {\n this.position.copyFrom(this._deferredPositionUpdate);\n }\n else {\n this._deferredUpdated = true;\n }\n }\n /** @internal */\n _checkInputs() {\n const directionMultiplier = this.invertRotation ? -this.inverseRotationSpeed : 1.0;\n const needToMove = this._decideIfNeedsToMove();\n const needToRotate = this.cameraRotation.x || this.cameraRotation.y;\n this._deferredUpdated = false;\n this._deferredRotationUpdate.copyFrom(this.rotation);\n this._deferredPositionUpdate.copyFrom(this.position);\n if (this.rotationQuaternion) {\n this._deferredRotationQuaternionUpdate.copyFrom(this.rotationQuaternion);\n }\n // Move\n if (needToMove) {\n this._updatePosition();\n }\n // Rotate\n if (needToRotate) {\n //rotate, if quaternion is set and rotation was used\n if (this.rotationQuaternion) {\n this.rotationQuaternion.toEulerAnglesToRef(this._deferredRotationUpdate);\n }\n this._deferredRotationUpdate.x += this.cameraRotation.x * directionMultiplier;\n this._deferredRotationUpdate.y += this.cameraRotation.y * directionMultiplier;\n // Apply constraints\n if (!this.noRotationConstraint) {\n const limit = 1.570796;\n if (this._deferredRotationUpdate.x > limit) {\n this._deferredRotationUpdate.x = limit;\n }\n if (this._deferredRotationUpdate.x < -limit) {\n this._deferredRotationUpdate.x = -limit;\n }\n }\n if (!this._deferOnly) {\n this.rotation.copyFrom(this._deferredRotationUpdate);\n }\n else {\n this._deferredUpdated = true;\n }\n //rotate, if quaternion is set and rotation was used\n if (this.rotationQuaternion) {\n const len = this._deferredRotationUpdate.lengthSquared();\n if (len) {\n Quaternion.RotationYawPitchRollToRef(this._deferredRotationUpdate.y, this._deferredRotationUpdate.x, this._deferredRotationUpdate.z, this._deferredRotationQuaternionUpdate);\n if (!this._deferOnly) {\n this.rotationQuaternion.copyFrom(this._deferredRotationQuaternionUpdate);\n }\n else {\n this._deferredUpdated = true;\n }\n }\n }\n }\n // Inertia\n if (needToMove) {\n if (Math.abs(this.cameraDirection.x) < this.speed * Epsilon) {\n this.cameraDirection.x = 0;\n }\n if (Math.abs(this.cameraDirection.y) < this.speed * Epsilon) {\n this.cameraDirection.y = 0;\n }\n if (Math.abs(this.cameraDirection.z) < this.speed * Epsilon) {\n this.cameraDirection.z = 0;\n }\n this.cameraDirection.scaleInPlace(this.inertia);\n }\n if (needToRotate) {\n if (Math.abs(this.cameraRotation.x) < this.speed * Epsilon) {\n this.cameraRotation.x = 0;\n }\n if (Math.abs(this.cameraRotation.y) < this.speed * Epsilon) {\n this.cameraRotation.y = 0;\n }\n this.cameraRotation.scaleInPlace(this.inertia);\n }\n super._checkInputs();\n }\n _updateCameraRotationMatrix() {\n if (this.rotationQuaternion) {\n this.rotationQuaternion.toRotationMatrix(this._cameraRotationMatrix);\n }\n else {\n Matrix.RotationYawPitchRollToRef(this.rotation.y, this.rotation.x, this.rotation.z, this._cameraRotationMatrix);\n }\n }\n /**\n * Update the up vector to apply the rotation of the camera (So if you changed the camera rotation.z this will let you update the up vector as well)\n * @returns the current camera\n */\n _rotateUpVectorWithCameraRotationMatrix() {\n Vector3.TransformNormalToRef(this._defaultUp, this._cameraRotationMatrix, this.upVector);\n return this;\n }\n /** @internal */\n _getViewMatrix() {\n if (this.lockedTarget) {\n this.setTarget(this._getLockedTargetPosition());\n }\n // Compute\n this._updateCameraRotationMatrix();\n // Apply the changed rotation to the upVector\n if (this.rotationQuaternion && this._cachedQuaternionRotationZ != this.rotationQuaternion.z) {\n this._rotateUpVectorWithCameraRotationMatrix();\n this._cachedQuaternionRotationZ = this.rotationQuaternion.z;\n }\n else if (this._cachedRotationZ !== this.rotation.z) {\n this._rotateUpVectorWithCameraRotationMatrix();\n this._cachedRotationZ = this.rotation.z;\n }\n Vector3.TransformCoordinatesToRef(this._referencePoint, this._cameraRotationMatrix, this._transformedReferencePoint);\n // Computing target and final matrix\n this.position.addToRef(this._transformedReferencePoint, this._currentTarget);\n if (this.updateUpVectorFromRotation) {\n if (this.rotationQuaternion) {\n Axis.Y.rotateByQuaternionToRef(this.rotationQuaternion, this.upVector);\n }\n else {\n Quaternion.FromEulerVectorToRef(this.rotation, this._tmpQuaternion);\n Axis.Y.rotateByQuaternionToRef(this._tmpQuaternion, this.upVector);\n }\n }\n this._computeViewMatrix(this.position, this._currentTarget, this.upVector);\n return this._viewMatrix;\n }\n _computeViewMatrix(position, target, up) {\n if (this.ignoreParentScaling) {\n if (this.parent) {\n const parentWorldMatrix = this.parent.getWorldMatrix();\n Vector3.TransformCoordinatesToRef(position, parentWorldMatrix, this._globalPosition);\n Vector3.TransformCoordinatesToRef(target, parentWorldMatrix, this._tmpTargetVector);\n Vector3.TransformNormalToRef(up, parentWorldMatrix, this._tmpUpVector);\n this._markSyncedWithParent();\n }\n else {\n this._globalPosition.copyFrom(position);\n this._tmpTargetVector.copyFrom(target);\n this._tmpUpVector.copyFrom(up);\n }\n if (this.getScene().useRightHandedSystem) {\n Matrix.LookAtRHToRef(this._globalPosition, this._tmpTargetVector, this._tmpUpVector, this._viewMatrix);\n }\n else {\n Matrix.LookAtLHToRef(this._globalPosition, this._tmpTargetVector, this._tmpUpVector, this._viewMatrix);\n }\n return;\n }\n if (this.getScene().useRightHandedSystem) {\n Matrix.LookAtRHToRef(position, target, up, this._viewMatrix);\n }\n else {\n Matrix.LookAtLHToRef(position, target, up, this._viewMatrix);\n }\n if (this.parent) {\n const parentWorldMatrix = this.parent.getWorldMatrix();\n this._viewMatrix.invert();\n this._viewMatrix.multiplyToRef(parentWorldMatrix, this._viewMatrix);\n this._viewMatrix.getTranslationToRef(this._globalPosition);\n this._viewMatrix.invert();\n this._markSyncedWithParent();\n }\n else {\n this._globalPosition.copyFrom(position);\n }\n }\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n createRigCamera(name, cameraIndex) {\n if (this.cameraRigMode !== Camera.RIG_MODE_NONE) {\n const rigCamera = new TargetCamera(name, this.position.clone(), this.getScene());\n rigCamera.isRigCamera = true;\n rigCamera.rigParent = this;\n if (this.cameraRigMode === Camera.RIG_MODE_VR || this.cameraRigMode === Camera.RIG_MODE_WEBVR) {\n if (!this.rotationQuaternion) {\n this.rotationQuaternion = new Quaternion();\n }\n rigCamera._cameraRigParams = {};\n rigCamera.rotationQuaternion = new Quaternion();\n }\n rigCamera.mode = this.mode;\n rigCamera.orthoLeft = this.orthoLeft;\n rigCamera.orthoRight = this.orthoRight;\n rigCamera.orthoTop = this.orthoTop;\n rigCamera.orthoBottom = this.orthoBottom;\n return rigCamera;\n }\n return null;\n }\n /**\n * @internal\n */\n _updateRigCameras() {\n const camLeft = this._rigCameras[0];\n const camRight = this._rigCameras[1];\n this.computeWorldMatrix();\n switch (this.cameraRigMode) {\n case Camera.RIG_MODE_STEREOSCOPIC_ANAGLYPH:\n case Camera.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL:\n case Camera.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED:\n case Camera.RIG_MODE_STEREOSCOPIC_OVERUNDER:\n case Camera.RIG_MODE_STEREOSCOPIC_INTERLACED: {\n //provisionnaly using _cameraRigParams.stereoHalfAngle instead of calculations based on _cameraRigParams.interaxialDistance:\n const leftSign = this.cameraRigMode === Camera.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED ? 1 : -1;\n const rightSign = this.cameraRigMode === Camera.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED ? -1 : 1;\n this._getRigCamPositionAndTarget(this._cameraRigParams.stereoHalfAngle * leftSign, camLeft);\n this._getRigCamPositionAndTarget(this._cameraRigParams.stereoHalfAngle * rightSign, camRight);\n break;\n }\n case Camera.RIG_MODE_VR:\n if (camLeft.rotationQuaternion) {\n camLeft.rotationQuaternion.copyFrom(this.rotationQuaternion);\n camRight.rotationQuaternion.copyFrom(this.rotationQuaternion);\n }\n else {\n camLeft.rotation.copyFrom(this.rotation);\n camRight.rotation.copyFrom(this.rotation);\n }\n camLeft.position.copyFrom(this.position);\n camRight.position.copyFrom(this.position);\n break;\n }\n super._updateRigCameras();\n }\n _getRigCamPositionAndTarget(halfSpace, rigCamera) {\n const target = this.getTarget();\n target.subtractToRef(this.position, TargetCamera._TargetFocalPoint);\n TargetCamera._TargetFocalPoint.normalize().scaleInPlace(this._initialFocalDistance);\n const newFocalTarget = TargetCamera._TargetFocalPoint.addInPlace(this.position);\n Matrix.TranslationToRef(-newFocalTarget.x, -newFocalTarget.y, -newFocalTarget.z, TargetCamera._TargetTransformMatrix);\n TargetCamera._TargetTransformMatrix.multiplyToRef(Matrix.RotationAxis(rigCamera.upVector, halfSpace), TargetCamera._RigCamTransformMatrix);\n Matrix.TranslationToRef(newFocalTarget.x, newFocalTarget.y, newFocalTarget.z, TargetCamera._TargetTransformMatrix);\n TargetCamera._RigCamTransformMatrix.multiplyToRef(TargetCamera._TargetTransformMatrix, TargetCamera._RigCamTransformMatrix);\n Vector3.TransformCoordinatesToRef(this.position, TargetCamera._RigCamTransformMatrix, rigCamera.position);\n rigCamera.setTarget(newFocalTarget);\n }\n /**\n * Gets the current object class name.\n * @returns the class name\n */\n getClassName() {\n return \"TargetCamera\";\n }\n }\n TargetCamera._RigCamTransformMatrix = new Matrix();\n TargetCamera._TargetTransformMatrix = new Matrix();\n TargetCamera._TargetFocalPoint = new Vector3();\n __decorate$1([\n serializeAsVector3()\n ], TargetCamera.prototype, \"rotation\", void 0);\n __decorate$1([\n serialize()\n ], TargetCamera.prototype, \"speed\", void 0);\n __decorate$1([\n serializeAsMeshReference(\"lockedTargetId\")\n ], TargetCamera.prototype, \"lockedTarget\", void 0);\n\n /**\n * @ignore\n * This is a list of all the different input types that are available in the application.\n * Fo instance: ArcRotateCameraGamepadInput...\n */\n // eslint-disable-next-line no-var, @typescript-eslint/naming-convention\n var CameraInputTypes = {};\n /**\n * This represents the input manager used within a camera.\n * It helps dealing with all the different kind of input attached to a camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\n */\n class CameraInputsManager {\n /**\n * Instantiate a new Camera Input Manager.\n * @param camera Defines the camera the input manager belongs to\n */\n constructor(camera) {\n /**\n * Defines the dom element the camera is collecting inputs from.\n * This is null if the controls have not been attached.\n */\n this.attachedToElement = false;\n this.attached = {};\n this.camera = camera;\n this.checkInputs = () => { };\n }\n /**\n * Add an input method to a camera\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\n * @param input Camera input method\n */\n add(input) {\n const type = input.getSimpleName();\n if (this.attached[type]) {\n Logger.Warn(\"camera input of type \" + type + \" already exists on camera\");\n return;\n }\n this.attached[type] = input;\n input.camera = this.camera;\n // for checkInputs, we are dynamically creating a function\n // the goal is to avoid the performance penalty of looping for inputs in the render loop\n if (input.checkInputs) {\n this.checkInputs = this._addCheckInputs(input.checkInputs.bind(input));\n }\n if (this.attachedToElement) {\n input.attachControl(this.noPreventDefault);\n }\n }\n /**\n * Remove a specific input method from a camera\n * example: camera.inputs.remove(camera.inputs.attached.mouse);\n * @param inputToRemove camera input method\n */\n remove(inputToRemove) {\n for (const cam in this.attached) {\n const input = this.attached[cam];\n if (input === inputToRemove) {\n input.detachControl();\n input.camera = null;\n delete this.attached[cam];\n this.rebuildInputCheck();\n return;\n }\n }\n }\n /**\n * Remove a specific input type from a camera\n * example: camera.inputs.remove(\"ArcRotateCameraGamepadInput\");\n * @param inputType the type of the input to remove\n */\n removeByType(inputType) {\n for (const cam in this.attached) {\n const input = this.attached[cam];\n if (input.getClassName() === inputType) {\n input.detachControl();\n input.camera = null;\n delete this.attached[cam];\n this.rebuildInputCheck();\n }\n }\n }\n _addCheckInputs(fn) {\n const current = this.checkInputs;\n return () => {\n current();\n fn();\n };\n }\n /**\n * Attach the input controls to the currently attached dom element to listen the events from.\n * @param input Defines the input to attach\n */\n attachInput(input) {\n if (this.attachedToElement) {\n input.attachControl(this.noPreventDefault);\n }\n }\n /**\n * Attach the current manager inputs controls to a specific dom element to listen the events from.\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\n */\n attachElement(noPreventDefault = false) {\n if (this.attachedToElement) {\n return;\n }\n noPreventDefault = Camera.ForceAttachControlToAlwaysPreventDefault ? false : noPreventDefault;\n this.attachedToElement = true;\n this.noPreventDefault = noPreventDefault;\n for (const cam in this.attached) {\n this.attached[cam].attachControl(noPreventDefault);\n }\n }\n /**\n * Detach the current manager inputs controls from a specific dom element.\n * @param disconnect Defines whether the input should be removed from the current list of attached inputs\n */\n detachElement(disconnect = false) {\n for (const cam in this.attached) {\n this.attached[cam].detachControl();\n if (disconnect) {\n this.attached[cam].camera = null;\n }\n }\n this.attachedToElement = false;\n }\n /**\n * Rebuild the dynamic inputCheck function from the current list of\n * defined inputs in the manager.\n */\n rebuildInputCheck() {\n this.checkInputs = () => { };\n for (const cam in this.attached) {\n const input = this.attached[cam];\n if (input.checkInputs) {\n this.checkInputs = this._addCheckInputs(input.checkInputs.bind(input));\n }\n }\n }\n /**\n * Remove all attached input methods from a camera\n */\n clear() {\n if (this.attachedToElement) {\n this.detachElement(true);\n }\n this.attached = {};\n this.attachedToElement = false;\n this.checkInputs = () => { };\n }\n /**\n * Serialize the current input manager attached to a camera.\n * This ensures than once parsed,\n * the input associated to the camera will be identical to the current ones\n * @param serializedCamera Defines the camera serialization JSON the input serialization should write to\n */\n serialize(serializedCamera) {\n const inputs = {};\n for (const cam in this.attached) {\n const input = this.attached[cam];\n const res = SerializationHelper.Serialize(input);\n inputs[input.getClassName()] = res;\n }\n serializedCamera.inputsmgr = inputs;\n }\n /**\n * Parses an input manager serialized JSON to restore the previous list of inputs\n * and states associated to a camera.\n * @param parsedCamera Defines the JSON to parse\n */\n parse(parsedCamera) {\n const parsedInputs = parsedCamera.inputsmgr;\n if (parsedInputs) {\n this.clear();\n for (const n in parsedInputs) {\n const construct = CameraInputTypes[n];\n if (construct) {\n const parsedinput = parsedInputs[n];\n const input = SerializationHelper.Parse(() => {\n return new construct();\n }, parsedinput, null);\n this.add(input);\n }\n }\n }\n else {\n //2016-03-08 this part is for managing backward compatibility\n for (const n in this.attached) {\n const construct = CameraInputTypes[this.attached[n].getClassName()];\n if (construct) {\n const input = SerializationHelper.Parse(() => {\n return new construct();\n }, parsedCamera, null);\n this.remove(this.attached[n]);\n this.add(input);\n }\n }\n }\n }\n }\n\n /**\n * Base class for Camera Pointer Inputs.\n * See FollowCameraPointersInput in src/Cameras/Inputs/followCameraPointersInput.ts\n * for example usage.\n */\n class BaseCameraPointersInput {\n constructor() {\n this._currentActiveButton = -1;\n /**\n * Defines the buttons associated with the input to handle camera move.\n */\n this.buttons = [0, 1, 2];\n }\n /**\n * Attach the input controls to a specific dom element to get the input from.\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\n */\n attachControl(noPreventDefault) {\n // eslint-disable-next-line prefer-rest-params\n noPreventDefault = Tools.BackCompatCameraNoPreventDefault(arguments);\n const engine = this.camera.getEngine();\n const element = engine.getInputElement();\n let previousPinchSquaredDistance = 0;\n let previousMultiTouchPanPosition = null;\n this._pointA = null;\n this._pointB = null;\n this._altKey = false;\n this._ctrlKey = false;\n this._metaKey = false;\n this._shiftKey = false;\n this._buttonsPressed = 0;\n this._pointerInput = (p) => {\n var _a, _b;\n const evt = p.event;\n const isTouch = evt.pointerType === \"touch\";\n if (engine.isInVRExclusivePointerMode) {\n return;\n }\n if (p.type !== PointerEventTypes.POINTERMOVE && this.buttons.indexOf(evt.button) === -1) {\n return;\n }\n const srcElement = evt.target;\n this._altKey = evt.altKey;\n this._ctrlKey = evt.ctrlKey;\n this._metaKey = evt.metaKey;\n this._shiftKey = evt.shiftKey;\n this._buttonsPressed = evt.buttons;\n if (engine.isPointerLock) {\n const offsetX = evt.movementX;\n const offsetY = evt.movementY;\n this.onTouch(null, offsetX, offsetY);\n this._pointA = null;\n this._pointB = null;\n }\n else if (p.type !== PointerEventTypes.POINTERDOWN && isTouch && ((_a = this._pointA) === null || _a === void 0 ? void 0 : _a.pointerId) !== evt.pointerId && ((_b = this._pointB) === null || _b === void 0 ? void 0 : _b.pointerId) !== evt.pointerId) {\n return; // If we get a non-down event for a touch that we're not tracking, ignore it\n }\n else if (p.type === PointerEventTypes.POINTERDOWN && (this._currentActiveButton === -1 || isTouch)) {\n try {\n srcElement === null || srcElement === void 0 ? void 0 : srcElement.setPointerCapture(evt.pointerId);\n }\n catch (e) {\n //Nothing to do with the error. Execution will continue.\n }\n if (this._pointA === null) {\n this._pointA = {\n x: evt.clientX,\n y: evt.clientY,\n pointerId: evt.pointerId,\n type: evt.pointerType,\n };\n }\n else if (this._pointB === null) {\n this._pointB = {\n x: evt.clientX,\n y: evt.clientY,\n pointerId: evt.pointerId,\n type: evt.pointerType,\n };\n }\n else {\n return; // We are already tracking two pointers so ignore this one\n }\n if (this._currentActiveButton === -1 && !isTouch) {\n this._currentActiveButton = evt.button;\n }\n this.onButtonDown(evt);\n if (!noPreventDefault) {\n evt.preventDefault();\n element && element.focus();\n }\n }\n else if (p.type === PointerEventTypes.POINTERDOUBLETAP) {\n this.onDoubleTap(evt.pointerType);\n }\n else if (p.type === PointerEventTypes.POINTERUP && (this._currentActiveButton === evt.button || isTouch)) {\n try {\n srcElement === null || srcElement === void 0 ? void 0 : srcElement.releasePointerCapture(evt.pointerId);\n }\n catch (e) {\n //Nothing to do with the error.\n }\n if (!isTouch) {\n this._pointB = null; // Mouse and pen are mono pointer\n }\n //would be better to use pointers.remove(evt.pointerId) for multitouch gestures,\n //but emptying completely pointers collection is required to fix a bug on iPhone :\n //when changing orientation while pinching camera,\n //one pointer stay pressed forever if we don't release all pointers\n //will be ok to put back pointers.remove(evt.pointerId); when iPhone bug corrected\n if (engine._badOS) {\n this._pointA = this._pointB = null;\n }\n else {\n //only remove the impacted pointer in case of multitouch allowing on most\n //platforms switching from rotate to zoom and pan seamlessly.\n if (this._pointB && this._pointA && this._pointA.pointerId == evt.pointerId) {\n this._pointA = this._pointB;\n this._pointB = null;\n }\n else if (this._pointA && this._pointB && this._pointB.pointerId == evt.pointerId) {\n this._pointB = null;\n }\n else {\n this._pointA = this._pointB = null;\n }\n }\n if (previousPinchSquaredDistance !== 0 || previousMultiTouchPanPosition) {\n // Previous pinch data is populated but a button has been lifted\n // so pinch has ended.\n this.onMultiTouch(this._pointA, this._pointB, previousPinchSquaredDistance, 0, // pinchSquaredDistance\n previousMultiTouchPanPosition, null // multiTouchPanPosition\n );\n previousPinchSquaredDistance = 0;\n previousMultiTouchPanPosition = null;\n }\n this._currentActiveButton = -1;\n this.onButtonUp(evt);\n if (!noPreventDefault) {\n evt.preventDefault();\n }\n }\n else if (p.type === PointerEventTypes.POINTERMOVE) {\n if (!noPreventDefault) {\n evt.preventDefault();\n }\n // One button down\n if (this._pointA && this._pointB === null) {\n const offsetX = evt.clientX - this._pointA.x;\n const offsetY = evt.clientY - this._pointA.y;\n this.onTouch(this._pointA, offsetX, offsetY);\n this._pointA.x = evt.clientX;\n this._pointA.y = evt.clientY;\n }\n // Two buttons down: pinch\n else if (this._pointA && this._pointB) {\n const ed = this._pointA.pointerId === evt.pointerId ? this._pointA : this._pointB;\n ed.x = evt.clientX;\n ed.y = evt.clientY;\n const distX = this._pointA.x - this._pointB.x;\n const distY = this._pointA.y - this._pointB.y;\n const pinchSquaredDistance = distX * distX + distY * distY;\n const multiTouchPanPosition = {\n x: (this._pointA.x + this._pointB.x) / 2,\n y: (this._pointA.y + this._pointB.y) / 2,\n pointerId: evt.pointerId,\n type: p.type,\n };\n this.onMultiTouch(this._pointA, this._pointB, previousPinchSquaredDistance, pinchSquaredDistance, previousMultiTouchPanPosition, multiTouchPanPosition);\n previousMultiTouchPanPosition = multiTouchPanPosition;\n previousPinchSquaredDistance = pinchSquaredDistance;\n }\n }\n };\n this._observer = this.camera\n .getScene()\n ._inputManager._addCameraPointerObserver(this._pointerInput, PointerEventTypes.POINTERDOWN | PointerEventTypes.POINTERUP | PointerEventTypes.POINTERMOVE | PointerEventTypes.POINTERDOUBLETAP);\n this._onLostFocus = () => {\n this._pointA = this._pointB = null;\n previousPinchSquaredDistance = 0;\n previousMultiTouchPanPosition = null;\n this.onLostFocus();\n };\n this._contextMenuBind = this.onContextMenu.bind(this);\n element && element.addEventListener(\"contextmenu\", this._contextMenuBind, false);\n const hostWindow = this.camera.getScene().getEngine().getHostWindow();\n if (hostWindow) {\n Tools.RegisterTopRootEvents(hostWindow, [{ name: \"blur\", handler: this._onLostFocus }]);\n }\n }\n /**\n * Detach the current controls from the specified dom element.\n */\n detachControl() {\n if (this._onLostFocus) {\n const hostWindow = this.camera.getScene().getEngine().getHostWindow();\n if (hostWindow) {\n Tools.UnregisterTopRootEvents(hostWindow, [{ name: \"blur\", handler: this._onLostFocus }]);\n }\n }\n if (this._observer) {\n this.camera.getScene()._inputManager._removeCameraPointerObserver(this._observer);\n this._observer = null;\n if (this._contextMenuBind) {\n const inputElement = this.camera.getScene().getEngine().getInputElement();\n inputElement && inputElement.removeEventListener(\"contextmenu\", this._contextMenuBind);\n }\n this._onLostFocus = null;\n }\n this._altKey = false;\n this._ctrlKey = false;\n this._metaKey = false;\n this._shiftKey = false;\n this._buttonsPressed = 0;\n this._currentActiveButton = -1;\n }\n /**\n * Gets the class name of the current input.\n * @returns the class name\n */\n getClassName() {\n return \"BaseCameraPointersInput\";\n }\n /**\n * Get the friendly name associated with the input class.\n * @returns the input friendly name\n */\n getSimpleName() {\n return \"pointers\";\n }\n /**\n * Called on pointer POINTERDOUBLETAP event.\n * Override this method to provide functionality on POINTERDOUBLETAP event.\n * @param type\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n onDoubleTap(type) { }\n /**\n * Called on pointer POINTERMOVE event if only a single touch is active.\n * Override this method to provide functionality.\n * @param point\n * @param offsetX\n * @param offsetY\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n onTouch(point, offsetX, offsetY) { }\n /**\n * Called on pointer POINTERMOVE event if multiple touches are active.\n * Override this method to provide functionality.\n * @param _pointA\n * @param _pointB\n * @param previousPinchSquaredDistance\n * @param pinchSquaredDistance\n * @param previousMultiTouchPanPosition\n * @param multiTouchPanPosition\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n onMultiTouch(_pointA, _pointB, previousPinchSquaredDistance, pinchSquaredDistance, previousMultiTouchPanPosition, multiTouchPanPosition) { }\n /**\n * Called on JS contextmenu event.\n * Override this method to provide functionality.\n * @param evt\n */\n onContextMenu(evt) {\n evt.preventDefault();\n }\n /**\n * Called each time a new POINTERDOWN event occurs. Ie, for each button\n * press.\n * Override this method to provide functionality.\n * @param evt\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n onButtonDown(evt) { }\n /**\n * Called each time a new POINTERUP event occurs. Ie, for each button\n * release.\n * Override this method to provide functionality.\n * @param evt\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n onButtonUp(evt) { }\n /**\n * Called when window becomes inactive.\n * Override this method to provide functionality.\n */\n onLostFocus() { }\n }\n __decorate$1([\n serialize()\n ], BaseCameraPointersInput.prototype, \"buttons\", void 0);\n\n /**\n * Manage the pointers inputs to control an arc rotate camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\n */\n class ArcRotateCameraPointersInput extends BaseCameraPointersInput {\n constructor() {\n super(...arguments);\n /**\n * Defines the buttons associated with the input to handle camera move.\n */\n this.buttons = [0, 1, 2];\n /**\n * Defines the pointer angular sensibility along the X axis or how fast is\n * the camera rotating.\n */\n this.angularSensibilityX = 1000.0;\n /**\n * Defines the pointer angular sensibility along the Y axis or how fast is\n * the camera rotating.\n */\n this.angularSensibilityY = 1000.0;\n /**\n * Defines the pointer pinch precision or how fast is the camera zooming.\n */\n this.pinchPrecision = 12.0;\n /**\n * pinchDeltaPercentage will be used instead of pinchPrecision if different\n * from 0.\n * It defines the percentage of current camera.radius to use as delta when\n * pinch zoom is used.\n */\n this.pinchDeltaPercentage = 0;\n /**\n * When useNaturalPinchZoom is true, multi touch zoom will zoom in such\n * that any object in the plane at the camera's target point will scale\n * perfectly with finger motion.\n * Overrides pinchDeltaPercentage and pinchPrecision.\n */\n this.useNaturalPinchZoom = false;\n /**\n * Defines whether zoom (2 fingers pinch) is enabled through multitouch\n */\n this.pinchZoom = true;\n /**\n * Defines the pointer panning sensibility or how fast is the camera moving.\n */\n this.panningSensibility = 1000.0;\n /**\n * Defines whether panning (2 fingers swipe) is enabled through multitouch.\n */\n this.multiTouchPanning = true;\n /**\n * Defines whether panning is enabled for both pan (2 fingers swipe) and\n * zoom (pinch) through multitouch.\n */\n this.multiTouchPanAndZoom = true;\n /**\n * Revers pinch action direction.\n */\n this.pinchInwards = true;\n this._isPanClick = false;\n this._twoFingerActivityCount = 0;\n this._isPinching = false;\n }\n /**\n * Gets the class name of the current input.\n * @returns the class name\n */\n getClassName() {\n return \"ArcRotateCameraPointersInput\";\n }\n /**\n * Move camera from multi touch panning positions.\n * @param previousMultiTouchPanPosition\n * @param multiTouchPanPosition\n */\n _computeMultiTouchPanning(previousMultiTouchPanPosition, multiTouchPanPosition) {\n if (this.panningSensibility !== 0 && previousMultiTouchPanPosition && multiTouchPanPosition) {\n const moveDeltaX = multiTouchPanPosition.x - previousMultiTouchPanPosition.x;\n const moveDeltaY = multiTouchPanPosition.y - previousMultiTouchPanPosition.y;\n this.camera.inertialPanningX += -moveDeltaX / this.panningSensibility;\n this.camera.inertialPanningY += moveDeltaY / this.panningSensibility;\n }\n }\n /**\n * Move camera from pinch zoom distances.\n * @param previousPinchSquaredDistance\n * @param pinchSquaredDistance\n */\n _computePinchZoom(previousPinchSquaredDistance, pinchSquaredDistance) {\n const radius = this.camera.radius || ArcRotateCameraPointersInput.MinimumRadiusForPinch;\n if (this.useNaturalPinchZoom) {\n this.camera.radius = (radius * Math.sqrt(previousPinchSquaredDistance)) / Math.sqrt(pinchSquaredDistance);\n }\n else if (this.pinchDeltaPercentage) {\n this.camera.inertialRadiusOffset += (pinchSquaredDistance - previousPinchSquaredDistance) * 0.001 * radius * this.pinchDeltaPercentage;\n }\n else {\n this.camera.inertialRadiusOffset +=\n (pinchSquaredDistance - previousPinchSquaredDistance) /\n ((this.pinchPrecision * (this.pinchInwards ? 1 : -1) * (this.angularSensibilityX + this.angularSensibilityY)) / 2);\n }\n }\n /**\n * Called on pointer POINTERMOVE event if only a single touch is active.\n * @param point\n * @param offsetX\n * @param offsetY\n */\n onTouch(point, offsetX, offsetY) {\n if (this.panningSensibility !== 0 && ((this._ctrlKey && this.camera._useCtrlForPanning) || this._isPanClick)) {\n this.camera.inertialPanningX += -offsetX / this.panningSensibility;\n this.camera.inertialPanningY += offsetY / this.panningSensibility;\n }\n else {\n this.camera.inertialAlphaOffset -= offsetX / this.angularSensibilityX;\n this.camera.inertialBetaOffset -= offsetY / this.angularSensibilityY;\n }\n }\n /**\n * Called on pointer POINTERDOUBLETAP event.\n */\n onDoubleTap() {\n if (this.camera.useInputToRestoreState) {\n this.camera.restoreState();\n }\n }\n /**\n * Called on pointer POINTERMOVE event if multiple touches are active.\n * @param pointA\n * @param pointB\n * @param previousPinchSquaredDistance\n * @param pinchSquaredDistance\n * @param previousMultiTouchPanPosition\n * @param multiTouchPanPosition\n */\n onMultiTouch(pointA, pointB, previousPinchSquaredDistance, pinchSquaredDistance, previousMultiTouchPanPosition, multiTouchPanPosition) {\n if (previousPinchSquaredDistance === 0 && previousMultiTouchPanPosition === null) {\n // First time this method is called for new pinch.\n // Next time this is called there will be a\n // previousPinchSquaredDistance and pinchSquaredDistance to compare.\n return;\n }\n if (pinchSquaredDistance === 0 && multiTouchPanPosition === null) {\n // Last time this method is called at the end of a pinch.\n return;\n }\n // Zoom and panning enabled together\n if (this.multiTouchPanAndZoom) {\n this._computePinchZoom(previousPinchSquaredDistance, pinchSquaredDistance);\n this._computeMultiTouchPanning(previousMultiTouchPanPosition, multiTouchPanPosition);\n // Zoom and panning enabled but only one at a time\n }\n else if (this.multiTouchPanning && this.pinchZoom) {\n this._twoFingerActivityCount++;\n if (this._isPinching ||\n (this._twoFingerActivityCount < 20 && Math.abs(Math.sqrt(pinchSquaredDistance) - Math.sqrt(previousPinchSquaredDistance)) > this.camera.pinchToPanMaxDistance)) {\n // Since pinch has not been active long, assume we intend to zoom.\n this._computePinchZoom(previousPinchSquaredDistance, pinchSquaredDistance);\n // Since we are pinching, remain pinching on next iteration.\n this._isPinching = true;\n }\n else {\n // Pause between pinch starting and moving implies not a zoom event. Pan instead.\n this._computeMultiTouchPanning(previousMultiTouchPanPosition, multiTouchPanPosition);\n }\n // Panning enabled, zoom disabled\n }\n else if (this.multiTouchPanning) {\n this._computeMultiTouchPanning(previousMultiTouchPanPosition, multiTouchPanPosition);\n // Zoom enabled, panning disabled\n }\n else if (this.pinchZoom) {\n this._computePinchZoom(previousPinchSquaredDistance, pinchSquaredDistance);\n }\n }\n /**\n * Called each time a new POINTERDOWN event occurs. Ie, for each button\n * press.\n * @param evt\n */\n onButtonDown(evt) {\n this._isPanClick = evt.button === this.camera._panningMouseButton;\n }\n /**\n * Called each time a new POINTERUP event occurs. Ie, for each button\n * release.\n */\n onButtonUp() {\n this._twoFingerActivityCount = 0;\n this._isPinching = false;\n }\n /**\n * Called when window becomes inactive.\n */\n onLostFocus() {\n this._isPanClick = false;\n this._twoFingerActivityCount = 0;\n this._isPinching = false;\n }\n }\n /**\n * The minimum radius used for pinch, to avoid radius lock at 0\n */\n ArcRotateCameraPointersInput.MinimumRadiusForPinch = 0.001;\n __decorate$1([\n serialize()\n ], ArcRotateCameraPointersInput.prototype, \"buttons\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCameraPointersInput.prototype, \"angularSensibilityX\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCameraPointersInput.prototype, \"angularSensibilityY\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCameraPointersInput.prototype, \"pinchPrecision\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCameraPointersInput.prototype, \"pinchDeltaPercentage\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCameraPointersInput.prototype, \"useNaturalPinchZoom\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCameraPointersInput.prototype, \"pinchZoom\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCameraPointersInput.prototype, \"panningSensibility\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCameraPointersInput.prototype, \"multiTouchPanning\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCameraPointersInput.prototype, \"multiTouchPanAndZoom\", void 0);\n CameraInputTypes[\"ArcRotateCameraPointersInput\"] = ArcRotateCameraPointersInput;\n\n /**\n * Manage the keyboard inputs to control the movement of an arc rotate camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\n */\n class ArcRotateCameraKeyboardMoveInput {\n constructor() {\n /**\n * Defines the list of key codes associated with the up action (increase alpha)\n */\n this.keysUp = [38];\n /**\n * Defines the list of key codes associated with the down action (decrease alpha)\n */\n this.keysDown = [40];\n /**\n * Defines the list of key codes associated with the left action (increase beta)\n */\n this.keysLeft = [37];\n /**\n * Defines the list of key codes associated with the right action (decrease beta)\n */\n this.keysRight = [39];\n /**\n * Defines the list of key codes associated with the reset action.\n * Those keys reset the camera to its last stored state (with the method camera.storeState())\n */\n this.keysReset = [220];\n /**\n * Defines the panning sensibility of the inputs.\n * (How fast is the camera panning)\n */\n this.panningSensibility = 50.0;\n /**\n * Defines the zooming sensibility of the inputs.\n * (How fast is the camera zooming)\n */\n this.zoomingSensibility = 25.0;\n /**\n * Defines whether maintaining the alt key down switch the movement mode from\n * orientation to zoom.\n */\n this.useAltToZoom = true;\n /**\n * Rotation speed of the camera\n */\n this.angularSpeed = 0.01;\n this._keys = new Array();\n }\n /**\n * Attach the input controls to a specific dom element to get the input from.\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\n */\n attachControl(noPreventDefault) {\n // was there a second variable defined?\n // eslint-disable-next-line prefer-rest-params\n noPreventDefault = Tools.BackCompatCameraNoPreventDefault(arguments);\n if (this._onCanvasBlurObserver) {\n return;\n }\n this._scene = this.camera.getScene();\n this._engine = this._scene.getEngine();\n this._onCanvasBlurObserver = this._engine.onCanvasBlurObservable.add(() => {\n this._keys.length = 0;\n });\n this._onKeyboardObserver = this._scene.onKeyboardObservable.add((info) => {\n const evt = info.event;\n if (!evt.metaKey) {\n if (info.type === KeyboardEventTypes.KEYDOWN) {\n this._ctrlPressed = evt.ctrlKey;\n this._altPressed = evt.altKey;\n if (this.keysUp.indexOf(evt.keyCode) !== -1 ||\n this.keysDown.indexOf(evt.keyCode) !== -1 ||\n this.keysLeft.indexOf(evt.keyCode) !== -1 ||\n this.keysRight.indexOf(evt.keyCode) !== -1 ||\n this.keysReset.indexOf(evt.keyCode) !== -1) {\n const index = this._keys.indexOf(evt.keyCode);\n if (index === -1) {\n this._keys.push(evt.keyCode);\n }\n if (evt.preventDefault) {\n if (!noPreventDefault) {\n evt.preventDefault();\n }\n }\n }\n }\n else {\n if (this.keysUp.indexOf(evt.keyCode) !== -1 ||\n this.keysDown.indexOf(evt.keyCode) !== -1 ||\n this.keysLeft.indexOf(evt.keyCode) !== -1 ||\n this.keysRight.indexOf(evt.keyCode) !== -1 ||\n this.keysReset.indexOf(evt.keyCode) !== -1) {\n const index = this._keys.indexOf(evt.keyCode);\n if (index >= 0) {\n this._keys.splice(index, 1);\n }\n if (evt.preventDefault) {\n if (!noPreventDefault) {\n evt.preventDefault();\n }\n }\n }\n }\n }\n });\n }\n /**\n * Detach the current controls from the specified dom element.\n */\n detachControl() {\n if (this._scene) {\n if (this._onKeyboardObserver) {\n this._scene.onKeyboardObservable.remove(this._onKeyboardObserver);\n }\n if (this._onCanvasBlurObserver) {\n this._engine.onCanvasBlurObservable.remove(this._onCanvasBlurObserver);\n }\n this._onKeyboardObserver = null;\n this._onCanvasBlurObserver = null;\n }\n this._keys.length = 0;\n }\n /**\n * Update the current camera state depending on the inputs that have been used this frame.\n * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop.\n */\n checkInputs() {\n if (this._onKeyboardObserver) {\n const camera = this.camera;\n for (let index = 0; index < this._keys.length; index++) {\n const keyCode = this._keys[index];\n if (this.keysLeft.indexOf(keyCode) !== -1) {\n if (this._ctrlPressed && this.camera._useCtrlForPanning) {\n camera.inertialPanningX -= 1 / this.panningSensibility;\n }\n else {\n camera.inertialAlphaOffset -= this.angularSpeed;\n }\n }\n else if (this.keysUp.indexOf(keyCode) !== -1) {\n if (this._ctrlPressed && this.camera._useCtrlForPanning) {\n camera.inertialPanningY += 1 / this.panningSensibility;\n }\n else if (this._altPressed && this.useAltToZoom) {\n camera.inertialRadiusOffset += 1 / this.zoomingSensibility;\n }\n else {\n camera.inertialBetaOffset -= this.angularSpeed;\n }\n }\n else if (this.keysRight.indexOf(keyCode) !== -1) {\n if (this._ctrlPressed && this.camera._useCtrlForPanning) {\n camera.inertialPanningX += 1 / this.panningSensibility;\n }\n else {\n camera.inertialAlphaOffset += this.angularSpeed;\n }\n }\n else if (this.keysDown.indexOf(keyCode) !== -1) {\n if (this._ctrlPressed && this.camera._useCtrlForPanning) {\n camera.inertialPanningY -= 1 / this.panningSensibility;\n }\n else if (this._altPressed && this.useAltToZoom) {\n camera.inertialRadiusOffset -= 1 / this.zoomingSensibility;\n }\n else {\n camera.inertialBetaOffset += this.angularSpeed;\n }\n }\n else if (this.keysReset.indexOf(keyCode) !== -1) {\n if (camera.useInputToRestoreState) {\n camera.restoreState();\n }\n }\n }\n }\n }\n /**\n * Gets the class name of the current input.\n * @returns the class name\n */\n getClassName() {\n return \"ArcRotateCameraKeyboardMoveInput\";\n }\n /**\n * Get the friendly name associated with the input class.\n * @returns the input friendly name\n */\n getSimpleName() {\n return \"keyboard\";\n }\n }\n __decorate$1([\n serialize()\n ], ArcRotateCameraKeyboardMoveInput.prototype, \"keysUp\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCameraKeyboardMoveInput.prototype, \"keysDown\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCameraKeyboardMoveInput.prototype, \"keysLeft\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCameraKeyboardMoveInput.prototype, \"keysRight\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCameraKeyboardMoveInput.prototype, \"keysReset\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCameraKeyboardMoveInput.prototype, \"panningSensibility\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCameraKeyboardMoveInput.prototype, \"zoomingSensibility\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCameraKeyboardMoveInput.prototype, \"useAltToZoom\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCameraKeyboardMoveInput.prototype, \"angularSpeed\", void 0);\n CameraInputTypes[\"ArcRotateCameraKeyboardMoveInput\"] = ArcRotateCameraKeyboardMoveInput;\n\n /**\n * Firefox uses a different scheme to report scroll distances to other\n * browsers. Rather than use complicated methods to calculate the exact\n * multiple we need to apply, let's just cheat and use a constant.\n * https://developer.mozilla.org/en-US/docs/Web/API/WheelEvent/deltaMode\n * https://stackoverflow.com/questions/20110224/what-is-the-height-of-a-line-in-a-wheel-event-deltamode-dom-delta-line\n */\n const ffMultiplier = 40;\n /**\n * Manage the mouse wheel inputs to control an arc rotate camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\n */\n class ArcRotateCameraMouseWheelInput {\n constructor() {\n /**\n * Gets or Set the mouse wheel precision or how fast is the camera zooming.\n */\n this.wheelPrecision = 3.0;\n /**\n * Gets or Set the boolean value that controls whether or not the mouse wheel\n * zooms to the location of the mouse pointer or not. The default is false.\n */\n this.zoomToMouseLocation = false;\n /**\n * wheelDeltaPercentage will be used instead of wheelPrecision if different from 0.\n * It defines the percentage of current camera.radius to use as delta when wheel is used.\n */\n this.wheelDeltaPercentage = 0;\n /**\n * If set, this function will be used to set the radius delta that will be added to the current camera radius\n */\n this.customComputeDeltaFromMouseWheel = null;\n this._inertialPanning = Vector3.Zero();\n }\n _computeDeltaFromMouseWheelLegacyEvent(mouseWheelDelta, radius) {\n let delta = 0;\n const wheelDelta = mouseWheelDelta * 0.01 * this.wheelDeltaPercentage * radius;\n if (mouseWheelDelta > 0) {\n delta = wheelDelta / (1.0 + this.wheelDeltaPercentage);\n }\n else {\n delta = wheelDelta * (1.0 + this.wheelDeltaPercentage);\n }\n return delta;\n }\n /**\n * Attach the input controls to a specific dom element to get the input from.\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\n */\n attachControl(noPreventDefault) {\n noPreventDefault = Tools.BackCompatCameraNoPreventDefault(arguments);\n this._wheel = (p) => {\n //sanity check - this should be a PointerWheel event.\n if (p.type !== PointerEventTypes.POINTERWHEEL) {\n return;\n }\n const event = p.event;\n let delta = 0;\n const platformScale = event.deltaMode === EventConstants.DOM_DELTA_LINE ? ffMultiplier : 1; // If this happens to be set to DOM_DELTA_LINE, adjust accordingly\n const wheelDelta = -(event.deltaY * platformScale);\n if (this.customComputeDeltaFromMouseWheel) {\n delta = this.customComputeDeltaFromMouseWheel(wheelDelta, this, event);\n }\n else {\n if (this.wheelDeltaPercentage) {\n delta = this._computeDeltaFromMouseWheelLegacyEvent(wheelDelta, this.camera.radius);\n // If zooming in, estimate the target radius and use that to compute the delta for inertia\n // this will stop multiple scroll events zooming in from adding too much inertia\n if (delta > 0) {\n let estimatedTargetRadius = this.camera.radius;\n let targetInertia = this.camera.inertialRadiusOffset + delta;\n for (let i = 0; i < 20 && Math.abs(targetInertia) > 0.001; i++) {\n estimatedTargetRadius -= targetInertia;\n targetInertia *= this.camera.inertia;\n }\n estimatedTargetRadius = Scalar.Clamp(estimatedTargetRadius, 0, Number.MAX_VALUE);\n delta = this._computeDeltaFromMouseWheelLegacyEvent(wheelDelta, estimatedTargetRadius);\n }\n }\n else {\n delta = wheelDelta / (this.wheelPrecision * 40);\n }\n }\n if (delta) {\n if (this.zoomToMouseLocation) {\n // If we are zooming to the mouse location, then we need to get the hit plane at the start of the zoom gesture if it doesn't exist\n // The hit plane is normally calculated after the first motion and each time there's motion so if we don't do this first,\n // the first zoom will be to the center of the screen\n if (!this._hitPlane) {\n this._updateHitPlane();\n }\n this._zoomToMouse(delta);\n }\n else {\n this.camera.inertialRadiusOffset += delta;\n }\n }\n if (event.preventDefault) {\n if (!noPreventDefault) {\n event.preventDefault();\n }\n }\n };\n this._observer = this.camera.getScene()._inputManager._addCameraPointerObserver(this._wheel, PointerEventTypes.POINTERWHEEL);\n if (this.zoomToMouseLocation) {\n this._inertialPanning.setAll(0);\n }\n }\n /**\n * Detach the current controls from the specified dom element.\n */\n detachControl() {\n if (this._observer) {\n this.camera.getScene()._inputManager._removeCameraPointerObserver(this._observer);\n this._observer = null;\n this._wheel = null;\n }\n }\n /**\n * Update the current camera state depending on the inputs that have been used this frame.\n * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop.\n */\n checkInputs() {\n if (!this.zoomToMouseLocation) {\n return;\n }\n const camera = this.camera;\n const motion = 0.0 + camera.inertialAlphaOffset + camera.inertialBetaOffset + camera.inertialRadiusOffset;\n if (motion) {\n // if zooming is still happening as a result of inertia, then we also need to update\n // the hit plane.\n this._updateHitPlane();\n // Note we cannot use arcRotateCamera.inertialPlanning here because arcRotateCamera panning\n // uses a different panningInertia which could cause this panning to get out of sync with\n // the zooming, and for this to work they must be exactly in sync.\n camera.target.addInPlace(this._inertialPanning);\n this._inertialPanning.scaleInPlace(camera.inertia);\n this._zeroIfClose(this._inertialPanning);\n }\n }\n /**\n * Gets the class name of the current input.\n * @returns the class name\n */\n getClassName() {\n return \"ArcRotateCameraMouseWheelInput\";\n }\n /**\n * Get the friendly name associated with the input class.\n * @returns the input friendly name\n */\n getSimpleName() {\n return \"mousewheel\";\n }\n _updateHitPlane() {\n const camera = this.camera;\n const direction = camera.target.subtract(camera.position);\n this._hitPlane = Plane.FromPositionAndNormal(camera.target, direction);\n }\n // Get position on the hit plane\n _getPosition() {\n var _a;\n const camera = this.camera;\n const scene = camera.getScene();\n // since the _hitPlane is always updated to be orthogonal to the camera position vector\n // we don't have to worry about this ray shooting off to infinity. This ray creates\n // a vector defining where we want to zoom to.\n const ray = scene.createPickingRay(scene.pointerX, scene.pointerY, Matrix.Identity(), camera, false);\n // Since the camera is the origin of the picking ray, we need to offset it by the camera's offset manually\n ray.origin.x -= camera.targetScreenOffset.x;\n ray.origin.y -= camera.targetScreenOffset.y;\n let distance = 0;\n if (this._hitPlane) {\n distance = (_a = ray.intersectsPlane(this._hitPlane)) !== null && _a !== void 0 ? _a : 0;\n }\n // not using this ray again, so modifying its vectors here is fine\n return ray.origin.addInPlace(ray.direction.scaleInPlace(distance));\n }\n _zoomToMouse(delta) {\n var _a, _b;\n const camera = this.camera;\n const inertiaComp = 1 - camera.inertia;\n if (camera.lowerRadiusLimit) {\n const lowerLimit = (_a = camera.lowerRadiusLimit) !== null && _a !== void 0 ? _a : 0;\n if (camera.radius - (camera.inertialRadiusOffset + delta) / inertiaComp < lowerLimit) {\n delta = (camera.radius - lowerLimit) * inertiaComp - camera.inertialRadiusOffset;\n }\n }\n if (camera.upperRadiusLimit) {\n const upperLimit = (_b = camera.upperRadiusLimit) !== null && _b !== void 0 ? _b : 0;\n if (camera.radius - (camera.inertialRadiusOffset + delta) / inertiaComp > upperLimit) {\n delta = (camera.radius - upperLimit) * inertiaComp - camera.inertialRadiusOffset;\n }\n }\n const zoomDistance = delta / inertiaComp;\n const ratio = zoomDistance / camera.radius;\n const vec = this._getPosition();\n // Now this vector tells us how much we also need to pan the camera\n // so the targeted mouse location becomes the center of zooming.\n const directionToZoomLocation = TmpVectors.Vector3[6];\n vec.subtractToRef(camera.target, directionToZoomLocation);\n directionToZoomLocation.scaleInPlace(ratio);\n directionToZoomLocation.scaleInPlace(inertiaComp);\n this._inertialPanning.addInPlace(directionToZoomLocation);\n camera.inertialRadiusOffset += delta;\n }\n // Sets x y or z of passed in vector to zero if less than Epsilon.\n _zeroIfClose(vec) {\n if (Math.abs(vec.x) < Epsilon) {\n vec.x = 0;\n }\n if (Math.abs(vec.y) < Epsilon) {\n vec.y = 0;\n }\n if (Math.abs(vec.z) < Epsilon) {\n vec.z = 0;\n }\n }\n }\n __decorate$1([\n serialize()\n ], ArcRotateCameraMouseWheelInput.prototype, \"wheelPrecision\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCameraMouseWheelInput.prototype, \"zoomToMouseLocation\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCameraMouseWheelInput.prototype, \"wheelDeltaPercentage\", void 0);\n CameraInputTypes[\"ArcRotateCameraMouseWheelInput\"] = ArcRotateCameraMouseWheelInput;\n\n /**\n * Default Inputs manager for the ArcRotateCamera.\n * It groups all the default supported inputs for ease of use.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\n */\n class ArcRotateCameraInputsManager extends CameraInputsManager {\n /**\n * Instantiates a new ArcRotateCameraInputsManager.\n * @param camera Defines the camera the inputs belong to\n */\n constructor(camera) {\n super(camera);\n }\n /**\n * Add mouse wheel input support to the input manager.\n * @returns the current input manager\n */\n addMouseWheel() {\n this.add(new ArcRotateCameraMouseWheelInput());\n return this;\n }\n /**\n * Add pointers input support to the input manager.\n * @returns the current input manager\n */\n addPointers() {\n this.add(new ArcRotateCameraPointersInput());\n return this;\n }\n /**\n * Add keyboard input support to the input manager.\n * @returns the current input manager\n */\n addKeyboard() {\n this.add(new ArcRotateCameraKeyboardMoveInput());\n return this;\n }\n }\n\n Node.AddNodeConstructor(\"ArcRotateCamera\", (name, scene) => {\n return () => new ArcRotateCamera(name, 0, 0, 1.0, Vector3.Zero(), scene);\n });\n /**\n * This represents an orbital type of camera.\n *\n * This camera always points towards a given target position and can be rotated around that target with the target as the centre of rotation. It can be controlled with cursors and mouse, or with touch events.\n * Think of this camera as one orbiting its target position, or more imaginatively as a spy satellite orbiting the earth. Its position relative to the target (earth) can be set by three parameters, alpha (radians) the longitudinal rotation, beta (radians) the latitudinal rotation and radius the distance from the target position.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#arc-rotate-camera\n */\n class ArcRotateCamera extends TargetCamera {\n /**\n * Defines the target point of the camera.\n * The camera looks towards it from the radius distance.\n */\n get target() {\n return this._target;\n }\n set target(value) {\n this.setTarget(value);\n }\n /**\n * Defines the target mesh of the camera.\n * The camera looks towards it from the radius distance.\n * Please note that setting a target host will disable panning.\n */\n get targetHost() {\n return this._targetHost;\n }\n set targetHost(value) {\n if (value) {\n this.setTarget(value);\n }\n }\n /**\n * Return the current target position of the camera. This value is expressed in local space.\n * @returns the target position\n */\n getTarget() {\n return this.target;\n }\n /**\n * Define the current local position of the camera in the scene\n */\n get position() {\n return this._position;\n }\n set position(newPosition) {\n this.setPosition(newPosition);\n }\n /**\n * The vector the camera should consider as up. (default is Vector3(0, 1, 0) as returned by Vector3.Up())\n * Setting this will copy the given vector to the camera's upVector, and set rotation matrices to and from Y up.\n * DO NOT set the up vector using copyFrom or copyFromFloats, as this bypasses setting the above matrices.\n */\n set upVector(vec) {\n if (!this._upToYMatrix) {\n this._yToUpMatrix = new Matrix();\n this._upToYMatrix = new Matrix();\n this._upVector = Vector3.Zero();\n }\n vec.normalize();\n this._upVector.copyFrom(vec);\n this.setMatUp();\n }\n get upVector() {\n return this._upVector;\n }\n /**\n * Sets the Y-up to camera up-vector rotation matrix, and the up-vector to Y-up rotation matrix.\n */\n setMatUp() {\n // from y-up to custom-up (used in _getViewMatrix)\n Matrix.RotationAlignToRef(Vector3.UpReadOnly, this._upVector, this._yToUpMatrix);\n // from custom-up to y-up (used in rebuildAnglesAndRadius)\n Matrix.RotationAlignToRef(this._upVector, Vector3.UpReadOnly, this._upToYMatrix);\n }\n //-- begin properties for backward compatibility for inputs\n /**\n * Gets or Set the pointer angular sensibility along the X axis or how fast is the camera rotating.\n */\n get angularSensibilityX() {\n const pointers = this.inputs.attached[\"pointers\"];\n if (pointers) {\n return pointers.angularSensibilityX;\n }\n return 0;\n }\n set angularSensibilityX(value) {\n const pointers = this.inputs.attached[\"pointers\"];\n if (pointers) {\n pointers.angularSensibilityX = value;\n }\n }\n /**\n * Gets or Set the pointer angular sensibility along the Y axis or how fast is the camera rotating.\n */\n get angularSensibilityY() {\n const pointers = this.inputs.attached[\"pointers\"];\n if (pointers) {\n return pointers.angularSensibilityY;\n }\n return 0;\n }\n set angularSensibilityY(value) {\n const pointers = this.inputs.attached[\"pointers\"];\n if (pointers) {\n pointers.angularSensibilityY = value;\n }\n }\n /**\n * Gets or Set the pointer pinch precision or how fast is the camera zooming.\n */\n get pinchPrecision() {\n const pointers = this.inputs.attached[\"pointers\"];\n if (pointers) {\n return pointers.pinchPrecision;\n }\n return 0;\n }\n set pinchPrecision(value) {\n const pointers = this.inputs.attached[\"pointers\"];\n if (pointers) {\n pointers.pinchPrecision = value;\n }\n }\n /**\n * Gets or Set the pointer pinch delta percentage or how fast is the camera zooming.\n * It will be used instead of pinchDeltaPrecision if different from 0.\n * It defines the percentage of current camera.radius to use as delta when pinch zoom is used.\n */\n get pinchDeltaPercentage() {\n const pointers = this.inputs.attached[\"pointers\"];\n if (pointers) {\n return pointers.pinchDeltaPercentage;\n }\n return 0;\n }\n set pinchDeltaPercentage(value) {\n const pointers = this.inputs.attached[\"pointers\"];\n if (pointers) {\n pointers.pinchDeltaPercentage = value;\n }\n }\n /**\n * Gets or Set the pointer use natural pinch zoom to override the pinch precision\n * and pinch delta percentage.\n * When useNaturalPinchZoom is true, multi touch zoom will zoom in such\n * that any object in the plane at the camera's target point will scale\n * perfectly with finger motion.\n */\n get useNaturalPinchZoom() {\n const pointers = this.inputs.attached[\"pointers\"];\n if (pointers) {\n return pointers.useNaturalPinchZoom;\n }\n return false;\n }\n set useNaturalPinchZoom(value) {\n const pointers = this.inputs.attached[\"pointers\"];\n if (pointers) {\n pointers.useNaturalPinchZoom = value;\n }\n }\n /**\n * Gets or Set the pointer panning sensibility or how fast is the camera moving.\n */\n get panningSensibility() {\n const pointers = this.inputs.attached[\"pointers\"];\n if (pointers) {\n return pointers.panningSensibility;\n }\n return 0;\n }\n set panningSensibility(value) {\n const pointers = this.inputs.attached[\"pointers\"];\n if (pointers) {\n pointers.panningSensibility = value;\n }\n }\n /**\n * Gets or Set the list of keyboard keys used to control beta angle in a positive direction.\n */\n get keysUp() {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n return keyboard.keysUp;\n }\n return [];\n }\n set keysUp(value) {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n keyboard.keysUp = value;\n }\n }\n /**\n * Gets or Set the list of keyboard keys used to control beta angle in a negative direction.\n */\n get keysDown() {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n return keyboard.keysDown;\n }\n return [];\n }\n set keysDown(value) {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n keyboard.keysDown = value;\n }\n }\n /**\n * Gets or Set the list of keyboard keys used to control alpha angle in a negative direction.\n */\n get keysLeft() {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n return keyboard.keysLeft;\n }\n return [];\n }\n set keysLeft(value) {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n keyboard.keysLeft = value;\n }\n }\n /**\n * Gets or Set the list of keyboard keys used to control alpha angle in a positive direction.\n */\n get keysRight() {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n return keyboard.keysRight;\n }\n return [];\n }\n set keysRight(value) {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n keyboard.keysRight = value;\n }\n }\n /**\n * Gets or Set the mouse wheel precision or how fast is the camera zooming.\n */\n get wheelPrecision() {\n const mousewheel = this.inputs.attached[\"mousewheel\"];\n if (mousewheel) {\n return mousewheel.wheelPrecision;\n }\n return 0;\n }\n set wheelPrecision(value) {\n const mousewheel = this.inputs.attached[\"mousewheel\"];\n if (mousewheel) {\n mousewheel.wheelPrecision = value;\n }\n }\n /**\n * Gets or Set the boolean value that controls whether or not the mouse wheel\n * zooms to the location of the mouse pointer or not. The default is false.\n */\n get zoomToMouseLocation() {\n const mousewheel = this.inputs.attached[\"mousewheel\"];\n if (mousewheel) {\n return mousewheel.zoomToMouseLocation;\n }\n return false;\n }\n set zoomToMouseLocation(value) {\n const mousewheel = this.inputs.attached[\"mousewheel\"];\n if (mousewheel) {\n mousewheel.zoomToMouseLocation = value;\n }\n }\n /**\n * Gets or Set the mouse wheel delta percentage or how fast is the camera zooming.\n * It will be used instead of pinchDeltaPrecision if different from 0.\n * It defines the percentage of current camera.radius to use as delta when pinch zoom is used.\n */\n get wheelDeltaPercentage() {\n const mousewheel = this.inputs.attached[\"mousewheel\"];\n if (mousewheel) {\n return mousewheel.wheelDeltaPercentage;\n }\n return 0;\n }\n set wheelDeltaPercentage(value) {\n const mousewheel = this.inputs.attached[\"mousewheel\"];\n if (mousewheel) {\n mousewheel.wheelDeltaPercentage = value;\n }\n }\n /**\n * Gets the bouncing behavior of the camera if it has been enabled.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#bouncing-behavior\n */\n get bouncingBehavior() {\n return this._bouncingBehavior;\n }\n /**\n * Defines if the bouncing behavior of the camera is enabled on the camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#bouncing-behavior\n */\n get useBouncingBehavior() {\n return this._bouncingBehavior != null;\n }\n set useBouncingBehavior(value) {\n if (value === this.useBouncingBehavior) {\n return;\n }\n if (value) {\n this._bouncingBehavior = new BouncingBehavior();\n this.addBehavior(this._bouncingBehavior);\n }\n else if (this._bouncingBehavior) {\n this.removeBehavior(this._bouncingBehavior);\n this._bouncingBehavior = null;\n }\n }\n /**\n * Gets the framing behavior of the camera if it has been enabled.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#framing-behavior\n */\n get framingBehavior() {\n return this._framingBehavior;\n }\n /**\n * Defines if the framing behavior of the camera is enabled on the camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#framing-behavior\n */\n get useFramingBehavior() {\n return this._framingBehavior != null;\n }\n set useFramingBehavior(value) {\n if (value === this.useFramingBehavior) {\n return;\n }\n if (value) {\n this._framingBehavior = new FramingBehavior();\n this.addBehavior(this._framingBehavior);\n }\n else if (this._framingBehavior) {\n this.removeBehavior(this._framingBehavior);\n this._framingBehavior = null;\n }\n }\n /**\n * Gets the auto rotation behavior of the camera if it has been enabled.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#autorotation-behavior\n */\n get autoRotationBehavior() {\n return this._autoRotationBehavior;\n }\n /**\n * Defines if the auto rotation behavior of the camera is enabled on the camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#autorotation-behavior\n */\n get useAutoRotationBehavior() {\n return this._autoRotationBehavior != null;\n }\n set useAutoRotationBehavior(value) {\n if (value === this.useAutoRotationBehavior) {\n return;\n }\n if (value) {\n this._autoRotationBehavior = new AutoRotationBehavior();\n this.addBehavior(this._autoRotationBehavior);\n }\n else if (this._autoRotationBehavior) {\n this.removeBehavior(this._autoRotationBehavior);\n this._autoRotationBehavior = null;\n }\n }\n /**\n * Instantiates a new ArcRotateCamera in a given scene\n * @param name Defines the name of the camera\n * @param alpha Defines the camera rotation along the longitudinal axis\n * @param beta Defines the camera rotation along the latitudinal axis\n * @param radius Defines the camera distance from its target\n * @param target Defines the camera target\n * @param scene Defines the scene the camera belongs to\n * @param setActiveOnSceneIfNoneActive Defines whether the camera should be marked as active if not other active cameras have been defined\n */\n constructor(name, alpha, beta, radius, target, scene, setActiveOnSceneIfNoneActive = true) {\n super(name, Vector3.Zero(), scene, setActiveOnSceneIfNoneActive);\n /**\n * Current inertia value on the longitudinal axis.\n * The bigger this number the longer it will take for the camera to stop.\n */\n this.inertialAlphaOffset = 0;\n /**\n * Current inertia value on the latitudinal axis.\n * The bigger this number the longer it will take for the camera to stop.\n */\n this.inertialBetaOffset = 0;\n /**\n * Current inertia value on the radius axis.\n * The bigger this number the longer it will take for the camera to stop.\n */\n this.inertialRadiusOffset = 0;\n /**\n * Minimum allowed angle on the longitudinal axis.\n * This can help limiting how the Camera is able to move in the scene.\n */\n this.lowerAlphaLimit = null;\n /**\n * Maximum allowed angle on the longitudinal axis.\n * This can help limiting how the Camera is able to move in the scene.\n */\n this.upperAlphaLimit = null;\n /**\n * Minimum allowed angle on the latitudinal axis.\n * This can help limiting how the Camera is able to move in the scene.\n */\n this.lowerBetaLimit = 0.01;\n /**\n * Maximum allowed angle on the latitudinal axis.\n * This can help limiting how the Camera is able to move in the scene.\n */\n this.upperBetaLimit = Math.PI - 0.01;\n /**\n * Minimum allowed distance of the camera to the target (The camera can not get closer).\n * This can help limiting how the Camera is able to move in the scene.\n */\n this.lowerRadiusLimit = null;\n /**\n * Maximum allowed distance of the camera to the target (The camera can not get further).\n * This can help limiting how the Camera is able to move in the scene.\n */\n this.upperRadiusLimit = null;\n /**\n * Defines the current inertia value used during panning of the camera along the X axis.\n */\n this.inertialPanningX = 0;\n /**\n * Defines the current inertia value used during panning of the camera along the Y axis.\n */\n this.inertialPanningY = 0;\n /**\n * Defines the distance used to consider the camera in pan mode vs pinch/zoom.\n * Basically if your fingers moves away from more than this distance you will be considered\n * in pinch mode.\n */\n this.pinchToPanMaxDistance = 20;\n /**\n * Defines the maximum distance the camera can pan.\n * This could help keeping the camera always in your scene.\n */\n this.panningDistanceLimit = null;\n /**\n * Defines the target of the camera before panning.\n */\n this.panningOriginTarget = Vector3.Zero();\n /**\n * Defines the value of the inertia used during panning.\n * 0 would mean stop inertia and one would mean no deceleration at all.\n */\n this.panningInertia = 0.9;\n //-- end properties for backward compatibility for inputs\n /**\n * Defines how much the radius should be scaled while zooming on a particular mesh (through the zoomOn function)\n */\n this.zoomOnFactor = 1;\n /**\n * Defines a screen offset for the camera position.\n */\n this.targetScreenOffset = Vector2.Zero();\n /**\n * Allows the camera to be completely reversed.\n * If false the camera can not arrive upside down.\n */\n this.allowUpsideDown = true;\n /**\n * Define if double tap/click is used to restore the previously saved state of the camera.\n */\n this.useInputToRestoreState = true;\n /** @internal */\n this._viewMatrix = new Matrix();\n /**\n * Defines the allowed panning axis.\n */\n this.panningAxis = new Vector3(1, 1, 0);\n this._transformedDirection = new Vector3();\n /**\n * Defines if camera will eliminate transform on y axis.\n */\n this.mapPanning = false;\n /**\n * Observable triggered when the mesh target has been changed on the camera.\n */\n this.onMeshTargetChangedObservable = new Observable$1();\n /**\n * Defines whether the camera should check collision with the objects oh the scene.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions#how-can-i-do-this-\n */\n this.checkCollisions = false;\n /**\n * Defines the collision radius of the camera.\n * This simulates a sphere around the camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions#arcrotatecamera\n */\n this.collisionRadius = new Vector3(0.5, 0.5, 0.5);\n this._previousPosition = Vector3.Zero();\n this._collisionVelocity = Vector3.Zero();\n this._newPosition = Vector3.Zero();\n this._computationVector = Vector3.Zero();\n this._onCollisionPositionChange = (collisionId, newPosition, collidedMesh = null) => {\n if (!collidedMesh) {\n this._previousPosition.copyFrom(this._position);\n }\n else {\n this.setPosition(newPosition);\n if (this.onCollide) {\n this.onCollide(collidedMesh);\n }\n }\n // Recompute because of constraints\n const cosa = Math.cos(this.alpha);\n const sina = Math.sin(this.alpha);\n const cosb = Math.cos(this.beta);\n let sinb = Math.sin(this.beta);\n if (sinb === 0) {\n sinb = 0.0001;\n }\n const target = this._getTargetPosition();\n this._computationVector.copyFromFloats(this.radius * cosa * sinb, this.radius * cosb, this.radius * sina * sinb);\n target.addToRef(this._computationVector, this._newPosition);\n this._position.copyFrom(this._newPosition);\n let up = this.upVector;\n if (this.allowUpsideDown && this.beta < 0) {\n up = up.clone();\n up = up.negate();\n }\n this._computeViewMatrix(this._position, target, up);\n this._viewMatrix.addAtIndex(12, this.targetScreenOffset.x);\n this._viewMatrix.addAtIndex(13, this.targetScreenOffset.y);\n this._collisionTriggered = false;\n };\n this._target = Vector3.Zero();\n if (target) {\n this.setTarget(target);\n }\n this.alpha = alpha;\n this.beta = beta;\n this.radius = radius;\n this.getViewMatrix();\n this.inputs = new ArcRotateCameraInputsManager(this);\n this.inputs.addKeyboard().addMouseWheel().addPointers();\n }\n // Cache\n /** @internal */\n _initCache() {\n super._initCache();\n this._cache._target = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\n this._cache.alpha = undefined;\n this._cache.beta = undefined;\n this._cache.radius = undefined;\n this._cache.targetScreenOffset = Vector2.Zero();\n }\n /**\n * @internal\n */\n _updateCache(ignoreParentClass) {\n if (!ignoreParentClass) {\n super._updateCache();\n }\n this._cache._target.copyFrom(this._getTargetPosition());\n this._cache.alpha = this.alpha;\n this._cache.beta = this.beta;\n this._cache.radius = this.radius;\n this._cache.targetScreenOffset.copyFrom(this.targetScreenOffset);\n }\n _getTargetPosition() {\n if (this._targetHost && this._targetHost.getAbsolutePosition) {\n const pos = this._targetHost.getAbsolutePosition();\n if (this._targetBoundingCenter) {\n pos.addToRef(this._targetBoundingCenter, this._target);\n }\n else {\n this._target.copyFrom(pos);\n }\n }\n const lockedTargetPosition = this._getLockedTargetPosition();\n if (lockedTargetPosition) {\n return lockedTargetPosition;\n }\n return this._target;\n }\n /**\n * Stores the current state of the camera (alpha, beta, radius and target)\n * @returns the camera itself\n */\n storeState() {\n this._storedAlpha = this.alpha;\n this._storedBeta = this.beta;\n this._storedRadius = this.radius;\n this._storedTarget = this._getTargetPosition().clone();\n this._storedTargetScreenOffset = this.targetScreenOffset.clone();\n return super.storeState();\n }\n /**\n * @internal\n * Restored camera state. You must call storeState() first\n */\n _restoreStateValues() {\n if (!super._restoreStateValues()) {\n return false;\n }\n this.setTarget(this._storedTarget.clone());\n this.alpha = this._storedAlpha;\n this.beta = this._storedBeta;\n this.radius = this._storedRadius;\n this.targetScreenOffset = this._storedTargetScreenOffset.clone();\n this.inertialAlphaOffset = 0;\n this.inertialBetaOffset = 0;\n this.inertialRadiusOffset = 0;\n this.inertialPanningX = 0;\n this.inertialPanningY = 0;\n return true;\n }\n // Synchronized\n /** @internal */\n _isSynchronizedViewMatrix() {\n if (!super._isSynchronizedViewMatrix()) {\n return false;\n }\n return (this._cache._target.equals(this._getTargetPosition()) &&\n this._cache.alpha === this.alpha &&\n this._cache.beta === this.beta &&\n this._cache.radius === this.radius &&\n this._cache.targetScreenOffset.equals(this.targetScreenOffset));\n }\n /**\n * Attached controls to the current camera.\n * @param ignored defines an ignored parameter kept for backward compatibility.\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\n * @param useCtrlForPanning Defines whether ctrl is used for panning within the controls\n * @param panningMouseButton Defines whether panning is allowed through mouse click button\n */\n attachControl(ignored, noPreventDefault, useCtrlForPanning = true, panningMouseButton = 2) {\n // eslint-disable-next-line prefer-rest-params\n const args = arguments;\n noPreventDefault = Tools.BackCompatCameraNoPreventDefault(args);\n this._useCtrlForPanning = useCtrlForPanning;\n this._panningMouseButton = panningMouseButton;\n // backwards compatibility\n if (typeof args[0] === \"boolean\") {\n if (args.length > 1) {\n this._useCtrlForPanning = args[1];\n }\n if (args.length > 2) {\n this._panningMouseButton = args[2];\n }\n }\n this.inputs.attachElement(noPreventDefault);\n this._reset = () => {\n this.inertialAlphaOffset = 0;\n this.inertialBetaOffset = 0;\n this.inertialRadiusOffset = 0;\n this.inertialPanningX = 0;\n this.inertialPanningY = 0;\n };\n }\n /**\n * Detach the current controls from the specified dom element.\n */\n detachControl() {\n this.inputs.detachElement();\n if (this._reset) {\n this._reset();\n }\n }\n /** @internal */\n _checkInputs() {\n //if (async) collision inspection was triggered, don't update the camera's position - until the collision callback was called.\n if (this._collisionTriggered) {\n return;\n }\n this.inputs.checkInputs();\n // Inertia\n if (this.inertialAlphaOffset !== 0 || this.inertialBetaOffset !== 0 || this.inertialRadiusOffset !== 0) {\n const directionModifier = this.invertRotation ? -1 : 1;\n let inertialAlphaOffset = this.inertialAlphaOffset;\n if (this.beta <= 0) {\n inertialAlphaOffset *= -1;\n }\n if (this.getScene().useRightHandedSystem) {\n inertialAlphaOffset *= -1;\n }\n if (this.parent && this.parent._getWorldMatrixDeterminant() < 0) {\n inertialAlphaOffset *= -1;\n }\n this.alpha += inertialAlphaOffset * directionModifier;\n this.beta += this.inertialBetaOffset * directionModifier;\n this.radius -= this.inertialRadiusOffset;\n this.inertialAlphaOffset *= this.inertia;\n this.inertialBetaOffset *= this.inertia;\n this.inertialRadiusOffset *= this.inertia;\n if (Math.abs(this.inertialAlphaOffset) < Epsilon) {\n this.inertialAlphaOffset = 0;\n }\n if (Math.abs(this.inertialBetaOffset) < Epsilon) {\n this.inertialBetaOffset = 0;\n }\n if (Math.abs(this.inertialRadiusOffset) < this.speed * Epsilon) {\n this.inertialRadiusOffset = 0;\n }\n }\n // Panning inertia\n if (this.inertialPanningX !== 0 || this.inertialPanningY !== 0) {\n const localDirection = new Vector3(this.inertialPanningX, this.inertialPanningY, this.inertialPanningY);\n this._viewMatrix.invertToRef(this._cameraTransformMatrix);\n localDirection.multiplyInPlace(this.panningAxis);\n Vector3.TransformNormalToRef(localDirection, this._cameraTransformMatrix, this._transformedDirection);\n // Eliminate y if mapPanning is enabled\n if (this.mapPanning || !this.panningAxis.y) {\n this._transformedDirection.y = 0;\n }\n if (!this._targetHost) {\n if (this.panningDistanceLimit) {\n this._transformedDirection.addInPlace(this._target);\n const distanceSquared = Vector3.DistanceSquared(this._transformedDirection, this.panningOriginTarget);\n if (distanceSquared <= this.panningDistanceLimit * this.panningDistanceLimit) {\n this._target.copyFrom(this._transformedDirection);\n }\n }\n else {\n this._target.addInPlace(this._transformedDirection);\n }\n }\n this.inertialPanningX *= this.panningInertia;\n this.inertialPanningY *= this.panningInertia;\n if (Math.abs(this.inertialPanningX) < this.speed * Epsilon) {\n this.inertialPanningX = 0;\n }\n if (Math.abs(this.inertialPanningY) < this.speed * Epsilon) {\n this.inertialPanningY = 0;\n }\n }\n // Limits\n this._checkLimits();\n super._checkInputs();\n }\n _checkLimits() {\n if (this.lowerBetaLimit === null || this.lowerBetaLimit === undefined) {\n if (this.allowUpsideDown && this.beta > Math.PI) {\n this.beta = this.beta - 2 * Math.PI;\n }\n }\n else {\n if (this.beta < this.lowerBetaLimit) {\n this.beta = this.lowerBetaLimit;\n }\n }\n if (this.upperBetaLimit === null || this.upperBetaLimit === undefined) {\n if (this.allowUpsideDown && this.beta < -Math.PI) {\n this.beta = this.beta + 2 * Math.PI;\n }\n }\n else {\n if (this.beta > this.upperBetaLimit) {\n this.beta = this.upperBetaLimit;\n }\n }\n if (this.lowerAlphaLimit !== null && this.alpha < this.lowerAlphaLimit) {\n this.alpha = this.lowerAlphaLimit;\n }\n if (this.upperAlphaLimit !== null && this.alpha > this.upperAlphaLimit) {\n this.alpha = this.upperAlphaLimit;\n }\n if (this.lowerRadiusLimit !== null && this.radius < this.lowerRadiusLimit) {\n this.radius = this.lowerRadiusLimit;\n this.inertialRadiusOffset = 0;\n }\n if (this.upperRadiusLimit !== null && this.radius > this.upperRadiusLimit) {\n this.radius = this.upperRadiusLimit;\n this.inertialRadiusOffset = 0;\n }\n }\n /**\n * Rebuilds angles (alpha, beta) and radius from the give position and target\n */\n rebuildAnglesAndRadius() {\n this._position.subtractToRef(this._getTargetPosition(), this._computationVector);\n // need to rotate to Y up equivalent if up vector not Axis.Y\n if (this._upVector.x !== 0 || this._upVector.y !== 1.0 || this._upVector.z !== 0) {\n Vector3.TransformCoordinatesToRef(this._computationVector, this._upToYMatrix, this._computationVector);\n }\n this.radius = this._computationVector.length();\n if (this.radius === 0) {\n this.radius = 0.0001; // Just to avoid division by zero\n }\n // Alpha\n const previousAlpha = this.alpha;\n if (this._computationVector.x === 0 && this._computationVector.z === 0) {\n this.alpha = Math.PI / 2; // avoid division by zero when looking along up axis, and set to acos(0)\n }\n else {\n this.alpha = Math.acos(this._computationVector.x / Math.sqrt(Math.pow(this._computationVector.x, 2) + Math.pow(this._computationVector.z, 2)));\n }\n if (this._computationVector.z < 0) {\n this.alpha = 2 * Math.PI - this.alpha;\n }\n // Calculate the number of revolutions between the new and old alpha values.\n const alphaCorrectionTurns = Math.round((previousAlpha - this.alpha) / (2.0 * Math.PI));\n // Adjust alpha so that its numerical representation is the closest one to the old value.\n this.alpha += alphaCorrectionTurns * 2.0 * Math.PI;\n // Beta\n this.beta = Math.acos(this._computationVector.y / this.radius);\n this._checkLimits();\n }\n /**\n * Use a position to define the current camera related information like alpha, beta and radius\n * @param position Defines the position to set the camera at\n */\n setPosition(position) {\n if (this._position.equals(position)) {\n return;\n }\n this._position.copyFrom(position);\n this.rebuildAnglesAndRadius();\n }\n /**\n * Defines the target the camera should look at.\n * This will automatically adapt alpha beta and radius to fit within the new target.\n * Please note that setting a target as a mesh will disable panning.\n * @param target Defines the new target as a Vector or a mesh\n * @param toBoundingCenter In case of a mesh target, defines whether to target the mesh position or its bounding information center\n * @param allowSamePosition If false, prevents reapplying the new computed position if it is identical to the current one (optim)\n * @param cloneAlphaBetaRadius If true, replicate the current setup (alpha, beta, radius) on the new target\n */\n setTarget(target, toBoundingCenter = false, allowSamePosition = false, cloneAlphaBetaRadius = false) {\n var _a;\n cloneAlphaBetaRadius = (_a = this.overrideCloneAlphaBetaRadius) !== null && _a !== void 0 ? _a : cloneAlphaBetaRadius;\n if (target.getBoundingInfo) {\n if (toBoundingCenter) {\n this._targetBoundingCenter = target.getBoundingInfo().boundingBox.centerWorld.clone();\n }\n else {\n this._targetBoundingCenter = null;\n }\n target.computeWorldMatrix();\n this._targetHost = target;\n this._target = this._getTargetPosition();\n this.onMeshTargetChangedObservable.notifyObservers(this._targetHost);\n }\n else {\n const newTarget = target;\n const currentTarget = this._getTargetPosition();\n if (currentTarget && !allowSamePosition && currentTarget.equals(newTarget)) {\n return;\n }\n this._targetHost = null;\n this._target = newTarget;\n this._targetBoundingCenter = null;\n this.onMeshTargetChangedObservable.notifyObservers(null);\n }\n if (!cloneAlphaBetaRadius) {\n this.rebuildAnglesAndRadius();\n }\n }\n /** @internal */\n _getViewMatrix() {\n // Compute\n const cosa = Math.cos(this.alpha);\n const sina = Math.sin(this.alpha);\n const cosb = Math.cos(this.beta);\n let sinb = Math.sin(this.beta);\n if (sinb === 0) {\n sinb = 0.0001;\n }\n if (this.radius === 0) {\n this.radius = 0.0001; // Just to avoid division by zero\n }\n const target = this._getTargetPosition();\n this._computationVector.copyFromFloats(this.radius * cosa * sinb, this.radius * cosb, this.radius * sina * sinb);\n // Rotate according to up vector\n if (this._upVector.x !== 0 || this._upVector.y !== 1.0 || this._upVector.z !== 0) {\n Vector3.TransformCoordinatesToRef(this._computationVector, this._yToUpMatrix, this._computationVector);\n }\n target.addToRef(this._computationVector, this._newPosition);\n if (this.getScene().collisionsEnabled && this.checkCollisions) {\n const coordinator = this.getScene().collisionCoordinator;\n if (!this._collider) {\n this._collider = coordinator.createCollider();\n }\n this._collider._radius = this.collisionRadius;\n this._newPosition.subtractToRef(this._position, this._collisionVelocity);\n this._collisionTriggered = true;\n coordinator.getNewPosition(this._position, this._collisionVelocity, this._collider, 3, null, this._onCollisionPositionChange, this.uniqueId);\n }\n else {\n this._position.copyFrom(this._newPosition);\n let up = this.upVector;\n if (this.allowUpsideDown && sinb < 0) {\n up = up.negate();\n }\n this._computeViewMatrix(this._position, target, up);\n this._viewMatrix.addAtIndex(12, this.targetScreenOffset.x);\n this._viewMatrix.addAtIndex(13, this.targetScreenOffset.y);\n }\n this._currentTarget = target;\n return this._viewMatrix;\n }\n /**\n * Zooms on a mesh to be at the min distance where we could see it fully in the current viewport.\n * @param meshes Defines the mesh to zoom on\n * @param doNotUpdateMaxZ Defines whether or not maxZ should be updated whilst zooming on the mesh (this can happen if the mesh is big and the maxradius pretty small for instance)\n */\n zoomOn(meshes, doNotUpdateMaxZ = false) {\n meshes = meshes || this.getScene().meshes;\n const minMaxVector = Mesh.MinMax(meshes);\n const distance = Vector3.Distance(minMaxVector.min, minMaxVector.max);\n this.radius = distance * this.zoomOnFactor;\n this.focusOn({ min: minMaxVector.min, max: minMaxVector.max, distance: distance }, doNotUpdateMaxZ);\n }\n /**\n * Focus on a mesh or a bounding box. This adapts the target and maxRadius if necessary but does not update the current radius.\n * The target will be changed but the radius\n * @param meshesOrMinMaxVectorAndDistance Defines the mesh or bounding info to focus on\n * @param doNotUpdateMaxZ Defines whether or not maxZ should be updated whilst zooming on the mesh (this can happen if the mesh is big and the maxradius pretty small for instance)\n */\n focusOn(meshesOrMinMaxVectorAndDistance, doNotUpdateMaxZ = false) {\n let meshesOrMinMaxVector;\n let distance;\n if (meshesOrMinMaxVectorAndDistance.min === undefined) {\n // meshes\n const meshes = meshesOrMinMaxVectorAndDistance || this.getScene().meshes;\n meshesOrMinMaxVector = Mesh.MinMax(meshes);\n distance = Vector3.Distance(meshesOrMinMaxVector.min, meshesOrMinMaxVector.max);\n }\n else {\n //minMaxVector and distance\n const minMaxVectorAndDistance = meshesOrMinMaxVectorAndDistance;\n meshesOrMinMaxVector = minMaxVectorAndDistance;\n distance = minMaxVectorAndDistance.distance;\n }\n this._target = Mesh.Center(meshesOrMinMaxVector);\n if (!doNotUpdateMaxZ) {\n this.maxZ = distance * 2;\n }\n }\n /**\n * @override\n * Override Camera.createRigCamera\n */\n createRigCamera(name, cameraIndex) {\n let alphaShift = 0;\n switch (this.cameraRigMode) {\n case Camera.RIG_MODE_STEREOSCOPIC_ANAGLYPH:\n case Camera.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL:\n case Camera.RIG_MODE_STEREOSCOPIC_OVERUNDER:\n case Camera.RIG_MODE_STEREOSCOPIC_INTERLACED:\n case Camera.RIG_MODE_VR:\n alphaShift = this._cameraRigParams.stereoHalfAngle * (cameraIndex === 0 ? 1 : -1);\n break;\n case Camera.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED:\n alphaShift = this._cameraRigParams.stereoHalfAngle * (cameraIndex === 0 ? -1 : 1);\n break;\n }\n const rigCam = new ArcRotateCamera(name, this.alpha + alphaShift, this.beta, this.radius, this._target, this.getScene());\n rigCam._cameraRigParams = {};\n rigCam.isRigCamera = true;\n rigCam.rigParent = this;\n rigCam.upVector = this.upVector;\n rigCam.mode = this.mode;\n rigCam.orthoLeft = this.orthoLeft;\n rigCam.orthoRight = this.orthoRight;\n rigCam.orthoBottom = this.orthoBottom;\n rigCam.orthoTop = this.orthoTop;\n return rigCam;\n }\n /**\n * @internal\n * @override\n * Override Camera._updateRigCameras\n */\n _updateRigCameras() {\n const camLeft = this._rigCameras[0];\n const camRight = this._rigCameras[1];\n camLeft.beta = camRight.beta = this.beta;\n switch (this.cameraRigMode) {\n case Camera.RIG_MODE_STEREOSCOPIC_ANAGLYPH:\n case Camera.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL:\n case Camera.RIG_MODE_STEREOSCOPIC_OVERUNDER:\n case Camera.RIG_MODE_STEREOSCOPIC_INTERLACED:\n case Camera.RIG_MODE_VR:\n camLeft.alpha = this.alpha - this._cameraRigParams.stereoHalfAngle;\n camRight.alpha = this.alpha + this._cameraRigParams.stereoHalfAngle;\n break;\n case Camera.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED:\n camLeft.alpha = this.alpha + this._cameraRigParams.stereoHalfAngle;\n camRight.alpha = this.alpha - this._cameraRigParams.stereoHalfAngle;\n break;\n }\n super._updateRigCameras();\n }\n /**\n * Destroy the camera and release the current resources hold by it.\n */\n dispose() {\n this.inputs.clear();\n super.dispose();\n }\n /**\n * Gets the current object class name.\n * @returns the class name\n */\n getClassName() {\n return \"ArcRotateCamera\";\n }\n }\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"alpha\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"beta\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"radius\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"overrideCloneAlphaBetaRadius\", void 0);\n __decorate$1([\n serializeAsVector3(\"target\")\n ], ArcRotateCamera.prototype, \"_target\", void 0);\n __decorate$1([\n serializeAsMeshReference(\"targetHost\")\n ], ArcRotateCamera.prototype, \"_targetHost\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"inertialAlphaOffset\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"inertialBetaOffset\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"inertialRadiusOffset\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"lowerAlphaLimit\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"upperAlphaLimit\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"lowerBetaLimit\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"upperBetaLimit\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"lowerRadiusLimit\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"upperRadiusLimit\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"inertialPanningX\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"inertialPanningY\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"pinchToPanMaxDistance\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"panningDistanceLimit\", void 0);\n __decorate$1([\n serializeAsVector3()\n ], ArcRotateCamera.prototype, \"panningOriginTarget\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"panningInertia\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"zoomToMouseLocation\", null);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"zoomOnFactor\", void 0);\n __decorate$1([\n serializeAsVector2()\n ], ArcRotateCamera.prototype, \"targetScreenOffset\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"allowUpsideDown\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCamera.prototype, \"useInputToRestoreState\", void 0);\n\n /**\n * Class representing a ray with position and direction\n */\n class Ray {\n /**\n * Creates a new ray\n * @param origin origin point\n * @param direction direction\n * @param length length of the ray\n */\n constructor(\n /** origin point */\n origin, \n /** direction */\n direction, \n /** length of the ray */\n length = Number.MAX_VALUE) {\n this.origin = origin;\n this.direction = direction;\n this.length = length;\n }\n // Methods\n /**\n * Clone the current ray\n * @returns a new ray\n */\n clone() {\n return new Ray(this.origin.clone(), this.direction.clone(), this.length);\n }\n /**\n * Checks if the ray intersects a box\n * This does not account for the ray length by design to improve perfs.\n * @param minimum bound of the box\n * @param maximum bound of the box\n * @param intersectionTreshold extra extend to be added to the box in all direction\n * @returns if the box was hit\n */\n intersectsBoxMinMax(minimum, maximum, intersectionTreshold = 0) {\n const newMinimum = Ray._TmpVector3[0].copyFromFloats(minimum.x - intersectionTreshold, minimum.y - intersectionTreshold, minimum.z - intersectionTreshold);\n const newMaximum = Ray._TmpVector3[1].copyFromFloats(maximum.x + intersectionTreshold, maximum.y + intersectionTreshold, maximum.z + intersectionTreshold);\n let d = 0.0;\n let maxValue = Number.MAX_VALUE;\n let inv;\n let min;\n let max;\n let temp;\n if (Math.abs(this.direction.x) < 0.0000001) {\n if (this.origin.x < newMinimum.x || this.origin.x > newMaximum.x) {\n return false;\n }\n }\n else {\n inv = 1.0 / this.direction.x;\n min = (newMinimum.x - this.origin.x) * inv;\n max = (newMaximum.x - this.origin.x) * inv;\n if (max === -Infinity) {\n max = Infinity;\n }\n if (min > max) {\n temp = min;\n min = max;\n max = temp;\n }\n d = Math.max(min, d);\n maxValue = Math.min(max, maxValue);\n if (d > maxValue) {\n return false;\n }\n }\n if (Math.abs(this.direction.y) < 0.0000001) {\n if (this.origin.y < newMinimum.y || this.origin.y > newMaximum.y) {\n return false;\n }\n }\n else {\n inv = 1.0 / this.direction.y;\n min = (newMinimum.y - this.origin.y) * inv;\n max = (newMaximum.y - this.origin.y) * inv;\n if (max === -Infinity) {\n max = Infinity;\n }\n if (min > max) {\n temp = min;\n min = max;\n max = temp;\n }\n d = Math.max(min, d);\n maxValue = Math.min(max, maxValue);\n if (d > maxValue) {\n return false;\n }\n }\n if (Math.abs(this.direction.z) < 0.0000001) {\n if (this.origin.z < newMinimum.z || this.origin.z > newMaximum.z) {\n return false;\n }\n }\n else {\n inv = 1.0 / this.direction.z;\n min = (newMinimum.z - this.origin.z) * inv;\n max = (newMaximum.z - this.origin.z) * inv;\n if (max === -Infinity) {\n max = Infinity;\n }\n if (min > max) {\n temp = min;\n min = max;\n max = temp;\n }\n d = Math.max(min, d);\n maxValue = Math.min(max, maxValue);\n if (d > maxValue) {\n return false;\n }\n }\n return true;\n }\n /**\n * Checks if the ray intersects a box\n * This does not account for the ray lenght by design to improve perfs.\n * @param box the bounding box to check\n * @param intersectionTreshold extra extend to be added to the BoundingBox in all direction\n * @returns if the box was hit\n */\n intersectsBox(box, intersectionTreshold = 0) {\n return this.intersectsBoxMinMax(box.minimum, box.maximum, intersectionTreshold);\n }\n /**\n * If the ray hits a sphere\n * @param sphere the bounding sphere to check\n * @param intersectionTreshold extra extend to be added to the BoundingSphere in all direction\n * @returns true if it hits the sphere\n */\n intersectsSphere(sphere, intersectionTreshold = 0) {\n const x = sphere.center.x - this.origin.x;\n const y = sphere.center.y - this.origin.y;\n const z = sphere.center.z - this.origin.z;\n const pyth = x * x + y * y + z * z;\n const radius = sphere.radius + intersectionTreshold;\n const rr = radius * radius;\n if (pyth <= rr) {\n return true;\n }\n const dot = x * this.direction.x + y * this.direction.y + z * this.direction.z;\n if (dot < 0.0) {\n return false;\n }\n const temp = pyth - dot * dot;\n return temp <= rr;\n }\n /**\n * If the ray hits a triange\n * @param vertex0 triangle vertex\n * @param vertex1 triangle vertex\n * @param vertex2 triangle vertex\n * @returns intersection information if hit\n */\n intersectsTriangle(vertex0, vertex1, vertex2) {\n const edge1 = Ray._TmpVector3[0];\n const edge2 = Ray._TmpVector3[1];\n const pvec = Ray._TmpVector3[2];\n const tvec = Ray._TmpVector3[3];\n const qvec = Ray._TmpVector3[4];\n vertex1.subtractToRef(vertex0, edge1);\n vertex2.subtractToRef(vertex0, edge2);\n Vector3.CrossToRef(this.direction, edge2, pvec);\n const det = Vector3.Dot(edge1, pvec);\n if (det === 0) {\n return null;\n }\n const invdet = 1 / det;\n this.origin.subtractToRef(vertex0, tvec);\n const bv = Vector3.Dot(tvec, pvec) * invdet;\n if (bv < 0 || bv > 1.0) {\n return null;\n }\n Vector3.CrossToRef(tvec, edge1, qvec);\n const bw = Vector3.Dot(this.direction, qvec) * invdet;\n if (bw < 0 || bv + bw > 1.0) {\n return null;\n }\n //check if the distance is longer than the predefined length.\n const distance = Vector3.Dot(edge2, qvec) * invdet;\n if (distance > this.length) {\n return null;\n }\n return new IntersectionInfo(1 - bv - bw, bv, distance);\n }\n /**\n * Checks if ray intersects a plane\n * @param plane the plane to check\n * @returns the distance away it was hit\n */\n intersectsPlane(plane) {\n let distance;\n const result1 = Vector3.Dot(plane.normal, this.direction);\n if (Math.abs(result1) < 9.99999997475243e-7) {\n return null;\n }\n else {\n const result2 = Vector3.Dot(plane.normal, this.origin);\n distance = (-plane.d - result2) / result1;\n if (distance < 0.0) {\n if (distance < -9.99999997475243e-7) {\n return null;\n }\n else {\n return 0;\n }\n }\n return distance;\n }\n }\n /**\n * Calculate the intercept of a ray on a given axis\n * @param axis to check 'x' | 'y' | 'z'\n * @param offset from axis interception (i.e. an offset of 1y is intercepted above ground)\n * @returns a vector containing the coordinates where 'axis' is equal to zero (else offset), or null if there is no intercept.\n */\n intersectsAxis(axis, offset = 0) {\n switch (axis) {\n case \"y\": {\n const t = (this.origin.y - offset) / this.direction.y;\n if (t > 0) {\n return null;\n }\n return new Vector3(this.origin.x + this.direction.x * -t, offset, this.origin.z + this.direction.z * -t);\n }\n case \"x\": {\n const t = (this.origin.x - offset) / this.direction.x;\n if (t > 0) {\n return null;\n }\n return new Vector3(offset, this.origin.y + this.direction.y * -t, this.origin.z + this.direction.z * -t);\n }\n case \"z\": {\n const t = (this.origin.z - offset) / this.direction.z;\n if (t > 0) {\n return null;\n }\n return new Vector3(this.origin.x + this.direction.x * -t, this.origin.y + this.direction.y * -t, offset);\n }\n default:\n return null;\n }\n }\n /**\n * Checks if ray intersects a mesh. The ray is defined in WORLD space. A mesh triangle can be picked both from its front and back sides,\n * irrespective of orientation.\n * @param mesh the mesh to check\n * @param fastCheck defines if the first intersection will be used (and not the closest)\n * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected\n * @param onlyBoundingInfo defines a boolean indicating if picking should only happen using bounding info (false by default)\n * @param worldToUse defines the world matrix to use to get the world coordinate of the intersection point\n * @param skipBoundingInfo a boolean indicating if we should skip the bounding info check\n * @returns picking info of the intersection\n */\n intersectsMesh(mesh, fastCheck, trianglePredicate, onlyBoundingInfo = false, worldToUse, skipBoundingInfo = false) {\n const tm = TmpVectors.Matrix[0];\n mesh.getWorldMatrix().invertToRef(tm);\n if (this._tmpRay) {\n Ray.TransformToRef(this, tm, this._tmpRay);\n }\n else {\n this._tmpRay = Ray.Transform(this, tm);\n }\n return mesh.intersects(this._tmpRay, fastCheck, trianglePredicate, onlyBoundingInfo, worldToUse, skipBoundingInfo);\n }\n /**\n * Checks if ray intersects a mesh\n * @param meshes the meshes to check\n * @param fastCheck defines if the first intersection will be used (and not the closest)\n * @param results array to store result in\n * @returns Array of picking infos\n */\n intersectsMeshes(meshes, fastCheck, results) {\n if (results) {\n results.length = 0;\n }\n else {\n results = [];\n }\n for (let i = 0; i < meshes.length; i++) {\n const pickInfo = this.intersectsMesh(meshes[i], fastCheck);\n if (pickInfo.hit) {\n results.push(pickInfo);\n }\n }\n results.sort(this._comparePickingInfo);\n return results;\n }\n _comparePickingInfo(pickingInfoA, pickingInfoB) {\n if (pickingInfoA.distance < pickingInfoB.distance) {\n return -1;\n }\n else if (pickingInfoA.distance > pickingInfoB.distance) {\n return 1;\n }\n else {\n return 0;\n }\n }\n /**\n * Intersection test between the ray and a given segment within a given tolerance (threshold)\n * @param sega the first point of the segment to test the intersection against\n * @param segb the second point of the segment to test the intersection against\n * @param threshold the tolerance margin, if the ray doesn't intersect the segment but is close to the given threshold, the intersection is successful\n * @returns the distance from the ray origin to the intersection point if there's intersection, or -1 if there's no intersection\n */\n intersectionSegment(sega, segb, threshold) {\n const o = this.origin;\n const u = TmpVectors.Vector3[0];\n const rsegb = TmpVectors.Vector3[1];\n const v = TmpVectors.Vector3[2];\n const w = TmpVectors.Vector3[3];\n segb.subtractToRef(sega, u);\n this.direction.scaleToRef(Ray._Rayl, v);\n o.addToRef(v, rsegb);\n sega.subtractToRef(o, w);\n const a = Vector3.Dot(u, u); // always >= 0\n const b = Vector3.Dot(u, v);\n const c = Vector3.Dot(v, v); // always >= 0\n const d = Vector3.Dot(u, w);\n const e = Vector3.Dot(v, w);\n const D = a * c - b * b; // always >= 0\n let sN, sD = D; // sc = sN / sD, default sD = D >= 0\n let tN, tD = D; // tc = tN / tD, default tD = D >= 0\n // compute the line parameters of the two closest points\n if (D < Ray._Smallnum) {\n // the lines are almost parallel\n sN = 0.0; // force using point P0 on segment S1\n sD = 1.0; // to prevent possible division by 0.0 later\n tN = e;\n tD = c;\n }\n else {\n // get the closest points on the infinite lines\n sN = b * e - c * d;\n tN = a * e - b * d;\n if (sN < 0.0) {\n // sc < 0 => the s=0 edge is visible\n sN = 0.0;\n tN = e;\n tD = c;\n }\n else if (sN > sD) {\n // sc > 1 => the s=1 edge is visible\n sN = sD;\n tN = e + b;\n tD = c;\n }\n }\n if (tN < 0.0) {\n // tc < 0 => the t=0 edge is visible\n tN = 0.0;\n // recompute sc for this edge\n if (-d < 0.0) {\n sN = 0.0;\n }\n else if (-d > a) {\n sN = sD;\n }\n else {\n sN = -d;\n sD = a;\n }\n }\n else if (tN > tD) {\n // tc > 1 => the t=1 edge is visible\n tN = tD;\n // recompute sc for this edge\n if (-d + b < 0.0) {\n sN = 0;\n }\n else if (-d + b > a) {\n sN = sD;\n }\n else {\n sN = -d + b;\n sD = a;\n }\n }\n // finally do the division to get sc and tc\n const sc = Math.abs(sN) < Ray._Smallnum ? 0.0 : sN / sD;\n const tc = Math.abs(tN) < Ray._Smallnum ? 0.0 : tN / tD;\n // get the difference of the two closest points\n const qtc = TmpVectors.Vector3[4];\n v.scaleToRef(tc, qtc);\n const qsc = TmpVectors.Vector3[5];\n u.scaleToRef(sc, qsc);\n qsc.addInPlace(w);\n const dP = TmpVectors.Vector3[6];\n qsc.subtractToRef(qtc, dP); // = S1(sc) - S2(tc)\n const isIntersected = tc > 0 && tc <= this.length && dP.lengthSquared() < threshold * threshold; // return intersection result\n if (isIntersected) {\n return qsc.length();\n }\n return -1;\n }\n /**\n * Update the ray from viewport position\n * @param x position\n * @param y y position\n * @param viewportWidth viewport width\n * @param viewportHeight viewport height\n * @param world world matrix\n * @param view view matrix\n * @param projection projection matrix\n * @param enableDistantPicking defines if picking should handle large values for mesh position/scaling (false by default)\n * @returns this ray updated\n */\n update(x, y, viewportWidth, viewportHeight, world, view, projection, enableDistantPicking = false) {\n if (enableDistantPicking) {\n // With world matrices having great values (like 8000000000 on 1 or more scaling or position axis),\n // multiplying view/projection/world and doing invert will result in loss of float precision in the matrix.\n // One way to fix it is to compute the ray with world at identity then transform the ray in object space.\n // This is slower (2 matrix inverts instead of 1) but precision is preserved.\n // This is hidden behind `EnableDistantPicking` flag (default is false)\n if (!Ray._RayDistant) {\n Ray._RayDistant = Ray.Zero();\n }\n Ray._RayDistant.unprojectRayToRef(x, y, viewportWidth, viewportHeight, Matrix.IdentityReadOnly, view, projection);\n const tm = TmpVectors.Matrix[0];\n world.invertToRef(tm);\n Ray.TransformToRef(Ray._RayDistant, tm, this);\n }\n else {\n this.unprojectRayToRef(x, y, viewportWidth, viewportHeight, world, view, projection);\n }\n return this;\n }\n // Statics\n /**\n * Creates a ray with origin and direction of 0,0,0\n * @returns the new ray\n */\n static Zero() {\n return new Ray(Vector3.Zero(), Vector3.Zero());\n }\n /**\n * Creates a new ray from screen space and viewport\n * @param x position\n * @param y y position\n * @param viewportWidth viewport width\n * @param viewportHeight viewport height\n * @param world world matrix\n * @param view view matrix\n * @param projection projection matrix\n * @returns new ray\n */\n static CreateNew(x, y, viewportWidth, viewportHeight, world, view, projection) {\n const result = Ray.Zero();\n return result.update(x, y, viewportWidth, viewportHeight, world, view, projection);\n }\n /**\n * Function will create a new transformed ray starting from origin and ending at the end point. Ray's length will be set, and ray will be\n * transformed to the given world matrix.\n * @param origin The origin point\n * @param end The end point\n * @param world a matrix to transform the ray to. Default is the identity matrix.\n * @returns the new ray\n */\n static CreateNewFromTo(origin, end, world = Matrix.IdentityReadOnly) {\n const direction = end.subtract(origin);\n const length = Math.sqrt(direction.x * direction.x + direction.y * direction.y + direction.z * direction.z);\n direction.normalize();\n return Ray.Transform(new Ray(origin, direction, length), world);\n }\n /**\n * Transforms a ray by a matrix\n * @param ray ray to transform\n * @param matrix matrix to apply\n * @returns the resulting new ray\n */\n static Transform(ray, matrix) {\n const result = new Ray(new Vector3(0, 0, 0), new Vector3(0, 0, 0));\n Ray.TransformToRef(ray, matrix, result);\n return result;\n }\n /**\n * Transforms a ray by a matrix\n * @param ray ray to transform\n * @param matrix matrix to apply\n * @param result ray to store result in\n */\n static TransformToRef(ray, matrix, result) {\n Vector3.TransformCoordinatesToRef(ray.origin, matrix, result.origin);\n Vector3.TransformNormalToRef(ray.direction, matrix, result.direction);\n result.length = ray.length;\n const dir = result.direction;\n const len = dir.length();\n if (!(len === 0 || len === 1)) {\n const num = 1.0 / len;\n dir.x *= num;\n dir.y *= num;\n dir.z *= num;\n result.length *= len;\n }\n }\n /**\n * Unproject a ray from screen space to object space\n * @param sourceX defines the screen space x coordinate to use\n * @param sourceY defines the screen space y coordinate to use\n * @param viewportWidth defines the current width of the viewport\n * @param viewportHeight defines the current height of the viewport\n * @param world defines the world matrix to use (can be set to Identity to go to world space)\n * @param view defines the view matrix to use\n * @param projection defines the projection matrix to use\n */\n unprojectRayToRef(sourceX, sourceY, viewportWidth, viewportHeight, world, view, projection) {\n const matrix = TmpVectors.Matrix[0];\n world.multiplyToRef(view, matrix);\n matrix.multiplyToRef(projection, matrix);\n matrix.invert();\n const engine = EngineStore.LastCreatedEngine;\n const nearScreenSource = TmpVectors.Vector3[0];\n nearScreenSource.x = (sourceX / viewportWidth) * 2 - 1;\n nearScreenSource.y = -((sourceY / viewportHeight) * 2 - 1);\n nearScreenSource.z = (engine === null || engine === void 0 ? void 0 : engine.useReverseDepthBuffer) ? 1 : (engine === null || engine === void 0 ? void 0 : engine.isNDCHalfZRange) ? 0 : -1;\n // far Z need to be close but < to 1 or camera projection matrix with maxZ = 0 will NaN\n const farScreenSource = TmpVectors.Vector3[1].copyFromFloats(nearScreenSource.x, nearScreenSource.y, 1.0 - 1e-8);\n const nearVec3 = TmpVectors.Vector3[2];\n const farVec3 = TmpVectors.Vector3[3];\n Vector3._UnprojectFromInvertedMatrixToRef(nearScreenSource, matrix, nearVec3);\n Vector3._UnprojectFromInvertedMatrixToRef(farScreenSource, matrix, farVec3);\n this.origin.copyFrom(nearVec3);\n farVec3.subtractToRef(nearVec3, this.direction);\n this.direction.normalize();\n }\n }\n Ray._TmpVector3 = ArrayTools.BuildArray(6, Vector3.Zero);\n Ray._RayDistant = Ray.Zero();\n Ray._Smallnum = 0.00000001;\n Ray._Rayl = 10e8;\n Scene.prototype.createPickingRay = function (x, y, world, camera, cameraViewSpace = false) {\n const result = Ray.Zero();\n this.createPickingRayToRef(x, y, world, result, camera, cameraViewSpace);\n return result;\n };\n Scene.prototype.createPickingRayToRef = function (x, y, world, result, camera, cameraViewSpace = false, enableDistantPicking = false) {\n const engine = this.getEngine();\n if (!camera) {\n if (!this.activeCamera) {\n return this;\n }\n camera = this.activeCamera;\n }\n const cameraViewport = camera.viewport;\n const viewport = cameraViewport.toGlobal(engine.getRenderWidth(), engine.getRenderHeight());\n // Moving coordinates to local viewport world\n x = x / engine.getHardwareScalingLevel() - viewport.x;\n y = y / engine.getHardwareScalingLevel() - (engine.getRenderHeight() - viewport.y - viewport.height);\n result.update(x, y, viewport.width, viewport.height, world ? world : Matrix.IdentityReadOnly, cameraViewSpace ? Matrix.IdentityReadOnly : camera.getViewMatrix(), camera.getProjectionMatrix(), enableDistantPicking);\n return this;\n };\n Scene.prototype.createPickingRayInCameraSpace = function (x, y, camera) {\n const result = Ray.Zero();\n this.createPickingRayInCameraSpaceToRef(x, y, result, camera);\n return result;\n };\n Scene.prototype.createPickingRayInCameraSpaceToRef = function (x, y, result, camera) {\n if (!PickingInfo) {\n return this;\n }\n const engine = this.getEngine();\n if (!camera) {\n if (!this.activeCamera) {\n throw new Error(\"Active camera not set\");\n }\n camera = this.activeCamera;\n }\n const cameraViewport = camera.viewport;\n const viewport = cameraViewport.toGlobal(engine.getRenderWidth(), engine.getRenderHeight());\n const identity = Matrix.Identity();\n // Moving coordinates to local viewport world\n x = x / engine.getHardwareScalingLevel() - viewport.x;\n y = y / engine.getHardwareScalingLevel() - (engine.getRenderHeight() - viewport.y - viewport.height);\n result.update(x, y, viewport.width, viewport.height, identity, identity, camera.getProjectionMatrix());\n return this;\n };\n Scene.prototype._internalPickForMesh = function (pickingInfo, rayFunction, mesh, world, fastCheck, onlyBoundingInfo, trianglePredicate, skipBoundingInfo) {\n const ray = rayFunction(world, mesh.enableDistantPicking);\n const result = mesh.intersects(ray, fastCheck, trianglePredicate, onlyBoundingInfo, world, skipBoundingInfo);\n if (!result || !result.hit) {\n return null;\n }\n if (!fastCheck && pickingInfo != null && result.distance >= pickingInfo.distance) {\n return null;\n }\n return result;\n };\n Scene.prototype._internalPick = function (rayFunction, predicate, fastCheck, onlyBoundingInfo, trianglePredicate) {\n let pickingInfo = null;\n const computeWorldMatrixForCamera = !!(this.activeCameras && this.activeCameras.length > 1 && this.cameraToUseForPointers !== this.activeCamera);\n const currentCamera = this.cameraToUseForPointers || this.activeCamera;\n for (let meshIndex = 0; meshIndex < this.meshes.length; meshIndex++) {\n const mesh = this.meshes[meshIndex];\n if (predicate) {\n if (!predicate(mesh)) {\n continue;\n }\n }\n else if (!mesh.isEnabled() || !mesh.isVisible || !mesh.isPickable) {\n continue;\n }\n const forceCompute = computeWorldMatrixForCamera && mesh.isWorldMatrixCameraDependent();\n const world = mesh.computeWorldMatrix(forceCompute, currentCamera);\n if (mesh.hasThinInstances && mesh.thinInstanceEnablePicking) {\n // first check if the ray intersects the whole bounding box/sphere of the mesh\n const result = this._internalPickForMesh(pickingInfo, rayFunction, mesh, world, true, true, trianglePredicate);\n if (result) {\n if (onlyBoundingInfo) {\n // the user only asked for a bounding info check so we can return\n return result;\n }\n const tmpMatrix = TmpVectors.Matrix[1];\n const thinMatrices = mesh.thinInstanceGetWorldMatrices();\n for (let index = 0; index < thinMatrices.length; index++) {\n const thinMatrix = thinMatrices[index];\n thinMatrix.multiplyToRef(world, tmpMatrix);\n const result = this._internalPickForMesh(pickingInfo, rayFunction, mesh, tmpMatrix, fastCheck, onlyBoundingInfo, trianglePredicate, true);\n if (result) {\n pickingInfo = result;\n pickingInfo.thinInstanceIndex = index;\n if (fastCheck) {\n return pickingInfo;\n }\n }\n }\n }\n }\n else {\n const result = this._internalPickForMesh(pickingInfo, rayFunction, mesh, world, fastCheck, onlyBoundingInfo, trianglePredicate);\n if (result) {\n pickingInfo = result;\n if (fastCheck) {\n return pickingInfo;\n }\n }\n }\n }\n return pickingInfo || new PickingInfo();\n };\n Scene.prototype._internalMultiPick = function (rayFunction, predicate, trianglePredicate) {\n if (!PickingInfo) {\n return null;\n }\n const pickingInfos = new Array();\n const computeWorldMatrixForCamera = !!(this.activeCameras && this.activeCameras.length > 1 && this.cameraToUseForPointers !== this.activeCamera);\n const currentCamera = this.cameraToUseForPointers || this.activeCamera;\n for (let meshIndex = 0; meshIndex < this.meshes.length; meshIndex++) {\n const mesh = this.meshes[meshIndex];\n if (predicate) {\n if (!predicate(mesh)) {\n continue;\n }\n }\n else if (!mesh.isEnabled() || !mesh.isVisible || !mesh.isPickable) {\n continue;\n }\n const forceCompute = computeWorldMatrixForCamera && mesh.isWorldMatrixCameraDependent();\n const world = mesh.computeWorldMatrix(forceCompute, currentCamera);\n if (mesh.hasThinInstances && mesh.thinInstanceEnablePicking) {\n const result = this._internalPickForMesh(null, rayFunction, mesh, world, true, true, trianglePredicate);\n if (result) {\n const tmpMatrix = TmpVectors.Matrix[1];\n const thinMatrices = mesh.thinInstanceGetWorldMatrices();\n for (let index = 0; index < thinMatrices.length; index++) {\n const thinMatrix = thinMatrices[index];\n thinMatrix.multiplyToRef(world, tmpMatrix);\n const result = this._internalPickForMesh(null, rayFunction, mesh, tmpMatrix, false, false, trianglePredicate, true);\n if (result) {\n result.thinInstanceIndex = index;\n pickingInfos.push(result);\n }\n }\n }\n }\n else {\n const result = this._internalPickForMesh(null, rayFunction, mesh, world, false, false, trianglePredicate);\n if (result) {\n pickingInfos.push(result);\n }\n }\n }\n return pickingInfos;\n };\n Scene.prototype.pickWithBoundingInfo = function (x, y, predicate, fastCheck, camera) {\n if (!PickingInfo) {\n return null;\n }\n const result = this._internalPick((world) => {\n if (!this._tempPickingRay) {\n this._tempPickingRay = Ray.Zero();\n }\n this.createPickingRayToRef(x, y, world, this._tempPickingRay, camera || null);\n return this._tempPickingRay;\n }, predicate, fastCheck, true);\n if (result) {\n result.ray = this.createPickingRay(x, y, Matrix.Identity(), camera || null);\n }\n return result;\n };\n Object.defineProperty(Scene.prototype, \"_pickingAvailable\", {\n get: () => true,\n enumerable: false,\n configurable: false,\n });\n Scene.prototype.pick = function (x, y, predicate, fastCheck, camera, trianglePredicate, _enableDistantPicking = false) {\n const result = this._internalPick((world, enableDistantPicking) => {\n if (!this._tempPickingRay) {\n this._tempPickingRay = Ray.Zero();\n }\n this.createPickingRayToRef(x, y, world, this._tempPickingRay, camera || null, false, enableDistantPicking);\n return this._tempPickingRay;\n }, predicate, fastCheck, false, trianglePredicate);\n if (result) {\n result.ray = this.createPickingRay(x, y, Matrix.Identity(), camera || null);\n }\n return result;\n };\n Scene.prototype.pickWithRay = function (ray, predicate, fastCheck, trianglePredicate) {\n const result = this._internalPick((world) => {\n if (!this._pickWithRayInverseMatrix) {\n this._pickWithRayInverseMatrix = Matrix.Identity();\n }\n world.invertToRef(this._pickWithRayInverseMatrix);\n if (!this._cachedRayForTransform) {\n this._cachedRayForTransform = Ray.Zero();\n }\n Ray.TransformToRef(ray, this._pickWithRayInverseMatrix, this._cachedRayForTransform);\n return this._cachedRayForTransform;\n }, predicate, fastCheck, false, trianglePredicate);\n if (result) {\n result.ray = ray;\n }\n return result;\n };\n Scene.prototype.multiPick = function (x, y, predicate, camera, trianglePredicate) {\n return this._internalMultiPick((world) => this.createPickingRay(x, y, world, camera || null), predicate, trianglePredicate);\n };\n Scene.prototype.multiPickWithRay = function (ray, predicate, trianglePredicate) {\n return this._internalMultiPick((world) => {\n if (!this._pickWithRayInverseMatrix) {\n this._pickWithRayInverseMatrix = Matrix.Identity();\n }\n world.invertToRef(this._pickWithRayInverseMatrix);\n if (!this._cachedRayForTransform) {\n this._cachedRayForTransform = Ray.Zero();\n }\n Ray.TransformToRef(ray, this._pickWithRayInverseMatrix, this._cachedRayForTransform);\n return this._cachedRayForTransform;\n }, predicate, trianglePredicate);\n };\n Camera.prototype.getForwardRay = function (length = 100, transform, origin) {\n return this.getForwardRayToRef(new Ray(Vector3.Zero(), Vector3.Zero(), length), length, transform, origin);\n };\n Camera.prototype.getForwardRayToRef = function (refRay, length = 100, transform, origin) {\n if (!transform) {\n transform = this.getWorldMatrix();\n }\n refRay.length = length;\n if (!origin) {\n refRay.origin.copyFrom(this.position);\n }\n else {\n refRay.origin.copyFrom(origin);\n }\n TmpVectors.Vector3[2].set(0, 0, this._scene.useRightHandedSystem ? -1 : 1);\n Vector3.TransformNormalToRef(TmpVectors.Vector3[2], transform, TmpVectors.Vector3[3]);\n Vector3.NormalizeToRef(TmpVectors.Vector3[3], refRay.direction);\n return refRay;\n };\n\n /* eslint-disable @typescript-eslint/naming-convention */\n // https://dickyjim.wordpress.com/2013/09/04/spherical-harmonics-for-beginners/\n // http://silviojemma.com/public/papers/lighting/spherical-harmonic-lighting.pdf\n // https://www.ppsloan.org/publications/StupidSH36.pdf\n // http://cseweb.ucsd.edu/~ravir/papers/envmap/envmap.pdf\n // https://www.ppsloan.org/publications/SHJCGT.pdf\n // https://www.ppsloan.org/publications/shdering.pdf\n // https://google.github.io/filament/Filament.md.html#annex/sphericalharmonics\n // https://patapom.com/blog/SHPortal/\n // https://imdoingitwrong.wordpress.com/2011/04/14/spherical-harmonics-wtf/\n // Using real SH basis:\n // m>0 m m\n // y = sqrt(2) * K * P * cos(m*phi) * cos(theta)\n // l l l\n //\n // m<0 m |m|\n // y = sqrt(2) * K * P * sin(m*phi) * cos(theta)\n // l l l\n //\n // m=0 0 0\n // y = K * P * trigono terms\n // l l l\n //\n // m (2l + 1)(l - |m|)!\n // K = sqrt(------------------)\n // l 4pi(l + |m|)!\n //\n // and P by recursion:\n //\n // P00(x) = 1\n // P01(x) = x\n // Pll(x) = (-1^l)(2l - 1)!!(1-x*x)^(1/2)\n // ((2l - 1)x[Pl-1/m]-(l + m - 1)[Pl-2/m])\n // Plm(x) = ---------------------------------------\n // l - m\n // Leaving the trigonometric terms aside we can precompute the constants to :\n const SH3ylmBasisConstants = [\n Math.sqrt(1 / (4 * Math.PI)),\n -Math.sqrt(3 / (4 * Math.PI)),\n Math.sqrt(3 / (4 * Math.PI)),\n -Math.sqrt(3 / (4 * Math.PI)),\n Math.sqrt(15 / (4 * Math.PI)),\n -Math.sqrt(15 / (4 * Math.PI)),\n Math.sqrt(5 / (16 * Math.PI)),\n -Math.sqrt(15 / (4 * Math.PI)),\n Math.sqrt(15 / (16 * Math.PI)), // l22\n ];\n // cm = cos(m * phi)\n // sm = sin(m * phi)\n // {x,y,z} = {cos(phi)sin(theta), sin(phi)sin(theta), cos(theta)}\n // By recursion on using trigo identities:\n const SH3ylmBasisTrigonometricTerms = [\n () => 1,\n (direction) => direction.y,\n (direction) => direction.z,\n (direction) => direction.x,\n (direction) => direction.x * direction.y,\n (direction) => direction.y * direction.z,\n (direction) => 3 * direction.z * direction.z - 1,\n (direction) => direction.x * direction.z,\n (direction) => direction.x * direction.x - direction.y * direction.y, // l22\n ];\n // Wrap the full compute\n const applySH3 = (lm, direction) => {\n return SH3ylmBasisConstants[lm] * SH3ylmBasisTrigonometricTerms[lm](direction);\n };\n // Derived from the integration of the a kernel convolution to SH.\n // Great explanation here: https://patapom.com/blog/SHPortal/#about-distant-radiance-and-irradiance-environments\n const SHCosKernelConvolution = [Math.PI, (2 * Math.PI) / 3, (2 * Math.PI) / 3, (2 * Math.PI) / 3, Math.PI / 4, Math.PI / 4, Math.PI / 4, Math.PI / 4, Math.PI / 4];\n /**\n * Class representing spherical harmonics coefficients to the 3rd degree\n */\n class SphericalHarmonics {\n constructor() {\n /**\n * Defines whether or not the harmonics have been prescaled for rendering.\n */\n this.preScaled = false;\n /**\n * The l0,0 coefficients of the spherical harmonics\n */\n this.l00 = Vector3.Zero();\n /**\n * The l1,-1 coefficients of the spherical harmonics\n */\n this.l1_1 = Vector3.Zero();\n /**\n * The l1,0 coefficients of the spherical harmonics\n */\n this.l10 = Vector3.Zero();\n /**\n * The l1,1 coefficients of the spherical harmonics\n */\n this.l11 = Vector3.Zero();\n /**\n * The l2,-2 coefficients of the spherical harmonics\n */\n this.l2_2 = Vector3.Zero();\n /**\n * The l2,-1 coefficients of the spherical harmonics\n */\n this.l2_1 = Vector3.Zero();\n /**\n * The l2,0 coefficients of the spherical harmonics\n */\n this.l20 = Vector3.Zero();\n /**\n * The l2,1 coefficients of the spherical harmonics\n */\n this.l21 = Vector3.Zero();\n /**\n * The l2,2 coefficients of the spherical harmonics\n */\n this.l22 = Vector3.Zero();\n }\n /**\n * Adds a light to the spherical harmonics\n * @param direction the direction of the light\n * @param color the color of the light\n * @param deltaSolidAngle the delta solid angle of the light\n */\n addLight(direction, color, deltaSolidAngle) {\n TmpVectors.Vector3[0].set(color.r, color.g, color.b);\n const colorVector = TmpVectors.Vector3[0];\n const c = TmpVectors.Vector3[1];\n colorVector.scaleToRef(deltaSolidAngle, c);\n c.scaleToRef(applySH3(0, direction), TmpVectors.Vector3[2]);\n this.l00.addInPlace(TmpVectors.Vector3[2]);\n c.scaleToRef(applySH3(1, direction), TmpVectors.Vector3[2]);\n this.l1_1.addInPlace(TmpVectors.Vector3[2]);\n c.scaleToRef(applySH3(2, direction), TmpVectors.Vector3[2]);\n this.l10.addInPlace(TmpVectors.Vector3[2]);\n c.scaleToRef(applySH3(3, direction), TmpVectors.Vector3[2]);\n this.l11.addInPlace(TmpVectors.Vector3[2]);\n c.scaleToRef(applySH3(4, direction), TmpVectors.Vector3[2]);\n this.l2_2.addInPlace(TmpVectors.Vector3[2]);\n c.scaleToRef(applySH3(5, direction), TmpVectors.Vector3[2]);\n this.l2_1.addInPlace(TmpVectors.Vector3[2]);\n c.scaleToRef(applySH3(6, direction), TmpVectors.Vector3[2]);\n this.l20.addInPlace(TmpVectors.Vector3[2]);\n c.scaleToRef(applySH3(7, direction), TmpVectors.Vector3[2]);\n this.l21.addInPlace(TmpVectors.Vector3[2]);\n c.scaleToRef(applySH3(8, direction), TmpVectors.Vector3[2]);\n this.l22.addInPlace(TmpVectors.Vector3[2]);\n }\n /**\n * Scales the spherical harmonics by the given amount\n * @param scale the amount to scale\n */\n scaleInPlace(scale) {\n this.l00.scaleInPlace(scale);\n this.l1_1.scaleInPlace(scale);\n this.l10.scaleInPlace(scale);\n this.l11.scaleInPlace(scale);\n this.l2_2.scaleInPlace(scale);\n this.l2_1.scaleInPlace(scale);\n this.l20.scaleInPlace(scale);\n this.l21.scaleInPlace(scale);\n this.l22.scaleInPlace(scale);\n }\n /**\n * Convert from incident radiance (Li) to irradiance (E) by applying convolution with the cosine-weighted hemisphere.\n *\n * ```\n * E_lm = A_l * L_lm\n * ```\n *\n * In spherical harmonics this convolution amounts to scaling factors for each frequency band.\n * This corresponds to equation 5 in \"An Efficient Representation for Irradiance Environment Maps\", where\n * the scaling factors are given in equation 9.\n */\n convertIncidentRadianceToIrradiance() {\n // Constant (Band 0)\n this.l00.scaleInPlace(SHCosKernelConvolution[0]);\n // Linear (Band 1)\n this.l1_1.scaleInPlace(SHCosKernelConvolution[1]);\n this.l10.scaleInPlace(SHCosKernelConvolution[2]);\n this.l11.scaleInPlace(SHCosKernelConvolution[3]);\n // Quadratic (Band 2)\n this.l2_2.scaleInPlace(SHCosKernelConvolution[4]);\n this.l2_1.scaleInPlace(SHCosKernelConvolution[5]);\n this.l20.scaleInPlace(SHCosKernelConvolution[6]);\n this.l21.scaleInPlace(SHCosKernelConvolution[7]);\n this.l22.scaleInPlace(SHCosKernelConvolution[8]);\n }\n /**\n * Convert from irradiance to outgoing radiance for Lambertian BDRF, suitable for efficient shader evaluation.\n *\n * ```\n * L = (1/pi) * E * rho\n * ```\n *\n * This is done by an additional scale by 1/pi, so is a fairly trivial operation but important conceptually.\n */\n convertIrradianceToLambertianRadiance() {\n this.scaleInPlace(1.0 / Math.PI);\n // The resultant SH now represents outgoing radiance, so includes the Lambert 1/pi normalisation factor but without albedo (rho) applied\n // (The pixel shader must apply albedo after texture fetches, etc).\n }\n /**\n * Integrates the reconstruction coefficients directly in to the SH preventing further\n * required operations at run time.\n *\n * This is simply done by scaling back the SH with Ylm constants parameter.\n * The trigonometric part being applied by the shader at run time.\n */\n preScaleForRendering() {\n this.preScaled = true;\n this.l00.scaleInPlace(SH3ylmBasisConstants[0]);\n this.l1_1.scaleInPlace(SH3ylmBasisConstants[1]);\n this.l10.scaleInPlace(SH3ylmBasisConstants[2]);\n this.l11.scaleInPlace(SH3ylmBasisConstants[3]);\n this.l2_2.scaleInPlace(SH3ylmBasisConstants[4]);\n this.l2_1.scaleInPlace(SH3ylmBasisConstants[5]);\n this.l20.scaleInPlace(SH3ylmBasisConstants[6]);\n this.l21.scaleInPlace(SH3ylmBasisConstants[7]);\n this.l22.scaleInPlace(SH3ylmBasisConstants[8]);\n }\n /**\n * update the spherical harmonics coefficients from the given array\n * @param data defines the 9x3 coefficients (l00, l1-1, l10, l11, l2-2, l2-1, l20, l21, l22)\n * @returns the spherical harmonics (this)\n */\n updateFromArray(data) {\n Vector3.FromArrayToRef(data[0], 0, this.l00);\n Vector3.FromArrayToRef(data[1], 0, this.l1_1);\n Vector3.FromArrayToRef(data[2], 0, this.l10);\n Vector3.FromArrayToRef(data[3], 0, this.l11);\n Vector3.FromArrayToRef(data[4], 0, this.l2_2);\n Vector3.FromArrayToRef(data[5], 0, this.l2_1);\n Vector3.FromArrayToRef(data[6], 0, this.l20);\n Vector3.FromArrayToRef(data[7], 0, this.l21);\n Vector3.FromArrayToRef(data[8], 0, this.l22);\n return this;\n }\n /**\n * update the spherical harmonics coefficients from the given floats array\n * @param data defines the 9x3 coefficients (l00, l1-1, l10, l11, l2-2, l2-1, l20, l21, l22)\n * @returns the spherical harmonics (this)\n */\n updateFromFloatsArray(data) {\n Vector3.FromFloatsToRef(data[0], data[1], data[2], this.l00);\n Vector3.FromFloatsToRef(data[3], data[4], data[5], this.l1_1);\n Vector3.FromFloatsToRef(data[6], data[7], data[8], this.l10);\n Vector3.FromFloatsToRef(data[9], data[10], data[11], this.l11);\n Vector3.FromFloatsToRef(data[12], data[13], data[14], this.l2_2);\n Vector3.FromFloatsToRef(data[15], data[16], data[17], this.l2_1);\n Vector3.FromFloatsToRef(data[18], data[19], data[20], this.l20);\n Vector3.FromFloatsToRef(data[21], data[22], data[23], this.l21);\n Vector3.FromFloatsToRef(data[24], data[25], data[26], this.l22);\n return this;\n }\n /**\n * Constructs a spherical harmonics from an array.\n * @param data defines the 9x3 coefficients (l00, l1-1, l10, l11, l2-2, l2-1, l20, l21, l22)\n * @returns the spherical harmonics\n */\n static FromArray(data) {\n const sh = new SphericalHarmonics();\n return sh.updateFromArray(data);\n }\n // Keep for references.\n /**\n * Gets the spherical harmonics from polynomial\n * @param polynomial the spherical polynomial\n * @returns the spherical harmonics\n */\n static FromPolynomial(polynomial) {\n const result = new SphericalHarmonics();\n result.l00 = polynomial.xx.scale(0.376127).add(polynomial.yy.scale(0.376127)).add(polynomial.zz.scale(0.376126));\n result.l1_1 = polynomial.y.scale(0.977204);\n result.l10 = polynomial.z.scale(0.977204);\n result.l11 = polynomial.x.scale(0.977204);\n result.l2_2 = polynomial.xy.scale(1.16538);\n result.l2_1 = polynomial.yz.scale(1.16538);\n result.l20 = polynomial.zz.scale(1.34567).subtract(polynomial.xx.scale(0.672834)).subtract(polynomial.yy.scale(0.672834));\n result.l21 = polynomial.zx.scale(1.16538);\n result.l22 = polynomial.xx.scale(1.16538).subtract(polynomial.yy.scale(1.16538));\n result.l1_1.scaleInPlace(-1);\n result.l11.scaleInPlace(-1);\n result.l2_1.scaleInPlace(-1);\n result.l21.scaleInPlace(-1);\n result.scaleInPlace(Math.PI);\n return result;\n }\n }\n /**\n * Class representing spherical polynomial coefficients to the 3rd degree\n */\n class SphericalPolynomial {\n constructor() {\n /**\n * The x coefficients of the spherical polynomial\n */\n this.x = Vector3.Zero();\n /**\n * The y coefficients of the spherical polynomial\n */\n this.y = Vector3.Zero();\n /**\n * The z coefficients of the spherical polynomial\n */\n this.z = Vector3.Zero();\n /**\n * The xx coefficients of the spherical polynomial\n */\n this.xx = Vector3.Zero();\n /**\n * The yy coefficients of the spherical polynomial\n */\n this.yy = Vector3.Zero();\n /**\n * The zz coefficients of the spherical polynomial\n */\n this.zz = Vector3.Zero();\n /**\n * The xy coefficients of the spherical polynomial\n */\n this.xy = Vector3.Zero();\n /**\n * The yz coefficients of the spherical polynomial\n */\n this.yz = Vector3.Zero();\n /**\n * The zx coefficients of the spherical polynomial\n */\n this.zx = Vector3.Zero();\n }\n /**\n * The spherical harmonics used to create the polynomials.\n */\n get preScaledHarmonics() {\n if (!this._harmonics) {\n this._harmonics = SphericalHarmonics.FromPolynomial(this);\n }\n if (!this._harmonics.preScaled) {\n this._harmonics.preScaleForRendering();\n }\n return this._harmonics;\n }\n /**\n * Adds an ambient color to the spherical polynomial\n * @param color the color to add\n */\n addAmbient(color) {\n TmpVectors.Vector3[0].copyFromFloats(color.r, color.g, color.b);\n const colorVector = TmpVectors.Vector3[0];\n this.xx.addInPlace(colorVector);\n this.yy.addInPlace(colorVector);\n this.zz.addInPlace(colorVector);\n }\n /**\n * Scales the spherical polynomial by the given amount\n * @param scale the amount to scale\n */\n scaleInPlace(scale) {\n this.x.scaleInPlace(scale);\n this.y.scaleInPlace(scale);\n this.z.scaleInPlace(scale);\n this.xx.scaleInPlace(scale);\n this.yy.scaleInPlace(scale);\n this.zz.scaleInPlace(scale);\n this.yz.scaleInPlace(scale);\n this.zx.scaleInPlace(scale);\n this.xy.scaleInPlace(scale);\n }\n /**\n * Updates the spherical polynomial from harmonics\n * @param harmonics the spherical harmonics\n * @returns the spherical polynomial\n */\n updateFromHarmonics(harmonics) {\n this._harmonics = harmonics;\n this.x.copyFrom(harmonics.l11);\n this.x.scaleInPlace(1.02333).scaleInPlace(-1);\n this.y.copyFrom(harmonics.l1_1);\n this.y.scaleInPlace(1.02333).scaleInPlace(-1);\n this.z.copyFrom(harmonics.l10);\n this.z.scaleInPlace(1.02333);\n this.xx.copyFrom(harmonics.l00);\n TmpVectors.Vector3[0].copyFrom(harmonics.l20).scaleInPlace(0.247708);\n TmpVectors.Vector3[1].copyFrom(harmonics.l22).scaleInPlace(0.429043);\n this.xx.scaleInPlace(0.886277).subtractInPlace(TmpVectors.Vector3[0]).addInPlace(TmpVectors.Vector3[1]);\n this.yy.copyFrom(harmonics.l00);\n this.yy.scaleInPlace(0.886277).subtractInPlace(TmpVectors.Vector3[0]).subtractInPlace(TmpVectors.Vector3[1]);\n this.zz.copyFrom(harmonics.l00);\n TmpVectors.Vector3[0].copyFrom(harmonics.l20).scaleInPlace(0.495417);\n this.zz.scaleInPlace(0.886277).addInPlace(TmpVectors.Vector3[0]);\n this.yz.copyFrom(harmonics.l2_1);\n this.yz.scaleInPlace(0.858086).scaleInPlace(-1);\n this.zx.copyFrom(harmonics.l21);\n this.zx.scaleInPlace(0.858086).scaleInPlace(-1);\n this.xy.copyFrom(harmonics.l2_2);\n this.xy.scaleInPlace(0.858086);\n this.scaleInPlace(1.0 / Math.PI);\n return this;\n }\n /**\n * Gets the spherical polynomial from harmonics\n * @param harmonics the spherical harmonics\n * @returns the spherical polynomial\n */\n static FromHarmonics(harmonics) {\n const result = new SphericalPolynomial();\n return result.updateFromHarmonics(harmonics);\n }\n /**\n * Constructs a spherical polynomial from an array.\n * @param data defines the 9x3 coefficients (x, y, z, xx, yy, zz, yz, zx, xy)\n * @returns the spherical polynomial\n */\n static FromArray(data) {\n const sp = new SphericalPolynomial();\n Vector3.FromArrayToRef(data[0], 0, sp.x);\n Vector3.FromArrayToRef(data[1], 0, sp.y);\n Vector3.FromArrayToRef(data[2], 0, sp.z);\n Vector3.FromArrayToRef(data[3], 0, sp.xx);\n Vector3.FromArrayToRef(data[4], 0, sp.yy);\n Vector3.FromArrayToRef(data[5], 0, sp.zz);\n Vector3.FromArrayToRef(data[6], 0, sp.yz);\n Vector3.FromArrayToRef(data[7], 0, sp.zx);\n Vector3.FromArrayToRef(data[8], 0, sp.xy);\n return sp;\n }\n }\n\n const ray1 = Ray.Zero();\n const vector1 = Vector3.Zero();\n const vector2 = Vector3.Zero();\n const vector3 = Vector3.Zero();\n function getCameraPositionWithOffset(camera) {\n if (camera instanceof ArcRotateCamera) {\n camera.getForwardRayToRef(ray1);\n Vector3.CrossToRef(ray1.direction, Vector3.Up(), vector1);\n Vector3.CrossToRef(ray1.direction, vector1, vector2);\n vector1.normalize().scaleInPlace(-camera.targetScreenOffset.x);\n vector2.normalize().scaleInPlace(camera.targetScreenOffset.y);\n vector3.copyFrom(camera.globalPosition);\n vector3.addInPlace(vector1).addInPlace(vector2);\n return vector3;\n }\n else {\n return camera.globalPosition;\n }\n }\n function toRadians(degrees) {\n const scalar = Math.PI / 180;\n return degrees * scalar;\n }\n function roundVectorToPlaces(vector, significance = 1, direction) {\n vector.x = roundToPlaces(vector.x, significance, direction);\n vector.y = roundToPlaces(vector.y, significance, direction);\n vector.z = roundToPlaces(vector.z, significance, direction);\n }\n function roundToPlaces(value, significance = 1, direction = 'round') {\n const isRound = Math.round(value / significance) * significance === value;\n if (isRound) {\n return value;\n }\n const n = (value / significance) * (1 + Number.EPSILON);\n let tmp;\n if (direction === 'ceil') {\n tmp = Math.ceil(n);\n }\n else if (direction === 'floor') {\n tmp = Math.floor(n);\n }\n else {\n tmp = Math.round(n);\n }\n return tmp * significance;\n }\n function rotationToVector(toVector, fromVector) {\n const result = new Quaternion();\n if (!fromVector) {\n fromVector = new Vector3(0, 1, 0);\n }\n rotationToVectorToRef(toVector, fromVector, result);\n return result;\n }\n function transformVectorWithQuat(vector, quat) {\n const transform = new Matrix();\n quat.toRotationMatrix(transform);\n const transformVector = Vector3.TransformCoordinates(vector, transform);\n return transformVector;\n }\n function applyTransformToVector(vector, transform) {\n const v = new Vector3(vector.x, vector.y, vector.z);\n const matrix = Matrix.Compose(new Vector3(1, 1, 1), Quaternion.Identity(), v);\n return applyTransformToSpace(matrix, transform);\n }\n function applyTransformToSpace(worldTransform, inSpace) {\n if (!inSpace) {\n inSpace = Matrix.Identity();\n }\n const inSpaceInv = inSpace.clone().invert();\n if (Array.isArray(worldTransform)) {\n worldTransform = Matrix.FromArray(worldTransform);\n }\n const spaceTransform = inSpace.multiply(worldTransform);\n spaceTransform.multiplyToRef(inSpaceInv, spaceTransform);\n const t = new Vector3();\n const q = new Quaternion();\n const s = new Vector3();\n spaceTransform.decompose(s, q, t);\n return {\n position: { x: t.x, y: t.y, z: t.z },\n rotation: exports.KbQuaternion.FromQuat(q),\n scaling: { x: s.x, y: s.y, z: s.z },\n };\n // const fromSpaceRotation = fromSpace.rotationQuaternion ? fromSpace.rotationQuaternion : Quaternion.Identity();\n // const toSpaceRotation = toSpace.rotationQuaternion ? toSpace.rotationQuaternion : Quaternion.Identity();\n // const fromSpaceRotationInverse = fromSpaceRotation.clone().invert();\n // const toSpaceRotationInverse = toSpaceRotation.clone().invert();\n // const q1 = q.clone().multiply(fromSpaceRotationInverse);\n // const q2 = toSpaceRotationInverse.multiply(q1);\n // const q3 = q2.multiply(toSpaceRotation);\n // return q3;\n }\n function computeLocalTransform(node) {\n let rotation;\n if (node.useEuler) {\n rotation = Quaternion.FromEulerAngles(toRadians(node.eulerRotation.x), toRadians(node.eulerRotation.y), toRadians(node.eulerRotation.z));\n }\n else {\n rotation = Quaternion.RotationAxis(node.rotationAxis.toVec3(), node.rotationAngle * (Math.PI / 180));\n }\n const pivotTransform = node.pivot ? Matrix.Translation(-node.pivot.x, -node.pivot.y, -node.pivot.z) : undefined;\n const scalingTransform = Matrix.Scaling(node.scaling, node.scaling, node.scaling);\n const rotationTransform = new Matrix();\n Matrix.FromQuaternionToRef(rotation, rotationTransform);\n const translationTransform = Matrix.Translation(node.position.x, node.position.y, node.position.z);\n const transform = Matrix.Identity();\n if (pivotTransform) {\n transform.multiplyToRef(pivotTransform, transform);\n }\n transform.multiplyToRef(scalingTransform, transform);\n transform.multiplyToRef(rotationTransform, transform);\n transform.multiplyToRef(translationTransform, transform);\n if (pivotTransform) {\n transform.multiplyToRef(pivotTransform.invert(), transform);\n }\n return transform;\n }\n function rotationToVectorToRef(_toVector, _fromVector, result) {\n const toVector = _toVector.clone().normalize();\n const fromVector = _fromVector.clone().normalize();\n if (Vector3.Dot(fromVector, toVector) > 0.9999999) {\n result.set(0, 0, 0, 1); // Identity Quaternion\n }\n if (Vector3.Dot(fromVector, toVector) < -0.9999999) {\n // Rotate 180 degrees about arbitrary axis\n return Quaternion.RotationAxisToRef(Vector3.Right().equals(toVector) ? Vector3.Up().cross(toVector) : Vector3.Right().cross(toVector), Math.PI, result);\n }\n const rotAxis = toVector.cross(fromVector);\n const w = -Math.acos(Vector3.Dot(fromVector, toVector));\n return Quaternion.RotationAxisToRef(rotAxis, w, result);\n }\n function degreeToRadian(vectorDegree, optionalResult) {\n optionalResult = optionalResult || new Vector3();\n if (vectorDegree !== optionalResult) {\n optionalResult.copyFrom(vectorDegree);\n }\n return multiplyScalar(optionalResult, Math.PI / 180);\n }\n function multiplyScalar(vec, scalar) {\n vec.x *= scalar;\n vec.y *= scalar;\n vec.z *= scalar;\n return vec;\n }\n function computeNormal(normals) {\n const vertex1 = normals[0];\n const vertex2 = normals[1];\n const vertex3 = normals[2];\n const p1p2 = vertex1.subtract(vertex2);\n const p3p2 = vertex3.subtract(vertex2);\n const normal = Vector3.Normalize(Vector3.Cross(p1p2, p3p2));\n return normal;\n }\n function applyMatrix4(thisVect, m) {\n const x = thisVect.x, y = thisVect.y, z = thisVect.z;\n const e = m.asArray();\n thisVect.x = e[0] * x + e[4] * y + e[8] * z + e[12];\n thisVect.y = e[1] * x + e[5] * y + e[9] * z + e[13];\n thisVect.z = e[2] * x + e[6] * y + e[10] * z + e[14];\n return thisVect;\n }\n function mathClamp(x, a, b) {\n return x < a ? a : x > b ? b : x;\n }\n function makeRotationAxis(axis, angle) {\n const c = Math.cos(angle);\n const s = Math.sin(angle);\n const t = 1 - c;\n const x = axis.x, y = axis.y, z = axis.z;\n const tx = t * x, ty = t * y;\n const matrix = Matrix.FromValues(tx * x + c, tx * y - s * z, tx * z + s * y, 0, tx * y + s * z, ty * y + c, ty * z - s * x, 0, tx * z - s * y, ty * z + s * x, t * z * z + c, 0, 0, 0, 0, 1);\n return matrix;\n }\n function getRotationBtwVectors(u, v) {\n //http://lolengine.net/blog/2014/02/24/quaternion-from-two-vectors-final\n v.clone().normalize();\n u.clone().normalize();\n const dot = Vector3.Dot(u, v);\n const k = 1; //we are using unit vectors, so just use \"1\" instead of: Math.sqrt(u.length()^2 * v.length()^2);\n let w = k + dot;\n let r;\n if (w < 0.0000001) {\n /* If u and v are exactly opposite, rotate 180 degrees\n * around an arbitrary orthogonal axis. Axis normalisation\n * can happen later, when we normalise the quaternion. */\n w = 0;\n r = Math.abs(u.x) > Math.abs(u.z) ? new Vector3(-u.y, u.x, 0) : new Vector3(0, -u.z, u.y);\n }\n else {\n r = Vector3.Cross(u, v);\n }\n return new Quaternion(r.x, r.y, r.z, w).normalize();\n }\n function addArrays(arrayA, arrayB) {\n if (arrayA.length == 0) {\n return arrayB;\n }\n if (arrayB.length == 0) {\n return arrayA;\n }\n const result = [];\n const n = Math.min(arrayA.length, arrayB.length);\n for (let i = 0; i < n; i++) {\n result[i] = arrayA[i] + arrayB[i];\n }\n return result;\n }\n (function (AxisType) {\n AxisType[AxisType[\"X\"] = 1] = \"X\";\n AxisType[AxisType[\"Y\"] = 2] = \"Y\";\n AxisType[AxisType[\"Z\"] = 3] = \"Z\";\n AxisType[AxisType[\"Custom\"] = 4] = \"Custom\";\n })(exports.AxisType || (exports.AxisType = {}));\n\n var KbVector_1;\n exports.KbVector = KbVector_1 = class KbVector {\n constructor(x, y, z) {\n this.x = x;\n this.y = y;\n this.z = z;\n if (!isFinite(x) || isNaN(x)) {\n this.x = 0;\n console.error('Error: Non numeric or infinite value detected in x-value of a vector, please check snapcode for mistakes.');\n }\n if (!isFinite(y) || isNaN(y)) {\n this.y = 0;\n console.error('Error: Non numeric or infinite value detected in y-value of a vector, please check snapcode for mistakes.');\n }\n if (!isFinite(z) || isNaN(z)) {\n this.z = 0;\n console.error('Error: Non numeric or infinite value detected in z-value of a vector, please check snapcode for mistakes.');\n }\n }\n add(v) {\n return new KbVector_1(this.x + v.x, this.y + v.y, this.z + v.z);\n }\n subtract(v) {\n return new KbVector_1(this.x - v.x, this.y - v.y, this.z - v.z);\n }\n multiply(v) {\n return new KbVector_1(this.x * v.x, this.y * v.y, this.z * v.z);\n }\n divide(v) {\n const newX = v.x !== 0 ? this.x / v.x : 0;\n const newY = v.y !== 0 ? this.y / v.y : 0;\n const newZ = v.z !== 0 ? this.z / v.z : 0;\n return new KbVector_1(newX, newY, newZ);\n }\n crossProduct(v) {\n const x = this.y * v.z - this.z * v.y;\n const y = this.z * v.x - this.x * v.z;\n const z = this.x * v.y - this.y * v.x;\n return new KbVector_1(x, y, z);\n }\n dotProduct(v) {\n return this.x * v.x + this.y * v.y + this.z * v.z;\n }\n length() {\n return Math.sqrt(this.x * this.x + this.y * this.y + this.z * this.z);\n }\n normalize() {\n const l = this.length();\n if (l === 0 || l === 1.0) {\n return this.clone();\n }\n return this.scale(1.0 / l);\n }\n scale(s) {\n return new KbVector_1(this.x * s, this.y * s, this.z * s);\n }\n equal(v) {\n return v.x == this.x && v.y == this.y && v.z == this.z;\n }\n clone() {\n return new KbVector_1(this.x, this.y, this.z);\n }\n assign(v) {\n return new KbVector_1(typeof v.x === 'number' ? v.x : this.x, typeof v.y === 'number' ? v.y : this.y, typeof v.z === 'number' ? v.z : this.z);\n }\n shiftClone(component, amount) {\n if (component === 'x') {\n return new KbVector_1(amount, this.y, this.z);\n }\n else if (component === 'y') {\n return new KbVector_1(this.x, amount, this.z);\n }\n else {\n return new KbVector_1(this.x, this.y, amount);\n }\n }\n toString() {\n return `{ x: ${this.x}, y: ${this.y}, z: ${this.z} }`;\n }\n toObject() {\n return { x: this.x, y: this.y, z: this.z };\n }\n round(significance = 1, direction) {\n return new KbVector_1(roundToPlaces(this.x, significance, direction), roundToPlaces(this.y, significance, direction), roundToPlaces(this.z, significance, direction));\n }\n static Zero() {\n return new KbVector_1(0, 0, 0);\n }\n static Identity() {\n return new KbVector_1(1, 1, 1);\n }\n static From(v) {\n return new KbVector_1(v.x, v.y, v.z);\n }\n };\n exports.KbVector = KbVector_1 = __decorate([\n Kb3dModel({\n token: Token.KbVector,\n trackChanges: false,\n create: args => new KbVector_1(args.x || 0, args.y || 0, args.z || 0),\n }),\n __param(0, Ignore()),\n __param(1, Ignore()),\n __param(2, Ignore())\n ], exports.KbVector);\n\n var KbQuaternion_1;\n exports.KbQuaternion = KbQuaternion_1 = class KbQuaternion {\n constructor(x, y, z, w) {\n this.x = x;\n this.y = y;\n this.z = z;\n this.w = w;\n if (this.w > 1) {\n this.w = 1;\n }\n if (this.w < -1) {\n this.w = -1;\n }\n }\n clone() {\n return new KbQuaternion_1(this.x, this.y, this.z, this.w);\n }\n assign(v) {\n return new KbQuaternion_1(typeof v.x === 'number' ? v.x : this.x, typeof v.y === 'number' ? v.y : this.y, typeof v.z === 'number' ? v.z : this.z, typeof v.w === 'number' ? v.w : this.w);\n }\n toEuler() {\n let x, y, z;\n const qz = this.z;\n const qx = this.x;\n const qy = this.y;\n const qw = this.w;\n const sqw = qw * qw;\n const sqz = qz * qz;\n const sqx = qx * qx;\n const sqy = qy * qy;\n const zAxisY = qy * qz - qx * qw;\n const limit = 0.4999999;\n if (zAxisY < -limit) {\n y = 2 * Math.atan2(qy, qw);\n x = Math.PI / 2;\n z = 0;\n }\n else if (zAxisY > limit) {\n y = 2 * Math.atan2(qy, qw);\n x = -Math.PI / 2;\n z = 0;\n }\n else {\n z = Math.atan2(2.0 * (qx * qy + qz * qw), -sqz - sqx + sqy + sqw);\n x = Math.asin(-2.0 * (qz * qy - qx * qw));\n y = Math.atan2(2.0 * (qz * qx + qy * qw), sqz - sqx - sqy + sqw);\n }\n const scalar = 180 / Math.PI;\n return new exports.KbVector(x * scalar, y * scalar, z * scalar);\n }\n multiply(q) {\n const x = this.x * q.w + this.y * q.z - this.z * q.y + this.w * q.x;\n const y = -this.x * q.z + this.y * q.w + this.z * q.x + this.w * q.y;\n const z = this.x * q.y - this.y * q.x + this.z * q.w + this.w * q.z;\n const w = -this.x * q.x - this.y * q.y - this.z * q.z + this.w * q.w;\n return new KbQuaternion_1(x, y, z, w);\n }\n equal(q) {\n return (!!q &&\n basicallyEqual(q.x, this.x) &&\n basicallyEqual(q.y, this.y) &&\n basicallyEqual(q.z, this.z) &&\n basicallyEqual(q.w, this.w));\n }\n toObject() {\n return { x: this.x, y: this.y, z: this.z, w: this.w };\n }\n getAngle() {\n return this._getRadAngle() * (180 / Math.PI);\n }\n _getRadAngle() {\n return Math.acos(this.w) * 2;\n }\n getAxis() {\n const angle = this._getRadAngle();\n const sin = Math.sin(angle / 2);\n if (sin === 0) {\n return exports.KbVector.Zero();\n }\n else {\n return new exports.KbVector(this.x / sin, this.y / sin, this.z / sin).normalize();\n }\n }\n static FromRotationAxis(axis, angle) {\n // var sin = Math.sin(angle / 2);\n // axis.normalize();\n // result.w = Math.cos(angle / 2);\n // result.x = axis._x * sin;\n // result.y = axis._y * sin;\n // result.z = axis._z * sin;\n const rad = (angle * Math.PI) / 180;\n const sin = Math.sin(rad / 2);\n const nAxis = axis.normalize();\n const result = new KbQuaternion_1(nAxis.x * sin, nAxis.y * sin, nAxis.z * sin, Math.cos(rad / 2));\n return result;\n }\n static FromYawPitchRoll(dV) {\n const v = dV.scale(Math.PI / 180);\n const halfRoll = v.z * 0.5;\n const halfPitch = v.x * 0.5;\n const halfYaw = v.y * 0.5;\n const sinRoll = Math.sin(halfRoll);\n const cosRoll = Math.cos(halfRoll);\n const sinPitch = Math.sin(halfPitch);\n const cosPitch = Math.cos(halfPitch);\n const sinYaw = Math.sin(halfYaw);\n const cosYaw = Math.cos(halfYaw);\n const result = new KbQuaternion_1(cosYaw * sinPitch * cosRoll + sinYaw * cosPitch * sinRoll, sinYaw * cosPitch * cosRoll - cosYaw * sinPitch * sinRoll, cosYaw * cosPitch * sinRoll - sinYaw * sinPitch * cosRoll, cosYaw * cosPitch * cosRoll + sinYaw * sinPitch * sinRoll);\n return result;\n }\n static Identity() {\n return new KbQuaternion_1(0, 0, 0, 1);\n }\n static From(q) {\n return new KbQuaternion_1(q.x, q.y, q.z, q.w);\n }\n };\n exports.KbQuaternion = KbQuaternion_1 = __decorate([\n Kb3dModel({\n token: Token.KbQuaternion,\n trackChanges: false,\n create: args => new KbQuaternion_1(args.x || 0, args.y || 0, args.z || 0, args.w || 1),\n }),\n __param(0, Ignore()),\n __param(1, Ignore()),\n __param(2, Ignore()),\n __param(3, Ignore())\n ], exports.KbQuaternion);\n function basicallyEqual(a, b) {\n return Math.abs(a - b) < 0.0000000001; // To deal with floating point errors\n }\n\n exports.SpaceNode = class SpaceNode extends exports.Kb3dObject {\n constructor() {\n super();\n this._showPivot = false;\n // this._rotationQuaternion = nodeConfig.rotationQuaternion ? KbQuaternion.FromVec3(nodeConfig.rotationQuaternion) : KbQuaternion.Identity();\n // this._rotation = this._rotationQuaternion.toEuler();\n }\n getRotationQuaternion() {\n if (this.useEuler && this.eulerRotation) {\n return exports.KbQuaternion.FromYawPitchRoll(this.eulerRotation);\n }\n else if (this.rotationAxis && this.rotationAngle !== undefined) {\n return exports.KbQuaternion.FromRotationAxis(this.rotationAxis, this.rotationAngle);\n }\n return exports.KbQuaternion.Identity();\n }\n setRotationQuaternion(q) {\n if (this.useEuler) {\n this.eulerRotation = q.toEuler();\n }\n else {\n this.rotationAngle = q.getAngle();\n this.rotationAxis = q.getAxis();\n }\n }\n get features() {\n if (!this._features)\n this._features = [];\n return this._features;\n }\n set features(val) {\n this._features = val;\n }\n get connectors() {\n if (!this._connectors)\n this._connectors = [];\n return this._connectors;\n }\n set connectors(val) {\n this._connectors = val;\n }\n getChildren() {\n return [...this.features, ...this.connectors];\n }\n getChildArrayForType(token) {\n if (token.isFeature()) {\n return this.features;\n }\n else if (token == Token.Connector) {\n return this.connectors;\n }\n else {\n throw Error('unexpected token');\n }\n }\n getAncestors() {\n return super.getAncestors();\n }\n copyPropertiesTo(target) {\n target.name = this.name;\n target.visible = this.visible;\n target.position = this.position;\n target.useEuler = this.useEuler;\n target.eulerRotation = this.eulerRotation;\n target.rotationAngle = this.rotationAngle;\n target.rotationAxis = this.rotationAxis;\n target.scaling = this.scaling;\n target.pivot = this.pivot;\n target.fixed = this.fixed;\n target._primeMover = this._primeMover;\n target._showPivot = this._showPivot;\n target.preTransform = this.preTransform;\n }\n getFeatureByName(name) {\n return [...this.features, ...this.connectors].find(f => f.name.isEqual(name));\n }\n rotate(axis, angle) {\n axis.normalize();\n const q = exports.KbQuaternion.FromRotationAxis(axis, angle);\n this.setRotationQuaternion(this.getRotationQuaternion().multiply(q));\n }\n };\n __decorate([\n Enumerable()\n ], exports.SpaceNode.prototype, \"features\", null);\n __decorate([\n Enumerable()\n ], exports.SpaceNode.prototype, \"connectors\", null);\n exports.SpaceNode = __decorate([\n BaseKb3dModel({\n token: Token.SpaceNode,\n })\n ], exports.SpaceNode);\n\n var MeshNode_1;\n exports.MeshNode = MeshNode_1 = class MeshNode extends exports.SpaceNode {\n constructor() {\n super();\n this._polygonCount = 0;\n this._verticesCount = 0;\n this.materialId = null;\n }\n getSpaceChildren() {\n return new Array();\n }\n copyPropertiesTo(target) {\n super.copyPropertiesTo(target);\n if (target instanceof MeshNode_1) {\n target.flipSurfaceOrientation = this.flipSurfaceOrientation;\n target.receiveShadows = this.receiveShadows;\n target.castShadows = this.castShadows;\n target.checkCollisions = this.checkCollisions;\n target.materialId = this.materialId;\n target.opacity = this.opacity;\n }\n }\n };\n __decorate([\n NodeReference(exports.MaterialNode, '')\n ], exports.MeshNode.prototype, \"materialId\", void 0);\n exports.MeshNode = MeshNode_1 = __decorate([\n BaseKb3dModel({\n token: Token.MeshNode,\n })\n ], exports.MeshNode);\n\n var PrimitiveMeshNode_1;\n exports.PrimitiveMeshNode = PrimitiveMeshNode_1 = class PrimitiveMeshNode extends exports.MeshNode {\n constructor() {\n super();\n }\n copyPropertiesTo(target) {\n super.copyPropertiesTo(target);\n if (target instanceof PrimitiveMeshNode_1) {\n target.uvOverrides = this.uvOverrides;\n target.sideOrientation = this.sideOrientation;\n }\n }\n };\n exports.PrimitiveMeshNode = PrimitiveMeshNode_1 = __decorate([\n BaseKb3dModel({\n token: Token.PrimitiveMeshNode,\n })\n ], exports.PrimitiveMeshNode);\n\n exports.BoxNode = class BoxNode extends exports.PrimitiveMeshNode {\n constructor() {\n super();\n }\n };\n exports.BoxNode = __decorate([\n Kb3dModel({\n token: Token.BoxNode,\n })\n ], exports.BoxNode);\n function isBoxMesh(node) {\n return instanceOf(node, Token.BoxNode);\n }\n\n exports.ConeNode = class ConeNode extends exports.PrimitiveMeshNode {\n constructor() {\n super();\n }\n };\n exports.ConeNode = __decorate([\n Kb3dModel({\n token: Token.ConeNode,\n })\n ], exports.ConeNode);\n function isConeMesh(node) {\n return instanceOf(node, Token.ConeNode);\n }\n\n var CustomMeshNode_1;\n /**\n * A Mesh Node. Must be a descendant of the MeshRoot node.\n * @param {Object} nodeConfig The configuration object of the node.\n * @param {string} nodeConfig.name The name of the node. Required.\n * @param {Object} parent The parent of this node. Optional, but mesh nodes should have parents if they are to be displayed.\n * @property\n */\n exports.CustomMeshNode = CustomMeshNode_1 = class CustomMeshNode extends exports.MeshNode {\n constructor() {\n super();\n /** Loading state of mesh. Used to keep track of when a request is in flight */\n this._loading = null;\n }\n get hasGeometry() {\n return this.geometryId != null && this.geometryId != '';\n }\n get geometryLoaded() {\n return this._geometry != null;\n }\n copyPropertiesTo(target) {\n super.copyPropertiesTo(target);\n if (target instanceof CustomMeshNode_1) {\n target.geometryId = this.geometryId;\n }\n }\n };\n __decorate([\n IgnoreChanges\n ], exports.CustomMeshNode.prototype, \"_loading\", void 0);\n exports.CustomMeshNode = CustomMeshNode_1 = __decorate([\n Kb3dModel({\n token: Token.CustomMeshNode,\n })\n ], exports.CustomMeshNode);\n\n exports.CylinderNode = class CylinderNode extends exports.PrimitiveMeshNode {\n constructor() {\n super();\n }\n };\n exports.CylinderNode = __decorate([\n Kb3dModel({\n token: Token.CylinderNode,\n })\n ], exports.CylinderNode);\n function isCylinderMesh(node) {\n return instanceOf(node, Token.CylinderNode);\n }\n\n exports.DiscNode = class DiscNode extends exports.PrimitiveMeshNode {\n constructor() {\n super();\n }\n };\n exports.DiscNode = __decorate([\n Kb3dModel({\n token: Token.DiscNode,\n })\n ], exports.DiscNode);\n function isDiscMesh(node) {\n return instanceOf(node, Token.DiscNode);\n }\n\n exports.MeshGroupNode = class MeshGroupNode extends exports.SpaceNode {\n constructor() {\n super();\n }\n get meshes() {\n if (!this._meshes)\n this._meshes = [];\n return this._meshes;\n }\n set meshes(val) {\n this._meshes = val;\n }\n getChildren() {\n return super.getChildren().concat(this.meshes);\n }\n getChildArrayForType(token) {\n if (token.isMeshOrGroup() || token.isLight()) {\n return this.meshes;\n }\n else if (token.isConnector()) {\n return this.connectors;\n }\n else {\n throw 'unexpected token';\n }\n }\n getSpaceChildren() {\n return this.meshes;\n }\n };\n __decorate([\n Enumerable()\n ], exports.MeshGroupNode.prototype, \"meshes\", null);\n exports.MeshGroupNode = __decorate([\n Kb3dModel({\n token: Token.MeshGroupNode,\n })\n ], exports.MeshGroupNode);\n\n function isMeshGroupNode(node) {\n return instanceOf(node, Token.MeshGroupNode);\n }\n function isCustomMeshNode(node) {\n return instanceOf(node, Token.CustomMeshNode);\n }\n function isSketchNode(node) {\n return instanceOf(node, Token.SketchNode);\n }\n function isTextNode(node) {\n return instanceOf(node, Token.TextNode);\n }\n function isSketchControlPoint(node) {\n return instanceOf(node, Token.SketchControlPoint);\n }\n function isMeshNode(node) {\n return instanceOfAny(node, ...Token.MeshTokens);\n }\n function isPrimitiveMeshNode(node) {\n return instanceOf(node, Token.PrimitiveMeshNode);\n }\n function isSpaceNode(node) {\n return instanceOf(node, Token.SpaceNode);\n }\n function isPickable(node) {\n if (node instanceof exports.SpaceNode || node instanceof exports.AnnotationNode) {\n return true;\n }\n else {\n return false;\n }\n }\n function isTrackedMeshNode(node) {\n return isMeshNode(node) && isTrackedInstanceOfAny(node, ...Token.MeshAndGroupTokens);\n }\n\n exports.PlaneNode = class PlaneNode extends exports.PrimitiveMeshNode {\n constructor() {\n super();\n }\n };\n exports.PlaneNode = __decorate([\n Kb3dModel({\n token: Token.PlaneNode,\n })\n ], exports.PlaneNode);\n function isPlaneMesh(node) {\n return instanceOf(node, Token.PlaneNode);\n }\n\n var KbColor_1;\n exports.KbColor = KbColor_1 = class KbColor {\n constructor(r, g, b, a = 1) {\n this.r = r;\n this.g = g;\n this.b = b;\n this.a = a;\n }\n clone() {\n return new KbColor_1(this.r, this.g, this.b, this.a);\n }\n assign(c) {\n return new KbColor_1(typeof c.r === 'number' ? c.r : this.r, typeof c.g === 'number' ? c.g : this.g, typeof c.b === 'number' ? c.b : this.b, typeof c.a === 'number' ? c.a : this.a);\n }\n equal(c) {\n return !!c && c.r == this.r && c.g == this.g && c.b == this.b && c.a == this.a;\n }\n cssString() {\n if (this.a < 1) {\n return `rgba(${this.r * 255}, ${this.g * 255}, ${this.b * 255}, ${this.a})`;\n }\n else {\n return `rgb(${this.r * 255}, ${this.g * 255}, ${this.b * 255})`;\n }\n }\n toObject() {\n return { r: this.r, g: this.g, b: this.b, a: this.a };\n }\n toIntObject() {\n return { r: this.r * 255, g: this.g * 255, b: this.b * 255, a: this.a };\n }\n static From(c) {\n return new KbColor_1(c.r, c.g, c.b, c.a);\n }\n static FromIntObject(c) {\n return new KbColor_1(c.r / 255, c.g / 255, c.b / 255, c.a != null ? c.a : 1);\n }\n static White() {\n return new KbColor_1(1, 1, 1, 1);\n }\n static Black() {\n return new KbColor_1(0, 0, 0, 1);\n }\n static Transparent() {\n return new KbColor_1(0, 0, 0, 0);\n }\n static Green() {\n return new KbColor_1(0, 1, 0);\n }\n static Red() {\n return new KbColor_1(1, 0, 0);\n }\n static KbmaxOrange() {\n return new KbColor_1(227 / 255, 131 / 255, 28 / 255);\n }\n static KbmaxBlue() {\n return new KbColor_1(8 / 255, 72 / 255, 96 / 255);\n }\n };\n exports.KbColor = KbColor_1 = __decorate([\n Kb3dModel({\n token: Token.KbColor,\n trackChanges: false,\n create: args => new KbColor_1(args.r || 0, args.g || 0, args.b || 0, args.a),\n }),\n __param(0, Ignore()),\n __param(1, Ignore()),\n __param(2, Ignore()),\n __param(3, Ignore())\n ], exports.KbColor);\n\n var KbGeometry_1;\n const NEW_GEOMETRY = 'NEW_GEOMETRY';\n const MAX_INT16 = 32767;\n const MAX_UINT16 = 65535;\n exports.KbGeometry = KbGeometry_1 = class KbGeometry {\n /**\n * A geometry object. Geometries are immutable, once they are created they should not be modified.\n */\n constructor() {\n this.shared = false;\n }\n clone() {\n const newGeometry = new KbGeometry_1();\n newGeometry.indices = this.indices;\n newGeometry.positions = this.positions;\n newGeometry.normals = this.normals;\n newGeometry.tangents = this.tangents;\n newGeometry.uvs = this.uvs;\n return newGeometry;\n }\n /**\n * Freeze a geometry to enforce the immutability of geometries\n */\n // freeze() {\n // Object.freeze(this);\n // Object.freeze(this.vertexData);\n // }\n // update(vertexData: Partial) {\n // const newVertexData = { ...this.vertexData, ...vertexData };\n // const newGeometry = new KbGeometry(newVertexData, this.id);\n // return newGeometry;\n // }\n extractFromBinary(fileData, meshNode) {\n const dataView = new DataView(fileData);\n let curPos;\n // indices\n const indicesOffset = dataView.getInt32(0, true);\n curPos = 4;\n const indicesDataSize = dataView.getInt32(curPos, true);\n curPos += 4;\n let indices;\n if (indicesDataSize === 16) {\n const indices16 = new Uint16Array(fileData, curPos, indicesOffset / 2);\n indices = Uint32Array.from(indices16);\n curPos += indicesOffset;\n }\n else if (indicesDataSize === 32) {\n indices = new Uint32Array(fileData, curPos, indicesOffset / 4);\n curPos += indicesOffset;\n }\n else {\n throw Error('Unknown indices data size ' + meshNode.name);\n }\n // positions\n const posOffset = dataView.getInt32(curPos, true);\n curPos += 4;\n const posDataSize = dataView.getInt32(curPos, true);\n curPos += 4;\n const posRatio = dataView.getInt32(curPos, true);\n curPos += 4;\n let positions;\n if (posDataSize === 32) {\n positions = new Float32Array(fileData.slice(curPos, curPos + posOffset));\n for (let i = 0; i < positions.length; i++) {\n positions[i] /= posRatio;\n }\n curPos += posOffset;\n }\n else {\n throw Error('Invalid positions in binary ' + meshNode.name);\n }\n // normals\n const norOffset = dataView.getInt32(curPos, true);\n curPos += 4;\n const norDataSize = dataView.getInt32(curPos, true);\n curPos += 4;\n const norRatio = dataView.getInt32(curPos, true);\n curPos += 4;\n let normals;\n if (norDataSize === 16) {\n const normalsInt = new Int16Array(fileData.slice(curPos, curPos + norOffset));\n normals = new Float32Array(normalsInt.length);\n for (let i = 0; i < normalsInt.length; i++) {\n normals[i] = normalsInt[i] / norRatio;\n }\n curPos += norOffset;\n }\n else {\n throw Error('Invalid normals in binary ' + meshNode.name);\n }\n const uvsOffset = dataView.getInt32(curPos, true);\n curPos += 4;\n const uvsDataSize = dataView.getInt32(curPos, true);\n curPos += 4;\n const uvsRatio = dataView.getInt32(curPos, true);\n curPos += 4;\n let uvs;\n if (uvsDataSize === 16) {\n const uvsInt = new Int16Array(fileData.slice(curPos, curPos + uvsOffset));\n uvs = new Float32Array(uvsInt.length);\n for (let i = 0; i < uvsInt.length; i++) {\n uvs[i] = uvsInt[i] / uvsRatio;\n }\n curPos += uvsOffset;\n }\n else {\n throw Error('Invalid uvs in binary ' + meshNode.name);\n }\n // Error reporting\n if (indices.length % 3 > 0) {\n console.warn('Incomplete indices triangles in mesh ' + meshNode.name);\n }\n if (positions.length % 3 > 0) {\n console.warn('Incomplete position vectors in mesh ' + meshNode.name);\n }\n if (normals.length % 3 > 0) {\n console.warn('Incomplete normal vectors in mesh ' + meshNode.name);\n }\n if (uvs.length % 2 > 0) {\n console.warn('Incomplete uv vectors in mesh ' + meshNode.name);\n }\n if (uvs.length > 0 && positions.length !== normals.length) {\n console.warn('Normals and Positions array size mismatch in ' + meshNode.name);\n }\n if (uvs.length > 0 && positions.length / 3 !== uvs.length / 2) {\n console.warn('UV array size mismatch in ' + meshNode.name);\n }\n // End error reporting\n this.positions = positions;\n this.indices = indices;\n if (normals.length > 0) {\n this.normals = normals;\n }\n if (uvs.length > 0) {\n this.uvs = uvs;\n }\n }\n toBinary() {\n if (!this.positions || !this.indices) {\n return;\n }\n const positions = this.positions;\n const indices = this.indices;\n const normals = this.normals || new Float32Array([]);\n const uvs = this.uvs || new Float32Array([]);\n let fileLen = 0;\n let indicesOffset = 8; // indices data markers\n let indicesBits = 0;\n let indicesLength = 0;\n let posOffset = 0;\n let posBits = 0;\n let posLength = 0;\n let normalsOffset = 0;\n let normalsBits = 0;\n let normalsLength = 0;\n let uvsOffset = 0;\n let uvsBits = 0;\n let uvsLength = 0;\n // calculate the total size of the array\n fileLen += indicesOffset;\n if (canFitInUInt16Bit(indices)) {\n // write a 16bit array\n indicesBits = 16;\n indicesLength = indices.length * 2;\n fileLen += indicesLength;\n }\n else {\n // write a 32bit array\n indicesBits = 32;\n indicesLength = indices.length * 4;\n fileLen += indicesLength;\n }\n fileLen += 12; // position data markers\n posOffset = fileLen;\n posBits = 32;\n fileLen += positions.length * 4;\n posLength = positions.length * 4;\n fileLen += 12; // normals data markers\n normalsOffset = fileLen;\n // write the normals 16 bit array\n normalsBits = 16;\n fileLen += normals.length * 2;\n normalsLength = normals.length * 2;\n fileLen += 12; // uv data markers\n uvsOffset = fileLen;\n uvsBits = 16;\n fileLen += uvs.length * 2;\n uvsLength = uvs.length * 2;\n // create the buffer that represent the binary file\n const buffer = new ArrayBuffer(fileLen);\n const bufferView = new DataView(buffer);\n // let's write the data into the bufferView\n // write the indices\n bufferView.setInt32(indicesOffset - 8, indicesLength, true); // NOTICE that by default JS is big endian, but since our servers are x86 we are little endian\n bufferView.setInt32(indicesOffset - 4, indicesBits, true);\n if (indicesBits == 16) {\n for (let i = 0; i < indices.length; i++) {\n bufferView.setUint16(indicesOffset, indices[i], true);\n indicesOffset += 2;\n }\n }\n else if (indicesBits == 32) {\n for (let i = 0; i < indices.length; i++) {\n bufferView.setUint32(indicesOffset, indices[i], true);\n indicesOffset += 4;\n }\n }\n // write the positions\n bufferView.setInt32(posOffset - 12, posLength, true);\n bufferView.setInt32(posOffset - 8, posBits, true);\n bufferView.setInt32(posOffset - 4, 1, true); // ratio\n if (posBits == 32) {\n for (let i = 0; i < positions.length; i++) {\n bufferView.setFloat32(posOffset, positions[i], true);\n posOffset += 4;\n }\n }\n else {\n console.error('Fix me - non-32bit positions not supported');\n }\n // write the normals\n bufferView.setInt32(normalsOffset - 12, normalsLength, true);\n bufferView.setInt32(normalsOffset - 8, normalsBits, true);\n bufferView.setInt32(normalsOffset - 4, MAX_INT16, true);\n if (normalsBits == 16) {\n for (let i = 0; i < normals.length; i++) {\n const n = floatToIntStore(normals[i], MAX_INT16);\n bufferView.setInt16(normalsOffset, n, true);\n normalsOffset += 2;\n }\n }\n else {\n console.error('Fix me - non-16bit normals not supported');\n }\n // write the UVs\n bufferView.setInt32(uvsOffset - 12, uvsLength, true);\n bufferView.setInt32(uvsOffset - 8, uvsBits, true);\n bufferView.setInt32(uvsOffset - 4, MAX_INT16, true);\n if (uvsBits == 16) {\n for (let i = 0; i < uvs.length; i++) {\n const n = floatToIntStore(uvs[i], MAX_INT16);\n bufferView.setInt16(uvsOffset, n, true);\n uvsOffset += 2;\n }\n }\n else {\n console.error('Fix me - non-16bit uvs not supported');\n }\n return buffer;\n }\n };\n exports.KbGeometry = KbGeometry_1 = __decorate([\n Kb3dModel({\n token: Token.KbGeometry,\n trackChanges: false,\n })\n ], exports.KbGeometry);\n // function canFitInInt16Bit(inArray: Float32Array): boolean {\n // for (let i = 0; i < inArray.length; i++) {\n // const val = inArray[i];\n // if (val > 327.68 || val < -327.68) {\n // return false;\n // }\n // }\n // return true;\n // }\n // function canFitInInt8Bit(inArray: Float32Array): boolean {\n // for (let i = 0; i < inArray.length; i++) {\n // const val = inArray[i];\n // if (val > 1.28 || val < -1.28) {\n // return false;\n // }\n // }\n // return true;\n // }\n function canFitInUInt16Bit(inArray) {\n for (let i = 0; i < inArray.length; i++) {\n const val = inArray[i];\n if (val > MAX_UINT16) {\n return false;\n }\n }\n return true;\n }\n function floatToIntStore(float, maxInt, maxVal = 1) {\n // this function convert a value between -327.68 and 327.68 to an int 16. we use the convention to have only 2 decimal numbers\n float = (Math.min(Math.max(-maxVal, float), maxVal) * maxInt) / maxVal;\n const outVal = Math.trunc(float);\n return outVal;\n }\n\n var SketchNode_1;\n exports.SketchNode = SketchNode_1 = class SketchNode extends exports.MeshNode {\n constructor() {\n super();\n this._boundingBoxMin = exports.KbVector.Zero();\n this._boundingBoxMax = exports.KbVector.Zero();\n this._boundingBoxChanges = new Subject();\n this._controlPoints = [];\n this._lineSegmentStartingPoints = [];\n // todo remove\n this._sketchPoints = new Float32Array();\n this.paths = [];\n /** Whether to generate a face for this path */\n this.addFace = false;\n this.closePath = false;\n this.flipPath = false;\n this.tracePath = false;\n }\n getChildren() {\n return super.getChildren().concat(this.paths);\n }\n getChildArrayForType(token) {\n if (token.isSketchPath()) {\n return this.paths;\n }\n else {\n return super.getChildArrayForType(token);\n }\n }\n getBoundingBox() {\n const min = new exports.KbVector(this.boundingXOverride == null ? this._boundingBoxMin.x : -this.boundingXOverride / 2, this.boundingYOverride == null ? this._boundingBoxMin.y : -this.boundingYOverride / 2, this.boundingZOverride == null ? this._boundingBoxMin.z : -this.boundingZOverride / 2);\n const max = new exports.KbVector(this.boundingXOverride == null ? this._boundingBoxMax.x : this.boundingXOverride / 2, this.boundingYOverride == null ? this._boundingBoxMax.y : this.boundingYOverride / 2, this.boundingZOverride == null ? this._boundingBoxMax.z : this.boundingZOverride / 2);\n const centerBB = this._boundingBoxMin.add(this._boundingBoxMax).scale(0.5);\n const center = new exports.KbVector(this.boundingXOverride == null ? centerBB.x : 0, this.boundingYOverride == null ? centerBB.y : 0, this.boundingZOverride == null ? centerBB.z : 0);\n return {\n min,\n max,\n center,\n };\n }\n getSketchLength() {\n let sketchLength = 0;\n this.paths.forEach(path => {\n let lastPosition = undefined;\n path._points.forEach(point => {\n if (!lastPosition) {\n lastPosition = point;\n }\n else {\n const currentPos = point;\n const distance = Math.sqrt(Math.pow(lastPosition.x - currentPos.x, 2) +\n Math.pow(lastPosition.y - currentPos.y, 2) +\n Math.pow(lastPosition.z - currentPos.z, 2));\n sketchLength += distance;\n lastPosition = currentPos;\n }\n });\n });\n return sketchLength;\n }\n };\n __decorate([\n NodeReference(SketchNode_1)\n ], exports.SketchNode.prototype, \"clonePath\", void 0);\n exports.SketchNode = SketchNode_1 = __decorate([\n Kb3dModel({\n token: Token.SketchNode,\n })\n ], exports.SketchNode);\n\n exports.SketchPath = class SketchPath extends exports.Kb3dObject {\n constructor() {\n super(...arguments);\n /** Path control points */\n this.controlPoints = [];\n /** How smooth to make this path when rasterizing to geometry */\n this.tessellation = 16;\n /** For grouping paths as one contiguous shape. Only required for hole-path matching */\n this.pathGroupKey = 'default';\n this._expanded = false;\n this._points = [];\n }\n getChildren() {\n return super.getChildren().concat(this.controlPoints);\n }\n getChildArrayForType(token) {\n if (token.isConnector()) {\n return this.controlPoints;\n }\n else {\n return super.getChildArrayForType(token);\n }\n }\n getAncestors() {\n return super.getAncestors();\n }\n };\n exports.SketchPath = __decorate([\n Kb3dModel({\n token: Token.SketchPath,\n })\n ], exports.SketchPath);\n\n exports.SphereNode = class SphereNode extends exports.PrimitiveMeshNode {\n constructor() {\n super();\n }\n };\n exports.SphereNode = __decorate([\n Kb3dModel({\n token: Token.SphereNode,\n })\n ], exports.SphereNode);\n function isSphereMesh(node) {\n return instanceOf(node, Token.SphereNode);\n }\n\n exports.TextNode = class TextNode extends exports.MeshNode {\n constructor() {\n super(...arguments);\n this.curveRadius = 0;\n this.charSpacing = 1;\n this.text = '';\n }\n };\n exports.TextNode = __decorate([\n Kb3dModel({\n token: Token.TextNode,\n })\n ], exports.TextNode);\n\n exports.TorusNode = class TorusNode extends exports.PrimitiveMeshNode {\n constructor() {\n super();\n }\n };\n exports.TorusNode = __decorate([\n Kb3dModel({\n token: Token.TorusNode,\n })\n ], exports.TorusNode);\n function isTorusMesh(node) {\n return instanceOf(node, Token.TorusNode);\n }\n\n class PatternFeature extends exports.Feature {\n }\n PatternFeature._affects = new Set(ALL_GEOMETRY);\n PatternFeature._affectedBy = new Set(ALL_GEOMETRY);\n __decorate([\n NodeReference(exports.SketchNode)\n ], PatternFeature.prototype, \"meshId\", void 0);\n function isPattern(o) {\n return instanceOfAny(o, Token.LinearPatternFeature, Token.CircularPatternFeature);\n }\n\n exports.CircularPatternFeature = class CircularPatternFeature extends PatternFeature {\n };\n exports.CircularPatternFeature._affects = new Set(ALL_GEOMETRY);\n exports.CircularPatternFeature._affectedBy = new Set(ALL_GEOMETRY);\n exports.CircularPatternFeature = __decorate([\n Kb3dModel({\n token: Token.CircularPatternFeature,\n })\n ], exports.CircularPatternFeature);\n function isCircularPattern(o) {\n return instanceOf(o, Token.CircularPatternFeature);\n }\n\n exports.Connector = class Connector extends exports.BaseConnector {\n getAncestors() {\n return super.getAncestors();\n }\n getSpaceParent() {\n return this._parent;\n }\n };\n exports.Connector = __decorate([\n Kb3dModel({\n token: Token.Connector,\n })\n ], exports.Connector);\n function isConnector(o) {\n return instanceOf(o, Token.Connector);\n }\n\n exports.DeleteFacesFeature = class DeleteFacesFeature extends exports.Feature {\n constructor() {\n super(...arguments);\n this.vertexGroups = [];\n }\n };\n exports.DeleteFacesFeature._dependsOn = new Set([exports.eMeshChangeTypes.VertexIndices]);\n exports.DeleteFacesFeature._affects = new Set([exports.eMeshChangeTypes.VertexIndices, exports.eMeshChangeTypes.Submeshes]);\n exports.DeleteFacesFeature._affectedBy = new Set([exports.eMeshChangeTypes.VertexIndices, exports.eMeshChangeTypes.Submeshes]);\n exports.DeleteFacesFeature = __decorate([\n Kb3dModel({\n token: Token.DeleteFacesFeature,\n })\n ], exports.DeleteFacesFeature);\n\n exports.DisplacementMapFeature = class DisplacementMapFeature extends exports.Feature {\n constructor(_materialService) {\n super();\n this._materialService = _materialService;\n this._initialized = false;\n }\n get textures() {\n if (!this._textures)\n this._textures = [];\n return this._textures;\n }\n set textures(val) {\n this._textures = val;\n }\n getChildren() {\n return [...this.textures];\n }\n getChildArrayForType(token) {\n if (token.isTexture()) {\n return this.textures;\n }\n else {\n throw 'unexpected token';\n }\n }\n };\n exports.DisplacementMapFeature._affects = new Set([exports.eMeshChangeTypes.VertexNormals, exports.eMeshChangeTypes.VertexPositions]);\n exports.DisplacementMapFeature._affectedBy = new Set(ALL_GEOMETRY);\n __decorate([\n Enumerable()\n ], exports.DisplacementMapFeature.prototype, \"textures\", null);\n exports.DisplacementMapFeature = __decorate([\n Kb3dModel({\n token: Token.DisplacementMapFeature,\n }),\n __param(0, Inject(Token.MaterialService))\n ], exports.DisplacementMapFeature);\n\n exports.ExtrudeFeature = class ExtrudeFeature extends exports.Feature {\n constructor() {\n super(...arguments);\n this.vertexGroups = [];\n }\n };\n exports.ExtrudeFeature._affects = new Set(ALL_GEOMETRY);\n exports.ExtrudeFeature._affectedBy = new Set(ALL_GEOMETRY);\n exports.ExtrudeFeature = __decorate([\n Kb3dModel({\n token: Token.ExtrudeFeature,\n })\n ], exports.ExtrudeFeature);\n\n exports.JoinGeometryFeature = class JoinGeometryFeature extends exports.Feature {\n };\n exports.JoinGeometryFeature._affects = new Set(ALL_GEOMETRY);\n exports.JoinGeometryFeature._affectedBy = new Set(ALL_CHANGES);\n __decorate([\n NodeReference(exports.MeshNode)\n ], exports.JoinGeometryFeature.prototype, \"meshId\", void 0);\n exports.JoinGeometryFeature = __decorate([\n Kb3dModel({\n token: Token.JoinGeometryFeature,\n })\n ], exports.JoinGeometryFeature);\n\n exports.LinearPatternFeature = class LinearPatternFeature extends PatternFeature {\n };\n exports.LinearPatternFeature._affects = new Set(ALL_GEOMETRY);\n exports.LinearPatternFeature._affectedBy = new Set(ALL_GEOMETRY);\n exports.LinearPatternFeature = __decorate([\n Kb3dModel({\n token: Token.LinearPatternFeature,\n })\n ], exports.LinearPatternFeature);\n function isLinearPattern(o) {\n return instanceOf(o, Token.LinearPatternFeature);\n }\n\n exports.SliceFeature = class SliceFeature extends exports.Feature {\n };\n exports.SliceFeature._affectedBy = new Set(ALL_GEOMETRY);\n exports.SliceFeature._affects = new Set(ALL_GEOMETRY);\n exports.SliceFeature = __decorate([\n Kb3dModel({\n token: Token.SliceFeature,\n })\n ], exports.SliceFeature);\n\n exports.MirrorFeature = class MirrorFeature extends exports.SliceFeature {\n };\n exports.MirrorFeature._affectedBy = new Set(ALL_GEOMETRY);\n exports.MirrorFeature._affects = new Set(ALL_GEOMETRY);\n exports.MirrorFeature = __decorate([\n Kb3dModel({\n token: Token.MirrorFeature,\n })\n ], exports.MirrorFeature);\n\n exports.MoveVerticesFeature = class MoveVerticesFeature extends exports.Feature {\n constructor() {\n super(...arguments);\n this._previewParticleSize = 0.005;\n }\n get vertexIndices() {\n if (!this._vertexIndices)\n this._vertexIndices = [];\n return this._vertexIndices;\n }\n set vertexIndices(val) {\n this._vertexIndices = val;\n }\n };\n exports.MoveVerticesFeature._dependsOn = new Set([exports.eMeshChangeTypes.VertexIndices]);\n exports.MoveVerticesFeature._affectedBy = new Set([exports.eMeshChangeTypes.VertexPositions]);\n exports.MoveVerticesFeature._affects = new Set([exports.eMeshChangeTypes.VertexPositions]);\n __decorate([\n IgnoreChanges\n ], exports.MoveVerticesFeature.prototype, \"_vertexIndices\", void 0);\n __decorate([\n Enumerable()\n ], exports.MoveVerticesFeature.prototype, \"vertexIndices\", null);\n exports.MoveVerticesFeature = __decorate([\n Kb3dModel({\n token: Token.MoveVerticesFeature,\n })\n ], exports.MoveVerticesFeature);\n\n exports.NormalSmoothingFeature = class NormalSmoothingFeature extends exports.Feature {\n constructor() {\n super(...arguments);\n this.vertexGroups = [];\n this.smooth = true;\n }\n };\n //static _dependsOn = new Set([eMeshChangeTypes.VertexIndices]);\n exports.NormalSmoothingFeature._affects = new Set(ALL_GEOMETRY);\n exports.NormalSmoothingFeature._affectedBy = new Set(ALL_GEOMETRY);\n exports.NormalSmoothingFeature = __decorate([\n Kb3dModel({\n token: Token.NormalSmoothingFeature,\n })\n ], exports.NormalSmoothingFeature);\n\n (function (eContainerItemLayout) {\n eContainerItemLayout[\"horizontal\"] = \"horizontal\";\n eContainerItemLayout[\"vertical\"] = \"vertical\";\n eContainerItemLayout[\"absolute\"] = \"absolute\";\n })(exports.eContainerItemLayout || (exports.eContainerItemLayout = {}));\n (function (eContainerChildAlignment) {\n eContainerChildAlignment[\"inherit\"] = \"inherit\";\n eContainerChildAlignment[\"stretch\"] = \"stretch\";\n eContainerChildAlignment[\"start\"] = \"start\";\n eContainerChildAlignment[\"center\"] = \"center\";\n eContainerChildAlignment[\"end\"] = \"end\";\n })(exports.eContainerChildAlignment || (exports.eContainerChildAlignment = {}));\n (function (eContainerItemAlignment) {\n eContainerItemAlignment[\"stretch\"] = \"stretch\";\n eContainerItemAlignment[\"start\"] = \"start\";\n eContainerItemAlignment[\"center\"] = \"center\";\n eContainerItemAlignment[\"end\"] = \"end\";\n })(exports.eContainerItemAlignment || (exports.eContainerItemAlignment = {}));\n (function (eContainerItemJustify) {\n eContainerItemJustify[\"start\"] = \"start\";\n eContainerItemJustify[\"center\"] = \"center\";\n eContainerItemJustify[\"end\"] = \"end\";\n })(exports.eContainerItemJustify || (exports.eContainerItemJustify = {}));\n (function (eScroll) {\n eScroll[\"none\"] = \"none\";\n eScroll[\"auto\"] = \"auto\";\n eScroll[\"x\"] = \"x\";\n eScroll[\"y\"] = \"y\";\n })(exports.eScroll || (exports.eScroll = {}));\n (function (eRelativeSize) {\n eRelativeSize[\"auto\"] = \"auto\";\n eRelativeSize[\"x1\"] = \"x1\";\n eRelativeSize[\"x2\"] = \"x2\";\n eRelativeSize[\"x3\"] = \"x3\";\n eRelativeSize[\"x4\"] = \"x4\";\n eRelativeSize[\"x5\"] = \"x5\";\n eRelativeSize[\"x6\"] = \"x6\";\n eRelativeSize[\"x7\"] = \"x7\";\n eRelativeSize[\"x8\"] = \"x8\";\n eRelativeSize[\"x9\"] = \"x9\";\n eRelativeSize[\"x10\"] = \"x10\";\n eRelativeSize[\"x11\"] = \"x11\";\n eRelativeSize[\"x12\"] = \"x12\";\n eRelativeSize[\"custom\"] = \"custom\";\n })(exports.eRelativeSize || (exports.eRelativeSize = {}));\n (function (ePadding) {\n ePadding[\"none\"] = \"none\";\n ePadding[\"small\"] = \"small\";\n ePadding[\"medium\"] = \"medium\";\n ePadding[\"large\"] = \"large\";\n })(exports.ePadding || (exports.ePadding = {}));\n (function (eColorShade) {\n eColorShade[\"default\"] = \"default\";\n eColorShade[\"lighter1\"] = \"lighter1\";\n eColorShade[\"lighter2\"] = \"lighter2\";\n eColorShade[\"lighter3\"] = \"lighter3\";\n eColorShade[\"lighter4\"] = \"lighter4\";\n eColorShade[\"darker1\"] = \"darker1\";\n eColorShade[\"darker2\"] = \"darker2\";\n eColorShade[\"darker3\"] = \"darker3\";\n eColorShade[\"darker4\"] = \"darker4\";\n })(exports.eColorShade || (exports.eColorShade = {}));\n (function (eBackgroundColor) {\n eBackgroundColor[\"background\"] = \"background\";\n eBackgroundColor[\"primary\"] = \"primary\";\n eBackgroundColor[\"accent\"] = \"accent\";\n eBackgroundColor[\"warning\"] = \"warning\";\n eBackgroundColor[\"success\"] = \"success\";\n eBackgroundColor[\"input\"] = \"input\";\n eBackgroundColor[\"header\"] = \"header\";\n })(exports.eBackgroundColor || (exports.eBackgroundColor = {}));\n (function (eElevation) {\n eElevation[\"none\"] = \"none\";\n eElevation[\"z1\"] = \"z1\";\n eElevation[\"z2\"] = \"z2\";\n eElevation[\"z3\"] = \"z3\";\n eElevation[\"z4\"] = \"z4\";\n eElevation[\"z5\"] = \"z5\";\n eElevation[\"z6\"] = \"z6\";\n eElevation[\"z7\"] = \"z7\";\n eElevation[\"z8\"] = \"z8\";\n })(exports.eElevation || (exports.eElevation = {}));\n (function (eRuleType) {\n eRuleType[\"value\"] = \"value\";\n eRuleType[\"visibility\"] = \"visibility\";\n eRuleType[\"validation\"] = \"validation\";\n eRuleType[\"pricing\"] = \"pricing\";\n eRuleType[\"message\"] = \"message\";\n eRuleType[\"loaded\"] = \"loaded\";\n eRuleType[\"submit\"] = \"submit\";\n eRuleType[\"pageChanged\"] = \"pageChanged\";\n eRuleType[\"tabChanged\"] = \"tabChanged\";\n eRuleType[\"expanderChanged\"] = \"expanderChanged\";\n eRuleType[\"output\"] = \"output\";\n eRuleType[\"action\"] = \"action\";\n eRuleType[\"function\"] = \"function\";\n eRuleType[\"global\"] = \"global\";\n eRuleType[\"globalEvent\"] = \"globalEvent\";\n eRuleType[\"safeFunction\"] = \"safeFunction\";\n eRuleType[\"naming\"] = \"naming\";\n eRuleType[\"field\"] = \"field\";\n eRuleType[\"button\"] = \"button\";\n eRuleType[\"svgViewer\"] = \"svgViewer\";\n eRuleType[\"dragStart\"] = \"dragStart\";\n eRuleType[\"dragMove\"] = \"dragMove\";\n eRuleType[\"dragEnd\"] = \"dragEnd\";\n eRuleType[\"optionFilter\"] = \"optionFilter\";\n eRuleType[\"nested\"] = \"nested\";\n eRuleType[\"resize\"] = \"resize\";\n eRuleType[\"nestedSetSelected\"] = \"nestedSetSelected\";\n eRuleType[\"keyboard\"] = \"keyboard\";\n eRuleType[\"manufacture\"] = \"manufacture\";\n eRuleType[\"kineticDocumentRules\"] = \"kineticDocumentRules\";\n eRuleType[\"layout\"] = \"layout\";\n eRuleType[\"configuratorAdded\"] = \"configuratorAdded\";\n eRuleType[\"configuratorRemoved\"] = \"configuratorRemoved\";\n eRuleType[\"configuratorCopied\"] = \"configuratorCopied\";\n eRuleType[\"product\"] = \"product\";\n eRuleType[\"workflow\"] = \"workflow\";\n eRuleType[\"beforeBuild\"] = \"beforeBuild\";\n eRuleType[\"globalDefinition\"] = \"globalDefinition\";\n eRuleType[\"quote\"] = \"quote\";\n eRuleType[\"scene\"] = \"scene\";\n eRuleType[\"sceneLoaded\"] = \"sceneLoaded\";\n eRuleType[\"xrEnter\"] = \"xrEnter\";\n eRuleType[\"xrExit\"] = \"xrExit\";\n })(exports.eRuleType || (exports.eRuleType = {}));\n (function (eToolbarPosition) {\n eToolbarPosition[\"hidden\"] = \"hidden\";\n eToolbarPosition[\"topLeft\"] = \"topLeft\";\n eToolbarPosition[\"topRight\"] = \"topRight\";\n eToolbarPosition[\"bottomLeft\"] = \"bottomLeft\";\n eToolbarPosition[\"bottomRight\"] = \"bottomRight\";\n })(exports.eToolbarPosition || (exports.eToolbarPosition = {}));\n (function (eValidationTiming) {\n eValidationTiming[\"always\"] = \"always\";\n eValidationTiming[\"onTouched\"] = \"onTouched\";\n })(exports.eValidationTiming || (exports.eValidationTiming = {}));\n (function (eViewerMode) {\n eViewerMode[\"none\"] = \"none\";\n eViewerMode[\"media\"] = \"media\";\n eViewerMode[\"scene\"] = \"scene\";\n })(exports.eViewerMode || (exports.eViewerMode = {}));\n (function (ePosition) {\n ePosition[\"left\"] = \"left\";\n ePosition[\"right\"] = \"right\";\n })(exports.ePosition || (exports.ePosition = {}));\n (function (eIconStyle) {\n eIconStyle[\"none\"] = \"none\";\n eIconStyle[\"icon\"] = \"icon\";\n eIconStyle[\"media\"] = \"media\";\n })(exports.eIconStyle || (exports.eIconStyle = {}));\n (function (eExportType) {\n eExportType[\"_custom\"] = \"_custom\";\n eExportType[\"pdf\"] = \"pdf\";\n eExportType[\"bmp\"] = \"bmp\";\n eExportType[\"doc\"] = \"doc\";\n eExportType[\"docm\"] = \"docm\";\n eExportType[\"dot\"] = \"dot\";\n eExportType[\"dotm\"] = \"dotm\";\n eExportType[\"dotx\"] = \"dotx\";\n eExportType[\"emf\"] = \"emf\";\n eExportType[\"epub\"] = \"epub\";\n eExportType[\"html\"] = \"html\";\n eExportType[\"jpg\"] = \"jpg\";\n eExportType[\"gif\"] = \"gif\";\n eExportType[\"tiff\"] = \"tiff\";\n eExportType[\"dwf\"] = \"dwf\";\n eExportType[\"edrw\"] = \"edrw\";\n eExportType[\"png\"] = \"png\";\n eExportType[\"odt\"] = \"odt\";\n eExportType[\"rtf\"] = \"rtf\";\n eExportType[\"tif\"] = \"tif\";\n eExportType[\"xaml\"] = \"xaml\";\n eExportType[\"xps\"] = \"xps\";\n eExportType[\"ods\"] = \"ods\";\n eExportType[\"svg\"] = \"svg\";\n eExportType[\"txt\"] = \"txt\";\n eExportType[\"easm\"] = \"easm\";\n eExportType[\"prt\"] = \"prt\";\n eExportType[\"g\"] = \"g\";\n eExportType[\"dwg\"] = \"dwg\";\n eExportType[\"dxf\"] = \"dxf\";\n eExportType[\"sldprt\"] = \"sldprt\";\n eExportType[\"sldasm\"] = \"sldasm\";\n eExportType[\"slddrw\"] = \"slddrw\";\n eExportType[\"sldlfp\"] = \"sldlfp\";\n eExportType[\"x_t\"] = \"x_t\";\n eExportType[\"x_b\"] = \"x_b\";\n eExportType[\"igs\"] = \"igs\";\n eExportType[\"stl\"] = \"stl\";\n eExportType[\"wrl\"] = \"wrl\";\n eExportType[\"u3d\"] = \"u3d\";\n eExportType[\"psd\"] = \"psd\";\n eExportType[\"ai\"] = \"ai\";\n eExportType[\"cgr\"] = \"cgr\";\n eExportType[\"asm\"] = \"asm\";\n eExportType[\"hcg\"] = \"hcg\";\n eExportType[\"hsf\"] = \"hsf\";\n eExportType[\"sat\"] = \"sat\";\n eExportType[\"stp\"] = \"stp\";\n eExportType[\"step\"] = \"step\";\n eExportType[\"vda\"] = \"vda\";\n eExportType[\"neu\"] = \"neu\";\n eExportType[\"iv\"] = \"iv\";\n eExportType[\"slp\"] = \"slp\";\n eExportType[\"eps\"] = \"eps\";\n eExportType[\"catproduct\"] = \"catproduct\";\n eExportType[\"jt\"] = \"jt\";\n eExportType[\"zgl\"] = \"zgl\";\n eExportType[\"xgl\"] = \"xgl\";\n eExportType[\"csv\"] = \"csv\";\n eExportType[\"xml\"] = \"xml\";\n })(exports.eExportType || (exports.eExportType = {}));\n (function (eHotspotSource) {\n eHotspotSource[\"none\"] = \"none\";\n eHotspotSource[\"text\"] = \"text\";\n eHotspotSource[\"media\"] = \"media\";\n eHotspotSource[\"element\"] = \"element\";\n })(exports.eHotspotSource || (exports.eHotspotSource = {}));\n (function (eHotspotAttach) {\n eHotspotAttach[\"topLeft\"] = \"topLeft\";\n eHotspotAttach[\"top\"] = \"top\";\n eHotspotAttach[\"topRight\"] = \"topRight\";\n eHotspotAttach[\"left\"] = \"left\";\n eHotspotAttach[\"center\"] = \"center\";\n eHotspotAttach[\"right\"] = \"right\";\n eHotspotAttach[\"bottomLeft\"] = \"bottomLeft\";\n eHotspotAttach[\"bottom\"] = \"bottom\";\n eHotspotAttach[\"bottomRight\"] = \"bottomRight\";\n })(exports.eHotspotAttach || (exports.eHotspotAttach = {}));\n (function (eHotspotPosition) {\n eHotspotPosition[\"manual\"] = \"manual\";\n eHotspotPosition[\"target\"] = \"target\";\n })(exports.eHotspotPosition || (exports.eHotspotPosition = {}));\n (function (eHotspotShape) {\n eHotspotShape[\"circle\"] = \"circle\";\n eHotspotShape[\"circleWithText\"] = \"circleWithText\";\n eHotspotShape[\"image\"] = \"image\";\n eHotspotShape[\"mesh\"] = \"mesh\";\n eHotspotShape[\"text\"] = \"text\";\n })(exports.eHotspotShape || (exports.eHotspotShape = {}));\n (function (eAnimationEasing) {\n eAnimationEasing[\"linear\"] = \"linear\";\n eAnimationEasing[\"easeInQuad\"] = \"easeInQuad\";\n eAnimationEasing[\"easeOutQuad\"] = \"easeOutQuad\";\n eAnimationEasing[\"easeInOutQuad\"] = \"easeInOutQuad\";\n eAnimationEasing[\"easeInCubic\"] = \"easeInCubic\";\n eAnimationEasing[\"easeOutCubic\"] = \"easeOutCubic\";\n eAnimationEasing[\"easeInOutCubic\"] = \"easeInOutCubic\";\n eAnimationEasing[\"easeInQuart\"] = \"easeInQuart\";\n eAnimationEasing[\"easeOutQuart\"] = \"easeOutQuart\";\n eAnimationEasing[\"easeInOutQuart\"] = \"easeInOutQuart\";\n eAnimationEasing[\"easeInQuint\"] = \"easeInQuint\";\n eAnimationEasing[\"easeOutQuint\"] = \"easeOutQuint\";\n eAnimationEasing[\"easeInOutQuint\"] = \"easeInOutQuint\";\n eAnimationEasing[\"easeInSine\"] = \"easeInSine\";\n eAnimationEasing[\"easeOutSine\"] = \"easeOutSine\";\n eAnimationEasing[\"easeInOutSine\"] = \"easeInOutSine\";\n eAnimationEasing[\"easeInExpo\"] = \"easeInExpo\";\n eAnimationEasing[\"easeOutExpo\"] = \"easeOutExpo\";\n eAnimationEasing[\"easeInOutExpo\"] = \"easeInOutExpo\";\n eAnimationEasing[\"easeInCirc\"] = \"easeInCirc\";\n eAnimationEasing[\"easeOutCirc\"] = \"easeOutCirc\";\n eAnimationEasing[\"easeInOutCirc\"] = \"easeInOutCirc\";\n eAnimationEasing[\"easeInElastic\"] = \"easeInElastic\";\n eAnimationEasing[\"easeOutElastic\"] = \"easeOutElastic\";\n eAnimationEasing[\"easeInOutElastic\"] = \"easeInOutElastic\";\n eAnimationEasing[\"easeInBack\"] = \"easeInBack\";\n eAnimationEasing[\"easeOutBack\"] = \"easeOutBack\";\n eAnimationEasing[\"easeInOutBack\"] = \"easeInOutBack\";\n eAnimationEasing[\"easeInBounce\"] = \"easeInBounce\";\n eAnimationEasing[\"easeOutBounce\"] = \"easeOutBounce\";\n eAnimationEasing[\"easeInOutBounce\"] = \"easeInOutBounce\";\n })(exports.eAnimationEasing || (exports.eAnimationEasing = {}));\n (function (eQueryMode) {\n eQueryMode[\"none\"] = \"none\";\n eQueryMode[\"table\"] = \"table\";\n eQueryMode[\"database\"] = \"database\";\n eQueryMode[\"array\"] = \"array\";\n eQueryMode[\"optionFilter\"] = \"optionFilter\";\n eQueryMode[\"parentOptionFilter\"] = \"parentOptionFilter\";\n })(exports.eQueryMode || (exports.eQueryMode = {}));\n (function (eOptionFilterSource) {\n eOptionFilterSource[\"table\"] = \"table\";\n eOptionFilterSource[\"database\"] = \"database\";\n eOptionFilterSource[\"query\"] = \"query\";\n eOptionFilterSource[\"safeFunction\"] = \"safeFunction\";\n eOptionFilterSource[\"optionFilter\"] = \"optionFilter\";\n })(exports.eOptionFilterSource || (exports.eOptionFilterSource = {}));\n (function (eOutputType) {\n eOutputType[\"none\"] = \"none\";\n eOutputType[\"word\"] = \"word\";\n eOutputType[\"excel\"] = \"excel\";\n eOutputType[\"image\"] = \"image\";\n eOutputType[\"solidworks\"] = \"solidworks\";\n eOutputType[\"inventor\"] = \"inventor\";\n eOutputType[\"creo\"] = \"creo\";\n eOutputType[\"creo3\"] = \"creo3\";\n eOutputType[\"cadDrawing\"] = \"cadDrawing\";\n eOutputType[\"bom\"] = \"bom\";\n eOutputType[\"sceneRender\"] = \"sceneRender\";\n eOutputType[\"pdf\"] = \"pdf\";\n eOutputType[\"salesforce\"] = \"salesforce\";\n eOutputType[\"autoCad\"] = \"autoCad\";\n eOutputType[\"zip\"] = \"zip\";\n eOutputType[\"svg\"] = \"svg\";\n eOutputType[\"text\"] = \"text\";\n })(exports.eOutputType || (exports.eOutputType = {}));\n (function (eOutputFieldType) {\n eOutputFieldType[\"wordText\"] = \"wordText\";\n eOutputFieldType[\"wordContainer\"] = \"wordContainer\";\n eOutputFieldType[\"wordImage\"] = \"wordImage\";\n eOutputFieldType[\"wordList\"] = \"wordList\";\n eOutputFieldType[\"excelSheet\"] = \"excelSheet\";\n eOutputFieldType[\"excelCell\"] = \"excelCell\";\n eOutputFieldType[\"excelImage\"] = \"excelImage\";\n eOutputFieldType[\"excelTable\"] = \"excelTable\";\n eOutputFieldType[\"excelChart\"] = \"excelChart\";\n eOutputFieldType[\"cadReplaceComponent\"] = \"cadReplaceComponent\";\n eOutputFieldType[\"cadAngleMate\"] = \"cadAngleMate\";\n eOutputFieldType[\"cadCoincidentMate\"] = \"cadCoincidentMate\";\n eOutputFieldType[\"cadCoordinateSystemMate\"] = \"cadCoordinateSystemMate\";\n eOutputFieldType[\"cadDistanceMate\"] = \"cadDistanceMate\";\n eOutputFieldType[\"cadPositionConstraint\"] = \"cadPositionConstraint\";\n eOutputFieldType[\"cadDimension\"] = \"cadDimension\";\n eOutputFieldType[\"cadFeature\"] = \"cadFeature\";\n eOutputFieldType[\"cadLibFeature\"] = \"cadLibFeature\";\n eOutputFieldType[\"cadCustomProperty\"] = \"cadCustomProperty\";\n eOutputFieldType[\"cadComponent\"] = \"cadComponent\";\n eOutputFieldType[\"cadDrawingView\"] = \"cadDrawingView\";\n eOutputFieldType[\"cadDrawingSheet\"] = \"cadDrawingSheet\";\n eOutputFieldType[\"cadDrawingAnnotation\"] = \"cadDrawingAnnotation\";\n eOutputFieldType[\"cadDynamicDrawingDimension\"] = \"cadDynamicDrawingDimension\";\n eOutputFieldType[\"cadDynamicDrawingAnnotation\"] = \"cadDynamicDrawingAnnotation\";\n eOutputFieldType[\"cadDynamicComponent\"] = \"cadDynamicComponent\";\n eOutputFieldType[\"cadNestedComponent\"] = \"cadNestedComponent\";\n eOutputFieldType[\"cadScaleFeature\"] = \"cadScaleFeature\";\n eOutputFieldType[\"cadAnnotation\"] = \"cadAnnotation\";\n eOutputFieldType[\"cadText\"] = \"cadText\";\n eOutputFieldType[\"cadParameter\"] = \"cadParameter\";\n eOutputFieldType[\"cadRepresentation\"] = \"cadRepresentation\";\n eOutputFieldType[\"cadPoint\"] = \"cadPoint\";\n eOutputFieldType[\"cadLayer\"] = \"cadLayer\";\n eOutputFieldType[\"bomMapping\"] = \"bomMapping\";\n eOutputFieldType[\"salesForceObject\"] = \"salesForceObject\";\n eOutputFieldType[\"cadDrawingDimension\"] = \"cadDrawingDimension\";\n eOutputFieldType[\"pdfSection\"] = \"pdfSection\";\n eOutputFieldType[\"cadLibReference\"] = \"cadLibReference\";\n eOutputFieldType[\"cadLibDimension\"] = \"cadLibDimension\";\n eOutputFieldType[\"cadBlock\"] = \"cadBlock\";\n eOutputFieldType[\"cadBlockArray\"] = \"cadBlockArray\";\n eOutputFieldType[\"zipSection\"] = \"zipSection\";\n })(exports.eOutputFieldType || (exports.eOutputFieldType = {}));\n (function (eSingleOptionBehavior) {\n eSingleOptionBehavior[\"nothing\"] = \"nothing\";\n eSingleOptionBehavior[\"disable\"] = \"disable\";\n eSingleOptionBehavior[\"hide\"] = \"hide\";\n })(exports.eSingleOptionBehavior || (exports.eSingleOptionBehavior = {}));\n (function (eConfiguratorImageSource) {\n eConfiguratorImageSource[\"none\"] = \"none\";\n eConfiguratorImageSource[\"media\"] = \"media\";\n eConfiguratorImageSource[\"scene\"] = \"scene\";\n })(exports.eConfiguratorImageSource || (exports.eConfiguratorImageSource = {}));\n (function (eTextColor) {\n eTextColor[\"normal\"] = \"normal\";\n eTextColor[\"header\"] = \"header\";\n eTextColor[\"primary\"] = \"primary\";\n eTextColor[\"accent\"] = \"accent\";\n eTextColor[\"error\"] = \"error\";\n eTextColor[\"success\"] = \"success\";\n })(exports.eTextColor || (exports.eTextColor = {}));\n (function (eConfiguratorEntityType) {\n eConfiguratorEntityType[\"product\"] = \"product\";\n eConfiguratorEntityType[\"quoteHeader\"] = \"quoteHeader\";\n })(exports.eConfiguratorEntityType || (exports.eConfiguratorEntityType = {}));\n (function (eBackground) {\n eBackground[\"color\"] = \"color\";\n eBackground[\"transparent\"] = \"transparent\";\n eBackground[\"environment\"] = \"environment\";\n })(exports.eBackground || (exports.eBackground = {}));\n (function (eSideOrientation) {\n eSideOrientation[\"frontSide\"] = \"frontSide\";\n eSideOrientation[\"backSide\"] = \"backSide\";\n eSideOrientation[\"doubleSide\"] = \"doubleSide\";\n })(exports.eSideOrientation || (exports.eSideOrientation = {}));\n (function (eUvMode) {\n eUvMode[\"planar\"] = \"planar\";\n eUvMode[\"box\"] = \"box\";\n eUvMode[\"cylindrical\"] = \"cylindrical\";\n eUvMode[\"spherical\"] = \"spherical\";\n eUvMode[\"useExisting\"] = \"useExisting\";\n })(exports.eUvMode || (exports.eUvMode = {}));\n (function (eMaterialViewMode) {\n eMaterialViewMode[\"shaded\"] = \"shaded\";\n eMaterialViewMode[\"edges\"] = \"edges\";\n eMaterialViewMode[\"wireframe\"] = \"wireframe\";\n })(exports.eMaterialViewMode || (exports.eMaterialViewMode = {}));\n (function (eMaterialLocation) {\n eMaterialLocation[\"local\"] = \"local\";\n eMaterialLocation[\"global\"] = \"global\";\n eMaterialLocation[\"library\"] = \"library\";\n })(exports.eMaterialLocation || (exports.eMaterialLocation = {}));\n (function (eAlphaMode) {\n eAlphaMode[\"disable\"] = \"disable\";\n eAlphaMode[\"add\"] = \"add\";\n eAlphaMode[\"combine\"] = \"combine\";\n eAlphaMode[\"subtract\"] = \"subtract\";\n eAlphaMode[\"multiply\"] = \"multiply\";\n })(exports.eAlphaMode || (exports.eAlphaMode = {}));\n (function (eViewpointType) {\n eViewpointType[\"orbit\"] = \"orbit\";\n eViewpointType[\"firstPerson\"] = \"firstPerson\";\n eViewpointType[\"overhead\"] = \"overhead\";\n })(exports.eViewpointType || (exports.eViewpointType = {}));\n (function (eViewpointProjection) {\n eViewpointProjection[\"perspective\"] = \"perspective\";\n eViewpointProjection[\"orthographic\"] = \"orthographic\";\n })(exports.eViewpointProjection || (exports.eViewpointProjection = {}));\n (function (eViewpointTargetMode) {\n eViewpointTargetMode[\"rotation\"] = \"rotation\";\n eViewpointTargetMode[\"point\"] = \"point\";\n eViewpointTargetMode[\"mesh\"] = \"mesh\";\n })(exports.eViewpointTargetMode || (exports.eViewpointTargetMode = {}));\n (function (eConnectorPositionMode) {\n eConnectorPositionMode[\"boundingBox\"] = \"boundingBox\";\n eConnectorPositionMode[\"absolute\"] = \"absolute\";\n eConnectorPositionMode[\"moveAlongSketch\"] = \"moveAlongSketch\";\n })(exports.eConnectorPositionMode || (exports.eConnectorPositionMode = {}));\n (function (eDragMode) {\n eDragMode[\"plane\"] = \"plane\";\n eDragMode[\"free\"] = \"free\";\n eDragMode[\"surface\"] = \"surface\";\n })(exports.eDragMode || (exports.eDragMode = {}));\n (function (eConnectionMode) {\n eConnectionMode[\"none\"] = \"none\";\n eConnectionMode[\"world\"] = \"world\";\n eConnectionMode[\"screen\"] = \"screen\";\n })(exports.eConnectionMode || (exports.eConnectionMode = {}));\n (function (eEasing) {\n eEasing[\"linear\"] = \"linear\";\n eEasing[\"quadraticIn\"] = \"quadraticIn\";\n eEasing[\"quadraticOut\"] = \"quadraticOut\";\n eEasing[\"quadraticInOut\"] = \"quadraticInOut\";\n eEasing[\"cubicIn\"] = \"cubicIn\";\n eEasing[\"cubicOut\"] = \"cubicOut\";\n eEasing[\"cubicInOut\"] = \"cubicInOut\";\n eEasing[\"quarticIn\"] = \"quarticIn\";\n eEasing[\"quarticOut\"] = \"quarticOut\";\n eEasing[\"quarticInOut\"] = \"quarticInOut\";\n eEasing[\"quinticIn\"] = \"quinticIn\";\n eEasing[\"quinticOut\"] = \"quinticOut\";\n eEasing[\"quinticInOut\"] = \"quinticInOut\";\n eEasing[\"sinusoidalIn\"] = \"sinusoidalIn\";\n eEasing[\"sinusoidalOut\"] = \"sinusoidalOut\";\n eEasing[\"sinusoidalInOut\"] = \"sinusoidalInOut\";\n eEasing[\"exponentialIn\"] = \"exponentialIn\";\n eEasing[\"exponentialOut\"] = \"exponentialOut\";\n eEasing[\"exponentialInOut\"] = \"exponentialInOut\";\n eEasing[\"circularIn\"] = \"circularIn\";\n eEasing[\"circularOut\"] = \"circularOut\";\n eEasing[\"circularInOut\"] = \"circularInOut\";\n eEasing[\"elasticIn\"] = \"elasticIn\";\n eEasing[\"elasticOut\"] = \"elasticOut\";\n eEasing[\"elasticInOut\"] = \"elasticInOut\";\n eEasing[\"backIn\"] = \"backIn\";\n eEasing[\"backOut\"] = \"backOut\";\n eEasing[\"backInOut\"] = \"backInOut\";\n eEasing[\"bounceIn\"] = \"bounceIn\";\n eEasing[\"bounceOut\"] = \"bounceOut\";\n eEasing[\"bounceInOut\"] = \"bounceInOut\";\n })(exports.eEasing || (exports.eEasing = {}));\n (function (eCylinderCap) {\n eCylinderCap[\"none\"] = \"none\";\n eCylinderCap[\"start\"] = \"start\";\n eCylinderCap[\"end\"] = \"end\";\n eCylinderCap[\"both\"] = \"both\";\n })(exports.eCylinderCap || (exports.eCylinderCap = {}));\n (function (eJoinType) {\n eJoinType[\"union\"] = \"union\";\n eJoinType[\"intersect\"] = \"intersect\";\n eJoinType[\"cut\"] = \"cut\";\n })(exports.eJoinType || (exports.eJoinType = {}));\n (function (eShadowQuality) {\n eShadowQuality[\"low\"] = \"low\";\n eShadowQuality[\"medium\"] = \"medium\";\n eShadowQuality[\"high\"] = \"high\";\n })(exports.eShadowQuality || (exports.eShadowQuality = {}));\n (function (eShadowResolution) {\n eShadowResolution[\"low\"] = \"low\";\n eShadowResolution[\"medium\"] = \"medium\";\n eShadowResolution[\"high\"] = \"high\";\n })(exports.eShadowResolution || (exports.eShadowResolution = {}));\n (function (eShadowFilter) {\n eShadowFilter[\"none\"] = \"none\";\n eShadowFilter[\"simple\"] = \"simple\";\n eShadowFilter[\"advanced\"] = \"advanced\";\n })(exports.eShadowFilter || (exports.eShadowFilter = {}));\n (function (eMaterialWorkflow) {\n eMaterialWorkflow[\"metallicRoughness\"] = \"metallicRoughness\";\n eMaterialWorkflow[\"specularGlossiness\"] = \"specularGlossiness\";\n })(exports.eMaterialWorkflow || (exports.eMaterialWorkflow = {}));\n (function (eFeatureCenter) {\n eFeatureCenter[\"barycenter\"] = \"barycenter\";\n eFeatureCenter[\"meshPivotPoint\"] = \"meshPivotPoint\";\n })(exports.eFeatureCenter || (exports.eFeatureCenter = {}));\n (function (eJobStatus) {\n eJobStatus[\"none\"] = \"none\";\n eJobStatus[\"waiting\"] = \"waiting\";\n eJobStatus[\"queued\"] = \"queued\";\n eJobStatus[\"working\"] = \"working\";\n eJobStatus[\"complete\"] = \"complete\";\n eJobStatus[\"failed\"] = \"failed\";\n })(exports.eJobStatus || (exports.eJobStatus = {}));\n (function (eJobType) {\n eJobType[\"build\"] = \"build\";\n eJobType[\"render\"] = \"render\";\n eJobType[\"notification\"] = \"notification\";\n eJobType[\"upsertProductsToSfdc\"] = \"upsertProductsToSfdc\";\n eJobType[\"pullSfdcProducts\"] = \"pullSfdcProducts\";\n eJobType[\"submitFromSfdcCpq\"] = \"submitFromSfdcCpq\";\n eJobType[\"completeBuildState\"] = \"completeBuildState\";\n eJobType[\"cleanDb\"] = \"cleanDb\";\n eJobType[\"deployment\"] = \"deployment\";\n eJobType[\"sceneImport\"] = \"sceneImport\";\n eJobType[\"sceneExport\"] = \"sceneExport\";\n eJobType[\"sceneConvert\"] = \"sceneConvert\";\n eJobType[\"completeQuoteProductBuild\"] = \"completeQuoteProductBuild\";\n eJobType[\"configuratorExport\"] = \"configuratorExport\";\n eJobType[\"configuratorImport\"] = \"configuratorImport\";\n })(exports.eJobType || (exports.eJobType = {}));\n (function (eEnvironment) {\n eEnvironment[\"prod\"] = \"prod\";\n eEnvironment[\"test\"] = \"test\";\n eEnvironment[\"dev\"] = \"dev\";\n })(exports.eEnvironment || (exports.eEnvironment = {}));\n (function (eImportType) {\n eImportType[\"external\"] = \"external\";\n eImportType[\"kb3d\"] = \"kb3d\";\n })(exports.eImportType || (exports.eImportType = {}));\n (function (eQueueMessageType) {\n eQueueMessageType[\"builder\"] = \"builder\";\n eQueueMessageType[\"cleanDb\"] = \"cleanDb\";\n eQueueMessageType[\"completeBuildState\"] = \"completeBuildState\";\n eQueueMessageType[\"notification\"] = \"notification\";\n eQueueMessageType[\"render\"] = \"render\";\n eQueueMessageType[\"pullSfdcProducts\"] = \"pullSfdcProducts\";\n eQueueMessageType[\"upsertProductsToSfdc\"] = \"upsertProductsToSfdc\";\n eQueueMessageType[\"submitFromSfdcCpq\"] = \"submitFromSfdcCpq\";\n eQueueMessageType[\"deployment\"] = \"deployment\";\n eQueueMessageType[\"completeQuoteProductBuild\"] = \"completeQuoteProductBuild\";\n eQueueMessageType[\"sceneImport\"] = \"sceneImport\";\n eQueueMessageType[\"sceneExport\"] = \"sceneExport\";\n eQueueMessageType[\"sceneConvert\"] = \"sceneConvert\";\n })(exports.eQueueMessageType || (exports.eQueueMessageType = {}));\n (function (eMateType) {\n eMateType[\"fastened\"] = \"fastened\";\n eMateType[\"revolute\"] = \"revolute\";\n eMateType[\"slider\"] = \"slider\";\n eMateType[\"planar\"] = \"planar\";\n eMateType[\"cylindrical\"] = \"cylindrical\";\n eMateType[\"ball\"] = \"ball\";\n eMateType[\"parallel\"] = \"parallel\";\n })(exports.eMateType || (exports.eMateType = {}));\n (function (eValidationType) {\n eValidationType[\"error\"] = \"error\";\n eValidationType[\"warning\"] = \"warning\";\n eValidationType[\"info\"] = \"info\";\n })(exports.eValidationType || (exports.eValidationType = {}));\n (function (eValidationSource) {\n eValidationSource[\"snap\"] = \"snap\";\n eValidationSource[\"configurator\"] = \"configurator\";\n eValidationSource[\"scene\"] = \"scene\";\n })(exports.eValidationSource || (exports.eValidationSource = {}));\n (function (eTextureChannel) {\n eTextureChannel[\"emissive\"] = \"emissive\";\n eTextureChannel[\"albedo\"] = \"albedo\";\n eTextureChannel[\"ambient\"] = \"ambient\";\n eTextureChannel[\"opacity\"] = \"opacity\";\n eTextureChannel[\"reflection\"] = \"reflection\";\n eTextureChannel[\"specular\"] = \"specular\";\n eTextureChannel[\"metallic\"] = \"metallic\";\n eTextureChannel[\"roughness\"] = \"roughness\";\n eTextureChannel[\"glossiness\"] = \"glossiness\";\n eTextureChannel[\"normal\"] = \"normal\";\n eTextureChannel[\"ambientOcclusion\"] = \"ambientOcclusion\";\n eTextureChannel[\"anisotropyTangent\"] = \"anisotropyTangent\";\n eTextureChannel[\"subsurfaceThickness\"] = \"subsurfaceThickness\";\n eTextureChannel[\"clearCoatNormal\"] = \"clearCoatNormal\";\n })(exports.eTextureChannel || (exports.eTextureChannel = {}));\n (function (eTextureHorizontalAlign) {\n eTextureHorizontalAlign[\"left\"] = \"left\";\n eTextureHorizontalAlign[\"center\"] = \"center\";\n eTextureHorizontalAlign[\"right\"] = \"right\";\n })(exports.eTextureHorizontalAlign || (exports.eTextureHorizontalAlign = {}));\n (function (eTextureVerticalAlign) {\n eTextureVerticalAlign[\"top\"] = \"top\";\n eTextureVerticalAlign[\"middle\"] = \"middle\";\n eTextureVerticalAlign[\"bottom\"] = \"bottom\";\n })(exports.eTextureVerticalAlign || (exports.eTextureVerticalAlign = {}));\n (function (eTextureFit) {\n eTextureFit[\"none\"] = \"none\";\n eTextureFit[\"stretch\"] = \"stretch\";\n eTextureFit[\"contain\"] = \"contain\";\n })(exports.eTextureFit || (exports.eTextureFit = {}));\n (function (eTextureBlendMode) {\n eTextureBlendMode[\"normal\"] = \"normal\";\n eTextureBlendMode[\"multiply\"] = \"multiply\";\n eTextureBlendMode[\"screen\"] = \"screen\";\n eTextureBlendMode[\"overlay\"] = \"overlay\";\n eTextureBlendMode[\"darken\"] = \"darken\";\n eTextureBlendMode[\"lighten\"] = \"lighten\";\n eTextureBlendMode[\"hue\"] = \"hue\";\n eTextureBlendMode[\"saturation\"] = \"saturation\";\n eTextureBlendMode[\"color\"] = \"color\";\n eTextureBlendMode[\"luminosity\"] = \"luminosity\";\n eTextureBlendMode[\"difference\"] = \"difference\";\n eTextureBlendMode[\"mask\"] = \"mask\";\n eTextureBlendMode[\"vector\"] = \"vector\";\n })(exports.eTextureBlendMode || (exports.eTextureBlendMode = {}));\n (function (eTextureWrapAddressMode) {\n eTextureWrapAddressMode[\"repeat\"] = \"repeat\";\n eTextureWrapAddressMode[\"clamp\"] = \"clamp\";\n })(exports.eTextureWrapAddressMode || (exports.eTextureWrapAddressMode = {}));\n (function (eAnnotationLineStyle) {\n eAnnotationLineStyle[\"solid\"] = \"solid\";\n eAnnotationLineStyle[\"dashed\"] = \"dashed\";\n eAnnotationLineStyle[\"dotted\"] = \"dotted\";\n eAnnotationLineStyle[\"none\"] = \"none\";\n })(exports.eAnnotationLineStyle || (exports.eAnnotationLineStyle = {}));\n (function (eDimensionLineCapStyle) {\n eDimensionLineCapStyle[\"none\"] = \"none\";\n eDimensionLineCapStyle[\"arrow\"] = \"arrow\";\n eDimensionLineCapStyle[\"line\"] = \"line\";\n })(exports.eDimensionLineCapStyle || (exports.eDimensionLineCapStyle = {}));\n (function (eDimensionAxis) {\n eDimensionAxis[\"free\"] = \"free\";\n eDimensionAxis[\"x\"] = \"x\";\n eDimensionAxis[\"y\"] = \"y\";\n eDimensionAxis[\"z\"] = \"z\";\n })(exports.eDimensionAxis || (exports.eDimensionAxis = {}));\n (function (eLoadingExperience) {\n eLoadingExperience[\"none\"] = \"none\";\n eLoadingExperience[\"fade\"] = \"fade\";\n eLoadingExperience[\"spinner\"] = \"spinner\";\n eLoadingExperience[\"image\"] = \"image\";\n })(exports.eLoadingExperience || (exports.eLoadingExperience = {}));\n (function (eReflectionType) {\n eReflectionType[\"plane\"] = \"plane\";\n eReflectionType[\"probe\"] = \"probe\";\n })(exports.eReflectionType || (exports.eReflectionType = {}));\n (function (eRenderMode) {\n eRenderMode[\"shaded\"] = \"shaded\";\n eRenderMode[\"wireframe\"] = \"wireframe\";\n eRenderMode[\"edge\"] = \"edge\";\n eRenderMode[\"hiddenEdges\"] = \"hiddenEdges\";\n })(exports.eRenderMode || (exports.eRenderMode = {}));\n (function (eTextureSamplingMode) {\n eTextureSamplingMode[\"nearest\"] = \"nearest\";\n eTextureSamplingMode[\"bilinear\"] = \"bilinear\";\n eTextureSamplingMode[\"trilinear\"] = \"trilinear\";\n })(exports.eTextureSamplingMode || (exports.eTextureSamplingMode = {}));\n (function (eAxis) {\n eAxis[\"x\"] = \"x\";\n eAxis[\"y\"] = \"y\";\n eAxis[\"z\"] = \"z\";\n })(exports.eAxis || (exports.eAxis = {}));\n (function (eSketchPathType) {\n eSketchPathType[\"points\"] = \"points\";\n eSketchPathType[\"arcThrough3Points\"] = \"arcThrough3Points\";\n eSketchPathType[\"centerRadiusCircle\"] = \"centerRadiusCircle\";\n eSketchPathType[\"catmullRom\"] = \"catmullRom\";\n eSketchPathType[\"cubicBezier\"] = \"cubicBezier\";\n eSketchPathType[\"quadraticBezier\"] = \"quadraticBezier\";\n eSketchPathType[\"hermite\"] = \"hermite\";\n })(exports.eSketchPathType || (exports.eSketchPathType = {}));\n (function (eFilterControl) {\n eFilterControl[\"none\"] = \"none\";\n eFilterControl[\"text\"] = \"text\";\n eFilterControl[\"select\"] = \"select\";\n eFilterControl[\"multiSelect\"] = \"multiSelect\";\n eFilterControl[\"dateRange\"] = \"dateRange\";\n eFilterControl[\"numberRange\"] = \"numberRange\";\n eFilterControl[\"suggest\"] = \"suggest\";\n eFilterControl[\"checkbox\"] = \"checkbox\";\n })(exports.eFilterControl || (exports.eFilterControl = {}));\n (function (eFilterSource) {\n eFilterSource[\"none\"] = \"none\";\n eFilterSource[\"enum\"] = \"enum\";\n eFilterSource[\"properties\"] = \"properties\";\n eFilterSource[\"users\"] = \"users\";\n eFilterSource[\"states\"] = \"states\";\n eFilterSource[\"configurators\"] = \"configurators\";\n eFilterSource[\"fields\"] = \"fields\";\n eFilterSource[\"customers\"] = \"customers\";\n eFilterSource[\"roles\"] = \"roles\";\n eFilterSource[\"productSorts\"] = \"productSorts\";\n eFilterSource[\"manufacturers\"] = \"manufacturers\";\n eFilterSource[\"attributes\"] = \"attributes\";\n eFilterSource[\"suggestContacts\"] = \"suggestContacts\";\n eFilterSource[\"suggestCustomers\"] = \"suggestCustomers\";\n eFilterSource[\"suggestQuotes\"] = \"suggestQuotes\";\n eFilterSource[\"suggestProducts\"] = \"suggestProducts\";\n eFilterSource[\"headerFields\"] = \"headerFields\";\n })(exports.eFilterSource || (exports.eFilterSource = {}));\n (function (eFilterType) {\n eFilterType[\"property\"] = \"property\";\n eFilterType[\"fieldValue\"] = \"fieldValue\";\n eFilterType[\"sort\"] = \"sort\";\n eFilterType[\"attribute\"] = \"attribute\";\n eFilterType[\"headerValue\"] = \"headerValue\";\n })(exports.eFilterType || (exports.eFilterType = {}));\n (function (eFieldType) {\n eFieldType[\"text\"] = \"text\";\n eFieldType[\"number\"] = \"number\";\n eFieldType[\"boolean\"] = \"boolean\";\n eFieldType[\"color\"] = \"color\";\n eFieldType[\"date\"] = \"date\";\n eFieldType[\"upload\"] = \"upload\";\n eFieldType[\"textArray\"] = \"textArray\";\n eFieldType[\"numberArray\"] = \"numberArray\";\n eFieldType[\"colorArray\"] = \"colorArray\";\n })(exports.eFieldType || (exports.eFieldType = {}));\n\n var _kb3dEnums = /*#__PURE__*/Object.freeze({\n __proto__: null,\n get eContainerItemLayout () { return exports.eContainerItemLayout; },\n get eContainerChildAlignment () { return exports.eContainerChildAlignment; },\n get eContainerItemAlignment () { return exports.eContainerItemAlignment; },\n get eContainerItemJustify () { return exports.eContainerItemJustify; },\n get eScroll () { return exports.eScroll; },\n get eRelativeSize () { return exports.eRelativeSize; },\n get ePadding () { return exports.ePadding; },\n get eColorShade () { return exports.eColorShade; },\n get eBackgroundColor () { return exports.eBackgroundColor; },\n get eElevation () { return exports.eElevation; },\n get eRuleType () { return exports.eRuleType; },\n get eToolbarPosition () { return exports.eToolbarPosition; },\n get eValidationTiming () { return exports.eValidationTiming; },\n get eViewerMode () { return exports.eViewerMode; },\n get ePosition () { return exports.ePosition; },\n get eIconStyle () { return exports.eIconStyle; },\n get eExportType () { return exports.eExportType; },\n get eHotspotSource () { return exports.eHotspotSource; },\n get eHotspotAttach () { return exports.eHotspotAttach; },\n get eHotspotPosition () { return exports.eHotspotPosition; },\n get eHotspotShape () { return exports.eHotspotShape; },\n get eAnimationEasing () { return exports.eAnimationEasing; },\n get eQueryMode () { return exports.eQueryMode; },\n get eOptionFilterSource () { return exports.eOptionFilterSource; },\n get eOutputType () { return exports.eOutputType; },\n get eOutputFieldType () { return exports.eOutputFieldType; },\n get eSingleOptionBehavior () { return exports.eSingleOptionBehavior; },\n get eConfiguratorImageSource () { return exports.eConfiguratorImageSource; },\n get eTextColor () { return exports.eTextColor; },\n get eConfiguratorEntityType () { return exports.eConfiguratorEntityType; },\n get eBackground () { return exports.eBackground; },\n get eSideOrientation () { return exports.eSideOrientation; },\n get eUvMode () { return exports.eUvMode; },\n get eMaterialViewMode () { return exports.eMaterialViewMode; },\n get eMaterialLocation () { return exports.eMaterialLocation; },\n get eAlphaMode () { return exports.eAlphaMode; },\n get eViewpointType () { return exports.eViewpointType; },\n get eViewpointProjection () { return exports.eViewpointProjection; },\n get eViewpointTargetMode () { return exports.eViewpointTargetMode; },\n get eConnectorPositionMode () { return exports.eConnectorPositionMode; },\n get eDragMode () { return exports.eDragMode; },\n get eConnectionMode () { return exports.eConnectionMode; },\n get eEasing () { return exports.eEasing; },\n get eCylinderCap () { return exports.eCylinderCap; },\n get eJoinType () { return exports.eJoinType; },\n get eShadowQuality () { return exports.eShadowQuality; },\n get eShadowResolution () { return exports.eShadowResolution; },\n get eShadowFilter () { return exports.eShadowFilter; },\n get eMaterialWorkflow () { return exports.eMaterialWorkflow; },\n get eFeatureCenter () { return exports.eFeatureCenter; },\n get eJobStatus () { return exports.eJobStatus; },\n get eJobType () { return exports.eJobType; },\n get eEnvironment () { return exports.eEnvironment; },\n get eImportType () { return exports.eImportType; },\n get eQueueMessageType () { return exports.eQueueMessageType; },\n get eMateType () { return exports.eMateType; },\n get eValidationType () { return exports.eValidationType; },\n get eValidationSource () { return exports.eValidationSource; },\n get eTextureChannel () { return exports.eTextureChannel; },\n get eTextureHorizontalAlign () { return exports.eTextureHorizontalAlign; },\n get eTextureVerticalAlign () { return exports.eTextureVerticalAlign; },\n get eTextureFit () { return exports.eTextureFit; },\n get eTextureBlendMode () { return exports.eTextureBlendMode; },\n get eTextureWrapAddressMode () { return exports.eTextureWrapAddressMode; },\n get eAnnotationLineStyle () { return exports.eAnnotationLineStyle; },\n get eDimensionLineCapStyle () { return exports.eDimensionLineCapStyle; },\n get eDimensionAxis () { return exports.eDimensionAxis; },\n get eLoadingExperience () { return exports.eLoadingExperience; },\n get eReflectionType () { return exports.eReflectionType; },\n get eRenderMode () { return exports.eRenderMode; },\n get eTextureSamplingMode () { return exports.eTextureSamplingMode; },\n get eAxis () { return exports.eAxis; },\n get eSketchPathType () { return exports.eSketchPathType; },\n get eFilterControl () { return exports.eFilterControl; },\n get eFilterSource () { return exports.eFilterSource; },\n get eFilterType () { return exports.eFilterType; },\n get eFieldType () { return exports.eFieldType; }\n });\n\n exports.SketchControlPoint = class SketchControlPoint extends exports.BaseConnector {\n constructor() {\n super(...arguments);\n this.enabled = true; // non-configurable\n this.positionMode = exports.eConnectorPositionMode.absolute;\n }\n getSketch() {\n var _a;\n return (_a = this._parent) === null || _a === void 0 ? void 0 : _a._parent;\n }\n getSpaceParent() {\n return this.getSketch();\n }\n getAncestors() {\n return super.getAncestors();\n }\n };\n exports.SketchControlPoint = __decorate([\n Kb3dModel({\n token: Token.SketchControlPoint,\n })\n ], exports.SketchControlPoint);\n\n exports.UvMapFeature = class UvMapFeature extends exports.Feature {\n constructor() {\n var _a;\n super(...arguments);\n this.worldPosition = (_a = this._parent) === null || _a === void 0 ? void 0 : _a.position;\n }\n };\n exports.UvMapFeature._affects = new Set(ALL_GEOMETRY);\n exports.UvMapFeature._affectedBy = new Set(ALL_GEOMETRY);\n exports.UvMapFeature = __decorate([\n Kb3dModel({\n token: Token.UvMapFeature,\n })\n ], exports.UvMapFeature);\n\n exports.SubmeshFeature = class SubmeshFeature extends exports.UvMapFeature {\n constructor() {\n super(...arguments);\n this.vertexGroups = [];\n }\n };\n exports.SubmeshFeature._dependsOn = new Set([exports.eMeshChangeTypes.VertexIndices]);\n exports.SubmeshFeature._affects = new Set(ALL_GEOMETRY);\n exports.SubmeshFeature._affectedBy = new Set(ALL_GEOMETRY);\n exports.SubmeshFeature = __decorate([\n Kb3dModel({\n token: Token.SubmeshFeature,\n })\n ], exports.SubmeshFeature);\n function isSubmesh(o) {\n return instanceOf(o, Token.SubmeshFeature);\n }\n\n exports.SweepFeature = class SweepFeature extends exports.ExtrudeFeature {\n constructor() {\n super(...arguments);\n this.size = 0;\n }\n };\n exports.SweepFeature._affects = new Set(ALL_GEOMETRY);\n exports.SweepFeature._affectedBy = new Set([...ALL_GEOMETRY, exports.eMeshChangeTypes.Transform]);\n __decorate([\n NodeReference(exports.BaseConnector)\n ], exports.SweepFeature.prototype, \"start\", void 0);\n __decorate([\n NodeReference(exports.BaseConnector)\n ], exports.SweepFeature.prototype, \"end\", void 0);\n __decorate([\n NodeReference(exports.SketchNode)\n ], exports.SweepFeature.prototype, \"path\", void 0);\n exports.SweepFeature = __decorate([\n Kb3dModel({\n token: Token.SweepFeature,\n })\n ], exports.SweepFeature);\n\n exports.TransformFeature = class TransformFeature extends exports.Feature {\n };\n exports.TransformFeature._affectedBy = new Set([exports.eMeshChangeTypes.Transform, exports.eMeshChangeTypes.VertexPositions]);\n exports.TransformFeature._affects = new Set([exports.eMeshChangeTypes.VertexPositions]);\n exports.TransformFeature = __decorate([\n Kb3dModel({\n token: Token.TransformFeature,\n })\n ], exports.TransformFeature);\n\n exports.WeldVerticesFeature = class WeldVerticesFeature extends exports.Feature {\n get vertexIndices() {\n if (!this._vertexIndices)\n this._vertexIndices = [];\n return this._vertexIndices;\n }\n set vertexIndices(val) {\n this._vertexIndices = val;\n }\n };\n exports.WeldVerticesFeature._dependsOn = new Set([exports.eMeshChangeTypes.VertexIndices]);\n exports.WeldVerticesFeature._affectedBy = new Set([exports.eMeshChangeTypes.VertexPositions, exports.eMeshChangeTypes.VertexIndices]);\n exports.WeldVerticesFeature._affects = new Set([\n exports.eMeshChangeTypes.VertexPositions,\n exports.eMeshChangeTypes.VertexIndices,\n exports.eMeshChangeTypes.VertexUvs,\n exports.eMeshChangeTypes.VertexNormals,\n exports.eMeshChangeTypes.Submeshes,\n ]);\n __decorate([\n IgnoreChanges\n ], exports.WeldVerticesFeature.prototype, \"_vertexIndices\", void 0);\n __decorate([\n Enumerable()\n ], exports.WeldVerticesFeature.prototype, \"vertexIndices\", null);\n exports.WeldVerticesFeature = __decorate([\n Kb3dModel({\n token: Token.WeldVerticesFeature,\n })\n ], exports.WeldVerticesFeature);\n\n exports.AnnotationNode = class AnnotationNode extends exports.Kb3dObject {\n constructor() {\n super();\n this.label = '';\n this.preloadFontStylesheet = '';\n }\n };\n __decorate([\n NodeReference(exports.Connector, 'position', 'direction', 'angle')\n ], exports.AnnotationNode.prototype, \"targetId\", void 0);\n exports.AnnotationNode = __decorate([\n Kb3dModel({\n token: Token.AnnotationNode,\n })\n ], exports.AnnotationNode);\n function isAnnotation(node) {\n return instanceOfAny(node, Token.AnnotationNode, Token.DimensionNode);\n }\n\n exports.DimensionNode = class DimensionNode extends exports.AnnotationNode {\n constructor() {\n super();\n }\n };\n __decorate([\n NodeReference(exports.Connector, 'position')\n ], exports.DimensionNode.prototype, \"target2Id\", void 0);\n exports.DimensionNode = __decorate([\n Kb3dModel({\n token: Token.DimensionNode,\n })\n ], exports.DimensionNode);\n\n (function (eHandler) {\n eHandler[\"click\"] = \"click\";\n eHandler[\"doubleClick\"] = \"doubleClick\";\n eHandler[\"draggable\"] = \"draggable\";\n eHandler[\"hotspot\"] = \"hotspot\";\n eHandler[\"boundingBox\"] = \"boundingBox\";\n })(exports.eHandler || (exports.eHandler = {}));\n /** A 2 layer map that stores values by a key 1st, then a 2nd level of a string tag, then an array of values */\n class TagArrayMap {\n constructor() {\n this._db = new Map();\n }\n get(key, tag) {\n var _a;\n const v = (_a = this._db.get(key)) === null || _a === void 0 ? void 0 : _a.get(tag);\n return v || [];\n }\n deleteByKeyValue(key, value) {\n const e = this._db.get(key);\n if (e) {\n for (const values of e.values()) {\n values.removeWhere(v => v == value);\n }\n this.deleteEntryIfEmpty(key);\n }\n }\n getByKey(key) {\n if (this._db.has(key)) {\n return this._db.get(key).all();\n }\n return [];\n }\n set(key, tag, value) {\n var _a;\n if (!this._db.has(key))\n this._db.set(key, new ArrayMap());\n (_a = this._db.get(key)) === null || _a === void 0 ? void 0 : _a.add(tag, value);\n }\n deleteByTag(...tags) {\n const result = [];\n for (const [key, value] of this._db.entries()) {\n const nodeResult = {};\n result.push({ node: key, handlers: nodeResult });\n for (const tag of tags) {\n const handler = value.get(tag);\n if (handler) {\n nodeResult[tag] = handler;\n }\n value.delete(tag);\n }\n this.deleteEntryIfEmpty(key);\n }\n return result;\n }\n delete(key, tag) {\n const e = this._db.get(key);\n if (e) {\n const result = e.get(tag);\n e.delete(tag);\n this.deleteEntryIfEmpty(key);\n return result;\n }\n return [];\n }\n deleteByKey(key) {\n const handlers = this._db.get(key);\n const result = {};\n if (handlers) {\n handlers.forEach((value, key) => {\n result[key] = value;\n });\n }\n this._db.delete(key);\n return result;\n }\n clear() {\n this._db.clear();\n }\n has(key, tag) {\n return this.get(key, tag).length > 0;\n }\n deleteEntryIfEmpty(key) {\n const e = this._db.get(key);\n if (e && !e.size)\n this._db.delete(key);\n }\n }\n class ArrayMap extends Map {\n get(key) {\n return super.get(key) || [];\n }\n add(key, ...values) {\n if (!this.has(key)) {\n super.set(key, [...values]);\n }\n else {\n this.get(key).push(...values);\n }\n }\n removeValue(key, ...values) {\n const arr = super.get(key);\n if (arr) {\n arr.removeWhere(i => values.some(v => v == i));\n }\n }\n all() {\n return [...this.values()].flat();\n }\n }\n class SetMap extends Map {\n get(key) {\n const set = super.get(key);\n if (!set)\n super.set(key, new Set());\n return super.get(key);\n }\n add(key, ...values) {\n const s = this.get(key);\n for (const v of values)\n s.add(v);\n }\n removeValue(key, ...values) {\n const s = super.get(key);\n if (s) {\n for (const v of values)\n s.delete(v);\n }\n }\n all() {\n return [...this.values()].flat();\n }\n }\n class HandlerMap extends TagArrayMap {\n constructor(manager) {\n super();\n this.manager = manager;\n }\n add(idNameOrNode, tag, handler) {\n let nodes = new Array();\n if (typeof idNameOrNode == 'string') {\n nodes = this.manager.getNodesByIdOrName(idNameOrNode, true);\n }\n else {\n nodes.push(idNameOrNode);\n }\n for (const n of nodes) {\n if (isPickable(n)) {\n this.set(n, tag, handler);\n }\n }\n }\n }\n\n exports.ImportInfo = class ImportInfo {\n };\n exports.ImportInfo = __decorate([\n Kb3dModel({\n token: Token.ImportInfo,\n trackChanges: false,\n })\n ], exports.ImportInfo);\n\n const fontFilesLoading = new Map();\n const stylesheetsLoading = new Map();\n const fontFamilySrcs = new Map();\n class FontService {\n constructor(httpService, assetRoot = '') {\n this.httpService = httpService;\n this.assetRoot = assetRoot;\n /** Can be both the font face name as well as the font file URL */\n this.readyFonts = new Map();\n }\n loadFont(stylesheetUrl, fontFamilies) {\n const fontFaceSet = document.fonts;\n if (typeof fontFaceSet === 'undefined') {\n return Promise.reject('FontFaceSet api not available');\n }\n return this.loadStylesheet(stylesheetUrl).then(() => this.loadFontFamily(fontFaceSet, fontFamilies));\n }\n loadFontByFile(path) {\n const absPath = isAbsoluteUrl(path) ? path : this.getAbsoluteFontPath(path);\n let loadPromise;\n if (fontFilesLoading.has(absPath)) {\n loadPromise = fontFilesLoading.get(absPath);\n }\n else {\n const promise = this.httpService.getPublicStatic(absPath, 'arraybuffer');\n fontFilesLoading.set(absPath, promise);\n loadPromise = promise;\n }\n return loadPromise.then(buffer => {\n if (buffer.byteLength === 0) {\n return Promise.reject('Could not fetch font');\n }\n else {\n this.readyFonts.set(path, buffer);\n }\n return buffer;\n });\n }\n loadFontData(stylesheetUrl, fontFamilies) {\n return this.loadStylesheet(stylesheetUrl).then(() => this.getFontData(fontFamilies));\n }\n getAbsoluteFontPath(relativePath) {\n return `${this.assetRoot}/media/${relativePath}`; //it's a relative path to the media folder\n }\n loadStylesheet(stylesheetUrl) {\n let stylesheetLoad = stylesheetsLoading.get(stylesheetUrl);\n if (!stylesheetLoad) {\n stylesheetLoad = this.httpService.getPublicStatic(stylesheetUrl).then(res => {\n // Parse the css and store the srcs to the font files\n const fontFaces = res.matchAll(/@font-face\\s*{([\\s\\S]*?)}/gm);\n for (const face of fontFaces) {\n const fontFamilyMatch = face[1].match(/font-family\\s*:\\s*['\"]?(.*?)['\"]?[;,\\n]/m);\n if (fontFamilyMatch) {\n const fontFamilyName = fontFamilyMatch[1];\n const srcMatch = face[1].match(/src\\s*:.*?url\\s*\\([\\s\"']*(.*?)[\\s\"']*\\)/m);\n if (srcMatch) {\n const fontSrc = srcMatch[1];\n fontFamilySrcs.set(fontFamilyName.trim().toLowerCase(), fontSrc.trim());\n }\n }\n }\n });\n stylesheetsLoading.set(stylesheetUrl, stylesheetLoad);\n }\n return stylesheetLoad;\n }\n getFontData(fontFamilies) {\n const loadFaces = [];\n const loadPromises = [];\n fontFamilies.forEach(fontFamily => {\n const fontSrc = fontFamilySrcs.get(fontFamily.trim().toLowerCase());\n if (!fontSrc) {\n console.warn(`Font family ${fontFamily} not found. Is it being loaded by the preloaded stylesheet?`);\n }\n else if (fontFilesLoading.has(fontFamily)) {\n loadPromises.push(fontFilesLoading.get(fontFamily));\n loadFaces.push(fontFamily);\n }\n else {\n const promise = this.httpService.getPublicStatic(fontSrc, 'arraybuffer');\n fontFilesLoading.set(fontFamily, promise);\n loadPromises.push(promise);\n loadFaces.push(fontFamily);\n }\n });\n const resMap = {};\n return Promise.all(loadPromises).then(buffers => {\n buffers.forEach((buffer, index) => {\n resMap[loadFaces[index]] = buffer;\n this.readyFonts.set(loadFaces[index], buffer);\n });\n return resMap;\n });\n }\n loadFontFamily(fontFaceSet, fontFamilies) {\n return this.getFontData(fontFamilies).then(resultData => {\n const resultFonts = [];\n fontFamilies.forEach(fontFamily => {\n if (resultData[fontFamily]) {\n const newFontFace = new FontFace(fontFamily, resultData[fontFamily]);\n // @ts-expect-error TS 5.1.6 definition does not include add on FontFaceSet\n fontFaceSet.add(newFontFace);\n resultFonts.push(newFontFace);\n }\n else {\n console.warn('Could not load font ' + fontFamily);\n }\n });\n return resultFonts;\n });\n }\n }\n\n (function (eContentType) {\n eContentType[\"json\"] = \"application/json\";\n eContentType[\"formData\"] = \"multipart/form-data\";\n })(exports.eContentType || (exports.eContentType = {}));\n const ABORTED = 'ABORTED';\n class HttpService {\n constructor(environment) {\n this.environment = environment;\n this.trackedAssetCalls = new Array();\n this.deferredAssetCalls = new Array();\n this.httpCallsToAbort = new Map();\n this.deferredExecuted = false;\n }\n addCacheParams(url) {\n return this.getUrlWithParams(url, this.getEnvironmentCacheParams());\n }\n get(url, params, trackAssetCall = true) {\n if (!isAbsoluteUrl(url) && this.environment) {\n // api calls in the test and prod environments should send the last deploy,\n // which allows the browser cache to store results for some calls\n if (this.environment) {\n const addParams = this.getEnvironmentCacheParams();\n params = params ? { ...params, ...addParams } : addParams;\n }\n }\n const u = this.getUrlWithParams(url, params);\n const p = this.fetchWrapper(u, 'GET');\n if (trackAssetCall) {\n this.trackAssetCall(p);\n }\n return p;\n }\n getPublicStatic(url, type = 'text', params, trackAssetCall = true) {\n const u = this.getUrlWithParams(url, params);\n const requestInit = {\n method: 'GET',\n credentials: 'omit',\n };\n const p = fetch(u, requestInit).then(res => {\n if (type === 'text') {\n return res.text();\n }\n else if (type === 'blob') {\n return res.blob();\n }\n else if (type === 'arraybuffer') {\n return res.arrayBuffer();\n }\n else {\n return res.json();\n }\n });\n if (trackAssetCall) {\n this.trackAssetCall(p);\n }\n return p;\n }\n post(url, body, contentType = exports.eContentType.json) {\n return this.fetchWrapper(url, 'POST', body, contentType);\n }\n put(url, body, contentType = exports.eContentType.json) {\n return this.fetchWrapper(url, 'PUT', body, contentType);\n }\n delete(url, params) {\n const u = this.getUrlWithParams(url, params);\n return this.fetchWrapper(u, 'DELETE');\n }\n getBinary(url, responseType, trackAssetCall, appendCacheParameters = false, abortSubject) {\n const xhr = new XMLHttpRequest();\n if (appendCacheParameters && !isAbsoluteUrl(url) && this.environment) {\n if (this.environment) {\n const addParams = this.getEnvironmentCacheParams();\n url = this.getUrlWithParams(url, addParams);\n }\n }\n const abortSub = abortSubject === null || abortSubject === void 0 ? void 0 : abortSubject.subscribe(() => {\n xhr.abort();\n });\n const p = new Promise((resolve, reject) => {\n xhr.open('GET', url, true);\n xhr.responseType = responseType;\n xhr.onload = function (oEvent) {\n if (xhr.status >= 200 && xhr.status < 300) {\n const allHeaders = xhr.getAllResponseHeaders();\n let height = null, width = null;\n if (allHeaders.match(/Original-Height/i)) {\n height = xhr.getResponseHeader('X-Original-Height');\n width = xhr.getResponseHeader('X-Original-Width');\n }\n const response = {\n data: xhr.response,\n };\n if (height && width) {\n response.height = parseInt(height);\n response.width = parseInt(width);\n }\n resolve(response);\n }\n else {\n reject(xhr.statusText);\n }\n abortSub === null || abortSub === void 0 ? void 0 : abortSub.unsubscribe();\n };\n xhr.onerror = e => {\n reject(e);\n abortSub === null || abortSub === void 0 ? void 0 : abortSub.unsubscribe();\n };\n xhr.onabort = () => {\n reject(ABORTED);\n // do nothing\n };\n xhr.send(null);\n });\n if (trackAssetCall) {\n this.trackAssetCall(p);\n }\n return p;\n }\n /**\n * delays execution of parameter function until all blocking assets have been loaded for the scene.\n * executes immediately if scene is already loaded.\n */\n defer(func, priority = 100) {\n if (this.deferredExecuted) {\n return func();\n }\n else {\n return new Promise((resolve, reject) => {\n this.deferredAssetCalls.push({ func, resolve, reject, priority });\n });\n }\n }\n executeDeferredCalls() {\n this.deferredAssetCalls.sort((a, b) => a.priority - b.priority);\n for (const call of this.deferredAssetCalls) {\n call.func().then(call.resolve).catch(call.reject);\n }\n this.deferredAssetCalls = [];\n this.deferredExecuted = true;\n }\n trackAssetCall(p) {\n const trackedPromise = p.finally(() => {\n this.trackedAssetCalls.remove(trackedPromise);\n });\n this.trackedAssetCalls.push(trackedPromise);\n }\n getEnvironmentCacheParams() {\n if (this.environment) {\n const params = {};\n if (this.environment.lastDeploy) {\n params.d = this.environment.lastDeploy.toString();\n }\n if (this.environment.cacheBuster) {\n params.v = this.environment.cacheBuster;\n }\n return params;\n }\n return {};\n }\n getUrlWithParams(url, params) {\n if (!params)\n return url;\n const urlParams = new URLSearchParams(Object.entries(params));\n let prefix = '?';\n if (url.match(/\\?/)) {\n prefix = '&';\n }\n return `${url}${prefix}${urlParams.toString()}`;\n }\n fetchWrapper(url, method, body, contentType = exports.eContentType.json) {\n const headers = new Headers();\n const requestInit = {\n method,\n headers,\n credentials: 'include',\n };\n if (method.equalsAny('POST', 'PUT')) {\n if (contentType == exports.eContentType.json) {\n headers.append('Content-Type', contentType);\n requestInit.body = JSON.stringify(body);\n }\n else if (contentType == exports.eContentType.formData) {\n requestInit.body = body;\n }\n }\n const p = fetch(url, requestInit).then(r => {\n if (r.ok) {\n return r.json().catch(() => body);\n }\n else {\n throw new Error(JSON.stringify(r));\n }\n });\n return p;\n }\n }\n\n const MOBILE_WIDTH = 847;\n function isMobile() {\n return window != null && window.innerWidth < MOBILE_WIDTH;\n }\n\n function getRotatedDimensions(width, height, rAngle) {\n const rad = (rAngle * Math.PI) / 180;\n return {\n width: width * Math.cos(rad) + height * Math.sin(rad),\n height: width * Math.sin(rad) + height * Math.cos(rad),\n };\n }\n function scaleImagePreserveAspect(targetWidth, targetHeight, naturalWidth, naturalHeight) {\n let drawWidth, drawHeight;\n const imageWidth = naturalWidth || 256;\n const imageHeight = naturalHeight || 256;\n if (imageWidth > imageHeight) {\n drawWidth = targetWidth;\n drawHeight = (imageHeight / imageWidth) * targetWidth;\n }\n else {\n drawWidth = (imageWidth / imageHeight) * targetHeight;\n drawHeight = targetHeight;\n }\n return {\n drawWidth: Math.ceil(drawWidth),\n drawHeight: Math.ceil(drawHeight),\n };\n }\n function getBinClampedSize(width, height, rounding = 'floor') {\n if (rounding === 'floor') {\n const log2 = Math.min(Math.log2(width), Math.log2(height));\n return Math.pow(2, Math.floor(log2));\n }\n else {\n const log2 = Math.max(Math.log2(width), Math.log2(height));\n return Math.pow(2, Math.ceil(log2));\n }\n }\n\n const KBMAXPREFIX = '____kbm____';\n const LOCAL_ENVIRONMENT_DOMAIN = 'library://';\n\n const SLOW_NETWORK_THRESHOLD = 100; // kB per second\n let slowWarningThrown = false;\n function attachLoadingTimer(p) {\n const startTime = performance.now();\n p.then(result => {\n if (slowWarningThrown) {\n return;\n }\n let fileSize = 0;\n const endTime = performance.now();\n if ('byteLength' in result.data) {\n fileSize = result.data.byteLength;\n }\n else if ('size' in result.data) {\n fileSize = result.data.size;\n }\n if (startTime && endTime && fileSize > 50000) {\n const resultTime = endTime - startTime; // in milliseconds\n // if download speed is less than 100 kB per second, throw slow loading warning\n if (fileSize / resultTime < SLOW_NETWORK_THRESHOLD) {\n console.warn('Slow network detected. Scene experience may be degraded.');\n slowWarningThrown = true;\n }\n }\n }).catch(() => {\n // Do nothing\n });\n return p;\n }\n\n const LIBRARY_PATH = '%LIBRARY%';\n // If this is a raster image, we store the bitmap. If this is an SVG, then we store the blob so it can be resized efficiently\n const textureCache = new Map();\n const materialCache = new Map();\n const TEXTURE_SIZES = [128, 256, 512, 1024, 2048, null];\n const LIBRARY_TEXTURE_SIZES = [256, 256, 512, 1024, 2048, 2048];\n const LIBRARY_ENV_SIZES = [128, 256, 512, 1024];\n const LIBRARY_MOBILE_ENV_SIZES = [128, 128, 256, 512];\n const MAX_DETAIL_LEVEL = 5;\n exports.MaterialService = class MaterialService {\n constructor(httpService, assetRoot, forceFullSizeTextures = false) {\n this.httpService = httpService;\n this.assetRoot = assetRoot;\n this.forceFullSizeTextures = forceFullSizeTextures;\n this.trackedAssetCalls = new Array();\n this.materialUpdateChannel = new Map();\n this.abortSubjects = new Map();\n this.textureDetailLevel = MAX_DETAIL_LEVEL;\n if (forceFullSizeTextures) {\n this.textureDetailLevel = -1;\n }\n else if (window.screen.width <= 1024 || window.screen.height <= 1024) {\n this.textureDetailLevel = 2;\n }\n }\n getMaterialLibraryCategories() {\n const categoryIds = Object.keys(materialDefinitionSet);\n return categoryIds.map(id => ({\n id,\n label: materialDefinitionSet[id].label,\n }));\n }\n getMaterialLibraryOptions(category) {\n return LibraryMaterialsByCategory.get(category) || [];\n }\n getMaterialById(guid) {\n if (materialCache.has(guid)) {\n return materialCache.get(guid);\n }\n else {\n let obs;\n const material = LibraryMaterials.get(guid);\n if (material) {\n //it's a library material\n obs = of(material\n //asapScheduler\n );\n }\n else {\n //maybe a global material\n obs = from(this.httpService.get(`/api/materials/${guid}`));\n }\n const updateChannel = new BehaviorSubject(null);\n this.materialUpdateChannel.set(guid, updateChannel);\n obs = combineLatest([obs, updateChannel])\n .pipe(map(([material, updatedMaterial]) => {\n if (!material) {\n return undefined;\n }\n return { ...material, ...(updatedMaterial || {}) };\n }))\n .pipe(shareReplay(1));\n materialCache.set(guid, obs);\n return obs;\n }\n }\n getTexturePath(node, detailLevel) {\n let width = null;\n let height = null;\n if (!node.path) {\n return '';\n }\n if (detailLevel === undefined) {\n detailLevel = this.textureDetailLevel;\n }\n detailLevel = Math.min(MAX_DETAIL_LEVEL, detailLevel);\n if (node.path.startsWith(LIBRARY_PATH)) {\n return this.getLibraryTexturePath(node.path, detailLevel);\n }\n else if (!isAbsoluteUrl(node.path)) {\n const url = this.getAbsoluteTexturePath(node.path);\n if (detailLevel >= 0) {\n width = TEXTURE_SIZES[detailLevel];\n }\n if (detailLevel >= 0) {\n height = TEXTURE_SIZES[detailLevel];\n }\n const params = [];\n if (width != null)\n params.push(`width=${width}`);\n if (height != null)\n params.push(`height=${height}`);\n if (params.length > 0) {\n return url + '?' + params.join('&');\n }\n else {\n return url;\n }\n }\n else {\n return node.path;\n }\n }\n getTextureImage(node, trackAssetCall = true) {\n const initialUrl = this.getTexturePath(node, 0);\n const url = this.getTexturePath(node);\n let bitmapObs;\n if (textureCache.has(url)) {\n bitmapObs = textureCache.get(url);\n }\n else {\n bitmapObs = new Observable(subject => {\n const activeLinkAbort = this.abortSubjects.get(node._dynamicId);\n // if there already is an httpService call on this node. Abort the previous one\n if (activeLinkAbort) {\n activeLinkAbort.next();\n }\n const abortTextureLoad = new Subject();\n this.abortSubjects.set(node._dynamicId, abortTextureLoad);\n // Textures are absolute so we have to add the cache params before passing to http service\n // With this code library textures are always progressively loaded regardless of the scene setting because they do not have a root scene\n // revisit this if we want to change this behavior but it is non-trivial to do so\n if (this.forceFullSizeTextures || (node._rootScene && !node._rootScene.progressiveTextureLoad)) {\n const cacheParamUrl = this.assetRoot && url.startsWith(this.assetRoot) ? this.httpService.addCacheParams(url) : url;\n attachLoadingTimer(this.httpService.getBinary(cacheParamUrl, 'blob', false, true, abortTextureLoad)).then(blobResult => {\n // clear the abort once the full image is loaded\n const activeLinkAbort = this.abortSubjects.get(node._dynamicId);\n if (activeLinkAbort) {\n activeLinkAbort.complete();\n this.abortSubjects.delete(node._dynamicId);\n }\n processImageBlob(blobResult).then(result => {\n if (result) {\n subject.next({ bitmap: result, url: cacheParamUrl });\n subject.complete();\n }\n });\n }, err => {\n if (err !== ABORTED) {\n console.warn(`Could not load texture ${cacheParamUrl}: ` + err.toString());\n }\n subject.complete();\n });\n }\n else {\n const initialCacheParamUrl = this.assetRoot && url.startsWith(this.assetRoot)\n ? this.httpService.addCacheParams(initialUrl)\n : initialUrl;\n const loadStartTime = performance.now();\n attachLoadingTimer(this.httpService.getBinary(initialCacheParamUrl, 'blob', false, true, abortTextureLoad)).then(initialBlobResult => {\n processImageBlob(initialBlobResult).then(result => {\n const loadTime = performance.now() - loadStartTime;\n if (loadTime > 2000) {\n console.warn(`Texture ${initialCacheParamUrl} took ${loadTime}ms to load`);\n }\n if (result) {\n subject.next({ bitmap: result, url: initialCacheParamUrl });\n }\n const cacheParamUrl = this.assetRoot && url.startsWith(this.assetRoot)\n ? this.httpService.addCacheParams(url)\n : url;\n this.httpService\n .defer(() => this.httpService.getBinary(cacheParamUrl, 'blob', false, true, abortTextureLoad))\n .then(blobResult => {\n // clear the abort once the full image is loaded\n const activeLinkAbort = this.abortSubjects.get(node._dynamicId);\n if (activeLinkAbort) {\n activeLinkAbort.complete();\n this.abortSubjects.delete(node._dynamicId);\n }\n processImageBlob(blobResult).then(result => {\n if (result) {\n subject.next({ bitmap: result, url: cacheParamUrl });\n subject.complete();\n }\n });\n }, err => {\n if (err !== ABORTED) {\n console.warn(`Could not load texture ${cacheParamUrl}: ` + err.toString());\n }\n subject.complete();\n });\n });\n }, err => {\n if (err !== ABORTED) {\n console.warn(`Could not load texture ${initialCacheParamUrl}: ` + err.toString());\n }\n subject.complete();\n });\n }\n }).pipe(shareReplay(1));\n }\n function processImageBlob(blob) {\n if (blob && blob.data) {\n if (blob.data.type === 'image/svg+xml') {\n return Promise.resolve(blob.data);\n }\n else {\n if (blob.data.type.match(/image/)) {\n const opts = blob.width && blob.height\n ? {\n resizeWidth: blob.width,\n resizeHeight: blob.height,\n resizeQuality: 'medium',\n }\n : undefined;\n return createImageBitmap(blob.data, opts).then(bitmap => {\n return Promise.resolve(bitmap);\n });\n }\n else {\n console.warn('source file is not an image: ', url);\n }\n }\n }\n return Promise.resolve(null);\n }\n textureCache.set(url, bitmapObs);\n const obs = bitmapObs.pipe(switchMap(res => {\n /** bitmap is returned as blob for SVGs and imageBitmap for raster images */\n if (res.bitmap instanceof Blob) {\n const blobToConvert = res.bitmap;\n return new Promise((resolve, reject) => {\n const imageElement = document.createElement('img');\n const blobUrl = URL.createObjectURL(blobToConvert);\n imageElement.onload = () => {\n // Figure out what resolution to actually draw the SVG at to fit within the envelope\n const scaled = scaleImagePreserveAspect(node.width, node.height, imageElement.naturalWidth, imageElement.naturalHeight);\n createImageBitmap(imageElement, {\n resizeWidth: scaled.drawWidth,\n resizeHeight: scaled.drawHeight,\n }).then(bitmap => {\n resolve({ bitmap: bitmap, url: res.url });\n });\n };\n imageElement.onerror = e => {\n reject(e);\n };\n imageElement.src = blobUrl;\n });\n }\n else {\n return of({ bitmap: res.bitmap, url: res.url });\n }\n }));\n if (trackAssetCall) {\n this.trackAssetCall(obs.pipe(take(1)).toPromise());\n }\n return obs;\n }\n getLibraryTexturePath(relativePath, quality = 0) {\n const path = relativePath.substring(LIBRARY_PATH.length);\n let res;\n if (quality < 0) {\n res = LIBRARY_TEXTURE_SIZES[LIBRARY_TEXTURE_SIZES.length - 1];\n }\n else {\n res = LIBRARY_TEXTURE_SIZES[quality];\n }\n return `${this.assetRoot}/assets/textures/${path}_${res}.jpg`;\n }\n getAbsoluteTexturePath(relativePath) {\n return `${this.assetRoot}/media/${relativePath}`; //it's a relative path to the media folder\n }\n getAbsoluteEnvironmentPath(path, quality) {\n if (path) {\n let url;\n if (path.startsWith(LOCAL_ENVIRONMENT_DOMAIN)) {\n const res = isMobile() ? LIBRARY_MOBILE_ENV_SIZES[quality] : LIBRARY_ENV_SIZES[quality];\n url = `${this.assetRoot}/assets/environments/${path.replace(LOCAL_ENVIRONMENT_DOMAIN, '')}_${res}.env`;\n }\n else {\n url = this.getAbsoluteTexturePath(path);\n }\n return this.httpService.addCacheParams(url);\n }\n return '';\n }\n searchMaterialByName(name, loadArchivedMaterials) {\n var _a;\n const archivedFilter = { type: 'property', property: 'archived', values: [false] };\n const searchArgs = { query: name + '*', sortField: 'Name', filters: [], descending: false };\n if (!loadArchivedMaterials) {\n (_a = searchArgs.filters) === null || _a === void 0 ? void 0 : _a.push(archivedFilter);\n }\n let result = null;\n if (name) {\n result = this.httpService.post('/api/materials/search', searchArgs);\n }\n return result;\n }\n getMaterialNameList() {\n const archivedFilter = { type: 'property', property: 'archived', values: [false] };\n const searchArgs = { fields: ['Guid', 'Name'], take: 10000, filters: [archivedFilter] };\n return this.httpService.post('/api/materials/search', searchArgs);\n }\n setMaterialCache(guid, mat) {\n materialCache.set(guid, of(mat));\n }\n trackAssetCall(p) {\n const trackedPromise = p.finally(() => {\n this.trackedAssetCalls.remove(trackedPromise);\n });\n this.trackedAssetCalls.push(trackedPromise);\n }\n };\n exports.MaterialService = __decorate([\n Kb3dService({\n token: Token.MaterialService,\n })\n ], exports.MaterialService);\n const LibraryMaterials = new Map();\n const LibraryMaterialsByCategory = new Map();\n const LibraryEnvironments = new Array();\n const LibraryEnvironmentsMap = new Map();\n function makeLibraryMaterial(tag, index) {\n //pad the index string with zeros\n let indexStr = index + '';\n while (indexStr.length < 2)\n indexStr = '0' + indexStr;\n const iname = tag + indexStr;\n const baseUrl = `${tag}/${iname}`;\n return {\n id: -1,\n name: iname,\n guid: iname,\n tags: [{ name: tag }],\n data: {\n $type: Token.AdvancedMaterialNode.key,\n name: iname,\n id: iname,\n location: exports.eMaterialLocation.library,\n textures: [\n {\n $type: 'FileTextureLayer',\n channel: exports.eTextureChannel.ambientOcclusion,\n path: `${LIBRARY_PATH}${baseUrl}_ao`,\n },\n {\n $type: 'FileTextureLayer',\n channel: exports.eTextureChannel.albedo,\n path: `${LIBRARY_PATH}${baseUrl}_col`,\n },\n {\n $type: 'FileTextureLayer',\n channel: exports.eTextureChannel.metallic,\n path: `${LIBRARY_PATH}${baseUrl}_met`,\n },\n {\n $type: 'FileTextureLayer',\n channel: exports.eTextureChannel.normal,\n path: `${LIBRARY_PATH}${baseUrl}_nrm`,\n },\n {\n $type: 'FileTextureLayer',\n channel: exports.eTextureChannel.roughness,\n path: `${LIBRARY_PATH}${baseUrl}_rgh`,\n },\n //{ $type: \"FileTextureLayer\", channel: eTextureChannel.displacement, path: `${LIBRARY_PATH}${baseUrl}_disp` },\n ],\n },\n };\n }\n function fillLibraryMaterialsMap() {\n for (const tag in materialDefinitionSet) {\n const entry = materialDefinitionSet[tag];\n for (let index = 1; index <= entry.count; index++) {\n const mat = makeLibraryMaterial(tag, index);\n const matData = mat.data;\n matData.roughness = 1;\n matData.metallic = 0;\n if (entry.omit) {\n const omit = entry.omit;\n matData.textures = matData.textures.filter(t => {\n if (omit[t.channel]) {\n if (omit[t.channel] === true) {\n return false;\n }\n if (Array.isArray(omit[t.channel]) && omit[t.channel].some(i => i === index)) {\n return false;\n }\n }\n return true;\n });\n }\n // Metallic is multiplied against the texture, so if the material has a metallic texture then set metallic to 1.\n if (matData.textures.findIndex(t => t.channel === exports.eTextureChannel.metallic) > -1) {\n matData.metallic = 1;\n }\n if (entry.props) {\n for (const p in entry.props) {\n mat.data[p] = entry.props[p];\n }\n }\n LibraryMaterials.set(mat.guid, mat);\n let categoryList = LibraryMaterialsByCategory.get(tag);\n if (!categoryList) {\n categoryList = [];\n }\n categoryList.push(mat);\n LibraryMaterialsByCategory.set(tag, categoryList);\n }\n }\n }\n const materialDefinitionSet = {\n asphalt: {\n label: 'Asphalt',\n count: 15,\n omit: {\n [exports.eTextureChannel.metallic]: true,\n [exports.eTextureChannel.ambientOcclusion]: [1, 2, 9, 10, 11, 12, 13, 14, 15],\n },\n },\n bark: {\n label: 'Bark',\n count: 6,\n omit: {\n [exports.eTextureChannel.metallic]: true,\n [exports.eTextureChannel.ambientOcclusion]: [1],\n },\n },\n bricks: {\n label: 'Bricks',\n count: 17,\n omit: {\n [exports.eTextureChannel.metallic]: true,\n [exports.eTextureChannel.ambientOcclusion]: [1, 2, 3, 4],\n },\n },\n chipboard: {\n label: 'Chipboard',\n count: 3,\n omit: {\n [exports.eTextureChannel.metallic]: true,\n },\n },\n concrete: {\n label: 'Concrete',\n count: 17,\n omit: {\n [exports.eTextureChannel.metallic]: true,\n [exports.eTextureChannel.ambientOcclusion]: [1, 2, 12, 14, 15, 16, 17],\n [exports.eTextureChannel.roughness]: [1],\n },\n },\n corrugatedsteel: {\n label: 'Corrugated Steel',\n count: 5,\n },\n diamondplate: {\n label: 'Diamond Plate',\n count: 4,\n omit: {\n [exports.eTextureChannel.ambientOcclusion]: true,\n [exports.eTextureChannel.albedo]: [4],\n },\n },\n fabric: {\n label: 'Fabric',\n count: 29,\n omit: {\n [exports.eTextureChannel.ambientOcclusion]: true,\n [exports.eTextureChannel.metallic]: true,\n },\n },\n gravel: {\n label: 'Gravel',\n count: 8,\n omit: {\n [exports.eTextureChannel.metallic]: true,\n [exports.eTextureChannel.ambientOcclusion]: [1, 2],\n },\n },\n ground: {\n label: 'Ground',\n count: 28,\n omit: {\n [exports.eTextureChannel.metallic]: true,\n [exports.eTextureChannel.ambientOcclusion]: [1, 2, 3, 17],\n },\n },\n ice: {\n label: 'Ice',\n count: 4,\n omit: {\n [exports.eTextureChannel.metallic]: true,\n [exports.eTextureChannel.ambientOcclusion]: true,\n },\n },\n leather: {\n label: 'Leather',\n count: 14,\n omit: {\n [exports.eTextureChannel.metallic]: true,\n [exports.eTextureChannel.ambientOcclusion]: [1, 2, 3, 4, 5, 6, 7, 9, 10, 11, 12, 13, 14],\n },\n },\n marble: {\n label: 'Marble',\n count: 5,\n omit: {\n [exports.eTextureChannel.metallic]: true,\n [exports.eTextureChannel.ambientOcclusion]: true,\n },\n },\n metal: {\n label: 'Metal',\n count: 27,\n omit: {\n [exports.eTextureChannel.ambientOcclusion]: true,\n },\n },\n metalplates: {\n label: 'Metal Plate',\n count: 3,\n omit: {\n [exports.eTextureChannel.ambientOcclusion]: true,\n },\n },\n paint: {\n label: 'Paint',\n count: 6,\n omit: {\n [exports.eTextureChannel.metallic]: true,\n },\n },\n paintedmetal: {\n label: 'Painted Metal',\n count: 5,\n omit: {\n [exports.eTextureChannel.ambientOcclusion]: true,\n },\n },\n paintedplaster: {\n label: 'Painted Plaster',\n count: 5,\n omit: {\n [exports.eTextureChannel.ambientOcclusion]: true,\n [exports.eTextureChannel.metallic]: true,\n },\n },\n pavingstones: {\n label: 'Paving Stones',\n count: 41,\n omit: {\n [exports.eTextureChannel.metallic]: true,\n [exports.eTextureChannel.ambientOcclusion]: [1, 3, 4],\n },\n },\n pipe: {\n label: 'Pipe',\n count: 4,\n omit: {\n [exports.eTextureChannel.ambientOcclusion]: true,\n },\n },\n planks: {\n label: 'Planks',\n count: 9,\n omit: {\n [exports.eTextureChannel.ambientOcclusion]: [1, 2],\n },\n },\n plastic: {\n label: 'Plastic',\n count: 5,\n omit: {\n [exports.eTextureChannel.metallic]: true,\n [exports.eTextureChannel.ambientOcclusion]: true,\n },\n },\n rock: {\n label: 'Rock',\n count: 14,\n omit: {\n [exports.eTextureChannel.metallic]: true,\n [exports.eTextureChannel.ambientOcclusion]: [1, 6, 7, 8, 9],\n [exports.eTextureChannel.albedo]: [4],\n },\n },\n rocks: {\n label: 'Rocks',\n count: 6,\n omit: {\n [exports.eTextureChannel.metallic]: true,\n },\n },\n roofingtiles: {\n label: 'Roofing Tiles',\n count: 5,\n omit: {\n [exports.eTextureChannel.metallic]: true,\n [exports.eTextureChannel.ambientOcclusion]: true,\n },\n },\n rust: {\n label: 'Rust',\n count: 5,\n omit: {\n [exports.eTextureChannel.ambientOcclusion]: true,\n },\n },\n scratches: {\n label: 'Scratches',\n count: 5,\n omit: {\n [exports.eTextureChannel.ambientOcclusion]: true,\n [exports.eTextureChannel.metallic]: true,\n [exports.eTextureChannel.albedo]: true,\n },\n },\n terrazzo: {\n label: 'Terrazzo',\n count: 8,\n omit: {\n [exports.eTextureChannel.ambientOcclusion]: true,\n [exports.eTextureChannel.metallic]: true,\n },\n },\n tiles: {\n label: 'Tiles',\n count: 37,\n omit: {\n [exports.eTextureChannel.metallic]: true,\n [exports.eTextureChannel.ambientOcclusion]: [\n 1, 2, 3, 4, 5, 12, 13, 14, 15, 20, 21, 22, 23, 24, 25, 29, 30, 31, 32, 33, 34, 35, 36, 37,\n ],\n },\n },\n wood: {\n label: 'Wood',\n count: 20,\n omit: {\n [exports.eTextureChannel.metallic]: true,\n [exports.eTextureChannel.ambientOcclusion]: [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 12, 13, 14, 15, 16, 17, 18, 19, 20],\n },\n },\n woodfloor: {\n label: 'Wood Floor',\n count: 28,\n omit: {\n [exports.eTextureChannel.metallic]: true,\n [exports.eTextureChannel.ambientOcclusion]: [1, 2, 4, 5, 6, 8],\n },\n },\n woodsiding: {\n label: 'Wood Siding',\n count: 8,\n omit: {\n [exports.eTextureChannel.metallic]: [5, 6, 7],\n },\n },\n };\n fillLibraryMaterialsMap();\n /** create library environments */\n for (const env of [\n // \"country\",\n // \"environment\",\n // \"forest\",\n // \"night\",\n // \"parking\",\n // \"room\",\n 'adams_place_bridge',\n 'aerodynamics_workshop',\n 'autoshop',\n 'ballroom',\n 'boiler_room',\n 'cape_hill',\n 'colorful_studio',\n 'fireplace',\n 'greenwich_park',\n 'hospital_room',\n 'industrial_pipe_and_valve',\n 'leadenhall_market',\n 'machine_shop',\n 'modern_buildings_night',\n 'museum_of_ethnography',\n 'mutianyu',\n 'old_bus_depot',\n 'photo_studio',\n 'preller_drive',\n 'red_wall',\n 'rooitou_park',\n 'royal_esplanade',\n 'schadowplatz',\n 'small_cathedral',\n 'spruit_sunrise',\n 'studio_small',\n 'surgery',\n 'venice_sunset',\n 'winter_evening',\n 'wooden_lounge',\n ]) {\n const label = env\n .split('_')\n .map(s => s.capitalize())\n .join(' ');\n const o = {\n name: env,\n label,\n path: `${LOCAL_ENVIRONMENT_DOMAIN}${env}`,\n previewUrl: `environments/${env}.jpg`,\n };\n LibraryEnvironments.push(o);\n LibraryEnvironmentsMap.set(o.path, o);\n }\n\n const geometryCache = new Map();\n class MeshService {\n constructor(httpService, assetRoot) {\n this.httpService = httpService;\n this.assetRoot = assetRoot;\n }\n /**\n * Loads the geometry for a mesh node and adds it to the node, mutating the node object.\n * @param meshNode Mesh node to load geometry for\n */\n hydrateGeometry(meshNode) {\n if (!isCustomMeshNode(meshNode) || !meshNode.geometryId) {\n return Promise.resolve(meshNode);\n }\n else {\n let request = geometryCache.get(meshNode.geometryId);\n let shared = false;\n if (!request) {\n request = attachLoadingTimer(this.httpService.getBinary(`${this.assetRoot}/meshes/${meshNode.geometryId}`, 'arraybuffer', true)).then(response => {\n if (response && response.data) {\n const geometry = meshNode._manager.create(Token.KbGeometry);\n geometry.extractFromBinary(response.data, meshNode);\n return geometry;\n }\n return Promise.reject();\n }, err => {\n console.error(`Geometry for \"${meshNode.name}\" could not be loaded`, err);\n return null;\n });\n geometryCache.set(meshNode.geometryId, request);\n }\n else {\n shared = true;\n }\n return request.then(geometry => {\n if (geometry) {\n if (shared) {\n geometry.shared = true;\n }\n meshNode._geometry = geometry;\n }\n return meshNode;\n });\n }\n }\n }\n\n exports.LightNode = class LightNode extends exports.SpaceNode {\n constructor() {\n super();\n }\n getSpaceChildren() {\n return [];\n }\n };\n exports.LightNode = __decorate([\n BaseKb3dModel({\n token: Token.LightNode,\n })\n ], exports.LightNode);\n function isLightNode(node) {\n return instanceOfAny(node, ...Token.LightTokens);\n }\n\n exports.SceneNode = class SceneNode extends exports.MeshGroupNode {\n constructor() {\n super();\n this._forceGrid = false;\n }\n get materials() {\n if (!this._materials)\n this._materials = [];\n return this._materials;\n }\n set materials(val) {\n this._materials = val;\n }\n get lights() {\n if (!this._lights)\n this._lights = [];\n return this._lights;\n }\n set lights(val) {\n this._lights = val;\n }\n get viewpoints() {\n if (!this._viewpoints)\n this._viewpoints = [];\n return this._viewpoints;\n }\n set viewpoints(val) {\n this._viewpoints = val;\n }\n get scenes() {\n if (!this._scenes)\n this._scenes = [];\n return this._scenes;\n }\n set scenes(val) {\n this._scenes = val;\n }\n get mates() {\n if (!this._mates)\n this._mates = [];\n return this._mates;\n }\n set mates(val) {\n this._mates = val;\n }\n get annotations() {\n if (!this._annotations)\n this._annotations = [];\n return this._annotations;\n }\n set annotations(val) {\n this._annotations = val;\n }\n get importHistory() {\n if (!this._importHistory)\n this._importHistory = [];\n return this._importHistory;\n }\n set importHistory(val) {\n this._importHistory = val;\n }\n get exportHistory() {\n if (!this._exportHistory)\n this._exportHistory = [];\n return this._importHistory;\n }\n set exportHistory(val) {\n this._exportHistory = val;\n }\n getChildren() {\n return [\n ...this.meshes,\n ...this.materials,\n ...this.lights,\n ...this.viewpoints,\n ...this.mates,\n ...this.connectors,\n ...this.annotations,\n ];\n }\n getChildArrayForType(token) {\n if (token.isMeshOrGroup() || token.isLight()) {\n return this.meshes;\n }\n else if (token.isMaterial()) {\n return this.materials;\n }\n else if (token.isViewpoint()) {\n return this.viewpoints;\n }\n else if (token.isMate()) {\n return this.mates;\n }\n else if (token.isConnector()) {\n return this.connectors;\n }\n else if (token.isAnnotation()) {\n return this.annotations;\n }\n else {\n throw 'unexpected token';\n }\n }\n getChildArray(node) {\n if (Token.withKey(node.$type).isLight()) {\n // Lights can be defined as meshes but can be in the light array as well, so we have to check there\n if (this.lights.contains(node)) {\n return this.lights;\n }\n }\n return super.getChildArray(node);\n }\n isNested() {\n return this._parent != null;\n }\n runOnAllScenes(action) {\n action(this);\n for (const nested of this.scenes) {\n nested.runOnAllScenes(action);\n }\n }\n getSpaceChildren() {\n return this.meshes.concat(this.scenes);\n }\n getNestedPathTo(node) {\n let path = node.id;\n let parent = node._parentScene;\n let isPath = false;\n while (parent) {\n if (parent === this)\n break;\n path = parent.nestedId + '/' + path;\n isPath = true;\n parent = parent._parent;\n }\n return `${isPath ? '@' : ''}${path}`;\n }\n getAllConnectors() {\n const connectors = new Array();\n this.runOnAllScenes(s => {\n connectors.push(...s._manager.getByToken(Token.Connector).filter(c => c.enabled));\n });\n return connectors;\n }\n getAllSketchControlPoints() {\n const points = new Array();\n this.runOnAllScenes(s => {\n points.push(...s._manager.getByToken(Token.SketchControlPoint).filter(c => c.enabled));\n });\n return points;\n }\n };\n __decorate([\n IgnoreChanges\n ], exports.SceneNode.prototype, \"conversionMap\", void 0);\n __decorate([\n IgnoreChanges\n ], exports.SceneNode.prototype, \"_configurator\", void 0);\n __decorate([\n Enumerable()\n ], exports.SceneNode.prototype, \"materials\", null);\n __decorate([\n Enumerable()\n ], exports.SceneNode.prototype, \"lights\", null);\n __decorate([\n Enumerable()\n ], exports.SceneNode.prototype, \"viewpoints\", null);\n __decorate([\n Enumerable()\n ], exports.SceneNode.prototype, \"scenes\", null);\n __decorate([\n Enumerable()\n ], exports.SceneNode.prototype, \"mates\", null);\n __decorate([\n Enumerable()\n ], exports.SceneNode.prototype, \"annotations\", null);\n __decorate([\n Enumerable()\n ], exports.SceneNode.prototype, \"importHistory\", null);\n __decorate([\n Enumerable()\n ], exports.SceneNode.prototype, \"exportHistory\", null);\n exports.SceneNode = __decorate([\n Kb3dModel({\n token: Token.SceneNode,\n })\n ], exports.SceneNode);\n function isSceneNode(node) {\n return instanceOf(node, Token.SceneNode);\n }\n\n const SET_TYPE = '__set';\n /** contains all kb3dobjects keyed by their dynamic Id, to fetch nodes from the 3d engine ID */\n const dynamicDb = new Map();\n function getByDynamicId(dynamicId, throwOnNotFound = false) {\n if (throwOnNotFound && !dynamicDb.has(dynamicId)) {\n throw `kb3d object with id ${dynamicId} not found`;\n }\n return dynamicDb.get(dynamicId);\n }\n (function (ImportOperation) {\n ImportOperation[ImportOperation[\"ignore\"] = 0] = \"ignore\";\n ImportOperation[ImportOperation[\"add\"] = 1] = \"add\";\n ImportOperation[ImportOperation[\"replace\"] = 2] = \"replace\";\n })(exports.ImportOperation || (exports.ImportOperation = {}));\n class Kb3dManager {\n constructor(options) {\n this.isDevEnvironment = true;\n /** contains all kb3dobjects */\n this._db = new Map();\n /** contains all objects keyed by their token */\n this._tokenDb = new SetMap();\n /** contains all nodes (but not other kinds of objects) */\n this._nodeDb = new Map();\n /** holds a cache of name searches that have been performed with their results */\n this._nameSearchCache = new Map();\n /** contains nested scenes keyed by their nestedId */\n this._nestedSceneDb = new Map();\n /** holds a cache of click handlers that have been added to nodes in the context of this manager */\n this.clickHandlers = new HandlerMap(this);\n this.doubleClickHandlers = new HandlerMap(this);\n this.draggables = new HandlerMap(this);\n this.hotspotClickHandlers = new TagArrayMap();\n this.boundingboxChangeHandlers = new HandlerMap(this);\n //if no services are provided, then use the default services\n if (options && options.services) {\n this.services = options.services;\n }\n else {\n const httpService = new HttpService(options === null || options === void 0 ? void 0 : options.environment);\n if ((options === null || options === void 0 ? void 0 : options.environment) && options.environment.lastDeploy) {\n this.isDevEnvironment = false;\n }\n this.services = new Map([\n [Token.HttpService, httpService],\n [\n Token.MaterialService,\n new exports.MaterialService(httpService, options === null || options === void 0 ? void 0 : options.assetRoot, options === null || options === void 0 ? void 0 : options.forceFullSizeTextures),\n ],\n [Token.MeshService, new MeshService(httpService, options === null || options === void 0 ? void 0 : options.assetRoot)],\n [Token.FontService, new FontService(httpService, options === null || options === void 0 ? void 0 : options.assetRoot)],\n ]);\n }\n }\n getService(serviceToken) {\n return this.services.get(serviceToken);\n }\n dispose() {\n /**only clear items out of the dynamic db that were part of this manager (as it is static and holds dynamic ids for all managers) */\n for (const [, value] of this._db) {\n dynamicDb.delete(value._dynamicId);\n }\n this._db.clear();\n this._nodeDb.clear();\n this.clickHandlers.clear();\n this.doubleClickHandlers.clear();\n this.draggables.clear();\n this.hotspotClickHandlers.clear();\n this.boundingboxChangeHandlers.clear();\n this._nameSearchCache.clear();\n this._tokenDb.clear();\n this._nestedSceneDb.clear();\n }\n /**\n * Accepts either a node id or a name to search for nodes. If using\n * node name wildcards * are allowed. The action provided will be run\n * on each of the matching nodes, with the args passed in to that action.\n * @param idOrName\n * @param action\n * @param args\n * @param useCache\n */\n runOnNodes(idOrName, action, args, useCache = true) {\n if (isGuid(idOrName)) {\n const node = this.getById(idOrName);\n if (node)\n action(node, args);\n }\n else {\n const nodes = this.getNodesByName(idOrName, useCache);\n for (const node of nodes) {\n action(node, args);\n }\n }\n }\n // @child1/xxxxx-xxxxxx-xxxx\n getById(id) {\n if (id && id.startsWith('@')) {\n //it's a nested path id\n const segments = id.split('/');\n const nestedId = segments.shift().substring(1).toLowerCase();\n const nestedScene = this._nestedSceneDb.get(nestedId);\n if (nestedScene) {\n let subId = segments.join('/');\n if (segments.length > 1)\n subId = '@' + subId;\n return nestedScene._manager.getNodeByIdOrName(subId);\n }\n }\n else {\n return this._db.get(id);\n }\n }\n getAll() {\n return [...this._db.values()];\n }\n getAllNodes() {\n return [...this._nodeDb.values()];\n }\n /**\n * Accepts either a node id or a name to search for nodes. Only returns the first\n * matching node. If using node name wildcards * are allowed.\n * @param idOrName\n * @param useCache\n */\n getNodeByIdOrName(idOrName, useCache = true) {\n if (typeof idOrName == 'string') {\n if (isGuid(idOrName) || idOrName.startsWith('@')) {\n return this.getById(idOrName);\n }\n else {\n return this.getFirstNodeByName(idOrName, useCache);\n }\n }\n else if (idOrName instanceof exports.Kb3dObject) {\n return idOrName;\n }\n }\n /**\n * Accepts either a node id or a name to search for nodes. If using\n * node name wildcards * are allowed.\n * @param idOrName\n * @param useCache\n */\n getNodesByIdOrName(idOrName, useCache = true) {\n const res = [];\n if (typeof idOrName == 'string') {\n if (isGuid(idOrName)) {\n const n = this.getById(idOrName);\n if (n)\n res.push(n);\n }\n else {\n res.push(...this.getNodesByName(idOrName, useCache));\n }\n }\n else if (idOrName instanceof exports.Kb3dObject) {\n res.push(idOrName);\n }\n return res;\n }\n /**\n * Get's the first node that matches the name given. Wildcards * are allowed.\n * @param name\n * @param useCache\n */\n getFirstNodeByName(name, useCache = true) {\n const nodes = this.getNodesByName(name, useCache);\n if (nodes && nodes.length) {\n return nodes[0];\n }\n }\n /**\n * Returns a list of nodes that match the name given.\n * Wildcards * are allowed.\n * Results will be fed from a search cache if useCache is true\n * @param name\n * @param useCache\n */\n getNodesByName(name, useCache = true) {\n if (useCache && this._nameSearchCache.has(name)) {\n return this._nameSearchCache.get(name);\n }\n const results = new Array();\n const re = this.getWildcardRegex(name);\n for (const [, node] of this._db) {\n if (re.test(node.name)) {\n results.push(node);\n }\n }\n this._nameSearchCache.set(name, results);\n return results;\n }\n getNodesContainingName(name, useCache = true) {\n if (useCache && this._nameSearchCache.has(name)) {\n return this._nameSearchCache.get(name);\n }\n const results = new Array();\n const re = this.getWildcardRegex(name);\n for (const [, node] of this._db) {\n if (re.test(node.name)) {\n results.push(node);\n }\n }\n this._nameSearchCache.set(name, results);\n return results;\n }\n getByToken(token) {\n return [...this._tokenDb.get(token).values()];\n }\n getByTokens(...tokens) {\n return tokens.flatMap(t => [...this._tokenDb.get(t).values()]);\n }\n /** escapes the given string so that any special characters will be processed literally in a regex search */\n escapeRegex(str) {\n return str.replace(/([.*+?^=!:${}()|[\\]/\\\\])/g, '\\\\$1');\n }\n /**\n * Given a string with wildcards *, will provide a regex that can be used to\n * match strings.\n * @param rule\n */\n getWildcardRegex(rule) {\n // \".\" => Find a single character, except newline or line terminator\n // \".*\" => Matches any string that contains zero or more characters\n rule = rule.split('*').map(this.escapeRegex).join('.*');\n // \"^\" => Matches any string with the following at the beginning of it\n // \"$\" => Matches any string with that in front at the end of it\n rule = '^' + rule + '$';\n //Create a regular expression object for matching string\n return new RegExp(rule, 'i');\n }\n /**\n * absorbs a kb3d object that already exists into the manager.\n * @param o\n */\n register(o) {\n if (o instanceof exports.Kb3dObject) {\n if (!this._db.has(o.id)) {\n if (isTrackedClass(o)) {\n o.isModified = true; //force a full round of change detection if this is a newly registered object\n }\n }\n this._db.set(o.id, o);\n dynamicDb.set(o._dynamicId, o);\n this._tokenDb.add(Token.withKey(o.$type), o);\n if (o.isNode()) {\n this._nodeDb.set(o.id, o);\n }\n }\n }\n unregister(o) {\n if (o instanceof exports.Kb3dObject) {\n this.clickHandlers.deleteByKey(o);\n this.doubleClickHandlers.deleteByKey(o);\n this.draggables.deleteByKey(o);\n this.boundingboxChangeHandlers.deleteByKey(o);\n this._db.delete(o.id);\n dynamicDb.delete(o._dynamicId);\n this._tokenDb.removeValue(Token.withKey(o.$type), o);\n if (o.isNode()) {\n this._nodeDb.delete(o.id);\n }\n if (o instanceof exports.SceneNode && o.isNested()) {\n this._nestedSceneDb.delete(o.nestedId);\n }\n }\n }\n /**\n * Creates an instance of a kb3d model, given a token\n * and an anonymous object in the shape of the model that\n * can be used to set properties on it.\n *\n * As part of the creation process, we do things in this order:\n *\n * 1. Create an instance of the class. If a create function was provided\n * in the decorator for the class, that is used to create the instance.\n * If not, then it's created by calling the constructor of the class\n * with no arguments.\n * 2. We then apply default values to the properties of the class\n * 3. We then apply the args that were passed in to set properties on the class\n * 4. We then register the object into the map using it's id so it can be quickly retrieved\n */\n create(token, args = {}, options) {\n var _a;\n options = { includeChildNodes: true, ...options };\n const modelDef = Kb3dModels.get(token);\n if (!modelDef) {\n throw `could not find model with token: ${token.key}`;\n }\n const constructorParameters = getConstructorParamsForModel(modelDef);\n const hydratedParameters = [];\n for (const i of constructorParameters) {\n if (!i.ignore) {\n const service = (_a = this.services) === null || _a === void 0 ? void 0 : _a.get(i.token);\n if (!service) {\n throw `Injected service of type '${i.token.key} could not be found to inject into the constructor of type '${token.key}`;\n }\n hydratedParameters.push(service);\n }\n }\n let model;\n if (modelDef.create) {\n model = modelDef.create(args);\n }\n else {\n model = new modelDef.classConstructor(...hydratedParameters);\n }\n //now initialize the model for when no create function was given\n if (!modelDef.create) {\n this.initializeModel(isTrackedClass(model) ? model.rawNode : model, args, modelDef, options);\n }\n if (model instanceof exports.SceneNode) {\n model._manager = this;\n model.calculateParents(); //because calculateParents also registers the item, we must call this after initialization, otherwise it won't have the right id\n }\n if (modelDef.afterCreate) {\n modelDef.afterCreate(model);\n }\n //need to clear the name cache since we just created a new node\n //no need to clear if the node is a primitive as they are never fetched by name\n if (!token.isPrimitive()) {\n this._nameSearchCache.clear();\n }\n return model;\n }\n serializeToJson(instance, options) {\n const o = this.serialize(instance, options);\n const json = JSON.stringify(o, null, (options === null || options === void 0 ? void 0 : options.prettyPrint) ? 2 : undefined);\n return json;\n }\n serialize(instance, options) {\n options = {\n includeChildNodes: true,\n isRoot: true,\n omitDefaultValues: true,\n ...options,\n };\n let result = { $type: instance.$type };\n const defaultValues = defaults[instance.$type];\n for (const key in instance) {\n if (\n // Never persist properties beginning with an underscore\n key.substr(0, 1) !== '_' &&\n // If a property is the same as a default value, remove it\n (!options.omitDefaultValues ||\n !defaultValues ||\n !this.shallowEquals(defaultValues[key], instance[key]))) {\n if (!isObject$1(instance[key])) {\n // For primitives, just copy the values over\n result[key] = instance[key];\n }\n else if (Array.isArray(instance[key])) {\n const sourceArray = instance[key];\n if (sourceArray.length && typeof sourceArray[0] !== 'object') {\n //it's an array of primitives\n result[key] = sourceArray.slice(0); //clone the array of primitives\n }\n else {\n result[key] = []; //make sure there is an array on target prop. Recreate it if one is there already.\n for (const sourceItem of sourceArray) {\n if (!sourceItem.$type ||\n !Token.withKey(sourceItem.$type).isNode() ||\n options.includeChildNodes) {\n // exclude it if it's a kb3d model and includeChildren is false\n const targetItem = this.serialize(sourceItem, { ...options, isRoot: false });\n if (targetItem) {\n result[key].push(targetItem);\n }\n }\n }\n }\n }\n else if (instance[key] instanceof Map) {\n console.error('implement serializing maps');\n }\n else if (instance[key] instanceof Set) {\n result[key] = {\n $type: SET_TYPE,\n values: Array.from(instance[key].values()),\n };\n }\n else {\n // For anything else, try deserializing. The deserializer will just copy the property if it is not a serialized class.\n result[key] = this.serialize(instance[key], { ...options, isRoot: false });\n }\n }\n }\n if (instance.$type) {\n const modelDef = Kb3dModels.get(Token.withKey(instance.$type));\n if (modelDef && modelDef.onSerializing) {\n const include = modelDef.onSerializing(instance, result);\n // Always serialize the object if it is the root object. Otherwise, let the class decide.\n if (include === false && !options.isRoot) {\n result = undefined;\n }\n }\n }\n return result;\n }\n /**\n * See notes for the create() function as this is virtually the same.\n * The only difference is that the type here is discovered through the $type\n * property of the args object instead of an explicit token.\n * @param object\n */\n deserialize(object, options) {\n if (!object || !object.$type) {\n return object;\n }\n else if (object.$type === SET_TYPE) {\n return new Set(object.values);\n }\n return this.create(Token.withKey(object.$type), object, options);\n }\n deserializeJson(json) {\n const object = JSON.parse(json);\n return this.deserialize(object);\n }\n clone(objectToClone) {\n //make sure it's a kb3d model\n if (!objectToClone)\n throw 'no object provided to clone';\n if (!('$type' in objectToClone)) {\n throw 'object passed to clone function must be a kb3d model and have a $type defined';\n }\n const json = this.serializeToJson(objectToClone);\n const object = JSON.parse(json);\n return this.deserialize(object, { issueNewIds: true });\n }\n nestScene(nestedId, parent, serializedNested = {}) {\n const nestedMan = new Kb3dManager({ services: this.services });\n if (serializedNested.lights) {\n serializedNested.lights.clear();\n }\n const nested = nestedMan.create(Token.SceneNode, serializedNested);\n nested.nestedId = nestedId.replace('/', '');\n parent.scenes.push(nested);\n this._nestedSceneDb.set(nested.nestedId.toLowerCase(), nested);\n return nested;\n }\n getNodeNamePathMap() {\n const meshMap = new Map();\n const matMap = new Map();\n const lightMap = new Map();\n const allNodes = this._nodeDb.values();\n for (const n of allNodes) {\n if (isLightNode(n)) {\n lightMap.set(n.getNamePath(), n);\n }\n else if (isMaterialNode(n)) {\n matMap.set(n.getNamePath(), n);\n }\n else {\n meshMap.set(n.getNamePath(), n);\n }\n }\n return {\n meshes: meshMap,\n lights: lightMap,\n materials: matMap,\n };\n }\n previewImport(parent, imported) {\n const sceneNode = parent._parentScene;\n if (!sceneNode || !imported.scene)\n return;\n const nMap = this.getNodeNamePathMap();\n const processNode = (node, parentNamePath) => {\n const token = Token.withKey(node['$type']);\n const namePath = token !== Token.SceneNode && node.name\n ? [parentNamePath, node.name].filter(Boolean).join('/')\n : parentNamePath;\n let matchingNode;\n if (token.isLight()) {\n matchingNode = nMap.lights.get(namePath);\n }\n else if (token.isMaterial()) {\n matchingNode = nMap.materials.get(namePath);\n }\n else {\n matchingNode = nMap.meshes.get(namePath);\n }\n const p = {\n node,\n type: Token.withKey(node['$type']),\n namePath,\n matchingNode,\n operation: matchingNode ? exports.ImportOperation.replace : exports.ImportOperation.add,\n };\n p.children = ['meshes', 'materials', 'lights', 'viewpoints'].flatMap(childProp => {\n const childArr = node[childProp];\n return childArr\n ? childArr.map(c => {\n const cNode = processNode(c, childProp === 'meshes' ? namePath : '');\n cNode.parent = p;\n return cNode;\n })\n : [];\n });\n return p;\n };\n const pathToImportParent = parent.getNamePath();\n imported.scene['$type'] = 'SceneNode';\n return processNode(imported.scene, pathToImportParent).children;\n }\n import(previews, parentNode, root = true) {\n const nMap = this.getNodeNamePathMap();\n // Materials have to come first so we know how they reconnect to the meshes to them\n previews.sort((a, b) => {\n if (a.type.isMaterial() && !b.type.isMaterial()) {\n return -1;\n }\n else {\n return 0;\n }\n });\n for (const p of previews) {\n if (p.operation !== exports.ImportOperation.ignore) {\n let thisNode;\n if (p.operation === exports.ImportOperation.add || !p.matchingNode) {\n const parentPath = p.namePath.substring(0, p.namePath.lastIndexOf('/'));\n let parent;\n if (!parentPath) {\n parent = parentNode;\n }\n else if (p.type.isLight()) {\n parent = nMap.lights.get(parentPath);\n }\n else if (p.type.isMaterial()) {\n parent = nMap.materials.get(parentPath);\n }\n else {\n parent = nMap.meshes.get(parentPath);\n }\n const newNode = this.deserialize(p.node, { includeChildNodes: false });\n if (instanceOfAny(newNode, ...Token.GlobalLightTokens) &&\n instanceOf(parentNode, Token.SceneNode)) {\n parentNode.lights.push(newNode);\n }\n else if ((parent === null || parent === void 0 ? void 0 : parent.id) === parentNode.id) {\n parentNode === null || parentNode === void 0 ? void 0 : parentNode.addChildren(newNode);\n }\n thisNode = newNode;\n }\n else {\n // replace\n const originalId = p.matchingNode.id;\n const matchingMeshNode = isMeshNode(p.matchingNode) ? p.matchingNode : undefined;\n const originalMaterialId = matchingMeshNode ? matchingMeshNode.materialId : undefined;\n this.assign(p.matchingNode, p.node, { includeChildNodes: false });\n p.matchingNode.id = originalId;\n // If the new materialId from the import exists, that means the material was imported and added and can be used.\n // If the new materialId from the import does not exist, use the old materialId because the material was updated but\n // kept the original ID.\n if (matchingMeshNode && originalMaterialId && matchingMeshNode.materialId) {\n if (!this.getById(matchingMeshNode.materialId)) {\n matchingMeshNode.materialId = originalMaterialId;\n }\n }\n // Trigger change detection on this existing node\n if (isTrackedClass(p.matchingNode)) {\n p.matchingNode.isModified = true;\n }\n thisNode = p.matchingNode;\n }\n if (thisNode instanceof exports.SpaceNode || thisNode instanceof exports.MaterialNode) {\n this.import(p.children, thisNode, false);\n }\n else if (p.children && p.children.length > 0) {\n throw Error('Children on non-space node or non-material-node, how did this happen');\n }\n }\n }\n }\n initializeModel(model, args, modelDef, options) {\n this.setModelDefaultValues(model);\n this.assign(model, args, options);\n }\n setModelDefaultValues(o) {\n const values = defaults[o.$type];\n if (values) {\n for (const prop in values) {\n if (values[prop] && values[prop].$type) {\n o[prop] = this.deserialize(values[prop]);\n }\n else if (isObject$1(values[prop])) {\n o[prop] = { ...values[prop] }; //need to clone the object from default values otherwise the class and defaults will both be pointing to same instance!\n }\n else {\n o[prop] = values[prop];\n }\n }\n }\n }\n assign(model, args, options) {\n options = {\n includeChildNodes: true,\n issueNewIds: false,\n copyingToDifferentNode: false,\n ...options,\n };\n if (args) {\n if (options.copyingToDifferentNode) {\n // If we are using assign to copy properties from one node to another, serialize the data\n if (args instanceof exports.Kb3dObject) {\n const defaultValues = { $type: args.$type };\n this.setModelDefaultValues(defaultValues);\n args = { ...defaultValues, ...this.serialize(args) };\n }\n }\n for (const prop in args) {\n // Never persist properties beginning with an underscore\n if (prop.substr(0, 1) == '_')\n continue;\n if (options.copyingToDifferentNode) {\n if ((prop.isEqual('id') && model.id) || prop.isEqual('$type')) {\n continue; // skip id's if copying to an existing node so the node retains its ID\n }\n }\n if (options.issueNewIds && prop.isEqual('id') && args.$type)\n continue; //skip id's if options specify so it will use the new one it generates during construction\n const sourceProp = args[prop];\n if (typeof model === 'undefined') {\n throw Error(); // this should never happen, this means a reference is being lost somewhere\n }\n if (prop === 'parent' || sourceProp === model[prop]) {\n // Don't break parent relationships\n continue; // Don't break parent relationships\n }\n if (!isObject$1(args[prop])) {\n // For primitives, just copy the values over\n model[prop] = args[prop];\n }\n else if (Array.isArray(sourceProp)) {\n // make sure there is an array on target prop. Recreate it if one is there already.\n if (model[prop]) {\n if (options.includeChildNodes ||\n !(model[prop].length > 0 &&\n model[prop][0].$type &&\n Token.withKey(model[prop][0].$type).isNode())) {\n model[prop].clear();\n }\n }\n else {\n model[prop] = [];\n }\n for (const sourceItem of sourceProp) {\n let targetItem;\n if (sourceItem.$type) {\n //it's one of our types so deserialize it\n if (options.includeChildNodes || !Token.withKey(sourceItem.$type).isNode()) {\n targetItem = this.deserialize(sourceItem, options);\n model[prop].push(targetItem);\n }\n }\n else if (isObject$1(sourceItem)) {\n targetItem = {};\n model[prop].push(targetItem);\n this.assign(targetItem, sourceItem, options);\n }\n else {\n //primitive type\n model[prop].push(sourceItem);\n }\n }\n }\n else {\n if (sourceProp.$type) {\n //it's one of our types so deserialize it\n if (options.includeChildNodes || !Token.withKey(sourceProp.$type).isNode()) {\n model[prop] = this.deserialize(sourceProp, options);\n }\n }\n else if (isObject$1(sourceProp)) {\n const targetItem = {};\n model[prop] = targetItem;\n this.assign(targetItem, sourceProp, options);\n }\n else {\n //primitive type\n model[prop] = sourceProp;\n }\n }\n }\n }\n }\n shallowEquals(a, b) {\n if (!isObject$1(a)) {\n return a === b;\n }\n for (const aProp in a) {\n if (aProp.substring(0, 1) !== '_') {\n if (!(aProp in b)) {\n return false;\n }\n if (a[aProp] !== b[aProp]) {\n return false;\n }\n }\n }\n for (const bProp in b) {\n if (bProp.substring(0, 1) !== '_') {\n if (!(bProp in a)) {\n return false;\n }\n if (b[bProp] !== a[bProp]) {\n return false;\n }\n }\n }\n return true;\n }\n }\n\n class ShadowLightNode extends exports.LightNode {\n constructor() {\n super();\n }\n }\n\n exports.DirectionalLightNode = class DirectionalLightNode extends ShadowLightNode {\n constructor() {\n super();\n }\n };\n exports.DirectionalLightNode = __decorate([\n Kb3dModel({\n token: Token.DirectionalLightNode,\n })\n ], exports.DirectionalLightNode);\n\n exports.HemisphericLightNode = class HemisphericLightNode extends exports.LightNode {\n constructor() {\n super();\n }\n };\n exports.HemisphericLightNode = __decorate([\n Kb3dModel({\n token: Token.HemisphericLightNode,\n })\n ], exports.HemisphericLightNode);\n\n exports.PointLightNode = class PointLightNode extends ShadowLightNode {\n constructor() {\n super();\n }\n };\n exports.PointLightNode = __decorate([\n Kb3dModel({\n token: Token.PointLightNode,\n })\n ], exports.PointLightNode);\n\n exports.SpotLightNode = class SpotLightNode extends exports.DirectionalLightNode {\n constructor() {\n super();\n }\n };\n exports.SpotLightNode = __decorate([\n Kb3dModel({\n token: Token.SpotLightNode,\n })\n ], exports.SpotLightNode);\n\n exports.TextureLayer = class TextureLayer extends exports.MaterialMapping {\n constructor() {\n super(...arguments);\n this._init = false;\n this._external = false;\n }\n init(parent) {\n if (!this._init) {\n this._parent = parent;\n // Normally this function is called when the node is added to an array on the parent, but textures live in defined properties on a material.\n // So we have to explicitly call it here, on texture init. The texture won't be doing anything before init anyways.\n this.calculateParents();\n if (isTrackedClass(this)) {\n // Always track changes on initialized textures because they may be attached to global materials\n this.trackChanges(true);\n this._init = true;\n }\n }\n }\n };\n __decorate([\n IgnoreHistory\n ], exports.TextureLayer.prototype, \"_init\", void 0);\n __decorate([\n IgnoreHistory\n ], exports.TextureLayer.prototype, \"_external\", void 0);\n exports.TextureLayer = __decorate([\n BaseKb3dModel({\n token: Token.TextureLayer,\n })\n ], exports.TextureLayer);\n function isTextureLayer(node) {\n return instanceOfAny(node, ...Token.TextureLayerTokens);\n }\n\n exports.AdvancedMaterialNode = class AdvancedMaterialNode extends exports.MaterialNode {\n constructor(_materialService) {\n super(_materialService);\n this._initialized = false;\n this._initialized = false;\n }\n get materials() {\n if (!this._materials)\n this._materials = [];\n return this._materials;\n }\n set materials(val) {\n this._materials = val;\n }\n };\n __decorate([\n Enumerable()\n ], exports.AdvancedMaterialNode.prototype, \"materials\", null);\n __decorate([\n IgnoreChanges\n ], exports.AdvancedMaterialNode.prototype, \"_initialized\", void 0);\n exports.AdvancedMaterialNode = __decorate([\n Kb3dModel({\n token: Token.AdvancedMaterialNode,\n onSerializing: (node, o) => {\n var _a;\n if (node.parentMaterialId == null) {\n return true;\n }\n // remove any textures that are from the parent material\n o.textures = (_a = o.textures) === null || _a === void 0 ? void 0 : _a.filter(texture => !texture._external);\n // remove any properties from the serialization that are inherited from a global or library material\n const keys = Object.keys(o);\n for (const p of keys) {\n if (node._overriddenPropertiesSet && !node._overriddenPropertiesSet.has(p) && p !== 'textures') {\n delete o[p];\n }\n }\n if (node._overriddenPropertiesSet) {\n o.overriddenProperties = Array.from(node._overriddenPropertiesSet).map(val => val.toString());\n }\n return node.location == exports.eMaterialLocation.local;\n },\n addedToChildArray: (node, child) => {\n if (child instanceof exports.TextureLayer) {\n if (node._initialized) {\n child.init(node);\n }\n }\n },\n afterCreate: node => {\n node.overriddenProperties.forEach(val => node._overriddenPropertiesSet.add(val));\n node.calculateParents(); //parents need to be calculated again because the texture children don't exist yet the first time it is run\n node.refreshParentMaterial();\n },\n }),\n __param(0, Inject(Token.MaterialService))\n ], exports.AdvancedMaterialNode);\n\n exports.ColorFillTextureLayer = class ColorFillTextureLayer extends exports.TextureLayer {\n };\n exports.ColorFillTextureLayer = __decorate([\n Kb3dModel({\n token: Token.ColorFillTextureLayer,\n })\n ], exports.ColorFillTextureLayer);\n\n exports.FileTextureLayer = class FileTextureLayer extends exports.TextureLayer {\n constructor() {\n super(...arguments);\n this.path = '';\n this._needsImageUpdate = false;\n this.flipNormalsRg = false;\n }\n };\n __decorate([\n IgnoreChanges\n ], exports.FileTextureLayer.prototype, \"_needsImageUpdate\", void 0);\n __decorate([\n IgnoreHistory\n ], exports.FileTextureLayer.prototype, \"_buffer\", void 0);\n exports.FileTextureLayer = __decorate([\n Kb3dModel({\n token: Token.FileTextureLayer,\n onSerializing: (node, o) => {\n return !node._external; // TODO Copy this to other layer types\n },\n })\n ], exports.FileTextureLayer);\n\n exports.SvgTextureLayer = class SvgTextureLayer extends exports.TextureLayer {\n constructor() {\n super(...arguments);\n this.svgXml = '';\n this.height = 256;\n this.width = 256;\n this.preloadFontStylesheet = '';\n }\n };\n exports.SvgTextureLayer = __decorate([\n Kb3dModel({\n token: Token.SvgTextureLayer,\n })\n ], exports.SvgTextureLayer);\n\n exports.TextTextureLayer = class TextTextureLayer extends exports.TextureLayer {\n };\n exports.TextTextureLayer = __decorate([\n Kb3dModel({\n token: Token.TextTextureLayer,\n })\n ], exports.TextTextureLayer);\n\n exports.MateNode = class MateNode extends exports.Kb3dObject {\n constructor() {\n super();\n }\n getC1() {\n var _a;\n return (_a = this._manager) === null || _a === void 0 ? void 0 : _a.getById(this.connector1);\n }\n getC2() {\n var _a;\n return (_a = this._manager) === null || _a === void 0 ? void 0 : _a.getById(this.connector2);\n }\n isActive() {\n return this.enabled && this.getC1() != null && this.getC2() != null;\n }\n };\n __decorate([\n NodeReference(exports.BaseConnector, 'position', 'direction', 'angle')\n ], exports.MateNode.prototype, \"connector1\", void 0);\n __decorate([\n NodeReference(exports.BaseConnector, 'position', 'direction', 'angle')\n ], exports.MateNode.prototype, \"connector2\", void 0);\n __decorate([\n IgnoreChanges\n ], exports.MateNode.prototype, \"_swap\", void 0);\n exports.MateNode = __decorate([\n Kb3dModel({\n token: Token.MateNode,\n validate: (n, v) => {\n var _a, _b, _c, _d, _e, _f, _g;\n if (n.changes.hasAny('connector1', 'connector2')) {\n v.assert(n, 'connector1', n.connector1 != null && ((_a = n._manager) === null || _a === void 0 ? void 0 : _a.getById(n.connector1)) != null, 'Connector is required');\n v.assert(n, 'connector2', n.connector2 != null && ((_b = n._manager) === null || _b === void 0 ? void 0 : _b.getById(n.connector2)) != null, 'Connector is required');\n v.assert(n, 'connector2', ((_d = (_c = n._manager) === null || _c === void 0 ? void 0 : _c.getById(n.connector1)) === null || _d === void 0 ? void 0 : _d._parent) !== ((_f = (_e = n._manager) === null || _e === void 0 ? void 0 : _e.getById(n.connector2)) === null || _f === void 0 ? void 0 : _f._parent) ||\n ((_g = n._manager) === null || _g === void 0 ? void 0 : _g.getById(n.connector1)) == null, 'Connectors cannot have the same parent node');\n }\n },\n })\n ], exports.MateNode);\n function isMate(node) {\n return instanceOf(node, Token.MateNode);\n }\n\n class Validator {\n constructor() {\n this.vmap = new Map();\n }\n getAll() {\n return this.vmap;\n }\n validateNode(node) {\n const model = Kb3dModels.get(node._token);\n if (model && model.validate) {\n model.validate(node, this);\n }\n }\n nodeIsValid(node) {\n return (!this.vmap.has(node.id) ||\n !this.vmap.get(node.id).size ||\n [...this.vmap.get(node.id).values()].every(a => a == null || a.length == 0));\n }\n getErrors(nodeId, prop) {\n var _a;\n return (_a = this.vmap.get(nodeId)) === null || _a === void 0 ? void 0 : _a.get(prop);\n }\n /** removes all validation errors for the given property */\n clearProp(node, prop) {\n var _a;\n if (this.vmap.has(node.id) && this.vmap.get(node.id).has(prop)) {\n (_a = this.vmap.get(node.id)) === null || _a === void 0 ? void 0 : _a.delete(prop);\n }\n return this;\n }\n /** removes a specific validation errors for the given property */\n clearPropError(node, prop, ...errors) {\n return this.clearPropErrorWhere(node, prop, e => e.equalsAny(...errors));\n }\n clearPropErrorWhere(node, prop, predicate) {\n var _a, _b;\n if (this.vmap.has(node.id) && this.vmap.get(node.id).has(prop)) {\n (_b = (_a = this.vmap\n .get(node.id)) === null || _a === void 0 ? void 0 : _a.get(prop)) === null || _b === void 0 ? void 0 : _b.removeWhere(e => predicate(e));\n }\n return this;\n }\n clearNode(node) {\n this.vmap.delete(node.id);\n }\n addError(node, prop, ...errors) {\n if (!this.vmap.has(node.id)) {\n this.vmap.set(node.id, new Map());\n }\n const entry = this.vmap.get(node.id);\n const sprop = prop;\n if (!entry.has(sprop)) {\n entry.set(sprop, [...errors]);\n }\n else {\n const existingErrors = entry.get(sprop);\n for (const err of errors) {\n if (!existingErrors.contains(err)) {\n existingErrors.push(err);\n }\n }\n }\n return this;\n }\n assert(node, prop, expression, ...errors) {\n if (expression) {\n this.clearPropError(node, prop, ...errors);\n }\n else {\n this.addError(node, prop, ...errors);\n }\n return this;\n }\n }\n\n exports.ViewpointNode = class ViewpointNode extends exports.Kb3dObject {\n constructor() {\n super();\n this.offsetX = 0; // For panning\n this.offsetY = 0;\n this.allowOrbit = true;\n this.allowPan = true;\n this.allowZoom = true;\n this.limitMinimumTargetDistance = false;\n this.limitLatOrbit = false;\n this.limitLngOrbit = false;\n this.type = exports.eViewpointType.orbit;\n this.targetMode = exports.eViewpointTargetMode.point;\n this.position = exports.KbVector.Zero();\n this.rotation = exports.KbVector.Zero();\n this.targetPoint = exports.KbVector.Zero();\n this.fov = defaults.ViewpointNode.fov;\n this.projection = exports.eViewpointProjection.perspective;\n }\n get type() {\n return this._type;\n }\n set type(newType) {\n this._type = newType;\n switch (newType) {\n case exports.eViewpointType.firstPerson:\n this.targetMode = exports.eViewpointTargetMode.rotation;\n this.projection = exports.eViewpointProjection.perspective;\n break;\n }\n }\n };\n __decorate([\n Enumerable()\n ], exports.ViewpointNode.prototype, \"type\", null);\n exports.ViewpointNode = __decorate([\n Kb3dModel({\n token: Token.ViewpointNode,\n onSerializing: (node, o) => {\n // remove any target properties not relevant to the target mode\n if (node.targetMode === exports.eViewpointTargetMode.rotation) {\n delete o.targetPoint;\n delete node.targetMeshId;\n }\n else if (node.targetMode === exports.eViewpointTargetMode.point) {\n delete o.rotation;\n delete node.targetMeshId;\n }\n else if (node.targetMode === exports.eViewpointTargetMode.mesh) {\n delete o.targetPoint;\n delete o.rotation;\n }\n return true;\n },\n validate: (n, v) => {\n var _a;\n if (n.changes.hasAny('targetMode', 'targetMeshId')) {\n const exp = n.targetMode != exports.eViewpointTargetMode.mesh ||\n (n.targetMeshId != null && ((_a = n._manager) === null || _a === void 0 ? void 0 : _a.getById(n.targetMeshId)) != null);\n v.assert(n, 'targetMeshId', exp, 'Target mesh required');\n }\n },\n })\n ], exports.ViewpointNode);\n function isViewpointNode(node) {\n return instanceOf(node, Token.ViewpointNode);\n }\n\n /**\n * Does a topological sort of directed graph with cycle detection and warning.\n * @param nodes a map where the key is the node and value is an array containing the nodes the key node depends on\n * @param nameSelector function that returns a descriptive name for the node\n */\n function topologicalSort(nodes, nameSelector) {\n const visited = new Set();\n const path = new Set();\n const sorted = new Set();\n const cycles = new Array();\n for (const [node] of nodes) {\n recurse(nodes, nameSelector, node, visited, path, sorted, cycles);\n }\n // console.log(`Original graph: ${getDescriptor(nameSelector, ...nodes.keys())}`);\n // console.log(`Sorted graph: ${getDescriptor(nameSelector, ...sorted)}`);\n return {\n sorted,\n cycles,\n };\n }\n function recurse(nodes, nameSelector, node, visited, path, sorted, cycles) {\n if (path.has(node)) {\n const items = [...path].reverse();\n cycles.push(items);\n //console.warn(`Cycle detected: ${getDescriptor(nameSelector, ...path, node)}`);\n }\n if (visited.has(node))\n return;\n visited.add(node);\n path.add(node);\n const edges = nodes.get(node);\n if (edges) {\n for (const edge of edges) {\n recurse(nodes, nameSelector, edge, visited, path, sorted, cycles);\n }\n }\n path.delete(node);\n sorted.add(node);\n }\n // function getDescriptor(nameSelector: (n: any) => string, ...nodes: any[]){\n // return nodes.map(n => nameSelector(n)).join(\" -> \");\n // }\n\n /**\n * Base class of all the textures in babylon.\n * It groups all the common properties required to work with Thin Engine.\n */\n class ThinTexture {\n /**\n * | Value | Type | Description |\n * | ----- | ------------------ | ----------- |\n * | 0 | CLAMP_ADDRESSMODE | |\n * | 1 | WRAP_ADDRESSMODE | |\n * | 2 | MIRROR_ADDRESSMODE | |\n */\n get wrapU() {\n return this._wrapU;\n }\n set wrapU(value) {\n this._wrapU = value;\n }\n /**\n * | Value | Type | Description |\n * | ----- | ------------------ | ----------- |\n * | 0 | CLAMP_ADDRESSMODE | |\n * | 1 | WRAP_ADDRESSMODE | |\n * | 2 | MIRROR_ADDRESSMODE | |\n */\n get wrapV() {\n return this._wrapV;\n }\n set wrapV(value) {\n this._wrapV = value;\n }\n /**\n * How a texture is mapped.\n * Unused in thin texture mode.\n */\n get coordinatesMode() {\n return 0;\n }\n /**\n * Define if the texture is a cube texture or if false a 2d texture.\n */\n get isCube() {\n if (!this._texture) {\n return false;\n }\n return this._texture.isCube;\n }\n set isCube(value) {\n if (!this._texture) {\n return;\n }\n this._texture.isCube = value;\n }\n /**\n * Define if the texture is a 3d texture (webgl 2) or if false a 2d texture.\n */\n get is3D() {\n if (!this._texture) {\n return false;\n }\n return this._texture.is3D;\n }\n set is3D(value) {\n if (!this._texture) {\n return;\n }\n this._texture.is3D = value;\n }\n /**\n * Define if the texture is a 2d array texture (webgl 2) or if false a 2d texture.\n */\n get is2DArray() {\n if (!this._texture) {\n return false;\n }\n return this._texture.is2DArray;\n }\n set is2DArray(value) {\n if (!this._texture) {\n return;\n }\n this._texture.is2DArray = value;\n }\n /**\n * Get the class name of the texture.\n * @returns \"ThinTexture\"\n */\n getClassName() {\n return \"ThinTexture\";\n }\n static _IsRenderTargetWrapper(texture) {\n return (texture === null || texture === void 0 ? void 0 : texture._shareDepth) !== undefined;\n }\n /**\n * Instantiates a new ThinTexture.\n * Base class of all the textures in babylon.\n * This can be used as an internal texture wrapper in ThinEngine to benefit from the cache\n * @param internalTexture Define the internalTexture to wrap. You can also pass a RenderTargetWrapper, in which case the texture will be the render target's texture\n */\n constructor(internalTexture) {\n this._wrapU = 1;\n this._wrapV = 1;\n /**\n * | Value | Type | Description |\n * | ----- | ------------------ | ----------- |\n * | 0 | CLAMP_ADDRESSMODE | |\n * | 1 | WRAP_ADDRESSMODE | |\n * | 2 | MIRROR_ADDRESSMODE | |\n */\n this.wrapR = 1;\n /**\n * With compliant hardware and browser (supporting anisotropic filtering)\n * this defines the level of anisotropic filtering in the texture.\n * The higher the better but the slower. This defaults to 4 as it seems to be the best tradeoff.\n */\n this.anisotropicFilteringLevel = 4;\n /**\n * Define the current state of the loading sequence when in delayed load mode.\n */\n this.delayLoadState = 0;\n /** @internal */\n this._texture = null;\n this._engine = null;\n this._cachedSize = Size.Zero();\n this._cachedBaseSize = Size.Zero();\n /** @internal */\n this._initialSamplingMode = 2;\n this._texture = ThinTexture._IsRenderTargetWrapper(internalTexture) ? internalTexture.texture : internalTexture;\n if (this._texture) {\n this._engine = this._texture.getEngine();\n }\n }\n /**\n * Get if the texture is ready to be used (downloaded, converted, mip mapped...).\n * @returns true if fully ready\n */\n isReady() {\n if (this.delayLoadState === 4) {\n this.delayLoad();\n return false;\n }\n if (this._texture) {\n return this._texture.isReady;\n }\n return false;\n }\n /**\n * Triggers the load sequence in delayed load mode.\n */\n delayLoad() { }\n /**\n * Get the underlying lower level texture from Babylon.\n * @returns the internal texture\n */\n getInternalTexture() {\n return this._texture;\n }\n /**\n * Get the size of the texture.\n * @returns the texture size.\n */\n getSize() {\n if (this._texture) {\n if (this._texture.width) {\n this._cachedSize.width = this._texture.width;\n this._cachedSize.height = this._texture.height;\n return this._cachedSize;\n }\n if (this._texture._size) {\n this._cachedSize.width = this._texture._size;\n this._cachedSize.height = this._texture._size;\n return this._cachedSize;\n }\n }\n return this._cachedSize;\n }\n /**\n * Get the base size of the texture.\n * It can be different from the size if the texture has been resized for POT for instance\n * @returns the base size\n */\n getBaseSize() {\n if (!this.isReady() || !this._texture) {\n this._cachedBaseSize.width = 0;\n this._cachedBaseSize.height = 0;\n return this._cachedBaseSize;\n }\n if (this._texture._size) {\n this._cachedBaseSize.width = this._texture._size;\n this._cachedBaseSize.height = this._texture._size;\n return this._cachedBaseSize;\n }\n this._cachedBaseSize.width = this._texture.baseWidth;\n this._cachedBaseSize.height = this._texture.baseHeight;\n return this._cachedBaseSize;\n }\n /**\n * Get the current sampling mode associated with the texture.\n */\n get samplingMode() {\n if (!this._texture) {\n return this._initialSamplingMode;\n }\n return this._texture.samplingMode;\n }\n /**\n * Update the sampling mode of the texture.\n * Default is Trilinear mode.\n *\n * | Value | Type | Description |\n * | ----- | ------------------ | ----------- |\n * | 1 | NEAREST_SAMPLINGMODE or NEAREST_NEAREST_MIPLINEAR | Nearest is: mag = nearest, min = nearest, mip = linear |\n * | 2 | BILINEAR_SAMPLINGMODE or LINEAR_LINEAR_MIPNEAREST | Bilinear is: mag = linear, min = linear, mip = nearest |\n * | 3 | TRILINEAR_SAMPLINGMODE or LINEAR_LINEAR_MIPLINEAR | Trilinear is: mag = linear, min = linear, mip = linear |\n * | 4 | NEAREST_NEAREST_MIPNEAREST | |\n * | 5 | NEAREST_LINEAR_MIPNEAREST | |\n * | 6 | NEAREST_LINEAR_MIPLINEAR | |\n * | 7 | NEAREST_LINEAR | |\n * | 8 | NEAREST_NEAREST | |\n * | 9 | LINEAR_NEAREST_MIPNEAREST | |\n * | 10 | LINEAR_NEAREST_MIPLINEAR | |\n * | 11 | LINEAR_LINEAR | |\n * | 12 | LINEAR_NEAREST | |\n *\n * > _mag_: magnification filter (close to the viewer)\n * > _min_: minification filter (far from the viewer)\n * > _mip_: filter used between mip map levels\n *@param samplingMode Define the new sampling mode of the texture\n */\n updateSamplingMode(samplingMode) {\n if (this._texture && this._engine) {\n this._engine.updateTextureSamplingMode(samplingMode, this._texture);\n }\n }\n /**\n * Release and destroy the underlying lower level texture aka internalTexture.\n */\n releaseInternalTexture() {\n if (this._texture) {\n this._texture.dispose();\n this._texture = null;\n }\n }\n /**\n * Dispose the texture and release its associated resources.\n */\n dispose() {\n if (this._texture) {\n this.releaseInternalTexture();\n this._engine = null;\n }\n }\n }\n\n /**\n * Base class of all the textures in babylon.\n * It groups all the common properties the materials, post process, lights... might need\n * in order to make a correct use of the texture.\n */\n class BaseTexture extends ThinTexture {\n /**\n * Define if the texture is having a usable alpha value (can be use for transparency or glossiness for instance).\n */\n set hasAlpha(value) {\n if (this._hasAlpha === value) {\n return;\n }\n this._hasAlpha = value;\n if (this._scene) {\n this._scene.markAllMaterialsAsDirty(1, (mat) => {\n return mat.hasTexture(this);\n });\n }\n }\n get hasAlpha() {\n return this._hasAlpha;\n }\n /**\n * Defines if the alpha value should be determined via the rgb values.\n * If true the luminance of the pixel might be used to find the corresponding alpha value.\n */\n set getAlphaFromRGB(value) {\n if (this._getAlphaFromRGB === value) {\n return;\n }\n this._getAlphaFromRGB = value;\n if (this._scene) {\n this._scene.markAllMaterialsAsDirty(1, (mat) => {\n return mat.hasTexture(this);\n });\n }\n }\n get getAlphaFromRGB() {\n return this._getAlphaFromRGB;\n }\n /**\n * Define the UV channel to use starting from 0 and defaulting to 0.\n * This is part of the texture as textures usually maps to one uv set.\n */\n set coordinatesIndex(value) {\n if (this._coordinatesIndex === value) {\n return;\n }\n this._coordinatesIndex = value;\n if (this._scene) {\n this._scene.markAllMaterialsAsDirty(1, (mat) => {\n return mat.hasTexture(this);\n });\n }\n }\n get coordinatesIndex() {\n return this._coordinatesIndex;\n }\n /**\n * How a texture is mapped.\n *\n * | Value | Type | Description |\n * | ----- | ----------------------------------- | ----------- |\n * | 0 | EXPLICIT_MODE | |\n * | 1 | SPHERICAL_MODE | |\n * | 2 | PLANAR_MODE | |\n * | 3 | CUBIC_MODE | |\n * | 4 | PROJECTION_MODE | |\n * | 5 | SKYBOX_MODE | |\n * | 6 | INVCUBIC_MODE | |\n * | 7 | EQUIRECTANGULAR_MODE | |\n * | 8 | FIXED_EQUIRECTANGULAR_MODE | |\n * | 9 | FIXED_EQUIRECTANGULAR_MIRRORED_MODE | |\n */\n set coordinatesMode(value) {\n if (this._coordinatesMode === value) {\n return;\n }\n this._coordinatesMode = value;\n if (this._scene) {\n this._scene.markAllMaterialsAsDirty(1, (mat) => {\n return mat.hasTexture(this);\n });\n }\n }\n get coordinatesMode() {\n return this._coordinatesMode;\n }\n /**\n * | Value | Type | Description |\n * | ----- | ------------------ | ----------- |\n * | 0 | CLAMP_ADDRESSMODE | |\n * | 1 | WRAP_ADDRESSMODE | |\n * | 2 | MIRROR_ADDRESSMODE | |\n */\n get wrapU() {\n return this._wrapU;\n }\n set wrapU(value) {\n this._wrapU = value;\n }\n /**\n * | Value | Type | Description |\n * | ----- | ------------------ | ----------- |\n * | 0 | CLAMP_ADDRESSMODE | |\n * | 1 | WRAP_ADDRESSMODE | |\n * | 2 | MIRROR_ADDRESSMODE | |\n */\n get wrapV() {\n return this._wrapV;\n }\n set wrapV(value) {\n this._wrapV = value;\n }\n /**\n * Define if the texture is a cube texture or if false a 2d texture.\n */\n get isCube() {\n if (!this._texture) {\n return this._isCube;\n }\n return this._texture.isCube;\n }\n set isCube(value) {\n if (!this._texture) {\n this._isCube = value;\n }\n else {\n this._texture.isCube = value;\n }\n }\n /**\n * Define if the texture is a 3d texture (webgl 2) or if false a 2d texture.\n */\n get is3D() {\n if (!this._texture) {\n return false;\n }\n return this._texture.is3D;\n }\n set is3D(value) {\n if (!this._texture) {\n return;\n }\n this._texture.is3D = value;\n }\n /**\n * Define if the texture is a 2d array texture (webgl 2) or if false a 2d texture.\n */\n get is2DArray() {\n if (!this._texture) {\n return false;\n }\n return this._texture.is2DArray;\n }\n set is2DArray(value) {\n if (!this._texture) {\n return;\n }\n this._texture.is2DArray = value;\n }\n /**\n * Define if the texture contains data in gamma space (most of the png/jpg aside bump).\n * HDR texture are usually stored in linear space.\n * This only impacts the PBR and Background materials\n */\n get gammaSpace() {\n if (!this._texture) {\n return this._gammaSpace;\n }\n else {\n if (this._texture._gammaSpace === null) {\n this._texture._gammaSpace = this._gammaSpace;\n }\n }\n return this._texture._gammaSpace && !this._texture._useSRGBBuffer;\n }\n set gammaSpace(gamma) {\n if (!this._texture) {\n if (this._gammaSpace === gamma) {\n return;\n }\n this._gammaSpace = gamma;\n }\n else {\n if (this._texture._gammaSpace === gamma) {\n return;\n }\n this._texture._gammaSpace = gamma;\n }\n this._markAllSubMeshesAsTexturesDirty();\n }\n /**\n * Gets or sets whether or not the texture contains RGBD data.\n */\n get isRGBD() {\n return this._texture != null && this._texture._isRGBD;\n }\n set isRGBD(value) {\n if (this._texture) {\n this._texture._isRGBD = value;\n }\n }\n /**\n * Are mip maps generated for this texture or not.\n */\n get noMipmap() {\n return false;\n }\n /**\n * With prefiltered texture, defined the offset used during the prefiltering steps.\n */\n get lodGenerationOffset() {\n if (this._texture) {\n return this._texture._lodGenerationOffset;\n }\n return 0.0;\n }\n set lodGenerationOffset(value) {\n if (this._texture) {\n this._texture._lodGenerationOffset = value;\n }\n }\n /**\n * With prefiltered texture, defined the scale used during the prefiltering steps.\n */\n get lodGenerationScale() {\n if (this._texture) {\n return this._texture._lodGenerationScale;\n }\n return 0.0;\n }\n set lodGenerationScale(value) {\n if (this._texture) {\n this._texture._lodGenerationScale = value;\n }\n }\n /**\n * With prefiltered texture, defined if the specular generation is based on a linear ramp.\n * By default we are using a log2 of the linear roughness helping to keep a better resolution for\n * average roughness values.\n */\n get linearSpecularLOD() {\n if (this._texture) {\n return this._texture._linearSpecularLOD;\n }\n return false;\n }\n set linearSpecularLOD(value) {\n if (this._texture) {\n this._texture._linearSpecularLOD = value;\n }\n }\n /**\n * In case a better definition than spherical harmonics is required for the diffuse part of the environment.\n * You can set the irradiance texture to rely on a texture instead of the spherical approach.\n * This texture need to have the same characteristics than its parent (Cube vs 2d, coordinates mode, Gamma/Linear, RGBD).\n */\n get irradianceTexture() {\n if (this._texture) {\n return this._texture._irradianceTexture;\n }\n return null;\n }\n set irradianceTexture(value) {\n if (this._texture) {\n this._texture._irradianceTexture = value;\n }\n }\n /**\n * Define the unique id of the texture in the scene.\n */\n get uid() {\n if (!this._uid) {\n this._uid = RandomGUID();\n }\n return this._uid;\n }\n /**\n * Return a string representation of the texture.\n * @returns the texture as a string\n */\n toString() {\n return this.name;\n }\n /**\n * Get the class name of the texture.\n * @returns \"BaseTexture\"\n */\n getClassName() {\n return \"BaseTexture\";\n }\n /**\n * Callback triggered when the texture has been disposed.\n * Kept for back compatibility, you can use the onDisposeObservable instead.\n */\n set onDispose(callback) {\n if (this._onDisposeObserver) {\n this.onDisposeObservable.remove(this._onDisposeObserver);\n }\n this._onDisposeObserver = this.onDisposeObservable.add(callback);\n }\n /**\n * Define if the texture is preventing a material to render or not.\n * If not and the texture is not ready, the engine will use a default black texture instead.\n */\n get isBlocking() {\n return true;\n }\n /**\n * Was there any loading error?\n */\n get loadingError() {\n return this._loadingError;\n }\n /**\n * If a loading error occurred this object will be populated with information about the error.\n */\n get errorObject() {\n return this._errorObject;\n }\n /**\n * Instantiates a new BaseTexture.\n * Base class of all the textures in babylon.\n * It groups all the common properties the materials, post process, lights... might need\n * in order to make a correct use of the texture.\n * @param sceneOrEngine Define the scene or engine the texture belongs to\n * @param internalTexture Define the internal texture associated with the texture\n */\n constructor(sceneOrEngine, internalTexture = null) {\n super(null);\n /**\n * Gets or sets an object used to store user defined information.\n */\n this.metadata = null;\n /**\n * For internal use only. Please do not use.\n */\n this.reservedDataStore = null;\n this._hasAlpha = false;\n this._getAlphaFromRGB = false;\n /**\n * Intensity or strength of the texture.\n * It is commonly used by materials to fine tune the intensity of the texture\n */\n this.level = 1;\n this._coordinatesIndex = 0;\n /**\n * Gets or sets a boolean indicating that the texture should try to reduce shader code if there is no UV manipulation.\n * (ie. when texture.getTextureMatrix().isIdentityAs3x2() returns true)\n */\n this.optimizeUVAllocation = true;\n this._coordinatesMode = 0;\n /**\n * | Value | Type | Description |\n * | ----- | ------------------ | ----------- |\n * | 0 | CLAMP_ADDRESSMODE | |\n * | 1 | WRAP_ADDRESSMODE | |\n * | 2 | MIRROR_ADDRESSMODE | |\n */\n this.wrapR = 1;\n /**\n * With compliant hardware and browser (supporting anisotropic filtering)\n * this defines the level of anisotropic filtering in the texture.\n * The higher the better but the slower. This defaults to 4 as it seems to be the best tradeoff.\n */\n this.anisotropicFilteringLevel = BaseTexture.DEFAULT_ANISOTROPIC_FILTERING_LEVEL;\n /** @internal */\n this._isCube = false;\n this._gammaSpace = true;\n /**\n * Is Z inverted in the texture (useful in a cube texture).\n */\n this.invertZ = false;\n /**\n * @internal\n */\n this.lodLevelInAlpha = false;\n /**\n * Define if the texture is a render target.\n */\n this.isRenderTarget = false;\n /** @internal */\n this._prefiltered = false;\n /** @internal */\n this._forceSerialize = false;\n /**\n * Define the list of animation attached to the texture.\n */\n this.animations = new Array();\n /**\n * An event triggered when the texture is disposed.\n */\n this.onDisposeObservable = new Observable$1();\n this._onDisposeObserver = null;\n this._scene = null;\n /** @internal */\n this._uid = null;\n /** @internal */\n this._parentContainer = null;\n this._loadingError = false;\n if (sceneOrEngine) {\n if (BaseTexture._IsScene(sceneOrEngine)) {\n this._scene = sceneOrEngine;\n }\n else {\n this._engine = sceneOrEngine;\n }\n }\n else {\n this._scene = EngineStore.LastCreatedScene;\n }\n if (this._scene) {\n this.uniqueId = this._scene.getUniqueId();\n this._scene.addTexture(this);\n this._engine = this._scene.getEngine();\n }\n this._texture = internalTexture;\n this._uid = null;\n }\n /**\n * Get the scene the texture belongs to.\n * @returns the scene or null if undefined\n */\n getScene() {\n return this._scene;\n }\n /** @internal */\n _getEngine() {\n return this._engine;\n }\n /**\n * Checks if the texture has the same transform matrix than another texture\n * @param texture texture to check against\n * @returns true if the transforms are the same, else false\n */\n checkTransformsAreIdentical(texture) {\n return texture !== null;\n }\n /**\n * Get the texture transform matrix used to offset tile the texture for instance.\n * @returns the transformation matrix\n */\n getTextureMatrix() {\n return Matrix.IdentityReadOnly;\n }\n /**\n * Get the texture reflection matrix used to rotate/transform the reflection.\n * @returns the reflection matrix\n */\n getReflectionTextureMatrix() {\n return Matrix.IdentityReadOnly;\n }\n /**\n * Gets a suitable rotate/transform matrix when the texture is used for refraction.\n * There's a separate function from getReflectionTextureMatrix because refraction requires a special configuration of the matrix in right-handed mode.\n * @returns The refraction matrix\n */\n getRefractionTextureMatrix() {\n return this.getReflectionTextureMatrix();\n }\n /**\n * Get if the texture is ready to be consumed (either it is ready or it is not blocking)\n * @returns true if ready, not blocking or if there was an error loading the texture\n */\n isReadyOrNotBlocking() {\n return !this.isBlocking || this.isReady() || this.loadingError;\n }\n /**\n * Scales the texture if is `canRescale()`\n * @param ratio the resize factor we want to use to rescale\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n scale(ratio) { }\n /**\n * Get if the texture can rescale.\n */\n get canRescale() {\n return false;\n }\n /**\n * @internal\n */\n _getFromCache(url, noMipmap, sampling, invertY, useSRGBBuffer, isCube) {\n const engine = this._getEngine();\n if (!engine) {\n return null;\n }\n const correctedUseSRGBBuffer = engine._getUseSRGBBuffer(!!useSRGBBuffer, noMipmap);\n const texturesCache = engine.getLoadedTexturesCache();\n for (let index = 0; index < texturesCache.length; index++) {\n const texturesCacheEntry = texturesCache[index];\n if (useSRGBBuffer === undefined || correctedUseSRGBBuffer === texturesCacheEntry._useSRGBBuffer) {\n if (invertY === undefined || invertY === texturesCacheEntry.invertY) {\n if (texturesCacheEntry.url === url && texturesCacheEntry.generateMipMaps === !noMipmap) {\n if (!sampling || sampling === texturesCacheEntry.samplingMode) {\n if (isCube === undefined || isCube === texturesCacheEntry.isCube) {\n texturesCacheEntry.incrementReferences();\n return texturesCacheEntry;\n }\n }\n }\n }\n }\n }\n return null;\n }\n /** @internal */\n _rebuild() { }\n /**\n * Clones the texture.\n * @returns the cloned texture\n */\n clone() {\n return null;\n }\n /**\n * Get the texture underlying type (INT, FLOAT...)\n */\n get textureType() {\n if (!this._texture) {\n return 0;\n }\n return this._texture.type !== undefined ? this._texture.type : 0;\n }\n /**\n * Get the texture underlying format (RGB, RGBA...)\n */\n get textureFormat() {\n if (!this._texture) {\n return 5;\n }\n return this._texture.format !== undefined ? this._texture.format : 5;\n }\n /**\n * Indicates that textures need to be re-calculated for all materials\n */\n _markAllSubMeshesAsTexturesDirty() {\n const scene = this.getScene();\n if (!scene) {\n return;\n }\n scene.markAllMaterialsAsDirty(1);\n }\n /**\n * Reads the pixels stored in the webgl texture and returns them as an ArrayBuffer.\n * This will returns an RGBA array buffer containing either in values (0-255) or\n * float values (0-1) depending of the underlying buffer type.\n * @param faceIndex defines the face of the texture to read (in case of cube texture)\n * @param level defines the LOD level of the texture to read (in case of Mip Maps)\n * @param buffer defines a user defined buffer to fill with data (can be null)\n * @param flushRenderer true to flush the renderer from the pending commands before reading the pixels\n * @param noDataConversion false to convert the data to Uint8Array (if texture type is UNSIGNED_BYTE) or to Float32Array (if texture type is anything but UNSIGNED_BYTE). If true, the type of the generated buffer (if buffer==null) will depend on the type of the texture\n * @param x defines the region x coordinates to start reading from (default to 0)\n * @param y defines the region y coordinates to start reading from (default to 0)\n * @param width defines the region width to read from (default to the texture size at level)\n * @param height defines the region width to read from (default to the texture size at level)\n * @returns The Array buffer promise containing the pixels data.\n */\n readPixels(faceIndex = 0, level = 0, buffer = null, flushRenderer = true, noDataConversion = false, x = 0, y = 0, width = Number.MAX_VALUE, height = Number.MAX_VALUE) {\n if (!this._texture) {\n return null;\n }\n const engine = this._getEngine();\n if (!engine) {\n return null;\n }\n const size = this.getSize();\n let maxWidth = size.width;\n let maxHeight = size.height;\n if (level !== 0) {\n maxWidth = maxWidth / Math.pow(2, level);\n maxHeight = maxHeight / Math.pow(2, level);\n maxWidth = Math.round(maxWidth);\n maxHeight = Math.round(maxHeight);\n }\n width = Math.min(maxWidth, width);\n height = Math.min(maxHeight, height);\n try {\n if (this._texture.isCube) {\n return engine._readTexturePixels(this._texture, width, height, faceIndex, level, buffer, flushRenderer, noDataConversion, x, y);\n }\n return engine._readTexturePixels(this._texture, width, height, -1, level, buffer, flushRenderer, noDataConversion, x, y);\n }\n catch (e) {\n return null;\n }\n }\n /**\n * @internal\n */\n _readPixelsSync(faceIndex = 0, level = 0, buffer = null, flushRenderer = true, noDataConversion = false) {\n if (!this._texture) {\n return null;\n }\n const size = this.getSize();\n let width = size.width;\n let height = size.height;\n const engine = this._getEngine();\n if (!engine) {\n return null;\n }\n if (level != 0) {\n width = width / Math.pow(2, level);\n height = height / Math.pow(2, level);\n width = Math.round(width);\n height = Math.round(height);\n }\n try {\n if (this._texture.isCube) {\n return engine._readTexturePixelsSync(this._texture, width, height, faceIndex, level, buffer, flushRenderer, noDataConversion);\n }\n return engine._readTexturePixelsSync(this._texture, width, height, -1, level, buffer, flushRenderer, noDataConversion);\n }\n catch (e) {\n return null;\n }\n }\n /** @internal */\n get _lodTextureHigh() {\n if (this._texture) {\n return this._texture._lodTextureHigh;\n }\n return null;\n }\n /** @internal */\n get _lodTextureMid() {\n if (this._texture) {\n return this._texture._lodTextureMid;\n }\n return null;\n }\n /** @internal */\n get _lodTextureLow() {\n if (this._texture) {\n return this._texture._lodTextureLow;\n }\n return null;\n }\n /**\n * Dispose the texture and release its associated resources.\n */\n dispose() {\n if (this._scene) {\n // Animations\n if (this._scene.stopAnimation) {\n this._scene.stopAnimation(this);\n }\n // Remove from scene\n this._scene.removePendingData(this);\n const index = this._scene.textures.indexOf(this);\n if (index >= 0) {\n this._scene.textures.splice(index, 1);\n }\n this._scene.onTextureRemovedObservable.notifyObservers(this);\n this._scene = null;\n if (this._parentContainer) {\n const index = this._parentContainer.textures.indexOf(this);\n if (index > -1) {\n this._parentContainer.textures.splice(index, 1);\n }\n this._parentContainer = null;\n }\n }\n // Callback\n this.onDisposeObservable.notifyObservers(this);\n this.onDisposeObservable.clear();\n this.metadata = null;\n super.dispose();\n }\n /**\n * Serialize the texture into a JSON representation that can be parsed later on.\n * @param allowEmptyName True to force serialization even if name is empty. Default: false\n * @returns the JSON representation of the texture\n */\n serialize(allowEmptyName = false) {\n if (!this.name && !allowEmptyName) {\n return null;\n }\n const serializationObject = SerializationHelper.Serialize(this);\n // Animations\n SerializationHelper.AppendSerializedAnimations(this, serializationObject);\n return serializationObject;\n }\n /**\n * Helper function to be called back once a list of texture contains only ready textures.\n * @param textures Define the list of textures to wait for\n * @param callback Define the callback triggered once the entire list will be ready\n */\n static WhenAllReady(textures, callback) {\n let numRemaining = textures.length;\n if (numRemaining === 0) {\n callback();\n return;\n }\n for (let i = 0; i < textures.length; i++) {\n const texture = textures[i];\n if (texture.isReady()) {\n if (--numRemaining === 0) {\n callback();\n }\n }\n else {\n const onLoadObservable = texture.onLoadObservable;\n if (onLoadObservable) {\n onLoadObservable.addOnce(() => {\n if (--numRemaining === 0) {\n callback();\n }\n });\n }\n else {\n if (--numRemaining === 0) {\n callback();\n }\n }\n }\n }\n }\n static _IsScene(sceneOrEngine) {\n return sceneOrEngine.getClassName() === \"Scene\";\n }\n }\n /**\n * Default anisotropic filtering level for the application.\n * It is set to 4 as a good tradeoff between perf and quality.\n */\n BaseTexture.DEFAULT_ANISOTROPIC_FILTERING_LEVEL = 4;\n __decorate$1([\n serialize()\n ], BaseTexture.prototype, \"uniqueId\", void 0);\n __decorate$1([\n serialize()\n ], BaseTexture.prototype, \"name\", void 0);\n __decorate$1([\n serialize()\n ], BaseTexture.prototype, \"metadata\", void 0);\n __decorate$1([\n serialize(\"hasAlpha\")\n ], BaseTexture.prototype, \"_hasAlpha\", void 0);\n __decorate$1([\n serialize(\"getAlphaFromRGB\")\n ], BaseTexture.prototype, \"_getAlphaFromRGB\", void 0);\n __decorate$1([\n serialize()\n ], BaseTexture.prototype, \"level\", void 0);\n __decorate$1([\n serialize(\"coordinatesIndex\")\n ], BaseTexture.prototype, \"_coordinatesIndex\", void 0);\n __decorate$1([\n serialize()\n ], BaseTexture.prototype, \"optimizeUVAllocation\", void 0);\n __decorate$1([\n serialize(\"coordinatesMode\")\n ], BaseTexture.prototype, \"_coordinatesMode\", void 0);\n __decorate$1([\n serialize()\n ], BaseTexture.prototype, \"wrapU\", null);\n __decorate$1([\n serialize()\n ], BaseTexture.prototype, \"wrapV\", null);\n __decorate$1([\n serialize()\n ], BaseTexture.prototype, \"wrapR\", void 0);\n __decorate$1([\n serialize()\n ], BaseTexture.prototype, \"anisotropicFilteringLevel\", void 0);\n __decorate$1([\n serialize()\n ], BaseTexture.prototype, \"isCube\", null);\n __decorate$1([\n serialize()\n ], BaseTexture.prototype, \"is3D\", null);\n __decorate$1([\n serialize()\n ], BaseTexture.prototype, \"is2DArray\", null);\n __decorate$1([\n serialize()\n ], BaseTexture.prototype, \"gammaSpace\", null);\n __decorate$1([\n serialize()\n ], BaseTexture.prototype, \"invertZ\", void 0);\n __decorate$1([\n serialize()\n ], BaseTexture.prototype, \"lodLevelInAlpha\", void 0);\n __decorate$1([\n serialize()\n ], BaseTexture.prototype, \"lodGenerationOffset\", null);\n __decorate$1([\n serialize()\n ], BaseTexture.prototype, \"lodGenerationScale\", null);\n __decorate$1([\n serialize()\n ], BaseTexture.prototype, \"linearSpecularLOD\", null);\n __decorate$1([\n serializeAsTexture()\n ], BaseTexture.prototype, \"irradianceTexture\", null);\n __decorate$1([\n serialize()\n ], BaseTexture.prototype, \"isRenderTarget\", void 0);\n\n /**\n * Transform some pixel data to a base64 string\n * @param pixels defines the pixel data to transform to base64\n * @param size defines the width and height of the (texture) data\n * @param invertY true if the data must be inverted for the Y coordinate during the conversion\n * @returns The base64 encoded string or null\n */\n function GenerateBase64StringFromPixelData(pixels, size, invertY = false) {\n const width = size.width;\n const height = size.height;\n if (pixels instanceof Float32Array) {\n let len = pixels.byteLength / pixels.BYTES_PER_ELEMENT;\n const npixels = new Uint8Array(len);\n while (--len >= 0) {\n let val = pixels[len];\n if (val < 0) {\n val = 0;\n }\n else if (val > 1) {\n val = 1;\n }\n npixels[len] = val * 255;\n }\n pixels = npixels;\n }\n const canvas = document.createElement(\"canvas\");\n canvas.width = width;\n canvas.height = height;\n const ctx = canvas.getContext(\"2d\");\n if (!ctx) {\n return null;\n }\n const imageData = ctx.createImageData(width, height);\n const castData = imageData.data;\n castData.set(pixels);\n ctx.putImageData(imageData, 0, 0);\n if (invertY) {\n const canvas2 = document.createElement(\"canvas\");\n canvas2.width = width;\n canvas2.height = height;\n const ctx2 = canvas2.getContext(\"2d\");\n if (!ctx2) {\n return null;\n }\n ctx2.translate(0, height);\n ctx2.scale(1, -1);\n ctx2.drawImage(canvas, 0, 0);\n return canvas2.toDataURL(\"image/png\");\n }\n return canvas.toDataURL(\"image/png\");\n }\n /**\n * Reads the pixels stored in the webgl texture and returns them as a base64 string\n * @param texture defines the texture to read pixels from\n * @param faceIndex defines the face of the texture to read (in case of cube texture)\n * @param level defines the LOD level of the texture to read (in case of Mip Maps)\n * @returns The base64 encoded string or null\n */\n function GenerateBase64StringFromTexture(texture, faceIndex = 0, level = 0) {\n const internalTexture = texture.getInternalTexture();\n if (!internalTexture) {\n return null;\n }\n const pixels = texture._readPixelsSync(faceIndex, level);\n if (!pixels) {\n return null;\n }\n return GenerateBase64StringFromPixelData(pixels, texture.getSize(), internalTexture.invertY);\n }\n /**\n * Reads the pixels stored in the webgl texture and returns them as a base64 string\n * @param texture defines the texture to read pixels from\n * @param faceIndex defines the face of the texture to read (in case of cube texture)\n * @param level defines the LOD level of the texture to read (in case of Mip Maps)\n * @returns The base64 encoded string or null wrapped in a promise\n */\n async function GenerateBase64StringFromTextureAsync(texture, faceIndex = 0, level = 0) {\n const internalTexture = texture.getInternalTexture();\n if (!internalTexture) {\n return null;\n }\n const pixels = await texture.readPixels(faceIndex, level);\n if (!pixels) {\n return null;\n }\n return GenerateBase64StringFromPixelData(pixels, texture.getSize(), internalTexture.invertY);\n }\n\n /**\n * This represents a texture in babylon. It can be easily loaded from a network, base64 or html input.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/materials_introduction#texture\n */\n class Texture extends BaseTexture {\n /**\n * Are mip maps generated for this texture or not.\n */\n get noMipmap() {\n return this._noMipmap;\n }\n /** Returns the texture mime type if it was defined by a loader (undefined else) */\n get mimeType() {\n return this._mimeType;\n }\n /**\n * Is the texture preventing material to render while loading.\n * If false, a default texture will be used instead of the loading one during the preparation step.\n */\n set isBlocking(value) {\n this._isBlocking = value;\n }\n get isBlocking() {\n return this._isBlocking;\n }\n /**\n * Gets a boolean indicating if the texture needs to be inverted on the y axis during loading\n */\n get invertY() {\n return this._invertY;\n }\n /**\n * Instantiates a new texture.\n * This represents a texture in babylon. It can be easily loaded from a network, base64 or html input.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/materials_introduction#texture\n * @param url defines the url of the picture to load as a texture\n * @param sceneOrEngine defines the scene or engine the texture will belong to\n * @param noMipmapOrOptions defines if the texture will require mip maps or not or set of all options to create the texture\n * @param invertY defines if the texture needs to be inverted on the y axis during loading\n * @param samplingMode defines the sampling mode we want for the texture while fetching from it (Texture.NEAREST_SAMPLINGMODE...)\n * @param onLoad defines a callback triggered when the texture has been loaded\n * @param onError defines a callback triggered when an error occurred during the loading session\n * @param buffer defines the buffer to load the texture from in case the texture is loaded from a buffer representation\n * @param deleteBuffer defines if the buffer we are loading the texture from should be deleted after load\n * @param format defines the format of the texture we are trying to load (Engine.TEXTUREFORMAT_RGBA...)\n * @param mimeType defines an optional mime type information\n * @param loaderOptions options to be passed to the loader\n * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg)\n * @param forcedExtension defines the extension to use to pick the right loader\n */\n constructor(url, sceneOrEngine, noMipmapOrOptions, invertY, samplingMode = Texture.TRILINEAR_SAMPLINGMODE, onLoad = null, onError = null, buffer = null, deleteBuffer = false, format, mimeType, loaderOptions, creationFlags, forcedExtension) {\n var _a, _b, _c, _d, _e, _f, _g, _h, _j;\n super(sceneOrEngine);\n /**\n * Define the url of the texture.\n */\n this.url = null;\n /**\n * Define an offset on the texture to offset the u coordinates of the UVs\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/moreMaterials#offsetting\n */\n this.uOffset = 0;\n /**\n * Define an offset on the texture to offset the v coordinates of the UVs\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/moreMaterials#offsetting\n */\n this.vOffset = 0;\n /**\n * Define an offset on the texture to scale the u coordinates of the UVs\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/moreMaterials#tiling\n */\n this.uScale = 1.0;\n /**\n * Define an offset on the texture to scale the v coordinates of the UVs\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/moreMaterials#tiling\n */\n this.vScale = 1.0;\n /**\n * Define an offset on the texture to rotate around the u coordinates of the UVs\n * The angle is defined in radians.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/moreMaterials\n */\n this.uAng = 0;\n /**\n * Define an offset on the texture to rotate around the v coordinates of the UVs\n * The angle is defined in radians.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/moreMaterials\n */\n this.vAng = 0;\n /**\n * Define an offset on the texture to rotate around the w coordinates of the UVs (in case of 3d texture)\n * The angle is defined in radians.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/moreMaterials\n */\n this.wAng = 0;\n /**\n * Defines the center of rotation (U)\n */\n this.uRotationCenter = 0.5;\n /**\n * Defines the center of rotation (V)\n */\n this.vRotationCenter = 0.5;\n /**\n * Defines the center of rotation (W)\n */\n this.wRotationCenter = 0.5;\n /**\n * Sets this property to true to avoid deformations when rotating the texture with non-uniform scaling\n */\n this.homogeneousRotationInUVTransform = false;\n /**\n * List of inspectable custom properties (used by the Inspector)\n * @see https://doc.babylonjs.com/toolsAndResources/inspector#extensibility\n */\n this.inspectableCustomProperties = null;\n /** @internal */\n this._noMipmap = false;\n /** @internal */\n this._invertY = false;\n this._rowGenerationMatrix = null;\n this._cachedTextureMatrix = null;\n this._projectionModeMatrix = null;\n this._t0 = null;\n this._t1 = null;\n this._t2 = null;\n this._cachedUOffset = -1;\n this._cachedVOffset = -1;\n this._cachedUScale = 0;\n this._cachedVScale = 0;\n this._cachedUAng = -1;\n this._cachedVAng = -1;\n this._cachedWAng = -1;\n this._cachedReflectionProjectionMatrixId = -1;\n this._cachedURotationCenter = -1;\n this._cachedVRotationCenter = -1;\n this._cachedWRotationCenter = -1;\n this._cachedHomogeneousRotationInUVTransform = false;\n this._cachedReflectionTextureMatrix = null;\n this._cachedReflectionUOffset = -1;\n this._cachedReflectionVOffset = -1;\n this._cachedReflectionUScale = 0;\n this._cachedReflectionVScale = 0;\n this._cachedReflectionCoordinatesMode = -1;\n /** @internal */\n this._buffer = null;\n this._deleteBuffer = false;\n this._format = null;\n this._delayedOnLoad = null;\n this._delayedOnError = null;\n /**\n * Observable triggered once the texture has been loaded.\n */\n this.onLoadObservable = new Observable$1();\n this._isBlocking = true;\n this.name = url || \"\";\n this.url = url;\n let noMipmap;\n let useSRGBBuffer = false;\n let internalTexture = null;\n if (typeof noMipmapOrOptions === \"object\" && noMipmapOrOptions !== null) {\n noMipmap = (_a = noMipmapOrOptions.noMipmap) !== null && _a !== void 0 ? _a : false;\n invertY = (_b = noMipmapOrOptions.invertY) !== null && _b !== void 0 ? _b : (CompatibilityOptions.UseOpenGLOrientationForUV ? false : true);\n samplingMode = (_c = noMipmapOrOptions.samplingMode) !== null && _c !== void 0 ? _c : Texture.TRILINEAR_SAMPLINGMODE;\n onLoad = (_d = noMipmapOrOptions.onLoad) !== null && _d !== void 0 ? _d : null;\n onError = (_e = noMipmapOrOptions.onError) !== null && _e !== void 0 ? _e : null;\n buffer = (_f = noMipmapOrOptions.buffer) !== null && _f !== void 0 ? _f : null;\n deleteBuffer = (_g = noMipmapOrOptions.deleteBuffer) !== null && _g !== void 0 ? _g : false;\n format = noMipmapOrOptions.format;\n mimeType = noMipmapOrOptions.mimeType;\n loaderOptions = noMipmapOrOptions.loaderOptions;\n creationFlags = noMipmapOrOptions.creationFlags;\n useSRGBBuffer = (_h = noMipmapOrOptions.useSRGBBuffer) !== null && _h !== void 0 ? _h : false;\n internalTexture = (_j = noMipmapOrOptions.internalTexture) !== null && _j !== void 0 ? _j : null;\n }\n else {\n noMipmap = !!noMipmapOrOptions;\n }\n this._noMipmap = noMipmap;\n this._invertY = invertY === undefined ? (CompatibilityOptions.UseOpenGLOrientationForUV ? false : true) : invertY;\n this._initialSamplingMode = samplingMode;\n this._buffer = buffer;\n this._deleteBuffer = deleteBuffer;\n this._mimeType = mimeType;\n this._loaderOptions = loaderOptions;\n this._creationFlags = creationFlags;\n this._useSRGBBuffer = useSRGBBuffer;\n this._forcedExtension = forcedExtension;\n if (format) {\n this._format = format;\n }\n const scene = this.getScene();\n const engine = this._getEngine();\n if (!engine) {\n return;\n }\n engine.onBeforeTextureInitObservable.notifyObservers(this);\n const load = () => {\n if (this._texture) {\n if (this._texture._invertVScale) {\n this.vScale *= -1;\n this.vOffset += 1;\n }\n // Update texture to match internal texture's wrapping\n if (this._texture._cachedWrapU !== null) {\n this.wrapU = this._texture._cachedWrapU;\n this._texture._cachedWrapU = null;\n }\n if (this._texture._cachedWrapV !== null) {\n this.wrapV = this._texture._cachedWrapV;\n this._texture._cachedWrapV = null;\n }\n if (this._texture._cachedWrapR !== null) {\n this.wrapR = this._texture._cachedWrapR;\n this._texture._cachedWrapR = null;\n }\n }\n if (this.onLoadObservable.hasObservers()) {\n this.onLoadObservable.notifyObservers(this);\n }\n if (onLoad) {\n onLoad();\n }\n if (!this.isBlocking && scene) {\n scene.resetCachedMaterial();\n }\n };\n const errorHandler = (message, exception) => {\n this._loadingError = true;\n this._errorObject = { message, exception };\n if (onError) {\n onError(message, exception);\n }\n Texture.OnTextureLoadErrorObservable.notifyObservers(this);\n };\n if (!this.url && !internalTexture) {\n this._delayedOnLoad = load;\n this._delayedOnError = errorHandler;\n return;\n }\n this._texture = internalTexture !== null && internalTexture !== void 0 ? internalTexture : this._getFromCache(this.url, noMipmap, samplingMode, this._invertY, useSRGBBuffer);\n if (!this._texture) {\n if (!scene || !scene.useDelayedTextureLoading) {\n try {\n this._texture = engine.createTexture(this.url, noMipmap, this._invertY, scene, samplingMode, load, errorHandler, this._buffer, undefined, this._format, this._forcedExtension, mimeType, loaderOptions, creationFlags, useSRGBBuffer);\n }\n catch (e) {\n errorHandler(\"error loading\", e);\n throw e;\n }\n if (deleteBuffer) {\n this._buffer = null;\n }\n }\n else {\n this.delayLoadState = 4;\n this._delayedOnLoad = load;\n this._delayedOnError = errorHandler;\n }\n }\n else {\n if (this._texture.isReady) {\n TimingTools.SetImmediate(() => load());\n }\n else {\n const loadObserver = this._texture.onLoadedObservable.add(load);\n this._texture.onErrorObservable.add((e) => {\n var _a;\n errorHandler(e.message, e.exception);\n (_a = this._texture) === null || _a === void 0 ? void 0 : _a.onLoadedObservable.remove(loadObserver);\n });\n }\n }\n }\n /**\n * Update the url (and optional buffer) of this texture if url was null during construction.\n * @param url the url of the texture\n * @param buffer the buffer of the texture (defaults to null)\n * @param onLoad callback called when the texture is loaded (defaults to null)\n * @param forcedExtension defines the extension to use to pick the right loader\n */\n updateURL(url, buffer = null, onLoad, forcedExtension) {\n if (this.url) {\n this.releaseInternalTexture();\n this.getScene().markAllMaterialsAsDirty(1);\n }\n if (!this.name || this.name.startsWith(\"data:\")) {\n this.name = url;\n }\n this.url = url;\n this._buffer = buffer;\n this._forcedExtension = forcedExtension;\n this.delayLoadState = 4;\n if (onLoad) {\n this._delayedOnLoad = onLoad;\n }\n this.delayLoad();\n }\n /**\n * Finish the loading sequence of a texture flagged as delayed load.\n * @internal\n */\n delayLoad() {\n if (this.delayLoadState !== 4) {\n return;\n }\n const scene = this.getScene();\n if (!scene) {\n return;\n }\n this.delayLoadState = 1;\n this._texture = this._getFromCache(this.url, this._noMipmap, this.samplingMode, this._invertY, this._useSRGBBuffer);\n if (!this._texture) {\n this._texture = scene\n .getEngine()\n .createTexture(this.url, this._noMipmap, this._invertY, scene, this.samplingMode, this._delayedOnLoad, this._delayedOnError, this._buffer, null, this._format, this._forcedExtension, this._mimeType, this._loaderOptions, this._creationFlags, this._useSRGBBuffer);\n if (this._deleteBuffer) {\n this._buffer = null;\n }\n }\n else {\n if (this._delayedOnLoad) {\n if (this._texture.isReady) {\n TimingTools.SetImmediate(this._delayedOnLoad);\n }\n else {\n this._texture.onLoadedObservable.add(this._delayedOnLoad);\n }\n }\n }\n this._delayedOnLoad = null;\n this._delayedOnError = null;\n }\n _prepareRowForTextureGeneration(x, y, z, t) {\n x *= this._cachedUScale;\n y *= this._cachedVScale;\n x -= this.uRotationCenter * this._cachedUScale;\n y -= this.vRotationCenter * this._cachedVScale;\n z -= this.wRotationCenter;\n Vector3.TransformCoordinatesFromFloatsToRef(x, y, z, this._rowGenerationMatrix, t);\n t.x += this.uRotationCenter * this._cachedUScale + this._cachedUOffset;\n t.y += this.vRotationCenter * this._cachedVScale + this._cachedVOffset;\n t.z += this.wRotationCenter;\n }\n /**\n * Checks if the texture has the same transform matrix than another texture\n * @param texture texture to check against\n * @returns true if the transforms are the same, else false\n */\n checkTransformsAreIdentical(texture) {\n return (texture !== null &&\n this.uOffset === texture.uOffset &&\n this.vOffset === texture.vOffset &&\n this.uScale === texture.uScale &&\n this.vScale === texture.vScale &&\n this.uAng === texture.uAng &&\n this.vAng === texture.vAng &&\n this.wAng === texture.wAng);\n }\n /**\n * Get the current texture matrix which includes the requested offsetting, tiling and rotation components.\n * @param uBase\n * @returns the transform matrix of the texture.\n */\n getTextureMatrix(uBase = 1) {\n if (this.uOffset === this._cachedUOffset &&\n this.vOffset === this._cachedVOffset &&\n this.uScale * uBase === this._cachedUScale &&\n this.vScale === this._cachedVScale &&\n this.uAng === this._cachedUAng &&\n this.vAng === this._cachedVAng &&\n this.wAng === this._cachedWAng &&\n this.uRotationCenter === this._cachedURotationCenter &&\n this.vRotationCenter === this._cachedVRotationCenter &&\n this.wRotationCenter === this._cachedWRotationCenter &&\n this.homogeneousRotationInUVTransform === this._cachedHomogeneousRotationInUVTransform) {\n return this._cachedTextureMatrix;\n }\n this._cachedUOffset = this.uOffset;\n this._cachedVOffset = this.vOffset;\n this._cachedUScale = this.uScale * uBase;\n this._cachedVScale = this.vScale;\n this._cachedUAng = this.uAng;\n this._cachedVAng = this.vAng;\n this._cachedWAng = this.wAng;\n this._cachedURotationCenter = this.uRotationCenter;\n this._cachedVRotationCenter = this.vRotationCenter;\n this._cachedWRotationCenter = this.wRotationCenter;\n this._cachedHomogeneousRotationInUVTransform = this.homogeneousRotationInUVTransform;\n if (!this._cachedTextureMatrix || !this._rowGenerationMatrix) {\n this._cachedTextureMatrix = Matrix.Zero();\n this._rowGenerationMatrix = new Matrix();\n this._t0 = Vector3.Zero();\n this._t1 = Vector3.Zero();\n this._t2 = Vector3.Zero();\n }\n Matrix.RotationYawPitchRollToRef(this.vAng, this.uAng, this.wAng, this._rowGenerationMatrix);\n if (this.homogeneousRotationInUVTransform) {\n Matrix.TranslationToRef(-this._cachedURotationCenter, -this._cachedVRotationCenter, -this._cachedWRotationCenter, TmpVectors.Matrix[0]);\n Matrix.TranslationToRef(this._cachedURotationCenter, this._cachedVRotationCenter, this._cachedWRotationCenter, TmpVectors.Matrix[1]);\n Matrix.ScalingToRef(this._cachedUScale, this._cachedVScale, 0, TmpVectors.Matrix[2]);\n Matrix.TranslationToRef(this._cachedUOffset, this._cachedVOffset, 0, TmpVectors.Matrix[3]);\n TmpVectors.Matrix[0].multiplyToRef(this._rowGenerationMatrix, this._cachedTextureMatrix);\n this._cachedTextureMatrix.multiplyToRef(TmpVectors.Matrix[1], this._cachedTextureMatrix);\n this._cachedTextureMatrix.multiplyToRef(TmpVectors.Matrix[2], this._cachedTextureMatrix);\n this._cachedTextureMatrix.multiplyToRef(TmpVectors.Matrix[3], this._cachedTextureMatrix);\n // copy the translation row to the 3rd row of the matrix so that we don't need to update the shaders (which expects the translation to be on the 3rd row)\n this._cachedTextureMatrix.setRowFromFloats(2, this._cachedTextureMatrix.m[12], this._cachedTextureMatrix.m[13], this._cachedTextureMatrix.m[14], 1);\n }\n else {\n this._prepareRowForTextureGeneration(0, 0, 0, this._t0);\n this._prepareRowForTextureGeneration(1.0, 0, 0, this._t1);\n this._prepareRowForTextureGeneration(0, 1.0, 0, this._t2);\n this._t1.subtractInPlace(this._t0);\n this._t2.subtractInPlace(this._t0);\n Matrix.FromValuesToRef(this._t1.x, this._t1.y, this._t1.z, 0.0, this._t2.x, this._t2.y, this._t2.z, 0.0, this._t0.x, this._t0.y, this._t0.z, 0.0, 0.0, 0.0, 0.0, 1.0, this._cachedTextureMatrix);\n }\n const scene = this.getScene();\n if (!scene) {\n return this._cachedTextureMatrix;\n }\n if (this.optimizeUVAllocation) {\n // We flag the materials that are using this texture as \"texture dirty\" because depending on the fact that the matrix is the identity or not, some defines\n // will get different values (see MaterialHelper.PrepareDefinesForMergedUV), meaning we should regenerate the effect accordingly\n scene.markAllMaterialsAsDirty(1, (mat) => {\n return mat.hasTexture(this);\n });\n }\n return this._cachedTextureMatrix;\n }\n /**\n * Get the current matrix used to apply reflection. This is useful to rotate an environment texture for instance.\n * @returns The reflection texture transform\n */\n getReflectionTextureMatrix() {\n const scene = this.getScene();\n if (!scene) {\n return this._cachedReflectionTextureMatrix;\n }\n if (this.uOffset === this._cachedReflectionUOffset &&\n this.vOffset === this._cachedReflectionVOffset &&\n this.uScale === this._cachedReflectionUScale &&\n this.vScale === this._cachedReflectionVScale &&\n this.coordinatesMode === this._cachedReflectionCoordinatesMode) {\n if (this.coordinatesMode === Texture.PROJECTION_MODE) {\n if (this._cachedReflectionProjectionMatrixId === scene.getProjectionMatrix().updateFlag) {\n return this._cachedReflectionTextureMatrix;\n }\n }\n else {\n return this._cachedReflectionTextureMatrix;\n }\n }\n if (!this._cachedReflectionTextureMatrix) {\n this._cachedReflectionTextureMatrix = Matrix.Zero();\n }\n if (!this._projectionModeMatrix) {\n this._projectionModeMatrix = Matrix.Zero();\n }\n const flagMaterialsAsTextureDirty = this._cachedReflectionCoordinatesMode !== this.coordinatesMode;\n this._cachedReflectionUOffset = this.uOffset;\n this._cachedReflectionVOffset = this.vOffset;\n this._cachedReflectionUScale = this.uScale;\n this._cachedReflectionVScale = this.vScale;\n this._cachedReflectionCoordinatesMode = this.coordinatesMode;\n switch (this.coordinatesMode) {\n case Texture.PLANAR_MODE: {\n Matrix.IdentityToRef(this._cachedReflectionTextureMatrix);\n this._cachedReflectionTextureMatrix[0] = this.uScale;\n this._cachedReflectionTextureMatrix[5] = this.vScale;\n this._cachedReflectionTextureMatrix[12] = this.uOffset;\n this._cachedReflectionTextureMatrix[13] = this.vOffset;\n break;\n }\n case Texture.PROJECTION_MODE: {\n Matrix.FromValuesToRef(0.5, 0.0, 0.0, 0.0, 0.0, -0.5, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.5, 0.5, 1.0, 1.0, this._projectionModeMatrix);\n const projectionMatrix = scene.getProjectionMatrix();\n this._cachedReflectionProjectionMatrixId = projectionMatrix.updateFlag;\n projectionMatrix.multiplyToRef(this._projectionModeMatrix, this._cachedReflectionTextureMatrix);\n break;\n }\n default:\n Matrix.IdentityToRef(this._cachedReflectionTextureMatrix);\n break;\n }\n if (flagMaterialsAsTextureDirty) {\n // We flag the materials that are using this texture as \"texture dirty\" if the coordinatesMode has changed.\n // Indeed, this property is used to set the value of some defines used to generate the effect (in material.isReadyForSubMesh), so we must make sure this code will be re-executed and the effect recreated if necessary\n scene.markAllMaterialsAsDirty(1, (mat) => {\n return mat.getActiveTextures().indexOf(this) !== -1;\n });\n }\n return this._cachedReflectionTextureMatrix;\n }\n /**\n * Clones the texture.\n * @returns the cloned texture\n */\n clone() {\n const options = {\n noMipmap: this._noMipmap,\n invertY: this._invertY,\n samplingMode: this.samplingMode,\n onLoad: undefined,\n onError: undefined,\n buffer: this._texture ? this._texture._buffer : undefined,\n deleteBuffer: this._deleteBuffer,\n format: this.textureFormat,\n mimeType: this.mimeType,\n loaderOptions: this._loaderOptions,\n creationFlags: this._creationFlags,\n useSRGBBuffer: this._useSRGBBuffer,\n };\n return SerializationHelper.Clone(() => {\n return new Texture(this._texture ? this._texture.url : null, this.getScene(), options);\n }, this);\n }\n /**\n * Serialize the texture to a JSON representation we can easily use in the respective Parse function.\n * @returns The JSON representation of the texture\n */\n serialize() {\n var _a, _b;\n const savedName = this.name;\n if (!Texture.SerializeBuffers) {\n if (this.name.startsWith(\"data:\")) {\n this.name = \"\";\n }\n }\n if (this.name.startsWith(\"data:\") && this.url === this.name) {\n this.url = \"\";\n }\n const serializationObject = super.serialize(Texture._SerializeInternalTextureUniqueId);\n if (!serializationObject) {\n return null;\n }\n if (Texture.SerializeBuffers || Texture.ForceSerializeBuffers) {\n if (typeof this._buffer === \"string\" && this._buffer.substr(0, 5) === \"data:\") {\n serializationObject.base64String = this._buffer;\n serializationObject.name = serializationObject.name.replace(\"data:\", \"\");\n }\n else if (this.url && this.url.startsWith(\"data:\") && this._buffer instanceof Uint8Array) {\n serializationObject.base64String = \"data:image/png;base64,\" + EncodeArrayBufferToBase64(this._buffer);\n }\n else if (Texture.ForceSerializeBuffers || (this.url && this.url.startsWith(\"blob:\")) || this._forceSerialize) {\n serializationObject.base64String =\n !this._engine || this._engine._features.supportSyncTextureRead ? GenerateBase64StringFromTexture(this) : GenerateBase64StringFromTextureAsync(this);\n }\n }\n serializationObject.invertY = this._invertY;\n serializationObject.samplingMode = this.samplingMode;\n serializationObject._creationFlags = this._creationFlags;\n serializationObject._useSRGBBuffer = this._useSRGBBuffer;\n if (Texture._SerializeInternalTextureUniqueId) {\n serializationObject.internalTextureUniqueId = (_b = (_a = this._texture) === null || _a === void 0 ? void 0 : _a.uniqueId) !== null && _b !== void 0 ? _b : undefined;\n }\n this.name = savedName;\n return serializationObject;\n }\n /**\n * Get the current class name of the texture useful for serialization or dynamic coding.\n * @returns \"Texture\"\n */\n getClassName() {\n return \"Texture\";\n }\n /**\n * Dispose the texture and release its associated resources.\n */\n dispose() {\n super.dispose();\n this.onLoadObservable.clear();\n this._delayedOnLoad = null;\n this._delayedOnError = null;\n this._buffer = null;\n }\n /**\n * Parse the JSON representation of a texture in order to recreate the texture in the given scene.\n * @param parsedTexture Define the JSON representation of the texture\n * @param scene Define the scene the parsed texture should be instantiated in\n * @param rootUrl Define the root url of the parsing sequence in the case of relative dependencies\n * @returns The parsed texture if successful\n */\n static Parse(parsedTexture, scene, rootUrl) {\n if (parsedTexture.customType) {\n const customTexture = InstantiationTools.Instantiate(parsedTexture.customType);\n // Update Sampling Mode\n const parsedCustomTexture = customTexture.Parse(parsedTexture, scene, rootUrl);\n if (parsedTexture.samplingMode && parsedCustomTexture.updateSamplingMode && parsedCustomTexture._samplingMode) {\n if (parsedCustomTexture._samplingMode !== parsedTexture.samplingMode) {\n parsedCustomTexture.updateSamplingMode(parsedTexture.samplingMode);\n }\n }\n return parsedCustomTexture;\n }\n if (parsedTexture.isCube && !parsedTexture.isRenderTarget) {\n return Texture._CubeTextureParser(parsedTexture, scene, rootUrl);\n }\n const hasInternalTextureUniqueId = parsedTexture.internalTextureUniqueId !== undefined;\n if (!parsedTexture.name && !parsedTexture.isRenderTarget && !hasInternalTextureUniqueId) {\n return null;\n }\n let internalTexture;\n if (hasInternalTextureUniqueId) {\n const cache = scene.getEngine().getLoadedTexturesCache();\n for (const texture of cache) {\n if (texture.uniqueId === parsedTexture.internalTextureUniqueId) {\n internalTexture = texture;\n break;\n }\n }\n }\n const onLoaded = (texture) => {\n var _a;\n // Clear cache\n if (texture && texture._texture) {\n texture._texture._cachedWrapU = null;\n texture._texture._cachedWrapV = null;\n texture._texture._cachedWrapR = null;\n }\n // Update Sampling Mode\n if (parsedTexture.samplingMode) {\n const sampling = parsedTexture.samplingMode;\n if (texture && texture.samplingMode !== sampling) {\n texture.updateSamplingMode(sampling);\n }\n }\n // Animations\n if (texture && parsedTexture.animations) {\n for (let animationIndex = 0; animationIndex < parsedTexture.animations.length; animationIndex++) {\n const parsedAnimation = parsedTexture.animations[animationIndex];\n const internalClass = GetClass(\"BABYLON.Animation\");\n if (internalClass) {\n texture.animations.push(internalClass.Parse(parsedAnimation));\n }\n }\n }\n if (hasInternalTextureUniqueId && !internalTexture) {\n (_a = texture === null || texture === void 0 ? void 0 : texture._texture) === null || _a === void 0 ? void 0 : _a._setUniqueId(parsedTexture.internalTextureUniqueId);\n }\n };\n const texture = SerializationHelper.Parse(() => {\n var _a, _b, _c;\n let generateMipMaps = true;\n if (parsedTexture.noMipmap) {\n generateMipMaps = false;\n }\n if (parsedTexture.mirrorPlane) {\n const mirrorTexture = Texture._CreateMirror(parsedTexture.name, parsedTexture.renderTargetSize, scene, generateMipMaps);\n mirrorTexture._waitingRenderList = parsedTexture.renderList;\n mirrorTexture.mirrorPlane = Plane.FromArray(parsedTexture.mirrorPlane);\n onLoaded(mirrorTexture);\n return mirrorTexture;\n }\n else if (parsedTexture.isRenderTarget) {\n let renderTargetTexture = null;\n if (parsedTexture.isCube) {\n // Search for an existing reflection probe (which contains a cube render target texture)\n if (scene.reflectionProbes) {\n for (let index = 0; index < scene.reflectionProbes.length; index++) {\n const probe = scene.reflectionProbes[index];\n if (probe.name === parsedTexture.name) {\n return probe.cubeTexture;\n }\n }\n }\n }\n else {\n renderTargetTexture = Texture._CreateRenderTargetTexture(parsedTexture.name, parsedTexture.renderTargetSize, scene, generateMipMaps, (_a = parsedTexture._creationFlags) !== null && _a !== void 0 ? _a : 0);\n renderTargetTexture._waitingRenderList = parsedTexture.renderList;\n }\n onLoaded(renderTargetTexture);\n return renderTargetTexture;\n }\n else {\n let texture;\n if (parsedTexture.base64String && !internalTexture) {\n // name and url are the same to ensure caching happens from the actual base64 string\n texture = Texture.CreateFromBase64String(parsedTexture.base64String, parsedTexture.base64String, scene, !generateMipMaps, parsedTexture.invertY, parsedTexture.samplingMode, () => {\n onLoaded(texture);\n }, (_b = parsedTexture._creationFlags) !== null && _b !== void 0 ? _b : 0, (_c = parsedTexture._useSRGBBuffer) !== null && _c !== void 0 ? _c : false);\n // prettier name to fit with the loaded data\n texture.name = parsedTexture.name;\n }\n else {\n let url;\n if (parsedTexture.name && (parsedTexture.name.indexOf(\"://\") > 0 || parsedTexture.name.startsWith(\"data:\"))) {\n url = parsedTexture.name;\n }\n else {\n url = rootUrl + parsedTexture.name;\n }\n if (parsedTexture.url && (parsedTexture.url.startsWith(\"data:\") || Texture.UseSerializedUrlIfAny)) {\n url = parsedTexture.url;\n }\n const options = {\n noMipmap: !generateMipMaps,\n invertY: parsedTexture.invertY,\n samplingMode: parsedTexture.samplingMode,\n onLoad: () => {\n onLoaded(texture);\n },\n internalTexture,\n };\n texture = new Texture(url, scene, options);\n }\n return texture;\n }\n }, parsedTexture, scene);\n return texture;\n }\n /**\n * Creates a texture from its base 64 representation.\n * @param data Define the base64 payload without the data: prefix\n * @param name Define the name of the texture in the scene useful fo caching purpose for instance\n * @param scene Define the scene the texture should belong to\n * @param noMipmapOrOptions defines if the texture will require mip maps or not or set of all options to create the texture\n * @param invertY define if the texture needs to be inverted on the y axis during loading\n * @param samplingMode define the sampling mode we want for the texture while fetching from it (Texture.NEAREST_SAMPLINGMODE...)\n * @param onLoad define a callback triggered when the texture has been loaded\n * @param onError define a callback triggered when an error occurred during the loading session\n * @param format define the format of the texture we are trying to load (Engine.TEXTUREFORMAT_RGBA...)\n * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg)\n * @returns the created texture\n */\n static CreateFromBase64String(data, name, scene, noMipmapOrOptions, invertY, samplingMode = Texture.TRILINEAR_SAMPLINGMODE, onLoad = null, onError = null, format = 5, creationFlags) {\n return new Texture(\"data:\" + name, scene, noMipmapOrOptions, invertY, samplingMode, onLoad, onError, data, false, format, undefined, undefined, creationFlags);\n }\n /**\n * Creates a texture from its data: representation. (data: will be added in case only the payload has been passed in)\n * @param name Define the name of the texture in the scene useful fo caching purpose for instance\n * @param buffer define the buffer to load the texture from in case the texture is loaded from a buffer representation\n * @param scene Define the scene the texture should belong to\n * @param deleteBuffer define if the buffer we are loading the texture from should be deleted after load\n * @param noMipmapOrOptions defines if the texture will require mip maps or not or set of all options to create the texture\n * @param invertY define if the texture needs to be inverted on the y axis during loading\n * @param samplingMode define the sampling mode we want for the texture while fetching from it (Texture.NEAREST_SAMPLINGMODE...)\n * @param onLoad define a callback triggered when the texture has been loaded\n * @param onError define a callback triggered when an error occurred during the loading session\n * @param format define the format of the texture we are trying to load (Engine.TEXTUREFORMAT_RGBA...)\n * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg)\n * @returns the created texture\n */\n static LoadFromDataString(name, buffer, scene, deleteBuffer = false, noMipmapOrOptions, invertY = true, samplingMode = Texture.TRILINEAR_SAMPLINGMODE, onLoad = null, onError = null, format = 5, creationFlags) {\n if (name.substr(0, 5) !== \"data:\") {\n name = \"data:\" + name;\n }\n return new Texture(name, scene, noMipmapOrOptions, invertY, samplingMode, onLoad, onError, buffer, deleteBuffer, format, undefined, undefined, creationFlags);\n }\n }\n /**\n * Gets or sets a general boolean used to indicate that textures containing direct data (buffers) must be saved as part of the serialization process\n */\n Texture.SerializeBuffers = true;\n /**\n * Gets or sets a general boolean used to indicate that texture buffers must be saved as part of the serialization process.\n * If no buffer exists, one will be created as base64 string from the internal webgl data.\n */\n Texture.ForceSerializeBuffers = false;\n /**\n * This observable will notify when any texture had a loading error\n */\n Texture.OnTextureLoadErrorObservable = new Observable$1();\n /** @internal */\n Texture._SerializeInternalTextureUniqueId = false;\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n Texture._CubeTextureParser = (jsonTexture, scene, rootUrl) => {\n throw _WarnImport(\"CubeTexture\");\n };\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n Texture._CreateMirror = (name, renderTargetSize, scene, generateMipMaps) => {\n throw _WarnImport(\"MirrorTexture\");\n };\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n Texture._CreateRenderTargetTexture = (name, renderTargetSize, scene, generateMipMaps, creationFlags) => {\n throw _WarnImport(\"RenderTargetTexture\");\n };\n /** nearest is mag = nearest and min = nearest and no mip */\n Texture.NEAREST_SAMPLINGMODE = 1;\n /** nearest is mag = nearest and min = nearest and mip = linear */\n Texture.NEAREST_NEAREST_MIPLINEAR = 8; // nearest is mag = nearest and min = nearest and mip = linear\n /** Bilinear is mag = linear and min = linear and no mip */\n Texture.BILINEAR_SAMPLINGMODE = 2;\n /** Bilinear is mag = linear and min = linear and mip = nearest */\n Texture.LINEAR_LINEAR_MIPNEAREST = 11; // Bilinear is mag = linear and min = linear and mip = nearest\n /** Trilinear is mag = linear and min = linear and mip = linear */\n Texture.TRILINEAR_SAMPLINGMODE = 3;\n /** Trilinear is mag = linear and min = linear and mip = linear */\n Texture.LINEAR_LINEAR_MIPLINEAR = 3; // Trilinear is mag = linear and min = linear and mip = linear\n /** mag = nearest and min = nearest and mip = nearest */\n Texture.NEAREST_NEAREST_MIPNEAREST = 4;\n /** mag = nearest and min = linear and mip = nearest */\n Texture.NEAREST_LINEAR_MIPNEAREST = 5;\n /** mag = nearest and min = linear and mip = linear */\n Texture.NEAREST_LINEAR_MIPLINEAR = 6;\n /** mag = nearest and min = linear and mip = none */\n Texture.NEAREST_LINEAR = 7;\n /** mag = nearest and min = nearest and mip = none */\n Texture.NEAREST_NEAREST = 1;\n /** mag = linear and min = nearest and mip = nearest */\n Texture.LINEAR_NEAREST_MIPNEAREST = 9;\n /** mag = linear and min = nearest and mip = linear */\n Texture.LINEAR_NEAREST_MIPLINEAR = 10;\n /** mag = linear and min = linear and mip = none */\n Texture.LINEAR_LINEAR = 2;\n /** mag = linear and min = nearest and mip = none */\n Texture.LINEAR_NEAREST = 12;\n /** Explicit coordinates mode */\n Texture.EXPLICIT_MODE = 0;\n /** Spherical coordinates mode */\n Texture.SPHERICAL_MODE = 1;\n /** Planar coordinates mode */\n Texture.PLANAR_MODE = 2;\n /** Cubic coordinates mode */\n Texture.CUBIC_MODE = 3;\n /** Projection coordinates mode */\n Texture.PROJECTION_MODE = 4;\n /** Inverse Cubic coordinates mode */\n Texture.SKYBOX_MODE = 5;\n /** Inverse Cubic coordinates mode */\n Texture.INVCUBIC_MODE = 6;\n /** Equirectangular coordinates mode */\n Texture.EQUIRECTANGULAR_MODE = 7;\n /** Equirectangular Fixed coordinates mode */\n Texture.FIXED_EQUIRECTANGULAR_MODE = 8;\n /** Equirectangular Fixed Mirrored coordinates mode */\n Texture.FIXED_EQUIRECTANGULAR_MIRRORED_MODE = 9;\n /** Texture is not repeating outside of 0..1 UVs */\n Texture.CLAMP_ADDRESSMODE = 0;\n /** Texture is repeating outside of 0..1 UVs */\n Texture.WRAP_ADDRESSMODE = 1;\n /** Texture is repeating and mirrored */\n Texture.MIRROR_ADDRESSMODE = 2;\n /**\n * Gets or sets a boolean which defines if the texture url must be build from the serialized URL instead of just using the name and loading them side by side with the scene file\n */\n Texture.UseSerializedUrlIfAny = false;\n __decorate$1([\n serialize()\n ], Texture.prototype, \"url\", void 0);\n __decorate$1([\n serialize()\n ], Texture.prototype, \"uOffset\", void 0);\n __decorate$1([\n serialize()\n ], Texture.prototype, \"vOffset\", void 0);\n __decorate$1([\n serialize()\n ], Texture.prototype, \"uScale\", void 0);\n __decorate$1([\n serialize()\n ], Texture.prototype, \"vScale\", void 0);\n __decorate$1([\n serialize()\n ], Texture.prototype, \"uAng\", void 0);\n __decorate$1([\n serialize()\n ], Texture.prototype, \"vAng\", void 0);\n __decorate$1([\n serialize()\n ], Texture.prototype, \"wAng\", void 0);\n __decorate$1([\n serialize()\n ], Texture.prototype, \"uRotationCenter\", void 0);\n __decorate$1([\n serialize()\n ], Texture.prototype, \"vRotationCenter\", void 0);\n __decorate$1([\n serialize()\n ], Texture.prototype, \"wRotationCenter\", void 0);\n __decorate$1([\n serialize()\n ], Texture.prototype, \"homogeneousRotationInUVTransform\", void 0);\n __decorate$1([\n serialize()\n ], Texture.prototype, \"isBlocking\", null);\n // References the dependencies.\n RegisterClass(\"BABYLON.Texture\", Texture);\n SerializationHelper._TextureParser = Texture.Parse;\n\n /**\n * Configuration needed for prepass-capable materials\n */\n class PrePassConfiguration {\n constructor() {\n /**\n * Previous world matrices of meshes carrying this material\n * Used for computing velocity\n */\n this.previousWorldMatrices = {};\n /**\n * Previous bones of meshes carrying this material\n * Used for computing velocity\n */\n this.previousBones = {};\n }\n /**\n * Add the required uniforms to the current list.\n * @param uniforms defines the current uniform list.\n */\n static AddUniforms(uniforms) {\n uniforms.push(\"previousWorld\", \"previousViewProjection\", \"mPreviousBones\");\n }\n /**\n * Add the required samplers to the current list.\n * @param samplers defines the current sampler list.\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n static AddSamplers(samplers) {\n // pass\n }\n /**\n * Binds the material data.\n * @param effect defines the effect to update\n * @param scene defines the scene the material belongs to.\n * @param mesh The mesh\n * @param world World matrix of this mesh\n * @param isFrozen Is the material frozen\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n bindForSubMesh(effect, scene, mesh, world, isFrozen) {\n if (scene.prePassRenderer && scene.prePassRenderer.enabled && scene.prePassRenderer.currentRTisSceneRT) {\n if (scene.prePassRenderer.getIndex(2) !== -1) {\n if (!this.previousWorldMatrices[mesh.uniqueId]) {\n this.previousWorldMatrices[mesh.uniqueId] = world.clone();\n }\n if (!this.previousViewProjection) {\n this.previousViewProjection = scene.getTransformMatrix().clone();\n this.currentViewProjection = scene.getTransformMatrix().clone();\n }\n const engine = scene.getEngine();\n if (this.currentViewProjection.updateFlag !== scene.getTransformMatrix().updateFlag) {\n // First update of the prepass configuration for this rendering pass\n this._lastUpdateFrameId = engine.frameId;\n this.previousViewProjection.copyFrom(this.currentViewProjection);\n this.currentViewProjection.copyFrom(scene.getTransformMatrix());\n }\n else if (this._lastUpdateFrameId !== engine.frameId) {\n // The scene transformation did not change from the previous frame (so no camera motion), we must update previousViewProjection accordingly\n this._lastUpdateFrameId = engine.frameId;\n this.previousViewProjection.copyFrom(this.currentViewProjection);\n }\n effect.setMatrix(\"previousWorld\", this.previousWorldMatrices[mesh.uniqueId]);\n effect.setMatrix(\"previousViewProjection\", this.previousViewProjection);\n this.previousWorldMatrices[mesh.uniqueId] = world.clone();\n }\n }\n }\n }\n\n /**\n * Base class of materials working in push mode in babylon JS\n * @internal\n */\n class PushMaterial extends Material {\n constructor(name, scene, storeEffectOnSubMeshes = true) {\n super(name, scene);\n this._normalMatrix = new Matrix();\n this._storeEffectOnSubMeshes = storeEffectOnSubMeshes;\n }\n getEffect() {\n return this._storeEffectOnSubMeshes ? this._activeEffect : super.getEffect();\n }\n isReady(mesh, useInstances) {\n if (!mesh) {\n return false;\n }\n if (!this._storeEffectOnSubMeshes) {\n return true;\n }\n if (!mesh.subMeshes || mesh.subMeshes.length === 0) {\n return true;\n }\n return this.isReadyForSubMesh(mesh, mesh.subMeshes[0], useInstances);\n }\n _isReadyForSubMesh(subMesh) {\n const defines = subMesh.materialDefines;\n if (!this.checkReadyOnEveryCall && subMesh.effect && defines) {\n if (defines._renderId === this.getScene().getRenderId()) {\n return true;\n }\n }\n return false;\n }\n /**\n * Binds the given world matrix to the active effect\n *\n * @param world the matrix to bind\n */\n bindOnlyWorldMatrix(world) {\n this._activeEffect.setMatrix(\"world\", world);\n }\n /**\n * Binds the given normal matrix to the active effect\n *\n * @param normalMatrix the matrix to bind\n */\n bindOnlyNormalMatrix(normalMatrix) {\n this._activeEffect.setMatrix(\"normalMatrix\", normalMatrix);\n }\n bind(world, mesh) {\n if (!mesh) {\n return;\n }\n this.bindForSubMesh(world, mesh, mesh.subMeshes[0]);\n }\n _afterBind(mesh, effect = null) {\n super._afterBind(mesh, effect);\n this.getScene()._cachedEffect = effect;\n if (effect) {\n effect._forceRebindOnNextCall = false;\n }\n }\n _mustRebind(scene, effect, visibility = 1) {\n return scene.isCachedMaterialInvalid(this, effect, visibility);\n }\n dispose(forceDisposeEffect, forceDisposeTextures, notBoundToMesh) {\n this._activeEffect = undefined;\n super.dispose(forceDisposeEffect, forceDisposeTextures, notBoundToMesh);\n }\n }\n\n /**\n * This groups all the flags used to control the materials channel.\n */\n class MaterialFlags {\n /**\n * Are diffuse textures enabled in the application.\n */\n static get DiffuseTextureEnabled() {\n return this._DiffuseTextureEnabled;\n }\n static set DiffuseTextureEnabled(value) {\n if (this._DiffuseTextureEnabled === value) {\n return;\n }\n this._DiffuseTextureEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n /**\n * Are detail textures enabled in the application.\n */\n static get DetailTextureEnabled() {\n return this._DetailTextureEnabled;\n }\n static set DetailTextureEnabled(value) {\n if (this._DetailTextureEnabled === value) {\n return;\n }\n this._DetailTextureEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n /**\n * Are decal maps enabled in the application.\n */\n static get DecalMapEnabled() {\n return this._DecalMapEnabled;\n }\n static set DecalMapEnabled(value) {\n if (this._DecalMapEnabled === value) {\n return;\n }\n this._DecalMapEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n /**\n * Are ambient textures enabled in the application.\n */\n static get AmbientTextureEnabled() {\n return this._AmbientTextureEnabled;\n }\n static set AmbientTextureEnabled(value) {\n if (this._AmbientTextureEnabled === value) {\n return;\n }\n this._AmbientTextureEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n /**\n * Are opacity textures enabled in the application.\n */\n static get OpacityTextureEnabled() {\n return this._OpacityTextureEnabled;\n }\n static set OpacityTextureEnabled(value) {\n if (this._OpacityTextureEnabled === value) {\n return;\n }\n this._OpacityTextureEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n /**\n * Are reflection textures enabled in the application.\n */\n static get ReflectionTextureEnabled() {\n return this._ReflectionTextureEnabled;\n }\n static set ReflectionTextureEnabled(value) {\n if (this._ReflectionTextureEnabled === value) {\n return;\n }\n this._ReflectionTextureEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n /**\n * Are emissive textures enabled in the application.\n */\n static get EmissiveTextureEnabled() {\n return this._EmissiveTextureEnabled;\n }\n static set EmissiveTextureEnabled(value) {\n if (this._EmissiveTextureEnabled === value) {\n return;\n }\n this._EmissiveTextureEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n /**\n * Are specular textures enabled in the application.\n */\n static get SpecularTextureEnabled() {\n return this._SpecularTextureEnabled;\n }\n static set SpecularTextureEnabled(value) {\n if (this._SpecularTextureEnabled === value) {\n return;\n }\n this._SpecularTextureEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n /**\n * Are bump textures enabled in the application.\n */\n static get BumpTextureEnabled() {\n return this._BumpTextureEnabled;\n }\n static set BumpTextureEnabled(value) {\n if (this._BumpTextureEnabled === value) {\n return;\n }\n this._BumpTextureEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n /**\n * Are lightmap textures enabled in the application.\n */\n static get LightmapTextureEnabled() {\n return this._LightmapTextureEnabled;\n }\n static set LightmapTextureEnabled(value) {\n if (this._LightmapTextureEnabled === value) {\n return;\n }\n this._LightmapTextureEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n /**\n * Are refraction textures enabled in the application.\n */\n static get RefractionTextureEnabled() {\n return this._RefractionTextureEnabled;\n }\n static set RefractionTextureEnabled(value) {\n if (this._RefractionTextureEnabled === value) {\n return;\n }\n this._RefractionTextureEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n /**\n * Are color grading textures enabled in the application.\n */\n static get ColorGradingTextureEnabled() {\n return this._ColorGradingTextureEnabled;\n }\n static set ColorGradingTextureEnabled(value) {\n if (this._ColorGradingTextureEnabled === value) {\n return;\n }\n this._ColorGradingTextureEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n /**\n * Are fresnels enabled in the application.\n */\n static get FresnelEnabled() {\n return this._FresnelEnabled;\n }\n static set FresnelEnabled(value) {\n if (this._FresnelEnabled === value) {\n return;\n }\n this._FresnelEnabled = value;\n Engine.MarkAllMaterialsAsDirty(4);\n }\n /**\n * Are clear coat textures enabled in the application.\n */\n static get ClearCoatTextureEnabled() {\n return this._ClearCoatTextureEnabled;\n }\n static set ClearCoatTextureEnabled(value) {\n if (this._ClearCoatTextureEnabled === value) {\n return;\n }\n this._ClearCoatTextureEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n /**\n * Are clear coat bump textures enabled in the application.\n */\n static get ClearCoatBumpTextureEnabled() {\n return this._ClearCoatBumpTextureEnabled;\n }\n static set ClearCoatBumpTextureEnabled(value) {\n if (this._ClearCoatBumpTextureEnabled === value) {\n return;\n }\n this._ClearCoatBumpTextureEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n /**\n * Are clear coat tint textures enabled in the application.\n */\n static get ClearCoatTintTextureEnabled() {\n return this._ClearCoatTintTextureEnabled;\n }\n static set ClearCoatTintTextureEnabled(value) {\n if (this._ClearCoatTintTextureEnabled === value) {\n return;\n }\n this._ClearCoatTintTextureEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n /**\n * Are sheen textures enabled in the application.\n */\n static get SheenTextureEnabled() {\n return this._SheenTextureEnabled;\n }\n static set SheenTextureEnabled(value) {\n if (this._SheenTextureEnabled === value) {\n return;\n }\n this._SheenTextureEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n /**\n * Are anisotropic textures enabled in the application.\n */\n static get AnisotropicTextureEnabled() {\n return this._AnisotropicTextureEnabled;\n }\n static set AnisotropicTextureEnabled(value) {\n if (this._AnisotropicTextureEnabled === value) {\n return;\n }\n this._AnisotropicTextureEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n /**\n * Are thickness textures enabled in the application.\n */\n static get ThicknessTextureEnabled() {\n return this._ThicknessTextureEnabled;\n }\n static set ThicknessTextureEnabled(value) {\n if (this._ThicknessTextureEnabled === value) {\n return;\n }\n this._ThicknessTextureEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n /**\n * Are refraction intensity textures enabled in the application.\n */\n static get RefractionIntensityTextureEnabled() {\n return this._ThicknessTextureEnabled;\n }\n static set RefractionIntensityTextureEnabled(value) {\n if (this._RefractionIntensityTextureEnabled === value) {\n return;\n }\n this._RefractionIntensityTextureEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n /**\n * Are translucency intensity textures enabled in the application.\n */\n static get TranslucencyIntensityTextureEnabled() {\n return this._ThicknessTextureEnabled;\n }\n static set TranslucencyIntensityTextureEnabled(value) {\n if (this._TranslucencyIntensityTextureEnabled === value) {\n return;\n }\n this._TranslucencyIntensityTextureEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n /**\n * Are translucency intensity textures enabled in the application.\n */\n static get IridescenceTextureEnabled() {\n return this._IridescenceTextureEnabled;\n }\n static set IridescenceTextureEnabled(value) {\n if (this._IridescenceTextureEnabled === value) {\n return;\n }\n this._IridescenceTextureEnabled = value;\n Engine.MarkAllMaterialsAsDirty(1);\n }\n }\n // Flags used to enable or disable a type of texture for all Standard Materials\n MaterialFlags._DiffuseTextureEnabled = true;\n MaterialFlags._DetailTextureEnabled = true;\n MaterialFlags._DecalMapEnabled = true;\n MaterialFlags._AmbientTextureEnabled = true;\n MaterialFlags._OpacityTextureEnabled = true;\n MaterialFlags._ReflectionTextureEnabled = true;\n MaterialFlags._EmissiveTextureEnabled = true;\n MaterialFlags._SpecularTextureEnabled = true;\n MaterialFlags._BumpTextureEnabled = true;\n MaterialFlags._LightmapTextureEnabled = true;\n MaterialFlags._RefractionTextureEnabled = true;\n MaterialFlags._ColorGradingTextureEnabled = true;\n MaterialFlags._FresnelEnabled = true;\n MaterialFlags._ClearCoatTextureEnabled = true;\n MaterialFlags._ClearCoatBumpTextureEnabled = true;\n MaterialFlags._ClearCoatTintTextureEnabled = true;\n MaterialFlags._SheenTextureEnabled = true;\n MaterialFlags._AnisotropicTextureEnabled = true;\n MaterialFlags._ThicknessTextureEnabled = true;\n MaterialFlags._RefractionIntensityTextureEnabled = true;\n MaterialFlags._TranslucencyIntensityTextureEnabled = true;\n MaterialFlags._IridescenceTextureEnabled = true;\n\n // Do not edit.\n const name = \"decalFragmentDeclaration\";\n const shader = `#ifdef DECAL\nuniform vec4 vDecalInfos;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name] = shader;\n\n // Do not edit.\n const name$1 = \"defaultFragmentDeclaration\";\n const shader$1 = `uniform vec4 vEyePosition;uniform vec4 vDiffuseColor;\n#ifdef SPECULARTERM\nuniform vec4 vSpecularColor;\n#endif\nuniform vec3 vEmissiveColor;uniform vec3 vAmbientColor;uniform float visibility;\n#ifdef DIFFUSE\nuniform vec2 vDiffuseInfos;\n#endif\n#ifdef AMBIENT\nuniform vec2 vAmbientInfos;\n#endif\n#ifdef OPACITY \nuniform vec2 vOpacityInfos;\n#endif\n#ifdef EMISSIVE\nuniform vec2 vEmissiveInfos;\n#endif\n#ifdef LIGHTMAP\nuniform vec2 vLightmapInfos;\n#endif\n#ifdef BUMP\nuniform vec3 vBumpInfos;uniform vec2 vTangentSpaceParams;\n#endif\n#ifdef ALPHATEST\nuniform float alphaCutOff;\n#endif\n#if defined(REFLECTIONMAP_SPHERICAL) || defined(REFLECTIONMAP_PROJECTION) || defined(REFRACTION) || defined(PREPASS)\nuniform mat4 view;\n#endif\n#ifdef REFRACTION\nuniform vec4 vRefractionInfos;\n#ifndef REFRACTIONMAP_3D\nuniform mat4 refractionMatrix;\n#endif\n#ifdef REFRACTIONFRESNEL\nuniform vec4 refractionLeftColor;uniform vec4 refractionRightColor;\n#endif\n#if defined(USE_LOCAL_REFRACTIONMAP_CUBIC) && defined(REFRACTIONMAP_3D)\nuniform vec3 vRefractionPosition;uniform vec3 vRefractionSize; \n#endif\n#endif\n#if defined(SPECULAR) && defined(SPECULARTERM)\nuniform vec2 vSpecularInfos;\n#endif\n#ifdef DIFFUSEFRESNEL\nuniform vec4 diffuseLeftColor;uniform vec4 diffuseRightColor;\n#endif\n#ifdef OPACITYFRESNEL\nuniform vec4 opacityParts;\n#endif\n#ifdef EMISSIVEFRESNEL\nuniform vec4 emissiveLeftColor;uniform vec4 emissiveRightColor;\n#endif\n#ifdef REFLECTION\nuniform vec2 vReflectionInfos;\n#if defined(REFLECTIONMAP_PLANAR) || defined(REFLECTIONMAP_CUBIC) || defined(REFLECTIONMAP_PROJECTION) || defined(REFLECTIONMAP_EQUIRECTANGULAR) || defined(REFLECTIONMAP_SPHERICAL) || defined(REFLECTIONMAP_SKYBOX)\nuniform mat4 reflectionMatrix;\n#endif\n#ifndef REFLECTIONMAP_SKYBOX\n#if defined(USE_LOCAL_REFLECTIONMAP_CUBIC) && defined(REFLECTIONMAP_CUBIC)\nuniform vec3 vReflectionPosition;uniform vec3 vReflectionSize; \n#endif\n#endif\n#ifdef REFLECTIONFRESNEL\nuniform vec4 reflectionLeftColor;uniform vec4 reflectionRightColor;\n#endif\n#endif\n#ifdef DETAIL\nuniform vec4 vDetailInfos;\n#endif\n#include\n#define ADDITIONAL_FRAGMENT_DECLARATION\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1] = shader$1;\n\n // Do not edit.\n const name$2 = \"sceneUboDeclaration\";\n const shader$2 = `layout(std140,column_major) uniform;uniform Scene {mat4 viewProjection;\n#ifdef MULTIVIEW\nmat4 viewProjectionR;\n#endif \nmat4 view;mat4 projection;vec4 vEyePosition;};\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$2] = shader$2;\n\n // Do not edit.\n const name$3 = \"meshUboDeclaration\";\n const shader$3 = `#ifdef WEBGL2\nuniform mat4 world;uniform float visibility;\n#else\nlayout(std140,column_major) uniform;uniform Mesh\n{mat4 world;float visibility;};\n#endif\n#define WORLD_UBO\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$3] = shader$3;\n\n // Do not edit.\n const name$4 = \"defaultUboDeclaration\";\n const shader$4 = `layout(std140,column_major) uniform;uniform Material\n{vec4 diffuseLeftColor;vec4 diffuseRightColor;vec4 opacityParts;vec4 reflectionLeftColor;vec4 reflectionRightColor;vec4 refractionLeftColor;vec4 refractionRightColor;vec4 emissiveLeftColor;vec4 emissiveRightColor;vec2 vDiffuseInfos;vec2 vAmbientInfos;vec2 vOpacityInfos;vec2 vReflectionInfos;vec3 vReflectionPosition;vec3 vReflectionSize;vec2 vEmissiveInfos;vec2 vLightmapInfos;vec2 vSpecularInfos;vec3 vBumpInfos;mat4 diffuseMatrix;mat4 ambientMatrix;mat4 opacityMatrix;mat4 reflectionMatrix;mat4 emissiveMatrix;mat4 lightmapMatrix;mat4 specularMatrix;mat4 bumpMatrix;vec2 vTangentSpaceParams;float pointSize;float alphaCutOff;mat4 refractionMatrix;vec4 vRefractionInfos;vec3 vRefractionPosition;vec3 vRefractionSize;vec4 vSpecularColor;vec3 vEmissiveColor;vec4 vDiffuseColor;vec3 vAmbientColor;\n#define ADDITIONAL_UBO_DECLARATION\n};\n#include\n#include\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$4] = shader$4;\n\n // Do not edit.\n const name$5 = \"prePassDeclaration\";\n const shader$5 = `#ifdef PREPASS\n#extension GL_EXT_draw_buffers : require\nlayout(location=0) out highp vec4 glFragData[{X}];highp vec4 gl_FragColor;\n#ifdef PREPASS_DEPTH\nvarying highp vec3 vViewPos;\n#endif\n#ifdef PREPASS_VELOCITY\nvarying highp vec4 vCurrentPosition;varying highp vec4 vPreviousPosition;\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$5] = shader$5;\n\n // Do not edit.\n const name$6 = \"oitDeclaration\";\n const shader$6 = `#ifdef ORDER_INDEPENDENT_TRANSPARENCY\n#extension GL_EXT_draw_buffers : require\nlayout(location=0) out vec2 depth; \nlayout(location=1) out vec4 frontColor;layout(location=2) out vec4 backColor;\n#define MAX_DEPTH 99999.0\nhighp vec4 gl_FragColor;uniform sampler2D oitDepthSampler;uniform sampler2D oitFrontColorSampler;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$6] = shader$6;\n\n // Do not edit.\n const name$7 = \"mainUVVaryingDeclaration\";\n const shader$7 = `#ifdef MAINUV{X}\nvarying vec2 vMainUV{X};\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$7] = shader$7;\n\n // Do not edit.\n const name$8 = \"helperFunctions\";\n const shader$8 = `const float PI=3.1415926535897932384626433832795;const float HALF_MIN=5.96046448e-08; \nconst float LinearEncodePowerApprox=2.2;const float GammaEncodePowerApprox=1.0/LinearEncodePowerApprox;const vec3 LuminanceEncodeApprox=vec3(0.2126,0.7152,0.0722);const float Epsilon=0.0000001;\n#define saturate(x) clamp(x,0.0,1.0)\n#define absEps(x) abs(x)+Epsilon\n#define maxEps(x) max(x,Epsilon)\n#define saturateEps(x) clamp(x,Epsilon,1.0)\nmat3 transposeMat3(mat3 inMatrix) {vec3 i0=inMatrix[0];vec3 i1=inMatrix[1];vec3 i2=inMatrix[2];mat3 outMatrix=mat3(\nvec3(i0.x,i1.x,i2.x),\nvec3(i0.y,i1.y,i2.y),\nvec3(i0.z,i1.z,i2.z)\n);return outMatrix;}\nmat3 inverseMat3(mat3 inMatrix) {float a00=inMatrix[0][0],a01=inMatrix[0][1],a02=inMatrix[0][2];float a10=inMatrix[1][0],a11=inMatrix[1][1],a12=inMatrix[1][2];float a20=inMatrix[2][0],a21=inMatrix[2][1],a22=inMatrix[2][2];float b01=a22*a11-a12*a21;float b11=-a22*a10+a12*a20;float b21=a21*a10-a11*a20;float det=a00*b01+a01*b11+a02*b21;return mat3(b01,(-a22*a01+a02*a21),(a12*a01-a02*a11),\nb11,(a22*a00-a02*a20),(-a12*a00+a02*a10),\nb21,(-a21*a00+a01*a20),(a11*a00-a01*a10))/det;}\n#if USE_EXACT_SRGB_CONVERSIONS\nvec3 toLinearSpaceExact(vec3 color)\n{vec3 nearZeroSection=0.0773993808*color;vec3 remainingSection=pow(0.947867299*(color+vec3(0.055)),vec3(2.4));\n#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE)\nreturn mix(remainingSection,nearZeroSection,lessThanEqual(color,vec3(0.04045)));\n#else\nreturn\nvec3(\ncolor.r<=0.04045 ? nearZeroSection.r : remainingSection.r,\ncolor.g<=0.04045 ? nearZeroSection.g : remainingSection.g,\ncolor.b<=0.04045 ? nearZeroSection.b : remainingSection.b);\n#endif\n}\nvec3 toGammaSpaceExact(vec3 color)\n{vec3 nearZeroSection=12.92*color;vec3 remainingSection=1.055*pow(color,vec3(0.41666))-vec3(0.055);\n#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE)\nreturn mix(remainingSection,nearZeroSection,lessThanEqual(color,vec3(0.0031308)));\n#else\nreturn\nvec3(\ncolor.r<=0.0031308 ? nearZeroSection.r : remainingSection.r,\ncolor.g<=0.0031308 ? nearZeroSection.g : remainingSection.g,\ncolor.b<=0.0031308 ? nearZeroSection.b : remainingSection.b);\n#endif\n}\n#endif\nfloat toLinearSpace(float color)\n{\n#if USE_EXACT_SRGB_CONVERSIONS\nfloat nearZeroSection=0.0773993808*color;float remainingSection=pow(0.947867299*(color+0.055),2.4);return color<=0.04045 ? nearZeroSection : remainingSection;\n#else\nreturn pow(color,LinearEncodePowerApprox);\n#endif\n}\nvec3 toLinearSpace(vec3 color)\n{\n#if USE_EXACT_SRGB_CONVERSIONS\nreturn toLinearSpaceExact(color);\n#else\nreturn pow(color,vec3(LinearEncodePowerApprox));\n#endif\n}\nvec4 toLinearSpace(vec4 color)\n{\n#if USE_EXACT_SRGB_CONVERSIONS\nreturn vec4(toLinearSpaceExact(color.rgb),color.a);\n#else\nreturn vec4(pow(color.rgb,vec3(LinearEncodePowerApprox)),color.a);\n#endif\n}\nfloat toGammaSpace(float color)\n{\n#if USE_EXACT_SRGB_CONVERSIONS\nfloat nearZeroSection=12.92*color;float remainingSection=1.055*pow(color,0.41666)-0.055;return color<=0.0031308 ? nearZeroSection : remainingSection;\n#else\nreturn pow(color,GammaEncodePowerApprox);\n#endif\n}\nvec3 toGammaSpace(vec3 color)\n{\n#if USE_EXACT_SRGB_CONVERSIONS\nreturn toGammaSpaceExact(color);\n#else\nreturn pow(color,vec3(GammaEncodePowerApprox));\n#endif\n}\nvec4 toGammaSpace(vec4 color)\n{\n#if USE_EXACT_SRGB_CONVERSIONS\nreturn vec4(toGammaSpaceExact(color.rgb),color.a);\n#else\nreturn vec4(pow(color.rgb,vec3(GammaEncodePowerApprox)),color.a);\n#endif\n}\nfloat square(float value)\n{return value*value;}\nvec3 square(vec3 value)\n{return value*value;}\nfloat pow5(float value) {float sq=value*value;return sq*sq*value;}\nfloat getLuminance(vec3 color)\n{return clamp(dot(color,LuminanceEncodeApprox),0.,1.);}\nfloat getRand(vec2 seed) {return fract(sin(dot(seed.xy ,vec2(12.9898,78.233)))*43758.5453);}\nfloat dither(vec2 seed,float varianceAmount) {float rand=getRand(seed);float normVariance=varianceAmount/255.0;float dither=mix(-normVariance,normVariance,rand);return dither;}\nconst float rgbdMaxRange=255.0;vec4 toRGBD(vec3 color) {float maxRGB=maxEps(max(color.r,max(color.g,color.b)));float D =max(rgbdMaxRange/maxRGB,1.);D =clamp(floor(D)/255.0,0.,1.);vec3 rgb=color.rgb*D;rgb=toGammaSpace(rgb);return vec4(clamp(rgb,0.,1.),D); }\nvec3 fromRGBD(vec4 rgbd) {rgbd.rgb=toLinearSpace(rgbd.rgb);return rgbd.rgb/rgbd.a;}\nvec3 parallaxCorrectNormal( vec3 vertexPos,vec3 origVec,vec3 cubeSize,vec3 cubePos ) {vec3 invOrigVec=vec3(1.0,1.0,1.0)/origVec;vec3 halfSize=cubeSize*0.5;vec3 intersecAtMaxPlane=(cubePos+halfSize-vertexPos)*invOrigVec;vec3 intersecAtMinPlane=(cubePos-halfSize-vertexPos)*invOrigVec;vec3 largestIntersec=max(intersecAtMaxPlane,intersecAtMinPlane);float distance=min(min(largestIntersec.x,largestIntersec.y),largestIntersec.z);vec3 intersectPositionWS=vertexPos+origVec*distance;return intersectPositionWS-cubePos;}\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$8] = shader$8;\n\n // Do not edit.\n const name$9 = \"lightFragmentDeclaration\";\n const shader$9 = `#ifdef LIGHT{X}\nuniform vec4 vLightData{X};uniform vec4 vLightDiffuse{X};\n#ifdef SPECULARTERM\nuniform vec4 vLightSpecular{X};\n#else\nvec4 vLightSpecular{X}=vec4(0.);\n#endif\n#ifdef SHADOW{X}\n#ifdef SHADOWCSM{X}\nuniform mat4 lightMatrix{X}[SHADOWCSMNUM_CASCADES{X}];uniform float viewFrustumZ{X}[SHADOWCSMNUM_CASCADES{X}];uniform float frustumLengths{X}[SHADOWCSMNUM_CASCADES{X}];uniform float cascadeBlendFactor{X};varying vec4 vPositionFromLight{X}[SHADOWCSMNUM_CASCADES{X}];varying float vDepthMetric{X}[SHADOWCSMNUM_CASCADES{X}];varying vec4 vPositionFromCamera{X};\n#if defined(SHADOWPCSS{X})\nuniform highp sampler2DArrayShadow shadowSampler{X};uniform highp sampler2DArray depthSampler{X};uniform vec2 lightSizeUVCorrection{X}[SHADOWCSMNUM_CASCADES{X}];uniform float depthCorrection{X}[SHADOWCSMNUM_CASCADES{X}];uniform float penumbraDarkness{X};\n#elif defined(SHADOWPCF{X})\nuniform highp sampler2DArrayShadow shadowSampler{X};\n#else\nuniform highp sampler2DArray shadowSampler{X};\n#endif\n#ifdef SHADOWCSMDEBUG{X}\nconst vec3 vCascadeColorsMultiplier{X}[8]=vec3[8]\n(\nvec3 ( 1.5,0.0,0.0 ),\nvec3 ( 0.0,1.5,0.0 ),\nvec3 ( 0.0,0.0,5.5 ),\nvec3 ( 1.5,0.0,5.5 ),\nvec3 ( 1.5,1.5,0.0 ),\nvec3 ( 1.0,1.0,1.0 ),\nvec3 ( 0.0,1.0,5.5 ),\nvec3 ( 0.5,3.5,0.75 )\n);vec3 shadowDebug{X};\n#endif\n#ifdef SHADOWCSMUSESHADOWMAXZ{X}\nint index{X}=-1;\n#else\nint index{X}=SHADOWCSMNUM_CASCADES{X}-1;\n#endif\nfloat diff{X}=0.;\n#elif defined(SHADOWCUBE{X})\nuniform samplerCube shadowSampler{X};\n#else\nvarying vec4 vPositionFromLight{X};varying float vDepthMetric{X};\n#if defined(SHADOWPCSS{X})\nuniform highp sampler2DShadow shadowSampler{X};uniform highp sampler2D depthSampler{X};\n#elif defined(SHADOWPCF{X})\nuniform highp sampler2DShadow shadowSampler{X};\n#else\nuniform sampler2D shadowSampler{X};\n#endif\nuniform mat4 lightMatrix{X};\n#endif\nuniform vec4 shadowsInfo{X};uniform vec2 depthValues{X};\n#endif\n#ifdef SPOTLIGHT{X}\nuniform vec4 vLightDirection{X};uniform vec4 vLightFalloff{X};\n#elif defined(POINTLIGHT{X})\nuniform vec4 vLightFalloff{X};\n#elif defined(HEMILIGHT{X})\nuniform vec3 vLightGround{X};\n#endif\n#ifdef PROJECTEDLIGHTTEXTURE{X}\nuniform mat4 textureProjectionMatrix{X};uniform sampler2D projectionLightSampler{X};\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$9] = shader$9;\n\n // Do not edit.\n const name$a = \"lightUboDeclaration\";\n const shader$a = `#ifdef LIGHT{X}\nuniform Light{X}\n{vec4 vLightData;vec4 vLightDiffuse;vec4 vLightSpecular;\n#ifdef SPOTLIGHT{X}\nvec4 vLightDirection;vec4 vLightFalloff;\n#elif defined(POINTLIGHT{X})\nvec4 vLightFalloff;\n#elif defined(HEMILIGHT{X})\nvec3 vLightGround;\n#endif\nvec4 shadowsInfo;vec2 depthValues;} light{X};\n#ifdef PROJECTEDLIGHTTEXTURE{X}\nuniform mat4 textureProjectionMatrix{X};uniform sampler2D projectionLightSampler{X};\n#endif\n#ifdef SHADOW{X}\n#ifdef SHADOWCSM{X}\nuniform mat4 lightMatrix{X}[SHADOWCSMNUM_CASCADES{X}];uniform float viewFrustumZ{X}[SHADOWCSMNUM_CASCADES{X}];uniform float frustumLengths{X}[SHADOWCSMNUM_CASCADES{X}];uniform float cascadeBlendFactor{X};varying vec4 vPositionFromLight{X}[SHADOWCSMNUM_CASCADES{X}];varying float vDepthMetric{X}[SHADOWCSMNUM_CASCADES{X}];varying vec4 vPositionFromCamera{X};\n#if defined(SHADOWPCSS{X})\nuniform highp sampler2DArrayShadow shadowSampler{X};uniform highp sampler2DArray depthSampler{X};uniform vec2 lightSizeUVCorrection{X}[SHADOWCSMNUM_CASCADES{X}];uniform float depthCorrection{X}[SHADOWCSMNUM_CASCADES{X}];uniform float penumbraDarkness{X};\n#elif defined(SHADOWPCF{X})\nuniform highp sampler2DArrayShadow shadowSampler{X};\n#else\nuniform highp sampler2DArray shadowSampler{X};\n#endif\n#ifdef SHADOWCSMDEBUG{X}\nconst vec3 vCascadeColorsMultiplier{X}[8]=vec3[8]\n(\nvec3 ( 1.5,0.0,0.0 ),\nvec3 ( 0.0,1.5,0.0 ),\nvec3 ( 0.0,0.0,5.5 ),\nvec3 ( 1.5,0.0,5.5 ),\nvec3 ( 1.5,1.5,0.0 ),\nvec3 ( 1.0,1.0,1.0 ),\nvec3 ( 0.0,1.0,5.5 ),\nvec3 ( 0.5,3.5,0.75 )\n);vec3 shadowDebug{X};\n#endif\n#ifdef SHADOWCSMUSESHADOWMAXZ{X}\nint index{X}=-1;\n#else\nint index{X}=SHADOWCSMNUM_CASCADES{X}-1;\n#endif\nfloat diff{X}=0.;\n#elif defined(SHADOWCUBE{X})\nuniform samplerCube shadowSampler{X}; \n#else\nvarying vec4 vPositionFromLight{X};varying float vDepthMetric{X};\n#if defined(SHADOWPCSS{X})\nuniform highp sampler2DShadow shadowSampler{X};uniform highp sampler2D depthSampler{X};\n#elif defined(SHADOWPCF{X})\nuniform highp sampler2DShadow shadowSampler{X};\n#else\nuniform sampler2D shadowSampler{X};\n#endif\nuniform mat4 lightMatrix{X};\n#endif\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$a] = shader$a;\n\n // Do not edit.\n const name$b = \"lightsFragmentFunctions\";\n const shader$b = `struct lightingInfo\n{vec3 diffuse;\n#ifdef SPECULARTERM\nvec3 specular;\n#endif\n#ifdef NDOTL\nfloat ndl;\n#endif\n};lightingInfo computeLighting(vec3 viewDirectionW,vec3 vNormal,vec4 lightData,vec3 diffuseColor,vec3 specularColor,float range,float glossiness) {lightingInfo result;vec3 lightVectorW;float attenuation=1.0;if (lightData.w==0.)\n{vec3 direction=lightData.xyz-vPositionW;attenuation=max(0.,1.0-length(direction)/range);lightVectorW=normalize(direction);}\nelse\n{lightVectorW=normalize(-lightData.xyz);}\nfloat ndl=max(0.,dot(vNormal,lightVectorW));\n#ifdef NDOTL\nresult.ndl=ndl;\n#endif\nresult.diffuse=ndl*diffuseColor*attenuation;\n#ifdef SPECULARTERM\nvec3 angleW=normalize(viewDirectionW+lightVectorW);float specComp=max(0.,dot(vNormal,angleW));specComp=pow(specComp,max(1.,glossiness));result.specular=specComp*specularColor*attenuation;\n#endif\nreturn result;}\nlightingInfo computeSpotLighting(vec3 viewDirectionW,vec3 vNormal,vec4 lightData,vec4 lightDirection,vec3 diffuseColor,vec3 specularColor,float range,float glossiness) {lightingInfo result;vec3 direction=lightData.xyz-vPositionW;vec3 lightVectorW=normalize(direction);float attenuation=max(0.,1.0-length(direction)/range);float cosAngle=max(0.,dot(lightDirection.xyz,-lightVectorW));if (cosAngle>=lightDirection.w)\n{cosAngle=max(0.,pow(cosAngle,lightData.w));attenuation*=cosAngle;float ndl=max(0.,dot(vNormal,lightVectorW));\n#ifdef NDOTL\nresult.ndl=ndl;\n#endif\nresult.diffuse=ndl*diffuseColor*attenuation;\n#ifdef SPECULARTERM\nvec3 angleW=normalize(viewDirectionW+lightVectorW);float specComp=max(0.,dot(vNormal,angleW));specComp=pow(specComp,max(1.,glossiness));result.specular=specComp*specularColor*attenuation;\n#endif\nreturn result;}\nresult.diffuse=vec3(0.);\n#ifdef SPECULARTERM\nresult.specular=vec3(0.);\n#endif\n#ifdef NDOTL\nresult.ndl=0.;\n#endif\nreturn result;}\nlightingInfo computeHemisphericLighting(vec3 viewDirectionW,vec3 vNormal,vec4 lightData,vec3 diffuseColor,vec3 specularColor,vec3 groundColor,float glossiness) {lightingInfo result;float ndl=dot(vNormal,lightData.xyz)*0.5+0.5;\n#ifdef NDOTL\nresult.ndl=ndl;\n#endif\nresult.diffuse=mix(groundColor,diffuseColor,ndl);\n#ifdef SPECULARTERM\nvec3 angleW=normalize(viewDirectionW+lightData.xyz);float specComp=max(0.,dot(vNormal,angleW));specComp=pow(specComp,max(1.,glossiness));result.specular=specComp*specularColor;\n#endif\nreturn result;}\n#define inline\nvec3 computeProjectionTextureDiffuseLighting(sampler2D projectionLightSampler,mat4 textureProjectionMatrix){vec4 strq=textureProjectionMatrix*vec4(vPositionW,1.0);strq/=strq.w;vec3 textureColor=texture2D(projectionLightSampler,strq.xy).rgb;return textureColor;}`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$b] = shader$b;\n\n // Do not edit.\n const name$c = \"shadowsFragmentFunctions\";\n const shader$c = `#ifdef SHADOWS\n#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE)\n#define TEXTUREFUNC(s,c,l) texture2DLodEXT(s,c,l)\n#else\n#define TEXTUREFUNC(s,c,b) texture2D(s,c,b)\n#endif\n#ifndef SHADOWFLOAT\nfloat unpack(vec4 color)\n{const vec4 bit_shift=vec4(1.0/(255.0*255.0*255.0),1.0/(255.0*255.0),1.0/255.0,1.0);return dot(color,bit_shift);}\n#endif\nfloat computeFallOff(float value,vec2 clipSpace,float frustumEdgeFalloff)\n{float mask=smoothstep(1.0-frustumEdgeFalloff,1.00000012,clamp(dot(clipSpace,clipSpace),0.,1.));return mix(value,1.0,mask);}\n#define inline\nfloat computeShadowCube(vec3 lightPosition,samplerCube shadowSampler,float darkness,vec2 depthValues)\n{vec3 directionToLight=vPositionW-lightPosition;float depth=length(directionToLight);depth=(depth+depthValues.x)/(depthValues.y);depth=clamp(depth,0.,1.0);directionToLight=normalize(directionToLight);directionToLight.y=-directionToLight.y;\n#ifndef SHADOWFLOAT\nfloat shadow=unpack(textureCube(shadowSampler,directionToLight));\n#else\nfloat shadow=textureCube(shadowSampler,directionToLight).x;\n#endif\nreturn depth>shadow ? darkness : 1.0;}\n#define inline\nfloat computeShadowWithPoissonSamplingCube(vec3 lightPosition,samplerCube shadowSampler,float mapSize,float darkness,vec2 depthValues)\n{vec3 directionToLight=vPositionW-lightPosition;float depth=length(directionToLight);depth=(depth+depthValues.x)/(depthValues.y);depth=clamp(depth,0.,1.0);directionToLight=normalize(directionToLight);directionToLight.y=-directionToLight.y;float visibility=1.;vec3 poissonDisk[4];poissonDisk[0]=vec3(-1.0,1.0,-1.0);poissonDisk[1]=vec3(1.0,-1.0,-1.0);poissonDisk[2]=vec3(-1.0,-1.0,-1.0);poissonDisk[3]=vec3(1.0,-1.0,1.0);\n#ifndef SHADOWFLOAT\nif (unpack(textureCube(shadowSampler,directionToLight+poissonDisk[0]*mapSize))shadow ? computeFallOff(darkness,clipSpace.xy,frustumEdgeFalloff) : 1.;}\n#endif\n#define inline\nfloat computeShadow(vec4 vPositionFromLight,float depthMetric,sampler2D shadowSampler,float darkness,float frustumEdgeFalloff)\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec2 uv=0.5*clipSpace.xy+vec2(0.5);if (uv.x<0. || uv.x>1.0 || uv.y<0. || uv.y>1.0)\n{return 1.0;}\nelse\n{float shadowPixelDepth=clamp(depthMetric,0.,1.0);\n#ifndef SHADOWFLOAT\nfloat shadow=unpack(TEXTUREFUNC(shadowSampler,uv,0.));\n#else\nfloat shadow=TEXTUREFUNC(shadowSampler,uv,0.).x;\n#endif\nreturn shadowPixelDepth>shadow ? computeFallOff(darkness,clipSpace.xy,frustumEdgeFalloff) : 1.;}}\n#define inline\nfloat computeShadowWithPoissonSampling(vec4 vPositionFromLight,float depthMetric,sampler2D shadowSampler,float mapSize,float darkness,float frustumEdgeFalloff)\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec2 uv=0.5*clipSpace.xy+vec2(0.5);if (uv.x<0. || uv.x>1.0 || uv.y<0. || uv.y>1.0)\n{return 1.0;}\nelse\n{float shadowPixelDepth=clamp(depthMetric,0.,1.0);float visibility=1.;vec2 poissonDisk[4];poissonDisk[0]=vec2(-0.94201624,-0.39906216);poissonDisk[1]=vec2(0.94558609,-0.76890725);poissonDisk[2]=vec2(-0.094184101,-0.92938870);poissonDisk[3]=vec2(0.34495938,0.29387760);\n#ifndef SHADOWFLOAT\nif (unpack(TEXTUREFUNC(shadowSampler,uv+poissonDisk[0]*mapSize,0.))1.0 || uv.y<0. || uv.y>1.0)\n{return 1.0;}\nelse\n{float shadowPixelDepth=clamp(depthMetric,0.,1.0);\n#ifndef SHADOWFLOAT\nfloat shadowMapSample=unpack(TEXTUREFUNC(shadowSampler,uv,0.));\n#else\nfloat shadowMapSample=TEXTUREFUNC(shadowSampler,uv,0.).x;\n#endif\nfloat esm=1.0-clamp(exp(min(87.,depthScale*shadowPixelDepth))*shadowMapSample,0.,1.-darkness);return computeFallOff(esm,clipSpace.xy,frustumEdgeFalloff);}}\n#define inline\nfloat computeShadowWithCloseESM(vec4 vPositionFromLight,float depthMetric,sampler2D shadowSampler,float darkness,float depthScale,float frustumEdgeFalloff)\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec2 uv=0.5*clipSpace.xy+vec2(0.5);if (uv.x<0. || uv.x>1.0 || uv.y<0. || uv.y>1.0)\n{return 1.0;}\nelse\n{float shadowPixelDepth=clamp(depthMetric,0.,1.0); \n#ifndef SHADOWFLOAT\nfloat shadowMapSample=unpack(TEXTUREFUNC(shadowSampler,uv,0.));\n#else\nfloat shadowMapSample=TEXTUREFUNC(shadowSampler,uv,0.).x;\n#endif\nfloat esm=clamp(exp(min(87.,-depthScale*(shadowPixelDepth-shadowMapSample))),darkness,1.);return computeFallOff(esm,clipSpace.xy,frustumEdgeFalloff);}}\n#ifdef IS_NDC_HALF_ZRANGE\n#define ZINCLIP clipSpace.z\n#else\n#define ZINCLIP uvDepth.z\n#endif\n#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE)\n#define GREATEST_LESS_THAN_ONE 0.99999994\n#define inline\nfloat computeShadowWithCSMPCF1(float layer,vec4 vPositionFromLight,float depthMetric,highp sampler2DArrayShadow shadowSampler,float darkness,float frustumEdgeFalloff)\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=clamp(ZINCLIP,0.,GREATEST_LESS_THAN_ONE);vec4 uvDepthLayer=vec4(uvDepth.x,uvDepth.y,layer,uvDepth.z);float shadow=texture2D(shadowSampler,uvDepthLayer);shadow=mix(darkness,1.,shadow);return computeFallOff(shadow,clipSpace.xy,frustumEdgeFalloff);}\n#define inline\nfloat computeShadowWithCSMPCF3(float layer,vec4 vPositionFromLight,float depthMetric,highp sampler2DArrayShadow shadowSampler,vec2 shadowMapSizeAndInverse,float darkness,float frustumEdgeFalloff)\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=clamp(ZINCLIP,0.,GREATEST_LESS_THAN_ONE);vec2 uv=uvDepth.xy*shadowMapSizeAndInverse.x; \nuv+=0.5; \nvec2 st=fract(uv); \nvec2 base_uv=floor(uv)-0.5; \nbase_uv*=shadowMapSizeAndInverse.y; \nvec2 uvw0=3.-2.*st;vec2 uvw1=1.+2.*st;vec2 u=vec2((2.-st.x)/uvw0.x-1.,st.x/uvw1.x+1.)*shadowMapSizeAndInverse.y;vec2 v=vec2((2.-st.y)/uvw0.y-1.,st.y/uvw1.y+1.)*shadowMapSizeAndInverse.y;float shadow=0.;shadow+=uvw0.x*uvw0.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[0],v[0]),layer,uvDepth.z));shadow+=uvw1.x*uvw0.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[1],v[0]),layer,uvDepth.z));shadow+=uvw0.x*uvw1.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[0],v[1]),layer,uvDepth.z));shadow+=uvw1.x*uvw1.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[1],v[1]),layer,uvDepth.z));shadow=shadow/16.;shadow=mix(darkness,1.,shadow);return computeFallOff(shadow,clipSpace.xy,frustumEdgeFalloff);}\n#define inline\nfloat computeShadowWithCSMPCF5(float layer,vec4 vPositionFromLight,float depthMetric,highp sampler2DArrayShadow shadowSampler,vec2 shadowMapSizeAndInverse,float darkness,float frustumEdgeFalloff)\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=clamp(ZINCLIP,0.,GREATEST_LESS_THAN_ONE);vec2 uv=uvDepth.xy*shadowMapSizeAndInverse.x; \nuv+=0.5; \nvec2 st=fract(uv); \nvec2 base_uv=floor(uv)-0.5; \nbase_uv*=shadowMapSizeAndInverse.y; \nvec2 uvw0=4.-3.*st;vec2 uvw1=vec2(7.);vec2 uvw2=1.+3.*st;vec3 u=vec3((3.-2.*st.x)/uvw0.x-2.,(3.+st.x)/uvw1.x,st.x/uvw2.x+2.)*shadowMapSizeAndInverse.y;vec3 v=vec3((3.-2.*st.y)/uvw0.y-2.,(3.+st.y)/uvw1.y,st.y/uvw2.y+2.)*shadowMapSizeAndInverse.y;float shadow=0.;shadow+=uvw0.x*uvw0.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[0],v[0]),layer,uvDepth.z));shadow+=uvw1.x*uvw0.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[1],v[0]),layer,uvDepth.z));shadow+=uvw2.x*uvw0.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[2],v[0]),layer,uvDepth.z));shadow+=uvw0.x*uvw1.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[0],v[1]),layer,uvDepth.z));shadow+=uvw1.x*uvw1.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[1],v[1]),layer,uvDepth.z));shadow+=uvw2.x*uvw1.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[2],v[1]),layer,uvDepth.z));shadow+=uvw0.x*uvw2.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[0],v[2]),layer,uvDepth.z));shadow+=uvw1.x*uvw2.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[1],v[2]),layer,uvDepth.z));shadow+=uvw2.x*uvw2.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[2],v[2]),layer,uvDepth.z));shadow=shadow/144.;shadow=mix(darkness,1.,shadow);return computeFallOff(shadow,clipSpace.xy,frustumEdgeFalloff);}\n#define inline\nfloat computeShadowWithPCF1(vec4 vPositionFromLight,float depthMetric,highp sampler2DShadow shadowSampler,float darkness,float frustumEdgeFalloff)\n{if (depthMetric>1.0 || depthMetric<0.0) {return 1.0;}\nelse\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=ZINCLIP;float shadow=TEXTUREFUNC(shadowSampler,uvDepth,0.);shadow=mix(darkness,1.,shadow);return computeFallOff(shadow,clipSpace.xy,frustumEdgeFalloff);}}\n#define inline\nfloat computeShadowWithPCF3(vec4 vPositionFromLight,float depthMetric,highp sampler2DShadow shadowSampler,vec2 shadowMapSizeAndInverse,float darkness,float frustumEdgeFalloff)\n{if (depthMetric>1.0 || depthMetric<0.0) {return 1.0;}\nelse\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=ZINCLIP;vec2 uv=uvDepth.xy*shadowMapSizeAndInverse.x; \nuv+=0.5; \nvec2 st=fract(uv); \nvec2 base_uv=floor(uv)-0.5; \nbase_uv*=shadowMapSizeAndInverse.y; \nvec2 uvw0=3.-2.*st;vec2 uvw1=1.+2.*st;vec2 u=vec2((2.-st.x)/uvw0.x-1.,st.x/uvw1.x+1.)*shadowMapSizeAndInverse.y;vec2 v=vec2((2.-st.y)/uvw0.y-1.,st.y/uvw1.y+1.)*shadowMapSizeAndInverse.y;float shadow=0.;shadow+=uvw0.x*uvw0.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[0],v[0]),uvDepth.z),0.);shadow+=uvw1.x*uvw0.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[1],v[0]),uvDepth.z),0.);shadow+=uvw0.x*uvw1.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[0],v[1]),uvDepth.z),0.);shadow+=uvw1.x*uvw1.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[1],v[1]),uvDepth.z),0.);shadow=shadow/16.;shadow=mix(darkness,1.,shadow);return computeFallOff(shadow,clipSpace.xy,frustumEdgeFalloff);}}\n#define inline\nfloat computeShadowWithPCF5(vec4 vPositionFromLight,float depthMetric,highp sampler2DShadow shadowSampler,vec2 shadowMapSizeAndInverse,float darkness,float frustumEdgeFalloff)\n{if (depthMetric>1.0 || depthMetric<0.0) {return 1.0;}\nelse\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=ZINCLIP;vec2 uv=uvDepth.xy*shadowMapSizeAndInverse.x; \nuv+=0.5; \nvec2 st=fract(uv); \nvec2 base_uv=floor(uv)-0.5; \nbase_uv*=shadowMapSizeAndInverse.y; \nvec2 uvw0=4.-3.*st;vec2 uvw1=vec2(7.);vec2 uvw2=1.+3.*st;vec3 u=vec3((3.-2.*st.x)/uvw0.x-2.,(3.+st.x)/uvw1.x,st.x/uvw2.x+2.)*shadowMapSizeAndInverse.y;vec3 v=vec3((3.-2.*st.y)/uvw0.y-2.,(3.+st.y)/uvw1.y,st.y/uvw2.y+2.)*shadowMapSizeAndInverse.y;float shadow=0.;shadow+=uvw0.x*uvw0.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[0],v[0]),uvDepth.z),0.);shadow+=uvw1.x*uvw0.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[1],v[0]),uvDepth.z),0.);shadow+=uvw2.x*uvw0.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[2],v[0]),uvDepth.z),0.);shadow+=uvw0.x*uvw1.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[0],v[1]),uvDepth.z),0.);shadow+=uvw1.x*uvw1.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[1],v[1]),uvDepth.z),0.);shadow+=uvw2.x*uvw1.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[2],v[1]),uvDepth.z),0.);shadow+=uvw0.x*uvw2.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[0],v[2]),uvDepth.z),0.);shadow+=uvw1.x*uvw2.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[1],v[2]),uvDepth.z),0.);shadow+=uvw2.x*uvw2.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[2],v[2]),uvDepth.z),0.);shadow=shadow/144.;shadow=mix(darkness,1.,shadow);return computeFallOff(shadow,clipSpace.xy,frustumEdgeFalloff);}}\nconst vec3 PoissonSamplers32[64]=vec3[64](\nvec3(0.06407013,0.05409927,0.),\nvec3(0.7366577,0.5789394,0.),\nvec3(-0.6270542,-0.5320278,0.),\nvec3(-0.4096107,0.8411095,0.),\nvec3(0.6849564,-0.4990818,0.),\nvec3(-0.874181,-0.04579735,0.),\nvec3(0.9989998,0.0009880066,0.),\nvec3(-0.004920578,-0.9151649,0.),\nvec3(0.1805763,0.9747483,0.),\nvec3(-0.2138451,0.2635818,0.),\nvec3(0.109845,0.3884785,0.),\nvec3(0.06876755,-0.3581074,0.),\nvec3(0.374073,-0.7661266,0.),\nvec3(0.3079132,-0.1216763,0.),\nvec3(-0.3794335,-0.8271583,0.),\nvec3(-0.203878,-0.07715034,0.),\nvec3(0.5912697,0.1469799,0.),\nvec3(-0.88069,0.3031784,0.),\nvec3(0.5040108,0.8283722,0.),\nvec3(-0.5844124,0.5494877,0.),\nvec3(0.6017799,-0.1726654,0.),\nvec3(-0.5554981,0.1559997,0.),\nvec3(-0.3016369,-0.3900928,0.),\nvec3(-0.5550632,-0.1723762,0.),\nvec3(0.925029,0.2995041,0.),\nvec3(-0.2473137,0.5538505,0.),\nvec3(0.9183037,-0.2862392,0.),\nvec3(0.2469421,0.6718712,0.),\nvec3(0.3916397,-0.4328209,0.),\nvec3(-0.03576927,-0.6220032,0.),\nvec3(-0.04661255,0.7995201,0.),\nvec3(0.4402924,0.3640312,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.)\n);const vec3 PoissonSamplers64[64]=vec3[64](\nvec3(-0.613392,0.617481,0.),\nvec3(0.170019,-0.040254,0.),\nvec3(-0.299417,0.791925,0.),\nvec3(0.645680,0.493210,0.),\nvec3(-0.651784,0.717887,0.),\nvec3(0.421003,0.027070,0.),\nvec3(-0.817194,-0.271096,0.),\nvec3(-0.705374,-0.668203,0.),\nvec3(0.977050,-0.108615,0.),\nvec3(0.063326,0.142369,0.),\nvec3(0.203528,0.214331,0.),\nvec3(-0.667531,0.326090,0.),\nvec3(-0.098422,-0.295755,0.),\nvec3(-0.885922,0.215369,0.),\nvec3(0.566637,0.605213,0.),\nvec3(0.039766,-0.396100,0.),\nvec3(0.751946,0.453352,0.),\nvec3(0.078707,-0.715323,0.),\nvec3(-0.075838,-0.529344,0.),\nvec3(0.724479,-0.580798,0.),\nvec3(0.222999,-0.215125,0.),\nvec3(-0.467574,-0.405438,0.),\nvec3(-0.248268,-0.814753,0.),\nvec3(0.354411,-0.887570,0.),\nvec3(0.175817,0.382366,0.),\nvec3(0.487472,-0.063082,0.),\nvec3(-0.084078,0.898312,0.),\nvec3(0.488876,-0.783441,0.),\nvec3(0.470016,0.217933,0.),\nvec3(-0.696890,-0.549791,0.),\nvec3(-0.149693,0.605762,0.),\nvec3(0.034211,0.979980,0.),\nvec3(0.503098,-0.308878,0.),\nvec3(-0.016205,-0.872921,0.),\nvec3(0.385784,-0.393902,0.),\nvec3(-0.146886,-0.859249,0.),\nvec3(0.643361,0.164098,0.),\nvec3(0.634388,-0.049471,0.),\nvec3(-0.688894,0.007843,0.),\nvec3(0.464034,-0.188818,0.),\nvec3(-0.440840,0.137486,0.),\nvec3(0.364483,0.511704,0.),\nvec3(0.034028,0.325968,0.),\nvec3(0.099094,-0.308023,0.),\nvec3(0.693960,-0.366253,0.),\nvec3(0.678884,-0.204688,0.),\nvec3(0.001801,0.780328,0.),\nvec3(0.145177,-0.898984,0.),\nvec3(0.062655,-0.611866,0.),\nvec3(0.315226,-0.604297,0.),\nvec3(-0.780145,0.486251,0.),\nvec3(-0.371868,0.882138,0.),\nvec3(0.200476,0.494430,0.),\nvec3(-0.494552,-0.711051,0.),\nvec3(0.612476,0.705252,0.),\nvec3(-0.578845,-0.768792,0.),\nvec3(-0.772454,-0.090976,0.),\nvec3(0.504440,0.372295,0.),\nvec3(0.155736,0.065157,0.),\nvec3(0.391522,0.849605,0.),\nvec3(-0.620106,-0.328104,0.),\nvec3(0.789239,-0.419965,0.),\nvec3(-0.545396,0.538133,0.),\nvec3(-0.178564,-0.596057,0.)\n);\n#define inline\nfloat computeShadowWithCSMPCSS(float layer,vec4 vPositionFromLight,float depthMetric,highp sampler2DArray depthSampler,highp sampler2DArrayShadow shadowSampler,float shadowMapSizeInverse,float lightSizeUV,float darkness,float frustumEdgeFalloff,int searchTapCount,int pcfTapCount,vec3[64] poissonSamplers,vec2 lightSizeUVCorrection,float depthCorrection,float penumbraDarkness)\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=clamp(ZINCLIP,0.,GREATEST_LESS_THAN_ONE);vec4 uvDepthLayer=vec4(uvDepth.x,uvDepth.y,layer,uvDepth.z);float blockerDepth=0.0;float sumBlockerDepth=0.0;float numBlocker=0.0;for (int i=0; i1.0 || depthMetric<0.0) {return 1.0;}\nelse\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=ZINCLIP;float blockerDepth=0.0;float sumBlockerDepth=0.0;float numBlocker=0.0;for (int i=0; i(_DEFINENAME_,BUMP,_VARYINGNAME_,Bump,_SAMPLERNAME_,bump)\n#endif\n#if defined(DETAIL)\n#include(_DEFINENAME_,DETAIL,_VARYINGNAME_,Detail,_SAMPLERNAME_,detail)\n#endif\n#if defined(BUMP) && defined(PARALLAX)\nconst float minSamples=4.;const float maxSamples=15.;const int iMaxSamples=15;vec2 parallaxOcclusion(vec3 vViewDirCoT,vec3 vNormalCoT,vec2 texCoord,float parallaxScale) {float parallaxLimit=length(vViewDirCoT.xy)/vViewDirCoT.z;parallaxLimit*=parallaxScale;vec2 vOffsetDir=normalize(vViewDirCoT.xy);vec2 vMaxOffset=vOffsetDir*parallaxLimit;float numSamples=maxSamples+(dot(vViewDirCoT,vNormalCoT)*(minSamples-maxSamples));float stepSize=1.0/numSamples;float currRayHeight=1.0;vec2 vCurrOffset=vec2(0,0);vec2 vLastOffset=vec2(0,0);float lastSampledHeight=1.0;float currSampledHeight=1.0;bool keepWorking=true;for (int i=0; icurrRayHeight)\n{float delta1=currSampledHeight-currRayHeight;float delta2=(currRayHeight+stepSize)-lastSampledHeight;float ratio=delta1/(delta1+delta2);vCurrOffset=(ratio)* vLastOffset+(1.0-ratio)*vCurrOffset;keepWorking=false;}\nelse\n{currRayHeight-=stepSize;vLastOffset=vCurrOffset;vCurrOffset+=stepSize*vMaxOffset;lastSampledHeight=currSampledHeight;}}\nreturn vCurrOffset;}\nvec2 parallaxOffset(vec3 viewDir,float heightScale)\n{float height=texture2D(bumpSampler,vBumpUV).w;vec2 texCoordOffset=heightScale*viewDir.xy*height;return -texCoordOffset;}\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$j] = shader$j;\n\n // Do not edit.\n const name$k = \"clipPlaneFragmentDeclaration\";\n const shader$k = `#ifdef CLIPPLANE\nvarying float fClipDistance;\n#endif\n#ifdef CLIPPLANE2\nvarying float fClipDistance2;\n#endif\n#ifdef CLIPPLANE3\nvarying float fClipDistance3;\n#endif\n#ifdef CLIPPLANE4\nvarying float fClipDistance4;\n#endif\n#ifdef CLIPPLANE5\nvarying float fClipDistance5;\n#endif\n#ifdef CLIPPLANE6\nvarying float fClipDistance6;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$k] = shader$k;\n\n // Do not edit.\n const name$l = \"logDepthDeclaration\";\n const shader$l = `#ifdef LOGARITHMICDEPTH\nuniform float logarithmicDepthConstant;varying float vFragmentDepth;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$l] = shader$l;\n\n // Do not edit.\n const name$m = \"fogFragmentDeclaration\";\n const shader$m = `#ifdef FOG\n#define FOGMODE_NONE 0.\n#define FOGMODE_EXP 1.\n#define FOGMODE_EXP2 2.\n#define FOGMODE_LINEAR 3.\n#define E 2.71828\nuniform vec4 vFogInfos;uniform vec3 vFogColor;varying vec3 vFogDistance;float CalcFogFactor()\n{float fogCoeff=1.0;float fogStart=vFogInfos.y;float fogEnd=vFogInfos.z;float fogDensity=vFogInfos.w;float fogDistance=length(vFogDistance);if (FOGMODE_LINEAR==vFogInfos.x)\n{fogCoeff=(fogEnd-fogDistance)/(fogEnd-fogStart);}\nelse if (FOGMODE_EXP==vFogInfos.x)\n{fogCoeff=1.0/pow(E,fogDistance*fogDensity);}\nelse if (FOGMODE_EXP2==vFogInfos.x)\n{fogCoeff=1.0/pow(E,fogDistance*fogDistance*fogDensity*fogDensity);}\nreturn clamp(fogCoeff,0.0,1.0);}\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$m] = shader$m;\n\n // Do not edit.\n const name$n = \"clipPlaneFragment\";\n const shader$n = `#if defined(CLIPPLANE) || defined(CLIPPLANE2) || defined(CLIPPLANE3) || defined(CLIPPLANE4) || defined(CLIPPLANE5) || defined(CLIPPLANE6)\nif (false) {}\n#endif\n#ifdef CLIPPLANE\nelse if (fClipDistance>0.0)\n{discard;}\n#endif\n#ifdef CLIPPLANE2\nelse if (fClipDistance2>0.0)\n{discard;}\n#endif\n#ifdef CLIPPLANE3\nelse if (fClipDistance3>0.0)\n{discard;}\n#endif\n#ifdef CLIPPLANE4\nelse if (fClipDistance4>0.0)\n{discard;}\n#endif\n#ifdef CLIPPLANE5\nelse if (fClipDistance5>0.0)\n{discard;}\n#endif\n#ifdef CLIPPLANE6\nelse if (fClipDistance6>0.0)\n{discard;}\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$n] = shader$n;\n\n // Do not edit.\n const name$o = \"bumpFragment\";\n const shader$o = `vec2 uvOffset=vec2(0.0,0.0);\n#if defined(BUMP) || defined(PARALLAX) || defined(DETAIL)\n#ifdef NORMALXYSCALE\nfloat normalScale=1.0;\n#elif defined(BUMP)\nfloat normalScale=vBumpInfos.y;\n#else\nfloat normalScale=1.0;\n#endif\n#if defined(TANGENT) && defined(NORMAL)\nmat3 TBN=vTBN;\n#elif defined(BUMP)\nvec2 TBNUV=gl_FrontFacing ? vBumpUV : -vBumpUV;mat3 TBN=cotangent_frame(normalW*normalScale,vPositionW,TBNUV,vTangentSpaceParams);\n#else\nvec2 TBNUV=gl_FrontFacing ? vDetailUV : -vDetailUV;mat3 TBN=cotangent_frame(normalW*normalScale,vPositionW,TBNUV,vec2(1.,1.));\n#endif\n#elif defined(ANISOTROPIC)\n#if defined(TANGENT) && defined(NORMAL)\nmat3 TBN=vTBN;\n#else\nvec2 TBNUV=gl_FrontFacing ? vMainUV1 : -vMainUV1;mat3 TBN=cotangent_frame(normalW,vPositionW,TBNUV,vec2(1.,1.));\n#endif\n#endif\n#ifdef PARALLAX\nmat3 invTBN=transposeMat3(TBN);\n#ifdef PARALLAXOCCLUSION\nuvOffset=parallaxOcclusion(invTBN*-viewDirectionW,invTBN*normalW,vBumpUV,vBumpInfos.z);\n#else\nuvOffset=parallaxOffset(invTBN*viewDirectionW,vBumpInfos.z);\n#endif\n#endif\n#ifdef DETAIL\nvec4 detailColor=texture2D(detailSampler,vDetailUV+uvOffset);vec2 detailNormalRG=detailColor.wy*2.0-1.0;float detailNormalB=sqrt(1.-saturate(dot(detailNormalRG,detailNormalRG)));vec3 detailNormal=vec3(detailNormalRG,detailNormalB);\n#endif\n#ifdef BUMP\n#ifdef OBJECTSPACE_NORMALMAP\n#define CUSTOM_FRAGMENT_BUMP_FRAGMENT\nnormalW=normalize(texture2D(bumpSampler,vBumpUV).xyz *2.0-1.0);normalW=normalize(mat3(normalMatrix)*normalW);\n#elif !defined(DETAIL)\nnormalW=perturbNormal(TBN,texture2D(bumpSampler,vBumpUV+uvOffset).xyz,vBumpInfos.y);\n#else\nvec3 bumpNormal=texture2D(bumpSampler,vBumpUV+uvOffset).xyz*2.0-1.0;\n#if DETAIL_NORMALBLENDMETHOD==0 \ndetailNormal.xy*=vDetailInfos.z;vec3 blendedNormal=normalize(vec3(bumpNormal.xy+detailNormal.xy,bumpNormal.z*detailNormal.z));\n#elif DETAIL_NORMALBLENDMETHOD==1 \ndetailNormal.xy*=vDetailInfos.z;bumpNormal+=vec3(0.0,0.0,1.0);detailNormal*=vec3(-1.0,-1.0,1.0);vec3 blendedNormal=bumpNormal*dot(bumpNormal,detailNormal)/bumpNormal.z-detailNormal;\n#endif\nnormalW=perturbNormalBase(TBN,blendedNormal,vBumpInfos.y);\n#endif\n#elif defined(DETAIL)\ndetailNormal.xy*=vDetailInfos.z;normalW=perturbNormalBase(TBN,detailNormal,vDetailInfos.z);\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$o] = shader$o;\n\n // Do not edit.\n const name$p = \"decalFragment\";\n const shader$p = `#ifdef DECAL\n#ifdef GAMMADECAL\ndecalColor.rgb=toLinearSpace(decalColor.rgb);\n#endif\n#ifdef DECAL_SMOOTHALPHA\ndecalColor.a*=decalColor.a;\n#endif\nsurfaceAlbedo.rgb=mix(surfaceAlbedo.rgb,decalColor.rgb,decalColor.a);\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$p] = shader$p;\n\n // Do not edit.\n const name$q = \"depthPrePass\";\n const shader$q = `#ifdef DEPTHPREPASS\ngl_FragColor=vec4(0.,0.,0.,1.0);return;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$q] = shader$q;\n\n // Do not edit.\n const name$r = \"lightFragment\";\n const shader$r = `#ifdef LIGHT{X}\n#if defined(SHADOWONLY) || defined(LIGHTMAP) && defined(LIGHTMAPEXCLUDED{X}) && defined(LIGHTMAPNOSPECULAR{X})\n#else\n#ifdef PBR\n#ifdef SPOTLIGHT{X}\npreInfo=computePointAndSpotPreLightingInfo(light{X}.vLightData,viewDirectionW,normalW);\n#elif defined(POINTLIGHT{X})\npreInfo=computePointAndSpotPreLightingInfo(light{X}.vLightData,viewDirectionW,normalW);\n#elif defined(HEMILIGHT{X})\npreInfo=computeHemisphericPreLightingInfo(light{X}.vLightData,viewDirectionW,normalW);\n#elif defined(DIRLIGHT{X})\npreInfo=computeDirectionalPreLightingInfo(light{X}.vLightData,viewDirectionW,normalW);\n#endif\npreInfo.NdotV=NdotV;\n#ifdef SPOTLIGHT{X}\n#ifdef LIGHT_FALLOFF_GLTF{X}\npreInfo.attenuation=computeDistanceLightFalloff_GLTF(preInfo.lightDistanceSquared,light{X}.vLightFalloff.y);preInfo.attenuation*=computeDirectionalLightFalloff_GLTF(light{X}.vLightDirection.xyz,preInfo.L,light{X}.vLightFalloff.z,light{X}.vLightFalloff.w);\n#elif defined(LIGHT_FALLOFF_PHYSICAL{X})\npreInfo.attenuation=computeDistanceLightFalloff_Physical(preInfo.lightDistanceSquared);preInfo.attenuation*=computeDirectionalLightFalloff_Physical(light{X}.vLightDirection.xyz,preInfo.L,light{X}.vLightDirection.w);\n#elif defined(LIGHT_FALLOFF_STANDARD{X})\npreInfo.attenuation=computeDistanceLightFalloff_Standard(preInfo.lightOffset,light{X}.vLightFalloff.x);preInfo.attenuation*=computeDirectionalLightFalloff_Standard(light{X}.vLightDirection.xyz,preInfo.L,light{X}.vLightDirection.w,light{X}.vLightData.w);\n#else\npreInfo.attenuation=computeDistanceLightFalloff(preInfo.lightOffset,preInfo.lightDistanceSquared,light{X}.vLightFalloff.x,light{X}.vLightFalloff.y);preInfo.attenuation*=computeDirectionalLightFalloff(light{X}.vLightDirection.xyz,preInfo.L,light{X}.vLightDirection.w,light{X}.vLightData.w,light{X}.vLightFalloff.z,light{X}.vLightFalloff.w);\n#endif\n#elif defined(POINTLIGHT{X})\n#ifdef LIGHT_FALLOFF_GLTF{X}\npreInfo.attenuation=computeDistanceLightFalloff_GLTF(preInfo.lightDistanceSquared,light{X}.vLightFalloff.y);\n#elif defined(LIGHT_FALLOFF_PHYSICAL{X})\npreInfo.attenuation=computeDistanceLightFalloff_Physical(preInfo.lightDistanceSquared);\n#elif defined(LIGHT_FALLOFF_STANDARD{X})\npreInfo.attenuation=computeDistanceLightFalloff_Standard(preInfo.lightOffset,light{X}.vLightFalloff.x);\n#else\npreInfo.attenuation=computeDistanceLightFalloff(preInfo.lightOffset,preInfo.lightDistanceSquared,light{X}.vLightFalloff.x,light{X}.vLightFalloff.y);\n#endif\n#else\npreInfo.attenuation=1.0;\n#endif\n#ifdef HEMILIGHT{X}\npreInfo.roughness=roughness;\n#else\npreInfo.roughness=adjustRoughnessFromLightProperties(roughness,light{X}.vLightSpecular.a,preInfo.lightDistance);\n#endif\n#ifdef IRIDESCENCE\npreInfo.iridescenceIntensity=iridescenceIntensity;\n#endif\n#ifdef HEMILIGHT{X}\ninfo.diffuse=computeHemisphericDiffuseLighting(preInfo,light{X}.vLightDiffuse.rgb,light{X}.vLightGround);\n#elif defined(SS_TRANSLUCENCY)\ninfo.diffuse=computeDiffuseAndTransmittedLighting(preInfo,light{X}.vLightDiffuse.rgb,subSurfaceOut.transmittance);\n#else\ninfo.diffuse=computeDiffuseLighting(preInfo,light{X}.vLightDiffuse.rgb);\n#endif\n#ifdef SPECULARTERM\n#ifdef ANISOTROPIC\ninfo.specular=computeAnisotropicSpecularLighting(preInfo,viewDirectionW,normalW,anisotropicOut.anisotropicTangent,anisotropicOut.anisotropicBitangent,anisotropicOut.anisotropy,clearcoatOut.specularEnvironmentR0,specularEnvironmentR90,AARoughnessFactors.x,light{X}.vLightDiffuse.rgb);\n#else\ninfo.specular=computeSpecularLighting(preInfo,normalW,clearcoatOut.specularEnvironmentR0,specularEnvironmentR90,AARoughnessFactors.x,light{X}.vLightDiffuse.rgb);\n#endif\n#endif\n#ifdef SHEEN\n#ifdef SHEEN_LINKWITHALBEDO\npreInfo.roughness=sheenOut.sheenIntensity;\n#else\n#ifdef HEMILIGHT{X}\npreInfo.roughness=sheenOut.sheenRoughness;\n#else\npreInfo.roughness=adjustRoughnessFromLightProperties(sheenOut.sheenRoughness,light{X}.vLightSpecular.a,preInfo.lightDistance);\n#endif\n#endif\ninfo.sheen=computeSheenLighting(preInfo,normalW,sheenOut.sheenColor,specularEnvironmentR90,AARoughnessFactors.x,light{X}.vLightDiffuse.rgb);\n#endif\n#ifdef CLEARCOAT\n#ifdef HEMILIGHT{X}\npreInfo.roughness=clearcoatOut.clearCoatRoughness;\n#else\npreInfo.roughness=adjustRoughnessFromLightProperties(clearcoatOut.clearCoatRoughness,light{X}.vLightSpecular.a,preInfo.lightDistance);\n#endif\ninfo.clearCoat=computeClearCoatLighting(preInfo,clearcoatOut.clearCoatNormalW,clearcoatOut.clearCoatAARoughnessFactors.x,clearcoatOut.clearCoatIntensity,light{X}.vLightDiffuse.rgb);\n#ifdef CLEARCOAT_TINT\nabsorption=computeClearCoatLightingAbsorption(clearcoatOut.clearCoatNdotVRefract,preInfo.L,clearcoatOut.clearCoatNormalW,clearcoatOut.clearCoatColor,clearcoatOut.clearCoatThickness,clearcoatOut.clearCoatIntensity);info.diffuse*=absorption;\n#ifdef SPECULARTERM\ninfo.specular*=absorption;\n#endif\n#endif\ninfo.diffuse*=info.clearCoat.w;\n#ifdef SPECULARTERM\ninfo.specular*=info.clearCoat.w;\n#endif\n#ifdef SHEEN\ninfo.sheen*=info.clearCoat.w;\n#endif\n#endif\n#else\n#ifdef SPOTLIGHT{X}\ninfo=computeSpotLighting(viewDirectionW,normalW,light{X}.vLightData,light{X}.vLightDirection,light{X}.vLightDiffuse.rgb,light{X}.vLightSpecular.rgb,light{X}.vLightDiffuse.a,glossiness);\n#elif defined(HEMILIGHT{X})\ninfo=computeHemisphericLighting(viewDirectionW,normalW,light{X}.vLightData,light{X}.vLightDiffuse.rgb,light{X}.vLightSpecular.rgb,light{X}.vLightGround,glossiness);\n#elif defined(POINTLIGHT{X}) || defined(DIRLIGHT{X})\ninfo=computeLighting(viewDirectionW,normalW,light{X}.vLightData,light{X}.vLightDiffuse.rgb,light{X}.vLightSpecular.rgb,light{X}.vLightDiffuse.a,glossiness);\n#endif\n#endif\n#ifdef PROJECTEDLIGHTTEXTURE{X}\ninfo.diffuse*=computeProjectionTextureDiffuseLighting(projectionLightSampler{X},textureProjectionMatrix{X});\n#endif\n#endif\n#ifdef SHADOW{X}\n#ifdef SHADOWCSM{X}\nfor (int i=0; i=0.) {index{X}=i;break;}}\n#ifdef SHADOWCSMUSESHADOWMAXZ{X}\nif (index{X}>=0)\n#endif\n{\n#if defined(SHADOWPCF{X})\n#if defined(SHADOWLOWQUALITY{X})\nshadow=computeShadowWithCSMPCF1(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#elif defined(SHADOWMEDIUMQUALITY{X})\nshadow=computeShadowWithCSMPCF3(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.yz,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#else\nshadow=computeShadowWithCSMPCF5(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.yz,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#endif\n#elif defined(SHADOWPCSS{X})\n#if defined(SHADOWLOWQUALITY{X})\nshadow=computeShadowWithCSMPCSS16(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w,lightSizeUVCorrection{X}[index{X}],depthCorrection{X}[index{X}],penumbraDarkness{X});\n#elif defined(SHADOWMEDIUMQUALITY{X})\nshadow=computeShadowWithCSMPCSS32(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w,lightSizeUVCorrection{X}[index{X}],depthCorrection{X}[index{X}],penumbraDarkness{X});\n#else\nshadow=computeShadowWithCSMPCSS64(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w,lightSizeUVCorrection{X}[index{X}],depthCorrection{X}[index{X}],penumbraDarkness{X});\n#endif\n#else\nshadow=computeShadowCSM(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#endif\n#ifdef SHADOWCSMDEBUG{X}\nshadowDebug{X}=vec3(shadow)*vCascadeColorsMultiplier{X}[index{X}];\n#endif\n#ifndef SHADOWCSMNOBLEND{X}\nfloat frustumLength=frustumLengths{X}[index{X}];float diffRatio=clamp(diff{X}/frustumLength,0.,1.)*cascadeBlendFactor{X};if (index{X}<(SHADOWCSMNUM_CASCADES{X}-1) && diffRatio<1.)\n{index{X}+=1;float nextShadow=0.;\n#if defined(SHADOWPCF{X})\n#if defined(SHADOWLOWQUALITY{X})\nnextShadow=computeShadowWithCSMPCF1(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#elif defined(SHADOWMEDIUMQUALITY{X})\nnextShadow=computeShadowWithCSMPCF3(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.yz,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#else\nnextShadow=computeShadowWithCSMPCF5(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.yz,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#endif\n#elif defined(SHADOWPCSS{X})\n#if defined(SHADOWLOWQUALITY{X})\nnextShadow=computeShadowWithCSMPCSS16(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w,lightSizeUVCorrection{X}[index{X}],depthCorrection{X}[index{X}],penumbraDarkness{X});\n#elif defined(SHADOWMEDIUMQUALITY{X})\nnextShadow=computeShadowWithCSMPCSS32(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w,lightSizeUVCorrection{X}[index{X}],depthCorrection{X}[index{X}],penumbraDarkness{X});\n#else\nnextShadow=computeShadowWithCSMPCSS64(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w,lightSizeUVCorrection{X}[index{X}],depthCorrection{X}[index{X}],penumbraDarkness{X});\n#endif\n#else\nnextShadow=computeShadowCSM(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#endif\nshadow=mix(nextShadow,shadow,diffRatio);\n#ifdef SHADOWCSMDEBUG{X}\nshadowDebug{X}=mix(vec3(nextShadow)*vCascadeColorsMultiplier{X}[index{X}],shadowDebug{X},diffRatio);\n#endif\n}\n#endif\n}\n#elif defined(SHADOWCLOSEESM{X})\n#if defined(SHADOWCUBE{X})\nshadow=computeShadowWithCloseESMCube(light{X}.vLightData.xyz,shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.z,light{X}.depthValues);\n#else\nshadow=computeShadowWithCloseESM(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.z,light{X}.shadowsInfo.w);\n#endif\n#elif defined(SHADOWESM{X})\n#if defined(SHADOWCUBE{X})\nshadow=computeShadowWithESMCube(light{X}.vLightData.xyz,shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.z,light{X}.depthValues);\n#else\nshadow=computeShadowWithESM(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.z,light{X}.shadowsInfo.w);\n#endif\n#elif defined(SHADOWPOISSON{X})\n#if defined(SHADOWCUBE{X})\nshadow=computeShadowWithPoissonSamplingCube(light{X}.vLightData.xyz,shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.x,light{X}.depthValues);\n#else\nshadow=computeShadowWithPoissonSampling(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#endif\n#elif defined(SHADOWPCF{X})\n#if defined(SHADOWLOWQUALITY{X})\nshadow=computeShadowWithPCF1(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#elif defined(SHADOWMEDIUMQUALITY{X})\nshadow=computeShadowWithPCF3(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.yz,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#else\nshadow=computeShadowWithPCF5(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.yz,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#endif\n#elif defined(SHADOWPCSS{X})\n#if defined(SHADOWLOWQUALITY{X})\nshadow=computeShadowWithPCSS16(vPositionFromLight{X},vDepthMetric{X},depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#elif defined(SHADOWMEDIUMQUALITY{X})\nshadow=computeShadowWithPCSS32(vPositionFromLight{X},vDepthMetric{X},depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#else\nshadow=computeShadowWithPCSS64(vPositionFromLight{X},vDepthMetric{X},depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#endif\n#else\n#if defined(SHADOWCUBE{X})\nshadow=computeShadowCube(light{X}.vLightData.xyz,shadowSampler{X},light{X}.shadowsInfo.x,light{X}.depthValues);\n#else\nshadow=computeShadow(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#endif\n#endif\n#ifdef SHADOWONLY\n#ifndef SHADOWINUSE\n#define SHADOWINUSE\n#endif\nglobalShadow+=shadow;shadowLightCount+=1.0;\n#endif\n#else\nshadow=1.;\n#endif\n#ifndef SHADOWONLY\n#ifdef CUSTOMUSERLIGHTING\ndiffuseBase+=computeCustomDiffuseLighting(info,diffuseBase,shadow);\n#ifdef SPECULARTERM\nspecularBase+=computeCustomSpecularLighting(info,specularBase,shadow);\n#endif\n#elif defined(LIGHTMAP) && defined(LIGHTMAPEXCLUDED{X})\ndiffuseBase+=lightmapColor.rgb*shadow;\n#ifdef SPECULARTERM\n#ifndef LIGHTMAPNOSPECULAR{X}\nspecularBase+=info.specular*shadow*lightmapColor.rgb;\n#endif\n#endif\n#ifdef CLEARCOAT\n#ifndef LIGHTMAPNOSPECULAR{X}\nclearCoatBase+=info.clearCoat.rgb*shadow*lightmapColor.rgb;\n#endif\n#endif\n#ifdef SHEEN\n#ifndef LIGHTMAPNOSPECULAR{X}\nsheenBase+=info.sheen.rgb*shadow;\n#endif\n#endif\n#else\n#ifdef SHADOWCSMDEBUG{X}\ndiffuseBase+=info.diffuse*shadowDebug{X};\n#else \ndiffuseBase+=info.diffuse*shadow;\n#endif\n#ifdef SPECULARTERM\nspecularBase+=info.specular*shadow;\n#endif\n#ifdef CLEARCOAT\nclearCoatBase+=info.clearCoat.rgb*shadow;\n#endif\n#ifdef SHEEN\nsheenBase+=info.sheen.rgb*shadow;\n#endif\n#endif\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$r] = shader$r;\n\n // Do not edit.\n const name$s = \"logDepthFragment\";\n const shader$s = `#ifdef LOGARITHMICDEPTH\ngl_FragDepthEXT=log2(vFragmentDepth)*logarithmicDepthConstant*0.5;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$s] = shader$s;\n\n // Do not edit.\n const name$t = \"fogFragment\";\n const shader$t = `#ifdef FOG\nfloat fog=CalcFogFactor();\n#ifdef PBR\nfog=toLinearSpace(fog);\n#endif\ncolor.rgb=mix(vFogColor,color.rgb,fog);\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$t] = shader$t;\n\n // Do not edit.\n const name$u = \"oitFragment\";\n const shader$u = `#ifdef ORDER_INDEPENDENT_TRANSPARENCY\nfloat fragDepth=gl_FragCoord.z; \n#ifdef ORDER_INDEPENDENT_TRANSPARENCY_16BITS\nuint halfFloat=packHalf2x16(vec2(fragDepth));vec2 full=unpackHalf2x16(halfFloat);fragDepth=full.x;\n#endif\nivec2 fragCoord=ivec2(gl_FragCoord.xy);vec2 lastDepth=texelFetch(oitDepthSampler,fragCoord,0).rg;vec4 lastFrontColor=texelFetch(oitFrontColorSampler,fragCoord,0);depth.rg=vec2(-MAX_DEPTH);frontColor=lastFrontColor;backColor=vec4(0.0);\n#ifdef USE_REVERSE_DEPTHBUFFER\nfloat furthestDepth=-lastDepth.x;float nearestDepth=lastDepth.y;\n#else\nfloat nearestDepth=-lastDepth.x;float furthestDepth=lastDepth.y;\n#endif\nfloat alphaMultiplier=1.0-lastFrontColor.a;\n#ifdef USE_REVERSE_DEPTHBUFFER\nif (fragDepth>nearestDepth || fragDepthfurthestDepth) {\n#endif\nreturn;}\n#ifdef USE_REVERSE_DEPTHBUFFER\nif (fragDepthfurthestDepth) {\n#else\nif (fragDepth>nearestDepth && fragDepth\n#if defined(BUMP) || !defined(NORMAL)\n#extension GL_OES_standard_derivatives : enable\n#endif\n#include[SCENE_MRT_COUNT]\n#include\n#define CUSTOM_FRAGMENT_BEGIN\n#ifdef LOGARITHMICDEPTH\n#extension GL_EXT_frag_depth : enable\n#endif\n#define RECIPROCAL_PI2 0.15915494\nvarying vec3 vPositionW;\n#ifdef NORMAL\nvarying vec3 vNormalW;\n#endif\n#if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES)\nvarying vec4 vColor;\n#endif\n#include[1..7]\n#include\n#include<__decl__lightFragment>[0..maxSimultaneousLights]\n#include\n#include\n#include(_DEFINENAME_,DIFFUSE,_VARYINGNAME_,Diffuse,_SAMPLERNAME_,diffuse)\n#include(_DEFINENAME_,AMBIENT,_VARYINGNAME_,Ambient,_SAMPLERNAME_,ambient)\n#include(_DEFINENAME_,OPACITY,_VARYINGNAME_,Opacity,_SAMPLERNAME_,opacity)\n#include(_DEFINENAME_,EMISSIVE,_VARYINGNAME_,Emissive,_SAMPLERNAME_,emissive)\n#include(_DEFINENAME_,LIGHTMAP,_VARYINGNAME_,Lightmap,_SAMPLERNAME_,lightmap)\n#include(_DEFINENAME_,DECAL,_VARYINGNAME_,Decal,_SAMPLERNAME_,decal)\n#ifdef REFRACTION\n#ifdef REFRACTIONMAP_3D\nuniform samplerCube refractionCubeSampler;\n#else\nuniform sampler2D refraction2DSampler;\n#endif\n#endif\n#if defined(SPECULARTERM)\n#include(_DEFINENAME_,SPECULAR,_VARYINGNAME_,Specular,_SAMPLERNAME_,specular)\n#endif\n#include\n#ifdef REFLECTION\n#ifdef REFLECTIONMAP_3D\nuniform samplerCube reflectionCubeSampler;\n#else\nuniform sampler2D reflection2DSampler;\n#endif\n#ifdef REFLECTIONMAP_SKYBOX\nvarying vec3 vPositionUVW;\n#else\n#if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED)\nvarying vec3 vDirectionW;\n#endif\n#endif\n#include\n#endif\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_FRAGMENT_MAIN_BEGIN\n#include\nvec3 viewDirectionW=normalize(vEyePosition.xyz-vPositionW);vec4 baseColor=vec4(1.,1.,1.,1.);vec3 diffuseColor=vDiffuseColor.rgb;float alpha=vDiffuseColor.a;\n#ifdef NORMAL\nvec3 normalW=normalize(vNormalW);\n#else\nvec3 normalW=normalize(-cross(dFdx(vPositionW),dFdy(vPositionW)));\n#endif\n#include\n#ifdef TWOSIDEDLIGHTING\nnormalW=gl_FrontFacing ? normalW : -normalW;\n#endif\n#ifdef DIFFUSE\nbaseColor=texture2D(diffuseSampler,vDiffuseUV+uvOffset);\n#if defined(ALPHATEST) && !defined(ALPHATEST_AFTERALLALPHACOMPUTATIONS)\nif (baseColor.a(surfaceAlbedo,baseColor,GAMMADECAL,_GAMMADECAL_NOTUSED_)\n#endif\n#include\n#if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES)\nbaseColor.rgb*=vColor.rgb;\n#endif\n#ifdef DETAIL\nbaseColor.rgb=baseColor.rgb*2.0*mix(0.5,detailColor.r,vDetailInfos.y);\n#endif\n#if defined(DECAL) && defined(DECAL_AFTER_DETAIL)\nvec4 decalColor=texture2D(decalSampler,vDecalUV+uvOffset);\n#include(surfaceAlbedo,baseColor,GAMMADECAL,_GAMMADECAL_NOTUSED_)\n#endif\n#define CUSTOM_FRAGMENT_UPDATE_DIFFUSE\nvec3 baseAmbientColor=vec3(1.,1.,1.);\n#ifdef AMBIENT\nbaseAmbientColor=texture2D(ambientSampler,vAmbientUV+uvOffset).rgb*vAmbientInfos.y;\n#endif\n#define CUSTOM_FRAGMENT_BEFORE_LIGHTS\n#ifdef SPECULARTERM\nfloat glossiness=vSpecularColor.a;vec3 specularColor=vSpecularColor.rgb;\n#ifdef SPECULAR\nvec4 specularMapColor=texture2D(specularSampler,vSpecularUV+uvOffset);specularColor=specularMapColor.rgb;\n#ifdef GLOSSINESS\nglossiness=glossiness*specularMapColor.a;\n#endif\n#endif\n#else\nfloat glossiness=0.;\n#endif\nvec3 diffuseBase=vec3(0.,0.,0.);lightingInfo info;\n#ifdef SPECULARTERM\nvec3 specularBase=vec3(0.,0.,0.);\n#endif\nfloat shadow=1.;\n#ifdef LIGHTMAP\nvec4 lightmapColor=texture2D(lightmapSampler,vLightmapUV+uvOffset);\n#ifdef RGBDLIGHTMAP\nlightmapColor.rgb=fromRGBD(lightmapColor);\n#endif\nlightmapColor.rgb*=vLightmapInfos.y;\n#endif\n#include[0..maxSimultaneousLights]\nvec4 refractionColor=vec4(0.,0.,0.,1.);\n#ifdef REFRACTION\nvec3 refractionVector=normalize(refract(-viewDirectionW,normalW,vRefractionInfos.y));\n#ifdef REFRACTIONMAP_3D\n#ifdef USE_LOCAL_REFRACTIONMAP_CUBIC\nrefractionVector=parallaxCorrectNormal(vPositionW,refractionVector,vRefractionSize,vRefractionPosition);\n#endif\nrefractionVector.y=refractionVector.y*vRefractionInfos.w;vec4 refractionLookup=textureCube(refractionCubeSampler,refractionVector);if (dot(refractionVector,viewDirectionW)<1.0) {refractionColor=refractionLookup;}\n#else\nvec3 vRefractionUVW=vec3(refractionMatrix*(view*vec4(vPositionW+refractionVector*vRefractionInfos.z,1.0)));vec2 refractionCoords=vRefractionUVW.xy/vRefractionUVW.z;refractionCoords.y=1.0-refractionCoords.y;refractionColor=texture2D(refraction2DSampler,refractionCoords);\n#endif\n#ifdef RGBDREFRACTION\nrefractionColor.rgb=fromRGBD(refractionColor);\n#endif\n#ifdef IS_REFRACTION_LINEAR\nrefractionColor.rgb=toGammaSpace(refractionColor.rgb);\n#endif\nrefractionColor.rgb*=vRefractionInfos.x;\n#endif\nvec4 reflectionColor=vec4(0.,0.,0.,1.);\n#ifdef REFLECTION\nvec3 vReflectionUVW=computeReflectionCoords(vec4(vPositionW,1.0),normalW);\n#ifdef REFLECTIONMAP_OPPOSITEZ\nvReflectionUVW.z*=-1.0;\n#endif\n#ifdef REFLECTIONMAP_3D\n#ifdef ROUGHNESS\nfloat bias=vReflectionInfos.y;\n#ifdef SPECULARTERM\n#ifdef SPECULAR\n#ifdef GLOSSINESS\nbias*=(1.0-specularMapColor.a);\n#endif\n#endif\n#endif\nreflectionColor=textureCube(reflectionCubeSampler,vReflectionUVW,bias);\n#else\nreflectionColor=textureCube(reflectionCubeSampler,vReflectionUVW);\n#endif\n#else\nvec2 coords=vReflectionUVW.xy;\n#ifdef REFLECTIONMAP_PROJECTION\ncoords/=vReflectionUVW.z;\n#endif\ncoords.y=1.0-coords.y;reflectionColor=texture2D(reflection2DSampler,coords);\n#endif\n#ifdef RGBDREFLECTION\nreflectionColor.rgb=fromRGBD(reflectionColor);\n#endif\n#ifdef IS_REFLECTION_LINEAR\nreflectionColor.rgb=toGammaSpace(reflectionColor.rgb);\n#endif\nreflectionColor.rgb*=vReflectionInfos.x;\n#ifdef REFLECTIONFRESNEL\nfloat reflectionFresnelTerm=computeFresnelTerm(viewDirectionW,normalW,reflectionRightColor.a,reflectionLeftColor.a);\n#ifdef REFLECTIONFRESNELFROMSPECULAR\n#ifdef SPECULARTERM\nreflectionColor.rgb*=specularColor.rgb*(1.0-reflectionFresnelTerm)+reflectionFresnelTerm*reflectionRightColor.rgb;\n#else\nreflectionColor.rgb*=reflectionLeftColor.rgb*(1.0-reflectionFresnelTerm)+reflectionFresnelTerm*reflectionRightColor.rgb;\n#endif\n#else\nreflectionColor.rgb*=reflectionLeftColor.rgb*(1.0-reflectionFresnelTerm)+reflectionFresnelTerm*reflectionRightColor.rgb;\n#endif\n#endif\n#endif\n#ifdef REFRACTIONFRESNEL\nfloat refractionFresnelTerm=computeFresnelTerm(viewDirectionW,normalW,refractionRightColor.a,refractionLeftColor.a);refractionColor.rgb*=refractionLeftColor.rgb*(1.0-refractionFresnelTerm)+refractionFresnelTerm*refractionRightColor.rgb;\n#endif\n#ifdef OPACITY\nvec4 opacityMap=texture2D(opacitySampler,vOpacityUV+uvOffset);\n#ifdef OPACITYRGB\nopacityMap.rgb=opacityMap.rgb*vec3(0.3,0.59,0.11);alpha*=(opacityMap.x+opacityMap.y+opacityMap.z)* vOpacityInfos.y;\n#else\nalpha*=opacityMap.a*vOpacityInfos.y;\n#endif\n#endif\n#if defined(VERTEXALPHA) || defined(INSTANCESCOLOR) && defined(INSTANCES)\nalpha*=vColor.a;\n#endif\n#ifdef OPACITYFRESNEL\nfloat opacityFresnelTerm=computeFresnelTerm(viewDirectionW,normalW,opacityParts.z,opacityParts.w);alpha+=opacityParts.x*(1.0-opacityFresnelTerm)+opacityFresnelTerm*opacityParts.y;\n#endif\n#ifdef ALPHATEST\n#ifdef ALPHATEST_AFTERALLALPHACOMPUTATIONS\nif (alpha\n#include\n#ifdef IMAGEPROCESSINGPOSTPROCESS\ncolor.rgb=toLinearSpace(color.rgb);\n#else\n#ifdef IMAGEPROCESSING\ncolor.rgb=toLinearSpace(color.rgb);color=applyImageProcessing(color);\n#endif\n#endif\ncolor.a*=visibility;\n#ifdef PREMULTIPLYALPHA\ncolor.rgb*=color.a;\n#endif\n#define CUSTOM_FRAGMENT_BEFORE_FRAGCOLOR\n#ifdef PREPASS\nfloat writeGeometryInfo=color.a>0.4 ? 1.0 : 0.0;gl_FragData[0]=color; \n#ifdef PREPASS_POSITION\ngl_FragData[PREPASS_POSITION_INDEX]=vec4(vPositionW,writeGeometryInfo);\n#endif\n#ifdef PREPASS_VELOCITY\nvec2 a=(vCurrentPosition.xy/vCurrentPosition.w)*0.5+0.5;vec2 b=(vPreviousPosition.xy/vPreviousPosition.w)*0.5+0.5;vec2 velocity=abs(a-b);velocity=vec2(pow(velocity.x,1.0/3.0),pow(velocity.y,1.0/3.0))*sign(a-b)*0.5+0.5;gl_FragData[PREPASS_VELOCITY_INDEX]=vec4(velocity,0.0,writeGeometryInfo);\n#endif\n#ifdef PREPASS_IRRADIANCE\ngl_FragData[PREPASS_IRRADIANCE_INDEX]=vec4(0.0,0.0,0.0,writeGeometryInfo); \n#endif\n#ifdef PREPASS_DEPTH\ngl_FragData[PREPASS_DEPTH_INDEX]=vec4(vViewPos.z,0.0,0.0,writeGeometryInfo); \n#endif\n#ifdef PREPASS_NORMAL\ngl_FragData[PREPASS_NORMAL_INDEX]=vec4(normalize((view*vec4(normalW,0.0)).rgb),writeGeometryInfo); \n#endif\n#ifdef PREPASS_ALBEDO_SQRT\ngl_FragData[PREPASS_ALBEDO_SQRT_INDEX]=vec4(0.0,0.0,0.0,writeGeometryInfo); \n#endif\n#ifdef PREPASS_REFLECTIVITY\n#if defined(SPECULARTERM)\n#if defined(SPECULAR)\ngl_FragData[PREPASS_REFLECTIVITY_INDEX]=vec4(toLinearSpace(specularMapColor))*writeGeometryInfo; \n#else\ngl_FragData[PREPASS_REFLECTIVITY_INDEX]=vec4(toLinearSpace(specularColor),1.0)*writeGeometryInfo;\n#endif\n#else\ngl_FragData[PREPASS_REFLECTIVITY_INDEX]=vec4(0.0,0.0,0.0,1.0)*writeGeometryInfo;\n#endif\n#endif\n#endif\n#if !defined(PREPASS) || defined(WEBGL2)\ngl_FragColor=color;\n#endif\n#include\n#if ORDER_INDEPENDENT_TRANSPARENCY\nif (fragDepth==nearestDepth) {frontColor.rgb+=color.rgb*color.a*alphaMultiplier;frontColor.a=1.0-alphaMultiplier*(1.0-color.a);} else {backColor+=color;}\n#endif\n#define CUSTOM_FRAGMENT_MAIN_END\n}\n`;\n // Sideeffect\n ShaderStore.ShadersStore[name$v] = shader$v;\n\n // Do not edit.\n const name$w = \"decalVertexDeclaration\";\n const shader$w = `#ifdef DECAL\nuniform vec4 vDecalInfos;uniform mat4 decalMatrix;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$w] = shader$w;\n\n // Do not edit.\n const name$x = \"defaultVertexDeclaration\";\n const shader$x = `uniform mat4 viewProjection;uniform mat4 view;\n#ifdef DIFFUSE\nuniform mat4 diffuseMatrix;uniform vec2 vDiffuseInfos;\n#endif\n#ifdef AMBIENT\nuniform mat4 ambientMatrix;uniform vec2 vAmbientInfos;\n#endif\n#ifdef OPACITY\nuniform mat4 opacityMatrix;uniform vec2 vOpacityInfos;\n#endif\n#ifdef EMISSIVE\nuniform vec2 vEmissiveInfos;uniform mat4 emissiveMatrix;\n#endif\n#ifdef LIGHTMAP\nuniform vec2 vLightmapInfos;uniform mat4 lightmapMatrix;\n#endif\n#if defined(SPECULAR) && defined(SPECULARTERM)\nuniform vec2 vSpecularInfos;uniform mat4 specularMatrix;\n#endif\n#ifdef BUMP\nuniform vec3 vBumpInfos;uniform mat4 bumpMatrix;\n#endif\n#ifdef REFLECTION\nuniform mat4 reflectionMatrix;\n#endif\n#ifdef POINTSIZE\nuniform float pointSize;\n#endif\n#ifdef DETAIL\nuniform vec4 vDetailInfos;uniform mat4 detailMatrix;\n#endif\n#include\n#define ADDITIONAL_VERTEX_DECLARATION\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$x] = shader$x;\n\n // Do not edit.\n const name$y = \"uvAttributeDeclaration\";\n const shader$y = `#ifdef UV{X}\nattribute vec2 uv{X};\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$y] = shader$y;\n\n // Do not edit.\n const name$z = \"bonesDeclaration\";\n const shader$z = `#if NUM_BONE_INFLUENCERS>0\nattribute vec4 matricesIndices;attribute vec4 matricesWeights;\n#if NUM_BONE_INFLUENCERS>4\nattribute vec4 matricesIndicesExtra;attribute vec4 matricesWeightsExtra;\n#endif\n#ifndef BAKED_VERTEX_ANIMATION_TEXTURE\n#ifdef BONETEXTURE\nuniform sampler2D boneSampler;uniform float boneTextureWidth;\n#else\nuniform mat4 mBones[BonesPerMesh];\n#ifdef BONES_VELOCITY_ENABLED\nuniform mat4 mPreviousBones[BonesPerMesh];\n#endif\n#endif\n#ifdef BONETEXTURE\n#define inline\nmat4 readMatrixFromRawSampler(sampler2D smp,float index)\n{float offset=index *4.0;float dx=1.0/boneTextureWidth;vec4 m0=texture2D(smp,vec2(dx*(offset+0.5),0.));vec4 m1=texture2D(smp,vec2(dx*(offset+1.5),0.));vec4 m2=texture2D(smp,vec2(dx*(offset+2.5),0.));vec4 m3=texture2D(smp,vec2(dx*(offset+3.5),0.));return mat4(m0,m1,m2,m3);}\n#endif\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$z] = shader$z;\n\n // Do not edit.\n const name$A = \"bakedVertexAnimationDeclaration\";\n const shader$A = `#ifdef BAKED_VERTEX_ANIMATION_TEXTURE\nuniform float bakedVertexAnimationTime;uniform vec2 bakedVertexAnimationTextureSizeInverted;uniform vec4 bakedVertexAnimationSettings;uniform sampler2D bakedVertexAnimationTexture;\n#ifdef INSTANCES\nattribute vec4 bakedVertexAnimationSettingsInstanced;\n#endif\n#define inline\nmat4 readMatrixFromRawSamplerVAT(sampler2D smp,float index,float frame)\n{float offset=index*4.0;float frameUV=(frame+0.5)*bakedVertexAnimationTextureSizeInverted.y;float dx=bakedVertexAnimationTextureSizeInverted.x;vec4 m0=texture2D(smp,vec2(dx*(offset+0.5),frameUV));vec4 m1=texture2D(smp,vec2(dx*(offset+1.5),frameUV));vec4 m2=texture2D(smp,vec2(dx*(offset+2.5),frameUV));vec4 m3=texture2D(smp,vec2(dx*(offset+3.5),frameUV));return mat4(m0,m1,m2,m3);}\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$A] = shader$A;\n\n // Do not edit.\n const name$B = \"instancesDeclaration\";\n const shader$B = `#ifdef INSTANCES\nattribute vec4 world0;attribute vec4 world1;attribute vec4 world2;attribute vec4 world3;\n#ifdef INSTANCESCOLOR\nattribute vec4 instanceColor;\n#endif\n#if defined(THIN_INSTANCES) && !defined(WORLD_UBO)\nuniform mat4 world;\n#endif\n#if defined(VELOCITY) || defined(PREPASS_VELOCITY)\nattribute vec4 previousWorld0;attribute vec4 previousWorld1;attribute vec4 previousWorld2;attribute vec4 previousWorld3;\n#ifdef THIN_INSTANCES\nuniform mat4 previousWorld;\n#endif\n#endif\n#else\n#if !defined(WORLD_UBO)\nuniform mat4 world;\n#endif\n#if defined(VELOCITY) || defined(PREPASS_VELOCITY)\nuniform mat4 previousWorld;\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$B] = shader$B;\n\n // Do not edit.\n const name$C = \"prePassVertexDeclaration\";\n const shader$C = `#ifdef PREPASS\n#ifdef PREPASS_DEPTH\nvarying vec3 vViewPos;\n#endif\n#ifdef PREPASS_VELOCITY\nuniform mat4 previousViewProjection;varying vec4 vCurrentPosition;varying vec4 vPreviousPosition;\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$C] = shader$C;\n\n // Do not edit.\n const name$D = \"samplerVertexDeclaration\";\n const shader$D = `#if defined(_DEFINENAME_) && _DEFINENAME_DIRECTUV==0\nvarying vec2 v_VARYINGNAME_UV;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$D] = shader$D;\n\n // Do not edit.\n const name$E = \"bumpVertexDeclaration\";\n const shader$E = `#if defined(BUMP) || defined(PARALLAX) || defined(CLEARCOAT_BUMP) || defined(ANISOTROPIC)\n#if defined(TANGENT) && defined(NORMAL) \nvarying mat3 vTBN;\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$E] = shader$E;\n\n // Do not edit.\n const name$F = \"clipPlaneVertexDeclaration\";\n const shader$F = `#ifdef CLIPPLANE\nuniform vec4 vClipPlane;varying float fClipDistance;\n#endif\n#ifdef CLIPPLANE2\nuniform vec4 vClipPlane2;varying float fClipDistance2;\n#endif\n#ifdef CLIPPLANE3\nuniform vec4 vClipPlane3;varying float fClipDistance3;\n#endif\n#ifdef CLIPPLANE4\nuniform vec4 vClipPlane4;varying float fClipDistance4;\n#endif\n#ifdef CLIPPLANE5\nuniform vec4 vClipPlane5;varying float fClipDistance5;\n#endif\n#ifdef CLIPPLANE6\nuniform vec4 vClipPlane6;varying float fClipDistance6;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$F] = shader$F;\n\n // Do not edit.\n const name$G = \"fogVertexDeclaration\";\n const shader$G = `#ifdef FOG\nvarying vec3 vFogDistance;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$G] = shader$G;\n\n // Do not edit.\n const name$H = \"lightVxFragmentDeclaration\";\n const shader$H = `#ifdef LIGHT{X}\nuniform vec4 vLightData{X};uniform vec4 vLightDiffuse{X};\n#ifdef SPECULARTERM\nuniform vec4 vLightSpecular{X};\n#else\nvec4 vLightSpecular{X}=vec4(0.);\n#endif\n#ifdef SHADOW{X}\n#ifdef SHADOWCSM{X}\nuniform mat4 lightMatrix{X}[SHADOWCSMNUM_CASCADES{X}];varying vec4 vPositionFromLight{X}[SHADOWCSMNUM_CASCADES{X}];varying float vDepthMetric{X}[SHADOWCSMNUM_CASCADES{X}];varying vec4 vPositionFromCamera{X};\n#elif defined(SHADOWCUBE{X})\n#else\nvarying vec4 vPositionFromLight{X};varying float vDepthMetric{X};uniform mat4 lightMatrix{X};\n#endif\nuniform vec4 shadowsInfo{X};uniform vec2 depthValues{X};\n#endif\n#ifdef SPOTLIGHT{X}\nuniform vec4 vLightDirection{X};uniform vec4 vLightFalloff{X};\n#elif defined(POINTLIGHT{X})\nuniform vec4 vLightFalloff{X};\n#elif defined(HEMILIGHT{X})\nuniform vec3 vLightGround{X};\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$H] = shader$H;\n\n // Do not edit.\n const name$I = \"lightVxUboDeclaration\";\n const shader$I = `#ifdef LIGHT{X}\nuniform Light{X}\n{vec4 vLightData;vec4 vLightDiffuse;vec4 vLightSpecular;\n#ifdef SPOTLIGHT{X}\nvec4 vLightDirection;vec4 vLightFalloff;\n#elif defined(POINTLIGHT{X})\nvec4 vLightFalloff;\n#elif defined(HEMILIGHT{X})\nvec3 vLightGround;\n#endif\nvec4 shadowsInfo;vec2 depthValues;} light{X};\n#ifdef SHADOW{X}\n#ifdef SHADOWCSM{X}\nuniform mat4 lightMatrix{X}[SHADOWCSMNUM_CASCADES{X}];varying vec4 vPositionFromLight{X}[SHADOWCSMNUM_CASCADES{X}];varying float vDepthMetric{X}[SHADOWCSMNUM_CASCADES{X}];varying vec4 vPositionFromCamera{X};\n#elif defined(SHADOWCUBE{X})\n#else\nvarying vec4 vPositionFromLight{X};varying float vDepthMetric{X};uniform mat4 lightMatrix{X};\n#endif\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$I] = shader$I;\n\n // Do not edit.\n const name$J = \"morphTargetsVertexGlobalDeclaration\";\n const shader$J = `#ifdef MORPHTARGETS\nuniform float morphTargetInfluences[NUM_MORPH_INFLUENCERS];\n#ifdef MORPHTARGETS_TEXTURE \nprecision mediump sampler2DArray; \nuniform float morphTargetTextureIndices[NUM_MORPH_INFLUENCERS];uniform vec3 morphTargetTextureInfo;uniform sampler2DArray morphTargets;vec3 readVector3FromRawSampler(int targetIndex,float vertexIndex)\n{ \nfloat y=floor(vertexIndex/morphTargetTextureInfo.y);float x=vertexIndex-y*morphTargetTextureInfo.y;vec3 textureUV=vec3((x+0.5)/morphTargetTextureInfo.y,(y+0.5)/morphTargetTextureInfo.z,morphTargetTextureIndices[targetIndex]);return texture(morphTargets,textureUV).xyz;}\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$J] = shader$J;\n\n // Do not edit.\n const name$K = \"morphTargetsVertexDeclaration\";\n const shader$K = `#ifdef MORPHTARGETS\n#ifndef MORPHTARGETS_TEXTURE\nattribute vec3 position{X};\n#ifdef MORPHTARGETS_NORMAL\nattribute vec3 normal{X};\n#endif\n#ifdef MORPHTARGETS_TANGENT\nattribute vec3 tangent{X};\n#endif\n#ifdef MORPHTARGETS_UV\nattribute vec2 uv_{X};\n#endif\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$K] = shader$K;\n\n // Do not edit.\n const name$L = \"morphTargetsVertexGlobal\";\n const shader$L = `#ifdef MORPHTARGETS\n#ifdef MORPHTARGETS_TEXTURE\nfloat vertexID;\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$L] = shader$L;\n\n // Do not edit.\n const name$M = \"morphTargetsVertex\";\n const shader$M = `#ifdef MORPHTARGETS\n#ifdef MORPHTARGETS_TEXTURE \nvertexID=float(gl_VertexID)*morphTargetTextureInfo.x;positionUpdated+=(readVector3FromRawSampler({X},vertexID)-position)*morphTargetInfluences[{X}];vertexID+=1.0;\n#ifdef MORPHTARGETS_NORMAL\nnormalUpdated+=(readVector3FromRawSampler({X},vertexID) -normal)*morphTargetInfluences[{X}];vertexID+=1.0;\n#endif\n#ifdef MORPHTARGETS_UV\nuvUpdated+=(readVector3FromRawSampler({X},vertexID).xy-uv)*morphTargetInfluences[{X}];vertexID+=1.0;\n#endif\n#ifdef MORPHTARGETS_TANGENT\ntangentUpdated.xyz+=(readVector3FromRawSampler({X},vertexID) -tangent.xyz)*morphTargetInfluences[{X}];\n#endif\n#else\npositionUpdated+=(position{X}-position)*morphTargetInfluences[{X}];\n#ifdef MORPHTARGETS_NORMAL\nnormalUpdated+=(normal{X}-normal)*morphTargetInfluences[{X}];\n#endif\n#ifdef MORPHTARGETS_TANGENT\ntangentUpdated.xyz+=(tangent{X}-tangent.xyz)*morphTargetInfluences[{X}];\n#endif\n#ifdef MORPHTARGETS_UV\nuvUpdated+=(uv_{X}-uv)*morphTargetInfluences[{X}];\n#endif\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$M] = shader$M;\n\n // Do not edit.\n const name$N = \"instancesVertex\";\n const shader$N = `#ifdef INSTANCES\nmat4 finalWorld=mat4(world0,world1,world2,world3);\n#if defined(PREPASS_VELOCITY) || defined(VELOCITY)\nmat4 finalPreviousWorld=mat4(previousWorld0,previousWorld1,previousWorld2,previousWorld3);\n#endif\n#ifdef THIN_INSTANCES\nfinalWorld=world*finalWorld;\n#if defined(PREPASS_VELOCITY) || defined(VELOCITY)\nfinalPreviousWorld=previousWorld*finalPreviousWorld;\n#endif\n#endif\n#else\nmat4 finalWorld=world;\n#if defined(PREPASS_VELOCITY) || defined(VELOCITY)\nmat4 finalPreviousWorld=previousWorld;\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$N] = shader$N;\n\n // Do not edit.\n const name$O = \"bonesVertex\";\n const shader$O = `#ifndef BAKED_VERTEX_ANIMATION_TEXTURE\n#if NUM_BONE_INFLUENCERS>0\nmat4 influence;\n#ifdef BONETEXTURE\ninfluence=readMatrixFromRawSampler(boneSampler,matricesIndices[0])*matricesWeights[0];\n#if NUM_BONE_INFLUENCERS>1\ninfluence+=readMatrixFromRawSampler(boneSampler,matricesIndices[1])*matricesWeights[1];\n#endif\n#if NUM_BONE_INFLUENCERS>2\ninfluence+=readMatrixFromRawSampler(boneSampler,matricesIndices[2])*matricesWeights[2];\n#endif\n#if NUM_BONE_INFLUENCERS>3\ninfluence+=readMatrixFromRawSampler(boneSampler,matricesIndices[3])*matricesWeights[3];\n#endif\n#if NUM_BONE_INFLUENCERS>4\ninfluence+=readMatrixFromRawSampler(boneSampler,matricesIndicesExtra[0])*matricesWeightsExtra[0];\n#endif\n#if NUM_BONE_INFLUENCERS>5\ninfluence+=readMatrixFromRawSampler(boneSampler,matricesIndicesExtra[1])*matricesWeightsExtra[1];\n#endif\n#if NUM_BONE_INFLUENCERS>6\ninfluence+=readMatrixFromRawSampler(boneSampler,matricesIndicesExtra[2])*matricesWeightsExtra[2];\n#endif\n#if NUM_BONE_INFLUENCERS>7\ninfluence+=readMatrixFromRawSampler(boneSampler,matricesIndicesExtra[3])*matricesWeightsExtra[3];\n#endif\n#else\ninfluence=mBones[int(matricesIndices[0])]*matricesWeights[0];\n#if NUM_BONE_INFLUENCERS>1\ninfluence+=mBones[int(matricesIndices[1])]*matricesWeights[1];\n#endif\n#if NUM_BONE_INFLUENCERS>2\ninfluence+=mBones[int(matricesIndices[2])]*matricesWeights[2];\n#endif\n#if NUM_BONE_INFLUENCERS>3\ninfluence+=mBones[int(matricesIndices[3])]*matricesWeights[3];\n#endif\n#if NUM_BONE_INFLUENCERS>4\ninfluence+=mBones[int(matricesIndicesExtra[0])]*matricesWeightsExtra[0];\n#endif\n#if NUM_BONE_INFLUENCERS>5\ninfluence+=mBones[int(matricesIndicesExtra[1])]*matricesWeightsExtra[1];\n#endif\n#if NUM_BONE_INFLUENCERS>6\ninfluence+=mBones[int(matricesIndicesExtra[2])]*matricesWeightsExtra[2];\n#endif\n#if NUM_BONE_INFLUENCERS>7\ninfluence+=mBones[int(matricesIndicesExtra[3])]*matricesWeightsExtra[3];\n#endif\n#endif\nfinalWorld=finalWorld*influence;\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$O] = shader$O;\n\n // Do not edit.\n const name$P = \"bakedVertexAnimation\";\n const shader$P = `#ifdef BAKED_VERTEX_ANIMATION_TEXTURE\n{\n#ifdef INSTANCES\n#define BVASNAME bakedVertexAnimationSettingsInstanced\n#else\n#define BVASNAME bakedVertexAnimationSettings\n#endif\nfloat VATStartFrame=BVASNAME.x;float VATEndFrame=BVASNAME.y;float VATOffsetFrame=BVASNAME.z;float VATSpeed=BVASNAME.w;float totalFrames=VATEndFrame-VATStartFrame+1.0;float time=bakedVertexAnimationTime*VATSpeed/totalFrames;float frameCorrection=time<1.0 ? 0.0 : 1.0;float numOfFrames=totalFrames-frameCorrection;float VATFrameNum=fract(time)*numOfFrames;VATFrameNum=mod(VATFrameNum+VATOffsetFrame,numOfFrames);VATFrameNum=floor(VATFrameNum);VATFrameNum+=VATStartFrame+frameCorrection;mat4 VATInfluence;VATInfluence=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndices[0],VATFrameNum)*matricesWeights[0];\n#if NUM_BONE_INFLUENCERS>1\nVATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndices[1],VATFrameNum)*matricesWeights[1];\n#endif\n#if NUM_BONE_INFLUENCERS>2\nVATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndices[2],VATFrameNum)*matricesWeights[2];\n#endif\n#if NUM_BONE_INFLUENCERS>3\nVATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndices[3],VATFrameNum)*matricesWeights[3];\n#endif\n#if NUM_BONE_INFLUENCERS>4\nVATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndicesExtra[0],VATFrameNum)*matricesWeightsExtra[0];\n#endif\n#if NUM_BONE_INFLUENCERS>5\nVATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndicesExtra[1],VATFrameNum)*matricesWeightsExtra[1];\n#endif\n#if NUM_BONE_INFLUENCERS>6\nVATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndicesExtra[2],VATFrameNum)*matricesWeightsExtra[2];\n#endif\n#if NUM_BONE_INFLUENCERS>7\nVATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndicesExtra[3],VATFrameNum)*matricesWeightsExtra[3];\n#endif\nfinalWorld=finalWorld*VATInfluence;}\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$P] = shader$P;\n\n // Do not edit.\n const name$Q = \"prePassVertex\";\n const shader$Q = `#ifdef PREPASS_DEPTH\nvViewPos=(view*worldPos).rgb;\n#endif\n#if defined(PREPASS_VELOCITY) && defined(BONES_VELOCITY_ENABLED)\nvCurrentPosition=viewProjection*worldPos;\n#if NUM_BONE_INFLUENCERS>0\nmat4 previousInfluence;previousInfluence=mPreviousBones[int(matricesIndices[0])]*matricesWeights[0];\n#if NUM_BONE_INFLUENCERS>1\npreviousInfluence+=mPreviousBones[int(matricesIndices[1])]*matricesWeights[1];\n#endif \n#if NUM_BONE_INFLUENCERS>2\npreviousInfluence+=mPreviousBones[int(matricesIndices[2])]*matricesWeights[2];\n#endif \n#if NUM_BONE_INFLUENCERS>3\npreviousInfluence+=mPreviousBones[int(matricesIndices[3])]*matricesWeights[3];\n#endif\n#if NUM_BONE_INFLUENCERS>4\npreviousInfluence+=mPreviousBones[int(matricesIndicesExtra[0])]*matricesWeightsExtra[0];\n#endif \n#if NUM_BONE_INFLUENCERS>5\npreviousInfluence+=mPreviousBones[int(matricesIndicesExtra[1])]*matricesWeightsExtra[1];\n#endif \n#if NUM_BONE_INFLUENCERS>6\npreviousInfluence+=mPreviousBones[int(matricesIndicesExtra[2])]*matricesWeightsExtra[2];\n#endif \n#if NUM_BONE_INFLUENCERS>7\npreviousInfluence+=mPreviousBones[int(matricesIndicesExtra[3])]*matricesWeightsExtra[3];\n#endif\nvPreviousPosition=previousViewProjection*finalPreviousWorld*previousInfluence*vec4(positionUpdated,1.0);\n#else\nvPreviousPosition=previousViewProjection*finalPreviousWorld*vec4(positionUpdated,1.0);\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$Q] = shader$Q;\n\n // Do not edit.\n const name$R = \"uvVariableDeclaration\";\n const shader$R = `#if !defined(UV{X}) && defined(MAINUV{X})\nvec2 uv{X}=vec2(0.,0.);\n#endif\n#ifdef MAINUV{X}\nvMainUV{X}=uv{X};\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$R] = shader$R;\n\n // Do not edit.\n const name$S = \"samplerVertexImplementation\";\n const shader$S = `#if defined(_DEFINENAME_) && _DEFINENAME_DIRECTUV==0\nif (v_INFONAME_==0.)\n{v_VARYINGNAME_UV=vec2(_MATRIXNAME_Matrix*vec4(uvUpdated,1.0,0.0));}\n#ifdef UV2\nelse if (v_INFONAME_==1.)\n{v_VARYINGNAME_UV=vec2(_MATRIXNAME_Matrix*vec4(uv2,1.0,0.0));}\n#endif\n#ifdef UV3\nelse if (v_INFONAME_==2.)\n{v_VARYINGNAME_UV=vec2(_MATRIXNAME_Matrix*vec4(uv3,1.0,0.0));}\n#endif\n#ifdef UV4\nelse if (v_INFONAME_==3.)\n{v_VARYINGNAME_UV=vec2(_MATRIXNAME_Matrix*vec4(uv4,1.0,0.0));}\n#endif\n#ifdef UV5\nelse if (v_INFONAME_==4.)\n{v_VARYINGNAME_UV=vec2(_MATRIXNAME_Matrix*vec4(uv5,1.0,0.0));}\n#endif\n#ifdef UV6\nelse if (v_INFONAME_==5.)\n{v_VARYINGNAME_UV=vec2(_MATRIXNAME_Matrix*vec4(uv6,1.0,0.0));}\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$S] = shader$S;\n\n // Do not edit.\n const name$T = \"bumpVertex\";\n const shader$T = `#if defined(BUMP) || defined(PARALLAX) || defined(CLEARCOAT_BUMP) || defined(ANISOTROPIC)\n#if defined(TANGENT) && defined(NORMAL)\nvec3 tbnNormal=normalize(normalUpdated);vec3 tbnTangent=normalize(tangentUpdated.xyz);vec3 tbnBitangent=cross(tbnNormal,tbnTangent)*tangentUpdated.w;vTBN=mat3(finalWorld)*mat3(tbnTangent,tbnBitangent,tbnNormal);\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$T] = shader$T;\n\n // Do not edit.\n const name$U = \"clipPlaneVertex\";\n const shader$U = `#ifdef CLIPPLANE\nfClipDistance=dot(worldPos,vClipPlane);\n#endif\n#ifdef CLIPPLANE2\nfClipDistance2=dot(worldPos,vClipPlane2);\n#endif\n#ifdef CLIPPLANE3\nfClipDistance3=dot(worldPos,vClipPlane3);\n#endif\n#ifdef CLIPPLANE4\nfClipDistance4=dot(worldPos,vClipPlane4);\n#endif\n#ifdef CLIPPLANE5\nfClipDistance5=dot(worldPos,vClipPlane5);\n#endif\n#ifdef CLIPPLANE6\nfClipDistance6=dot(worldPos,vClipPlane6);\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$U] = shader$U;\n\n // Do not edit.\n const name$V = \"fogVertex\";\n const shader$V = `#ifdef FOG\nvFogDistance=(view*worldPos).xyz;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$V] = shader$V;\n\n // Do not edit.\n const name$W = \"shadowsVertex\";\n const shader$W = `#ifdef SHADOWS\n#if defined(SHADOWCSM{X})\nvPositionFromCamera{X}=view*worldPos;for (int i=0; i\n#define CUSTOM_VERTEX_BEGIN\nattribute vec3 position;\n#ifdef NORMAL\nattribute vec3 normal;\n#endif\n#ifdef TANGENT\nattribute vec4 tangent;\n#endif\n#ifdef UV1\nattribute vec2 uv;\n#endif\n#include[2..7]\n#ifdef VERTEXCOLOR\nattribute vec4 color;\n#endif\n#include\n#include\n#include\n#include\n#include\n#include[1..7]\n#include(_DEFINENAME_,DIFFUSE,_VARYINGNAME_,Diffuse)\n#include(_DEFINENAME_,DETAIL,_VARYINGNAME_,Detail)\n#include(_DEFINENAME_,AMBIENT,_VARYINGNAME_,Ambient)\n#include(_DEFINENAME_,OPACITY,_VARYINGNAME_,Opacity)\n#include(_DEFINENAME_,EMISSIVE,_VARYINGNAME_,Emissive)\n#include(_DEFINENAME_,LIGHTMAP,_VARYINGNAME_,Lightmap)\n#if defined(SPECULARTERM)\n#include(_DEFINENAME_,SPECULAR,_VARYINGNAME_,Specular)\n#endif\n#include(_DEFINENAME_,BUMP,_VARYINGNAME_,Bump)\n#include(_DEFINENAME_,DECAL,_VARYINGNAME_,Decal)\nvarying vec3 vPositionW;\n#ifdef NORMAL\nvarying vec3 vNormalW;\n#endif\n#if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES)\nvarying vec4 vColor;\n#endif\n#include\n#include\n#include\n#include<__decl__lightVxFragment>[0..maxSimultaneousLights]\n#include\n#include[0..maxSimultaneousMorphTargets]\n#ifdef REFLECTIONMAP_SKYBOX\nvarying vec3 vPositionUVW;\n#endif\n#if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED)\nvarying vec3 vDirectionW;\n#endif\n#include\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\nvec3 positionUpdated=position;\n#ifdef NORMAL\nvec3 normalUpdated=normal;\n#endif\n#ifdef TANGENT\nvec4 tangentUpdated=tangent;\n#endif\n#ifdef UV1\nvec2 uvUpdated=uv;\n#endif\n#include\n#include[0..maxSimultaneousMorphTargets]\n#ifdef REFLECTIONMAP_SKYBOX\nvPositionUVW=positionUpdated;\n#endif\n#define CUSTOM_VERTEX_UPDATE_POSITION\n#define CUSTOM_VERTEX_UPDATE_NORMAL\n#include\n#if defined(PREPASS) && defined(PREPASS_VELOCITY) && !defined(BONES_VELOCITY_ENABLED)\nvCurrentPosition=viewProjection*finalWorld*vec4(positionUpdated,1.0);vPreviousPosition=previousViewProjection*finalPreviousWorld*vec4(positionUpdated,1.0);\n#endif\n#include\n#include\nvec4 worldPos=finalWorld*vec4(positionUpdated,1.0);\n#ifdef NORMAL\nmat3 normalWorld=mat3(finalWorld);\n#if defined(INSTANCES) && defined(THIN_INSTANCES)\nvNormalW=normalUpdated/vec3(dot(normalWorld[0],normalWorld[0]),dot(normalWorld[1],normalWorld[1]),dot(normalWorld[2],normalWorld[2]));vNormalW=normalize(normalWorld*vNormalW);\n#else\n#ifdef NONUNIFORMSCALING\nnormalWorld=transposeMat3(inverseMat3(normalWorld));\n#endif\nvNormalW=normalize(normalWorld*normalUpdated);\n#endif\n#endif\n#define CUSTOM_VERTEX_UPDATE_WORLDPOS\n#ifdef MULTIVIEW\nif (gl_ViewID_OVR==0u) {gl_Position=viewProjection*worldPos;} else {gl_Position=viewProjectionR*worldPos;}\n#else\ngl_Position=viewProjection*worldPos;\n#endif\nvPositionW=vec3(worldPos);\n#include\n#if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED)\nvDirectionW=normalize(vec3(finalWorld*vec4(positionUpdated,0.0)));\n#endif\n#ifndef UV1\nvec2 uvUpdated=vec2(0.,0.);\n#endif\n#ifdef MAINUV1\nvMainUV1=uvUpdated;\n#endif\n#include[2..7]\n#include(_DEFINENAME_,DIFFUSE,_VARYINGNAME_,Diffuse,_MATRIXNAME_,diffuse,_INFONAME_,DiffuseInfos.x)\n#include(_DEFINENAME_,DETAIL,_VARYINGNAME_,Detail,_MATRIXNAME_,detail,_INFONAME_,DetailInfos.x)\n#include(_DEFINENAME_,AMBIENT,_VARYINGNAME_,Ambient,_MATRIXNAME_,ambient,_INFONAME_,AmbientInfos.x)\n#include(_DEFINENAME_,OPACITY,_VARYINGNAME_,Opacity,_MATRIXNAME_,opacity,_INFONAME_,OpacityInfos.x)\n#include(_DEFINENAME_,EMISSIVE,_VARYINGNAME_,Emissive,_MATRIXNAME_,emissive,_INFONAME_,EmissiveInfos.x)\n#include(_DEFINENAME_,LIGHTMAP,_VARYINGNAME_,Lightmap,_MATRIXNAME_,lightmap,_INFONAME_,LightmapInfos.x)\n#if defined(SPECULARTERM)\n#include(_DEFINENAME_,SPECULAR,_VARYINGNAME_,Specular,_MATRIXNAME_,specular,_INFONAME_,SpecularInfos.x)\n#endif\n#include(_DEFINENAME_,BUMP,_VARYINGNAME_,Bump,_MATRIXNAME_,bump,_INFONAME_,BumpInfos.x)\n#include(_DEFINENAME_,DECAL,_VARYINGNAME_,Decal,_MATRIXNAME_,decal,_INFONAME_,DecalInfos.x)\n#include\n#include\n#include\n#include[0..maxSimultaneousLights]\n#include\n#include\n#include\n#define CUSTOM_VERTEX_MAIN_END\n}\n`;\n // Sideeffect\n ShaderStore.ShadersStore[name$_] = shader$_;\n\n /**\n * EffectFallbacks can be used to add fallbacks (properties to disable) to certain properties when desired to improve performance.\n * (Eg. Start at high quality with reflection and fog, if fps is low, remove reflection, if still low remove fog)\n */\n class EffectFallbacks {\n constructor() {\n this._defines = {};\n this._currentRank = 32;\n this._maxRank = -1;\n this._mesh = null;\n }\n /**\n * Removes the fallback from the bound mesh.\n */\n unBindMesh() {\n this._mesh = null;\n }\n /**\n * Adds a fallback on the specified property.\n * @param rank The rank of the fallback (Lower ranks will be fallbacked to first)\n * @param define The name of the define in the shader\n */\n addFallback(rank, define) {\n if (!this._defines[rank]) {\n if (rank < this._currentRank) {\n this._currentRank = rank;\n }\n if (rank > this._maxRank) {\n this._maxRank = rank;\n }\n this._defines[rank] = new Array();\n }\n this._defines[rank].push(define);\n }\n /**\n * Sets the mesh to use CPU skinning when needing to fallback.\n * @param rank The rank of the fallback (Lower ranks will be fallbacked to first)\n * @param mesh The mesh to use the fallbacks.\n */\n addCPUSkinningFallback(rank, mesh) {\n this._mesh = mesh;\n if (rank < this._currentRank) {\n this._currentRank = rank;\n }\n if (rank > this._maxRank) {\n this._maxRank = rank;\n }\n }\n /**\n * Checks to see if more fallbacks are still available.\n */\n get hasMoreFallbacks() {\n return this._currentRank <= this._maxRank;\n }\n /**\n * Removes the defines that should be removed when falling back.\n * @param currentDefines defines the current define statements for the shader.\n * @param effect defines the current effect we try to compile\n * @returns The resulting defines with defines of the current rank removed.\n */\n reduce(currentDefines, effect) {\n // First we try to switch to CPU skinning\n if (this._mesh && this._mesh.computeBonesUsingShaders && this._mesh.numBoneInfluencers > 0) {\n this._mesh.computeBonesUsingShaders = false;\n currentDefines = currentDefines.replace(\"#define NUM_BONE_INFLUENCERS \" + this._mesh.numBoneInfluencers, \"#define NUM_BONE_INFLUENCERS 0\");\n effect._bonesComputationForcedToCPU = true;\n const scene = this._mesh.getScene();\n for (let index = 0; index < scene.meshes.length; index++) {\n const otherMesh = scene.meshes[index];\n if (!otherMesh.material) {\n if (!this._mesh.material && otherMesh.computeBonesUsingShaders && otherMesh.numBoneInfluencers > 0) {\n otherMesh.computeBonesUsingShaders = false;\n }\n continue;\n }\n if (!otherMesh.computeBonesUsingShaders || otherMesh.numBoneInfluencers === 0) {\n continue;\n }\n if (otherMesh.material.getEffect() === effect) {\n otherMesh.computeBonesUsingShaders = false;\n }\n else if (otherMesh.subMeshes) {\n for (const subMesh of otherMesh.subMeshes) {\n const subMeshEffect = subMesh.effect;\n if (subMeshEffect === effect) {\n otherMesh.computeBonesUsingShaders = false;\n break;\n }\n }\n }\n }\n }\n else {\n const currentFallbacks = this._defines[this._currentRank];\n if (currentFallbacks) {\n for (let index = 0; index < currentFallbacks.length; index++) {\n currentDefines = currentDefines.replace(\"#define \" + currentFallbacks[index], \"\");\n }\n }\n this._currentRank++;\n }\n return currentDefines;\n }\n }\n\n const rxOption = new RegExp(\"^([gimus]+)!\");\n /**\n * Class that manages the plugins of a material\n * @since 5.0\n */\n class MaterialPluginManager {\n /**\n * Creates a new instance of the plugin manager\n * @param material material that this manager will manage the plugins for\n */\n constructor(material) {\n /** @internal */\n this._plugins = [];\n this._activePlugins = [];\n this._activePluginsForExtraEvents = [];\n this._material = material;\n this._scene = material.getScene();\n this._engine = this._scene.getEngine();\n }\n /**\n * @internal\n */\n _addPlugin(plugin) {\n for (let i = 0; i < this._plugins.length; ++i) {\n if (this._plugins[i].name === plugin.name) {\n return false;\n }\n }\n if (this._material._uniformBufferLayoutBuilt) {\n throw `The plugin \"${plugin.name}\" can't be added to the material \"${this._material.name}\" because this material has already been used for rendering! Please add plugins to materials before any rendering with this material occurs.`;\n }\n const pluginClassName = plugin.getClassName();\n if (!MaterialPluginManager._MaterialPluginClassToMainDefine[pluginClassName]) {\n MaterialPluginManager._MaterialPluginClassToMainDefine[pluginClassName] = \"MATERIALPLUGIN_\" + ++MaterialPluginManager._MaterialPluginCounter;\n }\n this._material._callbackPluginEventGeneric = this._handlePluginEvent.bind(this);\n this._plugins.push(plugin);\n this._plugins.sort((a, b) => a.priority - b.priority);\n this._codeInjectionPoints = {};\n const defineNamesFromPlugins = {};\n defineNamesFromPlugins[MaterialPluginManager._MaterialPluginClassToMainDefine[pluginClassName]] = {\n type: \"boolean\",\n default: true,\n };\n for (const plugin of this._plugins) {\n plugin.collectDefines(defineNamesFromPlugins);\n this._collectPointNames(\"vertex\", plugin.getCustomCode(\"vertex\"));\n this._collectPointNames(\"fragment\", plugin.getCustomCode(\"fragment\"));\n }\n this._defineNamesFromPlugins = defineNamesFromPlugins;\n return true;\n }\n /**\n * @internal\n */\n _activatePlugin(plugin) {\n if (this._activePlugins.indexOf(plugin) === -1) {\n this._activePlugins.push(plugin);\n this._activePlugins.sort((a, b) => a.priority - b.priority);\n this._material._callbackPluginEventIsReadyForSubMesh = this._handlePluginEventIsReadyForSubMesh.bind(this);\n this._material._callbackPluginEventPrepareDefinesBeforeAttributes = this._handlePluginEventPrepareDefinesBeforeAttributes.bind(this);\n this._material._callbackPluginEventPrepareDefines = this._handlePluginEventPrepareDefines.bind(this);\n this._material._callbackPluginEventBindForSubMesh = this._handlePluginEventBindForSubMesh.bind(this);\n if (plugin.registerForExtraEvents) {\n this._activePluginsForExtraEvents.push(plugin);\n this._activePluginsForExtraEvents.sort((a, b) => a.priority - b.priority);\n this._material._callbackPluginEventHasRenderTargetTextures = this._handlePluginEventHasRenderTargetTextures.bind(this);\n this._material._callbackPluginEventFillRenderTargetTextures = this._handlePluginEventFillRenderTargetTextures.bind(this);\n this._material._callbackPluginEventHardBindForSubMesh = this._handlePluginEventHardBindForSubMesh.bind(this);\n }\n }\n }\n /**\n * Gets a plugin from the list of plugins managed by this manager\n * @param name name of the plugin\n * @returns the plugin if found, else null\n */\n getPlugin(name) {\n for (let i = 0; i < this._plugins.length; ++i) {\n if (this._plugins[i].name === name) {\n return this._plugins[i];\n }\n }\n return null;\n }\n _handlePluginEventIsReadyForSubMesh(eventData) {\n let isReady = true;\n for (const plugin of this._activePlugins) {\n isReady = isReady && plugin.isReadyForSubMesh(eventData.defines, this._scene, this._engine, eventData.subMesh);\n }\n eventData.isReadyForSubMesh = isReady;\n }\n _handlePluginEventPrepareDefinesBeforeAttributes(eventData) {\n for (const plugin of this._activePlugins) {\n plugin.prepareDefinesBeforeAttributes(eventData.defines, this._scene, eventData.mesh);\n }\n }\n _handlePluginEventPrepareDefines(eventData) {\n for (const plugin of this._activePlugins) {\n plugin.prepareDefines(eventData.defines, this._scene, eventData.mesh);\n }\n }\n _handlePluginEventHardBindForSubMesh(eventData) {\n for (const plugin of this._activePluginsForExtraEvents) {\n plugin.hardBindForSubMesh(this._material._uniformBuffer, this._scene, this._engine, eventData.subMesh);\n }\n }\n _handlePluginEventBindForSubMesh(eventData) {\n for (const plugin of this._activePlugins) {\n plugin.bindForSubMesh(this._material._uniformBuffer, this._scene, this._engine, eventData.subMesh);\n }\n }\n _handlePluginEventHasRenderTargetTextures(eventData) {\n let hasRenderTargetTextures = false;\n for (const plugin of this._activePluginsForExtraEvents) {\n hasRenderTargetTextures = plugin.hasRenderTargetTextures();\n if (hasRenderTargetTextures) {\n break;\n }\n }\n eventData.hasRenderTargetTextures = hasRenderTargetTextures;\n }\n _handlePluginEventFillRenderTargetTextures(eventData) {\n for (const plugin of this._activePluginsForExtraEvents) {\n plugin.fillRenderTargetTextures(eventData.renderTargets);\n }\n }\n _handlePluginEvent(id, info) {\n var _a;\n switch (id) {\n case MaterialPluginEvent.GetActiveTextures: {\n const eventData = info;\n for (const plugin of this._activePlugins) {\n plugin.getActiveTextures(eventData.activeTextures);\n }\n break;\n }\n case MaterialPluginEvent.GetAnimatables: {\n const eventData = info;\n for (const plugin of this._activePlugins) {\n plugin.getAnimatables(eventData.animatables);\n }\n break;\n }\n case MaterialPluginEvent.HasTexture: {\n const eventData = info;\n let hasTexture = false;\n for (const plugin of this._activePlugins) {\n hasTexture = plugin.hasTexture(eventData.texture);\n if (hasTexture) {\n break;\n }\n }\n eventData.hasTexture = hasTexture;\n break;\n }\n case MaterialPluginEvent.Disposed: {\n const eventData = info;\n for (const plugin of this._plugins) {\n plugin.dispose(eventData.forceDisposeTextures);\n }\n break;\n }\n case MaterialPluginEvent.GetDefineNames: {\n const eventData = info;\n eventData.defineNames = this._defineNamesFromPlugins;\n break;\n }\n case MaterialPluginEvent.PrepareEffect: {\n const eventData = info;\n for (const plugin of this._activePlugins) {\n eventData.fallbackRank = plugin.addFallbacks(eventData.defines, eventData.fallbacks, eventData.fallbackRank);\n plugin.getAttributes(eventData.attributes, this._scene, eventData.mesh);\n }\n if (this._uniformList.length > 0) {\n eventData.uniforms.push(...this._uniformList);\n }\n if (this._samplerList.length > 0) {\n eventData.samplers.push(...this._samplerList);\n }\n if (this._uboList.length > 0) {\n eventData.uniformBuffersNames.push(...this._uboList);\n }\n eventData.customCode = this._injectCustomCode(eventData, eventData.customCode);\n break;\n }\n case MaterialPluginEvent.PrepareUniformBuffer: {\n const eventData = info;\n this._uboDeclaration = \"\";\n this._vertexDeclaration = \"\";\n this._fragmentDeclaration = \"\";\n this._uniformList = [];\n this._samplerList = [];\n this._uboList = [];\n for (const plugin of this._plugins) {\n const uniforms = plugin.getUniforms();\n if (uniforms) {\n if (uniforms.ubo) {\n for (const uniform of uniforms.ubo) {\n if (uniform.size && uniform.type) {\n const arraySize = (_a = uniform.arraySize) !== null && _a !== void 0 ? _a : 0;\n eventData.ubo.addUniform(uniform.name, uniform.size, arraySize);\n this._uboDeclaration += `${uniform.type} ${uniform.name}${arraySize > 0 ? `[${arraySize}]` : \"\"};\\n`;\n }\n this._uniformList.push(uniform.name);\n }\n }\n if (uniforms.vertex) {\n this._vertexDeclaration += uniforms.vertex + \"\\n\";\n }\n if (uniforms.fragment) {\n this._fragmentDeclaration += uniforms.fragment + \"\\n\";\n }\n }\n plugin.getSamplers(this._samplerList);\n plugin.getUniformBuffersNames(this._uboList);\n }\n break;\n }\n }\n }\n _collectPointNames(shaderType, customCode) {\n if (!customCode) {\n return;\n }\n for (const pointName in customCode) {\n if (!this._codeInjectionPoints[shaderType]) {\n this._codeInjectionPoints[shaderType] = {};\n }\n this._codeInjectionPoints[shaderType][pointName] = true;\n }\n }\n _injectCustomCode(eventData, existingCallback) {\n return (shaderType, code) => {\n var _a, _b;\n if (existingCallback) {\n code = existingCallback(shaderType, code);\n }\n if (this._uboDeclaration) {\n code = code.replace(\"#define ADDITIONAL_UBO_DECLARATION\", this._uboDeclaration);\n }\n if (this._vertexDeclaration) {\n code = code.replace(\"#define ADDITIONAL_VERTEX_DECLARATION\", this._vertexDeclaration);\n }\n if (this._fragmentDeclaration) {\n code = code.replace(\"#define ADDITIONAL_FRAGMENT_DECLARATION\", this._fragmentDeclaration);\n }\n const points = (_a = this._codeInjectionPoints) === null || _a === void 0 ? void 0 : _a[shaderType];\n if (!points) {\n return code;\n }\n let processorOptions = null;\n for (let pointName in points) {\n let injectedCode = \"\";\n for (const plugin of this._activePlugins) {\n let customCode = (_b = plugin.getCustomCode(shaderType)) === null || _b === void 0 ? void 0 : _b[pointName];\n if (!customCode) {\n continue;\n }\n if (plugin.resolveIncludes) {\n if (processorOptions === null) {\n const shaderLanguage = ShaderLanguage.GLSL;\n processorOptions = {\n defines: [],\n indexParameters: eventData.indexParameters,\n isFragment: false,\n shouldUseHighPrecisionShader: this._engine._shouldUseHighPrecisionShader,\n processor: undefined,\n supportsUniformBuffers: this._engine.supportsUniformBuffers,\n shadersRepository: ShaderStore.GetShadersRepository(shaderLanguage),\n includesShadersStore: ShaderStore.GetIncludesShadersStore(shaderLanguage),\n version: undefined,\n platformName: this._engine.shaderPlatformName,\n processingContext: undefined,\n isNDCHalfZRange: this._engine.isNDCHalfZRange,\n useReverseDepthBuffer: this._engine.useReverseDepthBuffer,\n processCodeAfterIncludes: undefined, // not used by _ProcessIncludes\n };\n }\n processorOptions.isFragment = shaderType === \"fragment\";\n ShaderProcessor._ProcessIncludes(customCode, processorOptions, (code) => (customCode = code));\n }\n injectedCode += customCode + \"\\n\";\n }\n if (injectedCode.length > 0) {\n if (pointName.charAt(0) === \"!\") {\n // pointName is a regular expression\n pointName = pointName.substring(1);\n let regexFlags = \"g\";\n if (pointName.charAt(0) === \"!\") {\n // no flags\n regexFlags = \"\";\n pointName = pointName.substring(1);\n }\n else {\n // get the flag(s)\n const matchOption = rxOption.exec(pointName);\n if (matchOption && matchOption.length >= 2) {\n regexFlags = matchOption[1];\n pointName = pointName.substring(regexFlags.length + 1);\n }\n }\n if (regexFlags.indexOf(\"g\") < 0) {\n // we force the \"g\" flag so that the regexp object is stateful!\n regexFlags += \"g\";\n }\n const sourceCode = code;\n const rx = new RegExp(pointName, regexFlags);\n let match = rx.exec(sourceCode);\n while (match !== null) {\n let newCode = injectedCode;\n for (let i = 0; i < match.length; ++i) {\n newCode = newCode.replace(\"$\" + i, match[i]);\n }\n code = code.replace(match[0], newCode);\n match = rx.exec(sourceCode);\n }\n }\n else {\n const fullPointName = \"#define \" + pointName;\n code = code.replace(fullPointName, \"\\n\" + injectedCode + \"\\n\" + fullPointName);\n }\n }\n }\n return code;\n };\n }\n }\n /** Map a plugin class name to a #define name (used in the vertex/fragment shaders as a marker of the plugin usage) */\n MaterialPluginManager._MaterialPluginClassToMainDefine = {};\n MaterialPluginManager._MaterialPluginCounter = 0;\n (() => {\n EngineStore.OnEnginesDisposedObservable.add(() => {\n UnregisterAllMaterialPlugins();\n });\n })();\n const plugins = [];\n let observer = null;\n /**\n * Clear the list of global material plugins\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n function UnregisterAllMaterialPlugins() {\n plugins.length = 0;\n Material.OnEventObservable.remove(observer);\n observer = null;\n }\n\n /**\n * Base class for material plugins.\n * @since 5.0\n */\n class MaterialPluginBase {\n _enable(enable) {\n if (enable) {\n this._pluginManager._activatePlugin(this);\n }\n }\n /**\n * Creates a new material plugin\n * @param material parent material of the plugin\n * @param name name of the plugin\n * @param priority priority of the plugin\n * @param defines list of defines used by the plugin. The value of the property is the default value for this property\n * @param addToPluginList true to add the plugin to the list of plugins managed by the material plugin manager of the material (default: true)\n * @param enable true to enable the plugin (it is handy if the plugin does not handle properties to switch its current activation)\n * @param resolveIncludes Indicates that any #include directive in the plugin code must be replaced by the corresponding code (default: false)\n */\n constructor(material, name, priority, defines, addToPluginList = true, enable = false, resolveIncludes = false) {\n /**\n * Defines the priority of the plugin. Lower numbers run first.\n */\n this.priority = 500;\n /**\n * Indicates that any #include directive in the plugin code must be replaced by the corresponding code.\n */\n this.resolveIncludes = false;\n /**\n * Indicates that this plugin should be notified for the extra events (HasRenderTargetTextures / FillRenderTargetTextures / HardBindForSubMesh)\n */\n this.registerForExtraEvents = false;\n this._material = material;\n this.name = name;\n this.priority = priority;\n this.resolveIncludes = resolveIncludes;\n if (!material.pluginManager) {\n material.pluginManager = new MaterialPluginManager(material);\n material.onDisposeObservable.add(() => {\n material.pluginManager = undefined;\n });\n }\n this._pluginDefineNames = defines;\n this._pluginManager = material.pluginManager;\n if (addToPluginList) {\n this._pluginManager._addPlugin(this);\n }\n if (enable) {\n this._enable(true);\n }\n this.markAllDefinesAsDirty = material._dirtyCallbacks[63];\n }\n /**\n * Gets the current class name useful for serialization or dynamic coding.\n * @returns The class name.\n */\n getClassName() {\n return \"MaterialPluginBase\";\n }\n /**\n * Specifies that the submesh is ready to be used.\n * @param defines the list of \"defines\" to update.\n * @param scene defines the scene the material belongs to.\n * @param engine the engine this scene belongs to.\n * @param subMesh the submesh to check for readiness\n * @returns - boolean indicating that the submesh is ready or not.\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n isReadyForSubMesh(defines, scene, engine, subMesh) {\n return true;\n }\n /**\n * Binds the material data (this function is called even if mustRebind() returns false)\n * @param uniformBuffer defines the Uniform buffer to fill in.\n * @param scene defines the scene the material belongs to.\n * @param engine defines the engine the material belongs to.\n * @param subMesh the submesh to bind data for\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n hardBindForSubMesh(uniformBuffer, scene, engine, subMesh) { }\n /**\n * Binds the material data.\n * @param uniformBuffer defines the Uniform buffer to fill in.\n * @param scene defines the scene the material belongs to.\n * @param engine the engine this scene belongs to.\n * @param subMesh the submesh to bind data for\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n bindForSubMesh(uniformBuffer, scene, engine, subMesh) { }\n /**\n * Disposes the resources of the material.\n * @param forceDisposeTextures - Forces the disposal of all textures.\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n dispose(forceDisposeTextures) { }\n /**\n * Returns a list of custom shader code fragments to customize the shader.\n * @param shaderType \"vertex\" or \"fragment\"\n * @returns null if no code to be added, or a list of pointName =\\> code.\n * Note that `pointName` can also be a regular expression if it starts with a `!`.\n * In that case, the string found by the regular expression (if any) will be\n * replaced by the code provided.\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n getCustomCode(shaderType) {\n return null;\n }\n /**\n * Collects all defines.\n * @param defines The object to append to.\n */\n collectDefines(defines) {\n if (!this._pluginDefineNames) {\n return;\n }\n for (const key of Object.keys(this._pluginDefineNames)) {\n if (key[0] === \"_\") {\n continue;\n }\n const type = typeof this._pluginDefineNames[key];\n defines[key] = {\n type: type === \"number\" ? \"number\" : type === \"string\" ? \"string\" : type === \"boolean\" ? \"boolean\" : \"object\",\n default: this._pluginDefineNames[key],\n };\n }\n }\n /**\n * Sets the defines for the next rendering. Called before MaterialHelper.PrepareDefinesForAttributes is called.\n * @param defines the list of \"defines\" to update.\n * @param scene defines the scene to the material belongs to.\n * @param mesh the mesh being rendered\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n prepareDefinesBeforeAttributes(defines, scene, mesh) { }\n /**\n * Sets the defines for the next rendering\n * @param defines the list of \"defines\" to update.\n * @param scene defines the scene to the material belongs to.\n * @param mesh the mesh being rendered\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n prepareDefines(defines, scene, mesh) { }\n /**\n * Checks to see if a texture is used in the material.\n * @param texture - Base texture to use.\n * @returns - Boolean specifying if a texture is used in the material.\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n hasTexture(texture) {\n return false;\n }\n /**\n * Gets a boolean indicating that current material needs to register RTT\n * @returns true if this uses a render target otherwise false.\n */\n hasRenderTargetTextures() {\n return false;\n }\n /**\n * Fills the list of render target textures.\n * @param renderTargets the list of render targets to update\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n fillRenderTargetTextures(renderTargets) { }\n /**\n * Returns an array of the actively used textures.\n * @param activeTextures Array of BaseTextures\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n getActiveTextures(activeTextures) { }\n /**\n * Returns the animatable textures.\n * @param animatables Array of animatable textures.\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n getAnimatables(animatables) { }\n /**\n * Add fallbacks to the effect fallbacks list.\n * @param defines defines the Base texture to use.\n * @param fallbacks defines the current fallback list.\n * @param currentRank defines the current fallback rank.\n * @returns the new fallback rank.\n */\n addFallbacks(defines, fallbacks, currentRank) {\n return currentRank;\n }\n /**\n * Gets the samplers used by the plugin.\n * @param samplers list that the sampler names should be added to.\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n getSamplers(samplers) { }\n /**\n * Gets the attributes used by the plugin.\n * @param attributes list that the attribute names should be added to.\n * @param scene the scene that the material belongs to.\n * @param mesh the mesh being rendered.\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n getAttributes(attributes, scene, mesh) { }\n /**\n * Gets the uniform buffers names added by the plugin.\n * @param ubos list that the ubo names should be added to.\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n getUniformBuffersNames(ubos) { }\n /**\n * Gets the description of the uniforms to add to the ubo (if engine supports ubos) or to inject directly in the vertex/fragment shaders (if engine does not support ubos)\n * @returns the description of the uniforms\n */\n getUniforms() {\n return {};\n }\n /**\n * Makes a duplicate of the current configuration into another one.\n * @param plugin define the config where to copy the info\n */\n copyTo(plugin) {\n SerializationHelper.Clone(() => plugin, this);\n }\n /**\n * Serializes this plugin configuration.\n * @returns - An object with the serialized config.\n */\n serialize() {\n return SerializationHelper.Serialize(this);\n }\n /**\n * Parses a plugin configuration from a serialized object.\n * @param source - Serialized object.\n * @param scene Defines the scene we are parsing for\n * @param rootUrl Defines the rootUrl to load from\n */\n parse(source, scene, rootUrl) {\n SerializationHelper.Parse(() => this, source, scene, rootUrl);\n }\n }\n __decorate$1([\n serialize()\n ], MaterialPluginBase.prototype, \"name\", void 0);\n __decorate$1([\n serialize()\n ], MaterialPluginBase.prototype, \"priority\", void 0);\n __decorate$1([\n serialize()\n ], MaterialPluginBase.prototype, \"resolveIncludes\", void 0);\n __decorate$1([\n serialize()\n ], MaterialPluginBase.prototype, \"registerForExtraEvents\", void 0);\n\n /**\n * @internal\n */\n class MaterialDetailMapDefines extends MaterialDefines {\n constructor() {\n super(...arguments);\n this.DETAIL = false;\n this.DETAILDIRECTUV = 0;\n this.DETAIL_NORMALBLENDMETHOD = 0;\n }\n }\n /**\n * Plugin that implements the detail map component of a material\n *\n * Inspired from:\n * Unity: https://docs.unity3d.com/Packages/com.unity.render-pipelines.high-definition@9.0/manual/Mask-Map-and-Detail-Map.html and https://docs.unity3d.com/Manual/StandardShaderMaterialParameterDetail.html\n * Unreal: https://docs.unrealengine.com/en-US/Engine/Rendering/Materials/HowTo/DetailTexturing/index.html\n * Cryengine: https://docs.cryengine.com/display/SDKDOC2/Detail+Maps\n */\n class DetailMapConfiguration extends MaterialPluginBase {\n /** @internal */\n _markAllSubMeshesAsTexturesDirty() {\n this._enable(this._isEnabled);\n this._internalMarkAllSubMeshesAsTexturesDirty();\n }\n constructor(material, addToPluginList = true) {\n super(material, \"DetailMap\", 140, new MaterialDetailMapDefines(), addToPluginList);\n this._texture = null;\n /**\n * Defines how strongly the detail diffuse/albedo channel is blended with the regular diffuse/albedo texture\n * Bigger values mean stronger blending\n */\n this.diffuseBlendLevel = 1;\n /**\n * Defines how strongly the detail roughness channel is blended with the regular roughness value\n * Bigger values mean stronger blending. Only used with PBR materials\n */\n this.roughnessBlendLevel = 1;\n /**\n * Defines how strong the bump effect from the detail map is\n * Bigger values mean stronger effect\n */\n this.bumpLevel = 1;\n this._normalBlendMethod = Material.MATERIAL_NORMALBLENDMETHOD_WHITEOUT;\n this._isEnabled = false;\n /**\n * Enable or disable the detail map on this material\n */\n this.isEnabled = false;\n this._internalMarkAllSubMeshesAsTexturesDirty = material._dirtyCallbacks[1];\n }\n isReadyForSubMesh(defines, scene, engine) {\n if (!this._isEnabled) {\n return true;\n }\n if (defines._areTexturesDirty && scene.texturesEnabled) {\n if (engine.getCaps().standardDerivatives && this._texture && MaterialFlags.DetailTextureEnabled) {\n // Detail texture cannot be not blocking.\n if (!this._texture.isReady()) {\n return false;\n }\n }\n }\n return true;\n }\n prepareDefines(defines, scene) {\n if (this._isEnabled) {\n defines.DETAIL_NORMALBLENDMETHOD = this._normalBlendMethod;\n const engine = scene.getEngine();\n if (defines._areTexturesDirty) {\n if (engine.getCaps().standardDerivatives && this._texture && MaterialFlags.DetailTextureEnabled && this._isEnabled) {\n MaterialHelper.PrepareDefinesForMergedUV(this._texture, defines, \"DETAIL\");\n defines.DETAIL_NORMALBLENDMETHOD = this._normalBlendMethod;\n }\n else {\n defines.DETAIL = false;\n }\n }\n }\n else {\n defines.DETAIL = false;\n }\n }\n bindForSubMesh(uniformBuffer, scene) {\n if (!this._isEnabled) {\n return;\n }\n const isFrozen = this._material.isFrozen;\n if (!uniformBuffer.useUbo || !isFrozen || !uniformBuffer.isSync) {\n if (this._texture && MaterialFlags.DetailTextureEnabled) {\n uniformBuffer.updateFloat4(\"vDetailInfos\", this._texture.coordinatesIndex, this.diffuseBlendLevel, this.bumpLevel, this.roughnessBlendLevel);\n MaterialHelper.BindTextureMatrix(this._texture, uniformBuffer, \"detail\");\n }\n }\n // Textures\n if (scene.texturesEnabled) {\n if (this._texture && MaterialFlags.DetailTextureEnabled) {\n uniformBuffer.setTexture(\"detailSampler\", this._texture);\n }\n }\n }\n hasTexture(texture) {\n if (this._texture === texture) {\n return true;\n }\n return false;\n }\n getActiveTextures(activeTextures) {\n if (this._texture) {\n activeTextures.push(this._texture);\n }\n }\n getAnimatables(animatables) {\n if (this._texture && this._texture.animations && this._texture.animations.length > 0) {\n animatables.push(this._texture);\n }\n }\n dispose(forceDisposeTextures) {\n var _a;\n if (forceDisposeTextures) {\n (_a = this._texture) === null || _a === void 0 ? void 0 : _a.dispose();\n }\n }\n getClassName() {\n return \"DetailMapConfiguration\";\n }\n getSamplers(samplers) {\n samplers.push(\"detailSampler\");\n }\n getUniforms() {\n return {\n ubo: [\n { name: \"vDetailInfos\", size: 4, type: \"vec4\" },\n { name: \"detailMatrix\", size: 16, type: \"mat4\" },\n ],\n };\n }\n }\n __decorate$1([\n serializeAsTexture(\"detailTexture\"),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], DetailMapConfiguration.prototype, \"texture\", void 0);\n __decorate$1([\n serialize()\n ], DetailMapConfiguration.prototype, \"diffuseBlendLevel\", void 0);\n __decorate$1([\n serialize()\n ], DetailMapConfiguration.prototype, \"roughnessBlendLevel\", void 0);\n __decorate$1([\n serialize()\n ], DetailMapConfiguration.prototype, \"bumpLevel\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], DetailMapConfiguration.prototype, \"normalBlendMethod\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], DetailMapConfiguration.prototype, \"isEnabled\", void 0);\n\n const onCreatedEffectParameters = { effect: null, subMesh: null };\n /** @internal */\n class StandardMaterialDefines extends MaterialDefines {\n /**\n * Initializes the Standard Material defines.\n * @param externalProperties The external properties\n */\n constructor(externalProperties) {\n super(externalProperties);\n this.MAINUV1 = false;\n this.MAINUV2 = false;\n this.MAINUV3 = false;\n this.MAINUV4 = false;\n this.MAINUV5 = false;\n this.MAINUV6 = false;\n this.DIFFUSE = false;\n this.DIFFUSEDIRECTUV = 0;\n this.BAKED_VERTEX_ANIMATION_TEXTURE = false;\n this.AMBIENT = false;\n this.AMBIENTDIRECTUV = 0;\n this.OPACITY = false;\n this.OPACITYDIRECTUV = 0;\n this.OPACITYRGB = false;\n this.REFLECTION = false;\n this.EMISSIVE = false;\n this.EMISSIVEDIRECTUV = 0;\n this.SPECULAR = false;\n this.SPECULARDIRECTUV = 0;\n this.BUMP = false;\n this.BUMPDIRECTUV = 0;\n this.PARALLAX = false;\n this.PARALLAXOCCLUSION = false;\n this.SPECULAROVERALPHA = false;\n this.CLIPPLANE = false;\n this.CLIPPLANE2 = false;\n this.CLIPPLANE3 = false;\n this.CLIPPLANE4 = false;\n this.CLIPPLANE5 = false;\n this.CLIPPLANE6 = false;\n this.ALPHATEST = false;\n this.DEPTHPREPASS = false;\n this.ALPHAFROMDIFFUSE = false;\n this.POINTSIZE = false;\n this.FOG = false;\n this.SPECULARTERM = false;\n this.DIFFUSEFRESNEL = false;\n this.OPACITYFRESNEL = false;\n this.REFLECTIONFRESNEL = false;\n this.REFRACTIONFRESNEL = false;\n this.EMISSIVEFRESNEL = false;\n this.FRESNEL = false;\n this.NORMAL = false;\n this.TANGENT = false;\n this.UV1 = false;\n this.UV2 = false;\n this.UV3 = false;\n this.UV4 = false;\n this.UV5 = false;\n this.UV6 = false;\n this.VERTEXCOLOR = false;\n this.VERTEXALPHA = false;\n this.NUM_BONE_INFLUENCERS = 0;\n this.BonesPerMesh = 0;\n this.BONETEXTURE = false;\n this.BONES_VELOCITY_ENABLED = false;\n this.INSTANCES = false;\n this.THIN_INSTANCES = false;\n this.INSTANCESCOLOR = false;\n this.GLOSSINESS = false;\n this.ROUGHNESS = false;\n this.EMISSIVEASILLUMINATION = false;\n this.LINKEMISSIVEWITHDIFFUSE = false;\n this.REFLECTIONFRESNELFROMSPECULAR = false;\n this.LIGHTMAP = false;\n this.LIGHTMAPDIRECTUV = 0;\n this.OBJECTSPACE_NORMALMAP = false;\n this.USELIGHTMAPASSHADOWMAP = false;\n this.REFLECTIONMAP_3D = false;\n this.REFLECTIONMAP_SPHERICAL = false;\n this.REFLECTIONMAP_PLANAR = false;\n this.REFLECTIONMAP_CUBIC = false;\n this.USE_LOCAL_REFLECTIONMAP_CUBIC = false;\n this.USE_LOCAL_REFRACTIONMAP_CUBIC = false;\n this.REFLECTIONMAP_PROJECTION = false;\n this.REFLECTIONMAP_SKYBOX = false;\n this.REFLECTIONMAP_EXPLICIT = false;\n this.REFLECTIONMAP_EQUIRECTANGULAR = false;\n this.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = false;\n this.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = false;\n this.REFLECTIONMAP_OPPOSITEZ = false;\n this.INVERTCUBICMAP = false;\n this.LOGARITHMICDEPTH = false;\n this.REFRACTION = false;\n this.REFRACTIONMAP_3D = false;\n this.REFLECTIONOVERALPHA = false;\n this.TWOSIDEDLIGHTING = false;\n this.SHADOWFLOAT = false;\n this.MORPHTARGETS = false;\n this.MORPHTARGETS_NORMAL = false;\n this.MORPHTARGETS_TANGENT = false;\n this.MORPHTARGETS_UV = false;\n this.NUM_MORPH_INFLUENCERS = 0;\n this.MORPHTARGETS_TEXTURE = false;\n this.NONUNIFORMSCALING = false; // https://playground.babylonjs.com#V6DWIH\n this.PREMULTIPLYALPHA = false; // https://playground.babylonjs.com#LNVJJ7\n this.ALPHATEST_AFTERALLALPHACOMPUTATIONS = false;\n this.ALPHABLEND = true;\n this.PREPASS = false;\n this.PREPASS_IRRADIANCE = false;\n this.PREPASS_IRRADIANCE_INDEX = -1;\n this.PREPASS_ALBEDO_SQRT = false;\n this.PREPASS_ALBEDO_SQRT_INDEX = -1;\n this.PREPASS_DEPTH = false;\n this.PREPASS_DEPTH_INDEX = -1;\n this.PREPASS_NORMAL = false;\n this.PREPASS_NORMAL_INDEX = -1;\n this.PREPASS_POSITION = false;\n this.PREPASS_POSITION_INDEX = -1;\n this.PREPASS_VELOCITY = false;\n this.PREPASS_VELOCITY_INDEX = -1;\n this.PREPASS_REFLECTIVITY = false;\n this.PREPASS_REFLECTIVITY_INDEX = -1;\n this.SCENE_MRT_COUNT = 0;\n this.RGBDLIGHTMAP = false;\n this.RGBDREFLECTION = false;\n this.RGBDREFRACTION = false;\n this.IMAGEPROCESSING = false;\n this.VIGNETTE = false;\n this.VIGNETTEBLENDMODEMULTIPLY = false;\n this.VIGNETTEBLENDMODEOPAQUE = false;\n this.TONEMAPPING = false;\n this.TONEMAPPING_ACES = false;\n this.CONTRAST = false;\n this.COLORCURVES = false;\n this.COLORGRADING = false;\n this.COLORGRADING3D = false;\n this.SAMPLER3DGREENDEPTH = false;\n this.SAMPLER3DBGRMAP = false;\n this.DITHER = false;\n this.IMAGEPROCESSINGPOSTPROCESS = false;\n this.SKIPFINALCOLORCLAMP = false;\n this.MULTIVIEW = false;\n this.ORDER_INDEPENDENT_TRANSPARENCY = false;\n this.ORDER_INDEPENDENT_TRANSPARENCY_16BITS = false;\n this.CAMERA_ORTHOGRAPHIC = false;\n this.CAMERA_PERSPECTIVE = false;\n /**\n * If the reflection texture on this material is in linear color space\n * @internal\n */\n this.IS_REFLECTION_LINEAR = false;\n /**\n * If the refraction texture on this material is in linear color space\n * @internal\n */\n this.IS_REFRACTION_LINEAR = false;\n this.EXPOSURE = false;\n this.DECAL_AFTER_DETAIL = false;\n this.rebuild();\n }\n setReflectionMode(modeToEnable) {\n const modes = [\n \"REFLECTIONMAP_CUBIC\",\n \"REFLECTIONMAP_EXPLICIT\",\n \"REFLECTIONMAP_PLANAR\",\n \"REFLECTIONMAP_PROJECTION\",\n \"REFLECTIONMAP_PROJECTION\",\n \"REFLECTIONMAP_SKYBOX\",\n \"REFLECTIONMAP_SPHERICAL\",\n \"REFLECTIONMAP_EQUIRECTANGULAR\",\n \"REFLECTIONMAP_EQUIRECTANGULAR_FIXED\",\n \"REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED\",\n ];\n for (const mode of modes) {\n this[mode] = mode === modeToEnable;\n }\n }\n }\n /**\n * This is the default material used in Babylon. It is the best trade off between quality\n * and performances.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/materials_introduction\n */\n class StandardMaterial extends PushMaterial {\n /**\n * Gets the image processing configuration used either in this material.\n */\n get imageProcessingConfiguration() {\n return this._imageProcessingConfiguration;\n }\n /**\n * Sets the Default image processing configuration used either in the this material.\n *\n * If sets to null, the scene one is in use.\n */\n set imageProcessingConfiguration(value) {\n this._attachImageProcessingConfiguration(value);\n // Ensure the effect will be rebuilt.\n this._markAllSubMeshesAsTexturesDirty();\n }\n /**\n * Attaches a new image processing configuration to the Standard Material.\n * @param configuration\n */\n _attachImageProcessingConfiguration(configuration) {\n if (configuration === this._imageProcessingConfiguration) {\n return;\n }\n // Detaches observer\n if (this._imageProcessingConfiguration && this._imageProcessingObserver) {\n this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver);\n }\n // Pick the scene configuration if needed\n if (!configuration) {\n this._imageProcessingConfiguration = this.getScene().imageProcessingConfiguration;\n }\n else {\n this._imageProcessingConfiguration = configuration;\n }\n // Attaches observer\n if (this._imageProcessingConfiguration) {\n this._imageProcessingObserver = this._imageProcessingConfiguration.onUpdateParameters.add(() => {\n this._markAllSubMeshesAsImageProcessingDirty();\n });\n }\n }\n /**\n * Can this material render to prepass\n */\n get isPrePassCapable() {\n return !this.disableDepthWrite;\n }\n /**\n * Gets whether the color curves effect is enabled.\n */\n get cameraColorCurvesEnabled() {\n return this.imageProcessingConfiguration.colorCurvesEnabled;\n }\n /**\n * Sets whether the color curves effect is enabled.\n */\n set cameraColorCurvesEnabled(value) {\n this.imageProcessingConfiguration.colorCurvesEnabled = value;\n }\n /**\n * Gets whether the color grading effect is enabled.\n */\n get cameraColorGradingEnabled() {\n return this.imageProcessingConfiguration.colorGradingEnabled;\n }\n /**\n * Gets whether the color grading effect is enabled.\n */\n set cameraColorGradingEnabled(value) {\n this.imageProcessingConfiguration.colorGradingEnabled = value;\n }\n /**\n * Gets whether tonemapping is enabled or not.\n */\n get cameraToneMappingEnabled() {\n return this._imageProcessingConfiguration.toneMappingEnabled;\n }\n /**\n * Sets whether tonemapping is enabled or not\n */\n set cameraToneMappingEnabled(value) {\n this._imageProcessingConfiguration.toneMappingEnabled = value;\n }\n /**\n * The camera exposure used on this material.\n * This property is here and not in the camera to allow controlling exposure without full screen post process.\n * This corresponds to a photographic exposure.\n */\n get cameraExposure() {\n return this._imageProcessingConfiguration.exposure;\n }\n /**\n * The camera exposure used on this material.\n * This property is here and not in the camera to allow controlling exposure without full screen post process.\n * This corresponds to a photographic exposure.\n */\n set cameraExposure(value) {\n this._imageProcessingConfiguration.exposure = value;\n }\n /**\n * Gets The camera contrast used on this material.\n */\n get cameraContrast() {\n return this._imageProcessingConfiguration.contrast;\n }\n /**\n * Sets The camera contrast used on this material.\n */\n set cameraContrast(value) {\n this._imageProcessingConfiguration.contrast = value;\n }\n /**\n * Gets the Color Grading 2D Lookup Texture.\n */\n get cameraColorGradingTexture() {\n return this._imageProcessingConfiguration.colorGradingTexture;\n }\n /**\n * Sets the Color Grading 2D Lookup Texture.\n */\n set cameraColorGradingTexture(value) {\n this._imageProcessingConfiguration.colorGradingTexture = value;\n }\n /**\n * The color grading curves provide additional color adjustmnent that is applied after any color grading transform (3D LUT).\n * They allow basic adjustment of saturation and small exposure adjustments, along with color filter tinting to provide white balance adjustment or more stylistic effects.\n * These are similar to controls found in many professional imaging or colorist software. The global controls are applied to the entire image. For advanced tuning, extra controls are provided to adjust the shadow, midtone and highlight areas of the image;\n * corresponding to low luminance, medium luminance, and high luminance areas respectively.\n */\n get cameraColorCurves() {\n return this._imageProcessingConfiguration.colorCurves;\n }\n /**\n * The color grading curves provide additional color adjustment that is applied after any color grading transform (3D LUT).\n * They allow basic adjustment of saturation and small exposure adjustments, along with color filter tinting to provide white balance adjustment or more stylistic effects.\n * These are similar to controls found in many professional imaging or colorist software. The global controls are applied to the entire image. For advanced tuning, extra controls are provided to adjust the shadow, midtone and highlight areas of the image;\n * corresponding to low luminance, medium luminance, and high luminance areas respectively.\n */\n set cameraColorCurves(value) {\n this._imageProcessingConfiguration.colorCurves = value;\n }\n /**\n * Can this material render to several textures at once\n */\n get canRenderToMRT() {\n return true;\n }\n /**\n * Instantiates a new standard material.\n * This is the default material used in Babylon. It is the best trade off between quality\n * and performances.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/materials_introduction\n * @param name Define the name of the material in the scene\n * @param scene Define the scene the material belong to\n */\n constructor(name, scene) {\n super(name, scene);\n this._diffuseTexture = null;\n this._ambientTexture = null;\n this._opacityTexture = null;\n this._reflectionTexture = null;\n this._emissiveTexture = null;\n this._specularTexture = null;\n this._bumpTexture = null;\n this._lightmapTexture = null;\n this._refractionTexture = null;\n /**\n * The color of the material lit by the environmental background lighting.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/materials_introduction#ambient-color-example\n */\n this.ambientColor = new Color3(0, 0, 0);\n /**\n * The basic color of the material as viewed under a light.\n */\n this.diffuseColor = new Color3(1, 1, 1);\n /**\n * Define how the color and intensity of the highlight given by the light in the material.\n */\n this.specularColor = new Color3(1, 1, 1);\n /**\n * Define the color of the material as if self lit.\n * This will be mixed in the final result even in the absence of light.\n */\n this.emissiveColor = new Color3(0, 0, 0);\n /**\n * Defines how sharp are the highlights in the material.\n * The bigger the value the sharper giving a more glossy feeling to the result.\n * Reversely, the smaller the value the blurrier giving a more rough feeling to the result.\n */\n this.specularPower = 64;\n this._useAlphaFromDiffuseTexture = false;\n this._useEmissiveAsIllumination = false;\n this._linkEmissiveWithDiffuse = false;\n this._useSpecularOverAlpha = false;\n this._useReflectionOverAlpha = false;\n this._disableLighting = false;\n this._useObjectSpaceNormalMap = false;\n this._useParallax = false;\n this._useParallaxOcclusion = false;\n /**\n * Apply a scaling factor that determine which \"depth\" the height map should reprensent. A value between 0.05 and 0.1 is reasonnable in Parallax, you can reach 0.2 using Parallax Occlusion.\n */\n this.parallaxScaleBias = 0.05;\n this._roughness = 0;\n /**\n * In case of refraction, define the value of the index of refraction.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#how-to-obtain-reflections-and-refractions\n */\n this.indexOfRefraction = 0.98;\n /**\n * Invert the refraction texture alongside the y axis.\n * It can be useful with procedural textures or probe for instance.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#how-to-obtain-reflections-and-refractions\n */\n this.invertRefractionY = true;\n /**\n * Defines the alpha limits in alpha test mode.\n */\n this.alphaCutOff = 0.4;\n this._useLightmapAsShadowmap = false;\n this._useReflectionFresnelFromSpecular = false;\n this._useGlossinessFromSpecularMapAlpha = false;\n this._maxSimultaneousLights = 4;\n this._invertNormalMapX = false;\n this._invertNormalMapY = false;\n this._twoSidedLighting = false;\n this._applyDecalMapAfterDetailMap = false;\n this._renderTargets = new SmartArray(16);\n this._worldViewProjectionMatrix = Matrix.Zero();\n this._globalAmbientColor = new Color3(0, 0, 0);\n this._cacheHasRenderTargetTextures = false;\n this.detailMap = new DetailMapConfiguration(this);\n // Setup the default processing configuration to the scene.\n this._attachImageProcessingConfiguration(null);\n this.prePassConfiguration = new PrePassConfiguration();\n this.getRenderTargetTextures = () => {\n this._renderTargets.reset();\n if (StandardMaterial.ReflectionTextureEnabled && this._reflectionTexture && this._reflectionTexture.isRenderTarget) {\n this._renderTargets.push(this._reflectionTexture);\n }\n if (StandardMaterial.RefractionTextureEnabled && this._refractionTexture && this._refractionTexture.isRenderTarget) {\n this._renderTargets.push(this._refractionTexture);\n }\n this._eventInfo.renderTargets = this._renderTargets;\n this._callbackPluginEventFillRenderTargetTextures(this._eventInfo);\n return this._renderTargets;\n };\n }\n /**\n * Gets a boolean indicating that current material needs to register RTT\n */\n get hasRenderTargetTextures() {\n if (StandardMaterial.ReflectionTextureEnabled && this._reflectionTexture && this._reflectionTexture.isRenderTarget) {\n return true;\n }\n if (StandardMaterial.RefractionTextureEnabled && this._refractionTexture && this._refractionTexture.isRenderTarget) {\n return true;\n }\n return this._cacheHasRenderTargetTextures;\n }\n /**\n * Gets the current class name of the material e.g. \"StandardMaterial\"\n * Mainly use in serialization.\n * @returns the class name\n */\n getClassName() {\n return \"StandardMaterial\";\n }\n /**\n * In case the depth buffer does not allow enough depth precision for your scene (might be the case in large scenes)\n * You can try switching to logarithmic depth.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/advanced/logarithmicDepthBuffer\n */\n get useLogarithmicDepth() {\n return this._useLogarithmicDepth;\n }\n set useLogarithmicDepth(value) {\n this._useLogarithmicDepth = value && this.getScene().getEngine().getCaps().fragmentDepthSupported;\n this._markAllSubMeshesAsMiscDirty();\n }\n /**\n * Specifies if the material will require alpha blending\n * @returns a boolean specifying if alpha blending is needed\n */\n needAlphaBlending() {\n if (this._disableAlphaBlending) {\n return false;\n }\n return (this.alpha < 1.0 ||\n this._opacityTexture != null ||\n this._shouldUseAlphaFromDiffuseTexture() ||\n (this._opacityFresnelParameters && this._opacityFresnelParameters.isEnabled));\n }\n /**\n * Specifies if this material should be rendered in alpha test mode\n * @returns a boolean specifying if an alpha test is needed.\n */\n needAlphaTesting() {\n if (this._forceAlphaTest) {\n return true;\n }\n return this._hasAlphaChannel() && (this._transparencyMode == null || this._transparencyMode === Material.MATERIAL_ALPHATEST);\n }\n /**\n * Specifies whether or not the alpha value of the diffuse texture should be used for alpha blending.\n */\n _shouldUseAlphaFromDiffuseTexture() {\n return this._diffuseTexture != null && this._diffuseTexture.hasAlpha && this._useAlphaFromDiffuseTexture && this._transparencyMode !== Material.MATERIAL_OPAQUE;\n }\n /**\n * Specifies whether or not there is a usable alpha channel for transparency.\n */\n _hasAlphaChannel() {\n return (this._diffuseTexture != null && this._diffuseTexture.hasAlpha) || this._opacityTexture != null;\n }\n /**\n * Get the texture used for alpha test purpose.\n * @returns the diffuse texture in case of the standard material.\n */\n getAlphaTestTexture() {\n return this._diffuseTexture;\n }\n /**\n * Get if the submesh is ready to be used and all its information available.\n * Child classes can use it to update shaders\n * @param mesh defines the mesh to check\n * @param subMesh defines which submesh to check\n * @param useInstances specifies that instances should be used\n * @returns a boolean indicating that the submesh is ready or not\n */\n isReadyForSubMesh(mesh, subMesh, useInstances = false) {\n if (!this._uniformBufferLayoutBuilt) {\n this.buildUniformLayout();\n }\n if (subMesh.effect && this.isFrozen) {\n if (subMesh.effect._wasPreviouslyReady && subMesh.effect._wasPreviouslyUsingInstances === useInstances) {\n return true;\n }\n }\n if (!subMesh.materialDefines) {\n this._callbackPluginEventGeneric(MaterialPluginEvent.GetDefineNames, this._eventInfo);\n subMesh.materialDefines = new StandardMaterialDefines(this._eventInfo.defineNames);\n }\n const scene = this.getScene();\n const defines = subMesh.materialDefines;\n if (this._isReadyForSubMesh(subMesh)) {\n return true;\n }\n const engine = scene.getEngine();\n // Lights\n defines._needNormals = MaterialHelper.PrepareDefinesForLights(scene, mesh, defines, true, this._maxSimultaneousLights, this._disableLighting);\n // Multiview\n MaterialHelper.PrepareDefinesForMultiview(scene, defines);\n // PrePass\n const oit = this.needAlphaBlendingForMesh(mesh) && this.getScene().useOrderIndependentTransparency;\n MaterialHelper.PrepareDefinesForPrePass(scene, defines, this.canRenderToMRT && !oit);\n // Order independant transparency\n MaterialHelper.PrepareDefinesForOIT(scene, defines, oit);\n // Textures\n if (defines._areTexturesDirty) {\n this._eventInfo.hasRenderTargetTextures = false;\n this._callbackPluginEventHasRenderTargetTextures(this._eventInfo);\n this._cacheHasRenderTargetTextures = this._eventInfo.hasRenderTargetTextures;\n defines._needUVs = false;\n for (let i = 1; i <= 6; ++i) {\n defines[\"MAINUV\" + i] = false;\n }\n if (scene.texturesEnabled) {\n defines.DIFFUSEDIRECTUV = 0;\n defines.BUMPDIRECTUV = 0;\n defines.AMBIENTDIRECTUV = 0;\n defines.OPACITYDIRECTUV = 0;\n defines.EMISSIVEDIRECTUV = 0;\n defines.SPECULARDIRECTUV = 0;\n defines.LIGHTMAPDIRECTUV = 0;\n if (this._diffuseTexture && StandardMaterial.DiffuseTextureEnabled) {\n if (!this._diffuseTexture.isReadyOrNotBlocking()) {\n return false;\n }\n else {\n MaterialHelper.PrepareDefinesForMergedUV(this._diffuseTexture, defines, \"DIFFUSE\");\n }\n }\n else {\n defines.DIFFUSE = false;\n }\n if (this._ambientTexture && StandardMaterial.AmbientTextureEnabled) {\n if (!this._ambientTexture.isReadyOrNotBlocking()) {\n return false;\n }\n else {\n MaterialHelper.PrepareDefinesForMergedUV(this._ambientTexture, defines, \"AMBIENT\");\n }\n }\n else {\n defines.AMBIENT = false;\n }\n if (this._opacityTexture && StandardMaterial.OpacityTextureEnabled) {\n if (!this._opacityTexture.isReadyOrNotBlocking()) {\n return false;\n }\n else {\n MaterialHelper.PrepareDefinesForMergedUV(this._opacityTexture, defines, \"OPACITY\");\n defines.OPACITYRGB = this._opacityTexture.getAlphaFromRGB;\n }\n }\n else {\n defines.OPACITY = false;\n }\n if (this._reflectionTexture && StandardMaterial.ReflectionTextureEnabled) {\n if (!this._reflectionTexture.isReadyOrNotBlocking()) {\n return false;\n }\n else {\n defines._needNormals = true;\n defines.REFLECTION = true;\n defines.ROUGHNESS = this._roughness > 0;\n defines.REFLECTIONOVERALPHA = this._useReflectionOverAlpha;\n defines.INVERTCUBICMAP = this._reflectionTexture.coordinatesMode === Texture.INVCUBIC_MODE;\n defines.REFLECTIONMAP_3D = this._reflectionTexture.isCube;\n defines.REFLECTIONMAP_OPPOSITEZ =\n defines.REFLECTIONMAP_3D && this.getScene().useRightHandedSystem ? !this._reflectionTexture.invertZ : this._reflectionTexture.invertZ;\n defines.RGBDREFLECTION = this._reflectionTexture.isRGBD;\n switch (this._reflectionTexture.coordinatesMode) {\n case Texture.EXPLICIT_MODE:\n defines.setReflectionMode(\"REFLECTIONMAP_EXPLICIT\");\n break;\n case Texture.PLANAR_MODE:\n defines.setReflectionMode(\"REFLECTIONMAP_PLANAR\");\n break;\n case Texture.PROJECTION_MODE:\n defines.setReflectionMode(\"REFLECTIONMAP_PROJECTION\");\n break;\n case Texture.SKYBOX_MODE:\n defines.setReflectionMode(\"REFLECTIONMAP_SKYBOX\");\n break;\n case Texture.SPHERICAL_MODE:\n defines.setReflectionMode(\"REFLECTIONMAP_SPHERICAL\");\n break;\n case Texture.EQUIRECTANGULAR_MODE:\n defines.setReflectionMode(\"REFLECTIONMAP_EQUIRECTANGULAR\");\n break;\n case Texture.FIXED_EQUIRECTANGULAR_MODE:\n defines.setReflectionMode(\"REFLECTIONMAP_EQUIRECTANGULAR_FIXED\");\n break;\n case Texture.FIXED_EQUIRECTANGULAR_MIRRORED_MODE:\n defines.setReflectionMode(\"REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED\");\n break;\n case Texture.CUBIC_MODE:\n case Texture.INVCUBIC_MODE:\n default:\n defines.setReflectionMode(\"REFLECTIONMAP_CUBIC\");\n break;\n }\n defines.USE_LOCAL_REFLECTIONMAP_CUBIC = this._reflectionTexture.boundingBoxSize ? true : false;\n }\n }\n else {\n defines.REFLECTION = false;\n defines.REFLECTIONMAP_OPPOSITEZ = false;\n }\n if (this._emissiveTexture && StandardMaterial.EmissiveTextureEnabled) {\n if (!this._emissiveTexture.isReadyOrNotBlocking()) {\n return false;\n }\n else {\n MaterialHelper.PrepareDefinesForMergedUV(this._emissiveTexture, defines, \"EMISSIVE\");\n }\n }\n else {\n defines.EMISSIVE = false;\n }\n if (this._lightmapTexture && StandardMaterial.LightmapTextureEnabled) {\n if (!this._lightmapTexture.isReadyOrNotBlocking()) {\n return false;\n }\n else {\n MaterialHelper.PrepareDefinesForMergedUV(this._lightmapTexture, defines, \"LIGHTMAP\");\n defines.USELIGHTMAPASSHADOWMAP = this._useLightmapAsShadowmap;\n defines.RGBDLIGHTMAP = this._lightmapTexture.isRGBD;\n }\n }\n else {\n defines.LIGHTMAP = false;\n }\n if (this._specularTexture && StandardMaterial.SpecularTextureEnabled) {\n if (!this._specularTexture.isReadyOrNotBlocking()) {\n return false;\n }\n else {\n MaterialHelper.PrepareDefinesForMergedUV(this._specularTexture, defines, \"SPECULAR\");\n defines.GLOSSINESS = this._useGlossinessFromSpecularMapAlpha;\n }\n }\n else {\n defines.SPECULAR = false;\n }\n if (scene.getEngine().getCaps().standardDerivatives && this._bumpTexture && StandardMaterial.BumpTextureEnabled) {\n // Bump texture can not be not blocking.\n if (!this._bumpTexture.isReady()) {\n return false;\n }\n else {\n MaterialHelper.PrepareDefinesForMergedUV(this._bumpTexture, defines, \"BUMP\");\n defines.PARALLAX = this._useParallax;\n defines.PARALLAXOCCLUSION = this._useParallaxOcclusion;\n }\n defines.OBJECTSPACE_NORMALMAP = this._useObjectSpaceNormalMap;\n }\n else {\n defines.BUMP = false;\n defines.PARALLAX = false;\n defines.PARALLAXOCCLUSION = false;\n }\n if (this._refractionTexture && StandardMaterial.RefractionTextureEnabled) {\n if (!this._refractionTexture.isReadyOrNotBlocking()) {\n return false;\n }\n else {\n defines._needUVs = true;\n defines.REFRACTION = true;\n defines.REFRACTIONMAP_3D = this._refractionTexture.isCube;\n defines.RGBDREFRACTION = this._refractionTexture.isRGBD;\n defines.USE_LOCAL_REFRACTIONMAP_CUBIC = this._refractionTexture.boundingBoxSize ? true : false;\n }\n }\n else {\n defines.REFRACTION = false;\n }\n defines.TWOSIDEDLIGHTING = !this._backFaceCulling && this._twoSidedLighting;\n }\n else {\n defines.DIFFUSE = false;\n defines.AMBIENT = false;\n defines.OPACITY = false;\n defines.REFLECTION = false;\n defines.EMISSIVE = false;\n defines.LIGHTMAP = false;\n defines.BUMP = false;\n defines.REFRACTION = false;\n }\n defines.ALPHAFROMDIFFUSE = this._shouldUseAlphaFromDiffuseTexture();\n defines.EMISSIVEASILLUMINATION = this._useEmissiveAsIllumination;\n defines.LINKEMISSIVEWITHDIFFUSE = this._linkEmissiveWithDiffuse;\n defines.SPECULAROVERALPHA = this._useSpecularOverAlpha;\n defines.PREMULTIPLYALPHA = this.alphaMode === 7 || this.alphaMode === 8;\n defines.ALPHATEST_AFTERALLALPHACOMPUTATIONS = this.transparencyMode !== null;\n defines.ALPHABLEND = this.transparencyMode === null || this.needAlphaBlendingForMesh(mesh); // check on null for backward compatibility\n }\n this._eventInfo.isReadyForSubMesh = true;\n this._eventInfo.defines = defines;\n this._eventInfo.subMesh = subMesh;\n this._callbackPluginEventIsReadyForSubMesh(this._eventInfo);\n if (!this._eventInfo.isReadyForSubMesh) {\n return false;\n }\n if (defines._areImageProcessingDirty && this._imageProcessingConfiguration) {\n if (!this._imageProcessingConfiguration.isReady()) {\n return false;\n }\n this._imageProcessingConfiguration.prepareDefines(defines);\n defines.IS_REFLECTION_LINEAR = this.reflectionTexture != null && !this.reflectionTexture.gammaSpace;\n defines.IS_REFRACTION_LINEAR = this.refractionTexture != null && !this.refractionTexture.gammaSpace;\n }\n if (defines._areFresnelDirty) {\n if (StandardMaterial.FresnelEnabled) {\n // Fresnel\n if (this._diffuseFresnelParameters ||\n this._opacityFresnelParameters ||\n this._emissiveFresnelParameters ||\n this._refractionFresnelParameters ||\n this._reflectionFresnelParameters) {\n defines.DIFFUSEFRESNEL = this._diffuseFresnelParameters && this._diffuseFresnelParameters.isEnabled;\n defines.OPACITYFRESNEL = this._opacityFresnelParameters && this._opacityFresnelParameters.isEnabled;\n defines.REFLECTIONFRESNEL = this._reflectionFresnelParameters && this._reflectionFresnelParameters.isEnabled;\n defines.REFLECTIONFRESNELFROMSPECULAR = this._useReflectionFresnelFromSpecular;\n defines.REFRACTIONFRESNEL = this._refractionFresnelParameters && this._refractionFresnelParameters.isEnabled;\n defines.EMISSIVEFRESNEL = this._emissiveFresnelParameters && this._emissiveFresnelParameters.isEnabled;\n defines._needNormals = true;\n defines.FRESNEL = true;\n }\n }\n else {\n defines.FRESNEL = false;\n }\n }\n // Misc.\n MaterialHelper.PrepareDefinesForMisc(mesh, scene, this._useLogarithmicDepth, this.pointsCloud, this.fogEnabled, this._shouldTurnAlphaTestOn(mesh) || this._forceAlphaTest, defines, this._applyDecalMapAfterDetailMap);\n // Values that need to be evaluated on every frame\n MaterialHelper.PrepareDefinesForFrameBoundValues(scene, engine, this, defines, useInstances, null, subMesh.getRenderingMesh().hasThinInstances);\n // External config\n this._eventInfo.defines = defines;\n this._eventInfo.mesh = mesh;\n this._callbackPluginEventPrepareDefinesBeforeAttributes(this._eventInfo);\n // Attribs\n MaterialHelper.PrepareDefinesForAttributes(mesh, defines, true, true, true);\n // External config\n this._callbackPluginEventPrepareDefines(this._eventInfo);\n // Get correct effect\n let forceWasNotReadyPreviously = false;\n if (defines.isDirty) {\n const lightDisposed = defines._areLightsDisposed;\n defines.markAsProcessed();\n // Fallbacks\n const fallbacks = new EffectFallbacks();\n if (defines.REFLECTION) {\n fallbacks.addFallback(0, \"REFLECTION\");\n }\n if (defines.SPECULAR) {\n fallbacks.addFallback(0, \"SPECULAR\");\n }\n if (defines.BUMP) {\n fallbacks.addFallback(0, \"BUMP\");\n }\n if (defines.PARALLAX) {\n fallbacks.addFallback(1, \"PARALLAX\");\n }\n if (defines.PARALLAXOCCLUSION) {\n fallbacks.addFallback(0, \"PARALLAXOCCLUSION\");\n }\n if (defines.SPECULAROVERALPHA) {\n fallbacks.addFallback(0, \"SPECULAROVERALPHA\");\n }\n if (defines.FOG) {\n fallbacks.addFallback(1, \"FOG\");\n }\n if (defines.POINTSIZE) {\n fallbacks.addFallback(0, \"POINTSIZE\");\n }\n if (defines.LOGARITHMICDEPTH) {\n fallbacks.addFallback(0, \"LOGARITHMICDEPTH\");\n }\n MaterialHelper.HandleFallbacksForShadows(defines, fallbacks, this._maxSimultaneousLights);\n if (defines.SPECULARTERM) {\n fallbacks.addFallback(0, \"SPECULARTERM\");\n }\n if (defines.DIFFUSEFRESNEL) {\n fallbacks.addFallback(1, \"DIFFUSEFRESNEL\");\n }\n if (defines.OPACITYFRESNEL) {\n fallbacks.addFallback(2, \"OPACITYFRESNEL\");\n }\n if (defines.REFLECTIONFRESNEL) {\n fallbacks.addFallback(3, \"REFLECTIONFRESNEL\");\n }\n if (defines.EMISSIVEFRESNEL) {\n fallbacks.addFallback(4, \"EMISSIVEFRESNEL\");\n }\n if (defines.FRESNEL) {\n fallbacks.addFallback(4, \"FRESNEL\");\n }\n if (defines.MULTIVIEW) {\n fallbacks.addFallback(0, \"MULTIVIEW\");\n }\n //Attributes\n const attribs = [VertexBuffer.PositionKind];\n if (defines.NORMAL) {\n attribs.push(VertexBuffer.NormalKind);\n }\n if (defines.TANGENT) {\n attribs.push(VertexBuffer.TangentKind);\n }\n for (let i = 1; i <= 6; ++i) {\n if (defines[\"UV\" + i]) {\n attribs.push(`uv${i === 1 ? \"\" : i}`);\n }\n }\n if (defines.VERTEXCOLOR) {\n attribs.push(VertexBuffer.ColorKind);\n }\n MaterialHelper.PrepareAttributesForBones(attribs, mesh, defines, fallbacks);\n MaterialHelper.PrepareAttributesForInstances(attribs, defines);\n MaterialHelper.PrepareAttributesForMorphTargets(attribs, mesh, defines);\n MaterialHelper.PrepareAttributesForBakedVertexAnimation(attribs, mesh, defines);\n let shaderName = \"default\";\n const uniforms = [\n \"world\",\n \"view\",\n \"viewProjection\",\n \"vEyePosition\",\n \"vLightsType\",\n \"vAmbientColor\",\n \"vDiffuseColor\",\n \"vSpecularColor\",\n \"vEmissiveColor\",\n \"visibility\",\n \"vFogInfos\",\n \"vFogColor\",\n \"pointSize\",\n \"vDiffuseInfos\",\n \"vAmbientInfos\",\n \"vOpacityInfos\",\n \"vReflectionInfos\",\n \"vEmissiveInfos\",\n \"vSpecularInfos\",\n \"vBumpInfos\",\n \"vLightmapInfos\",\n \"vRefractionInfos\",\n \"mBones\",\n \"diffuseMatrix\",\n \"ambientMatrix\",\n \"opacityMatrix\",\n \"reflectionMatrix\",\n \"emissiveMatrix\",\n \"specularMatrix\",\n \"bumpMatrix\",\n \"normalMatrix\",\n \"lightmapMatrix\",\n \"refractionMatrix\",\n \"diffuseLeftColor\",\n \"diffuseRightColor\",\n \"opacityParts\",\n \"reflectionLeftColor\",\n \"reflectionRightColor\",\n \"emissiveLeftColor\",\n \"emissiveRightColor\",\n \"refractionLeftColor\",\n \"refractionRightColor\",\n \"vReflectionPosition\",\n \"vReflectionSize\",\n \"vRefractionPosition\",\n \"vRefractionSize\",\n \"logarithmicDepthConstant\",\n \"vTangentSpaceParams\",\n \"alphaCutOff\",\n \"boneTextureWidth\",\n \"morphTargetTextureInfo\",\n \"morphTargetTextureIndices\",\n ];\n const samplers = [\n \"diffuseSampler\",\n \"ambientSampler\",\n \"opacitySampler\",\n \"reflectionCubeSampler\",\n \"reflection2DSampler\",\n \"emissiveSampler\",\n \"specularSampler\",\n \"bumpSampler\",\n \"lightmapSampler\",\n \"refractionCubeSampler\",\n \"refraction2DSampler\",\n \"boneSampler\",\n \"morphTargets\",\n \"oitDepthSampler\",\n \"oitFrontColorSampler\",\n ];\n const uniformBuffers = [\"Material\", \"Scene\", \"Mesh\"];\n const indexParameters = { maxSimultaneousLights: this._maxSimultaneousLights, maxSimultaneousMorphTargets: defines.NUM_MORPH_INFLUENCERS };\n this._eventInfo.fallbacks = fallbacks;\n this._eventInfo.fallbackRank = 0;\n this._eventInfo.defines = defines;\n this._eventInfo.uniforms = uniforms;\n this._eventInfo.attributes = attribs;\n this._eventInfo.samplers = samplers;\n this._eventInfo.uniformBuffersNames = uniformBuffers;\n this._eventInfo.customCode = undefined;\n this._eventInfo.mesh = mesh;\n this._eventInfo.indexParameters = indexParameters;\n this._callbackPluginEventGeneric(MaterialPluginEvent.PrepareEffect, this._eventInfo);\n PrePassConfiguration.AddUniforms(uniforms);\n PrePassConfiguration.AddSamplers(samplers);\n if (ImageProcessingConfiguration) {\n ImageProcessingConfiguration.PrepareUniforms(uniforms, defines);\n ImageProcessingConfiguration.PrepareSamplers(samplers, defines);\n }\n MaterialHelper.PrepareUniformsAndSamplersList({\n uniformsNames: uniforms,\n uniformBuffersNames: uniformBuffers,\n samplers: samplers,\n defines: defines,\n maxSimultaneousLights: this._maxSimultaneousLights,\n });\n addClipPlaneUniforms(uniforms);\n const csnrOptions = {};\n if (this.customShaderNameResolve) {\n shaderName = this.customShaderNameResolve(shaderName, uniforms, uniformBuffers, samplers, defines, attribs, csnrOptions);\n }\n const join = defines.toString();\n const previousEffect = subMesh.effect;\n let effect = scene.getEngine().createEffect(shaderName, {\n attributes: attribs,\n uniformsNames: uniforms,\n uniformBuffersNames: uniformBuffers,\n samplers: samplers,\n defines: join,\n fallbacks: fallbacks,\n onCompiled: this.onCompiled,\n onError: this.onError,\n indexParameters,\n processFinalCode: csnrOptions.processFinalCode,\n processCodeAfterIncludes: this._eventInfo.customCode,\n multiTarget: defines.PREPASS,\n }, engine);\n this._eventInfo.customCode = undefined;\n if (effect) {\n if (this._onEffectCreatedObservable) {\n onCreatedEffectParameters.effect = effect;\n onCreatedEffectParameters.subMesh = subMesh;\n this._onEffectCreatedObservable.notifyObservers(onCreatedEffectParameters);\n }\n // Use previous effect while new one is compiling\n if (this.allowShaderHotSwapping && previousEffect && !effect.isReady()) {\n effect = previousEffect;\n defines.markAsUnprocessed();\n forceWasNotReadyPreviously = this.isFrozen;\n if (lightDisposed) {\n // re register in case it takes more than one frame.\n defines._areLightsDisposed = true;\n return false;\n }\n }\n else {\n scene.resetCachedMaterial();\n subMesh.setEffect(effect, defines, this._materialContext);\n }\n }\n }\n if (!subMesh.effect || !subMesh.effect.isReady()) {\n return false;\n }\n defines._renderId = scene.getRenderId();\n subMesh.effect._wasPreviouslyReady = forceWasNotReadyPreviously ? false : true;\n subMesh.effect._wasPreviouslyUsingInstances = useInstances;\n this._checkScenePerformancePriority();\n return true;\n }\n /**\n * Builds the material UBO layouts.\n * Used internally during the effect preparation.\n */\n buildUniformLayout() {\n // Order is important !\n const ubo = this._uniformBuffer;\n ubo.addUniform(\"diffuseLeftColor\", 4);\n ubo.addUniform(\"diffuseRightColor\", 4);\n ubo.addUniform(\"opacityParts\", 4);\n ubo.addUniform(\"reflectionLeftColor\", 4);\n ubo.addUniform(\"reflectionRightColor\", 4);\n ubo.addUniform(\"refractionLeftColor\", 4);\n ubo.addUniform(\"refractionRightColor\", 4);\n ubo.addUniform(\"emissiveLeftColor\", 4);\n ubo.addUniform(\"emissiveRightColor\", 4);\n ubo.addUniform(\"vDiffuseInfos\", 2);\n ubo.addUniform(\"vAmbientInfos\", 2);\n ubo.addUniform(\"vOpacityInfos\", 2);\n ubo.addUniform(\"vReflectionInfos\", 2);\n ubo.addUniform(\"vReflectionPosition\", 3);\n ubo.addUniform(\"vReflectionSize\", 3);\n ubo.addUniform(\"vEmissiveInfos\", 2);\n ubo.addUniform(\"vLightmapInfos\", 2);\n ubo.addUniform(\"vSpecularInfos\", 2);\n ubo.addUniform(\"vBumpInfos\", 3);\n ubo.addUniform(\"diffuseMatrix\", 16);\n ubo.addUniform(\"ambientMatrix\", 16);\n ubo.addUniform(\"opacityMatrix\", 16);\n ubo.addUniform(\"reflectionMatrix\", 16);\n ubo.addUniform(\"emissiveMatrix\", 16);\n ubo.addUniform(\"lightmapMatrix\", 16);\n ubo.addUniform(\"specularMatrix\", 16);\n ubo.addUniform(\"bumpMatrix\", 16);\n ubo.addUniform(\"vTangentSpaceParams\", 2);\n ubo.addUniform(\"pointSize\", 1);\n ubo.addUniform(\"alphaCutOff\", 1);\n ubo.addUniform(\"refractionMatrix\", 16);\n ubo.addUniform(\"vRefractionInfos\", 4);\n ubo.addUniform(\"vRefractionPosition\", 3);\n ubo.addUniform(\"vRefractionSize\", 3);\n ubo.addUniform(\"vSpecularColor\", 4);\n ubo.addUniform(\"vEmissiveColor\", 3);\n ubo.addUniform(\"vDiffuseColor\", 4);\n ubo.addUniform(\"vAmbientColor\", 3);\n super.buildUniformLayout();\n }\n /**\n * Binds the submesh to this material by preparing the effect and shader to draw\n * @param world defines the world transformation matrix\n * @param mesh defines the mesh containing the submesh\n * @param subMesh defines the submesh to bind the material to\n */\n bindForSubMesh(world, mesh, subMesh) {\n var _a;\n const scene = this.getScene();\n const defines = subMesh.materialDefines;\n if (!defines) {\n return;\n }\n const effect = subMesh.effect;\n if (!effect) {\n return;\n }\n this._activeEffect = effect;\n // Matrices Mesh.\n mesh.getMeshUniformBuffer().bindToEffect(effect, \"Mesh\");\n mesh.transferToEffect(world);\n // Binding unconditionally\n this._uniformBuffer.bindToEffect(effect, \"Material\");\n this.prePassConfiguration.bindForSubMesh(this._activeEffect, scene, mesh, world, this.isFrozen);\n this._eventInfo.subMesh = subMesh;\n this._callbackPluginEventHardBindForSubMesh(this._eventInfo);\n // Normal Matrix\n if (defines.OBJECTSPACE_NORMALMAP) {\n world.toNormalMatrix(this._normalMatrix);\n this.bindOnlyNormalMatrix(this._normalMatrix);\n }\n const mustRebind = effect._forceRebindOnNextCall || this._mustRebind(scene, effect, mesh.visibility);\n // Bones\n MaterialHelper.BindBonesParameters(mesh, effect);\n const ubo = this._uniformBuffer;\n if (mustRebind) {\n this.bindViewProjection(effect);\n if (!ubo.useUbo || !this.isFrozen || !ubo.isSync || effect._forceRebindOnNextCall) {\n if (StandardMaterial.FresnelEnabled && defines.FRESNEL) {\n // Fresnel\n if (this.diffuseFresnelParameters && this.diffuseFresnelParameters.isEnabled) {\n ubo.updateColor4(\"diffuseLeftColor\", this.diffuseFresnelParameters.leftColor, this.diffuseFresnelParameters.power);\n ubo.updateColor4(\"diffuseRightColor\", this.diffuseFresnelParameters.rightColor, this.diffuseFresnelParameters.bias);\n }\n if (this.opacityFresnelParameters && this.opacityFresnelParameters.isEnabled) {\n ubo.updateColor4(\"opacityParts\", new Color3(this.opacityFresnelParameters.leftColor.toLuminance(), this.opacityFresnelParameters.rightColor.toLuminance(), this.opacityFresnelParameters.bias), this.opacityFresnelParameters.power);\n }\n if (this.reflectionFresnelParameters && this.reflectionFresnelParameters.isEnabled) {\n ubo.updateColor4(\"reflectionLeftColor\", this.reflectionFresnelParameters.leftColor, this.reflectionFresnelParameters.power);\n ubo.updateColor4(\"reflectionRightColor\", this.reflectionFresnelParameters.rightColor, this.reflectionFresnelParameters.bias);\n }\n if (this.refractionFresnelParameters && this.refractionFresnelParameters.isEnabled) {\n ubo.updateColor4(\"refractionLeftColor\", this.refractionFresnelParameters.leftColor, this.refractionFresnelParameters.power);\n ubo.updateColor4(\"refractionRightColor\", this.refractionFresnelParameters.rightColor, this.refractionFresnelParameters.bias);\n }\n if (this.emissiveFresnelParameters && this.emissiveFresnelParameters.isEnabled) {\n ubo.updateColor4(\"emissiveLeftColor\", this.emissiveFresnelParameters.leftColor, this.emissiveFresnelParameters.power);\n ubo.updateColor4(\"emissiveRightColor\", this.emissiveFresnelParameters.rightColor, this.emissiveFresnelParameters.bias);\n }\n }\n // Textures\n if (scene.texturesEnabled) {\n if (this._diffuseTexture && StandardMaterial.DiffuseTextureEnabled) {\n ubo.updateFloat2(\"vDiffuseInfos\", this._diffuseTexture.coordinatesIndex, this._diffuseTexture.level);\n MaterialHelper.BindTextureMatrix(this._diffuseTexture, ubo, \"diffuse\");\n }\n if (this._ambientTexture && StandardMaterial.AmbientTextureEnabled) {\n ubo.updateFloat2(\"vAmbientInfos\", this._ambientTexture.coordinatesIndex, this._ambientTexture.level);\n MaterialHelper.BindTextureMatrix(this._ambientTexture, ubo, \"ambient\");\n }\n if (this._opacityTexture && StandardMaterial.OpacityTextureEnabled) {\n ubo.updateFloat2(\"vOpacityInfos\", this._opacityTexture.coordinatesIndex, this._opacityTexture.level);\n MaterialHelper.BindTextureMatrix(this._opacityTexture, ubo, \"opacity\");\n }\n if (this._hasAlphaChannel()) {\n ubo.updateFloat(\"alphaCutOff\", this.alphaCutOff);\n }\n if (this._reflectionTexture && StandardMaterial.ReflectionTextureEnabled) {\n ubo.updateFloat2(\"vReflectionInfos\", this._reflectionTexture.level, this.roughness);\n ubo.updateMatrix(\"reflectionMatrix\", this._reflectionTexture.getReflectionTextureMatrix());\n if (this._reflectionTexture.boundingBoxSize) {\n const cubeTexture = this._reflectionTexture;\n ubo.updateVector3(\"vReflectionPosition\", cubeTexture.boundingBoxPosition);\n ubo.updateVector3(\"vReflectionSize\", cubeTexture.boundingBoxSize);\n }\n }\n if (this._emissiveTexture && StandardMaterial.EmissiveTextureEnabled) {\n ubo.updateFloat2(\"vEmissiveInfos\", this._emissiveTexture.coordinatesIndex, this._emissiveTexture.level);\n MaterialHelper.BindTextureMatrix(this._emissiveTexture, ubo, \"emissive\");\n }\n if (this._lightmapTexture && StandardMaterial.LightmapTextureEnabled) {\n ubo.updateFloat2(\"vLightmapInfos\", this._lightmapTexture.coordinatesIndex, this._lightmapTexture.level);\n MaterialHelper.BindTextureMatrix(this._lightmapTexture, ubo, \"lightmap\");\n }\n if (this._specularTexture && StandardMaterial.SpecularTextureEnabled) {\n ubo.updateFloat2(\"vSpecularInfos\", this._specularTexture.coordinatesIndex, this._specularTexture.level);\n MaterialHelper.BindTextureMatrix(this._specularTexture, ubo, \"specular\");\n }\n if (this._bumpTexture && scene.getEngine().getCaps().standardDerivatives && StandardMaterial.BumpTextureEnabled) {\n ubo.updateFloat3(\"vBumpInfos\", this._bumpTexture.coordinatesIndex, 1.0 / this._bumpTexture.level, this.parallaxScaleBias);\n MaterialHelper.BindTextureMatrix(this._bumpTexture, ubo, \"bump\");\n if (scene._mirroredCameraPosition) {\n ubo.updateFloat2(\"vTangentSpaceParams\", this._invertNormalMapX ? 1.0 : -1.0, this._invertNormalMapY ? 1.0 : -1.0);\n }\n else {\n ubo.updateFloat2(\"vTangentSpaceParams\", this._invertNormalMapX ? -1.0 : 1.0, this._invertNormalMapY ? -1.0 : 1.0);\n }\n }\n if (this._refractionTexture && StandardMaterial.RefractionTextureEnabled) {\n let depth = 1.0;\n if (!this._refractionTexture.isCube) {\n ubo.updateMatrix(\"refractionMatrix\", this._refractionTexture.getReflectionTextureMatrix());\n if (this._refractionTexture.depth) {\n depth = this._refractionTexture.depth;\n }\n }\n ubo.updateFloat4(\"vRefractionInfos\", this._refractionTexture.level, this.indexOfRefraction, depth, this.invertRefractionY ? -1 : 1);\n if (this._refractionTexture.boundingBoxSize) {\n const cubeTexture = this._refractionTexture;\n ubo.updateVector3(\"vRefractionPosition\", cubeTexture.boundingBoxPosition);\n ubo.updateVector3(\"vRefractionSize\", cubeTexture.boundingBoxSize);\n }\n }\n }\n // Point size\n if (this.pointsCloud) {\n ubo.updateFloat(\"pointSize\", this.pointSize);\n }\n if (defines.SPECULARTERM) {\n ubo.updateColor4(\"vSpecularColor\", this.specularColor, this.specularPower);\n }\n ubo.updateColor3(\"vEmissiveColor\", StandardMaterial.EmissiveTextureEnabled ? this.emissiveColor : Color3.BlackReadOnly);\n ubo.updateColor4(\"vDiffuseColor\", this.diffuseColor, this.alpha);\n scene.ambientColor.multiplyToRef(this.ambientColor, this._globalAmbientColor);\n ubo.updateColor3(\"vAmbientColor\", this._globalAmbientColor);\n }\n // Textures\n if (scene.texturesEnabled) {\n if (this._diffuseTexture && StandardMaterial.DiffuseTextureEnabled) {\n effect.setTexture(\"diffuseSampler\", this._diffuseTexture);\n }\n if (this._ambientTexture && StandardMaterial.AmbientTextureEnabled) {\n effect.setTexture(\"ambientSampler\", this._ambientTexture);\n }\n if (this._opacityTexture && StandardMaterial.OpacityTextureEnabled) {\n effect.setTexture(\"opacitySampler\", this._opacityTexture);\n }\n if (this._reflectionTexture && StandardMaterial.ReflectionTextureEnabled) {\n if (this._reflectionTexture.isCube) {\n effect.setTexture(\"reflectionCubeSampler\", this._reflectionTexture);\n }\n else {\n effect.setTexture(\"reflection2DSampler\", this._reflectionTexture);\n }\n }\n if (this._emissiveTexture && StandardMaterial.EmissiveTextureEnabled) {\n effect.setTexture(\"emissiveSampler\", this._emissiveTexture);\n }\n if (this._lightmapTexture && StandardMaterial.LightmapTextureEnabled) {\n effect.setTexture(\"lightmapSampler\", this._lightmapTexture);\n }\n if (this._specularTexture && StandardMaterial.SpecularTextureEnabled) {\n effect.setTexture(\"specularSampler\", this._specularTexture);\n }\n if (this._bumpTexture && scene.getEngine().getCaps().standardDerivatives && StandardMaterial.BumpTextureEnabled) {\n effect.setTexture(\"bumpSampler\", this._bumpTexture);\n }\n if (this._refractionTexture && StandardMaterial.RefractionTextureEnabled) {\n if (this._refractionTexture.isCube) {\n effect.setTexture(\"refractionCubeSampler\", this._refractionTexture);\n }\n else {\n effect.setTexture(\"refraction2DSampler\", this._refractionTexture);\n }\n }\n }\n // OIT with depth peeling\n if (this.getScene().useOrderIndependentTransparency && this.needAlphaBlendingForMesh(mesh)) {\n this.getScene().depthPeelingRenderer.bind(effect);\n }\n this._eventInfo.subMesh = subMesh;\n this._callbackPluginEventBindForSubMesh(this._eventInfo);\n // Clip plane\n bindClipPlane(effect, this, scene);\n // Colors\n this.bindEyePosition(effect);\n }\n else if (scene.getEngine()._features.needToAlwaysBindUniformBuffers) {\n this._needToBindSceneUbo = true;\n }\n if (mustRebind || !this.isFrozen) {\n // Lights\n if (scene.lightsEnabled && !this._disableLighting) {\n MaterialHelper.BindLights(scene, mesh, effect, defines, this._maxSimultaneousLights);\n }\n // View\n if ((scene.fogEnabled && mesh.applyFog && scene.fogMode !== Scene.FOGMODE_NONE) ||\n this._reflectionTexture ||\n this._refractionTexture ||\n mesh.receiveShadows ||\n defines.PREPASS) {\n this.bindView(effect);\n }\n // Fog\n MaterialHelper.BindFogParameters(scene, mesh, effect);\n // Morph targets\n if (defines.NUM_MORPH_INFLUENCERS) {\n MaterialHelper.BindMorphTargetParameters(mesh, effect);\n }\n if (defines.BAKED_VERTEX_ANIMATION_TEXTURE) {\n (_a = mesh.bakedVertexAnimationManager) === null || _a === void 0 ? void 0 : _a.bind(effect, defines.INSTANCES);\n }\n // Log. depth\n if (this.useLogarithmicDepth) {\n MaterialHelper.BindLogDepth(defines, effect, scene);\n }\n // image processing\n if (this._imageProcessingConfiguration && !this._imageProcessingConfiguration.applyByPostProcess) {\n this._imageProcessingConfiguration.bind(this._activeEffect);\n }\n }\n this._afterBind(mesh, this._activeEffect);\n ubo.update();\n }\n /**\n * Get the list of animatables in the material.\n * @returns the list of animatables object used in the material\n */\n getAnimatables() {\n const results = super.getAnimatables();\n if (this._diffuseTexture && this._diffuseTexture.animations && this._diffuseTexture.animations.length > 0) {\n results.push(this._diffuseTexture);\n }\n if (this._ambientTexture && this._ambientTexture.animations && this._ambientTexture.animations.length > 0) {\n results.push(this._ambientTexture);\n }\n if (this._opacityTexture && this._opacityTexture.animations && this._opacityTexture.animations.length > 0) {\n results.push(this._opacityTexture);\n }\n if (this._reflectionTexture && this._reflectionTexture.animations && this._reflectionTexture.animations.length > 0) {\n results.push(this._reflectionTexture);\n }\n if (this._emissiveTexture && this._emissiveTexture.animations && this._emissiveTexture.animations.length > 0) {\n results.push(this._emissiveTexture);\n }\n if (this._specularTexture && this._specularTexture.animations && this._specularTexture.animations.length > 0) {\n results.push(this._specularTexture);\n }\n if (this._bumpTexture && this._bumpTexture.animations && this._bumpTexture.animations.length > 0) {\n results.push(this._bumpTexture);\n }\n if (this._lightmapTexture && this._lightmapTexture.animations && this._lightmapTexture.animations.length > 0) {\n results.push(this._lightmapTexture);\n }\n if (this._refractionTexture && this._refractionTexture.animations && this._refractionTexture.animations.length > 0) {\n results.push(this._refractionTexture);\n }\n return results;\n }\n /**\n * Gets the active textures from the material\n * @returns an array of textures\n */\n getActiveTextures() {\n const activeTextures = super.getActiveTextures();\n if (this._diffuseTexture) {\n activeTextures.push(this._diffuseTexture);\n }\n if (this._ambientTexture) {\n activeTextures.push(this._ambientTexture);\n }\n if (this._opacityTexture) {\n activeTextures.push(this._opacityTexture);\n }\n if (this._reflectionTexture) {\n activeTextures.push(this._reflectionTexture);\n }\n if (this._emissiveTexture) {\n activeTextures.push(this._emissiveTexture);\n }\n if (this._specularTexture) {\n activeTextures.push(this._specularTexture);\n }\n if (this._bumpTexture) {\n activeTextures.push(this._bumpTexture);\n }\n if (this._lightmapTexture) {\n activeTextures.push(this._lightmapTexture);\n }\n if (this._refractionTexture) {\n activeTextures.push(this._refractionTexture);\n }\n return activeTextures;\n }\n /**\n * Specifies if the material uses a texture\n * @param texture defines the texture to check against the material\n * @returns a boolean specifying if the material uses the texture\n */\n hasTexture(texture) {\n if (super.hasTexture(texture)) {\n return true;\n }\n if (this._diffuseTexture === texture) {\n return true;\n }\n if (this._ambientTexture === texture) {\n return true;\n }\n if (this._opacityTexture === texture) {\n return true;\n }\n if (this._reflectionTexture === texture) {\n return true;\n }\n if (this._emissiveTexture === texture) {\n return true;\n }\n if (this._specularTexture === texture) {\n return true;\n }\n if (this._bumpTexture === texture) {\n return true;\n }\n if (this._lightmapTexture === texture) {\n return true;\n }\n if (this._refractionTexture === texture) {\n return true;\n }\n return false;\n }\n /**\n * Disposes the material\n * @param forceDisposeEffect specifies if effects should be forcefully disposed\n * @param forceDisposeTextures specifies if textures should be forcefully disposed\n */\n dispose(forceDisposeEffect, forceDisposeTextures) {\n var _a, _b, _c, _d, _e, _f, _g, _h, _j;\n if (forceDisposeTextures) {\n (_a = this._diffuseTexture) === null || _a === void 0 ? void 0 : _a.dispose();\n (_b = this._ambientTexture) === null || _b === void 0 ? void 0 : _b.dispose();\n (_c = this._opacityTexture) === null || _c === void 0 ? void 0 : _c.dispose();\n (_d = this._reflectionTexture) === null || _d === void 0 ? void 0 : _d.dispose();\n (_e = this._emissiveTexture) === null || _e === void 0 ? void 0 : _e.dispose();\n (_f = this._specularTexture) === null || _f === void 0 ? void 0 : _f.dispose();\n (_g = this._bumpTexture) === null || _g === void 0 ? void 0 : _g.dispose();\n (_h = this._lightmapTexture) === null || _h === void 0 ? void 0 : _h.dispose();\n (_j = this._refractionTexture) === null || _j === void 0 ? void 0 : _j.dispose();\n }\n if (this._imageProcessingConfiguration && this._imageProcessingObserver) {\n this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver);\n }\n super.dispose(forceDisposeEffect, forceDisposeTextures);\n }\n /**\n * Makes a duplicate of the material, and gives it a new name\n * @param name defines the new name for the duplicated material\n * @param cloneTexturesOnlyOnce - if a texture is used in more than one channel (e.g diffuse and opacity), only clone it once and reuse it on the other channels. Default false.\n * @param rootUrl defines the root URL to use to load textures\n * @returns the cloned material\n */\n clone(name, cloneTexturesOnlyOnce = true, rootUrl = \"\") {\n const result = SerializationHelper.Clone(() => new StandardMaterial(name, this.getScene()), this, { cloneTexturesOnlyOnce });\n result.name = name;\n result.id = name;\n this.stencil.copyTo(result.stencil);\n this._clonePlugins(result, rootUrl);\n return result;\n }\n /**\n * Creates a standard material from parsed material data\n * @param source defines the JSON representation of the material\n * @param scene defines the hosting scene\n * @param rootUrl defines the root URL to use to load textures and relative dependencies\n * @returns a new standard material\n */\n static Parse(source, scene, rootUrl) {\n const material = SerializationHelper.Parse(() => new StandardMaterial(source.name, scene), source, scene, rootUrl);\n if (source.stencil) {\n material.stencil.parse(source.stencil, scene, rootUrl);\n }\n Material._parsePlugins(source, material, scene, rootUrl);\n return material;\n }\n // Flags used to enable or disable a type of texture for all Standard Materials\n /**\n * Are diffuse textures enabled in the application.\n */\n static get DiffuseTextureEnabled() {\n return MaterialFlags.DiffuseTextureEnabled;\n }\n static set DiffuseTextureEnabled(value) {\n MaterialFlags.DiffuseTextureEnabled = value;\n }\n /**\n * Are detail textures enabled in the application.\n */\n static get DetailTextureEnabled() {\n return MaterialFlags.DetailTextureEnabled;\n }\n static set DetailTextureEnabled(value) {\n MaterialFlags.DetailTextureEnabled = value;\n }\n /**\n * Are ambient textures enabled in the application.\n */\n static get AmbientTextureEnabled() {\n return MaterialFlags.AmbientTextureEnabled;\n }\n static set AmbientTextureEnabled(value) {\n MaterialFlags.AmbientTextureEnabled = value;\n }\n /**\n * Are opacity textures enabled in the application.\n */\n static get OpacityTextureEnabled() {\n return MaterialFlags.OpacityTextureEnabled;\n }\n static set OpacityTextureEnabled(value) {\n MaterialFlags.OpacityTextureEnabled = value;\n }\n /**\n * Are reflection textures enabled in the application.\n */\n static get ReflectionTextureEnabled() {\n return MaterialFlags.ReflectionTextureEnabled;\n }\n static set ReflectionTextureEnabled(value) {\n MaterialFlags.ReflectionTextureEnabled = value;\n }\n /**\n * Are emissive textures enabled in the application.\n */\n static get EmissiveTextureEnabled() {\n return MaterialFlags.EmissiveTextureEnabled;\n }\n static set EmissiveTextureEnabled(value) {\n MaterialFlags.EmissiveTextureEnabled = value;\n }\n /**\n * Are specular textures enabled in the application.\n */\n static get SpecularTextureEnabled() {\n return MaterialFlags.SpecularTextureEnabled;\n }\n static set SpecularTextureEnabled(value) {\n MaterialFlags.SpecularTextureEnabled = value;\n }\n /**\n * Are bump textures enabled in the application.\n */\n static get BumpTextureEnabled() {\n return MaterialFlags.BumpTextureEnabled;\n }\n static set BumpTextureEnabled(value) {\n MaterialFlags.BumpTextureEnabled = value;\n }\n /**\n * Are lightmap textures enabled in the application.\n */\n static get LightmapTextureEnabled() {\n return MaterialFlags.LightmapTextureEnabled;\n }\n static set LightmapTextureEnabled(value) {\n MaterialFlags.LightmapTextureEnabled = value;\n }\n /**\n * Are refraction textures enabled in the application.\n */\n static get RefractionTextureEnabled() {\n return MaterialFlags.RefractionTextureEnabled;\n }\n static set RefractionTextureEnabled(value) {\n MaterialFlags.RefractionTextureEnabled = value;\n }\n /**\n * Are color grading textures enabled in the application.\n */\n static get ColorGradingTextureEnabled() {\n return MaterialFlags.ColorGradingTextureEnabled;\n }\n static set ColorGradingTextureEnabled(value) {\n MaterialFlags.ColorGradingTextureEnabled = value;\n }\n /**\n * Are fresnels enabled in the application.\n */\n static get FresnelEnabled() {\n return MaterialFlags.FresnelEnabled;\n }\n static set FresnelEnabled(value) {\n MaterialFlags.FresnelEnabled = value;\n }\n }\n __decorate$1([\n serializeAsTexture(\"diffuseTexture\")\n ], StandardMaterial.prototype, \"_diffuseTexture\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesAndMiscDirty\")\n ], StandardMaterial.prototype, \"diffuseTexture\", void 0);\n __decorate$1([\n serializeAsTexture(\"ambientTexture\")\n ], StandardMaterial.prototype, \"_ambientTexture\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], StandardMaterial.prototype, \"ambientTexture\", void 0);\n __decorate$1([\n serializeAsTexture(\"opacityTexture\")\n ], StandardMaterial.prototype, \"_opacityTexture\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesAndMiscDirty\")\n ], StandardMaterial.prototype, \"opacityTexture\", void 0);\n __decorate$1([\n serializeAsTexture(\"reflectionTexture\")\n ], StandardMaterial.prototype, \"_reflectionTexture\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], StandardMaterial.prototype, \"reflectionTexture\", void 0);\n __decorate$1([\n serializeAsTexture(\"emissiveTexture\")\n ], StandardMaterial.prototype, \"_emissiveTexture\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], StandardMaterial.prototype, \"emissiveTexture\", void 0);\n __decorate$1([\n serializeAsTexture(\"specularTexture\")\n ], StandardMaterial.prototype, \"_specularTexture\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], StandardMaterial.prototype, \"specularTexture\", void 0);\n __decorate$1([\n serializeAsTexture(\"bumpTexture\")\n ], StandardMaterial.prototype, \"_bumpTexture\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], StandardMaterial.prototype, \"bumpTexture\", void 0);\n __decorate$1([\n serializeAsTexture(\"lightmapTexture\")\n ], StandardMaterial.prototype, \"_lightmapTexture\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], StandardMaterial.prototype, \"lightmapTexture\", void 0);\n __decorate$1([\n serializeAsTexture(\"refractionTexture\")\n ], StandardMaterial.prototype, \"_refractionTexture\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], StandardMaterial.prototype, \"refractionTexture\", void 0);\n __decorate$1([\n serializeAsColor3(\"ambient\")\n ], StandardMaterial.prototype, \"ambientColor\", void 0);\n __decorate$1([\n serializeAsColor3(\"diffuse\")\n ], StandardMaterial.prototype, \"diffuseColor\", void 0);\n __decorate$1([\n serializeAsColor3(\"specular\")\n ], StandardMaterial.prototype, \"specularColor\", void 0);\n __decorate$1([\n serializeAsColor3(\"emissive\")\n ], StandardMaterial.prototype, \"emissiveColor\", void 0);\n __decorate$1([\n serialize()\n ], StandardMaterial.prototype, \"specularPower\", void 0);\n __decorate$1([\n serialize(\"useAlphaFromDiffuseTexture\")\n ], StandardMaterial.prototype, \"_useAlphaFromDiffuseTexture\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesAndMiscDirty\")\n ], StandardMaterial.prototype, \"useAlphaFromDiffuseTexture\", void 0);\n __decorate$1([\n serialize(\"useEmissiveAsIllumination\")\n ], StandardMaterial.prototype, \"_useEmissiveAsIllumination\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], StandardMaterial.prototype, \"useEmissiveAsIllumination\", void 0);\n __decorate$1([\n serialize(\"linkEmissiveWithDiffuse\")\n ], StandardMaterial.prototype, \"_linkEmissiveWithDiffuse\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], StandardMaterial.prototype, \"linkEmissiveWithDiffuse\", void 0);\n __decorate$1([\n serialize(\"useSpecularOverAlpha\")\n ], StandardMaterial.prototype, \"_useSpecularOverAlpha\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], StandardMaterial.prototype, \"useSpecularOverAlpha\", void 0);\n __decorate$1([\n serialize(\"useReflectionOverAlpha\")\n ], StandardMaterial.prototype, \"_useReflectionOverAlpha\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], StandardMaterial.prototype, \"useReflectionOverAlpha\", void 0);\n __decorate$1([\n serialize(\"disableLighting\")\n ], StandardMaterial.prototype, \"_disableLighting\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsLightsDirty\")\n ], StandardMaterial.prototype, \"disableLighting\", void 0);\n __decorate$1([\n serialize(\"useObjectSpaceNormalMap\")\n ], StandardMaterial.prototype, \"_useObjectSpaceNormalMap\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], StandardMaterial.prototype, \"useObjectSpaceNormalMap\", void 0);\n __decorate$1([\n serialize(\"useParallax\")\n ], StandardMaterial.prototype, \"_useParallax\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], StandardMaterial.prototype, \"useParallax\", void 0);\n __decorate$1([\n serialize(\"useParallaxOcclusion\")\n ], StandardMaterial.prototype, \"_useParallaxOcclusion\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], StandardMaterial.prototype, \"useParallaxOcclusion\", void 0);\n __decorate$1([\n serialize()\n ], StandardMaterial.prototype, \"parallaxScaleBias\", void 0);\n __decorate$1([\n serialize(\"roughness\")\n ], StandardMaterial.prototype, \"_roughness\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], StandardMaterial.prototype, \"roughness\", void 0);\n __decorate$1([\n serialize()\n ], StandardMaterial.prototype, \"indexOfRefraction\", void 0);\n __decorate$1([\n serialize()\n ], StandardMaterial.prototype, \"invertRefractionY\", void 0);\n __decorate$1([\n serialize()\n ], StandardMaterial.prototype, \"alphaCutOff\", void 0);\n __decorate$1([\n serialize(\"useLightmapAsShadowmap\")\n ], StandardMaterial.prototype, \"_useLightmapAsShadowmap\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], StandardMaterial.prototype, \"useLightmapAsShadowmap\", void 0);\n __decorate$1([\n serializeAsFresnelParameters(\"diffuseFresnelParameters\")\n ], StandardMaterial.prototype, \"_diffuseFresnelParameters\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsFresnelDirty\")\n ], StandardMaterial.prototype, \"diffuseFresnelParameters\", void 0);\n __decorate$1([\n serializeAsFresnelParameters(\"opacityFresnelParameters\")\n ], StandardMaterial.prototype, \"_opacityFresnelParameters\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsFresnelAndMiscDirty\")\n ], StandardMaterial.prototype, \"opacityFresnelParameters\", void 0);\n __decorate$1([\n serializeAsFresnelParameters(\"reflectionFresnelParameters\")\n ], StandardMaterial.prototype, \"_reflectionFresnelParameters\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsFresnelDirty\")\n ], StandardMaterial.prototype, \"reflectionFresnelParameters\", void 0);\n __decorate$1([\n serializeAsFresnelParameters(\"refractionFresnelParameters\")\n ], StandardMaterial.prototype, \"_refractionFresnelParameters\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsFresnelDirty\")\n ], StandardMaterial.prototype, \"refractionFresnelParameters\", void 0);\n __decorate$1([\n serializeAsFresnelParameters(\"emissiveFresnelParameters\")\n ], StandardMaterial.prototype, \"_emissiveFresnelParameters\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsFresnelDirty\")\n ], StandardMaterial.prototype, \"emissiveFresnelParameters\", void 0);\n __decorate$1([\n serialize(\"useReflectionFresnelFromSpecular\")\n ], StandardMaterial.prototype, \"_useReflectionFresnelFromSpecular\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsFresnelDirty\")\n ], StandardMaterial.prototype, \"useReflectionFresnelFromSpecular\", void 0);\n __decorate$1([\n serialize(\"useGlossinessFromSpecularMapAlpha\")\n ], StandardMaterial.prototype, \"_useGlossinessFromSpecularMapAlpha\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], StandardMaterial.prototype, \"useGlossinessFromSpecularMapAlpha\", void 0);\n __decorate$1([\n serialize(\"maxSimultaneousLights\")\n ], StandardMaterial.prototype, \"_maxSimultaneousLights\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsLightsDirty\")\n ], StandardMaterial.prototype, \"maxSimultaneousLights\", void 0);\n __decorate$1([\n serialize(\"invertNormalMapX\")\n ], StandardMaterial.prototype, \"_invertNormalMapX\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], StandardMaterial.prototype, \"invertNormalMapX\", void 0);\n __decorate$1([\n serialize(\"invertNormalMapY\")\n ], StandardMaterial.prototype, \"_invertNormalMapY\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], StandardMaterial.prototype, \"invertNormalMapY\", void 0);\n __decorate$1([\n serialize(\"twoSidedLighting\")\n ], StandardMaterial.prototype, \"_twoSidedLighting\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], StandardMaterial.prototype, \"twoSidedLighting\", void 0);\n __decorate$1([\n serialize(\"applyDecalMapAfterDetailMap\")\n ], StandardMaterial.prototype, \"_applyDecalMapAfterDetailMap\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsMiscDirty\")\n ], StandardMaterial.prototype, \"applyDecalMapAfterDetailMap\", void 0);\n __decorate$1([\n serialize()\n ], StandardMaterial.prototype, \"useLogarithmicDepth\", null);\n RegisterClass(\"BABYLON.StandardMaterial\", StandardMaterial);\n Scene.DefaultMaterialFactory = (scene) => {\n return new StandardMaterial(\"default material\", scene);\n };\n\n // Do not edit.\n const name$$ = \"postprocessVertexShader\";\n const shader$$ = `attribute vec2 position;uniform vec2 scale;varying vec2 vUV;const vec2 madd=vec2(0.5,0.5);\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\nvUV=(position*madd+madd)*scale;gl_Position=vec4(position,0.0,1.0);\n#define CUSTOM_VERTEX_MAIN_END\n}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$$] = shader$$;\n\n /**\n * Wrapper around a render target (either single or multi textures)\n */\n class RenderTargetWrapper {\n /**\n * Gets the depth/stencil texture (if created by a createDepthStencilTexture() call)\n */\n get depthStencilTexture() {\n return this._depthStencilTexture;\n }\n /**\n * Indicates if the depth/stencil texture has a stencil aspect\n */\n get depthStencilTextureWithStencil() {\n return this._depthStencilTextureWithStencil;\n }\n /**\n * Defines if the render target wrapper is for a cube texture or if false a 2d texture\n */\n get isCube() {\n return this._isCube;\n }\n /**\n * Defines if the render target wrapper is for a single or multi target render wrapper\n */\n get isMulti() {\n return this._isMulti;\n }\n /**\n * Defines if the render target wrapper is for a single or an array of textures\n */\n get is2DArray() {\n return this.layers > 0;\n }\n /**\n * Gets the size of the render target wrapper (used for cubes, as width=height in this case)\n */\n get size() {\n return this.width;\n }\n /**\n * Gets the width of the render target wrapper\n */\n get width() {\n return this._size.width || this._size;\n }\n /**\n * Gets the height of the render target wrapper\n */\n get height() {\n return this._size.height || this._size;\n }\n /**\n * Gets the number of layers of the render target wrapper (only used if is2DArray is true and wrapper is not a multi render target)\n */\n get layers() {\n return this._size.layers || 0;\n }\n /**\n * Gets the render texture. If this is a multi render target, gets the first texture\n */\n get texture() {\n var _a, _b;\n return (_b = (_a = this._textures) === null || _a === void 0 ? void 0 : _a[0]) !== null && _b !== void 0 ? _b : null;\n }\n /**\n * Gets the list of render textures. If we are not in a multi render target, the list will be null (use the texture getter instead)\n */\n get textures() {\n return this._textures;\n }\n /**\n * Gets the face indices that correspond to the list of render textures. If we are not in a multi render target, the list will be null\n */\n get faceIndices() {\n return this._faceIndices;\n }\n /**\n * Gets the layer indices that correspond to the list of render textures. If we are not in a multi render target, the list will be null\n */\n get layerIndices() {\n return this._layerIndices;\n }\n /**\n * Gets the sample count of the render target\n */\n get samples() {\n return this._samples;\n }\n /**\n * Sets the sample count of the render target\n * @param value sample count\n * @param initializeBuffers If set to true, the engine will make an initializing call to drawBuffers (only used when isMulti=true).\n * @param force true to force calling the update sample count engine function even if the current sample count is equal to value\n * @returns the sample count that has been set\n */\n setSamples(value, initializeBuffers = true, force = false) {\n if (this.samples === value && !force) {\n return value;\n }\n const result = this._isMulti\n ? this._engine.updateMultipleRenderTargetTextureSampleCount(this, value, initializeBuffers)\n : this._engine.updateRenderTargetTextureSampleCount(this, value);\n this._samples = value;\n return result;\n }\n /**\n * Initializes the render target wrapper\n * @param isMulti true if the wrapper is a multi render target\n * @param isCube true if the wrapper should render to a cube texture\n * @param size size of the render target (width/height/layers)\n * @param engine engine used to create the render target\n */\n constructor(isMulti, isCube, size, engine) {\n this._textures = null;\n this._faceIndices = null;\n this._layerIndices = null;\n /** @internal */\n this._samples = 1;\n /** @internal */\n this._attachments = null;\n /** @internal */\n this._generateStencilBuffer = false;\n /** @internal */\n this._generateDepthBuffer = false;\n /** @internal */\n this._depthStencilTextureWithStencil = false;\n this._isMulti = isMulti;\n this._isCube = isCube;\n this._size = size;\n this._engine = engine;\n this._depthStencilTexture = null;\n }\n /**\n * Sets the render target texture(s)\n * @param textures texture(s) to set\n */\n setTextures(textures) {\n if (Array.isArray(textures)) {\n this._textures = textures;\n }\n else if (textures) {\n this._textures = [textures];\n }\n else {\n this._textures = null;\n }\n }\n /**\n * Set a texture in the textures array\n * @param texture The texture to set\n * @param index The index in the textures array to set\n * @param disposePrevious If this function should dispose the previous texture\n */\n setTexture(texture, index = 0, disposePrevious = true) {\n if (!this._textures) {\n this._textures = [];\n }\n if (this._textures[index] && disposePrevious) {\n this._textures[index].dispose();\n }\n this._textures[index] = texture;\n }\n /**\n * Sets the layer and face indices of every render target texture bound to each color attachment\n * @param layers The layers of each texture to be set\n * @param faces The faces of each texture to be set\n */\n setLayerAndFaceIndices(layers, faces) {\n this._layerIndices = layers;\n this._faceIndices = faces;\n }\n /**\n * Sets the layer and face indices of a texture in the textures array that should be bound to each color attachment\n * @param index The index of the texture in the textures array to modify\n * @param layer The layer of the texture to be set\n * @param face The face of the texture to be set\n */\n setLayerAndFaceIndex(index = 0, layer, face) {\n if (!this._layerIndices) {\n this._layerIndices = [];\n }\n if (!this._faceIndices) {\n this._faceIndices = [];\n }\n if (layer !== undefined && layer >= 0) {\n this._layerIndices[index] = layer;\n }\n if (face !== undefined && face >= 0) {\n this._faceIndices[index] = face;\n }\n }\n /**\n * Creates the depth/stencil texture\n * @param comparisonFunction Comparison function to use for the texture\n * @param bilinearFiltering true if bilinear filtering should be used when sampling the texture\n * @param generateStencil true if the stencil aspect should also be created\n * @param samples sample count to use when creating the texture\n * @param format format of the depth texture\n * @param label defines the label to use for the texture (for debugging purpose only)\n * @returns the depth/stencil created texture\n */\n createDepthStencilTexture(comparisonFunction = 0, bilinearFiltering = true, generateStencil = false, samples = 1, format = 14, label) {\n var _a;\n (_a = this._depthStencilTexture) === null || _a === void 0 ? void 0 : _a.dispose();\n this._depthStencilTextureWithStencil = generateStencil;\n this._depthStencilTexture = this._engine.createDepthStencilTexture(this._size, {\n bilinearFiltering,\n comparisonFunction,\n generateStencil,\n isCube: this._isCube,\n samples,\n depthTextureFormat: format,\n label,\n }, this);\n return this._depthStencilTexture;\n }\n /**\n * Shares the depth buffer of this render target with another render target.\n * @internal\n * @param renderTarget Destination renderTarget\n */\n _shareDepth(renderTarget) {\n if (this._depthStencilTexture) {\n if (renderTarget._depthStencilTexture) {\n renderTarget._depthStencilTexture.dispose();\n }\n renderTarget._depthStencilTexture = this._depthStencilTexture;\n this._depthStencilTexture.incrementReferences();\n }\n }\n /**\n * @internal\n */\n _swapAndDie(target) {\n if (this.texture) {\n this.texture._swapAndDie(target);\n }\n this._textures = null;\n this.dispose(true);\n }\n _cloneRenderTargetWrapper() {\n var _a, _b, _c, _d, _e, _f, _g, _h;\n let rtw = null;\n if (this._isMulti) {\n const textureArray = this.textures;\n if (textureArray && textureArray.length > 0) {\n let generateDepthTexture = false;\n let textureCount = textureArray.length;\n const lastTextureSource = textureArray[textureArray.length - 1]._source;\n if (lastTextureSource === InternalTextureSource.Depth || lastTextureSource === InternalTextureSource.DepthStencil) {\n generateDepthTexture = true;\n textureCount--;\n }\n const samplingModes = [];\n const types = [];\n const formats = [];\n const targetTypes = [];\n const faceIndex = [];\n const layerIndex = [];\n const layerCounts = [];\n const internalTexture2Index = {};\n for (let i = 0; i < textureCount; ++i) {\n const texture = textureArray[i];\n samplingModes.push(texture.samplingMode);\n types.push(texture.type);\n formats.push(texture.format);\n const index = internalTexture2Index[texture.uniqueId];\n if (index !== undefined) {\n targetTypes.push(-1);\n layerCounts.push(0);\n }\n else {\n internalTexture2Index[texture.uniqueId] = i;\n if (texture.is2DArray) {\n targetTypes.push(35866);\n layerCounts.push(texture.depth);\n }\n else if (texture.isCube) {\n targetTypes.push(34067);\n layerCounts.push(0);\n } /*else if (texture.isCubeArray) {\n targetTypes.push(3735928559);\n layerCounts.push(texture.depth);\n }*/\n else if (texture.is3D) {\n targetTypes.push(32879);\n layerCounts.push(texture.depth);\n }\n else {\n targetTypes.push(3553);\n layerCounts.push(0);\n }\n }\n if (this._faceIndices) {\n faceIndex.push((_a = this._faceIndices[i]) !== null && _a !== void 0 ? _a : 0);\n }\n if (this._layerIndices) {\n layerIndex.push((_b = this._layerIndices[i]) !== null && _b !== void 0 ? _b : 0);\n }\n }\n const optionsMRT = {\n samplingModes,\n generateMipMaps: textureArray[0].generateMipMaps,\n generateDepthBuffer: this._generateDepthBuffer,\n generateStencilBuffer: this._generateStencilBuffer,\n generateDepthTexture,\n types,\n formats,\n textureCount,\n targetTypes,\n faceIndex,\n layerIndex,\n layerCounts,\n };\n const size = {\n width: this.width,\n height: this.height,\n };\n rtw = this._engine.createMultipleRenderTarget(size, optionsMRT);\n for (let i = 0; i < textureCount; ++i) {\n if (targetTypes[i] !== -1) {\n continue;\n }\n const index = internalTexture2Index[textureArray[i].uniqueId];\n rtw.setTexture(rtw.textures[index], i);\n }\n }\n }\n else {\n const options = {};\n options.generateDepthBuffer = this._generateDepthBuffer;\n options.generateMipMaps = (_d = (_c = this.texture) === null || _c === void 0 ? void 0 : _c.generateMipMaps) !== null && _d !== void 0 ? _d : false;\n options.generateStencilBuffer = this._generateStencilBuffer;\n options.samplingMode = (_e = this.texture) === null || _e === void 0 ? void 0 : _e.samplingMode;\n options.type = (_f = this.texture) === null || _f === void 0 ? void 0 : _f.type;\n options.format = (_g = this.texture) === null || _g === void 0 ? void 0 : _g.format;\n if (this.isCube) {\n rtw = this._engine.createRenderTargetCubeTexture(this.width, options);\n }\n else {\n const size = {\n width: this.width,\n height: this.height,\n layers: this.is2DArray ? (_h = this.texture) === null || _h === void 0 ? void 0 : _h.depth : undefined,\n };\n rtw = this._engine.createRenderTargetTexture(size, options);\n }\n rtw.texture.isReady = true;\n }\n return rtw;\n }\n _swapRenderTargetWrapper(target) {\n if (this._textures && target._textures) {\n for (let i = 0; i < this._textures.length; ++i) {\n this._textures[i]._swapAndDie(target._textures[i], false);\n target._textures[i].isReady = true;\n }\n }\n if (this._depthStencilTexture && target._depthStencilTexture) {\n this._depthStencilTexture._swapAndDie(target._depthStencilTexture);\n target._depthStencilTexture.isReady = true;\n }\n this._textures = null;\n this._depthStencilTexture = null;\n }\n /** @internal */\n _rebuild() {\n const rtw = this._cloneRenderTargetWrapper();\n if (!rtw) {\n return;\n }\n if (this._depthStencilTexture) {\n const samplingMode = this._depthStencilTexture.samplingMode;\n const bilinear = samplingMode === 2 ||\n samplingMode === 3 ||\n samplingMode === 11;\n rtw.createDepthStencilTexture(this._depthStencilTexture._comparisonFunction, bilinear, this._depthStencilTextureWithStencil, this._depthStencilTexture.samples);\n }\n if (this.samples > 1) {\n rtw.setSamples(this.samples);\n }\n rtw._swapRenderTargetWrapper(this);\n rtw.dispose();\n }\n /**\n * Releases the internal render textures\n */\n releaseTextures() {\n var _a, _b;\n if (this._textures) {\n for (let i = 0; (_b = i < ((_a = this._textures) === null || _a === void 0 ? void 0 : _a.length)) !== null && _b !== void 0 ? _b : 0; ++i) {\n this._textures[i].dispose();\n }\n }\n this._textures = null;\n }\n /**\n * Disposes the whole render target wrapper\n * @param disposeOnlyFramebuffers true if only the frame buffers should be released (used for the WebGL engine). If false, all the textures will also be released\n */\n dispose(disposeOnlyFramebuffers = false) {\n var _a;\n if (!disposeOnlyFramebuffers) {\n (_a = this._depthStencilTexture) === null || _a === void 0 ? void 0 : _a.dispose();\n this._depthStencilTexture = null;\n this.releaseTextures();\n }\n this._engine._releaseRenderTargetWrapper(this);\n }\n }\n\n /** @internal */\n class WebGLRenderTargetWrapper extends RenderTargetWrapper {\n constructor(isMulti, isCube, size, engine, context) {\n super(isMulti, isCube, size, engine);\n /**\n * @internal\n */\n this._framebuffer = null;\n /**\n * @internal\n */\n this._depthStencilBuffer = null;\n // eslint-disable-next-line @typescript-eslint/naming-convention\n /**\n * @internal\n */\n this._MSAAFramebuffer = null;\n // Multiview\n /**\n * @internal\n */\n this._colorTextureArray = null;\n /**\n * @internal\n */\n this._depthStencilTextureArray = null;\n /**\n * @internal\n */\n this._disposeOnlyFramebuffers = false;\n this._context = context;\n }\n _cloneRenderTargetWrapper() {\n let rtw = null;\n if (this._colorTextureArray && this._depthStencilTextureArray) {\n rtw = this._engine.createMultiviewRenderTargetTexture(this.width, this.height);\n rtw.texture.isReady = true;\n }\n else {\n rtw = super._cloneRenderTargetWrapper();\n }\n return rtw;\n }\n _swapRenderTargetWrapper(target) {\n super._swapRenderTargetWrapper(target);\n target._framebuffer = this._framebuffer;\n target._depthStencilBuffer = this._depthStencilBuffer;\n target._MSAAFramebuffer = this._MSAAFramebuffer;\n target._colorTextureArray = this._colorTextureArray;\n target._depthStencilTextureArray = this._depthStencilTextureArray;\n this._framebuffer = this._depthStencilBuffer = this._MSAAFramebuffer = this._colorTextureArray = this._depthStencilTextureArray = null;\n }\n /**\n * Shares the depth buffer of this render target with another render target.\n * @internal\n * @param renderTarget Destination renderTarget\n */\n _shareDepth(renderTarget) {\n super._shareDepth(renderTarget);\n const gl = this._context;\n const depthbuffer = this._depthStencilBuffer;\n const framebuffer = renderTarget._MSAAFramebuffer || renderTarget._framebuffer;\n if (renderTarget._depthStencilBuffer) {\n gl.deleteRenderbuffer(renderTarget._depthStencilBuffer);\n }\n renderTarget._depthStencilBuffer = this._depthStencilBuffer;\n this._engine._bindUnboundFramebuffer(framebuffer);\n gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_ATTACHMENT, gl.RENDERBUFFER, depthbuffer);\n this._engine._bindUnboundFramebuffer(null);\n }\n /**\n * Binds a texture to this render target on a specific attachment\n * @param texture The texture to bind to the framebuffer\n * @param attachmentIndex Index of the attachment\n * @param faceIndexOrLayer The face or layer of the texture to render to in case of cube texture or array texture\n * @param lodLevel defines the lod level to bind to the frame buffer\n */\n _bindTextureRenderTarget(texture, attachmentIndex = 0, faceIndexOrLayer, lodLevel = 0) {\n var _a, _b, _c, _d;\n if (!texture._hardwareTexture) {\n return;\n }\n const framebuffer = this._framebuffer;\n const currentFB = this._engine._currentFramebuffer;\n this._engine._bindUnboundFramebuffer(framebuffer);\n if (this._engine.webGLVersion > 1) {\n const gl = this._context;\n const attachment = gl[\"COLOR_ATTACHMENT\" + attachmentIndex];\n if (texture.is2DArray || texture.is3D) {\n faceIndexOrLayer = (_b = faceIndexOrLayer !== null && faceIndexOrLayer !== void 0 ? faceIndexOrLayer : (_a = this.layerIndices) === null || _a === void 0 ? void 0 : _a[attachmentIndex]) !== null && _b !== void 0 ? _b : 0;\n gl.framebufferTextureLayer(gl.FRAMEBUFFER, attachment, texture._hardwareTexture.underlyingResource, lodLevel, faceIndexOrLayer);\n }\n else if (texture.isCube) {\n // if face index is not specified, try to query it from faceIndices\n // default is face 0\n faceIndexOrLayer = (_d = faceIndexOrLayer !== null && faceIndexOrLayer !== void 0 ? faceIndexOrLayer : (_c = this.faceIndices) === null || _c === void 0 ? void 0 : _c[attachmentIndex]) !== null && _d !== void 0 ? _d : 0;\n gl.framebufferTexture2D(gl.FRAMEBUFFER, attachment, gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndexOrLayer, texture._hardwareTexture.underlyingResource, lodLevel);\n }\n else {\n gl.framebufferTexture2D(gl.FRAMEBUFFER, attachment, gl.TEXTURE_2D, texture._hardwareTexture.underlyingResource, lodLevel);\n }\n }\n else {\n // Default behavior (WebGL)\n const gl = this._context;\n const attachment = gl[\"COLOR_ATTACHMENT\" + attachmentIndex + \"_WEBGL\"];\n const target = faceIndexOrLayer !== undefined ? gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndexOrLayer : gl.TEXTURE_2D;\n gl.framebufferTexture2D(gl.FRAMEBUFFER, attachment, target, texture._hardwareTexture.underlyingResource, lodLevel);\n }\n this._engine._bindUnboundFramebuffer(currentFB);\n }\n /**\n * Set a texture in the textures array\n * @param texture the texture to set\n * @param index the index in the textures array to set\n * @param disposePrevious If this function should dispose the previous texture\n */\n setTexture(texture, index = 0, disposePrevious = true) {\n super.setTexture(texture, index, disposePrevious);\n this._bindTextureRenderTarget(texture, index);\n }\n /**\n * Sets the layer and face indices of every render target texture\n * @param layers The layer of the texture to be set (make negative to not modify)\n * @param faces The face of the texture to be set (make negative to not modify)\n */\n setLayerAndFaceIndices(layers, faces) {\n var _a, _b;\n super.setLayerAndFaceIndices(layers, faces);\n if (!this.textures || !this.layerIndices || !this.faceIndices) {\n return;\n }\n // the length of this._attachments is the right one as it does not count the depth texture, in case we generated it\n const textureCount = (_b = (_a = this._attachments) === null || _a === void 0 ? void 0 : _a.length) !== null && _b !== void 0 ? _b : this.textures.length;\n for (let index = 0; index < textureCount; index++) {\n const texture = this.textures[index];\n if (!texture) {\n // The target type was probably -1 at creation time and setTexture has not been called yet for this index\n continue;\n }\n if (texture.is2DArray || texture.is3D) {\n this._bindTextureRenderTarget(texture, index, this.layerIndices[index]);\n }\n else if (texture.isCube) {\n this._bindTextureRenderTarget(texture, index, this.faceIndices[index]);\n }\n else {\n this._bindTextureRenderTarget(texture, index);\n }\n }\n }\n /**\n * Set the face and layer indices of a texture in the textures array\n * @param index The index of the texture in the textures array to modify\n * @param layer The layer of the texture to be set\n * @param face The face of the texture to be set\n */\n setLayerAndFaceIndex(index = 0, layer, face) {\n super.setLayerAndFaceIndex(index, layer, face);\n if (!this.textures || !this.layerIndices || !this.faceIndices) {\n return;\n }\n const texture = this.textures[index];\n if (texture.is2DArray || texture.is3D) {\n this._bindTextureRenderTarget(this.textures[index], index, this.layerIndices[index]);\n }\n else if (texture.isCube) {\n this._bindTextureRenderTarget(this.textures[index], index, this.faceIndices[index]);\n }\n }\n dispose(disposeOnlyFramebuffers = this._disposeOnlyFramebuffers) {\n const gl = this._context;\n if (!disposeOnlyFramebuffers) {\n if (this._colorTextureArray) {\n this._context.deleteTexture(this._colorTextureArray);\n this._colorTextureArray = null;\n }\n if (this._depthStencilTextureArray) {\n this._context.deleteTexture(this._depthStencilTextureArray);\n this._depthStencilTextureArray = null;\n }\n }\n if (this._framebuffer) {\n gl.deleteFramebuffer(this._framebuffer);\n this._framebuffer = null;\n }\n if (this._depthStencilBuffer) {\n gl.deleteRenderbuffer(this._depthStencilBuffer);\n this._depthStencilBuffer = null;\n }\n if (this._MSAAFramebuffer) {\n gl.deleteFramebuffer(this._MSAAFramebuffer);\n this._MSAAFramebuffer = null;\n }\n super.dispose(disposeOnlyFramebuffers);\n }\n }\n\n ThinEngine.prototype._createHardwareRenderTargetWrapper = function (isMulti, isCube, size) {\n const rtWrapper = new WebGLRenderTargetWrapper(isMulti, isCube, size, this, this._gl);\n this._renderTargetWrapperCache.push(rtWrapper);\n return rtWrapper;\n };\n ThinEngine.prototype.createRenderTargetTexture = function (size, options) {\n var _a, _b;\n const rtWrapper = this._createHardwareRenderTargetWrapper(false, false, size);\n let generateDepthBuffer = true;\n let generateStencilBuffer = false;\n let noColorAttachment = false;\n let colorAttachment = undefined;\n let samples = 1;\n if (options !== undefined && typeof options === \"object\") {\n generateDepthBuffer = (_a = options.generateDepthBuffer) !== null && _a !== void 0 ? _a : true;\n generateStencilBuffer = !!options.generateStencilBuffer;\n noColorAttachment = !!options.noColorAttachment;\n colorAttachment = options.colorAttachment;\n samples = (_b = options.samples) !== null && _b !== void 0 ? _b : 1;\n }\n const texture = colorAttachment || (noColorAttachment ? null : this._createInternalTexture(size, options, true, InternalTextureSource.RenderTarget));\n const width = size.width || size;\n const height = size.height || size;\n const currentFrameBuffer = this._currentFramebuffer;\n const gl = this._gl;\n // Create the framebuffer\n const framebuffer = gl.createFramebuffer();\n this._bindUnboundFramebuffer(framebuffer);\n rtWrapper._depthStencilBuffer = this._setupFramebufferDepthAttachments(generateStencilBuffer, generateDepthBuffer, width, height);\n // No need to rebind on every frame\n if (texture && !texture.is2DArray) {\n gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, texture._hardwareTexture.underlyingResource, 0);\n }\n this._bindUnboundFramebuffer(currentFrameBuffer);\n rtWrapper._framebuffer = framebuffer;\n rtWrapper._generateDepthBuffer = generateDepthBuffer;\n rtWrapper._generateStencilBuffer = generateStencilBuffer;\n rtWrapper.setTextures(texture);\n this.updateRenderTargetTextureSampleCount(rtWrapper, samples);\n return rtWrapper;\n };\n ThinEngine.prototype.createDepthStencilTexture = function (size, options, rtWrapper) {\n if (options.isCube) {\n const width = size.width || size;\n return this._createDepthStencilCubeTexture(width, options, rtWrapper);\n }\n else {\n return this._createDepthStencilTexture(size, options, rtWrapper);\n }\n };\n ThinEngine.prototype._createDepthStencilTexture = function (size, options, rtWrapper) {\n const gl = this._gl;\n const layers = size.layers || 0;\n const target = layers !== 0 ? gl.TEXTURE_2D_ARRAY : gl.TEXTURE_2D;\n const internalTexture = new InternalTexture(this, InternalTextureSource.DepthStencil);\n if (!this._caps.depthTextureExtension) {\n Logger.Error(\"Depth texture is not supported by your browser or hardware.\");\n return internalTexture;\n }\n const internalOptions = Object.assign({ bilinearFiltering: false, comparisonFunction: 0, generateStencil: false }, options);\n this._bindTextureDirectly(target, internalTexture, true);\n this._setupDepthStencilTexture(internalTexture, size, internalOptions.generateStencil, internalOptions.comparisonFunction === 0 ? false : internalOptions.bilinearFiltering, internalOptions.comparisonFunction, internalOptions.samples);\n if (internalOptions.depthTextureFormat !== undefined) {\n if (internalOptions.depthTextureFormat !== 15 &&\n internalOptions.depthTextureFormat !== 16 &&\n internalOptions.depthTextureFormat !== 17 &&\n internalOptions.depthTextureFormat !== 13 &&\n internalOptions.depthTextureFormat !== 14 &&\n internalOptions.depthTextureFormat !== 18) {\n Logger.Error(\"Depth texture format is not supported.\");\n return internalTexture;\n }\n internalTexture.format = internalOptions.depthTextureFormat;\n }\n else {\n internalTexture.format = internalOptions.generateStencil ? 13 : 16;\n }\n const hasStencil = internalTexture.format === 17 ||\n internalTexture.format === 13 ||\n internalTexture.format === 18;\n rtWrapper._depthStencilTexture = internalTexture;\n rtWrapper._depthStencilTextureWithStencil = hasStencil;\n let type = gl.UNSIGNED_INT;\n if (internalTexture.format === 15) {\n type = gl.UNSIGNED_SHORT;\n }\n else if (internalTexture.format === 17 || internalTexture.format === 13) {\n type = gl.UNSIGNED_INT_24_8;\n }\n else if (internalTexture.format === 14) {\n type = gl.FLOAT;\n }\n else if (internalTexture.format === 18) {\n type = gl.FLOAT_32_UNSIGNED_INT_24_8_REV;\n }\n const format = hasStencil ? gl.DEPTH_STENCIL : gl.DEPTH_COMPONENT;\n let internalFormat = format;\n if (this.webGLVersion > 1) {\n if (internalTexture.format === 15) {\n internalFormat = gl.DEPTH_COMPONENT16;\n }\n else if (internalTexture.format === 16) {\n internalFormat = gl.DEPTH_COMPONENT24;\n }\n else if (internalTexture.format === 17 || internalTexture.format === 13) {\n internalFormat = gl.DEPTH24_STENCIL8;\n }\n else if (internalTexture.format === 14) {\n internalFormat = gl.DEPTH_COMPONENT32F;\n }\n else if (internalTexture.format === 18) {\n internalFormat = gl.DEPTH32F_STENCIL8;\n }\n }\n if (internalTexture.is2DArray) {\n gl.texImage3D(target, 0, internalFormat, internalTexture.width, internalTexture.height, layers, 0, format, type, null);\n }\n else {\n gl.texImage2D(target, 0, internalFormat, internalTexture.width, internalTexture.height, 0, format, type, null);\n }\n this._bindTextureDirectly(target, null);\n this._internalTexturesCache.push(internalTexture);\n // Dispose previous depth/stencil render buffers and clear the corresponding attachment.\n // Next time this framebuffer is bound, the new depth/stencil texture will be attached.\n const glRtWrapper = rtWrapper;\n if (glRtWrapper._depthStencilBuffer) {\n const currentFrameBuffer = this._currentFramebuffer;\n this._bindUnboundFramebuffer(glRtWrapper._framebuffer);\n gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_STENCIL_ATTACHMENT, gl.RENDERBUFFER, null);\n gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_ATTACHMENT, gl.RENDERBUFFER, null);\n gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.STENCIL_ATTACHMENT, gl.RENDERBUFFER, null);\n this._bindUnboundFramebuffer(currentFrameBuffer);\n gl.deleteRenderbuffer(glRtWrapper._depthStencilBuffer);\n glRtWrapper._depthStencilBuffer = null;\n }\n return internalTexture;\n };\n ThinEngine.prototype.updateRenderTargetTextureSampleCount = function (rtWrapper, samples) {\n if (this.webGLVersion < 2 || !rtWrapper || !rtWrapper.texture) {\n return 1;\n }\n if (rtWrapper.samples === samples) {\n return samples;\n }\n const gl = this._gl;\n samples = Math.min(samples, this.getCaps().maxMSAASamples);\n // Dispose previous render buffers\n if (rtWrapper._depthStencilBuffer) {\n gl.deleteRenderbuffer(rtWrapper._depthStencilBuffer);\n rtWrapper._depthStencilBuffer = null;\n }\n if (rtWrapper._MSAAFramebuffer) {\n gl.deleteFramebuffer(rtWrapper._MSAAFramebuffer);\n rtWrapper._MSAAFramebuffer = null;\n }\n const hardwareTexture = rtWrapper.texture._hardwareTexture;\n hardwareTexture.releaseMSAARenderBuffers();\n if (samples > 1 && typeof gl.renderbufferStorageMultisample === \"function\") {\n const framebuffer = gl.createFramebuffer();\n if (!framebuffer) {\n throw new Error(\"Unable to create multi sampled framebuffer\");\n }\n rtWrapper._MSAAFramebuffer = framebuffer;\n this._bindUnboundFramebuffer(rtWrapper._MSAAFramebuffer);\n const colorRenderbuffer = this._createRenderBuffer(rtWrapper.texture.width, rtWrapper.texture.height, samples, -1 /* not used */, this._getRGBAMultiSampleBufferFormat(rtWrapper.texture.type), gl.COLOR_ATTACHMENT0, false);\n if (!colorRenderbuffer) {\n throw new Error(\"Unable to create multi sampled framebuffer\");\n }\n hardwareTexture.addMSAARenderBuffer(colorRenderbuffer);\n }\n else {\n this._bindUnboundFramebuffer(rtWrapper._framebuffer);\n }\n rtWrapper.texture.samples = samples;\n rtWrapper._samples = samples;\n rtWrapper._depthStencilBuffer = this._setupFramebufferDepthAttachments(rtWrapper._generateStencilBuffer, rtWrapper._generateDepthBuffer, rtWrapper.texture.width, rtWrapper.texture.height, samples);\n this._bindUnboundFramebuffer(null);\n return samples;\n };\n\n /**\n * PostProcess can be used to apply a shader to a texture after it has been rendered\n * See https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/usePostProcesses\n */\n class PostProcess {\n /**\n * Registers a shader code processing with a post process name.\n * @param postProcessName name of the post process. Use null for the fallback shader code processing. This is the shader code processing that will be used in case no specific shader code processing has been associated to a post process name\n * @param customShaderCodeProcessing shader code processing to associate to the post process name\n * @returns\n */\n static RegisterShaderCodeProcessing(postProcessName, customShaderCodeProcessing) {\n if (!customShaderCodeProcessing) {\n delete PostProcess._CustomShaderCodeProcessing[postProcessName !== null && postProcessName !== void 0 ? postProcessName : \"\"];\n return;\n }\n PostProcess._CustomShaderCodeProcessing[postProcessName !== null && postProcessName !== void 0 ? postProcessName : \"\"] = customShaderCodeProcessing;\n }\n static _GetShaderCodeProcessing(postProcessName) {\n var _a;\n return (_a = PostProcess._CustomShaderCodeProcessing[postProcessName]) !== null && _a !== void 0 ? _a : PostProcess._CustomShaderCodeProcessing[\"\"];\n }\n /**\n * Number of sample textures (default: 1)\n */\n get samples() {\n return this._samples;\n }\n set samples(n) {\n this._samples = Math.min(n, this._engine.getCaps().maxMSAASamples);\n this._textures.forEach((texture) => {\n texture.setSamples(this._samples);\n });\n }\n /**\n * Returns the fragment url or shader name used in the post process.\n * @returns the fragment url or name in the shader store.\n */\n getEffectName() {\n return this._fragmentUrl;\n }\n /**\n * A function that is added to the onActivateObservable\n */\n set onActivate(callback) {\n if (this._onActivateObserver) {\n this.onActivateObservable.remove(this._onActivateObserver);\n }\n if (callback) {\n this._onActivateObserver = this.onActivateObservable.add(callback);\n }\n }\n /**\n * A function that is added to the onSizeChangedObservable\n */\n set onSizeChanged(callback) {\n if (this._onSizeChangedObserver) {\n this.onSizeChangedObservable.remove(this._onSizeChangedObserver);\n }\n this._onSizeChangedObserver = this.onSizeChangedObservable.add(callback);\n }\n /**\n * A function that is added to the onApplyObservable\n */\n set onApply(callback) {\n if (this._onApplyObserver) {\n this.onApplyObservable.remove(this._onApplyObserver);\n }\n this._onApplyObserver = this.onApplyObservable.add(callback);\n }\n /**\n * A function that is added to the onBeforeRenderObservable\n */\n set onBeforeRender(callback) {\n if (this._onBeforeRenderObserver) {\n this.onBeforeRenderObservable.remove(this._onBeforeRenderObserver);\n }\n this._onBeforeRenderObserver = this.onBeforeRenderObservable.add(callback);\n }\n /**\n * A function that is added to the onAfterRenderObservable\n */\n set onAfterRender(callback) {\n if (this._onAfterRenderObserver) {\n this.onAfterRenderObservable.remove(this._onAfterRenderObserver);\n }\n this._onAfterRenderObserver = this.onAfterRenderObservable.add(callback);\n }\n /**\n * The input texture for this post process and the output texture of the previous post process. When added to a pipeline the previous post process will\n * render it's output into this texture and this texture will be used as textureSampler in the fragment shader of this post process.\n */\n get inputTexture() {\n return this._textures.data[this._currentRenderTextureInd];\n }\n set inputTexture(value) {\n this._forcedOutputTexture = value;\n }\n /**\n * Since inputTexture should always be defined, if we previously manually set `inputTexture`,\n * the only way to unset it is to use this function to restore its internal state\n */\n restoreDefaultInputTexture() {\n if (this._forcedOutputTexture) {\n this._forcedOutputTexture = null;\n this.markTextureDirty();\n }\n }\n /**\n * Gets the camera which post process is applied to.\n * @returns The camera the post process is applied to.\n */\n getCamera() {\n return this._camera;\n }\n /**\n * Gets the texel size of the postprocess.\n * See https://en.wikipedia.org/wiki/Texel_(graphics)\n */\n get texelSize() {\n if (this._shareOutputWithPostProcess) {\n return this._shareOutputWithPostProcess.texelSize;\n }\n if (this._forcedOutputTexture) {\n this._texelSize.copyFromFloats(1.0 / this._forcedOutputTexture.width, 1.0 / this._forcedOutputTexture.height);\n }\n return this._texelSize;\n }\n /**\n * Creates a new instance PostProcess\n * @param name The name of the PostProcess.\n * @param fragmentUrl The url of the fragment shader to be used.\n * @param parameters Array of the names of uniform non-sampler2D variables that will be passed to the shader.\n * @param samplers Array of the names of uniform sampler2D variables that will be passed to the shader.\n * @param options The required width/height ratio to downsize to before computing the render pass. (Use 1.0 for full size)\n * @param camera The camera to apply the render pass to.\n * @param samplingMode The sampling mode to be used when computing the pass. (default: 0)\n * @param engine The engine which the post process will be applied. (default: current engine)\n * @param reusable If the post process can be reused on the same frame. (default: false)\n * @param defines String of defines that will be set when running the fragment shader. (default: null)\n * @param textureType Type of textures used when performing the post process. (default: 0)\n * @param vertexUrl The url of the vertex shader to be used. (default: \"postprocess\")\n * @param indexParameters The index parameters to be used for babylons include syntax \"#include[0..varyingCount]\". (default: undefined) See usage in babylon.blurPostProcess.ts and kernelBlur.vertex.fx\n * @param blockCompilation If the shader should not be compiled immediatly. (default: false)\n * @param textureFormat Format of textures used when performing the post process. (default: TEXTUREFORMAT_RGBA)\n */\n constructor(name, fragmentUrl, parameters, samplers, options, camera, samplingMode = 1, engine, reusable, defines = null, textureType = 0, vertexUrl = \"postprocess\", indexParameters, blockCompilation = false, textureFormat = 5, shaderLanguage = ShaderLanguage.GLSL) {\n /** @internal */\n this._parentContainer = null;\n /**\n * Width of the texture to apply the post process on\n */\n this.width = -1;\n /**\n * Height of the texture to apply the post process on\n */\n this.height = -1;\n /**\n * Gets the node material used to create this postprocess (null if the postprocess was manually created)\n */\n this.nodeMaterialSource = null;\n /**\n * Internal, reference to the location where this postprocess was output to. (Typically the texture on the next postprocess in the chain)\n * @internal\n */\n this._outputTexture = null;\n /**\n * If the buffer needs to be cleared before applying the post process. (default: true)\n * Should be set to false if shader will overwrite all previous pixels.\n */\n this.autoClear = true;\n /**\n * If clearing the buffer should be forced in autoClear mode, even when alpha mode is enabled (default: false).\n * By default, the buffer will only be cleared if alpha mode is disabled (and autoClear is true).\n */\n this.forceAutoClearInAlphaMode = false;\n /**\n * Type of alpha mode to use when performing the post process (default: Engine.ALPHA_DISABLE)\n */\n this.alphaMode = 0;\n /**\n * Animations to be used for the post processing\n */\n this.animations = new Array();\n /**\n * Enable Pixel Perfect mode where texture is not scaled to be power of 2.\n * Can only be used on a single postprocess or on the last one of a chain. (default: false)\n */\n this.enablePixelPerfectMode = false;\n /**\n * Force the postprocess to be applied without taking in account viewport\n */\n this.forceFullscreenViewport = true;\n /**\n * Scale mode for the post process (default: Engine.SCALEMODE_FLOOR)\n *\n * | Value | Type | Description |\n * | ----- | ----------------------------------- | ----------- |\n * | 1 | SCALEMODE_FLOOR | [engine.scalemode_floor](https://doc.babylonjs.com/api/classes/babylon.engine#scalemode_floor) |\n * | 2 | SCALEMODE_NEAREST | [engine.scalemode_nearest](https://doc.babylonjs.com/api/classes/babylon.engine#scalemode_nearest) |\n * | 3 | SCALEMODE_CEILING | [engine.scalemode_ceiling](https://doc.babylonjs.com/api/classes/babylon.engine#scalemode_ceiling) |\n *\n */\n this.scaleMode = 1;\n /**\n * Force textures to be a power of two (default: false)\n */\n this.alwaysForcePOT = false;\n this._samples = 1;\n /**\n * Modify the scale of the post process to be the same as the viewport (default: false)\n */\n this.adaptScaleToCurrentViewport = false;\n this._reusable = false;\n this._renderId = 0;\n /**\n * if externalTextureSamplerBinding is true, the \"apply\" method won't bind the textureSampler texture, it is expected to be done by the \"outside\" (by the onApplyObservable observer most probably).\n * counter-productive in some cases because if the texture bound by \"apply\" is different from the currently texture bound, (the one set by the onApplyObservable observer, for eg) some\n * internal structures (materialContext) will be dirtified, which may impact performances\n */\n this.externalTextureSamplerBinding = false;\n /**\n * Smart array of input and output textures for the post process.\n * @internal\n */\n this._textures = new SmartArray(2);\n /**\n * Smart array of input and output textures for the post process.\n * @internal\n */\n this._textureCache = [];\n /**\n * The index in _textures that corresponds to the output texture.\n * @internal\n */\n this._currentRenderTextureInd = 0;\n this._scaleRatio = new Vector2(1, 1);\n this._texelSize = Vector2.Zero();\n // Events\n /**\n * An event triggered when the postprocess is activated.\n */\n this.onActivateObservable = new Observable$1();\n /**\n * An event triggered when the postprocess changes its size.\n */\n this.onSizeChangedObservable = new Observable$1();\n /**\n * An event triggered when the postprocess applies its effect.\n */\n this.onApplyObservable = new Observable$1();\n /**\n * An event triggered before rendering the postprocess\n */\n this.onBeforeRenderObservable = new Observable$1();\n /**\n * An event triggered after rendering the postprocess\n */\n this.onAfterRenderObservable = new Observable$1();\n this.name = name;\n if (camera != null) {\n this._camera = camera;\n this._scene = camera.getScene();\n camera.attachPostProcess(this);\n this._engine = this._scene.getEngine();\n this._scene.postProcesses.push(this);\n this.uniqueId = this._scene.getUniqueId();\n }\n else if (engine) {\n this._engine = engine;\n this._engine.postProcesses.push(this);\n }\n this._options = options;\n this.renderTargetSamplingMode = samplingMode ? samplingMode : 1;\n this._reusable = reusable || false;\n this._textureType = textureType;\n this._textureFormat = textureFormat;\n this._shaderLanguage = shaderLanguage;\n this._samplers = samplers || [];\n this._samplers.push(\"textureSampler\");\n this._fragmentUrl = fragmentUrl;\n this._vertexUrl = vertexUrl;\n this._parameters = parameters || [];\n this._parameters.push(\"scale\");\n this._indexParameters = indexParameters;\n this._drawWrapper = new DrawWrapper(this._engine);\n if (!blockCompilation) {\n this.updateEffect(defines);\n }\n }\n /**\n * Gets a string identifying the name of the class\n * @returns \"PostProcess\" string\n */\n getClassName() {\n return \"PostProcess\";\n }\n /**\n * Gets the engine which this post process belongs to.\n * @returns The engine the post process was enabled with.\n */\n getEngine() {\n return this._engine;\n }\n /**\n * The effect that is created when initializing the post process.\n * @returns The created effect corresponding the the postprocess.\n */\n getEffect() {\n return this._drawWrapper.effect;\n }\n /**\n * To avoid multiple redundant textures for multiple post process, the output the output texture for this post process can be shared with another.\n * @param postProcess The post process to share the output with.\n * @returns This post process.\n */\n shareOutputWith(postProcess) {\n this._disposeTextures();\n this._shareOutputWithPostProcess = postProcess;\n return this;\n }\n /**\n * Reverses the effect of calling shareOutputWith and returns the post process back to its original state.\n * This should be called if the post process that shares output with this post process is disabled/disposed.\n */\n useOwnOutput() {\n if (this._textures.length == 0) {\n this._textures = new SmartArray(2);\n }\n this._shareOutputWithPostProcess = null;\n }\n /**\n * Updates the effect with the current post process compile time values and recompiles the shader.\n * @param defines Define statements that should be added at the beginning of the shader. (default: null)\n * @param uniforms Set of uniform variables that will be passed to the shader. (default: null)\n * @param samplers Set of Texture2D variables that will be passed to the shader. (default: null)\n * @param indexParameters The index parameters to be used for babylons include syntax \"#include[0..varyingCount]\". (default: undefined) See usage in babylon.blurPostProcess.ts and kernelBlur.vertex.fx\n * @param onCompiled Called when the shader has been compiled.\n * @param onError Called if there is an error when compiling a shader.\n * @param vertexUrl The url of the vertex shader to be used (default: the one given at construction time)\n * @param fragmentUrl The url of the fragment shader to be used (default: the one given at construction time)\n */\n updateEffect(defines = null, uniforms = null, samplers = null, indexParameters, onCompiled, onError, vertexUrl, fragmentUrl) {\n var _a, _b;\n const customShaderCodeProcessing = PostProcess._GetShaderCodeProcessing(this.name);\n if (customShaderCodeProcessing === null || customShaderCodeProcessing === void 0 ? void 0 : customShaderCodeProcessing.defineCustomBindings) {\n const newUniforms = (_a = uniforms === null || uniforms === void 0 ? void 0 : uniforms.slice()) !== null && _a !== void 0 ? _a : [];\n newUniforms.push(...this._parameters);\n const newSamplers = (_b = samplers === null || samplers === void 0 ? void 0 : samplers.slice()) !== null && _b !== void 0 ? _b : [];\n newSamplers.push(...this._samplers);\n defines = customShaderCodeProcessing.defineCustomBindings(this.name, defines, newUniforms, newSamplers);\n uniforms = newUniforms;\n samplers = newSamplers;\n }\n this._postProcessDefines = defines;\n this._drawWrapper.effect = this._engine.createEffect({ vertex: vertexUrl !== null && vertexUrl !== void 0 ? vertexUrl : this._vertexUrl, fragment: fragmentUrl !== null && fragmentUrl !== void 0 ? fragmentUrl : this._fragmentUrl }, {\n attributes: [\"position\"],\n uniformsNames: uniforms || this._parameters,\n uniformBuffersNames: [],\n samplers: samplers || this._samplers,\n defines: defines !== null ? defines : \"\",\n fallbacks: null,\n onCompiled: onCompiled !== null && onCompiled !== void 0 ? onCompiled : null,\n onError: onError !== null && onError !== void 0 ? onError : null,\n indexParameters: indexParameters || this._indexParameters,\n processCodeAfterIncludes: (customShaderCodeProcessing === null || customShaderCodeProcessing === void 0 ? void 0 : customShaderCodeProcessing.processCodeAfterIncludes)\n ? (shaderType, code) => customShaderCodeProcessing.processCodeAfterIncludes(this.name, shaderType, code)\n : null,\n processFinalCode: (customShaderCodeProcessing === null || customShaderCodeProcessing === void 0 ? void 0 : customShaderCodeProcessing.processFinalCode)\n ? (shaderType, code) => customShaderCodeProcessing.processFinalCode(this.name, shaderType, code)\n : null,\n shaderLanguage: this._shaderLanguage,\n }, this._engine);\n }\n /**\n * The post process is reusable if it can be used multiple times within one frame.\n * @returns If the post process is reusable\n */\n isReusable() {\n return this._reusable;\n }\n /** invalidate frameBuffer to hint the postprocess to create a depth buffer */\n markTextureDirty() {\n this.width = -1;\n }\n _createRenderTargetTexture(textureSize, textureOptions, channel = 0) {\n for (let i = 0; i < this._textureCache.length; i++) {\n if (this._textureCache[i].texture.width === textureSize.width &&\n this._textureCache[i].texture.height === textureSize.height &&\n this._textureCache[i].postProcessChannel === channel &&\n this._textureCache[i].texture._generateDepthBuffer === textureOptions.generateDepthBuffer &&\n this._textureCache[i].texture.samples === textureOptions.samples) {\n return this._textureCache[i].texture;\n }\n }\n const tex = this._engine.createRenderTargetTexture(textureSize, textureOptions);\n this._textureCache.push({ texture: tex, postProcessChannel: channel, lastUsedRenderId: -1 });\n return tex;\n }\n _flushTextureCache() {\n const currentRenderId = this._renderId;\n for (let i = this._textureCache.length - 1; i >= 0; i--) {\n if (currentRenderId - this._textureCache[i].lastUsedRenderId > 100) {\n let currentlyUsed = false;\n for (let j = 0; j < this._textures.length; j++) {\n if (this._textures.data[j] === this._textureCache[i].texture) {\n currentlyUsed = true;\n break;\n }\n }\n if (!currentlyUsed) {\n this._textureCache[i].texture.dispose();\n this._textureCache.splice(i, 1);\n }\n }\n }\n }\n _resize(width, height, camera, needMipMaps, forceDepthStencil) {\n if (this._textures.length > 0) {\n this._textures.reset();\n }\n this.width = width;\n this.height = height;\n let firstPP = null;\n for (let i = 0; i < camera._postProcesses.length; i++) {\n if (camera._postProcesses[i] !== null) {\n firstPP = camera._postProcesses[i];\n break;\n }\n }\n const textureSize = { width: this.width, height: this.height };\n const textureOptions = {\n generateMipMaps: needMipMaps,\n generateDepthBuffer: forceDepthStencil || firstPP === this,\n generateStencilBuffer: (forceDepthStencil || firstPP === this) && this._engine.isStencilEnable,\n samplingMode: this.renderTargetSamplingMode,\n type: this._textureType,\n format: this._textureFormat,\n samples: this._samples,\n label: \"PostProcessRTT-\" + this.name,\n };\n this._textures.push(this._createRenderTargetTexture(textureSize, textureOptions, 0));\n if (this._reusable) {\n this._textures.push(this._createRenderTargetTexture(textureSize, textureOptions, 1));\n }\n this._texelSize.copyFromFloats(1.0 / this.width, 1.0 / this.height);\n this.onSizeChangedObservable.notifyObservers(this);\n }\n /**\n * Activates the post process by intializing the textures to be used when executed. Notifies onActivateObservable.\n * When this post process is used in a pipeline, this is call will bind the input texture of this post process to the output of the previous.\n * @param camera The camera that will be used in the post process. This camera will be used when calling onActivateObservable.\n * @param sourceTexture The source texture to be inspected to get the width and height if not specified in the post process constructor. (default: null)\n * @param forceDepthStencil If true, a depth and stencil buffer will be generated. (default: false)\n * @returns The render target wrapper that was bound to be written to.\n */\n activate(camera, sourceTexture = null, forceDepthStencil) {\n var _a, _b;\n camera = camera || this._camera;\n const scene = camera.getScene();\n const engine = scene.getEngine();\n const maxSize = engine.getCaps().maxTextureSize;\n let requiredWidth = ((sourceTexture ? sourceTexture.width : this._engine.getRenderWidth(true)) * this._options) | 0;\n const requiredHeight = ((sourceTexture ? sourceTexture.height : this._engine.getRenderHeight(true)) * this._options) | 0;\n // If rendering to a webvr camera's left or right eye only half the width should be used to avoid resize when rendered to screen\n const webVRCamera = camera.parent;\n if (webVRCamera && (webVRCamera.leftCamera == camera || webVRCamera.rightCamera == camera)) {\n requiredWidth /= 2;\n }\n let desiredWidth = this._options.width || requiredWidth;\n let desiredHeight = this._options.height || requiredHeight;\n const needMipMaps = this.renderTargetSamplingMode !== 7 &&\n this.renderTargetSamplingMode !== 1 &&\n this.renderTargetSamplingMode !== 2;\n if (!this._shareOutputWithPostProcess && !this._forcedOutputTexture) {\n if (this.adaptScaleToCurrentViewport) {\n const currentViewport = engine.currentViewport;\n if (currentViewport) {\n desiredWidth *= currentViewport.width;\n desiredHeight *= currentViewport.height;\n }\n }\n if (needMipMaps || this.alwaysForcePOT) {\n if (!this._options.width) {\n desiredWidth = engine.needPOTTextures ? Engine.GetExponentOfTwo(desiredWidth, maxSize, this.scaleMode) : desiredWidth;\n }\n if (!this._options.height) {\n desiredHeight = engine.needPOTTextures ? Engine.GetExponentOfTwo(desiredHeight, maxSize, this.scaleMode) : desiredHeight;\n }\n }\n if (this.width !== desiredWidth || this.height !== desiredHeight) {\n this._resize(desiredWidth, desiredHeight, camera, needMipMaps, forceDepthStencil);\n }\n this._textures.forEach((texture) => {\n if (texture.samples !== this.samples) {\n this._engine.updateRenderTargetTextureSampleCount(texture, this.samples);\n }\n });\n this._flushTextureCache();\n this._renderId++;\n }\n let target;\n if (this._shareOutputWithPostProcess) {\n target = this._shareOutputWithPostProcess.inputTexture;\n }\n else if (this._forcedOutputTexture) {\n target = this._forcedOutputTexture;\n this.width = this._forcedOutputTexture.width;\n this.height = this._forcedOutputTexture.height;\n }\n else {\n target = this.inputTexture;\n let cache;\n for (let i = 0; i < this._textureCache.length; i++) {\n if (this._textureCache[i].texture === target) {\n cache = this._textureCache[i];\n break;\n }\n }\n if (cache) {\n cache.lastUsedRenderId = this._renderId;\n }\n }\n // Bind the input of this post process to be used as the output of the previous post process.\n if (this.enablePixelPerfectMode) {\n this._scaleRatio.copyFromFloats(requiredWidth / desiredWidth, requiredHeight / desiredHeight);\n this._engine.bindFramebuffer(target, 0, requiredWidth, requiredHeight, this.forceFullscreenViewport);\n }\n else {\n this._scaleRatio.copyFromFloats(1, 1);\n this._engine.bindFramebuffer(target, 0, undefined, undefined, this.forceFullscreenViewport);\n }\n (_b = (_a = this._engine)._debugInsertMarker) === null || _b === void 0 ? void 0 : _b.call(_a, `post process ${this.name} input`);\n this.onActivateObservable.notifyObservers(camera);\n // Clear\n if (this.autoClear && (this.alphaMode === 0 || this.forceAutoClearInAlphaMode)) {\n this._engine.clear(this.clearColor ? this.clearColor : scene.clearColor, scene._allowPostProcessClearColor, true, true);\n }\n if (this._reusable) {\n this._currentRenderTextureInd = (this._currentRenderTextureInd + 1) % 2;\n }\n return target;\n }\n /**\n * If the post process is supported.\n */\n get isSupported() {\n return this._drawWrapper.effect.isSupported;\n }\n /**\n * The aspect ratio of the output texture.\n */\n get aspectRatio() {\n if (this._shareOutputWithPostProcess) {\n return this._shareOutputWithPostProcess.aspectRatio;\n }\n if (this._forcedOutputTexture) {\n return this._forcedOutputTexture.width / this._forcedOutputTexture.height;\n }\n return this.width / this.height;\n }\n /**\n * Get a value indicating if the post-process is ready to be used\n * @returns true if the post-process is ready (shader is compiled)\n */\n isReady() {\n var _a, _b;\n return (_b = (_a = this._drawWrapper.effect) === null || _a === void 0 ? void 0 : _a.isReady()) !== null && _b !== void 0 ? _b : false;\n }\n /**\n * Binds all textures and uniforms to the shader, this will be run on every pass.\n * @returns the effect corresponding to this post process. Null if not compiled or not ready.\n */\n apply() {\n var _a, _b, _c;\n // Check\n if (!((_a = this._drawWrapper.effect) === null || _a === void 0 ? void 0 : _a.isReady())) {\n return null;\n }\n // States\n this._engine.enableEffect(this._drawWrapper);\n this._engine.setState(false);\n this._engine.setDepthBuffer(false);\n this._engine.setDepthWrite(false);\n // Alpha\n this._engine.setAlphaMode(this.alphaMode);\n if (this.alphaConstants) {\n this.getEngine().setAlphaConstants(this.alphaConstants.r, this.alphaConstants.g, this.alphaConstants.b, this.alphaConstants.a);\n }\n // Bind the output texture of the preivous post process as the input to this post process.\n let source;\n if (this._shareOutputWithPostProcess) {\n source = this._shareOutputWithPostProcess.inputTexture;\n }\n else if (this._forcedOutputTexture) {\n source = this._forcedOutputTexture;\n }\n else {\n source = this.inputTexture;\n }\n if (!this.externalTextureSamplerBinding) {\n this._drawWrapper.effect._bindTexture(\"textureSampler\", source === null || source === void 0 ? void 0 : source.texture);\n }\n // Parameters\n this._drawWrapper.effect.setVector2(\"scale\", this._scaleRatio);\n this.onApplyObservable.notifyObservers(this._drawWrapper.effect);\n (_c = (_b = PostProcess._GetShaderCodeProcessing(this.name)) === null || _b === void 0 ? void 0 : _b.bindCustomBindings) === null || _c === void 0 ? void 0 : _c.call(_b, this.name, this._drawWrapper.effect);\n return this._drawWrapper.effect;\n }\n _disposeTextures() {\n if (this._shareOutputWithPostProcess || this._forcedOutputTexture) {\n this._disposeTextureCache();\n return;\n }\n this._disposeTextureCache();\n this._textures.dispose();\n }\n _disposeTextureCache() {\n for (let i = this._textureCache.length - 1; i >= 0; i--) {\n this._textureCache[i].texture.dispose();\n }\n this._textureCache.length = 0;\n }\n /**\n * Sets the required values to the prepass renderer.\n * @param prePassRenderer defines the prepass renderer to setup.\n * @returns true if the pre pass is needed.\n */\n setPrePassRenderer(prePassRenderer) {\n if (this._prePassEffectConfiguration) {\n this._prePassEffectConfiguration = prePassRenderer.addEffectConfiguration(this._prePassEffectConfiguration);\n this._prePassEffectConfiguration.enabled = true;\n return true;\n }\n return false;\n }\n /**\n * Disposes the post process.\n * @param camera The camera to dispose the post process on.\n */\n dispose(camera) {\n camera = camera || this._camera;\n this._disposeTextures();\n let index;\n if (this._scene) {\n index = this._scene.postProcesses.indexOf(this);\n if (index !== -1) {\n this._scene.postProcesses.splice(index, 1);\n }\n }\n if (this._parentContainer) {\n const index = this._parentContainer.postProcesses.indexOf(this);\n if (index > -1) {\n this._parentContainer.postProcesses.splice(index, 1);\n }\n this._parentContainer = null;\n }\n index = this._engine.postProcesses.indexOf(this);\n if (index !== -1) {\n this._engine.postProcesses.splice(index, 1);\n }\n if (!camera) {\n return;\n }\n camera.detachPostProcess(this);\n index = camera._postProcesses.indexOf(this);\n if (index === 0 && camera._postProcesses.length > 0) {\n const firstPostProcess = this._camera._getFirstPostProcess();\n if (firstPostProcess) {\n firstPostProcess.markTextureDirty();\n }\n }\n this.onActivateObservable.clear();\n this.onAfterRenderObservable.clear();\n this.onApplyObservable.clear();\n this.onBeforeRenderObservable.clear();\n this.onSizeChangedObservable.clear();\n }\n /**\n * Serializes the post process to a JSON object\n * @returns the JSON object\n */\n serialize() {\n const serializationObject = SerializationHelper.Serialize(this);\n const camera = this.getCamera() || (this._scene && this._scene.activeCamera);\n serializationObject.customType = \"BABYLON.\" + this.getClassName();\n serializationObject.cameraId = camera ? camera.id : null;\n serializationObject.reusable = this._reusable;\n serializationObject.textureType = this._textureType;\n serializationObject.fragmentUrl = this._fragmentUrl;\n serializationObject.parameters = this._parameters;\n serializationObject.samplers = this._samplers;\n serializationObject.options = this._options;\n serializationObject.defines = this._postProcessDefines;\n serializationObject.textureFormat = this._textureFormat;\n serializationObject.vertexUrl = this._vertexUrl;\n serializationObject.indexParameters = this._indexParameters;\n return serializationObject;\n }\n /**\n * Clones this post process\n * @returns a new post process similar to this one\n */\n clone() {\n const serializationObject = this.serialize();\n serializationObject._engine = this._engine;\n serializationObject.cameraId = null;\n const result = PostProcess.Parse(serializationObject, this._scene, \"\");\n if (!result) {\n return null;\n }\n result.onActivateObservable = this.onActivateObservable.clone();\n result.onSizeChangedObservable = this.onSizeChangedObservable.clone();\n result.onApplyObservable = this.onApplyObservable.clone();\n result.onBeforeRenderObservable = this.onBeforeRenderObservable.clone();\n result.onAfterRenderObservable = this.onAfterRenderObservable.clone();\n result._prePassEffectConfiguration = this._prePassEffectConfiguration;\n return result;\n }\n /**\n * Creates a material from parsed material data\n * @param parsedPostProcess defines parsed post process data\n * @param scene defines the hosting scene\n * @param rootUrl defines the root URL to use to load textures\n * @returns a new post process\n */\n static Parse(parsedPostProcess, scene, rootUrl) {\n const postProcessType = GetClass(parsedPostProcess.customType);\n if (!postProcessType || !postProcessType._Parse) {\n return null;\n }\n const camera = scene ? scene.getCameraById(parsedPostProcess.cameraId) : null;\n return postProcessType._Parse(parsedPostProcess, camera, scene, rootUrl);\n }\n /**\n * @internal\n */\n static _Parse(parsedPostProcess, targetCamera, scene, rootUrl) {\n return SerializationHelper.Parse(() => {\n return new PostProcess(parsedPostProcess.name, parsedPostProcess.fragmentUrl, parsedPostProcess.parameters, parsedPostProcess.samplers, parsedPostProcess.options, targetCamera, parsedPostProcess.renderTargetSamplingMode, parsedPostProcess._engine, parsedPostProcess.reusable, parsedPostProcess.defines, parsedPostProcess.textureType, parsedPostProcess.vertexUrl, parsedPostProcess.indexParameters, false, parsedPostProcess.textureFormat);\n }, parsedPostProcess, scene, rootUrl);\n }\n }\n PostProcess._CustomShaderCodeProcessing = {};\n __decorate$1([\n serialize()\n ], PostProcess.prototype, \"uniqueId\", void 0);\n __decorate$1([\n serialize()\n ], PostProcess.prototype, \"name\", void 0);\n __decorate$1([\n serialize()\n ], PostProcess.prototype, \"width\", void 0);\n __decorate$1([\n serialize()\n ], PostProcess.prototype, \"height\", void 0);\n __decorate$1([\n serialize()\n ], PostProcess.prototype, \"renderTargetSamplingMode\", void 0);\n __decorate$1([\n serializeAsColor4()\n ], PostProcess.prototype, \"clearColor\", void 0);\n __decorate$1([\n serialize()\n ], PostProcess.prototype, \"autoClear\", void 0);\n __decorate$1([\n serialize()\n ], PostProcess.prototype, \"forceAutoClearInAlphaMode\", void 0);\n __decorate$1([\n serialize()\n ], PostProcess.prototype, \"alphaMode\", void 0);\n __decorate$1([\n serialize()\n ], PostProcess.prototype, \"alphaConstants\", void 0);\n __decorate$1([\n serialize()\n ], PostProcess.prototype, \"enablePixelPerfectMode\", void 0);\n __decorate$1([\n serialize()\n ], PostProcess.prototype, \"forceFullscreenViewport\", void 0);\n __decorate$1([\n serialize()\n ], PostProcess.prototype, \"scaleMode\", void 0);\n __decorate$1([\n serialize()\n ], PostProcess.prototype, \"alwaysForcePOT\", void 0);\n __decorate$1([\n serialize(\"samples\")\n ], PostProcess.prototype, \"_samples\", void 0);\n __decorate$1([\n serialize()\n ], PostProcess.prototype, \"adaptScaleToCurrentViewport\", void 0);\n RegisterClass(\"BABYLON.PostProcess\", PostProcess);\n\n // Do not edit.\n const name$10 = \"rgbdDecodePixelShader\";\n const shader$10 = `varying vec2 vUV;uniform sampler2D textureSampler;\n#include\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) \n{gl_FragColor=vec4(fromRGBD(texture2D(textureSampler,vUV)),1.0);}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$10] = shader$10;\n\n ThinEngine.prototype.createRenderTargetCubeTexture = function (size, options) {\n const rtWrapper = this._createHardwareRenderTargetWrapper(false, true, size);\n const fullOptions = Object.assign({ generateMipMaps: true, generateDepthBuffer: true, generateStencilBuffer: false, type: 0, samplingMode: 3, format: 5 }, options);\n fullOptions.generateStencilBuffer = fullOptions.generateDepthBuffer && fullOptions.generateStencilBuffer;\n if (fullOptions.type === 1 && !this._caps.textureFloatLinearFiltering) {\n // if floating point linear (gl.FLOAT) then force to NEAREST_SAMPLINGMODE\n fullOptions.samplingMode = 1;\n }\n else if (fullOptions.type === 2 && !this._caps.textureHalfFloatLinearFiltering) {\n // if floating point linear (HALF_FLOAT) then force to NEAREST_SAMPLINGMODE\n fullOptions.samplingMode = 1;\n }\n const gl = this._gl;\n const texture = new InternalTexture(this, InternalTextureSource.RenderTarget);\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, texture, true);\n const filters = this._getSamplingParameters(fullOptions.samplingMode, fullOptions.generateMipMaps);\n if (fullOptions.type === 1 && !this._caps.textureFloat) {\n fullOptions.type = 0;\n Logger.Warn(\"Float textures are not supported. Cube render target forced to TEXTURETYPE_UNESIGNED_BYTE type\");\n }\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_MAG_FILTER, filters.mag);\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_MIN_FILTER, filters.min);\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n for (let face = 0; face < 6; face++) {\n gl.texImage2D(gl.TEXTURE_CUBE_MAP_POSITIVE_X + face, 0, this._getRGBABufferInternalSizedFormat(fullOptions.type, fullOptions.format), size, size, 0, this._getInternalFormat(fullOptions.format), this._getWebGLTextureType(fullOptions.type), null);\n }\n // Create the framebuffer\n const framebuffer = gl.createFramebuffer();\n this._bindUnboundFramebuffer(framebuffer);\n rtWrapper._depthStencilBuffer = this._setupFramebufferDepthAttachments(fullOptions.generateStencilBuffer, fullOptions.generateDepthBuffer, size, size);\n // MipMaps\n if (fullOptions.generateMipMaps) {\n gl.generateMipmap(gl.TEXTURE_CUBE_MAP);\n }\n // Unbind\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, null);\n this._bindUnboundFramebuffer(null);\n rtWrapper._framebuffer = framebuffer;\n rtWrapper._generateDepthBuffer = fullOptions.generateDepthBuffer;\n rtWrapper._generateStencilBuffer = fullOptions.generateStencilBuffer;\n texture.width = size;\n texture.height = size;\n texture.isReady = true;\n texture.isCube = true;\n texture.samples = 1;\n texture.generateMipMaps = fullOptions.generateMipMaps;\n texture.samplingMode = fullOptions.samplingMode;\n texture.type = fullOptions.type;\n texture.format = fullOptions.format;\n this._internalTexturesCache.push(texture);\n rtWrapper.setTextures(texture);\n return rtWrapper;\n };\n\n // Fullscreen quad buffers by default.\n const defaultOptions = {\n positions: [1, 1, -1, 1, -1, -1, 1, -1],\n indices: [0, 1, 2, 0, 2, 3],\n };\n /**\n * Helper class to render one or more effects.\n * You can access the previous rendering in your shader by declaring a sampler named textureSampler\n */\n class EffectRenderer {\n /**\n * Creates an effect renderer\n * @param engine the engine to use for rendering\n * @param options defines the options of the effect renderer\n */\n constructor(engine, options = defaultOptions) {\n var _a, _b;\n this._fullscreenViewport = new Viewport(0, 0, 1, 1);\n const positions = (_a = options.positions) !== null && _a !== void 0 ? _a : defaultOptions.positions;\n const indices = (_b = options.indices) !== null && _b !== void 0 ? _b : defaultOptions.indices;\n this.engine = engine;\n this._vertexBuffers = {\n [VertexBuffer.PositionKind]: new VertexBuffer(engine, positions, VertexBuffer.PositionKind, false, false, 2),\n };\n this._indexBuffer = engine.createIndexBuffer(indices);\n this._onContextRestoredObserver = engine.onContextRestoredObservable.add(() => {\n this._indexBuffer = engine.createIndexBuffer(indices);\n for (const key in this._vertexBuffers) {\n const vertexBuffer = this._vertexBuffers[key];\n vertexBuffer._rebuild();\n }\n });\n }\n /**\n * Sets the current viewport in normalized coordinates 0-1\n * @param viewport Defines the viewport to set (defaults to 0 0 1 1)\n */\n setViewport(viewport = this._fullscreenViewport) {\n this.engine.setViewport(viewport);\n }\n /**\n * Binds the embedded attributes buffer to the effect.\n * @param effect Defines the effect to bind the attributes for\n */\n bindBuffers(effect) {\n this.engine.bindBuffers(this._vertexBuffers, this._indexBuffer, effect);\n }\n /**\n * Sets the current effect wrapper to use during draw.\n * The effect needs to be ready before calling this api.\n * This also sets the default full screen position attribute.\n * @param effectWrapper Defines the effect to draw with\n */\n applyEffectWrapper(effectWrapper) {\n this.engine.setState(true);\n this.engine.depthCullingState.depthTest = false;\n this.engine.stencilState.stencilTest = false;\n this.engine.enableEffect(effectWrapper._drawWrapper);\n this.bindBuffers(effectWrapper.effect);\n effectWrapper.onApplyObservable.notifyObservers({});\n }\n /**\n * Saves engine states\n */\n saveStates() {\n this._savedStateDepthTest = this.engine.depthCullingState.depthTest;\n this._savedStateStencilTest = this.engine.stencilState.stencilTest;\n }\n /**\n * Restores engine states\n */\n restoreStates() {\n this.engine.depthCullingState.depthTest = this._savedStateDepthTest;\n this.engine.stencilState.stencilTest = this._savedStateStencilTest;\n }\n /**\n * Draws a full screen quad.\n */\n draw() {\n this.engine.drawElementsType(0, 0, 6);\n }\n _isRenderTargetTexture(texture) {\n return texture.renderTarget !== undefined;\n }\n /**\n * renders one or more effects to a specified texture\n * @param effectWrapper the effect to renderer\n * @param outputTexture texture to draw to, if null it will render to the screen.\n */\n render(effectWrapper, outputTexture = null) {\n // Ensure effect is ready\n if (!effectWrapper.effect.isReady()) {\n return;\n }\n this.saveStates();\n // Reset state\n this.setViewport();\n const out = outputTexture === null ? null : this._isRenderTargetTexture(outputTexture) ? outputTexture.renderTarget : outputTexture;\n if (out) {\n this.engine.bindFramebuffer(out);\n }\n this.applyEffectWrapper(effectWrapper);\n this.draw();\n if (out) {\n this.engine.unBindFramebuffer(out);\n }\n this.restoreStates();\n }\n /**\n * Disposes of the effect renderer\n */\n dispose() {\n const vertexBuffer = this._vertexBuffers[VertexBuffer.PositionKind];\n if (vertexBuffer) {\n vertexBuffer.dispose();\n delete this._vertexBuffers[VertexBuffer.PositionKind];\n }\n if (this._indexBuffer) {\n this.engine._releaseBuffer(this._indexBuffer);\n }\n if (this._onContextRestoredObserver) {\n this.engine.onContextRestoredObservable.remove(this._onContextRestoredObserver);\n this._onContextRestoredObserver = null;\n }\n }\n }\n /**\n * Wraps an effect to be used for rendering\n */\n class EffectWrapper {\n /**\n * The underlying effect\n */\n get effect() {\n return this._drawWrapper.effect;\n }\n set effect(effect) {\n this._drawWrapper.effect = effect;\n }\n /**\n * Creates an effect to be renderer\n * @param creationOptions options to create the effect\n */\n constructor(creationOptions) {\n /**\n * Event that is fired right before the effect is drawn (should be used to update uniforms)\n */\n this.onApplyObservable = new Observable$1();\n let effectCreationOptions;\n const uniformNames = creationOptions.uniformNames || [];\n if (creationOptions.vertexShader) {\n effectCreationOptions = {\n fragmentSource: creationOptions.fragmentShader,\n vertexSource: creationOptions.vertexShader,\n spectorName: creationOptions.name || \"effectWrapper\",\n };\n }\n else {\n // Default scale to use in post process vertex shader.\n uniformNames.push(\"scale\");\n effectCreationOptions = {\n fragmentSource: creationOptions.fragmentShader,\n vertex: \"postprocess\",\n spectorName: creationOptions.name || \"effectWrapper\",\n };\n // Sets the default scale to identity for the post process vertex shader.\n this.onApplyObservable.add(() => {\n this.effect.setFloat2(\"scale\", 1, 1);\n });\n }\n const defines = creationOptions.defines ? creationOptions.defines.join(\"\\n\") : \"\";\n this._drawWrapper = new DrawWrapper(creationOptions.engine);\n if (creationOptions.useShaderStore) {\n effectCreationOptions.fragment = effectCreationOptions.fragmentSource;\n if (!effectCreationOptions.vertex) {\n effectCreationOptions.vertex = effectCreationOptions.vertexSource;\n }\n delete effectCreationOptions.fragmentSource;\n delete effectCreationOptions.vertexSource;\n this.effect = creationOptions.engine.createEffect(effectCreationOptions, creationOptions.attributeNames || [\"position\"], uniformNames, creationOptions.samplerNames, defines, undefined, creationOptions.onCompiled, undefined, undefined, creationOptions.shaderLanguage);\n }\n else {\n this.effect = new Effect(effectCreationOptions, creationOptions.attributeNames || [\"position\"], uniformNames, creationOptions.samplerNames, creationOptions.engine, defines, undefined, creationOptions.onCompiled, undefined, undefined, undefined, creationOptions.shaderLanguage);\n this._onContextRestoredObserver = creationOptions.engine.onContextRestoredObservable.add(() => {\n this.effect._pipelineContext = null; // because _prepareEffect will try to dispose this pipeline before recreating it and that would lead to webgl errors\n this.effect._wasPreviouslyReady = false;\n this.effect._prepareEffect();\n });\n }\n }\n /**\n * Disposes of the effect wrapper\n */\n dispose() {\n if (this._onContextRestoredObserver) {\n this.effect.getEngine().onContextRestoredObservable.remove(this._onContextRestoredObserver);\n this._onContextRestoredObserver = null;\n }\n this.effect.dispose();\n }\n }\n\n // Do not edit.\n const name$11 = \"passPixelShader\";\n const shader$11 = `varying vec2 vUV;uniform sampler2D textureSampler;\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) \n{gl_FragColor=texture2D(textureSampler,vUV);}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$11] = shader$11;\n /** @internal */\n const passPixelShader = { name: name$11, shader: shader$11 };\n\n /**\n * Class containing a set of static utilities functions to dump data from a canvas\n */\n class DumpTools {\n static _CreateDumpRenderer() {\n if (!DumpTools._DumpToolsEngine) {\n let canvas = new OffscreenCanvas(100, 100); // will be resized later\n let engine = null;\n const options = {\n preserveDrawingBuffer: true,\n depth: false,\n stencil: false,\n alpha: true,\n premultipliedAlpha: false,\n antialias: false,\n failIfMajorPerformanceCaveat: false,\n };\n try {\n engine = new ThinEngine(canvas, false, options);\n }\n catch (e) {\n // The browser does not support WebGL context in OffscreenCanvas, fallback on a regular canvas\n canvas = document.createElement(\"canvas\");\n engine = new ThinEngine(canvas, false, options);\n }\n engine.getCaps().parallelShaderCompile = undefined;\n const renderer = new EffectRenderer(engine);\n const wrapper = new EffectWrapper({\n engine,\n name: passPixelShader.name,\n fragmentShader: passPixelShader.shader,\n samplerNames: [\"textureSampler\"],\n });\n DumpTools._DumpToolsEngine = {\n canvas,\n engine,\n renderer,\n wrapper,\n };\n }\n return DumpTools._DumpToolsEngine;\n }\n /**\n * Dumps the current bound framebuffer\n * @param width defines the rendering width\n * @param height defines the rendering height\n * @param engine defines the hosting engine\n * @param successCallback defines the callback triggered once the data are available\n * @param mimeType defines the mime type of the result\n * @param fileName defines the filename to download. If present, the result will automatically be downloaded\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\n * @returns a void promise\n */\n static async DumpFramebuffer(width, height, engine, successCallback, mimeType = \"image/png\", fileName, quality) {\n // Read the contents of the framebuffer\n const bufferView = await engine.readPixels(0, 0, width, height);\n const data = new Uint8Array(bufferView.buffer);\n DumpTools.DumpData(width, height, data, successCallback, mimeType, fileName, true, undefined, quality);\n }\n /**\n * Dumps an array buffer\n * @param width defines the rendering width\n * @param height defines the rendering height\n * @param data the data array\n * @param mimeType defines the mime type of the result\n * @param fileName defines the filename to download. If present, the result will automatically be downloaded\n * @param invertY true to invert the picture in the Y dimension\n * @param toArrayBuffer true to convert the data to an ArrayBuffer (encoded as `mimeType`) instead of a base64 string\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\n * @returns a promise that resolve to the final data\n */\n static DumpDataAsync(width, height, data, mimeType = \"image/png\", fileName, invertY = false, toArrayBuffer = false, quality) {\n return new Promise((resolve) => {\n DumpTools.DumpData(width, height, data, (result) => resolve(result), mimeType, fileName, invertY, toArrayBuffer, quality);\n });\n }\n /**\n * Dumps an array buffer\n * @param width defines the rendering width\n * @param height defines the rendering height\n * @param data the data array\n * @param successCallback defines the callback triggered once the data are available\n * @param mimeType defines the mime type of the result\n * @param fileName defines the filename to download. If present, the result will automatically be downloaded\n * @param invertY true to invert the picture in the Y dimension\n * @param toArrayBuffer true to convert the data to an ArrayBuffer (encoded as `mimeType`) instead of a base64 string\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\n */\n static DumpData(width, height, data, successCallback, mimeType = \"image/png\", fileName, invertY = false, toArrayBuffer = false, quality) {\n const renderer = DumpTools._CreateDumpRenderer();\n renderer.engine.setSize(width, height, true);\n // Convert if data are float32\n if (data instanceof Float32Array) {\n const data2 = new Uint8Array(data.length);\n let n = data.length;\n while (n--) {\n const v = data[n];\n data2[n] = Math.round(Scalar.Clamp(v) * 255);\n }\n data = data2;\n }\n // Create the image\n const texture = renderer.engine.createRawTexture(data, width, height, 5, false, !invertY, 1);\n renderer.renderer.setViewport();\n renderer.renderer.applyEffectWrapper(renderer.wrapper);\n renderer.wrapper.effect._bindTexture(\"textureSampler\", texture);\n renderer.renderer.draw();\n if (toArrayBuffer) {\n Tools.ToBlob(renderer.canvas, (blob) => {\n const fileReader = new FileReader();\n fileReader.onload = (event) => {\n const arrayBuffer = event.target.result;\n if (successCallback) {\n successCallback(arrayBuffer);\n }\n };\n fileReader.readAsArrayBuffer(blob);\n }, mimeType, quality);\n }\n else {\n Tools.EncodeScreenshotCanvasData(renderer.canvas, successCallback, mimeType, fileName, quality);\n }\n texture.dispose();\n }\n /**\n * Dispose the dump tools associated resources\n */\n static Dispose() {\n if (DumpTools._DumpToolsEngine) {\n DumpTools._DumpToolsEngine.wrapper.dispose();\n DumpTools._DumpToolsEngine.renderer.dispose();\n DumpTools._DumpToolsEngine.engine.dispose();\n }\n DumpTools._DumpToolsEngine = null;\n }\n }\n /**\n * This will be executed automatically for UMD and es5.\n * If esm dev wants the side effects to execute they will have to run it manually\n * Once we build native modules those need to be exported.\n * @internal\n */\n const initSideEffects$1 = () => {\n // References the dependencies.\n Tools.DumpData = DumpTools.DumpData;\n Tools.DumpDataAsync = DumpTools.DumpDataAsync;\n Tools.DumpFramebuffer = DumpTools.DumpFramebuffer;\n };\n initSideEffects$1();\n\n /**\n * This Helps creating a texture that will be created from a camera in your scene.\n * It is basically a dynamic texture that could be used to create special effects for instance.\n * Actually, It is the base of lot of effects in the framework like post process, shadows, effect layers and rendering pipelines...\n */\n class RenderTargetTexture extends Texture {\n /**\n * Use this list to define the list of mesh you want to render.\n */\n get renderList() {\n return this._renderList;\n }\n set renderList(value) {\n if (this._unObserveRenderList) {\n this._unObserveRenderList();\n this._unObserveRenderList = null;\n }\n if (value) {\n this._unObserveRenderList = _ObserveArray(value, this._renderListHasChanged);\n }\n this._renderList = value;\n }\n /**\n * Post-processes for this render target\n */\n get postProcesses() {\n return this._postProcesses;\n }\n get _prePassEnabled() {\n return !!this._prePassRenderTarget && this._prePassRenderTarget.enabled;\n }\n /**\n * Set a after unbind callback in the texture.\n * This has been kept for backward compatibility and use of onAfterUnbindObservable is recommended.\n */\n set onAfterUnbind(callback) {\n if (this._onAfterUnbindObserver) {\n this.onAfterUnbindObservable.remove(this._onAfterUnbindObserver);\n }\n this._onAfterUnbindObserver = this.onAfterUnbindObservable.add(callback);\n }\n /**\n * Set a before render callback in the texture.\n * This has been kept for backward compatibility and use of onBeforeRenderObservable is recommended.\n */\n set onBeforeRender(callback) {\n if (this._onBeforeRenderObserver) {\n this.onBeforeRenderObservable.remove(this._onBeforeRenderObserver);\n }\n this._onBeforeRenderObserver = this.onBeforeRenderObservable.add(callback);\n }\n /**\n * Set a after render callback in the texture.\n * This has been kept for backward compatibility and use of onAfterRenderObservable is recommended.\n */\n set onAfterRender(callback) {\n if (this._onAfterRenderObserver) {\n this.onAfterRenderObservable.remove(this._onAfterRenderObserver);\n }\n this._onAfterRenderObserver = this.onAfterRenderObservable.add(callback);\n }\n /**\n * Set a clear callback in the texture.\n * This has been kept for backward compatibility and use of onClearObservable is recommended.\n */\n set onClear(callback) {\n if (this._onClearObserver) {\n this.onClearObservable.remove(this._onClearObserver);\n }\n this._onClearObserver = this.onClearObservable.add(callback);\n }\n /**\n * Gets the render pass ids used by the render target texture. For a single render target the array length will be 1, for a cube texture it will be 6 and for\n * a 2D texture array it will return an array of ids the size of the 2D texture array\n */\n get renderPassIds() {\n return this._renderPassIds;\n }\n /**\n * Gets the current value of the refreshId counter\n */\n get currentRefreshId() {\n return this._currentRefreshId;\n }\n /**\n * Sets a specific material to be used to render a mesh/a list of meshes in this render target texture\n * @param mesh mesh or array of meshes\n * @param material material or array of materials to use for this render pass. If undefined is passed, no specific material will be used but the regular material instead (mesh.material). It's possible to provide an array of materials to use a different material for each rendering in the case of a cube texture (6 rendering) and a 2D texture array (as many rendering as the length of the array)\n */\n setMaterialForRendering(mesh, material) {\n let meshes;\n if (!Array.isArray(mesh)) {\n meshes = [mesh];\n }\n else {\n meshes = mesh;\n }\n for (let j = 0; j < meshes.length; ++j) {\n for (let i = 0; i < this._renderPassIds.length; ++i) {\n meshes[j].setMaterialForRenderPass(this._renderPassIds[i], material !== undefined ? (Array.isArray(material) ? material[i] : material) : undefined);\n }\n }\n }\n /**\n * Define if the texture has multiple draw buffers or if false a single draw buffer.\n */\n get isMulti() {\n var _a, _b;\n return (_b = (_a = this._renderTarget) === null || _a === void 0 ? void 0 : _a.isMulti) !== null && _b !== void 0 ? _b : false;\n }\n /**\n * Gets render target creation options that were used.\n */\n get renderTargetOptions() {\n return this._renderTargetOptions;\n }\n /**\n * Gets the render target wrapper associated with this render target\n */\n get renderTarget() {\n return this._renderTarget;\n }\n _onRatioRescale() {\n if (this._sizeRatio) {\n this.resize(this._initialSizeParameter);\n }\n }\n /**\n * Gets or sets the size of the bounding box associated with the texture (when in cube mode)\n * When defined, the cubemap will switch to local mode\n * @see https://community.arm.com/graphics/b/blog/posts/reflections-based-on-local-cubemaps-in-unity\n * @example https://www.babylonjs-playground.com/#RNASML\n */\n set boundingBoxSize(value) {\n if (this._boundingBoxSize && this._boundingBoxSize.equals(value)) {\n return;\n }\n this._boundingBoxSize = value;\n const scene = this.getScene();\n if (scene) {\n scene.markAllMaterialsAsDirty(1);\n }\n }\n get boundingBoxSize() {\n return this._boundingBoxSize;\n }\n /**\n * In case the RTT has been created with a depth texture, get the associated\n * depth texture.\n * Otherwise, return null.\n */\n get depthStencilTexture() {\n var _a, _b;\n return (_b = (_a = this._renderTarget) === null || _a === void 0 ? void 0 : _a._depthStencilTexture) !== null && _b !== void 0 ? _b : null;\n }\n /** @internal */\n constructor(name, size, scene, generateMipMaps = false, doNotChangeAspectRatio = true, type = 0, isCube = false, samplingMode = Texture.TRILINEAR_SAMPLINGMODE, generateDepthBuffer = true, generateStencilBuffer = false, isMulti = false, format = 5, delayAllocation = false, samples, creationFlags, noColorAttachment = false, useSRGBBuffer = false) {\n var _a, _b, _c, _d, _e, _f;\n let colorAttachment = undefined;\n if (typeof generateMipMaps === \"object\") {\n const options = generateMipMaps;\n generateMipMaps = !!options.generateMipMaps;\n doNotChangeAspectRatio = (_a = options.doNotChangeAspectRatio) !== null && _a !== void 0 ? _a : true;\n type = (_b = options.type) !== null && _b !== void 0 ? _b : 0;\n isCube = !!options.isCube;\n samplingMode = (_c = options.samplingMode) !== null && _c !== void 0 ? _c : Texture.TRILINEAR_SAMPLINGMODE;\n generateDepthBuffer = (_d = options.generateDepthBuffer) !== null && _d !== void 0 ? _d : true;\n generateStencilBuffer = !!options.generateStencilBuffer;\n isMulti = !!options.isMulti;\n format = (_e = options.format) !== null && _e !== void 0 ? _e : 5;\n delayAllocation = !!options.delayAllocation;\n samples = options.samples;\n creationFlags = options.creationFlags;\n noColorAttachment = !!options.noColorAttachment;\n useSRGBBuffer = !!options.useSRGBBuffer;\n colorAttachment = options.colorAttachment;\n }\n super(null, scene, !generateMipMaps, undefined, samplingMode, undefined, undefined, undefined, undefined, format);\n this._unObserveRenderList = null;\n this._renderListHasChanged = (_functionName, previousLength) => {\n var _a;\n const newLength = this._renderList ? this._renderList.length : 0;\n if ((previousLength === 0 && newLength > 0) || newLength === 0) {\n (_a = this.getScene()) === null || _a === void 0 ? void 0 : _a.meshes.forEach((mesh) => {\n mesh._markSubMeshesAsLightDirty();\n });\n }\n };\n /**\n * Define if particles should be rendered in your texture.\n */\n this.renderParticles = true;\n /**\n * Define if sprites should be rendered in your texture.\n */\n this.renderSprites = false;\n /**\n * Force checking the layerMask property even if a custom list of meshes is provided (ie. if renderList is not undefined)\n */\n this.forceLayerMaskCheck = false;\n /**\n * Define if the camera viewport should be respected while rendering the texture or if the render should be done to the entire texture.\n */\n this.ignoreCameraViewport = false;\n /**\n * An event triggered when the texture is unbind.\n */\n this.onBeforeBindObservable = new Observable$1();\n /**\n * An event triggered when the texture is unbind.\n */\n this.onAfterUnbindObservable = new Observable$1();\n /**\n * An event triggered before rendering the texture\n */\n this.onBeforeRenderObservable = new Observable$1();\n /**\n * An event triggered after rendering the texture\n */\n this.onAfterRenderObservable = new Observable$1();\n /**\n * An event triggered after the texture clear\n */\n this.onClearObservable = new Observable$1();\n /**\n * An event triggered when the texture is resized.\n */\n this.onResizeObservable = new Observable$1();\n /** @internal */\n this._cleared = false;\n /**\n * Skip the initial clear of the rtt at the beginning of the frame render loop\n */\n this.skipInitialClear = false;\n this._currentRefreshId = -1;\n this._refreshRate = 1;\n this._samples = 1;\n this._canRescale = true;\n this._renderTarget = null;\n /**\n * Gets or sets the center of the bounding box associated with the texture (when in cube mode)\n * It must define where the camera used to render the texture is set\n */\n this.boundingBoxPosition = Vector3.Zero();\n scene = this.getScene();\n if (!scene) {\n return;\n }\n const engine = this.getScene().getEngine();\n this._coordinatesMode = Texture.PROJECTION_MODE;\n this.renderList = new Array();\n this.name = name;\n this.isRenderTarget = true;\n this._initialSizeParameter = size;\n this._renderPassIds = [];\n this._isCubeData = isCube;\n this._processSizeParameter(size);\n this.renderPassId = this._renderPassIds[0];\n this._resizeObserver = engine.onResizeObservable.add(() => { });\n this._generateMipMaps = generateMipMaps ? true : false;\n this._doNotChangeAspectRatio = doNotChangeAspectRatio;\n // Rendering groups\n this._renderingManager = new RenderingManager(scene);\n this._renderingManager._useSceneAutoClearSetup = true;\n if (isMulti) {\n return;\n }\n this._renderTargetOptions = {\n generateMipMaps: generateMipMaps,\n type: type,\n format: (_f = this._format) !== null && _f !== void 0 ? _f : undefined,\n samplingMode: this.samplingMode,\n generateDepthBuffer: generateDepthBuffer,\n generateStencilBuffer: generateStencilBuffer,\n samples,\n creationFlags,\n noColorAttachment: noColorAttachment,\n useSRGBBuffer,\n colorAttachment: colorAttachment,\n label: this.name,\n };\n if (this.samplingMode === Texture.NEAREST_SAMPLINGMODE) {\n this.wrapU = Texture.CLAMP_ADDRESSMODE;\n this.wrapV = Texture.CLAMP_ADDRESSMODE;\n }\n if (!delayAllocation) {\n if (isCube) {\n this._renderTarget = scene.getEngine().createRenderTargetCubeTexture(this.getRenderSize(), this._renderTargetOptions);\n this.coordinatesMode = Texture.INVCUBIC_MODE;\n this._textureMatrix = Matrix.Identity();\n }\n else {\n this._renderTarget = scene.getEngine().createRenderTargetTexture(this._size, this._renderTargetOptions);\n }\n this._texture = this._renderTarget.texture;\n if (samples !== undefined) {\n this.samples = samples;\n }\n }\n }\n /**\n * Creates a depth stencil texture.\n * This is only available in WebGL 2 or with the depth texture extension available.\n * @param comparisonFunction Specifies the comparison function to set on the texture. If 0 or undefined, the texture is not in comparison mode (default: 0)\n * @param bilinearFiltering Specifies whether or not bilinear filtering is enable on the texture (default: true)\n * @param generateStencil Specifies whether or not a stencil should be allocated in the texture (default: false)\n * @param samples sample count of the depth/stencil texture (default: 1)\n * @param format format of the depth texture (default: 14)\n */\n createDepthStencilTexture(comparisonFunction = 0, bilinearFiltering = true, generateStencil = false, samples = 1, format = 14) {\n var _a;\n (_a = this._renderTarget) === null || _a === void 0 ? void 0 : _a.createDepthStencilTexture(comparisonFunction, bilinearFiltering, generateStencil, samples, format);\n }\n _releaseRenderPassId() {\n if (this._scene) {\n const engine = this._scene.getEngine();\n for (let i = 0; i < this._renderPassIds.length; ++i) {\n engine.releaseRenderPassId(this._renderPassIds[i]);\n }\n }\n this._renderPassIds = [];\n }\n _createRenderPassId() {\n this._releaseRenderPassId();\n const engine = this._scene.getEngine(); // scene can't be null in a RenderTargetTexture, see constructor\n const numPasses = this._isCubeData ? 6 : this.getRenderLayers() || 1;\n for (let i = 0; i < numPasses; ++i) {\n this._renderPassIds[i] = engine.createRenderPassId(`RenderTargetTexture - ${this.name}#${i}`);\n }\n }\n _processSizeParameter(size, createRenderPassIds = true) {\n if (size.ratio) {\n this._sizeRatio = size.ratio;\n const engine = this._getEngine();\n this._size = {\n width: this._bestReflectionRenderTargetDimension(engine.getRenderWidth(), this._sizeRatio),\n height: this._bestReflectionRenderTargetDimension(engine.getRenderHeight(), this._sizeRatio),\n };\n }\n else {\n this._size = size;\n }\n if (createRenderPassIds) {\n this._createRenderPassId();\n }\n }\n /**\n * Define the number of samples to use in case of MSAA.\n * It defaults to one meaning no MSAA has been enabled.\n */\n get samples() {\n var _a, _b;\n return (_b = (_a = this._renderTarget) === null || _a === void 0 ? void 0 : _a.samples) !== null && _b !== void 0 ? _b : this._samples;\n }\n set samples(value) {\n if (this._renderTarget) {\n this._samples = this._renderTarget.setSamples(value);\n }\n }\n /**\n * Resets the refresh counter of the texture and start bak from scratch.\n * Could be useful to regenerate the texture if it is setup to render only once.\n */\n resetRefreshCounter() {\n this._currentRefreshId = -1;\n }\n /**\n * Define the refresh rate of the texture or the rendering frequency.\n * Use 0 to render just once, 1 to render on every frame, 2 to render every two frames and so on...\n */\n get refreshRate() {\n return this._refreshRate;\n }\n set refreshRate(value) {\n this._refreshRate = value;\n this.resetRefreshCounter();\n }\n /**\n * Adds a post process to the render target rendering passes.\n * @param postProcess define the post process to add\n */\n addPostProcess(postProcess) {\n if (!this._postProcessManager) {\n const scene = this.getScene();\n if (!scene) {\n return;\n }\n this._postProcessManager = new PostProcessManager(scene);\n this._postProcesses = new Array();\n }\n this._postProcesses.push(postProcess);\n this._postProcesses[0].autoClear = false;\n }\n /**\n * Clear all the post processes attached to the render target\n * @param dispose define if the cleared post processes should also be disposed (false by default)\n */\n clearPostProcesses(dispose = false) {\n if (!this._postProcesses) {\n return;\n }\n if (dispose) {\n for (const postProcess of this._postProcesses) {\n postProcess.dispose();\n }\n }\n this._postProcesses = [];\n }\n /**\n * Remove one of the post process from the list of attached post processes to the texture\n * @param postProcess define the post process to remove from the list\n */\n removePostProcess(postProcess) {\n if (!this._postProcesses) {\n return;\n }\n const index = this._postProcesses.indexOf(postProcess);\n if (index === -1) {\n return;\n }\n this._postProcesses.splice(index, 1);\n if (this._postProcesses.length > 0) {\n this._postProcesses[0].autoClear = false;\n }\n }\n /** @internal */\n _shouldRender() {\n if (this._currentRefreshId === -1) {\n // At least render once\n this._currentRefreshId = 1;\n return true;\n }\n if (this.refreshRate === this._currentRefreshId) {\n this._currentRefreshId = 1;\n return true;\n }\n this._currentRefreshId++;\n return false;\n }\n /**\n * Gets the actual render size of the texture.\n * @returns the width of the render size\n */\n getRenderSize() {\n return this.getRenderWidth();\n }\n /**\n * Gets the actual render width of the texture.\n * @returns the width of the render size\n */\n getRenderWidth() {\n if (this._size.width) {\n return this._size.width;\n }\n return this._size;\n }\n /**\n * Gets the actual render height of the texture.\n * @returns the height of the render size\n */\n getRenderHeight() {\n if (this._size.width) {\n return this._size.height;\n }\n return this._size;\n }\n /**\n * Gets the actual number of layers of the texture.\n * @returns the number of layers\n */\n getRenderLayers() {\n const layers = this._size.layers;\n if (layers) {\n return layers;\n }\n return 0;\n }\n /**\n * Don't allow this render target texture to rescale. Mainly used to prevent rescaling by the scene optimizer.\n */\n disableRescaling() {\n this._canRescale = false;\n }\n /**\n * Get if the texture can be rescaled or not.\n */\n get canRescale() {\n return this._canRescale;\n }\n /**\n * Resize the texture using a ratio.\n * @param ratio the ratio to apply to the texture size in order to compute the new target size\n */\n scale(ratio) {\n const newSize = Math.max(1, this.getRenderSize() * ratio);\n this.resize(newSize);\n }\n /**\n * Get the texture reflection matrix used to rotate/transform the reflection.\n * @returns the reflection matrix\n */\n getReflectionTextureMatrix() {\n if (this.isCube) {\n return this._textureMatrix;\n }\n return super.getReflectionTextureMatrix();\n }\n /**\n * Resize the texture to a new desired size.\n * Be careful as it will recreate all the data in the new texture.\n * @param size Define the new size. It can be:\n * - a number for squared texture,\n * - an object containing { width: number, height: number }\n * - or an object containing a ratio { ratio: number }\n */\n resize(size) {\n var _a;\n const wasCube = this.isCube;\n (_a = this._renderTarget) === null || _a === void 0 ? void 0 : _a.dispose();\n this._renderTarget = null;\n const scene = this.getScene();\n if (!scene) {\n return;\n }\n this._processSizeParameter(size, false);\n if (wasCube) {\n this._renderTarget = scene.getEngine().createRenderTargetCubeTexture(this.getRenderSize(), this._renderTargetOptions);\n }\n else {\n this._renderTarget = scene.getEngine().createRenderTargetTexture(this._size, this._renderTargetOptions);\n }\n this._texture = this._renderTarget.texture;\n if (this._renderTargetOptions.samples !== undefined) {\n this.samples = this._renderTargetOptions.samples;\n }\n if (this.onResizeObservable.hasObservers()) {\n this.onResizeObservable.notifyObservers(this);\n }\n }\n /**\n * Renders all the objects from the render list into the texture.\n * @param useCameraPostProcess Define if camera post processes should be used during the rendering\n * @param dumpForDebug Define if the rendering result should be dumped (copied) for debugging purpose\n */\n render(useCameraPostProcess = false, dumpForDebug = false) {\n this._render(useCameraPostProcess, dumpForDebug);\n }\n /**\n * This function will check if the render target texture can be rendered (textures are loaded, shaders are compiled)\n * @returns true if all required resources are ready\n */\n isReadyForRendering() {\n return this._render(false, false, true);\n }\n _render(useCameraPostProcess = false, dumpForDebug = false, checkReadiness = false) {\n var _a;\n const scene = this.getScene();\n if (!scene) {\n return checkReadiness;\n }\n const engine = scene.getEngine();\n if (this.useCameraPostProcesses !== undefined) {\n useCameraPostProcess = this.useCameraPostProcesses;\n }\n if (this._waitingRenderList) {\n this.renderList = [];\n for (let index = 0; index < this._waitingRenderList.length; index++) {\n const id = this._waitingRenderList[index];\n const mesh = scene.getMeshById(id);\n if (mesh) {\n this.renderList.push(mesh);\n }\n }\n this._waitingRenderList = undefined;\n }\n // Is predicate defined?\n if (this.renderListPredicate) {\n if (this.renderList) {\n this.renderList.length = 0; // Clear previous renderList\n }\n else {\n this.renderList = [];\n }\n const scene = this.getScene();\n if (!scene) {\n return checkReadiness;\n }\n const sceneMeshes = scene.meshes;\n for (let index = 0; index < sceneMeshes.length; index++) {\n const mesh = sceneMeshes[index];\n if (this.renderListPredicate(mesh)) {\n this.renderList.push(mesh);\n }\n }\n }\n const currentRenderPassId = engine.currentRenderPassId;\n this.onBeforeBindObservable.notifyObservers(this);\n // Set custom projection.\n // Needs to be before binding to prevent changing the aspect ratio.\n const camera = (_a = this.activeCamera) !== null && _a !== void 0 ? _a : scene.activeCamera;\n const sceneCamera = scene.activeCamera;\n if (camera) {\n if (camera !== scene.activeCamera) {\n scene.setTransformMatrix(camera.getViewMatrix(), camera.getProjectionMatrix(true));\n scene.activeCamera = camera;\n }\n engine.setViewport(camera.rigParent ? camera.rigParent.viewport : camera.viewport, this.getRenderWidth(), this.getRenderHeight());\n }\n this._defaultRenderListPrepared = false;\n let returnValue = checkReadiness;\n if (!checkReadiness) {\n if (this.is2DArray && !this.isMulti) {\n for (let layer = 0; layer < this.getRenderLayers(); layer++) {\n this._renderToTarget(0, useCameraPostProcess, dumpForDebug, layer, camera);\n scene.incrementRenderId();\n scene.resetCachedMaterial();\n }\n }\n else if (this.isCube && !this.isMulti) {\n for (let face = 0; face < 6; face++) {\n this._renderToTarget(face, useCameraPostProcess, dumpForDebug, undefined, camera);\n scene.incrementRenderId();\n scene.resetCachedMaterial();\n }\n }\n else {\n this._renderToTarget(0, useCameraPostProcess, dumpForDebug, undefined, camera);\n }\n }\n else {\n if (!scene.getViewMatrix()) {\n // We probably didn't execute scene.render() yet, so make sure we have a view/projection matrix setup for the scene\n scene.updateTransformMatrix();\n }\n const numLayers = this.is2DArray ? this.getRenderLayers() : this.isCube ? 6 : 1;\n for (let layer = 0; layer < numLayers && returnValue; layer++) {\n let currentRenderList = null;\n const defaultRenderList = this.renderList ? this.renderList : scene.getActiveMeshes().data;\n const defaultRenderListLength = this.renderList ? this.renderList.length : scene.getActiveMeshes().length;\n engine.currentRenderPassId = this._renderPassIds[layer];\n this.onBeforeRenderObservable.notifyObservers(layer);\n if (this.getCustomRenderList) {\n currentRenderList = this.getCustomRenderList(layer, defaultRenderList, defaultRenderListLength);\n }\n if (!currentRenderList) {\n currentRenderList = defaultRenderList;\n }\n if (!this._doNotChangeAspectRatio) {\n scene.updateTransformMatrix(true);\n }\n for (let i = 0; i < currentRenderList.length && returnValue; ++i) {\n const mesh = currentRenderList[i];\n if (!mesh.isEnabled() || mesh.isBlocked || !mesh.isVisible || !mesh.subMeshes) {\n continue;\n }\n if (this.customIsReadyFunction) {\n if (!this.customIsReadyFunction(mesh, this.refreshRate, checkReadiness)) {\n returnValue = false;\n continue;\n }\n }\n else if (!mesh.isReady(true)) {\n returnValue = false;\n continue;\n }\n }\n this.onAfterRenderObservable.notifyObservers(layer);\n if (this.is2DArray || this.isCube) {\n scene.incrementRenderId();\n scene.resetCachedMaterial();\n }\n }\n }\n this.onAfterUnbindObservable.notifyObservers(this);\n engine.currentRenderPassId = currentRenderPassId;\n if (sceneCamera) {\n scene.activeCamera = sceneCamera;\n // Do not avoid setting uniforms when multiple scenes are active as another camera may have overwrite these\n if (scene.getEngine().scenes.length > 1 || (this.activeCamera && this.activeCamera !== scene.activeCamera)) {\n scene.setTransformMatrix(scene.activeCamera.getViewMatrix(), scene.activeCamera.getProjectionMatrix(true));\n }\n engine.setViewport(scene.activeCamera.viewport);\n }\n scene.resetCachedMaterial();\n return returnValue;\n }\n _bestReflectionRenderTargetDimension(renderDimension, scale) {\n const minimum = 128;\n const x = renderDimension * scale;\n const curved = Engine.NearestPOT(x + (minimum * minimum) / (minimum + x));\n // Ensure we don't exceed the render dimension (while staying POT)\n return Math.min(Engine.FloorPOT(renderDimension), curved);\n }\n _prepareRenderingManager(currentRenderList, currentRenderListLength, camera, checkLayerMask) {\n const scene = this.getScene();\n if (!scene) {\n return;\n }\n this._renderingManager.reset();\n const sceneRenderId = scene.getRenderId();\n for (let meshIndex = 0; meshIndex < currentRenderListLength; meshIndex++) {\n const mesh = currentRenderList[meshIndex];\n if (mesh && !mesh.isBlocked) {\n if (this.customIsReadyFunction) {\n if (!this.customIsReadyFunction(mesh, this.refreshRate, false)) {\n this.resetRefreshCounter();\n continue;\n }\n }\n else if (!mesh.isReady(this.refreshRate === 0)) {\n this.resetRefreshCounter();\n continue;\n }\n if (!mesh._internalAbstractMeshDataInfo._currentLODIsUpToDate && scene.activeCamera) {\n mesh._internalAbstractMeshDataInfo._currentLOD = scene.customLODSelector\n ? scene.customLODSelector(mesh, this.activeCamera || scene.activeCamera)\n : mesh.getLOD(this.activeCamera || scene.activeCamera);\n mesh._internalAbstractMeshDataInfo._currentLODIsUpToDate = true;\n }\n if (!mesh._internalAbstractMeshDataInfo._currentLOD) {\n continue;\n }\n let meshToRender = mesh._internalAbstractMeshDataInfo._currentLOD;\n meshToRender._preActivateForIntermediateRendering(sceneRenderId);\n let isMasked;\n if (checkLayerMask && camera) {\n isMasked = (mesh.layerMask & camera.layerMask) === 0;\n }\n else {\n isMasked = false;\n }\n if (mesh.isEnabled() && mesh.isVisible && mesh.subMeshes && !isMasked) {\n if (meshToRender !== mesh) {\n meshToRender._activate(sceneRenderId, true);\n }\n if (mesh._activate(sceneRenderId, true) && mesh.subMeshes.length) {\n if (!mesh.isAnInstance) {\n meshToRender._internalAbstractMeshDataInfo._onlyForInstancesIntermediate = false;\n }\n else {\n if (mesh._internalAbstractMeshDataInfo._actAsRegularMesh) {\n meshToRender = mesh;\n }\n }\n meshToRender._internalAbstractMeshDataInfo._isActiveIntermediate = true;\n for (let subIndex = 0; subIndex < meshToRender.subMeshes.length; subIndex++) {\n const subMesh = meshToRender.subMeshes[subIndex];\n this._renderingManager.dispatch(subMesh, meshToRender);\n }\n }\n }\n }\n }\n for (let particleIndex = 0; particleIndex < scene.particleSystems.length; particleIndex++) {\n const particleSystem = scene.particleSystems[particleIndex];\n const emitter = particleSystem.emitter;\n if (!particleSystem.isStarted() || !emitter || (emitter.position && !emitter.isEnabled())) {\n continue;\n }\n this._renderingManager.dispatchParticles(particleSystem);\n }\n }\n /**\n * @internal\n * @param faceIndex face index to bind to if this is a cubetexture\n * @param layer defines the index of the texture to bind in the array\n */\n _bindFrameBuffer(faceIndex = 0, layer = 0) {\n const scene = this.getScene();\n if (!scene) {\n return;\n }\n const engine = scene.getEngine();\n if (this._renderTarget) {\n engine.bindFramebuffer(this._renderTarget, this.isCube ? faceIndex : undefined, undefined, undefined, this.ignoreCameraViewport, 0, layer);\n }\n }\n _unbindFrameBuffer(engine, faceIndex) {\n if (!this._renderTarget) {\n return;\n }\n engine.unBindFramebuffer(this._renderTarget, this.isCube, () => {\n this.onAfterRenderObservable.notifyObservers(faceIndex);\n });\n }\n /**\n * @internal\n */\n _prepareFrame(scene, faceIndex, layer, useCameraPostProcess) {\n if (this._postProcessManager) {\n if (!this._prePassEnabled) {\n this._postProcessManager._prepareFrame(this._texture, this._postProcesses);\n }\n }\n else if (!useCameraPostProcess || !scene.postProcessManager._prepareFrame(this._texture)) {\n this._bindFrameBuffer(faceIndex, layer);\n }\n }\n _renderToTarget(faceIndex, useCameraPostProcess, dumpForDebug, layer = 0, camera = null) {\n var _a, _b, _c, _d, _e, _f;\n const scene = this.getScene();\n if (!scene) {\n return;\n }\n const engine = scene.getEngine();\n (_a = engine._debugPushGroup) === null || _a === void 0 ? void 0 : _a.call(engine, `render to face #${faceIndex} layer #${layer}`, 1);\n // Bind\n this._prepareFrame(scene, faceIndex, layer, useCameraPostProcess);\n if (this.is2DArray) {\n engine.currentRenderPassId = this._renderPassIds[layer];\n this.onBeforeRenderObservable.notifyObservers(layer);\n }\n else {\n engine.currentRenderPassId = this._renderPassIds[faceIndex];\n this.onBeforeRenderObservable.notifyObservers(faceIndex);\n }\n const fastPath = engine.snapshotRendering && engine.snapshotRenderingMode === 1;\n if (!fastPath) {\n // Get the list of meshes to render\n let currentRenderList = null;\n const defaultRenderList = this.renderList ? this.renderList : scene.getActiveMeshes().data;\n const defaultRenderListLength = this.renderList ? this.renderList.length : scene.getActiveMeshes().length;\n if (this.getCustomRenderList) {\n currentRenderList = this.getCustomRenderList(this.is2DArray ? layer : faceIndex, defaultRenderList, defaultRenderListLength);\n }\n if (!currentRenderList) {\n // No custom render list provided, we prepare the rendering for the default list, but check\n // first if we did not already performed the preparation before so as to avoid re-doing it several times\n if (!this._defaultRenderListPrepared) {\n this._prepareRenderingManager(defaultRenderList, defaultRenderListLength, camera, !this.renderList || this.forceLayerMaskCheck);\n this._defaultRenderListPrepared = true;\n }\n currentRenderList = defaultRenderList;\n }\n else {\n // Prepare the rendering for the custom render list provided\n this._prepareRenderingManager(currentRenderList, currentRenderList.length, camera, this.forceLayerMaskCheck);\n }\n // Before clear\n for (const step of scene._beforeRenderTargetClearStage) {\n step.action(this, faceIndex, layer);\n }\n // Clear\n if (this.onClearObservable.hasObservers()) {\n this.onClearObservable.notifyObservers(engine);\n }\n else {\n if (!this.skipInitialClear) {\n engine.clear(this.clearColor || scene.clearColor, true, true, true);\n }\n }\n if (!this._doNotChangeAspectRatio) {\n scene.updateTransformMatrix(true);\n }\n // Before Camera Draw\n for (const step of scene._beforeRenderTargetDrawStage) {\n step.action(this, faceIndex, layer);\n }\n // Render\n this._renderingManager.render(this.customRenderFunction, currentRenderList, this.renderParticles, this.renderSprites);\n // After Camera Draw\n for (const step of scene._afterRenderTargetDrawStage) {\n step.action(this, faceIndex, layer);\n }\n const saveGenerateMipMaps = (_c = (_b = this._texture) === null || _b === void 0 ? void 0 : _b.generateMipMaps) !== null && _c !== void 0 ? _c : false;\n if (this._texture) {\n this._texture.generateMipMaps = false; // if left true, the mipmaps will be generated (if this._texture.generateMipMaps = true) when the first post process binds its own RTT: by doing so it will unbind the current RTT,\n // which will trigger a mipmap generation. We don't want this because it's a wasted work, we will do an unbind of the current RTT at the end of the process (see unbindFrameBuffer) which will\n // trigger the generation of the final mipmaps\n }\n if (this._postProcessManager) {\n this._postProcessManager._finalizeFrame(false, (_d = this._renderTarget) !== null && _d !== void 0 ? _d : undefined, faceIndex, this._postProcesses, this.ignoreCameraViewport);\n }\n else if (useCameraPostProcess) {\n scene.postProcessManager._finalizeFrame(false, (_e = this._renderTarget) !== null && _e !== void 0 ? _e : undefined, faceIndex);\n }\n for (const step of scene._afterRenderTargetPostProcessStage) {\n step.action(this, faceIndex, layer);\n }\n if (this._texture) {\n this._texture.generateMipMaps = saveGenerateMipMaps;\n }\n if (!this._doNotChangeAspectRatio) {\n scene.updateTransformMatrix(true);\n }\n // Dump ?\n if (dumpForDebug) {\n DumpTools.DumpFramebuffer(this.getRenderWidth(), this.getRenderHeight(), engine);\n }\n }\n else {\n // Clear\n if (this.onClearObservable.hasObservers()) {\n this.onClearObservable.notifyObservers(engine);\n }\n else {\n if (!this.skipInitialClear) {\n engine.clear(this.clearColor || scene.clearColor, true, true, true);\n }\n }\n }\n // Unbind\n this._unbindFrameBuffer(engine, faceIndex);\n if (this._texture && this.isCube && faceIndex === 5) {\n engine.generateMipMapsForCubemap(this._texture);\n }\n (_f = engine._debugPopGroup) === null || _f === void 0 ? void 0 : _f.call(engine, 1);\n }\n /**\n * Overrides the default sort function applied in the rendering group to prepare the meshes.\n * This allowed control for front to back rendering or reversely depending of the special needs.\n *\n * @param renderingGroupId The rendering group id corresponding to its index\n * @param opaqueSortCompareFn The opaque queue comparison function use to sort.\n * @param alphaTestSortCompareFn The alpha test queue comparison function use to sort.\n * @param transparentSortCompareFn The transparent queue comparison function use to sort.\n */\n setRenderingOrder(renderingGroupId, opaqueSortCompareFn = null, alphaTestSortCompareFn = null, transparentSortCompareFn = null) {\n this._renderingManager.setRenderingOrder(renderingGroupId, opaqueSortCompareFn, alphaTestSortCompareFn, transparentSortCompareFn);\n }\n /**\n * Specifies whether or not the stencil and depth buffer are cleared between two rendering groups.\n *\n * @param renderingGroupId The rendering group id corresponding to its index\n * @param autoClearDepthStencil Automatically clears depth and stencil between groups if true.\n */\n setRenderingAutoClearDepthStencil(renderingGroupId, autoClearDepthStencil) {\n this._renderingManager.setRenderingAutoClearDepthStencil(renderingGroupId, autoClearDepthStencil);\n this._renderingManager._useSceneAutoClearSetup = false;\n }\n /**\n * Clones the texture.\n * @returns the cloned texture\n */\n clone() {\n const textureSize = this.getSize();\n const newTexture = new RenderTargetTexture(this.name, textureSize, this.getScene(), this._renderTargetOptions.generateMipMaps, this._doNotChangeAspectRatio, this._renderTargetOptions.type, this.isCube, this._renderTargetOptions.samplingMode, this._renderTargetOptions.generateDepthBuffer, this._renderTargetOptions.generateStencilBuffer, undefined, this._renderTargetOptions.format, undefined, this._renderTargetOptions.samples);\n // Base texture\n newTexture.hasAlpha = this.hasAlpha;\n newTexture.level = this.level;\n // RenderTarget Texture\n newTexture.coordinatesMode = this.coordinatesMode;\n if (this.renderList) {\n newTexture.renderList = this.renderList.slice(0);\n }\n return newTexture;\n }\n /**\n * Serialize the texture to a JSON representation we can easily use in the respective Parse function.\n * @returns The JSON representation of the texture\n */\n serialize() {\n if (!this.name) {\n return null;\n }\n const serializationObject = super.serialize();\n serializationObject.renderTargetSize = this.getRenderSize();\n serializationObject.renderList = [];\n if (this.renderList) {\n for (let index = 0; index < this.renderList.length; index++) {\n serializationObject.renderList.push(this.renderList[index].id);\n }\n }\n return serializationObject;\n }\n /**\n * This will remove the attached framebuffer objects. The texture will not be able to be used as render target anymore\n */\n disposeFramebufferObjects() {\n var _a;\n (_a = this._renderTarget) === null || _a === void 0 ? void 0 : _a.dispose(true);\n }\n /**\n * Release and destroy the underlying lower level texture aka internalTexture.\n */\n releaseInternalTexture() {\n var _a;\n (_a = this._renderTarget) === null || _a === void 0 ? void 0 : _a.releaseTextures();\n this._texture = null;\n }\n /**\n * Dispose the texture and release its associated resources.\n */\n dispose() {\n var _a;\n this.onResizeObservable.clear();\n this.onClearObservable.clear();\n this.onAfterRenderObservable.clear();\n this.onAfterUnbindObservable.clear();\n this.onBeforeBindObservable.clear();\n this.onBeforeRenderObservable.clear();\n if (this._postProcessManager) {\n this._postProcessManager.dispose();\n this._postProcessManager = null;\n }\n if (this._prePassRenderTarget) {\n this._prePassRenderTarget.dispose();\n }\n this._releaseRenderPassId();\n this.clearPostProcesses(true);\n if (this._resizeObserver) {\n this.getScene().getEngine().onResizeObservable.remove(this._resizeObserver);\n this._resizeObserver = null;\n }\n this.renderList = null;\n // Remove from custom render targets\n const scene = this.getScene();\n if (!scene) {\n return;\n }\n let index = scene.customRenderTargets.indexOf(this);\n if (index >= 0) {\n scene.customRenderTargets.splice(index, 1);\n }\n for (const camera of scene.cameras) {\n index = camera.customRenderTargets.indexOf(this);\n if (index >= 0) {\n camera.customRenderTargets.splice(index, 1);\n }\n }\n (_a = this._renderTarget) === null || _a === void 0 ? void 0 : _a.dispose();\n this._renderTarget = null;\n this._texture = null;\n super.dispose();\n }\n /** @internal */\n _rebuild() {\n if (this.refreshRate === RenderTargetTexture.REFRESHRATE_RENDER_ONCE) {\n this.refreshRate = RenderTargetTexture.REFRESHRATE_RENDER_ONCE;\n }\n if (this._postProcessManager) {\n this._postProcessManager._rebuild();\n }\n }\n /**\n * Clear the info related to rendering groups preventing retention point in material dispose.\n */\n freeRenderingGroups() {\n if (this._renderingManager) {\n this._renderingManager.freeRenderingGroups();\n }\n }\n /**\n * Gets the number of views the corresponding to the texture (eg. a MultiviewRenderTarget will have > 1)\n * @returns the view count\n */\n getViewCount() {\n return 1;\n }\n }\n /**\n * The texture will only be rendered once which can be useful to improve performance if everything in your render is static for instance.\n */\n RenderTargetTexture.REFRESHRATE_RENDER_ONCE = 0;\n /**\n * The texture will only be rendered rendered every frame and is recommended for dynamic contents.\n */\n RenderTargetTexture.REFRESHRATE_RENDER_ONEVERYFRAME = 1;\n /**\n * The texture will be rendered every 2 frames which could be enough if your dynamic objects are not\n * the central point of your effect and can save a lot of performances.\n */\n RenderTargetTexture.REFRESHRATE_RENDER_ONEVERYTWOFRAMES = 2;\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n Texture._CreateRenderTargetTexture = (name, renderTargetSize, scene, generateMipMaps, creationFlags) => {\n return new RenderTargetTexture(name, renderTargetSize, scene, generateMipMaps);\n };\n\n // Do not edit.\n const name$12 = \"passCubePixelShader\";\n const shader$12 = `varying vec2 vUV;uniform samplerCube textureSampler;\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) \n{vec2 uv=vUV*2.0-1.0;\n#ifdef POSITIVEX\ngl_FragColor=textureCube(textureSampler,vec3(1.001,uv.y,uv.x));\n#endif\n#ifdef NEGATIVEX\ngl_FragColor=textureCube(textureSampler,vec3(-1.001,uv.y,uv.x));\n#endif\n#ifdef POSITIVEY\ngl_FragColor=textureCube(textureSampler,vec3(uv.y,1.001,uv.x));\n#endif\n#ifdef NEGATIVEY\ngl_FragColor=textureCube(textureSampler,vec3(uv.y,-1.001,uv.x));\n#endif\n#ifdef POSITIVEZ\ngl_FragColor=textureCube(textureSampler,vec3(uv,1.001));\n#endif\n#ifdef NEGATIVEZ\ngl_FragColor=textureCube(textureSampler,vec3(uv,-1.001));\n#endif\n}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$12] = shader$12;\n\n /**\n * PassPostProcess which produces an output the same as it's input\n */\n class PassPostProcess extends PostProcess {\n /**\n * Gets a string identifying the name of the class\n * @returns \"PassPostProcess\" string\n */\n getClassName() {\n return \"PassPostProcess\";\n }\n /**\n * Creates the PassPostProcess\n * @param name The name of the effect.\n * @param options The required width/height ratio to downsize to before computing the render pass.\n * @param camera The camera to apply the render pass to.\n * @param samplingMode The sampling mode to be used when computing the pass. (default: 0)\n * @param engine The engine which the post process will be applied. (default: current engine)\n * @param reusable If the post process can be reused on the same frame. (default: false)\n * @param textureType The type of texture to be used when performing the post processing.\n * @param blockCompilation If compilation of the shader should not be done in the constructor. The updateEffect method can be used to compile the shader at a later time. (default: false)\n */\n constructor(name, options, camera = null, samplingMode, engine, reusable, textureType = 0, blockCompilation = false) {\n super(name, \"pass\", null, null, options, camera, samplingMode, engine, reusable, undefined, textureType, undefined, null, blockCompilation);\n }\n /**\n * @internal\n */\n static _Parse(parsedPostProcess, targetCamera, scene, rootUrl) {\n return SerializationHelper.Parse(() => {\n return new PassPostProcess(parsedPostProcess.name, parsedPostProcess.options, targetCamera, parsedPostProcess.renderTargetSamplingMode, parsedPostProcess._engine, parsedPostProcess.reusable);\n }, parsedPostProcess, scene, rootUrl);\n }\n }\n RegisterClass(\"BABYLON.PassPostProcess\", PassPostProcess);\n Engine._RescalePostProcessFactory = (engine) => {\n return new PassPostProcess(\"rescale\", 1, null, 2, engine, false, 0);\n };\n\n /**\n * Uses the GPU to create a copy texture rescaled at a given size\n * @param texture Texture to copy from\n * @param width defines the desired width\n * @param height defines the desired height\n * @param useBilinearMode defines if bilinear mode has to be used\n * @returns the generated texture\n */\n function CreateResizedCopy(texture, width, height, useBilinearMode = true) {\n const scene = texture.getScene();\n const engine = scene.getEngine();\n const rtt = new RenderTargetTexture(\"resized\" + texture.name, { width: width, height: height }, scene, !texture.noMipmap, true, texture._texture.type, false, texture.samplingMode, false);\n rtt.wrapU = texture.wrapU;\n rtt.wrapV = texture.wrapV;\n rtt.uOffset = texture.uOffset;\n rtt.vOffset = texture.vOffset;\n rtt.uScale = texture.uScale;\n rtt.vScale = texture.vScale;\n rtt.uAng = texture.uAng;\n rtt.vAng = texture.vAng;\n rtt.wAng = texture.wAng;\n rtt.coordinatesIndex = texture.coordinatesIndex;\n rtt.level = texture.level;\n rtt.anisotropicFilteringLevel = texture.anisotropicFilteringLevel;\n rtt._texture.isReady = false;\n texture.wrapU = Texture.CLAMP_ADDRESSMODE;\n texture.wrapV = Texture.CLAMP_ADDRESSMODE;\n const passPostProcess = new PassPostProcess(\"pass\", 1, null, useBilinearMode ? Texture.BILINEAR_SAMPLINGMODE : Texture.NEAREST_SAMPLINGMODE, engine, false, 0);\n passPostProcess.externalTextureSamplerBinding = true;\n passPostProcess.getEffect().executeWhenCompiled(() => {\n passPostProcess.onApply = function (effect) {\n effect.setTexture(\"textureSampler\", texture);\n };\n const internalTexture = rtt.renderTarget;\n if (internalTexture) {\n scene.postProcessManager.directRender([passPostProcess], internalTexture);\n engine.unBindFramebuffer(internalTexture);\n rtt.disposeFramebufferObjects();\n passPostProcess.dispose();\n rtt.getInternalTexture().isReady = true;\n }\n });\n return rtt;\n }\n /**\n * Apply a post process to a texture\n * @param postProcessName name of the fragment post process\n * @param internalTexture the texture to encode\n * @param scene the scene hosting the texture\n * @param type type of the output texture. If not provided, use the one from internalTexture\n * @param samplingMode sampling mode to use to sample the source texture. If not provided, use the one from internalTexture\n * @param format format of the output texture. If not provided, use the one from internalTexture\n * @returns a promise with the internalTexture having its texture replaced by the result of the processing\n */\n function ApplyPostProcess(postProcessName, internalTexture, scene, type, samplingMode, format, width, height) {\n // Gets everything ready.\n const engine = internalTexture.getEngine();\n internalTexture.isReady = false;\n samplingMode = samplingMode !== null && samplingMode !== void 0 ? samplingMode : internalTexture.samplingMode;\n type = type !== null && type !== void 0 ? type : internalTexture.type;\n format = format !== null && format !== void 0 ? format : internalTexture.format;\n width = width !== null && width !== void 0 ? width : internalTexture.width;\n height = height !== null && height !== void 0 ? height : internalTexture.height;\n if (type === -1) {\n type = 0;\n }\n return new Promise((resolve) => {\n // Create the post process\n const postProcess = new PostProcess(\"postprocess\", postProcessName, null, null, 1, null, samplingMode, engine, false, undefined, type, undefined, null, false, format);\n postProcess.externalTextureSamplerBinding = true;\n // Hold the output of the decoding.\n const encodedTexture = engine.createRenderTargetTexture({ width: width, height: height }, {\n generateDepthBuffer: false,\n generateMipMaps: false,\n generateStencilBuffer: false,\n samplingMode,\n type,\n format,\n });\n postProcess.getEffect().executeWhenCompiled(() => {\n // PP Render Pass\n postProcess.onApply = (effect) => {\n effect._bindTexture(\"textureSampler\", internalTexture);\n effect.setFloat2(\"scale\", 1, 1);\n };\n scene.postProcessManager.directRender([postProcess], encodedTexture, true);\n // Cleanup\n engine.restoreDefaultFramebuffer();\n engine._releaseTexture(internalTexture);\n if (postProcess) {\n postProcess.dispose();\n }\n // Internal Swap\n encodedTexture._swapAndDie(internalTexture);\n // Ready to get rolling again.\n internalTexture.type = type;\n internalTexture.format = 5;\n internalTexture.isReady = true;\n resolve(internalTexture);\n });\n });\n }\n // ref: http://stackoverflow.com/questions/32633585/how-do-you-convert-to-half-floats-in-javascript\n let floatView;\n let int32View;\n /**\n * Converts a number to half float\n * @param value number to convert\n * @returns converted number\n */\n function ToHalfFloat(value) {\n if (!floatView) {\n floatView = new Float32Array(1);\n int32View = new Int32Array(floatView.buffer);\n }\n floatView[0] = value;\n const x = int32View[0];\n let bits = (x >> 16) & 0x8000; /* Get the sign */\n let m = (x >> 12) & 0x07ff; /* Keep one extra bit for rounding */\n const e = (x >> 23) & 0xff; /* Using int is faster here */\n /* If zero, or denormal, or exponent underflows too much for a denormal\n * half, return signed zero. */\n if (e < 103) {\n return bits;\n }\n /* If NaN, return NaN. If Inf or exponent overflow, return Inf. */\n if (e > 142) {\n bits |= 0x7c00;\n /* If exponent was 0xff and one mantissa bit was set, it means NaN,\n * not Inf, so make sure we set one mantissa bit too. */\n bits |= (e == 255 ? 0 : 1) && x & 0x007fffff;\n return bits;\n }\n /* If exponent underflows but not too much, return a denormal */\n if (e < 113) {\n m |= 0x0800;\n /* Extra rounding may overflow and set mantissa to 0 and exponent\n * to 1, which is OK. */\n bits |= (m >> (114 - e)) + ((m >> (113 - e)) & 1);\n return bits;\n }\n bits |= ((e - 112) << 10) | (m >> 1);\n bits += m & 1;\n return bits;\n }\n /**\n * Converts a half float to a number\n * @param value half float to convert\n * @returns converted half float\n */\n function FromHalfFloat(value) {\n const s = (value & 0x8000) >> 15;\n const e = (value & 0x7c00) >> 10;\n const f = value & 0x03ff;\n if (e === 0) {\n return (s ? -1 : 1) * Math.pow(2, -14) * (f / Math.pow(2, 10));\n }\n else if (e == 0x1f) {\n return f ? NaN : (s ? -1 : 1) * Infinity;\n }\n return (s ? -1 : 1) * Math.pow(2, e - 15) * (1 + f / Math.pow(2, 10));\n }\n /**\n * Class used to host texture specific utilities\n */\n const TextureTools = {\n /**\n * Uses the GPU to create a copy texture rescaled at a given size\n * @param texture Texture to copy from\n * @param width defines the desired width\n * @param height defines the desired height\n * @param useBilinearMode defines if bilinear mode has to be used\n * @returns the generated texture\n */\n CreateResizedCopy,\n /**\n * Apply a post process to a texture\n * @param postProcessName name of the fragment post process\n * @param internalTexture the texture to encode\n * @param scene the scene hosting the texture\n * @param type type of the output texture. If not provided, use the one from internalTexture\n * @param samplingMode sampling mode to use to sample the source texture. If not provided, use the one from internalTexture\n * @param format format of the output texture. If not provided, use the one from internalTexture\n * @returns a promise with the internalTexture having its texture replaced by the result of the processing\n */\n ApplyPostProcess,\n /**\n * Converts a number to half float\n * @param value number to convert\n * @returns converted number\n */\n ToHalfFloat,\n /**\n * Converts a half float to a number\n * @param value half float to convert\n * @returns converted half float\n */\n FromHalfFloat,\n };\n\n /**\n * Class used to host RGBD texture specific utilities\n */\n class RGBDTextureTools {\n /**\n * Expand the RGBD Texture from RGBD to Half Float if possible.\n * @param texture the texture to expand.\n */\n static ExpandRGBDTexture(texture) {\n const internalTexture = texture._texture;\n if (!internalTexture || !texture.isRGBD) {\n return;\n }\n // Gets everything ready.\n const engine = internalTexture.getEngine();\n const caps = engine.getCaps();\n const isReady = internalTexture.isReady;\n let expandTexture = false;\n // If half float available we can uncompress the texture\n if (caps.textureHalfFloatRender && caps.textureHalfFloatLinearFiltering) {\n expandTexture = true;\n internalTexture.type = 2;\n }\n // If full float available we can uncompress the texture\n else if (caps.textureFloatRender && caps.textureFloatLinearFiltering) {\n expandTexture = true;\n internalTexture.type = 1;\n }\n if (expandTexture) {\n // Do not use during decode.\n internalTexture.isReady = false;\n internalTexture._isRGBD = false;\n internalTexture.invertY = false;\n }\n const expandRGBDTexture = () => {\n // Expand the texture if possible\n if (expandTexture) {\n // Simply run through the decode PP.\n const rgbdPostProcess = new PostProcess(\"rgbdDecode\", \"rgbdDecode\", null, null, 1, null, 3, engine, false, undefined, internalTexture.type, undefined, null, false);\n rgbdPostProcess.externalTextureSamplerBinding = true;\n // Hold the output of the decoding.\n const expandedTexture = engine.createRenderTargetTexture(internalTexture.width, {\n generateDepthBuffer: false,\n generateMipMaps: false,\n generateStencilBuffer: false,\n samplingMode: internalTexture.samplingMode,\n type: internalTexture.type,\n format: 5,\n });\n rgbdPostProcess.getEffect().executeWhenCompiled(() => {\n // PP Render Pass\n rgbdPostProcess.onApply = (effect) => {\n effect._bindTexture(\"textureSampler\", internalTexture);\n effect.setFloat2(\"scale\", 1, 1);\n };\n texture.getScene().postProcessManager.directRender([rgbdPostProcess], expandedTexture, true);\n // Cleanup\n engine.restoreDefaultFramebuffer();\n engine._releaseTexture(internalTexture);\n if (rgbdPostProcess) {\n rgbdPostProcess.dispose();\n }\n // Internal Swap\n expandedTexture._swapAndDie(internalTexture);\n // Ready to get rolling again.\n internalTexture.isReady = true;\n });\n }\n };\n if (isReady) {\n expandRGBDTexture();\n }\n else {\n texture.onLoadObservable.addOnce(expandRGBDTexture);\n }\n }\n /**\n * Encode the texture to RGBD if possible.\n * @param internalTexture the texture to encode\n * @param scene the scene hosting the texture\n * @param outputTextureType type of the texture in which the encoding is performed\n * @returns a promise with the internalTexture having its texture replaced by the result of the processing\n */\n static EncodeTextureToRGBD(internalTexture, scene, outputTextureType = 0) {\n return ApplyPostProcess(\"rgbdEncode\", internalTexture, scene, outputTextureType, 1, 5);\n }\n }\n\n // eslint-disable-next-line @typescript-eslint/naming-convention\n const _environmentBRDFBase64Texture = \"\";\n let _instanceNumber = 0;\n /**\n * Gets a default environment BRDF for MS-BRDF Height Correlated BRDF\n * @param scene defines the hosting scene\n * @returns the environment BRDF texture\n */\n const GetEnvironmentBRDFTexture = (scene) => {\n if (!scene.environmentBRDFTexture) {\n // Forces Delayed Texture Loading to prevent undefined error whilst setting RGBD values.\n const useDelayedTextureLoading = scene.useDelayedTextureLoading;\n scene.useDelayedTextureLoading = false;\n const previousState = scene._blockEntityCollection;\n scene._blockEntityCollection = false;\n const texture = Texture.CreateFromBase64String(_environmentBRDFBase64Texture, \"EnvironmentBRDFTexture\" + _instanceNumber++, scene, true, false, Texture.BILINEAR_SAMPLINGMODE);\n scene._blockEntityCollection = previousState;\n // BRDF Texture should not be cached here due to pre processing and redundant scene caches.\n const texturesCache = scene.getEngine().getLoadedTexturesCache();\n const index = texturesCache.indexOf(texture.getInternalTexture());\n if (index !== -1) {\n texturesCache.splice(index, 1);\n }\n texture.isRGBD = true;\n texture.wrapU = Texture.CLAMP_ADDRESSMODE;\n texture.wrapV = Texture.CLAMP_ADDRESSMODE;\n scene.environmentBRDFTexture = texture;\n scene.useDelayedTextureLoading = useDelayedTextureLoading;\n RGBDTextureTools.ExpandRGBDTexture(texture);\n const observer = scene.getEngine().onContextRestoredObservable.add(() => {\n texture.isRGBD = true;\n const checkReady = () => {\n if (texture.isReady()) {\n RGBDTextureTools.ExpandRGBDTexture(texture);\n }\n else {\n Tools.SetImmediate(checkReady);\n }\n };\n checkReady();\n });\n scene.onDisposeObservable.add(() => {\n scene.getEngine().onContextRestoredObservable.remove(observer);\n });\n }\n return scene.environmentBRDFTexture;\n };\n\n /**\n * @internal\n */\n class MaterialBRDFDefines extends MaterialDefines {\n constructor() {\n super(...arguments);\n this.BRDF_V_HEIGHT_CORRELATED = false;\n this.MS_BRDF_ENERGY_CONSERVATION = false;\n this.SPHERICAL_HARMONICS = false;\n this.SPECULAR_GLOSSINESS_ENERGY_CONSERVATION = false;\n }\n }\n /**\n * Plugin that implements the BRDF component of the PBR material\n */\n class PBRBRDFConfiguration extends MaterialPluginBase {\n /** @internal */\n _markAllSubMeshesAsMiscDirty() {\n this._internalMarkAllSubMeshesAsMiscDirty();\n }\n constructor(material, addToPluginList = true) {\n super(material, \"PBRBRDF\", 90, new MaterialBRDFDefines(), addToPluginList);\n this._useEnergyConservation = PBRBRDFConfiguration.DEFAULT_USE_ENERGY_CONSERVATION;\n /**\n * Defines if the material uses energy conservation.\n */\n this.useEnergyConservation = PBRBRDFConfiguration.DEFAULT_USE_ENERGY_CONSERVATION;\n this._useSmithVisibilityHeightCorrelated = PBRBRDFConfiguration.DEFAULT_USE_SMITH_VISIBILITY_HEIGHT_CORRELATED;\n /**\n * LEGACY Mode set to false\n * Defines if the material uses height smith correlated visibility term.\n * If you intent to not use our default BRDF, you need to load a separate BRDF Texture for the PBR\n * You can either load https://assets.babylonjs.com/environments/uncorrelatedBRDF.png\n * or https://assets.babylonjs.com/environments/uncorrelatedBRDF.dds to have more precision\n * Not relying on height correlated will also disable energy conservation.\n */\n this.useSmithVisibilityHeightCorrelated = PBRBRDFConfiguration.DEFAULT_USE_SMITH_VISIBILITY_HEIGHT_CORRELATED;\n this._useSphericalHarmonics = PBRBRDFConfiguration.DEFAULT_USE_SPHERICAL_HARMONICS;\n /**\n * LEGACY Mode set to false\n * Defines if the material uses spherical harmonics vs spherical polynomials for the\n * diffuse part of the IBL.\n * The harmonics despite a tiny bigger cost has been proven to provide closer results\n * to the ground truth.\n */\n this.useSphericalHarmonics = PBRBRDFConfiguration.DEFAULT_USE_SPHERICAL_HARMONICS;\n this._useSpecularGlossinessInputEnergyConservation = PBRBRDFConfiguration.DEFAULT_USE_SPECULAR_GLOSSINESS_INPUT_ENERGY_CONSERVATION;\n /**\n * Defines if the material uses energy conservation, when the specular workflow is active.\n * If activated, the albedo color is multiplied with (1. - maxChannel(specular color)).\n * If deactivated, a material is only physically plausible, when (albedo color + specular color) < 1.\n * In the deactivated case, the material author has to ensure energy conservation, for a physically plausible rendering.\n */\n this.useSpecularGlossinessInputEnergyConservation = PBRBRDFConfiguration.DEFAULT_USE_SPECULAR_GLOSSINESS_INPUT_ENERGY_CONSERVATION;\n this._internalMarkAllSubMeshesAsMiscDirty = material._dirtyCallbacks[16];\n this._enable(true);\n }\n prepareDefines(defines) {\n defines.BRDF_V_HEIGHT_CORRELATED = this._useSmithVisibilityHeightCorrelated;\n defines.MS_BRDF_ENERGY_CONSERVATION = this._useEnergyConservation && this._useSmithVisibilityHeightCorrelated;\n defines.SPHERICAL_HARMONICS = this._useSphericalHarmonics;\n defines.SPECULAR_GLOSSINESS_ENERGY_CONSERVATION = this._useSpecularGlossinessInputEnergyConservation;\n }\n getClassName() {\n return \"PBRBRDFConfiguration\";\n }\n }\n /**\n * Default value used for the energy conservation.\n * This should only be changed to adapt to the type of texture in scene.environmentBRDFTexture.\n */\n PBRBRDFConfiguration.DEFAULT_USE_ENERGY_CONSERVATION = true;\n /**\n * Default value used for the Smith Visibility Height Correlated mode.\n * This should only be changed to adapt to the type of texture in scene.environmentBRDFTexture.\n */\n PBRBRDFConfiguration.DEFAULT_USE_SMITH_VISIBILITY_HEIGHT_CORRELATED = true;\n /**\n * Default value used for the IBL diffuse part.\n * This can help switching back to the polynomials mode globally which is a tiny bit\n * less GPU intensive at the drawback of a lower quality.\n */\n PBRBRDFConfiguration.DEFAULT_USE_SPHERICAL_HARMONICS = true;\n /**\n * Default value used for activating energy conservation for the specular workflow.\n * If activated, the albedo color is multiplied with (1. - maxChannel(specular color)).\n * If deactivated, a material is only physically plausible, when (albedo color + specular color) < 1.\n */\n PBRBRDFConfiguration.DEFAULT_USE_SPECULAR_GLOSSINESS_INPUT_ENERGY_CONSERVATION = true;\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsMiscDirty\")\n ], PBRBRDFConfiguration.prototype, \"useEnergyConservation\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsMiscDirty\")\n ], PBRBRDFConfiguration.prototype, \"useSmithVisibilityHeightCorrelated\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsMiscDirty\")\n ], PBRBRDFConfiguration.prototype, \"useSphericalHarmonics\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsMiscDirty\")\n ], PBRBRDFConfiguration.prototype, \"useSpecularGlossinessInputEnergyConservation\", void 0);\n\n class FileFaceOrientation {\n constructor(name, worldAxisForNormal, worldAxisForFileX, worldAxisForFileY) {\n this.name = name;\n this.worldAxisForNormal = worldAxisForNormal;\n this.worldAxisForFileX = worldAxisForFileX;\n this.worldAxisForFileY = worldAxisForFileY;\n }\n }\n /**\n * Helper class dealing with the extraction of spherical polynomial dataArray\n * from a cube map.\n */\n class CubeMapToSphericalPolynomialTools {\n /**\n * Converts a texture to the according Spherical Polynomial data.\n * This extracts the first 3 orders only as they are the only one used in the lighting.\n *\n * @param texture The texture to extract the information from.\n * @returns The Spherical Polynomial data.\n */\n static ConvertCubeMapTextureToSphericalPolynomial(texture) {\n var _a;\n if (!texture.isCube) {\n // Only supports cube Textures currently.\n return null;\n }\n (_a = texture.getScene()) === null || _a === void 0 ? void 0 : _a.getEngine().flushFramebuffer();\n const size = texture.getSize().width;\n const rightPromise = texture.readPixels(0, undefined, undefined, false);\n const leftPromise = texture.readPixels(1, undefined, undefined, false);\n let upPromise;\n let downPromise;\n if (texture.isRenderTarget) {\n upPromise = texture.readPixels(3, undefined, undefined, false);\n downPromise = texture.readPixels(2, undefined, undefined, false);\n }\n else {\n upPromise = texture.readPixels(2, undefined, undefined, false);\n downPromise = texture.readPixels(3, undefined, undefined, false);\n }\n const frontPromise = texture.readPixels(4, undefined, undefined, false);\n const backPromise = texture.readPixels(5, undefined, undefined, false);\n const gammaSpace = texture.gammaSpace;\n // Always read as RGBA.\n const format = 5;\n let type = 0;\n if (texture.textureType == 1 || texture.textureType == 2) {\n type = 1;\n }\n return new Promise((resolve) => {\n Promise.all([leftPromise, rightPromise, upPromise, downPromise, frontPromise, backPromise]).then(([left, right, up, down, front, back]) => {\n const cubeInfo = {\n size,\n right,\n left,\n up,\n down,\n front,\n back,\n format,\n type,\n gammaSpace,\n };\n resolve(this.ConvertCubeMapToSphericalPolynomial(cubeInfo));\n });\n });\n }\n /**\n * Compute the area on the unit sphere of the rectangle defined by (x,y) and the origin\n * See https://www.rorydriscoll.com/2012/01/15/cubemap-texel-solid-angle/\n * @param x\n * @param y\n */\n static _AreaElement(x, y) {\n return Math.atan2(x * y, Math.sqrt(x * x + y * y + 1));\n }\n /**\n * Converts a cubemap to the according Spherical Polynomial data.\n * This extracts the first 3 orders only as they are the only one used in the lighting.\n *\n * @param cubeInfo The Cube map to extract the information from.\n * @returns The Spherical Polynomial data.\n */\n static ConvertCubeMapToSphericalPolynomial(cubeInfo) {\n const sphericalHarmonics = new SphericalHarmonics();\n let totalSolidAngle = 0.0;\n // The (u,v) range is [-1,+1], so the distance between each texel is 2/Size.\n const du = 2.0 / cubeInfo.size;\n const dv = du;\n const halfTexel = 0.5 * du;\n // The (u,v) of the first texel is half a texel from the corner (-1,-1).\n const minUV = halfTexel - 1.0;\n for (let faceIndex = 0; faceIndex < 6; faceIndex++) {\n const fileFace = this._FileFaces[faceIndex];\n const dataArray = cubeInfo[fileFace.name];\n let v = minUV;\n // TODO: we could perform the summation directly into a SphericalPolynomial (SP), which is more efficient than SphericalHarmonic (SH).\n // This is possible because during the summation we do not need the SH-specific properties, e.g. orthogonality.\n // Because SP is still linear, so summation is fine in that basis.\n const stride = cubeInfo.format === 5 ? 4 : 3;\n for (let y = 0; y < cubeInfo.size; y++) {\n let u = minUV;\n for (let x = 0; x < cubeInfo.size; x++) {\n // World direction (not normalised)\n const worldDirection = fileFace.worldAxisForFileX.scale(u).add(fileFace.worldAxisForFileY.scale(v)).add(fileFace.worldAxisForNormal);\n worldDirection.normalize();\n const deltaSolidAngle = this._AreaElement(u - halfTexel, v - halfTexel) -\n this._AreaElement(u - halfTexel, v + halfTexel) -\n this._AreaElement(u + halfTexel, v - halfTexel) +\n this._AreaElement(u + halfTexel, v + halfTexel);\n let r = dataArray[y * cubeInfo.size * stride + x * stride + 0];\n let g = dataArray[y * cubeInfo.size * stride + x * stride + 1];\n let b = dataArray[y * cubeInfo.size * stride + x * stride + 2];\n // Prevent NaN harmonics with extreme HDRI data.\n if (isNaN(r)) {\n r = 0;\n }\n if (isNaN(g)) {\n g = 0;\n }\n if (isNaN(b)) {\n b = 0;\n }\n // Handle Integer types.\n if (cubeInfo.type === 0) {\n r /= 255;\n g /= 255;\n b /= 255;\n }\n // Handle Gamma space textures.\n if (cubeInfo.gammaSpace) {\n r = Math.pow(Scalar.Clamp(r), ToLinearSpace);\n g = Math.pow(Scalar.Clamp(g), ToLinearSpace);\n b = Math.pow(Scalar.Clamp(b), ToLinearSpace);\n }\n // Prevent to explode in case of really high dynamic ranges.\n // sh 3 would not be enough to accurately represent it.\n const max = this.MAX_HDRI_VALUE;\n if (this.PRESERVE_CLAMPED_COLORS) {\n const currentMax = Math.max(r, g, b);\n if (currentMax > max) {\n const factor = max / currentMax;\n r *= factor;\n g *= factor;\n b *= factor;\n }\n }\n else {\n r = Scalar.Clamp(r, 0, max);\n g = Scalar.Clamp(g, 0, max);\n b = Scalar.Clamp(b, 0, max);\n }\n const color = new Color3(r, g, b);\n sphericalHarmonics.addLight(worldDirection, color, deltaSolidAngle);\n totalSolidAngle += deltaSolidAngle;\n u += du;\n }\n v += dv;\n }\n }\n // Solid angle for entire sphere is 4*pi\n const sphereSolidAngle = 4.0 * Math.PI;\n // Adjust the solid angle to allow for how many faces we processed.\n const facesProcessed = 6.0;\n const expectedSolidAngle = (sphereSolidAngle * facesProcessed) / 6.0;\n // Adjust the harmonics so that the accumulated solid angle matches the expected solid angle.\n // This is needed because the numerical integration over the cube uses a\n // small angle approximation of solid angle for each texel (see deltaSolidAngle),\n // and also to compensate for accumulative error due to float precision in the summation.\n const correctionFactor = expectedSolidAngle / totalSolidAngle;\n sphericalHarmonics.scaleInPlace(correctionFactor);\n sphericalHarmonics.convertIncidentRadianceToIrradiance();\n sphericalHarmonics.convertIrradianceToLambertianRadiance();\n return SphericalPolynomial.FromHarmonics(sphericalHarmonics);\n }\n }\n CubeMapToSphericalPolynomialTools._FileFaces = [\n new FileFaceOrientation(\"right\", new Vector3(1, 0, 0), new Vector3(0, 0, -1), new Vector3(0, -1, 0)),\n new FileFaceOrientation(\"left\", new Vector3(-1, 0, 0), new Vector3(0, 0, 1), new Vector3(0, -1, 0)),\n new FileFaceOrientation(\"up\", new Vector3(0, 1, 0), new Vector3(1, 0, 0), new Vector3(0, 0, 1)),\n new FileFaceOrientation(\"down\", new Vector3(0, -1, 0), new Vector3(1, 0, 0), new Vector3(0, 0, -1)),\n new FileFaceOrientation(\"front\", new Vector3(0, 0, 1), new Vector3(1, 0, 0), new Vector3(0, -1, 0)),\n new FileFaceOrientation(\"back\", new Vector3(0, 0, -1), new Vector3(-1, 0, 0), new Vector3(0, -1, 0)), // -Z bottom\n ];\n /** @internal */\n CubeMapToSphericalPolynomialTools.MAX_HDRI_VALUE = 4096;\n /** @internal */\n CubeMapToSphericalPolynomialTools.PRESERVE_CLAMPED_COLORS = false;\n\n BaseTexture.prototype.forceSphericalPolynomialsRecompute = function () {\n if (this._texture) {\n this._texture._sphericalPolynomial = null;\n this._texture._sphericalPolynomialPromise = null;\n this._texture._sphericalPolynomialComputed = false;\n }\n };\n Object.defineProperty(BaseTexture.prototype, \"sphericalPolynomial\", {\n get: function () {\n if (this._texture) {\n if (this._texture._sphericalPolynomial || this._texture._sphericalPolynomialComputed) {\n return this._texture._sphericalPolynomial;\n }\n if (this._texture.isReady) {\n if (!this._texture._sphericalPolynomialPromise) {\n this._texture._sphericalPolynomialPromise = CubeMapToSphericalPolynomialTools.ConvertCubeMapTextureToSphericalPolynomial(this);\n if (this._texture._sphericalPolynomialPromise === null) {\n this._texture._sphericalPolynomialComputed = true;\n }\n else {\n this._texture._sphericalPolynomialPromise.then((sphericalPolynomial) => {\n this._texture._sphericalPolynomial = sphericalPolynomial;\n this._texture._sphericalPolynomialComputed = true;\n });\n }\n }\n return null;\n }\n }\n return null;\n },\n set: function (value) {\n if (this._texture) {\n this._texture._sphericalPolynomial = value;\n }\n },\n enumerable: true,\n configurable: true,\n });\n\n // Do not edit.\n const name$13 = \"pbrFragmentDeclaration\";\n const shader$13 = `uniform vec4 vEyePosition;uniform vec3 vReflectionColor;uniform vec4 vAlbedoColor;uniform vec4 vLightingIntensity;uniform vec4 vReflectivityColor;uniform vec4 vMetallicReflectanceFactors;uniform vec3 vEmissiveColor;uniform float visibility;uniform vec3 vAmbientColor;\n#ifdef ALBEDO\nuniform vec2 vAlbedoInfos;\n#endif\n#ifdef AMBIENT\nuniform vec4 vAmbientInfos;\n#endif\n#ifdef BUMP\nuniform vec3 vBumpInfos;uniform vec2 vTangentSpaceParams;\n#endif\n#ifdef OPACITY\nuniform vec2 vOpacityInfos;\n#endif\n#ifdef EMISSIVE\nuniform vec2 vEmissiveInfos;\n#endif\n#ifdef LIGHTMAP\nuniform vec2 vLightmapInfos;\n#endif\n#ifdef REFLECTIVITY\nuniform vec3 vReflectivityInfos;\n#endif\n#ifdef MICROSURFACEMAP\nuniform vec2 vMicroSurfaceSamplerInfos;\n#endif\n#if defined(REFLECTIONMAP_SPHERICAL) || defined(REFLECTIONMAP_PROJECTION) || defined(SS_REFRACTION) || defined(PREPASS)\nuniform mat4 view;\n#endif\n#ifdef REFLECTION\nuniform vec2 vReflectionInfos;\n#ifdef REALTIME_FILTERING\nuniform vec2 vReflectionFilteringInfo;\n#endif\nuniform mat4 reflectionMatrix;uniform vec3 vReflectionMicrosurfaceInfos;\n#if defined(USE_LOCAL_REFLECTIONMAP_CUBIC) && defined(REFLECTIONMAP_CUBIC)\nuniform vec3 vReflectionPosition;uniform vec3 vReflectionSize; \n#endif\n#endif\n#if defined(SS_REFRACTION) && defined(SS_USE_LOCAL_REFRACTIONMAP_CUBIC)\nuniform vec3 vRefractionPosition;uniform vec3 vRefractionSize; \n#endif\n#ifdef CLEARCOAT\nuniform vec2 vClearCoatParams;uniform vec4 vClearCoatRefractionParams;\n#if defined(CLEARCOAT_TEXTURE) || defined(CLEARCOAT_TEXTURE_ROUGHNESS)\nuniform vec4 vClearCoatInfos;\n#endif\n#ifdef CLEARCOAT_TEXTURE\nuniform mat4 clearCoatMatrix;\n#endif\n#ifdef CLEARCOAT_TEXTURE_ROUGHNESS\nuniform mat4 clearCoatRoughnessMatrix;\n#endif\n#ifdef CLEARCOAT_BUMP\nuniform vec2 vClearCoatBumpInfos;uniform vec2 vClearCoatTangentSpaceParams;uniform mat4 clearCoatBumpMatrix;\n#endif\n#ifdef CLEARCOAT_TINT\nuniform vec4 vClearCoatTintParams;uniform float clearCoatColorAtDistance;\n#ifdef CLEARCOAT_TINT_TEXTURE\nuniform vec2 vClearCoatTintInfos;uniform mat4 clearCoatTintMatrix;\n#endif\n#endif\n#endif\n#ifdef IRIDESCENCE\nuniform vec4 vIridescenceParams;\n#if defined(IRIDESCENCE_TEXTURE) || defined(IRIDESCENCE_THICKNESS_TEXTURE)\nuniform vec4 vIridescenceInfos;\n#endif\n#ifdef IRIDESCENCE_TEXTURE\nuniform mat4 iridescenceMatrix;\n#endif\n#ifdef IRIDESCENCE_THICKNESS_TEXTURE\nuniform mat4 iridescenceThicknessMatrix;\n#endif\n#endif\n#ifdef ANISOTROPIC\nuniform vec3 vAnisotropy;\n#ifdef ANISOTROPIC_TEXTURE\nuniform vec2 vAnisotropyInfos;uniform mat4 anisotropyMatrix;\n#endif\n#endif\n#ifdef SHEEN\nuniform vec4 vSheenColor;\n#ifdef SHEEN_ROUGHNESS\nuniform float vSheenRoughness;\n#endif\n#if defined(SHEEN_TEXTURE) || defined(SHEEN_TEXTURE_ROUGHNESS)\nuniform vec4 vSheenInfos;\n#endif\n#ifdef SHEEN_TEXTURE\nuniform mat4 sheenMatrix;\n#endif\n#ifdef SHEEN_TEXTURE_ROUGHNESS\nuniform mat4 sheenRoughnessMatrix;\n#endif\n#endif\n#ifdef SUBSURFACE\n#ifdef SS_REFRACTION\nuniform vec4 vRefractionMicrosurfaceInfos;uniform vec4 vRefractionInfos;uniform mat4 refractionMatrix;\n#ifdef REALTIME_FILTERING\nuniform vec2 vRefractionFilteringInfo;\n#endif\n#endif\n#ifdef SS_THICKNESSANDMASK_TEXTURE\nuniform vec2 vThicknessInfos;uniform mat4 thicknessMatrix;\n#endif\n#ifdef SS_REFRACTIONINTENSITY_TEXTURE\nuniform vec2 vRefractionIntensityInfos;uniform mat4 refractionIntensityMatrix;\n#endif\n#ifdef SS_TRANSLUCENCYINTENSITY_TEXTURE\nuniform vec2 vTranslucencyIntensityInfos;uniform mat4 translucencyIntensityMatrix;\n#endif\nuniform vec2 vThicknessParam;uniform vec3 vDiffusionDistance;uniform vec4 vTintColor;uniform vec3 vSubSurfaceIntensity;\n#endif\n#ifdef PREPASS\n#ifdef SS_SCATTERING\nuniform float scatteringDiffusionProfile;\n#endif\n#endif\n#if DEBUGMODE>0\nuniform vec2 vDebugMode;\n#endif\n#ifdef DETAIL\nuniform vec4 vDetailInfos;\n#endif\n#include\n#ifdef USESPHERICALFROMREFLECTIONMAP\n#ifdef SPHERICAL_HARMONICS\nuniform vec3 vSphericalL00;uniform vec3 vSphericalL1_1;uniform vec3 vSphericalL10;uniform vec3 vSphericalL11;uniform vec3 vSphericalL2_2;uniform vec3 vSphericalL2_1;uniform vec3 vSphericalL20;uniform vec3 vSphericalL21;uniform vec3 vSphericalL22;\n#else\nuniform vec3 vSphericalX;uniform vec3 vSphericalY;uniform vec3 vSphericalZ;uniform vec3 vSphericalXX_ZZ;uniform vec3 vSphericalYY_ZZ;uniform vec3 vSphericalZZ;uniform vec3 vSphericalXY;uniform vec3 vSphericalYZ;uniform vec3 vSphericalZX;\n#endif\n#endif\n#define ADDITIONAL_FRAGMENT_DECLARATION\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$13] = shader$13;\n\n // Do not edit.\n const name$14 = \"pbrUboDeclaration\";\n const shader$14 = `layout(std140,column_major) uniform;uniform Material {vec2 vAlbedoInfos;vec4 vAmbientInfos;vec2 vOpacityInfos;vec2 vEmissiveInfos;vec2 vLightmapInfos;vec3 vReflectivityInfos;vec2 vMicroSurfaceSamplerInfos;vec2 vReflectionInfos;vec2 vReflectionFilteringInfo;vec3 vReflectionPosition;vec3 vReflectionSize;vec3 vBumpInfos;mat4 albedoMatrix;mat4 ambientMatrix;mat4 opacityMatrix;mat4 emissiveMatrix;mat4 lightmapMatrix;mat4 reflectivityMatrix;mat4 microSurfaceSamplerMatrix;mat4 bumpMatrix;vec2 vTangentSpaceParams;mat4 reflectionMatrix;vec3 vReflectionColor;vec4 vAlbedoColor;vec4 vLightingIntensity;vec3 vReflectionMicrosurfaceInfos;float pointSize;vec4 vReflectivityColor;vec3 vEmissiveColor;vec3 vAmbientColor;vec2 vDebugMode;vec4 vMetallicReflectanceFactors;vec2 vMetallicReflectanceInfos;mat4 metallicReflectanceMatrix;vec2 vReflectanceInfos;mat4 reflectanceMatrix;vec3 vSphericalL00;vec3 vSphericalL1_1;vec3 vSphericalL10;vec3 vSphericalL11;vec3 vSphericalL2_2;vec3 vSphericalL2_1;vec3 vSphericalL20;vec3 vSphericalL21;vec3 vSphericalL22;vec3 vSphericalX;vec3 vSphericalY;vec3 vSphericalZ;vec3 vSphericalXX_ZZ;vec3 vSphericalYY_ZZ;vec3 vSphericalZZ;vec3 vSphericalXY;vec3 vSphericalYZ;vec3 vSphericalZX;\n#define ADDITIONAL_UBO_DECLARATION\n};\n#include\n#include\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$14] = shader$14;\n\n // Do not edit.\n const name$15 = \"pbrFragmentExtraDeclaration\";\n const shader$15 = `varying vec3 vPositionW;\n#if DEBUGMODE>0\nvarying vec4 vClipSpacePosition;\n#endif\n#include[1..7]\n#ifdef NORMAL\nvarying vec3 vNormalW;\n#if defined(USESPHERICALFROMREFLECTIONMAP) && defined(USESPHERICALINVERTEX)\nvarying vec3 vEnvironmentIrradiance;\n#endif\n#endif\n#if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES)\nvarying vec4 vColor;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$15] = shader$15;\n\n // Do not edit.\n const name$16 = \"samplerFragmentAlternateDeclaration\";\n const shader$16 = `#ifdef _DEFINENAME_\n#if _DEFINENAME_DIRECTUV==1\n#define v_VARYINGNAME_UV vMainUV1\n#elif _DEFINENAME_DIRECTUV==2\n#define v_VARYINGNAME_UV vMainUV2\n#elif _DEFINENAME_DIRECTUV==3\n#define v_VARYINGNAME_UV vMainUV3\n#elif _DEFINENAME_DIRECTUV==4\n#define v_VARYINGNAME_UV vMainUV4\n#elif _DEFINENAME_DIRECTUV==5\n#define v_VARYINGNAME_UV vMainUV5\n#elif _DEFINENAME_DIRECTUV==6\n#define v_VARYINGNAME_UV vMainUV6\n#else\nvarying vec2 v_VARYINGNAME_UV;\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$16] = shader$16;\n\n // Do not edit.\n const name$17 = \"pbrFragmentSamplersDeclaration\";\n const shader$17 = `#include(_DEFINENAME_,ALBEDO,_VARYINGNAME_,Albedo,_SAMPLERNAME_,albedo)\n#include(_DEFINENAME_,AMBIENT,_VARYINGNAME_,Ambient,_SAMPLERNAME_,ambient)\n#include(_DEFINENAME_,OPACITY,_VARYINGNAME_,Opacity,_SAMPLERNAME_,opacity)\n#include(_DEFINENAME_,EMISSIVE,_VARYINGNAME_,Emissive,_SAMPLERNAME_,emissive)\n#include(_DEFINENAME_,LIGHTMAP,_VARYINGNAME_,Lightmap,_SAMPLERNAME_,lightmap)\n#include(_DEFINENAME_,REFLECTIVITY,_VARYINGNAME_,Reflectivity,_SAMPLERNAME_,reflectivity)\n#include(_DEFINENAME_,MICROSURFACEMAP,_VARYINGNAME_,MicroSurfaceSampler,_SAMPLERNAME_,microSurface)\n#include(_DEFINENAME_,METALLIC_REFLECTANCE,_VARYINGNAME_,MetallicReflectance,_SAMPLERNAME_,metallicReflectance)\n#include(_DEFINENAME_,REFLECTANCE,_VARYINGNAME_,Reflectance,_SAMPLERNAME_,reflectance)\n#include(_DEFINENAME_,DECAL,_VARYINGNAME_,Decal,_SAMPLERNAME_,decal)\n#ifdef CLEARCOAT\n#include(_DEFINENAME_,CLEARCOAT_TEXTURE,_VARYINGNAME_,ClearCoat,_SAMPLERNAME_,clearCoat)\n#include(_DEFINENAME_,CLEARCOAT_TEXTURE_ROUGHNESS,_VARYINGNAME_,ClearCoatRoughness)\n#if defined(CLEARCOAT_TEXTURE_ROUGHNESS) && !defined(CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL)\nuniform sampler2D clearCoatRoughnessSampler;\n#endif\n#include(_DEFINENAME_,CLEARCOAT_BUMP,_VARYINGNAME_,ClearCoatBump,_SAMPLERNAME_,clearCoatBump)\n#include(_DEFINENAME_,CLEARCOAT_TINT_TEXTURE,_VARYINGNAME_,ClearCoatTint,_SAMPLERNAME_,clearCoatTint)\n#endif\n#ifdef IRIDESCENCE\n#include(_DEFINENAME_,IRIDESCENCE_TEXTURE,_VARYINGNAME_,Iridescence,_SAMPLERNAME_,iridescence)\n#include(_DEFINENAME_,IRIDESCENCE_THICKNESS_TEXTURE,_VARYINGNAME_,IridescenceThickness,_SAMPLERNAME_,iridescenceThickness)\n#endif\n#ifdef SHEEN\n#include(_DEFINENAME_,SHEEN_TEXTURE,_VARYINGNAME_,Sheen,_SAMPLERNAME_,sheen)\n#include(_DEFINENAME_,SHEEN_TEXTURE_ROUGHNESS,_VARYINGNAME_,SheenRoughness)\n#if defined(SHEEN_ROUGHNESS) && defined(SHEEN_TEXTURE_ROUGHNESS) && !defined(SHEEN_TEXTURE_ROUGHNESS_IDENTICAL)\nuniform sampler2D sheenRoughnessSampler;\n#endif\n#endif\n#ifdef ANISOTROPIC\n#include(_DEFINENAME_,ANISOTROPIC_TEXTURE,_VARYINGNAME_,Anisotropy,_SAMPLERNAME_,anisotropy)\n#endif\n#ifdef REFLECTION\n#ifdef REFLECTIONMAP_3D\n#define sampleReflection(s,c) textureCube(s,c)\nuniform samplerCube reflectionSampler;\n#ifdef LODBASEDMICROSFURACE\n#define sampleReflectionLod(s,c,l) textureCubeLodEXT(s,c,l)\n#else\nuniform samplerCube reflectionSamplerLow;uniform samplerCube reflectionSamplerHigh;\n#endif\n#ifdef USEIRRADIANCEMAP\nuniform samplerCube irradianceSampler;\n#endif\n#else\n#define sampleReflection(s,c) texture2D(s,c)\nuniform sampler2D reflectionSampler;\n#ifdef LODBASEDMICROSFURACE\n#define sampleReflectionLod(s,c,l) texture2DLodEXT(s,c,l)\n#else\nuniform sampler2D reflectionSamplerLow;uniform sampler2D reflectionSamplerHigh;\n#endif\n#ifdef USEIRRADIANCEMAP\nuniform sampler2D irradianceSampler;\n#endif\n#endif\n#ifdef REFLECTIONMAP_SKYBOX\nvarying vec3 vPositionUVW;\n#else\n#if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED)\nvarying vec3 vDirectionW;\n#endif\n#endif\n#endif\n#ifdef ENVIRONMENTBRDF\nuniform sampler2D environmentBrdfSampler;\n#endif\n#ifdef SUBSURFACE\n#ifdef SS_REFRACTION\n#ifdef SS_REFRACTIONMAP_3D\n#define sampleRefraction(s,c) textureCube(s,c)\nuniform samplerCube refractionSampler;\n#ifdef LODBASEDMICROSFURACE\n#define sampleRefractionLod(s,c,l) textureCubeLodEXT(s,c,l)\n#else\nuniform samplerCube refractionSamplerLow;uniform samplerCube refractionSamplerHigh;\n#endif\n#else\n#define sampleRefraction(s,c) texture2D(s,c)\nuniform sampler2D refractionSampler;\n#ifdef LODBASEDMICROSFURACE\n#define sampleRefractionLod(s,c,l) texture2DLodEXT(s,c,l)\n#else\nuniform sampler2D refractionSamplerLow;uniform sampler2D refractionSamplerHigh;\n#endif\n#endif\n#endif\n#include(_DEFINENAME_,SS_THICKNESSANDMASK_TEXTURE,_VARYINGNAME_,Thickness,_SAMPLERNAME_,thickness)\n#include(_DEFINENAME_,SS_REFRACTIONINTENSITY_TEXTURE,_VARYINGNAME_,RefractionIntensity,_SAMPLERNAME_,refractionIntensity)\n#include(_DEFINENAME_,SS_TRANSLUCENCYINTENSITY_TEXTURE,_VARYINGNAME_,TranslucencyIntensity,_SAMPLERNAME_,translucencyIntensity)\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$17] = shader$17;\n\n // Do not edit.\n const name$18 = \"subSurfaceScatteringFunctions\";\n const shader$18 = `bool testLightingForSSS(float diffusionProfile)\n{return diffusionProfile<1.;}`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$18] = shader$18;\n\n // Do not edit.\n const name$19 = \"importanceSampling\";\n const shader$19 = `vec3 hemisphereCosSample(vec2 u) {float phi=2.*PI*u.x;float cosTheta2=1.-u.y;float cosTheta=sqrt(cosTheta2);float sinTheta=sqrt(1.-cosTheta2);return vec3(sinTheta*cos(phi),sinTheta*sin(phi),cosTheta);}\nvec3 hemisphereImportanceSampleDggx(vec2 u,float a) {float phi=2.*PI*u.x;float cosTheta2=(1.-u.y)/(1.+(a+1.)*((a-1.)*u.y));float cosTheta=sqrt(cosTheta2);float sinTheta=sqrt(1.-cosTheta2);return vec3(sinTheta*cos(phi),sinTheta*sin(phi),cosTheta);}\nvec3 hemisphereImportanceSampleDCharlie(vec2 u,float a) { \nfloat phi=2.*PI*u.x;float sinTheta=pow(u.y,a/(2.*a+1.));float cosTheta=sqrt(1.-sinTheta*sinTheta);return vec3(sinTheta*cos(phi),sinTheta*sin(phi),cosTheta);}`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$19] = shader$19;\n\n // Do not edit.\n const name$1a = \"pbrHelperFunctions\";\n const shader$1a = `#define RECIPROCAL_PI2 0.15915494\n#define RECIPROCAL_PI 0.31830988618\n#define MINIMUMVARIANCE 0.0005\nfloat convertRoughnessToAverageSlope(float roughness)\n{return square(roughness)+MINIMUMVARIANCE;}\nfloat fresnelGrazingReflectance(float reflectance0) {float reflectance90=saturate(reflectance0*25.0);return reflectance90;}\nvec2 getAARoughnessFactors(vec3 normalVector) {\n#ifdef SPECULARAA\nvec3 nDfdx=dFdx(normalVector.xyz);vec3 nDfdy=dFdy(normalVector.xyz);float slopeSquare=max(dot(nDfdx,nDfdx),dot(nDfdy,nDfdy));float geometricRoughnessFactor=pow(saturate(slopeSquare),0.333);float geometricAlphaGFactor=sqrt(slopeSquare);geometricAlphaGFactor*=0.75;return vec2(geometricRoughnessFactor,geometricAlphaGFactor);\n#else\nreturn vec2(0.);\n#endif\n}\n#ifdef ANISOTROPIC\n#ifdef ANISOTROPIC_LEGACY\nvec2 getAnisotropicRoughness(float alphaG,float anisotropy) {float alphaT=max(alphaG*(1.0+anisotropy),MINIMUMVARIANCE);float alphaB=max(alphaG*(1.0-anisotropy),MINIMUMVARIANCE);return vec2(alphaT,alphaB);}\nvec3 getAnisotropicBentNormals(const vec3 T,const vec3 B,const vec3 N,const vec3 V,float anisotropy,float roughness) {vec3 anisotropicFrameDirection=anisotropy>=0.0 ? B : T;vec3 anisotropicFrameTangent=cross(normalize(anisotropicFrameDirection),V);vec3 anisotropicFrameNormal=cross(anisotropicFrameTangent,anisotropicFrameDirection);vec3 anisotropicNormal=normalize(mix(N,anisotropicFrameNormal,abs(anisotropy)));return anisotropicNormal;}\n#else\nvec2 getAnisotropicRoughness(float alphaG,float anisotropy) {float alphaT=max(mix(alphaG,1.0,anisotropy*anisotropy),MINIMUMVARIANCE);float alphaB=max(alphaG,MINIMUMVARIANCE);return vec2(alphaT,alphaB);}\nvec3 getAnisotropicBentNormals(const vec3 T,const vec3 B,const vec3 N,const vec3 V,float anisotropy,float roughness) {vec3 bentNormal=cross(B,V);bentNormal=normalize(cross(bentNormal,B));float a=square(square(1.0-anisotropy*(1.0-roughness)));bentNormal=normalize(mix(bentNormal,N,a));return bentNormal;}\n#endif\n#endif\n#if defined(CLEARCOAT) || defined(SS_REFRACTION)\nvec3 cocaLambert(vec3 alpha,float distance) {return exp(-alpha*distance);}\nvec3 cocaLambert(float NdotVRefract,float NdotLRefract,vec3 alpha,float thickness) {return cocaLambert(alpha,(thickness*((NdotLRefract+NdotVRefract)/(NdotLRefract*NdotVRefract))));}\nvec3 computeColorAtDistanceInMedia(vec3 color,float distance) {return -log(color)/distance;}\nvec3 computeClearCoatAbsorption(float NdotVRefract,float NdotLRefract,vec3 clearCoatColor,float clearCoatThickness,float clearCoatIntensity) {vec3 clearCoatAbsorption=mix(vec3(1.0),\ncocaLambert(NdotVRefract,NdotLRefract,clearCoatColor,clearCoatThickness),\nclearCoatIntensity);return clearCoatAbsorption;}\n#endif\n#ifdef MICROSURFACEAUTOMATIC\nfloat computeDefaultMicroSurface(float microSurface,vec3 reflectivityColor)\n{const float kReflectivityNoAlphaWorkflow_SmoothnessMax=0.95;float reflectivityLuminance=getLuminance(reflectivityColor);float reflectivityLuma=sqrt(reflectivityLuminance);microSurface=reflectivityLuma*kReflectivityNoAlphaWorkflow_SmoothnessMax;return microSurface;}\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1a] = shader$1a;\n\n // Do not edit.\n const name$1b = \"harmonicsFunctions\";\n const shader$1b = `#ifdef USESPHERICALFROMREFLECTIONMAP\n#ifdef SPHERICAL_HARMONICS\nvec3 computeEnvironmentIrradiance(vec3 normal) {return vSphericalL00\n+ vSphericalL1_1*(normal.y)\n+ vSphericalL10*(normal.z)\n+ vSphericalL11*(normal.x)\n+ vSphericalL2_2*(normal.y*normal.x)\n+ vSphericalL2_1*(normal.y*normal.z)\n+ vSphericalL20*((3.0*normal.z*normal.z)-1.0)\n+ vSphericalL21*(normal.z*normal.x)\n+ vSphericalL22*(normal.x*normal.x-(normal.y*normal.y));}\n#else\nvec3 computeEnvironmentIrradiance(vec3 normal) {float Nx=normal.x;float Ny=normal.y;float Nz=normal.z;vec3 C1=vSphericalZZ.rgb;vec3 Cx=vSphericalX.rgb;vec3 Cy=vSphericalY.rgb;vec3 Cz=vSphericalZ.rgb;vec3 Cxx_zz=vSphericalXX_ZZ.rgb;vec3 Cyy_zz=vSphericalYY_ZZ.rgb;vec3 Cxy=vSphericalXY.rgb;vec3 Cyz=vSphericalYZ.rgb;vec3 Czx=vSphericalZX.rgb;vec3 a1=Cyy_zz*Ny+Cy;vec3 a2=Cyz*Nz+a1;vec3 b1=Czx*Nz+Cx;vec3 b2=Cxy*Ny+b1;vec3 b3=Cxx_zz*Nx+b2;vec3 t1=Cz *Nz+C1;vec3 t2=a2 *Ny+t1;vec3 t3=b3 *Nx+t2;return t3;}\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1b] = shader$1b;\n\n // Do not edit.\n const name$1c = \"pbrDirectLightingSetupFunctions\";\n const shader$1c = `struct preLightingInfo\n{vec3 lightOffset;float lightDistanceSquared;float lightDistance;float attenuation;vec3 L;vec3 H;float NdotV;float NdotLUnclamped;float NdotL;float VdotH;float roughness;\n#ifdef IRIDESCENCE\nfloat iridescenceIntensity;\n#endif\n};preLightingInfo computePointAndSpotPreLightingInfo(vec4 lightData,vec3 V,vec3 N) {preLightingInfo result;result.lightOffset=lightData.xyz-vPositionW;result.lightDistanceSquared=dot(result.lightOffset,result.lightOffset);result.lightDistance=sqrt(result.lightDistanceSquared);result.L=normalize(result.lightOffset);result.H=normalize(V+result.L);result.VdotH=saturate(dot(V,result.H));result.NdotLUnclamped=dot(N,result.L);result.NdotL=saturateEps(result.NdotLUnclamped);return result;}\npreLightingInfo computeDirectionalPreLightingInfo(vec4 lightData,vec3 V,vec3 N) {preLightingInfo result;result.lightDistance=length(-lightData.xyz);result.L=normalize(-lightData.xyz);result.H=normalize(V+result.L);result.VdotH=saturate(dot(V,result.H));result.NdotLUnclamped=dot(N,result.L);result.NdotL=saturateEps(result.NdotLUnclamped);return result;}\npreLightingInfo computeHemisphericPreLightingInfo(vec4 lightData,vec3 V,vec3 N) {preLightingInfo result;result.NdotL=dot(N,lightData.xyz)*0.5+0.5;result.NdotL=saturateEps(result.NdotL);result.NdotLUnclamped=result.NdotL;\n#ifdef SPECULARTERM\nresult.L=normalize(lightData.xyz);result.H=normalize(V+result.L);result.VdotH=saturate(dot(V,result.H));\n#endif\nreturn result;}`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1c] = shader$1c;\n\n // Do not edit.\n const name$1d = \"pbrDirectLightingFalloffFunctions\";\n const shader$1d = `float computeDistanceLightFalloff_Standard(vec3 lightOffset,float range)\n{return max(0.,1.0-length(lightOffset)/range);}\nfloat computeDistanceLightFalloff_Physical(float lightDistanceSquared)\n{return 1.0/maxEps(lightDistanceSquared);}\nfloat computeDistanceLightFalloff_GLTF(float lightDistanceSquared,float inverseSquaredRange)\n{float lightDistanceFalloff=1.0/maxEps(lightDistanceSquared);float factor=lightDistanceSquared*inverseSquaredRange;float attenuation=saturate(1.0-factor*factor);attenuation*=attenuation;lightDistanceFalloff*=attenuation;return lightDistanceFalloff;}\nfloat computeDistanceLightFalloff(vec3 lightOffset,float lightDistanceSquared,float range,float inverseSquaredRange)\n{\n#ifdef USEPHYSICALLIGHTFALLOFF\nreturn computeDistanceLightFalloff_Physical(lightDistanceSquared);\n#elif defined(USEGLTFLIGHTFALLOFF)\nreturn computeDistanceLightFalloff_GLTF(lightDistanceSquared,inverseSquaredRange);\n#else\nreturn computeDistanceLightFalloff_Standard(lightOffset,range);\n#endif\n}\nfloat computeDirectionalLightFalloff_Standard(vec3 lightDirection,vec3 directionToLightCenterW,float cosHalfAngle,float exponent)\n{float falloff=0.0;float cosAngle=maxEps(dot(-lightDirection,directionToLightCenterW));if (cosAngle>=cosHalfAngle)\n{falloff=max(0.,pow(cosAngle,exponent));}\nreturn falloff;}\nfloat computeDirectionalLightFalloff_Physical(vec3 lightDirection,vec3 directionToLightCenterW,float cosHalfAngle)\n{const float kMinusLog2ConeAngleIntensityRatio=6.64385618977; \nfloat concentrationKappa=kMinusLog2ConeAngleIntensityRatio/(1.0-cosHalfAngle);vec4 lightDirectionSpreadSG=vec4(-lightDirection*concentrationKappa,-concentrationKappa);float falloff=exp2(dot(vec4(directionToLightCenterW,1.0),lightDirectionSpreadSG));return falloff;}\nfloat computeDirectionalLightFalloff_GLTF(vec3 lightDirection,vec3 directionToLightCenterW,float lightAngleScale,float lightAngleOffset)\n{float cd=dot(-lightDirection,directionToLightCenterW);float falloff=saturate(cd*lightAngleScale+lightAngleOffset);falloff*=falloff;return falloff;}\nfloat computeDirectionalLightFalloff(vec3 lightDirection,vec3 directionToLightCenterW,float cosHalfAngle,float exponent,float lightAngleScale,float lightAngleOffset)\n{\n#ifdef USEPHYSICALLIGHTFALLOFF\nreturn computeDirectionalLightFalloff_Physical(lightDirection,directionToLightCenterW,cosHalfAngle);\n#elif defined(USEGLTFLIGHTFALLOFF)\nreturn computeDirectionalLightFalloff_GLTF(lightDirection,directionToLightCenterW,lightAngleScale,lightAngleOffset);\n#else\nreturn computeDirectionalLightFalloff_Standard(lightDirection,directionToLightCenterW,cosHalfAngle,exponent);\n#endif\n}`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1d] = shader$1d;\n\n // Do not edit.\n const name$1e = \"pbrBRDFFunctions\";\n const shader$1e = `#define FRESNEL_MAXIMUM_ON_ROUGH 0.25\n#ifdef MS_BRDF_ENERGY_CONSERVATION\nvec3 getEnergyConservationFactor(const vec3 specularEnvironmentR0,const vec3 environmentBrdf) {return 1.0+specularEnvironmentR0*(1.0/environmentBrdf.y-1.0);}\n#endif\n#ifdef ENVIRONMENTBRDF\nvec3 getBRDFLookup(float NdotV,float perceptualRoughness) {vec2 UV=vec2(NdotV,perceptualRoughness);vec4 brdfLookup=texture2D(environmentBrdfSampler,UV);\n#ifdef ENVIRONMENTBRDF_RGBD\nbrdfLookup.rgb=fromRGBD(brdfLookup.rgba);\n#endif\nreturn brdfLookup.rgb;}\nvec3 getReflectanceFromBRDFLookup(const vec3 specularEnvironmentR0,const vec3 specularEnvironmentR90,const vec3 environmentBrdf) {\n#ifdef BRDF_V_HEIGHT_CORRELATED\nvec3 reflectance=(specularEnvironmentR90-specularEnvironmentR0)*environmentBrdf.x+specularEnvironmentR0*environmentBrdf.y;\n#else\nvec3 reflectance=specularEnvironmentR0*environmentBrdf.x+specularEnvironmentR90*environmentBrdf.y;\n#endif\nreturn reflectance;}\nvec3 getReflectanceFromBRDFLookup(const vec3 specularEnvironmentR0,const vec3 environmentBrdf) {\n#ifdef BRDF_V_HEIGHT_CORRELATED\nvec3 reflectance=mix(environmentBrdf.xxx,environmentBrdf.yyy,specularEnvironmentR0);\n#else\nvec3 reflectance=specularEnvironmentR0*environmentBrdf.x+environmentBrdf.y;\n#endif\nreturn reflectance;}\n#endif\n/* NOT USED\n#if defined(SHEEN) && defined(SHEEN_SOFTER)\nfloat getBRDFLookupCharlieSheen(float NdotV,float perceptualRoughness)\n{float c=1.0-NdotV;float c3=c*c*c;return 0.65584461*c3+1.0/(4.16526551+exp(-7.97291361*perceptualRoughness+6.33516894));}\n#endif\n*/\n#if !defined(ENVIRONMENTBRDF) || defined(REFLECTIONMAP_SKYBOX) || defined(ALPHAFRESNEL)\nvec3 getReflectanceFromAnalyticalBRDFLookup_Jones(float VdotN,vec3 reflectance0,vec3 reflectance90,float smoothness)\n{float weight=mix(FRESNEL_MAXIMUM_ON_ROUGH,1.0,smoothness);return reflectance0+weight*(reflectance90-reflectance0)*pow5(saturate(1.0-VdotN));}\n#endif\n#if defined(SHEEN) && defined(ENVIRONMENTBRDF)\n/**\n* The sheen BRDF not containing F can be easily stored in the blue channel of the BRDF texture.\n* The blue channel contains DCharlie*VAshikhmin*NdotL as a lokkup table\n*/\nvec3 getSheenReflectanceFromBRDFLookup(const vec3 reflectance0,const vec3 environmentBrdf) {vec3 sheenEnvironmentReflectance=reflectance0*environmentBrdf.b;return sheenEnvironmentReflectance;}\n#endif\nvec3 fresnelSchlickGGX(float VdotH,vec3 reflectance0,vec3 reflectance90)\n{return reflectance0+(reflectance90-reflectance0)*pow5(1.0-VdotH);}\nfloat fresnelSchlickGGX(float VdotH,float reflectance0,float reflectance90)\n{return reflectance0+(reflectance90-reflectance0)*pow5(1.0-VdotH);}\n#ifdef CLEARCOAT\nvec3 getR0RemappedForClearCoat(vec3 f0) {\n#ifdef CLEARCOAT_DEFAULTIOR\n#ifdef MOBILE\nreturn saturate(f0*(f0*0.526868+0.529324)-0.0482256);\n#else\nreturn saturate(f0*(f0*(0.941892-0.263008*f0)+0.346479)-0.0285998);\n#endif\n#else\nvec3 s=sqrt(f0);vec3 t=(vClearCoatRefractionParams.z+vClearCoatRefractionParams.w*s)/(vClearCoatRefractionParams.w+vClearCoatRefractionParams.z*s);return square(t);\n#endif\n}\n#endif\n#ifdef IRIDESCENCE\nconst mat3 XYZ_TO_REC709=mat3(\n3.2404542,-0.9692660, 0.0556434,\n-1.5371385, 1.8760108,-0.2040259,\n-0.4985314, 0.0415560, 1.0572252\n);vec3 getIORTfromAirToSurfaceR0(vec3 f0) {vec3 sqrtF0=sqrt(f0);return (1.+sqrtF0)/(1.-sqrtF0);}\nvec3 getR0fromIORs(vec3 iorT,float iorI) {return square((iorT-vec3(iorI))/(iorT+vec3(iorI)));}\nfloat getR0fromIORs(float iorT,float iorI) {return square((iorT-iorI)/(iorT+iorI));}\nvec3 evalSensitivity(float opd,vec3 shift) {float phase=2.0*PI*opd*1.0e-9;const vec3 val=vec3(5.4856e-13,4.4201e-13,5.2481e-13);const vec3 pos=vec3(1.6810e+06,1.7953e+06,2.2084e+06);const vec3 var=vec3(4.3278e+09,9.3046e+09,6.6121e+09);vec3 xyz=val*sqrt(2.0*PI*var)*cos(pos*phase+shift)*exp(-square(phase)*var);xyz.x+=9.7470e-14*sqrt(2.0*PI*4.5282e+09)*cos(2.2399e+06*phase+shift[0])*exp(-4.5282e+09*square(phase));xyz/=1.0685e-7;vec3 srgb=XYZ_TO_REC709*xyz;return srgb;}\nvec3 evalIridescence(float outsideIOR,float eta2,float cosTheta1,float thinFilmThickness,vec3 baseF0) {vec3 I=vec3(1.0);float iridescenceIOR=mix(outsideIOR,eta2,smoothstep(0.0,0.03,thinFilmThickness));float sinTheta2Sq=square(outsideIOR/iridescenceIOR)*(1.0-square(cosTheta1));float cosTheta2Sq=1.0-sinTheta2Sq;if (cosTheta2Sq<0.0) {return I;}\nfloat cosTheta2=sqrt(cosTheta2Sq);float R0=getR0fromIORs(iridescenceIOR,outsideIOR);float R12=fresnelSchlickGGX(cosTheta1,R0,1.);float R21=R12;float T121=1.0-R12;float phi12=0.0;if (iridescenceIOR0\n#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE)\nfloat radicalInverse_VdC(uint bits) \n{bits=(bits<<16u) | (bits>>16u);bits=((bits & 0x55555555u)<<1u) | ((bits & 0xAAAAAAAAu)>>1u);bits=((bits & 0x33333333u)<<2u) | ((bits & 0xCCCCCCCCu)>>2u);bits=((bits & 0x0F0F0F0Fu)<<4u) | ((bits & 0xF0F0F0F0u)>>4u);bits=((bits & 0x00FF00FFu)<<8u) | ((bits & 0xFF00FF00u)>>8u);return float(bits)*2.3283064365386963e-10; }\nvec2 hammersley(uint i,uint N)\n{return vec2(float(i)/float(N),radicalInverse_VdC(i));}\n#else\nfloat vanDerCorpus(int n,int base)\n{float invBase=1.0/float(base);float denom =1.0;float result =0.0;for(int i=0; i<32; ++i)\n{if(n>0)\n{denom =mod(float(n),2.0);result+=denom*invBase;invBase=invBase/2.0;n =int(float(n)/2.0);}}\nreturn result;}\nvec2 hammersley(int i,int N)\n{return vec2(float(i)/float(N),vanDerCorpus(i,2));}\n#endif\nfloat log4(float x) {return log2(x)/2.;}\nconst float NUM_SAMPLES_FLOAT=float(NUM_SAMPLES);const float NUM_SAMPLES_FLOAT_INVERSED=1./NUM_SAMPLES_FLOAT;const float K=4.;\n#define inline\nvec3 irradiance(samplerCube inputTexture,vec3 inputN,vec2 filteringInfo)\n{vec3 n=normalize(inputN);vec3 result=vec3(0.0);vec3 tangent=abs(n.z)<0.999 ? vec3(0.,0.,1.) : vec3(1.,0.,0.);tangent=normalize(cross(tangent,n));vec3 bitangent=cross(n,tangent);mat3 tbn=mat3(tangent,bitangent,n);float maxLevel=filteringInfo.y;float dim0=filteringInfo.x;float omegaP=(4.*PI)/(6.*dim0*dim0);\n#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE)\nfor(uint i=0u; i0.) {float pdf_inversed=PI/NoL;float omegaS=NUM_SAMPLES_FLOAT_INVERSED*pdf_inversed;float l=log4(omegaS)-log4(omegaP)+log4(K);float mipLevel=clamp(l,0.0,maxLevel);vec3 c=textureCubeLodEXT(inputTexture,tbn*Ls,mipLevel).rgb;\n#ifdef GAMMA_INPUT\nc=toLinearSpace(c);\n#endif\nresult+=c;}}\nresult=result*NUM_SAMPLES_FLOAT_INVERSED;return result;}\n#define inline\nvec3 radiance(float alphaG,samplerCube inputTexture,vec3 inputN,vec2 filteringInfo)\n{vec3 n=normalize(inputN);if (alphaG==0.) {vec3 c=textureCube(inputTexture,n).rgb;\n#ifdef GAMMA_INPUT\nc=toLinearSpace(c);\n#endif\nreturn c;} else {vec3 result=vec3(0.);vec3 tangent=abs(n.z)<0.999 ? vec3(0.,0.,1.) : vec3(1.,0.,0.);tangent=normalize(cross(tangent,n));vec3 bitangent=cross(n,tangent);mat3 tbn=mat3(tangent,bitangent,n);float maxLevel=filteringInfo.y;float dim0=filteringInfo.x;float omegaP=(4.*PI)/(6.*dim0*dim0);float weight=0.;\n#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE)\nfor(uint i=0u; i0.) {float pdf_inversed=4./normalDistributionFunction_TrowbridgeReitzGGX(NoH,alphaG);float omegaS=NUM_SAMPLES_FLOAT_INVERSED*pdf_inversed;float l=log4(omegaS)-log4(omegaP)+log4(K);float mipLevel=clamp(float(l),0.0,maxLevel);weight+=NoL;vec3 c=textureCubeLodEXT(inputTexture,tbn*L,mipLevel).rgb;\n#ifdef GAMMA_INPUT\nc=toLinearSpace(c);\n#endif\nresult+=c*NoL;}}\nresult=result/weight;return result;}}\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1f] = shader$1f;\n\n // Do not edit.\n const name$1g = \"pbrDirectLightingFunctions\";\n const shader$1g = `#define CLEARCOATREFLECTANCE90 1.0\nstruct lightingInfo\n{vec3 diffuse;\n#ifdef SPECULARTERM\nvec3 specular;\n#endif\n#ifdef CLEARCOAT\nvec4 clearCoat;\n#endif\n#ifdef SHEEN\nvec3 sheen;\n#endif\n};float adjustRoughnessFromLightProperties(float roughness,float lightRadius,float lightDistance) {\n#if defined(USEPHYSICALLIGHTFALLOFF) || defined(USEGLTFLIGHTFALLOFF)\nfloat lightRoughness=lightRadius/lightDistance;float totalRoughness=saturate(lightRoughness+roughness);return totalRoughness;\n#else\nreturn roughness;\n#endif\n}\nvec3 computeHemisphericDiffuseLighting(preLightingInfo info,vec3 lightColor,vec3 groundColor) {return mix(groundColor,lightColor,info.NdotL);}\nvec3 computeDiffuseLighting(preLightingInfo info,vec3 lightColor) {float diffuseTerm=diffuseBRDF_Burley(info.NdotL,info.NdotV,info.VdotH,info.roughness);return diffuseTerm*info.attenuation*info.NdotL*lightColor;}\n#define inline\nvec3 computeProjectionTextureDiffuseLighting(sampler2D projectionLightSampler,mat4 textureProjectionMatrix){vec4 strq=textureProjectionMatrix*vec4(vPositionW,1.0);strq/=strq.w;vec3 textureColor=texture2D(projectionLightSampler,strq.xy).rgb;return toLinearSpace(textureColor);}\n#ifdef SS_TRANSLUCENCY\nvec3 computeDiffuseAndTransmittedLighting(preLightingInfo info,vec3 lightColor,vec3 transmittance) {float NdotL=absEps(info.NdotLUnclamped);float wrapNdotL=computeWrappedDiffuseNdotL(NdotL,0.02);float trAdapt=step(0.,info.NdotLUnclamped);vec3 transmittanceNdotL=mix(transmittance*wrapNdotL,vec3(wrapNdotL),trAdapt);float diffuseTerm=diffuseBRDF_Burley(NdotL,info.NdotV,info.VdotH,info.roughness);return diffuseTerm*transmittanceNdotL*info.attenuation*lightColor;}\n#endif\n#ifdef SPECULARTERM\nvec3 computeSpecularLighting(preLightingInfo info,vec3 N,vec3 reflectance0,vec3 reflectance90,float geometricRoughnessFactor,vec3 lightColor) {float NdotH=saturateEps(dot(N,info.H));float roughness=max(info.roughness,geometricRoughnessFactor);float alphaG=convertRoughnessToAverageSlope(roughness);vec3 fresnel=fresnelSchlickGGX(info.VdotH,reflectance0,reflectance90);\n#ifdef IRIDESCENCE\nfresnel=mix(fresnel,reflectance0,info.iridescenceIntensity);\n#endif\nfloat distribution=normalDistributionFunction_TrowbridgeReitzGGX(NdotH,alphaG);\n#ifdef BRDF_V_HEIGHT_CORRELATED\nfloat smithVisibility=smithVisibility_GGXCorrelated(info.NdotL,info.NdotV,alphaG);\n#else\nfloat smithVisibility=smithVisibility_TrowbridgeReitzGGXFast(info.NdotL,info.NdotV,alphaG);\n#endif\nvec3 specTerm=fresnel*distribution*smithVisibility;return specTerm*info.attenuation*info.NdotL*lightColor;}\n#endif\n#ifdef ANISOTROPIC\nvec3 computeAnisotropicSpecularLighting(preLightingInfo info,vec3 V,vec3 N,vec3 T,vec3 B,float anisotropy,vec3 reflectance0,vec3 reflectance90,float geometricRoughnessFactor,vec3 lightColor) {float NdotH=saturateEps(dot(N,info.H));float TdotH=dot(T,info.H);float BdotH=dot(B,info.H);float TdotV=dot(T,V);float BdotV=dot(B,V);float TdotL=dot(T,info.L);float BdotL=dot(B,info.L);float alphaG=convertRoughnessToAverageSlope(info.roughness);vec2 alphaTB=getAnisotropicRoughness(alphaG,anisotropy);alphaTB=max(alphaTB,square(geometricRoughnessFactor));vec3 fresnel=fresnelSchlickGGX(info.VdotH,reflectance0,reflectance90);\n#ifdef IRIDESCENCE\nfresnel=mix(fresnel,reflectance0,info.iridescenceIntensity);\n#endif\nfloat distribution=normalDistributionFunction_BurleyGGX_Anisotropic(NdotH,TdotH,BdotH,alphaTB);float smithVisibility=smithVisibility_GGXCorrelated_Anisotropic(info.NdotL,info.NdotV,TdotV,BdotV,TdotL,BdotL,alphaTB);vec3 specTerm=fresnel*distribution*smithVisibility;return specTerm*info.attenuation*info.NdotL*lightColor;}\n#endif\n#ifdef CLEARCOAT\nvec4 computeClearCoatLighting(preLightingInfo info,vec3 Ncc,float geometricRoughnessFactor,float clearCoatIntensity,vec3 lightColor) {float NccdotL=saturateEps(dot(Ncc,info.L));float NccdotH=saturateEps(dot(Ncc,info.H));float clearCoatRoughness=max(info.roughness,geometricRoughnessFactor);float alphaG=convertRoughnessToAverageSlope(clearCoatRoughness);float fresnel=fresnelSchlickGGX(info.VdotH,vClearCoatRefractionParams.x,CLEARCOATREFLECTANCE90);fresnel*=clearCoatIntensity;float distribution=normalDistributionFunction_TrowbridgeReitzGGX(NccdotH,alphaG);float kelemenVisibility=visibility_Kelemen(info.VdotH);float clearCoatTerm=fresnel*distribution*kelemenVisibility;return vec4(\nclearCoatTerm*info.attenuation*NccdotL*lightColor,\n1.0-fresnel\n);}\nvec3 computeClearCoatLightingAbsorption(float NdotVRefract,vec3 L,vec3 Ncc,vec3 clearCoatColor,float clearCoatThickness,float clearCoatIntensity) {vec3 LRefract=-refract(L,Ncc,vClearCoatRefractionParams.y);float NdotLRefract=saturateEps(dot(Ncc,LRefract));vec3 absorption=computeClearCoatAbsorption(NdotVRefract,NdotLRefract,clearCoatColor,clearCoatThickness,clearCoatIntensity);return absorption;}\n#endif\n#ifdef SHEEN\nvec3 computeSheenLighting(preLightingInfo info,vec3 N,vec3 reflectance0,vec3 reflectance90,float geometricRoughnessFactor,vec3 lightColor) {float NdotH=saturateEps(dot(N,info.H));float roughness=max(info.roughness,geometricRoughnessFactor);float alphaG=convertRoughnessToAverageSlope(roughness);float fresnel=1.;float distribution=normalDistributionFunction_CharlieSheen(NdotH,alphaG);/*#ifdef SHEEN_SOFTER\nfloat visibility=visibility_CharlieSheen(info.NdotL,info.NdotV,alphaG);\n#else */\nfloat visibility=visibility_Ashikhmin(info.NdotL,info.NdotV);/* #endif */\nfloat sheenTerm=fresnel*distribution*visibility;return sheenTerm*info.attenuation*info.NdotL*lightColor;}\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1g] = shader$1g;\n\n // Do not edit.\n const name$1h = \"pbrIBLFunctions\";\n const shader$1h = `#if defined(REFLECTION) || defined(SS_REFRACTION)\nfloat getLodFromAlphaG(float cubeMapDimensionPixels,float microsurfaceAverageSlope) {float microsurfaceAverageSlopeTexels=cubeMapDimensionPixels*microsurfaceAverageSlope;float lod=log2(microsurfaceAverageSlopeTexels);return lod;}\nfloat getLinearLodFromRoughness(float cubeMapDimensionPixels,float roughness) {float lod=log2(cubeMapDimensionPixels)*roughness;return lod;}\n#endif\n#if defined(ENVIRONMENTBRDF) && defined(RADIANCEOCCLUSION)\nfloat environmentRadianceOcclusion(float ambientOcclusion,float NdotVUnclamped) {float temp=NdotVUnclamped+ambientOcclusion;return saturate(square(temp)-1.0+ambientOcclusion);}\n#endif\n#if defined(ENVIRONMENTBRDF) && defined(HORIZONOCCLUSION)\nfloat environmentHorizonOcclusion(vec3 view,vec3 normal,vec3 geometricNormal) {vec3 reflection=reflect(view,normal);float temp=saturate(1.0+1.1*dot(reflection,geometricNormal));return square(temp);}\n#endif\n#if defined(LODINREFLECTIONALPHA) || defined(SS_LODINREFRACTIONALPHA)\n#define UNPACK_LOD(x) (1.0-x)*255.0\nfloat getLodFromAlphaG(float cubeMapDimensionPixels,float alphaG,float NdotV) {float microsurfaceAverageSlope=alphaG;microsurfaceAverageSlope*=sqrt(abs(NdotV));return getLodFromAlphaG(cubeMapDimensionPixels,microsurfaceAverageSlope);}\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1h] = shader$1h;\n\n // Do not edit.\n const name$1i = \"pbrBlockAlbedoOpacity\";\n const shader$1i = `struct albedoOpacityOutParams\n{vec3 surfaceAlbedo;float alpha;};\n#define pbr_inline\nvoid albedoOpacityBlock(\nin vec4 vAlbedoColor,\n#ifdef ALBEDO\nin vec4 albedoTexture,\nin vec2 albedoInfos,\n#endif\n#ifdef OPACITY\nin vec4 opacityMap,\nin vec2 vOpacityInfos,\n#endif\n#ifdef DETAIL\nin vec4 detailColor,\nin vec4 vDetailInfos,\n#endif\n#ifdef DECAL\nin vec4 decalColor,\nin vec4 vDecalInfos,\n#endif\nout albedoOpacityOutParams outParams\n)\n{vec3 surfaceAlbedo=vAlbedoColor.rgb;float alpha=vAlbedoColor.a;\n#ifdef ALBEDO\n#if defined(ALPHAFROMALBEDO) || defined(ALPHATEST)\nalpha*=albedoTexture.a;\n#endif\n#ifdef GAMMAALBEDO\nsurfaceAlbedo*=toLinearSpace(albedoTexture.rgb);\n#else\nsurfaceAlbedo*=albedoTexture.rgb;\n#endif\nsurfaceAlbedo*=albedoInfos.y;\n#endif\n#ifndef DECAL_AFTER_DETAIL\n#include\n#endif\n#if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES)\nsurfaceAlbedo*=vColor.rgb;\n#endif\n#ifdef DETAIL\nfloat detailAlbedo=2.0*mix(0.5,detailColor.r,vDetailInfos.y);surfaceAlbedo.rgb=surfaceAlbedo.rgb*detailAlbedo*detailAlbedo; \n#endif\n#ifdef DECAL_AFTER_DETAIL\n#include\n#endif\n#define CUSTOM_FRAGMENT_UPDATE_ALBEDO\n#ifdef OPACITY\n#ifdef OPACITYRGB\nalpha=getLuminance(opacityMap.rgb);\n#else\nalpha*=opacityMap.a;\n#endif\nalpha*=vOpacityInfos.y;\n#endif\n#if defined(VERTEXALPHA) || defined(INSTANCESCOLOR) && defined(INSTANCES)\nalpha*=vColor.a;\n#endif\n#if !defined(SS_LINKREFRACTIONTOTRANSPARENCY) && !defined(ALPHAFRESNEL)\n#ifdef ALPHATEST \n#if DEBUGMODE != 88\nif (alpha0\n#ifdef METALLICWORKFLOW\nvec2 metallicRoughness;\n#ifdef REFLECTIVITY\nvec4 surfaceMetallicColorMap;\n#endif\n#ifndef FROSTBITE_REFLECTANCE\nvec3 metallicF0;\n#endif\n#else\n#ifdef REFLECTIVITY\nvec4 surfaceReflectivityColorMap;\n#endif\n#endif\n#endif\n};\n#define pbr_inline\nvoid reflectivityBlock(\nin vec4 vReflectivityColor,\n#ifdef METALLICWORKFLOW\nin vec3 surfaceAlbedo,\nin vec4 metallicReflectanceFactors,\n#endif\n#ifdef REFLECTIVITY\nin vec3 reflectivityInfos,\nin vec4 surfaceMetallicOrReflectivityColorMap,\n#endif\n#if defined(METALLICWORKFLOW) && defined(REFLECTIVITY) && defined(AOSTOREINMETALMAPRED)\nin vec3 ambientOcclusionColorIn,\n#endif\n#ifdef MICROSURFACEMAP\nin vec4 microSurfaceTexel,\n#endif\n#ifdef DETAIL\nin vec4 detailColor,\nin vec4 vDetailInfos,\n#endif\nout reflectivityOutParams outParams\n)\n{float microSurface=vReflectivityColor.a;vec3 surfaceReflectivityColor=vReflectivityColor.rgb;\n#ifdef METALLICWORKFLOW\nvec2 metallicRoughness=surfaceReflectivityColor.rg;\n#ifdef REFLECTIVITY\n#if DEBUGMODE>0\noutParams.surfaceMetallicColorMap=surfaceMetallicOrReflectivityColorMap;\n#endif\n#ifdef AOSTOREINMETALMAPRED\nvec3 aoStoreInMetalMap=vec3(surfaceMetallicOrReflectivityColorMap.r,surfaceMetallicOrReflectivityColorMap.r,surfaceMetallicOrReflectivityColorMap.r);outParams.ambientOcclusionColor=mix(ambientOcclusionColorIn,aoStoreInMetalMap,reflectivityInfos.z);\n#endif\n#ifdef METALLNESSSTOREINMETALMAPBLUE\nmetallicRoughness.r*=surfaceMetallicOrReflectivityColorMap.b;\n#else\nmetallicRoughness.r*=surfaceMetallicOrReflectivityColorMap.r;\n#endif\n#ifdef ROUGHNESSSTOREINMETALMAPALPHA\nmetallicRoughness.g*=surfaceMetallicOrReflectivityColorMap.a;\n#else\n#ifdef ROUGHNESSSTOREINMETALMAPGREEN\nmetallicRoughness.g*=surfaceMetallicOrReflectivityColorMap.g;\n#endif\n#endif\n#endif\n#ifdef DETAIL\nfloat detailRoughness=mix(0.5,detailColor.b,vDetailInfos.w);float loLerp=mix(0.,metallicRoughness.g,detailRoughness*2.);float hiLerp=mix(metallicRoughness.g,1.,(detailRoughness-0.5)*2.);metallicRoughness.g=mix(loLerp,hiLerp,step(detailRoughness,0.5));\n#endif\n#ifdef MICROSURFACEMAP\nmetallicRoughness.g*=microSurfaceTexel.r;\n#endif\n#if DEBUGMODE>0\noutParams.metallicRoughness=metallicRoughness;\n#endif\n#define CUSTOM_FRAGMENT_UPDATE_METALLICROUGHNESS\nmicroSurface=1.0-metallicRoughness.g;vec3 baseColor=surfaceAlbedo;\n#ifdef FROSTBITE_REFLECTANCE\noutParams.surfaceAlbedo=baseColor.rgb*(1.0-metallicRoughness.r);surfaceReflectivityColor=mix(0.16*reflectance*reflectance,baseColor,metallicRoughness.r);\n#else\nvec3 metallicF0=metallicReflectanceFactors.rgb;\n#if DEBUGMODE>0\noutParams.metallicF0=metallicF0;\n#endif\noutParams.surfaceAlbedo=mix(baseColor.rgb*(1.0-metallicF0),vec3(0.,0.,0.),metallicRoughness.r);surfaceReflectivityColor=mix(metallicF0,baseColor,metallicRoughness.r);\n#endif\n#else\n#ifdef REFLECTIVITY\nsurfaceReflectivityColor*=surfaceMetallicOrReflectivityColorMap.rgb;\n#if DEBUGMODE>0\noutParams.surfaceReflectivityColorMap=surfaceMetallicOrReflectivityColorMap;\n#endif\n#ifdef MICROSURFACEFROMREFLECTIVITYMAP\nmicroSurface*=surfaceMetallicOrReflectivityColorMap.a;microSurface*=reflectivityInfos.z;\n#else\n#ifdef MICROSURFACEAUTOMATIC\nmicroSurface*=computeDefaultMicroSurface(microSurface,surfaceReflectivityColor);\n#endif\n#ifdef MICROSURFACEMAP\nmicroSurface*=microSurfaceTexel.r;\n#endif\n#define CUSTOM_FRAGMENT_UPDATE_MICROSURFACE\n#endif\n#endif\n#endif\nmicroSurface=saturate(microSurface);float roughness=1.-microSurface;outParams.microSurface=microSurface;outParams.roughness=roughness;outParams.surfaceReflectivityColor=surfaceReflectivityColor;}\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1j] = shader$1j;\n\n // Do not edit.\n const name$1k = \"pbrBlockAmbientOcclusion\";\n const shader$1k = `struct ambientOcclusionOutParams\n{vec3 ambientOcclusionColor;\n#if DEBUGMODE>0 && defined(AMBIENT)\nvec3 ambientOcclusionColorMap;\n#endif\n};\n#define pbr_inline\nvoid ambientOcclusionBlock(\n#ifdef AMBIENT\nin vec3 ambientOcclusionColorMap_,\nin vec4 vAmbientInfos,\n#endif\nout ambientOcclusionOutParams outParams\n)\n{vec3 ambientOcclusionColor=vec3(1.,1.,1.);\n#ifdef AMBIENT\nvec3 ambientOcclusionColorMap=ambientOcclusionColorMap_*vAmbientInfos.y;\n#ifdef AMBIENTINGRAYSCALE\nambientOcclusionColorMap=vec3(ambientOcclusionColorMap.r,ambientOcclusionColorMap.r,ambientOcclusionColorMap.r);\n#endif\nambientOcclusionColor=mix(ambientOcclusionColor,ambientOcclusionColorMap,vAmbientInfos.z);\n#if DEBUGMODE>0\noutParams.ambientOcclusionColorMap=ambientOcclusionColorMap;\n#endif\n#endif\noutParams.ambientOcclusionColor=ambientOcclusionColor;}\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1k] = shader$1k;\n\n // Do not edit.\n const name$1l = \"pbrBlockAlphaFresnel\";\n const shader$1l = `#ifdef ALPHAFRESNEL\n#if defined(ALPHATEST) || defined(ALPHABLEND)\nstruct alphaFresnelOutParams\n{float alpha;};\n#define pbr_inline\nvoid alphaFresnelBlock(\nin vec3 normalW,\nin vec3 viewDirectionW,\nin float alpha,\nin float microSurface,\nout alphaFresnelOutParams outParams\n)\n{float opacityPerceptual=alpha;\n#ifdef LINEARALPHAFRESNEL\nfloat opacity0=opacityPerceptual;\n#else\nfloat opacity0=opacityPerceptual*opacityPerceptual;\n#endif\nfloat opacity90=fresnelGrazingReflectance(opacity0);vec3 normalForward=faceforward(normalW,-viewDirectionW,normalW);outParams.alpha=getReflectanceFromAnalyticalBRDFLookup_Jones(saturate(dot(viewDirectionW,normalForward)),vec3(opacity0),vec3(opacity90),sqrt(microSurface)).x;\n#ifdef ALPHATEST\nif (outParams.alpha0 && defined(ANISOTROPIC_TEXTURE)\nvec3 anisotropyMapData;\n#endif\n};\n#define pbr_inline\nvoid anisotropicBlock(\nin vec3 vAnisotropy,\nin float roughness,\n#ifdef ANISOTROPIC_TEXTURE\nin vec3 anisotropyMapData,\n#endif\nin mat3 TBN,\nin vec3 normalW,\nin vec3 viewDirectionW,\nout anisotropicOutParams outParams\n)\n{float anisotropy=vAnisotropy.b;vec3 anisotropyDirection=vec3(vAnisotropy.xy,0.);\n#ifdef ANISOTROPIC_TEXTURE\nanisotropy*=anisotropyMapData.b;\n#if DEBUGMODE>0\noutParams.anisotropyMapData=anisotropyMapData;\n#endif\nanisotropyMapData.rg=anisotropyMapData.rg*2.0-1.0;\n#ifdef ANISOTROPIC_LEGACY\nanisotropyDirection.rg*=anisotropyMapData.rg;\n#else\nanisotropyDirection.xy=mat2(anisotropyDirection.x,anisotropyDirection.y,-anisotropyDirection.y,anisotropyDirection.x)*normalize(anisotropyMapData.rg);\n#endif\n#endif\nmat3 anisoTBN=mat3(normalize(TBN[0]),normalize(TBN[1]),normalize(TBN[2]));vec3 anisotropicTangent=normalize(anisoTBN*anisotropyDirection);vec3 anisotropicBitangent=normalize(cross(anisoTBN[2],anisotropicTangent));outParams.anisotropy=anisotropy;outParams.anisotropicTangent=anisotropicTangent;outParams.anisotropicBitangent=anisotropicBitangent;outParams.anisotropicNormal=getAnisotropicBentNormals(anisotropicTangent,anisotropicBitangent,normalW,viewDirectionW,anisotropy,roughness);}\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1m] = shader$1m;\n\n // Do not edit.\n const name$1n = \"pbrBlockReflection\";\n const shader$1n = `#ifdef REFLECTION\nstruct reflectionOutParams\n{vec4 environmentRadiance;vec3 environmentIrradiance;\n#ifdef REFLECTIONMAP_3D\nvec3 reflectionCoords;\n#else\nvec2 reflectionCoords;\n#endif\n#ifdef SS_TRANSLUCENCY\n#ifdef USESPHERICALFROMREFLECTIONMAP\n#if !defined(NORMAL) || !defined(USESPHERICALINVERTEX)\nvec3 irradianceVector;\n#endif\n#endif\n#endif\n};\n#define pbr_inline\nvoid createReflectionCoords(\nin vec3 vPositionW,\nin vec3 normalW,\n#ifdef ANISOTROPIC\nin anisotropicOutParams anisotropicOut,\n#endif\n#ifdef REFLECTIONMAP_3D\nout vec3 reflectionCoords\n#else\nout vec2 reflectionCoords\n#endif\n)\n{\n#ifdef ANISOTROPIC\nvec3 reflectionVector=computeReflectionCoords(vec4(vPositionW,1.0),anisotropicOut.anisotropicNormal);\n#else\nvec3 reflectionVector=computeReflectionCoords(vec4(vPositionW,1.0),normalW);\n#endif\n#ifdef REFLECTIONMAP_OPPOSITEZ\nreflectionVector.z*=-1.0;\n#endif\n#ifdef REFLECTIONMAP_3D\nreflectionCoords=reflectionVector;\n#else\nreflectionCoords=reflectionVector.xy;\n#ifdef REFLECTIONMAP_PROJECTION\nreflectionCoords/=reflectionVector.z;\n#endif\nreflectionCoords.y=1.0-reflectionCoords.y;\n#endif\n}\n#define pbr_inline\n#define inline\nvoid sampleReflectionTexture(\nin float alphaG,\nin vec3 vReflectionMicrosurfaceInfos,\nin vec2 vReflectionInfos,\nin vec3 vReflectionColor,\n#if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX)\nin float NdotVUnclamped,\n#endif\n#ifdef LINEARSPECULARREFLECTION\nin float roughness,\n#endif\n#ifdef REFLECTIONMAP_3D\nin samplerCube reflectionSampler,\nconst vec3 reflectionCoords,\n#else\nin sampler2D reflectionSampler,\nconst vec2 reflectionCoords,\n#endif\n#ifndef LODBASEDMICROSFURACE\n#ifdef REFLECTIONMAP_3D\nin samplerCube reflectionSamplerLow,\nin samplerCube reflectionSamplerHigh,\n#else\nin sampler2D reflectionSamplerLow,\nin sampler2D reflectionSamplerHigh,\n#endif\n#endif\n#ifdef REALTIME_FILTERING\nin vec2 vReflectionFilteringInfo,\n#endif\nout vec4 environmentRadiance\n)\n{\n#if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX)\nfloat reflectionLOD=getLodFromAlphaG(vReflectionMicrosurfaceInfos.x,alphaG,NdotVUnclamped);\n#elif defined(LINEARSPECULARREFLECTION)\nfloat reflectionLOD=getLinearLodFromRoughness(vReflectionMicrosurfaceInfos.x,roughness);\n#else\nfloat reflectionLOD=getLodFromAlphaG(vReflectionMicrosurfaceInfos.x,alphaG);\n#endif\n#ifdef LODBASEDMICROSFURACE\nreflectionLOD=reflectionLOD*vReflectionMicrosurfaceInfos.y+vReflectionMicrosurfaceInfos.z;\n#ifdef LODINREFLECTIONALPHA\nfloat automaticReflectionLOD=UNPACK_LOD(sampleReflection(reflectionSampler,reflectionCoords).a);float requestedReflectionLOD=max(automaticReflectionLOD,reflectionLOD);\n#else\nfloat requestedReflectionLOD=reflectionLOD;\n#endif\n#ifdef REALTIME_FILTERING\nenvironmentRadiance=vec4(radiance(alphaG,reflectionSampler,reflectionCoords,vReflectionFilteringInfo),1.0);\n#else\nenvironmentRadiance=sampleReflectionLod(reflectionSampler,reflectionCoords,reflectionLOD);\n#endif\n#else\nfloat lodReflectionNormalized=saturate(reflectionLOD/log2(vReflectionMicrosurfaceInfos.x));float lodReflectionNormalizedDoubled=lodReflectionNormalized*2.0;vec4 environmentMid=sampleReflection(reflectionSampler,reflectionCoords);if (lodReflectionNormalizedDoubled<1.0){environmentRadiance=mix(\nsampleReflection(reflectionSamplerHigh,reflectionCoords),\nenvironmentMid,\nlodReflectionNormalizedDoubled\n);} else {environmentRadiance=mix(\nenvironmentMid,\nsampleReflection(reflectionSamplerLow,reflectionCoords),\nlodReflectionNormalizedDoubled-1.0\n);}\n#endif\n#ifdef RGBDREFLECTION\nenvironmentRadiance.rgb=fromRGBD(environmentRadiance);\n#endif\n#ifdef GAMMAREFLECTION\nenvironmentRadiance.rgb=toLinearSpace(environmentRadiance.rgb);\n#endif\nenvironmentRadiance.rgb*=vReflectionInfos.x;environmentRadiance.rgb*=vReflectionColor.rgb;}\n#define pbr_inline\n#define inline\nvoid reflectionBlock(\nin vec3 vPositionW,\nin vec3 normalW,\nin float alphaG,\nin vec3 vReflectionMicrosurfaceInfos,\nin vec2 vReflectionInfos,\nin vec3 vReflectionColor,\n#ifdef ANISOTROPIC\nin anisotropicOutParams anisotropicOut,\n#endif\n#if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX)\nin float NdotVUnclamped,\n#endif\n#ifdef LINEARSPECULARREFLECTION\nin float roughness,\n#endif\n#ifdef REFLECTIONMAP_3D\nin samplerCube reflectionSampler,\n#else\nin sampler2D reflectionSampler,\n#endif\n#if defined(NORMAL) && defined(USESPHERICALINVERTEX)\nin vec3 vEnvironmentIrradiance,\n#endif\n#ifdef USESPHERICALFROMREFLECTIONMAP\n#if !defined(NORMAL) || !defined(USESPHERICALINVERTEX)\nin mat4 reflectionMatrix,\n#endif\n#endif\n#ifdef USEIRRADIANCEMAP\n#ifdef REFLECTIONMAP_3D\nin samplerCube irradianceSampler,\n#else\nin sampler2D irradianceSampler,\n#endif\n#endif\n#ifndef LODBASEDMICROSFURACE\n#ifdef REFLECTIONMAP_3D\nin samplerCube reflectionSamplerLow,\nin samplerCube reflectionSamplerHigh,\n#else\nin sampler2D reflectionSamplerLow,\nin sampler2D reflectionSamplerHigh,\n#endif\n#endif\n#ifdef REALTIME_FILTERING\nin vec2 vReflectionFilteringInfo,\n#endif\nout reflectionOutParams outParams\n)\n{vec4 environmentRadiance=vec4(0.,0.,0.,0.);\n#ifdef REFLECTIONMAP_3D\nvec3 reflectionCoords=vec3(0.);\n#else\nvec2 reflectionCoords=vec2(0.);\n#endif\ncreateReflectionCoords(\nvPositionW,\nnormalW,\n#ifdef ANISOTROPIC\nanisotropicOut,\n#endif\nreflectionCoords\n);sampleReflectionTexture(\nalphaG,\nvReflectionMicrosurfaceInfos,\nvReflectionInfos,\nvReflectionColor,\n#if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX)\nNdotVUnclamped,\n#endif\n#ifdef LINEARSPECULARREFLECTION\nroughness,\n#endif\n#ifdef REFLECTIONMAP_3D\nreflectionSampler,\nreflectionCoords,\n#else\nreflectionSampler,\nreflectionCoords,\n#endif\n#ifndef LODBASEDMICROSFURACE\nreflectionSamplerLow,\nreflectionSamplerHigh,\n#endif\n#ifdef REALTIME_FILTERING\nvReflectionFilteringInfo,\n#endif\nenvironmentRadiance\n);vec3 environmentIrradiance=vec3(0.,0.,0.);\n#ifdef USESPHERICALFROMREFLECTIONMAP\n#if defined(NORMAL) && defined(USESPHERICALINVERTEX)\nenvironmentIrradiance=vEnvironmentIrradiance;\n#else\n#ifdef ANISOTROPIC\nvec3 irradianceVector=vec3(reflectionMatrix*vec4(anisotropicOut.anisotropicNormal,0)).xyz;\n#else\nvec3 irradianceVector=vec3(reflectionMatrix*vec4(normalW,0)).xyz;\n#endif\n#ifdef REFLECTIONMAP_OPPOSITEZ\nirradianceVector.z*=-1.0;\n#endif\n#ifdef INVERTCUBICMAP\nirradianceVector.y*=-1.0;\n#endif\n#if defined(REALTIME_FILTERING)\nenvironmentIrradiance=irradiance(reflectionSampler,irradianceVector,vReflectionFilteringInfo);\n#else\nenvironmentIrradiance=computeEnvironmentIrradiance(irradianceVector);\n#endif\n#ifdef SS_TRANSLUCENCY\noutParams.irradianceVector=irradianceVector;\n#endif\n#endif\n#elif defined(USEIRRADIANCEMAP)\nvec4 environmentIrradiance4=sampleReflection(irradianceSampler,reflectionCoords);environmentIrradiance=environmentIrradiance4.rgb;\n#ifdef RGBDREFLECTION\nenvironmentIrradiance.rgb=fromRGBD(environmentIrradiance4);\n#endif\n#ifdef GAMMAREFLECTION\nenvironmentIrradiance.rgb=toLinearSpace(environmentIrradiance.rgb);\n#endif\n#endif\nenvironmentIrradiance*=vReflectionColor.rgb;outParams.environmentRadiance=environmentRadiance;outParams.environmentIrradiance=environmentIrradiance;outParams.reflectionCoords=reflectionCoords;}\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1n] = shader$1n;\n\n // Do not edit.\n const name$1o = \"pbrBlockSheen\";\n const shader$1o = `#ifdef SHEEN\nstruct sheenOutParams\n{float sheenIntensity;vec3 sheenColor;float sheenRoughness;\n#ifdef SHEEN_LINKWITHALBEDO\nvec3 surfaceAlbedo;\n#endif\n#if defined(ENVIRONMENTBRDF) && defined(SHEEN_ALBEDOSCALING)\nfloat sheenAlbedoScaling;\n#endif\n#if defined(REFLECTION) && defined(ENVIRONMENTBRDF)\nvec3 finalSheenRadianceScaled;\n#endif\n#if DEBUGMODE>0\n#ifdef SHEEN_TEXTURE\nvec4 sheenMapData;\n#endif\n#if defined(REFLECTION) && defined(ENVIRONMENTBRDF)\nvec3 sheenEnvironmentReflectance;\n#endif\n#endif\n};\n#define pbr_inline\n#define inline\nvoid sheenBlock(\nin vec4 vSheenColor,\n#ifdef SHEEN_ROUGHNESS\nin float vSheenRoughness,\n#if defined(SHEEN_TEXTURE_ROUGHNESS) && !defined(SHEEN_TEXTURE_ROUGHNESS_IDENTICAL) && !defined(SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE)\nin vec4 sheenMapRoughnessData,\n#endif\n#endif\nin float roughness,\n#ifdef SHEEN_TEXTURE\nin vec4 sheenMapData,\nin float sheenMapLevel,\n#endif\nin float reflectance,\n#ifdef SHEEN_LINKWITHALBEDO\nin vec3 baseColor,\nin vec3 surfaceAlbedo,\n#endif\n#ifdef ENVIRONMENTBRDF\nin float NdotV,\nin vec3 environmentBrdf,\n#endif\n#if defined(REFLECTION) && defined(ENVIRONMENTBRDF)\nin vec2 AARoughnessFactors,\nin vec3 vReflectionMicrosurfaceInfos,\nin vec2 vReflectionInfos,\nin vec3 vReflectionColor,\nin vec4 vLightingIntensity,\n#ifdef REFLECTIONMAP_3D\nin samplerCube reflectionSampler,\nin vec3 reflectionCoords,\n#else\nin sampler2D reflectionSampler,\nin vec2 reflectionCoords,\n#endif\nin float NdotVUnclamped,\n#ifndef LODBASEDMICROSFURACE\n#ifdef REFLECTIONMAP_3D\nin samplerCube reflectionSamplerLow,\nin samplerCube reflectionSamplerHigh,\n#else\nin sampler2D reflectionSamplerLow,\nin sampler2D reflectionSamplerHigh,\n#endif\n#endif\n#ifdef REALTIME_FILTERING\nin vec2 vReflectionFilteringInfo,\n#endif\n#if !defined(REFLECTIONMAP_SKYBOX) && defined(RADIANCEOCCLUSION)\nin float seo,\n#endif\n#if !defined(REFLECTIONMAP_SKYBOX) && defined(HORIZONOCCLUSION) && defined(BUMP) && defined(REFLECTIONMAP_3D)\nin float eho,\n#endif\n#endif\nout sheenOutParams outParams\n)\n{float sheenIntensity=vSheenColor.a;\n#ifdef SHEEN_TEXTURE\n#if DEBUGMODE>0\noutParams.sheenMapData=sheenMapData;\n#endif\n#endif\n#ifdef SHEEN_LINKWITHALBEDO\nfloat sheenFactor=pow5(1.0-sheenIntensity);vec3 sheenColor=baseColor.rgb*(1.0-sheenFactor);float sheenRoughness=sheenIntensity;outParams.surfaceAlbedo=surfaceAlbedo*sheenFactor;\n#ifdef SHEEN_TEXTURE\nsheenIntensity*=sheenMapData.a;\n#endif\n#else\nvec3 sheenColor=vSheenColor.rgb;\n#ifdef SHEEN_TEXTURE\n#ifdef SHEEN_GAMMATEXTURE\nsheenColor.rgb*=toLinearSpace(sheenMapData.rgb);\n#else\nsheenColor.rgb*=sheenMapData.rgb;\n#endif\nsheenColor.rgb*=sheenMapLevel;\n#endif\n#ifdef SHEEN_ROUGHNESS\nfloat sheenRoughness=vSheenRoughness;\n#ifdef SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE\n#if defined(SHEEN_TEXTURE)\nsheenRoughness*=sheenMapData.a;\n#endif\n#elif defined(SHEEN_TEXTURE_ROUGHNESS)\n#ifdef SHEEN_TEXTURE_ROUGHNESS_IDENTICAL\nsheenRoughness*=sheenMapData.a;\n#else\nsheenRoughness*=sheenMapRoughnessData.a;\n#endif\n#endif\n#else\nfloat sheenRoughness=roughness;\n#ifdef SHEEN_TEXTURE\nsheenIntensity*=sheenMapData.a;\n#endif\n#endif\n#if !defined(SHEEN_ALBEDOSCALING)\nsheenIntensity*=(1.-reflectance);\n#endif\nsheenColor*=sheenIntensity;\n#endif\n#ifdef ENVIRONMENTBRDF\n/*#ifdef SHEEN_SOFTER\nvec3 environmentSheenBrdf=vec3(0.,0.,getBRDFLookupCharlieSheen(NdotV,sheenRoughness));\n#else*/\n#ifdef SHEEN_ROUGHNESS\nvec3 environmentSheenBrdf=getBRDFLookup(NdotV,sheenRoughness);\n#else\nvec3 environmentSheenBrdf=environmentBrdf;\n#endif\n/*#endif*/\n#endif\n#if defined(REFLECTION) && defined(ENVIRONMENTBRDF)\nfloat sheenAlphaG=convertRoughnessToAverageSlope(sheenRoughness);\n#ifdef SPECULARAA\nsheenAlphaG+=AARoughnessFactors.y;\n#endif\nvec4 environmentSheenRadiance=vec4(0.,0.,0.,0.);sampleReflectionTexture(\nsheenAlphaG,\nvReflectionMicrosurfaceInfos,\nvReflectionInfos,\nvReflectionColor,\n#if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX)\nNdotVUnclamped,\n#endif\n#ifdef LINEARSPECULARREFLECTION\nsheenRoughness,\n#endif\nreflectionSampler,\nreflectionCoords,\n#ifndef LODBASEDMICROSFURACE\nreflectionSamplerLow,\nreflectionSamplerHigh,\n#endif\n#ifdef REALTIME_FILTERING\nvReflectionFilteringInfo,\n#endif\nenvironmentSheenRadiance\n);vec3 sheenEnvironmentReflectance=getSheenReflectanceFromBRDFLookup(sheenColor,environmentSheenBrdf);\n#if !defined(REFLECTIONMAP_SKYBOX) && defined(RADIANCEOCCLUSION)\nsheenEnvironmentReflectance*=seo;\n#endif\n#if !defined(REFLECTIONMAP_SKYBOX) && defined(HORIZONOCCLUSION) && defined(BUMP) && defined(REFLECTIONMAP_3D)\nsheenEnvironmentReflectance*=eho;\n#endif\n#if DEBUGMODE>0\noutParams.sheenEnvironmentReflectance=sheenEnvironmentReflectance;\n#endif\noutParams.finalSheenRadianceScaled=\nenvironmentSheenRadiance.rgb *\nsheenEnvironmentReflectance *\nvLightingIntensity.z;\n#endif\n#if defined(ENVIRONMENTBRDF) && defined(SHEEN_ALBEDOSCALING)\noutParams.sheenAlbedoScaling=1.0-sheenIntensity*max(max(sheenColor.r,sheenColor.g),sheenColor.b)*environmentSheenBrdf.b;\n#endif\noutParams.sheenIntensity=sheenIntensity;outParams.sheenColor=sheenColor;outParams.sheenRoughness=sheenRoughness;}\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1o] = shader$1o;\n\n // Do not edit.\n const name$1p = \"pbrBlockClearcoat\";\n const shader$1p = `struct clearcoatOutParams\n{vec3 specularEnvironmentR0;float conservationFactor;vec3 clearCoatNormalW;vec2 clearCoatAARoughnessFactors;float clearCoatIntensity;float clearCoatRoughness;\n#ifdef REFLECTION\nvec3 finalClearCoatRadianceScaled;\n#endif\n#ifdef CLEARCOAT_TINT\nvec3 absorption;float clearCoatNdotVRefract;vec3 clearCoatColor;float clearCoatThickness;\n#endif\n#if defined(ENVIRONMENTBRDF) && defined(MS_BRDF_ENERGY_CONSERVATION)\nvec3 energyConservationFactorClearCoat;\n#endif\n#if DEBUGMODE>0\n#ifdef CLEARCOAT_BUMP\nmat3 TBNClearCoat;\n#endif\n#ifdef CLEARCOAT_TEXTURE\nvec2 clearCoatMapData;\n#endif\n#if defined(CLEARCOAT_TINT) && defined(CLEARCOAT_TINT_TEXTURE)\nvec4 clearCoatTintMapData;\n#endif\n#ifdef REFLECTION\nvec4 environmentClearCoatRadiance;vec3 clearCoatEnvironmentReflectance;\n#endif\nfloat clearCoatNdotV;\n#endif\n};\n#ifdef CLEARCOAT\n#define pbr_inline\n#define inline\nvoid clearcoatBlock(\nin vec3 vPositionW,\nin vec3 geometricNormalW,\nin vec3 viewDirectionW,\nin vec2 vClearCoatParams,\n#if defined(CLEARCOAT_TEXTURE_ROUGHNESS) && !defined(CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL) && !defined(CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE)\nin vec4 clearCoatMapRoughnessData,\n#endif\nin vec3 specularEnvironmentR0,\n#ifdef CLEARCOAT_TEXTURE\nin vec2 clearCoatMapData,\n#endif\n#ifdef CLEARCOAT_TINT\nin vec4 vClearCoatTintParams,\nin float clearCoatColorAtDistance,\nin vec4 vClearCoatRefractionParams,\n#ifdef CLEARCOAT_TINT_TEXTURE\nin vec4 clearCoatTintMapData,\n#endif\n#endif\n#ifdef CLEARCOAT_BUMP\nin vec2 vClearCoatBumpInfos,\nin vec4 clearCoatBumpMapData,\nin vec2 vClearCoatBumpUV,\n#if defined(TANGENT) && defined(NORMAL)\nin mat3 vTBN,\n#else\nin vec2 vClearCoatTangentSpaceParams,\n#endif\n#ifdef OBJECTSPACE_NORMALMAP\nin mat4 normalMatrix,\n#endif\n#endif\n#if defined(FORCENORMALFORWARD) && defined(NORMAL)\nin vec3 faceNormal,\n#endif\n#ifdef REFLECTION\nin vec3 vReflectionMicrosurfaceInfos,\nin vec2 vReflectionInfos,\nin vec3 vReflectionColor,\nin vec4 vLightingIntensity,\n#ifdef REFLECTIONMAP_3D\nin samplerCube reflectionSampler,\n#else\nin sampler2D reflectionSampler,\n#endif\n#ifndef LODBASEDMICROSFURACE\n#ifdef REFLECTIONMAP_3D\nin samplerCube reflectionSamplerLow,\nin samplerCube reflectionSamplerHigh,\n#else\nin sampler2D reflectionSamplerLow,\nin sampler2D reflectionSamplerHigh,\n#endif\n#endif\n#ifdef REALTIME_FILTERING\nin vec2 vReflectionFilteringInfo,\n#endif\n#endif\n#if defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX)\n#ifdef RADIANCEOCCLUSION\nin float ambientMonochrome,\n#endif\n#endif\n#if defined(CLEARCOAT_BUMP) || defined(TWOSIDEDLIGHTING)\nin float frontFacingMultiplier,\n#endif\nout clearcoatOutParams outParams\n)\n{float clearCoatIntensity=vClearCoatParams.x;float clearCoatRoughness=vClearCoatParams.y;\n#ifdef CLEARCOAT_TEXTURE\nclearCoatIntensity*=clearCoatMapData.x;\n#ifdef CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE\nclearCoatRoughness*=clearCoatMapData.y;\n#endif\n#if DEBUGMODE>0\noutParams.clearCoatMapData=clearCoatMapData;\n#endif\n#endif\n#if defined(CLEARCOAT_TEXTURE_ROUGHNESS) && !defined(CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE)\n#ifdef CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL\nclearCoatRoughness*=clearCoatMapData.y;\n#else\nclearCoatRoughness*=clearCoatMapRoughnessData.y;\n#endif\n#endif\noutParams.clearCoatIntensity=clearCoatIntensity;outParams.clearCoatRoughness=clearCoatRoughness;\n#ifdef CLEARCOAT_TINT\nvec3 clearCoatColor=vClearCoatTintParams.rgb;float clearCoatThickness=vClearCoatTintParams.a;\n#ifdef CLEARCOAT_TINT_TEXTURE\n#ifdef CLEARCOAT_TINT_GAMMATEXTURE\nclearCoatColor*=toLinearSpace(clearCoatTintMapData.rgb);\n#else\nclearCoatColor*=clearCoatTintMapData.rgb;\n#endif\nclearCoatThickness*=clearCoatTintMapData.a;\n#if DEBUGMODE>0\noutParams.clearCoatTintMapData=clearCoatTintMapData;\n#endif\n#endif\noutParams.clearCoatColor=computeColorAtDistanceInMedia(clearCoatColor,clearCoatColorAtDistance);outParams.clearCoatThickness=clearCoatThickness;\n#endif\n#ifdef CLEARCOAT_REMAP_F0\nvec3 specularEnvironmentR0Updated=getR0RemappedForClearCoat(specularEnvironmentR0);\n#else\nvec3 specularEnvironmentR0Updated=specularEnvironmentR0;\n#endif\noutParams.specularEnvironmentR0=mix(specularEnvironmentR0,specularEnvironmentR0Updated,clearCoatIntensity);vec3 clearCoatNormalW=geometricNormalW;\n#ifdef CLEARCOAT_BUMP\n#ifdef NORMALXYSCALE\nfloat clearCoatNormalScale=1.0;\n#else\nfloat clearCoatNormalScale=vClearCoatBumpInfos.y;\n#endif\n#if defined(TANGENT) && defined(NORMAL)\nmat3 TBNClearCoat=vTBN;\n#else\nvec2 TBNClearCoatUV=vClearCoatBumpUV*frontFacingMultiplier;mat3 TBNClearCoat=cotangent_frame(clearCoatNormalW*clearCoatNormalScale,vPositionW,TBNClearCoatUV,vClearCoatTangentSpaceParams);\n#endif\n#if DEBUGMODE>0\noutParams.TBNClearCoat=TBNClearCoat;\n#endif\n#ifdef OBJECTSPACE_NORMALMAP\nclearCoatNormalW=normalize(clearCoatBumpMapData.xyz *2.0-1.0);clearCoatNormalW=normalize(mat3(normalMatrix)*clearCoatNormalW);\n#else\nclearCoatNormalW=perturbNormal(TBNClearCoat,clearCoatBumpMapData.xyz,vClearCoatBumpInfos.y);\n#endif\n#endif\n#if defined(FORCENORMALFORWARD) && defined(NORMAL)\nclearCoatNormalW*=sign(dot(clearCoatNormalW,faceNormal));\n#endif\n#if defined(TWOSIDEDLIGHTING) && defined(NORMAL)\nclearCoatNormalW=clearCoatNormalW*frontFacingMultiplier;\n#endif\noutParams.clearCoatNormalW=clearCoatNormalW;outParams.clearCoatAARoughnessFactors=getAARoughnessFactors(clearCoatNormalW.xyz);float clearCoatNdotVUnclamped=dot(clearCoatNormalW,viewDirectionW);float clearCoatNdotV=absEps(clearCoatNdotVUnclamped);\n#if DEBUGMODE>0\noutParams.clearCoatNdotV=clearCoatNdotV;\n#endif\n#ifdef CLEARCOAT_TINT\nvec3 clearCoatVRefract=refract(-viewDirectionW,clearCoatNormalW,vClearCoatRefractionParams.y);outParams.clearCoatNdotVRefract=absEps(dot(clearCoatNormalW,clearCoatVRefract));\n#endif\n#if defined(ENVIRONMENTBRDF) && (!defined(REFLECTIONMAP_SKYBOX) || defined(MS_BRDF_ENERGY_CONSERVATION))\nvec3 environmentClearCoatBrdf=getBRDFLookup(clearCoatNdotV,clearCoatRoughness);\n#endif\n#if defined(REFLECTION)\nfloat clearCoatAlphaG=convertRoughnessToAverageSlope(clearCoatRoughness);\n#ifdef SPECULARAA\nclearCoatAlphaG+=outParams.clearCoatAARoughnessFactors.y;\n#endif\nvec4 environmentClearCoatRadiance=vec4(0.,0.,0.,0.);vec3 clearCoatReflectionVector=computeReflectionCoords(vec4(vPositionW,1.0),clearCoatNormalW);\n#ifdef REFLECTIONMAP_OPPOSITEZ\nclearCoatReflectionVector.z*=-1.0;\n#endif\n#ifdef REFLECTIONMAP_3D\nvec3 clearCoatReflectionCoords=clearCoatReflectionVector;\n#else\nvec2 clearCoatReflectionCoords=clearCoatReflectionVector.xy;\n#ifdef REFLECTIONMAP_PROJECTION\nclearCoatReflectionCoords/=clearCoatReflectionVector.z;\n#endif\nclearCoatReflectionCoords.y=1.0-clearCoatReflectionCoords.y;\n#endif\nsampleReflectionTexture(\nclearCoatAlphaG,\nvReflectionMicrosurfaceInfos,\nvReflectionInfos,\nvReflectionColor,\n#if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX)\nclearCoatNdotVUnclamped,\n#endif\n#ifdef LINEARSPECULARREFLECTION\nclearCoatRoughness,\n#endif\nreflectionSampler,\nclearCoatReflectionCoords,\n#ifndef LODBASEDMICROSFURACE\nreflectionSamplerLow,\nreflectionSamplerHigh,\n#endif\n#ifdef REALTIME_FILTERING\nvReflectionFilteringInfo,\n#endif\nenvironmentClearCoatRadiance\n);\n#if DEBUGMODE>0\noutParams.environmentClearCoatRadiance=environmentClearCoatRadiance;\n#endif\n#if defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX)\nvec3 clearCoatEnvironmentReflectance=getReflectanceFromBRDFLookup(vec3(vClearCoatRefractionParams.x),environmentClearCoatBrdf);\n#ifdef HORIZONOCCLUSION\n#ifdef BUMP\n#ifdef REFLECTIONMAP_3D\nfloat clearCoatEho=environmentHorizonOcclusion(-viewDirectionW,clearCoatNormalW,geometricNormalW);clearCoatEnvironmentReflectance*=clearCoatEho;\n#endif\n#endif\n#endif\n#else\nvec3 clearCoatEnvironmentReflectance=getReflectanceFromAnalyticalBRDFLookup_Jones(clearCoatNdotV,vec3(1.),vec3(1.),sqrt(1.-clearCoatRoughness));\n#endif\nclearCoatEnvironmentReflectance*=clearCoatIntensity;\n#if DEBUGMODE>0\noutParams.clearCoatEnvironmentReflectance=clearCoatEnvironmentReflectance;\n#endif\noutParams.finalClearCoatRadianceScaled=\nenvironmentClearCoatRadiance.rgb *\nclearCoatEnvironmentReflectance *\nvLightingIntensity.z;\n#endif\n#if defined(CLEARCOAT_TINT)\noutParams.absorption=computeClearCoatAbsorption(outParams.clearCoatNdotVRefract,outParams.clearCoatNdotVRefract,outParams.clearCoatColor,clearCoatThickness,clearCoatIntensity);\n#endif\nfloat fresnelIBLClearCoat=fresnelSchlickGGX(clearCoatNdotV,vClearCoatRefractionParams.x,CLEARCOATREFLECTANCE90);fresnelIBLClearCoat*=clearCoatIntensity;outParams.conservationFactor=(1.-fresnelIBLClearCoat);\n#if defined(ENVIRONMENTBRDF) && defined(MS_BRDF_ENERGY_CONSERVATION)\noutParams.energyConservationFactorClearCoat=getEnergyConservationFactor(outParams.specularEnvironmentR0,environmentClearCoatBrdf);\n#endif\n}\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1p] = shader$1p;\n\n // Do not edit.\n const name$1q = \"pbrBlockIridescence\";\n const shader$1q = `struct iridescenceOutParams\n{float iridescenceIntensity;float iridescenceIOR;float iridescenceThickness;vec3 specularEnvironmentR0;};\n#ifdef IRIDESCENCE\n#define pbr_inline\n#define inline\nvoid iridescenceBlock(\nin vec4 vIridescenceParams,\nin float viewAngle,\nin vec3 specularEnvironmentR0,\n#ifdef IRIDESCENCE_TEXTURE\nin vec2 iridescenceMapData,\n#endif\n#ifdef IRIDESCENCE_THICKNESS_TEXTURE\nin vec2 iridescenceThicknessMapData,\n#endif\n#ifdef CLEARCOAT\nin float NdotVUnclamped,\n#ifdef CLEARCOAT_TEXTURE\nin vec2 clearCoatMapData,\n#endif\n#endif\nout iridescenceOutParams outParams\n)\n{float iridescenceIntensity=vIridescenceParams.x;float iridescenceIOR=vIridescenceParams.y;float iridescenceThicknessMin=vIridescenceParams.z;float iridescenceThicknessMax=vIridescenceParams.w;float iridescenceThicknessWeight=1.;\n#ifdef IRIDESCENCE_TEXTURE\niridescenceIntensity*=iridescenceMapData.x;\n#ifdef IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE\niridescenceThicknessWeight=iridescenceMapData.g;\n#endif\n#endif\n#if defined(IRIDESCENCE_THICKNESS_TEXTURE)\niridescenceThicknessWeight=iridescenceThicknessMapData.g;\n#endif\nfloat iridescenceThickness=mix(iridescenceThicknessMin,iridescenceThicknessMax,iridescenceThicknessWeight);float topIor=1.; \n#ifdef CLEARCOAT\nfloat clearCoatIntensity=vClearCoatParams.x;\n#ifdef CLEARCOAT_TEXTURE\nclearCoatIntensity*=clearCoatMapData.x;\n#endif\ntopIor=mix(1.0,vClearCoatRefractionParams.w-1.,clearCoatIntensity);viewAngle=sqrt(1.0+square(1.0/topIor)*(square(NdotVUnclamped)-1.0));\n#endif\nvec3 iridescenceFresnel=evalIridescence(topIor,iridescenceIOR,viewAngle,iridescenceThickness,specularEnvironmentR0);outParams.specularEnvironmentR0=mix(specularEnvironmentR0,iridescenceFresnel,iridescenceIntensity);outParams.iridescenceIntensity=iridescenceIntensity;outParams.iridescenceThickness=iridescenceThickness;outParams.iridescenceIOR=iridescenceIOR;}\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1q] = shader$1q;\n\n // Do not edit.\n const name$1r = \"pbrBlockSubSurface\";\n const shader$1r = `struct subSurfaceOutParams\n{vec3 specularEnvironmentReflectance;\n#ifdef SS_REFRACTION\nvec3 finalRefraction;vec3 surfaceAlbedo;\n#ifdef SS_LINKREFRACTIONTOTRANSPARENCY\nfloat alpha;\n#endif\n#ifdef REFLECTION\nfloat refractionFactorForIrradiance;\n#endif\n#endif\n#ifdef SS_TRANSLUCENCY\nvec3 transmittance;float translucencyIntensity;\n#ifdef REFLECTION\nvec3 refractionIrradiance;\n#endif\n#endif\n#if DEBUGMODE>0\n#ifdef SS_THICKNESSANDMASK_TEXTURE\nvec4 thicknessMap;\n#endif\n#ifdef SS_REFRACTION\nvec4 environmentRefraction;vec3 refractionTransmittance;\n#endif\n#endif\n};\n#ifdef SUBSURFACE\n#define pbr_inline\n#define inline\nvoid subSurfaceBlock(\nin vec3 vSubSurfaceIntensity,\nin vec2 vThicknessParam,\nin vec4 vTintColor,\nin vec3 normalW,\nin vec3 specularEnvironmentReflectance,\n#ifdef SS_THICKNESSANDMASK_TEXTURE\nin vec4 thicknessMap,\n#endif\n#ifdef SS_REFRACTIONINTENSITY_TEXTURE\nin vec4 refractionIntensityMap,\n#endif\n#ifdef SS_TRANSLUCENCYINTENSITY_TEXTURE\nin vec4 translucencyIntensityMap,\n#endif\n#ifdef REFLECTION\n#ifdef SS_TRANSLUCENCY\nin mat4 reflectionMatrix,\n#ifdef USESPHERICALFROMREFLECTIONMAP\n#if !defined(NORMAL) || !defined(USESPHERICALINVERTEX)\nin vec3 irradianceVector_,\n#endif\n#if defined(REALTIME_FILTERING)\nin samplerCube reflectionSampler,\nin vec2 vReflectionFilteringInfo,\n#endif\n#endif\n#ifdef USEIRRADIANCEMAP\n#ifdef REFLECTIONMAP_3D\nin samplerCube irradianceSampler,\n#else\nin sampler2D irradianceSampler,\n#endif\n#endif\n#endif\n#endif\n#if defined(SS_REFRACTION) || defined(SS_TRANSLUCENCY)\nin vec3 surfaceAlbedo,\n#endif\n#ifdef SS_REFRACTION\nin vec3 vPositionW,\nin vec3 viewDirectionW,\nin mat4 view,\nin vec4 vRefractionInfos,\nin mat4 refractionMatrix,\nin vec4 vRefractionMicrosurfaceInfos,\nin vec4 vLightingIntensity,\n#ifdef SS_LINKREFRACTIONTOTRANSPARENCY\nin float alpha,\n#endif\n#ifdef SS_LODINREFRACTIONALPHA\nin float NdotVUnclamped,\n#endif\n#ifdef SS_LINEARSPECULARREFRACTION\nin float roughness,\n#endif\nin float alphaG,\n#ifdef SS_REFRACTIONMAP_3D\nin samplerCube refractionSampler,\n#ifndef LODBASEDMICROSFURACE\nin samplerCube refractionSamplerLow,\nin samplerCube refractionSamplerHigh,\n#endif\n#else\nin sampler2D refractionSampler,\n#ifndef LODBASEDMICROSFURACE\nin sampler2D refractionSamplerLow,\nin sampler2D refractionSamplerHigh,\n#endif\n#endif\n#ifdef ANISOTROPIC\nin anisotropicOutParams anisotropicOut,\n#endif\n#ifdef REALTIME_FILTERING\nin vec2 vRefractionFilteringInfo,\n#endif\n#ifdef SS_USE_LOCAL_REFRACTIONMAP_CUBIC\nin vec3 refractionPosition,\nin vec3 refractionSize,\n#endif\n#endif\n#ifdef SS_TRANSLUCENCY\nin vec3 vDiffusionDistance,\n#endif\nout subSurfaceOutParams outParams\n)\n{outParams.specularEnvironmentReflectance=specularEnvironmentReflectance;\n#ifdef SS_REFRACTION\nfloat refractionIntensity=vSubSurfaceIntensity.x;\n#ifdef SS_LINKREFRACTIONTOTRANSPARENCY\nrefractionIntensity*=(1.0-alpha);outParams.alpha=1.0;\n#endif\n#endif\n#ifdef SS_TRANSLUCENCY\nfloat translucencyIntensity=vSubSurfaceIntensity.y;\n#endif\n#ifdef SS_THICKNESSANDMASK_TEXTURE\n#if defined(SS_USE_GLTF_TEXTURES)\nfloat thickness=thicknessMap.g*vThicknessParam.y+vThicknessParam.x;\n#else\nfloat thickness=thicknessMap.r*vThicknessParam.y+vThicknessParam.x;\n#endif\n#if DEBUGMODE>0\noutParams.thicknessMap=thicknessMap;\n#endif\n#ifdef SS_MASK_FROM_THICKNESS_TEXTURE\n#if defined(SS_REFRACTION) && defined(SS_REFRACTION_USE_INTENSITY_FROM_TEXTURE)\n#if defined(SS_USE_GLTF_TEXTURES)\nrefractionIntensity*=thicknessMap.r;\n#else\nrefractionIntensity*=thicknessMap.g;\n#endif\n#endif\n#if defined(SS_TRANSLUCENCY) && defined(SS_TRANSLUCENCY_USE_INTENSITY_FROM_TEXTURE)\ntranslucencyIntensity*=thicknessMap.b;\n#endif\n#endif\n#else\nfloat thickness=vThicknessParam.y;\n#endif\n#ifdef SS_REFRACTIONINTENSITY_TEXTURE\n#ifdef SS_USE_GLTF_TEXTURES\nrefractionIntensity*=refractionIntensityMap.r;\n#else\nrefractionIntensity*=refractionIntensityMap.g;\n#endif\n#endif\n#ifdef SS_TRANSLUCENCYINTENSITY_TEXTURE\ntranslucencyIntensity*=translucencyIntensityMap.b;\n#endif\n#ifdef SS_TRANSLUCENCY\nthickness=maxEps(thickness);vec3 transmittance=transmittanceBRDF_Burley(vTintColor.rgb,vDiffusionDistance,thickness);transmittance*=translucencyIntensity;outParams.transmittance=transmittance;outParams.translucencyIntensity=translucencyIntensity;\n#endif\n#ifdef SS_REFRACTION\nvec4 environmentRefraction=vec4(0.,0.,0.,0.);\n#ifdef ANISOTROPIC\nvec3 refractionVector=refract(-viewDirectionW,anisotropicOut.anisotropicNormal,vRefractionInfos.y);\n#else\nvec3 refractionVector=refract(-viewDirectionW,normalW,vRefractionInfos.y);\n#endif\n#ifdef SS_REFRACTIONMAP_OPPOSITEZ\nrefractionVector.z*=-1.0;\n#endif\n#ifdef SS_REFRACTIONMAP_3D\n#ifdef SS_USE_LOCAL_REFRACTIONMAP_CUBIC\nrefractionVector=parallaxCorrectNormal(vPositionW,refractionVector,refractionSize,refractionPosition);\n#endif\nrefractionVector.y=refractionVector.y*vRefractionInfos.w;vec3 refractionCoords=refractionVector;refractionCoords=vec3(refractionMatrix*vec4(refractionCoords,0));\n#else\n#ifdef SS_USE_THICKNESS_AS_DEPTH\nvec3 vRefractionUVW=vec3(refractionMatrix*(view*vec4(vPositionW+refractionVector*thickness,1.0)));\n#else\nvec3 vRefractionUVW=vec3(refractionMatrix*(view*vec4(vPositionW+refractionVector*vRefractionInfos.z,1.0)));\n#endif\nvec2 refractionCoords=vRefractionUVW.xy/vRefractionUVW.z;refractionCoords.y=1.0-refractionCoords.y;\n#endif\n#ifdef SS_HAS_THICKNESS\nfloat ior=vRefractionInfos.y;\n#else\nfloat ior=vRefractionMicrosurfaceInfos.w;\n#endif\n#ifdef SS_LODINREFRACTIONALPHA\nfloat refractionAlphaG=alphaG;refractionAlphaG=mix(alphaG,0.0,clamp(ior*3.0-2.0,0.0,1.0));float refractionLOD=getLodFromAlphaG(vRefractionMicrosurfaceInfos.x,refractionAlphaG,NdotVUnclamped);\n#elif defined(SS_LINEARSPECULARREFRACTION)\nfloat refractionRoughness=alphaG;refractionRoughness=mix(alphaG,0.0,clamp(ior*3.0-2.0,0.0,1.0));float refractionLOD=getLinearLodFromRoughness(vRefractionMicrosurfaceInfos.x,refractionRoughness);\n#else\nfloat refractionAlphaG=alphaG;refractionAlphaG=mix(alphaG,0.0,clamp(ior*3.0-2.0,0.0,1.0));float refractionLOD=getLodFromAlphaG(vRefractionMicrosurfaceInfos.x,refractionAlphaG);\n#endif\n#ifdef LODBASEDMICROSFURACE\nrefractionLOD=refractionLOD*vRefractionMicrosurfaceInfos.y+vRefractionMicrosurfaceInfos.z;\n#ifdef SS_LODINREFRACTIONALPHA\nfloat automaticRefractionLOD=UNPACK_LOD(sampleRefraction(refractionSampler,refractionCoords).a);float requestedRefractionLOD=max(automaticRefractionLOD,refractionLOD);\n#else\nfloat requestedRefractionLOD=refractionLOD;\n#endif\n#if defined(REALTIME_FILTERING) && defined(SS_REFRACTIONMAP_3D)\nenvironmentRefraction=vec4(radiance(alphaG,refractionSampler,refractionCoords,vRefractionFilteringInfo),1.0);\n#else\nenvironmentRefraction=sampleRefractionLod(refractionSampler,refractionCoords,requestedRefractionLOD);\n#endif\n#else\nfloat lodRefractionNormalized=saturate(refractionLOD/log2(vRefractionMicrosurfaceInfos.x));float lodRefractionNormalizedDoubled=lodRefractionNormalized*2.0;vec4 environmentRefractionMid=sampleRefraction(refractionSampler,refractionCoords);if (lodRefractionNormalizedDoubled<1.0){environmentRefraction=mix(\nsampleRefraction(refractionSamplerHigh,refractionCoords),\nenvironmentRefractionMid,\nlodRefractionNormalizedDoubled\n);} else {environmentRefraction=mix(\nenvironmentRefractionMid,\nsampleRefraction(refractionSamplerLow,refractionCoords),\nlodRefractionNormalizedDoubled-1.0\n);}\n#endif\n#ifdef SS_RGBDREFRACTION\nenvironmentRefraction.rgb=fromRGBD(environmentRefraction);\n#endif\n#ifdef SS_GAMMAREFRACTION\nenvironmentRefraction.rgb=toLinearSpace(environmentRefraction.rgb);\n#endif\nenvironmentRefraction.rgb*=vRefractionInfos.x;\n#endif\n#ifdef SS_REFRACTION\nvec3 refractionTransmittance=vec3(refractionIntensity);\n#ifdef SS_THICKNESSANDMASK_TEXTURE\nvec3 volumeAlbedo=computeColorAtDistanceInMedia(vTintColor.rgb,vTintColor.w);refractionTransmittance*=cocaLambert(volumeAlbedo,thickness);\n#elif defined(SS_LINKREFRACTIONTOTRANSPARENCY)\nfloat maxChannel=max(max(surfaceAlbedo.r,surfaceAlbedo.g),surfaceAlbedo.b);vec3 volumeAlbedo=saturate(maxChannel*surfaceAlbedo);environmentRefraction.rgb*=volumeAlbedo;\n#else\nvec3 volumeAlbedo=computeColorAtDistanceInMedia(vTintColor.rgb,vTintColor.w);refractionTransmittance*=cocaLambert(volumeAlbedo,vThicknessParam.y);\n#endif\n#ifdef SS_ALBEDOFORREFRACTIONTINT\nenvironmentRefraction.rgb*=surfaceAlbedo.rgb;\n#endif\noutParams.surfaceAlbedo=surfaceAlbedo*(1.-refractionIntensity);\n#ifdef REFLECTION\noutParams.refractionFactorForIrradiance=(1.-refractionIntensity);\n#endif\n#ifdef UNUSED_MULTIPLEBOUNCES\nvec3 bounceSpecularEnvironmentReflectance=(2.0*specularEnvironmentReflectance)/(1.0+specularEnvironmentReflectance);outParams.specularEnvironmentReflectance=mix(bounceSpecularEnvironmentReflectance,specularEnvironmentReflectance,refractionIntensity);\n#endif\nrefractionTransmittance*=1.0-outParams.specularEnvironmentReflectance;\n#if DEBUGMODE>0\noutParams.refractionTransmittance=refractionTransmittance;\n#endif\noutParams.finalRefraction=environmentRefraction.rgb*refractionTransmittance*vLightingIntensity.z;\n#if DEBUGMODE>0\noutParams.environmentRefraction=environmentRefraction;\n#endif\n#endif\n#if defined(REFLECTION) && defined(SS_TRANSLUCENCY)\n#if defined(NORMAL) && defined(USESPHERICALINVERTEX) || !defined(USESPHERICALFROMREFLECTIONMAP)\nvec3 irradianceVector=vec3(reflectionMatrix*vec4(normalW,0)).xyz;\n#ifdef REFLECTIONMAP_OPPOSITEZ\nirradianceVector.z*=-1.0;\n#endif\n#ifdef INVERTCUBICMAP\nirradianceVector.y*=-1.0;\n#endif\n#else\nvec3 irradianceVector=irradianceVector_;\n#endif\n#if defined(USESPHERICALFROMREFLECTIONMAP)\n#if defined(REALTIME_FILTERING)\nvec3 refractionIrradiance=irradiance(reflectionSampler,-irradianceVector,vReflectionFilteringInfo);\n#else\nvec3 refractionIrradiance=computeEnvironmentIrradiance(-irradianceVector);\n#endif\n#elif defined(USEIRRADIANCEMAP)\n#ifdef REFLECTIONMAP_3D\nvec3 irradianceCoords=irradianceVector;\n#else\nvec2 irradianceCoords=irradianceVector.xy;\n#ifdef REFLECTIONMAP_PROJECTION\nirradianceCoords/=irradianceVector.z;\n#endif\nirradianceCoords.y=1.0-irradianceCoords.y;\n#endif\nvec4 refractionIrradiance=sampleReflection(irradianceSampler,-irradianceCoords);\n#ifdef RGBDREFLECTION\nrefractionIrradiance.rgb=fromRGBD(refractionIrradiance);\n#endif\n#ifdef GAMMAREFLECTION\nrefractionIrradiance.rgb=toLinearSpace(refractionIrradiance.rgb);\n#endif\n#else\nvec4 refractionIrradiance=vec4(0.);\n#endif\nrefractionIrradiance.rgb*=transmittance;\n#ifdef SS_ALBEDOFORTRANSLUCENCYTINT\nrefractionIrradiance.rgb*=surfaceAlbedo.rgb;\n#endif\noutParams.refractionIrradiance=refractionIrradiance.rgb;\n#endif\n}\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1r] = shader$1r;\n\n // Do not edit.\n const name$1s = \"pbrBlockNormalGeometric\";\n const shader$1s = `vec3 viewDirectionW=normalize(vEyePosition.xyz-vPositionW);\n#ifdef NORMAL\nvec3 normalW=normalize(vNormalW);\n#else\nvec3 normalW=normalize(cross(dFdx(vPositionW),dFdy(vPositionW)))*vEyePosition.w;\n#endif\nvec3 geometricNormalW=normalW;\n#if defined(TWOSIDEDLIGHTING) && defined(NORMAL)\ngeometricNormalW=gl_FrontFacing ? geometricNormalW : -geometricNormalW;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1s] = shader$1s;\n\n // Do not edit.\n const name$1t = \"pbrBlockNormalFinal\";\n const shader$1t = `#if defined(FORCENORMALFORWARD) && defined(NORMAL)\nvec3 faceNormal=normalize(cross(dFdx(vPositionW),dFdy(vPositionW)))*vEyePosition.w;\n#if defined(TWOSIDEDLIGHTING)\nfaceNormal=gl_FrontFacing ? faceNormal : -faceNormal;\n#endif\nnormalW*=sign(dot(normalW,faceNormal));\n#endif\n#if defined(TWOSIDEDLIGHTING) && defined(NORMAL)\nnormalW=gl_FrontFacing ? normalW : -normalW;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1t] = shader$1t;\n\n // Do not edit.\n const name$1u = \"pbrBlockLightmapInit\";\n const shader$1u = `#ifdef LIGHTMAP\nvec4 lightmapColor=texture2D(lightmapSampler,vLightmapUV+uvOffset);\n#ifdef RGBDLIGHTMAP\nlightmapColor.rgb=fromRGBD(lightmapColor);\n#endif\n#ifdef GAMMALIGHTMAP\nlightmapColor.rgb=toLinearSpace(lightmapColor.rgb);\n#endif\nlightmapColor.rgb*=vLightmapInfos.y;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1u] = shader$1u;\n\n // Do not edit.\n const name$1v = \"pbrBlockGeometryInfo\";\n const shader$1v = `float NdotVUnclamped=dot(normalW,viewDirectionW);float NdotV=absEps(NdotVUnclamped);float alphaG=convertRoughnessToAverageSlope(roughness);vec2 AARoughnessFactors=getAARoughnessFactors(normalW.xyz);\n#ifdef SPECULARAA\nalphaG+=AARoughnessFactors.y;\n#endif\n#if defined(ENVIRONMENTBRDF)\nvec3 environmentBrdf=getBRDFLookup(NdotV,roughness);\n#endif\n#if defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX)\n#ifdef RADIANCEOCCLUSION\n#ifdef AMBIENTINGRAYSCALE\nfloat ambientMonochrome=aoOut.ambientOcclusionColor.r;\n#else\nfloat ambientMonochrome=getLuminance(aoOut.ambientOcclusionColor);\n#endif\nfloat seo=environmentRadianceOcclusion(ambientMonochrome,NdotVUnclamped);\n#endif\n#ifdef HORIZONOCCLUSION\n#ifdef BUMP\n#ifdef REFLECTIONMAP_3D\nfloat eho=environmentHorizonOcclusion(-viewDirectionW,normalW,geometricNormalW);\n#endif\n#endif\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1v] = shader$1v;\n\n // Do not edit.\n const name$1w = \"pbrBlockReflectance0\";\n const shader$1w = `float reflectance=max(max(reflectivityOut.surfaceReflectivityColor.r,reflectivityOut.surfaceReflectivityColor.g),reflectivityOut.surfaceReflectivityColor.b);vec3 specularEnvironmentR0=reflectivityOut.surfaceReflectivityColor.rgb;\n#ifdef METALLICWORKFLOW\nvec3 specularEnvironmentR90=vec3(metallicReflectanceFactors.a);\n#else \nvec3 specularEnvironmentR90=vec3(1.0,1.0,1.0);\n#endif\n#ifdef ALPHAFRESNEL\nfloat reflectance90=fresnelGrazingReflectance(reflectance);specularEnvironmentR90=specularEnvironmentR90*reflectance90;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1w] = shader$1w;\n\n // Do not edit.\n const name$1x = \"pbrBlockReflectance\";\n const shader$1x = `#if defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX)\nvec3 specularEnvironmentReflectance=getReflectanceFromBRDFLookup(clearcoatOut.specularEnvironmentR0,specularEnvironmentR90,environmentBrdf);\n#ifdef RADIANCEOCCLUSION\nspecularEnvironmentReflectance*=seo;\n#endif\n#ifdef HORIZONOCCLUSION\n#ifdef BUMP\n#ifdef REFLECTIONMAP_3D\nspecularEnvironmentReflectance*=eho;\n#endif\n#endif\n#endif\n#else\nvec3 specularEnvironmentReflectance=getReflectanceFromAnalyticalBRDFLookup_Jones(NdotV,clearcoatOut.specularEnvironmentR0,specularEnvironmentR90,sqrt(microSurface));\n#endif\n#ifdef CLEARCOAT\nspecularEnvironmentReflectance*=clearcoatOut.conservationFactor;\n#if defined(CLEARCOAT_TINT)\nspecularEnvironmentReflectance*=clearcoatOut.absorption;\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1x] = shader$1x;\n\n // Do not edit.\n const name$1y = \"pbrBlockDirectLighting\";\n const shader$1y = `vec3 diffuseBase=vec3(0.,0.,0.);\n#ifdef SPECULARTERM\nvec3 specularBase=vec3(0.,0.,0.);\n#endif\n#ifdef CLEARCOAT\nvec3 clearCoatBase=vec3(0.,0.,0.);\n#endif\n#ifdef SHEEN\nvec3 sheenBase=vec3(0.,0.,0.);\n#endif\npreLightingInfo preInfo;lightingInfo info;float shadow=1.; \n#if defined(CLEARCOAT) && defined(CLEARCOAT_TINT)\nvec3 absorption=vec3(0.);\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1y] = shader$1y;\n\n // Do not edit.\n const name$1z = \"pbrBlockFinalLitComponents\";\n const shader$1z = `#if defined(ENVIRONMENTBRDF)\n#ifdef MS_BRDF_ENERGY_CONSERVATION\nvec3 energyConservationFactor=getEnergyConservationFactor(clearcoatOut.specularEnvironmentR0,environmentBrdf);\n#endif\n#endif\n#ifndef METALLICWORKFLOW\n#ifdef SPECULAR_GLOSSINESS_ENERGY_CONSERVATION\nsurfaceAlbedo.rgb=(1.-reflectance)*surfaceAlbedo.rgb;\n#endif\n#endif\n#if defined(SHEEN) && defined(SHEEN_ALBEDOSCALING) && defined(ENVIRONMENTBRDF)\nsurfaceAlbedo.rgb=sheenOut.sheenAlbedoScaling*surfaceAlbedo.rgb;\n#endif\n#ifdef REFLECTION\nvec3 finalIrradiance=reflectionOut.environmentIrradiance;\n#if defined(CLEARCOAT)\nfinalIrradiance*=clearcoatOut.conservationFactor;\n#if defined(CLEARCOAT_TINT)\nfinalIrradiance*=clearcoatOut.absorption;\n#endif\n#endif\n#if defined(SS_REFRACTION)\nfinalIrradiance*=subSurfaceOut.refractionFactorForIrradiance;\n#endif\n#if defined(SS_TRANSLUCENCY)\nfinalIrradiance*=(1.0-subSurfaceOut.translucencyIntensity);finalIrradiance+=subSurfaceOut.refractionIrradiance;\n#endif\nfinalIrradiance*=surfaceAlbedo.rgb;finalIrradiance*=vLightingIntensity.z;finalIrradiance*=aoOut.ambientOcclusionColor;\n#endif\n#ifdef SPECULARTERM\nvec3 finalSpecular=specularBase;finalSpecular=max(finalSpecular,0.0);vec3 finalSpecularScaled=finalSpecular*vLightingIntensity.x*vLightingIntensity.w;\n#if defined(ENVIRONMENTBRDF) && defined(MS_BRDF_ENERGY_CONSERVATION)\nfinalSpecularScaled*=energyConservationFactor;\n#endif\n#if defined(SHEEN) && defined(ENVIRONMENTBRDF) && defined(SHEEN_ALBEDOSCALING)\nfinalSpecularScaled*=sheenOut.sheenAlbedoScaling;\n#endif\n#endif\n#ifdef REFLECTION\nvec3 finalRadiance=reflectionOut.environmentRadiance.rgb;finalRadiance*=subSurfaceOut.specularEnvironmentReflectance;vec3 finalRadianceScaled=finalRadiance*vLightingIntensity.z;\n#if defined(ENVIRONMENTBRDF) && defined(MS_BRDF_ENERGY_CONSERVATION)\nfinalRadianceScaled*=energyConservationFactor;\n#endif\n#if defined(SHEEN) && defined(ENVIRONMENTBRDF) && defined(SHEEN_ALBEDOSCALING)\nfinalRadianceScaled*=sheenOut.sheenAlbedoScaling;\n#endif\n#endif\n#ifdef SHEEN\nvec3 finalSheen=sheenBase*sheenOut.sheenColor;finalSheen=max(finalSheen,0.0);vec3 finalSheenScaled=finalSheen*vLightingIntensity.x*vLightingIntensity.w;\n#if defined(CLEARCOAT) && defined(REFLECTION) && defined(ENVIRONMENTBRDF)\nsheenOut.finalSheenRadianceScaled*=clearcoatOut.conservationFactor;\n#if defined(CLEARCOAT_TINT)\nsheenOut.finalSheenRadianceScaled*=clearcoatOut.absorption;\n#endif\n#endif\n#endif\n#ifdef CLEARCOAT\nvec3 finalClearCoat=clearCoatBase;finalClearCoat=max(finalClearCoat,0.0);vec3 finalClearCoatScaled=finalClearCoat*vLightingIntensity.x*vLightingIntensity.w;\n#if defined(ENVIRONMENTBRDF) && defined(MS_BRDF_ENERGY_CONSERVATION)\nfinalClearCoatScaled*=clearcoatOut.energyConservationFactorClearCoat;\n#endif\n#ifdef SS_REFRACTION\nsubSurfaceOut.finalRefraction*=clearcoatOut.conservationFactor;\n#ifdef CLEARCOAT_TINT\nsubSurfaceOut.finalRefraction*=clearcoatOut.absorption;\n#endif\n#endif\n#endif\n#ifdef ALPHABLEND\nfloat luminanceOverAlpha=0.0;\n#if defined(REFLECTION) && defined(RADIANCEOVERALPHA)\nluminanceOverAlpha+=getLuminance(finalRadianceScaled);\n#if defined(CLEARCOAT)\nluminanceOverAlpha+=getLuminance(clearcoatOut.finalClearCoatRadianceScaled);\n#endif\n#endif\n#if defined(SPECULARTERM) && defined(SPECULAROVERALPHA)\nluminanceOverAlpha+=getLuminance(finalSpecularScaled);\n#endif\n#if defined(CLEARCOAT) && defined(CLEARCOATOVERALPHA)\nluminanceOverAlpha+=getLuminance(finalClearCoatScaled);\n#endif\n#if defined(RADIANCEOVERALPHA) || defined(SPECULAROVERALPHA) || defined(CLEARCOATOVERALPHA)\nalpha=saturate(alpha+luminanceOverAlpha*luminanceOverAlpha);\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1z] = shader$1z;\n\n // Do not edit.\n const name$1A = \"pbrBlockFinalUnlitComponents\";\n const shader$1A = `vec3 finalDiffuse=diffuseBase;finalDiffuse*=surfaceAlbedo.rgb;finalDiffuse=max(finalDiffuse,0.0);finalDiffuse*=vLightingIntensity.x;vec3 finalAmbient=vAmbientColor;finalAmbient*=surfaceAlbedo.rgb;vec3 finalEmissive=vEmissiveColor;\n#ifdef EMISSIVE\nvec3 emissiveColorTex=texture2D(emissiveSampler,vEmissiveUV+uvOffset).rgb;\n#ifdef GAMMAEMISSIVE\nfinalEmissive*=toLinearSpace(emissiveColorTex.rgb);\n#else\nfinalEmissive*=emissiveColorTex.rgb;\n#endif\nfinalEmissive*= vEmissiveInfos.y;\n#endif\nfinalEmissive*=vLightingIntensity.y;\n#ifdef AMBIENT\nvec3 ambientOcclusionForDirectDiffuse=mix(vec3(1.),aoOut.ambientOcclusionColor,vAmbientInfos.w);\n#else\nvec3 ambientOcclusionForDirectDiffuse=aoOut.ambientOcclusionColor;\n#endif\nfinalAmbient*=aoOut.ambientOcclusionColor;finalDiffuse*=ambientOcclusionForDirectDiffuse;\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1A] = shader$1A;\n\n // Do not edit.\n const name$1B = \"pbrBlockFinalColorComposition\";\n const shader$1B = `vec4 finalColor=vec4(\n#ifndef UNLIT\n#ifdef REFLECTION\nfinalIrradiance +\n#endif\n#ifdef SPECULARTERM\nfinalSpecularScaled +\n#endif\n#ifdef SHEEN\nfinalSheenScaled +\n#endif\n#ifdef CLEARCOAT\nfinalClearCoatScaled +\n#endif\n#ifdef REFLECTION\nfinalRadianceScaled +\n#if defined(SHEEN) && defined(ENVIRONMENTBRDF)\nsheenOut.finalSheenRadianceScaled +\n#endif\n#ifdef CLEARCOAT\nclearcoatOut.finalClearCoatRadianceScaled +\n#endif\n#endif\n#ifdef SS_REFRACTION\nsubSurfaceOut.finalRefraction +\n#endif\n#endif\nfinalAmbient +\nfinalDiffuse,\nalpha);\n#ifdef LIGHTMAP\n#ifndef LIGHTMAPEXCLUDED\n#ifdef USELIGHTMAPASSHADOWMAP\nfinalColor.rgb*=lightmapColor.rgb;\n#else\nfinalColor.rgb+=lightmapColor.rgb;\n#endif\n#endif\n#endif\nfinalColor.rgb+=finalEmissive;\n#define CUSTOM_FRAGMENT_BEFORE_FOG\nfinalColor=max(finalColor,0.0);\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1B] = shader$1B;\n\n // Do not edit.\n const name$1C = \"pbrBlockImageProcessing\";\n const shader$1C = `#if defined(IMAGEPROCESSINGPOSTPROCESS) || defined(SS_SCATTERING)\n#if !defined(SKIPFINALCOLORCLAMP)\nfinalColor.rgb=clamp(finalColor.rgb,0.,30.0);\n#endif\n#else\nfinalColor=applyImageProcessing(finalColor);\n#endif\nfinalColor.a*=visibility;\n#ifdef PREMULTIPLYALPHA\nfinalColor.rgb*=finalColor.a;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1C] = shader$1C;\n\n // Do not edit.\n const name$1D = \"pbrDebug\";\n const shader$1D = `#if DEBUGMODE>0\nif (vClipSpacePosition.x/vClipSpacePosition.w>=vDebugMode.x) {\n#if DEBUGMODE==1\ngl_FragColor.rgb=vPositionW.rgb;\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==2 && defined(NORMAL)\ngl_FragColor.rgb=vNormalW.rgb;\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==3 && defined(BUMP) || DEBUGMODE==3 && defined(PARALLAX) || DEBUGMODE==3 && defined(ANISOTROPIC)\ngl_FragColor.rgb=TBN[0];\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==4 && defined(BUMP) || DEBUGMODE==4 && defined(PARALLAX) || DEBUGMODE==4 && defined(ANISOTROPIC)\ngl_FragColor.rgb=TBN[1];\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==5\ngl_FragColor.rgb=normalW;\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==6 && defined(MAINUV1)\ngl_FragColor.rgb=vec3(vMainUV1,0.0);\n#elif DEBUGMODE==7 && defined(MAINUV2)\ngl_FragColor.rgb=vec3(vMainUV2,0.0);\n#elif DEBUGMODE==8 && defined(CLEARCOAT) && defined(CLEARCOAT_BUMP)\ngl_FragColor.rgb=clearcoatOut.TBNClearCoat[0];\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==9 && defined(CLEARCOAT) && defined(CLEARCOAT_BUMP)\ngl_FragColor.rgb=clearcoatOut.TBNClearCoat[1];\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==10 && defined(CLEARCOAT)\ngl_FragColor.rgb=clearcoatOut.clearCoatNormalW;\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==11 && defined(ANISOTROPIC)\ngl_FragColor.rgb=anisotropicOut.anisotropicNormal;\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==12 && defined(ANISOTROPIC)\ngl_FragColor.rgb=anisotropicOut.anisotropicTangent;\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==13 && defined(ANISOTROPIC)\ngl_FragColor.rgb=anisotropicOut.anisotropicBitangent;\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==20 && defined(ALBEDO)\ngl_FragColor.rgb=albedoTexture.rgb;\n#ifndef GAMMAALBEDO\n#define DEBUGMODE_GAMMA\n#endif\n#elif DEBUGMODE==21 && defined(AMBIENT)\ngl_FragColor.rgb=aoOut.ambientOcclusionColorMap.rgb;\n#elif DEBUGMODE==22 && defined(OPACITY)\ngl_FragColor.rgb=opacityMap.rgb;\n#elif DEBUGMODE==23 && defined(EMISSIVE)\ngl_FragColor.rgb=emissiveColorTex.rgb;\n#ifndef GAMMAEMISSIVE\n#define DEBUGMODE_GAMMA\n#endif\n#elif DEBUGMODE==24 && defined(LIGHTMAP)\ngl_FragColor.rgb=lightmapColor.rgb;\n#ifndef GAMMALIGHTMAP\n#define DEBUGMODE_GAMMA\n#endif\n#elif DEBUGMODE==25 && defined(REFLECTIVITY) && defined(METALLICWORKFLOW)\ngl_FragColor.rgb=reflectivityOut.surfaceMetallicColorMap.rgb;\n#elif DEBUGMODE==26 && defined(REFLECTIVITY) && !defined(METALLICWORKFLOW)\ngl_FragColor.rgb=reflectivityOut.surfaceReflectivityColorMap.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==27 && defined(CLEARCOAT) && defined(CLEARCOAT_TEXTURE)\ngl_FragColor.rgb=vec3(clearcoatOut.clearCoatMapData.rg,0.0);\n#elif DEBUGMODE==28 && defined(CLEARCOAT) && defined(CLEARCOAT_TINT) && defined(CLEARCOAT_TINT_TEXTURE)\ngl_FragColor.rgb=clearcoatOut.clearCoatTintMapData.rgb;\n#elif DEBUGMODE==29 && defined(SHEEN) && defined(SHEEN_TEXTURE)\ngl_FragColor.rgb=sheenOut.sheenMapData.rgb;\n#elif DEBUGMODE==30 && defined(ANISOTROPIC) && defined(ANISOTROPIC_TEXTURE)\ngl_FragColor.rgb=anisotropicOut.anisotropyMapData.rgb;\n#elif DEBUGMODE==31 && defined(SUBSURFACE) && defined(SS_THICKNESSANDMASK_TEXTURE)\ngl_FragColor.rgb=subSurfaceOut.thicknessMap.rgb;\n#elif DEBUGMODE==32 && defined(BUMP)\ngl_FragColor.rgb=texture2D(bumpSampler,vBumpUV).rgb;\n#elif DEBUGMODE==40 && defined(SS_REFRACTION)\ngl_FragColor.rgb=subSurfaceOut.environmentRefraction.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==41 && defined(REFLECTION)\ngl_FragColor.rgb=reflectionOut.environmentRadiance.rgb;\n#ifndef GAMMAREFLECTION\n#define DEBUGMODE_GAMMA\n#endif\n#elif DEBUGMODE==42 && defined(CLEARCOAT) && defined(REFLECTION)\ngl_FragColor.rgb=clearcoatOut.environmentClearCoatRadiance.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==50\ngl_FragColor.rgb=diffuseBase.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==51 && defined(SPECULARTERM)\ngl_FragColor.rgb=specularBase.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==52 && defined(CLEARCOAT)\ngl_FragColor.rgb=clearCoatBase.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==53 && defined(SHEEN)\ngl_FragColor.rgb=sheenBase.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==54 && defined(REFLECTION)\ngl_FragColor.rgb=reflectionOut.environmentIrradiance.rgb;\n#ifndef GAMMAREFLECTION\n#define DEBUGMODE_GAMMA\n#endif\n#elif DEBUGMODE==60\ngl_FragColor.rgb=surfaceAlbedo.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==61\ngl_FragColor.rgb=clearcoatOut.specularEnvironmentR0;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==62 && defined(METALLICWORKFLOW)\ngl_FragColor.rgb=vec3(reflectivityOut.metallicRoughness.r);\n#elif DEBUGMODE==71 && defined(METALLICWORKFLOW)\ngl_FragColor.rgb=reflectivityOut.metallicF0;\n#elif DEBUGMODE==63\ngl_FragColor.rgb=vec3(roughness);\n#elif DEBUGMODE==64\ngl_FragColor.rgb=vec3(alphaG);\n#elif DEBUGMODE==65\ngl_FragColor.rgb=vec3(NdotV);\n#elif DEBUGMODE==66 && defined(CLEARCOAT) && defined(CLEARCOAT_TINT)\ngl_FragColor.rgb=clearcoatOut.clearCoatColor.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==67 && defined(CLEARCOAT)\ngl_FragColor.rgb=vec3(clearcoatOut.clearCoatRoughness);\n#elif DEBUGMODE==68 && defined(CLEARCOAT)\ngl_FragColor.rgb=vec3(clearcoatOut.clearCoatNdotV);\n#elif DEBUGMODE==69 && defined(SUBSURFACE) && defined(SS_TRANSLUCENCY)\ngl_FragColor.rgb=subSurfaceOut.transmittance;\n#elif DEBUGMODE==70 && defined(SUBSURFACE) && defined(SS_REFRACTION)\ngl_FragColor.rgb=subSurfaceOut.refractionTransmittance;\n#elif DEBUGMODE==72\ngl_FragColor.rgb=vec3(microSurface);\n#elif DEBUGMODE==73\ngl_FragColor.rgb=vAlbedoColor.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==74 && !defined(METALLICWORKFLOW)\ngl_FragColor.rgb=vReflectivityColor.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==75\ngl_FragColor.rgb=vEmissiveColor.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==80 && defined(RADIANCEOCCLUSION)\ngl_FragColor.rgb=vec3(seo);\n#elif DEBUGMODE==81 && defined(HORIZONOCCLUSION)\ngl_FragColor.rgb=vec3(eho);\n#elif DEBUGMODE==82 && defined(MS_BRDF_ENERGY_CONSERVATION)\ngl_FragColor.rgb=vec3(energyConservationFactor);\n#elif DEBUGMODE==83 && defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX)\ngl_FragColor.rgb=specularEnvironmentReflectance;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==84 && defined(CLEARCOAT) && defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX)\ngl_FragColor.rgb=clearcoatOut.clearCoatEnvironmentReflectance;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==85 && defined(SHEEN) && defined(REFLECTION)\ngl_FragColor.rgb=sheenOut.sheenEnvironmentReflectance;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==86 && defined(ALPHABLEND)\ngl_FragColor.rgb=vec3(luminanceOverAlpha);\n#elif DEBUGMODE==87\ngl_FragColor.rgb=vec3(alpha);\n#elif DEBUGMODE==88 && defined(ALBEDO)\ngl_FragColor.rgb=vec3(albedoTexture.a);\n#else\nfloat stripeWidth=30.;float stripePos=floor((gl_FragCoord.x+gl_FragCoord.y)/stripeWidth);float whichColor=mod(stripePos,2.);vec3 color1=vec3(.6,.2,.2);vec3 color2=vec3(.3,.1,.1);gl_FragColor.rgb=mix(color1,color2,whichColor);\n#endif\ngl_FragColor.rgb*=vDebugMode.y;\n#ifdef DEBUGMODE_NORMALIZE\ngl_FragColor.rgb=normalize(gl_FragColor.rgb)*0.5+0.5;\n#endif\n#ifdef DEBUGMODE_GAMMA\ngl_FragColor.rgb=toGammaSpace(gl_FragColor.rgb);\n#endif\ngl_FragColor.a=1.0;\n#ifdef PREPASS\ngl_FragData[0]=toLinearSpace(gl_FragColor); \ngl_FragData[1]=vec4(0.,0.,0.,0.); \n#endif\n}\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1D] = shader$1D;\n\n // Do not edit.\n const name$1E = \"pbrPixelShader\";\n const shader$1E = `#if defined(BUMP) || !defined(NORMAL) || defined(FORCENORMALFORWARD) || defined(SPECULARAA) || defined(CLEARCOAT_BUMP) || defined(ANISOTROPIC)\n#extension GL_OES_standard_derivatives : enable\n#endif\n#ifdef LODBASEDMICROSFURACE\n#extension GL_EXT_shader_texture_lod : enable\n#endif\n#define CUSTOM_FRAGMENT_BEGIN\n#ifdef LOGARITHMICDEPTH\n#extension GL_EXT_frag_depth : enable\n#endif\n#include[SCENE_MRT_COUNT]\nprecision highp float;\n#include\n#ifndef FROMLINEARSPACE\n#define FROMLINEARSPACE\n#endif\n#include<__decl__pbrFragment>\n#include\n#include<__decl__lightFragment>[0..maxSimultaneousLights]\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#ifdef REFLECTION\n#include\n#endif\n#define CUSTOM_FRAGMENT_DEFINITIONS\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\nvoid main(void) {\n#define CUSTOM_FRAGMENT_MAIN_BEGIN\n#include\n#include\n#include\n#include\nalbedoOpacityOutParams albedoOpacityOut;\n#ifdef ALBEDO\nvec4 albedoTexture=texture2D(albedoSampler,vAlbedoUV+uvOffset);\n#endif\n#ifdef OPACITY\nvec4 opacityMap=texture2D(opacitySampler,vOpacityUV+uvOffset);\n#endif\n#ifdef DECAL\nvec4 decalColor=texture2D(decalSampler,vDecalUV+uvOffset);\n#endif\nalbedoOpacityBlock(\nvAlbedoColor,\n#ifdef ALBEDO\nalbedoTexture,\nvAlbedoInfos,\n#endif\n#ifdef OPACITY\nopacityMap,\nvOpacityInfos,\n#endif\n#ifdef DETAIL\ndetailColor,\nvDetailInfos,\n#endif\n#ifdef DECAL\ndecalColor,\nvDecalInfos,\n#endif\nalbedoOpacityOut\n);vec3 surfaceAlbedo=albedoOpacityOut.surfaceAlbedo;float alpha=albedoOpacityOut.alpha;\n#define CUSTOM_FRAGMENT_UPDATE_ALPHA\n#include\n#define CUSTOM_FRAGMENT_BEFORE_LIGHTS\nambientOcclusionOutParams aoOut;\n#ifdef AMBIENT\nvec3 ambientOcclusionColorMap=texture2D(ambientSampler,vAmbientUV+uvOffset).rgb;\n#endif\nambientOcclusionBlock(\n#ifdef AMBIENT\nambientOcclusionColorMap,\nvAmbientInfos,\n#endif\naoOut\n);\n#include\n#ifdef UNLIT\nvec3 diffuseBase=vec3(1.,1.,1.);\n#else\nvec3 baseColor=surfaceAlbedo;reflectivityOutParams reflectivityOut;\n#if defined(REFLECTIVITY)\nvec4 surfaceMetallicOrReflectivityColorMap=texture2D(reflectivitySampler,vReflectivityUV+uvOffset);vec4 baseReflectivity=surfaceMetallicOrReflectivityColorMap;\n#ifndef METALLICWORKFLOW\n#ifdef REFLECTIVITY_GAMMA\nsurfaceMetallicOrReflectivityColorMap=toLinearSpace(surfaceMetallicOrReflectivityColorMap);\n#endif\nsurfaceMetallicOrReflectivityColorMap.rgb*=vReflectivityInfos.y;\n#endif\n#endif\n#if defined(MICROSURFACEMAP)\nvec4 microSurfaceTexel=texture2D(microSurfaceSampler,vMicroSurfaceSamplerUV+uvOffset)*vMicroSurfaceSamplerInfos.y;\n#endif\n#ifdef METALLICWORKFLOW\nvec4 metallicReflectanceFactors=vMetallicReflectanceFactors;\n#ifdef REFLECTANCE\nvec4 reflectanceFactorsMap=texture2D(reflectanceSampler,vReflectanceUV+uvOffset);\n#ifdef REFLECTANCE_GAMMA\nreflectanceFactorsMap=toLinearSpace(reflectanceFactorsMap);\n#endif\nmetallicReflectanceFactors.rgb*=reflectanceFactorsMap.rgb;\n#endif\n#ifdef METALLIC_REFLECTANCE\nvec4 metallicReflectanceFactorsMap=texture2D(metallicReflectanceSampler,vMetallicReflectanceUV+uvOffset);\n#ifdef METALLIC_REFLECTANCE_GAMMA\nmetallicReflectanceFactorsMap=toLinearSpace(metallicReflectanceFactorsMap);\n#endif\n#ifndef METALLIC_REFLECTANCE_USE_ALPHA_ONLY\nmetallicReflectanceFactors.rgb*=metallicReflectanceFactorsMap.rgb;\n#endif\nmetallicReflectanceFactors*=metallicReflectanceFactorsMap.a;\n#endif\n#endif\nreflectivityBlock(\nvReflectivityColor,\n#ifdef METALLICWORKFLOW\nsurfaceAlbedo,\nmetallicReflectanceFactors,\n#endif\n#ifdef REFLECTIVITY\nvReflectivityInfos,\nsurfaceMetallicOrReflectivityColorMap,\n#endif\n#if defined(METALLICWORKFLOW) && defined(REFLECTIVITY) && defined(AOSTOREINMETALMAPRED)\naoOut.ambientOcclusionColor,\n#endif\n#ifdef MICROSURFACEMAP\nmicroSurfaceTexel,\n#endif\n#ifdef DETAIL\ndetailColor,\nvDetailInfos,\n#endif\nreflectivityOut\n);float microSurface=reflectivityOut.microSurface;float roughness=reflectivityOut.roughness;\n#ifdef METALLICWORKFLOW\nsurfaceAlbedo=reflectivityOut.surfaceAlbedo;\n#endif\n#if defined(METALLICWORKFLOW) && defined(REFLECTIVITY) && defined(AOSTOREINMETALMAPRED)\naoOut.ambientOcclusionColor=reflectivityOut.ambientOcclusionColor;\n#endif\n#ifdef ALPHAFRESNEL\n#if defined(ALPHATEST) || defined(ALPHABLEND)\nalphaFresnelOutParams alphaFresnelOut;alphaFresnelBlock(\nnormalW,\nviewDirectionW,\nalpha,\nmicroSurface,\nalphaFresnelOut\n);alpha=alphaFresnelOut.alpha;\n#endif\n#endif\n#include\n#ifdef ANISOTROPIC\nanisotropicOutParams anisotropicOut;\n#ifdef ANISOTROPIC_TEXTURE\nvec3 anisotropyMapData=texture2D(anisotropySampler,vAnisotropyUV+uvOffset).rgb*vAnisotropyInfos.y;\n#endif\nanisotropicBlock(\nvAnisotropy,\nroughness,\n#ifdef ANISOTROPIC_TEXTURE\nanisotropyMapData,\n#endif\nTBN,\nnormalW,\nviewDirectionW,\nanisotropicOut\n);\n#endif\n#ifdef REFLECTION\nreflectionOutParams reflectionOut;\n#ifndef USE_CUSTOM_REFLECTION\nreflectionBlock(\nvPositionW,\nnormalW,\nalphaG,\nvReflectionMicrosurfaceInfos,\nvReflectionInfos,\nvReflectionColor,\n#ifdef ANISOTROPIC\nanisotropicOut,\n#endif\n#if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX)\nNdotVUnclamped,\n#endif\n#ifdef LINEARSPECULARREFLECTION\nroughness,\n#endif\nreflectionSampler,\n#if defined(NORMAL) && defined(USESPHERICALINVERTEX)\nvEnvironmentIrradiance,\n#endif\n#ifdef USESPHERICALFROMREFLECTIONMAP\n#if !defined(NORMAL) || !defined(USESPHERICALINVERTEX)\nreflectionMatrix,\n#endif\n#endif\n#ifdef USEIRRADIANCEMAP\nirradianceSampler,\n#endif\n#ifndef LODBASEDMICROSFURACE\nreflectionSamplerLow,\nreflectionSamplerHigh,\n#endif\n#ifdef REALTIME_FILTERING\nvReflectionFilteringInfo,\n#endif\nreflectionOut\n);\n#else\n#define CUSTOM_REFLECTION\n#endif\n#endif\n#include\n#ifdef SHEEN\nsheenOutParams sheenOut;\n#ifdef SHEEN_TEXTURE\nvec4 sheenMapData=texture2D(sheenSampler,vSheenUV+uvOffset);\n#endif\n#if defined(SHEEN_ROUGHNESS) && defined(SHEEN_TEXTURE_ROUGHNESS) && !defined(SHEEN_TEXTURE_ROUGHNESS_IDENTICAL) && !defined(SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE)\nvec4 sheenMapRoughnessData=texture2D(sheenRoughnessSampler,vSheenRoughnessUV+uvOffset)*vSheenInfos.w;\n#endif\nsheenBlock(\nvSheenColor,\n#ifdef SHEEN_ROUGHNESS\nvSheenRoughness,\n#if defined(SHEEN_TEXTURE_ROUGHNESS) && !defined(SHEEN_TEXTURE_ROUGHNESS_IDENTICAL) && !defined(SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE)\nsheenMapRoughnessData,\n#endif\n#endif\nroughness,\n#ifdef SHEEN_TEXTURE\nsheenMapData,\nvSheenInfos.y,\n#endif\nreflectance,\n#ifdef SHEEN_LINKWITHALBEDO\nbaseColor,\nsurfaceAlbedo,\n#endif\n#ifdef ENVIRONMENTBRDF\nNdotV,\nenvironmentBrdf,\n#endif\n#if defined(REFLECTION) && defined(ENVIRONMENTBRDF)\nAARoughnessFactors,\nvReflectionMicrosurfaceInfos,\nvReflectionInfos,\nvReflectionColor,\nvLightingIntensity,\nreflectionSampler,\nreflectionOut.reflectionCoords,\nNdotVUnclamped,\n#ifndef LODBASEDMICROSFURACE\nreflectionSamplerLow,\nreflectionSamplerHigh,\n#endif\n#ifdef REALTIME_FILTERING\nvReflectionFilteringInfo,\n#endif\n#if !defined(REFLECTIONMAP_SKYBOX) && defined(RADIANCEOCCLUSION)\nseo,\n#endif\n#if !defined(REFLECTIONMAP_SKYBOX) && defined(HORIZONOCCLUSION) && defined(BUMP) && defined(REFLECTIONMAP_3D)\neho,\n#endif\n#endif\nsheenOut\n);\n#ifdef SHEEN_LINKWITHALBEDO\nsurfaceAlbedo=sheenOut.surfaceAlbedo;\n#endif\n#endif\n#ifdef CLEARCOAT\n#ifdef CLEARCOAT_TEXTURE\nvec2 clearCoatMapData=texture2D(clearCoatSampler,vClearCoatUV+uvOffset).rg*vClearCoatInfos.y;\n#endif\n#endif\n#ifdef IRIDESCENCE\niridescenceOutParams iridescenceOut;\n#ifdef IRIDESCENCE_TEXTURE\nvec2 iridescenceMapData=texture2D(iridescenceSampler,vIridescenceUV+uvOffset).rg*vIridescenceInfos.y;\n#endif\n#ifdef IRIDESCENCE_THICKNESS_TEXTURE\nvec2 iridescenceThicknessMapData=texture2D(iridescenceThicknessSampler,vIridescenceThicknessUV+uvOffset).rg*vIridescenceInfos.w;\n#endif\niridescenceBlock(\nvIridescenceParams,\nNdotV,\nspecularEnvironmentR0,\n#ifdef IRIDESCENCE_TEXTURE\niridescenceMapData,\n#endif\n#ifdef IRIDESCENCE_THICKNESS_TEXTURE\niridescenceThicknessMapData,\n#endif\n#ifdef CLEARCOAT\nNdotVUnclamped,\n#ifdef CLEARCOAT_TEXTURE\nclearCoatMapData,\n#endif\n#endif\niridescenceOut\n);float iridescenceIntensity=iridescenceOut.iridescenceIntensity;specularEnvironmentR0=iridescenceOut.specularEnvironmentR0;\n#endif\nclearcoatOutParams clearcoatOut;\n#ifdef CLEARCOAT\n#if defined(CLEARCOAT_TEXTURE_ROUGHNESS) && !defined(CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL) && !defined(CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE)\nvec4 clearCoatMapRoughnessData=texture2D(clearCoatRoughnessSampler,vClearCoatRoughnessUV+uvOffset)*vClearCoatInfos.w;\n#endif\n#if defined(CLEARCOAT_TINT) && defined(CLEARCOAT_TINT_TEXTURE)\nvec4 clearCoatTintMapData=texture2D(clearCoatTintSampler,vClearCoatTintUV+uvOffset);\n#endif\n#ifdef CLEARCOAT_BUMP\nvec4 clearCoatBumpMapData=texture2D(clearCoatBumpSampler,vClearCoatBumpUV+uvOffset);\n#endif\nclearcoatBlock(\nvPositionW,\ngeometricNormalW,\nviewDirectionW,\nvClearCoatParams,\n#if defined(CLEARCOAT_TEXTURE_ROUGHNESS) && !defined(CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL) && !defined(CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE)\nclearCoatMapRoughnessData,\n#endif\nspecularEnvironmentR0,\n#ifdef CLEARCOAT_TEXTURE\nclearCoatMapData,\n#endif\n#ifdef CLEARCOAT_TINT\nvClearCoatTintParams,\nclearCoatColorAtDistance,\nvClearCoatRefractionParams,\n#ifdef CLEARCOAT_TINT_TEXTURE\nclearCoatTintMapData,\n#endif\n#endif\n#ifdef CLEARCOAT_BUMP\nvClearCoatBumpInfos,\nclearCoatBumpMapData,\nvClearCoatBumpUV,\n#if defined(TANGENT) && defined(NORMAL)\nvTBN,\n#else\nvClearCoatTangentSpaceParams,\n#endif\n#ifdef OBJECTSPACE_NORMALMAP\nnormalMatrix,\n#endif\n#endif\n#if defined(FORCENORMALFORWARD) && defined(NORMAL)\nfaceNormal,\n#endif\n#ifdef REFLECTION\nvReflectionMicrosurfaceInfos,\nvReflectionInfos,\nvReflectionColor,\nvLightingIntensity,\nreflectionSampler,\n#ifndef LODBASEDMICROSFURACE\nreflectionSamplerLow,\nreflectionSamplerHigh,\n#endif\n#ifdef REALTIME_FILTERING\nvReflectionFilteringInfo,\n#endif\n#endif\n#if defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX)\n#ifdef RADIANCEOCCLUSION\nambientMonochrome,\n#endif\n#endif\n#if defined(CLEARCOAT_BUMP) || defined(TWOSIDEDLIGHTING)\n(gl_FrontFacing ? 1. : -1.),\n#endif\nclearcoatOut\n);\n#else\nclearcoatOut.specularEnvironmentR0=specularEnvironmentR0;\n#endif\n#include\nsubSurfaceOutParams subSurfaceOut;\n#ifdef SUBSURFACE\n#ifdef SS_THICKNESSANDMASK_TEXTURE\nvec4 thicknessMap=texture2D(thicknessSampler,vThicknessUV+uvOffset);\n#endif\n#ifdef SS_REFRACTIONINTENSITY_TEXTURE\nvec4 refractionIntensityMap=texture2D(refractionIntensitySampler,vRefractionIntensityUV+uvOffset);\n#endif\n#ifdef SS_TRANSLUCENCYINTENSITY_TEXTURE\nvec4 translucencyIntensityMap=texture2D(translucencyIntensitySampler,vTranslucencyIntensityUV+uvOffset);\n#endif\nsubSurfaceBlock(\nvSubSurfaceIntensity,\nvThicknessParam,\nvTintColor,\nnormalW,\nspecularEnvironmentReflectance,\n#ifdef SS_THICKNESSANDMASK_TEXTURE\nthicknessMap,\n#endif\n#ifdef SS_REFRACTIONINTENSITY_TEXTURE\nrefractionIntensityMap,\n#endif\n#ifdef SS_TRANSLUCENCYINTENSITY_TEXTURE\ntranslucencyIntensityMap,\n#endif\n#ifdef REFLECTION\n#ifdef SS_TRANSLUCENCY\nreflectionMatrix,\n#ifdef USESPHERICALFROMREFLECTIONMAP\n#if !defined(NORMAL) || !defined(USESPHERICALINVERTEX)\nreflectionOut.irradianceVector,\n#endif\n#if defined(REALTIME_FILTERING)\nreflectionSampler,\nvReflectionFilteringInfo,\n#endif\n#endif\n#ifdef USEIRRADIANCEMAP\nirradianceSampler,\n#endif\n#endif\n#endif\n#if defined(SS_REFRACTION) || defined(SS_TRANSLUCENCY)\nsurfaceAlbedo,\n#endif\n#ifdef SS_REFRACTION\nvPositionW,\nviewDirectionW,\nview,\nvRefractionInfos,\nrefractionMatrix,\nvRefractionMicrosurfaceInfos,\nvLightingIntensity,\n#ifdef SS_LINKREFRACTIONTOTRANSPARENCY\nalpha,\n#endif\n#ifdef SS_LODINREFRACTIONALPHA\nNdotVUnclamped,\n#endif\n#ifdef SS_LINEARSPECULARREFRACTION\nroughness,\n#endif\nalphaG,\nrefractionSampler,\n#ifndef LODBASEDMICROSFURACE\nrefractionSamplerLow,\nrefractionSamplerHigh,\n#endif\n#ifdef ANISOTROPIC\nanisotropicOut,\n#endif\n#ifdef REALTIME_FILTERING\nvRefractionFilteringInfo,\n#endif\n#ifdef SS_USE_LOCAL_REFRACTIONMAP_CUBIC\nvRefractionPosition,\nvRefractionSize,\n#endif\n#endif\n#ifdef SS_TRANSLUCENCY\nvDiffusionDistance,\n#endif\nsubSurfaceOut\n);\n#ifdef SS_REFRACTION\nsurfaceAlbedo=subSurfaceOut.surfaceAlbedo;\n#ifdef SS_LINKREFRACTIONTOTRANSPARENCY\nalpha=subSurfaceOut.alpha;\n#endif\n#endif\n#else\nsubSurfaceOut.specularEnvironmentReflectance=specularEnvironmentReflectance;\n#endif\n#include\n#include[0..maxSimultaneousLights]\n#include\n#endif \n#include\n#define CUSTOM_FRAGMENT_BEFORE_FINALCOLORCOMPOSITION\n#include\n#include\n#include(color,finalColor)\n#include\n#define CUSTOM_FRAGMENT_BEFORE_FRAGCOLOR\n#ifdef PREPASS\nfloat writeGeometryInfo=finalColor.a>0.4 ? 1.0 : 0.0;\n#ifdef PREPASS_POSITION\ngl_FragData[PREPASS_POSITION_INDEX]=vec4(vPositionW,writeGeometryInfo);\n#endif\n#ifdef PREPASS_VELOCITY\nvec2 a=(vCurrentPosition.xy/vCurrentPosition.w)*0.5+0.5;vec2 b=(vPreviousPosition.xy/vPreviousPosition.w)*0.5+0.5;vec2 velocity=abs(a-b);velocity=vec2(pow(velocity.x,1.0/3.0),pow(velocity.y,1.0/3.0))*sign(a-b)*0.5+0.5;gl_FragData[PREPASS_VELOCITY_INDEX]=vec4(velocity,0.0,writeGeometryInfo);\n#endif\n#ifdef PREPASS_ALBEDO_SQRT\nvec3 sqAlbedo=sqrt(surfaceAlbedo); \n#endif\n#ifdef PREPASS_IRRADIANCE\nvec3 irradiance=finalDiffuse;\n#ifndef UNLIT\n#ifdef REFLECTION\nirradiance+=finalIrradiance;\n#endif\n#endif\n#ifdef SS_SCATTERING\ngl_FragData[0]=vec4(finalColor.rgb-irradiance,finalColor.a); \nirradiance/=sqAlbedo;\n#else\ngl_FragData[0]=finalColor; \nfloat scatteringDiffusionProfile=255.;\n#endif\ngl_FragData[PREPASS_IRRADIANCE_INDEX]=vec4(clamp(irradiance,vec3(0.),vec3(1.)),writeGeometryInfo*scatteringDiffusionProfile/255.); \n#else\ngl_FragData[0]=vec4(finalColor.rgb,finalColor.a);\n#endif\n#ifdef PREPASS_DEPTH\ngl_FragData[PREPASS_DEPTH_INDEX]=vec4(vViewPos.z,0.0,0.0,writeGeometryInfo); \n#endif\n#ifdef PREPASS_NORMAL\ngl_FragData[PREPASS_NORMAL_INDEX]=vec4(normalize((view*vec4(normalW,0.0)).rgb),writeGeometryInfo); \n#endif\n#ifdef PREPASS_ALBEDO_SQRT\ngl_FragData[PREPASS_ALBEDO_SQRT_INDEX]=vec4(sqAlbedo,writeGeometryInfo); \n#endif\n#ifdef PREPASS_REFLECTIVITY\n#ifndef UNLIT\ngl_FragData[PREPASS_REFLECTIVITY_INDEX]=vec4(specularEnvironmentR0,microSurface)*writeGeometryInfo;\n#else\ngl_FragData[PREPASS_REFLECTIVITY_INDEX]=vec4( 0.0,0.0,0.0,1.0 )*writeGeometryInfo;\n#endif\n#endif\n#endif\n#if !defined(PREPASS) || defined(WEBGL2)\ngl_FragColor=finalColor;\n#endif\n#include\n#if ORDER_INDEPENDENT_TRANSPARENCY\nif (fragDepth==nearestDepth) {frontColor.rgb+=finalColor.rgb*finalColor.a*alphaMultiplier;frontColor.a=1.0-alphaMultiplier*(1.0-finalColor.a);} else {backColor+=finalColor;}\n#endif\n#include\n#define CUSTOM_FRAGMENT_MAIN_END\n}\n`;\n // Sideeffect\n ShaderStore.ShadersStore[name$1E] = shader$1E;\n\n // Do not edit.\n const name$1F = \"pbrVertexDeclaration\";\n const shader$1F = `uniform mat4 view;uniform mat4 viewProjection;\n#ifdef ALBEDO\nuniform mat4 albedoMatrix;uniform vec2 vAlbedoInfos;\n#endif\n#ifdef AMBIENT\nuniform mat4 ambientMatrix;uniform vec4 vAmbientInfos;\n#endif\n#ifdef OPACITY\nuniform mat4 opacityMatrix;uniform vec2 vOpacityInfos;\n#endif\n#ifdef EMISSIVE\nuniform vec2 vEmissiveInfos;uniform mat4 emissiveMatrix;\n#endif\n#ifdef LIGHTMAP\nuniform vec2 vLightmapInfos;uniform mat4 lightmapMatrix;\n#endif\n#ifdef REFLECTIVITY \nuniform vec3 vReflectivityInfos;uniform mat4 reflectivityMatrix;\n#endif\n#ifdef METALLIC_REFLECTANCE\nuniform vec2 vMetallicReflectanceInfos;uniform mat4 metallicReflectanceMatrix;\n#endif\n#ifdef REFLECTANCE\nuniform vec2 vReflectanceInfos;uniform mat4 reflectanceMatrix;\n#endif\n#ifdef MICROSURFACEMAP\nuniform vec2 vMicroSurfaceSamplerInfos;uniform mat4 microSurfaceSamplerMatrix;\n#endif\n#ifdef BUMP\nuniform vec3 vBumpInfos;uniform mat4 bumpMatrix;\n#endif\n#ifdef POINTSIZE\nuniform float pointSize;\n#endif\n#ifdef REFLECTION\nuniform vec2 vReflectionInfos;uniform mat4 reflectionMatrix;\n#endif\n#ifdef CLEARCOAT\n#if defined(CLEARCOAT_TEXTURE) || defined(CLEARCOAT_TEXTURE_ROUGHNESS)\nuniform vec4 vClearCoatInfos;\n#endif\n#ifdef CLEARCOAT_TEXTURE\nuniform mat4 clearCoatMatrix;\n#endif\n#ifdef CLEARCOAT_TEXTURE_ROUGHNESS\nuniform mat4 clearCoatRoughnessMatrix;\n#endif\n#ifdef CLEARCOAT_BUMP\nuniform vec2 vClearCoatBumpInfos;uniform mat4 clearCoatBumpMatrix;\n#endif\n#ifdef CLEARCOAT_TINT_TEXTURE\nuniform vec2 vClearCoatTintInfos;uniform mat4 clearCoatTintMatrix;\n#endif\n#endif\n#ifdef IRIDESCENCE\n#if defined(IRIDESCENCE_TEXTURE) || defined(IRIDESCENCE_THICKNESS_TEXTURE)\nuniform vec4 vIridescenceInfos;\n#endif\n#ifdef IRIDESCENCE_TEXTURE\nuniform mat4 iridescenceMatrix;\n#endif\n#ifdef IRIDESCENCE_THICKNESS_TEXTURE\nuniform mat4 iridescenceThicknessMatrix;\n#endif\n#endif\n#ifdef ANISOTROPIC\n#ifdef ANISOTROPIC_TEXTURE\nuniform vec2 vAnisotropyInfos;uniform mat4 anisotropyMatrix;\n#endif\n#endif\n#ifdef SHEEN\n#if defined(SHEEN_TEXTURE) || defined(SHEEN_TEXTURE_ROUGHNESS)\nuniform vec4 vSheenInfos;\n#endif\n#ifdef SHEEN_TEXTURE\nuniform mat4 sheenMatrix;\n#endif\n#ifdef SHEEN_TEXTURE_ROUGHNESS\nuniform mat4 sheenRoughnessMatrix;\n#endif\n#endif\n#ifdef SUBSURFACE\n#ifdef SS_REFRACTION\nuniform vec4 vRefractionInfos;uniform mat4 refractionMatrix;\n#endif\n#ifdef SS_THICKNESSANDMASK_TEXTURE\nuniform vec2 vThicknessInfos;uniform mat4 thicknessMatrix;\n#endif\n#ifdef SS_REFRACTIONINTENSITY_TEXTURE\nuniform vec2 vRefractionIntensityInfos;uniform mat4 refractionIntensityMatrix;\n#endif\n#ifdef SS_TRANSLUCENCYINTENSITY_TEXTURE\nuniform vec2 vTranslucencyIntensityInfos;uniform mat4 translucencyIntensityMatrix;\n#endif\n#endif\n#ifdef NORMAL\n#if defined(USESPHERICALFROMREFLECTIONMAP) && defined(USESPHERICALINVERTEX)\n#ifdef USESPHERICALFROMREFLECTIONMAP\n#ifdef SPHERICAL_HARMONICS\nuniform vec3 vSphericalL00;uniform vec3 vSphericalL1_1;uniform vec3 vSphericalL10;uniform vec3 vSphericalL11;uniform vec3 vSphericalL2_2;uniform vec3 vSphericalL2_1;uniform vec3 vSphericalL20;uniform vec3 vSphericalL21;uniform vec3 vSphericalL22;\n#else\nuniform vec3 vSphericalX;uniform vec3 vSphericalY;uniform vec3 vSphericalZ;uniform vec3 vSphericalXX_ZZ;uniform vec3 vSphericalYY_ZZ;uniform vec3 vSphericalZZ;uniform vec3 vSphericalXY;uniform vec3 vSphericalYZ;uniform vec3 vSphericalZX;\n#endif\n#endif\n#endif\n#endif\n#ifdef DETAIL\nuniform vec4 vDetailInfos;uniform mat4 detailMatrix;\n#endif\n#include\n#define ADDITIONAL_VERTEX_DECLARATION\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1F] = shader$1F;\n\n // Do not edit.\n const name$1G = \"pbrVertexShader\";\n const shader$1G = `precision highp float;\n#include<__decl__pbrVertex>\n#define CUSTOM_VERTEX_BEGIN\nattribute vec3 position;\n#ifdef NORMAL\nattribute vec3 normal;\n#endif\n#ifdef TANGENT\nattribute vec4 tangent;\n#endif\n#ifdef UV1\nattribute vec2 uv;\n#endif\n#include[2..7]\n#include[1..7]\n#ifdef VERTEXCOLOR\nattribute vec4 color;\n#endif\n#include\n#include\n#include\n#include\n#include\n#include(_DEFINENAME_,ALBEDO,_VARYINGNAME_,Albedo)\n#include(_DEFINENAME_,DETAIL,_VARYINGNAME_,Detail)\n#include(_DEFINENAME_,AMBIENT,_VARYINGNAME_,Ambient)\n#include(_DEFINENAME_,OPACITY,_VARYINGNAME_,Opacity)\n#include(_DEFINENAME_,EMISSIVE,_VARYINGNAME_,Emissive)\n#include(_DEFINENAME_,LIGHTMAP,_VARYINGNAME_,Lightmap)\n#include(_DEFINENAME_,REFLECTIVITY,_VARYINGNAME_,Reflectivity)\n#include(_DEFINENAME_,MICROSURFACEMAP,_VARYINGNAME_,MicroSurfaceSampler)\n#include(_DEFINENAME_,METALLIC_REFLECTANCE,_VARYINGNAME_,MetallicReflectance)\n#include(_DEFINENAME_,REFLECTANCE,_VARYINGNAME_,Reflectance)\n#include(_DEFINENAME_,BUMP,_VARYINGNAME_,Bump)\n#include(_DEFINENAME_,DECAL,_VARYINGNAME_,Decal)\n#ifdef CLEARCOAT\n#include(_DEFINENAME_,CLEARCOAT_TEXTURE,_VARYINGNAME_,ClearCoat)\n#include(_DEFINENAME_,CLEARCOAT_TEXTURE_ROUGHNESS,_VARYINGNAME_,ClearCoatRoughness)\n#include(_DEFINENAME_,CLEARCOAT_BUMP,_VARYINGNAME_,ClearCoatBump)\n#include(_DEFINENAME_,CLEARCOAT_TINT_TEXTURE,_VARYINGNAME_,ClearCoatTint)\n#endif\n#ifdef IRIDESCENCE\n#include(_DEFINENAME_,IRIDESCENCE_TEXTURE,_VARYINGNAME_,Iridescence)\n#include(_DEFINENAME_,IRIDESCENCE_THICKNESS_TEXTURE,_VARYINGNAME_,IridescenceThickness)\n#endif\n#ifdef SHEEN\n#include(_DEFINENAME_,SHEEN_TEXTURE,_VARYINGNAME_,Sheen)\n#include(_DEFINENAME_,SHEEN_TEXTURE_ROUGHNESS,_VARYINGNAME_,SheenRoughness)\n#endif\n#ifdef ANISOTROPIC\n#include(_DEFINENAME_,ANISOTROPIC_TEXTURE,_VARYINGNAME_,Anisotropy)\n#endif\n#ifdef SUBSURFACE\n#include(_DEFINENAME_,SS_THICKNESSANDMASK_TEXTURE,_VARYINGNAME_,Thickness)\n#include(_DEFINENAME_,SS_REFRACTIONINTENSITY_TEXTURE,_VARYINGNAME_,RefractionIntensity)\n#include(_DEFINENAME_,SS_TRANSLUCENCYINTENSITY_TEXTURE,_VARYINGNAME_,TranslucencyIntensity)\n#endif\nvarying vec3 vPositionW;\n#if DEBUGMODE>0\nvarying vec4 vClipSpacePosition;\n#endif\n#ifdef NORMAL\nvarying vec3 vNormalW;\n#if defined(USESPHERICALFROMREFLECTIONMAP) && defined(USESPHERICALINVERTEX)\nvarying vec3 vEnvironmentIrradiance;\n#include\n#endif\n#endif\n#if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES)\nvarying vec4 vColor;\n#endif\n#include\n#include\n#include\n#include<__decl__lightVxFragment>[0..maxSimultaneousLights]\n#include\n#include[0..maxSimultaneousMorphTargets]\n#ifdef REFLECTIONMAP_SKYBOX\nvarying vec3 vPositionUVW;\n#endif\n#if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED)\nvarying vec3 vDirectionW;\n#endif\n#include\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\nvec3 positionUpdated=position;\n#ifdef NORMAL\nvec3 normalUpdated=normal;\n#endif\n#ifdef TANGENT\nvec4 tangentUpdated=tangent;\n#endif\n#ifdef UV1\nvec2 uvUpdated=uv;\n#endif\n#include\n#include[0..maxSimultaneousMorphTargets]\n#ifdef REFLECTIONMAP_SKYBOX\nvPositionUVW=positionUpdated;\n#endif\n#define CUSTOM_VERTEX_UPDATE_POSITION\n#define CUSTOM_VERTEX_UPDATE_NORMAL\n#include\n#if defined(PREPASS) && defined(PREPASS_VELOCITY) && !defined(BONES_VELOCITY_ENABLED)\nvCurrentPosition=viewProjection*finalWorld*vec4(positionUpdated,1.0);vPreviousPosition=previousViewProjection*finalPreviousWorld*vec4(positionUpdated,1.0);\n#endif\n#include\n#include\nvec4 worldPos=finalWorld*vec4(positionUpdated,1.0);vPositionW=vec3(worldPos);\n#include\n#ifdef NORMAL\nmat3 normalWorld=mat3(finalWorld);\n#if defined(INSTANCES) && defined(THIN_INSTANCES)\nvNormalW=normalUpdated/vec3(dot(normalWorld[0],normalWorld[0]),dot(normalWorld[1],normalWorld[1]),dot(normalWorld[2],normalWorld[2]));vNormalW=normalize(normalWorld*vNormalW);\n#else\n#ifdef NONUNIFORMSCALING\nnormalWorld=transposeMat3(inverseMat3(normalWorld));\n#endif\nvNormalW=normalize(normalWorld*normalUpdated);\n#endif\n#if defined(USESPHERICALFROMREFLECTIONMAP) && defined(USESPHERICALINVERTEX)\nvec3 reflectionVector=vec3(reflectionMatrix*vec4(vNormalW,0)).xyz;\n#ifdef REFLECTIONMAP_OPPOSITEZ\nreflectionVector.z*=-1.0;\n#endif\nvEnvironmentIrradiance=computeEnvironmentIrradiance(reflectionVector);\n#endif\n#endif\n#define CUSTOM_VERTEX_UPDATE_WORLDPOS\n#ifdef MULTIVIEW\nif (gl_ViewID_OVR==0u) {gl_Position=viewProjection*worldPos;} else {gl_Position=viewProjectionR*worldPos;}\n#else\ngl_Position=viewProjection*worldPos;\n#endif\n#if DEBUGMODE>0\nvClipSpacePosition=gl_Position;\n#endif\n#if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED)\nvDirectionW=normalize(vec3(finalWorld*vec4(positionUpdated,0.0)));\n#endif\n#ifndef UV1\nvec2 uvUpdated=vec2(0.,0.);\n#endif\n#ifdef MAINUV1\nvMainUV1=uvUpdated;\n#endif\n#include[2..7]\n#include(_DEFINENAME_,ALBEDO,_VARYINGNAME_,Albedo,_MATRIXNAME_,albedo,_INFONAME_,AlbedoInfos.x)\n#include(_DEFINENAME_,DETAIL,_VARYINGNAME_,Detail,_MATRIXNAME_,detail,_INFONAME_,DetailInfos.x)\n#include(_DEFINENAME_,AMBIENT,_VARYINGNAME_,Ambient,_MATRIXNAME_,ambient,_INFONAME_,AmbientInfos.x)\n#include(_DEFINENAME_,OPACITY,_VARYINGNAME_,Opacity,_MATRIXNAME_,opacity,_INFONAME_,OpacityInfos.x)\n#include(_DEFINENAME_,EMISSIVE,_VARYINGNAME_,Emissive,_MATRIXNAME_,emissive,_INFONAME_,EmissiveInfos.x)\n#include(_DEFINENAME_,LIGHTMAP,_VARYINGNAME_,Lightmap,_MATRIXNAME_,lightmap,_INFONAME_,LightmapInfos.x)\n#include(_DEFINENAME_,REFLECTIVITY,_VARYINGNAME_,Reflectivity,_MATRIXNAME_,reflectivity,_INFONAME_,ReflectivityInfos.x)\n#include(_DEFINENAME_,MICROSURFACEMAP,_VARYINGNAME_,MicroSurfaceSampler,_MATRIXNAME_,microSurfaceSampler,_INFONAME_,MicroSurfaceSamplerInfos.x)\n#include(_DEFINENAME_,METALLIC_REFLECTANCE,_VARYINGNAME_,MetallicReflectance,_MATRIXNAME_,metallicReflectance,_INFONAME_,MetallicReflectanceInfos.x)\n#include(_DEFINENAME_,REFLECTANCE,_VARYINGNAME_,Reflectance,_MATRIXNAME_,reflectance,_INFONAME_,ReflectanceInfos.x)\n#include(_DEFINENAME_,BUMP,_VARYINGNAME_,Bump,_MATRIXNAME_,bump,_INFONAME_,BumpInfos.x)\n#include(_DEFINENAME_,DECAL,_VARYINGNAME_,Decal,_MATRIXNAME_,decal,_INFONAME_,DecalInfos.x)\n#ifdef CLEARCOAT\n#include(_DEFINENAME_,CLEARCOAT_TEXTURE,_VARYINGNAME_,ClearCoat,_MATRIXNAME_,clearCoat,_INFONAME_,ClearCoatInfos.x)\n#include(_DEFINENAME_,CLEARCOAT_TEXTURE_ROUGHNESS,_VARYINGNAME_,ClearCoatRoughness,_MATRIXNAME_,clearCoatRoughness,_INFONAME_,ClearCoatInfos.z)\n#include(_DEFINENAME_,CLEARCOAT_BUMP,_VARYINGNAME_,ClearCoatBump,_MATRIXNAME_,clearCoatBump,_INFONAME_,ClearCoatBumpInfos.x)\n#include(_DEFINENAME_,CLEARCOAT_TINT_TEXTURE,_VARYINGNAME_,ClearCoatTint,_MATRIXNAME_,clearCoatTint,_INFONAME_,ClearCoatTintInfos.x)\n#endif\n#ifdef IRIDESCENCE\n#include(_DEFINENAME_,IRIDESCENCE_TEXTURE,_VARYINGNAME_,Iridescence,_MATRIXNAME_,iridescence,_INFONAME_,IridescenceInfos.x)\n#include(_DEFINENAME_,IRIDESCENCE_THICKNESS_TEXTURE,_VARYINGNAME_,IridescenceThickness,_MATRIXNAME_,iridescenceThickness,_INFONAME_,IridescenceInfos.z)\n#endif\n#ifdef SHEEN\n#include(_DEFINENAME_,SHEEN_TEXTURE,_VARYINGNAME_,Sheen,_MATRIXNAME_,sheen,_INFONAME_,SheenInfos.x)\n#include(_DEFINENAME_,SHEEN_TEXTURE_ROUGHNESS,_VARYINGNAME_,SheenRoughness,_MATRIXNAME_,sheen,_INFONAME_,SheenInfos.z)\n#endif\n#ifdef ANISOTROPIC\n#include(_DEFINENAME_,ANISOTROPIC_TEXTURE,_VARYINGNAME_,Anisotropy,_MATRIXNAME_,anisotropy,_INFONAME_,AnisotropyInfos.x)\n#endif\n#ifdef SUBSURFACE\n#include(_DEFINENAME_,SS_THICKNESSANDMASK_TEXTURE,_VARYINGNAME_,Thickness,_MATRIXNAME_,thickness,_INFONAME_,ThicknessInfos.x)\n#include(_DEFINENAME_,SS_REFRACTIONINTENSITY_TEXTURE,_VARYINGNAME_,RefractionIntensity,_MATRIXNAME_,refractionIntensity,_INFONAME_,RefractionIntensityInfos.x)\n#include(_DEFINENAME_,SS_TRANSLUCENCYINTENSITY_TEXTURE,_VARYINGNAME_,TranslucencyIntensity,_MATRIXNAME_,translucencyIntensity,_INFONAME_,TranslucencyIntensityInfos.x)\n#endif\n#include\n#include\n#include\n#include[0..maxSimultaneousLights]\n#include\n#if defined(POINTSIZE) && !defined(WEBGPU)\ngl_PointSize=pointSize;\n#endif\n#include\n#define CUSTOM_VERTEX_MAIN_END\n}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$1G] = shader$1G;\n\n /**\n * @internal\n */\n class MaterialClearCoatDefines extends MaterialDefines {\n constructor() {\n super(...arguments);\n this.CLEARCOAT = false;\n this.CLEARCOAT_DEFAULTIOR = false;\n this.CLEARCOAT_TEXTURE = false;\n this.CLEARCOAT_TEXTURE_ROUGHNESS = false;\n this.CLEARCOAT_TEXTUREDIRECTUV = 0;\n this.CLEARCOAT_TEXTURE_ROUGHNESSDIRECTUV = 0;\n this.CLEARCOAT_BUMP = false;\n this.CLEARCOAT_BUMPDIRECTUV = 0;\n this.CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE = false;\n this.CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL = false;\n this.CLEARCOAT_REMAP_F0 = false;\n this.CLEARCOAT_TINT = false;\n this.CLEARCOAT_TINT_TEXTURE = false;\n this.CLEARCOAT_TINT_TEXTUREDIRECTUV = 0;\n this.CLEARCOAT_TINT_GAMMATEXTURE = false;\n }\n }\n /**\n * Plugin that implements the clear coat component of the PBR material\n */\n class PBRClearCoatConfiguration extends MaterialPluginBase {\n /** @internal */\n _markAllSubMeshesAsTexturesDirty() {\n this._enable(this._isEnabled);\n this._internalMarkAllSubMeshesAsTexturesDirty();\n }\n constructor(material, addToPluginList = true) {\n super(material, \"PBRClearCoat\", 100, new MaterialClearCoatDefines(), addToPluginList);\n this._isEnabled = false;\n /**\n * Defines if the clear coat is enabled in the material.\n */\n this.isEnabled = false;\n /**\n * Defines the clear coat layer strength (between 0 and 1) it defaults to 1.\n */\n this.intensity = 1;\n /**\n * Defines the clear coat layer roughness.\n */\n this.roughness = 0;\n this._indexOfRefraction = PBRClearCoatConfiguration._DefaultIndexOfRefraction;\n /**\n * Defines the index of refraction of the clear coat.\n * This defaults to 1.5 corresponding to a 0.04 f0 or a 4% reflectance at normal incidence\n * The default fits with a polyurethane material.\n * Changing the default value is more performance intensive.\n */\n this.indexOfRefraction = PBRClearCoatConfiguration._DefaultIndexOfRefraction;\n this._texture = null;\n /**\n * Stores the clear coat values in a texture (red channel is intensity and green channel is roughness)\n * If useRoughnessFromMainTexture is false, the green channel of texture is not used and the green channel of textureRoughness is used instead\n * if textureRoughness is not empty, else no texture roughness is used\n */\n this.texture = null;\n this._useRoughnessFromMainTexture = true;\n /**\n * Indicates that the green channel of the texture property will be used for roughness (default: true)\n * If false, the green channel from textureRoughness is used for roughness\n */\n this.useRoughnessFromMainTexture = true;\n this._textureRoughness = null;\n /**\n * Stores the clear coat roughness in a texture (green channel)\n * Not used if useRoughnessFromMainTexture is true\n */\n this.textureRoughness = null;\n this._remapF0OnInterfaceChange = true;\n /**\n * Defines if the F0 value should be remapped to account for the interface change in the material.\n */\n this.remapF0OnInterfaceChange = true;\n this._bumpTexture = null;\n /**\n * Define the clear coat specific bump texture.\n */\n this.bumpTexture = null;\n this._isTintEnabled = false;\n /**\n * Defines if the clear coat tint is enabled in the material.\n */\n this.isTintEnabled = false;\n /**\n * Defines the clear coat tint of the material.\n * This is only use if tint is enabled\n */\n this.tintColor = Color3.White();\n /**\n * Defines the distance at which the tint color should be found in the\n * clear coat media.\n * This is only use if tint is enabled\n */\n this.tintColorAtDistance = 1;\n /**\n * Defines the clear coat layer thickness.\n * This is only use if tint is enabled\n */\n this.tintThickness = 1;\n this._tintTexture = null;\n /**\n * Stores the clear tint values in a texture.\n * rgb is tint\n * a is a thickness factor\n */\n this.tintTexture = null;\n this._internalMarkAllSubMeshesAsTexturesDirty = material._dirtyCallbacks[1];\n }\n isReadyForSubMesh(defines, scene, engine) {\n if (!this._isEnabled) {\n return true;\n }\n const disableBumpMap = this._material._disableBumpMap;\n if (defines._areTexturesDirty) {\n if (scene.texturesEnabled) {\n if (this._texture && MaterialFlags.ClearCoatTextureEnabled) {\n if (!this._texture.isReadyOrNotBlocking()) {\n return false;\n }\n }\n if (this._textureRoughness && MaterialFlags.ClearCoatTextureEnabled) {\n if (!this._textureRoughness.isReadyOrNotBlocking()) {\n return false;\n }\n }\n if (engine.getCaps().standardDerivatives && this._bumpTexture && MaterialFlags.ClearCoatBumpTextureEnabled && !disableBumpMap) {\n // Bump texture cannot be not blocking.\n if (!this._bumpTexture.isReady()) {\n return false;\n }\n }\n if (this._isTintEnabled && this._tintTexture && MaterialFlags.ClearCoatTintTextureEnabled) {\n if (!this._tintTexture.isReadyOrNotBlocking()) {\n return false;\n }\n }\n }\n }\n return true;\n }\n prepareDefinesBeforeAttributes(defines, scene) {\n var _a;\n if (this._isEnabled) {\n defines.CLEARCOAT = true;\n defines.CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE = this._useRoughnessFromMainTexture;\n defines.CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL =\n this._texture !== null && this._texture._texture === ((_a = this._textureRoughness) === null || _a === void 0 ? void 0 : _a._texture) && this._texture.checkTransformsAreIdentical(this._textureRoughness);\n defines.CLEARCOAT_REMAP_F0 = this._remapF0OnInterfaceChange;\n if (defines._areTexturesDirty) {\n if (scene.texturesEnabled) {\n if (this._texture && MaterialFlags.ClearCoatTextureEnabled) {\n MaterialHelper.PrepareDefinesForMergedUV(this._texture, defines, \"CLEARCOAT_TEXTURE\");\n }\n else {\n defines.CLEARCOAT_TEXTURE = false;\n }\n if (this._textureRoughness && MaterialFlags.ClearCoatTextureEnabled) {\n MaterialHelper.PrepareDefinesForMergedUV(this._textureRoughness, defines, \"CLEARCOAT_TEXTURE_ROUGHNESS\");\n }\n else {\n defines.CLEARCOAT_TEXTURE_ROUGHNESS = false;\n }\n if (this._bumpTexture && MaterialFlags.ClearCoatBumpTextureEnabled) {\n MaterialHelper.PrepareDefinesForMergedUV(this._bumpTexture, defines, \"CLEARCOAT_BUMP\");\n }\n else {\n defines.CLEARCOAT_BUMP = false;\n }\n defines.CLEARCOAT_DEFAULTIOR = this._indexOfRefraction === PBRClearCoatConfiguration._DefaultIndexOfRefraction;\n if (this._isTintEnabled) {\n defines.CLEARCOAT_TINT = true;\n if (this._tintTexture && MaterialFlags.ClearCoatTintTextureEnabled) {\n MaterialHelper.PrepareDefinesForMergedUV(this._tintTexture, defines, \"CLEARCOAT_TINT_TEXTURE\");\n defines.CLEARCOAT_TINT_GAMMATEXTURE = this._tintTexture.gammaSpace;\n }\n else {\n defines.CLEARCOAT_TINT_TEXTURE = false;\n }\n }\n else {\n defines.CLEARCOAT_TINT = false;\n defines.CLEARCOAT_TINT_TEXTURE = false;\n }\n }\n }\n }\n else {\n defines.CLEARCOAT = false;\n defines.CLEARCOAT_TEXTURE = false;\n defines.CLEARCOAT_TEXTURE_ROUGHNESS = false;\n defines.CLEARCOAT_BUMP = false;\n defines.CLEARCOAT_TINT = false;\n defines.CLEARCOAT_TINT_TEXTURE = false;\n defines.CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE = false;\n defines.CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL = false;\n defines.CLEARCOAT_DEFAULTIOR = false;\n defines.CLEARCOAT_TEXTUREDIRECTUV = 0;\n defines.CLEARCOAT_TEXTURE_ROUGHNESSDIRECTUV = 0;\n defines.CLEARCOAT_BUMPDIRECTUV = 0;\n defines.CLEARCOAT_REMAP_F0 = false;\n defines.CLEARCOAT_TINT_TEXTUREDIRECTUV = 0;\n defines.CLEARCOAT_TINT_GAMMATEXTURE = false;\n }\n }\n bindForSubMesh(uniformBuffer, scene, engine, subMesh) {\n var _a, _b, _c, _d, _e, _f, _g, _h;\n if (!this._isEnabled) {\n return;\n }\n const defines = subMesh.materialDefines;\n const isFrozen = this._material.isFrozen;\n const disableBumpMap = this._material._disableBumpMap;\n const invertNormalMapX = this._material._invertNormalMapX;\n const invertNormalMapY = this._material._invertNormalMapY;\n const identicalTextures = defines.CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL;\n if (!uniformBuffer.useUbo || !isFrozen || !uniformBuffer.isSync) {\n if (identicalTextures && MaterialFlags.ClearCoatTextureEnabled) {\n uniformBuffer.updateFloat4(\"vClearCoatInfos\", this._texture.coordinatesIndex, this._texture.level, -1, -1);\n MaterialHelper.BindTextureMatrix(this._texture, uniformBuffer, \"clearCoat\");\n }\n else if ((this._texture || this._textureRoughness) && MaterialFlags.ClearCoatTextureEnabled) {\n uniformBuffer.updateFloat4(\"vClearCoatInfos\", (_b = (_a = this._texture) === null || _a === void 0 ? void 0 : _a.coordinatesIndex) !== null && _b !== void 0 ? _b : 0, (_d = (_c = this._texture) === null || _c === void 0 ? void 0 : _c.level) !== null && _d !== void 0 ? _d : 0, (_f = (_e = this._textureRoughness) === null || _e === void 0 ? void 0 : _e.coordinatesIndex) !== null && _f !== void 0 ? _f : 0, (_h = (_g = this._textureRoughness) === null || _g === void 0 ? void 0 : _g.level) !== null && _h !== void 0 ? _h : 0);\n if (this._texture) {\n MaterialHelper.BindTextureMatrix(this._texture, uniformBuffer, \"clearCoat\");\n }\n if (this._textureRoughness && !identicalTextures && !defines.CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE) {\n MaterialHelper.BindTextureMatrix(this._textureRoughness, uniformBuffer, \"clearCoatRoughness\");\n }\n }\n if (this._bumpTexture && engine.getCaps().standardDerivatives && MaterialFlags.ClearCoatTextureEnabled && !disableBumpMap) {\n uniformBuffer.updateFloat2(\"vClearCoatBumpInfos\", this._bumpTexture.coordinatesIndex, this._bumpTexture.level);\n MaterialHelper.BindTextureMatrix(this._bumpTexture, uniformBuffer, \"clearCoatBump\");\n if (scene._mirroredCameraPosition) {\n uniformBuffer.updateFloat2(\"vClearCoatTangentSpaceParams\", invertNormalMapX ? 1.0 : -1.0, invertNormalMapY ? 1.0 : -1.0);\n }\n else {\n uniformBuffer.updateFloat2(\"vClearCoatTangentSpaceParams\", invertNormalMapX ? -1.0 : 1.0, invertNormalMapY ? -1.0 : 1.0);\n }\n }\n if (this._tintTexture && MaterialFlags.ClearCoatTintTextureEnabled) {\n uniformBuffer.updateFloat2(\"vClearCoatTintInfos\", this._tintTexture.coordinatesIndex, this._tintTexture.level);\n MaterialHelper.BindTextureMatrix(this._tintTexture, uniformBuffer, \"clearCoatTint\");\n }\n // Clear Coat General params\n uniformBuffer.updateFloat2(\"vClearCoatParams\", this.intensity, this.roughness);\n // Clear Coat Refraction params\n const a = 1 - this._indexOfRefraction;\n const b = 1 + this._indexOfRefraction;\n const f0 = Math.pow(-a / b, 2); // Schlicks approx: (ior1 - ior2) / (ior1 + ior2) where ior2 for air is close to vacuum = 1.\n const eta = 1 / this._indexOfRefraction;\n uniformBuffer.updateFloat4(\"vClearCoatRefractionParams\", f0, eta, a, b);\n if (this._isTintEnabled) {\n uniformBuffer.updateFloat4(\"vClearCoatTintParams\", this.tintColor.r, this.tintColor.g, this.tintColor.b, Math.max(0.00001, this.tintThickness));\n uniformBuffer.updateFloat(\"clearCoatColorAtDistance\", Math.max(0.00001, this.tintColorAtDistance));\n }\n }\n // Textures\n if (scene.texturesEnabled) {\n if (this._texture && MaterialFlags.ClearCoatTextureEnabled) {\n uniformBuffer.setTexture(\"clearCoatSampler\", this._texture);\n }\n if (this._textureRoughness && !identicalTextures && !defines.CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE && MaterialFlags.ClearCoatTextureEnabled) {\n uniformBuffer.setTexture(\"clearCoatRoughnessSampler\", this._textureRoughness);\n }\n if (this._bumpTexture && engine.getCaps().standardDerivatives && MaterialFlags.ClearCoatBumpTextureEnabled && !disableBumpMap) {\n uniformBuffer.setTexture(\"clearCoatBumpSampler\", this._bumpTexture);\n }\n if (this._isTintEnabled && this._tintTexture && MaterialFlags.ClearCoatTintTextureEnabled) {\n uniformBuffer.setTexture(\"clearCoatTintSampler\", this._tintTexture);\n }\n }\n }\n hasTexture(texture) {\n if (this._texture === texture) {\n return true;\n }\n if (this._textureRoughness === texture) {\n return true;\n }\n if (this._bumpTexture === texture) {\n return true;\n }\n if (this._tintTexture === texture) {\n return true;\n }\n return false;\n }\n getActiveTextures(activeTextures) {\n if (this._texture) {\n activeTextures.push(this._texture);\n }\n if (this._textureRoughness) {\n activeTextures.push(this._textureRoughness);\n }\n if (this._bumpTexture) {\n activeTextures.push(this._bumpTexture);\n }\n if (this._tintTexture) {\n activeTextures.push(this._tintTexture);\n }\n }\n getAnimatables(animatables) {\n if (this._texture && this._texture.animations && this._texture.animations.length > 0) {\n animatables.push(this._texture);\n }\n if (this._textureRoughness && this._textureRoughness.animations && this._textureRoughness.animations.length > 0) {\n animatables.push(this._textureRoughness);\n }\n if (this._bumpTexture && this._bumpTexture.animations && this._bumpTexture.animations.length > 0) {\n animatables.push(this._bumpTexture);\n }\n if (this._tintTexture && this._tintTexture.animations && this._tintTexture.animations.length > 0) {\n animatables.push(this._tintTexture);\n }\n }\n dispose(forceDisposeTextures) {\n var _a, _b, _c, _d;\n if (forceDisposeTextures) {\n (_a = this._texture) === null || _a === void 0 ? void 0 : _a.dispose();\n (_b = this._textureRoughness) === null || _b === void 0 ? void 0 : _b.dispose();\n (_c = this._bumpTexture) === null || _c === void 0 ? void 0 : _c.dispose();\n (_d = this._tintTexture) === null || _d === void 0 ? void 0 : _d.dispose();\n }\n }\n getClassName() {\n return \"PBRClearCoatConfiguration\";\n }\n addFallbacks(defines, fallbacks, currentRank) {\n if (defines.CLEARCOAT_BUMP) {\n fallbacks.addFallback(currentRank++, \"CLEARCOAT_BUMP\");\n }\n if (defines.CLEARCOAT_TINT) {\n fallbacks.addFallback(currentRank++, \"CLEARCOAT_TINT\");\n }\n if (defines.CLEARCOAT) {\n fallbacks.addFallback(currentRank++, \"CLEARCOAT\");\n }\n return currentRank;\n }\n getSamplers(samplers) {\n samplers.push(\"clearCoatSampler\", \"clearCoatRoughnessSampler\", \"clearCoatBumpSampler\", \"clearCoatTintSampler\");\n }\n getUniforms() {\n return {\n ubo: [\n { name: \"vClearCoatParams\", size: 2, type: \"vec2\" },\n { name: \"vClearCoatRefractionParams\", size: 4, type: \"vec4\" },\n { name: \"vClearCoatInfos\", size: 4, type: \"vec4\" },\n { name: \"clearCoatMatrix\", size: 16, type: \"mat4\" },\n { name: \"clearCoatRoughnessMatrix\", size: 16, type: \"mat4\" },\n { name: \"vClearCoatBumpInfos\", size: 2, type: \"vec2\" },\n { name: \"vClearCoatTangentSpaceParams\", size: 2, type: \"vec2\" },\n { name: \"clearCoatBumpMatrix\", size: 16, type: \"mat4\" },\n { name: \"vClearCoatTintParams\", size: 4, type: \"vec4\" },\n { name: \"clearCoatColorAtDistance\", size: 1, type: \"float\" },\n { name: \"vClearCoatTintInfos\", size: 2, type: \"vec2\" },\n { name: \"clearCoatTintMatrix\", size: 16, type: \"mat4\" },\n ],\n };\n }\n }\n /**\n * This defaults to 1.5 corresponding to a 0.04 f0 or a 4% reflectance at normal incidence\n * The default fits with a polyurethane material.\n * @internal\n */\n PBRClearCoatConfiguration._DefaultIndexOfRefraction = 1.5;\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRClearCoatConfiguration.prototype, \"isEnabled\", void 0);\n __decorate$1([\n serialize()\n ], PBRClearCoatConfiguration.prototype, \"intensity\", void 0);\n __decorate$1([\n serialize()\n ], PBRClearCoatConfiguration.prototype, \"roughness\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRClearCoatConfiguration.prototype, \"indexOfRefraction\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRClearCoatConfiguration.prototype, \"texture\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRClearCoatConfiguration.prototype, \"useRoughnessFromMainTexture\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRClearCoatConfiguration.prototype, \"textureRoughness\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRClearCoatConfiguration.prototype, \"remapF0OnInterfaceChange\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRClearCoatConfiguration.prototype, \"bumpTexture\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRClearCoatConfiguration.prototype, \"isTintEnabled\", void 0);\n __decorate$1([\n serializeAsColor3()\n ], PBRClearCoatConfiguration.prototype, \"tintColor\", void 0);\n __decorate$1([\n serialize()\n ], PBRClearCoatConfiguration.prototype, \"tintColorAtDistance\", void 0);\n __decorate$1([\n serialize()\n ], PBRClearCoatConfiguration.prototype, \"tintThickness\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRClearCoatConfiguration.prototype, \"tintTexture\", void 0);\n\n /**\n * @internal\n */\n class MaterialIridescenceDefines extends MaterialDefines {\n constructor() {\n super(...arguments);\n this.IRIDESCENCE = false;\n this.IRIDESCENCE_TEXTURE = false;\n this.IRIDESCENCE_TEXTUREDIRECTUV = 0;\n this.IRIDESCENCE_THICKNESS_TEXTURE = false;\n this.IRIDESCENCE_THICKNESS_TEXTUREDIRECTUV = 0;\n this.IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE = false;\n }\n }\n /**\n * Plugin that implements the iridescence (thin film) component of the PBR material\n */\n class PBRIridescenceConfiguration extends MaterialPluginBase {\n /** @internal */\n _markAllSubMeshesAsTexturesDirty() {\n this._enable(this._isEnabled);\n this._internalMarkAllSubMeshesAsTexturesDirty();\n }\n constructor(material, addToPluginList = true) {\n super(material, \"PBRIridescence\", 110, new MaterialIridescenceDefines(), addToPluginList);\n this._isEnabled = false;\n /**\n * Defines if the iridescence is enabled in the material.\n */\n this.isEnabled = false;\n /**\n * Defines the iridescence layer strength (between 0 and 1) it defaults to 1.\n */\n this.intensity = 1;\n /**\n * Defines the minimum thickness of the thin-film layer given in nanometers (nm).\n */\n this.minimumThickness = PBRIridescenceConfiguration._DefaultMinimumThickness;\n /**\n * Defines the maximum thickness of the thin-film layer given in nanometers (nm). This will be the thickness used if not thickness texture has been set.\n */\n this.maximumThickness = PBRIridescenceConfiguration._DefaultMaximumThickness;\n /**\n * Defines the maximum thickness of the thin-film layer given in nanometers (nm).\n */\n this.indexOfRefraction = PBRIridescenceConfiguration._DefaultIndexOfRefraction;\n this._texture = null;\n /**\n * Stores the iridescence intensity in a texture (red channel)\n */\n this.texture = null;\n this._thicknessTexture = null;\n /**\n * Stores the iridescence thickness in a texture (green channel)\n */\n this.thicknessTexture = null;\n this._internalMarkAllSubMeshesAsTexturesDirty = material._dirtyCallbacks[1];\n }\n isReadyForSubMesh(defines, scene) {\n if (!this._isEnabled) {\n return true;\n }\n if (defines._areTexturesDirty) {\n if (scene.texturesEnabled) {\n if (this._texture && MaterialFlags.IridescenceTextureEnabled) {\n if (!this._texture.isReadyOrNotBlocking()) {\n return false;\n }\n }\n if (this._thicknessTexture && MaterialFlags.IridescenceTextureEnabled) {\n if (!this._thicknessTexture.isReadyOrNotBlocking()) {\n return false;\n }\n }\n }\n }\n return true;\n }\n prepareDefinesBeforeAttributes(defines, scene) {\n var _a;\n if (this._isEnabled) {\n defines.IRIDESCENCE = true;\n defines.IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE =\n this._texture !== null && this._texture._texture === ((_a = this._thicknessTexture) === null || _a === void 0 ? void 0 : _a._texture) && this._texture.checkTransformsAreIdentical(this._thicknessTexture);\n if (defines._areTexturesDirty) {\n if (scene.texturesEnabled) {\n if (this._texture && MaterialFlags.IridescenceTextureEnabled) {\n MaterialHelper.PrepareDefinesForMergedUV(this._texture, defines, \"IRIDESCENCE_TEXTURE\");\n }\n else {\n defines.IRIDESCENCE_TEXTURE = false;\n }\n if (!defines.IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE && this._thicknessTexture && MaterialFlags.IridescenceTextureEnabled) {\n MaterialHelper.PrepareDefinesForMergedUV(this._thicknessTexture, defines, \"IRIDESCENCE_THICKNESS_TEXTURE\");\n }\n else {\n defines.IRIDESCENCE_THICKNESS_TEXTURE = false;\n }\n }\n }\n }\n else {\n defines.IRIDESCENCE = false;\n defines.IRIDESCENCE_TEXTURE = false;\n defines.IRIDESCENCE_THICKNESS_TEXTURE = false;\n defines.IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE = false;\n defines.IRIDESCENCE_TEXTUREDIRECTUV = 0;\n defines.IRIDESCENCE_THICKNESS_TEXTUREDIRECTUV = 0;\n }\n }\n bindForSubMesh(uniformBuffer, scene, engine, subMesh) {\n var _a, _b, _c, _d, _e, _f, _g, _h;\n if (!this._isEnabled) {\n return;\n }\n const defines = subMesh.materialDefines;\n const isFrozen = this._material.isFrozen;\n const identicalTextures = defines.IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE;\n if (!uniformBuffer.useUbo || !isFrozen || !uniformBuffer.isSync) {\n if (identicalTextures && MaterialFlags.IridescenceTextureEnabled) {\n uniformBuffer.updateFloat4(\"vIridescenceInfos\", this._texture.coordinatesIndex, this._texture.level, -1, -1);\n MaterialHelper.BindTextureMatrix(this._texture, uniformBuffer, \"iridescence\");\n }\n else if ((this._texture || this._thicknessTexture) && MaterialFlags.IridescenceTextureEnabled) {\n uniformBuffer.updateFloat4(\"vIridescenceInfos\", (_b = (_a = this._texture) === null || _a === void 0 ? void 0 : _a.coordinatesIndex) !== null && _b !== void 0 ? _b : 0, (_d = (_c = this._texture) === null || _c === void 0 ? void 0 : _c.level) !== null && _d !== void 0 ? _d : 0, (_f = (_e = this._thicknessTexture) === null || _e === void 0 ? void 0 : _e.coordinatesIndex) !== null && _f !== void 0 ? _f : 0, (_h = (_g = this._thicknessTexture) === null || _g === void 0 ? void 0 : _g.level) !== null && _h !== void 0 ? _h : 0);\n if (this._texture) {\n MaterialHelper.BindTextureMatrix(this._texture, uniformBuffer, \"iridescence\");\n }\n if (this._thicknessTexture && !identicalTextures && !defines.IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE) {\n MaterialHelper.BindTextureMatrix(this._thicknessTexture, uniformBuffer, \"iridescenceThickness\");\n }\n }\n // Clear Coat General params\n uniformBuffer.updateFloat4(\"vIridescenceParams\", this.intensity, this.indexOfRefraction, this.minimumThickness, this.maximumThickness);\n }\n // Textures\n if (scene.texturesEnabled) {\n if (this._texture && MaterialFlags.IridescenceTextureEnabled) {\n uniformBuffer.setTexture(\"iridescenceSampler\", this._texture);\n }\n if (this._thicknessTexture && !identicalTextures && !defines.IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE && MaterialFlags.IridescenceTextureEnabled) {\n uniformBuffer.setTexture(\"iridescenceThicknessSampler\", this._thicknessTexture);\n }\n }\n }\n hasTexture(texture) {\n if (this._texture === texture) {\n return true;\n }\n if (this._thicknessTexture === texture) {\n return true;\n }\n return false;\n }\n getActiveTextures(activeTextures) {\n if (this._texture) {\n activeTextures.push(this._texture);\n }\n if (this._thicknessTexture) {\n activeTextures.push(this._thicknessTexture);\n }\n }\n getAnimatables(animatables) {\n if (this._texture && this._texture.animations && this._texture.animations.length > 0) {\n animatables.push(this._texture);\n }\n if (this._thicknessTexture && this._thicknessTexture.animations && this._thicknessTexture.animations.length > 0) {\n animatables.push(this._thicknessTexture);\n }\n }\n dispose(forceDisposeTextures) {\n var _a, _b;\n if (forceDisposeTextures) {\n (_a = this._texture) === null || _a === void 0 ? void 0 : _a.dispose();\n (_b = this._thicknessTexture) === null || _b === void 0 ? void 0 : _b.dispose();\n }\n }\n getClassName() {\n return \"PBRIridescenceConfiguration\";\n }\n addFallbacks(defines, fallbacks, currentRank) {\n if (defines.IRIDESCENCE) {\n fallbacks.addFallback(currentRank++, \"IRIDESCENCE\");\n }\n return currentRank;\n }\n getSamplers(samplers) {\n samplers.push(\"iridescenceSampler\", \"iridescenceThicknessSampler\");\n }\n getUniforms() {\n return {\n ubo: [\n { name: \"vIridescenceParams\", size: 4, type: \"vec4\" },\n { name: \"vIridescenceInfos\", size: 4, type: \"vec4\" },\n { name: \"iridescenceMatrix\", size: 16, type: \"mat4\" },\n { name: \"iridescenceThicknessMatrix\", size: 16, type: \"mat4\" },\n ],\n };\n }\n }\n /**\n * The default minimum thickness of the thin-film layer given in nanometers (nm).\n * Defaults to 100 nm.\n * @internal\n */\n PBRIridescenceConfiguration._DefaultMinimumThickness = 100;\n /**\n * The default maximum thickness of the thin-film layer given in nanometers (nm).\n * Defaults to 400 nm.\n * @internal\n */\n PBRIridescenceConfiguration._DefaultMaximumThickness = 400;\n /**\n * The default index of refraction of the thin-film layer.\n * Defaults to 1.3\n * @internal\n */\n PBRIridescenceConfiguration._DefaultIndexOfRefraction = 1.3;\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRIridescenceConfiguration.prototype, \"isEnabled\", void 0);\n __decorate$1([\n serialize()\n ], PBRIridescenceConfiguration.prototype, \"intensity\", void 0);\n __decorate$1([\n serialize()\n ], PBRIridescenceConfiguration.prototype, \"minimumThickness\", void 0);\n __decorate$1([\n serialize()\n ], PBRIridescenceConfiguration.prototype, \"maximumThickness\", void 0);\n __decorate$1([\n serialize()\n ], PBRIridescenceConfiguration.prototype, \"indexOfRefraction\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRIridescenceConfiguration.prototype, \"texture\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRIridescenceConfiguration.prototype, \"thicknessTexture\", void 0);\n\n /**\n * @internal\n */\n class MaterialAnisotropicDefines extends MaterialDefines {\n constructor() {\n super(...arguments);\n this.ANISOTROPIC = false;\n this.ANISOTROPIC_TEXTURE = false;\n this.ANISOTROPIC_TEXTUREDIRECTUV = 0;\n this.ANISOTROPIC_LEGACY = false;\n this.MAINUV1 = false;\n }\n }\n /**\n * Plugin that implements the anisotropic component of the PBR material\n */\n class PBRAnisotropicConfiguration extends MaterialPluginBase {\n /**\n * Sets the anisotropy direction as an angle.\n */\n set angle(value) {\n this.direction.x = Math.cos(value);\n this.direction.y = Math.sin(value);\n }\n /**\n * Gets the anisotropy angle value in radians.\n * @returns the anisotropy angle value in radians.\n */\n get angle() {\n return Math.atan2(this.direction.y, this.direction.x);\n }\n /** @internal */\n _markAllSubMeshesAsTexturesDirty() {\n this._enable(this._isEnabled);\n this._internalMarkAllSubMeshesAsTexturesDirty();\n }\n /** @internal */\n _markAllSubMeshesAsMiscDirty() {\n this._enable(this._isEnabled);\n this._internalMarkAllSubMeshesAsMiscDirty();\n }\n constructor(material, addToPluginList = true) {\n super(material, \"PBRAnisotropic\", 110, new MaterialAnisotropicDefines(), addToPluginList);\n this._isEnabled = false;\n /**\n * Defines if the anisotropy is enabled in the material.\n */\n this.isEnabled = false;\n /**\n * Defines the anisotropy strength (between 0 and 1) it defaults to 1.\n */\n this.intensity = 1;\n /**\n * Defines if the effect is along the tangents, bitangents or in between.\n * By default, the effect is \"stretching\" the highlights along the tangents.\n */\n this.direction = new Vector2(1, 0);\n this._texture = null;\n /**\n * Stores the anisotropy values in a texture.\n * rg is direction (like normal from -1 to 1)\n * b is a intensity\n */\n this.texture = null;\n this._legacy = false;\n /**\n * Defines if the anisotropy is in legacy mode for backwards compatibility before 6.4.0.\n */\n this.legacy = false;\n this._internalMarkAllSubMeshesAsTexturesDirty = material._dirtyCallbacks[1];\n this._internalMarkAllSubMeshesAsMiscDirty = material._dirtyCallbacks[16];\n }\n isReadyForSubMesh(defines, scene) {\n if (!this._isEnabled) {\n return true;\n }\n if (defines._areTexturesDirty) {\n if (scene.texturesEnabled) {\n if (this._texture && MaterialFlags.AnisotropicTextureEnabled) {\n if (!this._texture.isReadyOrNotBlocking()) {\n return false;\n }\n }\n }\n }\n return true;\n }\n prepareDefinesBeforeAttributes(defines, scene, mesh) {\n if (this._isEnabled) {\n defines.ANISOTROPIC = this._isEnabled;\n if (this._isEnabled && !mesh.isVerticesDataPresent(VertexBuffer.TangentKind)) {\n defines._needUVs = true;\n defines.MAINUV1 = true;\n }\n if (defines._areTexturesDirty) {\n if (scene.texturesEnabled) {\n if (this._texture && MaterialFlags.AnisotropicTextureEnabled) {\n MaterialHelper.PrepareDefinesForMergedUV(this._texture, defines, \"ANISOTROPIC_TEXTURE\");\n }\n else {\n defines.ANISOTROPIC_TEXTURE = false;\n }\n }\n }\n if (defines._areMiscDirty) {\n defines.ANISOTROPIC_LEGACY = this._legacy;\n }\n }\n else {\n defines.ANISOTROPIC = false;\n defines.ANISOTROPIC_TEXTURE = false;\n defines.ANISOTROPIC_TEXTUREDIRECTUV = 0;\n defines.ANISOTROPIC_LEGACY = false;\n }\n }\n bindForSubMesh(uniformBuffer, scene) {\n if (!this._isEnabled) {\n return;\n }\n const isFrozen = this._material.isFrozen;\n if (!uniformBuffer.useUbo || !isFrozen || !uniformBuffer.isSync) {\n if (this._texture && MaterialFlags.AnisotropicTextureEnabled) {\n uniformBuffer.updateFloat2(\"vAnisotropyInfos\", this._texture.coordinatesIndex, this._texture.level);\n MaterialHelper.BindTextureMatrix(this._texture, uniformBuffer, \"anisotropy\");\n }\n // Anisotropy\n uniformBuffer.updateFloat3(\"vAnisotropy\", this.direction.x, this.direction.y, this.intensity);\n }\n // Textures\n if (scene.texturesEnabled) {\n if (this._texture && MaterialFlags.AnisotropicTextureEnabled) {\n uniformBuffer.setTexture(\"anisotropySampler\", this._texture);\n }\n }\n }\n hasTexture(texture) {\n if (this._texture === texture) {\n return true;\n }\n return false;\n }\n getActiveTextures(activeTextures) {\n if (this._texture) {\n activeTextures.push(this._texture);\n }\n }\n getAnimatables(animatables) {\n if (this._texture && this._texture.animations && this._texture.animations.length > 0) {\n animatables.push(this._texture);\n }\n }\n dispose(forceDisposeTextures) {\n if (forceDisposeTextures) {\n if (this._texture) {\n this._texture.dispose();\n }\n }\n }\n getClassName() {\n return \"PBRAnisotropicConfiguration\";\n }\n addFallbacks(defines, fallbacks, currentRank) {\n if (defines.ANISOTROPIC) {\n fallbacks.addFallback(currentRank++, \"ANISOTROPIC\");\n }\n return currentRank;\n }\n getSamplers(samplers) {\n samplers.push(\"anisotropySampler\");\n }\n getUniforms() {\n return {\n ubo: [\n { name: \"vAnisotropy\", size: 3, type: \"vec3\" },\n { name: \"vAnisotropyInfos\", size: 2, type: \"vec2\" },\n { name: \"anisotropyMatrix\", size: 16, type: \"mat4\" },\n ],\n };\n }\n /**\n * Parses a anisotropy Configuration from a serialized object.\n * @param source - Serialized object.\n * @param scene Defines the scene we are parsing for\n * @param rootUrl Defines the rootUrl to load from\n */\n parse(source, scene, rootUrl) {\n super.parse(source, scene, rootUrl);\n // Backward compatibility\n if (source.legacy === undefined) {\n this.legacy = true;\n }\n }\n }\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRAnisotropicConfiguration.prototype, \"isEnabled\", void 0);\n __decorate$1([\n serialize()\n ], PBRAnisotropicConfiguration.prototype, \"intensity\", void 0);\n __decorate$1([\n serializeAsVector2()\n ], PBRAnisotropicConfiguration.prototype, \"direction\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRAnisotropicConfiguration.prototype, \"texture\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsMiscDirty\")\n ], PBRAnisotropicConfiguration.prototype, \"legacy\", void 0);\n\n /**\n * @internal\n */\n class MaterialSheenDefines extends MaterialDefines {\n constructor() {\n super(...arguments);\n this.SHEEN = false;\n this.SHEEN_TEXTURE = false;\n this.SHEEN_GAMMATEXTURE = false;\n this.SHEEN_TEXTURE_ROUGHNESS = false;\n this.SHEEN_TEXTUREDIRECTUV = 0;\n this.SHEEN_TEXTURE_ROUGHNESSDIRECTUV = 0;\n this.SHEEN_LINKWITHALBEDO = false;\n this.SHEEN_ROUGHNESS = false;\n this.SHEEN_ALBEDOSCALING = false;\n this.SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE = false;\n this.SHEEN_TEXTURE_ROUGHNESS_IDENTICAL = false;\n }\n }\n /**\n * Plugin that implements the sheen component of the PBR material.\n */\n class PBRSheenConfiguration extends MaterialPluginBase {\n /** @internal */\n _markAllSubMeshesAsTexturesDirty() {\n this._enable(this._isEnabled);\n this._internalMarkAllSubMeshesAsTexturesDirty();\n }\n constructor(material, addToPluginList = true) {\n super(material, \"Sheen\", 120, new MaterialSheenDefines(), addToPluginList);\n this._isEnabled = false;\n /**\n * Defines if the material uses sheen.\n */\n this.isEnabled = false;\n this._linkSheenWithAlbedo = false;\n /**\n * Defines if the sheen is linked to the sheen color.\n */\n this.linkSheenWithAlbedo = false;\n /**\n * Defines the sheen intensity.\n */\n this.intensity = 1;\n /**\n * Defines the sheen color.\n */\n this.color = Color3.White();\n this._texture = null;\n /**\n * Stores the sheen tint values in a texture.\n * rgb is tint\n * a is a intensity or roughness if the roughness property has been defined and useRoughnessFromTexture is true (in that case, textureRoughness won't be used)\n * If the roughness property has been defined and useRoughnessFromTexture is false then the alpha channel is not used to modulate roughness\n */\n this.texture = null;\n this._useRoughnessFromMainTexture = true;\n /**\n * Indicates that the alpha channel of the texture property will be used for roughness.\n * Has no effect if the roughness (and texture!) property is not defined\n */\n this.useRoughnessFromMainTexture = true;\n this._roughness = null;\n /**\n * Defines the sheen roughness.\n * It is not taken into account if linkSheenWithAlbedo is true.\n * To stay backward compatible, material roughness is used instead if sheen roughness = null\n */\n this.roughness = null;\n this._textureRoughness = null;\n /**\n * Stores the sheen roughness in a texture.\n * alpha channel is the roughness. This texture won't be used if the texture property is not empty and useRoughnessFromTexture is true\n */\n this.textureRoughness = null;\n this._albedoScaling = false;\n /**\n * If true, the sheen effect is layered above the base BRDF with the albedo-scaling technique.\n * It allows the strength of the sheen effect to not depend on the base color of the material,\n * making it easier to setup and tweak the effect\n */\n this.albedoScaling = false;\n this._internalMarkAllSubMeshesAsTexturesDirty = material._dirtyCallbacks[1];\n }\n isReadyForSubMesh(defines, scene) {\n if (!this._isEnabled) {\n return true;\n }\n if (defines._areTexturesDirty) {\n if (scene.texturesEnabled) {\n if (this._texture && MaterialFlags.SheenTextureEnabled) {\n if (!this._texture.isReadyOrNotBlocking()) {\n return false;\n }\n }\n if (this._textureRoughness && MaterialFlags.SheenTextureEnabled) {\n if (!this._textureRoughness.isReadyOrNotBlocking()) {\n return false;\n }\n }\n }\n }\n return true;\n }\n prepareDefinesBeforeAttributes(defines, scene) {\n var _a;\n if (this._isEnabled) {\n defines.SHEEN = true;\n defines.SHEEN_LINKWITHALBEDO = this._linkSheenWithAlbedo;\n defines.SHEEN_ROUGHNESS = this._roughness !== null;\n defines.SHEEN_ALBEDOSCALING = this._albedoScaling;\n defines.SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE = this._useRoughnessFromMainTexture;\n defines.SHEEN_TEXTURE_ROUGHNESS_IDENTICAL =\n this._texture !== null && this._texture._texture === ((_a = this._textureRoughness) === null || _a === void 0 ? void 0 : _a._texture) && this._texture.checkTransformsAreIdentical(this._textureRoughness);\n if (defines._areTexturesDirty) {\n if (scene.texturesEnabled) {\n if (this._texture && MaterialFlags.SheenTextureEnabled) {\n MaterialHelper.PrepareDefinesForMergedUV(this._texture, defines, \"SHEEN_TEXTURE\");\n defines.SHEEN_GAMMATEXTURE = this._texture.gammaSpace;\n }\n else {\n defines.SHEEN_TEXTURE = false;\n }\n if (this._textureRoughness && MaterialFlags.SheenTextureEnabled) {\n MaterialHelper.PrepareDefinesForMergedUV(this._textureRoughness, defines, \"SHEEN_TEXTURE_ROUGHNESS\");\n }\n else {\n defines.SHEEN_TEXTURE_ROUGHNESS = false;\n }\n }\n }\n }\n else {\n defines.SHEEN = false;\n defines.SHEEN_TEXTURE = false;\n defines.SHEEN_TEXTURE_ROUGHNESS = false;\n defines.SHEEN_LINKWITHALBEDO = false;\n defines.SHEEN_ROUGHNESS = false;\n defines.SHEEN_ALBEDOSCALING = false;\n defines.SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE = false;\n defines.SHEEN_TEXTURE_ROUGHNESS_IDENTICAL = false;\n defines.SHEEN_GAMMATEXTURE = false;\n defines.SHEEN_TEXTUREDIRECTUV = 0;\n defines.SHEEN_TEXTURE_ROUGHNESSDIRECTUV = 0;\n }\n }\n bindForSubMesh(uniformBuffer, scene, engine, subMesh) {\n var _a, _b, _c, _d, _e, _f, _g, _h;\n if (!this._isEnabled) {\n return;\n }\n const defines = subMesh.materialDefines;\n const isFrozen = this._material.isFrozen;\n const identicalTextures = defines.SHEEN_TEXTURE_ROUGHNESS_IDENTICAL;\n if (!uniformBuffer.useUbo || !isFrozen || !uniformBuffer.isSync) {\n if (identicalTextures && MaterialFlags.SheenTextureEnabled) {\n uniformBuffer.updateFloat4(\"vSheenInfos\", this._texture.coordinatesIndex, this._texture.level, -1, -1);\n MaterialHelper.BindTextureMatrix(this._texture, uniformBuffer, \"sheen\");\n }\n else if ((this._texture || this._textureRoughness) && MaterialFlags.SheenTextureEnabled) {\n uniformBuffer.updateFloat4(\"vSheenInfos\", (_b = (_a = this._texture) === null || _a === void 0 ? void 0 : _a.coordinatesIndex) !== null && _b !== void 0 ? _b : 0, (_d = (_c = this._texture) === null || _c === void 0 ? void 0 : _c.level) !== null && _d !== void 0 ? _d : 0, (_f = (_e = this._textureRoughness) === null || _e === void 0 ? void 0 : _e.coordinatesIndex) !== null && _f !== void 0 ? _f : 0, (_h = (_g = this._textureRoughness) === null || _g === void 0 ? void 0 : _g.level) !== null && _h !== void 0 ? _h : 0);\n if (this._texture) {\n MaterialHelper.BindTextureMatrix(this._texture, uniformBuffer, \"sheen\");\n }\n if (this._textureRoughness && !identicalTextures && !defines.SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE) {\n MaterialHelper.BindTextureMatrix(this._textureRoughness, uniformBuffer, \"sheenRoughness\");\n }\n }\n // Sheen\n uniformBuffer.updateFloat4(\"vSheenColor\", this.color.r, this.color.g, this.color.b, this.intensity);\n if (this._roughness !== null) {\n uniformBuffer.updateFloat(\"vSheenRoughness\", this._roughness);\n }\n }\n // Textures\n if (scene.texturesEnabled) {\n if (this._texture && MaterialFlags.SheenTextureEnabled) {\n uniformBuffer.setTexture(\"sheenSampler\", this._texture);\n }\n if (this._textureRoughness && !identicalTextures && !defines.SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE && MaterialFlags.SheenTextureEnabled) {\n uniformBuffer.setTexture(\"sheenRoughnessSampler\", this._textureRoughness);\n }\n }\n }\n hasTexture(texture) {\n if (this._texture === texture) {\n return true;\n }\n if (this._textureRoughness === texture) {\n return true;\n }\n return false;\n }\n getActiveTextures(activeTextures) {\n if (this._texture) {\n activeTextures.push(this._texture);\n }\n if (this._textureRoughness) {\n activeTextures.push(this._textureRoughness);\n }\n }\n getAnimatables(animatables) {\n if (this._texture && this._texture.animations && this._texture.animations.length > 0) {\n animatables.push(this._texture);\n }\n if (this._textureRoughness && this._textureRoughness.animations && this._textureRoughness.animations.length > 0) {\n animatables.push(this._textureRoughness);\n }\n }\n dispose(forceDisposeTextures) {\n var _a, _b;\n if (forceDisposeTextures) {\n (_a = this._texture) === null || _a === void 0 ? void 0 : _a.dispose();\n (_b = this._textureRoughness) === null || _b === void 0 ? void 0 : _b.dispose();\n }\n }\n getClassName() {\n return \"PBRSheenConfiguration\";\n }\n addFallbacks(defines, fallbacks, currentRank) {\n if (defines.SHEEN) {\n fallbacks.addFallback(currentRank++, \"SHEEN\");\n }\n return currentRank;\n }\n getSamplers(samplers) {\n samplers.push(\"sheenSampler\", \"sheenRoughnessSampler\");\n }\n getUniforms() {\n return {\n ubo: [\n { name: \"vSheenColor\", size: 4, type: \"vec4\" },\n { name: \"vSheenRoughness\", size: 1, type: \"float\" },\n { name: \"vSheenInfos\", size: 4, type: \"vec4\" },\n { name: \"sheenMatrix\", size: 16, type: \"mat4\" },\n { name: \"sheenRoughnessMatrix\", size: 16, type: \"mat4\" },\n ],\n };\n }\n }\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRSheenConfiguration.prototype, \"isEnabled\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRSheenConfiguration.prototype, \"linkSheenWithAlbedo\", void 0);\n __decorate$1([\n serialize()\n ], PBRSheenConfiguration.prototype, \"intensity\", void 0);\n __decorate$1([\n serializeAsColor3()\n ], PBRSheenConfiguration.prototype, \"color\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRSheenConfiguration.prototype, \"texture\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRSheenConfiguration.prototype, \"useRoughnessFromMainTexture\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRSheenConfiguration.prototype, \"roughness\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRSheenConfiguration.prototype, \"textureRoughness\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRSheenConfiguration.prototype, \"albedoScaling\", void 0);\n\n /**\n * @internal\n */\n class MaterialSubSurfaceDefines extends MaterialDefines {\n constructor() {\n super(...arguments);\n this.SUBSURFACE = false;\n this.SS_REFRACTION = false;\n this.SS_REFRACTION_USE_INTENSITY_FROM_TEXTURE = false;\n this.SS_TRANSLUCENCY = false;\n this.SS_TRANSLUCENCY_USE_INTENSITY_FROM_TEXTURE = false;\n this.SS_SCATTERING = false;\n this.SS_THICKNESSANDMASK_TEXTURE = false;\n this.SS_THICKNESSANDMASK_TEXTUREDIRECTUV = 0;\n this.SS_HAS_THICKNESS = false;\n this.SS_REFRACTIONINTENSITY_TEXTURE = false;\n this.SS_REFRACTIONINTENSITY_TEXTUREDIRECTUV = 0;\n this.SS_TRANSLUCENCYINTENSITY_TEXTURE = false;\n this.SS_TRANSLUCENCYINTENSITY_TEXTUREDIRECTUV = 0;\n this.SS_REFRACTIONMAP_3D = false;\n this.SS_REFRACTIONMAP_OPPOSITEZ = false;\n this.SS_LODINREFRACTIONALPHA = false;\n this.SS_GAMMAREFRACTION = false;\n this.SS_RGBDREFRACTION = false;\n this.SS_LINEARSPECULARREFRACTION = false;\n this.SS_LINKREFRACTIONTOTRANSPARENCY = false;\n this.SS_ALBEDOFORREFRACTIONTINT = false;\n this.SS_ALBEDOFORTRANSLUCENCYTINT = false;\n this.SS_USE_LOCAL_REFRACTIONMAP_CUBIC = false;\n this.SS_USE_THICKNESS_AS_DEPTH = false;\n this.SS_MASK_FROM_THICKNESS_TEXTURE = false;\n this.SS_USE_GLTF_TEXTURES = false;\n }\n }\n /**\n * Plugin that implements the sub surface component of the PBR material\n */\n class PBRSubSurfaceConfiguration extends MaterialPluginBase {\n /**\n * Diffusion profile for subsurface scattering.\n * Useful for better scattering in the skins or foliages.\n */\n get scatteringDiffusionProfile() {\n if (!this._scene.subSurfaceConfiguration) {\n return null;\n }\n return this._scene.subSurfaceConfiguration.ssDiffusionProfileColors[this._scatteringDiffusionProfileIndex];\n }\n set scatteringDiffusionProfile(c) {\n if (!this._scene.enableSubSurfaceForPrePass()) {\n // Not supported\n return;\n }\n // addDiffusionProfile automatically checks for doubles\n if (c) {\n this._scatteringDiffusionProfileIndex = this._scene.subSurfaceConfiguration.addDiffusionProfile(c);\n }\n }\n /**\n * Index of refraction of the material's volume.\n * https://en.wikipedia.org/wiki/List_of_refractive_indices\n *\n * This ONLY impacts refraction. If not provided or given a non-valid value,\n * the volume will use the same IOR as the surface.\n */\n get volumeIndexOfRefraction() {\n if (this._volumeIndexOfRefraction >= 1.0) {\n return this._volumeIndexOfRefraction;\n }\n return this._indexOfRefraction;\n }\n set volumeIndexOfRefraction(value) {\n if (value >= 1.0) {\n this._volumeIndexOfRefraction = value;\n }\n else {\n this._volumeIndexOfRefraction = -1.0;\n }\n }\n /** @internal */\n _markAllSubMeshesAsTexturesDirty() {\n this._enable(this._isRefractionEnabled || this._isTranslucencyEnabled || this._isScatteringEnabled);\n this._internalMarkAllSubMeshesAsTexturesDirty();\n }\n /** @internal */\n _markScenePrePassDirty() {\n this._internalMarkAllSubMeshesAsTexturesDirty();\n this._internalMarkScenePrePassDirty();\n }\n constructor(material, addToPluginList = true) {\n super(material, \"PBRSubSurface\", 130, new MaterialSubSurfaceDefines(), addToPluginList);\n this._isRefractionEnabled = false;\n /**\n * Defines if the refraction is enabled in the material.\n */\n this.isRefractionEnabled = false;\n this._isTranslucencyEnabled = false;\n /**\n * Defines if the translucency is enabled in the material.\n */\n this.isTranslucencyEnabled = false;\n this._isScatteringEnabled = false;\n /**\n * Defines if the sub surface scattering is enabled in the material.\n */\n this.isScatteringEnabled = false;\n this._scatteringDiffusionProfileIndex = 0;\n /**\n * Defines the refraction intensity of the material.\n * The refraction when enabled replaces the Diffuse part of the material.\n * The intensity helps transitioning between diffuse and refraction.\n */\n this.refractionIntensity = 1;\n /**\n * Defines the translucency intensity of the material.\n * When translucency has been enabled, this defines how much of the \"translucency\"\n * is added to the diffuse part of the material.\n */\n this.translucencyIntensity = 1;\n /**\n * When enabled, transparent surfaces will be tinted with the albedo colour (independent of thickness)\n */\n this.useAlbedoToTintRefraction = false;\n /**\n * When enabled, translucent surfaces will be tinted with the albedo colour (independent of thickness)\n */\n this.useAlbedoToTintTranslucency = false;\n this._thicknessTexture = null;\n /**\n * Stores the average thickness of a mesh in a texture (The texture is holding the values linearly).\n * The red (or green if useGltfStyleTextures=true) channel of the texture should contain the thickness remapped between 0 and 1.\n * 0 would mean minimumThickness\n * 1 would mean maximumThickness\n * The other channels might be use as a mask to vary the different effects intensity.\n */\n this.thicknessTexture = null;\n this._refractionTexture = null;\n /**\n * Defines the texture to use for refraction.\n */\n this.refractionTexture = null;\n /** @internal */\n this._indexOfRefraction = 1.5;\n /**\n * Index of refraction of the material base layer.\n * https://en.wikipedia.org/wiki/List_of_refractive_indices\n *\n * This does not only impact refraction but also the Base F0 of Dielectric Materials.\n *\n * From dielectric fresnel rules: F0 = square((iorT - iorI) / (iorT + iorI))\n */\n this.indexOfRefraction = 1.5;\n this._volumeIndexOfRefraction = -1.0;\n this._invertRefractionY = false;\n /**\n * Controls if refraction needs to be inverted on Y. This could be useful for procedural texture.\n */\n this.invertRefractionY = false;\n /** @internal */\n this._linkRefractionWithTransparency = false;\n /**\n * This parameters will make the material used its opacity to control how much it is refracting against not.\n * Materials half opaque for instance using refraction could benefit from this control.\n */\n this.linkRefractionWithTransparency = false;\n /**\n * Defines the minimum thickness stored in the thickness map.\n * If no thickness map is defined, this value will be used to simulate thickness.\n */\n this.minimumThickness = 0;\n /**\n * Defines the maximum thickness stored in the thickness map.\n */\n this.maximumThickness = 1;\n /**\n * Defines that the thickness should be used as a measure of the depth volume.\n */\n this.useThicknessAsDepth = false;\n /**\n * Defines the volume tint of the material.\n * This is used for both translucency and scattering.\n */\n this.tintColor = Color3.White();\n /**\n * Defines the distance at which the tint color should be found in the media.\n * This is used for refraction only.\n */\n this.tintColorAtDistance = 1;\n /**\n * Defines how far each channel transmit through the media.\n * It is defined as a color to simplify it selection.\n */\n this.diffusionDistance = Color3.White();\n this._useMaskFromThicknessTexture = false;\n /**\n * Stores the intensity of the different subsurface effects in the thickness texture.\n * Note that if refractionIntensityTexture and/or translucencyIntensityTexture is provided it takes precedence over thicknessTexture + useMaskFromThicknessTexture\n * * the green (red if useGltfStyleTextures = true) channel is the refraction intensity.\n * * the blue channel is the translucency intensity.\n */\n this.useMaskFromThicknessTexture = false;\n this._refractionIntensityTexture = null;\n /**\n * Stores the intensity of the refraction. If provided, it takes precedence over thicknessTexture + useMaskFromThicknessTexture\n * * the green (red if useGltfStyleTextures = true) channel is the refraction intensity.\n */\n this.refractionIntensityTexture = null;\n this._translucencyIntensityTexture = null;\n /**\n * Stores the intensity of the translucency. If provided, it takes precedence over thicknessTexture + useMaskFromThicknessTexture\n * * the blue channel is the translucency intensity.\n */\n this.translucencyIntensityTexture = null;\n this._useGltfStyleTextures = false;\n /**\n * Use channels layout used by glTF:\n * * thicknessTexture: the green (instead of red) channel is the thickness\n * * thicknessTexture/refractionIntensityTexture: the red (instead of green) channel is the refraction intensity\n * * thicknessTexture/translucencyIntensityTexture: no change, use the blue channel for the translucency intensity\n */\n this.useGltfStyleTextures = false;\n this._scene = material.getScene();\n this.registerForExtraEvents = true;\n this._internalMarkAllSubMeshesAsTexturesDirty = material._dirtyCallbacks[1];\n this._internalMarkScenePrePassDirty = material._dirtyCallbacks[32];\n }\n isReadyForSubMesh(defines, scene) {\n if (!this._isRefractionEnabled && !this._isTranslucencyEnabled && !this._isScatteringEnabled) {\n return true;\n }\n if (defines._areTexturesDirty) {\n if (scene.texturesEnabled) {\n if (this._thicknessTexture && MaterialFlags.ThicknessTextureEnabled) {\n if (!this._thicknessTexture.isReadyOrNotBlocking()) {\n return false;\n }\n }\n const refractionTexture = this._getRefractionTexture(scene);\n if (refractionTexture && MaterialFlags.RefractionTextureEnabled) {\n if (!refractionTexture.isReadyOrNotBlocking()) {\n return false;\n }\n }\n }\n }\n return true;\n }\n prepareDefinesBeforeAttributes(defines, scene) {\n if (!this._isRefractionEnabled && !this._isTranslucencyEnabled && !this._isScatteringEnabled) {\n defines.SUBSURFACE = false;\n defines.SS_TRANSLUCENCY = false;\n defines.SS_SCATTERING = false;\n defines.SS_REFRACTION = false;\n defines.SS_REFRACTION_USE_INTENSITY_FROM_TEXTURE = false;\n defines.SS_TRANSLUCENCY_USE_INTENSITY_FROM_TEXTURE = false;\n defines.SS_THICKNESSANDMASK_TEXTURE = false;\n defines.SS_THICKNESSANDMASK_TEXTUREDIRECTUV = 0;\n defines.SS_HAS_THICKNESS = false;\n defines.SS_REFRACTIONINTENSITY_TEXTURE = false;\n defines.SS_REFRACTIONINTENSITY_TEXTUREDIRECTUV = 0;\n defines.SS_TRANSLUCENCYINTENSITY_TEXTURE = false;\n defines.SS_TRANSLUCENCYINTENSITY_TEXTUREDIRECTUV = 0;\n defines.SS_REFRACTIONMAP_3D = false;\n defines.SS_REFRACTIONMAP_OPPOSITEZ = false;\n defines.SS_LODINREFRACTIONALPHA = false;\n defines.SS_GAMMAREFRACTION = false;\n defines.SS_RGBDREFRACTION = false;\n defines.SS_LINEARSPECULARREFRACTION = false;\n defines.SS_LINKREFRACTIONTOTRANSPARENCY = false;\n defines.SS_ALBEDOFORREFRACTIONTINT = false;\n defines.SS_ALBEDOFORTRANSLUCENCYTINT = false;\n defines.SS_USE_LOCAL_REFRACTIONMAP_CUBIC = false;\n defines.SS_USE_THICKNESS_AS_DEPTH = false;\n defines.SS_MASK_FROM_THICKNESS_TEXTURE = false;\n defines.SS_USE_GLTF_TEXTURES = false;\n return;\n }\n if (defines._areTexturesDirty) {\n defines.SUBSURFACE = true;\n defines.SS_TRANSLUCENCY = this._isTranslucencyEnabled;\n defines.SS_TRANSLUCENCY_USE_INTENSITY_FROM_TEXTURE = false;\n defines.SS_SCATTERING = this._isScatteringEnabled;\n defines.SS_THICKNESSANDMASK_TEXTURE = false;\n defines.SS_REFRACTIONINTENSITY_TEXTURE = false;\n defines.SS_TRANSLUCENCYINTENSITY_TEXTURE = false;\n defines.SS_HAS_THICKNESS = false;\n defines.SS_MASK_FROM_THICKNESS_TEXTURE = false;\n defines.SS_USE_GLTF_TEXTURES = false;\n defines.SS_REFRACTION = false;\n defines.SS_REFRACTION_USE_INTENSITY_FROM_TEXTURE = false;\n defines.SS_REFRACTIONMAP_3D = false;\n defines.SS_GAMMAREFRACTION = false;\n defines.SS_RGBDREFRACTION = false;\n defines.SS_LINEARSPECULARREFRACTION = false;\n defines.SS_REFRACTIONMAP_OPPOSITEZ = false;\n defines.SS_LODINREFRACTIONALPHA = false;\n defines.SS_LINKREFRACTIONTOTRANSPARENCY = false;\n defines.SS_ALBEDOFORREFRACTIONTINT = false;\n defines.SS_ALBEDOFORTRANSLUCENCYTINT = false;\n defines.SS_USE_LOCAL_REFRACTIONMAP_CUBIC = false;\n defines.SS_USE_THICKNESS_AS_DEPTH = false;\n const refractionIntensityTextureIsThicknessTexture = !!this._thicknessTexture &&\n !!this._refractionIntensityTexture &&\n this._refractionIntensityTexture.checkTransformsAreIdentical(this._thicknessTexture) &&\n this._refractionIntensityTexture._texture === this._thicknessTexture._texture;\n const translucencyIntensityTextureIsThicknessTexture = !!this._thicknessTexture &&\n !!this._translucencyIntensityTexture &&\n this._translucencyIntensityTexture.checkTransformsAreIdentical(this._thicknessTexture) &&\n this._translucencyIntensityTexture._texture === this._thicknessTexture._texture;\n // if true, it means the refraction/translucency textures are the same than the thickness texture so there's no need to pass them to the shader, only thicknessTexture\n const useOnlyThicknessTexture = (refractionIntensityTextureIsThicknessTexture || !this._refractionIntensityTexture) &&\n (translucencyIntensityTextureIsThicknessTexture || !this._translucencyIntensityTexture);\n if (defines._areTexturesDirty) {\n if (scene.texturesEnabled) {\n if (this._thicknessTexture && MaterialFlags.ThicknessTextureEnabled) {\n MaterialHelper.PrepareDefinesForMergedUV(this._thicknessTexture, defines, \"SS_THICKNESSANDMASK_TEXTURE\");\n }\n if (this._refractionIntensityTexture && MaterialFlags.RefractionIntensityTextureEnabled && !useOnlyThicknessTexture) {\n MaterialHelper.PrepareDefinesForMergedUV(this._refractionIntensityTexture, defines, \"SS_REFRACTIONINTENSITY_TEXTURE\");\n }\n if (this._translucencyIntensityTexture && MaterialFlags.TranslucencyIntensityTextureEnabled && !useOnlyThicknessTexture) {\n MaterialHelper.PrepareDefinesForMergedUV(this._translucencyIntensityTexture, defines, \"SS_TRANSLUCENCYINTENSITY_TEXTURE\");\n }\n }\n }\n defines.SS_HAS_THICKNESS = this.maximumThickness - this.minimumThickness !== 0.0;\n defines.SS_MASK_FROM_THICKNESS_TEXTURE =\n (this._useMaskFromThicknessTexture || !!this._refractionIntensityTexture || !!this._translucencyIntensityTexture) && useOnlyThicknessTexture;\n defines.SS_USE_GLTF_TEXTURES = this._useGltfStyleTextures;\n defines.SS_REFRACTION_USE_INTENSITY_FROM_TEXTURE = (this._useMaskFromThicknessTexture || !!this._refractionIntensityTexture) && useOnlyThicknessTexture;\n defines.SS_TRANSLUCENCY_USE_INTENSITY_FROM_TEXTURE = (this._useMaskFromThicknessTexture || !!this._translucencyIntensityTexture) && useOnlyThicknessTexture;\n if (this._isRefractionEnabled) {\n if (scene.texturesEnabled) {\n const refractionTexture = this._getRefractionTexture(scene);\n if (refractionTexture && MaterialFlags.RefractionTextureEnabled) {\n defines.SS_REFRACTION = true;\n defines.SS_REFRACTIONMAP_3D = refractionTexture.isCube;\n defines.SS_GAMMAREFRACTION = refractionTexture.gammaSpace;\n defines.SS_RGBDREFRACTION = refractionTexture.isRGBD;\n defines.SS_LINEARSPECULARREFRACTION = refractionTexture.linearSpecularLOD;\n defines.SS_REFRACTIONMAP_OPPOSITEZ = this._scene.useRightHandedSystem && refractionTexture.isCube ? !refractionTexture.invertZ : refractionTexture.invertZ;\n defines.SS_LODINREFRACTIONALPHA = refractionTexture.lodLevelInAlpha;\n defines.SS_LINKREFRACTIONTOTRANSPARENCY = this._linkRefractionWithTransparency;\n defines.SS_ALBEDOFORREFRACTIONTINT = this.useAlbedoToTintRefraction;\n defines.SS_USE_LOCAL_REFRACTIONMAP_CUBIC = refractionTexture.isCube && refractionTexture.boundingBoxSize;\n defines.SS_USE_THICKNESS_AS_DEPTH = this.useThicknessAsDepth;\n }\n }\n }\n if (this._isTranslucencyEnabled) {\n defines.SS_ALBEDOFORTRANSLUCENCYTINT = this.useAlbedoToTintTranslucency;\n }\n }\n }\n /**\n * Binds the material data (this function is called even if mustRebind() returns false)\n * @param uniformBuffer defines the Uniform buffer to fill in.\n * @param scene defines the scene the material belongs to.\n * @param engine defines the engine the material belongs to.\n * @param subMesh the submesh to bind data for\n */\n hardBindForSubMesh(uniformBuffer, scene, engine, subMesh) {\n if (!this._isRefractionEnabled && !this._isTranslucencyEnabled && !this._isScatteringEnabled) {\n return;\n }\n subMesh.getRenderingMesh().getWorldMatrix().decompose(TmpVectors.Vector3[0]);\n const thicknessScale = Math.max(Math.abs(TmpVectors.Vector3[0].x), Math.abs(TmpVectors.Vector3[0].y), Math.abs(TmpVectors.Vector3[0].z));\n uniformBuffer.updateFloat2(\"vThicknessParam\", this.minimumThickness * thicknessScale, (this.maximumThickness - this.minimumThickness) * thicknessScale);\n }\n bindForSubMesh(uniformBuffer, scene, engine, subMesh) {\n if (!this._isRefractionEnabled && !this._isTranslucencyEnabled && !this._isScatteringEnabled) {\n return;\n }\n const defines = subMesh.materialDefines;\n const isFrozen = this._material.isFrozen;\n const realTimeFiltering = this._material.realTimeFiltering;\n const lodBasedMicrosurface = defines.LODBASEDMICROSFURACE;\n const refractionTexture = this._getRefractionTexture(scene);\n if (!uniformBuffer.useUbo || !isFrozen || !uniformBuffer.isSync) {\n if (this._thicknessTexture && MaterialFlags.ThicknessTextureEnabled) {\n uniformBuffer.updateFloat2(\"vThicknessInfos\", this._thicknessTexture.coordinatesIndex, this._thicknessTexture.level);\n MaterialHelper.BindTextureMatrix(this._thicknessTexture, uniformBuffer, \"thickness\");\n }\n if (this._refractionIntensityTexture && MaterialFlags.RefractionIntensityTextureEnabled && defines.SS_REFRACTIONINTENSITY_TEXTURE) {\n uniformBuffer.updateFloat2(\"vRefractionIntensityInfos\", this._refractionIntensityTexture.coordinatesIndex, this._refractionIntensityTexture.level);\n MaterialHelper.BindTextureMatrix(this._refractionIntensityTexture, uniformBuffer, \"refractionIntensity\");\n }\n if (this._translucencyIntensityTexture && MaterialFlags.TranslucencyIntensityTextureEnabled && defines.SS_TRANSLUCENCYINTENSITY_TEXTURE) {\n uniformBuffer.updateFloat2(\"vTranslucencyIntensityInfos\", this._translucencyIntensityTexture.coordinatesIndex, this._translucencyIntensityTexture.level);\n MaterialHelper.BindTextureMatrix(this._translucencyIntensityTexture, uniformBuffer, \"translucencyIntensity\");\n }\n if (refractionTexture && MaterialFlags.RefractionTextureEnabled) {\n uniformBuffer.updateMatrix(\"refractionMatrix\", refractionTexture.getRefractionTextureMatrix());\n let depth = 1.0;\n if (!refractionTexture.isCube) {\n if (refractionTexture.depth) {\n depth = refractionTexture.depth;\n }\n }\n const width = refractionTexture.getSize().width;\n const refractionIor = this.volumeIndexOfRefraction;\n uniformBuffer.updateFloat4(\"vRefractionInfos\", refractionTexture.level, 1 / refractionIor, depth, this._invertRefractionY ? -1 : 1);\n uniformBuffer.updateFloat4(\"vRefractionMicrosurfaceInfos\", width, refractionTexture.lodGenerationScale, refractionTexture.lodGenerationOffset, 1.0 / this.indexOfRefraction);\n if (realTimeFiltering) {\n uniformBuffer.updateFloat2(\"vRefractionFilteringInfo\", width, Scalar.Log2(width));\n }\n if (refractionTexture.boundingBoxSize) {\n const cubeTexture = refractionTexture;\n uniformBuffer.updateVector3(\"vRefractionPosition\", cubeTexture.boundingBoxPosition);\n uniformBuffer.updateVector3(\"vRefractionSize\", cubeTexture.boundingBoxSize);\n }\n }\n if (this._isScatteringEnabled) {\n uniformBuffer.updateFloat(\"scatteringDiffusionProfile\", this._scatteringDiffusionProfileIndex);\n }\n uniformBuffer.updateColor3(\"vDiffusionDistance\", this.diffusionDistance);\n uniformBuffer.updateFloat4(\"vTintColor\", this.tintColor.r, this.tintColor.g, this.tintColor.b, Math.max(0.00001, this.tintColorAtDistance));\n uniformBuffer.updateFloat3(\"vSubSurfaceIntensity\", this.refractionIntensity, this.translucencyIntensity, 0);\n }\n // Textures\n if (scene.texturesEnabled) {\n if (this._thicknessTexture && MaterialFlags.ThicknessTextureEnabled) {\n uniformBuffer.setTexture(\"thicknessSampler\", this._thicknessTexture);\n }\n if (this._refractionIntensityTexture && MaterialFlags.RefractionIntensityTextureEnabled && defines.SS_REFRACTIONINTENSITY_TEXTURE) {\n uniformBuffer.setTexture(\"refractionIntensitySampler\", this._refractionIntensityTexture);\n }\n if (this._translucencyIntensityTexture && MaterialFlags.TranslucencyIntensityTextureEnabled && defines.SS_TRANSLUCENCYINTENSITY_TEXTURE) {\n uniformBuffer.setTexture(\"translucencyIntensitySampler\", this._translucencyIntensityTexture);\n }\n if (refractionTexture && MaterialFlags.RefractionTextureEnabled) {\n if (lodBasedMicrosurface) {\n uniformBuffer.setTexture(\"refractionSampler\", refractionTexture);\n }\n else {\n uniformBuffer.setTexture(\"refractionSampler\", refractionTexture._lodTextureMid || refractionTexture);\n uniformBuffer.setTexture(\"refractionSamplerLow\", refractionTexture._lodTextureLow || refractionTexture);\n uniformBuffer.setTexture(\"refractionSamplerHigh\", refractionTexture._lodTextureHigh || refractionTexture);\n }\n }\n }\n }\n /**\n * Returns the texture used for refraction or null if none is used.\n * @param scene defines the scene the material belongs to.\n * @returns - Refraction texture if present. If no refraction texture and refraction\n * is linked with transparency, returns environment texture. Otherwise, returns null.\n */\n _getRefractionTexture(scene) {\n if (this._refractionTexture) {\n return this._refractionTexture;\n }\n if (this._isRefractionEnabled) {\n return scene.environmentTexture;\n }\n return null;\n }\n /**\n * Returns true if alpha blending should be disabled.\n */\n get disableAlphaBlending() {\n return this._isRefractionEnabled && this._linkRefractionWithTransparency;\n }\n /**\n * Fills the list of render target textures.\n * @param renderTargets the list of render targets to update\n */\n fillRenderTargetTextures(renderTargets) {\n if (MaterialFlags.RefractionTextureEnabled && this._refractionTexture && this._refractionTexture.isRenderTarget) {\n renderTargets.push(this._refractionTexture);\n }\n }\n hasTexture(texture) {\n if (this._thicknessTexture === texture) {\n return true;\n }\n if (this._refractionTexture === texture) {\n return true;\n }\n return false;\n }\n hasRenderTargetTextures() {\n if (MaterialFlags.RefractionTextureEnabled && this._refractionTexture && this._refractionTexture.isRenderTarget) {\n return true;\n }\n return false;\n }\n getActiveTextures(activeTextures) {\n if (this._thicknessTexture) {\n activeTextures.push(this._thicknessTexture);\n }\n if (this._refractionTexture) {\n activeTextures.push(this._refractionTexture);\n }\n }\n getAnimatables(animatables) {\n if (this._thicknessTexture && this._thicknessTexture.animations && this._thicknessTexture.animations.length > 0) {\n animatables.push(this._thicknessTexture);\n }\n if (this._refractionTexture && this._refractionTexture.animations && this._refractionTexture.animations.length > 0) {\n animatables.push(this._refractionTexture);\n }\n }\n dispose(forceDisposeTextures) {\n if (forceDisposeTextures) {\n if (this._thicknessTexture) {\n this._thicknessTexture.dispose();\n }\n if (this._refractionTexture) {\n this._refractionTexture.dispose();\n }\n }\n }\n getClassName() {\n return \"PBRSubSurfaceConfiguration\";\n }\n addFallbacks(defines, fallbacks, currentRank) {\n if (defines.SS_SCATTERING) {\n fallbacks.addFallback(currentRank++, \"SS_SCATTERING\");\n }\n if (defines.SS_TRANSLUCENCY) {\n fallbacks.addFallback(currentRank++, \"SS_TRANSLUCENCY\");\n }\n return currentRank;\n }\n getSamplers(samplers) {\n samplers.push(\"thicknessSampler\", \"refractionIntensitySampler\", \"translucencyIntensitySampler\", \"refractionSampler\", \"refractionSamplerLow\", \"refractionSamplerHigh\");\n }\n getUniforms() {\n return {\n ubo: [\n { name: \"vRefractionMicrosurfaceInfos\", size: 4, type: \"vec4\" },\n { name: \"vRefractionFilteringInfo\", size: 2, type: \"vec2\" },\n { name: \"vTranslucencyIntensityInfos\", size: 2, type: \"vec2\" },\n { name: \"vRefractionInfos\", size: 4, type: \"vec4\" },\n { name: \"refractionMatrix\", size: 16, type: \"mat4\" },\n { name: \"vThicknessInfos\", size: 2, type: \"vec2\" },\n { name: \"vRefractionIntensityInfos\", size: 2, type: \"vec2\" },\n { name: \"thicknessMatrix\", size: 16, type: \"mat4\" },\n { name: \"refractionIntensityMatrix\", size: 16, type: \"mat4\" },\n { name: \"translucencyIntensityMatrix\", size: 16, type: \"mat4\" },\n { name: \"vThicknessParam\", size: 2, type: \"vec2\" },\n { name: \"vDiffusionDistance\", size: 3, type: \"vec3\" },\n { name: \"vTintColor\", size: 4, type: \"vec4\" },\n { name: \"vSubSurfaceIntensity\", size: 3, type: \"vec3\" },\n { name: \"vRefractionPosition\", size: 3, type: \"vec3\" },\n { name: \"vRefractionSize\", size: 3, type: \"vec3\" },\n { name: \"scatteringDiffusionProfile\", size: 1, type: \"float\" },\n ],\n };\n }\n }\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRSubSurfaceConfiguration.prototype, \"isRefractionEnabled\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRSubSurfaceConfiguration.prototype, \"isTranslucencyEnabled\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markScenePrePassDirty\")\n ], PBRSubSurfaceConfiguration.prototype, \"isScatteringEnabled\", void 0);\n __decorate$1([\n serialize()\n ], PBRSubSurfaceConfiguration.prototype, \"_scatteringDiffusionProfileIndex\", void 0);\n __decorate$1([\n serialize()\n ], PBRSubSurfaceConfiguration.prototype, \"refractionIntensity\", void 0);\n __decorate$1([\n serialize()\n ], PBRSubSurfaceConfiguration.prototype, \"translucencyIntensity\", void 0);\n __decorate$1([\n serialize()\n ], PBRSubSurfaceConfiguration.prototype, \"useAlbedoToTintRefraction\", void 0);\n __decorate$1([\n serialize()\n ], PBRSubSurfaceConfiguration.prototype, \"useAlbedoToTintTranslucency\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRSubSurfaceConfiguration.prototype, \"thicknessTexture\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRSubSurfaceConfiguration.prototype, \"refractionTexture\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRSubSurfaceConfiguration.prototype, \"indexOfRefraction\", void 0);\n __decorate$1([\n serialize()\n ], PBRSubSurfaceConfiguration.prototype, \"_volumeIndexOfRefraction\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRSubSurfaceConfiguration.prototype, \"volumeIndexOfRefraction\", null);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRSubSurfaceConfiguration.prototype, \"invertRefractionY\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRSubSurfaceConfiguration.prototype, \"linkRefractionWithTransparency\", void 0);\n __decorate$1([\n serialize()\n ], PBRSubSurfaceConfiguration.prototype, \"minimumThickness\", void 0);\n __decorate$1([\n serialize()\n ], PBRSubSurfaceConfiguration.prototype, \"maximumThickness\", void 0);\n __decorate$1([\n serialize()\n ], PBRSubSurfaceConfiguration.prototype, \"useThicknessAsDepth\", void 0);\n __decorate$1([\n serializeAsColor3()\n ], PBRSubSurfaceConfiguration.prototype, \"tintColor\", void 0);\n __decorate$1([\n serialize()\n ], PBRSubSurfaceConfiguration.prototype, \"tintColorAtDistance\", void 0);\n __decorate$1([\n serializeAsColor3()\n ], PBRSubSurfaceConfiguration.prototype, \"diffusionDistance\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRSubSurfaceConfiguration.prototype, \"useMaskFromThicknessTexture\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRSubSurfaceConfiguration.prototype, \"refractionIntensityTexture\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRSubSurfaceConfiguration.prototype, \"translucencyIntensityTexture\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRSubSurfaceConfiguration.prototype, \"useGltfStyleTextures\", void 0);\n\n const onCreatedEffectParameters$1 = { effect: null, subMesh: null };\n /**\n * Manages the defines for the PBR Material.\n * @internal\n */\n class PBRMaterialDefines extends MaterialDefines {\n /**\n * Initializes the PBR Material defines.\n * @param externalProperties The external properties\n */\n constructor(externalProperties) {\n super(externalProperties);\n this.PBR = true;\n this.NUM_SAMPLES = \"0\";\n this.REALTIME_FILTERING = false;\n this.MAINUV1 = false;\n this.MAINUV2 = false;\n this.MAINUV3 = false;\n this.MAINUV4 = false;\n this.MAINUV5 = false;\n this.MAINUV6 = false;\n this.UV1 = false;\n this.UV2 = false;\n this.UV3 = false;\n this.UV4 = false;\n this.UV5 = false;\n this.UV6 = false;\n this.ALBEDO = false;\n this.GAMMAALBEDO = false;\n this.ALBEDODIRECTUV = 0;\n this.VERTEXCOLOR = false;\n this.BAKED_VERTEX_ANIMATION_TEXTURE = false;\n this.AMBIENT = false;\n this.AMBIENTDIRECTUV = 0;\n this.AMBIENTINGRAYSCALE = false;\n this.OPACITY = false;\n this.VERTEXALPHA = false;\n this.OPACITYDIRECTUV = 0;\n this.OPACITYRGB = false;\n this.ALPHATEST = false;\n this.DEPTHPREPASS = false;\n this.ALPHABLEND = false;\n this.ALPHAFROMALBEDO = false;\n this.ALPHATESTVALUE = \"0.5\";\n this.SPECULAROVERALPHA = false;\n this.RADIANCEOVERALPHA = false;\n this.ALPHAFRESNEL = false;\n this.LINEARALPHAFRESNEL = false;\n this.PREMULTIPLYALPHA = false;\n this.EMISSIVE = false;\n this.EMISSIVEDIRECTUV = 0;\n this.GAMMAEMISSIVE = false;\n this.REFLECTIVITY = false;\n this.REFLECTIVITY_GAMMA = false;\n this.REFLECTIVITYDIRECTUV = 0;\n this.SPECULARTERM = false;\n this.MICROSURFACEFROMREFLECTIVITYMAP = false;\n this.MICROSURFACEAUTOMATIC = false;\n this.LODBASEDMICROSFURACE = false;\n this.MICROSURFACEMAP = false;\n this.MICROSURFACEMAPDIRECTUV = 0;\n this.METALLICWORKFLOW = false;\n this.ROUGHNESSSTOREINMETALMAPALPHA = false;\n this.ROUGHNESSSTOREINMETALMAPGREEN = false;\n this.METALLNESSSTOREINMETALMAPBLUE = false;\n this.AOSTOREINMETALMAPRED = false;\n this.METALLIC_REFLECTANCE = false;\n this.METALLIC_REFLECTANCE_GAMMA = false;\n this.METALLIC_REFLECTANCEDIRECTUV = 0;\n this.METALLIC_REFLECTANCE_USE_ALPHA_ONLY = false;\n this.REFLECTANCE = false;\n this.REFLECTANCE_GAMMA = false;\n this.REFLECTANCEDIRECTUV = 0;\n this.ENVIRONMENTBRDF = false;\n this.ENVIRONMENTBRDF_RGBD = false;\n this.NORMAL = false;\n this.TANGENT = false;\n this.BUMP = false;\n this.BUMPDIRECTUV = 0;\n this.OBJECTSPACE_NORMALMAP = false;\n this.PARALLAX = false;\n this.PARALLAXOCCLUSION = false;\n this.NORMALXYSCALE = true;\n this.LIGHTMAP = false;\n this.LIGHTMAPDIRECTUV = 0;\n this.USELIGHTMAPASSHADOWMAP = false;\n this.GAMMALIGHTMAP = false;\n this.RGBDLIGHTMAP = false;\n this.REFLECTION = false;\n this.REFLECTIONMAP_3D = false;\n this.REFLECTIONMAP_SPHERICAL = false;\n this.REFLECTIONMAP_PLANAR = false;\n this.REFLECTIONMAP_CUBIC = false;\n this.USE_LOCAL_REFLECTIONMAP_CUBIC = false;\n this.REFLECTIONMAP_PROJECTION = false;\n this.REFLECTIONMAP_SKYBOX = false;\n this.REFLECTIONMAP_EXPLICIT = false;\n this.REFLECTIONMAP_EQUIRECTANGULAR = false;\n this.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = false;\n this.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = false;\n this.INVERTCUBICMAP = false;\n this.USESPHERICALFROMREFLECTIONMAP = false;\n this.USEIRRADIANCEMAP = false;\n this.USESPHERICALINVERTEX = false;\n this.REFLECTIONMAP_OPPOSITEZ = false;\n this.LODINREFLECTIONALPHA = false;\n this.GAMMAREFLECTION = false;\n this.RGBDREFLECTION = false;\n this.LINEARSPECULARREFLECTION = false;\n this.RADIANCEOCCLUSION = false;\n this.HORIZONOCCLUSION = false;\n this.INSTANCES = false;\n this.THIN_INSTANCES = false;\n this.INSTANCESCOLOR = false;\n this.PREPASS = false;\n this.PREPASS_IRRADIANCE = false;\n this.PREPASS_IRRADIANCE_INDEX = -1;\n this.PREPASS_ALBEDO_SQRT = false;\n this.PREPASS_ALBEDO_SQRT_INDEX = -1;\n this.PREPASS_DEPTH = false;\n this.PREPASS_DEPTH_INDEX = -1;\n this.PREPASS_NORMAL = false;\n this.PREPASS_NORMAL_INDEX = -1;\n this.PREPASS_POSITION = false;\n this.PREPASS_POSITION_INDEX = -1;\n this.PREPASS_VELOCITY = false;\n this.PREPASS_VELOCITY_INDEX = -1;\n this.PREPASS_REFLECTIVITY = false;\n this.PREPASS_REFLECTIVITY_INDEX = -1;\n this.SCENE_MRT_COUNT = 0;\n this.NUM_BONE_INFLUENCERS = 0;\n this.BonesPerMesh = 0;\n this.BONETEXTURE = false;\n this.BONES_VELOCITY_ENABLED = false;\n this.NONUNIFORMSCALING = false;\n this.MORPHTARGETS = false;\n this.MORPHTARGETS_NORMAL = false;\n this.MORPHTARGETS_TANGENT = false;\n this.MORPHTARGETS_UV = false;\n this.NUM_MORPH_INFLUENCERS = 0;\n this.MORPHTARGETS_TEXTURE = false;\n this.IMAGEPROCESSING = false;\n this.VIGNETTE = false;\n this.VIGNETTEBLENDMODEMULTIPLY = false;\n this.VIGNETTEBLENDMODEOPAQUE = false;\n this.TONEMAPPING = false;\n this.TONEMAPPING_ACES = false;\n this.CONTRAST = false;\n this.COLORCURVES = false;\n this.COLORGRADING = false;\n this.COLORGRADING3D = false;\n this.SAMPLER3DGREENDEPTH = false;\n this.SAMPLER3DBGRMAP = false;\n this.DITHER = false;\n this.IMAGEPROCESSINGPOSTPROCESS = false;\n this.SKIPFINALCOLORCLAMP = false;\n this.EXPOSURE = false;\n this.MULTIVIEW = false;\n this.ORDER_INDEPENDENT_TRANSPARENCY = false;\n this.ORDER_INDEPENDENT_TRANSPARENCY_16BITS = false;\n this.USEPHYSICALLIGHTFALLOFF = false;\n this.USEGLTFLIGHTFALLOFF = false;\n this.TWOSIDEDLIGHTING = false;\n this.SHADOWFLOAT = false;\n this.CLIPPLANE = false;\n this.CLIPPLANE2 = false;\n this.CLIPPLANE3 = false;\n this.CLIPPLANE4 = false;\n this.CLIPPLANE5 = false;\n this.CLIPPLANE6 = false;\n this.POINTSIZE = false;\n this.FOG = false;\n this.LOGARITHMICDEPTH = false;\n this.CAMERA_ORTHOGRAPHIC = false;\n this.CAMERA_PERSPECTIVE = false;\n this.FORCENORMALFORWARD = false;\n this.SPECULARAA = false;\n this.UNLIT = false;\n this.DECAL_AFTER_DETAIL = false;\n this.DEBUGMODE = 0;\n this.rebuild();\n }\n /**\n * Resets the PBR Material defines.\n */\n reset() {\n super.reset();\n this.ALPHATESTVALUE = \"0.5\";\n this.PBR = true;\n this.NORMALXYSCALE = true;\n }\n }\n /**\n * The Physically based material base class of BJS.\n *\n * This offers the main features of a standard PBR material.\n * For more information, please refer to the documentation :\n * https://doc.babylonjs.com/features/featuresDeepDive/materials/using/introToPBR\n */\n class PBRBaseMaterial extends PushMaterial {\n /**\n * Enables realtime filtering on the texture.\n */\n get realTimeFiltering() {\n return this._realTimeFiltering;\n }\n set realTimeFiltering(b) {\n this._realTimeFiltering = b;\n this.markAsDirty(1);\n }\n /**\n * Quality switch for realtime filtering\n */\n get realTimeFilteringQuality() {\n return this._realTimeFilteringQuality;\n }\n set realTimeFilteringQuality(n) {\n this._realTimeFilteringQuality = n;\n this.markAsDirty(1);\n }\n /**\n * Can this material render to several textures at once\n */\n get canRenderToMRT() {\n return true;\n }\n /**\n * Attaches a new image processing configuration to the PBR Material.\n * @param configuration\n */\n _attachImageProcessingConfiguration(configuration) {\n if (configuration === this._imageProcessingConfiguration) {\n return;\n }\n // Detaches observer.\n if (this._imageProcessingConfiguration && this._imageProcessingObserver) {\n this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver);\n }\n // Pick the scene configuration if needed.\n if (!configuration) {\n this._imageProcessingConfiguration = this.getScene().imageProcessingConfiguration;\n }\n else {\n this._imageProcessingConfiguration = configuration;\n }\n // Attaches observer.\n if (this._imageProcessingConfiguration) {\n this._imageProcessingObserver = this._imageProcessingConfiguration.onUpdateParameters.add(() => {\n this._markAllSubMeshesAsImageProcessingDirty();\n });\n }\n }\n /**\n * Instantiates a new PBRMaterial instance.\n *\n * @param name The material name\n * @param scene The scene the material will be use in.\n */\n constructor(name, scene) {\n super(name, scene);\n /**\n * Intensity of the direct lights e.g. the four lights available in your scene.\n * This impacts both the direct diffuse and specular highlights.\n * @internal\n */\n this._directIntensity = 1.0;\n /**\n * Intensity of the emissive part of the material.\n * This helps controlling the emissive effect without modifying the emissive color.\n * @internal\n */\n this._emissiveIntensity = 1.0;\n /**\n * Intensity of the environment e.g. how much the environment will light the object\n * either through harmonics for rough material or through the reflection for shiny ones.\n * @internal\n */\n this._environmentIntensity = 1.0;\n /**\n * This is a special control allowing the reduction of the specular highlights coming from the\n * four lights of the scene. Those highlights may not be needed in full environment lighting.\n * @internal\n */\n this._specularIntensity = 1.0;\n /**\n * This stores the direct, emissive, environment, and specular light intensities into a Vector4.\n */\n this._lightingInfos = new Vector4(this._directIntensity, this._emissiveIntensity, this._environmentIntensity, this._specularIntensity);\n /**\n * Debug Control allowing disabling the bump map on this material.\n * @internal\n */\n this._disableBumpMap = false;\n /**\n * AKA Diffuse Texture in standard nomenclature.\n * @internal\n */\n this._albedoTexture = null;\n /**\n * AKA Occlusion Texture in other nomenclature.\n * @internal\n */\n this._ambientTexture = null;\n /**\n * AKA Occlusion Texture Intensity in other nomenclature.\n * @internal\n */\n this._ambientTextureStrength = 1.0;\n /**\n * Defines how much the AO map is occluding the analytical lights (point spot...).\n * 1 means it completely occludes it\n * 0 mean it has no impact\n * @internal\n */\n this._ambientTextureImpactOnAnalyticalLights = PBRBaseMaterial.DEFAULT_AO_ON_ANALYTICAL_LIGHTS;\n /**\n * Stores the alpha values in a texture.\n * @internal\n */\n this._opacityTexture = null;\n /**\n * Stores the reflection values in a texture.\n * @internal\n */\n this._reflectionTexture = null;\n /**\n * Stores the emissive values in a texture.\n * @internal\n */\n this._emissiveTexture = null;\n /**\n * AKA Specular texture in other nomenclature.\n * @internal\n */\n this._reflectivityTexture = null;\n /**\n * Used to switch from specular/glossiness to metallic/roughness workflow.\n * @internal\n */\n this._metallicTexture = null;\n /**\n * Specifies the metallic scalar of the metallic/roughness workflow.\n * Can also be used to scale the metalness values of the metallic texture.\n * @internal\n */\n this._metallic = null;\n /**\n * Specifies the roughness scalar of the metallic/roughness workflow.\n * Can also be used to scale the roughness values of the metallic texture.\n * @internal\n */\n this._roughness = null;\n /**\n * In metallic workflow, specifies an F0 factor to help configuring the material F0.\n * By default the indexOfrefraction is used to compute F0;\n *\n * This is used as a factor against the default reflectance at normal incidence to tweak it.\n *\n * F0 = defaultF0 * metallicF0Factor * metallicReflectanceColor;\n * F90 = metallicReflectanceColor;\n * @internal\n */\n this._metallicF0Factor = 1;\n /**\n * In metallic workflow, specifies an F0 color.\n * By default the F90 is always 1;\n *\n * Please note that this factor is also used as a factor against the default reflectance at normal incidence.\n *\n * F0 = defaultF0_from_IOR * metallicF0Factor * metallicReflectanceColor\n * F90 = metallicF0Factor;\n * @internal\n */\n this._metallicReflectanceColor = Color3.White();\n /**\n * Specifies that only the A channel from _metallicReflectanceTexture should be used.\n * If false, both RGB and A channels will be used\n * @internal\n */\n this._useOnlyMetallicFromMetallicReflectanceTexture = false;\n /**\n * Defines to store metallicReflectanceColor in RGB and metallicF0Factor in A\n * This is multiply against the scalar values defined in the material.\n * @internal\n */\n this._metallicReflectanceTexture = null;\n /**\n * Defines to store reflectanceColor in RGB\n * This is multiplied against the scalar values defined in the material.\n * If both _reflectanceTexture and _metallicReflectanceTexture textures are provided and _useOnlyMetallicFromMetallicReflectanceTexture\n * is false, _metallicReflectanceTexture takes precedence and _reflectanceTexture is not used\n * @internal\n */\n this._reflectanceTexture = null;\n /**\n * Used to enable roughness/glossiness fetch from a separate channel depending on the current mode.\n * Gray Scale represents roughness in metallic mode and glossiness in specular mode.\n * @internal\n */\n this._microSurfaceTexture = null;\n /**\n * Stores surface normal data used to displace a mesh in a texture.\n * @internal\n */\n this._bumpTexture = null;\n /**\n * Stores the pre-calculated light information of a mesh in a texture.\n * @internal\n */\n this._lightmapTexture = null;\n /**\n * The color of a material in ambient lighting.\n * @internal\n */\n this._ambientColor = new Color3(0, 0, 0);\n /**\n * AKA Diffuse Color in other nomenclature.\n * @internal\n */\n this._albedoColor = new Color3(1, 1, 1);\n /**\n * AKA Specular Color in other nomenclature.\n * @internal\n */\n this._reflectivityColor = new Color3(1, 1, 1);\n /**\n * The color applied when light is reflected from a material.\n * @internal\n */\n this._reflectionColor = new Color3(1, 1, 1);\n /**\n * The color applied when light is emitted from a material.\n * @internal\n */\n this._emissiveColor = new Color3(0, 0, 0);\n /**\n * AKA Glossiness in other nomenclature.\n * @internal\n */\n this._microSurface = 0.9;\n /**\n * Specifies that the material will use the light map as a show map.\n * @internal\n */\n this._useLightmapAsShadowmap = false;\n /**\n * This parameters will enable/disable Horizon occlusion to prevent normal maps to look shiny when the normal\n * makes the reflect vector face the model (under horizon).\n * @internal\n */\n this._useHorizonOcclusion = true;\n /**\n * This parameters will enable/disable radiance occlusion by preventing the radiance to lit\n * too much the area relying on ambient texture to define their ambient occlusion.\n * @internal\n */\n this._useRadianceOcclusion = true;\n /**\n * Specifies that the alpha is coming form the albedo channel alpha channel for alpha blending.\n * @internal\n */\n this._useAlphaFromAlbedoTexture = false;\n /**\n * Specifies that the material will keeps the specular highlights over a transparent surface (only the most luminous ones).\n * A car glass is a good example of that. When sun reflects on it you can not see what is behind.\n * @internal\n */\n this._useSpecularOverAlpha = true;\n /**\n * Specifies if the reflectivity texture contains the glossiness information in its alpha channel.\n * @internal\n */\n this._useMicroSurfaceFromReflectivityMapAlpha = false;\n /**\n * Specifies if the metallic texture contains the roughness information in its alpha channel.\n * @internal\n */\n this._useRoughnessFromMetallicTextureAlpha = true;\n /**\n * Specifies if the metallic texture contains the roughness information in its green channel.\n * @internal\n */\n this._useRoughnessFromMetallicTextureGreen = false;\n /**\n * Specifies if the metallic texture contains the metallness information in its blue channel.\n * @internal\n */\n this._useMetallnessFromMetallicTextureBlue = false;\n /**\n * Specifies if the metallic texture contains the ambient occlusion information in its red channel.\n * @internal\n */\n this._useAmbientOcclusionFromMetallicTextureRed = false;\n /**\n * Specifies if the ambient texture contains the ambient occlusion information in its red channel only.\n * @internal\n */\n this._useAmbientInGrayScale = false;\n /**\n * In case the reflectivity map does not contain the microsurface information in its alpha channel,\n * The material will try to infer what glossiness each pixel should be.\n * @internal\n */\n this._useAutoMicroSurfaceFromReflectivityMap = false;\n /**\n * Defines the falloff type used in this material.\n * It by default is Physical.\n * @internal\n */\n this._lightFalloff = PBRBaseMaterial.LIGHTFALLOFF_PHYSICAL;\n /**\n * Specifies that the material will keeps the reflection highlights over a transparent surface (only the most luminous ones).\n * A car glass is a good example of that. When the street lights reflects on it you can not see what is behind.\n * @internal\n */\n this._useRadianceOverAlpha = true;\n /**\n * Allows using an object space normal map (instead of tangent space).\n * @internal\n */\n this._useObjectSpaceNormalMap = false;\n /**\n * Allows using the bump map in parallax mode.\n * @internal\n */\n this._useParallax = false;\n /**\n * Allows using the bump map in parallax occlusion mode.\n * @internal\n */\n this._useParallaxOcclusion = false;\n /**\n * Controls the scale bias of the parallax mode.\n * @internal\n */\n this._parallaxScaleBias = 0.05;\n /**\n * If sets to true, disables all the lights affecting the material.\n * @internal\n */\n this._disableLighting = false;\n /**\n * Number of Simultaneous lights allowed on the material.\n * @internal\n */\n this._maxSimultaneousLights = 4;\n /**\n * If sets to true, x component of normal map value will be inverted (x = 1.0 - x).\n * @internal\n */\n this._invertNormalMapX = false;\n /**\n * If sets to true, y component of normal map value will be inverted (y = 1.0 - y).\n * @internal\n */\n this._invertNormalMapY = false;\n /**\n * If sets to true and backfaceCulling is false, normals will be flipped on the backside.\n * @internal\n */\n this._twoSidedLighting = false;\n /**\n * Defines the alpha limits in alpha test mode.\n * @internal\n */\n this._alphaCutOff = 0.4;\n /**\n * Enforces alpha test in opaque or blend mode in order to improve the performances of some situations.\n * @internal\n */\n this._forceAlphaTest = false;\n /**\n * A fresnel is applied to the alpha of the model to ensure grazing angles edges are not alpha tested.\n * And/Or occlude the blended part. (alpha is converted to gamma to compute the fresnel)\n * @internal\n */\n this._useAlphaFresnel = false;\n /**\n * A fresnel is applied to the alpha of the model to ensure grazing angles edges are not alpha tested.\n * And/Or occlude the blended part. (alpha stays linear to compute the fresnel)\n * @internal\n */\n this._useLinearAlphaFresnel = false;\n /**\n * Specifies the environment BRDF texture used to compute the scale and offset roughness values\n * from cos theta and roughness:\n * http://blog.selfshadow.com/publications/s2013-shading-course/karis/s2013_pbs_epic_notes_v2.pdf\n * @internal\n */\n this._environmentBRDFTexture = null;\n /**\n * Force the shader to compute irradiance in the fragment shader in order to take bump in account.\n * @internal\n */\n this._forceIrradianceInFragment = false;\n this._realTimeFiltering = false;\n this._realTimeFilteringQuality = 8;\n /**\n * Force normal to face away from face.\n * @internal\n */\n this._forceNormalForward = false;\n /**\n * Enables specular anti aliasing in the PBR shader.\n * It will both interacts on the Geometry for analytical and IBL lighting.\n * It also prefilter the roughness map based on the bump values.\n * @internal\n */\n this._enableSpecularAntiAliasing = false;\n /**\n * Keep track of the image processing observer to allow dispose and replace.\n */\n this._imageProcessingObserver = null;\n /**\n * Stores the available render targets.\n */\n this._renderTargets = new SmartArray(16);\n /**\n * Sets the global ambient color for the material used in lighting calculations.\n */\n this._globalAmbientColor = new Color3(0, 0, 0);\n /**\n * Enables the use of logarithmic depth buffers, which is good for wide depth buffers.\n */\n this._useLogarithmicDepth = false;\n /**\n * If set to true, no lighting calculations will be applied.\n */\n this._unlit = false;\n /**\n * If sets to true, the decal map will be applied after the detail map. Else, it is applied before (default: false)\n */\n this._applyDecalMapAfterDetailMap = false;\n this._debugMode = 0;\n /**\n * @internal\n * This is reserved for the inspector.\n * Defines the material debug mode.\n * It helps seeing only some components of the material while troubleshooting.\n */\n this.debugMode = 0;\n /**\n * @internal\n * This is reserved for the inspector.\n * Specify from where on screen the debug mode should start.\n * The value goes from -1 (full screen) to 1 (not visible)\n * It helps with side by side comparison against the final render\n * This defaults to -1\n */\n this.debugLimit = -1;\n /**\n * @internal\n * This is reserved for the inspector.\n * As the default viewing range might not be enough (if the ambient is really small for instance)\n * You can use the factor to better multiply the final value.\n */\n this.debugFactor = 1;\n this._cacheHasRenderTargetTextures = false;\n this.brdf = new PBRBRDFConfiguration(this);\n this.clearCoat = new PBRClearCoatConfiguration(this);\n this.iridescence = new PBRIridescenceConfiguration(this);\n this.anisotropy = new PBRAnisotropicConfiguration(this);\n this.sheen = new PBRSheenConfiguration(this);\n this.subSurface = new PBRSubSurfaceConfiguration(this);\n this.detailMap = new DetailMapConfiguration(this);\n // Setup the default processing configuration to the scene.\n this._attachImageProcessingConfiguration(null);\n this.getRenderTargetTextures = () => {\n this._renderTargets.reset();\n if (MaterialFlags.ReflectionTextureEnabled && this._reflectionTexture && this._reflectionTexture.isRenderTarget) {\n this._renderTargets.push(this._reflectionTexture);\n }\n this._eventInfo.renderTargets = this._renderTargets;\n this._callbackPluginEventFillRenderTargetTextures(this._eventInfo);\n return this._renderTargets;\n };\n this._environmentBRDFTexture = GetEnvironmentBRDFTexture(this.getScene());\n this.prePassConfiguration = new PrePassConfiguration();\n }\n /**\n * Gets a boolean indicating that current material needs to register RTT\n */\n get hasRenderTargetTextures() {\n if (MaterialFlags.ReflectionTextureEnabled && this._reflectionTexture && this._reflectionTexture.isRenderTarget) {\n return true;\n }\n return this._cacheHasRenderTargetTextures;\n }\n /**\n * Can this material render to prepass\n */\n get isPrePassCapable() {\n return !this.disableDepthWrite;\n }\n /**\n * Gets the name of the material class.\n */\n getClassName() {\n return \"PBRBaseMaterial\";\n }\n /**\n * Enabled the use of logarithmic depth buffers, which is good for wide depth buffers.\n */\n get useLogarithmicDepth() {\n return this._useLogarithmicDepth;\n }\n /**\n * Enabled the use of logarithmic depth buffers, which is good for wide depth buffers.\n */\n set useLogarithmicDepth(value) {\n this._useLogarithmicDepth = value && this.getScene().getEngine().getCaps().fragmentDepthSupported;\n }\n /**\n * Returns true if alpha blending should be disabled.\n */\n get _disableAlphaBlending() {\n var _a;\n return (this._transparencyMode === PBRBaseMaterial.PBRMATERIAL_OPAQUE ||\n this._transparencyMode === PBRBaseMaterial.PBRMATERIAL_ALPHATEST ||\n ((_a = this.subSurface) === null || _a === void 0 ? void 0 : _a.disableAlphaBlending));\n }\n /**\n * Specifies whether or not this material should be rendered in alpha blend mode.\n */\n needAlphaBlending() {\n if (this._disableAlphaBlending) {\n return false;\n }\n return this.alpha < 1.0 || this._opacityTexture != null || this._shouldUseAlphaFromAlbedoTexture();\n }\n /**\n * Specifies whether or not this material should be rendered in alpha test mode.\n */\n needAlphaTesting() {\n var _a;\n if (this._forceAlphaTest) {\n return true;\n }\n if ((_a = this.subSurface) === null || _a === void 0 ? void 0 : _a.disableAlphaBlending) {\n return false;\n }\n return this._hasAlphaChannel() && (this._transparencyMode == null || this._transparencyMode === PBRBaseMaterial.PBRMATERIAL_ALPHATEST);\n }\n /**\n * Specifies whether or not the alpha value of the albedo texture should be used for alpha blending.\n */\n _shouldUseAlphaFromAlbedoTexture() {\n return this._albedoTexture != null && this._albedoTexture.hasAlpha && this._useAlphaFromAlbedoTexture && this._transparencyMode !== PBRBaseMaterial.PBRMATERIAL_OPAQUE;\n }\n /**\n * Specifies whether or not there is a usable alpha channel for transparency.\n */\n _hasAlphaChannel() {\n return (this._albedoTexture != null && this._albedoTexture.hasAlpha) || this._opacityTexture != null;\n }\n /**\n * Gets the texture used for the alpha test.\n */\n getAlphaTestTexture() {\n return this._albedoTexture;\n }\n /**\n * Specifies that the submesh is ready to be used.\n * @param mesh - BJS mesh.\n * @param subMesh - A submesh of the BJS mesh. Used to check if it is ready.\n * @param useInstances - Specifies that instances should be used.\n * @returns - boolean indicating that the submesh is ready or not.\n */\n isReadyForSubMesh(mesh, subMesh, useInstances) {\n var _a;\n if (!this._uniformBufferLayoutBuilt) {\n this.buildUniformLayout();\n }\n if (subMesh.effect && this.isFrozen) {\n if (subMesh.effect._wasPreviouslyReady && subMesh.effect._wasPreviouslyUsingInstances === useInstances) {\n return true;\n }\n }\n if (!subMesh.materialDefines) {\n this._callbackPluginEventGeneric(MaterialPluginEvent.GetDefineNames, this._eventInfo);\n subMesh.materialDefines = new PBRMaterialDefines(this._eventInfo.defineNames);\n }\n const defines = subMesh.materialDefines;\n if (this._isReadyForSubMesh(subMesh)) {\n return true;\n }\n const scene = this.getScene();\n const engine = scene.getEngine();\n if (defines._areTexturesDirty) {\n this._eventInfo.hasRenderTargetTextures = false;\n this._callbackPluginEventHasRenderTargetTextures(this._eventInfo);\n this._cacheHasRenderTargetTextures = this._eventInfo.hasRenderTargetTextures;\n if (scene.texturesEnabled) {\n if (this._albedoTexture && MaterialFlags.DiffuseTextureEnabled) {\n if (!this._albedoTexture.isReadyOrNotBlocking()) {\n return false;\n }\n }\n if (this._ambientTexture && MaterialFlags.AmbientTextureEnabled) {\n if (!this._ambientTexture.isReadyOrNotBlocking()) {\n return false;\n }\n }\n if (this._opacityTexture && MaterialFlags.OpacityTextureEnabled) {\n if (!this._opacityTexture.isReadyOrNotBlocking()) {\n return false;\n }\n }\n const reflectionTexture = this._getReflectionTexture();\n if (reflectionTexture && MaterialFlags.ReflectionTextureEnabled) {\n if (!reflectionTexture.isReadyOrNotBlocking()) {\n return false;\n }\n if (reflectionTexture.irradianceTexture) {\n if (!reflectionTexture.irradianceTexture.isReadyOrNotBlocking()) {\n return false;\n }\n }\n else {\n // Not ready until spherical are ready too.\n if (!reflectionTexture.sphericalPolynomial && ((_a = reflectionTexture.getInternalTexture()) === null || _a === void 0 ? void 0 : _a._sphericalPolynomialPromise)) {\n return false;\n }\n }\n }\n if (this._lightmapTexture && MaterialFlags.LightmapTextureEnabled) {\n if (!this._lightmapTexture.isReadyOrNotBlocking()) {\n return false;\n }\n }\n if (this._emissiveTexture && MaterialFlags.EmissiveTextureEnabled) {\n if (!this._emissiveTexture.isReadyOrNotBlocking()) {\n return false;\n }\n }\n if (MaterialFlags.SpecularTextureEnabled) {\n if (this._metallicTexture) {\n if (!this._metallicTexture.isReadyOrNotBlocking()) {\n return false;\n }\n }\n else if (this._reflectivityTexture) {\n if (!this._reflectivityTexture.isReadyOrNotBlocking()) {\n return false;\n }\n }\n if (this._metallicReflectanceTexture) {\n if (!this._metallicReflectanceTexture.isReadyOrNotBlocking()) {\n return false;\n }\n }\n if (this._reflectanceTexture) {\n if (!this._reflectanceTexture.isReadyOrNotBlocking()) {\n return false;\n }\n }\n if (this._microSurfaceTexture) {\n if (!this._microSurfaceTexture.isReadyOrNotBlocking()) {\n return false;\n }\n }\n }\n if (engine.getCaps().standardDerivatives && this._bumpTexture && MaterialFlags.BumpTextureEnabled && !this._disableBumpMap) {\n // Bump texture cannot be not blocking.\n if (!this._bumpTexture.isReady()) {\n return false;\n }\n }\n if (this._environmentBRDFTexture && MaterialFlags.ReflectionTextureEnabled) {\n // This is blocking.\n if (!this._environmentBRDFTexture.isReady()) {\n return false;\n }\n }\n }\n }\n this._eventInfo.isReadyForSubMesh = true;\n this._eventInfo.defines = defines;\n this._eventInfo.subMesh = subMesh;\n this._callbackPluginEventIsReadyForSubMesh(this._eventInfo);\n if (!this._eventInfo.isReadyForSubMesh) {\n return false;\n }\n if (defines._areImageProcessingDirty && this._imageProcessingConfiguration) {\n if (!this._imageProcessingConfiguration.isReady()) {\n return false;\n }\n }\n if (!engine.getCaps().standardDerivatives && !mesh.isVerticesDataPresent(VertexBuffer.NormalKind)) {\n mesh.createNormals(true);\n Logger.Warn(\"PBRMaterial: Normals have been created for the mesh: \" + mesh.name);\n }\n const previousEffect = subMesh.effect;\n const lightDisposed = defines._areLightsDisposed;\n let effect = this._prepareEffect(mesh, defines, this.onCompiled, this.onError, useInstances, null, subMesh.getRenderingMesh().hasThinInstances);\n let forceWasNotReadyPreviously = false;\n if (effect) {\n if (this._onEffectCreatedObservable) {\n onCreatedEffectParameters$1.effect = effect;\n onCreatedEffectParameters$1.subMesh = subMesh;\n this._onEffectCreatedObservable.notifyObservers(onCreatedEffectParameters$1);\n }\n // Use previous effect while new one is compiling\n if (this.allowShaderHotSwapping && previousEffect && !effect.isReady()) {\n effect = previousEffect;\n defines.markAsUnprocessed();\n forceWasNotReadyPreviously = this.isFrozen;\n if (lightDisposed) {\n // re register in case it takes more than one frame.\n defines._areLightsDisposed = true;\n return false;\n }\n }\n else {\n scene.resetCachedMaterial();\n subMesh.setEffect(effect, defines, this._materialContext);\n }\n }\n if (!subMesh.effect || !subMesh.effect.isReady()) {\n return false;\n }\n defines._renderId = scene.getRenderId();\n subMesh.effect._wasPreviouslyReady = forceWasNotReadyPreviously ? false : true;\n subMesh.effect._wasPreviouslyUsingInstances = !!useInstances;\n this._checkScenePerformancePriority();\n return true;\n }\n /**\n * Specifies if the material uses metallic roughness workflow.\n * @returns boolean specifying if the material uses metallic roughness workflow.\n */\n isMetallicWorkflow() {\n if (this._metallic != null || this._roughness != null || this._metallicTexture) {\n return true;\n }\n return false;\n }\n _prepareEffect(mesh, defines, onCompiled = null, onError = null, useInstances = null, useClipPlane = null, useThinInstances) {\n this._prepareDefines(mesh, defines, useInstances, useClipPlane, useThinInstances);\n if (!defines.isDirty) {\n return null;\n }\n defines.markAsProcessed();\n const scene = this.getScene();\n const engine = scene.getEngine();\n // Fallbacks\n const fallbacks = new EffectFallbacks();\n let fallbackRank = 0;\n if (defines.USESPHERICALINVERTEX) {\n fallbacks.addFallback(fallbackRank++, \"USESPHERICALINVERTEX\");\n }\n if (defines.FOG) {\n fallbacks.addFallback(fallbackRank, \"FOG\");\n }\n if (defines.SPECULARAA) {\n fallbacks.addFallback(fallbackRank, \"SPECULARAA\");\n }\n if (defines.POINTSIZE) {\n fallbacks.addFallback(fallbackRank, \"POINTSIZE\");\n }\n if (defines.LOGARITHMICDEPTH) {\n fallbacks.addFallback(fallbackRank, \"LOGARITHMICDEPTH\");\n }\n if (defines.PARALLAX) {\n fallbacks.addFallback(fallbackRank, \"PARALLAX\");\n }\n if (defines.PARALLAXOCCLUSION) {\n fallbacks.addFallback(fallbackRank++, \"PARALLAXOCCLUSION\");\n }\n if (defines.ENVIRONMENTBRDF) {\n fallbacks.addFallback(fallbackRank++, \"ENVIRONMENTBRDF\");\n }\n if (defines.TANGENT) {\n fallbacks.addFallback(fallbackRank++, \"TANGENT\");\n }\n if (defines.BUMP) {\n fallbacks.addFallback(fallbackRank++, \"BUMP\");\n }\n fallbackRank = MaterialHelper.HandleFallbacksForShadows(defines, fallbacks, this._maxSimultaneousLights, fallbackRank++);\n if (defines.SPECULARTERM) {\n fallbacks.addFallback(fallbackRank++, \"SPECULARTERM\");\n }\n if (defines.USESPHERICALFROMREFLECTIONMAP) {\n fallbacks.addFallback(fallbackRank++, \"USESPHERICALFROMREFLECTIONMAP\");\n }\n if (defines.USEIRRADIANCEMAP) {\n fallbacks.addFallback(fallbackRank++, \"USEIRRADIANCEMAP\");\n }\n if (defines.LIGHTMAP) {\n fallbacks.addFallback(fallbackRank++, \"LIGHTMAP\");\n }\n if (defines.NORMAL) {\n fallbacks.addFallback(fallbackRank++, \"NORMAL\");\n }\n if (defines.AMBIENT) {\n fallbacks.addFallback(fallbackRank++, \"AMBIENT\");\n }\n if (defines.EMISSIVE) {\n fallbacks.addFallback(fallbackRank++, \"EMISSIVE\");\n }\n if (defines.VERTEXCOLOR) {\n fallbacks.addFallback(fallbackRank++, \"VERTEXCOLOR\");\n }\n if (defines.MORPHTARGETS) {\n fallbacks.addFallback(fallbackRank++, \"MORPHTARGETS\");\n }\n if (defines.MULTIVIEW) {\n fallbacks.addFallback(0, \"MULTIVIEW\");\n }\n //Attributes\n const attribs = [VertexBuffer.PositionKind];\n if (defines.NORMAL) {\n attribs.push(VertexBuffer.NormalKind);\n }\n if (defines.TANGENT) {\n attribs.push(VertexBuffer.TangentKind);\n }\n for (let i = 1; i <= 6; ++i) {\n if (defines[\"UV\" + i]) {\n attribs.push(`uv${i === 1 ? \"\" : i}`);\n }\n }\n if (defines.VERTEXCOLOR) {\n attribs.push(VertexBuffer.ColorKind);\n }\n if (defines.INSTANCESCOLOR) {\n attribs.push(VertexBuffer.ColorInstanceKind);\n }\n MaterialHelper.PrepareAttributesForBones(attribs, mesh, defines, fallbacks);\n MaterialHelper.PrepareAttributesForInstances(attribs, defines);\n MaterialHelper.PrepareAttributesForMorphTargets(attribs, mesh, defines);\n MaterialHelper.PrepareAttributesForBakedVertexAnimation(attribs, mesh, defines);\n let shaderName = \"pbr\";\n const uniforms = [\n \"world\",\n \"view\",\n \"viewProjection\",\n \"vEyePosition\",\n \"vLightsType\",\n \"vAmbientColor\",\n \"vAlbedoColor\",\n \"vReflectivityColor\",\n \"vMetallicReflectanceFactors\",\n \"vEmissiveColor\",\n \"visibility\",\n \"vReflectionColor\",\n \"vFogInfos\",\n \"vFogColor\",\n \"pointSize\",\n \"vAlbedoInfos\",\n \"vAmbientInfos\",\n \"vOpacityInfos\",\n \"vReflectionInfos\",\n \"vReflectionPosition\",\n \"vReflectionSize\",\n \"vEmissiveInfos\",\n \"vReflectivityInfos\",\n \"vReflectionFilteringInfo\",\n \"vMetallicReflectanceInfos\",\n \"vReflectanceInfos\",\n \"vMicroSurfaceSamplerInfos\",\n \"vBumpInfos\",\n \"vLightmapInfos\",\n \"mBones\",\n \"albedoMatrix\",\n \"ambientMatrix\",\n \"opacityMatrix\",\n \"reflectionMatrix\",\n \"emissiveMatrix\",\n \"reflectivityMatrix\",\n \"normalMatrix\",\n \"microSurfaceSamplerMatrix\",\n \"bumpMatrix\",\n \"lightmapMatrix\",\n \"metallicReflectanceMatrix\",\n \"reflectanceMatrix\",\n \"vLightingIntensity\",\n \"logarithmicDepthConstant\",\n \"vSphericalX\",\n \"vSphericalY\",\n \"vSphericalZ\",\n \"vSphericalXX_ZZ\",\n \"vSphericalYY_ZZ\",\n \"vSphericalZZ\",\n \"vSphericalXY\",\n \"vSphericalYZ\",\n \"vSphericalZX\",\n \"vSphericalL00\",\n \"vSphericalL1_1\",\n \"vSphericalL10\",\n \"vSphericalL11\",\n \"vSphericalL2_2\",\n \"vSphericalL2_1\",\n \"vSphericalL20\",\n \"vSphericalL21\",\n \"vSphericalL22\",\n \"vReflectionMicrosurfaceInfos\",\n \"vTangentSpaceParams\",\n \"boneTextureWidth\",\n \"vDebugMode\",\n \"morphTargetTextureInfo\",\n \"morphTargetTextureIndices\",\n ];\n const samplers = [\n \"albedoSampler\",\n \"reflectivitySampler\",\n \"ambientSampler\",\n \"emissiveSampler\",\n \"bumpSampler\",\n \"lightmapSampler\",\n \"opacitySampler\",\n \"reflectionSampler\",\n \"reflectionSamplerLow\",\n \"reflectionSamplerHigh\",\n \"irradianceSampler\",\n \"microSurfaceSampler\",\n \"environmentBrdfSampler\",\n \"boneSampler\",\n \"metallicReflectanceSampler\",\n \"reflectanceSampler\",\n \"morphTargets\",\n \"oitDepthSampler\",\n \"oitFrontColorSampler\",\n ];\n const uniformBuffers = [\"Material\", \"Scene\", \"Mesh\"];\n const indexParameters = { maxSimultaneousLights: this._maxSimultaneousLights, maxSimultaneousMorphTargets: defines.NUM_MORPH_INFLUENCERS };\n this._eventInfo.fallbacks = fallbacks;\n this._eventInfo.fallbackRank = fallbackRank;\n this._eventInfo.defines = defines;\n this._eventInfo.uniforms = uniforms;\n this._eventInfo.attributes = attribs;\n this._eventInfo.samplers = samplers;\n this._eventInfo.uniformBuffersNames = uniformBuffers;\n this._eventInfo.customCode = undefined;\n this._eventInfo.mesh = mesh;\n this._eventInfo.indexParameters = indexParameters;\n this._callbackPluginEventGeneric(MaterialPluginEvent.PrepareEffect, this._eventInfo);\n PrePassConfiguration.AddUniforms(uniforms);\n PrePassConfiguration.AddSamplers(samplers);\n addClipPlaneUniforms(uniforms);\n if (ImageProcessingConfiguration) {\n ImageProcessingConfiguration.PrepareUniforms(uniforms, defines);\n ImageProcessingConfiguration.PrepareSamplers(samplers, defines);\n }\n MaterialHelper.PrepareUniformsAndSamplersList({\n uniformsNames: uniforms,\n uniformBuffersNames: uniformBuffers,\n samplers: samplers,\n defines: defines,\n maxSimultaneousLights: this._maxSimultaneousLights,\n });\n const csnrOptions = {};\n if (this.customShaderNameResolve) {\n shaderName = this.customShaderNameResolve(shaderName, uniforms, uniformBuffers, samplers, defines, attribs, csnrOptions);\n }\n const join = defines.toString();\n const effect = engine.createEffect(shaderName, {\n attributes: attribs,\n uniformsNames: uniforms,\n uniformBuffersNames: uniformBuffers,\n samplers: samplers,\n defines: join,\n fallbacks: fallbacks,\n onCompiled: onCompiled,\n onError: onError,\n indexParameters,\n processFinalCode: csnrOptions.processFinalCode,\n processCodeAfterIncludes: this._eventInfo.customCode,\n multiTarget: defines.PREPASS,\n }, engine);\n this._eventInfo.customCode = undefined;\n return effect;\n }\n _prepareDefines(mesh, defines, useInstances = null, useClipPlane = null, useThinInstances = false) {\n var _a;\n const scene = this.getScene();\n const engine = scene.getEngine();\n // Lights\n MaterialHelper.PrepareDefinesForLights(scene, mesh, defines, true, this._maxSimultaneousLights, this._disableLighting);\n defines._needNormals = true;\n // Multiview\n MaterialHelper.PrepareDefinesForMultiview(scene, defines);\n // PrePass\n const oit = this.needAlphaBlendingForMesh(mesh) && this.getScene().useOrderIndependentTransparency;\n MaterialHelper.PrepareDefinesForPrePass(scene, defines, this.canRenderToMRT && !oit);\n // Order independant transparency\n MaterialHelper.PrepareDefinesForOIT(scene, defines, oit);\n // Textures\n defines.METALLICWORKFLOW = this.isMetallicWorkflow();\n if (defines._areTexturesDirty) {\n defines._needUVs = false;\n for (let i = 1; i <= 6; ++i) {\n defines[\"MAINUV\" + i] = false;\n }\n if (scene.texturesEnabled) {\n defines.ALBEDODIRECTUV = 0;\n defines.AMBIENTDIRECTUV = 0;\n defines.OPACITYDIRECTUV = 0;\n defines.EMISSIVEDIRECTUV = 0;\n defines.REFLECTIVITYDIRECTUV = 0;\n defines.MICROSURFACEMAPDIRECTUV = 0;\n defines.METALLIC_REFLECTANCEDIRECTUV = 0;\n defines.REFLECTANCEDIRECTUV = 0;\n defines.BUMPDIRECTUV = 0;\n defines.LIGHTMAPDIRECTUV = 0;\n if (engine.getCaps().textureLOD) {\n defines.LODBASEDMICROSFURACE = true;\n }\n if (this._albedoTexture && MaterialFlags.DiffuseTextureEnabled) {\n MaterialHelper.PrepareDefinesForMergedUV(this._albedoTexture, defines, \"ALBEDO\");\n defines.GAMMAALBEDO = this._albedoTexture.gammaSpace;\n }\n else {\n defines.ALBEDO = false;\n }\n if (this._ambientTexture && MaterialFlags.AmbientTextureEnabled) {\n MaterialHelper.PrepareDefinesForMergedUV(this._ambientTexture, defines, \"AMBIENT\");\n defines.AMBIENTINGRAYSCALE = this._useAmbientInGrayScale;\n }\n else {\n defines.AMBIENT = false;\n }\n if (this._opacityTexture && MaterialFlags.OpacityTextureEnabled) {\n MaterialHelper.PrepareDefinesForMergedUV(this._opacityTexture, defines, \"OPACITY\");\n defines.OPACITYRGB = this._opacityTexture.getAlphaFromRGB;\n }\n else {\n defines.OPACITY = false;\n }\n const reflectionTexture = this._getReflectionTexture();\n if (reflectionTexture && MaterialFlags.ReflectionTextureEnabled) {\n defines.REFLECTION = true;\n defines.GAMMAREFLECTION = reflectionTexture.gammaSpace;\n defines.RGBDREFLECTION = reflectionTexture.isRGBD;\n defines.LODINREFLECTIONALPHA = reflectionTexture.lodLevelInAlpha;\n defines.LINEARSPECULARREFLECTION = reflectionTexture.linearSpecularLOD;\n if (this.realTimeFiltering && this.realTimeFilteringQuality > 0) {\n defines.NUM_SAMPLES = \"\" + this.realTimeFilteringQuality;\n if (engine._features.needTypeSuffixInShaderConstants) {\n defines.NUM_SAMPLES = defines.NUM_SAMPLES + \"u\";\n }\n defines.REALTIME_FILTERING = true;\n }\n else {\n defines.REALTIME_FILTERING = false;\n }\n defines.INVERTCUBICMAP = reflectionTexture.coordinatesMode === Texture.INVCUBIC_MODE;\n defines.REFLECTIONMAP_3D = reflectionTexture.isCube;\n defines.REFLECTIONMAP_OPPOSITEZ = defines.REFLECTIONMAP_3D && this.getScene().useRightHandedSystem ? !reflectionTexture.invertZ : reflectionTexture.invertZ;\n defines.REFLECTIONMAP_CUBIC = false;\n defines.REFLECTIONMAP_EXPLICIT = false;\n defines.REFLECTIONMAP_PLANAR = false;\n defines.REFLECTIONMAP_PROJECTION = false;\n defines.REFLECTIONMAP_SKYBOX = false;\n defines.REFLECTIONMAP_SPHERICAL = false;\n defines.REFLECTIONMAP_EQUIRECTANGULAR = false;\n defines.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = false;\n defines.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = false;\n switch (reflectionTexture.coordinatesMode) {\n case Texture.EXPLICIT_MODE:\n defines.REFLECTIONMAP_EXPLICIT = true;\n break;\n case Texture.PLANAR_MODE:\n defines.REFLECTIONMAP_PLANAR = true;\n break;\n case Texture.PROJECTION_MODE:\n defines.REFLECTIONMAP_PROJECTION = true;\n break;\n case Texture.SKYBOX_MODE:\n defines.REFLECTIONMAP_SKYBOX = true;\n break;\n case Texture.SPHERICAL_MODE:\n defines.REFLECTIONMAP_SPHERICAL = true;\n break;\n case Texture.EQUIRECTANGULAR_MODE:\n defines.REFLECTIONMAP_EQUIRECTANGULAR = true;\n break;\n case Texture.FIXED_EQUIRECTANGULAR_MODE:\n defines.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = true;\n break;\n case Texture.FIXED_EQUIRECTANGULAR_MIRRORED_MODE:\n defines.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = true;\n break;\n case Texture.CUBIC_MODE:\n case Texture.INVCUBIC_MODE:\n default:\n defines.REFLECTIONMAP_CUBIC = true;\n defines.USE_LOCAL_REFLECTIONMAP_CUBIC = reflectionTexture.boundingBoxSize ? true : false;\n break;\n }\n if (reflectionTexture.coordinatesMode !== Texture.SKYBOX_MODE) {\n if (reflectionTexture.irradianceTexture) {\n defines.USEIRRADIANCEMAP = true;\n defines.USESPHERICALFROMREFLECTIONMAP = false;\n }\n // Assume using spherical polynomial if the reflection texture is a cube map\n else if (reflectionTexture.isCube) {\n defines.USESPHERICALFROMREFLECTIONMAP = true;\n defines.USEIRRADIANCEMAP = false;\n if (this._forceIrradianceInFragment || this.realTimeFiltering || engine.getCaps().maxVaryingVectors <= 8) {\n defines.USESPHERICALINVERTEX = false;\n }\n else {\n defines.USESPHERICALINVERTEX = true;\n }\n }\n }\n }\n else {\n defines.REFLECTION = false;\n defines.REFLECTIONMAP_3D = false;\n defines.REFLECTIONMAP_SPHERICAL = false;\n defines.REFLECTIONMAP_PLANAR = false;\n defines.REFLECTIONMAP_CUBIC = false;\n defines.USE_LOCAL_REFLECTIONMAP_CUBIC = false;\n defines.REFLECTIONMAP_PROJECTION = false;\n defines.REFLECTIONMAP_SKYBOX = false;\n defines.REFLECTIONMAP_EXPLICIT = false;\n defines.REFLECTIONMAP_EQUIRECTANGULAR = false;\n defines.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = false;\n defines.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = false;\n defines.INVERTCUBICMAP = false;\n defines.USESPHERICALFROMREFLECTIONMAP = false;\n defines.USEIRRADIANCEMAP = false;\n defines.USESPHERICALINVERTEX = false;\n defines.REFLECTIONMAP_OPPOSITEZ = false;\n defines.LODINREFLECTIONALPHA = false;\n defines.GAMMAREFLECTION = false;\n defines.RGBDREFLECTION = false;\n defines.LINEARSPECULARREFLECTION = false;\n }\n if (this._lightmapTexture && MaterialFlags.LightmapTextureEnabled) {\n MaterialHelper.PrepareDefinesForMergedUV(this._lightmapTexture, defines, \"LIGHTMAP\");\n defines.USELIGHTMAPASSHADOWMAP = this._useLightmapAsShadowmap;\n defines.GAMMALIGHTMAP = this._lightmapTexture.gammaSpace;\n defines.RGBDLIGHTMAP = this._lightmapTexture.isRGBD;\n }\n else {\n defines.LIGHTMAP = false;\n }\n if (this._emissiveTexture && MaterialFlags.EmissiveTextureEnabled) {\n MaterialHelper.PrepareDefinesForMergedUV(this._emissiveTexture, defines, \"EMISSIVE\");\n defines.GAMMAEMISSIVE = this._emissiveTexture.gammaSpace;\n }\n else {\n defines.EMISSIVE = false;\n }\n if (MaterialFlags.SpecularTextureEnabled) {\n if (this._metallicTexture) {\n MaterialHelper.PrepareDefinesForMergedUV(this._metallicTexture, defines, \"REFLECTIVITY\");\n defines.ROUGHNESSSTOREINMETALMAPALPHA = this._useRoughnessFromMetallicTextureAlpha;\n defines.ROUGHNESSSTOREINMETALMAPGREEN = !this._useRoughnessFromMetallicTextureAlpha && this._useRoughnessFromMetallicTextureGreen;\n defines.METALLNESSSTOREINMETALMAPBLUE = this._useMetallnessFromMetallicTextureBlue;\n defines.AOSTOREINMETALMAPRED = this._useAmbientOcclusionFromMetallicTextureRed;\n defines.REFLECTIVITY_GAMMA = false;\n }\n else if (this._reflectivityTexture) {\n MaterialHelper.PrepareDefinesForMergedUV(this._reflectivityTexture, defines, \"REFLECTIVITY\");\n defines.MICROSURFACEFROMREFLECTIVITYMAP = this._useMicroSurfaceFromReflectivityMapAlpha;\n defines.MICROSURFACEAUTOMATIC = this._useAutoMicroSurfaceFromReflectivityMap;\n defines.REFLECTIVITY_GAMMA = this._reflectivityTexture.gammaSpace;\n }\n else {\n defines.REFLECTIVITY = false;\n }\n if (this._metallicReflectanceTexture || this._reflectanceTexture) {\n const identicalTextures = this._metallicReflectanceTexture !== null &&\n this._metallicReflectanceTexture._texture === ((_a = this._reflectanceTexture) === null || _a === void 0 ? void 0 : _a._texture) &&\n this._metallicReflectanceTexture.checkTransformsAreIdentical(this._reflectanceTexture);\n defines.METALLIC_REFLECTANCE_USE_ALPHA_ONLY = this._useOnlyMetallicFromMetallicReflectanceTexture && !identicalTextures;\n if (this._metallicReflectanceTexture) {\n MaterialHelper.PrepareDefinesForMergedUV(this._metallicReflectanceTexture, defines, \"METALLIC_REFLECTANCE\");\n defines.METALLIC_REFLECTANCE_GAMMA = this._metallicReflectanceTexture.gammaSpace;\n }\n else {\n defines.METALLIC_REFLECTANCE = false;\n }\n if (this._reflectanceTexture &&\n !identicalTextures &&\n (!this._metallicReflectanceTexture || (this._metallicReflectanceTexture && this._useOnlyMetallicFromMetallicReflectanceTexture))) {\n MaterialHelper.PrepareDefinesForMergedUV(this._reflectanceTexture, defines, \"REFLECTANCE\");\n defines.REFLECTANCE_GAMMA = this._reflectanceTexture.gammaSpace;\n }\n else {\n defines.REFLECTANCE = false;\n }\n }\n else {\n defines.METALLIC_REFLECTANCE = false;\n defines.REFLECTANCE = false;\n }\n if (this._microSurfaceTexture) {\n MaterialHelper.PrepareDefinesForMergedUV(this._microSurfaceTexture, defines, \"MICROSURFACEMAP\");\n }\n else {\n defines.MICROSURFACEMAP = false;\n }\n }\n else {\n defines.REFLECTIVITY = false;\n defines.MICROSURFACEMAP = false;\n }\n if (engine.getCaps().standardDerivatives && this._bumpTexture && MaterialFlags.BumpTextureEnabled && !this._disableBumpMap) {\n MaterialHelper.PrepareDefinesForMergedUV(this._bumpTexture, defines, \"BUMP\");\n if (this._useParallax && this._albedoTexture && MaterialFlags.DiffuseTextureEnabled) {\n defines.PARALLAX = true;\n defines.PARALLAXOCCLUSION = !!this._useParallaxOcclusion;\n }\n else {\n defines.PARALLAX = false;\n }\n defines.OBJECTSPACE_NORMALMAP = this._useObjectSpaceNormalMap;\n }\n else {\n defines.BUMP = false;\n defines.PARALLAX = false;\n defines.PARALLAXOCCLUSION = false;\n defines.OBJECTSPACE_NORMALMAP = false;\n }\n if (this._environmentBRDFTexture && MaterialFlags.ReflectionTextureEnabled) {\n defines.ENVIRONMENTBRDF = true;\n defines.ENVIRONMENTBRDF_RGBD = this._environmentBRDFTexture.isRGBD;\n }\n else {\n defines.ENVIRONMENTBRDF = false;\n defines.ENVIRONMENTBRDF_RGBD = false;\n }\n if (this._shouldUseAlphaFromAlbedoTexture()) {\n defines.ALPHAFROMALBEDO = true;\n }\n else {\n defines.ALPHAFROMALBEDO = false;\n }\n }\n defines.SPECULAROVERALPHA = this._useSpecularOverAlpha;\n if (this._lightFalloff === PBRBaseMaterial.LIGHTFALLOFF_STANDARD) {\n defines.USEPHYSICALLIGHTFALLOFF = false;\n defines.USEGLTFLIGHTFALLOFF = false;\n }\n else if (this._lightFalloff === PBRBaseMaterial.LIGHTFALLOFF_GLTF) {\n defines.USEPHYSICALLIGHTFALLOFF = false;\n defines.USEGLTFLIGHTFALLOFF = true;\n }\n else {\n defines.USEPHYSICALLIGHTFALLOFF = true;\n defines.USEGLTFLIGHTFALLOFF = false;\n }\n defines.RADIANCEOVERALPHA = this._useRadianceOverAlpha;\n if (!this.backFaceCulling && this._twoSidedLighting) {\n defines.TWOSIDEDLIGHTING = true;\n }\n else {\n defines.TWOSIDEDLIGHTING = false;\n }\n defines.SPECULARAA = engine.getCaps().standardDerivatives && this._enableSpecularAntiAliasing;\n }\n if (defines._areTexturesDirty || defines._areMiscDirty) {\n defines.ALPHATESTVALUE = `${this._alphaCutOff}${this._alphaCutOff % 1 === 0 ? \".\" : \"\"}`;\n defines.PREMULTIPLYALPHA = this.alphaMode === 7 || this.alphaMode === 8;\n defines.ALPHABLEND = this.needAlphaBlendingForMesh(mesh);\n defines.ALPHAFRESNEL = this._useAlphaFresnel || this._useLinearAlphaFresnel;\n defines.LINEARALPHAFRESNEL = this._useLinearAlphaFresnel;\n }\n if (defines._areImageProcessingDirty && this._imageProcessingConfiguration) {\n this._imageProcessingConfiguration.prepareDefines(defines);\n }\n defines.FORCENORMALFORWARD = this._forceNormalForward;\n defines.RADIANCEOCCLUSION = this._useRadianceOcclusion;\n defines.HORIZONOCCLUSION = this._useHorizonOcclusion;\n // Misc.\n if (defines._areMiscDirty) {\n MaterialHelper.PrepareDefinesForMisc(mesh, scene, this._useLogarithmicDepth, this.pointsCloud, this.fogEnabled, this._shouldTurnAlphaTestOn(mesh) || this._forceAlphaTest, defines, this._applyDecalMapAfterDetailMap);\n defines.UNLIT = this._unlit || ((this.pointsCloud || this.wireframe) && !mesh.isVerticesDataPresent(VertexBuffer.NormalKind));\n defines.DEBUGMODE = this._debugMode;\n }\n // Values that need to be evaluated on every frame\n MaterialHelper.PrepareDefinesForFrameBoundValues(scene, engine, this, defines, useInstances ? true : false, useClipPlane, useThinInstances);\n // External config\n this._eventInfo.defines = defines;\n this._eventInfo.mesh = mesh;\n this._callbackPluginEventPrepareDefinesBeforeAttributes(this._eventInfo);\n // Attribs\n MaterialHelper.PrepareDefinesForAttributes(mesh, defines, true, true, true, this._transparencyMode !== PBRBaseMaterial.PBRMATERIAL_OPAQUE);\n // External config\n this._callbackPluginEventPrepareDefines(this._eventInfo);\n }\n /**\n * Force shader compilation\n * @param mesh\n * @param onCompiled\n * @param options\n */\n forceCompilation(mesh, onCompiled, options) {\n const localOptions = Object.assign({ clipPlane: false, useInstances: false }, options);\n if (!this._uniformBufferLayoutBuilt) {\n this.buildUniformLayout();\n }\n this._callbackPluginEventGeneric(MaterialPluginEvent.GetDefineNames, this._eventInfo);\n const defines = new PBRMaterialDefines(this._eventInfo.defineNames);\n const effect = this._prepareEffect(mesh, defines, undefined, undefined, localOptions.useInstances, localOptions.clipPlane, mesh.hasThinInstances);\n if (this._onEffectCreatedObservable) {\n onCreatedEffectParameters$1.effect = effect;\n onCreatedEffectParameters$1.subMesh = null;\n this._onEffectCreatedObservable.notifyObservers(onCreatedEffectParameters$1);\n }\n if (effect.isReady()) {\n if (onCompiled) {\n onCompiled(this);\n }\n }\n else {\n effect.onCompileObservable.add(() => {\n if (onCompiled) {\n onCompiled(this);\n }\n });\n }\n }\n /**\n * Initializes the uniform buffer layout for the shader.\n */\n buildUniformLayout() {\n // Order is important !\n const ubo = this._uniformBuffer;\n ubo.addUniform(\"vAlbedoInfos\", 2);\n ubo.addUniform(\"vAmbientInfos\", 4);\n ubo.addUniform(\"vOpacityInfos\", 2);\n ubo.addUniform(\"vEmissiveInfos\", 2);\n ubo.addUniform(\"vLightmapInfos\", 2);\n ubo.addUniform(\"vReflectivityInfos\", 3);\n ubo.addUniform(\"vMicroSurfaceSamplerInfos\", 2);\n ubo.addUniform(\"vReflectionInfos\", 2);\n ubo.addUniform(\"vReflectionFilteringInfo\", 2);\n ubo.addUniform(\"vReflectionPosition\", 3);\n ubo.addUniform(\"vReflectionSize\", 3);\n ubo.addUniform(\"vBumpInfos\", 3);\n ubo.addUniform(\"albedoMatrix\", 16);\n ubo.addUniform(\"ambientMatrix\", 16);\n ubo.addUniform(\"opacityMatrix\", 16);\n ubo.addUniform(\"emissiveMatrix\", 16);\n ubo.addUniform(\"lightmapMatrix\", 16);\n ubo.addUniform(\"reflectivityMatrix\", 16);\n ubo.addUniform(\"microSurfaceSamplerMatrix\", 16);\n ubo.addUniform(\"bumpMatrix\", 16);\n ubo.addUniform(\"vTangentSpaceParams\", 2);\n ubo.addUniform(\"reflectionMatrix\", 16);\n ubo.addUniform(\"vReflectionColor\", 3);\n ubo.addUniform(\"vAlbedoColor\", 4);\n ubo.addUniform(\"vLightingIntensity\", 4);\n ubo.addUniform(\"vReflectionMicrosurfaceInfos\", 3);\n ubo.addUniform(\"pointSize\", 1);\n ubo.addUniform(\"vReflectivityColor\", 4);\n ubo.addUniform(\"vEmissiveColor\", 3);\n ubo.addUniform(\"vAmbientColor\", 3);\n ubo.addUniform(\"vDebugMode\", 2);\n ubo.addUniform(\"vMetallicReflectanceFactors\", 4);\n ubo.addUniform(\"vMetallicReflectanceInfos\", 2);\n ubo.addUniform(\"metallicReflectanceMatrix\", 16);\n ubo.addUniform(\"vReflectanceInfos\", 2);\n ubo.addUniform(\"reflectanceMatrix\", 16);\n ubo.addUniform(\"vSphericalL00\", 3);\n ubo.addUniform(\"vSphericalL1_1\", 3);\n ubo.addUniform(\"vSphericalL10\", 3);\n ubo.addUniform(\"vSphericalL11\", 3);\n ubo.addUniform(\"vSphericalL2_2\", 3);\n ubo.addUniform(\"vSphericalL2_1\", 3);\n ubo.addUniform(\"vSphericalL20\", 3);\n ubo.addUniform(\"vSphericalL21\", 3);\n ubo.addUniform(\"vSphericalL22\", 3);\n ubo.addUniform(\"vSphericalX\", 3);\n ubo.addUniform(\"vSphericalY\", 3);\n ubo.addUniform(\"vSphericalZ\", 3);\n ubo.addUniform(\"vSphericalXX_ZZ\", 3);\n ubo.addUniform(\"vSphericalYY_ZZ\", 3);\n ubo.addUniform(\"vSphericalZZ\", 3);\n ubo.addUniform(\"vSphericalXY\", 3);\n ubo.addUniform(\"vSphericalYZ\", 3);\n ubo.addUniform(\"vSphericalZX\", 3);\n super.buildUniformLayout();\n }\n /**\n * Binds the submesh data.\n * @param world - The world matrix.\n * @param mesh - The BJS mesh.\n * @param subMesh - A submesh of the BJS mesh.\n */\n bindForSubMesh(world, mesh, subMesh) {\n var _a, _b, _c, _d;\n const scene = this.getScene();\n const defines = subMesh.materialDefines;\n if (!defines) {\n return;\n }\n const effect = subMesh.effect;\n if (!effect) {\n return;\n }\n this._activeEffect = effect;\n // Matrices Mesh.\n mesh.getMeshUniformBuffer().bindToEffect(effect, \"Mesh\");\n mesh.transferToEffect(world);\n const engine = scene.getEngine();\n // Binding unconditionally\n this._uniformBuffer.bindToEffect(effect, \"Material\");\n this.prePassConfiguration.bindForSubMesh(this._activeEffect, scene, mesh, world, this.isFrozen);\n this._eventInfo.subMesh = subMesh;\n this._callbackPluginEventHardBindForSubMesh(this._eventInfo);\n // Normal Matrix\n if (defines.OBJECTSPACE_NORMALMAP) {\n world.toNormalMatrix(this._normalMatrix);\n this.bindOnlyNormalMatrix(this._normalMatrix);\n }\n const mustRebind = effect._forceRebindOnNextCall || this._mustRebind(scene, effect, mesh.visibility);\n // Bones\n MaterialHelper.BindBonesParameters(mesh, this._activeEffect, this.prePassConfiguration);\n let reflectionTexture = null;\n const ubo = this._uniformBuffer;\n if (mustRebind) {\n this.bindViewProjection(effect);\n reflectionTexture = this._getReflectionTexture();\n if (!ubo.useUbo || !this.isFrozen || !ubo.isSync || effect._forceRebindOnNextCall) {\n // Texture uniforms\n if (scene.texturesEnabled) {\n if (this._albedoTexture && MaterialFlags.DiffuseTextureEnabled) {\n ubo.updateFloat2(\"vAlbedoInfos\", this._albedoTexture.coordinatesIndex, this._albedoTexture.level);\n MaterialHelper.BindTextureMatrix(this._albedoTexture, ubo, \"albedo\");\n }\n if (this._ambientTexture && MaterialFlags.AmbientTextureEnabled) {\n ubo.updateFloat4(\"vAmbientInfos\", this._ambientTexture.coordinatesIndex, this._ambientTexture.level, this._ambientTextureStrength, this._ambientTextureImpactOnAnalyticalLights);\n MaterialHelper.BindTextureMatrix(this._ambientTexture, ubo, \"ambient\");\n }\n if (this._opacityTexture && MaterialFlags.OpacityTextureEnabled) {\n ubo.updateFloat2(\"vOpacityInfos\", this._opacityTexture.coordinatesIndex, this._opacityTexture.level);\n MaterialHelper.BindTextureMatrix(this._opacityTexture, ubo, \"opacity\");\n }\n if (reflectionTexture && MaterialFlags.ReflectionTextureEnabled) {\n ubo.updateMatrix(\"reflectionMatrix\", reflectionTexture.getReflectionTextureMatrix());\n ubo.updateFloat2(\"vReflectionInfos\", reflectionTexture.level, 0);\n if (reflectionTexture.boundingBoxSize) {\n const cubeTexture = reflectionTexture;\n ubo.updateVector3(\"vReflectionPosition\", cubeTexture.boundingBoxPosition);\n ubo.updateVector3(\"vReflectionSize\", cubeTexture.boundingBoxSize);\n }\n if (this.realTimeFiltering) {\n const width = reflectionTexture.getSize().width;\n ubo.updateFloat2(\"vReflectionFilteringInfo\", width, Scalar.Log2(width));\n }\n if (!defines.USEIRRADIANCEMAP) {\n const polynomials = reflectionTexture.sphericalPolynomial;\n if (defines.USESPHERICALFROMREFLECTIONMAP && polynomials) {\n if (defines.SPHERICAL_HARMONICS) {\n const preScaledHarmonics = polynomials.preScaledHarmonics;\n ubo.updateVector3(\"vSphericalL00\", preScaledHarmonics.l00);\n ubo.updateVector3(\"vSphericalL1_1\", preScaledHarmonics.l1_1);\n ubo.updateVector3(\"vSphericalL10\", preScaledHarmonics.l10);\n ubo.updateVector3(\"vSphericalL11\", preScaledHarmonics.l11);\n ubo.updateVector3(\"vSphericalL2_2\", preScaledHarmonics.l2_2);\n ubo.updateVector3(\"vSphericalL2_1\", preScaledHarmonics.l2_1);\n ubo.updateVector3(\"vSphericalL20\", preScaledHarmonics.l20);\n ubo.updateVector3(\"vSphericalL21\", preScaledHarmonics.l21);\n ubo.updateVector3(\"vSphericalL22\", preScaledHarmonics.l22);\n }\n else {\n ubo.updateFloat3(\"vSphericalX\", polynomials.x.x, polynomials.x.y, polynomials.x.z);\n ubo.updateFloat3(\"vSphericalY\", polynomials.y.x, polynomials.y.y, polynomials.y.z);\n ubo.updateFloat3(\"vSphericalZ\", polynomials.z.x, polynomials.z.y, polynomials.z.z);\n ubo.updateFloat3(\"vSphericalXX_ZZ\", polynomials.xx.x - polynomials.zz.x, polynomials.xx.y - polynomials.zz.y, polynomials.xx.z - polynomials.zz.z);\n ubo.updateFloat3(\"vSphericalYY_ZZ\", polynomials.yy.x - polynomials.zz.x, polynomials.yy.y - polynomials.zz.y, polynomials.yy.z - polynomials.zz.z);\n ubo.updateFloat3(\"vSphericalZZ\", polynomials.zz.x, polynomials.zz.y, polynomials.zz.z);\n ubo.updateFloat3(\"vSphericalXY\", polynomials.xy.x, polynomials.xy.y, polynomials.xy.z);\n ubo.updateFloat3(\"vSphericalYZ\", polynomials.yz.x, polynomials.yz.y, polynomials.yz.z);\n ubo.updateFloat3(\"vSphericalZX\", polynomials.zx.x, polynomials.zx.y, polynomials.zx.z);\n }\n }\n }\n ubo.updateFloat3(\"vReflectionMicrosurfaceInfos\", reflectionTexture.getSize().width, reflectionTexture.lodGenerationScale, reflectionTexture.lodGenerationOffset);\n }\n if (this._emissiveTexture && MaterialFlags.EmissiveTextureEnabled) {\n ubo.updateFloat2(\"vEmissiveInfos\", this._emissiveTexture.coordinatesIndex, this._emissiveTexture.level);\n MaterialHelper.BindTextureMatrix(this._emissiveTexture, ubo, \"emissive\");\n }\n if (this._lightmapTexture && MaterialFlags.LightmapTextureEnabled) {\n ubo.updateFloat2(\"vLightmapInfos\", this._lightmapTexture.coordinatesIndex, this._lightmapTexture.level);\n MaterialHelper.BindTextureMatrix(this._lightmapTexture, ubo, \"lightmap\");\n }\n if (MaterialFlags.SpecularTextureEnabled) {\n if (this._metallicTexture) {\n ubo.updateFloat3(\"vReflectivityInfos\", this._metallicTexture.coordinatesIndex, this._metallicTexture.level, this._ambientTextureStrength);\n MaterialHelper.BindTextureMatrix(this._metallicTexture, ubo, \"reflectivity\");\n }\n else if (this._reflectivityTexture) {\n ubo.updateFloat3(\"vReflectivityInfos\", this._reflectivityTexture.coordinatesIndex, this._reflectivityTexture.level, 1.0);\n MaterialHelper.BindTextureMatrix(this._reflectivityTexture, ubo, \"reflectivity\");\n }\n if (this._metallicReflectanceTexture) {\n ubo.updateFloat2(\"vMetallicReflectanceInfos\", this._metallicReflectanceTexture.coordinatesIndex, this._metallicReflectanceTexture.level);\n MaterialHelper.BindTextureMatrix(this._metallicReflectanceTexture, ubo, \"metallicReflectance\");\n }\n if (this._reflectanceTexture && defines.REFLECTANCE) {\n ubo.updateFloat2(\"vReflectanceInfos\", this._reflectanceTexture.coordinatesIndex, this._reflectanceTexture.level);\n MaterialHelper.BindTextureMatrix(this._reflectanceTexture, ubo, \"reflectance\");\n }\n if (this._microSurfaceTexture) {\n ubo.updateFloat2(\"vMicroSurfaceSamplerInfos\", this._microSurfaceTexture.coordinatesIndex, this._microSurfaceTexture.level);\n MaterialHelper.BindTextureMatrix(this._microSurfaceTexture, ubo, \"microSurfaceSampler\");\n }\n }\n if (this._bumpTexture && engine.getCaps().standardDerivatives && MaterialFlags.BumpTextureEnabled && !this._disableBumpMap) {\n ubo.updateFloat3(\"vBumpInfos\", this._bumpTexture.coordinatesIndex, this._bumpTexture.level, this._parallaxScaleBias);\n MaterialHelper.BindTextureMatrix(this._bumpTexture, ubo, \"bump\");\n if (scene._mirroredCameraPosition) {\n ubo.updateFloat2(\"vTangentSpaceParams\", this._invertNormalMapX ? 1.0 : -1.0, this._invertNormalMapY ? 1.0 : -1.0);\n }\n else {\n ubo.updateFloat2(\"vTangentSpaceParams\", this._invertNormalMapX ? -1.0 : 1.0, this._invertNormalMapY ? -1.0 : 1.0);\n }\n }\n }\n // Point size\n if (this.pointsCloud) {\n ubo.updateFloat(\"pointSize\", this.pointSize);\n }\n // Colors\n if (defines.METALLICWORKFLOW) {\n TmpColors.Color3[0].r = this._metallic === undefined || this._metallic === null ? 1 : this._metallic;\n TmpColors.Color3[0].g = this._roughness === undefined || this._roughness === null ? 1 : this._roughness;\n ubo.updateColor4(\"vReflectivityColor\", TmpColors.Color3[0], 1);\n const ior = (_b = (_a = this.subSurface) === null || _a === void 0 ? void 0 : _a._indexOfRefraction) !== null && _b !== void 0 ? _b : 1.5;\n const outsideIOR = 1; // consider air as clear coat and other layers would remap in the shader.\n // We are here deriving our default reflectance from a common value for none metallic surface.\n // Based of the schlick fresnel approximation model\n // for dielectrics.\n const f0 = Math.pow((ior - outsideIOR) / (ior + outsideIOR), 2);\n // Tweak the default F0 and F90 based on our given setup\n this._metallicReflectanceColor.scaleToRef(f0 * this._metallicF0Factor, TmpColors.Color3[0]);\n const metallicF90 = this._metallicF0Factor;\n ubo.updateColor4(\"vMetallicReflectanceFactors\", TmpColors.Color3[0], metallicF90);\n }\n else {\n ubo.updateColor4(\"vReflectivityColor\", this._reflectivityColor, this._microSurface);\n }\n ubo.updateColor3(\"vEmissiveColor\", MaterialFlags.EmissiveTextureEnabled ? this._emissiveColor : Color3.BlackReadOnly);\n ubo.updateColor3(\"vReflectionColor\", this._reflectionColor);\n if (!defines.SS_REFRACTION && ((_c = this.subSurface) === null || _c === void 0 ? void 0 : _c._linkRefractionWithTransparency)) {\n ubo.updateColor4(\"vAlbedoColor\", this._albedoColor, 1);\n }\n else {\n ubo.updateColor4(\"vAlbedoColor\", this._albedoColor, this.alpha);\n }\n // Misc\n this._lightingInfos.x = this._directIntensity;\n this._lightingInfos.y = this._emissiveIntensity;\n this._lightingInfos.z = this._environmentIntensity * scene.environmentIntensity;\n this._lightingInfos.w = this._specularIntensity;\n ubo.updateVector4(\"vLightingIntensity\", this._lightingInfos);\n // Colors\n scene.ambientColor.multiplyToRef(this._ambientColor, this._globalAmbientColor);\n ubo.updateColor3(\"vAmbientColor\", this._globalAmbientColor);\n ubo.updateFloat2(\"vDebugMode\", this.debugLimit, this.debugFactor);\n }\n // Textures\n if (scene.texturesEnabled) {\n if (this._albedoTexture && MaterialFlags.DiffuseTextureEnabled) {\n ubo.setTexture(\"albedoSampler\", this._albedoTexture);\n }\n if (this._ambientTexture && MaterialFlags.AmbientTextureEnabled) {\n ubo.setTexture(\"ambientSampler\", this._ambientTexture);\n }\n if (this._opacityTexture && MaterialFlags.OpacityTextureEnabled) {\n ubo.setTexture(\"opacitySampler\", this._opacityTexture);\n }\n if (reflectionTexture && MaterialFlags.ReflectionTextureEnabled) {\n if (defines.LODBASEDMICROSFURACE) {\n ubo.setTexture(\"reflectionSampler\", reflectionTexture);\n }\n else {\n ubo.setTexture(\"reflectionSampler\", reflectionTexture._lodTextureMid || reflectionTexture);\n ubo.setTexture(\"reflectionSamplerLow\", reflectionTexture._lodTextureLow || reflectionTexture);\n ubo.setTexture(\"reflectionSamplerHigh\", reflectionTexture._lodTextureHigh || reflectionTexture);\n }\n if (defines.USEIRRADIANCEMAP) {\n ubo.setTexture(\"irradianceSampler\", reflectionTexture.irradianceTexture);\n }\n }\n if (defines.ENVIRONMENTBRDF) {\n ubo.setTexture(\"environmentBrdfSampler\", this._environmentBRDFTexture);\n }\n if (this._emissiveTexture && MaterialFlags.EmissiveTextureEnabled) {\n ubo.setTexture(\"emissiveSampler\", this._emissiveTexture);\n }\n if (this._lightmapTexture && MaterialFlags.LightmapTextureEnabled) {\n ubo.setTexture(\"lightmapSampler\", this._lightmapTexture);\n }\n if (MaterialFlags.SpecularTextureEnabled) {\n if (this._metallicTexture) {\n ubo.setTexture(\"reflectivitySampler\", this._metallicTexture);\n }\n else if (this._reflectivityTexture) {\n ubo.setTexture(\"reflectivitySampler\", this._reflectivityTexture);\n }\n if (this._metallicReflectanceTexture) {\n ubo.setTexture(\"metallicReflectanceSampler\", this._metallicReflectanceTexture);\n }\n if (this._reflectanceTexture && defines.REFLECTANCE) {\n ubo.setTexture(\"reflectanceSampler\", this._reflectanceTexture);\n }\n if (this._microSurfaceTexture) {\n ubo.setTexture(\"microSurfaceSampler\", this._microSurfaceTexture);\n }\n }\n if (this._bumpTexture && engine.getCaps().standardDerivatives && MaterialFlags.BumpTextureEnabled && !this._disableBumpMap) {\n ubo.setTexture(\"bumpSampler\", this._bumpTexture);\n }\n }\n // OIT with depth peeling\n if (this.getScene().useOrderIndependentTransparency && this.needAlphaBlendingForMesh(mesh)) {\n this.getScene().depthPeelingRenderer.bind(effect);\n }\n this._eventInfo.subMesh = subMesh;\n this._callbackPluginEventBindForSubMesh(this._eventInfo);\n // Clip plane\n bindClipPlane(this._activeEffect, this, scene);\n this.bindEyePosition(effect);\n }\n else if (scene.getEngine()._features.needToAlwaysBindUniformBuffers) {\n this._needToBindSceneUbo = true;\n }\n if (mustRebind || !this.isFrozen) {\n // Lights\n if (scene.lightsEnabled && !this._disableLighting) {\n MaterialHelper.BindLights(scene, mesh, this._activeEffect, defines, this._maxSimultaneousLights);\n }\n // View\n if ((scene.fogEnabled && mesh.applyFog && scene.fogMode !== Scene.FOGMODE_NONE) || reflectionTexture || mesh.receiveShadows || defines.PREPASS) {\n this.bindView(effect);\n }\n // Fog\n MaterialHelper.BindFogParameters(scene, mesh, this._activeEffect, true);\n // Morph targets\n if (defines.NUM_MORPH_INFLUENCERS) {\n MaterialHelper.BindMorphTargetParameters(mesh, this._activeEffect);\n }\n if (defines.BAKED_VERTEX_ANIMATION_TEXTURE) {\n (_d = mesh.bakedVertexAnimationManager) === null || _d === void 0 ? void 0 : _d.bind(effect, defines.INSTANCES);\n }\n // image processing\n this._imageProcessingConfiguration.bind(this._activeEffect);\n // Log. depth\n MaterialHelper.BindLogDepth(defines, this._activeEffect, scene);\n }\n this._afterBind(mesh, this._activeEffect);\n ubo.update();\n }\n /**\n * Returns the animatable textures.\n * If material have animatable metallic texture, then reflectivity texture will not be returned, even if it has animations.\n * @returns - Array of animatable textures.\n */\n getAnimatables() {\n const results = super.getAnimatables();\n if (this._albedoTexture && this._albedoTexture.animations && this._albedoTexture.animations.length > 0) {\n results.push(this._albedoTexture);\n }\n if (this._ambientTexture && this._ambientTexture.animations && this._ambientTexture.animations.length > 0) {\n results.push(this._ambientTexture);\n }\n if (this._opacityTexture && this._opacityTexture.animations && this._opacityTexture.animations.length > 0) {\n results.push(this._opacityTexture);\n }\n if (this._reflectionTexture && this._reflectionTexture.animations && this._reflectionTexture.animations.length > 0) {\n results.push(this._reflectionTexture);\n }\n if (this._emissiveTexture && this._emissiveTexture.animations && this._emissiveTexture.animations.length > 0) {\n results.push(this._emissiveTexture);\n }\n if (this._metallicTexture && this._metallicTexture.animations && this._metallicTexture.animations.length > 0) {\n results.push(this._metallicTexture);\n }\n else if (this._reflectivityTexture && this._reflectivityTexture.animations && this._reflectivityTexture.animations.length > 0) {\n results.push(this._reflectivityTexture);\n }\n if (this._bumpTexture && this._bumpTexture.animations && this._bumpTexture.animations.length > 0) {\n results.push(this._bumpTexture);\n }\n if (this._lightmapTexture && this._lightmapTexture.animations && this._lightmapTexture.animations.length > 0) {\n results.push(this._lightmapTexture);\n }\n if (this._metallicReflectanceTexture && this._metallicReflectanceTexture.animations && this._metallicReflectanceTexture.animations.length > 0) {\n results.push(this._metallicReflectanceTexture);\n }\n if (this._reflectanceTexture && this._reflectanceTexture.animations && this._reflectanceTexture.animations.length > 0) {\n results.push(this._reflectanceTexture);\n }\n if (this._microSurfaceTexture && this._microSurfaceTexture.animations && this._microSurfaceTexture.animations.length > 0) {\n results.push(this._microSurfaceTexture);\n }\n return results;\n }\n /**\n * Returns the texture used for reflections.\n * @returns - Reflection texture if present. Otherwise, returns the environment texture.\n */\n _getReflectionTexture() {\n if (this._reflectionTexture) {\n return this._reflectionTexture;\n }\n return this.getScene().environmentTexture;\n }\n /**\n * Returns an array of the actively used textures.\n * @returns - Array of BaseTextures\n */\n getActiveTextures() {\n const activeTextures = super.getActiveTextures();\n if (this._albedoTexture) {\n activeTextures.push(this._albedoTexture);\n }\n if (this._ambientTexture) {\n activeTextures.push(this._ambientTexture);\n }\n if (this._opacityTexture) {\n activeTextures.push(this._opacityTexture);\n }\n if (this._reflectionTexture) {\n activeTextures.push(this._reflectionTexture);\n }\n if (this._emissiveTexture) {\n activeTextures.push(this._emissiveTexture);\n }\n if (this._reflectivityTexture) {\n activeTextures.push(this._reflectivityTexture);\n }\n if (this._metallicTexture) {\n activeTextures.push(this._metallicTexture);\n }\n if (this._metallicReflectanceTexture) {\n activeTextures.push(this._metallicReflectanceTexture);\n }\n if (this._reflectanceTexture) {\n activeTextures.push(this._reflectanceTexture);\n }\n if (this._microSurfaceTexture) {\n activeTextures.push(this._microSurfaceTexture);\n }\n if (this._bumpTexture) {\n activeTextures.push(this._bumpTexture);\n }\n if (this._lightmapTexture) {\n activeTextures.push(this._lightmapTexture);\n }\n return activeTextures;\n }\n /**\n * Checks to see if a texture is used in the material.\n * @param texture - Base texture to use.\n * @returns - Boolean specifying if a texture is used in the material.\n */\n hasTexture(texture) {\n if (super.hasTexture(texture)) {\n return true;\n }\n if (this._albedoTexture === texture) {\n return true;\n }\n if (this._ambientTexture === texture) {\n return true;\n }\n if (this._opacityTexture === texture) {\n return true;\n }\n if (this._reflectionTexture === texture) {\n return true;\n }\n if (this._emissiveTexture === texture) {\n return true;\n }\n if (this._reflectivityTexture === texture) {\n return true;\n }\n if (this._metallicTexture === texture) {\n return true;\n }\n if (this._metallicReflectanceTexture === texture) {\n return true;\n }\n if (this._reflectanceTexture === texture) {\n return true;\n }\n if (this._microSurfaceTexture === texture) {\n return true;\n }\n if (this._bumpTexture === texture) {\n return true;\n }\n if (this._lightmapTexture === texture) {\n return true;\n }\n return false;\n }\n /**\n * Sets the required values to the prepass renderer.\n * It can't be sets when subsurface scattering of this material is disabled.\n * When scene have ability to enable subsurface prepass effect, it will enable.\n */\n setPrePassRenderer() {\n var _a;\n if (!((_a = this.subSurface) === null || _a === void 0 ? void 0 : _a.isScatteringEnabled)) {\n return false;\n }\n const subSurfaceConfiguration = this.getScene().enableSubSurfaceForPrePass();\n if (subSurfaceConfiguration) {\n subSurfaceConfiguration.enabled = true;\n }\n return true;\n }\n /**\n * Disposes the resources of the material.\n * @param forceDisposeEffect - Forces the disposal of effects.\n * @param forceDisposeTextures - Forces the disposal of all textures.\n */\n dispose(forceDisposeEffect, forceDisposeTextures) {\n var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m;\n if (forceDisposeTextures) {\n if (this._environmentBRDFTexture && this.getScene().environmentBRDFTexture !== this._environmentBRDFTexture) {\n this._environmentBRDFTexture.dispose();\n }\n (_a = this._albedoTexture) === null || _a === void 0 ? void 0 : _a.dispose();\n (_b = this._ambientTexture) === null || _b === void 0 ? void 0 : _b.dispose();\n (_c = this._opacityTexture) === null || _c === void 0 ? void 0 : _c.dispose();\n (_d = this._reflectionTexture) === null || _d === void 0 ? void 0 : _d.dispose();\n (_e = this._emissiveTexture) === null || _e === void 0 ? void 0 : _e.dispose();\n (_f = this._metallicTexture) === null || _f === void 0 ? void 0 : _f.dispose();\n (_g = this._reflectivityTexture) === null || _g === void 0 ? void 0 : _g.dispose();\n (_h = this._bumpTexture) === null || _h === void 0 ? void 0 : _h.dispose();\n (_j = this._lightmapTexture) === null || _j === void 0 ? void 0 : _j.dispose();\n (_k = this._metallicReflectanceTexture) === null || _k === void 0 ? void 0 : _k.dispose();\n (_l = this._reflectanceTexture) === null || _l === void 0 ? void 0 : _l.dispose();\n (_m = this._microSurfaceTexture) === null || _m === void 0 ? void 0 : _m.dispose();\n }\n this._renderTargets.dispose();\n if (this._imageProcessingConfiguration && this._imageProcessingObserver) {\n this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver);\n }\n super.dispose(forceDisposeEffect, forceDisposeTextures);\n }\n }\n /**\n * PBRMaterialTransparencyMode: No transparency mode, Alpha channel is not use.\n */\n PBRBaseMaterial.PBRMATERIAL_OPAQUE = Material.MATERIAL_OPAQUE;\n /**\n * PBRMaterialTransparencyMode: Alpha Test mode, pixel are discarded below a certain threshold defined by the alpha cutoff value.\n */\n PBRBaseMaterial.PBRMATERIAL_ALPHATEST = Material.MATERIAL_ALPHATEST;\n /**\n * PBRMaterialTransparencyMode: Pixels are blended (according to the alpha mode) with the already drawn pixels in the current frame buffer.\n */\n PBRBaseMaterial.PBRMATERIAL_ALPHABLEND = Material.MATERIAL_ALPHABLEND;\n /**\n * PBRMaterialTransparencyMode: Pixels are blended (according to the alpha mode) with the already drawn pixels in the current frame buffer.\n * They are also discarded below the alpha cutoff threshold to improve performances.\n */\n PBRBaseMaterial.PBRMATERIAL_ALPHATESTANDBLEND = Material.MATERIAL_ALPHATESTANDBLEND;\n /**\n * Defines the default value of how much AO map is occluding the analytical lights\n * (point spot...).\n */\n PBRBaseMaterial.DEFAULT_AO_ON_ANALYTICAL_LIGHTS = 0;\n /**\n * PBRMaterialLightFalloff Physical: light is falling off following the inverse squared distance law.\n */\n PBRBaseMaterial.LIGHTFALLOFF_PHYSICAL = 0;\n /**\n * PBRMaterialLightFalloff gltf: light is falling off as described in the gltf moving to PBR document\n * to enhance interoperability with other engines.\n */\n PBRBaseMaterial.LIGHTFALLOFF_GLTF = 1;\n /**\n * PBRMaterialLightFalloff Standard: light is falling off like in the standard material\n * to enhance interoperability with other materials.\n */\n PBRBaseMaterial.LIGHTFALLOFF_STANDARD = 2;\n __decorate$1([\n serializeAsImageProcessingConfiguration()\n ], PBRBaseMaterial.prototype, \"_imageProcessingConfiguration\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsMiscDirty\")\n ], PBRBaseMaterial.prototype, \"debugMode\", void 0);\n __decorate$1([\n serialize()\n ], PBRBaseMaterial.prototype, \"useLogarithmicDepth\", null);\n\n /**\n * The Physically based material of BJS.\n *\n * This offers the main features of a standard PBR material.\n * For more information, please refer to the documentation :\n * https://doc.babylonjs.com/features/featuresDeepDive/materials/using/introToPBR\n */\n class PBRMaterial extends PBRBaseMaterial {\n /**\n * Stores the refracted light information in a texture.\n */\n get refractionTexture() {\n return this.subSurface.refractionTexture;\n }\n set refractionTexture(value) {\n this.subSurface.refractionTexture = value;\n if (value) {\n this.subSurface.isRefractionEnabled = true;\n }\n else if (!this.subSurface.linkRefractionWithTransparency) {\n this.subSurface.isRefractionEnabled = false;\n }\n }\n /**\n * Index of refraction of the material base layer.\n * https://en.wikipedia.org/wiki/List_of_refractive_indices\n *\n * This does not only impact refraction but also the Base F0 of Dielectric Materials.\n *\n * From dielectric fresnel rules: F0 = square((iorT - iorI) / (iorT + iorI))\n */\n get indexOfRefraction() {\n return this.subSurface.indexOfRefraction;\n }\n set indexOfRefraction(value) {\n this.subSurface.indexOfRefraction = value;\n }\n /**\n * Controls if refraction needs to be inverted on Y. This could be useful for procedural texture.\n */\n get invertRefractionY() {\n return this.subSurface.invertRefractionY;\n }\n set invertRefractionY(value) {\n this.subSurface.invertRefractionY = value;\n }\n /**\n * This parameters will make the material used its opacity to control how much it is refracting against not.\n * Materials half opaque for instance using refraction could benefit from this control.\n */\n get linkRefractionWithTransparency() {\n return this.subSurface.linkRefractionWithTransparency;\n }\n set linkRefractionWithTransparency(value) {\n this.subSurface.linkRefractionWithTransparency = value;\n if (value) {\n this.subSurface.isRefractionEnabled = true;\n }\n }\n /**\n * BJS is using an hardcoded light falloff based on a manually sets up range.\n * In PBR, one way to represents the falloff is to use the inverse squared root algorithm.\n * This parameter can help you switch back to the BJS mode in order to create scenes using both materials.\n */\n get usePhysicalLightFalloff() {\n return this._lightFalloff === PBRBaseMaterial.LIGHTFALLOFF_PHYSICAL;\n }\n /**\n * BJS is using an hardcoded light falloff based on a manually sets up range.\n * In PBR, one way to represents the falloff is to use the inverse squared root algorithm.\n * This parameter can help you switch back to the BJS mode in order to create scenes using both materials.\n */\n set usePhysicalLightFalloff(value) {\n if (value !== this.usePhysicalLightFalloff) {\n // Ensure the effect will be rebuilt.\n this._markAllSubMeshesAsTexturesDirty();\n if (value) {\n this._lightFalloff = PBRBaseMaterial.LIGHTFALLOFF_PHYSICAL;\n }\n else {\n this._lightFalloff = PBRBaseMaterial.LIGHTFALLOFF_STANDARD;\n }\n }\n }\n /**\n * In order to support the falloff compatibility with gltf, a special mode has been added\n * to reproduce the gltf light falloff.\n */\n get useGLTFLightFalloff() {\n return this._lightFalloff === PBRBaseMaterial.LIGHTFALLOFF_GLTF;\n }\n /**\n * In order to support the falloff compatibility with gltf, a special mode has been added\n * to reproduce the gltf light falloff.\n */\n set useGLTFLightFalloff(value) {\n if (value !== this.useGLTFLightFalloff) {\n // Ensure the effect will be rebuilt.\n this._markAllSubMeshesAsTexturesDirty();\n if (value) {\n this._lightFalloff = PBRBaseMaterial.LIGHTFALLOFF_GLTF;\n }\n else {\n this._lightFalloff = PBRBaseMaterial.LIGHTFALLOFF_STANDARD;\n }\n }\n }\n /**\n * Gets the image processing configuration used either in this material.\n */\n get imageProcessingConfiguration() {\n return this._imageProcessingConfiguration;\n }\n /**\n * Sets the Default image processing configuration used either in the this material.\n *\n * If sets to null, the scene one is in use.\n */\n set imageProcessingConfiguration(value) {\n this._attachImageProcessingConfiguration(value);\n // Ensure the effect will be rebuilt.\n this._markAllSubMeshesAsTexturesDirty();\n }\n /**\n * Gets whether the color curves effect is enabled.\n */\n get cameraColorCurvesEnabled() {\n return this.imageProcessingConfiguration.colorCurvesEnabled;\n }\n /**\n * Sets whether the color curves effect is enabled.\n */\n set cameraColorCurvesEnabled(value) {\n this.imageProcessingConfiguration.colorCurvesEnabled = value;\n }\n /**\n * Gets whether the color grading effect is enabled.\n */\n get cameraColorGradingEnabled() {\n return this.imageProcessingConfiguration.colorGradingEnabled;\n }\n /**\n * Gets whether the color grading effect is enabled.\n */\n set cameraColorGradingEnabled(value) {\n this.imageProcessingConfiguration.colorGradingEnabled = value;\n }\n /**\n * Gets whether tonemapping is enabled or not.\n */\n get cameraToneMappingEnabled() {\n return this._imageProcessingConfiguration.toneMappingEnabled;\n }\n /**\n * Sets whether tonemapping is enabled or not\n */\n set cameraToneMappingEnabled(value) {\n this._imageProcessingConfiguration.toneMappingEnabled = value;\n }\n /**\n * The camera exposure used on this material.\n * This property is here and not in the camera to allow controlling exposure without full screen post process.\n * This corresponds to a photographic exposure.\n */\n get cameraExposure() {\n return this._imageProcessingConfiguration.exposure;\n }\n /**\n * The camera exposure used on this material.\n * This property is here and not in the camera to allow controlling exposure without full screen post process.\n * This corresponds to a photographic exposure.\n */\n set cameraExposure(value) {\n this._imageProcessingConfiguration.exposure = value;\n }\n /**\n * Gets The camera contrast used on this material.\n */\n get cameraContrast() {\n return this._imageProcessingConfiguration.contrast;\n }\n /**\n * Sets The camera contrast used on this material.\n */\n set cameraContrast(value) {\n this._imageProcessingConfiguration.contrast = value;\n }\n /**\n * Gets the Color Grading 2D Lookup Texture.\n */\n get cameraColorGradingTexture() {\n return this._imageProcessingConfiguration.colorGradingTexture;\n }\n /**\n * Sets the Color Grading 2D Lookup Texture.\n */\n set cameraColorGradingTexture(value) {\n this._imageProcessingConfiguration.colorGradingTexture = value;\n }\n /**\n * The color grading curves provide additional color adjustment that is applied after any color grading transform (3D LUT).\n * They allow basic adjustment of saturation and small exposure adjustments, along with color filter tinting to provide white balance adjustment or more stylistic effects.\n * These are similar to controls found in many professional imaging or colorist software. The global controls are applied to the entire image. For advanced tuning, extra controls are provided to adjust the shadow, midtone and highlight areas of the image;\n * corresponding to low luminance, medium luminance, and high luminance areas respectively.\n */\n get cameraColorCurves() {\n return this._imageProcessingConfiguration.colorCurves;\n }\n /**\n * The color grading curves provide additional color adjustment that is applied after any color grading transform (3D LUT).\n * They allow basic adjustment of saturation and small exposure adjustments, along with color filter tinting to provide white balance adjustment or more stylistic effects.\n * These are similar to controls found in many professional imaging or colorist software. The global controls are applied to the entire image. For advanced tuning, extra controls are provided to adjust the shadow, midtone and highlight areas of the image;\n * corresponding to low luminance, medium luminance, and high luminance areas respectively.\n */\n set cameraColorCurves(value) {\n this._imageProcessingConfiguration.colorCurves = value;\n }\n /**\n * Instantiates a new PBRMaterial instance.\n *\n * @param name The material name\n * @param scene The scene the material will be use in.\n */\n constructor(name, scene) {\n super(name, scene);\n /**\n * Intensity of the direct lights e.g. the four lights available in your scene.\n * This impacts both the direct diffuse and specular highlights.\n */\n this.directIntensity = 1.0;\n /**\n * Intensity of the emissive part of the material.\n * This helps controlling the emissive effect without modifying the emissive color.\n */\n this.emissiveIntensity = 1.0;\n /**\n * Intensity of the environment e.g. how much the environment will light the object\n * either through harmonics for rough material or through the reflection for shiny ones.\n */\n this.environmentIntensity = 1.0;\n /**\n * This is a special control allowing the reduction of the specular highlights coming from the\n * four lights of the scene. Those highlights may not be needed in full environment lighting.\n */\n this.specularIntensity = 1.0;\n /**\n * Debug Control allowing disabling the bump map on this material.\n */\n this.disableBumpMap = false;\n /**\n * AKA Occlusion Texture Intensity in other nomenclature.\n */\n this.ambientTextureStrength = 1.0;\n /**\n * Defines how much the AO map is occluding the analytical lights (point spot...).\n * 1 means it completely occludes it\n * 0 mean it has no impact\n */\n this.ambientTextureImpactOnAnalyticalLights = PBRMaterial.DEFAULT_AO_ON_ANALYTICAL_LIGHTS;\n /**\n * In metallic workflow, specifies an F0 factor to help configuring the material F0.\n * By default the indexOfrefraction is used to compute F0;\n *\n * This is used as a factor against the default reflectance at normal incidence to tweak it.\n *\n * F0 = defaultF0 * metallicF0Factor * metallicReflectanceColor;\n * F90 = metallicReflectanceColor;\n */\n this.metallicF0Factor = 1;\n /**\n * In metallic workflow, specifies an F0 color.\n * By default the F90 is always 1;\n *\n * Please note that this factor is also used as a factor against the default reflectance at normal incidence.\n *\n * F0 = defaultF0_from_IOR * metallicF0Factor * metallicReflectanceColor\n * F90 = metallicF0Factor;\n */\n this.metallicReflectanceColor = Color3.White();\n /**\n * Specifies that only the A channel from metallicReflectanceTexture should be used.\n * If false, both RGB and A channels will be used\n */\n this.useOnlyMetallicFromMetallicReflectanceTexture = false;\n /**\n * The color of a material in ambient lighting.\n */\n this.ambientColor = new Color3(0, 0, 0);\n /**\n * AKA Diffuse Color in other nomenclature.\n */\n this.albedoColor = new Color3(1, 1, 1);\n /**\n * AKA Specular Color in other nomenclature.\n */\n this.reflectivityColor = new Color3(1, 1, 1);\n /**\n * The color reflected from the material.\n */\n this.reflectionColor = new Color3(1.0, 1.0, 1.0);\n /**\n * The color emitted from the material.\n */\n this.emissiveColor = new Color3(0, 0, 0);\n /**\n * AKA Glossiness in other nomenclature.\n */\n this.microSurface = 1.0;\n /**\n * If true, the light map contains occlusion information instead of lighting info.\n */\n this.useLightmapAsShadowmap = false;\n /**\n * Specifies that the alpha is coming form the albedo channel alpha channel for alpha blending.\n */\n this.useAlphaFromAlbedoTexture = false;\n /**\n * Enforces alpha test in opaque or blend mode in order to improve the performances of some situations.\n */\n this.forceAlphaTest = false;\n /**\n * Defines the alpha limits in alpha test mode.\n */\n this.alphaCutOff = 0.4;\n /**\n * Specifies that the material will keep the specular highlights over a transparent surface (only the most luminous ones).\n * A car glass is a good example of that. When sun reflects on it you can not see what is behind.\n */\n this.useSpecularOverAlpha = true;\n /**\n * Specifies if the reflectivity texture contains the glossiness information in its alpha channel.\n */\n this.useMicroSurfaceFromReflectivityMapAlpha = false;\n /**\n * Specifies if the metallic texture contains the roughness information in its alpha channel.\n */\n this.useRoughnessFromMetallicTextureAlpha = true;\n /**\n * Specifies if the metallic texture contains the roughness information in its green channel.\n */\n this.useRoughnessFromMetallicTextureGreen = false;\n /**\n * Specifies if the metallic texture contains the metallness information in its blue channel.\n */\n this.useMetallnessFromMetallicTextureBlue = false;\n /**\n * Specifies if the metallic texture contains the ambient occlusion information in its red channel.\n */\n this.useAmbientOcclusionFromMetallicTextureRed = false;\n /**\n * Specifies if the ambient texture contains the ambient occlusion information in its red channel only.\n */\n this.useAmbientInGrayScale = false;\n /**\n * In case the reflectivity map does not contain the microsurface information in its alpha channel,\n * The material will try to infer what glossiness each pixel should be.\n */\n this.useAutoMicroSurfaceFromReflectivityMap = false;\n /**\n * Specifies that the material will keeps the reflection highlights over a transparent surface (only the most luminous ones).\n * A car glass is a good example of that. When the street lights reflects on it you can not see what is behind.\n */\n this.useRadianceOverAlpha = true;\n /**\n * Allows using an object space normal map (instead of tangent space).\n */\n this.useObjectSpaceNormalMap = false;\n /**\n * Allows using the bump map in parallax mode.\n */\n this.useParallax = false;\n /**\n * Allows using the bump map in parallax occlusion mode.\n */\n this.useParallaxOcclusion = false;\n /**\n * Controls the scale bias of the parallax mode.\n */\n this.parallaxScaleBias = 0.05;\n /**\n * If sets to true, disables all the lights affecting the material.\n */\n this.disableLighting = false;\n /**\n * Force the shader to compute irradiance in the fragment shader in order to take bump in account.\n */\n this.forceIrradianceInFragment = false;\n /**\n * Number of Simultaneous lights allowed on the material.\n */\n this.maxSimultaneousLights = 4;\n /**\n * If sets to true, x component of normal map value will invert (x = 1.0 - x).\n */\n this.invertNormalMapX = false;\n /**\n * If sets to true, y component of normal map value will invert (y = 1.0 - y).\n */\n this.invertNormalMapY = false;\n /**\n * If sets to true and backfaceCulling is false, normals will be flipped on the backside.\n */\n this.twoSidedLighting = false;\n /**\n * A fresnel is applied to the alpha of the model to ensure grazing angles edges are not alpha tested.\n * And/Or occlude the blended part. (alpha is converted to gamma to compute the fresnel)\n */\n this.useAlphaFresnel = false;\n /**\n * A fresnel is applied to the alpha of the model to ensure grazing angles edges are not alpha tested.\n * And/Or occlude the blended part. (alpha stays linear to compute the fresnel)\n */\n this.useLinearAlphaFresnel = false;\n /**\n * Let user defines the brdf lookup texture used for IBL.\n * A default 8bit version is embedded but you could point at :\n * * Default texture: https://assets.babylonjs.com/environments/correlatedMSBRDF_RGBD.png\n * * Default 16bit pixel depth texture: https://assets.babylonjs.com/environments/correlatedMSBRDF.dds\n * * LEGACY Default None correlated https://assets.babylonjs.com/environments/uncorrelatedBRDF_RGBD.png\n * * LEGACY Default None correlated 16bit pixel depth https://assets.babylonjs.com/environments/uncorrelatedBRDF.dds\n */\n this.environmentBRDFTexture = null;\n /**\n * Force normal to face away from face.\n */\n this.forceNormalForward = false;\n /**\n * Enables specular anti aliasing in the PBR shader.\n * It will both interacts on the Geometry for analytical and IBL lighting.\n * It also prefilter the roughness map based on the bump values.\n */\n this.enableSpecularAntiAliasing = false;\n /**\n * This parameters will enable/disable Horizon occlusion to prevent normal maps to look shiny when the normal\n * makes the reflect vector face the model (under horizon).\n */\n this.useHorizonOcclusion = true;\n /**\n * This parameters will enable/disable radiance occlusion by preventing the radiance to lit\n * too much the area relying on ambient texture to define their ambient occlusion.\n */\n this.useRadianceOcclusion = true;\n /**\n * If set to true, no lighting calculations will be applied.\n */\n this.unlit = false;\n /**\n * If sets to true, the decal map will be applied after the detail map. Else, it is applied before (default: false)\n */\n this.applyDecalMapAfterDetailMap = false;\n this._environmentBRDFTexture = GetEnvironmentBRDFTexture(this.getScene());\n }\n /**\n * Returns the name of this material class.\n */\n getClassName() {\n return \"PBRMaterial\";\n }\n /**\n * Makes a duplicate of the current material.\n * @param name - name to use for the new material.\n * @param cloneTexturesOnlyOnce - if a texture is used in more than one channel (e.g diffuse and opacity), only clone it once and reuse it on the other channels. Default false.\n * @param rootUrl defines the root URL to use to load textures\n */\n clone(name, cloneTexturesOnlyOnce = true, rootUrl = \"\") {\n const clone = SerializationHelper.Clone(() => new PBRMaterial(name, this.getScene()), this, { cloneTexturesOnlyOnce });\n clone.id = name;\n clone.name = name;\n this.stencil.copyTo(clone.stencil);\n this._clonePlugins(clone, rootUrl);\n return clone;\n }\n /**\n * Serializes this PBR Material.\n * @returns - An object with the serialized material.\n */\n serialize() {\n const serializationObject = super.serialize();\n serializationObject.customType = \"BABYLON.PBRMaterial\";\n return serializationObject;\n }\n // Statics\n /**\n * Parses a PBR Material from a serialized object.\n * @param source - Serialized object.\n * @param scene - BJS scene instance.\n * @param rootUrl - url for the scene object\n * @returns - PBRMaterial\n */\n static Parse(source, scene, rootUrl) {\n const material = SerializationHelper.Parse(() => new PBRMaterial(source.name, scene), source, scene, rootUrl);\n if (source.stencil) {\n material.stencil.parse(source.stencil, scene, rootUrl);\n }\n Material._parsePlugins(source, material, scene, rootUrl);\n // The code block below ensures backward compatibility with serialized materials before plugins are automatically serialized.\n if (source.clearCoat) {\n material.clearCoat.parse(source.clearCoat, scene, rootUrl);\n }\n if (source.anisotropy) {\n material.anisotropy.parse(source.anisotropy, scene, rootUrl);\n }\n if (source.brdf) {\n material.brdf.parse(source.brdf, scene, rootUrl);\n }\n if (source.sheen) {\n material.sheen.parse(source.sheen, scene, rootUrl);\n }\n if (source.subSurface) {\n material.subSurface.parse(source.subSurface, scene, rootUrl);\n }\n if (source.iridescence) {\n material.iridescence.parse(source.iridescence, scene, rootUrl);\n }\n return material;\n }\n }\n /**\n * PBRMaterialTransparencyMode: No transparency mode, Alpha channel is not use.\n */\n PBRMaterial.PBRMATERIAL_OPAQUE = PBRBaseMaterial.PBRMATERIAL_OPAQUE;\n /**\n * PBRMaterialTransparencyMode: Alpha Test mode, pixel are discarded below a certain threshold defined by the alpha cutoff value.\n */\n PBRMaterial.PBRMATERIAL_ALPHATEST = PBRBaseMaterial.PBRMATERIAL_ALPHATEST;\n /**\n * PBRMaterialTransparencyMode: Pixels are blended (according to the alpha mode) with the already drawn pixels in the current frame buffer.\n */\n PBRMaterial.PBRMATERIAL_ALPHABLEND = PBRBaseMaterial.PBRMATERIAL_ALPHABLEND;\n /**\n * PBRMaterialTransparencyMode: Pixels are blended (according to the alpha mode) with the already drawn pixels in the current frame buffer.\n * They are also discarded below the alpha cutoff threshold to improve performances.\n */\n PBRMaterial.PBRMATERIAL_ALPHATESTANDBLEND = PBRBaseMaterial.PBRMATERIAL_ALPHATESTANDBLEND;\n /**\n * Defines the default value of how much AO map is occluding the analytical lights\n * (point spot...).\n */\n PBRMaterial.DEFAULT_AO_ON_ANALYTICAL_LIGHTS = PBRBaseMaterial.DEFAULT_AO_ON_ANALYTICAL_LIGHTS;\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"directIntensity\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"emissiveIntensity\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"environmentIntensity\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"specularIntensity\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"disableBumpMap\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"albedoTexture\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"ambientTexture\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"ambientTextureStrength\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"ambientTextureImpactOnAnalyticalLights\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesAndMiscDirty\")\n ], PBRMaterial.prototype, \"opacityTexture\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"reflectionTexture\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"emissiveTexture\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"reflectivityTexture\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"metallicTexture\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"metallic\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"roughness\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"metallicF0Factor\", void 0);\n __decorate$1([\n serializeAsColor3(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"metallicReflectanceColor\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"useOnlyMetallicFromMetallicReflectanceTexture\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"metallicReflectanceTexture\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"reflectanceTexture\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"microSurfaceTexture\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"bumpTexture\", void 0);\n __decorate$1([\n serializeAsTexture(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\", null)\n ], PBRMaterial.prototype, \"lightmapTexture\", void 0);\n __decorate$1([\n serializeAsColor3(\"ambient\"),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"ambientColor\", void 0);\n __decorate$1([\n serializeAsColor3(\"albedo\"),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"albedoColor\", void 0);\n __decorate$1([\n serializeAsColor3(\"reflectivity\"),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"reflectivityColor\", void 0);\n __decorate$1([\n serializeAsColor3(\"reflection\"),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"reflectionColor\", void 0);\n __decorate$1([\n serializeAsColor3(\"emissive\"),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"emissiveColor\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"microSurface\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"useLightmapAsShadowmap\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesAndMiscDirty\")\n ], PBRMaterial.prototype, \"useAlphaFromAlbedoTexture\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesAndMiscDirty\")\n ], PBRMaterial.prototype, \"forceAlphaTest\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesAndMiscDirty\")\n ], PBRMaterial.prototype, \"alphaCutOff\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"useSpecularOverAlpha\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"useMicroSurfaceFromReflectivityMapAlpha\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"useRoughnessFromMetallicTextureAlpha\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"useRoughnessFromMetallicTextureGreen\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"useMetallnessFromMetallicTextureBlue\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"useAmbientOcclusionFromMetallicTextureRed\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"useAmbientInGrayScale\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"useAutoMicroSurfaceFromReflectivityMap\", void 0);\n __decorate$1([\n serialize()\n ], PBRMaterial.prototype, \"usePhysicalLightFalloff\", null);\n __decorate$1([\n serialize()\n ], PBRMaterial.prototype, \"useGLTFLightFalloff\", null);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"useRadianceOverAlpha\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"useObjectSpaceNormalMap\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"useParallax\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"useParallaxOcclusion\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"parallaxScaleBias\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsLightsDirty\")\n ], PBRMaterial.prototype, \"disableLighting\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"forceIrradianceInFragment\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsLightsDirty\")\n ], PBRMaterial.prototype, \"maxSimultaneousLights\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"invertNormalMapX\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"invertNormalMapY\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"twoSidedLighting\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"useAlphaFresnel\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"useLinearAlphaFresnel\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"environmentBRDFTexture\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"forceNormalForward\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"enableSpecularAntiAliasing\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"useHorizonOcclusion\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], PBRMaterial.prototype, \"useRadianceOcclusion\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsMiscDirty\")\n ], PBRMaterial.prototype, \"unlit\", void 0);\n __decorate$1([\n serialize(),\n expandToProperty(\"_markAllSubMeshesAsMiscDirty\")\n ], PBRMaterial.prototype, \"applyDecalMapAfterDetailMap\", void 0);\n RegisterClass(\"BABYLON.PBRMaterial\", PBRMaterial);\n\n /**\n * Base class of all the lights in Babylon. It groups all the generic information about lights.\n * Lights are used, as you would expect, to affect how meshes are seen, in terms of both illumination and colour.\n * All meshes allow light to pass through them unless shadow generation is activated. The default number of lights allowed is four but this can be increased.\n */\n class Light extends Node {\n /**\n * Defines how far from the source the light is impacting in scene units.\n * Note: Unused in PBR material as the distance light falloff is defined following the inverse squared falloff.\n */\n get range() {\n return this._range;\n }\n /**\n * Defines how far from the source the light is impacting in scene units.\n * Note: Unused in PBR material as the distance light falloff is defined following the inverse squared falloff.\n */\n set range(value) {\n this._range = value;\n this._inverseSquaredRange = 1.0 / (this.range * this.range);\n }\n /**\n * Gets the photometric scale used to interpret the intensity.\n * This is only relevant with PBR Materials where the light intensity can be defined in a physical way.\n */\n get intensityMode() {\n return this._intensityMode;\n }\n /**\n * Sets the photometric scale used to interpret the intensity.\n * This is only relevant with PBR Materials where the light intensity can be defined in a physical way.\n */\n set intensityMode(value) {\n this._intensityMode = value;\n this._computePhotometricScale();\n }\n /**\n * Gets the light radius used by PBR Materials to simulate soft area lights.\n */\n get radius() {\n return this._radius;\n }\n /**\n * sets the light radius used by PBR Materials to simulate soft area lights.\n */\n set radius(value) {\n this._radius = value;\n this._computePhotometricScale();\n }\n /**\n * Gets whether or not the shadows are enabled for this light. This can help turning off/on shadow without detaching\n * the current shadow generator.\n */\n get shadowEnabled() {\n return this._shadowEnabled;\n }\n /**\n * Sets whether or not the shadows are enabled for this light. This can help turning off/on shadow without detaching\n * the current shadow generator.\n */\n set shadowEnabled(value) {\n if (this._shadowEnabled === value) {\n return;\n }\n this._shadowEnabled = value;\n this._markMeshesAsLightDirty();\n }\n /**\n * Gets the only meshes impacted by this light.\n */\n get includedOnlyMeshes() {\n return this._includedOnlyMeshes;\n }\n /**\n * Sets the only meshes impacted by this light.\n */\n set includedOnlyMeshes(value) {\n this._includedOnlyMeshes = value;\n this._hookArrayForIncludedOnly(value);\n }\n /**\n * Gets the meshes not impacted by this light.\n */\n get excludedMeshes() {\n return this._excludedMeshes;\n }\n /**\n * Sets the meshes not impacted by this light.\n */\n set excludedMeshes(value) {\n this._excludedMeshes = value;\n this._hookArrayForExcluded(value);\n }\n /**\n * Gets the layer id use to find what meshes are not impacted by the light.\n * Inactive if 0\n */\n get excludeWithLayerMask() {\n return this._excludeWithLayerMask;\n }\n /**\n * Sets the layer id use to find what meshes are not impacted by the light.\n * Inactive if 0\n */\n set excludeWithLayerMask(value) {\n this._excludeWithLayerMask = value;\n this._resyncMeshes();\n }\n /**\n * Gets the layer id use to find what meshes are impacted by the light.\n * Inactive if 0\n */\n get includeOnlyWithLayerMask() {\n return this._includeOnlyWithLayerMask;\n }\n /**\n * Sets the layer id use to find what meshes are impacted by the light.\n * Inactive if 0\n */\n set includeOnlyWithLayerMask(value) {\n this._includeOnlyWithLayerMask = value;\n this._resyncMeshes();\n }\n /**\n * Gets the lightmap mode of this light (should be one of the constants defined by Light.LIGHTMAP_x)\n */\n get lightmapMode() {\n return this._lightmapMode;\n }\n /**\n * Sets the lightmap mode of this light (should be one of the constants defined by Light.LIGHTMAP_x)\n */\n set lightmapMode(value) {\n if (this._lightmapMode === value) {\n return;\n }\n this._lightmapMode = value;\n this._markMeshesAsLightDirty();\n }\n /**\n * Creates a Light object in the scene.\n * Documentation : https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction\n * @param name The friendly name of the light\n * @param scene The scene the light belongs too\n */\n constructor(name, scene) {\n super(name, scene);\n /**\n * Diffuse gives the basic color to an object.\n */\n this.diffuse = new Color3(1.0, 1.0, 1.0);\n /**\n * Specular produces a highlight color on an object.\n * Note: This is not affecting PBR materials.\n */\n this.specular = new Color3(1.0, 1.0, 1.0);\n /**\n * Defines the falloff type for this light. This lets overriding how punctual light are\n * falling off base on range or angle.\n * This can be set to any values in Light.FALLOFF_x.\n *\n * Note: This is only useful for PBR Materials at the moment. This could be extended if required to\n * other types of materials.\n */\n this.falloffType = Light.FALLOFF_DEFAULT;\n /**\n * Strength of the light.\n * Note: By default it is define in the framework own unit.\n * Note: In PBR materials the intensityMode can be use to chose what unit the intensity is defined in.\n */\n this.intensity = 1.0;\n this._range = Number.MAX_VALUE;\n this._inverseSquaredRange = 0;\n /**\n * Cached photometric scale default to 1.0 as the automatic intensity mode defaults to 1.0 for every type\n * of light.\n */\n this._photometricScale = 1.0;\n this._intensityMode = Light.INTENSITYMODE_AUTOMATIC;\n this._radius = 0.00001;\n /**\n * Defines the rendering priority of the lights. It can help in case of fallback or number of lights\n * exceeding the number allowed of the materials.\n */\n this.renderPriority = 0;\n this._shadowEnabled = true;\n this._excludeWithLayerMask = 0;\n this._includeOnlyWithLayerMask = 0;\n this._lightmapMode = 0;\n /**\n * Shadow generators associated to the light.\n * @internal Internal use only.\n */\n this._shadowGenerators = null;\n /**\n * @internal Internal use only.\n */\n this._excludedMeshesIds = new Array();\n /**\n * @internal Internal use only.\n */\n this._includedOnlyMeshesIds = new Array();\n /** @internal */\n this._isLight = true;\n this.getScene().addLight(this);\n this._uniformBuffer = new UniformBuffer(this.getScene().getEngine(), undefined, undefined, name);\n this._buildUniformLayout();\n this.includedOnlyMeshes = new Array();\n this.excludedMeshes = new Array();\n this._resyncMeshes();\n }\n /**\n * Sets the passed Effect \"effect\" with the Light textures.\n * @param effect The effect to update\n * @param lightIndex The index of the light in the effect to update\n * @returns The light\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n transferTexturesToEffect(effect, lightIndex) {\n // Do nothing by default.\n return this;\n }\n /**\n * Binds the lights information from the scene to the effect for the given mesh.\n * @param lightIndex Light index\n * @param scene The scene where the light belongs to\n * @param effect The effect we are binding the data to\n * @param useSpecular Defines if specular is supported\n * @param receiveShadows Defines if the effect (mesh) we bind the light for receives shadows\n */\n _bindLight(lightIndex, scene, effect, useSpecular, receiveShadows = true) {\n var _a;\n const iAsString = lightIndex.toString();\n let needUpdate = false;\n this._uniformBuffer.bindToEffect(effect, \"Light\" + iAsString);\n if (this._renderId !== scene.getRenderId() || this._lastUseSpecular !== useSpecular || !this._uniformBuffer.useUbo) {\n this._renderId = scene.getRenderId();\n this._lastUseSpecular = useSpecular;\n const scaledIntensity = this.getScaledIntensity();\n this.transferToEffect(effect, iAsString);\n this.diffuse.scaleToRef(scaledIntensity, TmpColors.Color3[0]);\n this._uniformBuffer.updateColor4(\"vLightDiffuse\", TmpColors.Color3[0], this.range, iAsString);\n if (useSpecular) {\n this.specular.scaleToRef(scaledIntensity, TmpColors.Color3[1]);\n this._uniformBuffer.updateColor4(\"vLightSpecular\", TmpColors.Color3[1], this.radius, iAsString);\n }\n needUpdate = true;\n }\n // Textures might still need to be rebound.\n this.transferTexturesToEffect(effect, iAsString);\n // Shadows\n if (scene.shadowsEnabled && this.shadowEnabled && receiveShadows) {\n const shadowGenerator = (_a = this.getShadowGenerator(scene.activeCamera)) !== null && _a !== void 0 ? _a : this.getShadowGenerator();\n if (shadowGenerator) {\n shadowGenerator.bindShadowLight(iAsString, effect);\n needUpdate = true;\n }\n }\n if (needUpdate) {\n this._uniformBuffer.update();\n }\n else {\n this._uniformBuffer.bindUniformBuffer();\n }\n }\n /**\n * Returns the string \"Light\".\n * @returns the class name\n */\n getClassName() {\n return \"Light\";\n }\n /**\n * Converts the light information to a readable string for debug purpose.\n * @param fullDetails Supports for multiple levels of logging within scene loading\n * @returns the human readable light info\n */\n toString(fullDetails) {\n let ret = \"Name: \" + this.name;\n ret += \", type: \" + [\"Point\", \"Directional\", \"Spot\", \"Hemispheric\"][this.getTypeID()];\n if (this.animations) {\n for (let i = 0; i < this.animations.length; i++) {\n ret += \", animation[0]: \" + this.animations[i].toString(fullDetails);\n }\n }\n return ret;\n }\n /** @internal */\n _syncParentEnabledState() {\n super._syncParentEnabledState();\n if (!this.isDisposed()) {\n this._resyncMeshes();\n }\n }\n /**\n * Set the enabled state of this node.\n * @param value - the new enabled state\n */\n setEnabled(value) {\n super.setEnabled(value);\n this._resyncMeshes();\n }\n /**\n * Returns the Light associated shadow generator if any.\n * @param camera Camera for which the shadow generator should be retrieved (default: null). If null, retrieves the default shadow generator\n * @returns the associated shadow generator.\n */\n getShadowGenerator(camera = null) {\n var _a;\n if (this._shadowGenerators === null) {\n return null;\n }\n return (_a = this._shadowGenerators.get(camera)) !== null && _a !== void 0 ? _a : null;\n }\n /**\n * Returns all the shadow generators associated to this light\n * @returns\n */\n getShadowGenerators() {\n return this._shadowGenerators;\n }\n /**\n * Returns a Vector3, the absolute light position in the World.\n * @returns the world space position of the light\n */\n getAbsolutePosition() {\n return Vector3.Zero();\n }\n /**\n * Specifies if the light will affect the passed mesh.\n * @param mesh The mesh to test against the light\n * @returns true the mesh is affected otherwise, false.\n */\n canAffectMesh(mesh) {\n if (!mesh) {\n return true;\n }\n if (this.includedOnlyMeshes && this.includedOnlyMeshes.length > 0 && this.includedOnlyMeshes.indexOf(mesh) === -1) {\n return false;\n }\n if (this.excludedMeshes && this.excludedMeshes.length > 0 && this.excludedMeshes.indexOf(mesh) !== -1) {\n return false;\n }\n if (this.includeOnlyWithLayerMask !== 0 && (this.includeOnlyWithLayerMask & mesh.layerMask) === 0) {\n return false;\n }\n if (this.excludeWithLayerMask !== 0 && this.excludeWithLayerMask & mesh.layerMask) {\n return false;\n }\n return true;\n }\n /**\n * Releases resources associated with this node.\n * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default)\n * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default)\n */\n dispose(doNotRecurse, disposeMaterialAndTextures = false) {\n if (this._shadowGenerators) {\n const iterator = this._shadowGenerators.values();\n for (let key = iterator.next(); key.done !== true; key = iterator.next()) {\n const shadowGenerator = key.value;\n shadowGenerator.dispose();\n }\n this._shadowGenerators = null;\n }\n // Animations\n this.getScene().stopAnimation(this);\n if (this._parentContainer) {\n const index = this._parentContainer.lights.indexOf(this);\n if (index > -1) {\n this._parentContainer.lights.splice(index, 1);\n }\n this._parentContainer = null;\n }\n // Remove from meshes\n for (const mesh of this.getScene().meshes) {\n mesh._removeLightSource(this, true);\n }\n this._uniformBuffer.dispose();\n // Remove from scene\n this.getScene().removeLight(this);\n super.dispose(doNotRecurse, disposeMaterialAndTextures);\n }\n /**\n * Returns the light type ID (integer).\n * @returns The light Type id as a constant defines in Light.LIGHTTYPEID_x\n */\n getTypeID() {\n return 0;\n }\n /**\n * Returns the intensity scaled by the Photometric Scale according to the light type and intensity mode.\n * @returns the scaled intensity in intensity mode unit\n */\n getScaledIntensity() {\n return this._photometricScale * this.intensity;\n }\n /**\n * Returns a new Light object, named \"name\", from the current one.\n * @param name The name of the cloned light\n * @param newParent The parent of this light, if it has one\n * @returns the new created light\n */\n clone(name, newParent = null) {\n const constructor = Light.GetConstructorFromName(this.getTypeID(), name, this.getScene());\n if (!constructor) {\n return null;\n }\n const clonedLight = SerializationHelper.Clone(constructor, this);\n if (name) {\n clonedLight.name = name;\n }\n if (newParent) {\n clonedLight.parent = newParent;\n }\n clonedLight.setEnabled(this.isEnabled());\n this.onClonedObservable.notifyObservers(clonedLight);\n return clonedLight;\n }\n /**\n * Serializes the current light into a Serialization object.\n * @returns the serialized object.\n */\n serialize() {\n const serializationObject = SerializationHelper.Serialize(this);\n serializationObject.uniqueId = this.uniqueId;\n // Type\n serializationObject.type = this.getTypeID();\n // Parent\n if (this.parent) {\n this.parent._serializeAsParent(serializationObject);\n }\n // Inclusion / exclusions\n if (this.excludedMeshes.length > 0) {\n serializationObject.excludedMeshesIds = [];\n this.excludedMeshes.forEach((mesh) => {\n serializationObject.excludedMeshesIds.push(mesh.id);\n });\n }\n if (this.includedOnlyMeshes.length > 0) {\n serializationObject.includedOnlyMeshesIds = [];\n this.includedOnlyMeshes.forEach((mesh) => {\n serializationObject.includedOnlyMeshesIds.push(mesh.id);\n });\n }\n // Animations\n SerializationHelper.AppendSerializedAnimations(this, serializationObject);\n serializationObject.ranges = this.serializeAnimationRanges();\n serializationObject.isEnabled = this.isEnabled();\n return serializationObject;\n }\n /**\n * Creates a new typed light from the passed type (integer) : point light = 0, directional light = 1, spot light = 2, hemispheric light = 3.\n * This new light is named \"name\" and added to the passed scene.\n * @param type Type according to the types available in Light.LIGHTTYPEID_x\n * @param name The friendly name of the light\n * @param scene The scene the new light will belong to\n * @returns the constructor function\n */\n static GetConstructorFromName(type, name, scene) {\n const constructorFunc = Node.Construct(\"Light_Type_\" + type, name, scene);\n if (constructorFunc) {\n return constructorFunc;\n }\n // Default to no light for none present once.\n return null;\n }\n /**\n * Parses the passed \"parsedLight\" and returns a new instanced Light from this parsing.\n * @param parsedLight The JSON representation of the light\n * @param scene The scene to create the parsed light in\n * @returns the created light after parsing\n */\n static Parse(parsedLight, scene) {\n const constructor = Light.GetConstructorFromName(parsedLight.type, parsedLight.name, scene);\n if (!constructor) {\n return null;\n }\n const light = SerializationHelper.Parse(constructor, parsedLight, scene);\n // Inclusion / exclusions\n if (parsedLight.excludedMeshesIds) {\n light._excludedMeshesIds = parsedLight.excludedMeshesIds;\n }\n if (parsedLight.includedOnlyMeshesIds) {\n light._includedOnlyMeshesIds = parsedLight.includedOnlyMeshesIds;\n }\n // Parent\n if (parsedLight.parentId !== undefined) {\n light._waitingParentId = parsedLight.parentId;\n }\n if (parsedLight.parentInstanceIndex !== undefined) {\n light._waitingParentInstanceIndex = parsedLight.parentInstanceIndex;\n }\n // Falloff\n if (parsedLight.falloffType !== undefined) {\n light.falloffType = parsedLight.falloffType;\n }\n // Lightmaps\n if (parsedLight.lightmapMode !== undefined) {\n light.lightmapMode = parsedLight.lightmapMode;\n }\n // Animations\n if (parsedLight.animations) {\n for (let animationIndex = 0; animationIndex < parsedLight.animations.length; animationIndex++) {\n const parsedAnimation = parsedLight.animations[animationIndex];\n const internalClass = GetClass(\"BABYLON.Animation\");\n if (internalClass) {\n light.animations.push(internalClass.Parse(parsedAnimation));\n }\n }\n Node.ParseAnimationRanges(light, parsedLight, scene);\n }\n if (parsedLight.autoAnimate) {\n scene.beginAnimation(light, parsedLight.autoAnimateFrom, parsedLight.autoAnimateTo, parsedLight.autoAnimateLoop, parsedLight.autoAnimateSpeed || 1.0);\n }\n // Check if isEnabled is defined to be back compatible with prior serialized versions.\n if (parsedLight.isEnabled !== undefined) {\n light.setEnabled(parsedLight.isEnabled);\n }\n return light;\n }\n _hookArrayForExcluded(array) {\n const oldPush = array.push;\n array.push = (...items) => {\n const result = oldPush.apply(array, items);\n for (const item of items) {\n item._resyncLightSource(this);\n }\n return result;\n };\n const oldSplice = array.splice;\n array.splice = (index, deleteCount) => {\n const deleted = oldSplice.apply(array, [index, deleteCount]);\n for (const item of deleted) {\n item._resyncLightSource(this);\n }\n return deleted;\n };\n for (const item of array) {\n item._resyncLightSource(this);\n }\n }\n _hookArrayForIncludedOnly(array) {\n const oldPush = array.push;\n array.push = (...items) => {\n const result = oldPush.apply(array, items);\n this._resyncMeshes();\n return result;\n };\n const oldSplice = array.splice;\n array.splice = (index, deleteCount) => {\n const deleted = oldSplice.apply(array, [index, deleteCount]);\n this._resyncMeshes();\n return deleted;\n };\n this._resyncMeshes();\n }\n _resyncMeshes() {\n for (const mesh of this.getScene().meshes) {\n mesh._resyncLightSource(this);\n }\n }\n /**\n * Forces the meshes to update their light related information in their rendering used effects\n * @internal Internal Use Only\n */\n _markMeshesAsLightDirty() {\n for (const mesh of this.getScene().meshes) {\n if (mesh.lightSources.indexOf(this) !== -1) {\n mesh._markSubMeshesAsLightDirty();\n }\n }\n }\n /**\n * Recomputes the cached photometric scale if needed.\n */\n _computePhotometricScale() {\n this._photometricScale = this._getPhotometricScale();\n this.getScene().resetCachedMaterial();\n }\n /**\n * Returns the Photometric Scale according to the light type and intensity mode.\n */\n _getPhotometricScale() {\n let photometricScale = 0.0;\n const lightTypeID = this.getTypeID();\n //get photometric mode\n let photometricMode = this.intensityMode;\n if (photometricMode === Light.INTENSITYMODE_AUTOMATIC) {\n if (lightTypeID === Light.LIGHTTYPEID_DIRECTIONALLIGHT) {\n photometricMode = Light.INTENSITYMODE_ILLUMINANCE;\n }\n else {\n photometricMode = Light.INTENSITYMODE_LUMINOUSINTENSITY;\n }\n }\n //compute photometric scale\n switch (lightTypeID) {\n case Light.LIGHTTYPEID_POINTLIGHT:\n case Light.LIGHTTYPEID_SPOTLIGHT:\n switch (photometricMode) {\n case Light.INTENSITYMODE_LUMINOUSPOWER:\n photometricScale = 1.0 / (4.0 * Math.PI);\n break;\n case Light.INTENSITYMODE_LUMINOUSINTENSITY:\n photometricScale = 1.0;\n break;\n case Light.INTENSITYMODE_LUMINANCE:\n photometricScale = this.radius * this.radius;\n break;\n }\n break;\n case Light.LIGHTTYPEID_DIRECTIONALLIGHT:\n switch (photometricMode) {\n case Light.INTENSITYMODE_ILLUMINANCE:\n photometricScale = 1.0;\n break;\n case Light.INTENSITYMODE_LUMINANCE: {\n // When radius (and therefore solid angle) is non-zero a directional lights brightness can be specified via central (peak) luminance.\n // For a directional light the 'radius' defines the angular radius (in radians) rather than world-space radius (e.g. in metres).\n let apexAngleRadians = this.radius;\n // Impose a minimum light angular size to avoid the light becoming an infinitely small angular light source (i.e. a dirac delta function).\n apexAngleRadians = Math.max(apexAngleRadians, 0.001);\n const solidAngle = 2.0 * Math.PI * (1.0 - Math.cos(apexAngleRadians));\n photometricScale = solidAngle;\n break;\n }\n }\n break;\n case Light.LIGHTTYPEID_HEMISPHERICLIGHT:\n // No fall off in hemispheric light.\n photometricScale = 1.0;\n break;\n }\n return photometricScale;\n }\n /**\n * Reorder the light in the scene according to their defined priority.\n * @internal Internal Use Only\n */\n _reorderLightsInScene() {\n const scene = this.getScene();\n if (this._renderPriority != 0) {\n scene.requireLightSorting = true;\n }\n this.getScene().sortLightsByPriority();\n }\n }\n /**\n * Falloff Default: light is falling off following the material specification:\n * standard material is using standard falloff whereas pbr material can request special falloff per materials.\n */\n Light.FALLOFF_DEFAULT = LightConstants.FALLOFF_DEFAULT;\n /**\n * Falloff Physical: light is falling off following the inverse squared distance law.\n */\n Light.FALLOFF_PHYSICAL = LightConstants.FALLOFF_PHYSICAL;\n /**\n * Falloff gltf: light is falling off as described in the gltf moving to PBR document\n * to enhance interoperability with other engines.\n */\n Light.FALLOFF_GLTF = LightConstants.FALLOFF_GLTF;\n /**\n * Falloff Standard: light is falling off like in the standard material\n * to enhance interoperability with other materials.\n */\n Light.FALLOFF_STANDARD = LightConstants.FALLOFF_STANDARD;\n //lightmapMode Consts\n /**\n * If every light affecting the material is in this lightmapMode,\n * material.lightmapTexture adds or multiplies\n * (depends on material.useLightmapAsShadowmap)\n * after every other light calculations.\n */\n Light.LIGHTMAP_DEFAULT = LightConstants.LIGHTMAP_DEFAULT;\n /**\n * material.lightmapTexture as only diffuse lighting from this light\n * adds only specular lighting from this light\n * adds dynamic shadows\n */\n Light.LIGHTMAP_SPECULAR = LightConstants.LIGHTMAP_SPECULAR;\n /**\n * material.lightmapTexture as only lighting\n * no light calculation from this light\n * only adds dynamic shadows from this light\n */\n Light.LIGHTMAP_SHADOWSONLY = LightConstants.LIGHTMAP_SHADOWSONLY;\n // Intensity Mode Consts\n /**\n * Each light type uses the default quantity according to its type:\n * point/spot lights use luminous intensity\n * directional lights use illuminance\n */\n Light.INTENSITYMODE_AUTOMATIC = LightConstants.INTENSITYMODE_AUTOMATIC;\n /**\n * lumen (lm)\n */\n Light.INTENSITYMODE_LUMINOUSPOWER = LightConstants.INTENSITYMODE_LUMINOUSPOWER;\n /**\n * candela (lm/sr)\n */\n Light.INTENSITYMODE_LUMINOUSINTENSITY = LightConstants.INTENSITYMODE_LUMINOUSINTENSITY;\n /**\n * lux (lm/m^2)\n */\n Light.INTENSITYMODE_ILLUMINANCE = LightConstants.INTENSITYMODE_ILLUMINANCE;\n /**\n * nit (cd/m^2)\n */\n Light.INTENSITYMODE_LUMINANCE = LightConstants.INTENSITYMODE_LUMINANCE;\n // Light types ids const.\n /**\n * Light type const id of the point light.\n */\n Light.LIGHTTYPEID_POINTLIGHT = LightConstants.LIGHTTYPEID_POINTLIGHT;\n /**\n * Light type const id of the directional light.\n */\n Light.LIGHTTYPEID_DIRECTIONALLIGHT = LightConstants.LIGHTTYPEID_DIRECTIONALLIGHT;\n /**\n * Light type const id of the spot light.\n */\n Light.LIGHTTYPEID_SPOTLIGHT = LightConstants.LIGHTTYPEID_SPOTLIGHT;\n /**\n * Light type const id of the hemispheric light.\n */\n Light.LIGHTTYPEID_HEMISPHERICLIGHT = LightConstants.LIGHTTYPEID_HEMISPHERICLIGHT;\n __decorate$1([\n serializeAsColor3()\n ], Light.prototype, \"diffuse\", void 0);\n __decorate$1([\n serializeAsColor3()\n ], Light.prototype, \"specular\", void 0);\n __decorate$1([\n serialize()\n ], Light.prototype, \"falloffType\", void 0);\n __decorate$1([\n serialize()\n ], Light.prototype, \"intensity\", void 0);\n __decorate$1([\n serialize()\n ], Light.prototype, \"range\", null);\n __decorate$1([\n serialize()\n ], Light.prototype, \"intensityMode\", null);\n __decorate$1([\n serialize()\n ], Light.prototype, \"radius\", null);\n __decorate$1([\n serialize()\n ], Light.prototype, \"_renderPriority\", void 0);\n __decorate$1([\n expandToProperty(\"_reorderLightsInScene\")\n ], Light.prototype, \"renderPriority\", void 0);\n __decorate$1([\n serialize(\"shadowEnabled\")\n ], Light.prototype, \"_shadowEnabled\", void 0);\n __decorate$1([\n serialize(\"excludeWithLayerMask\")\n ], Light.prototype, \"_excludeWithLayerMask\", void 0);\n __decorate$1([\n serialize(\"includeOnlyWithLayerMask\")\n ], Light.prototype, \"_includeOnlyWithLayerMask\", void 0);\n __decorate$1([\n serialize(\"lightmapMode\")\n ], Light.prototype, \"_lightmapMode\", void 0);\n\n Node.AddNodeConstructor(\"Light_Type_3\", (name, scene) => {\n return () => new HemisphericLight(name, Vector3.Zero(), scene);\n });\n /**\n * The HemisphericLight simulates the ambient environment light,\n * so the passed direction is the light reflection direction, not the incoming direction.\n */\n class HemisphericLight extends Light {\n /**\n * Creates a HemisphericLight object in the scene according to the passed direction (Vector3).\n * The HemisphericLight simulates the ambient environment light, so the passed direction is the light reflection direction, not the incoming direction.\n * The HemisphericLight can't cast shadows.\n * Documentation : https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction\n * @param name The friendly name of the light\n * @param direction The direction of the light reflection\n * @param scene The scene the light belongs to\n */\n constructor(name, direction, scene) {\n super(name, scene);\n /**\n * The groundColor is the light in the opposite direction to the one specified during creation.\n * You can think of the diffuse and specular light as coming from the centre of the object in the given direction and the groundColor light in the opposite direction.\n */\n this.groundColor = new Color3(0.0, 0.0, 0.0);\n this.direction = direction || Vector3.Up();\n }\n _buildUniformLayout() {\n this._uniformBuffer.addUniform(\"vLightData\", 4);\n this._uniformBuffer.addUniform(\"vLightDiffuse\", 4);\n this._uniformBuffer.addUniform(\"vLightSpecular\", 4);\n this._uniformBuffer.addUniform(\"vLightGround\", 3);\n this._uniformBuffer.addUniform(\"shadowsInfo\", 3);\n this._uniformBuffer.addUniform(\"depthValues\", 2);\n this._uniformBuffer.create();\n }\n /**\n * Returns the string \"HemisphericLight\".\n * @returns The class name\n */\n getClassName() {\n return \"HemisphericLight\";\n }\n /**\n * Sets the HemisphericLight direction towards the passed target (Vector3).\n * Returns the updated direction.\n * @param target The target the direction should point to\n * @returns The computed direction\n */\n setDirectionToTarget(target) {\n this.direction = Vector3.Normalize(target.subtract(Vector3.Zero()));\n return this.direction;\n }\n /**\n * Returns the shadow generator associated to the light.\n * @returns Always null for hemispheric lights because it does not support shadows.\n */\n getShadowGenerator() {\n return null;\n }\n /**\n * Sets the passed Effect object with the HemisphericLight normalized direction and color and the passed name (string).\n * @param _effect The effect to update\n * @param lightIndex The index of the light in the effect to update\n * @returns The hemispheric light\n */\n transferToEffect(_effect, lightIndex) {\n const normalizeDirection = Vector3.Normalize(this.direction);\n this._uniformBuffer.updateFloat4(\"vLightData\", normalizeDirection.x, normalizeDirection.y, normalizeDirection.z, 0.0, lightIndex);\n this._uniformBuffer.updateColor3(\"vLightGround\", this.groundColor.scale(this.intensity), lightIndex);\n return this;\n }\n transferToNodeMaterialEffect(effect, lightDataUniformName) {\n const normalizeDirection = Vector3.Normalize(this.direction);\n effect.setFloat3(lightDataUniformName, normalizeDirection.x, normalizeDirection.y, normalizeDirection.z);\n return this;\n }\n /**\n * Computes the world matrix of the node\n * @returns the world matrix\n */\n computeWorldMatrix() {\n if (!this._worldMatrix) {\n this._worldMatrix = Matrix.Identity();\n }\n return this._worldMatrix;\n }\n /**\n * Returns the integer 3.\n * @returns The light Type id as a constant defines in Light.LIGHTTYPEID_x\n */\n getTypeID() {\n return Light.LIGHTTYPEID_HEMISPHERICLIGHT;\n }\n /**\n * Prepares the list of defines specific to the light type.\n * @param defines the list of defines\n * @param lightIndex defines the index of the light for the effect\n */\n prepareLightSpecificDefines(defines, lightIndex) {\n defines[\"HEMILIGHT\" + lightIndex] = true;\n }\n }\n __decorate$1([\n serializeAsColor3()\n ], HemisphericLight.prototype, \"groundColor\", void 0);\n __decorate$1([\n serializeAsVector3()\n ], HemisphericLight.prototype, \"direction\", void 0);\n\n // Do not edit.\n const name$1H = \"kernelBlurVaryingDeclaration\";\n const shader$1H = `varying vec2 sampleCoord{X};`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1H] = shader$1H;\n\n // Do not edit.\n const name$1I = \"packingFunctions\";\n const shader$1I = `vec4 pack(float depth)\n{const vec4 bit_shift=vec4(255.0*255.0*255.0,255.0*255.0,255.0,1.0);const vec4 bit_mask=vec4(0.0,1.0/255.0,1.0/255.0,1.0/255.0);vec4 res=fract(depth*bit_shift);res-=res.xxyz*bit_mask;return res;}\nfloat unpack(vec4 color)\n{const vec4 bit_shift=vec4(1.0/(255.0*255.0*255.0),1.0/(255.0*255.0),1.0/255.0,1.0);return dot(color,bit_shift);}`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1I] = shader$1I;\n\n // Do not edit.\n const name$1J = \"kernelBlurFragment\";\n const shader$1J = `#ifdef DOF\nfactor=sampleCoC(sampleCoord{X}); \ncomputedWeight=KERNEL_WEIGHT{X}*factor;sumOfWeights+=computedWeight;\n#else\ncomputedWeight=KERNEL_WEIGHT{X};\n#endif\n#ifdef PACKEDFLOAT\nblend+=unpack(texture2D(textureSampler,sampleCoord{X}))*computedWeight;\n#else\nblend+=texture2D(textureSampler,sampleCoord{X})*computedWeight;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1J] = shader$1J;\n\n // Do not edit.\n const name$1K = \"kernelBlurFragment2\";\n const shader$1K = `#ifdef DOF\nfactor=sampleCoC(sampleCenter+delta*KERNEL_DEP_OFFSET{X});computedWeight=KERNEL_DEP_WEIGHT{X}*factor;sumOfWeights+=computedWeight;\n#else\ncomputedWeight=KERNEL_DEP_WEIGHT{X};\n#endif\n#ifdef PACKEDFLOAT\nblend+=unpack(texture2D(textureSampler,sampleCenter+delta*KERNEL_DEP_OFFSET{X}))*computedWeight;\n#else\nblend+=texture2D(textureSampler,sampleCenter+delta*KERNEL_DEP_OFFSET{X})*computedWeight;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1K] = shader$1K;\n\n // Do not edit.\n const name$1L = \"kernelBlurPixelShader\";\n const shader$1L = `uniform sampler2D textureSampler;uniform vec2 delta;varying vec2 sampleCenter;\n#ifdef DOF\nuniform sampler2D circleOfConfusionSampler;float sampleCoC(in vec2 offset) {float coc=texture2D(circleOfConfusionSampler,offset).r;return coc; }\n#endif\n#include[0..varyingCount]\n#ifdef PACKEDFLOAT\n#include\n#endif\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{float computedWeight=0.0;\n#ifdef PACKEDFLOAT\nfloat blend=0.;\n#else\nvec4 blend=vec4(0.);\n#endif\n#ifdef DOF\nfloat sumOfWeights=CENTER_WEIGHT; \nfloat factor=0.0;\n#ifdef PACKEDFLOAT\nblend+=unpack(texture2D(textureSampler,sampleCenter))*CENTER_WEIGHT;\n#else\nblend+=texture2D(textureSampler,sampleCenter)*CENTER_WEIGHT;\n#endif\n#endif\n#include[0..varyingCount]\n#include[0..depCount]\n#ifdef PACKEDFLOAT\ngl_FragColor=pack(blend);\n#else\ngl_FragColor=blend;\n#endif\n#ifdef DOF\ngl_FragColor/=sumOfWeights;\n#endif\n}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$1L] = shader$1L;\n\n // Do not edit.\n const name$1M = \"kernelBlurVertex\";\n const shader$1M = `sampleCoord{X}=sampleCenter+delta*KERNEL_OFFSET{X};`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1M] = shader$1M;\n\n // Do not edit.\n const name$1N = \"kernelBlurVertexShader\";\n const shader$1N = `attribute vec2 position;uniform vec2 delta;varying vec2 sampleCenter;\n#include[0..varyingCount]\nconst vec2 madd=vec2(0.5,0.5);\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\nsampleCenter=(position*madd+madd);\n#include[0..varyingCount]\ngl_Position=vec4(position,0.0,1.0);\n#define CUSTOM_VERTEX_MAIN_END\n}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$1N] = shader$1N;\n\n /**\n * The Blur Post Process which blurs an image based on a kernel and direction.\n * Can be used twice in x and y directions to perform a gaussian blur in two passes.\n */\n class BlurPostProcess extends PostProcess {\n /**\n * Sets the length in pixels of the blur sample region\n */\n set kernel(v) {\n if (this._idealKernel === v) {\n return;\n }\n v = Math.max(v, 1);\n this._idealKernel = v;\n this._kernel = this._nearestBestKernel(v);\n if (!this._blockCompilation) {\n this._updateParameters();\n }\n }\n /**\n * Gets the length in pixels of the blur sample region\n */\n get kernel() {\n return this._idealKernel;\n }\n /**\n * Sets whether or not the blur needs to unpack/repack floats\n */\n set packedFloat(v) {\n if (this._packedFloat === v) {\n return;\n }\n this._packedFloat = v;\n if (!this._blockCompilation) {\n this._updateParameters();\n }\n }\n /**\n * Gets whether or not the blur is unpacking/repacking floats\n */\n get packedFloat() {\n return this._packedFloat;\n }\n /**\n * Gets a string identifying the name of the class\n * @returns \"BlurPostProcess\" string\n */\n getClassName() {\n return \"BlurPostProcess\";\n }\n /**\n * Creates a new instance BlurPostProcess\n * @param name The name of the effect.\n * @param direction The direction in which to blur the image.\n * @param kernel The size of the kernel to be used when computing the blur. eg. Size of 3 will blur the center pixel by 2 pixels surrounding it.\n * @param options The required width/height ratio to downsize to before computing the render pass. (Use 1.0 for full size)\n * @param camera The camera to apply the render pass to.\n * @param samplingMode The sampling mode to be used when computing the pass. (default: 0)\n * @param engine The engine which the post process will be applied. (default: current engine)\n * @param reusable If the post process can be reused on the same frame. (default: false)\n * @param textureType Type of textures used when performing the post process. (default: 0)\n * @param defines\n * @param _blockCompilation If compilation of the shader should not be done in the constructor. The updateEffect method can be used to compile the shader at a later time. (default: false)\n * @param textureFormat Format of textures used when performing the post process. (default: TEXTUREFORMAT_RGBA)\n */\n constructor(name, direction, kernel, options, camera, samplingMode = Texture.BILINEAR_SAMPLINGMODE, engine, reusable, textureType = 0, defines = \"\", _blockCompilation = false, textureFormat = 5) {\n super(name, \"kernelBlur\", [\"delta\", \"direction\"], [\"circleOfConfusionSampler\"], options, camera, samplingMode, engine, reusable, null, textureType, \"kernelBlur\", { varyingCount: 0, depCount: 0 }, true, textureFormat);\n this._blockCompilation = _blockCompilation;\n this._packedFloat = false;\n this._staticDefines = \"\";\n this._staticDefines = defines;\n this.direction = direction;\n this.onApplyObservable.add((effect) => {\n if (this._outputTexture) {\n effect.setFloat2(\"delta\", (1 / this._outputTexture.width) * this.direction.x, (1 / this._outputTexture.height) * this.direction.y);\n }\n else {\n effect.setFloat2(\"delta\", (1 / this.width) * this.direction.x, (1 / this.height) * this.direction.y);\n }\n });\n this.kernel = kernel;\n }\n /**\n * Updates the effect with the current post process compile time values and recompiles the shader.\n * @param defines Define statements that should be added at the beginning of the shader. (default: null)\n * @param uniforms Set of uniform variables that will be passed to the shader. (default: null)\n * @param samplers Set of Texture2D variables that will be passed to the shader. (default: null)\n * @param indexParameters The index parameters to be used for babylons include syntax \"#include[0..varyingCount]\". (default: undefined) See usage in babylon.blurPostProcess.ts and kernelBlur.vertex.fx\n * @param onCompiled Called when the shader has been compiled.\n * @param onError Called if there is an error when compiling a shader.\n */\n updateEffect(defines = null, uniforms = null, samplers = null, indexParameters, onCompiled, onError) {\n this._updateParameters(onCompiled, onError);\n }\n _updateParameters(onCompiled, onError) {\n // Generate sampling offsets and weights\n const N = this._kernel;\n const centerIndex = (N - 1) / 2;\n // Generate Gaussian sampling weights over kernel\n let offsets = [];\n let weights = [];\n let totalWeight = 0;\n for (let i = 0; i < N; i++) {\n const u = i / (N - 1);\n const w = this._gaussianWeight(u * 2.0 - 1);\n offsets[i] = i - centerIndex;\n weights[i] = w;\n totalWeight += w;\n }\n // Normalize weights\n for (let i = 0; i < weights.length; i++) {\n weights[i] /= totalWeight;\n }\n // Optimize: combine samples to take advantage of hardware linear sampling\n // Walk from left to center, combining pairs (symmetrically)\n const linearSamplingWeights = [];\n const linearSamplingOffsets = [];\n const linearSamplingMap = [];\n for (let i = 0; i <= centerIndex; i += 2) {\n const j = Math.min(i + 1, Math.floor(centerIndex));\n const singleCenterSample = i === j;\n if (singleCenterSample) {\n linearSamplingMap.push({ o: offsets[i], w: weights[i] });\n }\n else {\n const sharedCell = j === centerIndex;\n const weightLinear = weights[i] + weights[j] * (sharedCell ? 0.5 : 1);\n const offsetLinear = offsets[i] + 1 / (1 + weights[i] / weights[j]);\n if (offsetLinear === 0) {\n linearSamplingMap.push({ o: offsets[i], w: weights[i] });\n linearSamplingMap.push({ o: offsets[i + 1], w: weights[i + 1] });\n }\n else {\n linearSamplingMap.push({ o: offsetLinear, w: weightLinear });\n linearSamplingMap.push({ o: -offsetLinear, w: weightLinear });\n }\n }\n }\n for (let i = 0; i < linearSamplingMap.length; i++) {\n linearSamplingOffsets[i] = linearSamplingMap[i].o;\n linearSamplingWeights[i] = linearSamplingMap[i].w;\n }\n // Replace with optimized\n offsets = linearSamplingOffsets;\n weights = linearSamplingWeights;\n // Generate shaders\n const maxVaryingRows = this.getEngine().getCaps().maxVaryingVectors;\n const freeVaryingVec2 = Math.max(maxVaryingRows, 0) - 1; // Because of sampleCenter\n let varyingCount = Math.min(offsets.length, freeVaryingVec2);\n let defines = \"\";\n defines += this._staticDefines;\n // The DOF fragment should ignore the center pixel when looping as it is handled manually in the fragment shader.\n if (this._staticDefines.indexOf(\"DOF\") != -1) {\n defines += `#define CENTER_WEIGHT ${this._glslFloat(weights[varyingCount - 1])}\\n`;\n varyingCount--;\n }\n for (let i = 0; i < varyingCount; i++) {\n defines += `#define KERNEL_OFFSET${i} ${this._glslFloat(offsets[i])}\\n`;\n defines += `#define KERNEL_WEIGHT${i} ${this._glslFloat(weights[i])}\\n`;\n }\n let depCount = 0;\n for (let i = freeVaryingVec2; i < offsets.length; i++) {\n defines += `#define KERNEL_DEP_OFFSET${depCount} ${this._glslFloat(offsets[i])}\\n`;\n defines += `#define KERNEL_DEP_WEIGHT${depCount} ${this._glslFloat(weights[i])}\\n`;\n depCount++;\n }\n if (this.packedFloat) {\n defines += `#define PACKEDFLOAT 1`;\n }\n this._blockCompilation = false;\n super.updateEffect(defines, null, null, {\n varyingCount: varyingCount,\n depCount: depCount,\n }, onCompiled, onError);\n }\n /**\n * Best kernels are odd numbers that when divided by 2, their integer part is even, so 5, 9 or 13.\n * Other odd kernels optimize correctly but require proportionally more samples, even kernels are\n * possible but will produce minor visual artifacts. Since each new kernel requires a new shader we\n * want to minimize kernel changes, having gaps between physical kernels is helpful in that regard.\n * The gaps between physical kernels are compensated for in the weighting of the samples\n * @param idealKernel Ideal blur kernel.\n * @returns Nearest best kernel.\n */\n _nearestBestKernel(idealKernel) {\n const v = Math.round(idealKernel);\n for (const k of [v, v - 1, v + 1, v - 2, v + 2]) {\n if (k % 2 !== 0 && Math.floor(k / 2) % 2 === 0 && k > 0) {\n return Math.max(k, 3);\n }\n }\n return Math.max(v, 3);\n }\n /**\n * Calculates the value of a Gaussian distribution with sigma 3 at a given point.\n * @param x The point on the Gaussian distribution to sample.\n * @returns the value of the Gaussian function at x.\n */\n _gaussianWeight(x) {\n //reference: Engines/ImageProcessingBlur.cpp #dcc760\n // We are evaluating the Gaussian (normal) distribution over a kernel parameter space of [-1,1],\n // so we truncate at three standard deviations by setting stddev (sigma) to 1/3.\n // The choice of 3-sigma truncation is common but arbitrary, and means that the signal is\n // truncated at around 1.3% of peak strength.\n //the distribution is scaled to account for the difference between the actual kernel size and the requested kernel size\n const sigma = 1 / 3;\n const denominator = Math.sqrt(2.0 * Math.PI) * sigma;\n const exponent = -((x * x) / (2.0 * sigma * sigma));\n const weight = (1.0 / denominator) * Math.exp(exponent);\n return weight;\n }\n /**\n * Generates a string that can be used as a floating point number in GLSL.\n * @param x Value to print.\n * @param decimalFigures Number of decimal places to print the number to (excluding trailing 0s).\n * @returns GLSL float string.\n */\n _glslFloat(x, decimalFigures = 8) {\n return x.toFixed(decimalFigures).replace(/0+$/, \"\");\n }\n /**\n * @internal\n */\n static _Parse(parsedPostProcess, targetCamera, scene, rootUrl) {\n return SerializationHelper.Parse(() => {\n return new BlurPostProcess(parsedPostProcess.name, parsedPostProcess.direction, parsedPostProcess.kernel, parsedPostProcess.options, targetCamera, parsedPostProcess.renderTargetSamplingMode, scene.getEngine(), parsedPostProcess.reusable, parsedPostProcess.textureType, undefined, false);\n }, parsedPostProcess, scene, rootUrl);\n }\n }\n __decorate$1([\n serialize(\"kernel\")\n ], BlurPostProcess.prototype, \"_kernel\", void 0);\n __decorate$1([\n serialize(\"packedFloat\")\n ], BlurPostProcess.prototype, \"_packedFloat\", void 0);\n __decorate$1([\n serializeAsVector2()\n ], BlurPostProcess.prototype, \"direction\", void 0);\n RegisterClass(\"BABYLON.BlurPostProcess\", BlurPostProcess);\n\n /**\n * Mirror texture can be used to simulate the view from a mirror in a scene.\n * It will dynamically be rendered every frame to adapt to the camera point of view.\n * You can then easily use it as a reflectionTexture on a flat surface.\n * In case the surface is not a plane, please consider relying on reflection probes.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#mirrortexture\n */\n class MirrorTexture extends RenderTargetTexture {\n /**\n * Define the blur ratio used to blur the reflection if needed.\n */\n set blurRatio(value) {\n if (this._blurRatio === value) {\n return;\n }\n this._blurRatio = value;\n this._preparePostProcesses();\n }\n get blurRatio() {\n return this._blurRatio;\n }\n /**\n * Define the adaptive blur kernel used to blur the reflection if needed.\n * This will autocompute the closest best match for the `blurKernel`\n */\n set adaptiveBlurKernel(value) {\n this._adaptiveBlurKernel = value;\n this._autoComputeBlurKernel();\n }\n /**\n * Define the blur kernel used to blur the reflection if needed.\n * Please consider using `adaptiveBlurKernel` as it could find the closest best value for you.\n */\n set blurKernel(value) {\n this.blurKernelX = value;\n this.blurKernelY = value;\n }\n /**\n * Define the blur kernel on the X Axis used to blur the reflection if needed.\n * Please consider using `adaptiveBlurKernel` as it could find the closest best value for you.\n */\n set blurKernelX(value) {\n if (this._blurKernelX === value) {\n return;\n }\n this._blurKernelX = value;\n this._preparePostProcesses();\n }\n get blurKernelX() {\n return this._blurKernelX;\n }\n /**\n * Define the blur kernel on the Y Axis used to blur the reflection if needed.\n * Please consider using `adaptiveBlurKernel` as it could find the closest best value for you.\n */\n set blurKernelY(value) {\n if (this._blurKernelY === value) {\n return;\n }\n this._blurKernelY = value;\n this._preparePostProcesses();\n }\n get blurKernelY() {\n return this._blurKernelY;\n }\n _autoComputeBlurKernel() {\n const engine = this.getScene().getEngine();\n const dw = this.getRenderWidth() / engine.getRenderWidth();\n const dh = this.getRenderHeight() / engine.getRenderHeight();\n this.blurKernelX = this._adaptiveBlurKernel * dw;\n this.blurKernelY = this._adaptiveBlurKernel * dh;\n }\n _onRatioRescale() {\n if (this._sizeRatio) {\n this.resize(this._initialSizeParameter);\n if (!this._adaptiveBlurKernel) {\n this._preparePostProcesses();\n }\n }\n if (this._adaptiveBlurKernel) {\n this._autoComputeBlurKernel();\n }\n }\n _updateGammaSpace() {\n const scene = this.getScene();\n if (!scene) {\n return;\n }\n this.gammaSpace = !scene.imageProcessingConfiguration.isEnabled || !scene.imageProcessingConfiguration.applyByPostProcess;\n }\n /**\n * Instantiates a Mirror Texture.\n * Mirror texture can be used to simulate the view from a mirror in a scene.\n * It will dynamically be rendered every frame to adapt to the camera point of view.\n * You can then easily use it as a reflectionTexture on a flat surface.\n * In case the surface is not a plane, please consider relying on reflection probes.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#mirrors\n * @param name\n * @param size\n * @param scene\n * @param generateMipMaps\n * @param type\n * @param samplingMode\n * @param generateDepthBuffer\n */\n constructor(name, size, scene, generateMipMaps, type = 0, samplingMode = Texture.BILINEAR_SAMPLINGMODE, generateDepthBuffer = true) {\n super(name, size, scene, generateMipMaps, true, type, false, samplingMode, generateDepthBuffer);\n /**\n * Define the reflection plane we want to use. The mirrorPlane is usually set to the constructed reflector.\n * It is possible to directly set the mirrorPlane by directly using a Plane(a, b, c, d) where a, b and c give the plane normal vector (a, b, c) and d is a scalar displacement from the mirrorPlane to the origin. However in all but the very simplest of situations it is more straight forward to set it to the reflector as stated in the doc.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#mirrors\n */\n this.mirrorPlane = new Plane(0, 1, 0, 1);\n this._transformMatrix = Matrix.Zero();\n this._mirrorMatrix = Matrix.Zero();\n this._adaptiveBlurKernel = 0;\n this._blurKernelX = 0;\n this._blurKernelY = 0;\n this._blurRatio = 1.0;\n scene = this.getScene();\n if (!scene) {\n return this;\n }\n this.ignoreCameraViewport = true;\n this._updateGammaSpace();\n this._imageProcessingConfigChangeObserver = scene.imageProcessingConfiguration.onUpdateParameters.add(() => {\n this._updateGammaSpace();\n });\n const engine = scene.getEngine();\n if (engine.supportsUniformBuffers) {\n this._sceneUBO = scene.createSceneUniformBuffer(`Scene for Mirror Texture (name \"${name}\")`);\n }\n this.onBeforeBindObservable.add(() => {\n var _a;\n (_a = engine._debugPushGroup) === null || _a === void 0 ? void 0 : _a.call(engine, `mirror generation for ${name}`, 1);\n });\n this.onAfterUnbindObservable.add(() => {\n var _a;\n (_a = engine._debugPopGroup) === null || _a === void 0 ? void 0 : _a.call(engine, 1);\n });\n let saveClipPlane;\n this.onBeforeRenderObservable.add(() => {\n if (this._sceneUBO) {\n this._currentSceneUBO = scene.getSceneUniformBuffer();\n scene.setSceneUniformBuffer(this._sceneUBO);\n scene.getSceneUniformBuffer().unbindEffect();\n }\n Matrix.ReflectionToRef(this.mirrorPlane, this._mirrorMatrix);\n this._mirrorMatrix.multiplyToRef(scene.getViewMatrix(), this._transformMatrix);\n scene.setTransformMatrix(this._transformMatrix, scene.getProjectionMatrix());\n saveClipPlane = scene.clipPlane;\n scene.clipPlane = this.mirrorPlane;\n scene._mirroredCameraPosition = Vector3.TransformCoordinates(scene.activeCamera.globalPosition, this._mirrorMatrix);\n });\n this.onAfterRenderObservable.add(() => {\n if (this._sceneUBO) {\n scene.setSceneUniformBuffer(this._currentSceneUBO);\n }\n scene.updateTransformMatrix();\n scene._mirroredCameraPosition = null;\n scene.clipPlane = saveClipPlane;\n });\n }\n _preparePostProcesses() {\n this.clearPostProcesses(true);\n if (this._blurKernelX && this._blurKernelY) {\n const engine = this.getScene().getEngine();\n const textureType = engine.getCaps().textureFloatRender && engine.getCaps().textureFloatLinearFiltering ? 1 : 2;\n this._blurX = new BlurPostProcess(\"horizontal blur\", new Vector2(1.0, 0), this._blurKernelX, this._blurRatio, null, Texture.BILINEAR_SAMPLINGMODE, engine, false, textureType);\n this._blurX.autoClear = false;\n if (this._blurRatio === 1 && this.samples < 2 && this._texture) {\n this._blurX.inputTexture = this._renderTarget;\n }\n else {\n this._blurX.alwaysForcePOT = true;\n }\n this._blurY = new BlurPostProcess(\"vertical blur\", new Vector2(0, 1.0), this._blurKernelY, this._blurRatio, null, Texture.BILINEAR_SAMPLINGMODE, engine, false, textureType);\n this._blurY.autoClear = false;\n this._blurY.alwaysForcePOT = this._blurRatio !== 1;\n this.addPostProcess(this._blurX);\n this.addPostProcess(this._blurY);\n }\n else {\n if (this._blurY) {\n this.removePostProcess(this._blurY);\n this._blurY.dispose();\n this._blurY = null;\n }\n if (this._blurX) {\n this.removePostProcess(this._blurX);\n this._blurX.dispose();\n this._blurX = null;\n }\n }\n }\n /**\n * Clone the mirror texture.\n * @returns the cloned texture\n */\n clone() {\n const scene = this.getScene();\n if (!scene) {\n return this;\n }\n const textureSize = this.getSize();\n const newTexture = new MirrorTexture(this.name, textureSize.width, scene, this._renderTargetOptions.generateMipMaps, this._renderTargetOptions.type, this._renderTargetOptions.samplingMode, this._renderTargetOptions.generateDepthBuffer);\n // Base texture\n newTexture.hasAlpha = this.hasAlpha;\n newTexture.level = this.level;\n // Mirror Texture\n newTexture.mirrorPlane = this.mirrorPlane.clone();\n if (this.renderList) {\n newTexture.renderList = this.renderList.slice(0);\n }\n return newTexture;\n }\n /**\n * Serialize the texture to a JSON representation you could use in Parse later on\n * @returns the serialized JSON representation\n */\n serialize() {\n if (!this.name) {\n return null;\n }\n const serializationObject = super.serialize();\n serializationObject.mirrorPlane = this.mirrorPlane.asArray();\n return serializationObject;\n }\n /**\n * Dispose the texture and release its associated resources.\n */\n dispose() {\n var _a;\n super.dispose();\n const scene = this.getScene();\n if (scene) {\n scene.imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingConfigChangeObserver);\n }\n (_a = this._sceneUBO) === null || _a === void 0 ? void 0 : _a.dispose();\n }\n }\n Texture._CreateMirror = (name, renderTargetSize, scene, generateMipMaps) => {\n return new MirrorTexture(name, renderTargetSize, scene, generateMipMaps);\n };\n\n ThinEngine.prototype._createDepthStencilCubeTexture = function (size, options, rtWrapper) {\n const internalTexture = new InternalTexture(this, InternalTextureSource.DepthStencil);\n internalTexture.isCube = true;\n if (this.webGLVersion === 1) {\n Logger.Error(\"Depth cube texture is not supported by WebGL 1.\");\n return internalTexture;\n }\n const internalOptions = Object.assign({ bilinearFiltering: false, comparisonFunction: 0, generateStencil: false }, options);\n const gl = this._gl;\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, internalTexture, true);\n this._setupDepthStencilTexture(internalTexture, size, internalOptions.generateStencil, internalOptions.bilinearFiltering, internalOptions.comparisonFunction);\n rtWrapper._depthStencilTexture = internalTexture;\n rtWrapper._depthStencilTextureWithStencil = internalOptions.generateStencil;\n // Create the depth/stencil buffer\n for (let face = 0; face < 6; face++) {\n if (internalOptions.generateStencil) {\n gl.texImage2D(gl.TEXTURE_CUBE_MAP_POSITIVE_X + face, 0, gl.DEPTH24_STENCIL8, size, size, 0, gl.DEPTH_STENCIL, gl.UNSIGNED_INT_24_8, null);\n }\n else {\n gl.texImage2D(gl.TEXTURE_CUBE_MAP_POSITIVE_X + face, 0, gl.DEPTH_COMPONENT24, size, size, 0, gl.DEPTH_COMPONENT, gl.UNSIGNED_INT, null);\n }\n }\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, null);\n this._internalTexturesCache.push(internalTexture);\n return internalTexture;\n };\n ThinEngine.prototype._partialLoadFile = function (url, index, loadedFiles, onfinish, onErrorCallBack = null) {\n const onload = (data) => {\n loadedFiles[index] = data;\n loadedFiles._internalCount++;\n if (loadedFiles._internalCount === 6) {\n onfinish(loadedFiles);\n }\n };\n const onerror = (request, exception) => {\n if (onErrorCallBack && request) {\n onErrorCallBack(request.status + \" \" + request.statusText, exception);\n }\n };\n this._loadFile(url, onload, undefined, undefined, true, onerror);\n };\n ThinEngine.prototype._cascadeLoadFiles = function (scene, onfinish, files, onError = null) {\n const loadedFiles = [];\n loadedFiles._internalCount = 0;\n for (let index = 0; index < 6; index++) {\n this._partialLoadFile(files[index], index, loadedFiles, onfinish, onError);\n }\n };\n ThinEngine.prototype._cascadeLoadImgs = function (scene, texture, onfinish, files, onError = null, mimeType) {\n const loadedImages = [];\n loadedImages._internalCount = 0;\n for (let index = 0; index < 6; index++) {\n this._partialLoadImg(files[index], index, loadedImages, scene, texture, onfinish, onError, mimeType);\n }\n };\n ThinEngine.prototype._partialLoadImg = function (url, index, loadedImages, scene, texture, onfinish, onErrorCallBack = null, mimeType) {\n const tokenPendingData = RandomGUID();\n const onload = (img) => {\n loadedImages[index] = img;\n loadedImages._internalCount++;\n if (scene) {\n scene.removePendingData(tokenPendingData);\n }\n if (loadedImages._internalCount === 6 && onfinish) {\n onfinish(texture, loadedImages);\n }\n };\n const onerror = (message, exception) => {\n if (scene) {\n scene.removePendingData(tokenPendingData);\n }\n if (onErrorCallBack) {\n onErrorCallBack(message, exception);\n }\n };\n LoadImage(url, onload, onerror, scene ? scene.offlineProvider : null, mimeType);\n if (scene) {\n scene.addPendingData(tokenPendingData);\n }\n };\n ThinEngine.prototype._setCubeMapTextureParams = function (texture, loadMipmap, maxLevel) {\n const gl = this._gl;\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_MAG_FILTER, gl.LINEAR);\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_MIN_FILTER, loadMipmap ? gl.LINEAR_MIPMAP_LINEAR : gl.LINEAR);\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n texture.samplingMode = loadMipmap ? 3 : 2;\n if (loadMipmap && this.getCaps().textureMaxLevel && maxLevel !== undefined && maxLevel > 0) {\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_MAX_LEVEL, maxLevel);\n texture._maxLodLevel = maxLevel;\n }\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, null);\n };\n ThinEngine.prototype.createCubeTextureBase = function (rootUrl, scene, files, noMipmap, onLoad = null, onError = null, format, forcedExtension = null, createPolynomials = false, lodScale = 0, lodOffset = 0, fallback = null, beforeLoadCubeDataCallback = null, imageHandler = null, useSRGBBuffer = false) {\n const texture = fallback ? fallback : new InternalTexture(this, InternalTextureSource.Cube);\n texture.isCube = true;\n texture.url = rootUrl;\n texture.generateMipMaps = !noMipmap;\n texture._lodGenerationScale = lodScale;\n texture._lodGenerationOffset = lodOffset;\n texture._useSRGBBuffer = !!useSRGBBuffer && this._caps.supportSRGBBuffers && (this.webGLVersion > 1 || this.isWebGPU || !!noMipmap);\n if (texture !== fallback) {\n texture.label = rootUrl.substring(0, 60); // default label, can be overriden by the caller\n }\n if (!this._doNotHandleContextLost) {\n texture._extension = forcedExtension;\n texture._files = files;\n }\n const originalRootUrl = rootUrl;\n if (this._transformTextureUrl && !fallback) {\n rootUrl = this._transformTextureUrl(rootUrl);\n }\n const rootUrlWithoutUriParams = rootUrl.split(\"?\")[0];\n const lastDot = rootUrlWithoutUriParams.lastIndexOf(\".\");\n const extension = forcedExtension ? forcedExtension : lastDot > -1 ? rootUrlWithoutUriParams.substring(lastDot).toLowerCase() : \"\";\n let loader = null;\n for (const availableLoader of ThinEngine._TextureLoaders) {\n if (availableLoader.canLoad(extension)) {\n loader = availableLoader;\n break;\n }\n }\n const onInternalError = (request, exception) => {\n if (rootUrl === originalRootUrl) {\n if (onError && request) {\n onError(request.status + \" \" + request.statusText, exception);\n }\n }\n else {\n // fall back to the original url if the transformed url fails to load\n Logger.Warn(`Failed to load ${rootUrl}, falling back to the ${originalRootUrl}`);\n this.createCubeTextureBase(originalRootUrl, scene, files, !!noMipmap, onLoad, onError, format, forcedExtension, createPolynomials, lodScale, lodOffset, texture, beforeLoadCubeDataCallback, imageHandler, useSRGBBuffer);\n }\n };\n if (loader) {\n const onloaddata = (data) => {\n if (beforeLoadCubeDataCallback) {\n beforeLoadCubeDataCallback(texture, data);\n }\n loader.loadCubeData(data, texture, createPolynomials, onLoad, onError);\n };\n if (files && files.length === 6) {\n if (loader.supportCascades) {\n this._cascadeLoadFiles(scene, (images) => onloaddata(images.map((image) => new Uint8Array(image))), files, onError);\n }\n else {\n if (onError) {\n onError(\"Textures type does not support cascades.\");\n }\n else {\n Logger.Warn(\"Texture loader does not support cascades.\");\n }\n }\n }\n else {\n this._loadFile(rootUrl, (data) => onloaddata(new Uint8Array(data)), undefined, undefined, true, onInternalError);\n }\n }\n else {\n if (!files) {\n throw new Error(\"Cannot load cubemap because files were not defined\");\n }\n this._cascadeLoadImgs(scene, texture, (texture, imgs) => {\n if (imageHandler) {\n imageHandler(texture, imgs);\n }\n }, files, onError);\n }\n this._internalTexturesCache.push(texture);\n return texture;\n };\n ThinEngine.prototype.createCubeTexture = function (rootUrl, scene, files, noMipmap, onLoad = null, onError = null, format, forcedExtension = null, createPolynomials = false, lodScale = 0, lodOffset = 0, fallback = null, loaderOptions, useSRGBBuffer = false) {\n const gl = this._gl;\n return this.createCubeTextureBase(rootUrl, scene, files, !!noMipmap, onLoad, onError, format, forcedExtension, createPolynomials, lodScale, lodOffset, fallback, (texture) => this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, texture, true), (texture, imgs) => {\n const width = this.needPOTTextures ? ThinEngine.GetExponentOfTwo(imgs[0].width, this._caps.maxCubemapTextureSize) : imgs[0].width;\n const height = width;\n const faces = [\n gl.TEXTURE_CUBE_MAP_POSITIVE_X,\n gl.TEXTURE_CUBE_MAP_POSITIVE_Y,\n gl.TEXTURE_CUBE_MAP_POSITIVE_Z,\n gl.TEXTURE_CUBE_MAP_NEGATIVE_X,\n gl.TEXTURE_CUBE_MAP_NEGATIVE_Y,\n gl.TEXTURE_CUBE_MAP_NEGATIVE_Z,\n ];\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, texture, true);\n this._unpackFlipY(false);\n const internalFormat = format ? this._getInternalFormat(format, texture._useSRGBBuffer) : texture._useSRGBBuffer ? this._glSRGBExtensionValues.SRGB8_ALPHA8 : gl.RGBA;\n let texelFormat = format ? this._getInternalFormat(format) : gl.RGBA;\n if (texture._useSRGBBuffer && this.webGLVersion === 1) {\n texelFormat = internalFormat;\n }\n for (let index = 0; index < faces.length; index++) {\n if (imgs[index].width !== width || imgs[index].height !== height) {\n this._prepareWorkingCanvas();\n if (!this._workingCanvas || !this._workingContext) {\n Logger.Warn(\"Cannot create canvas to resize texture.\");\n return;\n }\n this._workingCanvas.width = width;\n this._workingCanvas.height = height;\n this._workingContext.drawImage(imgs[index], 0, 0, imgs[index].width, imgs[index].height, 0, 0, width, height);\n gl.texImage2D(faces[index], 0, internalFormat, texelFormat, gl.UNSIGNED_BYTE, this._workingCanvas);\n }\n else {\n gl.texImage2D(faces[index], 0, internalFormat, texelFormat, gl.UNSIGNED_BYTE, imgs[index]);\n }\n }\n if (!noMipmap) {\n gl.generateMipmap(gl.TEXTURE_CUBE_MAP);\n }\n this._setCubeMapTextureParams(texture, !noMipmap);\n texture.width = width;\n texture.height = height;\n texture.isReady = true;\n if (format) {\n texture.format = format;\n }\n texture.onLoadedObservable.notifyObservers(texture);\n texture.onLoadedObservable.clear();\n if (onLoad) {\n onLoad();\n }\n }, !!useSRGBBuffer);\n };\n\n /**\n * Class for creating a cube texture\n */\n class CubeTexture extends BaseTexture {\n /**\n * Gets or sets the size of the bounding box associated with the cube texture\n * When defined, the cubemap will switch to local mode\n * @see https://community.arm.com/graphics/b/blog/posts/reflections-based-on-local-cubemaps-in-unity\n * @example https://www.babylonjs-playground.com/#RNASML\n */\n set boundingBoxSize(value) {\n if (this._boundingBoxSize && this._boundingBoxSize.equals(value)) {\n return;\n }\n this._boundingBoxSize = value;\n const scene = this.getScene();\n if (scene) {\n scene.markAllMaterialsAsDirty(1);\n }\n }\n /**\n * Returns the bounding box size\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#using-local-cubemap-mode\n */\n get boundingBoxSize() {\n return this._boundingBoxSize;\n }\n /**\n * Sets texture matrix rotation angle around Y axis in radians.\n */\n set rotationY(value) {\n this._rotationY = value;\n this.setReflectionTextureMatrix(Matrix.RotationY(this._rotationY));\n }\n /**\n * Gets texture matrix rotation angle around Y axis radians.\n */\n get rotationY() {\n return this._rotationY;\n }\n /**\n * Are mip maps generated for this texture or not.\n */\n get noMipmap() {\n return this._noMipmap;\n }\n /**\n * Gets the forced extension (if any)\n */\n get forcedExtension() {\n return this._forcedExtension;\n }\n /**\n * Creates a cube texture from an array of image urls\n * @param files defines an array of image urls\n * @param scene defines the hosting scene\n * @param noMipmap specifies if mip maps are not used\n * @returns a cube texture\n */\n static CreateFromImages(files, scene, noMipmap) {\n let rootUrlKey = \"\";\n files.forEach((url) => (rootUrlKey += url));\n return new CubeTexture(rootUrlKey, scene, null, noMipmap, files);\n }\n /**\n * Creates and return a texture created from prefilterd data by tools like IBL Baker or Lys.\n * @param url defines the url of the prefiltered texture\n * @param scene defines the scene the texture is attached to\n * @param forcedExtension defines the extension of the file if different from the url\n * @param createPolynomials defines whether or not to create polynomial harmonics from the texture data if necessary\n * @returns the prefiltered texture\n */\n static CreateFromPrefilteredData(url, scene, forcedExtension = null, createPolynomials = true) {\n const oldValue = scene.useDelayedTextureLoading;\n scene.useDelayedTextureLoading = false;\n const result = new CubeTexture(url, scene, null, false, null, null, null, undefined, true, forcedExtension, createPolynomials);\n scene.useDelayedTextureLoading = oldValue;\n return result;\n }\n /**\n * Creates a cube texture to use with reflection for instance. It can be based upon dds or six images as well\n * as prefiltered data.\n * @param rootUrl defines the url of the texture or the root name of the six images\n * @param sceneOrEngine defines the scene or engine the texture is attached to\n * @param extensions defines the suffixes add to the picture name in case six images are in use like _px.jpg...\n * @param noMipmap defines if mipmaps should be created or not\n * @param files defines the six files to load for the different faces in that order: px, py, pz, nx, ny, nz\n * @param onLoad defines a callback triggered at the end of the file load if no errors occurred\n * @param onError defines a callback triggered in case of error during load\n * @param format defines the internal format to use for the texture once loaded\n * @param prefiltered defines whether or not the texture is created from prefiltered data\n * @param forcedExtension defines the extensions to use (force a special type of file to load) in case it is different from the file name\n * @param createPolynomials defines whether or not to create polynomial harmonics from the texture data if necessary\n * @param lodScale defines the scale applied to environment texture. This manages the range of LOD level used for IBL according to the roughness\n * @param lodOffset defines the offset applied to environment texture. This manages first LOD level used for IBL according to the roughness\n * @param loaderOptions options to be passed to the loader\n * @param useSRGBBuffer Defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU) (default: false)\n * @returns the cube texture\n */\n constructor(rootUrl, sceneOrEngine, extensions = null, noMipmap = false, files = null, onLoad = null, onError = null, format = 5, prefiltered = false, forcedExtension = null, createPolynomials = false, lodScale = 0.8, lodOffset = 0, loaderOptions, useSRGBBuffer) {\n var _a;\n super(sceneOrEngine);\n this._lodScale = 0.8;\n this._lodOffset = 0;\n /**\n * Observable triggered once the texture has been loaded.\n */\n this.onLoadObservable = new Observable$1();\n /**\n * Gets or sets the center of the bounding box associated with the cube texture.\n * It must define where the camera used to render the texture was set\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#using-local-cubemap-mode\n */\n this.boundingBoxPosition = Vector3.Zero();\n this._rotationY = 0;\n /** @internal */\n this._files = null;\n this._forcedExtension = null;\n this._extensions = null;\n this._textureMatrixRefraction = new Matrix();\n this.name = rootUrl;\n this.url = rootUrl;\n this._noMipmap = noMipmap;\n this.hasAlpha = false;\n this._format = format;\n this.isCube = true;\n this._textureMatrix = Matrix.Identity();\n this._createPolynomials = createPolynomials;\n this.coordinatesMode = Texture.CUBIC_MODE;\n this._extensions = extensions;\n this._files = files;\n this._forcedExtension = forcedExtension;\n this._loaderOptions = loaderOptions;\n this._useSRGBBuffer = useSRGBBuffer;\n this._lodScale = lodScale;\n this._lodOffset = lodOffset;\n if (!rootUrl && !files) {\n return;\n }\n this.updateURL(rootUrl, forcedExtension, onLoad, prefiltered, onError, extensions, (_a = this.getScene()) === null || _a === void 0 ? void 0 : _a.useDelayedTextureLoading, files);\n }\n /**\n * Get the current class name of the texture useful for serialization or dynamic coding.\n * @returns \"CubeTexture\"\n */\n getClassName() {\n return \"CubeTexture\";\n }\n /**\n * Update the url (and optional buffer) of this texture if url was null during construction.\n * @param url the url of the texture\n * @param forcedExtension defines the extension to use\n * @param onLoad callback called when the texture is loaded (defaults to null)\n * @param prefiltered Defines whether the updated texture is prefiltered or not\n * @param onError callback called if there was an error during the loading process (defaults to null)\n * @param extensions defines the suffixes add to the picture name in case six images are in use like _px.jpg...\n * @param delayLoad defines if the texture should be loaded now (false by default)\n * @param files defines the six files to load for the different faces in that order: px, py, pz, nx, ny, nz\n */\n updateURL(url, forcedExtension, onLoad = null, prefiltered = false, onError = null, extensions = null, delayLoad = false, files = null) {\n if (!this.name || this.name.startsWith(\"data:\")) {\n this.name = url;\n }\n this.url = url;\n if (forcedExtension) {\n this._forcedExtension = forcedExtension;\n }\n const lastDot = url.lastIndexOf(\".\");\n const extension = forcedExtension ? forcedExtension : lastDot > -1 ? url.substring(lastDot).toLowerCase() : \"\";\n const isDDS = extension.indexOf(\".dds\") === 0;\n const isEnv = extension.indexOf(\".env\") === 0;\n const isBasis = extension.indexOf(\".basis\") === 0;\n if (isEnv) {\n this.gammaSpace = false;\n this._prefiltered = false;\n this.anisotropicFilteringLevel = 1;\n }\n else {\n this._prefiltered = prefiltered;\n if (prefiltered) {\n this.gammaSpace = false;\n this.anisotropicFilteringLevel = 1;\n }\n }\n if (files) {\n this._files = files;\n }\n else {\n if (!isBasis && !isEnv && !isDDS && !extensions) {\n extensions = [\"_px.jpg\", \"_py.jpg\", \"_pz.jpg\", \"_nx.jpg\", \"_ny.jpg\", \"_nz.jpg\"];\n }\n this._files = this._files || [];\n this._files.length = 0;\n if (extensions) {\n for (let index = 0; index < extensions.length; index++) {\n this._files.push(url + extensions[index]);\n }\n this._extensions = extensions;\n }\n }\n if (delayLoad) {\n this.delayLoadState = 4;\n this._delayedOnLoad = onLoad;\n this._delayedOnError = onError;\n }\n else {\n this._loadTexture(onLoad, onError);\n }\n }\n /**\n * Delays loading of the cube texture\n * @param forcedExtension defines the extension to use\n */\n delayLoad(forcedExtension) {\n if (this.delayLoadState !== 4) {\n return;\n }\n if (forcedExtension) {\n this._forcedExtension = forcedExtension;\n }\n this.delayLoadState = 1;\n this._loadTexture(this._delayedOnLoad, this._delayedOnError);\n }\n /**\n * Returns the reflection texture matrix\n * @returns the reflection texture matrix\n */\n getReflectionTextureMatrix() {\n return this._textureMatrix;\n }\n /**\n * Sets the reflection texture matrix\n * @param value Reflection texture matrix\n */\n setReflectionTextureMatrix(value) {\n var _a, _b;\n if (value.updateFlag === this._textureMatrix.updateFlag) {\n return;\n }\n if (value.isIdentity() !== this._textureMatrix.isIdentity()) {\n (_a = this.getScene()) === null || _a === void 0 ? void 0 : _a.markAllMaterialsAsDirty(1, (mat) => mat.getActiveTextures().indexOf(this) !== -1);\n }\n this._textureMatrix = value;\n if (!((_b = this.getScene()) === null || _b === void 0 ? void 0 : _b.useRightHandedSystem)) {\n return;\n }\n const scale = TmpVectors.Vector3[0];\n const quat = TmpVectors.Quaternion[0];\n const trans = TmpVectors.Vector3[1];\n this._textureMatrix.decompose(scale, quat, trans);\n quat.z *= -1; // these two operations correspond to negating the x and y euler angles\n quat.w *= -1;\n Matrix.ComposeToRef(scale, quat, trans, this._textureMatrixRefraction);\n }\n /**\n * Gets a suitable rotate/transform matrix when the texture is used for refraction.\n * There's a separate function from getReflectionTextureMatrix because refraction requires a special configuration of the matrix in right-handed mode.\n * @returns The refraction matrix\n */\n getRefractionTextureMatrix() {\n var _a;\n return ((_a = this.getScene()) === null || _a === void 0 ? void 0 : _a.useRightHandedSystem) ? this._textureMatrixRefraction : this._textureMatrix;\n }\n _loadTexture(onLoad = null, onError = null) {\n var _a;\n const scene = this.getScene();\n const oldTexture = this._texture;\n this._texture = this._getFromCache(this.url, this._noMipmap, undefined, undefined, this._useSRGBBuffer, this.isCube);\n const onLoadProcessing = () => {\n var _a;\n this.onLoadObservable.notifyObservers(this);\n if (oldTexture) {\n oldTexture.dispose();\n (_a = this.getScene()) === null || _a === void 0 ? void 0 : _a.markAllMaterialsAsDirty(1);\n }\n if (onLoad) {\n onLoad();\n }\n };\n const errorHandler = (message, exception) => {\n this._loadingError = true;\n this._errorObject = { message, exception };\n if (onError) {\n onError(message, exception);\n }\n Texture.OnTextureLoadErrorObservable.notifyObservers(this);\n };\n if (!this._texture) {\n if (this._prefiltered) {\n this._texture = this._getEngine().createPrefilteredCubeTexture(this.url, scene, this._lodScale, this._lodOffset, onLoad, errorHandler, this._format, this._forcedExtension, this._createPolynomials);\n }\n else {\n this._texture = this._getEngine().createCubeTexture(this.url, scene, this._files, this._noMipmap, onLoad, errorHandler, this._format, this._forcedExtension, false, this._lodScale, this._lodOffset, null, this._loaderOptions, !!this._useSRGBBuffer);\n }\n (_a = this._texture) === null || _a === void 0 ? void 0 : _a.onLoadedObservable.add(() => this.onLoadObservable.notifyObservers(this));\n }\n else {\n if (this._texture.isReady) {\n Tools.SetImmediate(() => onLoadProcessing());\n }\n else {\n this._texture.onLoadedObservable.add(() => onLoadProcessing());\n }\n }\n }\n /**\n * Parses text to create a cube texture\n * @param parsedTexture define the serialized text to read from\n * @param scene defines the hosting scene\n * @param rootUrl defines the root url of the cube texture\n * @returns a cube texture\n */\n static Parse(parsedTexture, scene, rootUrl) {\n const texture = SerializationHelper.Parse(() => {\n var _a;\n let prefiltered = false;\n if (parsedTexture.prefiltered) {\n prefiltered = parsedTexture.prefiltered;\n }\n return new CubeTexture(rootUrl + ((_a = parsedTexture.url) !== null && _a !== void 0 ? _a : parsedTexture.name), scene, parsedTexture.extensions, false, parsedTexture.files || null, null, null, undefined, prefiltered, parsedTexture.forcedExtension);\n }, parsedTexture, scene);\n // Local Cubemaps\n if (parsedTexture.boundingBoxPosition) {\n texture.boundingBoxPosition = Vector3.FromArray(parsedTexture.boundingBoxPosition);\n }\n if (parsedTexture.boundingBoxSize) {\n texture.boundingBoxSize = Vector3.FromArray(parsedTexture.boundingBoxSize);\n }\n // Animations\n if (parsedTexture.animations) {\n for (let animationIndex = 0; animationIndex < parsedTexture.animations.length; animationIndex++) {\n const parsedAnimation = parsedTexture.animations[animationIndex];\n const internalClass = GetClass(\"BABYLON.Animation\");\n if (internalClass) {\n texture.animations.push(internalClass.Parse(parsedAnimation));\n }\n }\n }\n return texture;\n }\n /**\n * Makes a clone, or deep copy, of the cube texture\n * @returns a new cube texture\n */\n clone() {\n let uniqueId = 0;\n const newCubeTexture = SerializationHelper.Clone(() => {\n const cubeTexture = new CubeTexture(this.url, this.getScene() || this._getEngine(), this._extensions, this._noMipmap, this._files);\n uniqueId = cubeTexture.uniqueId;\n return cubeTexture;\n }, this);\n newCubeTexture.uniqueId = uniqueId;\n return newCubeTexture;\n }\n }\n __decorate$1([\n serialize()\n ], CubeTexture.prototype, \"url\", void 0);\n __decorate$1([\n serializeAsVector3()\n ], CubeTexture.prototype, \"boundingBoxPosition\", void 0);\n __decorate$1([\n serializeAsVector3()\n ], CubeTexture.prototype, \"boundingBoxSize\", null);\n __decorate$1([\n serialize(\"rotationY\")\n ], CubeTexture.prototype, \"rotationY\", null);\n __decorate$1([\n serialize(\"files\")\n ], CubeTexture.prototype, \"_files\", void 0);\n __decorate$1([\n serialize(\"forcedExtension\")\n ], CubeTexture.prototype, \"_forcedExtension\", void 0);\n __decorate$1([\n serialize(\"extensions\")\n ], CubeTexture.prototype, \"_extensions\", void 0);\n __decorate$1([\n serializeAsMatrix(\"textureMatrix\")\n ], CubeTexture.prototype, \"_textureMatrix\", void 0);\n __decorate$1([\n serializeAsMatrix(\"textureMatrixRefraction\")\n ], CubeTexture.prototype, \"_textureMatrixRefraction\", void 0);\n Texture._CubeTextureParser = CubeTexture.Parse;\n // Some exporters relies on Tools.Instantiate\n RegisterClass(\"BABYLON.CubeTexture\", CubeTexture);\n\n // Do not edit.\n const name$1O = \"backgroundFragmentDeclaration\";\n const shader$1O = `uniform vec4 vEyePosition;uniform vec4 vPrimaryColor;\n#ifdef USEHIGHLIGHTANDSHADOWCOLORS\nuniform vec4 vPrimaryColorShadow;\n#endif\nuniform float shadowLevel;uniform float alpha;\n#ifdef DIFFUSE\nuniform vec2 vDiffuseInfos;\n#endif\n#ifdef REFLECTION\nuniform vec2 vReflectionInfos;uniform mat4 reflectionMatrix;uniform vec3 vReflectionMicrosurfaceInfos;\n#endif\n#if defined(REFLECTIONFRESNEL) || defined(OPACITYFRESNEL)\nuniform vec3 vBackgroundCenter;\n#endif\n#ifdef REFLECTIONFRESNEL\nuniform vec4 vReflectionControl;\n#endif\n#if defined(REFLECTIONMAP_SPHERICAL) || defined(REFLECTIONMAP_PROJECTION) || defined(REFRACTION)\nuniform mat4 view;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1O] = shader$1O;\n\n // Do not edit.\n const name$1P = \"backgroundUboDeclaration\";\n const shader$1P = `layout(std140,column_major) uniform;uniform Material\n{uniform vec4 vPrimaryColor;uniform vec4 vPrimaryColorShadow;uniform vec2 vDiffuseInfos;uniform vec2 vReflectionInfos;uniform mat4 diffuseMatrix;uniform mat4 reflectionMatrix;uniform vec3 vReflectionMicrosurfaceInfos;uniform float fFovMultiplier;uniform float pointSize;uniform float shadowLevel;uniform float alpha;\n#if defined(REFLECTIONFRESNEL) || defined(OPACITYFRESNEL)\nuniform vec3 vBackgroundCenter;\n#endif\n#ifdef REFLECTIONFRESNEL\nuniform vec4 vReflectionControl;\n#endif\n};\n#include\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1P] = shader$1P;\n\n // Do not edit.\n const name$1Q = \"backgroundPixelShader\";\n const shader$1Q = `#ifdef TEXTURELODSUPPORT\n#extension GL_EXT_shader_texture_lod : enable\n#endif\nprecision highp float;\n#include<__decl__backgroundFragment>\n#include\n#define RECIPROCAL_PI2 0.15915494\nvarying vec3 vPositionW;\n#ifdef MAINUV1\nvarying vec2 vMainUV1;\n#endif \n#ifdef MAINUV2 \nvarying vec2 vMainUV2; \n#endif \n#ifdef NORMAL\nvarying vec3 vNormalW;\n#endif\n#ifdef DIFFUSE\n#if DIFFUSEDIRECTUV==1\n#define vDiffuseUV vMainUV1\n#elif DIFFUSEDIRECTUV==2\n#define vDiffuseUV vMainUV2\n#else\nvarying vec2 vDiffuseUV;\n#endif\nuniform sampler2D diffuseSampler;\n#endif\n#ifdef REFLECTION\n#ifdef REFLECTIONMAP_3D\n#define sampleReflection(s,c) textureCube(s,c)\nuniform samplerCube reflectionSampler;\n#ifdef TEXTURELODSUPPORT\n#define sampleReflectionLod(s,c,l) textureCubeLodEXT(s,c,l)\n#else\nuniform samplerCube reflectionSamplerLow;uniform samplerCube reflectionSamplerHigh;\n#endif\n#else\n#define sampleReflection(s,c) texture2D(s,c)\nuniform sampler2D reflectionSampler;\n#ifdef TEXTURELODSUPPORT\n#define sampleReflectionLod(s,c,l) texture2DLodEXT(s,c,l)\n#else\nuniform samplerCube reflectionSamplerLow;uniform samplerCube reflectionSamplerHigh;\n#endif\n#endif\n#ifdef REFLECTIONMAP_SKYBOX\nvarying vec3 vPositionUVW;\n#else\n#if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED)\nvarying vec3 vDirectionW;\n#endif\n#endif\n#include\n#endif\n#ifndef FROMLINEARSPACE\n#define FROMLINEARSPACE;\n#endif\n#ifndef SHADOWONLY\n#define SHADOWONLY;\n#endif\n#include\n#include<__decl__lightFragment>[0..maxSimultaneousLights]\n#include\n#include\n#include\n#include\n#include\n#ifdef REFLECTIONFRESNEL\n#define FRESNEL_MAXIMUM_ON_ROUGH 0.25\nvec3 fresnelSchlickEnvironmentGGX(float VdotN,vec3 reflectance0,vec3 reflectance90,float smoothness)\n{float weight=mix(FRESNEL_MAXIMUM_ON_ROUGH,1.0,smoothness);return reflectance0+weight*(reflectance90-reflectance0)*pow5(saturate(1.0-VdotN));}\n#endif\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_FRAGMENT_MAIN_BEGIN\n#include\nvec3 viewDirectionW=normalize(vEyePosition.xyz-vPositionW);\n#ifdef NORMAL\nvec3 normalW=normalize(vNormalW);\n#else\nvec3 normalW=vec3(0.0,1.0,0.0);\n#endif\nfloat shadow=1.;float globalShadow=0.;float shadowLightCount=0.;\n#include[0..maxSimultaneousLights]\n#ifdef SHADOWINUSE\nglobalShadow/=shadowLightCount;\n#else\nglobalShadow=1.0;\n#endif\n#ifndef BACKMAT_SHADOWONLY\nvec4 reflectionColor=vec4(1.,1.,1.,1.);\n#ifdef REFLECTION\nvec3 reflectionVector=computeReflectionCoords(vec4(vPositionW,1.0),normalW);\n#ifdef REFLECTIONMAP_OPPOSITEZ\nreflectionVector.z*=-1.0;\n#endif\n#ifdef REFLECTIONMAP_3D\nvec3 reflectionCoords=reflectionVector;\n#else\nvec2 reflectionCoords=reflectionVector.xy;\n#ifdef REFLECTIONMAP_PROJECTION\nreflectionCoords/=reflectionVector.z;\n#endif\nreflectionCoords.y=1.0-reflectionCoords.y;\n#endif\n#ifdef REFLECTIONBLUR\nfloat reflectionLOD=vReflectionInfos.y;\n#ifdef TEXTURELODSUPPORT\nreflectionLOD=reflectionLOD*log2(vReflectionMicrosurfaceInfos.x)*vReflectionMicrosurfaceInfos.y+vReflectionMicrosurfaceInfos.z;reflectionColor=sampleReflectionLod(reflectionSampler,reflectionCoords,reflectionLOD);\n#else\nfloat lodReflectionNormalized=saturate(reflectionLOD);float lodReflectionNormalizedDoubled=lodReflectionNormalized*2.0;vec4 reflectionSpecularMid=sampleReflection(reflectionSampler,reflectionCoords);if(lodReflectionNormalizedDoubled<1.0){reflectionColor=mix(\nsampleReflection(reflectionSamplerHigh,reflectionCoords),\nreflectionSpecularMid,\nlodReflectionNormalizedDoubled\n);} else {reflectionColor=mix(\nreflectionSpecularMid,\nsampleReflection(reflectionSamplerLow,reflectionCoords),\nlodReflectionNormalizedDoubled-1.0\n);}\n#endif\n#else\nvec4 reflectionSample=sampleReflection(reflectionSampler,reflectionCoords);reflectionColor=reflectionSample;\n#endif\n#ifdef RGBDREFLECTION\nreflectionColor.rgb=fromRGBD(reflectionColor);\n#endif\n#ifdef GAMMAREFLECTION\nreflectionColor.rgb=toLinearSpace(reflectionColor.rgb);\n#endif\n#ifdef REFLECTIONBGR\nreflectionColor.rgb=reflectionColor.bgr;\n#endif\nreflectionColor.rgb*=vReflectionInfos.x;\n#endif\nvec3 diffuseColor=vec3(1.,1.,1.);float finalAlpha=alpha;\n#ifdef DIFFUSE\nvec4 diffuseMap=texture2D(diffuseSampler,vDiffuseUV);\n#ifdef GAMMADIFFUSE\ndiffuseMap.rgb=toLinearSpace(diffuseMap.rgb);\n#endif\ndiffuseMap.rgb*=vDiffuseInfos.y;\n#ifdef DIFFUSEHASALPHA\nfinalAlpha*=diffuseMap.a;\n#endif\ndiffuseColor=diffuseMap.rgb;\n#endif\n#ifdef REFLECTIONFRESNEL\nvec3 colorBase=diffuseColor;\n#else\nvec3 colorBase=reflectionColor.rgb*diffuseColor;\n#endif\ncolorBase=max(colorBase,0.0);\n#ifdef USERGBCOLOR\nvec3 finalColor=colorBase;\n#else\n#ifdef USEHIGHLIGHTANDSHADOWCOLORS\nvec3 mainColor=mix(vPrimaryColorShadow.rgb,vPrimaryColor.rgb,colorBase);\n#else\nvec3 mainColor=vPrimaryColor.rgb;\n#endif\nvec3 finalColor=colorBase*mainColor;\n#endif\n#ifdef REFLECTIONFRESNEL\nvec3 reflectionAmount=vReflectionControl.xxx;vec3 reflectionReflectance0=vReflectionControl.yyy;vec3 reflectionReflectance90=vReflectionControl.zzz;float VdotN=dot(normalize(vEyePosition.xyz),normalW);vec3 planarReflectionFresnel=fresnelSchlickEnvironmentGGX(saturate(VdotN),reflectionReflectance0,reflectionReflectance90,1.0);reflectionAmount*=planarReflectionFresnel;\n#ifdef REFLECTIONFALLOFF\nfloat reflectionDistanceFalloff=1.0-saturate(length(vPositionW.xyz-vBackgroundCenter)*vReflectionControl.w);reflectionDistanceFalloff*=reflectionDistanceFalloff;reflectionAmount*=reflectionDistanceFalloff;\n#endif\nfinalColor=mix(finalColor,reflectionColor.rgb,saturate(reflectionAmount));\n#endif\n#ifdef OPACITYFRESNEL\nfloat viewAngleToFloor=dot(normalW,normalize(vEyePosition.xyz-vBackgroundCenter));const float startAngle=0.1;float fadeFactor=saturate(viewAngleToFloor/startAngle);finalAlpha*=fadeFactor*fadeFactor;\n#endif\n#ifdef SHADOWINUSE\nfinalColor=mix(finalColor*shadowLevel,finalColor,globalShadow);\n#endif\nvec4 color=vec4(finalColor,finalAlpha);\n#else\nvec4 color=vec4(vPrimaryColor.rgb,(1.0-clamp(globalShadow,0.,1.))*alpha);\n#endif\n#include\n#ifdef IMAGEPROCESSINGPOSTPROCESS\n#if !defined(SKIPFINALCOLORCLAMP)\ncolor.rgb=clamp(color.rgb,0.,30.0);\n#endif\n#else\ncolor=applyImageProcessing(color);\n#endif\n#ifdef PREMULTIPLYALPHA\ncolor.rgb*=color.a;\n#endif\n#ifdef NOISE\ncolor.rgb+=dither(vPositionW.xy,0.5);color=max(color,0.0);\n#endif\ngl_FragColor=color;\n#define CUSTOM_FRAGMENT_MAIN_END\n}\n`;\n // Sideeffect\n ShaderStore.ShadersStore[name$1Q] = shader$1Q;\n\n // Do not edit.\n const name$1R = \"backgroundVertexDeclaration\";\n const shader$1R = `uniform mat4 view;uniform mat4 viewProjection;uniform float shadowLevel;\n#ifdef DIFFUSE\nuniform mat4 diffuseMatrix;uniform vec2 vDiffuseInfos;\n#endif\n#ifdef REFLECTION\nuniform vec2 vReflectionInfos;uniform mat4 reflectionMatrix;uniform vec3 vReflectionMicrosurfaceInfos;uniform float fFovMultiplier;\n#endif\n#ifdef POINTSIZE\nuniform float pointSize;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$1R] = shader$1R;\n\n // Do not edit.\n const name$1S = \"backgroundVertexShader\";\n const shader$1S = `precision highp float;\n#include<__decl__backgroundVertex>\n#include\nattribute vec3 position;\n#ifdef NORMAL\nattribute vec3 normal;\n#endif\n#include\n#include\n#include\nvarying vec3 vPositionW;\n#ifdef NORMAL\nvarying vec3 vNormalW;\n#endif\n#ifdef UV1\nattribute vec2 uv;\n#endif\n#ifdef UV2\nattribute vec2 uv2;\n#endif\n#ifdef MAINUV1\nvarying vec2 vMainUV1;\n#endif\n#ifdef MAINUV2\nvarying vec2 vMainUV2;\n#endif\n#if defined(DIFFUSE) && DIFFUSEDIRECTUV==0\nvarying vec2 vDiffuseUV;\n#endif\n#include\n#include\n#include<__decl__lightVxFragment>[0..maxSimultaneousLights]\n#ifdef REFLECTIONMAP_SKYBOX\nvarying vec3 vPositionUVW;\n#endif\n#if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED)\nvarying vec3 vDirectionW;\n#endif\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\n#ifdef REFLECTIONMAP_SKYBOX\nvPositionUVW=position;\n#endif\n#include\n#include\n#include\n#ifdef MULTIVIEW\nif (gl_ViewID_OVR==0u) {gl_Position=viewProjection*finalWorld*vec4(position,1.0);} else {gl_Position=viewProjectionR*finalWorld*vec4(position,1.0);}\n#else\ngl_Position=viewProjection*finalWorld*vec4(position,1.0);\n#endif\nvec4 worldPos=finalWorld*vec4(position,1.0);vPositionW=vec3(worldPos);\n#ifdef NORMAL\nmat3 normalWorld=mat3(finalWorld);\n#ifdef NONUNIFORMSCALING\nnormalWorld=transposeMat3(inverseMat3(normalWorld));\n#endif\nvNormalW=normalize(normalWorld*normal);\n#endif\n#if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED)\nvDirectionW=normalize(vec3(finalWorld*vec4(position,0.0)));\n#ifdef EQUIRECTANGULAR_RELFECTION_FOV\nmat3 screenToWorld=inverseMat3(mat3(finalWorld*viewProjection));vec3 segment=mix(vDirectionW,screenToWorld*vec3(0.0,0.0,1.0),abs(fFovMultiplier-1.0));if (fFovMultiplier<=1.0) {vDirectionW=normalize(segment);} else {vDirectionW=normalize(vDirectionW+(vDirectionW-segment));}\n#endif\n#endif\n#ifndef UV1\nvec2 uv=vec2(0.,0.);\n#endif\n#ifndef UV2\nvec2 uv2=vec2(0.,0.);\n#endif\n#ifdef MAINUV1\nvMainUV1=uv;\n#endif\n#ifdef MAINUV2\nvMainUV2=uv2;\n#endif\n#if defined(DIFFUSE) && DIFFUSEDIRECTUV==0\nif (vDiffuseInfos.x==0.)\n{vDiffuseUV=vec2(diffuseMatrix*vec4(uv,1.0,0.0));}\nelse\n{vDiffuseUV=vec2(diffuseMatrix*vec4(uv2,1.0,0.0));}\n#endif\n#include\n#include\n#include[0..maxSimultaneousLights]\n#ifdef VERTEXCOLOR\nvColor=color;\n#endif\n#if defined(POINTSIZE) && !defined(WEBGPU)\ngl_PointSize=pointSize;\n#endif\n#define CUSTOM_VERTEX_MAIN_END\n}\n`;\n // Sideeffect\n ShaderStore.ShadersStore[name$1S] = shader$1S;\n\n /**\n * Background material defines definition.\n * @internal Mainly internal Use\n */\n class BackgroundMaterialDefines extends MaterialDefines {\n /**\n * Constructor of the defines.\n */\n constructor() {\n super();\n /**\n * True if the diffuse texture is in use.\n */\n this.DIFFUSE = false;\n /**\n * The direct UV channel to use.\n */\n this.DIFFUSEDIRECTUV = 0;\n /**\n * True if the diffuse texture is in gamma space.\n */\n this.GAMMADIFFUSE = false;\n /**\n * True if the diffuse texture has opacity in the alpha channel.\n */\n this.DIFFUSEHASALPHA = false;\n /**\n * True if you want the material to fade to transparent at grazing angle.\n */\n this.OPACITYFRESNEL = false;\n /**\n * True if an extra blur needs to be added in the reflection.\n */\n this.REFLECTIONBLUR = false;\n /**\n * True if you want the material to fade to reflection at grazing angle.\n */\n this.REFLECTIONFRESNEL = false;\n /**\n * True if you want the material to falloff as far as you move away from the scene center.\n */\n this.REFLECTIONFALLOFF = false;\n /**\n * False if the current Webgl implementation does not support the texture lod extension.\n */\n this.TEXTURELODSUPPORT = false;\n /**\n * True to ensure the data are premultiplied.\n */\n this.PREMULTIPLYALPHA = false;\n /**\n * True if the texture contains cooked RGB values and not gray scaled multipliers.\n */\n this.USERGBCOLOR = false;\n /**\n * True if highlight and shadow levels have been specified. It can help ensuring the main perceived color\n * stays aligned with the desired configuration.\n */\n this.USEHIGHLIGHTANDSHADOWCOLORS = false;\n /**\n * True if only shadows must be rendered\n */\n this.BACKMAT_SHADOWONLY = false;\n /**\n * True to add noise in order to reduce the banding effect.\n */\n this.NOISE = false;\n /**\n * is the reflection texture in BGR color scheme?\n * Mainly used to solve a bug in ios10 video tag\n */\n this.REFLECTIONBGR = false;\n this.IMAGEPROCESSING = false;\n this.VIGNETTE = false;\n this.VIGNETTEBLENDMODEMULTIPLY = false;\n this.VIGNETTEBLENDMODEOPAQUE = false;\n this.TONEMAPPING = false;\n this.TONEMAPPING_ACES = false;\n this.CONTRAST = false;\n this.COLORCURVES = false;\n this.COLORGRADING = false;\n this.COLORGRADING3D = false;\n this.SAMPLER3DGREENDEPTH = false;\n this.SAMPLER3DBGRMAP = false;\n this.DITHER = false;\n this.IMAGEPROCESSINGPOSTPROCESS = false;\n this.SKIPFINALCOLORCLAMP = false;\n this.EXPOSURE = false;\n this.MULTIVIEW = false;\n // Reflection.\n this.REFLECTION = false;\n this.REFLECTIONMAP_3D = false;\n this.REFLECTIONMAP_SPHERICAL = false;\n this.REFLECTIONMAP_PLANAR = false;\n this.REFLECTIONMAP_CUBIC = false;\n this.REFLECTIONMAP_PROJECTION = false;\n this.REFLECTIONMAP_SKYBOX = false;\n this.REFLECTIONMAP_EXPLICIT = false;\n this.REFLECTIONMAP_EQUIRECTANGULAR = false;\n this.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = false;\n this.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = false;\n this.INVERTCUBICMAP = false;\n this.REFLECTIONMAP_OPPOSITEZ = false;\n this.LODINREFLECTIONALPHA = false;\n this.GAMMAREFLECTION = false;\n this.RGBDREFLECTION = false;\n this.EQUIRECTANGULAR_RELFECTION_FOV = false;\n // Default BJS.\n this.MAINUV1 = false;\n this.MAINUV2 = false;\n this.UV1 = false;\n this.UV2 = false;\n this.CLIPPLANE = false;\n this.CLIPPLANE2 = false;\n this.CLIPPLANE3 = false;\n this.CLIPPLANE4 = false;\n this.CLIPPLANE5 = false;\n this.CLIPPLANE6 = false;\n this.POINTSIZE = false;\n this.FOG = false;\n this.NORMAL = false;\n this.NUM_BONE_INFLUENCERS = 0;\n this.BonesPerMesh = 0;\n this.INSTANCES = false;\n this.SHADOWFLOAT = false;\n this.LOGARITHMICDEPTH = false;\n this.NONUNIFORMSCALING = false;\n this.ALPHATEST = false;\n this.rebuild();\n }\n }\n /**\n * Background material used to create an efficient environment around your scene.\n */\n class BackgroundMaterial extends PushMaterial {\n /**\n * Experimental Internal Use Only.\n *\n * Key light Color in \"perceptual value\" meaning the color you would like to see on screen.\n * This acts as a helper to set the primary color to a more \"human friendly\" value.\n * Conversion to linear space as well as exposure and tone mapping correction will be applied to keep the\n * output color as close as possible from the chosen value.\n * (This does not account for contrast color grading and color curves as they are considered post effect and not directly\n * part of lighting setup.)\n */\n get _perceptualColor() {\n return this.__perceptualColor;\n }\n set _perceptualColor(value) {\n this.__perceptualColor = value;\n this._computePrimaryColorFromPerceptualColor();\n this._markAllSubMeshesAsLightsDirty();\n }\n /**\n * Defines the level of the shadows (dark area of the reflection map) in order to help scaling the colors.\n * The color opposite to the primary color is used at the level chosen to define what the black area would look.\n */\n get primaryColorShadowLevel() {\n return this._primaryColorShadowLevel;\n }\n set primaryColorShadowLevel(value) {\n this._primaryColorShadowLevel = value;\n this._computePrimaryColors();\n this._markAllSubMeshesAsLightsDirty();\n }\n /**\n * Defines the level of the highlights (highlight area of the reflection map) in order to help scaling the colors.\n * The primary color is used at the level chosen to define what the white area would look.\n */\n get primaryColorHighlightLevel() {\n return this._primaryColorHighlightLevel;\n }\n set primaryColorHighlightLevel(value) {\n this._primaryColorHighlightLevel = value;\n this._computePrimaryColors();\n this._markAllSubMeshesAsLightsDirty();\n }\n /**\n * Sets the reflection reflectance fresnel values according to the default standard\n * empirically know to work well :-)\n */\n set reflectionStandardFresnelWeight(value) {\n let reflectionWeight = value;\n if (reflectionWeight < 0.5) {\n reflectionWeight = reflectionWeight * 2.0;\n this.reflectionReflectance0 = BackgroundMaterial.StandardReflectance0 * reflectionWeight;\n this.reflectionReflectance90 = BackgroundMaterial.StandardReflectance90 * reflectionWeight;\n }\n else {\n reflectionWeight = reflectionWeight * 2.0 - 1.0;\n this.reflectionReflectance0 = BackgroundMaterial.StandardReflectance0 + (1.0 - BackgroundMaterial.StandardReflectance0) * reflectionWeight;\n this.reflectionReflectance90 = BackgroundMaterial.StandardReflectance90 + (1.0 - BackgroundMaterial.StandardReflectance90) * reflectionWeight;\n }\n }\n /**\n * The current fov(field of view) multiplier, 0.0 - 2.0. Defaults to 1.0. Lower values \"zoom in\" and higher values \"zoom out\".\n * Best used when trying to implement visual zoom effects like fish-eye or binoculars while not adjusting camera fov.\n * Recommended to be keep at 1.0 except for special cases.\n */\n get fovMultiplier() {\n return this._fovMultiplier;\n }\n set fovMultiplier(value) {\n if (isNaN(value)) {\n value = 1.0;\n }\n this._fovMultiplier = Math.max(0.0, Math.min(2.0, value));\n }\n /**\n * Attaches a new image processing configuration to the PBR Material.\n * @param configuration (if null the scene configuration will be use)\n */\n _attachImageProcessingConfiguration(configuration) {\n if (configuration === this._imageProcessingConfiguration) {\n return;\n }\n // Detaches observer.\n if (this._imageProcessingConfiguration && this._imageProcessingObserver) {\n this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver);\n }\n // Pick the scene configuration if needed.\n if (!configuration) {\n this._imageProcessingConfiguration = this.getScene().imageProcessingConfiguration;\n }\n else {\n this._imageProcessingConfiguration = configuration;\n }\n // Attaches observer.\n if (this._imageProcessingConfiguration) {\n this._imageProcessingObserver = this._imageProcessingConfiguration.onUpdateParameters.add(() => {\n this._computePrimaryColorFromPerceptualColor();\n this._markAllSubMeshesAsImageProcessingDirty();\n });\n }\n }\n /**\n * Gets the image processing configuration used either in this material.\n */\n get imageProcessingConfiguration() {\n return this._imageProcessingConfiguration;\n }\n /**\n * Sets the Default image processing configuration used either in the this material.\n *\n * If sets to null, the scene one is in use.\n */\n set imageProcessingConfiguration(value) {\n this._attachImageProcessingConfiguration(value);\n // Ensure the effect will be rebuilt.\n this._markAllSubMeshesAsTexturesDirty();\n }\n /**\n * Gets whether the color curves effect is enabled.\n */\n get cameraColorCurvesEnabled() {\n return this.imageProcessingConfiguration.colorCurvesEnabled;\n }\n /**\n * Sets whether the color curves effect is enabled.\n */\n set cameraColorCurvesEnabled(value) {\n this.imageProcessingConfiguration.colorCurvesEnabled = value;\n }\n /**\n * Gets whether the color grading effect is enabled.\n */\n get cameraColorGradingEnabled() {\n return this.imageProcessingConfiguration.colorGradingEnabled;\n }\n /**\n * Gets whether the color grading effect is enabled.\n */\n set cameraColorGradingEnabled(value) {\n this.imageProcessingConfiguration.colorGradingEnabled = value;\n }\n /**\n * Gets whether tonemapping is enabled or not.\n */\n get cameraToneMappingEnabled() {\n return this._imageProcessingConfiguration.toneMappingEnabled;\n }\n /**\n * Sets whether tonemapping is enabled or not\n */\n set cameraToneMappingEnabled(value) {\n this._imageProcessingConfiguration.toneMappingEnabled = value;\n }\n /**\n * The camera exposure used on this material.\n * This property is here and not in the camera to allow controlling exposure without full screen post process.\n * This corresponds to a photographic exposure.\n */\n get cameraExposure() {\n return this._imageProcessingConfiguration.exposure;\n }\n /**\n * The camera exposure used on this material.\n * This property is here and not in the camera to allow controlling exposure without full screen post process.\n * This corresponds to a photographic exposure.\n */\n set cameraExposure(value) {\n this._imageProcessingConfiguration.exposure = value;\n }\n /**\n * Gets The camera contrast used on this material.\n */\n get cameraContrast() {\n return this._imageProcessingConfiguration.contrast;\n }\n /**\n * Sets The camera contrast used on this material.\n */\n set cameraContrast(value) {\n this._imageProcessingConfiguration.contrast = value;\n }\n /**\n * Gets the Color Grading 2D Lookup Texture.\n */\n get cameraColorGradingTexture() {\n return this._imageProcessingConfiguration.colorGradingTexture;\n }\n /**\n * Sets the Color Grading 2D Lookup Texture.\n */\n set cameraColorGradingTexture(value) {\n this.imageProcessingConfiguration.colorGradingTexture = value;\n }\n /**\n * The color grading curves provide additional color adjustment that is applied after any color grading transform (3D LUT).\n * They allow basic adjustment of saturation and small exposure adjustments, along with color filter tinting to provide white balance adjustment or more stylistic effects.\n * These are similar to controls found in many professional imaging or colorist software. The global controls are applied to the entire image. For advanced tuning, extra controls are provided to adjust the shadow, midtone and highlight areas of the image;\n * corresponding to low luminance, medium luminance, and high luminance areas respectively.\n */\n get cameraColorCurves() {\n return this.imageProcessingConfiguration.colorCurves;\n }\n /**\n * The color grading curves provide additional color adjustment that is applied after any color grading transform (3D LUT).\n * They allow basic adjustment of saturation and small exposure adjustments, along with color filter tinting to provide white balance adjustment or more stylistic effects.\n * These are similar to controls found in many professional imaging or colorist software. The global controls are applied to the entire image. For advanced tuning, extra controls are provided to adjust the shadow, midtone and highlight areas of the image;\n * corresponding to low luminance, medium luminance, and high luminance areas respectively.\n */\n set cameraColorCurves(value) {\n this.imageProcessingConfiguration.colorCurves = value;\n }\n /**\n * Instantiates a Background Material in the given scene\n * @param name The friendly name of the material\n * @param scene The scene to add the material to\n */\n constructor(name, scene) {\n super(name, scene);\n /**\n * Key light Color (multiply against the environment texture)\n */\n this.primaryColor = Color3.White();\n this._primaryColorShadowLevel = 0;\n this._primaryColorHighlightLevel = 0;\n /**\n * Reflection Texture used in the material.\n * Should be author in a specific way for the best result (refer to the documentation).\n */\n this.reflectionTexture = null;\n /**\n * Reflection Texture level of blur.\n *\n * Can be use to reuse an existing HDR Texture and target a specific LOD to prevent authoring the\n * texture twice.\n */\n this.reflectionBlur = 0;\n /**\n * Diffuse Texture used in the material.\n * Should be author in a specific way for the best result (refer to the documentation).\n */\n this.diffuseTexture = null;\n this._shadowLights = null;\n /**\n * Specify the list of lights casting shadow on the material.\n * All scene shadow lights will be included if null.\n */\n this.shadowLights = null;\n /**\n * Helps adjusting the shadow to a softer level if required.\n * 0 means black shadows and 1 means no shadows.\n */\n this.shadowLevel = 0;\n /**\n * In case of opacity Fresnel or reflection falloff, this is use as a scene center.\n * It is usually zero but might be interesting to modify according to your setup.\n */\n this.sceneCenter = Vector3.Zero();\n /**\n * This helps specifying that the material is falling off to the sky box at grazing angle.\n * This helps ensuring a nice transition when the camera goes under the ground.\n */\n this.opacityFresnel = true;\n /**\n * This helps specifying that the material is falling off from diffuse to the reflection texture at grazing angle.\n * This helps adding a mirror texture on the ground.\n */\n this.reflectionFresnel = false;\n /**\n * This helps specifying the falloff radius off the reflection texture from the sceneCenter.\n * This helps adding a nice falloff effect to the reflection if used as a mirror for instance.\n */\n this.reflectionFalloffDistance = 0.0;\n /**\n * This specifies the weight of the reflection against the background in case of reflection Fresnel.\n */\n this.reflectionAmount = 1.0;\n /**\n * This specifies the weight of the reflection at grazing angle.\n */\n this.reflectionReflectance0 = 0.05;\n /**\n * This specifies the weight of the reflection at a perpendicular point of view.\n */\n this.reflectionReflectance90 = 0.5;\n /**\n * Helps to directly use the maps channels instead of their level.\n */\n this.useRGBColor = true;\n /**\n * This helps reducing the banding effect that could occur on the background.\n */\n this.enableNoise = false;\n this._fovMultiplier = 1.0;\n /**\n * Enable the FOV adjustment feature controlled by fovMultiplier.\n */\n this.useEquirectangularFOV = false;\n this._maxSimultaneousLights = 4;\n /**\n * Number of Simultaneous lights allowed on the material.\n */\n this.maxSimultaneousLights = 4;\n this._shadowOnly = false;\n /**\n * Make the material only render shadows\n */\n this.shadowOnly = false;\n /**\n * Keep track of the image processing observer to allow dispose and replace.\n */\n this._imageProcessingObserver = null;\n /**\n * Due to a bug in iOS10, video tags (which are using the background material) are in BGR and not RGB.\n * Setting this flag to true (not done automatically!) will convert it back to RGB.\n */\n this.switchToBGR = false;\n // Temp values kept as cache in the material.\n this._renderTargets = new SmartArray(16);\n this._reflectionControls = Vector4.Zero();\n this._white = Color3.White();\n this._primaryShadowColor = Color3.Black();\n this._primaryHighlightColor = Color3.Black();\n // Setup the default processing configuration to the scene.\n this._attachImageProcessingConfiguration(null);\n this.getRenderTargetTextures = () => {\n this._renderTargets.reset();\n if (this._diffuseTexture && this._diffuseTexture.isRenderTarget) {\n this._renderTargets.push(this._diffuseTexture);\n }\n if (this._reflectionTexture && this._reflectionTexture.isRenderTarget) {\n this._renderTargets.push(this._reflectionTexture);\n }\n return this._renderTargets;\n };\n }\n /**\n * Gets a boolean indicating that current material needs to register RTT\n */\n get hasRenderTargetTextures() {\n if (this._diffuseTexture && this._diffuseTexture.isRenderTarget) {\n return true;\n }\n if (this._reflectionTexture && this._reflectionTexture.isRenderTarget) {\n return true;\n }\n return false;\n }\n /**\n * The entire material has been created in order to prevent overdraw.\n * @returns false\n */\n needAlphaTesting() {\n return true;\n }\n /**\n * The entire material has been created in order to prevent overdraw.\n * @returns true if blending is enable\n */\n needAlphaBlending() {\n return this.alpha < 1 || (this._diffuseTexture != null && this._diffuseTexture.hasAlpha) || this._shadowOnly;\n }\n /**\n * Checks whether the material is ready to be rendered for a given mesh.\n * @param mesh The mesh to render\n * @param subMesh The submesh to check against\n * @param useInstances Specify wether or not the material is used with instances\n * @returns true if all the dependencies are ready (Textures, Effects...)\n */\n isReadyForSubMesh(mesh, subMesh, useInstances = false) {\n if (subMesh.effect && this.isFrozen) {\n if (subMesh.effect._wasPreviouslyReady && subMesh.effect._wasPreviouslyUsingInstances === useInstances) {\n return true;\n }\n }\n if (!subMesh.materialDefines) {\n subMesh.materialDefines = new BackgroundMaterialDefines();\n }\n const scene = this.getScene();\n const defines = subMesh.materialDefines;\n if (this._isReadyForSubMesh(subMesh)) {\n return true;\n }\n const engine = scene.getEngine();\n // Lights\n MaterialHelper.PrepareDefinesForLights(scene, mesh, defines, false, this._maxSimultaneousLights);\n defines._needNormals = true;\n // Multiview\n MaterialHelper.PrepareDefinesForMultiview(scene, defines);\n // Textures\n if (defines._areTexturesDirty) {\n defines._needUVs = false;\n if (scene.texturesEnabled) {\n if (scene.getEngine().getCaps().textureLOD) {\n defines.TEXTURELODSUPPORT = true;\n }\n if (this._diffuseTexture && MaterialFlags.DiffuseTextureEnabled) {\n if (!this._diffuseTexture.isReadyOrNotBlocking()) {\n return false;\n }\n MaterialHelper.PrepareDefinesForMergedUV(this._diffuseTexture, defines, \"DIFFUSE\");\n defines.DIFFUSEHASALPHA = this._diffuseTexture.hasAlpha;\n defines.GAMMADIFFUSE = this._diffuseTexture.gammaSpace;\n defines.OPACITYFRESNEL = this._opacityFresnel;\n }\n else {\n defines.DIFFUSE = false;\n defines.DIFFUSEDIRECTUV = 0;\n defines.DIFFUSEHASALPHA = false;\n defines.GAMMADIFFUSE = false;\n defines.OPACITYFRESNEL = false;\n }\n const reflectionTexture = this._reflectionTexture;\n if (reflectionTexture && MaterialFlags.ReflectionTextureEnabled) {\n if (!reflectionTexture.isReadyOrNotBlocking()) {\n return false;\n }\n defines.REFLECTION = true;\n defines.GAMMAREFLECTION = reflectionTexture.gammaSpace;\n defines.RGBDREFLECTION = reflectionTexture.isRGBD;\n defines.REFLECTIONBLUR = this._reflectionBlur > 0;\n defines.LODINREFLECTIONALPHA = reflectionTexture.lodLevelInAlpha;\n defines.EQUIRECTANGULAR_RELFECTION_FOV = this.useEquirectangularFOV;\n defines.REFLECTIONBGR = this.switchToBGR;\n if (reflectionTexture.coordinatesMode === Texture.INVCUBIC_MODE) {\n defines.INVERTCUBICMAP = true;\n }\n defines.REFLECTIONMAP_3D = reflectionTexture.isCube;\n defines.REFLECTIONMAP_OPPOSITEZ = defines.REFLECTIONMAP_3D && this.getScene().useRightHandedSystem ? !reflectionTexture.invertZ : reflectionTexture.invertZ;\n switch (reflectionTexture.coordinatesMode) {\n case Texture.EXPLICIT_MODE:\n defines.REFLECTIONMAP_EXPLICIT = true;\n break;\n case Texture.PLANAR_MODE:\n defines.REFLECTIONMAP_PLANAR = true;\n break;\n case Texture.PROJECTION_MODE:\n defines.REFLECTIONMAP_PROJECTION = true;\n break;\n case Texture.SKYBOX_MODE:\n defines.REFLECTIONMAP_SKYBOX = true;\n break;\n case Texture.SPHERICAL_MODE:\n defines.REFLECTIONMAP_SPHERICAL = true;\n break;\n case Texture.EQUIRECTANGULAR_MODE:\n defines.REFLECTIONMAP_EQUIRECTANGULAR = true;\n break;\n case Texture.FIXED_EQUIRECTANGULAR_MODE:\n defines.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = true;\n break;\n case Texture.FIXED_EQUIRECTANGULAR_MIRRORED_MODE:\n defines.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = true;\n break;\n case Texture.CUBIC_MODE:\n case Texture.INVCUBIC_MODE:\n default:\n defines.REFLECTIONMAP_CUBIC = true;\n break;\n }\n if (this.reflectionFresnel) {\n defines.REFLECTIONFRESNEL = true;\n defines.REFLECTIONFALLOFF = this.reflectionFalloffDistance > 0;\n this._reflectionControls.x = this.reflectionAmount;\n this._reflectionControls.y = this.reflectionReflectance0;\n this._reflectionControls.z = this.reflectionReflectance90;\n this._reflectionControls.w = 1 / this.reflectionFalloffDistance;\n }\n else {\n defines.REFLECTIONFRESNEL = false;\n defines.REFLECTIONFALLOFF = false;\n }\n }\n else {\n defines.REFLECTION = false;\n defines.REFLECTIONFRESNEL = false;\n defines.REFLECTIONFALLOFF = false;\n defines.REFLECTIONBLUR = false;\n defines.REFLECTIONMAP_3D = false;\n defines.REFLECTIONMAP_SPHERICAL = false;\n defines.REFLECTIONMAP_PLANAR = false;\n defines.REFLECTIONMAP_CUBIC = false;\n defines.REFLECTIONMAP_PROJECTION = false;\n defines.REFLECTIONMAP_SKYBOX = false;\n defines.REFLECTIONMAP_EXPLICIT = false;\n defines.REFLECTIONMAP_EQUIRECTANGULAR = false;\n defines.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = false;\n defines.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = false;\n defines.INVERTCUBICMAP = false;\n defines.REFLECTIONMAP_OPPOSITEZ = false;\n defines.LODINREFLECTIONALPHA = false;\n defines.GAMMAREFLECTION = false;\n defines.RGBDREFLECTION = false;\n }\n }\n defines.PREMULTIPLYALPHA = this.alphaMode === 7 || this.alphaMode === 8;\n defines.USERGBCOLOR = this._useRGBColor;\n defines.NOISE = this._enableNoise;\n }\n if (defines._areLightsDirty) {\n defines.USEHIGHLIGHTANDSHADOWCOLORS = !this._useRGBColor && (this._primaryColorShadowLevel !== 0 || this._primaryColorHighlightLevel !== 0);\n defines.BACKMAT_SHADOWONLY = this._shadowOnly;\n }\n if (defines._areImageProcessingDirty && this._imageProcessingConfiguration) {\n if (!this._imageProcessingConfiguration.isReady()) {\n return false;\n }\n this._imageProcessingConfiguration.prepareDefines(defines);\n }\n // Misc.\n MaterialHelper.PrepareDefinesForMisc(mesh, scene, false, this.pointsCloud, this.fogEnabled, this._shouldTurnAlphaTestOn(mesh), defines);\n // Values that need to be evaluated on every frame\n MaterialHelper.PrepareDefinesForFrameBoundValues(scene, engine, this, defines, useInstances, null, subMesh.getRenderingMesh().hasThinInstances);\n // Attribs\n if (MaterialHelper.PrepareDefinesForAttributes(mesh, defines, false, true, false)) {\n if (mesh) {\n if (!scene.getEngine().getCaps().standardDerivatives && !mesh.isVerticesDataPresent(VertexBuffer.NormalKind)) {\n mesh.createNormals(true);\n Logger.Warn(\"BackgroundMaterial: Normals have been created for the mesh: \" + mesh.name);\n }\n }\n }\n // Get correct effect\n if (defines.isDirty) {\n defines.markAsProcessed();\n scene.resetCachedMaterial();\n // Fallbacks\n const fallbacks = new EffectFallbacks();\n if (defines.FOG) {\n fallbacks.addFallback(0, \"FOG\");\n }\n if (defines.POINTSIZE) {\n fallbacks.addFallback(1, \"POINTSIZE\");\n }\n if (defines.MULTIVIEW) {\n fallbacks.addFallback(0, \"MULTIVIEW\");\n }\n MaterialHelper.HandleFallbacksForShadows(defines, fallbacks, this._maxSimultaneousLights);\n //Attributes\n const attribs = [VertexBuffer.PositionKind];\n if (defines.NORMAL) {\n attribs.push(VertexBuffer.NormalKind);\n }\n if (defines.UV1) {\n attribs.push(VertexBuffer.UVKind);\n }\n if (defines.UV2) {\n attribs.push(VertexBuffer.UV2Kind);\n }\n MaterialHelper.PrepareAttributesForBones(attribs, mesh, defines, fallbacks);\n MaterialHelper.PrepareAttributesForInstances(attribs, defines);\n const uniforms = [\n \"world\",\n \"view\",\n \"viewProjection\",\n \"vEyePosition\",\n \"vLightsType\",\n \"vFogInfos\",\n \"vFogColor\",\n \"pointSize\",\n \"mBones\",\n \"vPrimaryColor\",\n \"vPrimaryColorShadow\",\n \"vReflectionInfos\",\n \"reflectionMatrix\",\n \"vReflectionMicrosurfaceInfos\",\n \"fFovMultiplier\",\n \"shadowLevel\",\n \"alpha\",\n \"vBackgroundCenter\",\n \"vReflectionControl\",\n \"vDiffuseInfos\",\n \"diffuseMatrix\",\n ];\n addClipPlaneUniforms(uniforms);\n const samplers = [\"diffuseSampler\", \"reflectionSampler\", \"reflectionSamplerLow\", \"reflectionSamplerHigh\"];\n const uniformBuffers = [\"Material\", \"Scene\"];\n if (ImageProcessingConfiguration) {\n ImageProcessingConfiguration.PrepareUniforms(uniforms, defines);\n ImageProcessingConfiguration.PrepareSamplers(samplers, defines);\n }\n MaterialHelper.PrepareUniformsAndSamplersList({\n uniformsNames: uniforms,\n uniformBuffersNames: uniformBuffers,\n samplers: samplers,\n defines: defines,\n maxSimultaneousLights: this._maxSimultaneousLights,\n });\n const join = defines.toString();\n const effect = scene.getEngine().createEffect(\"background\", {\n attributes: attribs,\n uniformsNames: uniforms,\n uniformBuffersNames: uniformBuffers,\n samplers: samplers,\n defines: join,\n fallbacks: fallbacks,\n onCompiled: this.onCompiled,\n onError: this.onError,\n indexParameters: { maxSimultaneousLights: this._maxSimultaneousLights },\n }, engine);\n subMesh.setEffect(effect, defines, this._materialContext);\n this.buildUniformLayout();\n }\n if (!subMesh.effect || !subMesh.effect.isReady()) {\n return false;\n }\n defines._renderId = scene.getRenderId();\n subMesh.effect._wasPreviouslyReady = true;\n subMesh.effect._wasPreviouslyUsingInstances = useInstances;\n this._checkScenePerformancePriority();\n return true;\n }\n /**\n * Compute the primary color according to the chosen perceptual color.\n */\n _computePrimaryColorFromPerceptualColor() {\n if (!this.__perceptualColor) {\n return;\n }\n this._primaryColor.copyFrom(this.__perceptualColor);\n // Revert gamma space.\n this._primaryColor.toLinearSpaceToRef(this._primaryColor, this.getScene().getEngine().useExactSrgbConversions);\n // Revert image processing configuration.\n if (this._imageProcessingConfiguration) {\n // Revert Exposure.\n this._primaryColor.scaleToRef(1 / this._imageProcessingConfiguration.exposure, this._primaryColor);\n }\n this._computePrimaryColors();\n }\n /**\n * Compute the highlights and shadow colors according to their chosen levels.\n */\n _computePrimaryColors() {\n if (this._primaryColorShadowLevel === 0 && this._primaryColorHighlightLevel === 0) {\n return;\n }\n // Find the highlight color based on the configuration.\n this._primaryColor.scaleToRef(this._primaryColorShadowLevel, this._primaryShadowColor);\n this._primaryColor.subtractToRef(this._primaryShadowColor, this._primaryShadowColor);\n // Find the shadow color based on the configuration.\n this._white.subtractToRef(this._primaryColor, this._primaryHighlightColor);\n this._primaryHighlightColor.scaleToRef(this._primaryColorHighlightLevel, this._primaryHighlightColor);\n this._primaryColor.addToRef(this._primaryHighlightColor, this._primaryHighlightColor);\n }\n /**\n * Build the uniform buffer used in the material.\n */\n buildUniformLayout() {\n // Order is important !\n this._uniformBuffer.addUniform(\"vPrimaryColor\", 4);\n this._uniformBuffer.addUniform(\"vPrimaryColorShadow\", 4);\n this._uniformBuffer.addUniform(\"vDiffuseInfos\", 2);\n this._uniformBuffer.addUniform(\"vReflectionInfos\", 2);\n this._uniformBuffer.addUniform(\"diffuseMatrix\", 16);\n this._uniformBuffer.addUniform(\"reflectionMatrix\", 16);\n this._uniformBuffer.addUniform(\"vReflectionMicrosurfaceInfos\", 3);\n this._uniformBuffer.addUniform(\"fFovMultiplier\", 1);\n this._uniformBuffer.addUniform(\"pointSize\", 1);\n this._uniformBuffer.addUniform(\"shadowLevel\", 1);\n this._uniformBuffer.addUniform(\"alpha\", 1);\n this._uniformBuffer.addUniform(\"vBackgroundCenter\", 3);\n this._uniformBuffer.addUniform(\"vReflectionControl\", 4);\n this._uniformBuffer.create();\n }\n /**\n * Unbind the material.\n */\n unbind() {\n if (this._diffuseTexture && this._diffuseTexture.isRenderTarget) {\n this._uniformBuffer.setTexture(\"diffuseSampler\", null);\n }\n if (this._reflectionTexture && this._reflectionTexture.isRenderTarget) {\n this._uniformBuffer.setTexture(\"reflectionSampler\", null);\n }\n super.unbind();\n }\n /**\n * Bind only the world matrix to the material.\n * @param world The world matrix to bind.\n */\n bindOnlyWorldMatrix(world) {\n this._activeEffect.setMatrix(\"world\", world);\n }\n /**\n * Bind the material for a dedicated submeh (every used meshes will be considered opaque).\n * @param world The world matrix to bind.\n * @param mesh\n * @param subMesh The submesh to bind for.\n */\n bindForSubMesh(world, mesh, subMesh) {\n const scene = this.getScene();\n const defines = subMesh.materialDefines;\n if (!defines) {\n return;\n }\n const effect = subMesh.effect;\n if (!effect) {\n return;\n }\n this._activeEffect = effect;\n // Matrices\n this.bindOnlyWorldMatrix(world);\n // Bones\n MaterialHelper.BindBonesParameters(mesh, this._activeEffect);\n const mustRebind = this._mustRebind(scene, effect, mesh.visibility);\n if (mustRebind) {\n this._uniformBuffer.bindToEffect(effect, \"Material\");\n this.bindViewProjection(effect);\n const reflectionTexture = this._reflectionTexture;\n if (!this._uniformBuffer.useUbo || !this.isFrozen || !this._uniformBuffer.isSync) {\n // Texture uniforms\n if (scene.texturesEnabled) {\n if (this._diffuseTexture && MaterialFlags.DiffuseTextureEnabled) {\n this._uniformBuffer.updateFloat2(\"vDiffuseInfos\", this._diffuseTexture.coordinatesIndex, this._diffuseTexture.level);\n MaterialHelper.BindTextureMatrix(this._diffuseTexture, this._uniformBuffer, \"diffuse\");\n }\n if (reflectionTexture && MaterialFlags.ReflectionTextureEnabled) {\n this._uniformBuffer.updateMatrix(\"reflectionMatrix\", reflectionTexture.getReflectionTextureMatrix());\n this._uniformBuffer.updateFloat2(\"vReflectionInfos\", reflectionTexture.level, this._reflectionBlur);\n this._uniformBuffer.updateFloat3(\"vReflectionMicrosurfaceInfos\", reflectionTexture.getSize().width, reflectionTexture.lodGenerationScale, reflectionTexture.lodGenerationOffset);\n }\n }\n if (this.shadowLevel > 0) {\n this._uniformBuffer.updateFloat(\"shadowLevel\", this.shadowLevel);\n }\n this._uniformBuffer.updateFloat(\"alpha\", this.alpha);\n // Point size\n if (this.pointsCloud) {\n this._uniformBuffer.updateFloat(\"pointSize\", this.pointSize);\n }\n if (defines.USEHIGHLIGHTANDSHADOWCOLORS) {\n this._uniformBuffer.updateColor4(\"vPrimaryColor\", this._primaryHighlightColor, 1.0);\n this._uniformBuffer.updateColor4(\"vPrimaryColorShadow\", this._primaryShadowColor, 1.0);\n }\n else {\n this._uniformBuffer.updateColor4(\"vPrimaryColor\", this._primaryColor, 1.0);\n }\n }\n this._uniformBuffer.updateFloat(\"fFovMultiplier\", this._fovMultiplier);\n // Textures\n if (scene.texturesEnabled) {\n if (this._diffuseTexture && MaterialFlags.DiffuseTextureEnabled) {\n this._uniformBuffer.setTexture(\"diffuseSampler\", this._diffuseTexture);\n }\n if (reflectionTexture && MaterialFlags.ReflectionTextureEnabled) {\n if (defines.REFLECTIONBLUR && defines.TEXTURELODSUPPORT) {\n this._uniformBuffer.setTexture(\"reflectionSampler\", reflectionTexture);\n }\n else if (!defines.REFLECTIONBLUR) {\n this._uniformBuffer.setTexture(\"reflectionSampler\", reflectionTexture);\n }\n else {\n this._uniformBuffer.setTexture(\"reflectionSampler\", reflectionTexture._lodTextureMid || reflectionTexture);\n this._uniformBuffer.setTexture(\"reflectionSamplerLow\", reflectionTexture._lodTextureLow || reflectionTexture);\n this._uniformBuffer.setTexture(\"reflectionSamplerHigh\", reflectionTexture._lodTextureHigh || reflectionTexture);\n }\n if (defines.REFLECTIONFRESNEL) {\n this._uniformBuffer.updateFloat3(\"vBackgroundCenter\", this.sceneCenter.x, this.sceneCenter.y, this.sceneCenter.z);\n this._uniformBuffer.updateFloat4(\"vReflectionControl\", this._reflectionControls.x, this._reflectionControls.y, this._reflectionControls.z, this._reflectionControls.w);\n }\n }\n }\n // Clip plane\n bindClipPlane(this._activeEffect, this, scene);\n scene.bindEyePosition(effect);\n }\n else if (scene.getEngine()._features.needToAlwaysBindUniformBuffers) {\n this._uniformBuffer.bindToEffect(effect, \"Material\");\n this._needToBindSceneUbo = true;\n }\n if (mustRebind || !this.isFrozen) {\n if (scene.lightsEnabled) {\n MaterialHelper.BindLights(scene, mesh, this._activeEffect, defines, this._maxSimultaneousLights);\n }\n // View\n this.bindView(effect);\n // Fog\n MaterialHelper.BindFogParameters(scene, mesh, this._activeEffect, true);\n // image processing\n if (this._imageProcessingConfiguration) {\n this._imageProcessingConfiguration.bind(this._activeEffect);\n }\n }\n this._afterBind(mesh, this._activeEffect);\n this._uniformBuffer.update();\n }\n /**\n * Checks to see if a texture is used in the material.\n * @param texture - Base texture to use.\n * @returns - Boolean specifying if a texture is used in the material.\n */\n hasTexture(texture) {\n if (super.hasTexture(texture)) {\n return true;\n }\n if (this._reflectionTexture === texture) {\n return true;\n }\n if (this._diffuseTexture === texture) {\n return true;\n }\n return false;\n }\n /**\n * Dispose the material.\n * @param forceDisposeEffect Force disposal of the associated effect.\n * @param forceDisposeTextures Force disposal of the associated textures.\n */\n dispose(forceDisposeEffect = false, forceDisposeTextures = false) {\n if (forceDisposeTextures) {\n if (this.diffuseTexture) {\n this.diffuseTexture.dispose();\n }\n if (this.reflectionTexture) {\n this.reflectionTexture.dispose();\n }\n }\n this._renderTargets.dispose();\n if (this._imageProcessingConfiguration && this._imageProcessingObserver) {\n this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver);\n }\n super.dispose(forceDisposeEffect);\n }\n /**\n * Clones the material.\n * @param name The cloned name.\n * @returns The cloned material.\n */\n clone(name) {\n return SerializationHelper.Clone(() => new BackgroundMaterial(name, this.getScene()), this);\n }\n /**\n * Serializes the current material to its JSON representation.\n * @returns The JSON representation.\n */\n serialize() {\n const serializationObject = super.serialize();\n serializationObject.customType = \"BABYLON.BackgroundMaterial\";\n return serializationObject;\n }\n /**\n * Gets the class name of the material\n * @returns \"BackgroundMaterial\"\n */\n getClassName() {\n return \"BackgroundMaterial\";\n }\n /**\n * Parse a JSON input to create back a background material.\n * @param source The JSON data to parse\n * @param scene The scene to create the parsed material in\n * @param rootUrl The root url of the assets the material depends upon\n * @returns the instantiated BackgroundMaterial.\n */\n static Parse(source, scene, rootUrl) {\n return SerializationHelper.Parse(() => new BackgroundMaterial(source.name, scene), source, scene, rootUrl);\n }\n }\n /**\n * Standard reflectance value at parallel view angle.\n */\n BackgroundMaterial.StandardReflectance0 = 0.05;\n /**\n * Standard reflectance value at grazing angle.\n */\n BackgroundMaterial.StandardReflectance90 = 0.5;\n __decorate$1([\n serializeAsColor3()\n ], BackgroundMaterial.prototype, \"_primaryColor\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsLightsDirty\")\n ], BackgroundMaterial.prototype, \"primaryColor\", void 0);\n __decorate$1([\n serializeAsColor3()\n ], BackgroundMaterial.prototype, \"__perceptualColor\", void 0);\n __decorate$1([\n serialize()\n ], BackgroundMaterial.prototype, \"_primaryColorShadowLevel\", void 0);\n __decorate$1([\n serialize()\n ], BackgroundMaterial.prototype, \"_primaryColorHighlightLevel\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsLightsDirty\")\n ], BackgroundMaterial.prototype, \"primaryColorHighlightLevel\", null);\n __decorate$1([\n serializeAsTexture()\n ], BackgroundMaterial.prototype, \"_reflectionTexture\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], BackgroundMaterial.prototype, \"reflectionTexture\", void 0);\n __decorate$1([\n serialize()\n ], BackgroundMaterial.prototype, \"_reflectionBlur\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], BackgroundMaterial.prototype, \"reflectionBlur\", void 0);\n __decorate$1([\n serializeAsTexture()\n ], BackgroundMaterial.prototype, \"_diffuseTexture\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], BackgroundMaterial.prototype, \"diffuseTexture\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], BackgroundMaterial.prototype, \"shadowLights\", void 0);\n __decorate$1([\n serialize()\n ], BackgroundMaterial.prototype, \"_shadowLevel\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], BackgroundMaterial.prototype, \"shadowLevel\", void 0);\n __decorate$1([\n serializeAsVector3()\n ], BackgroundMaterial.prototype, \"_sceneCenter\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], BackgroundMaterial.prototype, \"sceneCenter\", void 0);\n __decorate$1([\n serialize()\n ], BackgroundMaterial.prototype, \"_opacityFresnel\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], BackgroundMaterial.prototype, \"opacityFresnel\", void 0);\n __decorate$1([\n serialize()\n ], BackgroundMaterial.prototype, \"_reflectionFresnel\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], BackgroundMaterial.prototype, \"reflectionFresnel\", void 0);\n __decorate$1([\n serialize()\n ], BackgroundMaterial.prototype, \"_reflectionFalloffDistance\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], BackgroundMaterial.prototype, \"reflectionFalloffDistance\", void 0);\n __decorate$1([\n serialize()\n ], BackgroundMaterial.prototype, \"_reflectionAmount\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], BackgroundMaterial.prototype, \"reflectionAmount\", void 0);\n __decorate$1([\n serialize()\n ], BackgroundMaterial.prototype, \"_reflectionReflectance0\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], BackgroundMaterial.prototype, \"reflectionReflectance0\", void 0);\n __decorate$1([\n serialize()\n ], BackgroundMaterial.prototype, \"_reflectionReflectance90\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], BackgroundMaterial.prototype, \"reflectionReflectance90\", void 0);\n __decorate$1([\n serialize()\n ], BackgroundMaterial.prototype, \"_useRGBColor\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], BackgroundMaterial.prototype, \"useRGBColor\", void 0);\n __decorate$1([\n serialize()\n ], BackgroundMaterial.prototype, \"_enableNoise\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], BackgroundMaterial.prototype, \"enableNoise\", void 0);\n __decorate$1([\n serialize()\n ], BackgroundMaterial.prototype, \"_maxSimultaneousLights\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\n ], BackgroundMaterial.prototype, \"maxSimultaneousLights\", void 0);\n __decorate$1([\n serialize()\n ], BackgroundMaterial.prototype, \"_shadowOnly\", void 0);\n __decorate$1([\n expandToProperty(\"_markAllSubMeshesAsLightsDirty\")\n ], BackgroundMaterial.prototype, \"shadowOnly\", void 0);\n __decorate$1([\n serializeAsImageProcessingConfiguration()\n ], BackgroundMaterial.prototype, \"_imageProcessingConfiguration\", void 0);\n RegisterClass(\"BABYLON.BackgroundMaterial\", BackgroundMaterial);\n\n /**\n * Creates the VertexData for a Plane\n * @param options an object used to set the following optional parameters for the plane, required but can be empty\n * * size sets the width and height of the plane to the value of size, optional default 1\n * * width sets the width (x direction) of the plane, overwrites the width set by size, optional, default size\n * * height sets the height (y direction) of the plane, overwrites the height set by size, optional, default size\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.size\n * @param options.width\n * @param options.height\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @returns the VertexData of the box\n */\n function CreatePlaneVertexData(options) {\n const indices = [];\n const positions = [];\n const normals = [];\n const uvs = [];\n const width = options.width || options.size || 1;\n const height = options.height || options.size || 1;\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\n // Vertices\n const halfWidth = width / 2.0;\n const halfHeight = height / 2.0;\n positions.push(-halfWidth, -halfHeight, 0);\n normals.push(0, 0, -1.0);\n uvs.push(0.0, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 : 0.0);\n positions.push(halfWidth, -halfHeight, 0);\n normals.push(0, 0, -1.0);\n uvs.push(1.0, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 : 0.0);\n positions.push(halfWidth, halfHeight, 0);\n normals.push(0, 0, -1.0);\n uvs.push(1.0, CompatibilityOptions.UseOpenGLOrientationForUV ? 0.0 : 1.0);\n positions.push(-halfWidth, halfHeight, 0);\n normals.push(0, 0, -1.0);\n uvs.push(0.0, CompatibilityOptions.UseOpenGLOrientationForUV ? 0.0 : 1.0);\n // Indices\n indices.push(0);\n indices.push(1);\n indices.push(2);\n indices.push(0);\n indices.push(2);\n indices.push(3);\n // Sides\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\n // Result\n const vertexData = new VertexData();\n vertexData.indices = indices;\n vertexData.positions = positions;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n return vertexData;\n }\n /**\n * Creates a plane mesh\n * * The parameter `size` sets the size (float) of both sides of the plane at once (default 1)\n * * You can set some different plane dimensions by using the parameters `width` and `height` (both by default have the same value of `size`)\n * * The parameter `sourcePlane` is a Plane instance. It builds a mesh plane from a Math plane\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param options.size\n * @param options.width\n * @param options.height\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param options.updatable\n * @param options.sourcePlane\n * @param scene defines the hosting scene\n * @returns the plane mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#plane\n */\n function CreatePlane(name, options = {}, scene = null) {\n const plane = new Mesh(name, scene);\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n plane._originalBuilderSideOrientation = options.sideOrientation;\n const vertexData = CreatePlaneVertexData(options);\n vertexData.applyToMesh(plane, options.updatable);\n if (options.sourcePlane) {\n plane.translate(options.sourcePlane.normal, -options.sourcePlane.d);\n plane.setDirection(options.sourcePlane.normal.scale(-1));\n }\n return plane;\n }\n /**\n * Class containing static functions to help procedurally build meshes\n * @deprecated use the function directly from the module\n */\n const PlaneBuilder = {\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CreatePlane,\n };\n VertexData.CreatePlane = CreatePlaneVertexData;\n Mesh.CreatePlane = (name, size, scene, updatable, sideOrientation) => {\n const options = {\n size,\n width: size,\n height: size,\n sideOrientation,\n updatable,\n };\n return CreatePlane(name, options, scene);\n };\n\n Mesh._GroundMeshParser = (parsedMesh, scene) => {\n return GroundMesh.Parse(parsedMesh, scene);\n };\n /**\n * Mesh representing the ground\n */\n class GroundMesh extends Mesh {\n constructor(name, scene) {\n super(name, scene);\n /** If octree should be generated */\n this.generateOctree = false;\n }\n /**\n * \"GroundMesh\"\n * @returns \"GroundMesh\"\n */\n getClassName() {\n return \"GroundMesh\";\n }\n /**\n * The minimum of x and y subdivisions\n */\n get subdivisions() {\n return Math.min(this._subdivisionsX, this._subdivisionsY);\n }\n /**\n * X subdivisions\n */\n get subdivisionsX() {\n return this._subdivisionsX;\n }\n /**\n * Y subdivisions\n */\n get subdivisionsY() {\n return this._subdivisionsY;\n }\n /**\n * This function will divide the mesh into submeshes and update an octree to help to select the right submeshes\n * for rendering, picking and collision computations. Please note that you must have a decent number of submeshes\n * to get performance improvements when using an octree.\n * @param chunksCount the number of submeshes the mesh will be divided into\n * @param octreeBlocksSize the maximum size of the octree blocks (Default: 32)\n */\n optimize(chunksCount, octreeBlocksSize = 32) {\n this._subdivisionsX = chunksCount;\n this._subdivisionsY = chunksCount;\n this.subdivide(chunksCount);\n // Call the octree system optimization if it is defined.\n const thisAsAny = this;\n if (thisAsAny.createOrUpdateSubmeshesOctree) {\n thisAsAny.createOrUpdateSubmeshesOctree(octreeBlocksSize);\n }\n }\n /**\n * Returns a height (y) value in the World system :\n * the ground altitude at the coordinates (x, z) expressed in the World system.\n * @param x x coordinate\n * @param z z coordinate\n * @returns the ground y position if (x, z) are outside the ground surface.\n */\n getHeightAtCoordinates(x, z) {\n const world = this.getWorldMatrix();\n const invMat = TmpVectors.Matrix[5];\n world.invertToRef(invMat);\n const tmpVect = TmpVectors.Vector3[8];\n Vector3.TransformCoordinatesFromFloatsToRef(x, 0.0, z, invMat, tmpVect); // transform x,z in the mesh local space\n x = tmpVect.x;\n z = tmpVect.z;\n if (x < this._minX || x >= this._maxX || z <= this._minZ || z > this._maxZ) {\n return this.position.y;\n }\n if (!this._heightQuads || this._heightQuads.length == 0) {\n this._initHeightQuads();\n this._computeHeightQuads();\n }\n const facet = this._getFacetAt(x, z);\n const y = -(facet.x * x + facet.z * z + facet.w) / facet.y;\n // return y in the World system\n Vector3.TransformCoordinatesFromFloatsToRef(0.0, y, 0.0, world, tmpVect);\n return tmpVect.y;\n }\n /**\n * Returns a normalized vector (Vector3) orthogonal to the ground\n * at the ground coordinates (x, z) expressed in the World system.\n * @param x x coordinate\n * @param z z coordinate\n * @returns Vector3(0.0, 1.0, 0.0) if (x, z) are outside the ground surface.\n */\n getNormalAtCoordinates(x, z) {\n const normal = new Vector3(0.0, 1.0, 0.0);\n this.getNormalAtCoordinatesToRef(x, z, normal);\n return normal;\n }\n /**\n * Updates the Vector3 passed a reference with a normalized vector orthogonal to the ground\n * at the ground coordinates (x, z) expressed in the World system.\n * Doesn't update the reference Vector3 if (x, z) are outside the ground surface.\n * @param x x coordinate\n * @param z z coordinate\n * @param ref vector to store the result\n * @returns the GroundMesh.\n */\n getNormalAtCoordinatesToRef(x, z, ref) {\n const world = this.getWorldMatrix();\n const tmpMat = TmpVectors.Matrix[5];\n world.invertToRef(tmpMat);\n const tmpVect = TmpVectors.Vector3[8];\n Vector3.TransformCoordinatesFromFloatsToRef(x, 0.0, z, tmpMat, tmpVect); // transform x,z in the mesh local space\n x = tmpVect.x;\n z = tmpVect.z;\n if (x < this._minX || x > this._maxX || z < this._minZ || z > this._maxZ) {\n return this;\n }\n if (!this._heightQuads || this._heightQuads.length == 0) {\n this._initHeightQuads();\n this._computeHeightQuads();\n }\n const facet = this._getFacetAt(x, z);\n Vector3.TransformNormalFromFloatsToRef(facet.x, facet.y, facet.z, world, ref);\n return this;\n }\n /**\n * Force the heights to be recomputed for getHeightAtCoordinates() or getNormalAtCoordinates()\n * if the ground has been updated.\n * This can be used in the render loop.\n * @returns the GroundMesh.\n */\n updateCoordinateHeights() {\n if (!this._heightQuads || this._heightQuads.length == 0) {\n this._initHeightQuads();\n }\n this._computeHeightQuads();\n return this;\n }\n // Returns the element \"facet\" from the heightQuads array relative to (x, z) local coordinates\n _getFacetAt(x, z) {\n // retrieve col and row from x, z coordinates in the ground local system\n const col = Math.floor(((x + this._maxX) * this._subdivisionsX) / this._width);\n const row = Math.floor((-(z + this._maxZ) * this._subdivisionsY) / this._height + this._subdivisionsY);\n const quad = this._heightQuads[row * this._subdivisionsX + col];\n let facet;\n if (z < quad.slope.x * x + quad.slope.y) {\n facet = quad.facet1;\n }\n else {\n facet = quad.facet2;\n }\n return facet;\n }\n // Creates and populates the heightMap array with \"facet\" elements :\n // a quad is two triangular facets separated by a slope, so a \"facet\" element is 1 slope + 2 facets\n // slope : Vector2(c, h) = 2D diagonal line equation setting apart two triangular facets in a quad : z = cx + h\n // facet1 : Vector4(a, b, c, d) = first facet 3D plane equation : ax + by + cz + d = 0\n // facet2 : Vector4(a, b, c, d) = second facet 3D plane equation : ax + by + cz + d = 0\n // Returns the GroundMesh.\n _initHeightQuads() {\n const subdivisionsX = this._subdivisionsX;\n const subdivisionsY = this._subdivisionsY;\n this._heightQuads = new Array();\n for (let row = 0; row < subdivisionsY; row++) {\n for (let col = 0; col < subdivisionsX; col++) {\n const quad = { slope: Vector2.Zero(), facet1: new Vector4(0.0, 0.0, 0.0, 0.0), facet2: new Vector4(0.0, 0.0, 0.0, 0.0) };\n this._heightQuads[row * subdivisionsX + col] = quad;\n }\n }\n return this;\n }\n // Compute each quad element values and update the the heightMap array :\n // slope : Vector2(c, h) = 2D diagonal line equation setting apart two triangular facets in a quad : z = cx + h\n // facet1 : Vector4(a, b, c, d) = first facet 3D plane equation : ax + by + cz + d = 0\n // facet2 : Vector4(a, b, c, d) = second facet 3D plane equation : ax + by + cz + d = 0\n // Returns the GroundMesh.\n _computeHeightQuads() {\n const positions = this.getVerticesData(VertexBuffer.PositionKind);\n if (!positions) {\n return this;\n }\n const v1 = TmpVectors.Vector3[3];\n const v2 = TmpVectors.Vector3[2];\n const v3 = TmpVectors.Vector3[1];\n const v4 = TmpVectors.Vector3[0];\n const v1v2 = TmpVectors.Vector3[4];\n const v1v3 = TmpVectors.Vector3[5];\n const v1v4 = TmpVectors.Vector3[6];\n const norm1 = TmpVectors.Vector3[7];\n const norm2 = TmpVectors.Vector3[8];\n let i = 0;\n let j = 0;\n let k = 0;\n let cd = 0; // 2D slope coefficient : z = cd * x + h\n let h = 0;\n let d1 = 0; // facet plane equation : ax + by + cz + d = 0\n let d2 = 0;\n const subdivisionsX = this._subdivisionsX;\n const subdivisionsY = this._subdivisionsY;\n for (let row = 0; row < subdivisionsY; row++) {\n for (let col = 0; col < subdivisionsX; col++) {\n i = col * 3;\n j = row * (subdivisionsX + 1) * 3;\n k = (row + 1) * (subdivisionsX + 1) * 3;\n v1.x = positions[j + i];\n v1.y = positions[j + i + 1];\n v1.z = positions[j + i + 2];\n v2.x = positions[j + i + 3];\n v2.y = positions[j + i + 4];\n v2.z = positions[j + i + 5];\n v3.x = positions[k + i];\n v3.y = positions[k + i + 1];\n v3.z = positions[k + i + 2];\n v4.x = positions[k + i + 3];\n v4.y = positions[k + i + 4];\n v4.z = positions[k + i + 5];\n // 2D slope V1V4\n cd = (v4.z - v1.z) / (v4.x - v1.x);\n h = v1.z - cd * v1.x; // v1 belongs to the slope\n // facet equations :\n // we compute each facet normal vector\n // the equation of the facet plane is : norm.x * x + norm.y * y + norm.z * z + d = 0\n // we compute the value d by applying the equation to v1 which belongs to the plane\n // then we store the facet equation in a Vector4\n v2.subtractToRef(v1, v1v2);\n v3.subtractToRef(v1, v1v3);\n v4.subtractToRef(v1, v1v4);\n Vector3.CrossToRef(v1v4, v1v3, norm1); // caution : CrossToRef uses the Tmp class\n Vector3.CrossToRef(v1v2, v1v4, norm2);\n norm1.normalize();\n norm2.normalize();\n d1 = -(norm1.x * v1.x + norm1.y * v1.y + norm1.z * v1.z);\n d2 = -(norm2.x * v2.x + norm2.y * v2.y + norm2.z * v2.z);\n const quad = this._heightQuads[row * subdivisionsX + col];\n quad.slope.copyFromFloats(cd, h);\n quad.facet1.copyFromFloats(norm1.x, norm1.y, norm1.z, d1);\n quad.facet2.copyFromFloats(norm2.x, norm2.y, norm2.z, d2);\n }\n }\n return this;\n }\n /**\n * Serializes this ground mesh\n * @param serializationObject object to write serialization to\n */\n serialize(serializationObject) {\n super.serialize(serializationObject);\n serializationObject.subdivisionsX = this._subdivisionsX;\n serializationObject.subdivisionsY = this._subdivisionsY;\n serializationObject.minX = this._minX;\n serializationObject.maxX = this._maxX;\n serializationObject.minZ = this._minZ;\n serializationObject.maxZ = this._maxZ;\n serializationObject.width = this._width;\n serializationObject.height = this._height;\n }\n /**\n * Parses a serialized ground mesh\n * @param parsedMesh the serialized mesh\n * @param scene the scene to create the ground mesh in\n * @returns the created ground mesh\n */\n static Parse(parsedMesh, scene) {\n const result = new GroundMesh(parsedMesh.name, scene);\n result._subdivisionsX = parsedMesh.subdivisionsX || 1;\n result._subdivisionsY = parsedMesh.subdivisionsY || 1;\n result._minX = parsedMesh.minX;\n result._maxX = parsedMesh.maxX;\n result._minZ = parsedMesh.minZ;\n result._maxZ = parsedMesh.maxZ;\n result._width = parsedMesh.width;\n result._height = parsedMesh.height;\n return result;\n }\n }\n\n /**\n * Creates the VertexData for a Ground\n * @param options an object used to set the following optional parameters for the Ground, required but can be empty\n * - width the width (x direction) of the ground, optional, default 1\n * - height the height (z direction) of the ground, optional, default 1\n * - subdivisions the number of subdivisions per side, optional, default 1\n * @param options.width\n * @param options.height\n * @param options.subdivisions\n * @param options.subdivisionsX\n * @param options.subdivisionsY\n * @returns the VertexData of the Ground\n */\n function CreateGroundVertexData(options) {\n const indices = [];\n const positions = [];\n const normals = [];\n const uvs = [];\n let row, col;\n const width = options.width || 1;\n const height = options.height || 1;\n const subdivisionsX = (options.subdivisionsX || options.subdivisions || 1) | 0;\n const subdivisionsY = (options.subdivisionsY || options.subdivisions || 1) | 0;\n for (row = 0; row <= subdivisionsY; row++) {\n for (col = 0; col <= subdivisionsX; col++) {\n const position = new Vector3((col * width) / subdivisionsX - width / 2.0, 0, ((subdivisionsY - row) * height) / subdivisionsY - height / 2.0);\n const normal = new Vector3(0, 1.0, 0);\n positions.push(position.x, position.y, position.z);\n normals.push(normal.x, normal.y, normal.z);\n uvs.push(col / subdivisionsX, CompatibilityOptions.UseOpenGLOrientationForUV ? row / subdivisionsY : 1.0 - row / subdivisionsY);\n }\n }\n for (row = 0; row < subdivisionsY; row++) {\n for (col = 0; col < subdivisionsX; col++) {\n indices.push(col + 1 + (row + 1) * (subdivisionsX + 1));\n indices.push(col + 1 + row * (subdivisionsX + 1));\n indices.push(col + row * (subdivisionsX + 1));\n indices.push(col + (row + 1) * (subdivisionsX + 1));\n indices.push(col + 1 + (row + 1) * (subdivisionsX + 1));\n indices.push(col + row * (subdivisionsX + 1));\n }\n }\n // Result\n const vertexData = new VertexData();\n vertexData.indices = indices;\n vertexData.positions = positions;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n return vertexData;\n }\n /**\n * Creates the VertexData for a TiledGround by subdividing the ground into tiles\n * @param options an object used to set the following optional parameters for the Ground, required but can be empty\n * * xmin the ground minimum X coordinate, optional, default -1\n * * zmin the ground minimum Z coordinate, optional, default -1\n * * xmax the ground maximum X coordinate, optional, default 1\n * * zmax the ground maximum Z coordinate, optional, default 1\n * * subdivisions a javascript object {w: positive integer, h: positive integer}, `w` and `h` are the numbers of subdivisions on the ground width and height creating 'tiles', default {w: 6, h: 6}\n * * precision a javascript object {w: positive integer, h: positive integer}, `w` and `h` are the numbers of subdivisions on the tile width and height, default {w: 2, h: 2}\n * @param options.xmin\n * @param options.zmin\n * @param options.xmax\n * @param options.zmax\n * @param options.subdivisions\n * @param options.subdivisions.w\n * @param options.subdivisions.h\n * @param options.precision\n * @param options.precision.w\n * @param options.precision.h\n * @returns the VertexData of the TiledGround\n */\n function CreateTiledGroundVertexData(options) {\n const xmin = options.xmin !== undefined && options.xmin !== null ? options.xmin : -1.0;\n const zmin = options.zmin !== undefined && options.zmin !== null ? options.zmin : -1.0;\n const xmax = options.xmax !== undefined && options.xmax !== null ? options.xmax : 1.0;\n const zmax = options.zmax !== undefined && options.zmax !== null ? options.zmax : 1.0;\n const subdivisions = options.subdivisions || { w: 1, h: 1 };\n const precision = options.precision || { w: 1, h: 1 };\n const indices = new Array();\n const positions = new Array();\n const normals = new Array();\n const uvs = new Array();\n let row, col, tileRow, tileCol;\n subdivisions.h = subdivisions.h < 1 ? 1 : subdivisions.h;\n subdivisions.w = subdivisions.w < 1 ? 1 : subdivisions.w;\n precision.w = precision.w < 1 ? 1 : precision.w;\n precision.h = precision.h < 1 ? 1 : precision.h;\n const tileSize = {\n w: (xmax - xmin) / subdivisions.w,\n h: (zmax - zmin) / subdivisions.h,\n };\n function applyTile(xTileMin, zTileMin, xTileMax, zTileMax) {\n // Indices\n const base = positions.length / 3;\n const rowLength = precision.w + 1;\n for (row = 0; row < precision.h; row++) {\n for (col = 0; col < precision.w; col++) {\n const square = [base + col + row * rowLength, base + (col + 1) + row * rowLength, base + (col + 1) + (row + 1) * rowLength, base + col + (row + 1) * rowLength];\n indices.push(square[1]);\n indices.push(square[2]);\n indices.push(square[3]);\n indices.push(square[0]);\n indices.push(square[1]);\n indices.push(square[3]);\n }\n }\n // Position, normals and uvs\n const position = Vector3.Zero();\n const normal = new Vector3(0, 1.0, 0);\n for (row = 0; row <= precision.h; row++) {\n position.z = (row * (zTileMax - zTileMin)) / precision.h + zTileMin;\n for (col = 0; col <= precision.w; col++) {\n position.x = (col * (xTileMax - xTileMin)) / precision.w + xTileMin;\n position.y = 0;\n positions.push(position.x, position.y, position.z);\n normals.push(normal.x, normal.y, normal.z);\n uvs.push(col / precision.w, row / precision.h);\n }\n }\n }\n for (tileRow = 0; tileRow < subdivisions.h; tileRow++) {\n for (tileCol = 0; tileCol < subdivisions.w; tileCol++) {\n applyTile(xmin + tileCol * tileSize.w, zmin + tileRow * tileSize.h, xmin + (tileCol + 1) * tileSize.w, zmin + (tileRow + 1) * tileSize.h);\n }\n }\n // Result\n const vertexData = new VertexData();\n vertexData.indices = indices;\n vertexData.positions = positions;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n return vertexData;\n }\n /**\n * Creates the VertexData of the Ground designed from a heightmap\n * @param options an object used to set the following parameters for the Ground, required and provided by CreateGroundFromHeightMap\n * * width the width (x direction) of the ground\n * * height the height (z direction) of the ground\n * * subdivisions the number of subdivisions per side\n * * minHeight the minimum altitude on the ground, optional, default 0\n * * maxHeight the maximum altitude on the ground, optional default 1\n * * colorFilter the filter to apply to the image pixel colors to compute the height, optional Color3, default (0.3, 0.59, 0.11)\n * * buffer the array holding the image color data\n * * bufferWidth the width of image\n * * bufferHeight the height of image\n * * alphaFilter Remove any data where the alpha channel is below this value, defaults 0 (all data visible)\n * @param options.width\n * @param options.height\n * @param options.subdivisions\n * @param options.minHeight\n * @param options.maxHeight\n * @param options.colorFilter\n * @param options.buffer\n * @param options.bufferWidth\n * @param options.bufferHeight\n * @param options.alphaFilter\n * @returns the VertexData of the Ground designed from a heightmap\n */\n function CreateGroundFromHeightMapVertexData(options) {\n const indices = [];\n const positions = [];\n const normals = [];\n const uvs = [];\n let row, col;\n const filter = options.colorFilter || new Color3(0.3, 0.59, 0.11);\n const alphaFilter = options.alphaFilter || 0.0;\n let invert = false;\n if (options.minHeight > options.maxHeight) {\n invert = true;\n const temp = options.maxHeight;\n options.maxHeight = options.minHeight;\n options.minHeight = temp;\n }\n // Vertices\n for (row = 0; row <= options.subdivisions; row++) {\n for (col = 0; col <= options.subdivisions; col++) {\n const position = new Vector3((col * options.width) / options.subdivisions - options.width / 2.0, 0, ((options.subdivisions - row) * options.height) / options.subdivisions - options.height / 2.0);\n // Compute height\n const heightMapX = (((position.x + options.width / 2) / options.width) * (options.bufferWidth - 1)) | 0;\n const heightMapY = ((1.0 - (position.z + options.height / 2) / options.height) * (options.bufferHeight - 1)) | 0;\n const pos = (heightMapX + heightMapY * options.bufferWidth) * 4;\n let r = options.buffer[pos] / 255.0;\n let g = options.buffer[pos + 1] / 255.0;\n let b = options.buffer[pos + 2] / 255.0;\n const a = options.buffer[pos + 3] / 255.0;\n if (invert) {\n r = 1.0 - r;\n g = 1.0 - g;\n b = 1.0 - b;\n }\n const gradient = r * filter.r + g * filter.g + b * filter.b;\n // If our alpha channel is not within our filter then we will assign a 'special' height\n // Then when building the indices, we will ignore any vertex that is using the special height\n if (a >= alphaFilter) {\n position.y = options.minHeight + (options.maxHeight - options.minHeight) * gradient;\n }\n else {\n position.y = options.minHeight - Epsilon; // We can't have a height below minHeight, normally.\n }\n // Add vertex\n positions.push(position.x, position.y, position.z);\n normals.push(0, 0, 0);\n uvs.push(col / options.subdivisions, 1.0 - row / options.subdivisions);\n }\n }\n // Indices\n for (row = 0; row < options.subdivisions; row++) {\n for (col = 0; col < options.subdivisions; col++) {\n // Calculate Indices\n const idx1 = col + 1 + (row + 1) * (options.subdivisions + 1);\n const idx2 = col + 1 + row * (options.subdivisions + 1);\n const idx3 = col + row * (options.subdivisions + 1);\n const idx4 = col + (row + 1) * (options.subdivisions + 1);\n // Check that all indices are visible (based on our special height)\n // Only display the vertex if all Indices are visible\n // Positions are stored x,y,z for each vertex, hence the * 3 and + 1 for height\n const isVisibleIdx1 = positions[idx1 * 3 + 1] >= options.minHeight;\n const isVisibleIdx2 = positions[idx2 * 3 + 1] >= options.minHeight;\n const isVisibleIdx3 = positions[idx3 * 3 + 1] >= options.minHeight;\n if (isVisibleIdx1 && isVisibleIdx2 && isVisibleIdx3) {\n indices.push(idx1);\n indices.push(idx2);\n indices.push(idx3);\n }\n const isVisibleIdx4 = positions[idx4 * 3 + 1] >= options.minHeight;\n if (isVisibleIdx4 && isVisibleIdx1 && isVisibleIdx3) {\n indices.push(idx4);\n indices.push(idx1);\n indices.push(idx3);\n }\n }\n }\n // Normals\n VertexData.ComputeNormals(positions, indices, normals);\n // Result\n const vertexData = new VertexData();\n vertexData.indices = indices;\n vertexData.positions = positions;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n return vertexData;\n }\n /**\n * Creates a ground mesh\n * * The parameters `width` and `height` (floats, default 1) set the width and height sizes of the ground\n * * The parameter `subdivisions` (positive integer) sets the number of subdivisions per side\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param options.width\n * @param options.height\n * @param options.subdivisions\n * @param options.subdivisionsX\n * @param options.subdivisionsY\n * @param options.updatable\n * @param scene defines the hosting scene\n * @returns the ground mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#ground\n */\n function CreateGround(name, options = {}, scene) {\n const ground = new GroundMesh(name, scene);\n ground._setReady(false);\n ground._subdivisionsX = options.subdivisionsX || options.subdivisions || 1;\n ground._subdivisionsY = options.subdivisionsY || options.subdivisions || 1;\n ground._width = options.width || 1;\n ground._height = options.height || 1;\n ground._maxX = ground._width / 2;\n ground._maxZ = ground._height / 2;\n ground._minX = -ground._maxX;\n ground._minZ = -ground._maxZ;\n const vertexData = CreateGroundVertexData(options);\n vertexData.applyToMesh(ground, options.updatable);\n ground._setReady(true);\n return ground;\n }\n /**\n * Creates a tiled ground mesh\n * * The parameters `xmin` and `xmax` (floats, default -1 and 1) set the ground minimum and maximum X coordinates\n * * The parameters `zmin` and `zmax` (floats, default -1 and 1) set the ground minimum and maximum Z coordinates\n * * The parameter `subdivisions` is a javascript object `{w: positive integer, h: positive integer}` (default `{w: 6, h: 6}`). `w` and `h` are the numbers of subdivisions on the ground width and height. Each subdivision is called a tile\n * * The parameter `precision` is a javascript object `{w: positive integer, h: positive integer}` (default `{w: 2, h: 2}`). `w` and `h` are the numbers of subdivisions on the ground width and height of each tile\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param options.xmin\n * @param options.zmin\n * @param options.xmax\n * @param options.zmax\n * @param options.subdivisions\n * @param options.subdivisions.w\n * @param options.subdivisions.h\n * @param options.precision\n * @param options.precision.w\n * @param options.precision.h\n * @param options.updatable\n * @param scene defines the hosting scene\n * @returns the tiled ground mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#tiled-ground\n */\n function CreateTiledGround(name, options, scene = null) {\n const tiledGround = new Mesh(name, scene);\n const vertexData = CreateTiledGroundVertexData(options);\n vertexData.applyToMesh(tiledGround, options.updatable);\n return tiledGround;\n }\n /**\n * Creates a ground mesh from a height map\n * * The parameter `url` sets the URL of the height map image resource.\n * * The parameters `width` and `height` (positive floats, default 10) set the ground width and height sizes.\n * * The parameter `subdivisions` (positive integer, default 1) sets the number of subdivision per side.\n * * The parameter `minHeight` (float, default 0) is the minimum altitude on the ground.\n * * The parameter `maxHeight` (float, default 1) is the maximum altitude on the ground.\n * * The parameter `colorFilter` (optional Color3, default (0.3, 0.59, 0.11) ) is the filter to apply to the image pixel colors to compute the height.\n * * The parameter `onReady` is a javascript callback function that will be called once the mesh is just built (the height map download can last some time).\n * * The parameter `alphaFilter` will filter any data where the alpha channel is below this value, defaults 0 (all data visible)\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\n * @param name defines the name of the mesh\n * @param url defines the url to the height map\n * @param options defines the options used to create the mesh\n * @param options.width\n * @param options.height\n * @param options.subdivisions\n * @param options.minHeight\n * @param options.maxHeight\n * @param options.colorFilter\n * @param options.alphaFilter\n * @param options.updatable\n * @param options.onReady\n * @param scene defines the hosting scene\n * @returns the ground mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set/height_map\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#ground-from-a-height-map\n */\n function CreateGroundFromHeightMap(name, url, options = {}, scene = null) {\n const width = options.width || 10.0;\n const height = options.height || 10.0;\n const subdivisions = options.subdivisions || 1 | 0;\n const minHeight = options.minHeight || 0.0;\n const maxHeight = options.maxHeight || 1.0;\n const filter = options.colorFilter || new Color3(0.3, 0.59, 0.11);\n const alphaFilter = options.alphaFilter || 0.0;\n const updatable = options.updatable;\n const onReady = options.onReady;\n scene = scene || EngineStore.LastCreatedScene;\n const ground = new GroundMesh(name, scene);\n ground._subdivisionsX = subdivisions;\n ground._subdivisionsY = subdivisions;\n ground._width = width;\n ground._height = height;\n ground._maxX = ground._width / 2.0;\n ground._maxZ = ground._height / 2.0;\n ground._minX = -ground._maxX;\n ground._minZ = -ground._maxZ;\n ground._setReady(false);\n const onload = (img) => {\n const bufferWidth = img.width;\n const bufferHeight = img.height;\n if (scene.isDisposed) {\n return;\n }\n const buffer = scene === null || scene === void 0 ? void 0 : scene.getEngine().resizeImageBitmap(img, bufferWidth, bufferHeight);\n const vertexData = CreateGroundFromHeightMapVertexData({\n width: width,\n height: height,\n subdivisions: subdivisions,\n minHeight: minHeight,\n maxHeight: maxHeight,\n colorFilter: filter,\n buffer: buffer,\n bufferWidth: bufferWidth,\n bufferHeight: bufferHeight,\n alphaFilter: alphaFilter,\n });\n vertexData.applyToMesh(ground, updatable);\n //execute ready callback, if set\n if (onReady) {\n onReady(ground);\n }\n ground._setReady(true);\n };\n Tools.LoadImage(url, onload, () => { }, scene.offlineProvider);\n return ground;\n }\n /**\n * Class containing static functions to help procedurally build meshes\n * @deprecated use the functions directly from the module\n */\n const GroundBuilder = {\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CreateGround,\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CreateGroundFromHeightMap,\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CreateTiledGround,\n };\n VertexData.CreateGround = CreateGroundVertexData;\n VertexData.CreateTiledGround = CreateTiledGroundVertexData;\n VertexData.CreateGroundFromHeightMap = CreateGroundFromHeightMapVertexData;\n Mesh.CreateGround = (name, width, height, subdivisions, scene, updatable) => {\n const options = {\n width,\n height,\n subdivisions,\n updatable,\n };\n return CreateGround(name, options, scene);\n };\n Mesh.CreateTiledGround = (name, xmin, zmin, xmax, zmax, subdivisions, precision, scene, updatable) => {\n const options = {\n xmin,\n zmin,\n xmax,\n zmax,\n subdivisions,\n precision,\n updatable,\n };\n return CreateTiledGround(name, options, scene);\n };\n Mesh.CreateGroundFromHeightMap = (name, url, width, height, subdivisions, minHeight, maxHeight, scene, updatable, onReady, alphaFilter) => {\n const options = {\n width,\n height,\n subdivisions,\n minHeight,\n maxHeight,\n updatable,\n onReady,\n alphaFilter,\n };\n return CreateGroundFromHeightMap(name, url, options, scene);\n };\n\n /**\n * Creates the VertexData for a box\n * @param options an object used to set the following optional parameters for the box, required but can be empty\n * * size sets the width, height and depth of the box to the value of size, optional default 1\n * * width sets the width (x direction) of the box, overwrites the width set by size, optional, default size\n * * height sets the height (y direction) of the box, overwrites the height set by size, optional, default size\n * * depth sets the depth (z direction) of the box, overwrites the depth set by size, optional, default size\n * * faceUV an array of 6 Vector4 elements used to set different images to each box side\n * * faceColors an array of 6 Color3 elements used to set different colors to each box side\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.size\n * @param options.width\n * @param options.height\n * @param options.depth\n * @param options.faceUV\n * @param options.faceColors\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param options.wrap\n * @param options.topBaseAt\n * @param options.bottomBaseAt\n * @returns the VertexData of the box\n */\n function CreateBoxVertexData(options) {\n const nbFaces = 6;\n let indices = [0, 1, 2, 0, 2, 3, 4, 5, 6, 4, 6, 7, 8, 9, 10, 8, 10, 11, 12, 13, 14, 12, 14, 15, 16, 17, 18, 16, 18, 19, 20, 21, 22, 20, 22, 23];\n const normals = [\n 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, 0, 1, 0, 0, 1, 0, 0,\n 1, 0, 0, 1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0,\n ];\n const uvs = [];\n let positions = [];\n const width = options.width || options.size || 1;\n const height = options.height || options.size || 1;\n const depth = options.depth || options.size || 1;\n const wrap = options.wrap || false;\n let topBaseAt = options.topBaseAt === void 0 ? 1 : options.topBaseAt;\n let bottomBaseAt = options.bottomBaseAt === void 0 ? 0 : options.bottomBaseAt;\n topBaseAt = (topBaseAt + 4) % 4; // places values as 0 to 3\n bottomBaseAt = (bottomBaseAt + 4) % 4; // places values as 0 to 3\n const topOrder = [2, 0, 3, 1];\n const bottomOrder = [2, 0, 1, 3];\n let topIndex = topOrder[topBaseAt];\n let bottomIndex = bottomOrder[bottomBaseAt];\n let basePositions = [\n 1, -1, 1, -1, -1, 1, -1, 1, 1, 1, 1, 1, 1, 1, -1, -1, 1, -1, -1, -1, -1, 1, -1, -1, 1, 1, -1, 1, -1, -1, 1, -1, 1, 1, 1, 1, -1, 1, 1, -1, -1, 1, -1, -1, -1, -1, 1, -1, -1,\n 1, 1, -1, 1, -1, 1, 1, -1, 1, 1, 1, 1, -1, 1, 1, -1, -1, -1, -1, -1, -1, -1, 1,\n ];\n if (wrap) {\n indices = [2, 3, 0, 2, 0, 1, 4, 5, 6, 4, 6, 7, 9, 10, 11, 9, 11, 8, 12, 14, 15, 12, 13, 14];\n basePositions = [\n -1, 1, 1, 1, 1, 1, 1, -1, 1, -1, -1, 1, 1, 1, -1, -1, 1, -1, -1, -1, -1, 1, -1, -1, 1, 1, 1, 1, 1, -1, 1, -1, -1, 1, -1, 1, -1, 1, -1, -1, 1, 1, -1, -1, 1, -1, -1, -1,\n ];\n let topFaceBase = [\n [1, 1, 1],\n [-1, 1, 1],\n [-1, 1, -1],\n [1, 1, -1],\n ];\n let bottomFaceBase = [\n [-1, -1, 1],\n [1, -1, 1],\n [1, -1, -1],\n [-1, -1, -1],\n ];\n const topFaceOrder = [17, 18, 19, 16];\n const bottomFaceOrder = [22, 23, 20, 21];\n while (topIndex > 0) {\n topFaceBase.unshift(topFaceBase.pop());\n topFaceOrder.unshift(topFaceOrder.pop());\n topIndex--;\n }\n while (bottomIndex > 0) {\n bottomFaceBase.unshift(bottomFaceBase.pop());\n bottomFaceOrder.unshift(bottomFaceOrder.pop());\n bottomIndex--;\n }\n topFaceBase = topFaceBase.flat();\n bottomFaceBase = bottomFaceBase.flat();\n basePositions = basePositions.concat(topFaceBase).concat(bottomFaceBase);\n indices.push(topFaceOrder[0], topFaceOrder[2], topFaceOrder[3], topFaceOrder[0], topFaceOrder[1], topFaceOrder[2]);\n indices.push(bottomFaceOrder[0], bottomFaceOrder[2], bottomFaceOrder[3], bottomFaceOrder[0], bottomFaceOrder[1], bottomFaceOrder[2]);\n }\n const scaleArray = [width / 2, height / 2, depth / 2];\n positions = basePositions.reduce((accumulator, currentValue, currentIndex) => accumulator.concat(currentValue * scaleArray[currentIndex % 3]), []);\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\n const faceUV = options.faceUV || new Array(6);\n const faceColors = options.faceColors;\n const colors = [];\n // default face colors and UV if undefined\n for (let f = 0; f < 6; f++) {\n if (faceUV[f] === undefined) {\n faceUV[f] = new Vector4(0, 0, 1, 1);\n }\n if (faceColors && faceColors[f] === undefined) {\n faceColors[f] = new Color4(1, 1, 1, 1);\n }\n }\n // Create each face in turn.\n for (let index = 0; index < nbFaces; index++) {\n uvs.push(faceUV[index].z, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - faceUV[index].w : faceUV[index].w);\n uvs.push(faceUV[index].x, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - faceUV[index].w : faceUV[index].w);\n uvs.push(faceUV[index].x, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - faceUV[index].y : faceUV[index].y);\n uvs.push(faceUV[index].z, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - faceUV[index].y : faceUV[index].y);\n if (faceColors) {\n for (let c = 0; c < 4; c++) {\n colors.push(faceColors[index].r, faceColors[index].g, faceColors[index].b, faceColors[index].a);\n }\n }\n }\n // sides\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\n // Result\n const vertexData = new VertexData();\n vertexData.indices = indices;\n vertexData.positions = positions;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n if (faceColors) {\n const totalColors = sideOrientation === VertexData.DOUBLESIDE ? colors.concat(colors) : colors;\n vertexData.colors = totalColors;\n }\n return vertexData;\n }\n /**\n * Creates a box mesh\n * * The parameter `size` sets the size (float) of each box side (default 1)\n * * You can set some different box dimensions by using the parameters `width`, `height` and `depth` (all by default have the same value of `size`)\n * * You can set different colors and different images to each box side by using the parameters `faceColors` (an array of 6 Color3 elements) and `faceUV` (an array of 6 Vector4 elements)\n * * Please read this tutorial : https://doc.babylonjs.com/features/featuresDeepDive/materials/using/texturePerBoxFace\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#box\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param options.size\n * @param options.width\n * @param options.height\n * @param options.depth\n * @param options.faceUV\n * @param options.faceColors\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param options.wrap\n * @param options.topBaseAt\n * @param options.bottomBaseAt\n * @param options.updatable\n * @param scene defines the hosting scene\n * @returns the box mesh\n */\n function CreateBox(name, options = {}, scene = null) {\n const box = new Mesh(name, scene);\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n box._originalBuilderSideOrientation = options.sideOrientation;\n const vertexData = CreateBoxVertexData(options);\n vertexData.applyToMesh(box, options.updatable);\n return box;\n }\n /**\n * Class containing static functions to help procedurally build meshes\n * @deprecated please use CreateBox directly\n */\n const BoxBuilder = {\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CreateBox,\n };\n // Side effects\n VertexData.CreateBox = CreateBoxVertexData;\n Mesh.CreateBox = (name, size, scene = null, updatable, sideOrientation) => {\n const options = {\n size,\n sideOrientation,\n updatable,\n };\n return CreateBox(name, options, scene);\n };\n\n /**\n * The EnvironmentHelper class can be used to add a fully featured non-expensive background to your scene.\n * It includes by default a skybox and a ground relying on the BackgroundMaterial.\n * It also helps with the default setup of your ImageProcessingConfiguration.\n */\n class EnvironmentHelper {\n /**\n * Creates the default options for the helper.\n * @param scene The scene the environment helper belongs to.\n */\n static _GetDefaultOptions(scene) {\n return {\n createGround: true,\n groundSize: 15,\n groundTexture: this._GroundTextureCDNUrl,\n groundColor: new Color3(0.2, 0.2, 0.3).toLinearSpace(scene.getEngine().useExactSrgbConversions).scale(3),\n groundOpacity: 0.9,\n enableGroundShadow: true,\n groundShadowLevel: 0.5,\n enableGroundMirror: false,\n groundMirrorSizeRatio: 0.3,\n groundMirrorBlurKernel: 64,\n groundMirrorAmount: 1,\n groundMirrorFresnelWeight: 1,\n groundMirrorFallOffDistance: 0,\n groundMirrorTextureType: 0,\n groundYBias: 0.00001,\n createSkybox: true,\n skyboxSize: 20,\n skyboxTexture: this._SkyboxTextureCDNUrl,\n skyboxColor: new Color3(0.2, 0.2, 0.3).toLinearSpace(scene.getEngine().useExactSrgbConversions).scale(3),\n backgroundYRotation: 0,\n sizeAuto: true,\n rootPosition: Vector3.Zero(),\n setupImageProcessing: true,\n environmentTexture: this._EnvironmentTextureCDNUrl,\n cameraExposure: 0.8,\n cameraContrast: 1.2,\n toneMappingEnabled: true,\n };\n }\n /**\n * Gets the root mesh created by the helper.\n */\n get rootMesh() {\n return this._rootMesh;\n }\n /**\n * Gets the skybox created by the helper.\n */\n get skybox() {\n return this._skybox;\n }\n /**\n * Gets the skybox texture created by the helper.\n */\n get skyboxTexture() {\n return this._skyboxTexture;\n }\n /**\n * Gets the skybox material created by the helper.\n */\n get skyboxMaterial() {\n return this._skyboxMaterial;\n }\n /**\n * Gets the ground mesh created by the helper.\n */\n get ground() {\n return this._ground;\n }\n /**\n * Gets the ground texture created by the helper.\n */\n get groundTexture() {\n return this._groundTexture;\n }\n /**\n * Gets the ground mirror created by the helper.\n */\n get groundMirror() {\n return this._groundMirror;\n }\n /**\n * Gets the ground mirror render list to helps pushing the meshes\n * you wish in the ground reflection.\n */\n get groundMirrorRenderList() {\n if (this._groundMirror) {\n return this._groundMirror.renderList;\n }\n return null;\n }\n /**\n * Gets the ground material created by the helper.\n */\n get groundMaterial() {\n return this._groundMaterial;\n }\n /**\n * constructor\n * @param options Defines the options we want to customize the helper\n * @param scene The scene to add the material to\n */\n constructor(options, scene) {\n this._errorHandler = (message, exception) => {\n this.onErrorObservable.notifyObservers({ message: message, exception: exception });\n };\n this._options = Object.assign(Object.assign({}, EnvironmentHelper._GetDefaultOptions(scene)), options);\n this._scene = scene;\n this.onErrorObservable = new Observable$1();\n this._setupBackground();\n this._setupImageProcessing();\n }\n /**\n * Updates the background according to the new options\n * @param options\n */\n updateOptions(options) {\n const newOptions = Object.assign(Object.assign({}, this._options), options);\n if (this._ground && !newOptions.createGround) {\n this._ground.dispose();\n this._ground = null;\n }\n if (this._groundMaterial && !newOptions.createGround) {\n this._groundMaterial.dispose();\n this._groundMaterial = null;\n }\n if (this._groundTexture) {\n if (this._options.groundTexture != newOptions.groundTexture) {\n this._groundTexture.dispose();\n this._groundTexture = null;\n }\n }\n if (this._skybox && !newOptions.createSkybox) {\n this._skybox.dispose();\n this._skybox = null;\n }\n if (this._skyboxMaterial && !newOptions.createSkybox) {\n this._skyboxMaterial.dispose();\n this._skyboxMaterial = null;\n }\n if (this._skyboxTexture) {\n if (this._options.skyboxTexture != newOptions.skyboxTexture) {\n this._skyboxTexture.dispose();\n this._skyboxTexture = null;\n }\n }\n if (this._groundMirror && !newOptions.enableGroundMirror) {\n this._groundMirror.dispose();\n this._groundMirror = null;\n }\n if (this._scene.environmentTexture) {\n if (this._options.environmentTexture != newOptions.environmentTexture) {\n this._scene.environmentTexture.dispose();\n }\n }\n this._options = newOptions;\n this._setupBackground();\n this._setupImageProcessing();\n }\n /**\n * Sets the primary color of all the available elements.\n * @param color the main color to affect to the ground and the background\n */\n setMainColor(color) {\n if (this.groundMaterial) {\n this.groundMaterial.primaryColor = color;\n }\n if (this.skyboxMaterial) {\n this.skyboxMaterial.primaryColor = color;\n }\n if (this.groundMirror) {\n this.groundMirror.clearColor = new Color4(color.r, color.g, color.b, 1.0);\n }\n }\n /**\n * Setup the image processing according to the specified options.\n */\n _setupImageProcessing() {\n if (this._options.setupImageProcessing) {\n this._scene.imageProcessingConfiguration.contrast = this._options.cameraContrast;\n this._scene.imageProcessingConfiguration.exposure = this._options.cameraExposure;\n this._scene.imageProcessingConfiguration.toneMappingEnabled = this._options.toneMappingEnabled;\n this._setupEnvironmentTexture();\n }\n }\n /**\n * Setup the environment texture according to the specified options.\n */\n _setupEnvironmentTexture() {\n if (this._scene.environmentTexture) {\n return;\n }\n if (this._options.environmentTexture instanceof BaseTexture) {\n this._scene.environmentTexture = this._options.environmentTexture;\n return;\n }\n const environmentTexture = CubeTexture.CreateFromPrefilteredData(this._options.environmentTexture, this._scene);\n this._scene.environmentTexture = environmentTexture;\n }\n /**\n * Setup the background according to the specified options.\n */\n _setupBackground() {\n if (!this._rootMesh) {\n this._rootMesh = new Mesh(\"BackgroundHelper\", this._scene);\n }\n this._rootMesh.rotation.y = this._options.backgroundYRotation;\n const sceneSize = this._getSceneSize();\n if (this._options.createGround) {\n this._setupGround(sceneSize);\n this._setupGroundMaterial();\n this._setupGroundDiffuseTexture();\n if (this._options.enableGroundMirror) {\n this._setupGroundMirrorTexture(sceneSize);\n }\n this._setupMirrorInGroundMaterial();\n }\n if (this._options.createSkybox) {\n this._setupSkybox(sceneSize);\n this._setupSkyboxMaterial();\n this._setupSkyboxReflectionTexture();\n }\n this._rootMesh.position.x = sceneSize.rootPosition.x;\n this._rootMesh.position.z = sceneSize.rootPosition.z;\n this._rootMesh.position.y = sceneSize.rootPosition.y;\n }\n /**\n * Get the scene sizes according to the setup.\n */\n _getSceneSize() {\n let groundSize = this._options.groundSize;\n let skyboxSize = this._options.skyboxSize;\n let rootPosition = this._options.rootPosition;\n if (!this._scene.meshes || this._scene.meshes.length === 1) {\n // 1 only means the root of the helper.\n return { groundSize, skyboxSize, rootPosition };\n }\n const sceneExtends = this._scene.getWorldExtends((mesh) => {\n return mesh !== this._ground && mesh !== this._rootMesh && mesh !== this._skybox;\n });\n const sceneDiagonal = sceneExtends.max.subtract(sceneExtends.min);\n if (this._options.sizeAuto) {\n if (this._scene.activeCamera instanceof ArcRotateCamera && this._scene.activeCamera.upperRadiusLimit) {\n groundSize = this._scene.activeCamera.upperRadiusLimit * 2;\n skyboxSize = groundSize;\n }\n const sceneDiagonalLenght = sceneDiagonal.length();\n if (sceneDiagonalLenght > groundSize) {\n groundSize = sceneDiagonalLenght * 2;\n skyboxSize = groundSize;\n }\n // 10 % bigger.\n groundSize *= 1.1;\n skyboxSize *= 1.5;\n rootPosition = sceneExtends.min.add(sceneDiagonal.scale(0.5));\n rootPosition.y = sceneExtends.min.y - this._options.groundYBias;\n }\n return { groundSize, skyboxSize, rootPosition };\n }\n /**\n * Setup the ground according to the specified options.\n * @param sceneSize\n */\n _setupGround(sceneSize) {\n if (!this._ground || this._ground.isDisposed()) {\n this._ground = CreatePlane(\"BackgroundPlane\", { size: sceneSize.groundSize }, this._scene);\n this._ground.rotation.x = Math.PI / 2; // Face up by default.\n this._ground.parent = this._rootMesh;\n this._ground.onDisposeObservable.add(() => {\n this._ground = null;\n });\n }\n this._ground.receiveShadows = this._options.enableGroundShadow;\n }\n /**\n * Setup the ground material according to the specified options.\n */\n _setupGroundMaterial() {\n if (!this._groundMaterial) {\n this._groundMaterial = new BackgroundMaterial(\"BackgroundPlaneMaterial\", this._scene);\n }\n this._groundMaterial.alpha = this._options.groundOpacity;\n this._groundMaterial.alphaMode = 8;\n this._groundMaterial.shadowLevel = this._options.groundShadowLevel;\n this._groundMaterial.primaryColor = this._options.groundColor;\n this._groundMaterial.useRGBColor = false;\n this._groundMaterial.enableNoise = true;\n if (this._ground) {\n this._ground.material = this._groundMaterial;\n }\n }\n /**\n * Setup the ground diffuse texture according to the specified options.\n */\n _setupGroundDiffuseTexture() {\n if (!this._groundMaterial) {\n return;\n }\n if (this._groundTexture) {\n return;\n }\n if (this._options.groundTexture instanceof BaseTexture) {\n this._groundMaterial.diffuseTexture = this._options.groundTexture;\n return;\n }\n this._groundTexture = new Texture(this._options.groundTexture, this._scene, undefined, undefined, undefined, undefined, this._errorHandler);\n this._groundTexture.gammaSpace = false;\n this._groundTexture.hasAlpha = true;\n this._groundMaterial.diffuseTexture = this._groundTexture;\n }\n /**\n * Setup the ground mirror texture according to the specified options.\n * @param sceneSize\n */\n _setupGroundMirrorTexture(sceneSize) {\n const wrapping = Texture.CLAMP_ADDRESSMODE;\n if (!this._groundMirror) {\n this._groundMirror = new MirrorTexture(\"BackgroundPlaneMirrorTexture\", { ratio: this._options.groundMirrorSizeRatio }, this._scene, false, this._options.groundMirrorTextureType, Texture.BILINEAR_SAMPLINGMODE, true);\n this._groundMirror.mirrorPlane = new Plane(0, -1, 0, sceneSize.rootPosition.y);\n this._groundMirror.anisotropicFilteringLevel = 1;\n this._groundMirror.wrapU = wrapping;\n this._groundMirror.wrapV = wrapping;\n if (this._groundMirror.renderList) {\n for (let i = 0; i < this._scene.meshes.length; i++) {\n const mesh = this._scene.meshes[i];\n if (mesh !== this._ground && mesh !== this._skybox && mesh !== this._rootMesh) {\n this._groundMirror.renderList.push(mesh);\n }\n }\n }\n }\n const gammaGround = this._options.groundColor.toGammaSpace(this._scene.getEngine().useExactSrgbConversions);\n this._groundMirror.clearColor = new Color4(gammaGround.r, gammaGround.g, gammaGround.b, 1);\n this._groundMirror.adaptiveBlurKernel = this._options.groundMirrorBlurKernel;\n }\n /**\n * Setup the ground to receive the mirror texture.\n */\n _setupMirrorInGroundMaterial() {\n if (this._groundMaterial) {\n this._groundMaterial.reflectionTexture = this._groundMirror;\n this._groundMaterial.reflectionFresnel = true;\n this._groundMaterial.reflectionAmount = this._options.groundMirrorAmount;\n this._groundMaterial.reflectionStandardFresnelWeight = this._options.groundMirrorFresnelWeight;\n this._groundMaterial.reflectionFalloffDistance = this._options.groundMirrorFallOffDistance;\n }\n }\n /**\n * Setup the skybox according to the specified options.\n * @param sceneSize\n */\n _setupSkybox(sceneSize) {\n if (!this._skybox || this._skybox.isDisposed()) {\n this._skybox = CreateBox(\"BackgroundSkybox\", { size: sceneSize.skyboxSize, sideOrientation: Mesh.BACKSIDE }, this._scene);\n this._skybox.onDisposeObservable.add(() => {\n this._skybox = null;\n });\n }\n this._skybox.parent = this._rootMesh;\n }\n /**\n * Setup the skybox material according to the specified options.\n */\n _setupSkyboxMaterial() {\n if (!this._skybox) {\n return;\n }\n if (!this._skyboxMaterial) {\n this._skyboxMaterial = new BackgroundMaterial(\"BackgroundSkyboxMaterial\", this._scene);\n }\n this._skyboxMaterial.useRGBColor = false;\n this._skyboxMaterial.primaryColor = this._options.skyboxColor;\n this._skyboxMaterial.enableNoise = true;\n this._skybox.material = this._skyboxMaterial;\n }\n /**\n * Setup the skybox reflection texture according to the specified options.\n */\n _setupSkyboxReflectionTexture() {\n if (!this._skyboxMaterial) {\n return;\n }\n if (this._skyboxTexture) {\n return;\n }\n if (this._options.skyboxTexture instanceof BaseTexture) {\n this._skyboxMaterial.reflectionTexture = this._options.skyboxTexture;\n return;\n }\n this._skyboxTexture = new CubeTexture(this._options.skyboxTexture, this._scene, undefined, undefined, undefined, undefined, this._errorHandler);\n this._skyboxTexture.coordinatesMode = Texture.SKYBOX_MODE;\n this._skyboxTexture.gammaSpace = false;\n this._skyboxMaterial.reflectionTexture = this._skyboxTexture;\n }\n /**\n * Dispose all the elements created by the Helper.\n */\n dispose() {\n if (this._groundMaterial) {\n this._groundMaterial.dispose(true, true);\n }\n if (this._skyboxMaterial) {\n this._skyboxMaterial.dispose(true, true);\n }\n this._rootMesh.dispose(false);\n }\n }\n /**\n * Default ground texture URL.\n */\n EnvironmentHelper._GroundTextureCDNUrl = \"https://assets.babylonjs.com/environments/backgroundGround.png\";\n /**\n * Default skybox texture URL.\n */\n EnvironmentHelper._SkyboxTextureCDNUrl = \"https://assets.babylonjs.com/environments/backgroundSkybox.dds\";\n /**\n * Default environment texture URL.\n */\n EnvironmentHelper._EnvironmentTextureCDNUrl = \"https://assets.babylonjs.com/environments/environmentSpecular.env\";\n\n /**\n * Manage the keyboard inputs to control the movement of a free camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\n */\n class FreeCameraKeyboardMoveInput {\n constructor() {\n /**\n * Gets or Set the list of keyboard keys used to control the forward move of the camera.\n */\n this.keysUp = [38];\n /**\n * Gets or Set the list of keyboard keys used to control the upward move of the camera.\n */\n this.keysUpward = [33];\n /**\n * Gets or Set the list of keyboard keys used to control the backward move of the camera.\n */\n this.keysDown = [40];\n /**\n * Gets or Set the list of keyboard keys used to control the downward move of the camera.\n */\n this.keysDownward = [34];\n /**\n * Gets or Set the list of keyboard keys used to control the left strafe move of the camera.\n */\n this.keysLeft = [37];\n /**\n * Gets or Set the list of keyboard keys used to control the right strafe move of the camera.\n */\n this.keysRight = [39];\n /**\n * Defines the pointer angular sensibility along the X and Y axis or how fast is the camera rotating.\n */\n this.rotationSpeed = 0.5;\n /**\n * Gets or Set the list of keyboard keys used to control the left rotation move of the camera.\n */\n this.keysRotateLeft = [];\n /**\n * Gets or Set the list of keyboard keys used to control the right rotation move of the camera.\n */\n this.keysRotateRight = [];\n /**\n * Gets or Set the list of keyboard keys used to control the up rotation move of the camera.\n */\n this.keysRotateUp = [];\n /**\n * Gets or Set the list of keyboard keys used to control the down rotation move of the camera.\n */\n this.keysRotateDown = [];\n this._keys = new Array();\n }\n /**\n * Attach the input controls to a specific dom element to get the input from.\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\n */\n attachControl(noPreventDefault) {\n // eslint-disable-next-line prefer-rest-params\n noPreventDefault = Tools.BackCompatCameraNoPreventDefault(arguments);\n if (this._onCanvasBlurObserver) {\n return;\n }\n this._scene = this.camera.getScene();\n this._engine = this._scene.getEngine();\n this._onCanvasBlurObserver = this._engine.onCanvasBlurObservable.add(() => {\n this._keys.length = 0;\n });\n this._onKeyboardObserver = this._scene.onKeyboardObservable.add((info) => {\n const evt = info.event;\n if (!evt.metaKey) {\n if (info.type === KeyboardEventTypes.KEYDOWN) {\n if (this.keysUp.indexOf(evt.keyCode) !== -1 ||\n this.keysDown.indexOf(evt.keyCode) !== -1 ||\n this.keysLeft.indexOf(evt.keyCode) !== -1 ||\n this.keysRight.indexOf(evt.keyCode) !== -1 ||\n this.keysUpward.indexOf(evt.keyCode) !== -1 ||\n this.keysDownward.indexOf(evt.keyCode) !== -1 ||\n this.keysRotateLeft.indexOf(evt.keyCode) !== -1 ||\n this.keysRotateRight.indexOf(evt.keyCode) !== -1 ||\n this.keysRotateUp.indexOf(evt.keyCode) !== -1 ||\n this.keysRotateDown.indexOf(evt.keyCode) !== -1) {\n const index = this._keys.indexOf(evt.keyCode);\n if (index === -1) {\n this._keys.push(evt.keyCode);\n }\n if (!noPreventDefault) {\n evt.preventDefault();\n }\n }\n }\n else {\n if (this.keysUp.indexOf(evt.keyCode) !== -1 ||\n this.keysDown.indexOf(evt.keyCode) !== -1 ||\n this.keysLeft.indexOf(evt.keyCode) !== -1 ||\n this.keysRight.indexOf(evt.keyCode) !== -1 ||\n this.keysUpward.indexOf(evt.keyCode) !== -1 ||\n this.keysDownward.indexOf(evt.keyCode) !== -1 ||\n this.keysRotateLeft.indexOf(evt.keyCode) !== -1 ||\n this.keysRotateRight.indexOf(evt.keyCode) !== -1 ||\n this.keysRotateUp.indexOf(evt.keyCode) !== -1 ||\n this.keysRotateDown.indexOf(evt.keyCode) !== -1) {\n const index = this._keys.indexOf(evt.keyCode);\n if (index >= 0) {\n this._keys.splice(index, 1);\n }\n if (!noPreventDefault) {\n evt.preventDefault();\n }\n }\n }\n }\n });\n }\n /**\n * Detach the current controls from the specified dom element.\n */\n detachControl() {\n if (this._scene) {\n if (this._onKeyboardObserver) {\n this._scene.onKeyboardObservable.remove(this._onKeyboardObserver);\n }\n if (this._onCanvasBlurObserver) {\n this._engine.onCanvasBlurObservable.remove(this._onCanvasBlurObserver);\n }\n this._onKeyboardObserver = null;\n this._onCanvasBlurObserver = null;\n }\n this._keys.length = 0;\n }\n /**\n * Update the current camera state depending on the inputs that have been used this frame.\n * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop.\n */\n checkInputs() {\n if (this._onKeyboardObserver) {\n const camera = this.camera;\n // Keyboard\n for (let index = 0; index < this._keys.length; index++) {\n const keyCode = this._keys[index];\n const speed = camera._computeLocalCameraSpeed();\n if (this.keysLeft.indexOf(keyCode) !== -1) {\n camera._localDirection.copyFromFloats(-speed, 0, 0);\n }\n else if (this.keysUp.indexOf(keyCode) !== -1) {\n camera._localDirection.copyFromFloats(0, 0, speed);\n }\n else if (this.keysRight.indexOf(keyCode) !== -1) {\n camera._localDirection.copyFromFloats(speed, 0, 0);\n }\n else if (this.keysDown.indexOf(keyCode) !== -1) {\n camera._localDirection.copyFromFloats(0, 0, -speed);\n }\n else if (this.keysUpward.indexOf(keyCode) !== -1) {\n camera._localDirection.copyFromFloats(0, speed, 0);\n }\n else if (this.keysDownward.indexOf(keyCode) !== -1) {\n camera._localDirection.copyFromFloats(0, -speed, 0);\n }\n else if (this.keysRotateLeft.indexOf(keyCode) !== -1) {\n camera._localDirection.copyFromFloats(0, 0, 0);\n camera.cameraRotation.y -= this._getLocalRotation();\n }\n else if (this.keysRotateRight.indexOf(keyCode) !== -1) {\n camera._localDirection.copyFromFloats(0, 0, 0);\n camera.cameraRotation.y += this._getLocalRotation();\n }\n else if (this.keysRotateUp.indexOf(keyCode) !== -1) {\n camera._localDirection.copyFromFloats(0, 0, 0);\n camera.cameraRotation.x -= this._getLocalRotation();\n }\n else if (this.keysRotateDown.indexOf(keyCode) !== -1) {\n camera._localDirection.copyFromFloats(0, 0, 0);\n camera.cameraRotation.x += this._getLocalRotation();\n }\n if (camera.getScene().useRightHandedSystem) {\n camera._localDirection.z *= -1;\n }\n camera.getViewMatrix().invertToRef(camera._cameraTransformMatrix);\n Vector3.TransformNormalToRef(camera._localDirection, camera._cameraTransformMatrix, camera._transformedDirection);\n camera.cameraDirection.addInPlace(camera._transformedDirection);\n }\n }\n }\n /**\n * Gets the class name of the current input.\n * @returns the class name\n */\n getClassName() {\n return \"FreeCameraKeyboardMoveInput\";\n }\n /** @internal */\n _onLostFocus() {\n this._keys.length = 0;\n }\n /**\n * Get the friendly name associated with the input class.\n * @returns the input friendly name\n */\n getSimpleName() {\n return \"keyboard\";\n }\n _getLocalRotation() {\n let rotation = (this.rotationSpeed * this._engine.getDeltaTime()) / 1000;\n if (this.camera.getScene().useRightHandedSystem) {\n rotation *= -1;\n }\n if (this.camera.parent && this.camera.parent._getWorldMatrixDeterminant() < 0) {\n rotation *= -1;\n }\n return rotation;\n }\n }\n __decorate$1([\n serialize()\n ], FreeCameraKeyboardMoveInput.prototype, \"keysUp\", void 0);\n __decorate$1([\n serialize()\n ], FreeCameraKeyboardMoveInput.prototype, \"keysUpward\", void 0);\n __decorate$1([\n serialize()\n ], FreeCameraKeyboardMoveInput.prototype, \"keysDown\", void 0);\n __decorate$1([\n serialize()\n ], FreeCameraKeyboardMoveInput.prototype, \"keysDownward\", void 0);\n __decorate$1([\n serialize()\n ], FreeCameraKeyboardMoveInput.prototype, \"keysLeft\", void 0);\n __decorate$1([\n serialize()\n ], FreeCameraKeyboardMoveInput.prototype, \"keysRight\", void 0);\n __decorate$1([\n serialize()\n ], FreeCameraKeyboardMoveInput.prototype, \"rotationSpeed\", void 0);\n __decorate$1([\n serialize()\n ], FreeCameraKeyboardMoveInput.prototype, \"keysRotateLeft\", void 0);\n __decorate$1([\n serialize()\n ], FreeCameraKeyboardMoveInput.prototype, \"keysRotateRight\", void 0);\n __decorate$1([\n serialize()\n ], FreeCameraKeyboardMoveInput.prototype, \"keysRotateUp\", void 0);\n __decorate$1([\n serialize()\n ], FreeCameraKeyboardMoveInput.prototype, \"keysRotateDown\", void 0);\n CameraInputTypes[\"FreeCameraKeyboardMoveInput\"] = FreeCameraKeyboardMoveInput;\n\n /**\n * Manage the mouse inputs to control the movement of a free camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\n */\n class FreeCameraMouseInput {\n /**\n * Manage the mouse inputs to control the movement of a free camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\n * @param touchEnabled Defines if touch is enabled or not\n */\n constructor(\n /**\n * Define if touch is enabled in the mouse input\n */\n touchEnabled = true) {\n this.touchEnabled = touchEnabled;\n /**\n * Defines the buttons associated with the input to handle camera move.\n */\n this.buttons = [0, 1, 2];\n /**\n * Defines the pointer angular sensibility along the X and Y axis or how fast is the camera rotating.\n */\n this.angularSensibility = 2000.0;\n this._previousPosition = null;\n /**\n * Observable for when a pointer move event occurs containing the move offset\n */\n this.onPointerMovedObservable = new Observable$1();\n /**\n * @internal\n * If the camera should be rotated automatically based on pointer movement\n */\n this._allowCameraRotation = true;\n this._currentActiveButton = -1;\n this._activePointerId = -1;\n }\n /**\n * Attach the input controls to a specific dom element to get the input from.\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\n */\n attachControl(noPreventDefault) {\n // eslint-disable-next-line prefer-rest-params\n noPreventDefault = Tools.BackCompatCameraNoPreventDefault(arguments);\n const engine = this.camera.getEngine();\n const element = engine.getInputElement();\n if (!this._pointerInput) {\n this._pointerInput = (p) => {\n const evt = p.event;\n const isTouch = evt.pointerType === \"touch\";\n if (engine.isInVRExclusivePointerMode) {\n return;\n }\n if (!this.touchEnabled && isTouch) {\n return;\n }\n if (p.type !== PointerEventTypes.POINTERMOVE && this.buttons.indexOf(evt.button) === -1) {\n return;\n }\n const srcElement = evt.target;\n if (p.type === PointerEventTypes.POINTERDOWN) {\n // If the input is touch with more than one touch OR if the input is mouse and there is already an active button, return\n if ((isTouch && this._activePointerId !== -1) || (!isTouch && this._currentActiveButton !== -1)) {\n return;\n }\n this._activePointerId = evt.pointerId;\n try {\n srcElement === null || srcElement === void 0 ? void 0 : srcElement.setPointerCapture(evt.pointerId);\n }\n catch (e) {\n //Nothing to do with the error. Execution will continue.\n }\n if (this._currentActiveButton === -1) {\n this._currentActiveButton = evt.button;\n }\n this._previousPosition = {\n x: evt.clientX,\n y: evt.clientY,\n };\n if (!noPreventDefault) {\n evt.preventDefault();\n element && element.focus();\n }\n // This is required to move while pointer button is down\n if (engine.isPointerLock && this._onMouseMove) {\n this._onMouseMove(p.event);\n }\n }\n else if (p.type === PointerEventTypes.POINTERUP) {\n // If input is touch with a different touch id OR if input is mouse with a different button, return\n if ((isTouch && this._activePointerId !== evt.pointerId) || (!isTouch && this._currentActiveButton !== evt.button)) {\n return;\n }\n try {\n srcElement === null || srcElement === void 0 ? void 0 : srcElement.releasePointerCapture(evt.pointerId);\n }\n catch (e) {\n //Nothing to do with the error.\n }\n this._currentActiveButton = -1;\n this._previousPosition = null;\n if (!noPreventDefault) {\n evt.preventDefault();\n }\n this._activePointerId = -1;\n }\n else if (p.type === PointerEventTypes.POINTERMOVE && (this._activePointerId === evt.pointerId || !isTouch)) {\n if (engine.isPointerLock && this._onMouseMove) {\n this._onMouseMove(p.event);\n }\n else if (this._previousPosition) {\n let offsetX = evt.clientX - this._previousPosition.x;\n const offsetY = evt.clientY - this._previousPosition.y;\n if (this.camera.getScene().useRightHandedSystem) {\n offsetX *= -1;\n }\n if (this.camera.parent && this.camera.parent._getWorldMatrixDeterminant() < 0) {\n offsetX *= -1;\n }\n if (this._allowCameraRotation) {\n this.camera.cameraRotation.y += offsetX / this.angularSensibility;\n this.camera.cameraRotation.x += offsetY / this.angularSensibility;\n }\n this.onPointerMovedObservable.notifyObservers({ offsetX: offsetX, offsetY: offsetY });\n this._previousPosition = {\n x: evt.clientX,\n y: evt.clientY,\n };\n if (!noPreventDefault) {\n evt.preventDefault();\n }\n }\n }\n };\n }\n this._onMouseMove = (evt) => {\n if (!engine.isPointerLock) {\n return;\n }\n if (engine.isInVRExclusivePointerMode) {\n return;\n }\n let offsetX = evt.movementX;\n if (this.camera.getScene().useRightHandedSystem) {\n offsetX *= -1;\n }\n if (this.camera.parent && this.camera.parent._getWorldMatrixDeterminant() < 0) {\n offsetX *= -1;\n }\n this.camera.cameraRotation.y += offsetX / this.angularSensibility;\n const offsetY = evt.movementY;\n this.camera.cameraRotation.x += offsetY / this.angularSensibility;\n this._previousPosition = null;\n if (!noPreventDefault) {\n evt.preventDefault();\n }\n };\n this._observer = this.camera\n .getScene()\n ._inputManager._addCameraPointerObserver(this._pointerInput, PointerEventTypes.POINTERDOWN | PointerEventTypes.POINTERUP | PointerEventTypes.POINTERMOVE);\n if (element) {\n this._contextMenuBind = this.onContextMenu.bind(this);\n element.addEventListener(\"contextmenu\", this._contextMenuBind, false); // TODO: We need to figure out how to handle this for Native\n }\n }\n /**\n * Called on JS contextmenu event.\n * Override this method to provide functionality.\n * @param evt\n */\n onContextMenu(evt) {\n evt.preventDefault();\n }\n /**\n * Detach the current controls from the specified dom element.\n */\n detachControl() {\n if (this._observer) {\n this.camera.getScene()._inputManager._removeCameraPointerObserver(this._observer);\n if (this._contextMenuBind) {\n const engine = this.camera.getEngine();\n const element = engine.getInputElement();\n element && element.removeEventListener(\"contextmenu\", this._contextMenuBind);\n }\n if (this.onPointerMovedObservable) {\n this.onPointerMovedObservable.clear();\n }\n this._observer = null;\n this._onMouseMove = null;\n this._previousPosition = null;\n }\n this._activePointerId = -1;\n this._currentActiveButton = -1;\n }\n /**\n * Gets the class name of the current input.\n * @returns the class name\n */\n getClassName() {\n return \"FreeCameraMouseInput\";\n }\n /**\n * Get the friendly name associated with the input class.\n * @returns the input friendly name\n */\n getSimpleName() {\n return \"mouse\";\n }\n }\n __decorate$1([\n serialize()\n ], FreeCameraMouseInput.prototype, \"buttons\", void 0);\n __decorate$1([\n serialize()\n ], FreeCameraMouseInput.prototype, \"angularSensibility\", void 0);\n CameraInputTypes[\"FreeCameraMouseInput\"] = FreeCameraMouseInput;\n\n /**\n * Base class for mouse wheel input..\n * See FollowCameraMouseWheelInput in src/Cameras/Inputs/freeCameraMouseWheelInput.ts\n * for example usage.\n */\n class BaseCameraMouseWheelInput {\n constructor() {\n /**\n * How fast is the camera moves in relation to X axis mouseWheel events.\n * Use negative value to reverse direction.\n */\n this.wheelPrecisionX = 3.0;\n /**\n * How fast is the camera moves in relation to Y axis mouseWheel events.\n * Use negative value to reverse direction.\n */\n this.wheelPrecisionY = 3.0;\n /**\n * How fast is the camera moves in relation to Z axis mouseWheel events.\n * Use negative value to reverse direction.\n */\n this.wheelPrecisionZ = 3.0;\n /**\n * Observable for when a mouse wheel move event occurs.\n */\n this.onChangedObservable = new Observable$1();\n /**\n * Incremental value of multiple mouse wheel movements of the X axis.\n * Should be zero-ed when read.\n */\n this._wheelDeltaX = 0;\n /**\n * Incremental value of multiple mouse wheel movements of the Y axis.\n * Should be zero-ed when read.\n */\n this._wheelDeltaY = 0;\n /**\n * Incremental value of multiple mouse wheel movements of the Z axis.\n * Should be zero-ed when read.\n */\n this._wheelDeltaZ = 0;\n /**\n * Firefox uses a different scheme to report scroll distances to other\n * browsers. Rather than use complicated methods to calculate the exact\n * multiple we need to apply, let's just cheat and use a constant.\n * https://developer.mozilla.org/en-US/docs/Web/API/WheelEvent/deltaMode\n * https://stackoverflow.com/questions/20110224/what-is-the-height-of-a-line-in-a-wheel-event-deltamode-dom-delta-line\n */\n this._ffMultiplier = 12;\n /**\n * Different event attributes for wheel data fall into a few set ranges.\n * Some relevant but dated date here:\n * https://stackoverflow.com/questions/5527601/normalizing-mousewheel-speed-across-browsers\n */\n this._normalize = 120;\n }\n /**\n * Attach the input controls to a specific dom element to get the input from.\n * @param noPreventDefault Defines whether event caught by the controls\n * should call preventdefault().\n * (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\n */\n attachControl(noPreventDefault) {\n noPreventDefault = Tools.BackCompatCameraNoPreventDefault(arguments);\n this._wheel = (pointer) => {\n // sanity check - this should be a PointerWheel event.\n if (pointer.type !== PointerEventTypes.POINTERWHEEL) {\n return;\n }\n const event = pointer.event;\n const platformScale = event.deltaMode === EventConstants.DOM_DELTA_LINE ? this._ffMultiplier : 1; // If this happens to be set to DOM_DELTA_LINE, adjust accordingly\n this._wheelDeltaX += (this.wheelPrecisionX * platformScale * event.deltaX) / this._normalize;\n this._wheelDeltaY -= (this.wheelPrecisionY * platformScale * event.deltaY) / this._normalize;\n this._wheelDeltaZ += (this.wheelPrecisionZ * platformScale * event.deltaZ) / this._normalize;\n if (event.preventDefault) {\n if (!noPreventDefault) {\n event.preventDefault();\n }\n }\n };\n this._observer = this.camera.getScene()._inputManager._addCameraPointerObserver(this._wheel, PointerEventTypes.POINTERWHEEL);\n }\n /**\n * Detach the current controls from the specified dom element.\n */\n detachControl() {\n if (this._observer) {\n this.camera.getScene()._inputManager._removeCameraPointerObserver(this._observer);\n this._observer = null;\n this._wheel = null;\n }\n if (this.onChangedObservable) {\n this.onChangedObservable.clear();\n }\n }\n /**\n * Called for each rendered frame.\n */\n checkInputs() {\n this.onChangedObservable.notifyObservers({\n wheelDeltaX: this._wheelDeltaX,\n wheelDeltaY: this._wheelDeltaY,\n wheelDeltaZ: this._wheelDeltaZ,\n });\n // Clear deltas.\n this._wheelDeltaX = 0;\n this._wheelDeltaY = 0;\n this._wheelDeltaZ = 0;\n }\n /**\n * Gets the class name of the current input.\n * @returns the class name\n */\n getClassName() {\n return \"BaseCameraMouseWheelInput\";\n }\n /**\n * Get the friendly name associated with the input class.\n * @returns the input friendly name\n */\n getSimpleName() {\n return \"mousewheel\";\n }\n }\n __decorate$1([\n serialize()\n ], BaseCameraMouseWheelInput.prototype, \"wheelPrecisionX\", void 0);\n __decorate$1([\n serialize()\n ], BaseCameraMouseWheelInput.prototype, \"wheelPrecisionY\", void 0);\n __decorate$1([\n serialize()\n ], BaseCameraMouseWheelInput.prototype, \"wheelPrecisionZ\", void 0);\n\n // eslint-disable-next-line @typescript-eslint/naming-convention\n var _CameraProperty;\n (function (_CameraProperty) {\n _CameraProperty[_CameraProperty[\"MoveRelative\"] = 0] = \"MoveRelative\";\n _CameraProperty[_CameraProperty[\"RotateRelative\"] = 1] = \"RotateRelative\";\n _CameraProperty[_CameraProperty[\"MoveScene\"] = 2] = \"MoveScene\";\n })(_CameraProperty || (_CameraProperty = {}));\n /**\n * Manage the mouse wheel inputs to control a free camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\n */\n class FreeCameraMouseWheelInput extends BaseCameraMouseWheelInput {\n constructor() {\n super(...arguments);\n this._moveRelative = Vector3.Zero();\n this._rotateRelative = Vector3.Zero();\n this._moveScene = Vector3.Zero();\n /**\n * These are set to the desired default behaviour.\n */\n this._wheelXAction = _CameraProperty.MoveRelative;\n this._wheelXActionCoordinate = Coordinate.X;\n this._wheelYAction = _CameraProperty.MoveRelative;\n this._wheelYActionCoordinate = Coordinate.Z;\n this._wheelZAction = null;\n this._wheelZActionCoordinate = null;\n }\n /**\n * Gets the class name of the current input.\n * @returns the class name\n */\n getClassName() {\n return \"FreeCameraMouseWheelInput\";\n }\n /**\n * Set which movement axis (relative to camera's orientation) the mouse\n * wheel's X axis controls.\n * @param axis The axis to be moved. Set null to clear.\n */\n set wheelXMoveRelative(axis) {\n if (axis === null && this._wheelXAction !== _CameraProperty.MoveRelative) {\n // Attempting to clear different _wheelXAction.\n return;\n }\n this._wheelXAction = _CameraProperty.MoveRelative;\n this._wheelXActionCoordinate = axis;\n }\n /**\n * Get the configured movement axis (relative to camera's orientation) the\n * mouse wheel's X axis controls.\n * @returns The configured axis or null if none.\n */\n get wheelXMoveRelative() {\n if (this._wheelXAction !== _CameraProperty.MoveRelative) {\n return null;\n }\n return this._wheelXActionCoordinate;\n }\n /**\n * Set which movement axis (relative to camera's orientation) the mouse\n * wheel's Y axis controls.\n * @param axis The axis to be moved. Set null to clear.\n */\n set wheelYMoveRelative(axis) {\n if (axis === null && this._wheelYAction !== _CameraProperty.MoveRelative) {\n // Attempting to clear different _wheelYAction.\n return;\n }\n this._wheelYAction = _CameraProperty.MoveRelative;\n this._wheelYActionCoordinate = axis;\n }\n /**\n * Get the configured movement axis (relative to camera's orientation) the\n * mouse wheel's Y axis controls.\n * @returns The configured axis or null if none.\n */\n get wheelYMoveRelative() {\n if (this._wheelYAction !== _CameraProperty.MoveRelative) {\n return null;\n }\n return this._wheelYActionCoordinate;\n }\n /**\n * Set which movement axis (relative to camera's orientation) the mouse\n * wheel's Z axis controls.\n * @param axis The axis to be moved. Set null to clear.\n */\n set wheelZMoveRelative(axis) {\n if (axis === null && this._wheelZAction !== _CameraProperty.MoveRelative) {\n // Attempting to clear different _wheelZAction.\n return;\n }\n this._wheelZAction = _CameraProperty.MoveRelative;\n this._wheelZActionCoordinate = axis;\n }\n /**\n * Get the configured movement axis (relative to camera's orientation) the\n * mouse wheel's Z axis controls.\n * @returns The configured axis or null if none.\n */\n get wheelZMoveRelative() {\n if (this._wheelZAction !== _CameraProperty.MoveRelative) {\n return null;\n }\n return this._wheelZActionCoordinate;\n }\n /**\n * Set which rotation axis (relative to camera's orientation) the mouse\n * wheel's X axis controls.\n * @param axis The axis to be moved. Set null to clear.\n */\n set wheelXRotateRelative(axis) {\n if (axis === null && this._wheelXAction !== _CameraProperty.RotateRelative) {\n // Attempting to clear different _wheelXAction.\n return;\n }\n this._wheelXAction = _CameraProperty.RotateRelative;\n this._wheelXActionCoordinate = axis;\n }\n /**\n * Get the configured rotation axis (relative to camera's orientation) the\n * mouse wheel's X axis controls.\n * @returns The configured axis or null if none.\n */\n get wheelXRotateRelative() {\n if (this._wheelXAction !== _CameraProperty.RotateRelative) {\n return null;\n }\n return this._wheelXActionCoordinate;\n }\n /**\n * Set which rotation axis (relative to camera's orientation) the mouse\n * wheel's Y axis controls.\n * @param axis The axis to be moved. Set null to clear.\n */\n set wheelYRotateRelative(axis) {\n if (axis === null && this._wheelYAction !== _CameraProperty.RotateRelative) {\n // Attempting to clear different _wheelYAction.\n return;\n }\n this._wheelYAction = _CameraProperty.RotateRelative;\n this._wheelYActionCoordinate = axis;\n }\n /**\n * Get the configured rotation axis (relative to camera's orientation) the\n * mouse wheel's Y axis controls.\n * @returns The configured axis or null if none.\n */\n get wheelYRotateRelative() {\n if (this._wheelYAction !== _CameraProperty.RotateRelative) {\n return null;\n }\n return this._wheelYActionCoordinate;\n }\n /**\n * Set which rotation axis (relative to camera's orientation) the mouse\n * wheel's Z axis controls.\n * @param axis The axis to be moved. Set null to clear.\n */\n set wheelZRotateRelative(axis) {\n if (axis === null && this._wheelZAction !== _CameraProperty.RotateRelative) {\n // Attempting to clear different _wheelZAction.\n return;\n }\n this._wheelZAction = _CameraProperty.RotateRelative;\n this._wheelZActionCoordinate = axis;\n }\n /**\n * Get the configured rotation axis (relative to camera's orientation) the\n * mouse wheel's Z axis controls.\n * @returns The configured axis or null if none.\n */\n get wheelZRotateRelative() {\n if (this._wheelZAction !== _CameraProperty.RotateRelative) {\n return null;\n }\n return this._wheelZActionCoordinate;\n }\n /**\n * Set which movement axis (relative to the scene) the mouse wheel's X axis\n * controls.\n * @param axis The axis to be moved. Set null to clear.\n */\n set wheelXMoveScene(axis) {\n if (axis === null && this._wheelXAction !== _CameraProperty.MoveScene) {\n // Attempting to clear different _wheelXAction.\n return;\n }\n this._wheelXAction = _CameraProperty.MoveScene;\n this._wheelXActionCoordinate = axis;\n }\n /**\n * Get the configured movement axis (relative to the scene) the mouse wheel's\n * X axis controls.\n * @returns The configured axis or null if none.\n */\n get wheelXMoveScene() {\n if (this._wheelXAction !== _CameraProperty.MoveScene) {\n return null;\n }\n return this._wheelXActionCoordinate;\n }\n /**\n * Set which movement axis (relative to the scene) the mouse wheel's Y axis\n * controls.\n * @param axis The axis to be moved. Set null to clear.\n */\n set wheelYMoveScene(axis) {\n if (axis === null && this._wheelYAction !== _CameraProperty.MoveScene) {\n // Attempting to clear different _wheelYAction.\n return;\n }\n this._wheelYAction = _CameraProperty.MoveScene;\n this._wheelYActionCoordinate = axis;\n }\n /**\n * Get the configured movement axis (relative to the scene) the mouse wheel's\n * Y axis controls.\n * @returns The configured axis or null if none.\n */\n get wheelYMoveScene() {\n if (this._wheelYAction !== _CameraProperty.MoveScene) {\n return null;\n }\n return this._wheelYActionCoordinate;\n }\n /**\n * Set which movement axis (relative to the scene) the mouse wheel's Z axis\n * controls.\n * @param axis The axis to be moved. Set null to clear.\n */\n set wheelZMoveScene(axis) {\n if (axis === null && this._wheelZAction !== _CameraProperty.MoveScene) {\n // Attempting to clear different _wheelZAction.\n return;\n }\n this._wheelZAction = _CameraProperty.MoveScene;\n this._wheelZActionCoordinate = axis;\n }\n /**\n * Get the configured movement axis (relative to the scene) the mouse wheel's\n * Z axis controls.\n * @returns The configured axis or null if none.\n */\n get wheelZMoveScene() {\n if (this._wheelZAction !== _CameraProperty.MoveScene) {\n return null;\n }\n return this._wheelZActionCoordinate;\n }\n /**\n * Called for each rendered frame.\n */\n checkInputs() {\n if (this._wheelDeltaX === 0 && this._wheelDeltaY === 0 && this._wheelDeltaZ == 0) {\n return;\n }\n // Clear the camera properties that we might be updating.\n this._moveRelative.setAll(0);\n this._rotateRelative.setAll(0);\n this._moveScene.setAll(0);\n // Set the camera properties that are to be updated.\n this._updateCamera();\n if (this.camera.getScene().useRightHandedSystem) {\n // TODO: Does this need done for worldUpdate too?\n this._moveRelative.z *= -1;\n }\n // Convert updates relative to camera to world position update.\n const cameraTransformMatrix = Matrix.Zero();\n this.camera.getViewMatrix().invertToRef(cameraTransformMatrix);\n const transformedDirection = Vector3.Zero();\n Vector3.TransformNormalToRef(this._moveRelative, cameraTransformMatrix, transformedDirection);\n // Apply updates to camera position.\n this.camera.cameraRotation.x += this._rotateRelative.x / 200;\n this.camera.cameraRotation.y += this._rotateRelative.y / 200;\n this.camera.cameraDirection.addInPlace(transformedDirection);\n this.camera.cameraDirection.addInPlace(this._moveScene);\n // Call the base class implementation to handle observers and do cleanup.\n super.checkInputs();\n }\n /**\n * Update the camera according to any configured properties for the 3\n * mouse-wheel axis.\n */\n _updateCamera() {\n // Do the camera updates for each of the 3 touch-wheel axis.\n this._updateCameraProperty(this._wheelDeltaX, this._wheelXAction, this._wheelXActionCoordinate);\n this._updateCameraProperty(this._wheelDeltaY, this._wheelYAction, this._wheelYActionCoordinate);\n this._updateCameraProperty(this._wheelDeltaZ, this._wheelZAction, this._wheelZActionCoordinate);\n }\n /**\n * Update one property of the camera.\n * @param value\n * @param cameraProperty\n * @param coordinate\n */\n _updateCameraProperty(\n /* Mouse-wheel delta. */\n value, \n /* Camera property to be changed. */\n cameraProperty, \n /* Axis of Camera property to be changed. */\n coordinate) {\n if (value === 0) {\n // Mouse wheel has not moved.\n return;\n }\n if (cameraProperty === null || coordinate === null) {\n // Mouse wheel axis not configured.\n return;\n }\n let action = null;\n switch (cameraProperty) {\n case _CameraProperty.MoveRelative:\n action = this._moveRelative;\n break;\n case _CameraProperty.RotateRelative:\n action = this._rotateRelative;\n break;\n case _CameraProperty.MoveScene:\n action = this._moveScene;\n break;\n }\n switch (coordinate) {\n case Coordinate.X:\n action.set(value, 0, 0);\n break;\n case Coordinate.Y:\n action.set(0, value, 0);\n break;\n case Coordinate.Z:\n action.set(0, 0, value);\n break;\n }\n }\n }\n __decorate$1([\n serialize()\n ], FreeCameraMouseWheelInput.prototype, \"wheelXMoveRelative\", null);\n __decorate$1([\n serialize()\n ], FreeCameraMouseWheelInput.prototype, \"wheelYMoveRelative\", null);\n __decorate$1([\n serialize()\n ], FreeCameraMouseWheelInput.prototype, \"wheelZMoveRelative\", null);\n __decorate$1([\n serialize()\n ], FreeCameraMouseWheelInput.prototype, \"wheelXRotateRelative\", null);\n __decorate$1([\n serialize()\n ], FreeCameraMouseWheelInput.prototype, \"wheelYRotateRelative\", null);\n __decorate$1([\n serialize()\n ], FreeCameraMouseWheelInput.prototype, \"wheelZRotateRelative\", null);\n __decorate$1([\n serialize()\n ], FreeCameraMouseWheelInput.prototype, \"wheelXMoveScene\", null);\n __decorate$1([\n serialize()\n ], FreeCameraMouseWheelInput.prototype, \"wheelYMoveScene\", null);\n __decorate$1([\n serialize()\n ], FreeCameraMouseWheelInput.prototype, \"wheelZMoveScene\", null);\n CameraInputTypes[\"FreeCameraMouseWheelInput\"] = FreeCameraMouseWheelInput;\n\n /**\n * Manage the touch inputs to control the movement of a free camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\n */\n class FreeCameraTouchInput {\n /**\n * Manage the touch inputs to control the movement of a free camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\n * @param allowMouse Defines if mouse events can be treated as touch events\n */\n constructor(\n /**\n * Define if mouse events can be treated as touch events\n */\n allowMouse = false) {\n this.allowMouse = allowMouse;\n /**\n * Defines the touch sensibility for rotation.\n * The lower the faster.\n */\n this.touchAngularSensibility = 200000.0;\n /**\n * Defines the touch sensibility for move.\n * The lower the faster.\n */\n this.touchMoveSensibility = 250.0;\n /**\n * Swap touch actions so that one touch is used for rotation and multiple for movement\n */\n this.singleFingerRotate = false;\n this._offsetX = null;\n this._offsetY = null;\n this._pointerPressed = new Array();\n this._isSafari = Tools.IsSafari();\n }\n /**\n * Attach the input controls to a specific dom element to get the input from.\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\n */\n attachControl(noPreventDefault) {\n // eslint-disable-next-line prefer-rest-params\n noPreventDefault = Tools.BackCompatCameraNoPreventDefault(arguments);\n let previousPosition = null;\n if (this._pointerInput === undefined) {\n this._onLostFocus = () => {\n this._offsetX = null;\n this._offsetY = null;\n };\n this._pointerInput = (p) => {\n const evt = p.event;\n const isMouseEvent = evt.pointerType === \"mouse\" || (this._isSafari && typeof evt.pointerType === \"undefined\");\n if (!this.allowMouse && isMouseEvent) {\n return;\n }\n if (p.type === PointerEventTypes.POINTERDOWN) {\n if (!noPreventDefault) {\n evt.preventDefault();\n }\n this._pointerPressed.push(evt.pointerId);\n if (this._pointerPressed.length !== 1) {\n return;\n }\n previousPosition = {\n x: evt.clientX,\n y: evt.clientY,\n };\n }\n else if (p.type === PointerEventTypes.POINTERUP) {\n if (!noPreventDefault) {\n evt.preventDefault();\n }\n const index = this._pointerPressed.indexOf(evt.pointerId);\n if (index === -1) {\n return;\n }\n this._pointerPressed.splice(index, 1);\n if (index != 0) {\n return;\n }\n previousPosition = null;\n this._offsetX = null;\n this._offsetY = null;\n }\n else if (p.type === PointerEventTypes.POINTERMOVE) {\n if (!noPreventDefault) {\n evt.preventDefault();\n }\n if (!previousPosition) {\n return;\n }\n const index = this._pointerPressed.indexOf(evt.pointerId);\n if (index != 0) {\n return;\n }\n this._offsetX = evt.clientX - previousPosition.x;\n this._offsetY = -(evt.clientY - previousPosition.y);\n }\n };\n }\n this._observer = this.camera\n .getScene()\n ._inputManager._addCameraPointerObserver(this._pointerInput, PointerEventTypes.POINTERDOWN | PointerEventTypes.POINTERUP | PointerEventTypes.POINTERMOVE);\n if (this._onLostFocus) {\n const engine = this.camera.getEngine();\n const element = engine.getInputElement();\n element && element.addEventListener(\"blur\", this._onLostFocus);\n }\n }\n /**\n * Detach the current controls from the specified dom element.\n */\n detachControl() {\n if (this._pointerInput) {\n if (this._observer) {\n this.camera.getScene()._inputManager._removeCameraPointerObserver(this._observer);\n this._observer = null;\n }\n if (this._onLostFocus) {\n const engine = this.camera.getEngine();\n const element = engine.getInputElement();\n element && element.removeEventListener(\"blur\", this._onLostFocus);\n this._onLostFocus = null;\n }\n this._pointerPressed.length = 0;\n this._offsetX = null;\n this._offsetY = null;\n }\n }\n /**\n * Update the current camera state depending on the inputs that have been used this frame.\n * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop.\n */\n checkInputs() {\n if (this._offsetX === null || this._offsetY === null) {\n return;\n }\n if (this._offsetX === 0 && this._offsetY === 0) {\n return;\n }\n const camera = this.camera;\n camera.cameraRotation.y = this._offsetX / this.touchAngularSensibility;\n const rotateCamera = (this.singleFingerRotate && this._pointerPressed.length === 1) || (!this.singleFingerRotate && this._pointerPressed.length > 1);\n if (rotateCamera) {\n camera.cameraRotation.x = -this._offsetY / this.touchAngularSensibility;\n }\n else {\n const speed = camera._computeLocalCameraSpeed();\n const direction = new Vector3(0, 0, this.touchMoveSensibility !== 0 ? (speed * this._offsetY) / this.touchMoveSensibility : 0);\n Matrix.RotationYawPitchRollToRef(camera.rotation.y, camera.rotation.x, 0, camera._cameraRotationMatrix);\n camera.cameraDirection.addInPlace(Vector3.TransformCoordinates(direction, camera._cameraRotationMatrix));\n }\n }\n /**\n * Gets the class name of the current input.\n * @returns the class name\n */\n getClassName() {\n return \"FreeCameraTouchInput\";\n }\n /**\n * Get the friendly name associated with the input class.\n * @returns the input friendly name\n */\n getSimpleName() {\n return \"touch\";\n }\n }\n __decorate$1([\n serialize()\n ], FreeCameraTouchInput.prototype, \"touchAngularSensibility\", void 0);\n __decorate$1([\n serialize()\n ], FreeCameraTouchInput.prototype, \"touchMoveSensibility\", void 0);\n CameraInputTypes[\"FreeCameraTouchInput\"] = FreeCameraTouchInput;\n\n /**\n * Default Inputs manager for the FreeCamera.\n * It groups all the default supported inputs for ease of use.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\n */\n class FreeCameraInputsManager extends CameraInputsManager {\n /**\n * Instantiates a new FreeCameraInputsManager.\n * @param camera Defines the camera the inputs belong to\n */\n constructor(camera) {\n super(camera);\n /**\n * @internal\n */\n this._mouseInput = null;\n /**\n * @internal\n */\n this._mouseWheelInput = null;\n }\n /**\n * Add keyboard input support to the input manager.\n * @returns the current input manager\n */\n addKeyboard() {\n this.add(new FreeCameraKeyboardMoveInput());\n return this;\n }\n /**\n * Add mouse input support to the input manager.\n * @param touchEnabled if the FreeCameraMouseInput should support touch (default: true)\n * @returns the current input manager\n */\n addMouse(touchEnabled = true) {\n if (!this._mouseInput) {\n this._mouseInput = new FreeCameraMouseInput(touchEnabled);\n this.add(this._mouseInput);\n }\n return this;\n }\n /**\n * Removes the mouse input support from the manager\n * @returns the current input manager\n */\n removeMouse() {\n if (this._mouseInput) {\n this.remove(this._mouseInput);\n }\n return this;\n }\n /**\n * Add mouse wheel input support to the input manager.\n * @returns the current input manager\n */\n addMouseWheel() {\n if (!this._mouseWheelInput) {\n this._mouseWheelInput = new FreeCameraMouseWheelInput();\n this.add(this._mouseWheelInput);\n }\n return this;\n }\n /**\n * Removes the mouse wheel input support from the manager\n * @returns the current input manager\n */\n removeMouseWheel() {\n if (this._mouseWheelInput) {\n this.remove(this._mouseWheelInput);\n }\n return this;\n }\n /**\n * Add touch input support to the input manager.\n * @returns the current input manager\n */\n addTouch() {\n this.add(new FreeCameraTouchInput());\n return this;\n }\n /**\n * Remove all attached input methods from a camera\n */\n clear() {\n super.clear();\n this._mouseInput = null;\n }\n }\n\n /**\n * This represents a free type of camera. It can be useful in First Person Shooter game for instance.\n * Please consider using the new UniversalCamera instead as it adds more functionality like the gamepad.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#universal-camera\n */\n class FreeCamera extends TargetCamera {\n /**\n * Gets the input sensibility for a mouse input. (default is 2000.0)\n * Higher values reduce sensitivity.\n */\n get angularSensibility() {\n const mouse = this.inputs.attached[\"mouse\"];\n if (mouse) {\n return mouse.angularSensibility;\n }\n return 0;\n }\n /**\n * Sets the input sensibility for a mouse input. (default is 2000.0)\n * Higher values reduce sensitivity.\n */\n set angularSensibility(value) {\n const mouse = this.inputs.attached[\"mouse\"];\n if (mouse) {\n mouse.angularSensibility = value;\n }\n }\n /**\n * Gets or Set the list of keyboard keys used to control the forward move of the camera.\n */\n get keysUp() {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n return keyboard.keysUp;\n }\n return [];\n }\n set keysUp(value) {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n keyboard.keysUp = value;\n }\n }\n /**\n * Gets or Set the list of keyboard keys used to control the upward move of the camera.\n */\n get keysUpward() {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n return keyboard.keysUpward;\n }\n return [];\n }\n set keysUpward(value) {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n keyboard.keysUpward = value;\n }\n }\n /**\n * Gets or Set the list of keyboard keys used to control the backward move of the camera.\n */\n get keysDown() {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n return keyboard.keysDown;\n }\n return [];\n }\n set keysDown(value) {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n keyboard.keysDown = value;\n }\n }\n /**\n * Gets or Set the list of keyboard keys used to control the downward move of the camera.\n */\n get keysDownward() {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n return keyboard.keysDownward;\n }\n return [];\n }\n set keysDownward(value) {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n keyboard.keysDownward = value;\n }\n }\n /**\n * Gets or Set the list of keyboard keys used to control the left strafe move of the camera.\n */\n get keysLeft() {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n return keyboard.keysLeft;\n }\n return [];\n }\n set keysLeft(value) {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n keyboard.keysLeft = value;\n }\n }\n /**\n * Gets or Set the list of keyboard keys used to control the right strafe move of the camera.\n */\n get keysRight() {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n return keyboard.keysRight;\n }\n return [];\n }\n set keysRight(value) {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n keyboard.keysRight = value;\n }\n }\n /**\n * Gets or Set the list of keyboard keys used to control the left rotation move of the camera.\n */\n get keysRotateLeft() {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n return keyboard.keysRotateLeft;\n }\n return [];\n }\n set keysRotateLeft(value) {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n keyboard.keysRotateLeft = value;\n }\n }\n /**\n * Gets or Set the list of keyboard keys used to control the right rotation move of the camera.\n */\n get keysRotateRight() {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n return keyboard.keysRotateRight;\n }\n return [];\n }\n set keysRotateRight(value) {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n keyboard.keysRotateRight = value;\n }\n }\n /**\n * Gets or Set the list of keyboard keys used to control the up rotation move of the camera.\n */\n get keysRotateUp() {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n return keyboard.keysRotateUp;\n }\n return [];\n }\n set keysRotateUp(value) {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n keyboard.keysRotateUp = value;\n }\n }\n /**\n * Gets or Set the list of keyboard keys used to control the down rotation move of the camera.\n */\n get keysRotateDown() {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n return keyboard.keysRotateDown;\n }\n return [];\n }\n set keysRotateDown(value) {\n const keyboard = this.inputs.attached[\"keyboard\"];\n if (keyboard) {\n keyboard.keysRotateDown = value;\n }\n }\n /**\n * Instantiates a Free Camera.\n * This represents a free type of camera. It can be useful in First Person Shooter game for instance.\n * Please consider using the new UniversalCamera instead as it adds more functionality like touch to this camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#universal-camera\n * @param name Define the name of the camera in the scene\n * @param position Define the start position of the camera in the scene\n * @param scene Define the scene the camera belongs to\n * @param setActiveOnSceneIfNoneActive Defines whether the camera should be marked as active if not other active cameras have been defined\n */\n constructor(name, position, scene, setActiveOnSceneIfNoneActive = true) {\n super(name, position, scene, setActiveOnSceneIfNoneActive);\n /**\n * Define the collision ellipsoid of the camera.\n * This is helpful to simulate a camera body like the player body around the camera\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions#arcrotatecamera\n */\n this.ellipsoid = new Vector3(0.5, 1, 0.5);\n /**\n * Define an offset for the position of the ellipsoid around the camera.\n * This can be helpful to determine the center of the body near the gravity center of the body\n * instead of its head.\n */\n this.ellipsoidOffset = new Vector3(0, 0, 0);\n /**\n * Enable or disable collisions of the camera with the rest of the scene objects.\n */\n this.checkCollisions = false;\n /**\n * Enable or disable gravity on the camera.\n */\n this.applyGravity = false;\n this._needMoveForGravity = false;\n this._oldPosition = Vector3.Zero();\n this._diffPosition = Vector3.Zero();\n this._newPosition = Vector3.Zero();\n // Collisions\n this._collisionMask = -1;\n this._onCollisionPositionChange = (collisionId, newPosition, collidedMesh = null) => {\n this._newPosition.copyFrom(newPosition);\n this._newPosition.subtractToRef(this._oldPosition, this._diffPosition);\n if (this._diffPosition.length() > Engine.CollisionsEpsilon) {\n this.position.addToRef(this._diffPosition, this._deferredPositionUpdate);\n if (!this._deferOnly) {\n this.position.copyFrom(this._deferredPositionUpdate);\n }\n else {\n this._deferredUpdated = true;\n }\n // call onCollide, if defined. Note that in case of deferred update, the actual position change might happen in the next frame.\n if (this.onCollide && collidedMesh) {\n this.onCollide(collidedMesh);\n }\n }\n };\n this.inputs = new FreeCameraInputsManager(this);\n this.inputs.addKeyboard().addMouse();\n }\n /**\n * Attached controls to the current camera.\n * @param ignored defines an ignored parameter kept for backward compatibility.\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\n */\n attachControl(ignored, noPreventDefault) {\n // eslint-disable-next-line prefer-rest-params\n noPreventDefault = Tools.BackCompatCameraNoPreventDefault(arguments);\n this.inputs.attachElement(noPreventDefault);\n }\n /**\n * Detach the current controls from the specified dom element.\n */\n detachControl() {\n this.inputs.detachElement();\n this.cameraDirection = new Vector3(0, 0, 0);\n this.cameraRotation = new Vector2(0, 0);\n }\n /**\n * Define a collision mask to limit the list of object the camera can collide with\n */\n get collisionMask() {\n return this._collisionMask;\n }\n set collisionMask(mask) {\n this._collisionMask = !isNaN(mask) ? mask : -1;\n }\n /**\n * @internal\n */\n _collideWithWorld(displacement) {\n let globalPosition;\n if (this.parent) {\n globalPosition = Vector3.TransformCoordinates(this.position, this.parent.getWorldMatrix());\n }\n else {\n globalPosition = this.position;\n }\n globalPosition.subtractFromFloatsToRef(0, this.ellipsoid.y, 0, this._oldPosition);\n this._oldPosition.addInPlace(this.ellipsoidOffset);\n const coordinator = this.getScene().collisionCoordinator;\n if (!this._collider) {\n this._collider = coordinator.createCollider();\n }\n this._collider._radius = this.ellipsoid;\n this._collider.collisionMask = this._collisionMask;\n //no need for clone, as long as gravity is not on.\n let actualDisplacement = displacement;\n //add gravity to the direction to prevent the dual-collision checking\n if (this.applyGravity) {\n //this prevents mending with cameraDirection, a global variable of the free camera class.\n actualDisplacement = displacement.add(this.getScene().gravity);\n }\n coordinator.getNewPosition(this._oldPosition, actualDisplacement, this._collider, 3, null, this._onCollisionPositionChange, this.uniqueId);\n }\n /** @internal */\n _checkInputs() {\n if (!this._localDirection) {\n this._localDirection = Vector3.Zero();\n this._transformedDirection = Vector3.Zero();\n }\n this.inputs.checkInputs();\n super._checkInputs();\n }\n /**\n * Enable movement without a user input. This allows gravity to always be applied.\n */\n set needMoveForGravity(value) {\n this._needMoveForGravity = value;\n }\n /**\n * When true, gravity is applied whether there is user input or not.\n */\n get needMoveForGravity() {\n return this._needMoveForGravity;\n }\n /** @internal */\n _decideIfNeedsToMove() {\n return this._needMoveForGravity || Math.abs(this.cameraDirection.x) > 0 || Math.abs(this.cameraDirection.y) > 0 || Math.abs(this.cameraDirection.z) > 0;\n }\n /** @internal */\n _updatePosition() {\n if (this.checkCollisions && this.getScene().collisionsEnabled) {\n this._collideWithWorld(this.cameraDirection);\n }\n else {\n super._updatePosition();\n }\n }\n /**\n * Destroy the camera and release the current resources hold by it.\n */\n dispose() {\n this.inputs.clear();\n super.dispose();\n }\n /**\n * Gets the current object class name.\n * @returns the class name\n */\n getClassName() {\n return \"FreeCamera\";\n }\n }\n __decorate$1([\n serializeAsVector3()\n ], FreeCamera.prototype, \"ellipsoid\", void 0);\n __decorate$1([\n serializeAsVector3()\n ], FreeCamera.prototype, \"ellipsoidOffset\", void 0);\n __decorate$1([\n serialize()\n ], FreeCamera.prototype, \"checkCollisions\", void 0);\n __decorate$1([\n serialize()\n ], FreeCamera.prototype, \"applyGravity\", void 0);\n\n /**\n * Add orientation input support to the input manager.\n * @param smoothFactor deviceOrientation smoothing. 0: no smoothing, 1: new data ignored, 0.9 recommended for smoothing\n * @returns the current input manager\n */\n FreeCameraInputsManager.prototype.addDeviceOrientation = function (smoothFactor) {\n if (!this._deviceOrientationInput) {\n this._deviceOrientationInput = new FreeCameraDeviceOrientationInput();\n if (smoothFactor) {\n this._deviceOrientationInput.smoothFactor = smoothFactor;\n }\n this.add(this._deviceOrientationInput);\n }\n return this;\n };\n /**\n * Takes information about the orientation of the device as reported by the deviceorientation event to orient the camera.\n * Screen rotation is taken into account.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\n */\n class FreeCameraDeviceOrientationInput {\n /**\n * Can be used to detect if a device orientation sensor is available on a device\n * @param timeout amount of time in milliseconds to wait for a response from the sensor (default: infinite)\n * @returns a promise that will resolve on orientation change\n */\n static WaitForOrientationChangeAsync(timeout) {\n return new Promise((res, rej) => {\n let gotValue = false;\n const eventHandler = () => {\n window.removeEventListener(\"deviceorientation\", eventHandler);\n gotValue = true;\n res();\n };\n // If timeout is populated reject the promise\n if (timeout) {\n setTimeout(() => {\n if (!gotValue) {\n window.removeEventListener(\"deviceorientation\", eventHandler);\n rej(\"WaitForOrientationChangeAsync timed out\");\n }\n }, timeout);\n }\n if (typeof DeviceOrientationEvent !== \"undefined\" && typeof DeviceOrientationEvent.requestPermission === \"function\") {\n DeviceOrientationEvent\n .requestPermission()\n .then((response) => {\n if (response == \"granted\") {\n window.addEventListener(\"deviceorientation\", eventHandler);\n }\n else {\n Tools.Warn(\"Permission not granted.\");\n }\n })\n .catch((error) => {\n Tools.Error(error);\n });\n }\n else {\n window.addEventListener(\"deviceorientation\", eventHandler);\n }\n });\n }\n /**\n * Instantiates a new input\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\n */\n constructor() {\n this._screenOrientationAngle = 0;\n this._screenQuaternion = new Quaternion();\n this._alpha = 0;\n this._beta = 0;\n this._gamma = 0;\n /** alpha+beta+gamma smoothing. 0: no smoothing, 1: new data ignored, 0.9 recommended for smoothing */\n this.smoothFactor = 0;\n /**\n * @internal\n */\n this._onDeviceOrientationChangedObservable = new Observable$1();\n this._orientationChanged = () => {\n this._screenOrientationAngle =\n window.orientation !== undefined\n ? +window.orientation\n : window.screen.orientation && window.screen.orientation[\"angle\"]\n ? window.screen.orientation.angle\n : 0;\n this._screenOrientationAngle = -Tools.ToRadians(this._screenOrientationAngle / 2);\n this._screenQuaternion.copyFromFloats(0, Math.sin(this._screenOrientationAngle), 0, Math.cos(this._screenOrientationAngle));\n };\n this._deviceOrientation = (evt) => {\n if (this.smoothFactor) {\n this._alpha = evt.alpha !== null ? Tools.SmoothAngleChange(this._alpha, evt.alpha, this.smoothFactor) : 0;\n this._beta = evt.beta !== null ? Tools.SmoothAngleChange(this._beta, evt.beta, this.smoothFactor) : 0;\n this._gamma = evt.gamma !== null ? Tools.SmoothAngleChange(this._gamma, evt.gamma, this.smoothFactor) : 0;\n }\n else {\n this._alpha = evt.alpha !== null ? evt.alpha : 0;\n this._beta = evt.beta !== null ? evt.beta : 0;\n this._gamma = evt.gamma !== null ? evt.gamma : 0;\n }\n if (evt.alpha !== null) {\n this._onDeviceOrientationChangedObservable.notifyObservers();\n }\n };\n this._constantTranform = new Quaternion(-Math.sqrt(0.5), 0, 0, Math.sqrt(0.5));\n this._orientationChanged();\n }\n /**\n * Define the camera controlled by the input.\n */\n get camera() {\n return this._camera;\n }\n set camera(camera) {\n this._camera = camera;\n if (this._camera != null && !this._camera.rotationQuaternion) {\n this._camera.rotationQuaternion = new Quaternion();\n }\n if (this._camera) {\n this._camera.onDisposeObservable.add(() => {\n this._onDeviceOrientationChangedObservable.clear();\n });\n }\n }\n /**\n * Attach the input controls to a specific dom element to get the input from.\n */\n attachControl() {\n const hostWindow = this.camera.getScene().getEngine().getHostWindow();\n if (hostWindow) {\n const eventHandler = () => {\n hostWindow.addEventListener(\"orientationchange\", this._orientationChanged);\n hostWindow.addEventListener(\"deviceorientation\", this._deviceOrientation);\n //In certain cases, the attach control is called AFTER orientation was changed,\n //So this is needed.\n this._orientationChanged();\n };\n if (typeof DeviceOrientationEvent !== \"undefined\" && typeof DeviceOrientationEvent.requestPermission === \"function\") {\n DeviceOrientationEvent\n .requestPermission()\n .then((response) => {\n if (response === \"granted\") {\n eventHandler();\n }\n else {\n Tools.Warn(\"Permission not granted.\");\n }\n })\n .catch((error) => {\n Tools.Error(error);\n });\n }\n else {\n eventHandler();\n }\n }\n }\n /**\n * Detach the current controls from the specified dom element.\n */\n detachControl() {\n window.removeEventListener(\"orientationchange\", this._orientationChanged);\n window.removeEventListener(\"deviceorientation\", this._deviceOrientation);\n this._alpha = 0;\n }\n /**\n * Update the current camera state depending on the inputs that have been used this frame.\n * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop.\n */\n checkInputs() {\n //if no device orientation provided, don't update the rotation.\n //Only testing against alpha under the assumption thatnorientation will never be so exact when set.\n if (!this._alpha) {\n return;\n }\n Quaternion.RotationYawPitchRollToRef(Tools.ToRadians(this._alpha), Tools.ToRadians(this._beta), -Tools.ToRadians(this._gamma), this.camera.rotationQuaternion);\n this._camera.rotationQuaternion.multiplyInPlace(this._screenQuaternion);\n this._camera.rotationQuaternion.multiplyInPlace(this._constantTranform);\n //Mirror on XY Plane\n this._camera.rotationQuaternion.z *= -1;\n this._camera.rotationQuaternion.w *= -1;\n }\n /**\n * Gets the class name of the current input.\n * @returns the class name\n */\n getClassName() {\n return \"FreeCameraDeviceOrientationInput\";\n }\n /**\n * Get the friendly name associated with the input class.\n * @returns the input friendly name\n */\n getSimpleName() {\n return \"deviceOrientation\";\n }\n }\n CameraInputTypes[\"FreeCameraDeviceOrientationInput\"] = FreeCameraDeviceOrientationInput;\n\n Node.AddNodeConstructor(\"DeviceOrientationCamera\", (name, scene) => {\n return () => new DeviceOrientationCamera(name, Vector3.Zero(), scene);\n });\n // We're mainly based on the logic defined into the FreeCamera code\n /**\n * This is a camera specifically designed to react to device orientation events such as a modern mobile device\n * being tilted forward or back and left or right.\n */\n class DeviceOrientationCamera extends FreeCamera {\n /**\n * Creates a new device orientation camera\n * @param name The name of the camera\n * @param position The start position camera\n * @param scene The scene the camera belongs to\n */\n constructor(name, position, scene) {\n super(name, position, scene);\n this._tmpDragQuaternion = new Quaternion();\n this._disablePointerInputWhenUsingDeviceOrientation = true;\n this._dragFactor = 0;\n this._quaternionCache = new Quaternion();\n this.inputs.addDeviceOrientation();\n // When the orientation sensor fires it's first event, disable mouse input\n if (this.inputs._deviceOrientationInput) {\n this.inputs._deviceOrientationInput._onDeviceOrientationChangedObservable.addOnce(() => {\n if (this._disablePointerInputWhenUsingDeviceOrientation) {\n if (this.inputs._mouseInput) {\n this.inputs._mouseInput._allowCameraRotation = false;\n this.inputs._mouseInput.onPointerMovedObservable.add((e) => {\n if (this._dragFactor != 0) {\n if (!this._initialQuaternion) {\n this._initialQuaternion = new Quaternion();\n }\n // Rotate the initial space around the y axis to allow users to \"turn around\" via touch/mouse\n Quaternion.FromEulerAnglesToRef(0, e.offsetX * this._dragFactor, 0, this._tmpDragQuaternion);\n this._initialQuaternion.multiplyToRef(this._tmpDragQuaternion, this._initialQuaternion);\n }\n });\n }\n }\n });\n }\n }\n /**\n * Gets or sets a boolean indicating that pointer input must be disabled on first orientation sensor update (Default: true)\n */\n get disablePointerInputWhenUsingDeviceOrientation() {\n return this._disablePointerInputWhenUsingDeviceOrientation;\n }\n set disablePointerInputWhenUsingDeviceOrientation(value) {\n this._disablePointerInputWhenUsingDeviceOrientation = value;\n }\n /**\n * Enabled turning on the y axis when the orientation sensor is active\n * @param dragFactor the factor that controls the turn speed (default: 1/300)\n */\n enableHorizontalDragging(dragFactor = 1 / 300) {\n this._dragFactor = dragFactor;\n }\n /**\n * Gets the current instance class name (\"DeviceOrientationCamera\").\n * This helps avoiding instanceof at run time.\n * @returns the class name\n */\n getClassName() {\n return \"DeviceOrientationCamera\";\n }\n /**\n * @internal\n * Checks and applies the current values of the inputs to the camera. (Internal use only)\n */\n _checkInputs() {\n super._checkInputs();\n this._quaternionCache.copyFrom(this.rotationQuaternion);\n if (this._initialQuaternion) {\n this._initialQuaternion.multiplyToRef(this.rotationQuaternion, this.rotationQuaternion);\n }\n }\n /**\n * Reset the camera to its default orientation on the specified axis only.\n * @param axis The axis to reset\n */\n resetToCurrentRotation(axis = Axis.Y) {\n //can only work if this camera has a rotation quaternion already.\n if (!this.rotationQuaternion) {\n return;\n }\n if (!this._initialQuaternion) {\n this._initialQuaternion = new Quaternion();\n }\n this._initialQuaternion.copyFrom(this._quaternionCache || this.rotationQuaternion);\n [\"x\", \"y\", \"z\"].forEach((axisName) => {\n if (!axis[axisName]) {\n this._initialQuaternion[axisName] = 0;\n }\n else {\n this._initialQuaternion[axisName] *= -1;\n }\n });\n this._initialQuaternion.normalize();\n //force rotation update\n this._initialQuaternion.multiplyToRef(this.rotationQuaternion, this.rotationQuaternion);\n }\n }\n\n /**\n * This represents all the required metrics to create a VR camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#device-orientation-camera\n */\n class VRCameraMetrics {\n constructor() {\n /**\n * Define if the current vr camera should compensate the distortion of the lens or not.\n */\n this.compensateDistortion = true;\n /**\n * Defines if multiview should be enabled when rendering (Default: false)\n */\n this.multiviewEnabled = false;\n }\n /**\n * Gets the rendering aspect ratio based on the provided resolutions.\n */\n get aspectRatio() {\n return this.hResolution / (2 * this.vResolution);\n }\n /**\n * Gets the aspect ratio based on the FOV, scale factors, and real screen sizes.\n */\n get aspectRatioFov() {\n return 2 * Math.atan((this.postProcessScaleFactor * this.vScreenSize) / (2 * this.eyeToScreenDistance));\n }\n /**\n * @internal\n */\n get leftHMatrix() {\n const meters = this.hScreenSize / 4 - this.lensSeparationDistance / 2;\n const h = (4 * meters) / this.hScreenSize;\n return Matrix.Translation(h, 0, 0);\n }\n /**\n * @internal\n */\n get rightHMatrix() {\n const meters = this.hScreenSize / 4 - this.lensSeparationDistance / 2;\n const h = (4 * meters) / this.hScreenSize;\n return Matrix.Translation(-h, 0, 0);\n }\n /**\n * @internal\n */\n get leftPreViewMatrix() {\n return Matrix.Translation(0.5 * this.interpupillaryDistance, 0, 0);\n }\n /**\n * @internal\n */\n get rightPreViewMatrix() {\n return Matrix.Translation(-0.5 * this.interpupillaryDistance, 0, 0);\n }\n /**\n * Get the default VRMetrics based on the most generic setup.\n * @returns the default vr metrics\n */\n static GetDefault() {\n const result = new VRCameraMetrics();\n result.hResolution = 1280;\n result.vResolution = 800;\n result.hScreenSize = 0.149759993;\n result.vScreenSize = 0.0935999975;\n result.vScreenCenter = 0.0467999987;\n result.eyeToScreenDistance = 0.0410000011;\n result.lensSeparationDistance = 0.063500002;\n result.interpupillaryDistance = 0.064000003;\n result.distortionK = [1.0, 0.219999999, 0.239999995, 0.0];\n result.chromaAbCorrection = [0.995999992, -0.00400000019, 1.01400006, 0.0];\n result.postProcessScaleFactor = 1.714605507808412;\n result.lensCenterOffset = 0.151976421;\n return result;\n }\n }\n\n // Do not edit.\n const name$1T = \"vrDistortionCorrectionPixelShader\";\n const shader$1T = `varying vec2 vUV;uniform sampler2D textureSampler;uniform vec2 LensCenter;uniform vec2 Scale;uniform vec2 ScaleIn;uniform vec4 HmdWarpParam;vec2 HmdWarp(vec2 in01) {vec2 theta=(in01-LensCenter)*ScaleIn; \nfloat rSq=theta.x*theta.x+theta.y*theta.y;vec2 rvector=theta*(HmdWarpParam.x+HmdWarpParam.y*rSq+HmdWarpParam.z*rSq*rSq+HmdWarpParam.w*rSq*rSq*rSq);return LensCenter+Scale*rvector;}\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{vec2 tc=HmdWarp(vUV);if (tc.x <0.0 || tc.x>1.0 || tc.y<0.0 || tc.y>1.0)\ngl_FragColor=vec4(0.0,0.0,0.0,0.0);else{gl_FragColor=texture2D(textureSampler,tc);}}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$1T] = shader$1T;\n\n /**\n * VRDistortionCorrectionPostProcess used for mobile VR\n */\n class VRDistortionCorrectionPostProcess extends PostProcess {\n /**\n * Gets a string identifying the name of the class\n * @returns \"VRDistortionCorrectionPostProcess\" string\n */\n getClassName() {\n return \"VRDistortionCorrectionPostProcess\";\n }\n /**\n * Initializes the VRDistortionCorrectionPostProcess\n * @param name The name of the effect.\n * @param camera The camera to apply the render pass to.\n * @param isRightEye If this is for the right eye distortion\n * @param vrMetrics All the required metrics for the VR camera\n */\n constructor(name, camera, isRightEye, vrMetrics) {\n super(name, \"vrDistortionCorrection\", [\"LensCenter\", \"Scale\", \"ScaleIn\", \"HmdWarpParam\"], null, vrMetrics.postProcessScaleFactor, camera, Texture.BILINEAR_SAMPLINGMODE);\n this._isRightEye = isRightEye;\n this._distortionFactors = vrMetrics.distortionK;\n this._postProcessScaleFactor = vrMetrics.postProcessScaleFactor;\n this._lensCenterOffset = vrMetrics.lensCenterOffset;\n this.adaptScaleToCurrentViewport = true;\n this.onSizeChangedObservable.add(() => {\n this._scaleIn = new Vector2(2, 2 / this.aspectRatio);\n this._scaleFactor = new Vector2(0.5 * (1 / this._postProcessScaleFactor), 0.5 * (1 / this._postProcessScaleFactor) * this.aspectRatio);\n this._lensCenter = new Vector2(this._isRightEye ? 0.5 - this._lensCenterOffset * 0.5 : 0.5 + this._lensCenterOffset * 0.5, 0.5);\n });\n this.onApplyObservable.add((effect) => {\n effect.setFloat2(\"LensCenter\", this._lensCenter.x, this._lensCenter.y);\n effect.setFloat2(\"Scale\", this._scaleFactor.x, this._scaleFactor.y);\n effect.setFloat2(\"ScaleIn\", this._scaleIn.x, this._scaleIn.y);\n effect.setFloat4(\"HmdWarpParam\", this._distortionFactors[0], this._distortionFactors[1], this._distortionFactors[2], this._distortionFactors[3]);\n });\n }\n }\n\n // Do not edit.\n const name$1U = \"vrMultiviewToSingleviewPixelShader\";\n const shader$1U = `precision mediump sampler2DArray;varying vec2 vUV;uniform sampler2DArray multiviewSampler;uniform int imageIndex;\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{gl_FragColor=texture2D(multiviewSampler,vec3(vUV,imageIndex));}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$1U] = shader$1U;\n\n /**\n * Renders to multiple views with a single draw call\n * @see https://www.khronos.org/registry/webgl/extensions/OVR_multiview2/\n */\n class MultiviewRenderTarget extends RenderTargetTexture {\n set samples(value) {\n // We override this setter because multisampling is handled by framebufferTextureMultisampleMultiviewOVR\n this._samples = value;\n }\n get samples() {\n return this._samples;\n }\n /**\n * Creates a multiview render target\n * @param scene scene used with the render target\n * @param size the size of the render target (used for each view)\n */\n constructor(scene, size = 512) {\n super(\"multiview rtt\", size, scene, false, true, 0, false, undefined, false, false, true, undefined, true);\n this._renderTarget = this.getScene().getEngine().createMultiviewRenderTargetTexture(this.getRenderWidth(), this.getRenderHeight());\n this._texture = this._renderTarget.texture;\n this._texture.isMultiview = true;\n this._texture.format = 5;\n this.samples = this._getEngine().getCaps().maxSamples || this.samples;\n this._texture.samples = this._samples;\n }\n /**\n * @internal\n */\n _bindFrameBuffer() {\n if (!this._renderTarget) {\n return;\n }\n this.getScene().getEngine().bindMultiviewFramebuffer(this._renderTarget);\n }\n /**\n * Gets the number of views the corresponding to the texture (eg. a MultiviewRenderTarget will have > 1)\n * @returns the view count\n */\n getViewCount() {\n return 2;\n }\n }\n\n Engine.prototype.createMultiviewRenderTargetTexture = function (width, height, colorTexture, depthStencilTexture) {\n const gl = this._gl;\n if (!this.getCaps().multiview) {\n throw \"Multiview is not supported\";\n }\n const rtWrapper = this._createHardwareRenderTargetWrapper(false, false, { width, height });\n rtWrapper._framebuffer = gl.createFramebuffer();\n const internalTexture = new InternalTexture(this, InternalTextureSource.Unknown, true);\n internalTexture.width = width;\n internalTexture.height = height;\n internalTexture.isMultiview = true;\n if (!colorTexture) {\n colorTexture = gl.createTexture();\n gl.bindTexture(gl.TEXTURE_2D_ARRAY, colorTexture);\n gl.texStorage3D(gl.TEXTURE_2D_ARRAY, 1, gl.RGBA8, width, height, 2);\n }\n rtWrapper._colorTextureArray = colorTexture;\n if (!depthStencilTexture) {\n depthStencilTexture = gl.createTexture();\n gl.bindTexture(gl.TEXTURE_2D_ARRAY, depthStencilTexture);\n gl.texStorage3D(gl.TEXTURE_2D_ARRAY, 1, gl.DEPTH24_STENCIL8, width, height, 2);\n }\n rtWrapper._depthStencilTextureArray = depthStencilTexture;\n internalTexture.isReady = true;\n rtWrapper.setTextures(internalTexture);\n rtWrapper._depthStencilTexture = internalTexture;\n return rtWrapper;\n };\n Engine.prototype.bindMultiviewFramebuffer = function (_multiviewTexture) {\n const multiviewTexture = _multiviewTexture;\n const gl = this._gl;\n const ext = this.getCaps().oculusMultiview || this.getCaps().multiview;\n this.bindFramebuffer(multiviewTexture, undefined, undefined, undefined, true);\n gl.bindFramebuffer(gl.DRAW_FRAMEBUFFER, multiviewTexture._framebuffer);\n if (multiviewTexture._colorTextureArray && multiviewTexture._depthStencilTextureArray) {\n if (this.getCaps().oculusMultiview) {\n ext.framebufferTextureMultisampleMultiviewOVR(gl.DRAW_FRAMEBUFFER, gl.COLOR_ATTACHMENT0, multiviewTexture._colorTextureArray, 0, multiviewTexture.samples, 0, 2);\n ext.framebufferTextureMultisampleMultiviewOVR(gl.DRAW_FRAMEBUFFER, gl.DEPTH_STENCIL_ATTACHMENT, multiviewTexture._depthStencilTextureArray, 0, multiviewTexture.samples, 0, 2);\n }\n else {\n ext.framebufferTextureMultiviewOVR(gl.DRAW_FRAMEBUFFER, gl.COLOR_ATTACHMENT0, multiviewTexture._colorTextureArray, 0, 0, 2);\n ext.framebufferTextureMultiviewOVR(gl.DRAW_FRAMEBUFFER, gl.DEPTH_STENCIL_ATTACHMENT, multiviewTexture._depthStencilTextureArray, 0, 0, 2);\n }\n }\n else {\n throw \"Invalid multiview frame buffer\";\n }\n };\n Engine.prototype.bindSpaceWarpFramebuffer = function (_spaceWarpTexture) {\n const spaceWarpTexture = _spaceWarpTexture;\n const gl = this._gl;\n const ext = this.getCaps().oculusMultiview || this.getCaps().multiview;\n this.bindFramebuffer(spaceWarpTexture, undefined, undefined, undefined, true);\n gl.bindFramebuffer(gl.DRAW_FRAMEBUFFER, spaceWarpTexture._framebuffer);\n if (spaceWarpTexture._colorTextureArray && spaceWarpTexture._depthStencilTextureArray) {\n ext.framebufferTextureMultiviewOVR(gl.DRAW_FRAMEBUFFER, gl.COLOR_ATTACHMENT0, spaceWarpTexture._colorTextureArray, 0, 0, 2);\n ext.framebufferTextureMultiviewOVR(gl.DRAW_FRAMEBUFFER, gl.DEPTH_ATTACHMENT, spaceWarpTexture._depthStencilTextureArray, 0, 0, 2);\n }\n else {\n throw new Error(\"Invalid Space Warp framebuffer\");\n }\n };\n Camera.prototype._useMultiviewToSingleView = false;\n Camera.prototype._multiviewTexture = null;\n Camera.prototype._resizeOrCreateMultiviewTexture = function (width, height) {\n if (!this._multiviewTexture) {\n this._multiviewTexture = new MultiviewRenderTarget(this.getScene(), { width: width, height: height });\n }\n else if (this._multiviewTexture.getRenderWidth() != width || this._multiviewTexture.getRenderHeight() != height) {\n this._multiviewTexture.dispose();\n this._multiviewTexture = new MultiviewRenderTarget(this.getScene(), { width: width, height: height });\n }\n };\n function createMultiviewUbo(engine, name) {\n const ubo = new UniformBuffer(engine, undefined, true, name);\n ubo.addUniform(\"viewProjection\", 16);\n ubo.addUniform(\"viewProjectionR\", 16);\n ubo.addUniform(\"view\", 16);\n ubo.addUniform(\"projection\", 16);\n ubo.addUniform(\"vEyePosition\", 4);\n return ubo;\n }\n const currentCreateSceneUniformBuffer = Scene.prototype.createSceneUniformBuffer;\n Scene.prototype._transformMatrixR = Matrix.Zero();\n Scene.prototype._multiviewSceneUbo = null;\n Scene.prototype._createMultiviewUbo = function () {\n this._multiviewSceneUbo = createMultiviewUbo(this.getEngine(), \"scene_multiview\");\n };\n Scene.prototype.createSceneUniformBuffer = function (name) {\n if (this._multiviewSceneUbo) {\n return createMultiviewUbo(this.getEngine(), name);\n }\n return currentCreateSceneUniformBuffer.bind(this)(name);\n };\n Scene.prototype._updateMultiviewUbo = function (viewR, projectionR) {\n if (viewR && projectionR) {\n viewR.multiplyToRef(projectionR, this._transformMatrixR);\n }\n if (viewR && projectionR) {\n viewR.multiplyToRef(projectionR, TmpVectors.Matrix[0]);\n Frustum.GetRightPlaneToRef(TmpVectors.Matrix[0], this._frustumPlanes[3]); // Replace right plane by second camera right plane\n }\n if (this._multiviewSceneUbo) {\n this._multiviewSceneUbo.updateMatrix(\"viewProjection\", this.getTransformMatrix());\n this._multiviewSceneUbo.updateMatrix(\"viewProjectionR\", this._transformMatrixR);\n this._multiviewSceneUbo.updateMatrix(\"view\", this._viewMatrix);\n this._multiviewSceneUbo.updateMatrix(\"projection\", this._projectionMatrix);\n }\n };\n Scene.prototype._renderMultiviewToSingleView = function (camera) {\n // Multiview is only able to be displayed directly for API's such as webXR\n // This displays a multiview image by rendering to the multiview image and then\n // copying the result into the sub cameras instead of rendering them and proceeding as normal from there\n // Render to a multiview texture\n camera._resizeOrCreateMultiviewTexture(camera._rigPostProcess && camera._rigPostProcess && camera._rigPostProcess.width > 0 ? camera._rigPostProcess.width : this.getEngine().getRenderWidth(true), camera._rigPostProcess && camera._rigPostProcess && camera._rigPostProcess.height > 0 ? camera._rigPostProcess.height : this.getEngine().getRenderHeight(true));\n if (!this._multiviewSceneUbo) {\n this._createMultiviewUbo();\n }\n camera.outputRenderTarget = camera._multiviewTexture;\n this._renderForCamera(camera);\n camera.outputRenderTarget = null;\n // Consume the multiview texture through a shader for each eye\n for (let index = 0; index < camera._rigCameras.length; index++) {\n const engine = this.getEngine();\n this._activeCamera = camera._rigCameras[index];\n engine.setViewport(this._activeCamera.viewport);\n if (this.postProcessManager) {\n this.postProcessManager._prepareFrame();\n this.postProcessManager._finalizeFrame(this._activeCamera.isIntermediate);\n }\n }\n };\n\n /**\n * VRMultiviewToSingleview used to convert multiview texture arrays to standard textures for scenarios such as webVR\n * This will not be used for webXR as it supports displaying texture arrays directly\n */\n class VRMultiviewToSingleviewPostProcess extends PostProcess {\n /**\n * Gets a string identifying the name of the class\n * @returns \"VRMultiviewToSingleviewPostProcess\" string\n */\n getClassName() {\n return \"VRMultiviewToSingleviewPostProcess\";\n }\n /**\n * Initializes a VRMultiviewToSingleview\n * @param name name of the post process\n * @param camera camera to be applied to\n * @param scaleFactor scaling factor to the size of the output texture\n */\n constructor(name, camera, scaleFactor) {\n super(name, \"vrMultiviewToSingleview\", [\"imageIndex\"], [\"multiviewSampler\"], scaleFactor, camera, Texture.BILINEAR_SAMPLINGMODE);\n const cam = camera !== null && camera !== void 0 ? camera : this.getCamera();\n this.onSizeChangedObservable.add(() => { });\n this.onApplyObservable.add((effect) => {\n if (cam._scene.activeCamera && cam._scene.activeCamera.isLeftCamera) {\n effect.setInt(\"imageIndex\", 0);\n }\n else {\n effect.setInt(\"imageIndex\", 1);\n }\n effect.setTexture(\"multiviewSampler\", cam._multiviewTexture);\n });\n }\n }\n\n /**\n * @internal\n */\n function setVRRigMode(camera, rigParams) {\n const metrics = rigParams.vrCameraMetrics || VRCameraMetrics.GetDefault();\n camera._rigCameras[0]._cameraRigParams.vrMetrics = metrics;\n camera._rigCameras[0].viewport = new Viewport(0, 0, 0.5, 1.0);\n camera._rigCameras[0]._cameraRigParams.vrWorkMatrix = new Matrix();\n camera._rigCameras[0]._cameraRigParams.vrHMatrix = metrics.leftHMatrix;\n camera._rigCameras[0]._cameraRigParams.vrPreViewMatrix = metrics.leftPreViewMatrix;\n camera._rigCameras[0].getProjectionMatrix = camera._rigCameras[0]._getVRProjectionMatrix;\n camera._rigCameras[1]._cameraRigParams.vrMetrics = metrics;\n camera._rigCameras[1].viewport = new Viewport(0.5, 0, 0.5, 1.0);\n camera._rigCameras[1]._cameraRigParams.vrWorkMatrix = new Matrix();\n camera._rigCameras[1]._cameraRigParams.vrHMatrix = metrics.rightHMatrix;\n camera._rigCameras[1]._cameraRigParams.vrPreViewMatrix = metrics.rightPreViewMatrix;\n camera._rigCameras[1].getProjectionMatrix = camera._rigCameras[1]._getVRProjectionMatrix;\n // For multiview on a webVR camera\n // First multiview will be rendered to camera._multiviewTexture\n // Then this postprocess will run on each eye to copy the right texture to each eye\n if (metrics.multiviewEnabled) {\n if (!camera.getScene().getEngine().getCaps().multiview) {\n Logger.Warn(\"Multiview is not supported, falling back to standard rendering\");\n metrics.multiviewEnabled = false;\n }\n else {\n camera._useMultiviewToSingleView = true;\n camera._rigPostProcess = new VRMultiviewToSingleviewPostProcess(\"VRMultiviewToSingleview\", camera, metrics.postProcessScaleFactor);\n }\n }\n if (metrics.compensateDistortion) {\n camera._rigCameras[0]._rigPostProcess = new VRDistortionCorrectionPostProcess(\"VR_Distort_Compensation_Left\", camera._rigCameras[0], false, metrics);\n camera._rigCameras[1]._rigPostProcess = new VRDistortionCorrectionPostProcess(\"VR_Distort_Compensation_Right\", camera._rigCameras[1], true, metrics);\n }\n }\n\n Node.AddNodeConstructor(\"VRDeviceOrientationFreeCamera\", (name, scene) => {\n return () => new VRDeviceOrientationFreeCamera(name, Vector3.Zero(), scene);\n });\n /**\n * Camera used to simulate VR rendering (based on FreeCamera)\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#vr-device-orientation-cameras\n */\n class VRDeviceOrientationFreeCamera extends DeviceOrientationCamera {\n /**\n * Creates a new VRDeviceOrientationFreeCamera\n * @param name defines camera name\n * @param position defines the start position of the camera\n * @param scene defines the scene the camera belongs to\n * @param compensateDistortion defines if the camera needs to compensate the lens distortion\n * @param vrCameraMetrics defines the vr metrics associated to the camera\n */\n constructor(name, position, scene, compensateDistortion = true, vrCameraMetrics = VRCameraMetrics.GetDefault()) {\n super(name, position, scene);\n this._setRigMode = setVRRigMode.bind(null, this);\n vrCameraMetrics.compensateDistortion = compensateDistortion;\n this.setCameraRigMode(Camera.RIG_MODE_VR, { vrCameraMetrics: vrCameraMetrics });\n }\n /**\n * Gets camera class name\n * @returns VRDeviceOrientationFreeCamera\n */\n getClassName() {\n return \"VRDeviceOrientationFreeCamera\";\n }\n }\n\n /**\n * Represents a gamepad\n */\n class Gamepad {\n /**\n * Specifies if the gamepad has been connected\n */\n get isConnected() {\n return this._isConnected;\n }\n /**\n * Initializes the gamepad\n * @param id The id of the gamepad\n * @param index The index of the gamepad\n * @param browserGamepad The browser gamepad\n * @param leftStickX The x component of the left joystick\n * @param leftStickY The y component of the left joystick\n * @param rightStickX The x component of the right joystick\n * @param rightStickY The y component of the right joystick\n */\n constructor(\n /**\n * The id of the gamepad\n */\n id, \n /**\n * The index of the gamepad\n */\n index, \n /**\n * The browser gamepad\n */\n browserGamepad, leftStickX = 0, leftStickY = 1, rightStickX = 2, rightStickY = 3) {\n this.id = id;\n this.index = index;\n this.browserGamepad = browserGamepad;\n this._leftStick = { x: 0, y: 0 };\n this._rightStick = { x: 0, y: 0 };\n /** @internal */\n this._isConnected = true;\n /**\n * Specifies whether the left control stick should be Y-inverted\n */\n this._invertLeftStickY = false;\n this.type = Gamepad.GAMEPAD;\n this._leftStickAxisX = leftStickX;\n this._leftStickAxisY = leftStickY;\n this._rightStickAxisX = rightStickX;\n this._rightStickAxisY = rightStickY;\n if (this.browserGamepad.axes.length >= 2) {\n this._leftStick = { x: this.browserGamepad.axes[this._leftStickAxisX], y: this.browserGamepad.axes[this._leftStickAxisY] };\n }\n if (this.browserGamepad.axes.length >= 4) {\n this._rightStick = { x: this.browserGamepad.axes[this._rightStickAxisX], y: this.browserGamepad.axes[this._rightStickAxisY] };\n }\n }\n /**\n * Callback triggered when the left joystick has changed\n * @param callback\n */\n onleftstickchanged(callback) {\n this._onleftstickchanged = callback;\n }\n /**\n * Callback triggered when the right joystick has changed\n * @param callback\n */\n onrightstickchanged(callback) {\n this._onrightstickchanged = callback;\n }\n /**\n * Gets the left joystick\n */\n get leftStick() {\n return this._leftStick;\n }\n /**\n * Sets the left joystick values\n */\n set leftStick(newValues) {\n if (this._onleftstickchanged && (this._leftStick.x !== newValues.x || this._leftStick.y !== newValues.y)) {\n this._onleftstickchanged(newValues);\n }\n this._leftStick = newValues;\n }\n /**\n * Gets the right joystick\n */\n get rightStick() {\n return this._rightStick;\n }\n /**\n * Sets the right joystick value\n */\n set rightStick(newValues) {\n if (this._onrightstickchanged && (this._rightStick.x !== newValues.x || this._rightStick.y !== newValues.y)) {\n this._onrightstickchanged(newValues);\n }\n this._rightStick = newValues;\n }\n /**\n * Updates the gamepad joystick positions\n */\n update() {\n if (this._leftStick) {\n this.leftStick = { x: this.browserGamepad.axes[this._leftStickAxisX], y: this.browserGamepad.axes[this._leftStickAxisY] };\n if (this._invertLeftStickY) {\n this.leftStick.y *= -1;\n }\n }\n if (this._rightStick) {\n this.rightStick = { x: this.browserGamepad.axes[this._rightStickAxisX], y: this.browserGamepad.axes[this._rightStickAxisY] };\n }\n }\n /**\n * Disposes the gamepad\n */\n dispose() { }\n }\n /**\n * Represents a gamepad controller\n */\n Gamepad.GAMEPAD = 0;\n /**\n * Represents a generic controller\n */\n Gamepad.GENERIC = 1;\n /**\n * Represents an XBox controller\n */\n Gamepad.XBOX = 2;\n /**\n * Represents a pose-enabled controller\n */\n Gamepad.POSE_ENABLED = 3;\n /**\n * Represents an Dual Shock controller\n */\n Gamepad.DUALSHOCK = 4;\n /**\n * Represents a generic gamepad\n */\n class GenericPad extends Gamepad {\n /**\n * Callback triggered when a button has been pressed\n * @param callback Called when a button has been pressed\n */\n onbuttondown(callback) {\n this._onbuttondown = callback;\n }\n /**\n * Callback triggered when a button has been released\n * @param callback Called when a button has been released\n */\n onbuttonup(callback) {\n this._onbuttonup = callback;\n }\n /**\n * Initializes the generic gamepad\n * @param id The id of the generic gamepad\n * @param index The index of the generic gamepad\n * @param browserGamepad The browser gamepad\n */\n constructor(id, index, browserGamepad) {\n super(id, index, browserGamepad);\n /**\n * Observable triggered when a button has been pressed\n */\n this.onButtonDownObservable = new Observable$1();\n /**\n * Observable triggered when a button has been released\n */\n this.onButtonUpObservable = new Observable$1();\n this.type = Gamepad.GENERIC;\n this._buttons = new Array(browserGamepad.buttons.length);\n }\n _setButtonValue(newValue, currentValue, buttonIndex) {\n if (newValue !== currentValue) {\n if (newValue === 1) {\n if (this._onbuttondown) {\n this._onbuttondown(buttonIndex);\n }\n this.onButtonDownObservable.notifyObservers(buttonIndex);\n }\n if (newValue === 0) {\n if (this._onbuttonup) {\n this._onbuttonup(buttonIndex);\n }\n this.onButtonUpObservable.notifyObservers(buttonIndex);\n }\n }\n return newValue;\n }\n /**\n * Updates the generic gamepad\n */\n update() {\n super.update();\n for (let index = 0; index < this._buttons.length; index++) {\n this._buttons[index] = this._setButtonValue(this.browserGamepad.buttons[index].value, this._buttons[index], index);\n }\n }\n /**\n * Disposes the generic gamepad\n */\n dispose() {\n super.dispose();\n this.onButtonDownObservable.clear();\n this.onButtonUpObservable.clear();\n }\n }\n\n /**\n * Defines the types of pose enabled controllers that are supported\n */\n var PoseEnabledControllerType;\n (function (PoseEnabledControllerType) {\n /**\n * HTC Vive\n */\n PoseEnabledControllerType[PoseEnabledControllerType[\"VIVE\"] = 0] = \"VIVE\";\n /**\n * Oculus Rift\n */\n PoseEnabledControllerType[PoseEnabledControllerType[\"OCULUS\"] = 1] = \"OCULUS\";\n /**\n * Windows mixed reality\n */\n PoseEnabledControllerType[PoseEnabledControllerType[\"WINDOWS\"] = 2] = \"WINDOWS\";\n /**\n * Samsung gear VR\n */\n PoseEnabledControllerType[PoseEnabledControllerType[\"GEAR_VR\"] = 3] = \"GEAR_VR\";\n /**\n * Google Daydream\n */\n PoseEnabledControllerType[PoseEnabledControllerType[\"DAYDREAM\"] = 4] = \"DAYDREAM\";\n /**\n * Generic\n */\n PoseEnabledControllerType[PoseEnabledControllerType[\"GENERIC\"] = 5] = \"GENERIC\";\n })(PoseEnabledControllerType || (PoseEnabledControllerType = {}));\n /**\n * Defines the PoseEnabledControllerHelper object that is used initialize a gamepad as the controller type it is specified as (eg. windows mixed reality controller)\n */\n class PoseEnabledControllerHelper {\n /**\n * Initializes a gamepad as the controller type it is specified as (eg. windows mixed reality controller)\n * @param vrGamepad the gamepad to initialized\n * @returns a vr controller of the type the gamepad identified as\n */\n static InitiateController(vrGamepad) {\n for (const factory of this._ControllerFactories) {\n if (factory.canCreate(vrGamepad)) {\n return factory.create(vrGamepad);\n }\n }\n if (this._DefaultControllerFactory) {\n return this._DefaultControllerFactory(vrGamepad);\n }\n throw \"The type of gamepad you are trying to load needs to be imported first or is not supported.\";\n }\n }\n /** @internal */\n PoseEnabledControllerHelper._ControllerFactories = [];\n /** @internal */\n PoseEnabledControllerHelper._DefaultControllerFactory = null;\n /**\n * Defines the PoseEnabledController object that contains state of a vr capable controller\n */\n class PoseEnabledController extends Gamepad {\n /**\n * @internal\n */\n _disableTrackPosition(fixedPosition) {\n if (this._trackPosition) {\n this._calculatedPosition.copyFrom(fixedPosition);\n this._trackPosition = false;\n }\n }\n /**\n * Creates a new PoseEnabledController from a gamepad\n * @param browserGamepad the gamepad that the PoseEnabledController should be created from\n */\n constructor(browserGamepad) {\n super(browserGamepad.id, browserGamepad.index, browserGamepad);\n /**\n * If the controller is used in a webXR session\n */\n this.isXR = false;\n // Represents device position and rotation in room space. Should only be used to help calculate babylon space values\n this._deviceRoomPosition = Vector3.Zero();\n this._deviceRoomRotationQuaternion = new Quaternion();\n /**\n * The device position in babylon space\n */\n this.devicePosition = Vector3.Zero();\n /**\n * The device rotation in babylon space\n */\n this.deviceRotationQuaternion = new Quaternion();\n /**\n * The scale factor of the device in babylon space\n */\n this.deviceScaleFactor = 1;\n // Used to convert 6dof controllers to 3dof\n this._trackPosition = true;\n this._maxRotationDistFromHeadset = Math.PI / 5;\n this._draggedRoomRotation = 0;\n this._leftHandSystemQuaternion = new Quaternion();\n /**\n * Internal, matrix used to convert room space to babylon space\n * @internal\n */\n this._deviceToWorld = Matrix.Identity();\n /**\n * Node to be used when casting a ray from the controller\n * @internal\n */\n this._pointingPoseNode = null;\n this._workingMatrix = Matrix.Identity();\n /**\n * @internal\n */\n this._meshAttachedObservable = new Observable$1();\n this.type = Gamepad.POSE_ENABLED;\n this.controllerType = PoseEnabledControllerType.GENERIC;\n this.position = Vector3.Zero();\n this.rotationQuaternion = new Quaternion();\n this._calculatedPosition = Vector3.Zero();\n this._calculatedRotation = new Quaternion();\n Quaternion.RotationYawPitchRollToRef(Math.PI, 0, 0, this._leftHandSystemQuaternion);\n }\n /**\n * Updates the state of the pose enabled controller and mesh based on the current position and rotation of the controller\n */\n update() {\n super.update();\n this._updatePoseAndMesh();\n }\n /**\n * Updates only the pose device and mesh without doing any button event checking\n */\n _updatePoseAndMesh() {\n if (this.isXR) {\n return;\n }\n const pose = this.browserGamepad.pose;\n this.updateFromDevice(pose);\n if (!this._trackPosition &&\n EngineStore.LastCreatedScene &&\n EngineStore.LastCreatedScene.activeCamera &&\n EngineStore.LastCreatedScene.activeCamera.devicePosition) {\n const camera = EngineStore.LastCreatedScene.activeCamera;\n camera._computeDevicePosition();\n this._deviceToWorld.setTranslation(camera.devicePosition);\n if (camera.deviceRotationQuaternion) {\n camera._deviceRoomRotationQuaternion.toEulerAnglesToRef(TmpVectors.Vector3[0]);\n // Find the radian distance away that the headset is from the controllers rotation\n const distanceAway = Math.atan2(Math.sin(TmpVectors.Vector3[0].y - this._draggedRoomRotation), Math.cos(TmpVectors.Vector3[0].y - this._draggedRoomRotation));\n if (Math.abs(distanceAway) > this._maxRotationDistFromHeadset) {\n // Only rotate enouph to be within the _maxRotationDistFromHeadset\n const rotationAmount = distanceAway - (distanceAway < 0 ? -this._maxRotationDistFromHeadset : this._maxRotationDistFromHeadset);\n this._draggedRoomRotation += rotationAmount;\n // Rotate controller around headset\n const sin = Math.sin(-rotationAmount);\n const cos = Math.cos(-rotationAmount);\n this._calculatedPosition.x = this._calculatedPosition.x * cos - this._calculatedPosition.z * sin;\n this._calculatedPosition.z = this._calculatedPosition.x * sin + this._calculatedPosition.z * cos;\n }\n }\n }\n Vector3.TransformCoordinatesToRef(this._calculatedPosition, this._deviceToWorld, this.devicePosition);\n this._deviceToWorld.getRotationMatrixToRef(this._workingMatrix);\n Quaternion.FromRotationMatrixToRef(this._workingMatrix, this.deviceRotationQuaternion);\n this.deviceRotationQuaternion.multiplyInPlace(this._calculatedRotation);\n if (this._mesh) {\n this._mesh.position.copyFrom(this.devicePosition);\n if (this._mesh.rotationQuaternion) {\n this._mesh.rotationQuaternion.copyFrom(this.deviceRotationQuaternion);\n }\n }\n }\n /**\n * Updates the state of the pose enbaled controller based on the raw pose data from the device\n * @param poseData raw pose fromthe device\n */\n updateFromDevice(poseData) {\n if (this.isXR) {\n return;\n }\n if (poseData) {\n this.rawPose = poseData;\n if (poseData.position) {\n this._deviceRoomPosition.copyFromFloats(poseData.position[0], poseData.position[1], -poseData.position[2]);\n if (this._mesh && this._mesh.getScene().useRightHandedSystem) {\n this._deviceRoomPosition.z *= -1;\n }\n if (this._trackPosition) {\n this._deviceRoomPosition.scaleToRef(this.deviceScaleFactor, this._calculatedPosition);\n }\n this._calculatedPosition.addInPlace(this.position);\n }\n const pose = this.rawPose;\n if (poseData.orientation && pose.orientation && pose.orientation.length === 4) {\n this._deviceRoomRotationQuaternion.copyFromFloats(pose.orientation[0], pose.orientation[1], -pose.orientation[2], -pose.orientation[3]);\n if (this._mesh) {\n if (this._mesh.getScene().useRightHandedSystem) {\n this._deviceRoomRotationQuaternion.z *= -1;\n this._deviceRoomRotationQuaternion.w *= -1;\n }\n else {\n this._deviceRoomRotationQuaternion.multiplyToRef(this._leftHandSystemQuaternion, this._deviceRoomRotationQuaternion);\n }\n }\n // if the camera is set, rotate to the camera's rotation\n this._deviceRoomRotationQuaternion.multiplyToRef(this.rotationQuaternion, this._calculatedRotation);\n }\n }\n }\n /**\n * Attaches a mesh to the controller\n * @param mesh the mesh to be attached\n */\n attachToMesh(mesh) {\n if (this._mesh) {\n this._mesh.parent = null;\n }\n this._mesh = mesh;\n if (this._poseControlledCamera) {\n this._mesh.parent = this._poseControlledCamera;\n }\n if (!this._mesh.rotationQuaternion) {\n this._mesh.rotationQuaternion = new Quaternion();\n }\n // Sync controller mesh and pointing pose node's state with controller, this is done to avoid a frame where position is 0,0,0 when attaching mesh\n if (!this.isXR) {\n this._updatePoseAndMesh();\n if (this._pointingPoseNode) {\n const parents = [];\n let obj = this._pointingPoseNode;\n while (obj.parent) {\n parents.push(obj.parent);\n obj = obj.parent;\n }\n parents.reverse().forEach((p) => {\n p.computeWorldMatrix(true);\n });\n }\n }\n this._meshAttachedObservable.notifyObservers(mesh);\n }\n /**\n * Attaches the controllers mesh to a camera\n * @param camera the camera the mesh should be attached to\n */\n attachToPoseControlledCamera(camera) {\n this._poseControlledCamera = camera;\n if (this._mesh) {\n this._mesh.parent = this._poseControlledCamera;\n }\n }\n /**\n * Disposes of the controller\n */\n dispose() {\n if (this._mesh) {\n this._mesh.dispose();\n }\n this._mesh = null;\n super.dispose();\n }\n /**\n * The mesh that is attached to the controller\n */\n get mesh() {\n return this._mesh;\n }\n /**\n * Gets the ray of the controller in the direction the controller is pointing\n * @param length the length the resulting ray should be\n * @returns a ray in the direction the controller is pointing\n */\n getForwardRay(length = 100) {\n if (!this.mesh) {\n return new Ray(Vector3.Zero(), new Vector3(0, 0, 1), length);\n }\n const m = this._pointingPoseNode ? this._pointingPoseNode.getWorldMatrix() : this.mesh.getWorldMatrix();\n const origin = m.getTranslation();\n const forward = new Vector3(0, 0, -1);\n const forwardWorld = Vector3.TransformNormal(forward, m);\n const direction = Vector3.Normalize(forwardWorld);\n return new Ray(origin, direction, length);\n }\n }\n /**\n * Name of the child mesh that can be used to cast a ray from the controller\n */\n PoseEnabledController.POINTING_POSE = \"POINTING_POSE\";\n\n /**\n * @internal\n */\n function setWebVRRigMode(camera, rigParams) {\n if (rigParams.vrDisplay) {\n const leftEye = rigParams.vrDisplay.getEyeParameters(\"left\");\n const rightEye = rigParams.vrDisplay.getEyeParameters(\"right\");\n //Left eye\n camera._rigCameras[0].viewport = new Viewport(0, 0, 0.5, 1.0);\n camera._rigCameras[0].setCameraRigParameter(\"left\", true);\n //leaving this for future reference\n camera._rigCameras[0].setCameraRigParameter(\"specs\", rigParams.specs);\n camera._rigCameras[0].setCameraRigParameter(\"eyeParameters\", leftEye);\n camera._rigCameras[0].setCameraRigParameter(\"frameData\", rigParams.frameData);\n camera._rigCameras[0].setCameraRigParameter(\"parentCamera\", rigParams.parentCamera);\n camera._rigCameras[0]._cameraRigParams.vrWorkMatrix = new Matrix();\n camera._rigCameras[0].getProjectionMatrix = camera._getWebVRProjectionMatrix;\n camera._rigCameras[0].parent = camera;\n camera._rigCameras[0]._getViewMatrix = camera._getWebVRViewMatrix;\n //Right eye\n camera._rigCameras[1].viewport = new Viewport(0.5, 0, 0.5, 1.0);\n camera._rigCameras[1].setCameraRigParameter(\"eyeParameters\", rightEye);\n camera._rigCameras[1].setCameraRigParameter(\"specs\", rigParams.specs);\n camera._rigCameras[1].setCameraRigParameter(\"frameData\", rigParams.frameData);\n camera._rigCameras[1].setCameraRigParameter(\"parentCamera\", rigParams.parentCamera);\n camera._rigCameras[1]._cameraRigParams.vrWorkMatrix = new Matrix();\n camera._rigCameras[1].getProjectionMatrix = camera._getWebVRProjectionMatrix;\n camera._rigCameras[1].parent = camera;\n camera._rigCameras[1]._getViewMatrix = camera._getWebVRViewMatrix;\n }\n }\n\n Object.defineProperty(Engine.prototype, \"isInVRExclusivePointerMode\", {\n get: function () {\n return this._vrExclusivePointerMode;\n },\n enumerable: true,\n configurable: true,\n });\n Engine.prototype._prepareVRComponent = function () {\n this._vrSupported = false;\n this._vrExclusivePointerMode = false;\n this.onVRDisplayChangedObservable = new Observable$1();\n this.onVRRequestPresentComplete = new Observable$1();\n this.onVRRequestPresentStart = new Observable$1();\n };\n Engine.prototype.isVRDevicePresent = function () {\n return !!this._vrDisplay;\n };\n Engine.prototype.getVRDevice = function () {\n return this._vrDisplay;\n };\n Engine.prototype.initWebVR = function () {\n this.initWebVRAsync();\n return this.onVRDisplayChangedObservable;\n };\n Engine.prototype.initWebVRAsync = function () {\n const notifyObservers = () => {\n const eventArgs = {\n vrDisplay: this._vrDisplay,\n vrSupported: this._vrSupported,\n };\n this.onVRDisplayChangedObservable.notifyObservers(eventArgs);\n this._webVRInitPromise = new Promise((res) => {\n res(eventArgs);\n });\n };\n if (!this._onVrDisplayConnect) {\n this._onVrDisplayConnect = (event) => {\n this._vrDisplay = event.display;\n notifyObservers();\n };\n this._onVrDisplayDisconnect = () => {\n this._vrDisplay.cancelAnimationFrame(this._frameHandler);\n this._vrDisplay = undefined;\n this._frameHandler = Engine.QueueNewFrame(this._boundRenderFunction);\n notifyObservers();\n };\n this._onVrDisplayPresentChange = () => {\n this._vrExclusivePointerMode = this._vrDisplay && this._vrDisplay.isPresenting;\n };\n const hostWindow = this.getHostWindow();\n if (hostWindow) {\n hostWindow.addEventListener(\"vrdisplayconnect\", this._onVrDisplayConnect);\n hostWindow.addEventListener(\"vrdisplaydisconnect\", this._onVrDisplayDisconnect);\n hostWindow.addEventListener(\"vrdisplaypresentchange\", this._onVrDisplayPresentChange);\n }\n }\n this._webVRInitPromise = this._webVRInitPromise || this._getVRDisplaysAsync();\n this._webVRInitPromise.then(notifyObservers);\n return this._webVRInitPromise;\n };\n Engine.prototype._getVRDisplaysAsync = function () {\n return new Promise((res) => {\n if (navigator.getVRDisplays) {\n navigator.getVRDisplays().then((devices) => {\n this._vrSupported = true;\n // note that devices may actually be an empty array. This is fine;\n // we expect this._vrDisplay to be undefined in this case.\n this._vrDisplay = devices[0];\n res({\n vrDisplay: this._vrDisplay,\n vrSupported: this._vrSupported,\n });\n });\n }\n else {\n this._vrDisplay = undefined;\n this._vrSupported = false;\n res({\n vrDisplay: this._vrDisplay,\n vrSupported: this._vrSupported,\n });\n }\n });\n };\n Engine.prototype.enableVR = function (options) {\n if (this._vrDisplay && !this._vrDisplay.isPresenting) {\n const onResolved = () => {\n this.onVRRequestPresentComplete.notifyObservers(true);\n this._onVRFullScreenTriggered();\n };\n const onRejected = () => {\n this.onVRRequestPresentComplete.notifyObservers(false);\n };\n this.onVRRequestPresentStart.notifyObservers(this);\n const presentationAttributes = {\n highRefreshRate: this.vrPresentationAttributes ? this.vrPresentationAttributes.highRefreshRate : false,\n foveationLevel: this.vrPresentationAttributes ? this.vrPresentationAttributes.foveationLevel : 1,\n multiview: (this.getCaps().multiview || this.getCaps().oculusMultiview) && options.useMultiview,\n };\n this._vrDisplay\n .requestPresent([\n Object.assign({ source: this.getRenderingCanvas(), attributes: presentationAttributes }, presentationAttributes),\n ])\n .then(onResolved)\n .catch(onRejected);\n }\n };\n Engine.prototype._onVRFullScreenTriggered = function () {\n if (this._vrDisplay && this._vrDisplay.isPresenting) {\n //get the old size before we change\n this._oldSize = new Size(this.getRenderWidth(), this.getRenderHeight());\n this._oldHardwareScaleFactor = this.getHardwareScalingLevel();\n //get the width and height, change the render size\n const leftEye = this._vrDisplay.getEyeParameters(\"left\");\n this.setHardwareScalingLevel(1);\n this.setSize(leftEye.renderWidth * 2, leftEye.renderHeight);\n }\n else {\n this.setHardwareScalingLevel(this._oldHardwareScaleFactor);\n this.setSize(this._oldSize.width, this._oldSize.height);\n }\n };\n Engine.prototype.disableVR = function () {\n if (this._vrDisplay && this._vrDisplay.isPresenting) {\n this._vrDisplay\n .exitPresent()\n .then(() => this._onVRFullScreenTriggered())\n .catch(() => this._onVRFullScreenTriggered());\n }\n if (IsWindowObjectExist()) {\n window.removeEventListener(\"vrdisplaypointerrestricted\", this._onVRDisplayPointerRestricted);\n window.removeEventListener(\"vrdisplaypointerunrestricted\", this._onVRDisplayPointerUnrestricted);\n if (this._onVrDisplayConnect) {\n window.removeEventListener(\"vrdisplayconnect\", this._onVrDisplayConnect);\n if (this._onVrDisplayDisconnect) {\n window.removeEventListener(\"vrdisplaydisconnect\", this._onVrDisplayDisconnect);\n }\n if (this._onVrDisplayPresentChange) {\n window.removeEventListener(\"vrdisplaypresentchange\", this._onVrDisplayPresentChange);\n }\n this._onVrDisplayConnect = null;\n this._onVrDisplayDisconnect = null;\n }\n }\n };\n Engine.prototype._connectVREvents = function (canvas, document) {\n this._onVRDisplayPointerRestricted = () => {\n if (canvas) {\n canvas.requestPointerLock();\n }\n };\n this._onVRDisplayPointerUnrestricted = () => {\n // Edge fix - for some reason document is not present and this is window\n if (!document) {\n const hostWindow = this.getHostWindow();\n if (hostWindow.document && hostWindow.document.exitPointerLock) {\n hostWindow.document.exitPointerLock();\n }\n return;\n }\n if (!document.exitPointerLock) {\n return;\n }\n document.exitPointerLock();\n };\n if (IsWindowObjectExist()) {\n const hostWindow = this.getHostWindow();\n hostWindow.addEventListener(\"vrdisplaypointerrestricted\", this._onVRDisplayPointerRestricted, false);\n hostWindow.addEventListener(\"vrdisplaypointerunrestricted\", this._onVRDisplayPointerUnrestricted, false);\n }\n };\n Engine.prototype._submitVRFrame = function () {\n // Submit frame to the vr device, if enabled\n if (this._vrDisplay && this._vrDisplay.isPresenting) {\n // TODO: We should only submit the frame if we read frameData successfully.\n try {\n this._vrDisplay.submitFrame();\n }\n catch (e) {\n Tools.Warn(\"webVR submitFrame has had an unexpected failure: \" + e);\n }\n }\n };\n Engine.prototype.isVRPresenting = function () {\n return this._vrDisplay && this._vrDisplay.isPresenting;\n };\n Engine.prototype._requestVRFrame = function () {\n this._frameHandler = Engine.QueueNewFrame(this._boundRenderFunction, this._vrDisplay);\n };\n\n Node.AddNodeConstructor(\"WebVRFreeCamera\", (name, scene) => {\n return () => new WebVRFreeCamera(name, Vector3.Zero(), scene);\n });\n Node.AddNodeConstructor(\"WebVRGamepadCamera\", (name, scene) => {\n return () => new WebVRFreeCamera(name, Vector3.Zero(), scene);\n });\n /**\n * This represents a WebVR camera.\n * The WebVR camera is Babylon's simple interface to interaction with Windows Mixed Reality, HTC Vive and Oculus Rift.\n * @deprecated Use WebXR instead - https://doc.babylonjs.com/features/featuresDeepDive/webXR\n * @example https://doc.babylonjs.com/features/featuresDeepDive/cameras/webVRCamera\n */\n class WebVRFreeCamera extends FreeCamera {\n /**\n * Instantiates a WebVRFreeCamera.\n * @param name The name of the WebVRFreeCamera\n * @param position The starting anchor position for the camera\n * @param scene The scene the camera belongs to\n * @param _webVROptions a set of customizable options for the webVRCamera\n */\n constructor(name, position, scene, _webVROptions = {}) {\n super(name, position, scene);\n this._webVROptions = _webVROptions;\n /**\n * @internal\n * The vrDisplay tied to the camera. See https://developer.mozilla.org/en-US/docs/Web/API/VRDisplay\n */\n this._vrDevice = null;\n /**\n * The rawPose of the vrDevice.\n */\n this.rawPose = null;\n this._specsVersion = \"1.1\";\n this._attached = false;\n this._descendants = [];\n // Represents device position and rotation in room space. Should only be used to help calculate babylon space values\n this._deviceRoomPosition = Vector3.Zero();\n /** @internal */\n this._deviceRoomRotationQuaternion = Quaternion.Identity();\n this._standingMatrix = null;\n /**\n * Represents device position in babylon space.\n */\n this.devicePosition = Vector3.Zero();\n /**\n * Represents device rotation in babylon space.\n */\n this.deviceRotationQuaternion = Quaternion.Identity();\n /**\n * The scale of the device to be used when translating from device space to babylon space.\n */\n this.deviceScaleFactor = 1;\n this._deviceToWorld = Matrix.Identity();\n this._worldToDevice = Matrix.Identity();\n /**\n * References to the webVR controllers for the vrDevice.\n */\n this.controllers = [];\n /**\n * Emits an event when a controller is attached.\n */\n this.onControllersAttachedObservable = new Observable$1();\n /**\n * Emits an event when a controller's mesh has been loaded;\n */\n this.onControllerMeshLoadedObservable = new Observable$1();\n /**\n * Emits an event when the HMD's pose has been updated.\n */\n this.onPoseUpdatedFromDeviceObservable = new Observable$1();\n this._poseSet = false;\n /**\n * If the rig cameras be used as parent instead of this camera.\n */\n this.rigParenting = true;\n this._defaultHeight = undefined;\n this._setRigMode = setWebVRRigMode.bind(null, this);\n this._detachIfAttached = () => {\n const vrDisplay = this.getEngine().getVRDevice();\n if (vrDisplay && !vrDisplay.isPresenting) {\n this.detachControl();\n }\n };\n this._workingVector = Vector3.Zero();\n this._oneVector = Vector3.One();\n this._workingMatrix = Matrix.Identity();\n this._tmpMatrix = new Matrix();\n this._cache.position = Vector3.Zero();\n if (_webVROptions.defaultHeight) {\n this._defaultHeight = _webVROptions.defaultHeight;\n this.position.y = this._defaultHeight;\n }\n this.minZ = 0.1;\n //legacy support - the compensation boolean was removed.\n if (arguments.length === 5) {\n // eslint-disable-next-line prefer-rest-params\n this._webVROptions = arguments[4];\n }\n // default webVR options\n if (this._webVROptions.trackPosition == undefined) {\n this._webVROptions.trackPosition = true;\n }\n if (this._webVROptions.controllerMeshes == undefined) {\n this._webVROptions.controllerMeshes = true;\n }\n if (this._webVROptions.defaultLightingOnControllers == undefined) {\n this._webVROptions.defaultLightingOnControllers = true;\n }\n this.rotationQuaternion = new Quaternion();\n if (this._webVROptions && this._webVROptions.positionScale) {\n this.deviceScaleFactor = this._webVROptions.positionScale;\n }\n //enable VR\n const engine = this.getEngine();\n this._onVREnabled = (success) => {\n if (success) {\n this.initControllers();\n }\n };\n engine.onVRRequestPresentComplete.add(this._onVREnabled);\n engine.initWebVR().add((event) => {\n if (!event.vrDisplay || this._vrDevice === event.vrDisplay) {\n return;\n }\n this._vrDevice = event.vrDisplay;\n //reset the rig parameters.\n this.setCameraRigMode(Camera.RIG_MODE_WEBVR, { parentCamera: this, vrDisplay: this._vrDevice, frameData: this._frameData, specs: this._specsVersion });\n if (this._attached) {\n this.getEngine().enableVR(this._webVROptions);\n }\n });\n if (typeof VRFrameData !== \"undefined\") {\n this._frameData = new VRFrameData();\n }\n if (_webVROptions.useMultiview) {\n if (!this.getScene().getEngine().getCaps().multiview) {\n Logger.Warn(\"Multiview is not supported, falling back to standard rendering\");\n this._useMultiviewToSingleView = false;\n }\n else {\n this._useMultiviewToSingleView = true;\n this._rigPostProcess = new VRMultiviewToSingleviewPostProcess(\"VRMultiviewToSingleview\", this, 1.0);\n }\n }\n /**\n * The idea behind the following lines:\n * objects that have the camera as parent should actually have the rig cameras as a parent.\n * BUT, each of those cameras has a different view matrix, which means that if we set the parent to the first rig camera,\n * the second will not show it correctly.\n *\n * To solve this - each object that has the camera as parent will be added to a protected array.\n * When the rig camera renders, it will take this array and set all of those to be its children.\n * This way, the right camera will be used as a parent, and the mesh will be rendered correctly.\n * Amazing!\n */\n this.getScene().onBeforeCameraRenderObservable.add((camera) => {\n if (camera.parent === this && this.rigParenting) {\n this._descendants = this.getDescendants(true, (n) => {\n // don't take the cameras or the controllers!\n const isController = this.controllers.some((controller) => {\n return controller._mesh === n;\n });\n const isRigCamera = this._rigCameras.indexOf(n) !== -1;\n return !isController && !isRigCamera;\n });\n this._descendants.forEach((node) => {\n node.parent = camera;\n });\n }\n });\n this.getScene().onAfterCameraRenderObservable.add((camera) => {\n if (camera.parent === this && this.rigParenting) {\n this._descendants.forEach((node) => {\n node.parent = this;\n });\n }\n });\n }\n /**\n * Gets the device distance from the ground in meters.\n * @returns the distance in meters from the vrDevice to ground in device space. If standing matrix is not supported for the vrDevice 0 is returned.\n */\n deviceDistanceToRoomGround() {\n if (this._standingMatrix) {\n // Add standing matrix offset to get real offset from ground in room\n this._standingMatrix.getTranslationToRef(this._workingVector);\n return this._deviceRoomPosition.y + this._workingVector.y;\n }\n //If VRDisplay does not inform stage parameters and no default height is set we fallback to zero.\n return this._defaultHeight || 0;\n }\n /**\n * Enables the standing matrix when supported. This can be used to position the user's view the correct height from the ground.\n * @param callback will be called when the standing matrix is set. Callback parameter is if the standing matrix is supported.\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n useStandingMatrix(callback = (bool) => { }) {\n // Use standing matrix if available\n this.getEngine()\n .initWebVRAsync()\n .then((result) => {\n if (!result.vrDisplay || !result.vrDisplay.stageParameters || !result.vrDisplay.stageParameters.sittingToStandingTransform || !this._webVROptions.trackPosition) {\n callback(false);\n }\n else {\n this._standingMatrix = new Matrix();\n Matrix.FromFloat32ArrayToRefScaled(result.vrDisplay.stageParameters.sittingToStandingTransform, 0, 1, this._standingMatrix);\n if (!this.getScene().useRightHandedSystem) {\n if (this._standingMatrix) {\n this._standingMatrix.toggleModelMatrixHandInPlace();\n }\n }\n callback(true);\n }\n });\n }\n /**\n * Enables the standing matrix when supported. This can be used to position the user's view the correct height from the ground.\n * @returns A promise with a boolean set to if the standing matrix is supported.\n */\n useStandingMatrixAsync() {\n return new Promise((res) => {\n this.useStandingMatrix((supported) => {\n res(supported);\n });\n });\n }\n /**\n * Disposes the camera\n */\n dispose() {\n this._detachIfAttached();\n this.getEngine().onVRRequestPresentComplete.removeCallback(this._onVREnabled);\n if (this._updateCacheWhenTrackingDisabledObserver) {\n this._scene.onBeforeRenderObservable.remove(this._updateCacheWhenTrackingDisabledObserver);\n }\n super.dispose();\n }\n /**\n * Gets a vrController by name.\n * @param name The name of the controller to retrieve\n * @returns the controller matching the name specified or null if not found\n */\n getControllerByName(name) {\n for (const gp of this.controllers) {\n if (gp.hand === name) {\n return gp;\n }\n }\n return null;\n }\n /**\n * The controller corresponding to the users left hand.\n */\n get leftController() {\n if (!this._leftController) {\n this._leftController = this.getControllerByName(\"left\");\n }\n return this._leftController;\n }\n /**\n * The controller corresponding to the users right hand.\n */\n get rightController() {\n if (!this._rightController) {\n this._rightController = this.getControllerByName(\"right\");\n }\n return this._rightController;\n }\n /**\n * Casts a ray forward from the vrCamera's gaze.\n * @param length Length of the ray (default: 100)\n * @returns the ray corresponding to the gaze\n */\n getForwardRay(length = 100) {\n if (this.leftCamera) {\n // Use left eye to avoid computation to compute center on every call\n return super.getForwardRay(length, this.leftCamera.getWorldMatrix(), this.leftCamera.globalPosition); // Need the actual rendered camera\n }\n else {\n return super.getForwardRay(length);\n }\n }\n /**\n * @internal\n * Updates the camera based on device's frame data\n */\n _checkInputs() {\n if (this._vrDevice && this._vrDevice.isPresenting) {\n this._vrDevice.getFrameData(this._frameData);\n this.updateFromDevice(this._frameData.pose);\n }\n super._checkInputs();\n }\n /**\n * Updates the poseControlled values based on the input device pose.\n * @param poseData Pose coming from the device\n */\n updateFromDevice(poseData) {\n if (poseData && poseData.orientation && poseData.orientation.length === 4) {\n this.rawPose = poseData;\n this._deviceRoomRotationQuaternion.copyFromFloats(poseData.orientation[0], poseData.orientation[1], -poseData.orientation[2], -poseData.orientation[3]);\n if (this.getScene().useRightHandedSystem) {\n this._deviceRoomRotationQuaternion.z *= -1;\n this._deviceRoomRotationQuaternion.w *= -1;\n }\n if (this._webVROptions.trackPosition && this.rawPose.position) {\n this._deviceRoomPosition.copyFromFloats(this.rawPose.position[0], this.rawPose.position[1], -this.rawPose.position[2]);\n if (this.getScene().useRightHandedSystem) {\n this._deviceRoomPosition.z *= -1;\n }\n }\n this._poseSet = true;\n }\n }\n /**\n * WebVR's attach control will start broadcasting frames to the device.\n * Note that in certain browsers (chrome for example) this function must be called\n * within a user-interaction callback. Example:\n *
 scene.onPointerDown = function() { camera.attachControl(canvas); }
\n *\n * @param noPreventDefault prevent the default html element operation when attaching the vrDevice\n */\n attachControl(noPreventDefault) {\n // eslint-disable-next-line prefer-rest-params\n noPreventDefault = Tools.BackCompatCameraNoPreventDefault(arguments);\n super.attachControl(noPreventDefault);\n this._attached = true;\n noPreventDefault = Camera.ForceAttachControlToAlwaysPreventDefault ? false : noPreventDefault;\n if (this._vrDevice) {\n this.getEngine().enableVR(this._webVROptions);\n }\n const hostWindow = this._scene.getEngine().getHostWindow();\n if (hostWindow) {\n hostWindow.addEventListener(\"vrdisplaypresentchange\", this._detachIfAttached);\n }\n }\n /**\n * Detach the current controls from the specified dom element.\n */\n detachControl() {\n this.getScene().gamepadManager.onGamepadConnectedObservable.remove(this._onGamepadConnectedObserver);\n this.getScene().gamepadManager.onGamepadDisconnectedObservable.remove(this._onGamepadDisconnectedObserver);\n super.detachControl();\n this._attached = false;\n this.getEngine().disableVR();\n window.removeEventListener(\"vrdisplaypresentchange\", this._detachIfAttached);\n }\n /**\n * @returns the name of this class\n */\n getClassName() {\n return \"WebVRFreeCamera\";\n }\n /**\n * Calls resetPose on the vrDisplay\n * See: https://developer.mozilla.org/en-US/docs/Web/API/VRDisplay/resetPose\n */\n resetToCurrentRotation() {\n //uses the vrDisplay's \"resetPose()\".\n //pitch and roll won't be affected.\n this._vrDevice.resetPose();\n }\n /**\n * @internal\n * Updates the rig cameras (left and right eye)\n */\n _updateRigCameras() {\n const camLeft = this._rigCameras[0];\n const camRight = this._rigCameras[1];\n camLeft.rotationQuaternion.copyFrom(this._deviceRoomRotationQuaternion);\n camRight.rotationQuaternion.copyFrom(this._deviceRoomRotationQuaternion);\n camLeft.position.copyFrom(this._deviceRoomPosition);\n camRight.position.copyFrom(this._deviceRoomPosition);\n }\n // Remove translation from 6dof headset if trackposition is set to false\n _correctPositionIfNotTrackPosition(matrix, isViewMatrix = false) {\n if (this.rawPose && this.rawPose.position && !this._webVROptions.trackPosition) {\n Matrix.TranslationToRef(this.rawPose.position[0], this.rawPose.position[1], -this.rawPose.position[2], this._tmpMatrix);\n if (!isViewMatrix) {\n this._tmpMatrix.invert();\n }\n this._tmpMatrix.multiplyToRef(matrix, matrix);\n }\n }\n /**\n * @internal\n * Updates the cached values of the camera\n * @param ignoreParentClass ignores updating the parent class's cache (default: false)\n */\n _updateCache(ignoreParentClass) {\n if (!this.rotationQuaternion.equals(this._cache.rotationQuaternion) || !this.position.equals(this._cache.position)) {\n // Update to ensure devicePosition is up to date with most recent _deviceRoomPosition\n if (!this._updateCacheCalled) {\n // make sure it is only called once per loop. this.update() might cause an infinite loop.\n this._updateCacheCalled = true;\n this.update();\n }\n // Set working vector to the device position in room space rotated by the new rotation\n this.rotationQuaternion.toRotationMatrix(this._workingMatrix);\n Vector3.TransformCoordinatesToRef(this._deviceRoomPosition, this._workingMatrix, this._workingVector);\n // Subtract this vector from the current device position in world to get the translation for the device world matrix\n this.devicePosition.subtractToRef(this._workingVector, this._workingVector);\n Matrix.ComposeToRef(this._oneVector, this.rotationQuaternion, this._workingVector, this._deviceToWorld);\n // Add translation from anchor position\n this._deviceToWorld.getTranslationToRef(this._workingVector);\n this._workingVector.addInPlace(this.position);\n this._workingVector.subtractInPlace(this._cache.position);\n this._deviceToWorld.setTranslation(this._workingVector);\n // Set an inverted matrix to be used when updating the camera\n this._deviceToWorld.invertToRef(this._worldToDevice);\n // Update the gamepad to ensure the mesh is updated on the same frame as camera\n this.controllers.forEach((controller) => {\n controller._deviceToWorld.copyFrom(this._deviceToWorld);\n this._correctPositionIfNotTrackPosition(controller._deviceToWorld);\n controller.update();\n });\n }\n if (!ignoreParentClass) {\n super._updateCache();\n }\n this._updateCacheCalled = false;\n }\n /**\n * @internal\n * Get current device position in babylon world\n */\n _computeDevicePosition() {\n Vector3.TransformCoordinatesToRef(this._deviceRoomPosition, this._deviceToWorld, this.devicePosition);\n }\n /**\n * Updates the current device position and rotation in the babylon world\n */\n update() {\n this._computeDevicePosition();\n // Get current device rotation in babylon world\n Matrix.FromQuaternionToRef(this._deviceRoomRotationQuaternion, this._workingMatrix);\n this._workingMatrix.multiplyToRef(this._deviceToWorld, this._workingMatrix);\n Quaternion.FromRotationMatrixToRef(this._workingMatrix, this.deviceRotationQuaternion);\n if (this._poseSet) {\n this.onPoseUpdatedFromDeviceObservable.notifyObservers(null);\n }\n super.update();\n }\n /**\n * @internal\n * Gets the view matrix of this camera (Always set to identity as left and right eye cameras contain the actual view matrix)\n * @returns an identity matrix\n */\n _getViewMatrix() {\n return Matrix.Identity();\n }\n /**\n * This function is called by the two RIG cameras.\n * 'this' is the left or right camera (and NOT (!!!) the WebVRFreeCamera instance)\n * @internal\n */\n _getWebVRViewMatrix() {\n // Update the parent camera prior to using a child camera to avoid desynchronization\n const parentCamera = this._cameraRigParams[\"parentCamera\"];\n parentCamera._updateCache();\n //WebVR 1.1\n const viewArray = this._cameraRigParams[\"left\"] ? this._cameraRigParams[\"frameData\"].leftViewMatrix : this._cameraRigParams[\"frameData\"].rightViewMatrix;\n Matrix.FromArrayToRef(viewArray, 0, this._webvrViewMatrix);\n if (!this.getScene().useRightHandedSystem) {\n this._webvrViewMatrix.toggleModelMatrixHandInPlace();\n }\n // update the camera rotation matrix\n this._webvrViewMatrix.getRotationMatrixToRef(this._cameraRotationMatrix);\n Vector3.TransformCoordinatesToRef(this._referencePoint, this._cameraRotationMatrix, this._transformedReferencePoint);\n // Computing target and final matrix\n this.position.addToRef(this._transformedReferencePoint, this._currentTarget);\n // should the view matrix be updated with scale and position offset?\n if (parentCamera.deviceScaleFactor !== 1) {\n this._webvrViewMatrix.invert();\n // scale the position, if set\n if (parentCamera.deviceScaleFactor) {\n this._webvrViewMatrix.multiplyAtIndex(12, parentCamera.deviceScaleFactor);\n this._webvrViewMatrix.multiplyAtIndex(13, parentCamera.deviceScaleFactor);\n this._webvrViewMatrix.multiplyAtIndex(14, parentCamera.deviceScaleFactor);\n }\n this._webvrViewMatrix.invert();\n }\n // Remove translation from 6dof headset if trackposition is set to false\n parentCamera._correctPositionIfNotTrackPosition(this._webvrViewMatrix, true);\n parentCamera._worldToDevice.multiplyToRef(this._webvrViewMatrix, this._webvrViewMatrix);\n // Compute global position\n this._workingMatrix = this._workingMatrix || Matrix.Identity();\n this._webvrViewMatrix.invertToRef(this._workingMatrix);\n this._workingMatrix.multiplyToRef(parentCamera.getWorldMatrix(), this._workingMatrix);\n this._workingMatrix.getTranslationToRef(this._globalPosition);\n this._markSyncedWithParent();\n return this._webvrViewMatrix;\n }\n /** @internal */\n _getWebVRProjectionMatrix() {\n const parentCamera = this.parent;\n parentCamera._vrDevice.depthNear = parentCamera.minZ;\n parentCamera._vrDevice.depthFar = parentCamera.maxZ;\n const projectionArray = this._cameraRigParams[\"left\"] ? this._cameraRigParams[\"frameData\"].leftProjectionMatrix : this._cameraRigParams[\"frameData\"].rightProjectionMatrix;\n Matrix.FromArrayToRef(projectionArray, 0, this._projectionMatrix);\n //babylon compatible matrix\n if (!this.getScene().useRightHandedSystem) {\n this._projectionMatrix.toggleProjectionMatrixHandInPlace();\n }\n return this._projectionMatrix;\n }\n /**\n * Initializes the controllers and their meshes\n */\n initControllers() {\n this.controllers.length = 0;\n const manager = this.getScene().gamepadManager;\n this._onGamepadDisconnectedObserver = manager.onGamepadDisconnectedObservable.add((gamepad) => {\n if (gamepad.type === Gamepad.POSE_ENABLED) {\n const webVrController = gamepad;\n if (webVrController.defaultModel) {\n webVrController.defaultModel.setEnabled(false);\n }\n if (webVrController.hand === \"right\") {\n this._rightController = null;\n }\n if (webVrController.hand === \"left\") {\n this._leftController = null;\n }\n const controllerIndex = this.controllers.indexOf(webVrController);\n if (controllerIndex !== -1) {\n this.controllers.splice(controllerIndex, 1);\n }\n }\n });\n this._onGamepadConnectedObserver = manager.onGamepadConnectedObservable.add((gamepad) => {\n if (gamepad.type === Gamepad.POSE_ENABLED) {\n const webVrController = gamepad;\n if (!this._webVROptions.trackPosition) {\n webVrController._disableTrackPosition(new Vector3(webVrController.hand == \"left\" ? -0.15 : 0.15, -0.5, 0.25));\n // Cache must be updated before rendering controllers to avoid them being one frame behind\n if (!this._updateCacheWhenTrackingDisabledObserver) {\n this._updateCacheWhenTrackingDisabledObserver = this._scene.onBeforeRenderObservable.add(() => {\n this._updateCache();\n });\n }\n }\n webVrController.deviceScaleFactor = this.deviceScaleFactor;\n webVrController._deviceToWorld.copyFrom(this._deviceToWorld);\n this._correctPositionIfNotTrackPosition(webVrController._deviceToWorld);\n if (this._webVROptions.controllerMeshes) {\n if (webVrController.defaultModel) {\n webVrController.defaultModel.setEnabled(true);\n }\n else {\n // Load the meshes\n webVrController.initControllerMesh(this.getScene(), (loadedMesh) => {\n loadedMesh.scaling.scaleInPlace(this.deviceScaleFactor);\n this.onControllerMeshLoadedObservable.notifyObservers(webVrController);\n if (this._webVROptions.defaultLightingOnControllers) {\n if (!this._lightOnControllers) {\n this._lightOnControllers = new HemisphericLight(\"vrControllersLight\", new Vector3(0, 1, 0), this.getScene());\n }\n const activateLightOnSubMeshes = function (mesh, light) {\n const children = mesh.getChildren();\n if (children && children.length !== 0) {\n children.forEach((mesh) => {\n light.includedOnlyMeshes.push(mesh);\n activateLightOnSubMeshes(mesh, light);\n });\n }\n };\n this._lightOnControllers.includedOnlyMeshes.push(loadedMesh);\n activateLightOnSubMeshes(loadedMesh, this._lightOnControllers);\n }\n });\n }\n }\n webVrController.attachToPoseControlledCamera(this);\n // since this is async - sanity check. Is the controller already stored?\n if (this.controllers.indexOf(webVrController) === -1) {\n //add to the controllers array\n this.controllers.push(webVrController);\n // Forced to add some control code for Vive as it doesn't always fill properly the \"hand\" property\n // Sometimes, both controllers are set correctly (left and right), sometimes none, sometimes only one of them...\n // So we're overriding setting left & right manually to be sure\n let firstViveWandDetected = false;\n for (let i = 0; i < this.controllers.length; i++) {\n if (this.controllers[i].controllerType === PoseEnabledControllerType.VIVE) {\n if (!firstViveWandDetected) {\n firstViveWandDetected = true;\n this.controllers[i].hand = \"left\";\n }\n else {\n this.controllers[i].hand = \"right\";\n }\n }\n }\n //did we find enough controllers? Great! let the developer know.\n if (this.controllers.length >= 2) {\n this.onControllersAttachedObservable.notifyObservers(this.controllers);\n }\n }\n }\n });\n }\n }\n\n /**\n * Defines the WebVRController object that represents controllers tracked in 3D space\n * @deprecated Use WebXR instead\n */\n class WebVRController extends PoseEnabledController {\n /**\n * Fired when a controller button's state has changed\n * @param callback the callback containing the button that was modified\n */\n onButtonStateChange(callback) {\n this._onButtonStateChange = callback;\n }\n /**\n * The default controller model for the controller\n */\n get defaultModel() {\n return this._defaultModel;\n }\n /**\n * Creates a new WebVRController from a gamepad\n * @param vrGamepad the gamepad that the WebVRController should be created from\n */\n constructor(vrGamepad) {\n super(vrGamepad);\n // Observables\n /**\n * Fired when the trigger state has changed\n */\n this.onTriggerStateChangedObservable = new Observable$1();\n /**\n * Fired when the main button state has changed\n */\n this.onMainButtonStateChangedObservable = new Observable$1();\n /**\n * Fired when the secondary button state has changed\n */\n this.onSecondaryButtonStateChangedObservable = new Observable$1();\n /**\n * Fired when the pad state has changed\n */\n this.onPadStateChangedObservable = new Observable$1();\n /**\n * Fired when controllers stick values have changed\n */\n this.onPadValuesChangedObservable = new Observable$1();\n /**\n * X and Y axis corresponding to the controllers joystick\n */\n this.pad = { x: 0, y: 0 };\n // avoid GC, store state in a tmp object\n this._changes = {\n pressChanged: false,\n touchChanged: false,\n valueChanged: false,\n changed: false,\n };\n this._buttons = new Array(vrGamepad.buttons.length);\n this.hand = vrGamepad.hand;\n }\n /**\n * Updates the state of the controller and mesh based on the current position and rotation of the controller\n */\n update() {\n super.update();\n for (let index = 0; index < this._buttons.length; index++) {\n this._setButtonValue(this.browserGamepad.buttons[index], this._buttons[index], index);\n }\n if (this.leftStick.x !== this.pad.x || this.leftStick.y !== this.pad.y) {\n this.pad.x = this.leftStick.x;\n this.pad.y = this.leftStick.y;\n this.onPadValuesChangedObservable.notifyObservers(this.pad);\n }\n }\n _setButtonValue(newState, currentState, buttonIndex) {\n if (!newState) {\n newState = {\n pressed: false,\n touched: false,\n value: 0,\n };\n }\n if (!currentState) {\n this._buttons[buttonIndex] = {\n pressed: newState.pressed,\n touched: newState.touched,\n value: newState.value,\n };\n return;\n }\n this._checkChanges(newState, currentState);\n if (this._changes.changed) {\n this._onButtonStateChange && this._onButtonStateChange(this.index, buttonIndex, newState);\n this._handleButtonChange(buttonIndex, newState, this._changes);\n }\n this._buttons[buttonIndex].pressed = newState.pressed;\n this._buttons[buttonIndex].touched = newState.touched;\n // oculus triggers are never 0, thou not touched.\n this._buttons[buttonIndex].value = newState.value < 0.00000001 ? 0 : newState.value;\n }\n _checkChanges(newState, currentState) {\n this._changes.pressChanged = newState.pressed !== currentState.pressed;\n this._changes.touchChanged = newState.touched !== currentState.touched;\n this._changes.valueChanged = newState.value !== currentState.value;\n this._changes.changed = this._changes.pressChanged || this._changes.touchChanged || this._changes.valueChanged;\n return this._changes;\n }\n /**\n * Disposes of th webVRController\n */\n dispose() {\n super.dispose();\n this._defaultModel = null;\n this.onTriggerStateChangedObservable.clear();\n this.onMainButtonStateChangedObservable.clear();\n this.onSecondaryButtonStateChangedObservable.clear();\n this.onPadStateChangedObservable.clear();\n this.onPadValuesChangedObservable.clear();\n }\n }\n\n /**\n * Defines supported buttons for XBox360 compatible gamepads\n */\n var Xbox360Button;\n (function (Xbox360Button) {\n /** A */\n Xbox360Button[Xbox360Button[\"A\"] = 0] = \"A\";\n /** B */\n Xbox360Button[Xbox360Button[\"B\"] = 1] = \"B\";\n /** X */\n Xbox360Button[Xbox360Button[\"X\"] = 2] = \"X\";\n /** Y */\n Xbox360Button[Xbox360Button[\"Y\"] = 3] = \"Y\";\n /** Left button */\n Xbox360Button[Xbox360Button[\"LB\"] = 4] = \"LB\";\n /** Right button */\n Xbox360Button[Xbox360Button[\"RB\"] = 5] = \"RB\";\n /** Back */\n Xbox360Button[Xbox360Button[\"Back\"] = 8] = \"Back\";\n /** Start */\n Xbox360Button[Xbox360Button[\"Start\"] = 9] = \"Start\";\n /** Left stick */\n Xbox360Button[Xbox360Button[\"LeftStick\"] = 10] = \"LeftStick\";\n /** Right stick */\n Xbox360Button[Xbox360Button[\"RightStick\"] = 11] = \"RightStick\";\n })(Xbox360Button || (Xbox360Button = {}));\n /** Defines values for XBox360 DPad */\n var Xbox360Dpad;\n (function (Xbox360Dpad) {\n /** Up */\n Xbox360Dpad[Xbox360Dpad[\"Up\"] = 12] = \"Up\";\n /** Down */\n Xbox360Dpad[Xbox360Dpad[\"Down\"] = 13] = \"Down\";\n /** Left */\n Xbox360Dpad[Xbox360Dpad[\"Left\"] = 14] = \"Left\";\n /** Right */\n Xbox360Dpad[Xbox360Dpad[\"Right\"] = 15] = \"Right\";\n })(Xbox360Dpad || (Xbox360Dpad = {}));\n /**\n * Defines a XBox360 gamepad\n */\n class Xbox360Pad extends Gamepad {\n /**\n * Creates a new XBox360 gamepad object\n * @param id defines the id of this gamepad\n * @param index defines its index\n * @param gamepad defines the internal HTML gamepad object\n * @param xboxOne defines if it is a XBox One gamepad\n */\n constructor(id, index, gamepad, xboxOne = false) {\n super(id, index, gamepad, 0, 1, 2, 3);\n this._leftTrigger = 0;\n this._rightTrigger = 0;\n /** Observable raised when a button is pressed */\n this.onButtonDownObservable = new Observable$1();\n /** Observable raised when a button is released */\n this.onButtonUpObservable = new Observable$1();\n /** Observable raised when a pad is pressed */\n this.onPadDownObservable = new Observable$1();\n /** Observable raised when a pad is released */\n this.onPadUpObservable = new Observable$1();\n this._buttonA = 0;\n this._buttonB = 0;\n this._buttonX = 0;\n this._buttonY = 0;\n this._buttonBack = 0;\n this._buttonStart = 0;\n this._buttonLB = 0;\n this._buttonRB = 0;\n this._buttonLeftStick = 0;\n this._buttonRightStick = 0;\n this._dPadUp = 0;\n this._dPadDown = 0;\n this._dPadLeft = 0;\n this._dPadRight = 0;\n this._isXboxOnePad = false;\n this.type = Gamepad.XBOX;\n this._isXboxOnePad = xboxOne;\n }\n /**\n * Defines the callback to call when left trigger is pressed\n * @param callback defines the callback to use\n */\n onlefttriggerchanged(callback) {\n this._onlefttriggerchanged = callback;\n }\n /**\n * Defines the callback to call when right trigger is pressed\n * @param callback defines the callback to use\n */\n onrighttriggerchanged(callback) {\n this._onrighttriggerchanged = callback;\n }\n /**\n * Gets the left trigger value\n */\n get leftTrigger() {\n return this._leftTrigger;\n }\n /**\n * Sets the left trigger value\n */\n set leftTrigger(newValue) {\n if (this._onlefttriggerchanged && this._leftTrigger !== newValue) {\n this._onlefttriggerchanged(newValue);\n }\n this._leftTrigger = newValue;\n }\n /**\n * Gets the right trigger value\n */\n get rightTrigger() {\n return this._rightTrigger;\n }\n /**\n * Sets the right trigger value\n */\n set rightTrigger(newValue) {\n if (this._onrighttriggerchanged && this._rightTrigger !== newValue) {\n this._onrighttriggerchanged(newValue);\n }\n this._rightTrigger = newValue;\n }\n /**\n * Defines the callback to call when a button is pressed\n * @param callback defines the callback to use\n */\n onbuttondown(callback) {\n this._onbuttondown = callback;\n }\n /**\n * Defines the callback to call when a button is released\n * @param callback defines the callback to use\n */\n onbuttonup(callback) {\n this._onbuttonup = callback;\n }\n /**\n * Defines the callback to call when a pad is pressed\n * @param callback defines the callback to use\n */\n ondpaddown(callback) {\n this._ondpaddown = callback;\n }\n /**\n * Defines the callback to call when a pad is released\n * @param callback defines the callback to use\n */\n ondpadup(callback) {\n this._ondpadup = callback;\n }\n _setButtonValue(newValue, currentValue, buttonType) {\n if (newValue !== currentValue) {\n if (newValue === 1) {\n if (this._onbuttondown) {\n this._onbuttondown(buttonType);\n }\n this.onButtonDownObservable.notifyObservers(buttonType);\n }\n if (newValue === 0) {\n if (this._onbuttonup) {\n this._onbuttonup(buttonType);\n }\n this.onButtonUpObservable.notifyObservers(buttonType);\n }\n }\n return newValue;\n }\n _setDPadValue(newValue, currentValue, buttonType) {\n if (newValue !== currentValue) {\n if (newValue === 1) {\n if (this._ondpaddown) {\n this._ondpaddown(buttonType);\n }\n this.onPadDownObservable.notifyObservers(buttonType);\n }\n if (newValue === 0) {\n if (this._ondpadup) {\n this._ondpadup(buttonType);\n }\n this.onPadUpObservable.notifyObservers(buttonType);\n }\n }\n return newValue;\n }\n /**\n * Gets the value of the `A` button\n */\n get buttonA() {\n return this._buttonA;\n }\n /**\n * Sets the value of the `A` button\n */\n set buttonA(value) {\n this._buttonA = this._setButtonValue(value, this._buttonA, Xbox360Button.A);\n }\n /**\n * Gets the value of the `B` button\n */\n get buttonB() {\n return this._buttonB;\n }\n /**\n * Sets the value of the `B` button\n */\n set buttonB(value) {\n this._buttonB = this._setButtonValue(value, this._buttonB, Xbox360Button.B);\n }\n /**\n * Gets the value of the `X` button\n */\n get buttonX() {\n return this._buttonX;\n }\n /**\n * Sets the value of the `X` button\n */\n set buttonX(value) {\n this._buttonX = this._setButtonValue(value, this._buttonX, Xbox360Button.X);\n }\n /**\n * Gets the value of the `Y` button\n */\n get buttonY() {\n return this._buttonY;\n }\n /**\n * Sets the value of the `Y` button\n */\n set buttonY(value) {\n this._buttonY = this._setButtonValue(value, this._buttonY, Xbox360Button.Y);\n }\n /**\n * Gets the value of the `Start` button\n */\n get buttonStart() {\n return this._buttonStart;\n }\n /**\n * Sets the value of the `Start` button\n */\n set buttonStart(value) {\n this._buttonStart = this._setButtonValue(value, this._buttonStart, Xbox360Button.Start);\n }\n /**\n * Gets the value of the `Back` button\n */\n get buttonBack() {\n return this._buttonBack;\n }\n /**\n * Sets the value of the `Back` button\n */\n set buttonBack(value) {\n this._buttonBack = this._setButtonValue(value, this._buttonBack, Xbox360Button.Back);\n }\n /**\n * Gets the value of the `Left` button\n */\n get buttonLB() {\n return this._buttonLB;\n }\n /**\n * Sets the value of the `Left` button\n */\n set buttonLB(value) {\n this._buttonLB = this._setButtonValue(value, this._buttonLB, Xbox360Button.LB);\n }\n /**\n * Gets the value of the `Right` button\n */\n get buttonRB() {\n return this._buttonRB;\n }\n /**\n * Sets the value of the `Right` button\n */\n set buttonRB(value) {\n this._buttonRB = this._setButtonValue(value, this._buttonRB, Xbox360Button.RB);\n }\n /**\n * Gets the value of the Left joystick\n */\n get buttonLeftStick() {\n return this._buttonLeftStick;\n }\n /**\n * Sets the value of the Left joystick\n */\n set buttonLeftStick(value) {\n this._buttonLeftStick = this._setButtonValue(value, this._buttonLeftStick, Xbox360Button.LeftStick);\n }\n /**\n * Gets the value of the Right joystick\n */\n get buttonRightStick() {\n return this._buttonRightStick;\n }\n /**\n * Sets the value of the Right joystick\n */\n set buttonRightStick(value) {\n this._buttonRightStick = this._setButtonValue(value, this._buttonRightStick, Xbox360Button.RightStick);\n }\n /**\n * Gets the value of D-pad up\n */\n get dPadUp() {\n return this._dPadUp;\n }\n /**\n * Sets the value of D-pad up\n */\n set dPadUp(value) {\n this._dPadUp = this._setDPadValue(value, this._dPadUp, Xbox360Dpad.Up);\n }\n /**\n * Gets the value of D-pad down\n */\n get dPadDown() {\n return this._dPadDown;\n }\n /**\n * Sets the value of D-pad down\n */\n set dPadDown(value) {\n this._dPadDown = this._setDPadValue(value, this._dPadDown, Xbox360Dpad.Down);\n }\n /**\n * Gets the value of D-pad left\n */\n get dPadLeft() {\n return this._dPadLeft;\n }\n /**\n * Sets the value of D-pad left\n */\n set dPadLeft(value) {\n this._dPadLeft = this._setDPadValue(value, this._dPadLeft, Xbox360Dpad.Left);\n }\n /**\n * Gets the value of D-pad right\n */\n get dPadRight() {\n return this._dPadRight;\n }\n /**\n * Sets the value of D-pad right\n */\n set dPadRight(value) {\n this._dPadRight = this._setDPadValue(value, this._dPadRight, Xbox360Dpad.Right);\n }\n /**\n * Force the gamepad to synchronize with device values\n */\n update() {\n super.update();\n if (this._isXboxOnePad) {\n this.buttonA = this.browserGamepad.buttons[0].value;\n this.buttonB = this.browserGamepad.buttons[1].value;\n this.buttonX = this.browserGamepad.buttons[2].value;\n this.buttonY = this.browserGamepad.buttons[3].value;\n this.buttonLB = this.browserGamepad.buttons[4].value;\n this.buttonRB = this.browserGamepad.buttons[5].value;\n this.leftTrigger = this.browserGamepad.buttons[6].value;\n this.rightTrigger = this.browserGamepad.buttons[7].value;\n this.buttonBack = this.browserGamepad.buttons[8].value;\n this.buttonStart = this.browserGamepad.buttons[9].value;\n this.buttonLeftStick = this.browserGamepad.buttons[10].value;\n this.buttonRightStick = this.browserGamepad.buttons[11].value;\n this.dPadUp = this.browserGamepad.buttons[12].value;\n this.dPadDown = this.browserGamepad.buttons[13].value;\n this.dPadLeft = this.browserGamepad.buttons[14].value;\n this.dPadRight = this.browserGamepad.buttons[15].value;\n }\n else {\n this.buttonA = this.browserGamepad.buttons[0].value;\n this.buttonB = this.browserGamepad.buttons[1].value;\n this.buttonX = this.browserGamepad.buttons[2].value;\n this.buttonY = this.browserGamepad.buttons[3].value;\n this.buttonLB = this.browserGamepad.buttons[4].value;\n this.buttonRB = this.browserGamepad.buttons[5].value;\n this.leftTrigger = this.browserGamepad.buttons[6].value;\n this.rightTrigger = this.browserGamepad.buttons[7].value;\n this.buttonBack = this.browserGamepad.buttons[8].value;\n this.buttonStart = this.browserGamepad.buttons[9].value;\n this.buttonLeftStick = this.browserGamepad.buttons[10].value;\n this.buttonRightStick = this.browserGamepad.buttons[11].value;\n this.dPadUp = this.browserGamepad.buttons[12].value;\n this.dPadDown = this.browserGamepad.buttons[13].value;\n this.dPadLeft = this.browserGamepad.buttons[14].value;\n this.dPadRight = this.browserGamepad.buttons[15].value;\n }\n }\n /**\n * Disposes the gamepad\n */\n dispose() {\n super.dispose();\n this.onButtonDownObservable.clear();\n this.onButtonUpObservable.clear();\n this.onPadDownObservable.clear();\n this.onPadUpObservable.clear();\n }\n }\n\n ThinEngine.prototype.createDynamicTexture = function (width, height, generateMipMaps, samplingMode) {\n const texture = new InternalTexture(this, InternalTextureSource.Dynamic);\n texture.baseWidth = width;\n texture.baseHeight = height;\n if (generateMipMaps) {\n width = this.needPOTTextures ? ThinEngine.GetExponentOfTwo(width, this._caps.maxTextureSize) : width;\n height = this.needPOTTextures ? ThinEngine.GetExponentOfTwo(height, this._caps.maxTextureSize) : height;\n }\n // this.resetTextureCache();\n texture.width = width;\n texture.height = height;\n texture.isReady = false;\n texture.generateMipMaps = generateMipMaps;\n texture.samplingMode = samplingMode;\n this.updateTextureSamplingMode(samplingMode, texture);\n this._internalTexturesCache.push(texture);\n return texture;\n };\n ThinEngine.prototype.updateDynamicTexture = function (texture, source, invertY, premulAlpha = false, format, forceBindTexture = false, \n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n allowGPUOptimization = false) {\n if (!texture) {\n return;\n }\n const gl = this._gl;\n const target = gl.TEXTURE_2D;\n const wasPreviouslyBound = this._bindTextureDirectly(target, texture, true, forceBindTexture);\n this._unpackFlipY(invertY === undefined ? texture.invertY : invertY);\n if (premulAlpha) {\n gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, 1);\n }\n const textureType = this._getWebGLTextureType(texture.type);\n const glformat = this._getInternalFormat(format ? format : texture.format);\n const internalFormat = this._getRGBABufferInternalSizedFormat(texture.type, glformat);\n gl.texImage2D(target, 0, internalFormat, glformat, textureType, source);\n if (texture.generateMipMaps) {\n gl.generateMipmap(target);\n }\n if (!wasPreviouslyBound) {\n this._bindTextureDirectly(target, null);\n }\n if (premulAlpha) {\n gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, 0);\n }\n texture.isReady = true;\n };\n\n /**\n * A class extending Texture allowing drawing on a texture\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/dynamicTexture\n */\n class DynamicTexture extends Texture {\n /**\n * Creates a DynamicTexture\n * @param name defines the name of the texture\n * @param options provides 3 alternatives for width and height of texture, a canvas, object with width and height properties, number for both width and height\n * @param scene defines the scene where you want the texture\n * @param generateMipMaps defines the use of MinMaps or not (default is false)\n * @param samplingMode defines the sampling mode to use (default is Texture.TRILINEAR_SAMPLINGMODE)\n * @param format defines the texture format to use (default is Engine.TEXTUREFORMAT_RGBA)\n * @param invertY defines if the texture needs to be inverted on the y axis during loading\n */\n constructor(name, options, scene = null, generateMipMaps = false, samplingMode = 3, format = 5, invertY) {\n super(null, scene, !generateMipMaps, invertY, samplingMode, undefined, undefined, undefined, undefined, format);\n this.name = name;\n this.wrapU = Texture.CLAMP_ADDRESSMODE;\n this.wrapV = Texture.CLAMP_ADDRESSMODE;\n this._generateMipMaps = generateMipMaps;\n const engine = this._getEngine();\n if (!engine) {\n return;\n }\n if (options.getContext) {\n this._canvas = options;\n this._texture = engine.createDynamicTexture(options.width, options.height, generateMipMaps, samplingMode);\n }\n else {\n this._canvas = engine.createCanvas(1, 1);\n if (options.width || options.width === 0) {\n this._texture = engine.createDynamicTexture(options.width, options.height, generateMipMaps, samplingMode);\n }\n else {\n this._texture = engine.createDynamicTexture(options, options, generateMipMaps, samplingMode);\n }\n }\n const textureSize = this.getSize();\n if (this._canvas.width !== textureSize.width) {\n this._canvas.width = textureSize.width;\n }\n if (this._canvas.height !== textureSize.height) {\n this._canvas.height = textureSize.height;\n }\n this._context = this._canvas.getContext(\"2d\");\n }\n /**\n * Get the current class name of the texture useful for serialization or dynamic coding.\n * @returns \"DynamicTexture\"\n */\n getClassName() {\n return \"DynamicTexture\";\n }\n /**\n * Gets the current state of canRescale\n */\n get canRescale() {\n return true;\n }\n _recreate(textureSize) {\n this._canvas.width = textureSize.width;\n this._canvas.height = textureSize.height;\n this.releaseInternalTexture();\n this._texture = this._getEngine().createDynamicTexture(textureSize.width, textureSize.height, this._generateMipMaps, this.samplingMode);\n }\n /**\n * Scales the texture\n * @param ratio the scale factor to apply to both width and height\n */\n scale(ratio) {\n const textureSize = this.getSize();\n textureSize.width *= ratio;\n textureSize.height *= ratio;\n this._recreate(textureSize);\n }\n /**\n * Resizes the texture\n * @param width the new width\n * @param height the new height\n */\n scaleTo(width, height) {\n const textureSize = this.getSize();\n textureSize.width = width;\n textureSize.height = height;\n this._recreate(textureSize);\n }\n /**\n * Gets the context of the canvas used by the texture\n * @returns the canvas context of the dynamic texture\n */\n getContext() {\n return this._context;\n }\n /**\n * Clears the texture\n */\n clear() {\n const size = this.getSize();\n this._context.fillRect(0, 0, size.width, size.height);\n }\n /**\n * Updates the texture\n * @param invertY defines the direction for the Y axis (default is true - y increases downwards)\n * @param premulAlpha defines if alpha is stored as premultiplied (default is false)\n * @param allowGPUOptimization true to allow some specific GPU optimizations (subject to engine feature \"allowGPUOptimizationsForGUI\" being true)\n */\n update(invertY, premulAlpha = false, allowGPUOptimization = false) {\n this._getEngine().updateDynamicTexture(this._texture, this._canvas, invertY === undefined ? true : invertY, premulAlpha, this._format || undefined, undefined, allowGPUOptimization);\n }\n /**\n * Draws text onto the texture\n * @param text defines the text to be drawn\n * @param x defines the placement of the text from the left\n * @param y defines the placement of the text from the top when invertY is true and from the bottom when false\n * @param font defines the font to be used with font-style, font-size, font-name\n * @param color defines the color used for the text\n * @param clearColor defines the color for the canvas, use null to not overwrite canvas\n * @param invertY defines the direction for the Y axis (default is true - y increases downwards)\n * @param update defines whether texture is immediately update (default is true)\n */\n drawText(text, x, y, font, color, clearColor, invertY, update = true) {\n const size = this.getSize();\n if (clearColor) {\n this._context.fillStyle = clearColor;\n this._context.fillRect(0, 0, size.width, size.height);\n }\n this._context.font = font;\n if (x === null || x === undefined) {\n const textSize = this._context.measureText(text);\n x = (size.width - textSize.width) / 2;\n }\n if (y === null || y === undefined) {\n const fontSize = parseInt(font.replace(/\\D/g, \"\"));\n y = size.height / 2 + fontSize / 3.65;\n }\n this._context.fillStyle = color || \"\";\n this._context.fillText(text, x, y);\n if (update) {\n this.update(invertY);\n }\n }\n /**\n * Clones the texture\n * @returns the clone of the texture.\n */\n clone() {\n const scene = this.getScene();\n if (!scene) {\n return this;\n }\n const textureSize = this.getSize();\n const newTexture = new DynamicTexture(this.name, textureSize, scene, this._generateMipMaps);\n // Base texture\n newTexture.hasAlpha = this.hasAlpha;\n newTexture.level = this.level;\n // Dynamic Texture\n newTexture.wrapU = this.wrapU;\n newTexture.wrapV = this.wrapV;\n return newTexture;\n }\n /**\n * Serializes the dynamic texture. The scene should be ready before the dynamic texture is serialized\n * @returns a serialized dynamic texture object\n */\n serialize() {\n const scene = this.getScene();\n if (scene && !scene.isReady()) {\n Logger.Warn(\"The scene must be ready before serializing the dynamic texture\");\n }\n const serializationObject = super.serialize();\n if (DynamicTexture._IsCanvasElement(this._canvas)) {\n serializationObject.base64String = this._canvas.toDataURL();\n }\n serializationObject.invertY = this._invertY;\n serializationObject.samplingMode = this.samplingMode;\n return serializationObject;\n }\n static _IsCanvasElement(canvas) {\n return canvas.toDataURL !== undefined;\n }\n /** @internal */\n _rebuild() {\n this.update();\n }\n }\n\n // Do not edit.\n const name$1V = \"imageProcessingPixelShader\";\n const shader$1V = `varying vec2 vUV;uniform sampler2D textureSampler;\n#include\n#include\n#include\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{vec4 result=texture2D(textureSampler,vUV);\n#ifdef IMAGEPROCESSING\n#ifndef FROMLINEARSPACE\nresult.rgb=toLinearSpace(result.rgb);\n#endif\nresult=applyImageProcessing(result);\n#else\n#ifdef FROMLINEARSPACE\nresult=applyImageProcessing(result);\n#endif\n#endif\ngl_FragColor=result;}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$1V] = shader$1V;\n\n /**\n * ImageProcessingPostProcess\n * @see https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/usePostProcesses#imageprocessing\n */\n class ImageProcessingPostProcess extends PostProcess {\n /**\n * Gets the image processing configuration used either in this material.\n */\n get imageProcessingConfiguration() {\n return this._imageProcessingConfiguration;\n }\n /**\n * Sets the Default image processing configuration used either in the this material.\n *\n * If sets to null, the scene one is in use.\n */\n set imageProcessingConfiguration(value) {\n // We are almost sure it is applied by post process as\n // We are in the post process :-)\n value.applyByPostProcess = true;\n this._attachImageProcessingConfiguration(value);\n }\n /**\n * Attaches a new image processing configuration to the PBR Material.\n * @param configuration\n * @param doNotBuild\n */\n _attachImageProcessingConfiguration(configuration, doNotBuild = false) {\n if (configuration === this._imageProcessingConfiguration) {\n return;\n }\n // Detaches observer.\n if (this._imageProcessingConfiguration && this._imageProcessingObserver) {\n this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver);\n }\n // Pick the scene configuration if needed.\n if (!configuration) {\n let scene = null;\n const engine = this.getEngine();\n const camera = this.getCamera();\n if (camera) {\n scene = camera.getScene();\n }\n else if (engine && engine.scenes) {\n const scenes = engine.scenes;\n scene = scenes[scenes.length - 1];\n }\n else {\n scene = EngineStore.LastCreatedScene;\n }\n if (scene) {\n this._imageProcessingConfiguration = scene.imageProcessingConfiguration;\n }\n else {\n this._imageProcessingConfiguration = new ImageProcessingConfiguration();\n }\n }\n else {\n this._imageProcessingConfiguration = configuration;\n }\n // Attaches observer.\n if (this._imageProcessingConfiguration) {\n this._imageProcessingObserver = this._imageProcessingConfiguration.onUpdateParameters.add(() => {\n this._updateParameters();\n });\n }\n // Ensure the effect will be rebuilt.\n if (!doNotBuild) {\n this._updateParameters();\n }\n }\n /**\n * If the post process is supported.\n */\n get isSupported() {\n const effect = this.getEffect();\n return !effect || effect.isSupported;\n }\n /**\n * Gets Color curves setup used in the effect if colorCurvesEnabled is set to true .\n */\n get colorCurves() {\n return this.imageProcessingConfiguration.colorCurves;\n }\n /**\n * Sets Color curves setup used in the effect if colorCurvesEnabled is set to true .\n */\n set colorCurves(value) {\n this.imageProcessingConfiguration.colorCurves = value;\n }\n /**\n * Gets whether the color curves effect is enabled.\n */\n get colorCurvesEnabled() {\n return this.imageProcessingConfiguration.colorCurvesEnabled;\n }\n /**\n * Sets whether the color curves effect is enabled.\n */\n set colorCurvesEnabled(value) {\n this.imageProcessingConfiguration.colorCurvesEnabled = value;\n }\n /**\n * Gets Color grading LUT texture used in the effect if colorGradingEnabled is set to true.\n */\n get colorGradingTexture() {\n return this.imageProcessingConfiguration.colorGradingTexture;\n }\n /**\n * Sets Color grading LUT texture used in the effect if colorGradingEnabled is set to true.\n */\n set colorGradingTexture(value) {\n this.imageProcessingConfiguration.colorGradingTexture = value;\n }\n /**\n * Gets whether the color grading effect is enabled.\n */\n get colorGradingEnabled() {\n return this.imageProcessingConfiguration.colorGradingEnabled;\n }\n /**\n * Gets whether the color grading effect is enabled.\n */\n set colorGradingEnabled(value) {\n this.imageProcessingConfiguration.colorGradingEnabled = value;\n }\n /**\n * Gets exposure used in the effect.\n */\n get exposure() {\n return this.imageProcessingConfiguration.exposure;\n }\n /**\n * Sets exposure used in the effect.\n */\n set exposure(value) {\n this.imageProcessingConfiguration.exposure = value;\n }\n /**\n * Gets whether tonemapping is enabled or not.\n */\n get toneMappingEnabled() {\n return this._imageProcessingConfiguration.toneMappingEnabled;\n }\n /**\n * Sets whether tonemapping is enabled or not\n */\n set toneMappingEnabled(value) {\n this._imageProcessingConfiguration.toneMappingEnabled = value;\n }\n /**\n * Gets the type of tone mapping effect.\n */\n get toneMappingType() {\n return this._imageProcessingConfiguration.toneMappingType;\n }\n /**\n * Sets the type of tone mapping effect.\n */\n set toneMappingType(value) {\n this._imageProcessingConfiguration.toneMappingType = value;\n }\n /**\n * Gets contrast used in the effect.\n */\n get contrast() {\n return this.imageProcessingConfiguration.contrast;\n }\n /**\n * Sets contrast used in the effect.\n */\n set contrast(value) {\n this.imageProcessingConfiguration.contrast = value;\n }\n /**\n * Gets Vignette stretch size.\n */\n get vignetteStretch() {\n return this.imageProcessingConfiguration.vignetteStretch;\n }\n /**\n * Sets Vignette stretch size.\n */\n set vignetteStretch(value) {\n this.imageProcessingConfiguration.vignetteStretch = value;\n }\n /**\n * Gets Vignette center X Offset.\n * @deprecated use vignetteCenterX instead\n */\n get vignetteCentreX() {\n return this.imageProcessingConfiguration.vignetteCenterX;\n }\n /**\n * Sets Vignette center X Offset.\n * @deprecated use vignetteCenterX instead\n */\n set vignetteCentreX(value) {\n this.imageProcessingConfiguration.vignetteCenterX = value;\n }\n /**\n * Gets Vignette center Y Offset.\n * @deprecated use vignetteCenterY instead\n */\n get vignetteCentreY() {\n return this.imageProcessingConfiguration.vignetteCenterY;\n }\n /**\n * Sets Vignette center Y Offset.\n * @deprecated use vignetteCenterY instead\n */\n set vignetteCentreY(value) {\n this.imageProcessingConfiguration.vignetteCenterY = value;\n }\n /**\n * Vignette center Y Offset.\n */\n get vignetteCenterY() {\n return this.imageProcessingConfiguration.vignetteCenterY;\n }\n set vignetteCenterY(value) {\n this.imageProcessingConfiguration.vignetteCenterY = value;\n }\n /**\n * Vignette center X Offset.\n */\n get vignetteCenterX() {\n return this.imageProcessingConfiguration.vignetteCenterX;\n }\n set vignetteCenterX(value) {\n this.imageProcessingConfiguration.vignetteCenterX = value;\n }\n /**\n * Gets Vignette weight or intensity of the vignette effect.\n */\n get vignetteWeight() {\n return this.imageProcessingConfiguration.vignetteWeight;\n }\n /**\n * Sets Vignette weight or intensity of the vignette effect.\n */\n set vignetteWeight(value) {\n this.imageProcessingConfiguration.vignetteWeight = value;\n }\n /**\n * Gets Color of the vignette applied on the screen through the chosen blend mode (vignetteBlendMode)\n * if vignetteEnabled is set to true.\n */\n get vignetteColor() {\n return this.imageProcessingConfiguration.vignetteColor;\n }\n /**\n * Sets Color of the vignette applied on the screen through the chosen blend mode (vignetteBlendMode)\n * if vignetteEnabled is set to true.\n */\n set vignetteColor(value) {\n this.imageProcessingConfiguration.vignetteColor = value;\n }\n /**\n * Gets Camera field of view used by the Vignette effect.\n */\n get vignetteCameraFov() {\n return this.imageProcessingConfiguration.vignetteCameraFov;\n }\n /**\n * Sets Camera field of view used by the Vignette effect.\n */\n set vignetteCameraFov(value) {\n this.imageProcessingConfiguration.vignetteCameraFov = value;\n }\n /**\n * Gets the vignette blend mode allowing different kind of effect.\n */\n get vignetteBlendMode() {\n return this.imageProcessingConfiguration.vignetteBlendMode;\n }\n /**\n * Sets the vignette blend mode allowing different kind of effect.\n */\n set vignetteBlendMode(value) {\n this.imageProcessingConfiguration.vignetteBlendMode = value;\n }\n /**\n * Gets whether the vignette effect is enabled.\n */\n get vignetteEnabled() {\n return this.imageProcessingConfiguration.vignetteEnabled;\n }\n /**\n * Sets whether the vignette effect is enabled.\n */\n set vignetteEnabled(value) {\n this.imageProcessingConfiguration.vignetteEnabled = value;\n }\n /**\n * Gets intensity of the dithering effect.\n */\n get ditheringIntensity() {\n return this.imageProcessingConfiguration.ditheringIntensity;\n }\n /**\n * Sets intensity of the dithering effect.\n */\n set ditheringIntensity(value) {\n this.imageProcessingConfiguration.ditheringIntensity = value;\n }\n /**\n * Gets whether the dithering effect is enabled.\n */\n get ditheringEnabled() {\n return this.imageProcessingConfiguration.ditheringEnabled;\n }\n /**\n * Sets whether the dithering effect is enabled.\n */\n set ditheringEnabled(value) {\n this.imageProcessingConfiguration.ditheringEnabled = value;\n }\n /**\n * Gets whether the input of the processing is in Gamma or Linear Space.\n */\n get fromLinearSpace() {\n return this._fromLinearSpace;\n }\n /**\n * Sets whether the input of the processing is in Gamma or Linear Space.\n */\n set fromLinearSpace(value) {\n if (this._fromLinearSpace === value) {\n return;\n }\n this._fromLinearSpace = value;\n this._updateParameters();\n }\n constructor(name, options, camera = null, samplingMode, engine, reusable, textureType = 0, imageProcessingConfiguration) {\n super(name, \"imageProcessing\", [], [], options, camera, samplingMode, engine, reusable, null, textureType, \"postprocess\", null, true);\n this._fromLinearSpace = true;\n /**\n * Defines cache preventing GC.\n */\n this._defines = {\n IMAGEPROCESSING: false,\n VIGNETTE: false,\n VIGNETTEBLENDMODEMULTIPLY: false,\n VIGNETTEBLENDMODEOPAQUE: false,\n TONEMAPPING: false,\n TONEMAPPING_ACES: false,\n CONTRAST: false,\n COLORCURVES: false,\n COLORGRADING: false,\n COLORGRADING3D: false,\n FROMLINEARSPACE: false,\n SAMPLER3DGREENDEPTH: false,\n SAMPLER3DBGRMAP: false,\n DITHER: false,\n IMAGEPROCESSINGPOSTPROCESS: false,\n EXPOSURE: false,\n SKIPFINALCOLORCLAMP: false,\n };\n // Setup the configuration as forced by the constructor. This would then not force the\n // scene materials output in linear space and let untouched the default forward pass.\n if (imageProcessingConfiguration) {\n imageProcessingConfiguration.applyByPostProcess = true;\n this._attachImageProcessingConfiguration(imageProcessingConfiguration, true);\n // This will cause the shader to be compiled\n this._updateParameters();\n }\n // Setup the default processing configuration to the scene.\n else {\n this._attachImageProcessingConfiguration(null, true);\n this.imageProcessingConfiguration.applyByPostProcess = true;\n }\n this.onApply = (effect) => {\n this.imageProcessingConfiguration.bind(effect, this.aspectRatio);\n };\n }\n /**\n * \"ImageProcessingPostProcess\"\n * @returns \"ImageProcessingPostProcess\"\n */\n getClassName() {\n return \"ImageProcessingPostProcess\";\n }\n /**\n * @internal\n */\n _updateParameters() {\n this._defines.FROMLINEARSPACE = this._fromLinearSpace;\n this.imageProcessingConfiguration.prepareDefines(this._defines, true);\n let defines = \"\";\n for (const define in this._defines) {\n if (this._defines[define]) {\n defines += `#define ${define};\\n`;\n }\n }\n const samplers = [\"textureSampler\"];\n const uniforms = [\"scale\"];\n if (ImageProcessingConfiguration) {\n ImageProcessingConfiguration.PrepareSamplers(samplers, this._defines);\n ImageProcessingConfiguration.PrepareUniforms(uniforms, this._defines);\n }\n this.updateEffect(defines, uniforms, samplers);\n }\n dispose(camera) {\n super.dispose(camera);\n if (this._imageProcessingConfiguration && this._imageProcessingObserver) {\n this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver);\n }\n if (this._imageProcessingConfiguration) {\n this.imageProcessingConfiguration.applyByPostProcess = false;\n }\n }\n }\n __decorate$1([\n serialize()\n ], ImageProcessingPostProcess.prototype, \"_fromLinearSpace\", void 0);\n\n /**\n * Defines supported buttons for DualShock compatible gamepads\n */\n var DualShockButton;\n (function (DualShockButton) {\n /** Cross */\n DualShockButton[DualShockButton[\"Cross\"] = 0] = \"Cross\";\n /** Circle */\n DualShockButton[DualShockButton[\"Circle\"] = 1] = \"Circle\";\n /** Square */\n DualShockButton[DualShockButton[\"Square\"] = 2] = \"Square\";\n /** Triangle */\n DualShockButton[DualShockButton[\"Triangle\"] = 3] = \"Triangle\";\n /** L1 */\n DualShockButton[DualShockButton[\"L1\"] = 4] = \"L1\";\n /** R1 */\n DualShockButton[DualShockButton[\"R1\"] = 5] = \"R1\";\n /** Share */\n DualShockButton[DualShockButton[\"Share\"] = 8] = \"Share\";\n /** Options */\n DualShockButton[DualShockButton[\"Options\"] = 9] = \"Options\";\n /** Left stick */\n DualShockButton[DualShockButton[\"LeftStick\"] = 10] = \"LeftStick\";\n /** Right stick */\n DualShockButton[DualShockButton[\"RightStick\"] = 11] = \"RightStick\";\n })(DualShockButton || (DualShockButton = {}));\n /** Defines values for DualShock DPad */\n var DualShockDpad;\n (function (DualShockDpad) {\n /** Up */\n DualShockDpad[DualShockDpad[\"Up\"] = 12] = \"Up\";\n /** Down */\n DualShockDpad[DualShockDpad[\"Down\"] = 13] = \"Down\";\n /** Left */\n DualShockDpad[DualShockDpad[\"Left\"] = 14] = \"Left\";\n /** Right */\n DualShockDpad[DualShockDpad[\"Right\"] = 15] = \"Right\";\n })(DualShockDpad || (DualShockDpad = {}));\n /**\n * Defines a DualShock gamepad\n */\n class DualShockPad extends Gamepad {\n /**\n * Creates a new DualShock gamepad object\n * @param id defines the id of this gamepad\n * @param index defines its index\n * @param gamepad defines the internal HTML gamepad object\n */\n constructor(id, index, gamepad) {\n super(id.replace(\"STANDARD GAMEPAD\", \"SONY PLAYSTATION DUALSHOCK\"), index, gamepad, 0, 1, 2, 3);\n this._leftTrigger = 0;\n this._rightTrigger = 0;\n /** Observable raised when a button is pressed */\n this.onButtonDownObservable = new Observable$1();\n /** Observable raised when a button is released */\n this.onButtonUpObservable = new Observable$1();\n /** Observable raised when a pad is pressed */\n this.onPadDownObservable = new Observable$1();\n /** Observable raised when a pad is released */\n this.onPadUpObservable = new Observable$1();\n this._buttonCross = 0;\n this._buttonCircle = 0;\n this._buttonSquare = 0;\n this._buttonTriangle = 0;\n this._buttonShare = 0;\n this._buttonOptions = 0;\n this._buttonL1 = 0;\n this._buttonR1 = 0;\n this._buttonLeftStick = 0;\n this._buttonRightStick = 0;\n this._dPadUp = 0;\n this._dPadDown = 0;\n this._dPadLeft = 0;\n this._dPadRight = 0;\n this.type = Gamepad.DUALSHOCK;\n }\n /**\n * Defines the callback to call when left trigger is pressed\n * @param callback defines the callback to use\n */\n onlefttriggerchanged(callback) {\n this._onlefttriggerchanged = callback;\n }\n /**\n * Defines the callback to call when right trigger is pressed\n * @param callback defines the callback to use\n */\n onrighttriggerchanged(callback) {\n this._onrighttriggerchanged = callback;\n }\n /**\n * Gets the left trigger value\n */\n get leftTrigger() {\n return this._leftTrigger;\n }\n /**\n * Sets the left trigger value\n */\n set leftTrigger(newValue) {\n if (this._onlefttriggerchanged && this._leftTrigger !== newValue) {\n this._onlefttriggerchanged(newValue);\n }\n this._leftTrigger = newValue;\n }\n /**\n * Gets the right trigger value\n */\n get rightTrigger() {\n return this._rightTrigger;\n }\n /**\n * Sets the right trigger value\n */\n set rightTrigger(newValue) {\n if (this._onrighttriggerchanged && this._rightTrigger !== newValue) {\n this._onrighttriggerchanged(newValue);\n }\n this._rightTrigger = newValue;\n }\n /**\n * Defines the callback to call when a button is pressed\n * @param callback defines the callback to use\n */\n onbuttondown(callback) {\n this._onbuttondown = callback;\n }\n /**\n * Defines the callback to call when a button is released\n * @param callback defines the callback to use\n */\n onbuttonup(callback) {\n this._onbuttonup = callback;\n }\n /**\n * Defines the callback to call when a pad is pressed\n * @param callback defines the callback to use\n */\n ondpaddown(callback) {\n this._ondpaddown = callback;\n }\n /**\n * Defines the callback to call when a pad is released\n * @param callback defines the callback to use\n */\n ondpadup(callback) {\n this._ondpadup = callback;\n }\n _setButtonValue(newValue, currentValue, buttonType) {\n if (newValue !== currentValue) {\n if (newValue === 1) {\n if (this._onbuttondown) {\n this._onbuttondown(buttonType);\n }\n this.onButtonDownObservable.notifyObservers(buttonType);\n }\n if (newValue === 0) {\n if (this._onbuttonup) {\n this._onbuttonup(buttonType);\n }\n this.onButtonUpObservable.notifyObservers(buttonType);\n }\n }\n return newValue;\n }\n _setDPadValue(newValue, currentValue, buttonType) {\n if (newValue !== currentValue) {\n if (newValue === 1) {\n if (this._ondpaddown) {\n this._ondpaddown(buttonType);\n }\n this.onPadDownObservable.notifyObservers(buttonType);\n }\n if (newValue === 0) {\n if (this._ondpadup) {\n this._ondpadup(buttonType);\n }\n this.onPadUpObservable.notifyObservers(buttonType);\n }\n }\n return newValue;\n }\n /**\n * Gets the value of the `Cross` button\n */\n get buttonCross() {\n return this._buttonCross;\n }\n /**\n * Sets the value of the `Cross` button\n */\n set buttonCross(value) {\n this._buttonCross = this._setButtonValue(value, this._buttonCross, DualShockButton.Cross);\n }\n /**\n * Gets the value of the `Circle` button\n */\n get buttonCircle() {\n return this._buttonCircle;\n }\n /**\n * Sets the value of the `Circle` button\n */\n set buttonCircle(value) {\n this._buttonCircle = this._setButtonValue(value, this._buttonCircle, DualShockButton.Circle);\n }\n /**\n * Gets the value of the `Square` button\n */\n get buttonSquare() {\n return this._buttonSquare;\n }\n /**\n * Sets the value of the `Square` button\n */\n set buttonSquare(value) {\n this._buttonSquare = this._setButtonValue(value, this._buttonSquare, DualShockButton.Square);\n }\n /**\n * Gets the value of the `Triangle` button\n */\n get buttonTriangle() {\n return this._buttonTriangle;\n }\n /**\n * Sets the value of the `Triangle` button\n */\n set buttonTriangle(value) {\n this._buttonTriangle = this._setButtonValue(value, this._buttonTriangle, DualShockButton.Triangle);\n }\n /**\n * Gets the value of the `Options` button\n */\n get buttonOptions() {\n return this._buttonOptions;\n }\n /**\n * Sets the value of the `Options` button\n */\n set buttonOptions(value) {\n this._buttonOptions = this._setButtonValue(value, this._buttonOptions, DualShockButton.Options);\n }\n /**\n * Gets the value of the `Share` button\n */\n get buttonShare() {\n return this._buttonShare;\n }\n /**\n * Sets the value of the `Share` button\n */\n set buttonShare(value) {\n this._buttonShare = this._setButtonValue(value, this._buttonShare, DualShockButton.Share);\n }\n /**\n * Gets the value of the `L1` button\n */\n get buttonL1() {\n return this._buttonL1;\n }\n /**\n * Sets the value of the `L1` button\n */\n set buttonL1(value) {\n this._buttonL1 = this._setButtonValue(value, this._buttonL1, DualShockButton.L1);\n }\n /**\n * Gets the value of the `R1` button\n */\n get buttonR1() {\n return this._buttonR1;\n }\n /**\n * Sets the value of the `R1` button\n */\n set buttonR1(value) {\n this._buttonR1 = this._setButtonValue(value, this._buttonR1, DualShockButton.R1);\n }\n /**\n * Gets the value of the Left joystick\n */\n get buttonLeftStick() {\n return this._buttonLeftStick;\n }\n /**\n * Sets the value of the Left joystick\n */\n set buttonLeftStick(value) {\n this._buttonLeftStick = this._setButtonValue(value, this._buttonLeftStick, DualShockButton.LeftStick);\n }\n /**\n * Gets the value of the Right joystick\n */\n get buttonRightStick() {\n return this._buttonRightStick;\n }\n /**\n * Sets the value of the Right joystick\n */\n set buttonRightStick(value) {\n this._buttonRightStick = this._setButtonValue(value, this._buttonRightStick, DualShockButton.RightStick);\n }\n /**\n * Gets the value of D-pad up\n */\n get dPadUp() {\n return this._dPadUp;\n }\n /**\n * Sets the value of D-pad up\n */\n set dPadUp(value) {\n this._dPadUp = this._setDPadValue(value, this._dPadUp, DualShockDpad.Up);\n }\n /**\n * Gets the value of D-pad down\n */\n get dPadDown() {\n return this._dPadDown;\n }\n /**\n * Sets the value of D-pad down\n */\n set dPadDown(value) {\n this._dPadDown = this._setDPadValue(value, this._dPadDown, DualShockDpad.Down);\n }\n /**\n * Gets the value of D-pad left\n */\n get dPadLeft() {\n return this._dPadLeft;\n }\n /**\n * Sets the value of D-pad left\n */\n set dPadLeft(value) {\n this._dPadLeft = this._setDPadValue(value, this._dPadLeft, DualShockDpad.Left);\n }\n /**\n * Gets the value of D-pad right\n */\n get dPadRight() {\n return this._dPadRight;\n }\n /**\n * Sets the value of D-pad right\n */\n set dPadRight(value) {\n this._dPadRight = this._setDPadValue(value, this._dPadRight, DualShockDpad.Right);\n }\n /**\n * Force the gamepad to synchronize with device values\n */\n update() {\n super.update();\n this.buttonCross = this.browserGamepad.buttons[0].value;\n this.buttonCircle = this.browserGamepad.buttons[1].value;\n this.buttonSquare = this.browserGamepad.buttons[2].value;\n this.buttonTriangle = this.browserGamepad.buttons[3].value;\n this.buttonL1 = this.browserGamepad.buttons[4].value;\n this.buttonR1 = this.browserGamepad.buttons[5].value;\n this.leftTrigger = this.browserGamepad.buttons[6].value;\n this.rightTrigger = this.browserGamepad.buttons[7].value;\n this.buttonShare = this.browserGamepad.buttons[8].value;\n this.buttonOptions = this.browserGamepad.buttons[9].value;\n this.buttonLeftStick = this.browserGamepad.buttons[10].value;\n this.buttonRightStick = this.browserGamepad.buttons[11].value;\n this.dPadUp = this.browserGamepad.buttons[12].value;\n this.dPadDown = this.browserGamepad.buttons[13].value;\n this.dPadLeft = this.browserGamepad.buttons[14].value;\n this.dPadRight = this.browserGamepad.buttons[15].value;\n }\n /**\n * Disposes the gamepad\n */\n dispose() {\n super.dispose();\n this.onButtonDownObservable.clear();\n this.onButtonUpObservable.clear();\n this.onPadDownObservable.clear();\n this.onPadUpObservable.clear();\n }\n }\n\n /**\n * Manager for handling gamepads\n */\n class GamepadManager {\n /**\n * Initializes the gamepad manager\n * @param _scene BabylonJS scene\n */\n constructor(_scene) {\n this._scene = _scene;\n this._babylonGamepads = [];\n this._oneGamepadConnected = false;\n /** @internal */\n this._isMonitoring = false;\n /**\n * observable to be triggered when the gamepad controller has been disconnected\n */\n this.onGamepadDisconnectedObservable = new Observable$1();\n if (!IsWindowObjectExist()) {\n this._gamepadEventSupported = false;\n }\n else {\n this._gamepadEventSupported = \"GamepadEvent\" in window;\n this._gamepadSupport = navigator && navigator.getGamepads;\n }\n this.onGamepadConnectedObservable = new Observable$1((observer) => {\n // This will be used to raise the onGamepadConnected for all gamepads ALREADY connected\n for (const i in this._babylonGamepads) {\n const gamepad = this._babylonGamepads[i];\n if (gamepad && gamepad._isConnected) {\n this.onGamepadConnectedObservable.notifyObserver(observer, gamepad);\n }\n }\n });\n this._onGamepadConnectedEvent = (evt) => {\n const gamepad = evt.gamepad;\n if (gamepad.index in this._babylonGamepads) {\n if (this._babylonGamepads[gamepad.index].isConnected) {\n return;\n }\n }\n let newGamepad;\n if (this._babylonGamepads[gamepad.index]) {\n newGamepad = this._babylonGamepads[gamepad.index];\n newGamepad.browserGamepad = gamepad;\n newGamepad._isConnected = true;\n }\n else {\n newGamepad = this._addNewGamepad(gamepad);\n }\n this.onGamepadConnectedObservable.notifyObservers(newGamepad);\n this._startMonitoringGamepads();\n };\n this._onGamepadDisconnectedEvent = (evt) => {\n const gamepad = evt.gamepad;\n // Remove the gamepad from the list of gamepads to monitor.\n for (const i in this._babylonGamepads) {\n if (this._babylonGamepads[i].index === gamepad.index) {\n const disconnectedGamepad = this._babylonGamepads[i];\n disconnectedGamepad._isConnected = false;\n this.onGamepadDisconnectedObservable.notifyObservers(disconnectedGamepad);\n disconnectedGamepad.dispose && disconnectedGamepad.dispose();\n break;\n }\n }\n };\n if (this._gamepadSupport) {\n //first add already-connected gamepads\n this._updateGamepadObjects();\n if (this._babylonGamepads.length) {\n this._startMonitoringGamepads();\n }\n // Checking if the gamepad connected event is supported (like in Firefox)\n if (this._gamepadEventSupported) {\n const hostWindow = this._scene ? this._scene.getEngine().getHostWindow() : window;\n if (hostWindow) {\n hostWindow.addEventListener(\"gamepadconnected\", this._onGamepadConnectedEvent, false);\n hostWindow.addEventListener(\"gamepaddisconnected\", this._onGamepadDisconnectedEvent, false);\n }\n }\n else {\n this._startMonitoringGamepads();\n }\n }\n }\n /**\n * The gamepads in the game pad manager\n */\n get gamepads() {\n return this._babylonGamepads;\n }\n /**\n * Get the gamepad controllers based on type\n * @param type The type of gamepad controller\n * @returns Nullable gamepad\n */\n getGamepadByType(type = Gamepad.XBOX) {\n for (const gamepad of this._babylonGamepads) {\n if (gamepad && gamepad.type === type) {\n return gamepad;\n }\n }\n return null;\n }\n /**\n * Disposes the gamepad manager\n */\n dispose() {\n if (this._gamepadEventSupported) {\n if (this._onGamepadConnectedEvent) {\n window.removeEventListener(\"gamepadconnected\", this._onGamepadConnectedEvent);\n }\n if (this._onGamepadDisconnectedEvent) {\n window.removeEventListener(\"gamepaddisconnected\", this._onGamepadDisconnectedEvent);\n }\n this._onGamepadConnectedEvent = null;\n this._onGamepadDisconnectedEvent = null;\n }\n this._babylonGamepads.forEach((gamepad) => {\n gamepad.dispose();\n });\n this.onGamepadConnectedObservable.clear();\n this.onGamepadDisconnectedObservable.clear();\n this._oneGamepadConnected = false;\n this._stopMonitoringGamepads();\n this._babylonGamepads = [];\n }\n _addNewGamepad(gamepad) {\n if (!this._oneGamepadConnected) {\n this._oneGamepadConnected = true;\n }\n let newGamepad;\n const dualShock = gamepad.id.search(\"054c\") !== -1 && gamepad.id.search(\"0ce6\") === -1;\n const xboxOne = gamepad.id.search(\"Xbox One\") !== -1;\n if (xboxOne ||\n gamepad.id.search(\"Xbox 360\") !== -1 ||\n gamepad.id.search(\"xinput\") !== -1 ||\n (gamepad.id.search(\"045e\") !== -1 && gamepad.id.search(\"Surface Dock\") === -1)) {\n // make sure the Surface Dock Extender is not detected as an xbox controller\n newGamepad = new Xbox360Pad(gamepad.id, gamepad.index, gamepad, xboxOne);\n }\n else if (dualShock) {\n newGamepad = new DualShockPad(gamepad.id, gamepad.index, gamepad);\n }\n // if pose is supported, use the (WebVR) pose enabled controller\n else if (gamepad.pose) {\n newGamepad = PoseEnabledControllerHelper.InitiateController(gamepad);\n }\n else {\n newGamepad = new GenericPad(gamepad.id, gamepad.index, gamepad);\n }\n this._babylonGamepads[newGamepad.index] = newGamepad;\n return newGamepad;\n }\n _startMonitoringGamepads() {\n if (!this._isMonitoring) {\n this._isMonitoring = true;\n //back-comp\n this._checkGamepadsStatus();\n }\n }\n _stopMonitoringGamepads() {\n this._isMonitoring = false;\n }\n /** @internal */\n _checkGamepadsStatus() {\n // Hack to be compatible Chrome\n this._updateGamepadObjects();\n for (const i in this._babylonGamepads) {\n const gamepad = this._babylonGamepads[i];\n if (!gamepad || !gamepad.isConnected) {\n continue;\n }\n try {\n gamepad.update();\n }\n catch (_a) {\n if (this._loggedErrors.indexOf(gamepad.index) === -1) {\n Tools.Warn(`Error updating gamepad ${gamepad.id}`);\n this._loggedErrors.push(gamepad.index);\n }\n }\n }\n if (this._isMonitoring) {\n Engine.QueueNewFrame(() => {\n this._checkGamepadsStatus();\n });\n }\n }\n // This function is called only on Chrome, which does not properly support\n // connection/disconnection events and forces you to recopy again the gamepad object\n _updateGamepadObjects() {\n const gamepads = navigator.getGamepads ? navigator.getGamepads() : [];\n for (let i = 0; i < gamepads.length; i++) {\n const gamepad = gamepads[i];\n if (gamepad) {\n if (!this._babylonGamepads[gamepad.index]) {\n const newGamepad = this._addNewGamepad(gamepad);\n this.onGamepadConnectedObservable.notifyObservers(newGamepad);\n }\n else {\n // Forced to copy again this object for Chrome for unknown reason\n this._babylonGamepads[i].browserGamepad = gamepad;\n if (!this._babylonGamepads[i].isConnected) {\n this._babylonGamepads[i]._isConnected = true;\n this.onGamepadConnectedObservable.notifyObservers(this._babylonGamepads[i]);\n }\n }\n }\n }\n }\n }\n\n /**\n * Manage the gamepad inputs to control a free camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\n */\n class FreeCameraGamepadInput {\n constructor() {\n /**\n * Defines the gamepad rotation sensibility.\n * This is the threshold from when rotation starts to be accounted for to prevent jittering.\n */\n this.gamepadAngularSensibility = 200;\n /**\n * Defines the gamepad move sensibility.\n * This is the threshold from when moving starts to be accounted for for to prevent jittering.\n */\n this.gamepadMoveSensibility = 40;\n /**\n * Defines the minimum value at which any analog stick input is ignored.\n * Note: This value should only be a value between 0 and 1.\n */\n this.deadzoneDelta = 0.1;\n this._yAxisScale = 1.0;\n this._cameraTransform = Matrix.Identity();\n this._deltaTransform = Vector3.Zero();\n this._vector3 = Vector3.Zero();\n this._vector2 = Vector2.Zero();\n }\n /**\n * Gets or sets a boolean indicating that Yaxis (for right stick) should be inverted\n */\n get invertYAxis() {\n return this._yAxisScale !== 1.0;\n }\n set invertYAxis(value) {\n this._yAxisScale = value ? -1.0 : 1.0;\n }\n /**\n * Attach the input controls to a specific dom element to get the input from.\n */\n attachControl() {\n const manager = this.camera.getScene().gamepadManager;\n this._onGamepadConnectedObserver = manager.onGamepadConnectedObservable.add((gamepad) => {\n if (gamepad.type !== Gamepad.POSE_ENABLED) {\n // prioritize XBOX gamepads.\n if (!this.gamepad || gamepad.type === Gamepad.XBOX) {\n this.gamepad = gamepad;\n }\n }\n });\n this._onGamepadDisconnectedObserver = manager.onGamepadDisconnectedObservable.add((gamepad) => {\n if (this.gamepad === gamepad) {\n this.gamepad = null;\n }\n });\n // check if there are already other controllers connected\n this.gamepad = manager.getGamepadByType(Gamepad.XBOX);\n // if no xbox controller was found, but there are gamepad controllers, take the first one\n if (!this.gamepad && manager.gamepads.length) {\n this.gamepad = manager.gamepads[0];\n }\n }\n /**\n * Detach the current controls from the specified dom element.\n */\n detachControl() {\n this.camera.getScene().gamepadManager.onGamepadConnectedObservable.remove(this._onGamepadConnectedObserver);\n this.camera.getScene().gamepadManager.onGamepadDisconnectedObservable.remove(this._onGamepadDisconnectedObserver);\n this.gamepad = null;\n }\n /**\n * Update the current camera state depending on the inputs that have been used this frame.\n * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop.\n */\n checkInputs() {\n if (this.gamepad && this.gamepad.leftStick) {\n const camera = this.camera;\n const lsValues = this.gamepad.leftStick;\n if (this.gamepadMoveSensibility !== 0) {\n lsValues.x = Math.abs(lsValues.x) > this.deadzoneDelta ? lsValues.x / this.gamepadMoveSensibility : 0;\n lsValues.y = Math.abs(lsValues.y) > this.deadzoneDelta ? lsValues.y / this.gamepadMoveSensibility : 0;\n }\n let rsValues = this.gamepad.rightStick;\n if (rsValues && this.gamepadAngularSensibility !== 0) {\n rsValues.x = Math.abs(rsValues.x) > this.deadzoneDelta ? rsValues.x / this.gamepadAngularSensibility : 0;\n rsValues.y = (Math.abs(rsValues.y) > this.deadzoneDelta ? rsValues.y / this.gamepadAngularSensibility : 0) * this._yAxisScale;\n }\n else {\n rsValues = { x: 0, y: 0 };\n }\n if (!camera.rotationQuaternion) {\n Matrix.RotationYawPitchRollToRef(camera.rotation.y, camera.rotation.x, 0, this._cameraTransform);\n }\n else {\n camera.rotationQuaternion.toRotationMatrix(this._cameraTransform);\n }\n const speed = camera._computeLocalCameraSpeed() * 50.0;\n this._vector3.copyFromFloats(lsValues.x * speed, 0, -lsValues.y * speed);\n Vector3.TransformCoordinatesToRef(this._vector3, this._cameraTransform, this._deltaTransform);\n camera.cameraDirection.addInPlace(this._deltaTransform);\n this._vector2.copyFromFloats(rsValues.y, rsValues.x);\n camera.cameraRotation.addInPlace(this._vector2);\n }\n }\n /**\n * Gets the class name of the current input.\n * @returns the class name\n */\n getClassName() {\n return \"FreeCameraGamepadInput\";\n }\n /**\n * Get the friendly name associated with the input class.\n * @returns the input friendly name\n */\n getSimpleName() {\n return \"gamepad\";\n }\n }\n __decorate$1([\n serialize()\n ], FreeCameraGamepadInput.prototype, \"gamepadAngularSensibility\", void 0);\n __decorate$1([\n serialize()\n ], FreeCameraGamepadInput.prototype, \"gamepadMoveSensibility\", void 0);\n CameraInputTypes[\"FreeCameraGamepadInput\"] = FreeCameraGamepadInput;\n\n /**\n * Manage the gamepad inputs to control an arc rotate camera.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\n */\n class ArcRotateCameraGamepadInput {\n constructor() {\n /**\n * Defines the gamepad rotation sensibility.\n * This is the threshold from when rotation starts to be accounted for to prevent jittering.\n */\n this.gamepadRotationSensibility = 80;\n /**\n * Defines the gamepad move sensibility.\n * This is the threshold from when moving starts to be accounted for for to prevent jittering.\n */\n this.gamepadMoveSensibility = 40;\n this._yAxisScale = 1.0;\n }\n /**\n * Gets or sets a boolean indicating that Yaxis (for right stick) should be inverted\n */\n get invertYAxis() {\n return this._yAxisScale !== 1.0;\n }\n set invertYAxis(value) {\n this._yAxisScale = value ? -1.0 : 1.0;\n }\n /**\n * Attach the input controls to a specific dom element to get the input from.\n */\n attachControl() {\n const manager = this.camera.getScene().gamepadManager;\n this._onGamepadConnectedObserver = manager.onGamepadConnectedObservable.add((gamepad) => {\n if (gamepad.type !== Gamepad.POSE_ENABLED) {\n // prioritize XBOX gamepads.\n if (!this.gamepad || gamepad.type === Gamepad.XBOX) {\n this.gamepad = gamepad;\n }\n }\n });\n this._onGamepadDisconnectedObserver = manager.onGamepadDisconnectedObservable.add((gamepad) => {\n if (this.gamepad === gamepad) {\n this.gamepad = null;\n }\n });\n this.gamepad = manager.getGamepadByType(Gamepad.XBOX);\n }\n /**\n * Detach the current controls from the specified dom element.\n */\n detachControl() {\n this.camera.getScene().gamepadManager.onGamepadConnectedObservable.remove(this._onGamepadConnectedObserver);\n this.camera.getScene().gamepadManager.onGamepadDisconnectedObservable.remove(this._onGamepadDisconnectedObserver);\n this.gamepad = null;\n }\n /**\n * Update the current camera state depending on the inputs that have been used this frame.\n * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop.\n */\n checkInputs() {\n if (this.gamepad) {\n const camera = this.camera;\n const rsValues = this.gamepad.rightStick;\n if (rsValues) {\n if (rsValues.x != 0) {\n const normalizedRX = rsValues.x / this.gamepadRotationSensibility;\n if (normalizedRX != 0 && Math.abs(normalizedRX) > 0.005) {\n camera.inertialAlphaOffset += normalizedRX;\n }\n }\n if (rsValues.y != 0) {\n const normalizedRY = (rsValues.y / this.gamepadRotationSensibility) * this._yAxisScale;\n if (normalizedRY != 0 && Math.abs(normalizedRY) > 0.005) {\n camera.inertialBetaOffset += normalizedRY;\n }\n }\n }\n const lsValues = this.gamepad.leftStick;\n if (lsValues && lsValues.y != 0) {\n const normalizedLY = lsValues.y / this.gamepadMoveSensibility;\n if (normalizedLY != 0 && Math.abs(normalizedLY) > 0.005) {\n this.camera.inertialRadiusOffset -= normalizedLY;\n }\n }\n }\n }\n /**\n * Gets the class name of the current intput.\n * @returns the class name\n */\n getClassName() {\n return \"ArcRotateCameraGamepadInput\";\n }\n /**\n * Get the friendly name associated with the input class.\n * @returns the input friendly name\n */\n getSimpleName() {\n return \"gamepad\";\n }\n }\n __decorate$1([\n serialize()\n ], ArcRotateCameraGamepadInput.prototype, \"gamepadRotationSensibility\", void 0);\n __decorate$1([\n serialize()\n ], ArcRotateCameraGamepadInput.prototype, \"gamepadMoveSensibility\", void 0);\n CameraInputTypes[\"ArcRotateCameraGamepadInput\"] = ArcRotateCameraGamepadInput;\n\n Object.defineProperty(Scene.prototype, \"gamepadManager\", {\n get: function () {\n if (!this._gamepadManager) {\n this._gamepadManager = new GamepadManager(this);\n let component = this._getComponent(SceneComponentConstants.NAME_GAMEPAD);\n if (!component) {\n component = new GamepadSystemSceneComponent(this);\n this._addComponent(component);\n }\n }\n return this._gamepadManager;\n },\n enumerable: true,\n configurable: true,\n });\n /**\n * Adds a gamepad to the free camera inputs manager\n */\n FreeCameraInputsManager.prototype.addGamepad = function () {\n this.add(new FreeCameraGamepadInput());\n return this;\n };\n /**\n * Adds a gamepad to the arc rotate camera inputs manager\n */\n ArcRotateCameraInputsManager.prototype.addGamepad = function () {\n this.add(new ArcRotateCameraGamepadInput());\n return this;\n };\n /**\n * Defines the gamepad scene component responsible to manage gamepads in a given scene\n */\n class GamepadSystemSceneComponent {\n /**\n * Creates a new instance of the component for the given scene\n * @param scene Defines the scene to register the component in\n */\n constructor(scene) {\n /**\n * The component name helpfull to identify the component in the list of scene components.\n */\n this.name = SceneComponentConstants.NAME_GAMEPAD;\n this.scene = scene;\n }\n /**\n * Registers the component in a given scene\n */\n register() {\n this.scene._beforeCameraUpdateStage.registerStep(SceneComponentConstants.STEP_BEFORECAMERAUPDATE_GAMEPAD, this, this._beforeCameraUpdate);\n }\n /**\n * Rebuilds the elements related to this component in case of\n * context lost for instance.\n */\n rebuild() {\n // Nothing to do for gamepads\n }\n /**\n * Disposes the component and the associated resources\n */\n dispose() {\n const gamepadManager = this.scene._gamepadManager;\n if (gamepadManager) {\n gamepadManager.dispose();\n this.scene._gamepadManager = null;\n }\n }\n _beforeCameraUpdate() {\n const gamepadManager = this.scene._gamepadManager;\n if (gamepadManager && gamepadManager._isMonitoring) {\n gamepadManager._checkGamepadsStatus();\n }\n }\n }\n\n // Static values to help the garbage collector\n // Quaternion\n const _staticOffsetValueQuaternion = Object.freeze(new Quaternion(0, 0, 0, 0));\n // Vector3\n const _staticOffsetValueVector3 = Object.freeze(Vector3.Zero());\n // Vector2\n const _staticOffsetValueVector2 = Object.freeze(Vector2.Zero());\n // Size\n const _staticOffsetValueSize = Object.freeze(Size.Zero());\n // Color3\n const _staticOffsetValueColor3 = Object.freeze(Color3.Black());\n /**\n * Defines a runtime animation\n */\n class RuntimeAnimation {\n /**\n * Gets the current frame of the runtime animation\n */\n get currentFrame() {\n return this._currentFrame;\n }\n /**\n * Gets the weight of the runtime animation\n */\n get weight() {\n return this._weight;\n }\n /**\n * Gets the current value of the runtime animation\n */\n get currentValue() {\n return this._currentValue;\n }\n /**\n * Gets or sets the target path of the runtime animation\n */\n get targetPath() {\n return this._targetPath;\n }\n /**\n * Gets the actual target of the runtime animation\n */\n get target() {\n return this._currentActiveTarget;\n }\n /**\n * Gets the additive state of the runtime animation\n */\n get isAdditive() {\n return this._host && this._host.isAdditive;\n }\n /**\n * Create a new RuntimeAnimation object\n * @param target defines the target of the animation\n * @param animation defines the source animation object\n * @param scene defines the hosting scene\n * @param host defines the initiating Animatable\n */\n constructor(target, animation, scene, host) {\n this._events = new Array();\n /**\n * The current frame of the runtime animation\n */\n this._currentFrame = 0;\n /**\n * The original value of the runtime animation\n */\n this._originalValue = new Array();\n /**\n * The original blend value of the runtime animation\n */\n this._originalBlendValue = null;\n /**\n * The offsets cache of the runtime animation\n */\n this._offsetsCache = {};\n /**\n * The high limits cache of the runtime animation\n */\n this._highLimitsCache = {};\n /**\n * Specifies if the runtime animation has been stopped\n */\n this._stopped = false;\n /**\n * The blending factor of the runtime animation\n */\n this._blendingFactor = 0;\n /**\n * The current value of the runtime animation\n */\n this._currentValue = null;\n this._currentActiveTarget = null;\n this._directTarget = null;\n /**\n * The target path of the runtime animation\n */\n this._targetPath = \"\";\n /**\n * The weight of the runtime animation\n */\n this._weight = 1.0;\n /**\n * The absolute frame offset of the runtime animation\n */\n this._absoluteFrameOffset = 0;\n /**\n * The previous elapsed time (since start of animation) of the runtime animation\n */\n this._previousElapsedTime = 0;\n /**\n * The previous absolute frame of the runtime animation (meaning, without taking into account the from/to values, only the elapsed time and the fps)\n */\n this._previousAbsoluteFrame = 0;\n this._targetIsArray = false;\n this._animation = animation;\n this._target = target;\n this._scene = scene;\n this._host = host;\n this._activeTargets = [];\n animation._runtimeAnimations.push(this);\n // State\n this._animationState = {\n key: 0,\n repeatCount: 0,\n loopMode: this._getCorrectLoopMode(),\n };\n if (this._animation.dataType === Animation.ANIMATIONTYPE_MATRIX) {\n this._animationState.workValue = Matrix.Zero();\n }\n // Limits\n this._keys = this._animation.getKeys();\n this._minFrame = this._keys[0].frame;\n this._maxFrame = this._keys[this._keys.length - 1].frame;\n this._minValue = this._keys[0].value;\n this._maxValue = this._keys[this._keys.length - 1].value;\n // Add a start key at frame 0 if missing\n if (this._minFrame !== 0) {\n const newKey = { frame: 0, value: this._minValue };\n this._keys.splice(0, 0, newKey);\n }\n // Check data\n if (this._target instanceof Array) {\n let index = 0;\n for (const target of this._target) {\n this._preparePath(target, index);\n this._getOriginalValues(index);\n index++;\n }\n this._targetIsArray = true;\n }\n else {\n this._preparePath(this._target);\n this._getOriginalValues();\n this._targetIsArray = false;\n this._directTarget = this._activeTargets[0];\n }\n // Cloning events locally\n const events = animation.getEvents();\n if (events && events.length > 0) {\n events.forEach((e) => {\n this._events.push(e._clone());\n });\n }\n this._enableBlending = target && target.animationPropertiesOverride ? target.animationPropertiesOverride.enableBlending : this._animation.enableBlending;\n }\n _preparePath(target, targetIndex = 0) {\n const targetPropertyPath = this._animation.targetPropertyPath;\n if (targetPropertyPath.length > 1) {\n let property = target[targetPropertyPath[0]];\n for (let index = 1; index < targetPropertyPath.length - 1; index++) {\n property = property[targetPropertyPath[index]];\n }\n this._targetPath = targetPropertyPath[targetPropertyPath.length - 1];\n this._activeTargets[targetIndex] = property;\n }\n else {\n this._targetPath = targetPropertyPath[0];\n this._activeTargets[targetIndex] = target;\n }\n }\n /**\n * Gets the animation from the runtime animation\n */\n get animation() {\n return this._animation;\n }\n /**\n * Resets the runtime animation to the beginning\n * @param restoreOriginal defines whether to restore the target property to the original value\n */\n reset(restoreOriginal = false) {\n if (restoreOriginal) {\n if (this._target instanceof Array) {\n let index = 0;\n for (const target of this._target) {\n if (this._originalValue[index] !== undefined) {\n this._setValue(target, this._activeTargets[index], this._originalValue[index], -1, index);\n }\n index++;\n }\n }\n else {\n if (this._originalValue[0] !== undefined) {\n this._setValue(this._target, this._directTarget, this._originalValue[0], -1, 0);\n }\n }\n }\n this._offsetsCache = {};\n this._highLimitsCache = {};\n this._currentFrame = 0;\n this._blendingFactor = 0;\n // Events\n for (let index = 0; index < this._events.length; index++) {\n this._events[index].isDone = false;\n }\n }\n /**\n * Specifies if the runtime animation is stopped\n * @returns Boolean specifying if the runtime animation is stopped\n */\n isStopped() {\n return this._stopped;\n }\n /**\n * Disposes of the runtime animation\n */\n dispose() {\n const index = this._animation.runtimeAnimations.indexOf(this);\n if (index > -1) {\n this._animation.runtimeAnimations.splice(index, 1);\n }\n }\n /**\n * Apply the interpolated value to the target\n * @param currentValue defines the value computed by the animation\n * @param weight defines the weight to apply to this value (Defaults to 1.0)\n */\n setValue(currentValue, weight) {\n if (this._targetIsArray) {\n for (let index = 0; index < this._target.length; index++) {\n const target = this._target[index];\n this._setValue(target, this._activeTargets[index], currentValue, weight, index);\n }\n return;\n }\n this._setValue(this._target, this._directTarget, currentValue, weight, 0);\n }\n _getOriginalValues(targetIndex = 0) {\n let originalValue;\n const target = this._activeTargets[targetIndex];\n if (target.getRestPose && this._targetPath === \"_matrix\") {\n // For bones\n originalValue = target.getRestPose();\n }\n else {\n originalValue = target[this._targetPath];\n }\n if (originalValue && originalValue.clone) {\n this._originalValue[targetIndex] = originalValue.clone();\n }\n else {\n this._originalValue[targetIndex] = originalValue;\n }\n }\n _setValue(target, destination, currentValue, weight, targetIndex) {\n // Set value\n this._currentActiveTarget = destination;\n this._weight = weight;\n if (this._enableBlending && this._blendingFactor <= 1.0) {\n if (!this._originalBlendValue) {\n const originalValue = destination[this._targetPath];\n if (originalValue.clone) {\n this._originalBlendValue = originalValue.clone();\n }\n else {\n this._originalBlendValue = originalValue;\n }\n }\n if (this._originalBlendValue.m) {\n // Matrix\n if (Animation.AllowMatrixDecomposeForInterpolation) {\n if (this._currentValue) {\n Matrix.DecomposeLerpToRef(this._originalBlendValue, currentValue, this._blendingFactor, this._currentValue);\n }\n else {\n this._currentValue = Matrix.DecomposeLerp(this._originalBlendValue, currentValue, this._blendingFactor);\n }\n }\n else {\n if (this._currentValue) {\n Matrix.LerpToRef(this._originalBlendValue, currentValue, this._blendingFactor, this._currentValue);\n }\n else {\n this._currentValue = Matrix.Lerp(this._originalBlendValue, currentValue, this._blendingFactor);\n }\n }\n }\n else {\n this._currentValue = Animation._UniversalLerp(this._originalBlendValue, currentValue, this._blendingFactor);\n }\n const blendingSpeed = target && target.animationPropertiesOverride ? target.animationPropertiesOverride.blendingSpeed : this._animation.blendingSpeed;\n this._blendingFactor += blendingSpeed;\n }\n else {\n if (!this._currentValue) {\n if (currentValue === null || currentValue === void 0 ? void 0 : currentValue.clone) {\n this._currentValue = currentValue.clone();\n }\n else {\n this._currentValue = currentValue;\n }\n }\n else if (this._currentValue.copyFrom) {\n this._currentValue.copyFrom(currentValue);\n }\n else {\n this._currentValue = currentValue;\n }\n }\n if (weight !== -1.0) {\n this._scene._registerTargetForLateAnimationBinding(this, this._originalValue[targetIndex]);\n }\n else {\n destination[this._targetPath] = this._currentValue;\n }\n if (target.markAsDirty) {\n target.markAsDirty(this._animation.targetProperty);\n }\n }\n /**\n * Gets the loop pmode of the runtime animation\n * @returns Loop Mode\n */\n _getCorrectLoopMode() {\n if (this._target && this._target.animationPropertiesOverride) {\n return this._target.animationPropertiesOverride.loopMode;\n }\n return this._animation.loopMode;\n }\n /**\n * Move the current animation to a given frame\n * @param frame defines the frame to move to\n */\n goToFrame(frame) {\n const keys = this._animation.getKeys();\n if (frame < keys[0].frame) {\n frame = keys[0].frame;\n }\n else if (frame > keys[keys.length - 1].frame) {\n frame = keys[keys.length - 1].frame;\n }\n // Need to reset animation events\n const events = this._events;\n if (events.length) {\n for (let index = 0; index < events.length; index++) {\n if (!events[index].onlyOnce) {\n // reset events in the future\n events[index].isDone = events[index].frame < frame;\n }\n }\n }\n this._currentFrame = frame;\n const currentValue = this._animation._interpolate(frame, this._animationState);\n this.setValue(currentValue, -1);\n }\n /**\n * @internal Internal use only\n */\n _prepareForSpeedRatioChange(newSpeedRatio) {\n const newAbsoluteFrame = (this._previousElapsedTime * (this._animation.framePerSecond * newSpeedRatio)) / 1000.0;\n this._absoluteFrameOffset = this._previousAbsoluteFrame - newAbsoluteFrame;\n }\n /**\n * Execute the current animation\n * @param elapsedTimeSinceAnimationStart defines the elapsed time (in milliseconds) since the animation was started\n * @param from defines the lower frame of the animation range\n * @param to defines the upper frame of the animation range\n * @param loop defines if the current animation must loop\n * @param speedRatio defines the current speed ratio\n * @param weight defines the weight of the animation (default is -1 so no weight)\n * @returns a boolean indicating if the animation is running\n */\n animate(elapsedTimeSinceAnimationStart, from, to, loop, speedRatio, weight = -1.0) {\n const animation = this._animation;\n const targetPropertyPath = animation.targetPropertyPath;\n if (!targetPropertyPath || targetPropertyPath.length < 1) {\n this._stopped = true;\n return false;\n }\n let returnValue = true;\n // Check limits\n if (from < this._minFrame || from > this._maxFrame) {\n from = this._minFrame;\n }\n if (to < this._minFrame || to > this._maxFrame) {\n to = this._maxFrame;\n }\n const frameRange = to - from;\n let offsetValue;\n // Compute the frame according to the elapsed time and the fps of the animation (\"from\" and \"to\" are not factored in!)\n let absoluteFrame = (elapsedTimeSinceAnimationStart * (animation.framePerSecond * speedRatio)) / 1000.0 + this._absoluteFrameOffset;\n let highLimitValue = 0;\n // Apply the yoyo function if required\n if (loop && this._animationState.loopMode === Animation.ANIMATIONLOOPMODE_YOYO) {\n const position = (absoluteFrame - from) / frameRange;\n // Apply the yoyo curve\n const yoyoPosition = Math.abs(Math.sin(position * Math.PI));\n // Map the yoyo position back to the range\n absoluteFrame = yoyoPosition * frameRange + from;\n }\n this._previousElapsedTime = elapsedTimeSinceAnimationStart;\n this._previousAbsoluteFrame = absoluteFrame;\n if (!loop && to >= from && absoluteFrame >= frameRange) {\n // If we are out of range and not looping get back to caller\n returnValue = false;\n highLimitValue = animation._getKeyValue(this._maxValue);\n }\n else if (!loop && from >= to && absoluteFrame <= frameRange) {\n returnValue = false;\n highLimitValue = animation._getKeyValue(this._minValue);\n }\n else if (this._animationState.loopMode !== Animation.ANIMATIONLOOPMODE_CYCLE) {\n const keyOffset = to.toString() + from.toString();\n if (!this._offsetsCache[keyOffset]) {\n this._animationState.repeatCount = 0;\n this._animationState.loopMode = Animation.ANIMATIONLOOPMODE_CYCLE; // force a specific codepath in animation._interpolate()!\n const fromValue = animation._interpolate(from, this._animationState);\n const toValue = animation._interpolate(to, this._animationState);\n this._animationState.loopMode = this._getCorrectLoopMode();\n switch (animation.dataType) {\n // Float\n case Animation.ANIMATIONTYPE_FLOAT:\n this._offsetsCache[keyOffset] = toValue - fromValue;\n break;\n // Quaternion\n case Animation.ANIMATIONTYPE_QUATERNION:\n this._offsetsCache[keyOffset] = toValue.subtract(fromValue);\n break;\n // Vector3\n case Animation.ANIMATIONTYPE_VECTOR3:\n this._offsetsCache[keyOffset] = toValue.subtract(fromValue);\n break;\n // Vector2\n case Animation.ANIMATIONTYPE_VECTOR2:\n this._offsetsCache[keyOffset] = toValue.subtract(fromValue);\n break;\n // Size\n case Animation.ANIMATIONTYPE_SIZE:\n this._offsetsCache[keyOffset] = toValue.subtract(fromValue);\n break;\n // Color3\n case Animation.ANIMATIONTYPE_COLOR3:\n this._offsetsCache[keyOffset] = toValue.subtract(fromValue);\n break;\n }\n this._highLimitsCache[keyOffset] = toValue;\n }\n highLimitValue = this._highLimitsCache[keyOffset];\n offsetValue = this._offsetsCache[keyOffset];\n }\n if (offsetValue === undefined) {\n switch (animation.dataType) {\n // Float\n case Animation.ANIMATIONTYPE_FLOAT:\n offsetValue = 0;\n break;\n // Quaternion\n case Animation.ANIMATIONTYPE_QUATERNION:\n offsetValue = _staticOffsetValueQuaternion;\n break;\n // Vector3\n case Animation.ANIMATIONTYPE_VECTOR3:\n offsetValue = _staticOffsetValueVector3;\n break;\n // Vector2\n case Animation.ANIMATIONTYPE_VECTOR2:\n offsetValue = _staticOffsetValueVector2;\n break;\n // Size\n case Animation.ANIMATIONTYPE_SIZE:\n offsetValue = _staticOffsetValueSize;\n break;\n // Color3\n case Animation.ANIMATIONTYPE_COLOR3:\n offsetValue = _staticOffsetValueColor3;\n }\n }\n // Compute value\n let currentFrame;\n if (this._host && this._host.syncRoot) {\n // If we must sync with an animatable, calculate the current frame based on the frame of the root animatable\n const syncRoot = this._host.syncRoot;\n const hostNormalizedFrame = (syncRoot.masterFrame - syncRoot.fromFrame) / (syncRoot.toFrame - syncRoot.fromFrame);\n currentFrame = from + frameRange * hostNormalizedFrame;\n }\n else {\n if ((absoluteFrame > 0 && from > to) || (absoluteFrame < 0 && from < to)) {\n currentFrame = returnValue && frameRange !== 0 ? to + (absoluteFrame % frameRange) : from;\n }\n else {\n currentFrame = returnValue && frameRange !== 0 ? from + (absoluteFrame % frameRange) : to;\n }\n }\n const events = this._events;\n // Reset event/state if looping\n if ((speedRatio > 0 && this.currentFrame > currentFrame) || (speedRatio < 0 && this.currentFrame < currentFrame)) {\n this._onLoop();\n // Need to reset animation events\n for (let index = 0; index < events.length; index++) {\n if (!events[index].onlyOnce) {\n // reset event, the animation is looping\n events[index].isDone = false;\n }\n }\n this._animationState.key = speedRatio > 0 ? 0 : animation.getKeys().length - 1;\n }\n this._currentFrame = currentFrame;\n this._animationState.repeatCount = frameRange === 0 ? 0 : (absoluteFrame / frameRange) >> 0;\n this._animationState.highLimitValue = highLimitValue;\n this._animationState.offsetValue = offsetValue;\n const currentValue = animation._interpolate(currentFrame, this._animationState);\n // Set value\n this.setValue(currentValue, weight);\n // Check events\n if (events.length) {\n for (let index = 0; index < events.length; index++) {\n // Make sure current frame has passed event frame and that event frame is within the current range\n // Also, handle both forward and reverse animations\n if ((frameRange > 0 && currentFrame >= events[index].frame && events[index].frame >= from) ||\n (frameRange < 0 && currentFrame <= events[index].frame && events[index].frame <= from)) {\n const event = events[index];\n if (!event.isDone) {\n // If event should be done only once, remove it.\n if (event.onlyOnce) {\n events.splice(index, 1);\n index--;\n }\n event.isDone = true;\n event.action(currentFrame);\n } // Don't do anything if the event has already been done.\n }\n }\n }\n if (!returnValue) {\n this._stopped = true;\n }\n return returnValue;\n }\n }\n\n /**\n * Class used to store bone information\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/bonesSkeletons\n */\n class Bone extends Node {\n /** @internal */\n get _matrix() {\n this._compose();\n return this._localMatrix;\n }\n /** @internal */\n set _matrix(value) {\n // skip if the matrices are the same\n if (value.updateFlag === this._localMatrix.updateFlag && !this._needToCompose) {\n return;\n }\n this._needToCompose = false; // in case there was a pending compose\n this._localMatrix.copyFrom(value);\n this._markAsDirtyAndDecompose();\n }\n /**\n * Create a new bone\n * @param name defines the bone name\n * @param skeleton defines the parent skeleton\n * @param parentBone defines the parent (can be null if the bone is the root)\n * @param localMatrix defines the local matrix (default: identity)\n * @param restMatrix defines the rest matrix (default: localMatrix)\n * @param bindMatrix defines the bind matrix (default: localMatrix)\n * @param index defines index of the bone in the hierarchy (default: null)\n */\n constructor(\n /**\n * defines the bone name\n */\n name, skeleton, parentBone = null, localMatrix = null, restMatrix = null, bindMatrix = null, index = null) {\n var _a;\n super(name, skeleton.getScene());\n this.name = name;\n /**\n * Gets the list of child bones\n */\n this.children = new Array();\n /** Gets the animations associated with this bone */\n this.animations = new Array();\n /**\n * @internal Internal only\n * Set this value to map this bone to a different index in the transform matrices\n * Set this value to -1 to exclude the bone from the transform matrices\n */\n this._index = null;\n this._scalingDeterminant = 1;\n this._needToDecompose = true;\n this._needToCompose = false;\n /** @internal */\n this._linkedTransformNode = null;\n /** @internal */\n this._waitingTransformNodeId = null;\n this._skeleton = skeleton;\n this._localMatrix = (_a = localMatrix === null || localMatrix === void 0 ? void 0 : localMatrix.clone()) !== null && _a !== void 0 ? _a : Matrix.Identity();\n this._restMatrix = restMatrix !== null && restMatrix !== void 0 ? restMatrix : this._localMatrix.clone();\n this._bindMatrix = bindMatrix !== null && bindMatrix !== void 0 ? bindMatrix : this._localMatrix.clone();\n this._index = index;\n this._absoluteMatrix = new Matrix();\n this._absoluteBindMatrix = new Matrix();\n this._absoluteInverseBindMatrix = new Matrix();\n this._finalMatrix = new Matrix();\n skeleton.bones.push(this);\n this.setParent(parentBone, false);\n this._updateAbsoluteBindMatrices();\n }\n /**\n * Gets the current object class name.\n * @returns the class name\n */\n getClassName() {\n return \"Bone\";\n }\n // Members\n /**\n * Gets the parent skeleton\n * @returns a skeleton\n */\n getSkeleton() {\n return this._skeleton;\n }\n get parent() {\n return this._parentNode;\n }\n /**\n * Gets parent bone\n * @returns a bone or null if the bone is the root of the bone hierarchy\n */\n getParent() {\n return this.parent;\n }\n /**\n * Returns an array containing the children of the bone\n * @returns an array containing the children of the bone (can be empty if the bone has no children)\n */\n getChildren() {\n return this.children;\n }\n /**\n * Gets the node index in matrix array generated for rendering\n * @returns the node index\n */\n getIndex() {\n return this._index === null ? this.getSkeleton().bones.indexOf(this) : this._index;\n }\n set parent(newParent) {\n this.setParent(newParent);\n }\n /**\n * Sets the parent bone\n * @param parent defines the parent (can be null if the bone is the root)\n * @param updateAbsoluteBindMatrices defines if the absolute bind and absolute inverse bind matrices must be updated\n */\n setParent(parent, updateAbsoluteBindMatrices = true) {\n if (this.parent === parent) {\n return;\n }\n if (this.parent) {\n const index = this.parent.children.indexOf(this);\n if (index !== -1) {\n this.parent.children.splice(index, 1);\n }\n }\n this._parentNode = parent;\n if (this.parent) {\n this.parent.children.push(this);\n }\n if (updateAbsoluteBindMatrices) {\n this._updateAbsoluteBindMatrices();\n }\n this.markAsDirty();\n }\n /**\n * Gets the local matrix\n * @returns the local matrix\n */\n getLocalMatrix() {\n this._compose();\n return this._localMatrix;\n }\n /**\n * Gets the bind matrix\n * @returns the bind matrix\n */\n getBindMatrix() {\n return this._bindMatrix;\n }\n /**\n * Gets the bind matrix.\n * @returns the bind matrix\n * @deprecated Please use getBindMatrix instead\n */\n getBaseMatrix() {\n return this.getBindMatrix();\n }\n /**\n * Gets the rest matrix\n * @returns the rest matrix\n */\n getRestMatrix() {\n return this._restMatrix;\n }\n /**\n * Gets the rest matrix\n * @returns the rest matrix\n * @deprecated Please use getRestMatrix instead\n */\n getRestPose() {\n return this.getRestMatrix();\n }\n /**\n * Sets the rest matrix\n * @param matrix the local-space rest matrix to set for this bone\n */\n setRestMatrix(matrix) {\n this._restMatrix.copyFrom(matrix);\n }\n /**\n * Sets the rest matrix\n * @param matrix the local-space rest to set for this bone\n * @deprecated Please use setRestMatrix instead\n */\n setRestPose(matrix) {\n this.setRestMatrix(matrix);\n }\n /**\n * Gets the bind matrix\n * @returns the bind matrix\n * @deprecated Please use getBindMatrix instead\n */\n getBindPose() {\n return this.getBindMatrix();\n }\n /**\n * Sets the bind matrix\n * This will trigger a recomputation of the absolute bind and absolute inverse bind matrices for this bone and its children\n * Note that the local matrix will also be set with the matrix passed in parameter!\n * @param matrix the local-space bind matrix to set for this bone\n */\n setBindMatrix(matrix) {\n this.updateMatrix(matrix);\n }\n /**\n * Sets the bind matrix\n * @param matrix the local-space bind to set for this bone\n * @deprecated Please use setBindMatrix instead\n */\n setBindPose(matrix) {\n this.setBindMatrix(matrix);\n }\n /**\n * Gets the matrix used to store the final world transformation of the bone (ie. the matrix sent to shaders)\n */\n getFinalMatrix() {\n return this._finalMatrix;\n }\n /**\n * Gets the matrix used to store the final world transformation of the bone (ie. the matrix sent to shaders)\n * @deprecated Please use getFinalMatrix instead\n */\n getWorldMatrix() {\n return this.getFinalMatrix();\n }\n /**\n * Sets the local matrix to the rest matrix\n */\n returnToRest() {\n var _a;\n if (this._linkedTransformNode) {\n const localScaling = TmpVectors.Vector3[0];\n const localRotation = TmpVectors.Quaternion[0];\n const localPosition = TmpVectors.Vector3[1];\n this.getRestMatrix().decompose(localScaling, localRotation, localPosition);\n this._linkedTransformNode.position.copyFrom(localPosition);\n this._linkedTransformNode.rotationQuaternion = (_a = this._linkedTransformNode.rotationQuaternion) !== null && _a !== void 0 ? _a : Quaternion.Identity();\n this._linkedTransformNode.rotationQuaternion.copyFrom(localRotation);\n this._linkedTransformNode.scaling.copyFrom(localScaling);\n }\n else {\n this._matrix = this._restMatrix;\n }\n }\n /**\n * Gets the inverse of the bind matrix, in world space (relative to the skeleton root)\n * @returns the inverse bind matrix, in world space\n */\n getAbsoluteInverseBindMatrix() {\n return this._absoluteInverseBindMatrix;\n }\n /**\n * Gets the inverse of the bind matrix, in world space (relative to the skeleton root)\n * @returns the inverse bind matrix, in world space\n * @deprecated Please use getAbsoluteInverseBindMatrix instead\n */\n getInvertedAbsoluteTransform() {\n return this.getAbsoluteInverseBindMatrix();\n }\n /**\n * Gets the bone matrix, in world space (relative to the skeleton root)\n * @returns the bone matrix, in world space\n */\n getAbsoluteMatrix() {\n return this._absoluteMatrix;\n }\n /**\n * Gets the bone matrix, in world space (relative to the skeleton root)\n * @returns the bone matrix, in world space\n * @deprecated Please use getAbsoluteMatrix instead\n */\n getAbsoluteTransform() {\n return this._absoluteMatrix;\n }\n /**\n * Links with the given transform node.\n * The local matrix of this bone is overwritten by the transform of the node every frame.\n * @param transformNode defines the transform node to link to\n */\n linkTransformNode(transformNode) {\n if (this._linkedTransformNode) {\n this._skeleton._numBonesWithLinkedTransformNode--;\n }\n this._linkedTransformNode = transformNode;\n if (this._linkedTransformNode) {\n this._skeleton._numBonesWithLinkedTransformNode++;\n }\n }\n // Properties (matches TransformNode properties)\n /**\n * Gets the node used to drive the bone's transformation\n * @returns a transform node or null\n */\n getTransformNode() {\n return this._linkedTransformNode;\n }\n /** Gets or sets current position (in local space) */\n get position() {\n this._decompose();\n return this._localPosition;\n }\n set position(newPosition) {\n this._decompose();\n this._localPosition.copyFrom(newPosition);\n this._markAsDirtyAndCompose();\n }\n /** Gets or sets current rotation (in local space) */\n get rotation() {\n return this.getRotation();\n }\n set rotation(newRotation) {\n this.setRotation(newRotation);\n }\n /** Gets or sets current rotation quaternion (in local space) */\n get rotationQuaternion() {\n this._decompose();\n return this._localRotation;\n }\n set rotationQuaternion(newRotation) {\n this.setRotationQuaternion(newRotation);\n }\n /** Gets or sets current scaling (in local space) */\n get scaling() {\n return this.getScale();\n }\n set scaling(newScaling) {\n this.setScale(newScaling);\n }\n /**\n * Gets the animation properties override\n */\n get animationPropertiesOverride() {\n return this._skeleton.animationPropertiesOverride;\n }\n // Methods\n _decompose() {\n if (!this._needToDecompose) {\n return;\n }\n this._needToDecompose = false;\n if (!this._localScaling) {\n this._localScaling = Vector3.Zero();\n this._localRotation = Quaternion.Zero();\n this._localPosition = Vector3.Zero();\n }\n this._localMatrix.decompose(this._localScaling, this._localRotation, this._localPosition);\n }\n _compose() {\n if (!this._needToCompose) {\n return;\n }\n if (!this._localScaling) {\n this._needToCompose = false;\n return;\n }\n this._needToCompose = false;\n Matrix.ComposeToRef(this._localScaling, this._localRotation, this._localPosition, this._localMatrix);\n }\n /**\n * Update the bind (and optionally the local) matrix\n * @param bindMatrix defines the new matrix to set to the bind/local matrix, in local space\n * @param updateAbsoluteBindMatrices defines if the absolute bind and absolute inverse bind matrices must be recomputed (default: true)\n * @param updateLocalMatrix defines if the local matrix should also be updated with the matrix passed in parameter (default: true)\n */\n updateMatrix(bindMatrix, updateAbsoluteBindMatrices = true, updateLocalMatrix = true) {\n this._bindMatrix.copyFrom(bindMatrix);\n if (updateAbsoluteBindMatrices) {\n this._updateAbsoluteBindMatrices();\n }\n if (updateLocalMatrix) {\n this._matrix = bindMatrix;\n }\n else {\n this.markAsDirty();\n }\n }\n /**\n * @internal\n */\n _updateAbsoluteBindMatrices(bindMatrix, updateChildren = true) {\n if (!bindMatrix) {\n bindMatrix = this._bindMatrix;\n }\n if (this.parent) {\n bindMatrix.multiplyToRef(this.parent._absoluteBindMatrix, this._absoluteBindMatrix);\n }\n else {\n this._absoluteBindMatrix.copyFrom(bindMatrix);\n }\n this._absoluteBindMatrix.invertToRef(this._absoluteInverseBindMatrix);\n if (updateChildren) {\n for (let index = 0; index < this.children.length; index++) {\n this.children[index]._updateAbsoluteBindMatrices();\n }\n }\n this._scalingDeterminant = this._absoluteBindMatrix.determinant() < 0 ? -1 : 1;\n }\n /**\n * Flag the bone as dirty (Forcing it to update everything)\n * @returns this bone\n */\n markAsDirty() {\n this._currentRenderId++;\n this._childUpdateId++;\n this._skeleton._markAsDirty();\n return this;\n }\n /** @internal */\n _markAsDirtyAndCompose() {\n this.markAsDirty();\n this._needToCompose = true;\n }\n _markAsDirtyAndDecompose() {\n this.markAsDirty();\n this._needToDecompose = true;\n }\n _updatePosition(vec, space = Space.LOCAL, tNode, translationMode = true) {\n const lm = this.getLocalMatrix();\n if (space == Space.LOCAL) {\n if (translationMode) {\n lm.addAtIndex(12, vec.x);\n lm.addAtIndex(13, vec.y);\n lm.addAtIndex(14, vec.z);\n }\n else {\n lm.setTranslationFromFloats(vec.x, vec.y, vec.z);\n }\n }\n else {\n let wm = null;\n //tNode.getWorldMatrix() needs to be called before skeleton.computeAbsoluteMatrices()\n if (tNode) {\n wm = tNode.getWorldMatrix();\n }\n this._skeleton.computeAbsoluteMatrices();\n const tmat = Bone._TmpMats[0];\n const tvec = Bone._TmpVecs[0];\n if (this.parent) {\n if (tNode && wm) {\n tmat.copyFrom(this.parent.getAbsoluteMatrix());\n tmat.multiplyToRef(wm, tmat);\n }\n else {\n tmat.copyFrom(this.parent.getAbsoluteMatrix());\n }\n }\n else {\n Matrix.IdentityToRef(tmat);\n }\n if (translationMode) {\n tmat.setTranslationFromFloats(0, 0, 0);\n }\n tmat.invert();\n Vector3.TransformCoordinatesToRef(vec, tmat, tvec);\n if (translationMode) {\n lm.addAtIndex(12, tvec.x);\n lm.addAtIndex(13, tvec.y);\n lm.addAtIndex(14, tvec.z);\n }\n else {\n lm.setTranslationFromFloats(tvec.x, tvec.y, tvec.z);\n }\n }\n this._markAsDirtyAndDecompose();\n }\n /**\n * Translate the bone in local or world space\n * @param vec The amount to translate the bone\n * @param space The space that the translation is in (default: Space.LOCAL)\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n */\n translate(vec, space = Space.LOCAL, tNode) {\n this._updatePosition(vec, space, tNode, true);\n }\n /**\n * Set the position of the bone in local or world space\n * @param position The position to set the bone\n * @param space The space that the position is in (default: Space.LOCAL)\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n */\n setPosition(position, space = Space.LOCAL, tNode) {\n this._updatePosition(position, space, tNode, false);\n }\n /**\n * Set the absolute position of the bone (world space)\n * @param position The position to set the bone\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n */\n setAbsolutePosition(position, tNode) {\n this.setPosition(position, Space.WORLD, tNode);\n }\n /**\n * Scale the bone on the x, y and z axes (in local space)\n * @param x The amount to scale the bone on the x axis\n * @param y The amount to scale the bone on the y axis\n * @param z The amount to scale the bone on the z axis\n * @param scaleChildren sets this to true if children of the bone should be scaled as well (false by default)\n */\n scale(x, y, z, scaleChildren = false) {\n const locMat = this.getLocalMatrix();\n // Apply new scaling on top of current local matrix\n const scaleMat = Bone._TmpMats[0];\n Matrix.ScalingToRef(x, y, z, scaleMat);\n scaleMat.multiplyToRef(locMat, locMat);\n // Invert scaling matrix and apply the inverse to all children\n scaleMat.invert();\n for (const child of this.children) {\n const cm = child.getLocalMatrix();\n cm.multiplyToRef(scaleMat, cm);\n cm.multiplyAtIndex(12, x);\n cm.multiplyAtIndex(13, y);\n cm.multiplyAtIndex(14, z);\n child._markAsDirtyAndDecompose();\n }\n this._markAsDirtyAndDecompose();\n if (scaleChildren) {\n for (const child of this.children) {\n child.scale(x, y, z, scaleChildren);\n }\n }\n }\n /**\n * Set the bone scaling in local space\n * @param scale defines the scaling vector\n */\n setScale(scale) {\n this._decompose();\n this._localScaling.copyFrom(scale);\n this._markAsDirtyAndCompose();\n }\n /**\n * Gets the current scaling in local space\n * @returns the current scaling vector\n */\n getScale() {\n this._decompose();\n return this._localScaling;\n }\n /**\n * Gets the current scaling in local space and stores it in a target vector\n * @param result defines the target vector\n */\n getScaleToRef(result) {\n this._decompose();\n result.copyFrom(this._localScaling);\n }\n /**\n * Set the yaw, pitch, and roll of the bone in local or world space\n * @param yaw The rotation of the bone on the y axis\n * @param pitch The rotation of the bone on the x axis\n * @param roll The rotation of the bone on the z axis\n * @param space The space that the axes of rotation are in\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n */\n setYawPitchRoll(yaw, pitch, roll, space = Space.LOCAL, tNode) {\n if (space === Space.LOCAL) {\n const quat = Bone._TmpQuat;\n Quaternion.RotationYawPitchRollToRef(yaw, pitch, roll, quat);\n this.setRotationQuaternion(quat, space, tNode);\n return;\n }\n const rotMatInv = Bone._TmpMats[0];\n if (!this._getAbsoluteInverseMatrixUnscaledToRef(rotMatInv, tNode)) {\n return;\n }\n const rotMat = Bone._TmpMats[1];\n Matrix.RotationYawPitchRollToRef(yaw, pitch, roll, rotMat);\n rotMatInv.multiplyToRef(rotMat, rotMat);\n this._rotateWithMatrix(rotMat, space, tNode);\n }\n /**\n * Add a rotation to the bone on an axis in local or world space\n * @param axis The axis to rotate the bone on\n * @param amount The amount to rotate the bone\n * @param space The space that the axis is in\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n */\n rotate(axis, amount, space = Space.LOCAL, tNode) {\n const rmat = Bone._TmpMats[0];\n rmat.setTranslationFromFloats(0, 0, 0);\n Matrix.RotationAxisToRef(axis, amount, rmat);\n this._rotateWithMatrix(rmat, space, tNode);\n }\n /**\n * Set the rotation of the bone to a particular axis angle in local or world space\n * @param axis The axis to rotate the bone on\n * @param angle The angle that the bone should be rotated to\n * @param space The space that the axis is in\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n */\n setAxisAngle(axis, angle, space = Space.LOCAL, tNode) {\n if (space === Space.LOCAL) {\n const quat = Bone._TmpQuat;\n Quaternion.RotationAxisToRef(axis, angle, quat);\n this.setRotationQuaternion(quat, space, tNode);\n return;\n }\n const rotMatInv = Bone._TmpMats[0];\n if (!this._getAbsoluteInverseMatrixUnscaledToRef(rotMatInv, tNode)) {\n return;\n }\n const rotMat = Bone._TmpMats[1];\n Matrix.RotationAxisToRef(axis, angle, rotMat);\n rotMatInv.multiplyToRef(rotMat, rotMat);\n this._rotateWithMatrix(rotMat, space, tNode);\n }\n /**\n * Set the euler rotation of the bone in local or world space\n * @param rotation The euler rotation that the bone should be set to\n * @param space The space that the rotation is in\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n */\n setRotation(rotation, space = Space.LOCAL, tNode) {\n this.setYawPitchRoll(rotation.y, rotation.x, rotation.z, space, tNode);\n }\n /**\n * Set the quaternion rotation of the bone in local or world space\n * @param quat The quaternion rotation that the bone should be set to\n * @param space The space that the rotation is in\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n */\n setRotationQuaternion(quat, space = Space.LOCAL, tNode) {\n if (space === Space.LOCAL) {\n this._decompose();\n this._localRotation.copyFrom(quat);\n this._markAsDirtyAndCompose();\n return;\n }\n const rotMatInv = Bone._TmpMats[0];\n if (!this._getAbsoluteInverseMatrixUnscaledToRef(rotMatInv, tNode)) {\n return;\n }\n const rotMat = Bone._TmpMats[1];\n Matrix.FromQuaternionToRef(quat, rotMat);\n rotMatInv.multiplyToRef(rotMat, rotMat);\n this._rotateWithMatrix(rotMat, space, tNode);\n }\n /**\n * Set the rotation matrix of the bone in local or world space\n * @param rotMat The rotation matrix that the bone should be set to\n * @param space The space that the rotation is in\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n */\n setRotationMatrix(rotMat, space = Space.LOCAL, tNode) {\n if (space === Space.LOCAL) {\n const quat = Bone._TmpQuat;\n Quaternion.FromRotationMatrixToRef(rotMat, quat);\n this.setRotationQuaternion(quat, space, tNode);\n return;\n }\n const rotMatInv = Bone._TmpMats[0];\n if (!this._getAbsoluteInverseMatrixUnscaledToRef(rotMatInv, tNode)) {\n return;\n }\n const rotMat2 = Bone._TmpMats[1];\n rotMat2.copyFrom(rotMat);\n rotMatInv.multiplyToRef(rotMat, rotMat2);\n this._rotateWithMatrix(rotMat2, space, tNode);\n }\n _rotateWithMatrix(rmat, space = Space.LOCAL, tNode) {\n const lmat = this.getLocalMatrix();\n const lx = lmat.m[12];\n const ly = lmat.m[13];\n const lz = lmat.m[14];\n const parent = this.getParent();\n const parentScale = Bone._TmpMats[3];\n const parentScaleInv = Bone._TmpMats[4];\n if (parent && space == Space.WORLD) {\n if (tNode) {\n parentScale.copyFrom(tNode.getWorldMatrix());\n parent.getAbsoluteMatrix().multiplyToRef(parentScale, parentScale);\n }\n else {\n parentScale.copyFrom(parent.getAbsoluteMatrix());\n }\n parentScaleInv.copyFrom(parentScale);\n parentScaleInv.invert();\n lmat.multiplyToRef(parentScale, lmat);\n lmat.multiplyToRef(rmat, lmat);\n lmat.multiplyToRef(parentScaleInv, lmat);\n }\n else {\n if (space == Space.WORLD && tNode) {\n parentScale.copyFrom(tNode.getWorldMatrix());\n parentScaleInv.copyFrom(parentScale);\n parentScaleInv.invert();\n lmat.multiplyToRef(parentScale, lmat);\n lmat.multiplyToRef(rmat, lmat);\n lmat.multiplyToRef(parentScaleInv, lmat);\n }\n else {\n lmat.multiplyToRef(rmat, lmat);\n }\n }\n lmat.setTranslationFromFloats(lx, ly, lz);\n this.computeAbsoluteMatrices();\n this._markAsDirtyAndDecompose();\n }\n _getAbsoluteInverseMatrixUnscaledToRef(rotMatInv, tNode) {\n const scaleMatrix = Bone._TmpMats[2];\n rotMatInv.copyFrom(this.getAbsoluteMatrix());\n if (tNode) {\n rotMatInv.multiplyToRef(tNode.getWorldMatrix(), rotMatInv);\n Matrix.ScalingToRef(tNode.scaling.x, tNode.scaling.y, tNode.scaling.z, scaleMatrix);\n }\n else {\n Matrix.IdentityToRef(scaleMatrix);\n }\n rotMatInv.invert();\n if (isNaN(rotMatInv.m[0])) {\n // Matrix failed to invert.\n // This can happen if scale is zero for example.\n return false;\n }\n scaleMatrix.multiplyAtIndex(0, this._scalingDeterminant);\n rotMatInv.multiplyToRef(scaleMatrix, rotMatInv);\n return true;\n }\n /**\n * Get the position of the bone in local or world space\n * @param space The space that the returned position is in\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n * @returns The position of the bone\n */\n getPosition(space = Space.LOCAL, tNode = null) {\n const pos = Vector3.Zero();\n this.getPositionToRef(space, tNode, pos);\n return pos;\n }\n /**\n * Copy the position of the bone to a vector3 in local or world space\n * @param space The space that the returned position is in\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n * @param result The vector3 to copy the position to\n */\n getPositionToRef(space = Space.LOCAL, tNode, result) {\n if (space == Space.LOCAL) {\n const lm = this.getLocalMatrix();\n result.x = lm.m[12];\n result.y = lm.m[13];\n result.z = lm.m[14];\n }\n else {\n let wm = null;\n //tNode.getWorldMatrix() needs to be called before skeleton.computeAbsoluteMatrices()\n if (tNode) {\n wm = tNode.getWorldMatrix();\n }\n this._skeleton.computeAbsoluteMatrices();\n let tmat = Bone._TmpMats[0];\n if (tNode && wm) {\n tmat.copyFrom(this.getAbsoluteMatrix());\n tmat.multiplyToRef(wm, tmat);\n }\n else {\n tmat = this.getAbsoluteMatrix();\n }\n result.x = tmat.m[12];\n result.y = tmat.m[13];\n result.z = tmat.m[14];\n }\n }\n /**\n * Get the absolute position of the bone (world space)\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n * @returns The absolute position of the bone\n */\n getAbsolutePosition(tNode = null) {\n const pos = Vector3.Zero();\n this.getPositionToRef(Space.WORLD, tNode, pos);\n return pos;\n }\n /**\n * Copy the absolute position of the bone (world space) to the result param\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n * @param result The vector3 to copy the absolute position to\n */\n getAbsolutePositionToRef(tNode, result) {\n this.getPositionToRef(Space.WORLD, tNode, result);\n }\n /**\n * Compute the absolute matrices of this bone and its children\n */\n computeAbsoluteMatrices() {\n this._compose();\n if (this.parent) {\n this._localMatrix.multiplyToRef(this.parent._absoluteMatrix, this._absoluteMatrix);\n }\n else {\n this._absoluteMatrix.copyFrom(this._localMatrix);\n const poseMatrix = this._skeleton.getPoseMatrix();\n if (poseMatrix) {\n this._absoluteMatrix.multiplyToRef(poseMatrix, this._absoluteMatrix);\n }\n }\n const children = this.children;\n const len = children.length;\n for (let i = 0; i < len; i++) {\n children[i].computeAbsoluteMatrices();\n }\n }\n /**\n * Compute the absolute matrices of this bone and its children\n * @deprecated Please use computeAbsoluteMatrices instead\n */\n computeAbsoluteTransforms() {\n this.computeAbsoluteMatrices();\n }\n /**\n * Get the world direction from an axis that is in the local space of the bone\n * @param localAxis The local direction that is used to compute the world direction\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n * @returns The world direction\n */\n getDirection(localAxis, tNode = null) {\n const result = Vector3.Zero();\n this.getDirectionToRef(localAxis, tNode, result);\n return result;\n }\n /**\n * Copy the world direction to a vector3 from an axis that is in the local space of the bone\n * @param localAxis The local direction that is used to compute the world direction\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n * @param result The vector3 that the world direction will be copied to\n */\n getDirectionToRef(localAxis, tNode = null, result) {\n let wm = null;\n //tNode.getWorldMatrix() needs to be called before skeleton.computeAbsoluteMatrices()\n if (tNode) {\n wm = tNode.getWorldMatrix();\n }\n this._skeleton.computeAbsoluteMatrices();\n const mat = Bone._TmpMats[0];\n mat.copyFrom(this.getAbsoluteMatrix());\n if (tNode && wm) {\n mat.multiplyToRef(wm, mat);\n }\n Vector3.TransformNormalToRef(localAxis, mat, result);\n result.normalize();\n }\n /**\n * Get the euler rotation of the bone in local or world space\n * @param space The space that the rotation should be in\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n * @returns The euler rotation\n */\n getRotation(space = Space.LOCAL, tNode = null) {\n const result = Vector3.Zero();\n this.getRotationToRef(space, tNode, result);\n return result;\n }\n /**\n * Copy the euler rotation of the bone to a vector3. The rotation can be in either local or world space\n * @param space The space that the rotation should be in\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n * @param result The vector3 that the rotation should be copied to\n */\n getRotationToRef(space = Space.LOCAL, tNode = null, result) {\n const quat = Bone._TmpQuat;\n this.getRotationQuaternionToRef(space, tNode, quat);\n quat.toEulerAnglesToRef(result);\n }\n /**\n * Get the quaternion rotation of the bone in either local or world space\n * @param space The space that the rotation should be in\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n * @returns The quaternion rotation\n */\n getRotationQuaternion(space = Space.LOCAL, tNode = null) {\n const result = Quaternion.Identity();\n this.getRotationQuaternionToRef(space, tNode, result);\n return result;\n }\n /**\n * Copy the quaternion rotation of the bone to a quaternion. The rotation can be in either local or world space\n * @param space The space that the rotation should be in\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n * @param result The quaternion that the rotation should be copied to\n */\n getRotationQuaternionToRef(space = Space.LOCAL, tNode = null, result) {\n if (space == Space.LOCAL) {\n this._decompose();\n result.copyFrom(this._localRotation);\n }\n else {\n const mat = Bone._TmpMats[0];\n const amat = this.getAbsoluteMatrix();\n if (tNode) {\n amat.multiplyToRef(tNode.getWorldMatrix(), mat);\n }\n else {\n mat.copyFrom(amat);\n }\n mat.multiplyAtIndex(0, this._scalingDeterminant);\n mat.multiplyAtIndex(1, this._scalingDeterminant);\n mat.multiplyAtIndex(2, this._scalingDeterminant);\n mat.decompose(undefined, result, undefined);\n }\n }\n /**\n * Get the rotation matrix of the bone in local or world space\n * @param space The space that the rotation should be in\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n * @returns The rotation matrix\n */\n getRotationMatrix(space = Space.LOCAL, tNode) {\n const result = Matrix.Identity();\n this.getRotationMatrixToRef(space, tNode, result);\n return result;\n }\n /**\n * Copy the rotation matrix of the bone to a matrix. The rotation can be in either local or world space\n * @param space The space that the rotation should be in\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n * @param result The quaternion that the rotation should be copied to\n */\n getRotationMatrixToRef(space = Space.LOCAL, tNode, result) {\n if (space == Space.LOCAL) {\n this.getLocalMatrix().getRotationMatrixToRef(result);\n }\n else {\n const mat = Bone._TmpMats[0];\n const amat = this.getAbsoluteMatrix();\n if (tNode) {\n amat.multiplyToRef(tNode.getWorldMatrix(), mat);\n }\n else {\n mat.copyFrom(amat);\n }\n mat.multiplyAtIndex(0, this._scalingDeterminant);\n mat.multiplyAtIndex(1, this._scalingDeterminant);\n mat.multiplyAtIndex(2, this._scalingDeterminant);\n mat.getRotationMatrixToRef(result);\n }\n }\n /**\n * Get the world position of a point that is in the local space of the bone\n * @param position The local position\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n * @returns The world position\n */\n getAbsolutePositionFromLocal(position, tNode = null) {\n const result = Vector3.Zero();\n this.getAbsolutePositionFromLocalToRef(position, tNode, result);\n return result;\n }\n /**\n * Get the world position of a point that is in the local space of the bone and copy it to the result param\n * @param position The local position\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n * @param result The vector3 that the world position should be copied to\n */\n getAbsolutePositionFromLocalToRef(position, tNode = null, result) {\n let wm = null;\n //tNode.getWorldMatrix() needs to be called before skeleton.computeAbsoluteMatrices()\n if (tNode) {\n wm = tNode.getWorldMatrix();\n }\n this._skeleton.computeAbsoluteMatrices();\n const tmat = Bone._TmpMats[0];\n tmat.copyFrom(this.getAbsoluteMatrix());\n if (tNode && wm) {\n tmat.multiplyToRef(wm, tmat);\n }\n Vector3.TransformCoordinatesToRef(position, tmat, result);\n }\n /**\n * Get the local position of a point that is in world space\n * @param position The world position\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n * @returns The local position\n */\n getLocalPositionFromAbsolute(position, tNode = null) {\n const result = Vector3.Zero();\n this.getLocalPositionFromAbsoluteToRef(position, tNode, result);\n return result;\n }\n /**\n * Get the local position of a point that is in world space and copy it to the result param\n * @param position The world position\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\n * @param result The vector3 that the local position should be copied to\n */\n getLocalPositionFromAbsoluteToRef(position, tNode = null, result) {\n let wm = null;\n //tNode.getWorldMatrix() needs to be called before skeleton.computeAbsoluteMatrices()\n if (tNode) {\n wm = tNode.getWorldMatrix();\n }\n this._skeleton.computeAbsoluteMatrices();\n const tmat = Bone._TmpMats[0];\n tmat.copyFrom(this.getAbsoluteMatrix());\n if (tNode && wm) {\n tmat.multiplyToRef(wm, tmat);\n }\n tmat.invert();\n Vector3.TransformCoordinatesToRef(position, tmat, result);\n }\n /**\n * Set the current local matrix as the restMatrix for this bone.\n */\n setCurrentPoseAsRest() {\n this.setRestMatrix(this.getLocalMatrix());\n }\n }\n Bone._TmpVecs = ArrayTools.BuildArray(2, Vector3.Zero);\n Bone._TmpQuat = Quaternion.Identity();\n Bone._TmpMats = ArrayTools.BuildArray(5, Matrix.Identity);\n\n /**\n * Class used to store an actual running animation\n */\n class Animatable {\n /**\n * Gets the root Animatable used to synchronize and normalize animations\n */\n get syncRoot() {\n return this._syncRoot;\n }\n /**\n * Gets the current frame of the first RuntimeAnimation\n * Used to synchronize Animatables\n */\n get masterFrame() {\n if (this._runtimeAnimations.length === 0) {\n return 0;\n }\n return this._runtimeAnimations[0].currentFrame;\n }\n /**\n * Gets or sets the animatable weight (-1.0 by default meaning not weighted)\n */\n get weight() {\n return this._weight;\n }\n set weight(value) {\n if (value === -1) {\n // -1 is ok and means no weight\n this._weight = -1;\n return;\n }\n // Else weight must be in [0, 1] range\n this._weight = Math.min(Math.max(value, 0), 1.0);\n }\n /**\n * Gets or sets the speed ratio to apply to the animatable (1.0 by default)\n */\n get speedRatio() {\n return this._speedRatio;\n }\n set speedRatio(value) {\n for (let index = 0; index < this._runtimeAnimations.length; index++) {\n const animation = this._runtimeAnimations[index];\n animation._prepareForSpeedRatioChange(value);\n }\n this._speedRatio = value;\n // Resync _manualJumpDelay in case goToFrame was called before speedRatio was set.\n if (this._goToFrame !== null) {\n this.goToFrame(this._goToFrame);\n }\n }\n /**\n * Creates a new Animatable\n * @param scene defines the hosting scene\n * @param target defines the target object\n * @param fromFrame defines the starting frame number (default is 0)\n * @param toFrame defines the ending frame number (default is 100)\n * @param loopAnimation defines if the animation must loop (default is false)\n * @param speedRatio defines the factor to apply to animation speed (default is 1)\n * @param onAnimationEnd defines a callback to call when animation ends if it is not looping\n * @param animations defines a group of animation to add to the new Animatable\n * @param onAnimationLoop defines a callback to call when animation loops\n * @param isAdditive defines whether the animation should be evaluated additively\n * @param playOrder defines the order in which this animatable should be processed in the list of active animatables (default: 0)\n */\n constructor(scene, \n /** defines the target object */\n target, \n /** defines the starting frame number (default is 0) */\n fromFrame = 0, \n /** defines the ending frame number (default is 100) */\n toFrame = 100, \n /** defines if the animation must loop (default is false) */\n loopAnimation = false, speedRatio = 1.0, \n /** defines a callback to call when animation ends if it is not looping */\n onAnimationEnd, animations, \n /** defines a callback to call when animation loops */\n onAnimationLoop, \n /** defines whether the animation should be evaluated additively */\n isAdditive = false, \n /** defines the order in which this animatable should be processed in the list of active animatables (default: 0) */\n playOrder = 0) {\n this.target = target;\n this.fromFrame = fromFrame;\n this.toFrame = toFrame;\n this.loopAnimation = loopAnimation;\n this.onAnimationEnd = onAnimationEnd;\n this.onAnimationLoop = onAnimationLoop;\n this.isAdditive = isAdditive;\n this.playOrder = playOrder;\n this._localDelayOffset = null;\n this._pausedDelay = null;\n this._manualJumpDelay = null;\n /** @hidden */\n this._runtimeAnimations = new Array();\n this._paused = false;\n this._speedRatio = 1;\n this._weight = -1.0;\n this._syncRoot = null;\n this._frameToSyncFromJump = null;\n this._goToFrame = null;\n /**\n * Gets or sets a boolean indicating if the animatable must be disposed and removed at the end of the animation.\n * This will only apply for non looping animation (default is true)\n */\n this.disposeOnEnd = true;\n /**\n * Gets a boolean indicating if the animation has started\n */\n this.animationStarted = false;\n /**\n * Observer raised when the animation ends\n */\n this.onAnimationEndObservable = new Observable$1();\n /**\n * Observer raised when the animation loops\n */\n this.onAnimationLoopObservable = new Observable$1();\n this._scene = scene;\n if (animations) {\n this.appendAnimations(target, animations);\n }\n this._speedRatio = speedRatio;\n scene._activeAnimatables.push(this);\n }\n // Methods\n /**\n * Synchronize and normalize current Animatable with a source Animatable\n * This is useful when using animation weights and when animations are not of the same length\n * @param root defines the root Animatable to synchronize with (null to stop synchronizing)\n * @returns the current Animatable\n */\n syncWith(root) {\n this._syncRoot = root;\n if (root) {\n // Make sure this animatable will animate after the root\n const index = this._scene._activeAnimatables.indexOf(this);\n if (index > -1) {\n this._scene._activeAnimatables.splice(index, 1);\n this._scene._activeAnimatables.push(this);\n }\n }\n return this;\n }\n /**\n * Gets the list of runtime animations\n * @returns an array of RuntimeAnimation\n */\n getAnimations() {\n return this._runtimeAnimations;\n }\n /**\n * Adds more animations to the current animatable\n * @param target defines the target of the animations\n * @param animations defines the new animations to add\n */\n appendAnimations(target, animations) {\n for (let index = 0; index < animations.length; index++) {\n const animation = animations[index];\n const newRuntimeAnimation = new RuntimeAnimation(target, animation, this._scene, this);\n newRuntimeAnimation._onLoop = () => {\n this.onAnimationLoopObservable.notifyObservers(this);\n if (this.onAnimationLoop) {\n this.onAnimationLoop();\n }\n };\n this._runtimeAnimations.push(newRuntimeAnimation);\n }\n }\n /**\n * Gets the source animation for a specific property\n * @param property defines the property to look for\n * @returns null or the source animation for the given property\n */\n getAnimationByTargetProperty(property) {\n const runtimeAnimations = this._runtimeAnimations;\n for (let index = 0; index < runtimeAnimations.length; index++) {\n if (runtimeAnimations[index].animation.targetProperty === property) {\n return runtimeAnimations[index].animation;\n }\n }\n return null;\n }\n /**\n * Gets the runtime animation for a specific property\n * @param property defines the property to look for\n * @returns null or the runtime animation for the given property\n */\n getRuntimeAnimationByTargetProperty(property) {\n const runtimeAnimations = this._runtimeAnimations;\n for (let index = 0; index < runtimeAnimations.length; index++) {\n if (runtimeAnimations[index].animation.targetProperty === property) {\n return runtimeAnimations[index];\n }\n }\n return null;\n }\n /**\n * Resets the animatable to its original state\n */\n reset() {\n const runtimeAnimations = this._runtimeAnimations;\n for (let index = 0; index < runtimeAnimations.length; index++) {\n runtimeAnimations[index].reset(true);\n }\n this._localDelayOffset = null;\n this._pausedDelay = null;\n }\n /**\n * Allows the animatable to blend with current running animations\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#animation-blending\n * @param blendingSpeed defines the blending speed to use\n */\n enableBlending(blendingSpeed) {\n const runtimeAnimations = this._runtimeAnimations;\n for (let index = 0; index < runtimeAnimations.length; index++) {\n runtimeAnimations[index].animation.enableBlending = true;\n runtimeAnimations[index].animation.blendingSpeed = blendingSpeed;\n }\n }\n /**\n * Disable animation blending\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#animation-blending\n */\n disableBlending() {\n const runtimeAnimations = this._runtimeAnimations;\n for (let index = 0; index < runtimeAnimations.length; index++) {\n runtimeAnimations[index].animation.enableBlending = false;\n }\n }\n /**\n * Jump directly to a given frame\n * @param frame defines the frame to jump to\n */\n goToFrame(frame) {\n var _a;\n const runtimeAnimations = this._runtimeAnimations;\n if (runtimeAnimations[0]) {\n const fps = runtimeAnimations[0].animation.framePerSecond;\n this._frameToSyncFromJump = (_a = this._frameToSyncFromJump) !== null && _a !== void 0 ? _a : runtimeAnimations[0].currentFrame;\n const delay = this.speedRatio === 0 ? 0 : (((frame - this._frameToSyncFromJump) / fps) * 1000) / this.speedRatio;\n this._manualJumpDelay = -delay;\n }\n for (let index = 0; index < runtimeAnimations.length; index++) {\n runtimeAnimations[index].goToFrame(frame);\n }\n this._goToFrame = frame;\n }\n /**\n * Pause the animation\n */\n pause() {\n if (this._paused) {\n return;\n }\n this._paused = true;\n }\n /**\n * Restart the animation\n */\n restart() {\n this._paused = false;\n }\n _raiseOnAnimationEnd() {\n if (this.onAnimationEnd) {\n this.onAnimationEnd();\n }\n this.onAnimationEndObservable.notifyObservers(this);\n }\n /**\n * Stop and delete the current animation\n * @param animationName defines a string used to only stop some of the runtime animations instead of all\n * @param targetMask a function that determines if the animation should be stopped based on its target (all animations will be stopped if both this and animationName are empty)\n * @param useGlobalSplice if true, the animatables will be removed by the caller of this function (false by default)\n */\n stop(animationName, targetMask, useGlobalSplice = false) {\n if (animationName || targetMask) {\n const idx = this._scene._activeAnimatables.indexOf(this);\n if (idx > -1) {\n const runtimeAnimations = this._runtimeAnimations;\n for (let index = runtimeAnimations.length - 1; index >= 0; index--) {\n const runtimeAnimation = runtimeAnimations[index];\n if (animationName && runtimeAnimation.animation.name != animationName) {\n continue;\n }\n if (targetMask && !targetMask(runtimeAnimation.target)) {\n continue;\n }\n runtimeAnimation.dispose();\n runtimeAnimations.splice(index, 1);\n }\n if (runtimeAnimations.length == 0) {\n if (!useGlobalSplice) {\n this._scene._activeAnimatables.splice(idx, 1);\n }\n this._raiseOnAnimationEnd();\n }\n }\n }\n else {\n const index = this._scene._activeAnimatables.indexOf(this);\n if (index > -1) {\n if (!useGlobalSplice) {\n this._scene._activeAnimatables.splice(index, 1);\n }\n const runtimeAnimations = this._runtimeAnimations;\n for (let index = 0; index < runtimeAnimations.length; index++) {\n runtimeAnimations[index].dispose();\n }\n this._runtimeAnimations.length = 0;\n this._raiseOnAnimationEnd();\n }\n }\n }\n /**\n * Wait asynchronously for the animation to end\n * @returns a promise which will be fulfilled when the animation ends\n */\n waitAsync() {\n return new Promise((resolve) => {\n this.onAnimationEndObservable.add(() => {\n resolve(this);\n }, undefined, undefined, this, true);\n });\n }\n /**\n * @internal\n */\n _animate(delay) {\n if (this._paused) {\n this.animationStarted = false;\n if (this._pausedDelay === null) {\n this._pausedDelay = delay;\n }\n return true;\n }\n if (this._localDelayOffset === null) {\n this._localDelayOffset = delay;\n this._pausedDelay = null;\n }\n else if (this._pausedDelay !== null) {\n this._localDelayOffset += delay - this._pausedDelay;\n this._pausedDelay = null;\n }\n if (this._manualJumpDelay !== null) {\n this._localDelayOffset += this._manualJumpDelay;\n this._manualJumpDelay = null;\n this._frameToSyncFromJump = null;\n }\n this._goToFrame = null;\n if (this._weight === 0) {\n // We consider that an animation with a weight === 0 is \"actively\" paused\n return true;\n }\n // Animating\n let running = false;\n const runtimeAnimations = this._runtimeAnimations;\n let index;\n for (index = 0; index < runtimeAnimations.length; index++) {\n const animation = runtimeAnimations[index];\n const isRunning = animation.animate(delay - this._localDelayOffset, this.fromFrame, this.toFrame, this.loopAnimation, this._speedRatio, this._weight);\n running = running || isRunning;\n }\n this.animationStarted = running;\n if (!running) {\n if (this.disposeOnEnd) {\n // Remove from active animatables\n index = this._scene._activeAnimatables.indexOf(this);\n this._scene._activeAnimatables.splice(index, 1);\n // Dispose all runtime animations\n for (index = 0; index < runtimeAnimations.length; index++) {\n runtimeAnimations[index].dispose();\n }\n }\n this._raiseOnAnimationEnd();\n if (this.disposeOnEnd) {\n this.onAnimationEnd = null;\n this.onAnimationLoop = null;\n this.onAnimationLoopObservable.clear();\n this.onAnimationEndObservable.clear();\n }\n }\n return running;\n }\n }\n Scene.prototype._animate = function () {\n if (!this.animationsEnabled) {\n return;\n }\n // Getting time\n const now = PrecisionDate.Now;\n if (!this._animationTimeLast) {\n if (this._pendingData.length > 0) {\n return;\n }\n this._animationTimeLast = now;\n }\n this.deltaTime = this.useConstantAnimationDeltaTime ? 16.0 : (now - this._animationTimeLast) * this.animationTimeScale;\n this._animationTimeLast = now;\n const animatables = this._activeAnimatables;\n if (animatables.length === 0) {\n return;\n }\n this._animationTime += this.deltaTime;\n const animationTime = this._animationTime;\n for (let index = 0; index < animatables.length; index++) {\n const animatable = animatables[index];\n if (!animatable._animate(animationTime) && animatable.disposeOnEnd) {\n index--; // Array was updated\n }\n }\n // Late animation bindings\n this._processLateAnimationBindings();\n };\n Scene.prototype.sortActiveAnimatables = function () {\n this._activeAnimatables.sort((a, b) => {\n return a.playOrder - b.playOrder;\n });\n };\n Scene.prototype.beginWeightedAnimation = function (target, from, to, weight = 1.0, loop, speedRatio = 1.0, onAnimationEnd, animatable, targetMask, onAnimationLoop, isAdditive = false) {\n const returnedAnimatable = this.beginAnimation(target, from, to, loop, speedRatio, onAnimationEnd, animatable, false, targetMask, onAnimationLoop, isAdditive);\n returnedAnimatable.weight = weight;\n return returnedAnimatable;\n };\n Scene.prototype.beginAnimation = function (target, from, to, loop, speedRatio = 1.0, onAnimationEnd, animatable, stopCurrent = true, targetMask, onAnimationLoop, isAdditive = false) {\n if (from > to && speedRatio > 0) {\n speedRatio *= -1;\n }\n if (stopCurrent) {\n this.stopAnimation(target, undefined, targetMask);\n }\n if (!animatable) {\n animatable = new Animatable(this, target, from, to, loop, speedRatio, onAnimationEnd, undefined, onAnimationLoop, isAdditive);\n }\n const shouldRunTargetAnimations = targetMask ? targetMask(target) : true;\n // Local animations\n if (target.animations && shouldRunTargetAnimations) {\n animatable.appendAnimations(target, target.animations);\n }\n // Children animations\n if (target.getAnimatables) {\n const animatables = target.getAnimatables();\n for (let index = 0; index < animatables.length; index++) {\n this.beginAnimation(animatables[index], from, to, loop, speedRatio, onAnimationEnd, animatable, stopCurrent, targetMask, onAnimationLoop);\n }\n }\n animatable.reset();\n return animatable;\n };\n Scene.prototype.beginHierarchyAnimation = function (target, directDescendantsOnly, from, to, loop, speedRatio = 1.0, onAnimationEnd, animatable, stopCurrent = true, targetMask, onAnimationLoop, isAdditive = false) {\n const children = target.getDescendants(directDescendantsOnly);\n const result = [];\n result.push(this.beginAnimation(target, from, to, loop, speedRatio, onAnimationEnd, animatable, stopCurrent, targetMask, undefined, isAdditive));\n for (const child of children) {\n result.push(this.beginAnimation(child, from, to, loop, speedRatio, onAnimationEnd, animatable, stopCurrent, targetMask, undefined, isAdditive));\n }\n return result;\n };\n Scene.prototype.beginDirectAnimation = function (target, animations, from, to, loop, speedRatio, onAnimationEnd, onAnimationLoop, isAdditive = false) {\n if (speedRatio === undefined) {\n speedRatio = 1.0;\n }\n if (from > to && speedRatio > 0) {\n speedRatio *= -1;\n }\n else if (to > from && speedRatio < 0) {\n const temp = to;\n to = from;\n from = temp;\n }\n const animatable = new Animatable(this, target, from, to, loop, speedRatio, onAnimationEnd, animations, onAnimationLoop, isAdditive);\n return animatable;\n };\n Scene.prototype.beginDirectHierarchyAnimation = function (target, directDescendantsOnly, animations, from, to, loop, speedRatio, onAnimationEnd, onAnimationLoop, isAdditive = false) {\n const children = target.getDescendants(directDescendantsOnly);\n const result = [];\n result.push(this.beginDirectAnimation(target, animations, from, to, loop, speedRatio, onAnimationEnd, onAnimationLoop, isAdditive));\n for (const child of children) {\n result.push(this.beginDirectAnimation(child, animations, from, to, loop, speedRatio, onAnimationEnd, onAnimationLoop, isAdditive));\n }\n return result;\n };\n Scene.prototype.getAnimatableByTarget = function (target) {\n for (let index = 0; index < this._activeAnimatables.length; index++) {\n if (this._activeAnimatables[index].target === target) {\n return this._activeAnimatables[index];\n }\n }\n return null;\n };\n Scene.prototype.getAllAnimatablesByTarget = function (target) {\n const result = [];\n for (let index = 0; index < this._activeAnimatables.length; index++) {\n if (this._activeAnimatables[index].target === target) {\n result.push(this._activeAnimatables[index]);\n }\n }\n return result;\n };\n /**\n * Will stop the animation of the given target\n * @param target - the target\n * @param animationName - the name of the animation to stop (all animations will be stopped if both this and targetMask are empty)\n * @param targetMask - a function that determines if the animation should be stopped based on its target (all animations will be stopped if both this and animationName are empty)\n */\n Scene.prototype.stopAnimation = function (target, animationName, targetMask) {\n const animatables = this.getAllAnimatablesByTarget(target);\n for (const animatable of animatables) {\n animatable.stop(animationName, targetMask);\n }\n };\n /**\n * Stops and removes all animations that have been applied to the scene\n */\n Scene.prototype.stopAllAnimations = function () {\n if (this._activeAnimatables) {\n for (let i = 0; i < this._activeAnimatables.length; i++) {\n this._activeAnimatables[i].stop(undefined, undefined, true);\n }\n this._activeAnimatables.length = 0;\n }\n for (const group of this.animationGroups) {\n group.stop();\n }\n };\n Scene.prototype._registerTargetForLateAnimationBinding = function (runtimeAnimation, originalValue) {\n const target = runtimeAnimation.target;\n this._registeredForLateAnimationBindings.pushNoDuplicate(target);\n if (!target._lateAnimationHolders) {\n target._lateAnimationHolders = {};\n }\n if (!target._lateAnimationHolders[runtimeAnimation.targetPath]) {\n target._lateAnimationHolders[runtimeAnimation.targetPath] = {\n totalWeight: 0,\n totalAdditiveWeight: 0,\n animations: [],\n additiveAnimations: [],\n originalValue: originalValue,\n };\n }\n if (runtimeAnimation.isAdditive) {\n target._lateAnimationHolders[runtimeAnimation.targetPath].additiveAnimations.push(runtimeAnimation);\n target._lateAnimationHolders[runtimeAnimation.targetPath].totalAdditiveWeight += runtimeAnimation.weight;\n }\n else {\n target._lateAnimationHolders[runtimeAnimation.targetPath].animations.push(runtimeAnimation);\n target._lateAnimationHolders[runtimeAnimation.targetPath].totalWeight += runtimeAnimation.weight;\n }\n };\n Scene.prototype._processLateAnimationBindingsForMatrices = function (holder) {\n if (holder.totalWeight === 0 && holder.totalAdditiveWeight === 0) {\n return holder.originalValue;\n }\n let normalizer = 1.0;\n const finalPosition = TmpVectors.Vector3[0];\n const finalScaling = TmpVectors.Vector3[1];\n const finalQuaternion = TmpVectors.Quaternion[0];\n let startIndex = 0;\n const originalAnimation = holder.animations[0];\n const originalValue = holder.originalValue;\n let scale = 1;\n let skipOverride = false;\n if (holder.totalWeight < 1.0) {\n // We need to mix the original value in\n scale = 1.0 - holder.totalWeight;\n originalValue.decompose(finalScaling, finalQuaternion, finalPosition);\n }\n else {\n startIndex = 1;\n // We need to normalize the weights\n normalizer = holder.totalWeight;\n scale = originalAnimation.weight / normalizer;\n if (scale == 1) {\n if (holder.totalAdditiveWeight) {\n skipOverride = true;\n }\n else {\n return originalAnimation.currentValue;\n }\n }\n originalAnimation.currentValue.decompose(finalScaling, finalQuaternion, finalPosition);\n }\n // Add up the override animations\n if (!skipOverride) {\n finalScaling.scaleInPlace(scale);\n finalPosition.scaleInPlace(scale);\n finalQuaternion.scaleInPlace(scale);\n for (let animIndex = startIndex; animIndex < holder.animations.length; animIndex++) {\n const runtimeAnimation = holder.animations[animIndex];\n if (runtimeAnimation.weight === 0) {\n continue;\n }\n scale = runtimeAnimation.weight / normalizer;\n const currentPosition = TmpVectors.Vector3[2];\n const currentScaling = TmpVectors.Vector3[3];\n const currentQuaternion = TmpVectors.Quaternion[1];\n runtimeAnimation.currentValue.decompose(currentScaling, currentQuaternion, currentPosition);\n currentScaling.scaleAndAddToRef(scale, finalScaling);\n currentQuaternion.scaleAndAddToRef(Quaternion.Dot(finalQuaternion, currentQuaternion) > 0 ? scale : -scale, finalQuaternion);\n currentPosition.scaleAndAddToRef(scale, finalPosition);\n }\n finalQuaternion.normalize();\n }\n // Add up the additive animations\n for (let animIndex = 0; animIndex < holder.additiveAnimations.length; animIndex++) {\n const runtimeAnimation = holder.additiveAnimations[animIndex];\n if (runtimeAnimation.weight === 0) {\n continue;\n }\n const currentPosition = TmpVectors.Vector3[2];\n const currentScaling = TmpVectors.Vector3[3];\n const currentQuaternion = TmpVectors.Quaternion[1];\n runtimeAnimation.currentValue.decompose(currentScaling, currentQuaternion, currentPosition);\n currentScaling.multiplyToRef(finalScaling, currentScaling);\n Vector3.LerpToRef(finalScaling, currentScaling, runtimeAnimation.weight, finalScaling);\n finalQuaternion.multiplyToRef(currentQuaternion, currentQuaternion);\n Quaternion.SlerpToRef(finalQuaternion, currentQuaternion, runtimeAnimation.weight, finalQuaternion);\n currentPosition.scaleAndAddToRef(runtimeAnimation.weight, finalPosition);\n }\n const workValue = originalAnimation ? originalAnimation._animationState.workValue : TmpVectors.Matrix[0].clone();\n Matrix.ComposeToRef(finalScaling, finalQuaternion, finalPosition, workValue);\n return workValue;\n };\n Scene.prototype._processLateAnimationBindingsForQuaternions = function (holder, refQuaternion) {\n if (holder.totalWeight === 0 && holder.totalAdditiveWeight === 0) {\n return refQuaternion;\n }\n const originalAnimation = holder.animations[0];\n const originalValue = holder.originalValue;\n let cumulativeQuaternion = refQuaternion;\n if (holder.totalWeight === 0 && holder.totalAdditiveWeight > 0) {\n cumulativeQuaternion.copyFrom(originalValue);\n }\n else if (holder.animations.length === 1) {\n Quaternion.SlerpToRef(originalValue, originalAnimation.currentValue, Math.min(1.0, holder.totalWeight), cumulativeQuaternion);\n if (holder.totalAdditiveWeight === 0) {\n return cumulativeQuaternion;\n }\n }\n else if (holder.animations.length > 1) {\n // Add up the override animations\n let normalizer = 1.0;\n let quaternions;\n let weights;\n if (holder.totalWeight < 1.0) {\n const scale = 1.0 - holder.totalWeight;\n quaternions = [];\n weights = [];\n quaternions.push(originalValue);\n weights.push(scale);\n }\n else {\n if (holder.animations.length === 2) {\n // Slerp as soon as we can\n Quaternion.SlerpToRef(holder.animations[0].currentValue, holder.animations[1].currentValue, holder.animations[1].weight / holder.totalWeight, refQuaternion);\n if (holder.totalAdditiveWeight === 0) {\n return refQuaternion;\n }\n }\n quaternions = [];\n weights = [];\n normalizer = holder.totalWeight;\n }\n for (let animIndex = 0; animIndex < holder.animations.length; animIndex++) {\n const runtimeAnimation = holder.animations[animIndex];\n quaternions.push(runtimeAnimation.currentValue);\n weights.push(runtimeAnimation.weight / normalizer);\n }\n // https://gamedev.stackexchange.com/questions/62354/method-for-interpolation-between-3-quaternions\n let cumulativeAmount = 0;\n for (let index = 0; index < quaternions.length;) {\n if (!index) {\n Quaternion.SlerpToRef(quaternions[index], quaternions[index + 1], weights[index + 1] / (weights[index] + weights[index + 1]), refQuaternion);\n cumulativeQuaternion = refQuaternion;\n cumulativeAmount = weights[index] + weights[index + 1];\n index += 2;\n continue;\n }\n cumulativeAmount += weights[index];\n Quaternion.SlerpToRef(cumulativeQuaternion, quaternions[index], weights[index] / cumulativeAmount, cumulativeQuaternion);\n index++;\n }\n }\n // Add up the additive animations\n for (let animIndex = 0; animIndex < holder.additiveAnimations.length; animIndex++) {\n const runtimeAnimation = holder.additiveAnimations[animIndex];\n if (runtimeAnimation.weight === 0) {\n continue;\n }\n cumulativeQuaternion.multiplyToRef(runtimeAnimation.currentValue, TmpVectors.Quaternion[0]);\n Quaternion.SlerpToRef(cumulativeQuaternion, TmpVectors.Quaternion[0], runtimeAnimation.weight, cumulativeQuaternion);\n }\n return cumulativeQuaternion;\n };\n Scene.prototype._processLateAnimationBindings = function () {\n if (!this._registeredForLateAnimationBindings.length) {\n return;\n }\n for (let index = 0; index < this._registeredForLateAnimationBindings.length; index++) {\n const target = this._registeredForLateAnimationBindings.data[index];\n for (const path in target._lateAnimationHolders) {\n const holder = target._lateAnimationHolders[path];\n const originalAnimation = holder.animations[0];\n const originalValue = holder.originalValue;\n if (originalValue === undefined || originalValue === null) {\n continue;\n }\n const matrixDecomposeMode = Animation.AllowMatrixDecomposeForInterpolation && originalValue.m; // ie. data is matrix\n let finalValue = target[path];\n if (matrixDecomposeMode) {\n finalValue = this._processLateAnimationBindingsForMatrices(holder);\n }\n else {\n const quaternionMode = originalValue.w !== undefined;\n if (quaternionMode) {\n finalValue = this._processLateAnimationBindingsForQuaternions(holder, finalValue || Quaternion.Identity());\n }\n else {\n let startIndex = 0;\n let normalizer = 1.0;\n if (holder.totalWeight < 1.0) {\n // We need to mix the original value in\n if (originalAnimation && originalValue.scale) {\n finalValue = originalValue.scale(1.0 - holder.totalWeight);\n }\n else if (originalAnimation) {\n finalValue = originalValue * (1.0 - holder.totalWeight);\n }\n else if (originalValue.clone) {\n finalValue = originalValue.clone();\n }\n else {\n finalValue = originalValue;\n }\n }\n else if (originalAnimation) {\n // We need to normalize the weights\n normalizer = holder.totalWeight;\n const scale = originalAnimation.weight / normalizer;\n if (scale !== 1) {\n if (originalAnimation.currentValue.scale) {\n finalValue = originalAnimation.currentValue.scale(scale);\n }\n else {\n finalValue = originalAnimation.currentValue * scale;\n }\n }\n else {\n finalValue = originalAnimation.currentValue;\n }\n startIndex = 1;\n }\n // Add up the override animations\n for (let animIndex = startIndex; animIndex < holder.animations.length; animIndex++) {\n const runtimeAnimation = holder.animations[animIndex];\n const scale = runtimeAnimation.weight / normalizer;\n if (!scale) {\n continue;\n }\n else if (runtimeAnimation.currentValue.scaleAndAddToRef) {\n runtimeAnimation.currentValue.scaleAndAddToRef(scale, finalValue);\n }\n else {\n finalValue += runtimeAnimation.currentValue * scale;\n }\n }\n // Add up the additive animations\n for (let animIndex = 0; animIndex < holder.additiveAnimations.length; animIndex++) {\n const runtimeAnimation = holder.additiveAnimations[animIndex];\n const scale = runtimeAnimation.weight;\n if (!scale) {\n continue;\n }\n else if (runtimeAnimation.currentValue.scaleAndAddToRef) {\n runtimeAnimation.currentValue.scaleAndAddToRef(scale, finalValue);\n }\n else {\n finalValue += runtimeAnimation.currentValue * scale;\n }\n }\n }\n }\n target[path] = finalValue;\n }\n target._lateAnimationHolders = {};\n }\n this._registeredForLateAnimationBindings.reset();\n };\n Bone.prototype.copyAnimationRange = function (source, rangeName, frameOffset, rescaleAsRequired = false, skelDimensionsRatio = null) {\n // all animation may be coming from a library skeleton, so may need to create animation\n if (this.animations.length === 0) {\n this.animations.push(new Animation(this.name, \"_matrix\", source.animations[0].framePerSecond, Animation.ANIMATIONTYPE_MATRIX, 0));\n this.animations[0].setKeys([]);\n }\n // get animation info / verify there is such a range from the source bone\n const sourceRange = source.animations[0].getRange(rangeName);\n if (!sourceRange) {\n return false;\n }\n const from = sourceRange.from;\n const to = sourceRange.to;\n const sourceKeys = source.animations[0].getKeys();\n // rescaling prep\n const sourceBoneLength = source.length;\n const sourceParent = source.getParent();\n const parent = this.getParent();\n const parentScalingReqd = rescaleAsRequired && sourceParent && sourceBoneLength && this.length && sourceBoneLength !== this.length;\n const parentRatio = parentScalingReqd && parent && sourceParent ? parent.length / sourceParent.length : 1;\n const dimensionsScalingReqd = rescaleAsRequired && !parent && skelDimensionsRatio && (skelDimensionsRatio.x !== 1 || skelDimensionsRatio.y !== 1 || skelDimensionsRatio.z !== 1);\n const destKeys = this.animations[0].getKeys();\n // loop vars declaration\n let orig;\n let origTranslation;\n let mat;\n for (let key = 0, nKeys = sourceKeys.length; key < nKeys; key++) {\n orig = sourceKeys[key];\n if (orig.frame >= from && orig.frame <= to) {\n if (rescaleAsRequired) {\n mat = orig.value.clone();\n // scale based on parent ratio, when bone has parent\n if (parentScalingReqd) {\n origTranslation = mat.getTranslation();\n mat.setTranslation(origTranslation.scaleInPlace(parentRatio));\n // scale based on skeleton dimension ratio when root bone, and value is passed\n }\n else if (dimensionsScalingReqd && skelDimensionsRatio) {\n origTranslation = mat.getTranslation();\n mat.setTranslation(origTranslation.multiplyInPlace(skelDimensionsRatio));\n // use original when root bone, and no data for skelDimensionsRatio\n }\n else {\n mat = orig.value;\n }\n }\n else {\n mat = orig.value;\n }\n destKeys.push({ frame: orig.frame + frameOffset, value: mat });\n }\n }\n this.animations[0].createRange(rangeName, from + frameOffset, to + frameOffset);\n return true;\n };\n\n /**\n * Wrapper over subclasses of XRLayer.\n * @internal\n */\n class WebXRLayerWrapper {\n /**\n * Check if fixed foveation is supported on this device\n */\n get isFixedFoveationSupported() {\n return this.layerType == \"XRWebGLLayer\" && typeof this.layer.fixedFoveation == \"number\";\n }\n /**\n * Get the fixed foveation currently set, as specified by the webxr specs\n * If this returns null, then fixed foveation is not supported\n */\n get fixedFoveation() {\n if (this.isFixedFoveationSupported) {\n return this.layer.fixedFoveation;\n }\n return null;\n }\n /**\n * Set the fixed foveation to the specified value, as specified by the webxr specs\n * This value will be normalized to be between 0 and 1, 1 being max foveation, 0 being no foveation\n */\n set fixedFoveation(value) {\n if (this.isFixedFoveationSupported) {\n const val = Math.max(0, Math.min(1, value || 0));\n this.layer.fixedFoveation = val;\n }\n }\n constructor(\n /** The width of the layer's framebuffer. */\n getWidth, \n /** The height of the layer's framebuffer. */\n getHeight, \n /** The XR layer that this WebXRLayerWrapper wraps. */\n layer, \n /** The type of XR layer that is being wrapped. */\n layerType, \n /** Create a render target provider for the wrapped layer. */\n createRenderTargetTextureProvider) {\n this.getWidth = getWidth;\n this.getHeight = getHeight;\n this.layer = layer;\n this.layerType = layerType;\n this.createRenderTargetTextureProvider = createRenderTargetTextureProvider;\n }\n }\n\n /**\n * Provides render target textures and other important rendering information for a given XRLayer.\n * @internal\n */\n class WebXRLayerRenderTargetTextureProvider {\n constructor(_scene, layerWrapper) {\n this._scene = _scene;\n this.layerWrapper = layerWrapper;\n this._renderTargetTextures = new Array();\n this._engine = _scene.getEngine();\n }\n _createInternalTexture(textureSize, texture) {\n const internalTexture = new InternalTexture(this._engine, InternalTextureSource.Unknown, true);\n internalTexture.width = textureSize.width;\n internalTexture.height = textureSize.height;\n internalTexture._hardwareTexture = new WebGLHardwareTexture(texture, this._engine._gl);\n internalTexture.isReady = true;\n return internalTexture;\n }\n _createRenderTargetTexture(width, height, framebuffer, colorTexture, depthStencilTexture, multiview) {\n if (!this._engine) {\n throw new Error(\"Engine is disposed\");\n }\n const textureSize = { width, height };\n // Create render target texture from the internal texture\n const renderTargetTexture = multiview ? new MultiviewRenderTarget(this._scene, textureSize) : new RenderTargetTexture(\"XR renderTargetTexture\", textureSize, this._scene);\n const renderTargetWrapper = renderTargetTexture.renderTarget;\n renderTargetWrapper._samples = renderTargetTexture.samples;\n // Set the framebuffer, make sure it works in all scenarios - emulator, no layers and layers\n if (framebuffer || !colorTexture) {\n renderTargetWrapper._framebuffer = framebuffer;\n }\n // Create internal texture\n if (colorTexture) {\n if (multiview) {\n renderTargetWrapper._colorTextureArray = colorTexture;\n }\n else {\n const internalTexture = this._createInternalTexture(textureSize, colorTexture);\n renderTargetWrapper.setTexture(internalTexture, 0);\n renderTargetTexture._texture = internalTexture;\n }\n }\n if (depthStencilTexture) {\n if (multiview) {\n renderTargetWrapper._depthStencilTextureArray = depthStencilTexture;\n }\n else {\n renderTargetWrapper._depthStencilTexture = this._createInternalTexture(textureSize, depthStencilTexture);\n }\n }\n renderTargetTexture.disableRescaling();\n // Firefox reality fails if skipInitialClear is set to true, so make sure only modern XR implementations set it.\n if (typeof XRWebGLBinding !== \"undefined\") {\n // WebXR pre-clears textures\n renderTargetTexture.skipInitialClear = true;\n }\n this._renderTargetTextures.push(renderTargetTexture);\n return renderTargetTexture;\n }\n _destroyRenderTargetTexture(renderTargetTexture) {\n this._renderTargetTextures.splice(this._renderTargetTextures.indexOf(renderTargetTexture), 1);\n renderTargetTexture.dispose();\n }\n getFramebufferDimensions() {\n return this._framebufferDimensions;\n }\n dispose() {\n this._renderTargetTextures.forEach((rtt) => rtt.dispose());\n this._renderTargetTextures.length = 0;\n }\n }\n\n /**\n * Wraps xr webgl layers.\n * @internal\n */\n class WebXRWebGLLayerWrapper extends WebXRLayerWrapper {\n /**\n * @param layer is the layer to be wrapped.\n * @returns a new WebXRLayerWrapper wrapping the provided XRWebGLLayer.\n */\n constructor(layer) {\n super(() => layer.framebufferWidth, () => layer.framebufferHeight, layer, \"XRWebGLLayer\", (sessionManager) => new WebXRWebGLLayerRenderTargetTextureProvider(sessionManager.scene, this));\n this.layer = layer;\n }\n }\n /**\n * Provides render target textures and other important rendering information for a given XRWebGLLayer.\n * @internal\n */\n class WebXRWebGLLayerRenderTargetTextureProvider extends WebXRLayerRenderTargetTextureProvider {\n constructor(scene, layerWrapper) {\n super(scene, layerWrapper);\n this.layerWrapper = layerWrapper;\n this._layer = layerWrapper.layer;\n this._framebufferDimensions = {\n framebufferWidth: this._layer.framebufferWidth,\n framebufferHeight: this._layer.framebufferHeight,\n };\n }\n trySetViewportForView(viewport, view) {\n const xrViewport = this._layer.getViewport(view);\n if (!xrViewport) {\n return false;\n }\n const framebufferWidth = this._framebufferDimensions.framebufferWidth;\n const framebufferHeight = this._framebufferDimensions.framebufferHeight;\n viewport.x = xrViewport.x / framebufferWidth;\n viewport.y = xrViewport.y / framebufferHeight;\n viewport.width = xrViewport.width / framebufferWidth;\n viewport.height = xrViewport.height / framebufferHeight;\n return true;\n }\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n getRenderTargetTextureForEye(eye) {\n const layerWidth = this._layer.framebufferWidth;\n const layerHeight = this._layer.framebufferHeight;\n const framebuffer = this._layer.framebuffer;\n if (!this._rtt ||\n layerWidth !== this._framebufferDimensions.framebufferWidth ||\n layerHeight !== this._framebufferDimensions.framebufferHeight ||\n framebuffer !== this._framebuffer) {\n this._rtt = this._createRenderTargetTexture(layerWidth, layerHeight, framebuffer);\n this._framebufferDimensions.framebufferWidth = layerWidth;\n this._framebufferDimensions.framebufferHeight = layerHeight;\n this._framebuffer = framebuffer;\n }\n return this._rtt;\n }\n getRenderTargetTextureForView(view) {\n return this.getRenderTargetTextureForEye(view.eye);\n }\n }\n\n /**\n * Configuration object for WebXR output canvas\n */\n class WebXRManagedOutputCanvasOptions {\n /**\n * Get the default values of the configuration object\n * @param engine defines the engine to use (can be null)\n * @returns default values of this configuration object\n */\n static GetDefaults(engine) {\n const defaults = new WebXRManagedOutputCanvasOptions();\n defaults.canvasOptions = {\n antialias: true,\n depth: true,\n stencil: engine ? engine.isStencilEnable : true,\n alpha: true,\n framebufferScaleFactor: 1,\n };\n defaults.newCanvasCssStyle = \"position:absolute; bottom:0px;right:0px;z-index:10;width:90%;height:100%;background-color: #000000;\";\n return defaults;\n }\n }\n /**\n * Creates a canvas that is added/removed from the webpage when entering/exiting XR\n */\n class WebXRManagedOutputCanvas {\n /**\n * Initializes the canvas to be added/removed upon entering/exiting xr\n * @param _xrSessionManager The XR Session manager\n * @param _options optional configuration for this canvas output. defaults will be used if not provided\n */\n constructor(_xrSessionManager, _options = WebXRManagedOutputCanvasOptions.GetDefaults()) {\n this._options = _options;\n this._canvas = null;\n this._engine = null;\n /**\n * xr layer for the canvas\n */\n this.xrLayer = null;\n this._xrLayerWrapper = null;\n /**\n * Observers registered here will be triggered when the xr layer was initialized\n */\n this.onXRLayerInitObservable = new Observable$1();\n this._engine = _xrSessionManager.scene.getEngine();\n this._engine.onDisposeObservable.addOnce(() => {\n this._engine = null;\n });\n if (!_options.canvasElement) {\n const canvas = document.createElement(\"canvas\");\n canvas.style.cssText = this._options.newCanvasCssStyle || \"position:absolute; bottom:0px;right:0px;\";\n this._setManagedOutputCanvas(canvas);\n }\n else {\n this._setManagedOutputCanvas(_options.canvasElement);\n }\n _xrSessionManager.onXRSessionInit.add(() => {\n this._addCanvas();\n });\n _xrSessionManager.onXRSessionEnded.add(() => {\n this._removeCanvas();\n });\n }\n /**\n * Disposes of the object\n */\n dispose() {\n this._removeCanvas();\n this._setManagedOutputCanvas(null);\n }\n /**\n * Initializes a XRWebGLLayer to be used as the session's baseLayer.\n * @param xrSession xr session\n * @returns a promise that will resolve once the XR Layer has been created\n */\n async initializeXRLayerAsync(xrSession) {\n const createLayer = () => {\n this.xrLayer = new XRWebGLLayer(xrSession, this.canvasContext, this._options.canvasOptions);\n this._xrLayerWrapper = new WebXRWebGLLayerWrapper(this.xrLayer);\n this.onXRLayerInitObservable.notifyObservers(this.xrLayer);\n return this.xrLayer;\n };\n // support canvases without makeXRCompatible\n if (!this.canvasContext.makeXRCompatible) {\n return Promise.resolve(createLayer());\n }\n return this.canvasContext\n .makeXRCompatible()\n .then(\n // catch any error and continue. When using the emulator is throws this error for no apparent reason.\n () => { }, () => {\n // log the error, continue nonetheless!\n Tools.Warn(\"Error executing makeXRCompatible. This does not mean that the session will work incorrectly.\");\n })\n .then(() => {\n return createLayer();\n });\n }\n _addCanvas() {\n if (this._canvas && this._engine && this._canvas !== this._engine.getRenderingCanvas()) {\n document.body.appendChild(this._canvas);\n }\n if (this.xrLayer) {\n this._setCanvasSize(true);\n }\n else {\n this.onXRLayerInitObservable.addOnce(() => {\n this._setCanvasSize(true);\n });\n }\n }\n _removeCanvas() {\n if (this._canvas && this._engine && document.body.contains(this._canvas) && this._canvas !== this._engine.getRenderingCanvas()) {\n document.body.removeChild(this._canvas);\n }\n this._setCanvasSize(false);\n }\n _setCanvasSize(init = true, xrLayer = this._xrLayerWrapper) {\n if (!this._canvas || !this._engine) {\n return;\n }\n if (init) {\n if (xrLayer) {\n if (this._canvas !== this._engine.getRenderingCanvas()) {\n this._canvas.style.width = xrLayer.getWidth() + \"px\";\n this._canvas.style.height = xrLayer.getHeight() + \"px\";\n }\n else {\n this._engine.setSize(xrLayer.getWidth(), xrLayer.getHeight());\n }\n }\n }\n else {\n if (this._originalCanvasSize) {\n if (this._canvas !== this._engine.getRenderingCanvas()) {\n this._canvas.style.width = this._originalCanvasSize.width + \"px\";\n this._canvas.style.height = this._originalCanvasSize.height + \"px\";\n }\n else {\n this._engine.setSize(this._originalCanvasSize.width, this._originalCanvasSize.height);\n }\n }\n }\n }\n _setManagedOutputCanvas(canvas) {\n this._removeCanvas();\n if (!canvas) {\n this._canvas = null;\n this.canvasContext = null;\n }\n else {\n this._originalCanvasSize = {\n width: canvas.offsetWidth,\n height: canvas.offsetHeight,\n };\n this._canvas = canvas;\n this.canvasContext = this._canvas.getContext(\"webgl2\");\n if (!this.canvasContext) {\n this.canvasContext = this._canvas.getContext(\"webgl\");\n }\n }\n }\n }\n\n /**\n * Wraps XRWebGLLayer's created by Babylon Native.\n * @internal\n */\n class NativeXRLayerWrapper extends WebXRLayerWrapper {\n constructor(layer) {\n super(() => layer.framebufferWidth, () => layer.framebufferHeight, layer, \"XRWebGLLayer\", (sessionManager) => new NativeXRLayerRenderTargetTextureProvider(sessionManager, this));\n this.layer = layer;\n }\n }\n /**\n * Provides render target textures for layers created by Babylon Native.\n * @internal\n */\n class NativeXRLayerRenderTargetTextureProvider extends WebXRLayerRenderTargetTextureProvider {\n constructor(sessionManager, layerWrapper) {\n super(sessionManager.scene, layerWrapper);\n this.layerWrapper = layerWrapper;\n this._nativeRTTProvider = navigator.xr.getNativeRenderTargetProvider(sessionManager.session, this._createRenderTargetTexture.bind(this), this._destroyRenderTargetTexture.bind(this));\n this._nativeLayer = layerWrapper.layer;\n }\n trySetViewportForView(viewport) {\n viewport.x = 0;\n viewport.y = 0;\n viewport.width = 1;\n viewport.height = 1;\n return true;\n }\n getRenderTargetTextureForEye(eye) {\n // TODO (rgerd): Update the contract on the BabylonNative side to call this \"getRenderTargetTextureForEye\"\n return this._nativeRTTProvider.getRenderTargetForEye(eye);\n }\n getRenderTargetTextureForView(view) {\n return this._nativeRTTProvider.getRenderTargetForEye(view.eye);\n }\n getFramebufferDimensions() {\n return {\n framebufferWidth: this._nativeLayer.framebufferWidth,\n framebufferHeight: this._nativeLayer.framebufferHeight,\n };\n }\n }\n /**\n * Creates the xr layer that will be used as the xr session's base layer.\n * @internal\n */\n class NativeXRRenderTarget {\n constructor(_xrSessionManager) {\n this._nativeRenderTarget = navigator.xr.getWebXRRenderTarget(_xrSessionManager.scene.getEngine());\n }\n async initializeXRLayerAsync(xrSession) {\n await this._nativeRenderTarget.initializeXRLayerAsync(xrSession);\n this.xrLayer = this._nativeRenderTarget.xrLayer;\n return this.xrLayer;\n }\n dispose() {\n /* empty */\n }\n }\n\n /**\n * Manages an XRSession to work with Babylon's engine\n * @see https://doc.babylonjs.com/features/featuresDeepDive/webXR/webXRSessionManagers\n */\n class WebXRSessionManager {\n /**\n * Constructs a WebXRSessionManager, this must be initialized within a user action before usage\n * @param scene The scene which the session should be created for\n */\n constructor(\n /** The scene which the session should be created for */\n scene) {\n this.scene = scene;\n /** WebXR timestamp updated every frame */\n this.currentTimestamp = -1;\n /**\n * Used just in case of a failure to initialize an immersive session.\n * The viewer reference space is compensated using this height, creating a kind of \"viewer-floor\" reference space\n */\n this.defaultHeightCompensation = 1.7;\n /**\n * Fires every time a new xrFrame arrives which can be used to update the camera\n */\n this.onXRFrameObservable = new Observable$1();\n /**\n * Fires when the reference space changed\n */\n this.onXRReferenceSpaceChanged = new Observable$1();\n /**\n * Fires when the xr session is ended either by the device or manually done\n */\n this.onXRSessionEnded = new Observable$1();\n /**\n * Fires when the xr session is initialized: right after requestSession was called and returned with a successful result\n */\n this.onXRSessionInit = new Observable$1();\n /**\n * Are we currently in the XR loop?\n */\n this.inXRFrameLoop = false;\n /**\n * Are we in an XR session?\n */\n this.inXRSession = false;\n this._engine = scene.getEngine();\n this._onEngineDisposedObserver = this._engine.onDisposeObservable.addOnce(() => {\n this._engine = null;\n });\n scene.onDisposeObservable.addOnce(() => {\n this.dispose();\n });\n }\n /**\n * The current reference space used in this session. This reference space can constantly change!\n * It is mainly used to offset the camera's position.\n */\n get referenceSpace() {\n return this._referenceSpace;\n }\n /**\n * Set a new reference space and triggers the observable\n */\n set referenceSpace(newReferenceSpace) {\n this._referenceSpace = newReferenceSpace;\n this.onXRReferenceSpaceChanged.notifyObservers(this._referenceSpace);\n }\n /**\n * The mode for the managed XR session\n */\n get sessionMode() {\n return this._sessionMode;\n }\n /**\n * Disposes of the session manager\n * This should be called explicitly by the dev, if required.\n */\n dispose() {\n var _a;\n // disposing without leaving XR? Exit XR first\n if (this.inXRSession) {\n this.exitXRAsync();\n }\n this.onXRFrameObservable.clear();\n this.onXRSessionEnded.clear();\n this.onXRReferenceSpaceChanged.clear();\n this.onXRSessionInit.clear();\n (_a = this._engine) === null || _a === void 0 ? void 0 : _a.onDisposeObservable.remove(this._onEngineDisposedObserver);\n this._engine = null;\n }\n /**\n * Stops the xrSession and restores the render loop\n * @returns Promise which resolves after it exits XR\n */\n exitXRAsync() {\n if (this.session && this.inXRSession) {\n this.inXRSession = false;\n return this.session.end().catch(() => {\n Logger.Warn(\"Could not end XR session.\");\n });\n }\n return Promise.resolve();\n }\n /**\n * Attempts to set the framebuffer-size-normalized viewport to be rendered this frame for this view.\n * In the event of a failure, the supplied viewport is not updated.\n * @param viewport the viewport to which the view will be rendered\n * @param view the view for which to set the viewport\n * @returns whether the operation was successful\n */\n trySetViewportForView(viewport, view) {\n var _a;\n return ((_a = this._baseLayerRTTProvider) === null || _a === void 0 ? void 0 : _a.trySetViewportForView(viewport, view)) || false;\n }\n /**\n * Gets the correct render target texture to be rendered this frame for this eye\n * @param eye the eye for which to get the render target\n * @returns the render target for the specified eye or null if not available\n */\n getRenderTargetTextureForEye(eye) {\n var _a;\n return ((_a = this._baseLayerRTTProvider) === null || _a === void 0 ? void 0 : _a.getRenderTargetTextureForEye(eye)) || null;\n }\n /**\n * Gets the correct render target texture to be rendered this frame for this view\n * @param view the view for which to get the render target\n * @returns the render target for the specified view or null if not available\n */\n getRenderTargetTextureForView(view) {\n var _a;\n return ((_a = this._baseLayerRTTProvider) === null || _a === void 0 ? void 0 : _a.getRenderTargetTextureForView(view)) || null;\n }\n /**\n * Creates a WebXRRenderTarget object for the XR session\n * @param options optional options to provide when creating a new render target\n * @returns a WebXR render target to which the session can render\n */\n getWebXRRenderTarget(options) {\n const engine = this.scene.getEngine();\n if (this._xrNavigator.xr.native) {\n return new NativeXRRenderTarget(this);\n }\n else {\n options = options || WebXRManagedOutputCanvasOptions.GetDefaults(engine);\n options.canvasElement = options.canvasElement || engine.getRenderingCanvas() || undefined;\n return new WebXRManagedOutputCanvas(this, options);\n }\n }\n /**\n * Initializes the manager\n * After initialization enterXR can be called to start an XR session\n * @returns Promise which resolves after it is initialized\n */\n initializeAsync() {\n // Check if the browser supports webXR\n this._xrNavigator = navigator;\n if (!this._xrNavigator.xr) {\n return Promise.reject(\"WebXR not available\");\n }\n return Promise.resolve();\n }\n /**\n * Initializes an xr session\n * @param xrSessionMode mode to initialize\n * @param xrSessionInit defines optional and required values to pass to the session builder\n * @returns a promise which will resolve once the session has been initialized\n */\n initializeSessionAsync(xrSessionMode = \"immersive-vr\", xrSessionInit = {}) {\n return this._xrNavigator.xr.requestSession(xrSessionMode, xrSessionInit).then((session) => {\n this.session = session;\n this._sessionMode = xrSessionMode;\n this.onXRSessionInit.notifyObservers(session);\n this.inXRSession = true;\n // handle when the session is ended (By calling session.end or device ends its own session eg. pressing home button on phone)\n this.session.addEventListener(\"end\", () => {\n var _a;\n this.inXRSession = false;\n // Notify frame observers\n this.onXRSessionEnded.notifyObservers(null);\n if (this._engine) {\n // make sure dimensions object is restored\n this._engine.framebufferDimensionsObject = null;\n // Restore frame buffer to avoid clear on xr framebuffer after session end\n this._engine.restoreDefaultFramebuffer();\n // Need to restart render loop as after the session is ended the last request for new frame will never call callback\n this._engine.customAnimationFrameRequester = null;\n this._engine._renderLoop();\n }\n // Dispose render target textures.\n // Only dispose on native because we can't destroy opaque textures on browser.\n if (this.isNative) {\n (_a = this._baseLayerRTTProvider) === null || _a === void 0 ? void 0 : _a.dispose();\n }\n this._baseLayerRTTProvider = null;\n this._baseLayerWrapper = null;\n }, { once: true });\n return this.session;\n });\n }\n /**\n * Checks if a session would be supported for the creation options specified\n * @param sessionMode session mode to check if supported eg. immersive-vr\n * @returns A Promise that resolves to true if supported and false if not\n */\n isSessionSupportedAsync(sessionMode) {\n return WebXRSessionManager.IsSessionSupportedAsync(sessionMode);\n }\n /**\n * Resets the reference space to the one started the session\n */\n resetReferenceSpace() {\n this.referenceSpace = this.baseReferenceSpace;\n }\n /**\n * Starts rendering to the xr layer\n */\n runXRRenderLoop() {\n var _a;\n if (!this.inXRSession || !this._engine) {\n return;\n }\n // Tell the engine's render loop to be driven by the xr session's refresh rate and provide xr pose information\n this._engine.customAnimationFrameRequester = {\n requestAnimationFrame: this.session.requestAnimationFrame.bind(this.session),\n renderFunction: (timestamp, xrFrame) => {\n var _a;\n if (!this.inXRSession || !this._engine) {\n return;\n }\n // Store the XR frame and timestamp in the session manager\n this.currentFrame = xrFrame;\n this.currentTimestamp = timestamp;\n if (xrFrame) {\n this.inXRFrameLoop = true;\n this._engine.framebufferDimensionsObject = ((_a = this._baseLayerRTTProvider) === null || _a === void 0 ? void 0 : _a.getFramebufferDimensions()) || null;\n this.onXRFrameObservable.notifyObservers(xrFrame);\n this._engine._renderLoop();\n this._engine.framebufferDimensionsObject = null;\n this.inXRFrameLoop = false;\n }\n },\n };\n this._engine.framebufferDimensionsObject = ((_a = this._baseLayerRTTProvider) === null || _a === void 0 ? void 0 : _a.getFramebufferDimensions()) || null;\n // Stop window's animation frame and trigger sessions animation frame\n if (typeof window !== \"undefined\" && window.cancelAnimationFrame) {\n window.cancelAnimationFrame(this._engine._frameHandler);\n }\n this._engine._renderLoop();\n }\n /**\n * Sets the reference space on the xr session\n * @param referenceSpaceType space to set\n * @returns a promise that will resolve once the reference space has been set\n */\n setReferenceSpaceTypeAsync(referenceSpaceType = \"local-floor\") {\n return this.session\n .requestReferenceSpace(referenceSpaceType)\n .then((referenceSpace) => {\n return referenceSpace;\n }, (rejectionReason) => {\n Logger.Error(\"XR.requestReferenceSpace failed for the following reason: \");\n Logger.Error(rejectionReason);\n Logger.Log('Defaulting to universally-supported \"viewer\" reference space type.');\n return this.session.requestReferenceSpace(\"viewer\").then((referenceSpace) => {\n const heightCompensation = new XRRigidTransform({ x: 0, y: -this.defaultHeightCompensation, z: 0 });\n return referenceSpace.getOffsetReferenceSpace(heightCompensation);\n }, (rejectionReason) => {\n Logger.Error(rejectionReason);\n throw 'XR initialization failed: required \"viewer\" reference space type not supported.';\n });\n })\n .then((referenceSpace) => {\n // create viewer reference space before setting the first reference space\n return this.session.requestReferenceSpace(\"viewer\").then((viewerReferenceSpace) => {\n this.viewerReferenceSpace = viewerReferenceSpace;\n return referenceSpace;\n });\n })\n .then((referenceSpace) => {\n // initialize the base and offset (currently the same)\n this.referenceSpace = this.baseReferenceSpace = referenceSpace;\n return this.referenceSpace;\n });\n }\n /**\n * Updates the render state of the session.\n * Note that this is deprecated in favor of WebXRSessionManager.updateRenderState().\n * @param state state to set\n * @returns a promise that resolves once the render state has been updated\n * @deprecated\n */\n updateRenderStateAsync(state) {\n return Promise.resolve(this.session.updateRenderState(state));\n }\n /**\n * @internal\n */\n _setBaseLayerWrapper(baseLayerWrapper) {\n var _a, _b;\n if (this.isNative) {\n (_a = this._baseLayerRTTProvider) === null || _a === void 0 ? void 0 : _a.dispose();\n }\n this._baseLayerWrapper = baseLayerWrapper;\n this._baseLayerRTTProvider = ((_b = this._baseLayerWrapper) === null || _b === void 0 ? void 0 : _b.createRenderTargetTextureProvider(this)) || null;\n }\n /**\n * @internal\n */\n _getBaseLayerWrapper() {\n return this._baseLayerWrapper;\n }\n /**\n * Updates the render state of the session\n * @param state state to set\n */\n updateRenderState(state) {\n if (state.baseLayer) {\n this._setBaseLayerWrapper(this.isNative ? new NativeXRLayerWrapper(state.baseLayer) : new WebXRWebGLLayerWrapper(state.baseLayer));\n }\n this.session.updateRenderState(state);\n }\n /**\n * Returns a promise that resolves with a boolean indicating if the provided session mode is supported by this browser\n * @param sessionMode defines the session to test\n * @returns a promise with boolean as final value\n */\n static IsSessionSupportedAsync(sessionMode) {\n if (!navigator.xr) {\n return Promise.resolve(false);\n }\n // When the specs are final, remove supportsSession!\n const functionToUse = navigator.xr.isSessionSupported || navigator.xr.supportsSession;\n if (!functionToUse) {\n return Promise.resolve(false);\n }\n else {\n return functionToUse\n .call(navigator.xr, sessionMode)\n .then((result) => {\n const returnValue = typeof result === \"undefined\" ? true : result;\n return Promise.resolve(returnValue);\n })\n .catch((e) => {\n Logger.Warn(e);\n return Promise.resolve(false);\n });\n }\n }\n /**\n * Returns true if Babylon.js is using the BabylonNative backend, otherwise false\n */\n get isNative() {\n var _a;\n return (_a = this._xrNavigator.xr.native) !== null && _a !== void 0 ? _a : false;\n }\n /**\n * The current frame rate as reported by the device\n */\n get currentFrameRate() {\n var _a;\n return (_a = this.session) === null || _a === void 0 ? void 0 : _a.frameRate;\n }\n /**\n * A list of supported frame rates (only available in-session!\n */\n get supportedFrameRates() {\n var _a;\n return (_a = this.session) === null || _a === void 0 ? void 0 : _a.supportedFrameRates;\n }\n /**\n * Set the framerate of the session.\n * @param rate the new framerate. This value needs to be in the supportedFrameRates array\n * @returns a promise that resolves once the framerate has been set\n */\n updateTargetFrameRate(rate) {\n return this.session.updateTargetFrameRate(rate);\n }\n /**\n * Run a callback in the xr render loop\n * @param callback the callback to call when in XR Frame\n * @param ignoreIfNotInSession if no session is currently running, run it first thing on the next session\n */\n runInXRFrame(callback, ignoreIfNotInSession = true) {\n if (this.inXRFrameLoop) {\n callback();\n }\n else if (this.inXRSession || !ignoreIfNotInSession) {\n this.onXRFrameObservable.addOnce(callback);\n }\n }\n /**\n * Check if fixed foveation is supported on this device\n */\n get isFixedFoveationSupported() {\n var _a;\n return ((_a = this._baseLayerWrapper) === null || _a === void 0 ? void 0 : _a.isFixedFoveationSupported) || false;\n }\n /**\n * Get the fixed foveation currently set, as specified by the webxr specs\n * If this returns null, then fixed foveation is not supported\n */\n get fixedFoveation() {\n var _a;\n return ((_a = this._baseLayerWrapper) === null || _a === void 0 ? void 0 : _a.fixedFoveation) || null;\n }\n /**\n * Set the fixed foveation to the specified value, as specified by the webxr specs\n * This value will be normalized to be between 0 and 1, 1 being max foveation, 0 being no foveation\n */\n set fixedFoveation(value) {\n const val = Math.max(0, Math.min(1, value || 0));\n if (this._baseLayerWrapper) {\n this._baseLayerWrapper.fixedFoveation = val;\n }\n }\n /**\n * Get the features enabled on the current session\n * This is only available in-session!\n * @see https://www.w3.org/TR/webxr/#dom-xrsession-enabledfeatures\n */\n get enabledFeatures() {\n var _a, _b;\n return (_b = (_a = this.session) === null || _a === void 0 ? void 0 : _a.enabledFeatures) !== null && _b !== void 0 ? _b : null;\n }\n }\n\n /**\n * States of the webXR experience\n */\n var WebXRState;\n (function (WebXRState) {\n /**\n * Transitioning to being in XR mode\n */\n WebXRState[WebXRState[\"ENTERING_XR\"] = 0] = \"ENTERING_XR\";\n /**\n * Transitioning to non XR mode\n */\n WebXRState[WebXRState[\"EXITING_XR\"] = 1] = \"EXITING_XR\";\n /**\n * In XR mode and presenting\n */\n WebXRState[WebXRState[\"IN_XR\"] = 2] = \"IN_XR\";\n /**\n * Not entered XR mode\n */\n WebXRState[WebXRState[\"NOT_IN_XR\"] = 3] = \"NOT_IN_XR\";\n })(WebXRState || (WebXRState = {}));\n /**\n * The state of the XR camera's tracking\n */\n var WebXRTrackingState;\n (function (WebXRTrackingState) {\n /**\n * No transformation received, device is not being tracked\n */\n WebXRTrackingState[WebXRTrackingState[\"NOT_TRACKING\"] = 0] = \"NOT_TRACKING\";\n /**\n * Tracking lost - using emulated position\n */\n WebXRTrackingState[WebXRTrackingState[\"TRACKING_LOST\"] = 1] = \"TRACKING_LOST\";\n /**\n * Transformation tracking works normally\n */\n WebXRTrackingState[WebXRTrackingState[\"TRACKING\"] = 2] = \"TRACKING\";\n })(WebXRTrackingState || (WebXRTrackingState = {}));\n\n /**\n * Creates the VertexData for a cylinder, cone or prism\n * @param options an object used to set the following optional parameters for the box, required but can be empty\n * * height sets the height (y direction) of the cylinder, optional, default 2\n * * diameterTop sets the diameter of the top of the cone, overwrites diameter, optional, default diameter\n * * diameterBottom sets the diameter of the bottom of the cone, overwrites diameter, optional, default diameter\n * * diameter sets the diameter of the top and bottom of the cone, optional default 1\n * * tessellation the number of prism sides, 3 for a triangular prism, optional, default 24\n * * subdivisions` the number of rings along the cylinder height, optional, default 1\n * * arc a number from 0 to 1, to create an unclosed cylinder based on the fraction of the circumference given by the arc value, optional, default 1\n * * faceColors an array of Color3 elements used to set different colors to the top, rings and bottom respectively\n * * faceUV an array of Vector4 elements used to set different images to the top, rings and bottom respectively\n * * hasRings when true makes each subdivision independently treated as a face for faceUV and faceColors, optional, default false\n * * enclose when true closes an open cylinder by adding extra flat faces between the height axis and vertical edges, think cut cake\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.height\n * @param options.diameterTop\n * @param options.diameterBottom\n * @param options.diameter\n * @param options.tessellation\n * @param options.subdivisions\n * @param options.arc\n * @param options.faceColors\n * @param options.faceUV\n * @param options.hasRings\n * @param options.enclose\n * @param options.cap\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @returns the VertexData of the cylinder, cone or prism\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n function CreateCylinderVertexData(options) {\n const height = options.height || 2;\n let diameterTop = options.diameterTop === 0 ? 0 : options.diameterTop || options.diameter || 1;\n let diameterBottom = options.diameterBottom === 0 ? 0 : options.diameterBottom || options.diameter || 1;\n diameterTop = diameterTop || 0.00001; // Prevent broken normals\n diameterBottom = diameterBottom || 0.00001; // Prevent broken normals\n const tessellation = (options.tessellation || 24) | 0;\n const subdivisions = (options.subdivisions || 1) | 0;\n const hasRings = options.hasRings ? true : false;\n const enclose = options.enclose ? true : false;\n const cap = options.cap === 0 ? 0 : options.cap || Mesh.CAP_ALL;\n const arc = options.arc && (options.arc <= 0 || options.arc > 1) ? 1.0 : options.arc || 1.0;\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\n const faceUV = options.faceUV || new Array(3);\n const faceColors = options.faceColors;\n // default face colors and UV if undefined\n const quadNb = arc !== 1 && enclose ? 2 : 0;\n const ringNb = hasRings ? subdivisions : 1;\n const surfaceNb = 2 + (1 + quadNb) * ringNb;\n let f;\n for (f = 0; f < surfaceNb; f++) {\n if (faceColors && faceColors[f] === undefined) {\n faceColors[f] = new Color4(1, 1, 1, 1);\n }\n }\n for (f = 0; f < surfaceNb; f++) {\n if (faceUV && faceUV[f] === undefined) {\n faceUV[f] = new Vector4(0, 0, 1, 1);\n }\n }\n const indices = new Array();\n const positions = new Array();\n const normals = new Array();\n const uvs = new Array();\n const colors = new Array();\n const angleStep = (Math.PI * 2 * arc) / tessellation;\n let angle;\n let h;\n let radius;\n const tan = (diameterBottom - diameterTop) / 2 / height;\n const ringVertex = Vector3.Zero();\n const ringNormal = Vector3.Zero();\n const ringFirstVertex = Vector3.Zero();\n const ringFirstNormal = Vector3.Zero();\n const quadNormal = Vector3.Zero();\n const Y = Axis.Y;\n // positions, normals, uvs\n let i;\n let j;\n let r;\n let ringIdx = 1;\n let s = 1; // surface index\n let cs = 0;\n let v = 0;\n for (i = 0; i <= subdivisions; i++) {\n h = i / subdivisions;\n radius = (h * (diameterTop - diameterBottom) + diameterBottom) / 2;\n ringIdx = hasRings && i !== 0 && i !== subdivisions ? 2 : 1;\n for (r = 0; r < ringIdx; r++) {\n if (hasRings) {\n s += r;\n }\n if (enclose) {\n s += 2 * r;\n }\n for (j = 0; j <= tessellation; j++) {\n angle = j * angleStep;\n // position\n ringVertex.x = Math.cos(-angle) * radius;\n ringVertex.y = -height / 2 + h * height;\n ringVertex.z = Math.sin(-angle) * radius;\n // normal\n if (diameterTop === 0 && i === subdivisions) {\n // if no top cap, reuse former normals\n ringNormal.x = normals[normals.length - (tessellation + 1) * 3];\n ringNormal.y = normals[normals.length - (tessellation + 1) * 3 + 1];\n ringNormal.z = normals[normals.length - (tessellation + 1) * 3 + 2];\n }\n else {\n ringNormal.x = ringVertex.x;\n ringNormal.z = ringVertex.z;\n ringNormal.y = Math.sqrt(ringNormal.x * ringNormal.x + ringNormal.z * ringNormal.z) * tan;\n ringNormal.normalize();\n }\n // keep first ring vertex values for enclose\n if (j === 0) {\n ringFirstVertex.copyFrom(ringVertex);\n ringFirstNormal.copyFrom(ringNormal);\n }\n positions.push(ringVertex.x, ringVertex.y, ringVertex.z);\n normals.push(ringNormal.x, ringNormal.y, ringNormal.z);\n if (hasRings) {\n v = cs !== s ? faceUV[s].y : faceUV[s].w;\n }\n else {\n v = faceUV[s].y + (faceUV[s].w - faceUV[s].y) * h;\n }\n uvs.push(faceUV[s].x + ((faceUV[s].z - faceUV[s].x) * j) / tessellation, CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - v : v);\n if (faceColors) {\n colors.push(faceColors[s].r, faceColors[s].g, faceColors[s].b, faceColors[s].a);\n }\n }\n // if enclose, add four vertices and their dedicated normals\n if (arc !== 1 && enclose) {\n positions.push(ringVertex.x, ringVertex.y, ringVertex.z);\n positions.push(0, ringVertex.y, 0);\n positions.push(0, ringVertex.y, 0);\n positions.push(ringFirstVertex.x, ringFirstVertex.y, ringFirstVertex.z);\n Vector3.CrossToRef(Y, ringNormal, quadNormal);\n quadNormal.normalize();\n normals.push(quadNormal.x, quadNormal.y, quadNormal.z, quadNormal.x, quadNormal.y, quadNormal.z);\n Vector3.CrossToRef(ringFirstNormal, Y, quadNormal);\n quadNormal.normalize();\n normals.push(quadNormal.x, quadNormal.y, quadNormal.z, quadNormal.x, quadNormal.y, quadNormal.z);\n if (hasRings) {\n v = cs !== s ? faceUV[s + 1].y : faceUV[s + 1].w;\n }\n else {\n v = faceUV[s + 1].y + (faceUV[s + 1].w - faceUV[s + 1].y) * h;\n }\n uvs.push(faceUV[s + 1].x, CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - v : v);\n uvs.push(faceUV[s + 1].z, CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - v : v);\n if (hasRings) {\n v = cs !== s ? faceUV[s + 2].y : faceUV[s + 2].w;\n }\n else {\n v = faceUV[s + 2].y + (faceUV[s + 2].w - faceUV[s + 2].y) * h;\n }\n uvs.push(faceUV[s + 2].x, CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - v : v);\n uvs.push(faceUV[s + 2].z, CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - v : v);\n if (faceColors) {\n colors.push(faceColors[s + 1].r, faceColors[s + 1].g, faceColors[s + 1].b, faceColors[s + 1].a);\n colors.push(faceColors[s + 1].r, faceColors[s + 1].g, faceColors[s + 1].b, faceColors[s + 1].a);\n colors.push(faceColors[s + 2].r, faceColors[s + 2].g, faceColors[s + 2].b, faceColors[s + 2].a);\n colors.push(faceColors[s + 2].r, faceColors[s + 2].g, faceColors[s + 2].b, faceColors[s + 2].a);\n }\n }\n if (cs !== s) {\n cs = s;\n }\n }\n }\n // indices\n const e = arc !== 1 && enclose ? tessellation + 4 : tessellation; // correction of number of iteration if enclose\n i = 0;\n for (s = 0; s < subdivisions; s++) {\n let i0 = 0;\n let i1 = 0;\n let i2 = 0;\n let i3 = 0;\n for (j = 0; j < tessellation; j++) {\n i0 = i * (e + 1) + j;\n i1 = (i + 1) * (e + 1) + j;\n i2 = i * (e + 1) + (j + 1);\n i3 = (i + 1) * (e + 1) + (j + 1);\n indices.push(i0, i1, i2);\n indices.push(i3, i2, i1);\n }\n if (arc !== 1 && enclose) {\n // if enclose, add two quads\n indices.push(i0 + 2, i1 + 2, i2 + 2);\n indices.push(i3 + 2, i2 + 2, i1 + 2);\n indices.push(i0 + 4, i1 + 4, i2 + 4);\n indices.push(i3 + 4, i2 + 4, i1 + 4);\n }\n i = hasRings ? i + 2 : i + 1;\n }\n // Caps\n const createCylinderCap = (isTop) => {\n const radius = isTop ? diameterTop / 2 : diameterBottom / 2;\n if (radius === 0) {\n return;\n }\n // Cap positions, normals & uvs\n let angle;\n let circleVector;\n let i;\n const u = isTop ? faceUV[surfaceNb - 1] : faceUV[0];\n let c = null;\n if (faceColors) {\n c = isTop ? faceColors[surfaceNb - 1] : faceColors[0];\n }\n // cap center\n const vbase = positions.length / 3;\n const offset = isTop ? height / 2 : -height / 2;\n const center = new Vector3(0, offset, 0);\n positions.push(center.x, center.y, center.z);\n normals.push(0, isTop ? 1 : -1, 0);\n const v = u.y + (u.w - u.y) * 0.5;\n uvs.push(u.x + (u.z - u.x) * 0.5, CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - v : v);\n if (c) {\n colors.push(c.r, c.g, c.b, c.a);\n }\n const textureScale = new Vector2(0.5, 0.5);\n for (i = 0; i <= tessellation; i++) {\n angle = (Math.PI * 2 * i * arc) / tessellation;\n const cos = Math.cos(-angle);\n const sin = Math.sin(-angle);\n circleVector = new Vector3(cos * radius, offset, sin * radius);\n const textureCoordinate = new Vector2(cos * textureScale.x + 0.5, sin * textureScale.y + 0.5);\n positions.push(circleVector.x, circleVector.y, circleVector.z);\n normals.push(0, isTop ? 1 : -1, 0);\n const v = u.y + (u.w - u.y) * textureCoordinate.y;\n uvs.push(u.x + (u.z - u.x) * textureCoordinate.x, CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - v : v);\n if (c) {\n colors.push(c.r, c.g, c.b, c.a);\n }\n }\n // Cap indices\n for (i = 0; i < tessellation; i++) {\n if (!isTop) {\n indices.push(vbase);\n indices.push(vbase + (i + 1));\n indices.push(vbase + (i + 2));\n }\n else {\n indices.push(vbase);\n indices.push(vbase + (i + 2));\n indices.push(vbase + (i + 1));\n }\n }\n };\n // add caps to geometry based on cap parameter\n if (cap === Mesh.CAP_START || cap === Mesh.CAP_ALL) {\n createCylinderCap(false);\n }\n if (cap === Mesh.CAP_END || cap === Mesh.CAP_ALL) {\n createCylinderCap(true);\n }\n // Sides\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\n const vertexData = new VertexData();\n vertexData.indices = indices;\n vertexData.positions = positions;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n if (faceColors) {\n vertexData.colors = colors;\n }\n return vertexData;\n }\n /**\n * Creates a cylinder or a cone mesh\n * * The parameter `height` sets the height size (float) of the cylinder/cone (float, default 2).\n * * The parameter `diameter` sets the diameter of the top and bottom cap at once (float, default 1).\n * * The parameters `diameterTop` and `diameterBottom` overwrite the parameter `diameter` and set respectively the top cap and bottom cap diameter (floats, default 1). The parameter \"diameterBottom\" can't be zero.\n * * The parameter `tessellation` sets the number of cylinder sides (positive integer, default 24). Set it to 3 to get a prism for instance.\n * * The parameter `subdivisions` sets the number of rings along the cylinder height (positive integer, default 1).\n * * The parameter `hasRings` (boolean, default false) makes the subdivisions independent from each other, so they become different faces.\n * * The parameter `enclose` (boolean, default false) adds two extra faces per subdivision to a sliced cylinder to close it around its height axis.\n * * The parameter `cap` sets the way the cylinder is capped. Possible values : BABYLON.Mesh.NO_CAP, BABYLON.Mesh.CAP_START, BABYLON.Mesh.CAP_END, BABYLON.Mesh.CAP_ALL (default).\n * * The parameter `arc` (float, default 1) is the ratio (max 1) to apply to the circumference to slice the cylinder.\n * * You can set different colors and different images to each box side by using the parameters `faceColors` (an array of n Color3 elements) and `faceUV` (an array of n Vector4 elements).\n * * The value of n is the number of cylinder faces. If the cylinder has only 1 subdivisions, n equals : top face + cylinder surface + bottom face = 3\n * * Now, if the cylinder has 5 independent subdivisions (hasRings = true), n equals : top face + 5 stripe surfaces + bottom face = 2 + 5 = 7\n * * Finally, if the cylinder has 5 independent subdivisions and is enclose, n equals : top face + 5 x (stripe surface + 2 closing faces) + bottom face = 2 + 5 * 3 = 17\n * * Each array (color or UVs) is always ordered the same way : the first element is the bottom cap, the last element is the top cap. The other elements are each a ring surface.\n * * If `enclose` is false, a ring surface is one element.\n * * If `enclose` is true, a ring surface is 3 successive elements in the array : the tubular surface, then the two closing faces.\n * * Example how to set colors and textures on a sliced cylinder : https://www.html5gamedevs.com/topic/17945-creating-a-closed-slice-of-a-cylinder/#comment-106379\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param options.height\n * @param options.diameterTop\n * @param options.diameterBottom\n * @param options.diameter\n * @param options.tessellation\n * @param options.subdivisions\n * @param options.arc\n * @param options.faceColors\n * @param options.faceUV\n * @param options.updatable\n * @param options.hasRings\n * @param options.enclose\n * @param options.cap\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param scene defines the hosting scene\n * @returns the cylinder mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#cylinder-or-cone\n */\n function CreateCylinder(name, options = {}, scene) {\n const cylinder = new Mesh(name, scene);\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n cylinder._originalBuilderSideOrientation = options.sideOrientation;\n const vertexData = CreateCylinderVertexData(options);\n vertexData.applyToMesh(cylinder, options.updatable);\n return cylinder;\n }\n /**\n * Class containing static functions to help procedurally build meshes\n * @deprecated Please use CreateCylinder directly\n */\n const CylinderBuilder = {\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CreateCylinder,\n };\n VertexData.CreateCylinder = CreateCylinderVertexData;\n Mesh.CreateCylinder = (name, height, diameterTop, diameterBottom, tessellation, subdivisions, scene, updatable, sideOrientation) => {\n if (scene === undefined || !(scene instanceof Scene)) {\n if (scene !== undefined) {\n sideOrientation = updatable || Mesh.DEFAULTSIDE;\n updatable = scene;\n }\n scene = subdivisions;\n subdivisions = 1;\n }\n const options = {\n height,\n diameterTop,\n diameterBottom,\n tessellation,\n subdivisions,\n sideOrientation,\n updatable,\n };\n return CreateCylinder(name, options, scene);\n };\n\n /**\n * Creates the VertexData for a torus\n * @param options an object used to set the following optional parameters for the box, required but can be empty\n * * diameter the diameter of the torus, optional default 1\n * * thickness the diameter of the tube forming the torus, optional default 0.5\n * * tessellation the number of prism sides, 3 for a triangular prism, optional, default 24\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.diameter\n * @param options.thickness\n * @param options.tessellation\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @returns the VertexData of the torus\n */\n function CreateTorusVertexData(options) {\n const indices = [];\n const positions = [];\n const normals = [];\n const uvs = [];\n const diameter = options.diameter || 1;\n const thickness = options.thickness || 0.5;\n const tessellation = (options.tessellation || 16) | 0;\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\n const stride = tessellation + 1;\n for (let i = 0; i <= tessellation; i++) {\n const u = i / tessellation;\n const outerAngle = (i * Math.PI * 2.0) / tessellation - Math.PI / 2.0;\n const transform = Matrix.Translation(diameter / 2.0, 0, 0).multiply(Matrix.RotationY(outerAngle));\n for (let j = 0; j <= tessellation; j++) {\n const v = 1 - j / tessellation;\n const innerAngle = (j * Math.PI * 2.0) / tessellation + Math.PI;\n const dx = Math.cos(innerAngle);\n const dy = Math.sin(innerAngle);\n // Create a vertex.\n let normal = new Vector3(dx, dy, 0);\n let position = normal.scale(thickness / 2);\n const textureCoordinate = new Vector2(u, v);\n position = Vector3.TransformCoordinates(position, transform);\n normal = Vector3.TransformNormal(normal, transform);\n positions.push(position.x, position.y, position.z);\n normals.push(normal.x, normal.y, normal.z);\n uvs.push(textureCoordinate.x, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - textureCoordinate.y : textureCoordinate.y);\n // And create indices for two triangles.\n const nextI = (i + 1) % stride;\n const nextJ = (j + 1) % stride;\n indices.push(i * stride + j);\n indices.push(i * stride + nextJ);\n indices.push(nextI * stride + j);\n indices.push(i * stride + nextJ);\n indices.push(nextI * stride + nextJ);\n indices.push(nextI * stride + j);\n }\n }\n // Sides\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\n // Result\n const vertexData = new VertexData();\n vertexData.indices = indices;\n vertexData.positions = positions;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n return vertexData;\n }\n /**\n * Creates a torus mesh\n * * The parameter `diameter` sets the diameter size (float) of the torus (default 1)\n * * The parameter `thickness` sets the diameter size of the tube of the torus (float, default 0.5)\n * * The parameter `tessellation` sets the number of torus sides (positive integer, default 16)\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param options.diameter\n * @param options.thickness\n * @param options.tessellation\n * @param options.updatable\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param scene defines the hosting scene\n * @returns the torus mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#torus\n */\n function CreateTorus(name, options = {}, scene) {\n const torus = new Mesh(name, scene);\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n torus._originalBuilderSideOrientation = options.sideOrientation;\n const vertexData = CreateTorusVertexData(options);\n vertexData.applyToMesh(torus, options.updatable);\n return torus;\n }\n VertexData.CreateTorus = CreateTorusVertexData;\n Mesh.CreateTorus = (name, diameter, thickness, tessellation, scene, updatable, sideOrientation) => {\n const options = {\n diameter,\n thickness,\n tessellation,\n sideOrientation,\n updatable,\n };\n return CreateTorus(name, options, scene);\n };\n\n class VRExperienceHelperGazer {\n constructor(scene, gazeTrackerToClone = null) {\n this.scene = scene;\n /** @internal */\n this._pointerDownOnMeshAsked = false;\n /** @internal */\n this._isActionableMesh = false;\n /** @internal */\n this._teleportationRequestInitiated = false;\n /** @internal */\n this._teleportationBackRequestInitiated = false;\n /** @internal */\n this._rotationRightAsked = false;\n /** @internal */\n this._rotationLeftAsked = false;\n /** @internal */\n this._dpadPressed = true;\n /** @internal */\n this._activePointer = false;\n this._id = VRExperienceHelperGazer._IdCounter++;\n // Gaze tracker\n if (!gazeTrackerToClone) {\n this._gazeTracker = CreateTorus(\"gazeTracker\", {\n diameter: 0.0035,\n thickness: 0.0025,\n tessellation: 20,\n updatable: false,\n }, scene);\n this._gazeTracker.bakeCurrentTransformIntoVertices();\n this._gazeTracker.isPickable = false;\n this._gazeTracker.isVisible = false;\n const targetMat = new StandardMaterial(\"targetMat\", scene);\n targetMat.specularColor = Color3.Black();\n targetMat.emissiveColor = new Color3(0.7, 0.7, 0.7);\n targetMat.backFaceCulling = false;\n this._gazeTracker.material = targetMat;\n }\n else {\n this._gazeTracker = gazeTrackerToClone.clone(\"gazeTracker\");\n }\n }\n /**\n * @internal\n */\n _getForwardRay(length) {\n return new Ray(Vector3.Zero(), new Vector3(0, 0, length));\n }\n /** @internal */\n _selectionPointerDown() {\n this._pointerDownOnMeshAsked = true;\n if (this._currentHit) {\n this.scene.simulatePointerDown(this._currentHit, { pointerId: this._id });\n }\n }\n /** @internal */\n _selectionPointerUp() {\n if (this._currentHit) {\n this.scene.simulatePointerUp(this._currentHit, { pointerId: this._id });\n }\n this._pointerDownOnMeshAsked = false;\n }\n /** @internal */\n _activatePointer() {\n this._activePointer = true;\n }\n /** @internal */\n _deactivatePointer() {\n this._activePointer = false;\n }\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _updatePointerDistance(distance = 100) { }\n dispose() {\n this._interactionsEnabled = false;\n this._teleportationEnabled = false;\n if (this._gazeTracker) {\n this._gazeTracker.dispose();\n }\n }\n }\n VRExperienceHelperGazer._IdCounter = 0;\n class VRExperienceHelperControllerGazer extends VRExperienceHelperGazer {\n constructor(webVRController, scene, gazeTrackerToClone) {\n super(scene, gazeTrackerToClone);\n this.webVRController = webVRController;\n // Laser pointer\n this._laserPointer = CreateCylinder(\"laserPointer\", {\n updatable: false,\n height: 1,\n diameterTop: 0.004,\n diameterBottom: 0.0002,\n tessellation: 20,\n subdivisions: 1,\n }, scene);\n const laserPointerMaterial = new StandardMaterial(\"laserPointerMat\", scene);\n laserPointerMaterial.emissiveColor = new Color3(0.7, 0.7, 0.7);\n laserPointerMaterial.alpha = 0.6;\n this._laserPointer.material = laserPointerMaterial;\n this._laserPointer.rotation.x = Math.PI / 2;\n this._laserPointer.position.z = -0.5;\n this._laserPointer.isVisible = false;\n this._laserPointer.isPickable = false;\n if (!webVRController.mesh) {\n // Create an empty mesh that is used prior to loading the high quality model\n const preloadMesh = new Mesh(\"preloadControllerMesh\", scene);\n const preloadPointerPose = new Mesh(PoseEnabledController.POINTING_POSE, scene);\n preloadPointerPose.rotation.x = -0.7;\n preloadMesh.addChild(preloadPointerPose);\n webVRController.attachToMesh(preloadMesh);\n }\n this._setLaserPointerParent(webVRController.mesh);\n this._meshAttachedObserver = webVRController._meshAttachedObservable.add((mesh) => {\n this._setLaserPointerParent(mesh);\n });\n }\n _getForwardRay(length) {\n return this.webVRController.getForwardRay(length);\n }\n /** @internal */\n _activatePointer() {\n super._activatePointer();\n this._laserPointer.isVisible = true;\n }\n /** @internal */\n _deactivatePointer() {\n super._deactivatePointer();\n this._laserPointer.isVisible = false;\n }\n /**\n * @internal\n */\n _setLaserPointerColor(color) {\n this._laserPointer.material.emissiveColor = color;\n }\n /**\n * @internal\n */\n _setLaserPointerLightingDisabled(disabled) {\n this._laserPointer.material.disableLighting = disabled;\n }\n /**\n * @internal\n */\n _setLaserPointerParent(mesh) {\n const makeNotPick = (root) => {\n root.isPickable = false;\n root.getChildMeshes().forEach((c) => {\n makeNotPick(c);\n });\n };\n makeNotPick(mesh);\n const meshChildren = mesh.getChildren(undefined, false);\n let laserParent = mesh;\n this.webVRController._pointingPoseNode = null;\n for (let i = 0; i < meshChildren.length; i++) {\n if (meshChildren[i].name && meshChildren[i].name.indexOf(PoseEnabledController.POINTING_POSE) >= 0) {\n laserParent = meshChildren[i];\n this.webVRController._pointingPoseNode = laserParent;\n break;\n }\n }\n this._laserPointer.parent = laserParent;\n }\n _updatePointerDistance(distance = 100) {\n this._laserPointer.scaling.y = distance;\n this._laserPointer.position.z = -distance / 2;\n }\n dispose() {\n super.dispose();\n this._laserPointer.dispose();\n if (this._meshAttachedObserver) {\n this.webVRController._meshAttachedObservable.remove(this._meshAttachedObserver);\n }\n }\n }\n class VRExperienceHelperCameraGazer extends VRExperienceHelperGazer {\n constructor(_getCamera, scene) {\n super(scene);\n this._getCamera = _getCamera;\n }\n _getForwardRay(length) {\n const camera = this._getCamera();\n if (camera) {\n return camera.getForwardRay(length);\n }\n else {\n return new Ray(Vector3.Zero(), Vector3.Forward());\n }\n }\n }\n /**\n * Helps to quickly add VR support to an existing scene.\n * See https://doc.babylonjs.com/features/featuresDeepDive/cameras/webVRHelper\n * @deprecated\n */\n class VRExperienceHelper {\n /** Return this.onEnteringVRObservable\n * Note: This one is for backward compatibility. Please use onEnteringVRObservable directly\n */\n get onEnteringVR() {\n return this.onEnteringVRObservable;\n }\n /** Return this.onExitingVRObservable\n * Note: This one is for backward compatibility. Please use onExitingVRObservable directly\n */\n get onExitingVR() {\n return this.onExitingVRObservable;\n }\n /** Return this.onControllerMeshLoadedObservable\n * Note: This one is for backward compatibility. Please use onControllerMeshLoadedObservable directly\n */\n get onControllerMeshLoaded() {\n return this.onControllerMeshLoadedObservable;\n }\n /**\n * The mesh used to display where the user is going to teleport.\n */\n get teleportationTarget() {\n return this._teleportationTarget;\n }\n /**\n * Sets the mesh to be used to display where the user is going to teleport.\n */\n set teleportationTarget(value) {\n if (value) {\n value.name = \"teleportationTarget\";\n this._isDefaultTeleportationTarget = false;\n this._teleportationTarget = value;\n }\n }\n /**\n * The mesh used to display where the user is selecting, this mesh will be cloned and set as the gazeTracker for the left and right controller\n * when set bakeCurrentTransformIntoVertices will be called on the mesh.\n * See https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/center_origin/bakingTransforms\n */\n get gazeTrackerMesh() {\n return this._cameraGazer._gazeTracker;\n }\n set gazeTrackerMesh(value) {\n if (value) {\n // Dispose of existing meshes\n if (this._cameraGazer._gazeTracker) {\n this._cameraGazer._gazeTracker.dispose();\n }\n if (this._leftController && this._leftController._gazeTracker) {\n this._leftController._gazeTracker.dispose();\n }\n if (this._rightController && this._rightController._gazeTracker) {\n this._rightController._gazeTracker.dispose();\n }\n // Set and create gaze trackers on head and controllers\n this._cameraGazer._gazeTracker = value;\n this._cameraGazer._gazeTracker.bakeCurrentTransformIntoVertices();\n this._cameraGazer._gazeTracker.isPickable = false;\n this._cameraGazer._gazeTracker.isVisible = false;\n this._cameraGazer._gazeTracker.name = \"gazeTracker\";\n if (this._leftController) {\n this._leftController._gazeTracker = this._cameraGazer._gazeTracker.clone(\"gazeTracker\");\n }\n if (this._rightController) {\n this._rightController._gazeTracker = this._cameraGazer._gazeTracker.clone(\"gazeTracker\");\n }\n }\n }\n /**\n * The gaze tracking mesh corresponding to the left controller\n */\n get leftControllerGazeTrackerMesh() {\n if (this._leftController) {\n return this._leftController._gazeTracker;\n }\n return null;\n }\n /**\n * The gaze tracking mesh corresponding to the right controller\n */\n get rightControllerGazeTrackerMesh() {\n if (this._rightController) {\n return this._rightController._gazeTracker;\n }\n return null;\n }\n /**\n * If the ray of the gaze should be displayed.\n */\n get displayGaze() {\n return this._displayGaze;\n }\n /**\n * Sets if the ray of the gaze should be displayed.\n */\n set displayGaze(value) {\n this._displayGaze = value;\n if (!value) {\n this._cameraGazer._gazeTracker.isVisible = false;\n if (this._leftController) {\n this._leftController._gazeTracker.isVisible = false;\n }\n if (this._rightController) {\n this._rightController._gazeTracker.isVisible = false;\n }\n }\n }\n /**\n * If the ray of the LaserPointer should be displayed.\n */\n get displayLaserPointer() {\n return this._displayLaserPointer;\n }\n /**\n * Sets if the ray of the LaserPointer should be displayed.\n */\n set displayLaserPointer(value) {\n this._displayLaserPointer = value;\n if (!value) {\n if (this._rightController) {\n this._rightController._deactivatePointer();\n this._rightController._gazeTracker.isVisible = false;\n }\n if (this._leftController) {\n this._leftController._deactivatePointer();\n this._leftController._gazeTracker.isVisible = false;\n }\n }\n else {\n if (this._rightController) {\n this._rightController._activatePointer();\n }\n if (this._leftController) {\n this._leftController._activatePointer();\n }\n }\n }\n /**\n * The deviceOrientationCamera used as the camera when not in VR.\n */\n get deviceOrientationCamera() {\n return this._deviceOrientationCamera;\n }\n /**\n * Based on the current WebVR support, returns the current VR camera used.\n */\n get currentVRCamera() {\n if (this._webVRready) {\n return this._webVRCamera;\n }\n else {\n return this._scene.activeCamera;\n }\n }\n /**\n * The webVRCamera which is used when in VR.\n */\n get webVRCamera() {\n return this._webVRCamera;\n }\n /**\n * The deviceOrientationCamera that is used as a fallback when vr device is not connected.\n */\n get vrDeviceOrientationCamera() {\n return this._vrDeviceOrientationCamera;\n }\n /**\n * The html button that is used to trigger entering into VR.\n */\n get vrButton() {\n return this._btnVR;\n }\n get _teleportationRequestInitiated() {\n const result = this._cameraGazer._teleportationRequestInitiated ||\n (this._leftController !== null && this._leftController._teleportationRequestInitiated) ||\n (this._rightController !== null && this._rightController._teleportationRequestInitiated);\n return result;\n }\n /**\n * Instantiates a VRExperienceHelper.\n * Helps to quickly add VR support to an existing scene.\n * @param scene The scene the VRExperienceHelper belongs to.\n * @param webVROptions Options to modify the vr experience helper's behavior.\n */\n constructor(scene, \n /** Options to modify the vr experience helper's behavior. */\n webVROptions = {}) {\n this.webVROptions = webVROptions;\n // Can the system support WebVR, even if a headset isn't plugged in?\n this._webVRsupported = false;\n // If WebVR is supported, is a headset plugged in and are we ready to present?\n this._webVRready = false;\n // Are we waiting for the requestPresent callback to complete?\n this._webVRrequesting = false;\n // Are we presenting to the headset right now? (this is the vrDevice state)\n this._webVRpresenting = false;\n // Are we presenting in the fullscreen fallback?\n this._fullscreenVRpresenting = false;\n /**\n * Gets or sets a boolean indicating that gaze can be enabled even if pointer lock is not engage (useful on iOS where fullscreen mode and pointer lock are not supported)\n */\n this.enableGazeEvenWhenNoPointerLock = false;\n /**\n * Gets or sets a boolean indicating that the VREXperienceHelper will exit VR if double tap is detected\n */\n this.exitVROnDoubleTap = true;\n /**\n * Observable raised right before entering VR.\n */\n this.onEnteringVRObservable = new Observable$1();\n /**\n * Observable raised when entering VR has completed.\n */\n this.onAfterEnteringVRObservable = new Observable$1();\n /**\n * Observable raised when exiting VR.\n */\n this.onExitingVRObservable = new Observable$1();\n /**\n * Observable raised when controller mesh is loaded.\n */\n this.onControllerMeshLoadedObservable = new Observable$1();\n this._useCustomVRButton = false;\n this._teleportationRequested = false;\n this._teleportActive = false;\n this._floorMeshesCollection = [];\n this._teleportationMode = VRExperienceHelper.TELEPORTATIONMODE_CONSTANTTIME;\n this._teleportationTime = 122;\n this._teleportationSpeed = 20;\n this._rotationAllowed = true;\n this._teleportBackwardsVector = new Vector3(0, -1, -1);\n this._isDefaultTeleportationTarget = true;\n this._teleportationFillColor = \"#444444\";\n this._teleportationBorderColor = \"#FFFFFF\";\n this._rotationAngle = 0;\n this._haloCenter = new Vector3(0, 0, 0);\n this._padSensibilityUp = 0.65;\n this._padSensibilityDown = 0.35;\n this._leftController = null;\n this._rightController = null;\n this._gazeColor = new Color3(0.7, 0.7, 0.7);\n this._laserColor = new Color3(0.7, 0.7, 0.7);\n this._pickedLaserColor = new Color3(0.2, 0.2, 1);\n this._pickedGazeColor = new Color3(0, 0, 1);\n /**\n * Observable raised when a new mesh is selected based on meshSelectionPredicate\n */\n this.onNewMeshSelected = new Observable$1();\n /**\n * Observable raised when a new mesh is selected based on meshSelectionPredicate.\n * This observable will provide the mesh and the controller used to select the mesh\n */\n this.onMeshSelectedWithController = new Observable$1();\n /**\n * Observable raised when a new mesh is picked based on meshSelectionPredicate\n */\n this.onNewMeshPicked = new Observable$1();\n /**\n * Observable raised before camera teleportation\n */\n this.onBeforeCameraTeleport = new Observable$1();\n /**\n * Observable raised after camera teleportation\n */\n this.onAfterCameraTeleport = new Observable$1();\n /**\n * Observable raised when current selected mesh gets unselected\n */\n this.onSelectedMeshUnselected = new Observable$1();\n /**\n * Set teleportation enabled. If set to false camera teleportation will be disabled but camera rotation will be kept.\n */\n this.teleportationEnabled = true;\n this._teleportationInitialized = false;\n this._interactionsEnabled = false;\n this._interactionsRequested = false;\n this._displayGaze = true;\n this._displayLaserPointer = true;\n /**\n * If the gaze trackers scale should be updated to be constant size when pointing at near/far meshes\n */\n this.updateGazeTrackerScale = true;\n /**\n * If the gaze trackers color should be updated when selecting meshes\n */\n this.updateGazeTrackerColor = true;\n /**\n * If the controller laser color should be updated when selecting meshes\n */\n this.updateControllerLaserColor = true;\n /**\n * Defines whether or not Pointer lock should be requested when switching to\n * full screen.\n */\n this.requestPointerLockOnFullScreen = true;\n /**\n * Was the XR test done already. If this is true AND this.xr exists, xr is initialized.\n * If this is true and no this.xr, xr exists but is not supported, using WebVR.\n */\n this.xrTestDone = false;\n this._onResize = () => {\n this._moveButtonToBottomRight();\n if (this._fullscreenVRpresenting && this._webVRready) {\n this.exitVR();\n }\n };\n this._onFullscreenChange = () => {\n this._fullscreenVRpresenting = !!document.fullscreenElement;\n if (!this._fullscreenVRpresenting && this._inputElement) {\n this.exitVR();\n if (!this._useCustomVRButton && this._btnVR) {\n this._btnVR.style.top = this._inputElement.offsetTop + this._inputElement.offsetHeight - 70 + \"px\";\n this._btnVR.style.left = this._inputElement.offsetLeft + this._inputElement.offsetWidth - 100 + \"px\";\n // make sure the button is visible after setting its position\n this._updateButtonVisibility();\n }\n }\n };\n this._cachedAngularSensibility = { angularSensibilityX: null, angularSensibilityY: null, angularSensibility: null };\n this._beforeRender = () => {\n if (this._leftController && this._leftController._activePointer) {\n this._castRayAndSelectObject(this._leftController);\n }\n if (this._rightController && this._rightController._activePointer) {\n this._castRayAndSelectObject(this._rightController);\n }\n if (this._noControllerIsActive && (this._scene.getEngine().isPointerLock || this.enableGazeEvenWhenNoPointerLock)) {\n this._castRayAndSelectObject(this._cameraGazer);\n }\n else {\n this._cameraGazer._gazeTracker.isVisible = false;\n }\n };\n this._onNewGamepadConnected = (gamepad) => {\n if (gamepad.type !== Gamepad.POSE_ENABLED) {\n if (gamepad.leftStick) {\n gamepad.onleftstickchanged((stickValues) => {\n if (this._teleportationInitialized && this.teleportationEnabled) {\n // Listening to classic/xbox gamepad only if no VR controller is active\n if ((!this._leftController && !this._rightController) ||\n (this._leftController && !this._leftController._activePointer && this._rightController && !this._rightController._activePointer)) {\n this._checkTeleportWithRay(stickValues, this._cameraGazer);\n this._checkTeleportBackwards(stickValues, this._cameraGazer);\n }\n }\n });\n }\n if (gamepad.rightStick) {\n gamepad.onrightstickchanged((stickValues) => {\n if (this._teleportationInitialized) {\n this._checkRotate(stickValues, this._cameraGazer);\n }\n });\n }\n if (gamepad.type === Gamepad.XBOX) {\n gamepad.onbuttondown((buttonPressed) => {\n if (this._interactionsEnabled && buttonPressed === Xbox360Button.A) {\n this._cameraGazer._selectionPointerDown();\n }\n });\n gamepad.onbuttonup((buttonPressed) => {\n if (this._interactionsEnabled && buttonPressed === Xbox360Button.A) {\n this._cameraGazer._selectionPointerUp();\n }\n });\n }\n }\n else {\n const webVRController = gamepad;\n const controller = new VRExperienceHelperControllerGazer(webVRController, this._scene, this._cameraGazer._gazeTracker);\n if (webVRController.hand === \"right\" || (this._leftController && this._leftController.webVRController != webVRController)) {\n this._rightController = controller;\n }\n else {\n this._leftController = controller;\n }\n this._tryEnableInteractionOnController(controller);\n }\n };\n // This only succeeds if the controller's mesh exists for the controller so this must be called whenever new controller is connected or when mesh is loaded\n this._tryEnableInteractionOnController = (controller) => {\n if (this._interactionsRequested && !controller._interactionsEnabled) {\n this._enableInteractionOnController(controller);\n }\n if (this._teleportationRequested && !controller._teleportationEnabled) {\n this._enableTeleportationOnController(controller);\n }\n };\n this._onNewGamepadDisconnected = (gamepad) => {\n if (gamepad instanceof WebVRController) {\n if (gamepad.hand === \"left\" && this._leftController != null) {\n this._leftController.dispose();\n this._leftController = null;\n }\n if (gamepad.hand === \"right\" && this._rightController != null) {\n this._rightController.dispose();\n this._rightController = null;\n }\n }\n };\n this._workingVector = Vector3.Zero();\n this._workingQuaternion = Quaternion.Identity();\n this._workingMatrix = Matrix.Identity();\n Logger.Warn(\"WebVR is deprecated. Please avoid using this experience helper and use the WebXR experience helper instead\");\n this._scene = scene;\n this._inputElement = scene.getEngine().getInputElement();\n // check for VR support:\n const vrSupported = \"getVRDisplays\" in navigator;\n // no VR support? force XR but only when it is not set because web vr can work without the getVRDisplays\n if (!vrSupported && webVROptions.useXR === undefined) {\n webVROptions.useXR = true;\n }\n // Parse options\n if (webVROptions.createFallbackVRDeviceOrientationFreeCamera === undefined) {\n webVROptions.createFallbackVRDeviceOrientationFreeCamera = true;\n }\n if (webVROptions.createDeviceOrientationCamera === undefined) {\n webVROptions.createDeviceOrientationCamera = true;\n }\n if (webVROptions.laserToggle === undefined) {\n webVROptions.laserToggle = true;\n }\n if (webVROptions.defaultHeight === undefined) {\n webVROptions.defaultHeight = 1.7;\n }\n if (webVROptions.useCustomVRButton) {\n this._useCustomVRButton = true;\n if (webVROptions.customVRButton) {\n this._btnVR = webVROptions.customVRButton;\n }\n }\n if (webVROptions.rayLength) {\n this._rayLength = webVROptions.rayLength;\n }\n this._defaultHeight = webVROptions.defaultHeight;\n if (webVROptions.positionScale) {\n this._rayLength *= webVROptions.positionScale;\n this._defaultHeight *= webVROptions.positionScale;\n }\n this._hasEnteredVR = false;\n // Set position\n if (this._scene.activeCamera) {\n this._position = this._scene.activeCamera.position.clone();\n }\n else {\n this._position = new Vector3(0, this._defaultHeight, 0);\n }\n // Set non-vr camera\n if (webVROptions.createDeviceOrientationCamera || !this._scene.activeCamera) {\n this._deviceOrientationCamera = new DeviceOrientationCamera(\"deviceOrientationVRHelper\", this._position.clone(), scene);\n // Copy data from existing camera\n if (this._scene.activeCamera) {\n this._deviceOrientationCamera.minZ = this._scene.activeCamera.minZ;\n this._deviceOrientationCamera.maxZ = this._scene.activeCamera.maxZ;\n // Set rotation from previous camera\n if (this._scene.activeCamera instanceof TargetCamera && this._scene.activeCamera.rotation) {\n const targetCamera = this._scene.activeCamera;\n if (targetCamera.rotationQuaternion) {\n this._deviceOrientationCamera.rotationQuaternion.copyFrom(targetCamera.rotationQuaternion);\n }\n else {\n this._deviceOrientationCamera.rotationQuaternion.copyFrom(Quaternion.RotationYawPitchRoll(targetCamera.rotation.y, targetCamera.rotation.x, targetCamera.rotation.z));\n }\n this._deviceOrientationCamera.rotation = targetCamera.rotation.clone();\n }\n }\n this._scene.activeCamera = this._deviceOrientationCamera;\n if (this._inputElement) {\n this._scene.activeCamera.attachControl();\n }\n }\n else {\n this._existingCamera = this._scene.activeCamera;\n }\n if (this.webVROptions.useXR && navigator.xr) {\n // force-check XR session support\n WebXRSessionManager.IsSessionSupportedAsync(\"immersive-vr\").then((supported) => {\n if (supported) {\n Logger.Log(\"Using WebXR. It is recommended to use the WebXRDefaultExperience directly\");\n // it is possible to use XR, let's do it!\n scene\n .createDefaultXRExperienceAsync({\n floorMeshes: webVROptions.floorMeshes || [],\n })\n .then((xr) => {\n this.xr = xr;\n // connect observables\n this.xrTestDone = true;\n this._cameraGazer = new VRExperienceHelperCameraGazer(() => {\n return this.xr.baseExperience.camera;\n }, scene);\n this.xr.baseExperience.onStateChangedObservable.add((state) => {\n // support for entering / exiting\n switch (state) {\n case WebXRState.ENTERING_XR:\n this.onEnteringVRObservable.notifyObservers(this);\n if (!this._interactionsEnabled) {\n this.xr.pointerSelection.detach();\n }\n this.xr.pointerSelection.displayLaserPointer = this._displayLaserPointer;\n break;\n case WebXRState.EXITING_XR:\n this.onExitingVRObservable.notifyObservers(this);\n // resize to update width and height when exiting vr exits fullscreen\n this._scene.getEngine().resize();\n break;\n case WebXRState.IN_XR:\n this._hasEnteredVR = true;\n break;\n case WebXRState.NOT_IN_XR:\n this._hasEnteredVR = false;\n break;\n }\n });\n });\n }\n else {\n // XR not supported (thou exists), continue WebVR init\n this._completeVRInit(scene, webVROptions);\n }\n });\n }\n else {\n // no XR, continue init synchronous\n this._completeVRInit(scene, webVROptions);\n }\n }\n _completeVRInit(scene, webVROptions) {\n this.xrTestDone = true;\n // Create VR cameras\n if (webVROptions.createFallbackVRDeviceOrientationFreeCamera) {\n if (webVROptions.useMultiview) {\n if (!webVROptions.vrDeviceOrientationCameraMetrics) {\n webVROptions.vrDeviceOrientationCameraMetrics = VRCameraMetrics.GetDefault();\n }\n webVROptions.vrDeviceOrientationCameraMetrics.multiviewEnabled = true;\n }\n this._vrDeviceOrientationCamera = new VRDeviceOrientationFreeCamera(\"VRDeviceOrientationVRHelper\", this._position, this._scene, true, webVROptions.vrDeviceOrientationCameraMetrics);\n this._vrDeviceOrientationCamera.angularSensibility = Number.MAX_VALUE;\n }\n this._webVRCamera = new WebVRFreeCamera(\"WebVRHelper\", this._position, this._scene, webVROptions);\n this._webVRCamera.useStandingMatrix();\n this._cameraGazer = new VRExperienceHelperCameraGazer(() => {\n return this.currentVRCamera;\n }, scene);\n // Create default button\n if (!this._useCustomVRButton) {\n this._btnVR = document.createElement(\"BUTTON\");\n this._btnVR.className = \"babylonVRicon\";\n this._btnVR.id = \"babylonVRiconbtn\";\n this._btnVR.title = \"Click to switch to VR\";\n const url = !window.SVGSVGElement\n ? \"https://cdn.babylonjs.com/Assets/vrButton.png\"\n : \"data:image/svg+xml;charset=UTF-8,%3Csvg%20xmlns%3D%22http%3A//www.w3.org/2000/svg%22%20width%3D%222048%22%20height%3D%221152%22%20viewBox%3D%220%200%202048%201152%22%20version%3D%221.1%22%3E%3Cpath%20transform%3D%22rotate%28180%201024%2C576.0000000000001%29%22%20d%3D%22m1109%2C896q17%2C0%2030%2C-12t13%2C-30t-12.5%2C-30.5t-30.5%2C-12.5l-170%2C0q-18%2C0%20-30.5%2C12.5t-12.5%2C30.5t13%2C30t30%2C12l170%2C0zm-85%2C256q59%2C0%20132.5%2C-1.5t154.5%2C-5.5t164.5%2C-11.5t163%2C-20t150%2C-30t124.5%2C-41.5q23%2C-11%2042%2C-24t38%2C-30q27%2C-25%2041%2C-61.5t14%2C-72.5l0%2C-257q0%2C-123%20-47%2C-232t-128%2C-190t-190%2C-128t-232%2C-47l-81%2C0q-37%2C0%20-68.5%2C14t-60.5%2C34.5t-55.5%2C45t-53%2C45t-53%2C34.5t-55.5%2C14t-55.5%2C-14t-53%2C-34.5t-53%2C-45t-55.5%2C-45t-60.5%2C-34.5t-68.5%2C-14l-81%2C0q-123%2C0%20-232%2C47t-190%2C128t-128%2C190t-47%2C232l0%2C257q0%2C68%2038%2C115t97%2C73q54%2C24%20124.5%2C41.5t150%2C30t163%2C20t164.5%2C11.5t154.5%2C5.5t132.5%2C1.5zm939%2C-298q0%2C39%20-24.5%2C67t-58.5%2C42q-54%2C23%20-122%2C39.5t-143.5%2C28t-155.5%2C19t-157%2C11t-148.5%2C5t-129.5%2C1.5q-59%2C0%20-130%2C-1.5t-148%2C-5t-157%2C-11t-155.5%2C-19t-143.5%2C-28t-122%2C-39.5q-34%2C-14%20-58.5%2C-42t-24.5%2C-67l0%2C-257q0%2C-106%2040.5%2C-199t110%2C-162.5t162.5%2C-109.5t199%2C-40l81%2C0q27%2C0%2052%2C14t50%2C34.5t51%2C44.5t55.5%2C44.5t63.5%2C34.5t74%2C14t74%2C-14t63.5%2C-34.5t55.5%2C-44.5t51%2C-44.5t50%2C-34.5t52%2C-14l14%2C0q37%2C0%2070%2C0.5t64.5%2C4.5t63.5%2C12t68%2C23q71%2C30%20128.5%2C78.5t98.5%2C110t63.5%2C133.5t22.5%2C149l0%2C257z%22%20fill%3D%22white%22%20/%3E%3C/svg%3E%0A\";\n let css = \".babylonVRicon { position: absolute; right: 20px; height: 50px; width: 80px; background-color: rgba(51,51,51,0.7); background-image: url(\" +\n url +\n \"); background-size: 80%; background-repeat:no-repeat; background-position: center; border: none; outline: none; transition: transform 0.125s ease-out } .babylonVRicon:hover { transform: scale(1.05) } .babylonVRicon:active {background-color: rgba(51,51,51,1) } .babylonVRicon:focus {background-color: rgba(51,51,51,1) }\";\n css += \".babylonVRicon.vrdisplaypresenting { display: none; }\";\n // TODO: Add user feedback so that they know what state the VRDisplay is in (disconnected, connected, entering-VR)\n // css += \".babylonVRicon.vrdisplaysupported { }\";\n // css += \".babylonVRicon.vrdisplayready { }\";\n // css += \".babylonVRicon.vrdisplayrequesting { }\";\n const style = document.createElement(\"style\");\n style.appendChild(document.createTextNode(css));\n document.getElementsByTagName(\"head\")[0].appendChild(style);\n this._moveButtonToBottomRight();\n }\n // VR button click event\n if (this._btnVR) {\n this._btnVR.addEventListener(\"click\", () => {\n if (!this.isInVRMode) {\n this.enterVR();\n }\n else {\n this._scene.getEngine().disableVR();\n }\n });\n }\n // Window events\n const hostWindow = this._scene.getEngine().getHostWindow();\n if (!hostWindow) {\n return;\n }\n hostWindow.addEventListener(\"resize\", this._onResize);\n document.addEventListener(\"fullscreenchange\", this._onFullscreenChange, false);\n // Display vr button when headset is connected\n if (webVROptions.createFallbackVRDeviceOrientationFreeCamera) {\n this._displayVRButton();\n }\n else {\n this._scene.getEngine().onVRDisplayChangedObservable.add((e) => {\n if (e.vrDisplay) {\n this._displayVRButton();\n }\n });\n }\n // Exiting VR mode using 'ESC' key on desktop\n this._onKeyDown = (event) => {\n if (event.keyCode === 27 && this.isInVRMode) {\n this.exitVR();\n }\n };\n document.addEventListener(\"keydown\", this._onKeyDown);\n // Exiting VR mode double tapping the touch screen\n this._scene.onPrePointerObservable.add(() => {\n if (this._hasEnteredVR && this.exitVROnDoubleTap) {\n this.exitVR();\n if (this._fullscreenVRpresenting) {\n this._scene.getEngine().exitFullscreen();\n }\n }\n }, PointerEventTypes.POINTERDOUBLETAP, false);\n // Listen for WebVR display changes\n this._onVRDisplayChangedBind = (eventArgs) => this._onVRDisplayChanged(eventArgs);\n this._onVrDisplayPresentChangeBind = () => this._onVrDisplayPresentChange();\n this._onVRRequestPresentStart = () => {\n this._webVRrequesting = true;\n this._updateButtonVisibility();\n };\n this._onVRRequestPresentComplete = () => {\n this._webVRrequesting = false;\n this._updateButtonVisibility();\n };\n scene.getEngine().onVRDisplayChangedObservable.add(this._onVRDisplayChangedBind);\n scene.getEngine().onVRRequestPresentStart.add(this._onVRRequestPresentStart);\n scene.getEngine().onVRRequestPresentComplete.add(this._onVRRequestPresentComplete);\n hostWindow.addEventListener(\"vrdisplaypresentchange\", this._onVrDisplayPresentChangeBind);\n scene.onDisposeObservable.add(() => {\n this.dispose();\n });\n // Gamepad connection events\n this._webVRCamera.onControllerMeshLoadedObservable.add((webVRController) => this._onDefaultMeshLoaded(webVRController));\n this._scene.gamepadManager.onGamepadConnectedObservable.add(this._onNewGamepadConnected);\n this._scene.gamepadManager.onGamepadDisconnectedObservable.add(this._onNewGamepadDisconnected);\n this._updateButtonVisibility();\n //create easing functions\n this._circleEase = new CircleEase();\n this._circleEase.setEasingMode(EasingFunction.EASINGMODE_EASEINOUT);\n this._teleportationEasing = this._circleEase;\n // Allow clicking in the vrDeviceOrientationCamera\n scene.onPointerObservable.add((e) => {\n if (this._interactionsEnabled) {\n if (scene.activeCamera === this.vrDeviceOrientationCamera && e.event.pointerType === \"mouse\") {\n if (e.type === PointerEventTypes.POINTERDOWN) {\n this._cameraGazer._selectionPointerDown();\n }\n else if (e.type === PointerEventTypes.POINTERUP) {\n this._cameraGazer._selectionPointerUp();\n }\n }\n }\n });\n if (this.webVROptions.floorMeshes) {\n this.enableTeleportation({ floorMeshes: this.webVROptions.floorMeshes });\n }\n }\n // Raised when one of the controller has loaded successfully its associated default mesh\n _onDefaultMeshLoaded(webVRController) {\n if (this._leftController && this._leftController.webVRController == webVRController) {\n if (webVRController.mesh) {\n this._leftController._setLaserPointerParent(webVRController.mesh);\n }\n }\n if (this._rightController && this._rightController.webVRController == webVRController) {\n if (webVRController.mesh) {\n this._rightController._setLaserPointerParent(webVRController.mesh);\n }\n }\n try {\n this.onControllerMeshLoadedObservable.notifyObservers(webVRController);\n }\n catch (err) {\n Logger.Warn(\"Error in your custom logic onControllerMeshLoaded: \" + err);\n }\n }\n /**\n * Gets a value indicating if we are currently in VR mode.\n */\n get isInVRMode() {\n return (this.xr && this.webVROptions.useXR && this.xr.baseExperience.state === WebXRState.IN_XR) || this._webVRpresenting || this._fullscreenVRpresenting;\n }\n _onVrDisplayPresentChange() {\n const vrDisplay = this._scene.getEngine().getVRDevice();\n if (vrDisplay) {\n const wasPresenting = this._webVRpresenting;\n this._webVRpresenting = vrDisplay.isPresenting;\n if (wasPresenting && !this._webVRpresenting) {\n this.exitVR();\n }\n }\n else {\n Logger.Warn(\"Detected VRDisplayPresentChange on an unknown VRDisplay. Did you can enterVR on the vrExperienceHelper?\");\n }\n this._updateButtonVisibility();\n }\n _onVRDisplayChanged(eventArgs) {\n this._webVRsupported = eventArgs.vrSupported;\n this._webVRready = !!eventArgs.vrDisplay;\n this._webVRpresenting = eventArgs.vrDisplay && eventArgs.vrDisplay.isPresenting;\n this._updateButtonVisibility();\n }\n _moveButtonToBottomRight() {\n if (this._inputElement && !this._useCustomVRButton && this._btnVR) {\n const rect = this._inputElement.getBoundingClientRect();\n this._btnVR.style.top = rect.top + rect.height - 70 + \"px\";\n this._btnVR.style.left = rect.left + rect.width - 100 + \"px\";\n }\n }\n _displayVRButton() {\n if (!this._useCustomVRButton && !this._btnVRDisplayed && this._btnVR) {\n document.body.appendChild(this._btnVR);\n this._btnVRDisplayed = true;\n }\n }\n _updateButtonVisibility() {\n if (!this._btnVR || this._useCustomVRButton) {\n return;\n }\n this._btnVR.className = \"babylonVRicon\";\n if (this.isInVRMode) {\n this._btnVR.className += \" vrdisplaypresenting\";\n }\n else {\n if (this._webVRready) {\n this._btnVR.className += \" vrdisplayready\";\n }\n if (this._webVRsupported) {\n this._btnVR.className += \" vrdisplaysupported\";\n }\n if (this._webVRrequesting) {\n this._btnVR.className += \" vrdisplayrequesting\";\n }\n }\n }\n /**\n * Attempt to enter VR. If a headset is connected and ready, will request present on that.\n * Otherwise, will use the fullscreen API.\n */\n enterVR() {\n if (this.xr) {\n this.xr.baseExperience.enterXRAsync(\"immersive-vr\", \"local-floor\", this.xr.renderTarget);\n return;\n }\n if (this.onEnteringVRObservable) {\n try {\n this.onEnteringVRObservable.notifyObservers(this);\n }\n catch (err) {\n Logger.Warn(\"Error in your custom logic onEnteringVR: \" + err);\n }\n }\n if (this._scene.activeCamera) {\n this._position = this._scene.activeCamera.position.clone();\n if (this.vrDeviceOrientationCamera) {\n this.vrDeviceOrientationCamera.rotation = Quaternion.FromRotationMatrix(this._scene.activeCamera.getWorldMatrix().getRotationMatrix()).toEulerAngles();\n this.vrDeviceOrientationCamera.angularSensibility = 2000;\n }\n if (this.webVRCamera) {\n const currentYRotation = this.webVRCamera.deviceRotationQuaternion.toEulerAngles().y;\n const desiredYRotation = Quaternion.FromRotationMatrix(this._scene.activeCamera.getWorldMatrix().getRotationMatrix()).toEulerAngles().y;\n const delta = desiredYRotation - currentYRotation;\n const currentGlobalRotation = this.webVRCamera.rotationQuaternion.toEulerAngles().y;\n this.webVRCamera.rotationQuaternion = Quaternion.FromEulerAngles(0, currentGlobalRotation + delta, 0);\n }\n // make sure that we return to the last active camera\n this._existingCamera = this._scene.activeCamera;\n // Remove and cache angular sensability to avoid camera rotation when in VR\n if (this._existingCamera.angularSensibilityX) {\n this._cachedAngularSensibility.angularSensibilityX = this._existingCamera.angularSensibilityX;\n this._existingCamera.angularSensibilityX = Number.MAX_VALUE;\n }\n if (this._existingCamera.angularSensibilityY) {\n this._cachedAngularSensibility.angularSensibilityY = this._existingCamera.angularSensibilityY;\n this._existingCamera.angularSensibilityY = Number.MAX_VALUE;\n }\n if (this._existingCamera.angularSensibility) {\n this._cachedAngularSensibility.angularSensibility = this._existingCamera.angularSensibility;\n this._existingCamera.angularSensibility = Number.MAX_VALUE;\n }\n }\n if (this._webVRrequesting) {\n return;\n }\n // If WebVR is supported and a headset is connected\n if (this._webVRready) {\n if (!this._webVRpresenting) {\n this._scene.getEngine().onVRRequestPresentComplete.addOnce((result) => {\n this.onAfterEnteringVRObservable.notifyObservers({ success: result });\n });\n this._webVRCamera.position = this._position;\n this._scene.activeCamera = this._webVRCamera;\n }\n }\n else if (this._vrDeviceOrientationCamera) {\n this._vrDeviceOrientationCamera.position = this._position;\n if (this._scene.activeCamera) {\n this._vrDeviceOrientationCamera.minZ = this._scene.activeCamera.minZ;\n }\n this._scene.activeCamera = this._vrDeviceOrientationCamera;\n this._scene.getEngine().enterFullscreen(this.requestPointerLockOnFullScreen);\n this._updateButtonVisibility();\n this._vrDeviceOrientationCamera.onViewMatrixChangedObservable.addOnce(() => {\n this.onAfterEnteringVRObservable.notifyObservers({ success: true });\n });\n }\n if (this._scene.activeCamera && this._inputElement) {\n this._scene.activeCamera.attachControl();\n }\n if (this._interactionsEnabled) {\n this._scene.registerBeforeRender(this._beforeRender);\n }\n if (this._displayLaserPointer) {\n [this._leftController, this._rightController].forEach((controller) => {\n if (controller) {\n controller._activatePointer();\n }\n });\n }\n this._hasEnteredVR = true;\n }\n /**\n * Attempt to exit VR, or fullscreen.\n */\n exitVR() {\n if (this.xr) {\n this.xr.baseExperience.exitXRAsync();\n return;\n }\n if (this._hasEnteredVR) {\n if (this.onExitingVRObservable) {\n try {\n this.onExitingVRObservable.notifyObservers(this);\n }\n catch (err) {\n Logger.Warn(\"Error in your custom logic onExitingVR: \" + err);\n }\n }\n if (this._webVRpresenting) {\n this._scene.getEngine().disableVR();\n }\n if (this._scene.activeCamera) {\n this._position = this._scene.activeCamera.position.clone();\n }\n if (this.vrDeviceOrientationCamera) {\n this.vrDeviceOrientationCamera.angularSensibility = Number.MAX_VALUE;\n }\n if (this._deviceOrientationCamera) {\n this._deviceOrientationCamera.position = this._position;\n this._scene.activeCamera = this._deviceOrientationCamera;\n // Restore angular sensibility\n if (this._cachedAngularSensibility.angularSensibilityX) {\n this._deviceOrientationCamera.angularSensibilityX = this._cachedAngularSensibility.angularSensibilityX;\n this._cachedAngularSensibility.angularSensibilityX = null;\n }\n if (this._cachedAngularSensibility.angularSensibilityY) {\n this._deviceOrientationCamera.angularSensibilityY = this._cachedAngularSensibility.angularSensibilityY;\n this._cachedAngularSensibility.angularSensibilityY = null;\n }\n if (this._cachedAngularSensibility.angularSensibility) {\n this._deviceOrientationCamera.angularSensibility = this._cachedAngularSensibility.angularSensibility;\n this._cachedAngularSensibility.angularSensibility = null;\n }\n }\n else if (this._existingCamera) {\n this._existingCamera.position = this._position;\n this._scene.activeCamera = this._existingCamera;\n if (this._inputElement) {\n this._scene.activeCamera.attachControl();\n }\n // Restore angular sensibility\n if (this._cachedAngularSensibility.angularSensibilityX) {\n this._existingCamera.angularSensibilityX = this._cachedAngularSensibility.angularSensibilityX;\n this._cachedAngularSensibility.angularSensibilityX = null;\n }\n if (this._cachedAngularSensibility.angularSensibilityY) {\n this._existingCamera.angularSensibilityY = this._cachedAngularSensibility.angularSensibilityY;\n this._cachedAngularSensibility.angularSensibilityY = null;\n }\n if (this._cachedAngularSensibility.angularSensibility) {\n this._existingCamera.angularSensibility = this._cachedAngularSensibility.angularSensibility;\n this._cachedAngularSensibility.angularSensibility = null;\n }\n }\n this._updateButtonVisibility();\n if (this._interactionsEnabled) {\n this._scene.unregisterBeforeRender(this._beforeRender);\n this._cameraGazer._gazeTracker.isVisible = false;\n if (this._leftController) {\n this._leftController._gazeTracker.isVisible = false;\n }\n if (this._rightController) {\n this._rightController._gazeTracker.isVisible = false;\n }\n }\n // resize to update width and height when exiting vr exits fullscreen\n this._scene.getEngine().resize();\n [this._leftController, this._rightController].forEach((controller) => {\n if (controller) {\n controller._deactivatePointer();\n }\n });\n this._hasEnteredVR = false;\n // Update engine state to re enable non-vr camera input\n const engine = this._scene.getEngine();\n if (engine._onVrDisplayPresentChange) {\n engine._onVrDisplayPresentChange();\n }\n }\n }\n /**\n * The position of the vr experience helper.\n */\n get position() {\n return this._position;\n }\n /**\n * Sets the position of the vr experience helper.\n */\n set position(value) {\n this._position = value;\n if (this._scene.activeCamera) {\n this._scene.activeCamera.position = value;\n }\n }\n /**\n * Enables controllers and user interactions such as selecting and object or clicking on an object.\n */\n enableInteractions() {\n if (!this._interactionsEnabled) {\n this._interactionsRequested = true;\n // in XR it is enabled by default, but just to make sure, re-attach\n if (this.xr) {\n if (this.xr.baseExperience.state === WebXRState.IN_XR) {\n this.xr.pointerSelection.attach();\n }\n return;\n }\n if (this._leftController) {\n this._enableInteractionOnController(this._leftController);\n }\n if (this._rightController) {\n this._enableInteractionOnController(this._rightController);\n }\n this.raySelectionPredicate = (mesh) => {\n return mesh.isVisible && (mesh.isPickable || mesh.name === this._floorMeshName);\n };\n this.meshSelectionPredicate = () => {\n return true;\n };\n this._raySelectionPredicate = (mesh) => {\n if (this._isTeleportationFloor(mesh) ||\n (mesh.name.indexOf(\"gazeTracker\") === -1 && mesh.name.indexOf(\"teleportationTarget\") === -1 && mesh.name.indexOf(\"torusTeleportation\") === -1)) {\n return this.raySelectionPredicate(mesh);\n }\n return false;\n };\n this._interactionsEnabled = true;\n }\n }\n get _noControllerIsActive() {\n return !(this._leftController && this._leftController._activePointer) && !(this._rightController && this._rightController._activePointer);\n }\n _isTeleportationFloor(mesh) {\n for (let i = 0; i < this._floorMeshesCollection.length; i++) {\n if (this._floorMeshesCollection[i].id === mesh.id) {\n return true;\n }\n }\n if (this._floorMeshName && mesh.name === this._floorMeshName) {\n return true;\n }\n return false;\n }\n /**\n * Adds a floor mesh to be used for teleportation.\n * @param floorMesh the mesh to be used for teleportation.\n */\n addFloorMesh(floorMesh) {\n if (!this._floorMeshesCollection) {\n return;\n }\n if (this._floorMeshesCollection.indexOf(floorMesh) > -1) {\n return;\n }\n this._floorMeshesCollection.push(floorMesh);\n }\n /**\n * Removes a floor mesh from being used for teleportation.\n * @param floorMesh the mesh to be removed.\n */\n removeFloorMesh(floorMesh) {\n if (!this._floorMeshesCollection) {\n return;\n }\n const meshIndex = this._floorMeshesCollection.indexOf(floorMesh);\n if (meshIndex !== -1) {\n this._floorMeshesCollection.splice(meshIndex, 1);\n }\n }\n /**\n * Enables interactions and teleportation using the VR controllers and gaze.\n * @param vrTeleportationOptions options to modify teleportation behavior.\n */\n enableTeleportation(vrTeleportationOptions = {}) {\n if (!this._teleportationInitialized) {\n this._teleportationRequested = true;\n this.enableInteractions();\n if (this.webVROptions.useXR && (vrTeleportationOptions.floorMeshes || vrTeleportationOptions.floorMeshName)) {\n const floorMeshes = vrTeleportationOptions.floorMeshes || [];\n if (!floorMeshes.length) {\n const floorMesh = this._scene.getMeshByName(vrTeleportationOptions.floorMeshName);\n if (floorMesh) {\n floorMeshes.push(floorMesh);\n }\n }\n if (this.xr) {\n floorMeshes.forEach((mesh) => {\n this.xr.teleportation.addFloorMesh(mesh);\n });\n if (!this.xr.teleportation.attached) {\n this.xr.teleportation.attach();\n }\n return;\n }\n else if (!this.xrTestDone) {\n const waitForXr = () => {\n if (this.xrTestDone) {\n this._scene.unregisterBeforeRender(waitForXr);\n if (this.xr) {\n if (!this.xr.teleportation.attached) {\n this.xr.teleportation.attach();\n }\n }\n else {\n this.enableTeleportation(vrTeleportationOptions);\n }\n }\n };\n this._scene.registerBeforeRender(waitForXr);\n return;\n }\n }\n if (vrTeleportationOptions.floorMeshName) {\n this._floorMeshName = vrTeleportationOptions.floorMeshName;\n }\n if (vrTeleportationOptions.floorMeshes) {\n this._floorMeshesCollection = vrTeleportationOptions.floorMeshes;\n }\n if (vrTeleportationOptions.teleportationMode) {\n this._teleportationMode = vrTeleportationOptions.teleportationMode;\n }\n if (vrTeleportationOptions.teleportationTime && vrTeleportationOptions.teleportationTime > 0) {\n this._teleportationTime = vrTeleportationOptions.teleportationTime;\n }\n if (vrTeleportationOptions.teleportationSpeed && vrTeleportationOptions.teleportationSpeed > 0) {\n this._teleportationSpeed = vrTeleportationOptions.teleportationSpeed;\n }\n if (vrTeleportationOptions.easingFunction !== undefined) {\n this._teleportationEasing = vrTeleportationOptions.easingFunction;\n }\n if (this._leftController != null) {\n this._enableTeleportationOnController(this._leftController);\n }\n if (this._rightController != null) {\n this._enableTeleportationOnController(this._rightController);\n }\n // Creates an image processing post process for the vignette not relying\n // on the main scene configuration for image processing to reduce setup and spaces\n // (gamma/linear) conflicts.\n const imageProcessingConfiguration = new ImageProcessingConfiguration();\n imageProcessingConfiguration.vignetteColor = new Color4(0, 0, 0, 0);\n imageProcessingConfiguration.vignetteEnabled = true;\n this._postProcessMove = new ImageProcessingPostProcess(\"postProcessMove\", 1.0, this._webVRCamera, undefined, undefined, undefined, undefined, imageProcessingConfiguration);\n this._webVRCamera.detachPostProcess(this._postProcessMove);\n this._teleportationInitialized = true;\n if (this._isDefaultTeleportationTarget) {\n this._createTeleportationCircles();\n this._teleportationTarget.scaling.scaleInPlace(this._webVRCamera.deviceScaleFactor);\n }\n }\n }\n _enableInteractionOnController(controller) {\n const controllerMesh = controller.webVRController.mesh;\n if (controllerMesh) {\n controller._interactionsEnabled = true;\n if (this.isInVRMode && this._displayLaserPointer) {\n controller._activatePointer();\n }\n if (this.webVROptions.laserToggle) {\n controller.webVRController.onMainButtonStateChangedObservable.add((stateObject) => {\n // Enabling / disabling laserPointer\n if (this._displayLaserPointer && stateObject.value === 1) {\n if (controller._activePointer) {\n controller._deactivatePointer();\n }\n else {\n controller._activatePointer();\n }\n if (this.displayGaze) {\n controller._gazeTracker.isVisible = controller._activePointer;\n }\n }\n });\n }\n controller.webVRController.onTriggerStateChangedObservable.add((stateObject) => {\n let gazer = controller;\n if (this._noControllerIsActive) {\n gazer = this._cameraGazer;\n }\n if (!gazer._pointerDownOnMeshAsked) {\n if (stateObject.value > this._padSensibilityUp) {\n gazer._selectionPointerDown();\n }\n }\n else if (stateObject.value < this._padSensibilityDown) {\n gazer._selectionPointerUp();\n }\n });\n }\n }\n _checkTeleportWithRay(stateObject, gazer) {\n // Dont teleport if another gaze already requested teleportation\n if (this._teleportationRequestInitiated && !gazer._teleportationRequestInitiated) {\n return;\n }\n if (!gazer._teleportationRequestInitiated) {\n if (stateObject.y < -this._padSensibilityUp && gazer._dpadPressed) {\n gazer._activatePointer();\n gazer._teleportationRequestInitiated = true;\n }\n }\n else {\n // Listening to the proper controller values changes to confirm teleportation\n if (Math.sqrt(stateObject.y * stateObject.y + stateObject.x * stateObject.x) < this._padSensibilityDown) {\n if (this._teleportActive) {\n this.teleportCamera(this._haloCenter);\n }\n gazer._teleportationRequestInitiated = false;\n }\n }\n }\n _checkRotate(stateObject, gazer) {\n // Only rotate when user is not currently selecting a teleportation location\n if (gazer._teleportationRequestInitiated) {\n return;\n }\n if (!gazer._rotationLeftAsked) {\n if (stateObject.x < -this._padSensibilityUp && gazer._dpadPressed) {\n gazer._rotationLeftAsked = true;\n if (this._rotationAllowed) {\n this._rotateCamera(false);\n }\n }\n }\n else {\n if (stateObject.x > -this._padSensibilityDown) {\n gazer._rotationLeftAsked = false;\n }\n }\n if (!gazer._rotationRightAsked) {\n if (stateObject.x > this._padSensibilityUp && gazer._dpadPressed) {\n gazer._rotationRightAsked = true;\n if (this._rotationAllowed) {\n this._rotateCamera(true);\n }\n }\n }\n else {\n if (stateObject.x < this._padSensibilityDown) {\n gazer._rotationRightAsked = false;\n }\n }\n }\n _checkTeleportBackwards(stateObject, gazer) {\n // Only teleport backwards when user is not currently selecting a teleportation location\n if (gazer._teleportationRequestInitiated) {\n return;\n }\n // Teleport backwards\n if (stateObject.y > this._padSensibilityUp && gazer._dpadPressed) {\n if (!gazer._teleportationBackRequestInitiated) {\n if (!this.currentVRCamera) {\n return;\n }\n // Get rotation and position of the current camera\n let rotation = Quaternion.FromRotationMatrix(this.currentVRCamera.getWorldMatrix().getRotationMatrix());\n let position = this.currentVRCamera.position;\n // If the camera has device position, use that instead\n if (this.currentVRCamera.devicePosition && this.currentVRCamera.deviceRotationQuaternion) {\n rotation = this.currentVRCamera.deviceRotationQuaternion;\n position = this.currentVRCamera.devicePosition;\n }\n // Get matrix with only the y rotation of the device rotation\n rotation.toEulerAnglesToRef(this._workingVector);\n this._workingVector.z = 0;\n this._workingVector.x = 0;\n Quaternion.RotationYawPitchRollToRef(this._workingVector.y, this._workingVector.x, this._workingVector.z, this._workingQuaternion);\n this._workingQuaternion.toRotationMatrix(this._workingMatrix);\n // Rotate backwards ray by device rotation to cast at the ground behind the user\n Vector3.TransformCoordinatesToRef(this._teleportBackwardsVector, this._workingMatrix, this._workingVector);\n // Teleport if ray hit the ground and is not to far away eg. backwards off a cliff\n const ray = new Ray(position, this._workingVector);\n const hit = this._scene.pickWithRay(ray, this._raySelectionPredicate);\n if (hit && hit.pickedPoint && hit.pickedMesh && this._isTeleportationFloor(hit.pickedMesh) && hit.distance < 5) {\n this.teleportCamera(hit.pickedPoint);\n }\n gazer._teleportationBackRequestInitiated = true;\n }\n }\n else {\n gazer._teleportationBackRequestInitiated = false;\n }\n }\n _enableTeleportationOnController(controller) {\n const controllerMesh = controller.webVRController.mesh;\n if (controllerMesh) {\n if (!controller._interactionsEnabled) {\n this._enableInteractionOnController(controller);\n }\n controller._interactionsEnabled = true;\n controller._teleportationEnabled = true;\n if (controller.webVRController.controllerType === PoseEnabledControllerType.VIVE) {\n controller._dpadPressed = false;\n controller.webVRController.onPadStateChangedObservable.add((stateObject) => {\n controller._dpadPressed = stateObject.pressed;\n if (!controller._dpadPressed) {\n controller._rotationLeftAsked = false;\n controller._rotationRightAsked = false;\n controller._teleportationBackRequestInitiated = false;\n }\n });\n }\n controller.webVRController.onPadValuesChangedObservable.add((stateObject) => {\n if (this.teleportationEnabled) {\n this._checkTeleportBackwards(stateObject, controller);\n this._checkTeleportWithRay(stateObject, controller);\n }\n this._checkRotate(stateObject, controller);\n });\n }\n }\n _createTeleportationCircles() {\n this._teleportationTarget = CreateGround(\"teleportationTarget\", { width: 2, height: 2, subdivisions: 2 }, this._scene);\n this._teleportationTarget.isPickable = false;\n const length = 512;\n const dynamicTexture = new DynamicTexture(\"DynamicTexture\", length, this._scene, true);\n dynamicTexture.hasAlpha = true;\n const context = dynamicTexture.getContext();\n const centerX = length / 2;\n const centerY = length / 2;\n const radius = 200;\n context.beginPath();\n context.arc(centerX, centerY, radius, 0, 2 * Math.PI, false);\n context.fillStyle = this._teleportationFillColor;\n context.fill();\n context.lineWidth = 10;\n context.strokeStyle = this._teleportationBorderColor;\n context.stroke();\n context.closePath();\n dynamicTexture.update();\n const teleportationCircleMaterial = new StandardMaterial(\"TextPlaneMaterial\", this._scene);\n teleportationCircleMaterial.diffuseTexture = dynamicTexture;\n this._teleportationTarget.material = teleportationCircleMaterial;\n const torus = CreateTorus(\"torusTeleportation\", {\n diameter: 0.75,\n thickness: 0.1,\n tessellation: 25,\n updatable: false,\n }, this._scene);\n torus.isPickable = false;\n torus.parent = this._teleportationTarget;\n const animationInnerCircle = new Animation(\"animationInnerCircle\", \"position.y\", 30, Animation.ANIMATIONTYPE_FLOAT, Animation.ANIMATIONLOOPMODE_CYCLE);\n const keys = [];\n keys.push({\n frame: 0,\n value: 0,\n });\n keys.push({\n frame: 30,\n value: 0.4,\n });\n keys.push({\n frame: 60,\n value: 0,\n });\n animationInnerCircle.setKeys(keys);\n const easingFunction = new SineEase();\n easingFunction.setEasingMode(EasingFunction.EASINGMODE_EASEINOUT);\n animationInnerCircle.setEasingFunction(easingFunction);\n torus.animations = [];\n torus.animations.push(animationInnerCircle);\n this._scene.beginAnimation(torus, 0, 60, true);\n this._hideTeleportationTarget();\n }\n _displayTeleportationTarget() {\n this._teleportActive = true;\n if (this._teleportationInitialized) {\n this._teleportationTarget.isVisible = true;\n if (this._isDefaultTeleportationTarget) {\n this._teleportationTarget.getChildren()[0].isVisible = true;\n }\n }\n }\n _hideTeleportationTarget() {\n this._teleportActive = false;\n if (this._teleportationInitialized) {\n this._teleportationTarget.isVisible = false;\n if (this._isDefaultTeleportationTarget) {\n this._teleportationTarget.getChildren()[0].isVisible = false;\n }\n }\n }\n _rotateCamera(right) {\n if (!(this.currentVRCamera instanceof FreeCamera)) {\n return;\n }\n if (right) {\n this._rotationAngle++;\n }\n else {\n this._rotationAngle--;\n }\n this.currentVRCamera.animations = [];\n const target = Quaternion.FromRotationMatrix(Matrix.RotationY((Math.PI / 4) * this._rotationAngle));\n const animationRotation = new Animation(\"animationRotation\", \"rotationQuaternion\", 90, Animation.ANIMATIONTYPE_QUATERNION, Animation.ANIMATIONLOOPMODE_CONSTANT);\n const animationRotationKeys = [];\n animationRotationKeys.push({\n frame: 0,\n value: this.currentVRCamera.rotationQuaternion,\n });\n animationRotationKeys.push({\n frame: 6,\n value: target,\n });\n animationRotation.setKeys(animationRotationKeys);\n animationRotation.setEasingFunction(this._circleEase);\n this.currentVRCamera.animations.push(animationRotation);\n this._postProcessMove.animations = [];\n const animationPP = new Animation(\"animationPP\", \"vignetteWeight\", 90, Animation.ANIMATIONTYPE_FLOAT, Animation.ANIMATIONLOOPMODE_CONSTANT);\n const vignetteWeightKeys = [];\n vignetteWeightKeys.push({\n frame: 0,\n value: 0,\n });\n vignetteWeightKeys.push({\n frame: 3,\n value: 4,\n });\n vignetteWeightKeys.push({\n frame: 6,\n value: 0,\n });\n animationPP.setKeys(vignetteWeightKeys);\n animationPP.setEasingFunction(this._circleEase);\n this._postProcessMove.animations.push(animationPP);\n const animationPP2 = new Animation(\"animationPP2\", \"vignetteStretch\", 90, Animation.ANIMATIONTYPE_FLOAT, Animation.ANIMATIONLOOPMODE_CONSTANT);\n const vignetteStretchKeys = [];\n vignetteStretchKeys.push({\n frame: 0,\n value: 0,\n });\n vignetteStretchKeys.push({\n frame: 3,\n value: 10,\n });\n vignetteStretchKeys.push({\n frame: 6,\n value: 0,\n });\n animationPP2.setKeys(vignetteStretchKeys);\n animationPP2.setEasingFunction(this._circleEase);\n this._postProcessMove.animations.push(animationPP2);\n this._postProcessMove.imageProcessingConfiguration.vignetteWeight = 0;\n this._postProcessMove.imageProcessingConfiguration.vignetteStretch = 0;\n this._postProcessMove.samples = 4;\n this._webVRCamera.attachPostProcess(this._postProcessMove);\n this._scene.beginAnimation(this._postProcessMove, 0, 6, false, 1, () => {\n this._webVRCamera.detachPostProcess(this._postProcessMove);\n });\n this._scene.beginAnimation(this.currentVRCamera, 0, 6, false, 1);\n }\n _moveTeleportationSelectorTo(hit, gazer, ray) {\n if (hit.pickedPoint) {\n if (gazer._teleportationRequestInitiated) {\n this._displayTeleportationTarget();\n this._haloCenter.copyFrom(hit.pickedPoint);\n this._teleportationTarget.position.copyFrom(hit.pickedPoint);\n }\n const pickNormal = this._convertNormalToDirectionOfRay(hit.getNormal(true, false), ray);\n if (pickNormal) {\n const axis1 = Vector3.Cross(Axis.Y, pickNormal);\n const axis2 = Vector3.Cross(pickNormal, axis1);\n Vector3.RotationFromAxisToRef(axis2, pickNormal, axis1, this._teleportationTarget.rotation);\n }\n this._teleportationTarget.position.y += 0.1;\n }\n }\n /**\n * Teleports the users feet to the desired location\n * @param location The location where the user's feet should be placed\n */\n teleportCamera(location) {\n if (!(this.currentVRCamera instanceof FreeCamera)) {\n return;\n }\n // Teleport the hmd to where the user is looking by moving the anchor to where they are looking minus the\n // offset of the headset from the anchor.\n if (this.webVRCamera.leftCamera) {\n this._workingVector.copyFrom(this.webVRCamera.leftCamera.globalPosition);\n this._workingVector.subtractInPlace(this.webVRCamera.position);\n location.subtractToRef(this._workingVector, this._workingVector);\n }\n else {\n this._workingVector.copyFrom(location);\n }\n // Add height to account for user's height offset\n if (this.isInVRMode) {\n this._workingVector.y += this.webVRCamera.deviceDistanceToRoomGround() * this._webVRCamera.deviceScaleFactor;\n }\n else {\n this._workingVector.y += this._defaultHeight;\n }\n this.onBeforeCameraTeleport.notifyObservers(this._workingVector);\n // Animations FPS\n const FPS = 90;\n let speedRatio, lastFrame;\n if (this._teleportationMode == VRExperienceHelper.TELEPORTATIONMODE_CONSTANTSPEED) {\n lastFrame = FPS;\n const dist = Vector3.Distance(this.currentVRCamera.position, this._workingVector);\n speedRatio = this._teleportationSpeed / dist;\n }\n else {\n // teleportationMode is TELEPORTATIONMODE_CONSTANTTIME\n lastFrame = Math.round((this._teleportationTime * FPS) / 1000);\n speedRatio = 1;\n }\n // Create animation from the camera's position to the new location\n this.currentVRCamera.animations = [];\n const animationCameraTeleportation = new Animation(\"animationCameraTeleportation\", \"position\", FPS, Animation.ANIMATIONTYPE_VECTOR3, Animation.ANIMATIONLOOPMODE_CONSTANT);\n const animationCameraTeleportationKeys = [\n {\n frame: 0,\n value: this.currentVRCamera.position,\n },\n {\n frame: lastFrame,\n value: this._workingVector,\n },\n ];\n animationCameraTeleportation.setKeys(animationCameraTeleportationKeys);\n animationCameraTeleportation.setEasingFunction(this._teleportationEasing);\n this.currentVRCamera.animations.push(animationCameraTeleportation);\n this._postProcessMove.animations = [];\n // Calculate the mid frame for vignette animations\n const midFrame = Math.round(lastFrame / 2);\n const animationPP = new Animation(\"animationPP\", \"vignetteWeight\", FPS, Animation.ANIMATIONTYPE_FLOAT, Animation.ANIMATIONLOOPMODE_CONSTANT);\n const vignetteWeightKeys = [];\n vignetteWeightKeys.push({\n frame: 0,\n value: 0,\n });\n vignetteWeightKeys.push({\n frame: midFrame,\n value: 8,\n });\n vignetteWeightKeys.push({\n frame: lastFrame,\n value: 0,\n });\n animationPP.setKeys(vignetteWeightKeys);\n this._postProcessMove.animations.push(animationPP);\n const animationPP2 = new Animation(\"animationPP2\", \"vignetteStretch\", FPS, Animation.ANIMATIONTYPE_FLOAT, Animation.ANIMATIONLOOPMODE_CONSTANT);\n const vignetteStretchKeys = [];\n vignetteStretchKeys.push({\n frame: 0,\n value: 0,\n });\n vignetteStretchKeys.push({\n frame: midFrame,\n value: 10,\n });\n vignetteStretchKeys.push({\n frame: lastFrame,\n value: 0,\n });\n animationPP2.setKeys(vignetteStretchKeys);\n this._postProcessMove.animations.push(animationPP2);\n this._postProcessMove.imageProcessingConfiguration.vignetteWeight = 0;\n this._postProcessMove.imageProcessingConfiguration.vignetteStretch = 0;\n this._webVRCamera.attachPostProcess(this._postProcessMove);\n this._scene.beginAnimation(this._postProcessMove, 0, lastFrame, false, speedRatio, () => {\n this._webVRCamera.detachPostProcess(this._postProcessMove);\n });\n this._scene.beginAnimation(this.currentVRCamera, 0, lastFrame, false, speedRatio, () => {\n this.onAfterCameraTeleport.notifyObservers(this._workingVector);\n });\n this._hideTeleportationTarget();\n }\n _convertNormalToDirectionOfRay(normal, ray) {\n if (normal) {\n const angle = Math.acos(Vector3.Dot(normal, ray.direction));\n if (angle < Math.PI / 2) {\n normal.scaleInPlace(-1);\n }\n }\n return normal;\n }\n _castRayAndSelectObject(gazer) {\n if (!(this.currentVRCamera instanceof FreeCamera)) {\n return;\n }\n const ray = gazer._getForwardRay(this._rayLength);\n const hit = this._scene.pickWithRay(ray, this._raySelectionPredicate);\n if (hit) {\n this._scene.simulatePointerMove(hit, { pointerId: gazer._id });\n }\n gazer._currentHit = hit;\n // Moving the gazeTracker on the mesh face targetted\n if (hit && hit.pickedPoint) {\n if (this._displayGaze) {\n let multiplier = 1;\n gazer._gazeTracker.isVisible = true;\n if (gazer._isActionableMesh) {\n multiplier = 3;\n }\n if (this.updateGazeTrackerScale) {\n gazer._gazeTracker.scaling.x = hit.distance * multiplier;\n gazer._gazeTracker.scaling.y = hit.distance * multiplier;\n gazer._gazeTracker.scaling.z = hit.distance * multiplier;\n }\n const pickNormal = this._convertNormalToDirectionOfRay(hit.getNormal(), ray);\n // To avoid z-fighting\n const deltaFighting = 0.002;\n if (pickNormal) {\n const axis1 = Vector3.Cross(Axis.Y, pickNormal);\n const axis2 = Vector3.Cross(pickNormal, axis1);\n Vector3.RotationFromAxisToRef(axis2, pickNormal, axis1, gazer._gazeTracker.rotation);\n }\n gazer._gazeTracker.position.copyFrom(hit.pickedPoint);\n if (gazer._gazeTracker.position.x < 0) {\n gazer._gazeTracker.position.x += deltaFighting;\n }\n else {\n gazer._gazeTracker.position.x -= deltaFighting;\n }\n if (gazer._gazeTracker.position.y < 0) {\n gazer._gazeTracker.position.y += deltaFighting;\n }\n else {\n gazer._gazeTracker.position.y -= deltaFighting;\n }\n if (gazer._gazeTracker.position.z < 0) {\n gazer._gazeTracker.position.z += deltaFighting;\n }\n else {\n gazer._gazeTracker.position.z -= deltaFighting;\n }\n }\n // Changing the size of the laser pointer based on the distance from the targetted point\n gazer._updatePointerDistance(hit.distance);\n }\n else {\n gazer._updatePointerDistance();\n gazer._gazeTracker.isVisible = false;\n }\n if (hit && hit.pickedMesh) {\n // The object selected is the floor, we're in a teleportation scenario\n if (this._teleportationInitialized && this._isTeleportationFloor(hit.pickedMesh) && hit.pickedPoint) {\n // Moving the teleportation area to this targetted point\n //Raise onSelectedMeshUnselected observable if ray collided floor mesh/meshes and a non floor mesh was previously selected\n if (gazer._currentMeshSelected && !this._isTeleportationFloor(gazer._currentMeshSelected)) {\n this._notifySelectedMeshUnselected(gazer._currentMeshSelected);\n }\n gazer._currentMeshSelected = null;\n if (gazer._teleportationRequestInitiated) {\n this._moveTeleportationSelectorTo(hit, gazer, ray);\n }\n return;\n }\n // If not, we're in a selection scenario\n //this._teleportationAllowed = false;\n if (hit.pickedMesh !== gazer._currentMeshSelected) {\n if (this.meshSelectionPredicate(hit.pickedMesh)) {\n this.onNewMeshPicked.notifyObservers(hit);\n gazer._currentMeshSelected = hit.pickedMesh;\n if (hit.pickedMesh.isPickable && hit.pickedMesh.actionManager) {\n this.changeGazeColor(this._pickedGazeColor);\n this.changeLaserColor(this._pickedLaserColor);\n gazer._isActionableMesh = true;\n }\n else {\n this.changeGazeColor(this._gazeColor);\n this.changeLaserColor(this._laserColor);\n gazer._isActionableMesh = false;\n }\n try {\n this.onNewMeshSelected.notifyObservers(hit.pickedMesh);\n const gazerAsControllerGazer = gazer;\n if (gazerAsControllerGazer.webVRController) {\n this.onMeshSelectedWithController.notifyObservers({ mesh: hit.pickedMesh, controller: gazerAsControllerGazer.webVRController });\n }\n }\n catch (err) {\n Logger.Warn(\"Error while raising onNewMeshSelected or onMeshSelectedWithController: \" + err);\n }\n }\n else {\n this._notifySelectedMeshUnselected(gazer._currentMeshSelected);\n gazer._currentMeshSelected = null;\n this.changeGazeColor(this._gazeColor);\n this.changeLaserColor(this._laserColor);\n }\n }\n }\n else {\n this._notifySelectedMeshUnselected(gazer._currentMeshSelected);\n gazer._currentMeshSelected = null;\n //this._teleportationAllowed = false;\n this.changeGazeColor(this._gazeColor);\n this.changeLaserColor(this._laserColor);\n }\n }\n _notifySelectedMeshUnselected(mesh) {\n if (mesh) {\n this.onSelectedMeshUnselected.notifyObservers(mesh);\n }\n }\n /**\n * Permanently set new colors for the laser pointer\n * @param color the new laser color\n * @param pickedColor the new laser color when picked mesh detected\n */\n setLaserColor(color, pickedColor = this._pickedLaserColor) {\n this._laserColor = color;\n this._pickedLaserColor = pickedColor;\n }\n /**\n * Set lighting enabled / disabled on the laser pointer of both controllers\n * @param enabled should the lighting be enabled on the laser pointer\n */\n setLaserLightingState(enabled = true) {\n if (this._leftController) {\n this._leftController._setLaserPointerLightingDisabled(!enabled);\n }\n if (this._rightController) {\n this._rightController._setLaserPointerLightingDisabled(!enabled);\n }\n }\n /**\n * Permanently set new colors for the gaze pointer\n * @param color the new gaze color\n * @param pickedColor the new gaze color when picked mesh detected\n */\n setGazeColor(color, pickedColor = this._pickedGazeColor) {\n this._gazeColor = color;\n this._pickedGazeColor = pickedColor;\n }\n /**\n * Sets the color of the laser ray from the vr controllers.\n * @param color new color for the ray.\n */\n changeLaserColor(color) {\n if (!this.updateControllerLaserColor) {\n return;\n }\n if (this._leftController) {\n this._leftController._setLaserPointerColor(color);\n }\n if (this._rightController) {\n this._rightController._setLaserPointerColor(color);\n }\n }\n /**\n * Sets the color of the ray from the vr headsets gaze.\n * @param color new color for the ray.\n */\n changeGazeColor(color) {\n if (!this.updateGazeTrackerColor) {\n return;\n }\n if (!this._cameraGazer._gazeTracker.material) {\n return;\n }\n this._cameraGazer._gazeTracker.material.emissiveColor = color;\n if (this._leftController) {\n this._leftController._gazeTracker.material.emissiveColor = color;\n }\n if (this._rightController) {\n this._rightController._gazeTracker.material.emissiveColor = color;\n }\n }\n /**\n * Exits VR and disposes of the vr experience helper\n */\n dispose() {\n if (this.isInVRMode) {\n this.exitVR();\n }\n if (this._postProcessMove) {\n this._postProcessMove.dispose();\n }\n if (this._webVRCamera) {\n this._webVRCamera.dispose();\n }\n if (this._vrDeviceOrientationCamera) {\n this._vrDeviceOrientationCamera.dispose();\n }\n if (!this._useCustomVRButton && this._btnVR && this._btnVR.parentNode) {\n document.body.removeChild(this._btnVR);\n }\n if (this._deviceOrientationCamera && this._scene.activeCamera != this._deviceOrientationCamera) {\n this._deviceOrientationCamera.dispose();\n }\n if (this._cameraGazer) {\n this._cameraGazer.dispose();\n }\n if (this._leftController) {\n this._leftController.dispose();\n }\n if (this._rightController) {\n this._rightController.dispose();\n }\n if (this._teleportationTarget) {\n this._teleportationTarget.dispose();\n }\n if (this.xr) {\n this.xr.dispose();\n }\n this._floorMeshesCollection.length = 0;\n document.removeEventListener(\"keydown\", this._onKeyDown);\n window.removeEventListener(\"vrdisplaypresentchange\", this._onVrDisplayPresentChangeBind);\n window.removeEventListener(\"resize\", this._onResize);\n document.removeEventListener(\"fullscreenchange\", this._onFullscreenChange);\n this._scene.getEngine().onVRDisplayChangedObservable.removeCallback(this._onVRDisplayChangedBind);\n this._scene.getEngine().onVRRequestPresentStart.removeCallback(this._onVRRequestPresentStart);\n this._scene.getEngine().onVRRequestPresentComplete.removeCallback(this._onVRRequestPresentComplete);\n this._scene.gamepadManager.onGamepadConnectedObservable.removeCallback(this._onNewGamepadConnected);\n this._scene.gamepadManager.onGamepadDisconnectedObservable.removeCallback(this._onNewGamepadDisconnected);\n this._scene.unregisterBeforeRender(this._beforeRender);\n }\n /**\n * Gets the name of the VRExperienceHelper class\n * @returns \"VRExperienceHelper\"\n */\n getClassName() {\n return \"VRExperienceHelper\";\n }\n }\n /**\n * Time Constant Teleportation Mode\n */\n VRExperienceHelper.TELEPORTATIONMODE_CONSTANTTIME = 0;\n /**\n * Speed Constant Teleportation Mode\n */\n VRExperienceHelper.TELEPORTATIONMODE_CONSTANTSPEED = 1;\n\n /* eslint-disable @typescript-eslint/naming-convention */\n // Based on demo done by Brandon Jones - http://media.tojicode.com/webgl-samples/dds.html\n // All values and structures referenced from:\n // http://msdn.microsoft.com/en-us/library/bb943991.aspx/\n const DDS_MAGIC = 0x20534444;\n const //DDSD_CAPS = 0x1,\n //DDSD_HEIGHT = 0x2,\n //DDSD_WIDTH = 0x4,\n //DDSD_PITCH = 0x8,\n //DDSD_PIXELFORMAT = 0x1000,\n DDSD_MIPMAPCOUNT = 0x20000;\n //DDSD_LINEARSIZE = 0x80000,\n //DDSD_DEPTH = 0x800000;\n // var DDSCAPS_COMPLEX = 0x8,\n // DDSCAPS_MIPMAP = 0x400000,\n // DDSCAPS_TEXTURE = 0x1000;\n const DDSCAPS2_CUBEMAP = 0x200;\n // DDSCAPS2_CUBEMAP_POSITIVEX = 0x400,\n // DDSCAPS2_CUBEMAP_NEGATIVEX = 0x800,\n // DDSCAPS2_CUBEMAP_POSITIVEY = 0x1000,\n // DDSCAPS2_CUBEMAP_NEGATIVEY = 0x2000,\n // DDSCAPS2_CUBEMAP_POSITIVEZ = 0x4000,\n // DDSCAPS2_CUBEMAP_NEGATIVEZ = 0x8000,\n // DDSCAPS2_VOLUME = 0x200000;\n const //DDPF_ALPHAPIXELS = 0x1,\n //DDPF_ALPHA = 0x2,\n DDPF_FOURCC = 0x4, DDPF_RGB = 0x40, \n //DDPF_YUV = 0x200,\n DDPF_LUMINANCE = 0x20000;\n function FourCCToInt32(value) {\n return value.charCodeAt(0) + (value.charCodeAt(1) << 8) + (value.charCodeAt(2) << 16) + (value.charCodeAt(3) << 24);\n }\n function Int32ToFourCC(value) {\n return String.fromCharCode(value & 0xff, (value >> 8) & 0xff, (value >> 16) & 0xff, (value >> 24) & 0xff);\n }\n const FOURCC_DXT1 = FourCCToInt32(\"DXT1\");\n const FOURCC_DXT3 = FourCCToInt32(\"DXT3\");\n const FOURCC_DXT5 = FourCCToInt32(\"DXT5\");\n const FOURCC_DX10 = FourCCToInt32(\"DX10\");\n const FOURCC_D3DFMT_R16G16B16A16F = 113;\n const FOURCC_D3DFMT_R32G32B32A32F = 116;\n const DXGI_FORMAT_R32G32B32A32_FLOAT = 2;\n const DXGI_FORMAT_R16G16B16A16_FLOAT = 10;\n const DXGI_FORMAT_B8G8R8X8_UNORM = 88;\n const headerLengthInt = 31; // The header length in 32 bit ints\n // Offsets into the header array\n const off_magic = 0;\n const off_size = 1;\n const off_flags = 2;\n const off_height = 3;\n const off_width = 4;\n const off_mipmapCount = 7;\n const off_pfFlags = 20;\n const off_pfFourCC = 21;\n const off_RGBbpp = 22;\n const off_RMask = 23;\n const off_GMask = 24;\n const off_BMask = 25;\n const off_AMask = 26;\n // var off_caps1 = 27;\n const off_caps2 = 28;\n // var off_caps3 = 29;\n // var off_caps4 = 30;\n const off_dxgiFormat = 32;\n /**\n * Class used to provide DDS decompression tools\n */\n class DDSTools {\n /**\n * Gets DDS information from an array buffer\n * @param data defines the array buffer view to read data from\n * @returns the DDS information\n */\n static GetDDSInfo(data) {\n const header = new Int32Array(data.buffer, data.byteOffset, headerLengthInt);\n const extendedHeader = new Int32Array(data.buffer, data.byteOffset, headerLengthInt + 4);\n let mipmapCount = 1;\n if (header[off_flags] & DDSD_MIPMAPCOUNT) {\n mipmapCount = Math.max(1, header[off_mipmapCount]);\n }\n const fourCC = header[off_pfFourCC];\n const dxgiFormat = fourCC === FOURCC_DX10 ? extendedHeader[off_dxgiFormat] : 0;\n let textureType = 0;\n switch (fourCC) {\n case FOURCC_D3DFMT_R16G16B16A16F:\n textureType = 2;\n break;\n case FOURCC_D3DFMT_R32G32B32A32F:\n textureType = 1;\n break;\n case FOURCC_DX10:\n if (dxgiFormat === DXGI_FORMAT_R16G16B16A16_FLOAT) {\n textureType = 2;\n break;\n }\n if (dxgiFormat === DXGI_FORMAT_R32G32B32A32_FLOAT) {\n textureType = 1;\n break;\n }\n }\n return {\n width: header[off_width],\n height: header[off_height],\n mipmapCount: mipmapCount,\n isFourCC: (header[off_pfFlags] & DDPF_FOURCC) === DDPF_FOURCC,\n isRGB: (header[off_pfFlags] & DDPF_RGB) === DDPF_RGB,\n isLuminance: (header[off_pfFlags] & DDPF_LUMINANCE) === DDPF_LUMINANCE,\n isCube: (header[off_caps2] & DDSCAPS2_CUBEMAP) === DDSCAPS2_CUBEMAP,\n isCompressed: fourCC === FOURCC_DXT1 || fourCC === FOURCC_DXT3 || fourCC === FOURCC_DXT5,\n dxgiFormat: dxgiFormat,\n textureType: textureType,\n };\n }\n static _GetHalfFloatAsFloatRGBAArrayBuffer(width, height, dataOffset, dataLength, arrayBuffer, lod) {\n const destArray = new Float32Array(dataLength);\n const srcData = new Uint16Array(arrayBuffer, dataOffset);\n let index = 0;\n for (let y = 0; y < height; y++) {\n for (let x = 0; x < width; x++) {\n const srcPos = (x + y * width) * 4;\n destArray[index] = FromHalfFloat(srcData[srcPos]);\n destArray[index + 1] = FromHalfFloat(srcData[srcPos + 1]);\n destArray[index + 2] = FromHalfFloat(srcData[srcPos + 2]);\n if (DDSTools.StoreLODInAlphaChannel) {\n destArray[index + 3] = lod;\n }\n else {\n destArray[index + 3] = FromHalfFloat(srcData[srcPos + 3]);\n }\n index += 4;\n }\n }\n return destArray;\n }\n static _GetHalfFloatRGBAArrayBuffer(width, height, dataOffset, dataLength, arrayBuffer, lod) {\n if (DDSTools.StoreLODInAlphaChannel) {\n const destArray = new Uint16Array(dataLength);\n const srcData = new Uint16Array(arrayBuffer, dataOffset);\n let index = 0;\n for (let y = 0; y < height; y++) {\n for (let x = 0; x < width; x++) {\n const srcPos = (x + y * width) * 4;\n destArray[index] = srcData[srcPos];\n destArray[index + 1] = srcData[srcPos + 1];\n destArray[index + 2] = srcData[srcPos + 2];\n destArray[index + 3] = ToHalfFloat(lod);\n index += 4;\n }\n }\n return destArray;\n }\n return new Uint16Array(arrayBuffer, dataOffset, dataLength);\n }\n static _GetFloatRGBAArrayBuffer(width, height, dataOffset, dataLength, arrayBuffer, lod) {\n if (DDSTools.StoreLODInAlphaChannel) {\n const destArray = new Float32Array(dataLength);\n const srcData = new Float32Array(arrayBuffer, dataOffset);\n let index = 0;\n for (let y = 0; y < height; y++) {\n for (let x = 0; x < width; x++) {\n const srcPos = (x + y * width) * 4;\n destArray[index] = srcData[srcPos];\n destArray[index + 1] = srcData[srcPos + 1];\n destArray[index + 2] = srcData[srcPos + 2];\n destArray[index + 3] = lod;\n index += 4;\n }\n }\n return destArray;\n }\n return new Float32Array(arrayBuffer, dataOffset, dataLength);\n }\n static _GetFloatAsHalfFloatRGBAArrayBuffer(width, height, dataOffset, dataLength, arrayBuffer, lod) {\n const destArray = new Uint16Array(dataLength);\n const srcData = new Float32Array(arrayBuffer, dataOffset);\n let index = 0;\n for (let y = 0; y < height; y++) {\n for (let x = 0; x < width; x++) {\n destArray[index] = ToHalfFloat(srcData[index]);\n destArray[index + 1] = ToHalfFloat(srcData[index + 1]);\n destArray[index + 2] = ToHalfFloat(srcData[index + 2]);\n if (DDSTools.StoreLODInAlphaChannel) {\n destArray[index + 3] = ToHalfFloat(lod);\n }\n else {\n destArray[index + 3] = ToHalfFloat(srcData[index + 3]);\n }\n index += 4;\n }\n }\n return destArray;\n }\n static _GetFloatAsUIntRGBAArrayBuffer(width, height, dataOffset, dataLength, arrayBuffer, lod) {\n const destArray = new Uint8Array(dataLength);\n const srcData = new Float32Array(arrayBuffer, dataOffset);\n let index = 0;\n for (let y = 0; y < height; y++) {\n for (let x = 0; x < width; x++) {\n const srcPos = (x + y * width) * 4;\n destArray[index] = Scalar.Clamp(srcData[srcPos]) * 255;\n destArray[index + 1] = Scalar.Clamp(srcData[srcPos + 1]) * 255;\n destArray[index + 2] = Scalar.Clamp(srcData[srcPos + 2]) * 255;\n if (DDSTools.StoreLODInAlphaChannel) {\n destArray[index + 3] = lod;\n }\n else {\n destArray[index + 3] = Scalar.Clamp(srcData[srcPos + 3]) * 255;\n }\n index += 4;\n }\n }\n return destArray;\n }\n static _GetHalfFloatAsUIntRGBAArrayBuffer(width, height, dataOffset, dataLength, arrayBuffer, lod) {\n const destArray = new Uint8Array(dataLength);\n const srcData = new Uint16Array(arrayBuffer, dataOffset);\n let index = 0;\n for (let y = 0; y < height; y++) {\n for (let x = 0; x < width; x++) {\n const srcPos = (x + y * width) * 4;\n destArray[index] = Scalar.Clamp(FromHalfFloat(srcData[srcPos])) * 255;\n destArray[index + 1] = Scalar.Clamp(FromHalfFloat(srcData[srcPos + 1])) * 255;\n destArray[index + 2] = Scalar.Clamp(FromHalfFloat(srcData[srcPos + 2])) * 255;\n if (DDSTools.StoreLODInAlphaChannel) {\n destArray[index + 3] = lod;\n }\n else {\n destArray[index + 3] = Scalar.Clamp(FromHalfFloat(srcData[srcPos + 3])) * 255;\n }\n index += 4;\n }\n }\n return destArray;\n }\n static _GetRGBAArrayBuffer(width, height, dataOffset, dataLength, arrayBuffer, rOffset, gOffset, bOffset, aOffset) {\n const byteArray = new Uint8Array(dataLength);\n const srcData = new Uint8Array(arrayBuffer, dataOffset);\n let index = 0;\n for (let y = 0; y < height; y++) {\n for (let x = 0; x < width; x++) {\n const srcPos = (x + y * width) * 4;\n byteArray[index] = srcData[srcPos + rOffset];\n byteArray[index + 1] = srcData[srcPos + gOffset];\n byteArray[index + 2] = srcData[srcPos + bOffset];\n byteArray[index + 3] = srcData[srcPos + aOffset];\n index += 4;\n }\n }\n return byteArray;\n }\n static _ExtractLongWordOrder(value) {\n if (value === 0 || value === 255 || value === -16777216) {\n return 0;\n }\n return 1 + DDSTools._ExtractLongWordOrder(value >> 8);\n }\n static _GetRGBArrayBuffer(width, height, dataOffset, dataLength, arrayBuffer, rOffset, gOffset, bOffset) {\n const byteArray = new Uint8Array(dataLength);\n const srcData = new Uint8Array(arrayBuffer, dataOffset);\n let index = 0;\n for (let y = 0; y < height; y++) {\n for (let x = 0; x < width; x++) {\n const srcPos = (x + y * width) * 3;\n byteArray[index] = srcData[srcPos + rOffset];\n byteArray[index + 1] = srcData[srcPos + gOffset];\n byteArray[index + 2] = srcData[srcPos + bOffset];\n index += 3;\n }\n }\n return byteArray;\n }\n static _GetLuminanceArrayBuffer(width, height, dataOffset, dataLength, arrayBuffer) {\n const byteArray = new Uint8Array(dataLength);\n const srcData = new Uint8Array(arrayBuffer, dataOffset);\n let index = 0;\n for (let y = 0; y < height; y++) {\n for (let x = 0; x < width; x++) {\n const srcPos = x + y * width;\n byteArray[index] = srcData[srcPos];\n index++;\n }\n }\n return byteArray;\n }\n /**\n * Uploads DDS Levels to a Babylon Texture\n * @internal\n */\n static UploadDDSLevels(engine, texture, data, info, loadMipmaps, faces, lodIndex = -1, currentFace, destTypeMustBeFilterable = true) {\n let sphericalPolynomialFaces = null;\n if (info.sphericalPolynomial) {\n sphericalPolynomialFaces = new Array();\n }\n const ext = !!engine.getCaps().s3tc;\n // TODO WEBGPU Once generateMipMaps is split into generateMipMaps + hasMipMaps in InternalTexture this line can be removed\n texture.generateMipMaps = loadMipmaps;\n const header = new Int32Array(data.buffer, data.byteOffset, headerLengthInt);\n let fourCC, width, height, dataLength = 0, dataOffset;\n let byteArray, mipmapCount, mip;\n let internalCompressedFormat = 0;\n let blockBytes = 1;\n if (header[off_magic] !== DDS_MAGIC) {\n Logger.Error(\"Invalid magic number in DDS header\");\n return;\n }\n if (!info.isFourCC && !info.isRGB && !info.isLuminance) {\n Logger.Error(\"Unsupported format, must contain a FourCC, RGB or LUMINANCE code\");\n return;\n }\n if (info.isCompressed && !ext) {\n Logger.Error(\"Compressed textures are not supported on this platform.\");\n return;\n }\n let bpp = header[off_RGBbpp];\n dataOffset = header[off_size] + 4;\n let computeFormats = false;\n if (info.isFourCC) {\n fourCC = header[off_pfFourCC];\n switch (fourCC) {\n case FOURCC_DXT1:\n blockBytes = 8;\n internalCompressedFormat = 33777;\n break;\n case FOURCC_DXT3:\n blockBytes = 16;\n internalCompressedFormat = 33778;\n break;\n case FOURCC_DXT5:\n blockBytes = 16;\n internalCompressedFormat = 33779;\n break;\n case FOURCC_D3DFMT_R16G16B16A16F:\n computeFormats = true;\n bpp = 64;\n break;\n case FOURCC_D3DFMT_R32G32B32A32F:\n computeFormats = true;\n bpp = 128;\n break;\n case FOURCC_DX10: {\n // There is an additionnal header so dataOffset need to be changed\n dataOffset += 5 * 4; // 5 uints\n let supported = false;\n switch (info.dxgiFormat) {\n case DXGI_FORMAT_R16G16B16A16_FLOAT:\n computeFormats = true;\n bpp = 64;\n supported = true;\n break;\n case DXGI_FORMAT_R32G32B32A32_FLOAT:\n computeFormats = true;\n bpp = 128;\n supported = true;\n break;\n case DXGI_FORMAT_B8G8R8X8_UNORM:\n info.isRGB = true;\n info.isFourCC = false;\n bpp = 32;\n supported = true;\n break;\n }\n if (supported) {\n break;\n }\n }\n // eslint-disable-next-line no-fallthrough\n default:\n console.error(\"Unsupported FourCC code:\", Int32ToFourCC(fourCC));\n return;\n }\n }\n const rOffset = DDSTools._ExtractLongWordOrder(header[off_RMask]);\n const gOffset = DDSTools._ExtractLongWordOrder(header[off_GMask]);\n const bOffset = DDSTools._ExtractLongWordOrder(header[off_BMask]);\n const aOffset = DDSTools._ExtractLongWordOrder(header[off_AMask]);\n if (computeFormats) {\n internalCompressedFormat = engine._getRGBABufferInternalSizedFormat(info.textureType);\n }\n mipmapCount = 1;\n if (header[off_flags] & DDSD_MIPMAPCOUNT && loadMipmaps !== false) {\n mipmapCount = Math.max(1, header[off_mipmapCount]);\n }\n const startFace = currentFace || 0;\n const caps = engine.getCaps();\n for (let face = startFace; face < faces; face++) {\n width = header[off_width];\n height = header[off_height];\n for (mip = 0; mip < mipmapCount; ++mip) {\n if (lodIndex === -1 || lodIndex === mip) {\n // In case of fixed LOD, if the lod has just been uploaded, early exit.\n const i = lodIndex === -1 ? mip : 0;\n if (!info.isCompressed && info.isFourCC) {\n texture.format = 5;\n dataLength = width * height * 4;\n let floatArray = null;\n if (engine._badOS || engine._badDesktopOS || (!caps.textureHalfFloat && !caps.textureFloat)) {\n // Required because iOS has many issues with float and half float generation\n if (bpp === 128) {\n floatArray = DDSTools._GetFloatAsUIntRGBAArrayBuffer(width, height, data.byteOffset + dataOffset, dataLength, data.buffer, i);\n if (sphericalPolynomialFaces && i == 0) {\n sphericalPolynomialFaces.push(DDSTools._GetFloatRGBAArrayBuffer(width, height, data.byteOffset + dataOffset, dataLength, data.buffer, i));\n }\n }\n else if (bpp === 64) {\n floatArray = DDSTools._GetHalfFloatAsUIntRGBAArrayBuffer(width, height, data.byteOffset + dataOffset, dataLength, data.buffer, i);\n if (sphericalPolynomialFaces && i == 0) {\n sphericalPolynomialFaces.push(DDSTools._GetHalfFloatAsFloatRGBAArrayBuffer(width, height, data.byteOffset + dataOffset, dataLength, data.buffer, i));\n }\n }\n texture.type = 0;\n }\n else {\n const floatAvailable = caps.textureFloat && ((destTypeMustBeFilterable && caps.textureFloatLinearFiltering) || !destTypeMustBeFilterable);\n const halfFloatAvailable = caps.textureHalfFloat && ((destTypeMustBeFilterable && caps.textureHalfFloatLinearFiltering) || !destTypeMustBeFilterable);\n const destType = (bpp === 128 || (bpp === 64 && !halfFloatAvailable)) && floatAvailable\n ? 1\n : (bpp === 64 || (bpp === 128 && !floatAvailable)) && halfFloatAvailable\n ? 2\n : 0;\n let dataGetter;\n let dataGetterPolynomial = null;\n switch (bpp) {\n case 128: {\n switch (destType) {\n case 1:\n dataGetter = DDSTools._GetFloatRGBAArrayBuffer;\n dataGetterPolynomial = null;\n break;\n case 2:\n dataGetter = DDSTools._GetFloatAsHalfFloatRGBAArrayBuffer;\n dataGetterPolynomial = DDSTools._GetFloatRGBAArrayBuffer;\n break;\n case 0:\n dataGetter = DDSTools._GetFloatAsUIntRGBAArrayBuffer;\n dataGetterPolynomial = DDSTools._GetFloatRGBAArrayBuffer;\n break;\n }\n break;\n }\n default: {\n // 64 bpp\n switch (destType) {\n case 1:\n dataGetter = DDSTools._GetHalfFloatAsFloatRGBAArrayBuffer;\n dataGetterPolynomial = null;\n break;\n case 2:\n dataGetter = DDSTools._GetHalfFloatRGBAArrayBuffer;\n dataGetterPolynomial = DDSTools._GetHalfFloatAsFloatRGBAArrayBuffer;\n break;\n case 0:\n dataGetter = DDSTools._GetHalfFloatAsUIntRGBAArrayBuffer;\n dataGetterPolynomial = DDSTools._GetHalfFloatAsFloatRGBAArrayBuffer;\n break;\n }\n break;\n }\n }\n texture.type = destType;\n floatArray = dataGetter(width, height, data.byteOffset + dataOffset, dataLength, data.buffer, i);\n if (sphericalPolynomialFaces && i == 0) {\n sphericalPolynomialFaces.push(dataGetterPolynomial ? dataGetterPolynomial(width, height, data.byteOffset + dataOffset, dataLength, data.buffer, i) : floatArray);\n }\n }\n if (floatArray) {\n engine._uploadDataToTextureDirectly(texture, floatArray, face, i);\n }\n }\n else if (info.isRGB) {\n texture.type = 0;\n if (bpp === 24) {\n texture.format = 4;\n dataLength = width * height * 3;\n byteArray = DDSTools._GetRGBArrayBuffer(width, height, data.byteOffset + dataOffset, dataLength, data.buffer, rOffset, gOffset, bOffset);\n engine._uploadDataToTextureDirectly(texture, byteArray, face, i);\n }\n else {\n // 32\n texture.format = 5;\n dataLength = width * height * 4;\n byteArray = DDSTools._GetRGBAArrayBuffer(width, height, data.byteOffset + dataOffset, dataLength, data.buffer, rOffset, gOffset, bOffset, aOffset);\n engine._uploadDataToTextureDirectly(texture, byteArray, face, i);\n }\n }\n else if (info.isLuminance) {\n const unpackAlignment = engine._getUnpackAlignement();\n const unpaddedRowSize = width;\n const paddedRowSize = Math.floor((width + unpackAlignment - 1) / unpackAlignment) * unpackAlignment;\n dataLength = paddedRowSize * (height - 1) + unpaddedRowSize;\n byteArray = DDSTools._GetLuminanceArrayBuffer(width, height, data.byteOffset + dataOffset, dataLength, data.buffer);\n texture.format = 1;\n texture.type = 0;\n engine._uploadDataToTextureDirectly(texture, byteArray, face, i);\n }\n else {\n dataLength = (((Math.max(4, width) / 4) * Math.max(4, height)) / 4) * blockBytes;\n byteArray = new Uint8Array(data.buffer, data.byteOffset + dataOffset, dataLength);\n texture.type = 0;\n engine._uploadCompressedDataToTextureDirectly(texture, internalCompressedFormat, width, height, byteArray, face, i);\n }\n }\n dataOffset += bpp ? width * height * (bpp / 8) : dataLength;\n width *= 0.5;\n height *= 0.5;\n width = Math.max(1.0, width);\n height = Math.max(1.0, height);\n }\n if (currentFace !== undefined) {\n // Loading a single face\n break;\n }\n }\n if (sphericalPolynomialFaces && sphericalPolynomialFaces.length > 0) {\n info.sphericalPolynomial = CubeMapToSphericalPolynomialTools.ConvertCubeMapToSphericalPolynomial({\n size: header[off_width],\n right: sphericalPolynomialFaces[0],\n left: sphericalPolynomialFaces[1],\n up: sphericalPolynomialFaces[2],\n down: sphericalPolynomialFaces[3],\n front: sphericalPolynomialFaces[4],\n back: sphericalPolynomialFaces[5],\n format: 5,\n type: 1,\n gammaSpace: false,\n });\n }\n else {\n info.sphericalPolynomial = undefined;\n }\n }\n }\n /**\n * Gets or sets a boolean indicating that LOD info is stored in alpha channel (false by default)\n */\n DDSTools.StoreLODInAlphaChannel = false;\n /**\n * Create a cube texture from prefiltered data (ie. the mipmaps contain ready to use data for PBR reflection)\n * @param rootUrl defines the url where the file to load is located\n * @param scene defines the current scene\n * @param lodScale defines scale to apply to the mip map selection\n * @param lodOffset defines offset to apply to the mip map selection\n * @param onLoad defines an optional callback raised when the texture is loaded\n * @param onError defines an optional callback raised if there is an issue to load the texture\n * @param format defines the format of the data\n * @param forcedExtension defines the extension to use to pick the right loader\n * @param createPolynomials defines wheter or not to create polynomails harmonics for the texture\n * @returns the cube texture as an InternalTexture\n */\n ThinEngine.prototype.createPrefilteredCubeTexture = function (rootUrl, scene, lodScale, lodOffset, onLoad = null, onError = null, format, forcedExtension = null, createPolynomials = true) {\n const callback = (loadData) => {\n if (!loadData) {\n if (onLoad) {\n onLoad(null);\n }\n return;\n }\n const texture = loadData.texture;\n if (!createPolynomials) {\n texture._sphericalPolynomial = new SphericalPolynomial();\n }\n else if (loadData.info.sphericalPolynomial) {\n texture._sphericalPolynomial = loadData.info.sphericalPolynomial;\n }\n texture._source = InternalTextureSource.CubePrefiltered;\n if (this.getCaps().textureLOD) {\n // Do not add extra process if texture lod is supported.\n if (onLoad) {\n onLoad(texture);\n }\n return;\n }\n const mipSlices = 3;\n const gl = this._gl;\n const width = loadData.width;\n if (!width) {\n return;\n }\n const textures = [];\n for (let i = 0; i < mipSlices; i++) {\n //compute LOD from even spacing in smoothness (matching shader calculation)\n const smoothness = i / (mipSlices - 1);\n const roughness = 1 - smoothness;\n const minLODIndex = lodOffset; // roughness = 0\n const maxLODIndex = Scalar.Log2(width) * lodScale + lodOffset; // roughness = 1\n const lodIndex = minLODIndex + (maxLODIndex - minLODIndex) * roughness;\n const mipmapIndex = Math.round(Math.min(Math.max(lodIndex, 0), maxLODIndex));\n const glTextureFromLod = new InternalTexture(this, InternalTextureSource.Temp);\n glTextureFromLod.type = texture.type;\n glTextureFromLod.format = texture.format;\n glTextureFromLod.width = Math.pow(2, Math.max(Scalar.Log2(width) - mipmapIndex, 0));\n glTextureFromLod.height = glTextureFromLod.width;\n glTextureFromLod.isCube = true;\n glTextureFromLod._cachedWrapU = 0;\n glTextureFromLod._cachedWrapV = 0;\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, glTextureFromLod, true);\n glTextureFromLod.samplingMode = 2;\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_MAG_FILTER, gl.LINEAR);\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_MIN_FILTER, gl.LINEAR);\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n if (loadData.isDDS) {\n const info = loadData.info;\n const data = loadData.data;\n this._unpackFlipY(info.isCompressed);\n DDSTools.UploadDDSLevels(this, glTextureFromLod, data, info, true, 6, mipmapIndex);\n }\n else {\n Logger.Warn(\"DDS is the only prefiltered cube map supported so far.\");\n }\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, null);\n // Wrap in a base texture for easy binding.\n const lodTexture = new BaseTexture(scene);\n lodTexture._isCube = true;\n lodTexture._texture = glTextureFromLod;\n glTextureFromLod.isReady = true;\n textures.push(lodTexture);\n }\n texture._lodTextureHigh = textures[2];\n texture._lodTextureMid = textures[1];\n texture._lodTextureLow = textures[0];\n if (onLoad) {\n onLoad(texture);\n }\n };\n return this.createCubeTexture(rootUrl, scene, null, false, callback, onError, format, forcedExtension, createPolynomials, lodScale, lodOffset);\n };\n\n /**\n * Implementation of the DDS Texture Loader.\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class _DDSTextureLoader {\n constructor() {\n /**\n * Defines whether the loader supports cascade loading the different faces.\n */\n this.supportCascades = true;\n }\n /**\n * This returns if the loader support the current file information.\n * @param extension defines the file extension of the file being loaded\n * @returns true if the loader can load the specified file\n */\n canLoad(extension) {\n return extension.endsWith(\".dds\");\n }\n /**\n * Uploads the cube texture data to the WebGL texture. It has already been bound.\n * @param imgs contains the cube maps\n * @param texture defines the BabylonJS internal texture\n * @param createPolynomials will be true if polynomials have been requested\n * @param onLoad defines the callback to trigger once the texture is ready\n */\n loadCubeData(imgs, texture, createPolynomials, onLoad) {\n const engine = texture.getEngine();\n let info;\n let loadMipmap = false;\n let maxLevel = 1000;\n if (Array.isArray(imgs)) {\n for (let index = 0; index < imgs.length; index++) {\n const data = imgs[index];\n info = DDSTools.GetDDSInfo(data);\n texture.width = info.width;\n texture.height = info.height;\n loadMipmap = (info.isRGB || info.isLuminance || info.mipmapCount > 1) && texture.generateMipMaps;\n engine._unpackFlipY(info.isCompressed);\n DDSTools.UploadDDSLevels(engine, texture, data, info, loadMipmap, 6, -1, index);\n if (!info.isFourCC && info.mipmapCount === 1) {\n engine.generateMipMapsForCubemap(texture);\n }\n else {\n maxLevel = info.mipmapCount - 1;\n }\n }\n }\n else {\n const data = imgs;\n info = DDSTools.GetDDSInfo(data);\n texture.width = info.width;\n texture.height = info.height;\n if (createPolynomials) {\n info.sphericalPolynomial = new SphericalPolynomial();\n }\n loadMipmap = (info.isRGB || info.isLuminance || info.mipmapCount > 1) && texture.generateMipMaps;\n engine._unpackFlipY(info.isCompressed);\n DDSTools.UploadDDSLevels(engine, texture, data, info, loadMipmap, 6);\n if (!info.isFourCC && info.mipmapCount === 1) {\n // Do not unbind as we still need to set the parameters.\n engine.generateMipMapsForCubemap(texture, false);\n }\n else {\n maxLevel = info.mipmapCount - 1;\n }\n }\n engine._setCubeMapTextureParams(texture, loadMipmap, maxLevel);\n texture.isReady = true;\n texture.onLoadedObservable.notifyObservers(texture);\n texture.onLoadedObservable.clear();\n if (onLoad) {\n onLoad({ isDDS: true, width: texture.width, info, data: imgs, texture });\n }\n }\n /**\n * Uploads the 2D texture data to the WebGL texture. It has already been bound once in the callback.\n * @param data contains the texture data\n * @param texture defines the BabylonJS internal texture\n * @param callback defines the method to call once ready to upload\n */\n loadData(data, texture, callback) {\n const info = DDSTools.GetDDSInfo(data);\n const loadMipmap = (info.isRGB || info.isLuminance || info.mipmapCount > 1) && texture.generateMipMaps && info.width >> (info.mipmapCount - 1) === 1;\n callback(info.width, info.height, loadMipmap, info.isFourCC, () => {\n DDSTools.UploadDDSLevels(texture.getEngine(), texture, data, info, loadMipmap, 1);\n });\n }\n }\n // Register the loader.\n Engine._TextureLoaders.push(new _DDSTextureLoader());\n\n // Do not edit.\n const name$1W = \"rgbdEncodePixelShader\";\n const shader$1W = `varying vec2 vUV;uniform sampler2D textureSampler;\n#include\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) \n{gl_FragColor=toRGBD(texture2D(textureSampler,vUV).rgb);}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$1W] = shader$1W;\n\n const DefaultEnvironmentTextureImageType = \"image/png\";\n const CurrentVersion = 2;\n /**\n * Magic number identifying the env file.\n */\n const MagicBytes = [0x86, 0x16, 0x87, 0x96, 0xf6, 0xd6, 0x96, 0x36];\n /**\n * Gets the environment info from an env file.\n * @param data The array buffer containing the .env bytes.\n * @returns the environment file info (the json header) if successfully parsed, normalized to the latest supported version.\n */\n function GetEnvInfo(data) {\n const dataView = new DataView(data.buffer, data.byteOffset, data.byteLength);\n let pos = 0;\n for (let i = 0; i < MagicBytes.length; i++) {\n if (dataView.getUint8(pos++) !== MagicBytes[i]) {\n Logger.Error(\"Not a babylon environment map\");\n return null;\n }\n }\n // Read json manifest - collect characters up to null terminator\n let manifestString = \"\";\n let charCode = 0x00;\n while ((charCode = dataView.getUint8(pos++))) {\n manifestString += String.fromCharCode(charCode);\n }\n let manifest = JSON.parse(manifestString);\n manifest = normalizeEnvInfo(manifest);\n if (manifest.specular) {\n // Extend the header with the position of the payload.\n manifest.specular.specularDataPosition = pos;\n // Fallback to 0.8 exactly if lodGenerationScale is not defined for backward compatibility.\n manifest.specular.lodGenerationScale = manifest.specular.lodGenerationScale || 0.8;\n }\n return manifest;\n }\n /**\n * Normalizes any supported version of the environment file info to the latest version\n * @param info environment file info on any supported version\n * @returns environment file info in the latest supported version\n * @private\n */\n function normalizeEnvInfo(info) {\n if (info.version > CurrentVersion) {\n throw new Error(`Unsupported babylon environment map version \"${info.version}\". Latest supported version is \"${CurrentVersion}\".`);\n }\n if (info.version === 2) {\n return info;\n }\n // Migrate a v1 info to v2\n info = Object.assign(Object.assign({}, info), { version: 2, imageType: DefaultEnvironmentTextureImageType });\n return info;\n }\n /**\n * Creates an environment texture from a loaded cube texture.\n * @param texture defines the cube texture to convert in env file\n * @param options options for the conversion process\n * @param options.imageType the mime type for the encoded images, with support for \"image/png\" (default) and \"image/webp\"\n * @param options.imageQuality the image quality of encoded WebP images.\n * @returns a promise containing the environment data if successful.\n */\n async function CreateEnvTextureAsync(texture, options = {}) {\n var _a, _b;\n const internalTexture = texture.getInternalTexture();\n if (!internalTexture) {\n return Promise.reject(\"The cube texture is invalid.\");\n }\n const imageType = (_a = options.imageType) !== null && _a !== void 0 ? _a : DefaultEnvironmentTextureImageType;\n const engine = internalTexture.getEngine();\n if (texture.textureType !== 2 &&\n texture.textureType !== 1 &&\n texture.textureType !== 0 &&\n texture.textureType !== 0 &&\n texture.textureType !== 7 &&\n texture.textureType !== -1) {\n return Promise.reject(\"The cube texture should allow HDR (Full Float or Half Float).\");\n }\n let textureType = 1;\n if (!engine.getCaps().textureFloatRender) {\n textureType = 2;\n if (!engine.getCaps().textureHalfFloatRender) {\n return Promise.reject(\"Env texture can only be created when the browser supports half float or full float rendering.\");\n }\n }\n // sphericalPolynomial is lazy loaded so simply accessing it should trigger the computation.\n texture.sphericalPolynomial;\n // Lets keep track of the polynomial promise so we can wait for it to be ready before generating the pixels.\n const sphericalPolynomialPromise = (_b = texture.getInternalTexture()) === null || _b === void 0 ? void 0 : _b._sphericalPolynomialPromise;\n const cubeWidth = internalTexture.width;\n const hostingScene = new Scene(engine);\n const specularTextures = {};\n // As we are going to readPixels the faces of the cube, make sure the drawing/update commands for the cube texture are fully sent to the GPU in case it is drawn for the first time in this very frame!\n engine.flushFramebuffer();\n // Read and collect all mipmaps data from the cube.\n const mipmapsCount = Scalar.ILog2(internalTexture.width);\n for (let i = 0; i <= mipmapsCount; i++) {\n const faceWidth = Math.pow(2, mipmapsCount - i);\n // All faces of the cube.\n for (let face = 0; face < 6; face++) {\n let faceData = await texture.readPixels(face, i, undefined, false);\n if (faceData && faceData.byteLength === faceData.length) {\n const faceDataFloat = new Float32Array(faceData.byteLength * 4);\n for (let i = 0; i < faceData.byteLength; i++) {\n faceDataFloat[i] = faceData[i] / 255;\n // Gamma to linear\n faceDataFloat[i] = Math.pow(faceDataFloat[i], 2.2);\n }\n faceData = faceDataFloat;\n }\n else if (faceData && texture.gammaSpace) {\n const floatData = faceData;\n for (let i = 0; i < floatData.length; i++) {\n // Gamma to linear\n floatData[i] = Math.pow(floatData[i], 2.2);\n }\n }\n const tempTexture = engine.createRawTexture(faceData, faceWidth, faceWidth, 5, false, true, 1, null, textureType);\n await RGBDTextureTools.EncodeTextureToRGBD(tempTexture, hostingScene, textureType);\n const rgbdEncodedData = await engine._readTexturePixels(tempTexture, faceWidth, faceWidth);\n const imageEncodedData = await DumpTools.DumpDataAsync(faceWidth, faceWidth, rgbdEncodedData, imageType, undefined, false, true, options.imageQuality);\n specularTextures[i * 6 + face] = imageEncodedData;\n tempTexture.dispose();\n }\n }\n // We can delete the hosting scene keeping track of all the creation objects\n hostingScene.dispose();\n // Ensure completion of the polynomial creation promise.\n if (sphericalPolynomialPromise) {\n await sphericalPolynomialPromise;\n }\n // Creates the json header for the env texture\n const info = {\n version: CurrentVersion,\n width: cubeWidth,\n imageType,\n irradiance: _CreateEnvTextureIrradiance(texture),\n specular: {\n mipmaps: [],\n lodGenerationScale: texture.lodGenerationScale,\n },\n };\n // Sets the specular image data information\n let position = 0;\n for (let i = 0; i <= mipmapsCount; i++) {\n for (let face = 0; face < 6; face++) {\n const byteLength = specularTextures[i * 6 + face].byteLength;\n info.specular.mipmaps.push({\n length: byteLength,\n position: position,\n });\n position += byteLength;\n }\n }\n // Encode the JSON as an array buffer\n const infoString = JSON.stringify(info);\n const infoBuffer = new ArrayBuffer(infoString.length + 1);\n const infoView = new Uint8Array(infoBuffer); // Limited to ascii subset matching unicode.\n for (let i = 0, strLen = infoString.length; i < strLen; i++) {\n infoView[i] = infoString.charCodeAt(i);\n }\n // Ends up with a null terminator for easier parsing\n infoView[infoString.length] = 0x00;\n // Computes the final required size and creates the storage\n const totalSize = MagicBytes.length + position + infoBuffer.byteLength;\n const finalBuffer = new ArrayBuffer(totalSize);\n const finalBufferView = new Uint8Array(finalBuffer);\n const dataView = new DataView(finalBuffer);\n // Copy the magic bytes identifying the file in\n let pos = 0;\n for (let i = 0; i < MagicBytes.length; i++) {\n dataView.setUint8(pos++, MagicBytes[i]);\n }\n // Add the json info\n finalBufferView.set(new Uint8Array(infoBuffer), pos);\n pos += infoBuffer.byteLength;\n // Finally inserts the texture data\n for (let i = 0; i <= mipmapsCount; i++) {\n for (let face = 0; face < 6; face++) {\n const dataBuffer = specularTextures[i * 6 + face];\n finalBufferView.set(new Uint8Array(dataBuffer), pos);\n pos += dataBuffer.byteLength;\n }\n }\n // Voila\n return finalBuffer;\n }\n /**\n * Creates a JSON representation of the spherical data.\n * @param texture defines the texture containing the polynomials\n * @returns the JSON representation of the spherical info\n */\n function _CreateEnvTextureIrradiance(texture) {\n const polynmials = texture.sphericalPolynomial;\n if (polynmials == null) {\n return null;\n }\n return {\n x: [polynmials.x.x, polynmials.x.y, polynmials.x.z],\n y: [polynmials.y.x, polynmials.y.y, polynmials.y.z],\n z: [polynmials.z.x, polynmials.z.y, polynmials.z.z],\n xx: [polynmials.xx.x, polynmials.xx.y, polynmials.xx.z],\n yy: [polynmials.yy.x, polynmials.yy.y, polynmials.yy.z],\n zz: [polynmials.zz.x, polynmials.zz.y, polynmials.zz.z],\n yz: [polynmials.yz.x, polynmials.yz.y, polynmials.yz.z],\n zx: [polynmials.zx.x, polynmials.zx.y, polynmials.zx.z],\n xy: [polynmials.xy.x, polynmials.xy.y, polynmials.xy.z],\n };\n }\n /**\n * Creates the ArrayBufferViews used for initializing environment texture image data.\n * @param data the image data\n * @param info parameters that determine what views will be created for accessing the underlying buffer\n * @returns the views described by info providing access to the underlying buffer\n */\n function CreateImageDataArrayBufferViews(data, info) {\n info = normalizeEnvInfo(info);\n const specularInfo = info.specular;\n // Double checks the enclosed info\n let mipmapsCount = Scalar.Log2(info.width);\n mipmapsCount = Math.round(mipmapsCount) + 1;\n if (specularInfo.mipmaps.length !== 6 * mipmapsCount) {\n throw new Error(`Unsupported specular mipmaps number \"${specularInfo.mipmaps.length}\"`);\n }\n const imageData = new Array(mipmapsCount);\n for (let i = 0; i < mipmapsCount; i++) {\n imageData[i] = new Array(6);\n for (let face = 0; face < 6; face++) {\n const imageInfo = specularInfo.mipmaps[i * 6 + face];\n imageData[i][face] = new Uint8Array(data.buffer, data.byteOffset + specularInfo.specularDataPosition + imageInfo.position, imageInfo.length);\n }\n }\n return imageData;\n }\n /**\n * Uploads the texture info contained in the env file to the GPU.\n * @param texture defines the internal texture to upload to\n * @param data defines the data to load\n * @param info defines the texture info retrieved through the GetEnvInfo method\n * @returns a promise\n */\n function UploadEnvLevelsAsync(texture, data, info) {\n info = normalizeEnvInfo(info);\n const specularInfo = info.specular;\n if (!specularInfo) {\n // Nothing else parsed so far\n return Promise.resolve();\n }\n texture._lodGenerationScale = specularInfo.lodGenerationScale;\n const imageData = CreateImageDataArrayBufferViews(data, info);\n return UploadLevelsAsync(texture, imageData, info.imageType);\n }\n function _OnImageReadyAsync(image, engine, expandTexture, rgbdPostProcess, url, face, i, generateNonLODTextures, lodTextures, cubeRtt, texture) {\n return new Promise((resolve, reject) => {\n if (expandTexture) {\n const tempTexture = engine.createTexture(null, true, true, null, 1, null, (message) => {\n reject(message);\n }, image);\n rgbdPostProcess.getEffect().executeWhenCompiled(() => {\n // Uncompress the data to a RTT\n rgbdPostProcess.externalTextureSamplerBinding = true;\n rgbdPostProcess.onApply = (effect) => {\n effect._bindTexture(\"textureSampler\", tempTexture);\n effect.setFloat2(\"scale\", 1, engine._features.needsInvertingBitmap && image instanceof ImageBitmap ? -1 : 1);\n };\n if (!engine.scenes.length) {\n return;\n }\n engine.scenes[0].postProcessManager.directRender([rgbdPostProcess], cubeRtt, true, face, i);\n // Cleanup\n engine.restoreDefaultFramebuffer();\n tempTexture.dispose();\n URL.revokeObjectURL(url);\n resolve();\n });\n }\n else {\n engine._uploadImageToTexture(texture, image, face, i);\n // Upload the face to the non lod texture support\n if (generateNonLODTextures) {\n const lodTexture = lodTextures[i];\n if (lodTexture) {\n engine._uploadImageToTexture(lodTexture._texture, image, face, 0);\n }\n }\n resolve();\n }\n });\n }\n /**\n * Uploads the levels of image data to the GPU.\n * @param texture defines the internal texture to upload to\n * @param imageData defines the array buffer views of image data [mipmap][face]\n * @param imageType the mime type of the image data\n * @returns a promise\n */\n function UploadLevelsAsync(texture, imageData, imageType = DefaultEnvironmentTextureImageType) {\n if (!Tools.IsExponentOfTwo(texture.width)) {\n throw new Error(\"Texture size must be a power of two\");\n }\n const mipmapsCount = Scalar.ILog2(texture.width) + 1;\n // Gets everything ready.\n const engine = texture.getEngine();\n let expandTexture = false;\n let generateNonLODTextures = false;\n let rgbdPostProcess = null;\n let cubeRtt = null;\n let lodTextures = null;\n const caps = engine.getCaps();\n texture.format = 5;\n texture.type = 0;\n texture.generateMipMaps = true;\n texture._cachedAnisotropicFilteringLevel = null;\n engine.updateTextureSamplingMode(3, texture);\n // Add extra process if texture lod is not supported\n if (!caps.textureLOD) {\n expandTexture = false;\n generateNonLODTextures = true;\n lodTextures = {};\n }\n // in webgl 1 there are no ways to either render or copy lod level information for float textures.\n else if (!engine._features.supportRenderAndCopyToLodForFloatTextures) {\n expandTexture = false;\n }\n // If half float available we can uncompress the texture\n else if (caps.textureHalfFloatRender && caps.textureHalfFloatLinearFiltering) {\n expandTexture = true;\n texture.type = 2;\n }\n // If full float available we can uncompress the texture\n else if (caps.textureFloatRender && caps.textureFloatLinearFiltering) {\n expandTexture = true;\n texture.type = 1;\n }\n // Expand the texture if possible\n if (expandTexture) {\n // Simply run through the decode PP\n rgbdPostProcess = new PostProcess(\"rgbdDecode\", \"rgbdDecode\", null, null, 1, null, 3, engine, false, undefined, texture.type, undefined, null, false);\n texture._isRGBD = false;\n texture.invertY = false;\n cubeRtt = engine.createRenderTargetCubeTexture(texture.width, {\n generateDepthBuffer: false,\n generateMipMaps: true,\n generateStencilBuffer: false,\n samplingMode: 3,\n type: texture.type,\n format: 5,\n });\n }\n else {\n texture._isRGBD = true;\n texture.invertY = true;\n // In case of missing support, applies the same patch than DDS files.\n if (generateNonLODTextures) {\n const mipSlices = 3;\n const scale = texture._lodGenerationScale;\n const offset = texture._lodGenerationOffset;\n for (let i = 0; i < mipSlices; i++) {\n //compute LOD from even spacing in smoothness (matching shader calculation)\n const smoothness = i / (mipSlices - 1);\n const roughness = 1 - smoothness;\n const minLODIndex = offset; // roughness = 0\n const maxLODIndex = (mipmapsCount - 1) * scale + offset; // roughness = 1 (mipmaps start from 0)\n const lodIndex = minLODIndex + (maxLODIndex - minLODIndex) * roughness;\n const mipmapIndex = Math.round(Math.min(Math.max(lodIndex, 0), maxLODIndex));\n const glTextureFromLod = new InternalTexture(engine, InternalTextureSource.Temp);\n glTextureFromLod.isCube = true;\n glTextureFromLod.invertY = true;\n glTextureFromLod.generateMipMaps = false;\n engine.updateTextureSamplingMode(2, glTextureFromLod);\n // Wrap in a base texture for easy binding.\n const lodTexture = new BaseTexture(null);\n lodTexture._isCube = true;\n lodTexture._texture = glTextureFromLod;\n lodTextures[mipmapIndex] = lodTexture;\n switch (i) {\n case 0:\n texture._lodTextureLow = lodTexture;\n break;\n case 1:\n texture._lodTextureMid = lodTexture;\n break;\n case 2:\n texture._lodTextureHigh = lodTexture;\n break;\n }\n }\n }\n }\n const promises = [];\n // All mipmaps up to provided number of images\n for (let i = 0; i < imageData.length; i++) {\n // All faces\n for (let face = 0; face < 6; face++) {\n // Constructs an image element from image data\n const bytes = imageData[i][face];\n const blob = new Blob([bytes], { type: imageType });\n const url = URL.createObjectURL(blob);\n let promise;\n if (typeof Image === \"undefined\" || engine._features.forceBitmapOverHTMLImageElement) {\n promise = engine.createImageBitmap(blob, { premultiplyAlpha: \"none\" }).then((img) => {\n return _OnImageReadyAsync(img, engine, expandTexture, rgbdPostProcess, url, face, i, generateNonLODTextures, lodTextures, cubeRtt, texture);\n });\n }\n else {\n const image = new Image();\n image.src = url;\n // Enqueue promise to upload to the texture.\n promise = new Promise((resolve, reject) => {\n image.onload = () => {\n _OnImageReadyAsync(image, engine, expandTexture, rgbdPostProcess, url, face, i, generateNonLODTextures, lodTextures, cubeRtt, texture)\n .then(() => resolve())\n .catch((reason) => {\n reject(reason);\n });\n };\n image.onerror = (error) => {\n reject(error);\n };\n });\n }\n promises.push(promise);\n }\n }\n // Fill remaining mipmaps with black textures.\n if (imageData.length < mipmapsCount) {\n let data;\n const size = Math.pow(2, mipmapsCount - 1 - imageData.length);\n const dataLength = size * size * 4;\n switch (texture.type) {\n case 0: {\n data = new Uint8Array(dataLength);\n break;\n }\n case 2: {\n data = new Uint16Array(dataLength);\n break;\n }\n case 1: {\n data = new Float32Array(dataLength);\n break;\n }\n }\n for (let i = imageData.length; i < mipmapsCount; i++) {\n for (let face = 0; face < 6; face++) {\n engine._uploadArrayBufferViewToTexture(texture, data, face, i);\n }\n }\n }\n // Once all done, finishes the cleanup and return\n return Promise.all(promises).then(() => {\n // Release temp RTT.\n if (cubeRtt) {\n engine._releaseTexture(texture);\n cubeRtt._swapAndDie(texture);\n }\n // Release temp Post Process.\n if (rgbdPostProcess) {\n rgbdPostProcess.dispose();\n }\n // Flag internal texture as ready in case they are in use.\n if (generateNonLODTextures) {\n if (texture._lodTextureHigh && texture._lodTextureHigh._texture) {\n texture._lodTextureHigh._texture.isReady = true;\n }\n if (texture._lodTextureMid && texture._lodTextureMid._texture) {\n texture._lodTextureMid._texture.isReady = true;\n }\n if (texture._lodTextureLow && texture._lodTextureLow._texture) {\n texture._lodTextureLow._texture.isReady = true;\n }\n }\n });\n }\n /**\n * Uploads spherical polynomials information to the texture.\n * @param texture defines the texture we are trying to upload the information to\n * @param info defines the environment texture info retrieved through the GetEnvInfo method\n */\n function UploadEnvSpherical(texture, info) {\n info = normalizeEnvInfo(info);\n const irradianceInfo = info.irradiance;\n if (!irradianceInfo) {\n return;\n }\n const sp = new SphericalPolynomial();\n Vector3.FromArrayToRef(irradianceInfo.x, 0, sp.x);\n Vector3.FromArrayToRef(irradianceInfo.y, 0, sp.y);\n Vector3.FromArrayToRef(irradianceInfo.z, 0, sp.z);\n Vector3.FromArrayToRef(irradianceInfo.xx, 0, sp.xx);\n Vector3.FromArrayToRef(irradianceInfo.yy, 0, sp.yy);\n Vector3.FromArrayToRef(irradianceInfo.zz, 0, sp.zz);\n Vector3.FromArrayToRef(irradianceInfo.yz, 0, sp.yz);\n Vector3.FromArrayToRef(irradianceInfo.zx, 0, sp.zx);\n Vector3.FromArrayToRef(irradianceInfo.xy, 0, sp.xy);\n texture._sphericalPolynomial = sp;\n }\n /**\n * Sets of helpers addressing the serialization and deserialization of environment texture\n * stored in a BabylonJS env file.\n * Those files are usually stored as .env files.\n */\n const EnvironmentTextureTools = {\n /**\n * Gets the environment info from an env file.\n * @param data The array buffer containing the .env bytes.\n * @returns the environment file info (the json header) if successfully parsed, normalized to the latest supported version.\n */\n GetEnvInfo,\n /**\n * Creates an environment texture from a loaded cube texture.\n * @param texture defines the cube texture to convert in env file\n * @param options options for the conversion process\n * @param options.imageType the mime type for the encoded images, with support for \"image/png\" (default) and \"image/webp\"\n * @param options.imageQuality the image quality of encoded WebP images.\n * @returns a promise containing the environment data if successful.\n */\n CreateEnvTextureAsync,\n /**\n * Creates the ArrayBufferViews used for initializing environment texture image data.\n * @param data the image data\n * @param info parameters that determine what views will be created for accessing the underlying buffer\n * @returns the views described by info providing access to the underlying buffer\n */\n CreateImageDataArrayBufferViews,\n /**\n * Uploads the texture info contained in the env file to the GPU.\n * @param texture defines the internal texture to upload to\n * @param data defines the data to load\n * @param info defines the texture info retrieved through the GetEnvInfo method\n * @returns a promise\n */\n UploadEnvLevelsAsync,\n /**\n * Uploads the levels of image data to the GPU.\n * @param texture defines the internal texture to upload to\n * @param imageData defines the array buffer views of image data [mipmap][face]\n * @param imageType the mime type of the image data\n * @returns a promise\n */\n UploadLevelsAsync,\n /**\n * Uploads spherical polynomials information to the texture.\n * @param texture defines the texture we are trying to upload the information to\n * @param info defines the environment texture info retrieved through the GetEnvInfo method\n */\n UploadEnvSpherical,\n };\n\n /**\n * Implementation of the ENV Texture Loader.\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class _ENVTextureLoader {\n constructor() {\n /**\n * Defines whether the loader supports cascade loading the different faces.\n */\n this.supportCascades = false;\n }\n /**\n * This returns if the loader support the current file information.\n * @param extension defines the file extension of the file being loaded\n * @returns true if the loader can load the specified file\n */\n canLoad(extension) {\n return extension.endsWith(\".env\");\n }\n /**\n * Uploads the cube texture data to the WebGL texture. It has already been bound.\n * @param data contains the texture data\n * @param texture defines the BabylonJS internal texture\n * @param createPolynomials will be true if polynomials have been requested\n * @param onLoad defines the callback to trigger once the texture is ready\n * @param onError defines the callback to trigger in case of error\n */\n loadCubeData(data, texture, createPolynomials, onLoad, onError) {\n if (Array.isArray(data)) {\n return;\n }\n const info = GetEnvInfo(data);\n if (info) {\n texture.width = info.width;\n texture.height = info.width;\n try {\n UploadEnvSpherical(texture, info);\n UploadEnvLevelsAsync(texture, data, info).then(() => {\n texture.isReady = true;\n texture.onLoadedObservable.notifyObservers(texture);\n texture.onLoadedObservable.clear();\n if (onLoad) {\n onLoad();\n }\n }, (reason) => {\n onError === null || onError === void 0 ? void 0 : onError(\"Can not upload environment levels\", reason);\n });\n }\n catch (e) {\n onError === null || onError === void 0 ? void 0 : onError(\"Can not upload environment file\", e);\n }\n }\n else if (onError) {\n onError(\"Can not parse the environment file\", null);\n }\n }\n /**\n * Uploads the 2D texture data to the WebGL texture. It has already been bound once in the callback.\n */\n loadData() {\n throw \".env not supported in 2d.\";\n }\n }\n // Register the loader.\n Engine._TextureLoaders.push(new _ENVTextureLoader());\n\n /* eslint-disable @typescript-eslint/naming-convention */\n /**\n * for description see https://www.khronos.org/opengles/sdk/tools/KTX/\n * for file layout see https://www.khronos.org/opengles/sdk/tools/KTX/file_format_spec/\n */\n class KhronosTextureContainer {\n /**\n * Creates a new KhronosTextureContainer\n * @param data contents of the KTX container file\n * @param facesExpected should be either 1 or 6, based whether a cube texture or or\n */\n constructor(\n /** contents of the KTX container file */\n data, facesExpected) {\n this.data = data;\n /**\n * If the container has been made invalid (eg. constructor failed to correctly load array buffer)\n */\n this.isInvalid = false;\n if (!KhronosTextureContainer.IsValid(data)) {\n this.isInvalid = true;\n Logger.Error(\"texture missing KTX identifier\");\n return;\n }\n // load the reset of the header in native 32 bit uint\n const dataSize = Uint32Array.BYTES_PER_ELEMENT;\n const headerDataView = new DataView(this.data.buffer, this.data.byteOffset + 12, 13 * dataSize);\n const endianness = headerDataView.getUint32(0, true);\n const littleEndian = endianness === 0x04030201;\n this.glType = headerDataView.getUint32(1 * dataSize, littleEndian); // must be 0 for compressed textures\n this.glTypeSize = headerDataView.getUint32(2 * dataSize, littleEndian); // must be 1 for compressed textures\n this.glFormat = headerDataView.getUint32(3 * dataSize, littleEndian); // must be 0 for compressed textures\n this.glInternalFormat = headerDataView.getUint32(4 * dataSize, littleEndian); // the value of arg passed to gl.compressedTexImage2D(,,x,,,,)\n this.glBaseInternalFormat = headerDataView.getUint32(5 * dataSize, littleEndian); // specify GL_RGB, GL_RGBA, GL_ALPHA, etc (un-compressed only)\n this.pixelWidth = headerDataView.getUint32(6 * dataSize, littleEndian); // level 0 value of arg passed to gl.compressedTexImage2D(,,,x,,,)\n this.pixelHeight = headerDataView.getUint32(7 * dataSize, littleEndian); // level 0 value of arg passed to gl.compressedTexImage2D(,,,,x,,)\n this.pixelDepth = headerDataView.getUint32(8 * dataSize, littleEndian); // level 0 value of arg passed to gl.compressedTexImage3D(,,,,,x,,)\n this.numberOfArrayElements = headerDataView.getUint32(9 * dataSize, littleEndian); // used for texture arrays\n this.numberOfFaces = headerDataView.getUint32(10 * dataSize, littleEndian); // used for cubemap textures, should either be 1 or 6\n this.numberOfMipmapLevels = headerDataView.getUint32(11 * dataSize, littleEndian); // number of levels; disregard possibility of 0 for compressed textures\n this.bytesOfKeyValueData = headerDataView.getUint32(12 * dataSize, littleEndian); // the amount of space after the header for meta-data\n // Make sure we have a compressed type. Not only reduces work, but probably better to let dev know they are not compressing.\n if (this.glType !== 0) {\n Logger.Error(\"only compressed formats currently supported\");\n this.isInvalid = true;\n return;\n }\n else {\n // value of zero is an indication to generate mipmaps @ runtime. Not usually allowed for compressed, so disregard.\n this.numberOfMipmapLevels = Math.max(1, this.numberOfMipmapLevels);\n }\n if (this.pixelHeight === 0 || this.pixelDepth !== 0) {\n Logger.Error(\"only 2D textures currently supported\");\n this.isInvalid = true;\n return;\n }\n if (this.numberOfArrayElements !== 0) {\n Logger.Error(\"texture arrays not currently supported\");\n this.isInvalid = true;\n return;\n }\n if (this.numberOfFaces !== facesExpected) {\n Logger.Error(\"number of faces expected\" + facesExpected + \", but found \" + this.numberOfFaces);\n this.isInvalid = true;\n return;\n }\n // we now have a completely validated file, so could use existence of loadType as success\n // would need to make this more elaborate & adjust checks above to support more than one load type\n this.loadType = KhronosTextureContainer.COMPRESSED_2D;\n }\n /**\n * Uploads KTX content to a Babylon Texture.\n * It is assumed that the texture has already been created & is currently bound\n * @internal\n */\n uploadLevels(texture, loadMipmaps) {\n switch (this.loadType) {\n case KhronosTextureContainer.COMPRESSED_2D:\n this._upload2DCompressedLevels(texture, loadMipmaps);\n break;\n }\n }\n _upload2DCompressedLevels(texture, loadMipmaps) {\n // initialize width & height for level 1\n let dataOffset = KhronosTextureContainer.HEADER_LEN + this.bytesOfKeyValueData;\n let width = this.pixelWidth;\n let height = this.pixelHeight;\n const mipmapCount = loadMipmaps ? this.numberOfMipmapLevels : 1;\n for (let level = 0; level < mipmapCount; level++) {\n const imageSize = new Int32Array(this.data.buffer, this.data.byteOffset + dataOffset, 1)[0]; // size per face, since not supporting array cubemaps\n dataOffset += 4; //image data starts from next multiple of 4 offset. Each face refers to same imagesize field above.\n for (let face = 0; face < this.numberOfFaces; face++) {\n const byteArray = new Uint8Array(this.data.buffer, this.data.byteOffset + dataOffset, imageSize);\n const engine = texture.getEngine();\n engine._uploadCompressedDataToTextureDirectly(texture, texture.format, width, height, byteArray, face, level);\n dataOffset += imageSize; // add size of the image for the next face/mipmap\n dataOffset += 3 - ((imageSize + 3) % 4); // add padding for odd sized image\n }\n width = Math.max(1.0, width * 0.5);\n height = Math.max(1.0, height * 0.5);\n }\n }\n /**\n * Checks if the given data starts with a KTX file identifier.\n * @param data the data to check\n * @returns true if the data is a KTX file or false otherwise\n */\n static IsValid(data) {\n if (data.byteLength >= 12) {\n // '«', 'K', 'T', 'X', ' ', '1', '1', '»', '\\r', '\\n', '\\x1A', '\\n'\n const identifier = new Uint8Array(data.buffer, data.byteOffset, 12);\n if (identifier[0] === 0xab &&\n identifier[1] === 0x4b &&\n identifier[2] === 0x54 &&\n identifier[3] === 0x58 &&\n identifier[4] === 0x20 &&\n identifier[5] === 0x31 &&\n identifier[6] === 0x31 &&\n identifier[7] === 0xbb &&\n identifier[8] === 0x0d &&\n identifier[9] === 0x0a &&\n identifier[10] === 0x1a &&\n identifier[11] === 0x0a) {\n return true;\n }\n }\n return false;\n }\n }\n KhronosTextureContainer.HEADER_LEN = 12 + 13 * 4; // identifier + header elements (not including key value meta-data pairs)\n // load types\n KhronosTextureContainer.COMPRESSED_2D = 0; // uses a gl.compressedTexImage2D()\n KhronosTextureContainer.COMPRESSED_3D = 1; // uses a gl.compressedTexImage3D()\n KhronosTextureContainer.TEX_2D = 2; // uses a gl.texImage2D()\n KhronosTextureContainer.TEX_3D = 3; // uses a gl.texImage3D()\n\n /**\n * Helper class to push actions to a pool of workers.\n */\n class WorkerPool {\n /**\n * Constructor\n * @param workers Array of workers to use for actions\n */\n constructor(workers) {\n this._pendingActions = new Array();\n this._workerInfos = workers.map((worker) => ({\n workerPromise: Promise.resolve(worker),\n idle: true,\n }));\n }\n /**\n * Terminates all workers and clears any pending actions.\n */\n dispose() {\n for (const workerInfo of this._workerInfos) {\n workerInfo.workerPromise.then((worker) => {\n worker.terminate();\n });\n }\n this._workerInfos.length = 0;\n this._pendingActions.length = 0;\n }\n /**\n * Pushes an action to the worker pool. If all the workers are active, the action will be\n * pended until a worker has completed its action.\n * @param action The action to perform. Call onComplete when the action is complete.\n */\n push(action) {\n if (!this._executeOnIdleWorker(action)) {\n this._pendingActions.push(action);\n }\n }\n _executeOnIdleWorker(action) {\n for (const workerInfo of this._workerInfos) {\n if (workerInfo.idle) {\n this._execute(workerInfo, action);\n return true;\n }\n }\n return false;\n }\n _execute(workerInfo, action) {\n workerInfo.idle = false;\n workerInfo.workerPromise.then((worker) => {\n action(worker, () => {\n const nextAction = this._pendingActions.shift();\n if (nextAction) {\n this._execute(workerInfo, nextAction);\n }\n else {\n workerInfo.idle = true;\n }\n });\n });\n }\n }\n /**\n * Similar to the WorkerPool class except it creates and destroys workers automatically with a maximum of `maxWorkers` workers.\n * Workers are terminated when it is idle for at least `idleTimeElapsedBeforeRelease` milliseconds.\n */\n class AutoReleaseWorkerPool extends WorkerPool {\n constructor(maxWorkers, createWorkerAsync, options = AutoReleaseWorkerPool.DefaultOptions) {\n super([]);\n this._maxWorkers = maxWorkers;\n this._createWorkerAsync = createWorkerAsync;\n this._options = options;\n }\n push(action) {\n if (!this._executeOnIdleWorker(action)) {\n if (this._workerInfos.length < this._maxWorkers) {\n const workerInfo = {\n workerPromise: this._createWorkerAsync(),\n idle: false,\n };\n this._workerInfos.push(workerInfo);\n this._execute(workerInfo, action);\n }\n else {\n this._pendingActions.push(action);\n }\n }\n }\n _execute(workerInfo, action) {\n // Reset the idle timeout.\n if (workerInfo.timeoutId) {\n clearTimeout(workerInfo.timeoutId);\n delete workerInfo.timeoutId;\n }\n super._execute(workerInfo, (worker, onComplete) => {\n action(worker, () => {\n onComplete();\n if (workerInfo.idle) {\n // Schedule the worker to be terminated after the elapsed time.\n workerInfo.timeoutId = setTimeout(() => {\n workerInfo.workerPromise.then((worker) => {\n worker.terminate();\n });\n const indexOf = this._workerInfos.indexOf(workerInfo);\n if (indexOf !== -1) {\n this._workerInfos.splice(indexOf, 1);\n }\n }, this._options.idleTimeElapsedBeforeRelease);\n }\n });\n });\n }\n }\n /**\n * Default options for the constructor.\n * Override to change the defaults.\n */\n AutoReleaseWorkerPool.DefaultOptions = {\n idleTimeElapsedBeforeRelease: 1000,\n };\n\n var SourceTextureFormat;\n (function (SourceTextureFormat) {\n SourceTextureFormat[SourceTextureFormat[\"ETC1S\"] = 0] = \"ETC1S\";\n SourceTextureFormat[SourceTextureFormat[\"UASTC4x4\"] = 1] = \"UASTC4x4\";\n })(SourceTextureFormat || (SourceTextureFormat = {}));\n var TranscodeTarget;\n (function (TranscodeTarget) {\n TranscodeTarget[TranscodeTarget[\"ASTC_4X4_RGBA\"] = 0] = \"ASTC_4X4_RGBA\";\n TranscodeTarget[TranscodeTarget[\"BC7_RGBA\"] = 1] = \"BC7_RGBA\";\n TranscodeTarget[TranscodeTarget[\"BC3_RGBA\"] = 2] = \"BC3_RGBA\";\n TranscodeTarget[TranscodeTarget[\"BC1_RGB\"] = 3] = \"BC1_RGB\";\n TranscodeTarget[TranscodeTarget[\"PVRTC1_4_RGBA\"] = 4] = \"PVRTC1_4_RGBA\";\n TranscodeTarget[TranscodeTarget[\"PVRTC1_4_RGB\"] = 5] = \"PVRTC1_4_RGB\";\n TranscodeTarget[TranscodeTarget[\"ETC2_RGBA\"] = 6] = \"ETC2_RGBA\";\n TranscodeTarget[TranscodeTarget[\"ETC1_RGB\"] = 7] = \"ETC1_RGB\";\n TranscodeTarget[TranscodeTarget[\"RGBA32\"] = 8] = \"RGBA32\";\n TranscodeTarget[TranscodeTarget[\"R8\"] = 9] = \"R8\";\n TranscodeTarget[TranscodeTarget[\"RG8\"] = 10] = \"RG8\";\n })(TranscodeTarget || (TranscodeTarget = {}));\n var EngineFormat;\n (function (EngineFormat) {\n EngineFormat[EngineFormat[\"COMPRESSED_RGBA_BPTC_UNORM_EXT\"] = 36492] = \"COMPRESSED_RGBA_BPTC_UNORM_EXT\";\n EngineFormat[EngineFormat[\"COMPRESSED_RGBA_ASTC_4X4_KHR\"] = 37808] = \"COMPRESSED_RGBA_ASTC_4X4_KHR\";\n EngineFormat[EngineFormat[\"COMPRESSED_RGB_S3TC_DXT1_EXT\"] = 33776] = \"COMPRESSED_RGB_S3TC_DXT1_EXT\";\n EngineFormat[EngineFormat[\"COMPRESSED_RGBA_S3TC_DXT5_EXT\"] = 33779] = \"COMPRESSED_RGBA_S3TC_DXT5_EXT\";\n EngineFormat[EngineFormat[\"COMPRESSED_RGBA_PVRTC_4BPPV1_IMG\"] = 35842] = \"COMPRESSED_RGBA_PVRTC_4BPPV1_IMG\";\n EngineFormat[EngineFormat[\"COMPRESSED_RGB_PVRTC_4BPPV1_IMG\"] = 35840] = \"COMPRESSED_RGB_PVRTC_4BPPV1_IMG\";\n EngineFormat[EngineFormat[\"COMPRESSED_RGBA8_ETC2_EAC\"] = 37496] = \"COMPRESSED_RGBA8_ETC2_EAC\";\n EngineFormat[EngineFormat[\"COMPRESSED_RGB8_ETC2\"] = 37492] = \"COMPRESSED_RGB8_ETC2\";\n EngineFormat[EngineFormat[\"COMPRESSED_RGB_ETC1_WEBGL\"] = 36196] = \"COMPRESSED_RGB_ETC1_WEBGL\";\n EngineFormat[EngineFormat[\"RGBA8Format\"] = 32856] = \"RGBA8Format\";\n EngineFormat[EngineFormat[\"R8Format\"] = 33321] = \"R8Format\";\n EngineFormat[EngineFormat[\"RG8Format\"] = 33323] = \"RG8Format\";\n })(EngineFormat || (EngineFormat = {}));\n\n function getAbsoluteUrlOrNull(url) {\n return url ? Tools.GetAbsoluteUrl(url) : null;\n }\n function applyConfig(urls) {\n if (urls.wasmUASTCToASTC !== null) {\n KTX2DECODER.LiteTranscoder_UASTC_ASTC.WasmModuleURL = urls.wasmUASTCToASTC;\n }\n if (urls.wasmUASTCToBC7 !== null) {\n KTX2DECODER.LiteTranscoder_UASTC_BC7.WasmModuleURL = urls.wasmUASTCToBC7;\n }\n if (urls.wasmUASTCToRGBA_UNORM !== null) {\n KTX2DECODER.LiteTranscoder_UASTC_RGBA_UNORM.WasmModuleURL = urls.wasmUASTCToRGBA_UNORM;\n }\n if (urls.wasmUASTCToRGBA_SRGB !== null) {\n KTX2DECODER.LiteTranscoder_UASTC_RGBA_SRGB.WasmModuleURL = urls.wasmUASTCToRGBA_SRGB;\n }\n if (urls.wasmUASTCToR8_UNORM !== null) {\n KTX2DECODER.LiteTranscoder_UASTC_R8_UNORM.WasmModuleURL = urls.wasmUASTCToR8_UNORM;\n }\n if (urls.wasmUASTCToRG8_UNORM !== null) {\n KTX2DECODER.LiteTranscoder_UASTC_RG8_UNORM.WasmModuleURL = urls.wasmUASTCToRG8_UNORM;\n }\n if (urls.jsMSCTranscoder !== null) {\n KTX2DECODER.MSCTranscoder.JSModuleURL = urls.jsMSCTranscoder;\n }\n if (urls.wasmMSCTranscoder !== null) {\n KTX2DECODER.MSCTranscoder.WasmModuleURL = urls.wasmMSCTranscoder;\n }\n if (urls.wasmZSTDDecoder !== null) {\n KTX2DECODER.ZSTDDecoder.WasmModuleURL = urls.wasmZSTDDecoder;\n }\n }\n /**\n * Class that defines the default KTX2 decoder options.\n *\n * This class is useful for providing options to the KTX2 decoder to control how the source data is transcoded.\n */\n class DefaultKTX2DecoderOptions {\n constructor() {\n this._isDirty = true;\n this._useRGBAIfOnlyBC1BC3AvailableWhenUASTC = true;\n this._ktx2DecoderOptions = {};\n }\n /**\n * Gets the dirty flag\n */\n get isDirty() {\n return this._isDirty;\n }\n /**\n * force a (uncompressed) RGBA transcoded format if transcoding a UASTC source format and ASTC + BC7 are not available as a compressed transcoded format\n */\n get useRGBAIfASTCBC7NotAvailableWhenUASTC() {\n return this._useRGBAIfASTCBC7NotAvailableWhenUASTC;\n }\n set useRGBAIfASTCBC7NotAvailableWhenUASTC(value) {\n if (this._useRGBAIfASTCBC7NotAvailableWhenUASTC === value) {\n return;\n }\n this._useRGBAIfASTCBC7NotAvailableWhenUASTC = value;\n this._isDirty = true;\n }\n /**\n * force a (uncompressed) RGBA transcoded format if transcoding a UASTC source format and only BC1 or BC3 are available as a compressed transcoded format.\n * This property is true by default to favor speed over memory, because currently transcoding from UASTC to BC1/3 is slow because the transcoder transcodes\n * to uncompressed and then recompresses the texture\n */\n get useRGBAIfOnlyBC1BC3AvailableWhenUASTC() {\n return this._useRGBAIfOnlyBC1BC3AvailableWhenUASTC;\n }\n set useRGBAIfOnlyBC1BC3AvailableWhenUASTC(value) {\n if (this._useRGBAIfOnlyBC1BC3AvailableWhenUASTC === value) {\n return;\n }\n this._useRGBAIfOnlyBC1BC3AvailableWhenUASTC = value;\n this._isDirty = true;\n }\n /**\n * force to always use (uncompressed) RGBA for transcoded format\n */\n get forceRGBA() {\n return this._forceRGBA;\n }\n set forceRGBA(value) {\n if (this._forceRGBA === value) {\n return;\n }\n this._forceRGBA = value;\n this._isDirty = true;\n }\n /**\n * force to always use (uncompressed) R8 for transcoded format\n */\n get forceR8() {\n return this._forceR8;\n }\n set forceR8(value) {\n if (this._forceR8 === value) {\n return;\n }\n this._forceR8 = value;\n this._isDirty = true;\n }\n /**\n * force to always use (uncompressed) RG8 for transcoded format\n */\n get forceRG8() {\n return this._forceRG8;\n }\n set forceRG8(value) {\n if (this._forceRG8 === value) {\n return;\n }\n this._forceRG8 = value;\n this._isDirty = true;\n }\n /**\n * list of transcoders to bypass when looking for a suitable transcoder. The available transcoders are:\n * UniversalTranscoder_UASTC_ASTC\n * UniversalTranscoder_UASTC_BC7\n * UniversalTranscoder_UASTC_RGBA_UNORM\n * UniversalTranscoder_UASTC_RGBA_SRGB\n * UniversalTranscoder_UASTC_R8_UNORM\n * UniversalTranscoder_UASTC_RG8_UNORM\n * MSCTranscoder\n */\n get bypassTranscoders() {\n return this._bypassTranscoders;\n }\n set bypassTranscoders(value) {\n if (this._bypassTranscoders === value) {\n return;\n }\n this._bypassTranscoders = value;\n this._isDirty = true;\n }\n /** @internal */\n _getKTX2DecoderOptions() {\n if (!this._isDirty) {\n return this._ktx2DecoderOptions;\n }\n this._isDirty = false;\n const options = {\n useRGBAIfASTCBC7NotAvailableWhenUASTC: this._useRGBAIfASTCBC7NotAvailableWhenUASTC,\n forceRGBA: this._forceRGBA,\n forceR8: this._forceR8,\n forceRG8: this._forceRG8,\n bypassTranscoders: this._bypassTranscoders,\n };\n if (this.useRGBAIfOnlyBC1BC3AvailableWhenUASTC) {\n options.transcodeFormatDecisionTree = {\n UASTC: {\n transcodeFormat: [TranscodeTarget.BC1_RGB, TranscodeTarget.BC3_RGBA],\n yes: {\n transcodeFormat: TranscodeTarget.RGBA32,\n engineFormat: EngineFormat.RGBA8Format,\n roundToMultiple4: false,\n },\n },\n };\n }\n this._ktx2DecoderOptions = options;\n return options;\n }\n }\n /**\n * Class for loading KTX2 files\n */\n class KhronosTextureContainer2 {\n static GetDefaultNumWorkers() {\n if (typeof navigator !== \"object\" || !navigator.hardwareConcurrency) {\n return 1;\n }\n // Use 50% of the available logical processors but capped at 4.\n return Math.min(Math.floor(navigator.hardwareConcurrency * 0.5), 4);\n }\n static _Initialize(numWorkers) {\n if (KhronosTextureContainer2._WorkerPoolPromise || KhronosTextureContainer2._DecoderModulePromise) {\n return;\n }\n const urls = {\n jsDecoderModule: Tools.GetAbsoluteUrl(this.URLConfig.jsDecoderModule),\n wasmUASTCToASTC: getAbsoluteUrlOrNull(this.URLConfig.wasmUASTCToASTC),\n wasmUASTCToBC7: getAbsoluteUrlOrNull(this.URLConfig.wasmUASTCToBC7),\n wasmUASTCToRGBA_UNORM: getAbsoluteUrlOrNull(this.URLConfig.wasmUASTCToRGBA_UNORM),\n wasmUASTCToRGBA_SRGB: getAbsoluteUrlOrNull(this.URLConfig.wasmUASTCToRGBA_SRGB),\n wasmUASTCToR8_UNORM: getAbsoluteUrlOrNull(this.URLConfig.wasmUASTCToR8_UNORM),\n wasmUASTCToRG8_UNORM: getAbsoluteUrlOrNull(this.URLConfig.wasmUASTCToRG8_UNORM),\n jsMSCTranscoder: getAbsoluteUrlOrNull(this.URLConfig.jsMSCTranscoder),\n wasmMSCTranscoder: getAbsoluteUrlOrNull(this.URLConfig.wasmMSCTranscoder),\n wasmZSTDDecoder: getAbsoluteUrlOrNull(this.URLConfig.wasmZSTDDecoder),\n };\n if (numWorkers && typeof Worker === \"function\" && typeof URL !== \"undefined\") {\n KhronosTextureContainer2._WorkerPoolPromise = new Promise((resolve) => {\n const workerContent = `${applyConfig}(${workerFunc})()`;\n const workerBlobUrl = URL.createObjectURL(new Blob([workerContent], { type: \"application/javascript\" }));\n resolve(new AutoReleaseWorkerPool(numWorkers, () => new Promise((resolve, reject) => {\n const worker = new Worker(workerBlobUrl);\n const onError = (error) => {\n worker.removeEventListener(\"error\", onError);\n worker.removeEventListener(\"message\", onMessage);\n reject(error);\n };\n const onMessage = (message) => {\n if (message.data.action === \"init\") {\n worker.removeEventListener(\"error\", onError);\n worker.removeEventListener(\"message\", onMessage);\n resolve(worker);\n }\n };\n worker.addEventListener(\"error\", onError);\n worker.addEventListener(\"message\", onMessage);\n worker.postMessage({\n action: \"init\",\n urls: urls,\n });\n })));\n });\n }\n else if (typeof KTX2DECODER === \"undefined\") {\n KhronosTextureContainer2._DecoderModulePromise = Tools.LoadScriptAsync(urls.jsDecoderModule).then(() => {\n KTX2DECODER.MSCTranscoder.UseFromWorkerThread = false;\n KTX2DECODER.WASMMemoryManager.LoadBinariesFromCurrentThread = true;\n applyConfig(urls);\n return new KTX2DECODER.KTX2Decoder();\n });\n }\n else {\n KTX2DECODER.MSCTranscoder.UseFromWorkerThread = false;\n KTX2DECODER.WASMMemoryManager.LoadBinariesFromCurrentThread = true;\n KhronosTextureContainer2._DecoderModulePromise = Promise.resolve(new KTX2DECODER.KTX2Decoder());\n }\n }\n /**\n * Constructor\n * @param engine The engine to use\n * @param numWorkers The number of workers for async operations. Specify `0` to disable web workers and run synchronously in the current context.\n */\n constructor(engine, numWorkers = KhronosTextureContainer2.DefaultNumWorkers) {\n this._engine = engine;\n KhronosTextureContainer2._Initialize(numWorkers);\n }\n /**\n * @internal\n */\n uploadAsync(data, internalTexture, options) {\n const caps = this._engine.getCaps();\n const compressedTexturesCaps = {\n astc: !!caps.astc,\n bptc: !!caps.bptc,\n s3tc: !!caps.s3tc,\n pvrtc: !!caps.pvrtc,\n etc2: !!caps.etc2,\n etc1: !!caps.etc1,\n };\n if (KhronosTextureContainer2._WorkerPoolPromise) {\n return KhronosTextureContainer2._WorkerPoolPromise.then((workerPool) => {\n return new Promise((resolve, reject) => {\n workerPool.push((worker, onComplete) => {\n const onError = (error) => {\n worker.removeEventListener(\"error\", onError);\n worker.removeEventListener(\"message\", onMessage);\n reject(error);\n onComplete();\n };\n const onMessage = (message) => {\n if (message.data.action === \"decoded\") {\n worker.removeEventListener(\"error\", onError);\n worker.removeEventListener(\"message\", onMessage);\n if (!message.data.success) {\n reject({ message: message.data.msg });\n }\n else {\n try {\n this._createTexture(message.data.decodedData, internalTexture, options);\n resolve();\n }\n catch (err) {\n reject({ message: err });\n }\n }\n onComplete();\n }\n };\n worker.addEventListener(\"error\", onError);\n worker.addEventListener(\"message\", onMessage);\n worker.postMessage({ action: \"setDefaultDecoderOptions\", options: KhronosTextureContainer2.DefaultDecoderOptions._getKTX2DecoderOptions() });\n const dataCopy = new Uint8Array(data.byteLength);\n dataCopy.set(new Uint8Array(data.buffer, data.byteOffset, data.byteLength));\n worker.postMessage({ action: \"decode\", data: dataCopy, caps: compressedTexturesCaps, options }, [dataCopy.buffer]);\n });\n });\n });\n }\n else if (KhronosTextureContainer2._DecoderModulePromise) {\n return KhronosTextureContainer2._DecoderModulePromise.then((decoder) => {\n if (KhronosTextureContainer2.DefaultDecoderOptions.isDirty) {\n KTX2DECODER.KTX2Decoder.DefaultDecoderOptions = KhronosTextureContainer2.DefaultDecoderOptions._getKTX2DecoderOptions();\n }\n return new Promise((resolve, reject) => {\n decoder\n .decode(data, caps)\n .then((data) => {\n this._createTexture(data, internalTexture);\n resolve();\n })\n .catch((reason) => {\n reject({ message: reason });\n });\n });\n });\n }\n throw new Error(\"KTX2 decoder module is not available\");\n }\n _createTexture(data, internalTexture, options) {\n const oglTexture2D = 3553; // gl.TEXTURE_2D\n this._engine._bindTextureDirectly(oglTexture2D, internalTexture);\n if (options) {\n // return back some information about the decoded data\n options.transcodedFormat = data.transcodedFormat;\n options.isInGammaSpace = data.isInGammaSpace;\n options.hasAlpha = data.hasAlpha;\n options.transcoderName = data.transcoderName;\n }\n let isUncompressedFormat = true;\n switch (data.transcodedFormat) {\n case 0x8058 /* RGBA8 */:\n internalTexture.type = 0;\n internalTexture.format = 5;\n break;\n case 0x8229 /* R8 */:\n internalTexture.type = 0;\n internalTexture.format = 6;\n break;\n case 0x822b /* RG8 */:\n internalTexture.type = 0;\n internalTexture.format = 7;\n break;\n default:\n internalTexture.format = data.transcodedFormat;\n isUncompressedFormat = false;\n break;\n }\n internalTexture._gammaSpace = data.isInGammaSpace;\n internalTexture.generateMipMaps = data.mipmaps.length > 1;\n if (data.errors) {\n throw new Error(\"KTX2 container - could not transcode the data. \" + data.errors);\n }\n for (let t = 0; t < data.mipmaps.length; ++t) {\n const mipmap = data.mipmaps[t];\n if (!mipmap || !mipmap.data) {\n throw new Error(\"KTX2 container - could not transcode one of the image\");\n }\n if (isUncompressedFormat) {\n // uncompressed RGBA / R8 / RG8\n internalTexture.width = mipmap.width; // need to set width/height so that the call to _uploadDataToTextureDirectly uses the right dimensions\n internalTexture.height = mipmap.height;\n this._engine._uploadDataToTextureDirectly(internalTexture, mipmap.data, 0, t, undefined, true);\n }\n else {\n this._engine._uploadCompressedDataToTextureDirectly(internalTexture, data.transcodedFormat, mipmap.width, mipmap.height, mipmap.data, 0, t);\n }\n }\n internalTexture._extension = \".ktx2\";\n internalTexture.width = data.mipmaps[0].width;\n internalTexture.height = data.mipmaps[0].height;\n internalTexture.isReady = true;\n this._engine._bindTextureDirectly(oglTexture2D, null);\n }\n /**\n * Checks if the given data starts with a KTX2 file identifier.\n * @param data the data to check\n * @returns true if the data is a KTX2 file or false otherwise\n */\n static IsValid(data) {\n if (data.byteLength >= 12) {\n // '«', 'K', 'T', 'X', ' ', '2', '0', '»', '\\r', '\\n', '\\x1A', '\\n'\n const identifier = new Uint8Array(data.buffer, data.byteOffset, 12);\n if (identifier[0] === 0xab &&\n identifier[1] === 0x4b &&\n identifier[2] === 0x54 &&\n identifier[3] === 0x58 &&\n identifier[4] === 0x20 &&\n identifier[5] === 0x32 &&\n identifier[6] === 0x30 &&\n identifier[7] === 0xbb &&\n identifier[8] === 0x0d &&\n identifier[9] === 0x0a &&\n identifier[10] === 0x1a &&\n identifier[11] === 0x0a) {\n return true;\n }\n }\n return false;\n }\n }\n /**\n * URLs to use when loading the KTX2 decoder module as well as its dependencies\n * If a url is null, the default url is used (pointing to https://preview.babylonjs.com)\n * Note that jsDecoderModule can't be null and that the other dependencies will only be loaded if necessary\n * Urls you can change:\n * URLConfig.jsDecoderModule\n * URLConfig.wasmUASTCToASTC\n * URLConfig.wasmUASTCToBC7\n * URLConfig.wasmUASTCToRGBA_UNORM\n * URLConfig.wasmUASTCToRGBA_SRGB\n * URLConfig.wasmUASTCToR8_UNORM\n * URLConfig.wasmUASTCToRG8_UNORM\n * URLConfig.jsMSCTranscoder\n * URLConfig.wasmMSCTranscoder\n * URLConfig.wasmZSTDDecoder\n * You can see their default values in this PG: https://playground.babylonjs.com/#EIJH8L#29\n */\n KhronosTextureContainer2.URLConfig = {\n jsDecoderModule: \"https://preview.babylonjs.com/babylon.ktx2Decoder.js\",\n wasmUASTCToASTC: null,\n wasmUASTCToBC7: null,\n wasmUASTCToRGBA_UNORM: null,\n wasmUASTCToRGBA_SRGB: null,\n wasmUASTCToR8_UNORM: null,\n wasmUASTCToRG8_UNORM: null,\n jsMSCTranscoder: null,\n wasmMSCTranscoder: null,\n wasmZSTDDecoder: null,\n };\n /**\n * Default number of workers used to handle data decoding\n */\n KhronosTextureContainer2.DefaultNumWorkers = KhronosTextureContainer2.GetDefaultNumWorkers();\n /**\n * Default configuration for the KTX2 decoder.\n * The options defined in this way have priority over those passed when creating a KTX2 texture with new Texture(...).\n */\n KhronosTextureContainer2.DefaultDecoderOptions = new DefaultKTX2DecoderOptions();\n function workerFunc() {\n let ktx2Decoder;\n onmessage = (event) => {\n if (!event.data) {\n return;\n }\n switch (event.data.action) {\n case \"init\": {\n const urls = event.data.urls;\n importScripts(urls.jsDecoderModule);\n applyConfig(urls);\n ktx2Decoder = new KTX2DECODER.KTX2Decoder();\n postMessage({ action: \"init\" });\n break;\n }\n case \"setDefaultDecoderOptions\": {\n KTX2DECODER.KTX2Decoder.DefaultDecoderOptions = event.data.options;\n break;\n }\n case \"decode\":\n ktx2Decoder\n .decode(event.data.data, event.data.caps, event.data.options)\n .then((data) => {\n const buffers = [];\n for (let mip = 0; mip < data.mipmaps.length; ++mip) {\n const mipmap = data.mipmaps[mip];\n if (mipmap && mipmap.data) {\n buffers.push(mipmap.data.buffer);\n }\n }\n postMessage({ action: \"decoded\", success: true, decodedData: data }, buffers);\n })\n .catch((reason) => {\n postMessage({ action: \"decoded\", success: false, msg: reason });\n });\n break;\n }\n };\n }\n\n function mapSRGBToLinear(format) {\n switch (format) {\n case 35916:\n return 33776;\n case 35918:\n return 33778;\n case 35919:\n return 33779;\n case 37493:\n return 37492;\n case 37497:\n return 37496;\n case 37495:\n return 37494;\n case 37840:\n return 37808;\n case 36493:\n return 36492;\n }\n return null;\n }\n /**\n * Implementation of the KTX Texture Loader.\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class _KTXTextureLoader {\n constructor() {\n /**\n * Defines whether the loader supports cascade loading the different faces.\n */\n this.supportCascades = false;\n }\n /**\n * This returns if the loader support the current file information.\n * @param extension defines the file extension of the file being loaded\n * @param mimeType defines the optional mime type of the file being loaded\n * @returns true if the loader can load the specified file\n */\n canLoad(extension, mimeType) {\n // The \".ktx2\" file extension is still up for debate: https://github.com/KhronosGroup/KTX-Specification/issues/18\n return extension.endsWith(\".ktx\") || extension.endsWith(\".ktx2\") || mimeType === \"image/ktx\" || mimeType === \"image/ktx2\";\n }\n /**\n * Uploads the cube texture data to the WebGL texture. It has already been bound.\n * @param data contains the texture data\n * @param texture defines the BabylonJS internal texture\n * @param createPolynomials will be true if polynomials have been requested\n * @param onLoad defines the callback to trigger once the texture is ready\n */\n loadCubeData(data, texture, createPolynomials, onLoad) {\n if (Array.isArray(data)) {\n return;\n }\n // Need to invert vScale as invertY via UNPACK_FLIP_Y_WEBGL is not supported by compressed texture\n texture._invertVScale = !texture.invertY;\n const engine = texture.getEngine();\n const ktx = new KhronosTextureContainer(data, 6);\n const loadMipmap = ktx.numberOfMipmapLevels > 1 && texture.generateMipMaps;\n engine._unpackFlipY(true);\n ktx.uploadLevels(texture, texture.generateMipMaps);\n texture.width = ktx.pixelWidth;\n texture.height = ktx.pixelHeight;\n engine._setCubeMapTextureParams(texture, loadMipmap, ktx.numberOfMipmapLevels - 1);\n texture.isReady = true;\n texture.onLoadedObservable.notifyObservers(texture);\n texture.onLoadedObservable.clear();\n if (onLoad) {\n onLoad();\n }\n }\n /**\n * Uploads the 2D texture data to the WebGL texture. It has already been bound once in the callback.\n * @param data contains the texture data\n * @param texture defines the BabylonJS internal texture\n * @param callback defines the method to call once ready to upload\n * @param options\n */\n loadData(data, texture, callback, options) {\n if (KhronosTextureContainer.IsValid(data)) {\n // Need to invert vScale as invertY via UNPACK_FLIP_Y_WEBGL is not supported by compressed texture\n texture._invertVScale = !texture.invertY;\n const ktx = new KhronosTextureContainer(data, 1);\n const mappedFormat = mapSRGBToLinear(ktx.glInternalFormat);\n if (mappedFormat) {\n texture.format = mappedFormat;\n texture._useSRGBBuffer = texture.getEngine()._getUseSRGBBuffer(true, texture.generateMipMaps);\n texture._gammaSpace = true;\n }\n else {\n texture.format = ktx.glInternalFormat;\n }\n callback(ktx.pixelWidth, ktx.pixelHeight, texture.generateMipMaps, true, () => {\n ktx.uploadLevels(texture, texture.generateMipMaps);\n }, ktx.isInvalid);\n }\n else if (KhronosTextureContainer2.IsValid(data)) {\n const ktx2 = new KhronosTextureContainer2(texture.getEngine());\n ktx2.uploadAsync(data, texture, options).then(() => {\n callback(texture.width, texture.height, texture.generateMipMaps, true, () => { }, false);\n }, (error) => {\n Logger.Warn(`Failed to load KTX2 texture data: ${error.message}`);\n callback(0, 0, false, false, () => { }, true);\n });\n }\n else {\n Logger.Error(\"texture missing KTX identifier\");\n callback(0, 0, false, false, () => { }, true);\n }\n }\n }\n // Register the loader.\n Engine._TextureLoaders.unshift(new _KTXTextureLoader());\n\n /**\n * WebXR Camera which holds the views for the xrSession\n * @see https://doc.babylonjs.com/features/featuresDeepDive/webXR/webXRCamera\n */\n class WebXRCamera extends FreeCamera {\n /**\n * Creates a new webXRCamera, this should only be set at the camera after it has been updated by the xrSessionManager\n * @param name the name of the camera\n * @param scene the scene to add the camera to\n * @param _xrSessionManager a constructed xr session manager\n */\n constructor(name, scene, _xrSessionManager) {\n super(name, Vector3.Zero(), scene);\n this._xrSessionManager = _xrSessionManager;\n this._firstFrame = false;\n this._referenceQuaternion = Quaternion.Identity();\n this._referencedPosition = new Vector3();\n this._trackingState = WebXRTrackingState.NOT_TRACKING;\n /**\n * Observable raised before camera teleportation\n */\n this.onBeforeCameraTeleport = new Observable$1();\n /**\n * Observable raised after camera teleportation\n */\n this.onAfterCameraTeleport = new Observable$1();\n /**\n * Notifies when the camera's tracking state has changed.\n * Notice - will also be triggered when tracking has started (at the beginning of the session)\n */\n this.onTrackingStateChanged = new Observable$1();\n /**\n * Should position compensation execute on first frame.\n * This is used when copying the position from a native (non XR) camera\n */\n this.compensateOnFirstFrame = true;\n this._rotate180 = new Quaternion(0, 1, 0, 0);\n // Initial camera configuration\n this.minZ = 0.1;\n this.rotationQuaternion = new Quaternion();\n this.cameraRigMode = Camera.RIG_MODE_CUSTOM;\n this.updateUpVectorFromRotation = true;\n this._updateNumberOfRigCameras(1);\n // freeze projection matrix, which will be copied later\n this.freezeProjectionMatrix();\n this._deferOnly = true;\n this._xrSessionManager.onXRSessionInit.add(() => {\n this._referencedPosition.copyFromFloats(0, 0, 0);\n this._referenceQuaternion.copyFromFloats(0, 0, 0, 1);\n // first frame - camera's y position should be 0 for the correct offset\n this._firstFrame = this.compensateOnFirstFrame;\n });\n // Check transformation changes on each frame. Callback is added to be first so that the transformation will be\n // applied to the rest of the elements using the referenceSpace object\n this._xrSessionManager.onXRFrameObservable.add(() => {\n if (this._firstFrame) {\n this._updateFromXRSession();\n }\n if (this._deferredUpdated) {\n this.position.copyFrom(this._deferredPositionUpdate);\n this.rotationQuaternion.copyFrom(this._deferredRotationQuaternionUpdate);\n }\n this._updateReferenceSpace();\n this._updateFromXRSession();\n }, undefined, true);\n }\n /**\n * Get the current XR tracking state of the camera\n */\n get trackingState() {\n return this._trackingState;\n }\n _setTrackingState(newState) {\n if (this._trackingState !== newState) {\n this._trackingState = newState;\n this.onTrackingStateChanged.notifyObservers(newState);\n }\n }\n /**\n * Return the user's height, unrelated to the current ground.\n * This will be the y position of this camera, when ground level is 0.\n */\n get realWorldHeight() {\n const basePose = this._xrSessionManager.currentFrame && this._xrSessionManager.currentFrame.getViewerPose(this._xrSessionManager.baseReferenceSpace);\n if (basePose && basePose.transform) {\n return basePose.transform.position.y;\n }\n else {\n return 0;\n }\n }\n /** @internal */\n _updateForDualEyeDebugging( /*pupilDistance = 0.01*/) {\n // Create initial camera rigs\n this._updateNumberOfRigCameras(2);\n this.rigCameras[0].viewport = new Viewport(0, 0, 0.5, 1.0);\n // this.rigCameras[0].position.x = -pupilDistance / 2;\n this.rigCameras[0].outputRenderTarget = null;\n this.rigCameras[1].viewport = new Viewport(0.5, 0, 0.5, 1.0);\n // this.rigCameras[1].position.x = pupilDistance / 2;\n this.rigCameras[1].outputRenderTarget = null;\n }\n /**\n * Sets this camera's transformation based on a non-vr camera\n * @param otherCamera the non-vr camera to copy the transformation from\n * @param resetToBaseReferenceSpace should XR reset to the base reference space\n */\n setTransformationFromNonVRCamera(otherCamera = this.getScene().activeCamera, resetToBaseReferenceSpace = true) {\n if (!otherCamera || otherCamera === this) {\n return;\n }\n const mat = otherCamera.computeWorldMatrix();\n mat.decompose(undefined, this.rotationQuaternion, this.position);\n // set the ground level\n this.position.y = 0;\n Quaternion.FromEulerAnglesToRef(0, this.rotationQuaternion.toEulerAngles().y, 0, this.rotationQuaternion);\n this._firstFrame = true;\n if (resetToBaseReferenceSpace) {\n this._xrSessionManager.resetReferenceSpace();\n }\n }\n /**\n * Gets the current instance class name (\"WebXRCamera\").\n * @returns the class name\n */\n getClassName() {\n return \"WebXRCamera\";\n }\n /**\n * Set the target for the camera to look at.\n * Note that this only rotates around the Y axis, as opposed to the default behavior of other cameras\n * @param target the target to set the camera to look at\n */\n setTarget(target) {\n // only rotate around the y axis!\n const tmpVector = TmpVectors.Vector3[1];\n target.subtractToRef(this.position, tmpVector);\n tmpVector.y = 0;\n tmpVector.normalize();\n const yRotation = Math.atan2(tmpVector.x, tmpVector.z);\n this.rotationQuaternion.toEulerAnglesToRef(tmpVector);\n Quaternion.FromEulerAnglesToRef(tmpVector.x, yRotation, tmpVector.z, this.rotationQuaternion);\n }\n dispose() {\n super.dispose();\n this._lastXRViewerPose = undefined;\n }\n _updateFromXRSession() {\n const pose = this._xrSessionManager.currentFrame && this._xrSessionManager.currentFrame.getViewerPose(this._xrSessionManager.referenceSpace);\n this._lastXRViewerPose = pose || undefined;\n if (!pose) {\n this._setTrackingState(WebXRTrackingState.NOT_TRACKING);\n return;\n }\n // Set the tracking state. if it didn't change it is a no-op\n const trackingState = pose.emulatedPosition ? WebXRTrackingState.TRACKING_LOST : WebXRTrackingState.TRACKING;\n this._setTrackingState(trackingState);\n // check min/max Z and update if not the same as in cache\n if (this.minZ !== this._cache.minZ || this.maxZ !== this._cache.maxZ) {\n const xrRenderState = {\n // if maxZ is 0 it should be \"Infinity\", but it doesn't work with the WebXR API. Setting to a large number.\n depthFar: this.maxZ || 10000,\n depthNear: this.minZ,\n };\n this._xrSessionManager.updateRenderState(xrRenderState);\n this._cache.minZ = this.minZ;\n this._cache.maxZ = this.maxZ;\n }\n if (pose.transform) {\n const orientation = pose.transform.orientation;\n if (pose.transform.orientation.x === undefined) {\n // Babylon native polyfill can return an undefined orientation value\n // When not initialized\n return;\n }\n const pos = pose.transform.position;\n this._referencedPosition.set(pos.x, pos.y, pos.z);\n this._referenceQuaternion.set(orientation.x, orientation.y, orientation.z, orientation.w);\n if (!this._scene.useRightHandedSystem) {\n this._referencedPosition.z *= -1;\n this._referenceQuaternion.z *= -1;\n this._referenceQuaternion.w *= -1;\n }\n if (this._firstFrame) {\n this._firstFrame = false;\n // we have the XR reference, now use this to find the offset to get the camera to be\n // in the right position\n // set the height to correlate to the current height\n this.position.y += this._referencedPosition.y;\n // avoid using the head rotation on the first frame.\n this._referenceQuaternion.copyFromFloats(0, 0, 0, 1);\n }\n else {\n // update position and rotation as reference\n this.rotationQuaternion.copyFrom(this._referenceQuaternion);\n this.position.copyFrom(this._referencedPosition);\n }\n }\n // Update camera rigs\n if (this.rigCameras.length !== pose.views.length) {\n this._updateNumberOfRigCameras(pose.views.length);\n }\n pose.views.forEach((view, i) => {\n var _a;\n const currentRig = this.rigCameras[i];\n // update right and left, where applicable\n if (!currentRig.isLeftCamera && !currentRig.isRightCamera) {\n if (view.eye === \"right\") {\n currentRig._isRightCamera = true;\n }\n else if (view.eye === \"left\") {\n currentRig._isLeftCamera = true;\n }\n }\n // Update view/projection matrix\n const pos = view.transform.position;\n const orientation = view.transform.orientation;\n currentRig.parent = this.parent;\n currentRig.position.set(pos.x, pos.y, pos.z);\n currentRig.rotationQuaternion.set(orientation.x, orientation.y, orientation.z, orientation.w);\n if (!this._scene.useRightHandedSystem) {\n currentRig.position.z *= -1;\n currentRig.rotationQuaternion.z *= -1;\n currentRig.rotationQuaternion.w *= -1;\n }\n else {\n currentRig.rotationQuaternion.multiplyInPlace(this._rotate180);\n }\n Matrix.FromFloat32ArrayToRefScaled(view.projectionMatrix, 0, 1, currentRig._projectionMatrix);\n if (!this._scene.useRightHandedSystem) {\n currentRig._projectionMatrix.toggleProjectionMatrixHandInPlace();\n }\n // first camera?\n if (i === 0) {\n this._projectionMatrix.copyFrom(currentRig._projectionMatrix);\n }\n const renderTargetTexture = this._xrSessionManager.getRenderTargetTextureForView(view);\n this._renderingMultiview = ((_a = renderTargetTexture === null || renderTargetTexture === void 0 ? void 0 : renderTargetTexture._texture) === null || _a === void 0 ? void 0 : _a.isMultiview) || false;\n if (this._renderingMultiview) {\n // For multiview, the render target texture is the same per-view (just the slice index is different),\n // so we only need to set the output render target once for the rig parent.\n if (i == 0) {\n this._xrSessionManager.trySetViewportForView(this.viewport, view);\n this.outputRenderTarget = renderTargetTexture;\n }\n }\n else {\n // Update viewport\n this._xrSessionManager.trySetViewportForView(currentRig.viewport, view);\n // Set cameras to render to the session's render target\n currentRig.outputRenderTarget = renderTargetTexture || this._xrSessionManager.getRenderTargetTextureForView(view);\n }\n // Replicate parent rig camera behavior\n currentRig.layerMask = this.layerMask;\n });\n }\n _updateNumberOfRigCameras(viewCount = 1) {\n while (this.rigCameras.length < viewCount) {\n const newCamera = new TargetCamera(\"XR-RigCamera: \" + this.rigCameras.length, Vector3.Zero(), this.getScene());\n newCamera.minZ = 0.1;\n newCamera.rotationQuaternion = new Quaternion();\n newCamera.updateUpVectorFromRotation = true;\n newCamera.isRigCamera = true;\n newCamera.rigParent = this;\n // do not compute projection matrix, provided by XR\n newCamera.freezeProjectionMatrix();\n this.rigCameras.push(newCamera);\n }\n while (this.rigCameras.length > viewCount) {\n const removedCamera = this.rigCameras.pop();\n if (removedCamera) {\n removedCamera.dispose();\n }\n }\n }\n _updateReferenceSpace() {\n // were position & rotation updated OUTSIDE of the xr update loop\n if (!this.position.equals(this._referencedPosition) || !this.rotationQuaternion.equals(this._referenceQuaternion)) {\n const referencedMat = TmpVectors.Matrix[0];\n const poseMat = TmpVectors.Matrix[1];\n const transformMat = TmpVectors.Matrix[2];\n Matrix.ComposeToRef(WebXRCamera._ScaleReadOnly, this._referenceQuaternion, this._referencedPosition, referencedMat);\n Matrix.ComposeToRef(WebXRCamera._ScaleReadOnly, this.rotationQuaternion, this.position, poseMat);\n referencedMat.invert().multiplyToRef(poseMat, transformMat);\n transformMat.invert();\n if (!this._scene.useRightHandedSystem) {\n transformMat.toggleModelMatrixHandInPlace();\n }\n transformMat.decompose(undefined, this._referenceQuaternion, this._referencedPosition);\n const transform = new XRRigidTransform({\n x: this._referencedPosition.x,\n y: this._referencedPosition.y,\n z: this._referencedPosition.z,\n }, {\n x: this._referenceQuaternion.x,\n y: this._referenceQuaternion.y,\n z: this._referenceQuaternion.z,\n w: this._referenceQuaternion.w,\n });\n this._xrSessionManager.referenceSpace = this._xrSessionManager.referenceSpace.getOffsetReferenceSpace(transform);\n }\n }\n }\n WebXRCamera._ScaleReadOnly = Vector3.One();\n\n /**\n * A list of the currently available features without referencing them\n */\n class WebXRFeatureName {\n }\n /**\n * The name of the anchor system feature\n */\n WebXRFeatureName.ANCHOR_SYSTEM = \"xr-anchor-system\";\n /**\n * The name of the background remover feature\n */\n WebXRFeatureName.BACKGROUND_REMOVER = \"xr-background-remover\";\n /**\n * The name of the hit test feature\n */\n WebXRFeatureName.HIT_TEST = \"xr-hit-test\";\n /**\n * The name of the mesh detection feature\n */\n WebXRFeatureName.MESH_DETECTION = \"xr-mesh-detection\";\n /**\n * physics impostors for xr controllers feature\n */\n WebXRFeatureName.PHYSICS_CONTROLLERS = \"xr-physics-controller\";\n /**\n * The name of the plane detection feature\n */\n WebXRFeatureName.PLANE_DETECTION = \"xr-plane-detection\";\n /**\n * The name of the pointer selection feature\n */\n WebXRFeatureName.POINTER_SELECTION = \"xr-controller-pointer-selection\";\n /**\n * The name of the teleportation feature\n */\n WebXRFeatureName.TELEPORTATION = \"xr-controller-teleportation\";\n /**\n * The name of the feature points feature.\n */\n WebXRFeatureName.FEATURE_POINTS = \"xr-feature-points\";\n /**\n * The name of the hand tracking feature.\n */\n WebXRFeatureName.HAND_TRACKING = \"xr-hand-tracking\";\n /**\n * The name of the image tracking feature\n */\n WebXRFeatureName.IMAGE_TRACKING = \"xr-image-tracking\";\n /**\n * The name of the near interaction feature\n */\n WebXRFeatureName.NEAR_INTERACTION = \"xr-near-interaction\";\n /**\n * The name of the DOM overlay feature\n */\n WebXRFeatureName.DOM_OVERLAY = \"xr-dom-overlay\";\n /**\n * The name of the movement feature\n */\n WebXRFeatureName.MOVEMENT = \"xr-controller-movement\";\n /**\n * The name of the light estimation feature\n */\n WebXRFeatureName.LIGHT_ESTIMATION = \"xr-light-estimation\";\n /**\n * The name of the eye tracking feature\n */\n WebXRFeatureName.EYE_TRACKING = \"xr-eye-tracking\";\n /**\n * The name of the walking locomotion feature\n */\n WebXRFeatureName.WALKING_LOCOMOTION = \"xr-walking-locomotion\";\n /**\n * The name of the composition layers feature\n */\n WebXRFeatureName.LAYERS = \"xr-layers\";\n /**\n * The name of the depth sensing feature\n */\n WebXRFeatureName.DEPTH_SENSING = \"xr-depth-sensing\";\n /**\n * The name of the WebXR Space Warp feature\n */\n WebXRFeatureName.SPACE_WARP = \"xr-space-warp\";\n /**\n * The WebXR features manager is responsible of enabling or disabling features required for the current XR session.\n * It is mainly used in AR sessions.\n *\n * A feature can have a version that is defined by Babylon (and does not correspond with the webxr version).\n */\n class WebXRFeaturesManager {\n /**\n * constructs a new features manages.\n *\n * @param _xrSessionManager an instance of WebXRSessionManager\n */\n constructor(_xrSessionManager) {\n this._xrSessionManager = _xrSessionManager;\n this._features = {};\n // when session starts / initialized - attach\n this._xrSessionManager.onXRSessionInit.add(() => {\n this.getEnabledFeatures().forEach((featureName) => {\n const feature = this._features[featureName];\n if (feature.enabled && !feature.featureImplementation.attached && !feature.featureImplementation.disableAutoAttach) {\n this.attachFeature(featureName);\n }\n });\n });\n // when session ends - detach\n this._xrSessionManager.onXRSessionEnded.add(() => {\n this.getEnabledFeatures().forEach((featureName) => {\n const feature = this._features[featureName];\n if (feature.enabled && feature.featureImplementation.attached) {\n // detach, but don't disable!\n this.detachFeature(featureName);\n }\n });\n });\n }\n /**\n * Used to register a module. After calling this function a developer can use this feature in the scene.\n * Mainly used internally.\n *\n * @param featureName the name of the feature to register\n * @param constructorFunction the function used to construct the module\n * @param version the (babylon) version of the module\n * @param stable is that a stable version of this module\n */\n static AddWebXRFeature(featureName, constructorFunction, version = 1, stable = false) {\n this._AvailableFeatures[featureName] = this._AvailableFeatures[featureName] || { latest: version };\n if (version > this._AvailableFeatures[featureName].latest) {\n this._AvailableFeatures[featureName].latest = version;\n }\n if (stable) {\n this._AvailableFeatures[featureName].stable = version;\n }\n this._AvailableFeatures[featureName][version] = constructorFunction;\n }\n /**\n * Returns a constructor of a specific feature.\n *\n * @param featureName the name of the feature to construct\n * @param version the version of the feature to load\n * @param xrSessionManager the xrSessionManager. Used to construct the module\n * @param options optional options provided to the module.\n * @returns a function that, when called, will return a new instance of this feature\n */\n static ConstructFeature(featureName, version = 1, xrSessionManager, options) {\n const constructorFunction = this._AvailableFeatures[featureName][version];\n if (!constructorFunction) {\n // throw an error? return nothing?\n throw new Error(\"feature not found\");\n }\n return constructorFunction(xrSessionManager, options);\n }\n /**\n * Can be used to return the list of features currently registered\n *\n * @returns an Array of available features\n */\n static GetAvailableFeatures() {\n return Object.keys(this._AvailableFeatures);\n }\n /**\n * Gets the versions available for a specific feature\n * @param featureName the name of the feature\n * @returns an array with the available versions\n */\n static GetAvailableVersions(featureName) {\n return Object.keys(this._AvailableFeatures[featureName]);\n }\n /**\n * Return the latest unstable version of this feature\n * @param featureName the name of the feature to search\n * @returns the version number. if not found will return -1\n */\n static GetLatestVersionOfFeature(featureName) {\n return (this._AvailableFeatures[featureName] && this._AvailableFeatures[featureName].latest) || -1;\n }\n /**\n * Return the latest stable version of this feature\n * @param featureName the name of the feature to search\n * @returns the version number. if not found will return -1\n */\n static GetStableVersionOfFeature(featureName) {\n return (this._AvailableFeatures[featureName] && this._AvailableFeatures[featureName].stable) || -1;\n }\n /**\n * Attach a feature to the current session. Mainly used when session started to start the feature effect.\n * Can be used during a session to start a feature\n * @param featureName the name of feature to attach\n */\n attachFeature(featureName) {\n const feature = this._features[featureName];\n if (feature && feature.enabled && !feature.featureImplementation.attached) {\n feature.featureImplementation.attach();\n }\n }\n /**\n * Can be used inside a session or when the session ends to detach a specific feature\n * @param featureName the name of the feature to detach\n */\n detachFeature(featureName) {\n const feature = this._features[featureName];\n if (feature && feature.featureImplementation.attached) {\n feature.featureImplementation.detach();\n }\n }\n /**\n * Used to disable an already-enabled feature\n * The feature will be disposed and will be recreated once enabled.\n * @param featureName the feature to disable\n * @returns true if disable was successful\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n disableFeature(featureName) {\n const name = typeof featureName === \"string\" ? featureName : featureName.Name;\n const feature = this._features[name];\n if (feature && feature.enabled) {\n feature.enabled = false;\n this.detachFeature(name);\n feature.featureImplementation.dispose();\n delete this._features[name];\n return true;\n }\n return false;\n }\n /**\n * dispose this features manager\n */\n dispose() {\n this.getEnabledFeatures().forEach((feature) => {\n this.disableFeature(feature);\n });\n }\n /**\n * Enable a feature using its name and a version. This will enable it in the scene, and will be responsible to attach it when the session starts.\n * If used twice, the old version will be disposed and a new one will be constructed. This way you can re-enable with different configuration.\n *\n * @param featureName the name of the feature to load or the class of the feature\n * @param version optional version to load. if not provided the latest version will be enabled\n * @param moduleOptions options provided to the module. Ses the module documentation / constructor\n * @param attachIfPossible if set to true (default) the feature will be automatically attached, if it is currently possible\n * @param required is this feature required to the app. If set to true the session init will fail if the feature is not available.\n * @returns a new constructed feature or throws an error if feature not found or conflicts with another enabled feature.\n */\n enableFeature(\n // eslint-disable-next-line @typescript-eslint/naming-convention\n featureName, version = \"latest\", moduleOptions = {}, attachIfPossible = true, required = true) {\n const name = typeof featureName === \"string\" ? featureName : featureName.Name;\n let versionToLoad = 0;\n if (typeof version === \"string\") {\n if (!version) {\n throw new Error(`Error in provided version - ${name} (${version})`);\n }\n if (version === \"stable\") {\n versionToLoad = WebXRFeaturesManager.GetStableVersionOfFeature(name);\n }\n else if (version === \"latest\") {\n versionToLoad = WebXRFeaturesManager.GetLatestVersionOfFeature(name);\n }\n else {\n // try loading the number the string represents\n versionToLoad = +version;\n }\n if (versionToLoad === -1 || isNaN(versionToLoad)) {\n throw new Error(`feature not found - ${name} (${version})`);\n }\n }\n else {\n versionToLoad = version;\n }\n // check if there is a feature conflict\n const conflictingFeature = WebXRFeaturesManager._ConflictingFeatures[name];\n if (conflictingFeature !== undefined && this.getEnabledFeatures().indexOf(conflictingFeature) !== -1) {\n throw new Error(`Feature ${name} cannot be enabled while ${conflictingFeature} is enabled.`);\n }\n // check if already initialized\n const feature = this._features[name];\n const constructFunction = WebXRFeaturesManager.ConstructFeature(name, versionToLoad, this._xrSessionManager, moduleOptions);\n if (!constructFunction) {\n // report error?\n throw new Error(`feature not found - ${name}`);\n }\n /* If the feature is already enabled, detach and dispose it, and create a new one */\n if (feature) {\n this.disableFeature(name);\n }\n const constructed = constructFunction();\n if (constructed.dependsOn) {\n const dependentsFound = constructed.dependsOn.every((featureName) => !!this._features[featureName]);\n if (!dependentsFound) {\n throw new Error(`Dependant features missing. Make sure the following features are enabled - ${constructed.dependsOn.join(\", \")}`);\n }\n }\n if (constructed.isCompatible()) {\n this._features[name] = {\n featureImplementation: constructed,\n enabled: true,\n version: versionToLoad,\n required,\n };\n if (attachIfPossible) {\n // if session started already, request and enable\n if (this._xrSessionManager.session && !this._features[name].featureImplementation.attached) {\n // enable feature\n this.attachFeature(name);\n }\n }\n else {\n // disable auto-attach when session starts\n this._features[name].featureImplementation.disableAutoAttach = true;\n }\n return this._features[name].featureImplementation;\n }\n else {\n if (required) {\n throw new Error(\"required feature not compatible\");\n }\n else {\n Tools.Warn(`Feature ${name} not compatible with the current environment/browser and was not enabled.`);\n return constructed;\n }\n }\n }\n /**\n * get the implementation of an enabled feature.\n * @param featureName the name of the feature to load\n * @returns the feature class, if found\n */\n getEnabledFeature(featureName) {\n return this._features[featureName] && this._features[featureName].featureImplementation;\n }\n /**\n * Get the list of enabled features\n * @returns an array of enabled features\n */\n getEnabledFeatures() {\n return Object.keys(this._features);\n }\n /**\n * This function will extend the session creation configuration object with enabled features.\n * If, for example, the anchors feature is enabled, it will be automatically added to the optional or required features list,\n * according to the defined \"required\" variable, provided during enableFeature call\n * @param xrSessionInit the xr Session init object to extend\n *\n * @returns an extended XRSessionInit object\n */\n async _extendXRSessionInitObject(xrSessionInit) {\n const enabledFeatures = this.getEnabledFeatures();\n for (const featureName of enabledFeatures) {\n const feature = this._features[featureName];\n const nativeName = feature.featureImplementation.xrNativeFeatureName;\n if (nativeName) {\n if (feature.required) {\n xrSessionInit.requiredFeatures = xrSessionInit.requiredFeatures || [];\n if (xrSessionInit.requiredFeatures.indexOf(nativeName) === -1) {\n xrSessionInit.requiredFeatures.push(nativeName);\n }\n }\n else {\n xrSessionInit.optionalFeatures = xrSessionInit.optionalFeatures || [];\n if (xrSessionInit.optionalFeatures.indexOf(nativeName) === -1) {\n xrSessionInit.optionalFeatures.push(nativeName);\n }\n }\n }\n if (feature.featureImplementation.getXRSessionInitExtension) {\n const extended = await feature.featureImplementation.getXRSessionInitExtension();\n xrSessionInit = Object.assign(Object.assign({}, xrSessionInit), extended);\n }\n }\n return xrSessionInit;\n }\n }\n WebXRFeaturesManager._AvailableFeatures = {};\n /**\n * The key is the feature to check and the value is the feature that conflicts.\n */\n WebXRFeaturesManager._ConflictingFeatures = {\n [WebXRFeatureName.TELEPORTATION]: WebXRFeatureName.MOVEMENT,\n [WebXRFeatureName.MOVEMENT]: WebXRFeatureName.TELEPORTATION,\n };\n\n Node.AddNodeConstructor(\"TouchCamera\", (name, scene) => {\n return () => new TouchCamera(name, Vector3.Zero(), scene);\n });\n /**\n * This represents a FPS type of camera controlled by touch.\n * This is like a universal camera minus the Gamepad controls.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#universal-camera\n */\n class TouchCamera extends FreeCamera {\n /**\n * Defines the touch sensibility for rotation.\n * The higher the faster.\n */\n get touchAngularSensibility() {\n const touch = this.inputs.attached[\"touch\"];\n if (touch) {\n return touch.touchAngularSensibility;\n }\n return 0;\n }\n set touchAngularSensibility(value) {\n const touch = this.inputs.attached[\"touch\"];\n if (touch) {\n touch.touchAngularSensibility = value;\n }\n }\n /**\n * Defines the touch sensibility for move.\n * The higher the faster.\n */\n get touchMoveSensibility() {\n const touch = this.inputs.attached[\"touch\"];\n if (touch) {\n return touch.touchMoveSensibility;\n }\n return 0;\n }\n set touchMoveSensibility(value) {\n const touch = this.inputs.attached[\"touch\"];\n if (touch) {\n touch.touchMoveSensibility = value;\n }\n }\n /**\n * Instantiates a new touch camera.\n * This represents a FPS type of camera controlled by touch.\n * This is like a universal camera minus the Gamepad controls.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#universal-camera\n * @param name Define the name of the camera in the scene\n * @param position Define the start position of the camera in the scene\n * @param scene Define the scene the camera belongs to\n */\n constructor(name, position, scene) {\n super(name, position, scene);\n this.inputs.addTouch();\n this._setupInputs();\n }\n /**\n * Gets the current object class name.\n * @returns the class name\n */\n getClassName() {\n return \"TouchCamera\";\n }\n /** @internal */\n _setupInputs() {\n const touch = this.inputs.attached[\"touch\"];\n const mouse = this.inputs.attached[\"mouse\"];\n if (mouse) {\n mouse.touchEnabled = false;\n }\n else {\n touch.allowMouse = true;\n }\n }\n }\n\n Node.AddNodeConstructor(\"FreeCamera\", (name, scene) => {\n // Forcing to use the Universal camera\n return () => new UniversalCamera(name, Vector3.Zero(), scene);\n });\n /**\n * The Universal Camera is the one to choose for first person shooter type games, and works with all the keyboard, mouse, touch and gamepads. This replaces the earlier Free Camera,\n * which still works and will still be found in many Playgrounds.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#universal-camera\n */\n class UniversalCamera extends TouchCamera {\n /**\n * Defines the gamepad rotation sensibility.\n * This is the threshold from when rotation starts to be accounted for to prevent jittering.\n */\n get gamepadAngularSensibility() {\n const gamepad = this.inputs.attached[\"gamepad\"];\n if (gamepad) {\n return gamepad.gamepadAngularSensibility;\n }\n return 0;\n }\n set gamepadAngularSensibility(value) {\n const gamepad = this.inputs.attached[\"gamepad\"];\n if (gamepad) {\n gamepad.gamepadAngularSensibility = value;\n }\n }\n /**\n * Defines the gamepad move sensibility.\n * This is the threshold from when moving starts to be accounted for to prevent jittering.\n */\n get gamepadMoveSensibility() {\n const gamepad = this.inputs.attached[\"gamepad\"];\n if (gamepad) {\n return gamepad.gamepadMoveSensibility;\n }\n return 0;\n }\n set gamepadMoveSensibility(value) {\n const gamepad = this.inputs.attached[\"gamepad\"];\n if (gamepad) {\n gamepad.gamepadMoveSensibility = value;\n }\n }\n /**\n * The Universal Camera is the one to choose for first person shooter type games, and works with all the keyboard, mouse, touch and gamepads. This replaces the earlier Free Camera,\n * which still works and will still be found in many Playgrounds.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#universal-camera\n * @param name Define the name of the camera in the scene\n * @param position Define the start position of the camera in the scene\n * @param scene Define the scene the camera belongs to\n */\n constructor(name, position, scene) {\n super(name, position, scene);\n this.inputs.addGamepad();\n }\n /**\n * Gets the current object class name.\n * @returns the class name\n */\n getClassName() {\n return \"UniversalCamera\";\n }\n }\n Camera._CreateDefaultParsedCamera = (name, scene) => {\n return new UniversalCamera(name, Vector3.Zero(), scene);\n };\n\n /**\n * Base set of functionality needed to create an XR experience (WebXRSessionManager, Camera, StateManagement, etc.)\n * @see https://doc.babylonjs.com/features/featuresDeepDive/webXR/webXRExperienceHelpers\n */\n class WebXRExperienceHelper {\n /**\n * Creates a WebXRExperienceHelper\n * @param _scene The scene the helper should be created in\n */\n constructor(_scene) {\n this._scene = _scene;\n this._nonVRCamera = null;\n this._attachedToElement = false;\n this._spectatorCamera = null;\n this._originalSceneAutoClear = true;\n this._supported = false;\n this._spectatorMode = false;\n this._lastTimestamp = 0;\n /**\n * Observers registered here will be triggered after the camera's initial transformation is set\n * This can be used to set a different ground level or an extra rotation.\n *\n * Note that ground level is considered to be at 0. The height defined by the XR camera will be added\n * to the position set after this observable is done executing.\n */\n this.onInitialXRPoseSetObservable = new Observable$1();\n /**\n * Fires when the state of the experience helper has changed\n */\n this.onStateChangedObservable = new Observable$1();\n /**\n * The current state of the XR experience (eg. transitioning, in XR or not in XR)\n */\n this.state = WebXRState.NOT_IN_XR;\n this.sessionManager = new WebXRSessionManager(_scene);\n this.camera = new WebXRCamera(\"webxr\", _scene, this.sessionManager);\n this.featuresManager = new WebXRFeaturesManager(this.sessionManager);\n _scene.onDisposeObservable.addOnce(() => {\n this.dispose();\n });\n }\n /**\n * Creates the experience helper\n * @param scene the scene to attach the experience helper to\n * @returns a promise for the experience helper\n */\n static CreateAsync(scene) {\n const helper = new WebXRExperienceHelper(scene);\n return helper.sessionManager\n .initializeAsync()\n .then(() => {\n helper._supported = true;\n return helper;\n })\n .catch((e) => {\n helper._setState(WebXRState.NOT_IN_XR);\n helper.dispose();\n throw e;\n });\n }\n /**\n * Disposes of the experience helper\n */\n dispose() {\n var _a;\n this.exitXRAsync();\n this.camera.dispose();\n this.onStateChangedObservable.clear();\n this.onInitialXRPoseSetObservable.clear();\n this.sessionManager.dispose();\n (_a = this._spectatorCamera) === null || _a === void 0 ? void 0 : _a.dispose();\n if (this._nonVRCamera) {\n this._scene.activeCamera = this._nonVRCamera;\n }\n }\n /**\n * Enters XR mode (This must be done within a user interaction in most browsers eg. button click)\n * @param sessionMode options for the XR session\n * @param referenceSpaceType frame of reference of the XR session\n * @param renderTarget the output canvas that will be used to enter XR mode\n * @param sessionCreationOptions optional XRSessionInit object to init the session with\n * @returns promise that resolves after xr mode has entered\n */\n async enterXRAsync(sessionMode, referenceSpaceType, renderTarget = this.sessionManager.getWebXRRenderTarget(), sessionCreationOptions = {}) {\n var _a, _b, _c;\n if (!this._supported) {\n throw \"WebXR not supported in this browser or environment\";\n }\n this._setState(WebXRState.ENTERING_XR);\n if (referenceSpaceType !== \"viewer\" && referenceSpaceType !== \"local\") {\n sessionCreationOptions.optionalFeatures = sessionCreationOptions.optionalFeatures || [];\n sessionCreationOptions.optionalFeatures.push(referenceSpaceType);\n }\n sessionCreationOptions = await this.featuresManager._extendXRSessionInitObject(sessionCreationOptions);\n // we currently recommend \"unbounded\" space in AR (#7959)\n if (sessionMode === \"immersive-ar\" && referenceSpaceType !== \"unbounded\") {\n Logger.Warn(\"We recommend using 'unbounded' reference space type when using 'immersive-ar' session mode\");\n }\n // make sure that the session mode is supported\n try {\n await this.sessionManager.initializeSessionAsync(sessionMode, sessionCreationOptions);\n await this.sessionManager.setReferenceSpaceTypeAsync(referenceSpaceType);\n const baseLayer = await renderTarget.initializeXRLayerAsync(this.sessionManager.session);\n const xrRenderState = {\n // if maxZ is 0 it should be \"Infinity\", but it doesn't work with the WebXR API. Setting to a large number.\n depthFar: this.camera.maxZ || 10000,\n depthNear: this.camera.minZ,\n };\n // The layers feature will have already initialized the xr session's layers on session init.\n if (!this.featuresManager.getEnabledFeature(WebXRFeatureName.LAYERS)) {\n xrRenderState.baseLayer = baseLayer;\n }\n this.sessionManager.updateRenderState(xrRenderState);\n // run the render loop\n this.sessionManager.runXRRenderLoop();\n // Cache pre xr scene settings\n this._originalSceneAutoClear = this._scene.autoClear;\n this._nonVRCamera = this._scene.activeCamera;\n this._attachedToElement = !!((_b = (_a = this._nonVRCamera) === null || _a === void 0 ? void 0 : _a.inputs) === null || _b === void 0 ? void 0 : _b.attachedToElement);\n (_c = this._nonVRCamera) === null || _c === void 0 ? void 0 : _c.detachControl();\n this._scene.activeCamera = this.camera;\n // do not compensate when AR session is used\n if (sessionMode !== \"immersive-ar\") {\n this._nonXRToXRCamera();\n }\n else {\n // Kept here, TODO - check if needed\n this._scene.autoClear = false;\n this.camera.compensateOnFirstFrame = false;\n // reset the camera's position to the origin\n this.camera.position.set(0, 0, 0);\n this.camera.rotationQuaternion.set(0, 0, 0, 1);\n this.onInitialXRPoseSetObservable.notifyObservers(this.camera);\n }\n this.sessionManager.onXRSessionEnded.addOnce(() => {\n // when using the back button and not the exit button (default on mobile), the session is ending but the EXITING state was not set\n if (this.state !== WebXRState.EXITING_XR) {\n this._setState(WebXRState.EXITING_XR);\n }\n // Reset camera rigs output render target to ensure sessions render target is not drawn after it ends\n this.camera.rigCameras.forEach((c) => {\n c.outputRenderTarget = null;\n });\n // Restore scene settings\n this._scene.autoClear = this._originalSceneAutoClear;\n this._scene.activeCamera = this._nonVRCamera;\n if (this._attachedToElement && this._nonVRCamera) {\n this._nonVRCamera.attachControl(!!this._nonVRCamera.inputs.noPreventDefault);\n }\n if (sessionMode !== \"immersive-ar\" && this.camera.compensateOnFirstFrame) {\n if (this._nonVRCamera.setPosition) {\n this._nonVRCamera.setPosition(this.camera.position);\n }\n else {\n this._nonVRCamera.position.copyFrom(this.camera.position);\n }\n }\n this._setState(WebXRState.NOT_IN_XR);\n });\n // Wait until the first frame arrives before setting state to in xr\n this.sessionManager.onXRFrameObservable.addOnce(() => {\n this._setState(WebXRState.IN_XR);\n });\n return this.sessionManager;\n }\n catch (e) {\n console.log(e);\n console.log(e.message);\n this._setState(WebXRState.NOT_IN_XR);\n throw e;\n }\n }\n /**\n * Exits XR mode and returns the scene to its original state\n * @returns promise that resolves after xr mode has exited\n */\n exitXRAsync() {\n // only exit if state is IN_XR\n if (this.state !== WebXRState.IN_XR) {\n return Promise.resolve();\n }\n this._setState(WebXRState.EXITING_XR);\n return this.sessionManager.exitXRAsync();\n }\n /**\n * Enable spectator mode for desktop VR experiences.\n * When spectator mode is enabled a camera will be attached to the desktop canvas and will\n * display the first rig camera's view on the desktop canvas.\n * Please note that this will degrade performance, as it requires another camera render.\n * It is also not recommended to enable this in devices like the quest, as it brings no benefit there.\n * @param options giving WebXRSpectatorModeOption for specutator camera to setup when the spectator mode is enabled.\n */\n enableSpectatorMode(options) {\n if (!this._spectatorMode) {\n this._spectatorMode = true;\n this._switchSpectatorMode(options);\n }\n }\n /**\n * Disable spectator mode for desktop VR experiences.\n */\n disableSpecatatorMode() {\n if (this._spectatorMode) {\n this._spectatorMode = false;\n this._switchSpectatorMode();\n }\n }\n _switchSpectatorMode(options) {\n const fps = (options === null || options === void 0 ? void 0 : options.fps) ? options.fps : 1000.0;\n const refreshRate = (1.0 / fps) * 1000.0;\n const cameraIndex = (options === null || options === void 0 ? void 0 : options.preferredCameraIndex) ? options === null || options === void 0 ? void 0 : options.preferredCameraIndex : 0;\n const updateSpectatorCamera = () => {\n if (this._spectatorCamera) {\n const delta = this.sessionManager.currentTimestamp - this._lastTimestamp;\n if (delta >= refreshRate) {\n this._lastTimestamp = this.sessionManager.currentTimestamp;\n this._spectatorCamera.position.copyFrom(this.camera.rigCameras[cameraIndex].globalPosition);\n this._spectatorCamera.rotationQuaternion.copyFrom(this.camera.rigCameras[cameraIndex].absoluteRotation);\n }\n }\n };\n if (this._spectatorMode) {\n if (cameraIndex >= this.camera.rigCameras.length) {\n throw new Error(\"the preferred camera index is beyond the length of rig camera array.\");\n }\n const onStateChanged = () => {\n if (this.state === WebXRState.IN_XR) {\n this._spectatorCamera = new UniversalCamera(\"webxr-spectator\", Vector3.Zero(), this._scene);\n this._spectatorCamera.rotationQuaternion = new Quaternion();\n this._scene.activeCameras = [this.camera, this._spectatorCamera];\n this.sessionManager.onXRFrameObservable.add(updateSpectatorCamera);\n this._scene.onAfterRenderCameraObservable.add((camera) => {\n if (camera === this.camera) {\n // reset the dimensions object for correct resizing\n this._scene.getEngine().framebufferDimensionsObject = null;\n }\n });\n }\n else if (this.state === WebXRState.EXITING_XR) {\n this.sessionManager.onXRFrameObservable.removeCallback(updateSpectatorCamera);\n this._scene.activeCameras = null;\n }\n };\n this.onStateChangedObservable.add(onStateChanged);\n onStateChanged();\n }\n else {\n this.sessionManager.onXRFrameObservable.removeCallback(updateSpectatorCamera);\n this._scene.activeCameras = [this.camera];\n }\n }\n _nonXRToXRCamera() {\n this.camera.setTransformationFromNonVRCamera(this._nonVRCamera);\n this.onInitialXRPoseSetObservable.notifyObservers(this.camera);\n }\n _setState(val) {\n if (this.state === val) {\n return;\n }\n this.state = val;\n this.onStateChangedObservable.notifyObservers(this.state);\n }\n }\n\n /**\n * This class represents a single component (for example button or thumbstick) of a motion controller\n */\n class WebXRControllerComponent {\n /**\n * Creates a new component for a motion controller.\n * It is created by the motion controller itself\n *\n * @param id the id of this component\n * @param type the type of the component\n * @param _buttonIndex index in the buttons array of the gamepad\n * @param _axesIndices indices of the values in the axes array of the gamepad\n */\n constructor(\n /**\n * the id of this component\n */\n id, \n /**\n * the type of the component\n */\n type, _buttonIndex = -1, _axesIndices = []) {\n this.id = id;\n this.type = type;\n this._buttonIndex = _buttonIndex;\n this._axesIndices = _axesIndices;\n this._axes = {\n x: 0,\n y: 0,\n };\n this._changes = {};\n this._currentValue = 0;\n this._hasChanges = false;\n this._pressed = false;\n this._touched = false;\n /**\n * If axes are available for this component (like a touchpad or thumbstick) the observers will be notified when\n * the axes data changes\n */\n this.onAxisValueChangedObservable = new Observable$1();\n /**\n * Observers registered here will be triggered when the state of a button changes\n * State change is either pressed / touched / value\n */\n this.onButtonStateChangedObservable = new Observable$1();\n }\n /**\n * The current axes data. If this component has no axes it will still return an object { x: 0, y: 0 }\n */\n get axes() {\n return this._axes;\n }\n /**\n * Get the changes. Elements will be populated only if they changed with their previous and current value\n */\n get changes() {\n return this._changes;\n }\n /**\n * Return whether or not the component changed the last frame\n */\n get hasChanges() {\n return this._hasChanges;\n }\n /**\n * is the button currently pressed\n */\n get pressed() {\n return this._pressed;\n }\n /**\n * is the button currently touched\n */\n get touched() {\n return this._touched;\n }\n /**\n * Get the current value of this component\n */\n get value() {\n return this._currentValue;\n }\n /**\n * Dispose this component\n */\n dispose() {\n this.onAxisValueChangedObservable.clear();\n this.onButtonStateChangedObservable.clear();\n }\n /**\n * Are there axes correlating to this component\n * @returns true is axes data is available\n */\n isAxes() {\n return this._axesIndices.length !== 0;\n }\n /**\n * Is this component a button (hence - pressable)\n * @returns true if can be pressed\n */\n isButton() {\n return this._buttonIndex !== -1;\n }\n /**\n * update this component using the gamepad object it is in. Called on every frame\n * @param nativeController the native gamepad controller object\n */\n update(nativeController) {\n let buttonUpdated = false;\n let axesUpdate = false;\n this._hasChanges = false;\n this._changes = {};\n if (this.isButton()) {\n const button = nativeController.buttons[this._buttonIndex];\n // defensive, in case a profile was forced\n if (!button) {\n return;\n }\n if (this._currentValue !== button.value) {\n this.changes.value = {\n current: button.value,\n previous: this._currentValue,\n };\n buttonUpdated = true;\n this._currentValue = button.value;\n }\n if (this._touched !== button.touched) {\n this.changes.touched = {\n current: button.touched,\n previous: this._touched,\n };\n buttonUpdated = true;\n this._touched = button.touched;\n }\n if (this._pressed !== button.pressed) {\n this.changes.pressed = {\n current: button.pressed,\n previous: this._pressed,\n };\n buttonUpdated = true;\n this._pressed = button.pressed;\n }\n }\n if (this.isAxes()) {\n if (this._axes.x !== nativeController.axes[this._axesIndices[0]]) {\n this.changes.axes = {\n current: {\n x: nativeController.axes[this._axesIndices[0]],\n y: this._axes.y,\n },\n previous: {\n x: this._axes.x,\n y: this._axes.y,\n },\n };\n this._axes.x = nativeController.axes[this._axesIndices[0]];\n axesUpdate = true;\n }\n if (this._axes.y !== nativeController.axes[this._axesIndices[1]]) {\n if (this.changes.axes) {\n this.changes.axes.current.y = nativeController.axes[this._axesIndices[1]];\n }\n else {\n this.changes.axes = {\n current: {\n x: this._axes.x,\n y: nativeController.axes[this._axesIndices[1]],\n },\n previous: {\n x: this._axes.x,\n y: this._axes.y,\n },\n };\n }\n this._axes.y = nativeController.axes[this._axesIndices[1]];\n axesUpdate = true;\n }\n }\n if (buttonUpdated) {\n this._hasChanges = true;\n this.onButtonStateChangedObservable.notifyObservers(this);\n }\n if (axesUpdate) {\n this._hasChanges = true;\n this.onAxisValueChangedObservable.notifyObservers(this._axes);\n }\n }\n }\n /**\n * button component type\n */\n WebXRControllerComponent.BUTTON_TYPE = \"button\";\n /**\n * squeeze component type\n */\n WebXRControllerComponent.SQUEEZE_TYPE = \"squeeze\";\n /**\n * Thumbstick component type\n */\n WebXRControllerComponent.THUMBSTICK_TYPE = \"thumbstick\";\n /**\n * Touchpad component type\n */\n WebXRControllerComponent.TOUCHPAD_TYPE = \"touchpad\";\n /**\n * trigger component type\n */\n WebXRControllerComponent.TRIGGER_TYPE = \"trigger\";\n\n /**\n * Mode that determines how to handle old animation groups before loading new ones.\n */\n var SceneLoaderAnimationGroupLoadingMode;\n (function (SceneLoaderAnimationGroupLoadingMode) {\n /**\n * Reset all old animations to initial state then dispose them.\n */\n SceneLoaderAnimationGroupLoadingMode[SceneLoaderAnimationGroupLoadingMode[\"Clean\"] = 0] = \"Clean\";\n /**\n * Stop all old animations.\n */\n SceneLoaderAnimationGroupLoadingMode[SceneLoaderAnimationGroupLoadingMode[\"Stop\"] = 1] = \"Stop\";\n /**\n * Restart old animations from first frame.\n */\n SceneLoaderAnimationGroupLoadingMode[SceneLoaderAnimationGroupLoadingMode[\"Sync\"] = 2] = \"Sync\";\n /**\n * Old animations remains untouched.\n */\n SceneLoaderAnimationGroupLoadingMode[SceneLoaderAnimationGroupLoadingMode[\"NoSync\"] = 3] = \"NoSync\";\n })(SceneLoaderAnimationGroupLoadingMode || (SceneLoaderAnimationGroupLoadingMode = {}));\n /**\n * Class used to load scene from various file formats using registered plugins\n * @see https://doc.babylonjs.com/features/featuresDeepDive/importers/loadingFileTypes\n */\n class SceneLoader {\n /**\n * Gets or sets a boolean indicating if entire scene must be loaded even if scene contains incremental data\n */\n static get ForceFullSceneLoadingForIncremental() {\n return SceneLoaderFlags.ForceFullSceneLoadingForIncremental;\n }\n static set ForceFullSceneLoadingForIncremental(value) {\n SceneLoaderFlags.ForceFullSceneLoadingForIncremental = value;\n }\n /**\n * Gets or sets a boolean indicating if loading screen must be displayed while loading a scene\n */\n static get ShowLoadingScreen() {\n return SceneLoaderFlags.ShowLoadingScreen;\n }\n static set ShowLoadingScreen(value) {\n SceneLoaderFlags.ShowLoadingScreen = value;\n }\n /**\n * Defines the current logging level (while loading the scene)\n * @ignorenaming\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n static get loggingLevel() {\n return SceneLoaderFlags.loggingLevel;\n }\n // eslint-disable-next-line @typescript-eslint/naming-convention\n static set loggingLevel(value) {\n SceneLoaderFlags.loggingLevel = value;\n }\n /**\n * Gets or set a boolean indicating if matrix weights must be cleaned upon loading\n */\n static get CleanBoneMatrixWeights() {\n return SceneLoaderFlags.CleanBoneMatrixWeights;\n }\n static set CleanBoneMatrixWeights(value) {\n SceneLoaderFlags.CleanBoneMatrixWeights = value;\n }\n /**\n * Gets the default plugin (used to load Babylon files)\n * @returns the .babylon plugin\n */\n static GetDefaultPlugin() {\n return SceneLoader._RegisteredPlugins[\".babylon\"];\n }\n static _GetPluginForExtension(extension) {\n const registeredPlugin = SceneLoader._RegisteredPlugins[extension];\n if (registeredPlugin) {\n return registeredPlugin;\n }\n Logger.Warn(\"Unable to find a plugin to load \" +\n extension +\n \" files. Trying to use .babylon default plugin. To load from a specific filetype (eg. gltf) see: https://doc.babylonjs.com/features/featuresDeepDive/importers/loadingFileTypes\");\n return SceneLoader.GetDefaultPlugin();\n }\n static _GetPluginForDirectLoad(data) {\n for (const extension in SceneLoader._RegisteredPlugins) {\n const plugin = SceneLoader._RegisteredPlugins[extension].plugin;\n if (plugin.canDirectLoad && plugin.canDirectLoad(data)) {\n return SceneLoader._RegisteredPlugins[extension];\n }\n }\n return SceneLoader.GetDefaultPlugin();\n }\n static _GetPluginForFilename(sceneFilename) {\n const queryStringPosition = sceneFilename.indexOf(\"?\");\n if (queryStringPosition !== -1) {\n sceneFilename = sceneFilename.substring(0, queryStringPosition);\n }\n const dotPosition = sceneFilename.lastIndexOf(\".\");\n const extension = sceneFilename.substring(dotPosition, sceneFilename.length).toLowerCase();\n return SceneLoader._GetPluginForExtension(extension);\n }\n static _GetDirectLoad(sceneFilename) {\n if (sceneFilename.substr(0, 5) === \"data:\") {\n return sceneFilename.substr(5);\n }\n return null;\n }\n static _FormatErrorMessage(fileInfo, message, exception) {\n const fromLoad = fileInfo.rawData ? \"binary data\" : fileInfo.url;\n let errorMessage = \"Unable to load from \" + fromLoad;\n if (message) {\n errorMessage += `: ${message}`;\n }\n else if (exception) {\n errorMessage += `: ${exception}`;\n }\n return errorMessage;\n }\n static _LoadData(fileInfo, scene, onSuccess, onProgress, onError, onDispose, pluginExtension, name) {\n const directLoad = SceneLoader._GetDirectLoad(fileInfo.url);\n if (fileInfo.rawData && !pluginExtension) {\n throw \"When using ArrayBufferView to load data the file extension must be provided.\";\n }\n const registeredPlugin = pluginExtension\n ? SceneLoader._GetPluginForExtension(pluginExtension)\n : directLoad\n ? SceneLoader._GetPluginForDirectLoad(fileInfo.url)\n : SceneLoader._GetPluginForFilename(fileInfo.url);\n if (fileInfo.rawData && !registeredPlugin.isBinary) {\n throw \"Loading from ArrayBufferView can not be used with plugins that don't support binary loading.\";\n }\n let plugin;\n if (registeredPlugin.plugin.createPlugin !== undefined) {\n plugin = registeredPlugin.plugin.createPlugin();\n }\n else {\n plugin = registeredPlugin.plugin;\n }\n if (!plugin) {\n throw \"The loader plugin corresponding to the file type you are trying to load has not been found. If using es6, please import the plugin you wish to use before.\";\n }\n SceneLoader.OnPluginActivatedObservable.notifyObservers(plugin);\n // Check if we have a direct load url. If the plugin is registered to handle\n // it or it's not a base64 data url, then pass it through the direct load path.\n if (directLoad && ((plugin.canDirectLoad && plugin.canDirectLoad(fileInfo.url)) || !IsBase64DataUrl(fileInfo.url))) {\n if (plugin.directLoad) {\n const result = plugin.directLoad(scene, directLoad);\n if (result.then) {\n result\n .then((data) => {\n onSuccess(plugin, data);\n })\n .catch((error) => {\n onError(\"Error in directLoad of _loadData: \" + error, error);\n });\n }\n else {\n onSuccess(plugin, result);\n }\n }\n else {\n onSuccess(plugin, directLoad);\n }\n return plugin;\n }\n const useArrayBuffer = registeredPlugin.isBinary;\n const dataCallback = (data, responseURL) => {\n if (scene.isDisposed) {\n onError(\"Scene has been disposed\");\n return;\n }\n onSuccess(plugin, data, responseURL);\n };\n let request = null;\n let pluginDisposed = false;\n const onDisposeObservable = plugin.onDisposeObservable;\n if (onDisposeObservable) {\n onDisposeObservable.add(() => {\n pluginDisposed = true;\n if (request) {\n request.abort();\n request = null;\n }\n onDispose();\n });\n }\n const manifestChecked = () => {\n if (pluginDisposed) {\n return;\n }\n const errorCallback = (request, exception) => {\n onError(request === null || request === void 0 ? void 0 : request.statusText, exception);\n };\n if (!plugin.loadFile && fileInfo.rawData) {\n throw \"Plugin does not support loading ArrayBufferView.\";\n }\n request = plugin.loadFile\n ? plugin.loadFile(scene, fileInfo.rawData || fileInfo.file || fileInfo.url, fileInfo.rootUrl, dataCallback, onProgress, useArrayBuffer, errorCallback, name)\n : scene._loadFile(fileInfo.file || fileInfo.url, dataCallback, onProgress, true, useArrayBuffer, errorCallback);\n };\n const engine = scene.getEngine();\n let canUseOfflineSupport = engine.enableOfflineSupport;\n if (canUseOfflineSupport) {\n // Also check for exceptions\n let exceptionFound = false;\n for (const regex of scene.disableOfflineSupportExceptionRules) {\n if (regex.test(fileInfo.url)) {\n exceptionFound = true;\n break;\n }\n }\n canUseOfflineSupport = !exceptionFound;\n }\n if (canUseOfflineSupport && Engine.OfflineProviderFactory) {\n // Checking if a manifest file has been set for this scene and if offline mode has been requested\n scene.offlineProvider = Engine.OfflineProviderFactory(fileInfo.url, manifestChecked, engine.disableManifestCheck);\n }\n else {\n manifestChecked();\n }\n return plugin;\n }\n static _GetFileInfo(rootUrl, sceneFilename) {\n let url;\n let name;\n let file = null;\n let rawData = null;\n if (!sceneFilename) {\n url = rootUrl;\n name = Tools.GetFilename(rootUrl);\n rootUrl = Tools.GetFolderPath(rootUrl);\n }\n else if (sceneFilename.name) {\n const sceneFile = sceneFilename;\n url = `file:${sceneFile.name}`;\n name = sceneFile.name;\n file = sceneFile;\n }\n else if (ArrayBuffer.isView(sceneFilename)) {\n url = \"\";\n name = \"arrayBuffer\";\n rawData = sceneFilename;\n }\n else if (typeof sceneFilename === \"string\" && sceneFilename.startsWith(\"data:\")) {\n url = sceneFilename;\n name = \"\";\n }\n else {\n const filename = sceneFilename;\n if (filename.substr(0, 1) === \"/\") {\n Tools.Error(\"Wrong sceneFilename parameter\");\n return null;\n }\n url = rootUrl + filename;\n name = filename;\n }\n return {\n url: url,\n rootUrl: rootUrl,\n name: name,\n file: file,\n rawData,\n };\n }\n // Public functions\n /**\n * Gets a plugin that can load the given extension\n * @param extension defines the extension to load\n * @returns a plugin or null if none works\n */\n static GetPluginForExtension(extension) {\n return SceneLoader._GetPluginForExtension(extension).plugin;\n }\n /**\n * Gets a boolean indicating that the given extension can be loaded\n * @param extension defines the extension to load\n * @returns true if the extension is supported\n */\n static IsPluginForExtensionAvailable(extension) {\n return !!SceneLoader._RegisteredPlugins[extension];\n }\n /**\n * Adds a new plugin to the list of registered plugins\n * @param plugin defines the plugin to add\n */\n static RegisterPlugin(plugin) {\n if (typeof plugin.extensions === \"string\") {\n const extension = plugin.extensions;\n SceneLoader._RegisteredPlugins[extension.toLowerCase()] = {\n plugin: plugin,\n isBinary: false,\n };\n }\n else {\n const extensions = plugin.extensions;\n Object.keys(extensions).forEach((extension) => {\n SceneLoader._RegisteredPlugins[extension.toLowerCase()] = {\n plugin: plugin,\n isBinary: extensions[extension].isBinary,\n };\n });\n }\n }\n /**\n * Import meshes into a scene\n * @param meshNames an array of mesh names, a single mesh name, or empty string for all meshes that filter what meshes are imported\n * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb)\n * @param sceneFilename a string that defines the name of the scene file or starts with \"data:\" following by the stringified version of the scene or a File object (default: empty string)\n * @param scene the instance of BABYLON.Scene to append to\n * @param onSuccess a callback with a list of imported meshes, particleSystems, skeletons, and animationGroups when import succeeds\n * @param onProgress a callback with a progress event for each file being loaded\n * @param onError a callback with the scene, a message, and possibly an exception when import fails\n * @param pluginExtension the extension used to determine the plugin\n * @returns The loaded plugin\n */\n static ImportMesh(meshNames, rootUrl, sceneFilename = \"\", scene = EngineStore.LastCreatedScene, onSuccess = null, onProgress = null, onError = null, pluginExtension = null, name = \"\") {\n if (!scene) {\n Logger.Error(\"No scene available to import mesh to\");\n return null;\n }\n const fileInfo = SceneLoader._GetFileInfo(rootUrl, sceneFilename);\n if (!fileInfo) {\n return null;\n }\n const loadingToken = {};\n scene.addPendingData(loadingToken);\n const disposeHandler = () => {\n scene.removePendingData(loadingToken);\n };\n const errorHandler = (message, exception) => {\n const errorMessage = SceneLoader._FormatErrorMessage(fileInfo, message, exception);\n if (onError) {\n onError(scene, errorMessage, new RuntimeError(errorMessage, ErrorCodes.SceneLoaderError, exception));\n }\n else {\n Logger.Error(errorMessage);\n // should the exception be thrown?\n }\n disposeHandler();\n };\n const progressHandler = onProgress\n ? (event) => {\n try {\n onProgress(event);\n }\n catch (e) {\n errorHandler(\"Error in onProgress callback: \" + e, e);\n }\n }\n : undefined;\n const successHandler = (meshes, particleSystems, skeletons, animationGroups, transformNodes, geometries, lights) => {\n scene.importedMeshesFiles.push(fileInfo.url);\n if (onSuccess) {\n try {\n onSuccess(meshes, particleSystems, skeletons, animationGroups, transformNodes, geometries, lights);\n }\n catch (e) {\n errorHandler(\"Error in onSuccess callback: \" + e, e);\n }\n }\n scene.removePendingData(loadingToken);\n };\n return SceneLoader._LoadData(fileInfo, scene, (plugin, data, responseURL) => {\n if (plugin.rewriteRootURL) {\n fileInfo.rootUrl = plugin.rewriteRootURL(fileInfo.rootUrl, responseURL);\n }\n if (plugin.importMesh) {\n const syncedPlugin = plugin;\n const meshes = new Array();\n const particleSystems = new Array();\n const skeletons = new Array();\n if (!syncedPlugin.importMesh(meshNames, scene, data, fileInfo.rootUrl, meshes, particleSystems, skeletons, errorHandler)) {\n return;\n }\n scene.loadingPluginName = plugin.name;\n successHandler(meshes, particleSystems, skeletons, [], [], [], []);\n }\n else {\n const asyncedPlugin = plugin;\n asyncedPlugin\n .importMeshAsync(meshNames, scene, data, fileInfo.rootUrl, progressHandler, fileInfo.name)\n .then((result) => {\n scene.loadingPluginName = plugin.name;\n successHandler(result.meshes, result.particleSystems, result.skeletons, result.animationGroups, result.transformNodes, result.geometries, result.lights);\n })\n .catch((error) => {\n errorHandler(error.message, error);\n });\n }\n }, progressHandler, errorHandler, disposeHandler, pluginExtension, name);\n }\n /**\n * Import meshes into a scene\n * @param meshNames an array of mesh names, a single mesh name, or empty string for all meshes that filter what meshes are imported\n * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb)\n * @param sceneFilename a string that defines the name of the scene file or starts with \"data:\" following by the stringified version of the scene or a File object (default: empty string)\n * @param scene the instance of BABYLON.Scene to append to\n * @param onProgress a callback with a progress event for each file being loaded\n * @param pluginExtension the extension used to determine the plugin\n * @returns The loaded list of imported meshes, particle systems, skeletons, and animation groups\n */\n static ImportMeshAsync(meshNames, rootUrl, sceneFilename = \"\", scene = EngineStore.LastCreatedScene, onProgress = null, pluginExtension = null, name = \"\") {\n return new Promise((resolve, reject) => {\n SceneLoader.ImportMesh(meshNames, rootUrl, sceneFilename, scene, (meshes, particleSystems, skeletons, animationGroups, transformNodes, geometries, lights) => {\n resolve({\n meshes: meshes,\n particleSystems: particleSystems,\n skeletons: skeletons,\n animationGroups: animationGroups,\n transformNodes: transformNodes,\n geometries: geometries,\n lights: lights,\n });\n }, onProgress, (scene, message, exception) => {\n reject(exception || new Error(message));\n }, pluginExtension, name);\n });\n }\n /**\n * Load a scene\n * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb)\n * @param sceneFilename a string that defines the name of the scene file or starts with \"data:\" following by the stringified version of the scene or a File object (default: empty string)\n * @param engine is the instance of BABYLON.Engine to use to create the scene\n * @param onSuccess a callback with the scene when import succeeds\n * @param onProgress a callback with a progress event for each file being loaded\n * @param onError a callback with the scene, a message, and possibly an exception when import fails\n * @param pluginExtension the extension used to determine the plugin\n * @returns The loaded plugin\n */\n static Load(rootUrl, sceneFilename = \"\", engine = EngineStore.LastCreatedEngine, onSuccess = null, onProgress = null, onError = null, pluginExtension = null, name = \"\") {\n if (!engine) {\n Tools.Error(\"No engine available\");\n return null;\n }\n return SceneLoader.Append(rootUrl, sceneFilename, new Scene(engine), onSuccess, onProgress, onError, pluginExtension, name);\n }\n /**\n * Load a scene\n * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb)\n * @param sceneFilename a string that defines the name of the scene file or starts with \"data:\" following by the stringified version of the scene or a File object (default: empty string)\n * @param engine is the instance of BABYLON.Engine to use to create the scene\n * @param onProgress a callback with a progress event for each file being loaded\n * @param pluginExtension the extension used to determine the plugin\n * @returns The loaded scene\n */\n static LoadAsync(rootUrl, sceneFilename = \"\", engine = EngineStore.LastCreatedEngine, onProgress = null, pluginExtension = null, name = \"\") {\n return new Promise((resolve, reject) => {\n SceneLoader.Load(rootUrl, sceneFilename, engine, (scene) => {\n resolve(scene);\n }, onProgress, (scene, message, exception) => {\n reject(exception || new Error(message));\n }, pluginExtension, name);\n });\n }\n /**\n * Append a scene\n * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb)\n * @param sceneFilename a string that defines the name of the scene file or starts with \"data:\" following by the stringified version of the scene or a File object (default: empty string)\n * @param scene is the instance of BABYLON.Scene to append to\n * @param onSuccess a callback with the scene when import succeeds\n * @param onProgress a callback with a progress event for each file being loaded\n * @param onError a callback with the scene, a message, and possibly an exception when import fails\n * @param pluginExtension the extension used to determine the plugin\n * @returns The loaded plugin\n */\n static Append(rootUrl, sceneFilename = \"\", scene = EngineStore.LastCreatedScene, onSuccess = null, onProgress = null, onError = null, pluginExtension = null, name = \"\") {\n if (!scene) {\n Logger.Error(\"No scene available to append to\");\n return null;\n }\n const fileInfo = SceneLoader._GetFileInfo(rootUrl, sceneFilename);\n if (!fileInfo) {\n return null;\n }\n const loadingToken = {};\n scene.addPendingData(loadingToken);\n const disposeHandler = () => {\n scene.removePendingData(loadingToken);\n };\n if (SceneLoader.ShowLoadingScreen && !this._ShowingLoadingScreen) {\n this._ShowingLoadingScreen = true;\n scene.getEngine().displayLoadingUI();\n scene.executeWhenReady(() => {\n scene.getEngine().hideLoadingUI();\n this._ShowingLoadingScreen = false;\n });\n }\n const errorHandler = (message, exception) => {\n const errorMessage = SceneLoader._FormatErrorMessage(fileInfo, message, exception);\n if (onError) {\n onError(scene, errorMessage, new RuntimeError(errorMessage, ErrorCodes.SceneLoaderError, exception));\n }\n else {\n Logger.Error(errorMessage);\n // should the exception be thrown?\n }\n disposeHandler();\n };\n const progressHandler = onProgress\n ? (event) => {\n try {\n onProgress(event);\n }\n catch (e) {\n errorHandler(\"Error in onProgress callback\", e);\n }\n }\n : undefined;\n const successHandler = () => {\n if (onSuccess) {\n try {\n onSuccess(scene);\n }\n catch (e) {\n errorHandler(\"Error in onSuccess callback\", e);\n }\n }\n scene.removePendingData(loadingToken);\n };\n return SceneLoader._LoadData(fileInfo, scene, (plugin, data) => {\n if (plugin.load) {\n const syncedPlugin = plugin;\n if (!syncedPlugin.load(scene, data, fileInfo.rootUrl, errorHandler)) {\n return;\n }\n scene.loadingPluginName = plugin.name;\n successHandler();\n }\n else {\n const asyncedPlugin = plugin;\n asyncedPlugin\n .loadAsync(scene, data, fileInfo.rootUrl, progressHandler, fileInfo.name)\n .then(() => {\n scene.loadingPluginName = plugin.name;\n successHandler();\n })\n .catch((error) => {\n errorHandler(error.message, error);\n });\n }\n }, progressHandler, errorHandler, disposeHandler, pluginExtension, name);\n }\n /**\n * Append a scene\n * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb)\n * @param sceneFilename a string that defines the name of the scene file or starts with \"data:\" following by the stringified version of the scene or a File object (default: empty string)\n * @param scene is the instance of BABYLON.Scene to append to\n * @param onProgress a callback with a progress event for each file being loaded\n * @param pluginExtension the extension used to determine the plugin\n * @returns The given scene\n */\n static AppendAsync(rootUrl, sceneFilename = \"\", scene = EngineStore.LastCreatedScene, onProgress = null, pluginExtension = null, name = \"\") {\n return new Promise((resolve, reject) => {\n SceneLoader.Append(rootUrl, sceneFilename, scene, (scene) => {\n resolve(scene);\n }, onProgress, (scene, message, exception) => {\n reject(exception || new Error(message));\n }, pluginExtension, name);\n });\n }\n /**\n * Load a scene into an asset container\n * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb)\n * @param sceneFilename a string that defines the name of the scene file or starts with \"data:\" following by the stringified version of the scene or a File object (default: empty string)\n * @param scene is the instance of BABYLON.Scene to append to (default: last created scene)\n * @param onSuccess a callback with the scene when import succeeds\n * @param onProgress a callback with a progress event for each file being loaded\n * @param onError a callback with the scene, a message, and possibly an exception when import fails\n * @param pluginExtension the extension used to determine the plugin\n * @returns The loaded plugin\n */\n static LoadAssetContainer(rootUrl, sceneFilename = \"\", scene = EngineStore.LastCreatedScene, onSuccess = null, onProgress = null, onError = null, pluginExtension = null, name = \"\") {\n if (!scene) {\n Logger.Error(\"No scene available to load asset container to\");\n return null;\n }\n const fileInfo = SceneLoader._GetFileInfo(rootUrl, sceneFilename);\n if (!fileInfo) {\n return null;\n }\n const loadingToken = {};\n scene.addPendingData(loadingToken);\n const disposeHandler = () => {\n scene.removePendingData(loadingToken);\n };\n const errorHandler = (message, exception) => {\n const errorMessage = SceneLoader._FormatErrorMessage(fileInfo, message, exception);\n if (onError) {\n onError(scene, errorMessage, new RuntimeError(errorMessage, ErrorCodes.SceneLoaderError, exception));\n }\n else {\n Logger.Error(errorMessage);\n // should the exception be thrown?\n }\n disposeHandler();\n };\n const progressHandler = onProgress\n ? (event) => {\n try {\n onProgress(event);\n }\n catch (e) {\n errorHandler(\"Error in onProgress callback\", e);\n }\n }\n : undefined;\n const successHandler = (assets) => {\n if (onSuccess) {\n try {\n onSuccess(assets);\n }\n catch (e) {\n errorHandler(\"Error in onSuccess callback\", e);\n }\n }\n scene.removePendingData(loadingToken);\n };\n return SceneLoader._LoadData(fileInfo, scene, (plugin, data) => {\n if (plugin.loadAssetContainer) {\n const syncedPlugin = plugin;\n const assetContainer = syncedPlugin.loadAssetContainer(scene, data, fileInfo.rootUrl, errorHandler);\n if (!assetContainer) {\n return;\n }\n assetContainer.populateRootNodes();\n scene.loadingPluginName = plugin.name;\n successHandler(assetContainer);\n }\n else if (plugin.loadAssetContainerAsync) {\n const asyncedPlugin = plugin;\n asyncedPlugin\n .loadAssetContainerAsync(scene, data, fileInfo.rootUrl, progressHandler, fileInfo.name)\n .then((assetContainer) => {\n assetContainer.populateRootNodes();\n scene.loadingPluginName = plugin.name;\n successHandler(assetContainer);\n })\n .catch((error) => {\n errorHandler(error.message, error);\n });\n }\n else {\n errorHandler(\"LoadAssetContainer is not supported by this plugin. Plugin did not provide a loadAssetContainer or loadAssetContainerAsync method.\");\n }\n }, progressHandler, errorHandler, disposeHandler, pluginExtension, name);\n }\n /**\n * Load a scene into an asset container\n * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb)\n * @param sceneFilename a string that defines the name of the scene file or starts with \"data:\" following by the stringified version of the scene (default: empty string)\n * @param scene is the instance of Scene to append to\n * @param onProgress a callback with a progress event for each file being loaded\n * @param pluginExtension the extension used to determine the plugin\n * @returns The loaded asset container\n */\n static LoadAssetContainerAsync(rootUrl, sceneFilename = \"\", scene = EngineStore.LastCreatedScene, onProgress = null, pluginExtension = null) {\n return new Promise((resolve, reject) => {\n SceneLoader.LoadAssetContainer(rootUrl, sceneFilename, scene, (assetContainer) => {\n resolve(assetContainer);\n }, onProgress, (scene, message, exception) => {\n reject(exception || new Error(message));\n }, pluginExtension);\n });\n }\n /**\n * Import animations from a file into a scene\n * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb)\n * @param sceneFilename a string that defines the name of the scene file or starts with \"data:\" following by the stringified version of the scene or a File object (default: empty string)\n * @param scene is the instance of BABYLON.Scene to append to (default: last created scene)\n * @param overwriteAnimations when true, animations are cleaned before importing new ones. Animations are appended otherwise\n * @param animationGroupLoadingMode defines how to handle old animations groups before importing new ones\n * @param targetConverter defines a function used to convert animation targets from loaded scene to current scene (default: search node by name)\n * @param onSuccess a callback with the scene when import succeeds\n * @param onProgress a callback with a progress event for each file being loaded\n * @param onError a callback with the scene, a message, and possibly an exception when import fails\n * @param pluginExtension the extension used to determine the plugin\n */\n static ImportAnimations(rootUrl, sceneFilename = \"\", scene = EngineStore.LastCreatedScene, overwriteAnimations = true, animationGroupLoadingMode = SceneLoaderAnimationGroupLoadingMode.Clean, targetConverter = null, onSuccess = null, onProgress = null, onError = null, pluginExtension = null) {\n if (!scene) {\n Logger.Error(\"No scene available to load animations to\");\n return;\n }\n if (overwriteAnimations) {\n // Reset, stop and dispose all animations before loading new ones\n for (const animatable of scene.animatables) {\n animatable.reset();\n }\n scene.stopAllAnimations();\n scene.animationGroups.slice().forEach((animationGroup) => {\n animationGroup.dispose();\n });\n const nodes = scene.getNodes();\n nodes.forEach((node) => {\n if (node.animations) {\n node.animations = [];\n }\n });\n }\n else {\n switch (animationGroupLoadingMode) {\n case SceneLoaderAnimationGroupLoadingMode.Clean:\n scene.animationGroups.slice().forEach((animationGroup) => {\n animationGroup.dispose();\n });\n break;\n case SceneLoaderAnimationGroupLoadingMode.Stop:\n scene.animationGroups.forEach((animationGroup) => {\n animationGroup.stop();\n });\n break;\n case SceneLoaderAnimationGroupLoadingMode.Sync:\n scene.animationGroups.forEach((animationGroup) => {\n animationGroup.reset();\n animationGroup.restart();\n });\n break;\n case SceneLoaderAnimationGroupLoadingMode.NoSync:\n // nothing to do\n break;\n default:\n Logger.Error(\"Unknown animation group loading mode value '\" + animationGroupLoadingMode + \"'\");\n return;\n }\n }\n const startingIndexForNewAnimatables = scene.animatables.length;\n const onAssetContainerLoaded = (container) => {\n container.mergeAnimationsTo(scene, scene.animatables.slice(startingIndexForNewAnimatables), targetConverter);\n container.dispose();\n scene.onAnimationFileImportedObservable.notifyObservers(scene);\n if (onSuccess) {\n onSuccess(scene);\n }\n };\n this.LoadAssetContainer(rootUrl, sceneFilename, scene, onAssetContainerLoaded, onProgress, onError, pluginExtension);\n }\n /**\n * Import animations from a file into a scene\n * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb)\n * @param sceneFilename a string that defines the name of the scene file or starts with \"data:\" following by the stringified version of the scene or a File object (default: empty string)\n * @param scene is the instance of BABYLON.Scene to append to (default: last created scene)\n * @param overwriteAnimations when true, animations are cleaned before importing new ones. Animations are appended otherwise\n * @param animationGroupLoadingMode defines how to handle old animations groups before importing new ones\n * @param targetConverter defines a function used to convert animation targets from loaded scene to current scene (default: search node by name)\n * @param onSuccess a callback with the scene when import succeeds\n * @param onProgress a callback with a progress event for each file being loaded\n * @param onError a callback with the scene, a message, and possibly an exception when import fails\n * @param pluginExtension the extension used to determine the plugin\n * @returns the updated scene with imported animations\n */\n static ImportAnimationsAsync(rootUrl, sceneFilename = \"\", scene = EngineStore.LastCreatedScene, overwriteAnimations = true, animationGroupLoadingMode = SceneLoaderAnimationGroupLoadingMode.Clean, targetConverter = null, \n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n onSuccess = null, onProgress = null, \n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n onError = null, pluginExtension = null) {\n return new Promise((resolve, reject) => {\n SceneLoader.ImportAnimations(rootUrl, sceneFilename, scene, overwriteAnimations, animationGroupLoadingMode, targetConverter, (_scene) => {\n resolve(_scene);\n }, onProgress, (_scene, message, exception) => {\n reject(exception || new Error(message));\n }, pluginExtension);\n });\n }\n }\n /**\n * No logging while loading\n */\n SceneLoader.NO_LOGGING = 0;\n /**\n * Minimal logging while loading\n */\n SceneLoader.MINIMAL_LOGGING = 1;\n /**\n * Summary logging while loading\n */\n SceneLoader.SUMMARY_LOGGING = 2;\n /**\n * Detailed logging while loading\n */\n SceneLoader.DETAILED_LOGGING = 3;\n // Members\n /**\n * Event raised when a plugin is used to load a scene\n */\n SceneLoader.OnPluginActivatedObservable = new Observable$1();\n SceneLoader._RegisteredPlugins = {};\n SceneLoader._ShowingLoadingScreen = false;\n\n /**\n * An Abstract Motion controller\n * This class receives an xrInput and a profile layout and uses those to initialize the components\n * Each component has an observable to check for changes in value and state\n */\n class WebXRAbstractMotionController {\n /**\n * constructs a new abstract motion controller\n * @param scene the scene to which the model of the controller will be added\n * @param layout The profile layout to load\n * @param gamepadObject The gamepad object correlating to this controller\n * @param handedness handedness (left/right/none) of this controller\n * @param _doNotLoadControllerMesh set this flag to ignore the mesh loading\n * @param _controllerCache a cache holding controller models already loaded in this session\n */\n constructor(\n // eslint-disable-next-line @typescript-eslint/naming-convention\n scene, \n // eslint-disable-next-line @typescript-eslint/naming-convention\n layout, \n /**\n * The gamepad object correlating to this controller\n */\n gamepadObject, \n /**\n * handedness (left/right/none) of this controller\n */\n handedness, \n /**\n * @internal\n */\n _doNotLoadControllerMesh = false, _controllerCache) {\n this.scene = scene;\n this.layout = layout;\n this.gamepadObject = gamepadObject;\n this.handedness = handedness;\n this._doNotLoadControllerMesh = _doNotLoadControllerMesh;\n this._controllerCache = _controllerCache;\n this._initComponent = (id) => {\n if (!id) {\n return;\n }\n const componentDef = this.layout.components[id];\n const type = componentDef.type;\n const buttonIndex = componentDef.gamepadIndices.button;\n // search for axes\n const axes = [];\n if (componentDef.gamepadIndices.xAxis !== undefined && componentDef.gamepadIndices.yAxis !== undefined) {\n axes.push(componentDef.gamepadIndices.xAxis, componentDef.gamepadIndices.yAxis);\n }\n this.components[id] = new WebXRControllerComponent(id, type, buttonIndex, axes);\n };\n this._modelReady = false;\n /**\n * A map of components (WebXRControllerComponent) in this motion controller\n * Components have a ComponentType and can also have both button and axis definitions\n */\n this.components = {};\n /**\n * Disable the model's animation. Can be set at any time.\n */\n this.disableAnimation = false;\n /**\n * Observers registered here will be triggered when the model of this controller is done loading\n */\n this.onModelLoadedObservable = new Observable$1();\n // initialize the components\n if (layout.components) {\n Object.keys(layout.components).forEach(this._initComponent);\n }\n // Model is loaded in WebXRInput\n }\n /**\n * Dispose this controller, the model mesh and all its components\n */\n dispose() {\n this.getComponentIds().forEach((id) => this.getComponent(id).dispose());\n if (this.rootMesh) {\n this.rootMesh.getChildren(undefined, true).forEach((node) => {\n node.setEnabled(false);\n });\n this.rootMesh.dispose(!!this._controllerCache, !this._controllerCache);\n }\n }\n /**\n * Returns all components of specific type\n * @param type the type to search for\n * @returns an array of components with this type\n */\n getAllComponentsOfType(type) {\n return this.getComponentIds()\n .map((id) => this.components[id])\n .filter((component) => component.type === type);\n }\n /**\n * get a component based an its component id as defined in layout.components\n * @param id the id of the component\n * @returns the component correlates to the id or undefined if not found\n */\n getComponent(id) {\n return this.components[id];\n }\n /**\n * Get the list of components available in this motion controller\n * @returns an array of strings correlating to available components\n */\n getComponentIds() {\n return Object.keys(this.components);\n }\n /**\n * Get the first component of specific type\n * @param type type of component to find\n * @returns a controller component or null if not found\n */\n getComponentOfType(type) {\n return this.getAllComponentsOfType(type)[0] || null;\n }\n /**\n * Get the main (Select) component of this controller as defined in the layout\n * @returns the main component of this controller\n */\n getMainComponent() {\n return this.getComponent(this.layout.selectComponentId);\n }\n /**\n * Loads the model correlating to this controller\n * When the mesh is loaded, the onModelLoadedObservable will be triggered\n * @returns A promise fulfilled with the result of the model loading\n */\n async loadModel() {\n const useGeneric = !this._getModelLoadingConstraints();\n let loadingParams = this._getGenericFilenameAndPath();\n // Checking if GLB loader is present\n if (useGeneric) {\n Logger.Warn(\"Falling back to generic models\");\n }\n else {\n loadingParams = this._getFilenameAndPath();\n }\n return new Promise((resolve, reject) => {\n const meshesLoaded = (meshes) => {\n if (useGeneric) {\n this._getGenericParentMesh(meshes);\n }\n else {\n this._setRootMesh(meshes);\n }\n this._processLoadedModel(meshes);\n this._modelReady = true;\n this.onModelLoadedObservable.notifyObservers(this);\n resolve(true);\n };\n if (this._controllerCache) {\n // look for it in the cache\n const found = this._controllerCache.filter((c) => {\n return c.filename === loadingParams.filename && c.path === loadingParams.path;\n });\n if (found[0]) {\n found[0].meshes.forEach((mesh) => mesh.setEnabled(true));\n meshesLoaded(found[0].meshes);\n return;\n // found, don't continue to load\n }\n }\n SceneLoader.ImportMesh(\"\", loadingParams.path, loadingParams.filename, this.scene, (meshes) => {\n if (this._controllerCache) {\n this._controllerCache.push(Object.assign(Object.assign({}, loadingParams), { meshes }));\n }\n meshesLoaded(meshes);\n }, null, (_scene, message) => {\n Logger.Log(message);\n Logger.Warn(`Failed to retrieve controller model of type ${this.profileId} from the remote server: ${loadingParams.path}${loadingParams.filename}`);\n reject(message);\n });\n });\n }\n /**\n * Update this model using the current XRFrame\n * @param xrFrame the current xr frame to use and update the model\n */\n updateFromXRFrame(xrFrame) {\n this.getComponentIds().forEach((id) => this.getComponent(id).update(this.gamepadObject));\n this.updateModel(xrFrame);\n }\n /**\n * Backwards compatibility due to a deeply-integrated typo\n */\n get handness() {\n return this.handedness;\n }\n /**\n * Pulse (vibrate) this controller\n * If the controller does not support pulses, this function will fail silently and return Promise directly after called\n * Consecutive calls to this function will cancel the last pulse call\n *\n * @param value the strength of the pulse in 0.0...1.0 range\n * @param duration Duration of the pulse in milliseconds\n * @param hapticActuatorIndex optional index of actuator (will usually be 0)\n * @returns a promise that will send true when the pulse has ended and false if the device doesn't support pulse or an error accrued\n */\n pulse(value, duration, hapticActuatorIndex = 0) {\n if (this.gamepadObject.hapticActuators && this.gamepadObject.hapticActuators[hapticActuatorIndex]) {\n return this.gamepadObject.hapticActuators[hapticActuatorIndex].pulse(value, duration);\n }\n else {\n return Promise.resolve(false);\n }\n }\n // Look through all children recursively. This will return null if no mesh exists with the given name.\n _getChildByName(node, name) {\n return node.getChildren((n) => n.name === name, false)[0];\n }\n // Look through only immediate children. This will return null if no mesh exists with the given name.\n _getImmediateChildByName(node, name) {\n return node.getChildren((n) => n.name == name, true)[0];\n }\n /**\n * Moves the axis on the controller mesh based on its current state\n * @param axisMap\n * @param axisValue the value of the axis which determines the meshes new position\n * @internal\n */\n _lerpTransform(axisMap, axisValue, fixValueCoordinates) {\n if (!axisMap.minMesh || !axisMap.maxMesh || !axisMap.valueMesh) {\n return;\n }\n if (!axisMap.minMesh.rotationQuaternion || !axisMap.maxMesh.rotationQuaternion || !axisMap.valueMesh.rotationQuaternion) {\n return;\n }\n // Convert from gamepad value range (-1 to +1) to lerp range (0 to 1)\n const lerpValue = fixValueCoordinates ? axisValue * 0.5 + 0.5 : axisValue;\n Quaternion.SlerpToRef(axisMap.minMesh.rotationQuaternion, axisMap.maxMesh.rotationQuaternion, lerpValue, axisMap.valueMesh.rotationQuaternion);\n Vector3.LerpToRef(axisMap.minMesh.position, axisMap.maxMesh.position, lerpValue, axisMap.valueMesh.position);\n }\n /**\n * Update the model itself with the current frame data\n * @param xrFrame the frame to use for updating the model mesh\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n updateModel(xrFrame) {\n if (!this._modelReady) {\n return;\n }\n this._updateModel(xrFrame);\n }\n _getGenericFilenameAndPath() {\n return {\n filename: \"generic.babylon\",\n path: \"https://controllers.babylonjs.com/generic/\",\n };\n }\n _getGenericParentMesh(meshes) {\n this.rootMesh = new Mesh(this.profileId + \" \" + this.handedness, this.scene);\n meshes.forEach((mesh) => {\n if (!mesh.parent) {\n mesh.isPickable = false;\n mesh.setParent(this.rootMesh);\n }\n });\n this.rootMesh.rotationQuaternion = Quaternion.FromEulerAngles(0, Math.PI, 0);\n }\n }\n\n /**\n * A generic trigger-only motion controller for WebXR\n */\n class WebXRGenericTriggerMotionController extends WebXRAbstractMotionController {\n constructor(scene, gamepadObject, handedness) {\n super(scene, GenericTriggerLayout[handedness], gamepadObject, handedness);\n this.profileId = WebXRGenericTriggerMotionController.ProfileId;\n }\n _getFilenameAndPath() {\n return {\n filename: \"generic.babylon\",\n path: \"https://controllers.babylonjs.com/generic/\",\n };\n }\n _getModelLoadingConstraints() {\n return true;\n }\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _processLoadedModel(meshes) {\n // nothing to do\n }\n _setRootMesh(meshes) {\n this.rootMesh = new Mesh(this.profileId + \" \" + this.handedness, this.scene);\n meshes.forEach((mesh) => {\n mesh.isPickable = false;\n if (!mesh.parent) {\n mesh.setParent(this.rootMesh);\n }\n });\n this.rootMesh.rotationQuaternion = Quaternion.FromEulerAngles(0, Math.PI, 0);\n }\n _updateModel() {\n // no-op\n }\n }\n /**\n * Static version of the profile id of this controller\n */\n WebXRGenericTriggerMotionController.ProfileId = \"generic-trigger\";\n // https://github.com/immersive-web/webxr-input-profiles/blob/master/packages/registry/profiles/generic/generic-trigger-touchpad-thumbstick.json\n const GenericTriggerLayout = {\n left: {\n selectComponentId: \"xr-standard-trigger\",\n components: {\n // eslint-disable-next-line @typescript-eslint/naming-convention\n \"xr-standard-trigger\": {\n type: \"trigger\",\n gamepadIndices: {\n button: 0,\n },\n rootNodeName: \"xr_standard_trigger\",\n visualResponses: {},\n },\n },\n gamepadMapping: \"xr-standard\",\n rootNodeName: \"generic-trigger-left\",\n assetPath: \"left.glb\",\n },\n right: {\n selectComponentId: \"xr-standard-trigger\",\n components: {\n // eslint-disable-next-line @typescript-eslint/naming-convention\n \"xr-standard-trigger\": {\n type: \"trigger\",\n gamepadIndices: {\n button: 0,\n },\n rootNodeName: \"xr_standard_trigger\",\n visualResponses: {},\n },\n },\n gamepadMapping: \"xr-standard\",\n rootNodeName: \"generic-trigger-right\",\n assetPath: \"right.glb\",\n },\n none: {\n selectComponentId: \"xr-standard-trigger\",\n components: {\n // eslint-disable-next-line @typescript-eslint/naming-convention\n \"xr-standard-trigger\": {\n type: \"trigger\",\n gamepadIndices: {\n button: 0,\n },\n rootNodeName: \"xr_standard_trigger\",\n visualResponses: {},\n },\n },\n gamepadMapping: \"xr-standard\",\n rootNodeName: \"generic-trigger-none\",\n assetPath: \"none.glb\",\n },\n };\n\n /**\n * Creates the VertexData for an ellipsoid, defaults to a sphere\n * @param options an object used to set the following optional parameters for the box, required but can be empty\n * * segments sets the number of horizontal strips optional, default 32\n * * diameter sets the axes dimensions, diameterX, diameterY and diameterZ to the value of diameter, optional default 1\n * * diameterX sets the diameterX (x direction) of the ellipsoid, overwrites the diameterX set by diameter, optional, default diameter\n * * diameterY sets the diameterY (y direction) of the ellipsoid, overwrites the diameterY set by diameter, optional, default diameter\n * * diameterZ sets the diameterZ (z direction) of the ellipsoid, overwrites the diameterZ set by diameter, optional, default diameter\n * * arc a number from 0 to 1, to create an unclosed ellipsoid based on the fraction of the circumference (latitude) given by the arc value, optional, default 1\n * * slice a number from 0 to 1, to create an unclosed ellipsoid based on the fraction of the height (latitude) given by the arc value, optional, default 1\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.segments\n * @param options.diameter\n * @param options.diameterX\n * @param options.diameterY\n * @param options.diameterZ\n * @param options.arc\n * @param options.slice\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param options.dedupTopBottomIndices\n * @returns the VertexData of the ellipsoid\n */\n function CreateSphereVertexData(options) {\n const segments = (options.segments || 32) | 0;\n const diameterX = options.diameterX || options.diameter || 1;\n const diameterY = options.diameterY || options.diameter || 1;\n const diameterZ = options.diameterZ || options.diameter || 1;\n const arc = options.arc && (options.arc <= 0 || options.arc > 1) ? 1.0 : options.arc || 1.0;\n const slice = options.slice && options.slice <= 0 ? 1.0 : options.slice || 1.0;\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\n const dedupTopBottomIndices = !!options.dedupTopBottomIndices;\n const radius = new Vector3(diameterX / 2, diameterY / 2, diameterZ / 2);\n const totalZRotationSteps = 2 + segments;\n const totalYRotationSteps = 2 * totalZRotationSteps;\n const indices = [];\n const positions = [];\n const normals = [];\n const uvs = [];\n for (let zRotationStep = 0; zRotationStep <= totalZRotationSteps; zRotationStep++) {\n const normalizedZ = zRotationStep / totalZRotationSteps;\n const angleZ = normalizedZ * Math.PI * slice;\n for (let yRotationStep = 0; yRotationStep <= totalYRotationSteps; yRotationStep++) {\n const normalizedY = yRotationStep / totalYRotationSteps;\n const angleY = normalizedY * Math.PI * 2 * arc;\n const rotationZ = Matrix.RotationZ(-angleZ);\n const rotationY = Matrix.RotationY(angleY);\n const afterRotZ = Vector3.TransformCoordinates(Vector3.Up(), rotationZ);\n const complete = Vector3.TransformCoordinates(afterRotZ, rotationY);\n const vertex = complete.multiply(radius);\n const normal = complete.divide(radius).normalize();\n positions.push(vertex.x, vertex.y, vertex.z);\n normals.push(normal.x, normal.y, normal.z);\n uvs.push(normalizedY, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - normalizedZ : normalizedZ);\n }\n if (zRotationStep > 0) {\n const verticesCount = positions.length / 3;\n for (let firstIndex = verticesCount - 2 * (totalYRotationSteps + 1); firstIndex + totalYRotationSteps + 2 < verticesCount; firstIndex++) {\n if (dedupTopBottomIndices) {\n if (zRotationStep > 1) {\n indices.push(firstIndex);\n indices.push(firstIndex + 1);\n indices.push(firstIndex + totalYRotationSteps + 1);\n }\n if (zRotationStep < totalZRotationSteps || slice < 1.0) {\n indices.push(firstIndex + totalYRotationSteps + 1);\n indices.push(firstIndex + 1);\n indices.push(firstIndex + totalYRotationSteps + 2);\n }\n }\n else {\n indices.push(firstIndex);\n indices.push(firstIndex + 1);\n indices.push(firstIndex + totalYRotationSteps + 1);\n indices.push(firstIndex + totalYRotationSteps + 1);\n indices.push(firstIndex + 1);\n indices.push(firstIndex + totalYRotationSteps + 2);\n }\n }\n }\n }\n // Sides\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\n // Result\n const vertexData = new VertexData();\n vertexData.indices = indices;\n vertexData.positions = positions;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n return vertexData;\n }\n /**\n * Creates a sphere mesh\n * * The parameter `diameter` sets the diameter size (float) of the sphere (default 1)\n * * You can set some different sphere dimensions, for instance to build an ellipsoid, by using the parameters `diameterX`, `diameterY` and `diameterZ` (all by default have the same value of `diameter`)\n * * The parameter `segments` sets the sphere number of horizontal stripes (positive integer, default 32)\n * * You can create an unclosed sphere with the parameter `arc` (positive float, default 1), valued between 0 and 1, what is the ratio of the circumference (latitude) : 2 x PI x ratio\n * * You can create an unclosed sphere on its height with the parameter `slice` (positive float, default1), valued between 0 and 1, what is the height ratio (longitude)\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param options.segments\n * @param options.diameter\n * @param options.diameterX\n * @param options.diameterY\n * @param options.diameterZ\n * @param options.arc\n * @param options.slice\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param options.updatable\n * @param scene defines the hosting scene\n * @returns the sphere mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#sphere\n */\n function CreateSphere(name, options = {}, scene = null) {\n const sphere = new Mesh(name, scene);\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n sphere._originalBuilderSideOrientation = options.sideOrientation;\n const vertexData = CreateSphereVertexData(options);\n vertexData.applyToMesh(sphere, options.updatable);\n return sphere;\n }\n /**\n * Class containing static functions to help procedurally build meshes\n * @deprecated use CreateSphere directly\n */\n const SphereBuilder = {\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CreateSphere,\n };\n VertexData.CreateSphere = CreateSphereVertexData;\n Mesh.CreateSphere = (name, segments, diameter, scene, updatable, sideOrientation) => {\n const options = {\n segments: segments,\n diameterX: diameter,\n diameterY: diameter,\n diameterZ: diameter,\n sideOrientation: sideOrientation,\n updatable: updatable,\n };\n return CreateSphere(name, options, scene);\n };\n\n /**\n * A profiled motion controller has its profile loaded from an online repository.\n * The class is responsible of loading the model, mapping the keys and enabling model-animations\n */\n class WebXRProfiledMotionController extends WebXRAbstractMotionController {\n constructor(scene, xrInput, _profile, _repositoryUrl, \n // eslint-disable-next-line @typescript-eslint/naming-convention\n controllerCache) {\n super(scene, _profile.layouts[xrInput.handedness || \"none\"], xrInput.gamepad, xrInput.handedness, undefined, controllerCache);\n this._repositoryUrl = _repositoryUrl;\n this.controllerCache = controllerCache;\n this._buttonMeshMapping = {};\n this._touchDots = {};\n this.profileId = _profile.profileId;\n }\n dispose() {\n super.dispose();\n if (!this.controllerCache) {\n Object.keys(this._touchDots).forEach((visResKey) => {\n this._touchDots[visResKey].dispose();\n });\n }\n }\n _getFilenameAndPath() {\n return {\n filename: this.layout.assetPath,\n path: `${this._repositoryUrl}/profiles/${this.profileId}/`,\n };\n }\n _getModelLoadingConstraints() {\n const glbLoaded = SceneLoader.IsPluginForExtensionAvailable(\".glb\");\n if (!glbLoaded) {\n Logger.Warn(\"glTF / glb loader was not registered, using generic controller instead\");\n }\n return glbLoaded;\n }\n _processLoadedModel(_meshes) {\n this.getComponentIds().forEach((type) => {\n const componentInLayout = this.layout.components[type];\n this._buttonMeshMapping[type] = {\n mainMesh: this._getChildByName(this.rootMesh, componentInLayout.rootNodeName),\n states: {},\n };\n Object.keys(componentInLayout.visualResponses).forEach((visualResponseKey) => {\n const visResponse = componentInLayout.visualResponses[visualResponseKey];\n if (visResponse.valueNodeProperty === \"transform\") {\n this._buttonMeshMapping[type].states[visualResponseKey] = {\n valueMesh: this._getChildByName(this.rootMesh, visResponse.valueNodeName),\n minMesh: this._getChildByName(this.rootMesh, visResponse.minNodeName),\n maxMesh: this._getChildByName(this.rootMesh, visResponse.maxNodeName),\n };\n }\n else {\n // visibility, usually for touchpads\n const nameOfMesh = componentInLayout.type === WebXRControllerComponent.TOUCHPAD_TYPE && componentInLayout.touchPointNodeName\n ? componentInLayout.touchPointNodeName\n : visResponse.valueNodeName;\n this._buttonMeshMapping[type].states[visualResponseKey] = {\n valueMesh: this._getChildByName(this.rootMesh, nameOfMesh),\n };\n if (componentInLayout.type === WebXRControllerComponent.TOUCHPAD_TYPE && !this._touchDots[visualResponseKey]) {\n const dot = CreateSphere(visualResponseKey + \"dot\", {\n diameter: 0.0015,\n segments: 8,\n }, this.scene);\n dot.material = new StandardMaterial(visualResponseKey + \"mat\", this.scene);\n dot.material.diffuseColor = Color3.Red();\n dot.parent = this._buttonMeshMapping[type].states[visualResponseKey].valueMesh || null;\n dot.isVisible = false;\n this._touchDots[visualResponseKey] = dot;\n }\n }\n });\n });\n }\n _setRootMesh(meshes) {\n this.rootMesh = new Mesh(this.profileId + \"-\" + this.handedness, this.scene);\n this.rootMesh.isPickable = false;\n let rootMesh;\n // Find the root node in the loaded glTF scene, and attach it as a child of 'parentMesh'\n for (let i = 0; i < meshes.length; i++) {\n const mesh = meshes[i];\n mesh.isPickable = false;\n if (!mesh.parent) {\n // Handle root node, attach to the new parentMesh\n rootMesh = mesh;\n }\n }\n if (rootMesh) {\n rootMesh.setParent(this.rootMesh);\n }\n if (!this.scene.useRightHandedSystem) {\n this.rootMesh.rotate(Axis.Y, Math.PI, Space.WORLD);\n }\n }\n _updateModel(_xrFrame) {\n if (this.disableAnimation) {\n return;\n }\n this.getComponentIds().forEach((id) => {\n const component = this.getComponent(id);\n if (!component.hasChanges) {\n return;\n }\n const meshes = this._buttonMeshMapping[id];\n const componentInLayout = this.layout.components[id];\n Object.keys(componentInLayout.visualResponses).forEach((visualResponseKey) => {\n const visResponse = componentInLayout.visualResponses[visualResponseKey];\n let value = component.value;\n if (visResponse.componentProperty === \"xAxis\") {\n value = component.axes.x;\n }\n else if (visResponse.componentProperty === \"yAxis\") {\n value = component.axes.y;\n }\n if (visResponse.valueNodeProperty === \"transform\") {\n this._lerpTransform(meshes.states[visualResponseKey], value, visResponse.componentProperty !== \"button\");\n }\n else {\n // visibility\n const valueMesh = meshes.states[visualResponseKey].valueMesh;\n if (valueMesh) {\n valueMesh.isVisible = component.touched || component.pressed;\n }\n if (this._touchDots[visualResponseKey]) {\n this._touchDots[visualResponseKey].isVisible = component.touched || component.pressed;\n }\n }\n });\n });\n }\n }\n\n /**\n * The MotionController Manager manages all registered motion controllers and loads the right one when needed.\n *\n * When this repository is complete: https://github.com/immersive-web/webxr-input-profiles/tree/master/packages/assets\n * it should be replaced with auto-loaded controllers.\n *\n * When using a model try to stay as generic as possible. Eventually there will be no need in any of the controller classes\n */\n const controllerCache = [];\n /**\n * Motion controller manager is managing the different webxr profiles and makes sure the right\n * controller is being loaded.\n */\n class WebXRMotionControllerManager {\n /**\n * Clear the cache used for profile loading and reload when requested again\n */\n static ClearProfilesCache() {\n this._ProfilesList = null;\n this._ProfileLoadingPromises = {};\n }\n /**\n * Register the default fallbacks.\n * This function is called automatically when this file is imported.\n */\n static DefaultFallbacks() {\n this.RegisterFallbacksForProfileId(\"google-daydream\", [\"generic-touchpad\"]);\n this.RegisterFallbacksForProfileId(\"htc-vive-focus\", [\"generic-trigger-touchpad\"]);\n this.RegisterFallbacksForProfileId(\"htc-vive\", [\"generic-trigger-squeeze-touchpad\"]);\n this.RegisterFallbacksForProfileId(\"magicleap-one\", [\"generic-trigger-squeeze-touchpad\"]);\n this.RegisterFallbacksForProfileId(\"windows-mixed-reality\", [\"generic-trigger-squeeze-touchpad-thumbstick\"]);\n this.RegisterFallbacksForProfileId(\"microsoft-mixed-reality\", [\"windows-mixed-reality\", \"generic-trigger-squeeze-touchpad-thumbstick\"]);\n this.RegisterFallbacksForProfileId(\"oculus-go\", [\"generic-trigger-touchpad\"]);\n this.RegisterFallbacksForProfileId(\"oculus-touch-v2\", [\"oculus-touch\", \"generic-trigger-squeeze-thumbstick\"]);\n this.RegisterFallbacksForProfileId(\"oculus-touch\", [\"generic-trigger-squeeze-thumbstick\"]);\n this.RegisterFallbacksForProfileId(\"samsung-gearvr\", [\"windows-mixed-reality\", \"generic-trigger-squeeze-touchpad-thumbstick\"]);\n this.RegisterFallbacksForProfileId(\"samsung-odyssey\", [\"generic-touchpad\"]);\n this.RegisterFallbacksForProfileId(\"valve-index\", [\"generic-trigger-squeeze-touchpad-thumbstick\"]);\n this.RegisterFallbacksForProfileId(\"generic-hand-select\", [\"generic-trigger\"]);\n }\n /**\n * Find a fallback profile if the profile was not found. There are a few predefined generic profiles.\n * @param profileId the profile to which a fallback needs to be found\n * @returns an array with corresponding fallback profiles\n */\n static FindFallbackWithProfileId(profileId) {\n const returnArray = this._Fallbacks[profileId] || [];\n returnArray.unshift(profileId);\n return returnArray;\n }\n /**\n * When acquiring a new xrInput object (usually by the WebXRInput class), match it with the correct profile.\n * The order of search:\n *\n * 1) Iterate the profiles array of the xr input and try finding a corresponding motion controller\n * 2) (If not found) search in the gamepad id and try using it (legacy versions only)\n * 3) search for registered fallbacks (should be redundant, nonetheless it makes sense to check)\n * 4) return the generic trigger controller if none were found\n *\n * @param xrInput the xrInput to which a new controller is initialized\n * @param scene the scene to which the model will be added\n * @param forceProfile force a certain profile for this controller\n * @returns A promise that fulfils with the motion controller class for this profile id or the generic standard class if none was found\n */\n static GetMotionControllerWithXRInput(xrInput, scene, forceProfile) {\n const profileArray = [];\n if (forceProfile) {\n profileArray.push(forceProfile);\n }\n profileArray.push(...(xrInput.profiles || []));\n // emulator support\n if (profileArray.length && !profileArray[0]) {\n // remove the first \"undefined\" that the emulator is adding\n profileArray.pop();\n }\n // legacy support - try using the gamepad id\n if (xrInput.gamepad && xrInput.gamepad.id) {\n switch (xrInput.gamepad.id) {\n case xrInput.gamepad.id.match(/oculus touch/gi) ? xrInput.gamepad.id : undefined:\n // oculus in gamepad id\n profileArray.push(\"oculus-touch-v2\");\n break;\n }\n }\n // make sure microsoft/windows mixed reality works correctly\n const windowsMRIdx = profileArray.indexOf(\"windows-mixed-reality\");\n if (windowsMRIdx !== -1) {\n profileArray.splice(windowsMRIdx, 0, \"microsoft-mixed-reality\");\n }\n if (!profileArray.length) {\n profileArray.push(\"generic-trigger\");\n }\n if (this.UseOnlineRepository) {\n const firstFunction = this.PrioritizeOnlineRepository ? this._LoadProfileFromRepository : this._LoadProfilesFromAvailableControllers;\n const secondFunction = this.PrioritizeOnlineRepository ? this._LoadProfilesFromAvailableControllers : this._LoadProfileFromRepository;\n return firstFunction.call(this, profileArray, xrInput, scene).catch(() => {\n return secondFunction.call(this, profileArray, xrInput, scene);\n });\n }\n else {\n // use only available functions\n return this._LoadProfilesFromAvailableControllers(profileArray, xrInput, scene);\n }\n }\n /**\n * Register a new controller based on its profile. This function will be called by the controller classes themselves.\n *\n * If you are missing a profile, make sure it is imported in your source, otherwise it will not register.\n *\n * @param type the profile type to register\n * @param constructFunction the function to be called when loading this profile\n */\n static RegisterController(type, constructFunction) {\n this._AvailableControllers[type] = constructFunction;\n }\n /**\n * Register a fallback to a specific profile.\n * @param profileId the profileId that will receive the fallbacks\n * @param fallbacks A list of fallback profiles\n */\n static RegisterFallbacksForProfileId(profileId, fallbacks) {\n if (this._Fallbacks[profileId]) {\n this._Fallbacks[profileId].push(...fallbacks);\n }\n else {\n this._Fallbacks[profileId] = fallbacks;\n }\n }\n /**\n * Will update the list of profiles available in the repository\n * @returns a promise that resolves to a map of profiles available online\n */\n static UpdateProfilesList() {\n this._ProfilesList = Tools.LoadFileAsync(this.BaseRepositoryUrl + \"/profiles/profilesList.json\", false).then((data) => {\n return JSON.parse(data.toString());\n });\n return this._ProfilesList;\n }\n /**\n * Clear the controller's cache (usually happens at the end of a session)\n */\n static ClearControllerCache() {\n controllerCache.forEach((cacheItem) => {\n cacheItem.meshes.forEach((mesh) => {\n mesh.dispose(false, true);\n });\n });\n controllerCache.length = 0;\n }\n static _LoadProfileFromRepository(profileArray, xrInput, scene) {\n return Promise.resolve()\n .then(() => {\n if (!this._ProfilesList) {\n return this.UpdateProfilesList();\n }\n else {\n return this._ProfilesList;\n }\n })\n .then((profilesList) => {\n // load the right profile\n for (let i = 0; i < profileArray.length; ++i) {\n // defensive\n if (!profileArray[i]) {\n continue;\n }\n if (profilesList[profileArray[i]]) {\n return profileArray[i];\n }\n }\n throw new Error(`neither controller ${profileArray[0]} nor all fallbacks were found in the repository,`);\n })\n .then((profileToLoad) => {\n // load the profile\n if (!this._ProfileLoadingPromises[profileToLoad]) {\n this._ProfileLoadingPromises[profileToLoad] = Tools.LoadFileAsync(`${this.BaseRepositoryUrl}/profiles/${profileToLoad}/profile.json`, false).then((data) => JSON.parse(data));\n }\n return this._ProfileLoadingPromises[profileToLoad];\n })\n .then((profile) => {\n return new WebXRProfiledMotionController(scene, xrInput, profile, this.BaseRepositoryUrl, this.DisableControllerCache ? undefined : controllerCache);\n });\n }\n static _LoadProfilesFromAvailableControllers(profileArray, xrInput, scene) {\n // check fallbacks\n for (let i = 0; i < profileArray.length; ++i) {\n // defensive\n if (!profileArray[i]) {\n continue;\n }\n const fallbacks = this.FindFallbackWithProfileId(profileArray[i]);\n for (let j = 0; j < fallbacks.length; ++j) {\n const constructionFunction = this._AvailableControllers[fallbacks[j]];\n if (constructionFunction) {\n return Promise.resolve(constructionFunction(xrInput, scene));\n }\n }\n }\n throw new Error(`no controller requested was found in the available controllers list`);\n }\n }\n WebXRMotionControllerManager._AvailableControllers = {};\n WebXRMotionControllerManager._Fallbacks = {};\n // cache for loading\n WebXRMotionControllerManager._ProfileLoadingPromises = {};\n /**\n * The base URL of the online controller repository. Can be changed at any time.\n */\n WebXRMotionControllerManager.BaseRepositoryUrl = \"https://immersive-web.github.io/webxr-input-profiles/packages/viewer/dist\";\n /**\n * Which repository gets priority - local or online\n */\n WebXRMotionControllerManager.PrioritizeOnlineRepository = true;\n /**\n * Use the online repository, or use only locally-defined controllers\n */\n WebXRMotionControllerManager.UseOnlineRepository = true;\n /**\n * Disable the controller cache and load the models each time a new WebXRProfileMotionController is loaded.\n * Defaults to true.\n */\n WebXRMotionControllerManager.DisableControllerCache = true;\n // register the generic profile(s) here so we will at least have them\n WebXRMotionControllerManager.RegisterController(WebXRGenericTriggerMotionController.ProfileId, (xrInput, scene) => {\n return new WebXRGenericTriggerMotionController(scene, xrInput.gamepad, xrInput.handedness);\n });\n // register fallbacks\n WebXRMotionControllerManager.DefaultFallbacks();\n\n let idCount = 0;\n /**\n * Represents an XR controller\n */\n class WebXRInputSource {\n /**\n * Creates the input source object\n * @see https://doc.babylonjs.com/features/featuresDeepDive/webXR/webXRInputControllerSupport\n * @param _scene the scene which the controller should be associated to\n * @param inputSource the underlying input source for the controller\n * @param _options options for this controller creation\n */\n constructor(_scene, \n /** The underlying input source for the controller */\n inputSource, _options = {}) {\n this._scene = _scene;\n this.inputSource = inputSource;\n this._options = _options;\n this._tmpVector = new Vector3();\n this._disposed = false;\n /**\n * Event that fires when the controller is removed/disposed.\n * The object provided as event data is this controller, after associated assets were disposed.\n * uniqueId is still available.\n */\n this.onDisposeObservable = new Observable$1();\n /**\n * Will be triggered when the mesh associated with the motion controller is done loading.\n * It is also possible that this will never trigger (!) if no mesh was loaded, or if the developer decides to load a different mesh\n * A shortened version of controller -> motion controller -> on mesh loaded.\n */\n this.onMeshLoadedObservable = new Observable$1();\n /**\n * Observers registered here will trigger when a motion controller profile was assigned to this xr controller\n */\n this.onMotionControllerInitObservable = new Observable$1();\n this._uniqueId = `controller-${idCount++}-${inputSource.targetRayMode}-${inputSource.handedness}`;\n this.pointer = new AbstractMesh(`${this._uniqueId}-pointer`, _scene);\n this.pointer.rotationQuaternion = new Quaternion();\n if (this.inputSource.gripSpace) {\n this.grip = new AbstractMesh(`${this._uniqueId}-grip`, this._scene);\n this.grip.rotationQuaternion = new Quaternion();\n }\n this._tmpVector.set(0, 0, this._scene.useRightHandedSystem ? -1.0 : 1.0);\n // for now only load motion controllers if gamepad object available\n if (this.inputSource.gamepad && this.inputSource.targetRayMode === \"tracked-pointer\") {\n WebXRMotionControllerManager.GetMotionControllerWithXRInput(inputSource, _scene, this._options.forceControllerProfile).then((motionController) => {\n this.motionController = motionController;\n this.onMotionControllerInitObservable.notifyObservers(motionController);\n // should the model be loaded?\n if (!this._options.doNotLoadControllerMesh && !this.motionController._doNotLoadControllerMesh) {\n this.motionController.loadModel().then((success) => {\n var _a;\n if (success && this.motionController && this.motionController.rootMesh) {\n if (this._options.renderingGroupId) {\n // anything other than 0?\n this.motionController.rootMesh.renderingGroupId = this._options.renderingGroupId;\n this.motionController.rootMesh.getChildMeshes(false).forEach((mesh) => (mesh.renderingGroupId = this._options.renderingGroupId));\n }\n this.onMeshLoadedObservable.notifyObservers(this.motionController.rootMesh);\n this.motionController.rootMesh.parent = this.grip || this.pointer;\n this.motionController.disableAnimation = !!this._options.disableMotionControllerAnimation;\n }\n // make sure to dispose is the controller is already disposed\n if (this._disposed) {\n (_a = this.motionController) === null || _a === void 0 ? void 0 : _a.dispose();\n }\n });\n }\n }, () => {\n Tools.Warn(`Could not find a matching motion controller for the registered input source`);\n });\n }\n }\n /**\n * Get this controllers unique id\n */\n get uniqueId() {\n return this._uniqueId;\n }\n /**\n * Disposes of the object\n */\n dispose() {\n if (this.grip) {\n this.grip.dispose(true);\n }\n if (this.motionController) {\n this.motionController.dispose();\n }\n this.pointer.dispose(true);\n this.onMotionControllerInitObservable.clear();\n this.onMeshLoadedObservable.clear();\n this.onDisposeObservable.notifyObservers(this);\n this.onDisposeObservable.clear();\n this._disposed = true;\n }\n /**\n * Gets a world space ray coming from the pointer or grip\n * @param result the resulting ray\n * @param gripIfAvailable use the grip mesh instead of the pointer, if available\n */\n getWorldPointerRayToRef(result, gripIfAvailable = false) {\n const object = gripIfAvailable && this.grip ? this.grip : this.pointer;\n Vector3.TransformNormalToRef(this._tmpVector, object.getWorldMatrix(), result.direction);\n result.direction.normalize();\n result.origin.copyFrom(object.absolutePosition);\n result.length = 1000;\n }\n /**\n * Updates the controller pose based on the given XRFrame\n * @param xrFrame xr frame to update the pose with\n * @param referenceSpace reference space to use\n * @param xrCamera the xr camera, used for parenting\n */\n updateFromXRFrame(xrFrame, referenceSpace, xrCamera) {\n const pose = xrFrame.getPose(this.inputSource.targetRaySpace, referenceSpace);\n this._lastXRPose = pose;\n // Update the pointer mesh\n if (pose) {\n const pos = pose.transform.position;\n this.pointer.position.set(pos.x, pos.y, pos.z);\n const orientation = pose.transform.orientation;\n this.pointer.rotationQuaternion.set(orientation.x, orientation.y, orientation.z, orientation.w);\n if (!this._scene.useRightHandedSystem) {\n this.pointer.position.z *= -1;\n this.pointer.rotationQuaternion.z *= -1;\n this.pointer.rotationQuaternion.w *= -1;\n }\n this.pointer.parent = xrCamera.parent;\n }\n // Update the grip mesh if it exists\n if (this.inputSource.gripSpace && this.grip) {\n const pose = xrFrame.getPose(this.inputSource.gripSpace, referenceSpace);\n if (pose) {\n const pos = pose.transform.position;\n const orientation = pose.transform.orientation;\n this.grip.position.set(pos.x, pos.y, pos.z);\n this.grip.rotationQuaternion.set(orientation.x, orientation.y, orientation.z, orientation.w);\n if (!this._scene.useRightHandedSystem) {\n this.grip.position.z *= -1;\n this.grip.rotationQuaternion.z *= -1;\n this.grip.rotationQuaternion.w *= -1;\n }\n }\n this.grip.parent = xrCamera.parent;\n }\n if (this.motionController) {\n // either update buttons only or also position, if in gamepad mode\n this.motionController.updateFromXRFrame(xrFrame);\n }\n }\n }\n\n /**\n * XR input used to track XR inputs such as controllers/rays\n */\n class WebXRInput {\n /**\n * Initializes the WebXRInput\n * @param xrSessionManager the xr session manager for this session\n * @param xrCamera the WebXR camera for this session. Mainly used for teleportation\n * @param _options = initialization options for this xr input\n */\n constructor(\n /**\n * the xr session manager for this session\n */\n xrSessionManager, \n /**\n * the WebXR camera for this session. Mainly used for teleportation\n */\n xrCamera, _options = {}) {\n this.xrSessionManager = xrSessionManager;\n this.xrCamera = xrCamera;\n this._options = _options;\n /**\n * XR controllers being tracked\n */\n this.controllers = [];\n /**\n * Event when a controller has been connected/added\n */\n this.onControllerAddedObservable = new Observable$1();\n /**\n * Event when a controller has been removed/disconnected\n */\n this.onControllerRemovedObservable = new Observable$1();\n this._onInputSourcesChange = (event) => {\n this._addAndRemoveControllers(event.added, event.removed);\n };\n // Remove controllers when exiting XR\n this._sessionEndedObserver = this.xrSessionManager.onXRSessionEnded.add(() => {\n this._addAndRemoveControllers([], this.controllers.map((c) => {\n return c.inputSource;\n }));\n });\n this._sessionInitObserver = this.xrSessionManager.onXRSessionInit.add((session) => {\n session.addEventListener(\"inputsourceschange\", this._onInputSourcesChange);\n });\n this._frameObserver = this.xrSessionManager.onXRFrameObservable.add((frame) => {\n // Update controller pose info\n this.controllers.forEach((controller) => {\n controller.updateFromXRFrame(frame, this.xrSessionManager.referenceSpace, this.xrCamera);\n });\n });\n if (this._options.customControllersRepositoryURL) {\n WebXRMotionControllerManager.BaseRepositoryUrl = this._options.customControllersRepositoryURL;\n }\n WebXRMotionControllerManager.UseOnlineRepository = !this._options.disableOnlineControllerRepository;\n if (WebXRMotionControllerManager.UseOnlineRepository) {\n // pre-load the profiles list to load the controllers quicker afterwards\n try {\n WebXRMotionControllerManager.UpdateProfilesList().catch(() => {\n WebXRMotionControllerManager.UseOnlineRepository = false;\n });\n }\n catch (e) {\n WebXRMotionControllerManager.UseOnlineRepository = false;\n }\n }\n }\n _addAndRemoveControllers(addInputs, removeInputs) {\n // Add controllers if they don't already exist\n const sources = this.controllers.map((c) => {\n return c.inputSource;\n });\n for (const input of addInputs) {\n if (sources.indexOf(input) === -1) {\n const controller = new WebXRInputSource(this.xrSessionManager.scene, input, Object.assign(Object.assign({}, (this._options.controllerOptions || {})), { forceControllerProfile: this._options.forceInputProfile, doNotLoadControllerMesh: this._options.doNotLoadControllerMeshes, disableMotionControllerAnimation: this._options.disableControllerAnimation }));\n this.controllers.push(controller);\n this.onControllerAddedObservable.notifyObservers(controller);\n }\n }\n // Remove and dispose of controllers to be disposed\n const keepControllers = [];\n const removedControllers = [];\n this.controllers.forEach((c) => {\n if (removeInputs.indexOf(c.inputSource) === -1) {\n keepControllers.push(c);\n }\n else {\n removedControllers.push(c);\n }\n });\n this.controllers = keepControllers;\n removedControllers.forEach((c) => {\n this.onControllerRemovedObservable.notifyObservers(c);\n c.dispose();\n });\n }\n /**\n * Disposes of the object\n */\n dispose() {\n this.controllers.forEach((c) => {\n c.dispose();\n });\n this.xrSessionManager.onXRFrameObservable.remove(this._frameObserver);\n this.xrSessionManager.onXRSessionInit.remove(this._sessionInitObserver);\n this.xrSessionManager.onXRSessionEnded.remove(this._sessionEndedObserver);\n this.onControllerAddedObservable.clear();\n this.onControllerRemovedObservable.clear();\n // clear the controller cache\n WebXRMotionControllerManager.ClearControllerCache();\n }\n }\n\n /**\n * This is the base class for all WebXR features.\n * Since most features require almost the same resources and callbacks, this class can be used to simplify the development\n * Note that since the features manager is using the `IWebXRFeature` you are in no way obligated to use this class\n */\n class WebXRAbstractFeature {\n /**\n * Construct a new (abstract) WebXR feature\n * @param _xrSessionManager the xr session manager for this feature\n */\n constructor(_xrSessionManager) {\n this._xrSessionManager = _xrSessionManager;\n this._attached = false;\n this._removeOnDetach = [];\n /**\n * Is this feature disposed?\n */\n this.isDisposed = false;\n /**\n * Should auto-attach be disabled?\n */\n this.disableAutoAttach = false;\n /**\n * The name of the native xr feature name (like anchor, hit-test, or hand-tracking)\n */\n this.xrNativeFeatureName = \"\";\n }\n /**\n * Is this feature attached\n */\n get attached() {\n return this._attached;\n }\n /**\n * attach this feature\n *\n * @param force should attachment be forced (even when already attached)\n * @returns true if successful, false is failed or already attached\n */\n attach(force) {\n // do not attach a disposed feature\n if (this.isDisposed) {\n return false;\n }\n if (!force) {\n if (this.attached) {\n return false;\n }\n }\n else {\n if (this.attached) {\n // detach first, to be sure\n this.detach();\n }\n }\n this._attached = true;\n this._addNewAttachObserver(this._xrSessionManager.onXRFrameObservable, (frame) => this._onXRFrame(frame));\n return true;\n }\n /**\n * detach this feature.\n *\n * @returns true if successful, false if failed or already detached\n */\n detach() {\n if (!this._attached) {\n this.disableAutoAttach = true;\n return false;\n }\n this._attached = false;\n this._removeOnDetach.forEach((toRemove) => {\n toRemove.observable.remove(toRemove.observer);\n });\n return true;\n }\n /**\n * Dispose this feature and all of the resources attached\n */\n dispose() {\n this.detach();\n this.isDisposed = true;\n }\n /**\n * This function will be executed during before enabling the feature and can be used to not-allow enabling it.\n * Note that at this point the session has NOT started, so this is purely checking if the browser supports it\n *\n * @returns whether or not the feature is compatible in this environment\n */\n isCompatible() {\n return true;\n }\n /**\n * This is used to register callbacks that will automatically be removed when detach is called.\n * @param observable the observable to which the observer will be attached\n * @param callback the callback to register\n */\n _addNewAttachObserver(observable, callback) {\n this._removeOnDetach.push({\n observable,\n observer: observable.add(callback),\n });\n }\n }\n\n /**\n * Renders a layer on top of an existing scene\n */\n class UtilityLayerRenderer {\n /**\n * Gets the camera that is used to render the utility layer (when not set, this will be the last active camera)\n * @param getRigParentIfPossible if the current active camera is a rig camera, should its parent camera be returned\n * @returns the camera that is used when rendering the utility layer\n */\n getRenderCamera(getRigParentIfPossible) {\n if (this._renderCamera) {\n return this._renderCamera;\n }\n else {\n let activeCam;\n if (this.originalScene.activeCameras && this.originalScene.activeCameras.length > 1) {\n activeCam = this.originalScene.activeCameras[this.originalScene.activeCameras.length - 1];\n }\n else {\n activeCam = this.originalScene.activeCamera;\n }\n if (getRigParentIfPossible && activeCam && activeCam.isRigCamera) {\n return activeCam.rigParent;\n }\n return activeCam;\n }\n }\n /**\n * Sets the camera that should be used when rendering the utility layer (If set to null the last active camera will be used)\n * @param cam the camera that should be used when rendering the utility layer\n */\n setRenderCamera(cam) {\n this._renderCamera = cam;\n }\n /**\n * @internal\n * Light which used by gizmos to get light shading\n */\n _getSharedGizmoLight() {\n if (!this._sharedGizmoLight) {\n this._sharedGizmoLight = new HemisphericLight(\"shared gizmo light\", new Vector3(0, 1, 0), this.utilityLayerScene);\n this._sharedGizmoLight.intensity = 2;\n this._sharedGizmoLight.groundColor = Color3.Gray();\n }\n return this._sharedGizmoLight;\n }\n /**\n * A shared utility layer that can be used to overlay objects into a scene (Depth map of the previous scene is cleared before drawing on top of it)\n */\n static get DefaultUtilityLayer() {\n if (UtilityLayerRenderer._DefaultUtilityLayer == null) {\n return UtilityLayerRenderer._CreateDefaultUtilityLayerFromScene(EngineStore.LastCreatedScene);\n }\n return UtilityLayerRenderer._DefaultUtilityLayer;\n }\n /**\n * Creates an utility layer, and set it as a default utility layer\n * @param scene associated scene\n * @internal\n */\n static _CreateDefaultUtilityLayerFromScene(scene) {\n UtilityLayerRenderer._DefaultUtilityLayer = new UtilityLayerRenderer(scene);\n UtilityLayerRenderer._DefaultUtilityLayer.originalScene.onDisposeObservable.addOnce(() => {\n UtilityLayerRenderer._DefaultUtilityLayer = null;\n });\n return UtilityLayerRenderer._DefaultUtilityLayer;\n }\n /**\n * A shared utility layer that can be used to embed objects into a scene (Depth map of the previous scene is not cleared before drawing on top of it)\n */\n static get DefaultKeepDepthUtilityLayer() {\n if (UtilityLayerRenderer._DefaultKeepDepthUtilityLayer == null) {\n UtilityLayerRenderer._DefaultKeepDepthUtilityLayer = new UtilityLayerRenderer(EngineStore.LastCreatedScene);\n UtilityLayerRenderer._DefaultKeepDepthUtilityLayer.utilityLayerScene.autoClearDepthAndStencil = false;\n UtilityLayerRenderer._DefaultKeepDepthUtilityLayer.originalScene.onDisposeObservable.addOnce(() => {\n UtilityLayerRenderer._DefaultKeepDepthUtilityLayer = null;\n });\n }\n return UtilityLayerRenderer._DefaultKeepDepthUtilityLayer;\n }\n /**\n * Instantiates a UtilityLayerRenderer\n * @param originalScene the original scene that will be rendered on top of\n * @param handleEvents boolean indicating if the utility layer should handle events\n */\n constructor(\n /** the original scene that will be rendered on top of */\n originalScene, handleEvents = true) {\n this.originalScene = originalScene;\n this._pointerCaptures = {};\n this._lastPointerEvents = {};\n this._sharedGizmoLight = null;\n this._renderCamera = null;\n /**\n * If the picking should be done on the utility layer prior to the actual scene (Default: true)\n */\n this.pickUtilitySceneFirst = true;\n /**\n * If the utility layer should automatically be rendered on top of existing scene\n */\n this.shouldRender = true;\n /**\n * If set to true, only pointer down onPointerObservable events will be blocked when picking is occluded by original scene\n */\n this.onlyCheckPointerDownEvents = true;\n /**\n * If set to false, only pointerUp, pointerDown and pointerMove will be sent to the utilityLayerScene (false by default)\n */\n this.processAllEvents = false;\n /**\n * Set to false to disable picking\n */\n this.pickingEnabled = true;\n /**\n * Observable raised when the pointer moves from the utility layer scene to the main scene\n */\n this.onPointerOutObservable = new Observable$1();\n // Create scene which will be rendered in the foreground and remove it from being referenced by engine to avoid interfering with existing app\n this.utilityLayerScene = new Scene(originalScene.getEngine(), { virtual: true });\n this.utilityLayerScene.useRightHandedSystem = originalScene.useRightHandedSystem;\n this.utilityLayerScene._allowPostProcessClearColor = false;\n // Deactivate post processes\n this.utilityLayerScene.postProcessesEnabled = false;\n // Detach controls on utility scene, events will be fired by logic below to handle picking priority\n this.utilityLayerScene.detachControl();\n if (handleEvents) {\n this._originalPointerObserver = originalScene.onPrePointerObservable.add((prePointerInfo) => {\n if (!this.utilityLayerScene.activeCamera) {\n return;\n }\n if (!this.pickingEnabled) {\n return;\n }\n if (!this.processAllEvents) {\n if (prePointerInfo.type !== PointerEventTypes.POINTERMOVE &&\n prePointerInfo.type !== PointerEventTypes.POINTERUP &&\n prePointerInfo.type !== PointerEventTypes.POINTERDOWN &&\n prePointerInfo.type !== PointerEventTypes.POINTERDOUBLETAP) {\n return;\n }\n }\n this.utilityLayerScene.pointerX = originalScene.pointerX;\n this.utilityLayerScene.pointerY = originalScene.pointerY;\n const pointerEvent = prePointerInfo.event;\n if (originalScene.isPointerCaptured(pointerEvent.pointerId)) {\n this._pointerCaptures[pointerEvent.pointerId] = false;\n return;\n }\n const getNearPickDataForScene = (scene) => {\n let scenePick = null;\n if (prePointerInfo.nearInteractionPickingInfo) {\n if (prePointerInfo.nearInteractionPickingInfo.pickedMesh.getScene() == scene) {\n scenePick = prePointerInfo.nearInteractionPickingInfo;\n }\n else {\n scenePick = new PickingInfo();\n }\n }\n else if (scene !== this.utilityLayerScene && prePointerInfo.originalPickingInfo) {\n scenePick = prePointerInfo.originalPickingInfo;\n }\n else {\n let previousActiveCamera = null;\n // If a camera is set for rendering with this layer\n // it will also be used for the ray computation\n // To preserve back compat and because scene.pick always use activeCamera\n // it's substituted temporarily and a new scenePick is forced.\n // otherwise, the ray with previously active camera is always used.\n // It's set back to previous activeCamera after operation.\n if (this._renderCamera) {\n previousActiveCamera = scene._activeCamera;\n scene._activeCamera = this._renderCamera;\n prePointerInfo.ray = null;\n }\n scenePick = prePointerInfo.ray ? scene.pickWithRay(prePointerInfo.ray) : scene.pick(originalScene.pointerX, originalScene.pointerY);\n if (previousActiveCamera) {\n scene._activeCamera = previousActiveCamera;\n }\n }\n return scenePick;\n };\n const utilityScenePick = getNearPickDataForScene(this.utilityLayerScene);\n if (!prePointerInfo.ray && utilityScenePick) {\n prePointerInfo.ray = utilityScenePick.ray;\n }\n // always fire the prepointer observable\n this.utilityLayerScene.onPrePointerObservable.notifyObservers(prePointerInfo);\n // allow every non pointer down event to flow to the utility layer\n if (this.onlyCheckPointerDownEvents && prePointerInfo.type != PointerEventTypes.POINTERDOWN) {\n if (!prePointerInfo.skipOnPointerObservable) {\n this.utilityLayerScene.onPointerObservable.notifyObservers(new PointerInfo(prePointerInfo.type, prePointerInfo.event, utilityScenePick), prePointerInfo.type);\n }\n if (prePointerInfo.type === PointerEventTypes.POINTERUP && this._pointerCaptures[pointerEvent.pointerId]) {\n this._pointerCaptures[pointerEvent.pointerId] = false;\n }\n return;\n }\n if (this.utilityLayerScene.autoClearDepthAndStencil || this.pickUtilitySceneFirst) {\n // If this layer is an overlay, check if this layer was hit and if so, skip pointer events for the main scene\n if (utilityScenePick && utilityScenePick.hit) {\n if (!prePointerInfo.skipOnPointerObservable) {\n this.utilityLayerScene.onPointerObservable.notifyObservers(new PointerInfo(prePointerInfo.type, prePointerInfo.event, utilityScenePick), prePointerInfo.type);\n }\n prePointerInfo.skipOnPointerObservable = true;\n }\n }\n else {\n const originalScenePick = getNearPickDataForScene(originalScene);\n const pointerEvent = prePointerInfo.event;\n // If the layer can be occluded by the original scene, only fire pointer events to the first layer that hit they ray\n if (originalScenePick && utilityScenePick) {\n // No pick in utility scene\n if (utilityScenePick.distance === 0 && originalScenePick.pickedMesh) {\n if (this.mainSceneTrackerPredicate && this.mainSceneTrackerPredicate(originalScenePick.pickedMesh)) {\n // We touched an utility mesh present in the main scene\n this._notifyObservers(prePointerInfo, originalScenePick, pointerEvent);\n prePointerInfo.skipOnPointerObservable = true;\n }\n else if (prePointerInfo.type === PointerEventTypes.POINTERDOWN) {\n this._pointerCaptures[pointerEvent.pointerId] = true;\n }\n else if (prePointerInfo.type === PointerEventTypes.POINTERMOVE || prePointerInfo.type === PointerEventTypes.POINTERUP) {\n if (this._lastPointerEvents[pointerEvent.pointerId]) {\n // We need to send a last pointerup to the utilityLayerScene to make sure animations can complete\n this.onPointerOutObservable.notifyObservers(pointerEvent.pointerId);\n delete this._lastPointerEvents[pointerEvent.pointerId];\n }\n this._notifyObservers(prePointerInfo, originalScenePick, pointerEvent);\n }\n }\n else if (!this._pointerCaptures[pointerEvent.pointerId] && (utilityScenePick.distance < originalScenePick.distance || originalScenePick.distance === 0)) {\n // We pick something in utility scene or the pick in utility is closer than the one in main scene\n this._notifyObservers(prePointerInfo, utilityScenePick, pointerEvent);\n // If a previous utility layer set this, do not unset this\n if (!prePointerInfo.skipOnPointerObservable) {\n prePointerInfo.skipOnPointerObservable = utilityScenePick.distance > 0;\n }\n }\n else if (!this._pointerCaptures[pointerEvent.pointerId] && utilityScenePick.distance >= originalScenePick.distance) {\n // We have a pick in both scenes but main is closer than utility\n // We touched an utility mesh present in the main scene\n if (this.mainSceneTrackerPredicate && this.mainSceneTrackerPredicate(originalScenePick.pickedMesh)) {\n this._notifyObservers(prePointerInfo, originalScenePick, pointerEvent);\n prePointerInfo.skipOnPointerObservable = true;\n }\n else {\n if (prePointerInfo.type === PointerEventTypes.POINTERMOVE || prePointerInfo.type === PointerEventTypes.POINTERUP) {\n if (this._lastPointerEvents[pointerEvent.pointerId]) {\n // We need to send a last pointerup to the utilityLayerScene to make sure animations can complete\n this.onPointerOutObservable.notifyObservers(pointerEvent.pointerId);\n delete this._lastPointerEvents[pointerEvent.pointerId];\n }\n }\n this._notifyObservers(prePointerInfo, utilityScenePick, pointerEvent);\n }\n }\n if (prePointerInfo.type === PointerEventTypes.POINTERUP && this._pointerCaptures[pointerEvent.pointerId]) {\n this._pointerCaptures[pointerEvent.pointerId] = false;\n }\n }\n }\n });\n // As a newly added utility layer will be rendered over the screen last, it's pointer events should be processed first\n if (this._originalPointerObserver) {\n originalScene.onPrePointerObservable.makeObserverTopPriority(this._originalPointerObserver);\n }\n }\n // Render directly on top of existing scene without clearing\n this.utilityLayerScene.autoClear = false;\n this._afterRenderObserver = this.originalScene.onAfterRenderCameraObservable.add((camera) => {\n // Only render when the render camera finishes rendering\n if (this.shouldRender && camera == this.getRenderCamera()) {\n this.render();\n }\n });\n this._sceneDisposeObserver = this.originalScene.onDisposeObservable.add(() => {\n this.dispose();\n });\n this._updateCamera();\n }\n _notifyObservers(prePointerInfo, pickInfo, pointerEvent) {\n if (!prePointerInfo.skipOnPointerObservable) {\n this.utilityLayerScene.onPointerObservable.notifyObservers(new PointerInfo(prePointerInfo.type, prePointerInfo.event, pickInfo), prePointerInfo.type);\n this._lastPointerEvents[pointerEvent.pointerId] = true;\n }\n }\n /**\n * Renders the utility layers scene on top of the original scene\n */\n render() {\n this._updateCamera();\n if (this.utilityLayerScene.activeCamera) {\n // Set the camera's scene to utility layers scene\n const oldScene = this.utilityLayerScene.activeCamera.getScene();\n const camera = this.utilityLayerScene.activeCamera;\n camera._scene = this.utilityLayerScene;\n if (camera.leftCamera) {\n camera.leftCamera._scene = this.utilityLayerScene;\n }\n if (camera.rightCamera) {\n camera.rightCamera._scene = this.utilityLayerScene;\n }\n this.utilityLayerScene.render(false);\n // Reset camera's scene back to original\n camera._scene = oldScene;\n if (camera.leftCamera) {\n camera.leftCamera._scene = oldScene;\n }\n if (camera.rightCamera) {\n camera.rightCamera._scene = oldScene;\n }\n }\n }\n /**\n * Disposes of the renderer\n */\n dispose() {\n this.onPointerOutObservable.clear();\n if (this._afterRenderObserver) {\n this.originalScene.onAfterCameraRenderObservable.remove(this._afterRenderObserver);\n }\n if (this._sceneDisposeObserver) {\n this.originalScene.onDisposeObservable.remove(this._sceneDisposeObserver);\n }\n if (this._originalPointerObserver) {\n this.originalScene.onPrePointerObservable.remove(this._originalPointerObserver);\n }\n this.utilityLayerScene.dispose();\n }\n _updateCamera() {\n this.utilityLayerScene.cameraToUseForPointers = this.getRenderCamera();\n this.utilityLayerScene.activeCamera = this.getRenderCamera();\n }\n }\n /** @internal */\n UtilityLayerRenderer._DefaultUtilityLayer = null;\n /** @internal */\n UtilityLayerRenderer._DefaultKeepDepthUtilityLayer = null;\n\n /**\n * A module that will enable pointer selection for motion controllers of XR Input Sources\n */\n class WebXRControllerPointerSelection extends WebXRAbstractFeature {\n /**\n * constructs a new background remover module\n * @param _xrSessionManager the session manager for this module\n * @param _options read-only options to be used in this module\n */\n constructor(_xrSessionManager, _options) {\n super(_xrSessionManager);\n this._options = _options;\n this._attachController = (xrController) => {\n if (this._controllers[xrController.uniqueId]) {\n // already attached\n return;\n }\n const { laserPointer, selectionMesh } = this._generateNewMeshPair(xrController.pointer);\n // get two new meshes\n this._controllers[xrController.uniqueId] = {\n xrController,\n laserPointer,\n selectionMesh,\n meshUnderPointer: null,\n pick: null,\n tmpRay: new Ray(new Vector3(), new Vector3()),\n disabledByNearInteraction: false,\n id: WebXRControllerPointerSelection._IdCounter++,\n };\n if (this._attachedController) {\n if (!this._options.enablePointerSelectionOnAllControllers &&\n this._options.preferredHandedness &&\n xrController.inputSource.handedness === this._options.preferredHandedness) {\n this._attachedController = xrController.uniqueId;\n }\n }\n else {\n if (!this._options.enablePointerSelectionOnAllControllers) {\n this._attachedController = xrController.uniqueId;\n }\n }\n switch (xrController.inputSource.targetRayMode) {\n case \"tracked-pointer\":\n return this._attachTrackedPointerRayMode(xrController);\n case \"gaze\":\n return this._attachGazeMode(xrController);\n case \"screen\":\n return this._attachScreenRayMode(xrController);\n }\n };\n this._controllers = {};\n this._tmpVectorForPickCompare = new Vector3();\n /**\n * Disable lighting on the laser pointer (so it will always be visible)\n */\n this.disablePointerLighting = true;\n /**\n * Disable lighting on the selection mesh (so it will always be visible)\n */\n this.disableSelectionMeshLighting = true;\n /**\n * Should the laser pointer be displayed\n */\n this.displayLaserPointer = true;\n /**\n * Should the selection mesh be displayed (The ring at the end of the laser pointer)\n */\n this.displaySelectionMesh = true;\n /**\n * This color will be set to the laser pointer when selection is triggered\n */\n this.laserPointerPickedColor = new Color3(0.9, 0.9, 0.9);\n /**\n * Default color of the laser pointer\n */\n this.laserPointerDefaultColor = new Color3(0.7, 0.7, 0.7);\n /**\n * default color of the selection ring\n */\n this.selectionMeshDefaultColor = new Color3(0.8, 0.8, 0.8);\n /**\n * This color will be applied to the selection ring when selection is triggered\n */\n this.selectionMeshPickedColor = new Color3(0.3, 0.3, 1.0);\n this._identityMatrix = Matrix.Identity();\n this._screenCoordinatesRef = Vector3.Zero();\n this._viewportRef = new Viewport(0, 0, 0, 0);\n this._scene = this._xrSessionManager.scene;\n }\n /**\n * attach this feature\n * Will usually be called by the features manager\n *\n * @returns true if successful.\n */\n attach() {\n if (!super.attach()) {\n return false;\n }\n this._options.xrInput.controllers.forEach(this._attachController);\n this._addNewAttachObserver(this._options.xrInput.onControllerAddedObservable, this._attachController);\n this._addNewAttachObserver(this._options.xrInput.onControllerRemovedObservable, (controller) => {\n // REMOVE the controller\n this._detachController(controller.uniqueId);\n });\n this._scene.constantlyUpdateMeshUnderPointer = true;\n if (this._options.gazeCamera) {\n const webXRCamera = this._options.gazeCamera;\n const { laserPointer, selectionMesh } = this._generateNewMeshPair(webXRCamera);\n this._controllers[\"camera\"] = {\n webXRCamera,\n laserPointer,\n selectionMesh,\n meshUnderPointer: null,\n pick: null,\n tmpRay: new Ray(new Vector3(), new Vector3()),\n disabledByNearInteraction: false,\n id: WebXRControllerPointerSelection._IdCounter++,\n };\n this._attachGazeMode();\n }\n return true;\n }\n /**\n * detach this feature.\n * Will usually be called by the features manager\n *\n * @returns true if successful.\n */\n detach() {\n if (!super.detach()) {\n return false;\n }\n Object.keys(this._controllers).forEach((controllerId) => {\n this._detachController(controllerId);\n });\n return true;\n }\n /**\n * Will get the mesh under a specific pointer.\n * `scene.meshUnderPointer` will only return one mesh - either left or right.\n * @param controllerId the controllerId to check\n * @returns The mesh under pointer or null if no mesh is under the pointer\n */\n getMeshUnderPointer(controllerId) {\n if (this._controllers[controllerId]) {\n return this._controllers[controllerId].meshUnderPointer;\n }\n else {\n return null;\n }\n }\n /**\n * Get the xr controller that correlates to the pointer id in the pointer event\n *\n * @param id the pointer id to search for\n * @returns the controller that correlates to this id or null if not found\n */\n getXRControllerByPointerId(id) {\n const keys = Object.keys(this._controllers);\n for (let i = 0; i < keys.length; ++i) {\n if (this._controllers[keys[i]].id === id) {\n return this._controllers[keys[i]].xrController || null;\n }\n }\n return null;\n }\n /**\n * @internal\n */\n _getPointerSelectionDisabledByPointerId(id) {\n const keys = Object.keys(this._controllers);\n for (let i = 0; i < keys.length; ++i) {\n if (this._controllers[keys[i]].id === id) {\n return this._controllers[keys[i]].disabledByNearInteraction;\n }\n }\n return true;\n }\n /**\n * @internal\n */\n _setPointerSelectionDisabledByPointerId(id, state) {\n const keys = Object.keys(this._controllers);\n for (let i = 0; i < keys.length; ++i) {\n if (this._controllers[keys[i]].id === id) {\n this._controllers[keys[i]].disabledByNearInteraction = state;\n return;\n }\n }\n }\n _onXRFrame(_xrFrame) {\n Object.keys(this._controllers).forEach((id) => {\n // only do this for the selected pointer\n const controllerData = this._controllers[id];\n if ((!this._options.enablePointerSelectionOnAllControllers && id !== this._attachedController) || controllerData.disabledByNearInteraction) {\n controllerData.selectionMesh.isVisible = false;\n controllerData.laserPointer.isVisible = false;\n controllerData.pick = null;\n return;\n }\n controllerData.laserPointer.isVisible = this.displayLaserPointer;\n let controllerGlobalPosition;\n // Every frame check collisions/input\n if (controllerData.xrController) {\n controllerGlobalPosition = controllerData.xrController.pointer.position;\n controllerData.xrController.getWorldPointerRayToRef(controllerData.tmpRay);\n }\n else if (controllerData.webXRCamera) {\n controllerGlobalPosition = controllerData.webXRCamera.position;\n controllerData.webXRCamera.getForwardRayToRef(controllerData.tmpRay);\n }\n else {\n return;\n }\n if (this._options.maxPointerDistance) {\n controllerData.tmpRay.length = this._options.maxPointerDistance;\n }\n // update pointerX and pointerY of the scene. Only if the flag is set to true!\n if (!this._options.disableScenePointerVectorUpdate && controllerGlobalPosition) {\n const scene = this._xrSessionManager.scene;\n const camera = this._options.xrInput.xrCamera;\n if (camera) {\n camera.viewport.toGlobalToRef(scene.getEngine().getRenderWidth(), scene.getEngine().getRenderHeight(), this._viewportRef);\n Vector3.ProjectToRef(controllerGlobalPosition, this._identityMatrix, scene.getTransformMatrix(), this._viewportRef, this._screenCoordinatesRef);\n // stay safe\n if (typeof this._screenCoordinatesRef.x === \"number\" &&\n typeof this._screenCoordinatesRef.y === \"number\" &&\n !isNaN(this._screenCoordinatesRef.x) &&\n !isNaN(this._screenCoordinatesRef.y)) {\n scene.pointerX = this._screenCoordinatesRef.x;\n scene.pointerY = this._screenCoordinatesRef.y;\n controllerData.screenCoordinates = {\n x: this._screenCoordinatesRef.x,\n y: this._screenCoordinatesRef.y,\n };\n }\n }\n }\n let utilityScenePick = null;\n if (this._utilityLayerScene) {\n utilityScenePick = this._utilityLayerScene.pickWithRay(controllerData.tmpRay, this._utilityLayerScene.pointerMovePredicate || this.raySelectionPredicate);\n }\n const originalScenePick = this._scene.pickWithRay(controllerData.tmpRay, this._scene.pointerMovePredicate || this.raySelectionPredicate);\n if (!utilityScenePick || !utilityScenePick.hit) {\n // No hit in utility scene\n controllerData.pick = originalScenePick;\n }\n else if (!originalScenePick || !originalScenePick.hit) {\n // No hit in original scene\n controllerData.pick = utilityScenePick;\n }\n else if (utilityScenePick.distance < originalScenePick.distance) {\n // Hit is closer in utility scene\n controllerData.pick = utilityScenePick;\n }\n else {\n // Hit is closer in original scene\n controllerData.pick = originalScenePick;\n }\n if (controllerData.pick && controllerData.xrController) {\n controllerData.pick.aimTransform = controllerData.xrController.pointer;\n controllerData.pick.gripTransform = controllerData.xrController.grip || null;\n }\n const pick = controllerData.pick;\n if (pick && pick.pickedPoint && pick.hit) {\n // Update laser state\n this._updatePointerDistance(controllerData.laserPointer, pick.distance);\n // Update cursor state\n controllerData.selectionMesh.position.copyFrom(pick.pickedPoint);\n controllerData.selectionMesh.scaling.x = Math.sqrt(pick.distance);\n controllerData.selectionMesh.scaling.y = Math.sqrt(pick.distance);\n controllerData.selectionMesh.scaling.z = Math.sqrt(pick.distance);\n // To avoid z-fighting\n const pickNormal = this._convertNormalToDirectionOfRay(pick.getNormal(true), controllerData.tmpRay);\n const deltaFighting = 0.001;\n controllerData.selectionMesh.position.copyFrom(pick.pickedPoint);\n if (pickNormal) {\n const axis1 = Vector3.Cross(Axis.Y, pickNormal);\n const axis2 = Vector3.Cross(pickNormal, axis1);\n Vector3.RotationFromAxisToRef(axis2, pickNormal, axis1, controllerData.selectionMesh.rotation);\n controllerData.selectionMesh.position.addInPlace(pickNormal.scale(deltaFighting));\n }\n controllerData.selectionMesh.isVisible = this.displaySelectionMesh;\n controllerData.meshUnderPointer = pick.pickedMesh;\n }\n else {\n controllerData.selectionMesh.isVisible = false;\n this._updatePointerDistance(controllerData.laserPointer, 1);\n controllerData.meshUnderPointer = null;\n }\n });\n }\n get _utilityLayerScene() {\n return this._options.customUtilityLayerScene || UtilityLayerRenderer.DefaultUtilityLayer.utilityLayerScene;\n }\n _attachGazeMode(xrController) {\n const controllerData = this._controllers[(xrController && xrController.uniqueId) || \"camera\"];\n // attached when touched, detaches when raised\n const timeToSelect = this._options.timeToSelect || 3000;\n const sceneToRenderTo = this._options.useUtilityLayer ? this._utilityLayerScene : this._scene;\n let oldPick = new PickingInfo();\n const discMesh = CreateTorus(\"selection\", {\n diameter: 0.0035 * 15,\n thickness: 0.0025 * 6,\n tessellation: 20,\n }, sceneToRenderTo);\n discMesh.isVisible = false;\n discMesh.isPickable = false;\n discMesh.parent = controllerData.selectionMesh;\n let timer = 0;\n let downTriggered = false;\n const pointerEventInit = {\n pointerId: controllerData.id,\n pointerType: \"xr\",\n };\n controllerData.onFrameObserver = this._xrSessionManager.onXRFrameObservable.add(() => {\n if (!controllerData.pick) {\n return;\n }\n this._augmentPointerInit(pointerEventInit, controllerData.id, controllerData.screenCoordinates);\n controllerData.laserPointer.material.alpha = 0;\n discMesh.isVisible = false;\n if (controllerData.pick.hit) {\n if (!this._pickingMoved(oldPick, controllerData.pick)) {\n if (timer > timeToSelect / 10) {\n discMesh.isVisible = true;\n }\n timer += this._scene.getEngine().getDeltaTime();\n if (timer >= timeToSelect) {\n this._scene.simulatePointerDown(controllerData.pick, pointerEventInit);\n // this pointerdown event is not setting the controllerData.pointerDownTriggered to avoid a pointerUp event when this feature is detached\n downTriggered = true;\n // pointer up right after down, if disable on touch out\n if (this._options.disablePointerUpOnTouchOut) {\n this._scene.simulatePointerUp(controllerData.pick, pointerEventInit);\n }\n discMesh.isVisible = false;\n }\n else {\n const scaleFactor = 1 - timer / timeToSelect;\n discMesh.scaling.set(scaleFactor, scaleFactor, scaleFactor);\n }\n }\n else {\n if (downTriggered) {\n if (!this._options.disablePointerUpOnTouchOut) {\n this._scene.simulatePointerUp(controllerData.pick, pointerEventInit);\n }\n }\n downTriggered = false;\n timer = 0;\n }\n }\n else {\n downTriggered = false;\n timer = 0;\n }\n this._scene.simulatePointerMove(controllerData.pick, pointerEventInit);\n oldPick = controllerData.pick;\n });\n if (this._options.renderingGroupId !== undefined) {\n discMesh.renderingGroupId = this._options.renderingGroupId;\n }\n if (xrController) {\n xrController.onDisposeObservable.addOnce(() => {\n if (controllerData.pick && !this._options.disablePointerUpOnTouchOut && downTriggered) {\n this._scene.simulatePointerUp(controllerData.pick, pointerEventInit);\n controllerData.finalPointerUpTriggered = true;\n }\n discMesh.dispose();\n });\n }\n }\n _attachScreenRayMode(xrController) {\n const controllerData = this._controllers[xrController.uniqueId];\n let downTriggered = false;\n const pointerEventInit = {\n pointerId: controllerData.id,\n pointerType: \"xr\",\n };\n controllerData.onFrameObserver = this._xrSessionManager.onXRFrameObservable.add(() => {\n this._augmentPointerInit(pointerEventInit, controllerData.id, controllerData.screenCoordinates);\n if (!controllerData.pick || (this._options.disablePointerUpOnTouchOut && downTriggered)) {\n return;\n }\n if (!downTriggered) {\n this._scene.simulatePointerDown(controllerData.pick, pointerEventInit);\n controllerData.pointerDownTriggered = true;\n downTriggered = true;\n if (this._options.disablePointerUpOnTouchOut) {\n this._scene.simulatePointerUp(controllerData.pick, pointerEventInit);\n }\n }\n else {\n this._scene.simulatePointerMove(controllerData.pick, pointerEventInit);\n }\n });\n xrController.onDisposeObservable.addOnce(() => {\n this._augmentPointerInit(pointerEventInit, controllerData.id, controllerData.screenCoordinates);\n this._xrSessionManager.runInXRFrame(() => {\n if (controllerData.pick && !controllerData.finalPointerUpTriggered && downTriggered && !this._options.disablePointerUpOnTouchOut) {\n this._scene.simulatePointerUp(controllerData.pick, pointerEventInit);\n controllerData.finalPointerUpTriggered = true;\n }\n });\n });\n }\n _attachTrackedPointerRayMode(xrController) {\n const controllerData = this._controllers[xrController.uniqueId];\n if (this._options.forceGazeMode) {\n return this._attachGazeMode(xrController);\n }\n const pointerEventInit = {\n pointerId: controllerData.id,\n pointerType: \"xr\",\n };\n controllerData.onFrameObserver = this._xrSessionManager.onXRFrameObservable.add(() => {\n controllerData.laserPointer.material.disableLighting = this.disablePointerLighting;\n controllerData.selectionMesh.material.disableLighting = this.disableSelectionMeshLighting;\n if (controllerData.pick) {\n this._augmentPointerInit(pointerEventInit, controllerData.id, controllerData.screenCoordinates);\n this._scene.simulatePointerMove(controllerData.pick, pointerEventInit);\n }\n });\n if (xrController.inputSource.gamepad) {\n const init = (motionController) => {\n if (this._options.overrideButtonId) {\n controllerData.selectionComponent = motionController.getComponent(this._options.overrideButtonId);\n }\n if (!controllerData.selectionComponent) {\n controllerData.selectionComponent = motionController.getMainComponent();\n }\n controllerData.onButtonChangedObserver = controllerData.selectionComponent.onButtonStateChangedObservable.add((component) => {\n if (component.changes.pressed) {\n const pressed = component.changes.pressed.current;\n if (controllerData.pick) {\n if (this._options.enablePointerSelectionOnAllControllers || xrController.uniqueId === this._attachedController) {\n this._augmentPointerInit(pointerEventInit, controllerData.id, controllerData.screenCoordinates);\n if (pressed) {\n this._scene.simulatePointerDown(controllerData.pick, pointerEventInit);\n controllerData.pointerDownTriggered = true;\n controllerData.selectionMesh.material.emissiveColor = this.selectionMeshPickedColor;\n controllerData.laserPointer.material.emissiveColor = this.laserPointerPickedColor;\n }\n else {\n this._scene.simulatePointerUp(controllerData.pick, pointerEventInit);\n controllerData.selectionMesh.material.emissiveColor = this.selectionMeshDefaultColor;\n controllerData.laserPointer.material.emissiveColor = this.laserPointerDefaultColor;\n }\n }\n }\n else {\n if (pressed && !this._options.enablePointerSelectionOnAllControllers && !this._options.disableSwitchOnClick) {\n this._attachedController = xrController.uniqueId;\n }\n }\n }\n });\n };\n if (xrController.motionController) {\n init(xrController.motionController);\n }\n else {\n xrController.onMotionControllerInitObservable.add(init);\n }\n }\n else {\n // use the select and squeeze events\n const selectStartListener = (event) => {\n this._augmentPointerInit(pointerEventInit, controllerData.id, controllerData.screenCoordinates);\n if (controllerData.xrController && event.inputSource === controllerData.xrController.inputSource && controllerData.pick) {\n this._scene.simulatePointerDown(controllerData.pick, pointerEventInit);\n controllerData.pointerDownTriggered = true;\n controllerData.selectionMesh.material.emissiveColor = this.selectionMeshPickedColor;\n controllerData.laserPointer.material.emissiveColor = this.laserPointerPickedColor;\n }\n };\n const selectEndListener = (event) => {\n this._augmentPointerInit(pointerEventInit, controllerData.id, controllerData.screenCoordinates);\n if (controllerData.xrController && event.inputSource === controllerData.xrController.inputSource && controllerData.pick) {\n this._scene.simulatePointerUp(controllerData.pick, pointerEventInit);\n controllerData.selectionMesh.material.emissiveColor = this.selectionMeshDefaultColor;\n controllerData.laserPointer.material.emissiveColor = this.laserPointerDefaultColor;\n }\n };\n controllerData.eventListeners = {\n selectend: selectEndListener,\n selectstart: selectStartListener,\n };\n this._xrSessionManager.session.addEventListener(\"selectstart\", selectStartListener);\n this._xrSessionManager.session.addEventListener(\"selectend\", selectEndListener);\n }\n }\n _convertNormalToDirectionOfRay(normal, ray) {\n if (normal) {\n const angle = Math.acos(Vector3.Dot(normal, ray.direction));\n if (angle < Math.PI / 2) {\n normal.scaleInPlace(-1);\n }\n }\n return normal;\n }\n _detachController(xrControllerUniqueId) {\n const controllerData = this._controllers[xrControllerUniqueId];\n if (!controllerData) {\n return;\n }\n if (controllerData.selectionComponent) {\n if (controllerData.onButtonChangedObserver) {\n controllerData.selectionComponent.onButtonStateChangedObservable.remove(controllerData.onButtonChangedObserver);\n }\n }\n if (controllerData.onFrameObserver) {\n this._xrSessionManager.onXRFrameObservable.remove(controllerData.onFrameObserver);\n }\n if (controllerData.eventListeners) {\n Object.keys(controllerData.eventListeners).forEach((eventName) => {\n const func = controllerData.eventListeners && controllerData.eventListeners[eventName];\n if (func) {\n // For future reference - this is an issue in the WebXR typings.\n this._xrSessionManager.session.removeEventListener(eventName, func);\n }\n });\n }\n if (!controllerData.finalPointerUpTriggered && controllerData.pointerDownTriggered) {\n // Stay safe and fire a pointerup, in case it wasn't already triggered\n const pointerEventInit = {\n pointerId: controllerData.id,\n pointerType: \"xr\",\n };\n this._xrSessionManager.runInXRFrame(() => {\n this._augmentPointerInit(pointerEventInit, controllerData.id, controllerData.screenCoordinates);\n this._scene.simulatePointerUp(controllerData.pick || new PickingInfo(), pointerEventInit);\n controllerData.finalPointerUpTriggered = true;\n });\n }\n this._xrSessionManager.scene.onBeforeRenderObservable.addOnce(() => {\n try {\n controllerData.selectionMesh.dispose();\n controllerData.laserPointer.dispose();\n // remove from the map\n delete this._controllers[xrControllerUniqueId];\n if (this._attachedController === xrControllerUniqueId) {\n // check for other controllers\n const keys = Object.keys(this._controllers);\n if (keys.length) {\n this._attachedController = keys[0];\n }\n else {\n this._attachedController = \"\";\n }\n }\n }\n catch (e) {\n Tools.Warn(\"controller already detached.\");\n }\n });\n }\n _generateNewMeshPair(meshParent) {\n const sceneToRenderTo = this._options.useUtilityLayer ? this._options.customUtilityLayerScene || UtilityLayerRenderer.DefaultUtilityLayer.utilityLayerScene : this._scene;\n const laserPointer = this._options.customLasterPointerMeshGenerator\n ? this._options.customLasterPointerMeshGenerator()\n : CreateCylinder(\"laserPointer\", {\n height: 1,\n diameterTop: 0.0002,\n diameterBottom: 0.004,\n tessellation: 20,\n subdivisions: 1,\n }, sceneToRenderTo);\n laserPointer.parent = meshParent;\n const laserPointerMaterial = new StandardMaterial(\"laserPointerMat\", sceneToRenderTo);\n laserPointerMaterial.emissiveColor = this.laserPointerDefaultColor;\n laserPointerMaterial.alpha = 0.7;\n laserPointer.material = laserPointerMaterial;\n laserPointer.rotation.x = Math.PI / 2;\n this._updatePointerDistance(laserPointer, 1);\n laserPointer.isPickable = false;\n laserPointer.isVisible = false;\n // Create a gaze tracker for the XR controller\n const selectionMesh = this._options.customSelectionMeshGenerator\n ? this._options.customSelectionMeshGenerator()\n : CreateTorus(\"gazeTracker\", {\n diameter: 0.0035 * 3,\n thickness: 0.0025 * 3,\n tessellation: 20,\n }, sceneToRenderTo);\n selectionMesh.bakeCurrentTransformIntoVertices();\n selectionMesh.isPickable = false;\n selectionMesh.isVisible = false;\n const targetMat = new StandardMaterial(\"targetMat\", sceneToRenderTo);\n targetMat.specularColor = Color3.Black();\n targetMat.emissiveColor = this.selectionMeshDefaultColor;\n targetMat.backFaceCulling = false;\n selectionMesh.material = targetMat;\n if (this._options.renderingGroupId !== undefined) {\n laserPointer.renderingGroupId = this._options.renderingGroupId;\n selectionMesh.renderingGroupId = this._options.renderingGroupId;\n }\n return {\n laserPointer,\n selectionMesh,\n };\n }\n _pickingMoved(oldPick, newPick) {\n var _a;\n if (!oldPick.hit || !newPick.hit) {\n return true;\n }\n if (!oldPick.pickedMesh || !oldPick.pickedPoint || !newPick.pickedMesh || !newPick.pickedPoint) {\n return true;\n }\n if (oldPick.pickedMesh !== newPick.pickedMesh) {\n return true;\n }\n (_a = oldPick.pickedPoint) === null || _a === void 0 ? void 0 : _a.subtractToRef(newPick.pickedPoint, this._tmpVectorForPickCompare);\n this._tmpVectorForPickCompare.set(Math.abs(this._tmpVectorForPickCompare.x), Math.abs(this._tmpVectorForPickCompare.y), Math.abs(this._tmpVectorForPickCompare.z));\n const delta = (this._options.gazeModePointerMovedFactor || 1) * 0.01 * newPick.distance;\n const length = this._tmpVectorForPickCompare.length();\n if (length > delta) {\n return true;\n }\n return false;\n }\n _updatePointerDistance(_laserPointer, distance = 100) {\n _laserPointer.scaling.y = distance;\n // a bit of distance from the controller\n if (this._scene.useRightHandedSystem) {\n distance *= -1;\n }\n _laserPointer.position.z = distance / 2 + 0.05;\n }\n _augmentPointerInit(pointerEventInit, id, screenCoordinates) {\n pointerEventInit.pointerId = id;\n pointerEventInit.pointerType = \"xr\";\n if (screenCoordinates) {\n pointerEventInit.screenX = screenCoordinates.x;\n pointerEventInit.screenY = screenCoordinates.y;\n }\n }\n /** @internal */\n get lasterPointerDefaultColor() {\n // here due to a typo\n return this.laserPointerDefaultColor;\n }\n }\n WebXRControllerPointerSelection._IdCounter = 200;\n /**\n * The module's name\n */\n WebXRControllerPointerSelection.Name = WebXRFeatureName.POINTER_SELECTION;\n /**\n * The (Babylon) version of this module.\n * This is an integer representing the implementation version.\n * This number does not correspond to the WebXR specs version\n */\n WebXRControllerPointerSelection.Version = 1;\n //register the plugin\n WebXRFeaturesManager.AddWebXRFeature(WebXRControllerPointerSelection.Name, (xrSessionManager, options) => {\n return () => new WebXRControllerPointerSelection(xrSessionManager, options);\n }, WebXRControllerPointerSelection.Version, true);\n\n /**\n * Defines the kind of connection point for node based material\n */\n var NodeMaterialBlockConnectionPointTypes;\n (function (NodeMaterialBlockConnectionPointTypes) {\n /** Float */\n NodeMaterialBlockConnectionPointTypes[NodeMaterialBlockConnectionPointTypes[\"Float\"] = 1] = \"Float\";\n /** Int */\n NodeMaterialBlockConnectionPointTypes[NodeMaterialBlockConnectionPointTypes[\"Int\"] = 2] = \"Int\";\n /** Vector2 */\n NodeMaterialBlockConnectionPointTypes[NodeMaterialBlockConnectionPointTypes[\"Vector2\"] = 4] = \"Vector2\";\n /** Vector3 */\n NodeMaterialBlockConnectionPointTypes[NodeMaterialBlockConnectionPointTypes[\"Vector3\"] = 8] = \"Vector3\";\n /** Vector4 */\n NodeMaterialBlockConnectionPointTypes[NodeMaterialBlockConnectionPointTypes[\"Vector4\"] = 16] = \"Vector4\";\n /** Color3 */\n NodeMaterialBlockConnectionPointTypes[NodeMaterialBlockConnectionPointTypes[\"Color3\"] = 32] = \"Color3\";\n /** Color4 */\n NodeMaterialBlockConnectionPointTypes[NodeMaterialBlockConnectionPointTypes[\"Color4\"] = 64] = \"Color4\";\n /** Matrix */\n NodeMaterialBlockConnectionPointTypes[NodeMaterialBlockConnectionPointTypes[\"Matrix\"] = 128] = \"Matrix\";\n /** Custom object */\n NodeMaterialBlockConnectionPointTypes[NodeMaterialBlockConnectionPointTypes[\"Object\"] = 256] = \"Object\";\n /** Detect type based on connection */\n NodeMaterialBlockConnectionPointTypes[NodeMaterialBlockConnectionPointTypes[\"AutoDetect\"] = 1024] = \"AutoDetect\";\n /** Output type that will be defined by input type */\n NodeMaterialBlockConnectionPointTypes[NodeMaterialBlockConnectionPointTypes[\"BasedOnInput\"] = 2048] = \"BasedOnInput\";\n /** Bitmask of all types */\n NodeMaterialBlockConnectionPointTypes[NodeMaterialBlockConnectionPointTypes[\"All\"] = 4095] = \"All\";\n })(NodeMaterialBlockConnectionPointTypes || (NodeMaterialBlockConnectionPointTypes = {}));\n\n /**\n * Enum used to define the target of a block\n */\n var NodeMaterialBlockTargets;\n (function (NodeMaterialBlockTargets) {\n /** Vertex shader */\n NodeMaterialBlockTargets[NodeMaterialBlockTargets[\"Vertex\"] = 1] = \"Vertex\";\n /** Fragment shader */\n NodeMaterialBlockTargets[NodeMaterialBlockTargets[\"Fragment\"] = 2] = \"Fragment\";\n /** Neutral */\n NodeMaterialBlockTargets[NodeMaterialBlockTargets[\"Neutral\"] = 4] = \"Neutral\";\n /** Vertex and Fragment */\n NodeMaterialBlockTargets[NodeMaterialBlockTargets[\"VertexAndFragment\"] = 3] = \"VertexAndFragment\";\n })(NodeMaterialBlockTargets || (NodeMaterialBlockTargets = {}));\n\n /**\n * Class used to store node based material build state\n */\n class NodeMaterialBuildState {\n constructor() {\n /** Gets or sets a boolean indicating if the current state can emit uniform buffers */\n this.supportUniformBuffers = false;\n /**\n * Gets the list of emitted attributes\n */\n this.attributes = new Array();\n /**\n * Gets the list of emitted uniforms\n */\n this.uniforms = new Array();\n /**\n * Gets the list of emitted constants\n */\n this.constants = new Array();\n /**\n * Gets the list of emitted samplers\n */\n this.samplers = new Array();\n /**\n * Gets the list of emitted functions\n */\n this.functions = {};\n /**\n * Gets the list of emitted extensions\n */\n this.extensions = {};\n /**\n * Gets the list of emitted counters\n */\n this.counters = {};\n /** @internal */\n this._attributeDeclaration = \"\";\n /** @internal */\n this._uniformDeclaration = \"\";\n /** @internal */\n this._constantDeclaration = \"\";\n /** @internal */\n this._samplerDeclaration = \"\";\n /** @internal */\n this._varyingTransfer = \"\";\n /** @internal */\n this._injectAtEnd = \"\";\n this._repeatableContentAnchorIndex = 0;\n /** @internal */\n this._builtCompilationString = \"\";\n /**\n * Gets the emitted compilation strings\n */\n this.compilationString = \"\";\n }\n /**\n * Finalize the compilation strings\n * @param state defines the current compilation state\n */\n finalize(state) {\n const emitComments = state.sharedData.emitComments;\n const isFragmentMode = this.target === NodeMaterialBlockTargets.Fragment;\n this.compilationString = `\\n${emitComments ? \"//Entry point\\n\" : \"\"}void main(void) {\\n${this.compilationString}`;\n if (this._constantDeclaration) {\n this.compilationString = `\\n${emitComments ? \"//Constants\\n\" : \"\"}${this._constantDeclaration}\\n${this.compilationString}`;\n }\n let functionCode = \"\";\n for (const functionName in this.functions) {\n functionCode += this.functions[functionName] + `\\n`;\n }\n this.compilationString = `\\n${functionCode}\\n${this.compilationString}`;\n if (!isFragmentMode && this._varyingTransfer) {\n this.compilationString = `${this.compilationString}\\n${this._varyingTransfer}`;\n }\n if (this._injectAtEnd) {\n this.compilationString = `${this.compilationString}\\n${this._injectAtEnd}`;\n }\n this.compilationString = `${this.compilationString}\\n}`;\n if (this.sharedData.varyingDeclaration) {\n this.compilationString = `\\n${emitComments ? \"//Varyings\\n\" : \"\"}${this.sharedData.varyingDeclaration}\\n${this.compilationString}`;\n }\n if (this._samplerDeclaration) {\n this.compilationString = `\\n${emitComments ? \"//Samplers\\n\" : \"\"}${this._samplerDeclaration}\\n${this.compilationString}`;\n }\n if (this._uniformDeclaration) {\n this.compilationString = `\\n${emitComments ? \"//Uniforms\\n\" : \"\"}${this._uniformDeclaration}\\n${this.compilationString}`;\n }\n if (this._attributeDeclaration && !isFragmentMode) {\n this.compilationString = `\\n${emitComments ? \"//Attributes\\n\" : \"\"}${this._attributeDeclaration}\\n${this.compilationString}`;\n }\n this.compilationString = \"precision highp float;\\n\" + this.compilationString;\n this.compilationString = \"#if defined(WEBGL2) || defines(WEBGPU)\\nprecision highp sampler2DArray;\\n#endif\\n\" + this.compilationString;\n for (const extensionName in this.extensions) {\n const extension = this.extensions[extensionName];\n this.compilationString = `\\n${extension}\\n${this.compilationString}`;\n }\n this._builtCompilationString = this.compilationString;\n }\n /** @internal */\n get _repeatableContentAnchor() {\n return `###___ANCHOR${this._repeatableContentAnchorIndex++}___###`;\n }\n /**\n * @internal\n */\n _getFreeVariableName(prefix) {\n prefix = prefix.replace(/[^a-zA-Z_]+/g, \"\");\n if (this.sharedData.variableNames[prefix] === undefined) {\n this.sharedData.variableNames[prefix] = 0;\n // Check reserved words\n if (prefix === \"output\" || prefix === \"texture\") {\n return prefix + this.sharedData.variableNames[prefix];\n }\n return prefix;\n }\n else {\n this.sharedData.variableNames[prefix]++;\n }\n return prefix + this.sharedData.variableNames[prefix];\n }\n /**\n * @internal\n */\n _getFreeDefineName(prefix) {\n if (this.sharedData.defineNames[prefix] === undefined) {\n this.sharedData.defineNames[prefix] = 0;\n }\n else {\n this.sharedData.defineNames[prefix]++;\n }\n return prefix + this.sharedData.defineNames[prefix];\n }\n /**\n * @internal\n */\n _excludeVariableName(name) {\n this.sharedData.variableNames[name] = 0;\n }\n /**\n * @internal\n */\n _emit2DSampler(name) {\n if (this.samplers.indexOf(name) < 0) {\n this._samplerDeclaration += `uniform sampler2D ${name};\\n`;\n this.samplers.push(name);\n }\n }\n /**\n * @internal\n */\n _emit2DArraySampler(name) {\n if (this.samplers.indexOf(name) < 0) {\n this._samplerDeclaration += `uniform sampler2DArray ${name};\\n`;\n this.samplers.push(name);\n }\n }\n /**\n * @internal\n */\n _getGLType(type) {\n switch (type) {\n case NodeMaterialBlockConnectionPointTypes.Float:\n return \"float\";\n case NodeMaterialBlockConnectionPointTypes.Int:\n return \"int\";\n case NodeMaterialBlockConnectionPointTypes.Vector2:\n return \"vec2\";\n case NodeMaterialBlockConnectionPointTypes.Color3:\n case NodeMaterialBlockConnectionPointTypes.Vector3:\n return \"vec3\";\n case NodeMaterialBlockConnectionPointTypes.Color4:\n case NodeMaterialBlockConnectionPointTypes.Vector4:\n return \"vec4\";\n case NodeMaterialBlockConnectionPointTypes.Matrix:\n return \"mat4\";\n }\n return \"\";\n }\n /**\n * @internal\n */\n _emitExtension(name, extension, define = \"\") {\n if (this.extensions[name]) {\n return;\n }\n if (define) {\n extension = `#if ${define}\\n${extension}\\n#endif`;\n }\n this.extensions[name] = extension;\n }\n /**\n * @internal\n */\n _emitFunction(name, code, comments) {\n if (this.functions[name]) {\n return;\n }\n if (this.sharedData.emitComments) {\n code = comments + `\\n` + code;\n }\n this.functions[name] = code;\n }\n /**\n * @internal\n */\n _emitCodeFromInclude(includeName, comments, options) {\n if (options && options.repeatKey) {\n return `#include<${includeName}>${options.substitutionVars ? \"(\" + options.substitutionVars + \")\" : \"\"}[0..${options.repeatKey}]\\n`;\n }\n let code = Effect.IncludesShadersStore[includeName] + \"\\n\";\n if (this.sharedData.emitComments) {\n code = comments + `\\n` + code;\n }\n if (!options) {\n return code;\n }\n if (options.replaceStrings) {\n for (let index = 0; index < options.replaceStrings.length; index++) {\n const replaceString = options.replaceStrings[index];\n code = code.replace(replaceString.search, replaceString.replace);\n }\n }\n return code;\n }\n /**\n * @internal\n */\n _emitFunctionFromInclude(includeName, comments, options, storeKey = \"\") {\n const key = includeName + storeKey;\n if (this.functions[key]) {\n return;\n }\n if (!options || (!options.removeAttributes && !options.removeUniforms && !options.removeVaryings && !options.removeIfDef && !options.replaceStrings)) {\n if (options && options.repeatKey) {\n this.functions[key] = `#include<${includeName}>${options.substitutionVars ? \"(\" + options.substitutionVars + \")\" : \"\"}[0..${options.repeatKey}]\\n`;\n }\n else {\n this.functions[key] = `#include<${includeName}>${(options === null || options === void 0 ? void 0 : options.substitutionVars) ? \"(\" + (options === null || options === void 0 ? void 0 : options.substitutionVars) + \")\" : \"\"}\\n`;\n }\n if (this.sharedData.emitComments) {\n this.functions[key] = comments + `\\n` + this.functions[key];\n }\n return;\n }\n this.functions[key] = Effect.IncludesShadersStore[includeName];\n if (this.sharedData.emitComments) {\n this.functions[key] = comments + `\\n` + this.functions[key];\n }\n if (options.removeIfDef) {\n this.functions[key] = this.functions[key].replace(/^\\s*?#ifdef.+$/gm, \"\");\n this.functions[key] = this.functions[key].replace(/^\\s*?#endif.*$/gm, \"\");\n this.functions[key] = this.functions[key].replace(/^\\s*?#else.*$/gm, \"\");\n this.functions[key] = this.functions[key].replace(/^\\s*?#elif.*$/gm, \"\");\n }\n if (options.removeAttributes) {\n this.functions[key] = this.functions[key].replace(/\\s*?attribute .+?;/g, \"\\n\");\n }\n if (options.removeUniforms) {\n this.functions[key] = this.functions[key].replace(/\\s*?uniform .*?;/g, \"\\n\");\n }\n if (options.removeVaryings) {\n this.functions[key] = this.functions[key].replace(/\\s*?(varying|in) .+?;/g, \"\\n\");\n }\n if (options.replaceStrings) {\n for (let index = 0; index < options.replaceStrings.length; index++) {\n const replaceString = options.replaceStrings[index];\n this.functions[key] = this.functions[key].replace(replaceString.search, replaceString.replace);\n }\n }\n }\n /**\n * @internal\n */\n _registerTempVariable(name) {\n if (this.sharedData.temps.indexOf(name) !== -1) {\n return false;\n }\n this.sharedData.temps.push(name);\n return true;\n }\n /**\n * @internal\n */\n _emitVaryingFromString(name, type, define = \"\", notDefine = false) {\n if (this.sharedData.varyings.indexOf(name) !== -1) {\n return false;\n }\n this.sharedData.varyings.push(name);\n if (define) {\n if (define.startsWith(\"defined(\")) {\n this.sharedData.varyingDeclaration += `#if ${define}\\n`;\n }\n else {\n this.sharedData.varyingDeclaration += `${notDefine ? \"#ifndef\" : \"#ifdef\"} ${define}\\n`;\n }\n }\n this.sharedData.varyingDeclaration += `varying ${type} ${name};\\n`;\n if (define) {\n this.sharedData.varyingDeclaration += `#endif\\n`;\n }\n return true;\n }\n /**\n * @internal\n */\n _emitUniformFromString(name, type, define = \"\", notDefine = false) {\n if (this.uniforms.indexOf(name) !== -1) {\n return;\n }\n this.uniforms.push(name);\n if (define) {\n if (define.startsWith(\"defined(\")) {\n this._uniformDeclaration += `#if ${define}\\n`;\n }\n else {\n this._uniformDeclaration += `${notDefine ? \"#ifndef\" : \"#ifdef\"} ${define}\\n`;\n }\n }\n this._uniformDeclaration += `uniform ${type} ${name};\\n`;\n if (define) {\n this._uniformDeclaration += `#endif\\n`;\n }\n }\n /**\n * @internal\n */\n _emitFloat(value) {\n if (value.toString() === value.toFixed(0)) {\n return `${value}.0`;\n }\n return value.toString();\n }\n }\n\n /**\n * Class used to store shared data between 2 NodeMaterialBuildState\n */\n class NodeMaterialBuildStateSharedData {\n /** Creates a new shared data */\n constructor() {\n /**\n * Gets the list of emitted varyings\n */\n this.temps = new Array();\n /**\n * Gets the list of emitted varyings\n */\n this.varyings = new Array();\n /**\n * Gets the varying declaration string\n */\n this.varyingDeclaration = \"\";\n /**\n * Input blocks\n */\n this.inputBlocks = new Array();\n /**\n * Input blocks\n */\n this.textureBlocks = new Array();\n /**\n * Bindable blocks (Blocks that need to set data to the effect)\n */\n this.bindableBlocks = new Array();\n /**\n * Bindable blocks (Blocks that need to set data to the effect) that will always be called (by bindForSubMesh), contrary to bindableBlocks that won't be called if _mustRebind() returns false\n */\n this.forcedBindableBlocks = new Array();\n /**\n * List of blocks that can provide a compilation fallback\n */\n this.blocksWithFallbacks = new Array();\n /**\n * List of blocks that can provide a define update\n */\n this.blocksWithDefines = new Array();\n /**\n * List of blocks that can provide a repeatable content\n */\n this.repeatableContentBlocks = new Array();\n /**\n * List of blocks that can provide a dynamic list of uniforms\n */\n this.dynamicUniformBlocks = new Array();\n /**\n * List of blocks that can block the isReady function for the material\n */\n this.blockingBlocks = new Array();\n /**\n * Gets the list of animated inputs\n */\n this.animatedInputs = new Array();\n /** List of emitted variables */\n this.variableNames = {};\n /** List of emitted defines */\n this.defineNames = {};\n /**\n * Gets the compilation hints emitted at compilation time\n */\n this.hints = {\n needWorldViewMatrix: false,\n needWorldViewProjectionMatrix: false,\n needAlphaBlending: false,\n needAlphaTesting: false,\n };\n /**\n * List of compilation checks\n */\n this.checks = {\n emitVertex: false,\n emitFragment: false,\n notConnectedNonOptionalInputs: new Array(),\n };\n /**\n * Is vertex program allowed to be empty?\n */\n this.allowEmptyVertexProgram = false;\n // Exclude usual attributes from free variable names\n this.variableNames[\"position\"] = 0;\n this.variableNames[\"normal\"] = 0;\n this.variableNames[\"tangent\"] = 0;\n this.variableNames[\"uv\"] = 0;\n this.variableNames[\"uv2\"] = 0;\n this.variableNames[\"uv3\"] = 0;\n this.variableNames[\"uv4\"] = 0;\n this.variableNames[\"uv5\"] = 0;\n this.variableNames[\"uv6\"] = 0;\n this.variableNames[\"color\"] = 0;\n this.variableNames[\"matricesIndices\"] = 0;\n this.variableNames[\"matricesWeights\"] = 0;\n this.variableNames[\"matricesIndicesExtra\"] = 0;\n this.variableNames[\"matricesWeightsExtra\"] = 0;\n this.variableNames[\"diffuseBase\"] = 0;\n this.variableNames[\"specularBase\"] = 0;\n this.variableNames[\"worldPos\"] = 0;\n this.variableNames[\"shadow\"] = 0;\n this.variableNames[\"view\"] = 0;\n // Exclude known varyings\n this.variableNames[\"vTBN\"] = 0;\n // Exclude defines\n this.defineNames[\"MAINUV0\"] = 0;\n this.defineNames[\"MAINUV1\"] = 0;\n this.defineNames[\"MAINUV2\"] = 0;\n this.defineNames[\"MAINUV3\"] = 0;\n this.defineNames[\"MAINUV4\"] = 0;\n this.defineNames[\"MAINUV5\"] = 0;\n this.defineNames[\"MAINUV6\"] = 0;\n this.defineNames[\"MAINUV7\"] = 0;\n }\n /**\n * Emits console errors and exceptions if there is a failing check\n */\n emitErrors() {\n let errorMessage = \"\";\n if (!this.checks.emitVertex && !this.allowEmptyVertexProgram) {\n errorMessage += \"NodeMaterial does not have a vertex output. You need to at least add a block that generates a glPosition value.\\n\";\n }\n if (!this.checks.emitFragment) {\n errorMessage += \"NodeMaterial does not have a fragment output. You need to at least add a block that generates a glFragColor value.\\n\";\n }\n for (const notConnectedInput of this.checks.notConnectedNonOptionalInputs) {\n errorMessage += `input ${notConnectedInput.name} from block ${notConnectedInput.ownerBlock.name}[${notConnectedInput.ownerBlock.getClassName()}] is not connected and is not optional.\\n`;\n }\n if (errorMessage) {\n throw \"Build of NodeMaterial failed:\\n\" + errorMessage;\n }\n }\n }\n\n /**\n * Enum used to define the compatibility state between two connection points\n */\n var NodeMaterialConnectionPointCompatibilityStates;\n (function (NodeMaterialConnectionPointCompatibilityStates) {\n /** Points are compatibles */\n NodeMaterialConnectionPointCompatibilityStates[NodeMaterialConnectionPointCompatibilityStates[\"Compatible\"] = 0] = \"Compatible\";\n /** Points are incompatible because of their types */\n NodeMaterialConnectionPointCompatibilityStates[NodeMaterialConnectionPointCompatibilityStates[\"TypeIncompatible\"] = 1] = \"TypeIncompatible\";\n /** Points are incompatible because of their targets (vertex vs fragment) */\n NodeMaterialConnectionPointCompatibilityStates[NodeMaterialConnectionPointCompatibilityStates[\"TargetIncompatible\"] = 2] = \"TargetIncompatible\";\n /** Points are incompatible because they are in the same hierarchy **/\n NodeMaterialConnectionPointCompatibilityStates[NodeMaterialConnectionPointCompatibilityStates[\"HierarchyIssue\"] = 3] = \"HierarchyIssue\";\n })(NodeMaterialConnectionPointCompatibilityStates || (NodeMaterialConnectionPointCompatibilityStates = {}));\n /**\n * Defines the direction of a connection point\n */\n var NodeMaterialConnectionPointDirection;\n (function (NodeMaterialConnectionPointDirection) {\n /** Input */\n NodeMaterialConnectionPointDirection[NodeMaterialConnectionPointDirection[\"Input\"] = 0] = \"Input\";\n /** Output */\n NodeMaterialConnectionPointDirection[NodeMaterialConnectionPointDirection[\"Output\"] = 1] = \"Output\";\n })(NodeMaterialConnectionPointDirection || (NodeMaterialConnectionPointDirection = {}));\n /**\n * Defines a connection point for a block\n */\n class NodeMaterialConnectionPoint {\n /**\n * Checks if two types are equivalent\n * @param type1 type 1 to check\n * @param type2 type 2 to check\n * @returns true if both types are equivalent, else false\n */\n static AreEquivalentTypes(type1, type2) {\n switch (type1) {\n case NodeMaterialBlockConnectionPointTypes.Vector3: {\n if (type2 === NodeMaterialBlockConnectionPointTypes.Color3) {\n return true;\n }\n break;\n }\n case NodeMaterialBlockConnectionPointTypes.Vector4: {\n if (type2 === NodeMaterialBlockConnectionPointTypes.Color4) {\n return true;\n }\n break;\n }\n case NodeMaterialBlockConnectionPointTypes.Color3: {\n if (type2 === NodeMaterialBlockConnectionPointTypes.Vector3) {\n return true;\n }\n break;\n }\n case NodeMaterialBlockConnectionPointTypes.Color4: {\n if (type2 === NodeMaterialBlockConnectionPointTypes.Vector4) {\n return true;\n }\n break;\n }\n }\n return false;\n }\n /** Gets the direction of the point */\n get direction() {\n return this._direction;\n }\n /**\n * Gets or sets the associated variable name in the shader\n */\n get associatedVariableName() {\n if (this._ownerBlock.isInput) {\n return this._ownerBlock.associatedVariableName;\n }\n if ((!this._enforceAssociatedVariableName || !this._associatedVariableName) && this._connectedPoint) {\n return this._connectedPoint.associatedVariableName;\n }\n return this._associatedVariableName;\n }\n set associatedVariableName(value) {\n this._associatedVariableName = value;\n }\n /** Get the inner type (ie AutoDetect for instance instead of the inferred one) */\n get innerType() {\n if (this._linkedConnectionSource && this._linkedConnectionSource.isConnected) {\n return this.type;\n }\n return this._type;\n }\n /**\n * Gets or sets the connection point type (default is float)\n */\n get type() {\n if (this._type === NodeMaterialBlockConnectionPointTypes.AutoDetect) {\n if (this._ownerBlock.isInput) {\n return this._ownerBlock.type;\n }\n if (this._connectedPoint) {\n return this._connectedPoint.type;\n }\n if (this._linkedConnectionSource && this._linkedConnectionSource.isConnected) {\n return this._linkedConnectionSource.type;\n }\n }\n if (this._type === NodeMaterialBlockConnectionPointTypes.BasedOnInput) {\n if (this._typeConnectionSource) {\n if (!this._typeConnectionSource.isConnected && this._defaultConnectionPointType) {\n return this._defaultConnectionPointType;\n }\n return this._typeConnectionSource.type;\n }\n else if (this._defaultConnectionPointType) {\n return this._defaultConnectionPointType;\n }\n }\n return this._type;\n }\n set type(value) {\n this._type = value;\n }\n /** Gets or sets the target of that connection point */\n get target() {\n if (!this._prioritizeVertex || !this._ownerBlock) {\n return this._target;\n }\n if (this._target !== NodeMaterialBlockTargets.VertexAndFragment) {\n return this._target;\n }\n if (this._ownerBlock.target === NodeMaterialBlockTargets.Fragment) {\n return NodeMaterialBlockTargets.Fragment;\n }\n return NodeMaterialBlockTargets.Vertex;\n }\n set target(value) {\n this._target = value;\n }\n /**\n * Gets a boolean indicating that the current point is connected to another NodeMaterialBlock\n */\n get isConnected() {\n return this.connectedPoint !== null || this.hasEndpoints;\n }\n /**\n * Gets a boolean indicating that the current point is connected to an input block\n */\n get isConnectedToInputBlock() {\n return this.connectedPoint !== null && this.connectedPoint.ownerBlock.isInput;\n }\n /**\n * Gets a the connected input block (if any)\n */\n get connectInputBlock() {\n if (!this.isConnectedToInputBlock) {\n return null;\n }\n return this.connectedPoint.ownerBlock;\n }\n /** Get the other side of the connection (if any) */\n get connectedPoint() {\n return this._connectedPoint;\n }\n /** Get the block that owns this connection point */\n get ownerBlock() {\n return this._ownerBlock;\n }\n /** Get the block connected on the other side of this connection (if any) */\n get sourceBlock() {\n if (!this._connectedPoint) {\n return null;\n }\n return this._connectedPoint.ownerBlock;\n }\n /** Get the block connected on the endpoints of this connection (if any) */\n get connectedBlocks() {\n if (this._endpoints.length === 0) {\n return [];\n }\n return this._endpoints.map((e) => e.ownerBlock);\n }\n /** Gets the list of connected endpoints */\n get endpoints() {\n return this._endpoints;\n }\n /** Gets a boolean indicating if that output point is connected to at least one input */\n get hasEndpoints() {\n return this._endpoints && this._endpoints.length > 0;\n }\n /** Gets a boolean indicating that this connection has a path to the vertex output*/\n get isDirectlyConnectedToVertexOutput() {\n if (!this.hasEndpoints) {\n return false;\n }\n for (const endpoint of this._endpoints) {\n if (endpoint.ownerBlock.target === NodeMaterialBlockTargets.Vertex) {\n return true;\n }\n if (endpoint.ownerBlock.target === NodeMaterialBlockTargets.Neutral || endpoint.ownerBlock.target === NodeMaterialBlockTargets.VertexAndFragment) {\n if (endpoint.ownerBlock.outputs.some((o) => o.isDirectlyConnectedToVertexOutput)) {\n return true;\n }\n }\n }\n return false;\n }\n /** Gets a boolean indicating that this connection will be used in the vertex shader */\n get isConnectedInVertexShader() {\n if (this.target === NodeMaterialBlockTargets.Vertex) {\n return true;\n }\n if (!this.hasEndpoints) {\n return false;\n }\n for (const endpoint of this._endpoints) {\n if (endpoint.ownerBlock.target === NodeMaterialBlockTargets.Vertex) {\n return true;\n }\n if (endpoint.target === NodeMaterialBlockTargets.Vertex) {\n return true;\n }\n if (endpoint.ownerBlock.target === NodeMaterialBlockTargets.Neutral || endpoint.ownerBlock.target === NodeMaterialBlockTargets.VertexAndFragment) {\n if (endpoint.ownerBlock.outputs.some((o) => o.isConnectedInVertexShader)) {\n return true;\n }\n }\n }\n return false;\n }\n /** Gets a boolean indicating that this connection will be used in the fragment shader */\n get isConnectedInFragmentShader() {\n if (this.target === NodeMaterialBlockTargets.Fragment) {\n return true;\n }\n if (!this.hasEndpoints) {\n return false;\n }\n for (const endpoint of this._endpoints) {\n if (endpoint.ownerBlock.target === NodeMaterialBlockTargets.Fragment) {\n return true;\n }\n if (endpoint.ownerBlock.target === NodeMaterialBlockTargets.Neutral || endpoint.ownerBlock.target === NodeMaterialBlockTargets.VertexAndFragment) {\n if (endpoint.ownerBlock.isConnectedInFragmentShader()) {\n return true;\n }\n }\n }\n return false;\n }\n /**\n * Creates a block suitable to be used as an input for this input point.\n * If null is returned, a block based on the point type will be created.\n * @returns The returned string parameter is the name of the output point of NodeMaterialBlock (first parameter of the returned array) that can be connected to the input\n */\n createCustomInputBlock() {\n return null;\n }\n /**\n * Creates a new connection point\n * @param name defines the connection point name\n * @param ownerBlock defines the block hosting this connection point\n * @param direction defines the direction of the connection point\n */\n constructor(name, ownerBlock, direction) {\n /** @internal */\n this._connectedPoint = null;\n this._endpoints = new Array();\n /** @internal */\n this._typeConnectionSource = null;\n /** @internal */\n this._defaultConnectionPointType = null;\n /** @internal */\n this._linkedConnectionSource = null;\n /** @internal */\n this._acceptedConnectionPointType = null;\n this._type = NodeMaterialBlockConnectionPointTypes.Float;\n /** @internal */\n this._enforceAssociatedVariableName = false;\n /** Indicates that this connection point needs dual validation before being connected to another point */\n this.needDualDirectionValidation = false;\n /**\n * Gets or sets the additional types supported by this connection point\n */\n this.acceptedConnectionPointTypes = new Array();\n /**\n * Gets or sets the additional types excluded by this connection point\n */\n this.excludedConnectionPointTypes = new Array();\n /**\n * Observable triggered when this point is connected\n */\n this.onConnectionObservable = new Observable$1();\n /**\n * Gets or sets a boolean indicating that this connection point is exposed on a frame\n */\n this.isExposedOnFrame = false;\n /**\n * Gets or sets number indicating the position that the port is exposed to on a frame\n */\n this.exposedPortPosition = -1;\n /** @internal */\n this._prioritizeVertex = false;\n this._target = NodeMaterialBlockTargets.VertexAndFragment;\n this._ownerBlock = ownerBlock;\n this.name = name;\n this._direction = direction;\n }\n /**\n * Gets the current class name e.g. \"NodeMaterialConnectionPoint\"\n * @returns the class name\n */\n getClassName() {\n return \"NodeMaterialConnectionPoint\";\n }\n /**\n * Gets a boolean indicating if the current point can be connected to another point\n * @param connectionPoint defines the other connection point\n * @returns a boolean\n */\n canConnectTo(connectionPoint) {\n return this.checkCompatibilityState(connectionPoint) === NodeMaterialConnectionPointCompatibilityStates.Compatible;\n }\n /**\n * Gets a number indicating if the current point can be connected to another point\n * @param connectionPoint defines the other connection point\n * @returns a number defining the compatibility state\n */\n checkCompatibilityState(connectionPoint) {\n const ownerBlock = this._ownerBlock;\n const otherBlock = connectionPoint.ownerBlock;\n if (ownerBlock.target === NodeMaterialBlockTargets.Fragment) {\n // Let's check we are not going reverse\n if (otherBlock.target === NodeMaterialBlockTargets.Vertex) {\n return NodeMaterialConnectionPointCompatibilityStates.TargetIncompatible;\n }\n for (const output of otherBlock.outputs) {\n if (output.ownerBlock.target != NodeMaterialBlockTargets.Neutral && output.isConnectedInVertexShader) {\n return NodeMaterialConnectionPointCompatibilityStates.TargetIncompatible;\n }\n }\n }\n if (this.type !== connectionPoint.type && connectionPoint.innerType !== NodeMaterialBlockConnectionPointTypes.AutoDetect) {\n // Equivalents\n if (NodeMaterialConnectionPoint.AreEquivalentTypes(this.type, connectionPoint.type)) {\n return NodeMaterialConnectionPointCompatibilityStates.Compatible;\n }\n // Accepted types\n if ((connectionPoint.acceptedConnectionPointTypes && connectionPoint.acceptedConnectionPointTypes.indexOf(this.type) !== -1) ||\n (connectionPoint._acceptedConnectionPointType && NodeMaterialConnectionPoint.AreEquivalentTypes(connectionPoint._acceptedConnectionPointType.type, this.type))) {\n return NodeMaterialConnectionPointCompatibilityStates.Compatible;\n }\n else {\n return NodeMaterialConnectionPointCompatibilityStates.TypeIncompatible;\n }\n }\n // Excluded\n if (connectionPoint.excludedConnectionPointTypes && connectionPoint.excludedConnectionPointTypes.indexOf(this.type) !== -1) {\n return NodeMaterialConnectionPointCompatibilityStates.TypeIncompatible;\n }\n // Check hierarchy\n let targetBlock = otherBlock;\n let sourceBlock = ownerBlock;\n if (this.direction === NodeMaterialConnectionPointDirection.Input) {\n targetBlock = ownerBlock;\n sourceBlock = otherBlock;\n }\n if (targetBlock.isAnAncestorOf(sourceBlock)) {\n return NodeMaterialConnectionPointCompatibilityStates.HierarchyIssue;\n }\n return NodeMaterialConnectionPointCompatibilityStates.Compatible;\n }\n /**\n * Connect this point to another connection point\n * @param connectionPoint defines the other connection point\n * @param ignoreConstraints defines if the system will ignore connection type constraints (default is false)\n * @returns the current connection point\n */\n connectTo(connectionPoint, ignoreConstraints = false) {\n if (!ignoreConstraints && !this.canConnectTo(connectionPoint)) {\n throw \"Cannot connect these two connectors.\";\n }\n this._endpoints.push(connectionPoint);\n connectionPoint._connectedPoint = this;\n this._enforceAssociatedVariableName = false;\n this.onConnectionObservable.notifyObservers(connectionPoint);\n connectionPoint.onConnectionObservable.notifyObservers(this);\n return this;\n }\n /**\n * Disconnect this point from one of his endpoint\n * @param endpoint defines the other connection point\n * @returns the current connection point\n */\n disconnectFrom(endpoint) {\n const index = this._endpoints.indexOf(endpoint);\n if (index === -1) {\n return this;\n }\n this._endpoints.splice(index, 1);\n endpoint._connectedPoint = null;\n this._enforceAssociatedVariableName = false;\n endpoint._enforceAssociatedVariableName = false;\n return this;\n }\n /**\n * Fill the list of excluded connection point types with all types other than those passed in the parameter\n * @param mask Types (ORed values of NodeMaterialBlockConnectionPointTypes) that are allowed, and thus will not be pushed to the excluded list\n */\n addExcludedConnectionPointFromAllowedTypes(mask) {\n let bitmask = 1;\n while (bitmask < NodeMaterialBlockConnectionPointTypes.All) {\n if (!(mask & bitmask)) {\n this.excludedConnectionPointTypes.push(bitmask);\n }\n bitmask = bitmask << 1;\n }\n }\n /**\n * Serializes this point in a JSON representation\n * @param isInput defines if the connection point is an input (default is true)\n * @returns the serialized point object\n */\n serialize(isInput = true) {\n const serializationObject = {};\n serializationObject.name = this.name;\n serializationObject.displayName = this.displayName;\n if (isInput && this.connectedPoint) {\n serializationObject.inputName = this.name;\n serializationObject.targetBlockId = this.connectedPoint.ownerBlock.uniqueId;\n serializationObject.targetConnectionName = this.connectedPoint.name;\n serializationObject.isExposedOnFrame = true;\n serializationObject.exposedPortPosition = this.exposedPortPosition;\n }\n if (this.isExposedOnFrame || this.exposedPortPosition >= 0) {\n serializationObject.isExposedOnFrame = true;\n serializationObject.exposedPortPosition = this.exposedPortPosition;\n }\n return serializationObject;\n }\n /**\n * Release resources\n */\n dispose() {\n this.onConnectionObservable.clear();\n }\n }\n\n /**\n * Defines a block that can be used inside a node based material\n */\n class NodeMaterialBlock {\n /**\n * Gets the name of the block\n */\n get name() {\n return this._name;\n }\n /**\n * Sets the name of the block. Will check if the name is valid.\n */\n set name(newName) {\n if (!this.validateBlockName(newName)) {\n return;\n }\n this._name = newName;\n }\n /**\n * Gets a boolean indicating that this block can only be used once per NodeMaterial\n */\n get isUnique() {\n return this._isUnique;\n }\n /**\n * Gets a boolean indicating that this block is an end block (e.g. it is generating a system value)\n */\n get isFinalMerger() {\n return this._isFinalMerger;\n }\n /**\n * Gets a boolean indicating that this block is an input (e.g. it sends data to the shader)\n */\n get isInput() {\n return this._isInput;\n }\n /**\n * Gets a boolean indicating if this block is a teleport out\n */\n get isTeleportOut() {\n return this._isTeleportOut;\n }\n /**\n * Gets a boolean indicating if this block is a teleport in\n */\n get isTeleportIn() {\n return this._isTeleportIn;\n }\n /**\n * Gets or sets the build Id\n */\n get buildId() {\n return this._buildId;\n }\n set buildId(value) {\n this._buildId = value;\n }\n /**\n * Gets or sets the target of the block\n */\n get target() {\n return this._target;\n }\n set target(value) {\n if ((this._target & value) !== 0) {\n return;\n }\n this._target = value;\n }\n /**\n * Gets the list of input points\n */\n get inputs() {\n return this._inputs;\n }\n /** Gets the list of output points */\n get outputs() {\n return this._outputs;\n }\n /**\n * Find an input by its name\n * @param name defines the name of the input to look for\n * @returns the input or null if not found\n */\n getInputByName(name) {\n const filter = this._inputs.filter((e) => e.name === name);\n if (filter.length) {\n return filter[0];\n }\n return null;\n }\n /**\n * Find an output by its name\n * @param name defines the name of the output to look for\n * @returns the output or null if not found\n */\n getOutputByName(name) {\n const filter = this._outputs.filter((e) => e.name === name);\n if (filter.length) {\n return filter[0];\n }\n return null;\n }\n /**\n * Creates a new NodeMaterialBlock\n * @param name defines the block name\n * @param target defines the target of that block (Vertex by default)\n * @param isFinalMerger defines a boolean indicating that this block is an end block (e.g. it is generating a system value). Default is false\n */\n constructor(name, target = NodeMaterialBlockTargets.Vertex, isFinalMerger = false) {\n this._isFinalMerger = false;\n this._isInput = false;\n this._isTeleportOut = false;\n this._isTeleportIn = false;\n this._name = \"\";\n this._isUnique = false;\n /** Gets or sets a boolean indicating that only one input can be connected at a time */\n this.inputsAreExclusive = false;\n /** @internal */\n this._codeVariableName = \"\";\n /** @internal */\n this._inputs = new Array();\n /** @internal */\n this._outputs = new Array();\n /**\n * Gets or sets the comments associated with this block\n */\n this.comments = \"\";\n /** Gets or sets a boolean indicating that this input can be edited in the Inspector (false by default) */\n this.visibleInInspector = false;\n /** Gets or sets a boolean indicating that this input can be edited from a collapsed frame */\n this.visibleOnFrame = false;\n this._target = target;\n this._originalTargetIsNeutral = target === NodeMaterialBlockTargets.Neutral;\n this._isFinalMerger = isFinalMerger;\n this._isInput = this.getClassName() === \"InputBlock\";\n this._isTeleportOut = this.getClassName() === \"NodeMaterialTeleportOutBlock\";\n this._isTeleportIn = this.getClassName() === \"NodeMaterialTeleportInBlock\";\n this._name = name;\n this.uniqueId = UniqueIdGenerator.UniqueId;\n }\n /** @internal */\n _setInitialTarget(target) {\n this._target = target;\n this._originalTargetIsNeutral = target === NodeMaterialBlockTargets.Neutral;\n }\n /**\n * Initialize the block and prepare the context for build\n * @param state defines the state that will be used for the build\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n initialize(state) {\n // Do nothing\n }\n /**\n * Bind data to effect. Will only be called for blocks with isBindable === true\n * @param effect defines the effect to bind data to\n * @param nodeMaterial defines the hosting NodeMaterial\n * @param mesh defines the mesh that will be rendered\n * @param subMesh defines the submesh that will be rendered\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n bind(effect, nodeMaterial, mesh, subMesh) {\n // Do nothing\n }\n _declareOutput(output, state) {\n return `${state._getGLType(output.type)} ${output.associatedVariableName}`;\n }\n _writeVariable(currentPoint) {\n const connectionPoint = currentPoint.connectedPoint;\n if (connectionPoint) {\n return `${currentPoint.associatedVariableName}`;\n }\n return `0.`;\n }\n _writeFloat(value) {\n let stringVersion = value.toString();\n if (stringVersion.indexOf(\".\") === -1) {\n stringVersion += \".0\";\n }\n return `${stringVersion}`;\n }\n /**\n * Gets the current class name e.g. \"NodeMaterialBlock\"\n * @returns the class name\n */\n getClassName() {\n return \"NodeMaterialBlock\";\n }\n /** Gets a boolean indicating that this connection will be used in the fragment shader */\n isConnectedInFragmentShader() {\n return this.outputs.some((o) => o.isConnectedInFragmentShader);\n }\n /**\n * Register a new input. Must be called inside a block constructor\n * @param name defines the connection point name\n * @param type defines the connection point type\n * @param isOptional defines a boolean indicating that this input can be omitted\n * @param target defines the target to use to limit the connection point (will be VertexAndFragment by default)\n * @param point an already created connection point. If not provided, create a new one\n * @returns the current block\n */\n registerInput(name, type, isOptional = false, target, point) {\n point = point !== null && point !== void 0 ? point : new NodeMaterialConnectionPoint(name, this, NodeMaterialConnectionPointDirection.Input);\n point.type = type;\n point.isOptional = isOptional;\n if (target) {\n point.target = target;\n }\n this._inputs.push(point);\n return this;\n }\n /**\n * Register a new output. Must be called inside a block constructor\n * @param name defines the connection point name\n * @param type defines the connection point type\n * @param target defines the target to use to limit the connection point (will be VertexAndFragment by default)\n * @param point an already created connection point. If not provided, create a new one\n * @returns the current block\n */\n registerOutput(name, type, target, point) {\n point = point !== null && point !== void 0 ? point : new NodeMaterialConnectionPoint(name, this, NodeMaterialConnectionPointDirection.Output);\n point.type = type;\n if (target) {\n point.target = target;\n }\n this._outputs.push(point);\n return this;\n }\n /**\n * Will return the first available input e.g. the first one which is not an uniform or an attribute\n * @param forOutput defines an optional connection point to check compatibility with\n * @returns the first available input or null\n */\n getFirstAvailableInput(forOutput = null) {\n for (const input of this._inputs) {\n if (!input.connectedPoint) {\n if (!forOutput || forOutput.type === input.type || input.type === NodeMaterialBlockConnectionPointTypes.AutoDetect) {\n return input;\n }\n }\n }\n return null;\n }\n /**\n * Will return the first available output e.g. the first one which is not yet connected and not a varying\n * @param forBlock defines an optional block to check compatibility with\n * @returns the first available input or null\n */\n getFirstAvailableOutput(forBlock = null) {\n for (const output of this._outputs) {\n if (!forBlock || !forBlock.target || forBlock.target === NodeMaterialBlockTargets.Neutral || (forBlock.target & output.target) !== 0) {\n return output;\n }\n }\n return null;\n }\n /**\n * Gets the sibling of the given output\n * @param current defines the current output\n * @returns the next output in the list or null\n */\n getSiblingOutput(current) {\n const index = this._outputs.indexOf(current);\n if (index === -1 || index >= this._outputs.length) {\n return null;\n }\n return this._outputs[index + 1];\n }\n /**\n * Checks if the current block is an ancestor of a given block\n * @param block defines the potential descendant block to check\n * @returns true if block is a descendant\n */\n isAnAncestorOf(block) {\n for (const output of this._outputs) {\n if (!output.hasEndpoints) {\n continue;\n }\n for (const endpoint of output.endpoints) {\n if (endpoint.ownerBlock === block) {\n return true;\n }\n if (endpoint.ownerBlock.isAnAncestorOf(block)) {\n return true;\n }\n }\n }\n return false;\n }\n /**\n * Connect current block with another block\n * @param other defines the block to connect with\n * @param options define the various options to help pick the right connections\n * @param options.input\n * @param options.output\n * @param options.outputSwizzle\n * @returns the current block\n */\n connectTo(other, options) {\n if (this._outputs.length === 0) {\n return;\n }\n let output = options && options.output ? this.getOutputByName(options.output) : this.getFirstAvailableOutput(other);\n let notFound = true;\n while (notFound) {\n const input = options && options.input ? other.getInputByName(options.input) : other.getFirstAvailableInput(output);\n if (output && input && output.canConnectTo(input)) {\n output.connectTo(input);\n notFound = false;\n }\n else if (!output) {\n throw \"Unable to find a compatible match\";\n }\n else {\n output = this.getSiblingOutput(output);\n }\n }\n return this;\n }\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _buildBlock(state) {\n // Empty. Must be defined by child nodes\n }\n /**\n * Add uniforms, samplers and uniform buffers at compilation time\n * @param state defines the state to update\n * @param nodeMaterial defines the node material requesting the update\n * @param defines defines the material defines to update\n * @param uniformBuffers defines the list of uniform buffer names\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n updateUniformsAndSamples(state, nodeMaterial, defines, uniformBuffers) {\n // Do nothing\n }\n /**\n * Add potential fallbacks if shader compilation fails\n * @param mesh defines the mesh to be rendered\n * @param fallbacks defines the current prioritized list of fallbacks\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n provideFallbacks(mesh, fallbacks) {\n // Do nothing\n }\n /**\n * Initialize defines for shader compilation\n * @param mesh defines the mesh to be rendered\n * @param nodeMaterial defines the node material requesting the update\n * @param defines defines the material defines to update\n * @param useInstances specifies that instances should be used\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n initializeDefines(mesh, nodeMaterial, defines, useInstances = false) { }\n /**\n * Update defines for shader compilation\n * @param mesh defines the mesh to be rendered\n * @param nodeMaterial defines the node material requesting the update\n * @param defines defines the material defines to update\n * @param useInstances specifies that instances should be used\n * @param subMesh defines which submesh to render\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n prepareDefines(mesh, nodeMaterial, defines, useInstances = false, subMesh) {\n // Do nothing\n }\n /**\n * Lets the block try to connect some inputs automatically\n * @param material defines the hosting NodeMaterial\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n autoConfigure(material) {\n // Do nothing\n }\n /**\n * Function called when a block is declared as repeatable content generator\n * @param vertexShaderState defines the current compilation state for the vertex shader\n * @param fragmentShaderState defines the current compilation state for the fragment shader\n * @param mesh defines the mesh to be rendered\n * @param defines defines the material defines to update\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n replaceRepeatableContent(vertexShaderState, fragmentShaderState, mesh, defines) {\n // Do nothing\n }\n /** Gets a boolean indicating that the code of this block will be promoted to vertex shader even if connected to fragment output */\n get willBeGeneratedIntoVertexShaderFromFragmentShader() {\n if (this.isInput || this.isFinalMerger) {\n return false;\n }\n if (this._outputs.some((o) => o.isDirectlyConnectedToVertexOutput)) {\n return false;\n }\n if (this.target === NodeMaterialBlockTargets.Vertex) {\n return false;\n }\n if (this.target === NodeMaterialBlockTargets.VertexAndFragment || this.target === NodeMaterialBlockTargets.Neutral) {\n if (this._outputs.some((o) => o.isConnectedInVertexShader)) {\n return true;\n }\n }\n return false;\n }\n /**\n * Checks if the block is ready\n * @param mesh defines the mesh to be rendered\n * @param nodeMaterial defines the node material requesting the update\n * @param defines defines the material defines to update\n * @param useInstances specifies that instances should be used\n * @returns true if the block is ready\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n isReady(mesh, nodeMaterial, defines, useInstances = false) {\n return true;\n }\n _linkConnectionTypes(inputIndex0, inputIndex1, looseCoupling = false) {\n if (looseCoupling) {\n this._inputs[inputIndex1]._acceptedConnectionPointType = this._inputs[inputIndex0];\n }\n else {\n this._inputs[inputIndex0]._linkedConnectionSource = this._inputs[inputIndex1];\n }\n this._inputs[inputIndex1]._linkedConnectionSource = this._inputs[inputIndex0];\n }\n _processBuild(block, state, input, activeBlocks) {\n block.build(state, activeBlocks);\n const localBlockIsFragment = state._vertexState != null;\n const otherBlockWasGeneratedInVertexShader = block._buildTarget === NodeMaterialBlockTargets.Vertex && block.target !== NodeMaterialBlockTargets.VertexAndFragment;\n if (localBlockIsFragment &&\n ((block.target & block._buildTarget) === 0 ||\n (block.target & input.target) === 0 ||\n (this.target !== NodeMaterialBlockTargets.VertexAndFragment && otherBlockWasGeneratedInVertexShader))) {\n // context switch! We need a varying\n if ((!block.isInput && state.target !== block._buildTarget) || // block was already emitted by vertex shader\n (block.isInput && block.isAttribute && !block._noContextSwitch) // block is an attribute\n ) {\n const connectedPoint = input.connectedPoint;\n if (state._vertexState._emitVaryingFromString(\"v_\" + connectedPoint.associatedVariableName, state._getGLType(connectedPoint.type))) {\n state._vertexState.compilationString += `${\"v_\" + connectedPoint.associatedVariableName} = ${connectedPoint.associatedVariableName};\\n`;\n }\n input.associatedVariableName = \"v_\" + connectedPoint.associatedVariableName;\n input._enforceAssociatedVariableName = true;\n }\n }\n }\n /**\n * Validates the new name for the block node.\n * @param newName the new name to be given to the node.\n * @returns false if the name is a reserve word, else true.\n */\n validateBlockName(newName) {\n const reservedNames = [\n \"position\",\n \"normal\",\n \"tangent\",\n \"particle_positionw\",\n \"uv\",\n \"uv2\",\n \"uv3\",\n \"uv4\",\n \"uv5\",\n \"uv6\",\n \"position2d\",\n \"particle_uv\",\n \"matricesIndices\",\n \"matricesWeights\",\n \"world0\",\n \"world1\",\n \"world2\",\n \"world3\",\n \"particle_color\",\n \"particle_texturemask\",\n ];\n for (const reservedName of reservedNames) {\n if (newName === reservedName) {\n return false;\n }\n }\n return true;\n }\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _customBuildStep(state, activeBlocks) {\n // Must be implemented by children\n }\n /**\n * Compile the current node and generate the shader code\n * @param state defines the current compilation state (uniforms, samplers, current string)\n * @param activeBlocks defines the list of active blocks (i.e. blocks to compile)\n * @returns true if already built\n */\n build(state, activeBlocks) {\n if (this._buildId === state.sharedData.buildId) {\n return true;\n }\n if (!this.isInput) {\n /** Prepare outputs */\n for (const output of this._outputs) {\n if (!output.associatedVariableName) {\n output.associatedVariableName = state._getFreeVariableName(output.name);\n }\n }\n }\n // Check if \"parent\" blocks are compiled\n for (const input of this._inputs) {\n if (!input.connectedPoint) {\n if (!input.isOptional) {\n // Emit a warning\n state.sharedData.checks.notConnectedNonOptionalInputs.push(input);\n }\n continue;\n }\n if (this.target !== NodeMaterialBlockTargets.Neutral) {\n if ((input.target & this.target) === 0) {\n continue;\n }\n if ((input.target & state.target) === 0) {\n continue;\n }\n }\n const block = input.connectedPoint.ownerBlock;\n if (block && block !== this) {\n this._processBuild(block, state, input, activeBlocks);\n }\n }\n this._customBuildStep(state, activeBlocks);\n if (this._buildId === state.sharedData.buildId) {\n return true; // Need to check again as inputs can be connected multiple time to this endpoint\n }\n // Logs\n if (state.sharedData.verbose) {\n console.log(`${state.target === NodeMaterialBlockTargets.Vertex ? \"Vertex shader\" : \"Fragment shader\"}: Building ${this.name} [${this.getClassName()}]`);\n }\n // Checks final outputs\n if (this.isFinalMerger) {\n switch (state.target) {\n case NodeMaterialBlockTargets.Vertex:\n state.sharedData.checks.emitVertex = true;\n break;\n case NodeMaterialBlockTargets.Fragment:\n state.sharedData.checks.emitFragment = true;\n break;\n }\n }\n if (!this.isInput && state.sharedData.emitComments) {\n state.compilationString += `\\n//${this.name}\\n`;\n }\n this._buildBlock(state);\n this._buildId = state.sharedData.buildId;\n this._buildTarget = state.target;\n // Compile connected blocks\n for (const output of this._outputs) {\n if ((output.target & state.target) === 0) {\n continue;\n }\n for (const endpoint of output.endpoints) {\n const block = endpoint.ownerBlock;\n if (block && (block.target & state.target) !== 0 && activeBlocks.indexOf(block) !== -1) {\n this._processBuild(block, state, endpoint, activeBlocks);\n }\n }\n }\n return false;\n }\n _inputRename(name) {\n return name;\n }\n _outputRename(name) {\n return name;\n }\n _dumpPropertiesCode() {\n const variableName = this._codeVariableName;\n return `${variableName}.visibleInInspector = ${this.visibleInInspector};\\n${variableName}.visibleOnFrame = ${this.visibleOnFrame};\\n${variableName}.target = ${this.target};\\n`;\n }\n /**\n * @internal\n */\n _dumpCode(uniqueNames, alreadyDumped) {\n alreadyDumped.push(this);\n // Get unique name\n const nameAsVariableName = this.name.replace(/[^A-Za-z_]+/g, \"\");\n this._codeVariableName = nameAsVariableName || `${this.getClassName()}_${this.uniqueId}`;\n if (uniqueNames.indexOf(this._codeVariableName) !== -1) {\n let index = 0;\n do {\n index++;\n this._codeVariableName = nameAsVariableName + index;\n } while (uniqueNames.indexOf(this._codeVariableName) !== -1);\n }\n uniqueNames.push(this._codeVariableName);\n // Declaration\n let codeString = `\\n// ${this.getClassName()}\\n`;\n if (this.comments) {\n codeString += `// ${this.comments}\\n`;\n }\n codeString += `var ${this._codeVariableName} = new BABYLON.${this.getClassName()}(\"${this.name}\");\\n`;\n // Properties\n codeString += this._dumpPropertiesCode();\n // Inputs\n for (const input of this.inputs) {\n if (!input.isConnected) {\n continue;\n }\n const connectedOutput = input.connectedPoint;\n const connectedBlock = connectedOutput.ownerBlock;\n if (alreadyDumped.indexOf(connectedBlock) === -1) {\n codeString += connectedBlock._dumpCode(uniqueNames, alreadyDumped);\n }\n }\n // Outputs\n for (const output of this.outputs) {\n if (!output.hasEndpoints) {\n continue;\n }\n for (const endpoint of output.endpoints) {\n const connectedBlock = endpoint.ownerBlock;\n if (connectedBlock && alreadyDumped.indexOf(connectedBlock) === -1) {\n codeString += connectedBlock._dumpCode(uniqueNames, alreadyDumped);\n }\n }\n }\n return codeString;\n }\n /**\n * @internal\n */\n _dumpCodeForOutputConnections(alreadyDumped) {\n let codeString = \"\";\n if (alreadyDumped.indexOf(this) !== -1) {\n return codeString;\n }\n alreadyDumped.push(this);\n for (const input of this.inputs) {\n if (!input.isConnected) {\n continue;\n }\n const connectedOutput = input.connectedPoint;\n const connectedBlock = connectedOutput.ownerBlock;\n codeString += connectedBlock._dumpCodeForOutputConnections(alreadyDumped);\n codeString += `${connectedBlock._codeVariableName}.${connectedBlock._outputRename(connectedOutput.name)}.connectTo(${this._codeVariableName}.${this._inputRename(input.name)});\\n`;\n }\n return codeString;\n }\n /**\n * Clone the current block to a new identical block\n * @param scene defines the hosting scene\n * @param rootUrl defines the root URL to use to load textures and relative dependencies\n * @returns a copy of the current block\n */\n clone(scene, rootUrl = \"\") {\n const serializationObject = this.serialize();\n const blockType = GetClass(serializationObject.customType);\n if (blockType) {\n const block = new blockType();\n block._deserialize(serializationObject, scene, rootUrl);\n return block;\n }\n return null;\n }\n /**\n * Serializes this block in a JSON representation\n * @returns the serialized block object\n */\n serialize() {\n const serializationObject = {};\n serializationObject.customType = \"BABYLON.\" + this.getClassName();\n serializationObject.id = this.uniqueId;\n serializationObject.name = this.name;\n serializationObject.comments = this.comments;\n serializationObject.visibleInInspector = this.visibleInInspector;\n serializationObject.visibleOnFrame = this.visibleOnFrame;\n serializationObject.target = this.target;\n serializationObject.inputs = [];\n serializationObject.outputs = [];\n for (const input of this.inputs) {\n serializationObject.inputs.push(input.serialize());\n }\n for (const output of this.outputs) {\n serializationObject.outputs.push(output.serialize(false));\n }\n return serializationObject;\n }\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _deserialize(serializationObject, scene, rootUrl) {\n var _a;\n this.name = serializationObject.name;\n this.comments = serializationObject.comments;\n this.visibleInInspector = !!serializationObject.visibleInInspector;\n this.visibleOnFrame = !!serializationObject.visibleOnFrame;\n this._target = (_a = serializationObject.target) !== null && _a !== void 0 ? _a : this.target;\n this._deserializePortDisplayNamesAndExposedOnFrame(serializationObject);\n }\n _deserializePortDisplayNamesAndExposedOnFrame(serializationObject) {\n const serializedInputs = serializationObject.inputs;\n const serializedOutputs = serializationObject.outputs;\n if (serializedInputs) {\n serializedInputs.forEach((port, i) => {\n if (port.displayName) {\n this.inputs[i].displayName = port.displayName;\n }\n if (port.isExposedOnFrame) {\n this.inputs[i].isExposedOnFrame = port.isExposedOnFrame;\n this.inputs[i].exposedPortPosition = port.exposedPortPosition;\n }\n });\n }\n if (serializedOutputs) {\n serializedOutputs.forEach((port, i) => {\n if (port.displayName) {\n this.outputs[i].displayName = port.displayName;\n }\n if (port.isExposedOnFrame) {\n this.outputs[i].isExposedOnFrame = port.isExposedOnFrame;\n this.outputs[i].exposedPortPosition = port.exposedPortPosition;\n }\n });\n }\n }\n /**\n * Release resources\n */\n dispose() {\n for (const input of this.inputs) {\n input.dispose();\n }\n for (const output of this.outputs) {\n output.dispose();\n }\n }\n }\n\n /**\n * Block used to transform a vector (2, 3 or 4) with a matrix. It will generate a Vector4\n */\n class TransformBlock extends NodeMaterialBlock {\n /**\n * Creates a new TransformBlock\n * @param name defines the block name\n */\n constructor(name) {\n super(name, NodeMaterialBlockTargets.Neutral);\n /**\n * Defines the value to use to complement W value to transform it to a Vector4\n */\n this.complementW = 1;\n /**\n * Defines the value to use to complement z value to transform it to a Vector4\n */\n this.complementZ = 0;\n this.target = NodeMaterialBlockTargets.Vertex;\n this.registerInput(\"vector\", NodeMaterialBlockConnectionPointTypes.AutoDetect);\n this.registerInput(\"transform\", NodeMaterialBlockConnectionPointTypes.Matrix);\n this.registerOutput(\"output\", NodeMaterialBlockConnectionPointTypes.Vector4);\n this.registerOutput(\"xyz\", NodeMaterialBlockConnectionPointTypes.Vector3);\n this._inputs[0].onConnectionObservable.add((other) => {\n if (other.ownerBlock.isInput) {\n const otherAsInput = other.ownerBlock;\n if (otherAsInput.name === \"normal\" || otherAsInput.name === \"tangent\") {\n this.complementW = 0;\n }\n }\n });\n }\n /**\n * Gets the current class name\n * @returns the class name\n */\n getClassName() {\n return \"TransformBlock\";\n }\n /**\n * Gets the vector input\n */\n get vector() {\n return this._inputs[0];\n }\n /**\n * Gets the output component\n */\n get output() {\n return this._outputs[0];\n }\n /**\n * Gets the xyz output component\n */\n get xyz() {\n return this._outputs[1];\n }\n /**\n * Gets the matrix transform input\n */\n get transform() {\n return this._inputs[1];\n }\n _buildBlock(state) {\n super._buildBlock(state);\n const vector = this.vector;\n const transform = this.transform;\n if (vector.connectedPoint) {\n // None uniform scaling case.\n if (this.complementW === 0) {\n const comments = `//${this.name}`;\n state._emitFunctionFromInclude(\"helperFunctions\", comments);\n state.sharedData.blocksWithDefines.push(this);\n const transformName = state._getFreeVariableName(`${transform.associatedVariableName}_NUS`);\n state.compilationString += `mat3 ${transformName} = mat3(${transform.associatedVariableName});\\n`;\n state.compilationString += `#ifdef NONUNIFORMSCALING\\n`;\n state.compilationString += `${transformName} = transposeMat3(inverseMat3(${transformName}));\\n`;\n state.compilationString += `#endif\\n`;\n switch (vector.connectedPoint.type) {\n case NodeMaterialBlockConnectionPointTypes.Vector2:\n state.compilationString +=\n this._declareOutput(this.output, state) +\n ` = vec4(${transformName} * vec3(${vector.associatedVariableName}, ${this._writeFloat(this.complementZ)}), ${this._writeFloat(this.complementW)});\\n`;\n break;\n case NodeMaterialBlockConnectionPointTypes.Vector3:\n case NodeMaterialBlockConnectionPointTypes.Color3:\n state.compilationString +=\n this._declareOutput(this.output, state) + ` = vec4(${transformName} * ${vector.associatedVariableName}, ${this._writeFloat(this.complementW)});\\n`;\n break;\n default:\n state.compilationString +=\n this._declareOutput(this.output, state) + ` = vec4(${transformName} * ${vector.associatedVariableName}.xyz, ${this._writeFloat(this.complementW)});\\n`;\n break;\n }\n }\n else {\n const transformName = transform.associatedVariableName;\n switch (vector.connectedPoint.type) {\n case NodeMaterialBlockConnectionPointTypes.Vector2:\n state.compilationString +=\n this._declareOutput(this.output, state) +\n ` = ${transformName} * vec4(${vector.associatedVariableName}, ${this._writeFloat(this.complementZ)}, ${this._writeFloat(this.complementW)});\\n`;\n break;\n case NodeMaterialBlockConnectionPointTypes.Vector3:\n case NodeMaterialBlockConnectionPointTypes.Color3:\n state.compilationString +=\n this._declareOutput(this.output, state) + ` = ${transformName} * vec4(${vector.associatedVariableName}, ${this._writeFloat(this.complementW)});\\n`;\n break;\n default:\n state.compilationString += this._declareOutput(this.output, state) + ` = ${transformName} * ${vector.associatedVariableName};\\n`;\n break;\n }\n }\n if (this.xyz.hasEndpoints) {\n state.compilationString += this._declareOutput(this.xyz, state) + ` = ${this.output.associatedVariableName}.xyz;\\n`;\n }\n }\n return this;\n }\n /**\n * Update defines for shader compilation\n * @param mesh defines the mesh to be rendered\n * @param nodeMaterial defines the node material requesting the update\n * @param defines defines the material defines to update\n */\n prepareDefines(mesh, nodeMaterial, defines) {\n // Do nothing\n if (mesh.nonUniformScaling) {\n defines.setValue(\"NONUNIFORMSCALING\", true);\n }\n }\n serialize() {\n const serializationObject = super.serialize();\n serializationObject.complementZ = this.complementZ;\n serializationObject.complementW = this.complementW;\n return serializationObject;\n }\n _deserialize(serializationObject, scene, rootUrl) {\n super._deserialize(serializationObject, scene, rootUrl);\n this.complementZ = serializationObject.complementZ !== undefined ? serializationObject.complementZ : 0.0;\n this.complementW = serializationObject.complementW !== undefined ? serializationObject.complementW : 1.0;\n }\n _dumpPropertiesCode() {\n let codeString = super._dumpPropertiesCode() + `${this._codeVariableName}.complementZ = ${this.complementZ};\\n`;\n codeString += `${this._codeVariableName}.complementW = ${this.complementW};\\n`;\n return codeString;\n }\n }\n RegisterClass(\"BABYLON.TransformBlock\", TransformBlock);\n\n /**\n * Block used to output the vertex position\n */\n class VertexOutputBlock extends NodeMaterialBlock {\n /**\n * Creates a new VertexOutputBlock\n * @param name defines the block name\n */\n constructor(name) {\n super(name, NodeMaterialBlockTargets.Vertex, true);\n this.registerInput(\"vector\", NodeMaterialBlockConnectionPointTypes.Vector4);\n }\n /**\n * Gets the current class name\n * @returns the class name\n */\n getClassName() {\n return \"VertexOutputBlock\";\n }\n /**\n * Gets the vector input component\n */\n get vector() {\n return this._inputs[0];\n }\n _isLogarithmicDepthEnabled(nodeList) {\n for (const node of nodeList) {\n if (node.useLogarithmicDepth) {\n return true;\n }\n }\n return false;\n }\n _buildBlock(state) {\n super._buildBlock(state);\n const input = this.vector;\n state.compilationString += `gl_Position = ${input.associatedVariableName};\\n`;\n if (this._isLogarithmicDepthEnabled(state.sharedData.fragmentOutputNodes)) {\n state._emitUniformFromString(\"logarithmicDepthConstant\", \"float\");\n state._emitVaryingFromString(\"vFragmentDepth\", \"float\");\n state.compilationString += `vFragmentDepth = 1.0 + gl_Position.w;\\n`;\n state.compilationString += `gl_Position.z = log2(max(0.000001, vFragmentDepth)) * logarithmicDepthConstant;\\n`;\n }\n return this;\n }\n }\n RegisterClass(\"BABYLON.VertexOutputBlock\", VertexOutputBlock);\n\n /**\n * Enum defining the type of properties that can be edited in the property pages in the node editor\n */\n var PropertyTypeForEdition;\n (function (PropertyTypeForEdition) {\n /** property is a boolean */\n PropertyTypeForEdition[PropertyTypeForEdition[\"Boolean\"] = 0] = \"Boolean\";\n /** property is a float */\n PropertyTypeForEdition[PropertyTypeForEdition[\"Float\"] = 1] = \"Float\";\n /** property is a int */\n PropertyTypeForEdition[PropertyTypeForEdition[\"Int\"] = 2] = \"Int\";\n /** property is a Vector2 */\n PropertyTypeForEdition[PropertyTypeForEdition[\"Vector2\"] = 3] = \"Vector2\";\n /** property is a list of values */\n PropertyTypeForEdition[PropertyTypeForEdition[\"List\"] = 4] = \"List\";\n })(PropertyTypeForEdition || (PropertyTypeForEdition = {}));\n /**\n * Decorator that flags a property in a node block as being editable\n * @param displayName\n * @param propertyType\n * @param groupName\n * @param options\n */\n function editableInPropertyPage(displayName, propertyType = PropertyTypeForEdition.Boolean, groupName = \"PROPERTIES\", options) {\n return (target, propertyKey) => {\n let propStore = target._propStore;\n if (!propStore) {\n propStore = [];\n target._propStore = propStore;\n }\n propStore.push({\n propertyName: propertyKey,\n displayName: displayName,\n type: propertyType,\n groupName: groupName,\n options: options !== null && options !== void 0 ? options : {},\n });\n };\n }\n\n /**\n * Block used to output the final color\n */\n class FragmentOutputBlock extends NodeMaterialBlock {\n /**\n * Create a new FragmentOutputBlock\n * @param name defines the block name\n */\n constructor(name) {\n super(name, NodeMaterialBlockTargets.Fragment, true);\n /** Gets or sets a boolean indicating if content needs to be converted to gamma space */\n this.convertToGammaSpace = false;\n /** Gets or sets a boolean indicating if content needs to be converted to linear space */\n this.convertToLinearSpace = false;\n /** Gets or sets a boolean indicating if logarithmic depth should be used */\n this.useLogarithmicDepth = false;\n this.registerInput(\"rgba\", NodeMaterialBlockConnectionPointTypes.Color4, true);\n this.registerInput(\"rgb\", NodeMaterialBlockConnectionPointTypes.AutoDetect, true);\n this.registerInput(\"a\", NodeMaterialBlockConnectionPointTypes.Float, true);\n this.rgb.addExcludedConnectionPointFromAllowedTypes(NodeMaterialBlockConnectionPointTypes.Color3 | NodeMaterialBlockConnectionPointTypes.Vector3 | NodeMaterialBlockConnectionPointTypes.Float);\n }\n /**\n * Gets the current class name\n * @returns the class name\n */\n getClassName() {\n return \"FragmentOutputBlock\";\n }\n /**\n * Initialize the block and prepare the context for build\n * @param state defines the state that will be used for the build\n */\n initialize(state) {\n state._excludeVariableName(\"logarithmicDepthConstant\");\n state._excludeVariableName(\"vFragmentDepth\");\n }\n /**\n * Gets the rgba input component\n */\n get rgba() {\n return this._inputs[0];\n }\n /**\n * Gets the rgb input component\n */\n get rgb() {\n return this._inputs[1];\n }\n /**\n * Gets the a input component\n */\n get a() {\n return this._inputs[2];\n }\n prepareDefines(mesh, nodeMaterial, defines) {\n defines.setValue(this._linearDefineName, this.convertToLinearSpace, true);\n defines.setValue(this._gammaDefineName, this.convertToGammaSpace, true);\n }\n bind(effect, nodeMaterial, mesh) {\n if (this.useLogarithmicDepth && mesh) {\n MaterialHelper.BindLogDepth(undefined, effect, mesh.getScene());\n }\n }\n _buildBlock(state) {\n super._buildBlock(state);\n const rgba = this.rgba;\n const rgb = this.rgb;\n const a = this.a;\n state.sharedData.hints.needAlphaBlending = rgba.isConnected || a.isConnected;\n state.sharedData.blocksWithDefines.push(this);\n if (this.useLogarithmicDepth) {\n state._emitUniformFromString(\"logarithmicDepthConstant\", \"float\");\n state._emitVaryingFromString(\"vFragmentDepth\", \"float\");\n state.sharedData.bindableBlocks.push(this);\n }\n this._linearDefineName = state._getFreeDefineName(\"CONVERTTOLINEAR\");\n this._gammaDefineName = state._getFreeDefineName(\"CONVERTTOGAMMA\");\n const comments = `//${this.name}`;\n state._emitFunctionFromInclude(\"helperFunctions\", comments);\n if (rgba.connectedPoint) {\n if (a.isConnected) {\n state.compilationString += `gl_FragColor = vec4(${rgba.associatedVariableName}.rgb, ${a.associatedVariableName});\\n`;\n }\n else {\n state.compilationString += `gl_FragColor = ${rgba.associatedVariableName};\\n`;\n }\n }\n else if (rgb.connectedPoint) {\n let aValue = \"1.0\";\n if (a.connectedPoint) {\n aValue = a.associatedVariableName;\n }\n if (rgb.connectedPoint.type === NodeMaterialBlockConnectionPointTypes.Float) {\n state.compilationString += `gl_FragColor = vec4(${rgb.associatedVariableName}, ${rgb.associatedVariableName}, ${rgb.associatedVariableName}, ${aValue});\\n`;\n }\n else {\n state.compilationString += `gl_FragColor = vec4(${rgb.associatedVariableName}, ${aValue});\\n`;\n }\n }\n else {\n state.sharedData.checks.notConnectedNonOptionalInputs.push(rgba);\n }\n state.compilationString += `#ifdef ${this._linearDefineName}\\n`;\n state.compilationString += `gl_FragColor = toLinearSpace(gl_FragColor);\\n`;\n state.compilationString += `#endif\\n`;\n state.compilationString += `#ifdef ${this._gammaDefineName}\\n`;\n state.compilationString += `gl_FragColor = toGammaSpace(gl_FragColor);\\n`;\n state.compilationString += `#endif\\n`;\n if (this.useLogarithmicDepth) {\n state.compilationString += `gl_FragDepthEXT = log2(vFragmentDepth) * logarithmicDepthConstant * 0.5;\\n`;\n }\n return this;\n }\n _dumpPropertiesCode() {\n let codeString = super._dumpPropertiesCode();\n codeString += `${this._codeVariableName}.convertToGammaSpace = ${this.convertToGammaSpace};\\n`;\n codeString += `${this._codeVariableName}.convertToLinearSpace = ${this.convertToLinearSpace};\\n`;\n codeString += `${this._codeVariableName}.useLogarithmicDepth = ${this.useLogarithmicDepth};\\n`;\n return codeString;\n }\n serialize() {\n const serializationObject = super.serialize();\n serializationObject.convertToGammaSpace = this.convertToGammaSpace;\n serializationObject.convertToLinearSpace = this.convertToLinearSpace;\n serializationObject.useLogarithmicDepth = this.useLogarithmicDepth;\n return serializationObject;\n }\n _deserialize(serializationObject, scene, rootUrl) {\n var _a;\n super._deserialize(serializationObject, scene, rootUrl);\n this.convertToGammaSpace = serializationObject.convertToGammaSpace;\n this.convertToLinearSpace = serializationObject.convertToLinearSpace;\n this.useLogarithmicDepth = (_a = serializationObject.useLogarithmicDepth) !== null && _a !== void 0 ? _a : false;\n }\n }\n __decorate$1([\n editableInPropertyPage(\"Convert to gamma space\", PropertyTypeForEdition.Boolean, \"PROPERTIES\", { notifiers: { update: true } })\n ], FragmentOutputBlock.prototype, \"convertToGammaSpace\", void 0);\n __decorate$1([\n editableInPropertyPage(\"Convert to linear space\", PropertyTypeForEdition.Boolean, \"PROPERTIES\", { notifiers: { update: true } })\n ], FragmentOutputBlock.prototype, \"convertToLinearSpace\", void 0);\n __decorate$1([\n editableInPropertyPage(\"Use logarithmic depth\", PropertyTypeForEdition.Boolean, \"PROPERTIES\")\n ], FragmentOutputBlock.prototype, \"useLogarithmicDepth\", void 0);\n RegisterClass(\"BABYLON.FragmentOutputBlock\", FragmentOutputBlock);\n\n /**\n * Enum defining the mode of a NodeMaterialBlockConnectionPoint\n */\n var NodeMaterialBlockConnectionPointMode;\n (function (NodeMaterialBlockConnectionPointMode) {\n /** Value is an uniform */\n NodeMaterialBlockConnectionPointMode[NodeMaterialBlockConnectionPointMode[\"Uniform\"] = 0] = \"Uniform\";\n /** Value is a mesh attribute */\n NodeMaterialBlockConnectionPointMode[NodeMaterialBlockConnectionPointMode[\"Attribute\"] = 1] = \"Attribute\";\n /** Value is a varying between vertex and fragment shaders */\n NodeMaterialBlockConnectionPointMode[NodeMaterialBlockConnectionPointMode[\"Varying\"] = 2] = \"Varying\";\n /** Mode is undefined */\n NodeMaterialBlockConnectionPointMode[NodeMaterialBlockConnectionPointMode[\"Undefined\"] = 3] = \"Undefined\";\n })(NodeMaterialBlockConnectionPointMode || (NodeMaterialBlockConnectionPointMode = {}));\n\n /**\n * Enum used to define system values e.g. values automatically provided by the system\n */\n var NodeMaterialSystemValues;\n (function (NodeMaterialSystemValues) {\n /** World */\n NodeMaterialSystemValues[NodeMaterialSystemValues[\"World\"] = 1] = \"World\";\n /** View */\n NodeMaterialSystemValues[NodeMaterialSystemValues[\"View\"] = 2] = \"View\";\n /** Projection */\n NodeMaterialSystemValues[NodeMaterialSystemValues[\"Projection\"] = 3] = \"Projection\";\n /** ViewProjection */\n NodeMaterialSystemValues[NodeMaterialSystemValues[\"ViewProjection\"] = 4] = \"ViewProjection\";\n /** WorldView */\n NodeMaterialSystemValues[NodeMaterialSystemValues[\"WorldView\"] = 5] = \"WorldView\";\n /** WorldViewProjection */\n NodeMaterialSystemValues[NodeMaterialSystemValues[\"WorldViewProjection\"] = 6] = \"WorldViewProjection\";\n /** CameraPosition */\n NodeMaterialSystemValues[NodeMaterialSystemValues[\"CameraPosition\"] = 7] = \"CameraPosition\";\n /** Fog Color */\n NodeMaterialSystemValues[NodeMaterialSystemValues[\"FogColor\"] = 8] = \"FogColor\";\n /** Delta time */\n NodeMaterialSystemValues[NodeMaterialSystemValues[\"DeltaTime\"] = 9] = \"DeltaTime\";\n /** Camera parameters */\n NodeMaterialSystemValues[NodeMaterialSystemValues[\"CameraParameters\"] = 10] = \"CameraParameters\";\n /** Material alpha */\n NodeMaterialSystemValues[NodeMaterialSystemValues[\"MaterialAlpha\"] = 11] = \"MaterialAlpha\";\n })(NodeMaterialSystemValues || (NodeMaterialSystemValues = {}));\n\n /**\n * Enum defining the type of animations supported by InputBlock\n */\n var AnimatedInputBlockTypes;\n (function (AnimatedInputBlockTypes) {\n /** No animation */\n AnimatedInputBlockTypes[AnimatedInputBlockTypes[\"None\"] = 0] = \"None\";\n /** Time based animation (is incremented by 0.6 each second). Will only work for floats */\n AnimatedInputBlockTypes[AnimatedInputBlockTypes[\"Time\"] = 1] = \"Time\";\n /** Time elapsed (in seconds) since the engine was initialized. Will only work for floats */\n AnimatedInputBlockTypes[AnimatedInputBlockTypes[\"RealTime\"] = 2] = \"RealTime\";\n })(AnimatedInputBlockTypes || (AnimatedInputBlockTypes = {}));\n\n /* eslint-disable @typescript-eslint/naming-convention */\n const remapAttributeName = {\n position2d: \"position\",\n particle_uv: \"vUV\",\n particle_color: \"vColor\",\n particle_texturemask: \"textureMask\",\n particle_positionw: \"vPositionW\",\n };\n const attributeInFragmentOnly = {\n particle_uv: true,\n particle_color: true,\n particle_texturemask: true,\n particle_positionw: true,\n };\n const attributeAsUniform = {\n particle_texturemask: true,\n };\n /**\n * Block used to expose an input value\n */\n class InputBlock extends NodeMaterialBlock {\n /**\n * Gets or sets the connection point type (default is float)\n */\n get type() {\n if (this._type === NodeMaterialBlockConnectionPointTypes.AutoDetect) {\n if (this.isUniform && this.value != null) {\n if (!isNaN(this.value)) {\n this._type = NodeMaterialBlockConnectionPointTypes.Float;\n return this._type;\n }\n switch (this.value.getClassName()) {\n case \"Vector2\":\n this._type = NodeMaterialBlockConnectionPointTypes.Vector2;\n return this._type;\n case \"Vector3\":\n this._type = NodeMaterialBlockConnectionPointTypes.Vector3;\n return this._type;\n case \"Vector4\":\n this._type = NodeMaterialBlockConnectionPointTypes.Vector4;\n return this._type;\n case \"Color3\":\n this._type = NodeMaterialBlockConnectionPointTypes.Color3;\n return this._type;\n case \"Color4\":\n this._type = NodeMaterialBlockConnectionPointTypes.Color4;\n return this._type;\n case \"Matrix\":\n this._type = NodeMaterialBlockConnectionPointTypes.Matrix;\n return this._type;\n }\n }\n if (this.isAttribute) {\n switch (this.name) {\n case \"position\":\n case \"normal\":\n case \"particle_positionw\":\n this._type = NodeMaterialBlockConnectionPointTypes.Vector3;\n return this._type;\n case \"uv\":\n case \"uv2\":\n case \"uv3\":\n case \"uv4\":\n case \"uv5\":\n case \"uv6\":\n case \"position2d\":\n case \"particle_uv\":\n this._type = NodeMaterialBlockConnectionPointTypes.Vector2;\n return this._type;\n case \"matricesIndices\":\n case \"matricesWeights\":\n case \"matricesIndicesExtra\":\n case \"matricesWeightsExtra\":\n case \"world0\":\n case \"world1\":\n case \"world2\":\n case \"world3\":\n case \"tangent\":\n this._type = NodeMaterialBlockConnectionPointTypes.Vector4;\n return this._type;\n case \"color\":\n case \"instanceColor\":\n case \"particle_color\":\n case \"particle_texturemask\":\n this._type = NodeMaterialBlockConnectionPointTypes.Color4;\n return this._type;\n }\n }\n if (this.isSystemValue) {\n switch (this._systemValue) {\n case NodeMaterialSystemValues.World:\n case NodeMaterialSystemValues.WorldView:\n case NodeMaterialSystemValues.WorldViewProjection:\n case NodeMaterialSystemValues.View:\n case NodeMaterialSystemValues.ViewProjection:\n case NodeMaterialSystemValues.Projection:\n this._type = NodeMaterialBlockConnectionPointTypes.Matrix;\n return this._type;\n case NodeMaterialSystemValues.CameraPosition:\n this._type = NodeMaterialBlockConnectionPointTypes.Vector3;\n return this._type;\n case NodeMaterialSystemValues.FogColor:\n this._type = NodeMaterialBlockConnectionPointTypes.Color3;\n return this._type;\n case NodeMaterialSystemValues.DeltaTime:\n case NodeMaterialSystemValues.MaterialAlpha:\n this._type = NodeMaterialBlockConnectionPointTypes.Float;\n return this._type;\n case NodeMaterialSystemValues.CameraParameters:\n this._type = NodeMaterialBlockConnectionPointTypes.Vector4;\n return this._type;\n }\n }\n }\n return this._type;\n }\n /**\n * Creates a new InputBlock\n * @param name defines the block name\n * @param target defines the target of that block (Vertex by default)\n * @param type defines the type of the input (can be set to NodeMaterialBlockConnectionPointTypes.AutoDetect)\n */\n constructor(name, target = NodeMaterialBlockTargets.Vertex, type = NodeMaterialBlockConnectionPointTypes.AutoDetect) {\n super(name, target, false);\n this._mode = NodeMaterialBlockConnectionPointMode.Undefined;\n this._animationType = AnimatedInputBlockTypes.None;\n /** Gets or set a value used to limit the range of float values */\n this.min = 0;\n /** Gets or set a value used to limit the range of float values */\n this.max = 0;\n /** Gets or set a value indicating that this input can only get 0 and 1 values */\n this.isBoolean = false;\n /** Gets or sets a value used by the Node Material editor to determine how to configure the current value if it is a matrix */\n this.matrixMode = 0;\n /** @internal */\n this._systemValue = null;\n /** Gets or sets a boolean indicating that the value of this input will not change after a build */\n this.isConstant = false;\n /** Gets or sets the group to use to display this block in the Inspector */\n this.groupInInspector = \"\";\n /** Gets an observable raised when the value is changed */\n this.onValueChangedObservable = new Observable$1();\n /** Gets or sets a boolean indicating if content needs to be converted to gamma space (for color3/4 only) */\n this.convertToGammaSpace = false;\n /** Gets or sets a boolean indicating if content needs to be converted to linear space (for color3/4 only) */\n this.convertToLinearSpace = false;\n this._type = type;\n this.setDefaultValue();\n this.registerOutput(\"output\", type);\n }\n /**\n * Validates if a name is a reserve word.\n * @param newName the new name to be given to the node.\n * @returns false if the name is a reserve word, else true.\n */\n validateBlockName(newName) {\n if (!this.isAttribute) {\n return super.validateBlockName(newName);\n }\n return true;\n }\n /**\n * Gets the output component\n */\n get output() {\n return this._outputs[0];\n }\n /**\n * Set the source of this connection point to a vertex attribute\n * @param attributeName defines the attribute name (position, uv, normal, etc...). If not specified it will take the connection point name\n * @returns the current connection point\n */\n setAsAttribute(attributeName) {\n this._mode = NodeMaterialBlockConnectionPointMode.Attribute;\n if (attributeName) {\n this.name = attributeName;\n }\n return this;\n }\n /**\n * Set the source of this connection point to a system value\n * @param value define the system value to use (world, view, etc...) or null to switch to manual value\n * @returns the current connection point\n */\n setAsSystemValue(value) {\n this.systemValue = value;\n return this;\n }\n /**\n * Gets or sets the value of that point.\n * Please note that this value will be ignored if valueCallback is defined\n */\n get value() {\n return this._storedValue;\n }\n set value(value) {\n if (this.type === NodeMaterialBlockConnectionPointTypes.Float) {\n if (this.isBoolean) {\n value = value ? 1 : 0;\n }\n else if (this.min !== this.max) {\n value = Math.max(this.min, value);\n value = Math.min(this.max, value);\n }\n }\n this._storedValue = value;\n this._mode = NodeMaterialBlockConnectionPointMode.Uniform;\n this.onValueChangedObservable.notifyObservers(this);\n }\n /**\n * Gets or sets a callback used to get the value of that point.\n * Please note that setting this value will force the connection point to ignore the value property\n */\n get valueCallback() {\n return this._valueCallback;\n }\n set valueCallback(value) {\n this._valueCallback = value;\n this._mode = NodeMaterialBlockConnectionPointMode.Uniform;\n }\n /**\n * Gets or sets the associated variable name in the shader\n */\n get associatedVariableName() {\n return this._associatedVariableName;\n }\n set associatedVariableName(value) {\n this._associatedVariableName = value;\n }\n /** Gets or sets the type of animation applied to the input */\n get animationType() {\n return this._animationType;\n }\n set animationType(value) {\n this._animationType = value;\n }\n /**\n * Gets a boolean indicating that this connection point not defined yet\n */\n get isUndefined() {\n return this._mode === NodeMaterialBlockConnectionPointMode.Undefined;\n }\n /**\n * Gets or sets a boolean indicating that this connection point is coming from an uniform.\n * In this case the connection point name must be the name of the uniform to use.\n * Can only be set on inputs\n */\n get isUniform() {\n return this._mode === NodeMaterialBlockConnectionPointMode.Uniform;\n }\n set isUniform(value) {\n this._mode = value ? NodeMaterialBlockConnectionPointMode.Uniform : NodeMaterialBlockConnectionPointMode.Undefined;\n this.associatedVariableName = \"\";\n }\n /**\n * Gets or sets a boolean indicating that this connection point is coming from an attribute.\n * In this case the connection point name must be the name of the attribute to use\n * Can only be set on inputs\n */\n get isAttribute() {\n return this._mode === NodeMaterialBlockConnectionPointMode.Attribute;\n }\n set isAttribute(value) {\n this._mode = value ? NodeMaterialBlockConnectionPointMode.Attribute : NodeMaterialBlockConnectionPointMode.Undefined;\n this.associatedVariableName = \"\";\n }\n /**\n * Gets or sets a boolean indicating that this connection point is generating a varying variable.\n * Can only be set on exit points\n */\n get isVarying() {\n return this._mode === NodeMaterialBlockConnectionPointMode.Varying;\n }\n set isVarying(value) {\n this._mode = value ? NodeMaterialBlockConnectionPointMode.Varying : NodeMaterialBlockConnectionPointMode.Undefined;\n this.associatedVariableName = \"\";\n }\n /**\n * Gets a boolean indicating that the current connection point is a system value\n */\n get isSystemValue() {\n return this._systemValue != null;\n }\n /**\n * Gets or sets the current well known value or null if not defined as a system value\n */\n get systemValue() {\n return this._systemValue;\n }\n set systemValue(value) {\n this._mode = NodeMaterialBlockConnectionPointMode.Uniform;\n this.associatedVariableName = \"\";\n this._systemValue = value;\n }\n /**\n * Gets the current class name\n * @returns the class name\n */\n getClassName() {\n return \"InputBlock\";\n }\n /**\n * Animate the input if animationType !== None\n * @param scene defines the rendering scene\n */\n animate(scene) {\n switch (this._animationType) {\n case AnimatedInputBlockTypes.Time: {\n if (this.type === NodeMaterialBlockConnectionPointTypes.Float) {\n this.value += scene.getAnimationRatio() * 0.01;\n }\n break;\n }\n case AnimatedInputBlockTypes.RealTime: {\n if (this.type === NodeMaterialBlockConnectionPointTypes.Float) {\n this.value = (PrecisionDate.Now - scene.getEngine().startTime) / 1000;\n }\n break;\n }\n }\n }\n _emitDefine(define) {\n if (define[0] === \"!\") {\n return `#ifndef ${define.substring(1)}\\n`;\n }\n return `#ifdef ${define}\\n`;\n }\n initialize() {\n this.associatedVariableName = \"\";\n }\n /**\n * Set the input block to its default value (based on its type)\n */\n setDefaultValue() {\n switch (this.type) {\n case NodeMaterialBlockConnectionPointTypes.Float:\n this.value = 0;\n break;\n case NodeMaterialBlockConnectionPointTypes.Vector2:\n this.value = Vector2.Zero();\n break;\n case NodeMaterialBlockConnectionPointTypes.Vector3:\n this.value = Vector3.Zero();\n break;\n case NodeMaterialBlockConnectionPointTypes.Vector4:\n this.value = Vector4.Zero();\n break;\n case NodeMaterialBlockConnectionPointTypes.Color3:\n this.value = Color3.White();\n break;\n case NodeMaterialBlockConnectionPointTypes.Color4:\n this.value = new Color4(1, 1, 1, 1);\n break;\n case NodeMaterialBlockConnectionPointTypes.Matrix:\n this.value = Matrix.Identity();\n break;\n }\n }\n _emitConstant(state) {\n switch (this.type) {\n case NodeMaterialBlockConnectionPointTypes.Float:\n return `${state._emitFloat(this.value)}`;\n case NodeMaterialBlockConnectionPointTypes.Vector2:\n return `vec2(${this.value.x}, ${this.value.y})`;\n case NodeMaterialBlockConnectionPointTypes.Vector3:\n return `vec3(${this.value.x}, ${this.value.y}, ${this.value.z})`;\n case NodeMaterialBlockConnectionPointTypes.Vector4:\n return `vec4(${this.value.x}, ${this.value.y}, ${this.value.z}, ${this.value.w})`;\n case NodeMaterialBlockConnectionPointTypes.Color3:\n TmpColors.Color3[0].set(this.value.r, this.value.g, this.value.b);\n if (this.convertToGammaSpace) {\n TmpColors.Color3[0].toGammaSpaceToRef(TmpColors.Color3[0], state.sharedData.scene.getEngine().useExactSrgbConversions);\n }\n if (this.convertToLinearSpace) {\n TmpColors.Color3[0].toLinearSpaceToRef(TmpColors.Color3[0], state.sharedData.scene.getEngine().useExactSrgbConversions);\n }\n return `vec3(${TmpColors.Color3[0].r}, ${TmpColors.Color3[0].g}, ${TmpColors.Color3[0].b})`;\n case NodeMaterialBlockConnectionPointTypes.Color4:\n TmpColors.Color4[0].set(this.value.r, this.value.g, this.value.b, this.value.a);\n if (this.convertToGammaSpace) {\n TmpColors.Color4[0].toGammaSpaceToRef(TmpColors.Color4[0], state.sharedData.scene.getEngine().useExactSrgbConversions);\n }\n if (this.convertToLinearSpace) {\n TmpColors.Color4[0].toLinearSpaceToRef(TmpColors.Color4[0], state.sharedData.scene.getEngine().useExactSrgbConversions);\n }\n return `vec4(${TmpColors.Color4[0].r}, ${TmpColors.Color4[0].g}, ${TmpColors.Color4[0].b}, ${TmpColors.Color4[0].a})`;\n }\n return \"\";\n }\n /** @internal */\n get _noContextSwitch() {\n return attributeInFragmentOnly[this.name];\n }\n _emit(state, define) {\n var _a;\n // Uniforms\n if (this.isUniform) {\n if (!this.associatedVariableName) {\n this.associatedVariableName = state._getFreeVariableName(\"u_\" + this.name);\n }\n if (this.isConstant) {\n if (state.constants.indexOf(this.associatedVariableName) !== -1) {\n return;\n }\n state.constants.push(this.associatedVariableName);\n state._constantDeclaration += this._declareOutput(this.output, state) + ` = ${this._emitConstant(state)};\\n`;\n return;\n }\n if (state.uniforms.indexOf(this.associatedVariableName) !== -1) {\n return;\n }\n state.uniforms.push(this.associatedVariableName);\n if (define) {\n state._uniformDeclaration += this._emitDefine(define);\n }\n state._uniformDeclaration += `uniform ${state._getGLType(this.type)} ${this.associatedVariableName};\\n`;\n if (define) {\n state._uniformDeclaration += `#endif\\n`;\n }\n // well known\n const hints = state.sharedData.hints;\n if (this._systemValue !== null && this._systemValue !== undefined) {\n switch (this._systemValue) {\n case NodeMaterialSystemValues.WorldView:\n hints.needWorldViewMatrix = true;\n break;\n case NodeMaterialSystemValues.WorldViewProjection:\n hints.needWorldViewProjectionMatrix = true;\n break;\n }\n }\n else {\n if (this._animationType !== AnimatedInputBlockTypes.None) {\n state.sharedData.animatedInputs.push(this);\n }\n }\n return;\n }\n // Attribute\n if (this.isAttribute) {\n this.associatedVariableName = (_a = remapAttributeName[this.name]) !== null && _a !== void 0 ? _a : this.name;\n if (this.target === NodeMaterialBlockTargets.Vertex && state._vertexState) {\n // Attribute for fragment need to be carried over by varyings\n if (attributeInFragmentOnly[this.name]) {\n if (attributeAsUniform[this.name]) {\n state._emitUniformFromString(this.associatedVariableName, state._getGLType(this.type), define);\n }\n else {\n state._emitVaryingFromString(this.associatedVariableName, state._getGLType(this.type), define);\n }\n }\n else {\n this._emit(state._vertexState, define);\n }\n return;\n }\n if (state.attributes.indexOf(this.associatedVariableName) !== -1) {\n return;\n }\n state.attributes.push(this.associatedVariableName);\n if (attributeInFragmentOnly[this.name]) {\n if (attributeAsUniform[this.name]) {\n state._emitUniformFromString(this.associatedVariableName, state._getGLType(this.type), define);\n }\n else {\n state._emitVaryingFromString(this.associatedVariableName, state._getGLType(this.type), define);\n }\n }\n else {\n if (define) {\n state._attributeDeclaration += this._emitDefine(define);\n }\n state._attributeDeclaration += `attribute ${state._getGLType(this.type)} ${this.associatedVariableName};\\n`;\n if (define) {\n state._attributeDeclaration += `#endif\\n`;\n }\n }\n }\n }\n /**\n * @internal\n */\n _transmitWorld(effect, world, worldView, worldViewProjection) {\n if (!this._systemValue) {\n return;\n }\n const variableName = this.associatedVariableName;\n switch (this._systemValue) {\n case NodeMaterialSystemValues.World:\n effect.setMatrix(variableName, world);\n break;\n case NodeMaterialSystemValues.WorldView:\n effect.setMatrix(variableName, worldView);\n break;\n case NodeMaterialSystemValues.WorldViewProjection:\n effect.setMatrix(variableName, worldViewProjection);\n break;\n }\n }\n /**\n * @internal\n */\n _transmit(effect, scene, material) {\n if (this.isAttribute) {\n return;\n }\n const variableName = this.associatedVariableName;\n if (this._systemValue) {\n switch (this._systemValue) {\n case NodeMaterialSystemValues.World:\n case NodeMaterialSystemValues.WorldView:\n case NodeMaterialSystemValues.WorldViewProjection:\n return;\n case NodeMaterialSystemValues.View:\n effect.setMatrix(variableName, scene.getViewMatrix());\n break;\n case NodeMaterialSystemValues.Projection:\n effect.setMatrix(variableName, scene.getProjectionMatrix());\n break;\n case NodeMaterialSystemValues.ViewProjection:\n effect.setMatrix(variableName, scene.getTransformMatrix());\n break;\n case NodeMaterialSystemValues.CameraPosition:\n scene.bindEyePosition(effect, variableName, true);\n break;\n case NodeMaterialSystemValues.FogColor:\n effect.setColor3(variableName, scene.fogColor);\n break;\n case NodeMaterialSystemValues.DeltaTime:\n effect.setFloat(variableName, scene.deltaTime / 1000.0);\n break;\n case NodeMaterialSystemValues.CameraParameters:\n if (scene.activeCamera) {\n effect.setFloat4(variableName, scene.getEngine().hasOriginBottomLeft ? -1 : 1, scene.activeCamera.minZ, scene.activeCamera.maxZ, 1 / scene.activeCamera.maxZ);\n }\n break;\n case NodeMaterialSystemValues.MaterialAlpha:\n effect.setFloat(variableName, material.alpha);\n break;\n }\n return;\n }\n const value = this._valueCallback ? this._valueCallback() : this._storedValue;\n if (value === null) {\n return;\n }\n switch (this.type) {\n case NodeMaterialBlockConnectionPointTypes.Float:\n effect.setFloat(variableName, value);\n break;\n case NodeMaterialBlockConnectionPointTypes.Int:\n effect.setInt(variableName, value);\n break;\n case NodeMaterialBlockConnectionPointTypes.Color3:\n TmpColors.Color3[0].set(this.value.r, this.value.g, this.value.b);\n if (this.convertToGammaSpace) {\n TmpColors.Color3[0].toGammaSpaceToRef(TmpColors.Color3[0], scene.getEngine().useExactSrgbConversions);\n }\n if (this.convertToLinearSpace) {\n TmpColors.Color3[0].toLinearSpaceToRef(TmpColors.Color3[0], scene.getEngine().useExactSrgbConversions);\n }\n effect.setColor3(variableName, TmpColors.Color3[0]);\n break;\n case NodeMaterialBlockConnectionPointTypes.Color4:\n TmpColors.Color4[0].set(this.value.r, this.value.g, this.value.b, this.value.a);\n if (this.convertToGammaSpace) {\n TmpColors.Color4[0].toGammaSpaceToRef(TmpColors.Color4[0], scene.getEngine().useExactSrgbConversions);\n }\n if (this.convertToLinearSpace) {\n TmpColors.Color4[0].toLinearSpaceToRef(TmpColors.Color4[0], scene.getEngine().useExactSrgbConversions);\n }\n effect.setDirectColor4(variableName, TmpColors.Color4[0]);\n break;\n case NodeMaterialBlockConnectionPointTypes.Vector2:\n effect.setVector2(variableName, value);\n break;\n case NodeMaterialBlockConnectionPointTypes.Vector3:\n effect.setVector3(variableName, value);\n break;\n case NodeMaterialBlockConnectionPointTypes.Vector4:\n effect.setVector4(variableName, value);\n break;\n case NodeMaterialBlockConnectionPointTypes.Matrix:\n effect.setMatrix(variableName, value);\n break;\n }\n }\n _buildBlock(state) {\n super._buildBlock(state);\n if (this.isUniform || this.isSystemValue) {\n state.sharedData.inputBlocks.push(this);\n }\n this._emit(state);\n }\n _dumpPropertiesCode() {\n const variableName = this._codeVariableName;\n if (this.isAttribute) {\n return super._dumpPropertiesCode() + `${variableName}.setAsAttribute(\"${this.name}\");\\n`;\n }\n if (this.isSystemValue) {\n return super._dumpPropertiesCode() + `${variableName}.setAsSystemValue(BABYLON.NodeMaterialSystemValues.${NodeMaterialSystemValues[this._systemValue]});\\n`;\n }\n if (this.isUniform) {\n const codes = [];\n let valueString = \"\";\n switch (this.type) {\n case NodeMaterialBlockConnectionPointTypes.Float:\n valueString = `${this.value}`;\n break;\n case NodeMaterialBlockConnectionPointTypes.Vector2:\n valueString = `new BABYLON.Vector2(${this.value.x}, ${this.value.y})`;\n break;\n case NodeMaterialBlockConnectionPointTypes.Vector3:\n valueString = `new BABYLON.Vector3(${this.value.x}, ${this.value.y}, ${this.value.z})`;\n break;\n case NodeMaterialBlockConnectionPointTypes.Vector4:\n valueString = `new BABYLON.Vector4(${this.value.x}, ${this.value.y}, ${this.value.z}, ${this.value.w})`;\n break;\n case NodeMaterialBlockConnectionPointTypes.Color3:\n valueString = `new BABYLON.Color3(${this.value.r}, ${this.value.g}, ${this.value.b})`;\n if (this.convertToGammaSpace) {\n valueString += \".toGammaSpace()\";\n }\n if (this.convertToLinearSpace) {\n valueString += \".toLinearSpace()\";\n }\n break;\n case NodeMaterialBlockConnectionPointTypes.Color4:\n valueString = `new BABYLON.Color4(${this.value.r}, ${this.value.g}, ${this.value.b}, ${this.value.a})`;\n if (this.convertToGammaSpace) {\n valueString += \".toGammaSpace()\";\n }\n if (this.convertToLinearSpace) {\n valueString += \".toLinearSpace()\";\n }\n break;\n case NodeMaterialBlockConnectionPointTypes.Matrix:\n valueString = `BABYLON.Matrix.FromArray([${this.value.m}])`;\n break;\n }\n // Common Property \"Value\"\n codes.push(`${variableName}.value = ${valueString}`);\n // Float-Value-Specific Properties\n if (this.type === NodeMaterialBlockConnectionPointTypes.Float) {\n codes.push(`${variableName}.min = ${this.min}`, `${variableName}.max = ${this.max}`, `${variableName}.isBoolean = ${this.isBoolean}`, `${variableName}.matrixMode = ${this.matrixMode}`, `${variableName}.animationType = BABYLON.AnimatedInputBlockTypes.${AnimatedInputBlockTypes[this.animationType]}`);\n }\n // Common Property \"Type\"\n codes.push(`${variableName}.isConstant = ${this.isConstant}`);\n codes.push(\"\");\n return super._dumpPropertiesCode() + codes.join(\";\\n\");\n }\n return super._dumpPropertiesCode();\n }\n dispose() {\n this.onValueChangedObservable.clear();\n super.dispose();\n }\n serialize() {\n const serializationObject = super.serialize();\n serializationObject.type = this.type;\n serializationObject.mode = this._mode;\n serializationObject.systemValue = this._systemValue;\n serializationObject.animationType = this._animationType;\n serializationObject.min = this.min;\n serializationObject.max = this.max;\n serializationObject.isBoolean = this.isBoolean;\n serializationObject.matrixMode = this.matrixMode;\n serializationObject.isConstant = this.isConstant;\n serializationObject.groupInInspector = this.groupInInspector;\n serializationObject.convertToGammaSpace = this.convertToGammaSpace;\n serializationObject.convertToLinearSpace = this.convertToLinearSpace;\n if (this._storedValue != null && this._mode === NodeMaterialBlockConnectionPointMode.Uniform) {\n if (this._storedValue.asArray) {\n serializationObject.valueType = \"BABYLON.\" + this._storedValue.getClassName();\n serializationObject.value = this._storedValue.asArray();\n }\n else {\n serializationObject.valueType = \"number\";\n serializationObject.value = this._storedValue;\n }\n }\n return serializationObject;\n }\n _deserialize(serializationObject, scene, rootUrl) {\n this._mode = serializationObject.mode;\n super._deserialize(serializationObject, scene, rootUrl);\n this._type = serializationObject.type;\n this._systemValue = serializationObject.systemValue || serializationObject.wellKnownValue;\n this._animationType = serializationObject.animationType;\n this.min = serializationObject.min || 0;\n this.max = serializationObject.max || 0;\n this.isBoolean = !!serializationObject.isBoolean;\n this.matrixMode = serializationObject.matrixMode || 0;\n this.isConstant = !!serializationObject.isConstant;\n this.groupInInspector = serializationObject.groupInInspector || \"\";\n this.convertToGammaSpace = !!serializationObject.convertToGammaSpace;\n this.convertToLinearSpace = !!serializationObject.convertToLinearSpace;\n // Tangents back compat\n if (serializationObject.name === \"tangent\" &&\n serializationObject.mode === NodeMaterialBlockConnectionPointMode.Attribute &&\n serializationObject.type === NodeMaterialBlockConnectionPointTypes.Vector3) {\n this._type = NodeMaterialBlockConnectionPointTypes.Vector4;\n }\n if (!serializationObject.valueType) {\n return;\n }\n if (serializationObject.valueType === \"number\") {\n this._storedValue = serializationObject.value;\n }\n else {\n const valueType = GetClass(serializationObject.valueType);\n if (valueType) {\n this._storedValue = valueType.FromArray(serializationObject.value);\n }\n }\n }\n }\n RegisterClass(\"BABYLON.InputBlock\", InputBlock);\n\n /**\n * Base block used as input for post process\n */\n class CurrentScreenBlock extends NodeMaterialBlock {\n /**\n * Create a new CurrentScreenBlock\n * @param name defines the block name\n */\n constructor(name) {\n super(name, NodeMaterialBlockTargets.VertexAndFragment);\n this._samplerName = \"textureSampler\";\n /**\n * Gets or sets a boolean indicating if content needs to be converted to gamma space\n */\n this.convertToGammaSpace = false;\n /**\n * Gets or sets a boolean indicating if content needs to be converted to linear space\n */\n this.convertToLinearSpace = false;\n this._isUnique = false;\n this.registerInput(\"uv\", NodeMaterialBlockConnectionPointTypes.AutoDetect, false, NodeMaterialBlockTargets.VertexAndFragment);\n this.registerOutput(\"rgba\", NodeMaterialBlockConnectionPointTypes.Color4, NodeMaterialBlockTargets.Neutral);\n this.registerOutput(\"rgb\", NodeMaterialBlockConnectionPointTypes.Color3, NodeMaterialBlockTargets.Neutral);\n this.registerOutput(\"r\", NodeMaterialBlockConnectionPointTypes.Float, NodeMaterialBlockTargets.Neutral);\n this.registerOutput(\"g\", NodeMaterialBlockConnectionPointTypes.Float, NodeMaterialBlockTargets.Neutral);\n this.registerOutput(\"b\", NodeMaterialBlockConnectionPointTypes.Float, NodeMaterialBlockTargets.Neutral);\n this.registerOutput(\"a\", NodeMaterialBlockConnectionPointTypes.Float, NodeMaterialBlockTargets.Neutral);\n this._inputs[0].addExcludedConnectionPointFromAllowedTypes(NodeMaterialBlockConnectionPointTypes.Vector2 | NodeMaterialBlockConnectionPointTypes.Vector3 | NodeMaterialBlockConnectionPointTypes.Vector4);\n this._inputs[0]._prioritizeVertex = false;\n }\n /**\n * Gets the current class name\n * @returns the class name\n */\n getClassName() {\n return \"CurrentScreenBlock\";\n }\n /**\n * Gets the uv input component\n */\n get uv() {\n return this._inputs[0];\n }\n /**\n * Gets the rgba output component\n */\n get rgba() {\n return this._outputs[0];\n }\n /**\n * Gets the rgb output component\n */\n get rgb() {\n return this._outputs[1];\n }\n /**\n * Gets the r output component\n */\n get r() {\n return this._outputs[2];\n }\n /**\n * Gets the g output component\n */\n get g() {\n return this._outputs[3];\n }\n /**\n * Gets the b output component\n */\n get b() {\n return this._outputs[4];\n }\n /**\n * Gets the a output component\n */\n get a() {\n return this._outputs[5];\n }\n /**\n * Initialize the block and prepare the context for build\n * @param state defines the state that will be used for the build\n */\n initialize(state) {\n state._excludeVariableName(\"textureSampler\");\n }\n get target() {\n if (!this.uv.isConnected) {\n return NodeMaterialBlockTargets.VertexAndFragment;\n }\n if (this.uv.sourceBlock.isInput) {\n return NodeMaterialBlockTargets.VertexAndFragment;\n }\n return NodeMaterialBlockTargets.Fragment;\n }\n prepareDefines(mesh, nodeMaterial, defines) {\n defines.setValue(this._linearDefineName, this.convertToGammaSpace, true);\n defines.setValue(this._gammaDefineName, this.convertToLinearSpace, true);\n }\n isReady() {\n if (this.texture && !this.texture.isReadyOrNotBlocking()) {\n return false;\n }\n return true;\n }\n _injectVertexCode(state) {\n const uvInput = this.uv;\n if (uvInput.connectedPoint.ownerBlock.isInput) {\n const uvInputOwnerBlock = uvInput.connectedPoint.ownerBlock;\n if (!uvInputOwnerBlock.isAttribute) {\n state._emitUniformFromString(uvInput.associatedVariableName, \"vec2\");\n }\n }\n this._mainUVName = \"vMain\" + uvInput.associatedVariableName;\n state._emitVaryingFromString(this._mainUVName, \"vec2\");\n state.compilationString += `${this._mainUVName} = ${uvInput.associatedVariableName}.xy;\\n`;\n if (!this._outputs.some((o) => o.isConnectedInVertexShader)) {\n return;\n }\n this._writeTextureRead(state, true);\n for (const output of this._outputs) {\n if (output.hasEndpoints) {\n this._writeOutput(state, output, output.name, true);\n }\n }\n }\n _writeTextureRead(state, vertexMode = false) {\n const uvInput = this.uv;\n if (vertexMode) {\n if (state.target === NodeMaterialBlockTargets.Fragment) {\n return;\n }\n state.compilationString += `vec4 ${this._tempTextureRead} = texture2D(${this._samplerName}, ${uvInput.associatedVariableName});\\n`;\n return;\n }\n if (this.uv.ownerBlock.target === NodeMaterialBlockTargets.Fragment) {\n state.compilationString += `vec4 ${this._tempTextureRead} = texture2D(${this._samplerName}, ${uvInput.associatedVariableName});\\n`;\n return;\n }\n state.compilationString += `vec4 ${this._tempTextureRead} = texture2D(${this._samplerName}, ${this._mainUVName});\\n`;\n }\n _writeOutput(state, output, swizzle, vertexMode = false) {\n if (vertexMode) {\n if (state.target === NodeMaterialBlockTargets.Fragment) {\n return;\n }\n state.compilationString += `${this._declareOutput(output, state)} = ${this._tempTextureRead}.${swizzle};\\n`;\n return;\n }\n if (this.uv.ownerBlock.target === NodeMaterialBlockTargets.Fragment) {\n state.compilationString += `${this._declareOutput(output, state)} = ${this._tempTextureRead}.${swizzle};\\n`;\n return;\n }\n state.compilationString += `${this._declareOutput(output, state)} = ${this._tempTextureRead}.${swizzle};\\n`;\n state.compilationString += `#ifdef ${this._linearDefineName}\\n`;\n state.compilationString += `${output.associatedVariableName} = toGammaSpace(${output.associatedVariableName});\\n`;\n state.compilationString += `#endif\\n`;\n state.compilationString += `#ifdef ${this._gammaDefineName}\\n`;\n state.compilationString += `${output.associatedVariableName} = toLinearSpace(${output.associatedVariableName});\\n`;\n state.compilationString += `#endif\\n`;\n }\n _buildBlock(state) {\n super._buildBlock(state);\n this._tempTextureRead = state._getFreeVariableName(\"tempTextureRead\");\n if (state.sharedData.blockingBlocks.indexOf(this) < 0) {\n state.sharedData.blockingBlocks.push(this);\n }\n if (state.sharedData.textureBlocks.indexOf(this) < 0) {\n state.sharedData.textureBlocks.push(this);\n }\n if (state.sharedData.blocksWithDefines.indexOf(this) < 0) {\n state.sharedData.blocksWithDefines.push(this);\n }\n if (state.target !== NodeMaterialBlockTargets.Fragment) {\n // Vertex\n state._emit2DSampler(this._samplerName);\n this._injectVertexCode(state);\n return;\n }\n // Fragment\n if (!this._outputs.some((o) => o.isConnectedInFragmentShader)) {\n return;\n }\n state._emit2DSampler(this._samplerName);\n this._linearDefineName = state._getFreeDefineName(\"ISLINEAR\");\n this._gammaDefineName = state._getFreeDefineName(\"ISGAMMA\");\n const comments = `//${this.name}`;\n state._emitFunctionFromInclude(\"helperFunctions\", comments);\n this._writeTextureRead(state);\n for (const output of this._outputs) {\n if (output.hasEndpoints) {\n this._writeOutput(state, output, output.name);\n }\n }\n return this;\n }\n serialize() {\n const serializationObject = super.serialize();\n serializationObject.convertToGammaSpace = this.convertToGammaSpace;\n serializationObject.convertToLinearSpace = this.convertToLinearSpace;\n if (this.texture && !this.texture.isRenderTarget) {\n serializationObject.texture = this.texture.serialize();\n }\n return serializationObject;\n }\n _deserialize(serializationObject, scene, rootUrl) {\n super._deserialize(serializationObject, scene, rootUrl);\n this.convertToGammaSpace = serializationObject.convertToGammaSpace;\n this.convertToLinearSpace = !!serializationObject.convertToLinearSpace;\n if (serializationObject.texture) {\n rootUrl = serializationObject.texture.url.indexOf(\"data:\") === 0 ? \"\" : rootUrl;\n this.texture = Texture.Parse(serializationObject.texture, scene, rootUrl);\n }\n }\n }\n RegisterClass(\"BABYLON.CurrentScreenBlock\", CurrentScreenBlock);\n\n /**\n * Base block used for the particle texture\n */\n class ParticleTextureBlock extends NodeMaterialBlock {\n /**\n * Create a new ParticleTextureBlock\n * @param name defines the block name\n */\n constructor(name) {\n super(name, NodeMaterialBlockTargets.Fragment);\n this._samplerName = \"diffuseSampler\";\n /**\n * Gets or sets a boolean indicating if content needs to be converted to gamma space\n */\n this.convertToGammaSpace = false;\n /**\n * Gets or sets a boolean indicating if content needs to be converted to linear space\n */\n this.convertToLinearSpace = false;\n this._isUnique = false;\n this.registerInput(\"uv\", NodeMaterialBlockConnectionPointTypes.AutoDetect, false, NodeMaterialBlockTargets.VertexAndFragment);\n this.registerOutput(\"rgba\", NodeMaterialBlockConnectionPointTypes.Color4, NodeMaterialBlockTargets.Neutral);\n this.registerOutput(\"rgb\", NodeMaterialBlockConnectionPointTypes.Color3, NodeMaterialBlockTargets.Neutral);\n this.registerOutput(\"r\", NodeMaterialBlockConnectionPointTypes.Float, NodeMaterialBlockTargets.Neutral);\n this.registerOutput(\"g\", NodeMaterialBlockConnectionPointTypes.Float, NodeMaterialBlockTargets.Neutral);\n this.registerOutput(\"b\", NodeMaterialBlockConnectionPointTypes.Float, NodeMaterialBlockTargets.Neutral);\n this.registerOutput(\"a\", NodeMaterialBlockConnectionPointTypes.Float, NodeMaterialBlockTargets.Neutral);\n this._inputs[0].addExcludedConnectionPointFromAllowedTypes(NodeMaterialBlockConnectionPointTypes.Vector2 | NodeMaterialBlockConnectionPointTypes.Vector3 | NodeMaterialBlockConnectionPointTypes.Vector4);\n }\n /**\n * Gets the current class name\n * @returns the class name\n */\n getClassName() {\n return \"ParticleTextureBlock\";\n }\n /**\n * Gets the uv input component\n */\n get uv() {\n return this._inputs[0];\n }\n /**\n * Gets the rgba output component\n */\n get rgba() {\n return this._outputs[0];\n }\n /**\n * Gets the rgb output component\n */\n get rgb() {\n return this._outputs[1];\n }\n /**\n * Gets the r output component\n */\n get r() {\n return this._outputs[2];\n }\n /**\n * Gets the g output component\n */\n get g() {\n return this._outputs[3];\n }\n /**\n * Gets the b output component\n */\n get b() {\n return this._outputs[4];\n }\n /**\n * Gets the a output component\n */\n get a() {\n return this._outputs[5];\n }\n /**\n * Initialize the block and prepare the context for build\n * @param state defines the state that will be used for the build\n */\n initialize(state) {\n state._excludeVariableName(\"diffuseSampler\");\n }\n autoConfigure(material) {\n if (!this.uv.isConnected) {\n let uvInput = material.getInputBlockByPredicate((b) => b.isAttribute && b.name === \"particle_uv\");\n if (!uvInput) {\n uvInput = new InputBlock(\"uv\");\n uvInput.setAsAttribute(\"particle_uv\");\n }\n uvInput.output.connectTo(this.uv);\n }\n }\n prepareDefines(mesh, nodeMaterial, defines) {\n defines.setValue(this._linearDefineName, this.convertToGammaSpace, true);\n defines.setValue(this._gammaDefineName, this.convertToLinearSpace, true);\n }\n isReady() {\n if (this.texture && !this.texture.isReadyOrNotBlocking()) {\n return false;\n }\n return true;\n }\n _writeOutput(state, output, swizzle) {\n state.compilationString += `${this._declareOutput(output, state)} = ${this._tempTextureRead}.${swizzle};\\n`;\n state.compilationString += `#ifdef ${this._linearDefineName}\\n`;\n state.compilationString += `${output.associatedVariableName} = toGammaSpace(${output.associatedVariableName});\\n`;\n state.compilationString += `#endif\\n`;\n state.compilationString += `#ifdef ${this._gammaDefineName}\\n`;\n state.compilationString += `${output.associatedVariableName} = toLinearSpace(${output.associatedVariableName});\\n`;\n state.compilationString += `#endif\\n`;\n }\n _buildBlock(state) {\n super._buildBlock(state);\n if (state.target === NodeMaterialBlockTargets.Vertex) {\n return;\n }\n this._tempTextureRead = state._getFreeVariableName(\"tempTextureRead\");\n state._emit2DSampler(this._samplerName);\n state.sharedData.blockingBlocks.push(this);\n state.sharedData.textureBlocks.push(this);\n state.sharedData.blocksWithDefines.push(this);\n this._linearDefineName = state._getFreeDefineName(\"ISLINEAR\");\n this._gammaDefineName = state._getFreeDefineName(\"ISGAMMA\");\n const comments = `//${this.name}`;\n state._emitFunctionFromInclude(\"helperFunctions\", comments);\n state.compilationString += `vec4 ${this._tempTextureRead} = texture2D(${this._samplerName}, ${this.uv.associatedVariableName});\\n`;\n for (const output of this._outputs) {\n if (output.hasEndpoints) {\n this._writeOutput(state, output, output.name);\n }\n }\n return this;\n }\n serialize() {\n const serializationObject = super.serialize();\n serializationObject.convertToGammaSpace = this.convertToGammaSpace;\n serializationObject.convertToLinearSpace = this.convertToLinearSpace;\n if (this.texture && !this.texture.isRenderTarget) {\n serializationObject.texture = this.texture.serialize();\n }\n return serializationObject;\n }\n _deserialize(serializationObject, scene, rootUrl) {\n super._deserialize(serializationObject, scene, rootUrl);\n this.convertToGammaSpace = serializationObject.convertToGammaSpace;\n this.convertToLinearSpace = !!serializationObject.convertToLinearSpace;\n if (serializationObject.texture) {\n rootUrl = serializationObject.texture.url.indexOf(\"data:\") === 0 ? \"\" : rootUrl;\n this.texture = Texture.Parse(serializationObject.texture, scene, rootUrl);\n }\n }\n }\n RegisterClass(\"BABYLON.ParticleTextureBlock\", ParticleTextureBlock);\n\n /**\n * Block used for the particle ramp gradient section\n */\n class ParticleRampGradientBlock extends NodeMaterialBlock {\n /**\n * Create a new ParticleRampGradientBlock\n * @param name defines the block name\n */\n constructor(name) {\n super(name, NodeMaterialBlockTargets.Fragment);\n this._isUnique = true;\n this.registerInput(\"color\", NodeMaterialBlockConnectionPointTypes.Color4, false, NodeMaterialBlockTargets.Fragment);\n this.registerOutput(\"rampColor\", NodeMaterialBlockConnectionPointTypes.Color4, NodeMaterialBlockTargets.Fragment);\n }\n /**\n * Gets the current class name\n * @returns the class name\n */\n getClassName() {\n return \"ParticleRampGradientBlock\";\n }\n /**\n * Gets the color input component\n */\n get color() {\n return this._inputs[0];\n }\n /**\n * Gets the rampColor output component\n */\n get rampColor() {\n return this._outputs[0];\n }\n /**\n * Initialize the block and prepare the context for build\n * @param state defines the state that will be used for the build\n */\n initialize(state) {\n state._excludeVariableName(\"remapRanges\");\n state._excludeVariableName(\"rampSampler\");\n state._excludeVariableName(\"baseColor\");\n state._excludeVariableName(\"alpha\");\n state._excludeVariableName(\"remappedColorIndex\");\n state._excludeVariableName(\"rampColor\");\n state._excludeVariableName(\"finalAlpha\");\n }\n _buildBlock(state) {\n super._buildBlock(state);\n if (state.target === NodeMaterialBlockTargets.Vertex) {\n return;\n }\n state._emit2DSampler(\"rampSampler\");\n state._emitVaryingFromString(\"remapRanges\", \"vec4\", \"RAMPGRADIENT\");\n state.compilationString += `\r\n #ifdef RAMPGRADIENT\r\n vec4 baseColor = ${this.color.associatedVariableName};\r\n float alpha = ${this.color.associatedVariableName}.a;\r\n\r\n float remappedColorIndex = clamp((alpha - remapRanges.x) / remapRanges.y, 0.0, 1.0);\r\n\r\n vec4 rampColor = texture2D(rampSampler, vec2(1.0 - remappedColorIndex, 0.));\r\n baseColor.rgb *= rampColor.rgb;\r\n\r\n // Remapped alpha\r\n float finalAlpha = baseColor.a;\r\n baseColor.a = clamp((alpha * rampColor.a - remapRanges.z) / remapRanges.w, 0.0, 1.0);\r\n\r\n ${this._declareOutput(this.rampColor, state)} = baseColor;\r\n #else\r\n ${this._declareOutput(this.rampColor, state)} = ${this.color.associatedVariableName};\r\n #endif\r\n `;\n return this;\n }\n }\n RegisterClass(\"BABYLON.ParticleRampGradientBlock\", ParticleRampGradientBlock);\n\n /**\n * Block used for the particle blend multiply section\n */\n class ParticleBlendMultiplyBlock extends NodeMaterialBlock {\n /**\n * Create a new ParticleBlendMultiplyBlock\n * @param name defines the block name\n */\n constructor(name) {\n super(name, NodeMaterialBlockTargets.Fragment);\n this._isUnique = true;\n this.registerInput(\"color\", NodeMaterialBlockConnectionPointTypes.Color4, false, NodeMaterialBlockTargets.Fragment);\n this.registerInput(\"alphaTexture\", NodeMaterialBlockConnectionPointTypes.Float, false, NodeMaterialBlockTargets.Fragment);\n this.registerInput(\"alphaColor\", NodeMaterialBlockConnectionPointTypes.Float, false, NodeMaterialBlockTargets.Fragment);\n this.registerOutput(\"blendColor\", NodeMaterialBlockConnectionPointTypes.Color4, NodeMaterialBlockTargets.Fragment);\n }\n /**\n * Gets the current class name\n * @returns the class name\n */\n getClassName() {\n return \"ParticleBlendMultiplyBlock\";\n }\n /**\n * Gets the color input component\n */\n get color() {\n return this._inputs[0];\n }\n /**\n * Gets the alphaTexture input component\n */\n get alphaTexture() {\n return this._inputs[1];\n }\n /**\n * Gets the alphaColor input component\n */\n get alphaColor() {\n return this._inputs[2];\n }\n /**\n * Gets the blendColor output component\n */\n get blendColor() {\n return this._outputs[0];\n }\n /**\n * Initialize the block and prepare the context for build\n * @param state defines the state that will be used for the build\n */\n initialize(state) {\n state._excludeVariableName(\"sourceAlpha\");\n }\n _buildBlock(state) {\n super._buildBlock(state);\n if (state.target === NodeMaterialBlockTargets.Vertex) {\n return;\n }\n state.compilationString += `\r\n #ifdef BLENDMULTIPLYMODE\r\n ${this._declareOutput(this.blendColor, state)};\r\n float sourceAlpha = ${this.alphaColor.associatedVariableName} * ${this.alphaTexture.associatedVariableName};\r\n ${this.blendColor.associatedVariableName}.rgb = ${this.color.associatedVariableName}.rgb * sourceAlpha + vec3(1.0) * (1.0 - sourceAlpha);\r\n ${this.blendColor.associatedVariableName}.a = ${this.color.associatedVariableName}.a;\r\n #else\r\n ${this._declareOutput(this.blendColor, state)} = ${this.color.associatedVariableName};\r\n #endif\r\n `;\n return this;\n }\n }\n RegisterClass(\"BABYLON.ParticleBlendMultiplyBlock\", ParticleBlendMultiplyBlock);\n\n /**\n * Block used to create a Vector2/3/4 out of individual inputs (one for each component)\n */\n class VectorMergerBlock extends NodeMaterialBlock {\n /**\n * Create a new VectorMergerBlock\n * @param name defines the block name\n */\n constructor(name) {\n super(name, NodeMaterialBlockTargets.Neutral);\n /**\n * Gets or sets the swizzle for x (meaning which component to affect to the output.x)\n */\n this.xSwizzle = \"x\";\n /**\n * Gets or sets the swizzle for y (meaning which component to affect to the output.y)\n */\n this.ySwizzle = \"y\";\n /**\n * Gets or sets the swizzle for z (meaning which component to affect to the output.z)\n */\n this.zSwizzle = \"z\";\n /**\n * Gets or sets the swizzle for w (meaning which component to affect to the output.w)\n */\n this.wSwizzle = \"w\";\n this.registerInput(\"xyzw \", NodeMaterialBlockConnectionPointTypes.Vector4, true);\n this.registerInput(\"xyz \", NodeMaterialBlockConnectionPointTypes.Vector3, true);\n this.registerInput(\"xy \", NodeMaterialBlockConnectionPointTypes.Vector2, true);\n this.registerInput(\"zw \", NodeMaterialBlockConnectionPointTypes.Vector2, true);\n this.registerInput(\"x\", NodeMaterialBlockConnectionPointTypes.Float, true);\n this.registerInput(\"y\", NodeMaterialBlockConnectionPointTypes.Float, true);\n this.registerInput(\"z\", NodeMaterialBlockConnectionPointTypes.Float, true);\n this.registerInput(\"w\", NodeMaterialBlockConnectionPointTypes.Float, true);\n this.registerOutput(\"xyzw\", NodeMaterialBlockConnectionPointTypes.Vector4);\n this.registerOutput(\"xyz\", NodeMaterialBlockConnectionPointTypes.Vector3);\n this.registerOutput(\"xy\", NodeMaterialBlockConnectionPointTypes.Vector2);\n this.registerOutput(\"zw\", NodeMaterialBlockConnectionPointTypes.Vector2);\n }\n /**\n * Gets the current class name\n * @returns the class name\n */\n getClassName() {\n return \"VectorMergerBlock\";\n }\n /**\n * Gets the xyzw component (input)\n */\n get xyzwIn() {\n return this._inputs[0];\n }\n /**\n * Gets the xyz component (input)\n */\n get xyzIn() {\n return this._inputs[1];\n }\n /**\n * Gets the xy component (input)\n */\n get xyIn() {\n return this._inputs[2];\n }\n /**\n * Gets the zw component (input)\n */\n get zwIn() {\n return this._inputs[3];\n }\n /**\n * Gets the x component (input)\n */\n get x() {\n return this._inputs[4];\n }\n /**\n * Gets the y component (input)\n */\n get y() {\n return this._inputs[5];\n }\n /**\n * Gets the z component (input)\n */\n get z() {\n return this._inputs[6];\n }\n /**\n * Gets the w component (input)\n */\n get w() {\n return this._inputs[7];\n }\n /**\n * Gets the xyzw component (output)\n */\n get xyzw() {\n return this._outputs[0];\n }\n /**\n * Gets the xyz component (output)\n */\n get xyzOut() {\n return this._outputs[1];\n }\n /**\n * Gets the xy component (output)\n */\n get xyOut() {\n return this._outputs[2];\n }\n /**\n * Gets the zw component (output)\n */\n get zwOut() {\n return this._outputs[3];\n }\n /**\n * Gets the xy component (output)\n * @deprecated Please use xyOut instead.\n */\n get xy() {\n return this.xyOut;\n }\n /**\n * Gets the xyz component (output)\n * @deprecated Please use xyzOut instead.\n */\n get xyz() {\n return this.xyzOut;\n }\n _inputRename(name) {\n if (name === \"xyzw \") {\n return \"xyzwIn\";\n }\n if (name === \"xyz \") {\n return \"xyzIn\";\n }\n if (name === \"xy \") {\n return \"xyIn\";\n }\n if (name === \"zw \") {\n return \"zwIn\";\n }\n return name;\n }\n _buildSwizzle(len) {\n const swizzle = this.xSwizzle + this.ySwizzle + this.zSwizzle + this.wSwizzle;\n return \".\" + swizzle.substr(0, len);\n }\n _buildBlock(state) {\n super._buildBlock(state);\n const xInput = this.x;\n const yInput = this.y;\n const zInput = this.z;\n const wInput = this.w;\n const xyInput = this.xyIn;\n const zwInput = this.zwIn;\n const xyzInput = this.xyzIn;\n const xyzwInput = this.xyzwIn;\n const v4Output = this._outputs[0];\n const v3Output = this._outputs[1];\n const v2Output = this._outputs[2];\n const v2CompOutput = this._outputs[3];\n if (xyzwInput.isConnected) {\n if (v4Output.hasEndpoints) {\n state.compilationString += this._declareOutput(v4Output, state) + ` = ${xyzwInput.associatedVariableName}${this._buildSwizzle(4)};\\n`;\n }\n if (v3Output.hasEndpoints) {\n state.compilationString += this._declareOutput(v3Output, state) + ` = ${xyzwInput.associatedVariableName}${this._buildSwizzle(3)};\\n`;\n }\n if (v2Output.hasEndpoints) {\n state.compilationString += this._declareOutput(v2Output, state) + ` = ${xyzwInput.associatedVariableName}${this._buildSwizzle(2)};\\n`;\n }\n }\n else if (xyzInput.isConnected) {\n if (v4Output.hasEndpoints) {\n state.compilationString +=\n this._declareOutput(v4Output, state) +\n ` = vec4(${xyzInput.associatedVariableName}, ${wInput.isConnected ? this._writeVariable(wInput) : \"0.0\"})${this._buildSwizzle(4)};\\n`;\n }\n if (v3Output.hasEndpoints) {\n state.compilationString += this._declareOutput(v3Output, state) + ` = ${xyzInput.associatedVariableName}${this._buildSwizzle(3)};\\n`;\n }\n if (v2Output.hasEndpoints) {\n state.compilationString += this._declareOutput(v2Output, state) + ` = ${xyzInput.associatedVariableName}${this._buildSwizzle(2)};\\n`;\n }\n }\n else if (xyInput.isConnected) {\n if (v4Output.hasEndpoints) {\n if (zwInput.isConnected) {\n state.compilationString +=\n this._declareOutput(v4Output, state) + ` = vec4(${xyInput.associatedVariableName}, ${zwInput.associatedVariableName})${this._buildSwizzle(4)};\\n`;\n }\n else {\n state.compilationString +=\n this._declareOutput(v4Output, state) +\n ` = vec4(${xyInput.associatedVariableName}, ${zInput.isConnected ? this._writeVariable(zInput) : \"0.0\"}, ${wInput.isConnected ? this._writeVariable(wInput) : \"0.0\"})${this._buildSwizzle(4)};\\n`;\n }\n }\n if (v3Output.hasEndpoints) {\n state.compilationString +=\n this._declareOutput(v3Output, state) +\n ` = vec3(${xyInput.associatedVariableName}, ${zInput.isConnected ? this._writeVariable(zInput) : \"0.0\"})${this._buildSwizzle(3)};\\n`;\n }\n if (v2Output.hasEndpoints) {\n state.compilationString += this._declareOutput(v2Output, state) + ` = ${xyInput.associatedVariableName}${this._buildSwizzle(2)};\\n`;\n }\n if (v2CompOutput.hasEndpoints) {\n if (zwInput.isConnected) {\n state.compilationString += this._declareOutput(v2CompOutput, state) + ` = ${zwInput.associatedVariableName}${this._buildSwizzle(2)};\\n`;\n }\n else {\n state.compilationString +=\n this._declareOutput(v2CompOutput, state) +\n ` = vec2(${zInput.isConnected ? this._writeVariable(zInput) : \"0.0\"}, ${wInput.isConnected ? this._writeVariable(wInput) : \"0.0\"})${this._buildSwizzle(2)};\\n`;\n }\n }\n }\n else {\n if (v4Output.hasEndpoints) {\n if (zwInput.isConnected) {\n state.compilationString +=\n this._declareOutput(v4Output, state) +\n ` = vec4(${xInput.isConnected ? this._writeVariable(xInput) : \"0.0\"}, ${yInput.isConnected ? this._writeVariable(yInput) : \"0.0\"}, ${zwInput.associatedVariableName})${this._buildSwizzle(4)};\\n`;\n }\n else {\n state.compilationString +=\n this._declareOutput(v4Output, state) +\n ` = vec4(${xInput.isConnected ? this._writeVariable(xInput) : \"0.0\"}, ${yInput.isConnected ? this._writeVariable(yInput) : \"0.0\"}, ${zInput.isConnected ? this._writeVariable(zInput) : \"0.0\"}, ${wInput.isConnected ? this._writeVariable(wInput) : \"0.0\"})${this._buildSwizzle(4)};\\n`;\n }\n }\n if (v3Output.hasEndpoints) {\n state.compilationString +=\n this._declareOutput(v3Output, state) +\n ` = vec3(${xInput.isConnected ? this._writeVariable(xInput) : \"0.0\"}, ${yInput.isConnected ? this._writeVariable(yInput) : \"0.0\"}, ${zInput.isConnected ? this._writeVariable(zInput) : \"0.0\"})${this._buildSwizzle(3)};\\n`;\n }\n if (v2Output.hasEndpoints) {\n state.compilationString +=\n this._declareOutput(v2Output, state) +\n ` = vec2(${xInput.isConnected ? this._writeVariable(xInput) : \"0.0\"}, ${yInput.isConnected ? this._writeVariable(yInput) : \"0.0\"})${this._buildSwizzle(2)};\\n`;\n }\n if (v2CompOutput.hasEndpoints) {\n if (zwInput.isConnected) {\n state.compilationString += this._declareOutput(v2CompOutput, state) + ` = ${zwInput.associatedVariableName}${this._buildSwizzle(2)};\\n`;\n }\n else {\n state.compilationString +=\n this._declareOutput(v2CompOutput, state) +\n ` = vec2(${zInput.isConnected ? this._writeVariable(zInput) : \"0.0\"}, ${wInput.isConnected ? this._writeVariable(wInput) : \"0.0\"})${this._buildSwizzle(2)};\\n`;\n }\n }\n }\n return this;\n }\n serialize() {\n const serializationObject = super.serialize();\n serializationObject.xSwizzle = this.xSwizzle;\n serializationObject.ySwizzle = this.ySwizzle;\n serializationObject.zSwizzle = this.zSwizzle;\n serializationObject.wSwizzle = this.wSwizzle;\n return serializationObject;\n }\n _deserialize(serializationObject, scene, rootUrl) {\n var _a, _b, _c, _d;\n super._deserialize(serializationObject, scene, rootUrl);\n this.xSwizzle = (_a = serializationObject.xSwizzle) !== null && _a !== void 0 ? _a : \"x\";\n this.ySwizzle = (_b = serializationObject.ySwizzle) !== null && _b !== void 0 ? _b : \"y\";\n this.zSwizzle = (_c = serializationObject.zSwizzle) !== null && _c !== void 0 ? _c : \"z\";\n this.wSwizzle = (_d = serializationObject.wSwizzle) !== null && _d !== void 0 ? _d : \"w\";\n }\n _dumpPropertiesCode() {\n let codeString = super._dumpPropertiesCode();\n codeString += `${this._codeVariableName}.xSwizzle = \"${this.xSwizzle}\";\\n`;\n codeString += `${this._codeVariableName}.ySwizzle = \"${this.ySwizzle}\";\\n`;\n codeString += `${this._codeVariableName}.zSwizzle = \"${this.zSwizzle}\";\\n`;\n codeString += `${this._codeVariableName}.wSwizzle = \"${this.wSwizzle}\";\\n`;\n return codeString;\n }\n }\n RegisterClass(\"BABYLON.VectorMergerBlock\", VectorMergerBlock);\n\n /**\n * Block used to remap a float from a range to a new one\n */\n class RemapBlock extends NodeMaterialBlock {\n /**\n * Creates a new RemapBlock\n * @param name defines the block name\n */\n constructor(name) {\n super(name, NodeMaterialBlockTargets.Neutral);\n /**\n * Gets or sets the source range\n */\n this.sourceRange = new Vector2(-1, 1);\n /**\n * Gets or sets the target range\n */\n this.targetRange = new Vector2(0, 1);\n this.registerInput(\"input\", NodeMaterialBlockConnectionPointTypes.AutoDetect);\n this.registerInput(\"sourceMin\", NodeMaterialBlockConnectionPointTypes.Float, true);\n this.registerInput(\"sourceMax\", NodeMaterialBlockConnectionPointTypes.Float, true);\n this.registerInput(\"targetMin\", NodeMaterialBlockConnectionPointTypes.Float, true);\n this.registerInput(\"targetMax\", NodeMaterialBlockConnectionPointTypes.Float, true);\n this.registerOutput(\"output\", NodeMaterialBlockConnectionPointTypes.BasedOnInput);\n this._outputs[0]._typeConnectionSource = this._inputs[0];\n }\n /**\n * Gets the current class name\n * @returns the class name\n */\n getClassName() {\n return \"RemapBlock\";\n }\n /**\n * Gets the input component\n */\n get input() {\n return this._inputs[0];\n }\n /**\n * Gets the source min input component\n */\n get sourceMin() {\n return this._inputs[1];\n }\n /**\n * Gets the source max input component\n */\n get sourceMax() {\n return this._inputs[2];\n }\n /**\n * Gets the target min input component\n */\n get targetMin() {\n return this._inputs[3];\n }\n /**\n * Gets the target max input component\n */\n get targetMax() {\n return this._inputs[4];\n }\n /**\n * Gets the output component\n */\n get output() {\n return this._outputs[0];\n }\n _buildBlock(state) {\n super._buildBlock(state);\n const output = this._outputs[0];\n const sourceMin = this.sourceMin.isConnected ? this.sourceMin.associatedVariableName : this._writeFloat(this.sourceRange.x);\n const sourceMax = this.sourceMax.isConnected ? this.sourceMax.associatedVariableName : this._writeFloat(this.sourceRange.y);\n const targetMin = this.targetMin.isConnected ? this.targetMin.associatedVariableName : this._writeFloat(this.targetRange.x);\n const targetMax = this.targetMax.isConnected ? this.targetMax.associatedVariableName : this._writeFloat(this.targetRange.y);\n state.compilationString +=\n this._declareOutput(output, state) +\n ` = ${targetMin} + (${this._inputs[0].associatedVariableName} - ${sourceMin}) * (${targetMax} - ${targetMin}) / (${sourceMax} - ${sourceMin});\\n`;\n return this;\n }\n _dumpPropertiesCode() {\n let codeString = super._dumpPropertiesCode() + `${this._codeVariableName}.sourceRange = new BABYLON.Vector2(${this.sourceRange.x}, ${this.sourceRange.y});\\n`;\n codeString += `${this._codeVariableName}.targetRange = new BABYLON.Vector2(${this.targetRange.x}, ${this.targetRange.y});\\n`;\n return codeString;\n }\n serialize() {\n const serializationObject = super.serialize();\n serializationObject.sourceRange = this.sourceRange.asArray();\n serializationObject.targetRange = this.targetRange.asArray();\n return serializationObject;\n }\n _deserialize(serializationObject, scene, rootUrl) {\n super._deserialize(serializationObject, scene, rootUrl);\n this.sourceRange = Vector2.FromArray(serializationObject.sourceRange);\n this.targetRange = Vector2.FromArray(serializationObject.targetRange);\n }\n }\n __decorate$1([\n editableInPropertyPage(\"From\", PropertyTypeForEdition.Vector2)\n ], RemapBlock.prototype, \"sourceRange\", void 0);\n __decorate$1([\n editableInPropertyPage(\"To\", PropertyTypeForEdition.Vector2)\n ], RemapBlock.prototype, \"targetRange\", void 0);\n RegisterClass(\"BABYLON.RemapBlock\", RemapBlock);\n\n /**\n * Block used to multiply 2 values\n */\n class MultiplyBlock extends NodeMaterialBlock {\n /**\n * Creates a new MultiplyBlock\n * @param name defines the block name\n */\n constructor(name) {\n super(name, NodeMaterialBlockTargets.Neutral);\n this.registerInput(\"left\", NodeMaterialBlockConnectionPointTypes.AutoDetect);\n this.registerInput(\"right\", NodeMaterialBlockConnectionPointTypes.AutoDetect);\n this.registerOutput(\"output\", NodeMaterialBlockConnectionPointTypes.BasedOnInput);\n this._outputs[0]._typeConnectionSource = this._inputs[0];\n this._linkConnectionTypes(0, 1);\n this._inputs[0].acceptedConnectionPointTypes.push(NodeMaterialBlockConnectionPointTypes.Float);\n this._inputs[1].acceptedConnectionPointTypes.push(NodeMaterialBlockConnectionPointTypes.Float);\n }\n /**\n * Gets the current class name\n * @returns the class name\n */\n getClassName() {\n return \"MultiplyBlock\";\n }\n /**\n * Gets the left operand input component\n */\n get left() {\n return this._inputs[0];\n }\n /**\n * Gets the right operand input component\n */\n get right() {\n return this._inputs[1];\n }\n /**\n * Gets the output component\n */\n get output() {\n return this._outputs[0];\n }\n _buildBlock(state) {\n super._buildBlock(state);\n const output = this._outputs[0];\n state.compilationString += this._declareOutput(output, state) + ` = ${this.left.associatedVariableName} * ${this.right.associatedVariableName};\\n`;\n return this;\n }\n }\n RegisterClass(\"BABYLON.MultiplyBlock\", MultiplyBlock);\n\n /**\n * Enum used to define the material modes\n */\n var NodeMaterialModes;\n (function (NodeMaterialModes) {\n /** Regular material */\n NodeMaterialModes[NodeMaterialModes[\"Material\"] = 0] = \"Material\";\n /** For post process */\n NodeMaterialModes[NodeMaterialModes[\"PostProcess\"] = 1] = \"PostProcess\";\n /** For particle system */\n NodeMaterialModes[NodeMaterialModes[\"Particle\"] = 2] = \"Particle\";\n /** For procedural texture */\n NodeMaterialModes[NodeMaterialModes[\"ProceduralTexture\"] = 3] = \"ProceduralTexture\";\n })(NodeMaterialModes || (NodeMaterialModes = {}));\n\n /**\n * Particle emitter emitting particles from the inside of a box.\n * It emits the particles randomly between 2 given directions.\n */\n class BoxParticleEmitter {\n /**\n * Creates a new instance BoxParticleEmitter\n */\n constructor() {\n /**\n * Random direction of each particle after it has been emitted, between direction1 and direction2 vectors.\n */\n this.direction1 = new Vector3(0, 1.0, 0);\n /**\n * Random direction of each particle after it has been emitted, between direction1 and direction2 vectors.\n */\n this.direction2 = new Vector3(0, 1.0, 0);\n /**\n * Minimum box point around our emitter. Our emitter is the center of particles source, but if you want your particles to emit from more than one point, then you can tell it to do so.\n */\n this.minEmitBox = new Vector3(-0.5, -0.5, -0.5);\n /**\n * Maximum box point around our emitter. Our emitter is the center of particles source, but if you want your particles to emit from more than one point, then you can tell it to do so.\n */\n this.maxEmitBox = new Vector3(0.5, 0.5, 0.5);\n }\n /**\n * Called by the particle System when the direction is computed for the created particle.\n * @param worldMatrix is the world matrix of the particle system\n * @param directionToUpdate is the direction vector to update with the result\n * @param particle is the particle we are computed the direction for\n * @param isLocal defines if the direction should be set in local space\n */\n startDirectionFunction(worldMatrix, directionToUpdate, particle, isLocal) {\n const randX = Scalar.RandomRange(this.direction1.x, this.direction2.x);\n const randY = Scalar.RandomRange(this.direction1.y, this.direction2.y);\n const randZ = Scalar.RandomRange(this.direction1.z, this.direction2.z);\n if (isLocal) {\n directionToUpdate.x = randX;\n directionToUpdate.y = randY;\n directionToUpdate.z = randZ;\n return;\n }\n Vector3.TransformNormalFromFloatsToRef(randX, randY, randZ, worldMatrix, directionToUpdate);\n }\n /**\n * Called by the particle System when the position is computed for the created particle.\n * @param worldMatrix is the world matrix of the particle system\n * @param positionToUpdate is the position vector to update with the result\n * @param particle is the particle we are computed the position for\n * @param isLocal defines if the position should be set in local space\n */\n startPositionFunction(worldMatrix, positionToUpdate, particle, isLocal) {\n const randX = Scalar.RandomRange(this.minEmitBox.x, this.maxEmitBox.x);\n const randY = Scalar.RandomRange(this.minEmitBox.y, this.maxEmitBox.y);\n const randZ = Scalar.RandomRange(this.minEmitBox.z, this.maxEmitBox.z);\n if (isLocal) {\n positionToUpdate.x = randX;\n positionToUpdate.y = randY;\n positionToUpdate.z = randZ;\n return;\n }\n Vector3.TransformCoordinatesFromFloatsToRef(randX, randY, randZ, worldMatrix, positionToUpdate);\n }\n /**\n * Clones the current emitter and returns a copy of it\n * @returns the new emitter\n */\n clone() {\n const newOne = new BoxParticleEmitter();\n DeepCopier.DeepCopy(this, newOne);\n return newOne;\n }\n /**\n * Called by the GPUParticleSystem to setup the update shader\n * @param uboOrEffect defines the update shader\n */\n applyToShader(uboOrEffect) {\n uboOrEffect.setVector3(\"direction1\", this.direction1);\n uboOrEffect.setVector3(\"direction2\", this.direction2);\n uboOrEffect.setVector3(\"minEmitBox\", this.minEmitBox);\n uboOrEffect.setVector3(\"maxEmitBox\", this.maxEmitBox);\n }\n /**\n * Creates the structure of the ubo for this particle emitter\n * @param ubo ubo to create the structure for\n */\n buildUniformLayout(ubo) {\n ubo.addUniform(\"direction1\", 3);\n ubo.addUniform(\"direction2\", 3);\n ubo.addUniform(\"minEmitBox\", 3);\n ubo.addUniform(\"maxEmitBox\", 3);\n }\n /**\n * Returns a string to use to update the GPU particles update shader\n * @returns a string containing the defines string\n */\n getEffectDefines() {\n return \"#define BOXEMITTER\";\n }\n /**\n * Returns the string \"BoxParticleEmitter\"\n * @returns a string containing the class name\n */\n getClassName() {\n return \"BoxParticleEmitter\";\n }\n /**\n * Serializes the particle system to a JSON object.\n * @returns the JSON object\n */\n serialize() {\n const serializationObject = {};\n serializationObject.type = this.getClassName();\n serializationObject.direction1 = this.direction1.asArray();\n serializationObject.direction2 = this.direction2.asArray();\n serializationObject.minEmitBox = this.minEmitBox.asArray();\n serializationObject.maxEmitBox = this.maxEmitBox.asArray();\n return serializationObject;\n }\n /**\n * Parse properties from a JSON object\n * @param serializationObject defines the JSON object\n */\n parse(serializationObject) {\n Vector3.FromArrayToRef(serializationObject.direction1, 0, this.direction1);\n Vector3.FromArrayToRef(serializationObject.direction2, 0, this.direction2);\n Vector3.FromArrayToRef(serializationObject.minEmitBox, 0, this.minEmitBox);\n Vector3.FromArrayToRef(serializationObject.maxEmitBox, 0, this.maxEmitBox);\n }\n }\n\n /**\n * Particle emitter emitting particles from the inside of a cone.\n * It emits the particles alongside the cone volume from the base to the particle.\n * The emission direction might be randomized.\n */\n class ConeParticleEmitter {\n /**\n * Gets or sets the radius of the emission cone\n */\n get radius() {\n return this._radius;\n }\n set radius(value) {\n this._radius = value;\n this._buildHeight();\n }\n /**\n * Gets or sets the angle of the emission cone\n */\n get angle() {\n return this._angle;\n }\n set angle(value) {\n this._angle = value;\n this._buildHeight();\n }\n _buildHeight() {\n if (this._angle !== 0) {\n this._height = this._radius / Math.tan(this._angle / 2);\n }\n else {\n this._height = 1;\n }\n }\n /**\n * Creates a new instance ConeParticleEmitter\n * @param radius the radius of the emission cone (1 by default)\n * @param angle the cone base angle (PI by default)\n * @param directionRandomizer defines how much to randomize the particle direction [0-1] (default is 0)\n */\n constructor(radius = 1, angle = Math.PI, \n /** defines how much to randomize the particle direction [0-1] (default is 0) */\n directionRandomizer = 0) {\n this.directionRandomizer = directionRandomizer;\n /**\n * Gets or sets a value indicating where on the radius the start position should be picked (1 = everywhere, 0 = only surface)\n */\n this.radiusRange = 1;\n /**\n * Gets or sets a value indicating where on the height the start position should be picked (1 = everywhere, 0 = only surface)\n */\n this.heightRange = 1;\n /**\n * Gets or sets a value indicating if all the particles should be emitted from the spawn point only (the base of the cone)\n */\n this.emitFromSpawnPointOnly = false;\n this.angle = angle;\n this.radius = radius;\n }\n /**\n * Called by the particle System when the direction is computed for the created particle.\n * @param worldMatrix is the world matrix of the particle system\n * @param directionToUpdate is the direction vector to update with the result\n * @param particle is the particle we are computed the direction for\n * @param isLocal defines if the direction should be set in local space\n */\n startDirectionFunction(worldMatrix, directionToUpdate, particle, isLocal) {\n if (isLocal) {\n TmpVectors.Vector3[0].copyFrom(particle._localPosition).normalize();\n }\n else {\n particle.position.subtractToRef(worldMatrix.getTranslation(), TmpVectors.Vector3[0]).normalize();\n }\n const randX = Scalar.RandomRange(0, this.directionRandomizer);\n const randY = Scalar.RandomRange(0, this.directionRandomizer);\n const randZ = Scalar.RandomRange(0, this.directionRandomizer);\n directionToUpdate.x = TmpVectors.Vector3[0].x + randX;\n directionToUpdate.y = TmpVectors.Vector3[0].y + randY;\n directionToUpdate.z = TmpVectors.Vector3[0].z + randZ;\n directionToUpdate.normalize();\n }\n /**\n * Called by the particle System when the position is computed for the created particle.\n * @param worldMatrix is the world matrix of the particle system\n * @param positionToUpdate is the position vector to update with the result\n * @param particle is the particle we are computed the position for\n * @param isLocal defines if the position should be set in local space\n */\n startPositionFunction(worldMatrix, positionToUpdate, particle, isLocal) {\n const s = Scalar.RandomRange(0, Math.PI * 2);\n let h;\n if (!this.emitFromSpawnPointOnly) {\n h = Scalar.RandomRange(0, this.heightRange);\n // Better distribution in a cone at normal angles.\n h = 1 - h * h;\n }\n else {\n h = 0.0001;\n }\n let radius = this._radius - Scalar.RandomRange(0, this._radius * this.radiusRange);\n radius = radius * h;\n const randX = radius * Math.sin(s);\n const randZ = radius * Math.cos(s);\n const randY = h * this._height;\n if (isLocal) {\n positionToUpdate.x = randX;\n positionToUpdate.y = randY;\n positionToUpdate.z = randZ;\n return;\n }\n Vector3.TransformCoordinatesFromFloatsToRef(randX, randY, randZ, worldMatrix, positionToUpdate);\n }\n /**\n * Clones the current emitter and returns a copy of it\n * @returns the new emitter\n */\n clone() {\n const newOne = new ConeParticleEmitter(this._radius, this._angle, this.directionRandomizer);\n DeepCopier.DeepCopy(this, newOne);\n return newOne;\n }\n /**\n * Called by the GPUParticleSystem to setup the update shader\n * @param uboOrEffect defines the update shader\n */\n applyToShader(uboOrEffect) {\n uboOrEffect.setFloat2(\"radius\", this._radius, this.radiusRange);\n uboOrEffect.setFloat(\"coneAngle\", this._angle);\n uboOrEffect.setFloat2(\"height\", this._height, this.heightRange);\n uboOrEffect.setFloat(\"directionRandomizer\", this.directionRandomizer);\n }\n /**\n * Creates the structure of the ubo for this particle emitter\n * @param ubo ubo to create the structure for\n */\n buildUniformLayout(ubo) {\n ubo.addUniform(\"radius\", 2);\n ubo.addUniform(\"coneAngle\", 1);\n ubo.addUniform(\"height\", 2);\n ubo.addUniform(\"directionRandomizer\", 1);\n }\n /**\n * Returns a string to use to update the GPU particles update shader\n * @returns a string containing the defines string\n */\n getEffectDefines() {\n let defines = \"#define CONEEMITTER\";\n if (this.emitFromSpawnPointOnly) {\n defines += \"\\n#define CONEEMITTERSPAWNPOINT\";\n }\n return defines;\n }\n /**\n * Returns the string \"ConeParticleEmitter\"\n * @returns a string containing the class name\n */\n getClassName() {\n return \"ConeParticleEmitter\";\n }\n /**\n * Serializes the particle system to a JSON object.\n * @returns the JSON object\n */\n serialize() {\n const serializationObject = {};\n serializationObject.type = this.getClassName();\n serializationObject.radius = this._radius;\n serializationObject.angle = this._angle;\n serializationObject.directionRandomizer = this.directionRandomizer;\n serializationObject.radiusRange = this.radiusRange;\n serializationObject.heightRange = this.heightRange;\n serializationObject.emitFromSpawnPointOnly = this.emitFromSpawnPointOnly;\n return serializationObject;\n }\n /**\n * Parse properties from a JSON object\n * @param serializationObject defines the JSON object\n */\n parse(serializationObject) {\n this.radius = serializationObject.radius;\n this.angle = serializationObject.angle;\n this.directionRandomizer = serializationObject.directionRandomizer;\n this.radiusRange = serializationObject.radiusRange !== undefined ? serializationObject.radiusRange : 1;\n this.heightRange = serializationObject.radiusRange !== undefined ? serializationObject.heightRange : 1;\n this.emitFromSpawnPointOnly = serializationObject.emitFromSpawnPointOnly !== undefined ? serializationObject.emitFromSpawnPointOnly : false;\n }\n }\n\n /**\n * Particle emitter emitting particles from the inside of a cylinder.\n * It emits the particles alongside the cylinder radius. The emission direction might be randomized.\n */\n class CylinderParticleEmitter {\n /**\n * Creates a new instance CylinderParticleEmitter\n * @param radius the radius of the emission cylinder (1 by default)\n * @param height the height of the emission cylinder (1 by default)\n * @param radiusRange the range of the emission cylinder [0-1] 0 Surface only, 1 Entire Radius (1 by default)\n * @param directionRandomizer defines how much to randomize the particle direction [0-1]\n */\n constructor(\n /**\n * The radius of the emission cylinder.\n */\n radius = 1, \n /**\n * The height of the emission cylinder.\n */\n height = 1, \n /**\n * The range of emission [0-1] 0 Surface only, 1 Entire Radius.\n */\n radiusRange = 1, \n /**\n * How much to randomize the particle direction [0-1].\n */\n directionRandomizer = 0) {\n this.radius = radius;\n this.height = height;\n this.radiusRange = radiusRange;\n this.directionRandomizer = directionRandomizer;\n this._tempVector = Vector3.Zero();\n }\n /**\n * Called by the particle System when the direction is computed for the created particle.\n * @param worldMatrix is the world matrix of the particle system\n * @param directionToUpdate is the direction vector to update with the result\n * @param particle is the particle we are computed the direction for\n * @param isLocal defines if the direction should be set in local space\n * @param inverseWorldMatrix defines the inverted world matrix to use if isLocal is false\n */\n startDirectionFunction(worldMatrix, directionToUpdate, particle, isLocal, inverseWorldMatrix) {\n particle.position.subtractToRef(worldMatrix.getTranslation(), this._tempVector);\n this._tempVector.normalize();\n Vector3.TransformNormalToRef(this._tempVector, inverseWorldMatrix, this._tempVector);\n const randY = Scalar.RandomRange(-this.directionRandomizer / 2, this.directionRandomizer / 2);\n let angle = Math.atan2(this._tempVector.x, this._tempVector.z);\n angle += Scalar.RandomRange(-Math.PI / 2, Math.PI / 2) * this.directionRandomizer;\n this._tempVector.y = randY; // set direction y to rand y to mirror normal of cylinder surface\n this._tempVector.x = Math.sin(angle);\n this._tempVector.z = Math.cos(angle);\n this._tempVector.normalize();\n if (isLocal) {\n directionToUpdate.copyFrom(this._tempVector);\n return;\n }\n Vector3.TransformNormalFromFloatsToRef(this._tempVector.x, this._tempVector.y, this._tempVector.z, worldMatrix, directionToUpdate);\n }\n /**\n * Called by the particle System when the position is computed for the created particle.\n * @param worldMatrix is the world matrix of the particle system\n * @param positionToUpdate is the position vector to update with the result\n * @param particle is the particle we are computed the position for\n * @param isLocal defines if the position should be set in local space\n */\n startPositionFunction(worldMatrix, positionToUpdate, particle, isLocal) {\n const yPos = Scalar.RandomRange(-this.height / 2, this.height / 2);\n const angle = Scalar.RandomRange(0, 2 * Math.PI);\n // Pick a properly distributed point within the circle https://programming.guide/random-point-within-circle.html\n const radiusDistribution = Scalar.RandomRange((1 - this.radiusRange) * (1 - this.radiusRange), 1);\n const positionRadius = Math.sqrt(radiusDistribution) * this.radius;\n const xPos = positionRadius * Math.cos(angle);\n const zPos = positionRadius * Math.sin(angle);\n if (isLocal) {\n positionToUpdate.copyFromFloats(xPos, yPos, zPos);\n return;\n }\n Vector3.TransformCoordinatesFromFloatsToRef(xPos, yPos, zPos, worldMatrix, positionToUpdate);\n }\n /**\n * Clones the current emitter and returns a copy of it\n * @returns the new emitter\n */\n clone() {\n const newOne = new CylinderParticleEmitter(this.radius, this.directionRandomizer);\n DeepCopier.DeepCopy(this, newOne);\n return newOne;\n }\n /**\n * Called by the GPUParticleSystem to setup the update shader\n * @param uboOrEffect defines the update shader\n */\n applyToShader(uboOrEffect) {\n uboOrEffect.setFloat(\"radius\", this.radius);\n uboOrEffect.setFloat(\"height\", this.height);\n uboOrEffect.setFloat(\"radiusRange\", this.radiusRange);\n uboOrEffect.setFloat(\"directionRandomizer\", this.directionRandomizer);\n }\n /**\n * Creates the structure of the ubo for this particle emitter\n * @param ubo ubo to create the structure for\n */\n buildUniformLayout(ubo) {\n ubo.addUniform(\"radius\", 1);\n ubo.addUniform(\"height\", 1);\n ubo.addUniform(\"radiusRange\", 1);\n ubo.addUniform(\"directionRandomizer\", 1);\n }\n /**\n * Returns a string to use to update the GPU particles update shader\n * @returns a string containing the defines string\n */\n getEffectDefines() {\n return \"#define CYLINDEREMITTER\";\n }\n /**\n * Returns the string \"CylinderParticleEmitter\"\n * @returns a string containing the class name\n */\n getClassName() {\n return \"CylinderParticleEmitter\";\n }\n /**\n * Serializes the particle system to a JSON object.\n * @returns the JSON object\n */\n serialize() {\n const serializationObject = {};\n serializationObject.type = this.getClassName();\n serializationObject.radius = this.radius;\n serializationObject.height = this.height;\n serializationObject.radiusRange = this.radiusRange;\n serializationObject.directionRandomizer = this.directionRandomizer;\n return serializationObject;\n }\n /**\n * Parse properties from a JSON object\n * @param serializationObject defines the JSON object\n */\n parse(serializationObject) {\n this.radius = serializationObject.radius;\n this.height = serializationObject.height;\n this.radiusRange = serializationObject.radiusRange;\n this.directionRandomizer = serializationObject.directionRandomizer;\n }\n }\n /**\n * Particle emitter emitting particles from the inside of a cylinder.\n * It emits the particles randomly between two vectors.\n */\n class CylinderDirectedParticleEmitter extends CylinderParticleEmitter {\n /**\n * Creates a new instance CylinderDirectedParticleEmitter\n * @param radius the radius of the emission cylinder (1 by default)\n * @param height the height of the emission cylinder (1 by default)\n * @param radiusRange the range of the emission cylinder [0-1] 0 Surface only, 1 Entire Radius (1 by default)\n * @param direction1 the min limit of the emission direction (up vector by default)\n * @param direction2 the max limit of the emission direction (up vector by default)\n */\n constructor(radius = 1, height = 1, radiusRange = 1, \n /**\n * The min limit of the emission direction.\n */\n direction1 = new Vector3(0, 1, 0), \n /**\n * The max limit of the emission direction.\n */\n direction2 = new Vector3(0, 1, 0)) {\n super(radius, height, radiusRange);\n this.direction1 = direction1;\n this.direction2 = direction2;\n }\n /**\n * Called by the particle System when the direction is computed for the created particle.\n * @param worldMatrix is the world matrix of the particle system\n * @param directionToUpdate is the direction vector to update with the result\n */\n startDirectionFunction(worldMatrix, directionToUpdate) {\n const randX = Scalar.RandomRange(this.direction1.x, this.direction2.x);\n const randY = Scalar.RandomRange(this.direction1.y, this.direction2.y);\n const randZ = Scalar.RandomRange(this.direction1.z, this.direction2.z);\n Vector3.TransformNormalFromFloatsToRef(randX, randY, randZ, worldMatrix, directionToUpdate);\n }\n /**\n * Clones the current emitter and returns a copy of it\n * @returns the new emitter\n */\n clone() {\n const newOne = new CylinderDirectedParticleEmitter(this.radius, this.height, this.radiusRange, this.direction1, this.direction2);\n DeepCopier.DeepCopy(this, newOne);\n return newOne;\n }\n /**\n * Called by the GPUParticleSystem to setup the update shader\n * @param uboOrEffect defines the update shader\n */\n applyToShader(uboOrEffect) {\n uboOrEffect.setFloat(\"radius\", this.radius);\n uboOrEffect.setFloat(\"height\", this.height);\n uboOrEffect.setFloat(\"radiusRange\", this.radiusRange);\n uboOrEffect.setVector3(\"direction1\", this.direction1);\n uboOrEffect.setVector3(\"direction2\", this.direction2);\n }\n /**\n * Creates the structure of the ubo for this particle emitter\n * @param ubo ubo to create the structure for\n */\n buildUniformLayout(ubo) {\n ubo.addUniform(\"radius\", 1);\n ubo.addUniform(\"height\", 1);\n ubo.addUniform(\"radiusRange\", 1);\n ubo.addUniform(\"direction1\", 3);\n ubo.addUniform(\"direction2\", 3);\n }\n /**\n * Returns a string to use to update the GPU particles update shader\n * @returns a string containing the defines string\n */\n getEffectDefines() {\n return \"#define CYLINDEREMITTER\\n#define DIRECTEDCYLINDEREMITTER\";\n }\n /**\n * Returns the string \"CylinderDirectedParticleEmitter\"\n * @returns a string containing the class name\n */\n getClassName() {\n return \"CylinderDirectedParticleEmitter\";\n }\n /**\n * Serializes the particle system to a JSON object.\n * @returns the JSON object\n */\n serialize() {\n const serializationObject = super.serialize();\n serializationObject.direction1 = this.direction1.asArray();\n serializationObject.direction2 = this.direction2.asArray();\n return serializationObject;\n }\n /**\n * Parse properties from a JSON object\n * @param serializationObject defines the JSON object\n */\n parse(serializationObject) {\n super.parse(serializationObject);\n this.direction1.copyFrom(serializationObject.direction1);\n this.direction2.copyFrom(serializationObject.direction2);\n }\n }\n\n /**\n * Particle emitter emitting particles from the inside of a hemisphere.\n * It emits the particles alongside the hemisphere radius. The emission direction might be randomized.\n */\n class HemisphericParticleEmitter {\n /**\n * Creates a new instance HemisphericParticleEmitter\n * @param radius the radius of the emission hemisphere (1 by default)\n * @param radiusRange the range of the emission hemisphere [0-1] 0 Surface only, 1 Entire Radius (1 by default)\n * @param directionRandomizer defines how much to randomize the particle direction [0-1]\n */\n constructor(\n /**\n * The radius of the emission hemisphere.\n */\n radius = 1, \n /**\n * The range of emission [0-1] 0 Surface only, 1 Entire Radius.\n */\n radiusRange = 1, \n /**\n * How much to randomize the particle direction [0-1].\n */\n directionRandomizer = 0) {\n this.radius = radius;\n this.radiusRange = radiusRange;\n this.directionRandomizer = directionRandomizer;\n }\n /**\n * Called by the particle System when the direction is computed for the created particle.\n * @param worldMatrix is the world matrix of the particle system\n * @param directionToUpdate is the direction vector to update with the result\n * @param particle is the particle we are computed the direction for\n * @param isLocal defines if the direction should be set in local space\n */\n startDirectionFunction(worldMatrix, directionToUpdate, particle, isLocal) {\n const direction = particle.position.subtract(worldMatrix.getTranslation()).normalize();\n const randX = Scalar.RandomRange(0, this.directionRandomizer);\n const randY = Scalar.RandomRange(0, this.directionRandomizer);\n const randZ = Scalar.RandomRange(0, this.directionRandomizer);\n direction.x += randX;\n direction.y += randY;\n direction.z += randZ;\n direction.normalize();\n if (isLocal) {\n directionToUpdate.copyFrom(direction);\n return;\n }\n Vector3.TransformNormalFromFloatsToRef(direction.x, direction.y, direction.z, worldMatrix, directionToUpdate);\n }\n /**\n * Called by the particle System when the position is computed for the created particle.\n * @param worldMatrix is the world matrix of the particle system\n * @param positionToUpdate is the position vector to update with the result\n * @param particle is the particle we are computed the position for\n * @param isLocal defines if the position should be set in local space\n */\n startPositionFunction(worldMatrix, positionToUpdate, particle, isLocal) {\n const randRadius = this.radius - Scalar.RandomRange(0, this.radius * this.radiusRange);\n const v = Scalar.RandomRange(0, 1.0);\n const phi = Scalar.RandomRange(0, 2 * Math.PI);\n const theta = Math.acos(2 * v - 1);\n const randX = randRadius * Math.cos(phi) * Math.sin(theta);\n const randY = randRadius * Math.cos(theta);\n const randZ = randRadius * Math.sin(phi) * Math.sin(theta);\n if (isLocal) {\n positionToUpdate.copyFromFloats(randX, Math.abs(randY), randZ);\n return;\n }\n Vector3.TransformCoordinatesFromFloatsToRef(randX, Math.abs(randY), randZ, worldMatrix, positionToUpdate);\n }\n /**\n * Clones the current emitter and returns a copy of it\n * @returns the new emitter\n */\n clone() {\n const newOne = new HemisphericParticleEmitter(this.radius, this.directionRandomizer);\n DeepCopier.DeepCopy(this, newOne);\n return newOne;\n }\n /**\n * Called by the GPUParticleSystem to setup the update shader\n * @param uboOrEffect defines the update shader\n */\n applyToShader(uboOrEffect) {\n uboOrEffect.setFloat(\"radius\", this.radius);\n uboOrEffect.setFloat(\"radiusRange\", this.radiusRange);\n uboOrEffect.setFloat(\"directionRandomizer\", this.directionRandomizer);\n }\n /**\n * Creates the structure of the ubo for this particle emitter\n * @param ubo ubo to create the structure for\n */\n buildUniformLayout(ubo) {\n ubo.addUniform(\"radius\", 1);\n ubo.addUniform(\"radiusRange\", 1);\n ubo.addUniform(\"directionRandomizer\", 1);\n }\n /**\n * Returns a string to use to update the GPU particles update shader\n * @returns a string containing the defines string\n */\n getEffectDefines() {\n return \"#define HEMISPHERICEMITTER\";\n }\n /**\n * Returns the string \"HemisphericParticleEmitter\"\n * @returns a string containing the class name\n */\n getClassName() {\n return \"HemisphericParticleEmitter\";\n }\n /**\n * Serializes the particle system to a JSON object.\n * @returns the JSON object\n */\n serialize() {\n const serializationObject = {};\n serializationObject.type = this.getClassName();\n serializationObject.radius = this.radius;\n serializationObject.radiusRange = this.radiusRange;\n serializationObject.directionRandomizer = this.directionRandomizer;\n return serializationObject;\n }\n /**\n * Parse properties from a JSON object\n * @param serializationObject defines the JSON object\n */\n parse(serializationObject) {\n this.radius = serializationObject.radius;\n this.radiusRange = serializationObject.radiusRange;\n this.directionRandomizer = serializationObject.directionRandomizer;\n }\n }\n\n /**\n * Particle emitter emitting particles from a point.\n * It emits the particles randomly between 2 given directions.\n */\n class PointParticleEmitter {\n /**\n * Creates a new instance PointParticleEmitter\n */\n constructor() {\n /**\n * Random direction of each particle after it has been emitted, between direction1 and direction2 vectors.\n */\n this.direction1 = new Vector3(0, 1.0, 0);\n /**\n * Random direction of each particle after it has been emitted, between direction1 and direction2 vectors.\n */\n this.direction2 = new Vector3(0, 1.0, 0);\n }\n /**\n * Called by the particle System when the direction is computed for the created particle.\n * @param worldMatrix is the world matrix of the particle system\n * @param directionToUpdate is the direction vector to update with the result\n * @param particle is the particle we are computed the direction for\n * @param isLocal defines if the direction should be set in local space\n */\n startDirectionFunction(worldMatrix, directionToUpdate, particle, isLocal) {\n const randX = Scalar.RandomRange(this.direction1.x, this.direction2.x);\n const randY = Scalar.RandomRange(this.direction1.y, this.direction2.y);\n const randZ = Scalar.RandomRange(this.direction1.z, this.direction2.z);\n if (isLocal) {\n directionToUpdate.copyFromFloats(randX, randY, randZ);\n return;\n }\n Vector3.TransformNormalFromFloatsToRef(randX, randY, randZ, worldMatrix, directionToUpdate);\n }\n /**\n * Called by the particle System when the position is computed for the created particle.\n * @param worldMatrix is the world matrix of the particle system\n * @param positionToUpdate is the position vector to update with the result\n * @param particle is the particle we are computed the position for\n * @param isLocal defines if the position should be set in local space\n */\n startPositionFunction(worldMatrix, positionToUpdate, particle, isLocal) {\n if (isLocal) {\n positionToUpdate.copyFromFloats(0, 0, 0);\n return;\n }\n Vector3.TransformCoordinatesFromFloatsToRef(0, 0, 0, worldMatrix, positionToUpdate);\n }\n /**\n * Clones the current emitter and returns a copy of it\n * @returns the new emitter\n */\n clone() {\n const newOne = new PointParticleEmitter();\n DeepCopier.DeepCopy(this, newOne);\n return newOne;\n }\n /**\n * Called by the GPUParticleSystem to setup the update shader\n * @param uboOrEffect defines the update shader\n */\n applyToShader(uboOrEffect) {\n uboOrEffect.setVector3(\"direction1\", this.direction1);\n uboOrEffect.setVector3(\"direction2\", this.direction2);\n }\n /**\n * Creates the structure of the ubo for this particle emitter\n * @param ubo ubo to create the structure for\n */\n buildUniformLayout(ubo) {\n ubo.addUniform(\"direction1\", 3);\n ubo.addUniform(\"direction2\", 3);\n }\n /**\n * Returns a string to use to update the GPU particles update shader\n * @returns a string containing the defines string\n */\n getEffectDefines() {\n return \"#define POINTEMITTER\";\n }\n /**\n * Returns the string \"PointParticleEmitter\"\n * @returns a string containing the class name\n */\n getClassName() {\n return \"PointParticleEmitter\";\n }\n /**\n * Serializes the particle system to a JSON object.\n * @returns the JSON object\n */\n serialize() {\n const serializationObject = {};\n serializationObject.type = this.getClassName();\n serializationObject.direction1 = this.direction1.asArray();\n serializationObject.direction2 = this.direction2.asArray();\n return serializationObject;\n }\n /**\n * Parse properties from a JSON object\n * @param serializationObject defines the JSON object\n */\n parse(serializationObject) {\n Vector3.FromArrayToRef(serializationObject.direction1, 0, this.direction1);\n Vector3.FromArrayToRef(serializationObject.direction2, 0, this.direction2);\n }\n }\n\n /**\n * Particle emitter emitting particles from the inside of a sphere.\n * It emits the particles alongside the sphere radius. The emission direction might be randomized.\n */\n class SphereParticleEmitter {\n /**\n * Creates a new instance SphereParticleEmitter\n * @param radius the radius of the emission sphere (1 by default)\n * @param radiusRange the range of the emission sphere [0-1] 0 Surface only, 1 Entire Radius (1 by default)\n * @param directionRandomizer defines how much to randomize the particle direction [0-1]\n */\n constructor(\n /**\n * The radius of the emission sphere.\n */\n radius = 1, \n /**\n * The range of emission [0-1] 0 Surface only, 1 Entire Radius.\n */\n radiusRange = 1, \n /**\n * How much to randomize the particle direction [0-1].\n */\n directionRandomizer = 0) {\n this.radius = radius;\n this.radiusRange = radiusRange;\n this.directionRandomizer = directionRandomizer;\n }\n /**\n * Called by the particle System when the direction is computed for the created particle.\n * @param worldMatrix is the world matrix of the particle system\n * @param directionToUpdate is the direction vector to update with the result\n * @param particle is the particle we are computed the direction for\n * @param isLocal defines if the direction should be set in local space\n */\n startDirectionFunction(worldMatrix, directionToUpdate, particle, isLocal) {\n const direction = particle.position.subtract(worldMatrix.getTranslation()).normalize();\n const randX = Scalar.RandomRange(0, this.directionRandomizer);\n const randY = Scalar.RandomRange(0, this.directionRandomizer);\n const randZ = Scalar.RandomRange(0, this.directionRandomizer);\n direction.x += randX;\n direction.y += randY;\n direction.z += randZ;\n direction.normalize();\n if (isLocal) {\n directionToUpdate.copyFrom(direction);\n return;\n }\n Vector3.TransformNormalFromFloatsToRef(direction.x, direction.y, direction.z, worldMatrix, directionToUpdate);\n }\n /**\n * Called by the particle System when the position is computed for the created particle.\n * @param worldMatrix is the world matrix of the particle system\n * @param positionToUpdate is the position vector to update with the result\n * @param particle is the particle we are computed the position for\n * @param isLocal defines if the position should be set in local space\n */\n startPositionFunction(worldMatrix, positionToUpdate, particle, isLocal) {\n const randRadius = this.radius - Scalar.RandomRange(0, this.radius * this.radiusRange);\n const v = Scalar.RandomRange(0, 1.0);\n const phi = Scalar.RandomRange(0, 2 * Math.PI);\n const theta = Math.acos(2 * v - 1);\n const randX = randRadius * Math.cos(phi) * Math.sin(theta);\n const randY = randRadius * Math.cos(theta);\n const randZ = randRadius * Math.sin(phi) * Math.sin(theta);\n if (isLocal) {\n positionToUpdate.copyFromFloats(randX, randY, randZ);\n return;\n }\n Vector3.TransformCoordinatesFromFloatsToRef(randX, randY, randZ, worldMatrix, positionToUpdate);\n }\n /**\n * Clones the current emitter and returns a copy of it\n * @returns the new emitter\n */\n clone() {\n const newOne = new SphereParticleEmitter(this.radius, this.directionRandomizer);\n DeepCopier.DeepCopy(this, newOne);\n return newOne;\n }\n /**\n * Called by the GPUParticleSystem to setup the update shader\n * @param uboOrEffect defines the update shader\n */\n applyToShader(uboOrEffect) {\n uboOrEffect.setFloat(\"radius\", this.radius);\n uboOrEffect.setFloat(\"radiusRange\", this.radiusRange);\n uboOrEffect.setFloat(\"directionRandomizer\", this.directionRandomizer);\n }\n /**\n * Creates the structure of the ubo for this particle emitter\n * @param ubo ubo to create the structure for\n */\n buildUniformLayout(ubo) {\n ubo.addUniform(\"radius\", 1);\n ubo.addUniform(\"radiusRange\", 1);\n ubo.addUniform(\"directionRandomizer\", 1);\n }\n /**\n * Returns a string to use to update the GPU particles update shader\n * @returns a string containing the defines string\n */\n getEffectDefines() {\n return \"#define SPHEREEMITTER\";\n }\n /**\n * Returns the string \"SphereParticleEmitter\"\n * @returns a string containing the class name\n */\n getClassName() {\n return \"SphereParticleEmitter\";\n }\n /**\n * Serializes the particle system to a JSON object.\n * @returns the JSON object\n */\n serialize() {\n const serializationObject = {};\n serializationObject.type = this.getClassName();\n serializationObject.radius = this.radius;\n serializationObject.radiusRange = this.radiusRange;\n serializationObject.directionRandomizer = this.directionRandomizer;\n return serializationObject;\n }\n /**\n * Parse properties from a JSON object\n * @param serializationObject defines the JSON object\n */\n parse(serializationObject) {\n this.radius = serializationObject.radius;\n this.radiusRange = serializationObject.radiusRange;\n this.directionRandomizer = serializationObject.directionRandomizer;\n }\n }\n /**\n * Particle emitter emitting particles from the inside of a sphere.\n * It emits the particles randomly between two vectors.\n */\n class SphereDirectedParticleEmitter extends SphereParticleEmitter {\n /**\n * Creates a new instance SphereDirectedParticleEmitter\n * @param radius the radius of the emission sphere (1 by default)\n * @param direction1 the min limit of the emission direction (up vector by default)\n * @param direction2 the max limit of the emission direction (up vector by default)\n */\n constructor(radius = 1, \n /**\n * The min limit of the emission direction.\n */\n direction1 = new Vector3(0, 1, 0), \n /**\n * The max limit of the emission direction.\n */\n direction2 = new Vector3(0, 1, 0)) {\n super(radius);\n this.direction1 = direction1;\n this.direction2 = direction2;\n }\n /**\n * Called by the particle System when the direction is computed for the created particle.\n * @param worldMatrix is the world matrix of the particle system\n * @param directionToUpdate is the direction vector to update with the result\n */\n startDirectionFunction(worldMatrix, directionToUpdate) {\n const randX = Scalar.RandomRange(this.direction1.x, this.direction2.x);\n const randY = Scalar.RandomRange(this.direction1.y, this.direction2.y);\n const randZ = Scalar.RandomRange(this.direction1.z, this.direction2.z);\n Vector3.TransformNormalFromFloatsToRef(randX, randY, randZ, worldMatrix, directionToUpdate);\n }\n /**\n * Clones the current emitter and returns a copy of it\n * @returns the new emitter\n */\n clone() {\n const newOne = new SphereDirectedParticleEmitter(this.radius, this.direction1, this.direction2);\n DeepCopier.DeepCopy(this, newOne);\n return newOne;\n }\n /**\n * Called by the GPUParticleSystem to setup the update shader\n * @param uboOrEffect defines the update shader\n */\n applyToShader(uboOrEffect) {\n uboOrEffect.setFloat(\"radius\", this.radius);\n uboOrEffect.setFloat(\"radiusRange\", this.radiusRange);\n uboOrEffect.setVector3(\"direction1\", this.direction1);\n uboOrEffect.setVector3(\"direction2\", this.direction2);\n }\n /**\n * Creates the structure of the ubo for this particle emitter\n * @param ubo ubo to create the structure for\n */\n buildUniformLayout(ubo) {\n ubo.addUniform(\"radius\", 1);\n ubo.addUniform(\"radiusRange\", 1);\n ubo.addUniform(\"direction1\", 3);\n ubo.addUniform(\"direction2\", 3);\n }\n /**\n * Returns a string to use to update the GPU particles update shader\n * @returns a string containing the defines string\n */\n getEffectDefines() {\n return \"#define SPHEREEMITTER\\n#define DIRECTEDSPHEREEMITTER\";\n }\n /**\n * Returns the string \"SphereDirectedParticleEmitter\"\n * @returns a string containing the class name\n */\n getClassName() {\n return \"SphereDirectedParticleEmitter\";\n }\n /**\n * Serializes the particle system to a JSON object.\n * @returns the JSON object\n */\n serialize() {\n const serializationObject = super.serialize();\n serializationObject.direction1 = this.direction1.asArray();\n serializationObject.direction2 = this.direction2.asArray();\n return serializationObject;\n }\n /**\n * Parse properties from a JSON object\n * @param serializationObject defines the JSON object\n */\n parse(serializationObject) {\n super.parse(serializationObject);\n this.direction1.copyFrom(serializationObject.direction1);\n this.direction2.copyFrom(serializationObject.direction2);\n }\n }\n\n /**\n * Particle emitter emitting particles from the inside of a box.\n * It emits the particles randomly between 2 given directions.\n */\n class MeshParticleEmitter {\n /** Defines the mesh to use as source */\n get mesh() {\n return this._mesh;\n }\n set mesh(value) {\n if (this._mesh === value) {\n return;\n }\n this._mesh = value;\n if (value) {\n this._indices = value.getIndices();\n this._positions = value.getVerticesData(VertexBuffer.PositionKind);\n this._normals = value.getVerticesData(VertexBuffer.NormalKind);\n }\n else {\n this._indices = null;\n this._positions = null;\n this._normals = null;\n }\n }\n /**\n * Creates a new instance MeshParticleEmitter\n * @param mesh defines the mesh to use as source\n */\n constructor(mesh = null) {\n this._indices = null;\n this._positions = null;\n this._normals = null;\n this._storedNormal = Vector3.Zero();\n this._mesh = null;\n /**\n * Random direction of each particle after it has been emitted, between direction1 and direction2 vectors.\n */\n this.direction1 = new Vector3(0, 1.0, 0);\n /**\n * Random direction of each particle after it has been emitted, between direction1 and direction2 vectors.\n */\n this.direction2 = new Vector3(0, 1.0, 0);\n /**\n * Gets or sets a boolean indicating that particle directions must be built from mesh face normals\n */\n this.useMeshNormalsForDirection = true;\n this.mesh = mesh;\n }\n /**\n * Called by the particle System when the direction is computed for the created particle.\n * @param worldMatrix is the world matrix of the particle system\n * @param directionToUpdate is the direction vector to update with the result\n * @param particle is the particle we are computed the direction for\n * @param isLocal defines if the direction should be set in local space\n */\n startDirectionFunction(worldMatrix, directionToUpdate, particle, isLocal) {\n if (this.useMeshNormalsForDirection && this._normals) {\n Vector3.TransformNormalToRef(this._storedNormal, worldMatrix, directionToUpdate);\n return;\n }\n const randX = Scalar.RandomRange(this.direction1.x, this.direction2.x);\n const randY = Scalar.RandomRange(this.direction1.y, this.direction2.y);\n const randZ = Scalar.RandomRange(this.direction1.z, this.direction2.z);\n if (isLocal) {\n directionToUpdate.copyFromFloats(randX, randY, randZ);\n return;\n }\n Vector3.TransformNormalFromFloatsToRef(randX, randY, randZ, worldMatrix, directionToUpdate);\n }\n /**\n * Called by the particle System when the position is computed for the created particle.\n * @param worldMatrix is the world matrix of the particle system\n * @param positionToUpdate is the position vector to update with the result\n * @param particle is the particle we are computed the position for\n * @param isLocal defines if the position should be set in local space\n */\n startPositionFunction(worldMatrix, positionToUpdate, particle, isLocal) {\n if (!this._indices || !this._positions) {\n return;\n }\n const randomFaceIndex = (3 * Math.random() * (this._indices.length / 3)) | 0;\n const bu = Math.random();\n const bv = Math.random() * (1.0 - bu);\n const bw = 1.0 - bu - bv;\n const faceIndexA = this._indices[randomFaceIndex];\n const faceIndexB = this._indices[randomFaceIndex + 1];\n const faceIndexC = this._indices[randomFaceIndex + 2];\n const vertexA = TmpVectors.Vector3[0];\n const vertexB = TmpVectors.Vector3[1];\n const vertexC = TmpVectors.Vector3[2];\n const randomVertex = TmpVectors.Vector3[3];\n Vector3.FromArrayToRef(this._positions, faceIndexA * 3, vertexA);\n Vector3.FromArrayToRef(this._positions, faceIndexB * 3, vertexB);\n Vector3.FromArrayToRef(this._positions, faceIndexC * 3, vertexC);\n randomVertex.x = bu * vertexA.x + bv * vertexB.x + bw * vertexC.x;\n randomVertex.y = bu * vertexA.y + bv * vertexB.y + bw * vertexC.y;\n randomVertex.z = bu * vertexA.z + bv * vertexB.z + bw * vertexC.z;\n if (isLocal) {\n positionToUpdate.copyFromFloats(randomVertex.x, randomVertex.y, randomVertex.z);\n }\n else {\n Vector3.TransformCoordinatesFromFloatsToRef(randomVertex.x, randomVertex.y, randomVertex.z, worldMatrix, positionToUpdate);\n }\n if (this.useMeshNormalsForDirection && this._normals) {\n Vector3.FromArrayToRef(this._normals, faceIndexA * 3, vertexA);\n Vector3.FromArrayToRef(this._normals, faceIndexB * 3, vertexB);\n Vector3.FromArrayToRef(this._normals, faceIndexC * 3, vertexC);\n this._storedNormal.x = bu * vertexA.x + bv * vertexB.x + bw * vertexC.x;\n this._storedNormal.y = bu * vertexA.y + bv * vertexB.y + bw * vertexC.y;\n this._storedNormal.z = bu * vertexA.z + bv * vertexB.z + bw * vertexC.z;\n }\n }\n /**\n * Clones the current emitter and returns a copy of it\n * @returns the new emitter\n */\n clone() {\n const newOne = new MeshParticleEmitter(this.mesh);\n DeepCopier.DeepCopy(this, newOne);\n return newOne;\n }\n /**\n * Called by the GPUParticleSystem to setup the update shader\n * @param uboOrEffect defines the update shader\n */\n applyToShader(uboOrEffect) {\n uboOrEffect.setVector3(\"direction1\", this.direction1);\n uboOrEffect.setVector3(\"direction2\", this.direction2);\n }\n /**\n * Creates the structure of the ubo for this particle emitter\n * @param ubo ubo to create the structure for\n */\n buildUniformLayout(ubo) {\n ubo.addUniform(\"direction1\", 3);\n ubo.addUniform(\"direction2\", 3);\n }\n /**\n * Returns a string to use to update the GPU particles update shader\n * @returns a string containing the defines string\n */\n getEffectDefines() {\n return \"\";\n }\n /**\n * Returns the string \"BoxParticleEmitter\"\n * @returns a string containing the class name\n */\n getClassName() {\n return \"MeshParticleEmitter\";\n }\n /**\n * Serializes the particle system to a JSON object.\n * @returns the JSON object\n */\n serialize() {\n var _a;\n const serializationObject = {};\n serializationObject.type = this.getClassName();\n serializationObject.direction1 = this.direction1.asArray();\n serializationObject.direction2 = this.direction2.asArray();\n serializationObject.meshId = (_a = this.mesh) === null || _a === void 0 ? void 0 : _a.id;\n serializationObject.useMeshNormalsForDirection = this.useMeshNormalsForDirection;\n return serializationObject;\n }\n /**\n * Parse properties from a JSON object\n * @param serializationObject defines the JSON object\n * @param scene defines the hosting scene\n */\n parse(serializationObject, scene) {\n Vector3.FromArrayToRef(serializationObject.direction1, 0, this.direction1);\n Vector3.FromArrayToRef(serializationObject.direction2, 0, this.direction2);\n if (serializationObject.meshId && scene) {\n this.mesh = scene.getLastMeshById(serializationObject.meshId);\n }\n this.useMeshNormalsForDirection = serializationObject.useMeshNormalsForDirection;\n }\n }\n\n /**\n * This represents the base class for particle system in Babylon.\n * Particles are often small sprites used to simulate hard-to-reproduce phenomena like fire, smoke, water, or abstract visual effects like magic glitter and faery dust.\n * Particles can take different shapes while emitted like box, sphere, cone or you can write your custom function.\n * @example https://doc.babylonjs.com/features/featuresDeepDive/particles/particle_system/particle_system_intro\n */\n class BaseParticleSystem {\n /**\n * Gets or sets a texture used to add random noise to particle positions\n */\n get noiseTexture() {\n return this._noiseTexture;\n }\n set noiseTexture(value) {\n if (this._noiseTexture === value) {\n return;\n }\n this._noiseTexture = value;\n this._reset();\n }\n /**\n * Gets or sets whether an animation sprite sheet is enabled or not on the particle system\n */\n get isAnimationSheetEnabled() {\n return this._isAnimationSheetEnabled;\n }\n set isAnimationSheetEnabled(value) {\n if (this._isAnimationSheetEnabled == value) {\n return;\n }\n this._isAnimationSheetEnabled = value;\n this._reset();\n }\n /**\n * Gets or sets a boolean enabling the use of logarithmic depth buffers, which is good for wide depth buffers.\n */\n get useLogarithmicDepth() {\n return this._useLogarithmicDepth;\n }\n set useLogarithmicDepth(value) {\n this._useLogarithmicDepth = value && this.getScene().getEngine().getCaps().fragmentDepthSupported;\n }\n /**\n * Get hosting scene\n * @returns the scene\n */\n getScene() {\n return this._scene;\n }\n _hasTargetStopDurationDependantGradient() {\n return ((this._startSizeGradients && this._startSizeGradients.length > 0) ||\n (this._emitRateGradients && this._emitRateGradients.length > 0) ||\n (this._lifeTimeGradients && this._lifeTimeGradients.length > 0));\n }\n /**\n * Gets the current list of drag gradients.\n * You must use addDragGradient and removeDragGradient to update this list\n * @returns the list of drag gradients\n */\n getDragGradients() {\n return this._dragGradients;\n }\n /**\n * Gets the current list of limit velocity gradients.\n * You must use addLimitVelocityGradient and removeLimitVelocityGradient to update this list\n * @returns the list of limit velocity gradients\n */\n getLimitVelocityGradients() {\n return this._limitVelocityGradients;\n }\n /**\n * Gets the current list of color gradients.\n * You must use addColorGradient and removeColorGradient to update this list\n * @returns the list of color gradients\n */\n getColorGradients() {\n return this._colorGradients;\n }\n /**\n * Gets the current list of size gradients.\n * You must use addSizeGradient and removeSizeGradient to update this list\n * @returns the list of size gradients\n */\n getSizeGradients() {\n return this._sizeGradients;\n }\n /**\n * Gets the current list of color remap gradients.\n * You must use addColorRemapGradient and removeColorRemapGradient to update this list\n * @returns the list of color remap gradients\n */\n getColorRemapGradients() {\n return this._colorRemapGradients;\n }\n /**\n * Gets the current list of alpha remap gradients.\n * You must use addAlphaRemapGradient and removeAlphaRemapGradient to update this list\n * @returns the list of alpha remap gradients\n */\n getAlphaRemapGradients() {\n return this._alphaRemapGradients;\n }\n /**\n * Gets the current list of life time gradients.\n * You must use addLifeTimeGradient and removeLifeTimeGradient to update this list\n * @returns the list of life time gradients\n */\n getLifeTimeGradients() {\n return this._lifeTimeGradients;\n }\n /**\n * Gets the current list of angular speed gradients.\n * You must use addAngularSpeedGradient and removeAngularSpeedGradient to update this list\n * @returns the list of angular speed gradients\n */\n getAngularSpeedGradients() {\n return this._angularSpeedGradients;\n }\n /**\n * Gets the current list of velocity gradients.\n * You must use addVelocityGradient and removeVelocityGradient to update this list\n * @returns the list of velocity gradients\n */\n getVelocityGradients() {\n return this._velocityGradients;\n }\n /**\n * Gets the current list of start size gradients.\n * You must use addStartSizeGradient and removeStartSizeGradient to update this list\n * @returns the list of start size gradients\n */\n getStartSizeGradients() {\n return this._startSizeGradients;\n }\n /**\n * Gets the current list of emit rate gradients.\n * You must use addEmitRateGradient and removeEmitRateGradient to update this list\n * @returns the list of emit rate gradients\n */\n getEmitRateGradients() {\n return this._emitRateGradients;\n }\n /**\n * Random direction of each particle after it has been emitted, between direction1 and direction2 vectors.\n * This only works when particleEmitterTyps is a BoxParticleEmitter\n */\n get direction1() {\n if (this.particleEmitterType.direction1) {\n return this.particleEmitterType.direction1;\n }\n return Vector3.Zero();\n }\n set direction1(value) {\n if (this.particleEmitterType.direction1) {\n this.particleEmitterType.direction1 = value;\n }\n }\n /**\n * Random direction of each particle after it has been emitted, between direction1 and direction2 vectors.\n * This only works when particleEmitterTyps is a BoxParticleEmitter\n */\n get direction2() {\n if (this.particleEmitterType.direction2) {\n return this.particleEmitterType.direction2;\n }\n return Vector3.Zero();\n }\n set direction2(value) {\n if (this.particleEmitterType.direction2) {\n this.particleEmitterType.direction2 = value;\n }\n }\n /**\n * Minimum box point around our emitter. Our emitter is the center of particles source, but if you want your particles to emit from more than one point, then you can tell it to do so.\n * This only works when particleEmitterTyps is a BoxParticleEmitter\n */\n get minEmitBox() {\n if (this.particleEmitterType.minEmitBox) {\n return this.particleEmitterType.minEmitBox;\n }\n return Vector3.Zero();\n }\n set minEmitBox(value) {\n if (this.particleEmitterType.minEmitBox) {\n this.particleEmitterType.minEmitBox = value;\n }\n }\n /**\n * Maximum box point around our emitter. Our emitter is the center of particles source, but if you want your particles to emit from more than one point, then you can tell it to do so.\n * This only works when particleEmitterTyps is a BoxParticleEmitter\n */\n get maxEmitBox() {\n if (this.particleEmitterType.maxEmitBox) {\n return this.particleEmitterType.maxEmitBox;\n }\n return Vector3.Zero();\n }\n set maxEmitBox(value) {\n if (this.particleEmitterType.maxEmitBox) {\n this.particleEmitterType.maxEmitBox = value;\n }\n }\n /**\n * Gets or sets the billboard mode to use when isBillboardBased = true.\n * Value can be: ParticleSystem.BILLBOARDMODE_ALL, ParticleSystem.BILLBOARDMODE_Y, ParticleSystem.BILLBOARDMODE_STRETCHED\n */\n get billboardMode() {\n return this._billboardMode;\n }\n set billboardMode(value) {\n if (this._billboardMode === value) {\n return;\n }\n this._billboardMode = value;\n this._reset();\n }\n /**\n * Gets or sets a boolean indicating if the particles must be rendered as billboard or aligned with the direction\n */\n get isBillboardBased() {\n return this._isBillboardBased;\n }\n set isBillboardBased(value) {\n if (this._isBillboardBased === value) {\n return;\n }\n this._isBillboardBased = value;\n this._reset();\n }\n /**\n * Gets the image processing configuration used either in this material.\n */\n get imageProcessingConfiguration() {\n return this._imageProcessingConfiguration;\n }\n /**\n * Sets the Default image processing configuration used either in the this material.\n *\n * If sets to null, the scene one is in use.\n */\n set imageProcessingConfiguration(value) {\n this._attachImageProcessingConfiguration(value);\n }\n /**\n * Attaches a new image processing configuration to the Standard Material.\n * @param configuration\n */\n _attachImageProcessingConfiguration(configuration) {\n if (configuration === this._imageProcessingConfiguration) {\n return;\n }\n // Pick the scene configuration if needed.\n if (!configuration && this._scene) {\n this._imageProcessingConfiguration = this._scene.imageProcessingConfiguration;\n }\n else {\n this._imageProcessingConfiguration = configuration;\n }\n }\n /** @internal */\n _reset() { }\n /**\n * @internal\n */\n _removeGradientAndTexture(gradient, gradients, texture) {\n if (!gradients) {\n return this;\n }\n let index = 0;\n for (const valueGradient of gradients) {\n if (valueGradient.gradient === gradient) {\n gradients.splice(index, 1);\n break;\n }\n index++;\n }\n if (texture) {\n texture.dispose();\n }\n return this;\n }\n /**\n * Instantiates a particle system.\n * Particles are often small sprites used to simulate hard-to-reproduce phenomena like fire, smoke, water, or abstract visual effects like magic glitter and faery dust.\n * @param name The name of the particle system\n */\n constructor(name) {\n /**\n * List of animations used by the particle system.\n */\n this.animations = [];\n /**\n * The rendering group used by the Particle system to chose when to render.\n */\n this.renderingGroupId = 0;\n /**\n * The emitter represents the Mesh or position we are attaching the particle system to.\n */\n this.emitter = Vector3.Zero();\n /**\n * The maximum number of particles to emit per frame\n */\n this.emitRate = 10;\n /**\n * If you want to launch only a few particles at once, that can be done, as well.\n */\n this.manualEmitCount = -1;\n /**\n * The overall motion speed (0.01 is default update speed, faster updates = faster animation)\n */\n this.updateSpeed = 0.01;\n /**\n * The amount of time the particle system is running (depends of the overall update speed).\n */\n this.targetStopDuration = 0;\n /**\n * Specifies whether the particle system will be disposed once it reaches the end of the animation.\n */\n this.disposeOnStop = false;\n /**\n * Minimum power of emitting particles.\n */\n this.minEmitPower = 1;\n /**\n * Maximum power of emitting particles.\n */\n this.maxEmitPower = 1;\n /**\n * Minimum life time of emitting particles.\n */\n this.minLifeTime = 1;\n /**\n * Maximum life time of emitting particles.\n */\n this.maxLifeTime = 1;\n /**\n * Minimum Size of emitting particles.\n */\n this.minSize = 1;\n /**\n * Maximum Size of emitting particles.\n */\n this.maxSize = 1;\n /**\n * Minimum scale of emitting particles on X axis.\n */\n this.minScaleX = 1;\n /**\n * Maximum scale of emitting particles on X axis.\n */\n this.maxScaleX = 1;\n /**\n * Minimum scale of emitting particles on Y axis.\n */\n this.minScaleY = 1;\n /**\n * Maximum scale of emitting particles on Y axis.\n */\n this.maxScaleY = 1;\n /**\n * Gets or sets the minimal initial rotation in radians.\n */\n this.minInitialRotation = 0;\n /**\n * Gets or sets the maximal initial rotation in radians.\n */\n this.maxInitialRotation = 0;\n /**\n * Minimum angular speed of emitting particles (Z-axis rotation for each particle).\n */\n this.minAngularSpeed = 0;\n /**\n * Maximum angular speed of emitting particles (Z-axis rotation for each particle).\n */\n this.maxAngularSpeed = 0;\n /**\n * The layer mask we are rendering the particles through.\n */\n this.layerMask = 0x0fffffff;\n /**\n * This can help using your own shader to render the particle system.\n * The according effect will be created\n */\n this.customShader = null;\n /**\n * By default particle system starts as soon as they are created. This prevents the\n * automatic start to happen and let you decide when to start emitting particles.\n */\n this.preventAutoStart = false;\n /** @internal */\n this._wasDispatched = false;\n this._rootUrl = \"\";\n /** Gets or sets the strength to apply to the noise value (default is (10, 10, 10)) */\n this.noiseStrength = new Vector3(10, 10, 10);\n /**\n * Callback triggered when the particle animation is ending.\n */\n this.onAnimationEnd = null;\n /**\n * Blend mode use to render the particle, it can be either ParticleSystem.BLENDMODE_ONEONE or ParticleSystem.BLENDMODE_STANDARD.\n */\n this.blendMode = BaseParticleSystem.BLENDMODE_ONEONE;\n /**\n * Forces the particle to write their depth information to the depth buffer. This can help preventing other draw calls\n * to override the particles.\n */\n this.forceDepthWrite = false;\n /** Gets or sets a value indicating how many cycles (or frames) must be executed before first rendering (this value has to be set before starting the system). Default is 0 */\n this.preWarmCycles = 0;\n /** Gets or sets a value indicating the time step multiplier to use in pre-warm mode (default is 1) */\n this.preWarmStepOffset = 1;\n /**\n * If using a spritesheet (isAnimationSheetEnabled) defines the speed of the sprite loop (default is 1 meaning the animation will play once during the entire particle lifetime)\n */\n this.spriteCellChangeSpeed = 1;\n /**\n * If using a spritesheet (isAnimationSheetEnabled) defines the first sprite cell to display\n */\n this.startSpriteCellID = 0;\n /**\n * If using a spritesheet (isAnimationSheetEnabled) defines the last sprite cell to display\n */\n this.endSpriteCellID = 0;\n /**\n * If using a spritesheet (isAnimationSheetEnabled), defines the sprite cell width to use\n */\n this.spriteCellWidth = 0;\n /**\n * If using a spritesheet (isAnimationSheetEnabled), defines the sprite cell height to use\n */\n this.spriteCellHeight = 0;\n /**\n * If using a spritesheet (isAnimationSheetEnabled), defines wether the sprite animation is looping\n */\n this.spriteCellLoop = true;\n /**\n * This allows the system to random pick the start cell ID between startSpriteCellID and endSpriteCellID\n */\n this.spriteRandomStartCell = false;\n /** Gets or sets a Vector2 used to move the pivot (by default (0,0)) */\n this.translationPivot = new Vector2(0, 0);\n /**\n * Gets or sets a boolean indicating that hosted animations (in the system.animations array) must be started when system.start() is called\n */\n this.beginAnimationOnStart = false;\n /**\n * Gets or sets the frame to start the animation from when beginAnimationOnStart is true\n */\n this.beginAnimationFrom = 0;\n /**\n * Gets or sets the frame to end the animation on when beginAnimationOnStart is true\n */\n this.beginAnimationTo = 60;\n /**\n * Gets or sets a boolean indicating if animations must loop when beginAnimationOnStart is true\n */\n this.beginAnimationLoop = false;\n /**\n * Gets or sets a world offset applied to all particles\n */\n this.worldOffset = new Vector3(0, 0, 0);\n this._useLogarithmicDepth = false;\n /**\n * You can use gravity if you want to give an orientation to your particles.\n */\n this.gravity = Vector3.Zero();\n this._colorGradients = null;\n this._sizeGradients = null;\n this._lifeTimeGradients = null;\n this._angularSpeedGradients = null;\n this._velocityGradients = null;\n this._limitVelocityGradients = null;\n this._dragGradients = null;\n this._emitRateGradients = null;\n this._startSizeGradients = null;\n this._rampGradients = null;\n this._colorRemapGradients = null;\n this._alphaRemapGradients = null;\n /**\n * Defines the delay in milliseconds before starting the system (0 by default)\n */\n this.startDelay = 0;\n /** Gets or sets a value indicating the damping to apply if the limit velocity factor is reached */\n this.limitVelocityDamping = 0.4;\n /**\n * Random color of each particle after it has been emitted, between color1 and color2 vectors\n */\n this.color1 = new Color4(1.0, 1.0, 1.0, 1.0);\n /**\n * Random color of each particle after it has been emitted, between color1 and color2 vectors\n */\n this.color2 = new Color4(1.0, 1.0, 1.0, 1.0);\n /**\n * Color the particle will have at the end of its lifetime\n */\n this.colorDead = new Color4(0, 0, 0, 1.0);\n /**\n * An optional mask to filter some colors out of the texture, or filter a part of the alpha channel\n */\n this.textureMask = new Color4(1.0, 1.0, 1.0, 1.0);\n /** @internal */\n this._isSubEmitter = false;\n /** @internal */\n this._billboardMode = 7;\n /** @internal */\n this._isBillboardBased = true;\n /**\n * Local cache of defines for image processing.\n */\n this._imageProcessingConfigurationDefines = new ImageProcessingConfigurationDefines();\n this.id = name;\n this.name = name;\n }\n /**\n * Creates a Point Emitter for the particle system (emits directly from the emitter position)\n * @param direction1 Particles are emitted between the direction1 and direction2 from within the box\n * @param direction2 Particles are emitted between the direction1 and direction2 from within the box\n * @returns the emitter\n */\n createPointEmitter(direction1, direction2) {\n const particleEmitter = new PointParticleEmitter();\n particleEmitter.direction1 = direction1;\n particleEmitter.direction2 = direction2;\n this.particleEmitterType = particleEmitter;\n return particleEmitter;\n }\n /**\n * Creates a Hemisphere Emitter for the particle system (emits along the hemisphere radius)\n * @param radius The radius of the hemisphere to emit from\n * @param radiusRange The range of the hemisphere to emit from [0-1] 0 Surface Only, 1 Entire Radius\n * @returns the emitter\n */\n createHemisphericEmitter(radius = 1, radiusRange = 1) {\n const particleEmitter = new HemisphericParticleEmitter(radius, radiusRange);\n this.particleEmitterType = particleEmitter;\n return particleEmitter;\n }\n /**\n * Creates a Sphere Emitter for the particle system (emits along the sphere radius)\n * @param radius The radius of the sphere to emit from\n * @param radiusRange The range of the sphere to emit from [0-1] 0 Surface Only, 1 Entire Radius\n * @returns the emitter\n */\n createSphereEmitter(radius = 1, radiusRange = 1) {\n const particleEmitter = new SphereParticleEmitter(radius, radiusRange);\n this.particleEmitterType = particleEmitter;\n return particleEmitter;\n }\n /**\n * Creates a Directed Sphere Emitter for the particle system (emits between direction1 and direction2)\n * @param radius The radius of the sphere to emit from\n * @param direction1 Particles are emitted between the direction1 and direction2 from within the sphere\n * @param direction2 Particles are emitted between the direction1 and direction2 from within the sphere\n * @returns the emitter\n */\n createDirectedSphereEmitter(radius = 1, direction1 = new Vector3(0, 1.0, 0), direction2 = new Vector3(0, 1.0, 0)) {\n const particleEmitter = new SphereDirectedParticleEmitter(radius, direction1, direction2);\n this.particleEmitterType = particleEmitter;\n return particleEmitter;\n }\n /**\n * Creates a Cylinder Emitter for the particle system (emits from the cylinder to the particle position)\n * @param radius The radius of the emission cylinder\n * @param height The height of the emission cylinder\n * @param radiusRange The range of emission [0-1] 0 Surface only, 1 Entire Radius\n * @param directionRandomizer How much to randomize the particle direction [0-1]\n * @returns the emitter\n */\n createCylinderEmitter(radius = 1, height = 1, radiusRange = 1, directionRandomizer = 0) {\n const particleEmitter = new CylinderParticleEmitter(radius, height, radiusRange, directionRandomizer);\n this.particleEmitterType = particleEmitter;\n return particleEmitter;\n }\n /**\n * Creates a Directed Cylinder Emitter for the particle system (emits between direction1 and direction2)\n * @param radius The radius of the cylinder to emit from\n * @param height The height of the emission cylinder\n * @param radiusRange the range of the emission cylinder [0-1] 0 Surface only, 1 Entire Radius (1 by default)\n * @param direction1 Particles are emitted between the direction1 and direction2 from within the cylinder\n * @param direction2 Particles are emitted between the direction1 and direction2 from within the cylinder\n * @returns the emitter\n */\n createDirectedCylinderEmitter(radius = 1, height = 1, radiusRange = 1, direction1 = new Vector3(0, 1.0, 0), direction2 = new Vector3(0, 1.0, 0)) {\n const particleEmitter = new CylinderDirectedParticleEmitter(radius, height, radiusRange, direction1, direction2);\n this.particleEmitterType = particleEmitter;\n return particleEmitter;\n }\n /**\n * Creates a Cone Emitter for the particle system (emits from the cone to the particle position)\n * @param radius The radius of the cone to emit from\n * @param angle The base angle of the cone\n * @returns the emitter\n */\n createConeEmitter(radius = 1, angle = Math.PI / 4) {\n const particleEmitter = new ConeParticleEmitter(radius, angle);\n this.particleEmitterType = particleEmitter;\n return particleEmitter;\n }\n /**\n * Creates a Box Emitter for the particle system. (emits between direction1 and direction2 from withing the box defined by minEmitBox and maxEmitBox)\n * @param direction1 Particles are emitted between the direction1 and direction2 from within the box\n * @param direction2 Particles are emitted between the direction1 and direction2 from within the box\n * @param minEmitBox Particles are emitted from the box between minEmitBox and maxEmitBox\n * @param maxEmitBox Particles are emitted from the box between minEmitBox and maxEmitBox\n * @returns the emitter\n */\n createBoxEmitter(direction1, direction2, minEmitBox, maxEmitBox) {\n const particleEmitter = new BoxParticleEmitter();\n this.particleEmitterType = particleEmitter;\n this.direction1 = direction1;\n this.direction2 = direction2;\n this.minEmitBox = minEmitBox;\n this.maxEmitBox = maxEmitBox;\n return particleEmitter;\n }\n }\n /**\n * Source color is added to the destination color without alpha affecting the result\n */\n BaseParticleSystem.BLENDMODE_ONEONE = 0;\n /**\n * Blend current color and particle color using particle’s alpha\n */\n BaseParticleSystem.BLENDMODE_STANDARD = 1;\n /**\n * Add current color and particle color multiplied by particle’s alpha\n */\n BaseParticleSystem.BLENDMODE_ADD = 2;\n /**\n * Multiply current color with particle color\n */\n BaseParticleSystem.BLENDMODE_MULTIPLY = 3;\n /**\n * Multiply current color with particle color then add current color and particle color multiplied by particle’s alpha\n */\n BaseParticleSystem.BLENDMODE_MULTIPLYADD = 4;\n\n /**\n * Block used to expand a Color3/4 into 4 outputs (one for each component)\n */\n class ColorSplitterBlock extends NodeMaterialBlock {\n /**\n * Create a new ColorSplitterBlock\n * @param name defines the block name\n */\n constructor(name) {\n super(name, NodeMaterialBlockTargets.Neutral);\n this.registerInput(\"rgba\", NodeMaterialBlockConnectionPointTypes.Color4, true);\n this.registerInput(\"rgb \", NodeMaterialBlockConnectionPointTypes.Color3, true);\n this.registerOutput(\"rgb\", NodeMaterialBlockConnectionPointTypes.Color3);\n this.registerOutput(\"r\", NodeMaterialBlockConnectionPointTypes.Float);\n this.registerOutput(\"g\", NodeMaterialBlockConnectionPointTypes.Float);\n this.registerOutput(\"b\", NodeMaterialBlockConnectionPointTypes.Float);\n this.registerOutput(\"a\", NodeMaterialBlockConnectionPointTypes.Float);\n this.inputsAreExclusive = true;\n }\n /**\n * Gets the current class name\n * @returns the class name\n */\n getClassName() {\n return \"ColorSplitterBlock\";\n }\n /**\n * Gets the rgba component (input)\n */\n get rgba() {\n return this._inputs[0];\n }\n /**\n * Gets the rgb component (input)\n */\n get rgbIn() {\n return this._inputs[1];\n }\n /**\n * Gets the rgb component (output)\n */\n get rgbOut() {\n return this._outputs[0];\n }\n /**\n * Gets the r component (output)\n */\n get r() {\n return this._outputs[1];\n }\n /**\n * Gets the g component (output)\n */\n get g() {\n return this._outputs[2];\n }\n /**\n * Gets the b component (output)\n */\n get b() {\n return this._outputs[3];\n }\n /**\n * Gets the a component (output)\n */\n get a() {\n return this._outputs[4];\n }\n _inputRename(name) {\n if (name === \"rgb \") {\n return \"rgbIn\";\n }\n return name;\n }\n _outputRename(name) {\n if (name === \"rgb\") {\n return \"rgbOut\";\n }\n return name;\n }\n _buildBlock(state) {\n super._buildBlock(state);\n const input = this.rgba.isConnected ? this.rgba : this.rgbIn;\n if (!input.isConnected) {\n return;\n }\n const rgbOutput = this._outputs[0];\n const rOutput = this._outputs[1];\n const gOutput = this._outputs[2];\n const bOutput = this._outputs[3];\n const aOutput = this._outputs[4];\n if (rgbOutput.hasEndpoints) {\n state.compilationString += this._declareOutput(rgbOutput, state) + ` = ${input.associatedVariableName}.rgb;\\n`;\n }\n if (rOutput.hasEndpoints) {\n state.compilationString += this._declareOutput(rOutput, state) + ` = ${input.associatedVariableName}.r;\\n`;\n }\n if (gOutput.hasEndpoints) {\n state.compilationString += this._declareOutput(gOutput, state) + ` = ${input.associatedVariableName}.g;\\n`;\n }\n if (bOutput.hasEndpoints) {\n state.compilationString += this._declareOutput(bOutput, state) + ` = ${input.associatedVariableName}.b;\\n`;\n }\n if (aOutput.hasEndpoints) {\n state.compilationString += this._declareOutput(aOutput, state) + ` = ${input.associatedVariableName}.a;\\n`;\n }\n return this;\n }\n }\n RegisterClass(\"BABYLON.ColorSplitterBlock\", ColorSplitterBlock);\n\n /**\n * Defines the Procedural Texture scene component responsible to manage any Procedural Texture\n * in a given scene.\n */\n class ProceduralTextureSceneComponent {\n /**\n * Creates a new instance of the component for the given scene\n * @param scene Defines the scene to register the component in\n */\n constructor(scene) {\n /**\n * The component name helpful to identify the component in the list of scene components.\n */\n this.name = SceneComponentConstants.NAME_PROCEDURALTEXTURE;\n this.scene = scene;\n this.scene.proceduralTextures = new Array();\n }\n /**\n * Registers the component in a given scene\n */\n register() {\n this.scene._beforeClearStage.registerStep(SceneComponentConstants.STEP_BEFORECLEAR_PROCEDURALTEXTURE, this, this._beforeClear);\n }\n /**\n * Rebuilds the elements related to this component in case of\n * context lost for instance.\n */\n rebuild() {\n // Nothing to do here.\n }\n /**\n * Disposes the component and the associated resources.\n */\n dispose() {\n // Nothing to do here.\n }\n _beforeClear() {\n if (this.scene.proceduralTexturesEnabled) {\n Tools.StartPerformanceCounter(\"Procedural textures\", this.scene.proceduralTextures.length > 0);\n for (let proceduralIndex = 0; proceduralIndex < this.scene.proceduralTextures.length; proceduralIndex++) {\n const proceduralTexture = this.scene.proceduralTextures[proceduralIndex];\n if (proceduralTexture._shouldRender()) {\n proceduralTexture.render();\n }\n }\n Tools.EndPerformanceCounter(\"Procedural textures\", this.scene.proceduralTextures.length > 0);\n }\n }\n }\n\n // Do not edit.\n const name$1X = \"proceduralVertexShader\";\n const shader$1X = `attribute vec2 position;varying vec2 vPosition;varying vec2 vUV;const vec2 madd=vec2(0.5,0.5);\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\nvPosition=position;vUV=position*madd+madd;gl_Position=vec4(position,0.0,1.0);\n#define CUSTOM_VERTEX_MAIN_END\n}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$1X] = shader$1X;\n\n /**\n * Procedural texturing is a way to programmatically create a texture. There are 2 types of procedural textures: code-only, and code that references some classic 2D images, sometimes calmpler' images.\n * This is the base class of any Procedural texture and contains most of the shareable code.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/proceduralTextures\n */\n class ProceduralTexture extends Texture {\n /**\n * Instantiates a new procedural texture.\n * Procedural texturing is a way to programmatically create a texture. There are 2 types of procedural textures: code-only, and code that references some classic 2D images, sometimes called 'refMaps' or 'sampler' images.\n * This is the base class of any Procedural texture and contains most of the shareable code.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/proceduralTextures\n * @param name Define the name of the texture\n * @param size Define the size of the texture to create\n * @param fragment Define the fragment shader to use to generate the texture or null if it is defined later\n * @param scene Define the scene the texture belongs to\n * @param fallbackTexture Define a fallback texture in case there were issues to create the custom texture\n * @param generateMipMaps Define if the texture should creates mip maps or not\n * @param isCube Define if the texture is a cube texture or not (this will render each faces of the cube)\n * @param textureType The FBO internal texture type\n */\n constructor(name, size, fragment, scene, fallbackTexture = null, generateMipMaps = true, isCube = false, textureType = 0) {\n super(null, scene, !generateMipMaps);\n /**\n * Define if the texture is enabled or not (disabled texture will not render)\n */\n this.isEnabled = true;\n /**\n * Define if the texture must be cleared before rendering (default is true)\n */\n this.autoClear = true;\n /**\n * Event raised when the texture is generated\n */\n this.onGeneratedObservable = new Observable$1();\n /**\n * Event raised before the texture is generated\n */\n this.onBeforeGenerationObservable = new Observable$1();\n /**\n * Gets or sets the node material used to create this texture (null if the texture was manually created)\n */\n this.nodeMaterialSource = null;\n /** @internal */\n this._textures = {};\n this._currentRefreshId = -1;\n this._frameId = -1;\n this._refreshRate = 1;\n this._vertexBuffers = {};\n this._uniforms = new Array();\n this._samplers = new Array();\n this._floats = {};\n this._ints = {};\n this._floatsArrays = {};\n this._colors3 = {};\n this._colors4 = {};\n this._vectors2 = {};\n this._vectors3 = {};\n this._matrices = {};\n this._fallbackTextureUsed = false;\n this._cachedDefines = null;\n this._contentUpdateId = -1;\n this._rtWrapper = null;\n scene = this.getScene() || EngineStore.LastCreatedScene;\n let component = scene._getComponent(SceneComponentConstants.NAME_PROCEDURALTEXTURE);\n if (!component) {\n component = new ProceduralTextureSceneComponent(scene);\n scene._addComponent(component);\n }\n scene.proceduralTextures.push(this);\n this._fullEngine = scene.getEngine();\n this.name = name;\n this.isRenderTarget = true;\n this._size = size;\n this._textureType = textureType;\n this._generateMipMaps = generateMipMaps;\n this._drawWrapper = new DrawWrapper(this._fullEngine);\n this.setFragment(fragment);\n this._fallbackTexture = fallbackTexture;\n const rtWrapper = this._createRtWrapper(isCube, size, generateMipMaps, textureType);\n this._texture = rtWrapper.texture;\n // VBO\n const vertices = [];\n vertices.push(1, 1);\n vertices.push(-1, 1);\n vertices.push(-1, -1);\n vertices.push(1, -1);\n this._vertexBuffers[VertexBuffer.PositionKind] = new VertexBuffer(this._fullEngine, vertices, VertexBuffer.PositionKind, false, false, 2);\n this._createIndexBuffer();\n }\n _createRtWrapper(isCube, size, generateMipMaps, textureType) {\n if (isCube) {\n this._rtWrapper = this._fullEngine.createRenderTargetCubeTexture(size, {\n generateMipMaps: generateMipMaps,\n generateDepthBuffer: false,\n generateStencilBuffer: false,\n type: textureType,\n });\n this.setFloat(\"face\", 0);\n }\n else {\n this._rtWrapper = this._fullEngine.createRenderTargetTexture(size, {\n generateMipMaps: generateMipMaps,\n generateDepthBuffer: false,\n generateStencilBuffer: false,\n type: textureType,\n });\n }\n return this._rtWrapper;\n }\n /**\n * The effect that is created when initializing the post process.\n * @returns The created effect corresponding the the postprocess.\n */\n getEffect() {\n return this._drawWrapper.effect;\n }\n /**\n * @internal*\n */\n _setEffect(effect) {\n this._drawWrapper.effect = effect;\n }\n /**\n * Gets texture content (Use this function wisely as reading from a texture can be slow)\n * @returns an ArrayBufferView promise (Uint8Array or Float32Array)\n */\n getContent() {\n if (this._contentData && this._frameId === this._contentUpdateId) {\n return this._contentData;\n }\n if (this._contentData) {\n this._contentData.then((buffer) => {\n this._contentData = this.readPixels(0, 0, buffer);\n this._contentUpdateId = this._frameId;\n });\n }\n else {\n this._contentData = this.readPixels(0, 0);\n this._contentUpdateId = this._frameId;\n }\n return this._contentData;\n }\n _createIndexBuffer() {\n const engine = this._fullEngine;\n // Indices\n const indices = [];\n indices.push(0);\n indices.push(1);\n indices.push(2);\n indices.push(0);\n indices.push(2);\n indices.push(3);\n this._indexBuffer = engine.createIndexBuffer(indices);\n }\n /** @internal */\n _rebuild() {\n const vb = this._vertexBuffers[VertexBuffer.PositionKind];\n if (vb) {\n vb._rebuild();\n }\n this._createIndexBuffer();\n if (this.refreshRate === RenderTargetTexture.REFRESHRATE_RENDER_ONCE) {\n this.refreshRate = RenderTargetTexture.REFRESHRATE_RENDER_ONCE;\n }\n }\n /**\n * Resets the texture in order to recreate its associated resources.\n * This can be called in case of context loss\n */\n reset() {\n var _a;\n (_a = this._drawWrapper.effect) === null || _a === void 0 ? void 0 : _a.dispose();\n }\n _getDefines() {\n return \"\";\n }\n /**\n * Is the texture ready to be used ? (rendered at least once)\n * @returns true if ready, otherwise, false.\n */\n isReady() {\n const engine = this._fullEngine;\n let shaders;\n if (this.nodeMaterialSource) {\n return this._drawWrapper.effect.isReady();\n }\n if (!this._fragment) {\n return false;\n }\n if (this._fallbackTextureUsed) {\n return true;\n }\n if (!this._texture) {\n return false;\n }\n const defines = this._getDefines();\n if (this._drawWrapper.effect && defines === this._cachedDefines && this._drawWrapper.effect.isReady()) {\n return true;\n }\n if (this._fragment.fragmentElement !== undefined) {\n shaders = { vertex: \"procedural\", fragmentElement: this._fragment.fragmentElement };\n }\n else {\n shaders = { vertex: \"procedural\", fragment: this._fragment };\n }\n if (this._cachedDefines !== defines) {\n this._cachedDefines = defines;\n this._drawWrapper.effect = engine.createEffect(shaders, [VertexBuffer.PositionKind], this._uniforms, this._samplers, defines, undefined, undefined, () => {\n var _a;\n (_a = this._rtWrapper) === null || _a === void 0 ? void 0 : _a.dispose();\n this._rtWrapper = this._texture = null;\n if (this._fallbackTexture) {\n this._texture = this._fallbackTexture._texture;\n if (this._texture) {\n this._texture.incrementReferences();\n }\n }\n this._fallbackTextureUsed = true;\n });\n }\n return this._drawWrapper.effect.isReady();\n }\n /**\n * Resets the refresh counter of the texture and start bak from scratch.\n * Could be useful to regenerate the texture if it is setup to render only once.\n */\n resetRefreshCounter() {\n this._currentRefreshId = -1;\n }\n /**\n * Set the fragment shader to use in order to render the texture.\n * @param fragment This can be set to a path (into the shader store) or to a json object containing a fragmentElement property.\n */\n setFragment(fragment) {\n this._fragment = fragment;\n }\n /**\n * Define the refresh rate of the texture or the rendering frequency.\n * Use 0 to render just once, 1 to render on every frame, 2 to render every two frames and so on...\n */\n get refreshRate() {\n return this._refreshRate;\n }\n set refreshRate(value) {\n this._refreshRate = value;\n this.resetRefreshCounter();\n }\n /** @internal */\n _shouldRender() {\n if (!this.isEnabled || !this.isReady() || !this._texture) {\n if (this._texture) {\n this._texture.isReady = false;\n }\n return false;\n }\n if (this._fallbackTextureUsed) {\n return false;\n }\n if (this._currentRefreshId === -1) {\n // At least render once\n this._currentRefreshId = 1;\n this._frameId++;\n return true;\n }\n if (this.refreshRate === this._currentRefreshId) {\n this._currentRefreshId = 1;\n this._frameId++;\n return true;\n }\n this._currentRefreshId++;\n return false;\n }\n /**\n * Get the size the texture is rendering at.\n * @returns the size (on cube texture it is always squared)\n */\n getRenderSize() {\n return this._size;\n }\n /**\n * Resize the texture to new value.\n * @param size Define the new size the texture should have\n * @param generateMipMaps Define whether the new texture should create mip maps\n */\n resize(size, generateMipMaps) {\n if (this._fallbackTextureUsed || !this._rtWrapper || !this._texture) {\n return;\n }\n const isCube = this._texture.isCube;\n this._rtWrapper.dispose();\n const rtWrapper = this._createRtWrapper(isCube, size, generateMipMaps, this._textureType);\n this._texture = rtWrapper.texture;\n // Update properties\n this._size = size;\n this._generateMipMaps = generateMipMaps;\n }\n _checkUniform(uniformName) {\n if (this._uniforms.indexOf(uniformName) === -1) {\n this._uniforms.push(uniformName);\n }\n }\n /**\n * Set a texture in the shader program used to render.\n * @param name Define the name of the uniform samplers as defined in the shader\n * @param texture Define the texture to bind to this sampler\n * @returns the texture itself allowing \"fluent\" like uniform updates\n */\n setTexture(name, texture) {\n if (this._samplers.indexOf(name) === -1) {\n this._samplers.push(name);\n }\n this._textures[name] = texture;\n return this;\n }\n /**\n * Set a float in the shader.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the texture itself allowing \"fluent\" like uniform updates\n */\n setFloat(name, value) {\n this._checkUniform(name);\n this._floats[name] = value;\n return this;\n }\n /**\n * Set a int in the shader.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the texture itself allowing \"fluent\" like uniform updates\n */\n setInt(name, value) {\n this._checkUniform(name);\n this._ints[name] = value;\n return this;\n }\n /**\n * Set an array of floats in the shader.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the texture itself allowing \"fluent\" like uniform updates\n */\n setFloats(name, value) {\n this._checkUniform(name);\n this._floatsArrays[name] = value;\n return this;\n }\n /**\n * Set a vec3 in the shader from a Color3.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the texture itself allowing \"fluent\" like uniform updates\n */\n setColor3(name, value) {\n this._checkUniform(name);\n this._colors3[name] = value;\n return this;\n }\n /**\n * Set a vec4 in the shader from a Color4.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the texture itself allowing \"fluent\" like uniform updates\n */\n setColor4(name, value) {\n this._checkUniform(name);\n this._colors4[name] = value;\n return this;\n }\n /**\n * Set a vec2 in the shader from a Vector2.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the texture itself allowing \"fluent\" like uniform updates\n */\n setVector2(name, value) {\n this._checkUniform(name);\n this._vectors2[name] = value;\n return this;\n }\n /**\n * Set a vec3 in the shader from a Vector3.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the texture itself allowing \"fluent\" like uniform updates\n */\n setVector3(name, value) {\n this._checkUniform(name);\n this._vectors3[name] = value;\n return this;\n }\n /**\n * Set a mat4 in the shader from a MAtrix.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the texture itself allowing \"fluent\" like uniform updates\n */\n setMatrix(name, value) {\n this._checkUniform(name);\n this._matrices[name] = value;\n return this;\n }\n /**\n * Render the texture to its associated render target.\n * @param useCameraPostProcess Define if camera post process should be applied to the texture\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n render(useCameraPostProcess) {\n var _a, _b;\n const scene = this.getScene();\n if (!scene) {\n return;\n }\n const engine = this._fullEngine;\n // Render\n engine.enableEffect(this._drawWrapper);\n this.onBeforeGenerationObservable.notifyObservers(this);\n engine.setState(false);\n if (!this.nodeMaterialSource) {\n // Texture\n for (const name in this._textures) {\n this._drawWrapper.effect.setTexture(name, this._textures[name]);\n }\n // Float\n for (const name in this._ints) {\n this._drawWrapper.effect.setInt(name, this._ints[name]);\n }\n // Float\n for (const name in this._floats) {\n this._drawWrapper.effect.setFloat(name, this._floats[name]);\n }\n // Floats\n for (const name in this._floatsArrays) {\n this._drawWrapper.effect.setArray(name, this._floatsArrays[name]);\n }\n // Color3\n for (const name in this._colors3) {\n this._drawWrapper.effect.setColor3(name, this._colors3[name]);\n }\n // Color4\n for (const name in this._colors4) {\n const color = this._colors4[name];\n this._drawWrapper.effect.setFloat4(name, color.r, color.g, color.b, color.a);\n }\n // Vector2\n for (const name in this._vectors2) {\n this._drawWrapper.effect.setVector2(name, this._vectors2[name]);\n }\n // Vector3\n for (const name in this._vectors3) {\n this._drawWrapper.effect.setVector3(name, this._vectors3[name]);\n }\n // Matrix\n for (const name in this._matrices) {\n this._drawWrapper.effect.setMatrix(name, this._matrices[name]);\n }\n }\n if (!this._texture || !this._rtWrapper) {\n return;\n }\n (_a = engine._debugPushGroup) === null || _a === void 0 ? void 0 : _a.call(engine, `procedural texture generation for ${this.name}`, 1);\n const viewPort = engine.currentViewport;\n if (this.isCube) {\n for (let face = 0; face < 6; face++) {\n engine.bindFramebuffer(this._rtWrapper, face, undefined, undefined, true);\n // VBOs\n engine.bindBuffers(this._vertexBuffers, this._indexBuffer, this._drawWrapper.effect);\n this._drawWrapper.effect.setFloat(\"face\", face);\n // Clear\n if (this.autoClear) {\n engine.clear(scene.clearColor, true, false, false);\n }\n // Draw order\n engine.drawElementsType(Material.TriangleFillMode, 0, 6);\n }\n }\n else {\n engine.bindFramebuffer(this._rtWrapper, 0, undefined, undefined, true);\n // VBOs\n engine.bindBuffers(this._vertexBuffers, this._indexBuffer, this._drawWrapper.effect);\n // Clear\n if (this.autoClear) {\n engine.clear(scene.clearColor, true, false, false);\n }\n // Draw order\n engine.drawElementsType(Material.TriangleFillMode, 0, 6);\n }\n // Unbind and restore viewport\n engine.unBindFramebuffer(this._rtWrapper, this.isCube);\n if (viewPort) {\n engine.setViewport(viewPort);\n }\n // Mipmaps\n if (this.isCube) {\n engine.generateMipMapsForCubemap(this._texture);\n }\n (_b = engine._debugPopGroup) === null || _b === void 0 ? void 0 : _b.call(engine, 1);\n if (this.onGenerated) {\n this.onGenerated();\n }\n this.onGeneratedObservable.notifyObservers(this);\n }\n /**\n * Clone the texture.\n * @returns the cloned texture\n */\n clone() {\n const textureSize = this.getSize();\n const newTexture = new ProceduralTexture(this.name, textureSize.width, this._fragment, this.getScene(), this._fallbackTexture, this._generateMipMaps);\n // Base texture\n newTexture.hasAlpha = this.hasAlpha;\n newTexture.level = this.level;\n // RenderTarget Texture\n newTexture.coordinatesMode = this.coordinatesMode;\n return newTexture;\n }\n /**\n * Dispose the texture and release its associated resources.\n */\n dispose() {\n const scene = this.getScene();\n if (!scene) {\n return;\n }\n const index = scene.proceduralTextures.indexOf(this);\n if (index >= 0) {\n scene.proceduralTextures.splice(index, 1);\n }\n const vertexBuffer = this._vertexBuffers[VertexBuffer.PositionKind];\n if (vertexBuffer) {\n vertexBuffer.dispose();\n this._vertexBuffers[VertexBuffer.PositionKind] = null;\n }\n if (this._indexBuffer && this._fullEngine._releaseBuffer(this._indexBuffer)) {\n this._indexBuffer = null;\n }\n this.onGeneratedObservable.clear();\n this.onBeforeGenerationObservable.clear();\n super.dispose();\n }\n }\n __decorate$1([\n serialize()\n ], ProceduralTexture.prototype, \"isEnabled\", void 0);\n __decorate$1([\n serialize()\n ], ProceduralTexture.prototype, \"autoClear\", void 0);\n __decorate$1([\n serialize()\n ], ProceduralTexture.prototype, \"_generateMipMaps\", void 0);\n __decorate$1([\n serialize()\n ], ProceduralTexture.prototype, \"_size\", void 0);\n __decorate$1([\n serialize()\n ], ProceduralTexture.prototype, \"refreshRate\", null);\n RegisterClass(\"BABYLON.ProceduralTexture\", ProceduralTexture);\n\n /**\n * Operations supported by the Trigonometry block\n */\n var TrigonometryBlockOperations;\n (function (TrigonometryBlockOperations) {\n /** Cos */\n TrigonometryBlockOperations[TrigonometryBlockOperations[\"Cos\"] = 0] = \"Cos\";\n /** Sin */\n TrigonometryBlockOperations[TrigonometryBlockOperations[\"Sin\"] = 1] = \"Sin\";\n /** Abs */\n TrigonometryBlockOperations[TrigonometryBlockOperations[\"Abs\"] = 2] = \"Abs\";\n /** Exp */\n TrigonometryBlockOperations[TrigonometryBlockOperations[\"Exp\"] = 3] = \"Exp\";\n /** Exp2 */\n TrigonometryBlockOperations[TrigonometryBlockOperations[\"Exp2\"] = 4] = \"Exp2\";\n /** Round */\n TrigonometryBlockOperations[TrigonometryBlockOperations[\"Round\"] = 5] = \"Round\";\n /** Floor */\n TrigonometryBlockOperations[TrigonometryBlockOperations[\"Floor\"] = 6] = \"Floor\";\n /** Ceiling */\n TrigonometryBlockOperations[TrigonometryBlockOperations[\"Ceiling\"] = 7] = \"Ceiling\";\n /** Square root */\n TrigonometryBlockOperations[TrigonometryBlockOperations[\"Sqrt\"] = 8] = \"Sqrt\";\n /** Log */\n TrigonometryBlockOperations[TrigonometryBlockOperations[\"Log\"] = 9] = \"Log\";\n /** Tangent */\n TrigonometryBlockOperations[TrigonometryBlockOperations[\"Tan\"] = 10] = \"Tan\";\n /** Arc tangent */\n TrigonometryBlockOperations[TrigonometryBlockOperations[\"ArcTan\"] = 11] = \"ArcTan\";\n /** Arc cosinus */\n TrigonometryBlockOperations[TrigonometryBlockOperations[\"ArcCos\"] = 12] = \"ArcCos\";\n /** Arc sinus */\n TrigonometryBlockOperations[TrigonometryBlockOperations[\"ArcSin\"] = 13] = \"ArcSin\";\n /** Fraction */\n TrigonometryBlockOperations[TrigonometryBlockOperations[\"Fract\"] = 14] = \"Fract\";\n /** Sign */\n TrigonometryBlockOperations[TrigonometryBlockOperations[\"Sign\"] = 15] = \"Sign\";\n /** To radians (from degrees) */\n TrigonometryBlockOperations[TrigonometryBlockOperations[\"Radians\"] = 16] = \"Radians\";\n /** To degrees (from radians) */\n TrigonometryBlockOperations[TrigonometryBlockOperations[\"Degrees\"] = 17] = \"Degrees\";\n })(TrigonometryBlockOperations || (TrigonometryBlockOperations = {}));\n /**\n * Block used to apply trigonometry operation to floats\n */\n class TrigonometryBlock extends NodeMaterialBlock {\n /**\n * Creates a new TrigonometryBlock\n * @param name defines the block name\n */\n constructor(name) {\n super(name, NodeMaterialBlockTargets.Neutral);\n /**\n * Gets or sets the operation applied by the block\n */\n this.operation = TrigonometryBlockOperations.Cos;\n this.registerInput(\"input\", NodeMaterialBlockConnectionPointTypes.AutoDetect);\n this.registerOutput(\"output\", NodeMaterialBlockConnectionPointTypes.BasedOnInput);\n this._outputs[0]._typeConnectionSource = this._inputs[0];\n }\n /**\n * Gets the current class name\n * @returns the class name\n */\n getClassName() {\n return \"TrigonometryBlock\";\n }\n /**\n * Gets the input component\n */\n get input() {\n return this._inputs[0];\n }\n /**\n * Gets the output component\n */\n get output() {\n return this._outputs[0];\n }\n _buildBlock(state) {\n super._buildBlock(state);\n const output = this._outputs[0];\n let operation = \"\";\n switch (this.operation) {\n case TrigonometryBlockOperations.Cos: {\n operation = \"cos\";\n break;\n }\n case TrigonometryBlockOperations.Sin: {\n operation = \"sin\";\n break;\n }\n case TrigonometryBlockOperations.Abs: {\n operation = \"abs\";\n break;\n }\n case TrigonometryBlockOperations.Exp: {\n operation = \"exp\";\n break;\n }\n case TrigonometryBlockOperations.Exp2: {\n operation = \"exp2\";\n break;\n }\n case TrigonometryBlockOperations.Round: {\n operation = \"round\";\n break;\n }\n case TrigonometryBlockOperations.Floor: {\n operation = \"floor\";\n break;\n }\n case TrigonometryBlockOperations.Ceiling: {\n operation = \"ceil\";\n break;\n }\n case TrigonometryBlockOperations.Sqrt: {\n operation = \"sqrt\";\n break;\n }\n case TrigonometryBlockOperations.Log: {\n operation = \"log\";\n break;\n }\n case TrigonometryBlockOperations.Tan: {\n operation = \"tan\";\n break;\n }\n case TrigonometryBlockOperations.ArcTan: {\n operation = \"atan\";\n break;\n }\n case TrigonometryBlockOperations.ArcCos: {\n operation = \"acos\";\n break;\n }\n case TrigonometryBlockOperations.ArcSin: {\n operation = \"asin\";\n break;\n }\n case TrigonometryBlockOperations.Fract: {\n operation = \"fract\";\n break;\n }\n case TrigonometryBlockOperations.Sign: {\n operation = \"sign\";\n break;\n }\n case TrigonometryBlockOperations.Radians: {\n operation = \"radians\";\n break;\n }\n case TrigonometryBlockOperations.Degrees: {\n operation = \"degrees\";\n break;\n }\n }\n state.compilationString += this._declareOutput(output, state) + ` = ${operation}(${this.input.associatedVariableName});\\n`;\n return this;\n }\n serialize() {\n const serializationObject = super.serialize();\n serializationObject.operation = this.operation;\n return serializationObject;\n }\n _deserialize(serializationObject, scene, rootUrl) {\n super._deserialize(serializationObject, scene, rootUrl);\n this.operation = serializationObject.operation;\n }\n _dumpPropertiesCode() {\n const codeString = super._dumpPropertiesCode() + `${this._codeVariableName}.operation = BABYLON.TrigonometryBlockOperations.${TrigonometryBlockOperations[this.operation]};\\n`;\n return codeString;\n }\n }\n RegisterClass(\"BABYLON.TrigonometryBlock\", TrigonometryBlock);\n\n const onCreatedEffectParameters$2 = { effect: null, subMesh: null };\n /** @internal */\n class NodeMaterialDefines extends MaterialDefines {\n constructor() {\n super();\n this.NORMAL = false;\n this.TANGENT = false;\n this.VERTEXCOLOR_NME = false;\n this.UV1 = false;\n this.UV2 = false;\n this.UV3 = false;\n this.UV4 = false;\n this.UV5 = false;\n this.UV6 = false;\n /** BONES */\n this.NUM_BONE_INFLUENCERS = 0;\n this.BonesPerMesh = 0;\n this.BONETEXTURE = false;\n /** MORPH TARGETS */\n this.MORPHTARGETS = false;\n this.MORPHTARGETS_NORMAL = false;\n this.MORPHTARGETS_TANGENT = false;\n this.MORPHTARGETS_UV = false;\n this.NUM_MORPH_INFLUENCERS = 0;\n this.MORPHTARGETS_TEXTURE = false;\n /** IMAGE PROCESSING */\n this.IMAGEPROCESSING = false;\n this.VIGNETTE = false;\n this.VIGNETTEBLENDMODEMULTIPLY = false;\n this.VIGNETTEBLENDMODEOPAQUE = false;\n this.TONEMAPPING = false;\n this.TONEMAPPING_ACES = false;\n this.CONTRAST = false;\n this.EXPOSURE = false;\n this.COLORCURVES = false;\n this.COLORGRADING = false;\n this.COLORGRADING3D = false;\n this.SAMPLER3DGREENDEPTH = false;\n this.SAMPLER3DBGRMAP = false;\n this.DITHER = false;\n this.IMAGEPROCESSINGPOSTPROCESS = false;\n this.SKIPFINALCOLORCLAMP = false;\n /** MISC. */\n this.BUMPDIRECTUV = 0;\n this.CAMERA_ORTHOGRAPHIC = false;\n this.CAMERA_PERSPECTIVE = false;\n this.rebuild();\n }\n setValue(name, value, markAsUnprocessedIfDirty = false) {\n if (this[name] === undefined) {\n this._keys.push(name);\n }\n if (markAsUnprocessedIfDirty && this[name] !== value) {\n this.markAsUnprocessed();\n }\n this[name] = value;\n }\n }\n /**\n * Class used to create a node based material built by assembling shader blocks\n */\n class NodeMaterial extends PushMaterial {\n /**\n * Checks if a block is a texture block\n * @param block The block to check\n * @returns True if the block is a texture block\n */\n static _BlockIsTextureBlock(block) {\n return (block.getClassName() === \"TextureBlock\" ||\n block.getClassName() === \"ReflectionTextureBaseBlock\" ||\n block.getClassName() === \"RefractionBlock\" ||\n block.getClassName() === \"CurrentScreenBlock\" ||\n block.getClassName() === \"ParticleTextureBlock\" ||\n block.getClassName() === \"ImageSourceBlock\" ||\n block.getClassName() === \"TriPlanarBlock\" ||\n block.getClassName() === \"BiPlanarBlock\");\n }\n /** Get the inspector from bundle or global */\n _getGlobalNodeMaterialEditor() {\n // UMD Global name detection from Webpack Bundle UMD Name.\n if (typeof NODEEDITOR !== \"undefined\") {\n return NODEEDITOR;\n }\n // In case of module let's check the global emitted from the editor entry point.\n if (typeof BABYLON !== \"undefined\" && typeof BABYLON.NodeEditor !== \"undefined\") {\n return BABYLON;\n }\n return undefined;\n }\n /** Gets or sets options to control the node material overall behavior */\n get options() {\n return this._options;\n }\n set options(options) {\n this._options = options;\n }\n /**\n * Gets the image processing configuration used either in this material.\n */\n get imageProcessingConfiguration() {\n return this._imageProcessingConfiguration;\n }\n /**\n * Sets the Default image processing configuration used either in the this material.\n *\n * If sets to null, the scene one is in use.\n */\n set imageProcessingConfiguration(value) {\n this._attachImageProcessingConfiguration(value);\n // Ensure the effect will be rebuilt.\n this._markAllSubMeshesAsTexturesDirty();\n }\n /**\n * Gets or sets the mode property\n */\n get mode() {\n return this._mode;\n }\n set mode(value) {\n this._mode = value;\n }\n /** Gets or sets the unique identifier used to identified the effect associated with the material */\n get buildId() {\n return this._buildId;\n }\n set buildId(value) {\n this._buildId = value;\n }\n /**\n * Create a new node based material\n * @param name defines the material name\n * @param scene defines the hosting scene\n * @param options defines creation option\n */\n constructor(name, scene, options = {}) {\n super(name, scene || EngineStore.LastCreatedScene);\n this._buildId = NodeMaterial._BuildIdGenerator++;\n this._buildWasSuccessful = false;\n this._cachedWorldViewMatrix = new Matrix();\n this._cachedWorldViewProjectionMatrix = new Matrix();\n this._optimizers = new Array();\n this._animationFrame = -1;\n this.BJSNODEMATERIALEDITOR = this._getGlobalNodeMaterialEditor();\n /**\n * Gets or sets data used by visual editor\n * @see https://nme.babylonjs.com\n */\n this.editorData = null;\n /**\n * Gets or sets a boolean indicating that alpha value must be ignored (This will turn alpha blending off even if an alpha value is produced by the material)\n */\n this.ignoreAlpha = false;\n /**\n * Defines the maximum number of lights that can be used in the material\n */\n this.maxSimultaneousLights = 4;\n /**\n * Observable raised when the material is built\n */\n this.onBuildObservable = new Observable$1();\n /**\n * Gets or sets the root nodes of the material vertex shader\n */\n this._vertexOutputNodes = new Array();\n /**\n * Gets or sets the root nodes of the material fragment (pixel) shader\n */\n this._fragmentOutputNodes = new Array();\n /**\n * Gets an array of blocks that needs to be serialized even if they are not yet connected\n */\n this.attachedBlocks = new Array();\n /**\n * Specifies the mode of the node material\n * @internal\n */\n this._mode = NodeMaterialModes.Material;\n /**\n * Gets or sets a boolean indicating that alpha blending must be enabled no matter what alpha value or alpha channel of the FragmentBlock are\n */\n this.forceAlphaBlending = false;\n this._options = Object.assign({ emitComments: false }, options);\n // Setup the default processing configuration to the scene.\n this._attachImageProcessingConfiguration(null);\n }\n /**\n * Gets the current class name of the material e.g. \"NodeMaterial\"\n * @returns the class name\n */\n getClassName() {\n return \"NodeMaterial\";\n }\n /**\n * Attaches a new image processing configuration to the Standard Material.\n * @param configuration\n */\n _attachImageProcessingConfiguration(configuration) {\n if (configuration === this._imageProcessingConfiguration) {\n return;\n }\n // Detaches observer.\n if (this._imageProcessingConfiguration && this._imageProcessingObserver) {\n this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver);\n }\n // Pick the scene configuration if needed.\n if (!configuration) {\n this._imageProcessingConfiguration = this.getScene().imageProcessingConfiguration;\n }\n else {\n this._imageProcessingConfiguration = configuration;\n }\n // Attaches observer.\n if (this._imageProcessingConfiguration) {\n this._imageProcessingObserver = this._imageProcessingConfiguration.onUpdateParameters.add(() => {\n this._markAllSubMeshesAsImageProcessingDirty();\n });\n }\n }\n /**\n * Get a block by its name\n * @param name defines the name of the block to retrieve\n * @returns the required block or null if not found\n */\n getBlockByName(name) {\n let result = null;\n for (const block of this.attachedBlocks) {\n if (block.name === name) {\n if (!result) {\n result = block;\n }\n else {\n Tools.Warn(\"More than one block was found with the name `\" + name + \"`\");\n return result;\n }\n }\n }\n return result;\n }\n /**\n * Get a block by its name\n * @param predicate defines the predicate used to find the good candidate\n * @returns the required block or null if not found\n */\n getBlockByPredicate(predicate) {\n for (const block of this.attachedBlocks) {\n if (predicate(block)) {\n return block;\n }\n }\n return null;\n }\n /**\n * Get an input block by its name\n * @param predicate defines the predicate used to find the good candidate\n * @returns the required input block or null if not found\n */\n getInputBlockByPredicate(predicate) {\n for (const block of this.attachedBlocks) {\n if (block.isInput && predicate(block)) {\n return block;\n }\n }\n return null;\n }\n /**\n * Gets the list of input blocks attached to this material\n * @returns an array of InputBlocks\n */\n getInputBlocks() {\n const blocks = [];\n for (const block of this.attachedBlocks) {\n if (block.isInput) {\n blocks.push(block);\n }\n }\n return blocks;\n }\n /**\n * Adds a new optimizer to the list of optimizers\n * @param optimizer defines the optimizers to add\n * @returns the current material\n */\n registerOptimizer(optimizer) {\n const index = this._optimizers.indexOf(optimizer);\n if (index > -1) {\n return;\n }\n this._optimizers.push(optimizer);\n return this;\n }\n /**\n * Remove an optimizer from the list of optimizers\n * @param optimizer defines the optimizers to remove\n * @returns the current material\n */\n unregisterOptimizer(optimizer) {\n const index = this._optimizers.indexOf(optimizer);\n if (index === -1) {\n return;\n }\n this._optimizers.splice(index, 1);\n return this;\n }\n /**\n * Add a new block to the list of output nodes\n * @param node defines the node to add\n * @returns the current material\n */\n addOutputNode(node) {\n if (node.target === null) {\n throw \"This node is not meant to be an output node. You may want to explicitly set its target value.\";\n }\n if ((node.target & NodeMaterialBlockTargets.Vertex) !== 0) {\n this._addVertexOutputNode(node);\n }\n if ((node.target & NodeMaterialBlockTargets.Fragment) !== 0) {\n this._addFragmentOutputNode(node);\n }\n return this;\n }\n /**\n * Remove a block from the list of root nodes\n * @param node defines the node to remove\n * @returns the current material\n */\n removeOutputNode(node) {\n if (node.target === null) {\n return this;\n }\n if ((node.target & NodeMaterialBlockTargets.Vertex) !== 0) {\n this._removeVertexOutputNode(node);\n }\n if ((node.target & NodeMaterialBlockTargets.Fragment) !== 0) {\n this._removeFragmentOutputNode(node);\n }\n return this;\n }\n _addVertexOutputNode(node) {\n if (this._vertexOutputNodes.indexOf(node) !== -1) {\n return;\n }\n node.target = NodeMaterialBlockTargets.Vertex;\n this._vertexOutputNodes.push(node);\n return this;\n }\n _removeVertexOutputNode(node) {\n const index = this._vertexOutputNodes.indexOf(node);\n if (index === -1) {\n return;\n }\n this._vertexOutputNodes.splice(index, 1);\n return this;\n }\n _addFragmentOutputNode(node) {\n if (this._fragmentOutputNodes.indexOf(node) !== -1) {\n return;\n }\n node.target = NodeMaterialBlockTargets.Fragment;\n this._fragmentOutputNodes.push(node);\n return this;\n }\n _removeFragmentOutputNode(node) {\n const index = this._fragmentOutputNodes.indexOf(node);\n if (index === -1) {\n return;\n }\n this._fragmentOutputNodes.splice(index, 1);\n return this;\n }\n /**\n * Specifies if the material will require alpha blending\n * @returns a boolean specifying if alpha blending is needed\n */\n needAlphaBlending() {\n if (this.ignoreAlpha) {\n return false;\n }\n return this.forceAlphaBlending || this.alpha < 1.0 || (this._sharedData && this._sharedData.hints.needAlphaBlending);\n }\n /**\n * Specifies if this material should be rendered in alpha test mode\n * @returns a boolean specifying if an alpha test is needed.\n */\n needAlphaTesting() {\n return this._sharedData && this._sharedData.hints.needAlphaTesting;\n }\n _processInitializeOnLink(block, state, nodesToProcessForOtherBuildState, autoConfigure = true) {\n if (block.target === NodeMaterialBlockTargets.VertexAndFragment) {\n nodesToProcessForOtherBuildState.push(block);\n }\n else if (state.target === NodeMaterialBlockTargets.Fragment && block.target === NodeMaterialBlockTargets.Vertex && block._preparationId !== this._buildId) {\n nodesToProcessForOtherBuildState.push(block);\n }\n this._initializeBlock(block, state, nodesToProcessForOtherBuildState, autoConfigure);\n }\n _initializeBlock(node, state, nodesToProcessForOtherBuildState, autoConfigure = true) {\n node.initialize(state);\n if (autoConfigure) {\n node.autoConfigure(this);\n }\n node._preparationId = this._buildId;\n if (this.attachedBlocks.indexOf(node) === -1) {\n if (node.isUnique) {\n const className = node.getClassName();\n for (const other of this.attachedBlocks) {\n if (other.getClassName() === className) {\n throw `Cannot have multiple blocks of type ${className} in the same NodeMaterial`;\n }\n }\n }\n this.attachedBlocks.push(node);\n }\n for (const input of node.inputs) {\n input.associatedVariableName = \"\";\n const connectedPoint = input.connectedPoint;\n if (connectedPoint) {\n const block = connectedPoint.ownerBlock;\n if (block !== node) {\n this._processInitializeOnLink(block, state, nodesToProcessForOtherBuildState, autoConfigure);\n }\n }\n }\n // Teleportation\n if (node.isTeleportOut) {\n const teleport = node;\n if (teleport.entryPoint) {\n this._processInitializeOnLink(teleport.entryPoint, state, nodesToProcessForOtherBuildState, autoConfigure);\n }\n }\n for (const output of node.outputs) {\n output.associatedVariableName = \"\";\n }\n }\n _resetDualBlocks(node, id) {\n if (node.target === NodeMaterialBlockTargets.VertexAndFragment) {\n node.buildId = id;\n }\n for (const inputs of node.inputs) {\n const connectedPoint = inputs.connectedPoint;\n if (connectedPoint) {\n const block = connectedPoint.ownerBlock;\n if (block !== node) {\n this._resetDualBlocks(block, id);\n }\n }\n }\n // If this is a teleport out, we need to reset the connected block\n if (node.isTeleportOut) {\n const teleportOut = node;\n if (teleportOut.entryPoint) {\n this._resetDualBlocks(teleportOut.entryPoint, id);\n }\n }\n }\n /**\n * Remove a block from the current node material\n * @param block defines the block to remove\n */\n removeBlock(block) {\n const attachedBlockIndex = this.attachedBlocks.indexOf(block);\n if (attachedBlockIndex > -1) {\n this.attachedBlocks.splice(attachedBlockIndex, 1);\n }\n if (block.isFinalMerger) {\n this.removeOutputNode(block);\n }\n }\n /**\n * Build the material and generates the inner effect\n * @param verbose defines if the build should log activity\n * @param updateBuildId defines if the internal build Id should be updated (default is true)\n * @param autoConfigure defines if the autoConfigure method should be called when initializing blocks (default is true)\n */\n build(verbose = false, updateBuildId = true, autoConfigure = true) {\n this._buildWasSuccessful = false;\n const engine = this.getScene().getEngine();\n const allowEmptyVertexProgram = this._mode === NodeMaterialModes.Particle;\n if (this._vertexOutputNodes.length === 0 && !allowEmptyVertexProgram) {\n throw \"You must define at least one vertexOutputNode\";\n }\n if (this._fragmentOutputNodes.length === 0) {\n throw \"You must define at least one fragmentOutputNode\";\n }\n // Compilation state\n this._vertexCompilationState = new NodeMaterialBuildState();\n this._vertexCompilationState.supportUniformBuffers = engine.supportsUniformBuffers;\n this._vertexCompilationState.target = NodeMaterialBlockTargets.Vertex;\n this._fragmentCompilationState = new NodeMaterialBuildState();\n this._fragmentCompilationState.supportUniformBuffers = engine.supportsUniformBuffers;\n this._fragmentCompilationState.target = NodeMaterialBlockTargets.Fragment;\n // Shared data\n this._sharedData = new NodeMaterialBuildStateSharedData();\n this._sharedData.fragmentOutputNodes = this._fragmentOutputNodes;\n this._vertexCompilationState.sharedData = this._sharedData;\n this._fragmentCompilationState.sharedData = this._sharedData;\n this._sharedData.buildId = this._buildId;\n this._sharedData.emitComments = this._options.emitComments;\n this._sharedData.verbose = verbose;\n this._sharedData.scene = this.getScene();\n this._sharedData.allowEmptyVertexProgram = allowEmptyVertexProgram;\n // Initialize blocks\n const vertexNodes = [];\n const fragmentNodes = [];\n for (const vertexOutputNode of this._vertexOutputNodes) {\n vertexNodes.push(vertexOutputNode);\n this._initializeBlock(vertexOutputNode, this._vertexCompilationState, fragmentNodes, autoConfigure);\n }\n for (const fragmentOutputNode of this._fragmentOutputNodes) {\n fragmentNodes.push(fragmentOutputNode);\n this._initializeBlock(fragmentOutputNode, this._fragmentCompilationState, vertexNodes, autoConfigure);\n }\n // Optimize\n this.optimize();\n // Vertex\n for (const vertexOutputNode of vertexNodes) {\n vertexOutputNode.build(this._vertexCompilationState, vertexNodes);\n }\n // Fragment\n this._fragmentCompilationState.uniforms = this._vertexCompilationState.uniforms.slice(0);\n this._fragmentCompilationState._uniformDeclaration = this._vertexCompilationState._uniformDeclaration;\n this._fragmentCompilationState._constantDeclaration = this._vertexCompilationState._constantDeclaration;\n this._fragmentCompilationState._vertexState = this._vertexCompilationState;\n for (const fragmentOutputNode of fragmentNodes) {\n this._resetDualBlocks(fragmentOutputNode, this._buildId - 1);\n }\n for (const fragmentOutputNode of fragmentNodes) {\n fragmentOutputNode.build(this._fragmentCompilationState, fragmentNodes);\n }\n // Finalize\n this._vertexCompilationState.finalize(this._vertexCompilationState);\n this._fragmentCompilationState.finalize(this._fragmentCompilationState);\n if (updateBuildId) {\n this._buildId = NodeMaterial._BuildIdGenerator++;\n }\n // Errors\n this._sharedData.emitErrors();\n if (verbose) {\n console.log(\"Vertex shader:\");\n console.log(this._vertexCompilationState.compilationString);\n console.log(\"Fragment shader:\");\n console.log(this._fragmentCompilationState.compilationString);\n }\n this._buildWasSuccessful = true;\n this.onBuildObservable.notifyObservers(this);\n // Wipe defines\n const meshes = this.getScene().meshes;\n for (const mesh of meshes) {\n if (!mesh.subMeshes) {\n continue;\n }\n for (const subMesh of mesh.subMeshes) {\n if (subMesh.getMaterial() !== this) {\n continue;\n }\n if (!subMesh.materialDefines) {\n continue;\n }\n const defines = subMesh.materialDefines;\n defines.markAllAsDirty();\n defines.reset();\n }\n }\n }\n /**\n * Runs an otpimization phase to try to improve the shader code\n */\n optimize() {\n for (const optimizer of this._optimizers) {\n optimizer.optimize(this._vertexOutputNodes, this._fragmentOutputNodes);\n }\n }\n _prepareDefinesForAttributes(mesh, defines) {\n const oldNormal = defines[\"NORMAL\"];\n const oldTangent = defines[\"TANGENT\"];\n const oldColor = defines[\"VERTEXCOLOR_NME\"];\n defines[\"NORMAL\"] = mesh.isVerticesDataPresent(VertexBuffer.NormalKind);\n defines[\"TANGENT\"] = mesh.isVerticesDataPresent(VertexBuffer.TangentKind);\n const hasVertexColors = mesh.useVertexColors && mesh.isVerticesDataPresent(VertexBuffer.ColorKind);\n defines[\"VERTEXCOLOR_NME\"] = hasVertexColors;\n let uvChanged = false;\n for (let i = 1; i <= 6; ++i) {\n const oldUV = defines[\"UV\" + i];\n defines[\"UV\" + i] = mesh.isVerticesDataPresent(`uv${i === 1 ? \"\" : i}`);\n uvChanged = uvChanged || defines[\"UV\" + i] !== oldUV;\n }\n if (oldNormal !== defines[\"NORMAL\"] || oldTangent !== defines[\"TANGENT\"] || oldColor !== defines[\"VERTEXCOLOR_NME\"] || uvChanged) {\n defines.markAsAttributesDirty();\n }\n }\n /**\n * Create a post process from the material\n * @param camera The camera to apply the render pass to.\n * @param options The required width/height ratio to downsize to before computing the render pass. (Use 1.0 for full size)\n * @param samplingMode The sampling mode to be used when computing the pass. (default: 0)\n * @param engine The engine which the post process will be applied. (default: current engine)\n * @param reusable If the post process can be reused on the same frame. (default: false)\n * @param textureType Type of textures used when performing the post process. (default: 0)\n * @param textureFormat Format of textures used when performing the post process. (default: TEXTUREFORMAT_RGBA)\n * @returns the post process created\n */\n createPostProcess(camera, options = 1, samplingMode = 1, engine, reusable, textureType = 0, textureFormat = 5) {\n if (this.mode !== NodeMaterialModes.PostProcess) {\n console.log(\"Incompatible material mode\");\n return null;\n }\n return this._createEffectForPostProcess(null, camera, options, samplingMode, engine, reusable, textureType, textureFormat);\n }\n /**\n * Create the post process effect from the material\n * @param postProcess The post process to create the effect for\n */\n createEffectForPostProcess(postProcess) {\n this._createEffectForPostProcess(postProcess);\n }\n _createEffectForPostProcess(postProcess, camera, options = 1, samplingMode = 1, engine, reusable, textureType = 0, textureFormat = 5) {\n let tempName = this.name + this._buildId;\n const defines = new NodeMaterialDefines();\n const dummyMesh = new AbstractMesh(tempName + \"PostProcess\", this.getScene());\n let buildId = this._buildId;\n this._processDefines(dummyMesh, defines);\n Effect.RegisterShader(tempName, this._fragmentCompilationState._builtCompilationString, this._vertexCompilationState._builtCompilationString);\n if (!postProcess) {\n postProcess = new PostProcess(this.name + \"PostProcess\", tempName, this._fragmentCompilationState.uniforms, this._fragmentCompilationState.samplers, options, camera, samplingMode, engine, reusable, defines.toString(), textureType, tempName, { maxSimultaneousLights: this.maxSimultaneousLights }, false, textureFormat);\n }\n else {\n postProcess.updateEffect(defines.toString(), this._fragmentCompilationState.uniforms, this._fragmentCompilationState.samplers, { maxSimultaneousLights: this.maxSimultaneousLights }, undefined, undefined, tempName, tempName);\n }\n postProcess.nodeMaterialSource = this;\n postProcess.onApplyObservable.add((effect) => {\n if (buildId !== this._buildId) {\n delete Effect.ShadersStore[tempName + \"VertexShader\"];\n delete Effect.ShadersStore[tempName + \"PixelShader\"];\n tempName = this.name + this._buildId;\n defines.markAllAsDirty();\n buildId = this._buildId;\n }\n const result = this._processDefines(dummyMesh, defines);\n if (result) {\n Effect.RegisterShader(tempName, this._fragmentCompilationState._builtCompilationString, this._vertexCompilationState._builtCompilationString);\n TimingTools.SetImmediate(() => postProcess.updateEffect(defines.toString(), this._fragmentCompilationState.uniforms, this._fragmentCompilationState.samplers, { maxSimultaneousLights: this.maxSimultaneousLights }, undefined, undefined, tempName, tempName));\n }\n this._checkInternals(effect);\n });\n return postProcess;\n }\n /**\n * Create a new procedural texture based on this node material\n * @param size defines the size of the texture\n * @param scene defines the hosting scene\n * @returns the new procedural texture attached to this node material\n */\n createProceduralTexture(size, scene) {\n if (this.mode !== NodeMaterialModes.ProceduralTexture) {\n console.log(\"Incompatible material mode\");\n return null;\n }\n let tempName = this.name + this._buildId;\n const proceduralTexture = new ProceduralTexture(tempName, size, null, scene);\n const dummyMesh = new AbstractMesh(tempName + \"Procedural\", this.getScene());\n dummyMesh.reservedDataStore = {\n hidden: true,\n };\n const defines = new NodeMaterialDefines();\n const result = this._processDefines(dummyMesh, defines);\n Effect.RegisterShader(tempName, this._fragmentCompilationState._builtCompilationString, this._vertexCompilationState._builtCompilationString);\n let effect = this.getScene().getEngine().createEffect({\n vertexElement: tempName,\n fragmentElement: tempName,\n }, [VertexBuffer.PositionKind], this._fragmentCompilationState.uniforms, this._fragmentCompilationState.samplers, defines.toString(), result === null || result === void 0 ? void 0 : result.fallbacks, undefined);\n proceduralTexture.nodeMaterialSource = this;\n proceduralTexture._setEffect(effect);\n let buildId = this._buildId;\n proceduralTexture.onBeforeGenerationObservable.add(() => {\n if (buildId !== this._buildId) {\n delete Effect.ShadersStore[tempName + \"VertexShader\"];\n delete Effect.ShadersStore[tempName + \"PixelShader\"];\n tempName = this.name + this._buildId;\n defines.markAllAsDirty();\n buildId = this._buildId;\n }\n const result = this._processDefines(dummyMesh, defines);\n if (result) {\n Effect.RegisterShader(tempName, this._fragmentCompilationState._builtCompilationString, this._vertexCompilationState._builtCompilationString);\n TimingTools.SetImmediate(() => {\n effect = this.getScene().getEngine().createEffect({\n vertexElement: tempName,\n fragmentElement: tempName,\n }, [VertexBuffer.PositionKind], this._fragmentCompilationState.uniforms, this._fragmentCompilationState.samplers, defines.toString(), result === null || result === void 0 ? void 0 : result.fallbacks, undefined);\n proceduralTexture._setEffect(effect);\n });\n }\n this._checkInternals(effect);\n });\n return proceduralTexture;\n }\n _createEffectForParticles(particleSystem, blendMode, onCompiled, onError, effect, defines, dummyMesh, particleSystemDefinesJoined = \"\") {\n let tempName = this.name + this._buildId + \"_\" + blendMode;\n if (!defines) {\n defines = new NodeMaterialDefines();\n }\n if (!dummyMesh) {\n dummyMesh = this.getScene().getMeshByName(this.name + \"Particle\");\n if (!dummyMesh) {\n dummyMesh = new AbstractMesh(this.name + \"Particle\", this.getScene());\n dummyMesh.reservedDataStore = {\n hidden: true,\n };\n }\n }\n let buildId = this._buildId;\n const particleSystemDefines = [];\n let join = particleSystemDefinesJoined;\n if (!effect) {\n const result = this._processDefines(dummyMesh, defines);\n Effect.RegisterShader(tempName, this._fragmentCompilationState._builtCompilationString);\n particleSystem.fillDefines(particleSystemDefines, blendMode);\n join = particleSystemDefines.join(\"\\n\");\n effect = this.getScene()\n .getEngine()\n .createEffectForParticles(tempName, this._fragmentCompilationState.uniforms, this._fragmentCompilationState.samplers, defines.toString() + \"\\n\" + join, result === null || result === void 0 ? void 0 : result.fallbacks, onCompiled, onError, particleSystem);\n particleSystem.setCustomEffect(effect, blendMode);\n }\n effect.onBindObservable.add((effect) => {\n if (buildId !== this._buildId) {\n delete Effect.ShadersStore[tempName + \"PixelShader\"];\n tempName = this.name + this._buildId + \"_\" + blendMode;\n defines.markAllAsDirty();\n buildId = this._buildId;\n }\n particleSystemDefines.length = 0;\n particleSystem.fillDefines(particleSystemDefines, blendMode);\n const particleSystemDefinesJoinedCurrent = particleSystemDefines.join(\"\\n\");\n if (particleSystemDefinesJoinedCurrent !== join) {\n defines.markAllAsDirty();\n join = particleSystemDefinesJoinedCurrent;\n }\n const result = this._processDefines(dummyMesh, defines);\n if (result) {\n Effect.RegisterShader(tempName, this._fragmentCompilationState._builtCompilationString);\n effect = this.getScene()\n .getEngine()\n .createEffectForParticles(tempName, this._fragmentCompilationState.uniforms, this._fragmentCompilationState.samplers, defines.toString() + \"\\n\" + join, result === null || result === void 0 ? void 0 : result.fallbacks, onCompiled, onError, particleSystem);\n particleSystem.setCustomEffect(effect, blendMode);\n this._createEffectForParticles(particleSystem, blendMode, onCompiled, onError, effect, defines, dummyMesh, particleSystemDefinesJoined); // add the effect.onBindObservable observer\n return;\n }\n this._checkInternals(effect);\n });\n }\n _checkInternals(effect) {\n // Animated blocks\n if (this._sharedData.animatedInputs) {\n const scene = this.getScene();\n const frameId = scene.getFrameId();\n if (this._animationFrame !== frameId) {\n for (const input of this._sharedData.animatedInputs) {\n input.animate(scene);\n }\n this._animationFrame = frameId;\n }\n }\n // Bindable blocks\n for (const block of this._sharedData.bindableBlocks) {\n block.bind(effect, this);\n }\n // Connection points\n for (const inputBlock of this._sharedData.inputBlocks) {\n inputBlock._transmit(effect, this.getScene(), this);\n }\n }\n /**\n * Create the effect to be used as the custom effect for a particle system\n * @param particleSystem Particle system to create the effect for\n * @param onCompiled defines a function to call when the effect creation is successful\n * @param onError defines a function to call when the effect creation has failed\n */\n createEffectForParticles(particleSystem, onCompiled, onError) {\n if (this.mode !== NodeMaterialModes.Particle) {\n console.log(\"Incompatible material mode\");\n return;\n }\n this._createEffectForParticles(particleSystem, BaseParticleSystem.BLENDMODE_ONEONE, onCompiled, onError);\n this._createEffectForParticles(particleSystem, BaseParticleSystem.BLENDMODE_MULTIPLY, onCompiled, onError);\n }\n /**\n * Use this material as the shadow depth wrapper of a target material\n * @param targetMaterial defines the target material\n */\n createAsShadowDepthWrapper(targetMaterial) {\n if (this.mode !== NodeMaterialModes.Material) {\n console.log(\"Incompatible material mode\");\n return;\n }\n targetMaterial.shadowDepthWrapper = new BABYLON.ShadowDepthWrapper(this, this.getScene());\n }\n _processDefines(mesh, defines, useInstances = false, subMesh) {\n let result = null;\n // Global defines\n const scene = this.getScene();\n if (MaterialHelper.PrepareDefinesForCamera(scene, defines)) {\n defines.markAsMiscDirty();\n }\n // Shared defines\n this._sharedData.blocksWithDefines.forEach((b) => {\n b.initializeDefines(mesh, this, defines, useInstances);\n });\n this._sharedData.blocksWithDefines.forEach((b) => {\n b.prepareDefines(mesh, this, defines, useInstances, subMesh);\n });\n // Need to recompile?\n if (defines.isDirty) {\n const lightDisposed = defines._areLightsDisposed;\n defines.markAsProcessed();\n // Repeatable content generators\n this._vertexCompilationState.compilationString = this._vertexCompilationState._builtCompilationString;\n this._fragmentCompilationState.compilationString = this._fragmentCompilationState._builtCompilationString;\n this._sharedData.repeatableContentBlocks.forEach((b) => {\n b.replaceRepeatableContent(this._vertexCompilationState, this._fragmentCompilationState, mesh, defines);\n });\n // Uniforms\n const uniformBuffers = [];\n this._sharedData.dynamicUniformBlocks.forEach((b) => {\n b.updateUniformsAndSamples(this._vertexCompilationState, this, defines, uniformBuffers);\n });\n const mergedUniforms = this._vertexCompilationState.uniforms;\n this._fragmentCompilationState.uniforms.forEach((u) => {\n const index = mergedUniforms.indexOf(u);\n if (index === -1) {\n mergedUniforms.push(u);\n }\n });\n // Samplers\n const mergedSamplers = this._vertexCompilationState.samplers;\n this._fragmentCompilationState.samplers.forEach((s) => {\n const index = mergedSamplers.indexOf(s);\n if (index === -1) {\n mergedSamplers.push(s);\n }\n });\n const fallbacks = new EffectFallbacks();\n this._sharedData.blocksWithFallbacks.forEach((b) => {\n b.provideFallbacks(mesh, fallbacks);\n });\n result = {\n lightDisposed,\n uniformBuffers,\n mergedUniforms,\n mergedSamplers,\n fallbacks,\n };\n }\n return result;\n }\n /**\n * Get if the submesh is ready to be used and all its information available.\n * Child classes can use it to update shaders\n * @param mesh defines the mesh to check\n * @param subMesh defines which submesh to check\n * @param useInstances specifies that instances should be used\n * @returns a boolean indicating that the submesh is ready or not\n */\n isReadyForSubMesh(mesh, subMesh, useInstances = false) {\n if (!this._buildWasSuccessful) {\n return false;\n }\n const scene = this.getScene();\n if (this._sharedData.animatedInputs) {\n const frameId = scene.getFrameId();\n if (this._animationFrame !== frameId) {\n for (const input of this._sharedData.animatedInputs) {\n input.animate(scene);\n }\n this._animationFrame = frameId;\n }\n }\n if (subMesh.effect && this.isFrozen) {\n if (subMesh.effect._wasPreviouslyReady && subMesh.effect._wasPreviouslyUsingInstances === useInstances) {\n return true;\n }\n }\n if (!subMesh.materialDefines) {\n subMesh.materialDefines = new NodeMaterialDefines();\n }\n const defines = subMesh.materialDefines;\n if (this._isReadyForSubMesh(subMesh)) {\n return true;\n }\n const engine = scene.getEngine();\n this._prepareDefinesForAttributes(mesh, defines);\n // Check if blocks are ready\n if (this._sharedData.blockingBlocks.some((b) => !b.isReady(mesh, this, defines, useInstances))) {\n return false;\n }\n const result = this._processDefines(mesh, defines, useInstances, subMesh);\n if (result) {\n const previousEffect = subMesh.effect;\n // Compilation\n const join = defines.toString();\n let effect = engine.createEffect({\n vertex: \"nodeMaterial\" + this._buildId,\n fragment: \"nodeMaterial\" + this._buildId,\n vertexSource: this._vertexCompilationState.compilationString,\n fragmentSource: this._fragmentCompilationState.compilationString,\n }, {\n attributes: this._vertexCompilationState.attributes,\n uniformsNames: result.mergedUniforms,\n uniformBuffersNames: result.uniformBuffers,\n samplers: result.mergedSamplers,\n defines: join,\n fallbacks: result.fallbacks,\n onCompiled: this.onCompiled,\n onError: this.onError,\n indexParameters: { maxSimultaneousLights: this.maxSimultaneousLights, maxSimultaneousMorphTargets: defines.NUM_MORPH_INFLUENCERS },\n }, engine);\n if (effect) {\n if (this._onEffectCreatedObservable) {\n onCreatedEffectParameters$2.effect = effect;\n onCreatedEffectParameters$2.subMesh = subMesh;\n this._onEffectCreatedObservable.notifyObservers(onCreatedEffectParameters$2);\n }\n // Use previous effect while new one is compiling\n if (this.allowShaderHotSwapping && previousEffect && !effect.isReady()) {\n effect = previousEffect;\n defines.markAsUnprocessed();\n if (result.lightDisposed) {\n // re register in case it takes more than one frame.\n defines._areLightsDisposed = true;\n return false;\n }\n }\n else {\n scene.resetCachedMaterial();\n subMesh.setEffect(effect, defines, this._materialContext);\n }\n }\n }\n if (!subMesh.effect || !subMesh.effect.isReady()) {\n return false;\n }\n defines._renderId = scene.getRenderId();\n subMesh.effect._wasPreviouslyReady = true;\n subMesh.effect._wasPreviouslyUsingInstances = useInstances;\n this._checkScenePerformancePriority();\n return true;\n }\n /**\n * Get a string representing the shaders built by the current node graph\n */\n get compiledShaders() {\n return `// Vertex shader\\n${this._vertexCompilationState.compilationString}\\n\\n// Fragment shader\\n${this._fragmentCompilationState.compilationString}`;\n }\n /**\n * Binds the world matrix to the material\n * @param world defines the world transformation matrix\n */\n bindOnlyWorldMatrix(world) {\n const scene = this.getScene();\n if (!this._activeEffect) {\n return;\n }\n const hints = this._sharedData.hints;\n if (hints.needWorldViewMatrix) {\n world.multiplyToRef(scene.getViewMatrix(), this._cachedWorldViewMatrix);\n }\n if (hints.needWorldViewProjectionMatrix) {\n world.multiplyToRef(scene.getTransformMatrix(), this._cachedWorldViewProjectionMatrix);\n }\n // Connection points\n for (const inputBlock of this._sharedData.inputBlocks) {\n inputBlock._transmitWorld(this._activeEffect, world, this._cachedWorldViewMatrix, this._cachedWorldViewProjectionMatrix);\n }\n }\n /**\n * Binds the submesh to this material by preparing the effect and shader to draw\n * @param world defines the world transformation matrix\n * @param mesh defines the mesh containing the submesh\n * @param subMesh defines the submesh to bind the material to\n */\n bindForSubMesh(world, mesh, subMesh) {\n const scene = this.getScene();\n const effect = subMesh.effect;\n if (!effect) {\n return;\n }\n this._activeEffect = effect;\n // Matrices\n this.bindOnlyWorldMatrix(world);\n const mustRebind = this._mustRebind(scene, effect, mesh.visibility);\n const sharedData = this._sharedData;\n if (mustRebind) {\n // Bindable blocks\n for (const block of sharedData.bindableBlocks) {\n block.bind(effect, this, mesh, subMesh);\n }\n for (const block of sharedData.forcedBindableBlocks) {\n block.bind(effect, this, mesh, subMesh);\n }\n // Connection points\n for (const inputBlock of sharedData.inputBlocks) {\n inputBlock._transmit(effect, scene, this);\n }\n }\n else if (!this.isFrozen) {\n for (const block of sharedData.forcedBindableBlocks) {\n block.bind(effect, this, mesh, subMesh);\n }\n }\n this._afterBind(mesh, this._activeEffect);\n }\n /**\n * Gets the active textures from the material\n * @returns an array of textures\n */\n getActiveTextures() {\n const activeTextures = super.getActiveTextures();\n if (this._sharedData) {\n activeTextures.push(...this._sharedData.textureBlocks.filter((tb) => tb.texture).map((tb) => tb.texture));\n }\n return activeTextures;\n }\n /**\n * Gets the list of texture blocks\n * Note that this method will only return blocks that are reachable from the final block(s) and only after the material has been built!\n * @returns an array of texture blocks\n */\n getTextureBlocks() {\n if (!this._sharedData) {\n return [];\n }\n return this._sharedData.textureBlocks;\n }\n /**\n * Gets the list of all texture blocks\n * Note that this method will scan all attachedBlocks and return blocks that are texture blocks\n * @returns\n */\n getAllTextureBlocks() {\n const textureBlocks = [];\n for (const block of this.attachedBlocks) {\n if (NodeMaterial._BlockIsTextureBlock(block)) {\n textureBlocks.push(block);\n }\n }\n return textureBlocks;\n }\n /**\n * Specifies if the material uses a texture\n * @param texture defines the texture to check against the material\n * @returns a boolean specifying if the material uses the texture\n */\n hasTexture(texture) {\n if (super.hasTexture(texture)) {\n return true;\n }\n if (!this._sharedData) {\n return false;\n }\n for (const t of this._sharedData.textureBlocks) {\n if (t.texture === texture) {\n return true;\n }\n }\n return false;\n }\n /**\n * Disposes the material\n * @param forceDisposeEffect specifies if effects should be forcefully disposed\n * @param forceDisposeTextures specifies if textures should be forcefully disposed\n * @param notBoundToMesh specifies if the material that is being disposed is known to be not bound to any mesh\n */\n dispose(forceDisposeEffect, forceDisposeTextures, notBoundToMesh) {\n if (forceDisposeTextures) {\n for (const texture of this.getTextureBlocks()\n .filter((tb) => tb.texture)\n .map((tb) => tb.texture)) {\n texture.dispose();\n }\n }\n for (const block of this.attachedBlocks) {\n block.dispose();\n }\n this.attachedBlocks.length = 0;\n this._sharedData = null;\n this._vertexCompilationState = null;\n this._fragmentCompilationState = null;\n this.onBuildObservable.clear();\n if (this._imageProcessingObserver) {\n this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver);\n this._imageProcessingObserver = null;\n }\n super.dispose(forceDisposeEffect, forceDisposeTextures, notBoundToMesh);\n }\n /** Creates the node editor window. */\n _createNodeEditor(additionalConfig) {\n const nodeEditorConfig = Object.assign({ nodeMaterial: this }, additionalConfig);\n this.BJSNODEMATERIALEDITOR.NodeEditor.Show(nodeEditorConfig);\n }\n /**\n * Launch the node material editor\n * @param config Define the configuration of the editor\n * @returns a promise fulfilled when the node editor is visible\n */\n edit(config) {\n return new Promise((resolve) => {\n this.BJSNODEMATERIALEDITOR = this.BJSNODEMATERIALEDITOR || this._getGlobalNodeMaterialEditor();\n if (typeof this.BJSNODEMATERIALEDITOR == \"undefined\") {\n const editorUrl = config && config.editorURL ? config.editorURL : NodeMaterial.EditorURL;\n // Load editor and add it to the DOM\n Tools.LoadScript(editorUrl, () => {\n this.BJSNODEMATERIALEDITOR = this.BJSNODEMATERIALEDITOR || this._getGlobalNodeMaterialEditor();\n this._createNodeEditor(config === null || config === void 0 ? void 0 : config.nodeEditorConfig);\n resolve();\n });\n }\n else {\n // Otherwise creates the editor\n this._createNodeEditor(config === null || config === void 0 ? void 0 : config.nodeEditorConfig);\n resolve();\n }\n });\n }\n /**\n * Clear the current material\n */\n clear() {\n this._vertexOutputNodes.length = 0;\n this._fragmentOutputNodes.length = 0;\n this.attachedBlocks.length = 0;\n }\n /**\n * Clear the current material and set it to a default state\n */\n setToDefault() {\n this.clear();\n this.editorData = null;\n const positionInput = new InputBlock(\"Position\");\n positionInput.setAsAttribute(\"position\");\n const worldInput = new InputBlock(\"World\");\n worldInput.setAsSystemValue(NodeMaterialSystemValues.World);\n const worldPos = new TransformBlock(\"WorldPos\");\n positionInput.connectTo(worldPos);\n worldInput.connectTo(worldPos);\n const viewProjectionInput = new InputBlock(\"ViewProjection\");\n viewProjectionInput.setAsSystemValue(NodeMaterialSystemValues.ViewProjection);\n const worldPosdMultipliedByViewProjection = new TransformBlock(\"WorldPos * ViewProjectionTransform\");\n worldPos.connectTo(worldPosdMultipliedByViewProjection);\n viewProjectionInput.connectTo(worldPosdMultipliedByViewProjection);\n const vertexOutput = new VertexOutputBlock(\"VertexOutput\");\n worldPosdMultipliedByViewProjection.connectTo(vertexOutput);\n // Pixel\n const pixelColor = new InputBlock(\"color\");\n pixelColor.value = new Color4(0.8, 0.8, 0.8, 1);\n const fragmentOutput = new FragmentOutputBlock(\"FragmentOutput\");\n pixelColor.connectTo(fragmentOutput);\n // Add to nodes\n this.addOutputNode(vertexOutput);\n this.addOutputNode(fragmentOutput);\n this._mode = NodeMaterialModes.Material;\n }\n /**\n * Clear the current material and set it to a default state for post process\n */\n setToDefaultPostProcess() {\n this.clear();\n this.editorData = null;\n const position = new InputBlock(\"Position\");\n position.setAsAttribute(\"position2d\");\n const const1 = new InputBlock(\"Constant1\");\n const1.isConstant = true;\n const1.value = 1;\n const vmerger = new VectorMergerBlock(\"Position3D\");\n position.connectTo(vmerger);\n const1.connectTo(vmerger, { input: \"w\" });\n const vertexOutput = new VertexOutputBlock(\"VertexOutput\");\n vmerger.connectTo(vertexOutput);\n // Pixel\n const scale = new InputBlock(\"Scale\");\n scale.visibleInInspector = true;\n scale.value = new Vector2(1, 1);\n const uv0 = new RemapBlock(\"uv0\");\n position.connectTo(uv0);\n const uv = new MultiplyBlock(\"UV scale\");\n uv0.connectTo(uv);\n scale.connectTo(uv);\n const currentScreen = new CurrentScreenBlock(\"CurrentScreen\");\n uv.connectTo(currentScreen);\n currentScreen.texture = new Texture(\"https://assets.babylonjs.com/nme/currentScreenPostProcess.png\", this.getScene());\n const fragmentOutput = new FragmentOutputBlock(\"FragmentOutput\");\n currentScreen.connectTo(fragmentOutput, { output: \"rgba\" });\n // Add to nodes\n this.addOutputNode(vertexOutput);\n this.addOutputNode(fragmentOutput);\n this._mode = NodeMaterialModes.PostProcess;\n }\n /**\n * Clear the current material and set it to a default state for procedural texture\n */\n setToDefaultProceduralTexture() {\n this.clear();\n this.editorData = null;\n const position = new InputBlock(\"Position\");\n position.setAsAttribute(\"position2d\");\n const const1 = new InputBlock(\"Constant1\");\n const1.isConstant = true;\n const1.value = 1;\n const vmerger = new VectorMergerBlock(\"Position3D\");\n position.connectTo(vmerger);\n const1.connectTo(vmerger, { input: \"w\" });\n const vertexOutput = new VertexOutputBlock(\"VertexOutput\");\n vmerger.connectTo(vertexOutput);\n // Pixel\n const time = new InputBlock(\"Time\");\n time.value = 0;\n time.min = 0;\n time.max = 0;\n time.isBoolean = false;\n time.matrixMode = 0;\n time.animationType = AnimatedInputBlockTypes.Time;\n time.isConstant = false;\n const color = new InputBlock(\"Color3\");\n color.value = new Color3(1, 1, 1);\n color.isConstant = false;\n const fragmentOutput = new FragmentOutputBlock(\"FragmentOutput\");\n const vectorMerger = new VectorMergerBlock(\"VectorMerger\");\n vectorMerger.visibleInInspector = false;\n const cos = new TrigonometryBlock(\"Cos\");\n cos.operation = TrigonometryBlockOperations.Cos;\n position.connectTo(vectorMerger);\n time.output.connectTo(cos.input);\n cos.output.connectTo(vectorMerger.z);\n vectorMerger.xyzOut.connectTo(fragmentOutput.rgb);\n // Add to nodes\n this.addOutputNode(vertexOutput);\n this.addOutputNode(fragmentOutput);\n this._mode = NodeMaterialModes.ProceduralTexture;\n }\n /**\n * Clear the current material and set it to a default state for particle\n */\n setToDefaultParticle() {\n this.clear();\n this.editorData = null;\n // Pixel\n const uv = new InputBlock(\"uv\");\n uv.setAsAttribute(\"particle_uv\");\n const texture = new ParticleTextureBlock(\"ParticleTexture\");\n uv.connectTo(texture);\n const color = new InputBlock(\"Color\");\n color.setAsAttribute(\"particle_color\");\n const multiply = new MultiplyBlock(\"Texture * Color\");\n texture.connectTo(multiply);\n color.connectTo(multiply);\n const rampGradient = new ParticleRampGradientBlock(\"ParticleRampGradient\");\n multiply.connectTo(rampGradient);\n const cSplitter = new ColorSplitterBlock(\"ColorSplitter\");\n color.connectTo(cSplitter);\n const blendMultiply = new ParticleBlendMultiplyBlock(\"ParticleBlendMultiply\");\n rampGradient.connectTo(blendMultiply);\n texture.connectTo(blendMultiply, { output: \"a\" });\n cSplitter.connectTo(blendMultiply, { output: \"a\" });\n const fragmentOutput = new FragmentOutputBlock(\"FragmentOutput\");\n blendMultiply.connectTo(fragmentOutput);\n // Add to nodes\n this.addOutputNode(fragmentOutput);\n this._mode = NodeMaterialModes.Particle;\n }\n /**\n * Loads the current Node Material from a url pointing to a file save by the Node Material Editor\n * @deprecated Please use NodeMaterial.ParseFromFileAsync instead\n * @param url defines the url to load from\n * @param rootUrl defines the root URL for nested url in the node material\n * @returns a promise that will fulfil when the material is fully loaded\n */\n async loadAsync(url, rootUrl = \"\") {\n return NodeMaterial.ParseFromFileAsync(\"\", url, this.getScene(), rootUrl, true, this);\n }\n _gatherBlocks(rootNode, list) {\n if (list.indexOf(rootNode) !== -1) {\n return;\n }\n list.push(rootNode);\n for (const input of rootNode.inputs) {\n const connectedPoint = input.connectedPoint;\n if (connectedPoint) {\n const block = connectedPoint.ownerBlock;\n if (block !== rootNode) {\n this._gatherBlocks(block, list);\n }\n }\n }\n // Teleportation\n if (rootNode.isTeleportOut) {\n const block = rootNode;\n if (block.entryPoint) {\n this._gatherBlocks(block.entryPoint, list);\n }\n }\n }\n /**\n * Generate a string containing the code declaration required to create an equivalent of this material\n * @returns a string\n */\n generateCode() {\n let alreadyDumped = [];\n const vertexBlocks = [];\n const uniqueNames = [\"const\", \"var\", \"let\"];\n // Gets active blocks\n for (const outputNode of this._vertexOutputNodes) {\n this._gatherBlocks(outputNode, vertexBlocks);\n }\n const fragmentBlocks = [];\n for (const outputNode of this._fragmentOutputNodes) {\n this._gatherBlocks(outputNode, fragmentBlocks);\n }\n // Generate vertex shader\n let codeString = `var nodeMaterial = new BABYLON.NodeMaterial(\"${this.name || \"node material\"}\");\\n`;\n codeString += `nodeMaterial.mode = BABYLON.NodeMaterialModes.${NodeMaterialModes[this.mode]};\\n`;\n for (const node of vertexBlocks) {\n if (node.isInput && alreadyDumped.indexOf(node) === -1) {\n codeString += node._dumpCode(uniqueNames, alreadyDumped);\n }\n }\n // Generate fragment shader\n for (const node of fragmentBlocks) {\n if (node.isInput && alreadyDumped.indexOf(node) === -1) {\n codeString += node._dumpCode(uniqueNames, alreadyDumped);\n }\n }\n // Connections\n alreadyDumped = [];\n codeString += \"\\n// Connections\\n\";\n for (const node of this._vertexOutputNodes) {\n codeString += node._dumpCodeForOutputConnections(alreadyDumped);\n }\n for (const node of this._fragmentOutputNodes) {\n codeString += node._dumpCodeForOutputConnections(alreadyDumped);\n }\n // Output nodes\n codeString += \"\\n// Output nodes\\n\";\n for (const node of this._vertexOutputNodes) {\n codeString += `nodeMaterial.addOutputNode(${node._codeVariableName});\\n`;\n }\n for (const node of this._fragmentOutputNodes) {\n codeString += `nodeMaterial.addOutputNode(${node._codeVariableName});\\n`;\n }\n codeString += `nodeMaterial.build();\\n`;\n return codeString;\n }\n /**\n * Serializes this material in a JSON representation\n * @param selectedBlocks\n * @returns the serialized material object\n */\n serialize(selectedBlocks) {\n const serializationObject = selectedBlocks ? {} : SerializationHelper.Serialize(this);\n serializationObject.editorData = JSON.parse(JSON.stringify(this.editorData)); // Copy\n let blocks = [];\n if (selectedBlocks) {\n blocks = selectedBlocks;\n }\n else {\n serializationObject.customType = \"BABYLON.NodeMaterial\";\n serializationObject.outputNodes = [];\n // Outputs\n for (const outputNode of this._vertexOutputNodes) {\n this._gatherBlocks(outputNode, blocks);\n serializationObject.outputNodes.push(outputNode.uniqueId);\n }\n for (const outputNode of this._fragmentOutputNodes) {\n this._gatherBlocks(outputNode, blocks);\n if (serializationObject.outputNodes.indexOf(outputNode.uniqueId) === -1) {\n serializationObject.outputNodes.push(outputNode.uniqueId);\n }\n }\n }\n // Blocks\n serializationObject.blocks = [];\n for (const block of blocks) {\n serializationObject.blocks.push(block.serialize());\n }\n if (!selectedBlocks) {\n for (const block of this.attachedBlocks) {\n if (blocks.indexOf(block) !== -1) {\n continue;\n }\n serializationObject.blocks.push(block.serialize());\n }\n }\n return serializationObject;\n }\n _restoreConnections(block, source, map) {\n for (const outputPoint of block.outputs) {\n for (const candidate of source.blocks) {\n const target = map[candidate.id];\n if (!target) {\n continue;\n }\n for (const input of candidate.inputs) {\n if (map[input.targetBlockId] === block && input.targetConnectionName === outputPoint.name) {\n const inputPoint = target.getInputByName(input.inputName);\n if (!inputPoint || inputPoint.isConnected) {\n continue;\n }\n outputPoint.connectTo(inputPoint, true);\n this._restoreConnections(target, source, map);\n continue;\n }\n }\n }\n }\n }\n /**\n * Clear the current graph and load a new one from a serialization object\n * @param source defines the JSON representation of the material\n * @param rootUrl defines the root URL to use to load textures and relative dependencies\n * @param merge defines whether or not the source must be merged or replace the current content\n */\n parseSerializedObject(source, rootUrl = \"\", merge = false) {\n var _a;\n if (!merge) {\n this.clear();\n }\n const map = {};\n // Create blocks\n for (const parsedBlock of source.blocks) {\n const blockType = GetClass(parsedBlock.customType);\n if (blockType) {\n const block = new blockType();\n block._deserialize(parsedBlock, this.getScene(), rootUrl);\n map[parsedBlock.id] = block;\n this.attachedBlocks.push(block);\n }\n }\n // Reconnect teleportation\n for (const block of this.attachedBlocks) {\n if (block.isTeleportOut) {\n const teleportOut = block;\n const id = teleportOut._tempEntryPointUniqueId;\n if (id) {\n const source = map[id];\n source.attachToEndpoint(teleportOut);\n }\n }\n }\n // Connections - Starts with input blocks only (except if in \"merge\" mode where we scan all blocks)\n for (let blockIndex = 0; blockIndex < source.blocks.length; blockIndex++) {\n const parsedBlock = source.blocks[blockIndex];\n const block = map[parsedBlock.id];\n if (!block) {\n continue;\n }\n if (block.inputs.length && !merge) {\n continue;\n }\n this._restoreConnections(block, source, map);\n }\n // Outputs\n if (source.outputNodes) {\n for (const outputNodeId of source.outputNodes) {\n this.addOutputNode(map[outputNodeId]);\n }\n }\n // UI related info\n if (source.locations || (source.editorData && source.editorData.locations)) {\n const locations = source.locations || source.editorData.locations;\n for (const location of locations) {\n if (map[location.blockId]) {\n location.blockId = map[location.blockId].uniqueId;\n }\n }\n if (merge && this.editorData && this.editorData.locations) {\n locations.concat(this.editorData.locations);\n }\n if (source.locations) {\n this.editorData = {\n locations: locations,\n };\n }\n else {\n this.editorData = source.editorData;\n this.editorData.locations = locations;\n }\n const blockMap = [];\n for (const key in map) {\n blockMap[key] = map[key].uniqueId;\n }\n this.editorData.map = blockMap;\n }\n this.comment = source.comment;\n if (source.forceAlphaBlending !== undefined) {\n this.forceAlphaBlending = source.forceAlphaBlending;\n }\n if (!merge) {\n this._mode = (_a = source.mode) !== null && _a !== void 0 ? _a : NodeMaterialModes.Material;\n }\n }\n /**\n * Clear the current graph and load a new one from a serialization object\n * @param source defines the JSON representation of the material\n * @param rootUrl defines the root URL to use to load textures and relative dependencies\n * @param merge defines whether or not the source must be merged or replace the current content\n * @deprecated Please use the parseSerializedObject method instead\n */\n loadFromSerialization(source, rootUrl = \"\", merge = false) {\n this.parseSerializedObject(source, rootUrl, merge);\n }\n /**\n * Makes a duplicate of the current material.\n * @param name defines the name to use for the new material\n * @param shareEffect defines if the clone material should share the same effect (default is false)\n */\n clone(name, shareEffect = false) {\n const serializationObject = this.serialize();\n const clone = SerializationHelper.Clone(() => new NodeMaterial(name, this.getScene(), this.options), this);\n clone.id = name;\n clone.name = name;\n clone.parseSerializedObject(serializationObject);\n clone._buildId = this._buildId;\n clone.build(false, !shareEffect);\n return clone;\n }\n /**\n * Creates a node material from parsed material data\n * @param source defines the JSON representation of the material\n * @param scene defines the hosting scene\n * @param rootUrl defines the root URL to use to load textures and relative dependencies\n * @returns a new node material\n */\n static Parse(source, scene, rootUrl = \"\") {\n const nodeMaterial = SerializationHelper.Parse(() => new NodeMaterial(source.name, scene), source, scene, rootUrl);\n nodeMaterial.parseSerializedObject(source, rootUrl);\n nodeMaterial.build();\n return nodeMaterial;\n }\n /**\n * Creates a node material from a snippet saved in a remote file\n * @param name defines the name of the material to create\n * @param url defines the url to load from\n * @param scene defines the hosting scene\n * @param rootUrl defines the root URL for nested url in the node material\n * @param skipBuild defines whether to build the node material\n * @param targetMaterial defines a material to use instead of creating a new one\n * @returns a promise that will resolve to the new node material\n */\n static async ParseFromFileAsync(name, url, scene, rootUrl = \"\", skipBuild = false, targetMaterial) {\n const material = targetMaterial !== null && targetMaterial !== void 0 ? targetMaterial : new NodeMaterial(name, scene);\n const data = await scene._loadFileAsync(url);\n const serializationObject = JSON.parse(data);\n material.parseSerializedObject(serializationObject, rootUrl);\n if (!skipBuild) {\n material.build();\n }\n return material;\n }\n /**\n * Creates a node material from a snippet saved by the node material editor\n * @param snippetId defines the snippet to load\n * @param scene defines the hosting scene\n * @param rootUrl defines the root URL to use to load textures and relative dependencies\n * @param nodeMaterial defines a node material to update (instead of creating a new one)\n * @param skipBuild defines whether to build the node material\n * @returns a promise that will resolve to the new node material\n */\n static ParseFromSnippetAsync(snippetId, scene = EngineStore.LastCreatedScene, rootUrl = \"\", nodeMaterial, skipBuild = false) {\n if (snippetId === \"_BLANK\") {\n return Promise.resolve(NodeMaterial.CreateDefault(\"blank\", scene));\n }\n return new Promise((resolve, reject) => {\n const request = new WebRequest();\n request.addEventListener(\"readystatechange\", () => {\n if (request.readyState == 4) {\n if (request.status == 200) {\n const snippet = JSON.parse(JSON.parse(request.responseText).jsonPayload);\n const serializationObject = JSON.parse(snippet.nodeMaterial);\n if (!nodeMaterial) {\n nodeMaterial = SerializationHelper.Parse(() => new NodeMaterial(snippetId, scene), serializationObject, scene, rootUrl);\n nodeMaterial.uniqueId = scene.getUniqueId();\n }\n nodeMaterial.parseSerializedObject(serializationObject);\n nodeMaterial.snippetId = snippetId;\n try {\n if (!skipBuild) {\n nodeMaterial.build();\n }\n resolve(nodeMaterial);\n }\n catch (err) {\n reject(err);\n }\n }\n else {\n reject(\"Unable to load the snippet \" + snippetId);\n }\n }\n });\n request.open(\"GET\", this.SnippetUrl + \"/\" + snippetId.replace(/#/g, \"/\"));\n request.send();\n });\n }\n /**\n * Creates a new node material set to default basic configuration\n * @param name defines the name of the material\n * @param scene defines the hosting scene\n * @returns a new NodeMaterial\n */\n static CreateDefault(name, scene) {\n const newMaterial = new NodeMaterial(name, scene);\n newMaterial.setToDefault();\n newMaterial.build();\n return newMaterial;\n }\n }\n NodeMaterial._BuildIdGenerator = 0;\n /** Define the Url to load node editor script */\n NodeMaterial.EditorURL = `https://unpkg.com/babylonjs-node-editor@${Engine.Version}/babylon.nodeEditor.js`;\n /** Define the Url to load snippets */\n NodeMaterial.SnippetUrl = `https://snippet.babylonjs.com`;\n /** Gets or sets a boolean indicating that node materials should not deserialize textures from json / snippet content */\n NodeMaterial.IgnoreTexturesAtLoadTime = false;\n __decorate$1([\n serialize()\n ], NodeMaterial.prototype, \"ignoreAlpha\", void 0);\n __decorate$1([\n serialize()\n ], NodeMaterial.prototype, \"maxSimultaneousLights\", void 0);\n __decorate$1([\n serialize(\"mode\")\n ], NodeMaterial.prototype, \"_mode\", void 0);\n __decorate$1([\n serialize(\"comment\")\n ], NodeMaterial.prototype, \"comment\", void 0);\n __decorate$1([\n serialize()\n ], NodeMaterial.prototype, \"forceAlphaBlending\", void 0);\n RegisterClass(\"BABYLON.NodeMaterial\", NodeMaterial);\n\n /**\n * @internal\n */\n SubMesh.prototype._projectOnTrianglesToRef = function (vector, positions, indices, step, checkStopper, ref) {\n // Triangles test\n const proj = TmpVectors.Vector3[0];\n const tmp = TmpVectors.Vector3[1];\n let distance = +Infinity;\n for (let index = this.indexStart; index < this.indexStart + this.indexCount - (3 - step); index += step) {\n const indexA = indices[index];\n const indexB = indices[index + 1];\n const indexC = indices[index + 2];\n if (checkStopper && indexC === 0xffffffff) {\n index += 2;\n continue;\n }\n const p0 = positions[indexA];\n const p1 = positions[indexB];\n const p2 = positions[indexC];\n // stay defensive and don't check against undefined positions.\n if (!p0 || !p1 || !p2) {\n continue;\n }\n const tmpDist = Vector3.ProjectOnTriangleToRef(vector, p0, p1, p2, tmp);\n if (tmpDist < distance) {\n proj.copyFrom(tmp);\n distance = tmpDist;\n }\n }\n ref.copyFrom(proj);\n return distance;\n };\n /**\n * @internal\n */\n SubMesh.prototype._projectOnUnIndexedTrianglesToRef = function (vector, positions, indices, ref) {\n // Triangles test\n const proj = TmpVectors.Vector3[0];\n const tmp = TmpVectors.Vector3[1];\n let distance = +Infinity;\n for (let index = this.verticesStart; index < this.verticesStart + this.verticesCount; index += 3) {\n const p0 = positions[index];\n const p1 = positions[index + 1];\n const p2 = positions[index + 2];\n const tmpDist = Vector3.ProjectOnTriangleToRef(vector, p0, p1, p2, tmp);\n if (tmpDist < distance) {\n proj.copyFrom(tmp);\n distance = tmpDist;\n }\n }\n ref.copyFrom(proj);\n return distance;\n };\n SubMesh.prototype.projectToRef = function (vector, positions, indices, ref) {\n const material = this.getMaterial();\n if (!material) {\n return -1;\n }\n let step = 3;\n let checkStopper = false;\n switch (material.fillMode) {\n case 3:\n case 5:\n case 6:\n case 8:\n return -1;\n case 7:\n step = 1;\n checkStopper = true;\n break;\n }\n // LineMesh first as it's also a Mesh...\n if (material.fillMode === 4) {\n return -1;\n }\n else {\n // Check if mesh is unindexed\n if (!indices.length && this._mesh._unIndexed) {\n return this._projectOnUnIndexedTrianglesToRef(vector, positions, indices, ref);\n }\n return this._projectOnTrianglesToRef(vector, positions, indices, step, checkStopper, ref);\n }\n };\n\n // Tracks the interaction animation state when using a motion controller with a near interaction orb\n var ControllerOrbAnimationState;\n (function (ControllerOrbAnimationState) {\n /**\n * Orb is invisible\n */\n ControllerOrbAnimationState[ControllerOrbAnimationState[\"DEHYDRATED\"] = 0] = \"DEHYDRATED\";\n /**\n * Orb is visible and inside the hover range\n */\n ControllerOrbAnimationState[ControllerOrbAnimationState[\"HOVER\"] = 1] = \"HOVER\";\n /**\n * Orb is visible and touching a near interaction target\n */\n ControllerOrbAnimationState[ControllerOrbAnimationState[\"TOUCH\"] = 2] = \"TOUCH\";\n })(ControllerOrbAnimationState || (ControllerOrbAnimationState = {}));\n /**\n * Where should the near interaction mesh be attached to when using a motion controller for near interaction\n */\n var WebXRNearControllerMode;\n (function (WebXRNearControllerMode) {\n /**\n * Motion controllers will not support near interaction\n */\n WebXRNearControllerMode[WebXRNearControllerMode[\"DISABLED\"] = 0] = \"DISABLED\";\n /**\n * The interaction point for motion controllers will be inside of them\n */\n WebXRNearControllerMode[WebXRNearControllerMode[\"CENTERED_ON_CONTROLLER\"] = 1] = \"CENTERED_ON_CONTROLLER\";\n /**\n * The interaction point for motion controllers will be in front of the controller\n */\n WebXRNearControllerMode[WebXRNearControllerMode[\"CENTERED_IN_FRONT\"] = 2] = \"CENTERED_IN_FRONT\";\n })(WebXRNearControllerMode || (WebXRNearControllerMode = {}));\n /**\n * A module that will enable near interaction near interaction for hands and motion controllers of XR Input Sources\n */\n class WebXRNearInteraction extends WebXRAbstractFeature {\n /**\n * constructs a new background remover module\n * @param _xrSessionManager the session manager for this module\n * @param _options read-only options to be used in this module\n */\n constructor(_xrSessionManager, _options) {\n super(_xrSessionManager);\n this._options = _options;\n this._tmpRay = new Ray(new Vector3(), new Vector3());\n this._attachController = (xrController) => {\n if (this._controllers[xrController.uniqueId]) {\n // already attached\n return;\n }\n // get two new meshes\n const { touchCollisionMesh, touchCollisionMeshFunction, hydrateCollisionMeshFunction } = this._generateNewTouchPointMesh();\n const selectionMesh = this._generateVisualCue();\n this._controllers[xrController.uniqueId] = {\n xrController,\n meshUnderPointer: null,\n nearInteractionTargetMesh: null,\n pick: null,\n stalePick: null,\n touchCollisionMesh,\n touchCollisionMeshFunction: touchCollisionMeshFunction,\n hydrateCollisionMeshFunction: hydrateCollisionMeshFunction,\n currentAnimationState: ControllerOrbAnimationState.DEHYDRATED,\n grabRay: new Ray(new Vector3(), new Vector3()),\n hoverInteraction: false,\n nearInteraction: false,\n grabInteraction: false,\n id: WebXRNearInteraction._IdCounter++,\n pickedPointVisualCue: selectionMesh,\n };\n if (this._attachedController) {\n if (!this._options.enableNearInteractionOnAllControllers &&\n this._options.preferredHandedness &&\n xrController.inputSource.handedness === this._options.preferredHandedness) {\n this._attachedController = xrController.uniqueId;\n }\n }\n else {\n if (!this._options.enableNearInteractionOnAllControllers) {\n this._attachedController = xrController.uniqueId;\n }\n }\n switch (xrController.inputSource.targetRayMode) {\n case \"tracked-pointer\":\n return this._attachNearInteractionMode(xrController);\n case \"gaze\":\n return null;\n case \"screen\":\n return null;\n }\n };\n this._controllers = {};\n this._farInteractionFeature = null;\n /**\n * default color of the selection ring\n */\n this.selectionMeshDefaultColor = new Color3(0.8, 0.8, 0.8);\n /**\n * This color will be applied to the selection ring when selection is triggered\n */\n this.selectionMeshPickedColor = new Color3(0.3, 0.3, 1.0);\n this._hoverRadius = 0.1;\n this._pickRadius = 0.02;\n this._controllerPickRadius = 0.03; // The radius is slightly larger here to make it easier to manipulate since it's not tied to the hand position\n this._nearGrabLengthScale = 5;\n this._scene = this._xrSessionManager.scene;\n if (this._options.nearInteractionControllerMode === undefined) {\n this._options.nearInteractionControllerMode = WebXRNearControllerMode.CENTERED_IN_FRONT;\n }\n if (this._options.farInteractionFeature) {\n this._farInteractionFeature = this._options.farInteractionFeature;\n }\n }\n /**\n * Attach this feature\n * Will usually be called by the features manager\n *\n * @returns true if successful.\n */\n attach() {\n if (!super.attach()) {\n return false;\n }\n this._options.xrInput.controllers.forEach(this._attachController);\n this._addNewAttachObserver(this._options.xrInput.onControllerAddedObservable, this._attachController);\n this._addNewAttachObserver(this._options.xrInput.onControllerRemovedObservable, (controller) => {\n // REMOVE the controller\n this._detachController(controller.uniqueId);\n });\n this._scene.constantlyUpdateMeshUnderPointer = true;\n return true;\n }\n /**\n * Detach this feature.\n * Will usually be called by the features manager\n *\n * @returns true if successful.\n */\n detach() {\n if (!super.detach()) {\n return false;\n }\n Object.keys(this._controllers).forEach((controllerId) => {\n this._detachController(controllerId);\n });\n return true;\n }\n /**\n * Will get the mesh under a specific pointer.\n * `scene.meshUnderPointer` will only return one mesh - either left or right.\n * @param controllerId the controllerId to check\n * @returns The mesh under pointer or null if no mesh is under the pointer\n */\n getMeshUnderPointer(controllerId) {\n if (this._controllers[controllerId]) {\n return this._controllers[controllerId].meshUnderPointer;\n }\n else {\n return null;\n }\n }\n /**\n * Get the xr controller that correlates to the pointer id in the pointer event\n *\n * @param id the pointer id to search for\n * @returns the controller that correlates to this id or null if not found\n */\n getXRControllerByPointerId(id) {\n const keys = Object.keys(this._controllers);\n for (let i = 0; i < keys.length; ++i) {\n if (this._controllers[keys[i]].id === id) {\n return this._controllers[keys[i]].xrController || null;\n }\n }\n return null;\n }\n /**\n * This function sets webXRControllerPointerSelection feature that will be disabled when\n * the hover range is reached for a mesh and will be reattached when not in hover range.\n * This is used to remove the selection rays when moving.\n * @param farInteractionFeature the feature to disable when finger is in hover range for a mesh\n */\n setFarInteractionFeature(farInteractionFeature) {\n this._farInteractionFeature = farInteractionFeature;\n }\n /**\n * Filter used for near interaction pick and hover\n * @param mesh\n */\n _nearPickPredicate(mesh) {\n return mesh.isEnabled() && mesh.isVisible && mesh.isPickable && mesh.isNearPickable;\n }\n /**\n * Filter used for near interaction grab\n * @param mesh\n */\n _nearGrabPredicate(mesh) {\n return mesh.isEnabled() && mesh.isVisible && mesh.isPickable && mesh.isNearGrabbable;\n }\n /**\n * Filter used for any near interaction\n * @param mesh\n */\n _nearInteractionPredicate(mesh) {\n return mesh.isEnabled() && mesh.isVisible && mesh.isPickable && (mesh.isNearPickable || mesh.isNearGrabbable);\n }\n _controllerAvailablePredicate(mesh, controllerId) {\n let parent = mesh;\n while (parent) {\n if (parent.reservedDataStore && parent.reservedDataStore.nearInteraction && parent.reservedDataStore.nearInteraction.excludedControllerId === controllerId) {\n return false;\n }\n parent = parent.parent;\n }\n return true;\n }\n _handleTransitionAnimation(controllerData, newState) {\n var _a;\n if (controllerData.currentAnimationState === newState ||\n this._options.nearInteractionControllerMode !== WebXRNearControllerMode.CENTERED_IN_FRONT ||\n !!((_a = controllerData.xrController) === null || _a === void 0 ? void 0 : _a.inputSource.hand)) {\n return;\n }\n // Don't always break to allow for animation fallthrough on rare cases of multi-transitions\n if (newState > controllerData.currentAnimationState) {\n switch (controllerData.currentAnimationState) {\n case ControllerOrbAnimationState.DEHYDRATED: {\n controllerData.hydrateCollisionMeshFunction(true);\n if (newState === ControllerOrbAnimationState.HOVER) {\n break;\n }\n }\n // eslint-disable-next-line no-fallthrough\n case ControllerOrbAnimationState.HOVER: {\n controllerData.touchCollisionMeshFunction(true);\n if (newState === ControllerOrbAnimationState.TOUCH) {\n break;\n }\n }\n }\n }\n else {\n switch (controllerData.currentAnimationState) {\n case ControllerOrbAnimationState.TOUCH: {\n controllerData.touchCollisionMeshFunction(false);\n if (newState === ControllerOrbAnimationState.HOVER) {\n break;\n }\n }\n // eslint-disable-next-line no-fallthrough\n case ControllerOrbAnimationState.HOVER: {\n controllerData.hydrateCollisionMeshFunction(false);\n if (newState === ControllerOrbAnimationState.DEHYDRATED) {\n break;\n }\n }\n }\n }\n controllerData.currentAnimationState = newState;\n }\n _processTouchPoint(id, position, orientation) {\n var _a;\n const controllerData = this._controllers[id];\n // Position and orientation could be temporary values, se we take care of them before calling any functions that use temporary vectors/quaternions\n controllerData.grabRay.origin.copyFrom(position);\n orientation.toEulerAnglesToRef(TmpVectors.Vector3[0]);\n controllerData.grabRay.direction.copyFrom(TmpVectors.Vector3[0]);\n if (this._options.nearInteractionControllerMode === WebXRNearControllerMode.CENTERED_IN_FRONT && !((_a = controllerData.xrController) === null || _a === void 0 ? void 0 : _a.inputSource.hand)) {\n // offset the touch point in the direction the transform is facing\n controllerData.xrController.getWorldPointerRayToRef(this._tmpRay);\n controllerData.grabRay.origin.addInPlace(this._tmpRay.direction.scale(0.05));\n }\n controllerData.grabRay.length = this._nearGrabLengthScale * this._hoverRadius;\n controllerData.touchCollisionMesh.position.copyFrom(controllerData.grabRay.origin);\n }\n _onXRFrame(_xrFrame) {\n Object.keys(this._controllers).forEach((id) => {\n var _a;\n // only do this for the selected pointer\n const controllerData = this._controllers[id];\n const handData = (_a = controllerData.xrController) === null || _a === void 0 ? void 0 : _a.inputSource.hand;\n // If near interaction is not enabled/available for this controller, return early\n if ((!this._options.enableNearInteractionOnAllControllers && id !== this._attachedController) ||\n !controllerData.xrController ||\n (!handData && (!this._options.nearInteractionControllerMode || !controllerData.xrController.inputSource.gamepad))) {\n controllerData.pick = null;\n return;\n }\n controllerData.hoverInteraction = false;\n controllerData.nearInteraction = false;\n // Every frame check collisions/input\n if (controllerData.xrController) {\n if (handData) {\n const xrIndexTip = handData.get(\"index-finger-tip\");\n if (xrIndexTip) {\n const indexTipPose = _xrFrame.getJointPose(xrIndexTip, this._xrSessionManager.referenceSpace);\n if (indexTipPose && indexTipPose.transform) {\n const axisRHSMultiplier = this._scene.useRightHandedSystem ? 1 : -1;\n TmpVectors.Vector3[0].set(indexTipPose.transform.position.x, indexTipPose.transform.position.y, indexTipPose.transform.position.z * axisRHSMultiplier);\n TmpVectors.Quaternion[0].set(indexTipPose.transform.orientation.x, indexTipPose.transform.orientation.y, indexTipPose.transform.orientation.z * axisRHSMultiplier, indexTipPose.transform.orientation.w * axisRHSMultiplier);\n this._processTouchPoint(id, TmpVectors.Vector3[0], TmpVectors.Quaternion[0]);\n }\n }\n }\n else if (controllerData.xrController.inputSource.gamepad && this._options.nearInteractionControllerMode !== WebXRNearControllerMode.DISABLED) {\n let controllerPose = controllerData.xrController.pointer;\n if (controllerData.xrController.grip && this._options.nearInteractionControllerMode === WebXRNearControllerMode.CENTERED_ON_CONTROLLER) {\n controllerPose = controllerData.xrController.grip;\n }\n this._processTouchPoint(id, controllerPose.position, controllerPose.rotationQuaternion);\n }\n }\n else {\n return;\n }\n const accuratePickInfo = (originalScenePick, utilityScenePick) => {\n let pick = null;\n if (!utilityScenePick || !utilityScenePick.hit) {\n // No hit in utility scene\n pick = originalScenePick;\n }\n else if (!originalScenePick || !originalScenePick.hit) {\n // No hit in original scene\n pick = utilityScenePick;\n }\n else if (utilityScenePick.distance < originalScenePick.distance) {\n // Hit is closer in utility scene\n pick = utilityScenePick;\n }\n else {\n // Hit is closer in original scene\n pick = originalScenePick;\n }\n return pick;\n };\n const populateNearInteractionInfo = (nearInteractionInfo) => {\n let result = new PickingInfo();\n let nearInteractionAtOrigin = false;\n const nearInteraction = nearInteractionInfo && nearInteractionInfo.pickedPoint && nearInteractionInfo.hit;\n if (nearInteractionInfo === null || nearInteractionInfo === void 0 ? void 0 : nearInteractionInfo.pickedPoint) {\n nearInteractionAtOrigin = nearInteractionInfo.pickedPoint.x === 0 && nearInteractionInfo.pickedPoint.y === 0 && nearInteractionInfo.pickedPoint.z === 0;\n }\n if (nearInteraction && !nearInteractionAtOrigin) {\n result = nearInteractionInfo;\n }\n return result;\n };\n // Don't perform touch logic while grabbing, to prevent triggering touch interactions while in the middle of a grab interaction\n // Dont update cursor logic either - the cursor should already be visible for the grab to be in range,\n // and in order to maintain its position on the target mesh it is parented for the duration of the grab.\n if (!controllerData.grabInteraction) {\n let pick = null;\n // near interaction hover\n let utilitySceneHoverPick = null;\n if (this._options.useUtilityLayer && this._utilityLayerScene) {\n utilitySceneHoverPick = this._pickWithSphere(controllerData, this._hoverRadius, this._utilityLayerScene, (mesh) => this._nearInteractionPredicate(mesh));\n }\n const originalSceneHoverPick = this._pickWithSphere(controllerData, this._hoverRadius, this._scene, (mesh) => this._nearInteractionPredicate(mesh));\n const hoverPickInfo = accuratePickInfo(originalSceneHoverPick, utilitySceneHoverPick);\n if (hoverPickInfo && hoverPickInfo.hit) {\n pick = populateNearInteractionInfo(hoverPickInfo);\n if (pick.hit) {\n controllerData.hoverInteraction = true;\n }\n }\n // near interaction pick\n if (controllerData.hoverInteraction) {\n let utilitySceneNearPick = null;\n const radius = handData ? this._pickRadius : this._controllerPickRadius;\n if (this._options.useUtilityLayer && this._utilityLayerScene) {\n utilitySceneNearPick = this._pickWithSphere(controllerData, radius, this._utilityLayerScene, (mesh) => this._nearPickPredicate(mesh));\n }\n const originalSceneNearPick = this._pickWithSphere(controllerData, radius, this._scene, (mesh) => this._nearPickPredicate(mesh));\n const pickInfo = accuratePickInfo(originalSceneNearPick, utilitySceneNearPick);\n const nearPick = populateNearInteractionInfo(pickInfo);\n if (nearPick.hit) {\n // Near pick takes precedence over hover interaction\n pick = nearPick;\n controllerData.nearInteraction = true;\n }\n }\n controllerData.stalePick = controllerData.pick;\n controllerData.pick = pick;\n // Update mesh under pointer\n if (controllerData.pick && controllerData.pick.pickedPoint && controllerData.pick.hit) {\n controllerData.meshUnderPointer = controllerData.pick.pickedMesh;\n controllerData.pickedPointVisualCue.position.copyFrom(controllerData.pick.pickedPoint);\n controllerData.pickedPointVisualCue.isVisible = true;\n if (this._farInteractionFeature && this._farInteractionFeature.attached) {\n this._farInteractionFeature._setPointerSelectionDisabledByPointerId(controllerData.id, true);\n }\n }\n else {\n controllerData.meshUnderPointer = null;\n controllerData.pickedPointVisualCue.isVisible = false;\n if (this._farInteractionFeature && this._farInteractionFeature.attached) {\n this._farInteractionFeature._setPointerSelectionDisabledByPointerId(controllerData.id, false);\n }\n }\n }\n // Update the interaction animation. Only updates if the visible touch mesh is active\n let state = ControllerOrbAnimationState.DEHYDRATED;\n if (controllerData.grabInteraction || controllerData.nearInteraction) {\n state = ControllerOrbAnimationState.TOUCH;\n }\n else if (controllerData.hoverInteraction) {\n state = ControllerOrbAnimationState.HOVER;\n }\n this._handleTransitionAnimation(controllerData, state);\n });\n }\n get _utilityLayerScene() {\n return this._options.customUtilityLayerScene || UtilityLayerRenderer.DefaultUtilityLayer.utilityLayerScene;\n }\n _generateVisualCue() {\n const sceneToRenderTo = this._options.useUtilityLayer ? this._options.customUtilityLayerScene || UtilityLayerRenderer.DefaultUtilityLayer.utilityLayerScene : this._scene;\n const selectionMesh = CreateSphere(\"nearInteraction\", {\n diameter: 0.0035 * 3,\n }, sceneToRenderTo);\n selectionMesh.bakeCurrentTransformIntoVertices();\n selectionMesh.isPickable = false;\n selectionMesh.isVisible = false;\n selectionMesh.rotationQuaternion = Quaternion.Identity();\n const targetMat = new StandardMaterial(\"targetMat\", sceneToRenderTo);\n targetMat.specularColor = Color3.Black();\n targetMat.emissiveColor = this.selectionMeshDefaultColor;\n targetMat.backFaceCulling = false;\n selectionMesh.material = targetMat;\n return selectionMesh;\n }\n _isControllerReadyForNearInteraction(id) {\n if (this._farInteractionFeature) {\n return this._farInteractionFeature._getPointerSelectionDisabledByPointerId(id);\n }\n return true;\n }\n _attachNearInteractionMode(xrController) {\n const controllerData = this._controllers[xrController.uniqueId];\n const pointerEventInit = {\n pointerId: controllerData.id,\n pointerType: \"xr-near\",\n };\n controllerData.onFrameObserver = this._xrSessionManager.onXRFrameObservable.add(() => {\n if ((!this._options.enableNearInteractionOnAllControllers && xrController.uniqueId !== this._attachedController) ||\n !controllerData.xrController ||\n (!controllerData.xrController.inputSource.hand && (!this._options.nearInteractionControllerMode || !controllerData.xrController.inputSource.gamepad))) {\n return;\n }\n if (controllerData.pick) {\n controllerData.pick.ray = controllerData.grabRay;\n }\n if (controllerData.pick && this._isControllerReadyForNearInteraction(controllerData.id)) {\n this._scene.simulatePointerMove(controllerData.pick, pointerEventInit);\n }\n // Near pick pointer event\n if (controllerData.nearInteraction && controllerData.pick && controllerData.pick.hit) {\n if (!controllerData.nearInteractionTargetMesh) {\n this._scene.simulatePointerDown(controllerData.pick, pointerEventInit);\n controllerData.nearInteractionTargetMesh = controllerData.meshUnderPointer;\n }\n }\n else if (controllerData.nearInteractionTargetMesh && controllerData.stalePick) {\n this._scene.simulatePointerUp(controllerData.stalePick, pointerEventInit);\n controllerData.nearInteractionTargetMesh = null;\n }\n });\n const grabCheck = (pressed) => {\n if (this._options.enableNearInteractionOnAllControllers ||\n (xrController.uniqueId === this._attachedController && this._isControllerReadyForNearInteraction(controllerData.id))) {\n if (controllerData.pick) {\n controllerData.pick.ray = controllerData.grabRay;\n }\n if (pressed && controllerData.pick && controllerData.meshUnderPointer && this._nearGrabPredicate(controllerData.meshUnderPointer)) {\n controllerData.grabInteraction = true;\n controllerData.pickedPointVisualCue.isVisible = false;\n this._scene.simulatePointerDown(controllerData.pick, pointerEventInit);\n }\n else if (!pressed && controllerData.pick && controllerData.grabInteraction) {\n this._scene.simulatePointerUp(controllerData.pick, pointerEventInit);\n controllerData.grabInteraction = false;\n controllerData.pickedPointVisualCue.isVisible = true;\n }\n }\n else {\n if (pressed && !this._options.enableNearInteractionOnAllControllers && !this._options.disableSwitchOnClick) {\n this._attachedController = xrController.uniqueId;\n }\n }\n };\n if (xrController.inputSource.gamepad) {\n const init = (motionController) => {\n controllerData.squeezeComponent = motionController.getComponent(\"grasp\");\n if (controllerData.squeezeComponent) {\n controllerData.onSqueezeButtonChangedObserver = controllerData.squeezeComponent.onButtonStateChangedObservable.add((component) => {\n if (component.changes.pressed) {\n const pressed = component.changes.pressed.current;\n grabCheck(pressed);\n }\n });\n }\n else {\n controllerData.selectionComponent = motionController.getMainComponent();\n controllerData.onButtonChangedObserver = controllerData.selectionComponent.onButtonStateChangedObservable.add((component) => {\n if (component.changes.pressed) {\n const pressed = component.changes.pressed.current;\n grabCheck(pressed);\n }\n });\n }\n };\n if (xrController.motionController) {\n init(xrController.motionController);\n }\n else {\n xrController.onMotionControllerInitObservable.add(init);\n }\n }\n else {\n // use the select and squeeze events\n const selectStartListener = (event) => {\n if (controllerData.xrController &&\n event.inputSource === controllerData.xrController.inputSource &&\n controllerData.pick &&\n this._isControllerReadyForNearInteraction(controllerData.id) &&\n controllerData.meshUnderPointer &&\n this._nearGrabPredicate(controllerData.meshUnderPointer)) {\n controllerData.grabInteraction = true;\n controllerData.pickedPointVisualCue.isVisible = false;\n this._scene.simulatePointerDown(controllerData.pick, pointerEventInit);\n }\n };\n const selectEndListener = (event) => {\n if (controllerData.xrController &&\n event.inputSource === controllerData.xrController.inputSource &&\n controllerData.pick &&\n this._isControllerReadyForNearInteraction(controllerData.id)) {\n this._scene.simulatePointerUp(controllerData.pick, pointerEventInit);\n controllerData.grabInteraction = false;\n controllerData.pickedPointVisualCue.isVisible = true;\n }\n };\n controllerData.eventListeners = {\n selectend: selectEndListener,\n selectstart: selectStartListener,\n };\n this._xrSessionManager.session.addEventListener(\"selectstart\", selectStartListener);\n this._xrSessionManager.session.addEventListener(\"selectend\", selectEndListener);\n }\n }\n _detachController(xrControllerUniqueId) {\n const controllerData = this._controllers[xrControllerUniqueId];\n if (!controllerData) {\n return;\n }\n if (controllerData.squeezeComponent) {\n if (controllerData.onSqueezeButtonChangedObserver) {\n controllerData.squeezeComponent.onButtonStateChangedObservable.remove(controllerData.onSqueezeButtonChangedObserver);\n }\n }\n if (controllerData.selectionComponent) {\n if (controllerData.onButtonChangedObserver) {\n controllerData.selectionComponent.onButtonStateChangedObservable.remove(controllerData.onButtonChangedObserver);\n }\n }\n if (controllerData.onFrameObserver) {\n this._xrSessionManager.onXRFrameObservable.remove(controllerData.onFrameObserver);\n }\n if (controllerData.eventListeners) {\n Object.keys(controllerData.eventListeners).forEach((eventName) => {\n const func = controllerData.eventListeners && controllerData.eventListeners[eventName];\n if (func) {\n this._xrSessionManager.session.removeEventListener(eventName, func);\n }\n });\n }\n controllerData.touchCollisionMesh.dispose();\n controllerData.pickedPointVisualCue.dispose();\n this._xrSessionManager.runInXRFrame(() => {\n // Fire a pointerup\n const pointerEventInit = {\n pointerId: controllerData.id,\n pointerType: \"xr-near\",\n };\n this._scene.simulatePointerUp(new PickingInfo(), pointerEventInit);\n });\n // remove from the map\n delete this._controllers[xrControllerUniqueId];\n if (this._attachedController === xrControllerUniqueId) {\n // check for other controllers\n const keys = Object.keys(this._controllers);\n if (keys.length) {\n this._attachedController = keys[0];\n }\n else {\n this._attachedController = \"\";\n }\n }\n }\n _generateNewTouchPointMesh() {\n // populate information for near hover, pick and pinch\n const meshCreationScene = this._options.useUtilityLayer ? this._options.customUtilityLayerScene || UtilityLayerRenderer.DefaultUtilityLayer.utilityLayerScene : this._scene;\n const touchCollisionMesh = CreateSphere(\"PickSphere\", { diameter: 1 }, meshCreationScene);\n touchCollisionMesh.isVisible = false;\n // Generate the material for the touch mesh visuals\n if (this._options.motionControllerOrbMaterial) {\n touchCollisionMesh.material = this._options.motionControllerOrbMaterial;\n }\n else {\n NodeMaterial.ParseFromSnippetAsync(\"8RUNKL#3\", meshCreationScene).then((nodeMaterial) => {\n touchCollisionMesh.material = nodeMaterial;\n });\n }\n const easingFunction = new QuadraticEase();\n easingFunction.setEasingMode(EasingFunction.EASINGMODE_EASEINOUT);\n // Adjust the visual size based off of the size of the touch collision orb.\n // Having the size perfectly match for hover gives a more accurate tell for when the user will start interacting with the target\n // Sizes for other states are somewhat arbitrary, as they are based on what feels nice during an interaction\n const hoverSizeVec = new Vector3(this._controllerPickRadius, this._controllerPickRadius, this._controllerPickRadius);\n const touchSize = this._controllerPickRadius * (4 / 3);\n const touchSizeVec = new Vector3(touchSize, touchSize, touchSize);\n const hydrateTransitionSize = this._controllerPickRadius * (7 / 6);\n const hydrateTransitionSizeVec = new Vector3(hydrateTransitionSize, hydrateTransitionSize, hydrateTransitionSize);\n const touchHoverTransitionSize = this._controllerPickRadius * (4 / 5);\n const touchHoverTransitionSizeVec = new Vector3(touchHoverTransitionSize, touchHoverTransitionSize, touchHoverTransitionSize);\n const hoverTouchTransitionSize = this._controllerPickRadius * (3 / 2);\n const hoverTouchTransitionSizeVec = new Vector3(hoverTouchTransitionSize, hoverTouchTransitionSize, hoverTouchTransitionSize);\n const touchKeys = [\n { frame: 0, value: hoverSizeVec },\n { frame: 10, value: hoverTouchTransitionSizeVec },\n { frame: 18, value: touchSizeVec },\n ];\n const releaseKeys = [\n { frame: 0, value: touchSizeVec },\n { frame: 10, value: touchHoverTransitionSizeVec },\n { frame: 18, value: hoverSizeVec },\n ];\n const hydrateKeys = [\n { frame: 0, value: Vector3.ZeroReadOnly },\n { frame: 12, value: hydrateTransitionSizeVec },\n { frame: 15, value: hoverSizeVec },\n ];\n const dehydrateKeys = [\n { frame: 0, value: hoverSizeVec },\n { frame: 10, value: Vector3.ZeroReadOnly },\n { frame: 15, value: Vector3.ZeroReadOnly },\n ];\n const touchAction = new Animation(\"touch\", \"scaling\", 60, Animation.ANIMATIONTYPE_VECTOR3, Animation.ANIMATIONLOOPMODE_CONSTANT);\n const releaseAction = new Animation(\"release\", \"scaling\", 60, Animation.ANIMATIONTYPE_VECTOR3, Animation.ANIMATIONLOOPMODE_CONSTANT);\n const hydrateAction = new Animation(\"hydrate\", \"scaling\", 60, Animation.ANIMATIONTYPE_VECTOR3, Animation.ANIMATIONLOOPMODE_CONSTANT);\n const dehydrateAction = new Animation(\"dehydrate\", \"scaling\", 60, Animation.ANIMATIONTYPE_VECTOR3, Animation.ANIMATIONLOOPMODE_CONSTANT);\n touchAction.setEasingFunction(easingFunction);\n releaseAction.setEasingFunction(easingFunction);\n hydrateAction.setEasingFunction(easingFunction);\n dehydrateAction.setEasingFunction(easingFunction);\n touchAction.setKeys(touchKeys);\n releaseAction.setKeys(releaseKeys);\n hydrateAction.setKeys(hydrateKeys);\n dehydrateAction.setKeys(dehydrateKeys);\n const touchCollisionMeshFunction = (isTouch) => {\n const action = isTouch ? touchAction : releaseAction;\n meshCreationScene.beginDirectAnimation(touchCollisionMesh, [action], 0, 18, false, 1);\n };\n const hydrateCollisionMeshFunction = (isHydration) => {\n const action = isHydration ? hydrateAction : dehydrateAction;\n if (isHydration) {\n touchCollisionMesh.isVisible = true;\n }\n meshCreationScene.beginDirectAnimation(touchCollisionMesh, [action], 0, 15, false, 1, () => {\n if (!isHydration) {\n touchCollisionMesh.isVisible = false;\n }\n });\n };\n return { touchCollisionMesh, touchCollisionMeshFunction, hydrateCollisionMeshFunction };\n }\n _pickWithSphere(controllerData, radius, sceneToUse, predicate) {\n const pickingInfo = new PickingInfo();\n pickingInfo.distance = +Infinity;\n if (controllerData.touchCollisionMesh && controllerData.xrController) {\n const position = controllerData.touchCollisionMesh.position;\n const sphere = BoundingSphere.CreateFromCenterAndRadius(position, radius);\n for (let meshIndex = 0; meshIndex < sceneToUse.meshes.length; meshIndex++) {\n const mesh = sceneToUse.meshes[meshIndex];\n if (!predicate(mesh) || !this._controllerAvailablePredicate(mesh, controllerData.xrController.uniqueId)) {\n continue;\n }\n const result = WebXRNearInteraction.PickMeshWithSphere(mesh, sphere);\n if (result && result.hit && result.distance < pickingInfo.distance) {\n pickingInfo.hit = result.hit;\n pickingInfo.pickedMesh = mesh;\n pickingInfo.pickedPoint = result.pickedPoint;\n pickingInfo.aimTransform = controllerData.xrController.pointer;\n pickingInfo.gripTransform = controllerData.xrController.grip || null;\n pickingInfo.originMesh = controllerData.touchCollisionMesh;\n pickingInfo.distance = result.distance;\n }\n }\n }\n return pickingInfo;\n }\n /**\n * Picks a mesh with a sphere\n * @param mesh the mesh to pick\n * @param sphere picking sphere in world coordinates\n * @param skipBoundingInfo a boolean indicating if we should skip the bounding info check\n * @returns the picking info\n */\n static PickMeshWithSphere(mesh, sphere, skipBoundingInfo = false) {\n const subMeshes = mesh.subMeshes;\n const pi = new PickingInfo();\n const boundingInfo = mesh.getBoundingInfo();\n if (!mesh._generatePointsArray()) {\n return pi;\n }\n if (!mesh.subMeshes || !boundingInfo) {\n return pi;\n }\n if (!skipBoundingInfo && !BoundingSphere.Intersects(boundingInfo.boundingSphere, sphere)) {\n return pi;\n }\n const result = TmpVectors.Vector3[0];\n const tmpVec = TmpVectors.Vector3[1];\n let distance = +Infinity;\n let tmp, tmpDistanceSphereToCenter, tmpDistanceSurfaceToCenter;\n const center = TmpVectors.Vector3[2];\n const worldToMesh = TmpVectors.Matrix[0];\n worldToMesh.copyFrom(mesh.getWorldMatrix());\n worldToMesh.invert();\n Vector3.TransformCoordinatesToRef(sphere.center, worldToMesh, center);\n for (let index = 0; index < subMeshes.length; index++) {\n const subMesh = subMeshes[index];\n subMesh.projectToRef(center, mesh._positions, mesh.getIndices(), tmpVec);\n Vector3.TransformCoordinatesToRef(tmpVec, mesh.getWorldMatrix(), tmpVec);\n tmp = Vector3.Distance(tmpVec, sphere.center);\n // Check for finger inside of mesh\n tmpDistanceSurfaceToCenter = Vector3.Distance(tmpVec, mesh.getAbsolutePosition());\n tmpDistanceSphereToCenter = Vector3.Distance(sphere.center, mesh.getAbsolutePosition());\n if (tmpDistanceSphereToCenter !== -1 && tmpDistanceSurfaceToCenter !== -1 && tmpDistanceSurfaceToCenter > tmpDistanceSphereToCenter) {\n tmp = 0;\n tmpVec.copyFrom(sphere.center);\n }\n if (tmp !== -1 && tmp < distance) {\n distance = tmp;\n result.copyFrom(tmpVec);\n }\n }\n if (distance < sphere.radius) {\n pi.hit = true;\n pi.distance = distance;\n pi.pickedMesh = mesh;\n pi.pickedPoint = result.clone();\n }\n return pi;\n }\n }\n WebXRNearInteraction._IdCounter = 200;\n /**\n * The module's name\n */\n WebXRNearInteraction.Name = WebXRFeatureName.NEAR_INTERACTION;\n /**\n * The (Babylon) version of this module.\n * This is an integer representing the implementation version.\n * This number does not correspond to the WebXR specs version\n */\n WebXRNearInteraction.Version = 1;\n //Register the plugin\n WebXRFeaturesManager.AddWebXRFeature(WebXRNearInteraction.Name, (xrSessionManager, options) => {\n return () => new WebXRNearInteraction(xrSessionManager, options);\n }, WebXRNearInteraction.Version, true);\n\n /**\n * Button which can be used to enter a different mode of XR\n */\n class WebXREnterExitUIButton {\n /**\n * Creates a WebXREnterExitUIButton\n * @param element button element\n * @param sessionMode XR initialization session mode\n * @param referenceSpaceType the type of reference space to be used\n */\n constructor(\n /** button element */\n element, \n /** XR initialization options for the button */\n sessionMode, \n /** Reference space type */\n referenceSpaceType) {\n this.element = element;\n this.sessionMode = sessionMode;\n this.referenceSpaceType = referenceSpaceType;\n }\n /**\n * Extendable function which can be used to update the button's visuals when the state changes\n * @param activeButton the current active button in the UI\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n update(activeButton) { }\n }\n /**\n * UI to allow the user to enter/exit XR mode\n */\n class WebXREnterExitUI {\n /**\n * Construct a new EnterExit UI class\n *\n * @param _scene babylon scene object to use\n * @param options (read-only) version of the options passed to this UI\n */\n constructor(_scene, \n /** version of the options passed to this UI */\n options) {\n this._scene = _scene;\n this.options = options;\n this._activeButton = null;\n this._buttons = [];\n /**\n * Fired every time the active button is changed.\n *\n * When xr is entered via a button that launches xr that button will be the callback parameter\n *\n * When exiting xr the callback parameter will be null)\n */\n this.activeButtonChangedObservable = new Observable$1();\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n this._onSessionGranted = (evt) => {\n // This section is for future reference.\n // As per specs, evt.session.mode should have the supported session mode, but no browser supports it for now.\n // // check if the session granted is the same as the one requested\n // const grantedMode = (evt.session as any).mode;\n // if (grantedMode) {\n // this._buttons.some((btn, idx) => {\n // if (btn.sessionMode === grantedMode) {\n // this._enterXRWithButtonIndex(idx);\n // return true;\n // }\n // return false;\n // });\n // } else\n if (this._helper) {\n this._enterXRWithButtonIndex(0);\n }\n };\n this.overlay = document.createElement(\"div\");\n this.overlay.classList.add(\"xr-button-overlay\");\n // prepare for session granted event\n if (!options.ignoreSessionGrantedEvent && navigator.xr) {\n navigator.xr.addEventListener(\"sessiongranted\", this._onSessionGranted);\n }\n // if served over HTTP, warn people.\n // Hopefully the browsers will catch up\n if (typeof window !== \"undefined\") {\n if (window.location && window.location.protocol === \"http:\" && window.location.hostname !== \"localhost\") {\n Tools.Warn(\"WebXR can only be served over HTTPS\");\n throw new Error(\"WebXR can only be served over HTTPS\");\n }\n }\n if (options.customButtons) {\n this._buttons = options.customButtons;\n }\n else {\n this.overlay.style.cssText = \"z-index:11;position: absolute; right: 20px;bottom: 50px;\";\n const sessionMode = options.sessionMode || \"immersive-vr\";\n const referenceSpaceType = options.referenceSpaceType || \"local-floor\";\n const url = typeof SVGSVGElement === \"undefined\"\n ? \"https://cdn.babylonjs.com/Assets/vrButton.png\"\n : \"data:image/svg+xml;charset=UTF-8,%3Csvg%20xmlns%3D%22http%3A//www.w3.org/2000/svg%22%20width%3D%222048%22%20height%3D%221152%22%20viewBox%3D%220%200%202048%201152%22%20version%3D%221.1%22%3E%3Cpath%20transform%3D%22rotate%28180%201024%2C576.0000000000001%29%22%20d%3D%22m1109%2C896q17%2C0%2030%2C-12t13%2C-30t-12.5%2C-30.5t-30.5%2C-12.5l-170%2C0q-18%2C0%20-30.5%2C12.5t-12.5%2C30.5t13%2C30t30%2C12l170%2C0zm-85%2C256q59%2C0%20132.5%2C-1.5t154.5%2C-5.5t164.5%2C-11.5t163%2C-20t150%2C-30t124.5%2C-41.5q23%2C-11%2042%2C-24t38%2C-30q27%2C-25%2041%2C-61.5t14%2C-72.5l0%2C-257q0%2C-123%20-47%2C-232t-128%2C-190t-190%2C-128t-232%2C-47l-81%2C0q-37%2C0%20-68.5%2C14t-60.5%2C34.5t-55.5%2C45t-53%2C45t-53%2C34.5t-55.5%2C14t-55.5%2C-14t-53%2C-34.5t-53%2C-45t-55.5%2C-45t-60.5%2C-34.5t-68.5%2C-14l-81%2C0q-123%2C0%20-232%2C47t-190%2C128t-128%2C190t-47%2C232l0%2C257q0%2C68%2038%2C115t97%2C73q54%2C24%20124.5%2C41.5t150%2C30t163%2C20t164.5%2C11.5t154.5%2C5.5t132.5%2C1.5zm939%2C-298q0%2C39%20-24.5%2C67t-58.5%2C42q-54%2C23%20-122%2C39.5t-143.5%2C28t-155.5%2C19t-157%2C11t-148.5%2C5t-129.5%2C1.5q-59%2C0%20-130%2C-1.5t-148%2C-5t-157%2C-11t-155.5%2C-19t-143.5%2C-28t-122%2C-39.5q-34%2C-14%20-58.5%2C-42t-24.5%2C-67l0%2C-257q0%2C-106%2040.5%2C-199t110%2C-162.5t162.5%2C-109.5t199%2C-40l81%2C0q27%2C0%2052%2C14t50%2C34.5t51%2C44.5t55.5%2C44.5t63.5%2C34.5t74%2C14t74%2C-14t63.5%2C-34.5t55.5%2C-44.5t51%2C-44.5t50%2C-34.5t52%2C-14l14%2C0q37%2C0%2070%2C0.5t64.5%2C4.5t63.5%2C12t68%2C23q71%2C30%20128.5%2C78.5t98.5%2C110t63.5%2C133.5t22.5%2C149l0%2C257z%22%20fill%3D%22white%22%20/%3E%3C/svg%3E%0A\";\n let css = \".babylonVRicon { color: #868686; border-color: #868686; border-style: solid; margin-left: 10px; height: 50px; width: 80px; background-color: rgba(51,51,51,0.7); background-image: url(\" +\n url +\n \"); background-size: 80%; background-repeat:no-repeat; background-position: center; border: none; outline: none; transition: transform 0.125s ease-out } .babylonVRicon:hover { transform: scale(1.05) } .babylonVRicon:active {background-color: rgba(51,51,51,1) } .babylonVRicon:focus {background-color: rgba(51,51,51,1) }\";\n css += '.babylonVRicon.vrdisplaypresenting { background-image: none;} .vrdisplaypresenting::after { content: \"EXIT\"} .xr-error::after { content: \"ERROR\"}';\n const style = document.createElement(\"style\");\n style.appendChild(document.createTextNode(css));\n document.getElementsByTagName(\"head\")[0].appendChild(style);\n const hmdBtn = document.createElement(\"button\");\n hmdBtn.className = \"babylonVRicon\";\n hmdBtn.title = `${sessionMode} - ${referenceSpaceType}`;\n this._buttons.push(new WebXREnterExitUIButton(hmdBtn, sessionMode, referenceSpaceType));\n this._buttons[this._buttons.length - 1].update = function (activeButton) {\n this.element.style.display = activeButton === null || activeButton === this ? \"\" : \"none\";\n hmdBtn.className = \"babylonVRicon\" + (activeButton === this ? \" vrdisplaypresenting\" : \"\");\n };\n this._updateButtons(null);\n }\n const renderCanvas = _scene.getEngine().getInputElement();\n if (renderCanvas && renderCanvas.parentNode) {\n renderCanvas.parentNode.appendChild(this.overlay);\n _scene.onDisposeObservable.addOnce(() => {\n this.dispose();\n });\n }\n }\n /**\n * Set the helper to be used with this UI component.\n * The UI is bound to an experience helper. If not provided the UI can still be used but the events should be registered by the developer.\n *\n * @param helper the experience helper to attach\n * @param renderTarget an optional render target (in case it is created outside of the helper scope)\n * @returns a promise that resolves when the ui is ready\n */\n async setHelperAsync(helper, renderTarget) {\n this._helper = helper;\n this._renderTarget = renderTarget;\n const supportedPromises = this._buttons.map((btn) => {\n return helper.sessionManager.isSessionSupportedAsync(btn.sessionMode);\n });\n helper.onStateChangedObservable.add((state) => {\n if (state == WebXRState.NOT_IN_XR) {\n this._updateButtons(null);\n }\n });\n const results = await Promise.all(supportedPromises);\n results.forEach((supported, i) => {\n if (supported) {\n this.overlay.appendChild(this._buttons[i].element);\n this._buttons[i].element.onclick = this._enterXRWithButtonIndex.bind(this, i);\n }\n else {\n Tools.Warn(`Session mode \"${this._buttons[i].sessionMode}\" not supported in browser`);\n }\n });\n }\n /**\n * Creates UI to allow the user to enter/exit XR mode\n * @param scene the scene to add the ui to\n * @param helper the xr experience helper to enter/exit xr with\n * @param options options to configure the UI\n * @returns the created ui\n */\n static async CreateAsync(scene, helper, options) {\n const ui = new WebXREnterExitUI(scene, options);\n await ui.setHelperAsync(helper, options.renderTarget || undefined);\n return ui;\n }\n async _enterXRWithButtonIndex(idx = 0) {\n if (this._helper.state == WebXRState.IN_XR) {\n await this._helper.exitXRAsync();\n this._updateButtons(null);\n }\n else if (this._helper.state == WebXRState.NOT_IN_XR) {\n try {\n await this._helper.enterXRAsync(this._buttons[idx].sessionMode, this._buttons[idx].referenceSpaceType, this._renderTarget, {\n optionalFeatures: this.options.optionalFeatures,\n requiredFeatures: this.options.requiredFeatures,\n });\n this._updateButtons(this._buttons[idx]);\n }\n catch (e) {\n // make sure button is visible\n this._updateButtons(null);\n const element = this._buttons[idx].element;\n const prevTitle = element.title;\n element.title = \"Error entering XR session : \" + prevTitle;\n element.classList.add(\"xr-error\");\n if (this.options.onError) {\n this.options.onError(e);\n }\n }\n }\n }\n /**\n * Disposes of the XR UI component\n */\n dispose() {\n const renderCanvas = this._scene.getEngine().getInputElement();\n if (renderCanvas && renderCanvas.parentNode && renderCanvas.parentNode.contains(this.overlay)) {\n renderCanvas.parentNode.removeChild(this.overlay);\n }\n this.activeButtonChangedObservable.clear();\n navigator.xr.removeEventListener(\"sessiongranted\", this._onSessionGranted);\n }\n _updateButtons(activeButton) {\n this._activeButton = activeButton;\n this._buttons.forEach((b) => {\n b.update(this._activeButton);\n });\n this.activeButtonChangedObservable.notifyObservers(this._activeButton);\n }\n }\n\n Mesh._instancedMeshFactory = (name, mesh) => {\n const instance = new InstancedMesh(name, mesh);\n if (mesh.instancedBuffers) {\n instance.instancedBuffers = {};\n for (const key in mesh.instancedBuffers) {\n instance.instancedBuffers[key] = mesh.instancedBuffers[key];\n }\n }\n return instance;\n };\n /**\n * Creates an instance based on a source mesh.\n */\n class InstancedMesh extends AbstractMesh {\n /**\n * Creates a new InstancedMesh object from the mesh source.\n * @param name defines the name of the instance\n * @param source the mesh to create the instance from\n */\n constructor(name, source) {\n super(name, source.getScene());\n /** @internal */\n this._indexInSourceMeshInstanceArray = -1;\n /** @internal */\n this._distanceToCamera = 0;\n source.addInstance(this);\n this._sourceMesh = source;\n this._unIndexed = source._unIndexed;\n this.position.copyFrom(source.position);\n this.rotation.copyFrom(source.rotation);\n this.scaling.copyFrom(source.scaling);\n if (source.rotationQuaternion) {\n this.rotationQuaternion = source.rotationQuaternion.clone();\n }\n this.animations = source.animations.slice();\n for (const range of source.getAnimationRanges()) {\n if (range != null) {\n this.createAnimationRange(range.name, range.from, range.to);\n }\n }\n this.infiniteDistance = source.infiniteDistance;\n this.setPivotMatrix(source.getPivotMatrix());\n this.refreshBoundingInfo(true, true);\n this._syncSubMeshes();\n }\n /**\n * Returns the string \"InstancedMesh\".\n */\n getClassName() {\n return \"InstancedMesh\";\n }\n /** Gets the list of lights affecting that mesh */\n get lightSources() {\n return this._sourceMesh._lightSources;\n }\n _resyncLightSources() {\n // Do nothing as all the work will be done by source mesh\n }\n _resyncLightSource() {\n // Do nothing as all the work will be done by source mesh\n }\n _removeLightSource() {\n // Do nothing as all the work will be done by source mesh\n }\n // Methods\n /**\n * If the source mesh receives shadows\n */\n get receiveShadows() {\n return this._sourceMesh.receiveShadows;\n }\n set receiveShadows(_value) {\n var _a;\n if (((_a = this._sourceMesh) === null || _a === void 0 ? void 0 : _a.receiveShadows) !== _value) {\n Tools.Warn(\"Setting receiveShadows on an instanced mesh has no effect\");\n }\n }\n /**\n * The material of the source mesh\n */\n get material() {\n return this._sourceMesh.material;\n }\n set material(_value) {\n var _a;\n if (((_a = this._sourceMesh) === null || _a === void 0 ? void 0 : _a.material) !== _value) {\n Tools.Warn(\"Setting material on an instanced mesh has no effect\");\n }\n }\n /**\n * Visibility of the source mesh\n */\n get visibility() {\n return this._sourceMesh.visibility;\n }\n set visibility(_value) {\n var _a;\n if (((_a = this._sourceMesh) === null || _a === void 0 ? void 0 : _a.visibility) !== _value) {\n Tools.Warn(\"Setting visibility on an instanced mesh has no effect\");\n }\n }\n /**\n * Skeleton of the source mesh\n */\n get skeleton() {\n return this._sourceMesh.skeleton;\n }\n set skeleton(_value) {\n var _a;\n if (((_a = this._sourceMesh) === null || _a === void 0 ? void 0 : _a.skeleton) !== _value) {\n Tools.Warn(\"Setting skeleton on an instanced mesh has no effect\");\n }\n }\n /**\n * Rendering ground id of the source mesh\n */\n get renderingGroupId() {\n return this._sourceMesh.renderingGroupId;\n }\n set renderingGroupId(value) {\n if (!this._sourceMesh || value === this._sourceMesh.renderingGroupId) {\n return;\n }\n //no-op with warning\n Logger.Warn(\"Note - setting renderingGroupId of an instanced mesh has no effect on the scene\");\n }\n /**\n * Returns the total number of vertices (integer).\n */\n getTotalVertices() {\n return this._sourceMesh ? this._sourceMesh.getTotalVertices() : 0;\n }\n /**\n * Returns a positive integer : the total number of indices in this mesh geometry.\n * @returns the number of indices or zero if the mesh has no geometry.\n */\n getTotalIndices() {\n return this._sourceMesh.getTotalIndices();\n }\n /**\n * The source mesh of the instance\n */\n get sourceMesh() {\n return this._sourceMesh;\n }\n /**\n * Creates a new InstancedMesh object from the mesh model.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/copies/instances\n * @param name defines the name of the new instance\n * @returns a new InstancedMesh\n */\n createInstance(name) {\n return this._sourceMesh.createInstance(name);\n }\n /**\n * Is this node ready to be used/rendered\n * @param completeCheck defines if a complete check (including materials and lights) has to be done (false by default)\n * @returns {boolean} is it ready\n */\n isReady(completeCheck = false) {\n return this._sourceMesh.isReady(completeCheck, true);\n }\n /**\n * Returns an array of integers or a typed array (Int32Array, Uint32Array, Uint16Array) populated with the mesh indices.\n * @param kind kind of verticies to retrieve (eg. positions, normals, uvs, etc.)\n * @param copyWhenShared If true (default false) and and if the mesh geometry is shared among some other meshes, the returned array is a copy of the internal one.\n * @param forceCopy defines a boolean forcing the copy of the buffer no matter what the value of copyWhenShared is\n * @returns a float array or a Float32Array of the requested kind of data : positions, normals, uvs, etc.\n */\n getVerticesData(kind, copyWhenShared, forceCopy) {\n return this._sourceMesh.getVerticesData(kind, copyWhenShared, forceCopy);\n }\n /**\n * Sets the vertex data of the mesh geometry for the requested `kind`.\n * If the mesh has no geometry, a new Geometry object is set to the mesh and then passed this vertex data.\n * The `data` are either a numeric array either a Float32Array.\n * The parameter `updatable` is passed as is to the underlying Geometry object constructor (if initially none) or updater.\n * The parameter `stride` is an optional positive integer, it is usually automatically deducted from the `kind` (3 for positions or normals, 2 for UV, etc).\n * Note that a new underlying VertexBuffer object is created each call.\n * If the `kind` is the `PositionKind`, the mesh BoundingInfo is renewed, so the bounding box and sphere, and the mesh World Matrix is recomputed.\n *\n * Possible `kind` values :\n * - VertexBuffer.PositionKind\n * - VertexBuffer.UVKind\n * - VertexBuffer.UV2Kind\n * - VertexBuffer.UV3Kind\n * - VertexBuffer.UV4Kind\n * - VertexBuffer.UV5Kind\n * - VertexBuffer.UV6Kind\n * - VertexBuffer.ColorKind\n * - VertexBuffer.MatricesIndicesKind\n * - VertexBuffer.MatricesIndicesExtraKind\n * - VertexBuffer.MatricesWeightsKind\n * - VertexBuffer.MatricesWeightsExtraKind\n *\n * Returns the Mesh.\n * @param kind\n * @param data\n * @param updatable\n * @param stride\n */\n setVerticesData(kind, data, updatable, stride) {\n if (this.sourceMesh) {\n this.sourceMesh.setVerticesData(kind, data, updatable, stride);\n }\n return this.sourceMesh;\n }\n /**\n * Updates the existing vertex data of the mesh geometry for the requested `kind`.\n * If the mesh has no geometry, it is simply returned as it is.\n * The `data` are either a numeric array either a Float32Array.\n * No new underlying VertexBuffer object is created.\n * If the `kind` is the `PositionKind` and if `updateExtends` is true, the mesh BoundingInfo is renewed, so the bounding box and sphere, and the mesh World Matrix is recomputed.\n * If the parameter `makeItUnique` is true, a new global geometry is created from this positions and is set to the mesh.\n *\n * Possible `kind` values :\n * - VertexBuffer.PositionKind\n * - VertexBuffer.UVKind\n * - VertexBuffer.UV2Kind\n * - VertexBuffer.UV3Kind\n * - VertexBuffer.UV4Kind\n * - VertexBuffer.UV5Kind\n * - VertexBuffer.UV6Kind\n * - VertexBuffer.ColorKind\n * - VertexBuffer.MatricesIndicesKind\n * - VertexBuffer.MatricesIndicesExtraKind\n * - VertexBuffer.MatricesWeightsKind\n * - VertexBuffer.MatricesWeightsExtraKind\n *\n * Returns the Mesh.\n * @param kind\n * @param data\n * @param updateExtends\n * @param makeItUnique\n */\n updateVerticesData(kind, data, updateExtends, makeItUnique) {\n if (this.sourceMesh) {\n this.sourceMesh.updateVerticesData(kind, data, updateExtends, makeItUnique);\n }\n return this.sourceMesh;\n }\n /**\n * Sets the mesh indices.\n * Expects an array populated with integers or a typed array (Int32Array, Uint32Array, Uint16Array).\n * If the mesh has no geometry, a new Geometry object is created and set to the mesh.\n * This method creates a new index buffer each call.\n * Returns the Mesh.\n * @param indices\n * @param totalVertices\n */\n setIndices(indices, totalVertices = null) {\n if (this.sourceMesh) {\n this.sourceMesh.setIndices(indices, totalVertices);\n }\n return this.sourceMesh;\n }\n /**\n * Boolean : True if the mesh owns the requested kind of data.\n * @param kind\n */\n isVerticesDataPresent(kind) {\n return this._sourceMesh.isVerticesDataPresent(kind);\n }\n /**\n * Returns an array of indices (IndicesArray).\n */\n getIndices() {\n return this._sourceMesh.getIndices();\n }\n get _positions() {\n return this._sourceMesh._positions;\n }\n /**\n * This method recomputes and sets a new BoundingInfo to the mesh unless it is locked.\n * This means the mesh underlying bounding box and sphere are recomputed.\n * @param applySkeleton defines whether to apply the skeleton before computing the bounding info\n * @param applyMorph defines whether to apply the morph target before computing the bounding info\n * @returns the current mesh\n */\n refreshBoundingInfo(applySkeleton = false, applyMorph = false) {\n if (this.hasBoundingInfo && this.getBoundingInfo().isLocked) {\n return this;\n }\n const bias = this._sourceMesh.geometry ? this._sourceMesh.geometry.boundingBias : null;\n this._refreshBoundingInfo(this._sourceMesh._getPositionData(applySkeleton, applyMorph), bias);\n return this;\n }\n /** @internal */\n _preActivate() {\n if (this._currentLOD) {\n this._currentLOD._preActivate();\n }\n return this;\n }\n /**\n * @internal\n */\n _activate(renderId, intermediateRendering) {\n super._activate(renderId, intermediateRendering);\n if (!this._sourceMesh.subMeshes) {\n Logger.Warn(\"Instances should only be created for meshes with geometry.\");\n }\n if (this._currentLOD) {\n const differentSign = this._currentLOD._getWorldMatrixDeterminant() >= 0 !== this._getWorldMatrixDeterminant() >= 0;\n if (differentSign) {\n this._internalAbstractMeshDataInfo._actAsRegularMesh = true;\n return true;\n }\n this._internalAbstractMeshDataInfo._actAsRegularMesh = false;\n this._currentLOD._registerInstanceForRenderId(this, renderId);\n if (intermediateRendering) {\n if (!this._currentLOD._internalAbstractMeshDataInfo._isActiveIntermediate) {\n this._currentLOD._internalAbstractMeshDataInfo._onlyForInstancesIntermediate = true;\n return true;\n }\n }\n else {\n if (!this._currentLOD._internalAbstractMeshDataInfo._isActive) {\n this._currentLOD._internalAbstractMeshDataInfo._onlyForInstances = true;\n return true;\n }\n }\n }\n return false;\n }\n /** @internal */\n _postActivate() {\n if (this._sourceMesh.edgesShareWithInstances && this._sourceMesh._edgesRenderer && this._sourceMesh._edgesRenderer.isEnabled && this._sourceMesh._renderingGroup) {\n // we are using the edge renderer of the source mesh\n this._sourceMesh._renderingGroup._edgesRenderers.pushNoDuplicate(this._sourceMesh._edgesRenderer);\n this._sourceMesh._edgesRenderer.customInstances.push(this.getWorldMatrix());\n }\n else if (this._edgesRenderer && this._edgesRenderer.isEnabled && this._sourceMesh._renderingGroup) {\n // we are using the edge renderer defined for this instance\n this._sourceMesh._renderingGroup._edgesRenderers.push(this._edgesRenderer);\n }\n }\n getWorldMatrix() {\n if (this._currentLOD && this._currentLOD.billboardMode !== TransformNode.BILLBOARDMODE_NONE && this._currentLOD._masterMesh !== this) {\n if (!this._billboardWorldMatrix) {\n this._billboardWorldMatrix = new Matrix();\n }\n const tempMaster = this._currentLOD._masterMesh;\n this._currentLOD._masterMesh = this;\n TmpVectors.Vector3[7].copyFrom(this._currentLOD.position);\n this._currentLOD.position.set(0, 0, 0);\n this._billboardWorldMatrix.copyFrom(this._currentLOD.computeWorldMatrix(true));\n this._currentLOD.position.copyFrom(TmpVectors.Vector3[7]);\n this._currentLOD._masterMesh = tempMaster;\n return this._billboardWorldMatrix;\n }\n return super.getWorldMatrix();\n }\n get isAnInstance() {\n return true;\n }\n /**\n * Returns the current associated LOD AbstractMesh.\n * @param camera\n */\n getLOD(camera) {\n if (!camera) {\n return this;\n }\n const sourceMeshLODLevels = this.sourceMesh.getLODLevels();\n if (!sourceMeshLODLevels || sourceMeshLODLevels.length === 0) {\n this._currentLOD = this.sourceMesh;\n }\n else {\n const boundingInfo = this.getBoundingInfo();\n this._currentLOD = this.sourceMesh.getLOD(camera, boundingInfo.boundingSphere);\n }\n return this._currentLOD;\n }\n /**\n * @internal\n */\n _preActivateForIntermediateRendering(renderId) {\n return this.sourceMesh._preActivateForIntermediateRendering(renderId);\n }\n /** @internal */\n _syncSubMeshes() {\n this.releaseSubMeshes();\n if (this._sourceMesh.subMeshes) {\n for (let index = 0; index < this._sourceMesh.subMeshes.length; index++) {\n this._sourceMesh.subMeshes[index].clone(this, this._sourceMesh);\n }\n }\n return this;\n }\n /** @internal */\n _generatePointsArray() {\n return this._sourceMesh._generatePointsArray();\n }\n /** @internal */\n _updateBoundingInfo() {\n if (this.hasBoundingInfo) {\n this.getBoundingInfo().update(this.worldMatrixFromCache);\n }\n else {\n this.buildBoundingInfo(this.absolutePosition, this.absolutePosition, this.worldMatrixFromCache);\n }\n this._updateSubMeshesBoundingInfo(this.worldMatrixFromCache);\n return this;\n }\n /**\n * Creates a new InstancedMesh from the current mesh.\n *\n * Returns the clone.\n * @param name the cloned mesh name\n * @param newParent the optional Node to parent the clone to.\n * @param doNotCloneChildren if `true` the model children aren't cloned.\n * @param newSourceMesh if set this mesh will be used as the source mesh instead of ths instance's one\n * @returns the clone\n */\n clone(name, newParent = null, doNotCloneChildren, newSourceMesh) {\n const result = (newSourceMesh || this._sourceMesh).createInstance(name);\n // Deep copy\n DeepCopier.DeepCopy(this, result, [\n \"name\",\n \"subMeshes\",\n \"uniqueId\",\n \"parent\",\n \"lightSources\",\n \"receiveShadows\",\n \"material\",\n \"visibility\",\n \"skeleton\",\n \"sourceMesh\",\n \"isAnInstance\",\n \"facetNb\",\n \"isFacetDataEnabled\",\n \"isBlocked\",\n \"useBones\",\n \"hasInstances\",\n \"collider\",\n \"edgesRenderer\",\n \"forward\",\n \"up\",\n \"right\",\n \"absolutePosition\",\n \"absoluteScaling\",\n \"absoluteRotationQuaternion\",\n \"isWorldMatrixFrozen\",\n \"nonUniformScaling\",\n \"behaviors\",\n \"worldMatrixFromCache\",\n \"hasThinInstances\",\n \"hasBoundingInfo\",\n ], []);\n // Bounding info\n this.refreshBoundingInfo();\n // Parent\n if (newParent) {\n result.parent = newParent;\n }\n if (!doNotCloneChildren) {\n // Children\n for (let index = 0; index < this.getScene().meshes.length; index++) {\n const mesh = this.getScene().meshes[index];\n if (mesh.parent === this) {\n mesh.clone(mesh.name, result);\n }\n }\n }\n result.computeWorldMatrix(true);\n this.onClonedObservable.notifyObservers(result);\n return result;\n }\n /**\n * Disposes the InstancedMesh.\n * Returns nothing.\n * @param doNotRecurse\n * @param disposeMaterialAndTextures\n */\n dispose(doNotRecurse, disposeMaterialAndTextures = false) {\n // Remove from mesh\n this._sourceMesh.removeInstance(this);\n super.dispose(doNotRecurse, disposeMaterialAndTextures);\n }\n /**\n * @internal\n */\n _serializeAsParent(serializationObject) {\n super._serializeAsParent(serializationObject);\n serializationObject.parentId = this._sourceMesh.uniqueId;\n serializationObject.parentInstanceIndex = this._indexInSourceMeshInstanceArray;\n }\n /**\n * Instantiate (when possible) or clone that node with its hierarchy\n * @param newParent defines the new parent to use for the instance (or clone)\n * @param options defines options to configure how copy is done\n * @param options.doNotInstantiate defines if the model must be instantiated or just cloned\n * @param options.newSourcedMesh newSourcedMesh the new source mesh for the instance (or clone)\n * @param onNewNodeCreated defines an option callback to call when a clone or an instance is created\n * @returns an instance (or a clone) of the current node with its hierarchy\n */\n instantiateHierarchy(newParent = null, options, onNewNodeCreated) {\n const clone = this.clone(\"Clone of \" + (this.name || this.id), newParent || this.parent, true, options && options.newSourcedMesh);\n if (clone) {\n if (onNewNodeCreated) {\n onNewNodeCreated(this, clone);\n }\n }\n for (const child of this.getChildTransformNodes(true)) {\n child.instantiateHierarchy(clone, options, onNewNodeCreated);\n }\n return clone;\n }\n }\n Mesh.prototype.registerInstancedBuffer = function (kind, stride) {\n var _a, _b;\n // Remove existing one\n (_b = (_a = this._userInstancedBuffersStorage) === null || _a === void 0 ? void 0 : _a.vertexBuffers[kind]) === null || _b === void 0 ? void 0 : _b.dispose();\n // Creates the instancedBuffer field if not present\n if (!this.instancedBuffers) {\n this.instancedBuffers = {};\n for (const instance of this.instances) {\n instance.instancedBuffers = {};\n }\n if (!this._userInstancedBuffersStorage) {\n this._userInstancedBuffersStorage = {\n data: {},\n vertexBuffers: {},\n strides: {},\n sizes: {},\n vertexArrayObjects: this.getEngine().getCaps().vertexArrayObject ? {} : undefined,\n };\n }\n }\n // Creates an empty property for this kind\n this.instancedBuffers[kind] = null;\n this._userInstancedBuffersStorage.strides[kind] = stride;\n this._userInstancedBuffersStorage.sizes[kind] = stride * 32; // Initial size\n this._userInstancedBuffersStorage.data[kind] = new Float32Array(this._userInstancedBuffersStorage.sizes[kind]);\n this._userInstancedBuffersStorage.vertexBuffers[kind] = new VertexBuffer(this.getEngine(), this._userInstancedBuffersStorage.data[kind], kind, true, false, stride, true);\n for (const instance of this.instances) {\n instance.instancedBuffers[kind] = null;\n }\n this._invalidateInstanceVertexArrayObject();\n this._markSubMeshesAsAttributesDirty();\n };\n Mesh.prototype._processInstancedBuffers = function (visibleInstances, renderSelf) {\n const instanceCount = visibleInstances ? visibleInstances.length : 0;\n for (const kind in this.instancedBuffers) {\n let size = this._userInstancedBuffersStorage.sizes[kind];\n const stride = this._userInstancedBuffersStorage.strides[kind];\n // Resize if required\n const expectedSize = (instanceCount + 1) * stride;\n while (size < expectedSize) {\n size *= 2;\n }\n if (this._userInstancedBuffersStorage.data[kind].length != size) {\n this._userInstancedBuffersStorage.data[kind] = new Float32Array(size);\n this._userInstancedBuffersStorage.sizes[kind] = size;\n if (this._userInstancedBuffersStorage.vertexBuffers[kind]) {\n this._userInstancedBuffersStorage.vertexBuffers[kind].dispose();\n this._userInstancedBuffersStorage.vertexBuffers[kind] = null;\n }\n }\n const data = this._userInstancedBuffersStorage.data[kind];\n // Update data buffer\n let offset = 0;\n if (renderSelf) {\n const value = this.instancedBuffers[kind];\n if (value.toArray) {\n value.toArray(data, offset);\n }\n else if (value.copyToArray) {\n value.copyToArray(data, offset);\n }\n else {\n data[offset] = value;\n }\n offset += stride;\n }\n for (let instanceIndex = 0; instanceIndex < instanceCount; instanceIndex++) {\n const instance = visibleInstances[instanceIndex];\n const value = instance.instancedBuffers[kind];\n if (value.toArray) {\n value.toArray(data, offset);\n }\n else if (value.copyToArray) {\n value.copyToArray(data, offset);\n }\n else {\n data[offset] = value;\n }\n offset += stride;\n }\n // Update vertex buffer\n if (!this._userInstancedBuffersStorage.vertexBuffers[kind]) {\n this._userInstancedBuffersStorage.vertexBuffers[kind] = new VertexBuffer(this.getEngine(), this._userInstancedBuffersStorage.data[kind], kind, true, false, stride, true);\n this._invalidateInstanceVertexArrayObject();\n }\n else {\n this._userInstancedBuffersStorage.vertexBuffers[kind].updateDirectly(data, 0);\n }\n }\n };\n Mesh.prototype._invalidateInstanceVertexArrayObject = function () {\n if (!this._userInstancedBuffersStorage || this._userInstancedBuffersStorage.vertexArrayObjects === undefined) {\n return;\n }\n for (const kind in this._userInstancedBuffersStorage.vertexArrayObjects) {\n this.getEngine().releaseVertexArrayObject(this._userInstancedBuffersStorage.vertexArrayObjects[kind]);\n }\n this._userInstancedBuffersStorage.vertexArrayObjects = {};\n };\n Mesh.prototype._disposeInstanceSpecificData = function () {\n if (this._instanceDataStorage.instancesBuffer) {\n this._instanceDataStorage.instancesBuffer.dispose();\n this._instanceDataStorage.instancesBuffer = null;\n }\n while (this.instances.length) {\n this.instances[0].dispose();\n }\n for (const kind in this.instancedBuffers) {\n if (this._userInstancedBuffersStorage.vertexBuffers[kind]) {\n this._userInstancedBuffersStorage.vertexBuffers[kind].dispose();\n }\n }\n this._invalidateInstanceVertexArrayObject();\n this.instancedBuffers = {};\n };\n\n const onCreatedEffectParameters$3 = { effect: null, subMesh: null };\n /**\n * The ShaderMaterial object has the necessary methods to pass data from your scene to the Vertex and Fragment Shaders and returns a material that can be applied to any mesh.\n *\n * This returned material effects how the mesh will look based on the code in the shaders.\n *\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/shaders/shaderMaterial\n */\n class ShaderMaterial extends PushMaterial {\n /**\n * Instantiate a new shader material.\n * The ShaderMaterial object has the necessary methods to pass data from your scene to the Vertex and Fragment Shaders and returns a material that can be applied to any mesh.\n * This returned material effects how the mesh will look based on the code in the shaders.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/shaders/shaderMaterial\n * @param name Define the name of the material in the scene\n * @param scene Define the scene the material belongs to\n * @param shaderPath Defines the route to the shader code in one of three ways:\n * * object: \\{ vertex: \"custom\", fragment: \"custom\" \\}, used with Effect.ShadersStore[\"customVertexShader\"] and Effect.ShadersStore[\"customFragmentShader\"]\n * * object: \\{ vertexElement: \"vertexShaderCode\", fragmentElement: \"fragmentShaderCode\" \\}, used with shader code in script tags\n * * object: \\{ vertexSource: \"vertex shader code string\", fragmentSource: \"fragment shader code string\" \\} using with strings containing the shaders code\n * * string: \"./COMMON_NAME\", used with external files COMMON_NAME.vertex.fx and COMMON_NAME.fragment.fx in index.html folder.\n * @param options Define the options used to create the shader\n * @param storeEffectOnSubMeshes true to store effect on submeshes, false to store the effect directly in the material class.\n */\n constructor(name, scene, shaderPath, options = {}, storeEffectOnSubMeshes = true) {\n super(name, scene, storeEffectOnSubMeshes);\n this._textures = {};\n this._textureArrays = {};\n this._externalTextures = {};\n this._floats = {};\n this._ints = {};\n this._uints = {};\n this._floatsArrays = {};\n this._colors3 = {};\n this._colors3Arrays = {};\n this._colors4 = {};\n this._colors4Arrays = {};\n this._vectors2 = {};\n this._vectors3 = {};\n this._vectors4 = {};\n this._quaternions = {};\n this._quaternionsArrays = {};\n this._matrices = {};\n this._matrixArrays = {};\n this._matrices3x3 = {};\n this._matrices2x2 = {};\n this._vectors2Arrays = {};\n this._vectors3Arrays = {};\n this._vectors4Arrays = {};\n this._uniformBuffers = {};\n this._textureSamplers = {};\n this._storageBuffers = {};\n this._cachedWorldViewMatrix = new Matrix();\n this._cachedWorldViewProjectionMatrix = new Matrix();\n this._multiview = false;\n /**\n * @internal\n */\n this._materialHelperNeedsPreviousMatrices = false;\n this._shaderPath = shaderPath;\n this._options = Object.assign({ needAlphaBlending: false, needAlphaTesting: false, attributes: [\"position\", \"normal\", \"uv\"], uniforms: [\"worldViewProjection\"], uniformBuffers: [], samplers: [], externalTextures: [], samplerObjects: [], storageBuffers: [], defines: [], useClipPlane: false }, options);\n }\n /**\n * Gets the shader path used to define the shader code\n * It can be modified to trigger a new compilation\n */\n get shaderPath() {\n return this._shaderPath;\n }\n /**\n * Sets the shader path used to define the shader code\n * It can be modified to trigger a new compilation\n */\n set shaderPath(shaderPath) {\n this._shaderPath = shaderPath;\n }\n /**\n * Gets the options used to compile the shader.\n * They can be modified to trigger a new compilation\n */\n get options() {\n return this._options;\n }\n /**\n * is multiview set to true?\n */\n get isMultiview() {\n return this._multiview;\n }\n /**\n * Gets the current class name of the material e.g. \"ShaderMaterial\"\n * Mainly use in serialization.\n * @returns the class name\n */\n getClassName() {\n return \"ShaderMaterial\";\n }\n /**\n * Specifies if the material will require alpha blending\n * @returns a boolean specifying if alpha blending is needed\n */\n needAlphaBlending() {\n return this.alpha < 1.0 || this._options.needAlphaBlending;\n }\n /**\n * Specifies if this material should be rendered in alpha test mode\n * @returns a boolean specifying if an alpha test is needed.\n */\n needAlphaTesting() {\n return this._options.needAlphaTesting;\n }\n _checkUniform(uniformName) {\n if (this._options.uniforms.indexOf(uniformName) === -1) {\n this._options.uniforms.push(uniformName);\n }\n }\n /**\n * Set a texture in the shader.\n * @param name Define the name of the uniform samplers as defined in the shader\n * @param texture Define the texture to bind to this sampler\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setTexture(name, texture) {\n if (this._options.samplers.indexOf(name) === -1) {\n this._options.samplers.push(name);\n }\n this._textures[name] = texture;\n return this;\n }\n /**\n * Set a texture array in the shader.\n * @param name Define the name of the uniform sampler array as defined in the shader\n * @param textures Define the list of textures to bind to this sampler\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setTextureArray(name, textures) {\n if (this._options.samplers.indexOf(name) === -1) {\n this._options.samplers.push(name);\n }\n this._checkUniform(name);\n this._textureArrays[name] = textures;\n return this;\n }\n /**\n * Set an internal texture in the shader.\n * @param name Define the name of the uniform samplers as defined in the shader\n * @param texture Define the texture to bind to this sampler\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setExternalTexture(name, texture) {\n if (this._options.externalTextures.indexOf(name) === -1) {\n this._options.externalTextures.push(name);\n }\n this._externalTextures[name] = texture;\n return this;\n }\n /**\n * Set a float in the shader.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setFloat(name, value) {\n this._checkUniform(name);\n this._floats[name] = value;\n return this;\n }\n /**\n * Set a int in the shader.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setInt(name, value) {\n this._checkUniform(name);\n this._ints[name] = value;\n return this;\n }\n /**\n * Set a unsigned int in the shader.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setUInt(name, value) {\n this._checkUniform(name);\n this._uints[name] = value;\n return this;\n }\n /**\n * Set an array of floats in the shader.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setFloats(name, value) {\n this._checkUniform(name);\n this._floatsArrays[name] = value;\n return this;\n }\n /**\n * Set a vec3 in the shader from a Color3.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setColor3(name, value) {\n this._checkUniform(name);\n this._colors3[name] = value;\n return this;\n }\n /**\n * Set a vec3 array in the shader from a Color3 array.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setColor3Array(name, value) {\n this._checkUniform(name);\n this._colors3Arrays[name] = value.reduce((arr, color) => {\n color.toArray(arr, arr.length);\n return arr;\n }, []);\n return this;\n }\n /**\n * Set a vec4 in the shader from a Color4.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setColor4(name, value) {\n this._checkUniform(name);\n this._colors4[name] = value;\n return this;\n }\n /**\n * Set a vec4 array in the shader from a Color4 array.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setColor4Array(name, value) {\n this._checkUniform(name);\n this._colors4Arrays[name] = value.reduce((arr, color) => {\n color.toArray(arr, arr.length);\n return arr;\n }, []);\n return this;\n }\n /**\n * Set a vec2 in the shader from a Vector2.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setVector2(name, value) {\n this._checkUniform(name);\n this._vectors2[name] = value;\n return this;\n }\n /**\n * Set a vec3 in the shader from a Vector3.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setVector3(name, value) {\n this._checkUniform(name);\n this._vectors3[name] = value;\n return this;\n }\n /**\n * Set a vec4 in the shader from a Vector4.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setVector4(name, value) {\n this._checkUniform(name);\n this._vectors4[name] = value;\n return this;\n }\n /**\n * Set a vec4 in the shader from a Quaternion.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setQuaternion(name, value) {\n this._checkUniform(name);\n this._quaternions[name] = value;\n return this;\n }\n /**\n * Set a vec4 array in the shader from a Quaternion array.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setQuaternionArray(name, value) {\n this._checkUniform(name);\n this._quaternionsArrays[name] = value.reduce((arr, quaternion) => {\n quaternion.toArray(arr, arr.length);\n return arr;\n }, []);\n return this;\n }\n /**\n * Set a mat4 in the shader from a Matrix.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setMatrix(name, value) {\n this._checkUniform(name);\n this._matrices[name] = value;\n return this;\n }\n /**\n * Set a float32Array in the shader from a matrix array.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setMatrices(name, value) {\n this._checkUniform(name);\n const float32Array = new Float32Array(value.length * 16);\n for (let index = 0; index < value.length; index++) {\n const matrix = value[index];\n matrix.copyToArray(float32Array, index * 16);\n }\n this._matrixArrays[name] = float32Array;\n return this;\n }\n /**\n * Set a mat3 in the shader from a Float32Array.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setMatrix3x3(name, value) {\n this._checkUniform(name);\n this._matrices3x3[name] = value;\n return this;\n }\n /**\n * Set a mat2 in the shader from a Float32Array.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setMatrix2x2(name, value) {\n this._checkUniform(name);\n this._matrices2x2[name] = value;\n return this;\n }\n /**\n * Set a vec2 array in the shader from a number array.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setArray2(name, value) {\n this._checkUniform(name);\n this._vectors2Arrays[name] = value;\n return this;\n }\n /**\n * Set a vec3 array in the shader from a number array.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setArray3(name, value) {\n this._checkUniform(name);\n this._vectors3Arrays[name] = value;\n return this;\n }\n /**\n * Set a vec4 array in the shader from a number array.\n * @param name Define the name of the uniform as defined in the shader\n * @param value Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setArray4(name, value) {\n this._checkUniform(name);\n this._vectors4Arrays[name] = value;\n return this;\n }\n /**\n * Set a uniform buffer in the shader\n * @param name Define the name of the uniform as defined in the shader\n * @param buffer Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setUniformBuffer(name, buffer) {\n if (this._options.uniformBuffers.indexOf(name) === -1) {\n this._options.uniformBuffers.push(name);\n }\n this._uniformBuffers[name] = buffer;\n return this;\n }\n /**\n * Set a texture sampler in the shader\n * @param name Define the name of the uniform as defined in the shader\n * @param sampler Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setTextureSampler(name, sampler) {\n if (this._options.samplerObjects.indexOf(name) === -1) {\n this._options.samplerObjects.push(name);\n }\n this._textureSamplers[name] = sampler;\n return this;\n }\n /**\n * Set a storage buffer in the shader\n * @param name Define the name of the storage buffer as defined in the shader\n * @param buffer Define the value to give to the uniform\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setStorageBuffer(name, buffer) {\n if (this._options.storageBuffers.indexOf(name) === -1) {\n this._options.storageBuffers.push(name);\n }\n this._storageBuffers[name] = buffer;\n return this;\n }\n /**\n * Adds, removes, or replaces the specified shader define and value.\n * * setDefine(\"MY_DEFINE\", true); // enables a boolean define\n * * setDefine(\"MY_DEFINE\", \"0.5\"); // adds \"#define MY_DEFINE 0.5\" to the shader (or sets and replaces the value of any existing define with that name)\n * * setDefine(\"MY_DEFINE\", false); // disables and removes the define\n * Note if the active defines do change, the shader will be recompiled and this can be expensive.\n * @param define the define name e.g., \"OUTPUT_TO_SRGB\" or \"#define OUTPUT_TO_SRGB\". If the define was passed into the constructor already, the version used should match that, and in either case, it should not include any appended value.\n * @param value either the value of the define (e.g. a numerical value) or for booleans, true if the define should be enabled or false if it should be disabled\n * @returns the material itself allowing \"fluent\" like uniform updates\n */\n setDefine(define, value) {\n // First remove any existing define with this name.\n const defineName = define.trimEnd() + \" \";\n const existingDefineIdx = this.options.defines.findIndex((x) => x === define || x.startsWith(defineName));\n if (existingDefineIdx >= 0) {\n this.options.defines.splice(existingDefineIdx, 1);\n }\n // Then add the new define value. (If it's a boolean value and false, don't add it.)\n if (typeof value !== \"boolean\" || value) {\n this.options.defines.push(defineName + value);\n }\n return this;\n }\n /**\n * Specifies that the submesh is ready to be used\n * @param mesh defines the mesh to check\n * @param subMesh defines which submesh to check\n * @param useInstances specifies that instances should be used\n * @returns a boolean indicating that the submesh is ready or not\n */\n isReadyForSubMesh(mesh, subMesh, useInstances) {\n return this.isReady(mesh, useInstances, subMesh);\n }\n /**\n * Checks if the material is ready to render the requested mesh\n * @param mesh Define the mesh to render\n * @param useInstances Define whether or not the material is used with instances\n * @param subMesh defines which submesh to render\n * @returns true if ready, otherwise false\n */\n isReady(mesh, useInstances, subMesh) {\n var _a, _b, _c, _d;\n const storeEffectOnSubMeshes = subMesh && this._storeEffectOnSubMeshes;\n if (this.isFrozen) {\n if (storeEffectOnSubMeshes) {\n if (subMesh.effect && subMesh.effect._wasPreviouslyReady) {\n return true;\n }\n }\n else {\n const effect = this._drawWrapper.effect;\n if (effect && effect._wasPreviouslyReady && effect._wasPreviouslyUsingInstances === useInstances) {\n return true;\n }\n }\n }\n const scene = this.getScene();\n const engine = scene.getEngine();\n // Instances\n const defines = [];\n const attribs = [];\n const fallbacks = new EffectFallbacks();\n let shaderName = this._shaderPath, uniforms = this._options.uniforms, uniformBuffers = this._options.uniformBuffers, samplers = this._options.samplers;\n // global multiview\n if (engine.getCaps().multiview && scene.activeCamera && scene.activeCamera.outputRenderTarget && scene.activeCamera.outputRenderTarget.getViewCount() > 1) {\n this._multiview = true;\n defines.push(\"#define MULTIVIEW\");\n if (this._options.uniforms.indexOf(\"viewProjection\") !== -1 && this._options.uniforms.indexOf(\"viewProjectionR\") === -1) {\n this._options.uniforms.push(\"viewProjectionR\");\n }\n }\n for (let index = 0; index < this._options.defines.length; index++) {\n const defineToAdd = this._options.defines[index].indexOf(\"#define\") === 0 ? this._options.defines[index] : `#define ${this._options.defines[index]}`;\n defines.push(defineToAdd);\n }\n for (let index = 0; index < this._options.attributes.length; index++) {\n attribs.push(this._options.attributes[index]);\n }\n if (mesh && mesh.isVerticesDataPresent(VertexBuffer.ColorKind)) {\n attribs.push(VertexBuffer.ColorKind);\n defines.push(\"#define VERTEXCOLOR\");\n }\n if (useInstances) {\n defines.push(\"#define INSTANCES\");\n MaterialHelper.PushAttributesForInstances(attribs, this._materialHelperNeedsPreviousMatrices);\n if (mesh === null || mesh === void 0 ? void 0 : mesh.hasThinInstances) {\n defines.push(\"#define THIN_INSTANCES\");\n if (mesh && mesh.isVerticesDataPresent(VertexBuffer.ColorInstanceKind)) {\n attribs.push(VertexBuffer.ColorInstanceKind);\n defines.push(\"#define INSTANCESCOLOR\");\n }\n }\n }\n // Bones\n if (mesh && mesh.useBones && mesh.computeBonesUsingShaders && mesh.skeleton) {\n attribs.push(VertexBuffer.MatricesIndicesKind);\n attribs.push(VertexBuffer.MatricesWeightsKind);\n if (mesh.numBoneInfluencers > 4) {\n attribs.push(VertexBuffer.MatricesIndicesExtraKind);\n attribs.push(VertexBuffer.MatricesWeightsExtraKind);\n }\n const skeleton = mesh.skeleton;\n defines.push(\"#define NUM_BONE_INFLUENCERS \" + mesh.numBoneInfluencers);\n fallbacks.addCPUSkinningFallback(0, mesh);\n if (skeleton.isUsingTextureForMatrices) {\n defines.push(\"#define BONETEXTURE\");\n if (this._options.uniforms.indexOf(\"boneTextureWidth\") === -1) {\n this._options.uniforms.push(\"boneTextureWidth\");\n }\n if (this._options.samplers.indexOf(\"boneSampler\") === -1) {\n this._options.samplers.push(\"boneSampler\");\n }\n }\n else {\n defines.push(\"#define BonesPerMesh \" + (skeleton.bones.length + 1));\n if (this._options.uniforms.indexOf(\"mBones\") === -1) {\n this._options.uniforms.push(\"mBones\");\n }\n }\n }\n else {\n defines.push(\"#define NUM_BONE_INFLUENCERS 0\");\n }\n // Morph\n let numInfluencers = 0;\n const manager = mesh ? mesh.morphTargetManager : null;\n if (manager) {\n const uv = manager.supportsUVs && defines.indexOf(\"#define UV1\") !== -1;\n const tangent = manager.supportsTangents && defines.indexOf(\"#define TANGENT\") !== -1;\n const normal = manager.supportsNormals && defines.indexOf(\"#define NORMAL\") !== -1;\n numInfluencers = manager.numInfluencers;\n if (uv) {\n defines.push(\"#define MORPHTARGETS_UV\");\n }\n if (tangent) {\n defines.push(\"#define MORPHTARGETS_TANGENT\");\n }\n if (normal) {\n defines.push(\"#define MORPHTARGETS_NORMAL\");\n }\n if (numInfluencers > 0) {\n defines.push(\"#define MORPHTARGETS\");\n }\n if (manager.isUsingTextureForTargets) {\n defines.push(\"#define MORPHTARGETS_TEXTURE\");\n if (this._options.uniforms.indexOf(\"morphTargetTextureIndices\") === -1) {\n this._options.uniforms.push(\"morphTargetTextureIndices\");\n }\n if (this._options.samplers.indexOf(\"morphTargets\") === -1) {\n this._options.samplers.push(\"morphTargets\");\n }\n }\n defines.push(\"#define NUM_MORPH_INFLUENCERS \" + numInfluencers);\n for (let index = 0; index < numInfluencers; index++) {\n attribs.push(VertexBuffer.PositionKind + index);\n if (normal) {\n attribs.push(VertexBuffer.NormalKind + index);\n }\n if (tangent) {\n attribs.push(VertexBuffer.TangentKind + index);\n }\n if (uv) {\n attribs.push(VertexBuffer.UVKind + \"_\" + index);\n }\n }\n if (numInfluencers > 0) {\n uniforms = uniforms.slice();\n uniforms.push(\"morphTargetInfluences\");\n uniforms.push(\"morphTargetTextureInfo\");\n uniforms.push(\"morphTargetTextureIndices\");\n }\n }\n else {\n defines.push(\"#define NUM_MORPH_INFLUENCERS 0\");\n }\n // Baked Vertex Animation\n if (mesh) {\n const bvaManager = mesh.bakedVertexAnimationManager;\n if (bvaManager && bvaManager.isEnabled) {\n defines.push(\"#define BAKED_VERTEX_ANIMATION_TEXTURE\");\n if (this._options.uniforms.indexOf(\"bakedVertexAnimationSettings\") === -1) {\n this._options.uniforms.push(\"bakedVertexAnimationSettings\");\n }\n if (this._options.uniforms.indexOf(\"bakedVertexAnimationTextureSizeInverted\") === -1) {\n this._options.uniforms.push(\"bakedVertexAnimationTextureSizeInverted\");\n }\n if (this._options.uniforms.indexOf(\"bakedVertexAnimationTime\") === -1) {\n this._options.uniforms.push(\"bakedVertexAnimationTime\");\n }\n if (this._options.samplers.indexOf(\"bakedVertexAnimationTexture\") === -1) {\n this._options.samplers.push(\"bakedVertexAnimationTexture\");\n }\n }\n MaterialHelper.PrepareAttributesForBakedVertexAnimation(attribs, mesh, defines);\n }\n // Textures\n for (const name in this._textures) {\n if (!this._textures[name].isReady()) {\n return false;\n }\n }\n // Alpha test\n if (mesh && this._shouldTurnAlphaTestOn(mesh)) {\n defines.push(\"#define ALPHATEST\");\n }\n // Clip planes\n if (this._options.useClipPlane !== false) {\n addClipPlaneUniforms(uniforms);\n prepareStringDefinesForClipPlanes(this, scene, defines);\n }\n if (this.customShaderNameResolve) {\n uniforms = uniforms.slice();\n uniformBuffers = uniformBuffers.slice();\n samplers = samplers.slice();\n shaderName = this.customShaderNameResolve(shaderName, uniforms, uniformBuffers, samplers, defines, attribs);\n }\n const drawWrapper = storeEffectOnSubMeshes ? subMesh._getDrawWrapper() : this._drawWrapper;\n const previousEffect = (_a = drawWrapper === null || drawWrapper === void 0 ? void 0 : drawWrapper.effect) !== null && _a !== void 0 ? _a : null;\n const previousDefines = (_b = drawWrapper === null || drawWrapper === void 0 ? void 0 : drawWrapper.defines) !== null && _b !== void 0 ? _b : null;\n const join = defines.join(\"\\n\");\n let effect = previousEffect;\n if (previousDefines !== join) {\n effect = engine.createEffect(shaderName, {\n attributes: attribs,\n uniformsNames: uniforms,\n uniformBuffersNames: uniformBuffers,\n samplers: samplers,\n defines: join,\n fallbacks: fallbacks,\n onCompiled: this.onCompiled,\n onError: this.onError,\n indexParameters: { maxSimultaneousMorphTargets: numInfluencers },\n shaderLanguage: this._options.shaderLanguage,\n }, engine);\n if (storeEffectOnSubMeshes) {\n subMesh.setEffect(effect, join, this._materialContext);\n }\n else if (drawWrapper) {\n drawWrapper.setEffect(effect, join);\n }\n if (this._onEffectCreatedObservable) {\n onCreatedEffectParameters$3.effect = effect;\n onCreatedEffectParameters$3.subMesh = (_c = subMesh !== null && subMesh !== void 0 ? subMesh : mesh === null || mesh === void 0 ? void 0 : mesh.subMeshes[0]) !== null && _c !== void 0 ? _c : null;\n this._onEffectCreatedObservable.notifyObservers(onCreatedEffectParameters$3);\n }\n }\n effect._wasPreviouslyUsingInstances = !!useInstances;\n if ((_d = !(effect === null || effect === void 0 ? void 0 : effect.isReady())) !== null && _d !== void 0 ? _d : true) {\n return false;\n }\n if (previousEffect !== effect) {\n scene.resetCachedMaterial();\n }\n effect._wasPreviouslyReady = true;\n return true;\n }\n /**\n * Binds the world matrix to the material\n * @param world defines the world transformation matrix\n * @param effectOverride - If provided, use this effect instead of internal effect\n */\n bindOnlyWorldMatrix(world, effectOverride) {\n const scene = this.getScene();\n const effect = effectOverride !== null && effectOverride !== void 0 ? effectOverride : this.getEffect();\n if (!effect) {\n return;\n }\n if (this._options.uniforms.indexOf(\"world\") !== -1) {\n effect.setMatrix(\"world\", world);\n }\n if (this._options.uniforms.indexOf(\"worldView\") !== -1) {\n world.multiplyToRef(scene.getViewMatrix(), this._cachedWorldViewMatrix);\n effect.setMatrix(\"worldView\", this._cachedWorldViewMatrix);\n }\n if (this._options.uniforms.indexOf(\"worldViewProjection\") !== -1) {\n world.multiplyToRef(scene.getTransformMatrix(), this._cachedWorldViewProjectionMatrix);\n effect.setMatrix(\"worldViewProjection\", this._cachedWorldViewProjectionMatrix);\n }\n }\n /**\n * Binds the submesh to this material by preparing the effect and shader to draw\n * @param world defines the world transformation matrix\n * @param mesh defines the mesh containing the submesh\n * @param subMesh defines the submesh to bind the material to\n */\n bindForSubMesh(world, mesh, subMesh) {\n var _a;\n this.bind(world, mesh, (_a = subMesh._drawWrapperOverride) === null || _a === void 0 ? void 0 : _a.effect, subMesh);\n }\n /**\n * Binds the material to the mesh\n * @param world defines the world transformation matrix\n * @param mesh defines the mesh to bind the material to\n * @param effectOverride - If provided, use this effect instead of internal effect\n * @param subMesh defines the submesh to bind the material to\n */\n bind(world, mesh, effectOverride, subMesh) {\n var _a;\n // Std values\n const storeEffectOnSubMeshes = subMesh && this._storeEffectOnSubMeshes;\n const effect = effectOverride !== null && effectOverride !== void 0 ? effectOverride : (storeEffectOnSubMeshes ? subMesh.effect : this.getEffect());\n if (!effect) {\n return;\n }\n this._activeEffect = effect;\n this.bindOnlyWorldMatrix(world, effectOverride);\n const uniformBuffers = this._options.uniformBuffers;\n let useSceneUBO = false;\n if (effect && uniformBuffers && uniformBuffers.length > 0 && this.getScene().getEngine().supportsUniformBuffers) {\n for (let i = 0; i < uniformBuffers.length; ++i) {\n const bufferName = uniformBuffers[i];\n switch (bufferName) {\n case \"Mesh\":\n if (mesh) {\n mesh.getMeshUniformBuffer().bindToEffect(effect, \"Mesh\");\n mesh.transferToEffect(world);\n }\n break;\n case \"Scene\":\n MaterialHelper.BindSceneUniformBuffer(effect, this.getScene().getSceneUniformBuffer());\n this.getScene().finalizeSceneUbo();\n useSceneUBO = true;\n break;\n }\n }\n }\n const mustRebind = mesh && storeEffectOnSubMeshes ? this._mustRebind(this.getScene(), effect, mesh.visibility) : this.getScene().getCachedMaterial() !== this;\n if (effect && mustRebind) {\n if (!useSceneUBO && this._options.uniforms.indexOf(\"view\") !== -1) {\n effect.setMatrix(\"view\", this.getScene().getViewMatrix());\n }\n if (!useSceneUBO && this._options.uniforms.indexOf(\"projection\") !== -1) {\n effect.setMatrix(\"projection\", this.getScene().getProjectionMatrix());\n }\n if (!useSceneUBO && this._options.uniforms.indexOf(\"viewProjection\") !== -1) {\n effect.setMatrix(\"viewProjection\", this.getScene().getTransformMatrix());\n if (this._multiview) {\n effect.setMatrix(\"viewProjectionR\", this.getScene()._transformMatrixR);\n }\n }\n if (this.getScene().activeCamera && this._options.uniforms.indexOf(\"cameraPosition\") !== -1) {\n effect.setVector3(\"cameraPosition\", this.getScene().activeCamera.globalPosition);\n }\n // Bones\n MaterialHelper.BindBonesParameters(mesh, effect);\n // Clip plane\n bindClipPlane(effect, this, this.getScene());\n let name;\n // Texture\n for (name in this._textures) {\n effect.setTexture(name, this._textures[name]);\n }\n // Texture arrays\n for (name in this._textureArrays) {\n effect.setTextureArray(name, this._textureArrays[name]);\n }\n // External texture\n for (name in this._externalTextures) {\n effect.setExternalTexture(name, this._externalTextures[name]);\n }\n // Int\n for (name in this._ints) {\n effect.setInt(name, this._ints[name]);\n }\n // UInt\n for (name in this._uints) {\n effect.setUInt(name, this._uints[name]);\n }\n // Float\n for (name in this._floats) {\n effect.setFloat(name, this._floats[name]);\n }\n // Floats\n for (name in this._floatsArrays) {\n effect.setArray(name, this._floatsArrays[name]);\n }\n // Color3\n for (name in this._colors3) {\n effect.setColor3(name, this._colors3[name]);\n }\n // Color3Array\n for (name in this._colors3Arrays) {\n effect.setArray3(name, this._colors3Arrays[name]);\n }\n // Color4\n for (name in this._colors4) {\n const color = this._colors4[name];\n effect.setFloat4(name, color.r, color.g, color.b, color.a);\n }\n // Color4Array\n for (name in this._colors4Arrays) {\n effect.setArray4(name, this._colors4Arrays[name]);\n }\n // Vector2\n for (name in this._vectors2) {\n effect.setVector2(name, this._vectors2[name]);\n }\n // Vector3\n for (name in this._vectors3) {\n effect.setVector3(name, this._vectors3[name]);\n }\n // Vector4\n for (name in this._vectors4) {\n effect.setVector4(name, this._vectors4[name]);\n }\n // Quaternion\n for (name in this._quaternions) {\n effect.setQuaternion(name, this._quaternions[name]);\n }\n // Matrix\n for (name in this._matrices) {\n effect.setMatrix(name, this._matrices[name]);\n }\n // MatrixArray\n for (name in this._matrixArrays) {\n effect.setMatrices(name, this._matrixArrays[name]);\n }\n // Matrix 3x3\n for (name in this._matrices3x3) {\n effect.setMatrix3x3(name, this._matrices3x3[name]);\n }\n // Matrix 2x2\n for (name in this._matrices2x2) {\n effect.setMatrix2x2(name, this._matrices2x2[name]);\n }\n // Vector2Array\n for (name in this._vectors2Arrays) {\n effect.setArray2(name, this._vectors2Arrays[name]);\n }\n // Vector3Array\n for (name in this._vectors3Arrays) {\n effect.setArray3(name, this._vectors3Arrays[name]);\n }\n // Vector4Array\n for (name in this._vectors4Arrays) {\n effect.setArray4(name, this._vectors4Arrays[name]);\n }\n // QuaternionArray\n for (name in this._quaternionsArrays) {\n effect.setArray4(name, this._quaternionsArrays[name]);\n }\n // Uniform buffers\n for (name in this._uniformBuffers) {\n const buffer = this._uniformBuffers[name].getBuffer();\n if (buffer) {\n effect.bindUniformBuffer(buffer, name);\n }\n }\n // Samplers\n for (name in this._textureSamplers) {\n effect.setTextureSampler(name, this._textureSamplers[name]);\n }\n // Storage buffers\n for (name in this._storageBuffers) {\n effect.setStorageBuffer(name, this._storageBuffers[name]);\n }\n }\n if (effect && mesh && (mustRebind || !this.isFrozen)) {\n // Morph targets\n const manager = mesh.morphTargetManager;\n if (manager && manager.numInfluencers > 0) {\n MaterialHelper.BindMorphTargetParameters(mesh, effect);\n }\n const bvaManager = mesh.bakedVertexAnimationManager;\n if (bvaManager && bvaManager.isEnabled) {\n (_a = mesh.bakedVertexAnimationManager) === null || _a === void 0 ? void 0 : _a.bind(effect, !!effect._wasPreviouslyUsingInstances);\n }\n }\n this._afterBind(mesh, effect);\n }\n /**\n * Gets the active textures from the material\n * @returns an array of textures\n */\n getActiveTextures() {\n const activeTextures = super.getActiveTextures();\n for (const name in this._textures) {\n activeTextures.push(this._textures[name]);\n }\n for (const name in this._textureArrays) {\n const array = this._textureArrays[name];\n for (let index = 0; index < array.length; index++) {\n activeTextures.push(array[index]);\n }\n }\n return activeTextures;\n }\n /**\n * Specifies if the material uses a texture\n * @param texture defines the texture to check against the material\n * @returns a boolean specifying if the material uses the texture\n */\n hasTexture(texture) {\n if (super.hasTexture(texture)) {\n return true;\n }\n for (const name in this._textures) {\n if (this._textures[name] === texture) {\n return true;\n }\n }\n for (const name in this._textureArrays) {\n const array = this._textureArrays[name];\n for (let index = 0; index < array.length; index++) {\n if (array[index] === texture) {\n return true;\n }\n }\n }\n return false;\n }\n /**\n * Makes a duplicate of the material, and gives it a new name\n * @param name defines the new name for the duplicated material\n * @returns the cloned material\n */\n clone(name) {\n const result = SerializationHelper.Clone(() => new ShaderMaterial(name, this.getScene(), this._shaderPath, this._options, this._storeEffectOnSubMeshes), this);\n result.name = name;\n result.id = name;\n // Shader code path\n if (typeof result._shaderPath === \"object\") {\n result._shaderPath = Object.assign({}, result._shaderPath);\n }\n // Options\n this._options = Object.assign({}, this._options);\n Object.keys(this._options).forEach((propName) => {\n const propValue = this._options[propName];\n if (Array.isArray(propValue)) {\n this._options[propName] = propValue.slice(0);\n }\n });\n // Stencil\n this.stencil.copyTo(result.stencil);\n // Texture\n for (const key in this._textures) {\n result.setTexture(key, this._textures[key]);\n }\n // TextureArray\n for (const key in this._textureArrays) {\n result.setTextureArray(key, this._textureArrays[key]);\n }\n // External texture\n for (const key in this._externalTextures) {\n result.setExternalTexture(key, this._externalTextures[key]);\n }\n // Int\n for (const key in this._ints) {\n result.setInt(key, this._ints[key]);\n }\n // UInt\n for (const key in this._uints) {\n result.setUInt(key, this._uints[key]);\n }\n // Float\n for (const key in this._floats) {\n result.setFloat(key, this._floats[key]);\n }\n // Floats\n for (const key in this._floatsArrays) {\n result.setFloats(key, this._floatsArrays[key]);\n }\n // Color3\n for (const key in this._colors3) {\n result.setColor3(key, this._colors3[key]);\n }\n // Color3Array\n for (const key in this._colors3Arrays) {\n result._colors3Arrays[key] = this._colors3Arrays[key];\n }\n // Color4\n for (const key in this._colors4) {\n result.setColor4(key, this._colors4[key]);\n }\n // Color4Array\n for (const key in this._colors4Arrays) {\n result._colors4Arrays[key] = this._colors4Arrays[key];\n }\n // Vector2\n for (const key in this._vectors2) {\n result.setVector2(key, this._vectors2[key]);\n }\n // Vector3\n for (const key in this._vectors3) {\n result.setVector3(key, this._vectors3[key]);\n }\n // Vector4\n for (const key in this._vectors4) {\n result.setVector4(key, this._vectors4[key]);\n }\n // Quaternion\n for (const key in this._quaternions) {\n result.setQuaternion(key, this._quaternions[key]);\n }\n // QuaternionArray\n for (const key in this._quaternionsArrays) {\n result._quaternionsArrays[key] = this._quaternionsArrays[key];\n }\n // Matrix\n for (const key in this._matrices) {\n result.setMatrix(key, this._matrices[key]);\n }\n // MatrixArray\n for (const key in this._matrixArrays) {\n result._matrixArrays[key] = this._matrixArrays[key].slice();\n }\n // Matrix 3x3\n for (const key in this._matrices3x3) {\n result.setMatrix3x3(key, this._matrices3x3[key]);\n }\n // Matrix 2x2\n for (const key in this._matrices2x2) {\n result.setMatrix2x2(key, this._matrices2x2[key]);\n }\n // Vector2Array\n for (const key in this._vectors2Arrays) {\n result.setArray2(key, this._vectors2Arrays[key]);\n }\n // Vector3Array\n for (const key in this._vectors3Arrays) {\n result.setArray3(key, this._vectors3Arrays[key]);\n }\n // Vector4Array\n for (const key in this._vectors4Arrays) {\n result.setArray4(key, this._vectors4Arrays[key]);\n }\n // Uniform buffers\n for (const key in this._uniformBuffers) {\n result.setUniformBuffer(key, this._uniformBuffers[key]);\n }\n // Samplers\n for (const key in this._textureSamplers) {\n result.setTextureSampler(key, this._textureSamplers[key]);\n }\n // Storag buffers\n for (const key in this._storageBuffers) {\n result.setStorageBuffer(key, this._storageBuffers[key]);\n }\n return result;\n }\n /**\n * Disposes the material\n * @param forceDisposeEffect specifies if effects should be forcefully disposed\n * @param forceDisposeTextures specifies if textures should be forcefully disposed\n * @param notBoundToMesh specifies if the material that is being disposed is known to be not bound to any mesh\n */\n dispose(forceDisposeEffect, forceDisposeTextures, notBoundToMesh) {\n if (forceDisposeTextures) {\n let name;\n for (name in this._textures) {\n this._textures[name].dispose();\n }\n for (name in this._textureArrays) {\n const array = this._textureArrays[name];\n for (let index = 0; index < array.length; index++) {\n array[index].dispose();\n }\n }\n }\n this._textures = {};\n super.dispose(forceDisposeEffect, forceDisposeTextures, notBoundToMesh);\n }\n /**\n * Serializes this material in a JSON representation\n * @returns the serialized material object\n */\n serialize() {\n const serializationObject = SerializationHelper.Serialize(this);\n serializationObject.customType = \"BABYLON.ShaderMaterial\";\n serializationObject.uniqueId = this.uniqueId;\n serializationObject.options = this._options;\n serializationObject.shaderPath = this._shaderPath;\n serializationObject.storeEffectOnSubMeshes = this._storeEffectOnSubMeshes;\n let name;\n // Stencil\n serializationObject.stencil = this.stencil.serialize();\n // Texture\n serializationObject.textures = {};\n for (name in this._textures) {\n serializationObject.textures[name] = this._textures[name].serialize();\n }\n // Texture arrays\n serializationObject.textureArrays = {};\n for (name in this._textureArrays) {\n serializationObject.textureArrays[name] = [];\n const array = this._textureArrays[name];\n for (let index = 0; index < array.length; index++) {\n serializationObject.textureArrays[name].push(array[index].serialize());\n }\n }\n // Int\n serializationObject.ints = {};\n for (name in this._ints) {\n serializationObject.ints[name] = this._ints[name];\n }\n // UInt\n serializationObject.uints = {};\n for (name in this._uints) {\n serializationObject.uints[name] = this._uints[name];\n }\n // Float\n serializationObject.floats = {};\n for (name in this._floats) {\n serializationObject.floats[name] = this._floats[name];\n }\n // Floats\n serializationObject.FloatArrays = {};\n for (name in this._floatsArrays) {\n serializationObject.FloatArrays[name] = this._floatsArrays[name];\n }\n // Color3\n serializationObject.colors3 = {};\n for (name in this._colors3) {\n serializationObject.colors3[name] = this._colors3[name].asArray();\n }\n // Color3 array\n serializationObject.colors3Arrays = {};\n for (name in this._colors3Arrays) {\n serializationObject.colors3Arrays[name] = this._colors3Arrays[name];\n }\n // Color4\n serializationObject.colors4 = {};\n for (name in this._colors4) {\n serializationObject.colors4[name] = this._colors4[name].asArray();\n }\n // Color4 array\n serializationObject.colors4Arrays = {};\n for (name in this._colors4Arrays) {\n serializationObject.colors4Arrays[name] = this._colors4Arrays[name];\n }\n // Vector2\n serializationObject.vectors2 = {};\n for (name in this._vectors2) {\n serializationObject.vectors2[name] = this._vectors2[name].asArray();\n }\n // Vector3\n serializationObject.vectors3 = {};\n for (name in this._vectors3) {\n serializationObject.vectors3[name] = this._vectors3[name].asArray();\n }\n // Vector4\n serializationObject.vectors4 = {};\n for (name in this._vectors4) {\n serializationObject.vectors4[name] = this._vectors4[name].asArray();\n }\n // Quaternion\n serializationObject.quaternions = {};\n for (name in this._quaternions) {\n serializationObject.quaternions[name] = this._quaternions[name].asArray();\n }\n // Matrix\n serializationObject.matrices = {};\n for (name in this._matrices) {\n serializationObject.matrices[name] = this._matrices[name].asArray();\n }\n // MatrixArray\n serializationObject.matrixArray = {};\n for (name in this._matrixArrays) {\n serializationObject.matrixArray[name] = this._matrixArrays[name];\n }\n // Matrix 3x3\n serializationObject.matrices3x3 = {};\n for (name in this._matrices3x3) {\n serializationObject.matrices3x3[name] = this._matrices3x3[name];\n }\n // Matrix 2x2\n serializationObject.matrices2x2 = {};\n for (name in this._matrices2x2) {\n serializationObject.matrices2x2[name] = this._matrices2x2[name];\n }\n // Vector2Array\n serializationObject.vectors2Arrays = {};\n for (name in this._vectors2Arrays) {\n serializationObject.vectors2Arrays[name] = this._vectors2Arrays[name];\n }\n // Vector3Array\n serializationObject.vectors3Arrays = {};\n for (name in this._vectors3Arrays) {\n serializationObject.vectors3Arrays[name] = this._vectors3Arrays[name];\n }\n // Vector4Array\n serializationObject.vectors4Arrays = {};\n for (name in this._vectors4Arrays) {\n serializationObject.vectors4Arrays[name] = this._vectors4Arrays[name];\n }\n // QuaternionArray\n serializationObject.quaternionsArrays = {};\n for (name in this._quaternionsArrays) {\n serializationObject.quaternionsArrays[name] = this._quaternionsArrays[name];\n }\n return serializationObject;\n }\n /**\n * Creates a shader material from parsed shader material data\n * @param source defines the JSON representation of the material\n * @param scene defines the hosting scene\n * @param rootUrl defines the root URL to use to load textures and relative dependencies\n * @returns a new material\n */\n static Parse(source, scene, rootUrl) {\n const material = SerializationHelper.Parse(() => new ShaderMaterial(source.name, scene, source.shaderPath, source.options, source.storeEffectOnSubMeshes), source, scene, rootUrl);\n let name;\n // Stencil\n if (source.stencil) {\n material.stencil.parse(source.stencil, scene, rootUrl);\n }\n // Texture\n for (name in source.textures) {\n material.setTexture(name, Texture.Parse(source.textures[name], scene, rootUrl));\n }\n // Texture arrays\n for (name in source.textureArrays) {\n const array = source.textureArrays[name];\n const textureArray = new Array();\n for (let index = 0; index < array.length; index++) {\n textureArray.push(Texture.Parse(array[index], scene, rootUrl));\n }\n material.setTextureArray(name, textureArray);\n }\n // Int\n for (name in source.ints) {\n material.setInt(name, source.ints[name]);\n }\n // UInt\n for (name in source.uints) {\n material.setUInt(name, source.uints[name]);\n }\n // Float\n for (name in source.floats) {\n material.setFloat(name, source.floats[name]);\n }\n // Floats\n for (name in source.floatsArrays) {\n material.setFloats(name, source.floatsArrays[name]);\n }\n // Color3\n for (name in source.colors3) {\n material.setColor3(name, Color3.FromArray(source.colors3[name]));\n }\n // Color3 arrays\n for (name in source.colors3Arrays) {\n const colors = source.colors3Arrays[name]\n .reduce((arr, num, i) => {\n if (i % 3 === 0) {\n arr.push([num]);\n }\n else {\n arr[arr.length - 1].push(num);\n }\n return arr;\n }, [])\n .map((color) => Color3.FromArray(color));\n material.setColor3Array(name, colors);\n }\n // Color4\n for (name in source.colors4) {\n material.setColor4(name, Color4.FromArray(source.colors4[name]));\n }\n // Color4 arrays\n for (name in source.colors4Arrays) {\n const colors = source.colors4Arrays[name]\n .reduce((arr, num, i) => {\n if (i % 4 === 0) {\n arr.push([num]);\n }\n else {\n arr[arr.length - 1].push(num);\n }\n return arr;\n }, [])\n .map((color) => Color4.FromArray(color));\n material.setColor4Array(name, colors);\n }\n // Vector2\n for (name in source.vectors2) {\n material.setVector2(name, Vector2.FromArray(source.vectors2[name]));\n }\n // Vector3\n for (name in source.vectors3) {\n material.setVector3(name, Vector3.FromArray(source.vectors3[name]));\n }\n // Vector4\n for (name in source.vectors4) {\n material.setVector4(name, Vector4.FromArray(source.vectors4[name]));\n }\n // Quaternion\n for (name in source.quaternions) {\n material.setQuaternion(name, Quaternion.FromArray(source.quaternions[name]));\n }\n // Matrix\n for (name in source.matrices) {\n material.setMatrix(name, Matrix.FromArray(source.matrices[name]));\n }\n // MatrixArray\n for (name in source.matrixArray) {\n material._matrixArrays[name] = new Float32Array(source.matrixArray[name]);\n }\n // Matrix 3x3\n for (name in source.matrices3x3) {\n material.setMatrix3x3(name, source.matrices3x3[name]);\n }\n // Matrix 2x2\n for (name in source.matrices2x2) {\n material.setMatrix2x2(name, source.matrices2x2[name]);\n }\n // Vector2Array\n for (name in source.vectors2Arrays) {\n material.setArray2(name, source.vectors2Arrays[name]);\n }\n // Vector3Array\n for (name in source.vectors3Arrays) {\n material.setArray3(name, source.vectors3Arrays[name]);\n }\n // Vector4Array\n for (name in source.vectors4Arrays) {\n material.setArray4(name, source.vectors4Arrays[name]);\n }\n // QuaternionArray\n for (name in source.quaternionsArrays) {\n material.setArray4(name, source.quaternionsArrays[name]);\n }\n return material;\n }\n /**\n * Creates a new ShaderMaterial from a snippet saved in a remote file\n * @param name defines the name of the ShaderMaterial to create (can be null or empty to use the one from the json data)\n * @param url defines the url to load from\n * @param scene defines the hosting scene\n * @param rootUrl defines the root URL to use to load textures and relative dependencies\n * @returns a promise that will resolve to the new ShaderMaterial\n */\n static ParseFromFileAsync(name, url, scene, rootUrl = \"\") {\n return new Promise((resolve, reject) => {\n const request = new WebRequest();\n request.addEventListener(\"readystatechange\", () => {\n if (request.readyState == 4) {\n if (request.status == 200) {\n const serializationObject = JSON.parse(request.responseText);\n const output = this.Parse(serializationObject, scene || EngineStore.LastCreatedScene, rootUrl);\n if (name) {\n output.name = name;\n }\n resolve(output);\n }\n else {\n reject(\"Unable to load the ShaderMaterial\");\n }\n }\n });\n request.open(\"GET\", url);\n request.send();\n });\n }\n /**\n * Creates a ShaderMaterial from a snippet saved by the Inspector\n * @param snippetId defines the snippet to load\n * @param scene defines the hosting scene\n * @param rootUrl defines the root URL to use to load textures and relative dependencies\n * @returns a promise that will resolve to the new ShaderMaterial\n */\n static ParseFromSnippetAsync(snippetId, scene, rootUrl = \"\") {\n return new Promise((resolve, reject) => {\n const request = new WebRequest();\n request.addEventListener(\"readystatechange\", () => {\n if (request.readyState == 4) {\n if (request.status == 200) {\n const snippet = JSON.parse(JSON.parse(request.responseText).jsonPayload);\n const serializationObject = JSON.parse(snippet.shaderMaterial);\n const output = this.Parse(serializationObject, scene || EngineStore.LastCreatedScene, rootUrl);\n output.snippetId = snippetId;\n resolve(output);\n }\n else {\n reject(\"Unable to load the snippet \" + snippetId);\n }\n }\n });\n request.open(\"GET\", this.SnippetUrl + \"/\" + snippetId.replace(/#/g, \"/\"));\n request.send();\n });\n }\n }\n /** Define the Url to load snippets */\n ShaderMaterial.SnippetUrl = `https://snippet.babylonjs.com`;\n /**\n * Creates a ShaderMaterial from a snippet saved by the Inspector\n * @deprecated Please use ParseFromSnippetAsync instead\n * @param snippetId defines the snippet to load\n * @param scene defines the hosting scene\n * @param rootUrl defines the root URL to use to load textures and relative dependencies\n * @returns a promise that will resolve to the new ShaderMaterial\n */\n ShaderMaterial.CreateFromSnippetAsync = ShaderMaterial.ParseFromSnippetAsync;\n RegisterClass(\"BABYLON.ShaderMaterial\", ShaderMaterial);\n\n // Do not edit.\n const name$1Y = \"colorPixelShader\";\n const shader$1Y = `#if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES)\n#define VERTEXCOLOR\nvarying vec4 vColor;\n#else\nuniform vec4 color;\n#endif\n#include\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_FRAGMENT_MAIN_BEGIN\n#include\n#if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES)\ngl_FragColor=vColor;\n#else\ngl_FragColor=color;\n#endif\n#define CUSTOM_FRAGMENT_MAIN_END\n}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$1Y] = shader$1Y;\n\n // Do not edit.\n const name$1Z = \"colorVertexShader\";\n const shader$1Z = `attribute vec3 position;\n#ifdef VERTEXCOLOR\nattribute vec4 color;\n#endif\n#include\n#include\n#include\n#include\nuniform mat4 viewProjection;\n#ifdef MULTIVIEW\nuniform mat4 viewProjectionR;\n#endif\n#if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES)\nvarying vec4 vColor;\n#endif\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\n#include\n#include\n#include\nvec4 worldPos=finalWorld*vec4(position,1.0);\n#ifdef MULTIVIEW\nif (gl_ViewID_OVR==0u) {gl_Position=viewProjection*worldPos;} else {gl_Position=viewProjectionR*worldPos;}\n#else\ngl_Position=viewProjection*worldPos;\n#endif\n#include\n#include\n#define CUSTOM_VERTEX_MAIN_END\n}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$1Z] = shader$1Z;\n\n Mesh._LinesMeshParser = (parsedMesh, scene) => {\n return LinesMesh.Parse(parsedMesh, scene);\n };\n /**\n * Line mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param\n */\n class LinesMesh extends Mesh {\n _isShaderMaterial(shader) {\n return shader.getClassName() === \"ShaderMaterial\";\n }\n /**\n * Creates a new LinesMesh\n * @param name defines the name\n * @param scene defines the hosting scene\n * @param parent defines the parent mesh if any\n * @param source defines the optional source LinesMesh used to clone data from\n * @param doNotCloneChildren When cloning, skip cloning child meshes of source, default False.\n * When false, achieved by calling a clone(), also passing False.\n * This will make creation of children, recursive.\n * @param useVertexColor defines if this LinesMesh supports vertex color\n * @param useVertexAlpha defines if this LinesMesh supports vertex alpha\n * @param material material to use to draw the line. If not provided, will create a new one\n */\n constructor(name, scene = null, parent = null, source = null, doNotCloneChildren, \n /**\n * If vertex color should be applied to the mesh\n */\n useVertexColor, \n /**\n * If vertex alpha should be applied to the mesh\n */\n useVertexAlpha, material) {\n super(name, scene, parent, source, doNotCloneChildren);\n this.useVertexColor = useVertexColor;\n this.useVertexAlpha = useVertexAlpha;\n /**\n * Color of the line (Default: White)\n */\n this.color = new Color3(1, 1, 1);\n /**\n * Alpha of the line (Default: 1)\n */\n this.alpha = 1;\n if (source) {\n this.color = source.color.clone();\n this.alpha = source.alpha;\n this.useVertexColor = source.useVertexColor;\n this.useVertexAlpha = source.useVertexAlpha;\n }\n this.intersectionThreshold = 0.1;\n const defines = [];\n const options = {\n attributes: [VertexBuffer.PositionKind],\n uniforms: [\"world\", \"viewProjection\"],\n needAlphaBlending: true,\n defines: defines,\n useClipPlane: null,\n };\n if (useVertexAlpha === false) {\n options.needAlphaBlending = false;\n }\n else {\n options.defines.push(\"#define VERTEXALPHA\");\n }\n if (!useVertexColor) {\n options.uniforms.push(\"color\");\n this._color4 = new Color4();\n }\n else {\n options.defines.push(\"#define VERTEXCOLOR\");\n options.attributes.push(VertexBuffer.ColorKind);\n }\n if (material) {\n this.material = material;\n }\n else {\n this.material = new ShaderMaterial(\"colorShader\", this.getScene(), \"color\", options, false);\n this.material.doNotSerialize = true;\n }\n }\n isReady() {\n if (!this._lineMaterial.isReady(this, !!this._userInstancedBuffersStorage)) {\n return false;\n }\n return super.isReady();\n }\n /**\n * Returns the string \"LineMesh\"\n */\n getClassName() {\n return \"LinesMesh\";\n }\n /**\n * @internal\n */\n get material() {\n return this._lineMaterial;\n }\n /**\n * @internal\n */\n set material(value) {\n this._lineMaterial = value;\n this._lineMaterial.fillMode = Material.LineListDrawMode;\n }\n /**\n * @internal\n */\n get checkCollisions() {\n return false;\n }\n set checkCollisions(value) {\n // Just ignore it\n }\n /**\n * @internal\n */\n _bind(_subMesh, colorEffect) {\n if (!this._geometry) {\n return this;\n }\n // VBOs\n const indexToBind = this.isUnIndexed ? null : this._geometry.getIndexBuffer();\n if (!this._userInstancedBuffersStorage) {\n this._geometry._bind(colorEffect, indexToBind);\n }\n else {\n this._geometry._bind(colorEffect, indexToBind, this._userInstancedBuffersStorage.vertexBuffers, this._userInstancedBuffersStorage.vertexArrayObjects);\n }\n // Color\n if (!this.useVertexColor && this._isShaderMaterial(this._lineMaterial)) {\n const { r, g, b } = this.color;\n this._color4.set(r, g, b, this.alpha);\n this._lineMaterial.setColor4(\"color\", this._color4);\n }\n return this;\n }\n /**\n * @internal\n */\n _draw(subMesh, fillMode, instancesCount) {\n if (!this._geometry || !this._geometry.getVertexBuffers() || (!this._unIndexed && !this._geometry.getIndexBuffer())) {\n return this;\n }\n const engine = this.getScene().getEngine();\n // Draw order\n if (this._unIndexed) {\n engine.drawArraysType(Material.LineListDrawMode, subMesh.verticesStart, subMesh.verticesCount, instancesCount);\n }\n else {\n engine.drawElementsType(Material.LineListDrawMode, subMesh.indexStart, subMesh.indexCount, instancesCount);\n }\n return this;\n }\n /**\n * Disposes of the line mesh\n * @param doNotRecurse If children should be disposed\n * @param disposeMaterialAndTextures This parameter is not used by the LineMesh class\n * @param doNotDisposeMaterial If the material should not be disposed (default: false, meaning the material is disposed)\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n dispose(doNotRecurse, disposeMaterialAndTextures = false, doNotDisposeMaterial) {\n if (!doNotDisposeMaterial) {\n this._lineMaterial.dispose(false, false, true);\n }\n super.dispose(doNotRecurse);\n }\n /**\n * Returns a new LineMesh object cloned from the current one.\n * @param name\n * @param newParent\n * @param doNotCloneChildren\n */\n clone(name, newParent = null, doNotCloneChildren) {\n return new LinesMesh(name, this.getScene(), newParent, this, doNotCloneChildren);\n }\n /**\n * Creates a new InstancedLinesMesh object from the mesh model.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/copies/instances\n * @param name defines the name of the new instance\n * @returns a new InstancedLinesMesh\n */\n createInstance(name) {\n const instance = new InstancedLinesMesh(name, this);\n if (this.instancedBuffers) {\n instance.instancedBuffers = {};\n for (const key in this.instancedBuffers) {\n instance.instancedBuffers[key] = this.instancedBuffers[key];\n }\n }\n return instance;\n }\n /**\n * Serializes this ground mesh\n * @param serializationObject object to write serialization to\n */\n serialize(serializationObject) {\n super.serialize(serializationObject);\n serializationObject.color = this.color.asArray();\n serializationObject.alpha = this.alpha;\n }\n /**\n * Parses a serialized ground mesh\n * @param parsedMesh the serialized mesh\n * @param scene the scene to create the ground mesh in\n * @returns the created ground mesh\n */\n static Parse(parsedMesh, scene) {\n const result = new LinesMesh(parsedMesh.name, scene);\n result.color = Color3.FromArray(parsedMesh.color);\n result.alpha = parsedMesh.alpha;\n return result;\n }\n }\n /**\n * Creates an instance based on a source LinesMesh\n */\n class InstancedLinesMesh extends InstancedMesh {\n constructor(name, source) {\n super(name, source);\n this.intersectionThreshold = source.intersectionThreshold;\n }\n /**\n * Returns the string \"InstancedLinesMesh\".\n */\n getClassName() {\n return \"InstancedLinesMesh\";\n }\n }\n\n /* eslint-disable @typescript-eslint/naming-convention */\n /**\n * Creates the VertexData of the LineSystem\n * @param options an object used to set the following optional parameters for the LineSystem, required but can be empty\n * - lines an array of lines, each line being an array of successive Vector3\n * - colors an array of line colors, each of the line colors being an array of successive Color4, one per line point\n * @param options.lines\n * @param options.colors\n * @returns the VertexData of the LineSystem\n */\n function CreateLineSystemVertexData(options) {\n const indices = [];\n const positions = [];\n const lines = options.lines;\n const colors = options.colors;\n const vertexColors = [];\n let idx = 0;\n for (let l = 0; l < lines.length; l++) {\n const points = lines[l];\n for (let index = 0; index < points.length; index++) {\n positions.push(points[index].x, points[index].y, points[index].z);\n if (colors) {\n const color = colors[l];\n vertexColors.push(color[index].r, color[index].g, color[index].b, color[index].a);\n }\n if (index > 0) {\n indices.push(idx - 1);\n indices.push(idx);\n }\n idx++;\n }\n }\n const vertexData = new VertexData();\n vertexData.indices = indices;\n vertexData.positions = positions;\n if (colors) {\n vertexData.colors = vertexColors;\n }\n return vertexData;\n }\n /**\n * Create the VertexData for a DashedLines\n * @param options an object used to set the following optional parameters for the DashedLines, required but can be empty\n * - points an array successive Vector3\n * - dashSize the size of the dashes relative to the dash number, optional, default 3\n * - gapSize the size of the gap between two successive dashes relative to the dash number, optional, default 1\n * - dashNb the intended total number of dashes, optional, default 200\n * @param options.points\n * @param options.dashSize\n * @param options.gapSize\n * @param options.dashNb\n * @returns the VertexData for the DashedLines\n */\n function CreateDashedLinesVertexData(options) {\n const dashSize = options.dashSize || 3;\n const gapSize = options.gapSize || 1;\n const dashNb = options.dashNb || 200;\n const points = options.points;\n const positions = new Array();\n const indices = new Array();\n const curvect = Vector3.Zero();\n let lg = 0;\n let nb = 0;\n let shft = 0;\n let dashshft = 0;\n let curshft = 0;\n let idx = 0;\n let i = 0;\n for (i = 0; i < points.length - 1; i++) {\n points[i + 1].subtractToRef(points[i], curvect);\n lg += curvect.length();\n }\n shft = lg / dashNb;\n dashshft = (dashSize * shft) / (dashSize + gapSize);\n for (i = 0; i < points.length - 1; i++) {\n points[i + 1].subtractToRef(points[i], curvect);\n nb = Math.floor(curvect.length() / shft);\n curvect.normalize();\n for (let j = 0; j < nb; j++) {\n curshft = shft * j;\n positions.push(points[i].x + curshft * curvect.x, points[i].y + curshft * curvect.y, points[i].z + curshft * curvect.z);\n positions.push(points[i].x + (curshft + dashshft) * curvect.x, points[i].y + (curshft + dashshft) * curvect.y, points[i].z + (curshft + dashshft) * curvect.z);\n indices.push(idx, idx + 1);\n idx += 2;\n }\n }\n // Result\n const vertexData = new VertexData();\n vertexData.positions = positions;\n vertexData.indices = indices;\n return vertexData;\n }\n /**\n * Creates a line system mesh. A line system is a pool of many lines gathered in a single mesh\n * * A line system mesh is considered as a parametric shape since it has no predefined original shape. Its shape is determined by the passed array of lines as an input parameter\n * * Like every other parametric shape, it is dynamically updatable by passing an existing instance of LineSystem to this static function\n * * The parameter `lines` is an array of lines, each line being an array of successive Vector3\n * * The optional parameter `instance` is an instance of an existing LineSystem object to be updated with the passed `lines` parameter\n * * The optional parameter `colors` is an array of line colors, each line colors being an array of successive Color4, one per line point\n * * The optional parameter `useVertexAlpha` is to be set to `false` (default `true`) when you don't need the alpha blending (faster)\n * * The optional parameter `material` is the material to use to draw the lines if provided. If not, a default material will be created\n * * Updating a simple Line mesh, you just need to update every line in the `lines` array : https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#lines-and-dashedlines\n * * When updating an instance, remember that only line point positions can change, not the number of points, neither the number of lines\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#line-system\n * @param name defines the name of the new line system\n * @param options defines the options used to create the line system\n * @param options.lines\n * @param options.updatable\n * @param options.instance\n * @param options.colors\n * @param options.useVertexAlpha\n * @param options.material\n * @param scene defines the hosting scene\n * @returns a new line system mesh\n */\n function CreateLineSystem(name, options, scene = null) {\n const instance = options.instance;\n const lines = options.lines;\n const colors = options.colors;\n if (instance) {\n // lines update\n const positions = instance.getVerticesData(VertexBuffer.PositionKind);\n let vertexColor;\n let lineColors;\n if (colors) {\n vertexColor = instance.getVerticesData(VertexBuffer.ColorKind);\n }\n let i = 0;\n let c = 0;\n for (let l = 0; l < lines.length; l++) {\n const points = lines[l];\n for (let p = 0; p < points.length; p++) {\n positions[i] = points[p].x;\n positions[i + 1] = points[p].y;\n positions[i + 2] = points[p].z;\n if (colors && vertexColor) {\n lineColors = colors[l];\n vertexColor[c] = lineColors[p].r;\n vertexColor[c + 1] = lineColors[p].g;\n vertexColor[c + 2] = lineColors[p].b;\n vertexColor[c + 3] = lineColors[p].a;\n c += 4;\n }\n i += 3;\n }\n }\n instance.updateVerticesData(VertexBuffer.PositionKind, positions, false, false);\n if (colors && vertexColor) {\n instance.updateVerticesData(VertexBuffer.ColorKind, vertexColor, false, false);\n }\n return instance;\n }\n // line system creation\n const useVertexColor = colors ? true : false;\n const lineSystem = new LinesMesh(name, scene, null, undefined, undefined, useVertexColor, options.useVertexAlpha, options.material);\n const vertexData = CreateLineSystemVertexData(options);\n vertexData.applyToMesh(lineSystem, options.updatable);\n return lineSystem;\n }\n /**\n * Creates a line mesh\n * A line mesh is considered as a parametric shape since it has no predefined original shape. Its shape is determined by the passed array of points as an input parameter\n * * Like every other parametric shape, it is dynamically updatable by passing an existing instance of LineMesh to this static function\n * * The parameter `points` is an array successive Vector3\n * * The optional parameter `instance` is an instance of an existing LineMesh object to be updated with the passed `points` parameter : https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#lines-and-dashedlines\n * * The optional parameter `colors` is an array of successive Color4, one per line point\n * * The optional parameter `useVertexAlpha` is to be set to `false` (default `true`) when you don't need alpha blending (faster)\n * * The optional parameter `material` is the material to use to draw the lines if provided. If not, a default material will be created\n * * When updating an instance, remember that only point positions can change, not the number of points\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#lines\n * @param name defines the name of the new line system\n * @param options defines the options used to create the line system\n * @param options.points\n * @param options.updatable\n * @param options.instance\n * @param options.colors\n * @param options.useVertexAlpha\n * @param options.material\n * @param scene defines the hosting scene\n * @returns a new line mesh\n */\n function CreateLines(name, options, scene = null) {\n const colors = options.colors ? [options.colors] : null;\n const lines = CreateLineSystem(name, { lines: [options.points], updatable: options.updatable, instance: options.instance, colors: colors, useVertexAlpha: options.useVertexAlpha, material: options.material }, scene);\n return lines;\n }\n /**\n * Creates a dashed line mesh\n * * A dashed line mesh is considered as a parametric shape since it has no predefined original shape. Its shape is determined by the passed array of points as an input parameter\n * * Like every other parametric shape, it is dynamically updatable by passing an existing instance of LineMesh to this static function\n * * The parameter `points` is an array successive Vector3\n * * The parameter `dashNb` is the intended total number of dashes (positive integer, default 200)\n * * The parameter `dashSize` is the size of the dashes relatively the dash number (positive float, default 3)\n * * The parameter `gapSize` is the size of the gap between two successive dashes relatively the dash number (positive float, default 1)\n * * The optional parameter `instance` is an instance of an existing LineMesh object to be updated with the passed `points` parameter : https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#lines-and-dashedlines\n * * The optional parameter `useVertexAlpha` is to be set to `false` (default `true`) when you don't need the alpha blending (faster)\n * * The optional parameter `material` is the material to use to draw the lines if provided. If not, a default material will be created\n * * When updating an instance, remember that only point positions can change, not the number of points\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param options.points\n * @param options.dashSize\n * @param options.gapSize\n * @param options.dashNb\n * @param options.updatable\n * @param options.instance\n * @param options.useVertexAlpha\n * @param options.material\n * @param scene defines the hosting scene\n * @returns the dashed line mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#dashed-lines\n */\n function CreateDashedLines(name, options, scene = null) {\n const points = options.points;\n const instance = options.instance;\n const gapSize = options.gapSize || 1;\n const dashSize = options.dashSize || 3;\n if (instance) {\n // dashed lines update\n const positionFunction = (positions) => {\n const curvect = Vector3.Zero();\n const nbSeg = positions.length / 6;\n let lg = 0;\n let nb = 0;\n let shft = 0;\n let dashshft = 0;\n let curshft = 0;\n let p = 0;\n let i = 0;\n let j = 0;\n for (i = 0; i < points.length - 1; i++) {\n points[i + 1].subtractToRef(points[i], curvect);\n lg += curvect.length();\n }\n shft = lg / nbSeg;\n const dashSize = instance._creationDataStorage.dashSize;\n const gapSize = instance._creationDataStorage.gapSize;\n dashshft = (dashSize * shft) / (dashSize + gapSize);\n for (i = 0; i < points.length - 1; i++) {\n points[i + 1].subtractToRef(points[i], curvect);\n nb = Math.floor(curvect.length() / shft);\n curvect.normalize();\n j = 0;\n while (j < nb && p < positions.length) {\n curshft = shft * j;\n positions[p] = points[i].x + curshft * curvect.x;\n positions[p + 1] = points[i].y + curshft * curvect.y;\n positions[p + 2] = points[i].z + curshft * curvect.z;\n positions[p + 3] = points[i].x + (curshft + dashshft) * curvect.x;\n positions[p + 4] = points[i].y + (curshft + dashshft) * curvect.y;\n positions[p + 5] = points[i].z + (curshft + dashshft) * curvect.z;\n p += 6;\n j++;\n }\n }\n while (p < positions.length) {\n positions[p] = points[i].x;\n positions[p + 1] = points[i].y;\n positions[p + 2] = points[i].z;\n p += 3;\n }\n };\n if (options.dashNb || options.dashSize || options.gapSize || options.useVertexAlpha || options.material) {\n Logger.Warn(\"You have used an option other than points with the instance option. Please be aware that these other options will be ignored.\");\n }\n instance.updateMeshPositions(positionFunction, false);\n return instance;\n }\n // dashed lines creation\n const dashedLines = new LinesMesh(name, scene, null, undefined, undefined, undefined, options.useVertexAlpha, options.material);\n const vertexData = CreateDashedLinesVertexData(options);\n vertexData.applyToMesh(dashedLines, options.updatable);\n dashedLines._creationDataStorage = new _CreationDataStorage();\n dashedLines._creationDataStorage.dashSize = dashSize;\n dashedLines._creationDataStorage.gapSize = gapSize;\n return dashedLines;\n }\n VertexData.CreateLineSystem = CreateLineSystemVertexData;\n VertexData.CreateDashedLines = CreateDashedLinesVertexData;\n Mesh.CreateLines = (name, points, scene = null, updatable = false, instance = null) => {\n const options = {\n points,\n updatable,\n instance,\n };\n return CreateLines(name, options, scene);\n };\n Mesh.CreateDashedLines = (name, points, dashSize, gapSize, dashNb, scene = null, updatable, instance) => {\n const options = {\n points,\n dashSize,\n gapSize,\n dashNb,\n updatable,\n instance,\n };\n return CreateDashedLines(name, options, scene);\n };\n\n /**\n * The current state of the timer\n */\n var TimerState;\n (function (TimerState) {\n /**\n * Timer initialized, not yet started\n */\n TimerState[TimerState[\"INIT\"] = 0] = \"INIT\";\n /**\n * Timer started and counting\n */\n TimerState[TimerState[\"STARTED\"] = 1] = \"STARTED\";\n /**\n * Timer ended (whether aborted or time reached)\n */\n TimerState[TimerState[\"ENDED\"] = 2] = \"ENDED\";\n })(TimerState || (TimerState = {}));\n /**\n * A simple version of the timer. Will take options and start the timer immediately after calling it\n *\n * @param options options with which to initialize this timer\n */\n function setAndStartTimer(options) {\n var _a;\n let timer = 0;\n const startTime = Date.now();\n options.observableParameters = (_a = options.observableParameters) !== null && _a !== void 0 ? _a : {};\n const observer = options.contextObservable.add((payload) => {\n const now = Date.now();\n timer = now - startTime;\n const data = {\n startTime,\n currentTime: now,\n deltaTime: timer,\n completeRate: timer / options.timeout,\n payload,\n };\n options.onTick && options.onTick(data);\n if (options.breakCondition && options.breakCondition()) {\n options.contextObservable.remove(observer);\n options.onAborted && options.onAborted(data);\n }\n if (timer >= options.timeout) {\n options.contextObservable.remove(observer);\n options.onEnded && options.onEnded(data);\n }\n }, options.observableParameters.mask, options.observableParameters.insertFirst, options.observableParameters.scope);\n return observer;\n }\n\n /**\n * This is a teleportation feature to be used with WebXR-enabled motion controllers.\n * When enabled and attached, the feature will allow a user to move around and rotate in the scene using\n * the input of the attached controllers.\n */\n class WebXRMotionControllerTeleportation extends WebXRAbstractFeature {\n /**\n * Is rotation enabled when moving forward?\n * Disabling this feature will prevent the user from deciding the direction when teleporting\n */\n get rotationEnabled() {\n return this._rotationEnabled;\n }\n /**\n * Sets whether rotation is enabled or not\n * @param enabled is rotation enabled when teleportation is shown\n */\n set rotationEnabled(enabled) {\n this._rotationEnabled = enabled;\n if (this._options.teleportationTargetMesh) {\n const children = this._options.teleportationTargetMesh.getChildMeshes(false, (node) => node.name === \"rotationCone\");\n if (children[0]) {\n children[0].setEnabled(enabled);\n }\n }\n }\n /**\n * Exposes the currently set teleportation target mesh.\n */\n get teleportationTargetMesh() {\n return this._options.teleportationTargetMesh || null;\n }\n /**\n * constructs a new teleportation system\n * @param _xrSessionManager an instance of WebXRSessionManager\n * @param _options configuration object for this feature\n */\n constructor(_xrSessionManager, _options) {\n super(_xrSessionManager);\n this._options = _options;\n this._controllers = {};\n this._snappedToPoint = false;\n this._cachedColor4White = new Color4(1, 1, 1, 1);\n this._tmpRay = new Ray(new Vector3(), new Vector3());\n this._tmpVector = new Vector3();\n this._tmpQuaternion = new Quaternion();\n /**\n * Skip the next teleportation. This can be controlled by the user to prevent the user from teleportation\n * to sections that are not yet \"unlocked\", but should still show the teleportation mesh.\n */\n this.skipNextTeleportation = false;\n /**\n * Is movement backwards enabled\n */\n this.backwardsMovementEnabled = true;\n /**\n * Distance to travel when moving backwards\n */\n this.backwardsTeleportationDistance = 0.7;\n /**\n * The distance from the user to the inspection point in the direction of the controller\n * A higher number will allow the user to move further\n * defaults to 5 (meters, in xr units)\n */\n this.parabolicCheckRadius = 5;\n /**\n * Should the module support parabolic ray on top of direct ray\n * If enabled, the user will be able to point \"at the sky\" and move according to predefined radius distance\n * Very helpful when moving between floors / different heights\n */\n this.parabolicRayEnabled = true;\n /**\n * The second type of ray - straight line.\n * Should it be enabled or should the parabolic line be the only one.\n */\n this.straightRayEnabled = true;\n /**\n * How much rotation should be applied when rotating right and left\n */\n this.rotationAngle = Math.PI / 8;\n /**\n * This observable will notify when the target mesh position was updated.\n * The picking info it provides contains the point to which the target mesh will move ()\n */\n this.onTargetMeshPositionUpdatedObservable = new Observable$1();\n /**\n * Is teleportation enabled. Can be used to allow rotation only.\n */\n this.teleportationEnabled = true;\n this._rotationEnabled = true;\n this._attachController = (xrController) => {\n if (this._controllers[xrController.uniqueId] || (this._options.forceHandedness && xrController.inputSource.handedness !== this._options.forceHandedness)) {\n // already attached\n return;\n }\n this._controllers[xrController.uniqueId] = {\n xrController,\n teleportationState: {\n forward: false,\n backwards: false,\n rotating: false,\n currentRotation: 0,\n baseRotation: 0,\n blocked: false,\n },\n };\n const controllerData = this._controllers[xrController.uniqueId];\n // motion controller only available to gamepad-enabled input sources.\n if (controllerData.xrController.inputSource.targetRayMode === \"tracked-pointer\" && controllerData.xrController.inputSource.gamepad) {\n // motion controller support\n const initMotionController = () => {\n if (xrController.motionController) {\n const movementController = xrController.motionController.getComponentOfType(WebXRControllerComponent.THUMBSTICK_TYPE) ||\n xrController.motionController.getComponentOfType(WebXRControllerComponent.TOUCHPAD_TYPE);\n if (!movementController || this._options.useMainComponentOnly) {\n // use trigger to move on long press\n const mainComponent = xrController.motionController.getMainComponent();\n if (!mainComponent) {\n return;\n }\n controllerData.teleportationComponent = mainComponent;\n controllerData.onButtonChangedObserver = mainComponent.onButtonStateChangedObservable.add(() => {\n if (!this.teleportationEnabled) {\n return;\n }\n // did \"pressed\" changed?\n if (mainComponent.changes.pressed) {\n if (mainComponent.changes.pressed.current) {\n // simulate \"forward\" thumbstick push\n controllerData.teleportationState.forward = true;\n this._currentTeleportationControllerId = controllerData.xrController.uniqueId;\n controllerData.teleportationState.baseRotation = this._options.xrInput.xrCamera.rotationQuaternion.toEulerAngles().y;\n controllerData.teleportationState.currentRotation = 0;\n const timeToSelect = this._options.timeToTeleport || 3000;\n setAndStartTimer({\n timeout: timeToSelect,\n contextObservable: this._xrSessionManager.onXRFrameObservable,\n breakCondition: () => !mainComponent.pressed,\n onEnded: () => {\n if (this._currentTeleportationControllerId === controllerData.xrController.uniqueId && controllerData.teleportationState.forward) {\n this._teleportForward(xrController.uniqueId);\n }\n },\n });\n }\n else {\n controllerData.teleportationState.forward = false;\n this._currentTeleportationControllerId = \"\";\n }\n }\n });\n }\n else {\n controllerData.teleportationComponent = movementController;\n // use thumbstick (or touchpad if thumbstick not available)\n controllerData.onAxisChangedObserver = movementController.onAxisValueChangedObservable.add((axesData) => {\n if (axesData.y <= 0.7 && controllerData.teleportationState.backwards) {\n controllerData.teleportationState.backwards = false;\n }\n if (axesData.y > 0.7 && !controllerData.teleportationState.forward && this.backwardsMovementEnabled && !this.snapPointsOnly) {\n // teleport backwards\n // General gist: Go Back N units, cast a ray towards the floor. If collided, move.\n if (!controllerData.teleportationState.backwards) {\n controllerData.teleportationState.backwards = true;\n // teleport backwards ONCE\n this._tmpQuaternion.copyFrom(this._options.xrInput.xrCamera.rotationQuaternion);\n this._tmpQuaternion.toEulerAnglesToRef(this._tmpVector);\n // get only the y rotation\n this._tmpVector.x = 0;\n this._tmpVector.z = 0;\n // get the quaternion\n Quaternion.FromEulerVectorToRef(this._tmpVector, this._tmpQuaternion);\n this._tmpVector.set(0, 0, this.backwardsTeleportationDistance * (this._xrSessionManager.scene.useRightHandedSystem ? 1.0 : -1.0));\n this._tmpVector.rotateByQuaternionToRef(this._tmpQuaternion, this._tmpVector);\n this._tmpVector.addInPlace(this._options.xrInput.xrCamera.position);\n this._tmpRay.origin.copyFrom(this._tmpVector);\n // This will prevent the user from \"falling\" to a lower platform!\n // TODO - should this be a flag? 'allow falling to lower platforms'?\n this._tmpRay.length = this._options.xrInput.xrCamera.realWorldHeight + 0.1;\n // Right handed system had here \"1\" instead of -1. This is unneeded.\n this._tmpRay.direction.set(0, -1, 0);\n const pick = this._xrSessionManager.scene.pickWithRay(this._tmpRay, (o) => {\n return this._floorMeshes.indexOf(o) !== -1;\n });\n // pick must exist, but stay safe\n if (pick && pick.pickedPoint) {\n // Teleport the users feet to where they targeted. Ignore the Y axis.\n // If the \"falling to lower platforms\" feature is implemented the Y axis should be set here as well\n this._options.xrInput.xrCamera.position.x = pick.pickedPoint.x;\n this._options.xrInput.xrCamera.position.z = pick.pickedPoint.z;\n }\n }\n }\n if (axesData.y < -0.7 && !this._currentTeleportationControllerId && !controllerData.teleportationState.rotating && this.teleportationEnabled) {\n controllerData.teleportationState.forward = true;\n this._currentTeleportationControllerId = controllerData.xrController.uniqueId;\n controllerData.teleportationState.baseRotation = this._options.xrInput.xrCamera.rotationQuaternion.toEulerAngles().y;\n }\n if (axesData.x) {\n if (!controllerData.teleportationState.forward) {\n if (!controllerData.teleportationState.rotating && Math.abs(axesData.x) > 0.7) {\n // rotate in the right direction positive is right\n controllerData.teleportationState.rotating = true;\n const rotation = this.rotationAngle * (axesData.x > 0 ? 1 : -1) * (this._xrSessionManager.scene.useRightHandedSystem ? -1 : 1);\n Quaternion.FromEulerAngles(0, rotation, 0).multiplyToRef(this._options.xrInput.xrCamera.rotationQuaternion, this._options.xrInput.xrCamera.rotationQuaternion);\n }\n }\n else {\n if (this._currentTeleportationControllerId === controllerData.xrController.uniqueId) {\n // set the rotation of the forward movement\n if (this.rotationEnabled) {\n setTimeout(() => {\n controllerData.teleportationState.currentRotation = Math.atan2(axesData.x, axesData.y * (this._xrSessionManager.scene.useRightHandedSystem ? 1 : -1));\n });\n }\n else {\n controllerData.teleportationState.currentRotation = 0;\n }\n }\n }\n }\n else {\n controllerData.teleportationState.rotating = false;\n }\n if (axesData.x === 0 && axesData.y === 0) {\n if (controllerData.teleportationState.blocked) {\n controllerData.teleportationState.blocked = false;\n this._setTargetMeshVisibility(false);\n }\n if (controllerData.teleportationState.forward) {\n this._teleportForward(xrController.uniqueId);\n }\n }\n });\n }\n }\n };\n if (xrController.motionController) {\n initMotionController();\n }\n else {\n xrController.onMotionControllerInitObservable.addOnce(() => {\n initMotionController();\n });\n }\n }\n else {\n this._xrSessionManager.scene.onPointerObservable.add((pointerInfo) => {\n if (pointerInfo.type === PointerEventTypes.POINTERDOWN) {\n controllerData.teleportationState.forward = true;\n this._currentTeleportationControllerId = controllerData.xrController.uniqueId;\n controllerData.teleportationState.baseRotation = this._options.xrInput.xrCamera.rotationQuaternion.toEulerAngles().y;\n controllerData.teleportationState.currentRotation = 0;\n const timeToSelect = this._options.timeToTeleport || 3000;\n setAndStartTimer({\n timeout: timeToSelect,\n contextObservable: this._xrSessionManager.onXRFrameObservable,\n onEnded: () => {\n if (this._currentTeleportationControllerId === controllerData.xrController.uniqueId && controllerData.teleportationState.forward) {\n this._teleportForward(xrController.uniqueId);\n }\n },\n });\n }\n else if (pointerInfo.type === PointerEventTypes.POINTERUP) {\n controllerData.teleportationState.forward = false;\n this._currentTeleportationControllerId = \"\";\n }\n });\n }\n };\n // create default mesh if not provided\n if (!this._options.teleportationTargetMesh) {\n this._createDefaultTargetMesh();\n }\n this._floorMeshes = this._options.floorMeshes || [];\n this._snapToPositions = this._options.snapPositions || [];\n this._blockedRayColor = this._options.blockedRayColor || new Color4(1, 0, 0, 0.75);\n this._setTargetMeshVisibility(false);\n }\n /**\n * Get the snapPointsOnly flag\n */\n get snapPointsOnly() {\n return !!this._options.snapPointsOnly;\n }\n /**\n * Sets the snapPointsOnly flag\n * @param snapToPoints should teleportation be exclusively to snap points\n */\n set snapPointsOnly(snapToPoints) {\n this._options.snapPointsOnly = snapToPoints;\n }\n /**\n * Add a new mesh to the floor meshes array\n * @param mesh the mesh to use as floor mesh\n */\n addFloorMesh(mesh) {\n this._floorMeshes.push(mesh);\n }\n /**\n * Add a mesh to the list of meshes blocking the teleportation ray\n * @param mesh The mesh to add to the teleportation-blocking meshes\n */\n addBlockerMesh(mesh) {\n this._options.pickBlockerMeshes = this._options.pickBlockerMeshes || [];\n this._options.pickBlockerMeshes.push(mesh);\n }\n /**\n * Add a new snap-to point to fix teleportation to this position\n * @param newSnapPoint The new Snap-To point\n */\n addSnapPoint(newSnapPoint) {\n this._snapToPositions.push(newSnapPoint);\n }\n attach() {\n if (!super.attach()) {\n return false;\n }\n // Safety reset\n this._currentTeleportationControllerId = \"\";\n this._options.xrInput.controllers.forEach(this._attachController);\n this._addNewAttachObserver(this._options.xrInput.onControllerAddedObservable, this._attachController);\n this._addNewAttachObserver(this._options.xrInput.onControllerRemovedObservable, (controller) => {\n // REMOVE the controller\n this._detachController(controller.uniqueId);\n });\n return true;\n }\n detach() {\n if (!super.detach()) {\n return false;\n }\n Object.keys(this._controllers).forEach((controllerId) => {\n this._detachController(controllerId);\n });\n this._setTargetMeshVisibility(false);\n this._currentTeleportationControllerId = \"\";\n this._controllers = {};\n return true;\n }\n dispose() {\n super.dispose();\n this._options.teleportationTargetMesh && this._options.teleportationTargetMesh.dispose(false, true);\n }\n /**\n * Remove a mesh from the floor meshes array\n * @param mesh the mesh to remove\n */\n removeFloorMesh(mesh) {\n const index = this._floorMeshes.indexOf(mesh);\n if (index !== -1) {\n this._floorMeshes.splice(index, 1);\n }\n }\n /**\n * Remove a mesh from the blocker meshes array\n * @param mesh the mesh to remove\n */\n removeBlockerMesh(mesh) {\n this._options.pickBlockerMeshes = this._options.pickBlockerMeshes || [];\n const index = this._options.pickBlockerMeshes.indexOf(mesh);\n if (index !== -1) {\n this._options.pickBlockerMeshes.splice(index, 1);\n }\n }\n /**\n * Remove a mesh from the floor meshes array using its name\n * @param name the mesh name to remove\n */\n removeFloorMeshByName(name) {\n const mesh = this._xrSessionManager.scene.getMeshByName(name);\n if (mesh) {\n this.removeFloorMesh(mesh);\n }\n }\n /**\n * This function will iterate through the array, searching for this point or equal to it. It will then remove it from the snap-to array\n * @param snapPointToRemove the point (or a clone of it) to be removed from the array\n * @returns was the point found and removed or not\n */\n removeSnapPoint(snapPointToRemove) {\n // check if the object is in the array\n let index = this._snapToPositions.indexOf(snapPointToRemove);\n // if not found as an object, compare to the points\n if (index === -1) {\n for (let i = 0; i < this._snapToPositions.length; ++i) {\n // equals? index is i, break the loop\n if (this._snapToPositions[i].equals(snapPointToRemove)) {\n index = i;\n break;\n }\n }\n }\n // index is not -1? remove the object\n if (index !== -1) {\n this._snapToPositions.splice(index, 1);\n return true;\n }\n return false;\n }\n /**\n * This function sets a selection feature that will be disabled when\n * the forward ray is shown and will be reattached when hidden.\n * This is used to remove the selection rays when moving.\n * @param selectionFeature the feature to disable when forward movement is enabled\n */\n setSelectionFeature(selectionFeature) {\n this._selectionFeature = selectionFeature;\n }\n _onXRFrame(_xrFrame) {\n const frame = this._xrSessionManager.currentFrame;\n const scene = this._xrSessionManager.scene;\n if (!this.attach || !frame) {\n return;\n }\n // render target if needed\n const targetMesh = this._options.teleportationTargetMesh;\n if (this._currentTeleportationControllerId) {\n if (!targetMesh) {\n return;\n }\n targetMesh.rotationQuaternion = targetMesh.rotationQuaternion || new Quaternion();\n const controllerData = this._controllers[this._currentTeleportationControllerId];\n if (controllerData && controllerData.teleportationState.forward) {\n // set the rotation\n Quaternion.RotationYawPitchRollToRef(controllerData.teleportationState.currentRotation + controllerData.teleportationState.baseRotation, 0, 0, targetMesh.rotationQuaternion);\n // set the ray and position\n let hitPossible = false;\n controllerData.xrController.getWorldPointerRayToRef(this._tmpRay);\n if (this.straightRayEnabled) {\n // first check if direct ray possible\n // pick grounds that are LOWER only. upper will use parabolic path\n const pick = scene.pickWithRay(this._tmpRay, (o) => {\n // check for mesh-blockers\n if (this._options.pickBlockerMeshes && this._options.pickBlockerMeshes.indexOf(o) !== -1) {\n return true;\n }\n const index = this._floorMeshes.indexOf(o);\n if (index === -1) {\n return false;\n }\n return this._floorMeshes[index].absolutePosition.y < this._options.xrInput.xrCamera.globalPosition.y;\n });\n if (pick && pick.pickedMesh && this._options.pickBlockerMeshes && this._options.pickBlockerMeshes.indexOf(pick.pickedMesh) !== -1) {\n controllerData.teleportationState.blocked = true;\n this._setTargetMeshVisibility(false);\n this._showParabolicPath(pick);\n return;\n }\n else if (pick && pick.pickedPoint) {\n controllerData.teleportationState.blocked = false;\n hitPossible = true;\n this._setTargetMeshPosition(pick);\n this._setTargetMeshVisibility(true);\n this._showParabolicPath(pick);\n }\n }\n // straight ray is still the main ray, but disabling the straight line will force parabolic line.\n if (this.parabolicRayEnabled && !hitPossible) {\n // radius compensation according to pointer rotation around X\n const xRotation = controllerData.xrController.pointer.rotationQuaternion.toEulerAngles().x;\n const compensation = 1 + (Math.PI / 2 - Math.abs(xRotation));\n // check parabolic ray\n const radius = this.parabolicCheckRadius * compensation;\n this._tmpRay.origin.addToRef(this._tmpRay.direction.scale(radius * 2), this._tmpVector);\n this._tmpVector.y = this._tmpRay.origin.y;\n this._tmpRay.origin.addInPlace(this._tmpRay.direction.scale(radius));\n this._tmpVector.subtractToRef(this._tmpRay.origin, this._tmpRay.direction);\n this._tmpRay.direction.normalize();\n const pick = scene.pickWithRay(this._tmpRay, (o) => {\n // check for mesh-blockers\n if (this._options.pickBlockerMeshes && this._options.pickBlockerMeshes.indexOf(o) !== -1) {\n return true;\n }\n return this._floorMeshes.indexOf(o) !== -1;\n });\n if (pick && pick.pickedMesh && this._options.pickBlockerMeshes && this._options.pickBlockerMeshes.indexOf(pick.pickedMesh) !== -1) {\n controllerData.teleportationState.blocked = true;\n this._setTargetMeshVisibility(false);\n this._showParabolicPath(pick);\n return;\n }\n else if (pick && pick.pickedPoint) {\n controllerData.teleportationState.blocked = false;\n hitPossible = true;\n this._setTargetMeshPosition(pick);\n this._setTargetMeshVisibility(true);\n this._showParabolicPath(pick);\n }\n }\n // if needed, set visible:\n this._setTargetMeshVisibility(hitPossible);\n }\n else {\n this._setTargetMeshVisibility(false);\n }\n }\n else {\n this._disposeBezierCurve();\n this._setTargetMeshVisibility(false);\n }\n }\n _createDefaultTargetMesh() {\n // set defaults\n this._options.defaultTargetMeshOptions = this._options.defaultTargetMeshOptions || {};\n const sceneToRenderTo = this._options.useUtilityLayer\n ? this._options.customUtilityLayerScene || UtilityLayerRenderer.DefaultUtilityLayer.utilityLayerScene\n : this._xrSessionManager.scene;\n const teleportationTarget = CreateGround(\"teleportationTarget\", { width: 2, height: 2, subdivisions: 2 }, sceneToRenderTo);\n teleportationTarget.isPickable = false;\n if (this._options.defaultTargetMeshOptions.teleportationCircleMaterial) {\n teleportationTarget.material = this._options.defaultTargetMeshOptions.teleportationCircleMaterial;\n }\n else {\n const length = 512;\n const dynamicTexture = new DynamicTexture(\"teleportationPlaneDynamicTexture\", length, sceneToRenderTo, true);\n dynamicTexture.hasAlpha = true;\n const context = dynamicTexture.getContext();\n const centerX = length / 2;\n const centerY = length / 2;\n const radius = 200;\n context.beginPath();\n context.arc(centerX, centerY, radius, 0, 2 * Math.PI, false);\n context.fillStyle = this._options.defaultTargetMeshOptions.teleportationFillColor || \"#444444\";\n context.fill();\n context.lineWidth = 10;\n context.strokeStyle = this._options.defaultTargetMeshOptions.teleportationBorderColor || \"#FFFFFF\";\n context.stroke();\n context.closePath();\n dynamicTexture.update();\n const teleportationCircleMaterial = new StandardMaterial(\"teleportationPlaneMaterial\", sceneToRenderTo);\n teleportationCircleMaterial.diffuseTexture = dynamicTexture;\n teleportationTarget.material = teleportationCircleMaterial;\n }\n const torus = CreateTorus(\"torusTeleportation\", {\n diameter: 0.75,\n thickness: 0.1,\n tessellation: 20,\n }, sceneToRenderTo);\n torus.isPickable = false;\n torus.parent = teleportationTarget;\n if (!this._options.defaultTargetMeshOptions.disableAnimation) {\n const animationInnerCircle = new Animation(\"animationInnerCircle\", \"position.y\", 30, Animation.ANIMATIONTYPE_FLOAT, Animation.ANIMATIONLOOPMODE_CYCLE);\n const keys = [];\n keys.push({\n frame: 0,\n value: 0,\n });\n keys.push({\n frame: 30,\n value: 0.4,\n });\n keys.push({\n frame: 60,\n value: 0,\n });\n animationInnerCircle.setKeys(keys);\n const easingFunction = new SineEase();\n easingFunction.setEasingMode(EasingFunction.EASINGMODE_EASEINOUT);\n animationInnerCircle.setEasingFunction(easingFunction);\n torus.animations = [];\n torus.animations.push(animationInnerCircle);\n sceneToRenderTo.beginAnimation(torus, 0, 60, true);\n }\n const cone = CreateCylinder(\"rotationCone\", { diameterTop: 0, tessellation: 4 }, sceneToRenderTo);\n cone.isPickable = false;\n cone.scaling.set(0.5, 0.12, 0.2);\n cone.rotate(Axis.X, Math.PI / 2);\n cone.position.z = 0.6;\n cone.parent = torus;\n if (this._options.defaultTargetMeshOptions.torusArrowMaterial) {\n torus.material = this._options.defaultTargetMeshOptions.torusArrowMaterial;\n cone.material = this._options.defaultTargetMeshOptions.torusArrowMaterial;\n }\n else {\n const torusConeMaterial = new StandardMaterial(\"torusConsMat\", sceneToRenderTo);\n torusConeMaterial.disableLighting = !!this._options.defaultTargetMeshOptions.disableLighting;\n if (torusConeMaterial.disableLighting) {\n torusConeMaterial.emissiveColor = new Color3(0.3, 0.3, 1.0);\n }\n else {\n torusConeMaterial.diffuseColor = new Color3(0.3, 0.3, 1.0);\n }\n torusConeMaterial.alpha = 0.9;\n torus.material = torusConeMaterial;\n cone.material = torusConeMaterial;\n this._teleportationRingMaterial = torusConeMaterial;\n }\n if (this._options.renderingGroupId !== undefined) {\n teleportationTarget.renderingGroupId = this._options.renderingGroupId;\n torus.renderingGroupId = this._options.renderingGroupId;\n cone.renderingGroupId = this._options.renderingGroupId;\n }\n this._options.teleportationTargetMesh = teleportationTarget;\n // hide the teleportation target mesh right after creating it.\n this._setTargetMeshVisibility(false);\n }\n _detachController(xrControllerUniqueId) {\n const controllerData = this._controllers[xrControllerUniqueId];\n if (!controllerData) {\n return;\n }\n if (controllerData.teleportationComponent) {\n if (controllerData.onAxisChangedObserver) {\n controllerData.teleportationComponent.onAxisValueChangedObservable.remove(controllerData.onAxisChangedObserver);\n }\n if (controllerData.onButtonChangedObserver) {\n controllerData.teleportationComponent.onButtonStateChangedObservable.remove(controllerData.onButtonChangedObserver);\n }\n }\n // remove from the map\n delete this._controllers[xrControllerUniqueId];\n }\n _findClosestSnapPointWithRadius(realPosition, radius = this._options.snapToPositionRadius || 0.8) {\n let closestPoint = null;\n let closestDistance = Number.MAX_VALUE;\n if (this._snapToPositions.length) {\n const radiusSquared = radius * radius;\n this._snapToPositions.forEach((position) => {\n const dist = Vector3.DistanceSquared(position, realPosition);\n if (dist <= radiusSquared && dist < closestDistance) {\n closestDistance = dist;\n closestPoint = position;\n }\n });\n }\n return closestPoint;\n }\n _setTargetMeshPosition(pickInfo) {\n const newPosition = pickInfo.pickedPoint;\n if (!this._options.teleportationTargetMesh || !newPosition) {\n return;\n }\n const snapPosition = this._findClosestSnapPointWithRadius(newPosition);\n this._snappedToPoint = !!snapPosition;\n if (this.snapPointsOnly && !this._snappedToPoint && this._teleportationRingMaterial) {\n this._teleportationRingMaterial.diffuseColor.set(1.0, 0.3, 0.3);\n }\n else if (this.snapPointsOnly && this._snappedToPoint && this._teleportationRingMaterial) {\n this._teleportationRingMaterial.diffuseColor.set(0.3, 0.3, 1.0);\n }\n this._options.teleportationTargetMesh.position.copyFrom(snapPosition || newPosition);\n this._options.teleportationTargetMesh.position.y += 0.01;\n this.onTargetMeshPositionUpdatedObservable.notifyObservers(pickInfo);\n }\n _setTargetMeshVisibility(visible, force) {\n if (!this._options.teleportationTargetMesh) {\n return;\n }\n if (this._options.teleportationTargetMesh.isVisible === visible && !force) {\n return;\n }\n this._options.teleportationTargetMesh.isVisible = visible;\n this._options.teleportationTargetMesh.getChildren(undefined, false).forEach((m) => {\n m.isVisible = visible;\n });\n if (!visible) {\n if (this._quadraticBezierCurve) {\n this._quadraticBezierCurve.dispose();\n this._quadraticBezierCurve = null;\n }\n if (this._selectionFeature) {\n this._selectionFeature.attach();\n }\n }\n else {\n if (this._selectionFeature) {\n this._selectionFeature.detach();\n }\n }\n }\n _disposeBezierCurve() {\n if (this._quadraticBezierCurve) {\n this._quadraticBezierCurve.dispose();\n this._quadraticBezierCurve = null;\n }\n }\n _showParabolicPath(pickInfo) {\n if (!pickInfo.pickedPoint || !this._currentTeleportationControllerId) {\n return;\n }\n const sceneToRenderTo = this._options.useUtilityLayer\n ? this._options.customUtilityLayerScene || UtilityLayerRenderer.DefaultUtilityLayer.utilityLayerScene\n : this._xrSessionManager.scene;\n const controllerData = this._controllers[this._currentTeleportationControllerId];\n const quadraticBezierVectors = Curve3.CreateQuadraticBezier(controllerData.xrController.pointer.absolutePosition, pickInfo.ray.origin, pickInfo.pickedPoint, 25);\n const color = controllerData.teleportationState.blocked ? this._blockedRayColor : undefined;\n const colorsArray = new Array(26).fill(color || this._cachedColor4White);\n if (!this._options.generateRayPathMesh) {\n this._quadraticBezierCurve = CreateLines(\"teleportation path line\", { points: quadraticBezierVectors.getPoints(), instance: this._quadraticBezierCurve, updatable: true, colors: colorsArray }, sceneToRenderTo);\n }\n else {\n this._quadraticBezierCurve = this._options.generateRayPathMesh(quadraticBezierVectors.getPoints(), pickInfo);\n }\n this._quadraticBezierCurve.isPickable = false;\n if (this._options.renderingGroupId !== undefined) {\n this._quadraticBezierCurve.renderingGroupId = this._options.renderingGroupId;\n }\n }\n _teleportForward(controllerId) {\n const controllerData = this._controllers[controllerId];\n if (!controllerData || !controllerData.teleportationState.forward || !this.teleportationEnabled) {\n return;\n }\n controllerData.teleportationState.forward = false;\n this._currentTeleportationControllerId = \"\";\n if (this.snapPointsOnly && !this._snappedToPoint) {\n return;\n }\n if (this.skipNextTeleportation) {\n this.skipNextTeleportation = false;\n return;\n }\n // do the movement forward here\n if (this._options.teleportationTargetMesh && this._options.teleportationTargetMesh.isVisible) {\n const height = this._options.xrInput.xrCamera.realWorldHeight;\n this._options.xrInput.xrCamera.onBeforeCameraTeleport.notifyObservers(this._options.xrInput.xrCamera.position);\n this._options.xrInput.xrCamera.position.copyFrom(this._options.teleportationTargetMesh.position);\n this._options.xrInput.xrCamera.position.y += height;\n Quaternion.FromEulerAngles(0, controllerData.teleportationState.currentRotation - (this._xrSessionManager.scene.useRightHandedSystem ? Math.PI : 0), 0).multiplyToRef(this._options.xrInput.xrCamera.rotationQuaternion, this._options.xrInput.xrCamera.rotationQuaternion);\n this._options.xrInput.xrCamera.onAfterCameraTeleport.notifyObservers(this._options.xrInput.xrCamera.position);\n }\n }\n }\n /**\n * The module's name\n */\n WebXRMotionControllerTeleportation.Name = WebXRFeatureName.TELEPORTATION;\n /**\n * The (Babylon) version of this module.\n * This is an integer representing the implementation version.\n * This number does not correspond to the webxr specs version\n */\n WebXRMotionControllerTeleportation.Version = 1;\n WebXRFeaturesManager.AddWebXRFeature(WebXRMotionControllerTeleportation.Name, (xrSessionManager, options) => {\n return () => new WebXRMotionControllerTeleportation(xrSessionManager, options);\n }, WebXRMotionControllerTeleportation.Version, true);\n\n /**\n * Default experience which provides a similar setup to the previous webVRExperience\n */\n class WebXRDefaultExperience {\n constructor() { }\n /**\n * Creates the default xr experience\n * @param scene scene\n * @param options options for basic configuration\n * @returns resulting WebXRDefaultExperience\n */\n static CreateAsync(scene, options = {}) {\n const result = new WebXRDefaultExperience();\n scene.onDisposeObservable.addOnce(() => {\n result.dispose();\n });\n // init the UI right after construction\n if (!options.disableDefaultUI) {\n const uiOptions = Object.assign({ renderTarget: result.renderTarget }, (options.uiOptions || {}));\n if (options.optionalFeatures) {\n if (typeof options.optionalFeatures === \"boolean\") {\n uiOptions.optionalFeatures = [\"hit-test\", \"anchors\", \"plane-detection\", \"hand-tracking\"];\n }\n else {\n uiOptions.optionalFeatures = options.optionalFeatures;\n }\n }\n result.enterExitUI = new WebXREnterExitUI(scene, uiOptions);\n }\n // Create base experience\n return WebXRExperienceHelper.CreateAsync(scene)\n .then((xrHelper) => {\n result.baseExperience = xrHelper;\n if (options.ignoreNativeCameraTransformation) {\n result.baseExperience.camera.compensateOnFirstFrame = false;\n }\n // Add controller support\n result.input = new WebXRInput(xrHelper.sessionManager, xrHelper.camera, Object.assign({ controllerOptions: {\n renderingGroupId: options.renderingGroupId,\n } }, (options.inputOptions || {})));\n if (!options.disablePointerSelection) {\n // Add default pointer selection\n const pointerSelectionOptions = Object.assign(Object.assign({}, options.pointerSelectionOptions), { xrInput: result.input, renderingGroupId: options.renderingGroupId });\n result.pointerSelection = (result.baseExperience.featuresManager.enableFeature(WebXRControllerPointerSelection.Name, options.useStablePlugins ? \"stable\" : \"latest\", pointerSelectionOptions));\n if (!options.disableTeleportation) {\n // Add default teleportation, including rotation\n result.teleportation = result.baseExperience.featuresManager.enableFeature(WebXRMotionControllerTeleportation.Name, options.useStablePlugins ? \"stable\" : \"latest\", Object.assign({ floorMeshes: options.floorMeshes, xrInput: result.input, renderingGroupId: options.renderingGroupId }, options.teleportationOptions));\n result.teleportation.setSelectionFeature(result.pointerSelection);\n }\n }\n if (!options.disableNearInteraction) {\n // Add default pointer selection\n result.nearInteraction = result.baseExperience.featuresManager.enableFeature(WebXRNearInteraction.Name, options.useStablePlugins ? \"stable\" : \"latest\", Object.assign({ xrInput: result.input, farInteractionFeature: result.pointerSelection, renderingGroupId: options.renderingGroupId, useUtilityLayer: true, enableNearInteractionOnAllControllers: true }, options.nearInteractionOptions));\n }\n // Create the WebXR output target\n result.renderTarget = result.baseExperience.sessionManager.getWebXRRenderTarget(options.outputCanvasOptions);\n if (!options.disableDefaultUI) {\n // Create ui for entering/exiting xr\n return result.enterExitUI.setHelperAsync(result.baseExperience, result.renderTarget);\n }\n else {\n return;\n }\n })\n .then(() => {\n return result;\n })\n .catch((error) => {\n Logger.Error(\"Error initializing XR\");\n Logger.Error(error);\n return result;\n });\n }\n /**\n * Disposes of the experience helper\n */\n dispose() {\n if (this.baseExperience) {\n this.baseExperience.dispose();\n }\n if (this.input) {\n this.input.dispose();\n }\n if (this.enterExitUI) {\n this.enterExitUI.dispose();\n }\n if (this.renderTarget) {\n this.renderTarget.dispose();\n }\n }\n }\n\n Scene.prototype.createDefaultLight = function (replace = false) {\n // Dispose existing light in replace mode.\n if (replace) {\n if (this.lights) {\n for (let i = 0; i < this.lights.length; i++) {\n this.lights[i].dispose();\n }\n }\n }\n // Light\n if (this.lights.length === 0) {\n new HemisphericLight(\"default light\", Vector3.Up(), this);\n }\n };\n Scene.prototype.createDefaultCamera = function (createArcRotateCamera = false, replace = false, attachCameraControls = false) {\n // Dispose existing camera in replace mode.\n if (replace) {\n if (this.activeCamera) {\n this.activeCamera.dispose();\n this.activeCamera = null;\n }\n }\n // Camera\n if (!this.activeCamera) {\n const worldExtends = this.getWorldExtends((mesh) => mesh.isVisible && mesh.isEnabled());\n const worldSize = worldExtends.max.subtract(worldExtends.min);\n const worldCenter = worldExtends.min.add(worldSize.scale(0.5));\n let camera;\n let radius = worldSize.length() * 1.5;\n // empty scene scenario!\n if (!isFinite(radius)) {\n radius = 1;\n worldCenter.copyFromFloats(0, 0, 0);\n }\n if (createArcRotateCamera) {\n const arcRotateCamera = new ArcRotateCamera(\"default camera\", -(Math.PI / 2), Math.PI / 2, radius, worldCenter, this);\n arcRotateCamera.lowerRadiusLimit = radius * 0.01;\n arcRotateCamera.wheelPrecision = 100 / radius;\n camera = arcRotateCamera;\n }\n else {\n const freeCamera = new FreeCamera(\"default camera\", new Vector3(worldCenter.x, worldCenter.y, -radius), this);\n freeCamera.setTarget(worldCenter);\n camera = freeCamera;\n }\n camera.minZ = radius * 0.01;\n camera.maxZ = radius * 1000;\n camera.speed = radius * 0.2;\n this.activeCamera = camera;\n if (attachCameraControls) {\n camera.attachControl();\n }\n }\n };\n Scene.prototype.createDefaultCameraOrLight = function (createArcRotateCamera = false, replace = false, attachCameraControls = false) {\n this.createDefaultLight(replace);\n this.createDefaultCamera(createArcRotateCamera, replace, attachCameraControls);\n };\n Scene.prototype.createDefaultSkybox = function (environmentTexture, pbr = false, scale = 1000, blur = 0, setGlobalEnvTexture = true) {\n if (!environmentTexture) {\n Logger.Warn(\"Can not create default skybox without environment texture.\");\n return null;\n }\n if (setGlobalEnvTexture) {\n if (environmentTexture) {\n this.environmentTexture = environmentTexture;\n }\n }\n // Skybox\n const hdrSkybox = CreateBox(\"hdrSkyBox\", { size: scale }, this);\n if (pbr) {\n const hdrSkyboxMaterial = new PBRMaterial(\"skyBox\", this);\n hdrSkyboxMaterial.backFaceCulling = false;\n hdrSkyboxMaterial.reflectionTexture = environmentTexture.clone();\n if (hdrSkyboxMaterial.reflectionTexture) {\n hdrSkyboxMaterial.reflectionTexture.coordinatesMode = Texture.SKYBOX_MODE;\n }\n hdrSkyboxMaterial.microSurface = 1.0 - blur;\n hdrSkyboxMaterial.disableLighting = true;\n hdrSkyboxMaterial.twoSidedLighting = true;\n hdrSkybox.material = hdrSkyboxMaterial;\n }\n else {\n const skyboxMaterial = new StandardMaterial(\"skyBox\", this);\n skyboxMaterial.backFaceCulling = false;\n skyboxMaterial.reflectionTexture = environmentTexture.clone();\n if (skyboxMaterial.reflectionTexture) {\n skyboxMaterial.reflectionTexture.coordinatesMode = Texture.SKYBOX_MODE;\n }\n skyboxMaterial.disableLighting = true;\n hdrSkybox.material = skyboxMaterial;\n }\n hdrSkybox.isPickable = false;\n hdrSkybox.infiniteDistance = true;\n hdrSkybox.ignoreCameraMaxZ = true;\n return hdrSkybox;\n };\n Scene.prototype.createDefaultEnvironment = function (options) {\n if (EnvironmentHelper) {\n return new EnvironmentHelper(options, this);\n }\n return null;\n };\n Scene.prototype.createDefaultVRExperience = function (webVROptions = {}) {\n return new VRExperienceHelper(this, webVROptions);\n };\n Scene.prototype.createDefaultXRExperienceAsync = function (options = {}) {\n return WebXRDefaultExperience.CreateAsync(this, options).then((helper) => {\n return helper;\n });\n };\n\n // Do not edit.\n const name$1_ = \"glowMapGenerationPixelShader\";\n const shader$1_ = `#if defined(DIFFUSE_ISLINEAR) || defined(EMISSIVE_ISLINEAR)\n#include\n#endif\n#ifdef DIFFUSE\nvarying vec2 vUVDiffuse;uniform sampler2D diffuseSampler;\n#endif\n#ifdef OPACITY\nvarying vec2 vUVOpacity;uniform sampler2D opacitySampler;uniform float opacityIntensity;\n#endif\n#ifdef EMISSIVE\nvarying vec2 vUVEmissive;uniform sampler2D emissiveSampler;\n#endif\n#ifdef VERTEXALPHA\nvarying vec4 vColor;\n#endif\nuniform vec4 glowColor;uniform float glowIntensity;\n#include\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{\n#include\nvec4 finalColor=glowColor;\n#ifdef DIFFUSE\nvec4 albedoTexture=texture2D(diffuseSampler,vUVDiffuse);\n#ifdef DIFFUSE_ISLINEAR\nalbedoTexture=toGammaSpace(albedoTexture);\n#endif\n#ifdef GLOW\nfinalColor.a*=albedoTexture.a;\n#endif\n#ifdef HIGHLIGHT\nfinalColor.a=albedoTexture.a;\n#endif\n#endif\n#ifdef OPACITY\nvec4 opacityMap=texture2D(opacitySampler,vUVOpacity);\n#ifdef OPACITYRGB\nfinalColor.a*=getLuminance(opacityMap.rgb);\n#else\nfinalColor.a*=opacityMap.a;\n#endif\nfinalColor.a*=opacityIntensity;\n#endif\n#ifdef VERTEXALPHA\nfinalColor.a*=vColor.a;\n#endif\n#ifdef ALPHATEST\nif (finalColor.a\n#include\n#include\n#include[0..maxSimultaneousMorphTargets]\n#include\n#include\nuniform mat4 viewProjection;varying vec4 vPosition;\n#ifdef UV1\nattribute vec2 uv;\n#endif\n#ifdef UV2\nattribute vec2 uv2;\n#endif\n#ifdef DIFFUSE\nvarying vec2 vUVDiffuse;uniform mat4 diffuseMatrix;\n#endif\n#ifdef OPACITY\nvarying vec2 vUVOpacity;uniform mat4 opacityMatrix;\n#endif\n#ifdef EMISSIVE\nvarying vec2 vUVEmissive;uniform mat4 emissiveMatrix;\n#endif\n#ifdef VERTEXALPHA\nattribute vec4 color;varying vec4 vColor;\n#endif\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void)\n{vec3 positionUpdated=position;\n#ifdef UV1\nvec2 uvUpdated=uv;\n#endif\n#include\n#include[0..maxSimultaneousMorphTargets]\n#include\n#include\n#include\nvec4 worldPos=finalWorld*vec4(positionUpdated,1.0);\n#ifdef CUBEMAP\nvPosition=worldPos;gl_Position=viewProjection*finalWorld*vec4(position,1.0);\n#else\nvPosition=viewProjection*worldPos;gl_Position=vPosition;\n#endif\n#ifdef DIFFUSE\n#ifdef DIFFUSEUV1\nvUVDiffuse=vec2(diffuseMatrix*vec4(uvUpdated,1.0,0.0));\n#endif\n#ifdef DIFFUSEUV2\nvUVDiffuse=vec2(diffuseMatrix*vec4(uv2,1.0,0.0));\n#endif\n#endif\n#ifdef OPACITY\n#ifdef OPACITYUV1\nvUVOpacity=vec2(opacityMatrix*vec4(uvUpdated,1.0,0.0));\n#endif\n#ifdef OPACITYUV2\nvUVOpacity=vec2(opacityMatrix*vec4(uv2,1.0,0.0));\n#endif\n#endif\n#ifdef EMISSIVE\n#ifdef EMISSIVEUV1\nvUVEmissive=vec2(emissiveMatrix*vec4(uvUpdated,1.0,0.0));\n#endif\n#ifdef EMISSIVEUV2\nvUVEmissive=vec2(emissiveMatrix*vec4(uv2,1.0,0.0));\n#endif\n#endif\n#ifdef VERTEXALPHA\nvColor=color;\n#endif\n#include\n}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$1$] = shader$1$;\n\n /**\n * The effect layer Helps adding post process effect blended with the main pass.\n *\n * This can be for instance use to generate glow or highlight effects on the scene.\n *\n * The effect layer class can not be used directly and is intented to inherited from to be\n * customized per effects.\n */\n class EffectLayer {\n /**\n * Gets the camera attached to the layer.\n */\n get camera() {\n return this._effectLayerOptions.camera;\n }\n /**\n * Gets the rendering group id the layer should render in.\n */\n get renderingGroupId() {\n return this._effectLayerOptions.renderingGroupId;\n }\n set renderingGroupId(renderingGroupId) {\n this._effectLayerOptions.renderingGroupId = renderingGroupId;\n }\n /**\n * Gets the main texture where the effect is rendered\n */\n get mainTexture() {\n return this._mainTexture;\n }\n /**\n * Sets a specific material to be used to render a mesh/a list of meshes in the layer\n * @param mesh mesh or array of meshes\n * @param material material to use by the layer when rendering the mesh(es). If undefined is passed, the specific material created by the layer will be used.\n */\n setMaterialForRendering(mesh, material) {\n this._mainTexture.setMaterialForRendering(mesh, material);\n if (Array.isArray(mesh)) {\n for (let i = 0; i < mesh.length; ++i) {\n const currentMesh = mesh[i];\n if (!material) {\n delete this._materialForRendering[currentMesh.uniqueId];\n }\n else {\n this._materialForRendering[currentMesh.uniqueId] = [currentMesh, material];\n }\n }\n }\n else {\n if (!material) {\n delete this._materialForRendering[mesh.uniqueId];\n }\n else {\n this._materialForRendering[mesh.uniqueId] = [mesh, material];\n }\n }\n }\n /**\n * Gets the intensity of the effect for a specific mesh.\n * @param mesh The mesh to get the effect intensity for\n * @returns The intensity of the effect for the mesh\n */\n getEffectIntensity(mesh) {\n var _a;\n return (_a = this._effectIntensity[mesh.uniqueId]) !== null && _a !== void 0 ? _a : 1;\n }\n /**\n * Sets the intensity of the effect for a specific mesh.\n * @param mesh The mesh to set the effect intensity for\n * @param intensity The intensity of the effect for the mesh\n */\n setEffectIntensity(mesh, intensity) {\n this._effectIntensity[mesh.uniqueId] = intensity;\n }\n /**\n * Instantiates a new effect Layer and references it in the scene.\n * @param name The name of the layer\n * @param scene The scene to use the layer in\n */\n constructor(\n /** The Friendly of the effect in the scene */\n name, scene) {\n this._vertexBuffers = {};\n this._maxSize = 0;\n this._mainTextureDesiredSize = { width: 0, height: 0 };\n this._shouldRender = true;\n this._postProcesses = [];\n this._textures = [];\n this._emissiveTextureAndColor = { texture: null, color: new Color4() };\n this._effectIntensity = {};\n /**\n * The clear color of the texture used to generate the glow map.\n */\n this.neutralColor = new Color4();\n /**\n * Specifies whether the highlight layer is enabled or not.\n */\n this.isEnabled = true;\n /**\n * Specifies if the bounding boxes should be rendered normally or if they should undergo the effect of the layer\n */\n this.disableBoundingBoxesFromEffectLayer = false;\n /**\n * An event triggered when the effect layer has been disposed.\n */\n this.onDisposeObservable = new Observable$1();\n /**\n * An event triggered when the effect layer is about rendering the main texture with the glowy parts.\n */\n this.onBeforeRenderMainTextureObservable = new Observable$1();\n /**\n * An event triggered when the generated texture is being merged in the scene.\n */\n this.onBeforeComposeObservable = new Observable$1();\n /**\n * An event triggered when the mesh is rendered into the effect render target.\n */\n this.onBeforeRenderMeshToEffect = new Observable$1();\n /**\n * An event triggered after the mesh has been rendered into the effect render target.\n */\n this.onAfterRenderMeshToEffect = new Observable$1();\n /**\n * An event triggered when the generated texture has been merged in the scene.\n */\n this.onAfterComposeObservable = new Observable$1();\n /**\n * An event triggered when the effect layer changes its size.\n */\n this.onSizeChangedObservable = new Observable$1();\n this._materialForRendering = {};\n this.name = name;\n this._scene = scene || EngineStore.LastCreatedScene;\n EffectLayer._SceneComponentInitialization(this._scene);\n this._engine = this._scene.getEngine();\n this._maxSize = this._engine.getCaps().maxTextureSize;\n this._scene.effectLayers.push(this);\n this._mergeDrawWrapper = [];\n // Generate Buffers\n this._generateIndexBuffer();\n this._generateVertexBuffer();\n }\n /**\n * Number of times _internalRender will be called. Some effect layers need to render the mesh several times, so they should override this method with the number of times the mesh should be rendered\n * @returns Number of times a mesh must be rendered in the layer\n */\n _numInternalDraws() {\n return 1;\n }\n /**\n * Initializes the effect layer with the required options.\n * @param options Sets of none mandatory options to use with the layer (see IEffectLayerOptions for more information)\n */\n _init(options) {\n // Adapt options\n this._effectLayerOptions = Object.assign({ mainTextureRatio: 0.5, alphaBlendingMode: 2, camera: null, renderingGroupId: -1, mainTextureType: 0, generateStencilBuffer: false }, options);\n this._setMainTextureSize();\n this._createMainTexture();\n this._createTextureAndPostProcesses();\n }\n /**\n * Generates the index buffer of the full screen quad blending to the main canvas.\n */\n _generateIndexBuffer() {\n // Indices\n const indices = [];\n indices.push(0);\n indices.push(1);\n indices.push(2);\n indices.push(0);\n indices.push(2);\n indices.push(3);\n this._indexBuffer = this._engine.createIndexBuffer(indices);\n }\n /**\n * Generates the vertex buffer of the full screen quad blending to the main canvas.\n */\n _generateVertexBuffer() {\n // VBO\n const vertices = [];\n vertices.push(1, 1);\n vertices.push(-1, 1);\n vertices.push(-1, -1);\n vertices.push(1, -1);\n const vertexBuffer = new VertexBuffer(this._engine, vertices, VertexBuffer.PositionKind, false, false, 2);\n this._vertexBuffers[VertexBuffer.PositionKind] = vertexBuffer;\n }\n /**\n * Sets the main texture desired size which is the closest power of two\n * of the engine canvas size.\n */\n _setMainTextureSize() {\n if (this._effectLayerOptions.mainTextureFixedSize) {\n this._mainTextureDesiredSize.width = this._effectLayerOptions.mainTextureFixedSize;\n this._mainTextureDesiredSize.height = this._effectLayerOptions.mainTextureFixedSize;\n }\n else {\n this._mainTextureDesiredSize.width = this._engine.getRenderWidth() * this._effectLayerOptions.mainTextureRatio;\n this._mainTextureDesiredSize.height = this._engine.getRenderHeight() * this._effectLayerOptions.mainTextureRatio;\n this._mainTextureDesiredSize.width = this._engine.needPOTTextures\n ? Engine.GetExponentOfTwo(this._mainTextureDesiredSize.width, this._maxSize)\n : this._mainTextureDesiredSize.width;\n this._mainTextureDesiredSize.height = this._engine.needPOTTextures\n ? Engine.GetExponentOfTwo(this._mainTextureDesiredSize.height, this._maxSize)\n : this._mainTextureDesiredSize.height;\n }\n this._mainTextureDesiredSize.width = Math.floor(this._mainTextureDesiredSize.width);\n this._mainTextureDesiredSize.height = Math.floor(this._mainTextureDesiredSize.height);\n }\n /**\n * Creates the main texture for the effect layer.\n */\n _createMainTexture() {\n this._mainTexture = new RenderTargetTexture(\"EffectLayerMainRTT\", {\n width: this._mainTextureDesiredSize.width,\n height: this._mainTextureDesiredSize.height,\n }, this._scene, false, true, this._effectLayerOptions.mainTextureType, false, Texture.TRILINEAR_SAMPLINGMODE, true, this._effectLayerOptions.generateStencilBuffer);\n this._mainTexture.activeCamera = this._effectLayerOptions.camera;\n this._mainTexture.wrapU = Texture.CLAMP_ADDRESSMODE;\n this._mainTexture.wrapV = Texture.CLAMP_ADDRESSMODE;\n this._mainTexture.anisotropicFilteringLevel = 1;\n this._mainTexture.updateSamplingMode(Texture.BILINEAR_SAMPLINGMODE);\n this._mainTexture.renderParticles = false;\n this._mainTexture.renderList = null;\n this._mainTexture.ignoreCameraViewport = true;\n for (const id in this._materialForRendering) {\n const [mesh, material] = this._materialForRendering[id];\n this._mainTexture.setMaterialForRendering(mesh, material);\n }\n this._mainTexture.customIsReadyFunction = (mesh, refreshRate, preWarm) => {\n if ((preWarm || refreshRate === 0) && mesh.subMeshes) {\n for (let i = 0; i < mesh.subMeshes.length; ++i) {\n const subMesh = mesh.subMeshes[i];\n const material = subMesh.getMaterial();\n const renderingMesh = subMesh.getRenderingMesh();\n if (!material) {\n continue;\n }\n const batch = renderingMesh._getInstancesRenderList(subMesh._id, !!subMesh.getReplacementMesh());\n const hardwareInstancedRendering = batch.hardwareInstancedRendering[subMesh._id] || renderingMesh.hasThinInstances;\n this._setEmissiveTextureAndColor(renderingMesh, subMesh, material);\n if (!this._isReady(subMesh, hardwareInstancedRendering, this._emissiveTextureAndColor.texture)) {\n return false;\n }\n }\n }\n return true;\n };\n // Custom render function\n this._mainTexture.customRenderFunction = (opaqueSubMeshes, alphaTestSubMeshes, transparentSubMeshes, depthOnlySubMeshes) => {\n this.onBeforeRenderMainTextureObservable.notifyObservers(this);\n let index;\n const engine = this._scene.getEngine();\n if (depthOnlySubMeshes.length) {\n engine.setColorWrite(false);\n for (index = 0; index < depthOnlySubMeshes.length; index++) {\n this._renderSubMesh(depthOnlySubMeshes.data[index]);\n }\n engine.setColorWrite(true);\n }\n for (index = 0; index < opaqueSubMeshes.length; index++) {\n this._renderSubMesh(opaqueSubMeshes.data[index]);\n }\n for (index = 0; index < alphaTestSubMeshes.length; index++) {\n this._renderSubMesh(alphaTestSubMeshes.data[index]);\n }\n const previousAlphaMode = engine.getAlphaMode();\n for (index = 0; index < transparentSubMeshes.length; index++) {\n this._renderSubMesh(transparentSubMeshes.data[index], true);\n }\n engine.setAlphaMode(previousAlphaMode);\n };\n this._mainTexture.onClearObservable.add((engine) => {\n engine.clear(this.neutralColor, true, true, true);\n });\n // Prevent package size in es6 (getBoundingBoxRenderer might not be present)\n if (this._scene.getBoundingBoxRenderer) {\n const boundingBoxRendererEnabled = this._scene.getBoundingBoxRenderer().enabled;\n this._mainTexture.onBeforeBindObservable.add(() => {\n this._scene.getBoundingBoxRenderer().enabled = !this.disableBoundingBoxesFromEffectLayer && boundingBoxRendererEnabled;\n });\n this._mainTexture.onAfterUnbindObservable.add(() => {\n this._scene.getBoundingBoxRenderer().enabled = boundingBoxRendererEnabled;\n });\n }\n }\n /**\n * Adds specific effects defines.\n * @param defines The defines to add specifics to.\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _addCustomEffectDefines(defines) {\n // Nothing to add by default.\n }\n /**\n * Checks for the readiness of the element composing the layer.\n * @param subMesh the mesh to check for\n * @param useInstances specify whether or not to use instances to render the mesh\n * @param emissiveTexture the associated emissive texture used to generate the glow\n * @returns true if ready otherwise, false\n */\n _isReady(subMesh, useInstances, emissiveTexture) {\n var _a;\n const engine = this._scene.getEngine();\n const mesh = subMesh.getMesh();\n const renderingMaterial = (_a = mesh._internalAbstractMeshDataInfo._materialForRenderPass) === null || _a === void 0 ? void 0 : _a[engine.currentRenderPassId];\n if (renderingMaterial) {\n return renderingMaterial.isReadyForSubMesh(mesh, subMesh, useInstances);\n }\n const material = subMesh.getMaterial();\n if (!material) {\n return false;\n }\n if (this._useMeshMaterial(subMesh.getRenderingMesh())) {\n return material.isReadyForSubMesh(subMesh.getMesh(), subMesh, useInstances);\n }\n const defines = [];\n const attribs = [VertexBuffer.PositionKind];\n let uv1 = false;\n let uv2 = false;\n // Diffuse\n if (material) {\n const needAlphaTest = material.needAlphaTesting();\n const diffuseTexture = material.getAlphaTestTexture();\n const needAlphaBlendFromDiffuse = diffuseTexture && diffuseTexture.hasAlpha && (material.useAlphaFromDiffuseTexture || material._useAlphaFromAlbedoTexture);\n if (diffuseTexture && (needAlphaTest || needAlphaBlendFromDiffuse)) {\n defines.push(\"#define DIFFUSE\");\n if (mesh.isVerticesDataPresent(VertexBuffer.UV2Kind) && diffuseTexture.coordinatesIndex === 1) {\n defines.push(\"#define DIFFUSEUV2\");\n uv2 = true;\n }\n else if (mesh.isVerticesDataPresent(VertexBuffer.UVKind)) {\n defines.push(\"#define DIFFUSEUV1\");\n uv1 = true;\n }\n if (needAlphaTest) {\n defines.push(\"#define ALPHATEST\");\n defines.push(\"#define ALPHATESTVALUE 0.4\");\n }\n if (!diffuseTexture.gammaSpace) {\n defines.push(\"#define DIFFUSE_ISLINEAR\");\n }\n }\n const opacityTexture = material.opacityTexture;\n if (opacityTexture) {\n defines.push(\"#define OPACITY\");\n if (mesh.isVerticesDataPresent(VertexBuffer.UV2Kind) && opacityTexture.coordinatesIndex === 1) {\n defines.push(\"#define OPACITYUV2\");\n uv2 = true;\n }\n else if (mesh.isVerticesDataPresent(VertexBuffer.UVKind)) {\n defines.push(\"#define OPACITYUV1\");\n uv1 = true;\n }\n }\n }\n // Emissive\n if (emissiveTexture) {\n defines.push(\"#define EMISSIVE\");\n if (mesh.isVerticesDataPresent(VertexBuffer.UV2Kind) && emissiveTexture.coordinatesIndex === 1) {\n defines.push(\"#define EMISSIVEUV2\");\n uv2 = true;\n }\n else if (mesh.isVerticesDataPresent(VertexBuffer.UVKind)) {\n defines.push(\"#define EMISSIVEUV1\");\n uv1 = true;\n }\n if (!emissiveTexture.gammaSpace) {\n defines.push(\"#define EMISSIVE_ISLINEAR\");\n }\n }\n // Vertex\n if (mesh.useVertexColors && mesh.isVerticesDataPresent(VertexBuffer.ColorKind) && mesh.hasVertexAlpha && material.transparencyMode !== Material.MATERIAL_OPAQUE) {\n attribs.push(VertexBuffer.ColorKind);\n defines.push(\"#define VERTEXALPHA\");\n }\n if (uv1) {\n attribs.push(VertexBuffer.UVKind);\n defines.push(\"#define UV1\");\n }\n if (uv2) {\n attribs.push(VertexBuffer.UV2Kind);\n defines.push(\"#define UV2\");\n }\n // Bones\n const fallbacks = new EffectFallbacks();\n if (mesh.useBones && mesh.computeBonesUsingShaders) {\n attribs.push(VertexBuffer.MatricesIndicesKind);\n attribs.push(VertexBuffer.MatricesWeightsKind);\n if (mesh.numBoneInfluencers > 4) {\n attribs.push(VertexBuffer.MatricesIndicesExtraKind);\n attribs.push(VertexBuffer.MatricesWeightsExtraKind);\n }\n defines.push(\"#define NUM_BONE_INFLUENCERS \" + mesh.numBoneInfluencers);\n const skeleton = mesh.skeleton;\n if (skeleton && skeleton.isUsingTextureForMatrices) {\n defines.push(\"#define BONETEXTURE\");\n }\n else {\n defines.push(\"#define BonesPerMesh \" + (skeleton ? skeleton.bones.length + 1 : 0));\n }\n if (mesh.numBoneInfluencers > 0) {\n fallbacks.addCPUSkinningFallback(0, mesh);\n }\n }\n else {\n defines.push(\"#define NUM_BONE_INFLUENCERS 0\");\n }\n // Morph targets\n const manager = mesh.morphTargetManager;\n let morphInfluencers = 0;\n if (manager) {\n if (manager.numInfluencers > 0) {\n defines.push(\"#define MORPHTARGETS\");\n morphInfluencers = manager.numInfluencers;\n defines.push(\"#define NUM_MORPH_INFLUENCERS \" + morphInfluencers);\n if (manager.isUsingTextureForTargets) {\n defines.push(\"#define MORPHTARGETS_TEXTURE\");\n }\n MaterialHelper.PrepareAttributesForMorphTargetsInfluencers(attribs, mesh, morphInfluencers);\n }\n }\n // Instances\n if (useInstances) {\n defines.push(\"#define INSTANCES\");\n MaterialHelper.PushAttributesForInstances(attribs);\n if (subMesh.getRenderingMesh().hasThinInstances) {\n defines.push(\"#define THIN_INSTANCES\");\n }\n }\n // ClipPlanes\n prepareStringDefinesForClipPlanes(material, this._scene, defines);\n this._addCustomEffectDefines(defines);\n // Get correct effect\n const drawWrapper = subMesh._getDrawWrapper(undefined, true);\n const cachedDefines = drawWrapper.defines;\n const join = defines.join(\"\\n\");\n if (cachedDefines !== join) {\n const uniforms = [\n \"world\",\n \"mBones\",\n \"viewProjection\",\n \"glowColor\",\n \"morphTargetInfluences\",\n \"boneTextureWidth\",\n \"diffuseMatrix\",\n \"emissiveMatrix\",\n \"opacityMatrix\",\n \"opacityIntensity\",\n \"morphTargetTextureInfo\",\n \"morphTargetTextureIndices\",\n \"glowIntensity\",\n ];\n addClipPlaneUniforms(uniforms);\n drawWrapper.setEffect(this._engine.createEffect(\"glowMapGeneration\", attribs, uniforms, [\"diffuseSampler\", \"emissiveSampler\", \"opacitySampler\", \"boneSampler\", \"morphTargets\"], join, fallbacks, undefined, undefined, { maxSimultaneousMorphTargets: morphInfluencers }), join);\n }\n return drawWrapper.effect.isReady();\n }\n /**\n * Renders the glowing part of the scene by blending the blurred glowing meshes on top of the rendered scene.\n */\n render() {\n for (let i = 0; i < this._postProcesses.length; i++) {\n if (!this._postProcesses[i].isReady()) {\n return;\n }\n }\n const engine = this._scene.getEngine();\n const numDraws = this._numInternalDraws();\n // Check\n let isReady = true;\n for (let i = 0; i < numDraws; ++i) {\n let currentEffect = this._mergeDrawWrapper[i];\n if (!currentEffect) {\n currentEffect = this._mergeDrawWrapper[i] = new DrawWrapper(this._engine);\n currentEffect.setEffect(this._createMergeEffect());\n }\n isReady = isReady && currentEffect.effect.isReady();\n }\n if (!isReady) {\n return;\n }\n this.onBeforeComposeObservable.notifyObservers(this);\n const previousAlphaMode = engine.getAlphaMode();\n for (let i = 0; i < numDraws; ++i) {\n const currentEffect = this._mergeDrawWrapper[i];\n // Render\n engine.enableEffect(currentEffect);\n engine.setState(false);\n // VBOs\n engine.bindBuffers(this._vertexBuffers, this._indexBuffer, currentEffect.effect);\n // Go Blend.\n engine.setAlphaMode(this._effectLayerOptions.alphaBlendingMode);\n // Blends the map on the main canvas.\n this._internalRender(currentEffect.effect, i);\n }\n // Restore Alpha\n engine.setAlphaMode(previousAlphaMode);\n this.onAfterComposeObservable.notifyObservers(this);\n // Handle size changes.\n const size = this._mainTexture.getSize();\n this._setMainTextureSize();\n if ((size.width !== this._mainTextureDesiredSize.width || size.height !== this._mainTextureDesiredSize.height) &&\n this._mainTextureDesiredSize.width !== 0 &&\n this._mainTextureDesiredSize.height !== 0) {\n // Recreate RTT and post processes on size change.\n this.onSizeChangedObservable.notifyObservers(this);\n this._disposeTextureAndPostProcesses();\n this._createMainTexture();\n this._createTextureAndPostProcesses();\n }\n }\n /**\n * Determine if a given mesh will be used in the current effect.\n * @param mesh mesh to test\n * @returns true if the mesh will be used\n */\n hasMesh(mesh) {\n if (this.renderingGroupId === -1 || mesh.renderingGroupId === this.renderingGroupId) {\n return true;\n }\n return false;\n }\n /**\n * Returns true if the layer contains information to display, otherwise false.\n * @returns true if the glow layer should be rendered\n */\n shouldRender() {\n return this.isEnabled && this._shouldRender;\n }\n /**\n * Returns true if the mesh should render, otherwise false.\n * @param mesh The mesh to render\n * @returns true if it should render otherwise false\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _shouldRenderMesh(mesh) {\n return true;\n }\n /**\n * Returns true if the mesh can be rendered, otherwise false.\n * @param mesh The mesh to render\n * @param material The material used on the mesh\n * @returns true if it can be rendered otherwise false\n */\n _canRenderMesh(mesh, material) {\n return !material.needAlphaBlendingForMesh(mesh);\n }\n /**\n * Returns true if the mesh should render, otherwise false.\n * @returns true if it should render otherwise false\n */\n _shouldRenderEmissiveTextureForMesh() {\n return true;\n }\n /**\n * Renders the submesh passed in parameter to the generation map.\n * @param subMesh\n * @param enableAlphaMode\n */\n _renderSubMesh(subMesh, enableAlphaMode = false) {\n var _a, _b;\n if (!this.shouldRender()) {\n return;\n }\n const material = subMesh.getMaterial();\n const ownerMesh = subMesh.getMesh();\n const replacementMesh = subMesh.getReplacementMesh();\n const renderingMesh = subMesh.getRenderingMesh();\n const effectiveMesh = subMesh.getEffectiveMesh();\n const scene = this._scene;\n const engine = scene.getEngine();\n effectiveMesh._internalAbstractMeshDataInfo._isActiveIntermediate = false;\n if (!material) {\n return;\n }\n // Do not block in blend mode.\n if (!this._canRenderMesh(renderingMesh, material)) {\n return;\n }\n // Culling\n let sideOrientation = (_a = renderingMesh.overrideMaterialSideOrientation) !== null && _a !== void 0 ? _a : material.sideOrientation;\n const mainDeterminant = effectiveMesh._getWorldMatrixDeterminant();\n if (mainDeterminant < 0) {\n sideOrientation = sideOrientation === Material.ClockWiseSideOrientation ? Material.CounterClockWiseSideOrientation : Material.ClockWiseSideOrientation;\n }\n const reverse = sideOrientation === Material.ClockWiseSideOrientation;\n engine.setState(material.backFaceCulling, material.zOffset, undefined, reverse, material.cullBackFaces, undefined, material.zOffsetUnits);\n // Managing instances\n const batch = renderingMesh._getInstancesRenderList(subMesh._id, !!replacementMesh);\n if (batch.mustReturn) {\n return;\n }\n // Early Exit per mesh\n if (!this._shouldRenderMesh(renderingMesh)) {\n return;\n }\n const hardwareInstancedRendering = batch.hardwareInstancedRendering[subMesh._id] || renderingMesh.hasThinInstances;\n this._setEmissiveTextureAndColor(renderingMesh, subMesh, material);\n this.onBeforeRenderMeshToEffect.notifyObservers(ownerMesh);\n if (this._useMeshMaterial(renderingMesh)) {\n renderingMesh.render(subMesh, enableAlphaMode, replacementMesh || undefined);\n }\n else if (this._isReady(subMesh, hardwareInstancedRendering, this._emissiveTextureAndColor.texture)) {\n const renderingMaterial = (_b = effectiveMesh._internalAbstractMeshDataInfo._materialForRenderPass) === null || _b === void 0 ? void 0 : _b[engine.currentRenderPassId];\n let drawWrapper = subMesh._getDrawWrapper();\n if (!drawWrapper && renderingMaterial) {\n drawWrapper = renderingMaterial._getDrawWrapper();\n }\n if (!drawWrapper) {\n return;\n }\n const effect = drawWrapper.effect;\n engine.enableEffect(drawWrapper);\n if (!hardwareInstancedRendering) {\n renderingMesh._bind(subMesh, effect, material.fillMode);\n }\n if (!renderingMaterial) {\n effect.setMatrix(\"viewProjection\", scene.getTransformMatrix());\n effect.setMatrix(\"world\", effectiveMesh.getWorldMatrix());\n effect.setFloat4(\"glowColor\", this._emissiveTextureAndColor.color.r, this._emissiveTextureAndColor.color.g, this._emissiveTextureAndColor.color.b, this._emissiveTextureAndColor.color.a);\n }\n else {\n renderingMaterial.bindForSubMesh(effectiveMesh.getWorldMatrix(), effectiveMesh, subMesh);\n }\n if (!renderingMaterial) {\n const needAlphaTest = material.needAlphaTesting();\n const diffuseTexture = material.getAlphaTestTexture();\n const needAlphaBlendFromDiffuse = diffuseTexture && diffuseTexture.hasAlpha && (material.useAlphaFromDiffuseTexture || material._useAlphaFromAlbedoTexture);\n if (diffuseTexture && (needAlphaTest || needAlphaBlendFromDiffuse)) {\n effect.setTexture(\"diffuseSampler\", diffuseTexture);\n const textureMatrix = diffuseTexture.getTextureMatrix();\n if (textureMatrix) {\n effect.setMatrix(\"diffuseMatrix\", textureMatrix);\n }\n }\n const opacityTexture = material.opacityTexture;\n if (opacityTexture) {\n effect.setTexture(\"opacitySampler\", opacityTexture);\n effect.setFloat(\"opacityIntensity\", opacityTexture.level);\n const textureMatrix = opacityTexture.getTextureMatrix();\n if (textureMatrix) {\n effect.setMatrix(\"opacityMatrix\", textureMatrix);\n }\n }\n // Glow emissive only\n if (this._emissiveTextureAndColor.texture) {\n effect.setTexture(\"emissiveSampler\", this._emissiveTextureAndColor.texture);\n effect.setMatrix(\"emissiveMatrix\", this._emissiveTextureAndColor.texture.getTextureMatrix());\n }\n // Bones\n if (renderingMesh.useBones && renderingMesh.computeBonesUsingShaders && renderingMesh.skeleton) {\n const skeleton = renderingMesh.skeleton;\n if (skeleton.isUsingTextureForMatrices) {\n const boneTexture = skeleton.getTransformMatrixTexture(renderingMesh);\n if (!boneTexture) {\n return;\n }\n effect.setTexture(\"boneSampler\", boneTexture);\n effect.setFloat(\"boneTextureWidth\", 4.0 * (skeleton.bones.length + 1));\n }\n else {\n effect.setMatrices(\"mBones\", skeleton.getTransformMatrices(renderingMesh));\n }\n }\n // Morph targets\n MaterialHelper.BindMorphTargetParameters(renderingMesh, effect);\n if (renderingMesh.morphTargetManager && renderingMesh.morphTargetManager.isUsingTextureForTargets) {\n renderingMesh.morphTargetManager._bind(effect);\n }\n // Alpha mode\n if (enableAlphaMode) {\n engine.setAlphaMode(material.alphaMode);\n }\n // Intensity of effect\n effect.setFloat(\"glowIntensity\", this.getEffectIntensity(renderingMesh));\n // Clip planes\n bindClipPlane(effect, material, scene);\n }\n // Draw\n renderingMesh._processRendering(effectiveMesh, subMesh, effect, material.fillMode, batch, hardwareInstancedRendering, (isInstance, world) => effect.setMatrix(\"world\", world));\n }\n else {\n // Need to reset refresh rate of the main map\n this._mainTexture.resetRefreshCounter();\n }\n this.onAfterRenderMeshToEffect.notifyObservers(ownerMesh);\n }\n /**\n * Defines whether the current material of the mesh should be use to render the effect.\n * @param mesh defines the current mesh to render\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _useMeshMaterial(mesh) {\n return false;\n }\n /**\n * Rebuild the required buffers.\n * @internal Internal use only.\n */\n _rebuild() {\n const vb = this._vertexBuffers[VertexBuffer.PositionKind];\n if (vb) {\n vb._rebuild();\n }\n this._generateIndexBuffer();\n }\n /**\n * Dispose only the render target textures and post process.\n */\n _disposeTextureAndPostProcesses() {\n this._mainTexture.dispose();\n for (let i = 0; i < this._postProcesses.length; i++) {\n if (this._postProcesses[i]) {\n this._postProcesses[i].dispose();\n }\n }\n this._postProcesses = [];\n for (let i = 0; i < this._textures.length; i++) {\n if (this._textures[i]) {\n this._textures[i].dispose();\n }\n }\n this._textures = [];\n }\n /**\n * Dispose the highlight layer and free resources.\n */\n dispose() {\n const vertexBuffer = this._vertexBuffers[VertexBuffer.PositionKind];\n if (vertexBuffer) {\n vertexBuffer.dispose();\n this._vertexBuffers[VertexBuffer.PositionKind] = null;\n }\n if (this._indexBuffer) {\n this._scene.getEngine()._releaseBuffer(this._indexBuffer);\n this._indexBuffer = null;\n }\n for (const drawWrapper of this._mergeDrawWrapper) {\n drawWrapper.dispose();\n }\n this._mergeDrawWrapper = [];\n // Clean textures and post processes\n this._disposeTextureAndPostProcesses();\n // Remove from scene\n const index = this._scene.effectLayers.indexOf(this, 0);\n if (index > -1) {\n this._scene.effectLayers.splice(index, 1);\n }\n // Callback\n this.onDisposeObservable.notifyObservers(this);\n this.onDisposeObservable.clear();\n this.onBeforeRenderMainTextureObservable.clear();\n this.onBeforeComposeObservable.clear();\n this.onBeforeRenderMeshToEffect.clear();\n this.onAfterRenderMeshToEffect.clear();\n this.onAfterComposeObservable.clear();\n this.onSizeChangedObservable.clear();\n }\n /**\n * Gets the class name of the effect layer\n * @returns the string with the class name of the effect layer\n */\n getClassName() {\n return \"EffectLayer\";\n }\n /**\n * Creates an effect layer from parsed effect layer data\n * @param parsedEffectLayer defines effect layer data\n * @param scene defines the current scene\n * @param rootUrl defines the root URL containing the effect layer information\n * @returns a parsed effect Layer\n */\n static Parse(parsedEffectLayer, scene, rootUrl) {\n const effectLayerType = Tools.Instantiate(parsedEffectLayer.customType);\n return effectLayerType.Parse(parsedEffectLayer, scene, rootUrl);\n }\n }\n /**\n * @internal\n */\n EffectLayer._SceneComponentInitialization = (_) => {\n throw _WarnImport(\"EffectLayerSceneComponent\");\n };\n __decorate$1([\n serialize()\n ], EffectLayer.prototype, \"name\", void 0);\n __decorate$1([\n serializeAsColor4()\n ], EffectLayer.prototype, \"neutralColor\", void 0);\n __decorate$1([\n serialize()\n ], EffectLayer.prototype, \"isEnabled\", void 0);\n __decorate$1([\n serializeAsCameraReference()\n ], EffectLayer.prototype, \"camera\", null);\n __decorate$1([\n serialize()\n ], EffectLayer.prototype, \"renderingGroupId\", null);\n __decorate$1([\n serialize()\n ], EffectLayer.prototype, \"disableBoundingBoxesFromEffectLayer\", void 0);\n\n // Adds the parser to the scene parsers.\n AbstractScene.AddParser(SceneComponentConstants.NAME_EFFECTLAYER, (parsedData, scene, container, rootUrl) => {\n if (parsedData.effectLayers) {\n if (!container.effectLayers) {\n container.effectLayers = new Array();\n }\n for (let index = 0; index < parsedData.effectLayers.length; index++) {\n const effectLayer = EffectLayer.Parse(parsedData.effectLayers[index], scene, rootUrl);\n container.effectLayers.push(effectLayer);\n }\n }\n });\n AbstractScene.prototype.removeEffectLayer = function (toRemove) {\n const index = this.effectLayers.indexOf(toRemove);\n if (index !== -1) {\n this.effectLayers.splice(index, 1);\n }\n return index;\n };\n AbstractScene.prototype.addEffectLayer = function (newEffectLayer) {\n this.effectLayers.push(newEffectLayer);\n };\n /**\n * Defines the layer scene component responsible to manage any effect layers\n * in a given scene.\n */\n class EffectLayerSceneComponent {\n /**\n * Creates a new instance of the component for the given scene\n * @param scene Defines the scene to register the component in\n */\n constructor(scene) {\n /**\n * The component name helpful to identify the component in the list of scene components.\n */\n this.name = SceneComponentConstants.NAME_EFFECTLAYER;\n this._renderEffects = false;\n this._needStencil = false;\n this._previousStencilState = false;\n this.scene = scene || EngineStore.LastCreatedScene;\n if (!this.scene) {\n return;\n }\n this._engine = this.scene.getEngine();\n this.scene.effectLayers = new Array();\n }\n /**\n * Registers the component in a given scene\n */\n register() {\n this.scene._isReadyForMeshStage.registerStep(SceneComponentConstants.STEP_ISREADYFORMESH_EFFECTLAYER, this, this._isReadyForMesh);\n this.scene._cameraDrawRenderTargetStage.registerStep(SceneComponentConstants.STEP_CAMERADRAWRENDERTARGET_EFFECTLAYER, this, this._renderMainTexture);\n this.scene._beforeCameraDrawStage.registerStep(SceneComponentConstants.STEP_BEFORECAMERADRAW_EFFECTLAYER, this, this._setStencil);\n this.scene._afterRenderingGroupDrawStage.registerStep(SceneComponentConstants.STEP_AFTERRENDERINGGROUPDRAW_EFFECTLAYER_DRAW, this, this._drawRenderingGroup);\n this.scene._afterCameraDrawStage.registerStep(SceneComponentConstants.STEP_AFTERCAMERADRAW_EFFECTLAYER, this, this._setStencilBack);\n this.scene._afterCameraDrawStage.registerStep(SceneComponentConstants.STEP_AFTERCAMERADRAW_EFFECTLAYER_DRAW, this, this._drawCamera);\n }\n /**\n * Rebuilds the elements related to this component in case of\n * context lost for instance.\n */\n rebuild() {\n const layers = this.scene.effectLayers;\n for (const effectLayer of layers) {\n effectLayer._rebuild();\n }\n }\n /**\n * Serializes the component data to the specified json object\n * @param serializationObject The object to serialize to\n */\n serialize(serializationObject) {\n // Effect layers\n serializationObject.effectLayers = [];\n const layers = this.scene.effectLayers;\n for (const effectLayer of layers) {\n if (effectLayer.serialize) {\n serializationObject.effectLayers.push(effectLayer.serialize());\n }\n }\n }\n /**\n * Adds all the elements from the container to the scene\n * @param container the container holding the elements\n */\n addFromContainer(container) {\n if (!container.effectLayers) {\n return;\n }\n container.effectLayers.forEach((o) => {\n this.scene.addEffectLayer(o);\n });\n }\n /**\n * Removes all the elements in the container from the scene\n * @param container contains the elements to remove\n * @param dispose if the removed element should be disposed (default: false)\n */\n removeFromContainer(container, dispose) {\n if (!container.effectLayers) {\n return;\n }\n container.effectLayers.forEach((o) => {\n this.scene.removeEffectLayer(o);\n if (dispose) {\n o.dispose();\n }\n });\n }\n /**\n * Disposes the component and the associated resources.\n */\n dispose() {\n const layers = this.scene.effectLayers;\n while (layers.length) {\n layers[0].dispose();\n }\n }\n _isReadyForMesh(mesh, hardwareInstancedRendering) {\n const currentRenderPassId = this._engine.currentRenderPassId;\n const layers = this.scene.effectLayers;\n for (const layer of layers) {\n if (!layer.hasMesh(mesh)) {\n continue;\n }\n const renderTarget = layer._mainTexture;\n this._engine.currentRenderPassId = renderTarget.renderPassId;\n for (const subMesh of mesh.subMeshes) {\n if (!layer.isReady(subMesh, hardwareInstancedRendering)) {\n this._engine.currentRenderPassId = currentRenderPassId;\n return false;\n }\n }\n }\n this._engine.currentRenderPassId = currentRenderPassId;\n return true;\n }\n _renderMainTexture(camera) {\n this._renderEffects = false;\n this._needStencil = false;\n let needRebind = false;\n const layers = this.scene.effectLayers;\n if (layers && layers.length > 0) {\n this._previousStencilState = this._engine.getStencilBuffer();\n for (const effectLayer of layers) {\n if (effectLayer.shouldRender() &&\n (!effectLayer.camera ||\n (effectLayer.camera.cameraRigMode === Camera.RIG_MODE_NONE && camera === effectLayer.camera) ||\n (effectLayer.camera.cameraRigMode !== Camera.RIG_MODE_NONE && effectLayer.camera._rigCameras.indexOf(camera) > -1))) {\n this._renderEffects = true;\n this._needStencil = this._needStencil || effectLayer.needStencil();\n const renderTarget = effectLayer._mainTexture;\n if (renderTarget._shouldRender()) {\n this.scene.incrementRenderId();\n renderTarget.render(false, false);\n needRebind = true;\n }\n }\n }\n this.scene.incrementRenderId();\n }\n return needRebind;\n }\n _setStencil() {\n // Activate effect Layer stencil\n if (this._needStencil) {\n this._engine.setStencilBuffer(true);\n }\n }\n _setStencilBack() {\n // Restore effect Layer stencil\n if (this._needStencil) {\n this._engine.setStencilBuffer(this._previousStencilState);\n }\n }\n _draw(renderingGroupId) {\n if (this._renderEffects) {\n this._engine.setDepthBuffer(false);\n const layers = this.scene.effectLayers;\n for (let i = 0; i < layers.length; i++) {\n const effectLayer = layers[i];\n if (effectLayer.renderingGroupId === renderingGroupId) {\n if (effectLayer.shouldRender()) {\n effectLayer.render();\n }\n }\n }\n this._engine.setDepthBuffer(true);\n }\n }\n _drawCamera() {\n if (this._renderEffects) {\n this._draw(-1);\n }\n }\n _drawRenderingGroup(index) {\n if (!this.scene._isInIntermediateRendering() && this._renderEffects) {\n this._draw(index);\n }\n }\n }\n EffectLayer._SceneComponentInitialization = (scene) => {\n let component = scene._getComponent(SceneComponentConstants.NAME_EFFECTLAYER);\n if (!component) {\n component = new EffectLayerSceneComponent(scene);\n scene._addComponent(component);\n }\n };\n\n /**\n * Base implementation IShadowLight\n * It groups all the common behaviour in order to reduce duplication and better follow the DRY pattern.\n */\n class ShadowLight extends Light {\n constructor() {\n super(...arguments);\n this._needProjectionMatrixCompute = true;\n }\n _setPosition(value) {\n this._position = value;\n }\n /**\n * Sets the position the shadow will be casted from. Also use as the light position for both\n * point and spot lights.\n */\n get position() {\n return this._position;\n }\n /**\n * Sets the position the shadow will be casted from. Also use as the light position for both\n * point and spot lights.\n */\n set position(value) {\n this._setPosition(value);\n }\n _setDirection(value) {\n this._direction = value;\n }\n /**\n * In 2d mode (needCube being false), gets the direction used to cast the shadow.\n * Also use as the light direction on spot and directional lights.\n */\n get direction() {\n return this._direction;\n }\n /**\n * In 2d mode (needCube being false), sets the direction used to cast the shadow.\n * Also use as the light direction on spot and directional lights.\n */\n set direction(value) {\n this._setDirection(value);\n }\n /**\n * Gets the shadow projection clipping minimum z value.\n */\n get shadowMinZ() {\n return this._shadowMinZ;\n }\n /**\n * Sets the shadow projection clipping minimum z value.\n */\n set shadowMinZ(value) {\n this._shadowMinZ = value;\n this.forceProjectionMatrixCompute();\n }\n /**\n * Sets the shadow projection clipping maximum z value.\n */\n get shadowMaxZ() {\n return this._shadowMaxZ;\n }\n /**\n * Gets the shadow projection clipping maximum z value.\n */\n set shadowMaxZ(value) {\n this._shadowMaxZ = value;\n this.forceProjectionMatrixCompute();\n }\n /**\n * Computes the transformed information (transformedPosition and transformedDirection in World space) of the current light\n * @returns true if the information has been computed, false if it does not need to (no parenting)\n */\n computeTransformedInformation() {\n if (this.parent && this.parent.getWorldMatrix) {\n if (!this.transformedPosition) {\n this.transformedPosition = Vector3.Zero();\n }\n Vector3.TransformCoordinatesToRef(this.position, this.parent.getWorldMatrix(), this.transformedPosition);\n // In case the direction is present.\n if (this.direction) {\n if (!this.transformedDirection) {\n this.transformedDirection = Vector3.Zero();\n }\n Vector3.TransformNormalToRef(this.direction, this.parent.getWorldMatrix(), this.transformedDirection);\n }\n return true;\n }\n return false;\n }\n /**\n * Return the depth scale used for the shadow map.\n * @returns the depth scale.\n */\n getDepthScale() {\n return 50.0;\n }\n /**\n * Get the direction to use to render the shadow map. In case of cube texture, the face index can be passed.\n * @param faceIndex The index of the face we are computed the direction to generate shadow\n * @returns The set direction in 2d mode otherwise the direction to the cubemap face if needCube() is true\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n getShadowDirection(faceIndex) {\n return this.transformedDirection ? this.transformedDirection : this.direction;\n }\n /**\n * Returns the ShadowLight absolute position in the World.\n * @returns the position vector in world space\n */\n getAbsolutePosition() {\n return this.transformedPosition ? this.transformedPosition : this.position;\n }\n /**\n * Sets the ShadowLight direction toward the passed target.\n * @param target The point to target in local space\n * @returns the updated ShadowLight direction\n */\n setDirectionToTarget(target) {\n this.direction = Vector3.Normalize(target.subtract(this.position));\n return this.direction;\n }\n /**\n * Returns the light rotation in euler definition.\n * @returns the x y z rotation in local space.\n */\n getRotation() {\n this.direction.normalize();\n const xaxis = Vector3.Cross(this.direction, Axis.Y);\n const yaxis = Vector3.Cross(xaxis, this.direction);\n return Vector3.RotationFromAxis(xaxis, yaxis, this.direction);\n }\n /**\n * Returns whether or not the shadow generation require a cube texture or a 2d texture.\n * @returns true if a cube texture needs to be use\n */\n needCube() {\n return false;\n }\n /**\n * Detects if the projection matrix requires to be recomputed this frame.\n * @returns true if it requires to be recomputed otherwise, false.\n */\n needProjectionMatrixCompute() {\n return this._needProjectionMatrixCompute;\n }\n /**\n * Forces the shadow generator to recompute the projection matrix even if position and direction did not changed.\n */\n forceProjectionMatrixCompute() {\n this._needProjectionMatrixCompute = true;\n }\n /** @internal */\n _initCache() {\n super._initCache();\n this._cache.position = Vector3.Zero();\n }\n /** @internal */\n _isSynchronized() {\n if (!this._cache.position.equals(this.position)) {\n return false;\n }\n return true;\n }\n /**\n * Computes the world matrix of the node\n * @param force defines if the cache version should be invalidated forcing the world matrix to be created from scratch\n * @returns the world matrix\n */\n computeWorldMatrix(force) {\n if (!force && this.isSynchronized()) {\n this._currentRenderId = this.getScene().getRenderId();\n return this._worldMatrix;\n }\n this._updateCache();\n this._cache.position.copyFrom(this.position);\n if (!this._worldMatrix) {\n this._worldMatrix = Matrix.Identity();\n }\n Matrix.TranslationToRef(this.position.x, this.position.y, this.position.z, this._worldMatrix);\n if (this.parent && this.parent.getWorldMatrix) {\n this._worldMatrix.multiplyToRef(this.parent.getWorldMatrix(), this._worldMatrix);\n this._markSyncedWithParent();\n }\n // Cache the determinant\n this._worldMatrixDeterminantIsDirty = true;\n return this._worldMatrix;\n }\n /**\n * Gets the minZ used for shadow according to both the scene and the light.\n * @param activeCamera The camera we are returning the min for\n * @returns the depth min z\n */\n getDepthMinZ(activeCamera) {\n return this.shadowMinZ !== undefined ? this.shadowMinZ : activeCamera.minZ;\n }\n /**\n * Gets the maxZ used for shadow according to both the scene and the light.\n * @param activeCamera The camera we are returning the max for\n * @returns the depth max z\n */\n getDepthMaxZ(activeCamera) {\n return this.shadowMaxZ !== undefined ? this.shadowMaxZ : activeCamera.maxZ;\n }\n /**\n * Sets the shadow projection matrix in parameter to the generated projection matrix.\n * @param matrix The matrix to updated with the projection information\n * @param viewMatrix The transform matrix of the light\n * @param renderList The list of mesh to render in the map\n * @returns The current light\n */\n setShadowProjectionMatrix(matrix, viewMatrix, renderList) {\n if (this.customProjectionMatrixBuilder) {\n this.customProjectionMatrixBuilder(viewMatrix, renderList, matrix);\n }\n else {\n this._setDefaultShadowProjectionMatrix(matrix, viewMatrix, renderList);\n }\n return this;\n }\n /** @internal */\n _syncParentEnabledState() {\n super._syncParentEnabledState();\n if (!this.parent || !this.parent.getWorldMatrix) {\n this.transformedPosition = null;\n this.transformedDirection = null;\n }\n }\n }\n __decorate$1([\n serializeAsVector3()\n ], ShadowLight.prototype, \"position\", null);\n __decorate$1([\n serializeAsVector3()\n ], ShadowLight.prototype, \"direction\", null);\n __decorate$1([\n serialize()\n ], ShadowLight.prototype, \"shadowMinZ\", null);\n __decorate$1([\n serialize()\n ], ShadowLight.prototype, \"shadowMaxZ\", null);\n\n // Do not edit.\n const name$20 = \"bayerDitherFunctions\";\n const shader$20 = `float bayerDither2(vec2 _P) {return mod(2.0*_P.y+_P.x+1.0,4.0);}\nfloat bayerDither4(vec2 _P) {vec2 P1=mod(_P,2.0); \nvec2 P2=floor(0.5*mod(_P,4.0)); \nreturn 4.0*bayerDither2(P1)+bayerDither2(P2);}\nfloat bayerDither8(vec2 _P) {vec2 P1=mod(_P,2.0); \nvec2 P2=floor(0.5 *mod(_P,4.0)); \nvec2 P4=floor(0.25*mod(_P,8.0)); \nreturn 4.0*(4.0*bayerDither2(P1)+bayerDither2(P2))+bayerDither2(P4);}\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$20] = shader$20;\n\n // Do not edit.\n const name$21 = \"shadowMapFragmentExtraDeclaration\";\n const shader$21 = `#if SM_FLOAT==0\n#include\n#endif\n#if SM_SOFTTRANSPARENTSHADOW==1\n#include\nuniform float softTransparentShadowSM;\n#endif\nvarying float vDepthMetricSM;\n#if SM_USEDISTANCE==1\nuniform vec3 lightDataSM;varying vec3 vPositionWSM;\n#endif\nuniform vec3 biasAndScaleSM;uniform vec2 depthValuesSM;\n#if defined(SM_DEPTHCLAMP) && SM_DEPTHCLAMP==1\nvarying float zSM;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$21] = shader$21;\n\n // Do not edit.\n const name$22 = \"shadowMapFragment\";\n const shader$22 = `float depthSM=vDepthMetricSM;\n#if defined(SM_DEPTHCLAMP) && SM_DEPTHCLAMP==1\n#if SM_USEDISTANCE==1\ndepthSM=(length(vPositionWSM-lightDataSM)+depthValuesSM.x)/depthValuesSM.y+biasAndScaleSM.x;\n#else\n#ifdef USE_REVERSE_DEPTHBUFFER\ndepthSM=(-zSM+depthValuesSM.x)/depthValuesSM.y+biasAndScaleSM.x;\n#else\ndepthSM=(zSM+depthValuesSM.x)/depthValuesSM.y+biasAndScaleSM.x;\n#endif\n#endif\n#ifdef USE_REVERSE_DEPTHBUFFER\ngl_FragDepth=clamp(1.0-depthSM,0.0,1.0);\n#else\ngl_FragDepth=clamp(depthSM,0.0,1.0); \n#endif\n#elif SM_USEDISTANCE==1\ndepthSM=(length(vPositionWSM-lightDataSM)+depthValuesSM.x)/depthValuesSM.y+biasAndScaleSM.x;\n#endif\n#if SM_ESM==1\ndepthSM=clamp(exp(-min(87.,biasAndScaleSM.z*depthSM)),0.,1.);\n#endif\n#if SM_FLOAT==1\ngl_FragColor=vec4(depthSM,1.0,1.0,1.0);\n#else\ngl_FragColor=pack(depthSM);\n#endif\nreturn;`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$22] = shader$22;\n\n // Do not edit.\n const name$23 = \"shadowMapPixelShader\";\n const shader$23 = `#include\n#ifdef ALPHATEXTURE\nvarying vec2 vUV;uniform sampler2D diffuseSampler;\n#endif\n#include\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{\n#include\n#ifdef ALPHATEXTURE\nfloat alphaFromAlphaTexture=texture2D(diffuseSampler,vUV).a;\n#ifdef ALPHATESTVALUE\nif (alphaFromAlphaTexture=softTransparentShadowSM*alphaFromAlphaTexture) discard;\n#else\nif ((bayerDither8(floor(mod(gl_FragCoord.xy,8.0))))/64.0>=softTransparentShadowSM) discard;\n#endif\n#endif\n#include\n}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$23] = shader$23;\n\n // Do not edit.\n const name$24 = \"sceneVertexDeclaration\";\n const shader$24 = `uniform mat4 viewProjection;\n#ifdef MULTIVIEW\nuniform mat4 viewProjectionR;\n#endif\nuniform mat4 view;uniform mat4 projection;uniform vec4 vEyePosition;\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$24] = shader$24;\n\n // Do not edit.\n const name$25 = \"meshVertexDeclaration\";\n const shader$25 = `uniform mat4 world;uniform float visibility;\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$25] = shader$25;\n\n // Do not edit.\n const name$26 = \"shadowMapVertexDeclaration\";\n const shader$26 = `#include\n#include\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$26] = shader$26;\n\n // Do not edit.\n const name$27 = \"shadowMapUboDeclaration\";\n const shader$27 = `layout(std140,column_major) uniform;\n#include\n#include\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$27] = shader$27;\n\n // Do not edit.\n const name$28 = \"shadowMapVertexExtraDeclaration\";\n const shader$28 = `#if SM_NORMALBIAS==1\nuniform vec3 lightDataSM;\n#endif\nuniform vec3 biasAndScaleSM;uniform vec2 depthValuesSM;varying float vDepthMetricSM;\n#if SM_USEDISTANCE==1\nvarying vec3 vPositionWSM;\n#endif\n#if defined(SM_DEPTHCLAMP) && SM_DEPTHCLAMP==1\nvarying float zSM;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$28] = shader$28;\n\n // Do not edit.\n const name$29 = \"shadowMapVertexNormalBias\";\n const shader$29 = `#if SM_NORMALBIAS==1\n#if SM_DIRECTIONINLIGHTDATA==1\nvec3 worldLightDirSM=normalize(-lightDataSM.xyz);\n#else\nvec3 directionToLightSM=lightDataSM.xyz-worldPos.xyz;vec3 worldLightDirSM=normalize(directionToLightSM);\n#endif\nfloat ndlSM=dot(vNormalW,worldLightDirSM);float sinNLSM=sqrt(1.0-ndlSM*ndlSM);float normalBiasSM=biasAndScaleSM.y*sinNLSM;worldPos.xyz-=vNormalW*normalBiasSM;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$29] = shader$29;\n\n // Do not edit.\n const name$2a = \"shadowMapVertexMetric\";\n const shader$2a = `#if SM_USEDISTANCE==1\nvPositionWSM=worldPos.xyz;\n#endif\n#if SM_DEPTHTEXTURE==1\n#ifdef IS_NDC_HALF_ZRANGE\n#define BIASFACTOR 0.5\n#else\n#define BIASFACTOR 1.0\n#endif\n#ifdef USE_REVERSE_DEPTHBUFFER\ngl_Position.z-=biasAndScaleSM.x*gl_Position.w*BIASFACTOR;\n#else\ngl_Position.z+=biasAndScaleSM.x*gl_Position.w*BIASFACTOR;\n#endif\n#endif\n#if defined(SM_DEPTHCLAMP) && SM_DEPTHCLAMP==1\nzSM=gl_Position.z;gl_Position.z=0.0;\n#elif SM_USEDISTANCE==0\n#ifdef USE_REVERSE_DEPTHBUFFER\nvDepthMetricSM=(-gl_Position.z+depthValuesSM.x)/depthValuesSM.y+biasAndScaleSM.x;\n#else\nvDepthMetricSM=(gl_Position.z+depthValuesSM.x)/depthValuesSM.y+biasAndScaleSM.x;\n#endif\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$2a] = shader$2a;\n\n // Do not edit.\n const name$2b = \"shadowMapVertexShader\";\n const shader$2b = `attribute vec3 position;\n#ifdef NORMAL\nattribute vec3 normal;\n#endif\n#include\n#include\n#include\n#include[0..maxSimultaneousMorphTargets]\n#ifdef INSTANCES\nattribute vec4 world0;attribute vec4 world1;attribute vec4 world2;attribute vec4 world3;\n#endif\n#include\n#include<__decl__shadowMapVertex>\n#ifdef ALPHATEXTURE\nvarying vec2 vUV;uniform mat4 diffuseMatrix;\n#ifdef UV1\nattribute vec2 uv;\n#endif\n#ifdef UV2\nattribute vec2 uv2;\n#endif\n#endif\n#include\n#include\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void)\n{vec3 positionUpdated=position;\n#ifdef UV1\nvec2 uvUpdated=uv;\n#endif\n#ifdef NORMAL\nvec3 normalUpdated=normal;\n#endif\n#include\n#include[0..maxSimultaneousMorphTargets]\n#include\n#include\n#include\nvec4 worldPos=finalWorld*vec4(positionUpdated,1.0);\n#ifdef NORMAL\nmat3 normWorldSM=mat3(finalWorld);\n#if defined(INSTANCES) && defined(THIN_INSTANCES)\nvec3 vNormalW=normalUpdated/vec3(dot(normWorldSM[0],normWorldSM[0]),dot(normWorldSM[1],normWorldSM[1]),dot(normWorldSM[2],normWorldSM[2]));vNormalW=normalize(normWorldSM*vNormalW);\n#else\n#ifdef NONUNIFORMSCALING\nnormWorldSM=transposeMat3(inverseMat3(normWorldSM));\n#endif\nvec3 vNormalW=normalize(normWorldSM*normalUpdated);\n#endif\n#endif\n#include\ngl_Position=viewProjection*worldPos;\n#include\n#ifdef ALPHATEXTURE\n#ifdef UV1\nvUV=vec2(diffuseMatrix*vec4(uvUpdated,1.0,0.0));\n#endif\n#ifdef UV2\nvUV=vec2(diffuseMatrix*vec4(uv2,1.0,0.0));\n#endif\n#endif\n#include\n}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2b] = shader$2b;\n\n // Do not edit.\n const name$2c = \"depthBoxBlurPixelShader\";\n const shader$2c = `varying vec2 vUV;uniform sampler2D textureSampler;uniform vec2 screenSize;\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{vec4 colorDepth=vec4(0.0);for (int x=-OFFSET; x<=OFFSET; x++)\nfor (int y=-OFFSET; y<=OFFSET; y++)\ncolorDepth+=texture2D(textureSampler,vUV+vec2(x,y)/screenSize);gl_FragColor=(colorDepth/float((OFFSET*2+1)*(OFFSET*2+1)));}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2c] = shader$2c;\n\n // Do not edit.\n const name$2d = \"shadowMapFragmentSoftTransparentShadow\";\n const shader$2d = `#if SM_SOFTTRANSPARENTSHADOW==1\nif ((bayerDither8(floor(mod(gl_FragCoord.xy,8.0))))/64.0>=softTransparentShadowSM*alpha) discard;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$2d] = shader$2d;\n\n /**\n * Default implementation IShadowGenerator.\n * This is the main object responsible of generating shadows in the framework.\n * Documentation: https://doc.babylonjs.com/features/featuresDeepDive/lights/shadows\n */\n class ShadowGenerator {\n /**\n * Gets the bias: offset applied on the depth preventing acnea (in light direction).\n */\n get bias() {\n return this._bias;\n }\n /**\n * Sets the bias: offset applied on the depth preventing acnea (in light direction).\n */\n set bias(bias) {\n this._bias = bias;\n }\n /**\n * Gets the normalBias: offset applied on the depth preventing acnea (along side the normal direction and proportional to the light/normal angle).\n */\n get normalBias() {\n return this._normalBias;\n }\n /**\n * Sets the normalBias: offset applied on the depth preventing acnea (along side the normal direction and proportional to the light/normal angle).\n */\n set normalBias(normalBias) {\n this._normalBias = normalBias;\n }\n /**\n * Gets the blur box offset: offset applied during the blur pass.\n * Only useful if useKernelBlur = false\n */\n get blurBoxOffset() {\n return this._blurBoxOffset;\n }\n /**\n * Sets the blur box offset: offset applied during the blur pass.\n * Only useful if useKernelBlur = false\n */\n set blurBoxOffset(value) {\n if (this._blurBoxOffset === value) {\n return;\n }\n this._blurBoxOffset = value;\n this._disposeBlurPostProcesses();\n }\n /**\n * Gets the blur scale: scale of the blurred texture compared to the main shadow map.\n * 2 means half of the size.\n */\n get blurScale() {\n return this._blurScale;\n }\n /**\n * Sets the blur scale: scale of the blurred texture compared to the main shadow map.\n * 2 means half of the size.\n */\n set blurScale(value) {\n if (this._blurScale === value) {\n return;\n }\n this._blurScale = value;\n this._disposeBlurPostProcesses();\n }\n /**\n * Gets the blur kernel: kernel size of the blur pass.\n * Only useful if useKernelBlur = true\n */\n get blurKernel() {\n return this._blurKernel;\n }\n /**\n * Sets the blur kernel: kernel size of the blur pass.\n * Only useful if useKernelBlur = true\n */\n set blurKernel(value) {\n if (this._blurKernel === value) {\n return;\n }\n this._blurKernel = value;\n this._disposeBlurPostProcesses();\n }\n /**\n * Gets whether the blur pass is a kernel blur (if true) or box blur.\n * Only useful in filtered mode (useBlurExponentialShadowMap...)\n */\n get useKernelBlur() {\n return this._useKernelBlur;\n }\n /**\n * Sets whether the blur pass is a kernel blur (if true) or box blur.\n * Only useful in filtered mode (useBlurExponentialShadowMap...)\n */\n set useKernelBlur(value) {\n if (this._useKernelBlur === value) {\n return;\n }\n this._useKernelBlur = value;\n this._disposeBlurPostProcesses();\n }\n /**\n * Gets the depth scale used in ESM mode.\n */\n get depthScale() {\n return this._depthScale !== undefined ? this._depthScale : this._light.getDepthScale();\n }\n /**\n * Sets the depth scale used in ESM mode.\n * This can override the scale stored on the light.\n */\n set depthScale(value) {\n this._depthScale = value;\n }\n _validateFilter(filter) {\n return filter;\n }\n /**\n * Gets the current mode of the shadow generator (normal, PCF, ESM...).\n * The returned value is a number equal to one of the available mode defined in ShadowMap.FILTER_x like _FILTER_NONE\n */\n get filter() {\n return this._filter;\n }\n /**\n * Sets the current mode of the shadow generator (normal, PCF, ESM...).\n * The returned value is a number equal to one of the available mode defined in ShadowMap.FILTER_x like _FILTER_NONE\n */\n set filter(value) {\n value = this._validateFilter(value);\n // Blurring the cubemap is going to be too expensive. Reverting to unblurred version\n if (this._light.needCube()) {\n if (value === ShadowGenerator.FILTER_BLUREXPONENTIALSHADOWMAP) {\n this.useExponentialShadowMap = true;\n return;\n }\n else if (value === ShadowGenerator.FILTER_BLURCLOSEEXPONENTIALSHADOWMAP) {\n this.useCloseExponentialShadowMap = true;\n return;\n }\n // PCF on cubemap would also be expensive\n else if (value === ShadowGenerator.FILTER_PCF || value === ShadowGenerator.FILTER_PCSS) {\n this.usePoissonSampling = true;\n return;\n }\n }\n // Weblg1 fallback for PCF.\n if (value === ShadowGenerator.FILTER_PCF || value === ShadowGenerator.FILTER_PCSS) {\n if (!this._scene.getEngine()._features.supportShadowSamplers) {\n this.usePoissonSampling = true;\n return;\n }\n }\n if (this._filter === value) {\n return;\n }\n this._filter = value;\n this._disposeBlurPostProcesses();\n this._applyFilterValues();\n this._light._markMeshesAsLightDirty();\n }\n /**\n * Gets if the current filter is set to Poisson Sampling.\n */\n get usePoissonSampling() {\n return this.filter === ShadowGenerator.FILTER_POISSONSAMPLING;\n }\n /**\n * Sets the current filter to Poisson Sampling.\n */\n set usePoissonSampling(value) {\n const filter = this._validateFilter(ShadowGenerator.FILTER_POISSONSAMPLING);\n if (!value && this.filter !== ShadowGenerator.FILTER_POISSONSAMPLING) {\n return;\n }\n this.filter = value ? filter : ShadowGenerator.FILTER_NONE;\n }\n /**\n * Gets if the current filter is set to ESM.\n */\n get useExponentialShadowMap() {\n return this.filter === ShadowGenerator.FILTER_EXPONENTIALSHADOWMAP;\n }\n /**\n * Sets the current filter is to ESM.\n */\n set useExponentialShadowMap(value) {\n const filter = this._validateFilter(ShadowGenerator.FILTER_EXPONENTIALSHADOWMAP);\n if (!value && this.filter !== ShadowGenerator.FILTER_EXPONENTIALSHADOWMAP) {\n return;\n }\n this.filter = value ? filter : ShadowGenerator.FILTER_NONE;\n }\n /**\n * Gets if the current filter is set to filtered ESM.\n */\n get useBlurExponentialShadowMap() {\n return this.filter === ShadowGenerator.FILTER_BLUREXPONENTIALSHADOWMAP;\n }\n /**\n * Gets if the current filter is set to filtered ESM.\n */\n set useBlurExponentialShadowMap(value) {\n const filter = this._validateFilter(ShadowGenerator.FILTER_BLUREXPONENTIALSHADOWMAP);\n if (!value && this.filter !== ShadowGenerator.FILTER_BLUREXPONENTIALSHADOWMAP) {\n return;\n }\n this.filter = value ? filter : ShadowGenerator.FILTER_NONE;\n }\n /**\n * Gets if the current filter is set to \"close ESM\" (using the inverse of the\n * exponential to prevent steep falloff artifacts).\n */\n get useCloseExponentialShadowMap() {\n return this.filter === ShadowGenerator.FILTER_CLOSEEXPONENTIALSHADOWMAP;\n }\n /**\n * Sets the current filter to \"close ESM\" (using the inverse of the\n * exponential to prevent steep falloff artifacts).\n */\n set useCloseExponentialShadowMap(value) {\n const filter = this._validateFilter(ShadowGenerator.FILTER_CLOSEEXPONENTIALSHADOWMAP);\n if (!value && this.filter !== ShadowGenerator.FILTER_CLOSEEXPONENTIALSHADOWMAP) {\n return;\n }\n this.filter = value ? filter : ShadowGenerator.FILTER_NONE;\n }\n /**\n * Gets if the current filter is set to filtered \"close ESM\" (using the inverse of the\n * exponential to prevent steep falloff artifacts).\n */\n get useBlurCloseExponentialShadowMap() {\n return this.filter === ShadowGenerator.FILTER_BLURCLOSEEXPONENTIALSHADOWMAP;\n }\n /**\n * Sets the current filter to filtered \"close ESM\" (using the inverse of the\n * exponential to prevent steep falloff artifacts).\n */\n set useBlurCloseExponentialShadowMap(value) {\n const filter = this._validateFilter(ShadowGenerator.FILTER_BLURCLOSEEXPONENTIALSHADOWMAP);\n if (!value && this.filter !== ShadowGenerator.FILTER_BLURCLOSEEXPONENTIALSHADOWMAP) {\n return;\n }\n this.filter = value ? filter : ShadowGenerator.FILTER_NONE;\n }\n /**\n * Gets if the current filter is set to \"PCF\" (percentage closer filtering).\n */\n get usePercentageCloserFiltering() {\n return this.filter === ShadowGenerator.FILTER_PCF;\n }\n /**\n * Sets the current filter to \"PCF\" (percentage closer filtering).\n */\n set usePercentageCloserFiltering(value) {\n const filter = this._validateFilter(ShadowGenerator.FILTER_PCF);\n if (!value && this.filter !== ShadowGenerator.FILTER_PCF) {\n return;\n }\n this.filter = value ? filter : ShadowGenerator.FILTER_NONE;\n }\n /**\n * Gets the PCF or PCSS Quality.\n * Only valid if usePercentageCloserFiltering or usePercentageCloserFiltering is true.\n */\n get filteringQuality() {\n return this._filteringQuality;\n }\n /**\n * Sets the PCF or PCSS Quality.\n * Only valid if usePercentageCloserFiltering or usePercentageCloserFiltering is true.\n */\n set filteringQuality(filteringQuality) {\n if (this._filteringQuality === filteringQuality) {\n return;\n }\n this._filteringQuality = filteringQuality;\n this._disposeBlurPostProcesses();\n this._applyFilterValues();\n this._light._markMeshesAsLightDirty();\n }\n /**\n * Gets if the current filter is set to \"PCSS\" (contact hardening).\n */\n get useContactHardeningShadow() {\n return this.filter === ShadowGenerator.FILTER_PCSS;\n }\n /**\n * Sets the current filter to \"PCSS\" (contact hardening).\n */\n set useContactHardeningShadow(value) {\n const filter = this._validateFilter(ShadowGenerator.FILTER_PCSS);\n if (!value && this.filter !== ShadowGenerator.FILTER_PCSS) {\n return;\n }\n this.filter = value ? filter : ShadowGenerator.FILTER_NONE;\n }\n /**\n * Gets the Light Size (in shadow map uv unit) used in PCSS to determine the blocker search area and the penumbra size.\n * Using a ratio helps keeping shape stability independently of the map size.\n *\n * It does not account for the light projection as it was having too much\n * instability during the light setup or during light position changes.\n *\n * Only valid if useContactHardeningShadow is true.\n */\n get contactHardeningLightSizeUVRatio() {\n return this._contactHardeningLightSizeUVRatio;\n }\n /**\n * Sets the Light Size (in shadow map uv unit) used in PCSS to determine the blocker search area and the penumbra size.\n * Using a ratio helps keeping shape stability independently of the map size.\n *\n * It does not account for the light projection as it was having too much\n * instability during the light setup or during light position changes.\n *\n * Only valid if useContactHardeningShadow is true.\n */\n set contactHardeningLightSizeUVRatio(contactHardeningLightSizeUVRatio) {\n this._contactHardeningLightSizeUVRatio = contactHardeningLightSizeUVRatio;\n }\n /** Gets or sets the actual darkness of a shadow */\n get darkness() {\n return this._darkness;\n }\n set darkness(value) {\n this.setDarkness(value);\n }\n /**\n * Returns the darkness value (float). This can only decrease the actual darkness of a shadow.\n * 0 means strongest and 1 would means no shadow.\n * @returns the darkness.\n */\n getDarkness() {\n return this._darkness;\n }\n /**\n * Sets the darkness value (float). This can only decrease the actual darkness of a shadow.\n * @param darkness The darkness value 0 means strongest and 1 would means no shadow.\n * @returns the shadow generator allowing fluent coding.\n */\n setDarkness(darkness) {\n if (darkness >= 1.0) {\n this._darkness = 1.0;\n }\n else if (darkness <= 0.0) {\n this._darkness = 0.0;\n }\n else {\n this._darkness = darkness;\n }\n return this;\n }\n /** Gets or sets the ability to have transparent shadow */\n get transparencyShadow() {\n return this._transparencyShadow;\n }\n set transparencyShadow(value) {\n this.setTransparencyShadow(value);\n }\n /**\n * Sets the ability to have transparent shadow (boolean).\n * @param transparent True if transparent else False\n * @returns the shadow generator allowing fluent coding\n */\n setTransparencyShadow(transparent) {\n this._transparencyShadow = transparent;\n return this;\n }\n /**\n * Gets the main RTT containing the shadow map (usually storing depth from the light point of view).\n * @returns The render target texture if present otherwise, null\n */\n getShadowMap() {\n return this._shadowMap;\n }\n /**\n * Gets the RTT used during rendering (can be a blurred version of the shadow map or the shadow map itself).\n * @returns The render target texture if the shadow map is present otherwise, null\n */\n getShadowMapForRendering() {\n if (this._shadowMap2) {\n return this._shadowMap2;\n }\n return this._shadowMap;\n }\n /**\n * Gets the class name of that object\n * @returns \"ShadowGenerator\"\n */\n getClassName() {\n return ShadowGenerator.CLASSNAME;\n }\n /**\n * Helper function to add a mesh and its descendants to the list of shadow casters.\n * @param mesh Mesh to add\n * @param includeDescendants boolean indicating if the descendants should be added. Default to true\n * @returns the Shadow Generator itself\n */\n addShadowCaster(mesh, includeDescendants = true) {\n if (!this._shadowMap) {\n return this;\n }\n if (!this._shadowMap.renderList) {\n this._shadowMap.renderList = [];\n }\n if (this._shadowMap.renderList.indexOf(mesh) === -1) {\n this._shadowMap.renderList.push(mesh);\n }\n if (includeDescendants) {\n for (const childMesh of mesh.getChildMeshes()) {\n if (this._shadowMap.renderList.indexOf(childMesh) === -1) {\n this._shadowMap.renderList.push(childMesh);\n }\n }\n }\n return this;\n }\n /**\n * Helper function to remove a mesh and its descendants from the list of shadow casters\n * @param mesh Mesh to remove\n * @param includeDescendants boolean indicating if the descendants should be removed. Default to true\n * @returns the Shadow Generator itself\n */\n removeShadowCaster(mesh, includeDescendants = true) {\n if (!this._shadowMap || !this._shadowMap.renderList) {\n return this;\n }\n const index = this._shadowMap.renderList.indexOf(mesh);\n if (index !== -1) {\n this._shadowMap.renderList.splice(index, 1);\n }\n if (includeDescendants) {\n for (const child of mesh.getChildren()) {\n this.removeShadowCaster(child);\n }\n }\n return this;\n }\n /**\n * Returns the associated light object.\n * @returns the light generating the shadow\n */\n getLight() {\n return this._light;\n }\n _getCamera() {\n var _a;\n return (_a = this._camera) !== null && _a !== void 0 ? _a : this._scene.activeCamera;\n }\n /**\n * Gets or sets the size of the texture what stores the shadows\n */\n get mapSize() {\n return this._mapSize;\n }\n set mapSize(size) {\n this._mapSize = size;\n this._light._markMeshesAsLightDirty();\n this.recreateShadowMap();\n }\n /**\n * Creates a ShadowGenerator object.\n * A ShadowGenerator is the required tool to use the shadows.\n * Each light casting shadows needs to use its own ShadowGenerator.\n * Documentation : https://doc.babylonjs.com/features/featuresDeepDive/lights/shadows\n * @param mapSize The size of the texture what stores the shadows. Example : 1024.\n * @param light The light object generating the shadows.\n * @param usefullFloatFirst By default the generator will try to use half float textures but if you need precision (for self shadowing for instance), you can use this option to enforce full float texture.\n * @param camera Camera associated with this shadow generator (default: null). If null, takes the scene active camera at the time we need to access it\n * @param useRedTextureType Forces the generator to use a Red instead of a RGBA type for the shadow map texture format (default: false)\n */\n constructor(mapSize, light, usefullFloatFirst, camera, useRedTextureType) {\n /**\n * Observable triggered before the shadow is rendered. Can be used to update internal effect state\n */\n this.onBeforeShadowMapRenderObservable = new Observable$1();\n /**\n * Observable triggered after the shadow is rendered. Can be used to restore internal effect state\n */\n this.onAfterShadowMapRenderObservable = new Observable$1();\n /**\n * Observable triggered before a mesh is rendered in the shadow map.\n * Can be used to update internal effect state (that you can get from the onBeforeShadowMapRenderObservable)\n */\n this.onBeforeShadowMapRenderMeshObservable = new Observable$1();\n /**\n * Observable triggered after a mesh is rendered in the shadow map.\n * Can be used to update internal effect state (that you can get from the onAfterShadowMapRenderObservable)\n */\n this.onAfterShadowMapRenderMeshObservable = new Observable$1();\n this._bias = 0.00005;\n this._normalBias = 0;\n this._blurBoxOffset = 1;\n this._blurScale = 2;\n this._blurKernel = 1;\n this._useKernelBlur = false;\n this._filter = ShadowGenerator.FILTER_NONE;\n this._filteringQuality = ShadowGenerator.QUALITY_HIGH;\n this._contactHardeningLightSizeUVRatio = 0.1;\n this._darkness = 0;\n this._transparencyShadow = false;\n /**\n * Enables or disables shadows with varying strength based on the transparency\n * When it is enabled, the strength of the shadow is taken equal to mesh.visibility\n * If you enabled an alpha texture on your material, the alpha value red from the texture is also combined to compute the strength:\n * mesh.visibility * alphaTexture.a\n * The texture used is the diffuse by default, but it can be set to the opacity by setting useOpacityTextureForTransparentShadow\n * Note that by definition transparencyShadow must be set to true for enableSoftTransparentShadow to work!\n */\n this.enableSoftTransparentShadow = false;\n /**\n * If this is true, use the opacity texture's alpha channel for transparent shadows instead of the diffuse one\n */\n this.useOpacityTextureForTransparentShadow = false;\n /**\n * Controls the extent to which the shadows fade out at the edge of the frustum\n */\n this.frustumEdgeFalloff = 0;\n /**\n * If true the shadow map is generated by rendering the back face of the mesh instead of the front face.\n * This can help with self-shadowing as the geometry making up the back of objects is slightly offset.\n * It might on the other hand introduce peter panning.\n */\n this.forceBackFacesOnly = false;\n this._lightDirection = Vector3.Zero();\n this._viewMatrix = Matrix.Zero();\n this._projectionMatrix = Matrix.Zero();\n this._transformMatrix = Matrix.Zero();\n this._cachedPosition = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\n this._cachedDirection = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\n this._currentFaceIndex = 0;\n this._currentFaceIndexCache = 0;\n this._defaultTextureMatrix = Matrix.Identity();\n this._mapSize = mapSize;\n this._light = light;\n this._scene = light.getScene();\n this._camera = camera !== null && camera !== void 0 ? camera : null;\n this._useRedTextureType = !!useRedTextureType;\n let shadowGenerators = light._shadowGenerators;\n if (!shadowGenerators) {\n shadowGenerators = light._shadowGenerators = new Map();\n }\n shadowGenerators.set(this._camera, this);\n this.id = light.id;\n this._useUBO = this._scene.getEngine().supportsUniformBuffers;\n if (this._useUBO) {\n this._sceneUBOs = [];\n this._sceneUBOs.push(this._scene.createSceneUniformBuffer(`Scene for Shadow Generator (light \"${this._light.name}\")`));\n }\n ShadowGenerator._SceneComponentInitialization(this._scene);\n // Texture type fallback from float to int if not supported.\n const caps = this._scene.getEngine().getCaps();\n if (!usefullFloatFirst) {\n if (caps.textureHalfFloatRender && caps.textureHalfFloatLinearFiltering) {\n this._textureType = 2;\n }\n else if (caps.textureFloatRender && caps.textureFloatLinearFiltering) {\n this._textureType = 1;\n }\n else {\n this._textureType = 0;\n }\n }\n else {\n if (caps.textureFloatRender && caps.textureFloatLinearFiltering) {\n this._textureType = 1;\n }\n else if (caps.textureHalfFloatRender && caps.textureHalfFloatLinearFiltering) {\n this._textureType = 2;\n }\n else {\n this._textureType = 0;\n }\n }\n this._initializeGenerator();\n this._applyFilterValues();\n }\n _initializeGenerator() {\n this._light._markMeshesAsLightDirty();\n this._initializeShadowMap();\n }\n _createTargetRenderTexture() {\n const engine = this._scene.getEngine();\n if (engine._features.supportDepthStencilTexture) {\n this._shadowMap = new RenderTargetTexture(this._light.name + \"_shadowMap\", this._mapSize, this._scene, false, true, this._textureType, this._light.needCube(), undefined, false, false, undefined, this._useRedTextureType ? 6 : 5);\n this._shadowMap.createDepthStencilTexture(engine.useReverseDepthBuffer ? 516 : 513, true);\n }\n else {\n this._shadowMap = new RenderTargetTexture(this._light.name + \"_shadowMap\", this._mapSize, this._scene, false, true, this._textureType, this._light.needCube());\n }\n this._shadowMap.noPrePassRenderer = true;\n }\n _initializeShadowMap() {\n this._createTargetRenderTexture();\n if (this._shadowMap === null) {\n return;\n }\n this._shadowMap.wrapU = Texture.CLAMP_ADDRESSMODE;\n this._shadowMap.wrapV = Texture.CLAMP_ADDRESSMODE;\n this._shadowMap.anisotropicFilteringLevel = 1;\n this._shadowMap.updateSamplingMode(Texture.BILINEAR_SAMPLINGMODE);\n this._shadowMap.renderParticles = false;\n this._shadowMap.ignoreCameraViewport = true;\n if (this._storedUniqueId) {\n this._shadowMap.uniqueId = this._storedUniqueId;\n }\n // Custom render function.\n this._shadowMap.customRenderFunction = this._renderForShadowMap.bind(this);\n // Force the mesh is ready function to true as we are double checking it\n // in the custom render function. Also it prevents side effects and useless\n // shader variations in DEPTHPREPASS mode.\n this._shadowMap.customIsReadyFunction = () => {\n return true;\n };\n const engine = this._scene.getEngine();\n this._shadowMap.onBeforeBindObservable.add(() => {\n var _a;\n this._currentSceneUBO = this._scene.getSceneUniformBuffer();\n (_a = engine._debugPushGroup) === null || _a === void 0 ? void 0 : _a.call(engine, `shadow map generation for pass id ${engine.currentRenderPassId}`, 1);\n });\n // Record Face Index before render.\n this._shadowMap.onBeforeRenderObservable.add((faceIndex) => {\n if (this._sceneUBOs) {\n this._scene.setSceneUniformBuffer(this._sceneUBOs[0]);\n }\n this._currentFaceIndex = faceIndex;\n if (this._filter === ShadowGenerator.FILTER_PCF) {\n engine.setColorWrite(false);\n }\n this.getTransformMatrix(); // generate the view/projection matrix\n this._scene.setTransformMatrix(this._viewMatrix, this._projectionMatrix);\n if (this._useUBO) {\n this._scene.getSceneUniformBuffer().unbindEffect();\n this._scene.finalizeSceneUbo();\n }\n });\n // Blur if required after render.\n this._shadowMap.onAfterUnbindObservable.add(() => {\n var _a, _b;\n if (this._sceneUBOs) {\n this._scene.setSceneUniformBuffer(this._currentSceneUBO);\n }\n this._scene.updateTransformMatrix(); // restore the view/projection matrices of the active camera\n if (this._filter === ShadowGenerator.FILTER_PCF) {\n engine.setColorWrite(true);\n }\n if (!this.useBlurExponentialShadowMap && !this.useBlurCloseExponentialShadowMap) {\n (_a = engine._debugPopGroup) === null || _a === void 0 ? void 0 : _a.call(engine, 1);\n return;\n }\n const shadowMap = this.getShadowMapForRendering();\n if (shadowMap) {\n this._scene.postProcessManager.directRender(this._blurPostProcesses, shadowMap.renderTarget, true);\n engine.unBindFramebuffer(shadowMap.renderTarget, true);\n (_b = engine._debugPopGroup) === null || _b === void 0 ? void 0 : _b.call(engine, 1);\n }\n });\n // Clear according to the chosen filter.\n const clearZero = new Color4(0, 0, 0, 0);\n const clearOne = new Color4(1.0, 1.0, 1.0, 1.0);\n this._shadowMap.onClearObservable.add((engine) => {\n if (this._filter === ShadowGenerator.FILTER_PCF) {\n engine.clear(clearOne, false, true, false);\n }\n else if (this.useExponentialShadowMap || this.useBlurExponentialShadowMap) {\n engine.clear(clearZero, true, true, false);\n }\n else {\n engine.clear(clearOne, true, true, false);\n }\n });\n // Recreate on resize.\n this._shadowMap.onResizeObservable.add((rtt) => {\n this._storedUniqueId = this._shadowMap.uniqueId;\n this._mapSize = rtt.getRenderSize();\n this._light._markMeshesAsLightDirty();\n this.recreateShadowMap();\n });\n // Ensures rendering groupids do not erase the depth buffer\n // or we would lose the shadows information.\n for (let i = RenderingManager.MIN_RENDERINGGROUPS; i < RenderingManager.MAX_RENDERINGGROUPS; i++) {\n this._shadowMap.setRenderingAutoClearDepthStencil(i, false);\n }\n }\n _initializeBlurRTTAndPostProcesses() {\n const engine = this._scene.getEngine();\n const targetSize = this._mapSize / this.blurScale;\n if (!this.useKernelBlur || this.blurScale !== 1.0) {\n this._shadowMap2 = new RenderTargetTexture(this._light.name + \"_shadowMap2\", targetSize, this._scene, false, true, this._textureType, undefined, undefined, false);\n this._shadowMap2.wrapU = Texture.CLAMP_ADDRESSMODE;\n this._shadowMap2.wrapV = Texture.CLAMP_ADDRESSMODE;\n this._shadowMap2.updateSamplingMode(Texture.BILINEAR_SAMPLINGMODE);\n }\n if (this.useKernelBlur) {\n this._kernelBlurXPostprocess = new BlurPostProcess(this._light.name + \"KernelBlurX\", new Vector2(1, 0), this.blurKernel, 1.0, null, Texture.BILINEAR_SAMPLINGMODE, engine, false, this._textureType);\n this._kernelBlurXPostprocess.width = targetSize;\n this._kernelBlurXPostprocess.height = targetSize;\n this._kernelBlurXPostprocess.externalTextureSamplerBinding = true;\n this._kernelBlurXPostprocess.onApplyObservable.add((effect) => {\n effect.setTexture(\"textureSampler\", this._shadowMap);\n });\n this._kernelBlurYPostprocess = new BlurPostProcess(this._light.name + \"KernelBlurY\", new Vector2(0, 1), this.blurKernel, 1.0, null, Texture.BILINEAR_SAMPLINGMODE, engine, false, this._textureType);\n this._kernelBlurXPostprocess.autoClear = false;\n this._kernelBlurYPostprocess.autoClear = false;\n if (this._textureType === 0) {\n this._kernelBlurXPostprocess.packedFloat = true;\n this._kernelBlurYPostprocess.packedFloat = true;\n }\n this._blurPostProcesses = [this._kernelBlurXPostprocess, this._kernelBlurYPostprocess];\n }\n else {\n this._boxBlurPostprocess = new PostProcess(this._light.name + \"DepthBoxBlur\", \"depthBoxBlur\", [\"screenSize\", \"boxOffset\"], [], 1.0, null, Texture.BILINEAR_SAMPLINGMODE, engine, false, \"#define OFFSET \" + this._blurBoxOffset, this._textureType);\n this._boxBlurPostprocess.externalTextureSamplerBinding = true;\n this._boxBlurPostprocess.onApplyObservable.add((effect) => {\n effect.setFloat2(\"screenSize\", targetSize, targetSize);\n effect.setTexture(\"textureSampler\", this._shadowMap);\n });\n this._boxBlurPostprocess.autoClear = false;\n this._blurPostProcesses = [this._boxBlurPostprocess];\n }\n }\n _renderForShadowMap(opaqueSubMeshes, alphaTestSubMeshes, transparentSubMeshes, depthOnlySubMeshes) {\n let index;\n if (depthOnlySubMeshes.length) {\n for (index = 0; index < depthOnlySubMeshes.length; index++) {\n this._renderSubMeshForShadowMap(depthOnlySubMeshes.data[index]);\n }\n }\n for (index = 0; index < opaqueSubMeshes.length; index++) {\n this._renderSubMeshForShadowMap(opaqueSubMeshes.data[index]);\n }\n for (index = 0; index < alphaTestSubMeshes.length; index++) {\n this._renderSubMeshForShadowMap(alphaTestSubMeshes.data[index]);\n }\n if (this._transparencyShadow) {\n for (index = 0; index < transparentSubMeshes.length; index++) {\n this._renderSubMeshForShadowMap(transparentSubMeshes.data[index], true);\n }\n }\n else {\n for (index = 0; index < transparentSubMeshes.length; index++) {\n transparentSubMeshes.data[index].getEffectiveMesh()._internalAbstractMeshDataInfo._isActiveIntermediate = false;\n }\n }\n }\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _bindCustomEffectForRenderSubMeshForShadowMap(subMesh, effect, mesh) {\n effect.setMatrix(\"viewProjection\", this.getTransformMatrix());\n }\n _renderSubMeshForShadowMap(subMesh, isTransparent = false) {\n var _a, _b;\n const renderingMesh = subMesh.getRenderingMesh();\n const effectiveMesh = subMesh.getEffectiveMesh();\n const scene = this._scene;\n const engine = scene.getEngine();\n const material = subMesh.getMaterial();\n effectiveMesh._internalAbstractMeshDataInfo._isActiveIntermediate = false;\n if (!material || subMesh.verticesCount === 0 || subMesh._renderId === scene.getRenderId()) {\n return;\n }\n // Culling\n const detNeg = effectiveMesh._getWorldMatrixDeterminant() < 0;\n let sideOrientation = (_a = renderingMesh.overrideMaterialSideOrientation) !== null && _a !== void 0 ? _a : material.sideOrientation;\n if (detNeg) {\n sideOrientation =\n sideOrientation === 0 ? 1 : 0;\n }\n const reverseSideOrientation = sideOrientation === 0;\n engine.setState(material.backFaceCulling, undefined, undefined, reverseSideOrientation, material.cullBackFaces);\n // Managing instances\n const batch = renderingMesh._getInstancesRenderList(subMesh._id, !!subMesh.getReplacementMesh());\n if (batch.mustReturn) {\n return;\n }\n const hardwareInstancedRendering = engine.getCaps().instancedArrays &&\n ((batch.visibleInstances[subMesh._id] !== null && batch.visibleInstances[subMesh._id] !== undefined) || renderingMesh.hasThinInstances);\n if (this.customAllowRendering && !this.customAllowRendering(subMesh)) {\n return;\n }\n if (this.isReady(subMesh, hardwareInstancedRendering, isTransparent)) {\n subMesh._renderId = scene.getRenderId();\n const shadowDepthWrapper = material.shadowDepthWrapper;\n const drawWrapper = (_b = shadowDepthWrapper === null || shadowDepthWrapper === void 0 ? void 0 : shadowDepthWrapper.getEffect(subMesh, this, engine.currentRenderPassId)) !== null && _b !== void 0 ? _b : subMesh._getDrawWrapper();\n const effect = DrawWrapper.GetEffect(drawWrapper);\n engine.enableEffect(drawWrapper);\n if (!hardwareInstancedRendering) {\n renderingMesh._bind(subMesh, effect, material.fillMode);\n }\n this.getTransformMatrix(); // make sure _cachedDirection et _cachedPosition are up to date\n effect.setFloat3(\"biasAndScaleSM\", this.bias, this.normalBias, this.depthScale);\n if (this.getLight().getTypeID() === Light.LIGHTTYPEID_DIRECTIONALLIGHT) {\n effect.setVector3(\"lightDataSM\", this._cachedDirection);\n }\n else {\n effect.setVector3(\"lightDataSM\", this._cachedPosition);\n }\n const camera = this._getCamera();\n if (camera) {\n effect.setFloat2(\"depthValuesSM\", this.getLight().getDepthMinZ(camera), this.getLight().getDepthMinZ(camera) + this.getLight().getDepthMaxZ(camera));\n }\n if (isTransparent && this.enableSoftTransparentShadow) {\n effect.setFloat(\"softTransparentShadowSM\", effectiveMesh.visibility * material.alpha);\n }\n if (shadowDepthWrapper) {\n subMesh._setMainDrawWrapperOverride(drawWrapper);\n if (shadowDepthWrapper.standalone) {\n shadowDepthWrapper.baseMaterial.bindForSubMesh(effectiveMesh.getWorldMatrix(), renderingMesh, subMesh);\n }\n else {\n material.bindForSubMesh(effectiveMesh.getWorldMatrix(), renderingMesh, subMesh);\n }\n subMesh._setMainDrawWrapperOverride(null);\n }\n else {\n // Alpha test\n if (this._opacityTexture) {\n effect.setTexture(\"diffuseSampler\", this._opacityTexture);\n effect.setMatrix(\"diffuseMatrix\", this._opacityTexture.getTextureMatrix() || this._defaultTextureMatrix);\n }\n // Bones\n if (renderingMesh.useBones && renderingMesh.computeBonesUsingShaders && renderingMesh.skeleton) {\n const skeleton = renderingMesh.skeleton;\n if (skeleton.isUsingTextureForMatrices) {\n const boneTexture = skeleton.getTransformMatrixTexture(renderingMesh);\n if (!boneTexture) {\n return;\n }\n effect.setTexture(\"boneSampler\", boneTexture);\n effect.setFloat(\"boneTextureWidth\", 4.0 * (skeleton.bones.length + 1));\n }\n else {\n effect.setMatrices(\"mBones\", skeleton.getTransformMatrices(renderingMesh));\n }\n }\n // Morph targets\n MaterialHelper.BindMorphTargetParameters(renderingMesh, effect);\n if (renderingMesh.morphTargetManager && renderingMesh.morphTargetManager.isUsingTextureForTargets) {\n renderingMesh.morphTargetManager._bind(effect);\n }\n // Clip planes\n bindClipPlane(effect, material, scene);\n }\n if (!this._useUBO && !shadowDepthWrapper) {\n this._bindCustomEffectForRenderSubMeshForShadowMap(subMesh, effect, effectiveMesh);\n }\n MaterialHelper.BindSceneUniformBuffer(effect, this._scene.getSceneUniformBuffer());\n this._scene.getSceneUniformBuffer().bindUniformBuffer();\n const world = effectiveMesh.getWorldMatrix();\n // In the non hardware instanced mode, the Mesh ubo update is done by the callback passed to renderingMesh._processRendering (see below)\n if (hardwareInstancedRendering) {\n effectiveMesh.getMeshUniformBuffer().bindToEffect(effect, \"Mesh\");\n effectiveMesh.transferToEffect(world);\n }\n if (this.forceBackFacesOnly) {\n engine.setState(true, 0, false, true, material.cullBackFaces);\n }\n // Observables\n this.onBeforeShadowMapRenderMeshObservable.notifyObservers(renderingMesh);\n this.onBeforeShadowMapRenderObservable.notifyObservers(effect);\n // Draw\n renderingMesh._processRendering(effectiveMesh, subMesh, effect, material.fillMode, batch, hardwareInstancedRendering, (isInstance, worldOverride) => {\n if (effectiveMesh !== renderingMesh && !isInstance) {\n renderingMesh.getMeshUniformBuffer().bindToEffect(effect, \"Mesh\");\n renderingMesh.transferToEffect(worldOverride);\n }\n else {\n effectiveMesh.getMeshUniformBuffer().bindToEffect(effect, \"Mesh\");\n effectiveMesh.transferToEffect(isInstance ? worldOverride : world);\n }\n });\n if (this.forceBackFacesOnly) {\n engine.setState(true, 0, false, false, material.cullBackFaces);\n }\n // Observables\n this.onAfterShadowMapRenderObservable.notifyObservers(effect);\n this.onAfterShadowMapRenderMeshObservable.notifyObservers(renderingMesh);\n }\n else {\n // Need to reset refresh rate of the shadowMap\n if (this._shadowMap) {\n this._shadowMap.resetRefreshCounter();\n }\n }\n }\n _applyFilterValues() {\n if (!this._shadowMap) {\n return;\n }\n if (this.filter === ShadowGenerator.FILTER_NONE || this.filter === ShadowGenerator.FILTER_PCSS) {\n this._shadowMap.updateSamplingMode(Texture.NEAREST_SAMPLINGMODE);\n }\n else {\n this._shadowMap.updateSamplingMode(Texture.BILINEAR_SAMPLINGMODE);\n }\n }\n /**\n * Forces all the attached effect to compile to enable rendering only once ready vs. lazily compiling effects.\n * @param onCompiled Callback triggered at the and of the effects compilation\n * @param options Sets of optional options forcing the compilation with different modes\n */\n forceCompilation(onCompiled, options) {\n const localOptions = Object.assign({ useInstances: false }, options);\n const shadowMap = this.getShadowMap();\n if (!shadowMap) {\n if (onCompiled) {\n onCompiled(this);\n }\n return;\n }\n const renderList = shadowMap.renderList;\n if (!renderList) {\n if (onCompiled) {\n onCompiled(this);\n }\n return;\n }\n const subMeshes = new Array();\n for (const mesh of renderList) {\n subMeshes.push(...mesh.subMeshes);\n }\n if (subMeshes.length === 0) {\n if (onCompiled) {\n onCompiled(this);\n }\n return;\n }\n let currentIndex = 0;\n const checkReady = () => {\n var _a, _b;\n if (!this._scene || !this._scene.getEngine()) {\n return;\n }\n while (this.isReady(subMeshes[currentIndex], localOptions.useInstances, (_b = (_a = subMeshes[currentIndex].getMaterial()) === null || _a === void 0 ? void 0 : _a.needAlphaBlendingForMesh(subMeshes[currentIndex].getMesh())) !== null && _b !== void 0 ? _b : false)) {\n currentIndex++;\n if (currentIndex >= subMeshes.length) {\n if (onCompiled) {\n onCompiled(this);\n }\n return;\n }\n }\n setTimeout(checkReady, 16);\n };\n checkReady();\n }\n /**\n * Forces all the attached effect to compile to enable rendering only once ready vs. lazily compiling effects.\n * @param options Sets of optional options forcing the compilation with different modes\n * @returns A promise that resolves when the compilation completes\n */\n forceCompilationAsync(options) {\n return new Promise((resolve) => {\n this.forceCompilation(() => {\n resolve();\n }, options);\n });\n }\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _isReadyCustomDefines(defines, subMesh, useInstances) { }\n _prepareShadowDefines(subMesh, useInstances, defines, isTransparent) {\n defines.push(\"#define SM_LIGHTTYPE_\" + this._light.getClassName().toUpperCase());\n defines.push(\"#define SM_FLOAT \" + (this._textureType !== 0 ? \"1\" : \"0\"));\n defines.push(\"#define SM_ESM \" + (this.useExponentialShadowMap || this.useBlurExponentialShadowMap ? \"1\" : \"0\"));\n defines.push(\"#define SM_DEPTHTEXTURE \" + (this.usePercentageCloserFiltering || this.useContactHardeningShadow ? \"1\" : \"0\"));\n const mesh = subMesh.getMesh();\n // Normal bias.\n defines.push(\"#define SM_NORMALBIAS \" + (this.normalBias && mesh.isVerticesDataPresent(VertexBuffer.NormalKind) ? \"1\" : \"0\"));\n defines.push(\"#define SM_DIRECTIONINLIGHTDATA \" + (this.getLight().getTypeID() === Light.LIGHTTYPEID_DIRECTIONALLIGHT ? \"1\" : \"0\"));\n // Point light\n defines.push(\"#define SM_USEDISTANCE \" + (this._light.needCube() ? \"1\" : \"0\"));\n // Soft transparent shadows\n defines.push(\"#define SM_SOFTTRANSPARENTSHADOW \" + (this.enableSoftTransparentShadow && isTransparent ? \"1\" : \"0\"));\n this._isReadyCustomDefines(defines, subMesh, useInstances);\n return defines;\n }\n /**\n * Determine whether the shadow generator is ready or not (mainly all effects and related post processes needs to be ready).\n * @param subMesh The submesh we want to render in the shadow map\n * @param useInstances Defines whether will draw in the map using instances\n * @param isTransparent Indicates that isReady is called for a transparent subMesh\n * @returns true if ready otherwise, false\n */\n isReady(subMesh, useInstances, isTransparent) {\n var _a;\n const material = subMesh.getMaterial(), shadowDepthWrapper = material === null || material === void 0 ? void 0 : material.shadowDepthWrapper;\n this._opacityTexture = null;\n if (!material) {\n return false;\n }\n const defines = [];\n this._prepareShadowDefines(subMesh, useInstances, defines, isTransparent);\n if (shadowDepthWrapper) {\n if (!shadowDepthWrapper.isReadyForSubMesh(subMesh, defines, this, useInstances, this._scene.getEngine().currentRenderPassId)) {\n return false;\n }\n }\n else {\n const subMeshEffect = subMesh._getDrawWrapper(undefined, true);\n let effect = subMeshEffect.effect;\n let cachedDefines = subMeshEffect.defines;\n const attribs = [VertexBuffer.PositionKind];\n const mesh = subMesh.getMesh();\n // Normal bias.\n if (this.normalBias && mesh.isVerticesDataPresent(VertexBuffer.NormalKind)) {\n attribs.push(VertexBuffer.NormalKind);\n defines.push(\"#define NORMAL\");\n if (mesh.nonUniformScaling) {\n defines.push(\"#define NONUNIFORMSCALING\");\n }\n }\n // Alpha test\n const needAlphaTesting = material.needAlphaTesting();\n if (needAlphaTesting || material.needAlphaBlending()) {\n if (this.useOpacityTextureForTransparentShadow) {\n this._opacityTexture = material.opacityTexture;\n }\n else {\n this._opacityTexture = material.getAlphaTestTexture();\n }\n if (this._opacityTexture) {\n if (!this._opacityTexture.isReady()) {\n return false;\n }\n const alphaCutOff = (_a = material.alphaCutOff) !== null && _a !== void 0 ? _a : ShadowGenerator.DEFAULT_ALPHA_CUTOFF;\n defines.push(\"#define ALPHATEXTURE\");\n if (needAlphaTesting) {\n defines.push(`#define ALPHATESTVALUE ${alphaCutOff}${alphaCutOff % 1 === 0 ? \".\" : \"\"}`);\n }\n if (mesh.isVerticesDataPresent(VertexBuffer.UVKind)) {\n attribs.push(VertexBuffer.UVKind);\n defines.push(\"#define UV1\");\n }\n if (mesh.isVerticesDataPresent(VertexBuffer.UV2Kind)) {\n if (this._opacityTexture.coordinatesIndex === 1) {\n attribs.push(VertexBuffer.UV2Kind);\n defines.push(\"#define UV2\");\n }\n }\n }\n }\n // Bones\n const fallbacks = new EffectFallbacks();\n if (mesh.useBones && mesh.computeBonesUsingShaders && mesh.skeleton) {\n attribs.push(VertexBuffer.MatricesIndicesKind);\n attribs.push(VertexBuffer.MatricesWeightsKind);\n if (mesh.numBoneInfluencers > 4) {\n attribs.push(VertexBuffer.MatricesIndicesExtraKind);\n attribs.push(VertexBuffer.MatricesWeightsExtraKind);\n }\n const skeleton = mesh.skeleton;\n defines.push(\"#define NUM_BONE_INFLUENCERS \" + mesh.numBoneInfluencers);\n if (mesh.numBoneInfluencers > 0) {\n fallbacks.addCPUSkinningFallback(0, mesh);\n }\n if (skeleton.isUsingTextureForMatrices) {\n defines.push(\"#define BONETEXTURE\");\n }\n else {\n defines.push(\"#define BonesPerMesh \" + (skeleton.bones.length + 1));\n }\n }\n else {\n defines.push(\"#define NUM_BONE_INFLUENCERS 0\");\n }\n // Morph targets\n const manager = mesh.morphTargetManager;\n let morphInfluencers = 0;\n if (manager) {\n if (manager.numInfluencers > 0) {\n defines.push(\"#define MORPHTARGETS\");\n morphInfluencers = manager.numInfluencers;\n defines.push(\"#define NUM_MORPH_INFLUENCERS \" + morphInfluencers);\n if (manager.isUsingTextureForTargets) {\n defines.push(\"#define MORPHTARGETS_TEXTURE\");\n }\n MaterialHelper.PrepareAttributesForMorphTargetsInfluencers(attribs, mesh, morphInfluencers);\n }\n }\n // ClipPlanes\n prepareStringDefinesForClipPlanes(material, this._scene, defines);\n // Instances\n if (useInstances) {\n defines.push(\"#define INSTANCES\");\n MaterialHelper.PushAttributesForInstances(attribs);\n if (subMesh.getRenderingMesh().hasThinInstances) {\n defines.push(\"#define THIN_INSTANCES\");\n }\n }\n if (this.customShaderOptions) {\n if (this.customShaderOptions.defines) {\n for (const define of this.customShaderOptions.defines) {\n if (defines.indexOf(define) === -1) {\n defines.push(define);\n }\n }\n }\n }\n // Get correct effect\n const join = defines.join(\"\\n\");\n if (cachedDefines !== join) {\n cachedDefines = join;\n let shaderName = \"shadowMap\";\n const uniforms = [\n \"world\",\n \"mBones\",\n \"viewProjection\",\n \"diffuseMatrix\",\n \"lightDataSM\",\n \"depthValuesSM\",\n \"biasAndScaleSM\",\n \"morphTargetInfluences\",\n \"boneTextureWidth\",\n \"softTransparentShadowSM\",\n \"morphTargetTextureInfo\",\n \"morphTargetTextureIndices\",\n ];\n const samplers = [\"diffuseSampler\", \"boneSampler\", \"morphTargets\"];\n const uniformBuffers = [\"Scene\", \"Mesh\"];\n addClipPlaneUniforms(uniforms);\n // Custom shader?\n if (this.customShaderOptions) {\n shaderName = this.customShaderOptions.shaderName;\n if (this.customShaderOptions.attributes) {\n for (const attrib of this.customShaderOptions.attributes) {\n if (attribs.indexOf(attrib) === -1) {\n attribs.push(attrib);\n }\n }\n }\n if (this.customShaderOptions.uniforms) {\n for (const uniform of this.customShaderOptions.uniforms) {\n if (uniforms.indexOf(uniform) === -1) {\n uniforms.push(uniform);\n }\n }\n }\n if (this.customShaderOptions.samplers) {\n for (const sampler of this.customShaderOptions.samplers) {\n if (samplers.indexOf(sampler) === -1) {\n samplers.push(sampler);\n }\n }\n }\n }\n const engine = this._scene.getEngine();\n effect = engine.createEffect(shaderName, {\n attributes: attribs,\n uniformsNames: uniforms,\n uniformBuffersNames: uniformBuffers,\n samplers: samplers,\n defines: join,\n fallbacks: fallbacks,\n onCompiled: null,\n onError: null,\n indexParameters: { maxSimultaneousMorphTargets: morphInfluencers },\n }, engine);\n subMeshEffect.setEffect(effect, cachedDefines);\n }\n if (!effect.isReady()) {\n return false;\n }\n }\n if (this.useBlurExponentialShadowMap || this.useBlurCloseExponentialShadowMap) {\n if (!this._blurPostProcesses || !this._blurPostProcesses.length) {\n this._initializeBlurRTTAndPostProcesses();\n }\n }\n if (this._kernelBlurXPostprocess && !this._kernelBlurXPostprocess.isReady()) {\n return false;\n }\n if (this._kernelBlurYPostprocess && !this._kernelBlurYPostprocess.isReady()) {\n return false;\n }\n if (this._boxBlurPostprocess && !this._boxBlurPostprocess.isReady()) {\n return false;\n }\n return true;\n }\n /**\n * Prepare all the defines in a material relying on a shadow map at the specified light index.\n * @param defines Defines of the material we want to update\n * @param lightIndex Index of the light in the enabled light list of the material\n */\n prepareDefines(defines, lightIndex) {\n const scene = this._scene;\n const light = this._light;\n if (!scene.shadowsEnabled || !light.shadowEnabled) {\n return;\n }\n defines[\"SHADOW\" + lightIndex] = true;\n if (this.useContactHardeningShadow) {\n defines[\"SHADOWPCSS\" + lightIndex] = true;\n if (this._filteringQuality === ShadowGenerator.QUALITY_LOW) {\n defines[\"SHADOWLOWQUALITY\" + lightIndex] = true;\n }\n else if (this._filteringQuality === ShadowGenerator.QUALITY_MEDIUM) {\n defines[\"SHADOWMEDIUMQUALITY\" + lightIndex] = true;\n }\n // else default to high.\n }\n else if (this.usePercentageCloserFiltering) {\n defines[\"SHADOWPCF\" + lightIndex] = true;\n if (this._filteringQuality === ShadowGenerator.QUALITY_LOW) {\n defines[\"SHADOWLOWQUALITY\" + lightIndex] = true;\n }\n else if (this._filteringQuality === ShadowGenerator.QUALITY_MEDIUM) {\n defines[\"SHADOWMEDIUMQUALITY\" + lightIndex] = true;\n }\n // else default to high.\n }\n else if (this.usePoissonSampling) {\n defines[\"SHADOWPOISSON\" + lightIndex] = true;\n }\n else if (this.useExponentialShadowMap || this.useBlurExponentialShadowMap) {\n defines[\"SHADOWESM\" + lightIndex] = true;\n }\n else if (this.useCloseExponentialShadowMap || this.useBlurCloseExponentialShadowMap) {\n defines[\"SHADOWCLOSEESM\" + lightIndex] = true;\n }\n if (light.needCube()) {\n defines[\"SHADOWCUBE\" + lightIndex] = true;\n }\n }\n /**\n * Binds the shadow related information inside of an effect (information like near, far, darkness...\n * defined in the generator but impacting the effect).\n * @param lightIndex Index of the light in the enabled light list of the material owning the effect\n * @param effect The effect we are binding the information for\n */\n bindShadowLight(lightIndex, effect) {\n const light = this._light;\n const scene = this._scene;\n if (!scene.shadowsEnabled || !light.shadowEnabled) {\n return;\n }\n const camera = this._getCamera();\n if (!camera) {\n return;\n }\n const shadowMap = this.getShadowMap();\n if (!shadowMap) {\n return;\n }\n if (!light.needCube()) {\n effect.setMatrix(\"lightMatrix\" + lightIndex, this.getTransformMatrix());\n }\n // Only PCF uses depth stencil texture.\n if (this._filter === ShadowGenerator.FILTER_PCF) {\n effect.setDepthStencilTexture(\"shadowSampler\" + lightIndex, this.getShadowMapForRendering());\n light._uniformBuffer.updateFloat4(\"shadowsInfo\", this.getDarkness(), shadowMap.getSize().width, 1 / shadowMap.getSize().width, this.frustumEdgeFalloff, lightIndex);\n }\n else if (this._filter === ShadowGenerator.FILTER_PCSS) {\n effect.setDepthStencilTexture(\"shadowSampler\" + lightIndex, this.getShadowMapForRendering());\n effect.setTexture(\"depthSampler\" + lightIndex, this.getShadowMapForRendering());\n light._uniformBuffer.updateFloat4(\"shadowsInfo\", this.getDarkness(), 1 / shadowMap.getSize().width, this._contactHardeningLightSizeUVRatio * shadowMap.getSize().width, this.frustumEdgeFalloff, lightIndex);\n }\n else {\n effect.setTexture(\"shadowSampler\" + lightIndex, this.getShadowMapForRendering());\n light._uniformBuffer.updateFloat4(\"shadowsInfo\", this.getDarkness(), this.blurScale / shadowMap.getSize().width, this.depthScale, this.frustumEdgeFalloff, lightIndex);\n }\n light._uniformBuffer.updateFloat2(\"depthValues\", this.getLight().getDepthMinZ(camera), this.getLight().getDepthMinZ(camera) + this.getLight().getDepthMaxZ(camera), lightIndex);\n }\n /**\n * Gets the transformation matrix used to project the meshes into the map from the light point of view.\n * (eq to shadow projection matrix * light transform matrix)\n * @returns The transform matrix used to create the shadow map\n */\n getTransformMatrix() {\n const scene = this._scene;\n if (this._currentRenderId === scene.getRenderId() && this._currentFaceIndexCache === this._currentFaceIndex) {\n return this._transformMatrix;\n }\n this._currentRenderId = scene.getRenderId();\n this._currentFaceIndexCache = this._currentFaceIndex;\n let lightPosition = this._light.position;\n if (this._light.computeTransformedInformation()) {\n lightPosition = this._light.transformedPosition;\n }\n Vector3.NormalizeToRef(this._light.getShadowDirection(this._currentFaceIndex), this._lightDirection);\n if (Math.abs(Vector3.Dot(this._lightDirection, Vector3.Up())) === 1.0) {\n this._lightDirection.z = 0.0000000000001; // Required to avoid perfectly perpendicular light\n }\n if (this._light.needProjectionMatrixCompute() ||\n !this._cachedPosition ||\n !this._cachedDirection ||\n !lightPosition.equals(this._cachedPosition) ||\n !this._lightDirection.equals(this._cachedDirection)) {\n this._cachedPosition.copyFrom(lightPosition);\n this._cachedDirection.copyFrom(this._lightDirection);\n Matrix.LookAtLHToRef(lightPosition, lightPosition.add(this._lightDirection), Vector3.Up(), this._viewMatrix);\n const shadowMap = this.getShadowMap();\n if (shadowMap) {\n const renderList = shadowMap.renderList;\n if (renderList) {\n this._light.setShadowProjectionMatrix(this._projectionMatrix, this._viewMatrix, renderList);\n }\n }\n this._viewMatrix.multiplyToRef(this._projectionMatrix, this._transformMatrix);\n }\n return this._transformMatrix;\n }\n /**\n * Recreates the shadow map dependencies like RTT and post processes. This can be used during the switch between\n * Cube and 2D textures for instance.\n */\n recreateShadowMap() {\n const shadowMap = this._shadowMap;\n if (!shadowMap) {\n return;\n }\n // Track render list.\n const renderList = shadowMap.renderList;\n // Clean up existing data.\n this._disposeRTTandPostProcesses();\n // Reinitializes.\n this._initializeGenerator();\n // Reaffect the filter to ensure a correct fallback if necessary.\n this.filter = this._filter;\n // Reaffect the filter.\n this._applyFilterValues();\n // Reaffect Render List.\n if (renderList) {\n // Note: don't do this._shadowMap!.renderList = renderList;\n // The renderList hooked array is accessing the old RenderTargetTexture (see RenderTargetTexture._hookArray), which is disposed at this point (by the call to _disposeRTTandPostProcesses)\n if (!this._shadowMap.renderList) {\n this._shadowMap.renderList = [];\n }\n for (const mesh of renderList) {\n this._shadowMap.renderList.push(mesh);\n }\n }\n else {\n this._shadowMap.renderList = null;\n }\n }\n _disposeBlurPostProcesses() {\n if (this._shadowMap2) {\n this._shadowMap2.dispose();\n this._shadowMap2 = null;\n }\n if (this._boxBlurPostprocess) {\n this._boxBlurPostprocess.dispose();\n this._boxBlurPostprocess = null;\n }\n if (this._kernelBlurXPostprocess) {\n this._kernelBlurXPostprocess.dispose();\n this._kernelBlurXPostprocess = null;\n }\n if (this._kernelBlurYPostprocess) {\n this._kernelBlurYPostprocess.dispose();\n this._kernelBlurYPostprocess = null;\n }\n this._blurPostProcesses = [];\n }\n _disposeRTTandPostProcesses() {\n if (this._shadowMap) {\n this._shadowMap.dispose();\n this._shadowMap = null;\n }\n this._disposeBlurPostProcesses();\n }\n _disposeSceneUBOs() {\n if (this._sceneUBOs) {\n for (const ubo of this._sceneUBOs) {\n ubo.dispose();\n }\n this._sceneUBOs = [];\n }\n }\n /**\n * Disposes the ShadowGenerator.\n * Returns nothing.\n */\n dispose() {\n this._disposeRTTandPostProcesses();\n this._disposeSceneUBOs();\n if (this._light) {\n if (this._light._shadowGenerators) {\n const iterator = this._light._shadowGenerators.entries();\n for (let entry = iterator.next(); entry.done !== true; entry = iterator.next()) {\n const [camera, shadowGenerator] = entry.value;\n if (shadowGenerator === this) {\n this._light._shadowGenerators.delete(camera);\n }\n }\n if (this._light._shadowGenerators.size === 0) {\n this._light._shadowGenerators = null;\n }\n }\n this._light._markMeshesAsLightDirty();\n }\n this.onBeforeShadowMapRenderMeshObservable.clear();\n this.onBeforeShadowMapRenderObservable.clear();\n this.onAfterShadowMapRenderMeshObservable.clear();\n this.onAfterShadowMapRenderObservable.clear();\n }\n /**\n * Serializes the shadow generator setup to a json object.\n * @returns The serialized JSON object\n */\n serialize() {\n var _a;\n const serializationObject = {};\n const shadowMap = this.getShadowMap();\n if (!shadowMap) {\n return serializationObject;\n }\n serializationObject.className = this.getClassName();\n serializationObject.lightId = this._light.id;\n serializationObject.cameraId = (_a = this._camera) === null || _a === void 0 ? void 0 : _a.id;\n serializationObject.id = this.id;\n serializationObject.mapSize = shadowMap.getRenderSize();\n serializationObject.forceBackFacesOnly = this.forceBackFacesOnly;\n serializationObject.darkness = this.getDarkness();\n serializationObject.transparencyShadow = this._transparencyShadow;\n serializationObject.frustumEdgeFalloff = this.frustumEdgeFalloff;\n serializationObject.bias = this.bias;\n serializationObject.normalBias = this.normalBias;\n serializationObject.usePercentageCloserFiltering = this.usePercentageCloserFiltering;\n serializationObject.useContactHardeningShadow = this.useContactHardeningShadow;\n serializationObject.contactHardeningLightSizeUVRatio = this.contactHardeningLightSizeUVRatio;\n serializationObject.filteringQuality = this.filteringQuality;\n serializationObject.useExponentialShadowMap = this.useExponentialShadowMap;\n serializationObject.useBlurExponentialShadowMap = this.useBlurExponentialShadowMap;\n serializationObject.useCloseExponentialShadowMap = this.useBlurExponentialShadowMap;\n serializationObject.useBlurCloseExponentialShadowMap = this.useBlurExponentialShadowMap;\n serializationObject.usePoissonSampling = this.usePoissonSampling;\n serializationObject.depthScale = this.depthScale;\n serializationObject.blurBoxOffset = this.blurBoxOffset;\n serializationObject.blurKernel = this.blurKernel;\n serializationObject.blurScale = this.blurScale;\n serializationObject.useKernelBlur = this.useKernelBlur;\n serializationObject.renderList = [];\n if (shadowMap.renderList) {\n for (let meshIndex = 0; meshIndex < shadowMap.renderList.length; meshIndex++) {\n const mesh = shadowMap.renderList[meshIndex];\n serializationObject.renderList.push(mesh.id);\n }\n }\n return serializationObject;\n }\n /**\n * Parses a serialized ShadowGenerator and returns a new ShadowGenerator.\n * @param parsedShadowGenerator The JSON object to parse\n * @param scene The scene to create the shadow map for\n * @param constr A function that builds a shadow generator or undefined to create an instance of the default shadow generator\n * @returns The parsed shadow generator\n */\n static Parse(parsedShadowGenerator, scene, constr) {\n const light = scene.getLightById(parsedShadowGenerator.lightId);\n const camera = parsedShadowGenerator.cameraId !== undefined ? scene.getCameraById(parsedShadowGenerator.cameraId) : null;\n const shadowGenerator = constr ? constr(parsedShadowGenerator.mapSize, light, camera) : new ShadowGenerator(parsedShadowGenerator.mapSize, light, undefined, camera);\n const shadowMap = shadowGenerator.getShadowMap();\n for (let meshIndex = 0; meshIndex < parsedShadowGenerator.renderList.length; meshIndex++) {\n const meshes = scene.getMeshesById(parsedShadowGenerator.renderList[meshIndex]);\n meshes.forEach(function (mesh) {\n if (!shadowMap) {\n return;\n }\n if (!shadowMap.renderList) {\n shadowMap.renderList = [];\n }\n shadowMap.renderList.push(mesh);\n });\n }\n if (parsedShadowGenerator.id !== undefined) {\n shadowGenerator.id = parsedShadowGenerator.id;\n }\n shadowGenerator.forceBackFacesOnly = !!parsedShadowGenerator.forceBackFacesOnly;\n if (parsedShadowGenerator.darkness !== undefined) {\n shadowGenerator.setDarkness(parsedShadowGenerator.darkness);\n }\n if (parsedShadowGenerator.transparencyShadow) {\n shadowGenerator.setTransparencyShadow(true);\n }\n if (parsedShadowGenerator.frustumEdgeFalloff !== undefined) {\n shadowGenerator.frustumEdgeFalloff = parsedShadowGenerator.frustumEdgeFalloff;\n }\n if (parsedShadowGenerator.bias !== undefined) {\n shadowGenerator.bias = parsedShadowGenerator.bias;\n }\n if (parsedShadowGenerator.normalBias !== undefined) {\n shadowGenerator.normalBias = parsedShadowGenerator.normalBias;\n }\n if (parsedShadowGenerator.usePercentageCloserFiltering) {\n shadowGenerator.usePercentageCloserFiltering = true;\n }\n else if (parsedShadowGenerator.useContactHardeningShadow) {\n shadowGenerator.useContactHardeningShadow = true;\n }\n else if (parsedShadowGenerator.usePoissonSampling) {\n shadowGenerator.usePoissonSampling = true;\n }\n else if (parsedShadowGenerator.useExponentialShadowMap) {\n shadowGenerator.useExponentialShadowMap = true;\n }\n else if (parsedShadowGenerator.useBlurExponentialShadowMap) {\n shadowGenerator.useBlurExponentialShadowMap = true;\n }\n else if (parsedShadowGenerator.useCloseExponentialShadowMap) {\n shadowGenerator.useCloseExponentialShadowMap = true;\n }\n else if (parsedShadowGenerator.useBlurCloseExponentialShadowMap) {\n shadowGenerator.useBlurCloseExponentialShadowMap = true;\n }\n // Backward compat\n else if (parsedShadowGenerator.useVarianceShadowMap) {\n shadowGenerator.useExponentialShadowMap = true;\n }\n else if (parsedShadowGenerator.useBlurVarianceShadowMap) {\n shadowGenerator.useBlurExponentialShadowMap = true;\n }\n if (parsedShadowGenerator.contactHardeningLightSizeUVRatio !== undefined) {\n shadowGenerator.contactHardeningLightSizeUVRatio = parsedShadowGenerator.contactHardeningLightSizeUVRatio;\n }\n if (parsedShadowGenerator.filteringQuality !== undefined) {\n shadowGenerator.filteringQuality = parsedShadowGenerator.filteringQuality;\n }\n if (parsedShadowGenerator.depthScale) {\n shadowGenerator.depthScale = parsedShadowGenerator.depthScale;\n }\n if (parsedShadowGenerator.blurScale) {\n shadowGenerator.blurScale = parsedShadowGenerator.blurScale;\n }\n if (parsedShadowGenerator.blurBoxOffset) {\n shadowGenerator.blurBoxOffset = parsedShadowGenerator.blurBoxOffset;\n }\n if (parsedShadowGenerator.useKernelBlur) {\n shadowGenerator.useKernelBlur = parsedShadowGenerator.useKernelBlur;\n }\n if (parsedShadowGenerator.blurKernel) {\n shadowGenerator.blurKernel = parsedShadowGenerator.blurKernel;\n }\n return shadowGenerator;\n }\n }\n /**\n * Name of the shadow generator class\n */\n ShadowGenerator.CLASSNAME = \"ShadowGenerator\";\n /**\n * Shadow generator mode None: no filtering applied.\n */\n ShadowGenerator.FILTER_NONE = 0;\n /**\n * Shadow generator mode ESM: Exponential Shadow Mapping.\n * (http://developer.download.nvidia.com/presentations/2008/GDC/GDC08_SoftShadowMapping.pdf)\n */\n ShadowGenerator.FILTER_EXPONENTIALSHADOWMAP = 1;\n /**\n * Shadow generator mode Poisson Sampling: Percentage Closer Filtering.\n * (Multiple Tap around evenly distributed around the pixel are used to evaluate the shadow strength)\n */\n ShadowGenerator.FILTER_POISSONSAMPLING = 2;\n /**\n * Shadow generator mode ESM: Blurred Exponential Shadow Mapping.\n * (http://developer.download.nvidia.com/presentations/2008/GDC/GDC08_SoftShadowMapping.pdf)\n */\n ShadowGenerator.FILTER_BLUREXPONENTIALSHADOWMAP = 3;\n /**\n * Shadow generator mode ESM: Exponential Shadow Mapping using the inverse of the exponential preventing\n * edge artifacts on steep falloff.\n * (http://developer.download.nvidia.com/presentations/2008/GDC/GDC08_SoftShadowMapping.pdf)\n */\n ShadowGenerator.FILTER_CLOSEEXPONENTIALSHADOWMAP = 4;\n /**\n * Shadow generator mode ESM: Blurred Exponential Shadow Mapping using the inverse of the exponential preventing\n * edge artifacts on steep falloff.\n * (http://developer.download.nvidia.com/presentations/2008/GDC/GDC08_SoftShadowMapping.pdf)\n */\n ShadowGenerator.FILTER_BLURCLOSEEXPONENTIALSHADOWMAP = 5;\n /**\n * Shadow generator mode PCF: Percentage Closer Filtering\n * benefits from Webgl 2 shadow samplers. Fallback to Poisson Sampling in Webgl 1\n * (https://developer.nvidia.com/gpugems/GPUGems/gpugems_ch11.html)\n */\n ShadowGenerator.FILTER_PCF = 6;\n /**\n * Shadow generator mode PCSS: Percentage Closering Soft Shadow.\n * benefits from Webgl 2 shadow samplers. Fallback to Poisson Sampling in Webgl 1\n * Contact Hardening\n */\n ShadowGenerator.FILTER_PCSS = 7;\n /**\n * Reserved for PCF and PCSS\n * Highest Quality.\n *\n * Execute PCF on a 5*5 kernel improving a lot the shadow aliasing artifacts.\n *\n * Execute PCSS with 32 taps blocker search and 64 taps PCF.\n */\n ShadowGenerator.QUALITY_HIGH = 0;\n /**\n * Reserved for PCF and PCSS\n * Good tradeoff for quality/perf cross devices\n *\n * Execute PCF on a 3*3 kernel.\n *\n * Execute PCSS with 16 taps blocker search and 32 taps PCF.\n */\n ShadowGenerator.QUALITY_MEDIUM = 1;\n /**\n * Reserved for PCF and PCSS\n * The lowest quality but the fastest.\n *\n * Execute PCF on a 1*1 kernel.\n *\n * Execute PCSS with 16 taps blocker search and 16 taps PCF.\n */\n ShadowGenerator.QUALITY_LOW = 2;\n /**\n * Defines the default alpha cutoff value used for transparent alpha tested materials.\n */\n ShadowGenerator.DEFAULT_ALPHA_CUTOFF = 0.5;\n /**\n * @internal\n */\n ShadowGenerator._SceneComponentInitialization = (_) => {\n throw _WarnImport(\"ShadowGeneratorSceneComponent\");\n };\n\n // Do not edit.\n const name$2e = \"depthPixelShader\";\n const shader$2e = `#ifdef ALPHATEST\nvarying vec2 vUV;uniform sampler2D diffuseSampler;\n#endif\n#include\nvarying float vDepthMetric;\n#ifdef PACKED\n#include\n#endif\n#ifdef STORE_CAMERASPACE_Z\nvarying vec4 vViewPos;\n#endif\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{\n#include\n#ifdef ALPHATEST\nif (texture2D(diffuseSampler,vUV).a<0.4)\ndiscard;\n#endif\n#ifdef STORE_CAMERASPACE_Z\n#ifdef PACKED\ngl_FragColor=pack(vViewPos.z);\n#else\ngl_FragColor=vec4(vViewPos.z,0.0,0.0,1.0);\n#endif\n#else\n#ifdef NONLINEARDEPTH\n#ifdef PACKED\ngl_FragColor=pack(gl_FragCoord.z);\n#else\ngl_FragColor=vec4(gl_FragCoord.z,0.0,0.0,0.0);\n#endif\n#else\n#ifdef PACKED\ngl_FragColor=pack(vDepthMetric);\n#else\ngl_FragColor=vec4(vDepthMetric,0.0,0.0,1.0);\n#endif\n#endif\n#endif\n}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2e] = shader$2e;\n\n // Do not edit.\n const name$2f = \"depthVertexShader\";\n const shader$2f = `attribute vec3 position;\n#include\n#include\n#include\n#include[0..maxSimultaneousMorphTargets]\n#include\n#include\nuniform mat4 viewProjection;uniform vec2 depthValues;\n#if defined(ALPHATEST) || defined(NEED_UV)\nvarying vec2 vUV;uniform mat4 diffuseMatrix;\n#ifdef UV1\nattribute vec2 uv;\n#endif\n#ifdef UV2\nattribute vec2 uv2;\n#endif\n#endif\n#ifdef STORE_CAMERASPACE_Z\nuniform mat4 view;varying vec4 vViewPos;\n#endif\nvarying float vDepthMetric;\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void)\n{vec3 positionUpdated=position;\n#ifdef UV1\nvec2 uvUpdated=uv;\n#endif\n#include\n#include[0..maxSimultaneousMorphTargets]\n#include\n#include\n#include\nvec4 worldPos=finalWorld*vec4(positionUpdated,1.0);\n#include\ngl_Position=viewProjection*worldPos;\n#ifdef STORE_CAMERASPACE_Z\nvViewPos=view*worldPos;\n#else\n#ifdef USE_REVERSE_DEPTHBUFFER\nvDepthMetric=((-gl_Position.z+depthValues.x)/(depthValues.y));\n#else\nvDepthMetric=((gl_Position.z+depthValues.x)/(depthValues.y));\n#endif\n#endif\n#if defined(ALPHATEST) || defined(BASIC_RENDER)\n#ifdef UV1\nvUV=vec2(diffuseMatrix*vec4(uvUpdated,1.0,0.0));\n#endif\n#ifdef UV2\nvUV=vec2(diffuseMatrix*vec4(uv2,1.0,0.0));\n#endif\n#endif\n}\n`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2f] = shader$2f;\n\n /**\n * This represents a depth renderer in Babylon.\n * A depth renderer will render to it's depth map every frame which can be displayed or used in post processing\n */\n class DepthRenderer {\n /**\n * Sets a specific material to be used to render a mesh/a list of meshes by the depth renderer\n * @param mesh mesh or array of meshes\n * @param material material to use by the depth render when rendering the mesh(es). If undefined is passed, the specific material created by the depth renderer will be used.\n */\n setMaterialForRendering(mesh, material) {\n this._depthMap.setMaterialForRendering(mesh, material);\n }\n /**\n * Instantiates a depth renderer\n * @param scene The scene the renderer belongs to\n * @param type The texture type of the depth map (default: Engine.TEXTURETYPE_FLOAT)\n * @param camera The camera to be used to render the depth map (default: scene's active camera)\n * @param storeNonLinearDepth Defines whether the depth is stored linearly like in Babylon Shadows or directly like glFragCoord.z\n * @param samplingMode The sampling mode to be used with the render target (Linear, Nearest...) (default: TRILINEAR_SAMPLINGMODE)\n * @param storeCameraSpaceZ Defines whether the depth stored is the Z coordinate in camera space. If true, storeNonLinearDepth has no effect. (Default: false)\n * @param name Name of the render target (default: DepthRenderer)\n */\n constructor(scene, type = 1, camera = null, storeNonLinearDepth = false, samplingMode = Texture.TRILINEAR_SAMPLINGMODE, storeCameraSpaceZ = false, name) {\n /** Enable or disable the depth renderer. When disabled, the depth texture is not updated */\n this.enabled = true;\n /** Force writing the transparent objects into the depth map */\n this.forceDepthWriteTransparentMeshes = false;\n /**\n * Specifies that the depth renderer will only be used within\n * the camera it is created for.\n * This can help forcing its rendering during the camera processing.\n */\n this.useOnlyInActiveCamera = false;\n /** If true, reverse the culling of materials before writing to the depth texture.\n * So, basically, when \"true\", back facing instead of front facing faces are rasterized into the texture\n */\n this.reverseCulling = false;\n this._scene = scene;\n this._storeNonLinearDepth = storeNonLinearDepth;\n this._storeCameraSpaceZ = storeCameraSpaceZ;\n this.isPacked = type === 0;\n if (this.isPacked) {\n this.clearColor = new Color4(1.0, 1.0, 1.0, 1.0);\n }\n else {\n this.clearColor = new Color4(storeCameraSpaceZ ? 1e8 : 1.0, 0.0, 0.0, 1.0);\n }\n DepthRenderer._SceneComponentInitialization(this._scene);\n const engine = scene.getEngine();\n this._camera = camera;\n if (samplingMode !== Texture.NEAREST_SAMPLINGMODE) {\n if (type === 1 && !engine._caps.textureFloatLinearFiltering) {\n samplingMode = Texture.NEAREST_SAMPLINGMODE;\n }\n if (type === 2 && !engine._caps.textureHalfFloatLinearFiltering) {\n samplingMode = Texture.NEAREST_SAMPLINGMODE;\n }\n }\n // Render target\n const format = this.isPacked || !engine._features.supportExtendedTextureFormats ? 5 : 6;\n this._depthMap = new RenderTargetTexture(name !== null && name !== void 0 ? name : \"DepthRenderer\", { width: engine.getRenderWidth(), height: engine.getRenderHeight() }, this._scene, false, true, type, false, samplingMode, undefined, undefined, undefined, format);\n this._depthMap.wrapU = Texture.CLAMP_ADDRESSMODE;\n this._depthMap.wrapV = Texture.CLAMP_ADDRESSMODE;\n this._depthMap.refreshRate = 1;\n this._depthMap.renderParticles = false;\n this._depthMap.renderList = null;\n this._depthMap.noPrePassRenderer = true;\n // Camera to get depth map from to support multiple concurrent cameras\n this._depthMap.activeCamera = this._camera;\n this._depthMap.ignoreCameraViewport = true;\n this._depthMap.useCameraPostProcesses = false;\n // set default depth value to 1.0 (far away)\n this._depthMap.onClearObservable.add((engine) => {\n engine.clear(this.clearColor, true, true, true);\n });\n this._depthMap.onBeforeBindObservable.add(() => {\n var _a;\n (_a = engine._debugPushGroup) === null || _a === void 0 ? void 0 : _a.call(engine, \"depth renderer\", 1);\n });\n this._depthMap.onAfterUnbindObservable.add(() => {\n var _a;\n (_a = engine._debugPopGroup) === null || _a === void 0 ? void 0 : _a.call(engine, 1);\n });\n this._depthMap.customIsReadyFunction = (mesh, refreshRate, preWarm) => {\n if ((preWarm || refreshRate === 0) && mesh.subMeshes) {\n for (let i = 0; i < mesh.subMeshes.length; ++i) {\n const subMesh = mesh.subMeshes[i];\n const renderingMesh = subMesh.getRenderingMesh();\n const batch = renderingMesh._getInstancesRenderList(subMesh._id, !!subMesh.getReplacementMesh());\n const hardwareInstancedRendering = engine.getCaps().instancedArrays &&\n ((batch.visibleInstances[subMesh._id] !== null && batch.visibleInstances[subMesh._id] !== undefined) || renderingMesh.hasThinInstances);\n if (!this.isReady(subMesh, hardwareInstancedRendering)) {\n return false;\n }\n }\n }\n return true;\n };\n // Custom render function\n const renderSubMesh = (subMesh) => {\n var _a, _b;\n const renderingMesh = subMesh.getRenderingMesh();\n const effectiveMesh = subMesh.getEffectiveMesh();\n const scene = this._scene;\n const engine = scene.getEngine();\n const material = subMesh.getMaterial();\n effectiveMesh._internalAbstractMeshDataInfo._isActiveIntermediate = false;\n if (!material || effectiveMesh.infiniteDistance || material.disableDepthWrite || subMesh.verticesCount === 0 || subMesh._renderId === scene.getRenderId()) {\n return;\n }\n // Culling\n const detNeg = effectiveMesh._getWorldMatrixDeterminant() < 0;\n let sideOrientation = (_a = renderingMesh.overrideMaterialSideOrientation) !== null && _a !== void 0 ? _a : material.sideOrientation;\n if (detNeg) {\n sideOrientation =\n sideOrientation === 0\n ? 1\n : 0;\n }\n const reverseSideOrientation = sideOrientation === 0;\n engine.setState(material.backFaceCulling, 0, false, reverseSideOrientation, this.reverseCulling ? !material.cullBackFaces : material.cullBackFaces);\n // Managing instances\n const batch = renderingMesh._getInstancesRenderList(subMesh._id, !!subMesh.getReplacementMesh());\n if (batch.mustReturn) {\n return;\n }\n const hardwareInstancedRendering = engine.getCaps().instancedArrays &&\n ((batch.visibleInstances[subMesh._id] !== null && batch.visibleInstances[subMesh._id] !== undefined) || renderingMesh.hasThinInstances);\n const camera = this._camera || scene.activeCamera;\n if (this.isReady(subMesh, hardwareInstancedRendering) && camera) {\n subMesh._renderId = scene.getRenderId();\n const renderingMaterial = (_b = effectiveMesh._internalAbstractMeshDataInfo._materialForRenderPass) === null || _b === void 0 ? void 0 : _b[engine.currentRenderPassId];\n let drawWrapper = subMesh._getDrawWrapper();\n if (!drawWrapper && renderingMaterial) {\n drawWrapper = renderingMaterial._getDrawWrapper();\n }\n const cameraIsOrtho = camera.mode === Camera.ORTHOGRAPHIC_CAMERA;\n if (!drawWrapper) {\n return;\n }\n const effect = drawWrapper.effect;\n engine.enableEffect(drawWrapper);\n if (!hardwareInstancedRendering) {\n renderingMesh._bind(subMesh, effect, material.fillMode);\n }\n if (!renderingMaterial) {\n effect.setMatrix(\"viewProjection\", scene.getTransformMatrix());\n effect.setMatrix(\"world\", effectiveMesh.getWorldMatrix());\n if (this._storeCameraSpaceZ) {\n effect.setMatrix(\"view\", scene.getViewMatrix());\n }\n }\n else {\n renderingMaterial.bindForSubMesh(effectiveMesh.getWorldMatrix(), effectiveMesh, subMesh);\n }\n let minZ, maxZ;\n if (cameraIsOrtho) {\n minZ = !engine.useReverseDepthBuffer && engine.isNDCHalfZRange ? 0 : 1;\n maxZ = engine.useReverseDepthBuffer && engine.isNDCHalfZRange ? 0 : 1;\n }\n else {\n minZ = engine.useReverseDepthBuffer && engine.isNDCHalfZRange ? camera.minZ : engine.isNDCHalfZRange ? 0 : camera.minZ;\n maxZ = engine.useReverseDepthBuffer && engine.isNDCHalfZRange ? 0 : camera.maxZ;\n }\n effect.setFloat2(\"depthValues\", minZ, minZ + maxZ);\n if (!renderingMaterial) {\n // Alpha test\n if (material.needAlphaTesting()) {\n const alphaTexture = material.getAlphaTestTexture();\n if (alphaTexture) {\n effect.setTexture(\"diffuseSampler\", alphaTexture);\n effect.setMatrix(\"diffuseMatrix\", alphaTexture.getTextureMatrix());\n }\n }\n // Bones\n if (renderingMesh.useBones && renderingMesh.computeBonesUsingShaders && renderingMesh.skeleton) {\n const skeleton = renderingMesh.skeleton;\n if (skeleton.isUsingTextureForMatrices) {\n const boneTexture = skeleton.getTransformMatrixTexture(renderingMesh);\n if (!boneTexture) {\n return;\n }\n effect.setTexture(\"boneSampler\", boneTexture);\n effect.setFloat(\"boneTextureWidth\", 4.0 * (skeleton.bones.length + 1));\n }\n else {\n effect.setMatrices(\"mBones\", skeleton.getTransformMatrices(renderingMesh));\n }\n }\n // Clip planes\n bindClipPlane(effect, material, scene);\n // Morph targets\n MaterialHelper.BindMorphTargetParameters(renderingMesh, effect);\n if (renderingMesh.morphTargetManager && renderingMesh.morphTargetManager.isUsingTextureForTargets) {\n renderingMesh.morphTargetManager._bind(effect);\n }\n }\n // Draw\n renderingMesh._processRendering(effectiveMesh, subMesh, effect, material.fillMode, batch, hardwareInstancedRendering, (isInstance, world) => effect.setMatrix(\"world\", world));\n }\n };\n this._depthMap.customRenderFunction = (opaqueSubMeshes, alphaTestSubMeshes, transparentSubMeshes, depthOnlySubMeshes) => {\n let index;\n if (depthOnlySubMeshes.length) {\n for (index = 0; index < depthOnlySubMeshes.length; index++) {\n renderSubMesh(depthOnlySubMeshes.data[index]);\n }\n }\n for (index = 0; index < opaqueSubMeshes.length; index++) {\n renderSubMesh(opaqueSubMeshes.data[index]);\n }\n for (index = 0; index < alphaTestSubMeshes.length; index++) {\n renderSubMesh(alphaTestSubMeshes.data[index]);\n }\n if (this.forceDepthWriteTransparentMeshes) {\n for (index = 0; index < transparentSubMeshes.length; index++) {\n renderSubMesh(transparentSubMeshes.data[index]);\n }\n }\n else {\n for (index = 0; index < transparentSubMeshes.length; index++) {\n transparentSubMeshes.data[index].getEffectiveMesh()._internalAbstractMeshDataInfo._isActiveIntermediate = false;\n }\n }\n };\n }\n /**\n * Creates the depth rendering effect and checks if the effect is ready.\n * @param subMesh The submesh to be used to render the depth map of\n * @param useInstances If multiple world instances should be used\n * @returns if the depth renderer is ready to render the depth map\n */\n isReady(subMesh, useInstances) {\n var _a;\n const engine = this._scene.getEngine();\n const mesh = subMesh.getMesh();\n const scene = mesh.getScene();\n const renderingMaterial = (_a = mesh._internalAbstractMeshDataInfo._materialForRenderPass) === null || _a === void 0 ? void 0 : _a[engine.currentRenderPassId];\n if (renderingMaterial) {\n return renderingMaterial.isReadyForSubMesh(mesh, subMesh, useInstances);\n }\n const material = subMesh.getMaterial();\n if (!material || material.disableDepthWrite) {\n return false;\n }\n const defines = [];\n const attribs = [VertexBuffer.PositionKind];\n // Alpha test\n if (material && material.needAlphaTesting() && material.getAlphaTestTexture()) {\n defines.push(\"#define ALPHATEST\");\n if (mesh.isVerticesDataPresent(VertexBuffer.UVKind)) {\n attribs.push(VertexBuffer.UVKind);\n defines.push(\"#define UV1\");\n }\n if (mesh.isVerticesDataPresent(VertexBuffer.UV2Kind)) {\n attribs.push(VertexBuffer.UV2Kind);\n defines.push(\"#define UV2\");\n }\n }\n // Bones\n if (mesh.useBones && mesh.computeBonesUsingShaders) {\n attribs.push(VertexBuffer.MatricesIndicesKind);\n attribs.push(VertexBuffer.MatricesWeightsKind);\n if (mesh.numBoneInfluencers > 4) {\n attribs.push(VertexBuffer.MatricesIndicesExtraKind);\n attribs.push(VertexBuffer.MatricesWeightsExtraKind);\n }\n defines.push(\"#define NUM_BONE_INFLUENCERS \" + mesh.numBoneInfluencers);\n defines.push(\"#define BonesPerMesh \" + (mesh.skeleton ? mesh.skeleton.bones.length + 1 : 0));\n const skeleton = subMesh.getRenderingMesh().skeleton;\n if (skeleton === null || skeleton === void 0 ? void 0 : skeleton.isUsingTextureForMatrices) {\n defines.push(\"#define BONETEXTURE\");\n }\n }\n else {\n defines.push(\"#define NUM_BONE_INFLUENCERS 0\");\n }\n // Morph targets\n const morphTargetManager = mesh.morphTargetManager;\n let numMorphInfluencers = 0;\n if (morphTargetManager) {\n if (morphTargetManager.numInfluencers > 0) {\n numMorphInfluencers = morphTargetManager.numInfluencers;\n defines.push(\"#define MORPHTARGETS\");\n defines.push(\"#define NUM_MORPH_INFLUENCERS \" + numMorphInfluencers);\n if (morphTargetManager.isUsingTextureForTargets) {\n defines.push(\"#define MORPHTARGETS_TEXTURE\");\n }\n MaterialHelper.PrepareAttributesForMorphTargetsInfluencers(attribs, mesh, numMorphInfluencers);\n }\n }\n // Instances\n if (useInstances) {\n defines.push(\"#define INSTANCES\");\n MaterialHelper.PushAttributesForInstances(attribs);\n if (subMesh.getRenderingMesh().hasThinInstances) {\n defines.push(\"#define THIN_INSTANCES\");\n }\n }\n // None linear depth\n if (this._storeNonLinearDepth) {\n defines.push(\"#define NONLINEARDEPTH\");\n }\n // Store camera space Z coordinate instead of NDC Z\n if (this._storeCameraSpaceZ) {\n defines.push(\"#define STORE_CAMERASPACE_Z\");\n }\n // Float Mode\n if (this.isPacked) {\n defines.push(\"#define PACKED\");\n }\n // Clip planes\n prepareStringDefinesForClipPlanes(material, scene, defines);\n // Get correct effect\n const drawWrapper = subMesh._getDrawWrapper(undefined, true);\n const cachedDefines = drawWrapper.defines;\n const join = defines.join(\"\\n\");\n if (cachedDefines !== join) {\n const uniforms = [\n \"world\",\n \"mBones\",\n \"boneTextureWidth\",\n \"viewProjection\",\n \"view\",\n \"diffuseMatrix\",\n \"depthValues\",\n \"morphTargetInfluences\",\n \"morphTargetTextureInfo\",\n \"morphTargetTextureIndices\",\n ];\n addClipPlaneUniforms(uniforms);\n drawWrapper.setEffect(engine.createEffect(\"depth\", attribs, uniforms, [\"diffuseSampler\", \"morphTargets\", \"boneSampler\"], join, undefined, undefined, undefined, {\n maxSimultaneousMorphTargets: numMorphInfluencers,\n }), join);\n }\n return drawWrapper.effect.isReady();\n }\n /**\n * Gets the texture which the depth map will be written to.\n * @returns The depth map texture\n */\n getDepthMap() {\n return this._depthMap;\n }\n /**\n * Disposes of the depth renderer.\n */\n dispose() {\n const keysToDelete = [];\n for (const key in this._scene._depthRenderer) {\n const depthRenderer = this._scene._depthRenderer[key];\n if (depthRenderer === this) {\n keysToDelete.push(key);\n }\n }\n if (keysToDelete.length > 0) {\n this._depthMap.dispose();\n for (const key of keysToDelete) {\n delete this._scene._depthRenderer[key];\n }\n }\n }\n }\n /**\n * @internal\n */\n DepthRenderer._SceneComponentInitialization = (_) => {\n throw _WarnImport(\"DepthRendererSceneComponent\");\n };\n\n // Do not edit.\n const name$2g = \"minmaxReduxPixelShader\";\n const shader$2g = `varying vec2 vUV;uniform sampler2D textureSampler;\n#if defined(INITIAL)\nuniform sampler2D sourceTexture;uniform vec2 texSize;void main(void)\n{ivec2 coord=ivec2(vUV*(texSize-1.0));float f1=texelFetch(sourceTexture,coord,0).r;float f2=texelFetch(sourceTexture,coord+ivec2(1,0),0).r;float f3=texelFetch(sourceTexture,coord+ivec2(1,1),0).r;float f4=texelFetch(sourceTexture,coord+ivec2(0,1),0).r;float minz=min(min(min(f1,f2),f3),f4);\n#ifdef DEPTH_REDUX\nfloat maxz=max(max(max(sign(1.0-f1)*f1,sign(1.0-f2)*f2),sign(1.0-f3)*f3),sign(1.0-f4)*f4);\n#else\nfloat maxz=max(max(max(f1,f2),f3),f4);\n#endif\nglFragColor=vec4(minz,maxz,0.,0.);}\n#elif defined(MAIN)\nuniform vec2 texSize;void main(void)\n{ivec2 coord=ivec2(vUV*(texSize-1.0));vec2 f1=texelFetch(textureSampler,coord,0).rg;vec2 f2=texelFetch(textureSampler,coord+ivec2(1,0),0).rg;vec2 f3=texelFetch(textureSampler,coord+ivec2(1,1),0).rg;vec2 f4=texelFetch(textureSampler,coord+ivec2(0,1),0).rg;float minz=min(min(min(f1.x,f2.x),f3.x),f4.x);float maxz=max(max(max(f1.y,f2.y),f3.y),f4.y);glFragColor=vec4(minz,maxz,0.,0.);}\n#elif defined(ONEBEFORELAST)\nuniform ivec2 texSize;void main(void)\n{ivec2 coord=ivec2(vUV*vec2(texSize-1));vec2 f1=texelFetch(textureSampler,coord % texSize,0).rg;vec2 f2=texelFetch(textureSampler,(coord+ivec2(1,0)) % texSize,0).rg;vec2 f3=texelFetch(textureSampler,(coord+ivec2(1,1)) % texSize,0).rg;vec2 f4=texelFetch(textureSampler,(coord+ivec2(0,1)) % texSize,0).rg;float minz=min(f1.x,f2.x);float maxz=max(f1.y,f2.y);glFragColor=vec4(minz,maxz,0.,0.);}\n#elif defined(LAST)\nvoid main(void)\n{glFragColor=vec4(0.);if (true) { \ndiscard;}}\n#endif\n`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2g] = shader$2g;\n\n /**\n * This class computes a min/max reduction from a texture: it means it computes the minimum\n * and maximum values from all values of the texture.\n * It is performed on the GPU for better performances, thanks to a succession of post processes.\n * The source values are read from the red channel of the texture.\n */\n class MinMaxReducer {\n /**\n * Creates a min/max reducer\n * @param camera The camera to use for the post processes\n */\n constructor(camera) {\n /**\n * Observable triggered when the computation has been performed\n */\n this.onAfterReductionPerformed = new Observable$1();\n this._forceFullscreenViewport = true;\n this._activated = false;\n this._camera = camera;\n this._postProcessManager = new PostProcessManager(camera.getScene());\n this._onContextRestoredObserver = camera.getEngine().onContextRestoredObservable.add(() => {\n this._postProcessManager._rebuild();\n });\n }\n /**\n * Gets the texture used to read the values from.\n */\n get sourceTexture() {\n return this._sourceTexture;\n }\n /**\n * Sets the source texture to read the values from.\n * One must indicate if the texture is a depth texture or not through the depthRedux parameter\n * because in such textures '1' value must not be taken into account to compute the maximum\n * as this value is used to clear the texture.\n * Note that the computation is not activated by calling this function, you must call activate() for that!\n * @param sourceTexture The texture to read the values from. The values should be in the red channel.\n * @param depthRedux Indicates if the texture is a depth texture or not\n * @param type The type of the textures created for the reduction (defaults to TEXTURETYPE_HALF_FLOAT)\n * @param forceFullscreenViewport Forces the post processes used for the reduction to be applied without taking into account viewport (defaults to true)\n */\n setSourceTexture(sourceTexture, depthRedux, type = 2, forceFullscreenViewport = true) {\n if (sourceTexture === this._sourceTexture) {\n return;\n }\n this.dispose(false);\n this._sourceTexture = sourceTexture;\n this._reductionSteps = [];\n this._forceFullscreenViewport = forceFullscreenViewport;\n const scene = this._camera.getScene();\n // create the first step\n const reductionInitial = new PostProcess(\"Initial reduction phase\", \"minmaxRedux\", // shader\n [\"texSize\"], [\"sourceTexture\"], // textures\n 1.0, // options\n null, // camera\n 1, // sampling\n scene.getEngine(), // engine\n false, // reusable\n \"#define INITIAL\" + (depthRedux ? \"\\n#define DEPTH_REDUX\" : \"\"), // defines\n type, undefined, undefined, undefined, 7);\n reductionInitial.autoClear = false;\n reductionInitial.forceFullscreenViewport = forceFullscreenViewport;\n let w = this._sourceTexture.getRenderWidth(), h = this._sourceTexture.getRenderHeight();\n reductionInitial.onApply = ((w, h) => {\n return (effect) => {\n effect.setTexture(\"sourceTexture\", this._sourceTexture);\n effect.setFloat2(\"texSize\", w, h);\n };\n })(w, h);\n this._reductionSteps.push(reductionInitial);\n let index = 1;\n // create the additional steps\n while (w > 1 || h > 1) {\n w = Math.max(Math.round(w / 2), 1);\n h = Math.max(Math.round(h / 2), 1);\n const reduction = new PostProcess(\"Reduction phase \" + index, \"minmaxRedux\", // shader\n [\"texSize\"], null, { width: w, height: h }, // options\n null, // camera\n 1, // sampling\n scene.getEngine(), // engine\n false, // reusable\n \"#define \" + (w == 1 && h == 1 ? \"LAST\" : w == 1 || h == 1 ? \"ONEBEFORELAST\" : \"MAIN\"), // defines\n type, undefined, undefined, undefined, 7);\n reduction.autoClear = false;\n reduction.forceFullscreenViewport = forceFullscreenViewport;\n reduction.onApply = ((w, h) => {\n return (effect) => {\n if (w == 1 || h == 1) {\n effect.setInt2(\"texSize\", w, h);\n }\n else {\n effect.setFloat2(\"texSize\", w, h);\n }\n };\n })(w, h);\n this._reductionSteps.push(reduction);\n index++;\n if (w == 1 && h == 1) {\n const func = (w, h, reduction) => {\n const buffer = new Float32Array(4 * w * h), minmax = { min: 0, max: 0 };\n return () => {\n scene.getEngine()._readTexturePixels(reduction.inputTexture.texture, w, h, -1, 0, buffer, false);\n minmax.min = buffer[0];\n minmax.max = buffer[1];\n this.onAfterReductionPerformed.notifyObservers(minmax);\n };\n };\n reduction.onAfterRenderObservable.add(func(w, h, reduction));\n }\n }\n }\n /**\n * Defines the refresh rate of the computation.\n * Use 0 to compute just once, 1 to compute on every frame, 2 to compute every two frames and so on...\n */\n get refreshRate() {\n return this._sourceTexture ? this._sourceTexture.refreshRate : -1;\n }\n set refreshRate(value) {\n if (this._sourceTexture) {\n this._sourceTexture.refreshRate = value;\n }\n }\n /**\n * Gets the activation status of the reducer\n */\n get activated() {\n return this._activated;\n }\n /**\n * Activates the reduction computation.\n * When activated, the observers registered in onAfterReductionPerformed are\n * called after the computation is performed\n */\n activate() {\n if (this._onAfterUnbindObserver || !this._sourceTexture) {\n return;\n }\n this._onAfterUnbindObserver = this._sourceTexture.onAfterUnbindObservable.add(() => {\n var _a, _b;\n const engine = this._camera.getScene().getEngine();\n (_a = engine._debugPushGroup) === null || _a === void 0 ? void 0 : _a.call(engine, `min max reduction`, 1);\n this._reductionSteps[0].activate(this._camera);\n this._postProcessManager.directRender(this._reductionSteps, this._reductionSteps[0].inputTexture, this._forceFullscreenViewport);\n engine.unBindFramebuffer(this._reductionSteps[0].inputTexture, false);\n (_b = engine._debugPopGroup) === null || _b === void 0 ? void 0 : _b.call(engine, 1);\n });\n this._activated = true;\n }\n /**\n * Deactivates the reduction computation.\n */\n deactivate() {\n if (!this._onAfterUnbindObserver || !this._sourceTexture) {\n return;\n }\n this._sourceTexture.onAfterUnbindObservable.remove(this._onAfterUnbindObserver);\n this._onAfterUnbindObserver = null;\n this._activated = false;\n }\n /**\n * Disposes the min/max reducer\n * @param disposeAll true to dispose all the resources. You should always call this function with true as the parameter (or without any parameter as it is the default one). This flag is meant to be used internally.\n */\n dispose(disposeAll = true) {\n if (disposeAll) {\n this.onAfterReductionPerformed.clear();\n if (this._onContextRestoredObserver) {\n this._camera.getEngine().onContextRestoredObservable.remove(this._onContextRestoredObserver);\n this._onContextRestoredObserver = null;\n }\n }\n this.deactivate();\n if (this._reductionSteps) {\n for (let i = 0; i < this._reductionSteps.length; ++i) {\n this._reductionSteps[i].dispose();\n }\n this._reductionSteps = null;\n }\n if (this._postProcessManager && disposeAll) {\n this._postProcessManager.dispose();\n }\n this._sourceTexture = null;\n }\n }\n\n /**\n * This class is a small wrapper around the MinMaxReducer class to compute the min/max values of a depth texture\n */\n class DepthReducer extends MinMaxReducer {\n /**\n * Gets the depth renderer used for the computation.\n * Note that the result is null if you provide your own renderer when calling setDepthRenderer.\n */\n get depthRenderer() {\n return this._depthRenderer;\n }\n /**\n * Creates a depth reducer\n * @param camera The camera used to render the depth texture\n */\n constructor(camera) {\n super(camera);\n }\n /**\n * Sets the depth renderer to use to generate the depth map\n * @param depthRenderer The depth renderer to use. If not provided, a new one will be created automatically\n * @param type The texture type of the depth map (default: TEXTURETYPE_HALF_FLOAT)\n * @param forceFullscreenViewport Forces the post processes used for the reduction to be applied without taking into account viewport (defaults to true)\n */\n setDepthRenderer(depthRenderer = null, type = 2, forceFullscreenViewport = true) {\n const scene = this._camera.getScene();\n if (this._depthRenderer) {\n delete scene._depthRenderer[this._depthRendererId];\n this._depthRenderer.dispose();\n this._depthRenderer = null;\n }\n if (depthRenderer === null) {\n if (!scene._depthRenderer) {\n scene._depthRenderer = {};\n }\n depthRenderer = this._depthRenderer = new DepthRenderer(scene, type, this._camera, false, 1);\n depthRenderer.enabled = false;\n this._depthRendererId = \"minmax\" + this._camera.id;\n scene._depthRenderer[this._depthRendererId] = depthRenderer;\n }\n super.setSourceTexture(depthRenderer.getDepthMap(), true, type, forceFullscreenViewport);\n }\n /**\n * @internal\n */\n setSourceTexture(sourceTexture, depthRedux, type = 2, forceFullscreenViewport = true) {\n super.setSourceTexture(sourceTexture, depthRedux, type, forceFullscreenViewport);\n }\n /**\n * Activates the reduction computation.\n * When activated, the observers registered in onAfterReductionPerformed are\n * called after the computation is performed\n */\n activate() {\n if (this._depthRenderer) {\n this._depthRenderer.enabled = true;\n }\n super.activate();\n }\n /**\n * Deactivates the reduction computation.\n */\n deactivate() {\n super.deactivate();\n if (this._depthRenderer) {\n this._depthRenderer.enabled = false;\n }\n }\n /**\n * Disposes the depth reducer\n * @param disposeAll true to dispose all the resources. You should always call this function with true as the parameter (or without any parameter as it is the default one). This flag is meant to be used internally.\n */\n dispose(disposeAll = true) {\n super.dispose(disposeAll);\n if (this._depthRenderer && disposeAll) {\n const scene = this._depthRenderer.getDepthMap().getScene();\n if (scene) {\n delete scene._depthRenderer[this._depthRendererId];\n }\n this._depthRenderer.dispose();\n this._depthRenderer = null;\n }\n }\n }\n\n // eslint-disable-next-line @typescript-eslint/naming-convention\n const UpDir = Vector3.Up();\n // eslint-disable-next-line @typescript-eslint/naming-convention\n const ZeroVec = Vector3.Zero();\n const tmpv1 = new Vector3(), tmpv2 = new Vector3(), tmpMatrix = new Matrix();\n /**\n * A CSM implementation allowing casting shadows on large scenes.\n * Documentation : https://doc.babylonjs.com/babylon101/cascadedShadows\n * Based on: https://github.com/TheRealMJP/Shadows and https://johanmedestrom.wordpress.com/2016/03/18/opengl-cascaded-shadow-maps/\n */\n class CascadedShadowGenerator extends ShadowGenerator {\n _validateFilter(filter) {\n if (filter === ShadowGenerator.FILTER_NONE || filter === ShadowGenerator.FILTER_PCF || filter === ShadowGenerator.FILTER_PCSS) {\n return filter;\n }\n console.error('Unsupported filter \"' + filter + '\"!');\n return ShadowGenerator.FILTER_NONE;\n }\n /**\n * Gets or set the number of cascades used by the CSM.\n */\n get numCascades() {\n return this._numCascades;\n }\n set numCascades(value) {\n value = Math.min(Math.max(value, CascadedShadowGenerator.MIN_CASCADES_COUNT), CascadedShadowGenerator.MAX_CASCADES_COUNT);\n if (value === this._numCascades) {\n return;\n }\n this._numCascades = value;\n this.recreateShadowMap();\n this._recreateSceneUBOs();\n }\n /**\n * Enables or disables the shadow casters bounding info computation.\n * If your shadow casters don't move, you can disable this feature.\n * If it is enabled, the bounding box computation is done every frame.\n */\n get freezeShadowCastersBoundingInfo() {\n return this._freezeShadowCastersBoundingInfo;\n }\n set freezeShadowCastersBoundingInfo(freeze) {\n if (this._freezeShadowCastersBoundingInfoObservable && freeze) {\n this._scene.onBeforeRenderObservable.remove(this._freezeShadowCastersBoundingInfoObservable);\n this._freezeShadowCastersBoundingInfoObservable = null;\n }\n if (!this._freezeShadowCastersBoundingInfoObservable && !freeze) {\n this._freezeShadowCastersBoundingInfoObservable = this._scene.onBeforeRenderObservable.add(this._computeShadowCastersBoundingInfo.bind(this));\n }\n this._freezeShadowCastersBoundingInfo = freeze;\n if (freeze) {\n this._computeShadowCastersBoundingInfo();\n }\n }\n _computeShadowCastersBoundingInfo() {\n this._scbiMin.copyFromFloats(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\n this._scbiMax.copyFromFloats(Number.MIN_VALUE, Number.MIN_VALUE, Number.MIN_VALUE);\n if (this._shadowMap && this._shadowMap.renderList) {\n const renderList = this._shadowMap.renderList;\n for (let meshIndex = 0; meshIndex < renderList.length; meshIndex++) {\n const mesh = renderList[meshIndex];\n if (!mesh) {\n continue;\n }\n const boundingInfo = mesh.getBoundingInfo(), boundingBox = boundingInfo.boundingBox;\n this._scbiMin.minimizeInPlace(boundingBox.minimumWorld);\n this._scbiMax.maximizeInPlace(boundingBox.maximumWorld);\n }\n const meshes = this._scene.meshes;\n for (let meshIndex = 0; meshIndex < meshes.length; meshIndex++) {\n const mesh = meshes[meshIndex];\n if (!mesh || !mesh.isVisible || !mesh.isEnabled || !mesh.receiveShadows) {\n continue;\n }\n const boundingInfo = mesh.getBoundingInfo(), boundingBox = boundingInfo.boundingBox;\n this._scbiMin.minimizeInPlace(boundingBox.minimumWorld);\n this._scbiMax.maximizeInPlace(boundingBox.maximumWorld);\n }\n }\n this._shadowCastersBoundingInfo.reConstruct(this._scbiMin, this._scbiMax);\n }\n /**\n * Gets or sets the shadow casters bounding info.\n * If you provide your own shadow casters bounding info, first enable freezeShadowCastersBoundingInfo\n * so that the system won't overwrite the bounds you provide\n */\n get shadowCastersBoundingInfo() {\n return this._shadowCastersBoundingInfo;\n }\n set shadowCastersBoundingInfo(boundingInfo) {\n this._shadowCastersBoundingInfo = boundingInfo;\n }\n /**\n * Sets the minimal and maximal distances to use when computing the cascade breaks.\n *\n * The values of min / max are typically the depth zmin and zmax values of your scene, for a given frame.\n * If you don't know these values, simply leave them to their defaults and don't call this function.\n * @param min minimal distance for the breaks (default to 0.)\n * @param max maximal distance for the breaks (default to 1.)\n */\n setMinMaxDistance(min, max) {\n if (this._minDistance === min && this._maxDistance === max) {\n return;\n }\n if (min > max) {\n min = 0;\n max = 1;\n }\n if (min < 0) {\n min = 0;\n }\n if (max > 1) {\n max = 1;\n }\n this._minDistance = min;\n this._maxDistance = max;\n this._breaksAreDirty = true;\n }\n /** Gets the minimal distance used in the cascade break computation */\n get minDistance() {\n return this._minDistance;\n }\n /** Gets the maximal distance used in the cascade break computation */\n get maxDistance() {\n return this._maxDistance;\n }\n /**\n * Gets the class name of that object\n * @returns \"CascadedShadowGenerator\"\n */\n getClassName() {\n return CascadedShadowGenerator.CLASSNAME;\n }\n /**\n * Gets a cascade minimum extents\n * @param cascadeIndex index of the cascade\n * @returns the minimum cascade extents\n */\n getCascadeMinExtents(cascadeIndex) {\n return cascadeIndex >= 0 && cascadeIndex < this._numCascades ? this._cascadeMinExtents[cascadeIndex] : null;\n }\n /**\n * Gets a cascade maximum extents\n * @param cascadeIndex index of the cascade\n * @returns the maximum cascade extents\n */\n getCascadeMaxExtents(cascadeIndex) {\n return cascadeIndex >= 0 && cascadeIndex < this._numCascades ? this._cascadeMaxExtents[cascadeIndex] : null;\n }\n /**\n * Gets the shadow max z distance. It's the limit beyond which shadows are not displayed.\n * It defaults to camera.maxZ\n */\n get shadowMaxZ() {\n if (!this._getCamera()) {\n return 0;\n }\n return this._shadowMaxZ;\n }\n /**\n * Sets the shadow max z distance.\n */\n set shadowMaxZ(value) {\n const camera = this._getCamera();\n if (!camera) {\n this._shadowMaxZ = value;\n return;\n }\n if (this._shadowMaxZ === value || value < camera.minZ || value > camera.maxZ) {\n return;\n }\n this._shadowMaxZ = value;\n this._light._markMeshesAsLightDirty();\n this._breaksAreDirty = true;\n }\n /**\n * Gets or sets the debug flag.\n * When enabled, the cascades are materialized by different colors on the screen.\n */\n get debug() {\n return this._debug;\n }\n set debug(dbg) {\n this._debug = dbg;\n this._light._markMeshesAsLightDirty();\n }\n /**\n * Gets or sets the depth clamping value.\n *\n * When enabled, it improves the shadow quality because the near z plane of the light frustum don't need to be adjusted\n * to account for the shadow casters far away.\n *\n * Note that this property is incompatible with PCSS filtering, so it won't be used in that case.\n */\n get depthClamp() {\n return this._depthClamp;\n }\n set depthClamp(value) {\n this._depthClamp = value;\n }\n /**\n * Gets or sets the percentage of blending between two cascades (value between 0. and 1.).\n * It defaults to 0.1 (10% blending).\n */\n get cascadeBlendPercentage() {\n return this._cascadeBlendPercentage;\n }\n set cascadeBlendPercentage(value) {\n this._cascadeBlendPercentage = value;\n this._light._markMeshesAsLightDirty();\n }\n /**\n * Gets or set the lambda parameter.\n * This parameter is used to split the camera frustum and create the cascades.\n * It's a value between 0. and 1.: If 0, the split is a uniform split of the frustum, if 1 it is a logarithmic split.\n * For all values in-between, it's a linear combination of the uniform and logarithm split algorithm.\n */\n get lambda() {\n return this._lambda;\n }\n set lambda(value) {\n const lambda = Math.min(Math.max(value, 0), 1);\n if (this._lambda == lambda) {\n return;\n }\n this._lambda = lambda;\n this._breaksAreDirty = true;\n }\n /**\n * Gets the view matrix corresponding to a given cascade\n * @param cascadeNum cascade to retrieve the view matrix from\n * @returns the cascade view matrix\n */\n getCascadeViewMatrix(cascadeNum) {\n return cascadeNum >= 0 && cascadeNum < this._numCascades ? this._viewMatrices[cascadeNum] : null;\n }\n /**\n * Gets the projection matrix corresponding to a given cascade\n * @param cascadeNum cascade to retrieve the projection matrix from\n * @returns the cascade projection matrix\n */\n getCascadeProjectionMatrix(cascadeNum) {\n return cascadeNum >= 0 && cascadeNum < this._numCascades ? this._projectionMatrices[cascadeNum] : null;\n }\n /**\n * Gets the transformation matrix corresponding to a given cascade\n * @param cascadeNum cascade to retrieve the transformation matrix from\n * @returns the cascade transformation matrix\n */\n getCascadeTransformMatrix(cascadeNum) {\n return cascadeNum >= 0 && cascadeNum < this._numCascades ? this._transformMatrices[cascadeNum] : null;\n }\n /**\n * Sets the depth renderer to use when autoCalcDepthBounds is enabled.\n *\n * Note that if no depth renderer is set, a new one will be automatically created internally when necessary.\n *\n * You should call this function if you already have a depth renderer enabled in your scene, to avoid\n * doing multiple depth rendering each frame. If you provide your own depth renderer, make sure it stores linear depth!\n * @param depthRenderer The depth renderer to use when autoCalcDepthBounds is enabled. If you pass null or don't call this function at all, a depth renderer will be automatically created\n */\n setDepthRenderer(depthRenderer) {\n this._depthRenderer = depthRenderer;\n if (this._depthReducer) {\n this._depthReducer.setDepthRenderer(this._depthRenderer);\n }\n }\n /**\n * Gets or sets the autoCalcDepthBounds property.\n *\n * When enabled, a depth rendering pass is first performed (with an internally created depth renderer or with the one\n * you provide by calling setDepthRenderer). Then, a min/max reducing is applied on the depth map to compute the\n * minimal and maximal depth of the map and those values are used as inputs for the setMinMaxDistance() function.\n * It can greatly enhance the shadow quality, at the expense of more GPU works.\n * When using this option, you should increase the value of the lambda parameter, and even set it to 1 for best results.\n */\n get autoCalcDepthBounds() {\n return this._autoCalcDepthBounds;\n }\n set autoCalcDepthBounds(value) {\n const camera = this._getCamera();\n if (!camera) {\n return;\n }\n this._autoCalcDepthBounds = value;\n if (!value) {\n if (this._depthReducer) {\n this._depthReducer.deactivate();\n }\n this.setMinMaxDistance(0, 1);\n return;\n }\n if (!this._depthReducer) {\n this._depthReducer = new DepthReducer(camera);\n this._depthReducer.onAfterReductionPerformed.add((minmax) => {\n let min = minmax.min, max = minmax.max;\n if (min >= max) {\n min = 0;\n max = 1;\n }\n if (min != this._minDistance || max != this._maxDistance) {\n this.setMinMaxDistance(min, max);\n }\n });\n this._depthReducer.setDepthRenderer(this._depthRenderer);\n }\n this._depthReducer.activate();\n }\n /**\n * Defines the refresh rate of the min/max computation used when autoCalcDepthBounds is set to true\n * Use 0 to compute just once, 1 to compute on every frame, 2 to compute every two frames and so on...\n * Note that if you provided your own depth renderer through a call to setDepthRenderer, you are responsible\n * for setting the refresh rate on the renderer yourself!\n */\n get autoCalcDepthBoundsRefreshRate() {\n var _a, _b, _c;\n return (_c = (_b = (_a = this._depthReducer) === null || _a === void 0 ? void 0 : _a.depthRenderer) === null || _b === void 0 ? void 0 : _b.getDepthMap().refreshRate) !== null && _c !== void 0 ? _c : -1;\n }\n set autoCalcDepthBoundsRefreshRate(value) {\n var _a;\n if ((_a = this._depthReducer) === null || _a === void 0 ? void 0 : _a.depthRenderer) {\n this._depthReducer.depthRenderer.getDepthMap().refreshRate = value;\n }\n }\n /**\n * Create the cascade breaks according to the lambda, shadowMaxZ and min/max distance properties, as well as the camera near and far planes.\n * This function is automatically called when updating lambda, shadowMaxZ and min/max distances, however you should call it yourself if\n * you change the camera near/far planes!\n */\n splitFrustum() {\n this._breaksAreDirty = true;\n }\n _splitFrustum() {\n const camera = this._getCamera();\n if (!camera) {\n return;\n }\n const near = camera.minZ, far = camera.maxZ || this._shadowMaxZ, // account for infinite far plane (ie. maxZ = 0)\n cameraRange = far - near, minDistance = this._minDistance, maxDistance = this._shadowMaxZ < far && this._shadowMaxZ >= near ? Math.min((this._shadowMaxZ - near) / (far - near), this._maxDistance) : this._maxDistance;\n const minZ = near + minDistance * cameraRange, maxZ = near + maxDistance * cameraRange;\n const range = maxZ - minZ, ratio = maxZ / minZ;\n for (let cascadeIndex = 0; cascadeIndex < this._cascades.length; ++cascadeIndex) {\n const p = (cascadeIndex + 1) / this._numCascades, log = minZ * ratio ** p, uniform = minZ + range * p;\n const d = this._lambda * (log - uniform) + uniform;\n this._cascades[cascadeIndex].prevBreakDistance = cascadeIndex === 0 ? minDistance : this._cascades[cascadeIndex - 1].breakDistance;\n this._cascades[cascadeIndex].breakDistance = (d - near) / cameraRange;\n this._viewSpaceFrustumsZ[cascadeIndex] = d;\n this._frustumLengths[cascadeIndex] = (this._cascades[cascadeIndex].breakDistance - this._cascades[cascadeIndex].prevBreakDistance) * cameraRange;\n }\n this._breaksAreDirty = false;\n }\n _computeMatrices() {\n const scene = this._scene;\n const camera = this._getCamera();\n if (!camera) {\n return;\n }\n Vector3.NormalizeToRef(this._light.getShadowDirection(0), this._lightDirection);\n if (Math.abs(Vector3.Dot(this._lightDirection, Vector3.Up())) === 1.0) {\n this._lightDirection.z = 0.0000000000001; // Required to avoid perfectly perpendicular light\n }\n this._cachedDirection.copyFrom(this._lightDirection);\n const useReverseDepthBuffer = scene.getEngine().useReverseDepthBuffer;\n for (let cascadeIndex = 0; cascadeIndex < this._numCascades; ++cascadeIndex) {\n this._computeFrustumInWorldSpace(cascadeIndex);\n this._computeCascadeFrustum(cascadeIndex);\n this._cascadeMaxExtents[cascadeIndex].subtractToRef(this._cascadeMinExtents[cascadeIndex], tmpv1); // tmpv1 = cascadeExtents\n // Get position of the shadow camera\n this._frustumCenter[cascadeIndex].addToRef(this._lightDirection.scale(this._cascadeMinExtents[cascadeIndex].z), this._shadowCameraPos[cascadeIndex]);\n // Come up with a new orthographic camera for the shadow caster\n Matrix.LookAtLHToRef(this._shadowCameraPos[cascadeIndex], this._frustumCenter[cascadeIndex], UpDir, this._viewMatrices[cascadeIndex]);\n let minZ = 0, maxZ = tmpv1.z;\n // Try to tighten minZ and maxZ based on the bounding box of the shadow casters\n const boundingInfo = this._shadowCastersBoundingInfo;\n boundingInfo.update(this._viewMatrices[cascadeIndex]);\n maxZ = Math.min(maxZ, boundingInfo.boundingBox.maximumWorld.z);\n if (!this._depthClamp || this.filter === ShadowGenerator.FILTER_PCSS) {\n // If we don't use depth clamping, we must set minZ so that all shadow casters are in the light frustum\n minZ = Math.min(minZ, boundingInfo.boundingBox.minimumWorld.z);\n }\n else {\n // If using depth clamping, we can adjust minZ to reduce the [minZ, maxZ] range (and get some additional precision in the shadow map)\n minZ = Math.max(minZ, boundingInfo.boundingBox.minimumWorld.z);\n }\n Matrix.OrthoOffCenterLHToRef(this._cascadeMinExtents[cascadeIndex].x, this._cascadeMaxExtents[cascadeIndex].x, this._cascadeMinExtents[cascadeIndex].y, this._cascadeMaxExtents[cascadeIndex].y, useReverseDepthBuffer ? maxZ : minZ, useReverseDepthBuffer ? minZ : maxZ, this._projectionMatrices[cascadeIndex], scene.getEngine().isNDCHalfZRange);\n this._cascadeMinExtents[cascadeIndex].z = minZ;\n this._cascadeMaxExtents[cascadeIndex].z = maxZ;\n this._viewMatrices[cascadeIndex].multiplyToRef(this._projectionMatrices[cascadeIndex], this._transformMatrices[cascadeIndex]);\n // Create the rounding matrix, by projecting the world-space origin and determining\n // the fractional offset in texel space\n Vector3.TransformCoordinatesToRef(ZeroVec, this._transformMatrices[cascadeIndex], tmpv1); // tmpv1 = shadowOrigin\n tmpv1.scaleInPlace(this._mapSize / 2);\n tmpv2.copyFromFloats(Math.round(tmpv1.x), Math.round(tmpv1.y), Math.round(tmpv1.z)); // tmpv2 = roundedOrigin\n tmpv2.subtractInPlace(tmpv1).scaleInPlace(2 / this._mapSize); // tmpv2 = roundOffset\n Matrix.TranslationToRef(tmpv2.x, tmpv2.y, 0.0, tmpMatrix);\n this._projectionMatrices[cascadeIndex].multiplyToRef(tmpMatrix, this._projectionMatrices[cascadeIndex]);\n this._viewMatrices[cascadeIndex].multiplyToRef(this._projectionMatrices[cascadeIndex], this._transformMatrices[cascadeIndex]);\n this._transformMatrices[cascadeIndex].copyToArray(this._transformMatricesAsArray, cascadeIndex * 16);\n }\n }\n // Get the 8 points of the view frustum in world space\n _computeFrustumInWorldSpace(cascadeIndex) {\n const camera = this._getCamera();\n if (!camera) {\n return;\n }\n const prevSplitDist = this._cascades[cascadeIndex].prevBreakDistance, splitDist = this._cascades[cascadeIndex].breakDistance;\n const isNDCHalfZRange = this._scene.getEngine().isNDCHalfZRange;\n camera.getViewMatrix(); // make sure the transformation matrix we get when calling 'getTransformationMatrix()' is calculated with an up to date view matrix\n const cameraInfiniteFarPlane = camera.maxZ === 0;\n const saveCameraMaxZ = camera.maxZ;\n if (cameraInfiniteFarPlane) {\n camera.maxZ = this._shadowMaxZ;\n camera.getProjectionMatrix(true);\n }\n const invViewProj = Matrix.Invert(camera.getTransformationMatrix());\n if (cameraInfiniteFarPlane) {\n camera.maxZ = saveCameraMaxZ;\n camera.getProjectionMatrix(true);\n }\n const cornerIndexOffset = this._scene.getEngine().useReverseDepthBuffer ? 4 : 0;\n for (let cornerIndex = 0; cornerIndex < CascadedShadowGenerator._FrustumCornersNDCSpace.length; ++cornerIndex) {\n tmpv1.copyFrom(CascadedShadowGenerator._FrustumCornersNDCSpace[(cornerIndex + cornerIndexOffset) % CascadedShadowGenerator._FrustumCornersNDCSpace.length]);\n if (isNDCHalfZRange && tmpv1.z === -1) {\n tmpv1.z = 0;\n }\n Vector3.TransformCoordinatesToRef(tmpv1, invViewProj, this._frustumCornersWorldSpace[cascadeIndex][cornerIndex]);\n }\n // Get the corners of the current cascade slice of the view frustum\n for (let cornerIndex = 0; cornerIndex < CascadedShadowGenerator._FrustumCornersNDCSpace.length / 2; ++cornerIndex) {\n tmpv1.copyFrom(this._frustumCornersWorldSpace[cascadeIndex][cornerIndex + 4]).subtractInPlace(this._frustumCornersWorldSpace[cascadeIndex][cornerIndex]);\n tmpv2.copyFrom(tmpv1).scaleInPlace(prevSplitDist); // near corner ray\n tmpv1.scaleInPlace(splitDist); // far corner ray\n tmpv1.addInPlace(this._frustumCornersWorldSpace[cascadeIndex][cornerIndex]);\n this._frustumCornersWorldSpace[cascadeIndex][cornerIndex + 4].copyFrom(tmpv1);\n this._frustumCornersWorldSpace[cascadeIndex][cornerIndex].addInPlace(tmpv2);\n }\n }\n _computeCascadeFrustum(cascadeIndex) {\n this._cascadeMinExtents[cascadeIndex].copyFromFloats(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\n this._cascadeMaxExtents[cascadeIndex].copyFromFloats(Number.MIN_VALUE, Number.MIN_VALUE, Number.MIN_VALUE);\n this._frustumCenter[cascadeIndex].copyFromFloats(0, 0, 0);\n const camera = this._getCamera();\n if (!camera) {\n return;\n }\n // Calculate the centroid of the view frustum slice\n for (let cornerIndex = 0; cornerIndex < this._frustumCornersWorldSpace[cascadeIndex].length; ++cornerIndex) {\n this._frustumCenter[cascadeIndex].addInPlace(this._frustumCornersWorldSpace[cascadeIndex][cornerIndex]);\n }\n this._frustumCenter[cascadeIndex].scaleInPlace(1 / this._frustumCornersWorldSpace[cascadeIndex].length);\n if (this.stabilizeCascades) {\n // Calculate the radius of a bounding sphere surrounding the frustum corners\n let sphereRadius = 0;\n for (let cornerIndex = 0; cornerIndex < this._frustumCornersWorldSpace[cascadeIndex].length; ++cornerIndex) {\n const dist = this._frustumCornersWorldSpace[cascadeIndex][cornerIndex].subtractToRef(this._frustumCenter[cascadeIndex], tmpv1).length();\n sphereRadius = Math.max(sphereRadius, dist);\n }\n sphereRadius = Math.ceil(sphereRadius * 16) / 16;\n this._cascadeMaxExtents[cascadeIndex].copyFromFloats(sphereRadius, sphereRadius, sphereRadius);\n this._cascadeMinExtents[cascadeIndex].copyFromFloats(-sphereRadius, -sphereRadius, -sphereRadius);\n }\n else {\n // Create a temporary view matrix for the light\n const lightCameraPos = this._frustumCenter[cascadeIndex];\n this._frustumCenter[cascadeIndex].addToRef(this._lightDirection, tmpv1); // tmpv1 = look at\n Matrix.LookAtLHToRef(lightCameraPos, tmpv1, UpDir, tmpMatrix); // matrix = lightView\n // Calculate an AABB around the frustum corners\n for (let cornerIndex = 0; cornerIndex < this._frustumCornersWorldSpace[cascadeIndex].length; ++cornerIndex) {\n Vector3.TransformCoordinatesToRef(this._frustumCornersWorldSpace[cascadeIndex][cornerIndex], tmpMatrix, tmpv1);\n this._cascadeMinExtents[cascadeIndex].minimizeInPlace(tmpv1);\n this._cascadeMaxExtents[cascadeIndex].maximizeInPlace(tmpv1);\n }\n }\n }\n _recreateSceneUBOs() {\n this._disposeSceneUBOs();\n if (this._sceneUBOs) {\n for (let i = 0; i < this._numCascades; ++i) {\n this._sceneUBOs.push(this._scene.createSceneUniformBuffer(`Scene for CSM Shadow Generator (light \"${this._light.name}\" cascade #${i})`));\n }\n }\n }\n /**\n * Support test.\n */\n static get IsSupported() {\n const engine = EngineStore.LastCreatedEngine;\n if (!engine) {\n return false;\n }\n return engine._features.supportCSM;\n }\n /**\n * Creates a Cascaded Shadow Generator object.\n * A ShadowGenerator is the required tool to use the shadows.\n * Each directional light casting shadows needs to use its own ShadowGenerator.\n * Documentation : https://doc.babylonjs.com/babylon101/cascadedShadows\n * @param mapSize The size of the texture what stores the shadows. Example : 1024.\n * @param light The directional light object generating the shadows.\n * @param usefulFloatFirst By default the generator will try to use half float textures but if you need precision (for self shadowing for instance), you can use this option to enforce full float texture.\n * @param camera Camera associated with this shadow generator (default: null). If null, takes the scene active camera at the time we need to access it\n * @param useRedTextureType Forces the generator to use a Red instead of a RGBA type for the shadow map texture format (default: true)\n */\n constructor(mapSize, light, usefulFloatFirst, camera, useRedTextureType = true) {\n if (!CascadedShadowGenerator.IsSupported) {\n Logger.Error(\"CascadedShadowMap is not supported by the current engine.\");\n return;\n }\n super(mapSize, light, usefulFloatFirst, camera, useRedTextureType);\n this.usePercentageCloserFiltering = true;\n }\n _initializeGenerator() {\n var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m, _o, _p, _q, _r, _s, _t, _u, _v;\n this.penumbraDarkness = (_a = this.penumbraDarkness) !== null && _a !== void 0 ? _a : 1.0;\n this._numCascades = (_b = this._numCascades) !== null && _b !== void 0 ? _b : CascadedShadowGenerator.DEFAULT_CASCADES_COUNT;\n this.stabilizeCascades = (_c = this.stabilizeCascades) !== null && _c !== void 0 ? _c : false;\n this._freezeShadowCastersBoundingInfoObservable = (_d = this._freezeShadowCastersBoundingInfoObservable) !== null && _d !== void 0 ? _d : null;\n this.freezeShadowCastersBoundingInfo = (_e = this.freezeShadowCastersBoundingInfo) !== null && _e !== void 0 ? _e : false;\n this._scbiMin = (_f = this._scbiMin) !== null && _f !== void 0 ? _f : new Vector3(0, 0, 0);\n this._scbiMax = (_g = this._scbiMax) !== null && _g !== void 0 ? _g : new Vector3(0, 0, 0);\n this._shadowCastersBoundingInfo = (_h = this._shadowCastersBoundingInfo) !== null && _h !== void 0 ? _h : new BoundingInfo(new Vector3(0, 0, 0), new Vector3(0, 0, 0));\n this._breaksAreDirty = (_j = this._breaksAreDirty) !== null && _j !== void 0 ? _j : true;\n this._minDistance = (_k = this._minDistance) !== null && _k !== void 0 ? _k : 0;\n this._maxDistance = (_l = this._maxDistance) !== null && _l !== void 0 ? _l : 1;\n this._currentLayer = (_m = this._currentLayer) !== null && _m !== void 0 ? _m : 0;\n this._shadowMaxZ = (_q = (_o = this._shadowMaxZ) !== null && _o !== void 0 ? _o : (_p = this._getCamera()) === null || _p === void 0 ? void 0 : _p.maxZ) !== null && _q !== void 0 ? _q : 10000;\n this._debug = (_r = this._debug) !== null && _r !== void 0 ? _r : false;\n this._depthClamp = (_s = this._depthClamp) !== null && _s !== void 0 ? _s : true;\n this._cascadeBlendPercentage = (_t = this._cascadeBlendPercentage) !== null && _t !== void 0 ? _t : 0.1;\n this._lambda = (_u = this._lambda) !== null && _u !== void 0 ? _u : 0.5;\n this._autoCalcDepthBounds = (_v = this._autoCalcDepthBounds) !== null && _v !== void 0 ? _v : false;\n this._recreateSceneUBOs();\n super._initializeGenerator();\n }\n _createTargetRenderTexture() {\n const engine = this._scene.getEngine();\n const size = { width: this._mapSize, height: this._mapSize, layers: this.numCascades };\n this._shadowMap = new RenderTargetTexture(this._light.name + \"_CSMShadowMap\", size, this._scene, false, true, this._textureType, false, undefined, false, false, undefined, this._useRedTextureType ? 6 : 5);\n this._shadowMap.createDepthStencilTexture(engine.useReverseDepthBuffer ? 516 : 513, true);\n this._shadowMap.noPrePassRenderer = true;\n }\n _initializeShadowMap() {\n super._initializeShadowMap();\n if (this._shadowMap === null) {\n return;\n }\n this._transformMatricesAsArray = new Float32Array(this._numCascades * 16);\n this._viewSpaceFrustumsZ = new Array(this._numCascades);\n this._frustumLengths = new Array(this._numCascades);\n this._lightSizeUVCorrection = new Array(this._numCascades * 2);\n this._depthCorrection = new Array(this._numCascades);\n this._cascades = [];\n this._viewMatrices = [];\n this._projectionMatrices = [];\n this._transformMatrices = [];\n this._cascadeMinExtents = [];\n this._cascadeMaxExtents = [];\n this._frustumCenter = [];\n this._shadowCameraPos = [];\n this._frustumCornersWorldSpace = [];\n for (let cascadeIndex = 0; cascadeIndex < this._numCascades; ++cascadeIndex) {\n this._cascades[cascadeIndex] = {\n prevBreakDistance: 0,\n breakDistance: 0,\n };\n this._viewMatrices[cascadeIndex] = Matrix.Zero();\n this._projectionMatrices[cascadeIndex] = Matrix.Zero();\n this._transformMatrices[cascadeIndex] = Matrix.Zero();\n this._cascadeMinExtents[cascadeIndex] = new Vector3();\n this._cascadeMaxExtents[cascadeIndex] = new Vector3();\n this._frustumCenter[cascadeIndex] = new Vector3();\n this._shadowCameraPos[cascadeIndex] = new Vector3();\n this._frustumCornersWorldSpace[cascadeIndex] = new Array(CascadedShadowGenerator._FrustumCornersNDCSpace.length);\n for (let i = 0; i < CascadedShadowGenerator._FrustumCornersNDCSpace.length; ++i) {\n this._frustumCornersWorldSpace[cascadeIndex][i] = new Vector3();\n }\n }\n const engine = this._scene.getEngine();\n this._shadowMap.onBeforeBindObservable.clear();\n this._shadowMap.onBeforeRenderObservable.clear();\n this._shadowMap.onBeforeRenderObservable.add((layer) => {\n if (this._sceneUBOs) {\n this._scene.setSceneUniformBuffer(this._sceneUBOs[layer]);\n }\n this._currentLayer = layer;\n if (this._filter === ShadowGenerator.FILTER_PCF) {\n engine.setColorWrite(false);\n }\n this._scene.setTransformMatrix(this.getCascadeViewMatrix(layer), this.getCascadeProjectionMatrix(layer));\n if (this._useUBO) {\n this._scene.getSceneUniformBuffer().unbindEffect();\n this._scene.finalizeSceneUbo();\n }\n });\n this._shadowMap.onBeforeBindObservable.add(() => {\n var _a;\n this._currentSceneUBO = this._scene.getSceneUniformBuffer();\n (_a = engine._debugPushGroup) === null || _a === void 0 ? void 0 : _a.call(engine, `cascaded shadow map generation for pass id ${engine.currentRenderPassId}`, 1);\n if (this._breaksAreDirty) {\n this._splitFrustum();\n }\n this._computeMatrices();\n });\n this._splitFrustum();\n }\n _bindCustomEffectForRenderSubMeshForShadowMap(subMesh, effect) {\n effect.setMatrix(\"viewProjection\", this.getCascadeTransformMatrix(this._currentLayer));\n }\n _isReadyCustomDefines(defines) {\n defines.push(\"#define SM_DEPTHCLAMP \" + (this._depthClamp && this._filter !== ShadowGenerator.FILTER_PCSS ? \"1\" : \"0\"));\n }\n /**\n * Prepare all the defines in a material relying on a shadow map at the specified light index.\n * @param defines Defines of the material we want to update\n * @param lightIndex Index of the light in the enabled light list of the material\n */\n prepareDefines(defines, lightIndex) {\n super.prepareDefines(defines, lightIndex);\n const scene = this._scene;\n const light = this._light;\n if (!scene.shadowsEnabled || !light.shadowEnabled) {\n return;\n }\n defines[\"SHADOWCSM\" + lightIndex] = true;\n defines[\"SHADOWCSMDEBUG\" + lightIndex] = this.debug;\n defines[\"SHADOWCSMNUM_CASCADES\" + lightIndex] = this.numCascades;\n defines[\"SHADOWCSM_RIGHTHANDED\" + lightIndex] = scene.useRightHandedSystem;\n const camera = this._getCamera();\n if (camera && this._shadowMaxZ <= (camera.maxZ || this._shadowMaxZ)) {\n defines[\"SHADOWCSMUSESHADOWMAXZ\" + lightIndex] = true;\n }\n if (this.cascadeBlendPercentage === 0) {\n defines[\"SHADOWCSMNOBLEND\" + lightIndex] = true;\n }\n }\n /**\n * Binds the shadow related information inside of an effect (information like near, far, darkness...\n * defined in the generator but impacting the effect).\n * @param lightIndex Index of the light in the enabled light list of the material owning the effect\n * @param effect The effect we are binfing the information for\n */\n bindShadowLight(lightIndex, effect) {\n const light = this._light;\n const scene = this._scene;\n if (!scene.shadowsEnabled || !light.shadowEnabled) {\n return;\n }\n const camera = this._getCamera();\n if (!camera) {\n return;\n }\n const shadowMap = this.getShadowMap();\n if (!shadowMap) {\n return;\n }\n const width = shadowMap.getSize().width;\n effect.setMatrices(\"lightMatrix\" + lightIndex, this._transformMatricesAsArray);\n effect.setArray(\"viewFrustumZ\" + lightIndex, this._viewSpaceFrustumsZ);\n effect.setFloat(\"cascadeBlendFactor\" + lightIndex, this.cascadeBlendPercentage === 0 ? 10000 : 1 / this.cascadeBlendPercentage);\n effect.setArray(\"frustumLengths\" + lightIndex, this._frustumLengths);\n // Only PCF uses depth stencil texture.\n if (this._filter === ShadowGenerator.FILTER_PCF) {\n effect.setDepthStencilTexture(\"shadowSampler\" + lightIndex, shadowMap);\n light._uniformBuffer.updateFloat4(\"shadowsInfo\", this.getDarkness(), width, 1 / width, this.frustumEdgeFalloff, lightIndex);\n }\n else if (this._filter === ShadowGenerator.FILTER_PCSS) {\n for (let cascadeIndex = 0; cascadeIndex < this._numCascades; ++cascadeIndex) {\n this._lightSizeUVCorrection[cascadeIndex * 2 + 0] =\n cascadeIndex === 0\n ? 1\n : (this._cascadeMaxExtents[0].x - this._cascadeMinExtents[0].x) / (this._cascadeMaxExtents[cascadeIndex].x - this._cascadeMinExtents[cascadeIndex].x); // x correction\n this._lightSizeUVCorrection[cascadeIndex * 2 + 1] =\n cascadeIndex === 0\n ? 1\n : (this._cascadeMaxExtents[0].y - this._cascadeMinExtents[0].y) / (this._cascadeMaxExtents[cascadeIndex].y - this._cascadeMinExtents[cascadeIndex].y); // y correction\n this._depthCorrection[cascadeIndex] =\n cascadeIndex === 0\n ? 1\n : (this._cascadeMaxExtents[cascadeIndex].z - this._cascadeMinExtents[cascadeIndex].z) / (this._cascadeMaxExtents[0].z - this._cascadeMinExtents[0].z);\n }\n effect.setDepthStencilTexture(\"shadowSampler\" + lightIndex, shadowMap);\n effect.setTexture(\"depthSampler\" + lightIndex, shadowMap);\n effect.setArray2(\"lightSizeUVCorrection\" + lightIndex, this._lightSizeUVCorrection);\n effect.setArray(\"depthCorrection\" + lightIndex, this._depthCorrection);\n effect.setFloat(\"penumbraDarkness\" + lightIndex, this.penumbraDarkness);\n light._uniformBuffer.updateFloat4(\"shadowsInfo\", this.getDarkness(), 1 / width, this._contactHardeningLightSizeUVRatio * width, this.frustumEdgeFalloff, lightIndex);\n }\n else {\n effect.setTexture(\"shadowSampler\" + lightIndex, shadowMap);\n light._uniformBuffer.updateFloat4(\"shadowsInfo\", this.getDarkness(), width, 1 / width, this.frustumEdgeFalloff, lightIndex);\n }\n light._uniformBuffer.updateFloat2(\"depthValues\", this.getLight().getDepthMinZ(camera), this.getLight().getDepthMinZ(camera) + this.getLight().getDepthMaxZ(camera), lightIndex);\n }\n /**\n * Gets the transformation matrix of the first cascade used to project the meshes into the map from the light point of view.\n * (eq to view projection * shadow projection matrices)\n * @returns The transform matrix used to create the shadow map\n */\n getTransformMatrix() {\n return this.getCascadeTransformMatrix(0);\n }\n /**\n * Disposes the ShadowGenerator.\n * Returns nothing.\n */\n dispose() {\n super.dispose();\n if (this._freezeShadowCastersBoundingInfoObservable) {\n this._scene.onBeforeRenderObservable.remove(this._freezeShadowCastersBoundingInfoObservable);\n this._freezeShadowCastersBoundingInfoObservable = null;\n }\n if (this._depthReducer) {\n this._depthReducer.dispose();\n this._depthReducer = null;\n }\n }\n /**\n * Serializes the shadow generator setup to a json object.\n * @returns The serialized JSON object\n */\n serialize() {\n const serializationObject = super.serialize();\n const shadowMap = this.getShadowMap();\n if (!shadowMap) {\n return serializationObject;\n }\n serializationObject.numCascades = this._numCascades;\n serializationObject.debug = this._debug;\n serializationObject.stabilizeCascades = this.stabilizeCascades;\n serializationObject.lambda = this._lambda;\n serializationObject.cascadeBlendPercentage = this.cascadeBlendPercentage;\n serializationObject.depthClamp = this._depthClamp;\n serializationObject.autoCalcDepthBounds = this.autoCalcDepthBounds;\n serializationObject.shadowMaxZ = this._shadowMaxZ;\n serializationObject.penumbraDarkness = this.penumbraDarkness;\n serializationObject.freezeShadowCastersBoundingInfo = this._freezeShadowCastersBoundingInfo;\n serializationObject.minDistance = this.minDistance;\n serializationObject.maxDistance = this.maxDistance;\n serializationObject.renderList = [];\n if (shadowMap.renderList) {\n for (let meshIndex = 0; meshIndex < shadowMap.renderList.length; meshIndex++) {\n const mesh = shadowMap.renderList[meshIndex];\n serializationObject.renderList.push(mesh.id);\n }\n }\n return serializationObject;\n }\n /**\n * Parses a serialized ShadowGenerator and returns a new ShadowGenerator.\n * @param parsedShadowGenerator The JSON object to parse\n * @param scene The scene to create the shadow map for\n * @returns The parsed shadow generator\n */\n static Parse(parsedShadowGenerator, scene) {\n const shadowGenerator = ShadowGenerator.Parse(parsedShadowGenerator, scene, (mapSize, light, camera) => new CascadedShadowGenerator(mapSize, light, undefined, camera));\n if (parsedShadowGenerator.numCascades !== undefined) {\n shadowGenerator.numCascades = parsedShadowGenerator.numCascades;\n }\n if (parsedShadowGenerator.debug !== undefined) {\n shadowGenerator.debug = parsedShadowGenerator.debug;\n }\n if (parsedShadowGenerator.stabilizeCascades !== undefined) {\n shadowGenerator.stabilizeCascades = parsedShadowGenerator.stabilizeCascades;\n }\n if (parsedShadowGenerator.lambda !== undefined) {\n shadowGenerator.lambda = parsedShadowGenerator.lambda;\n }\n if (parsedShadowGenerator.cascadeBlendPercentage !== undefined) {\n shadowGenerator.cascadeBlendPercentage = parsedShadowGenerator.cascadeBlendPercentage;\n }\n if (parsedShadowGenerator.depthClamp !== undefined) {\n shadowGenerator.depthClamp = parsedShadowGenerator.depthClamp;\n }\n if (parsedShadowGenerator.autoCalcDepthBounds !== undefined) {\n shadowGenerator.autoCalcDepthBounds = parsedShadowGenerator.autoCalcDepthBounds;\n }\n if (parsedShadowGenerator.shadowMaxZ !== undefined) {\n shadowGenerator.shadowMaxZ = parsedShadowGenerator.shadowMaxZ;\n }\n if (parsedShadowGenerator.penumbraDarkness !== undefined) {\n shadowGenerator.penumbraDarkness = parsedShadowGenerator.penumbraDarkness;\n }\n if (parsedShadowGenerator.freezeShadowCastersBoundingInfo !== undefined) {\n shadowGenerator.freezeShadowCastersBoundingInfo = parsedShadowGenerator.freezeShadowCastersBoundingInfo;\n }\n if (parsedShadowGenerator.minDistance !== undefined && parsedShadowGenerator.maxDistance !== undefined) {\n shadowGenerator.setMinMaxDistance(parsedShadowGenerator.minDistance, parsedShadowGenerator.maxDistance);\n }\n return shadowGenerator;\n }\n }\n CascadedShadowGenerator._FrustumCornersNDCSpace = [\n new Vector3(-1.0, +1.0, -1.0),\n new Vector3(+1.0, +1.0, -1.0),\n new Vector3(+1.0, -1.0, -1.0),\n new Vector3(-1.0, -1.0, -1.0),\n new Vector3(-1.0, +1.0, +1.0),\n new Vector3(+1.0, +1.0, +1.0),\n new Vector3(+1.0, -1.0, +1.0),\n new Vector3(-1.0, -1.0, +1.0),\n ];\n /**\n * Name of the CSM class\n */\n CascadedShadowGenerator.CLASSNAME = \"CascadedShadowGenerator\";\n /**\n * Defines the default number of cascades used by the CSM.\n */\n CascadedShadowGenerator.DEFAULT_CASCADES_COUNT = 4;\n /**\n * Defines the minimum number of cascades used by the CSM.\n */\n CascadedShadowGenerator.MIN_CASCADES_COUNT = 2;\n /**\n * Defines the maximum number of cascades used by the CSM.\n */\n CascadedShadowGenerator.MAX_CASCADES_COUNT = 4;\n /**\n * @internal\n */\n CascadedShadowGenerator._SceneComponentInitialization = (_) => {\n throw _WarnImport(\"ShadowGeneratorSceneComponent\");\n };\n\n // Adds the parser to the scene parsers.\n AbstractScene.AddParser(SceneComponentConstants.NAME_SHADOWGENERATOR, (parsedData, scene) => {\n // Shadows\n if (parsedData.shadowGenerators !== undefined && parsedData.shadowGenerators !== null) {\n for (let index = 0, cache = parsedData.shadowGenerators.length; index < cache; index++) {\n const parsedShadowGenerator = parsedData.shadowGenerators[index];\n if (parsedShadowGenerator.className === CascadedShadowGenerator.CLASSNAME) {\n CascadedShadowGenerator.Parse(parsedShadowGenerator, scene);\n }\n else {\n ShadowGenerator.Parse(parsedShadowGenerator, scene);\n }\n // SG would be available on their associated lights\n }\n }\n });\n /**\n * Defines the shadow generator component responsible to manage any shadow generators\n * in a given scene.\n */\n class ShadowGeneratorSceneComponent {\n /**\n * Creates a new instance of the component for the given scene\n * @param scene Defines the scene to register the component in\n */\n constructor(scene) {\n /**\n * The component name helpful to identify the component in the list of scene components.\n */\n this.name = SceneComponentConstants.NAME_SHADOWGENERATOR;\n this.scene = scene;\n }\n /**\n * Registers the component in a given scene\n */\n register() {\n this.scene._gatherRenderTargetsStage.registerStep(SceneComponentConstants.STEP_GATHERRENDERTARGETS_SHADOWGENERATOR, this, this._gatherRenderTargets);\n }\n /**\n * Rebuilds the elements related to this component in case of\n * context lost for instance.\n */\n rebuild() {\n // Nothing To Do Here.\n }\n /**\n * Serializes the component data to the specified json object\n * @param serializationObject The object to serialize to\n */\n serialize(serializationObject) {\n // Shadows\n serializationObject.shadowGenerators = [];\n const lights = this.scene.lights;\n for (const light of lights) {\n const shadowGenerators = light.getShadowGenerators();\n if (shadowGenerators) {\n const iterator = shadowGenerators.values();\n for (let key = iterator.next(); key.done !== true; key = iterator.next()) {\n const shadowGenerator = key.value;\n serializationObject.shadowGenerators.push(shadowGenerator.serialize());\n }\n }\n }\n }\n /**\n * Adds all the elements from the container to the scene\n * @param container the container holding the elements\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n addFromContainer(container) {\n // Nothing To Do Here. (directly attached to a light)\n }\n /**\n * Removes all the elements in the container from the scene\n * @param container contains the elements to remove\n * @param dispose if the removed element should be disposed (default: false)\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n removeFromContainer(container, dispose) {\n // Nothing To Do Here. (directly attached to a light)\n }\n /**\n * Rebuilds the elements related to this component in case of\n * context lost for instance.\n */\n dispose() {\n // Nothing To Do Here.\n }\n _gatherRenderTargets(renderTargets) {\n // Shadows\n const scene = this.scene;\n if (this.scene.shadowsEnabled) {\n for (let lightIndex = 0; lightIndex < scene.lights.length; lightIndex++) {\n const light = scene.lights[lightIndex];\n const shadowGenerators = light.getShadowGenerators();\n if (light.isEnabled() && light.shadowEnabled && shadowGenerators) {\n const iterator = shadowGenerators.values();\n for (let key = iterator.next(); key.done !== true; key = iterator.next()) {\n const shadowGenerator = key.value;\n const shadowMap = shadowGenerator.getShadowMap();\n if (scene.textures.indexOf(shadowMap) !== -1) {\n renderTargets.push(shadowMap);\n }\n }\n }\n }\n }\n }\n }\n ShadowGenerator._SceneComponentInitialization = (scene) => {\n let component = scene._getComponent(SceneComponentConstants.NAME_SHADOWGENERATOR);\n if (!component) {\n component = new ShadowGeneratorSceneComponent(scene);\n scene._addComponent(component);\n }\n };\n\n Node.AddNodeConstructor(\"Light_Type_1\", (name, scene) => {\n return () => new DirectionalLight(name, Vector3.Zero(), scene);\n });\n /**\n * A directional light is defined by a direction (what a surprise!).\n * The light is emitted from everywhere in the specified direction, and has an infinite range.\n * An example of a directional light is when a distance planet is lit by the apparently parallel lines of light from its sun. Light in a downward direction will light the top of an object.\n * Documentation: https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction\n */\n class DirectionalLight extends ShadowLight {\n /**\n * Fix frustum size for the shadow generation. This is disabled if the value is 0.\n */\n get shadowFrustumSize() {\n return this._shadowFrustumSize;\n }\n /**\n * Specifies a fix frustum size for the shadow generation.\n */\n set shadowFrustumSize(value) {\n this._shadowFrustumSize = value;\n this.forceProjectionMatrixCompute();\n }\n /**\n * Gets the shadow projection scale against the optimal computed one.\n * 0.1 by default which means that the projection window is increase by 10% from the optimal size.\n * This does not impact in fixed frustum size (shadowFrustumSize being set)\n */\n get shadowOrthoScale() {\n return this._shadowOrthoScale;\n }\n /**\n * Sets the shadow projection scale against the optimal computed one.\n * 0.1 by default which means that the projection window is increase by 10% from the optimal size.\n * This does not impact in fixed frustum size (shadowFrustumSize being set)\n */\n set shadowOrthoScale(value) {\n this._shadowOrthoScale = value;\n this.forceProjectionMatrixCompute();\n }\n /**\n * Gets or sets the orthoLeft property used to build the light frustum\n */\n get orthoLeft() {\n return this._orthoLeft;\n }\n set orthoLeft(left) {\n this._orthoLeft = left;\n }\n /**\n * Gets or sets the orthoRight property used to build the light frustum\n */\n get orthoRight() {\n return this._orthoRight;\n }\n set orthoRight(right) {\n this._orthoRight = right;\n }\n /**\n * Gets or sets the orthoTop property used to build the light frustum\n */\n get orthoTop() {\n return this._orthoTop;\n }\n set orthoTop(top) {\n this._orthoTop = top;\n }\n /**\n * Gets or sets the orthoBottom property used to build the light frustum\n */\n get orthoBottom() {\n return this._orthoBottom;\n }\n set orthoBottom(bottom) {\n this._orthoBottom = bottom;\n }\n /**\n * Creates a DirectionalLight object in the scene, oriented towards the passed direction (Vector3).\n * The directional light is emitted from everywhere in the given direction.\n * It can cast shadows.\n * Documentation : https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction\n * @param name The friendly name of the light\n * @param direction The direction of the light\n * @param scene The scene the light belongs to\n */\n constructor(name, direction, scene) {\n super(name, scene);\n this._shadowFrustumSize = 0;\n this._shadowOrthoScale = 0.1;\n /**\n * Automatically compute the projection matrix to best fit (including all the casters)\n * on each frame.\n */\n this.autoUpdateExtends = true;\n /**\n * Automatically compute the shadowMinZ and shadowMaxZ for the projection matrix to best fit (including all the casters)\n * on each frame. autoUpdateExtends must be set to true for this to work\n */\n this.autoCalcShadowZBounds = false;\n // Cache\n this._orthoLeft = Number.MAX_VALUE;\n this._orthoRight = Number.MIN_VALUE;\n this._orthoTop = Number.MIN_VALUE;\n this._orthoBottom = Number.MAX_VALUE;\n this.position = direction.scale(-1.0);\n this.direction = direction;\n }\n /**\n * Returns the string \"DirectionalLight\".\n * @returns The class name\n */\n getClassName() {\n return \"DirectionalLight\";\n }\n /**\n * Returns the integer 1.\n * @returns The light Type id as a constant defines in Light.LIGHTTYPEID_x\n */\n getTypeID() {\n return Light.LIGHTTYPEID_DIRECTIONALLIGHT;\n }\n /**\n * Sets the passed matrix \"matrix\" as projection matrix for the shadows cast by the light according to the passed view matrix.\n * Returns the DirectionalLight Shadow projection matrix.\n * @param matrix\n * @param viewMatrix\n * @param renderList\n */\n _setDefaultShadowProjectionMatrix(matrix, viewMatrix, renderList) {\n if (this.shadowFrustumSize > 0) {\n this._setDefaultFixedFrustumShadowProjectionMatrix(matrix);\n }\n else {\n this._setDefaultAutoExtendShadowProjectionMatrix(matrix, viewMatrix, renderList);\n }\n }\n /**\n * Sets the passed matrix \"matrix\" as fixed frustum projection matrix for the shadows cast by the light according to the passed view matrix.\n * Returns the DirectionalLight Shadow projection matrix.\n * @param matrix\n */\n _setDefaultFixedFrustumShadowProjectionMatrix(matrix) {\n const activeCamera = this.getScene().activeCamera;\n if (!activeCamera) {\n return;\n }\n Matrix.OrthoLHToRef(this.shadowFrustumSize, this.shadowFrustumSize, this.shadowMinZ !== undefined ? this.shadowMinZ : activeCamera.minZ, this.shadowMaxZ !== undefined ? this.shadowMaxZ : activeCamera.maxZ, matrix, this.getScene().getEngine().isNDCHalfZRange);\n }\n /**\n * Sets the passed matrix \"matrix\" as auto extend projection matrix for the shadows cast by the light according to the passed view matrix.\n * Returns the DirectionalLight Shadow projection matrix.\n * @param matrix\n * @param viewMatrix\n * @param renderList\n */\n _setDefaultAutoExtendShadowProjectionMatrix(matrix, viewMatrix, renderList) {\n const activeCamera = this.getScene().activeCamera;\n if (!activeCamera) {\n return;\n }\n // Check extends\n if (this.autoUpdateExtends || this._orthoLeft === Number.MAX_VALUE) {\n const tempVector3 = Vector3.Zero();\n this._orthoLeft = Number.MAX_VALUE;\n this._orthoRight = Number.MIN_VALUE;\n this._orthoTop = Number.MIN_VALUE;\n this._orthoBottom = Number.MAX_VALUE;\n let shadowMinZ = Number.MAX_VALUE;\n let shadowMaxZ = Number.MIN_VALUE;\n for (let meshIndex = 0; meshIndex < renderList.length; meshIndex++) {\n const mesh = renderList[meshIndex];\n if (!mesh) {\n continue;\n }\n const boundingInfo = mesh.getBoundingInfo();\n const boundingBox = boundingInfo.boundingBox;\n for (let index = 0; index < boundingBox.vectorsWorld.length; index++) {\n Vector3.TransformCoordinatesToRef(boundingBox.vectorsWorld[index], viewMatrix, tempVector3);\n if (tempVector3.x < this._orthoLeft) {\n this._orthoLeft = tempVector3.x;\n }\n if (tempVector3.y < this._orthoBottom) {\n this._orthoBottom = tempVector3.y;\n }\n if (tempVector3.x > this._orthoRight) {\n this._orthoRight = tempVector3.x;\n }\n if (tempVector3.y > this._orthoTop) {\n this._orthoTop = tempVector3.y;\n }\n if (this.autoCalcShadowZBounds) {\n if (tempVector3.z < shadowMinZ) {\n shadowMinZ = tempVector3.z;\n }\n if (tempVector3.z > shadowMaxZ) {\n shadowMaxZ = tempVector3.z;\n }\n }\n }\n }\n if (this.autoCalcShadowZBounds) {\n this._shadowMinZ = shadowMinZ;\n this._shadowMaxZ = shadowMaxZ;\n }\n }\n const xOffset = this._orthoRight - this._orthoLeft;\n const yOffset = this._orthoTop - this._orthoBottom;\n const minZ = this.shadowMinZ !== undefined ? this.shadowMinZ : activeCamera.minZ;\n const maxZ = this.shadowMaxZ !== undefined ? this.shadowMaxZ : activeCamera.maxZ;\n const useReverseDepthBuffer = this.getScene().getEngine().useReverseDepthBuffer;\n Matrix.OrthoOffCenterLHToRef(this._orthoLeft - xOffset * this.shadowOrthoScale, this._orthoRight + xOffset * this.shadowOrthoScale, this._orthoBottom - yOffset * this.shadowOrthoScale, this._orthoTop + yOffset * this.shadowOrthoScale, useReverseDepthBuffer ? maxZ : minZ, useReverseDepthBuffer ? minZ : maxZ, matrix, this.getScene().getEngine().isNDCHalfZRange);\n }\n _buildUniformLayout() {\n this._uniformBuffer.addUniform(\"vLightData\", 4);\n this._uniformBuffer.addUniform(\"vLightDiffuse\", 4);\n this._uniformBuffer.addUniform(\"vLightSpecular\", 4);\n this._uniformBuffer.addUniform(\"shadowsInfo\", 3);\n this._uniformBuffer.addUniform(\"depthValues\", 2);\n this._uniformBuffer.create();\n }\n /**\n * Sets the passed Effect object with the DirectionalLight transformed position (or position if not parented) and the passed name.\n * @param effect The effect to update\n * @param lightIndex The index of the light in the effect to update\n * @returns The directional light\n */\n transferToEffect(effect, lightIndex) {\n if (this.computeTransformedInformation()) {\n this._uniformBuffer.updateFloat4(\"vLightData\", this.transformedDirection.x, this.transformedDirection.y, this.transformedDirection.z, 1, lightIndex);\n return this;\n }\n this._uniformBuffer.updateFloat4(\"vLightData\", this.direction.x, this.direction.y, this.direction.z, 1, lightIndex);\n return this;\n }\n transferToNodeMaterialEffect(effect, lightDataUniformName) {\n if (this.computeTransformedInformation()) {\n effect.setFloat3(lightDataUniformName, this.transformedDirection.x, this.transformedDirection.y, this.transformedDirection.z);\n return this;\n }\n effect.setFloat3(lightDataUniformName, this.direction.x, this.direction.y, this.direction.z);\n return this;\n }\n /**\n * Gets the minZ used for shadow according to both the scene and the light.\n *\n * Values are fixed on directional lights as it relies on an ortho projection hence the need to convert being\n * -1 and 1 to 0 and 1 doing (depth + min) / (min + max) -> (depth + 1) / (1 + 1) -> (depth * 0.5) + 0.5.\n * (when not using reverse depth buffer / NDC half Z range)\n * @param activeCamera The camera we are returning the min for\n * @returns the depth min z\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n getDepthMinZ(activeCamera) {\n const engine = this._scene.getEngine();\n return !engine.useReverseDepthBuffer && engine.isNDCHalfZRange ? 0 : 1;\n }\n /**\n * Gets the maxZ used for shadow according to both the scene and the light.\n *\n * Values are fixed on directional lights as it relies on an ortho projection hence the need to convert being\n * -1 and 1 to 0 and 1 doing (depth + min) / (min + max) -> (depth + 1) / (1 + 1) -> (depth * 0.5) + 0.5.\n * (when not using reverse depth buffer / NDC half Z range)\n * @param activeCamera The camera we are returning the max for\n * @returns the depth max z\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n getDepthMaxZ(activeCamera) {\n const engine = this._scene.getEngine();\n return engine.useReverseDepthBuffer && engine.isNDCHalfZRange ? 0 : 1;\n }\n /**\n * Prepares the list of defines specific to the light type.\n * @param defines the list of defines\n * @param lightIndex defines the index of the light for the effect\n */\n prepareLightSpecificDefines(defines, lightIndex) {\n defines[\"DIRLIGHT\" + lightIndex] = true;\n }\n }\n __decorate$1([\n serialize()\n ], DirectionalLight.prototype, \"shadowFrustumSize\", null);\n __decorate$1([\n serialize()\n ], DirectionalLight.prototype, \"shadowOrthoScale\", null);\n __decorate$1([\n serialize()\n ], DirectionalLight.prototype, \"autoUpdateExtends\", void 0);\n __decorate$1([\n serialize()\n ], DirectionalLight.prototype, \"autoCalcShadowZBounds\", void 0);\n __decorate$1([\n serialize(\"orthoLeft\")\n ], DirectionalLight.prototype, \"_orthoLeft\", void 0);\n __decorate$1([\n serialize(\"orthoRight\")\n ], DirectionalLight.prototype, \"_orthoRight\", void 0);\n __decorate$1([\n serialize(\"orthoTop\")\n ], DirectionalLight.prototype, \"_orthoTop\", void 0);\n __decorate$1([\n serialize(\"orthoBottom\")\n ], DirectionalLight.prototype, \"_orthoBottom\", void 0);\n\n Node.AddNodeConstructor(\"Light_Type_0\", (name, scene) => {\n return () => new PointLight(name, Vector3.Zero(), scene);\n });\n /**\n * A point light is a light defined by an unique point in world space.\n * The light is emitted in every direction from this point.\n * A good example of a point light is a standard light bulb.\n * Documentation: https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction\n */\n class PointLight extends ShadowLight {\n /**\n * Getter: In case of direction provided, the shadow will not use a cube texture but simulate a spot shadow as a fallback\n * This specifies what angle the shadow will use to be created.\n *\n * It default to 90 degrees to work nicely with the cube texture generation for point lights shadow maps.\n */\n get shadowAngle() {\n return this._shadowAngle;\n }\n /**\n * Setter: In case of direction provided, the shadow will not use a cube texture but simulate a spot shadow as a fallback\n * This specifies what angle the shadow will use to be created.\n *\n * It default to 90 degrees to work nicely with the cube texture generation for point lights shadow maps.\n */\n set shadowAngle(value) {\n this._shadowAngle = value;\n this.forceProjectionMatrixCompute();\n }\n /**\n * Gets the direction if it has been set.\n * In case of direction provided, the shadow will not use a cube texture but simulate a spot shadow as a fallback\n */\n get direction() {\n return this._direction;\n }\n /**\n * In case of direction provided, the shadow will not use a cube texture but simulate a spot shadow as a fallback\n */\n set direction(value) {\n const previousNeedCube = this.needCube();\n this._direction = value;\n if (this.needCube() !== previousNeedCube && this._shadowGenerators) {\n const iterator = this._shadowGenerators.values();\n for (let key = iterator.next(); key.done !== true; key = iterator.next()) {\n const shadowGenerator = key.value;\n shadowGenerator.recreateShadowMap();\n }\n }\n }\n /**\n * Creates a PointLight object from the passed name and position (Vector3) and adds it in the scene.\n * A PointLight emits the light in every direction.\n * It can cast shadows.\n * If the scene camera is already defined and you want to set your PointLight at the camera position, just set it :\n * ```javascript\n * var pointLight = new PointLight(\"pl\", camera.position, scene);\n * ```\n * Documentation : https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction\n * @param name The light friendly name\n * @param position The position of the point light in the scene\n * @param scene The scene the lights belongs to\n */\n constructor(name, position, scene) {\n super(name, scene);\n this._shadowAngle = Math.PI / 2;\n this.position = position;\n }\n /**\n * Returns the string \"PointLight\"\n * @returns the class name\n */\n getClassName() {\n return \"PointLight\";\n }\n /**\n * Returns the integer 0.\n * @returns The light Type id as a constant defines in Light.LIGHTTYPEID_x\n */\n getTypeID() {\n return Light.LIGHTTYPEID_POINTLIGHT;\n }\n /**\n * Specifies whether or not the shadowmap should be a cube texture.\n * @returns true if the shadowmap needs to be a cube texture.\n */\n needCube() {\n return !this.direction;\n }\n /**\n * Returns a new Vector3 aligned with the PointLight cube system according to the passed cube face index (integer).\n * @param faceIndex The index of the face we are computed the direction to generate shadow\n * @returns The set direction in 2d mode otherwise the direction to the cubemap face if needCube() is true\n */\n getShadowDirection(faceIndex) {\n if (this.direction) {\n return super.getShadowDirection(faceIndex);\n }\n else {\n switch (faceIndex) {\n case 0:\n return new Vector3(1.0, 0.0, 0.0);\n case 1:\n return new Vector3(-1.0, 0.0, 0.0);\n case 2:\n return new Vector3(0.0, -1.0, 0.0);\n case 3:\n return new Vector3(0.0, 1.0, 0.0);\n case 4:\n return new Vector3(0.0, 0.0, 1.0);\n case 5:\n return new Vector3(0.0, 0.0, -1.0);\n }\n }\n return Vector3.Zero();\n }\n /**\n * Sets the passed matrix \"matrix\" as a left-handed perspective projection matrix with the following settings :\n * - fov = PI / 2\n * - aspect ratio : 1.0\n * - z-near and far equal to the active camera minZ and maxZ.\n * Returns the PointLight.\n * @param matrix\n * @param viewMatrix\n * @param renderList\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _setDefaultShadowProjectionMatrix(matrix, viewMatrix, renderList) {\n const activeCamera = this.getScene().activeCamera;\n if (!activeCamera) {\n return;\n }\n const minZ = this.shadowMinZ !== undefined ? this.shadowMinZ : activeCamera.minZ;\n const maxZ = this.shadowMaxZ !== undefined ? this.shadowMaxZ : activeCamera.maxZ;\n const useReverseDepthBuffer = this.getScene().getEngine().useReverseDepthBuffer;\n Matrix.PerspectiveFovLHToRef(this.shadowAngle, 1.0, useReverseDepthBuffer ? maxZ : minZ, useReverseDepthBuffer ? minZ : maxZ, matrix, true, this._scene.getEngine().isNDCHalfZRange, undefined, useReverseDepthBuffer);\n }\n _buildUniformLayout() {\n this._uniformBuffer.addUniform(\"vLightData\", 4);\n this._uniformBuffer.addUniform(\"vLightDiffuse\", 4);\n this._uniformBuffer.addUniform(\"vLightSpecular\", 4);\n this._uniformBuffer.addUniform(\"vLightFalloff\", 4);\n this._uniformBuffer.addUniform(\"shadowsInfo\", 3);\n this._uniformBuffer.addUniform(\"depthValues\", 2);\n this._uniformBuffer.create();\n }\n /**\n * Sets the passed Effect \"effect\" with the PointLight transformed position (or position, if none) and passed name (string).\n * @param effect The effect to update\n * @param lightIndex The index of the light in the effect to update\n * @returns The point light\n */\n transferToEffect(effect, lightIndex) {\n if (this.computeTransformedInformation()) {\n this._uniformBuffer.updateFloat4(\"vLightData\", this.transformedPosition.x, this.transformedPosition.y, this.transformedPosition.z, 0.0, lightIndex);\n }\n else {\n this._uniformBuffer.updateFloat4(\"vLightData\", this.position.x, this.position.y, this.position.z, 0, lightIndex);\n }\n this._uniformBuffer.updateFloat4(\"vLightFalloff\", this.range, this._inverseSquaredRange, 0, 0, lightIndex);\n return this;\n }\n transferToNodeMaterialEffect(effect, lightDataUniformName) {\n if (this.computeTransformedInformation()) {\n effect.setFloat3(lightDataUniformName, this.transformedPosition.x, this.transformedPosition.y, this.transformedPosition.z);\n }\n else {\n effect.setFloat3(lightDataUniformName, this.position.x, this.position.y, this.position.z);\n }\n return this;\n }\n /**\n * Prepares the list of defines specific to the light type.\n * @param defines the list of defines\n * @param lightIndex defines the index of the light for the effect\n */\n prepareLightSpecificDefines(defines, lightIndex) {\n defines[\"POINTLIGHT\" + lightIndex] = true;\n }\n }\n __decorate$1([\n serialize()\n ], PointLight.prototype, \"shadowAngle\", null);\n\n Node.AddNodeConstructor(\"Light_Type_2\", (name, scene) => {\n return () => new SpotLight(name, Vector3.Zero(), Vector3.Zero(), 0, 0, scene);\n });\n /**\n * A spot light is defined by a position, a direction, an angle, and an exponent.\n * These values define a cone of light starting from the position, emitting toward the direction.\n * The angle, in radians, defines the size (field of illumination) of the spotlight's conical beam,\n * and the exponent defines the speed of the decay of the light with distance (reach).\n * Documentation: https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction\n */\n class SpotLight extends ShadowLight {\n /**\n * Gets the cone angle of the spot light in Radians.\n */\n get angle() {\n return this._angle;\n }\n /**\n * Sets the cone angle of the spot light in Radians.\n */\n set angle(value) {\n this._angle = value;\n this._cosHalfAngle = Math.cos(value * 0.5);\n this._projectionTextureProjectionLightDirty = true;\n this.forceProjectionMatrixCompute();\n this._computeAngleValues();\n }\n /**\n * Only used in gltf falloff mode, this defines the angle where\n * the directional falloff will start before cutting at angle which could be seen\n * as outer angle.\n */\n get innerAngle() {\n return this._innerAngle;\n }\n /**\n * Only used in gltf falloff mode, this defines the angle where\n * the directional falloff will start before cutting at angle which could be seen\n * as outer angle.\n */\n set innerAngle(value) {\n this._innerAngle = value;\n this._computeAngleValues();\n }\n /**\n * Allows scaling the angle of the light for shadow generation only.\n */\n get shadowAngleScale() {\n return this._shadowAngleScale;\n }\n /**\n * Allows scaling the angle of the light for shadow generation only.\n */\n set shadowAngleScale(value) {\n this._shadowAngleScale = value;\n this.forceProjectionMatrixCompute();\n }\n /**\n * Allows reading the projection texture\n */\n get projectionTextureMatrix() {\n return this._projectionTextureMatrix;\n }\n /**\n * Gets the near clip of the Spotlight for texture projection.\n */\n get projectionTextureLightNear() {\n return this._projectionTextureLightNear;\n }\n /**\n * Sets the near clip of the Spotlight for texture projection.\n */\n set projectionTextureLightNear(value) {\n this._projectionTextureLightNear = value;\n this._projectionTextureProjectionLightDirty = true;\n }\n /**\n * Gets the far clip of the Spotlight for texture projection.\n */\n get projectionTextureLightFar() {\n return this._projectionTextureLightFar;\n }\n /**\n * Sets the far clip of the Spotlight for texture projection.\n */\n set projectionTextureLightFar(value) {\n this._projectionTextureLightFar = value;\n this._projectionTextureProjectionLightDirty = true;\n }\n /**\n * Gets the Up vector of the Spotlight for texture projection.\n */\n get projectionTextureUpDirection() {\n return this._projectionTextureUpDirection;\n }\n /**\n * Sets the Up vector of the Spotlight for texture projection.\n */\n set projectionTextureUpDirection(value) {\n this._projectionTextureUpDirection = value;\n this._projectionTextureProjectionLightDirty = true;\n }\n /**\n * Gets the projection texture of the light.\n */\n get projectionTexture() {\n return this._projectionTexture;\n }\n /**\n * Sets the projection texture of the light.\n */\n set projectionTexture(value) {\n if (this._projectionTexture === value) {\n return;\n }\n this._projectionTexture = value;\n this._projectionTextureDirty = true;\n if (this._projectionTexture && !this._projectionTexture.isReady()) {\n if (SpotLight._IsProceduralTexture(this._projectionTexture)) {\n this._projectionTexture.getEffect().executeWhenCompiled(() => {\n this._markMeshesAsLightDirty();\n });\n }\n else if (SpotLight._IsTexture(this._projectionTexture)) {\n this._projectionTexture.onLoadObservable.addOnce(() => {\n this._markMeshesAsLightDirty();\n });\n }\n }\n }\n static _IsProceduralTexture(texture) {\n return texture.onGeneratedObservable !== undefined;\n }\n static _IsTexture(texture) {\n return texture.onLoadObservable !== undefined;\n }\n /**\n * Gets or sets the light projection matrix as used by the projection texture\n */\n get projectionTextureProjectionLightMatrix() {\n return this._projectionTextureProjectionLightMatrix;\n }\n set projectionTextureProjectionLightMatrix(projection) {\n this._projectionTextureProjectionLightMatrix = projection;\n this._projectionTextureProjectionLightDirty = false;\n this._projectionTextureDirty = true;\n }\n /**\n * Creates a SpotLight object in the scene. A spot light is a simply light oriented cone.\n * It can cast shadows.\n * Documentation : https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction\n * @param name The light friendly name\n * @param position The position of the spot light in the scene\n * @param direction The direction of the light in the scene\n * @param angle The cone angle of the light in Radians\n * @param exponent The light decay speed with the distance from the emission spot\n * @param scene The scene the lights belongs to\n */\n constructor(name, position, direction, angle, exponent, scene) {\n super(name, scene);\n this._innerAngle = 0;\n this._projectionTextureMatrix = Matrix.Zero();\n this._projectionTextureLightNear = 1e-6;\n this._projectionTextureLightFar = 1000.0;\n this._projectionTextureUpDirection = Vector3.Up();\n this._projectionTextureViewLightDirty = true;\n this._projectionTextureProjectionLightDirty = true;\n this._projectionTextureDirty = true;\n this._projectionTextureViewTargetVector = Vector3.Zero();\n this._projectionTextureViewLightMatrix = Matrix.Zero();\n this._projectionTextureProjectionLightMatrix = Matrix.Zero();\n this._projectionTextureScalingMatrix = Matrix.FromValues(0.5, 0.0, 0.0, 0.0, 0.0, 0.5, 0.0, 0.0, 0.0, 0.0, 0.5, 0.0, 0.5, 0.5, 0.5, 1.0);\n this.position = position;\n this.direction = direction;\n this.angle = angle;\n this.exponent = exponent;\n }\n /**\n * Returns the string \"SpotLight\".\n * @returns the class name\n */\n getClassName() {\n return \"SpotLight\";\n }\n /**\n * Returns the integer 2.\n * @returns The light Type id as a constant defines in Light.LIGHTTYPEID_x\n */\n getTypeID() {\n return Light.LIGHTTYPEID_SPOTLIGHT;\n }\n /**\n * Overrides the direction setter to recompute the projection texture view light Matrix.\n * @param value\n */\n _setDirection(value) {\n super._setDirection(value);\n this._projectionTextureViewLightDirty = true;\n }\n /**\n * Overrides the position setter to recompute the projection texture view light Matrix.\n * @param value\n */\n _setPosition(value) {\n super._setPosition(value);\n this._projectionTextureViewLightDirty = true;\n }\n /**\n * Sets the passed matrix \"matrix\" as perspective projection matrix for the shadows and the passed view matrix with the fov equal to the SpotLight angle and and aspect ratio of 1.0.\n * Returns the SpotLight.\n * @param matrix\n * @param viewMatrix\n * @param renderList\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _setDefaultShadowProjectionMatrix(matrix, viewMatrix, renderList) {\n const activeCamera = this.getScene().activeCamera;\n if (!activeCamera) {\n return;\n }\n this._shadowAngleScale = this._shadowAngleScale || 1;\n const angle = this._shadowAngleScale * this._angle;\n const minZ = this.shadowMinZ !== undefined ? this.shadowMinZ : activeCamera.minZ;\n const maxZ = this.shadowMaxZ !== undefined ? this.shadowMaxZ : activeCamera.maxZ;\n const useReverseDepthBuffer = this.getScene().getEngine().useReverseDepthBuffer;\n Matrix.PerspectiveFovLHToRef(angle, 1.0, useReverseDepthBuffer ? maxZ : minZ, useReverseDepthBuffer ? minZ : maxZ, matrix, true, this._scene.getEngine().isNDCHalfZRange, undefined, useReverseDepthBuffer);\n }\n _computeProjectionTextureViewLightMatrix() {\n this._projectionTextureViewLightDirty = false;\n this._projectionTextureDirty = true;\n this.getAbsolutePosition().addToRef(this.direction, this._projectionTextureViewTargetVector);\n Matrix.LookAtLHToRef(this.getAbsolutePosition(), this._projectionTextureViewTargetVector, this._projectionTextureUpDirection, this._projectionTextureViewLightMatrix);\n }\n _computeProjectionTextureProjectionLightMatrix() {\n this._projectionTextureProjectionLightDirty = false;\n this._projectionTextureDirty = true;\n const lightFar = this.projectionTextureLightFar;\n const lightNear = this.projectionTextureLightNear;\n const P = lightFar / (lightFar - lightNear);\n const Q = -P * lightNear;\n const S = 1.0 / Math.tan(this._angle / 2.0);\n const A = 1.0;\n Matrix.FromValuesToRef(S / A, 0.0, 0.0, 0.0, 0.0, S, 0.0, 0.0, 0.0, 0.0, P, 1.0, 0.0, 0.0, Q, 0.0, this._projectionTextureProjectionLightMatrix);\n }\n /**\n * Main function for light texture projection matrix computing.\n */\n _computeProjectionTextureMatrix() {\n this._projectionTextureDirty = false;\n this._projectionTextureViewLightMatrix.multiplyToRef(this._projectionTextureProjectionLightMatrix, this._projectionTextureMatrix);\n if (this._projectionTexture instanceof Texture) {\n const u = this._projectionTexture.uScale / 2.0;\n const v = this._projectionTexture.vScale / 2.0;\n Matrix.FromValuesToRef(u, 0.0, 0.0, 0.0, 0.0, v, 0.0, 0.0, 0.0, 0.0, 0.5, 0.0, 0.5, 0.5, 0.5, 1.0, this._projectionTextureScalingMatrix);\n }\n this._projectionTextureMatrix.multiplyToRef(this._projectionTextureScalingMatrix, this._projectionTextureMatrix);\n }\n _buildUniformLayout() {\n this._uniformBuffer.addUniform(\"vLightData\", 4);\n this._uniformBuffer.addUniform(\"vLightDiffuse\", 4);\n this._uniformBuffer.addUniform(\"vLightSpecular\", 4);\n this._uniformBuffer.addUniform(\"vLightDirection\", 3);\n this._uniformBuffer.addUniform(\"vLightFalloff\", 4);\n this._uniformBuffer.addUniform(\"shadowsInfo\", 3);\n this._uniformBuffer.addUniform(\"depthValues\", 2);\n this._uniformBuffer.create();\n }\n _computeAngleValues() {\n this._lightAngleScale = 1.0 / Math.max(0.001, Math.cos(this._innerAngle * 0.5) - this._cosHalfAngle);\n this._lightAngleOffset = -this._cosHalfAngle * this._lightAngleScale;\n }\n /**\n * Sets the passed Effect \"effect\" with the Light textures.\n * @param effect The effect to update\n * @param lightIndex The index of the light in the effect to update\n * @returns The light\n */\n transferTexturesToEffect(effect, lightIndex) {\n if (this.projectionTexture && this.projectionTexture.isReady()) {\n if (this._projectionTextureViewLightDirty) {\n this._computeProjectionTextureViewLightMatrix();\n }\n if (this._projectionTextureProjectionLightDirty) {\n this._computeProjectionTextureProjectionLightMatrix();\n }\n if (this._projectionTextureDirty) {\n this._computeProjectionTextureMatrix();\n }\n effect.setMatrix(\"textureProjectionMatrix\" + lightIndex, this._projectionTextureMatrix);\n effect.setTexture(\"projectionLightSampler\" + lightIndex, this.projectionTexture);\n }\n return this;\n }\n /**\n * Sets the passed Effect object with the SpotLight transformed position (or position if not parented) and normalized direction.\n * @param effect The effect to update\n * @param lightIndex The index of the light in the effect to update\n * @returns The spot light\n */\n transferToEffect(effect, lightIndex) {\n let normalizeDirection;\n if (this.computeTransformedInformation()) {\n this._uniformBuffer.updateFloat4(\"vLightData\", this.transformedPosition.x, this.transformedPosition.y, this.transformedPosition.z, this.exponent, lightIndex);\n normalizeDirection = Vector3.Normalize(this.transformedDirection);\n }\n else {\n this._uniformBuffer.updateFloat4(\"vLightData\", this.position.x, this.position.y, this.position.z, this.exponent, lightIndex);\n normalizeDirection = Vector3.Normalize(this.direction);\n }\n this._uniformBuffer.updateFloat4(\"vLightDirection\", normalizeDirection.x, normalizeDirection.y, normalizeDirection.z, this._cosHalfAngle, lightIndex);\n this._uniformBuffer.updateFloat4(\"vLightFalloff\", this.range, this._inverseSquaredRange, this._lightAngleScale, this._lightAngleOffset, lightIndex);\n return this;\n }\n transferToNodeMaterialEffect(effect, lightDataUniformName) {\n let normalizeDirection;\n if (this.computeTransformedInformation()) {\n normalizeDirection = Vector3.Normalize(this.transformedDirection);\n }\n else {\n normalizeDirection = Vector3.Normalize(this.direction);\n }\n if (this.getScene().useRightHandedSystem) {\n effect.setFloat3(lightDataUniformName, -normalizeDirection.x, -normalizeDirection.y, -normalizeDirection.z);\n }\n else {\n effect.setFloat3(lightDataUniformName, normalizeDirection.x, normalizeDirection.y, normalizeDirection.z);\n }\n return this;\n }\n /**\n * Disposes the light and the associated resources.\n */\n dispose() {\n super.dispose();\n if (this._projectionTexture) {\n this._projectionTexture.dispose();\n }\n }\n /**\n * Gets the minZ used for shadow according to both the scene and the light.\n * @param activeCamera The camera we are returning the min for\n * @returns the depth min z\n */\n getDepthMinZ(activeCamera) {\n const engine = this._scene.getEngine();\n const minZ = this.shadowMinZ !== undefined ? this.shadowMinZ : activeCamera.minZ;\n return engine.useReverseDepthBuffer && engine.isNDCHalfZRange ? minZ : this._scene.getEngine().isNDCHalfZRange ? 0 : minZ;\n }\n /**\n * Gets the maxZ used for shadow according to both the scene and the light.\n * @param activeCamera The camera we are returning the max for\n * @returns the depth max z\n */\n getDepthMaxZ(activeCamera) {\n const engine = this._scene.getEngine();\n const maxZ = this.shadowMaxZ !== undefined ? this.shadowMaxZ : activeCamera.maxZ;\n return engine.useReverseDepthBuffer && engine.isNDCHalfZRange ? 0 : maxZ;\n }\n /**\n * Prepares the list of defines specific to the light type.\n * @param defines the list of defines\n * @param lightIndex defines the index of the light for the effect\n */\n prepareLightSpecificDefines(defines, lightIndex) {\n defines[\"SPOTLIGHT\" + lightIndex] = true;\n defines[\"PROJECTEDLIGHTTEXTURE\" + lightIndex] = this.projectionTexture && this.projectionTexture.isReady() ? true : false;\n }\n }\n __decorate$1([\n serialize()\n ], SpotLight.prototype, \"angle\", null);\n __decorate$1([\n serialize()\n ], SpotLight.prototype, \"innerAngle\", null);\n __decorate$1([\n serialize()\n ], SpotLight.prototype, \"shadowAngleScale\", null);\n __decorate$1([\n serialize()\n ], SpotLight.prototype, \"exponent\", void 0);\n __decorate$1([\n serialize()\n ], SpotLight.prototype, \"projectionTextureLightNear\", null);\n __decorate$1([\n serialize()\n ], SpotLight.prototype, \"projectionTextureLightFar\", null);\n __decorate$1([\n serialize()\n ], SpotLight.prototype, \"projectionTextureUpDirection\", null);\n __decorate$1([\n serializeAsTexture(\"projectedLightTexture\")\n ], SpotLight.prototype, \"_projectionTexture\", void 0);\n\n /**\n * Defines the root class used to create scene optimization to use with SceneOptimizer\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\n */\n class SceneOptimization {\n /**\n * Gets a string describing the action executed by the current optimization\n * @returns description string\n */\n getDescription() {\n return \"\";\n }\n /**\n * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization\n * @param scene defines the current scene where to apply this optimization\n * @param optimizer defines the current optimizer\n * @returns true if everything that can be done was applied\n */\n apply(scene, optimizer) {\n return true;\n }\n /**\n * Creates the SceneOptimization object\n * @param priority defines the priority of this optimization (0 by default which means first in the list)\n */\n constructor(\n /**\n * Defines the priority of this optimization (0 by default which means first in the list)\n */\n priority = 0) {\n this.priority = priority;\n }\n }\n /**\n * Defines an optimization used to reduce the size of render target textures\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\n */\n class TextureOptimization extends SceneOptimization {\n /**\n * Gets a string describing the action executed by the current optimization\n * @returns description string\n */\n getDescription() {\n return \"Reducing render target texture size to \" + this.maximumSize;\n }\n /**\n * Creates the TextureOptimization object\n * @param priority defines the priority of this optimization (0 by default which means first in the list)\n * @param maximumSize defines the maximum sized allowed for textures (1024 is the default value). If a texture is bigger, it will be scaled down using a factor defined by the step parameter\n * @param step defines the factor (0.5 by default) used to scale down textures bigger than maximum sized allowed.\n */\n constructor(\n /**\n * Defines the priority of this optimization (0 by default which means first in the list)\n */\n priority = 0, \n /**\n * Defines the maximum sized allowed for textures (1024 is the default value). If a texture is bigger, it will be scaled down using a factor defined by the step parameter\n */\n maximumSize = 1024, \n /**\n * Defines the factor (0.5 by default) used to scale down textures bigger than maximum sized allowed.\n */\n step = 0.5) {\n super(priority);\n this.priority = priority;\n this.maximumSize = maximumSize;\n this.step = step;\n }\n /**\n * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization\n * @param scene defines the current scene where to apply this optimization\n * @param optimizer defines the current optimizer\n * @returns true if everything that can be done was applied\n */\n apply(scene, optimizer) {\n let allDone = true;\n for (let index = 0; index < scene.textures.length; index++) {\n const texture = scene.textures[index];\n if (!texture.canRescale || texture.getContext) {\n continue;\n }\n const currentSize = texture.getSize();\n const maxDimension = Math.max(currentSize.width, currentSize.height);\n if (maxDimension > this.maximumSize) {\n texture.scale(this.step);\n allDone = false;\n }\n }\n return allDone;\n }\n }\n /**\n * Defines an optimization used to increase or decrease the rendering resolution\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\n */\n class HardwareScalingOptimization extends SceneOptimization {\n /**\n * Gets a string describing the action executed by the current optimization\n * @returns description string\n */\n getDescription() {\n return \"Setting hardware scaling level to \" + this._currentScale;\n }\n /**\n * Creates the HardwareScalingOptimization object\n * @param priority defines the priority of this optimization (0 by default which means first in the list)\n * @param maximumScale defines the maximum scale to use (2 by default)\n * @param step defines the step to use between two passes (0.5 by default)\n */\n constructor(\n /**\n * Defines the priority of this optimization (0 by default which means first in the list)\n */\n priority = 0, \n /**\n * Defines the maximum scale to use (2 by default)\n */\n maximumScale = 2, \n /**\n * Defines the step to use between two passes (0.5 by default)\n */\n step = 0.25) {\n super(priority);\n this.priority = priority;\n this.maximumScale = maximumScale;\n this.step = step;\n this._currentScale = -1;\n this._directionOffset = 1;\n }\n /**\n * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization\n * @param scene defines the current scene where to apply this optimization\n * @param optimizer defines the current optimizer\n * @returns true if everything that can be done was applied\n */\n apply(scene, optimizer) {\n if (this._currentScale === -1) {\n this._currentScale = scene.getEngine().getHardwareScalingLevel();\n if (this._currentScale > this.maximumScale) {\n this._directionOffset = -1;\n }\n }\n this._currentScale += this._directionOffset * this.step;\n scene.getEngine().setHardwareScalingLevel(this._currentScale);\n return this._directionOffset === 1 ? this._currentScale >= this.maximumScale : this._currentScale <= this.maximumScale;\n }\n }\n /**\n * Defines an optimization used to remove shadows\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\n */\n class ShadowsOptimization extends SceneOptimization {\n /**\n * Gets a string describing the action executed by the current optimization\n * @returns description string\n */\n getDescription() {\n return \"Turning shadows on/off\";\n }\n /**\n * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization\n * @param scene defines the current scene where to apply this optimization\n * @param optimizer defines the current optimizer\n * @returns true if everything that can be done was applied\n */\n apply(scene, optimizer) {\n scene.shadowsEnabled = optimizer.isInImprovementMode;\n return true;\n }\n }\n /**\n * Defines an optimization used to turn post-processes off\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\n */\n class PostProcessesOptimization extends SceneOptimization {\n /**\n * Gets a string describing the action executed by the current optimization\n * @returns description string\n */\n getDescription() {\n return \"Turning post-processes on/off\";\n }\n /**\n * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization\n * @param scene defines the current scene where to apply this optimization\n * @param optimizer defines the current optimizer\n * @returns true if everything that can be done was applied\n */\n apply(scene, optimizer) {\n scene.postProcessesEnabled = optimizer.isInImprovementMode;\n return true;\n }\n }\n /**\n * Defines an optimization used to turn lens flares off\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\n */\n class LensFlaresOptimization extends SceneOptimization {\n /**\n * Gets a string describing the action executed by the current optimization\n * @returns description string\n */\n getDescription() {\n return \"Turning lens flares on/off\";\n }\n /**\n * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization\n * @param scene defines the current scene where to apply this optimization\n * @param optimizer defines the current optimizer\n * @returns true if everything that can be done was applied\n */\n apply(scene, optimizer) {\n scene.lensFlaresEnabled = optimizer.isInImprovementMode;\n return true;\n }\n }\n /**\n * Defines an optimization based on user defined callback.\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\n */\n class CustomOptimization extends SceneOptimization {\n /**\n * Gets a string describing the action executed by the current optimization\n * @returns description string\n */\n getDescription() {\n if (this.onGetDescription) {\n return this.onGetDescription();\n }\n return \"Running user defined callback\";\n }\n /**\n * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization\n * @param scene defines the current scene where to apply this optimization\n * @param optimizer defines the current optimizer\n * @returns true if everything that can be done was applied\n */\n apply(scene, optimizer) {\n if (this.onApply) {\n return this.onApply(scene, optimizer);\n }\n return true;\n }\n }\n /**\n * Defines an optimization used to turn particles off\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\n */\n class ParticlesOptimization extends SceneOptimization {\n /**\n * Gets a string describing the action executed by the current optimization\n * @returns description string\n */\n getDescription() {\n return \"Turning particles on/off\";\n }\n /**\n * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization\n * @param scene defines the current scene where to apply this optimization\n * @param optimizer defines the current optimizer\n * @returns true if everything that can be done was applied\n */\n apply(scene, optimizer) {\n scene.particlesEnabled = optimizer.isInImprovementMode;\n return true;\n }\n }\n /**\n * Defines an optimization used to turn render targets off\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\n */\n class RenderTargetsOptimization extends SceneOptimization {\n /**\n * Gets a string describing the action executed by the current optimization\n * @returns description string\n */\n getDescription() {\n return \"Turning render targets off\";\n }\n /**\n * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization\n * @param scene defines the current scene where to apply this optimization\n * @param optimizer defines the current optimizer\n * @returns true if everything that can be done was applied\n */\n apply(scene, optimizer) {\n scene.renderTargetsEnabled = optimizer.isInImprovementMode;\n return true;\n }\n }\n /**\n * Defines an optimization used to merge meshes with compatible materials\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\n */\n class MergeMeshesOptimization extends SceneOptimization {\n constructor() {\n super(...arguments);\n this._canBeMerged = (abstractMesh) => {\n if (!(abstractMesh instanceof Mesh)) {\n return false;\n }\n const mesh = abstractMesh;\n if (mesh.isDisposed()) {\n return false;\n }\n if (!mesh.isVisible || !mesh.isEnabled()) {\n return false;\n }\n if (mesh.instances.length > 0) {\n return false;\n }\n if (mesh.skeleton || mesh.hasLODLevels) {\n return false;\n }\n if (mesh.getTotalVertices() === 0) {\n return false;\n }\n return true;\n };\n }\n /**\n * Gets or sets a boolean which defines if optimization octree has to be updated\n */\n static get UpdateSelectionTree() {\n return MergeMeshesOptimization._UpdateSelectionTree;\n }\n /**\n * Gets or sets a boolean which defines if optimization octree has to be updated\n */\n static set UpdateSelectionTree(value) {\n MergeMeshesOptimization._UpdateSelectionTree = value;\n }\n /**\n * Gets a string describing the action executed by the current optimization\n * @returns description string\n */\n getDescription() {\n return \"Merging similar meshes together\";\n }\n /**\n * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization\n * @param scene defines the current scene where to apply this optimization\n * @param optimizer defines the current optimizer\n * @param updateSelectionTree defines that the selection octree has to be updated (false by default)\n * @returns true if everything that can be done was applied\n */\n apply(scene, optimizer, updateSelectionTree) {\n const globalPool = scene.meshes.slice(0);\n let globalLength = globalPool.length;\n for (let index = 0; index < globalLength; index++) {\n const currentPool = new Array();\n const current = globalPool[index];\n // Checks\n if (!this._canBeMerged(current)) {\n continue;\n }\n currentPool.push(current);\n // Find compatible meshes\n for (let subIndex = index + 1; subIndex < globalLength; subIndex++) {\n const otherMesh = globalPool[subIndex];\n if (!this._canBeMerged(otherMesh)) {\n continue;\n }\n if (otherMesh.material !== current.material) {\n continue;\n }\n if (otherMesh.checkCollisions !== current.checkCollisions) {\n continue;\n }\n currentPool.push(otherMesh);\n globalLength--;\n globalPool.splice(subIndex, 1);\n subIndex--;\n }\n if (currentPool.length < 2) {\n continue;\n }\n // Merge meshes\n Mesh.MergeMeshes(currentPool, undefined, true);\n }\n // Call the octree system optimization if it is defined.\n const sceneAsAny = scene;\n if (sceneAsAny.createOrUpdateSelectionOctree) {\n if (updateSelectionTree != undefined) {\n if (updateSelectionTree) {\n sceneAsAny.createOrUpdateSelectionOctree();\n }\n }\n else if (MergeMeshesOptimization.UpdateSelectionTree) {\n sceneAsAny.createOrUpdateSelectionOctree();\n }\n }\n return true;\n }\n }\n MergeMeshesOptimization._UpdateSelectionTree = false;\n /**\n * Defines a list of options used by SceneOptimizer\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\n */\n class SceneOptimizerOptions {\n /**\n * Creates a new list of options used by SceneOptimizer\n * @param targetFrameRate defines the target frame rate to reach (60 by default)\n * @param trackerDuration defines the interval between two checks (2000ms by default)\n */\n constructor(\n /**\n * Defines the target frame rate to reach (60 by default)\n */\n targetFrameRate = 60, \n /**\n * Defines the interval between two checks (2000ms by default)\n */\n trackerDuration = 2000) {\n this.targetFrameRate = targetFrameRate;\n this.trackerDuration = trackerDuration;\n /**\n * Gets the list of optimizations to apply\n */\n this.optimizations = new Array();\n }\n /**\n * Add a new optimization\n * @param optimization defines the SceneOptimization to add to the list of active optimizations\n * @returns the current SceneOptimizerOptions\n */\n addOptimization(optimization) {\n this.optimizations.push(optimization);\n return this;\n }\n /**\n * Add a new custom optimization\n * @param onApply defines the callback called to apply the custom optimization (true if everything that can be done was applied)\n * @param onGetDescription defines the callback called to get the description attached with the optimization.\n * @param priority defines the priority of this optimization (0 by default which means first in the list)\n * @returns the current SceneOptimizerOptions\n */\n addCustomOptimization(onApply, onGetDescription, priority = 0) {\n const optimization = new CustomOptimization(priority);\n optimization.onApply = onApply;\n optimization.onGetDescription = onGetDescription;\n this.optimizations.push(optimization);\n return this;\n }\n /**\n * Creates a list of pre-defined optimizations aimed to reduce the visual impact on the scene\n * @param targetFrameRate defines the target frame rate (60 by default)\n * @returns a SceneOptimizerOptions object\n */\n static LowDegradationAllowed(targetFrameRate) {\n const result = new SceneOptimizerOptions(targetFrameRate);\n let priority = 0;\n result.addOptimization(new MergeMeshesOptimization(priority));\n result.addOptimization(new ShadowsOptimization(priority));\n result.addOptimization(new LensFlaresOptimization(priority));\n // Next priority\n priority++;\n result.addOptimization(new PostProcessesOptimization(priority));\n result.addOptimization(new ParticlesOptimization(priority));\n // Next priority\n priority++;\n result.addOptimization(new TextureOptimization(priority, 1024));\n return result;\n }\n /**\n * Creates a list of pre-defined optimizations aimed to have a moderate impact on the scene visual\n * @param targetFrameRate defines the target frame rate (60 by default)\n * @returns a SceneOptimizerOptions object\n */\n static ModerateDegradationAllowed(targetFrameRate) {\n const result = new SceneOptimizerOptions(targetFrameRate);\n let priority = 0;\n result.addOptimization(new MergeMeshesOptimization(priority));\n result.addOptimization(new ShadowsOptimization(priority));\n result.addOptimization(new LensFlaresOptimization(priority));\n // Next priority\n priority++;\n result.addOptimization(new PostProcessesOptimization(priority));\n result.addOptimization(new ParticlesOptimization(priority));\n // Next priority\n priority++;\n result.addOptimization(new TextureOptimization(priority, 512));\n // Next priority\n priority++;\n result.addOptimization(new RenderTargetsOptimization(priority));\n // Next priority\n priority++;\n result.addOptimization(new HardwareScalingOptimization(priority, 2));\n return result;\n }\n /**\n * Creates a list of pre-defined optimizations aimed to have a big impact on the scene visual\n * @param targetFrameRate defines the target frame rate (60 by default)\n * @returns a SceneOptimizerOptions object\n */\n static HighDegradationAllowed(targetFrameRate) {\n const result = new SceneOptimizerOptions(targetFrameRate);\n let priority = 0;\n result.addOptimization(new MergeMeshesOptimization(priority));\n result.addOptimization(new ShadowsOptimization(priority));\n result.addOptimization(new LensFlaresOptimization(priority));\n // Next priority\n priority++;\n result.addOptimization(new PostProcessesOptimization(priority));\n result.addOptimization(new ParticlesOptimization(priority));\n // Next priority\n priority++;\n result.addOptimization(new TextureOptimization(priority, 256));\n // Next priority\n priority++;\n result.addOptimization(new RenderTargetsOptimization(priority));\n // Next priority\n priority++;\n result.addOptimization(new HardwareScalingOptimization(priority, 4));\n return result;\n }\n }\n /**\n * Class used to run optimizations in order to reach a target frame rate\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\n */\n class SceneOptimizer {\n /**\n * Gets or sets a boolean indicating if the optimizer is in improvement mode\n */\n get isInImprovementMode() {\n return this._improvementMode;\n }\n set isInImprovementMode(value) {\n this._improvementMode = value;\n }\n /**\n * Gets the current priority level (0 at start)\n */\n get currentPriorityLevel() {\n return this._currentPriorityLevel;\n }\n /**\n * Gets the current frame rate checked by the SceneOptimizer\n */\n get currentFrameRate() {\n return this._currentFrameRate;\n }\n /**\n * Gets or sets the current target frame rate (60 by default)\n */\n get targetFrameRate() {\n return this._targetFrameRate;\n }\n /**\n * Gets or sets the current target frame rate (60 by default)\n */\n set targetFrameRate(value) {\n this._targetFrameRate = value;\n }\n /**\n * Gets or sets the current interval between two checks (every 2000ms by default)\n */\n get trackerDuration() {\n return this._trackerDuration;\n }\n /**\n * Gets or sets the current interval between two checks (every 2000ms by default)\n */\n set trackerDuration(value) {\n this._trackerDuration = value;\n }\n /**\n * Gets the list of active optimizations\n */\n get optimizations() {\n return this._options.optimizations;\n }\n /**\n * Creates a new SceneOptimizer\n * @param scene defines the scene to work on\n * @param options defines the options to use with the SceneOptimizer\n * @param autoGeneratePriorities defines if priorities must be generated and not read from SceneOptimization property (true by default)\n * @param improvementMode defines if the scene optimizer must run the maximum optimization while staying over a target frame instead of trying to reach the target framerate (false by default)\n */\n constructor(scene, options, autoGeneratePriorities = true, improvementMode = false) {\n this._isRunning = false;\n this._currentPriorityLevel = 0;\n this._targetFrameRate = 60;\n this._trackerDuration = 2000;\n this._currentFrameRate = 0;\n this._improvementMode = false;\n /**\n * Defines an observable called when the optimizer reaches the target frame rate\n */\n this.onSuccessObservable = new Observable$1();\n /**\n * Defines an observable called when the optimizer enables an optimization\n */\n this.onNewOptimizationAppliedObservable = new Observable$1();\n /**\n * Defines an observable called when the optimizer is not able to reach the target frame rate\n */\n this.onFailureObservable = new Observable$1();\n if (!options) {\n this._options = new SceneOptimizerOptions();\n }\n else {\n this._options = options;\n }\n if (this._options.targetFrameRate) {\n this._targetFrameRate = this._options.targetFrameRate;\n }\n if (this._options.trackerDuration) {\n this._trackerDuration = this._options.trackerDuration;\n }\n if (autoGeneratePriorities) {\n let priority = 0;\n for (const optim of this._options.optimizations) {\n optim.priority = priority++;\n }\n }\n this._improvementMode = improvementMode;\n this._scene = scene || EngineStore.LastCreatedScene;\n this._sceneDisposeObserver = this._scene.onDisposeObservable.add(() => {\n this._sceneDisposeObserver = null;\n this.dispose();\n });\n }\n /**\n * Stops the current optimizer\n */\n stop() {\n this._isRunning = false;\n }\n /**\n * Reset the optimizer to initial step (current priority level = 0)\n */\n reset() {\n this._currentPriorityLevel = 0;\n }\n /**\n * Start the optimizer. By default it will try to reach a specific framerate\n * but if the optimizer is set with improvementMode === true then it will run all optimization while frame rate is above the target frame rate\n */\n start() {\n if (this._isRunning) {\n return;\n }\n this._isRunning = true;\n // Let's wait for the scene to be ready before running our check\n this._scene.executeWhenReady(() => {\n setTimeout(() => {\n this._checkCurrentState();\n }, this._trackerDuration);\n });\n }\n _checkCurrentState() {\n if (!this._isRunning) {\n return;\n }\n const scene = this._scene;\n const options = this._options;\n this._currentFrameRate = Math.round(scene.getEngine().getFps());\n if ((this._improvementMode && this._currentFrameRate <= this._targetFrameRate) || (!this._improvementMode && this._currentFrameRate >= this._targetFrameRate)) {\n this._isRunning = false;\n this.onSuccessObservable.notifyObservers(this);\n return;\n }\n // Apply current level of optimizations\n let allDone = true;\n let noOptimizationApplied = true;\n for (let index = 0; index < options.optimizations.length; index++) {\n const optimization = options.optimizations[index];\n if (optimization.priority === this._currentPriorityLevel) {\n noOptimizationApplied = false;\n allDone = allDone && optimization.apply(scene, this);\n this.onNewOptimizationAppliedObservable.notifyObservers(optimization);\n }\n }\n // If no optimization was applied, this is a failure :(\n if (noOptimizationApplied) {\n this._isRunning = false;\n this.onFailureObservable.notifyObservers(this);\n return;\n }\n // If all optimizations were done, move to next level\n if (allDone) {\n this._currentPriorityLevel++;\n }\n // Let's the system running for a specific amount of time before checking FPS\n scene.executeWhenReady(() => {\n setTimeout(() => {\n this._checkCurrentState();\n }, this._trackerDuration);\n });\n }\n /**\n * Release all resources\n */\n dispose() {\n this.stop();\n this.onSuccessObservable.clear();\n this.onFailureObservable.clear();\n this.onNewOptimizationAppliedObservable.clear();\n if (this._sceneDisposeObserver) {\n this._scene.onDisposeObservable.remove(this._sceneDisposeObserver);\n }\n }\n /**\n * Helper function to create a SceneOptimizer with one single line of code\n * @param scene defines the scene to work on\n * @param options defines the options to use with the SceneOptimizer\n * @param onSuccess defines a callback to call on success\n * @param onFailure defines a callback to call on failure\n * @returns the new SceneOptimizer object\n */\n static OptimizeAsync(scene, options, onSuccess, onFailure) {\n const optimizer = new SceneOptimizer(scene, options || SceneOptimizerOptions.ModerateDegradationAllowed(), false);\n if (onSuccess) {\n optimizer.onSuccessObservable.add(() => {\n onSuccess();\n });\n }\n if (onFailure) {\n optimizer.onFailureObservable.add(() => {\n onFailure();\n });\n }\n optimizer.start();\n return optimizer;\n }\n }\n\n let serializedGeometries = [];\n const SerializeGeometry = (geometry, serializationGeometries) => {\n if (geometry.doNotSerialize) {\n return;\n }\n serializationGeometries.vertexData.push(geometry.serializeVerticeData());\n serializedGeometries[geometry.id] = true;\n };\n const SerializeMesh = (mesh, serializationScene) => {\n const serializationObject = {};\n // Geometry\n const geometry = mesh._geometry;\n if (geometry) {\n if (!mesh.getScene().getGeometryById(geometry.id)) {\n // Geometry was in the memory but not added to the scene, nevertheless it's better to serialize to be able to reload the mesh with its geometry\n SerializeGeometry(geometry, serializationScene.geometries);\n }\n }\n // Custom\n if (mesh.serialize) {\n mesh.serialize(serializationObject);\n }\n return serializationObject;\n };\n const FinalizeSingleNode = (node, serializationObject) => {\n if (node._isMesh) {\n const mesh = node;\n //only works if the mesh is already loaded\n if (mesh.delayLoadState === 1 || mesh.delayLoadState === 0) {\n const serializeMaterial = (material) => {\n serializationObject.materials = serializationObject.materials || [];\n if (mesh.material && !serializationObject.materials.some((mat) => mat.id === mesh.material.id)) {\n serializationObject.materials.push(material.serialize());\n }\n };\n //serialize material\n if (mesh.material && !mesh.material.doNotSerialize) {\n if (mesh.material instanceof MultiMaterial) {\n serializationObject.multiMaterials = serializationObject.multiMaterials || [];\n if (!serializationObject.multiMaterials.some((mat) => mat.id === mesh.material.id)) {\n serializationObject.multiMaterials.push(mesh.material.serialize());\n for (const submaterial of mesh.material.subMaterials) {\n if (submaterial) {\n serializeMaterial(submaterial);\n }\n }\n }\n }\n else {\n serializeMaterial(mesh.material);\n }\n }\n else if (!mesh.material) {\n serializeMaterial(mesh.getScene().defaultMaterial);\n }\n //serialize geometry\n const geometry = mesh._geometry;\n if (geometry) {\n if (!serializationObject.geometries) {\n serializationObject.geometries = {};\n serializationObject.geometries.boxes = [];\n serializationObject.geometries.spheres = [];\n serializationObject.geometries.cylinders = [];\n serializationObject.geometries.toruses = [];\n serializationObject.geometries.grounds = [];\n serializationObject.geometries.planes = [];\n serializationObject.geometries.torusKnots = [];\n serializationObject.geometries.vertexData = [];\n }\n SerializeGeometry(geometry, serializationObject.geometries);\n }\n // Skeletons\n if (mesh.skeleton && !mesh.skeleton.doNotSerialize) {\n serializationObject.skeletons = serializationObject.skeletons || [];\n serializationObject.skeletons.push(mesh.skeleton.serialize());\n }\n //serialize the actual mesh\n serializationObject.meshes = serializationObject.meshes || [];\n serializationObject.meshes.push(SerializeMesh(mesh, serializationObject));\n }\n }\n else if (node.getClassName() === \"TransformNode\") {\n const transformNode = node;\n serializationObject.transformNodes.push(transformNode.serialize());\n }\n else if (node.getClassName().indexOf(\"Camera\") !== -1) {\n const camera = node;\n serializationObject.cameras.push(camera.serialize());\n }\n else if (node.getClassName().indexOf(\"Light\") !== -1) {\n const light = node;\n serializationObject.lights.push(light.serialize());\n }\n };\n /**\n * Class used to serialize a scene into a string\n */\n class SceneSerializer {\n /**\n * Clear cache used by a previous serialization\n */\n static ClearCache() {\n serializedGeometries = [];\n }\n /**\n * Serialize a scene into a JSON compatible object\n * Note that if the current engine does not support synchronous texture reading (like WebGPU), you should use SerializeAsync instead\n * as else you may not retrieve the proper base64 encoded texture data (when using the Texture.ForceSerializeBuffers flag)\n * @param scene defines the scene to serialize\n * @returns a JSON compatible object\n */\n static Serialize(scene) {\n return SceneSerializer._Serialize(scene);\n }\n static _Serialize(scene, checkSyncReadSupported = true) {\n const serializationObject = {};\n if (checkSyncReadSupported && !scene.getEngine()._features.supportSyncTextureRead && Texture.ForceSerializeBuffers) {\n console.warn(\"The serialization object may not contain the proper base64 encoded texture data! You should use the SerializeAsync method instead.\");\n }\n SceneSerializer.ClearCache();\n // Scene\n serializationObject.useDelayedTextureLoading = scene.useDelayedTextureLoading;\n serializationObject.autoClear = scene.autoClear;\n serializationObject.clearColor = scene.clearColor.asArray();\n serializationObject.ambientColor = scene.ambientColor.asArray();\n serializationObject.gravity = scene.gravity.asArray();\n serializationObject.collisionsEnabled = scene.collisionsEnabled;\n serializationObject.useRightHandedSystem = scene.useRightHandedSystem;\n // Fog\n if (scene.fogMode && scene.fogMode !== 0) {\n serializationObject.fogMode = scene.fogMode;\n serializationObject.fogColor = scene.fogColor.asArray();\n serializationObject.fogStart = scene.fogStart;\n serializationObject.fogEnd = scene.fogEnd;\n serializationObject.fogDensity = scene.fogDensity;\n }\n //Physics\n if (scene.isPhysicsEnabled && scene.isPhysicsEnabled()) {\n const physicEngine = scene.getPhysicsEngine();\n if (physicEngine) {\n serializationObject.physicsEnabled = true;\n serializationObject.physicsGravity = physicEngine.gravity.asArray();\n serializationObject.physicsEngine = physicEngine.getPhysicsPluginName();\n }\n }\n // Metadata\n if (scene.metadata) {\n serializationObject.metadata = scene.metadata;\n }\n // Morph targets\n serializationObject.morphTargetManagers = [];\n for (const abstractMesh of scene.meshes) {\n const manager = abstractMesh.morphTargetManager;\n if (manager) {\n serializationObject.morphTargetManagers.push(manager.serialize());\n }\n }\n // Lights\n serializationObject.lights = [];\n let index;\n let light;\n for (index = 0; index < scene.lights.length; index++) {\n light = scene.lights[index];\n if (!light.doNotSerialize) {\n serializationObject.lights.push(light.serialize());\n }\n }\n // Cameras\n serializationObject.cameras = [];\n for (index = 0; index < scene.cameras.length; index++) {\n const camera = scene.cameras[index];\n if (!camera.doNotSerialize) {\n serializationObject.cameras.push(camera.serialize());\n }\n }\n if (scene.activeCamera) {\n serializationObject.activeCameraID = scene.activeCamera.id;\n }\n // Animations\n SerializationHelper.AppendSerializedAnimations(scene, serializationObject);\n // Animation Groups\n if (scene.animationGroups && scene.animationGroups.length > 0) {\n serializationObject.animationGroups = [];\n for (let animationGroupIndex = 0; animationGroupIndex < scene.animationGroups.length; animationGroupIndex++) {\n const animationGroup = scene.animationGroups[animationGroupIndex];\n serializationObject.animationGroups.push(animationGroup.serialize());\n }\n }\n // Reflection probes\n if (scene.reflectionProbes && scene.reflectionProbes.length > 0) {\n serializationObject.reflectionProbes = [];\n for (index = 0; index < scene.reflectionProbes.length; index++) {\n const reflectionProbe = scene.reflectionProbes[index];\n serializationObject.reflectionProbes.push(reflectionProbe.serialize());\n }\n }\n // Materials\n serializationObject.materials = [];\n serializationObject.multiMaterials = [];\n let material;\n for (index = 0; index < scene.materials.length; index++) {\n material = scene.materials[index];\n if (!material.doNotSerialize) {\n serializationObject.materials.push(material.serialize());\n }\n }\n // MultiMaterials\n serializationObject.multiMaterials = [];\n for (index = 0; index < scene.multiMaterials.length; index++) {\n const multiMaterial = scene.multiMaterials[index];\n serializationObject.multiMaterials.push(multiMaterial.serialize());\n }\n // Environment texture\n if (scene.environmentTexture) {\n if (scene.environmentTexture._files) {\n serializationObject.environmentTexture = scene.environmentTexture.serialize();\n }\n else {\n serializationObject.environmentTexture = scene.environmentTexture.name;\n serializationObject.environmentTextureRotationY = scene.environmentTexture.rotationY;\n }\n }\n // Environment Intensity\n serializationObject.environmentIntensity = scene.environmentIntensity;\n // Skeletons\n serializationObject.skeletons = [];\n for (index = 0; index < scene.skeletons.length; index++) {\n const skeleton = scene.skeletons[index];\n if (!skeleton.doNotSerialize) {\n serializationObject.skeletons.push(skeleton.serialize());\n }\n }\n // Transform nodes\n serializationObject.transformNodes = [];\n for (index = 0; index < scene.transformNodes.length; index++) {\n if (!scene.transformNodes[index].doNotSerialize) {\n serializationObject.transformNodes.push(scene.transformNodes[index].serialize());\n }\n }\n // Geometries\n serializationObject.geometries = {};\n serializationObject.geometries.boxes = [];\n serializationObject.geometries.spheres = [];\n serializationObject.geometries.cylinders = [];\n serializationObject.geometries.toruses = [];\n serializationObject.geometries.grounds = [];\n serializationObject.geometries.planes = [];\n serializationObject.geometries.torusKnots = [];\n serializationObject.geometries.vertexData = [];\n serializedGeometries = [];\n const geometries = scene.getGeometries();\n for (index = 0; index < geometries.length; index++) {\n const geometry = geometries[index];\n if (geometry.isReady()) {\n SerializeGeometry(geometry, serializationObject.geometries);\n }\n }\n // Meshes\n serializationObject.meshes = [];\n for (index = 0; index < scene.meshes.length; index++) {\n const abstractMesh = scene.meshes[index];\n if (abstractMesh instanceof Mesh) {\n const mesh = abstractMesh;\n if (!mesh.doNotSerialize) {\n if (mesh.delayLoadState === 1 || mesh.delayLoadState === 0) {\n serializationObject.meshes.push(SerializeMesh(mesh, serializationObject));\n }\n }\n }\n }\n // Particles Systems\n serializationObject.particleSystems = [];\n for (index = 0; index < scene.particleSystems.length; index++) {\n serializationObject.particleSystems.push(scene.particleSystems[index].serialize(false));\n }\n // Post processes\n serializationObject.postProcesses = [];\n for (index = 0; index < scene.postProcesses.length; index++) {\n serializationObject.postProcesses.push(scene.postProcesses[index].serialize());\n }\n // Action Manager\n if (scene.actionManager) {\n serializationObject.actions = scene.actionManager.serialize(\"scene\");\n }\n // Components\n for (const component of scene._serializableComponents) {\n component.serialize(serializationObject);\n }\n return serializationObject;\n }\n /**\n * Serialize a scene into a JSON compatible object\n * @param scene defines the scene to serialize\n * @returns a JSON promise compatible object\n */\n static SerializeAsync(scene) {\n const serializationObject = SceneSerializer._Serialize(scene, false);\n const promises = [];\n this._CollectPromises(serializationObject, promises);\n return Promise.all(promises).then(() => serializationObject);\n }\n static _CollectPromises(obj, promises) {\n if (Array.isArray(obj)) {\n for (let i = 0; i < obj.length; ++i) {\n const o = obj[i];\n if (o instanceof Promise) {\n promises.push(o.then((res) => (obj[i] = res)));\n }\n else if (o instanceof Object || Array.isArray(o)) {\n this._CollectPromises(o, promises);\n }\n }\n }\n else if (obj instanceof Object) {\n for (const name in obj) {\n if (Object.prototype.hasOwnProperty.call(obj, name)) {\n const o = obj[name];\n if (o instanceof Promise) {\n promises.push(o.then((res) => (obj[name] = res)));\n }\n else if (o instanceof Object || Array.isArray(o)) {\n this._CollectPromises(o, promises);\n }\n }\n }\n }\n }\n /**\n * Serialize a mesh into a JSON compatible object\n * @param toSerialize defines the mesh to serialize\n * @param withParents defines if parents must be serialized as well\n * @param withChildren defines if children must be serialized as well\n * @returns a JSON compatible object\n */\n static SerializeMesh(toSerialize /* Mesh || Mesh[] */, withParents = false, withChildren = false) {\n const serializationObject = {};\n serializationObject.meshes = [];\n serializationObject.transformNodes = [];\n serializationObject.cameras = [];\n serializationObject.lights = [];\n SceneSerializer.ClearCache();\n toSerialize = toSerialize instanceof Array ? toSerialize : [toSerialize];\n if (withParents || withChildren) {\n //deliberate for loop! not for each, appended should be processed as well.\n for (let i = 0; i < toSerialize.length; ++i) {\n if (withChildren) {\n toSerialize[i].getDescendants().forEach((node) => {\n if (toSerialize.indexOf(node) < 0 && !node.doNotSerialize) {\n toSerialize.push(node);\n }\n });\n }\n //make sure the array doesn't contain the object already\n if (withParents && toSerialize[i].parent && toSerialize.indexOf(toSerialize[i].parent) < 0 && !toSerialize[i].parent.doNotSerialize) {\n toSerialize.push(toSerialize[i].parent);\n }\n }\n }\n toSerialize.forEach((mesh) => {\n FinalizeSingleNode(mesh, serializationObject);\n });\n return serializationObject;\n }\n }\n\n Scene.prototype.enableDepthRenderer = function (camera, storeNonLinearDepth = false, force32bitsFloat = false, samplingMode = 3, storeCameraSpaceZ = false) {\n camera = camera || this.activeCamera;\n if (!camera) {\n throw \"No camera available to enable depth renderer\";\n }\n if (!this._depthRenderer) {\n this._depthRenderer = {};\n }\n if (!this._depthRenderer[camera.id]) {\n const supportFullfloat = !!this.getEngine().getCaps().textureFloatRender;\n let textureType = 0;\n if (this.getEngine().getCaps().textureHalfFloatRender && (!force32bitsFloat || !supportFullfloat)) {\n textureType = 2;\n }\n else if (supportFullfloat) {\n textureType = 1;\n }\n else {\n textureType = 0;\n }\n this._depthRenderer[camera.id] = new DepthRenderer(this, textureType, camera, storeNonLinearDepth, samplingMode, storeCameraSpaceZ);\n }\n return this._depthRenderer[camera.id];\n };\n Scene.prototype.disableDepthRenderer = function (camera) {\n camera = camera || this.activeCamera;\n if (!camera || !this._depthRenderer || !this._depthRenderer[camera.id]) {\n return;\n }\n this._depthRenderer[camera.id].dispose();\n };\n /**\n * Defines the Depth Renderer scene component responsible to manage a depth buffer useful\n * in several rendering techniques.\n */\n class DepthRendererSceneComponent {\n /**\n * Creates a new instance of the component for the given scene\n * @param scene Defines the scene to register the component in\n */\n constructor(scene) {\n /**\n * The component name helpful to identify the component in the list of scene components.\n */\n this.name = SceneComponentConstants.NAME_DEPTHRENDERER;\n this.scene = scene;\n }\n /**\n * Registers the component in a given scene\n */\n register() {\n this.scene._gatherRenderTargetsStage.registerStep(SceneComponentConstants.STEP_GATHERRENDERTARGETS_DEPTHRENDERER, this, this._gatherRenderTargets);\n this.scene._gatherActiveCameraRenderTargetsStage.registerStep(SceneComponentConstants.STEP_GATHERACTIVECAMERARENDERTARGETS_DEPTHRENDERER, this, this._gatherActiveCameraRenderTargets);\n }\n /**\n * Rebuilds the elements related to this component in case of\n * context lost for instance.\n */\n rebuild() {\n // Nothing to do for this component\n }\n /**\n * Disposes the component and the associated resources\n */\n dispose() {\n for (const key in this.scene._depthRenderer) {\n this.scene._depthRenderer[key].dispose();\n }\n }\n _gatherRenderTargets(renderTargets) {\n if (this.scene._depthRenderer) {\n for (const key in this.scene._depthRenderer) {\n const depthRenderer = this.scene._depthRenderer[key];\n if (depthRenderer.enabled && !depthRenderer.useOnlyInActiveCamera) {\n renderTargets.push(depthRenderer.getDepthMap());\n }\n }\n }\n }\n _gatherActiveCameraRenderTargets(renderTargets) {\n if (this.scene._depthRenderer) {\n for (const key in this.scene._depthRenderer) {\n const depthRenderer = this.scene._depthRenderer[key];\n if (depthRenderer.enabled && depthRenderer.useOnlyInActiveCamera && this.scene.activeCamera.id === key) {\n renderTargets.push(depthRenderer.getDepthMap());\n }\n }\n }\n }\n }\n DepthRenderer._SceneComponentInitialization = (scene) => {\n // Register the G Buffer component to the scene.\n let component = scene._getComponent(SceneComponentConstants.NAME_DEPTHRENDERER);\n if (!component) {\n component = new DepthRendererSceneComponent(scene);\n scene._addComponent(component);\n }\n };\n\n // Do not edit.\n const name$2h = \"linePixelShader\";\n const shader$2h = `#include\nuniform vec4 color;\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_FRAGMENT_MAIN_BEGIN\n#include\ngl_FragColor=color;\n#define CUSTOM_FRAGMENT_MAIN_END\n}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2h] = shader$2h;\n\n // Do not edit.\n const name$2i = \"lineVertexShader\";\n const shader$2i = `#include\n#include\nattribute vec3 position;attribute vec4 normal;uniform mat4 viewProjection;uniform float width;uniform float aspectRatio;\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\n#include\nmat4 worldViewProjection=viewProjection*finalWorld;vec4 viewPosition=worldViewProjection*vec4(position,1.0);vec4 viewPositionNext=worldViewProjection*vec4(normal.xyz,1.0);vec2 currentScreen=viewPosition.xy/viewPosition.w;vec2 nextScreen=viewPositionNext.xy/viewPositionNext.w;currentScreen.x*=aspectRatio;nextScreen.x*=aspectRatio;vec2 dir=normalize(nextScreen-currentScreen);vec2 normalDir=vec2(-dir.y,dir.x);normalDir*=width/2.0;normalDir.x/=aspectRatio;vec4 offset=vec4(normalDir*normal.w,0.0,0.0);gl_Position=viewPosition+offset;\n#if defined(CLIPPLANE) || defined(CLIPPLANE2) || defined(CLIPPLANE3) || defined(CLIPPLANE4) || defined(CLIPPLANE5) || defined(CLIPPLANE6)\nvec4 worldPos=finalWorld*vec4(position,1.0);\n#include\n#endif\n#define CUSTOM_VERTEX_MAIN_END\n}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2i] = shader$2i;\n\n AbstractMesh.prototype.disableEdgesRendering = function () {\n if (this._edgesRenderer) {\n this._edgesRenderer.dispose();\n this._edgesRenderer = null;\n }\n return this;\n };\n AbstractMesh.prototype.enableEdgesRendering = function (epsilon = 0.95, checkVerticesInsteadOfIndices = false, options) {\n this.disableEdgesRendering();\n this._edgesRenderer = new EdgesRenderer(this, epsilon, checkVerticesInsteadOfIndices, true, options);\n return this;\n };\n Object.defineProperty(AbstractMesh.prototype, \"edgesRenderer\", {\n get: function () {\n return this._edgesRenderer;\n },\n enumerable: true,\n configurable: true,\n });\n LinesMesh.prototype.enableEdgesRendering = function (epsilon = 0.95, checkVerticesInsteadOfIndices = false) {\n this.disableEdgesRendering();\n this._edgesRenderer = new LineEdgesRenderer(this, epsilon, checkVerticesInsteadOfIndices);\n return this;\n };\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n InstancedLinesMesh.prototype.enableEdgesRendering = function (epsilon = 0.95, checkVerticesInsteadOfIndices = false) {\n LinesMesh.prototype.enableEdgesRendering.apply(this, arguments);\n return this;\n };\n /**\n * FaceAdjacencies Helper class to generate edges\n */\n class FaceAdjacencies {\n constructor() {\n this.edges = new Array();\n this.edgesConnectedCount = 0;\n }\n }\n /**\n * This class is used to generate edges of the mesh that could then easily be rendered in a scene.\n */\n class EdgesRenderer {\n /** Gets the vertices generated by the edge renderer */\n get linesPositions() {\n return this._linesPositions;\n }\n /** Gets the normals generated by the edge renderer */\n get linesNormals() {\n return this._linesNormals;\n }\n /** Gets the indices generated by the edge renderer */\n get linesIndices() {\n return this._linesIndices;\n }\n /**\n * Gets or sets the shader used to draw the lines\n */\n get lineShader() {\n return this._lineShader;\n }\n set lineShader(shader) {\n this._lineShader = shader;\n }\n static _GetShader(scene) {\n if (!scene._edgeRenderLineShader) {\n const shader = new ShaderMaterial(\"lineShader\", scene, \"line\", {\n attributes: [\"position\", \"normal\"],\n uniforms: [\"world\", \"viewProjection\", \"color\", \"width\", \"aspectRatio\"],\n }, false);\n shader.disableDepthWrite = true;\n shader.backFaceCulling = false;\n shader.checkReadyOnEveryCall = scene.getEngine().isWebGPU;\n scene._edgeRenderLineShader = shader;\n }\n return scene._edgeRenderLineShader;\n }\n /**\n * Creates an instance of the EdgesRenderer. It is primarily use to display edges of a mesh.\n * Beware when you use this class with complex objects as the adjacencies computation can be really long\n * @param source Mesh used to create edges\n * @param epsilon sum of angles in adjacency to check for edge\n * @param checkVerticesInsteadOfIndices bases the edges detection on vertices vs indices. Note that this parameter is not used if options.useAlternateEdgeFinder = true\n * @param generateEdgesLines - should generate Lines or only prepare resources.\n * @param options The options to apply when generating the edges\n */\n constructor(source, epsilon = 0.95, checkVerticesInsteadOfIndices = false, generateEdgesLines = true, options) {\n var _a;\n /**\n * Define the size of the edges with an orthographic camera\n */\n this.edgesWidthScalerForOrthographic = 1000.0;\n /**\n * Define the size of the edges with a perspective camera\n */\n this.edgesWidthScalerForPerspective = 50.0;\n this._linesPositions = new Array();\n this._linesNormals = new Array();\n this._linesIndices = new Array();\n this._buffers = {};\n this._buffersForInstances = {};\n this._checkVerticesInsteadOfIndices = false;\n /** Gets or sets a boolean indicating if the edgesRenderer is active */\n this.isEnabled = true;\n /**\n * List of instances to render in case the source mesh has instances\n */\n this.customInstances = new SmartArray(32);\n this._source = source;\n this._checkVerticesInsteadOfIndices = checkVerticesInsteadOfIndices;\n this._options = options !== null && options !== void 0 ? options : null;\n this._epsilon = epsilon;\n if (this._source.getScene().getEngine().isWebGPU) {\n this._drawWrapper = new DrawWrapper(source.getEngine());\n }\n this._prepareRessources();\n if (generateEdgesLines) {\n if ((_a = options === null || options === void 0 ? void 0 : options.useAlternateEdgeFinder) !== null && _a !== void 0 ? _a : true) {\n this._generateEdgesLinesAlternate();\n }\n else {\n this._generateEdgesLines();\n }\n }\n this._meshRebuildObserver = this._source.onRebuildObservable.add(() => {\n this._rebuild();\n });\n this._meshDisposeObserver = this._source.onDisposeObservable.add(() => {\n this.dispose();\n });\n }\n _prepareRessources() {\n if (this._lineShader) {\n return;\n }\n this._lineShader = EdgesRenderer._GetShader(this._source.getScene());\n }\n /** @internal */\n _rebuild() {\n let buffer = this._buffers[VertexBuffer.PositionKind];\n if (buffer) {\n buffer._rebuild();\n }\n buffer = this._buffers[VertexBuffer.NormalKind];\n if (buffer) {\n buffer._rebuild();\n }\n const scene = this._source.getScene();\n const engine = scene.getEngine();\n this._ib = engine.createIndexBuffer(this._linesIndices);\n }\n /**\n * Releases the required resources for the edges renderer\n */\n dispose() {\n var _a;\n this._source.onRebuildObservable.remove(this._meshRebuildObserver);\n this._source.onDisposeObservable.remove(this._meshDisposeObserver);\n let buffer = this._buffers[VertexBuffer.PositionKind];\n if (buffer) {\n buffer.dispose();\n this._buffers[VertexBuffer.PositionKind] = null;\n }\n buffer = this._buffers[VertexBuffer.NormalKind];\n if (buffer) {\n buffer.dispose();\n this._buffers[VertexBuffer.NormalKind] = null;\n }\n if (this._ib) {\n this._source.getScene().getEngine()._releaseBuffer(this._ib);\n }\n this._lineShader.dispose();\n (_a = this._drawWrapper) === null || _a === void 0 ? void 0 : _a.dispose();\n }\n _processEdgeForAdjacencies(pa, pb, p0, p1, p2) {\n if ((pa === p0 && pb === p1) || (pa === p1 && pb === p0)) {\n return 0;\n }\n if ((pa === p1 && pb === p2) || (pa === p2 && pb === p1)) {\n return 1;\n }\n if ((pa === p2 && pb === p0) || (pa === p0 && pb === p2)) {\n return 2;\n }\n return -1;\n }\n _processEdgeForAdjacenciesWithVertices(pa, pb, p0, p1, p2) {\n const eps = 1e-10;\n if ((pa.equalsWithEpsilon(p0, eps) && pb.equalsWithEpsilon(p1, eps)) || (pa.equalsWithEpsilon(p1, eps) && pb.equalsWithEpsilon(p0, eps))) {\n return 0;\n }\n if ((pa.equalsWithEpsilon(p1, eps) && pb.equalsWithEpsilon(p2, eps)) || (pa.equalsWithEpsilon(p2, eps) && pb.equalsWithEpsilon(p1, eps))) {\n return 1;\n }\n if ((pa.equalsWithEpsilon(p2, eps) && pb.equalsWithEpsilon(p0, eps)) || (pa.equalsWithEpsilon(p0, eps) && pb.equalsWithEpsilon(p2, eps))) {\n return 2;\n }\n return -1;\n }\n /**\n * Checks if the pair of p0 and p1 is en edge\n * @param faceIndex\n * @param edge\n * @param faceNormals\n * @param p0\n * @param p1\n * @private\n */\n _checkEdge(faceIndex, edge, faceNormals, p0, p1) {\n let needToCreateLine;\n if (edge === undefined) {\n needToCreateLine = true;\n }\n else {\n const dotProduct = Vector3.Dot(faceNormals[faceIndex], faceNormals[edge]);\n needToCreateLine = dotProduct < this._epsilon;\n }\n if (needToCreateLine) {\n this.createLine(p0, p1, this._linesPositions.length / 3);\n }\n }\n /**\n * push line into the position, normal and index buffer\n * @param p0\n * @param p1\n * @param offset\n * @protected\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n createLine(p0, p1, offset) {\n // Positions\n this._linesPositions.push(p0.x, p0.y, p0.z, p0.x, p0.y, p0.z, p1.x, p1.y, p1.z, p1.x, p1.y, p1.z);\n // Normals\n this._linesNormals.push(p1.x, p1.y, p1.z, -1, p1.x, p1.y, p1.z, 1, p0.x, p0.y, p0.z, -1, p0.x, p0.y, p0.z, 1);\n // Indices\n this._linesIndices.push(offset, offset + 1, offset + 2, offset, offset + 2, offset + 3);\n }\n /**\n * See https://playground.babylonjs.com/#R3JR6V#1 for a visual display of the algorithm\n * @param edgePoints\n * @param indexTriangle\n * @param indices\n * @param remapVertexIndices\n */\n _tessellateTriangle(edgePoints, indexTriangle, indices, remapVertexIndices) {\n const makePointList = (edgePoints, pointIndices, firstIndex) => {\n if (firstIndex >= 0) {\n pointIndices.push(firstIndex);\n }\n for (let i = 0; i < edgePoints.length; ++i) {\n pointIndices.push(edgePoints[i][0]);\n }\n };\n let startEdge = 0;\n if (edgePoints[1].length >= edgePoints[0].length && edgePoints[1].length >= edgePoints[2].length) {\n startEdge = 1;\n }\n else if (edgePoints[2].length >= edgePoints[0].length && edgePoints[2].length >= edgePoints[1].length) {\n startEdge = 2;\n }\n for (let e = 0; e < 3; ++e) {\n if (e === startEdge) {\n edgePoints[e].sort((a, b) => (a[1] < b[1] ? -1 : a[1] > b[1] ? 1 : 0));\n }\n else {\n edgePoints[e].sort((a, b) => (a[1] > b[1] ? -1 : a[1] < b[1] ? 1 : 0));\n }\n }\n const mainPointIndices = [], otherPointIndices = [];\n makePointList(edgePoints[startEdge], mainPointIndices, -1);\n const numMainPoints = mainPointIndices.length;\n for (let i = startEdge + 2; i >= startEdge + 1; --i) {\n makePointList(edgePoints[i % 3], otherPointIndices, i !== startEdge + 2 ? remapVertexIndices[indices[indexTriangle + ((i + 1) % 3)]] : -1);\n }\n const numOtherPoints = otherPointIndices.length;\n const idxMain = 0;\n const idxOther = 0;\n indices.push(remapVertexIndices[indices[indexTriangle + startEdge]], mainPointIndices[0], otherPointIndices[0]);\n indices.push(remapVertexIndices[indices[indexTriangle + ((startEdge + 1) % 3)]], otherPointIndices[numOtherPoints - 1], mainPointIndices[numMainPoints - 1]);\n const bucketIsMain = numMainPoints <= numOtherPoints;\n const bucketStep = bucketIsMain ? numMainPoints : numOtherPoints;\n const bucketLimit = bucketIsMain ? numOtherPoints : numMainPoints;\n const bucketIdxLimit = bucketIsMain ? numMainPoints - 1 : numOtherPoints - 1;\n const winding = bucketIsMain ? 0 : 1;\n let numTris = numMainPoints + numOtherPoints - 2;\n let bucketIdx = bucketIsMain ? idxMain : idxOther;\n let nbucketIdx = bucketIsMain ? idxOther : idxMain;\n const bucketPoints = bucketIsMain ? mainPointIndices : otherPointIndices;\n const nbucketPoints = bucketIsMain ? otherPointIndices : mainPointIndices;\n let bucket = 0;\n while (numTris-- > 0) {\n if (winding) {\n indices.push(bucketPoints[bucketIdx], nbucketPoints[nbucketIdx]);\n }\n else {\n indices.push(nbucketPoints[nbucketIdx], bucketPoints[bucketIdx]);\n }\n bucket += bucketStep;\n let lastIdx;\n if (bucket >= bucketLimit && bucketIdx < bucketIdxLimit) {\n lastIdx = bucketPoints[++bucketIdx];\n bucket -= bucketLimit;\n }\n else {\n lastIdx = nbucketPoints[++nbucketIdx];\n }\n indices.push(lastIdx);\n }\n indices[indexTriangle + 0] = indices[indices.length - 3];\n indices[indexTriangle + 1] = indices[indices.length - 2];\n indices[indexTriangle + 2] = indices[indices.length - 1];\n indices.length = indices.length - 3;\n }\n _generateEdgesLinesAlternate() {\n var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k;\n const positions = this._source.getVerticesData(VertexBuffer.PositionKind);\n let indices = this._source.getIndices();\n if (!indices || !positions) {\n return;\n }\n if (!Array.isArray(indices)) {\n indices = Array.from(indices);\n }\n /**\n * Find all vertices that are at the same location (with an epsilon) and remapp them on the same vertex\n */\n const useFastVertexMerger = (_b = (_a = this._options) === null || _a === void 0 ? void 0 : _a.useFastVertexMerger) !== null && _b !== void 0 ? _b : true;\n const epsVertexMerge = useFastVertexMerger ? Math.round(-Math.log((_d = (_c = this._options) === null || _c === void 0 ? void 0 : _c.epsilonVertexMerge) !== null && _d !== void 0 ? _d : 1e-6) / Math.log(10)) : (_f = (_e = this._options) === null || _e === void 0 ? void 0 : _e.epsilonVertexMerge) !== null && _f !== void 0 ? _f : 1e-6;\n const remapVertexIndices = [];\n const uniquePositions = []; // list of unique index of vertices - needed for tessellation\n if (useFastVertexMerger) {\n const mapVertices = {};\n for (let v1 = 0; v1 < positions.length; v1 += 3) {\n const x1 = positions[v1 + 0], y1 = positions[v1 + 1], z1 = positions[v1 + 2];\n const key = x1.toFixed(epsVertexMerge) + \"|\" + y1.toFixed(epsVertexMerge) + \"|\" + z1.toFixed(epsVertexMerge);\n if (mapVertices[key] !== undefined) {\n remapVertexIndices.push(mapVertices[key]);\n }\n else {\n const idx = v1 / 3;\n mapVertices[key] = idx;\n remapVertexIndices.push(idx);\n uniquePositions.push(idx);\n }\n }\n }\n else {\n for (let v1 = 0; v1 < positions.length; v1 += 3) {\n const x1 = positions[v1 + 0], y1 = positions[v1 + 1], z1 = positions[v1 + 2];\n let found = false;\n for (let v2 = 0; v2 < v1 && !found; v2 += 3) {\n const x2 = positions[v2 + 0], y2 = positions[v2 + 1], z2 = positions[v2 + 2];\n if (Math.abs(x1 - x2) < epsVertexMerge && Math.abs(y1 - y2) < epsVertexMerge && Math.abs(z1 - z2) < epsVertexMerge) {\n remapVertexIndices.push(v2 / 3);\n found = true;\n break;\n }\n }\n if (!found) {\n remapVertexIndices.push(v1 / 3);\n uniquePositions.push(v1 / 3);\n }\n }\n }\n if ((_g = this._options) === null || _g === void 0 ? void 0 : _g.applyTessellation) {\n /**\n * Tessellate triangles if necessary:\n *\n * A\n * +\n * |\\\n * | \\\n * | \\\n * E + \\\n * /| \\\n * / | \\\n * / | \\\n * +---+-------+ B\n * D C\n *\n * For the edges to be rendered correctly, the ABC triangle has to be split into ABE and BCE, else AC is considered to be an edge, whereas only AE should be.\n *\n * The tessellation process looks for the vertices like E that are in-between two other vertices making of an edge and create new triangles as necessary\n */\n // First step: collect the triangles to tessellate\n const epsVertexAligned = (_j = (_h = this._options) === null || _h === void 0 ? void 0 : _h.epsilonVertexAligned) !== null && _j !== void 0 ? _j : 1e-6;\n const mustTesselate = []; // liste of triangles that must be tessellated\n for (let index = 0; index < indices.length; index += 3) {\n // loop over all triangles\n let triangleToTessellate;\n for (let i = 0; i < 3; ++i) {\n // loop over the 3 edges of the triangle\n const p0Index = remapVertexIndices[indices[index + i]];\n const p1Index = remapVertexIndices[indices[index + ((i + 1) % 3)]];\n const p2Index = remapVertexIndices[indices[index + ((i + 2) % 3)]];\n if (p0Index === p1Index) {\n continue;\n } // degenerated triangle - don't process\n const p0x = positions[p0Index * 3 + 0], p0y = positions[p0Index * 3 + 1], p0z = positions[p0Index * 3 + 2];\n const p1x = positions[p1Index * 3 + 0], p1y = positions[p1Index * 3 + 1], p1z = positions[p1Index * 3 + 2];\n const p0p1 = Math.sqrt((p1x - p0x) * (p1x - p0x) + (p1y - p0y) * (p1y - p0y) + (p1z - p0z) * (p1z - p0z));\n for (let v = 0; v < uniquePositions.length - 1; v++) {\n // loop over all (unique) vertices and look for the ones that would be in-between p0 and p1\n const vIndex = uniquePositions[v];\n if (vIndex === p0Index || vIndex === p1Index || vIndex === p2Index) {\n continue;\n } // don't handle the vertex if it is a vertex of the current triangle\n const x = positions[vIndex * 3 + 0], y = positions[vIndex * 3 + 1], z = positions[vIndex * 3 + 2];\n const p0p = Math.sqrt((x - p0x) * (x - p0x) + (y - p0y) * (y - p0y) + (z - p0z) * (z - p0z));\n const pp1 = Math.sqrt((x - p1x) * (x - p1x) + (y - p1y) * (y - p1y) + (z - p1z) * (z - p1z));\n if (Math.abs(p0p + pp1 - p0p1) < epsVertexAligned) {\n // vertices are aligned and p in-between p0 and p1 if distance(p0, p) + distance (p, p1) ~ distance(p0, p1)\n if (!triangleToTessellate) {\n triangleToTessellate = {\n index: index,\n edgesPoints: [[], [], []],\n };\n mustTesselate.push(triangleToTessellate);\n }\n triangleToTessellate.edgesPoints[i].push([vIndex, p0p]);\n }\n }\n }\n }\n // Second step: tesselate the triangles\n for (let t = 0; t < mustTesselate.length; ++t) {\n const triangle = mustTesselate[t];\n this._tessellateTriangle(triangle.edgesPoints, triangle.index, indices, remapVertexIndices);\n }\n mustTesselate.length = 0;\n }\n /**\n * Collect the edges to render\n */\n const edges = {};\n for (let index = 0; index < indices.length; index += 3) {\n let faceNormal;\n for (let i = 0; i < 3; ++i) {\n let p0Index = remapVertexIndices[indices[index + i]];\n let p1Index = remapVertexIndices[indices[index + ((i + 1) % 3)]];\n const p2Index = remapVertexIndices[indices[index + ((i + 2) % 3)]];\n if (p0Index === p1Index || ((p0Index === p2Index || p1Index === p2Index) && ((_k = this._options) === null || _k === void 0 ? void 0 : _k.removeDegeneratedTriangles))) {\n continue;\n }\n TmpVectors.Vector3[0].copyFromFloats(positions[p0Index * 3 + 0], positions[p0Index * 3 + 1], positions[p0Index * 3 + 2]);\n TmpVectors.Vector3[1].copyFromFloats(positions[p1Index * 3 + 0], positions[p1Index * 3 + 1], positions[p1Index * 3 + 2]);\n TmpVectors.Vector3[2].copyFromFloats(positions[p2Index * 3 + 0], positions[p2Index * 3 + 1], positions[p2Index * 3 + 2]);\n if (!faceNormal) {\n TmpVectors.Vector3[1].subtractToRef(TmpVectors.Vector3[0], TmpVectors.Vector3[3]);\n TmpVectors.Vector3[2].subtractToRef(TmpVectors.Vector3[1], TmpVectors.Vector3[4]);\n faceNormal = Vector3.Cross(TmpVectors.Vector3[3], TmpVectors.Vector3[4]);\n faceNormal.normalize();\n }\n if (p0Index > p1Index) {\n const tmp = p0Index;\n p0Index = p1Index;\n p1Index = tmp;\n }\n const key = p0Index + \"_\" + p1Index;\n const ei = edges[key];\n if (ei) {\n if (!ei.done) {\n const dotProduct = Vector3.Dot(faceNormal, ei.normal);\n if (dotProduct < this._epsilon) {\n this.createLine(TmpVectors.Vector3[0], TmpVectors.Vector3[1], this._linesPositions.length / 3);\n }\n ei.done = true;\n }\n }\n else {\n edges[key] = { normal: faceNormal, done: false, index: index, i: i };\n }\n }\n }\n for (const key in edges) {\n const ei = edges[key];\n if (!ei.done) {\n // Orphaned edge - we must display it\n const p0Index = remapVertexIndices[indices[ei.index + ei.i]];\n const p1Index = remapVertexIndices[indices[ei.index + ((ei.i + 1) % 3)]];\n TmpVectors.Vector3[0].copyFromFloats(positions[p0Index * 3 + 0], positions[p0Index * 3 + 1], positions[p0Index * 3 + 2]);\n TmpVectors.Vector3[1].copyFromFloats(positions[p1Index * 3 + 0], positions[p1Index * 3 + 1], positions[p1Index * 3 + 2]);\n this.createLine(TmpVectors.Vector3[0], TmpVectors.Vector3[1], this._linesPositions.length / 3);\n }\n }\n /**\n * Merge into a single mesh\n */\n const engine = this._source.getScene().getEngine();\n this._buffers[VertexBuffer.PositionKind] = new VertexBuffer(engine, this._linesPositions, VertexBuffer.PositionKind, false);\n this._buffers[VertexBuffer.NormalKind] = new VertexBuffer(engine, this._linesNormals, VertexBuffer.NormalKind, false, false, 4);\n this._buffersForInstances[VertexBuffer.PositionKind] = this._buffers[VertexBuffer.PositionKind];\n this._buffersForInstances[VertexBuffer.NormalKind] = this._buffers[VertexBuffer.NormalKind];\n this._ib = engine.createIndexBuffer(this._linesIndices);\n this._indicesCount = this._linesIndices.length;\n }\n /**\n * Generates lines edges from adjacencjes\n * @private\n */\n _generateEdgesLines() {\n const positions = this._source.getVerticesData(VertexBuffer.PositionKind);\n const indices = this._source.getIndices();\n if (!indices || !positions) {\n return;\n }\n // First let's find adjacencies\n const adjacencies = new Array();\n const faceNormals = new Array();\n let index;\n let faceAdjacencies;\n // Prepare faces\n for (index = 0; index < indices.length; index += 3) {\n faceAdjacencies = new FaceAdjacencies();\n const p0Index = indices[index];\n const p1Index = indices[index + 1];\n const p2Index = indices[index + 2];\n faceAdjacencies.p0 = new Vector3(positions[p0Index * 3], positions[p0Index * 3 + 1], positions[p0Index * 3 + 2]);\n faceAdjacencies.p1 = new Vector3(positions[p1Index * 3], positions[p1Index * 3 + 1], positions[p1Index * 3 + 2]);\n faceAdjacencies.p2 = new Vector3(positions[p2Index * 3], positions[p2Index * 3 + 1], positions[p2Index * 3 + 2]);\n const faceNormal = Vector3.Cross(faceAdjacencies.p1.subtract(faceAdjacencies.p0), faceAdjacencies.p2.subtract(faceAdjacencies.p1));\n faceNormal.normalize();\n faceNormals.push(faceNormal);\n adjacencies.push(faceAdjacencies);\n }\n // Scan\n for (index = 0; index < adjacencies.length; index++) {\n faceAdjacencies = adjacencies[index];\n for (let otherIndex = index + 1; otherIndex < adjacencies.length; otherIndex++) {\n const otherFaceAdjacencies = adjacencies[otherIndex];\n if (faceAdjacencies.edgesConnectedCount === 3) {\n // Full\n break;\n }\n if (otherFaceAdjacencies.edgesConnectedCount === 3) {\n // Full\n continue;\n }\n const otherP0 = indices[otherIndex * 3];\n const otherP1 = indices[otherIndex * 3 + 1];\n const otherP2 = indices[otherIndex * 3 + 2];\n for (let edgeIndex = 0; edgeIndex < 3; edgeIndex++) {\n let otherEdgeIndex = 0;\n if (faceAdjacencies.edges[edgeIndex] !== undefined) {\n continue;\n }\n switch (edgeIndex) {\n case 0:\n if (this._checkVerticesInsteadOfIndices) {\n otherEdgeIndex = this._processEdgeForAdjacenciesWithVertices(faceAdjacencies.p0, faceAdjacencies.p1, otherFaceAdjacencies.p0, otherFaceAdjacencies.p1, otherFaceAdjacencies.p2);\n }\n else {\n otherEdgeIndex = this._processEdgeForAdjacencies(indices[index * 3], indices[index * 3 + 1], otherP0, otherP1, otherP2);\n }\n break;\n case 1:\n if (this._checkVerticesInsteadOfIndices) {\n otherEdgeIndex = this._processEdgeForAdjacenciesWithVertices(faceAdjacencies.p1, faceAdjacencies.p2, otherFaceAdjacencies.p0, otherFaceAdjacencies.p1, otherFaceAdjacencies.p2);\n }\n else {\n otherEdgeIndex = this._processEdgeForAdjacencies(indices[index * 3 + 1], indices[index * 3 + 2], otherP0, otherP1, otherP2);\n }\n break;\n case 2:\n if (this._checkVerticesInsteadOfIndices) {\n otherEdgeIndex = this._processEdgeForAdjacenciesWithVertices(faceAdjacencies.p2, faceAdjacencies.p0, otherFaceAdjacencies.p0, otherFaceAdjacencies.p1, otherFaceAdjacencies.p2);\n }\n else {\n otherEdgeIndex = this._processEdgeForAdjacencies(indices[index * 3 + 2], indices[index * 3], otherP0, otherP1, otherP2);\n }\n break;\n }\n if (otherEdgeIndex === -1) {\n continue;\n }\n faceAdjacencies.edges[edgeIndex] = otherIndex;\n otherFaceAdjacencies.edges[otherEdgeIndex] = index;\n faceAdjacencies.edgesConnectedCount++;\n otherFaceAdjacencies.edgesConnectedCount++;\n if (faceAdjacencies.edgesConnectedCount === 3) {\n break;\n }\n }\n }\n }\n // Create lines\n for (index = 0; index < adjacencies.length; index++) {\n // We need a line when a face has no adjacency on a specific edge or if all the adjacencies has an angle greater than epsilon\n const current = adjacencies[index];\n this._checkEdge(index, current.edges[0], faceNormals, current.p0, current.p1);\n this._checkEdge(index, current.edges[1], faceNormals, current.p1, current.p2);\n this._checkEdge(index, current.edges[2], faceNormals, current.p2, current.p0);\n }\n // Merge into a single mesh\n const engine = this._source.getScene().getEngine();\n this._buffers[VertexBuffer.PositionKind] = new VertexBuffer(engine, this._linesPositions, VertexBuffer.PositionKind, false);\n this._buffers[VertexBuffer.NormalKind] = new VertexBuffer(engine, this._linesNormals, VertexBuffer.NormalKind, false, false, 4);\n this._buffersForInstances[VertexBuffer.PositionKind] = this._buffers[VertexBuffer.PositionKind];\n this._buffersForInstances[VertexBuffer.NormalKind] = this._buffers[VertexBuffer.NormalKind];\n this._ib = engine.createIndexBuffer(this._linesIndices);\n this._indicesCount = this._linesIndices.length;\n }\n /**\n * Checks whether or not the edges renderer is ready to render.\n * @returns true if ready, otherwise false.\n */\n isReady() {\n return this._lineShader.isReady(this._source, (this._source.hasInstances && this.customInstances.length > 0) || this._source.hasThinInstances);\n }\n /**\n * Renders the edges of the attached mesh,\n */\n render() {\n const scene = this._source.getScene();\n const currentDrawWrapper = this._lineShader._getDrawWrapper();\n if (this._drawWrapper) {\n this._lineShader._setDrawWrapper(this._drawWrapper);\n }\n if (!this.isReady() || !scene.activeCamera) {\n this._lineShader._setDrawWrapper(currentDrawWrapper);\n return;\n }\n const hasInstances = this._source.hasInstances && this.customInstances.length > 0;\n const useBuffersWithInstances = hasInstances || this._source.hasThinInstances;\n let instanceCount = 0;\n if (useBuffersWithInstances) {\n this._buffersForInstances[\"world0\"] = this._source.getVertexBuffer(\"world0\");\n this._buffersForInstances[\"world1\"] = this._source.getVertexBuffer(\"world1\");\n this._buffersForInstances[\"world2\"] = this._source.getVertexBuffer(\"world2\");\n this._buffersForInstances[\"world3\"] = this._source.getVertexBuffer(\"world3\");\n if (hasInstances) {\n const instanceStorage = this._source._instanceDataStorage;\n instanceCount = this.customInstances.length;\n if (!instanceStorage.instancesData) {\n if (!this._source.getScene()._activeMeshesFrozen) {\n this.customInstances.reset();\n }\n return;\n }\n if (!instanceStorage.isFrozen) {\n let offset = 0;\n for (let i = 0; i < instanceCount; ++i) {\n this.customInstances.data[i].copyToArray(instanceStorage.instancesData, offset);\n offset += 16;\n }\n instanceStorage.instancesBuffer.updateDirectly(instanceStorage.instancesData, 0, instanceCount);\n }\n }\n else {\n instanceCount = this._source.thinInstanceCount;\n }\n }\n const engine = scene.getEngine();\n this._lineShader._preBind();\n if (this._source.edgesColor.a !== 1) {\n engine.setAlphaMode(2);\n }\n else {\n engine.setAlphaMode(0);\n }\n // VBOs\n engine.bindBuffers(useBuffersWithInstances ? this._buffersForInstances : this._buffers, this._ib, this._lineShader.getEffect());\n scene.resetCachedMaterial();\n this._lineShader.setColor4(\"color\", this._source.edgesColor);\n if (scene.activeCamera.mode === Camera.ORTHOGRAPHIC_CAMERA) {\n this._lineShader.setFloat(\"width\", this._source.edgesWidth / this.edgesWidthScalerForOrthographic);\n }\n else {\n this._lineShader.setFloat(\"width\", this._source.edgesWidth / this.edgesWidthScalerForPerspective);\n }\n this._lineShader.setFloat(\"aspectRatio\", engine.getAspectRatio(scene.activeCamera));\n this._lineShader.bind(this._source.getWorldMatrix());\n // Draw order\n engine.drawElementsType(Material.TriangleFillMode, 0, this._indicesCount, instanceCount);\n this._lineShader.unbind();\n if (useBuffersWithInstances) {\n engine.unbindInstanceAttributes();\n }\n if (!this._source.getScene()._activeMeshesFrozen) {\n this.customInstances.reset();\n }\n this._lineShader._setDrawWrapper(currentDrawWrapper);\n }\n }\n /**\n * LineEdgesRenderer for LineMeshes to remove unnecessary triangulation\n */\n class LineEdgesRenderer extends EdgesRenderer {\n /**\n * This constructor turns off auto generating edges line in Edges Renderer to make it here.\n * @param source LineMesh used to generate edges\n * @param epsilon not important (specified angle for edge detection)\n * @param checkVerticesInsteadOfIndices not important for LineMesh\n */\n constructor(source, epsilon = 0.95, checkVerticesInsteadOfIndices = false) {\n super(source, epsilon, checkVerticesInsteadOfIndices, false);\n this._generateEdgesLines();\n }\n /**\n * Generate edges for each line in LinesMesh. Every Line should be rendered as edge.\n */\n _generateEdgesLines() {\n const positions = this._source.getVerticesData(VertexBuffer.PositionKind);\n const indices = this._source.getIndices();\n if (!indices || !positions) {\n return;\n }\n const p0 = TmpVectors.Vector3[0];\n const p1 = TmpVectors.Vector3[1];\n const len = indices.length - 1;\n for (let i = 0, offset = 0; i < len; i += 2, offset += 4) {\n Vector3.FromArrayToRef(positions, 3 * indices[i], p0);\n Vector3.FromArrayToRef(positions, 3 * indices[i + 1], p1);\n this.createLine(p0, p1, offset);\n }\n // Merge into a single mesh\n const engine = this._source.getScene().getEngine();\n this._buffers[VertexBuffer.PositionKind] = new VertexBuffer(engine, this._linesPositions, VertexBuffer.PositionKind, false);\n this._buffers[VertexBuffer.NormalKind] = new VertexBuffer(engine, this._linesNormals, VertexBuffer.NormalKind, false, false, 4);\n this._ib = engine.createIndexBuffer(this._linesIndices);\n this._indicesCount = this._linesIndices.length;\n }\n }\n\n ThinEngine.prototype.restoreSingleAttachment = function () {\n const gl = this._gl;\n this.bindAttachments([gl.BACK]);\n };\n ThinEngine.prototype.restoreSingleAttachmentForRenderTarget = function () {\n const gl = this._gl;\n this.bindAttachments([gl.COLOR_ATTACHMENT0]);\n };\n ThinEngine.prototype.buildTextureLayout = function (textureStatus) {\n const gl = this._gl;\n const result = [];\n for (let i = 0; i < textureStatus.length; i++) {\n if (textureStatus[i]) {\n result.push(gl[\"COLOR_ATTACHMENT\" + i]);\n }\n else {\n result.push(gl.NONE);\n }\n }\n return result;\n };\n ThinEngine.prototype.bindAttachments = function (attachments) {\n const gl = this._gl;\n gl.drawBuffers(attachments);\n };\n ThinEngine.prototype.unBindMultiColorAttachmentFramebuffer = function (rtWrapper, disableGenerateMipMaps = false, onBeforeUnbind) {\n this._currentRenderTarget = null;\n // If MSAA, we need to bitblt back to main texture\n const gl = this._gl;\n const attachments = rtWrapper._attachments;\n const count = attachments.length;\n if (rtWrapper._MSAAFramebuffer) {\n gl.bindFramebuffer(gl.READ_FRAMEBUFFER, rtWrapper._MSAAFramebuffer);\n gl.bindFramebuffer(gl.DRAW_FRAMEBUFFER, rtWrapper._framebuffer);\n for (let i = 0; i < count; i++) {\n const texture = rtWrapper.textures[i];\n for (let j = 0; j < count; j++) {\n attachments[j] = gl.NONE;\n }\n attachments[i] = gl[this.webGLVersion > 1 ? \"COLOR_ATTACHMENT\" + i : \"COLOR_ATTACHMENT\" + i + \"_WEBGL\"];\n gl.readBuffer(attachments[i]);\n gl.drawBuffers(attachments);\n gl.blitFramebuffer(0, 0, texture.width, texture.height, 0, 0, texture.width, texture.height, gl.COLOR_BUFFER_BIT, gl.NEAREST);\n }\n for (let i = 0; i < count; i++) {\n attachments[i] = gl[this.webGLVersion > 1 ? \"COLOR_ATTACHMENT\" + i : \"COLOR_ATTACHMENT\" + i + \"_WEBGL\"];\n }\n gl.drawBuffers(attachments);\n }\n for (let i = 0; i < count; i++) {\n const texture = rtWrapper.textures[i];\n if ((texture === null || texture === void 0 ? void 0 : texture.generateMipMaps) && !disableGenerateMipMaps && !texture.isCube) {\n this._bindTextureDirectly(gl.TEXTURE_2D, texture, true);\n gl.generateMipmap(gl.TEXTURE_2D);\n this._bindTextureDirectly(gl.TEXTURE_2D, null);\n }\n }\n if (onBeforeUnbind) {\n if (rtWrapper._MSAAFramebuffer) {\n // Bind the correct framebuffer\n this._bindUnboundFramebuffer(rtWrapper._framebuffer);\n }\n onBeforeUnbind();\n }\n this._bindUnboundFramebuffer(null);\n };\n ThinEngine.prototype.createMultipleRenderTarget = function (size, options, initializeBuffers = true) {\n var _a;\n let generateMipMaps = false;\n let generateDepthBuffer = true;\n let generateStencilBuffer = false;\n let generateDepthTexture = false;\n let depthTextureFormat = 15;\n let textureCount = 1;\n const defaultType = 0;\n const defaultSamplingMode = 3;\n const defaultUseSRGBBuffer = false;\n const defaultFormat = 5;\n const defaultTarget = 3553;\n let types = new Array();\n let samplingModes = new Array();\n let useSRGBBuffers = new Array();\n let formats = new Array();\n let targets = new Array();\n let faceIndex = new Array();\n let layerIndex = new Array();\n let layers = new Array();\n const rtWrapper = this._createHardwareRenderTargetWrapper(true, false, size);\n if (options !== undefined) {\n generateMipMaps = options.generateMipMaps === undefined ? false : options.generateMipMaps;\n generateDepthBuffer = options.generateDepthBuffer === undefined ? true : options.generateDepthBuffer;\n generateStencilBuffer = options.generateStencilBuffer === undefined ? false : options.generateStencilBuffer;\n generateDepthTexture = options.generateDepthTexture === undefined ? false : options.generateDepthTexture;\n textureCount = options.textureCount || 1;\n if (options.types) {\n types = options.types;\n }\n if (options.samplingModes) {\n samplingModes = options.samplingModes;\n }\n if (options.useSRGBBuffers) {\n useSRGBBuffers = options.useSRGBBuffers;\n }\n if (options.formats) {\n formats = options.formats;\n }\n if (options.targetTypes) {\n targets = options.targetTypes;\n }\n if (options.faceIndex) {\n faceIndex = options.faceIndex;\n }\n if (options.layerIndex) {\n layerIndex = options.layerIndex;\n }\n if (options.layerCounts) {\n layers = options.layerCounts;\n }\n if (this.webGLVersion > 1 &&\n (options.depthTextureFormat === 13 ||\n options.depthTextureFormat === 17 ||\n options.depthTextureFormat === 16 ||\n options.depthTextureFormat === 14 ||\n options.depthTextureFormat === 18)) {\n depthTextureFormat = options.depthTextureFormat;\n }\n }\n const gl = this._gl;\n // Create the framebuffer\n const framebuffer = gl.createFramebuffer();\n this._bindUnboundFramebuffer(framebuffer);\n const width = size.width || size;\n const height = size.height || size;\n const textures = [];\n const attachments = [];\n const useStencilTexture = this.webGLVersion > 1 &&\n generateDepthTexture &&\n (options.depthTextureFormat === 13 ||\n options.depthTextureFormat === 17 ||\n options.depthTextureFormat === 18);\n const depthStencilBuffer = this._setupFramebufferDepthAttachments(!useStencilTexture && generateStencilBuffer, !generateDepthTexture && generateDepthBuffer, width, height);\n rtWrapper._framebuffer = framebuffer;\n rtWrapper._depthStencilBuffer = depthStencilBuffer;\n rtWrapper._generateDepthBuffer = !generateDepthTexture && generateDepthBuffer;\n rtWrapper._generateStencilBuffer = !useStencilTexture && generateStencilBuffer;\n rtWrapper._attachments = attachments;\n for (let i = 0; i < textureCount; i++) {\n let samplingMode = samplingModes[i] || defaultSamplingMode;\n let type = types[i] || defaultType;\n let useSRGBBuffer = useSRGBBuffers[i] || defaultUseSRGBBuffer;\n const format = formats[i] || defaultFormat;\n const target = targets[i] || defaultTarget;\n const layerCount = (_a = layers[i]) !== null && _a !== void 0 ? _a : 1;\n if (type === 1 && !this._caps.textureFloatLinearFiltering) {\n // if floating point linear (gl.FLOAT) then force to NEAREST_SAMPLINGMODE\n samplingMode = 1;\n }\n else if (type === 2 && !this._caps.textureHalfFloatLinearFiltering) {\n // if floating point linear (HALF_FLOAT) then force to NEAREST_SAMPLINGMODE\n samplingMode = 1;\n }\n const filters = this._getSamplingParameters(samplingMode, generateMipMaps);\n if (type === 1 && !this._caps.textureFloat) {\n type = 0;\n Logger.Warn(\"Float textures are not supported. Render target forced to TEXTURETYPE_UNSIGNED_BYTE type\");\n }\n useSRGBBuffer = useSRGBBuffer && this._caps.supportSRGBBuffers && (this.webGLVersion > 1 || this.isWebGPU);\n const isWebGL2 = this.webGLVersion > 1;\n const attachment = gl[isWebGL2 ? \"COLOR_ATTACHMENT\" + i : \"COLOR_ATTACHMENT\" + i + \"_WEBGL\"];\n attachments.push(attachment);\n if (target === -1) {\n continue;\n }\n const texture = new InternalTexture(this, InternalTextureSource.MultiRenderTarget);\n textures[i] = texture;\n gl.activeTexture(gl[\"TEXTURE\" + i]);\n gl.bindTexture(target, texture._hardwareTexture.underlyingResource);\n gl.texParameteri(target, gl.TEXTURE_MAG_FILTER, filters.mag);\n gl.texParameteri(target, gl.TEXTURE_MIN_FILTER, filters.min);\n gl.texParameteri(target, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n gl.texParameteri(target, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n const internalSizedFormat = this._getRGBABufferInternalSizedFormat(type, format, useSRGBBuffer);\n const internalFormat = this._getInternalFormat(format);\n const webGLTextureType = this._getWebGLTextureType(type);\n if (isWebGL2 && (target === 35866 || target === 32879)) {\n if (target === 35866) {\n texture.is2DArray = true;\n }\n else {\n texture.is3D = true;\n }\n texture.baseDepth = texture.depth = layerCount;\n gl.texImage3D(target, 0, internalSizedFormat, width, height, layerCount, 0, internalFormat, webGLTextureType, null);\n }\n else if (target === 34067) {\n // We have to generate all faces to complete the framebuffer\n for (let i = 0; i < 6; i++) {\n gl.texImage2D(gl.TEXTURE_CUBE_MAP_POSITIVE_X + i, 0, internalSizedFormat, width, height, 0, internalFormat, webGLTextureType, null);\n }\n texture.isCube = true;\n }\n else {\n gl.texImage2D(gl.TEXTURE_2D, 0, internalSizedFormat, width, height, 0, internalFormat, webGLTextureType, null);\n }\n if (generateMipMaps) {\n gl.generateMipmap(target);\n }\n // Unbind\n this._bindTextureDirectly(target, null);\n texture.baseWidth = width;\n texture.baseHeight = height;\n texture.width = width;\n texture.height = height;\n texture.isReady = true;\n texture.samples = 1;\n texture.generateMipMaps = generateMipMaps;\n texture.samplingMode = samplingMode;\n texture.type = type;\n texture._useSRGBBuffer = useSRGBBuffer;\n texture.format = format;\n this._internalTexturesCache.push(texture);\n }\n if (generateDepthTexture && this._caps.depthTextureExtension) {\n // Depth texture\n const depthTexture = new InternalTexture(this, InternalTextureSource.Depth);\n let depthTextureType = 5;\n let glDepthTextureInternalFormat = gl.DEPTH_COMPONENT16;\n let glDepthTextureFormat = gl.DEPTH_COMPONENT;\n let glDepthTextureType = gl.UNSIGNED_SHORT;\n let glDepthTextureAttachment = gl.DEPTH_ATTACHMENT;\n if (this.webGLVersion < 2) {\n glDepthTextureInternalFormat = gl.DEPTH_COMPONENT;\n }\n else {\n if (depthTextureFormat === 14) {\n depthTextureType = 1;\n glDepthTextureType = gl.FLOAT;\n glDepthTextureInternalFormat = gl.DEPTH_COMPONENT32F;\n }\n else if (depthTextureFormat === 18) {\n depthTextureType = 0;\n glDepthTextureType = gl.FLOAT_32_UNSIGNED_INT_24_8_REV;\n glDepthTextureInternalFormat = gl.DEPTH32F_STENCIL8;\n glDepthTextureFormat = gl.DEPTH_STENCIL;\n glDepthTextureAttachment = gl.DEPTH_STENCIL_ATTACHMENT;\n }\n else if (depthTextureFormat === 16) {\n depthTextureType = 0;\n glDepthTextureType = gl.UNSIGNED_INT;\n glDepthTextureInternalFormat = gl.DEPTH_COMPONENT24;\n glDepthTextureAttachment = gl.DEPTH_ATTACHMENT;\n }\n else if (depthTextureFormat === 13 || depthTextureFormat === 17) {\n depthTextureType = 12;\n glDepthTextureType = gl.UNSIGNED_INT_24_8;\n glDepthTextureInternalFormat = gl.DEPTH24_STENCIL8;\n glDepthTextureFormat = gl.DEPTH_STENCIL;\n glDepthTextureAttachment = gl.DEPTH_STENCIL_ATTACHMENT;\n }\n }\n gl.activeTexture(gl.TEXTURE0);\n gl.bindTexture(gl.TEXTURE_2D, depthTexture._hardwareTexture.underlyingResource);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n gl.texImage2D(gl.TEXTURE_2D, 0, glDepthTextureInternalFormat, width, height, 0, glDepthTextureFormat, glDepthTextureType, null);\n gl.framebufferTexture2D(gl.FRAMEBUFFER, glDepthTextureAttachment, gl.TEXTURE_2D, depthTexture._hardwareTexture.underlyingResource, 0);\n depthTexture.baseWidth = width;\n depthTexture.baseHeight = height;\n depthTexture.width = width;\n depthTexture.height = height;\n depthTexture.isReady = true;\n depthTexture.samples = 1;\n depthTexture.generateMipMaps = generateMipMaps;\n depthTexture.samplingMode = 1;\n depthTexture.format = depthTextureFormat;\n depthTexture.type = depthTextureType;\n textures[textureCount] = depthTexture;\n this._internalTexturesCache.push(depthTexture);\n }\n rtWrapper.setTextures(textures);\n if (initializeBuffers) {\n gl.drawBuffers(attachments);\n }\n this._bindUnboundFramebuffer(null);\n rtWrapper.setLayerAndFaceIndices(layerIndex, faceIndex);\n this.resetTextureCache();\n return rtWrapper;\n };\n ThinEngine.prototype.updateMultipleRenderTargetTextureSampleCount = function (rtWrapper, samples, initializeBuffers = true) {\n if (this.webGLVersion < 2 || !rtWrapper || !rtWrapper.texture) {\n return 1;\n }\n if (rtWrapper.samples === samples) {\n return samples;\n }\n const count = rtWrapper._attachments.length;\n if (count === 0) {\n return 1;\n }\n const gl = this._gl;\n samples = Math.min(samples, this.getCaps().maxMSAASamples);\n // Dispose previous render buffers\n const useDepthStencil = !!rtWrapper._depthStencilBuffer;\n if (useDepthStencil) {\n gl.deleteRenderbuffer(rtWrapper._depthStencilBuffer);\n rtWrapper._depthStencilBuffer = null;\n }\n if (rtWrapper._MSAAFramebuffer) {\n gl.deleteFramebuffer(rtWrapper._MSAAFramebuffer);\n rtWrapper._MSAAFramebuffer = null;\n }\n if (samples > 1 && typeof gl.renderbufferStorageMultisample === \"function\") {\n const framebuffer = gl.createFramebuffer();\n if (!framebuffer) {\n throw new Error(\"Unable to create multi sampled framebuffer\");\n }\n rtWrapper._MSAAFramebuffer = framebuffer;\n this._bindUnboundFramebuffer(framebuffer);\n const attachments = [];\n for (let i = 0; i < count; i++) {\n const texture = rtWrapper.textures[i];\n const hardwareTexture = texture._hardwareTexture;\n hardwareTexture.releaseMSAARenderBuffers();\n }\n for (let i = 0; i < count; i++) {\n const texture = rtWrapper.textures[i];\n const hardwareTexture = texture._hardwareTexture;\n const attachment = gl[this.webGLVersion > 1 ? \"COLOR_ATTACHMENT\" + i : \"COLOR_ATTACHMENT\" + i + \"_WEBGL\"];\n const colorRenderbuffer = this._createRenderBuffer(texture.width, texture.height, samples, -1 /* not used */, this._getRGBAMultiSampleBufferFormat(texture.type, texture.format), attachment);\n if (!colorRenderbuffer) {\n throw new Error(\"Unable to create multi sampled framebuffer\");\n }\n hardwareTexture.addMSAARenderBuffer(colorRenderbuffer);\n texture.samples = samples;\n attachments.push(attachment);\n }\n if (initializeBuffers) {\n gl.drawBuffers(attachments);\n }\n }\n else {\n this._bindUnboundFramebuffer(rtWrapper._framebuffer);\n }\n if (useDepthStencil) {\n rtWrapper._depthStencilBuffer = this._setupFramebufferDepthAttachments(rtWrapper._generateStencilBuffer, rtWrapper._generateDepthBuffer, rtWrapper.texture.width, rtWrapper.texture.height, samples);\n }\n this._bindUnboundFramebuffer(null);\n return samples;\n };\n\n /**\n * A multi render target, like a render target provides the ability to render to a texture.\n * Unlike the render target, it can render to several draw buffers in one draw.\n * This is specially interesting in deferred rendering or for any effects requiring more than\n * just one color from a single pass.\n */\n class MultiRenderTarget extends RenderTargetTexture {\n /**\n * Get if draw buffers are currently supported by the used hardware and browser.\n */\n get isSupported() {\n var _a, _b;\n return (_b = (_a = this._engine) === null || _a === void 0 ? void 0 : _a.getCaps().drawBuffersExtension) !== null && _b !== void 0 ? _b : false;\n }\n /**\n * Get the list of textures generated by the multi render target.\n */\n get textures() {\n return this._textures;\n }\n /**\n * Gets the number of textures in this MRT. This number can be different from `_textures.length` in case a depth texture is generated.\n */\n get count() {\n return this._count;\n }\n /**\n * Get the depth texture generated by the multi render target if options.generateDepthTexture has been set\n */\n get depthTexture() {\n return this._textures[this._textures.length - 1];\n }\n /**\n * Set the wrapping mode on U of all the textures we are rendering to.\n * Can be any of the Texture. (CLAMP_ADDRESSMODE, MIRROR_ADDRESSMODE or WRAP_ADDRESSMODE)\n */\n set wrapU(wrap) {\n if (this._textures) {\n for (let i = 0; i < this._textures.length; i++) {\n this._textures[i].wrapU = wrap;\n }\n }\n }\n /**\n * Set the wrapping mode on V of all the textures we are rendering to.\n * Can be any of the Texture. (CLAMP_ADDRESSMODE, MIRROR_ADDRESSMODE or WRAP_ADDRESSMODE)\n */\n set wrapV(wrap) {\n if (this._textures) {\n for (let i = 0; i < this._textures.length; i++) {\n this._textures[i].wrapV = wrap;\n }\n }\n }\n /**\n * Instantiate a new multi render target texture.\n * A multi render target, like a render target provides the ability to render to a texture.\n * Unlike the render target, it can render to several draw buffers in one draw.\n * This is specially interesting in deferred rendering or for any effects requiring more than\n * just one color from a single pass.\n * @param name Define the name of the texture\n * @param size Define the size of the buffers to render to\n * @param count Define the number of target we are rendering into\n * @param scene Define the scene the texture belongs to\n * @param options Define the options used to create the multi render target\n * @param textureNames Define the names to set to the textures (if count > 0 - optional)\n */\n constructor(name, size, count, scene, options, textureNames) {\n const generateMipMaps = options && options.generateMipMaps ? options.generateMipMaps : false;\n const generateDepthTexture = options && options.generateDepthTexture ? options.generateDepthTexture : false;\n const depthTextureFormat = options && options.depthTextureFormat ? options.depthTextureFormat : 15;\n const doNotChangeAspectRatio = !options || options.doNotChangeAspectRatio === undefined ? true : options.doNotChangeAspectRatio;\n const drawOnlyOnFirstAttachmentByDefault = options && options.drawOnlyOnFirstAttachmentByDefault ? options.drawOnlyOnFirstAttachmentByDefault : false;\n super(name, size, scene, generateMipMaps, doNotChangeAspectRatio, undefined, undefined, undefined, undefined, undefined, undefined, undefined, true);\n if (!this.isSupported) {\n this.dispose();\n return;\n }\n this._textureNames = textureNames;\n const types = [];\n const samplingModes = [];\n const useSRGBBuffers = [];\n const formats = [];\n const targetTypes = [];\n const faceIndex = [];\n const layerIndex = [];\n const layerCounts = [];\n this._initTypes(count, types, samplingModes, useSRGBBuffers, formats, targetTypes, faceIndex, layerIndex, layerCounts, options);\n const generateDepthBuffer = !options || options.generateDepthBuffer === undefined ? true : options.generateDepthBuffer;\n const generateStencilBuffer = !options || options.generateStencilBuffer === undefined ? false : options.generateStencilBuffer;\n this._multiRenderTargetOptions = {\n samplingModes: samplingModes,\n generateMipMaps: generateMipMaps,\n generateDepthBuffer: generateDepthBuffer,\n generateStencilBuffer: generateStencilBuffer,\n generateDepthTexture: generateDepthTexture,\n depthTextureFormat: depthTextureFormat,\n types: types,\n textureCount: count,\n useSRGBBuffers: useSRGBBuffers,\n formats: formats,\n targetTypes: targetTypes,\n faceIndex: faceIndex,\n layerIndex: layerIndex,\n layerCounts: layerCounts,\n };\n this._count = count;\n this._drawOnlyOnFirstAttachmentByDefault = drawOnlyOnFirstAttachmentByDefault;\n if (count > 0) {\n this._createInternalTextures();\n this._createTextures(textureNames);\n }\n }\n _initTypes(count, types, samplingModes, useSRGBBuffers, formats, targets, faceIndex, layerIndex, layerCounts, options) {\n for (let i = 0; i < count; i++) {\n if (options && options.types && options.types[i] !== undefined) {\n types.push(options.types[i]);\n }\n else {\n types.push(options && options.defaultType ? options.defaultType : 0);\n }\n if (options && options.samplingModes && options.samplingModes[i] !== undefined) {\n samplingModes.push(options.samplingModes[i]);\n }\n else {\n samplingModes.push(Texture.BILINEAR_SAMPLINGMODE);\n }\n if (options && options.useSRGBBuffers && options.useSRGBBuffers[i] !== undefined) {\n useSRGBBuffers.push(options.useSRGBBuffers[i]);\n }\n else {\n useSRGBBuffers.push(false);\n }\n if (options && options.formats && options.formats[i] !== undefined) {\n formats.push(options.formats[i]);\n }\n else {\n formats.push(5);\n }\n if (options && options.targetTypes && options.targetTypes[i] !== undefined) {\n targets.push(options.targetTypes[i]);\n }\n else {\n targets.push(3553);\n }\n if (options && options.faceIndex && options.faceIndex[i] !== undefined) {\n faceIndex.push(options.faceIndex[i]);\n }\n else {\n faceIndex.push(0);\n }\n if (options && options.layerIndex && options.layerIndex[i] !== undefined) {\n layerIndex.push(options.layerIndex[i]);\n }\n else {\n layerIndex.push(0);\n }\n if (options && options.layerCounts && options.layerCounts[i] !== undefined) {\n layerCounts.push(options.layerCounts[i]);\n }\n else {\n layerCounts.push(1);\n }\n }\n }\n _createInternaTextureIndexMapping() {\n const mapMainInternalTexture2Index = {};\n const mapInternalTexture2MainIndex = [];\n if (!this._renderTarget) {\n return mapInternalTexture2MainIndex;\n }\n const internalTextures = this._renderTarget.textures;\n for (let i = 0; i < internalTextures.length; i++) {\n const texture = internalTextures[i];\n if (!texture) {\n continue;\n }\n const mainIndex = mapMainInternalTexture2Index[texture.uniqueId];\n if (mainIndex !== undefined) {\n mapInternalTexture2MainIndex[i] = mainIndex;\n }\n else {\n mapMainInternalTexture2Index[texture.uniqueId] = i;\n }\n }\n return mapInternalTexture2MainIndex;\n }\n /**\n * @internal\n */\n _rebuild(forceFullRebuild = false, textureNames) {\n if (this._count < 1) {\n return;\n }\n const mapInternalTexture2MainIndex = this._createInternaTextureIndexMapping();\n this.releaseInternalTextures();\n this._createInternalTextures();\n if (forceFullRebuild) {\n this._releaseTextures();\n this._createTextures(textureNames);\n }\n const internalTextures = this._renderTarget.textures;\n for (let i = 0; i < internalTextures.length; i++) {\n const texture = this._textures[i];\n if (mapInternalTexture2MainIndex[i] !== undefined) {\n this._renderTarget.setTexture(internalTextures[mapInternalTexture2MainIndex[i]], i);\n }\n texture._texture = internalTextures[i];\n if (texture._texture) {\n texture._noMipmap = !texture._texture.useMipMaps;\n texture._useSRGBBuffer = texture._texture._useSRGBBuffer;\n }\n }\n if (this.samples !== 1) {\n this._renderTarget.setSamples(this.samples, !this._drawOnlyOnFirstAttachmentByDefault, true);\n }\n }\n _createInternalTextures() {\n this._renderTarget = this._getEngine().createMultipleRenderTarget(this._size, this._multiRenderTargetOptions, !this._drawOnlyOnFirstAttachmentByDefault);\n this._texture = this._renderTarget.texture;\n }\n _releaseTextures() {\n if (this._textures) {\n for (let i = 0; i < this._textures.length; i++) {\n this._textures[i]._texture = null; // internal textures are released by a call to releaseInternalTextures()\n this._textures[i].dispose();\n }\n }\n }\n _createTextures(textureNames) {\n const internalTextures = this._renderTarget.textures;\n this._textures = [];\n for (let i = 0; i < internalTextures.length; i++) {\n const texture = new Texture(null, this.getScene());\n if (textureNames === null || textureNames === void 0 ? void 0 : textureNames[i]) {\n texture.name = textureNames[i];\n }\n texture._texture = internalTextures[i];\n if (texture._texture) {\n texture._noMipmap = !texture._texture.useMipMaps;\n texture._useSRGBBuffer = texture._texture._useSRGBBuffer;\n }\n this._textures.push(texture);\n }\n }\n /**\n * Replaces an internal texture within the MRT. Useful to share textures between MultiRenderTarget.\n * @param texture The new texture to set in the MRT\n * @param index The index of the texture to replace\n * @param disposePrevious Set to true if the previous internal texture should be disposed\n */\n setInternalTexture(texture, index, disposePrevious = true) {\n var _a, _b;\n if (!this.renderTarget) {\n return;\n }\n if (index === 0) {\n this._texture = texture;\n }\n this.renderTarget.setTexture(texture, index, disposePrevious);\n if (!this.textures[index]) {\n this.textures[index] = new Texture(null, this.getScene());\n this.textures[index].name = (_b = (_a = this._textureNames) === null || _a === void 0 ? void 0 : _a[index]) !== null && _b !== void 0 ? _b : this.textures[index].name;\n }\n this.textures[index]._texture = texture;\n this.textures[index]._noMipmap = !texture.useMipMaps;\n this.textures[index]._useSRGBBuffer = texture._useSRGBBuffer;\n this._count = this.renderTarget.textures ? this.renderTarget.textures.length : 0;\n if (this._multiRenderTargetOptions.types) {\n this._multiRenderTargetOptions.types[index] = texture.type;\n }\n if (this._multiRenderTargetOptions.samplingModes) {\n this._multiRenderTargetOptions.samplingModes[index] = texture.samplingMode;\n }\n if (this._multiRenderTargetOptions.useSRGBBuffers) {\n this._multiRenderTargetOptions.useSRGBBuffers[index] = texture._useSRGBBuffer;\n }\n if (this._multiRenderTargetOptions.targetTypes && this._multiRenderTargetOptions.targetTypes[index] !== -1) {\n let target = 0;\n if (texture.is2DArray) {\n target = 35866;\n }\n else if (texture.isCube) {\n target = 34067;\n } /*else if (texture.isCubeArray) {\n target = 3735928559;\n }*/\n else if (texture.is3D) {\n target = 32879;\n }\n else {\n target = 3553;\n }\n this._multiRenderTargetOptions.targetTypes[index] = target;\n }\n }\n /**\n * Changes an attached texture's face index or layer.\n * @param index The index of the texture to modify the attachment of\n * @param layerIndex The layer index of the texture to be attached to the framebuffer\n * @param faceIndex The face index of the texture to be attached to the framebuffer\n */\n setLayerAndFaceIndex(index, layerIndex = -1, faceIndex = -1) {\n if (!this.textures[index] || !this.renderTarget) {\n return;\n }\n if (this._multiRenderTargetOptions.layerIndex) {\n this._multiRenderTargetOptions.layerIndex[index] = layerIndex;\n }\n if (this._multiRenderTargetOptions.faceIndex) {\n this._multiRenderTargetOptions.faceIndex[index] = faceIndex;\n }\n this.renderTarget.setLayerAndFaceIndex(index, layerIndex, faceIndex);\n }\n /**\n * Changes every attached texture's face index or layer.\n * @param layerIndices The layer indices of the texture to be attached to the framebuffer\n * @param faceIndices The face indices of the texture to be attached to the framebuffer\n */\n setLayerAndFaceIndices(layerIndices, faceIndices) {\n if (!this.renderTarget) {\n return;\n }\n this._multiRenderTargetOptions.layerIndex = layerIndices;\n this._multiRenderTargetOptions.faceIndex = faceIndices;\n this.renderTarget.setLayerAndFaceIndices(layerIndices, faceIndices);\n }\n /**\n * Define the number of samples used if MSAA is enabled.\n */\n get samples() {\n return this._samples;\n }\n set samples(value) {\n if (this._renderTarget) {\n this._samples = this._renderTarget.setSamples(value);\n }\n else {\n // In case samples are set with 0 textures created, we must save the desired samples value\n this._samples = value;\n }\n }\n /**\n * Resize all the textures in the multi render target.\n * Be careful as it will recreate all the data in the new texture.\n * @param size Define the new size\n */\n resize(size) {\n this._processSizeParameter(size, false);\n this._rebuild(undefined, this._textureNames);\n }\n /**\n * Changes the number of render targets in this MRT\n * Be careful as it will recreate all the data in the new texture.\n * @param count new texture count\n * @param options Specifies texture types and sampling modes for new textures\n * @param textureNames Specifies the names of the textures (optional)\n */\n updateCount(count, options, textureNames) {\n this._multiRenderTargetOptions.textureCount = count;\n this._count = count;\n const types = [];\n const samplingModes = [];\n const useSRGBBuffers = [];\n const formats = [];\n const targetTypes = [];\n const faceIndex = [];\n const layerIndex = [];\n const layerCounts = [];\n this._textureNames = textureNames;\n this._initTypes(count, types, samplingModes, useSRGBBuffers, formats, targetTypes, faceIndex, layerIndex, layerCounts, options);\n this._multiRenderTargetOptions.types = types;\n this._multiRenderTargetOptions.samplingModes = samplingModes;\n this._multiRenderTargetOptions.useSRGBBuffers = useSRGBBuffers;\n this._multiRenderTargetOptions.formats = formats;\n this._multiRenderTargetOptions.targetTypes = targetTypes;\n this._multiRenderTargetOptions.faceIndex = faceIndex;\n this._multiRenderTargetOptions.layerIndex = layerIndex;\n this._multiRenderTargetOptions.layerCounts = layerCounts;\n this._rebuild(true, textureNames);\n }\n _unbindFrameBuffer(engine, faceIndex) {\n if (this._renderTarget) {\n engine.unBindMultiColorAttachmentFramebuffer(this._renderTarget, this.isCube, () => {\n this.onAfterRenderObservable.notifyObservers(faceIndex);\n });\n }\n }\n /**\n * Dispose the render targets and their associated resources\n * @param doNotDisposeInternalTextures\n */\n dispose(doNotDisposeInternalTextures = false) {\n this._releaseTextures();\n if (!doNotDisposeInternalTextures) {\n this.releaseInternalTextures();\n }\n else {\n // Prevent internal texture dispose in super.dispose\n this._texture = null;\n }\n super.dispose();\n }\n /**\n * Release all the underlying texture used as draw buffers.\n */\n releaseInternalTextures() {\n var _a, _b;\n const internalTextures = (_a = this._renderTarget) === null || _a === void 0 ? void 0 : _a.textures;\n if (!internalTextures) {\n return;\n }\n for (let i = internalTextures.length - 1; i >= 0; i--) {\n this._textures[i]._texture = null;\n }\n (_b = this._renderTarget) === null || _b === void 0 ? void 0 : _b.dispose();\n this._renderTarget = null;\n }\n }\n\n // Do not edit.\n const name$2j = \"mrtFragmentDeclaration\";\n const shader$2j = `#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE)\nlayout(location=0) out vec4 glFragData[{X}];\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$2j] = shader$2j;\n\n // Do not edit.\n const name$2k = \"geometryPixelShader\";\n const shader$2k = `#extension GL_EXT_draw_buffers : require\n#if defined(BUMP) || !defined(NORMAL)\n#extension GL_OES_standard_derivatives : enable\n#endif\nprecision highp float;\n#ifdef BUMP\nvarying mat4 vWorldView;varying vec3 vNormalW;\n#else\nvarying vec3 vNormalV;\n#endif\nvarying vec4 vViewPos;\n#if defined(POSITION) || defined(BUMP)\nvarying vec3 vPositionW;\n#endif\n#ifdef VELOCITY\nvarying vec4 vCurrentPosition;varying vec4 vPreviousPosition;\n#endif\n#ifdef NEED_UV\nvarying vec2 vUV;\n#endif\n#ifdef BUMP\nuniform vec3 vBumpInfos;uniform vec2 vTangentSpaceParams;\n#endif\n#if defined(REFLECTIVITY)\n#if defined(ORMTEXTURE) || defined(SPECULARGLOSSINESSTEXTURE) || defined(REFLECTIVITYTEXTURE)\nuniform sampler2D reflectivitySampler;varying vec2 vReflectivityUV;\n#endif\n#ifdef ALBEDOTEXTURE\nvarying vec2 vAlbedoUV;uniform sampler2D albedoSampler;\n#endif\n#ifdef REFLECTIVITYCOLOR\nuniform vec3 reflectivityColor;\n#endif\n#ifdef ALBEDOCOLOR\nuniform vec3 albedoColor;\n#endif\n#ifdef METALLIC\nuniform float metallic;\n#endif\n#if defined(ROUGHNESS) || defined(GLOSSINESS)\nuniform float glossiness;\n#endif\n#endif\n#if defined(ALPHATEST) && defined(NEED_UV)\nuniform sampler2D diffuseSampler;\n#endif\n#include\n#include[RENDER_TARGET_COUNT]\n#include\n#include\n#include\nvoid main() {\n#include\n#ifdef ALPHATEST\nif (texture2D(diffuseSampler,vUV).a<0.4)\ndiscard;\n#endif\nvec3 normalOutput;\n#ifdef BUMP\nvec3 normalW=normalize(vNormalW);\n#include\nnormalOutput=normalize(vec3(vWorldView*vec4(normalW,0.0)));\n#else\nnormalOutput=normalize(vNormalV);\n#endif\n#ifdef PREPASS\n#ifdef PREPASS_DEPTH\ngl_FragData[DEPTH_INDEX]=vec4(vViewPos.z/vViewPos.w,0.0,0.0,1.0);\n#endif\n#ifdef PREPASS_NORMAL\ngl_FragData[NORMAL_INDEX]=vec4(normalOutput,1.0);\n#endif\n#else\ngl_FragData[0]=vec4(vViewPos.z/vViewPos.w,0.0,0.0,1.0);gl_FragData[1]=vec4(normalOutput,1.0);\n#endif\n#ifdef POSITION\ngl_FragData[POSITION_INDEX]=vec4(vPositionW,1.0);\n#endif\n#ifdef VELOCITY\nvec2 a=(vCurrentPosition.xy/vCurrentPosition.w)*0.5+0.5;vec2 b=(vPreviousPosition.xy/vPreviousPosition.w)*0.5+0.5;vec2 velocity=abs(a-b);velocity=vec2(pow(velocity.x,1.0/3.0),pow(velocity.y,1.0/3.0))*sign(a-b)*0.5+0.5;gl_FragData[VELOCITY_INDEX]=vec4(velocity,0.0,1.0);\n#endif\n#ifdef REFLECTIVITY\nvec4 reflectivity=vec4(0.0,0.0,0.0,1.0);\n#ifdef METALLICWORKFLOW\nfloat metal=1.0;float roughness=1.0;\n#ifdef ORMTEXTURE\nmetal*=texture2D(reflectivitySampler,vReflectivityUV).b;roughness*=texture2D(reflectivitySampler,vReflectivityUV).g;\n#endif\n#ifdef METALLIC\nmetal*=metallic;\n#endif\n#ifdef ROUGHNESS\nroughness*=(1.0-glossiness); \n#endif\nreflectivity.a-=roughness;vec3 color=vec3(1.0);\n#ifdef ALBEDOTEXTURE\ncolor=texture2D(albedoSampler,vAlbedoUV).rgb;\n#ifdef GAMMAALBEDO\ncolor=toLinearSpace(color);\n#endif\n#endif\n#ifdef ALBEDOCOLOR\ncolor*=albedoColor.xyz;\n#endif\nreflectivity.rgb=mix(vec3(0.04),color,metal);\n#else\n#if defined(SPECULARGLOSSINESSTEXTURE) || defined(REFLECTIVITYTEXTURE)\nreflectivity=texture2D(reflectivitySampler,vReflectivityUV);\n#ifdef GAMMAREFLECTIVITYTEXTURE\nreflectivity.rgb=toLinearSpace(reflectivity.rgb);\n#endif\n#else \n#ifdef REFLECTIVITYCOLOR\nreflectivity.rgb=toLinearSpace(reflectivityColor.xyz);reflectivity.a=1.0;\n#endif\n#endif\n#ifdef GLOSSINESSS\nreflectivity.a*=glossiness; \n#endif\n#endif\ngl_FragData[REFLECTIVITY_INDEX]=reflectivity;\n#endif\n}\n`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2k] = shader$2k;\n\n // Do not edit.\n const name$2l = \"geometryVertexDeclaration\";\n const shader$2l = `uniform mat4 viewProjection;uniform mat4 view;`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$2l] = shader$2l;\n\n // Do not edit.\n const name$2m = \"geometryUboDeclaration\";\n const shader$2m = `#include\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$2m] = shader$2m;\n\n // Do not edit.\n const name$2n = \"geometryVertexShader\";\n const shader$2n = `precision highp float;\n#include\n#include\n#include\n#include[0..maxSimultaneousMorphTargets]\n#include\n#include<__decl__geometryVertex>\n#include\nattribute vec3 position;attribute vec3 normal;\n#ifdef NEED_UV\nvarying vec2 vUV;\n#ifdef ALPHATEST\nuniform mat4 diffuseMatrix;\n#endif\n#ifdef BUMP\nuniform mat4 bumpMatrix;varying vec2 vBumpUV;\n#endif\n#ifdef REFLECTIVITY\nuniform mat4 reflectivityMatrix;uniform mat4 albedoMatrix;varying vec2 vReflectivityUV;varying vec2 vAlbedoUV;\n#endif\n#ifdef UV1\nattribute vec2 uv;\n#endif\n#ifdef UV2\nattribute vec2 uv2;\n#endif\n#endif\n#ifdef BUMP\nvarying mat4 vWorldView;\n#endif\n#ifdef BUMP\nvarying vec3 vNormalW;\n#else\nvarying vec3 vNormalV;\n#endif\nvarying vec4 vViewPos;\n#if defined(POSITION) || defined(BUMP)\nvarying vec3 vPositionW;\n#endif\n#ifdef VELOCITY\nuniform mat4 previousViewProjection;varying vec4 vCurrentPosition;varying vec4 vPreviousPosition;\n#endif\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void)\n{vec3 positionUpdated=position;vec3 normalUpdated=normal;\n#ifdef UV1\nvec2 uvUpdated=uv;\n#endif\n#include\n#include[0..maxSimultaneousMorphTargets]\n#include\n#if defined(VELOCITY) && !defined(BONES_VELOCITY_ENABLED)\nvCurrentPosition=viewProjection*finalWorld*vec4(positionUpdated,1.0);vPreviousPosition=previousViewProjection*finalPreviousWorld*vec4(positionUpdated,1.0);\n#endif\n#include\n#include\nvec4 worldPos=vec4(finalWorld*vec4(positionUpdated,1.0));\n#ifdef BUMP\nvWorldView=view*finalWorld;vNormalW=normalUpdated;\n#else\nvNormalV=normalize(vec3((view*finalWorld)*vec4(normalUpdated,0.0)));\n#endif\nvViewPos=view*worldPos;\n#if defined(VELOCITY) && defined(BONES_VELOCITY_ENABLED)\nvCurrentPosition=viewProjection*finalWorld*vec4(positionUpdated,1.0);\n#if NUM_BONE_INFLUENCERS>0\nmat4 previousInfluence;previousInfluence=mPreviousBones[int(matricesIndices[0])]*matricesWeights[0];\n#if NUM_BONE_INFLUENCERS>1\npreviousInfluence+=mPreviousBones[int(matricesIndices[1])]*matricesWeights[1];\n#endif\n#if NUM_BONE_INFLUENCERS>2\npreviousInfluence+=mPreviousBones[int(matricesIndices[2])]*matricesWeights[2];\n#endif\n#if NUM_BONE_INFLUENCERS>3\npreviousInfluence+=mPreviousBones[int(matricesIndices[3])]*matricesWeights[3];\n#endif\n#if NUM_BONE_INFLUENCERS>4\npreviousInfluence+=mPreviousBones[int(matricesIndicesExtra[0])]*matricesWeightsExtra[0];\n#endif\n#if NUM_BONE_INFLUENCERS>5\npreviousInfluence+=mPreviousBones[int(matricesIndicesExtra[1])]*matricesWeightsExtra[1];\n#endif\n#if NUM_BONE_INFLUENCERS>6\npreviousInfluence+=mPreviousBones[int(matricesIndicesExtra[2])]*matricesWeightsExtra[2];\n#endif\n#if NUM_BONE_INFLUENCERS>7\npreviousInfluence+=mPreviousBones[int(matricesIndicesExtra[3])]*matricesWeightsExtra[3];\n#endif\nvPreviousPosition=previousViewProjection*finalPreviousWorld*previousInfluence*vec4(positionUpdated,1.0);\n#else\nvPreviousPosition=previousViewProjection*finalPreviousWorld*vec4(positionUpdated,1.0);\n#endif\n#endif\n#if defined(POSITION) || defined(BUMP)\nvPositionW=worldPos.xyz/worldPos.w;\n#endif\ngl_Position=viewProjection*finalWorld*vec4(positionUpdated,1.0);\n#include\n#ifdef NEED_UV\n#ifdef UV1\n#if defined(ALPHATEST) && defined(ALPHATEST_UV1)\nvUV=vec2(diffuseMatrix*vec4(uvUpdated,1.0,0.0));\n#else\nvUV=uv;\n#endif\n#ifdef BUMP_UV1\nvBumpUV=vec2(bumpMatrix*vec4(uvUpdated,1.0,0.0));\n#endif\n#ifdef REFLECTIVITY_UV1\nvReflectivityUV=vec2(reflectivityMatrix*vec4(uvUpdated,1.0,0.0));\n#endif\n#ifdef ALBEDO_UV1\nvAlbedoUV=vec2(albedoMatrix*vec4(uvUpdated,1.0,0.0));\n#endif\n#endif\n#ifdef UV2\n#if defined(ALPHATEST) && defined(ALPHATEST_UV2)\nvUV=vec2(diffuseMatrix*vec4(uv2,1.0,0.0));\n#else\nvUV=uv2;\n#endif\n#ifdef BUMP_UV2\nvBumpUV=vec2(bumpMatrix*vec4(uv2,1.0,0.0));\n#endif\n#ifdef REFLECTIVITY_UV2\nvReflectivityUV=vec2(reflectivityMatrix*vec4(uv2,1.0,0.0));\n#endif\n#ifdef ALBEDO_UV2\nvAlbedoUV=vec2(albedoMatrix*vec4(uv2,1.0,0.0));\n#endif\n#endif\n#endif\n#include\n}\n`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2n] = shader$2n;\n\n /** list the uniforms used by the geometry renderer */\n const uniforms = [\n \"world\",\n \"mBones\",\n \"viewProjection\",\n \"diffuseMatrix\",\n \"view\",\n \"previousWorld\",\n \"previousViewProjection\",\n \"mPreviousBones\",\n \"bumpMatrix\",\n \"reflectivityMatrix\",\n \"albedoMatrix\",\n \"reflectivityColor\",\n \"albedoColor\",\n \"metallic\",\n \"glossiness\",\n \"vTangentSpaceParams\",\n \"vBumpInfos\",\n \"morphTargetInfluences\",\n \"morphTargetTextureInfo\",\n \"morphTargetTextureIndices\",\n ];\n addClipPlaneUniforms(uniforms);\n /**\n * This renderer is helpful to fill one of the render target with a geometry buffer.\n */\n class GeometryBufferRenderer {\n /**\n * @internal\n * Sets up internal structures to share outputs with PrePassRenderer\n * This method should only be called by the PrePassRenderer itself\n */\n _linkPrePassRenderer(prePassRenderer) {\n this._linkedWithPrePass = true;\n this._prePassRenderer = prePassRenderer;\n if (this._multiRenderTarget) {\n // prevents clearing of the RT since it's done by prepass\n this._multiRenderTarget.onClearObservable.clear();\n this._multiRenderTarget.onClearObservable.add(() => {\n // pass\n });\n }\n }\n /**\n * @internal\n * Separates internal structures from PrePassRenderer so the geometry buffer can now operate by itself.\n * This method should only be called by the PrePassRenderer itself\n */\n _unlinkPrePassRenderer() {\n this._linkedWithPrePass = false;\n this._createRenderTargets();\n }\n /**\n * @internal\n * Resets the geometry buffer layout\n */\n _resetLayout() {\n this._enablePosition = false;\n this._enableReflectivity = false;\n this._enableVelocity = false;\n this._attachmentsFromPrePass = [];\n }\n /**\n * @internal\n * Replaces a texture in the geometry buffer renderer\n * Useful when linking textures of the prepass renderer\n */\n _forceTextureType(geometryBufferType, index) {\n if (geometryBufferType === GeometryBufferRenderer.POSITION_TEXTURE_TYPE) {\n this._positionIndex = index;\n this._enablePosition = true;\n }\n else if (geometryBufferType === GeometryBufferRenderer.VELOCITY_TEXTURE_TYPE) {\n this._velocityIndex = index;\n this._enableVelocity = true;\n }\n else if (geometryBufferType === GeometryBufferRenderer.REFLECTIVITY_TEXTURE_TYPE) {\n this._reflectivityIndex = index;\n this._enableReflectivity = true;\n }\n else if (geometryBufferType === GeometryBufferRenderer.DEPTH_TEXTURE_TYPE) {\n this._depthIndex = index;\n }\n else if (geometryBufferType === GeometryBufferRenderer.NORMAL_TEXTURE_TYPE) {\n this._normalIndex = index;\n }\n }\n /**\n * @internal\n * Sets texture attachments\n * Useful when linking textures of the prepass renderer\n */\n _setAttachments(attachments) {\n this._attachmentsFromPrePass = attachments;\n }\n /**\n * @internal\n * Replaces the first texture which is hard coded as a depth texture in the geometry buffer\n * Useful when linking textures of the prepass renderer\n */\n _linkInternalTexture(internalTexture) {\n this._multiRenderTarget.setInternalTexture(internalTexture, 0, false);\n }\n /**\n * Gets the render list (meshes to be rendered) used in the G buffer.\n */\n get renderList() {\n return this._multiRenderTarget.renderList;\n }\n /**\n * Set the render list (meshes to be rendered) used in the G buffer.\n */\n set renderList(meshes) {\n this._multiRenderTarget.renderList = meshes;\n }\n /**\n * Gets whether or not G buffer are supported by the running hardware.\n * This requires draw buffer supports\n */\n get isSupported() {\n return this._multiRenderTarget.isSupported;\n }\n /**\n * Returns the index of the given texture type in the G-Buffer textures array\n * @param textureType The texture type constant. For example GeometryBufferRenderer.POSITION_TEXTURE_INDEX\n * @returns the index of the given texture type in the G-Buffer textures array\n */\n getTextureIndex(textureType) {\n switch (textureType) {\n case GeometryBufferRenderer.POSITION_TEXTURE_TYPE:\n return this._positionIndex;\n case GeometryBufferRenderer.VELOCITY_TEXTURE_TYPE:\n return this._velocityIndex;\n case GeometryBufferRenderer.REFLECTIVITY_TEXTURE_TYPE:\n return this._reflectivityIndex;\n default:\n return -1;\n }\n }\n /**\n * Gets a boolean indicating if objects positions are enabled for the G buffer.\n */\n get enablePosition() {\n return this._enablePosition;\n }\n /**\n * Sets whether or not objects positions are enabled for the G buffer.\n */\n set enablePosition(enable) {\n this._enablePosition = enable;\n // PrePass handles index and texture links\n if (!this._linkedWithPrePass) {\n this.dispose();\n this._createRenderTargets();\n }\n }\n /**\n * Gets a boolean indicating if objects velocities are enabled for the G buffer.\n */\n get enableVelocity() {\n return this._enableVelocity;\n }\n /**\n * Sets whether or not objects velocities are enabled for the G buffer.\n */\n set enableVelocity(enable) {\n this._enableVelocity = enable;\n if (!enable) {\n this._previousTransformationMatrices = {};\n }\n if (!this._linkedWithPrePass) {\n this.dispose();\n this._createRenderTargets();\n }\n this._scene.needsPreviousWorldMatrices = enable;\n }\n /**\n * Gets a boolean indicating if objects reflectivity are enabled in the G buffer.\n */\n get enableReflectivity() {\n return this._enableReflectivity;\n }\n /**\n * Sets whether or not objects reflectivity are enabled for the G buffer.\n * For Metallic-Roughness workflow with ORM texture, we assume that ORM texture is defined according to the default layout:\n * pbr.useRoughnessFromMetallicTextureAlpha = false;\n * pbr.useRoughnessFromMetallicTextureGreen = true;\n * pbr.useMetallnessFromMetallicTextureBlue = true;\n */\n set enableReflectivity(enable) {\n this._enableReflectivity = enable;\n if (!this._linkedWithPrePass) {\n this.dispose();\n this._createRenderTargets();\n }\n }\n /**\n * Gets the scene associated with the buffer.\n */\n get scene() {\n return this._scene;\n }\n /**\n * Gets the ratio used by the buffer during its creation.\n * How big is the buffer related to the main canvas.\n */\n get ratio() {\n return this._ratio;\n }\n /**\n * Creates a new G Buffer for the scene\n * @param scene The scene the buffer belongs to\n * @param ratio How big is the buffer related to the main canvas (default: 1)\n * @param depthFormat Format of the depth texture (default: 15)\n */\n constructor(scene, ratio = 1, depthFormat = 15) {\n /**\n * Dictionary used to store the previous transformation matrices of each rendered mesh\n * in order to compute objects velocities when enableVelocity is set to \"true\"\n * @internal\n */\n this._previousTransformationMatrices = {};\n /**\n * Dictionary used to store the previous bones transformation matrices of each rendered mesh\n * in order to compute objects velocities when enableVelocity is set to \"true\"\n * @internal\n */\n this._previousBonesTransformationMatrices = {};\n /**\n * Array used to store the ignored skinned meshes while computing velocity map (typically used by the motion blur post-process).\n * Avoids computing bones velocities and computes only mesh's velocity itself (position, rotation, scaling).\n */\n this.excludedSkinnedMeshesFromVelocity = [];\n /** Gets or sets a boolean indicating if transparent meshes should be rendered */\n this.renderTransparentMeshes = true;\n this._resizeObserver = null;\n this._enablePosition = false;\n this._enableVelocity = false;\n this._enableReflectivity = false;\n this._clearColor = new Color4(0, 0, 0, 0);\n this._clearDepthColor = new Color4(1e8, 0, 0, 1); // \"infinity\" value - depth in the depth texture is view.z, not a 0..1 value!\n this._positionIndex = -1;\n this._velocityIndex = -1;\n this._reflectivityIndex = -1;\n this._depthIndex = -1;\n this._normalIndex = -1;\n this._linkedWithPrePass = false;\n /**\n * If set to true (default: false), the depth texture will be cleared with the depth value corresponding to the far plane (1 in normal mode, 0 in reverse depth buffer mode)\n * If set to false, the depth texture is always cleared with 0.\n */\n this.useSpecificClearForDepthTexture = false;\n this._scene = scene;\n this._ratio = ratio;\n this._useUbo = scene.getEngine().supportsUniformBuffers;\n this._depthFormat = depthFormat;\n GeometryBufferRenderer._SceneComponentInitialization(this._scene);\n // Render target\n this._createRenderTargets();\n }\n /**\n * Checks whether everything is ready to render a submesh to the G buffer.\n * @param subMesh the submesh to check readiness for\n * @param useInstances is the mesh drawn using instance or not\n * @returns true if ready otherwise false\n */\n isReady(subMesh, useInstances) {\n const material = subMesh.getMaterial();\n if (material && material.disableDepthWrite) {\n return false;\n }\n const defines = [];\n const attribs = [VertexBuffer.PositionKind, VertexBuffer.NormalKind];\n const mesh = subMesh.getMesh();\n // Alpha test\n if (material) {\n let needUv = false;\n if (material.needAlphaTesting() && material.getAlphaTestTexture()) {\n defines.push(\"#define ALPHATEST\");\n defines.push(`#define ALPHATEST_UV${material.getAlphaTestTexture().coordinatesIndex + 1}`);\n needUv = true;\n }\n if (material.bumpTexture && MaterialFlags.BumpTextureEnabled) {\n defines.push(\"#define BUMP\");\n defines.push(`#define BUMP_UV${material.bumpTexture.coordinatesIndex + 1}`);\n needUv = true;\n }\n if (this._enableReflectivity) {\n let metallicWorkflow = false;\n // for PBR materials: cf. https://doc.babylonjs.com/features/featuresDeepDive/materials/using/masterPBR\n if (material.getClassName() === \"PBRMetallicRoughnessMaterial\") {\n // if it is a PBR material in MetallicRoughness Mode:\n if (material.metallicRoughnessTexture !== null) {\n defines.push(\"#define ORMTEXTURE\");\n defines.push(`#define REFLECTIVITY_UV${material.metallicRoughnessTexture.coordinatesIndex + 1}`);\n defines.push(\"#define METALLICWORKFLOW\");\n needUv = true;\n metallicWorkflow = true;\n }\n if (material.metallic !== null) {\n defines.push(\"#define METALLIC\");\n defines.push(\"#define METALLICWORKFLOW\");\n metallicWorkflow = true;\n }\n if (material.roughness !== null) {\n defines.push(\"#define ROUGHNESS\");\n defines.push(\"#define METALLICWORKFLOW\");\n metallicWorkflow = true;\n }\n if (metallicWorkflow) {\n if (material.baseTexture !== null) {\n defines.push(\"#define ALBEDOTEXTURE\");\n defines.push(`#define ALBEDO_UV${material.baseTexture.coordinatesIndex + 1}`);\n if (material.baseTexture.gammaSpace) {\n defines.push(\"#define GAMMAALBEDO\");\n }\n needUv = true;\n }\n if (material.baseColor !== null) {\n defines.push(\"#define ALBEDOCOLOR\");\n }\n }\n }\n else if (material.getClassName() === \"PBRSpecularGlossinessMaterial\") {\n // if it is a PBR material in Specular/Glossiness Mode:\n if (material.specularGlossinessTexture !== null) {\n defines.push(\"#define SPECULARGLOSSINESSTEXTURE\");\n defines.push(`#define REFLECTIVITY_UV${material.specularGlossinessTexture.coordinatesIndex + 1}`);\n needUv = true;\n if (material.specularGlossinessTexture.gammaSpace) {\n defines.push(\"#define GAMMAREFLECTIVITYTEXTURE\");\n }\n }\n else {\n if (material.specularColor !== null) {\n defines.push(\"#define REFLECTIVITYCOLOR\");\n }\n }\n if (material.glossiness !== null) {\n defines.push(\"#define GLOSSINESS\");\n }\n }\n else if (material.getClassName() === \"PBRMaterial\") {\n // if it is the bigger PBRMaterial\n if (material.metallicTexture !== null) {\n defines.push(\"#define ORMTEXTURE\");\n defines.push(`#define REFLECTIVITY_UV${material.metallicTexture.coordinatesIndex + 1}`);\n defines.push(\"#define METALLICWORKFLOW\");\n needUv = true;\n metallicWorkflow = true;\n }\n if (material.metallic !== null) {\n defines.push(\"#define METALLIC\");\n defines.push(\"#define METALLICWORKFLOW\");\n metallicWorkflow = true;\n }\n if (material.roughness !== null) {\n defines.push(\"#define ROUGHNESS\");\n defines.push(\"#define METALLICWORKFLOW\");\n metallicWorkflow = true;\n }\n if (metallicWorkflow) {\n if (material.albedoTexture !== null) {\n defines.push(\"#define ALBEDOTEXTURE\");\n defines.push(`#define ALBEDO_UV${material.albedoTexture.coordinatesIndex + 1}`);\n if (material.albedoTexture.gammaSpace) {\n defines.push(\"#define GAMMAALBEDO\");\n }\n needUv = true;\n }\n if (material.albedoColor !== null) {\n defines.push(\"#define ALBEDOCOLOR\");\n }\n }\n else {\n // SpecularGlossiness Model\n if (material.reflectivityTexture !== null) {\n defines.push(\"#define SPECULARGLOSSINESSTEXTURE\");\n defines.push(`#define REFLECTIVITY_UV${material.reflectivityTexture.coordinatesIndex + 1}`);\n if (material.reflectivityTexture.gammaSpace) {\n defines.push(\"#define GAMMAREFLECTIVITYTEXTURE\");\n }\n needUv = true;\n }\n else if (material.reflectivityColor !== null) {\n defines.push(\"#define REFLECTIVITYCOLOR\");\n }\n if (material.microSurface !== null) {\n defines.push(\"#define GLOSSINESS\");\n }\n }\n }\n else if (material.getClassName() === \"StandardMaterial\") {\n // if StandardMaterial:\n if (material.specularTexture !== null) {\n defines.push(\"#define REFLECTIVITYTEXTURE\");\n defines.push(`#define REFLECTIVITY_UV${material.specularTexture.coordinatesIndex + 1}`);\n if (material.specularTexture.gammaSpace) {\n defines.push(\"#define GAMMAREFLECTIVITYTEXTURE\");\n }\n needUv = true;\n }\n if (material.specularColor !== null) {\n defines.push(\"#define REFLECTIVITYCOLOR\");\n }\n }\n }\n if (needUv) {\n defines.push(\"#define NEED_UV\");\n if (mesh.isVerticesDataPresent(VertexBuffer.UVKind)) {\n attribs.push(VertexBuffer.UVKind);\n defines.push(\"#define UV1\");\n }\n if (mesh.isVerticesDataPresent(VertexBuffer.UV2Kind)) {\n attribs.push(VertexBuffer.UV2Kind);\n defines.push(\"#define UV2\");\n }\n }\n }\n // PrePass\n if (this._linkedWithPrePass) {\n defines.push(\"#define PREPASS\");\n if (this._depthIndex !== -1) {\n defines.push(\"#define DEPTH_INDEX \" + this._depthIndex);\n defines.push(\"#define PREPASS_DEPTH\");\n }\n if (this._normalIndex !== -1) {\n defines.push(\"#define NORMAL_INDEX \" + this._normalIndex);\n defines.push(\"#define PREPASS_NORMAL\");\n }\n }\n // Buffers\n if (this._enablePosition) {\n defines.push(\"#define POSITION\");\n defines.push(\"#define POSITION_INDEX \" + this._positionIndex);\n }\n if (this._enableVelocity) {\n defines.push(\"#define VELOCITY\");\n defines.push(\"#define VELOCITY_INDEX \" + this._velocityIndex);\n if (this.excludedSkinnedMeshesFromVelocity.indexOf(mesh) === -1) {\n defines.push(\"#define BONES_VELOCITY_ENABLED\");\n }\n }\n if (this._enableReflectivity) {\n defines.push(\"#define REFLECTIVITY\");\n defines.push(\"#define REFLECTIVITY_INDEX \" + this._reflectivityIndex);\n }\n // Bones\n if (mesh.useBones && mesh.computeBonesUsingShaders) {\n attribs.push(VertexBuffer.MatricesIndicesKind);\n attribs.push(VertexBuffer.MatricesWeightsKind);\n if (mesh.numBoneInfluencers > 4) {\n attribs.push(VertexBuffer.MatricesIndicesExtraKind);\n attribs.push(VertexBuffer.MatricesWeightsExtraKind);\n }\n defines.push(\"#define NUM_BONE_INFLUENCERS \" + mesh.numBoneInfluencers);\n defines.push(\"#define BonesPerMesh \" + (mesh.skeleton ? mesh.skeleton.bones.length + 1 : 0));\n }\n else {\n defines.push(\"#define NUM_BONE_INFLUENCERS 0\");\n }\n // Morph targets\n const morphTargetManager = mesh.morphTargetManager;\n let numMorphInfluencers = 0;\n if (morphTargetManager) {\n if (morphTargetManager.numInfluencers > 0) {\n numMorphInfluencers = morphTargetManager.numInfluencers;\n defines.push(\"#define MORPHTARGETS\");\n defines.push(\"#define NUM_MORPH_INFLUENCERS \" + numMorphInfluencers);\n if (morphTargetManager.isUsingTextureForTargets) {\n defines.push(\"#define MORPHTARGETS_TEXTURE\");\n }\n MaterialHelper.PrepareAttributesForMorphTargetsInfluencers(attribs, mesh, numMorphInfluencers);\n }\n }\n // Instances\n if (useInstances) {\n defines.push(\"#define INSTANCES\");\n MaterialHelper.PushAttributesForInstances(attribs, this._enableVelocity);\n if (subMesh.getRenderingMesh().hasThinInstances) {\n defines.push(\"#define THIN_INSTANCES\");\n }\n }\n // Setup textures count\n if (this._linkedWithPrePass) {\n defines.push(\"#define RENDER_TARGET_COUNT \" + this._attachmentsFromPrePass.length);\n }\n else {\n defines.push(\"#define RENDER_TARGET_COUNT \" + this._multiRenderTarget.textures.length);\n }\n prepareStringDefinesForClipPlanes(material, this._scene, defines);\n // Get correct effect\n const engine = this._scene.getEngine();\n const drawWrapper = subMesh._getDrawWrapper(undefined, true);\n const cachedDefines = drawWrapper.defines;\n const join = defines.join(\"\\n\");\n if (cachedDefines !== join) {\n drawWrapper.setEffect(engine.createEffect(\"geometry\", {\n attributes: attribs,\n uniformsNames: uniforms,\n samplers: [\"diffuseSampler\", \"bumpSampler\", \"reflectivitySampler\", \"albedoSampler\", \"morphTargets\"],\n defines: join,\n onCompiled: null,\n fallbacks: null,\n onError: null,\n uniformBuffersNames: [\"Scene\"],\n indexParameters: { buffersCount: this._multiRenderTarget.textures.length - 1, maxSimultaneousMorphTargets: numMorphInfluencers },\n }, engine), join);\n }\n return drawWrapper.effect.isReady();\n }\n /**\n * Gets the current underlying G Buffer.\n * @returns the buffer\n */\n getGBuffer() {\n return this._multiRenderTarget;\n }\n /**\n * Gets the number of samples used to render the buffer (anti aliasing).\n */\n get samples() {\n return this._multiRenderTarget.samples;\n }\n /**\n * Sets the number of samples used to render the buffer (anti aliasing).\n */\n set samples(value) {\n this._multiRenderTarget.samples = value;\n }\n /**\n * Disposes the renderer and frees up associated resources.\n */\n dispose() {\n if (this._resizeObserver) {\n const engine = this._scene.getEngine();\n engine.onResizeObservable.remove(this._resizeObserver);\n this._resizeObserver = null;\n }\n this.getGBuffer().dispose();\n }\n _assignRenderTargetIndices() {\n const textureNames = [];\n let count = 2;\n textureNames.push(\"gBuffer_Depth\", \"gBuffer_Normal\");\n if (this._enablePosition) {\n this._positionIndex = count;\n count++;\n textureNames.push(\"gBuffer_Position\");\n }\n if (this._enableVelocity) {\n this._velocityIndex = count;\n count++;\n textureNames.push(\"gBuffer_Velocity\");\n }\n if (this._enableReflectivity) {\n this._reflectivityIndex = count;\n count++;\n textureNames.push(\"gBuffer_Reflectivity\");\n }\n return [count, textureNames];\n }\n _createRenderTargets() {\n const engine = this._scene.getEngine();\n const [count, textureNames] = this._assignRenderTargetIndices();\n let type = 0;\n if (engine._caps.textureFloat && engine._caps.textureFloatLinearFiltering) {\n type = 1;\n }\n else if (engine._caps.textureHalfFloat && engine._caps.textureHalfFloatLinearFiltering) {\n type = 2;\n }\n this._multiRenderTarget = new MultiRenderTarget(\"gBuffer\", { width: engine.getRenderWidth() * this._ratio, height: engine.getRenderHeight() * this._ratio }, count, this._scene, { generateMipMaps: false, generateDepthTexture: true, defaultType: type, depthTextureFormat: this._depthFormat }, textureNames.concat(\"gBuffer_DepthBuffer\"));\n if (!this.isSupported) {\n return;\n }\n this._multiRenderTarget.wrapU = Texture.CLAMP_ADDRESSMODE;\n this._multiRenderTarget.wrapV = Texture.CLAMP_ADDRESSMODE;\n this._multiRenderTarget.refreshRate = 1;\n this._multiRenderTarget.renderParticles = false;\n this._multiRenderTarget.renderList = null;\n // Depth is always the first texture in the geometry buffer renderer!\n const layoutAttachmentsAll = [true];\n const layoutAttachmentsAllButDepth = [false];\n const layoutAttachmentsDepthOnly = [true];\n for (let i = 1; i < count; ++i) {\n layoutAttachmentsAll.push(true);\n layoutAttachmentsDepthOnly.push(false);\n layoutAttachmentsAllButDepth.push(true);\n }\n const attachmentsAll = engine.buildTextureLayout(layoutAttachmentsAll);\n const attachmentsAllButDepth = engine.buildTextureLayout(layoutAttachmentsAllButDepth);\n const attachmentsDepthOnly = engine.buildTextureLayout(layoutAttachmentsDepthOnly);\n this._multiRenderTarget.onClearObservable.add((engine) => {\n engine.bindAttachments(this.useSpecificClearForDepthTexture ? attachmentsAllButDepth : attachmentsAll);\n engine.clear(this._clearColor, true, true, true);\n if (this.useSpecificClearForDepthTexture) {\n engine.bindAttachments(attachmentsDepthOnly);\n engine.clear(this._clearDepthColor, true, true, true);\n }\n engine.bindAttachments(attachmentsAll);\n });\n this._resizeObserver = engine.onResizeObservable.add(() => {\n if (this._multiRenderTarget) {\n this._multiRenderTarget.resize({ width: engine.getRenderWidth() * this._ratio, height: engine.getRenderHeight() * this._ratio });\n }\n });\n // Custom render function\n const renderSubMesh = (subMesh) => {\n const renderingMesh = subMesh.getRenderingMesh();\n const effectiveMesh = subMesh.getEffectiveMesh();\n const scene = this._scene;\n const engine = scene.getEngine();\n const material = subMesh.getMaterial();\n if (!material) {\n return;\n }\n effectiveMesh._internalAbstractMeshDataInfo._isActiveIntermediate = false;\n // Velocity\n if (this._enableVelocity && !this._previousTransformationMatrices[effectiveMesh.uniqueId]) {\n this._previousTransformationMatrices[effectiveMesh.uniqueId] = {\n world: Matrix.Identity(),\n viewProjection: scene.getTransformMatrix(),\n };\n if (renderingMesh.skeleton) {\n const bonesTransformations = renderingMesh.skeleton.getTransformMatrices(renderingMesh);\n this._previousBonesTransformationMatrices[renderingMesh.uniqueId] = this._copyBonesTransformationMatrices(bonesTransformations, new Float32Array(bonesTransformations.length));\n }\n }\n // Managing instances\n const batch = renderingMesh._getInstancesRenderList(subMesh._id, !!subMesh.getReplacementMesh());\n if (batch.mustReturn) {\n return;\n }\n const hardwareInstancedRendering = engine.getCaps().instancedArrays && (batch.visibleInstances[subMesh._id] !== null || renderingMesh.hasThinInstances);\n const world = effectiveMesh.getWorldMatrix();\n if (this.isReady(subMesh, hardwareInstancedRendering)) {\n const drawWrapper = subMesh._getDrawWrapper();\n if (!drawWrapper) {\n return;\n }\n const effect = drawWrapper.effect;\n engine.enableEffect(drawWrapper);\n if (!hardwareInstancedRendering) {\n renderingMesh._bind(subMesh, effect, material.fillMode);\n }\n if (!this._useUbo) {\n effect.setMatrix(\"viewProjection\", scene.getTransformMatrix());\n effect.setMatrix(\"view\", scene.getViewMatrix());\n }\n else {\n MaterialHelper.BindSceneUniformBuffer(effect, this._scene.getSceneUniformBuffer());\n this._scene.finalizeSceneUbo();\n }\n let sideOrientation;\n const instanceDataStorage = renderingMesh._instanceDataStorage;\n if (!instanceDataStorage.isFrozen && (material.backFaceCulling || renderingMesh.overrideMaterialSideOrientation !== null)) {\n const mainDeterminant = effectiveMesh._getWorldMatrixDeterminant();\n sideOrientation = renderingMesh.overrideMaterialSideOrientation;\n if (sideOrientation === null) {\n sideOrientation = material.sideOrientation;\n }\n if (mainDeterminant < 0) {\n sideOrientation = sideOrientation === Material.ClockWiseSideOrientation ? Material.CounterClockWiseSideOrientation : Material.ClockWiseSideOrientation;\n }\n }\n else {\n sideOrientation = instanceDataStorage.sideOrientation;\n }\n material._preBind(drawWrapper, sideOrientation);\n // Alpha test\n if (material.needAlphaTesting()) {\n const alphaTexture = material.getAlphaTestTexture();\n if (alphaTexture) {\n effect.setTexture(\"diffuseSampler\", alphaTexture);\n effect.setMatrix(\"diffuseMatrix\", alphaTexture.getTextureMatrix());\n }\n }\n // Bump\n if (material.bumpTexture && scene.getEngine().getCaps().standardDerivatives && MaterialFlags.BumpTextureEnabled) {\n effect.setFloat3(\"vBumpInfos\", material.bumpTexture.coordinatesIndex, 1.0 / material.bumpTexture.level, material.parallaxScaleBias);\n effect.setMatrix(\"bumpMatrix\", material.bumpTexture.getTextureMatrix());\n effect.setTexture(\"bumpSampler\", material.bumpTexture);\n effect.setFloat2(\"vTangentSpaceParams\", material.invertNormalMapX ? -1.0 : 1.0, material.invertNormalMapY ? -1.0 : 1.0);\n }\n // Reflectivity\n if (this._enableReflectivity) {\n // for PBR materials: cf. https://doc.babylonjs.com/features/featuresDeepDive/materials/using/masterPBR\n if (material.getClassName() === \"PBRMetallicRoughnessMaterial\") {\n // if it is a PBR material in MetallicRoughness Mode:\n if (material.metallicRoughnessTexture !== null) {\n effect.setTexture(\"reflectivitySampler\", material.metallicRoughnessTexture);\n effect.setMatrix(\"reflectivityMatrix\", material.metallicRoughnessTexture.getTextureMatrix());\n }\n if (material.metallic !== null) {\n effect.setFloat(\"metallic\", material.metallic);\n }\n if (material.roughness !== null) {\n effect.setFloat(\"glossiness\", 1.0 - material.roughness);\n }\n if (material.baseTexture !== null) {\n effect.setTexture(\"albedoSampler\", material.baseTexture);\n effect.setMatrix(\"albedoMatrix\", material.baseTexture.getTextureMatrix());\n }\n if (material.baseColor !== null) {\n effect.setColor3(\"albedoColor\", material.baseColor);\n }\n }\n else if (material.getClassName() === \"PBRSpecularGlossinessMaterial\") {\n // if it is a PBR material in Specular/Glossiness Mode:\n if (material.specularGlossinessTexture !== null) {\n effect.setTexture(\"reflectivitySampler\", material.specularGlossinessTexture);\n effect.setMatrix(\"reflectivityMatrix\", material.specularGlossinessTexture.getTextureMatrix());\n }\n else {\n if (material.specularColor !== null) {\n effect.setColor3(\"reflectivityColor\", material.specularColor);\n }\n }\n if (material.glossiness !== null) {\n effect.setFloat(\"glossiness\", material.glossiness);\n }\n }\n else if (material.getClassName() === \"PBRMaterial\") {\n // if it is the bigger PBRMaterial\n if (material.metallicTexture !== null) {\n effect.setTexture(\"reflectivitySampler\", material.metallicTexture);\n effect.setMatrix(\"reflectivityMatrix\", material.metallicTexture.getTextureMatrix());\n }\n if (material.metallic !== null) {\n effect.setFloat(\"metallic\", material.metallic);\n }\n if (material.roughness !== null) {\n effect.setFloat(\"glossiness\", 1.0 - material.roughness);\n }\n if (material.roughness !== null || material.metallic !== null || material.metallicTexture !== null) {\n // MetallicRoughness Model\n if (material.albedoTexture !== null) {\n effect.setTexture(\"albedoSampler\", material.albedoTexture);\n effect.setMatrix(\"albedoMatrix\", material.albedoTexture.getTextureMatrix());\n }\n if (material.albedoColor !== null) {\n effect.setColor3(\"albedoColor\", material.albedoColor);\n }\n }\n else {\n // SpecularGlossiness Model\n if (material.reflectivityTexture !== null) {\n effect.setTexture(\"reflectivitySampler\", material.reflectivityTexture);\n effect.setMatrix(\"reflectivityMatrix\", material.reflectivityTexture.getTextureMatrix());\n }\n else if (material.reflectivityColor !== null) {\n effect.setColor3(\"reflectivityColor\", material.reflectivityColor);\n }\n if (material.microSurface !== null) {\n effect.setFloat(\"glossiness\", material.microSurface);\n }\n }\n }\n else if (material.getClassName() === \"StandardMaterial\") {\n // if StandardMaterial:\n if (material.specularTexture !== null) {\n effect.setTexture(\"reflectivitySampler\", material.specularTexture);\n effect.setMatrix(\"reflectivityMatrix\", material.specularTexture.getTextureMatrix());\n }\n if (material.specularColor !== null) {\n effect.setColor3(\"reflectivityColor\", material.specularColor);\n }\n }\n }\n // Clip plane\n bindClipPlane(effect, material, this._scene);\n // Bones\n if (renderingMesh.useBones && renderingMesh.computeBonesUsingShaders && renderingMesh.skeleton) {\n effect.setMatrices(\"mBones\", renderingMesh.skeleton.getTransformMatrices(renderingMesh));\n if (this._enableVelocity) {\n effect.setMatrices(\"mPreviousBones\", this._previousBonesTransformationMatrices[renderingMesh.uniqueId]);\n }\n }\n // Morph targets\n MaterialHelper.BindMorphTargetParameters(renderingMesh, effect);\n if (renderingMesh.morphTargetManager && renderingMesh.morphTargetManager.isUsingTextureForTargets) {\n renderingMesh.morphTargetManager._bind(effect);\n }\n // Velocity\n if (this._enableVelocity) {\n effect.setMatrix(\"previousWorld\", this._previousTransformationMatrices[effectiveMesh.uniqueId].world);\n effect.setMatrix(\"previousViewProjection\", this._previousTransformationMatrices[effectiveMesh.uniqueId].viewProjection);\n }\n if (hardwareInstancedRendering && renderingMesh.hasThinInstances) {\n effect.setMatrix(\"world\", world);\n }\n // Draw\n renderingMesh._processRendering(effectiveMesh, subMesh, effect, material.fillMode, batch, hardwareInstancedRendering, (isInstance, w) => {\n if (!isInstance) {\n effect.setMatrix(\"world\", w);\n }\n });\n }\n // Velocity\n if (this._enableVelocity) {\n this._previousTransformationMatrices[effectiveMesh.uniqueId].world = world.clone();\n this._previousTransformationMatrices[effectiveMesh.uniqueId].viewProjection = this._scene.getTransformMatrix().clone();\n if (renderingMesh.skeleton) {\n this._copyBonesTransformationMatrices(renderingMesh.skeleton.getTransformMatrices(renderingMesh), this._previousBonesTransformationMatrices[effectiveMesh.uniqueId]);\n }\n }\n };\n this._multiRenderTarget.customIsReadyFunction = (mesh, refreshRate, preWarm) => {\n if ((preWarm || refreshRate === 0) && mesh.subMeshes) {\n for (let i = 0; i < mesh.subMeshes.length; ++i) {\n const subMesh = mesh.subMeshes[i];\n const material = subMesh.getMaterial();\n const renderingMesh = subMesh.getRenderingMesh();\n if (!material) {\n continue;\n }\n const batch = renderingMesh._getInstancesRenderList(subMesh._id, !!subMesh.getReplacementMesh());\n const hardwareInstancedRendering = engine.getCaps().instancedArrays && (batch.visibleInstances[subMesh._id] !== null || renderingMesh.hasThinInstances);\n if (!this.isReady(subMesh, hardwareInstancedRendering)) {\n return false;\n }\n }\n }\n return true;\n };\n this._multiRenderTarget.customRenderFunction = (opaqueSubMeshes, alphaTestSubMeshes, transparentSubMeshes, depthOnlySubMeshes) => {\n let index;\n if (this._linkedWithPrePass) {\n if (!this._prePassRenderer.enabled) {\n return;\n }\n this._scene.getEngine().bindAttachments(this._attachmentsFromPrePass);\n }\n if (depthOnlySubMeshes.length) {\n engine.setColorWrite(false);\n for (index = 0; index < depthOnlySubMeshes.length; index++) {\n renderSubMesh(depthOnlySubMeshes.data[index]);\n }\n engine.setColorWrite(true);\n }\n for (index = 0; index < opaqueSubMeshes.length; index++) {\n renderSubMesh(opaqueSubMeshes.data[index]);\n }\n engine.setDepthWrite(false);\n for (index = 0; index < alphaTestSubMeshes.length; index++) {\n renderSubMesh(alphaTestSubMeshes.data[index]);\n }\n if (this.renderTransparentMeshes) {\n for (index = 0; index < transparentSubMeshes.length; index++) {\n renderSubMesh(transparentSubMeshes.data[index]);\n }\n }\n engine.setDepthWrite(true);\n };\n }\n // Copies the bones transformation matrices into the target array and returns the target's reference\n _copyBonesTransformationMatrices(source, target) {\n for (let i = 0; i < source.length; i++) {\n target[i] = source[i];\n }\n return target;\n }\n }\n /**\n * Constant used to retrieve the depth texture index in the G-Buffer textures array\n * using getIndex(GeometryBufferRenderer.DEPTH_TEXTURE_INDEX)\n */\n GeometryBufferRenderer.DEPTH_TEXTURE_TYPE = 0;\n /**\n * Constant used to retrieve the normal texture index in the G-Buffer textures array\n * using getIndex(GeometryBufferRenderer.NORMAL_TEXTURE_INDEX)\n */\n GeometryBufferRenderer.NORMAL_TEXTURE_TYPE = 1;\n /**\n * Constant used to retrieve the position texture index in the G-Buffer textures array\n * using getIndex(GeometryBufferRenderer.POSITION_TEXTURE_INDEX)\n */\n GeometryBufferRenderer.POSITION_TEXTURE_TYPE = 2;\n /**\n * Constant used to retrieve the velocity texture index in the G-Buffer textures array\n * using getIndex(GeometryBufferRenderer.VELOCITY_TEXTURE_INDEX)\n */\n GeometryBufferRenderer.VELOCITY_TEXTURE_TYPE = 3;\n /**\n * Constant used to retrieve the reflectivity texture index in the G-Buffer textures array\n * using the getIndex(GeometryBufferRenderer.REFLECTIVITY_TEXTURE_TYPE)\n */\n GeometryBufferRenderer.REFLECTIVITY_TEXTURE_TYPE = 4;\n /**\n * @internal\n */\n GeometryBufferRenderer._SceneComponentInitialization = (_) => {\n throw _WarnImport(\"GeometryBufferRendererSceneComponent\");\n };\n\n Object.defineProperty(Scene.prototype, \"geometryBufferRenderer\", {\n get: function () {\n return this._geometryBufferRenderer;\n },\n set: function (value) {\n if (value && value.isSupported) {\n this._geometryBufferRenderer = value;\n }\n },\n enumerable: true,\n configurable: true,\n });\n Scene.prototype.enableGeometryBufferRenderer = function (ratio = 1, depthFormat = 15) {\n if (this._geometryBufferRenderer) {\n return this._geometryBufferRenderer;\n }\n this._geometryBufferRenderer = new GeometryBufferRenderer(this, ratio, depthFormat);\n if (!this._geometryBufferRenderer.isSupported) {\n this._geometryBufferRenderer = null;\n }\n return this._geometryBufferRenderer;\n };\n Scene.prototype.disableGeometryBufferRenderer = function () {\n if (!this._geometryBufferRenderer) {\n return;\n }\n this._geometryBufferRenderer.dispose();\n this._geometryBufferRenderer = null;\n };\n /**\n * Defines the Geometry Buffer scene component responsible to manage a G-Buffer useful\n * in several rendering techniques.\n */\n class GeometryBufferRendererSceneComponent {\n /**\n * Creates a new instance of the component for the given scene\n * @param scene Defines the scene to register the component in\n */\n constructor(scene) {\n /**\n * The component name helpful to identify the component in the list of scene components.\n */\n this.name = SceneComponentConstants.NAME_GEOMETRYBUFFERRENDERER;\n this.scene = scene;\n }\n /**\n * Registers the component in a given scene\n */\n register() {\n this.scene._gatherRenderTargetsStage.registerStep(SceneComponentConstants.STEP_GATHERRENDERTARGETS_GEOMETRYBUFFERRENDERER, this, this._gatherRenderTargets);\n }\n /**\n * Rebuilds the elements related to this component in case of\n * context lost for instance.\n */\n rebuild() {\n // Nothing to do for this component\n }\n /**\n * Disposes the component and the associated resources\n */\n dispose() {\n // Nothing to do for this component\n }\n _gatherRenderTargets(renderTargets) {\n if (this.scene._geometryBufferRenderer) {\n renderTargets.push(this.scene._geometryBufferRenderer.getGBuffer());\n }\n }\n }\n GeometryBufferRenderer._SceneComponentInitialization = (scene) => {\n // Register the G Buffer component to the scene.\n let component = scene._getComponent(SceneComponentConstants.NAME_GEOMETRYBUFFERRENDERER);\n if (!component) {\n component = new GeometryBufferRendererSceneComponent(scene);\n scene._addComponent(component);\n }\n };\n\n // Do not edit.\n const name$2o = \"outlinePixelShader\";\n const shader$2o = `#ifdef LOGARITHMICDEPTH\n#extension GL_EXT_frag_depth : enable\n#endif\nuniform vec4 color;\n#ifdef ALPHATEST\nvarying vec2 vUV;uniform sampler2D diffuseSampler;\n#endif\n#include\n#include\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_FRAGMENT_MAIN_BEGIN\n#include\n#ifdef ALPHATEST\nif (texture2D(diffuseSampler,vUV).a<0.4)\ndiscard;\n#endif\n#include\ngl_FragColor=color;\n#define CUSTOM_FRAGMENT_MAIN_END\n}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2o] = shader$2o;\n\n // Do not edit.\n const name$2p = \"outlineVertexShader\";\n const shader$2p = `attribute vec3 position;attribute vec3 normal;\n#include\n#include\n#include\n#include[0..maxSimultaneousMorphTargets]\n#include\nuniform float offset;\n#include\nuniform mat4 viewProjection;\n#ifdef ALPHATEST\nvarying vec2 vUV;uniform mat4 diffuseMatrix;\n#ifdef UV1\nattribute vec2 uv;\n#endif\n#ifdef UV2\nattribute vec2 uv2;\n#endif\n#endif\n#include\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void)\n{vec3 positionUpdated=position;vec3 normalUpdated=normal;\n#ifdef UV1\nvec2 uvUpdated=uv;\n#endif\n#include\n#include[0..maxSimultaneousMorphTargets]\nvec3 offsetPosition=positionUpdated+(normalUpdated*offset);\n#include\n#include\n#include\nvec4 worldPos=finalWorld*vec4(offsetPosition,1.0);gl_Position=viewProjection*worldPos;\n#ifdef ALPHATEST\n#ifdef UV1\nvUV=vec2(diffuseMatrix*vec4(uvUpdated,1.0,0.0));\n#endif\n#ifdef UV2\nvUV=vec2(diffuseMatrix*vec4(uv2,1.0,0.0));\n#endif\n#endif\n#include\n#include\n}\n`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2p] = shader$2p;\n\n /**\n * Gets the outline renderer associated with the scene\n * @returns a OutlineRenderer\n */\n Scene.prototype.getOutlineRenderer = function () {\n if (!this._outlineRenderer) {\n this._outlineRenderer = new OutlineRenderer(this);\n }\n return this._outlineRenderer;\n };\n Object.defineProperty(Mesh.prototype, \"renderOutline\", {\n get: function () {\n return this._renderOutline;\n },\n set: function (value) {\n if (value) {\n // Lazy Load the component.\n this.getScene().getOutlineRenderer();\n }\n this._renderOutline = value;\n },\n enumerable: true,\n configurable: true,\n });\n Object.defineProperty(Mesh.prototype, \"renderOverlay\", {\n get: function () {\n return this._renderOverlay;\n },\n set: function (value) {\n if (value) {\n // Lazy Load the component.\n this.getScene().getOutlineRenderer();\n }\n this._renderOverlay = value;\n },\n enumerable: true,\n configurable: true,\n });\n /**\n * This class is responsible to draw the outline/overlay of meshes.\n * It should not be used directly but through the available method on mesh.\n */\n class OutlineRenderer {\n /**\n * Instantiates a new outline renderer. (There could be only one per scene).\n * @param scene Defines the scene it belongs to\n */\n constructor(scene) {\n /**\n * The name of the component. Each component must have a unique name.\n */\n this.name = SceneComponentConstants.NAME_OUTLINERENDERER;\n /**\n * Defines a zOffset default Factor to prevent zFighting between the overlay and the mesh.\n */\n this.zOffset = 1;\n /**\n * Defines a zOffset default Unit to prevent zFighting between the overlay and the mesh.\n */\n this.zOffsetUnits = 4; // 4 to account for projection a bit by default\n this.scene = scene;\n this._engine = scene.getEngine();\n this.scene._addComponent(this);\n this._passIdForDrawWrapper = [];\n for (let i = 0; i < 4; ++i) {\n this._passIdForDrawWrapper[i] = this._engine.createRenderPassId(`Outline Renderer (${i})`);\n }\n }\n /**\n * Register the component to one instance of a scene.\n */\n register() {\n this.scene._beforeRenderingMeshStage.registerStep(SceneComponentConstants.STEP_BEFORERENDERINGMESH_OUTLINE, this, this._beforeRenderingMesh);\n this.scene._afterRenderingMeshStage.registerStep(SceneComponentConstants.STEP_AFTERRENDERINGMESH_OUTLINE, this, this._afterRenderingMesh);\n }\n /**\n * Rebuilds the elements related to this component in case of\n * context lost for instance.\n */\n rebuild() {\n // Nothing to do here.\n }\n /**\n * Disposes the component and the associated resources.\n */\n dispose() {\n for (let i = 0; i < this._passIdForDrawWrapper.length; ++i) {\n this._engine.releaseRenderPassId(this._passIdForDrawWrapper[i]);\n }\n }\n /**\n * Renders the outline in the canvas.\n * @param subMesh Defines the sumesh to render\n * @param batch Defines the batch of meshes in case of instances\n * @param useOverlay Defines if the rendering is for the overlay or the outline\n * @param renderPassId Render pass id to use to render the mesh\n */\n render(subMesh, batch, useOverlay = false, renderPassId) {\n renderPassId = renderPassId !== null && renderPassId !== void 0 ? renderPassId : this._passIdForDrawWrapper[0];\n const scene = this.scene;\n const engine = scene.getEngine();\n const hardwareInstancedRendering = engine.getCaps().instancedArrays &&\n ((batch.visibleInstances[subMesh._id] !== null && batch.visibleInstances[subMesh._id] !== undefined) || subMesh.getRenderingMesh().hasThinInstances);\n if (!this.isReady(subMesh, hardwareInstancedRendering, renderPassId)) {\n return;\n }\n const ownerMesh = subMesh.getMesh();\n const replacementMesh = ownerMesh._internalAbstractMeshDataInfo._actAsRegularMesh ? ownerMesh : null;\n const renderingMesh = subMesh.getRenderingMesh();\n const effectiveMesh = replacementMesh ? replacementMesh : renderingMesh;\n const material = subMesh.getMaterial();\n if (!material || !scene.activeCamera) {\n return;\n }\n const drawWrapper = subMesh._getDrawWrapper(renderPassId);\n const effect = DrawWrapper.GetEffect(drawWrapper);\n engine.enableEffect(drawWrapper);\n // Logarithmic depth\n if (material.useLogarithmicDepth) {\n effect.setFloat(\"logarithmicDepthConstant\", 2.0 / (Math.log(scene.activeCamera.maxZ + 1.0) / Math.LN2));\n }\n effect.setFloat(\"offset\", useOverlay ? 0 : renderingMesh.outlineWidth);\n effect.setColor4(\"color\", useOverlay ? renderingMesh.overlayColor : renderingMesh.outlineColor, useOverlay ? renderingMesh.overlayAlpha : material.alpha);\n effect.setMatrix(\"viewProjection\", scene.getTransformMatrix());\n effect.setMatrix(\"world\", effectiveMesh.getWorldMatrix());\n // Bones\n if (renderingMesh.useBones && renderingMesh.computeBonesUsingShaders && renderingMesh.skeleton) {\n effect.setMatrices(\"mBones\", renderingMesh.skeleton.getTransformMatrices(renderingMesh));\n }\n if (renderingMesh.morphTargetManager && renderingMesh.morphTargetManager.isUsingTextureForTargets) {\n renderingMesh.morphTargetManager._bind(effect);\n }\n // Morph targets\n MaterialHelper.BindMorphTargetParameters(renderingMesh, effect);\n if (!hardwareInstancedRendering) {\n renderingMesh._bind(subMesh, effect, material.fillMode);\n }\n // Alpha test\n if (material && material.needAlphaTesting()) {\n const alphaTexture = material.getAlphaTestTexture();\n if (alphaTexture) {\n effect.setTexture(\"diffuseSampler\", alphaTexture);\n effect.setMatrix(\"diffuseMatrix\", alphaTexture.getTextureMatrix());\n }\n }\n // Clip plane\n bindClipPlane(effect, material, scene);\n engine.setZOffset(-this.zOffset);\n engine.setZOffsetUnits(-this.zOffsetUnits);\n renderingMesh._processRendering(effectiveMesh, subMesh, effect, material.fillMode, batch, hardwareInstancedRendering, (isInstance, world) => {\n effect.setMatrix(\"world\", world);\n });\n engine.setZOffset(0);\n engine.setZOffsetUnits(0);\n }\n /**\n * Returns whether or not the outline renderer is ready for a given submesh.\n * All the dependencies e.g. submeshes, texture, effect... mus be ready\n * @param subMesh Defines the submesh to check readiness for\n * @param useInstances Defines whether wee are trying to render instances or not\n * @param renderPassId Render pass id to use to render the mesh\n * @returns true if ready otherwise false\n */\n isReady(subMesh, useInstances, renderPassId) {\n renderPassId = renderPassId !== null && renderPassId !== void 0 ? renderPassId : this._passIdForDrawWrapper[0];\n const defines = [];\n const attribs = [VertexBuffer.PositionKind, VertexBuffer.NormalKind];\n const mesh = subMesh.getMesh();\n const material = subMesh.getMaterial();\n if (!material) {\n return false;\n }\n const scene = mesh.getScene();\n // Alpha test\n if (material.needAlphaTesting()) {\n defines.push(\"#define ALPHATEST\");\n if (mesh.isVerticesDataPresent(VertexBuffer.UVKind)) {\n attribs.push(VertexBuffer.UVKind);\n defines.push(\"#define UV1\");\n }\n if (mesh.isVerticesDataPresent(VertexBuffer.UV2Kind)) {\n attribs.push(VertexBuffer.UV2Kind);\n defines.push(\"#define UV2\");\n }\n }\n //Logarithmic depth\n if (material.useLogarithmicDepth) {\n defines.push(\"#define LOGARITHMICDEPTH\");\n }\n // Clip planes\n prepareStringDefinesForClipPlanes(material, scene, defines);\n // Bones\n if (mesh.useBones && mesh.computeBonesUsingShaders) {\n attribs.push(VertexBuffer.MatricesIndicesKind);\n attribs.push(VertexBuffer.MatricesWeightsKind);\n if (mesh.numBoneInfluencers > 4) {\n attribs.push(VertexBuffer.MatricesIndicesExtraKind);\n attribs.push(VertexBuffer.MatricesWeightsExtraKind);\n }\n defines.push(\"#define NUM_BONE_INFLUENCERS \" + mesh.numBoneInfluencers);\n defines.push(\"#define BonesPerMesh \" + (mesh.skeleton ? mesh.skeleton.bones.length + 1 : 0));\n }\n else {\n defines.push(\"#define NUM_BONE_INFLUENCERS 0\");\n }\n // Morph targets\n const morphTargetManager = mesh.morphTargetManager;\n let numMorphInfluencers = 0;\n if (morphTargetManager) {\n if (morphTargetManager.numInfluencers > 0) {\n numMorphInfluencers = morphTargetManager.numInfluencers;\n defines.push(\"#define MORPHTARGETS\");\n defines.push(\"#define NUM_MORPH_INFLUENCERS \" + numMorphInfluencers);\n if (morphTargetManager.isUsingTextureForTargets) {\n defines.push(\"#define MORPHTARGETS_TEXTURE\");\n }\n MaterialHelper.PrepareAttributesForMorphTargetsInfluencers(attribs, mesh, numMorphInfluencers);\n }\n }\n // Instances\n if (useInstances) {\n defines.push(\"#define INSTANCES\");\n MaterialHelper.PushAttributesForInstances(attribs);\n if (subMesh.getRenderingMesh().hasThinInstances) {\n defines.push(\"#define THIN_INSTANCES\");\n }\n }\n // Get correct effect\n const drawWrapper = subMesh._getDrawWrapper(renderPassId, true);\n const cachedDefines = drawWrapper.defines;\n const join = defines.join(\"\\n\");\n if (cachedDefines !== join) {\n const uniforms = [\n \"world\",\n \"mBones\",\n \"viewProjection\",\n \"diffuseMatrix\",\n \"offset\",\n \"color\",\n \"logarithmicDepthConstant\",\n \"morphTargetInfluences\",\n \"morphTargetTextureInfo\",\n \"morphTargetTextureIndices\",\n ];\n addClipPlaneUniforms(uniforms);\n drawWrapper.setEffect(this.scene.getEngine().createEffect(\"outline\", attribs, uniforms, [\"diffuseSampler\", \"morphTargets\"], join, undefined, undefined, undefined, {\n maxSimultaneousMorphTargets: numMorphInfluencers,\n }), join);\n }\n return drawWrapper.effect.isReady();\n }\n _beforeRenderingMesh(mesh, subMesh, batch) {\n // Outline - step 1\n this._savedDepthWrite = this._engine.getDepthWrite();\n if (mesh.renderOutline) {\n const material = subMesh.getMaterial();\n if (material && material.needAlphaBlendingForMesh(mesh)) {\n this._engine.cacheStencilState();\n // Draw only to stencil buffer for the original mesh\n // The resulting stencil buffer will be used so the outline is not visible inside the mesh when the mesh is transparent\n this._engine.setDepthWrite(false);\n this._engine.setColorWrite(false);\n this._engine.setStencilBuffer(true);\n this._engine.setStencilOperationPass(7681);\n this._engine.setStencilFunction(519);\n this._engine.setStencilMask(OutlineRenderer._StencilReference);\n this._engine.setStencilFunctionReference(OutlineRenderer._StencilReference);\n this._engine.stencilStateComposer.useStencilGlobalOnly = true;\n this.render(subMesh, batch, /* This sets offset to 0 */ true, this._passIdForDrawWrapper[1]);\n this._engine.setColorWrite(true);\n this._engine.setStencilFunction(517);\n }\n // Draw the outline using the above stencil if needed to avoid drawing within the mesh\n this._engine.setDepthWrite(false);\n this.render(subMesh, batch, false, this._passIdForDrawWrapper[0]);\n this._engine.setDepthWrite(this._savedDepthWrite);\n if (material && material.needAlphaBlendingForMesh(mesh)) {\n this._engine.stencilStateComposer.useStencilGlobalOnly = false;\n this._engine.restoreStencilState();\n }\n }\n }\n _afterRenderingMesh(mesh, subMesh, batch) {\n // Overlay\n if (mesh.renderOverlay) {\n const currentMode = this._engine.getAlphaMode();\n const alphaBlendState = this._engine.alphaState.alphaBlend;\n this._engine.setAlphaMode(2);\n this.render(subMesh, batch, true, this._passIdForDrawWrapper[3]);\n this._engine.setAlphaMode(currentMode);\n this._engine.setDepthWrite(this._savedDepthWrite);\n this._engine.alphaState.alphaBlend = alphaBlendState;\n }\n // Outline - step 2\n if (mesh.renderOutline && this._savedDepthWrite) {\n this._engine.setDepthWrite(true);\n this._engine.setColorWrite(false);\n this.render(subMesh, batch, false, this._passIdForDrawWrapper[2]);\n this._engine.setColorWrite(true);\n }\n }\n }\n /**\n * Stencil value used to avoid outline being seen within the mesh when the mesh is transparent\n */\n OutlineRenderer._StencilReference = 0x04;\n\n /**\n * A multi render target designed to render the prepass.\n * Prepass is a scene component used to render information in multiple textures\n * alongside with the scene materials rendering.\n * Note : This is an internal class, and you should NOT need to instanciate this.\n * Only the `PrePassRenderer` should instanciate this class.\n * It is more likely that you need a regular `MultiRenderTarget`\n * @internal\n */\n class PrePassRenderTarget extends MultiRenderTarget {\n constructor(name, renderTargetTexture, size, count, scene, options) {\n super(name, size, count, scene, options);\n /**\n * @internal\n */\n this._beforeCompositionPostProcesses = [];\n /**\n * @internal\n */\n this._internalTextureDirty = false;\n /**\n * Is this render target enabled for prepass rendering\n */\n this.enabled = false;\n /**\n * Render target associated with this prePassRenderTarget\n * If this is `null`, it means this prePassRenderTarget is associated with the scene\n */\n this.renderTargetTexture = null;\n this.renderTargetTexture = renderTargetTexture;\n }\n /**\n * Creates a composition effect for this RT\n * @internal\n */\n _createCompositionEffect() {\n this.imageProcessingPostProcess = new ImageProcessingPostProcess(\"prePassComposition\", 1, null, undefined, this._engine);\n this.imageProcessingPostProcess._updateParameters();\n }\n /**\n * Checks that the size of this RT is still adapted to the desired render size.\n * @internal\n */\n _checkSize() {\n const requiredWidth = this._engine.getRenderWidth(true);\n const requiredHeight = this._engine.getRenderHeight(true);\n const width = this.getRenderWidth();\n const height = this.getRenderHeight();\n if (width !== requiredWidth || height !== requiredHeight) {\n this.resize({ width: requiredWidth, height: requiredHeight });\n this._internalTextureDirty = true;\n }\n }\n /**\n * Changes the number of render targets in this MRT\n * Be careful as it will recreate all the data in the new texture.\n * @param count new texture count\n * @param options Specifies texture types and sampling modes for new textures\n * @param textureNames Specifies the names of the textures (optional)\n */\n updateCount(count, options, textureNames) {\n super.updateCount(count, options, textureNames);\n this._internalTextureDirty = true;\n }\n /**\n * Resets the post processes chains applied to this RT.\n * @internal\n */\n _resetPostProcessChain() {\n this._beforeCompositionPostProcesses.length = 0;\n }\n /**\n * Diposes this render target\n */\n dispose() {\n const scene = this._scene;\n super.dispose();\n if (scene && scene.prePassRenderer) {\n const index = scene.prePassRenderer.renderTargets.indexOf(this);\n if (index !== -1) {\n scene.prePassRenderer.renderTargets.splice(index, 1);\n }\n }\n if (this.imageProcessingPostProcess) {\n this.imageProcessingPostProcess.dispose();\n }\n if (this.renderTargetTexture) {\n this.renderTargetTexture._prePassRenderTarget = null;\n }\n if (this._outputPostProcess) {\n this._outputPostProcess.autoClear = true;\n this._outputPostProcess.restoreDefaultInputTexture();\n }\n }\n }\n\n /**\n * Renders a pre pass of the scene\n * This means every mesh in the scene will be rendered to a render target texture\n * And then this texture will be composited to the rendering canvas with post processes\n * It is necessary for effects like subsurface scattering or deferred shading\n */\n class PrePassRenderer {\n /**\n * Returns the index of a texture in the multi render target texture array.\n * @param type Texture type\n * @returns The index\n */\n getIndex(type) {\n return this._textureIndices[type];\n }\n /**\n * How many samples are used for MSAA of the scene render target\n */\n get samples() {\n return this.defaultRT.samples;\n }\n set samples(n) {\n this.defaultRT.samples = n;\n }\n /**\n * If set to true (default: false), the depth texture will be cleared with the depth value corresponding to the far plane (1 in normal mode, 0 in reverse depth buffer mode)\n * If set to false, the depth texture is always cleared with 0.\n */\n get useSpecificClearForDepthTexture() {\n return this._useSpecificClearForDepthTexture;\n }\n set useSpecificClearForDepthTexture(value) {\n if (this._useSpecificClearForDepthTexture === value) {\n return;\n }\n this._useSpecificClearForDepthTexture = value;\n this._isDirty = true;\n }\n /**\n * @returns the prepass render target for the rendering pass.\n * If we are currently rendering a render target, it returns the PrePassRenderTarget\n * associated with that render target. Otherwise, it returns the scene default PrePassRenderTarget\n */\n getRenderTarget() {\n return this._currentTarget;\n }\n /**\n * @internal\n * Managed by the scene component\n * @param prePassRenderTarget\n */\n _setRenderTarget(prePassRenderTarget) {\n var _a, _b;\n if (prePassRenderTarget) {\n this._currentTarget = prePassRenderTarget;\n }\n else {\n this._currentTarget = this.defaultRT;\n this._engine.currentRenderPassId = (_b = (_a = this._scene.activeCamera) === null || _a === void 0 ? void 0 : _a.renderPassId) !== null && _b !== void 0 ? _b : this._currentTarget.renderPassId;\n }\n }\n /**\n * Returns true if the currently rendered prePassRenderTarget is the one\n * associated with the scene.\n */\n get currentRTisSceneRT() {\n return this._currentTarget === this.defaultRT;\n }\n _refreshGeometryBufferRendererLink() {\n if (!this.doNotUseGeometryRendererFallback) {\n this._geometryBuffer = this._scene.enableGeometryBufferRenderer();\n if (!this._geometryBuffer) {\n // Not supported\n this.doNotUseGeometryRendererFallback = true;\n return;\n }\n this._geometryBuffer._linkPrePassRenderer(this);\n }\n else {\n if (this._geometryBuffer) {\n this._geometryBuffer._unlinkPrePassRenderer();\n }\n this._geometryBuffer = null;\n this._scene.disableGeometryBufferRenderer();\n }\n }\n /**\n * Indicates if the prepass is enabled\n */\n get enabled() {\n return this._enabled;\n }\n /**\n * Instantiates a prepass renderer\n * @param scene The scene\n */\n constructor(scene) {\n /**\n * To save performance, we can excluded skinned meshes from the prepass\n */\n this.excludedSkinnedMesh = [];\n /**\n * Force material to be excluded from the prepass\n * Can be useful when `useGeometryBufferFallback` is set to `true`\n * and you don't want a material to show in the effect.\n */\n this.excludedMaterials = [];\n /**\n * Number of textures in the multi render target texture where the scene is directly rendered\n */\n this.mrtCount = 0;\n this._mrtTypes = [];\n this._mrtFormats = [];\n this._mrtLayout = [];\n this._mrtNames = [];\n this._textureIndices = [];\n this._useSpecificClearForDepthTexture = false;\n this._isDirty = true;\n /**\n * Configuration for prepass effects\n */\n this._effectConfigurations = [];\n /**\n * Prevents the PrePassRenderer from using the GeometryBufferRenderer as a fallback\n */\n this.doNotUseGeometryRendererFallback = true;\n /**\n * All the render targets generated by prepass\n */\n this.renderTargets = [];\n this._clearColor = new Color4(0, 0, 0, 0);\n this._clearDepthColor = new Color4(1e8, 0, 0, 1); // \"infinity\" value - depth in the depth texture is view.z, not a 0..1 value!\n this._enabled = false;\n this._needsCompositionForThisPass = false;\n /**\n * Set to true to disable gamma transform in PrePass.\n * Can be useful in case you already proceed to gamma transform on a material level\n * and your post processes don't need to be in linear color space.\n */\n this.disableGammaTransform = false;\n this._scene = scene;\n this._engine = scene.getEngine();\n let type = 0;\n if (this._engine._caps.textureFloat && this._engine._caps.textureFloatLinearFiltering) {\n type = 1;\n }\n else if (this._engine._caps.textureHalfFloat && this._engine._caps.textureHalfFloatLinearFiltering) {\n type = 2;\n }\n if (type !== 1) {\n for (let i = 0; i < PrePassRenderer.TextureFormats.length; ++i) {\n if (PrePassRenderer.TextureFormats[i].type === 1) {\n PrePassRenderer.TextureFormats[5].type = type;\n }\n }\n }\n PrePassRenderer._SceneComponentInitialization(this._scene);\n this.defaultRT = this._createRenderTarget(\"sceneprePassRT\", null);\n this._currentTarget = this.defaultRT;\n }\n /**\n * Creates a new PrePassRenderTarget\n * This should be the only way to instantiate a `PrePassRenderTarget`\n * @param name Name of the `PrePassRenderTarget`\n * @param renderTargetTexture RenderTarget the `PrePassRenderTarget` will be attached to.\n * Can be `null` if the created `PrePassRenderTarget` is attached to the scene (default framebuffer).\n * @internal\n */\n _createRenderTarget(name, renderTargetTexture) {\n const rt = new PrePassRenderTarget(name, renderTargetTexture, { width: this._engine.getRenderWidth(), height: this._engine.getRenderHeight() }, 0, this._scene, {\n generateMipMaps: false,\n generateStencilBuffer: this._engine.isStencilEnable,\n defaultType: 0,\n types: [],\n drawOnlyOnFirstAttachmentByDefault: true,\n });\n this.renderTargets.push(rt);\n if (this._enabled) {\n // The pre-pass renderer is already enabled, so make sure we create the render target with the correct number of textures\n this._update();\n }\n return rt;\n }\n /**\n * Indicates if rendering a prepass is supported\n */\n get isSupported() {\n return this._scene.getEngine().getCaps().drawBuffersExtension;\n }\n /**\n * Sets the proper output textures to draw in the engine.\n * @param effect The effect that is drawn. It can be or not be compatible with drawing to several output textures.\n * @param subMesh Submesh on which the effect is applied\n */\n bindAttachmentsForEffect(effect, subMesh) {\n const material = subMesh.getMaterial();\n const isPrePassCapable = material && material.isPrePassCapable;\n const excluded = material && this.excludedMaterials.indexOf(material) !== -1;\n if (this.enabled && this._currentTarget.enabled) {\n if (effect._multiTarget && isPrePassCapable && !excluded) {\n this._engine.bindAttachments(this._multiRenderAttachments);\n }\n else {\n if (this._engine._currentRenderTarget) {\n this._engine.bindAttachments(this._defaultAttachments);\n }\n else {\n this._engine.restoreSingleAttachment();\n }\n if (this._geometryBuffer && this.currentRTisSceneRT && !excluded) {\n this._geometryBuffer.renderList.push(subMesh.getRenderingMesh());\n }\n }\n }\n }\n _reinitializeAttachments() {\n const multiRenderLayout = [];\n const clearLayout = [false];\n const clearDepthLayout = [false];\n const defaultLayout = [true];\n for (let i = 0; i < this.mrtCount; i++) {\n multiRenderLayout.push(true);\n if (i > 0) {\n if (this._useSpecificClearForDepthTexture && this._mrtLayout[i] === 5) {\n clearLayout.push(false);\n clearDepthLayout.push(true);\n }\n else {\n clearLayout.push(true);\n clearDepthLayout.push(false);\n }\n defaultLayout.push(false);\n }\n }\n this._multiRenderAttachments = this._engine.buildTextureLayout(multiRenderLayout);\n this._clearAttachments = this._engine.buildTextureLayout(clearLayout);\n this._clearDepthAttachments = this._engine.buildTextureLayout(clearDepthLayout);\n this._defaultAttachments = this._engine.buildTextureLayout(defaultLayout);\n }\n _resetLayout() {\n for (let i = 0; i < PrePassRenderer.TextureFormats.length; i++) {\n this._textureIndices[PrePassRenderer.TextureFormats[i].purpose] = -1;\n }\n this._textureIndices[4] = 0;\n this._mrtLayout = [4];\n this._mrtTypes = [PrePassRenderer.TextureFormats[4].type];\n this._mrtFormats = [PrePassRenderer.TextureFormats[4].format];\n this._mrtNames = [PrePassRenderer.TextureFormats[4].name];\n this.mrtCount = 1;\n }\n _updateGeometryBufferLayout() {\n this._refreshGeometryBufferRendererLink();\n if (this._geometryBuffer) {\n this._geometryBuffer._resetLayout();\n const texturesActivated = [];\n for (let i = 0; i < this._mrtLayout.length; i++) {\n texturesActivated.push(false);\n }\n this._geometryBuffer._linkInternalTexture(this.defaultRT.getInternalTexture());\n const matches = [\n {\n prePassConstant: 5,\n geometryBufferConstant: GeometryBufferRenderer.DEPTH_TEXTURE_TYPE,\n },\n {\n prePassConstant: 6,\n geometryBufferConstant: GeometryBufferRenderer.NORMAL_TEXTURE_TYPE,\n },\n {\n prePassConstant: 1,\n geometryBufferConstant: GeometryBufferRenderer.POSITION_TEXTURE_TYPE,\n },\n {\n prePassConstant: 3,\n geometryBufferConstant: GeometryBufferRenderer.REFLECTIVITY_TEXTURE_TYPE,\n },\n {\n prePassConstant: 2,\n geometryBufferConstant: GeometryBufferRenderer.VELOCITY_TEXTURE_TYPE,\n },\n ];\n // replace textures in the geometryBuffer RT\n for (let i = 0; i < matches.length; i++) {\n const index = this._mrtLayout.indexOf(matches[i].prePassConstant);\n if (index !== -1) {\n this._geometryBuffer._forceTextureType(matches[i].geometryBufferConstant, index);\n texturesActivated[index] = true;\n }\n }\n this._geometryBuffer._setAttachments(this._engine.buildTextureLayout(texturesActivated));\n }\n }\n /**\n * Restores attachments for single texture draw.\n */\n restoreAttachments() {\n if (this.enabled && this._currentTarget.enabled && this._defaultAttachments) {\n if (this._engine._currentRenderTarget) {\n this._engine.bindAttachments(this._defaultAttachments);\n }\n else {\n this._engine.restoreSingleAttachment();\n }\n }\n }\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _beforeDraw(camera, faceIndex, layer) {\n // const previousEnabled = this._enabled && this._currentTarget.enabled;\n if (this._isDirty) {\n this._update();\n }\n if (!this._enabled || !this._currentTarget.enabled) {\n return;\n }\n if (this._geometryBuffer) {\n this._geometryBuffer.renderList = [];\n }\n this._setupOutputForThisPass(this._currentTarget, camera);\n }\n _prepareFrame(prePassRenderTarget, faceIndex, layer) {\n if (prePassRenderTarget.renderTargetTexture) {\n prePassRenderTarget.renderTargetTexture._prepareFrame(this._scene, faceIndex, layer, prePassRenderTarget.renderTargetTexture.useCameraPostProcesses);\n }\n else if (this._postProcessesSourceForThisPass.length) {\n this._scene.postProcessManager._prepareFrame();\n }\n else {\n this._engine.restoreDefaultFramebuffer();\n }\n }\n /**\n * Sets an intermediary texture between prepass and postprocesses. This texture\n * will be used as input for post processes\n * @param rt\n * @returns true if there are postprocesses that will use this texture,\n * false if there is no postprocesses - and the function has no effect\n */\n setCustomOutput(rt) {\n const firstPP = this._postProcessesSourceForThisPass[0];\n if (!firstPP) {\n return false;\n }\n firstPP.inputTexture = rt.renderTarget;\n return true;\n }\n _renderPostProcesses(prePassRenderTarget, faceIndex) {\n var _a;\n const firstPP = this._postProcessesSourceForThisPass[0];\n const outputTexture = firstPP ? firstPP.inputTexture : prePassRenderTarget.renderTargetTexture ? prePassRenderTarget.renderTargetTexture.renderTarget : null;\n // Build post process chain for this prepass post draw\n let postProcessChain = this._currentTarget._beforeCompositionPostProcesses;\n if (this._needsCompositionForThisPass) {\n postProcessChain = postProcessChain.concat([this._currentTarget.imageProcessingPostProcess]);\n }\n // Activates and renders the chain\n if (postProcessChain.length) {\n this._scene.postProcessManager._prepareFrame((_a = this._currentTarget.renderTarget) === null || _a === void 0 ? void 0 : _a.texture, postProcessChain);\n this._scene.postProcessManager.directRender(postProcessChain, outputTexture, false, faceIndex);\n }\n }\n /**\n * @internal\n */\n _afterDraw(faceIndex, layer) {\n if (this._enabled && this._currentTarget.enabled) {\n this._prepareFrame(this._currentTarget, faceIndex, layer);\n this._renderPostProcesses(this._currentTarget, faceIndex);\n }\n }\n /**\n * Clears the current prepass render target (in the sense of settings pixels to the scene clear color value)\n * @internal\n */\n _clear() {\n if (this._enabled && this._currentTarget.enabled) {\n this._bindFrameBuffer(this._currentTarget);\n // Clearing other attachment with 0 on all other attachments\n this._engine.bindAttachments(this._clearAttachments);\n this._engine.clear(this._clearColor, true, false, false);\n if (this._useSpecificClearForDepthTexture) {\n this._engine.bindAttachments(this._clearDepthAttachments);\n this._engine.clear(this._clearDepthColor, true, false, false);\n }\n // Regular clear color with the scene clear color of the 1st attachment\n this._engine.bindAttachments(this._defaultAttachments);\n }\n }\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _bindFrameBuffer(prePassRenderTarget) {\n if (this._enabled && this._currentTarget.enabled) {\n this._currentTarget._checkSize();\n const internalTexture = this._currentTarget.renderTarget;\n if (internalTexture) {\n this._engine.bindFramebuffer(internalTexture);\n }\n }\n }\n _setEnabled(enabled) {\n this._enabled = enabled;\n }\n _setRenderTargetEnabled(prePassRenderTarget, enabled) {\n prePassRenderTarget.enabled = enabled;\n if (!enabled) {\n this._unlinkInternalTexture(prePassRenderTarget);\n }\n }\n /**\n * Adds an effect configuration to the prepass render target.\n * If an effect has already been added, it won't add it twice and will return the configuration\n * already present.\n * @param cfg the effect configuration\n * @returns the effect configuration now used by the prepass\n */\n addEffectConfiguration(cfg) {\n // Do not add twice\n for (let i = 0; i < this._effectConfigurations.length; i++) {\n if (this._effectConfigurations[i].name === cfg.name) {\n return this._effectConfigurations[i];\n }\n }\n this._effectConfigurations.push(cfg);\n return cfg;\n }\n _enable() {\n const previousMrtCount = this.mrtCount;\n for (let i = 0; i < this._effectConfigurations.length; i++) {\n if (this._effectConfigurations[i].enabled) {\n this._enableTextures(this._effectConfigurations[i].texturesRequired);\n }\n }\n for (let i = 0; i < this.renderTargets.length; i++) {\n if (this.mrtCount !== previousMrtCount || this.renderTargets[i].count !== this.mrtCount) {\n this.renderTargets[i].updateCount(this.mrtCount, { types: this._mrtTypes, formats: this._mrtFormats }, this._mrtNames.concat(\"prePass_DepthBuffer\"));\n }\n this.renderTargets[i]._resetPostProcessChain();\n for (let j = 0; j < this._effectConfigurations.length; j++) {\n if (this._effectConfigurations[j].enabled) {\n // TODO : subsurface scattering has 1 scene-wide effect configuration\n // solution : do not stock postProcess on effectConfiguration, but in the prepassRenderTarget (hashmap configuration => postProcess)\n // And call createPostProcess whenever the post process does not exist in the RT\n if (!this._effectConfigurations[j].postProcess && this._effectConfigurations[j].createPostProcess) {\n this._effectConfigurations[j].createPostProcess();\n }\n if (this._effectConfigurations[j].postProcess) {\n this.renderTargets[i]._beforeCompositionPostProcesses.push(this._effectConfigurations[j].postProcess);\n }\n }\n }\n }\n this._reinitializeAttachments();\n this._setEnabled(true);\n this._updateGeometryBufferLayout();\n }\n _disable() {\n this._setEnabled(false);\n for (let i = 0; i < this.renderTargets.length; i++) {\n this._setRenderTargetEnabled(this.renderTargets[i], false);\n }\n this._resetLayout();\n for (let i = 0; i < this._effectConfigurations.length; i++) {\n this._effectConfigurations[i].enabled = false;\n }\n }\n _getPostProcessesSource(prePassRenderTarget, camera) {\n if (camera) {\n return camera._postProcesses;\n }\n else if (prePassRenderTarget.renderTargetTexture) {\n if (prePassRenderTarget.renderTargetTexture.useCameraPostProcesses) {\n const camera = prePassRenderTarget.renderTargetTexture.activeCamera ? prePassRenderTarget.renderTargetTexture.activeCamera : this._scene.activeCamera;\n return camera ? camera._postProcesses : [];\n }\n else if (prePassRenderTarget.renderTargetTexture.postProcesses) {\n return prePassRenderTarget.renderTargetTexture.postProcesses;\n }\n else {\n return [];\n }\n }\n else {\n return this._scene.activeCamera ? this._scene.activeCamera._postProcesses : [];\n }\n }\n _setupOutputForThisPass(prePassRenderTarget, camera) {\n // Order is : draw ===> prePassRenderTarget._postProcesses ==> ipp ==> camera._postProcesses\n const secondaryCamera = camera && this._scene.activeCameras && !!this._scene.activeCameras.length && this._scene.activeCameras.indexOf(camera) !== 0;\n this._postProcessesSourceForThisPass = this._getPostProcessesSource(prePassRenderTarget, camera);\n this._postProcessesSourceForThisPass = this._postProcessesSourceForThisPass.filter((pp) => {\n return pp != null;\n });\n this._scene.autoClear = true;\n const cameraHasImageProcessing = this._hasImageProcessing(this._postProcessesSourceForThisPass);\n this._needsCompositionForThisPass = !cameraHasImageProcessing && !this.disableGammaTransform && this._needsImageProcessing() && !secondaryCamera;\n const firstCameraPP = this._getFirstPostProcess(this._postProcessesSourceForThisPass);\n const firstPrePassPP = prePassRenderTarget._beforeCompositionPostProcesses && prePassRenderTarget._beforeCompositionPostProcesses[0];\n let firstPP = null;\n // Setting the scene-wide post process configuration\n this._scene.imageProcessingConfiguration.applyByPostProcess = this._needsCompositionForThisPass || cameraHasImageProcessing;\n // Create composition effect if needed\n if (this._needsCompositionForThisPass && !prePassRenderTarget.imageProcessingPostProcess) {\n prePassRenderTarget._createCompositionEffect();\n }\n // Setting the prePassRenderTarget as input texture of the first PP\n if (firstPrePassPP) {\n firstPP = firstPrePassPP;\n }\n else if (this._needsCompositionForThisPass) {\n firstPP = prePassRenderTarget.imageProcessingPostProcess;\n }\n else if (firstCameraPP) {\n firstPP = firstCameraPP;\n }\n this._bindFrameBuffer(prePassRenderTarget);\n this._linkInternalTexture(prePassRenderTarget, firstPP);\n }\n _linkInternalTexture(prePassRenderTarget, postProcess) {\n if (postProcess) {\n postProcess.autoClear = false;\n postProcess.inputTexture = prePassRenderTarget.renderTarget;\n }\n if (prePassRenderTarget._outputPostProcess !== postProcess) {\n if (prePassRenderTarget._outputPostProcess) {\n this._unlinkInternalTexture(prePassRenderTarget);\n }\n prePassRenderTarget._outputPostProcess = postProcess;\n }\n if (prePassRenderTarget._internalTextureDirty) {\n this._updateGeometryBufferLayout();\n prePassRenderTarget._internalTextureDirty = false;\n }\n }\n /**\n * @internal\n */\n _unlinkInternalTexture(prePassRenderTarget) {\n if (prePassRenderTarget._outputPostProcess) {\n prePassRenderTarget._outputPostProcess.autoClear = true;\n prePassRenderTarget._outputPostProcess.restoreDefaultInputTexture();\n prePassRenderTarget._outputPostProcess = null;\n }\n }\n _needsImageProcessing() {\n for (let i = 0; i < this._effectConfigurations.length; i++) {\n if (this._effectConfigurations[i].enabled && this._effectConfigurations[i].needsImageProcessing) {\n return true;\n }\n }\n return false;\n }\n _hasImageProcessing(postProcesses) {\n var _a;\n let isIPPAlreadyPresent = false;\n if (postProcesses) {\n for (let i = 0; i < postProcesses.length; i++) {\n if (((_a = postProcesses[i]) === null || _a === void 0 ? void 0 : _a.getClassName()) === \"ImageProcessingPostProcess\") {\n isIPPAlreadyPresent = true;\n break;\n }\n }\n }\n return isIPPAlreadyPresent;\n }\n /**\n * Internal, gets the first post proces.\n * @param postProcesses\n * @returns the first post process to be run on this camera.\n */\n _getFirstPostProcess(postProcesses) {\n for (let ppIndex = 0; ppIndex < postProcesses.length; ppIndex++) {\n if (postProcesses[ppIndex] !== null) {\n return postProcesses[ppIndex];\n }\n }\n return null;\n }\n /**\n * Marks the prepass renderer as dirty, triggering a check if the prepass is necessary for the next rendering.\n */\n markAsDirty() {\n this._isDirty = true;\n }\n /**\n * Enables a texture on the MultiRenderTarget for prepass\n * @param types\n */\n _enableTextures(types) {\n // For velocity : enable storage of previous matrices for instances\n this._scene.needsPreviousWorldMatrices = false;\n for (let i = 0; i < types.length; i++) {\n const type = types[i];\n if (this._textureIndices[type] === -1) {\n this._textureIndices[type] = this._mrtLayout.length;\n this._mrtLayout.push(type);\n this._mrtTypes.push(PrePassRenderer.TextureFormats[type].type);\n this._mrtFormats.push(PrePassRenderer.TextureFormats[type].format);\n this._mrtNames.push(PrePassRenderer.TextureFormats[type].name);\n this.mrtCount++;\n }\n if (type === 2) {\n this._scene.needsPreviousWorldMatrices = true;\n }\n }\n }\n /**\n * Makes sure that the prepass renderer is up to date if it has been dirtified.\n */\n update() {\n if (this._isDirty) {\n this._update();\n }\n }\n _update() {\n this._disable();\n let enablePrePass = false;\n this._scene.imageProcessingConfiguration.applyByPostProcess = false;\n if (this._scene._depthPeelingRenderer && this._scene.useOrderIndependentTransparency) {\n this._scene._depthPeelingRenderer.setPrePassRenderer(this);\n enablePrePass = true;\n }\n for (let i = 0; i < this._scene.materials.length; i++) {\n if (this._scene.materials[i].setPrePassRenderer(this)) {\n enablePrePass = true;\n }\n }\n if (enablePrePass) {\n this._setRenderTargetEnabled(this.defaultRT, true);\n }\n let postProcesses;\n for (let i = 0; i < this.renderTargets.length; i++) {\n if (this.renderTargets[i].renderTargetTexture) {\n postProcesses = this._getPostProcessesSource(this.renderTargets[i]);\n }\n else {\n const camera = this._scene.activeCamera;\n if (!camera) {\n continue;\n }\n postProcesses = camera._postProcesses;\n }\n if (!postProcesses) {\n continue;\n }\n postProcesses = postProcesses.filter((pp) => {\n return pp != null;\n });\n if (postProcesses) {\n for (let j = 0; j < postProcesses.length; j++) {\n if (postProcesses[j].setPrePassRenderer(this)) {\n this._setRenderTargetEnabled(this.renderTargets[i], true);\n enablePrePass = true;\n }\n }\n if (this._hasImageProcessing(postProcesses)) {\n this._scene.imageProcessingConfiguration.applyByPostProcess = true;\n }\n }\n }\n this._markAllMaterialsAsPrePassDirty();\n this._isDirty = false;\n if (enablePrePass) {\n this._enable();\n }\n }\n _markAllMaterialsAsPrePassDirty() {\n const materials = this._scene.materials;\n for (let i = 0; i < materials.length; i++) {\n materials[i].markAsDirty(Material.PrePassDirtyFlag);\n }\n }\n /**\n * Disposes the prepass renderer.\n */\n dispose() {\n for (let i = this.renderTargets.length - 1; i >= 0; i--) {\n this.renderTargets[i].dispose();\n }\n for (let i = 0; i < this._effectConfigurations.length; i++) {\n if (this._effectConfigurations[i].dispose) {\n this._effectConfigurations[i].dispose();\n }\n }\n }\n }\n /**\n * @internal\n */\n PrePassRenderer._SceneComponentInitialization = (_) => {\n throw _WarnImport(\"PrePassRendererSceneComponent\");\n };\n /**\n * Describes the types and formats of the textures used by the pre-pass renderer\n */\n PrePassRenderer.TextureFormats = [\n {\n purpose: 0,\n type: 2,\n format: 5,\n name: \"prePass_Irradiance\",\n },\n {\n purpose: 1,\n type: 2,\n format: 5,\n name: \"prePass_Position\",\n },\n {\n purpose: 2,\n type: 0,\n format: 5,\n name: \"prePass_Velocity\",\n },\n {\n purpose: 3,\n type: 0,\n format: 5,\n name: \"prePass_Reflectivity\",\n },\n {\n purpose: 4,\n type: 2,\n format: 5,\n name: \"prePass_Color\",\n },\n {\n purpose: 5,\n type: 1,\n format: 6,\n name: \"prePass_Depth\",\n },\n {\n purpose: 6,\n type: 2,\n format: 5,\n name: \"prePass_Normal\",\n },\n {\n purpose: 7,\n type: 0,\n format: 5,\n name: \"prePass_Albedo\",\n },\n ];\n\n Object.defineProperty(Scene.prototype, \"prePassRenderer\", {\n get: function () {\n return this._prePassRenderer;\n },\n set: function (value) {\n if (value && value.isSupported) {\n this._prePassRenderer = value;\n }\n },\n enumerable: true,\n configurable: true,\n });\n Scene.prototype.enablePrePassRenderer = function () {\n if (this._prePassRenderer) {\n return this._prePassRenderer;\n }\n this._prePassRenderer = new PrePassRenderer(this);\n if (!this._prePassRenderer.isSupported) {\n this._prePassRenderer = null;\n Logger.Error(\"PrePassRenderer needs WebGL 2 support.\\n\" + \"Maybe you tried to use the following features that need the PrePassRenderer :\\n\" + \" + Subsurface Scattering\");\n }\n return this._prePassRenderer;\n };\n Scene.prototype.disablePrePassRenderer = function () {\n if (!this._prePassRenderer) {\n return;\n }\n this._prePassRenderer.dispose();\n this._prePassRenderer = null;\n };\n /**\n * Defines the Geometry Buffer scene component responsible to manage a G-Buffer useful\n * in several rendering techniques.\n */\n class PrePassRendererSceneComponent {\n /**\n * Creates a new instance of the component for the given scene\n * @param scene Defines the scene to register the component in\n */\n constructor(scene) {\n /**\n * The component name helpful to identify the component in the list of scene components.\n */\n this.name = SceneComponentConstants.NAME_PREPASSRENDERER;\n this.scene = scene;\n }\n /**\n * Registers the component in a given scene\n */\n register() {\n this.scene._beforeCameraDrawStage.registerStep(SceneComponentConstants.STEP_BEFORECAMERADRAW_PREPASS, this, this._beforeCameraDraw);\n this.scene._afterCameraDrawStage.registerStep(SceneComponentConstants.STEP_AFTERCAMERADRAW_PREPASS, this, this._afterCameraDraw);\n this.scene._beforeRenderTargetDrawStage.registerStep(SceneComponentConstants.STEP_BEFORERENDERTARGETDRAW_PREPASS, this, this._beforeRenderTargetDraw);\n this.scene._afterRenderTargetDrawStage.registerStep(SceneComponentConstants.STEP_AFTERCAMERADRAW_PREPASS, this, this._afterRenderTargetDraw);\n this.scene._beforeClearStage.registerStep(SceneComponentConstants.STEP_BEFORECLEAR_PREPASS, this, this._beforeClearStage);\n this.scene._beforeRenderTargetClearStage.registerStep(SceneComponentConstants.STEP_BEFORERENDERTARGETCLEAR_PREPASS, this, this._beforeRenderTargetClearStage);\n this.scene._beforeRenderingMeshStage.registerStep(SceneComponentConstants.STEP_BEFORERENDERINGMESH_PREPASS, this, this._beforeRenderingMeshStage);\n this.scene._afterRenderingMeshStage.registerStep(SceneComponentConstants.STEP_AFTERRENDERINGMESH_PREPASS, this, this._afterRenderingMeshStage);\n }\n _beforeRenderTargetDraw(renderTarget, faceIndex, layer) {\n if (this.scene.prePassRenderer && !renderTarget.noPrePassRenderer) {\n this.scene.prePassRenderer._setRenderTarget(renderTarget._prePassRenderTarget);\n this.scene.prePassRenderer._beforeDraw(undefined, faceIndex, layer);\n }\n }\n _afterRenderTargetDraw(renderTarget, faceIndex, layer) {\n if (this.scene.prePassRenderer && !renderTarget.noPrePassRenderer) {\n this.scene.prePassRenderer._afterDraw(faceIndex, layer);\n }\n }\n _beforeRenderTargetClearStage(renderTarget) {\n if (this.scene.prePassRenderer && !renderTarget.noPrePassRenderer) {\n if (!renderTarget._prePassRenderTarget) {\n renderTarget._prePassRenderTarget = this.scene.prePassRenderer._createRenderTarget(renderTarget.name + \"_prePassRTT\", renderTarget);\n }\n this.scene.prePassRenderer._setRenderTarget(renderTarget._prePassRenderTarget);\n this.scene.prePassRenderer._clear();\n }\n }\n _beforeCameraDraw(camera) {\n if (this.scene.prePassRenderer) {\n this.scene.prePassRenderer._setRenderTarget(null);\n this.scene.prePassRenderer._beforeDraw(camera);\n }\n }\n _afterCameraDraw() {\n if (this.scene.prePassRenderer) {\n this.scene.prePassRenderer._afterDraw();\n }\n }\n _beforeClearStage() {\n if (this.scene.prePassRenderer) {\n this.scene.prePassRenderer._setRenderTarget(null);\n this.scene.prePassRenderer._clear();\n }\n }\n _beforeRenderingMeshStage(mesh, subMesh, batch, effect) {\n if (!effect) {\n return;\n }\n // Render to MRT\n const scene = mesh.getScene();\n if (scene.prePassRenderer) {\n scene.prePassRenderer.bindAttachmentsForEffect(effect, subMesh);\n }\n }\n _afterRenderingMeshStage(mesh) {\n const scene = mesh.getScene();\n if (scene.prePassRenderer) {\n scene.prePassRenderer.restoreAttachments();\n }\n }\n /**\n * Rebuilds the elements related to this component in case of\n * context lost for instance.\n */\n rebuild() {\n // Release textures first\n this.scene.disablePrePassRenderer();\n // Re-enable\n this.scene.enablePrePassRenderer();\n }\n /**\n * Disposes the component and the associated resources\n */\n dispose() {\n this.scene.disablePrePassRenderer();\n }\n }\n PrePassRenderer._SceneComponentInitialization = (scene) => {\n // Register the G Buffer component to the scene.\n let component = scene._getComponent(SceneComponentConstants.NAME_PREPASSRENDERER);\n if (!component) {\n component = new PrePassRendererSceneComponent(scene);\n scene._addComponent(component);\n }\n };\n\n const settingDefaults = {\n parentChangeBehavior: exports.eParentChangeBehavior.PRESERVE_POSITION,\n };\n const settingsKey = 'KBVIEWER_SETTINGS';\n const savedSettings = localStorage.getItem(settingsKey);\n const settings = savedSettings\n ? { ...settingDefaults, ...JSON.parse(savedSettings) }\n : { ...settingDefaults };\n class SettingsManagerService {\n static get(name) {\n return settings[name];\n }\n static set(name, value) {\n settings[name] = value;\n localStorage.setItem(settingsKey, JSON.stringify(settings));\n }\n }\n\n /**\n * Creates the VertexData for a Ribbon\n * @param options an object used to set the following optional parameters for the ribbon, required but can be empty\n * * pathArray array of paths, each of which an array of successive Vector3\n * * closeArray creates a seam between the first and the last paths of the pathArray, optional, default false\n * * closePath creates a seam between the first and the last points of each path of the path array, optional, default false\n * * offset a positive integer, only used when pathArray contains a single path (offset = 10 means the point 1 is joined to the point 11), default rounded half size of the pathArray length\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * * invertUV swaps in the U and V coordinates when applying a texture, optional, default false\n * * uvs a linear array, of length 2 * number of vertices, of custom UV values, optional\n * * colors a linear array, of length 4 * number of vertices, of custom color values, optional\n * @param options.pathArray\n * @param options.closeArray\n * @param options.closePath\n * @param options.offset\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param options.invertUV\n * @param options.uvs\n * @param options.colors\n * @returns the VertexData of the ribbon\n */\n function CreateRibbonVertexData(options) {\n let pathArray = options.pathArray;\n const closeArray = options.closeArray || false;\n const closePath = options.closePath || false;\n const invertUV = options.invertUV || false;\n const defaultOffset = Math.floor(pathArray[0].length / 2);\n let offset = options.offset || defaultOffset;\n offset = offset > defaultOffset ? defaultOffset : Math.floor(offset); // offset max allowed : defaultOffset\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\n const customUV = options.uvs;\n const customColors = options.colors;\n const positions = [];\n const indices = [];\n const normals = [];\n const uvs = [];\n const us = []; // us[path_id] = [uDist1, uDist2, uDist3 ... ] distances between points on path path_id\n const vs = []; // vs[i] = [vDist1, vDist2, vDist3, ... ] distances between points i of consecutive paths from pathArray\n const uTotalDistance = []; // uTotalDistance[p] : total distance of path p\n const vTotalDistance = []; // vTotalDistance[i] : total distance between points i of first and last path from pathArray\n let minlg; // minimal length among all paths from pathArray\n const lg = []; // array of path lengths : nb of vertex per path\n const idx = []; // array of path indexes : index of each path (first vertex) in the total vertex number\n let p; // path iterator\n let i; // point iterator\n let j; // point iterator\n // if single path in pathArray\n if (pathArray.length < 2) {\n const ar1 = [];\n const ar2 = [];\n for (i = 0; i < pathArray[0].length - offset; i++) {\n ar1.push(pathArray[0][i]);\n ar2.push(pathArray[0][i + offset]);\n }\n pathArray = [ar1, ar2];\n }\n // positions and horizontal distances (u)\n let idc = 0;\n const closePathCorr = closePath ? 1 : 0; // the final index will be +1 if closePath\n let path;\n let l;\n minlg = pathArray[0].length;\n let vectlg;\n let dist;\n for (p = 0; p < pathArray.length; p++) {\n uTotalDistance[p] = 0;\n us[p] = [0];\n path = pathArray[p];\n l = path.length;\n minlg = minlg < l ? minlg : l;\n j = 0;\n while (j < l) {\n positions.push(path[j].x, path[j].y, path[j].z);\n if (j > 0) {\n vectlg = path[j].subtract(path[j - 1]).length();\n dist = vectlg + uTotalDistance[p];\n us[p].push(dist);\n uTotalDistance[p] = dist;\n }\n j++;\n }\n if (closePath) {\n // an extra hidden vertex is added in the \"positions\" array\n j--;\n positions.push(path[0].x, path[0].y, path[0].z);\n vectlg = path[j].subtract(path[0]).length();\n dist = vectlg + uTotalDistance[p];\n us[p].push(dist);\n uTotalDistance[p] = dist;\n }\n lg[p] = l + closePathCorr;\n idx[p] = idc;\n idc += l + closePathCorr;\n }\n // vertical distances (v)\n let path1;\n let path2;\n let vertex1 = null;\n let vertex2 = null;\n for (i = 0; i < minlg + closePathCorr; i++) {\n vTotalDistance[i] = 0;\n vs[i] = [0];\n for (p = 0; p < pathArray.length - 1; p++) {\n path1 = pathArray[p];\n path2 = pathArray[p + 1];\n if (i === minlg) {\n // closePath\n vertex1 = path1[0];\n vertex2 = path2[0];\n }\n else {\n vertex1 = path1[i];\n vertex2 = path2[i];\n }\n vectlg = vertex2.subtract(vertex1).length();\n dist = vectlg + vTotalDistance[i];\n vs[i].push(dist);\n vTotalDistance[i] = dist;\n }\n if (closeArray && vertex2 && vertex1) {\n path1 = pathArray[p];\n path2 = pathArray[0];\n if (i === minlg) {\n // closePath\n vertex2 = path2[0];\n }\n vectlg = vertex2.subtract(vertex1).length();\n dist = vectlg + vTotalDistance[i];\n vTotalDistance[i] = dist;\n }\n }\n // uvs\n let u;\n let v;\n if (customUV) {\n for (p = 0; p < customUV.length; p++) {\n uvs.push(customUV[p].x, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - customUV[p].y : customUV[p].y);\n }\n }\n else {\n for (p = 0; p < pathArray.length; p++) {\n for (i = 0; i < minlg + closePathCorr; i++) {\n u = uTotalDistance[p] != 0.0 ? us[p][i] / uTotalDistance[p] : 0.0;\n v = vTotalDistance[i] != 0.0 ? vs[i][p] / vTotalDistance[i] : 0.0;\n if (invertUV) {\n uvs.push(v, u);\n }\n else {\n uvs.push(u, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - v : v);\n }\n }\n }\n }\n // indices\n p = 0; // path index\n let pi = 0; // positions array index\n let l1 = lg[p] - 1; // path1 length\n let l2 = lg[p + 1] - 1; // path2 length\n let min = l1 < l2 ? l1 : l2; // current path stop index\n let shft = idx[1] - idx[0]; // shift\n const path1nb = closeArray ? lg.length : lg.length - 1; // number of path1 to iterate\ton\n while (pi <= min && p < path1nb) {\n // stay under min and don't go over next to last path\n // draw two triangles between path1 (p1) and path2 (p2) : (p1.pi, p2.pi, p1.pi+1) and (p2.pi+1, p1.pi+1, p2.pi) clockwise\n indices.push(pi, pi + shft, pi + 1);\n indices.push(pi + shft + 1, pi + 1, pi + shft);\n pi += 1;\n if (pi === min) {\n // if end of one of two consecutive paths reached, go to next existing path\n p++;\n if (p === lg.length - 1) {\n // last path of pathArray reached <=> closeArray == true\n shft = idx[0] - idx[p];\n l1 = lg[p] - 1;\n l2 = lg[0] - 1;\n }\n else {\n shft = idx[p + 1] - idx[p];\n l1 = lg[p] - 1;\n l2 = lg[p + 1] - 1;\n }\n pi = idx[p];\n min = l1 < l2 ? l1 + pi : l2 + pi;\n }\n }\n // normals\n VertexData.ComputeNormals(positions, indices, normals);\n if (closePath) {\n // update both the first and last vertex normals to their average value\n let indexFirst = 0;\n let indexLast = 0;\n for (p = 0; p < pathArray.length; p++) {\n indexFirst = idx[p] * 3;\n if (p + 1 < pathArray.length) {\n indexLast = (idx[p + 1] - 1) * 3;\n }\n else {\n indexLast = normals.length - 3;\n }\n normals[indexFirst] = (normals[indexFirst] + normals[indexLast]) * 0.5;\n normals[indexFirst + 1] = (normals[indexFirst + 1] + normals[indexLast + 1]) * 0.5;\n normals[indexFirst + 2] = (normals[indexFirst + 2] + normals[indexLast + 2]) * 0.5;\n normals[indexLast] = normals[indexFirst];\n normals[indexLast + 1] = normals[indexFirst + 1];\n normals[indexLast + 2] = normals[indexFirst + 2];\n }\n }\n // sides\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\n // Colors\n let colors = null;\n if (customColors) {\n colors = new Float32Array(customColors.length * 4);\n for (let c = 0; c < customColors.length; c++) {\n colors[c * 4] = customColors[c].r;\n colors[c * 4 + 1] = customColors[c].g;\n colors[c * 4 + 2] = customColors[c].b;\n colors[c * 4 + 3] = customColors[c].a;\n }\n }\n // Result\n const vertexData = new VertexData();\n const positions32 = new Float32Array(positions);\n const normals32 = new Float32Array(normals);\n const uvs32 = new Float32Array(uvs);\n vertexData.indices = indices;\n vertexData.positions = positions32;\n vertexData.normals = normals32;\n vertexData.uvs = uvs32;\n if (colors) {\n vertexData.set(colors, VertexBuffer.ColorKind);\n }\n if (closePath) {\n vertexData._idx = idx;\n }\n return vertexData;\n }\n /**\n * Creates a ribbon mesh. The ribbon is a parametric shape. It has no predefined shape. Its final shape will depend on the input parameters\n * * The parameter `pathArray` is a required array of paths, what are each an array of successive Vector3. The pathArray parameter depicts the ribbon geometry\n * * The parameter `closeArray` (boolean, default false) creates a seam between the first and the last paths of the path array\n * * The parameter `closePath` (boolean, default false) creates a seam between the first and the last points of each path of the path array\n * * The parameter `offset` (positive integer, default : rounded half size of the pathArray length), is taken in account only if the `pathArray` is containing a single path\n * * It's the offset to join the points from the same path. Ex : offset = 10 means the point 1 is joined to the point 11\n * * The optional parameter `instance` is an instance of an existing Ribbon object to be updated with the passed `pathArray` parameter : https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#ribbon\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\n * * The optional parameter `invertUV` (boolean, default false) swaps in the geometry the U and V coordinates to apply a texture\n * * The parameter `uvs` is an optional flat array of `Vector2` to update/set each ribbon vertex with its own custom UV values instead of the computed ones\n * * The parameters `colors` is an optional flat array of `Color4` to set/update each ribbon vertex with its own custom color values\n * * Note that if you use the parameters `uvs` or `colors`, the passed arrays must be populated with the right number of elements, it is to say the number of ribbon vertices. Remember that if you set `closePath` to `true`, there's one extra vertex per path in the geometry\n * * Moreover, you can use the parameter `color` with `instance` (to update the ribbon), only if you previously used it at creation time\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param options.pathArray\n * @param options.closeArray\n * @param options.closePath\n * @param options.offset\n * @param options.updatable\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param options.instance\n * @param options.invertUV\n * @param options.uvs\n * @param options.colors\n * @param scene defines the hosting scene\n * @returns the ribbon mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param/ribbon_extra\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param\n */\n function CreateRibbon(name, options, scene = null) {\n const pathArray = options.pathArray;\n const closeArray = options.closeArray;\n const closePath = options.closePath;\n const sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n const instance = options.instance;\n const updatable = options.updatable;\n if (instance) {\n // existing ribbon instance update\n // positionFunction : ribbon case\n // only pathArray and sideOrientation parameters are taken into account for positions update\n const minimum = TmpVectors.Vector3[0].setAll(Number.MAX_VALUE);\n const maximum = TmpVectors.Vector3[1].setAll(-Number.MAX_VALUE);\n const positionFunction = (positions) => {\n let minlg = pathArray[0].length;\n const mesh = instance;\n let i = 0;\n const ns = mesh._originalBuilderSideOrientation === Mesh.DOUBLESIDE ? 2 : 1;\n for (let si = 1; si <= ns; ++si) {\n for (let p = 0; p < pathArray.length; ++p) {\n const path = pathArray[p];\n const l = path.length;\n minlg = minlg < l ? minlg : l;\n for (let j = 0; j < minlg; ++j) {\n const pathPoint = path[j];\n positions[i] = pathPoint.x;\n positions[i + 1] = pathPoint.y;\n positions[i + 2] = pathPoint.z;\n minimum.minimizeInPlaceFromFloats(pathPoint.x, pathPoint.y, pathPoint.z);\n maximum.maximizeInPlaceFromFloats(pathPoint.x, pathPoint.y, pathPoint.z);\n i += 3;\n }\n if (mesh._creationDataStorage && mesh._creationDataStorage.closePath) {\n const pathPoint = path[0];\n positions[i] = pathPoint.x;\n positions[i + 1] = pathPoint.y;\n positions[i + 2] = pathPoint.z;\n i += 3;\n }\n }\n }\n };\n const positions = instance.getVerticesData(VertexBuffer.PositionKind);\n positionFunction(positions);\n if (instance.hasBoundingInfo) {\n instance.getBoundingInfo().reConstruct(minimum, maximum, instance._worldMatrix);\n }\n else {\n instance.buildBoundingInfo(minimum, maximum, instance._worldMatrix);\n }\n instance.updateVerticesData(VertexBuffer.PositionKind, positions, false, false);\n if (options.colors) {\n const colors = instance.getVerticesData(VertexBuffer.ColorKind);\n for (let c = 0, colorIndex = 0; c < options.colors.length; c++, colorIndex += 4) {\n const color = options.colors[c];\n colors[colorIndex] = color.r;\n colors[colorIndex + 1] = color.g;\n colors[colorIndex + 2] = color.b;\n colors[colorIndex + 3] = color.a;\n }\n instance.updateVerticesData(VertexBuffer.ColorKind, colors, false, false);\n }\n if (options.uvs) {\n const uvs = instance.getVerticesData(VertexBuffer.UVKind);\n for (let i = 0; i < options.uvs.length; i++) {\n uvs[i * 2] = options.uvs[i].x;\n uvs[i * 2 + 1] = CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - options.uvs[i].y : options.uvs[i].y;\n }\n instance.updateVerticesData(VertexBuffer.UVKind, uvs, false, false);\n }\n if (!instance.areNormalsFrozen || instance.isFacetDataEnabled) {\n const indices = instance.getIndices();\n const normals = instance.getVerticesData(VertexBuffer.NormalKind);\n const params = instance.isFacetDataEnabled ? instance.getFacetDataParameters() : null;\n VertexData.ComputeNormals(positions, indices, normals, params);\n if (instance._creationDataStorage && instance._creationDataStorage.closePath) {\n let indexFirst = 0;\n let indexLast = 0;\n for (let p = 0; p < pathArray.length; p++) {\n indexFirst = instance._creationDataStorage.idx[p] * 3;\n if (p + 1 < pathArray.length) {\n indexLast = (instance._creationDataStorage.idx[p + 1] - 1) * 3;\n }\n else {\n indexLast = normals.length - 3;\n }\n normals[indexFirst] = (normals[indexFirst] + normals[indexLast]) * 0.5;\n normals[indexFirst + 1] = (normals[indexFirst + 1] + normals[indexLast + 1]) * 0.5;\n normals[indexFirst + 2] = (normals[indexFirst + 2] + normals[indexLast + 2]) * 0.5;\n normals[indexLast] = normals[indexFirst];\n normals[indexLast + 1] = normals[indexFirst + 1];\n normals[indexLast + 2] = normals[indexFirst + 2];\n }\n }\n if (!instance.areNormalsFrozen) {\n instance.updateVerticesData(VertexBuffer.NormalKind, normals, false, false);\n }\n }\n return instance;\n }\n else {\n // new ribbon creation\n const ribbon = new Mesh(name, scene);\n ribbon._originalBuilderSideOrientation = sideOrientation;\n ribbon._creationDataStorage = new _CreationDataStorage();\n const vertexData = CreateRibbonVertexData(options);\n if (closePath) {\n ribbon._creationDataStorage.idx = vertexData._idx;\n }\n ribbon._creationDataStorage.closePath = closePath;\n ribbon._creationDataStorage.closeArray = closeArray;\n vertexData.applyToMesh(ribbon, updatable);\n return ribbon;\n }\n }\n VertexData.CreateRibbon = CreateRibbonVertexData;\n Mesh.CreateRibbon = (name, pathArray, closeArray = false, closePath, offset, scene, updatable = false, sideOrientation, instance) => {\n return CreateRibbon(name, {\n pathArray: pathArray,\n closeArray: closeArray,\n closePath: closePath,\n offset: offset,\n updatable: updatable,\n sideOrientation: sideOrientation,\n instance: instance,\n }, scene);\n };\n\n /**\n * Creates the VertexData of the Disc or regular Polygon\n * @param options an object used to set the following optional parameters for the disc, required but can be empty\n * * radius the radius of the disc, optional default 0.5\n * * tessellation the number of polygon sides, optional, default 64\n * * arc a number from 0 to 1, to create an unclosed polygon based on the fraction of the circumference given by the arc value, optional, default 1\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.radius\n * @param options.tessellation\n * @param options.arc\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @returns the VertexData of the box\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n function CreateDiscVertexData(options) {\n const positions = new Array();\n const indices = new Array();\n const normals = new Array();\n const uvs = new Array();\n const radius = options.radius || 0.5;\n const tessellation = options.tessellation || 64;\n const arc = options.arc && (options.arc <= 0 || options.arc > 1) ? 1.0 : options.arc || 1.0;\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\n // positions and uvs\n positions.push(0, 0, 0); // disc center first\n uvs.push(0.5, 0.5);\n const theta = Math.PI * 2 * arc;\n const step = arc === 1 ? theta / tessellation : theta / (tessellation - 1);\n let a = 0;\n for (let t = 0; t < tessellation; t++) {\n const x = Math.cos(a);\n const y = Math.sin(a);\n const u = (x + 1) / 2;\n const v = (1 - y) / 2;\n positions.push(radius * x, radius * y, 0);\n uvs.push(u, CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - v : v);\n a += step;\n }\n if (arc === 1) {\n positions.push(positions[3], positions[4], positions[5]); // close the circle\n uvs.push(uvs[2], CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - uvs[3] : uvs[3]);\n }\n //indices\n const vertexNb = positions.length / 3;\n for (let i = 1; i < vertexNb - 1; i++) {\n indices.push(i + 1, 0, i);\n }\n // result\n VertexData.ComputeNormals(positions, indices, normals);\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\n const vertexData = new VertexData();\n vertexData.indices = indices;\n vertexData.positions = positions;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n return vertexData;\n }\n /**\n * Creates a plane polygonal mesh. By default, this is a disc\n * * The parameter `radius` sets the radius size (float) of the polygon (default 0.5)\n * * The parameter `tessellation` sets the number of polygon sides (positive integer, default 64). So a tessellation valued to 3 will build a triangle, to 4 a square, etc\n * * You can create an unclosed polygon with the parameter `arc` (positive float, default 1), valued between 0 and 1, what is the ratio of the circumference : 2 x PI x ratio\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param options.radius\n * @param options.tessellation\n * @param options.arc\n * @param options.updatable\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param scene defines the hosting scene\n * @returns the plane polygonal mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#disc-or-regular-polygon\n */\n function CreateDisc(name, options = {}, scene = null) {\n const disc = new Mesh(name, scene);\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n disc._originalBuilderSideOrientation = options.sideOrientation;\n const vertexData = CreateDiscVertexData(options);\n vertexData.applyToMesh(disc, options.updatable);\n return disc;\n }\n VertexData.CreateDisc = CreateDiscVertexData;\n Mesh.CreateDisc = (name, radius, tessellation, scene = null, updatable, sideOrientation) => {\n const options = {\n radius,\n tessellation,\n sideOrientation,\n updatable,\n };\n return CreateDisc(name, options, scene);\n };\n\n /**\n * Creates the VertexData for a tiled plane\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set/tiled_plane\n * @param options an object used to set the following optional parameters for the tiled plane, required but can be empty\n * * pattern a limited pattern arrangement depending on the number\n * * size of the box\n * * width of the box, overwrites size\n * * height of the box, overwrites size\n * * tileSize sets the width, height and depth of the tile to the value of size, optional default 1\n * * tileWidth sets the width (x direction) of the tile, overwrites the width set by size, optional, default size\n * * tileHeight sets the height (y direction) of the tile, overwrites the height set by size, optional, default size\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * alignHorizontal places whole tiles aligned to the center, left or right of a row\n * * alignVertical places whole tiles aligned to the center, left or right of a column\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * @param options.pattern\n * @param options.tileSize\n * @param options.tileWidth\n * @param options.tileHeight\n * @param options.size\n * @param options.width\n * @param options.height\n * @param options.alignHorizontal\n * @param options.alignVertical\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @returns the VertexData of the tiled plane\n */\n function CreateTiledPlaneVertexData(options) {\n const flipTile = options.pattern || Mesh.NO_FLIP;\n const tileWidth = options.tileWidth || options.tileSize || 1;\n const tileHeight = options.tileHeight || options.tileSize || 1;\n const alignH = options.alignHorizontal || 0;\n const alignV = options.alignVertical || 0;\n const width = options.width || options.size || 1;\n const tilesX = Math.floor(width / tileWidth);\n let offsetX = width - tilesX * tileWidth;\n const height = options.height || options.size || 1;\n const tilesY = Math.floor(height / tileHeight);\n let offsetY = height - tilesY * tileHeight;\n const halfWidth = (tileWidth * tilesX) / 2;\n const halfHeight = (tileHeight * tilesY) / 2;\n let adjustX = 0;\n let adjustY = 0;\n let startX = 0;\n let startY = 0;\n let endX = 0;\n let endY = 0;\n //Part Tiles\n if (offsetX > 0 || offsetY > 0) {\n startX = -halfWidth;\n startY = -halfHeight;\n endX = halfWidth;\n endY = halfHeight;\n switch (alignH) {\n case Mesh.CENTER:\n offsetX /= 2;\n startX -= offsetX;\n endX += offsetX;\n break;\n case Mesh.LEFT:\n endX += offsetX;\n adjustX = -offsetX / 2;\n break;\n case Mesh.RIGHT:\n startX -= offsetX;\n adjustX = offsetX / 2;\n break;\n }\n switch (alignV) {\n case Mesh.CENTER:\n offsetY /= 2;\n startY -= offsetY;\n endY += offsetY;\n break;\n case Mesh.BOTTOM:\n endY += offsetY;\n adjustY = -offsetY / 2;\n break;\n case Mesh.TOP:\n startY -= offsetY;\n adjustY = offsetY / 2;\n break;\n }\n }\n const positions = [];\n const normals = [];\n const uvBase = [];\n uvBase[0] = [0, 0, 1, 0, 1, 1, 0, 1];\n uvBase[1] = [0, 0, 1, 0, 1, 1, 0, 1];\n if (flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.ROTATE_ROW) {\n uvBase[1] = [1, 1, 0, 1, 0, 0, 1, 0];\n }\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.FLIP_ROW) {\n uvBase[1] = [1, 0, 0, 0, 0, 1, 1, 1];\n }\n if (flipTile === Mesh.FLIP_N_ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvBase[1] = [0, 1, 1, 1, 1, 0, 0, 0];\n }\n let uvs = [];\n const colors = [];\n const indices = [];\n let index = 0;\n for (let y = 0; y < tilesY; y++) {\n for (let x = 0; x < tilesX; x++) {\n positions.push(-halfWidth + x * tileWidth + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\n positions.push(-halfWidth + x * tileWidth + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_TILE) {\n uvs = uvs.concat(uvBase[((x % 2) + (y % 2)) % 2]);\n }\n else if (flipTile === Mesh.FLIP_ROW || flipTile === Mesh.ROTATE_ROW || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvs = uvs.concat(uvBase[y % 2]);\n }\n else {\n uvs = uvs.concat(uvBase[0]);\n }\n colors.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1);\n normals.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1);\n index += 4;\n }\n }\n //Part Tiles\n if (offsetX > 0 || offsetY > 0) {\n const partialBottomRow = offsetY > 0 && (alignV === Mesh.CENTER || alignV === Mesh.TOP);\n const partialTopRow = offsetY > 0 && (alignV === Mesh.CENTER || alignV === Mesh.BOTTOM);\n const partialLeftCol = offsetX > 0 && (alignH === Mesh.CENTER || alignH === Mesh.RIGHT);\n const partialRightCol = offsetX > 0 && (alignH === Mesh.CENTER || alignH === Mesh.LEFT);\n let uvPart = [];\n let a, b, c, d;\n //corners\n if (partialBottomRow && partialLeftCol) {\n //bottom left corner\n positions.push(startX + adjustX, startY + adjustY, 0);\n positions.push(-halfWidth + adjustX, startY + adjustY, 0);\n positions.push(-halfWidth + adjustX, startY + offsetY + adjustY, 0);\n positions.push(startX + adjustX, startY + offsetY + adjustY, 0);\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\n index += 4;\n a = 1 - offsetX / tileWidth;\n b = 1 - offsetY / tileHeight;\n c = 1;\n d = 1;\n uvPart = [a, b, c, b, c, d, a, d];\n if (flipTile === Mesh.ROTATE_ROW) {\n uvPart = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\n }\n if (flipTile === Mesh.FLIP_ROW) {\n uvPart = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\n }\n if (flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvPart = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\n }\n uvs = uvs.concat(uvPart);\n colors.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1);\n normals.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1);\n }\n if (partialBottomRow && partialRightCol) {\n //bottom right corner\n positions.push(halfWidth + adjustX, startY + adjustY, 0);\n positions.push(endX + adjustX, startY + adjustY, 0);\n positions.push(endX + adjustX, startY + offsetY + adjustY, 0);\n positions.push(halfWidth + adjustX, startY + offsetY + adjustY, 0);\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\n index += 4;\n a = 0;\n b = 1 - offsetY / tileHeight;\n c = offsetX / tileWidth;\n d = 1;\n uvPart = [a, b, c, b, c, d, a, d];\n if (flipTile === Mesh.ROTATE_ROW || (flipTile === Mesh.ROTATE_TILE && tilesX % 2 === 0)) {\n uvPart = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\n }\n if (flipTile === Mesh.FLIP_ROW || (flipTile === Mesh.FLIP_TILE && tilesX % 2 === 0)) {\n uvPart = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\n }\n if (flipTile === Mesh.FLIP_N_ROTATE_ROW || (flipTile === Mesh.FLIP_N_ROTATE_TILE && tilesX % 2 === 0)) {\n uvPart = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\n }\n uvs = uvs.concat(uvPart);\n colors.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1);\n normals.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1);\n }\n if (partialTopRow && partialLeftCol) {\n //top left corner\n positions.push(startX + adjustX, halfHeight + adjustY, 0);\n positions.push(-halfWidth + adjustX, halfHeight + adjustY, 0);\n positions.push(-halfWidth + adjustX, endY + adjustY, 0);\n positions.push(startX + adjustX, endY + adjustY, 0);\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\n index += 4;\n a = 1 - offsetX / tileWidth;\n b = 0;\n c = 1;\n d = offsetY / tileHeight;\n uvPart = [a, b, c, b, c, d, a, d];\n if ((flipTile === Mesh.ROTATE_ROW && tilesY % 2 === 1) || (flipTile === Mesh.ROTATE_TILE && tilesY % 1 === 0)) {\n uvPart = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\n }\n if ((flipTile === Mesh.FLIP_ROW && tilesY % 2 === 1) || (flipTile === Mesh.FLIP_TILE && tilesY % 2 === 0)) {\n uvPart = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\n }\n if ((flipTile === Mesh.FLIP_N_ROTATE_ROW && tilesY % 2 === 1) || (flipTile === Mesh.FLIP_N_ROTATE_TILE && tilesY % 2 === 0)) {\n uvPart = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\n }\n uvs = uvs.concat(uvPart);\n colors.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1);\n normals.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1);\n }\n if (partialTopRow && partialRightCol) {\n //top right corner\n positions.push(halfWidth + adjustX, halfHeight + adjustY, 0);\n positions.push(endX + adjustX, halfHeight + adjustY, 0);\n positions.push(endX + adjustX, endY + adjustY, 0);\n positions.push(halfWidth + adjustX, endY + adjustY, 0);\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\n index += 4;\n a = 0;\n b = 0;\n c = offsetX / tileWidth;\n d = offsetY / tileHeight;\n uvPart = [a, b, c, b, c, d, a, d];\n if ((flipTile === Mesh.ROTATE_ROW && tilesY % 2 === 1) || (flipTile === Mesh.ROTATE_TILE && (tilesY + tilesX) % 2 === 1)) {\n uvPart = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\n }\n if ((flipTile === Mesh.FLIP_ROW && tilesY % 2 === 1) || (flipTile === Mesh.FLIP_TILE && (tilesY + tilesX) % 2 === 1)) {\n uvPart = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\n }\n if ((flipTile === Mesh.FLIP_N_ROTATE_ROW && tilesY % 2 === 1) || (flipTile === Mesh.FLIP_N_ROTATE_TILE && (tilesY + tilesX) % 2 === 1)) {\n uvPart = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\n }\n uvs = uvs.concat(uvPart);\n colors.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1);\n normals.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1);\n }\n //part rows\n if (partialBottomRow) {\n const uvBaseBR = [];\n a = 0;\n b = 1 - offsetY / tileHeight;\n c = 1;\n d = 1;\n uvBaseBR[0] = [a, b, c, b, c, d, a, d];\n uvBaseBR[1] = [a, b, c, b, c, d, a, d];\n if (flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.ROTATE_ROW) {\n uvBaseBR[1] = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\n }\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.FLIP_ROW) {\n uvBaseBR[1] = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\n }\n if (flipTile === Mesh.FLIP_N_ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvBaseBR[1] = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\n }\n for (let x = 0; x < tilesX; x++) {\n positions.push(-halfWidth + x * tileWidth + adjustX, startY + adjustY, 0);\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, startY + adjustY, 0);\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, startY + offsetY + adjustY, 0);\n positions.push(-halfWidth + x * tileWidth + adjustX, startY + offsetY + adjustY, 0);\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\n index += 4;\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_TILE) {\n uvs = uvs.concat(uvBaseBR[(x + 1) % 2]);\n }\n else if (flipTile === Mesh.FLIP_ROW || flipTile === Mesh.ROTATE_ROW || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvs = uvs.concat(uvBaseBR[1]);\n }\n else {\n uvs = uvs.concat(uvBaseBR[0]);\n }\n colors.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1);\n normals.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1);\n }\n }\n if (partialTopRow) {\n const uvBaseTR = [];\n a = 0;\n b = 0;\n c = 1;\n d = offsetY / tileHeight;\n uvBaseTR[0] = [a, b, c, b, c, d, a, d];\n uvBaseTR[1] = [a, b, c, b, c, d, a, d];\n if (flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.ROTATE_ROW) {\n uvBaseTR[1] = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\n }\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.FLIP_ROW) {\n uvBaseTR[1] = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\n }\n if (flipTile === Mesh.FLIP_N_ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvBaseTR[1] = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\n }\n for (let x = 0; x < tilesX; x++) {\n positions.push(-halfWidth + x * tileWidth + adjustX, endY - offsetY + adjustY, 0);\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, endY - offsetY + adjustY, 0);\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, endY + adjustY, 0);\n positions.push(-halfWidth + x * tileWidth + adjustX, endY + adjustY, 0);\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\n index += 4;\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_TILE) {\n uvs = uvs.concat(uvBaseTR[(x + tilesY) % 2]);\n }\n else if (flipTile === Mesh.FLIP_ROW || flipTile === Mesh.ROTATE_ROW || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvs = uvs.concat(uvBaseTR[tilesY % 2]);\n }\n else {\n uvs = uvs.concat(uvBaseTR[0]);\n }\n colors.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1);\n normals.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1);\n }\n }\n if (partialLeftCol) {\n const uvBaseLC = [];\n a = 1 - offsetX / tileWidth;\n b = 0;\n c = 1;\n d = 1;\n uvBaseLC[0] = [a, b, c, b, c, d, a, d];\n uvBaseLC[1] = [a, b, c, b, c, d, a, d];\n if (flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.ROTATE_ROW) {\n uvBaseLC[1] = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\n }\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.FLIP_ROW) {\n uvBaseLC[1] = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\n }\n if (flipTile === Mesh.FLIP_N_ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvBaseLC[1] = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\n }\n for (let y = 0; y < tilesY; y++) {\n positions.push(startX + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\n positions.push(startX + offsetX + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\n positions.push(startX + offsetX + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\n positions.push(startX + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\n index += 4;\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_TILE) {\n uvs = uvs.concat(uvBaseLC[(y + 1) % 2]);\n }\n else if (flipTile === Mesh.FLIP_ROW || flipTile === Mesh.ROTATE_ROW || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvs = uvs.concat(uvBaseLC[y % 2]);\n }\n else {\n uvs = uvs.concat(uvBaseLC[0]);\n }\n colors.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1);\n normals.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1);\n }\n }\n if (partialRightCol) {\n const uvBaseRC = [];\n a = 0;\n b = 0;\n c = offsetX / tileHeight;\n d = 1;\n uvBaseRC[0] = [a, b, c, b, c, d, a, d];\n uvBaseRC[1] = [a, b, c, b, c, d, a, d];\n if (flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.ROTATE_ROW) {\n uvBaseRC[1] = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\n }\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.FLIP_ROW) {\n uvBaseRC[1] = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\n }\n if (flipTile === Mesh.FLIP_N_ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvBaseRC[1] = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\n }\n for (let y = 0; y < tilesY; y++) {\n positions.push(endX - offsetX + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\n positions.push(endX + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\n positions.push(endX + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\n positions.push(endX - offsetX + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\n index += 4;\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_TILE) {\n uvs = uvs.concat(uvBaseRC[(y + tilesX) % 2]);\n }\n else if (flipTile === Mesh.FLIP_ROW || flipTile === Mesh.ROTATE_ROW || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvs = uvs.concat(uvBaseRC[y % 2]);\n }\n else {\n uvs = uvs.concat(uvBaseRC[0]);\n }\n colors.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1);\n normals.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1);\n }\n }\n }\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\n // sides\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\n // Result\n const vertexData = new VertexData();\n vertexData.indices = indices;\n vertexData.positions = positions;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n const totalColors = sideOrientation === VertexData.DOUBLESIDE ? colors.concat(colors) : colors;\n vertexData.colors = totalColors;\n return vertexData;\n }\n /**\n * Creates a tiled plane mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set/tiled_plane\n * @param name defines the name of the mesh\n * @param options an object used to set the following optional parameters for the tiled plane, required but can be empty\n * * pattern a limited pattern arrangement depending on the number\n * * size of the box\n * * width of the box, overwrites size\n * * height of the box, overwrites size\n * * tileSize sets the width, height and depth of the tile to the value of size, optional default 1\n * * tileWidth sets the width (x direction) of the tile, overwrites the width set by size, optional, default size\n * * tileHeight sets the height (y direction) of the tile, overwrites the height set by size, optional, default size\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * alignHorizontal places whole tiles aligned to the center, left or right of a row\n * * alignVertical places whole tiles aligned to the center, left or right of a column\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.pattern\n * @param options.tileSize\n * @param options.tileWidth\n * @param options.tileHeight\n * @param options.size\n * @param options.width\n * @param options.height\n * @param options.alignHorizontal\n * @param options.alignVertical\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param options.updatable\n * @param scene defines the hosting scene\n * @returns the box mesh\n */\n function CreateTiledPlane(name, options, scene = null) {\n const plane = new Mesh(name, scene);\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n plane._originalBuilderSideOrientation = options.sideOrientation;\n const vertexData = CreateTiledPlaneVertexData(options);\n vertexData.applyToMesh(plane, options.updatable);\n return plane;\n }\n VertexData.CreateTiledPlane = CreateTiledPlaneVertexData;\n\n /**\n * Creates the VertexData for a tiled box\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set/tiled_box\n * @param options an object used to set the following optional parameters for the tiled box, required but can be empty\n * * pattern sets the rotation or reflection pattern for the tiles,\n * * size of the box\n * * width of the box, overwrites size\n * * height of the box, overwrites size\n * * depth of the box, overwrites size\n * * tileSize sets the size of a tile\n * * tileWidth sets the tile width and overwrites tileSize\n * * tileHeight sets the tile width and overwrites tileSize\n * * faceUV an array of 6 Vector4 elements used to set different images to each box side\n * * faceColors an array of 6 Color3 elements used to set different colors to each box side\n * * alignHorizontal places whole tiles aligned to the center, left or right of a row\n * * alignVertical places whole tiles aligned to the center, left or right of a column\n * @param options.pattern\n * @param options.size\n * @param options.width\n * @param options.height\n * @param options.depth\n * @param options.tileSize\n * @param options.tileWidth\n * @param options.tileHeight\n * @param options.faceUV\n * @param options.faceColors\n * @param options.alignHorizontal\n * @param options.alignVertical\n * @param options.sideOrientation\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * @returns the VertexData of the TiledBox\n */\n function CreateTiledBoxVertexData(options) {\n const nbFaces = 6;\n const faceUV = options.faceUV || new Array(6);\n const faceColors = options.faceColors;\n const flipTile = options.pattern || Mesh.NO_FLIP;\n const width = options.width || options.size || 1;\n const height = options.height || options.size || 1;\n const depth = options.depth || options.size || 1;\n const tileWidth = options.tileWidth || options.tileSize || 1;\n const tileHeight = options.tileHeight || options.tileSize || 1;\n const alignH = options.alignHorizontal || 0;\n const alignV = options.alignVertical || 0;\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\n // default face colors and UV if undefined\n for (let f = 0; f < nbFaces; f++) {\n if (faceUV[f] === undefined) {\n faceUV[f] = new Vector4(0, 0, 1, 1);\n }\n if (faceColors && faceColors[f] === undefined) {\n faceColors[f] = new Color4(1, 1, 1, 1);\n }\n }\n const halfWidth = width / 2;\n const halfHeight = height / 2;\n const halfDepth = depth / 2;\n const faceVertexData = [];\n for (let f = 0; f < 2; f++) {\n //front and back\n faceVertexData[f] = CreateTiledPlaneVertexData({\n pattern: flipTile,\n tileWidth: tileWidth,\n tileHeight: tileHeight,\n width: width,\n height: height,\n alignVertical: alignV,\n alignHorizontal: alignH,\n sideOrientation: sideOrientation,\n });\n }\n for (let f = 2; f < 4; f++) {\n //sides\n faceVertexData[f] = CreateTiledPlaneVertexData({\n pattern: flipTile,\n tileWidth: tileWidth,\n tileHeight: tileHeight,\n width: depth,\n height: height,\n alignVertical: alignV,\n alignHorizontal: alignH,\n sideOrientation: sideOrientation,\n });\n }\n let baseAlignV = alignV;\n if (alignV === Mesh.BOTTOM) {\n baseAlignV = Mesh.TOP;\n }\n else if (alignV === Mesh.TOP) {\n baseAlignV = Mesh.BOTTOM;\n }\n for (let f = 4; f < 6; f++) {\n //top and bottom\n faceVertexData[f] = CreateTiledPlaneVertexData({\n pattern: flipTile,\n tileWidth: tileWidth,\n tileHeight: tileHeight,\n width: width,\n height: depth,\n alignVertical: baseAlignV,\n alignHorizontal: alignH,\n sideOrientation: sideOrientation,\n });\n }\n let positions = [];\n let normals = [];\n let uvs = [];\n let indices = [];\n const colors = [];\n const facePositions = [];\n const faceNormals = [];\n const newFaceUV = [];\n let lu = 0;\n let li = 0;\n for (let f = 0; f < nbFaces; f++) {\n const len = faceVertexData[f].positions.length;\n facePositions[f] = [];\n faceNormals[f] = [];\n for (let p = 0; p < len / 3; p++) {\n facePositions[f].push(new Vector3(faceVertexData[f].positions[3 * p], faceVertexData[f].positions[3 * p + 1], faceVertexData[f].positions[3 * p + 2]));\n faceNormals[f].push(new Vector3(faceVertexData[f].normals[3 * p], faceVertexData[f].normals[3 * p + 1], faceVertexData[f].normals[3 * p + 2]));\n }\n // uvs\n lu = faceVertexData[f].uvs.length;\n newFaceUV[f] = [];\n for (let i = 0; i < lu; i += 2) {\n newFaceUV[f][i] = faceUV[f].x + (faceUV[f].z - faceUV[f].x) * faceVertexData[f].uvs[i];\n newFaceUV[f][i + 1] = faceUV[f].y + (faceUV[f].w - faceUV[f].y) * faceVertexData[f].uvs[i + 1];\n if (CompatibilityOptions.UseOpenGLOrientationForUV) {\n newFaceUV[f][i + 1] = 1.0 - newFaceUV[f][i + 1];\n }\n }\n uvs = uvs.concat(newFaceUV[f]);\n indices = indices.concat(faceVertexData[f].indices.map((x) => x + li));\n li += facePositions[f].length;\n if (faceColors) {\n for (let c = 0; c < 4; c++) {\n colors.push(faceColors[f].r, faceColors[f].g, faceColors[f].b, faceColors[f].a);\n }\n }\n }\n const vec0 = new Vector3(0, 0, halfDepth);\n const mtrx0 = Matrix.RotationY(Math.PI);\n positions = facePositions[0]\n .map((entry) => Vector3.TransformNormal(entry, mtrx0).add(vec0))\n .map((entry) => [entry.x, entry.y, entry.z])\n .reduce((accumulator, currentValue) => accumulator.concat(currentValue), []);\n normals = faceNormals[0]\n .map((entry) => Vector3.TransformNormal(entry, mtrx0))\n .map((entry) => [entry.x, entry.y, entry.z])\n .reduce((accumulator, currentValue) => accumulator.concat(currentValue), []);\n positions = positions.concat(facePositions[1]\n .map((entry) => entry.subtract(vec0))\n .map((entry) => [entry.x, entry.y, entry.z])\n .reduce((accumulator, currentValue) => accumulator.concat(currentValue), []));\n normals = normals.concat(faceNormals[1].map((entry) => [entry.x, entry.y, entry.z]).reduce((accumulator, currentValue) => accumulator.concat(currentValue), []));\n const vec2 = new Vector3(halfWidth, 0, 0);\n const mtrx2 = Matrix.RotationY(-Math.PI / 2);\n positions = positions.concat(facePositions[2]\n .map((entry) => Vector3.TransformNormal(entry, mtrx2).add(vec2))\n .map((entry) => [entry.x, entry.y, entry.z])\n .reduce((accumulator, currentValue) => accumulator.concat(currentValue), []));\n normals = normals.concat(faceNormals[2]\n .map((entry) => Vector3.TransformNormal(entry, mtrx2))\n .map((entry) => [entry.x, entry.y, entry.z])\n .reduce((accumulator, currentValue) => accumulator.concat(currentValue), []));\n const mtrx3 = Matrix.RotationY(Math.PI / 2);\n positions = positions.concat(facePositions[3]\n .map((entry) => Vector3.TransformNormal(entry, mtrx3).subtract(vec2))\n .map((entry) => [entry.x, entry.y, entry.z])\n .reduce((accumulator, currentValue) => accumulator.concat(currentValue), []));\n normals = normals.concat(faceNormals[3]\n .map((entry) => Vector3.TransformNormal(entry, mtrx3))\n .map((entry) => [entry.x, entry.y, entry.z])\n .reduce((accumulator, currentValue) => accumulator.concat(currentValue), []));\n const vec4 = new Vector3(0, halfHeight, 0);\n const mtrx4 = Matrix.RotationX(Math.PI / 2);\n positions = positions.concat(facePositions[4]\n .map((entry) => Vector3.TransformNormal(entry, mtrx4).add(vec4))\n .map((entry) => [entry.x, entry.y, entry.z])\n .reduce((accumulator, currentValue) => accumulator.concat(currentValue), []));\n normals = normals.concat(faceNormals[4]\n .map((entry) => Vector3.TransformNormal(entry, mtrx4))\n .map((entry) => [entry.x, entry.y, entry.z])\n .reduce((accumulator, currentValue) => accumulator.concat(currentValue), []));\n const mtrx5 = Matrix.RotationX(-Math.PI / 2);\n positions = positions.concat(facePositions[5]\n .map((entry) => Vector3.TransformNormal(entry, mtrx5).subtract(vec4))\n .map((entry) => [entry.x, entry.y, entry.z])\n .reduce((accumulator, currentValue) => accumulator.concat(currentValue), []));\n normals = normals.concat(faceNormals[5]\n .map((entry) => Vector3.TransformNormal(entry, mtrx5))\n .map((entry) => [entry.x, entry.y, entry.z])\n .reduce((accumulator, currentValue) => accumulator.concat(currentValue), []));\n // sides\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs);\n // Result\n const vertexData = new VertexData();\n vertexData.indices = indices;\n vertexData.positions = positions;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n if (faceColors) {\n const totalColors = sideOrientation === VertexData.DOUBLESIDE ? colors.concat(colors) : colors;\n vertexData.colors = totalColors;\n }\n return vertexData;\n }\n /**\n * Creates a tiled box mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set/tiled_box\n * @param name defines the name of the mesh\n * @param options an object used to set the following optional parameters for the tiled box, required but can be empty\n * * pattern sets the rotation or reflection pattern for the tiles,\n * * size of the box\n * * width of the box, overwrites size\n * * height of the box, overwrites size\n * * depth of the box, overwrites size\n * * tileSize sets the size of a tile\n * * tileWidth sets the tile width and overwrites tileSize\n * * tileHeight sets the tile width and overwrites tileSize\n * * faceUV an array of 6 Vector4 elements used to set different images to each box side\n * * faceColors an array of 6 Color3 elements used to set different colors to each box side\n * * alignHorizontal places whole tiles aligned to the center, left or right of a row\n * * alignVertical places whole tiles aligned to the center, left or right of a column\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * @param options.pattern\n * @param options.width\n * @param options.height\n * @param options.depth\n * @param options.tileSize\n * @param options.tileWidth\n * @param options.tileHeight\n * @param options.alignHorizontal\n * @param options.alignVertical\n * @param options.faceUV\n * @param options.faceColors\n * @param options.sideOrientation\n * @param options.updatable\n * @param scene defines the hosting scene\n * @returns the box mesh\n */\n function CreateTiledBox(name, options, scene = null) {\n const box = new Mesh(name, scene);\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n box._originalBuilderSideOrientation = options.sideOrientation;\n const vertexData = CreateTiledBoxVertexData(options);\n vertexData.applyToMesh(box, options.updatable);\n return box;\n }\n VertexData.CreateTiledBox = CreateTiledBoxVertexData;\n\n // based on http://code.google.com/p/away3d/source/browse/trunk/fp10/Away3D/src/away3d/primitives/TorusKnot.as?spec=svn2473&r=2473\n /**\n * Creates the VertexData for a TorusKnot\n * @param options an object used to set the following optional parameters for the TorusKnot, required but can be empty\n * * radius the radius of the torus knot, optional, default 2\n * * tube the thickness of the tube, optional, default 0.5\n * * radialSegments the number of sides on each tube segments, optional, default 32\n * * tubularSegments the number of tubes to decompose the knot into, optional, default 32\n * * p the number of windings around the z axis, optional, default 2\n * * q the number of windings around the x axis, optional, default 3\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.radius\n * @param options.tube\n * @param options.radialSegments\n * @param options.tubularSegments\n * @param options.p\n * @param options.q\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @returns the VertexData of the Torus Knot\n */\n function CreateTorusKnotVertexData(options) {\n const indices = new Array();\n const positions = new Array();\n const normals = new Array();\n const uvs = new Array();\n const radius = options.radius || 2;\n const tube = options.tube || 0.5;\n const radialSegments = options.radialSegments || 32;\n const tubularSegments = options.tubularSegments || 32;\n const p = options.p || 2;\n const q = options.q || 3;\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\n // Helper\n const getPos = (angle) => {\n const cu = Math.cos(angle);\n const su = Math.sin(angle);\n const quOverP = (q / p) * angle;\n const cs = Math.cos(quOverP);\n const tx = radius * (2 + cs) * 0.5 * cu;\n const ty = radius * (2 + cs) * su * 0.5;\n const tz = radius * Math.sin(quOverP) * 0.5;\n return new Vector3(tx, ty, tz);\n };\n // Vertices\n let i;\n let j;\n for (i = 0; i <= radialSegments; i++) {\n const modI = i % radialSegments;\n const u = (modI / radialSegments) * 2 * p * Math.PI;\n const p1 = getPos(u);\n const p2 = getPos(u + 0.01);\n const tang = p2.subtract(p1);\n let n = p2.add(p1);\n const bitan = Vector3.Cross(tang, n);\n n = Vector3.Cross(bitan, tang);\n bitan.normalize();\n n.normalize();\n for (j = 0; j < tubularSegments; j++) {\n const modJ = j % tubularSegments;\n const v = (modJ / tubularSegments) * 2 * Math.PI;\n const cx = -tube * Math.cos(v);\n const cy = tube * Math.sin(v);\n positions.push(p1.x + cx * n.x + cy * bitan.x);\n positions.push(p1.y + cx * n.y + cy * bitan.y);\n positions.push(p1.z + cx * n.z + cy * bitan.z);\n uvs.push(i / radialSegments);\n uvs.push(CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - j / tubularSegments : j / tubularSegments);\n }\n }\n for (i = 0; i < radialSegments; i++) {\n for (j = 0; j < tubularSegments; j++) {\n const jNext = (j + 1) % tubularSegments;\n const a = i * tubularSegments + j;\n const b = (i + 1) * tubularSegments + j;\n const c = (i + 1) * tubularSegments + jNext;\n const d = i * tubularSegments + jNext;\n indices.push(d);\n indices.push(b);\n indices.push(a);\n indices.push(d);\n indices.push(c);\n indices.push(b);\n }\n }\n // Normals\n VertexData.ComputeNormals(positions, indices, normals);\n // Sides\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\n // Result\n const vertexData = new VertexData();\n vertexData.indices = indices;\n vertexData.positions = positions;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n return vertexData;\n }\n /**\n * Creates a torus knot mesh\n * * The parameter `radius` sets the global radius size (float) of the torus knot (default 2)\n * * The parameter `radialSegments` sets the number of sides on each tube segments (positive integer, default 32)\n * * The parameter `tubularSegments` sets the number of tubes to decompose the knot into (positive integer, default 32)\n * * The parameters `p` and `q` are the number of windings on each axis (positive integers, default 2 and 3)\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param options.radius\n * @param options.tube\n * @param options.radialSegments\n * @param options.tubularSegments\n * @param options.p\n * @param options.q\n * @param options.updatable\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param scene defines the hosting scene\n * @returns the torus knot mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#torus-knot\n */\n function CreateTorusKnot(name, options = {}, scene) {\n const torusKnot = new Mesh(name, scene);\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n torusKnot._originalBuilderSideOrientation = options.sideOrientation;\n const vertexData = CreateTorusKnotVertexData(options);\n vertexData.applyToMesh(torusKnot, options.updatable);\n return torusKnot;\n }\n VertexData.CreateTorusKnot = CreateTorusKnotVertexData;\n Mesh.CreateTorusKnot = (name, radius, tube, radialSegments, tubularSegments, p, q, scene, updatable, sideOrientation) => {\n const options = {\n radius,\n tube,\n radialSegments,\n tubularSegments,\n p,\n q,\n sideOrientation,\n updatable,\n };\n return CreateTorusKnot(name, options, scene);\n };\n\n /**\n * Vector2 wth index property\n */\n class IndexedVector2 extends Vector2 {\n constructor(original, \n /** Index of the vector2 */\n index) {\n super(original.x, original.y);\n this.index = index;\n }\n }\n /**\n * Defines points to create a polygon\n */\n class PolygonPoints {\n constructor() {\n this.elements = new Array();\n }\n add(originalPoints) {\n const result = new Array();\n originalPoints.forEach((point) => {\n const newPoint = new IndexedVector2(point, this.elements.length);\n result.push(newPoint);\n this.elements.push(newPoint);\n });\n return result;\n }\n computeBounds() {\n const lmin = new Vector2(this.elements[0].x, this.elements[0].y);\n const lmax = new Vector2(this.elements[0].x, this.elements[0].y);\n this.elements.forEach((point) => {\n // x\n if (point.x < lmin.x) {\n lmin.x = point.x;\n }\n else if (point.x > lmax.x) {\n lmax.x = point.x;\n }\n // y\n if (point.y < lmin.y) {\n lmin.y = point.y;\n }\n else if (point.y > lmax.y) {\n lmax.y = point.y;\n }\n });\n return {\n min: lmin,\n max: lmax,\n width: lmax.x - lmin.x,\n height: lmax.y - lmin.y,\n };\n }\n }\n /**\n * Builds a polygon\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param/polyMeshBuilder\n */\n class PolygonMeshBuilder {\n _addToepoint(points) {\n for (const p of points) {\n this._epoints.push(p.x, p.y);\n }\n }\n /**\n * Creates a PolygonMeshBuilder\n * @param name name of the builder\n * @param contours Path of the polygon\n * @param scene scene to add to when creating the mesh\n * @param earcutInjection can be used to inject your own earcut reference\n */\n constructor(name, contours, scene, earcutInjection = earcut) {\n this._points = new PolygonPoints();\n this._outlinepoints = new PolygonPoints();\n this._holes = new Array();\n this._epoints = new Array();\n this._eholes = new Array();\n this.bjsEarcut = earcutInjection;\n this._name = name;\n this._scene = scene || EngineStore.LastCreatedScene;\n let points;\n if (contours instanceof Path2) {\n points = contours.getPoints();\n }\n else {\n points = contours;\n }\n this._addToepoint(points);\n this._points.add(points);\n this._outlinepoints.add(points);\n if (typeof this.bjsEarcut === \"undefined\") {\n Logger.Warn(\"Earcut was not found, the polygon will not be built.\");\n }\n }\n /**\n * Adds a hole within the polygon\n * @param hole Array of points defining the hole\n * @returns this\n */\n addHole(hole) {\n this._points.add(hole);\n const holepoints = new PolygonPoints();\n holepoints.add(hole);\n this._holes.push(holepoints);\n this._eholes.push(this._epoints.length / 2);\n this._addToepoint(hole);\n return this;\n }\n /**\n * Creates the polygon\n * @param updatable If the mesh should be updatable\n * @param depth The depth of the mesh created\n * @param smoothingThreshold Dot product threshold for smoothed normals\n * @returns the created mesh\n */\n build(updatable = false, depth = 0, smoothingThreshold = 2) {\n const result = new Mesh(this._name, this._scene);\n const vertexData = this.buildVertexData(depth, smoothingThreshold);\n result.setVerticesData(VertexBuffer.PositionKind, vertexData.positions, updatable);\n result.setVerticesData(VertexBuffer.NormalKind, vertexData.normals, updatable);\n result.setVerticesData(VertexBuffer.UVKind, vertexData.uvs, updatable);\n result.setIndices(vertexData.indices);\n return result;\n }\n /**\n * Creates the polygon\n * @param depth The depth of the mesh created\n * @param smoothingThreshold Dot product threshold for smoothed normals\n * @returns the created VertexData\n */\n buildVertexData(depth = 0, smoothingThreshold = 2) {\n const result = new VertexData();\n const normals = new Array();\n const positions = new Array();\n const uvs = new Array();\n const bounds = this._points.computeBounds();\n this._points.elements.forEach((p) => {\n normals.push(0, 1.0, 0);\n positions.push(p.x, 0, p.y);\n uvs.push((p.x - bounds.min.x) / bounds.width, (p.y - bounds.min.y) / bounds.height);\n });\n const indices = new Array();\n const res = this.bjsEarcut(this._epoints, this._eholes, 2);\n for (let i = 0; i < res.length; i++) {\n indices.push(res[i]);\n }\n if (depth > 0) {\n const positionscount = positions.length / 3; //get the current pointcount\n this._points.elements.forEach((p) => {\n //add the elements at the depth\n normals.push(0, -1.0, 0);\n positions.push(p.x, -depth, p.y);\n uvs.push(1 - (p.x - bounds.min.x) / bounds.width, 1 - (p.y - bounds.min.y) / bounds.height);\n });\n const totalCount = indices.length;\n for (let i = 0; i < totalCount; i += 3) {\n const i0 = indices[i + 0];\n const i1 = indices[i + 1];\n const i2 = indices[i + 2];\n indices.push(i2 + positionscount);\n indices.push(i1 + positionscount);\n indices.push(i0 + positionscount);\n }\n //Add the sides\n this._addSide(positions, normals, uvs, indices, bounds, this._outlinepoints, depth, false, smoothingThreshold);\n this._holes.forEach((hole) => {\n this._addSide(positions, normals, uvs, indices, bounds, hole, depth, true, smoothingThreshold);\n });\n }\n result.indices = indices;\n result.positions = positions;\n result.normals = normals;\n result.uvs = uvs;\n return result;\n }\n /**\n * Adds a side to the polygon\n * @param positions points that make the polygon\n * @param normals normals of the polygon\n * @param uvs uvs of the polygon\n * @param indices indices of the polygon\n * @param bounds bounds of the polygon\n * @param points points of the polygon\n * @param depth depth of the polygon\n * @param flip flip of the polygon\n * @param smoothingThreshold\n */\n _addSide(positions, normals, uvs, indices, bounds, points, depth, flip, smoothingThreshold) {\n let startIndex = positions.length / 3;\n let ulength = 0;\n for (let i = 0; i < points.elements.length; i++) {\n const p = points.elements[i];\n const p1 = points.elements[(i + 1) % points.elements.length];\n positions.push(p.x, 0, p.y);\n positions.push(p.x, -depth, p.y);\n positions.push(p1.x, 0, p1.y);\n positions.push(p1.x, -depth, p1.y);\n const p0 = points.elements[(i + points.elements.length - 1) % points.elements.length];\n const p2 = points.elements[(i + 2) % points.elements.length];\n let vc = new Vector3(-(p1.y - p.y), 0, p1.x - p.x);\n let vp = new Vector3(-(p.y - p0.y), 0, p.x - p0.x);\n let vn = new Vector3(-(p2.y - p1.y), 0, p2.x - p1.x);\n if (!flip) {\n vc = vc.scale(-1);\n vp = vp.scale(-1);\n vn = vn.scale(-1);\n }\n const vc_norm = vc.normalizeToNew();\n let vp_norm = vp.normalizeToNew();\n let vn_norm = vn.normalizeToNew();\n const dotp = Vector3.Dot(vp_norm, vc_norm);\n if (dotp > smoothingThreshold) {\n if (dotp < Epsilon - 1) {\n vp_norm = new Vector3(p.x, 0, p.y).subtract(new Vector3(p1.x, 0, p1.y)).normalize();\n }\n else {\n // cheap average weighed by side length\n vp_norm = vp.add(vc).normalize();\n }\n }\n else {\n vp_norm = vc_norm;\n }\n const dotn = Vector3.Dot(vn, vc);\n if (dotn > smoothingThreshold) {\n if (dotn < Epsilon - 1) {\n // back to back\n vn_norm = new Vector3(p1.x, 0, p1.y).subtract(new Vector3(p.x, 0, p.y)).normalize();\n }\n else {\n // cheap average weighed by side length\n vn_norm = vn.add(vc).normalize();\n }\n }\n else {\n vn_norm = vc_norm;\n }\n uvs.push(ulength / bounds.width, 0);\n uvs.push(ulength / bounds.width, 1);\n ulength += vc.length();\n uvs.push(ulength / bounds.width, 0);\n uvs.push(ulength / bounds.width, 1);\n normals.push(vp_norm.x, vp_norm.y, vp_norm.z);\n normals.push(vp_norm.x, vp_norm.y, vp_norm.z);\n normals.push(vn_norm.x, vn_norm.y, vn_norm.z);\n normals.push(vn_norm.x, vn_norm.y, vn_norm.z);\n if (!flip) {\n indices.push(startIndex);\n indices.push(startIndex + 1);\n indices.push(startIndex + 2);\n indices.push(startIndex + 1);\n indices.push(startIndex + 3);\n indices.push(startIndex + 2);\n }\n else {\n indices.push(startIndex);\n indices.push(startIndex + 2);\n indices.push(startIndex + 1);\n indices.push(startIndex + 1);\n indices.push(startIndex + 2);\n indices.push(startIndex + 3);\n }\n startIndex += 4;\n }\n }\n }\n\n /**\n * Creates the VertexData for an irregular Polygon in the XoZ plane using a mesh built by polygonTriangulation.build()\n * All parameters are provided by CreatePolygon as needed\n * @param polygon a mesh built from polygonTriangulation.build()\n * @param sideOrientation takes the values Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * @param fUV an array of Vector4 elements used to set different images to the top, rings and bottom respectively\n * @param fColors an array of Color3 elements used to set different colors to the top, rings and bottom respectively\n * @param frontUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * @param backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param wrp a boolean, default false, when true and fUVs used texture is wrapped around all sides, when false texture is applied side\n * @returns the VertexData of the Polygon\n */\n function CreatePolygonVertexData(polygon, sideOrientation, fUV, fColors, frontUVs, backUVs, wrp) {\n const faceUV = fUV || new Array(3);\n const faceColors = fColors;\n const colors = [];\n const wrap = wrp || false;\n // default face colors and UV if undefined\n for (let f = 0; f < 3; f++) {\n if (faceUV[f] === undefined) {\n faceUV[f] = new Vector4(0, 0, 1, 1);\n }\n if (faceColors && faceColors[f] === undefined) {\n faceColors[f] = new Color4(1, 1, 1, 1);\n }\n }\n const positions = polygon.getVerticesData(VertexBuffer.PositionKind);\n const normals = polygon.getVerticesData(VertexBuffer.NormalKind);\n const uvs = polygon.getVerticesData(VertexBuffer.UVKind);\n const indices = polygon.getIndices();\n const startIndex = positions.length / 9;\n let disp = 0;\n let distX = 0;\n let distZ = 0;\n let dist = 0;\n let totalLen = 0;\n const cumulate = [0];\n if (wrap) {\n for (let idx = startIndex; idx < positions.length / 3; idx += 4) {\n distX = positions[3 * (idx + 2)] - positions[3 * idx];\n distZ = positions[3 * (idx + 2) + 2] - positions[3 * idx + 2];\n dist = Math.sqrt(distX * distX + distZ * distZ);\n totalLen += dist;\n cumulate.push(totalLen);\n }\n }\n // set face colours and textures\n let idx = 0;\n let face = 0;\n for (let index = 0; index < normals.length; index += 3) {\n //Edge Face no. 1\n if (Math.abs(normals[index + 1]) < 0.001) {\n face = 1;\n }\n //Top Face no. 0\n if (Math.abs(normals[index + 1] - 1) < 0.001) {\n face = 0;\n }\n //Bottom Face no. 2\n if (Math.abs(normals[index + 1] + 1) < 0.001) {\n face = 2;\n }\n idx = index / 3;\n if (face === 1) {\n disp = idx - startIndex;\n if (disp % 4 < 1.5) {\n if (wrap) {\n uvs[2 * idx] = faceUV[face].x + ((faceUV[face].z - faceUV[face].x) * cumulate[Math.floor(disp / 4)]) / totalLen;\n }\n else {\n uvs[2 * idx] = faceUV[face].x;\n }\n }\n else {\n if (wrap) {\n uvs[2 * idx] = faceUV[face].x + ((faceUV[face].z - faceUV[face].x) * cumulate[Math.floor(disp / 4) + 1]) / totalLen;\n }\n else {\n uvs[2 * idx] = faceUV[face].z;\n }\n }\n if (disp % 2 === 0) {\n uvs[2 * idx + 1] = CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - faceUV[face].w : faceUV[face].w;\n }\n else {\n uvs[2 * idx + 1] = CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - faceUV[face].y : faceUV[face].y;\n }\n }\n else {\n uvs[2 * idx] = (1 - uvs[2 * idx]) * faceUV[face].x + uvs[2 * idx] * faceUV[face].z;\n uvs[2 * idx + 1] = (1 - uvs[2 * idx + 1]) * faceUV[face].y + uvs[2 * idx + 1] * faceUV[face].w;\n if (CompatibilityOptions.UseOpenGLOrientationForUV) {\n uvs[2 * idx + 1] = 1.0 - uvs[2 * idx + 1];\n }\n }\n if (faceColors) {\n colors.push(faceColors[face].r, faceColors[face].g, faceColors[face].b, faceColors[face].a);\n }\n }\n // sides\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, frontUVs, backUVs);\n // Result\n const vertexData = new VertexData();\n vertexData.indices = indices;\n vertexData.positions = positions;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n if (faceColors) {\n const totalColors = sideOrientation === VertexData.DOUBLESIDE ? colors.concat(colors) : colors;\n vertexData.colors = totalColors;\n }\n return vertexData;\n }\n /**\n * Creates a polygon mesh\n * The polygon's shape will depend on the input parameters and is constructed parallel to a ground mesh\n * * The parameter `shape` is a required array of successive Vector3 representing the corners of the polygon in th XoZ plane, that is y = 0 for all vectors\n * * You can set the mesh side orientation with the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4)\n * * Remember you can only change the shape positions, not their number when updating a polygon\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param options.shape\n * @param options.holes\n * @param options.depth\n * @param options.smoothingThreshold\n * @param options.faceUV\n * @param options.faceColors\n * @param options.updatable\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param options.wrap\n * @param scene defines the hosting scene\n * @param earcutInjection can be used to inject your own earcut reference\n * @returns the polygon mesh\n */\n function CreatePolygon(name, options, scene = null, earcutInjection = earcut) {\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n const shape = options.shape;\n const holes = options.holes || [];\n const depth = options.depth || 0;\n const smoothingThreshold = options.smoothingThreshold || 2;\n const contours = [];\n let hole = [];\n for (let i = 0; i < shape.length; i++) {\n contours[i] = new Vector2(shape[i].x, shape[i].z);\n }\n const epsilon = 0.00000001;\n if (contours[0].equalsWithEpsilon(contours[contours.length - 1], epsilon)) {\n contours.pop();\n }\n const polygonTriangulation = new PolygonMeshBuilder(name, contours, scene || EngineStore.LastCreatedScene, earcutInjection);\n for (let hNb = 0; hNb < holes.length; hNb++) {\n hole = [];\n for (let hPoint = 0; hPoint < holes[hNb].length; hPoint++) {\n hole.push(new Vector2(holes[hNb][hPoint].x, holes[hNb][hPoint].z));\n }\n polygonTriangulation.addHole(hole);\n }\n //updatability is set during applyToMesh; setting to true in triangulation build produces errors\n const polygon = polygonTriangulation.build(false, depth, smoothingThreshold);\n polygon._originalBuilderSideOrientation = options.sideOrientation;\n const vertexData = CreatePolygonVertexData(polygon, options.sideOrientation, options.faceUV, options.faceColors, options.frontUVs, options.backUVs, options.wrap);\n vertexData.applyToMesh(polygon, options.updatable);\n return polygon;\n }\n /**\n * Creates an extruded polygon mesh, with depth in the Y direction.\n * * You can set different colors and different images to the top, bottom and extruded side by using the parameters `faceColors` (an array of 3 Color3 elements) and `faceUV` (an array of 3 Vector4 elements)\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/texturePerBoxFace\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param options.shape\n * @param options.holes\n * @param options.depth\n * @param options.faceUV\n * @param options.faceColors\n * @param options.updatable\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param options.wrap\n * @param scene defines the hosting scene\n * @param earcutInjection can be used to inject your own earcut reference\n * @returns the polygon mesh\n */\n function ExtrudePolygon(name, options, scene = null, earcutInjection = earcut) {\n return CreatePolygon(name, options, scene, earcutInjection);\n }\n VertexData.CreatePolygon = CreatePolygonVertexData;\n Mesh.CreatePolygon = (name, shape, scene, holes, updatable, sideOrientation, earcutInjection = earcut) => {\n const options = {\n shape: shape,\n holes: holes,\n updatable: updatable,\n sideOrientation: sideOrientation,\n };\n return CreatePolygon(name, options, scene, earcutInjection);\n };\n Mesh.ExtrudePolygon = (name, shape, depth, scene, holes, updatable, sideOrientation, earcutInjection = earcut) => {\n const options = {\n shape: shape,\n holes: holes,\n depth: depth,\n updatable: updatable,\n sideOrientation: sideOrientation,\n };\n return ExtrudePolygon(name, options, scene, earcutInjection);\n };\n\n /**\n * Creates an extruded shape mesh. The extrusion is a parametric shape. It has no predefined shape. Its final shape will depend on the input parameters.\n * * The parameter `shape` is a required array of successive Vector3. This array depicts the shape to be extruded in its local space : the shape must be designed in the xOy plane and will be extruded along the Z axis.\n * * The parameter `path` is a required array of successive Vector3. This is the axis curve the shape is extruded along.\n * * The parameter `rotation` (float, default 0 radians) is the angle value to rotate the shape each step (each path point), from the former step (so rotation added each step) along the curve.\n * * The parameter `scale` (float, default 1) is the value to scale the shape.\n * * The parameter `closeShape` (boolean, default false) closes the shape when true, since v5.0.0.\n * * The parameter `closePath` (boolean, default false) closes the path when true and no caps, since v5.0.0.\n * * The parameter `cap` sets the way the extruded shape is capped. Possible values : BABYLON.Mesh.NO_CAP (default), BABYLON.Mesh.CAP_START, BABYLON.Mesh.CAP_END, BABYLON.Mesh.CAP_ALL\n * * The optional parameter `instance` is an instance of an existing ExtrudedShape object to be updated with the passed `shape`, `path`, `scale` or `rotation` parameters : https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#extruded-shape\n * * Remember you can only change the shape or path point positions, not their number when updating an extruded shape.\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\n * * The optional parameter `invertUV` (boolean, default false) swaps in the geometry the U and V coordinates to apply a texture.\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\n * * The optional parameter `firstNormal` (Vector3) defines the direction of the first normal of the supplied path. Consider using this for any path that is straight, and particular for paths in the xy plane.\n * * The optional `adjustFrame` (boolean, default false) will cause the internally generated Path3D tangents, normals, and binormals to be adjusted so that a) they are always well-defined, and b) they do not reverse from one path point to the next. This prevents the extruded shape from being flipped and/or rotated with resulting mesh self-intersections. This is primarily useful for straight paths that can reverse direction.\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param options.shape\n * @param options.path\n * @param options.scale\n * @param options.rotation\n * @param options.closeShape\n * @param options.closePath\n * @param options.cap\n * @param options.updatable\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param options.instance\n * @param options.invertUV\n * @param options.firstNormal\n * @param options.adjustFrame\n * @param scene defines the hosting scene\n * @returns the extruded shape mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#extruded-shapes\n */\n function ExtrudeShape(name, options, scene = null) {\n const path = options.path;\n const shape = options.shape;\n const scale = options.scale || 1;\n const rotation = options.rotation || 0;\n const cap = options.cap === 0 ? 0 : options.cap || Mesh.NO_CAP;\n const updatable = options.updatable;\n const sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n const instance = options.instance || null;\n const invertUV = options.invertUV || false;\n const closeShape = options.closeShape || false;\n const closePath = options.closePath || false;\n return _ExtrudeShapeGeneric(name, shape, path, scale, rotation, null, null, closePath, closeShape, cap, false, scene, updatable ? true : false, sideOrientation, instance, invertUV, options.frontUVs || null, options.backUVs || null, options.firstNormal || null, options.adjustFrame ? true : false);\n }\n /**\n * Creates an custom extruded shape mesh.\n * The custom extrusion is a parametric shape. It has no predefined shape. Its final shape will depend on the input parameters.\n * * The parameter `shape` is a required array of successive Vector3. This array depicts the shape to be extruded in its local space : the shape must be designed in the xOy plane and will be extruded along the Z axis.\n * * The parameter `path` is a required array of successive Vector3. This is the axis curve the shape is extruded along.\n * * The parameter `rotationFunction` (JS function) is a custom Javascript function called on each path point. This function is passed the position i of the point in the path and the distance of this point from the beginning of the path\n * * It must returns a float value that will be the rotation in radians applied to the shape on each path point.\n * * The parameter `scaleFunction` (JS function) is a custom Javascript function called on each path point. This function is passed the position i of the point in the path and the distance of this point from the beginning of the path\n * * It must returns a float value that will be the scale value applied to the shape on each path point\n * * The parameter `closeShape` (boolean, default false) closes the shape when true, since v5.0.0.\n * * The parameter `closePath` (boolean, default false) closes the path when true and no caps, since v5.0.0.\n * * The parameter `ribbonClosePath` (boolean, default false) forces the extrusion underlying ribbon to close all the paths in its `pathArray` - depreciated in favor of closeShape\n * * The parameter `ribbonCloseArray` (boolean, default false) forces the extrusion underlying ribbon to close its `pathArray` - depreciated in favor of closePath\n * * The parameter `cap` sets the way the extruded shape is capped. Possible values : BABYLON.Mesh.NO_CAP (default), BABYLON.Mesh.CAP_START, BABYLON.Mesh.CAP_END, BABYLON.Mesh.CAP_ALL\n * * The optional parameter `instance` is an instance of an existing ExtrudedShape object to be updated with the passed `shape`, `path`, `scale` or `rotation` parameters : https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#extruded-shape\n * * Remember you can only change the shape or path point positions, not their number when updating an extruded shape\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\n * * The optional parameter `invertUV` (boolean, default false) swaps in the geometry the U and V coordinates to apply a texture\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\n * * The optional parameter `firstNormal` (Vector3) defines the direction of the first normal of the supplied path. It should be supplied when the path is in the xy plane, and particularly if these sections are straight, because the underlying Path3D object will pick a normal in the xy plane that causes the extrusion to be collapsed into the plane. This should be used for any path that is straight.\n * * The optional `adjustFrame` (boolean, default false) will cause the internally generated Path3D tangents, normals, and binormals to be adjusted so that a) they are always well-defined, and b) they do not reverse from one path point to the next. This prevents the extruded shape from being flipped and/or rotated with resulting mesh self-intersections. This is primarily useful for straight paths that can reverse direction.\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param options.shape\n * @param options.path\n * @param options.scaleFunction\n * @param options.rotationFunction\n * @param options.ribbonCloseArray\n * @param options.ribbonClosePath\n * @param options.closeShape\n * @param options.closePath\n * @param options.cap\n * @param options.updatable\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param options.instance\n * @param options.invertUV\n * @param options.firstNormal\n * @param options.adjustFrame\n * @param scene defines the hosting scene\n * @returns the custom extruded shape mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#custom-extruded-shapes\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#extruded-shapes\n */\n function ExtrudeShapeCustom(name, options, scene = null) {\n const path = options.path;\n const shape = options.shape;\n const scaleFunction = options.scaleFunction ||\n (() => {\n return 1;\n });\n const rotationFunction = options.rotationFunction ||\n (() => {\n return 0;\n });\n const ribbonCloseArray = options.closePath || options.ribbonCloseArray || false;\n const ribbonClosePath = options.closeShape || options.ribbonClosePath || false;\n const cap = options.cap === 0 ? 0 : options.cap || Mesh.NO_CAP;\n const updatable = options.updatable;\n const firstNormal = options.firstNormal || null;\n const adjustFrame = options.adjustFrame || false;\n const sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n const instance = options.instance;\n const invertUV = options.invertUV || false;\n return _ExtrudeShapeGeneric(name, shape, path, null, null, scaleFunction, rotationFunction, ribbonCloseArray, ribbonClosePath, cap, true, scene, updatable ? true : false, sideOrientation, instance || null, invertUV, options.frontUVs || null, options.backUVs || null, firstNormal, adjustFrame);\n }\n function _ExtrudeShapeGeneric(name, shape, curve, scale, rotation, scaleFunction, rotateFunction, rbCA, rbCP, cap, custom, scene, updtbl, side, instance, invertUV, frontUVs, backUVs, firstNormal, adjustFrame) {\n // extrusion geometry\n const extrusionPathArray = (shape, curve, path3D, shapePaths, scale, rotation, scaleFunction, rotateFunction, cap, custom, adjustFrame) => {\n const tangents = path3D.getTangents();\n const normals = path3D.getNormals();\n const binormals = path3D.getBinormals();\n const distances = path3D.getDistances();\n if (adjustFrame) {\n /* fix tangents,normals, binormals */\n for (let i = 0; i < tangents.length; i++) {\n if (tangents[i].x == 0 && tangents[i].y == 0 && tangents[i].z == 0) {\n tangents[i].copyFrom(tangents[i - 1]);\n }\n if (normals[i].x == 0 && normals[i].y == 0 && normals[i].z == 0) {\n normals[i].copyFrom(normals[i - 1]);\n }\n if (binormals[i].x == 0 && binormals[i].y == 0 && binormals[i].z == 0) {\n binormals[i].copyFrom(binormals[i - 1]);\n }\n if (i > 0) {\n let v = tangents[i - 1];\n if (Vector3.Dot(v, tangents[i]) < 0) {\n tangents[i].scaleInPlace(-1);\n }\n v = normals[i - 1];\n if (Vector3.Dot(v, normals[i]) < 0) {\n normals[i].scaleInPlace(-1);\n }\n v = binormals[i - 1];\n if (Vector3.Dot(v, binormals[i]) < 0) {\n binormals[i].scaleInPlace(-1);\n }\n }\n }\n }\n let angle = 0;\n const returnScale = () => {\n return scale !== null ? scale : 1;\n };\n const returnRotation = () => {\n return rotation !== null ? rotation : 0;\n };\n const rotate = custom && rotateFunction ? rotateFunction : returnRotation;\n const scl = custom && scaleFunction ? scaleFunction : returnScale;\n let index = cap === Mesh.NO_CAP || cap === Mesh.CAP_END ? 0 : 2;\n const rotationMatrix = TmpVectors.Matrix[0];\n for (let i = 0; i < curve.length; i++) {\n const shapePath = new Array();\n const angleStep = rotate(i, distances[i]);\n const scaleRatio = scl(i, distances[i]);\n Matrix.RotationAxisToRef(tangents[i], angle, rotationMatrix);\n for (let p = 0; p < shape.length; p++) {\n const planed = tangents[i].scale(shape[p].z).add(normals[i].scale(shape[p].x)).add(binormals[i].scale(shape[p].y));\n const rotated = Vector3.Zero();\n Vector3.TransformCoordinatesToRef(planed, rotationMatrix, rotated);\n rotated.scaleInPlace(scaleRatio).addInPlace(curve[i]);\n shapePath[p] = rotated;\n }\n shapePaths[index] = shapePath;\n angle += angleStep;\n index++;\n }\n // cap\n const capPath = (shapePath) => {\n const pointCap = Array();\n const barycenter = Vector3.Zero();\n let i;\n for (i = 0; i < shapePath.length; i++) {\n barycenter.addInPlace(shapePath[i]);\n }\n barycenter.scaleInPlace(1.0 / shapePath.length);\n for (i = 0; i < shapePath.length; i++) {\n pointCap.push(barycenter);\n }\n return pointCap;\n };\n switch (cap) {\n case Mesh.NO_CAP:\n break;\n case Mesh.CAP_START:\n shapePaths[0] = capPath(shapePaths[2]);\n shapePaths[1] = shapePaths[2];\n break;\n case Mesh.CAP_END:\n shapePaths[index] = shapePaths[index - 1];\n shapePaths[index + 1] = capPath(shapePaths[index - 1]);\n break;\n case Mesh.CAP_ALL:\n shapePaths[0] = capPath(shapePaths[2]);\n shapePaths[1] = shapePaths[2];\n shapePaths[index] = shapePaths[index - 1];\n shapePaths[index + 1] = capPath(shapePaths[index - 1]);\n break;\n }\n return shapePaths;\n };\n let path3D;\n let pathArray;\n if (instance) {\n // instance update\n const storage = instance._creationDataStorage;\n path3D = firstNormal ? storage.path3D.update(curve, firstNormal) : storage.path3D.update(curve);\n pathArray = extrusionPathArray(shape, curve, storage.path3D, storage.pathArray, scale, rotation, scaleFunction, rotateFunction, storage.cap, custom, adjustFrame);\n instance = CreateRibbon(\"\", { pathArray, closeArray: false, closePath: false, offset: 0, updatable: false, sideOrientation: 0, instance }, scene || undefined);\n return instance;\n }\n // extruded shape creation\n path3D = firstNormal ? new Path3D(curve, firstNormal) : new Path3D(curve);\n const newShapePaths = new Array();\n cap = cap < 0 || cap > 3 ? 0 : cap;\n pathArray = extrusionPathArray(shape, curve, path3D, newShapePaths, scale, rotation, scaleFunction, rotateFunction, cap, custom, adjustFrame);\n const extrudedGeneric = CreateRibbon(name, {\n pathArray: pathArray,\n closeArray: rbCA,\n closePath: rbCP,\n updatable: updtbl,\n sideOrientation: side,\n invertUV: invertUV,\n frontUVs: frontUVs || undefined,\n backUVs: backUVs || undefined,\n }, scene);\n extrudedGeneric._creationDataStorage.pathArray = pathArray;\n extrudedGeneric._creationDataStorage.path3D = path3D;\n extrudedGeneric._creationDataStorage.cap = cap;\n return extrudedGeneric;\n }\n Mesh.ExtrudeShape = (name, shape, path, scale, rotation, cap, scene = null, updatable, sideOrientation, instance) => {\n const options = {\n shape: shape,\n path: path,\n scale: scale,\n rotation: rotation,\n cap: cap === 0 ? 0 : cap || Mesh.NO_CAP,\n sideOrientation: sideOrientation,\n instance: instance,\n updatable: updatable,\n };\n return ExtrudeShape(name, options, scene);\n };\n Mesh.ExtrudeShapeCustom = (name, shape, path, scaleFunction, rotationFunction, ribbonCloseArray, ribbonClosePath, cap, scene, updatable, sideOrientation, instance) => {\n const options = {\n shape: shape,\n path: path,\n scaleFunction: scaleFunction,\n rotationFunction: rotationFunction,\n ribbonCloseArray: ribbonCloseArray,\n ribbonClosePath: ribbonClosePath,\n cap: cap === 0 ? 0 : cap || Mesh.NO_CAP,\n sideOrientation: sideOrientation,\n instance: instance,\n updatable: updatable,\n };\n return ExtrudeShapeCustom(name, options, scene);\n };\n\n /**\n * Creates lathe mesh.\n * The lathe is a shape with a symmetry axis : a 2D model shape is rotated around this axis to design the lathe\n * * The parameter `shape` is a required array of successive Vector3. This array depicts the shape to be rotated in its local space : the shape must be designed in the xOy plane and will be rotated around the Y axis. It's usually a 2D shape, so the Vector3 z coordinates are often set to zero\n * * The parameter `radius` (positive float, default 1) is the radius value of the lathe\n * * The parameter `tessellation` (positive integer, default 64) is the side number of the lathe\n * * The parameter `clip` (positive integer, default 0) is the number of sides to not create without effecting the general shape of the sides\n * * The parameter `arc` (positive float, default 1) is the ratio of the lathe. 0.5 builds for instance half a lathe, so an opened shape\n * * The parameter `closed` (boolean, default true) opens/closes the lathe circumference. This should be set to false when used with the parameter \"arc\"\n * * The parameter `cap` sets the way the extruded shape is capped. Possible values : BABYLON.Mesh.NO_CAP (default), BABYLON.Mesh.CAP_START, BABYLON.Mesh.CAP_END, BABYLON.Mesh.CAP_ALL\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\n * * The optional parameter `invertUV` (boolean, default false) swaps in the geometry the U and V coordinates to apply a texture\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param options.shape\n * @param options.radius\n * @param options.tessellation\n * @param options.clip\n * @param options.arc\n * @param options.closed\n * @param options.updatable\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param options.cap\n * @param options.invertUV\n * @param scene defines the hosting scene\n * @returns the lathe mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#lathe\n */\n function CreateLathe(name, options, scene = null) {\n const arc = options.arc ? (options.arc <= 0 || options.arc > 1 ? 1.0 : options.arc) : 1.0;\n const closed = options.closed === undefined ? true : options.closed;\n const shape = options.shape;\n const radius = options.radius || 1;\n const tessellation = options.tessellation || 64;\n const clip = options.clip || 0;\n const updatable = options.updatable;\n const sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n const cap = options.cap || Mesh.NO_CAP;\n const pi2 = Math.PI * 2;\n const paths = new Array();\n const invertUV = options.invertUV || false;\n let i = 0;\n let p = 0;\n const step = (pi2 / tessellation) * arc;\n let rotated;\n let path;\n for (i = 0; i <= tessellation - clip; i++) {\n path = [];\n if (cap == Mesh.CAP_START || cap == Mesh.CAP_ALL) {\n path.push(new Vector3(0, shape[0].y, 0));\n path.push(new Vector3(Math.cos(i * step) * shape[0].x * radius, shape[0].y, Math.sin(i * step) * shape[0].x * radius));\n }\n for (p = 0; p < shape.length; p++) {\n rotated = new Vector3(Math.cos(i * step) * shape[p].x * radius, shape[p].y, Math.sin(i * step) * shape[p].x * radius);\n path.push(rotated);\n }\n if (cap == Mesh.CAP_END || cap == Mesh.CAP_ALL) {\n path.push(new Vector3(Math.cos(i * step) * shape[shape.length - 1].x * radius, shape[shape.length - 1].y, Math.sin(i * step) * shape[shape.length - 1].x * radius));\n path.push(new Vector3(0, shape[shape.length - 1].y, 0));\n }\n paths.push(path);\n }\n // lathe ribbon\n const lathe = CreateRibbon(name, { pathArray: paths, closeArray: closed, sideOrientation: sideOrientation, updatable: updatable, invertUV: invertUV, frontUVs: options.frontUVs, backUVs: options.backUVs }, scene);\n return lathe;\n }\n Mesh.CreateLathe = (name, shape, radius, tessellation, scene, updatable, sideOrientation) => {\n const options = {\n shape: shape,\n radius: radius,\n tessellation: tessellation,\n sideOrientation: sideOrientation,\n updatable: updatable,\n };\n return CreateLathe(name, options, scene);\n };\n\n /**\n * Creates a tube mesh.\n * The tube is a parametric shape. It has no predefined shape. Its final shape will depend on the input parameters\n * * The parameter `path` is a required array of successive Vector3. It is the curve used as the axis of the tube\n * * The parameter `radius` (positive float, default 1) sets the tube radius size\n * * The parameter `tessellation` (positive float, default 64) is the number of sides on the tubular surface\n * * The parameter `radiusFunction` (javascript function, default null) is a vanilla javascript function. If it is not null, it overrides the parameter `radius`\n * * This function is called on each point of the tube path and is passed the index `i` of the i-th point and the distance of this point from the first point of the path. It must return a radius value (positive float)\n * * The parameter `arc` (positive float, maximum 1, default 1) is the ratio to apply to the tube circumference : 2 x PI x arc\n * * The parameter `cap` sets the way the extruded shape is capped. Possible values : BABYLON.Mesh.NO_CAP (default), BABYLON.Mesh.CAP_START, BABYLON.Mesh.CAP_END, BABYLON.Mesh.CAP_ALL\n * * The optional parameter `instance` is an instance of an existing Tube object to be updated with the passed `pathArray` parameter. The `path`Array HAS to have the SAME number of points as the previous one: https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#tube\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\n * * The optional parameter `invertUV` (boolean, default false) swaps in the geometry the U and V coordinates to apply a texture\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created. The NUMBER of points CAN'T CHANGE, only their positions.\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param options.path\n * @param options.radius\n * @param options.tessellation\n * @param options.radiusFunction\n * @param options.cap\n * @param options.arc\n * @param options.updatable\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param options.instance\n * @param options.invertUV\n * @param scene defines the hosting scene\n * @returns the tube mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#tube\n */\n function CreateTube(name, options, scene = null) {\n const path = options.path;\n let instance = options.instance;\n let radius = 1.0;\n if (options.radius !== undefined) {\n radius = options.radius;\n }\n else if (instance) {\n radius = instance._creationDataStorage.radius;\n }\n const tessellation = options.tessellation || 64 | 0;\n const radiusFunction = options.radiusFunction || null;\n let cap = options.cap || Mesh.NO_CAP;\n const invertUV = options.invertUV || false;\n const updatable = options.updatable;\n const sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n options.arc = options.arc && (options.arc <= 0.0 || options.arc > 1.0) ? 1.0 : options.arc || 1.0;\n // tube geometry\n const tubePathArray = (path, path3D, circlePaths, radius, tessellation, radiusFunction, cap, arc) => {\n const tangents = path3D.getTangents();\n const normals = path3D.getNormals();\n const distances = path3D.getDistances();\n const pi2 = Math.PI * 2;\n const step = (pi2 / tessellation) * arc;\n const returnRadius = () => radius;\n const radiusFunctionFinal = radiusFunction || returnRadius;\n let circlePath;\n let rad;\n let normal;\n let rotated;\n const rotationMatrix = TmpVectors.Matrix[0];\n let index = cap === Mesh.NO_CAP || cap === Mesh.CAP_END ? 0 : 2;\n for (let i = 0; i < path.length; i++) {\n rad = radiusFunctionFinal(i, distances[i]); // current radius\n circlePath = Array(); // current circle array\n normal = normals[i]; // current normal\n for (let t = 0; t < tessellation; t++) {\n Matrix.RotationAxisToRef(tangents[i], step * t, rotationMatrix);\n rotated = circlePath[t] ? circlePath[t] : Vector3.Zero();\n Vector3.TransformCoordinatesToRef(normal, rotationMatrix, rotated);\n rotated.scaleInPlace(rad).addInPlace(path[i]);\n circlePath[t] = rotated;\n }\n circlePaths[index] = circlePath;\n index++;\n }\n // cap\n const capPath = (nbPoints, pathIndex) => {\n const pointCap = Array();\n for (let i = 0; i < nbPoints; i++) {\n pointCap.push(path[pathIndex]);\n }\n return pointCap;\n };\n switch (cap) {\n case Mesh.NO_CAP:\n break;\n case Mesh.CAP_START:\n circlePaths[0] = capPath(tessellation, 0);\n circlePaths[1] = circlePaths[2].slice(0);\n break;\n case Mesh.CAP_END:\n circlePaths[index] = circlePaths[index - 1].slice(0);\n circlePaths[index + 1] = capPath(tessellation, path.length - 1);\n break;\n case Mesh.CAP_ALL:\n circlePaths[0] = capPath(tessellation, 0);\n circlePaths[1] = circlePaths[2].slice(0);\n circlePaths[index] = circlePaths[index - 1].slice(0);\n circlePaths[index + 1] = capPath(tessellation, path.length - 1);\n break;\n }\n return circlePaths;\n };\n let path3D;\n let pathArray;\n if (instance) {\n // tube update\n const storage = instance._creationDataStorage;\n const arc = options.arc || storage.arc;\n path3D = storage.path3D.update(path);\n pathArray = tubePathArray(path, path3D, storage.pathArray, radius, storage.tessellation, radiusFunction, storage.cap, arc);\n instance = CreateRibbon(\"\", { pathArray: pathArray, instance: instance });\n // Update mode, no need to recreate the storage.\n storage.path3D = path3D;\n storage.pathArray = pathArray;\n storage.arc = arc;\n storage.radius = radius;\n return instance;\n }\n // tube creation\n path3D = new Path3D(path);\n const newPathArray = new Array();\n cap = cap < 0 || cap > 3 ? 0 : cap;\n pathArray = tubePathArray(path, path3D, newPathArray, radius, tessellation, radiusFunction, cap, options.arc);\n const tube = CreateRibbon(name, {\n pathArray: pathArray,\n closePath: true,\n closeArray: false,\n updatable: updatable,\n sideOrientation: sideOrientation,\n invertUV: invertUV,\n frontUVs: options.frontUVs,\n backUVs: options.backUVs,\n }, scene);\n tube._creationDataStorage.pathArray = pathArray;\n tube._creationDataStorage.path3D = path3D;\n tube._creationDataStorage.tessellation = tessellation;\n tube._creationDataStorage.cap = cap;\n tube._creationDataStorage.arc = options.arc;\n tube._creationDataStorage.radius = radius;\n return tube;\n }\n Mesh.CreateTube = (name, path, radius, tessellation, radiusFunction, cap, scene, updatable, sideOrientation, instance) => {\n const options = {\n path: path,\n radius: radius,\n tessellation: tessellation,\n radiusFunction: radiusFunction,\n arc: 1,\n cap: cap,\n updatable: updatable,\n sideOrientation: sideOrientation,\n instance: instance,\n };\n return CreateTube(name, options, scene);\n };\n\n // inspired from // http://stemkoski.github.io/Three.js/Polyhedra.html\n /**\n * Creates the VertexData for a Polyhedron\n * @param options an object used to set the following optional parameters for the polyhedron, required but can be empty\n * * type provided types are:\n * * 0 : Tetrahedron, 1 : Octahedron, 2 : Dodecahedron, 3 : Icosahedron, 4 : Rhombicuboctahedron, 5 : Triangular Prism, 6 : Pentagonal Prism, 7 : Hexagonal Prism, 8 : Square Pyramid (J1)\n * * 9 : Pentagonal Pyramid (J2), 10 : Triangular Dipyramid (J12), 11 : Pentagonal Dipyramid (J13), 12 : Elongated Square Dipyramid (J15), 13 : Elongated Pentagonal Dipyramid (J16), 14 : Elongated Pentagonal Cupola (J20)\n * * size the size of the IcoSphere, optional default 1\n * * sizeX allows stretching in the x direction, optional, default size\n * * sizeY allows stretching in the y direction, optional, default size\n * * sizeZ allows stretching in the z direction, optional, default size\n * * custom a number that overwrites the type to create from an extended set of polyhedron from https://www.babylonjs-playground.com/#21QRSK#15 with minimised editor\n * * faceUV an array of Vector4 elements used to set different images to the top, rings and bottom respectively\n * * faceColors an array of Color3 elements used to set different colors to the top, rings and bottom respectively\n * * flat when true creates a flat shaded mesh, optional, default true\n * * subdivisions increasing the subdivisions increases the number of faces, optional, default 4\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.type\n * @param options.size\n * @param options.sizeX\n * @param options.sizeY\n * @param options.sizeZ\n * @param options.custom\n * @param options.faceUV\n * @param options.faceColors\n * @param options.flat\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @returns the VertexData of the Polyhedron\n */\n function CreatePolyhedronVertexData(options) {\n // provided polyhedron types :\n // 0 : Tetrahedron, 1 : Octahedron, 2 : Dodecahedron, 3 : Icosahedron, 4 : Rhombicuboctahedron, 5 : Triangular Prism, 6 : Pentagonal Prism, 7 : Hexagonal Prism, 8 : Square Pyramid (J1)\n // 9 : Pentagonal Pyramid (J2), 10 : Triangular Dipyramid (J12), 11 : Pentagonal Dipyramid (J13), 12 : Elongated Square Dipyramid (J15), 13 : Elongated Pentagonal Dipyramid (J16), 14 : Elongated Pentagonal Cupola (J20)\n const polyhedra = [];\n polyhedra[0] = {\n vertex: [\n [0, 0, 1.732051],\n [1.632993, 0, -0.5773503],\n [-0.8164966, 1.414214, -0.5773503],\n [-0.8164966, -1.414214, -0.5773503],\n ],\n face: [\n [0, 1, 2],\n [0, 2, 3],\n [0, 3, 1],\n [1, 3, 2],\n ],\n };\n polyhedra[1] = {\n vertex: [\n [0, 0, 1.414214],\n [1.414214, 0, 0],\n [0, 1.414214, 0],\n [-1.414214, 0, 0],\n [0, -1.414214, 0],\n [0, 0, -1.414214],\n ],\n face: [\n [0, 1, 2],\n [0, 2, 3],\n [0, 3, 4],\n [0, 4, 1],\n [1, 4, 5],\n [1, 5, 2],\n [2, 5, 3],\n [3, 5, 4],\n ],\n };\n polyhedra[2] = {\n vertex: [\n [0, 0, 1.070466],\n [0.7136442, 0, 0.7978784],\n [-0.3568221, 0.618034, 0.7978784],\n [-0.3568221, -0.618034, 0.7978784],\n [0.7978784, 0.618034, 0.3568221],\n [0.7978784, -0.618034, 0.3568221],\n [-0.9341724, 0.381966, 0.3568221],\n [0.1362939, 1, 0.3568221],\n [0.1362939, -1, 0.3568221],\n [-0.9341724, -0.381966, 0.3568221],\n [0.9341724, 0.381966, -0.3568221],\n [0.9341724, -0.381966, -0.3568221],\n [-0.7978784, 0.618034, -0.3568221],\n [-0.1362939, 1, -0.3568221],\n [-0.1362939, -1, -0.3568221],\n [-0.7978784, -0.618034, -0.3568221],\n [0.3568221, 0.618034, -0.7978784],\n [0.3568221, -0.618034, -0.7978784],\n [-0.7136442, 0, -0.7978784],\n [0, 0, -1.070466],\n ],\n face: [\n [0, 1, 4, 7, 2],\n [0, 2, 6, 9, 3],\n [0, 3, 8, 5, 1],\n [1, 5, 11, 10, 4],\n [2, 7, 13, 12, 6],\n [3, 9, 15, 14, 8],\n [4, 10, 16, 13, 7],\n [5, 8, 14, 17, 11],\n [6, 12, 18, 15, 9],\n [10, 11, 17, 19, 16],\n [12, 13, 16, 19, 18],\n [14, 15, 18, 19, 17],\n ],\n };\n polyhedra[3] = {\n vertex: [\n [0, 0, 1.175571],\n [1.051462, 0, 0.5257311],\n [0.3249197, 1, 0.5257311],\n [-0.8506508, 0.618034, 0.5257311],\n [-0.8506508, -0.618034, 0.5257311],\n [0.3249197, -1, 0.5257311],\n [0.8506508, 0.618034, -0.5257311],\n [0.8506508, -0.618034, -0.5257311],\n [-0.3249197, 1, -0.5257311],\n [-1.051462, 0, -0.5257311],\n [-0.3249197, -1, -0.5257311],\n [0, 0, -1.175571],\n ],\n face: [\n [0, 1, 2],\n [0, 2, 3],\n [0, 3, 4],\n [0, 4, 5],\n [0, 5, 1],\n [1, 5, 7],\n [1, 7, 6],\n [1, 6, 2],\n [2, 6, 8],\n [2, 8, 3],\n [3, 8, 9],\n [3, 9, 4],\n [4, 9, 10],\n [4, 10, 5],\n [5, 10, 7],\n [6, 7, 11],\n [6, 11, 8],\n [7, 10, 11],\n [8, 11, 9],\n [9, 11, 10],\n ],\n };\n polyhedra[4] = {\n vertex: [\n [0, 0, 1.070722],\n [0.7148135, 0, 0.7971752],\n [-0.104682, 0.7071068, 0.7971752],\n [-0.6841528, 0.2071068, 0.7971752],\n [-0.104682, -0.7071068, 0.7971752],\n [0.6101315, 0.7071068, 0.5236279],\n [1.04156, 0.2071068, 0.1367736],\n [0.6101315, -0.7071068, 0.5236279],\n [-0.3574067, 1, 0.1367736],\n [-0.7888348, -0.5, 0.5236279],\n [-0.9368776, 0.5, 0.1367736],\n [-0.3574067, -1, 0.1367736],\n [0.3574067, 1, -0.1367736],\n [0.9368776, -0.5, -0.1367736],\n [0.7888348, 0.5, -0.5236279],\n [0.3574067, -1, -0.1367736],\n [-0.6101315, 0.7071068, -0.5236279],\n [-1.04156, -0.2071068, -0.1367736],\n [-0.6101315, -0.7071068, -0.5236279],\n [0.104682, 0.7071068, -0.7971752],\n [0.6841528, -0.2071068, -0.7971752],\n [0.104682, -0.7071068, -0.7971752],\n [-0.7148135, 0, -0.7971752],\n [0, 0, -1.070722],\n ],\n face: [\n [0, 2, 3],\n [1, 6, 5],\n [4, 9, 11],\n [7, 15, 13],\n [8, 16, 10],\n [12, 14, 19],\n [17, 22, 18],\n [20, 21, 23],\n [0, 1, 5, 2],\n [0, 3, 9, 4],\n [0, 4, 7, 1],\n [1, 7, 13, 6],\n [2, 5, 12, 8],\n [2, 8, 10, 3],\n [3, 10, 17, 9],\n [4, 11, 15, 7],\n [5, 6, 14, 12],\n [6, 13, 20, 14],\n [8, 12, 19, 16],\n [9, 17, 18, 11],\n [10, 16, 22, 17],\n [11, 18, 21, 15],\n [13, 15, 21, 20],\n [14, 20, 23, 19],\n [16, 19, 23, 22],\n [18, 22, 23, 21],\n ],\n };\n polyhedra[5] = {\n vertex: [\n [0, 0, 1.322876],\n [1.309307, 0, 0.1889822],\n [-0.9819805, 0.8660254, 0.1889822],\n [0.1636634, -1.299038, 0.1889822],\n [0.3273268, 0.8660254, -0.9449112],\n [-0.8183171, -0.4330127, -0.9449112],\n ],\n face: [\n [0, 3, 1],\n [2, 4, 5],\n [0, 1, 4, 2],\n [0, 2, 5, 3],\n [1, 3, 5, 4],\n ],\n };\n polyhedra[6] = {\n vertex: [\n [0, 0, 1.159953],\n [1.013464, 0, 0.5642542],\n [-0.3501431, 0.9510565, 0.5642542],\n [-0.7715208, -0.6571639, 0.5642542],\n [0.6633206, 0.9510565, -0.03144481],\n [0.8682979, -0.6571639, -0.3996071],\n [-1.121664, 0.2938926, -0.03144481],\n [-0.2348831, -1.063314, -0.3996071],\n [0.5181548, 0.2938926, -0.9953061],\n [-0.5850262, -0.112257, -0.9953061],\n ],\n face: [\n [0, 1, 4, 2],\n [0, 2, 6, 3],\n [1, 5, 8, 4],\n [3, 6, 9, 7],\n [5, 7, 9, 8],\n [0, 3, 7, 5, 1],\n [2, 4, 8, 9, 6],\n ],\n };\n polyhedra[7] = {\n vertex: [\n [0, 0, 1.118034],\n [0.8944272, 0, 0.6708204],\n [-0.2236068, 0.8660254, 0.6708204],\n [-0.7826238, -0.4330127, 0.6708204],\n [0.6708204, 0.8660254, 0.2236068],\n [1.006231, -0.4330127, -0.2236068],\n [-1.006231, 0.4330127, 0.2236068],\n [-0.6708204, -0.8660254, -0.2236068],\n [0.7826238, 0.4330127, -0.6708204],\n [0.2236068, -0.8660254, -0.6708204],\n [-0.8944272, 0, -0.6708204],\n [0, 0, -1.118034],\n ],\n face: [\n [0, 1, 4, 2],\n [0, 2, 6, 3],\n [1, 5, 8, 4],\n [3, 6, 10, 7],\n [5, 9, 11, 8],\n [7, 10, 11, 9],\n [0, 3, 7, 9, 5, 1],\n [2, 4, 8, 11, 10, 6],\n ],\n };\n polyhedra[8] = {\n vertex: [\n [-0.729665, 0.670121, 0.319155],\n [-0.655235, -0.29213, -0.754096],\n [-0.093922, -0.607123, 0.537818],\n [0.702196, 0.595691, 0.485187],\n [0.776626, -0.36656, -0.588064],\n ],\n face: [\n [1, 4, 2],\n [0, 1, 2],\n [3, 0, 2],\n [4, 3, 2],\n [4, 1, 0, 3],\n ],\n };\n polyhedra[9] = {\n vertex: [\n [-0.868849, -0.100041, 0.61257],\n [-0.329458, 0.976099, 0.28078],\n [-0.26629, -0.013796, -0.477654],\n [-0.13392, -1.034115, 0.229829],\n [0.738834, 0.707117, -0.307018],\n [0.859683, -0.535264, -0.338508],\n ],\n face: [\n [3, 0, 2],\n [5, 3, 2],\n [4, 5, 2],\n [1, 4, 2],\n [0, 1, 2],\n [0, 3, 5, 4, 1],\n ],\n };\n polyhedra[10] = {\n vertex: [\n [-0.610389, 0.243975, 0.531213],\n [-0.187812, -0.48795, -0.664016],\n [-0.187812, 0.9759, -0.664016],\n [0.187812, -0.9759, 0.664016],\n [0.798201, 0.243975, 0.132803],\n ],\n face: [\n [1, 3, 0],\n [3, 4, 0],\n [3, 1, 4],\n [0, 2, 1],\n [0, 4, 2],\n [2, 4, 1],\n ],\n };\n polyhedra[11] = {\n vertex: [\n [-1.028778, 0.392027, -0.048786],\n [-0.640503, -0.646161, 0.621837],\n [-0.125162, -0.395663, -0.540059],\n [0.004683, 0.888447, -0.651988],\n [0.125161, 0.395663, 0.540059],\n [0.632925, -0.791376, 0.433102],\n [1.031672, 0.157063, -0.354165],\n ],\n face: [\n [3, 2, 0],\n [2, 1, 0],\n [2, 5, 1],\n [0, 4, 3],\n [0, 1, 4],\n [4, 1, 5],\n [2, 3, 6],\n [3, 4, 6],\n [5, 2, 6],\n [4, 5, 6],\n ],\n };\n polyhedra[12] = {\n vertex: [\n [-0.669867, 0.334933, -0.529576],\n [-0.669867, 0.334933, 0.529577],\n [-0.4043, 1.212901, 0],\n [-0.334933, -0.669867, -0.529576],\n [-0.334933, -0.669867, 0.529577],\n [0.334933, 0.669867, -0.529576],\n [0.334933, 0.669867, 0.529577],\n [0.4043, -1.212901, 0],\n [0.669867, -0.334933, -0.529576],\n [0.669867, -0.334933, 0.529577],\n ],\n face: [\n [8, 9, 7],\n [6, 5, 2],\n [3, 8, 7],\n [5, 0, 2],\n [4, 3, 7],\n [0, 1, 2],\n [9, 4, 7],\n [1, 6, 2],\n [9, 8, 5, 6],\n [8, 3, 0, 5],\n [3, 4, 1, 0],\n [4, 9, 6, 1],\n ],\n };\n polyhedra[13] = {\n vertex: [\n [-0.931836, 0.219976, -0.264632],\n [-0.636706, 0.318353, 0.692816],\n [-0.613483, -0.735083, -0.264632],\n [-0.326545, 0.979634, 0],\n [-0.318353, -0.636706, 0.692816],\n [-0.159176, 0.477529, -0.856368],\n [0.159176, -0.477529, -0.856368],\n [0.318353, 0.636706, 0.692816],\n [0.326545, -0.979634, 0],\n [0.613482, 0.735082, -0.264632],\n [0.636706, -0.318353, 0.692816],\n [0.931835, -0.219977, -0.264632],\n ],\n face: [\n [11, 10, 8],\n [7, 9, 3],\n [6, 11, 8],\n [9, 5, 3],\n [2, 6, 8],\n [5, 0, 3],\n [4, 2, 8],\n [0, 1, 3],\n [10, 4, 8],\n [1, 7, 3],\n [10, 11, 9, 7],\n [11, 6, 5, 9],\n [6, 2, 0, 5],\n [2, 4, 1, 0],\n [4, 10, 7, 1],\n ],\n };\n polyhedra[14] = {\n vertex: [\n [-0.93465, 0.300459, -0.271185],\n [-0.838689, -0.260219, -0.516017],\n [-0.711319, 0.717591, 0.128359],\n [-0.710334, -0.156922, 0.080946],\n [-0.599799, 0.556003, -0.725148],\n [-0.503838, -0.004675, -0.969981],\n [-0.487004, 0.26021, 0.48049],\n [-0.460089, -0.750282, -0.512622],\n [-0.376468, 0.973135, -0.325605],\n [-0.331735, -0.646985, 0.084342],\n [-0.254001, 0.831847, 0.530001],\n [-0.125239, -0.494738, -0.966586],\n [0.029622, 0.027949, 0.730817],\n [0.056536, -0.982543, -0.262295],\n [0.08085, 1.087391, 0.076037],\n [0.125583, -0.532729, 0.485984],\n [0.262625, 0.599586, 0.780328],\n [0.391387, -0.726999, -0.716259],\n [0.513854, -0.868287, 0.139347],\n [0.597475, 0.85513, 0.326364],\n [0.641224, 0.109523, 0.783723],\n [0.737185, -0.451155, 0.538891],\n [0.848705, -0.612742, -0.314616],\n [0.976075, 0.365067, 0.32976],\n [1.072036, -0.19561, 0.084927],\n ],\n face: [\n [15, 18, 21],\n [12, 20, 16],\n [6, 10, 2],\n [3, 0, 1],\n [9, 7, 13],\n [2, 8, 4, 0],\n [0, 4, 5, 1],\n [1, 5, 11, 7],\n [7, 11, 17, 13],\n [13, 17, 22, 18],\n [18, 22, 24, 21],\n [21, 24, 23, 20],\n [20, 23, 19, 16],\n [16, 19, 14, 10],\n [10, 14, 8, 2],\n [15, 9, 13, 18],\n [12, 15, 21, 20],\n [6, 12, 16, 10],\n [3, 6, 2, 0],\n [9, 3, 1, 7],\n [9, 15, 12, 6, 3],\n [22, 17, 11, 5, 4, 8, 14, 19, 23, 24],\n ],\n };\n const type = options.type && (options.type < 0 || options.type >= polyhedra.length) ? 0 : options.type || 0;\n const size = options.size;\n const sizeX = options.sizeX || size || 1;\n const sizeY = options.sizeY || size || 1;\n const sizeZ = options.sizeZ || size || 1;\n const data = options.custom || polyhedra[type];\n const nbfaces = data.face.length;\n const faceUV = options.faceUV || new Array(nbfaces);\n const faceColors = options.faceColors;\n const flat = options.flat === undefined ? true : options.flat;\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\n const positions = new Array();\n const indices = new Array();\n const normals = new Array();\n const uvs = new Array();\n const colors = new Array();\n let index = 0;\n let faceIdx = 0; // face cursor in the array \"indexes\"\n const indexes = new Array();\n let i = 0;\n let f = 0;\n let u, v, ang, x, y, tmp;\n // default face colors and UV if undefined\n if (flat) {\n for (f = 0; f < nbfaces; f++) {\n if (faceColors && faceColors[f] === undefined) {\n faceColors[f] = new Color4(1, 1, 1, 1);\n }\n if (faceUV && faceUV[f] === undefined) {\n faceUV[f] = new Vector4(0, 0, 1, 1);\n }\n }\n }\n if (!flat) {\n for (i = 0; i < data.vertex.length; i++) {\n positions.push(data.vertex[i][0] * sizeX, data.vertex[i][1] * sizeY, data.vertex[i][2] * sizeZ);\n uvs.push(0, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 : 0);\n }\n for (f = 0; f < nbfaces; f++) {\n for (i = 0; i < data.face[f].length - 2; i++) {\n indices.push(data.face[f][0], data.face[f][i + 2], data.face[f][i + 1]);\n }\n }\n }\n else {\n for (f = 0; f < nbfaces; f++) {\n const fl = data.face[f].length; // number of vertices of the current face\n ang = (2 * Math.PI) / fl;\n x = 0.5 * Math.tan(ang / 2);\n y = 0.5;\n // positions, uvs, colors\n for (i = 0; i < fl; i++) {\n // positions\n positions.push(data.vertex[data.face[f][i]][0] * sizeX, data.vertex[data.face[f][i]][1] * sizeY, data.vertex[data.face[f][i]][2] * sizeZ);\n indexes.push(index);\n index++;\n // uvs\n u = faceUV[f].x + (faceUV[f].z - faceUV[f].x) * (0.5 + x);\n v = faceUV[f].y + (faceUV[f].w - faceUV[f].y) * (y - 0.5);\n uvs.push(u, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - v : v);\n tmp = x * Math.cos(ang) - y * Math.sin(ang);\n y = x * Math.sin(ang) + y * Math.cos(ang);\n x = tmp;\n // colors\n if (faceColors) {\n colors.push(faceColors[f].r, faceColors[f].g, faceColors[f].b, faceColors[f].a);\n }\n }\n // indices from indexes\n for (i = 0; i < fl - 2; i++) {\n indices.push(indexes[0 + faceIdx], indexes[i + 2 + faceIdx], indexes[i + 1 + faceIdx]);\n }\n faceIdx += fl;\n }\n }\n VertexData.ComputeNormals(positions, indices, normals);\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\n const vertexData = new VertexData();\n vertexData.positions = positions;\n vertexData.indices = indices;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n if (faceColors && flat) {\n vertexData.colors = colors;\n }\n return vertexData;\n }\n /**\n * Creates a polyhedron mesh\n * * The parameter `type` (positive integer, max 14, default 0) sets the polyhedron type to build among the 15 embbeded types. Please refer to the type sheet in the tutorial to choose the wanted type\n * * The parameter `size` (positive float, default 1) sets the polygon size\n * * You can overwrite the `size` on each dimension bu using the parameters `sizeX`, `sizeY` or `sizeZ` (positive floats, default to `size` value)\n * * You can build other polyhedron types than the 15 embbeded ones by setting the parameter `custom` (`polyhedronObject`, default null). If you set the parameter `custom`, this overrides the parameter `type`\n * * A `polyhedronObject` is a formatted javascript object. You'll find a full file with pre-set polyhedra here : https://github.com/BabylonJS/Extensions/tree/master/Polyhedron\n * * You can set the color and the UV of each side of the polyhedron with the parameters `faceColors` (Color4, default `(1, 1, 1, 1)`) and faceUV (Vector4, default `(0, 0, 1, 1)`)\n * * To understand how to set `faceUV` or `faceColors`, please read this by considering the right number of faces of your polyhedron, instead of only 6 for the box : https://doc.babylonjs.com/features/featuresDeepDive/materials/using/texturePerBoxFace\n * * The parameter `flat` (boolean, default true). If set to false, it gives the polyhedron a single global face, so less vertices and shared normals. In this case, `faceColors` and `faceUV` are ignored\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param options.type\n * @param options.size\n * @param options.sizeX\n * @param options.sizeY\n * @param options.sizeZ\n * @param options.custom\n * @param options.faceUV\n * @param options.faceColors\n * @param options.flat\n * @param options.updatable\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param scene defines the hosting scene\n * @returns the polyhedron mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/polyhedra\n */\n function CreatePolyhedron(name, options = {}, scene = null) {\n const polyhedron = new Mesh(name, scene);\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n polyhedron._originalBuilderSideOrientation = options.sideOrientation;\n const vertexData = CreatePolyhedronVertexData(options);\n vertexData.applyToMesh(polyhedron, options.updatable);\n return polyhedron;\n }\n /**\n * Class containing static functions to help procedurally build meshes\n * @deprecated use the function directly from the module\n */\n const PolyhedronBuilder = {\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CreatePolyhedron,\n };\n VertexData.CreatePolyhedron = CreatePolyhedronVertexData;\n Mesh.CreatePolyhedron = (name, options, scene) => {\n return CreatePolyhedron(name, options, scene);\n };\n\n /**\n * Creates the VertexData of the IcoSphere\n * @param options an object used to set the following optional parameters for the IcoSphere, required but can be empty\n * * radius the radius of the IcoSphere, optional default 1\n * * radiusX allows stretching in the x direction, optional, default radius\n * * radiusY allows stretching in the y direction, optional, default radius\n * * radiusZ allows stretching in the z direction, optional, default radius\n * * flat when true creates a flat shaded mesh, optional, default true\n * * subdivisions increasing the subdivisions increases the number of faces, optional, default 4\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.radius\n * @param options.radiusX\n * @param options.radiusY\n * @param options.radiusZ\n * @param options.flat\n * @param options.subdivisions\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @returns the VertexData of the IcoSphere\n */\n function CreateIcoSphereVertexData(options) {\n const sideOrientation = options.sideOrientation || VertexData.DEFAULTSIDE;\n const radius = options.radius || 1;\n const flat = options.flat === undefined ? true : options.flat;\n const subdivisions = (options.subdivisions || 4) | 0;\n const radiusX = options.radiusX || radius;\n const radiusY = options.radiusY || radius;\n const radiusZ = options.radiusZ || radius;\n const t = (1 + Math.sqrt(5)) / 2;\n // 12 vertex x,y,z\n const icoVertices = [\n -1,\n t,\n -0,\n 1,\n t,\n 0,\n -1,\n -t,\n 0,\n 1,\n -t,\n 0,\n 0,\n -1,\n -t,\n 0,\n 1,\n -t,\n 0,\n -1,\n t,\n 0,\n 1,\n t,\n t,\n 0,\n 1,\n t,\n 0,\n -1,\n -t,\n 0,\n 1,\n -t,\n 0,\n -1, // v8-11\n ];\n // index of 3 vertex makes a face of icopshere\n const ico_indices = [\n 0, 11, 5, 0, 5, 1, 0, 1, 7, 0, 7, 10, 12, 22, 23, 1, 5, 20, 5, 11, 4, 23, 22, 13, 22, 18, 6, 7, 1, 8, 14, 21, 4, 14, 4, 2, 16, 13, 6, 15, 6, 19, 3, 8, 9, 4, 21, 5, 13, 17,\n 23, 6, 13, 22, 19, 6, 18, 9, 8, 1,\n ];\n // vertex for uv have aliased position, not for UV\n const vertices_unalias_id = [\n 0,\n 1,\n 2,\n 3,\n 4,\n 5,\n 6,\n 7,\n 8,\n 9,\n 10,\n 11,\n // vertex alias\n 0,\n 2,\n 3,\n 3,\n 3,\n 4,\n 7,\n 8,\n 9,\n 9,\n 10,\n 11, // 23: B + 12\n ];\n // uv as integer step (not pixels !)\n const ico_vertexuv = [\n 5,\n 1,\n 3,\n 1,\n 6,\n 4,\n 0,\n 0,\n 5,\n 3,\n 4,\n 2,\n 2,\n 2,\n 4,\n 0,\n 2,\n 0,\n 1,\n 1,\n 6,\n 0,\n 6,\n 2,\n // vertex alias (for same vertex on different faces)\n 0,\n 4,\n 3,\n 3,\n 4,\n 4,\n 3,\n 1,\n 4,\n 2,\n 4,\n 4,\n 0,\n 2,\n 1,\n 1,\n 2,\n 2,\n 3,\n 3,\n 1,\n 3,\n 2,\n 4, // 23: B + 12\n ];\n // Vertices[0, 1, ...9, A, B] : position on UV plane\n // '+' indicate duplicate position to be fixed (3,9:0,2,3,4,7,8,A,B)\n // First island of uv mapping\n // v = 4h 3+ 2\n // v = 3h 9+ 4\n // v = 2h 9+ 5 B\n // v = 1h 9 1 0\n // v = 0h 3 8 7 A\n // u = 0 1 2 3 4 5 6 *a\n // Second island of uv mapping\n // v = 4h 0+ B+ 4+\n // v = 3h A+ 2+\n // v = 2h 7+ 6 3+\n // v = 1h 8+ 3+\n // v = 0h\n // u = 0 1 2 3 4 5 6 *a\n // Face layout on texture UV mapping\n // ============\n // \\ 4 /\\ 16 / ======\n // \\ / \\ / /\\ 11 /\n // \\/ 7 \\/ / \\ /\n // ======= / 10 \\/\n // /\\ 17 /\\ =======\n // / \\ / \\ \\ 15 /\\\n // / 8 \\/ 12 \\ \\ / \\\n // ============ \\/ 6 \\\n // \\ 18 /\\ ============\n // \\ / \\ \\ 5 /\\ 0 /\n // \\/ 13 \\ \\ / \\ /\n // ======= \\/ 1 \\/\n // =============\n // /\\ 19 /\\ 2 /\\\n // / \\ / \\ / \\\n // / 14 \\/ 9 \\/ 3 \\\n // ===================\n // uv step is u:1 or 0.5, v:cos(30)=sqrt(3)/2, ratio approx is 84/97\n const ustep = 138 / 1024;\n const vstep = 239 / 1024;\n const uoffset = 60 / 1024;\n const voffset = 26 / 1024;\n // Second island should have margin, not to touch the first island\n // avoid any borderline artefact in pixel rounding\n const island_u_offset = -40 / 1024;\n const island_v_offset = +20 / 1024;\n // face is either island 0 or 1 :\n // second island is for faces : [4, 7, 8, 12, 13, 16, 17, 18]\n const island = [\n 0,\n 0,\n 0,\n 0,\n 1,\n 0,\n 0,\n 1,\n 1,\n 0,\n 0,\n 0,\n 1,\n 1,\n 0,\n 0,\n 1,\n 1,\n 1,\n 0, // 15 - 19\n ];\n const indices = new Array();\n const positions = new Array();\n const normals = new Array();\n const uvs = new Array();\n let current_indice = 0;\n // prepare array of 3 vector (empty) (to be worked in place, shared for each face)\n const face_vertex_pos = new Array(3);\n const face_vertex_uv = new Array(3);\n let v012;\n for (v012 = 0; v012 < 3; v012++) {\n face_vertex_pos[v012] = Vector3.Zero();\n face_vertex_uv[v012] = Vector2.Zero();\n }\n // create all with normals\n for (let face = 0; face < 20; face++) {\n // 3 vertex per face\n for (v012 = 0; v012 < 3; v012++) {\n // look up vertex 0,1,2 to its index in 0 to 11 (or 23 including alias)\n const v_id = ico_indices[3 * face + v012];\n // vertex have 3D position (x,y,z)\n face_vertex_pos[v012].copyFromFloats(icoVertices[3 * vertices_unalias_id[v_id]], icoVertices[3 * vertices_unalias_id[v_id] + 1], icoVertices[3 * vertices_unalias_id[v_id] + 2]);\n // Normalize to get normal\n face_vertex_pos[v012].normalize();\n // uv Coordinates from vertex ID\n face_vertex_uv[v012].copyFromFloats(ico_vertexuv[2 * v_id] * ustep + uoffset + island[face] * island_u_offset, ico_vertexuv[2 * v_id + 1] * vstep + voffset + island[face] * island_v_offset);\n }\n // Subdivide the face (interpolate pos, norm, uv)\n // - pos is linear interpolation, then projected to sphere (converge polyhedron to sphere)\n // - norm is linear interpolation of vertex corner normal\n // (to be checked if better to re-calc from face vertex, or if approximation is OK ??? )\n // - uv is linear interpolation\n //\n // Topology is as below for sub-divide by 2\n // vertex shown as v0,v1,v2\n // interp index is i1 to progress in range [v0,v1[\n // interp index is i2 to progress in range [v0,v2[\n // face index as (i1,i2) for /\\ : (i1,i2),(i1+1,i2),(i1,i2+1)\n // and (i1,i2)' for \\/ : (i1+1,i2),(i1+1,i2+1),(i1,i2+1)\n //\n //\n // i2 v2\n // ^ ^\n // / / \\\n // / / \\\n // / / \\\n // / / (0,1) \\\n // / #---------\\\n // / / \\ (0,0)'/ \\\n // / / \\ / \\\n // / / \\ / \\\n // / / (0,0) \\ / (1,0) \\\n // / #---------#---------\\\n // v0 v1\n //\n // --------------------> i1\n //\n // interp of (i1,i2):\n // along i2 : x0=lerp(v0,v2, i2/S) <---> x1=lerp(v1,v2, i2/S)\n // along i1 : lerp(x0,x1, i1/(S-i2))\n //\n // centroid of triangle is needed to get help normal computation\n // (c1,c2) are used for centroid location\n const interp_vertex = (i1, i2, c1, c2) => {\n // vertex is interpolated from\n // - face_vertex_pos[0..2]\n // - face_vertex_uv[0..2]\n const pos_x0 = Vector3.Lerp(face_vertex_pos[0], face_vertex_pos[2], i2 / subdivisions);\n const pos_x1 = Vector3.Lerp(face_vertex_pos[1], face_vertex_pos[2], i2 / subdivisions);\n const pos_interp = subdivisions === i2 ? face_vertex_pos[2] : Vector3.Lerp(pos_x0, pos_x1, i1 / (subdivisions - i2));\n pos_interp.normalize();\n let vertex_normal;\n if (flat) {\n // in flat mode, recalculate normal as face centroid normal\n const centroid_x0 = Vector3.Lerp(face_vertex_pos[0], face_vertex_pos[2], c2 / subdivisions);\n const centroid_x1 = Vector3.Lerp(face_vertex_pos[1], face_vertex_pos[2], c2 / subdivisions);\n vertex_normal = Vector3.Lerp(centroid_x0, centroid_x1, c1 / (subdivisions - c2));\n }\n else {\n // in smooth mode, recalculate normal from each single vertex position\n vertex_normal = new Vector3(pos_interp.x, pos_interp.y, pos_interp.z);\n }\n // Vertex normal need correction due to X,Y,Z radius scaling\n vertex_normal.x /= radiusX;\n vertex_normal.y /= radiusY;\n vertex_normal.z /= radiusZ;\n vertex_normal.normalize();\n const uv_x0 = Vector2.Lerp(face_vertex_uv[0], face_vertex_uv[2], i2 / subdivisions);\n const uv_x1 = Vector2.Lerp(face_vertex_uv[1], face_vertex_uv[2], i2 / subdivisions);\n const uv_interp = subdivisions === i2 ? face_vertex_uv[2] : Vector2.Lerp(uv_x0, uv_x1, i1 / (subdivisions - i2));\n positions.push(pos_interp.x * radiusX, pos_interp.y * radiusY, pos_interp.z * radiusZ);\n normals.push(vertex_normal.x, vertex_normal.y, vertex_normal.z);\n uvs.push(uv_interp.x, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - uv_interp.y : uv_interp.y);\n // push each vertex has member of a face\n // Same vertex can belong to multiple face, it is pushed multiple time (duplicate vertex are present)\n indices.push(current_indice);\n current_indice++;\n };\n for (let i2 = 0; i2 < subdivisions; i2++) {\n for (let i1 = 0; i1 + i2 < subdivisions; i1++) {\n // face : (i1,i2) for /\\ :\n // interp for : (i1,i2),(i1+1,i2),(i1,i2+1)\n interp_vertex(i1, i2, i1 + 1.0 / 3, i2 + 1.0 / 3);\n interp_vertex(i1 + 1, i2, i1 + 1.0 / 3, i2 + 1.0 / 3);\n interp_vertex(i1, i2 + 1, i1 + 1.0 / 3, i2 + 1.0 / 3);\n if (i1 + i2 + 1 < subdivisions) {\n // face : (i1,i2)' for \\/ :\n // interp for (i1+1,i2),(i1+1,i2+1),(i1,i2+1)\n interp_vertex(i1 + 1, i2, i1 + 2.0 / 3, i2 + 2.0 / 3);\n interp_vertex(i1 + 1, i2 + 1, i1 + 2.0 / 3, i2 + 2.0 / 3);\n interp_vertex(i1, i2 + 1, i1 + 2.0 / 3, i2 + 2.0 / 3);\n }\n }\n }\n }\n // Sides\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\n // Result\n const vertexData = new VertexData();\n vertexData.indices = indices;\n vertexData.positions = positions;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n return vertexData;\n }\n /**\n * Creates a sphere based upon an icosahedron with 20 triangular faces which can be subdivided\n * * The parameter `radius` sets the radius size (float) of the icosphere (default 1)\n * * You can set some different icosphere dimensions, for instance to build an ellipsoid, by using the parameters `radiusX`, `radiusY` and `radiusZ` (all by default have the same value of `radius`)\n * * The parameter `subdivisions` sets the number of subdivisions (positive integer, default 4). The more subdivisions, the more faces on the icosphere whatever its size\n * * The parameter `flat` (boolean, default true) gives each side its own normals. Set it to false to get a smooth continuous light reflection on the surface\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param options.radius\n * @param options.radiusX\n * @param options.radiusY\n * @param options.radiusZ\n * @param options.flat\n * @param options.subdivisions\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param options.updatable\n * @param scene defines the hosting scene\n * @returns the icosahedron mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/polyhedra#icosphere\n */\n function CreateIcoSphere(name, options = {}, scene = null) {\n const sphere = new Mesh(name, scene);\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n sphere._originalBuilderSideOrientation = options.sideOrientation;\n const vertexData = CreateIcoSphereVertexData(options);\n vertexData.applyToMesh(sphere, options.updatable);\n return sphere;\n }\n VertexData.CreateIcoSphere = CreateIcoSphereVertexData;\n Mesh.CreateIcoSphere = (name, options, scene) => {\n return CreateIcoSphere(name, options, scene);\n };\n\n const xpAxis = new Vector3(1, 0, 0);\n const xnAxis = new Vector3(-1, 0, 0);\n const ypAxis = new Vector3(0, 1, 0);\n const ynAxis = new Vector3(0, -1, 0);\n const zpAxis = new Vector3(0, 0, 1);\n const znAxis = new Vector3(0, 0, -1);\n /** @internal */\n class DecalVertex {\n constructor(position = Vector3.Zero(), normal = Vector3.Up(), uv = Vector2.Zero(), vertexIdx = 0, vertexIdxForBones = 0, localPositionOverride = null, localNormalOverride = null, matrixIndicesOverride = null, matrixWeightsOverride = null) {\n this.position = position;\n this.normal = normal;\n this.uv = uv;\n this.vertexIdx = vertexIdx;\n this.vertexIdxForBones = vertexIdxForBones;\n this.localPositionOverride = localPositionOverride;\n this.localNormalOverride = localNormalOverride;\n this.matrixIndicesOverride = matrixIndicesOverride;\n this.matrixWeightsOverride = matrixWeightsOverride;\n }\n clone() {\n var _a, _b, _c, _d;\n return new DecalVertex(this.position.clone(), this.normal.clone(), this.uv.clone(), this.vertexIdx, this.vertexIdxForBones, (_a = this.localPositionOverride) === null || _a === void 0 ? void 0 : _a.slice(), (_b = this.localNormalOverride) === null || _b === void 0 ? void 0 : _b.slice(), (_c = this.matrixIndicesOverride) === null || _c === void 0 ? void 0 : _c.slice(), (_d = this.matrixWeightsOverride) === null || _d === void 0 ? void 0 : _d.slice());\n }\n }\n /**\n * Creates a decal mesh.\n * A decal is a mesh usually applied as a model onto the surface of another mesh. So don't forget the parameter `sourceMesh` depicting the decal\n * * The parameter `position` (Vector3, default `(0, 0, 0)`) sets the position of the decal in World coordinates\n * * The parameter `normal` (Vector3, default `Vector3.Up`) sets the normal of the mesh where the decal is applied onto in World coordinates\n * * The parameter `size` (Vector3, default `(1, 1, 1)`) sets the decal scaling\n * * The parameter `angle` (float in radian, default 0) sets the angle to rotate the decal\n * * The parameter `captureUVS` defines if we need to capture the uvs or compute them\n * * The parameter `cullBackFaces` defines if the back faces should be removed from the decal mesh\n * * The parameter `localMode` defines that the computations should be done with the local mesh coordinates instead of the world space coordinates.\n * * Use this mode if you want the decal to be parented to the sourceMesh and move/rotate with it.\n * Note: Meshes with morph targets are not supported!\n * @param name defines the name of the mesh\n * @param sourceMesh defines the mesh where the decal must be applied\n * @param options defines the options used to create the mesh\n * @param options.position\n * @param options.normal\n * @param options.size\n * @param options.angle\n * @param options.captureUVS\n * @param options.cullBackFaces\n * @param options.localMode\n * @returns the decal mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/decals\n */\n function CreateDecal(name, sourceMesh, options) {\n var _a, _b, _c, _d;\n const hasSkeleton = !!sourceMesh.skeleton;\n const useLocalComputation = options.localMode || hasSkeleton;\n const meshHasOverridenMaterial = sourceMesh.overrideMaterialSideOrientation !== null && sourceMesh.overrideMaterialSideOrientation !== undefined;\n const indices = sourceMesh.getIndices();\n const positions = hasSkeleton ? sourceMesh.getPositionData(true, true) : sourceMesh.getVerticesData(VertexBuffer.PositionKind);\n const normals = hasSkeleton ? sourceMesh.getNormalsData(true, true) : sourceMesh.getVerticesData(VertexBuffer.NormalKind);\n const localPositions = useLocalComputation ? (hasSkeleton ? sourceMesh.getVerticesData(VertexBuffer.PositionKind) : positions) : null;\n const localNormals = useLocalComputation ? (hasSkeleton ? sourceMesh.getVerticesData(VertexBuffer.NormalKind) : normals) : null;\n const uvs = sourceMesh.getVerticesData(VertexBuffer.UVKind);\n const matIndices = hasSkeleton ? sourceMesh.getVerticesData(VertexBuffer.MatricesIndicesKind) : null;\n const matWeights = hasSkeleton ? sourceMesh.getVerticesData(VertexBuffer.MatricesWeightsKind) : null;\n const matIndicesExtra = hasSkeleton ? sourceMesh.getVerticesData(VertexBuffer.MatricesIndicesExtraKind) : null;\n const matWeightsExtra = hasSkeleton ? sourceMesh.getVerticesData(VertexBuffer.MatricesWeightsExtraKind) : null;\n const position = options.position || Vector3.Zero();\n let normal = options.normal || Vector3.Up();\n const size = options.size || Vector3.One();\n const angle = options.angle || 0;\n // Getting correct rotation\n if (!normal) {\n const target = new Vector3(0, 0, 1);\n const camera = sourceMesh.getScene().activeCamera;\n const cameraWorldTarget = Vector3.TransformCoordinates(target, camera.getWorldMatrix());\n normal = camera.globalPosition.subtract(cameraWorldTarget);\n }\n const yaw = -Math.atan2(normal.z, normal.x) - Math.PI / 2;\n const len = Math.sqrt(normal.x * normal.x + normal.z * normal.z);\n const pitch = Math.atan2(normal.y, len);\n const vertexData = new VertexData();\n vertexData.indices = [];\n vertexData.positions = [];\n vertexData.normals = [];\n vertexData.uvs = [];\n vertexData.matricesIndices = hasSkeleton ? [] : null;\n vertexData.matricesWeights = hasSkeleton ? [] : null;\n vertexData.matricesIndicesExtra = matIndicesExtra ? [] : null;\n vertexData.matricesWeightsExtra = matWeightsExtra ? [] : null;\n let currentVertexDataIndex = 0;\n const extractDecalVector3 = (indexId, transformMatrix) => {\n const result = new DecalVertex();\n if (!indices || !positions || !normals) {\n return result;\n }\n const vertexId = indices[indexId];\n result.vertexIdx = vertexId * 3;\n result.vertexIdxForBones = vertexId * 4;\n // Send vector to decal local world\n result.position = new Vector3(positions[vertexId * 3], positions[vertexId * 3 + 1], positions[vertexId * 3 + 2]);\n Vector3.TransformCoordinatesToRef(result.position, transformMatrix, result.position);\n // Get normal\n result.normal = new Vector3(normals[vertexId * 3], normals[vertexId * 3 + 1], normals[vertexId * 3 + 2]);\n Vector3.TransformNormalToRef(result.normal, transformMatrix, result.normal);\n if (options.captureUVS && uvs) {\n const v = uvs[vertexId * 2 + 1];\n result.uv = new Vector2(uvs[vertexId * 2], CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - v : v);\n }\n return result;\n };\n const emptyArray = [0, 0, 0, 0];\n // Inspired by https://github.com/mrdoob/three.js/blob/eee231960882f6f3b6113405f524956145148146/examples/js/geometries/DecalGeometry.js\n const clip = (vertices, axis) => {\n if (vertices.length === 0) {\n return vertices;\n }\n const clipSize = 0.5 * Math.abs(Vector3.Dot(size, axis));\n const indexOf = (arr, val, start, num) => {\n for (let i = 0; i < num; ++i) {\n if (arr[start + i] === val) {\n return start + i;\n }\n }\n return -1;\n };\n const clipVertices = (v0, v1) => {\n var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m, _o, _p, _q, _r;\n const clipFactor = Vector3.GetClipFactor(v0.position, v1.position, axis, clipSize);\n let indices = emptyArray;\n let weights = emptyArray;\n if (matIndices && matWeights) {\n const mat0Index = v0.matrixIndicesOverride ? 0 : v0.vertexIdxForBones;\n const v0Indices = (_a = v0.matrixIndicesOverride) !== null && _a !== void 0 ? _a : matIndices;\n const v0Weights = (_b = v0.matrixWeightsOverride) !== null && _b !== void 0 ? _b : matWeights;\n const mat1Index = v1.matrixIndicesOverride ? 0 : v1.vertexIdxForBones;\n const v1Indices = (_c = v1.matrixIndicesOverride) !== null && _c !== void 0 ? _c : matIndices;\n const v1Weights = (_d = v1.matrixWeightsOverride) !== null && _d !== void 0 ? _d : matWeights;\n indices = [0, 0, 0, 0];\n weights = [0, 0, 0, 0];\n let index = 0;\n for (let i = 0; i < 4; ++i) {\n if (v0Weights[mat0Index + i] > 0) {\n const idx = indexOf(v1Indices, v0Indices[mat0Index + i], mat1Index, 4);\n indices[index] = v0Indices[mat0Index + i];\n weights[index] = Scalar.Lerp(v0Weights[mat0Index + i], idx >= 0 ? v1Weights[idx] : 0, clipFactor);\n index++;\n }\n }\n for (let i = 0; i < 4 && index < 4; ++i) {\n const ind = v1Indices[mat1Index + i];\n if (indexOf(v0Indices, ind, mat0Index, 4) !== -1)\n continue;\n indices[index] = ind;\n weights[index] = Scalar.Lerp(0, v1Weights[mat1Index + i], clipFactor);\n index++;\n }\n const sumw = weights[0] + weights[1] + weights[2] + weights[3];\n weights[0] /= sumw;\n weights[1] /= sumw;\n weights[2] /= sumw;\n weights[3] /= sumw;\n }\n const v0LocalPositionX = v0.localPositionOverride ? v0.localPositionOverride[0] : (_e = localPositions === null || localPositions === void 0 ? void 0 : localPositions[v0.vertexIdx]) !== null && _e !== void 0 ? _e : 0;\n const v0LocalPositionY = v0.localPositionOverride ? v0.localPositionOverride[1] : (_f = localPositions === null || localPositions === void 0 ? void 0 : localPositions[v0.vertexIdx + 1]) !== null && _f !== void 0 ? _f : 0;\n const v0LocalPositionZ = v0.localPositionOverride ? v0.localPositionOverride[2] : (_g = localPositions === null || localPositions === void 0 ? void 0 : localPositions[v0.vertexIdx + 2]) !== null && _g !== void 0 ? _g : 0;\n const v1LocalPositionX = v1.localPositionOverride ? v1.localPositionOverride[0] : (_h = localPositions === null || localPositions === void 0 ? void 0 : localPositions[v1.vertexIdx]) !== null && _h !== void 0 ? _h : 0;\n const v1LocalPositionY = v1.localPositionOverride ? v1.localPositionOverride[1] : (_j = localPositions === null || localPositions === void 0 ? void 0 : localPositions[v1.vertexIdx + 1]) !== null && _j !== void 0 ? _j : 0;\n const v1LocalPositionZ = v1.localPositionOverride ? v1.localPositionOverride[2] : (_k = localPositions === null || localPositions === void 0 ? void 0 : localPositions[v1.vertexIdx + 2]) !== null && _k !== void 0 ? _k : 0;\n const v0LocalNormalX = v0.localNormalOverride ? v0.localNormalOverride[0] : (_l = localNormals === null || localNormals === void 0 ? void 0 : localNormals[v0.vertexIdx]) !== null && _l !== void 0 ? _l : 0;\n const v0LocalNormalY = v0.localNormalOverride ? v0.localNormalOverride[1] : (_m = localNormals === null || localNormals === void 0 ? void 0 : localNormals[v0.vertexIdx + 1]) !== null && _m !== void 0 ? _m : 0;\n const v0LocalNormalZ = v0.localNormalOverride ? v0.localNormalOverride[2] : (_o = localNormals === null || localNormals === void 0 ? void 0 : localNormals[v0.vertexIdx + 2]) !== null && _o !== void 0 ? _o : 0;\n const v1LocalNormalX = v1.localNormalOverride ? v1.localNormalOverride[0] : (_p = localNormals === null || localNormals === void 0 ? void 0 : localNormals[v1.vertexIdx]) !== null && _p !== void 0 ? _p : 0;\n const v1LocalNormalY = v1.localNormalOverride ? v1.localNormalOverride[1] : (_q = localNormals === null || localNormals === void 0 ? void 0 : localNormals[v1.vertexIdx + 1]) !== null && _q !== void 0 ? _q : 0;\n const v1LocalNormalZ = v1.localNormalOverride ? v1.localNormalOverride[2] : (_r = localNormals === null || localNormals === void 0 ? void 0 : localNormals[v1.vertexIdx + 2]) !== null && _r !== void 0 ? _r : 0;\n const interpNormalX = v0LocalNormalX + (v1LocalNormalX - v0LocalNormalX) * clipFactor;\n const interpNormalY = v0LocalNormalY + (v1LocalNormalY - v0LocalNormalY) * clipFactor;\n const interpNormalZ = v0LocalNormalZ + (v1LocalNormalZ - v0LocalNormalZ) * clipFactor;\n const norm = Math.sqrt(interpNormalX * interpNormalX + interpNormalY * interpNormalY + interpNormalZ * interpNormalZ);\n return new DecalVertex(Vector3.Lerp(v0.position, v1.position, clipFactor), Vector3.Lerp(v0.normal, v1.normal, clipFactor).normalize(), Vector2.Lerp(v0.uv, v1.uv, clipFactor), -1, -1, localPositions\n ? [\n v0LocalPositionX + (v1LocalPositionX - v0LocalPositionX) * clipFactor,\n v0LocalPositionY + (v1LocalPositionY - v0LocalPositionY) * clipFactor,\n v0LocalPositionZ + (v1LocalPositionZ - v0LocalPositionZ) * clipFactor,\n ]\n : null, localNormals ? [interpNormalX / norm, interpNormalY / norm, interpNormalZ / norm] : null, indices, weights);\n };\n let clipResult = null;\n if (vertices.length > 3) {\n clipResult = new Array();\n }\n for (let index = 0; index < vertices.length; index += 3) {\n let total = 0;\n let nV1 = null;\n let nV2 = null;\n let nV3 = null;\n let nV4 = null;\n const d1 = Vector3.Dot(vertices[index].position, axis) - clipSize;\n const d2 = Vector3.Dot(vertices[index + 1].position, axis) - clipSize;\n const d3 = Vector3.Dot(vertices[index + 2].position, axis) - clipSize;\n const v1Out = d1 > 0;\n const v2Out = d2 > 0;\n const v3Out = d3 > 0;\n total = (v1Out ? 1 : 0) + (v2Out ? 1 : 0) + (v3Out ? 1 : 0);\n switch (total) {\n case 0:\n if (vertices.length > 3) {\n clipResult.push(vertices[index]);\n clipResult.push(vertices[index + 1]);\n clipResult.push(vertices[index + 2]);\n }\n else {\n clipResult = vertices;\n }\n break;\n case 1:\n clipResult = clipResult !== null && clipResult !== void 0 ? clipResult : new Array();\n if (v1Out) {\n nV1 = vertices[index + 1];\n nV2 = vertices[index + 2];\n nV3 = clipVertices(vertices[index], nV1);\n nV4 = clipVertices(vertices[index], nV2);\n }\n if (v2Out) {\n nV1 = vertices[index];\n nV2 = vertices[index + 2];\n nV3 = clipVertices(vertices[index + 1], nV1);\n nV4 = clipVertices(vertices[index + 1], nV2);\n clipResult.push(nV3);\n clipResult.push(nV2.clone());\n clipResult.push(nV1.clone());\n clipResult.push(nV2.clone());\n clipResult.push(nV3.clone());\n clipResult.push(nV4);\n break;\n }\n if (v3Out) {\n nV1 = vertices[index];\n nV2 = vertices[index + 1];\n nV3 = clipVertices(vertices[index + 2], nV1);\n nV4 = clipVertices(vertices[index + 2], nV2);\n }\n if (nV1 && nV2 && nV3 && nV4) {\n clipResult.push(nV1.clone());\n clipResult.push(nV2.clone());\n clipResult.push(nV3);\n clipResult.push(nV4);\n clipResult.push(nV3.clone());\n clipResult.push(nV2.clone());\n }\n break;\n case 2:\n clipResult = clipResult !== null && clipResult !== void 0 ? clipResult : new Array();\n if (!v1Out) {\n nV1 = vertices[index].clone();\n nV2 = clipVertices(nV1, vertices[index + 1]);\n nV3 = clipVertices(nV1, vertices[index + 2]);\n clipResult.push(nV1);\n clipResult.push(nV2);\n clipResult.push(nV3);\n }\n if (!v2Out) {\n nV1 = vertices[index + 1].clone();\n nV2 = clipVertices(nV1, vertices[index + 2]);\n nV3 = clipVertices(nV1, vertices[index]);\n clipResult.push(nV1);\n clipResult.push(nV2);\n clipResult.push(nV3);\n }\n if (!v3Out) {\n nV1 = vertices[index + 2].clone();\n nV2 = clipVertices(nV1, vertices[index]);\n nV3 = clipVertices(nV1, vertices[index + 1]);\n clipResult.push(nV1);\n clipResult.push(nV2);\n clipResult.push(nV3);\n }\n break;\n }\n }\n return clipResult;\n };\n const sourceMeshAsMesh = sourceMesh instanceof Mesh ? sourceMesh : null;\n const matrixData = sourceMeshAsMesh === null || sourceMeshAsMesh === void 0 ? void 0 : sourceMeshAsMesh._thinInstanceDataStorage.matrixData;\n const numMatrices = (sourceMeshAsMesh === null || sourceMeshAsMesh === void 0 ? void 0 : sourceMeshAsMesh.thinInstanceCount) || 1;\n const thinInstanceMatrix = TmpVectors.Matrix[0];\n thinInstanceMatrix.copyFrom(Matrix.IdentityReadOnly);\n for (let m = 0; m < numMatrices; ++m) {\n if ((sourceMeshAsMesh === null || sourceMeshAsMesh === void 0 ? void 0 : sourceMeshAsMesh.hasThinInstances) && matrixData) {\n const ofst = m * 16;\n thinInstanceMatrix.setRowFromFloats(0, matrixData[ofst + 0], matrixData[ofst + 1], matrixData[ofst + 2], matrixData[ofst + 3]);\n thinInstanceMatrix.setRowFromFloats(1, matrixData[ofst + 4], matrixData[ofst + 5], matrixData[ofst + 6], matrixData[ofst + 7]);\n thinInstanceMatrix.setRowFromFloats(2, matrixData[ofst + 8], matrixData[ofst + 9], matrixData[ofst + 10], matrixData[ofst + 11]);\n thinInstanceMatrix.setRowFromFloats(3, matrixData[ofst + 12], matrixData[ofst + 13], matrixData[ofst + 14], matrixData[ofst + 15]);\n }\n // Matrix\n const decalWorldMatrix = Matrix.RotationYawPitchRoll(yaw, pitch, angle).multiply(Matrix.Translation(position.x, position.y, position.z));\n const inverseDecalWorldMatrix = Matrix.Invert(decalWorldMatrix);\n const meshWorldMatrix = sourceMesh.getWorldMatrix();\n const transformMatrix = thinInstanceMatrix.multiply(meshWorldMatrix).multiply(inverseDecalWorldMatrix);\n const oneFaceVertices = new Array(3);\n for (let index = 0; index < indices.length; index += 3) {\n let faceVertices = oneFaceVertices;\n faceVertices[0] = extractDecalVector3(index, transformMatrix);\n if (meshHasOverridenMaterial && useLocalComputation) {\n faceVertices[1] = extractDecalVector3(index + 2, transformMatrix);\n faceVertices[2] = extractDecalVector3(index + 1, transformMatrix);\n }\n else {\n faceVertices[1] = extractDecalVector3(index + 1, transformMatrix);\n faceVertices[2] = extractDecalVector3(index + 2, transformMatrix);\n }\n if (options.cullBackFaces) {\n // If all the normals of the vertices of the face are pointing away from the view direction we discard the face.\n // As computations are done in the decal coordinate space, the viewDirection is (0,0,1), so when dot(vertexNormal, -viewDirection) <= 0 the vertex is culled\n if (-faceVertices[0].normal.z <= 0 && -faceVertices[1].normal.z <= 0 && -faceVertices[2].normal.z <= 0) {\n continue;\n }\n }\n // Clip\n faceVertices = clip(faceVertices, xpAxis);\n if (!faceVertices)\n continue;\n faceVertices = clip(faceVertices, xnAxis);\n if (!faceVertices)\n continue;\n faceVertices = clip(faceVertices, ypAxis);\n if (!faceVertices)\n continue;\n faceVertices = clip(faceVertices, ynAxis);\n if (!faceVertices)\n continue;\n faceVertices = clip(faceVertices, zpAxis);\n if (!faceVertices)\n continue;\n faceVertices = clip(faceVertices, znAxis);\n if (!faceVertices)\n continue;\n // Add UVs and get back to world\n for (let vIndex = 0; vIndex < faceVertices.length; vIndex++) {\n const vertex = faceVertices[vIndex];\n //TODO check for Int32Array | Uint32Array | Uint16Array\n vertexData.indices.push(currentVertexDataIndex);\n if (useLocalComputation) {\n if (vertex.localPositionOverride) {\n vertexData.positions[currentVertexDataIndex * 3] = vertex.localPositionOverride[0];\n vertexData.positions[currentVertexDataIndex * 3 + 1] = vertex.localPositionOverride[1];\n vertexData.positions[currentVertexDataIndex * 3 + 2] = vertex.localPositionOverride[2];\n }\n else if (localPositions) {\n vertexData.positions[currentVertexDataIndex * 3] = localPositions[vertex.vertexIdx];\n vertexData.positions[currentVertexDataIndex * 3 + 1] = localPositions[vertex.vertexIdx + 1];\n vertexData.positions[currentVertexDataIndex * 3 + 2] = localPositions[vertex.vertexIdx + 2];\n }\n if (vertex.localNormalOverride) {\n vertexData.normals[currentVertexDataIndex * 3] = vertex.localNormalOverride[0];\n vertexData.normals[currentVertexDataIndex * 3 + 1] = vertex.localNormalOverride[1];\n vertexData.normals[currentVertexDataIndex * 3 + 2] = vertex.localNormalOverride[2];\n }\n else if (localNormals) {\n vertexData.normals[currentVertexDataIndex * 3] = localNormals[vertex.vertexIdx];\n vertexData.normals[currentVertexDataIndex * 3 + 1] = localNormals[vertex.vertexIdx + 1];\n vertexData.normals[currentVertexDataIndex * 3 + 2] = localNormals[vertex.vertexIdx + 2];\n }\n }\n else {\n vertex.position.toArray(vertexData.positions, currentVertexDataIndex * 3);\n vertex.normal.toArray(vertexData.normals, currentVertexDataIndex * 3);\n }\n if (vertexData.matricesIndices && vertexData.matricesWeights) {\n if (vertex.matrixIndicesOverride) {\n vertexData.matricesIndices[currentVertexDataIndex * 4] = vertex.matrixIndicesOverride[0];\n vertexData.matricesIndices[currentVertexDataIndex * 4 + 1] = vertex.matrixIndicesOverride[1];\n vertexData.matricesIndices[currentVertexDataIndex * 4 + 2] = vertex.matrixIndicesOverride[2];\n vertexData.matricesIndices[currentVertexDataIndex * 4 + 3] = vertex.matrixIndicesOverride[3];\n }\n else {\n if (matIndices) {\n vertexData.matricesIndices[currentVertexDataIndex * 4] = matIndices[vertex.vertexIdxForBones];\n vertexData.matricesIndices[currentVertexDataIndex * 4 + 1] = matIndices[vertex.vertexIdxForBones + 1];\n vertexData.matricesIndices[currentVertexDataIndex * 4 + 2] = matIndices[vertex.vertexIdxForBones + 2];\n vertexData.matricesIndices[currentVertexDataIndex * 4 + 3] = matIndices[vertex.vertexIdxForBones + 3];\n }\n if (matIndicesExtra && vertexData.matricesIndicesExtra) {\n vertexData.matricesIndicesExtra[currentVertexDataIndex * 4] = matIndicesExtra[vertex.vertexIdxForBones];\n vertexData.matricesIndicesExtra[currentVertexDataIndex * 4 + 1] = matIndicesExtra[vertex.vertexIdxForBones + 1];\n vertexData.matricesIndicesExtra[currentVertexDataIndex * 4 + 2] = matIndicesExtra[vertex.vertexIdxForBones + 2];\n vertexData.matricesIndicesExtra[currentVertexDataIndex * 4 + 3] = matIndicesExtra[vertex.vertexIdxForBones + 3];\n }\n }\n if (vertex.matrixWeightsOverride) {\n vertexData.matricesWeights[currentVertexDataIndex * 4] = vertex.matrixWeightsOverride[0];\n vertexData.matricesWeights[currentVertexDataIndex * 4 + 1] = vertex.matrixWeightsOverride[1];\n vertexData.matricesWeights[currentVertexDataIndex * 4 + 2] = vertex.matrixWeightsOverride[2];\n vertexData.matricesWeights[currentVertexDataIndex * 4 + 3] = vertex.matrixWeightsOverride[3];\n }\n else {\n if (matWeights) {\n vertexData.matricesWeights[currentVertexDataIndex * 4] = matWeights[vertex.vertexIdxForBones];\n vertexData.matricesWeights[currentVertexDataIndex * 4 + 1] = matWeights[vertex.vertexIdxForBones + 1];\n vertexData.matricesWeights[currentVertexDataIndex * 4 + 2] = matWeights[vertex.vertexIdxForBones + 2];\n vertexData.matricesWeights[currentVertexDataIndex * 4 + 3] = matWeights[vertex.vertexIdxForBones + 3];\n }\n if (matWeightsExtra && vertexData.matricesWeightsExtra) {\n vertexData.matricesWeightsExtra[currentVertexDataIndex * 4] = matWeightsExtra[vertex.vertexIdxForBones];\n vertexData.matricesWeightsExtra[currentVertexDataIndex * 4 + 1] = matWeightsExtra[vertex.vertexIdxForBones + 1];\n vertexData.matricesWeightsExtra[currentVertexDataIndex * 4 + 2] = matWeightsExtra[vertex.vertexIdxForBones + 2];\n vertexData.matricesWeightsExtra[currentVertexDataIndex * 4 + 3] = matWeightsExtra[vertex.vertexIdxForBones + 3];\n }\n }\n }\n if (!options.captureUVS) {\n vertexData.uvs.push(0.5 + vertex.position.x / size.x);\n const v = 0.5 + vertex.position.y / size.y;\n vertexData.uvs.push(CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - v : v);\n }\n else {\n vertex.uv.toArray(vertexData.uvs, currentVertexDataIndex * 2);\n }\n currentVertexDataIndex++;\n }\n }\n }\n // Avoid the \"Setting vertex data kind 'XXX' with an empty array\" warning when calling vertexData.applyToMesh\n if (vertexData.indices.length === 0)\n vertexData.indices = null;\n if (vertexData.positions.length === 0)\n vertexData.positions = null;\n if (vertexData.normals.length === 0)\n vertexData.normals = null;\n if (vertexData.uvs.length === 0)\n vertexData.uvs = null;\n if (((_a = vertexData.matricesIndices) === null || _a === void 0 ? void 0 : _a.length) === 0)\n vertexData.matricesIndices = null;\n if (((_b = vertexData.matricesWeights) === null || _b === void 0 ? void 0 : _b.length) === 0)\n vertexData.matricesWeights = null;\n if (((_c = vertexData.matricesIndicesExtra) === null || _c === void 0 ? void 0 : _c.length) === 0)\n vertexData.matricesIndicesExtra = null;\n if (((_d = vertexData.matricesWeightsExtra) === null || _d === void 0 ? void 0 : _d.length) === 0)\n vertexData.matricesWeightsExtra = null;\n // Return mesh\n const decal = new Mesh(name, sourceMesh.getScene());\n vertexData.applyToMesh(decal);\n if (useLocalComputation) {\n decal.skeleton = sourceMesh.skeleton;\n decal.parent = sourceMesh;\n }\n else {\n decal.position = position.clone();\n decal.rotation = new Vector3(pitch, yaw, angle);\n }\n decal.computeWorldMatrix(true);\n decal.refreshBoundingInfo(true, true);\n return decal;\n }\n Mesh.CreateDecal = (name, sourceMesh, position, normal, size, angle) => {\n const options = {\n position,\n normal,\n size,\n angle,\n };\n return CreateDecal(name, sourceMesh, options);\n };\n\n /**\n * Scripts based off of https://github.com/maximeq/three-js-capsule-geometry/blob/master/src/CapsuleBufferGeometry.js\n * @param options the constructors options used to shape the mesh.\n * @returns the capsule VertexData\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set/capsule\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n function CreateCapsuleVertexData(options = {\n subdivisions: 2,\n tessellation: 16,\n height: 1,\n radius: 0.25,\n capSubdivisions: 6,\n }) {\n const subdivisions = Math.max(options.subdivisions ? options.subdivisions : 2, 1) | 0;\n const tessellation = Math.max(options.tessellation ? options.tessellation : 16, 3) | 0;\n const height = Math.max(options.height ? options.height : 1, 0);\n const radius = Math.max(options.radius ? options.radius : 0.25, 0);\n const capDetail = Math.max(options.capSubdivisions ? options.capSubdivisions : 6, 1) | 0;\n const radialSegments = tessellation;\n const heightSegments = subdivisions;\n const radiusTop = Math.max(options.radiusTop ? options.radiusTop : radius, 0);\n const radiusBottom = Math.max(options.radiusBottom ? options.radiusBottom : radius, 0);\n const heightMinusCaps = height - (radiusTop + radiusBottom);\n const thetaStart = 0.0;\n const thetaLength = 2.0 * Math.PI;\n const capsTopSegments = Math.max(options.topCapSubdivisions ? options.topCapSubdivisions : capDetail, 1);\n const capsBottomSegments = Math.max(options.bottomCapSubdivisions ? options.bottomCapSubdivisions : capDetail, 1);\n const alpha = Math.acos((radiusBottom - radiusTop) / height);\n let indices = [];\n const vertices = [];\n const normals = [];\n const uvs = [];\n let index = 0;\n const indexArray = [], halfHeight = heightMinusCaps * 0.5;\n const pi2 = Math.PI * 0.5;\n let x, y;\n const normal = Vector3.Zero();\n const vertex = Vector3.Zero();\n const cosAlpha = Math.cos(alpha);\n const sinAlpha = Math.sin(alpha);\n const coneLength = new Vector2(radiusTop * sinAlpha, halfHeight + radiusTop * cosAlpha)\n .subtract(new Vector2(radiusBottom * sinAlpha, -halfHeight + radiusBottom * cosAlpha))\n .length();\n // Total length for v texture coord\n const vl = radiusTop * alpha + coneLength + radiusBottom * (pi2 - alpha);\n let v = 0;\n for (y = 0; y <= capsTopSegments; y++) {\n const indexRow = [];\n const a = pi2 - alpha * (y / capsTopSegments);\n v += (radiusTop * alpha) / capsTopSegments;\n const cosA = Math.cos(a);\n const sinA = Math.sin(a);\n // calculate the radius of the current row\n const _radius = cosA * radiusTop;\n for (x = 0; x <= radialSegments; x++) {\n const u = x / radialSegments;\n const theta = u * thetaLength + thetaStart;\n const sinTheta = Math.sin(theta);\n const cosTheta = Math.cos(theta);\n // vertex\n vertex.x = _radius * sinTheta;\n vertex.y = halfHeight + sinA * radiusTop;\n vertex.z = _radius * cosTheta;\n vertices.push(vertex.x, vertex.y, vertex.z);\n // normal\n normal.set(cosA * sinTheta, sinA, cosA * cosTheta);\n normals.push(normal.x, normal.y, normal.z);\n // uv\n uvs.push(u, CompatibilityOptions.UseOpenGLOrientationForUV ? v / vl : 1 - v / vl);\n // save index of vertex in respective row\n indexRow.push(index);\n // increase index\n index++;\n }\n // now save vertices of the row in our index array\n indexArray.push(indexRow);\n }\n const coneHeight = height - radiusTop - radiusBottom + cosAlpha * radiusTop - cosAlpha * radiusBottom;\n const slope = (sinAlpha * (radiusBottom - radiusTop)) / coneHeight;\n for (y = 1; y <= heightSegments; y++) {\n const indexRow = [];\n v += coneLength / heightSegments;\n // calculate the radius of the current row\n const _radius = sinAlpha * ((y * (radiusBottom - radiusTop)) / heightSegments + radiusTop);\n for (x = 0; x <= radialSegments; x++) {\n const u = x / radialSegments;\n const theta = u * thetaLength + thetaStart;\n const sinTheta = Math.sin(theta);\n const cosTheta = Math.cos(theta);\n // vertex\n vertex.x = _radius * sinTheta;\n vertex.y = halfHeight + cosAlpha * radiusTop - (y * coneHeight) / heightSegments;\n vertex.z = _radius * cosTheta;\n vertices.push(vertex.x, vertex.y, vertex.z);\n // normal\n normal.set(sinTheta, slope, cosTheta).normalize();\n normals.push(normal.x, normal.y, normal.z);\n // uv\n uvs.push(u, CompatibilityOptions.UseOpenGLOrientationForUV ? v / vl : 1 - v / vl);\n // save index of vertex in respective row\n indexRow.push(index);\n // increase index\n index++;\n }\n // now save vertices of the row in our index array\n indexArray.push(indexRow);\n }\n for (y = 1; y <= capsBottomSegments; y++) {\n const indexRow = [];\n const a = pi2 - alpha - (Math.PI - alpha) * (y / capsBottomSegments);\n v += (radiusBottom * alpha) / capsBottomSegments;\n const cosA = Math.cos(a);\n const sinA = Math.sin(a);\n // calculate the radius of the current row\n const _radius = cosA * radiusBottom;\n for (x = 0; x <= radialSegments; x++) {\n const u = x / radialSegments;\n const theta = u * thetaLength + thetaStart;\n const sinTheta = Math.sin(theta);\n const cosTheta = Math.cos(theta);\n // vertex\n vertex.x = _radius * sinTheta;\n vertex.y = -halfHeight + sinA * radiusBottom;\n vertex.z = _radius * cosTheta;\n vertices.push(vertex.x, vertex.y, vertex.z);\n // normal\n normal.set(cosA * sinTheta, sinA, cosA * cosTheta);\n normals.push(normal.x, normal.y, normal.z);\n // uv\n uvs.push(u, CompatibilityOptions.UseOpenGLOrientationForUV ? v / vl : 1 - v / vl);\n // save index of vertex in respective row\n indexRow.push(index);\n // increase index\n index++;\n }\n // now save vertices of the row in our index array\n indexArray.push(indexRow);\n }\n // generate indices\n for (x = 0; x < radialSegments; x++) {\n for (y = 0; y < capsTopSegments + heightSegments + capsBottomSegments; y++) {\n // we use the index array to access the correct indices\n const i1 = indexArray[y][x];\n const i2 = indexArray[y + 1][x];\n const i3 = indexArray[y + 1][x + 1];\n const i4 = indexArray[y][x + 1];\n // face one\n indices.push(i1);\n indices.push(i2);\n indices.push(i4);\n // face two\n indices.push(i2);\n indices.push(i3);\n indices.push(i4);\n }\n }\n indices = indices.reverse();\n if (options.orientation && !options.orientation.equals(Vector3.Up())) {\n const m = new Matrix();\n options.orientation\n .clone()\n .scale(Math.PI * 0.5)\n .cross(Vector3.Up())\n .toQuaternion()\n .toRotationMatrix(m);\n const v = Vector3.Zero();\n for (let i = 0; i < vertices.length; i += 3) {\n v.set(vertices[i], vertices[i + 1], vertices[i + 2]);\n Vector3.TransformCoordinatesToRef(v.clone(), m, v);\n vertices[i] = v.x;\n vertices[i + 1] = v.y;\n vertices[i + 2] = v.z;\n }\n }\n const vDat = new VertexData();\n vDat.positions = vertices;\n vDat.normals = normals;\n vDat.uvs = uvs;\n vDat.indices = indices;\n return vDat;\n }\n /**\n * Creates a capsule or a pill mesh\n * @param name defines the name of the mesh\n * @param options The constructors options.\n * @param scene The scene the mesh is scoped to.\n * @returns Capsule Mesh\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n function CreateCapsule(name, options = {\n orientation: Vector3.Up(),\n subdivisions: 2,\n tessellation: 16,\n height: 1,\n radius: 0.25,\n capSubdivisions: 6,\n updatable: false,\n }, scene = null) {\n const capsule = new Mesh(name, scene);\n const vertexData = CreateCapsuleVertexData(options);\n vertexData.applyToMesh(capsule, options.updatable);\n return capsule;\n }\n /**\n * Creates a capsule or a pill mesh\n * @param name defines the name of the mesh.\n * @param options the constructors options used to shape the mesh.\n * @param scene defines the scene the mesh is scoped to.\n * @returns the capsule mesh\n * @see https://doc.babylonjs.com/how_to/capsule_shape\n */\n Mesh.CreateCapsule = (name, options, scene) => {\n return CreateCapsule(name, options, scene);\n };\n VertexData.CreateCapsule = CreateCapsuleVertexData;\n\n /**\n * Class representing an isovector a vector containing 2 INTEGER coordinates\n * x axis is horizontal\n * y axis is 60 deg counter clockwise from positive y axis\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class _IsoVector {\n /**\n * Creates a new isovector from the given x and y coordinates\n * @param x defines the first coordinate, must be an integer\n * @param y defines the second coordinate, must be an integer\n */\n constructor(\n /** defines the first coordinate */\n x = 0, \n /** defines the second coordinate */\n y = 0) {\n this.x = x;\n this.y = y;\n if (x !== Math.floor(x)) {\n Logger.Warn(\"x is not an integer, floor(x) used\");\n }\n if (y !== Math.floor(y)) {\n Logger.Warn(\"y is not an integer, floor(y) used\");\n }\n }\n // Operators\n /**\n * Gets a new IsoVector copied from the IsoVector\n * @returns a new IsoVector\n */\n clone() {\n return new _IsoVector(this.x, this.y);\n }\n /**\n * Rotates one IsoVector 60 degrees counter clockwise about another\n * Please note that this is an in place operation\n * @param other an IsoVector a center of rotation\n * @returns the rotated IsoVector\n */\n rotate60About(other) {\n //other IsoVector\n const x = this.x;\n this.x = other.x + other.y - this.y;\n this.y = x + this.y - other.x;\n return this;\n }\n /**\n * Rotates one IsoVector 60 degrees clockwise about another\n * Please note that this is an in place operation\n * @param other an IsoVector as center of rotation\n * @returns the rotated IsoVector\n */\n rotateNeg60About(other) {\n const x = this.x;\n this.x = x + this.y - other.y;\n this.y = other.x + other.y - x;\n return this;\n }\n /**\n * For an equilateral triangle OAB with O at isovector (0, 0) and A at isovector (m, n)\n * Rotates one IsoVector 120 degrees counter clockwise about the center of the triangle\n * Please note that this is an in place operation\n * @param m integer a measure a Primary triangle of order (m, n) m > n\n * @param n >= 0 integer a measure for a Primary triangle of order (m, n)\n * @returns the rotated IsoVector\n */\n rotate120(m, n) {\n //m, n integers\n if (m !== Math.floor(m)) {\n Logger.Warn(\"m not an integer only floor(m) used\");\n }\n if (n !== Math.floor(n)) {\n Logger.Warn(\"n not an integer only floor(n) used\");\n }\n const x = this.x;\n this.x = m - x - this.y;\n this.y = n + x;\n return this;\n }\n /**\n * For an equilateral triangle OAB with O at isovector (0, 0) and A at isovector (m, n)\n * Rotates one IsoVector 120 degrees clockwise about the center of the triangle\n * Please note that this is an in place operation\n * @param m integer a measure a Primary triangle of order (m, n) m > n\n * @param n >= 0 integer a measure for a Primary triangle of order (m, n)\n * @returns the rotated IsoVector\n */\n rotateNeg120(m, n) {\n //m, n integers\n if (m !== Math.floor(m)) {\n Logger.Warn(\"m is not an integer, floor(m) used\");\n }\n if (n !== Math.floor(n)) {\n Logger.Warn(\"n is not an integer, floor(n) used\");\n }\n const x = this.x;\n this.x = this.y - n;\n this.y = m + n - x - this.y;\n return this;\n }\n /**\n * Transforms an IsoVector to one in Cartesian 3D space based on an isovector\n * @param origin an IsoVector\n * @param isoGridSize\n * @returns Point as a Vector3\n */\n toCartesianOrigin(origin, isoGridSize) {\n const point = Vector3.Zero();\n point.x = origin.x + 2 * this.x * isoGridSize + this.y * isoGridSize;\n point.y = origin.y + Math.sqrt(3) * this.y * isoGridSize;\n return point;\n }\n // Statics\n /**\n * Gets a new IsoVector(0, 0)\n * @returns a new IsoVector\n */\n static Zero() {\n return new _IsoVector(0, 0);\n }\n }\n\n /**\n * Class representing data for one face OAB of an equilateral icosahedron\n * When O is the isovector (0, 0), A is isovector (m, n)\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class _PrimaryIsoTriangle {\n constructor() {\n this.cartesian = [];\n this.vertices = [];\n this.max = [];\n this.min = [];\n this.closestTo = [];\n this.innerFacets = [];\n this.isoVecsABOB = [];\n this.isoVecsOBOA = [];\n this.isoVecsBAOA = [];\n this.vertexTypes = [];\n // eslint-disable-next-line @typescript-eslint/naming-convention\n this.IDATA = new PolyhedronData(\"icosahedron\", \"Regular\", [\n [0, PHI, -1],\n [-PHI, 1, 0],\n [-1, 0, -PHI],\n [1, 0, -PHI],\n [PHI, 1, 0],\n [0, PHI, 1],\n [-1, 0, PHI],\n [-PHI, -1, 0],\n [0, -PHI, -1],\n [PHI, -1, 0],\n [1, 0, PHI],\n [0, -PHI, 1],\n ], [\n [0, 2, 1],\n [0, 3, 2],\n [0, 4, 3],\n [0, 5, 4],\n [0, 1, 5],\n [7, 6, 1],\n [8, 7, 2],\n [9, 8, 3],\n [10, 9, 4],\n [6, 10, 5],\n [2, 7, 1],\n [3, 8, 2],\n [4, 9, 3],\n [5, 10, 4],\n [1, 6, 5],\n [11, 6, 7],\n [11, 7, 8],\n [11, 8, 9],\n [11, 9, 10],\n [11, 10, 6],\n ]);\n }\n /**\n * Creates the PrimaryIsoTriangle Triangle OAB\n * @param m an integer\n * @param n an integer\n */\n //operators\n setIndices() {\n let indexCount = 12; // 12 vertices already assigned\n const vecToidx = {}; //maps iso-vectors to indexCount;\n const m = this.m;\n const n = this.n;\n let g = m; // hcf of m, n when n != 0\n let m1 = 1;\n let n1 = 0;\n if (n !== 0) {\n g = Scalar.HCF(m, n);\n }\n m1 = m / g;\n n1 = n / g;\n let fr; //face to the right of current face\n let rot; //rotation about which vertex for fr\n let O;\n let A;\n let B;\n const oVec = _IsoVector.Zero();\n const aVec = new _IsoVector(m, n);\n const bVec = new _IsoVector(-n, m + n);\n const oaVec = _IsoVector.Zero();\n const abVec = _IsoVector.Zero();\n const obVec = _IsoVector.Zero();\n let verts = [];\n let idx;\n let idxR;\n let isoId;\n let isoIdR;\n const closestTo = [];\n const vDist = this.vertByDist;\n const matchIdx = (f, fr, isoId, isoIdR) => {\n idx = f + \"|\" + isoId;\n idxR = fr + \"|\" + isoIdR;\n if (!(idx in vecToidx || idxR in vecToidx)) {\n vecToidx[idx] = indexCount;\n vecToidx[idxR] = indexCount;\n indexCount++;\n }\n else if (idx in vecToidx && !(idxR in vecToidx)) {\n vecToidx[idxR] = vecToidx[idx];\n }\n else if (idxR in vecToidx && !(idx in vecToidx)) {\n vecToidx[idx] = vecToidx[idxR];\n }\n if (vDist[isoId][0] > 2) {\n closestTo[vecToidx[idx]] = [-vDist[isoId][0], vDist[isoId][1], vecToidx[idx]];\n }\n else {\n closestTo[vecToidx[idx]] = [verts[vDist[isoId][0]], vDist[isoId][1], vecToidx[idx]];\n }\n };\n this.IDATA.edgematch = [\n [1, \"B\"],\n [2, \"B\"],\n [3, \"B\"],\n [4, \"B\"],\n [0, \"B\"],\n [10, \"O\", 14, \"A\"],\n [11, \"O\", 10, \"A\"],\n [12, \"O\", 11, \"A\"],\n [13, \"O\", 12, \"A\"],\n [14, \"O\", 13, \"A\"],\n [0, \"O\"],\n [1, \"O\"],\n [2, \"O\"],\n [3, \"O\"],\n [4, \"O\"],\n [19, \"B\", 5, \"A\"],\n [15, \"B\", 6, \"A\"],\n [16, \"B\", 7, \"A\"],\n [17, \"B\", 8, \"A\"],\n [18, \"B\", 9, \"A\"],\n ];\n /***edges AB to OB***** rotation about B*/\n for (let f = 0; f < 20; f++) {\n //f current face\n verts = this.IDATA.face[f];\n O = verts[2];\n A = verts[1];\n B = verts[0];\n isoId = oVec.x + \"|\" + oVec.y;\n idx = f + \"|\" + isoId;\n if (!(idx in vecToidx)) {\n vecToidx[idx] = O;\n closestTo[O] = [verts[vDist[isoId][0]], vDist[isoId][1]];\n }\n isoId = aVec.x + \"|\" + aVec.y;\n idx = f + \"|\" + isoId;\n if (!(idx in vecToidx)) {\n vecToidx[idx] = A;\n closestTo[A] = [verts[vDist[isoId][0]], vDist[isoId][1]];\n }\n isoId = bVec.x + \"|\" + bVec.y;\n idx = f + \"|\" + isoId;\n if (!(idx in vecToidx)) {\n vecToidx[idx] = B;\n closestTo[B] = [verts[vDist[isoId][0]], vDist[isoId][1]];\n }\n //for edge vertices\n fr = this.IDATA.edgematch[f][0];\n rot = this.IDATA.edgematch[f][1];\n if (rot === \"B\") {\n for (let i = 1; i < g; i++) {\n abVec.x = m - i * (m1 + n1);\n abVec.y = n + i * m1;\n obVec.x = -i * n1;\n obVec.y = i * (m1 + n1);\n isoId = abVec.x + \"|\" + abVec.y;\n isoIdR = obVec.x + \"|\" + obVec.y;\n matchIdx(f, fr, isoId, isoIdR);\n }\n }\n if (rot === \"O\") {\n for (let i = 1; i < g; i++) {\n obVec.x = -i * n1;\n obVec.y = i * (m1 + n1);\n oaVec.x = i * m1;\n oaVec.y = i * n1;\n isoId = obVec.x + \"|\" + obVec.y;\n isoIdR = oaVec.x + \"|\" + oaVec.y;\n matchIdx(f, fr, isoId, isoIdR);\n }\n }\n fr = this.IDATA.edgematch[f][2];\n rot = this.IDATA.edgematch[f][3];\n if (rot && rot === \"A\") {\n for (let i = 1; i < g; i++) {\n oaVec.x = i * m1;\n oaVec.y = i * n1;\n abVec.x = m - (g - i) * (m1 + n1); //reversed for BA\n abVec.y = n + (g - i) * m1; //reversed for BA\n isoId = oaVec.x + \"|\" + oaVec.y;\n isoIdR = abVec.x + \"|\" + abVec.y;\n matchIdx(f, fr, isoId, isoIdR);\n }\n }\n for (let i = 0; i < this.vertices.length; i++) {\n isoId = this.vertices[i].x + \"|\" + this.vertices[i].y;\n idx = f + \"|\" + isoId;\n if (!(idx in vecToidx)) {\n vecToidx[idx] = indexCount++;\n if (vDist[isoId][0] > 2) {\n closestTo[vecToidx[idx]] = [-vDist[isoId][0], vDist[isoId][1], vecToidx[idx]];\n }\n else {\n closestTo[vecToidx[idx]] = [verts[vDist[isoId][0]], vDist[isoId][1], vecToidx[idx]];\n }\n }\n }\n }\n this.closestTo = closestTo;\n this.vecToidx = vecToidx;\n }\n calcCoeffs() {\n const m = this.m;\n const n = this.n;\n const thirdR3 = Math.sqrt(3) / 3;\n const LSQD = m * m + n * n + m * n;\n this.coau = (m + n) / LSQD;\n this.cobu = -n / LSQD;\n this.coav = (-thirdR3 * (m - n)) / LSQD;\n this.cobv = (thirdR3 * (2 * m + n)) / LSQD;\n }\n createInnerFacets() {\n const m = this.m;\n const n = this.n;\n for (let y = 0; y < n + m + 1; y++) {\n for (let x = this.min[y]; x < this.max[y] + 1; x++) {\n if (x < this.max[y] && x < this.max[y + 1] + 1) {\n this.innerFacets.push([\"|\" + x + \"|\" + y, \"|\" + x + \"|\" + (y + 1), \"|\" + (x + 1) + \"|\" + y]);\n }\n if (y > 0 && x < this.max[y - 1] && x + 1 < this.max[y] + 1) {\n this.innerFacets.push([\"|\" + x + \"|\" + y, \"|\" + (x + 1) + \"|\" + y, \"|\" + (x + 1) + \"|\" + (y - 1)]);\n }\n }\n }\n }\n edgeVecsABOB() {\n const m = this.m;\n const n = this.n;\n const B = new _IsoVector(-n, m + n);\n for (let y = 1; y < m + n; y++) {\n const point = new _IsoVector(this.min[y], y);\n const prev = new _IsoVector(this.min[y - 1], y - 1);\n const next = new _IsoVector(this.min[y + 1], y + 1);\n const pointR = point.clone();\n const prevR = prev.clone();\n const nextR = next.clone();\n pointR.rotate60About(B);\n prevR.rotate60About(B);\n nextR.rotate60About(B);\n const maxPoint = new _IsoVector(this.max[pointR.y], pointR.y);\n const maxPrev = new _IsoVector(this.max[pointR.y - 1], pointR.y - 1);\n const maxLeftPrev = new _IsoVector(this.max[pointR.y - 1] - 1, pointR.y - 1);\n if (pointR.x !== maxPoint.x || pointR.y !== maxPoint.y) {\n if (pointR.x !== maxPrev.x) {\n // type2\n //up\n this.vertexTypes.push([1, 0, 0]);\n this.isoVecsABOB.push([point, maxPrev, maxLeftPrev]);\n //down\n this.vertexTypes.push([1, 0, 0]);\n this.isoVecsABOB.push([point, maxLeftPrev, maxPoint]);\n }\n else if (pointR.y === nextR.y) {\n // type1\n //up\n this.vertexTypes.push([1, 1, 0]);\n this.isoVecsABOB.push([point, prev, maxPrev]);\n //down\n this.vertexTypes.push([1, 0, 1]);\n this.isoVecsABOB.push([point, maxPrev, next]);\n }\n else {\n // type 0\n //up\n this.vertexTypes.push([1, 1, 0]);\n this.isoVecsABOB.push([point, prev, maxPrev]);\n //down\n this.vertexTypes.push([1, 0, 0]);\n this.isoVecsABOB.push([point, maxPrev, maxPoint]);\n }\n }\n }\n }\n mapABOBtoOBOA() {\n const point = new _IsoVector(0, 0);\n for (let i = 0; i < this.isoVecsABOB.length; i++) {\n const temp = [];\n for (let j = 0; j < 3; j++) {\n point.x = this.isoVecsABOB[i][j].x;\n point.y = this.isoVecsABOB[i][j].y;\n if (this.vertexTypes[i][j] === 0) {\n point.rotateNeg120(this.m, this.n);\n }\n temp.push(point.clone());\n }\n this.isoVecsOBOA.push(temp);\n }\n }\n mapABOBtoBAOA() {\n const point = new _IsoVector(0, 0);\n for (let i = 0; i < this.isoVecsABOB.length; i++) {\n const temp = [];\n for (let j = 0; j < 3; j++) {\n point.x = this.isoVecsABOB[i][j].x;\n point.y = this.isoVecsABOB[i][j].y;\n if (this.vertexTypes[i][j] === 1) {\n point.rotate120(this.m, this.n);\n }\n temp.push(point.clone());\n }\n this.isoVecsBAOA.push(temp);\n }\n }\n // eslint-disable-next-line @typescript-eslint/naming-convention\n MapToFace(faceNb, geodesicData) {\n const F = this.IDATA.face[faceNb];\n const oidx = F[2];\n const aidx = F[1];\n const bidx = F[0];\n const O = Vector3.FromArray(this.IDATA.vertex[oidx]);\n const A = Vector3.FromArray(this.IDATA.vertex[aidx]);\n const B = Vector3.FromArray(this.IDATA.vertex[bidx]);\n const OA = A.subtract(O);\n const OB = B.subtract(O);\n const x = OA.scale(this.coau).add(OB.scale(this.cobu));\n const y = OA.scale(this.coav).add(OB.scale(this.cobv));\n const mapped = [];\n let idx;\n let tempVec = TmpVectors.Vector3[0];\n for (let i = 0; i < this.cartesian.length; i++) {\n tempVec = x.scale(this.cartesian[i].x).add(y.scale(this.cartesian[i].y)).add(O);\n mapped[i] = [tempVec.x, tempVec.y, tempVec.z];\n idx = faceNb + \"|\" + this.vertices[i].x + \"|\" + this.vertices[i].y;\n geodesicData.vertex[this.vecToidx[idx]] = [tempVec.x, tempVec.y, tempVec.z];\n }\n }\n //statics\n /**Creates a primary triangle\n * @internal\n */\n build(m, n) {\n const vertices = new Array();\n const O = _IsoVector.Zero();\n const A = new _IsoVector(m, n);\n const B = new _IsoVector(-n, m + n);\n vertices.push(O, A, B);\n //max internal isoceles triangle vertices\n for (let y = n; y < m + 1; y++) {\n for (let x = 0; x < m + 1 - y; x++) {\n vertices.push(new _IsoVector(x, y));\n }\n }\n //shared vertices along edges when needed\n if (n > 0) {\n const g = Scalar.HCF(m, n);\n const m1 = m / g;\n const n1 = n / g;\n for (let i = 1; i < g; i++) {\n vertices.push(new _IsoVector(i * m1, i * n1)); //OA\n vertices.push(new _IsoVector(-i * n1, i * (m1 + n1))); //OB\n vertices.push(new _IsoVector(m - i * (m1 + n1), n + i * m1)); // AB\n }\n //lower rows vertices and their rotations\n const ratio = m / n;\n for (let y = 1; y < n; y++) {\n for (let x = 0; x < y * ratio; x++) {\n vertices.push(new _IsoVector(x, y));\n vertices.push(new _IsoVector(x, y).rotate120(m, n));\n vertices.push(new _IsoVector(x, y).rotateNeg120(m, n));\n }\n }\n }\n //order vertices by x and then y\n vertices.sort((a, b) => {\n return a.x - b.x;\n });\n vertices.sort((a, b) => {\n return a.y - b.y;\n });\n const min = new Array(m + n + 1);\n const max = new Array(m + n + 1);\n for (let i = 0; i < min.length; i++) {\n min[i] = Infinity;\n max[i] = -Infinity;\n }\n let y = 0;\n let x = 0;\n const len = vertices.length;\n for (let i = 0; i < len; i++) {\n x = vertices[i].x;\n y = vertices[i].y;\n min[y] = Math.min(x, min[y]);\n max[y] = Math.max(x, max[y]);\n }\n //calculates the distance of a vertex from a given primary vertex\n const distFrom = (vert, primVert) => {\n const v = vert.clone();\n if (primVert === \"A\") {\n v.rotateNeg120(m, n);\n }\n if (primVert === \"B\") {\n v.rotate120(m, n);\n }\n if (v.x < 0) {\n return v.y;\n }\n return v.x + v.y;\n };\n const cartesian = [];\n const distFromO = [];\n const distFromA = [];\n const distFromB = [];\n const vertByDist = {};\n const vertData = [];\n let closest = -1;\n let dist = -1;\n for (let i = 0; i < len; i++) {\n cartesian[i] = vertices[i].toCartesianOrigin(new _IsoVector(0, 0), 0.5);\n distFromO[i] = distFrom(vertices[i], \"O\");\n distFromA[i] = distFrom(vertices[i], \"A\");\n distFromB[i] = distFrom(vertices[i], \"B\");\n if (distFromO[i] === distFromA[i] && distFromA[i] === distFromB[i]) {\n closest = 3;\n dist = distFromO[i];\n }\n else if (distFromO[i] === distFromA[i]) {\n closest = 4;\n dist = distFromO[i];\n }\n else if (distFromA[i] === distFromB[i]) {\n closest = 5;\n dist = distFromA[i];\n }\n else if (distFromB[i] === distFromO[i]) {\n closest = 6;\n dist = distFromO[i];\n }\n if (distFromO[i] < distFromA[i] && distFromO[i] < distFromB[i]) {\n closest = 2;\n dist = distFromO[i];\n }\n if (distFromA[i] < distFromO[i] && distFromA[i] < distFromB[i]) {\n closest = 1;\n dist = distFromA[i];\n }\n if (distFromB[i] < distFromA[i] && distFromB[i] < distFromO[i]) {\n closest = 0;\n dist = distFromB[i];\n }\n vertData.push([closest, dist, vertices[i].x, vertices[i].y]);\n }\n vertData.sort((a, b) => {\n return a[2] - b[2];\n });\n vertData.sort((a, b) => {\n return a[3] - b[3];\n });\n vertData.sort((a, b) => {\n return a[1] - b[1];\n });\n vertData.sort((a, b) => {\n return a[0] - b[0];\n });\n for (let v = 0; v < vertData.length; v++) {\n vertByDist[vertData[v][2] + \"|\" + vertData[v][3]] = [vertData[v][0], vertData[v][1], v];\n }\n this.m = m;\n this.n = n;\n this.vertices = vertices;\n this.vertByDist = vertByDist;\n this.cartesian = cartesian;\n this.min = min;\n this.max = max;\n return this;\n }\n }\n /** Builds Polyhedron Data\n * @internal\n */\n class PolyhedronData {\n constructor(name, category, vertex, face) {\n this.name = name;\n this.category = category;\n this.vertex = vertex;\n this.face = face;\n }\n }\n /**\n * This class Extends the PolyhedronData Class to provide measures for a Geodesic Polyhedron\n */\n class GeodesicData extends PolyhedronData {\n /**\n * @internal\n */\n innerToData(face, primTri) {\n for (let i = 0; i < primTri.innerFacets.length; i++) {\n this.face.push(primTri.innerFacets[i].map((el) => primTri.vecToidx[face + el]));\n }\n }\n /**\n * @internal\n */\n mapABOBtoDATA(faceNb, primTri) {\n const fr = primTri.IDATA.edgematch[faceNb][0];\n for (let i = 0; i < primTri.isoVecsABOB.length; i++) {\n const temp = [];\n for (let j = 0; j < 3; j++) {\n if (primTri.vertexTypes[i][j] === 0) {\n temp.push(faceNb + \"|\" + primTri.isoVecsABOB[i][j].x + \"|\" + primTri.isoVecsABOB[i][j].y);\n }\n else {\n temp.push(fr + \"|\" + primTri.isoVecsABOB[i][j].x + \"|\" + primTri.isoVecsABOB[i][j].y);\n }\n }\n this.face.push([primTri.vecToidx[temp[0]], primTri.vecToidx[temp[1]], primTri.vecToidx[temp[2]]]);\n }\n }\n /**\n * @internal\n */\n mapOBOAtoDATA(faceNb, primTri) {\n const fr = primTri.IDATA.edgematch[faceNb][0];\n for (let i = 0; i < primTri.isoVecsOBOA.length; i++) {\n const temp = [];\n for (let j = 0; j < 3; j++) {\n if (primTri.vertexTypes[i][j] === 1) {\n temp.push(faceNb + \"|\" + primTri.isoVecsOBOA[i][j].x + \"|\" + primTri.isoVecsOBOA[i][j].y);\n }\n else {\n temp.push(fr + \"|\" + primTri.isoVecsOBOA[i][j].x + \"|\" + primTri.isoVecsOBOA[i][j].y);\n }\n }\n this.face.push([primTri.vecToidx[temp[0]], primTri.vecToidx[temp[1]], primTri.vecToidx[temp[2]]]);\n }\n }\n /**\n * @internal\n */\n mapBAOAtoDATA(faceNb, primTri) {\n const fr = primTri.IDATA.edgematch[faceNb][2];\n for (let i = 0; i < primTri.isoVecsBAOA.length; i++) {\n const temp = [];\n for (let j = 0; j < 3; j++) {\n if (primTri.vertexTypes[i][j] === 1) {\n temp.push(faceNb + \"|\" + primTri.isoVecsBAOA[i][j].x + \"|\" + primTri.isoVecsBAOA[i][j].y);\n }\n else {\n temp.push(fr + \"|\" + primTri.isoVecsBAOA[i][j].x + \"|\" + primTri.isoVecsBAOA[i][j].y);\n }\n }\n this.face.push([primTri.vecToidx[temp[0]], primTri.vecToidx[temp[1]], primTri.vecToidx[temp[2]]]);\n }\n }\n /**\n * @internal\n */\n orderData(primTri) {\n const nearTo = [];\n for (let i = 0; i < 13; i++) {\n nearTo[i] = [];\n }\n const close = primTri.closestTo;\n for (let i = 0; i < close.length; i++) {\n if (close[i][0] > -1) {\n if (close[i][1] > 0) {\n nearTo[close[i][0]].push([i, close[i][1]]);\n }\n }\n else {\n nearTo[12].push([i, close[i][0]]);\n }\n }\n const near = [];\n for (let i = 0; i < 12; i++) {\n near[i] = i;\n }\n let nearIndex = 12;\n for (let i = 0; i < 12; i++) {\n nearTo[i].sort((a, b) => {\n return a[1] - b[1];\n });\n for (let j = 0; j < nearTo[i].length; j++) {\n near[nearTo[i][j][0]] = nearIndex++;\n }\n }\n for (let j = 0; j < nearTo[12].length; j++) {\n near[nearTo[12][j][0]] = nearIndex++;\n }\n for (let i = 0; i < this.vertex.length; i++) {\n this.vertex[i].push(near[i]);\n }\n this.vertex.sort((a, b) => {\n return a[3] - b[3];\n });\n for (let i = 0; i < this.vertex.length; i++) {\n this.vertex[i].pop();\n }\n for (let i = 0; i < this.face.length; i++) {\n for (let j = 0; j < this.face[i].length; j++) {\n this.face[i][j] = near[this.face[i][j]];\n }\n }\n this.sharedNodes = nearTo[12].length;\n this.poleNodes = this.vertex.length - this.sharedNodes;\n }\n /**\n * @internal\n */\n setOrder(m, faces) {\n const adjVerts = [];\n const dualFaces = [];\n let face = faces.pop();\n dualFaces.push(face);\n let index = this.face[face].indexOf(m);\n index = (index + 2) % 3;\n let v = this.face[face][index];\n adjVerts.push(v);\n let f = 0;\n while (faces.length > 0) {\n face = faces[f];\n if (this.face[face].indexOf(v) > -1) {\n // v is a vertex of face f\n index = (this.face[face].indexOf(v) + 1) % 3;\n v = this.face[face][index];\n adjVerts.push(v);\n dualFaces.push(face);\n faces.splice(f, 1);\n f = 0;\n }\n else {\n f++;\n }\n }\n this.adjacentFaces.push(adjVerts);\n return dualFaces;\n }\n /**\n * @internal\n */\n toGoldbergPolyhedronData() {\n const goldbergPolyhedronData = new PolyhedronData(\"GeoDual\", \"Goldberg\", [], []);\n goldbergPolyhedronData.name = \"GD dual\";\n const verticesNb = this.vertex.length;\n const map = new Array(verticesNb);\n for (let v = 0; v < verticesNb; v++) {\n map[v] = [];\n }\n for (let f = 0; f < this.face.length; f++) {\n for (let i = 0; i < 3; i++) {\n map[this.face[f][i]].push(f);\n }\n }\n let cx = 0;\n let cy = 0;\n let cz = 0;\n let face = [];\n let vertex = [];\n this.adjacentFaces = [];\n for (let m = 0; m < map.length; m++) {\n goldbergPolyhedronData.face[m] = this.setOrder(m, map[m].concat([]));\n map[m].forEach((el) => {\n cx = 0;\n cy = 0;\n cz = 0;\n face = this.face[el];\n for (let i = 0; i < 3; i++) {\n vertex = this.vertex[face[i]];\n cx += vertex[0];\n cy += vertex[1];\n cz += vertex[2];\n }\n goldbergPolyhedronData.vertex[el] = [cx / 3, cy / 3, cz / 3];\n });\n }\n return goldbergPolyhedronData;\n }\n //statics\n /**Builds the data for a Geodesic Polyhedron from a primary triangle\n * @param primTri the primary triangle\n * @internal\n */\n static BuildGeodesicData(primTri) {\n const geodesicData = new GeodesicData(\"Geodesic-m-n\", \"Geodesic\", [\n [0, PHI, -1],\n [-PHI, 1, 0],\n [-1, 0, -PHI],\n [1, 0, -PHI],\n [PHI, 1, 0],\n [0, PHI, 1],\n [-1, 0, PHI],\n [-PHI, -1, 0],\n [0, -PHI, -1],\n [PHI, -1, 0],\n [1, 0, PHI],\n [0, -PHI, 1],\n ], []);\n primTri.setIndices();\n primTri.calcCoeffs();\n primTri.createInnerFacets();\n primTri.edgeVecsABOB();\n primTri.mapABOBtoOBOA();\n primTri.mapABOBtoBAOA();\n for (let f = 0; f < primTri.IDATA.face.length; f++) {\n primTri.MapToFace(f, geodesicData);\n geodesicData.innerToData(f, primTri);\n if (primTri.IDATA.edgematch[f][1] === \"B\") {\n geodesicData.mapABOBtoDATA(f, primTri);\n }\n if (primTri.IDATA.edgematch[f][1] === \"O\") {\n geodesicData.mapOBOAtoDATA(f, primTri);\n }\n if (primTri.IDATA.edgematch[f][3] === \"A\") {\n geodesicData.mapBAOAtoDATA(f, primTri);\n }\n }\n geodesicData.orderData(primTri);\n const radius = 1;\n geodesicData.vertex = geodesicData.vertex.map(function (el) {\n const a = el[0];\n const b = el[1];\n const c = el[2];\n const d = Math.sqrt(a * a + b * b + c * c);\n el[0] *= radius / d;\n el[1] *= radius / d;\n el[2] *= radius / d;\n return el;\n });\n return geodesicData;\n }\n }\n\n /**\n * Creates the Mesh for a Geodesic Polyhedron\n * @see https://en.wikipedia.org/wiki/Geodesic_polyhedron\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/polyhedra/geodesic_poly\n * @param name defines the name of the mesh\n * @param options an object used to set the following optional parameters for the polyhedron, required but can be empty\n * * m number of horizontal steps along an isogrid\n * * n number of angled steps along an isogrid\n * * size the size of the Geodesic, optional default 1\n * * sizeX allows stretching in the x direction, optional, default size\n * * sizeY allows stretching in the y direction, optional, default size\n * * sizeZ allows stretching in the z direction, optional, default size\n * * faceUV an array of Vector4 elements used to set different images to the top, rings and bottom respectively\n * * faceColors an array of Color3 elements used to set different colors to the top, rings and bottom respectively\n * * flat when true creates a flat shaded mesh, optional, default true\n * * subdivisions increasing the subdivisions increases the number of faces, optional, default 4\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * @param options.n\n * @param options.size\n * @param options.sizeX\n * @param options.sizeY\n * @param options.sizeZ\n * @param options.faceUV\n * @param options.faceColors\n * @param options.flat\n * @param options.updatable\n * @param options.sideOrientation\n * @param options.frontUVs\n * @param options.backUVs\n * @param options.m\n * @param scene defines the hosting scene\n * @returns Geodesic mesh\n */\n function CreateGeodesic(name, options, scene = null) {\n let m = options.m || 1;\n if (m !== Math.floor(m)) {\n Logger.Warn(\"m not an integer only floor(m) used\");\n }\n let n = options.n || 0;\n if (n !== Math.floor(n)) {\n Logger.Warn(\"n not an integer only floor(n) used\");\n }\n if (n > m) {\n const temp = n;\n n = m;\n m = temp;\n Logger.Warn(\"n > m therefore m and n swapped\");\n }\n const primTri = new _PrimaryIsoTriangle();\n primTri.build(m, n);\n const geodesicData = GeodesicData.BuildGeodesicData(primTri);\n const geoOptions = {\n custom: geodesicData,\n size: options.size,\n sizeX: options.sizeX,\n sizeY: options.sizeY,\n sizeZ: options.sizeZ,\n faceUV: options.faceUV,\n faceColors: options.faceColors,\n flat: options.flat,\n updatable: options.updatable,\n sideOrientation: options.sideOrientation,\n frontUVs: options.frontUVs,\n backUVs: options.backUVs,\n };\n const geodesic = CreatePolyhedron(name, geoOptions, scene);\n return geodesic;\n }\n\n Mesh._GoldbergMeshParser = (parsedMesh, scene) => {\n return GoldbergMesh.Parse(parsedMesh, scene);\n };\n /**\n * Mesh for a Goldberg Polyhedron which is made from 12 pentagonal and the rest hexagonal faces\n * @see https://en.wikipedia.org/wiki/Goldberg_polyhedron\n */\n class GoldbergMesh extends Mesh {\n constructor() {\n super(...arguments);\n /**\n * Defines the specific Goldberg data used in this mesh construction.\n */\n this.goldbergData = {\n faceColors: [],\n faceCenters: [],\n faceZaxis: [],\n faceXaxis: [],\n faceYaxis: [],\n nbSharedFaces: 0,\n nbUnsharedFaces: 0,\n nbFaces: 0,\n nbFacesAtPole: 0,\n adjacentFaces: [],\n };\n }\n /**\n * Gets the related Goldberg face from pole infos\n * @param poleOrShared Defines the pole index or the shared face index if the fromPole parameter is passed in\n * @param fromPole Defines an optional pole index to find the related info from\n * @returns the goldberg face number\n */\n relatedGoldbergFace(poleOrShared, fromPole) {\n if (fromPole === void 0) {\n if (poleOrShared > this.goldbergData.nbUnsharedFaces - 1) {\n Logger.Warn(\"Maximum number of unshared faces used\");\n poleOrShared = this.goldbergData.nbUnsharedFaces - 1;\n }\n return this.goldbergData.nbUnsharedFaces + poleOrShared;\n }\n if (poleOrShared > 11) {\n Logger.Warn(\"Last pole used\");\n poleOrShared = 11;\n }\n if (fromPole > this.goldbergData.nbFacesAtPole - 1) {\n Logger.Warn(\"Maximum number of faces at a pole used\");\n fromPole = this.goldbergData.nbFacesAtPole - 1;\n }\n return 12 + poleOrShared * this.goldbergData.nbFacesAtPole + fromPole;\n }\n _changeGoldbergFaceColors(colorRange) {\n for (let i = 0; i < colorRange.length; i++) {\n const min = colorRange[i][0];\n const max = colorRange[i][1];\n const col = colorRange[i][2];\n for (let f = min; f < max + 1; f++) {\n this.goldbergData.faceColors[f] = col;\n }\n }\n const newCols = [];\n for (let f = 0; f < 12; f++) {\n for (let i = 0; i < 5; i++) {\n newCols.push(this.goldbergData.faceColors[f].r, this.goldbergData.faceColors[f].g, this.goldbergData.faceColors[f].b, this.goldbergData.faceColors[f].a);\n }\n }\n for (let f = 12; f < this.goldbergData.faceColors.length; f++) {\n for (let i = 0; i < 6; i++) {\n newCols.push(this.goldbergData.faceColors[f].r, this.goldbergData.faceColors[f].g, this.goldbergData.faceColors[f].b, this.goldbergData.faceColors[f].a);\n }\n }\n return newCols;\n }\n /**\n * Set new goldberg face colors\n * @param colorRange the new color to apply to the mesh\n */\n setGoldbergFaceColors(colorRange) {\n const newCols = this._changeGoldbergFaceColors(colorRange);\n this.setVerticesData(VertexBuffer.ColorKind, newCols);\n }\n /**\n * Updates new goldberg face colors\n * @param colorRange the new color to apply to the mesh\n */\n updateGoldbergFaceColors(colorRange) {\n const newCols = this._changeGoldbergFaceColors(colorRange);\n this.updateVerticesData(VertexBuffer.ColorKind, newCols);\n }\n _changeGoldbergFaceUVs(uvRange) {\n const uvs = this.getVerticesData(VertexBuffer.UVKind);\n for (let i = 0; i < uvRange.length; i++) {\n const min = uvRange[i][0];\n const max = uvRange[i][1];\n const center = uvRange[i][2];\n const radius = uvRange[i][3];\n const angle = uvRange[i][4];\n const points5 = [];\n const points6 = [];\n let u;\n let v;\n for (let p = 0; p < 5; p++) {\n u = center.x + radius * Math.cos(angle + (p * Math.PI) / 2.5);\n v = center.y + radius * Math.sin(angle + (p * Math.PI) / 2.5);\n if (u < 0) {\n u = 0;\n }\n if (u > 1) {\n u = 1;\n }\n points5.push(u, v);\n }\n for (let p = 0; p < 6; p++) {\n u = center.x + radius * Math.cos(angle + (p * Math.PI) / 3);\n v = center.y + radius * Math.sin(angle + (p * Math.PI) / 3);\n if (u < 0) {\n u = 0;\n }\n if (u > 1) {\n u = 1;\n }\n points6.push(u, v);\n }\n for (let f = min; f < Math.min(12, max + 1); f++) {\n for (let p = 0; p < 5; p++) {\n uvs[10 * f + 2 * p] = points5[2 * p];\n uvs[10 * f + 2 * p + 1] = points5[2 * p + 1];\n }\n }\n for (let f = Math.max(12, min); f < max + 1; f++) {\n for (let p = 0; p < 6; p++) {\n //120 + 12 * (f - 12) = 12 * f - 24\n uvs[12 * f - 24 + 2 * p] = points6[2 * p];\n uvs[12 * f - 23 + 2 * p] = points6[2 * p + 1];\n }\n }\n }\n return uvs;\n }\n /**\n * set new goldberg face UVs\n * @param uvRange the new UVs to apply to the mesh\n */\n setGoldbergFaceUVs(uvRange) {\n const newUVs = this._changeGoldbergFaceUVs(uvRange);\n this.setVerticesData(VertexBuffer.UVKind, newUVs);\n }\n /**\n * Updates new goldberg face UVs\n * @param uvRange the new UVs to apply to the mesh\n */\n updateGoldbergFaceUVs(uvRange) {\n const newUVs = this._changeGoldbergFaceUVs(uvRange);\n this.updateVerticesData(VertexBuffer.UVKind, newUVs);\n }\n /**\n * Places a mesh on a particular face of the goldberg polygon\n * @param mesh Defines the mesh to position\n * @param face Defines the face to position onto\n * @param position Defines the position relative to the face we are positioning the mesh onto\n */\n placeOnGoldbergFaceAt(mesh, face, position) {\n const orientation = Vector3.RotationFromAxis(this.goldbergData.faceXaxis[face], this.goldbergData.faceYaxis[face], this.goldbergData.faceZaxis[face]);\n mesh.rotation = orientation;\n mesh.position = this.goldbergData.faceCenters[face]\n .add(this.goldbergData.faceXaxis[face].scale(position.x))\n .add(this.goldbergData.faceYaxis[face].scale(position.y))\n .add(this.goldbergData.faceZaxis[face].scale(position.z));\n }\n /**\n * Serialize current mesh\n * @param serializationObject defines the object which will receive the serialization data\n */\n serialize(serializationObject) {\n super.serialize(serializationObject);\n serializationObject.type = \"GoldbergMesh\";\n const goldbergData = {};\n goldbergData.adjacentFaces = this.goldbergData.adjacentFaces;\n goldbergData.nbSharedFaces = this.goldbergData.nbSharedFaces;\n goldbergData.nbUnsharedFaces = this.goldbergData.nbUnsharedFaces;\n goldbergData.nbFaces = this.goldbergData.nbFaces;\n goldbergData.nbFacesAtPole = this.goldbergData.nbFacesAtPole;\n if (this.goldbergData.faceColors) {\n goldbergData.faceColors = [];\n for (const color of this.goldbergData.faceColors) {\n goldbergData.faceColors.push(color.asArray());\n }\n }\n if (this.goldbergData.faceCenters) {\n goldbergData.faceCenters = [];\n for (const vector of this.goldbergData.faceCenters) {\n goldbergData.faceCenters.push(vector.asArray());\n }\n }\n if (this.goldbergData.faceZaxis) {\n goldbergData.faceZaxis = [];\n for (const vector of this.goldbergData.faceZaxis) {\n goldbergData.faceZaxis.push(vector.asArray());\n }\n }\n if (this.goldbergData.faceYaxis) {\n goldbergData.faceYaxis = [];\n for (const vector of this.goldbergData.faceYaxis) {\n goldbergData.faceYaxis.push(vector.asArray());\n }\n }\n if (this.goldbergData.faceXaxis) {\n goldbergData.faceXaxis = [];\n for (const vector of this.goldbergData.faceXaxis) {\n goldbergData.faceXaxis.push(vector.asArray());\n }\n }\n serializationObject.goldbergData = goldbergData;\n }\n /**\n * Parses a serialized goldberg mesh\n * @param parsedMesh the serialized mesh\n * @param scene the scene to create the goldberg mesh in\n * @returns the created goldberg mesh\n */\n static Parse(parsedMesh, scene) {\n const goldbergData = parsedMesh.goldbergData;\n goldbergData.faceColors = goldbergData.faceColors.map((el) => Color4.FromArray(el));\n goldbergData.faceCenters = goldbergData.faceCenters.map((el) => Vector3.FromArray(el));\n goldbergData.faceZaxis = goldbergData.faceZaxis.map((el) => Vector3.FromArray(el));\n goldbergData.faceXaxis = goldbergData.faceXaxis.map((el) => Vector3.FromArray(el));\n goldbergData.faceYaxis = goldbergData.faceYaxis.map((el) => Vector3.FromArray(el));\n const goldberg = new GoldbergMesh(parsedMesh.name, scene);\n goldberg.goldbergData = goldbergData;\n return goldberg;\n }\n }\n\n /**\n * Creates the Mesh for a Goldberg Polyhedron\n * @param options an object used to set the following optional parameters for the polyhedron, required but can be empty\n * @param goldbergData polyhedronData defining the Goldberg polyhedron\n * @returns GoldbergSphere mesh\n */\n function CreateGoldbergVertexData(options, goldbergData) {\n const size = options.size;\n const sizeX = options.sizeX || size || 1;\n const sizeY = options.sizeY || size || 1;\n const sizeZ = options.sizeZ || size || 1;\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\n const positions = new Array();\n const indices = new Array();\n const normals = new Array();\n const uvs = new Array();\n let minX = Infinity;\n let maxX = -Infinity;\n let minY = Infinity;\n let maxY = -Infinity;\n for (let v = 0; v < goldbergData.vertex.length; v++) {\n minX = Math.min(minX, goldbergData.vertex[v][0] * sizeX);\n maxX = Math.max(maxX, goldbergData.vertex[v][0] * sizeX);\n minY = Math.min(minY, goldbergData.vertex[v][1] * sizeY);\n maxY = Math.max(maxY, goldbergData.vertex[v][1] * sizeY);\n }\n let index = 0;\n for (let f = 0; f < goldbergData.face.length; f++) {\n const verts = goldbergData.face[f];\n const a = Vector3.FromArray(goldbergData.vertex[verts[0]]);\n const b = Vector3.FromArray(goldbergData.vertex[verts[2]]);\n const c = Vector3.FromArray(goldbergData.vertex[verts[1]]);\n const ba = b.subtract(a);\n const ca = c.subtract(a);\n const norm = Vector3.Cross(ca, ba).normalize();\n for (let v = 0; v < verts.length; v++) {\n normals.push(norm.x, norm.y, norm.z);\n const pdata = goldbergData.vertex[verts[v]];\n positions.push(pdata[0] * sizeX, pdata[1] * sizeY, pdata[2] * sizeZ);\n const vCoord = (pdata[1] * sizeY - minY) / (maxY - minY);\n uvs.push((pdata[0] * sizeX - minX) / (maxX - minX), CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - vCoord : vCoord);\n }\n for (let v = 0; v < verts.length - 2; v++) {\n indices.push(index, index + v + 2, index + v + 1);\n }\n index += verts.length;\n }\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs);\n const vertexData = new VertexData();\n vertexData.positions = positions;\n vertexData.indices = indices;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n return vertexData;\n }\n /**\n * Creates the Mesh for a Goldberg Polyhedron which is made from 12 pentagonal and the rest hexagonal faces\n * @see https://en.wikipedia.org/wiki/Goldberg_polyhedron\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/polyhedra/goldberg_poly\n * @param name defines the name of the mesh\n * @param options an object used to set the following optional parameters for the polyhedron, required but can be empty\n * @param scene defines the hosting scene\n * @returns Goldberg mesh\n */\n function CreateGoldberg(name, options, scene = null) {\n const size = options.size;\n const sizeX = options.sizeX || size || 1;\n const sizeY = options.sizeY || size || 1;\n const sizeZ = options.sizeZ || size || 1;\n let m = options.m || 1;\n if (m !== Math.floor(m)) {\n Logger.Warn(\"m not an integer only floor(m) used\");\n }\n let n = options.n || 0;\n if (n !== Math.floor(n)) {\n Logger.Warn(\"n not an integer only floor(n) used\");\n }\n if (n > m) {\n const temp = n;\n n = m;\n m = temp;\n Logger.Warn(\"n > m therefore m and n swapped\");\n }\n const primTri = new _PrimaryIsoTriangle();\n primTri.build(m, n);\n const geodesicData = GeodesicData.BuildGeodesicData(primTri);\n const goldbergData = geodesicData.toGoldbergPolyhedronData();\n const goldberg = new GoldbergMesh(name, scene);\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n goldberg._originalBuilderSideOrientation = options.sideOrientation;\n const vertexData = CreateGoldbergVertexData(options, goldbergData);\n vertexData.applyToMesh(goldberg, options.updatable);\n goldberg.goldbergData.nbSharedFaces = geodesicData.sharedNodes;\n goldberg.goldbergData.nbUnsharedFaces = geodesicData.poleNodes;\n goldberg.goldbergData.adjacentFaces = geodesicData.adjacentFaces;\n goldberg.goldbergData.nbFaces = goldberg.goldbergData.nbSharedFaces + goldberg.goldbergData.nbUnsharedFaces;\n goldberg.goldbergData.nbFacesAtPole = (goldberg.goldbergData.nbUnsharedFaces - 12) / 12;\n for (let f = 0; f < geodesicData.vertex.length; f++) {\n goldberg.goldbergData.faceCenters.push(Vector3.FromArray(geodesicData.vertex[f]));\n goldberg.goldbergData.faceCenters[f].x *= sizeX;\n goldberg.goldbergData.faceCenters[f].y *= sizeY;\n goldberg.goldbergData.faceCenters[f].z *= sizeZ;\n goldberg.goldbergData.faceColors.push(new Color4(1, 1, 1, 1));\n }\n for (let f = 0; f < goldbergData.face.length; f++) {\n const verts = goldbergData.face[f];\n const a = Vector3.FromArray(goldbergData.vertex[verts[0]]);\n const b = Vector3.FromArray(goldbergData.vertex[verts[2]]);\n const c = Vector3.FromArray(goldbergData.vertex[verts[1]]);\n const ba = b.subtract(a);\n const ca = c.subtract(a);\n const norm = Vector3.Cross(ca, ba).normalize();\n const z = Vector3.Cross(ca, norm).normalize();\n goldberg.goldbergData.faceXaxis.push(ca.normalize());\n goldberg.goldbergData.faceYaxis.push(norm);\n goldberg.goldbergData.faceZaxis.push(z);\n }\n return goldberg;\n }\n\n // Shape functions\n class ShapePath {\n /** Create the ShapePath used to support glyphs */\n constructor(resolution) {\n this._paths = [];\n this._tempPaths = [];\n this._holes = [];\n this._resolution = resolution;\n }\n /** Move the virtual cursor to a coordinate */\n moveTo(x, y) {\n this._currentPath = new Path2(x, y);\n this._tempPaths.push(this._currentPath);\n }\n /** Draw a line from the virtual cursor to a given coordinate */\n lineTo(x, y) {\n this._currentPath.addLineTo(x, y);\n }\n /** Create a quadratic curve from the virtual cursor to a given coordinate */\n quadraticCurveTo(cpx, cpy, x, y) {\n this._currentPath.addQuadraticCurveTo(cpx, cpy, x, y, this._resolution);\n }\n /** Create a bezier curve from the virtual cursor to a given coordinate */\n bezierCurveTo(cpx1, cpy1, cpx2, cpy2, x, y) {\n this._currentPath.addBezierCurveTo(cpx1, cpy1, cpx2, cpy2, x, y, this._resolution);\n }\n /** Extract holes based on CW / CCW */\n extractHoles() {\n for (const path of this._tempPaths) {\n if (path.area() > 0) {\n this._holes.push(path);\n }\n else {\n this._paths.push(path);\n }\n }\n if (!this._paths.length && this._holes.length) {\n const temp = this._holes;\n this._holes = this._paths;\n this._paths = temp;\n }\n this._tempPaths.length = 0;\n }\n /** Gets the list of paths */\n get paths() {\n return this._paths;\n }\n /** Gets the list of holes */\n get holes() {\n return this._holes;\n }\n }\n // Utility functions\n function CreateShapePath(char, scale, offsetX, offsetY, resolution, fontData) {\n const glyph = fontData.glyphs[char] || fontData.glyphs[\"?\"];\n if (!glyph) {\n // return if there is no glyph data\n return null;\n }\n const shapePath = new ShapePath(resolution);\n if (glyph.o) {\n const outline = glyph.o.split(\" \");\n for (let i = 0, l = outline.length; i < l;) {\n const action = outline[i++];\n switch (action) {\n case \"m\": {\n // moveTo\n const x = parseInt(outline[i++]) * scale + offsetX;\n const y = parseInt(outline[i++]) * scale + offsetY;\n shapePath.moveTo(x, y);\n break;\n }\n case \"l\": {\n // lineTo\n const x = parseInt(outline[i++]) * scale + offsetX;\n const y = parseInt(outline[i++]) * scale + offsetY;\n shapePath.lineTo(x, y);\n break;\n }\n case \"q\": {\n // quadraticCurveTo\n const cpx = parseInt(outline[i++]) * scale + offsetX;\n const cpy = parseInt(outline[i++]) * scale + offsetY;\n const cpx1 = parseInt(outline[i++]) * scale + offsetX;\n const cpy1 = parseInt(outline[i++]) * scale + offsetY;\n shapePath.quadraticCurveTo(cpx1, cpy1, cpx, cpy);\n break;\n }\n case \"b\": {\n // bezierCurveTo\n const cpx = parseInt(outline[i++]) * scale + offsetX;\n const cpy = parseInt(outline[i++]) * scale + offsetY;\n const cpx1 = parseInt(outline[i++]) * scale + offsetX;\n const cpy1 = parseInt(outline[i++]) * scale + offsetY;\n const cpx2 = parseInt(outline[i++]) * scale + offsetX;\n const cpy2 = parseInt(outline[i++]) * scale + offsetY;\n shapePath.bezierCurveTo(cpx1, cpy1, cpx2, cpy2, cpx, cpy);\n break;\n }\n }\n }\n }\n // Extract holes (based on clockwise data)\n shapePath.extractHoles();\n return { offsetX: glyph.ha * scale, shapePath: shapePath };\n }\n /**\n * Creates shape paths from a text and font\n * @param text the text\n * @param size size of the font\n * @param resolution resolution of the font\n * @param fontData defines the font data (can be generated with http://gero3.github.io/facetype.js/)\n * @returns array of ShapePath objects\n */\n function CreateTextShapePaths(text, size, resolution, fontData) {\n const chars = Array.from(text);\n const scale = size / fontData.resolution;\n const line_height = (fontData.boundingBox.yMax - fontData.boundingBox.yMin + fontData.underlineThickness) * scale;\n const shapePaths = [];\n let offsetX = 0, offsetY = 0;\n for (let i = 0; i < chars.length; i++) {\n const char = chars[i];\n if (char === \"\\n\") {\n offsetX = 0;\n offsetY -= line_height;\n }\n else {\n const ret = CreateShapePath(char, scale, offsetX, offsetY, resolution, fontData);\n if (ret) {\n offsetX += ret.offsetX;\n shapePaths.push(ret.shapePath);\n }\n }\n }\n return shapePaths;\n }\n /**\n * Create a text mesh\n * @param name defines the name of the mesh\n * @param text defines the text to use to build the mesh\n * @param fontData defines the font data (can be generated with http://gero3.github.io/facetype.js/)\n * @param options defines options used to create the mesh\n * @param scene defines the hosting scene\n * @param earcutInjection can be used to inject your own earcut reference\n * @returns a new Mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set/text\n */\n function CreateText(name, text, fontData, options = {\n size: 50,\n resolution: 8,\n depth: 1.0,\n }, scene = null, earcutInjection = earcut) {\n // First we need to generate the paths\n const shapePaths = CreateTextShapePaths(text, options.size || 50, options.resolution || 8, fontData);\n // And extrude them\n const meshes = [];\n for (const shapePath of shapePaths) {\n if (!shapePath.paths.length) {\n continue;\n }\n const holes = shapePath.holes.slice(); // Copy it as we will update the copy\n for (const path of shapePath.paths) {\n const holeVectors = [];\n const shapeVectors = [];\n const points = path.getPoints();\n for (const point of points) {\n shapeVectors.push(new Vector3(point.x, 0, point.y)); // ExtrudePolygon expects data on the xz plane\n }\n // Holes\n const localHolesCopy = holes.slice();\n for (const hole of localHolesCopy) {\n const points = hole.getPoints();\n let found = false;\n for (const point of points) {\n if (path.isPointInside(point)) {\n found = true;\n break;\n }\n }\n if (!found) {\n continue;\n }\n const holePoints = [];\n for (const point of points) {\n holePoints.push(new Vector3(point.x, 0, point.y)); // ExtrudePolygon expects data on the xz plane\n }\n holeVectors.push(holePoints);\n // Remove the hole as it was already used\n holes.splice(holes.indexOf(hole), 1);\n }\n // There is at least a hole but it was unaffected\n if (!holeVectors.length && holes.length) {\n for (const hole of holes) {\n const points = hole.getPoints();\n const holePoints = [];\n for (const point of points) {\n holePoints.push(new Vector3(point.x, 0, point.y)); // ExtrudePolygon expects data on the xz plane\n }\n holeVectors.push(holePoints);\n }\n }\n // Extrusion!\n const mesh = ExtrudePolygon(name, {\n shape: shapeVectors,\n holes: holeVectors.length ? holeVectors : undefined,\n depth: options.depth || 1.0,\n sideOrientation: Mesh._GetDefaultSideOrientation(options.sideOrientation || Mesh.DOUBLESIDE),\n }, scene, earcutInjection);\n meshes.push(mesh);\n }\n }\n // Then we can merge everyone into one single mesh\n const newMesh = Mesh.MergeMeshes(meshes, true, true);\n if (newMesh) {\n // Move pivot to center\n const bbox = newMesh === null || newMesh === void 0 ? void 0 : newMesh.getBoundingInfo();\n newMesh.position.x = -(bbox === null || bbox === void 0 ? void 0 : bbox.boundingBox.extendSizeWorld._x);\n newMesh.position.y = -(bbox === null || bbox === void 0 ? void 0 : bbox.boundingBox.extendSizeWorld._y);\n newMesh.position.z = -(bbox === null || bbox === void 0 ? void 0 : bbox.boundingBox.extendSizeWorld._z);\n newMesh.name = name;\n newMesh.rotation.x = -Math.PI / 2;\n newMesh.bakeCurrentTransformIntoVertices();\n }\n return newMesh;\n }\n\n /* eslint-disable @typescript-eslint/naming-convention */\n /**\n * Class containing static functions to help procedurally build meshes\n */\n const MeshBuilder = {\n CreateBox,\n CreateTiledBox,\n CreateSphere,\n CreateDisc,\n CreateIcoSphere,\n CreateRibbon,\n CreateCylinder,\n CreateTorus,\n CreateTorusKnot,\n CreateLineSystem,\n CreateLines,\n CreateDashedLines,\n ExtrudeShape,\n ExtrudeShapeCustom,\n CreateLathe,\n CreateTiledPlane,\n CreatePlane,\n CreateGround,\n CreateTiledGround,\n CreateGroundFromHeightMap,\n CreatePolygon,\n ExtrudePolygon,\n CreateTube,\n CreatePolyhedron,\n CreateGeodesic,\n CreateGoldberg,\n CreateDecal,\n CreateCapsule,\n CreateText,\n };\n\n class GuiHelper {\n constructor(scene, highlighter) {\n this.scene = scene;\n this.highlighter = highlighter;\n this.textLabelMap = new Map();\n this.gridSettings = {};\n const cameraPosition = new Vector3();\n scene.registerBeforeRender(() => {\n if (scene.activeCamera && this.textLabelMap.size > 0) {\n this.textLabelMap.forEach(mesh => {\n const camera = scene.activeCamera;\n cameraPosition.copyFrom(getCameraPositionWithOffset(camera));\n mesh.lookAt(cameraPosition);\n if (camera.mode !== Camera.ORTHOGRAPHIC_CAMERA) {\n const size = cameraPosition.subtractInPlace(mesh.position).length() * 0.05;\n mesh.scaling = mesh.scaling.copyFromFloats(size, size, size);\n }\n });\n }\n });\n }\n enableGrid(sceneNode, skipOrigin = false) {\n if (this.gridSettings.gridMesh) {\n return;\n }\n const bgColor = sceneNode === null || sceneNode === void 0 ? void 0 : sceneNode.backgroundColor;\n let bgColor3;\n if (!bgColor) {\n bgColor3 = new Color3(this.scene.clearColor.r, this.scene.clearColor.g, this.scene.clearColor.b);\n }\n else {\n bgColor3 = bgColor.a === 0 ? Color3.Black() : new Color3(bgColor.r, bgColor.g, bgColor.b);\n }\n const gridStepSize = sceneNode.gridStepSize || 1;\n const gridSize = sceneNode.gridSize || 10;\n const gridNormal = sceneNode.gridNormal;\n const gridMajorInterval = sceneNode.gridMajorInterval || 0;\n const strongLineColor = Color4.FromColor3(bgColor3.toLuminance() > 0.5 ? Color3.Black() : Color3.White());\n const lightLineColor = strongLineColor.scale(0.4).add(Color4.FromColor3(bgColor3.scale(0.6)));\n const points = this.calculateGridPoints(gridStepSize, gridSize, gridNormal);\n const pointColors = [];\n const linesPerSide = points.length / 2;\n for (let s = 0; s < 2; s++) {\n for (let i = 0; i < linesPerSide; i++) {\n if (gridMajorInterval === 0) {\n pointColors.push([lightLineColor, lightLineColor]);\n }\n else if (Math.ceil(i / 2) % gridMajorInterval === 0) {\n pointColors.push([strongLineColor, strongLineColor]);\n }\n else {\n pointColors.push([lightLineColor, lightLineColor]);\n }\n }\n }\n if (skipOrigin) {\n points.splice(linesPerSide, 1);\n points.splice(0, 1);\n pointColors.splice(linesPerSide, 1);\n pointColors.splice(0, 1);\n }\n const gridMesh = MeshBuilder.CreateLineSystem('grid', {\n lines: points,\n colors: pointColors,\n updatable: false,\n }, this.scene);\n // gridMesh.color = lineColor.scale(0.4).add(bgColor3.scale(0.6));\n gridMesh.isPickable = false;\n this.gridSettings.gridMesh = gridMesh;\n }\n disableGrid() {\n if (this.gridSettings.gridMesh) {\n this.gridSettings.gridMesh.dispose();\n delete this.gridSettings.gridMesh;\n }\n }\n enableGridAxes(sceneNode) {\n if (this.gridSettings.gridAxesMesh) {\n return;\n }\n // const bgColor = sceneNode.backgroundColor;\n // let bgColor3: Color3;\n // if (!bgColor) {\n // bgColor3 = new Color3(this.scene.clearColor.r, this.scene.clearColor.g, this.scene.clearColor.b);\n // } else {\n // bgColor3 = bgColor.a === 0 ? Color3.Black() : new Color3(bgColor.r, bgColor.g, bgColor.b);\n // }\n // const lineColor = bgColor3.toLuminance() > 0.5 ? Color3.Black() : Color3.White();\n const gridStepSize = sceneNode.gridStepSize || 1;\n const gridSize = (sceneNode.gridSize || 10) * gridStepSize;\n const colors = [\n Color4.FromColor3(Color3.Red()),\n Color4.FromColor3(Color3.Green()),\n Color4.FromColor3(Color3.Blue()),\n ];\n const pointsAxes = [\n [new Vector3(-gridSize, 0, 0), new Vector3(gridSize, 0, 0)],\n [new Vector3(0, -gridSize, 0), new Vector3(0, gridSize, 0)],\n [new Vector3(0, 0, -gridSize), new Vector3(0, 0, gridSize)],\n ];\n const pointColors = [\n [colors[0], colors[0]],\n [colors[1], colors[1]],\n [colors[2], colors[2]],\n ];\n const gridMeshAxes = MeshBuilder.CreateLineSystem('gridAxes', {\n lines: pointsAxes,\n updatable: false,\n colors: pointColors,\n }, this.scene);\n gridMeshAxes.isPickable = false;\n this.gridSettings.gridAxesMesh = gridMeshAxes;\n this.createTextLabels([\n { text: 'x', position: new exports.KbVector(gridSize, 0.5, 0), color: colors[0] },\n { text: 'y', position: new exports.KbVector(0.5, gridSize, 0), color: colors[1] },\n { text: 'z', position: new exports.KbVector(0, 0.5, gridSize), color: colors[2] },\n ]);\n }\n disableGridAxes() {\n if (this.gridSettings.gridAxesMesh) {\n this.gridSettings.gridAxesMesh.dispose();\n delete this.gridSettings.gridAxesMesh;\n }\n }\n createTextLabels(items, style = {}) {\n if (!style.font) {\n style.font = 'bold 32px sans-serif';\n }\n items.forEach(label => {\n const font = label.style && label.style.font ? label.style.font : style.font;\n if (!label.id) {\n label.id = label.text;\n }\n if (this.textLabelMap.has(label.id)) {\n const labelMesh = this.textLabelMap.get(label.id);\n labelMesh.position = label.position.toVec3();\n }\n else {\n // Set height for plane\n const fontSize = parseInt(font.replace(/\\D+(\\d+)px.*/, '$1')) || 32;\n const planeHeight = fontSize / 32;\n const textureHeight = 2 * fontSize;\n const ratio = planeHeight / textureHeight;\n // Use a temporay canvas to calculate the length of the text\n const temp = document.createElement('canvas');\n temp.width = 64;\n temp.height = 64;\n const tmpctx = temp.getContext('2d');\n if (tmpctx) {\n tmpctx.font = font;\n const textureWidth = tmpctx.measureText(label.text).width + 8;\n temp.width = 0;\n temp.height = 0;\n // Create dynamic texture and write the text\n const dynamicId = `axisLabel${label.id}`;\n const planeWidth = textureWidth * ratio;\n const dynamicTexture = new DynamicTexture(dynamicId + '-labelTexture', { width: textureWidth, height: textureHeight }, this.scene, false);\n const mat = new StandardMaterial(dynamicId + '-labelMaterial', this.scene);\n mat.emissiveTexture = dynamicTexture;\n mat.diffuseTexture = dynamicTexture;\n mat.sideOrientation = Mesh.BACKSIDE;\n dynamicTexture.hasAlpha = true;\n dynamicTexture.drawText(label.text, null, null, font, label.color.toHexString(), 'rgba(0,0,0,0)', true);\n // Create plane and set dynamic texture as material\n const plane = MeshBuilder.CreatePlane(dynamicId, { width: planeWidth, height: planeHeight }, this.scene);\n plane.material = mat;\n plane.position = label.position.toVec3();\n plane.isPickable = false;\n this.textLabelMap.set(label.id, plane);\n }\n }\n });\n }\n calculateGridPoints(stepSize, gridCount, normal, gridFullSize) {\n const points = [];\n const wV1 = new Vector3(), wV2 = new Vector3(), vStep = new Vector3();\n if (gridFullSize === undefined) {\n gridFullSize = gridCount;\n }\n let uVec;\n let vVec;\n switch (normal) {\n case exports.eAxis.x:\n uVec = new Vector3(0, stepSize, 0);\n vVec = new Vector3(0, 0, stepSize);\n break;\n case exports.eAxis.y:\n uVec = new Vector3(stepSize, 0, 0);\n vVec = new Vector3(0, 0, stepSize);\n break;\n case exports.eAxis.z:\n uVec = new Vector3(stepSize, 0, 0);\n vVec = new Vector3(0, stepSize, 0);\n break;\n }\n wV1.setAll(0).addInPlace(vVec).scaleInPlace(-1).scaleInPlace(gridFullSize);\n wV2.setAll(0).scaleInPlace(-1).addInPlace(vVec).scaleInPlace(gridFullSize);\n for (let i = 0; i < gridCount * 2 + 1; i++) {\n const d = (2 * (i % 2) - 1) * Math.ceil(i / 2);\n vStep.copyFrom(uVec).scaleInPlace(d);\n points.push([wV1.add(vStep), wV2.add(vStep)]);\n }\n wV1.setAll(0).addInPlace(uVec).scaleInPlace(-1).scaleInPlace(gridFullSize);\n wV2.setAll(0).scaleInPlace(-1).addInPlace(uVec).scaleInPlace(gridFullSize);\n for (let i = 0; i < gridCount * 2 + 1; i++) {\n const d = (2 * (i % 2) - 1) * Math.ceil(i / 2);\n vStep.copyFrom(vVec).scaleInPlace(d);\n points.push([wV1.add(vStep), wV2.add(vStep)]);\n }\n return points;\n }\n }\n\n /*\n * Polyfill for createImageBitmap\n * https://developer.mozilla.org/en-US/docs/Web/API/WindowOrWorkerGlobalScope/createImageBitmap\n *\n * Supports CanvasImageSource (img, video, canvas) sources, Blobs, and ImageData.\n *\n * From:\n * - https://dev.to/nektro/createimagebitmap-polyfill-for-safari-and-edge-228\n * - https://gist.github.com/MonsieurV/fb640c29084c171b4444184858a91bc7\n * Updated by:\n * - Yoan Tournade \n * - diachedelic, https://gist.github.com/diachedelic\n * - Paul Ellis, https://pseudosavant.com\n */\n (function createImageBitmapIIFE(global) {\n function isCanvasImageSource(el) {\n const validElements = ['img', 'video', 'canvas'];\n return el && el.tagName && validElements.includes(el.tagName.toLowerCase());\n }\n function idealSize(currentValue, newValue, numerator, denominator) {\n if (typeof newValue === 'number')\n return newValue;\n if (typeof numerator !== 'number' || typeof denominator !== 'number')\n return currentValue;\n return (numerator / denominator) * currentValue;\n }\n if (!('createImageBitmap' in global)) {\n global.createImageBitmap = async function polyfillCreateImageBitmap(data, opts) {\n return new Promise((resolve, reject) => {\n opts = opts || {};\n let dataURL;\n const canvas = document.createElement('canvas');\n try {\n const ctx = canvas.getContext('2d');\n if (data instanceof Blob) {\n dataURL = URL.createObjectURL(data);\n }\n else if (isCanvasImageSource(data)) {\n const width = data.naturalWidth || data.videoWidth || data.clientWidth || data.width;\n const height = data.naturalHeight || data.videoHeight || data.clientHeight || data.height;\n canvas.width = idealSize(width, opts.resizeWidth, opts.resizeHeight, height);\n canvas.height = idealSize(height, opts.resizeHeight, opts.resizeWidth, width);\n ctx.drawImage(data, 0, 0, canvas.width, canvas.height);\n dataURL = canvas.toDataURL();\n }\n else if (data instanceof ImageData) {\n canvas.width = idealSize(data.width, opts.resizeWidth, opts.resizeHeight, data.height);\n canvas.height = idealSize(data.height, opts.resizeHeight, opts.resizeWidth, data.width);\n ctx.putImageData(data, 0, 0);\n dataURL = canvas.toDataURL();\n }\n else {\n reject('createImageBitmap does not handle the provided image source type');\n }\n const img = new Image();\n img.onerror = reject;\n img.onload = () => resolve(img);\n img.src = dataURL;\n }\n finally {\n // avoid memory leaks on iOS Safari, see https://stackoverflow.com/a/52586606\n canvas.width = 0;\n canvas.height = 0;\n }\n });\n };\n }\n })(window);\n\n /**\n * Generic utilities for the babylon scene\n */\n /** returns the x,y pixel position of the node on the canvas */\n function nodeToScreen(bNode, camera) {\n camera = getCamera(camera);\n const nodeWorldPos = bNode.getAbsolutePosition(); // Vector3.TransformCoordinates(bNode.position, bNode.getWorldMatrix());\n return toScreen(nodeWorldPos, camera);\n }\n /** converts screen units */\n function toPixels(viewportUnits, camera) {\n camera = getCamera(camera);\n const canvasRect = camera.getEngine().getRenderingCanvasClientRect();\n return {\n x: viewportUnits.x * canvasRect.width,\n y: viewportUnits.y * canvasRect.height,\n };\n }\n /** returns the screen position as normalized device units a ratio from 0 to 1 */\n function toNDC(worldPos, camera) {\n camera = getCamera(camera);\n const pos = Vector3.Project(worldPos, Matrix.IdentityReadOnly, camera.getTransformationMatrix(), camera.viewport);\n return {\n x: pos.x,\n y: pos.y,\n };\n }\n function toScreen(worldPos, camera) {\n camera = getCamera(camera);\n const scene = camera.getScene();\n const engine = scene.getEngine();\n const pos = Vector3.Project(worldPos, Matrix.IdentityReadOnly, scene.getTransformMatrix(), camera.viewport.toGlobal(engine.getRenderWidth(), engine.getRenderHeight()));\n return {\n x: pos.x,\n y: pos.y,\n };\n }\n function getScreenBoundingBox(node, camera, includeDescendants = true) {\n camera = getCamera(camera);\n const bv = getHierarchyBoundingVectors(node, includeDescendants);\n return boundingVectorsToScreen(bv, camera);\n }\n // export function getBoundingVectors(bNode: AbstractMesh | TransformNode | undefined, scene: Scene, includeDescendants = true) {\n // let bv: { min: Vector3, max: Vector3 } | undefined;\n // if (!bNode) {\n // bv = scene.getWorldExtends();\n // } else {\n // bv = bNode.getHierarchyBoundingVectors(true);\n // if (bv.min.x == Number.MAX_VALUE && bv.min.y == Number.MAX_VALUE && bv.min.z == Number.MAX_VALUE && bv.max.x == -Number.MAX_VALUE && bv.max.y == -Number.MAX_VALUE && bv.max.z == -Number.MAX_VALUE) {\n // bv = {\n // min: bNode.absolutePosition.clone(),\n // max: bNode.absolutePosition.clone()\n // }\n // }\n // }\n // return bv;\n // }\n /**\n * Babylon's getHierarchyBoundingVectors has a bug that surfaces when called on a transformNode that has no real meshes in it's child hierarchy.\n * Use this wrapper instead\n * @param bNode\n * @param includeDescendants\n */\n function getHierarchyBoundingVectors(bNode, includeDescendants = true) {\n let bv = bNode.getHierarchyBoundingVectors(includeDescendants);\n //to handle a babylon bug that returns maxvalues when the node is an empty transformnode\n if (bv.min.x == Number.MAX_VALUE &&\n bv.min.y == Number.MAX_VALUE &&\n bv.min.z == Number.MAX_VALUE &&\n bv.max.x == -Number.MAX_VALUE &&\n bv.max.y == -Number.MAX_VALUE &&\n bv.max.z == -Number.MAX_VALUE) {\n bv = {\n min: bNode.absolutePosition.clone(),\n max: bNode.absolutePosition.clone(),\n };\n }\n return bv;\n }\n function boundingVectorsToScreen(boundingVectors, camera) {\n camera = getCamera(camera);\n const bv = boundingVectors;\n const xs = new Array();\n const ys = new Array();\n for (let i = 0; i < 8; i++) {\n const x = bv[Math.floor(i / 4) ? 'max' : 'min'].x;\n const y = bv[Math.floor(i / 2) % 2 ? 'max' : 'min'].y;\n const z = bv[i % 2 ? 'max' : 'min'].z;\n const v = new Vector3(x, y, z);\n const sp = toNDC(v, camera);\n xs.push(sp.x);\n ys.push(sp.y);\n }\n const minx = Math.min(...xs);\n const maxx = Math.max(...xs);\n const miny = Math.min(...ys);\n const maxy = Math.max(...ys);\n const width = maxx - minx;\n const height = maxy - miny;\n const x = minx + width / 2;\n const y = miny + height / 2;\n const engine = camera.getEngine();\n const canvas = engine.getRenderingCanvas();\n const canvasWidth = canvas.width * engine.getHardwareScalingLevel();\n const canvasHeight = canvas.height * engine.getHardwareScalingLevel();\n return {\n x: x * canvasWidth,\n y: y * canvasHeight,\n top: miny * canvasHeight,\n left: minx * canvasWidth,\n width: width * canvasWidth,\n height: height * canvasHeight,\n };\n }\n function updateMatrixOfAncestorChain(ancestor, child) {\n while (child != ancestor && child instanceof TransformNode) {\n child.computeWorldMatrix(true);\n child = child.parent;\n }\n if (child) {\n child.computeWorldMatrix(true);\n }\n }\n function getCamera(camera) {\n if (camera instanceof WebXRCamera) {\n return camera.rigCameras[0];\n }\n return camera;\n }\n\n (function (eWorkerTasks) {\n eWorkerTasks[\"usdConvertGeometry\"] = \"usdConvertGeometry\";\n eWorkerTasks[\"usdPackage\"] = \"usdPackageStart\";\n })(exports.eWorkerTasks || (exports.eWorkerTasks = {}));\n (function (eKB3DMessageType) {\n eKB3DMessageType[\"taskStart\"] = \"taskStart\";\n eKB3DMessageType[\"taskError\"] = \"taskError\";\n eKB3DMessageType[\"taskProgress\"] = \"taskProgress\";\n eKB3DMessageType[\"taskComplete\"] = \"taskComplete\";\n })(exports.eKB3DMessageType || (exports.eKB3DMessageType = {}));\n const THREAD_COUNT = 2;\n class Kb3dWorker {\n constructor() {\n this.workerPool = [];\n this.toggle = 0;\n this.requestsInFlight = new Map();\n if (!Kb3dWorker.workerPath) {\n throw Error('Worker Script path not set');\n }\n const workerPath = Kb3dWorker.workerPath;\n const content = `importScripts( \"${workerPath}\" );`;\n const workerBlobUrl = URL.createObjectURL(new Blob([content], { type: 'text/javascript' }));\n const workerReturnMessage = (event) => {\n if ('task' in event.data) {\n const data = event.data;\n const sub = this.requestsInFlight.get(data.taskId);\n if (sub) {\n // We have to remove the request from the inflight map first because the sub.next can trigger the dispose\n // But the sub.complete has to be after the sub.next\n if (data.type === exports.eKB3DMessageType.taskComplete) {\n this.requestsInFlight.delete(data.taskId);\n }\n if (data.type === exports.eKB3DMessageType.taskError) {\n sub.error(data.payload);\n }\n else {\n sub.next(data);\n }\n if (data.type === exports.eKB3DMessageType.taskComplete) {\n sub.complete();\n }\n }\n }\n };\n for (let i = 0; i < THREAD_COUNT; i++) {\n const newWorker = new Worker(workerBlobUrl);\n newWorker.onmessage = workerReturnMessage;\n this.workerPool.push(newWorker);\n }\n URL.revokeObjectURL(workerBlobUrl);\n }\n startTask(task, taskPayload, transfer) {\n const messagePayload = {\n type: exports.eKB3DMessageType.taskStart,\n task,\n taskId: UUID.generate(),\n payload: taskPayload,\n };\n this.workerPool[this.toggle].postMessage(messagePayload, transfer ? { transfer } : undefined);\n const sub = new Subject();\n this.requestsInFlight.set(messagePayload.taskId, sub);\n this.toggle = (this.toggle + 1) % THREAD_COUNT;\n return sub;\n }\n dispose() {\n this.workerPool.forEach(worker => {\n worker.terminate();\n });\n this.requestsInFlight.forEach(sub => {\n sub.error('Worker Terminated');\n sub.complete();\n });\n this.requestsInFlight.clear();\n }\n }\n Kb3dWorker.workerPath = '';\n\n var version = '18.5.0';\n\n /**\n * Tween.js - Licensed under the MIT license\n * https://github.com/tweenjs/tween.js\n * ----------------------------------------------\n *\n * See https://github.com/tweenjs/tween.js/graphs/contributors for the full list of contributors.\n * Thank you all, you're awesome!\n */\n\n\n var _Group = function () {\n \tthis._tweens = {};\n \tthis._tweensAddedDuringUpdate = {};\n };\n\n _Group.prototype = {\n \tgetAll: function () {\n\n \t\treturn Object.keys(this._tweens).map(function (tweenId) {\n \t\t\treturn this._tweens[tweenId];\n \t\t}.bind(this));\n\n \t},\n\n \tremoveAll: function () {\n\n \t\tthis._tweens = {};\n\n \t},\n\n \tadd: function (tween) {\n\n \t\tthis._tweens[tween.getId()] = tween;\n \t\tthis._tweensAddedDuringUpdate[tween.getId()] = tween;\n\n \t},\n\n \tremove: function (tween) {\n\n \t\tdelete this._tweens[tween.getId()];\n \t\tdelete this._tweensAddedDuringUpdate[tween.getId()];\n\n \t},\n\n \tupdate: function (time, preserve) {\n\n \t\tvar tweenIds = Object.keys(this._tweens);\n\n \t\tif (tweenIds.length === 0) {\n \t\t\treturn false;\n \t\t}\n\n \t\ttime = time !== undefined ? time : TWEEN.now();\n\n \t\t// Tweens are updated in \"batches\". If you add a new tween during an\n \t\t// update, then the new tween will be updated in the next batch.\n \t\t// If you remove a tween during an update, it may or may not be updated.\n \t\t// However, if the removed tween was added during the current batch,\n \t\t// then it will not be updated.\n \t\twhile (tweenIds.length > 0) {\n \t\t\tthis._tweensAddedDuringUpdate = {};\n\n \t\t\tfor (var i = 0; i < tweenIds.length; i++) {\n\n \t\t\t\tvar tween = this._tweens[tweenIds[i]];\n\n \t\t\t\tif (tween && tween.update(time) === false) {\n \t\t\t\t\ttween._isPlaying = false;\n\n \t\t\t\t\tif (!preserve) {\n \t\t\t\t\t\tdelete this._tweens[tweenIds[i]];\n \t\t\t\t\t}\n \t\t\t\t}\n \t\t\t}\n\n \t\t\ttweenIds = Object.keys(this._tweensAddedDuringUpdate);\n \t\t}\n\n \t\treturn true;\n\n \t}\n };\n\n var TWEEN = new _Group();\n\n TWEEN.Group = _Group;\n TWEEN._nextId = 0;\n TWEEN.nextId = function () {\n \treturn TWEEN._nextId++;\n };\n\n\n // Include a performance.now polyfill.\n // In node.js, use process.hrtime.\n if (typeof (self) === 'undefined' && typeof (process) !== 'undefined' && process.hrtime) {\n \tTWEEN.now = function () {\n \t\tvar time = process.hrtime();\n\n \t\t// Convert [seconds, nanoseconds] to milliseconds.\n \t\treturn time[0] * 1000 + time[1] / 1000000;\n \t};\n }\n // In a browser, use self.performance.now if it is available.\n else if (typeof (self) !== 'undefined' &&\n self.performance !== undefined &&\n \t\t self.performance.now !== undefined) {\n \t// This must be bound, because directly assigning this function\n \t// leads to an invocation exception in Chrome.\n \tTWEEN.now = self.performance.now.bind(self.performance);\n }\n // Use Date.now if it is available.\n else if (Date.now !== undefined) {\n \tTWEEN.now = Date.now;\n }\n // Otherwise, use 'new Date().getTime()'.\n else {\n \tTWEEN.now = function () {\n \t\treturn new Date().getTime();\n \t};\n }\n\n\n TWEEN.Tween = function (object, group) {\n \tthis._isPaused = false;\n \tthis._pauseStart = null;\n \tthis._object = object;\n \tthis._valuesStart = {};\n \tthis._valuesEnd = {};\n \tthis._valuesStartRepeat = {};\n \tthis._duration = 1000;\n \tthis._repeat = 0;\n \tthis._repeatDelayTime = undefined;\n \tthis._yoyo = false;\n \tthis._isPlaying = false;\n \tthis._reversed = false;\n \tthis._delayTime = 0;\n \tthis._startTime = null;\n \tthis._easingFunction = TWEEN.Easing.Linear.None;\n \tthis._interpolationFunction = TWEEN.Interpolation.Linear;\n \tthis._chainedTweens = [];\n \tthis._onStartCallback = null;\n \tthis._onStartCallbackFired = false;\n \tthis._onUpdateCallback = null;\n \tthis._onRepeatCallback = null;\n \tthis._onCompleteCallback = null;\n \tthis._onStopCallback = null;\n \tthis._group = group || TWEEN;\n \tthis._id = TWEEN.nextId();\n\n };\n\n TWEEN.Tween.prototype = {\n \tgetId: function () {\n \t\treturn this._id;\n \t},\n\n \tisPlaying: function () {\n \t\treturn this._isPlaying;\n \t},\n\n \tisPaused: function () {\n \t\treturn this._isPaused;\n \t},\n\n \tto: function (properties, duration) {\n\n \t\tthis._valuesEnd = Object.create(properties);\n\n \t\tif (duration !== undefined) {\n \t\t\tthis._duration = duration;\n \t\t}\n\n \t\treturn this;\n\n \t},\n\n \tduration: function duration(d) {\n \t\tthis._duration = d;\n \t\treturn this;\n \t},\n\n \tstart: function (time) {\n\n \t\tthis._group.add(this);\n\n \t\tthis._isPlaying = true;\n\n \t\tthis._isPaused = false;\n\n \t\tthis._onStartCallbackFired = false;\n\n \t\tthis._startTime = time !== undefined ? typeof time === 'string' ? TWEEN.now() + parseFloat(time) : time : TWEEN.now();\n \t\tthis._startTime += this._delayTime;\n\n \t\tfor (var property in this._valuesEnd) {\n\n \t\t\t// Check if an Array was provided as property value\n \t\t\tif (this._valuesEnd[property] instanceof Array) {\n\n \t\t\t\tif (this._valuesEnd[property].length === 0) {\n \t\t\t\t\tcontinue;\n \t\t\t\t}\n\n \t\t\t\t// Create a local copy of the Array with the start value at the front\n \t\t\t\tthis._valuesEnd[property] = [this._object[property]].concat(this._valuesEnd[property]);\n\n \t\t\t}\n\n \t\t\t// If `to()` specifies a property that doesn't exist in the source object,\n \t\t\t// we should not set that property in the object\n \t\t\tif (this._object[property] === undefined) {\n \t\t\t\tcontinue;\n \t\t\t}\n\n \t\t\t// Save the starting value, but only once.\n \t\t\tif (typeof(this._valuesStart[property]) === 'undefined') {\n \t\t\t\tthis._valuesStart[property] = this._object[property];\n \t\t\t}\n\n \t\t\tif ((this._valuesStart[property] instanceof Array) === false) {\n \t\t\t\tthis._valuesStart[property] *= 1.0; // Ensures we're using numbers, not strings\n \t\t\t}\n\n \t\t\tthis._valuesStartRepeat[property] = this._valuesStart[property] || 0;\n\n \t\t}\n\n \t\treturn this;\n\n \t},\n\n \tstop: function () {\n\n \t\tif (!this._isPlaying) {\n \t\t\treturn this;\n \t\t}\n\n \t\tthis._group.remove(this);\n\n \t\tthis._isPlaying = false;\n\n \t\tthis._isPaused = false;\n\n \t\tif (this._onStopCallback !== null) {\n \t\t\tthis._onStopCallback(this._object);\n \t\t}\n\n \t\tthis.stopChainedTweens();\n \t\treturn this;\n\n \t},\n\n \tend: function () {\n\n \t\tthis.update(Infinity);\n \t\treturn this;\n\n \t},\n\n \tpause: function(time) {\n\n \t\tif (this._isPaused || !this._isPlaying) {\n \t\t\treturn this;\n \t\t}\n\n \t\tthis._isPaused = true;\n\n \t\tthis._pauseStart = time === undefined ? TWEEN.now() : time;\n\n \t\tthis._group.remove(this);\n\n \t\treturn this;\n\n \t},\n\n \tresume: function(time) {\n\n \t\tif (!this._isPaused || !this._isPlaying) {\n \t\t\treturn this;\n \t\t}\n\n \t\tthis._isPaused = false;\n\n \t\tthis._startTime += (time === undefined ? TWEEN.now() : time)\n \t\t\t- this._pauseStart;\n\n \t\tthis._pauseStart = 0;\n\n \t\tthis._group.add(this);\n\n \t\treturn this;\n\n \t},\n\n \tstopChainedTweens: function () {\n\n \t\tfor (var i = 0, numChainedTweens = this._chainedTweens.length; i < numChainedTweens; i++) {\n \t\t\tthis._chainedTweens[i].stop();\n \t\t}\n\n \t},\n\n \tgroup: function (group) {\n \t\tthis._group = group;\n \t\treturn this;\n \t},\n\n \tdelay: function (amount) {\n\n \t\tthis._delayTime = amount;\n \t\treturn this;\n\n \t},\n\n \trepeat: function (times) {\n\n \t\tthis._repeat = times;\n \t\treturn this;\n\n \t},\n\n \trepeatDelay: function (amount) {\n\n \t\tthis._repeatDelayTime = amount;\n \t\treturn this;\n\n \t},\n\n \tyoyo: function (yoyo) {\n\n \t\tthis._yoyo = yoyo;\n \t\treturn this;\n\n \t},\n\n \teasing: function (easingFunction) {\n\n \t\tthis._easingFunction = easingFunction;\n \t\treturn this;\n\n \t},\n\n \tinterpolation: function (interpolationFunction) {\n\n \t\tthis._interpolationFunction = interpolationFunction;\n \t\treturn this;\n\n \t},\n\n \tchain: function () {\n\n \t\tthis._chainedTweens = arguments;\n \t\treturn this;\n\n \t},\n\n \tonStart: function (callback) {\n\n \t\tthis._onStartCallback = callback;\n \t\treturn this;\n\n \t},\n\n \tonUpdate: function (callback) {\n\n \t\tthis._onUpdateCallback = callback;\n \t\treturn this;\n\n \t},\n\n \tonRepeat: function onRepeat(callback) {\n\n \t\tthis._onRepeatCallback = callback;\n \t\treturn this;\n\n \t},\n\n \tonComplete: function (callback) {\n\n \t\tthis._onCompleteCallback = callback;\n \t\treturn this;\n\n \t},\n\n \tonStop: function (callback) {\n\n \t\tthis._onStopCallback = callback;\n \t\treturn this;\n\n \t},\n\n \tupdate: function (time) {\n\n \t\tvar property;\n \t\tvar elapsed;\n \t\tvar value;\n\n \t\tif (time < this._startTime) {\n \t\t\treturn true;\n \t\t}\n\n \t\tif (this._onStartCallbackFired === false) {\n\n \t\t\tif (this._onStartCallback !== null) {\n \t\t\t\tthis._onStartCallback(this._object);\n \t\t\t}\n\n \t\t\tthis._onStartCallbackFired = true;\n \t\t}\n\n \t\telapsed = (time - this._startTime) / this._duration;\n \t\telapsed = (this._duration === 0 || elapsed > 1) ? 1 : elapsed;\n\n \t\tvalue = this._easingFunction(elapsed);\n\n \t\tfor (property in this._valuesEnd) {\n\n \t\t\t// Don't update properties that do not exist in the source object\n \t\t\tif (this._valuesStart[property] === undefined) {\n \t\t\t\tcontinue;\n \t\t\t}\n\n \t\t\tvar start = this._valuesStart[property] || 0;\n \t\t\tvar end = this._valuesEnd[property];\n\n \t\t\tif (end instanceof Array) {\n\n \t\t\t\tthis._object[property] = this._interpolationFunction(end, value);\n\n \t\t\t} else {\n\n \t\t\t\t// Parses relative end values with start as base (e.g.: +10, -3)\n \t\t\t\tif (typeof (end) === 'string') {\n\n \t\t\t\t\tif (end.charAt(0) === '+' || end.charAt(0) === '-') {\n \t\t\t\t\t\tend = start + parseFloat(end);\n \t\t\t\t\t} else {\n \t\t\t\t\t\tend = parseFloat(end);\n \t\t\t\t\t}\n \t\t\t\t}\n\n \t\t\t\t// Protect against non numeric properties.\n \t\t\t\tif (typeof (end) === 'number') {\n \t\t\t\t\tthis._object[property] = start + (end - start) * value;\n \t\t\t\t}\n\n \t\t\t}\n\n \t\t}\n\n \t\tif (this._onUpdateCallback !== null) {\n \t\t\tthis._onUpdateCallback(this._object, elapsed);\n \t\t}\n\n \t\tif (elapsed === 1) {\n\n \t\t\tif (this._repeat > 0) {\n\n \t\t\t\tif (isFinite(this._repeat)) {\n \t\t\t\t\tthis._repeat--;\n \t\t\t\t}\n\n \t\t\t\t// Reassign starting values, restart by making startTime = now\n \t\t\t\tfor (property in this._valuesStartRepeat) {\n\n \t\t\t\t\tif (typeof (this._valuesEnd[property]) === 'string') {\n \t\t\t\t\t\tthis._valuesStartRepeat[property] = this._valuesStartRepeat[property] + parseFloat(this._valuesEnd[property]);\n \t\t\t\t\t}\n\n \t\t\t\t\tif (this._yoyo) {\n \t\t\t\t\t\tvar tmp = this._valuesStartRepeat[property];\n\n \t\t\t\t\t\tthis._valuesStartRepeat[property] = this._valuesEnd[property];\n \t\t\t\t\t\tthis._valuesEnd[property] = tmp;\n \t\t\t\t\t}\n\n \t\t\t\t\tthis._valuesStart[property] = this._valuesStartRepeat[property];\n\n \t\t\t\t}\n\n \t\t\t\tif (this._yoyo) {\n \t\t\t\t\tthis._reversed = !this._reversed;\n \t\t\t\t}\n\n \t\t\t\tif (this._repeatDelayTime !== undefined) {\n \t\t\t\t\tthis._startTime = time + this._repeatDelayTime;\n \t\t\t\t} else {\n \t\t\t\t\tthis._startTime = time + this._delayTime;\n \t\t\t\t}\n\n \t\t\t\tif (this._onRepeatCallback !== null) {\n \t\t\t\t\tthis._onRepeatCallback(this._object);\n \t\t\t\t}\n\n \t\t\t\treturn true;\n\n \t\t\t} else {\n\n \t\t\t\tif (this._onCompleteCallback !== null) {\n\n \t\t\t\t\tthis._onCompleteCallback(this._object);\n \t\t\t\t}\n\n \t\t\t\tfor (var i = 0, numChainedTweens = this._chainedTweens.length; i < numChainedTweens; i++) {\n \t\t\t\t\t// Make the chained tweens start exactly at the time they should,\n \t\t\t\t\t// even if the `update()` method was called way past the duration of the tween\n \t\t\t\t\tthis._chainedTweens[i].start(this._startTime + this._duration);\n \t\t\t\t}\n\n \t\t\t\treturn false;\n\n \t\t\t}\n\n \t\t}\n\n \t\treturn true;\n\n \t}\n };\n\n\n TWEEN.Easing = {\n\n \tLinear: {\n\n \t\tNone: function (k) {\n\n \t\t\treturn k;\n\n \t\t}\n\n \t},\n\n \tQuadratic: {\n\n \t\tIn: function (k) {\n\n \t\t\treturn k * k;\n\n \t\t},\n\n \t\tOut: function (k) {\n\n \t\t\treturn k * (2 - k);\n\n \t\t},\n\n \t\tInOut: function (k) {\n\n \t\t\tif ((k *= 2) < 1) {\n \t\t\t\treturn 0.5 * k * k;\n \t\t\t}\n\n \t\t\treturn - 0.5 * (--k * (k - 2) - 1);\n\n \t\t}\n\n \t},\n\n \tCubic: {\n\n \t\tIn: function (k) {\n\n \t\t\treturn k * k * k;\n\n \t\t},\n\n \t\tOut: function (k) {\n\n \t\t\treturn --k * k * k + 1;\n\n \t\t},\n\n \t\tInOut: function (k) {\n\n \t\t\tif ((k *= 2) < 1) {\n \t\t\t\treturn 0.5 * k * k * k;\n \t\t\t}\n\n \t\t\treturn 0.5 * ((k -= 2) * k * k + 2);\n\n \t\t}\n\n \t},\n\n \tQuartic: {\n\n \t\tIn: function (k) {\n\n \t\t\treturn k * k * k * k;\n\n \t\t},\n\n \t\tOut: function (k) {\n\n \t\t\treturn 1 - (--k * k * k * k);\n\n \t\t},\n\n \t\tInOut: function (k) {\n\n \t\t\tif ((k *= 2) < 1) {\n \t\t\t\treturn 0.5 * k * k * k * k;\n \t\t\t}\n\n \t\t\treturn - 0.5 * ((k -= 2) * k * k * k - 2);\n\n \t\t}\n\n \t},\n\n \tQuintic: {\n\n \t\tIn: function (k) {\n\n \t\t\treturn k * k * k * k * k;\n\n \t\t},\n\n \t\tOut: function (k) {\n\n \t\t\treturn --k * k * k * k * k + 1;\n\n \t\t},\n\n \t\tInOut: function (k) {\n\n \t\t\tif ((k *= 2) < 1) {\n \t\t\t\treturn 0.5 * k * k * k * k * k;\n \t\t\t}\n\n \t\t\treturn 0.5 * ((k -= 2) * k * k * k * k + 2);\n\n \t\t}\n\n \t},\n\n \tSinusoidal: {\n\n \t\tIn: function (k) {\n\n \t\t\treturn 1 - Math.cos(k * Math.PI / 2);\n\n \t\t},\n\n \t\tOut: function (k) {\n\n \t\t\treturn Math.sin(k * Math.PI / 2);\n\n \t\t},\n\n \t\tInOut: function (k) {\n\n \t\t\treturn 0.5 * (1 - Math.cos(Math.PI * k));\n\n \t\t}\n\n \t},\n\n \tExponential: {\n\n \t\tIn: function (k) {\n\n \t\t\treturn k === 0 ? 0 : Math.pow(1024, k - 1);\n\n \t\t},\n\n \t\tOut: function (k) {\n\n \t\t\treturn k === 1 ? 1 : 1 - Math.pow(2, - 10 * k);\n\n \t\t},\n\n \t\tInOut: function (k) {\n\n \t\t\tif (k === 0) {\n \t\t\t\treturn 0;\n \t\t\t}\n\n \t\t\tif (k === 1) {\n \t\t\t\treturn 1;\n \t\t\t}\n\n \t\t\tif ((k *= 2) < 1) {\n \t\t\t\treturn 0.5 * Math.pow(1024, k - 1);\n \t\t\t}\n\n \t\t\treturn 0.5 * (- Math.pow(2, - 10 * (k - 1)) + 2);\n\n \t\t}\n\n \t},\n\n \tCircular: {\n\n \t\tIn: function (k) {\n\n \t\t\treturn 1 - Math.sqrt(1 - k * k);\n\n \t\t},\n\n \t\tOut: function (k) {\n\n \t\t\treturn Math.sqrt(1 - (--k * k));\n\n \t\t},\n\n \t\tInOut: function (k) {\n\n \t\t\tif ((k *= 2) < 1) {\n \t\t\t\treturn - 0.5 * (Math.sqrt(1 - k * k) - 1);\n \t\t\t}\n\n \t\t\treturn 0.5 * (Math.sqrt(1 - (k -= 2) * k) + 1);\n\n \t\t}\n\n \t},\n\n \tElastic: {\n\n \t\tIn: function (k) {\n\n \t\t\tif (k === 0) {\n \t\t\t\treturn 0;\n \t\t\t}\n\n \t\t\tif (k === 1) {\n \t\t\t\treturn 1;\n \t\t\t}\n\n \t\t\treturn -Math.pow(2, 10 * (k - 1)) * Math.sin((k - 1.1) * 5 * Math.PI);\n\n \t\t},\n\n \t\tOut: function (k) {\n\n \t\t\tif (k === 0) {\n \t\t\t\treturn 0;\n \t\t\t}\n\n \t\t\tif (k === 1) {\n \t\t\t\treturn 1;\n \t\t\t}\n\n \t\t\treturn Math.pow(2, -10 * k) * Math.sin((k - 0.1) * 5 * Math.PI) + 1;\n\n \t\t},\n\n \t\tInOut: function (k) {\n\n \t\t\tif (k === 0) {\n \t\t\t\treturn 0;\n \t\t\t}\n\n \t\t\tif (k === 1) {\n \t\t\t\treturn 1;\n \t\t\t}\n\n \t\t\tk *= 2;\n\n \t\t\tif (k < 1) {\n \t\t\t\treturn -0.5 * Math.pow(2, 10 * (k - 1)) * Math.sin((k - 1.1) * 5 * Math.PI);\n \t\t\t}\n\n \t\t\treturn 0.5 * Math.pow(2, -10 * (k - 1)) * Math.sin((k - 1.1) * 5 * Math.PI) + 1;\n\n \t\t}\n\n \t},\n\n \tBack: {\n\n \t\tIn: function (k) {\n\n \t\t\tvar s = 1.70158;\n\n \t\t\treturn k * k * ((s + 1) * k - s);\n\n \t\t},\n\n \t\tOut: function (k) {\n\n \t\t\tvar s = 1.70158;\n\n \t\t\treturn --k * k * ((s + 1) * k + s) + 1;\n\n \t\t},\n\n \t\tInOut: function (k) {\n\n \t\t\tvar s = 1.70158 * 1.525;\n\n \t\t\tif ((k *= 2) < 1) {\n \t\t\t\treturn 0.5 * (k * k * ((s + 1) * k - s));\n \t\t\t}\n\n \t\t\treturn 0.5 * ((k -= 2) * k * ((s + 1) * k + s) + 2);\n\n \t\t}\n\n \t},\n\n \tBounce: {\n\n \t\tIn: function (k) {\n\n \t\t\treturn 1 - TWEEN.Easing.Bounce.Out(1 - k);\n\n \t\t},\n\n \t\tOut: function (k) {\n\n \t\t\tif (k < (1 / 2.75)) {\n \t\t\t\treturn 7.5625 * k * k;\n \t\t\t} else if (k < (2 / 2.75)) {\n \t\t\t\treturn 7.5625 * (k -= (1.5 / 2.75)) * k + 0.75;\n \t\t\t} else if (k < (2.5 / 2.75)) {\n \t\t\t\treturn 7.5625 * (k -= (2.25 / 2.75)) * k + 0.9375;\n \t\t\t} else {\n \t\t\t\treturn 7.5625 * (k -= (2.625 / 2.75)) * k + 0.984375;\n \t\t\t}\n\n \t\t},\n\n \t\tInOut: function (k) {\n\n \t\t\tif (k < 0.5) {\n \t\t\t\treturn TWEEN.Easing.Bounce.In(k * 2) * 0.5;\n \t\t\t}\n\n \t\t\treturn TWEEN.Easing.Bounce.Out(k * 2 - 1) * 0.5 + 0.5;\n\n \t\t}\n\n \t}\n\n };\n\n TWEEN.Interpolation = {\n\n \tLinear: function (v, k) {\n\n \t\tvar m = v.length - 1;\n \t\tvar f = m * k;\n \t\tvar i = Math.floor(f);\n \t\tvar fn = TWEEN.Interpolation.Utils.Linear;\n\n \t\tif (k < 0) {\n \t\t\treturn fn(v[0], v[1], f);\n \t\t}\n\n \t\tif (k > 1) {\n \t\t\treturn fn(v[m], v[m - 1], m - f);\n \t\t}\n\n \t\treturn fn(v[i], v[i + 1 > m ? m : i + 1], f - i);\n\n \t},\n\n \tBezier: function (v, k) {\n\n \t\tvar b = 0;\n \t\tvar n = v.length - 1;\n \t\tvar pw = Math.pow;\n \t\tvar bn = TWEEN.Interpolation.Utils.Bernstein;\n\n \t\tfor (var i = 0; i <= n; i++) {\n \t\t\tb += pw(1 - k, n - i) * pw(k, i) * v[i] * bn(n, i);\n \t\t}\n\n \t\treturn b;\n\n \t},\n\n \tCatmullRom: function (v, k) {\n\n \t\tvar m = v.length - 1;\n \t\tvar f = m * k;\n \t\tvar i = Math.floor(f);\n \t\tvar fn = TWEEN.Interpolation.Utils.CatmullRom;\n\n \t\tif (v[0] === v[m]) {\n\n \t\t\tif (k < 0) {\n \t\t\t\ti = Math.floor(f = m * (1 + k));\n \t\t\t}\n\n \t\t\treturn fn(v[(i - 1 + m) % m], v[i], v[(i + 1) % m], v[(i + 2) % m], f - i);\n\n \t\t} else {\n\n \t\t\tif (k < 0) {\n \t\t\t\treturn v[0] - (fn(v[0], v[0], v[1], v[1], -f) - v[0]);\n \t\t\t}\n\n \t\t\tif (k > 1) {\n \t\t\t\treturn v[m] - (fn(v[m], v[m], v[m - 1], v[m - 1], f - m) - v[m]);\n \t\t\t}\n\n \t\t\treturn fn(v[i ? i - 1 : 0], v[i], v[m < i + 1 ? m : i + 1], v[m < i + 2 ? m : i + 2], f - i);\n\n \t\t}\n\n \t},\n\n \tUtils: {\n\n \t\tLinear: function (p0, p1, t) {\n\n \t\t\treturn (p1 - p0) * t + p0;\n\n \t\t},\n\n \t\tBernstein: function (n, i) {\n\n \t\t\tvar fc = TWEEN.Interpolation.Utils.Factorial;\n\n \t\t\treturn fc(n) / fc(i) / fc(n - i);\n\n \t\t},\n\n \t\tFactorial: (function () {\n\n \t\t\tvar a = [1];\n\n \t\t\treturn function (n) {\n\n \t\t\t\tvar s = 1;\n\n \t\t\t\tif (a[n]) {\n \t\t\t\t\treturn a[n];\n \t\t\t\t}\n\n \t\t\t\tfor (var i = n; i > 1; i--) {\n \t\t\t\t\ts *= i;\n \t\t\t\t}\n\n \t\t\t\ta[n] = s;\n \t\t\t\treturn s;\n\n \t\t\t};\n\n \t\t})(),\n\n \t\tCatmullRom: function (p0, p1, p2, p3, t) {\n\n \t\t\tvar v0 = (p2 - p0) * 0.5;\n \t\t\tvar v1 = (p3 - p1) * 0.5;\n \t\t\tvar t2 = t * t;\n \t\t\tvar t3 = t * t2;\n\n \t\t\treturn (2 * p1 - 2 * p2 + v0 + v1) * t3 + (- 3 * p1 + 3 * p2 - 2 * v0 - v1) * t2 + v0 * t + p1;\n\n \t\t}\n\n \t}\n\n };\n TWEEN.version = version;\n\n class AnimationModule {\n constructor(kbViewer, scene) {\n this.kbViewer = kbViewer;\n this.scene = scene;\n this.activeAnimations = new Set();\n }\n get animationsAreRunning() {\n return this.activeAnimations.size > 0;\n }\n queue(a) {\n if (!a.node)\n throw 'node required';\n if (a.toValue == null)\n throw 'value required';\n a = this.applyDefaults(a);\n return new Promise(resolve => {\n const origValue = a.node[a.prop];\n const origValueType = origValue === null || origValue === void 0 ? void 0 : origValue.$type;\n const val = this.valueIn(origValue);\n const toVal = this.valueIn(a.toValue);\n const tween = new TWEEN.Tween(val)\n .to(toVal, a.duration)\n .delay(a.delay)\n .repeat(a.iterations < 0 ? Infinity : a.iterations - 1)\n .easing(this.getTweenEasingFn(a.easing))\n .onUpdate(() => {\n a.node[a.prop] = this.valueOut(val, origValueType);\n })\n .onComplete(() => {\n this.activeAnimations.delete(activeAnim);\n resolve();\n })\n .start();\n const activeAnim = { animation: a, tween };\n this.activeAnimations.add(activeAnim);\n this.kbViewer.animationWarningsNeedsUpdate.next(this.activeAnimations.size);\n });\n }\n stop(name) {\n if (name != null) {\n const anims = [...this.activeAnimations.values()];\n for (const t of anims.filter(a => name.isEqual(a.animation.name))) {\n t.tween.stop();\n this.activeAnimations.delete(t);\n }\n }\n }\n stopAll() {\n for (const a of this.activeAnimations) {\n a.tween.stop();\n }\n this.activeAnimations.clear();\n }\n /** should be called by the render loop */\n update() {\n TWEEN.update(); //update all animations\n }\n applyDefaults(a) {\n return {\n duration: 1000,\n easing: exports.eEasing.linear,\n iterations: 1,\n delay: 0,\n ...a,\n };\n }\n valueIn(val) {\n if (val instanceof exports.KbVector) {\n return val.toObject();\n }\n else if (val instanceof exports.KbColor) {\n return val.toObject();\n }\n else if (val instanceof exports.KbQuaternion) {\n return val.toObject();\n }\n else {\n return { value: val };\n }\n }\n valueOut(val, $type) {\n if (!$type) {\n return val.value;\n }\n else {\n if ($type == Token.KbVector.key) {\n return exports.KbVector.From(val);\n }\n else if ($type == Token.KbColor.key) {\n return exports.KbColor.From(val);\n }\n else if ($type == Token.KbQuaternion.key) {\n return exports.KbQuaternion.From(val);\n }\n else {\n throw 'unrecognized type';\n }\n }\n }\n getTweenEasingFn(e) {\n const split = e.splitWords().map(s => s.capitalize());\n const inOut = split.length == 3 ? split[1] + split[2] : split.length == 2 ? split[1] : 'None';\n return TWEEN.Easing[split[0]][inOut];\n }\n }\n\n let anchorIdProvider = 0;\n /**\n * An implementation of the anchor system for WebXR.\n * For further information see https://github.com/immersive-web/anchors/\n */\n class WebXRAnchorSystem extends WebXRAbstractFeature {\n /**\n * Set the reference space to use for anchor creation, when not using a hit test.\n * Will default to the session's reference space if not defined\n */\n set referenceSpaceForFrameAnchors(referenceSpace) {\n this._referenceSpaceForFrameAnchors = referenceSpace;\n }\n /**\n * constructs a new anchor system\n * @param _xrSessionManager an instance of WebXRSessionManager\n * @param _options configuration object for this feature\n */\n constructor(_xrSessionManager, _options = {}) {\n super(_xrSessionManager);\n this._options = _options;\n this._lastFrameDetected = new Set();\n this._trackedAnchors = [];\n this._futureAnchors = [];\n /**\n * Observers registered here will be executed when a new anchor was added to the session\n */\n this.onAnchorAddedObservable = new Observable$1();\n /**\n * Observers registered here will be executed when an anchor was removed from the session\n */\n this.onAnchorRemovedObservable = new Observable$1();\n /**\n * Observers registered here will be executed when an existing anchor updates\n * This can execute N times every frame\n */\n this.onAnchorUpdatedObservable = new Observable$1();\n this._tmpVector = new Vector3();\n this._tmpQuaternion = new Quaternion();\n this.xrNativeFeatureName = \"anchors\";\n }\n _populateTmpTransformation(position, rotationQuaternion) {\n this._tmpVector.copyFrom(position);\n this._tmpQuaternion.copyFrom(rotationQuaternion);\n if (!this._xrSessionManager.scene.useRightHandedSystem) {\n this._tmpVector.z *= -1;\n this._tmpQuaternion.z *= -1;\n this._tmpQuaternion.w *= -1;\n }\n return {\n position: this._tmpVector,\n rotationQuaternion: this._tmpQuaternion,\n };\n }\n /**\n * Create a new anchor point using a hit test result at a specific point in the scene\n * An anchor is tracked only after it is added to the trackerAnchors in xrFrame. The promise returned here does not yet guaranty that.\n * Use onAnchorAddedObservable to get newly added anchors if you require tracking guaranty.\n *\n * @param hitTestResult The hit test result to use for this anchor creation\n * @param position an optional position offset for this anchor\n * @param rotationQuaternion an optional rotation offset for this anchor\n * @returns A promise that fulfills when babylon has created the corresponding WebXRAnchor object and tracking has begun\n */\n async addAnchorPointUsingHitTestResultAsync(hitTestResult, position = new Vector3(), rotationQuaternion = new Quaternion()) {\n // convert to XR space (right handed) if needed\n this._populateTmpTransformation(position, rotationQuaternion);\n // the matrix that we'll use\n const m = new XRRigidTransform({ x: this._tmpVector.x, y: this._tmpVector.y, z: this._tmpVector.z }, { x: this._tmpQuaternion.x, y: this._tmpQuaternion.y, z: this._tmpQuaternion.z, w: this._tmpQuaternion.w });\n if (!hitTestResult.xrHitResult.createAnchor) {\n this.detach();\n throw new Error(\"Anchors not enabled in this environment/browser\");\n }\n else {\n try {\n const nativeAnchor = await hitTestResult.xrHitResult.createAnchor(m);\n return new Promise((resolve, reject) => {\n this._futureAnchors.push({\n nativeAnchor,\n resolved: false,\n submitted: true,\n xrTransformation: m,\n resolve,\n reject,\n });\n });\n }\n catch (error) {\n throw new Error(error);\n }\n }\n }\n /**\n * Add a new anchor at a specific position and rotation\n * This function will add a new anchor per default in the next available frame. Unless forced, the createAnchor function\n * will be called in the next xrFrame loop to make sure that the anchor can be created correctly.\n * An anchor is tracked only after it is added to the trackerAnchors in xrFrame. The promise returned here does not yet guaranty that.\n * Use onAnchorAddedObservable to get newly added anchors if you require tracking guaranty.\n *\n * @param position the position in which to add an anchor\n * @param rotationQuaternion an optional rotation for the anchor transformation\n * @param forceCreateInCurrentFrame force the creation of this anchor in the current frame. Must be called inside xrFrame loop!\n * @returns A promise that fulfills when babylon has created the corresponding WebXRAnchor object and tracking has begun\n */\n async addAnchorAtPositionAndRotationAsync(position, rotationQuaternion = new Quaternion(), forceCreateInCurrentFrame = false) {\n // convert to XR space (right handed) if needed\n this._populateTmpTransformation(position, rotationQuaternion);\n // the matrix that we'll use\n const xrTransformation = new XRRigidTransform({ x: this._tmpVector.x, y: this._tmpVector.y, z: this._tmpVector.z }, { x: this._tmpQuaternion.x, y: this._tmpQuaternion.y, z: this._tmpQuaternion.z, w: this._tmpQuaternion.w });\n const xrAnchor = forceCreateInCurrentFrame && this.attached && this._xrSessionManager.currentFrame\n ? await this._createAnchorAtTransformation(xrTransformation, this._xrSessionManager.currentFrame)\n : undefined;\n // add the transformation to the future anchors list\n return new Promise((resolve, reject) => {\n this._futureAnchors.push({\n nativeAnchor: xrAnchor,\n resolved: false,\n submitted: false,\n xrTransformation,\n resolve,\n reject,\n });\n });\n }\n /**\n * Get the list of anchors currently being tracked by the system\n */\n get anchors() {\n return this._trackedAnchors;\n }\n /**\n * detach this feature.\n * Will usually be called by the features manager\n *\n * @returns true if successful.\n */\n detach() {\n if (!super.detach()) {\n return false;\n }\n if (!this._options.doNotRemoveAnchorsOnSessionEnded) {\n while (this._trackedAnchors.length) {\n const toRemove = this._trackedAnchors.pop();\n if (toRemove) {\n try {\n // try to natively remove it as well\n toRemove.remove();\n }\n catch (e) {\n // no-op\n }\n // as the xr frame loop is removed, we need to notify manually\n this.onAnchorRemovedObservable.notifyObservers(toRemove);\n }\n }\n }\n return true;\n }\n /**\n * Dispose this feature and all of the resources attached\n */\n dispose() {\n this._futureAnchors.length = 0;\n super.dispose();\n this.onAnchorAddedObservable.clear();\n this.onAnchorRemovedObservable.clear();\n this.onAnchorUpdatedObservable.clear();\n }\n _onXRFrame(frame) {\n if (!this.attached || !frame) {\n return;\n }\n const trackedAnchors = frame.trackedAnchors;\n if (trackedAnchors) {\n const toRemove = this._trackedAnchors\n .filter((anchor) => !trackedAnchors.has(anchor.xrAnchor))\n .map((anchor) => {\n const index = this._trackedAnchors.indexOf(anchor);\n return index;\n });\n let idxTracker = 0;\n toRemove.forEach((index) => {\n const anchor = this._trackedAnchors.splice(index - idxTracker, 1)[0];\n this.onAnchorRemovedObservable.notifyObservers(anchor);\n idxTracker++;\n });\n // now check for new ones\n trackedAnchors.forEach((xrAnchor) => {\n if (!this._lastFrameDetected.has(xrAnchor)) {\n const newAnchor = {\n id: anchorIdProvider++,\n xrAnchor: xrAnchor,\n remove: () => xrAnchor.delete(),\n };\n const anchor = this._updateAnchorWithXRFrame(xrAnchor, newAnchor, frame);\n this._trackedAnchors.push(anchor);\n this.onAnchorAddedObservable.notifyObservers(anchor);\n // search for the future anchor promise that matches this\n const results = this._futureAnchors.filter((futureAnchor) => futureAnchor.nativeAnchor === xrAnchor);\n const result = results[0];\n if (result) {\n result.resolve(anchor);\n result.resolved = true;\n }\n }\n else {\n const index = this._findIndexInAnchorArray(xrAnchor);\n const anchor = this._trackedAnchors[index];\n try {\n // anchors update every frame\n this._updateAnchorWithXRFrame(xrAnchor, anchor, frame);\n if (anchor.attachedNode) {\n anchor.attachedNode.rotationQuaternion = anchor.attachedNode.rotationQuaternion || new Quaternion();\n anchor.transformationMatrix.decompose(anchor.attachedNode.scaling, anchor.attachedNode.rotationQuaternion, anchor.attachedNode.position);\n }\n this.onAnchorUpdatedObservable.notifyObservers(anchor);\n }\n catch (e) {\n Tools.Warn(`Anchor could not be updated`);\n }\n }\n });\n this._lastFrameDetected = trackedAnchors;\n }\n // process future anchors\n this._futureAnchors.forEach((futureAnchor) => {\n if (!futureAnchor.resolved && !futureAnchor.submitted) {\n this._createAnchorAtTransformation(futureAnchor.xrTransformation, frame).then((nativeAnchor) => {\n futureAnchor.nativeAnchor = nativeAnchor;\n }, (error) => {\n futureAnchor.resolved = true;\n futureAnchor.reject(error);\n });\n futureAnchor.submitted = true;\n }\n });\n }\n /**\n * avoiding using Array.find for global support.\n * @param xrAnchor the plane to find in the array\n */\n _findIndexInAnchorArray(xrAnchor) {\n for (let i = 0; i < this._trackedAnchors.length; ++i) {\n if (this._trackedAnchors[i].xrAnchor === xrAnchor) {\n return i;\n }\n }\n return -1;\n }\n _updateAnchorWithXRFrame(xrAnchor, anchor, xrFrame) {\n // matrix\n const pose = xrFrame.getPose(xrAnchor.anchorSpace, this._xrSessionManager.referenceSpace);\n if (pose) {\n const mat = anchor.transformationMatrix || new Matrix();\n Matrix.FromArrayToRef(pose.transform.matrix, 0, mat);\n if (!this._xrSessionManager.scene.useRightHandedSystem) {\n mat.toggleModelMatrixHandInPlace();\n }\n anchor.transformationMatrix = mat;\n if (!this._options.worldParentNode) ;\n else {\n mat.multiplyToRef(this._options.worldParentNode.getWorldMatrix(), mat);\n }\n }\n return anchor;\n }\n async _createAnchorAtTransformation(xrTransformation, xrFrame) {\n var _a;\n if (xrFrame.createAnchor) {\n try {\n return xrFrame.createAnchor(xrTransformation, (_a = this._referenceSpaceForFrameAnchors) !== null && _a !== void 0 ? _a : this._xrSessionManager.referenceSpace);\n }\n catch (error) {\n throw new Error(error);\n }\n }\n else {\n this.detach();\n throw new Error(\"Anchors are not enabled in your browser\");\n }\n }\n }\n /**\n * The module's name\n */\n WebXRAnchorSystem.Name = WebXRFeatureName.ANCHOR_SYSTEM;\n /**\n * The (Babylon) version of this module.\n * This is an integer representing the implementation version.\n * This number does not correspond to the WebXR specs version\n */\n WebXRAnchorSystem.Version = 1;\n // register the plugin\n WebXRFeaturesManager.AddWebXRFeature(WebXRAnchorSystem.Name, (xrSessionManager, options) => {\n return () => new WebXRAnchorSystem(xrSessionManager, options);\n }, WebXRAnchorSystem.Version);\n\n /**\n * The currently-working hit-test module.\n * Hit test (or Ray-casting) is used to interact with the real world.\n * For further information read here - https://github.com/immersive-web/hit-test\n *\n * Tested on chrome (mobile) 80.\n */\n class WebXRHitTest extends WebXRAbstractFeature {\n /**\n * Creates a new instance of the hit test feature\n * @param _xrSessionManager an instance of WebXRSessionManager\n * @param options options to use when constructing this feature\n */\n constructor(_xrSessionManager, \n /**\n * options to use when constructing this feature\n */\n options = {}) {\n super(_xrSessionManager);\n this.options = options;\n this._tmpMat = new Matrix();\n this._tmpPos = new Vector3();\n this._tmpQuat = new Quaternion();\n this._initHitTestSource = (referenceSpace) => {\n if (!referenceSpace) {\n return;\n }\n const offsetRay = new XRRay(this.options.offsetRay || {});\n const hitTestOptions = {\n space: this.options.useReferenceSpace ? referenceSpace : this._xrSessionManager.viewerReferenceSpace,\n offsetRay: offsetRay,\n };\n if (this.options.entityTypes) {\n hitTestOptions.entityTypes = this.options.entityTypes;\n }\n if (!hitTestOptions.space) {\n Tools.Warn(\"waiting for viewer reference space to initialize\");\n return;\n }\n this._xrSessionManager.session.requestHitTestSource(hitTestOptions).then((hitTestSource) => {\n if (this._xrHitTestSource) {\n this._xrHitTestSource.cancel();\n }\n this._xrHitTestSource = hitTestSource;\n });\n };\n /**\n * When set to true, each hit test will have its own position/rotation objects\n * When set to false, position and rotation objects will be reused for each hit test. It is expected that\n * the developers will clone them or copy them as they see fit.\n */\n this.autoCloneTransformation = false;\n /**\n * Triggered when new babylon (transformed) hit test results are available\n * Note - this will be called when results come back from the device. It can be an empty array!!\n */\n this.onHitTestResultObservable = new Observable$1();\n /**\n * Use this to temporarily pause hit test checks.\n */\n this.paused = false;\n this.xrNativeFeatureName = \"hit-test\";\n Tools.Warn(\"Hit test is an experimental and unstable feature.\");\n }\n /**\n * attach this feature\n * Will usually be called by the features manager\n *\n * @returns true if successful.\n */\n attach() {\n if (!super.attach()) {\n return false;\n }\n // Feature enabled, but not available\n if (!this._xrSessionManager.session.requestHitTestSource) {\n return false;\n }\n if (!this.options.disablePermanentHitTest) {\n if (this._xrSessionManager.referenceSpace) {\n this._initHitTestSource(this._xrSessionManager.referenceSpace);\n }\n this._xrSessionManager.onXRReferenceSpaceChanged.add(this._initHitTestSource);\n }\n if (this.options.enableTransientHitTest) {\n const offsetRay = new XRRay(this.options.transientOffsetRay || {});\n this._xrSessionManager.session.requestHitTestSourceForTransientInput({\n profile: this.options.transientHitTestProfile || \"generic-touchscreen\",\n offsetRay,\n entityTypes: this.options.entityTypes,\n }).then((hitSource) => {\n this._transientXrHitTestSource = hitSource;\n });\n }\n return true;\n }\n /**\n * detach this feature.\n * Will usually be called by the features manager\n *\n * @returns true if successful.\n */\n detach() {\n if (!super.detach()) {\n return false;\n }\n if (this._xrHitTestSource) {\n this._xrHitTestSource.cancel();\n this._xrHitTestSource = null;\n }\n this._xrSessionManager.onXRReferenceSpaceChanged.removeCallback(this._initHitTestSource);\n if (this._transientXrHitTestSource) {\n this._transientXrHitTestSource.cancel();\n this._transientXrHitTestSource = null;\n }\n return true;\n }\n /**\n * Dispose this feature and all of the resources attached\n */\n dispose() {\n super.dispose();\n this.onHitTestResultObservable.clear();\n }\n _onXRFrame(frame) {\n // make sure we do nothing if (async) not attached\n if (!this.attached || this.paused) {\n return;\n }\n if (this._xrHitTestSource) {\n const results = frame.getHitTestResults(this._xrHitTestSource);\n this._processWebXRHitTestResult(results);\n }\n if (this._transientXrHitTestSource) {\n const hitTestResultsPerInputSource = frame.getHitTestResultsForTransientInput(this._transientXrHitTestSource);\n hitTestResultsPerInputSource.forEach((resultsPerInputSource) => {\n this._processWebXRHitTestResult(resultsPerInputSource.results, resultsPerInputSource.inputSource);\n });\n }\n }\n _processWebXRHitTestResult(hitTestResults, inputSource) {\n const results = [];\n hitTestResults.forEach((hitTestResult) => {\n const pose = hitTestResult.getPose(this._xrSessionManager.referenceSpace);\n if (!pose) {\n return;\n }\n const pos = pose.transform.position;\n const quat = pose.transform.orientation;\n this._tmpPos.set(pos.x, pos.y, pos.z);\n this._tmpQuat.set(quat.x, quat.y, quat.z, quat.w);\n Matrix.FromFloat32ArrayToRefScaled(pose.transform.matrix, 0, 1, this._tmpMat);\n if (!this._xrSessionManager.scene.useRightHandedSystem) {\n this._tmpPos.z *= -1;\n this._tmpQuat.z *= -1;\n this._tmpQuat.w *= -1;\n this._tmpMat.toggleModelMatrixHandInPlace();\n }\n const result = {\n position: this.autoCloneTransformation ? this._tmpPos.clone() : this._tmpPos,\n rotationQuaternion: this.autoCloneTransformation ? this._tmpQuat.clone() : this._tmpQuat,\n transformationMatrix: this.autoCloneTransformation ? this._tmpMat.clone() : this._tmpMat,\n inputSource: inputSource,\n isTransient: !!inputSource,\n xrHitResult: hitTestResult,\n };\n results.push(result);\n });\n this.onHitTestResultObservable.notifyObservers(results);\n }\n }\n /**\n * The module's name\n */\n WebXRHitTest.Name = WebXRFeatureName.HIT_TEST;\n /**\n * The (Babylon) version of this module.\n * This is an integer representing the implementation version.\n * This number does not correspond to the WebXR specs version\n */\n WebXRHitTest.Version = 2;\n //register the plugin versions\n WebXRFeaturesManager.AddWebXRFeature(WebXRHitTest.Name, (xrSessionManager, options) => {\n return () => new WebXRHitTest(xrSessionManager, options);\n }, WebXRHitTest.Version, false);\n\n let planeIdProvider = 0;\n /**\n * The plane detector is used to detect planes in the real world when in AR\n * For more information see https://github.com/immersive-web/real-world-geometry/\n */\n class WebXRPlaneDetector extends WebXRAbstractFeature {\n /**\n * construct a new Plane Detector\n * @param _xrSessionManager an instance of xr Session manager\n * @param _options configuration to use when constructing this feature\n */\n constructor(_xrSessionManager, _options = {}) {\n super(_xrSessionManager);\n this._options = _options;\n this._detectedPlanes = [];\n this._enabled = false;\n this._lastFrameDetected = new Set();\n /**\n * Observers registered here will be executed when a new plane was added to the session\n */\n this.onPlaneAddedObservable = new Observable$1();\n /**\n * Observers registered here will be executed when a plane is no longer detected in the session\n */\n this.onPlaneRemovedObservable = new Observable$1();\n /**\n * Observers registered here will be executed when an existing plane updates (for example - expanded)\n * This can execute N times every frame\n */\n this.onPlaneUpdatedObservable = new Observable$1();\n this.xrNativeFeatureName = \"plane-detection\";\n if (this._xrSessionManager.session) {\n this._init();\n }\n else {\n this._xrSessionManager.onXRSessionInit.addOnce(() => {\n this._init();\n });\n }\n }\n /**\n * detach this feature.\n * Will usually be called by the features manager\n *\n * @returns true if successful.\n */\n detach() {\n if (!super.detach()) {\n return false;\n }\n if (!this._options.doNotRemovePlanesOnSessionEnded) {\n while (this._detectedPlanes.length) {\n const toRemove = this._detectedPlanes.pop();\n if (toRemove) {\n this.onPlaneRemovedObservable.notifyObservers(toRemove);\n }\n }\n }\n return true;\n }\n /**\n * Dispose this feature and all of the resources attached\n */\n dispose() {\n super.dispose();\n this.onPlaneAddedObservable.clear();\n this.onPlaneRemovedObservable.clear();\n this.onPlaneUpdatedObservable.clear();\n }\n /**\n * Check if the needed objects are defined.\n * This does not mean that the feature is enabled, but that the objects needed are well defined.\n */\n isCompatible() {\n return typeof XRPlane !== \"undefined\";\n }\n _onXRFrame(frame) {\n var _a;\n if (!this.attached || !this._enabled || !frame) {\n return;\n }\n const detectedPlanes = frame.detectedPlanes || ((_a = frame.worldInformation) === null || _a === void 0 ? void 0 : _a.detectedPlanes);\n if (detectedPlanes) {\n // remove all planes that are not currently detected in the frame\n for (let planeIdx = 0; planeIdx < this._detectedPlanes.length; planeIdx++) {\n const plane = this._detectedPlanes[planeIdx];\n if (!detectedPlanes.has(plane.xrPlane)) {\n this._detectedPlanes.splice(planeIdx--, 1);\n this.onPlaneRemovedObservable.notifyObservers(plane);\n }\n }\n // now check for new ones\n detectedPlanes.forEach((xrPlane) => {\n if (!this._lastFrameDetected.has(xrPlane)) {\n const newPlane = {\n id: planeIdProvider++,\n xrPlane: xrPlane,\n polygonDefinition: [],\n };\n const plane = this._updatePlaneWithXRPlane(xrPlane, newPlane, frame);\n this._detectedPlanes.push(plane);\n this.onPlaneAddedObservable.notifyObservers(plane);\n }\n else {\n // updated?\n if (xrPlane.lastChangedTime === this._xrSessionManager.currentTimestamp) {\n const index = this._findIndexInPlaneArray(xrPlane);\n const plane = this._detectedPlanes[index];\n this._updatePlaneWithXRPlane(xrPlane, plane, frame);\n this.onPlaneUpdatedObservable.notifyObservers(plane);\n }\n }\n });\n this._lastFrameDetected = detectedPlanes;\n }\n }\n _init() {\n const internalInit = () => {\n this._enabled = true;\n if (this._detectedPlanes.length) {\n this._detectedPlanes.length = 0;\n }\n };\n // Only supported by BabylonNative\n if (!!this._xrSessionManager.isNative && !!this._options.preferredDetectorOptions && !!this._xrSessionManager.session.trySetPreferredPlaneDetectorOptions) {\n this._xrSessionManager.session.trySetPreferredPlaneDetectorOptions(this._options.preferredDetectorOptions);\n }\n if (!this._xrSessionManager.session.updateWorldTrackingState) {\n internalInit();\n return;\n }\n this._xrSessionManager.session.updateWorldTrackingState({ planeDetectionState: { enabled: true } });\n internalInit();\n }\n _updatePlaneWithXRPlane(xrPlane, plane, xrFrame) {\n plane.polygonDefinition = xrPlane.polygon.map((xrPoint) => {\n const rightHandedSystem = this._xrSessionManager.scene.useRightHandedSystem ? 1 : -1;\n return new Vector3(xrPoint.x, xrPoint.y, xrPoint.z * rightHandedSystem);\n });\n // matrix\n const pose = xrFrame.getPose(xrPlane.planeSpace, this._xrSessionManager.referenceSpace);\n if (pose) {\n const mat = plane.transformationMatrix || new Matrix();\n Matrix.FromArrayToRef(pose.transform.matrix, 0, mat);\n if (!this._xrSessionManager.scene.useRightHandedSystem) {\n mat.toggleModelMatrixHandInPlace();\n }\n plane.transformationMatrix = mat;\n if (this._options.worldParentNode) {\n mat.multiplyToRef(this._options.worldParentNode.getWorldMatrix(), mat);\n }\n }\n return plane;\n }\n /**\n * avoiding using Array.find for global support.\n * @param xrPlane the plane to find in the array\n */\n _findIndexInPlaneArray(xrPlane) {\n for (let i = 0; i < this._detectedPlanes.length; ++i) {\n if (this._detectedPlanes[i].xrPlane === xrPlane) {\n return i;\n }\n }\n return -1;\n }\n }\n /**\n * The module's name\n */\n WebXRPlaneDetector.Name = WebXRFeatureName.PLANE_DETECTION;\n /**\n * The (Babylon) version of this module.\n * This is an integer representing the implementation version.\n * This number does not correspond to the WebXR specs version\n */\n WebXRPlaneDetector.Version = 1;\n //register the plugin\n WebXRFeaturesManager.AddWebXRFeature(WebXRPlaneDetector.Name, (xrSessionManager, options) => {\n return () => new WebXRPlaneDetector(xrSessionManager, options);\n }, WebXRPlaneDetector.Version);\n\n /**\n * Defines a target to use with MorphTargetManager\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/morphTargets\n */\n class MorphTarget {\n /**\n * Gets or sets the influence of this target (ie. its weight in the overall morphing)\n */\n get influence() {\n return this._influence;\n }\n set influence(influence) {\n if (this._influence === influence) {\n return;\n }\n const previous = this._influence;\n this._influence = influence;\n if (this.onInfluenceChanged.hasObservers()) {\n this.onInfluenceChanged.notifyObservers(previous === 0 || influence === 0);\n }\n }\n /**\n * Gets or sets the animation properties override\n */\n get animationPropertiesOverride() {\n if (!this._animationPropertiesOverride && this._scene) {\n return this._scene.animationPropertiesOverride;\n }\n return this._animationPropertiesOverride;\n }\n set animationPropertiesOverride(value) {\n this._animationPropertiesOverride = value;\n }\n /**\n * Creates a new MorphTarget\n * @param name defines the name of the target\n * @param influence defines the influence to use\n * @param scene defines the scene the morphtarget belongs to\n */\n constructor(\n /** defines the name of the target */\n name, influence = 0, scene = null) {\n this.name = name;\n /**\n * Gets or sets the list of animations\n */\n this.animations = new Array();\n this._positions = null;\n this._normals = null;\n this._tangents = null;\n this._uvs = null;\n this._uniqueId = 0;\n /**\n * Observable raised when the influence changes\n */\n this.onInfluenceChanged = new Observable$1();\n /** @internal */\n this._onDataLayoutChanged = new Observable$1();\n this._animationPropertiesOverride = null;\n this._scene = scene || EngineStore.LastCreatedScene;\n this.influence = influence;\n if (this._scene) {\n this._uniqueId = this._scene.getUniqueId();\n }\n }\n /**\n * Gets the unique ID of this manager\n */\n get uniqueId() {\n return this._uniqueId;\n }\n /**\n * Gets a boolean defining if the target contains position data\n */\n get hasPositions() {\n return !!this._positions;\n }\n /**\n * Gets a boolean defining if the target contains normal data\n */\n get hasNormals() {\n return !!this._normals;\n }\n /**\n * Gets a boolean defining if the target contains tangent data\n */\n get hasTangents() {\n return !!this._tangents;\n }\n /**\n * Gets a boolean defining if the target contains texture coordinates data\n */\n get hasUVs() {\n return !!this._uvs;\n }\n /**\n * Affects position data to this target\n * @param data defines the position data to use\n */\n setPositions(data) {\n const hadPositions = this.hasPositions;\n this._positions = data;\n if (hadPositions !== this.hasPositions) {\n this._onDataLayoutChanged.notifyObservers(undefined);\n }\n }\n /**\n * Gets the position data stored in this target\n * @returns a FloatArray containing the position data (or null if not present)\n */\n getPositions() {\n return this._positions;\n }\n /**\n * Affects normal data to this target\n * @param data defines the normal data to use\n */\n setNormals(data) {\n const hadNormals = this.hasNormals;\n this._normals = data;\n if (hadNormals !== this.hasNormals) {\n this._onDataLayoutChanged.notifyObservers(undefined);\n }\n }\n /**\n * Gets the normal data stored in this target\n * @returns a FloatArray containing the normal data (or null if not present)\n */\n getNormals() {\n return this._normals;\n }\n /**\n * Affects tangent data to this target\n * @param data defines the tangent data to use\n */\n setTangents(data) {\n const hadTangents = this.hasTangents;\n this._tangents = data;\n if (hadTangents !== this.hasTangents) {\n this._onDataLayoutChanged.notifyObservers(undefined);\n }\n }\n /**\n * Gets the tangent data stored in this target\n * @returns a FloatArray containing the tangent data (or null if not present)\n */\n getTangents() {\n return this._tangents;\n }\n /**\n * Affects texture coordinates data to this target\n * @param data defines the texture coordinates data to use\n */\n setUVs(data) {\n const hadUVs = this.hasUVs;\n this._uvs = data;\n if (hadUVs !== this.hasUVs) {\n this._onDataLayoutChanged.notifyObservers(undefined);\n }\n }\n /**\n * Gets the texture coordinates data stored in this target\n * @returns a FloatArray containing the texture coordinates data (or null if not present)\n */\n getUVs() {\n return this._uvs;\n }\n /**\n * Clone the current target\n * @returns a new MorphTarget\n */\n clone() {\n const newOne = SerializationHelper.Clone(() => new MorphTarget(this.name, this.influence, this._scene), this);\n newOne._positions = this._positions;\n newOne._normals = this._normals;\n newOne._tangents = this._tangents;\n newOne._uvs = this._uvs;\n return newOne;\n }\n /**\n * Serializes the current target into a Serialization object\n * @returns the serialized object\n */\n serialize() {\n const serializationObject = {};\n serializationObject.name = this.name;\n serializationObject.influence = this.influence;\n serializationObject.positions = Array.prototype.slice.call(this.getPositions());\n if (this.id != null) {\n serializationObject.id = this.id;\n }\n if (this.hasNormals) {\n serializationObject.normals = Array.prototype.slice.call(this.getNormals());\n }\n if (this.hasTangents) {\n serializationObject.tangents = Array.prototype.slice.call(this.getTangents());\n }\n if (this.hasUVs) {\n serializationObject.uvs = Array.prototype.slice.call(this.getUVs());\n }\n // Animations\n SerializationHelper.AppendSerializedAnimations(this, serializationObject);\n return serializationObject;\n }\n /**\n * Returns the string \"MorphTarget\"\n * @returns \"MorphTarget\"\n */\n getClassName() {\n return \"MorphTarget\";\n }\n // Statics\n /**\n * Creates a new target from serialized data\n * @param serializationObject defines the serialized data to use\n * @param scene defines the hosting scene\n * @returns a new MorphTarget\n */\n static Parse(serializationObject, scene) {\n const result = new MorphTarget(serializationObject.name, serializationObject.influence);\n result.setPositions(serializationObject.positions);\n if (serializationObject.id != null) {\n result.id = serializationObject.id;\n }\n if (serializationObject.normals) {\n result.setNormals(serializationObject.normals);\n }\n if (serializationObject.tangents) {\n result.setTangents(serializationObject.tangents);\n }\n if (serializationObject.uvs) {\n result.setUVs(serializationObject.uvs);\n }\n // Animations\n if (serializationObject.animations) {\n for (let animationIndex = 0; animationIndex < serializationObject.animations.length; animationIndex++) {\n const parsedAnimation = serializationObject.animations[animationIndex];\n const internalClass = GetClass(\"BABYLON.Animation\");\n if (internalClass) {\n result.animations.push(internalClass.Parse(parsedAnimation));\n }\n }\n if (serializationObject.autoAnimate && scene) {\n scene.beginAnimation(result, serializationObject.autoAnimateFrom, serializationObject.autoAnimateTo, serializationObject.autoAnimateLoop, serializationObject.autoAnimateSpeed || 1.0);\n }\n }\n return result;\n }\n /**\n * Creates a MorphTarget from mesh data\n * @param mesh defines the source mesh\n * @param name defines the name to use for the new target\n * @param influence defines the influence to attach to the target\n * @returns a new MorphTarget\n */\n static FromMesh(mesh, name, influence) {\n if (!name) {\n name = mesh.name;\n }\n const result = new MorphTarget(name, influence, mesh.getScene());\n result.setPositions(mesh.getVerticesData(VertexBuffer.PositionKind));\n if (mesh.isVerticesDataPresent(VertexBuffer.NormalKind)) {\n result.setNormals(mesh.getVerticesData(VertexBuffer.NormalKind));\n }\n if (mesh.isVerticesDataPresent(VertexBuffer.TangentKind)) {\n result.setTangents(mesh.getVerticesData(VertexBuffer.TangentKind));\n }\n if (mesh.isVerticesDataPresent(VertexBuffer.UVKind)) {\n result.setUVs(mesh.getVerticesData(VertexBuffer.UVKind));\n }\n return result;\n }\n }\n __decorate$1([\n serialize()\n ], MorphTarget.prototype, \"id\", void 0);\n\n /**\n * @internal\n */\n class _GLTFUtilities {\n /**\n * Creates a buffer view based on the supplied arguments\n * @param bufferIndex index value of the specified buffer\n * @param byteOffset byte offset value\n * @param byteLength byte length of the bufferView\n * @param byteStride byte distance between conequential elements\n * @param name name of the buffer view\n * @returns bufferView for glTF\n */\n static _CreateBufferView(bufferIndex, byteOffset, byteLength, byteStride, name) {\n const bufferview = { buffer: bufferIndex, byteLength: byteLength };\n if (byteOffset) {\n bufferview.byteOffset = byteOffset;\n }\n if (name) {\n bufferview.name = name;\n }\n if (byteStride) {\n bufferview.byteStride = byteStride;\n }\n return bufferview;\n }\n /**\n * Creates an accessor based on the supplied arguments\n * @param bufferviewIndex The index of the bufferview referenced by this accessor\n * @param name The name of the accessor\n * @param type The type of the accessor\n * @param componentType The datatype of components in the attribute\n * @param count The number of attributes referenced by this accessor\n * @param byteOffset The offset relative to the start of the bufferView in bytes\n * @param min Minimum value of each component in this attribute\n * @param max Maximum value of each component in this attribute\n * @returns accessor for glTF\n */\n static _CreateAccessor(bufferviewIndex, name, type, componentType, count, byteOffset, min, max) {\n const accessor = { name: name, bufferView: bufferviewIndex, componentType: componentType, count: count, type: type };\n if (min != null) {\n accessor.min = min;\n }\n if (max != null) {\n accessor.max = max;\n }\n if (byteOffset != null) {\n accessor.byteOffset = byteOffset;\n }\n return accessor;\n }\n /**\n * Calculates the minimum and maximum values of an array of position floats\n * @param positions Positions array of a mesh\n * @param vertexStart Starting vertex offset to calculate min and max values\n * @param vertexCount Number of vertices to check for min and max values\n * @returns min number array and max number array\n */\n static _CalculateMinMaxPositions(positions, vertexStart, vertexCount) {\n const min = [Infinity, Infinity, Infinity];\n const max = [-Infinity, -Infinity, -Infinity];\n const positionStrideSize = 3;\n let indexOffset;\n let position;\n let vector;\n if (vertexCount) {\n for (let i = vertexStart, length = vertexStart + vertexCount; i < length; ++i) {\n indexOffset = positionStrideSize * i;\n position = Vector3.FromArray(positions, indexOffset);\n vector = position.asArray();\n for (let j = 0; j < positionStrideSize; ++j) {\n const num = vector[j];\n if (num < min[j]) {\n min[j] = num;\n }\n if (num > max[j]) {\n max[j] = num;\n }\n ++indexOffset;\n }\n }\n }\n return { min, max };\n }\n static _NormalizeTangentFromRef(tangent) {\n const length = Math.sqrt(tangent.x * tangent.x + tangent.y * tangent.y + tangent.z * tangent.z);\n if (length > 0) {\n tangent.x /= length;\n tangent.y /= length;\n tangent.z /= length;\n }\n }\n static _GetDataAccessorElementCount(accessorType) {\n switch (accessorType) {\n case \"MAT2\" /* AccessorType.MAT2 */:\n return 4;\n case \"MAT3\" /* AccessorType.MAT3 */:\n return 9;\n case \"MAT4\" /* AccessorType.MAT4 */:\n return 16;\n case \"SCALAR\" /* AccessorType.SCALAR */:\n return 1;\n case \"VEC2\" /* AccessorType.VEC2 */:\n return 2;\n case \"VEC3\" /* AccessorType.VEC3 */:\n return 3;\n case \"VEC4\" /* AccessorType.VEC4 */:\n return 4;\n }\n }\n }\n\n /**\n * @internal\n * Enum for handling in tangent and out tangent.\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n var _TangentType;\n (function (_TangentType) {\n /**\n * Specifies that input tangents are used.\n */\n _TangentType[_TangentType[\"INTANGENT\"] = 0] = \"INTANGENT\";\n /**\n * Specifies that output tangents are used.\n */\n _TangentType[_TangentType[\"OUTTANGENT\"] = 1] = \"OUTTANGENT\";\n })(_TangentType || (_TangentType = {}));\n /**\n * @internal\n * Utility class for generating glTF animation data from BabylonJS.\n */\n class _GLTFAnimation {\n /**\n * Determine if a node is transformable - ie has properties it should be part of animation of transformation.\n * @param babylonNode the node to test\n * @returns true if can be animated, false otherwise. False if the parameter is null or undefined.\n */\n static _IsTransformable(babylonNode) {\n return babylonNode && (babylonNode instanceof TransformNode || babylonNode instanceof Camera || babylonNode instanceof Light);\n }\n /**\n * @ignore\n *\n * Creates glTF channel animation from BabylonJS animation.\n * @param babylonTransformNode - BabylonJS mesh.\n * @param animation - animation.\n * @param animationChannelTargetPath - The target animation channel.\n * @param useQuaternion - Specifies if quaternions are used.\n * @returns nullable IAnimationData\n */\n static _CreateNodeAnimation(babylonTransformNode, animation, animationChannelTargetPath, useQuaternion, animationSampleRate) {\n if (this._IsTransformable(babylonTransformNode)) {\n const inputs = [];\n const outputs = [];\n const keyFrames = animation.getKeys();\n const minMaxKeyFrames = _GLTFAnimation._CalculateMinMaxKeyFrames(keyFrames);\n const interpolationOrBake = _GLTFAnimation._DeduceInterpolation(keyFrames, animationChannelTargetPath, useQuaternion);\n const interpolation = interpolationOrBake.interpolationType;\n const shouldBakeAnimation = interpolationOrBake.shouldBakeAnimation;\n if (shouldBakeAnimation) {\n _GLTFAnimation._CreateBakedAnimation(babylonTransformNode, animation, animationChannelTargetPath, minMaxKeyFrames.min, minMaxKeyFrames.max, animation.framePerSecond, animationSampleRate, inputs, outputs, minMaxKeyFrames, useQuaternion);\n }\n else {\n if (interpolation === \"LINEAR\" /* AnimationSamplerInterpolation.LINEAR */ || interpolation === \"STEP\" /* AnimationSamplerInterpolation.STEP */) {\n _GLTFAnimation._CreateLinearOrStepAnimation(babylonTransformNode, animation, animationChannelTargetPath, inputs, outputs, useQuaternion);\n }\n else if (interpolation === \"CUBICSPLINE\" /* AnimationSamplerInterpolation.CUBICSPLINE */) {\n _GLTFAnimation._CreateCubicSplineAnimation(babylonTransformNode, animation, animationChannelTargetPath, inputs, outputs, useQuaternion);\n }\n else {\n _GLTFAnimation._CreateBakedAnimation(babylonTransformNode, animation, animationChannelTargetPath, minMaxKeyFrames.min, minMaxKeyFrames.max, animation.framePerSecond, animationSampleRate, inputs, outputs, minMaxKeyFrames, useQuaternion);\n }\n }\n if (inputs.length && outputs.length) {\n const result = {\n inputs: inputs,\n outputs: outputs,\n samplerInterpolation: interpolation,\n inputsMin: shouldBakeAnimation ? minMaxKeyFrames.min : Tools.FloatRound(minMaxKeyFrames.min / animation.framePerSecond),\n inputsMax: shouldBakeAnimation ? minMaxKeyFrames.max : Tools.FloatRound(minMaxKeyFrames.max / animation.framePerSecond),\n };\n return result;\n }\n }\n return null;\n }\n static _DeduceAnimationInfo(animation) {\n let animationChannelTargetPath = null;\n let dataAccessorType = \"VEC3\" /* AccessorType.VEC3 */;\n let useQuaternion = false;\n const property = animation.targetProperty.split(\".\");\n switch (property[0]) {\n case \"scaling\": {\n animationChannelTargetPath = \"scale\" /* AnimationChannelTargetPath.SCALE */;\n break;\n }\n case \"position\": {\n animationChannelTargetPath = \"translation\" /* AnimationChannelTargetPath.TRANSLATION */;\n break;\n }\n case \"rotation\": {\n dataAccessorType = \"VEC4\" /* AccessorType.VEC4 */;\n animationChannelTargetPath = \"rotation\" /* AnimationChannelTargetPath.ROTATION */;\n break;\n }\n case \"rotationQuaternion\": {\n dataAccessorType = \"VEC4\" /* AccessorType.VEC4 */;\n useQuaternion = true;\n animationChannelTargetPath = \"rotation\" /* AnimationChannelTargetPath.ROTATION */;\n break;\n }\n case \"influence\": {\n dataAccessorType = \"SCALAR\" /* AccessorType.SCALAR */;\n animationChannelTargetPath = \"weights\" /* AnimationChannelTargetPath.WEIGHTS */;\n break;\n }\n default: {\n Tools.Error(`Unsupported animatable property ${property[0]}`);\n }\n }\n if (animationChannelTargetPath) {\n return { animationChannelTargetPath: animationChannelTargetPath, dataAccessorType: dataAccessorType, useQuaternion: useQuaternion };\n }\n else {\n Tools.Error(\"animation channel target path and data accessor type could be deduced\");\n }\n return null;\n }\n /**\n * @ignore\n * Create node animations from the transform node animations\n * @param babylonNode\n * @param runtimeGLTFAnimation\n * @param idleGLTFAnimations\n * @param nodeMap\n * @param nodes\n * @param binaryWriter\n * @param bufferViews\n * @param accessors\n * @param animationSampleRate\n */\n static _CreateNodeAnimationFromNodeAnimations(babylonNode, runtimeGLTFAnimation, idleGLTFAnimations, nodeMap, nodes, binaryWriter, bufferViews, accessors, animationSampleRate, shouldExportAnimation) {\n let glTFAnimation;\n if (_GLTFAnimation._IsTransformable(babylonNode)) {\n if (babylonNode.animations) {\n for (const animation of babylonNode.animations) {\n if (shouldExportAnimation && !shouldExportAnimation(animation)) {\n continue;\n }\n const animationInfo = _GLTFAnimation._DeduceAnimationInfo(animation);\n if (animationInfo) {\n glTFAnimation = {\n name: animation.name,\n samplers: [],\n channels: [],\n };\n _GLTFAnimation._AddAnimation(`${animation.name}`, animation.hasRunningRuntimeAnimations ? runtimeGLTFAnimation : glTFAnimation, babylonNode, animation, animationInfo.dataAccessorType, animationInfo.animationChannelTargetPath, nodeMap, binaryWriter, bufferViews, accessors, animationInfo.useQuaternion, animationSampleRate);\n if (glTFAnimation.samplers.length && glTFAnimation.channels.length) {\n idleGLTFAnimations.push(glTFAnimation);\n }\n }\n }\n }\n }\n }\n /**\n * @ignore\n * Create individual morph animations from the mesh's morph target animation tracks\n * @param babylonNode\n * @param runtimeGLTFAnimation\n * @param idleGLTFAnimations\n * @param nodeMap\n * @param nodes\n * @param binaryWriter\n * @param bufferViews\n * @param accessors\n * @param animationSampleRate\n */\n static _CreateMorphTargetAnimationFromMorphTargetAnimations(babylonNode, runtimeGLTFAnimation, idleGLTFAnimations, nodeMap, nodes, binaryWriter, bufferViews, accessors, animationSampleRate, shouldExportAnimation) {\n let glTFAnimation;\n if (babylonNode instanceof Mesh) {\n const morphTargetManager = babylonNode.morphTargetManager;\n if (morphTargetManager) {\n for (let i = 0; i < morphTargetManager.numTargets; ++i) {\n const morphTarget = morphTargetManager.getTarget(i);\n for (const animation of morphTarget.animations) {\n if (shouldExportAnimation && !shouldExportAnimation(animation)) {\n continue;\n }\n const combinedAnimation = new Animation(`${animation.name}`, \"influence\", animation.framePerSecond, animation.dataType, animation.loopMode, animation.enableBlending);\n const combinedAnimationKeys = [];\n const animationKeys = animation.getKeys();\n for (let j = 0; j < animationKeys.length; ++j) {\n const animationKey = animationKeys[j];\n for (let k = 0; k < morphTargetManager.numTargets; ++k) {\n if (k == i) {\n combinedAnimationKeys.push(animationKey);\n }\n else {\n combinedAnimationKeys.push({ frame: animationKey.frame, value: 0 });\n }\n }\n }\n combinedAnimation.setKeys(combinedAnimationKeys);\n const animationInfo = _GLTFAnimation._DeduceAnimationInfo(combinedAnimation);\n if (animationInfo) {\n glTFAnimation = {\n name: combinedAnimation.name,\n samplers: [],\n channels: [],\n };\n _GLTFAnimation._AddAnimation(animation.name, animation.hasRunningRuntimeAnimations ? runtimeGLTFAnimation : glTFAnimation, babylonNode, combinedAnimation, animationInfo.dataAccessorType, animationInfo.animationChannelTargetPath, nodeMap, binaryWriter, bufferViews, accessors, animationInfo.useQuaternion, animationSampleRate, morphTargetManager.numTargets);\n if (glTFAnimation.samplers.length && glTFAnimation.channels.length) {\n idleGLTFAnimations.push(glTFAnimation);\n }\n }\n }\n }\n }\n }\n }\n /**\n * @ignore\n * Create node and morph animations from the animation groups\n * @param babylonScene\n * @param glTFAnimations\n * @param nodeMap\n * @param nodes\n * @param binaryWriter\n * @param bufferViews\n * @param accessors\n * @param animationSampleRate\n */\n static _CreateNodeAndMorphAnimationFromAnimationGroups(babylonScene, glTFAnimations, nodeMap, binaryWriter, bufferViews, accessors, animationSampleRate, shouldExportAnimation) {\n var _a;\n let glTFAnimation;\n if (babylonScene.animationGroups) {\n const animationGroups = babylonScene.animationGroups;\n for (const animationGroup of animationGroups) {\n const morphAnimations = new Map();\n const sampleAnimations = new Map();\n const morphAnimationMeshes = new Set();\n const animationGroupFrameDiff = animationGroup.to - animationGroup.from;\n glTFAnimation = {\n name: animationGroup.name,\n channels: [],\n samplers: [],\n };\n for (let i = 0; i < animationGroup.targetedAnimations.length; ++i) {\n const targetAnimation = animationGroup.targetedAnimations[i];\n const target = targetAnimation.target;\n const animation = targetAnimation.animation;\n if (shouldExportAnimation && !shouldExportAnimation(animation)) {\n continue;\n }\n if (this._IsTransformable(target) || (target.length === 1 && this._IsTransformable(target[0]))) {\n const animationInfo = _GLTFAnimation._DeduceAnimationInfo(targetAnimation.animation);\n if (animationInfo) {\n const babylonTransformNode = this._IsTransformable(target) ? target : this._IsTransformable(target[0]) ? target[0] : null;\n if (babylonTransformNode) {\n _GLTFAnimation._AddAnimation(`${animation.name}`, glTFAnimation, babylonTransformNode, animation, animationInfo.dataAccessorType, animationInfo.animationChannelTargetPath, nodeMap, binaryWriter, bufferViews, accessors, animationInfo.useQuaternion, animationSampleRate);\n }\n }\n }\n else if (target instanceof MorphTarget || (target.length === 1 && target[0] instanceof MorphTarget)) {\n const animationInfo = _GLTFAnimation._DeduceAnimationInfo(targetAnimation.animation);\n if (animationInfo) {\n const babylonMorphTarget = target instanceof MorphTarget ? target : target[0];\n if (babylonMorphTarget) {\n const babylonMorphTargetManager = babylonScene.morphTargetManagers.find((morphTargetManager) => {\n for (let j = 0; j < morphTargetManager.numTargets; ++j) {\n if (morphTargetManager.getTarget(j) === babylonMorphTarget) {\n return true;\n }\n }\n return false;\n });\n if (babylonMorphTargetManager) {\n const babylonMesh = babylonScene.meshes.find((mesh) => {\n return mesh.morphTargetManager === babylonMorphTargetManager;\n });\n if (babylonMesh) {\n if (!morphAnimations.has(babylonMesh)) {\n morphAnimations.set(babylonMesh, new Map());\n }\n (_a = morphAnimations.get(babylonMesh)) === null || _a === void 0 ? void 0 : _a.set(babylonMorphTarget, animation);\n morphAnimationMeshes.add(babylonMesh);\n sampleAnimations.set(babylonMesh, animation);\n }\n }\n }\n }\n }\n else ;\n }\n morphAnimationMeshes.forEach((mesh) => {\n const morphTargetManager = mesh.morphTargetManager;\n let combinedAnimationGroup = null;\n const animationKeys = [];\n const sampleAnimation = sampleAnimations.get(mesh);\n const sampleAnimationKeys = sampleAnimation.getKeys();\n const numAnimationKeys = sampleAnimationKeys.length;\n /*\n Due to how glTF expects morph target animation data to be formatted, we need to rearrange the individual morph target animation tracks,\n such that we have a single animation, where a given keyframe input value has successive output values for each morph target belonging to the manager.\n See: https://github.com/KhronosGroup/glTF/tree/master/specification/2.0#animations\n\n We do this via constructing a new Animation track, and interleaving the frames of each morph target animation track in the current Animation Group\n We reuse the Babylon Animation data structure for ease of handling export of cubic spline animation keys, and to reuse the\n existing _GLTFAnimation.AddAnimation codepath with minimal modification, however the constructed Babylon Animation is NOT intended for use in-engine.\n */\n for (let i = 0; i < numAnimationKeys; ++i) {\n for (let j = 0; j < morphTargetManager.numTargets; ++j) {\n const morphTarget = morphTargetManager.getTarget(j);\n const animationsByMorphTarget = morphAnimations.get(mesh);\n if (animationsByMorphTarget) {\n const morphTargetAnimation = animationsByMorphTarget.get(morphTarget);\n if (morphTargetAnimation) {\n if (!combinedAnimationGroup) {\n combinedAnimationGroup = new Animation(`${animationGroup.name}_${mesh.name}_MorphWeightAnimation`, \"influence\", morphTargetAnimation.framePerSecond, Animation.ANIMATIONTYPE_FLOAT, morphTargetAnimation.loopMode, morphTargetAnimation.enableBlending);\n }\n animationKeys.push(morphTargetAnimation.getKeys()[i]);\n }\n else {\n animationKeys.push({\n frame: animationGroup.from + (animationGroupFrameDiff / numAnimationKeys) * i,\n value: morphTarget.influence,\n inTangent: sampleAnimationKeys[0].inTangent ? 0 : undefined,\n outTangent: sampleAnimationKeys[0].outTangent ? 0 : undefined,\n });\n }\n }\n }\n }\n combinedAnimationGroup.setKeys(animationKeys);\n const animationInfo = _GLTFAnimation._DeduceAnimationInfo(combinedAnimationGroup);\n if (animationInfo) {\n _GLTFAnimation._AddAnimation(`${animationGroup.name}_${mesh.name}_MorphWeightAnimation`, glTFAnimation, mesh, combinedAnimationGroup, animationInfo.dataAccessorType, animationInfo.animationChannelTargetPath, nodeMap, binaryWriter, bufferViews, accessors, animationInfo.useQuaternion, animationSampleRate, morphTargetManager === null || morphTargetManager === void 0 ? void 0 : morphTargetManager.numTargets);\n }\n });\n if (glTFAnimation.channels.length && glTFAnimation.samplers.length) {\n glTFAnimations.push(glTFAnimation);\n }\n }\n }\n }\n static _AddAnimation(name, glTFAnimation, babylonTransformNode, animation, dataAccessorType, animationChannelTargetPath, nodeMap, binaryWriter, bufferViews, accessors, useQuaternion, animationSampleRate, morphAnimationChannels) {\n const animationData = _GLTFAnimation._CreateNodeAnimation(babylonTransformNode, animation, animationChannelTargetPath, useQuaternion, animationSampleRate);\n let bufferView;\n let accessor;\n let keyframeAccessorIndex;\n let dataAccessorIndex;\n let outputLength;\n let animationSampler;\n let animationChannel;\n if (animationData) {\n /*\n * Now that we have the glTF converted morph target animation data,\n * we can remove redundant input data so that we have n input frames,\n * and morphAnimationChannels * n output frames\n */\n if (morphAnimationChannels) {\n let index = 0;\n let currentInput = 0;\n const newInputs = [];\n while (animationData.inputs.length > 0) {\n currentInput = animationData.inputs.shift();\n if (index % morphAnimationChannels == 0) {\n newInputs.push(currentInput);\n }\n index++;\n }\n animationData.inputs = newInputs;\n }\n const nodeIndex = nodeMap[babylonTransformNode.uniqueId];\n // Creates buffer view and accessor for key frames.\n let byteLength = animationData.inputs.length * 4;\n bufferView = _GLTFUtilities._CreateBufferView(0, binaryWriter.getByteOffset(), byteLength, undefined, `${name} keyframe data view`);\n bufferViews.push(bufferView);\n animationData.inputs.forEach(function (input) {\n binaryWriter.setFloat32(input);\n });\n accessor = _GLTFUtilities._CreateAccessor(bufferViews.length - 1, `${name} keyframes`, \"SCALAR\" /* AccessorType.SCALAR */, 5126 /* AccessorComponentType.FLOAT */, animationData.inputs.length, null, [animationData.inputsMin], [animationData.inputsMax]);\n accessors.push(accessor);\n keyframeAccessorIndex = accessors.length - 1;\n // create bufferview and accessor for keyed values.\n outputLength = animationData.outputs.length;\n byteLength = _GLTFUtilities._GetDataAccessorElementCount(dataAccessorType) * 4 * animationData.outputs.length;\n // check for in and out tangents\n bufferView = _GLTFUtilities._CreateBufferView(0, binaryWriter.getByteOffset(), byteLength, undefined, `${name} data view`);\n bufferViews.push(bufferView);\n animationData.outputs.forEach(function (output) {\n output.forEach(function (entry) {\n binaryWriter.setFloat32(entry);\n });\n });\n accessor = _GLTFUtilities._CreateAccessor(bufferViews.length - 1, `${name} data`, dataAccessorType, 5126 /* AccessorComponentType.FLOAT */, outputLength, null, null, null);\n accessors.push(accessor);\n dataAccessorIndex = accessors.length - 1;\n // create sampler\n animationSampler = {\n interpolation: animationData.samplerInterpolation,\n input: keyframeAccessorIndex,\n output: dataAccessorIndex,\n };\n glTFAnimation.samplers.push(animationSampler);\n // create channel\n animationChannel = {\n sampler: glTFAnimation.samplers.length - 1,\n target: {\n node: nodeIndex,\n path: animationChannelTargetPath,\n },\n };\n glTFAnimation.channels.push(animationChannel);\n }\n }\n /**\n * Create a baked animation\n * @param babylonTransformNode BabylonJS mesh\n * @param animation BabylonJS animation corresponding to the BabylonJS mesh\n * @param animationChannelTargetPath animation target channel\n * @param minFrame minimum animation frame\n * @param maxFrame maximum animation frame\n * @param fps frames per second of the animation\n * @param sampleRate\n * @param inputs input key frames of the animation\n * @param outputs output key frame data of the animation\n * @param minMaxFrames\n * @param minMaxFrames.min\n * @param minMaxFrames.max\n * @param useQuaternion specifies if quaternions should be used\n */\n static _CreateBakedAnimation(babylonTransformNode, animation, animationChannelTargetPath, minFrame, maxFrame, fps, sampleRate, inputs, outputs, minMaxFrames, useQuaternion) {\n let value;\n const quaternionCache = Quaternion.Identity();\n let previousTime = null;\n let time;\n let maxUsedFrame = null;\n let currKeyFrame = null;\n let nextKeyFrame = null;\n let prevKeyFrame = null;\n let endFrame = null;\n minMaxFrames.min = Tools.FloatRound(minFrame / fps);\n const keyFrames = animation.getKeys();\n for (let i = 0, length = keyFrames.length; i < length; ++i) {\n endFrame = null;\n currKeyFrame = keyFrames[i];\n if (i + 1 < length) {\n nextKeyFrame = keyFrames[i + 1];\n if ((currKeyFrame.value.equals && currKeyFrame.value.equals(nextKeyFrame.value)) || currKeyFrame.value === nextKeyFrame.value) {\n if (i === 0) {\n // set the first frame to itself\n endFrame = currKeyFrame.frame;\n }\n else {\n continue;\n }\n }\n else {\n endFrame = nextKeyFrame.frame;\n }\n }\n else {\n // at the last key frame\n prevKeyFrame = keyFrames[i - 1];\n if ((currKeyFrame.value.equals && currKeyFrame.value.equals(prevKeyFrame.value)) || currKeyFrame.value === prevKeyFrame.value) {\n continue;\n }\n else {\n endFrame = maxFrame;\n }\n }\n if (endFrame) {\n for (let f = currKeyFrame.frame; f <= endFrame; f += sampleRate) {\n time = Tools.FloatRound(f / fps);\n if (time === previousTime) {\n continue;\n }\n previousTime = time;\n maxUsedFrame = time;\n const state = {\n key: 0,\n repeatCount: 0,\n loopMode: animation.loopMode,\n };\n value = animation._interpolate(f, state);\n _GLTFAnimation._SetInterpolatedValue(babylonTransformNode, value, time, animation, animationChannelTargetPath, quaternionCache, inputs, outputs, useQuaternion);\n }\n }\n }\n if (maxUsedFrame) {\n minMaxFrames.max = maxUsedFrame;\n }\n }\n static _ConvertFactorToVector3OrQuaternion(factor, babylonTransformNode, animation, animationChannelTargetPath, useQuaternion) {\n const basePositionRotationOrScale = _GLTFAnimation._GetBasePositionRotationOrScale(babylonTransformNode, animationChannelTargetPath, useQuaternion);\n // handles single component x, y, z or w component animation by using a base property and animating over a component.\n const property = animation.targetProperty.split(\".\");\n const componentName = property ? property[1] : \"\"; // x, y, z, or w component\n const value = useQuaternion ? Quaternion.FromArray(basePositionRotationOrScale).normalize() : Vector3.FromArray(basePositionRotationOrScale);\n switch (componentName) {\n case \"x\":\n case \"y\":\n case \"z\": {\n value[componentName] = factor;\n break;\n }\n case \"w\": {\n value.w = factor;\n break;\n }\n default: {\n Tools.Error(`glTFAnimation: Unsupported component name \"${componentName}\"!`);\n }\n }\n return value;\n }\n static _SetInterpolatedValue(babylonTransformNode, value, time, animation, animationChannelTargetPath, quaternionCache, inputs, outputs, useQuaternion) {\n let cacheValue;\n inputs.push(time);\n if (animationChannelTargetPath === \"weights\" /* AnimationChannelTargetPath.WEIGHTS */) {\n outputs.push([value]);\n return;\n }\n if (animation.dataType === Animation.ANIMATIONTYPE_FLOAT) {\n value = this._ConvertFactorToVector3OrQuaternion(value, babylonTransformNode, animation, animationChannelTargetPath, useQuaternion);\n }\n if (animationChannelTargetPath === \"rotation\" /* AnimationChannelTargetPath.ROTATION */) {\n if (useQuaternion) {\n quaternionCache = value;\n }\n else {\n cacheValue = value;\n Quaternion.RotationYawPitchRollToRef(cacheValue.y, cacheValue.x, cacheValue.z, quaternionCache);\n }\n outputs.push(quaternionCache.asArray());\n }\n else {\n // scaling and position animation\n cacheValue = value;\n outputs.push(cacheValue.asArray());\n }\n }\n /**\n * Creates linear animation from the animation key frames\n * @param babylonTransformNode BabylonJS mesh\n * @param animation BabylonJS animation\n * @param animationChannelTargetPath The target animation channel\n * @param inputs Array to store the key frame times\n * @param outputs Array to store the key frame data\n * @param useQuaternion Specifies if quaternions are used in the animation\n */\n static _CreateLinearOrStepAnimation(babylonTransformNode, animation, animationChannelTargetPath, inputs, outputs, useQuaternion) {\n for (const keyFrame of animation.getKeys()) {\n inputs.push(keyFrame.frame / animation.framePerSecond); // keyframes in seconds.\n _GLTFAnimation._AddKeyframeValue(keyFrame, animation, outputs, animationChannelTargetPath, babylonTransformNode, useQuaternion);\n }\n }\n /**\n * Creates cubic spline animation from the animation key frames\n * @param babylonTransformNode BabylonJS mesh\n * @param animation BabylonJS animation\n * @param animationChannelTargetPath The target animation channel\n * @param frameDelta The difference between the last and first frame of the animation\n * @param inputs Array to store the key frame times\n * @param outputs Array to store the key frame data\n * @param useQuaternion Specifies if quaternions are used in the animation\n */\n static _CreateCubicSplineAnimation(babylonTransformNode, animation, animationChannelTargetPath, inputs, outputs, useQuaternion) {\n animation.getKeys().forEach(function (keyFrame) {\n inputs.push(keyFrame.frame / animation.framePerSecond); // keyframes in seconds.\n _GLTFAnimation._AddSplineTangent(_TangentType.INTANGENT, outputs, animationChannelTargetPath, \"CUBICSPLINE\" /* AnimationSamplerInterpolation.CUBICSPLINE */, keyFrame, useQuaternion);\n _GLTFAnimation._AddKeyframeValue(keyFrame, animation, outputs, animationChannelTargetPath, babylonTransformNode, useQuaternion);\n _GLTFAnimation._AddSplineTangent(_TangentType.OUTTANGENT, outputs, animationChannelTargetPath, \"CUBICSPLINE\" /* AnimationSamplerInterpolation.CUBICSPLINE */, keyFrame, useQuaternion);\n });\n }\n static _GetBasePositionRotationOrScale(babylonTransformNode, animationChannelTargetPath, useQuaternion) {\n let basePositionRotationOrScale;\n if (animationChannelTargetPath === \"rotation\" /* AnimationChannelTargetPath.ROTATION */) {\n if (useQuaternion) {\n const q = babylonTransformNode.rotationQuaternion;\n basePositionRotationOrScale = (q !== null && q !== void 0 ? q : Quaternion.Identity()).asArray();\n }\n else {\n const r = babylonTransformNode.rotation;\n basePositionRotationOrScale = (r !== null && r !== void 0 ? r : Vector3.Zero()).asArray();\n }\n }\n else if (animationChannelTargetPath === \"translation\" /* AnimationChannelTargetPath.TRANSLATION */) {\n const p = babylonTransformNode.position;\n basePositionRotationOrScale = (p !== null && p !== void 0 ? p : Vector3.Zero()).asArray();\n }\n else {\n // scale\n const s = babylonTransformNode.scaling;\n basePositionRotationOrScale = (s !== null && s !== void 0 ? s : Vector3.One()).asArray();\n }\n return basePositionRotationOrScale;\n }\n /**\n * Adds a key frame value\n * @param keyFrame\n * @param animation\n * @param outputs\n * @param animationChannelTargetPath\n * @param babylonTransformNode\n * @param useQuaternion\n */\n static _AddKeyframeValue(keyFrame, animation, outputs, animationChannelTargetPath, babylonTransformNode, useQuaternion) {\n let newPositionRotationOrScale;\n const animationType = animation.dataType;\n if (animationType === Animation.ANIMATIONTYPE_VECTOR3) {\n let value = keyFrame.value.asArray();\n if (animationChannelTargetPath === \"rotation\" /* AnimationChannelTargetPath.ROTATION */) {\n const array = Vector3.FromArray(value);\n const rotationQuaternion = Quaternion.RotationYawPitchRoll(array.y, array.x, array.z);\n value = rotationQuaternion.asArray();\n }\n outputs.push(value); // scale vector.\n }\n else if (animationType === Animation.ANIMATIONTYPE_FLOAT) {\n if (animationChannelTargetPath === \"weights\" /* AnimationChannelTargetPath.WEIGHTS */) {\n outputs.push([keyFrame.value]);\n }\n else {\n // handles single component x, y, z or w component animation by using a base property and animating over a component.\n newPositionRotationOrScale = this._ConvertFactorToVector3OrQuaternion(keyFrame.value, babylonTransformNode, animation, animationChannelTargetPath, useQuaternion);\n if (newPositionRotationOrScale) {\n if (animationChannelTargetPath === \"rotation\" /* AnimationChannelTargetPath.ROTATION */) {\n const posRotScale = useQuaternion\n ? newPositionRotationOrScale\n : Quaternion.RotationYawPitchRoll(newPositionRotationOrScale.y, newPositionRotationOrScale.x, newPositionRotationOrScale.z).normalize();\n outputs.push(posRotScale.asArray());\n }\n outputs.push(newPositionRotationOrScale.asArray());\n }\n }\n }\n else if (animationType === Animation.ANIMATIONTYPE_QUATERNION) {\n outputs.push(keyFrame.value.normalize().asArray());\n }\n else {\n Tools.Error(\"glTFAnimation: Unsupported key frame values for animation!\");\n }\n }\n /**\n * Determine the interpolation based on the key frames\n * @param keyFrames\n * @param animationChannelTargetPath\n * @param useQuaternion\n */\n static _DeduceInterpolation(keyFrames, animationChannelTargetPath, useQuaternion) {\n let interpolationType;\n let shouldBakeAnimation = false;\n let key;\n if (animationChannelTargetPath === \"rotation\" /* AnimationChannelTargetPath.ROTATION */ && !useQuaternion) {\n return { interpolationType: \"LINEAR\" /* AnimationSamplerInterpolation.LINEAR */, shouldBakeAnimation: true };\n }\n for (let i = 0, length = keyFrames.length; i < length; ++i) {\n key = keyFrames[i];\n if (key.inTangent || key.outTangent) {\n if (interpolationType) {\n if (interpolationType !== \"CUBICSPLINE\" /* AnimationSamplerInterpolation.CUBICSPLINE */) {\n interpolationType = \"LINEAR\" /* AnimationSamplerInterpolation.LINEAR */;\n shouldBakeAnimation = true;\n break;\n }\n }\n else {\n interpolationType = \"CUBICSPLINE\" /* AnimationSamplerInterpolation.CUBICSPLINE */;\n }\n }\n else {\n if (interpolationType) {\n if (interpolationType === \"CUBICSPLINE\" /* AnimationSamplerInterpolation.CUBICSPLINE */ ||\n (key.interpolation && key.interpolation === AnimationKeyInterpolation.STEP && interpolationType !== \"STEP\" /* AnimationSamplerInterpolation.STEP */)) {\n interpolationType = \"LINEAR\" /* AnimationSamplerInterpolation.LINEAR */;\n shouldBakeAnimation = true;\n break;\n }\n }\n else {\n if (key.interpolation && key.interpolation === AnimationKeyInterpolation.STEP) {\n interpolationType = \"STEP\" /* AnimationSamplerInterpolation.STEP */;\n }\n else {\n interpolationType = \"LINEAR\" /* AnimationSamplerInterpolation.LINEAR */;\n }\n }\n }\n }\n if (!interpolationType) {\n interpolationType = \"LINEAR\" /* AnimationSamplerInterpolation.LINEAR */;\n }\n return { interpolationType: interpolationType, shouldBakeAnimation: shouldBakeAnimation };\n }\n /**\n * Adds an input tangent or output tangent to the output data\n * If an input tangent or output tangent is missing, it uses the zero vector or zero quaternion\n * @param babylonTransformNode\n * @param tangentType Specifies which type of tangent to handle (inTangent or outTangent)\n * @param outputs The animation data by keyframe\n * @param animationChannelTargetPath The target animation channel\n * @param interpolation The interpolation type\n * @param keyFrame The key frame with the animation data\n * @param frameDelta Time difference between two frames used to scale the tangent by the frame delta\n * @param useQuaternion Specifies if quaternions are used\n */\n static _AddSplineTangent(tangentType, outputs, animationChannelTargetPath, interpolation, keyFrame, useQuaternion) {\n let tangent;\n const tangentValue = tangentType === _TangentType.INTANGENT ? keyFrame.inTangent : keyFrame.outTangent;\n if (interpolation === \"CUBICSPLINE\" /* AnimationSamplerInterpolation.CUBICSPLINE */) {\n if (animationChannelTargetPath === \"rotation\" /* AnimationChannelTargetPath.ROTATION */) {\n if (tangentValue) {\n if (useQuaternion) {\n tangent = tangentValue.asArray();\n }\n else {\n const array = tangentValue;\n tangent = Quaternion.RotationYawPitchRoll(array.y, array.x, array.z).asArray();\n }\n }\n else {\n tangent = [0, 0, 0, 0];\n }\n }\n else if (animationChannelTargetPath === \"weights\" /* AnimationChannelTargetPath.WEIGHTS */) {\n if (tangentValue) {\n tangent = [tangentValue];\n }\n else {\n tangent = [0];\n }\n }\n else {\n if (tangentValue) {\n tangent = tangentValue.asArray();\n }\n else {\n tangent = [0, 0, 0];\n }\n }\n outputs.push(tangent);\n }\n }\n /**\n * Get the minimum and maximum key frames' frame values\n * @param keyFrames animation key frames\n * @returns the minimum and maximum key frame value\n */\n static _CalculateMinMaxKeyFrames(keyFrames) {\n let min = Infinity;\n let max = -Infinity;\n keyFrames.forEach(function (keyFrame) {\n min = Math.min(min, keyFrame.frame);\n max = Math.max(max, keyFrame.frame);\n });\n return { min: min, max: max };\n }\n }\n\n /**\n * Class for holding and downloading glTF file data\n */\n class GLTFData {\n /**\n * Initializes the glTF file object\n */\n constructor() {\n this.glTFFiles = {};\n }\n /**\n * Downloads the glTF data as files based on their names and data\n */\n downloadFiles() {\n /**\n * Checks for a matching suffix at the end of a string (for ES5 and lower)\n * @param str Source string\n * @param suffix Suffix to search for in the source string\n * @returns Boolean indicating whether the suffix was found (true) or not (false)\n */\n function endsWith(str, suffix) {\n return str.indexOf(suffix, str.length - suffix.length) !== -1;\n }\n for (const key in this.glTFFiles) {\n const link = document.createElement(\"a\");\n document.body.appendChild(link);\n link.setAttribute(\"type\", \"hidden\");\n link.download = key;\n const blob = this.glTFFiles[key];\n let mimeType;\n if (endsWith(key, \".glb\")) {\n mimeType = { type: \"model/gltf-binary\" };\n }\n else if (endsWith(key, \".bin\")) {\n mimeType = { type: \"application/octet-stream\" };\n }\n else if (endsWith(key, \".gltf\")) {\n mimeType = { type: \"model/gltf+json\" };\n }\n else if (endsWith(key, \".jpeg\") || endsWith(key, \".jpg\")) {\n mimeType = { type: \"image/jpeg\" /* ImageMimeType.JPEG */ };\n }\n else if (endsWith(key, \".png\")) {\n mimeType = { type: \"image/png\" /* ImageMimeType.PNG */ };\n }\n link.href = window.URL.createObjectURL(new Blob([blob], mimeType));\n link.click();\n }\n }\n }\n\n ThinEngine.prototype.updateRawTexture = function (texture, data, format, invertY, compression = null, type = 0, useSRGBBuffer = false) {\n if (!texture) {\n return;\n }\n // Babylon's internalSizedFomat but gl's texImage2D internalFormat\n const internalSizedFomat = this._getRGBABufferInternalSizedFormat(type, format, useSRGBBuffer);\n // Babylon's internalFormat but gl's texImage2D format\n const internalFormat = this._getInternalFormat(format);\n const textureType = this._getWebGLTextureType(type);\n this._bindTextureDirectly(this._gl.TEXTURE_2D, texture, true);\n this._unpackFlipY(invertY === undefined ? true : invertY ? true : false);\n if (!this._doNotHandleContextLost) {\n texture._bufferView = data;\n texture.format = format;\n texture.type = type;\n texture.invertY = invertY;\n texture._compression = compression;\n }\n if (texture.width % 4 !== 0) {\n this._gl.pixelStorei(this._gl.UNPACK_ALIGNMENT, 1);\n }\n if (compression && data) {\n this._gl.compressedTexImage2D(this._gl.TEXTURE_2D, 0, this.getCaps().s3tc[compression], texture.width, texture.height, 0, data);\n }\n else {\n this._gl.texImage2D(this._gl.TEXTURE_2D, 0, internalSizedFomat, texture.width, texture.height, 0, internalFormat, textureType, data);\n }\n if (texture.generateMipMaps) {\n this._gl.generateMipmap(this._gl.TEXTURE_2D);\n }\n this._bindTextureDirectly(this._gl.TEXTURE_2D, null);\n // this.resetTextureCache();\n texture.isReady = true;\n };\n ThinEngine.prototype.createRawTexture = function (data, width, height, format, generateMipMaps, invertY, samplingMode, compression = null, type = 0, \n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n creationFlags = 0, useSRGBBuffer = false) {\n const texture = new InternalTexture(this, InternalTextureSource.Raw);\n texture.baseWidth = width;\n texture.baseHeight = height;\n texture.width = width;\n texture.height = height;\n texture.format = format;\n texture.generateMipMaps = generateMipMaps;\n texture.samplingMode = samplingMode;\n texture.invertY = invertY;\n texture._compression = compression;\n texture.type = type;\n texture._useSRGBBuffer = this._getUseSRGBBuffer(useSRGBBuffer, !generateMipMaps);\n if (!this._doNotHandleContextLost) {\n texture._bufferView = data;\n }\n this.updateRawTexture(texture, data, format, invertY, compression, type, texture._useSRGBBuffer);\n this._bindTextureDirectly(this._gl.TEXTURE_2D, texture, true);\n // Filters\n const filters = this._getSamplingParameters(samplingMode, generateMipMaps);\n this._gl.texParameteri(this._gl.TEXTURE_2D, this._gl.TEXTURE_MAG_FILTER, filters.mag);\n this._gl.texParameteri(this._gl.TEXTURE_2D, this._gl.TEXTURE_MIN_FILTER, filters.min);\n if (generateMipMaps) {\n this._gl.generateMipmap(this._gl.TEXTURE_2D);\n }\n this._bindTextureDirectly(this._gl.TEXTURE_2D, null);\n this._internalTexturesCache.push(texture);\n return texture;\n };\n ThinEngine.prototype.createRawCubeTexture = function (data, size, format, type, generateMipMaps, invertY, samplingMode, compression = null) {\n const gl = this._gl;\n const texture = new InternalTexture(this, InternalTextureSource.CubeRaw);\n texture.isCube = true;\n texture.format = format;\n texture.type = type;\n if (!this._doNotHandleContextLost) {\n texture._bufferViewArray = data;\n }\n const textureType = this._getWebGLTextureType(type);\n let internalFormat = this._getInternalFormat(format);\n if (internalFormat === gl.RGB) {\n internalFormat = gl.RGBA;\n }\n // Mipmap generation needs a sized internal format that is both color-renderable and texture-filterable\n if (textureType === gl.FLOAT && !this._caps.textureFloatLinearFiltering) {\n generateMipMaps = false;\n samplingMode = 1;\n Logger.Warn(\"Float texture filtering is not supported. Mipmap generation and sampling mode are forced to false and TEXTURE_NEAREST_SAMPLINGMODE, respectively.\");\n }\n else if (textureType === this._gl.HALF_FLOAT_OES && !this._caps.textureHalfFloatLinearFiltering) {\n generateMipMaps = false;\n samplingMode = 1;\n Logger.Warn(\"Half float texture filtering is not supported. Mipmap generation and sampling mode are forced to false and TEXTURE_NEAREST_SAMPLINGMODE, respectively.\");\n }\n else if (textureType === gl.FLOAT && !this._caps.textureFloatRender) {\n generateMipMaps = false;\n Logger.Warn(\"Render to float textures is not supported. Mipmap generation forced to false.\");\n }\n else if (textureType === gl.HALF_FLOAT && !this._caps.colorBufferFloat) {\n generateMipMaps = false;\n Logger.Warn(\"Render to half float textures is not supported. Mipmap generation forced to false.\");\n }\n const width = size;\n const height = width;\n texture.width = width;\n texture.height = height;\n texture.invertY = invertY;\n texture._compression = compression;\n // Double check on POT to generate Mips.\n const isPot = !this.needPOTTextures || (Tools.IsExponentOfTwo(texture.width) && Tools.IsExponentOfTwo(texture.height));\n if (!isPot) {\n generateMipMaps = false;\n }\n // Upload data if needed. The texture won't be ready until then.\n if (data) {\n this.updateRawCubeTexture(texture, data, format, type, invertY, compression);\n }\n else {\n const internalSizedFomat = this._getRGBABufferInternalSizedFormat(type);\n const level = 0;\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, texture, true);\n for (let faceIndex = 0; faceIndex < 6; faceIndex++) {\n if (compression) {\n gl.compressedTexImage2D(gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex, level, this.getCaps().s3tc[compression], texture.width, texture.height, 0, undefined);\n }\n else {\n gl.texImage2D(gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex, level, internalSizedFomat, texture.width, texture.height, 0, internalFormat, textureType, null);\n }\n }\n this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, null);\n }\n this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, texture, true);\n // Filters\n if (data && generateMipMaps) {\n this._gl.generateMipmap(this._gl.TEXTURE_CUBE_MAP);\n }\n const filters = this._getSamplingParameters(samplingMode, generateMipMaps);\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_MAG_FILTER, filters.mag);\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_MIN_FILTER, filters.min);\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, null);\n texture.generateMipMaps = generateMipMaps;\n texture.samplingMode = samplingMode;\n texture.isReady = true;\n return texture;\n };\n ThinEngine.prototype.updateRawCubeTexture = function (texture, data, format, type, invertY, compression = null, level = 0) {\n texture._bufferViewArray = data;\n texture.format = format;\n texture.type = type;\n texture.invertY = invertY;\n texture._compression = compression;\n const gl = this._gl;\n const textureType = this._getWebGLTextureType(type);\n let internalFormat = this._getInternalFormat(format);\n const internalSizedFomat = this._getRGBABufferInternalSizedFormat(type);\n let needConversion = false;\n if (internalFormat === gl.RGB) {\n internalFormat = gl.RGBA;\n needConversion = true;\n }\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, texture, true);\n this._unpackFlipY(invertY === undefined ? true : invertY ? true : false);\n if (texture.width % 4 !== 0) {\n gl.pixelStorei(gl.UNPACK_ALIGNMENT, 1);\n }\n // Data are known to be in +X +Y +Z -X -Y -Z\n for (let faceIndex = 0; faceIndex < 6; faceIndex++) {\n let faceData = data[faceIndex];\n if (compression) {\n gl.compressedTexImage2D(gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex, level, this.getCaps().s3tc[compression], texture.width, texture.height, 0, faceData);\n }\n else {\n if (needConversion) {\n faceData = _convertRGBtoRGBATextureData(faceData, texture.width, texture.height, type);\n }\n gl.texImage2D(gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex, level, internalSizedFomat, texture.width, texture.height, 0, internalFormat, textureType, faceData);\n }\n }\n const isPot = !this.needPOTTextures || (Tools.IsExponentOfTwo(texture.width) && Tools.IsExponentOfTwo(texture.height));\n if (isPot && texture.generateMipMaps && level === 0) {\n this._gl.generateMipmap(this._gl.TEXTURE_CUBE_MAP);\n }\n this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, null);\n // this.resetTextureCache();\n texture.isReady = true;\n };\n ThinEngine.prototype.createRawCubeTextureFromUrl = function (url, scene, size, format, type, noMipmap, callback, mipmapGenerator, onLoad = null, onError = null, samplingMode = 3, invertY = false) {\n const gl = this._gl;\n const texture = this.createRawCubeTexture(null, size, format, type, !noMipmap, invertY, samplingMode, null);\n scene === null || scene === void 0 ? void 0 : scene.addPendingData(texture);\n texture.url = url;\n texture.isReady = false;\n this._internalTexturesCache.push(texture);\n const onerror = (request, exception) => {\n scene === null || scene === void 0 ? void 0 : scene.removePendingData(texture);\n if (onError && request) {\n onError(request.status + \" \" + request.statusText, exception);\n }\n };\n const internalCallback = (data) => {\n const width = texture.width;\n const faceDataArrays = callback(data);\n if (!faceDataArrays) {\n return;\n }\n if (mipmapGenerator) {\n const textureType = this._getWebGLTextureType(type);\n let internalFormat = this._getInternalFormat(format);\n const internalSizedFomat = this._getRGBABufferInternalSizedFormat(type);\n let needConversion = false;\n if (internalFormat === gl.RGB) {\n internalFormat = gl.RGBA;\n needConversion = true;\n }\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, texture, true);\n this._unpackFlipY(false);\n const mipData = mipmapGenerator(faceDataArrays);\n for (let level = 0; level < mipData.length; level++) {\n const mipSize = width >> level;\n for (let faceIndex = 0; faceIndex < 6; faceIndex++) {\n let mipFaceData = mipData[level][faceIndex];\n if (needConversion) {\n mipFaceData = _convertRGBtoRGBATextureData(mipFaceData, mipSize, mipSize, type);\n }\n gl.texImage2D(faceIndex, level, internalSizedFomat, mipSize, mipSize, 0, internalFormat, textureType, mipFaceData);\n }\n }\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, null);\n }\n else {\n this.updateRawCubeTexture(texture, faceDataArrays, format, type, invertY);\n }\n texture.isReady = true;\n // this.resetTextureCache();\n scene === null || scene === void 0 ? void 0 : scene.removePendingData(texture);\n texture.onLoadedObservable.notifyObservers(texture);\n texture.onLoadedObservable.clear();\n if (onLoad) {\n onLoad();\n }\n };\n this._loadFile(url, (data) => {\n internalCallback(data);\n }, undefined, scene === null || scene === void 0 ? void 0 : scene.offlineProvider, true, onerror);\n return texture;\n };\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n function _convertRGBtoRGBATextureData(rgbData, width, height, textureType) {\n // Create new RGBA data container.\n let rgbaData;\n let val1 = 1;\n if (textureType === 1) {\n rgbaData = new Float32Array(width * height * 4);\n }\n else if (textureType === 2) {\n rgbaData = new Uint16Array(width * height * 4);\n val1 = 15360; // 15360 is the encoding of 1 in half float\n }\n else if (textureType === 7) {\n rgbaData = new Uint32Array(width * height * 4);\n }\n else {\n rgbaData = new Uint8Array(width * height * 4);\n }\n // Convert each pixel.\n for (let x = 0; x < width; x++) {\n for (let y = 0; y < height; y++) {\n const index = (y * width + x) * 3;\n const newIndex = (y * width + x) * 4;\n // Map Old Value to new value.\n rgbaData[newIndex + 0] = rgbData[index + 0];\n rgbaData[newIndex + 1] = rgbData[index + 1];\n rgbaData[newIndex + 2] = rgbData[index + 2];\n // Add fully opaque alpha channel.\n rgbaData[newIndex + 3] = val1;\n }\n }\n return rgbaData;\n }\n /**\n * Create a function for createRawTexture3D/createRawTexture2DArray\n * @param is3D true for TEXTURE_3D and false for TEXTURE_2D_ARRAY\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n function _makeCreateRawTextureFunction(is3D) {\n return function (data, width, height, depth, format, generateMipMaps, invertY, samplingMode, compression = null, textureType = 0) {\n const target = is3D ? this._gl.TEXTURE_3D : this._gl.TEXTURE_2D_ARRAY;\n const source = is3D ? InternalTextureSource.Raw3D : InternalTextureSource.Raw2DArray;\n const texture = new InternalTexture(this, source);\n texture.baseWidth = width;\n texture.baseHeight = height;\n texture.baseDepth = depth;\n texture.width = width;\n texture.height = height;\n texture.depth = depth;\n texture.format = format;\n texture.type = textureType;\n texture.generateMipMaps = generateMipMaps;\n texture.samplingMode = samplingMode;\n if (is3D) {\n texture.is3D = true;\n }\n else {\n texture.is2DArray = true;\n }\n if (!this._doNotHandleContextLost) {\n texture._bufferView = data;\n }\n if (is3D) {\n this.updateRawTexture3D(texture, data, format, invertY, compression, textureType);\n }\n else {\n this.updateRawTexture2DArray(texture, data, format, invertY, compression, textureType);\n }\n this._bindTextureDirectly(target, texture, true);\n // Filters\n const filters = this._getSamplingParameters(samplingMode, generateMipMaps);\n this._gl.texParameteri(target, this._gl.TEXTURE_MAG_FILTER, filters.mag);\n this._gl.texParameteri(target, this._gl.TEXTURE_MIN_FILTER, filters.min);\n if (generateMipMaps) {\n this._gl.generateMipmap(target);\n }\n this._bindTextureDirectly(target, null);\n this._internalTexturesCache.push(texture);\n return texture;\n };\n }\n ThinEngine.prototype.createRawTexture2DArray = _makeCreateRawTextureFunction(false);\n ThinEngine.prototype.createRawTexture3D = _makeCreateRawTextureFunction(true);\n /**\n * Create a function for updateRawTexture3D/updateRawTexture2DArray\n * @param is3D true for TEXTURE_3D and false for TEXTURE_2D_ARRAY\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n function _makeUpdateRawTextureFunction(is3D) {\n return function (texture, data, format, invertY, compression = null, textureType = 0) {\n const target = is3D ? this._gl.TEXTURE_3D : this._gl.TEXTURE_2D_ARRAY;\n const internalType = this._getWebGLTextureType(textureType);\n const internalFormat = this._getInternalFormat(format);\n const internalSizedFomat = this._getRGBABufferInternalSizedFormat(textureType, format);\n this._bindTextureDirectly(target, texture, true);\n this._unpackFlipY(invertY === undefined ? true : invertY ? true : false);\n if (!this._doNotHandleContextLost) {\n texture._bufferView = data;\n texture.format = format;\n texture.invertY = invertY;\n texture._compression = compression;\n }\n if (texture.width % 4 !== 0) {\n this._gl.pixelStorei(this._gl.UNPACK_ALIGNMENT, 1);\n }\n if (compression && data) {\n this._gl.compressedTexImage3D(target, 0, this.getCaps().s3tc[compression], texture.width, texture.height, texture.depth, 0, data);\n }\n else {\n this._gl.texImage3D(target, 0, internalSizedFomat, texture.width, texture.height, texture.depth, 0, internalFormat, internalType, data);\n }\n if (texture.generateMipMaps) {\n this._gl.generateMipmap(target);\n }\n this._bindTextureDirectly(target, null);\n // this.resetTextureCache();\n texture.isReady = true;\n };\n }\n ThinEngine.prototype.updateRawTexture2DArray = _makeUpdateRawTextureFunction(false);\n ThinEngine.prototype.updateRawTexture3D = _makeUpdateRawTextureFunction(true);\n\n /**\n * Raw texture can help creating a texture directly from an array of data.\n * This can be super useful if you either get the data from an uncompressed source or\n * if you wish to create your texture pixel by pixel.\n */\n class RawTexture extends Texture {\n /**\n * Instantiates a new RawTexture.\n * Raw texture can help creating a texture directly from an array of data.\n * This can be super useful if you either get the data from an uncompressed source or\n * if you wish to create your texture pixel by pixel.\n * @param data define the array of data to use to create the texture (null to create an empty texture)\n * @param width define the width of the texture\n * @param height define the height of the texture\n * @param format define the format of the data (RGB, RGBA... Engine.TEXTUREFORMAT_xxx)\n * @param sceneOrEngine defines the scene or engine the texture will belong to\n * @param generateMipMaps define whether mip maps should be generated or not\n * @param invertY define if the data should be flipped on Y when uploaded to the GPU\n * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE)\n * @param type define the format of the data (int, float... Engine.TEXTURETYPE_xxx)\n * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg)\n * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU).\n */\n constructor(data, width, height, \n /**\n * Define the format of the data (RGB, RGBA... Engine.TEXTUREFORMAT_xxx)\n */\n format, sceneOrEngine, generateMipMaps = true, invertY = false, samplingMode = 3, type = 0, creationFlags, useSRGBBuffer) {\n super(null, sceneOrEngine, !generateMipMaps, invertY, undefined, undefined, undefined, undefined, undefined, undefined, undefined, undefined, creationFlags);\n this.format = format;\n if (!this._engine) {\n return;\n }\n if (!this._engine._caps.textureFloatLinearFiltering && type === 1) {\n samplingMode = 1;\n }\n if (!this._engine._caps.textureHalfFloatLinearFiltering && type === 2) {\n samplingMode = 1;\n }\n this._texture = this._engine.createRawTexture(data, width, height, format, generateMipMaps, invertY, samplingMode, null, type, creationFlags !== null && creationFlags !== void 0 ? creationFlags : 0, useSRGBBuffer !== null && useSRGBBuffer !== void 0 ? useSRGBBuffer : false);\n this.wrapU = Texture.CLAMP_ADDRESSMODE;\n this.wrapV = Texture.CLAMP_ADDRESSMODE;\n }\n /**\n * Updates the texture underlying data.\n * @param data Define the new data of the texture\n */\n update(data) {\n this._getEngine().updateRawTexture(this._texture, data, this._texture.format, this._texture.invertY, null, this._texture.type, this._texture._useSRGBBuffer);\n }\n /**\n * Creates a luminance texture from some data.\n * @param data Define the texture data\n * @param width Define the width of the texture\n * @param height Define the height of the texture\n * @param sceneOrEngine defines the scene or engine the texture will belong to\n * @param generateMipMaps Define whether or not to create mip maps for the texture\n * @param invertY define if the data should be flipped on Y when uploaded to the GPU\n * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE)\n * @returns the luminance texture\n */\n static CreateLuminanceTexture(data, width, height, sceneOrEngine, generateMipMaps = true, invertY = false, samplingMode = 3) {\n return new RawTexture(data, width, height, 1, sceneOrEngine, generateMipMaps, invertY, samplingMode);\n }\n /**\n * Creates a luminance alpha texture from some data.\n * @param data Define the texture data\n * @param width Define the width of the texture\n * @param height Define the height of the texture\n * @param sceneOrEngine defines the scene or engine the texture will belong to\n * @param generateMipMaps Define whether or not to create mip maps for the texture\n * @param invertY define if the data should be flipped on Y when uploaded to the GPU\n * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE)\n * @returns the luminance alpha texture\n */\n static CreateLuminanceAlphaTexture(data, width, height, sceneOrEngine, generateMipMaps = true, invertY = false, samplingMode = 3) {\n return new RawTexture(data, width, height, 2, sceneOrEngine, generateMipMaps, invertY, samplingMode);\n }\n /**\n * Creates an alpha texture from some data.\n * @param data Define the texture data\n * @param width Define the width of the texture\n * @param height Define the height of the texture\n * @param sceneOrEngine defines the scene or engine the texture will belong to\n * @param generateMipMaps Define whether or not to create mip maps for the texture\n * @param invertY define if the data should be flipped on Y when uploaded to the GPU\n * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE)\n * @returns the alpha texture\n */\n static CreateAlphaTexture(data, width, height, sceneOrEngine, generateMipMaps = true, invertY = false, samplingMode = 3) {\n return new RawTexture(data, width, height, 0, sceneOrEngine, generateMipMaps, invertY, samplingMode);\n }\n /**\n * Creates a RGB texture from some data.\n * @param data Define the texture data\n * @param width Define the width of the texture\n * @param height Define the height of the texture\n * @param sceneOrEngine defines the scene or engine the texture will belong to\n * @param generateMipMaps Define whether or not to create mip maps for the texture\n * @param invertY define if the data should be flipped on Y when uploaded to the GPU\n * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE)\n * @param type define the format of the data (int, float... Engine.TEXTURETYPE_xxx)\n * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg)\n * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU).\n * @returns the RGB alpha texture\n */\n static CreateRGBTexture(data, width, height, sceneOrEngine, generateMipMaps = true, invertY = false, samplingMode = 3, type = 0, creationFlags = 0, useSRGBBuffer = false) {\n return new RawTexture(data, width, height, 4, sceneOrEngine, generateMipMaps, invertY, samplingMode, type, creationFlags, useSRGBBuffer);\n }\n /**\n * Creates a RGBA texture from some data.\n * @param data Define the texture data\n * @param width Define the width of the texture\n * @param height Define the height of the texture\n * @param sceneOrEngine defines the scene or engine the texture will belong to\n * @param generateMipMaps Define whether or not to create mip maps for the texture\n * @param invertY define if the data should be flipped on Y when uploaded to the GPU\n * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE)\n * @param type define the format of the data (int, float... Engine.TEXTURETYPE_xxx)\n * @param creationFlags specific flags to use when creating the texture (1 for storage textures, for eg)\n * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU).\n * @returns the RGBA texture\n */\n static CreateRGBATexture(data, width, height, sceneOrEngine, generateMipMaps = true, invertY = false, samplingMode = 3, type = 0, creationFlags = 0, useSRGBBuffer = false) {\n return new RawTexture(data, width, height, 5, sceneOrEngine, generateMipMaps, invertY, samplingMode, type, creationFlags, useSRGBBuffer);\n }\n /**\n * Creates a RGBA storage texture from some data.\n * @param data Define the texture data\n * @param width Define the width of the texture\n * @param height Define the height of the texture\n * @param sceneOrEngine defines the scene or engine the texture will belong to\n * @param generateMipMaps Define whether or not to create mip maps for the texture\n * @param invertY define if the data should be flipped on Y when uploaded to the GPU\n * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE)\n * @param type define the format of the data (int, float... Engine.TEXTURETYPE_xxx)\n * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU).\n * @returns the RGBA texture\n */\n static CreateRGBAStorageTexture(data, width, height, sceneOrEngine, generateMipMaps = true, invertY = false, samplingMode = 3, type = 0, useSRGBBuffer = false) {\n return new RawTexture(data, width, height, 5, sceneOrEngine, generateMipMaps, invertY, samplingMode, type, 1, useSRGBBuffer);\n }\n /**\n * Creates a R texture from some data.\n * @param data Define the texture data\n * @param width Define the width of the texture\n * @param height Define the height of the texture\n * @param sceneOrEngine defines the scene or engine the texture will belong to\n * @param generateMipMaps Define whether or not to create mip maps for the texture\n * @param invertY define if the data should be flipped on Y when uploaded to the GPU\n * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE)\n * @param type define the format of the data (int, float... Engine.TEXTURETYPE_xxx)\n * @returns the R texture\n */\n static CreateRTexture(data, width, height, sceneOrEngine, generateMipMaps = true, invertY = false, samplingMode = Texture.TRILINEAR_SAMPLINGMODE, type = 1) {\n return new RawTexture(data, width, height, 6, sceneOrEngine, generateMipMaps, invertY, samplingMode, type);\n }\n /**\n * Creates a R storage texture from some data.\n * @param data Define the texture data\n * @param width Define the width of the texture\n * @param height Define the height of the texture\n * @param sceneOrEngine defines the scene or engine the texture will belong to\n * @param generateMipMaps Define whether or not to create mip maps for the texture\n * @param invertY define if the data should be flipped on Y when uploaded to the GPU\n * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE)\n * @param type define the format of the data (int, float... Engine.TEXTURETYPE_xxx)\n * @returns the R texture\n */\n static CreateRStorageTexture(data, width, height, sceneOrEngine, generateMipMaps = true, invertY = false, samplingMode = Texture.TRILINEAR_SAMPLINGMODE, type = 1) {\n return new RawTexture(data, width, height, 6, sceneOrEngine, generateMipMaps, invertY, samplingMode, type, 1);\n }\n }\n\n /* eslint-disable @typescript-eslint/naming-convention */\n /** Defines the cross module used constants to avoid circular dependencies */\n class Constants {\n }\n /** Defines that alpha blending is disabled */\n Constants.ALPHA_DISABLE = 0;\n /** Defines that alpha blending is SRC ALPHA * SRC + DEST */\n Constants.ALPHA_ADD = 1;\n /** Defines that alpha blending is SRC ALPHA * SRC + (1 - SRC ALPHA) * DEST */\n Constants.ALPHA_COMBINE = 2;\n /** Defines that alpha blending is DEST - SRC * DEST */\n Constants.ALPHA_SUBTRACT = 3;\n /** Defines that alpha blending is SRC * DEST */\n Constants.ALPHA_MULTIPLY = 4;\n /** Defines that alpha blending is SRC ALPHA * SRC + (1 - SRC) * DEST */\n Constants.ALPHA_MAXIMIZED = 5;\n /** Defines that alpha blending is SRC + DEST */\n Constants.ALPHA_ONEONE = 6;\n /** Defines that alpha blending is SRC + (1 - SRC ALPHA) * DEST */\n Constants.ALPHA_PREMULTIPLIED = 7;\n /**\n * Defines that alpha blending is SRC + (1 - SRC ALPHA) * DEST\n * Alpha will be set to (1 - SRC ALPHA) * DEST ALPHA\n */\n Constants.ALPHA_PREMULTIPLIED_PORTERDUFF = 8;\n /** Defines that alpha blending is CST * SRC + (1 - CST) * DEST */\n Constants.ALPHA_INTERPOLATE = 9;\n /**\n * Defines that alpha blending is SRC + (1 - SRC) * DEST\n * Alpha will be set to SRC ALPHA + (1 - SRC ALPHA) * DEST ALPHA\n */\n Constants.ALPHA_SCREENMODE = 10;\n /**\n * Defines that alpha blending is SRC + DST\n * Alpha will be set to SRC ALPHA + DST ALPHA\n */\n Constants.ALPHA_ONEONE_ONEONE = 11;\n /**\n * Defines that alpha blending is SRC * DST ALPHA + DST\n * Alpha will be set to 0\n */\n Constants.ALPHA_ALPHATOCOLOR = 12;\n /**\n * Defines that alpha blending is SRC * (1 - DST) + DST * (1 - SRC)\n */\n Constants.ALPHA_REVERSEONEMINUS = 13;\n /**\n * Defines that alpha blending is SRC + DST * (1 - SRC ALPHA)\n * Alpha will be set to SRC ALPHA + DST ALPHA * (1 - SRC ALPHA)\n */\n Constants.ALPHA_SRC_DSTONEMINUSSRCALPHA = 14;\n /**\n * Defines that alpha blending is SRC + DST\n * Alpha will be set to SRC ALPHA\n */\n Constants.ALPHA_ONEONE_ONEZERO = 15;\n /**\n * Defines that alpha blending is SRC * (1 - DST) + DST * (1 - SRC)\n * Alpha will be set to DST ALPHA\n */\n Constants.ALPHA_EXCLUSION = 16;\n /**\n * Defines that alpha blending is SRC * SRC ALPHA + DST * (1 - SRC ALPHA)\n * Alpha will be set to SRC ALPHA + (1 - SRC ALPHA) * DST ALPHA\n */\n Constants.ALPHA_LAYER_ACCUMULATE = 17;\n /** Defines that alpha blending equation a SUM */\n Constants.ALPHA_EQUATION_ADD = 0;\n /** Defines that alpha blending equation a SUBSTRACTION */\n Constants.ALPHA_EQUATION_SUBSTRACT = 1;\n /** Defines that alpha blending equation a REVERSE SUBSTRACTION */\n Constants.ALPHA_EQUATION_REVERSE_SUBTRACT = 2;\n /** Defines that alpha blending equation a MAX operation */\n Constants.ALPHA_EQUATION_MAX = 3;\n /** Defines that alpha blending equation a MIN operation */\n Constants.ALPHA_EQUATION_MIN = 4;\n /**\n * Defines that alpha blending equation a DARKEN operation:\n * It takes the min of the src and sums the alpha channels.\n */\n Constants.ALPHA_EQUATION_DARKEN = 5;\n /** Defines that the resource is not delayed*/\n Constants.DELAYLOADSTATE_NONE = 0;\n /** Defines that the resource was successfully delay loaded */\n Constants.DELAYLOADSTATE_LOADED = 1;\n /** Defines that the resource is currently delay loading */\n Constants.DELAYLOADSTATE_LOADING = 2;\n /** Defines that the resource is delayed and has not started loading */\n Constants.DELAYLOADSTATE_NOTLOADED = 4;\n // Depth or Stencil test Constants.\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will never pass. i.e. Nothing will be drawn */\n Constants.NEVER = 0x0200;\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will always pass. i.e. Pixels will be drawn in the order they are drawn */\n Constants.ALWAYS = 0x0207;\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is less than the stored value */\n Constants.LESS = 0x0201;\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is equals to the stored value */\n Constants.EQUAL = 0x0202;\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is less than or equal to the stored value */\n Constants.LEQUAL = 0x0203;\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is greater than the stored value */\n Constants.GREATER = 0x0204;\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is greater than or equal to the stored value */\n Constants.GEQUAL = 0x0206;\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is not equal to the stored value */\n Constants.NOTEQUAL = 0x0205;\n // Stencil Actions Constants.\n /** Passed to stencilOperation to specify that stencil value must be kept */\n Constants.KEEP = 0x1e00;\n /** Passed to stencilOperation to specify that stencil value must be zero */\n Constants.ZERO = 0x0000;\n /** Passed to stencilOperation to specify that stencil value must be replaced */\n Constants.REPLACE = 0x1e01;\n /** Passed to stencilOperation to specify that stencil value must be incremented */\n Constants.INCR = 0x1e02;\n /** Passed to stencilOperation to specify that stencil value must be decremented */\n Constants.DECR = 0x1e03;\n /** Passed to stencilOperation to specify that stencil value must be inverted */\n Constants.INVERT = 0x150a;\n /** Passed to stencilOperation to specify that stencil value must be incremented with wrapping */\n Constants.INCR_WRAP = 0x8507;\n /** Passed to stencilOperation to specify that stencil value must be decremented with wrapping */\n Constants.DECR_WRAP = 0x8508;\n /** Texture is not repeating outside of 0..1 UVs */\n Constants.TEXTURE_CLAMP_ADDRESSMODE = 0;\n /** Texture is repeating outside of 0..1 UVs */\n Constants.TEXTURE_WRAP_ADDRESSMODE = 1;\n /** Texture is repeating and mirrored */\n Constants.TEXTURE_MIRROR_ADDRESSMODE = 2;\n /** Flag to create a storage texture */\n Constants.TEXTURE_CREATIONFLAG_STORAGE = 1;\n /** ALPHA */\n Constants.TEXTUREFORMAT_ALPHA = 0;\n /** LUMINANCE */\n Constants.TEXTUREFORMAT_LUMINANCE = 1;\n /** LUMINANCE_ALPHA */\n Constants.TEXTUREFORMAT_LUMINANCE_ALPHA = 2;\n /** RGB */\n Constants.TEXTUREFORMAT_RGB = 4;\n /** RGBA */\n Constants.TEXTUREFORMAT_RGBA = 5;\n /** RED */\n Constants.TEXTUREFORMAT_RED = 6;\n /** RED (2nd reference) */\n Constants.TEXTUREFORMAT_R = 6;\n /** RG */\n Constants.TEXTUREFORMAT_RG = 7;\n /** RED_INTEGER */\n Constants.TEXTUREFORMAT_RED_INTEGER = 8;\n /** RED_INTEGER (2nd reference) */\n Constants.TEXTUREFORMAT_R_INTEGER = 8;\n /** RG_INTEGER */\n Constants.TEXTUREFORMAT_RG_INTEGER = 9;\n /** RGB_INTEGER */\n Constants.TEXTUREFORMAT_RGB_INTEGER = 10;\n /** RGBA_INTEGER */\n Constants.TEXTUREFORMAT_RGBA_INTEGER = 11;\n /** BGRA */\n Constants.TEXTUREFORMAT_BGRA = 12;\n /** Depth 24 bits + Stencil 8 bits */\n Constants.TEXTUREFORMAT_DEPTH24_STENCIL8 = 13;\n /** Depth 32 bits float */\n Constants.TEXTUREFORMAT_DEPTH32_FLOAT = 14;\n /** Depth 16 bits */\n Constants.TEXTUREFORMAT_DEPTH16 = 15;\n /** Depth 24 bits */\n Constants.TEXTUREFORMAT_DEPTH24 = 16;\n /** Depth 24 bits unorm + Stencil 8 bits */\n Constants.TEXTUREFORMAT_DEPTH24UNORM_STENCIL8 = 17;\n /** Depth 32 bits float + Stencil 8 bits */\n Constants.TEXTUREFORMAT_DEPTH32FLOAT_STENCIL8 = 18;\n /** Stencil 8 bits */\n Constants.TEXTUREFORMAT_STENCIL8 = 19;\n /** Compressed BC7 */\n Constants.TEXTUREFORMAT_COMPRESSED_RGBA_BPTC_UNORM = 36492;\n /** Compressed BC7 (SRGB) */\n Constants.TEXTUREFORMAT_COMPRESSED_SRGB_ALPHA_BPTC_UNORM = 36493;\n /** Compressed BC6 unsigned float */\n Constants.TEXTUREFORMAT_COMPRESSED_RGB_BPTC_UNSIGNED_FLOAT = 36495;\n /** Compressed BC6 signed float */\n Constants.TEXTUREFORMAT_COMPRESSED_RGB_BPTC_SIGNED_FLOAT = 36494;\n /** Compressed BC3 */\n Constants.TEXTUREFORMAT_COMPRESSED_RGBA_S3TC_DXT5 = 33779;\n /** Compressed BC3 (SRGB) */\n Constants.TEXTUREFORMAT_COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT = 35919;\n /** Compressed BC2 */\n Constants.TEXTUREFORMAT_COMPRESSED_RGBA_S3TC_DXT3 = 33778;\n /** Compressed BC2 (SRGB) */\n Constants.TEXTUREFORMAT_COMPRESSED_SRGB_ALPHA_S3TC_DXT3_EXT = 35918;\n /** Compressed BC1 (RGBA) */\n Constants.TEXTUREFORMAT_COMPRESSED_RGBA_S3TC_DXT1 = 33777;\n /** Compressed BC1 (RGB) */\n Constants.TEXTUREFORMAT_COMPRESSED_RGB_S3TC_DXT1 = 33776;\n /** Compressed BC1 (SRGB+A) */\n Constants.TEXTUREFORMAT_COMPRESSED_SRGB_ALPHA_S3TC_DXT1_EXT = 35917;\n /** Compressed BC1 (SRGB) */\n Constants.TEXTUREFORMAT_COMPRESSED_SRGB_S3TC_DXT1_EXT = 35916;\n /** Compressed ASTC 4x4 */\n Constants.TEXTUREFORMAT_COMPRESSED_RGBA_ASTC_4x4 = 37808;\n /** Compressed ASTC 4x4 (SRGB) */\n Constants.TEXTUREFORMAT_COMPRESSED_SRGB8_ALPHA8_ASTC_4x4_KHR = 37840;\n /** Compressed ETC1 (RGB) */\n Constants.TEXTUREFORMAT_COMPRESSED_RGB_ETC1_WEBGL = 36196;\n /** Compressed ETC2 (RGB) */\n Constants.TEXTUREFORMAT_COMPRESSED_RGB8_ETC2 = 37492;\n /** Compressed ETC2 (SRGB) */\n Constants.TEXTUREFORMAT_COMPRESSED_SRGB8_ETC2 = 37493;\n /** Compressed ETC2 (RGB+A1) */\n Constants.TEXTUREFORMAT_COMPRESSED_RGB8_PUNCHTHROUGH_ALPHA1_ETC2 = 37494;\n /** Compressed ETC2 (SRGB+A1)*/\n Constants.TEXTUREFORMAT_COMPRESSED_SRGB8_PUNCHTHROUGH_ALPHA1_ETC2 = 37495;\n /** Compressed ETC2 (RGB+A) */\n Constants.TEXTUREFORMAT_COMPRESSED_RGBA8_ETC2_EAC = 37496;\n /** Compressed ETC2 (SRGB+1) */\n Constants.TEXTUREFORMAT_COMPRESSED_SRGB8_ALPHA8_ETC2_EAC = 37497;\n /** UNSIGNED_BYTE */\n Constants.TEXTURETYPE_UNSIGNED_BYTE = 0;\n /** UNSIGNED_BYTE (2nd reference) */\n Constants.TEXTURETYPE_UNSIGNED_INT = 0;\n /** FLOAT */\n Constants.TEXTURETYPE_FLOAT = 1;\n /** HALF_FLOAT */\n Constants.TEXTURETYPE_HALF_FLOAT = 2;\n /** BYTE */\n Constants.TEXTURETYPE_BYTE = 3;\n /** SHORT */\n Constants.TEXTURETYPE_SHORT = 4;\n /** UNSIGNED_SHORT */\n Constants.TEXTURETYPE_UNSIGNED_SHORT = 5;\n /** INT */\n Constants.TEXTURETYPE_INT = 6;\n /** UNSIGNED_INT */\n Constants.TEXTURETYPE_UNSIGNED_INTEGER = 7;\n /** UNSIGNED_SHORT_4_4_4_4 */\n Constants.TEXTURETYPE_UNSIGNED_SHORT_4_4_4_4 = 8;\n /** UNSIGNED_SHORT_5_5_5_1 */\n Constants.TEXTURETYPE_UNSIGNED_SHORT_5_5_5_1 = 9;\n /** UNSIGNED_SHORT_5_6_5 */\n Constants.TEXTURETYPE_UNSIGNED_SHORT_5_6_5 = 10;\n /** UNSIGNED_INT_2_10_10_10_REV */\n Constants.TEXTURETYPE_UNSIGNED_INT_2_10_10_10_REV = 11;\n /** UNSIGNED_INT_24_8 */\n Constants.TEXTURETYPE_UNSIGNED_INT_24_8 = 12;\n /** UNSIGNED_INT_10F_11F_11F_REV */\n Constants.TEXTURETYPE_UNSIGNED_INT_10F_11F_11F_REV = 13;\n /** UNSIGNED_INT_5_9_9_9_REV */\n Constants.TEXTURETYPE_UNSIGNED_INT_5_9_9_9_REV = 14;\n /** FLOAT_32_UNSIGNED_INT_24_8_REV */\n Constants.TEXTURETYPE_FLOAT_32_UNSIGNED_INT_24_8_REV = 15;\n /** UNDEFINED */\n Constants.TEXTURETYPE_UNDEFINED = 16;\n /** 2D Texture target*/\n Constants.TEXTURE_2D = 3553;\n /** 2D Array Texture target */\n Constants.TEXTURE_2D_ARRAY = 35866;\n /** Cube Map Texture target */\n Constants.TEXTURE_CUBE_MAP = 34067;\n /** Cube Map Array Texture target */\n Constants.TEXTURE_CUBE_MAP_ARRAY = 0xdeadbeef;\n /** 3D Texture target */\n Constants.TEXTURE_3D = 32879;\n /** nearest is mag = nearest and min = nearest and no mip */\n Constants.TEXTURE_NEAREST_SAMPLINGMODE = 1;\n /** mag = nearest and min = nearest and mip = none */\n Constants.TEXTURE_NEAREST_NEAREST = 1;\n /** Bilinear is mag = linear and min = linear and no mip */\n Constants.TEXTURE_BILINEAR_SAMPLINGMODE = 2;\n /** mag = linear and min = linear and mip = none */\n Constants.TEXTURE_LINEAR_LINEAR = 2;\n /** Trilinear is mag = linear and min = linear and mip = linear */\n Constants.TEXTURE_TRILINEAR_SAMPLINGMODE = 3;\n /** Trilinear is mag = linear and min = linear and mip = linear */\n Constants.TEXTURE_LINEAR_LINEAR_MIPLINEAR = 3;\n /** mag = nearest and min = nearest and mip = nearest */\n Constants.TEXTURE_NEAREST_NEAREST_MIPNEAREST = 4;\n /** mag = nearest and min = linear and mip = nearest */\n Constants.TEXTURE_NEAREST_LINEAR_MIPNEAREST = 5;\n /** mag = nearest and min = linear and mip = linear */\n Constants.TEXTURE_NEAREST_LINEAR_MIPLINEAR = 6;\n /** mag = nearest and min = linear and mip = none */\n Constants.TEXTURE_NEAREST_LINEAR = 7;\n /** nearest is mag = nearest and min = nearest and mip = linear */\n Constants.TEXTURE_NEAREST_NEAREST_MIPLINEAR = 8;\n /** mag = linear and min = nearest and mip = nearest */\n Constants.TEXTURE_LINEAR_NEAREST_MIPNEAREST = 9;\n /** mag = linear and min = nearest and mip = linear */\n Constants.TEXTURE_LINEAR_NEAREST_MIPLINEAR = 10;\n /** Bilinear is mag = linear and min = linear and mip = nearest */\n Constants.TEXTURE_LINEAR_LINEAR_MIPNEAREST = 11;\n /** mag = linear and min = nearest and mip = none */\n Constants.TEXTURE_LINEAR_NEAREST = 12;\n /** Explicit coordinates mode */\n Constants.TEXTURE_EXPLICIT_MODE = 0;\n /** Spherical coordinates mode */\n Constants.TEXTURE_SPHERICAL_MODE = 1;\n /** Planar coordinates mode */\n Constants.TEXTURE_PLANAR_MODE = 2;\n /** Cubic coordinates mode */\n Constants.TEXTURE_CUBIC_MODE = 3;\n /** Projection coordinates mode */\n Constants.TEXTURE_PROJECTION_MODE = 4;\n /** Skybox coordinates mode */\n Constants.TEXTURE_SKYBOX_MODE = 5;\n /** Inverse Cubic coordinates mode */\n Constants.TEXTURE_INVCUBIC_MODE = 6;\n /** Equirectangular coordinates mode */\n Constants.TEXTURE_EQUIRECTANGULAR_MODE = 7;\n /** Equirectangular Fixed coordinates mode */\n Constants.TEXTURE_FIXED_EQUIRECTANGULAR_MODE = 8;\n /** Equirectangular Fixed Mirrored coordinates mode */\n Constants.TEXTURE_FIXED_EQUIRECTANGULAR_MIRRORED_MODE = 9;\n /** Offline (baking) quality for texture filtering */\n Constants.TEXTURE_FILTERING_QUALITY_OFFLINE = 4096;\n /** High quality for texture filtering */\n Constants.TEXTURE_FILTERING_QUALITY_HIGH = 64;\n /** Medium quality for texture filtering */\n Constants.TEXTURE_FILTERING_QUALITY_MEDIUM = 16;\n /** Low quality for texture filtering */\n Constants.TEXTURE_FILTERING_QUALITY_LOW = 8;\n // Texture rescaling mode\n /** Defines that texture rescaling will use a floor to find the closer power of 2 size */\n Constants.SCALEMODE_FLOOR = 1;\n /** Defines that texture rescaling will look for the nearest power of 2 size */\n Constants.SCALEMODE_NEAREST = 2;\n /** Defines that texture rescaling will use a ceil to find the closer power of 2 size */\n Constants.SCALEMODE_CEILING = 3;\n /**\n * The dirty texture flag value\n */\n Constants.MATERIAL_TextureDirtyFlag = 1;\n /**\n * The dirty light flag value\n */\n Constants.MATERIAL_LightDirtyFlag = 2;\n /**\n * The dirty fresnel flag value\n */\n Constants.MATERIAL_FresnelDirtyFlag = 4;\n /**\n * The dirty attribute flag value\n */\n Constants.MATERIAL_AttributesDirtyFlag = 8;\n /**\n * The dirty misc flag value\n */\n Constants.MATERIAL_MiscDirtyFlag = 16;\n /**\n * The dirty prepass flag value\n */\n Constants.MATERIAL_PrePassDirtyFlag = 32;\n /**\n * The all dirty flag value\n */\n Constants.MATERIAL_AllDirtyFlag = 63;\n /**\n * Returns the triangle fill mode\n */\n Constants.MATERIAL_TriangleFillMode = 0;\n /**\n * Returns the wireframe mode\n */\n Constants.MATERIAL_WireFrameFillMode = 1;\n /**\n * Returns the point fill mode\n */\n Constants.MATERIAL_PointFillMode = 2;\n /**\n * Returns the point list draw mode\n */\n Constants.MATERIAL_PointListDrawMode = 3;\n /**\n * Returns the line list draw mode\n */\n Constants.MATERIAL_LineListDrawMode = 4;\n /**\n * Returns the line loop draw mode\n */\n Constants.MATERIAL_LineLoopDrawMode = 5;\n /**\n * Returns the line strip draw mode\n */\n Constants.MATERIAL_LineStripDrawMode = 6;\n /**\n * Returns the triangle strip draw mode\n */\n Constants.MATERIAL_TriangleStripDrawMode = 7;\n /**\n * Returns the triangle fan draw mode\n */\n Constants.MATERIAL_TriangleFanDrawMode = 8;\n /**\n * Stores the clock-wise side orientation\n */\n Constants.MATERIAL_ClockWiseSideOrientation = 0;\n /**\n * Stores the counter clock-wise side orientation\n */\n Constants.MATERIAL_CounterClockWiseSideOrientation = 1;\n /**\n * Nothing\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\n */\n Constants.ACTION_NothingTrigger = 0;\n /**\n * On pick\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\n */\n Constants.ACTION_OnPickTrigger = 1;\n /**\n * On left pick\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\n */\n Constants.ACTION_OnLeftPickTrigger = 2;\n /**\n * On right pick\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\n */\n Constants.ACTION_OnRightPickTrigger = 3;\n /**\n * On center pick\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\n */\n Constants.ACTION_OnCenterPickTrigger = 4;\n /**\n * On pick down\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\n */\n Constants.ACTION_OnPickDownTrigger = 5;\n /**\n * On double pick\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\n */\n Constants.ACTION_OnDoublePickTrigger = 6;\n /**\n * On pick up\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\n */\n Constants.ACTION_OnPickUpTrigger = 7;\n /**\n * On pick out.\n * This trigger will only be raised if you also declared a OnPickDown\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\n */\n Constants.ACTION_OnPickOutTrigger = 16;\n /**\n * On long press\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\n */\n Constants.ACTION_OnLongPressTrigger = 8;\n /**\n * On pointer over\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\n */\n Constants.ACTION_OnPointerOverTrigger = 9;\n /**\n * On pointer out\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\n */\n Constants.ACTION_OnPointerOutTrigger = 10;\n /**\n * On every frame\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\n */\n Constants.ACTION_OnEveryFrameTrigger = 11;\n /**\n * On intersection enter\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\n */\n Constants.ACTION_OnIntersectionEnterTrigger = 12;\n /**\n * On intersection exit\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\n */\n Constants.ACTION_OnIntersectionExitTrigger = 13;\n /**\n * On key down\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\n */\n Constants.ACTION_OnKeyDownTrigger = 14;\n /**\n * On key up\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\n */\n Constants.ACTION_OnKeyUpTrigger = 15;\n /**\n * Billboard mode will only apply to Y axis\n */\n Constants.PARTICLES_BILLBOARDMODE_Y = 2;\n /**\n * Billboard mode will apply to all axes\n */\n Constants.PARTICLES_BILLBOARDMODE_ALL = 7;\n /**\n * Special billboard mode where the particle will be biilboard to the camera but rotated to align with direction\n */\n Constants.PARTICLES_BILLBOARDMODE_STRETCHED = 8;\n /**\n * Special billboard mode where the particle will be billboard to the camera but only around the axis of the direction of particle emission\n */\n Constants.PARTICLES_BILLBOARDMODE_STRETCHED_LOCAL = 9;\n /** Default culling strategy : this is an exclusion test and it's the more accurate.\n * Test order :\n * Is the bounding sphere outside the frustum ?\n * If not, are the bounding box vertices outside the frustum ?\n * It not, then the cullable object is in the frustum.\n */\n Constants.MESHES_CULLINGSTRATEGY_STANDARD = 0;\n /** Culling strategy : Bounding Sphere Only.\n * This is an exclusion test. It's faster than the standard strategy because the bounding box is not tested.\n * It's also less accurate than the standard because some not visible objects can still be selected.\n * Test : is the bounding sphere outside the frustum ?\n * If not, then the cullable object is in the frustum.\n */\n Constants.MESHES_CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY = 1;\n /** Culling strategy : Optimistic Inclusion.\n * This in an inclusion test first, then the standard exclusion test.\n * This can be faster when a cullable object is expected to be almost always in the camera frustum.\n * This could also be a little slower than the standard test when the tested object center is not the frustum but one of its bounding box vertex is still inside.\n * Anyway, it's as accurate as the standard strategy.\n * Test :\n * Is the cullable object bounding sphere center in the frustum ?\n * If not, apply the default culling strategy.\n */\n Constants.MESHES_CULLINGSTRATEGY_OPTIMISTIC_INCLUSION = 2;\n /** Culling strategy : Optimistic Inclusion then Bounding Sphere Only.\n * This in an inclusion test first, then the bounding sphere only exclusion test.\n * This can be the fastest test when a cullable object is expected to be almost always in the camera frustum.\n * This could also be a little slower than the BoundingSphereOnly strategy when the tested object center is not in the frustum but its bounding sphere still intersects it.\n * It's less accurate than the standard strategy and as accurate as the BoundingSphereOnly strategy.\n * Test :\n * Is the cullable object bounding sphere center in the frustum ?\n * If not, apply the Bounding Sphere Only strategy. No Bounding Box is tested here.\n */\n Constants.MESHES_CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY = 3;\n /**\n * No logging while loading\n */\n Constants.SCENELOADER_NO_LOGGING = 0;\n /**\n * Minimal logging while loading\n */\n Constants.SCENELOADER_MINIMAL_LOGGING = 1;\n /**\n * Summary logging while loading\n */\n Constants.SCENELOADER_SUMMARY_LOGGING = 2;\n /**\n * Detailed logging while loading\n */\n Constants.SCENELOADER_DETAILED_LOGGING = 3;\n /**\n * Constant used to retrieve the irradiance texture index in the textures array in the prepass\n * using getIndex(Constants.PREPASS_IRRADIANCE_TEXTURE_TYPE)\n */\n Constants.PREPASS_IRRADIANCE_TEXTURE_TYPE = 0;\n /**\n * Constant used to retrieve the position texture index in the textures array in the prepass\n * using getIndex(Constants.PREPASS_POSITION_TEXTURE_INDEX)\n */\n Constants.PREPASS_POSITION_TEXTURE_TYPE = 1;\n /**\n * Constant used to retrieve the velocity texture index in the textures array in the prepass\n * using getIndex(Constants.PREPASS_VELOCITY_TEXTURE_INDEX)\n */\n Constants.PREPASS_VELOCITY_TEXTURE_TYPE = 2;\n /**\n * Constant used to retrieve the reflectivity texture index in the textures array in the prepass\n * using the getIndex(Constants.PREPASS_REFLECTIVITY_TEXTURE_TYPE)\n */\n Constants.PREPASS_REFLECTIVITY_TEXTURE_TYPE = 3;\n /**\n * Constant used to retrieve the lit color texture index in the textures array in the prepass\n * using the getIndex(Constants.PREPASS_COLOR_TEXTURE_TYPE)\n */\n Constants.PREPASS_COLOR_TEXTURE_TYPE = 4;\n /**\n * Constant used to retrieve depth index in the textures array in the prepass\n * using the getIndex(Constants.PREPASS_DEPTH_TEXTURE_TYPE)\n */\n Constants.PREPASS_DEPTH_TEXTURE_TYPE = 5;\n /**\n * Constant used to retrieve normal index in the textures array in the prepass\n * using the getIndex(Constants.PREPASS_NORMAL_TEXTURE_TYPE)\n */\n Constants.PREPASS_NORMAL_TEXTURE_TYPE = 6;\n /**\n * Constant used to retrieve albedo index in the textures array in the prepass\n * using the getIndex(Constants.PREPASS_ALBEDO_SQRT_TEXTURE_TYPE)\n */\n Constants.PREPASS_ALBEDO_SQRT_TEXTURE_TYPE = 7;\n /** Flag to create a readable buffer (the buffer can be the source of a copy) */\n Constants.BUFFER_CREATIONFLAG_READ = 1;\n /** Flag to create a writable buffer (the buffer can be the destination of a copy) */\n Constants.BUFFER_CREATIONFLAG_WRITE = 2;\n /** Flag to create a readable and writable buffer */\n Constants.BUFFER_CREATIONFLAG_READWRITE = 3;\n /** Flag to create a buffer suitable to be used as a uniform buffer */\n Constants.BUFFER_CREATIONFLAG_UNIFORM = 4;\n /** Flag to create a buffer suitable to be used as a vertex buffer */\n Constants.BUFFER_CREATIONFLAG_VERTEX = 8;\n /** Flag to create a buffer suitable to be used as an index buffer */\n Constants.BUFFER_CREATIONFLAG_INDEX = 16;\n /** Flag to create a buffer suitable to be used as a storage buffer */\n Constants.BUFFER_CREATIONFLAG_STORAGE = 32;\n /**\n * Prefixes used by the engine for sub mesh draw wrappers\n */\n /** @internal */\n Constants.RENDERPASS_MAIN = 0;\n /**\n * Constant used as key code for Alt key\n */\n Constants.INPUT_ALT_KEY = 18;\n /**\n * Constant used as key code for Ctrl key\n */\n Constants.INPUT_CTRL_KEY = 17;\n /**\n * Constant used as key code for Meta key (Left Win, Left Cmd)\n */\n Constants.INPUT_META_KEY1 = 91;\n /**\n * Constant used as key code for Meta key (Right Win)\n */\n Constants.INPUT_META_KEY2 = 92;\n /**\n * Constant used as key code for Meta key (Right Win, Right Cmd)\n */\n Constants.INPUT_META_KEY3 = 93;\n /**\n * Constant used as key code for Shift key\n */\n Constants.INPUT_SHIFT_KEY = 16;\n /** Standard snapshot rendering. In this mode, some form of dynamic behavior is possible (for eg, uniform buffers are still updated) */\n Constants.SNAPSHOTRENDERING_STANDARD = 0;\n /** Fast snapshot rendering. In this mode, everything is static and only some limited form of dynamic behaviour is possible */\n Constants.SNAPSHOTRENDERING_FAST = 1;\n /**\n * This is the default projection mode used by the cameras.\n * It helps recreating a feeling of perspective and better appreciate depth.\n * This is the best way to simulate real life cameras.\n */\n Constants.PERSPECTIVE_CAMERA = 0;\n /**\n * This helps creating camera with an orthographic mode.\n * Orthographic is commonly used in engineering as a means to produce object specifications that communicate dimensions unambiguously, each line of 1 unit length (cm, meter..whatever) will appear to have the same length everywhere on the drawing. This allows the drafter to dimension only a subset of lines and let the reader know that other lines of that length on the drawing are also that length in reality. Every parallel line in the drawing is also parallel in the object.\n */\n Constants.ORTHOGRAPHIC_CAMERA = 1;\n /**\n * This is the default FOV mode for perspective cameras.\n * This setting aligns the upper and lower bounds of the viewport to the upper and lower bounds of the camera frustum.\n */\n Constants.FOVMODE_VERTICAL_FIXED = 0;\n /**\n * This setting aligns the left and right bounds of the viewport to the left and right bounds of the camera frustum.\n */\n Constants.FOVMODE_HORIZONTAL_FIXED = 1;\n /**\n * This specifies there is no need for a camera rig.\n * Basically only one eye is rendered corresponding to the camera.\n */\n Constants.RIG_MODE_NONE = 0;\n /**\n * Simulates a camera Rig with one blue eye and one red eye.\n * This can be use with 3d blue and red glasses.\n */\n Constants.RIG_MODE_STEREOSCOPIC_ANAGLYPH = 10;\n /**\n * Defines that both eyes of the camera will be rendered side by side with a parallel target.\n */\n Constants.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL = 11;\n /**\n * Defines that both eyes of the camera will be rendered side by side with a none parallel target.\n */\n Constants.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED = 12;\n /**\n * Defines that both eyes of the camera will be rendered over under each other.\n */\n Constants.RIG_MODE_STEREOSCOPIC_OVERUNDER = 13;\n /**\n * Defines that both eyes of the camera will be rendered on successive lines interlaced for passive 3d monitors.\n */\n Constants.RIG_MODE_STEREOSCOPIC_INTERLACED = 14;\n /**\n * Defines that both eyes of the camera should be renderered in a VR mode (carbox).\n */\n Constants.RIG_MODE_VR = 20;\n /**\n * Defines that both eyes of the camera should be renderered in a VR mode (webVR).\n */\n Constants.RIG_MODE_WEBVR = 21;\n /**\n * Custom rig mode allowing rig cameras to be populated manually with any number of cameras\n */\n Constants.RIG_MODE_CUSTOM = 22;\n /**\n * Maximum number of uv sets supported\n */\n Constants.MAX_SUPPORTED_UV_SETS = 6;\n /**\n * GL constants\n */\n /** Alpha blend equation: ADD */\n Constants.GL_ALPHA_EQUATION_ADD = 0x8006;\n /** Alpha equation: MIN */\n Constants.GL_ALPHA_EQUATION_MIN = 0x8007;\n /** Alpha equation: MAX */\n Constants.GL_ALPHA_EQUATION_MAX = 0x8008;\n /** Alpha equation: SUBTRACT */\n Constants.GL_ALPHA_EQUATION_SUBTRACT = 0x800a;\n /** Alpha equation: REVERSE_SUBTRACT */\n Constants.GL_ALPHA_EQUATION_REVERSE_SUBTRACT = 0x800b;\n /** Alpha blend function: SRC */\n Constants.GL_ALPHA_FUNCTION_SRC = 0x0300;\n /** Alpha blend function: ONE_MINUS_SRC */\n Constants.GL_ALPHA_FUNCTION_ONE_MINUS_SRC_COLOR = 0x0301;\n /** Alpha blend function: SRC_ALPHA */\n Constants.GL_ALPHA_FUNCTION_SRC_ALPHA = 0x0302;\n /** Alpha blend function: ONE_MINUS_SRC_ALPHA */\n Constants.GL_ALPHA_FUNCTION_ONE_MINUS_SRC_ALPHA = 0x0303;\n /** Alpha blend function: DST_ALPHA */\n Constants.GL_ALPHA_FUNCTION_DST_ALPHA = 0x0304;\n /** Alpha blend function: ONE_MINUS_DST_ALPHA */\n Constants.GL_ALPHA_FUNCTION_ONE_MINUS_DST_ALPHA = 0x0305;\n /** Alpha blend function: ONE_MINUS_DST */\n Constants.GL_ALPHA_FUNCTION_DST_COLOR = 0x0306;\n /** Alpha blend function: ONE_MINUS_DST */\n Constants.GL_ALPHA_FUNCTION_ONE_MINUS_DST_COLOR = 0x0307;\n /** Alpha blend function: SRC_ALPHA_SATURATED */\n Constants.GL_ALPHA_FUNCTION_SRC_ALPHA_SATURATED = 0x0308;\n /** Alpha blend function: CONSTANT */\n Constants.GL_ALPHA_FUNCTION_CONSTANT_COLOR = 0x8001;\n /** Alpha blend function: ONE_MINUS_CONSTANT */\n Constants.GL_ALPHA_FUNCTION_ONE_MINUS_CONSTANT_COLOR = 0x8002;\n /** Alpha blend function: CONSTANT_ALPHA */\n Constants.GL_ALPHA_FUNCTION_CONSTANT_ALPHA = 0x8003;\n /** Alpha blend function: ONE_MINUS_CONSTANT_ALPHA */\n Constants.GL_ALPHA_FUNCTION_ONE_MINUS_CONSTANT_ALPHA = 0x8004;\n /** URL to the snippet server. Points to the public snippet server by default */\n Constants.SnippetUrl = \"https://snippet.babylonjs.com\";\n\n function getFileExtensionFromMimeType(mimeType) {\n switch (mimeType) {\n case \"image/jpeg\" /* ImageMimeType.JPEG */:\n return \".jpg\";\n case \"image/png\" /* ImageMimeType.PNG */:\n return \".png\";\n case \"image/webp\" /* ImageMimeType.WEBP */:\n return \".webp\";\n }\n }\n /**\n * Utility methods for working with glTF material conversion properties. This class should only be used internally\n * @internal\n */\n class _GLTFMaterialExporter {\n constructor(exporter) {\n /**\n * Mapping to store textures\n */\n this._textureMap = {};\n // Mapping of internal textures to images to avoid exporting duplicate images.\n this._internalTextureToImage = {};\n this._textureMap = {};\n this._exporter = exporter;\n }\n /**\n * Specifies if two colors are approximately equal in value\n * @param color1 first color to compare to\n * @param color2 second color to compare to\n * @param epsilon threshold value\n */\n static _FuzzyEquals(color1, color2, epsilon) {\n return Scalar.WithinEpsilon(color1.r, color2.r, epsilon) && Scalar.WithinEpsilon(color1.g, color2.g, epsilon) && Scalar.WithinEpsilon(color1.b, color2.b, epsilon);\n }\n /**\n * Gets the materials from a Babylon scene and converts them to glTF materials\n * @param exportMaterials\n * @param mimeType texture mime type\n * @param hasTextureCoords specifies if texture coordinates are present on the material\n */\n _convertMaterialsToGLTFAsync(exportMaterials, mimeType, hasTextureCoords) {\n const promises = [];\n exportMaterials.forEach((material) => {\n if (material.getClassName() === \"StandardMaterial\") {\n promises.push(this._convertStandardMaterialAsync(material, mimeType, hasTextureCoords));\n }\n else if (material.getClassName().indexOf(\"PBR\") !== -1) {\n promises.push(this._convertPBRMaterialAsync(material, mimeType, hasTextureCoords));\n }\n else {\n Tools.Warn(`Unsupported material type: ${material.name}`);\n }\n });\n return Promise.all(promises).then(() => {\n /* do nothing */\n });\n }\n /**\n * Makes a copy of the glTF material without the texture parameters\n * @param originalMaterial original glTF material\n * @returns glTF material without texture parameters\n */\n _stripTexturesFromMaterial(originalMaterial) {\n const newMaterial = {};\n if (originalMaterial) {\n newMaterial.name = originalMaterial.name;\n newMaterial.doubleSided = originalMaterial.doubleSided;\n newMaterial.alphaMode = originalMaterial.alphaMode;\n newMaterial.alphaCutoff = originalMaterial.alphaCutoff;\n newMaterial.emissiveFactor = originalMaterial.emissiveFactor;\n const originalPBRMetallicRoughness = originalMaterial.pbrMetallicRoughness;\n if (originalPBRMetallicRoughness) {\n newMaterial.pbrMetallicRoughness = {};\n newMaterial.pbrMetallicRoughness.baseColorFactor = originalPBRMetallicRoughness.baseColorFactor;\n newMaterial.pbrMetallicRoughness.metallicFactor = originalPBRMetallicRoughness.metallicFactor;\n newMaterial.pbrMetallicRoughness.roughnessFactor = originalPBRMetallicRoughness.roughnessFactor;\n }\n }\n return newMaterial;\n }\n /**\n * Specifies if the material has any texture parameters present\n * @param material glTF Material\n * @returns boolean specifying if texture parameters are present\n */\n _hasTexturesPresent(material) {\n var _a;\n if (material.emissiveTexture || material.normalTexture || material.occlusionTexture) {\n return true;\n }\n const pbrMat = material.pbrMetallicRoughness;\n if (pbrMat) {\n if (pbrMat.baseColorTexture || pbrMat.metallicRoughnessTexture) {\n return true;\n }\n }\n if (material.extensions) {\n for (const extension in material.extensions) {\n const extensionObject = material.extensions[extension];\n if (extensionObject) {\n return (_a = extensionObject.hasTextures) === null || _a === void 0 ? void 0 : _a.call(extensionObject);\n }\n }\n }\n return false;\n }\n _getTextureInfo(babylonTexture) {\n if (babylonTexture) {\n const textureUid = babylonTexture.uid;\n if (textureUid in this._textureMap) {\n return this._textureMap[textureUid];\n }\n }\n return null;\n }\n /**\n * Converts a Babylon StandardMaterial to a glTF Metallic Roughness Material\n * @param babylonStandardMaterial\n * @returns glTF Metallic Roughness Material representation\n */\n _convertToGLTFPBRMetallicRoughness(babylonStandardMaterial) {\n // Defines a cubic bezier curve where x is specular power and y is roughness\n const P0 = new Vector2(0, 1);\n const P1 = new Vector2(0, 0.1);\n const P2 = new Vector2(0, 0.1);\n const P3 = new Vector2(1300, 0.1);\n /**\n * Given the control points, solve for x based on a given t for a cubic bezier curve\n * @param t a value between 0 and 1\n * @param p0 first control point\n * @param p1 second control point\n * @param p2 third control point\n * @param p3 fourth control point\n * @returns number result of cubic bezier curve at the specified t\n */\n function cubicBezierCurve(t, p0, p1, p2, p3) {\n return (1 - t) * (1 - t) * (1 - t) * p0 + 3 * (1 - t) * (1 - t) * t * p1 + 3 * (1 - t) * t * t * p2 + t * t * t * p3;\n }\n /**\n * Evaluates a specified specular power value to determine the appropriate roughness value,\n * based on a pre-defined cubic bezier curve with specular on the abscissa axis (x-axis)\n * and roughness on the ordinant axis (y-axis)\n * @param specularPower specular power of standard material\n * @returns Number representing the roughness value\n */\n function solveForRoughness(specularPower) {\n // Given P0.x = 0, P1.x = 0, P2.x = 0\n // x = t * t * t * P3.x\n // t = (x / P3.x)^(1/3)\n const t = Math.pow(specularPower / P3.x, 0.333333);\n return cubicBezierCurve(t, P0.y, P1.y, P2.y, P3.y);\n }\n const diffuse = babylonStandardMaterial.diffuseColor.toLinearSpace(babylonStandardMaterial.getScene().getEngine().useExactSrgbConversions).scale(0.5);\n const opacity = babylonStandardMaterial.alpha;\n const specularPower = Scalar.Clamp(babylonStandardMaterial.specularPower, 0, _GLTFMaterialExporter._MaxSpecularPower);\n const roughness = solveForRoughness(specularPower);\n const glTFPbrMetallicRoughness = {\n baseColorFactor: [diffuse.r, diffuse.g, diffuse.b, opacity],\n metallicFactor: 0,\n roughnessFactor: roughness,\n };\n return glTFPbrMetallicRoughness;\n }\n /**\n * Computes the metallic factor\n * @param diffuse diffused value\n * @param specular specular value\n * @param oneMinusSpecularStrength one minus the specular strength\n * @returns metallic value\n */\n static _SolveMetallic(diffuse, specular, oneMinusSpecularStrength) {\n if (specular < this._DielectricSpecular.r) {\n this._DielectricSpecular;\n return 0;\n }\n const a = this._DielectricSpecular.r;\n const b = (diffuse * oneMinusSpecularStrength) / (1.0 - this._DielectricSpecular.r) + specular - 2.0 * this._DielectricSpecular.r;\n const c = this._DielectricSpecular.r - specular;\n const D = b * b - 4.0 * a * c;\n return Scalar.Clamp((-b + Math.sqrt(D)) / (2.0 * a), 0, 1);\n }\n /**\n * Sets the glTF alpha mode to a glTF material from the Babylon Material\n * @param glTFMaterial glTF material\n * @param babylonMaterial Babylon material\n */\n static _SetAlphaMode(glTFMaterial, babylonMaterial) {\n if (babylonMaterial.needAlphaBlending()) {\n glTFMaterial.alphaMode = \"BLEND\" /* MaterialAlphaMode.BLEND */;\n }\n else if (babylonMaterial.needAlphaTesting()) {\n glTFMaterial.alphaMode = \"MASK\" /* MaterialAlphaMode.MASK */;\n glTFMaterial.alphaCutoff = babylonMaterial.alphaCutOff;\n }\n }\n /**\n * Converts a Babylon Standard Material to a glTF Material\n * @param babylonStandardMaterial BJS Standard Material\n * @param mimeType mime type to use for the textures\n * @param hasTextureCoords specifies if texture coordinates are present on the submesh to determine if textures should be applied\n */\n _convertStandardMaterialAsync(babylonStandardMaterial, mimeType, hasTextureCoords) {\n const materialMap = this._exporter._materialMap;\n const materials = this._exporter._materials;\n const promises = [];\n const pbrMetallicRoughness = this._convertToGLTFPBRMetallicRoughness(babylonStandardMaterial);\n const material = { name: babylonStandardMaterial.name };\n if (babylonStandardMaterial.backFaceCulling != null && !babylonStandardMaterial.backFaceCulling) {\n if (!babylonStandardMaterial.twoSidedLighting) {\n Tools.Warn(babylonStandardMaterial.name + \": Back-face culling disabled and two-sided lighting disabled is not supported in glTF.\");\n }\n material.doubleSided = true;\n }\n if (hasTextureCoords) {\n if (babylonStandardMaterial.diffuseTexture) {\n promises.push(this._exportTextureAsync(babylonStandardMaterial.diffuseTexture, mimeType).then((textureInfo) => {\n if (textureInfo) {\n pbrMetallicRoughness.baseColorTexture = textureInfo;\n }\n }));\n }\n const bumpTexture = babylonStandardMaterial.bumpTexture;\n if (bumpTexture) {\n promises.push(this._exportTextureAsync(bumpTexture, mimeType).then((textureInfo) => {\n if (textureInfo) {\n material.normalTexture = textureInfo;\n if (bumpTexture.level !== 1) {\n material.normalTexture.scale = bumpTexture.level;\n }\n }\n }));\n }\n if (babylonStandardMaterial.emissiveTexture) {\n material.emissiveFactor = [1.0, 1.0, 1.0];\n promises.push(this._exportTextureAsync(babylonStandardMaterial.emissiveTexture, mimeType).then((textureInfo) => {\n if (textureInfo) {\n material.emissiveTexture = textureInfo;\n }\n }));\n }\n if (babylonStandardMaterial.ambientTexture) {\n promises.push(this._exportTextureAsync(babylonStandardMaterial.ambientTexture, mimeType).then((textureInfo) => {\n if (textureInfo) {\n const occlusionTexture = {\n index: textureInfo.index,\n };\n material.occlusionTexture = occlusionTexture;\n }\n }));\n }\n }\n if (babylonStandardMaterial.alpha < 1.0 || babylonStandardMaterial.opacityTexture) {\n if (babylonStandardMaterial.alphaMode === Constants.ALPHA_COMBINE) {\n material.alphaMode = \"BLEND\" /* MaterialAlphaMode.BLEND */;\n }\n else {\n Tools.Warn(babylonStandardMaterial.name + \": glTF 2.0 does not support alpha mode: \" + babylonStandardMaterial.alphaMode.toString());\n }\n }\n if (babylonStandardMaterial.emissiveColor && !_GLTFMaterialExporter._FuzzyEquals(babylonStandardMaterial.emissiveColor, Color3.Black(), _GLTFMaterialExporter._Epsilon)) {\n material.emissiveFactor = babylonStandardMaterial.emissiveColor.asArray();\n }\n material.pbrMetallicRoughness = pbrMetallicRoughness;\n _GLTFMaterialExporter._SetAlphaMode(material, babylonStandardMaterial);\n materials.push(material);\n materialMap[babylonStandardMaterial.uniqueId] = materials.length - 1;\n return this._finishMaterial(promises, material, babylonStandardMaterial, mimeType);\n }\n _finishMaterial(promises, glTFMaterial, babylonMaterial, mimeType) {\n return Promise.all(promises).then(() => {\n const textures = this._exporter._extensionsPostExportMaterialAdditionalTextures(\"exportMaterial\", glTFMaterial, babylonMaterial);\n let tasks = null;\n for (const texture of textures) {\n if (!tasks) {\n tasks = [];\n }\n tasks.push(this._exportTextureAsync(texture, mimeType));\n }\n if (!tasks) {\n tasks = [Promise.resolve(null)];\n }\n return Promise.all(tasks).then(() => {\n const extensionWork = this._exporter._extensionsPostExportMaterialAsync(\"exportMaterial\", glTFMaterial, babylonMaterial);\n if (!extensionWork) {\n return glTFMaterial;\n }\n return extensionWork.then(() => glTFMaterial);\n });\n });\n }\n /**\n * Converts an image typed array buffer to a base64 image\n * @param buffer typed array buffer\n * @param width width of the image\n * @param height height of the image\n * @param mimeType mimetype of the image\n * @returns base64 image string\n */\n async _getImageDataAsync(buffer, width, height, mimeType) {\n const textureType = Constants.TEXTURETYPE_UNSIGNED_INT;\n const hostingScene = this._exporter._babylonScene;\n const engine = hostingScene.getEngine();\n // Create a temporary texture with the texture buffer data\n const tempTexture = engine.createRawTexture(buffer, width, height, Constants.TEXTUREFORMAT_RGBA, false, true, Texture.NEAREST_SAMPLINGMODE, null, textureType);\n await TextureTools.ApplyPostProcess(\"pass\", tempTexture, hostingScene, textureType, Constants.TEXTURE_NEAREST_SAMPLINGMODE, Constants.TEXTUREFORMAT_RGBA);\n const data = await engine._readTexturePixels(tempTexture, width, height);\n return (await DumpTools.DumpDataAsync(width, height, data, mimeType, undefined, true, true));\n }\n /**\n * Generates a white texture based on the specified width and height\n * @param width width of the texture in pixels\n * @param height height of the texture in pixels\n * @param scene babylonjs scene\n * @returns white texture\n */\n _createWhiteTexture(width, height, scene) {\n const data = new Uint8Array(width * height * 4);\n for (let i = 0; i < data.length; i = i + 4) {\n data[i] = data[i + 1] = data[i + 2] = data[i + 3] = 0xff;\n }\n const rawTexture = RawTexture.CreateRGBATexture(data, width, height, scene);\n return rawTexture;\n }\n /**\n * Resizes the two source textures to the same dimensions. If a texture is null, a default white texture is generated. If both textures are null, returns null\n * @param texture1 first texture to resize\n * @param texture2 second texture to resize\n * @param scene babylonjs scene\n * @returns resized textures or null\n */\n _resizeTexturesToSameDimensions(texture1, texture2, scene) {\n const texture1Size = texture1 ? texture1.getSize() : { width: 0, height: 0 };\n const texture2Size = texture2 ? texture2.getSize() : { width: 0, height: 0 };\n let resizedTexture1;\n let resizedTexture2;\n if (texture1Size.width < texture2Size.width) {\n if (texture1 && texture1 instanceof Texture) {\n resizedTexture1 = TextureTools.CreateResizedCopy(texture1, texture2Size.width, texture2Size.height, true);\n }\n else {\n resizedTexture1 = this._createWhiteTexture(texture2Size.width, texture2Size.height, scene);\n }\n resizedTexture2 = texture2;\n }\n else if (texture1Size.width > texture2Size.width) {\n if (texture2 && texture2 instanceof Texture) {\n resizedTexture2 = TextureTools.CreateResizedCopy(texture2, texture1Size.width, texture1Size.height, true);\n }\n else {\n resizedTexture2 = this._createWhiteTexture(texture1Size.width, texture1Size.height, scene);\n }\n resizedTexture1 = texture1;\n }\n else {\n resizedTexture1 = texture1;\n resizedTexture2 = texture2;\n }\n return {\n texture1: resizedTexture1,\n texture2: resizedTexture2,\n };\n }\n /**\n * Converts an array of pixels to a Float32Array\n * Throws an error if the pixel format is not supported\n * @param pixels - array buffer containing pixel values\n * @returns Float32 of pixels\n */\n _convertPixelArrayToFloat32(pixels) {\n if (pixels instanceof Uint8Array) {\n const length = pixels.length;\n const buffer = new Float32Array(pixels.length);\n for (let i = 0; i < length; ++i) {\n buffer[i] = pixels[i] / 255;\n }\n return buffer;\n }\n else if (pixels instanceof Float32Array) {\n return pixels;\n }\n else {\n throw new Error(\"Unsupported pixel format!\");\n }\n }\n /**\n * Convert Specular Glossiness Textures to Metallic Roughness\n * See link below for info on the material conversions from PBR Metallic/Roughness and Specular/Glossiness\n * @link https://github.com/KhronosGroup/glTF/blob/main/extensions/2.0/Archived/KHR_materials_pbrSpecularGlossiness/examples/convert-between-workflows-.js/babylon.pbrUtilities.js\n * @param diffuseTexture texture used to store diffuse information\n * @param specularGlossinessTexture texture used to store specular and glossiness information\n * @param factors specular glossiness material factors\n * @param mimeType the mime type to use for the texture\n * @returns pbr metallic roughness interface or null\n */\n async _convertSpecularGlossinessTexturesToMetallicRoughnessAsync(diffuseTexture, specularGlossinessTexture, factors, mimeType) {\n var _a;\n const promises = new Array();\n if (!(diffuseTexture || specularGlossinessTexture)) {\n return Promise.reject(\"_ConvertSpecularGlosinessTexturesToMetallicRoughness: diffuse and specular glossiness textures are not defined!\");\n }\n const scene = diffuseTexture ? diffuseTexture.getScene() : specularGlossinessTexture ? specularGlossinessTexture.getScene() : null;\n if (scene) {\n const resizedTextures = this._resizeTexturesToSameDimensions(diffuseTexture, specularGlossinessTexture, scene);\n const diffuseSize = (_a = resizedTextures.texture1) === null || _a === void 0 ? void 0 : _a.getSize();\n let diffuseBuffer;\n let specularGlossinessBuffer;\n const width = diffuseSize.width;\n const height = diffuseSize.height;\n const diffusePixels = await resizedTextures.texture1.readPixels();\n const specularPixels = await resizedTextures.texture2.readPixels();\n if (diffusePixels) {\n diffuseBuffer = this._convertPixelArrayToFloat32(diffusePixels);\n }\n else {\n return Promise.reject(\"Failed to retrieve pixels from diffuse texture!\");\n }\n if (specularPixels) {\n specularGlossinessBuffer = this._convertPixelArrayToFloat32(specularPixels);\n }\n else {\n return Promise.reject(\"Failed to retrieve pixels from specular glossiness texture!\");\n }\n const byteLength = specularGlossinessBuffer.byteLength;\n const metallicRoughnessBuffer = new Uint8Array(byteLength);\n const baseColorBuffer = new Uint8Array(byteLength);\n const strideSize = 4;\n const maxBaseColor = Color3.Black();\n let maxMetallic = 0;\n let maxRoughness = 0;\n for (let h = 0; h < height; ++h) {\n for (let w = 0; w < width; ++w) {\n const offset = (width * h + w) * strideSize;\n const diffuseColor = new Color3(diffuseBuffer[offset], diffuseBuffer[offset + 1], diffuseBuffer[offset + 2])\n .toLinearSpace(scene.getEngine().useExactSrgbConversions)\n .multiply(factors.diffuseColor);\n const specularColor = new Color3(specularGlossinessBuffer[offset], specularGlossinessBuffer[offset + 1], specularGlossinessBuffer[offset + 2])\n .toLinearSpace(scene.getEngine().useExactSrgbConversions)\n .multiply(factors.specularColor);\n const glossiness = specularGlossinessBuffer[offset + 3] * factors.glossiness;\n const specularGlossiness = {\n diffuseColor: diffuseColor,\n specularColor: specularColor,\n glossiness: glossiness,\n };\n const metallicRoughness = this._convertSpecularGlossinessToMetallicRoughness(specularGlossiness);\n maxBaseColor.r = Math.max(maxBaseColor.r, metallicRoughness.baseColor.r);\n maxBaseColor.g = Math.max(maxBaseColor.g, metallicRoughness.baseColor.g);\n maxBaseColor.b = Math.max(maxBaseColor.b, metallicRoughness.baseColor.b);\n maxMetallic = Math.max(maxMetallic, metallicRoughness.metallic);\n maxRoughness = Math.max(maxRoughness, metallicRoughness.roughness);\n baseColorBuffer[offset] = metallicRoughness.baseColor.r * 255;\n baseColorBuffer[offset + 1] = metallicRoughness.baseColor.g * 255;\n baseColorBuffer[offset + 2] = metallicRoughness.baseColor.b * 255;\n baseColorBuffer[offset + 3] = resizedTextures.texture1.hasAlpha ? diffuseBuffer[offset + 3] * 255 : 255;\n metallicRoughnessBuffer[offset] = 0;\n metallicRoughnessBuffer[offset + 1] = metallicRoughness.roughness * 255;\n metallicRoughnessBuffer[offset + 2] = metallicRoughness.metallic * 255;\n metallicRoughnessBuffer[offset + 3] = 255;\n }\n }\n // Retrieves the metallic roughness factors from the maximum texture values.\n const metallicRoughnessFactors = {\n baseColor: maxBaseColor,\n metallic: maxMetallic,\n roughness: maxRoughness,\n };\n let writeOutMetallicRoughnessTexture = false;\n let writeOutBaseColorTexture = false;\n for (let h = 0; h < height; ++h) {\n for (let w = 0; w < width; ++w) {\n const destinationOffset = (width * h + w) * strideSize;\n baseColorBuffer[destinationOffset] /= metallicRoughnessFactors.baseColor.r > _GLTFMaterialExporter._Epsilon ? metallicRoughnessFactors.baseColor.r : 1;\n baseColorBuffer[destinationOffset + 1] /= metallicRoughnessFactors.baseColor.g > _GLTFMaterialExporter._Epsilon ? metallicRoughnessFactors.baseColor.g : 1;\n baseColorBuffer[destinationOffset + 2] /= metallicRoughnessFactors.baseColor.b > _GLTFMaterialExporter._Epsilon ? metallicRoughnessFactors.baseColor.b : 1;\n const linearBaseColorPixel = Color3.FromInts(baseColorBuffer[destinationOffset], baseColorBuffer[destinationOffset + 1], baseColorBuffer[destinationOffset + 2]);\n const sRGBBaseColorPixel = linearBaseColorPixel.toGammaSpace(scene.getEngine().useExactSrgbConversions);\n baseColorBuffer[destinationOffset] = sRGBBaseColorPixel.r * 255;\n baseColorBuffer[destinationOffset + 1] = sRGBBaseColorPixel.g * 255;\n baseColorBuffer[destinationOffset + 2] = sRGBBaseColorPixel.b * 255;\n if (!_GLTFMaterialExporter._FuzzyEquals(sRGBBaseColorPixel, Color3.White(), _GLTFMaterialExporter._Epsilon)) {\n writeOutBaseColorTexture = true;\n }\n metallicRoughnessBuffer[destinationOffset + 1] /=\n metallicRoughnessFactors.roughness > _GLTFMaterialExporter._Epsilon ? metallicRoughnessFactors.roughness : 1;\n metallicRoughnessBuffer[destinationOffset + 2] /= metallicRoughnessFactors.metallic > _GLTFMaterialExporter._Epsilon ? metallicRoughnessFactors.metallic : 1;\n const metallicRoughnessPixel = Color3.FromInts(255, metallicRoughnessBuffer[destinationOffset + 1], metallicRoughnessBuffer[destinationOffset + 2]);\n if (!_GLTFMaterialExporter._FuzzyEquals(metallicRoughnessPixel, Color3.White(), _GLTFMaterialExporter._Epsilon)) {\n writeOutMetallicRoughnessTexture = true;\n }\n }\n }\n if (writeOutMetallicRoughnessTexture) {\n promises.push(this._getImageDataAsync(metallicRoughnessBuffer, width, height, mimeType).then((data) => {\n metallicRoughnessFactors.metallicRoughnessTextureData = data;\n }));\n }\n if (writeOutBaseColorTexture) {\n promises.push(this._getImageDataAsync(baseColorBuffer, width, height, mimeType).then((data) => {\n metallicRoughnessFactors.baseColorTextureData = data;\n }));\n }\n return Promise.all(promises).then(() => {\n return metallicRoughnessFactors;\n });\n }\n else {\n return Promise.reject(\"_ConvertSpecularGlossinessTexturesToMetallicRoughness: Scene from textures is missing!\");\n }\n }\n /**\n * Converts specular glossiness material properties to metallic roughness\n * @param specularGlossiness interface with specular glossiness material properties\n * @returns interface with metallic roughness material properties\n */\n _convertSpecularGlossinessToMetallicRoughness(specularGlossiness) {\n const diffusePerceivedBrightness = this._getPerceivedBrightness(specularGlossiness.diffuseColor);\n const specularPerceivedBrightness = this._getPerceivedBrightness(specularGlossiness.specularColor);\n const oneMinusSpecularStrength = 1 - this._getMaxComponent(specularGlossiness.specularColor);\n const metallic = _GLTFMaterialExporter._SolveMetallic(diffusePerceivedBrightness, specularPerceivedBrightness, oneMinusSpecularStrength);\n const baseColorFromDiffuse = specularGlossiness.diffuseColor.scale(oneMinusSpecularStrength / (1.0 - _GLTFMaterialExporter._DielectricSpecular.r) / Math.max(1 - metallic, _GLTFMaterialExporter._Epsilon));\n const baseColorFromSpecular = specularGlossiness.specularColor\n .subtract(_GLTFMaterialExporter._DielectricSpecular.scale(1 - metallic))\n .scale(1 / Math.max(metallic, _GLTFMaterialExporter._Epsilon));\n let baseColor = Color3.Lerp(baseColorFromDiffuse, baseColorFromSpecular, metallic * metallic);\n baseColor = baseColor.clampToRef(0, 1, baseColor);\n const metallicRoughness = {\n baseColor: baseColor,\n metallic: metallic,\n roughness: 1 - specularGlossiness.glossiness,\n };\n return metallicRoughness;\n }\n /**\n * Calculates the surface reflectance, independent of lighting conditions\n * @param color Color source to calculate brightness from\n * @returns number representing the perceived brightness, or zero if color is undefined\n */\n _getPerceivedBrightness(color) {\n if (color) {\n return Math.sqrt(0.299 * color.r * color.r + 0.587 * color.g * color.g + 0.114 * color.b * color.b);\n }\n return 0;\n }\n /**\n * Returns the maximum color component value\n * @param color\n * @returns maximum color component value, or zero if color is null or undefined\n */\n _getMaxComponent(color) {\n if (color) {\n return Math.max(color.r, Math.max(color.g, color.b));\n }\n return 0;\n }\n /**\n * Convert a PBRMaterial (Metallic/Roughness) to Metallic Roughness factors\n * @param babylonPBRMaterial BJS PBR Metallic Roughness Material\n * @param mimeType mime type to use for the textures\n * @param glTFPbrMetallicRoughness glTF PBR Metallic Roughness interface\n * @param hasTextureCoords specifies if texture coordinates are present on the submesh to determine if textures should be applied\n * @returns glTF PBR Metallic Roughness factors\n */\n _convertMetalRoughFactorsToMetallicRoughnessAsync(babylonPBRMaterial, mimeType, glTFPbrMetallicRoughness, hasTextureCoords) {\n const promises = [];\n const baseColor = babylonPBRMaterial._albedoColor;\n const metallic = babylonPBRMaterial._metallic;\n const roughness = babylonPBRMaterial._roughness;\n const metallicRoughness = {\n baseColor: baseColor,\n metallic: metallic,\n roughness: roughness,\n };\n if (hasTextureCoords) {\n const albedoTexture = babylonPBRMaterial._albedoTexture;\n if (albedoTexture) {\n promises.push(this._exportTextureAsync(babylonPBRMaterial._albedoTexture, mimeType).then((glTFTexture) => {\n if (glTFTexture) {\n glTFPbrMetallicRoughness.baseColorTexture = glTFTexture;\n }\n }));\n }\n const metallicTexture = babylonPBRMaterial._metallicTexture;\n if (metallicTexture) {\n promises.push(this._exportTextureAsync(metallicTexture, mimeType).then((glTFTexture) => {\n if (glTFTexture) {\n glTFPbrMetallicRoughness.metallicRoughnessTexture = glTFTexture;\n }\n }));\n }\n }\n return Promise.all(promises).then(() => {\n return metallicRoughness;\n });\n }\n _getTextureSampler(texture) {\n const sampler = {};\n if (!texture || !(texture instanceof Texture)) {\n return sampler;\n }\n const wrapS = this._getGLTFTextureWrapMode(texture.wrapU);\n if (wrapS !== 10497 /* TextureWrapMode.REPEAT */) {\n sampler.wrapS = wrapS;\n }\n const wrapT = this._getGLTFTextureWrapMode(texture.wrapV);\n if (wrapT !== 10497 /* TextureWrapMode.REPEAT */) {\n sampler.wrapT = wrapT;\n }\n switch (texture.samplingMode) {\n case Texture.LINEAR_LINEAR: {\n sampler.magFilter = 9729 /* TextureMagFilter.LINEAR */;\n sampler.minFilter = 9729 /* TextureMinFilter.LINEAR */;\n break;\n }\n case Texture.LINEAR_NEAREST: {\n sampler.magFilter = 9729 /* TextureMagFilter.LINEAR */;\n sampler.minFilter = 9728 /* TextureMinFilter.NEAREST */;\n break;\n }\n case Texture.NEAREST_LINEAR: {\n sampler.magFilter = 9728 /* TextureMagFilter.NEAREST */;\n sampler.minFilter = 9729 /* TextureMinFilter.LINEAR */;\n break;\n }\n case Texture.NEAREST_LINEAR_MIPLINEAR: {\n sampler.magFilter = 9728 /* TextureMagFilter.NEAREST */;\n sampler.minFilter = 9987 /* TextureMinFilter.LINEAR_MIPMAP_LINEAR */;\n break;\n }\n case Texture.NEAREST_NEAREST: {\n sampler.magFilter = 9728 /* TextureMagFilter.NEAREST */;\n sampler.minFilter = 9728 /* TextureMinFilter.NEAREST */;\n break;\n }\n case Texture.NEAREST_LINEAR_MIPNEAREST: {\n sampler.magFilter = 9728 /* TextureMagFilter.NEAREST */;\n sampler.minFilter = 9985 /* TextureMinFilter.LINEAR_MIPMAP_NEAREST */;\n break;\n }\n case Texture.LINEAR_NEAREST_MIPNEAREST: {\n sampler.magFilter = 9729 /* TextureMagFilter.LINEAR */;\n sampler.minFilter = 9984 /* TextureMinFilter.NEAREST_MIPMAP_NEAREST */;\n break;\n }\n case Texture.LINEAR_NEAREST_MIPLINEAR: {\n sampler.magFilter = 9729 /* TextureMagFilter.LINEAR */;\n sampler.minFilter = 9986 /* TextureMinFilter.NEAREST_MIPMAP_LINEAR */;\n break;\n }\n case Texture.NEAREST_NEAREST_MIPLINEAR: {\n sampler.magFilter = 9728 /* TextureMagFilter.NEAREST */;\n sampler.minFilter = 9986 /* TextureMinFilter.NEAREST_MIPMAP_LINEAR */;\n break;\n }\n case Texture.LINEAR_LINEAR_MIPLINEAR: {\n sampler.magFilter = 9729 /* TextureMagFilter.LINEAR */;\n sampler.minFilter = 9987 /* TextureMinFilter.LINEAR_MIPMAP_LINEAR */;\n break;\n }\n case Texture.LINEAR_LINEAR_MIPNEAREST: {\n sampler.magFilter = 9729 /* TextureMagFilter.LINEAR */;\n sampler.minFilter = 9985 /* TextureMinFilter.LINEAR_MIPMAP_NEAREST */;\n break;\n }\n case Texture.NEAREST_NEAREST_MIPNEAREST: {\n sampler.magFilter = 9728 /* TextureMagFilter.NEAREST */;\n sampler.minFilter = 9984 /* TextureMinFilter.NEAREST_MIPMAP_NEAREST */;\n break;\n }\n }\n return sampler;\n }\n _getGLTFTextureWrapMode(wrapMode) {\n switch (wrapMode) {\n case Texture.WRAP_ADDRESSMODE: {\n return 10497 /* TextureWrapMode.REPEAT */;\n }\n case Texture.CLAMP_ADDRESSMODE: {\n return 33071 /* TextureWrapMode.CLAMP_TO_EDGE */;\n }\n case Texture.MIRROR_ADDRESSMODE: {\n return 33648 /* TextureWrapMode.MIRRORED_REPEAT */;\n }\n default: {\n Tools.Error(`Unsupported Texture Wrap Mode ${wrapMode}!`);\n return 10497 /* TextureWrapMode.REPEAT */;\n }\n }\n }\n /**\n * Convert a PBRMaterial (Specular/Glossiness) to Metallic Roughness factors\n * @param babylonPBRMaterial BJS PBR Metallic Roughness Material\n * @param mimeType mime type to use for the textures\n * @param glTFPbrMetallicRoughness glTF PBR Metallic Roughness interface\n * @param hasTextureCoords specifies if texture coordinates are present on the submesh to determine if textures should be applied\n * @returns glTF PBR Metallic Roughness factors\n */\n _convertSpecGlossFactorsToMetallicRoughnessAsync(babylonPBRMaterial, mimeType, pbrMetallicRoughness, hasTextureCoords) {\n return Promise.resolve().then(() => {\n const specGloss = {\n diffuseColor: babylonPBRMaterial._albedoColor,\n specularColor: babylonPBRMaterial._reflectivityColor,\n glossiness: babylonPBRMaterial._microSurface,\n };\n const albedoTexture = babylonPBRMaterial._albedoTexture;\n const reflectivityTexture = babylonPBRMaterial._reflectivityTexture;\n const useMicrosurfaceFromReflectivityMapAlpha = babylonPBRMaterial._useMicroSurfaceFromReflectivityMapAlpha;\n if (reflectivityTexture && !useMicrosurfaceFromReflectivityMapAlpha) {\n return Promise.reject(\"_ConvertPBRMaterial: Glossiness values not included in the reflectivity texture are currently not supported\");\n }\n if ((albedoTexture || reflectivityTexture) && hasTextureCoords) {\n const samplerIndex = this._exportTextureSampler(albedoTexture || reflectivityTexture);\n return this._convertSpecularGlossinessTexturesToMetallicRoughnessAsync(albedoTexture, reflectivityTexture, specGloss, mimeType).then((metallicRoughnessFactors) => {\n const textures = this._exporter._textures;\n if (metallicRoughnessFactors.baseColorTextureData) {\n const imageIndex = this._exportImage(`baseColor${textures.length}`, mimeType, metallicRoughnessFactors.baseColorTextureData);\n pbrMetallicRoughness.baseColorTexture = this._exportTextureInfo(imageIndex, samplerIndex, albedoTexture === null || albedoTexture === void 0 ? void 0 : albedoTexture.coordinatesIndex);\n }\n if (metallicRoughnessFactors.metallicRoughnessTextureData) {\n const imageIndex = this._exportImage(`metallicRoughness${textures.length}`, mimeType, metallicRoughnessFactors.metallicRoughnessTextureData);\n pbrMetallicRoughness.metallicRoughnessTexture = this._exportTextureInfo(imageIndex, samplerIndex, reflectivityTexture === null || reflectivityTexture === void 0 ? void 0 : reflectivityTexture.coordinatesIndex);\n }\n return metallicRoughnessFactors;\n });\n }\n else {\n return this._convertSpecularGlossinessToMetallicRoughness(specGloss);\n }\n });\n }\n /**\n * Converts a Babylon PBR Base Material to a glTF Material\n * @param babylonPBRMaterial BJS PBR Base Material\n * @param mimeType mime type to use for the textures\n * @param hasTextureCoords specifies if texture coordinates are present on the submesh to determine if textures should be applied\n */\n _convertPBRMaterialAsync(babylonPBRMaterial, mimeType, hasTextureCoords) {\n const glTFPbrMetallicRoughness = {};\n const glTFMaterial = {\n name: babylonPBRMaterial.name,\n };\n const useMetallicRoughness = babylonPBRMaterial.isMetallicWorkflow();\n if (useMetallicRoughness) {\n const albedoColor = babylonPBRMaterial._albedoColor;\n const alpha = babylonPBRMaterial.alpha;\n if (albedoColor) {\n glTFPbrMetallicRoughness.baseColorFactor = [albedoColor.r, albedoColor.g, albedoColor.b, alpha];\n }\n return this._convertMetalRoughFactorsToMetallicRoughnessAsync(babylonPBRMaterial, mimeType, glTFPbrMetallicRoughness, hasTextureCoords).then((metallicRoughness) => {\n return this._setMetallicRoughnessPbrMaterial(metallicRoughness, babylonPBRMaterial, glTFMaterial, glTFPbrMetallicRoughness, mimeType, hasTextureCoords);\n });\n }\n else {\n return this._convertSpecGlossFactorsToMetallicRoughnessAsync(babylonPBRMaterial, mimeType, glTFPbrMetallicRoughness, hasTextureCoords).then((metallicRoughness) => {\n return this._setMetallicRoughnessPbrMaterial(metallicRoughness, babylonPBRMaterial, glTFMaterial, glTFPbrMetallicRoughness, mimeType, hasTextureCoords);\n });\n }\n }\n _setMetallicRoughnessPbrMaterial(metallicRoughness, babylonPBRMaterial, glTFMaterial, glTFPbrMetallicRoughness, mimeType, hasTextureCoords) {\n const materialMap = this._exporter._materialMap;\n const materials = this._exporter._materials;\n const promises = [];\n if (metallicRoughness) {\n _GLTFMaterialExporter._SetAlphaMode(glTFMaterial, babylonPBRMaterial);\n if (!(_GLTFMaterialExporter._FuzzyEquals(metallicRoughness.baseColor, Color3.White(), _GLTFMaterialExporter._Epsilon) &&\n babylonPBRMaterial.alpha >= _GLTFMaterialExporter._Epsilon)) {\n glTFPbrMetallicRoughness.baseColorFactor = [metallicRoughness.baseColor.r, metallicRoughness.baseColor.g, metallicRoughness.baseColor.b, babylonPBRMaterial.alpha];\n }\n if (metallicRoughness.metallic != null && metallicRoughness.metallic !== 1) {\n glTFPbrMetallicRoughness.metallicFactor = metallicRoughness.metallic;\n }\n if (metallicRoughness.roughness != null && metallicRoughness.roughness !== 1) {\n glTFPbrMetallicRoughness.roughnessFactor = metallicRoughness.roughness;\n }\n if (babylonPBRMaterial.backFaceCulling != null && !babylonPBRMaterial.backFaceCulling) {\n if (!babylonPBRMaterial._twoSidedLighting) {\n Tools.Warn(babylonPBRMaterial.name + \": Back-face culling disabled and two-sided lighting disabled is not supported in glTF.\");\n }\n glTFMaterial.doubleSided = true;\n }\n if (hasTextureCoords) {\n const bumpTexture = babylonPBRMaterial._bumpTexture;\n if (bumpTexture) {\n const promise = this._exportTextureAsync(bumpTexture, mimeType).then((glTFTexture) => {\n if (glTFTexture) {\n glTFMaterial.normalTexture = glTFTexture;\n if (bumpTexture.level !== 1) {\n glTFMaterial.normalTexture.scale = bumpTexture.level;\n }\n }\n });\n promises.push(promise);\n }\n const ambientTexture = babylonPBRMaterial._ambientTexture;\n if (ambientTexture) {\n const promise = this._exportTextureAsync(ambientTexture, mimeType).then((glTFTexture) => {\n if (glTFTexture) {\n const occlusionTexture = {\n index: glTFTexture.index,\n texCoord: glTFTexture.texCoord,\n extensions: glTFTexture.extensions,\n };\n glTFMaterial.occlusionTexture = occlusionTexture;\n const ambientTextureStrength = babylonPBRMaterial._ambientTextureStrength;\n if (ambientTextureStrength) {\n occlusionTexture.strength = ambientTextureStrength;\n }\n }\n });\n promises.push(promise);\n }\n const emissiveTexture = babylonPBRMaterial._emissiveTexture;\n if (emissiveTexture) {\n const promise = this._exportTextureAsync(emissiveTexture, mimeType).then((glTFTexture) => {\n if (glTFTexture) {\n glTFMaterial.emissiveTexture = glTFTexture;\n }\n });\n promises.push(promise);\n }\n }\n const emissiveColor = babylonPBRMaterial._emissiveColor;\n if (!_GLTFMaterialExporter._FuzzyEquals(emissiveColor, Color3.Black(), _GLTFMaterialExporter._Epsilon)) {\n glTFMaterial.emissiveFactor = emissiveColor.asArray();\n }\n glTFMaterial.pbrMetallicRoughness = glTFPbrMetallicRoughness;\n materials.push(glTFMaterial);\n materialMap[babylonPBRMaterial.uniqueId] = materials.length - 1;\n }\n return this._finishMaterial(promises, glTFMaterial, babylonPBRMaterial, mimeType);\n }\n _getPixelsFromTexture(babylonTexture) {\n const pixels = babylonTexture.textureType === Constants.TEXTURETYPE_UNSIGNED_INT\n ? babylonTexture.readPixels()\n : babylonTexture.readPixels();\n return pixels;\n }\n /**\n * Extracts a texture from a Babylon texture into file data and glTF data\n * @param babylonTexture Babylon texture to extract\n * @param mimeType Mime Type of the babylonTexture\n * @returns glTF texture info, or null if the texture format is not supported\n */\n _exportTextureAsync(babylonTexture, mimeType) {\n const extensionPromise = this._exporter._extensionsPreExportTextureAsync(\"exporter\", babylonTexture, mimeType);\n if (!extensionPromise) {\n return this._exportTextureInfoAsync(babylonTexture, mimeType);\n }\n return extensionPromise.then((texture) => {\n if (!texture) {\n return this._exportTextureInfoAsync(babylonTexture, mimeType);\n }\n return this._exportTextureInfoAsync(texture, mimeType);\n });\n }\n async _exportTextureInfoAsync(babylonTexture, mimeType) {\n const textureUid = babylonTexture.uid;\n if (!(textureUid in this._textureMap)) {\n const pixels = await this._getPixelsFromTexture(babylonTexture);\n if (!pixels) {\n return null;\n }\n const samplerIndex = this._exportTextureSampler(babylonTexture);\n // Preserve texture mime type if defined\n const textureMimeType = babylonTexture.mimeType;\n if (textureMimeType) {\n switch (textureMimeType) {\n case \"image/jpeg\":\n case \"image/png\":\n case \"image/webp\":\n mimeType = textureMimeType;\n break;\n default:\n Tools.Warn(`Unsupported media type: ${textureMimeType}`);\n break;\n }\n }\n const internalTextureToImage = this._internalTextureToImage;\n const internalTextureUniqueId = babylonTexture.getInternalTexture().uniqueId;\n internalTextureToImage[internalTextureUniqueId] || (internalTextureToImage[internalTextureUniqueId] = {});\n let imageIndexPromise = internalTextureToImage[internalTextureUniqueId][mimeType];\n if (imageIndexPromise === undefined) {\n const size = babylonTexture.getSize();\n imageIndexPromise = (async () => {\n const data = await this._getImageDataAsync(pixels, size.width, size.height, mimeType);\n return this._exportImage(babylonTexture.name, mimeType, data);\n })();\n internalTextureToImage[internalTextureUniqueId][mimeType] = imageIndexPromise;\n }\n const textureInfo = this._exportTextureInfo(await imageIndexPromise, samplerIndex, babylonTexture.coordinatesIndex);\n this._textureMap[textureUid] = textureInfo;\n this._exporter._extensionsPostExportTextures(\"exporter\", this._textureMap[textureUid], babylonTexture);\n }\n return this._textureMap[textureUid];\n }\n _exportImage(name, mimeType, data) {\n const imageData = this._exporter._imageData;\n const baseName = name.replace(/\\.\\/|\\/|\\.\\\\|\\\\/g, \"_\");\n const extension = getFileExtensionFromMimeType(mimeType);\n let fileName = baseName + extension;\n if (fileName in imageData) {\n fileName = `${baseName}_${Tools.RandomId()}${extension}`;\n }\n imageData[fileName] = {\n data: data,\n mimeType: mimeType,\n };\n const images = this._exporter._images;\n images.push({\n name: name,\n uri: fileName,\n });\n return images.length - 1;\n }\n _exportTextureInfo(imageIndex, samplerIndex, coordinatesIndex) {\n const textures = this._exporter._textures;\n let textureIndex = textures.findIndex((t) => t.sampler == samplerIndex && t.source === imageIndex);\n if (textureIndex === -1) {\n textureIndex = textures.length;\n textures.push({\n source: imageIndex,\n sampler: samplerIndex,\n });\n }\n const textureInfo = { index: textureIndex };\n if (coordinatesIndex) {\n textureInfo.texCoord = coordinatesIndex;\n }\n return textureInfo;\n }\n _exportTextureSampler(texture) {\n const sampler = this._getTextureSampler(texture);\n // if a pre-existing sampler with identical parameters exists, then reuse the previous sampler\n const samplers = this._exporter._samplers;\n const samplerIndex = samplers.findIndex((s) => s.minFilter === sampler.minFilter && s.magFilter === sampler.magFilter && s.wrapS === sampler.wrapS && s.wrapT === sampler.wrapT);\n if (samplerIndex !== -1) {\n return samplerIndex;\n }\n samplers.push(sampler);\n return samplers.length - 1;\n }\n }\n /**\n * Represents the dielectric specular values for R, G and B\n */\n _GLTFMaterialExporter._DielectricSpecular = new Color3(0.04, 0.04, 0.04);\n /**\n * Allows the maximum specular power to be defined for material calculations\n */\n _GLTFMaterialExporter._MaxSpecularPower = 1024;\n /**\n * Numeric tolerance value\n */\n _GLTFMaterialExporter._Epsilon = 1e-6;\n\n // Matrix that converts handedness on the X-axis.\n const convertHandednessMatrix = Matrix.Compose(new Vector3(-1, 1, 1), Quaternion.Identity(), Vector3.Zero());\n function isNoopNode(node, useRightHandedSystem) {\n if (!(node instanceof TransformNode)) {\n return false;\n }\n // Transform\n if (useRightHandedSystem) {\n const matrix = node.getWorldMatrix();\n if (!matrix.isIdentity()) {\n return false;\n }\n }\n else {\n const matrix = node.getWorldMatrix().multiplyToRef(convertHandednessMatrix, TmpVectors.Matrix[0]);\n if (!matrix.isIdentity()) {\n return false;\n }\n }\n // Geometry\n if ((node instanceof Mesh && node.geometry) || (node instanceof InstancedMesh && node.sourceMesh.geometry)) {\n return false;\n }\n return true;\n }\n function convertNodeHandedness(node) {\n const translation = Vector3.FromArrayToRef(node.translation || [0, 0, 0], 0, TmpVectors.Vector3[0]);\n const rotation = Quaternion.FromArrayToRef(node.rotation || [0, 0, 0, 1], 0, TmpVectors.Quaternion[0]);\n const scale = Vector3.FromArrayToRef(node.scale || [1, 1, 1], 0, TmpVectors.Vector3[1]);\n const matrix = Matrix.ComposeToRef(scale, rotation, translation, TmpVectors.Matrix[0]).multiplyToRef(convertHandednessMatrix, TmpVectors.Matrix[0]);\n matrix.decompose(scale, rotation, translation);\n if (translation.equalsToFloats(0, 0, 0)) {\n delete node.translation;\n }\n else {\n node.translation = translation.asArray();\n }\n if (Quaternion.IsIdentity(rotation)) {\n delete node.rotation;\n }\n else {\n node.rotation = rotation.asArray();\n }\n if (scale.equalsToFloats(1, 1, 1)) {\n delete node.scale;\n }\n else {\n node.scale = scale.asArray();\n }\n }\n /**\n * Converts Babylon Scene into glTF 2.0.\n * @internal\n */\n class _Exporter {\n _applyExtension(node, extensions, index, actionAsync) {\n if (index >= extensions.length) {\n return Promise.resolve(node);\n }\n const currentPromise = actionAsync(extensions[index], node);\n if (!currentPromise) {\n return this._applyExtension(node, extensions, index + 1, actionAsync);\n }\n return currentPromise.then((newNode) => this._applyExtension(newNode, extensions, index + 1, actionAsync));\n }\n _applyExtensions(node, actionAsync) {\n const extensions = [];\n for (const name of _Exporter._ExtensionNames) {\n extensions.push(this._extensions[name]);\n }\n return this._applyExtension(node, extensions, 0, actionAsync);\n }\n _extensionsPreExportTextureAsync(context, babylonTexture, mimeType) {\n return this._applyExtensions(babylonTexture, (extension, node) => extension.preExportTextureAsync && extension.preExportTextureAsync(context, node, mimeType));\n }\n _extensionsPostExportMeshPrimitiveAsync(context, meshPrimitive, babylonSubMesh, binaryWriter) {\n return this._applyExtensions(meshPrimitive, (extension, node) => extension.postExportMeshPrimitiveAsync && extension.postExportMeshPrimitiveAsync(context, node, babylonSubMesh, binaryWriter));\n }\n _extensionsPostExportNodeAsync(context, node, babylonNode, nodeMap, binaryWriter) {\n return this._applyExtensions(node, (extension, node) => extension.postExportNodeAsync && extension.postExportNodeAsync(context, node, babylonNode, nodeMap, binaryWriter));\n }\n _extensionsPostExportMaterialAsync(context, material, babylonMaterial) {\n return this._applyExtensions(material, (extension, node) => extension.postExportMaterialAsync && extension.postExportMaterialAsync(context, node, babylonMaterial));\n }\n _extensionsPostExportMaterialAdditionalTextures(context, material, babylonMaterial) {\n const output = [];\n for (const name of _Exporter._ExtensionNames) {\n const extension = this._extensions[name];\n if (extension.postExportMaterialAdditionalTextures) {\n output.push(...extension.postExportMaterialAdditionalTextures(context, material, babylonMaterial));\n }\n }\n return output;\n }\n _extensionsPostExportTextures(context, textureInfo, babylonTexture) {\n for (const name of _Exporter._ExtensionNames) {\n const extension = this._extensions[name];\n if (extension.postExportTexture) {\n extension.postExportTexture(context, textureInfo, babylonTexture);\n }\n }\n }\n _forEachExtensions(action) {\n for (const name of _Exporter._ExtensionNames) {\n const extension = this._extensions[name];\n if (extension.enabled) {\n action(extension);\n }\n }\n }\n _extensionsOnExporting() {\n this._forEachExtensions((extension) => {\n if (extension.wasUsed) {\n if (this._glTF.extensionsUsed == null) {\n this._glTF.extensionsUsed = [];\n }\n if (this._glTF.extensionsUsed.indexOf(extension.name) === -1) {\n this._glTF.extensionsUsed.push(extension.name);\n }\n if (extension.required) {\n if (this._glTF.extensionsRequired == null) {\n this._glTF.extensionsRequired = [];\n }\n if (this._glTF.extensionsRequired.indexOf(extension.name) === -1) {\n this._glTF.extensionsRequired.push(extension.name);\n }\n }\n if (this._glTF.extensions == null) {\n this._glTF.extensions = {};\n }\n if (extension.onExporting) {\n extension.onExporting();\n }\n }\n });\n }\n /**\n * Load glTF serializer extensions\n */\n _loadExtensions() {\n for (const name of _Exporter._ExtensionNames) {\n const extension = _Exporter._ExtensionFactories[name](this);\n this._extensions[name] = extension;\n }\n }\n /**\n * Creates a glTF Exporter instance, which can accept optional exporter options\n * @param babylonScene Babylon scene object\n * @param options Options to modify the behavior of the exporter\n */\n constructor(babylonScene, options) {\n this._extensions = {};\n this._glTF = {\n asset: { generator: `Babylon.js v${Engine.Version}`, version: \"2.0\" },\n };\n babylonScene = babylonScene || EngineStore.LastCreatedScene;\n if (!babylonScene) {\n return;\n }\n this._babylonScene = babylonScene;\n this._bufferViews = [];\n this._accessors = [];\n this._meshes = [];\n this._scenes = [];\n this._cameras = [];\n this._nodes = [];\n this._images = [];\n this._materials = [];\n this._materialMap = [];\n this._textures = [];\n this._samplers = [];\n this._skins = [];\n this._animations = [];\n this._imageData = {};\n this._orderedImageData = [];\n this._options = options || {};\n this._animationSampleRate = this._options.animationSampleRate || 1 / 60;\n this._glTFMaterialExporter = new _GLTFMaterialExporter(this);\n this._loadExtensions();\n }\n dispose() {\n for (const extensionKey in this._extensions) {\n const extension = this._extensions[extensionKey];\n extension.dispose();\n }\n }\n get options() {\n return this._options;\n }\n /**\n * Registers a glTF exporter extension\n * @param name Name of the extension to export\n * @param factory The factory function that creates the exporter extension\n */\n static RegisterExtension(name, factory) {\n if (_Exporter.UnregisterExtension(name)) {\n Tools.Warn(`Extension with the name ${name} already exists`);\n }\n _Exporter._ExtensionFactories[name] = factory;\n _Exporter._ExtensionNames.push(name);\n }\n /**\n * Un-registers an exporter extension\n * @param name The name fo the exporter extension\n * @returns A boolean indicating whether the extension has been un-registered\n */\n static UnregisterExtension(name) {\n if (!_Exporter._ExtensionFactories[name]) {\n return false;\n }\n delete _Exporter._ExtensionFactories[name];\n const index = _Exporter._ExtensionNames.indexOf(name);\n if (index !== -1) {\n _Exporter._ExtensionNames.splice(index, 1);\n }\n return true;\n }\n _reorderIndicesBasedOnPrimitiveMode(submesh, primitiveMode, babylonIndices, byteOffset, binaryWriter) {\n switch (primitiveMode) {\n case Material.TriangleFillMode: {\n if (!byteOffset) {\n byteOffset = 0;\n }\n for (let i = submesh.indexStart, length = submesh.indexStart + submesh.indexCount; i < length; i = i + 3) {\n const index = byteOffset + i * 4;\n // swap the second and third indices\n const secondIndex = binaryWriter.getUInt32(index + 4);\n const thirdIndex = binaryWriter.getUInt32(index + 8);\n binaryWriter.setUInt32(thirdIndex, index + 4);\n binaryWriter.setUInt32(secondIndex, index + 8);\n }\n break;\n }\n case Material.TriangleFanDrawMode: {\n for (let i = submesh.indexStart + submesh.indexCount - 1, start = submesh.indexStart; i >= start; --i) {\n binaryWriter.setUInt32(babylonIndices[i], byteOffset);\n byteOffset += 4;\n }\n break;\n }\n case Material.TriangleStripDrawMode: {\n if (submesh.indexCount >= 3) {\n binaryWriter.setUInt32(babylonIndices[submesh.indexStart + 2], byteOffset + 4);\n binaryWriter.setUInt32(babylonIndices[submesh.indexStart + 1], byteOffset + 8);\n }\n break;\n }\n }\n }\n /**\n * Reorders the vertex attribute data based on the primitive mode. This is necessary when indices are not available and the winding order is\n * clock-wise during export to glTF\n * @param submesh BabylonJS submesh\n * @param primitiveMode Primitive mode of the mesh\n * @param vertexBufferKind The type of vertex attribute\n * @param meshAttributeArray The vertex attribute data\n * @param byteOffset The offset to the binary data\n * @param binaryWriter The binary data for the glTF file\n */\n _reorderVertexAttributeDataBasedOnPrimitiveMode(submesh, primitiveMode, vertexBufferKind, meshAttributeArray, byteOffset, binaryWriter) {\n switch (primitiveMode) {\n case Material.TriangleFillMode: {\n this._reorderTriangleFillMode(submesh, vertexBufferKind, meshAttributeArray, byteOffset, binaryWriter);\n break;\n }\n case Material.TriangleStripDrawMode: {\n this._reorderTriangleStripDrawMode(submesh, vertexBufferKind, meshAttributeArray, byteOffset, binaryWriter);\n break;\n }\n case Material.TriangleFanDrawMode: {\n this._reorderTriangleFanMode(submesh, vertexBufferKind, meshAttributeArray, byteOffset, binaryWriter);\n break;\n }\n }\n }\n /**\n * Reorders the vertex attributes in the correct triangle mode order . This is necessary when indices are not available and the winding order is\n * clock-wise during export to glTF\n * @param submesh BabylonJS submesh\n * @param vertexBufferKind The type of vertex attribute\n * @param meshAttributeArray The vertex attribute data\n * @param byteOffset The offset to the binary data\n * @param binaryWriter The binary data for the glTF file\n */\n _reorderTriangleFillMode(submesh, vertexBufferKind, meshAttributeArray, byteOffset, binaryWriter) {\n const vertexBuffer = this._getVertexBufferFromMesh(vertexBufferKind, submesh.getMesh());\n if (vertexBuffer) {\n const stride = vertexBuffer.byteStride / VertexBuffer.GetTypeByteLength(vertexBuffer.type);\n if (submesh.verticesCount % 3 !== 0) {\n Tools.Error(\"The submesh vertices for the triangle fill mode is not divisible by 3!\");\n }\n else {\n const vertexData = [];\n let index = 0;\n switch (vertexBufferKind) {\n case VertexBuffer.PositionKind:\n case VertexBuffer.NormalKind: {\n for (let x = submesh.verticesStart; x < submesh.verticesStart + submesh.verticesCount; x = x + 3) {\n index = x * stride;\n vertexData.push(Vector3.FromArray(meshAttributeArray, index));\n vertexData.push(Vector3.FromArray(meshAttributeArray, index + 2 * stride));\n vertexData.push(Vector3.FromArray(meshAttributeArray, index + stride));\n }\n break;\n }\n case VertexBuffer.TangentKind: {\n for (let x = submesh.verticesStart; x < submesh.verticesStart + submesh.verticesCount; x = x + 3) {\n index = x * stride;\n vertexData.push(Vector4.FromArray(meshAttributeArray, index));\n vertexData.push(Vector4.FromArray(meshAttributeArray, index + 2 * stride));\n vertexData.push(Vector4.FromArray(meshAttributeArray, index + stride));\n }\n break;\n }\n case VertexBuffer.ColorKind: {\n const size = vertexBuffer.getSize();\n for (let x = submesh.verticesStart; x < submesh.verticesStart + submesh.verticesCount; x = x + size) {\n index = x * stride;\n if (size === 4) {\n vertexData.push(Vector4.FromArray(meshAttributeArray, index));\n vertexData.push(Vector4.FromArray(meshAttributeArray, index + 2 * stride));\n vertexData.push(Vector4.FromArray(meshAttributeArray, index + stride));\n }\n else {\n vertexData.push(Vector3.FromArray(meshAttributeArray, index));\n vertexData.push(Vector3.FromArray(meshAttributeArray, index + 2 * stride));\n vertexData.push(Vector3.FromArray(meshAttributeArray, index + stride));\n }\n }\n break;\n }\n case VertexBuffer.UVKind:\n case VertexBuffer.UV2Kind: {\n for (let x = submesh.verticesStart; x < submesh.verticesStart + submesh.verticesCount; x = x + 3) {\n index = x * stride;\n vertexData.push(Vector2.FromArray(meshAttributeArray, index));\n vertexData.push(Vector2.FromArray(meshAttributeArray, index + 2 * stride));\n vertexData.push(Vector2.FromArray(meshAttributeArray, index + stride));\n }\n break;\n }\n default: {\n Tools.Error(`Unsupported Vertex Buffer type: ${vertexBufferKind}`);\n }\n }\n this._writeVertexAttributeData(vertexData, byteOffset, vertexBufferKind, binaryWriter);\n }\n }\n else {\n Tools.Warn(`reorderTriangleFillMode: Vertex Buffer Kind ${vertexBufferKind} not present!`);\n }\n }\n /**\n * Reorders the vertex attributes in the correct triangle strip order. This is necessary when indices are not available and the winding order is\n * clock-wise during export to glTF\n * @param submesh BabylonJS submesh\n * @param vertexBufferKind The type of vertex attribute\n * @param meshAttributeArray The vertex attribute data\n * @param byteOffset The offset to the binary data\n * @param binaryWriter The binary data for the glTF file\n */\n _reorderTriangleStripDrawMode(submesh, vertexBufferKind, meshAttributeArray, byteOffset, binaryWriter) {\n const vertexBuffer = this._getVertexBufferFromMesh(vertexBufferKind, submesh.getMesh());\n if (vertexBuffer) {\n const stride = vertexBuffer.byteStride / VertexBuffer.GetTypeByteLength(vertexBuffer.type);\n const vertexData = [];\n let index = 0;\n switch (vertexBufferKind) {\n case VertexBuffer.PositionKind:\n case VertexBuffer.NormalKind: {\n index = submesh.verticesStart;\n vertexData.push(Vector3.FromArray(meshAttributeArray, index + 2 * stride));\n vertexData.push(Vector3.FromArray(meshAttributeArray, index + stride));\n break;\n }\n case VertexBuffer.TangentKind: {\n for (let x = submesh.verticesStart + submesh.verticesCount - 1; x >= submesh.verticesStart; --x) {\n index = x * stride;\n vertexData.push(Vector4.FromArray(meshAttributeArray, index));\n }\n break;\n }\n case VertexBuffer.ColorKind: {\n for (let x = submesh.verticesStart + submesh.verticesCount - 1; x >= submesh.verticesStart; --x) {\n index = x * stride;\n vertexBuffer.getSize() === 4\n ? vertexData.push(Vector4.FromArray(meshAttributeArray, index))\n : vertexData.push(Vector3.FromArray(meshAttributeArray, index));\n }\n break;\n }\n case VertexBuffer.UVKind:\n case VertexBuffer.UV2Kind: {\n for (let x = submesh.verticesStart + submesh.verticesCount - 1; x >= submesh.verticesStart; --x) {\n index = x * stride;\n vertexData.push(Vector2.FromArray(meshAttributeArray, index));\n }\n break;\n }\n default: {\n Tools.Error(`Unsupported Vertex Buffer type: ${vertexBufferKind}`);\n }\n }\n this._writeVertexAttributeData(vertexData, byteOffset + 12, vertexBufferKind, binaryWriter);\n }\n else {\n Tools.Warn(`reorderTriangleStripDrawMode: Vertex buffer kind ${vertexBufferKind} not present!`);\n }\n }\n /**\n * Reorders the vertex attributes in the correct triangle fan order. This is necessary when indices are not available and the winding order is\n * clock-wise during export to glTF\n * @param submesh BabylonJS submesh\n * @param vertexBufferKind The type of vertex attribute\n * @param meshAttributeArray The vertex attribute data\n * @param byteOffset The offset to the binary data\n * @param binaryWriter The binary data for the glTF file\n */\n _reorderTriangleFanMode(submesh, vertexBufferKind, meshAttributeArray, byteOffset, binaryWriter) {\n const vertexBuffer = this._getVertexBufferFromMesh(vertexBufferKind, submesh.getMesh());\n if (vertexBuffer) {\n const stride = vertexBuffer.byteStride / VertexBuffer.GetTypeByteLength(vertexBuffer.type);\n const vertexData = [];\n let index = 0;\n switch (vertexBufferKind) {\n case VertexBuffer.PositionKind:\n case VertexBuffer.NormalKind: {\n for (let x = submesh.verticesStart + submesh.verticesCount - 1; x >= submesh.verticesStart; --x) {\n index = x * stride;\n vertexData.push(Vector3.FromArray(meshAttributeArray, index));\n }\n break;\n }\n case VertexBuffer.TangentKind: {\n for (let x = submesh.verticesStart + submesh.verticesCount - 1; x >= submesh.verticesStart; --x) {\n index = x * stride;\n vertexData.push(Vector4.FromArray(meshAttributeArray, index));\n }\n break;\n }\n case VertexBuffer.ColorKind: {\n for (let x = submesh.verticesStart + submesh.verticesCount - 1; x >= submesh.verticesStart; --x) {\n index = x * stride;\n vertexData.push(Vector4.FromArray(meshAttributeArray, index));\n vertexBuffer.getSize() === 4\n ? vertexData.push(Vector4.FromArray(meshAttributeArray, index))\n : vertexData.push(Vector3.FromArray(meshAttributeArray, index));\n }\n break;\n }\n case VertexBuffer.UVKind:\n case VertexBuffer.UV2Kind: {\n for (let x = submesh.verticesStart + submesh.verticesCount - 1; x >= submesh.verticesStart; --x) {\n index = x * stride;\n vertexData.push(Vector2.FromArray(meshAttributeArray, index));\n }\n break;\n }\n default: {\n Tools.Error(`Unsupported Vertex Buffer type: ${vertexBufferKind}`);\n }\n }\n this._writeVertexAttributeData(vertexData, byteOffset, vertexBufferKind, binaryWriter);\n }\n else {\n Tools.Warn(`reorderTriangleFanMode: Vertex buffer kind ${vertexBufferKind} not present!`);\n }\n }\n /**\n * Writes the vertex attribute data to binary\n * @param vertices The vertices to write to the binary writer\n * @param byteOffset The offset into the binary writer to overwrite binary data\n * @param vertexAttributeKind The vertex attribute type\n * @param binaryWriter The writer containing the binary data\n */\n _writeVertexAttributeData(vertices, byteOffset, vertexAttributeKind, binaryWriter) {\n for (const vertex of vertices) {\n if (vertexAttributeKind === VertexBuffer.NormalKind) {\n vertex.normalize();\n }\n else if (vertexAttributeKind === VertexBuffer.TangentKind && vertex instanceof Vector4) {\n _GLTFUtilities._NormalizeTangentFromRef(vertex);\n }\n for (const component of vertex.asArray()) {\n binaryWriter.setFloat32(component, byteOffset);\n byteOffset += 4;\n }\n }\n }\n /**\n * Writes mesh attribute data to a data buffer\n * Returns the bytelength of the data\n * @param vertexBufferKind Indicates what kind of vertex data is being passed in\n * @param attributeComponentKind\n * @param meshAttributeArray Array containing the attribute data\n * @param stride Specifies the space between data\n * @param binaryWriter The buffer to write the binary data to\n * @param babylonTransformNode\n */\n _writeAttributeData(vertexBufferKind, attributeComponentKind, meshAttributeArray, stride, binaryWriter, babylonTransformNode) {\n let vertexAttributes = [];\n let index;\n switch (vertexBufferKind) {\n case VertexBuffer.PositionKind: {\n for (let k = 0, length = meshAttributeArray.length / stride; k < length; ++k) {\n index = k * stride;\n const vertexData = Vector3.FromArray(meshAttributeArray, index);\n vertexAttributes.push(vertexData.asArray());\n }\n break;\n }\n case VertexBuffer.NormalKind: {\n for (let k = 0, length = meshAttributeArray.length / stride; k < length; ++k) {\n index = k * stride;\n const vertexData = Vector3.FromArray(meshAttributeArray, index);\n vertexAttributes.push(vertexData.normalize().asArray());\n }\n break;\n }\n case VertexBuffer.TangentKind: {\n for (let k = 0, length = meshAttributeArray.length / stride; k < length; ++k) {\n index = k * stride;\n const vertexData = Vector4.FromArray(meshAttributeArray, index);\n _GLTFUtilities._NormalizeTangentFromRef(vertexData);\n vertexAttributes.push(vertexData.asArray());\n }\n break;\n }\n case VertexBuffer.ColorKind: {\n const meshMaterial = babylonTransformNode.material;\n const convertToLinear = meshMaterial ? meshMaterial.getClassName() === \"StandardMaterial\" : true;\n const vertexData = stride === 3 ? new Color3() : new Color4();\n const useExactSrgbConversions = this._babylonScene.getEngine().useExactSrgbConversions;\n for (let k = 0, length = meshAttributeArray.length / stride; k < length; ++k) {\n index = k * stride;\n if (stride === 3) {\n Color3.FromArrayToRef(meshAttributeArray, index, vertexData);\n if (convertToLinear) {\n vertexData.toLinearSpaceToRef(vertexData, useExactSrgbConversions);\n }\n }\n else {\n Color4.FromArrayToRef(meshAttributeArray, index, vertexData);\n if (convertToLinear) {\n vertexData.toLinearSpaceToRef(vertexData, useExactSrgbConversions);\n }\n }\n vertexAttributes.push(vertexData.asArray());\n }\n break;\n }\n case VertexBuffer.UVKind:\n case VertexBuffer.UV2Kind: {\n for (let k = 0, length = meshAttributeArray.length / stride; k < length; ++k) {\n index = k * stride;\n const vertexData = Vector2.FromArray(meshAttributeArray, index);\n vertexAttributes.push(vertexData.asArray());\n }\n break;\n }\n case VertexBuffer.MatricesIndicesKind:\n case VertexBuffer.MatricesIndicesExtraKind: {\n for (let k = 0, length = meshAttributeArray.length / stride; k < length; ++k) {\n index = k * stride;\n const vertexData = Vector4.FromArray(meshAttributeArray, index);\n vertexAttributes.push(vertexData.asArray());\n }\n break;\n }\n case VertexBuffer.MatricesWeightsKind:\n case VertexBuffer.MatricesWeightsExtraKind: {\n for (let k = 0, length = meshAttributeArray.length / stride; k < length; ++k) {\n index = k * stride;\n const vertexData = Vector4.FromArray(meshAttributeArray, index);\n vertexAttributes.push(vertexData.asArray());\n }\n break;\n }\n default: {\n Tools.Warn(\"Unsupported Vertex Buffer Type: \" + vertexBufferKind);\n vertexAttributes = [];\n }\n }\n let writeBinaryFunc;\n switch (attributeComponentKind) {\n case 5121 /* AccessorComponentType.UNSIGNED_BYTE */: {\n writeBinaryFunc = binaryWriter.setUInt8.bind(binaryWriter);\n break;\n }\n case 5123 /* AccessorComponentType.UNSIGNED_SHORT */: {\n writeBinaryFunc = binaryWriter.setUInt16.bind(binaryWriter);\n break;\n }\n case 5125 /* AccessorComponentType.UNSIGNED_INT */: {\n writeBinaryFunc = binaryWriter.setUInt32.bind(binaryWriter);\n break;\n }\n case 5126 /* AccessorComponentType.FLOAT */: {\n writeBinaryFunc = binaryWriter.setFloat32.bind(binaryWriter);\n break;\n }\n default: {\n Tools.Warn(\"Unsupported Attribute Component kind: \" + attributeComponentKind);\n return;\n }\n }\n for (const vertexAttribute of vertexAttributes) {\n for (const component of vertexAttribute) {\n writeBinaryFunc(component);\n }\n }\n }\n /**\n * Writes mesh attribute data to a data buffer\n * Returns the bytelength of the data\n * @param vertexBufferKind Indicates what kind of vertex data is being passed in\n * @param attributeComponentKind\n * @param meshPrimitive\n * @param morphTarget\n * @param meshAttributeArray Array containing the attribute data\n * @param morphTargetAttributeArray\n * @param stride Specifies the space between data\n * @param binaryWriter The buffer to write the binary data to\n * @param minMax\n */\n writeMorphTargetAttributeData(vertexBufferKind, attributeComponentKind, meshPrimitive, meshAttributeArray, morphTargetAttributeArray, stride, binaryWriter, minMax) {\n let vertexAttributes = [];\n let index;\n let difference = new Vector3();\n let difference4 = new Vector4(0, 0, 0, 0);\n switch (vertexBufferKind) {\n case VertexBuffer.PositionKind: {\n for (let k = meshPrimitive.verticesStart; k < meshPrimitive.verticesCount; ++k) {\n index = meshPrimitive.indexStart + k * stride;\n const vertexData = Vector3.FromArray(meshAttributeArray, index);\n const morphData = Vector3.FromArray(morphTargetAttributeArray, index);\n difference = morphData.subtractToRef(vertexData, difference);\n if (minMax) {\n minMax.min.copyFromFloats(Math.min(difference.x, minMax.min.x), Math.min(difference.y, minMax.min.y), Math.min(difference.z, minMax.min.z));\n minMax.max.copyFromFloats(Math.max(difference.x, minMax.max.x), Math.max(difference.y, minMax.max.y), Math.max(difference.z, minMax.max.z));\n }\n vertexAttributes.push(difference.asArray());\n }\n break;\n }\n case VertexBuffer.NormalKind: {\n for (let k = meshPrimitive.verticesStart; k < meshPrimitive.verticesCount; ++k) {\n index = meshPrimitive.indexStart + k * stride;\n const vertexData = Vector3.FromArray(meshAttributeArray, index).normalize();\n const morphData = Vector3.FromArray(morphTargetAttributeArray, index).normalize();\n difference = morphData.subtractToRef(vertexData, difference);\n vertexAttributes.push(difference.asArray());\n }\n break;\n }\n case VertexBuffer.TangentKind: {\n for (let k = meshPrimitive.verticesStart; k < meshPrimitive.verticesCount; ++k) {\n index = meshPrimitive.indexStart + k * (stride + 1);\n const vertexData = Vector4.FromArray(meshAttributeArray, index);\n _GLTFUtilities._NormalizeTangentFromRef(vertexData);\n const morphData = Vector4.FromArray(morphTargetAttributeArray, index);\n _GLTFUtilities._NormalizeTangentFromRef(morphData);\n difference4 = morphData.subtractToRef(vertexData, difference4);\n vertexAttributes.push([difference4.x, difference4.y, difference4.z]);\n }\n break;\n }\n default: {\n Tools.Warn(\"Unsupported Vertex Buffer Type: \" + vertexBufferKind);\n vertexAttributes = [];\n }\n }\n let writeBinaryFunc;\n switch (attributeComponentKind) {\n case 5121 /* AccessorComponentType.UNSIGNED_BYTE */: {\n writeBinaryFunc = binaryWriter.setUInt8.bind(binaryWriter);\n break;\n }\n case 5123 /* AccessorComponentType.UNSIGNED_SHORT */: {\n writeBinaryFunc = binaryWriter.setUInt16.bind(binaryWriter);\n break;\n }\n case 5125 /* AccessorComponentType.UNSIGNED_INT */: {\n writeBinaryFunc = binaryWriter.setUInt32.bind(binaryWriter);\n break;\n }\n case 5126 /* AccessorComponentType.FLOAT */: {\n writeBinaryFunc = binaryWriter.setFloat32.bind(binaryWriter);\n break;\n }\n default: {\n Tools.Warn(\"Unsupported Attribute Component kind: \" + attributeComponentKind);\n return;\n }\n }\n for (const vertexAttribute of vertexAttributes) {\n for (const component of vertexAttribute) {\n writeBinaryFunc(component);\n }\n }\n }\n /**\n * Generates glTF json data\n * @param shouldUseGlb Indicates whether the json should be written for a glb file\n * @param glTFPrefix Text to use when prefixing a glTF file\n * @param prettyPrint Indicates whether the json file should be pretty printed (true) or not (false)\n * @returns json data as string\n */\n _generateJSON(shouldUseGlb, glTFPrefix, prettyPrint) {\n const buffer = { byteLength: this._totalByteLength };\n let imageName;\n let imageData;\n let bufferView;\n let byteOffset = this._totalByteLength;\n if (buffer.byteLength) {\n this._glTF.buffers = [buffer];\n }\n if (this._nodes && this._nodes.length) {\n this._glTF.nodes = this._nodes;\n }\n if (this._meshes && this._meshes.length) {\n this._glTF.meshes = this._meshes;\n }\n if (this._scenes && this._scenes.length) {\n this._glTF.scenes = this._scenes;\n this._glTF.scene = 0;\n }\n if (this._cameras && this._cameras.length) {\n this._glTF.cameras = this._cameras;\n }\n if (this._bufferViews && this._bufferViews.length) {\n this._glTF.bufferViews = this._bufferViews;\n }\n if (this._accessors && this._accessors.length) {\n this._glTF.accessors = this._accessors;\n }\n if (this._animations && this._animations.length) {\n this._glTF.animations = this._animations;\n }\n if (this._materials && this._materials.length) {\n this._glTF.materials = this._materials;\n }\n if (this._textures && this._textures.length) {\n this._glTF.textures = this._textures;\n }\n if (this._samplers && this._samplers.length) {\n this._glTF.samplers = this._samplers;\n }\n if (this._skins && this._skins.length) {\n this._glTF.skins = this._skins;\n }\n if (this._images && this._images.length) {\n if (!shouldUseGlb) {\n this._glTF.images = this._images;\n }\n else {\n this._glTF.images = [];\n this._images.forEach((image) => {\n if (image.uri) {\n imageData = this._imageData[image.uri];\n this._orderedImageData.push(imageData);\n imageName = image.uri.split(\".\")[0] + \" image\";\n bufferView = _GLTFUtilities._CreateBufferView(0, byteOffset, imageData.data.byteLength, undefined, imageName);\n byteOffset += imageData.data.byteLength;\n this._bufferViews.push(bufferView);\n image.bufferView = this._bufferViews.length - 1;\n image.name = imageName;\n image.mimeType = imageData.mimeType;\n image.uri = undefined;\n if (!this._glTF.images) {\n this._glTF.images = [];\n }\n this._glTF.images.push(image);\n }\n });\n // Replace uri with bufferview and mime type for glb\n buffer.byteLength = byteOffset;\n }\n }\n if (!shouldUseGlb) {\n buffer.uri = glTFPrefix + \".bin\";\n }\n const jsonText = prettyPrint ? JSON.stringify(this._glTF, null, 2) : JSON.stringify(this._glTF);\n return jsonText;\n }\n /**\n * Generates data for .gltf and .bin files based on the glTF prefix string\n * @param glTFPrefix Text to use when prefixing a glTF file\n * @param dispose Dispose the exporter\n * @returns GLTFData with glTF file data\n */\n _generateGLTFAsync(glTFPrefix, dispose = true) {\n return this._generateBinaryAsync().then((binaryBuffer) => {\n this._extensionsOnExporting();\n const jsonText = this._generateJSON(false, glTFPrefix, true);\n const bin = new Blob([binaryBuffer], { type: \"application/octet-stream\" });\n const glTFFileName = glTFPrefix + \".gltf\";\n const glTFBinFile = glTFPrefix + \".bin\";\n const container = new GLTFData();\n container.glTFFiles[glTFFileName] = jsonText;\n container.glTFFiles[glTFBinFile] = bin;\n if (this._imageData) {\n for (const image in this._imageData) {\n container.glTFFiles[image] = new Blob([this._imageData[image].data], { type: this._imageData[image].mimeType });\n }\n }\n if (dispose) {\n this.dispose();\n }\n return container;\n });\n }\n /**\n * Creates a binary buffer for glTF\n * @returns array buffer for binary data\n */\n _generateBinaryAsync() {\n const binaryWriter = new _BinaryWriter(4);\n return this._createSceneAsync(binaryWriter).then(() => {\n if (this._localEngine) {\n this._localEngine.dispose();\n }\n return binaryWriter.getArrayBuffer();\n });\n }\n /**\n * Pads the number to a multiple of 4\n * @param num number to pad\n * @returns padded number\n */\n _getPadding(num) {\n const remainder = num % 4;\n const padding = remainder === 0 ? remainder : 4 - remainder;\n return padding;\n }\n /**\n * @internal\n */\n _generateGLBAsync(glTFPrefix, dispose = true) {\n return this._generateBinaryAsync().then((binaryBuffer) => {\n this._extensionsOnExporting();\n const jsonText = this._generateJSON(true);\n const glbFileName = glTFPrefix + \".glb\";\n const headerLength = 12;\n const chunkLengthPrefix = 8;\n let jsonLength = jsonText.length;\n let encodedJsonText;\n let imageByteLength = 0;\n // make use of TextEncoder when available\n if (typeof TextEncoder !== \"undefined\") {\n const encoder = new TextEncoder();\n encodedJsonText = encoder.encode(jsonText);\n jsonLength = encodedJsonText.length;\n }\n for (let i = 0; i < this._orderedImageData.length; ++i) {\n imageByteLength += this._orderedImageData[i].data.byteLength;\n }\n const jsonPadding = this._getPadding(jsonLength);\n const binPadding = this._getPadding(binaryBuffer.byteLength);\n const imagePadding = this._getPadding(imageByteLength);\n const byteLength = headerLength + 2 * chunkLengthPrefix + jsonLength + jsonPadding + binaryBuffer.byteLength + binPadding + imageByteLength + imagePadding;\n //header\n const headerBuffer = new ArrayBuffer(headerLength);\n const headerBufferView = new DataView(headerBuffer);\n headerBufferView.setUint32(0, 0x46546c67, true); //glTF\n headerBufferView.setUint32(4, 2, true); // version\n headerBufferView.setUint32(8, byteLength, true); // total bytes in file\n //json chunk\n const jsonChunkBuffer = new ArrayBuffer(chunkLengthPrefix + jsonLength + jsonPadding);\n const jsonChunkBufferView = new DataView(jsonChunkBuffer);\n jsonChunkBufferView.setUint32(0, jsonLength + jsonPadding, true);\n jsonChunkBufferView.setUint32(4, 0x4e4f534a, true);\n //json chunk bytes\n const jsonData = new Uint8Array(jsonChunkBuffer, chunkLengthPrefix);\n // if TextEncoder was available, we can simply copy the encoded array\n if (encodedJsonText) {\n jsonData.set(encodedJsonText);\n }\n else {\n const blankCharCode = \"_\".charCodeAt(0);\n for (let i = 0; i < jsonLength; ++i) {\n const charCode = jsonText.charCodeAt(i);\n // if the character doesn't fit into a single UTF-16 code unit, just put a blank character\n if (charCode != jsonText.codePointAt(i)) {\n jsonData[i] = blankCharCode;\n }\n else {\n jsonData[i] = charCode;\n }\n }\n }\n //json padding\n const jsonPaddingView = new Uint8Array(jsonChunkBuffer, chunkLengthPrefix + jsonLength);\n for (let i = 0; i < jsonPadding; ++i) {\n jsonPaddingView[i] = 0x20;\n }\n //binary chunk\n const binaryChunkBuffer = new ArrayBuffer(chunkLengthPrefix);\n const binaryChunkBufferView = new DataView(binaryChunkBuffer);\n binaryChunkBufferView.setUint32(0, binaryBuffer.byteLength + imageByteLength + imagePadding, true);\n binaryChunkBufferView.setUint32(4, 0x004e4942, true);\n // binary padding\n const binPaddingBuffer = new ArrayBuffer(binPadding);\n const binPaddingView = new Uint8Array(binPaddingBuffer);\n for (let i = 0; i < binPadding; ++i) {\n binPaddingView[i] = 0;\n }\n const imagePaddingBuffer = new ArrayBuffer(imagePadding);\n const imagePaddingView = new Uint8Array(imagePaddingBuffer);\n for (let i = 0; i < imagePadding; ++i) {\n imagePaddingView[i] = 0;\n }\n const glbData = [headerBuffer, jsonChunkBuffer, binaryChunkBuffer, binaryBuffer];\n // binary data\n for (let i = 0; i < this._orderedImageData.length; ++i) {\n glbData.push(this._orderedImageData[i].data);\n }\n glbData.push(binPaddingBuffer);\n glbData.push(imagePaddingBuffer);\n const glbFile = new Blob(glbData, { type: \"application/octet-stream\" });\n const container = new GLTFData();\n container.glTFFiles[glbFileName] = glbFile;\n if (this._localEngine != null) {\n this._localEngine.dispose();\n }\n if (dispose) {\n this.dispose();\n }\n return container;\n });\n }\n /**\n * Sets the TRS for each node\n * @param node glTF Node for storing the transformation data\n * @param babylonTransformNode Babylon mesh used as the source for the transformation data\n */\n _setNodeTransformation(node, babylonTransformNode) {\n if (!babylonTransformNode.getPivotPoint().equalsToFloats(0, 0, 0)) {\n Tools.Warn(\"Pivot points are not supported in the glTF serializer\");\n }\n if (!babylonTransformNode.position.equalsToFloats(0, 0, 0)) {\n node.translation = babylonTransformNode.position.asArray();\n }\n if (!babylonTransformNode.scaling.equalsToFloats(1, 1, 1)) {\n node.scale = babylonTransformNode.scaling.asArray();\n }\n const rotationQuaternion = Quaternion.RotationYawPitchRoll(babylonTransformNode.rotation.y, babylonTransformNode.rotation.x, babylonTransformNode.rotation.z);\n if (babylonTransformNode.rotationQuaternion) {\n rotationQuaternion.multiplyInPlace(babylonTransformNode.rotationQuaternion);\n }\n if (!Quaternion.IsIdentity(rotationQuaternion)) {\n node.rotation = rotationQuaternion.normalize().asArray();\n }\n }\n _setCameraTransformation(node, babylonCamera) {\n if (!babylonCamera.position.equalsToFloats(0, 0, 0)) {\n node.translation = babylonCamera.position.asArray();\n }\n const rotationQuaternion = babylonCamera.rotationQuaternion; // we target the local transformation if one.\n if (rotationQuaternion && !Quaternion.IsIdentity(rotationQuaternion)) {\n node.rotation = rotationQuaternion.normalize().asArray();\n }\n }\n _getVertexBufferFromMesh(attributeKind, bufferMesh) {\n if (bufferMesh.isVerticesDataPresent(attributeKind, true)) {\n const vertexBuffer = bufferMesh.getVertexBuffer(attributeKind, true);\n if (vertexBuffer) {\n return vertexBuffer;\n }\n }\n return null;\n }\n /**\n * Creates a bufferview based on the vertices type for the Babylon mesh\n * @param kind Indicates the type of vertices data\n * @param attributeComponentKind Indicates the numerical type used to store the data\n * @param babylonTransformNode The Babylon mesh to get the vertices data from\n * @param binaryWriter The buffer to write the bufferview data to\n * @param byteStride\n */\n _createBufferViewKind(kind, attributeComponentKind, babylonTransformNode, binaryWriter, byteStride) {\n const bufferMesh = babylonTransformNode instanceof Mesh\n ? babylonTransformNode\n : babylonTransformNode instanceof InstancedMesh\n ? babylonTransformNode.sourceMesh\n : null;\n if (bufferMesh) {\n const vertexBuffer = bufferMesh.getVertexBuffer(kind, true);\n const vertexData = bufferMesh.getVerticesData(kind, undefined, undefined, true);\n if (vertexBuffer && vertexData) {\n const typeByteLength = VertexBuffer.GetTypeByteLength(attributeComponentKind);\n const byteLength = vertexData.length * typeByteLength;\n const bufferView = _GLTFUtilities._CreateBufferView(0, binaryWriter.getByteOffset(), byteLength, byteStride, kind + \" - \" + bufferMesh.name);\n this._bufferViews.push(bufferView);\n this._writeAttributeData(kind, attributeComponentKind, vertexData, byteStride / typeByteLength, binaryWriter, babylonTransformNode);\n }\n }\n }\n /**\n * Creates a bufferview based on the vertices type for the Babylon mesh\n * @param babylonSubMesh The Babylon submesh that the morph target is applied to\n * @param meshPrimitive\n * @param babylonMorphTarget the morph target to be exported\n * @param binaryWriter The buffer to write the bufferview data to\n */\n _setMorphTargetAttributes(babylonSubMesh, meshPrimitive, babylonMorphTarget, binaryWriter) {\n if (babylonMorphTarget) {\n if (!meshPrimitive.targets) {\n meshPrimitive.targets = [];\n }\n const target = {};\n const mesh = babylonSubMesh.getMesh();\n if (babylonMorphTarget.hasNormals) {\n const vertexNormals = mesh.getVerticesData(VertexBuffer.NormalKind, undefined, undefined, true);\n const morphNormals = babylonMorphTarget.getNormals();\n const count = babylonSubMesh.verticesCount;\n const byteStride = 12; // 3 x 4 byte floats\n const byteLength = count * byteStride;\n const bufferView = _GLTFUtilities._CreateBufferView(0, binaryWriter.getByteOffset(), byteLength, byteStride, babylonMorphTarget.name + \"_NORMAL\");\n this._bufferViews.push(bufferView);\n const bufferViewIndex = this._bufferViews.length - 1;\n const accessor = _GLTFUtilities._CreateAccessor(bufferViewIndex, babylonMorphTarget.name + \" - \" + \"NORMAL\", \"VEC3\" /* AccessorType.VEC3 */, 5126 /* AccessorComponentType.FLOAT */, count, 0, null, null);\n this._accessors.push(accessor);\n target.NORMAL = this._accessors.length - 1;\n this.writeMorphTargetAttributeData(VertexBuffer.NormalKind, 5126 /* AccessorComponentType.FLOAT */, babylonSubMesh, vertexNormals, morphNormals, byteStride / 4, binaryWriter);\n }\n if (babylonMorphTarget.hasPositions) {\n const vertexPositions = mesh.getVerticesData(VertexBuffer.PositionKind, undefined, undefined, true);\n const morphPositions = babylonMorphTarget.getPositions();\n const count = babylonSubMesh.verticesCount;\n const byteStride = 12; // 3 x 4 byte floats\n const byteLength = count * byteStride;\n const bufferView = _GLTFUtilities._CreateBufferView(0, binaryWriter.getByteOffset(), byteLength, byteStride, babylonMorphTarget.name + \"_POSITION\");\n this._bufferViews.push(bufferView);\n const bufferViewIndex = this._bufferViews.length - 1;\n const minMax = { min: new Vector3(Infinity, Infinity, Infinity), max: new Vector3(-Infinity, -Infinity, -Infinity) };\n const accessor = _GLTFUtilities._CreateAccessor(bufferViewIndex, babylonMorphTarget.name + \" - \" + \"POSITION\", \"VEC3\" /* AccessorType.VEC3 */, 5126 /* AccessorComponentType.FLOAT */, count, 0, null, null);\n this._accessors.push(accessor);\n target.POSITION = this._accessors.length - 1;\n this.writeMorphTargetAttributeData(VertexBuffer.PositionKind, 5126 /* AccessorComponentType.FLOAT */, babylonSubMesh, vertexPositions, morphPositions, byteStride / 4, binaryWriter, minMax);\n accessor.min = minMax.min.asArray();\n accessor.max = minMax.max.asArray();\n }\n if (babylonMorphTarget.hasTangents) {\n const vertexTangents = mesh.getVerticesData(VertexBuffer.TangentKind, undefined, undefined, true);\n const morphTangents = babylonMorphTarget.getTangents();\n const count = babylonSubMesh.verticesCount;\n const byteStride = 12; // 3 x 4 byte floats\n const byteLength = count * byteStride;\n const bufferView = _GLTFUtilities._CreateBufferView(0, binaryWriter.getByteOffset(), byteLength, byteStride, babylonMorphTarget.name + \"_NORMAL\");\n this._bufferViews.push(bufferView);\n const bufferViewIndex = this._bufferViews.length - 1;\n const accessor = _GLTFUtilities._CreateAccessor(bufferViewIndex, babylonMorphTarget.name + \" - \" + \"TANGENT\", \"VEC3\" /* AccessorType.VEC3 */, 5126 /* AccessorComponentType.FLOAT */, count, 0, null, null);\n this._accessors.push(accessor);\n target.TANGENT = this._accessors.length - 1;\n this.writeMorphTargetAttributeData(VertexBuffer.TangentKind, 5126 /* AccessorComponentType.FLOAT */, babylonSubMesh, vertexTangents, morphTangents, byteStride / 4, binaryWriter);\n }\n meshPrimitive.targets.push(target);\n }\n }\n /**\n * The primitive mode of the Babylon mesh\n * @param babylonMesh The BabylonJS mesh\n */\n _getMeshPrimitiveMode(babylonMesh) {\n if (babylonMesh instanceof LinesMesh) {\n return Material.LineListDrawMode;\n }\n if (babylonMesh instanceof InstancedMesh || babylonMesh instanceof Mesh) {\n const baseMesh = babylonMesh instanceof Mesh ? babylonMesh : babylonMesh.sourceMesh;\n if (typeof baseMesh.overrideRenderingFillMode === \"number\") {\n return baseMesh.overrideRenderingFillMode;\n }\n }\n return babylonMesh.material ? babylonMesh.material.fillMode : Material.TriangleFillMode;\n }\n /**\n * Sets the primitive mode of the glTF mesh primitive\n * @param meshPrimitive glTF mesh primitive\n * @param primitiveMode The primitive mode\n */\n _setPrimitiveMode(meshPrimitive, primitiveMode) {\n switch (primitiveMode) {\n case Material.TriangleFillMode: {\n // glTF defaults to using Triangle Mode\n break;\n }\n case Material.TriangleStripDrawMode: {\n meshPrimitive.mode = 5 /* MeshPrimitiveMode.TRIANGLE_STRIP */;\n break;\n }\n case Material.TriangleFanDrawMode: {\n meshPrimitive.mode = 6 /* MeshPrimitiveMode.TRIANGLE_FAN */;\n break;\n }\n case Material.PointListDrawMode: {\n meshPrimitive.mode = 0 /* MeshPrimitiveMode.POINTS */;\n break;\n }\n case Material.PointFillMode: {\n meshPrimitive.mode = 0 /* MeshPrimitiveMode.POINTS */;\n break;\n }\n case Material.LineLoopDrawMode: {\n meshPrimitive.mode = 2 /* MeshPrimitiveMode.LINE_LOOP */;\n break;\n }\n case Material.LineListDrawMode: {\n meshPrimitive.mode = 1 /* MeshPrimitiveMode.LINES */;\n break;\n }\n case Material.LineStripDrawMode: {\n meshPrimitive.mode = 3 /* MeshPrimitiveMode.LINE_STRIP */;\n break;\n }\n }\n }\n /**\n * Sets the vertex attribute accessor based of the glTF mesh primitive\n * @param meshPrimitive glTF mesh primitive\n * @param attributeKind vertex attribute\n * @returns boolean specifying if uv coordinates are present\n */\n _setAttributeKind(meshPrimitive, attributeKind) {\n switch (attributeKind) {\n case VertexBuffer.PositionKind: {\n meshPrimitive.attributes.POSITION = this._accessors.length - 1;\n break;\n }\n case VertexBuffer.NormalKind: {\n meshPrimitive.attributes.NORMAL = this._accessors.length - 1;\n break;\n }\n case VertexBuffer.ColorKind: {\n meshPrimitive.attributes.COLOR_0 = this._accessors.length - 1;\n break;\n }\n case VertexBuffer.TangentKind: {\n meshPrimitive.attributes.TANGENT = this._accessors.length - 1;\n break;\n }\n case VertexBuffer.UVKind: {\n meshPrimitive.attributes.TEXCOORD_0 = this._accessors.length - 1;\n break;\n }\n case VertexBuffer.UV2Kind: {\n meshPrimitive.attributes.TEXCOORD_1 = this._accessors.length - 1;\n break;\n }\n case VertexBuffer.MatricesIndicesKind: {\n meshPrimitive.attributes.JOINTS_0 = this._accessors.length - 1;\n break;\n }\n case VertexBuffer.MatricesIndicesExtraKind: {\n meshPrimitive.attributes.JOINTS_1 = this._accessors.length - 1;\n break;\n }\n case VertexBuffer.MatricesWeightsKind: {\n meshPrimitive.attributes.WEIGHTS_0 = this._accessors.length - 1;\n break;\n }\n case VertexBuffer.MatricesWeightsExtraKind: {\n meshPrimitive.attributes.WEIGHTS_1 = this._accessors.length - 1;\n break;\n }\n default: {\n Tools.Warn(\"Unsupported Vertex Buffer Type: \" + attributeKind);\n }\n }\n }\n /**\n * Sets data for the primitive attributes of each submesh\n * @param mesh glTF Mesh object to store the primitive attribute information\n * @param babylonTransformNode Babylon mesh to get the primitive attribute data from\n * @param binaryWriter Buffer to write the attribute data to\n */\n _setPrimitiveAttributesAsync(mesh, babylonTransformNode, binaryWriter) {\n const promises = [];\n let bufferMesh = null;\n let bufferView;\n let minMax;\n if (babylonTransformNode instanceof Mesh) {\n bufferMesh = babylonTransformNode;\n }\n else if (babylonTransformNode instanceof InstancedMesh) {\n bufferMesh = babylonTransformNode.sourceMesh;\n }\n const attributeData = [\n { kind: VertexBuffer.PositionKind, accessorType: \"VEC3\" /* AccessorType.VEC3 */, accessorComponentType: 5126 /* AccessorComponentType.FLOAT */, byteStride: 12 },\n { kind: VertexBuffer.NormalKind, accessorType: \"VEC3\" /* AccessorType.VEC3 */, accessorComponentType: 5126 /* AccessorComponentType.FLOAT */, byteStride: 12 },\n { kind: VertexBuffer.ColorKind, accessorType: \"VEC4\" /* AccessorType.VEC4 */, accessorComponentType: 5126 /* AccessorComponentType.FLOAT */, byteStride: 16 },\n { kind: VertexBuffer.TangentKind, accessorType: \"VEC4\" /* AccessorType.VEC4 */, accessorComponentType: 5126 /* AccessorComponentType.FLOAT */, byteStride: 16 },\n { kind: VertexBuffer.UVKind, accessorType: \"VEC2\" /* AccessorType.VEC2 */, accessorComponentType: 5126 /* AccessorComponentType.FLOAT */, byteStride: 8 },\n { kind: VertexBuffer.UV2Kind, accessorType: \"VEC2\" /* AccessorType.VEC2 */, accessorComponentType: 5126 /* AccessorComponentType.FLOAT */, byteStride: 8 },\n { kind: VertexBuffer.MatricesIndicesKind, accessorType: \"VEC4\" /* AccessorType.VEC4 */, accessorComponentType: 5123 /* AccessorComponentType.UNSIGNED_SHORT */, byteStride: 8 },\n { kind: VertexBuffer.MatricesIndicesExtraKind, accessorType: \"VEC4\" /* AccessorType.VEC4 */, accessorComponentType: 5123 /* AccessorComponentType.UNSIGNED_SHORT */, byteStride: 8 },\n { kind: VertexBuffer.MatricesWeightsKind, accessorType: \"VEC4\" /* AccessorType.VEC4 */, accessorComponentType: 5126 /* AccessorComponentType.FLOAT */, byteStride: 16 },\n { kind: VertexBuffer.MatricesWeightsExtraKind, accessorType: \"VEC4\" /* AccessorType.VEC4 */, accessorComponentType: 5126 /* AccessorComponentType.FLOAT */, byteStride: 16 },\n ];\n if (bufferMesh) {\n let indexBufferViewIndex = null;\n const primitiveMode = this._getMeshPrimitiveMode(bufferMesh);\n const vertexAttributeBufferViews = {};\n const morphTargetManager = bufferMesh.morphTargetManager;\n // For each BabylonMesh, create bufferviews for each 'kind'\n for (const attribute of attributeData) {\n const attributeKind = attribute.kind;\n const attributeComponentKind = attribute.accessorComponentType;\n if (bufferMesh.isVerticesDataPresent(attributeKind, true)) {\n const vertexBuffer = this._getVertexBufferFromMesh(attributeKind, bufferMesh);\n attribute.byteStride = vertexBuffer\n ? vertexBuffer.getSize() * VertexBuffer.GetTypeByteLength(attribute.accessorComponentType)\n : VertexBuffer.DeduceStride(attributeKind) * 4;\n if (attribute.byteStride === 12) {\n attribute.accessorType = \"VEC3\" /* AccessorType.VEC3 */;\n }\n this._createBufferViewKind(attributeKind, attributeComponentKind, babylonTransformNode, binaryWriter, attribute.byteStride);\n attribute.bufferViewIndex = this._bufferViews.length - 1;\n vertexAttributeBufferViews[attributeKind] = attribute.bufferViewIndex;\n }\n }\n if (bufferMesh.getTotalIndices()) {\n const indices = bufferMesh.getIndices();\n if (indices) {\n const byteLength = indices.length * 4;\n bufferView = _GLTFUtilities._CreateBufferView(0, binaryWriter.getByteOffset(), byteLength, undefined, \"Indices - \" + bufferMesh.name);\n this._bufferViews.push(bufferView);\n indexBufferViewIndex = this._bufferViews.length - 1;\n for (let k = 0, length = indices.length; k < length; ++k) {\n binaryWriter.setUInt32(indices[k]);\n }\n }\n }\n if (bufferMesh.subMeshes) {\n // go through all mesh primitives (submeshes)\n for (const submesh of bufferMesh.subMeshes) {\n let babylonMaterial = submesh.getMaterial() || bufferMesh.getScene().defaultMaterial;\n let materialIndex = null;\n if (babylonMaterial) {\n if (bufferMesh instanceof LinesMesh) {\n // get the color from the lines mesh and set it in the material\n const material = {\n name: bufferMesh.name + \" material\",\n };\n if (!bufferMesh.color.equals(Color3.White()) || bufferMesh.alpha < 1) {\n material.pbrMetallicRoughness = {\n baseColorFactor: bufferMesh.color.asArray().concat([bufferMesh.alpha]),\n };\n }\n this._materials.push(material);\n materialIndex = this._materials.length - 1;\n }\n else if (babylonMaterial instanceof MultiMaterial) {\n const subMaterial = babylonMaterial.subMaterials[submesh.materialIndex];\n if (subMaterial) {\n babylonMaterial = subMaterial;\n materialIndex = this._materialMap[babylonMaterial.uniqueId];\n }\n }\n else {\n materialIndex = this._materialMap[babylonMaterial.uniqueId];\n }\n }\n const glTFMaterial = materialIndex != null ? this._materials[materialIndex] : null;\n const meshPrimitive = { attributes: {} };\n this._setPrimitiveMode(meshPrimitive, primitiveMode);\n for (const attribute of attributeData) {\n const attributeKind = attribute.kind;\n if ((attributeKind === VertexBuffer.UVKind || attributeKind === VertexBuffer.UV2Kind) && !this._options.exportUnusedUVs) {\n if (!glTFMaterial || !this._glTFMaterialExporter._hasTexturesPresent(glTFMaterial)) {\n continue;\n }\n }\n const vertexData = bufferMesh.getVerticesData(attributeKind, undefined, undefined, true);\n if (vertexData) {\n const vertexBuffer = this._getVertexBufferFromMesh(attributeKind, bufferMesh);\n if (vertexBuffer) {\n const stride = vertexBuffer.getSize();\n const bufferViewIndex = attribute.bufferViewIndex;\n if (bufferViewIndex != undefined) {\n // check to see if bufferviewindex has a numeric value assigned.\n minMax = { min: null, max: null };\n if (attributeKind == VertexBuffer.PositionKind) {\n minMax = _GLTFUtilities._CalculateMinMaxPositions(vertexData, 0, vertexData.length / stride);\n }\n const accessor = _GLTFUtilities._CreateAccessor(bufferViewIndex, attributeKind + \" - \" + babylonTransformNode.name, attribute.accessorType, attribute.accessorComponentType, vertexData.length / stride, 0, minMax.min, minMax.max);\n this._accessors.push(accessor);\n this._setAttributeKind(meshPrimitive, attributeKind);\n }\n }\n }\n }\n if (indexBufferViewIndex) {\n // Create accessor\n const accessor = _GLTFUtilities._CreateAccessor(indexBufferViewIndex, \"indices - \" + babylonTransformNode.name, \"SCALAR\" /* AccessorType.SCALAR */, 5125 /* AccessorComponentType.UNSIGNED_INT */, submesh.indexCount, submesh.indexStart * 4, null, null);\n this._accessors.push(accessor);\n meshPrimitive.indices = this._accessors.length - 1;\n }\n if (materialIndex != null && Object.keys(meshPrimitive.attributes).length > 0) {\n const sideOrientation = bufferMesh.overrideMaterialSideOrientation !== null ? bufferMesh.overrideMaterialSideOrientation : babylonMaterial.sideOrientation;\n if (sideOrientation === (this._babylonScene.useRightHandedSystem ? Material.ClockWiseSideOrientation : Material.CounterClockWiseSideOrientation)) {\n let byteOffset = indexBufferViewIndex != null ? this._bufferViews[indexBufferViewIndex].byteOffset : null;\n if (byteOffset == null) {\n byteOffset = 0;\n }\n let babylonIndices = null;\n if (indexBufferViewIndex != null) {\n babylonIndices = bufferMesh.getIndices();\n }\n if (babylonIndices) {\n this._reorderIndicesBasedOnPrimitiveMode(submesh, primitiveMode, babylonIndices, byteOffset, binaryWriter);\n }\n else {\n for (const attribute of attributeData) {\n const vertexData = bufferMesh.getVerticesData(attribute.kind, undefined, undefined, true);\n if (vertexData) {\n const byteOffset = this._bufferViews[vertexAttributeBufferViews[attribute.kind]].byteOffset || 0;\n this._reorderVertexAttributeDataBasedOnPrimitiveMode(submesh, primitiveMode, attribute.kind, vertexData, byteOffset, binaryWriter);\n }\n }\n }\n }\n meshPrimitive.material = materialIndex;\n }\n if (morphTargetManager) {\n let target;\n for (let i = 0; i < morphTargetManager.numTargets; ++i) {\n target = morphTargetManager.getTarget(i);\n this._setMorphTargetAttributes(submesh, meshPrimitive, target, binaryWriter);\n }\n }\n mesh.primitives.push(meshPrimitive);\n this._extensionsPostExportMeshPrimitiveAsync(\"postExport\", meshPrimitive, submesh, binaryWriter);\n promises.push();\n }\n }\n }\n return Promise.all(promises).then(() => {\n /* do nothing */\n });\n }\n /**\n * Creates a glTF scene based on the array of meshes\n * Returns the the total byte offset\n * @param babylonScene Babylon scene to get the mesh data from\n * @param binaryWriter Buffer to write binary data to\n */\n _createSceneAsync(binaryWriter) {\n var _a;\n const scene = { nodes: [] };\n let glTFNodeIndex;\n let glTFNode;\n let directDescendents;\n const nodes = [...this._babylonScene.transformNodes, ...this._babylonScene.meshes, ...this._babylonScene.lights, ...this._babylonScene.cameras];\n const removedRootNodes = new Set();\n // Scene metadata\n if (this._babylonScene.metadata) {\n if (this._options.metadataSelector) {\n scene.extras = this._options.metadataSelector(this._babylonScene.metadata);\n }\n else if (this._babylonScene.metadata.gltf) {\n scene.extras = this._babylonScene.metadata.gltf.extras;\n }\n }\n // Remove no-op root nodes\n if (((_a = this._options.removeNoopRootNodes) !== null && _a !== void 0 ? _a : true) && !this._options.includeCoordinateSystemConversionNodes) {\n for (const rootNode of this._babylonScene.rootNodes) {\n if (isNoopNode(rootNode, this._babylonScene.useRightHandedSystem)) {\n removedRootNodes.add(rootNode);\n // Exclude the node from list of nodes to export\n nodes.splice(nodes.indexOf(rootNode), 1);\n }\n }\n }\n // Export babylon cameras to glTFCamera\n const cameraMap = new Map();\n this._babylonScene.cameras.forEach((camera) => {\n if (!this._options.shouldExportNode || this._options.shouldExportNode(camera)) {\n const glTFCamera = {\n type: camera.mode === Camera.PERSPECTIVE_CAMERA ? \"perspective\" /* CameraType.PERSPECTIVE */ : \"orthographic\" /* CameraType.ORTHOGRAPHIC */,\n };\n if (camera.name) {\n glTFCamera.name = camera.name;\n }\n if (glTFCamera.type === \"perspective\" /* CameraType.PERSPECTIVE */) {\n glTFCamera.perspective = {\n aspectRatio: camera.getEngine().getAspectRatio(camera),\n yfov: camera.fovMode === Camera.FOVMODE_VERTICAL_FIXED ? camera.fov : camera.fov * camera.getEngine().getAspectRatio(camera),\n znear: camera.minZ,\n zfar: camera.maxZ,\n };\n }\n else if (glTFCamera.type === \"orthographic\" /* CameraType.ORTHOGRAPHIC */) {\n const halfWidth = camera.orthoLeft && camera.orthoRight ? 0.5 * (camera.orthoRight - camera.orthoLeft) : camera.getEngine().getRenderWidth() * 0.5;\n const halfHeight = camera.orthoBottom && camera.orthoTop ? 0.5 * (camera.orthoTop - camera.orthoBottom) : camera.getEngine().getRenderHeight() * 0.5;\n glTFCamera.orthographic = {\n xmag: halfWidth,\n ymag: halfHeight,\n znear: camera.minZ,\n zfar: camera.maxZ,\n };\n }\n cameraMap.set(camera, this._cameras.length);\n this._cameras.push(glTFCamera);\n }\n });\n const [exportNodes, exportMaterials] = this._getExportNodes(nodes);\n return this._glTFMaterialExporter._convertMaterialsToGLTFAsync(exportMaterials, \"image/png\" /* ImageMimeType.PNG */, true).then(() => {\n return this._createNodeMapAndAnimationsAsync(exportNodes, binaryWriter).then((nodeMap) => {\n return this._createSkinsAsync(nodeMap, binaryWriter).then((skinMap) => {\n this._nodeMap = nodeMap;\n this._totalByteLength = binaryWriter.getByteOffset();\n if (this._totalByteLength == undefined) {\n throw new Error(\"undefined byte length!\");\n }\n // Build Hierarchy with the node map.\n for (const babylonNode of nodes) {\n glTFNodeIndex = this._nodeMap[babylonNode.uniqueId];\n if (glTFNodeIndex !== undefined) {\n glTFNode = this._nodes[glTFNodeIndex];\n if (babylonNode.metadata) {\n if (this._options.metadataSelector) {\n glTFNode.extras = this._options.metadataSelector(babylonNode.metadata);\n }\n else if (babylonNode.metadata.gltf) {\n glTFNode.extras = babylonNode.metadata.gltf.extras;\n }\n }\n if (babylonNode instanceof Camera) {\n glTFNode.camera = cameraMap.get(babylonNode);\n }\n if (this._options.shouldExportNode && !this._options.shouldExportNode(babylonNode)) {\n Tools.Log(\"Omitting \" + babylonNode.name + \" from scene.\");\n }\n else {\n if (!babylonNode.parent && !this._babylonScene.useRightHandedSystem) {\n convertNodeHandedness(glTFNode);\n }\n if (!babylonNode.parent || removedRootNodes.has(babylonNode.parent)) {\n scene.nodes.push(glTFNodeIndex);\n }\n }\n if (babylonNode instanceof Mesh) {\n if (babylonNode.skeleton) {\n glTFNode.skin = skinMap[babylonNode.skeleton.uniqueId];\n }\n }\n directDescendents = babylonNode.getDescendants(true);\n if (!glTFNode.children && directDescendents && directDescendents.length) {\n const children = [];\n for (const descendent of directDescendents) {\n if (this._nodeMap[descendent.uniqueId] != null) {\n children.push(this._nodeMap[descendent.uniqueId]);\n }\n }\n if (children.length) {\n glTFNode.children = children;\n }\n }\n }\n }\n if (scene.nodes.length) {\n this._scenes.push(scene);\n }\n });\n });\n });\n }\n /**\n * Getting the nodes and materials that would be exported.\n * @param nodes Babylon transform nodes\n * @returns Array of nodes which would be exported.\n * @returns Set of materials which would be exported.\n */\n _getExportNodes(nodes) {\n const exportNodes = [];\n const exportMaterials = new Set();\n for (const babylonNode of nodes) {\n if (!this._options.shouldExportNode || this._options.shouldExportNode(babylonNode)) {\n exportNodes.push(babylonNode);\n const babylonMesh = babylonNode;\n if (babylonMesh.subMeshes && babylonMesh.subMeshes.length > 0) {\n const material = babylonMesh.material || babylonMesh.getScene().defaultMaterial;\n if (material instanceof MultiMaterial) {\n for (const subMaterial of material.subMaterials) {\n if (subMaterial) {\n exportMaterials.add(subMaterial);\n }\n }\n }\n else {\n exportMaterials.add(material);\n }\n }\n }\n else {\n `Excluding node ${babylonNode.name}`;\n }\n }\n return [exportNodes, exportMaterials];\n }\n /**\n * Creates a mapping of Node unique id to node index and handles animations\n * @param nodes Babylon transform nodes\n * @param binaryWriter Buffer to write binary data to\n * @returns Node mapping of unique id to index\n */\n _createNodeMapAndAnimationsAsync(nodes, binaryWriter) {\n let promiseChain = Promise.resolve();\n const nodeMap = {};\n let nodeIndex;\n const runtimeGLTFAnimation = {\n name: \"runtime animations\",\n channels: [],\n samplers: [],\n };\n const idleGLTFAnimations = [];\n for (const babylonNode of nodes) {\n promiseChain = promiseChain.then(() => {\n return this._createNodeAsync(babylonNode, binaryWriter).then((node) => {\n const promise = this._extensionsPostExportNodeAsync(\"createNodeAsync\", node, babylonNode, nodeMap, binaryWriter);\n if (promise == null) {\n Tools.Warn(`Not exporting node ${babylonNode.name}`);\n return Promise.resolve();\n }\n else {\n return promise.then((node) => {\n if (!node) {\n return;\n }\n this._nodes.push(node);\n nodeIndex = this._nodes.length - 1;\n nodeMap[babylonNode.uniqueId] = nodeIndex;\n if (!this._babylonScene.animationGroups.length) {\n _GLTFAnimation._CreateMorphTargetAnimationFromMorphTargetAnimations(babylonNode, runtimeGLTFAnimation, idleGLTFAnimations, nodeMap, this._nodes, binaryWriter, this._bufferViews, this._accessors, this._animationSampleRate, this._options.shouldExportAnimation);\n if (babylonNode.animations.length) {\n _GLTFAnimation._CreateNodeAnimationFromNodeAnimations(babylonNode, runtimeGLTFAnimation, idleGLTFAnimations, nodeMap, this._nodes, binaryWriter, this._bufferViews, this._accessors, this._animationSampleRate, this._options.shouldExportAnimation);\n }\n }\n });\n }\n });\n });\n }\n return promiseChain.then(() => {\n if (runtimeGLTFAnimation.channels.length && runtimeGLTFAnimation.samplers.length) {\n this._animations.push(runtimeGLTFAnimation);\n }\n idleGLTFAnimations.forEach((idleGLTFAnimation) => {\n if (idleGLTFAnimation.channels.length && idleGLTFAnimation.samplers.length) {\n this._animations.push(idleGLTFAnimation);\n }\n });\n if (this._babylonScene.animationGroups.length) {\n _GLTFAnimation._CreateNodeAndMorphAnimationFromAnimationGroups(this._babylonScene, this._animations, nodeMap, binaryWriter, this._bufferViews, this._accessors, this._animationSampleRate, this._options.shouldExportAnimation);\n }\n return nodeMap;\n });\n }\n /**\n * Creates a glTF node from a Babylon mesh\n * @param babylonNode Source Babylon mesh\n * @param binaryWriter Buffer for storing geometry data\n * @returns glTF node\n */\n _createNodeAsync(babylonNode, binaryWriter) {\n return Promise.resolve().then(() => {\n // create node to hold translation/rotation/scale and the mesh\n const node = {};\n // create mesh\n const mesh = { primitives: [] };\n if (babylonNode.name) {\n node.name = babylonNode.name;\n }\n if (babylonNode instanceof TransformNode) {\n // Set transformation\n this._setNodeTransformation(node, babylonNode);\n if (babylonNode instanceof Mesh) {\n const morphTargetManager = babylonNode.morphTargetManager;\n if (morphTargetManager && morphTargetManager.numTargets > 0) {\n mesh.weights = [];\n for (let i = 0; i < morphTargetManager.numTargets; ++i) {\n mesh.weights.push(morphTargetManager.getTarget(i).influence);\n }\n }\n }\n return this._setPrimitiveAttributesAsync(mesh, babylonNode, binaryWriter).then(() => {\n if (mesh.primitives.length) {\n this._meshes.push(mesh);\n node.mesh = this._meshes.length - 1;\n }\n return node;\n });\n }\n else if (babylonNode instanceof Camera) {\n this._setCameraTransformation(node, babylonNode);\n return node;\n }\n else {\n return node;\n }\n });\n }\n /**\n * Creates a glTF skin from a Babylon skeleton\n * @param babylonScene Babylon Scene\n * @param nodeMap Babylon transform nodes\n * @param binaryWriter Buffer to write binary data to\n * @returns Node mapping of unique id to index\n */\n _createSkinsAsync(nodeMap, binaryWriter) {\n var _a;\n const promiseChain = Promise.resolve();\n const skinMap = {};\n for (const skeleton of this._babylonScene.skeletons) {\n if (skeleton.bones.length <= 0) {\n continue;\n }\n // create skin\n const skin = { joints: [] };\n const inverseBindMatrices = [];\n const boneIndexMap = {};\n let maxBoneIndex = -1;\n for (let i = 0; i < skeleton.bones.length; ++i) {\n const bone = skeleton.bones[i];\n const boneIndex = (_a = bone.getIndex()) !== null && _a !== void 0 ? _a : i;\n if (boneIndex !== -1) {\n boneIndexMap[boneIndex] = bone;\n if (boneIndex > maxBoneIndex) {\n maxBoneIndex = boneIndex;\n }\n }\n }\n for (let boneIndex = 0; boneIndex <= maxBoneIndex; ++boneIndex) {\n const bone = boneIndexMap[boneIndex];\n inverseBindMatrices.push(bone.getInvertedAbsoluteTransform());\n const transformNode = bone.getTransformNode();\n if (transformNode && nodeMap[transformNode.uniqueId] !== null && nodeMap[transformNode.uniqueId] !== undefined) {\n skin.joints.push(nodeMap[transformNode.uniqueId]);\n }\n else {\n Tools.Warn(\"Exporting a bone without a linked transform node is currently unsupported\");\n }\n }\n if (skin.joints.length > 0) {\n // create buffer view for inverse bind matrices\n const byteStride = 64; // 4 x 4 matrix of 32 bit float\n const byteLength = inverseBindMatrices.length * byteStride;\n const bufferViewOffset = binaryWriter.getByteOffset();\n const bufferView = _GLTFUtilities._CreateBufferView(0, bufferViewOffset, byteLength, undefined, \"InverseBindMatrices\" + \" - \" + skeleton.name);\n this._bufferViews.push(bufferView);\n const bufferViewIndex = this._bufferViews.length - 1;\n const bindMatrixAccessor = _GLTFUtilities._CreateAccessor(bufferViewIndex, \"InverseBindMatrices\" + \" - \" + skeleton.name, \"MAT4\" /* AccessorType.MAT4 */, 5126 /* AccessorComponentType.FLOAT */, inverseBindMatrices.length, null, null, null);\n const inverseBindAccessorIndex = this._accessors.push(bindMatrixAccessor) - 1;\n skin.inverseBindMatrices = inverseBindAccessorIndex;\n this._skins.push(skin);\n skinMap[skeleton.uniqueId] = this._skins.length - 1;\n inverseBindMatrices.forEach((mat) => {\n mat.m.forEach((cell) => {\n binaryWriter.setFloat32(cell);\n });\n });\n }\n }\n return promiseChain.then(() => {\n return skinMap;\n });\n }\n }\n _Exporter._ExtensionNames = new Array();\n _Exporter._ExtensionFactories = {};\n /**\n * @internal\n *\n * Stores glTF binary data. If the array buffer byte length is exceeded, it doubles in size dynamically\n */\n class _BinaryWriter {\n /**\n * Initialize binary writer with an initial byte length\n * @param byteLength Initial byte length of the array buffer\n */\n constructor(byteLength) {\n this._arrayBuffer = new ArrayBuffer(byteLength);\n this._dataView = new DataView(this._arrayBuffer);\n this._byteOffset = 0;\n }\n /**\n * Resize the array buffer to the specified byte length\n * @param byteLength\n */\n _resizeBuffer(byteLength) {\n const newBuffer = new ArrayBuffer(byteLength);\n const copyOldBufferSize = Math.min(this._arrayBuffer.byteLength, byteLength);\n const oldUint8Array = new Uint8Array(this._arrayBuffer, 0, copyOldBufferSize);\n const newUint8Array = new Uint8Array(newBuffer);\n newUint8Array.set(oldUint8Array, 0);\n this._arrayBuffer = newBuffer;\n this._dataView = new DataView(this._arrayBuffer);\n return newBuffer;\n }\n /**\n * Get an array buffer with the length of the byte offset\n * @returns ArrayBuffer resized to the byte offset\n */\n getArrayBuffer() {\n return this._resizeBuffer(this.getByteOffset());\n }\n /**\n * Get the byte offset of the array buffer\n * @returns byte offset\n */\n getByteOffset() {\n if (this._byteOffset == undefined) {\n throw new Error(\"Byte offset is undefined!\");\n }\n return this._byteOffset;\n }\n /**\n * Stores an UInt8 in the array buffer\n * @param entry\n * @param byteOffset If defined, specifies where to set the value as an offset.\n */\n setUInt8(entry, byteOffset) {\n if (byteOffset != null) {\n if (byteOffset < this._byteOffset) {\n this._dataView.setUint8(byteOffset, entry);\n }\n else {\n Tools.Error(\"BinaryWriter: byteoffset is greater than the current binary buffer length!\");\n }\n }\n else {\n if (this._byteOffset + 1 > this._arrayBuffer.byteLength) {\n this._resizeBuffer(this._arrayBuffer.byteLength * 2);\n }\n this._dataView.setUint8(this._byteOffset, entry);\n this._byteOffset += 1;\n }\n }\n /**\n * Stores an UInt16 in the array buffer\n * @param entry\n * @param byteOffset If defined, specifies where to set the value as an offset.\n */\n setUInt16(entry, byteOffset) {\n if (byteOffset != null) {\n if (byteOffset < this._byteOffset) {\n this._dataView.setUint16(byteOffset, entry, true);\n }\n else {\n Tools.Error(\"BinaryWriter: byteoffset is greater than the current binary buffer length!\");\n }\n }\n else {\n if (this._byteOffset + 2 > this._arrayBuffer.byteLength) {\n this._resizeBuffer(this._arrayBuffer.byteLength * 2);\n }\n this._dataView.setUint16(this._byteOffset, entry, true);\n this._byteOffset += 2;\n }\n }\n /**\n * Gets an UInt32 in the array buffer\n * @param byteOffset If defined, specifies where to set the value as an offset.\n */\n getUInt32(byteOffset) {\n if (byteOffset < this._byteOffset) {\n return this._dataView.getUint32(byteOffset, true);\n }\n else {\n Tools.Error(\"BinaryWriter: byteoffset is greater than the current binary buffer length!\");\n throw new Error(\"BinaryWriter: byteoffset is greater than the current binary buffer length!\");\n }\n }\n getVector3Float32FromRef(vector3, byteOffset) {\n if (byteOffset + 8 > this._byteOffset) {\n Tools.Error(`BinaryWriter: byteoffset is greater than the current binary buffer length!`);\n }\n else {\n vector3.x = this._dataView.getFloat32(byteOffset, true);\n vector3.y = this._dataView.getFloat32(byteOffset + 4, true);\n vector3.z = this._dataView.getFloat32(byteOffset + 8, true);\n }\n }\n setVector3Float32FromRef(vector3, byteOffset) {\n if (byteOffset + 8 > this._byteOffset) {\n Tools.Error(`BinaryWriter: byteoffset is greater than the current binary buffer length!`);\n }\n else {\n this._dataView.setFloat32(byteOffset, vector3.x, true);\n this._dataView.setFloat32(byteOffset + 4, vector3.y, true);\n this._dataView.setFloat32(byteOffset + 8, vector3.z, true);\n }\n }\n getVector4Float32FromRef(vector4, byteOffset) {\n if (byteOffset + 12 > this._byteOffset) {\n Tools.Error(`BinaryWriter: byteoffset is greater than the current binary buffer length!`);\n }\n else {\n vector4.x = this._dataView.getFloat32(byteOffset, true);\n vector4.y = this._dataView.getFloat32(byteOffset + 4, true);\n vector4.z = this._dataView.getFloat32(byteOffset + 8, true);\n vector4.w = this._dataView.getFloat32(byteOffset + 12, true);\n }\n }\n setVector4Float32FromRef(vector4, byteOffset) {\n if (byteOffset + 12 > this._byteOffset) {\n Tools.Error(`BinaryWriter: byteoffset is greater than the current binary buffer length!`);\n }\n else {\n this._dataView.setFloat32(byteOffset, vector4.x, true);\n this._dataView.setFloat32(byteOffset + 4, vector4.y, true);\n this._dataView.setFloat32(byteOffset + 8, vector4.z, true);\n this._dataView.setFloat32(byteOffset + 12, vector4.w, true);\n }\n }\n /**\n * Stores a Float32 in the array buffer\n * @param entry\n * @param byteOffset\n */\n setFloat32(entry, byteOffset) {\n if (isNaN(entry)) {\n Tools.Error(\"Invalid data being written!\");\n }\n if (byteOffset != null) {\n if (byteOffset < this._byteOffset) {\n this._dataView.setFloat32(byteOffset, entry, true);\n }\n else {\n Tools.Error(\"BinaryWriter: byteoffset is greater than the current binary length!\");\n }\n }\n if (this._byteOffset + 4 > this._arrayBuffer.byteLength) {\n this._resizeBuffer(this._arrayBuffer.byteLength * 2);\n }\n this._dataView.setFloat32(this._byteOffset, entry, true);\n this._byteOffset += 4;\n }\n /**\n * Stores an UInt32 in the array buffer\n * @param entry\n * @param byteOffset If defined, specifies where to set the value as an offset.\n */\n setUInt32(entry, byteOffset) {\n if (byteOffset != null) {\n if (byteOffset < this._byteOffset) {\n this._dataView.setUint32(byteOffset, entry, true);\n }\n else {\n Tools.Error(\"BinaryWriter: byteoffset is greater than the current binary buffer length!\");\n }\n }\n else {\n if (this._byteOffset + 4 > this._arrayBuffer.byteLength) {\n this._resizeBuffer(this._arrayBuffer.byteLength * 2);\n }\n this._dataView.setUint32(this._byteOffset, entry, true);\n this._byteOffset += 4;\n }\n }\n /**\n * Stores an Int16 in the array buffer\n * @param entry\n * @param byteOffset If defined, specifies where to set the value as an offset.\n */\n setInt16(entry, byteOffset) {\n if (byteOffset != null) {\n if (byteOffset < this._byteOffset) {\n this._dataView.setInt16(byteOffset, entry, true);\n }\n else {\n Tools.Error(\"BinaryWriter: byteoffset is greater than the current binary buffer length!\");\n }\n }\n else {\n if (this._byteOffset + 2 > this._arrayBuffer.byteLength) {\n this._resizeBuffer(this._arrayBuffer.byteLength * 2);\n }\n this._dataView.setInt16(this._byteOffset, entry, true);\n this._byteOffset += 2;\n }\n }\n /**\n * Stores a byte in the array buffer\n * @param entry\n * @param byteOffset If defined, specifies where to set the value as an offset.\n */\n setByte(entry, byteOffset) {\n if (byteOffset != null) {\n if (byteOffset < this._byteOffset) {\n this._dataView.setInt8(byteOffset, entry);\n }\n else {\n Tools.Error(\"BinaryWriter: byteoffset is greater than the current binary buffer length!\");\n }\n }\n else {\n if (this._byteOffset + 1 > this._arrayBuffer.byteLength) {\n this._resizeBuffer(this._arrayBuffer.byteLength * 2);\n }\n this._dataView.setInt8(this._byteOffset, entry);\n this._byteOffset++;\n }\n }\n }\n\n /**\n * Class for generating glTF data from a Babylon scene.\n */\n class GLTF2Export {\n /**\n * Exports the geometry of the scene to .gltf file format asynchronously\n * @param scene Babylon scene with scene hierarchy information\n * @param filePrefix File prefix to use when generating the glTF file\n * @param options Exporter options\n * @returns Returns an object with a .gltf file and associates texture names\n * as keys and their data and paths as values\n */\n static GLTFAsync(scene, filePrefix, options) {\n return scene.whenReadyAsync().then(() => {\n const glTFPrefix = filePrefix.replace(/\\.[^/.]+$/, \"\");\n const gltfGenerator = new _Exporter(scene, options);\n return gltfGenerator._generateGLTFAsync(glTFPrefix);\n });\n }\n static _PreExportAsync(scene, options) {\n return Promise.resolve().then(() => {\n if (options && options.exportWithoutWaitingForScene) {\n return Promise.resolve();\n }\n else {\n return scene.whenReadyAsync();\n }\n });\n }\n static _PostExportAsync(scene, glTFData, options) {\n return Promise.resolve().then(() => {\n if (options && options.exportWithoutWaitingForScene) {\n return glTFData;\n }\n else {\n return glTFData;\n }\n });\n }\n /**\n * Exports the geometry of the scene to .glb file format asychronously\n * @param scene Babylon scene with scene hierarchy information\n * @param filePrefix File prefix to use when generating glb file\n * @param options Exporter options\n * @returns Returns an object with a .glb filename as key and data as value\n */\n static GLBAsync(scene, filePrefix, options) {\n return this._PreExportAsync(scene, options).then(() => {\n const glTFPrefix = filePrefix.replace(/\\.[^/.]+$/, \"\");\n const gltfGenerator = new _Exporter(scene, options);\n return gltfGenerator._generateGLBAsync(glTFPrefix).then((glTFData) => {\n return this._PostExportAsync(scene, glTFData, options);\n });\n });\n }\n }\n\n const NAME = \"KHR_texture_transform\";\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class KHR_texture_transform {\n constructor() {\n /** Name of this extension */\n this.name = NAME;\n /** Defines whether this extension is enabled */\n this.enabled = true;\n /** Defines whether this extension is required */\n this.required = false;\n /** Reference to the glTF exporter */\n this._wasUsed = false;\n }\n dispose() { }\n /** @internal */\n get wasUsed() {\n return this._wasUsed;\n }\n postExportTexture(context, textureInfo, babylonTexture) {\n const canUseExtension = babylonTexture &&\n ((babylonTexture.uAng === 0 && babylonTexture.wAng === 0 && babylonTexture.vAng === 0) ||\n (babylonTexture.uRotationCenter === 0 && babylonTexture.vRotationCenter === 0));\n if (canUseExtension) {\n const textureTransform = {};\n let transformIsRequired = false;\n if (babylonTexture.uOffset !== 0 || babylonTexture.vOffset !== 0) {\n textureTransform.offset = [babylonTexture.uOffset, babylonTexture.vOffset];\n transformIsRequired = true;\n }\n if (babylonTexture.uScale !== 1 || babylonTexture.vScale !== 1) {\n textureTransform.scale = [babylonTexture.uScale, babylonTexture.vScale];\n transformIsRequired = true;\n }\n if (babylonTexture.wAng !== 0) {\n textureTransform.rotation = -babylonTexture.wAng;\n transformIsRequired = true;\n }\n if (babylonTexture.coordinatesIndex !== 0) {\n textureTransform.texCoord = babylonTexture.coordinatesIndex;\n transformIsRequired = true;\n }\n if (!transformIsRequired) {\n return;\n }\n this._wasUsed = true;\n if (!textureInfo.extensions) {\n textureInfo.extensions = {};\n }\n textureInfo.extensions[NAME] = textureTransform;\n }\n }\n preExportTextureAsync(context, babylonTexture) {\n return new Promise((resolve, reject) => {\n const scene = babylonTexture.getScene();\n if (!scene) {\n reject(`${context}: \"scene\" is not defined for Babylon texture ${babylonTexture.name}!`);\n return;\n }\n /*\n * The KHR_texture_transform schema only supports w rotation around the origin.\n * See https://github.com/KhronosGroup/glTF/tree/main/extensions/2.0/Khronos/KHR_texture_transform#gltf-schema-updates.\n */\n if (babylonTexture.uAng !== 0 || babylonTexture.vAng !== 0) {\n Tools.Warn(`${context}: Texture ${babylonTexture.name} with rotation in the u or v axis is not supported in glTF.`);\n resolve(null);\n }\n else if (babylonTexture.wAng !== 0 && (babylonTexture.uRotationCenter !== 0 || babylonTexture.vRotationCenter !== 0)) {\n Tools.Warn(`${context}: Texture ${babylonTexture.name} with rotation not centered at the origin cannot be exported with ${NAME}`);\n resolve(null);\n }\n else {\n resolve(babylonTexture);\n }\n });\n }\n }\n _Exporter.RegisterExtension(NAME, () => new KHR_texture_transform());\n\n const NAME$1 = \"KHR_lights_punctual\";\n /**\n * [Specification](https://github.com/KhronosGroup/glTF/blob/master/extensions/2.0/Khronos/KHR_lights_punctual/README.md)\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class KHR_lights_punctual {\n /**\n * @internal\n */\n constructor(exporter) {\n /** The name of this extension. */\n this.name = NAME$1;\n /** Defines whether this extension is enabled. */\n this.enabled = true;\n /** Defines whether this extension is required */\n this.required = false;\n this._exporter = exporter;\n }\n /** @internal */\n dispose() {\n this._lights = null;\n }\n /** @internal */\n get wasUsed() {\n return !!this._lights;\n }\n /** @internal */\n onExporting() {\n this._exporter._glTF.extensions[NAME$1] = this._lights;\n }\n /**\n * Define this method to modify the default behavior when exporting a node\n * @param context The context when exporting the node\n * @param node glTF node\n * @param babylonNode BabylonJS node\n * @param nodeMap Node mapping of unique id to glTF node index\n * @returns nullable INode promise\n */\n postExportNodeAsync(context, node, babylonNode, nodeMap) {\n return new Promise((resolve) => {\n if (node && babylonNode instanceof ShadowLight) {\n let light;\n const lightType = babylonNode.getTypeID() == Light.LIGHTTYPEID_POINTLIGHT\n ? \"point\" /* KHRLightsPunctual_LightType.POINT */\n : babylonNode.getTypeID() == Light.LIGHTTYPEID_DIRECTIONALLIGHT\n ? \"directional\" /* KHRLightsPunctual_LightType.DIRECTIONAL */\n : babylonNode.getTypeID() == Light.LIGHTTYPEID_SPOTLIGHT\n ? \"spot\" /* KHRLightsPunctual_LightType.SPOT */\n : null;\n if (lightType == null) {\n Logger.Warn(`${context}: Light ${babylonNode.name} is not supported in ${NAME$1}`);\n }\n else {\n if (!babylonNode.position.equalsToFloats(0, 0, 0)) {\n node.translation = babylonNode.position.asArray();\n }\n if (lightType !== \"point\" /* KHRLightsPunctual_LightType.POINT */) {\n const localAxis = babylonNode.direction;\n const yaw = -Math.atan2(localAxis.z, localAxis.x) + Math.PI / 2;\n const len = Math.sqrt(localAxis.x * localAxis.x + localAxis.z * localAxis.z);\n const pitch = -Math.atan2(localAxis.y, len);\n const lightRotationQuaternion = Quaternion.RotationYawPitchRoll(yaw + Math.PI, pitch, 0);\n if (!Quaternion.IsIdentity(lightRotationQuaternion)) {\n node.rotation = lightRotationQuaternion.asArray();\n }\n }\n if (babylonNode.falloffType !== Light.FALLOFF_GLTF) {\n Logger.Warn(`${context}: Light falloff for ${babylonNode.name} does not match the ${NAME$1} specification!`);\n }\n light = {\n type: lightType,\n };\n if (!babylonNode.diffuse.equals(Color3.White())) {\n light.color = babylonNode.diffuse.asArray();\n }\n if (babylonNode.intensity !== 1.0) {\n light.intensity = babylonNode.intensity;\n }\n if (babylonNode.range !== Number.MAX_VALUE) {\n light.range = babylonNode.range;\n }\n if (lightType === \"spot\" /* KHRLightsPunctual_LightType.SPOT */) {\n const babylonSpotLight = babylonNode;\n if (babylonSpotLight.angle !== Math.PI / 2.0) {\n if (light.spot == null) {\n light.spot = {};\n }\n light.spot.outerConeAngle = babylonSpotLight.angle / 2.0;\n }\n if (babylonSpotLight.innerAngle !== 0) {\n if (light.spot == null) {\n light.spot = {};\n }\n light.spot.innerConeAngle = babylonSpotLight.innerAngle / 2.0;\n }\n }\n this._lights || (this._lights = {\n lights: [],\n });\n this._lights.lights.push(light);\n const lightReference = {\n light: this._lights.lights.length - 1,\n };\n // Avoid duplicating the Light's parent node if possible.\n const parentBabylonNode = babylonNode.parent;\n if (parentBabylonNode && parentBabylonNode.getChildren().length == 1) {\n const parentNode = this._exporter._nodes[nodeMap[parentBabylonNode.uniqueId]];\n if (parentNode) {\n const parentTranslation = Vector3.FromArrayToRef(parentNode.translation || [0, 0, 0], 0, TmpVectors.Vector3[0]);\n const parentRotation = Quaternion.FromArrayToRef(parentNode.rotation || [0, 0, 0, 1], 0, TmpVectors.Quaternion[0]);\n const parentScale = Vector3.FromArrayToRef(parentNode.scale || [1, 1, 1], 0, TmpVectors.Vector3[1]);\n const parentMatrix = Matrix.ComposeToRef(parentScale, parentRotation, parentTranslation, TmpVectors.Matrix[0]);\n const translation = Vector3.FromArrayToRef(node.translation || [0, 0, 0], 0, TmpVectors.Vector3[2]);\n const rotation = Quaternion.FromArrayToRef(node.rotation || [0, 0, 0, 1], 0, TmpVectors.Quaternion[1]);\n const matrix = Matrix.ComposeToRef(Vector3.OneReadOnly, rotation, translation, TmpVectors.Matrix[1]);\n parentMatrix.multiplyToRef(matrix, matrix);\n matrix.decompose(parentScale, parentRotation, parentTranslation);\n if (parentTranslation.equalsToFloats(0, 0, 0)) {\n delete parentNode.translation;\n }\n else {\n parentNode.translation = parentTranslation.asArray();\n }\n if (Quaternion.IsIdentity(parentRotation)) {\n delete parentNode.rotation;\n }\n else {\n parentNode.rotation = parentRotation.asArray();\n }\n if (parentScale.equalsToFloats(1, 1, 1)) {\n delete parentNode.scale;\n }\n else {\n parentNode.scale = parentScale.asArray();\n }\n parentNode.extensions || (parentNode.extensions = {});\n parentNode.extensions[NAME$1] = lightReference;\n // Do not export the original node\n resolve(null);\n return;\n }\n }\n node.extensions || (node.extensions = {});\n node.extensions[NAME$1] = lightReference;\n }\n }\n resolve(node);\n });\n }\n }\n _Exporter.RegisterExtension(NAME$1, (exporter) => new KHR_lights_punctual(exporter));\n\n const NAME$2 = \"KHR_materials_clearcoat\";\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class KHR_materials_clearcoat {\n constructor(exporter) {\n /** Name of this extension */\n this.name = NAME$2;\n /** Defines whether this extension is enabled */\n this.enabled = true;\n /** Defines whether this extension is required */\n this.required = false;\n this._wasUsed = false;\n this._exporter = exporter;\n }\n dispose() { }\n /** @internal */\n get wasUsed() {\n return this._wasUsed;\n }\n postExportMaterialAdditionalTextures(context, node, babylonMaterial) {\n const additionalTextures = [];\n if (babylonMaterial instanceof PBRBaseMaterial) {\n if (babylonMaterial.clearCoat.isEnabled) {\n if (babylonMaterial.clearCoat.texture) {\n additionalTextures.push(babylonMaterial.clearCoat.texture);\n }\n if (!babylonMaterial.clearCoat.useRoughnessFromMainTexture && babylonMaterial.clearCoat.textureRoughness) {\n additionalTextures.push(babylonMaterial.clearCoat.textureRoughness);\n }\n if (babylonMaterial.clearCoat.bumpTexture) {\n additionalTextures.push(babylonMaterial.clearCoat.bumpTexture);\n }\n return additionalTextures;\n }\n }\n return [];\n }\n postExportMaterialAsync(context, node, babylonMaterial) {\n return new Promise((resolve) => {\n if (babylonMaterial instanceof PBRBaseMaterial) {\n if (!babylonMaterial.clearCoat.isEnabled) {\n resolve(node);\n return;\n }\n this._wasUsed = true;\n node.extensions = node.extensions || {};\n const clearCoatTextureInfo = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.clearCoat.texture);\n let clearCoatTextureRoughnessInfo;\n if (babylonMaterial.clearCoat.useRoughnessFromMainTexture) {\n clearCoatTextureRoughnessInfo = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.clearCoat.texture);\n }\n else {\n clearCoatTextureRoughnessInfo = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.clearCoat.textureRoughness);\n }\n if (babylonMaterial.clearCoat.isTintEnabled) {\n Tools.Warn(`Clear Color tint is not supported for glTF export. Ignoring for: ${babylonMaterial.name}`);\n }\n if (babylonMaterial.clearCoat.remapF0OnInterfaceChange) {\n Tools.Warn(`Clear Color F0 remapping is not supported for glTF export. Ignoring for: ${babylonMaterial.name}`);\n }\n const clearCoatNormalTextureInfo = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.clearCoat.bumpTexture);\n const clearCoatInfo = {\n clearcoatFactor: babylonMaterial.clearCoat.intensity,\n clearcoatTexture: clearCoatTextureInfo !== null && clearCoatTextureInfo !== void 0 ? clearCoatTextureInfo : undefined,\n clearcoatRoughnessFactor: babylonMaterial.clearCoat.roughness,\n clearcoatRoughnessTexture: clearCoatTextureRoughnessInfo !== null && clearCoatTextureRoughnessInfo !== void 0 ? clearCoatTextureRoughnessInfo : undefined,\n clearcoatNormalTexture: clearCoatNormalTextureInfo !== null && clearCoatNormalTextureInfo !== void 0 ? clearCoatNormalTextureInfo : undefined,\n hasTextures: () => {\n return clearCoatInfo.clearcoatTexture !== null || clearCoatInfo.clearcoatRoughnessTexture !== null || clearCoatInfo.clearcoatRoughnessTexture !== null;\n },\n };\n node.extensions[NAME$2] = clearCoatInfo;\n }\n resolve(node);\n });\n }\n }\n _Exporter.RegisterExtension(NAME$2, (exporter) => new KHR_materials_clearcoat(exporter));\n\n const NAME$3 = \"KHR_materials_iridescence\";\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class KHR_materials_iridescence {\n constructor(exporter) {\n /** Name of this extension */\n this.name = NAME$3;\n /** Defines whether this extension is enabled */\n this.enabled = true;\n /** Defines whether this extension is required */\n this.required = false;\n this._wasUsed = false;\n this._exporter = exporter;\n }\n dispose() { }\n /** @internal */\n get wasUsed() {\n return this._wasUsed;\n }\n postExportMaterialAdditionalTextures(context, node, babylonMaterial) {\n const additionalTextures = [];\n if (babylonMaterial instanceof PBRBaseMaterial) {\n if (babylonMaterial.iridescence.isEnabled) {\n if (babylonMaterial.iridescence.texture) {\n additionalTextures.push(babylonMaterial.iridescence.texture);\n }\n if (babylonMaterial.iridescence.thicknessTexture && babylonMaterial.iridescence.thicknessTexture !== babylonMaterial.iridescence.texture) {\n additionalTextures.push(babylonMaterial.iridescence.thicknessTexture);\n }\n return additionalTextures;\n }\n }\n return [];\n }\n postExportMaterialAsync(context, node, babylonMaterial) {\n return new Promise((resolve) => {\n if (babylonMaterial instanceof PBRBaseMaterial) {\n if (!babylonMaterial.iridescence.isEnabled) {\n resolve(node);\n return;\n }\n this._wasUsed = true;\n node.extensions = node.extensions || {};\n const iridescenceTextureInfo = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.iridescence.texture);\n const iridescenceThicknessTextureInfo = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.iridescence.thicknessTexture);\n const iridescenceInfo = {\n iridescenceFactor: babylonMaterial.iridescence.intensity,\n iridescenceIor: babylonMaterial.iridescence.indexOfRefraction,\n iridescenceThicknessMinimum: babylonMaterial.iridescence.minimumThickness,\n iridescenceThicknessMaximum: babylonMaterial.iridescence.maximumThickness,\n iridescenceTexture: iridescenceTextureInfo !== null && iridescenceTextureInfo !== void 0 ? iridescenceTextureInfo : undefined,\n iridescenceThicknessTexture: iridescenceThicknessTextureInfo !== null && iridescenceThicknessTextureInfo !== void 0 ? iridescenceThicknessTextureInfo : undefined,\n hasTextures: () => {\n return iridescenceInfo.iridescenceTexture !== null || iridescenceInfo.iridescenceThicknessTexture !== null;\n },\n };\n node.extensions[NAME$3] = iridescenceInfo;\n }\n resolve(node);\n });\n }\n }\n _Exporter.RegisterExtension(NAME$3, (exporter) => new KHR_materials_iridescence(exporter));\n\n const NAME$4 = \"KHR_materials_anisotropy\";\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class KHR_materials_anisotropy {\n constructor(exporter) {\n /** Name of this extension */\n this.name = NAME$4;\n /** Defines whether this extension is enabled */\n this.enabled = true;\n /** Defines whether this extension is required */\n this.required = false;\n this._wasUsed = false;\n this._exporter = exporter;\n }\n dispose() { }\n /** @internal */\n get wasUsed() {\n return this._wasUsed;\n }\n postExportMaterialAdditionalTextures(context, node, babylonMaterial) {\n const additionalTextures = [];\n if (babylonMaterial instanceof PBRBaseMaterial) {\n if (babylonMaterial.anisotropy.isEnabled && !babylonMaterial.anisotropy.legacy) {\n if (babylonMaterial.anisotropy.texture) {\n additionalTextures.push(babylonMaterial.anisotropy.texture);\n }\n return additionalTextures;\n }\n }\n return [];\n }\n postExportMaterialAsync(context, node, babylonMaterial) {\n return new Promise((resolve) => {\n if (babylonMaterial instanceof PBRBaseMaterial) {\n if (!babylonMaterial.anisotropy.isEnabled || babylonMaterial.anisotropy.legacy) {\n resolve(node);\n return;\n }\n this._wasUsed = true;\n node.extensions = node.extensions || {};\n const anisotropyTextureInfo = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.anisotropy.texture);\n const anisotropyInfo = {\n anisotropyStrength: babylonMaterial.anisotropy.intensity,\n anisotropyRotation: babylonMaterial.anisotropy.angle,\n anisotropyTexture: anisotropyTextureInfo !== null && anisotropyTextureInfo !== void 0 ? anisotropyTextureInfo : undefined,\n hasTextures: () => {\n return anisotropyInfo.anisotropyTexture !== null;\n },\n };\n node.extensions[NAME$4] = anisotropyInfo;\n }\n resolve(node);\n });\n }\n }\n _Exporter.RegisterExtension(NAME$4, (exporter) => new KHR_materials_anisotropy(exporter));\n\n const NAME$5 = \"KHR_materials_sheen\";\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class KHR_materials_sheen {\n constructor(exporter) {\n /** Name of this extension */\n this.name = NAME$5;\n /** Defines whether this extension is enabled */\n this.enabled = true;\n /** Defines whether this extension is required */\n this.required = false;\n this._wasUsed = false;\n this._exporter = exporter;\n }\n dispose() { }\n /** @internal */\n get wasUsed() {\n return this._wasUsed;\n }\n postExportMaterialAdditionalTextures(context, node, babylonMaterial) {\n if (babylonMaterial instanceof PBRMaterial) {\n if (babylonMaterial.sheen.isEnabled && babylonMaterial.sheen.texture) {\n return [babylonMaterial.sheen.texture];\n }\n }\n return [];\n }\n postExportMaterialAsync(context, node, babylonMaterial) {\n return new Promise((resolve) => {\n var _a, _b, _c, _d;\n if (babylonMaterial instanceof PBRMaterial) {\n if (!babylonMaterial.sheen.isEnabled) {\n resolve(node);\n return;\n }\n this._wasUsed = true;\n if (node.extensions == null) {\n node.extensions = {};\n }\n const sheenInfo = {\n sheenColorFactor: babylonMaterial.sheen.color.asArray(),\n sheenRoughnessFactor: (_a = babylonMaterial.sheen.roughness) !== null && _a !== void 0 ? _a : 0,\n hasTextures: () => {\n return sheenInfo.sheenColorTexture !== null || sheenInfo.sheenRoughnessTexture !== null;\n },\n };\n if (babylonMaterial.sheen.texture) {\n sheenInfo.sheenColorTexture = (_b = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.sheen.texture)) !== null && _b !== void 0 ? _b : undefined;\n }\n if (babylonMaterial.sheen.textureRoughness && !babylonMaterial.sheen.useRoughnessFromMainTexture) {\n sheenInfo.sheenRoughnessTexture = (_c = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.sheen.textureRoughness)) !== null && _c !== void 0 ? _c : undefined;\n }\n else if (babylonMaterial.sheen.texture && babylonMaterial.sheen.useRoughnessFromMainTexture) {\n sheenInfo.sheenRoughnessTexture = (_d = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.sheen.texture)) !== null && _d !== void 0 ? _d : undefined;\n }\n node.extensions[NAME$5] = sheenInfo;\n }\n resolve(node);\n });\n }\n }\n _Exporter.RegisterExtension(NAME$5, (exporter) => new KHR_materials_sheen(exporter));\n\n const NAME$6 = \"KHR_materials_unlit\";\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class KHR_materials_unlit {\n constructor() {\n /** Name of this extension */\n this.name = NAME$6;\n /** Defines whether this extension is enabled */\n this.enabled = true;\n /** Defines whether this extension is required */\n this.required = false;\n this._wasUsed = false;\n }\n /** @internal */\n get wasUsed() {\n return this._wasUsed;\n }\n dispose() { }\n postExportMaterialAsync(context, node, babylonMaterial) {\n return new Promise((resolve) => {\n let unlitMaterial = false;\n if (babylonMaterial instanceof PBRMaterial) {\n unlitMaterial = babylonMaterial.unlit;\n }\n else if (babylonMaterial instanceof StandardMaterial) {\n unlitMaterial = babylonMaterial.disableLighting;\n }\n if (unlitMaterial) {\n this._wasUsed = true;\n if (node.extensions == null) {\n node.extensions = {};\n }\n node.extensions[NAME$6] = {};\n }\n resolve(node);\n });\n }\n }\n _Exporter.RegisterExtension(NAME$6, () => new KHR_materials_unlit());\n\n const NAME$7 = \"KHR_materials_ior\";\n /**\n * [Specification](https://github.com/KhronosGroup/glTF/blob/main/extensions/2.0/Khronos/KHR_materials_ior/README.md)\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class KHR_materials_ior {\n constructor() {\n /** Name of this extension */\n this.name = NAME$7;\n /** Defines whether this extension is enabled */\n this.enabled = true;\n /** Defines whether this extension is required */\n this.required = false;\n this._wasUsed = false;\n }\n dispose() { }\n /** @internal */\n get wasUsed() {\n return this._wasUsed;\n }\n _isExtensionEnabled(mat) {\n // This extension must not be used on a material that also uses KHR_materials_unlit\n if (mat.unlit) {\n return false;\n }\n return mat.indexOfRefraction != undefined && mat.indexOfRefraction != 1.5; // 1.5 is normative default value.\n }\n postExportMaterialAsync(context, node, babylonMaterial) {\n return new Promise((resolve) => {\n if (babylonMaterial instanceof PBRMaterial && this._isExtensionEnabled(babylonMaterial)) {\n this._wasUsed = true;\n const iorInfo = {\n ior: babylonMaterial.indexOfRefraction,\n };\n node.extensions = node.extensions || {};\n node.extensions[NAME$7] = iorInfo;\n }\n resolve(node);\n });\n }\n }\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _Exporter.RegisterExtension(NAME$7, (exporter) => new KHR_materials_ior());\n\n const NAME$8 = \"KHR_materials_specular\";\n /**\n * [Specification](https://github.com/KhronosGroup/glTF/blob/main/extensions/2.0/Khronos/KHR_materials_specular/README.md)\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class KHR_materials_specular {\n constructor(exporter) {\n /** Name of this extension */\n this.name = NAME$8;\n /** Defines whether this extension is enabled */\n this.enabled = true;\n /** Defines whether this extension is required */\n this.required = false;\n this._wasUsed = false;\n this._exporter = exporter;\n }\n dispose() { }\n /** @internal */\n get wasUsed() {\n return this._wasUsed;\n }\n postExportMaterialAdditionalTextures(context, node, babylonMaterial) {\n const additionalTextures = [];\n if (babylonMaterial instanceof PBRMaterial) {\n if (this._isExtensionEnabled(babylonMaterial)) {\n if (babylonMaterial.metallicReflectanceTexture) {\n additionalTextures.push(babylonMaterial.metallicReflectanceTexture);\n }\n if (babylonMaterial.reflectanceTexture) {\n additionalTextures.push(babylonMaterial.reflectanceTexture);\n }\n return additionalTextures;\n }\n }\n return additionalTextures;\n }\n _isExtensionEnabled(mat) {\n // This extension must not be used on a material that also uses KHR_materials_unlit\n if (mat.unlit) {\n return false;\n }\n return ((mat.metallicF0Factor != undefined && mat.metallicF0Factor != 1.0) ||\n (mat.metallicReflectanceColor != undefined && !mat.metallicReflectanceColor.equalsFloats(1.0, 1.0, 1.0)) ||\n this._hasTexturesExtension(mat));\n }\n _hasTexturesExtension(mat) {\n return mat.metallicReflectanceTexture != null || mat.reflectanceTexture != null;\n }\n postExportMaterialAsync(context, node, babylonMaterial) {\n return new Promise((resolve) => {\n var _a, _b;\n if (babylonMaterial instanceof PBRMaterial && this._isExtensionEnabled(babylonMaterial)) {\n this._wasUsed = true;\n node.extensions = node.extensions || {};\n const metallicReflectanceTexture = (_a = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.metallicReflectanceTexture)) !== null && _a !== void 0 ? _a : undefined;\n const reflectanceTexture = (_b = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.reflectanceTexture)) !== null && _b !== void 0 ? _b : undefined;\n const metallicF0Factor = babylonMaterial.metallicF0Factor == 1.0 ? undefined : babylonMaterial.metallicF0Factor;\n const metallicReflectanceColor = babylonMaterial.metallicReflectanceColor.equalsFloats(1.0, 1.0, 1.0)\n ? undefined\n : babylonMaterial.metallicReflectanceColor.asArray();\n const specularInfo = {\n specularFactor: metallicF0Factor,\n specularTexture: metallicReflectanceTexture,\n specularColorFactor: metallicReflectanceColor,\n specularColorTexture: reflectanceTexture,\n hasTextures: () => {\n return this._hasTexturesExtension(babylonMaterial);\n },\n };\n node.extensions[NAME$8] = specularInfo;\n }\n resolve(node);\n });\n }\n }\n _Exporter.RegisterExtension(NAME$8, (exporter) => new KHR_materials_specular(exporter));\n\n const NAME$9 = \"KHR_materials_volume\";\n /**\n * [Specification](https://github.com/KhronosGroup/glTF/blob/main/extensions/2.0/Khronos/KHR_materials_volume/README.md)\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class KHR_materials_volume {\n constructor(exporter) {\n /** Name of this extension */\n this.name = NAME$9;\n /** Defines whether this extension is enabled */\n this.enabled = true;\n /** Defines whether this extension is required */\n this.required = false;\n this._wasUsed = false;\n this._exporter = exporter;\n }\n dispose() { }\n /** @internal */\n get wasUsed() {\n return this._wasUsed;\n }\n postExportMaterialAdditionalTextures(context, node, babylonMaterial) {\n const additionalTextures = [];\n if (babylonMaterial instanceof PBRMaterial) {\n if (this._isExtensionEnabled(babylonMaterial)) {\n if (babylonMaterial.subSurface.thicknessTexture) {\n additionalTextures.push(babylonMaterial.subSurface.thicknessTexture);\n }\n return additionalTextures;\n }\n }\n return additionalTextures;\n }\n _isExtensionEnabled(mat) {\n // This extension must not be used on a material that also uses KHR_materials_unlit\n if (mat.unlit) {\n return false;\n }\n const subs = mat.subSurface;\n // this extension requires either the KHR_materials_transmission or KHR_materials_translucency extensions.\n if (!subs.isRefractionEnabled && !subs.isTranslucencyEnabled) {\n return false;\n }\n return ((subs.maximumThickness != undefined && subs.maximumThickness != 0) ||\n (subs.tintColorAtDistance != undefined && subs.tintColorAtDistance != Number.POSITIVE_INFINITY) ||\n (subs.tintColor != undefined && subs.tintColor != Color3.White()) ||\n this._hasTexturesExtension(mat));\n }\n _hasTexturesExtension(mat) {\n return mat.subSurface.thicknessTexture != null;\n }\n postExportMaterialAsync(context, node, babylonMaterial) {\n return new Promise((resolve) => {\n var _a;\n if (babylonMaterial instanceof PBRMaterial && this._isExtensionEnabled(babylonMaterial)) {\n this._wasUsed = true;\n const subs = babylonMaterial.subSurface;\n const thicknessFactor = subs.maximumThickness == 0 ? undefined : subs.maximumThickness;\n const thicknessTexture = (_a = this._exporter._glTFMaterialExporter._getTextureInfo(subs.thicknessTexture)) !== null && _a !== void 0 ? _a : undefined;\n const attenuationDistance = subs.tintColorAtDistance == Number.POSITIVE_INFINITY ? undefined : subs.tintColorAtDistance;\n const attenuationColor = subs.tintColor.equalsFloats(1.0, 1.0, 1.0) ? undefined : subs.tintColor.asArray();\n const volumeInfo = {\n thicknessFactor: thicknessFactor,\n thicknessTexture: thicknessTexture,\n attenuationDistance: attenuationDistance,\n attenuationColor: attenuationColor,\n hasTextures: () => {\n return this._hasTexturesExtension(babylonMaterial);\n },\n };\n node.extensions = node.extensions || {};\n node.extensions[NAME$9] = volumeInfo;\n }\n resolve(node);\n });\n }\n }\n _Exporter.RegisterExtension(NAME$9, (exporter) => new KHR_materials_volume(exporter));\n\n const NAME$a = \"KHR_materials_transmission\";\n /**\n * [Specification](https://github.com/KhronosGroup/glTF/blob/main/extensions/2.0/Khronos/KHR_materials_transmission/README.md)\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class KHR_materials_transmission {\n constructor(exporter) {\n /** Name of this extension */\n this.name = NAME$a;\n /** Defines whether this extension is enabled */\n this.enabled = true;\n /** Defines whether this extension is required */\n this.required = false;\n this._wasUsed = false;\n this._exporter = exporter;\n }\n dispose() { }\n /** @internal */\n get wasUsed() {\n return this._wasUsed;\n }\n postExportMaterialAdditionalTextures(context, node, babylonMaterial) {\n const additionalTextures = [];\n if (babylonMaterial instanceof PBRMaterial) {\n if (this._isExtensionEnabled(babylonMaterial)) {\n if (babylonMaterial.subSurface.thicknessTexture) {\n additionalTextures.push(babylonMaterial.subSurface.thicknessTexture);\n }\n return additionalTextures;\n }\n }\n return additionalTextures;\n }\n _isExtensionEnabled(mat) {\n // This extension must not be used on a material that also uses KHR_materials_unlit\n if (mat.unlit) {\n return false;\n }\n const subs = mat.subSurface;\n return (subs.isRefractionEnabled && subs.refractionIntensity != undefined && subs.refractionIntensity != 0) || this._hasTexturesExtension(mat);\n }\n _hasTexturesExtension(mat) {\n return mat.subSurface.refractionIntensityTexture != null;\n }\n postExportMaterialAsync(context, node, babylonMaterial) {\n return new Promise((resolve) => {\n var _a;\n if (babylonMaterial instanceof PBRMaterial && this._isExtensionEnabled(babylonMaterial)) {\n this._wasUsed = true;\n const subs = babylonMaterial.subSurface;\n const transmissionFactor = subs.refractionIntensity === 0 ? undefined : subs.refractionIntensity;\n const transmissionTexture = (_a = this._exporter._glTFMaterialExporter._getTextureInfo(subs.refractionIntensityTexture)) !== null && _a !== void 0 ? _a : undefined;\n const volumeInfo = {\n transmissionFactor: transmissionFactor,\n transmissionTexture: transmissionTexture,\n hasTextures: () => {\n return this._hasTexturesExtension(babylonMaterial);\n },\n };\n node.extensions = node.extensions || {};\n node.extensions[NAME$a] = volumeInfo;\n }\n resolve(node);\n });\n }\n }\n _Exporter.RegisterExtension(NAME$a, (exporter) => new KHR_materials_transmission(exporter));\n\n Mesh.prototype.thinInstanceAdd = function (matrix, refresh = true) {\n if (!this.getScene().getEngine().getCaps().instancedArrays) {\n Logger.Error(\"Thin Instances are not supported on this device as Instanced Array extension not supported\");\n return -1;\n }\n this._thinInstanceUpdateBufferSize(\"matrix\", Array.isArray(matrix) ? matrix.length : 1);\n const index = this._thinInstanceDataStorage.instancesCount;\n if (Array.isArray(matrix)) {\n for (let i = 0; i < matrix.length; ++i) {\n this.thinInstanceSetMatrixAt(this._thinInstanceDataStorage.instancesCount++, matrix[i], i === matrix.length - 1 && refresh);\n }\n }\n else {\n this.thinInstanceSetMatrixAt(this._thinInstanceDataStorage.instancesCount++, matrix, refresh);\n }\n return index;\n };\n Mesh.prototype.thinInstanceAddSelf = function (refresh = true) {\n return this.thinInstanceAdd(Matrix.IdentityReadOnly, refresh);\n };\n Mesh.prototype.thinInstanceRegisterAttribute = function (kind, stride) {\n // preserve backward compatibility\n if (kind === VertexBuffer.ColorKind) {\n kind = VertexBuffer.ColorInstanceKind;\n }\n this.removeVerticesData(kind);\n this._thinInstanceInitializeUserStorage();\n this._userThinInstanceBuffersStorage.strides[kind] = stride;\n this._userThinInstanceBuffersStorage.sizes[kind] = stride * Math.max(32, this._thinInstanceDataStorage.instancesCount); // Initial size\n this._userThinInstanceBuffersStorage.data[kind] = new Float32Array(this._userThinInstanceBuffersStorage.sizes[kind]);\n this._userThinInstanceBuffersStorage.vertexBuffers[kind] = new VertexBuffer(this.getEngine(), this._userThinInstanceBuffersStorage.data[kind], kind, true, false, stride, true);\n this.setVerticesBuffer(this._userThinInstanceBuffersStorage.vertexBuffers[kind]);\n };\n Mesh.prototype.thinInstanceSetMatrixAt = function (index, matrix, refresh = true) {\n if (!this._thinInstanceDataStorage.matrixData || index >= this._thinInstanceDataStorage.instancesCount) {\n return false;\n }\n const matrixData = this._thinInstanceDataStorage.matrixData;\n matrix.copyToArray(matrixData, index * 16);\n if (this._thinInstanceDataStorage.worldMatrices) {\n this._thinInstanceDataStorage.worldMatrices[index] = matrix;\n }\n if (refresh) {\n this.thinInstanceBufferUpdated(\"matrix\");\n if (!this.doNotSyncBoundingInfo) {\n this.thinInstanceRefreshBoundingInfo(false);\n }\n }\n return true;\n };\n Mesh.prototype.thinInstanceSetAttributeAt = function (kind, index, value, refresh = true) {\n // preserve backward compatibility\n if (kind === VertexBuffer.ColorKind) {\n kind = VertexBuffer.ColorInstanceKind;\n }\n if (!this._userThinInstanceBuffersStorage || !this._userThinInstanceBuffersStorage.data[kind] || index >= this._thinInstanceDataStorage.instancesCount) {\n return false;\n }\n this._thinInstanceUpdateBufferSize(kind, 0); // make sur the buffer for the kind attribute is big enough\n this._userThinInstanceBuffersStorage.data[kind].set(value, index * this._userThinInstanceBuffersStorage.strides[kind]);\n if (refresh) {\n this.thinInstanceBufferUpdated(kind);\n }\n return true;\n };\n Object.defineProperty(Mesh.prototype, \"thinInstanceCount\", {\n get: function () {\n return this._thinInstanceDataStorage.instancesCount;\n },\n set: function (value) {\n var _a, _b;\n const matrixData = (_a = this._thinInstanceDataStorage.matrixData) !== null && _a !== void 0 ? _a : (_b = this.source) === null || _b === void 0 ? void 0 : _b._thinInstanceDataStorage.matrixData;\n const numMaxInstances = matrixData ? matrixData.length / 16 : 0;\n if (value <= numMaxInstances) {\n this._thinInstanceDataStorage.instancesCount = value;\n }\n },\n enumerable: true,\n configurable: true,\n });\n Mesh.prototype._thinInstanceCreateMatrixBuffer = function (kind, buffer, staticBuffer = false) {\n // preserve backward compatibility\n if (kind === VertexBuffer.ColorKind) {\n kind = VertexBuffer.ColorInstanceKind;\n }\n const matrixBuffer = new Buffer$1(this.getEngine(), buffer, !staticBuffer, 16, false, true);\n for (let i = 0; i < 4; i++) {\n this.setVerticesBuffer(matrixBuffer.createVertexBuffer(kind + i, i * 4, 4));\n }\n return matrixBuffer;\n };\n Mesh.prototype.thinInstanceSetBuffer = function (kind, buffer, stride = 0, staticBuffer = false) {\n var _a, _b, _c;\n stride = stride || 16;\n if (kind === \"matrix\") {\n (_a = this._thinInstanceDataStorage.matrixBuffer) === null || _a === void 0 ? void 0 : _a.dispose();\n this._thinInstanceDataStorage.matrixBuffer = null;\n this._thinInstanceDataStorage.matrixBufferSize = buffer ? buffer.length : 32 * stride;\n this._thinInstanceDataStorage.matrixData = buffer;\n this._thinInstanceDataStorage.worldMatrices = null;\n if (buffer !== null) {\n this._thinInstanceDataStorage.instancesCount = buffer.length / stride;\n this._thinInstanceDataStorage.matrixBuffer = this._thinInstanceCreateMatrixBuffer(\"world\", buffer, staticBuffer);\n if (!this.doNotSyncBoundingInfo) {\n this.thinInstanceRefreshBoundingInfo(false);\n }\n }\n else {\n this._thinInstanceDataStorage.instancesCount = 0;\n if (!this.doNotSyncBoundingInfo) {\n // mesh has no more thin instances, so need to recompute the bounding box because it's the regular mesh that will now be displayed\n this.refreshBoundingInfo();\n }\n }\n }\n else if (kind === \"previousMatrix\") {\n (_b = this._thinInstanceDataStorage.previousMatrixBuffer) === null || _b === void 0 ? void 0 : _b.dispose();\n this._thinInstanceDataStorage.previousMatrixBuffer = null;\n this._thinInstanceDataStorage.previousMatrixData = buffer;\n if (buffer !== null) {\n this._thinInstanceDataStorage.previousMatrixBuffer = this._thinInstanceCreateMatrixBuffer(\"previousWorld\", buffer, staticBuffer);\n }\n }\n else {\n // color for instanced mesh is ColorInstanceKind and not ColorKind because of native that needs to do the differenciation\n // hot switching kind here to preserve backward compatibility\n if (kind === VertexBuffer.ColorKind) {\n kind = VertexBuffer.ColorInstanceKind;\n }\n if (buffer === null) {\n if ((_c = this._userThinInstanceBuffersStorage) === null || _c === void 0 ? void 0 : _c.data[kind]) {\n this.removeVerticesData(kind);\n delete this._userThinInstanceBuffersStorage.data[kind];\n delete this._userThinInstanceBuffersStorage.strides[kind];\n delete this._userThinInstanceBuffersStorage.sizes[kind];\n delete this._userThinInstanceBuffersStorage.vertexBuffers[kind];\n }\n }\n else {\n this._thinInstanceInitializeUserStorage();\n this._userThinInstanceBuffersStorage.data[kind] = buffer;\n this._userThinInstanceBuffersStorage.strides[kind] = stride;\n this._userThinInstanceBuffersStorage.sizes[kind] = buffer.length;\n this._userThinInstanceBuffersStorage.vertexBuffers[kind] = new VertexBuffer(this.getEngine(), buffer, kind, !staticBuffer, false, stride, true);\n this.setVerticesBuffer(this._userThinInstanceBuffersStorage.vertexBuffers[kind]);\n }\n }\n };\n Mesh.prototype.thinInstanceBufferUpdated = function (kind) {\n var _a, _b, _c;\n if (kind === \"matrix\") {\n (_a = this._thinInstanceDataStorage.matrixBuffer) === null || _a === void 0 ? void 0 : _a.updateDirectly(this._thinInstanceDataStorage.matrixData, 0, this._thinInstanceDataStorage.instancesCount);\n }\n else if (kind === \"previousMatrix\") {\n (_b = this._thinInstanceDataStorage.previousMatrixBuffer) === null || _b === void 0 ? void 0 : _b.updateDirectly(this._thinInstanceDataStorage.previousMatrixData, 0, this._thinInstanceDataStorage.instancesCount);\n }\n else {\n // preserve backward compatibility\n if (kind === VertexBuffer.ColorKind) {\n kind = VertexBuffer.ColorInstanceKind;\n }\n if ((_c = this._userThinInstanceBuffersStorage) === null || _c === void 0 ? void 0 : _c.vertexBuffers[kind]) {\n this._userThinInstanceBuffersStorage.vertexBuffers[kind].updateDirectly(this._userThinInstanceBuffersStorage.data[kind], 0);\n }\n }\n };\n Mesh.prototype.thinInstancePartialBufferUpdate = function (kind, data, offset) {\n var _a;\n if (kind === \"matrix\") {\n if (this._thinInstanceDataStorage.matrixBuffer) {\n this._thinInstanceDataStorage.matrixBuffer.updateDirectly(data, offset);\n }\n }\n else {\n // preserve backward compatibility\n if (kind === VertexBuffer.ColorKind) {\n kind = VertexBuffer.ColorInstanceKind;\n }\n if ((_a = this._userThinInstanceBuffersStorage) === null || _a === void 0 ? void 0 : _a.vertexBuffers[kind]) {\n this._userThinInstanceBuffersStorage.vertexBuffers[kind].updateDirectly(data, offset);\n }\n }\n };\n Mesh.prototype.thinInstanceGetWorldMatrices = function () {\n if (!this._thinInstanceDataStorage.matrixData || !this._thinInstanceDataStorage.matrixBuffer) {\n return [];\n }\n const matrixData = this._thinInstanceDataStorage.matrixData;\n if (!this._thinInstanceDataStorage.worldMatrices) {\n this._thinInstanceDataStorage.worldMatrices = new Array();\n for (let i = 0; i < this._thinInstanceDataStorage.instancesCount; ++i) {\n this._thinInstanceDataStorage.worldMatrices[i] = Matrix.FromArray(matrixData, i * 16);\n }\n }\n return this._thinInstanceDataStorage.worldMatrices;\n };\n Mesh.prototype.thinInstanceRefreshBoundingInfo = function (forceRefreshParentInfo = false, applySkeleton = false, applyMorph = false) {\n if (!this._thinInstanceDataStorage.matrixData || !this._thinInstanceDataStorage.matrixBuffer) {\n return;\n }\n const vectors = this._thinInstanceDataStorage.boundingVectors;\n if (forceRefreshParentInfo || !this.rawBoundingInfo) {\n vectors.length = 0;\n this.refreshBoundingInfo(applySkeleton, applyMorph);\n const boundingInfo = this.getBoundingInfo();\n this.rawBoundingInfo = new BoundingInfo(boundingInfo.minimum, boundingInfo.maximum);\n }\n const boundingInfo = this.getBoundingInfo();\n const matrixData = this._thinInstanceDataStorage.matrixData;\n if (vectors.length === 0) {\n for (let v = 0; v < boundingInfo.boundingBox.vectors.length; ++v) {\n vectors.push(boundingInfo.boundingBox.vectors[v].clone());\n }\n }\n TmpVectors.Vector3[0].setAll(Number.POSITIVE_INFINITY); // min\n TmpVectors.Vector3[1].setAll(Number.NEGATIVE_INFINITY); // max\n for (let i = 0; i < this._thinInstanceDataStorage.instancesCount; ++i) {\n Matrix.FromArrayToRef(matrixData, i * 16, TmpVectors.Matrix[0]);\n for (let v = 0; v < vectors.length; ++v) {\n Vector3.TransformCoordinatesToRef(vectors[v], TmpVectors.Matrix[0], TmpVectors.Vector3[2]);\n TmpVectors.Vector3[0].minimizeInPlace(TmpVectors.Vector3[2]);\n TmpVectors.Vector3[1].maximizeInPlace(TmpVectors.Vector3[2]);\n }\n }\n boundingInfo.reConstruct(TmpVectors.Vector3[0], TmpVectors.Vector3[1]);\n this._updateBoundingInfo();\n };\n Mesh.prototype._thinInstanceUpdateBufferSize = function (kind, numInstances = 1) {\n var _a, _b, _c;\n // preserve backward compatibility\n if (kind === VertexBuffer.ColorKind) {\n kind = VertexBuffer.ColorInstanceKind;\n }\n const kindIsMatrix = kind === \"matrix\";\n if (!kindIsMatrix && (!this._userThinInstanceBuffersStorage || !this._userThinInstanceBuffersStorage.strides[kind])) {\n return;\n }\n const stride = kindIsMatrix ? 16 : this._userThinInstanceBuffersStorage.strides[kind];\n const currentSize = kindIsMatrix ? this._thinInstanceDataStorage.matrixBufferSize : this._userThinInstanceBuffersStorage.sizes[kind];\n let data = kindIsMatrix ? this._thinInstanceDataStorage.matrixData : this._userThinInstanceBuffersStorage.data[kind];\n const bufferSize = (this._thinInstanceDataStorage.instancesCount + numInstances) * stride;\n let newSize = currentSize;\n while (newSize < bufferSize) {\n newSize *= 2;\n }\n if (!data || currentSize != newSize) {\n if (!data) {\n data = new Float32Array(newSize);\n }\n else {\n const newData = new Float32Array(newSize);\n newData.set(data, 0);\n data = newData;\n }\n if (kindIsMatrix) {\n (_a = this._thinInstanceDataStorage.matrixBuffer) === null || _a === void 0 ? void 0 : _a.dispose();\n this._thinInstanceDataStorage.matrixBuffer = this._thinInstanceCreateMatrixBuffer(\"world\", data, false);\n this._thinInstanceDataStorage.matrixData = data;\n this._thinInstanceDataStorage.matrixBufferSize = newSize;\n if (this._scene.needsPreviousWorldMatrices && !this._thinInstanceDataStorage.previousMatrixData) {\n (_b = this._thinInstanceDataStorage.previousMatrixBuffer) === null || _b === void 0 ? void 0 : _b.dispose();\n this._thinInstanceDataStorage.previousMatrixBuffer = this._thinInstanceCreateMatrixBuffer(\"previousWorld\", data, false);\n }\n }\n else {\n (_c = this._userThinInstanceBuffersStorage.vertexBuffers[kind]) === null || _c === void 0 ? void 0 : _c.dispose();\n this._userThinInstanceBuffersStorage.data[kind] = data;\n this._userThinInstanceBuffersStorage.sizes[kind] = newSize;\n this._userThinInstanceBuffersStorage.vertexBuffers[kind] = new VertexBuffer(this.getEngine(), data, kind, true, false, stride, true);\n this.setVerticesBuffer(this._userThinInstanceBuffersStorage.vertexBuffers[kind]);\n }\n }\n };\n Mesh.prototype._thinInstanceInitializeUserStorage = function () {\n if (!this._userThinInstanceBuffersStorage) {\n this._userThinInstanceBuffersStorage = {\n data: {},\n sizes: {},\n vertexBuffers: {},\n strides: {},\n };\n }\n };\n Mesh.prototype._disposeThinInstanceSpecificData = function () {\n var _a;\n if ((_a = this._thinInstanceDataStorage) === null || _a === void 0 ? void 0 : _a.matrixBuffer) {\n this._thinInstanceDataStorage.matrixBuffer.dispose();\n this._thinInstanceDataStorage.matrixBuffer = null;\n }\n };\n\n const NAME$b = \"EXT_mesh_gpu_instancing\";\n /**\n * [Specification](https://github.com/KhronosGroup/glTF/blob/main/extensions/2.0/Vendor/EXT_mesh_gpu_instancing/README.md)\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class EXT_mesh_gpu_instancing {\n constructor(exporter) {\n /** Name of this extension */\n this.name = NAME$b;\n /** Defines whether this extension is enabled */\n this.enabled = true;\n /** Defines whether this extension is required */\n this.required = false;\n this._wasUsed = false;\n this._exporter = exporter;\n }\n dispose() { }\n /** @internal */\n get wasUsed() {\n return this._wasUsed;\n }\n postExportNodeAsync(context, node, babylonNode, nodeMap, binaryWriter) {\n return new Promise((resolve) => {\n if (node && babylonNode instanceof Mesh) {\n if (babylonNode.hasThinInstances && binaryWriter) {\n this._wasUsed = true;\n const noTranslation = Vector3.Zero();\n const noRotation = Quaternion.Identity();\n const noScale = Vector3.One();\n // retreive all the instance world matrix\n const matrix = babylonNode.thinInstanceGetWorldMatrices();\n const iwt = TmpVectors.Vector3[2];\n const iwr = TmpVectors.Quaternion[1];\n const iws = TmpVectors.Vector3[3];\n let hasAnyInstanceWorldTranslation = false;\n let hasAnyInstanceWorldRotation = false;\n let hasAnyInstanceWorldScale = false;\n // prepare temp buffers\n const translationBuffer = new Float32Array(babylonNode.thinInstanceCount * 3);\n const rotationBuffer = new Float32Array(babylonNode.thinInstanceCount * 4);\n const scaleBuffer = new Float32Array(babylonNode.thinInstanceCount * 3);\n let i = 0;\n for (const m of matrix) {\n m.decompose(iws, iwr, iwt);\n // fill the temp buffer\n translationBuffer.set(iwt.asArray(), i * 3);\n rotationBuffer.set(iwr.normalize().asArray(), i * 4); // ensure the quaternion is normalized\n scaleBuffer.set(iws.asArray(), i * 3);\n // this is where we decide if there is any transformation\n hasAnyInstanceWorldTranslation = hasAnyInstanceWorldTranslation || !iwt.equalsWithEpsilon(noTranslation);\n hasAnyInstanceWorldRotation = hasAnyInstanceWorldRotation || !iwr.equalsWithEpsilon(noRotation);\n hasAnyInstanceWorldScale = hasAnyInstanceWorldScale || !iws.equalsWithEpsilon(noScale);\n i++;\n }\n const extension = {\n attributes: {},\n };\n // do we need to write TRANSLATION ?\n if (hasAnyInstanceWorldTranslation) {\n extension.attributes[\"TRANSLATION\"] = this._buildAccessor(translationBuffer, \"VEC3\" /* AccessorType.VEC3 */, babylonNode.thinInstanceCount, binaryWriter, 5126 /* AccessorComponentType.FLOAT */);\n }\n // do we need to write ROTATION ?\n if (hasAnyInstanceWorldRotation) {\n const componentType = 5126 /* AccessorComponentType.FLOAT */; // we decided to stay on FLOAT for now see https://github.com/BabylonJS/Babylon.js/pull/12495\n extension.attributes[\"ROTATION\"] = this._buildAccessor(rotationBuffer, \"VEC4\" /* AccessorType.VEC4 */, babylonNode.thinInstanceCount, binaryWriter, componentType);\n }\n // do we need to write SCALE ?\n if (hasAnyInstanceWorldScale) {\n extension.attributes[\"SCALE\"] = this._buildAccessor(scaleBuffer, \"VEC3\" /* AccessorType.VEC3 */, babylonNode.thinInstanceCount, binaryWriter, 5126 /* AccessorComponentType.FLOAT */);\n }\n /* eslint-enable @typescript-eslint/naming-convention*/\n node.extensions = node.extensions || {};\n node.extensions[NAME$b] = extension;\n }\n }\n resolve(node);\n });\n }\n _buildAccessor(buffer, type, count, binaryWriter, componentType) {\n // write the buffer\n const bufferOffset = binaryWriter.getByteOffset();\n switch (componentType) {\n case 5126 /* AccessorComponentType.FLOAT */: {\n for (let i = 0; i != buffer.length; i++) {\n binaryWriter.setFloat32(buffer[i]);\n }\n break;\n }\n case 5120 /* AccessorComponentType.BYTE */: {\n for (let i = 0; i != buffer.length; i++) {\n binaryWriter.setByte(buffer[i] * 127);\n }\n break;\n }\n case 5122 /* AccessorComponentType.SHORT */: {\n for (let i = 0; i != buffer.length; i++) {\n binaryWriter.setInt16(buffer[i] * 32767);\n }\n break;\n }\n }\n // build the buffer view\n const bv = { buffer: 0, byteOffset: bufferOffset, byteLength: buffer.length * VertexBuffer.GetTypeByteLength(componentType) };\n const bufferViewIndex = this._exporter._bufferViews.length;\n this._exporter._bufferViews.push(bv);\n // finally build the accessor\n const accessorIndex = this._exporter._accessors.length;\n const accessor = {\n bufferView: bufferViewIndex,\n componentType: componentType,\n count: count,\n type: type,\n normalized: componentType == 5120 /* AccessorComponentType.BYTE */ || componentType == 5122 /* AccessorComponentType.SHORT */,\n };\n this._exporter._accessors.push(accessor);\n return accessorIndex;\n }\n }\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _Exporter.RegisterExtension(NAME$b, (exporter) => new EXT_mesh_gpu_instancing(exporter));\n\n const NAME$c = \"KHR_materials_emissive_strength\";\n /**\n * [Specification](https://github.com/KhronosGroup/glTF/blob/main/extensions/2.0/Khronos/KHR_materials_emissive_strength/README.md)\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class KHR_materials_emissive_strength {\n constructor() {\n /** Name of this extension */\n this.name = NAME$c;\n /** Defines whether this extension is enabled */\n this.enabled = true;\n /** Defines whether this extension is required */\n this.required = false;\n this._wasUsed = false;\n }\n dispose() { }\n /** @interal */\n get wasUsed() {\n return this._wasUsed;\n }\n postExportMaterialAsync(context, node, babylonMaterial) {\n return new Promise((resolve) => {\n if (!(babylonMaterial instanceof PBRMaterial)) {\n return resolve(node);\n }\n const emissiveColor = babylonMaterial.emissiveColor.asArray();\n const tempEmissiveStrength = Math.max(...emissiveColor);\n if (tempEmissiveStrength > 1) {\n this._wasUsed = true;\n node.extensions || (node.extensions = {});\n const emissiveStrengthInfo = {\n emissiveStrength: tempEmissiveStrength,\n };\n // Normalize each value of the emissive factor to have a max value of 1\n const newEmissiveFactor = babylonMaterial.emissiveColor.scale(1 / emissiveStrengthInfo.emissiveStrength);\n node.emissiveFactor = newEmissiveFactor.asArray();\n node.extensions[NAME$c] = emissiveStrengthInfo;\n }\n return resolve(node);\n });\n }\n }\n _Exporter.RegisterExtension(NAME$c, (exporter) => new KHR_materials_emissive_strength());\n\n class GLTFKb3dExporter {\n constructor(scene) {\n this.scene = scene;\n this._glbExportOptions = {\n shouldExportNode: n => this.shouldExportNode(n),\n };\n }\n /** generates the current scene into a glb file and returns an object url for it */\n generateGlb(filename = 'scene') {\n return GLTF2Export.GLBAsync(this.scene, filename, this._glbExportOptions).then(glbData => {\n return this.createGlbBlob(glbData);\n });\n }\n exportGlb(filename = 'scene') {\n return GLTF2Export.GLBAsync(this.scene, filename, this._glbExportOptions).then(glbData => {\n const dispose = this.downloadBlob(`${filename}.glb`, this.createGlbBlob(glbData));\n setTimeout(() => dispose());\n });\n }\n exportGltf(filename = 'scene') {\n return GLTF2Export.GLTFAsync(this.scene, filename, this._glbExportOptions).then(glbData => {\n const disposeFuncs = [];\n for (const key in glbData.glTFFiles) {\n const blob = glbData.glTFFiles[key];\n let mimeType;\n if (endsWith(key, '.glb')) {\n mimeType = { type: 'model/gltf-binary' };\n }\n else if (endsWith(key, '.bin')) {\n mimeType = { type: 'application/octet-stream' };\n }\n else if (endsWith(key, '.gltf')) {\n mimeType = { type: 'model/gltf+json' };\n }\n else if (endsWith(key, '.jpeg') || endsWith(key, '.jpg')) {\n mimeType = { type: 'image/jpeg' /* ImageMimeType.JPEG */ };\n }\n else if (endsWith(key, '.png')) {\n mimeType = { type: 'image/png' /* ImageMimeType.PNG */ };\n }\n disposeFuncs.push(this.downloadBlob(key, new Blob([blob], mimeType)));\n }\n setTimeout(() => disposeFuncs.forEach(dispose => dispose()));\n });\n function endsWith(str, suffix) {\n return str.indexOf(suffix, str.length - suffix.length) !== -1;\n }\n }\n shouldExportNode(node) {\n const kb3dNode = getByDynamicId(node.id);\n if (kb3dNode instanceof exports.Connector) {\n return false;\n }\n else if (kb3dNode instanceof exports.ViewpointNode) {\n return false;\n }\n return kb3dNode != null;\n }\n createGlbBlob(glbData) {\n //we're expecting one file since it's a GLB\n const key = Object.keys(glbData.glTFFiles)[0];\n const blob = glbData.glTFFiles[key];\n const mimeType = { type: 'model/gltf-binary' };\n return new Blob([blob], mimeType);\n }\n downloadBlob(fileName, blob) {\n const link = document.createElement('a');\n document.body.appendChild(link);\n link.setAttribute('type', 'hidden');\n link.download = fileName;\n link.href = window.URL.createObjectURL(blob);\n link.click();\n return dispose;\n function dispose() {\n window.URL.revokeObjectURL(link.href);\n link.remove();\n }\n }\n }\n\n /**\n * This represents a color grading texture. This acts as a lookup table LUT, useful during post process\n * It can help converting any input color in a desired output one. This can then be used to create effects\n * from sepia, black and white to sixties or futuristic rendering...\n *\n * The only supported format is currently 3dl.\n * More information on LUT: https://en.wikipedia.org/wiki/3D_lookup_table\n */\n class ColorGradingTexture extends BaseTexture {\n /**\n * Instantiates a ColorGradingTexture from the following parameters.\n *\n * @param url The location of the color grading data (currently only supporting 3dl)\n * @param sceneOrEngine The scene or engine the texture will be used in\n * @param onLoad defines a callback triggered when the texture has been loaded\n */\n constructor(url, sceneOrEngine, onLoad = null) {\n super(sceneOrEngine);\n if (!url) {\n return;\n }\n this._textureMatrix = Matrix.Identity();\n this.name = url;\n this.url = url;\n this._onLoad = onLoad;\n this._texture = this._getFromCache(url, true);\n if (!this._texture) {\n const scene = this.getScene();\n if (scene) {\n if (!scene.useDelayedTextureLoading) {\n this._loadTexture();\n }\n else {\n this.delayLoadState = 4;\n }\n }\n else {\n this._loadTexture();\n }\n }\n else {\n this._triggerOnLoad();\n }\n }\n /**\n * Fires the onload event from the constructor if requested.\n */\n _triggerOnLoad() {\n if (this._onLoad) {\n this._onLoad();\n }\n }\n /**\n * Returns the texture matrix used in most of the material.\n * This is not used in color grading but keep for troubleshooting purpose (easily swap diffuse by colorgrading to look in).\n */\n getTextureMatrix() {\n return this._textureMatrix;\n }\n /**\n * Occurs when the file being loaded is a .3dl LUT file.\n */\n _load3dlTexture() {\n const engine = this._getEngine();\n let texture;\n if (!engine._features.support3DTextures) {\n texture = engine.createRawTexture(null, 1, 1, 5, false, false, 2, null, 0);\n }\n else {\n texture = engine.createRawTexture3D(null, 1, 1, 1, 5, false, false, 2, null, 0);\n }\n this._texture = texture;\n this._texture.isReady = false;\n this.isCube = false;\n this.is3D = engine._features.support3DTextures;\n this.wrapU = 0;\n this.wrapV = 0;\n this.wrapR = 0;\n this.anisotropicFilteringLevel = 1;\n const callback = (text) => {\n if (typeof text !== \"string\") {\n return;\n }\n let data = null;\n let tempData = null;\n let line;\n const lines = text.split(\"\\n\");\n let size = 0, pixelIndexW = 0, pixelIndexH = 0, pixelIndexSlice = 0;\n let maxColor = 0;\n for (let i = 0; i < lines.length; i++) {\n line = lines[i];\n if (!ColorGradingTexture._NoneEmptyLineRegex.test(line)) {\n continue;\n }\n if (line.indexOf(\"#\") === 0) {\n continue;\n }\n const words = line.split(\" \");\n if (size === 0) {\n // Number of space + one\n size = words.length;\n data = new Uint8Array(size * size * size * 4); // volume texture of side size and rgb 8\n tempData = new Float32Array(size * size * size * 4);\n continue;\n }\n if (size != 0) {\n const r = Math.max(parseInt(words[0]), 0);\n const g = Math.max(parseInt(words[1]), 0);\n const b = Math.max(parseInt(words[2]), 0);\n maxColor = Math.max(r, maxColor);\n maxColor = Math.max(g, maxColor);\n maxColor = Math.max(b, maxColor);\n const pixelStorageIndex = (pixelIndexW + pixelIndexSlice * size + pixelIndexH * size * size) * 4;\n if (tempData) {\n tempData[pixelStorageIndex + 0] = r;\n tempData[pixelStorageIndex + 1] = g;\n tempData[pixelStorageIndex + 2] = b;\n }\n // Keep for reference in case of back compat problems.\n // pixelIndexSlice++;\n // if (pixelIndexSlice % size == 0) {\n // pixelIndexH++;\n // pixelIndexSlice = 0;\n // if (pixelIndexH % size == 0) {\n // pixelIndexW++;\n // pixelIndexH = 0;\n // }\n // }\n pixelIndexH++;\n if (pixelIndexH % size == 0) {\n pixelIndexSlice++;\n pixelIndexH = 0;\n if (pixelIndexSlice % size == 0) {\n pixelIndexW++;\n pixelIndexSlice = 0;\n }\n }\n }\n }\n if (tempData && data) {\n for (let i = 0; i < tempData.length; i++) {\n if (i > 0 && (i + 1) % 4 === 0) {\n data[i] = 255;\n }\n else {\n const value = tempData[i];\n data[i] = (value / maxColor) * 255;\n }\n }\n }\n if (texture.is3D) {\n texture.updateSize(size, size, size);\n engine.updateRawTexture3D(texture, data, 5, false);\n }\n else {\n texture.updateSize(size * size, size);\n engine.updateRawTexture(texture, data, 5, false);\n }\n texture.isReady = true;\n this._triggerOnLoad();\n };\n const scene = this.getScene();\n if (scene) {\n scene._loadFile(this.url, callback);\n }\n else {\n engine._loadFile(this.url, callback);\n }\n return this._texture;\n }\n /**\n * Starts the loading process of the texture.\n */\n _loadTexture() {\n if (this.url && this.url.toLocaleLowerCase().indexOf(\".3dl\") == this.url.length - 4) {\n this._load3dlTexture();\n }\n }\n /**\n * Clones the color grading texture.\n */\n clone() {\n const newTexture = new ColorGradingTexture(this.url, this.getScene() || this._getEngine());\n // Base texture\n newTexture.level = this.level;\n return newTexture;\n }\n /**\n * Called during delayed load for textures.\n */\n delayLoad() {\n if (this.delayLoadState !== 4) {\n return;\n }\n this.delayLoadState = 1;\n this._texture = this._getFromCache(this.url, true);\n if (!this._texture) {\n this._loadTexture();\n }\n }\n /**\n * Parses a color grading texture serialized by Babylon.\n * @param parsedTexture The texture information being parsedTexture\n * @param scene The scene to load the texture in\n * @returns A color grading texture\n */\n static Parse(parsedTexture, scene) {\n let texture = null;\n if (parsedTexture.name && !parsedTexture.isRenderTarget) {\n texture = new ColorGradingTexture(parsedTexture.name, scene);\n texture.name = parsedTexture.name;\n texture.level = parsedTexture.level;\n }\n return texture;\n }\n /**\n * Serializes the LUT texture to json format.\n */\n serialize() {\n if (!this.name) {\n return null;\n }\n const serializationObject = {};\n serializationObject.name = this.name;\n serializationObject.level = this.level;\n serializationObject.customType = \"BABYLON.ColorGradingTexture\";\n return serializationObject;\n }\n }\n /**\n * Empty line regex stored for GC.\n */\n ColorGradingTexture._NoneEmptyLineRegex = /\\S+/;\n RegisterClass(\"BABYLON.ColorGradingTexture\", ColorGradingTexture);\n\n /**\n * Helper class useful to convert panorama picture to their cubemap representation in 6 faces.\n */\n class PanoramaToCubeMapTools {\n /**\n * Converts a panorama stored in RGB right to left up to down format into a cubemap (6 faces).\n *\n * @param float32Array The source data.\n * @param inputWidth The width of the input panorama.\n * @param inputHeight The height of the input panorama.\n * @param size The willing size of the generated cubemap (each faces will be size * size pixels)\n * @returns The cubemap data\n */\n static ConvertPanoramaToCubemap(float32Array, inputWidth, inputHeight, size, supersample = false) {\n if (!float32Array) {\n throw \"ConvertPanoramaToCubemap: input cannot be null\";\n }\n if (float32Array.length != inputWidth * inputHeight * 3) {\n throw \"ConvertPanoramaToCubemap: input size is wrong\";\n }\n const textureFront = this.CreateCubemapTexture(size, this.FACE_FRONT, float32Array, inputWidth, inputHeight, supersample);\n const textureBack = this.CreateCubemapTexture(size, this.FACE_BACK, float32Array, inputWidth, inputHeight, supersample);\n const textureLeft = this.CreateCubemapTexture(size, this.FACE_LEFT, float32Array, inputWidth, inputHeight, supersample);\n const textureRight = this.CreateCubemapTexture(size, this.FACE_RIGHT, float32Array, inputWidth, inputHeight, supersample);\n const textureUp = this.CreateCubemapTexture(size, this.FACE_UP, float32Array, inputWidth, inputHeight, supersample);\n const textureDown = this.CreateCubemapTexture(size, this.FACE_DOWN, float32Array, inputWidth, inputHeight, supersample);\n return {\n front: textureFront,\n back: textureBack,\n left: textureLeft,\n right: textureRight,\n up: textureUp,\n down: textureDown,\n size: size,\n type: 1,\n format: 4,\n gammaSpace: false,\n };\n }\n static CreateCubemapTexture(texSize, faceData, float32Array, inputWidth, inputHeight, supersample = false) {\n const buffer = new ArrayBuffer(texSize * texSize * 4 * 3);\n const textureArray = new Float32Array(buffer);\n // If supersampling, determine number of samples needed when source texture width is divided for 4 cube faces\n const samples = supersample ? Math.max(1, Math.round(inputWidth / 4 / texSize)) : 1;\n const sampleFactor = 1 / samples;\n const sampleFactorSqr = sampleFactor * sampleFactor;\n const rotDX1 = faceData[1].subtract(faceData[0]).scale(sampleFactor / texSize);\n const rotDX2 = faceData[3].subtract(faceData[2]).scale(sampleFactor / texSize);\n const dy = 1 / texSize;\n let fy = 0;\n for (let y = 0; y < texSize; y++) {\n for (let sy = 0; sy < samples; sy++) {\n let xv1 = faceData[0];\n let xv2 = faceData[2];\n for (let x = 0; x < texSize; x++) {\n for (let sx = 0; sx < samples; sx++) {\n const v = xv2.subtract(xv1).scale(fy).add(xv1);\n v.normalize();\n const color = this.CalcProjectionSpherical(v, float32Array, inputWidth, inputHeight);\n // 3 channels per pixels\n textureArray[y * texSize * 3 + x * 3 + 0] += color.r * sampleFactorSqr;\n textureArray[y * texSize * 3 + x * 3 + 1] += color.g * sampleFactorSqr;\n textureArray[y * texSize * 3 + x * 3 + 2] += color.b * sampleFactorSqr;\n xv1 = xv1.add(rotDX1);\n xv2 = xv2.add(rotDX2);\n }\n }\n fy += dy * sampleFactor;\n }\n }\n return textureArray;\n }\n static CalcProjectionSpherical(vDir, float32Array, inputWidth, inputHeight) {\n let theta = Math.atan2(vDir.z, vDir.x);\n const phi = Math.acos(vDir.y);\n while (theta < -Math.PI) {\n theta += 2 * Math.PI;\n }\n while (theta > Math.PI) {\n theta -= 2 * Math.PI;\n }\n let dx = theta / Math.PI;\n const dy = phi / Math.PI;\n // recenter.\n dx = dx * 0.5 + 0.5;\n let px = Math.round(dx * inputWidth);\n if (px < 0) {\n px = 0;\n }\n else if (px >= inputWidth) {\n px = inputWidth - 1;\n }\n let py = Math.round(dy * inputHeight);\n if (py < 0) {\n py = 0;\n }\n else if (py >= inputHeight) {\n py = inputHeight - 1;\n }\n const inputY = inputHeight - py - 1;\n const r = float32Array[inputY * inputWidth * 3 + px * 3 + 0];\n const g = float32Array[inputY * inputWidth * 3 + px * 3 + 1];\n const b = float32Array[inputY * inputWidth * 3 + px * 3 + 2];\n return {\n r: r,\n g: g,\n b: b,\n };\n }\n }\n PanoramaToCubeMapTools.FACE_LEFT = [new Vector3(-1.0, -1.0, -1.0), new Vector3(1.0, -1.0, -1.0), new Vector3(-1.0, 1.0, -1.0), new Vector3(1.0, 1.0, -1.0)];\n PanoramaToCubeMapTools.FACE_RIGHT = [new Vector3(1.0, -1.0, 1.0), new Vector3(-1.0, -1.0, 1.0), new Vector3(1.0, 1.0, 1.0), new Vector3(-1.0, 1.0, 1.0)];\n PanoramaToCubeMapTools.FACE_FRONT = [new Vector3(1.0, -1.0, -1.0), new Vector3(1.0, -1.0, 1.0), new Vector3(1.0, 1.0, -1.0), new Vector3(1.0, 1.0, 1.0)];\n PanoramaToCubeMapTools.FACE_BACK = [new Vector3(-1.0, -1.0, 1.0), new Vector3(-1.0, -1.0, -1.0), new Vector3(-1.0, 1.0, 1.0), new Vector3(-1.0, 1.0, -1.0)];\n PanoramaToCubeMapTools.FACE_DOWN = [new Vector3(1.0, 1.0, -1.0), new Vector3(1.0, 1.0, 1.0), new Vector3(-1.0, 1.0, -1.0), new Vector3(-1.0, 1.0, 1.0)];\n PanoramaToCubeMapTools.FACE_UP = [new Vector3(-1.0, -1.0, -1.0), new Vector3(-1.0, -1.0, 1.0), new Vector3(1.0, -1.0, -1.0), new Vector3(1.0, -1.0, 1.0)];\n\n // Do not edit.\n const name$2q = \"hdrFilteringVertexShader\";\n const shader$2q = `attribute vec2 position;varying vec3 direction;uniform vec3 up;uniform vec3 right;uniform vec3 front;\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\nmat3 view=mat3(up,right,front);direction=view*vec3(position,1.0);gl_Position=vec4(position,0.0,1.0);\n#define CUSTOM_VERTEX_MAIN_END\n}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2q] = shader$2q;\n\n // Do not edit.\n const name$2r = \"hdrFilteringPixelShader\";\n const shader$2r = `#include\n#include\n#include\n#include\nuniform float alphaG;uniform samplerCube inputTexture;uniform vec2 vFilteringInfo;uniform float hdrScale;varying vec3 direction;void main() {vec3 color=radiance(alphaG,inputTexture,direction,vFilteringInfo);gl_FragColor=vec4(color*hdrScale,1.0);}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2r] = shader$2r;\n\n /**\n * Filters HDR maps to get correct renderings of PBR reflections\n */\n class HDRFiltering {\n /**\n * Instantiates HDR filter for reflection maps\n *\n * @param engine Thin engine\n * @param options Options\n */\n constructor(engine, options = {}) {\n this._lodGenerationOffset = 0;\n this._lodGenerationScale = 0.8;\n /**\n * Quality switch for prefiltering. Should be set to `4096` unless\n * you care about baking speed.\n */\n this.quality = 4096;\n /**\n * Scales pixel intensity for the input HDR map.\n */\n this.hdrScale = 1;\n // pass\n this._engine = engine;\n this.hdrScale = options.hdrScale || this.hdrScale;\n this.quality = options.quality || this.quality;\n }\n _createRenderTarget(size) {\n let textureType = 0;\n if (this._engine.getCaps().textureHalfFloatRender) {\n textureType = 2;\n }\n else if (this._engine.getCaps().textureFloatRender) {\n textureType = 1;\n }\n const rtWrapper = this._engine.createRenderTargetCubeTexture(size, {\n format: 5,\n type: textureType,\n createMipMaps: true,\n generateMipMaps: false,\n generateDepthBuffer: false,\n generateStencilBuffer: false,\n samplingMode: 1,\n });\n this._engine.updateTextureWrappingMode(rtWrapper.texture, 0, 0, 0);\n this._engine.updateTextureSamplingMode(3, rtWrapper.texture, true);\n return rtWrapper;\n }\n _prefilterInternal(texture) {\n const width = texture.getSize().width;\n const mipmapsCount = Scalar.ILog2(width) + 1;\n const effect = this._effectWrapper.effect;\n const outputTexture = this._createRenderTarget(width);\n this._effectRenderer.saveStates();\n this._effectRenderer.setViewport();\n const intTexture = texture.getInternalTexture();\n if (intTexture) {\n // Just in case generate fresh clean mips.\n this._engine.updateTextureSamplingMode(3, intTexture, true);\n }\n this._effectRenderer.applyEffectWrapper(this._effectWrapper);\n const directions = [\n [new Vector3(0, 0, -1), new Vector3(0, -1, 0), new Vector3(1, 0, 0)],\n [new Vector3(0, 0, 1), new Vector3(0, -1, 0), new Vector3(-1, 0, 0)],\n [new Vector3(1, 0, 0), new Vector3(0, 0, 1), new Vector3(0, 1, 0)],\n [new Vector3(1, 0, 0), new Vector3(0, 0, -1), new Vector3(0, -1, 0)],\n [new Vector3(1, 0, 0), new Vector3(0, -1, 0), new Vector3(0, 0, 1)],\n [new Vector3(-1, 0, 0), new Vector3(0, -1, 0), new Vector3(0, 0, -1)], // NegativeZ\n ];\n effect.setFloat(\"hdrScale\", this.hdrScale);\n effect.setFloat2(\"vFilteringInfo\", texture.getSize().width, mipmapsCount);\n effect.setTexture(\"inputTexture\", texture);\n for (let face = 0; face < 6; face++) {\n effect.setVector3(\"up\", directions[face][0]);\n effect.setVector3(\"right\", directions[face][1]);\n effect.setVector3(\"front\", directions[face][2]);\n for (let lod = 0; lod < mipmapsCount; lod++) {\n this._engine.bindFramebuffer(outputTexture, face, undefined, undefined, true, lod);\n this._effectRenderer.applyEffectWrapper(this._effectWrapper);\n let alpha = Math.pow(2, (lod - this._lodGenerationOffset) / this._lodGenerationScale) / width;\n if (lod === 0) {\n alpha = 0;\n }\n effect.setFloat(\"alphaG\", alpha);\n this._effectRenderer.draw();\n }\n }\n // Cleanup\n this._effectRenderer.restoreStates();\n this._engine.restoreDefaultFramebuffer();\n this._engine._releaseTexture(texture._texture);\n // Internal Swap\n const type = outputTexture.texture.type;\n const format = outputTexture.texture.format;\n outputTexture._swapAndDie(texture._texture);\n texture._texture.type = type;\n texture._texture.format = format;\n // New settings\n texture.gammaSpace = false;\n texture.lodGenerationOffset = this._lodGenerationOffset;\n texture.lodGenerationScale = this._lodGenerationScale;\n texture._prefiltered = true;\n return texture;\n }\n _createEffect(texture, onCompiled) {\n const defines = [];\n if (texture.gammaSpace) {\n defines.push(\"#define GAMMA_INPUT\");\n }\n defines.push(\"#define NUM_SAMPLES \" + this.quality + \"u\"); // unsigned int\n const effectWrapper = new EffectWrapper({\n engine: this._engine,\n name: \"hdrFiltering\",\n vertexShader: \"hdrFiltering\",\n fragmentShader: \"hdrFiltering\",\n samplerNames: [\"inputTexture\"],\n uniformNames: [\"vSampleDirections\", \"vWeights\", \"up\", \"right\", \"front\", \"vFilteringInfo\", \"hdrScale\", \"alphaG\"],\n useShaderStore: true,\n defines,\n onCompiled: onCompiled,\n });\n return effectWrapper;\n }\n /**\n * Get a value indicating if the filter is ready to be used\n * @param texture Texture to filter\n * @returns true if the filter is ready\n */\n isReady(texture) {\n return texture.isReady() && this._effectWrapper.effect.isReady();\n }\n /**\n * Prefilters a cube texture to have mipmap levels representing roughness values.\n * Prefiltering will be invoked at the end of next rendering pass.\n * This has to be done once the map is loaded, and has not been prefiltered by a third party software.\n * See http://blog.selfshadow.com/publications/s2013-shading-course/karis/s2013_pbs_epic_notes_v2.pdf for more information\n * @param texture Texture to filter\n * @param onFinished Callback when filtering is done\n * @returns Promise called when prefiltering is done\n */\n prefilter(texture, onFinished = null) {\n if (!this._engine._features.allowTexturePrefiltering) {\n Logger.Warn(\"HDR prefiltering is not available in WebGL 1., you can use real time filtering instead.\");\n return Promise.reject(\"HDR prefiltering is not available in WebGL 1., you can use real time filtering instead.\");\n }\n return new Promise((resolve) => {\n this._effectRenderer = new EffectRenderer(this._engine);\n this._effectWrapper = this._createEffect(texture);\n this._effectWrapper.effect.executeWhenCompiled(() => {\n this._prefilterInternal(texture);\n this._effectRenderer.dispose();\n this._effectWrapper.dispose();\n resolve();\n if (onFinished) {\n onFinished();\n }\n });\n });\n }\n }\n\n /**\n * This groups tools to convert HDR texture to native colors array.\n */\n class HDRTools {\n static _Ldexp(mantissa, exponent) {\n if (exponent > 1023) {\n return mantissa * Math.pow(2, 1023) * Math.pow(2, exponent - 1023);\n }\n if (exponent < -1074) {\n return mantissa * Math.pow(2, -1074) * Math.pow(2, exponent + 1074);\n }\n return mantissa * Math.pow(2, exponent);\n }\n static _Rgbe2float(float32array, red, green, blue, exponent, index) {\n if (exponent > 0) {\n /*nonzero pixel*/\n exponent = this._Ldexp(1.0, exponent - (128 + 8));\n float32array[index + 0] = red * exponent;\n float32array[index + 1] = green * exponent;\n float32array[index + 2] = blue * exponent;\n }\n else {\n float32array[index + 0] = 0;\n float32array[index + 1] = 0;\n float32array[index + 2] = 0;\n }\n }\n static _ReadStringLine(uint8array, startIndex) {\n let line = \"\";\n let character = \"\";\n for (let i = startIndex; i < uint8array.length - startIndex; i++) {\n character = String.fromCharCode(uint8array[i]);\n if (character == \"\\n\") {\n break;\n }\n line += character;\n }\n return line;\n }\n /**\n * Reads header information from an RGBE texture stored in a native array.\n * More information on this format are available here:\n * https://en.wikipedia.org/wiki/RGBE_image_format\n *\n * @param uint8array The binary file stored in native array.\n * @returns The header information.\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n static RGBE_ReadHeader(uint8array) {\n let height = 0;\n let width = 0;\n let line = this._ReadStringLine(uint8array, 0);\n if (line[0] != \"#\" || line[1] != \"?\") {\n throw \"Bad HDR Format.\";\n }\n let endOfHeader = false;\n let findFormat = false;\n let lineIndex = 0;\n do {\n lineIndex += line.length + 1;\n line = this._ReadStringLine(uint8array, lineIndex);\n if (line == \"FORMAT=32-bit_rle_rgbe\") {\n findFormat = true;\n }\n else if (line.length == 0) {\n endOfHeader = true;\n }\n } while (!endOfHeader);\n if (!findFormat) {\n throw \"HDR Bad header format, unsupported FORMAT\";\n }\n lineIndex += line.length + 1;\n line = this._ReadStringLine(uint8array, lineIndex);\n const sizeRegexp = /^-Y (.*) \\+X (.*)$/g;\n const match = sizeRegexp.exec(line);\n // TODO. Support +Y and -X if needed.\n if (!match || match.length < 3) {\n throw \"HDR Bad header format, no size\";\n }\n width = parseInt(match[2]);\n height = parseInt(match[1]);\n if (width < 8 || width > 0x7fff) {\n throw \"HDR Bad header format, unsupported size\";\n }\n lineIndex += line.length + 1;\n return {\n height: height,\n width: width,\n dataPosition: lineIndex,\n };\n }\n /**\n * Returns the cubemap information (each faces texture data) extracted from an RGBE texture.\n * This RGBE texture needs to store the information as a panorama.\n *\n * More information on this format are available here:\n * https://en.wikipedia.org/wiki/RGBE_image_format\n *\n * @param buffer The binary file stored in an array buffer.\n * @param size The expected size of the extracted cubemap.\n * @returns The Cube Map information.\n */\n static GetCubeMapTextureData(buffer, size, supersample = false) {\n const uint8array = new Uint8Array(buffer);\n const hdrInfo = this.RGBE_ReadHeader(uint8array);\n const data = this.RGBE_ReadPixels(uint8array, hdrInfo);\n const cubeMapData = PanoramaToCubeMapTools.ConvertPanoramaToCubemap(data, hdrInfo.width, hdrInfo.height, size, supersample);\n return cubeMapData;\n }\n /**\n * Returns the pixels data extracted from an RGBE texture.\n * This pixels will be stored left to right up to down in the R G B order in one array.\n *\n * More information on this format are available here:\n * https://en.wikipedia.org/wiki/RGBE_image_format\n *\n * @param uint8array The binary file stored in an array buffer.\n * @param hdrInfo The header information of the file.\n * @returns The pixels data in RGB right to left up to down order.\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n static RGBE_ReadPixels(uint8array, hdrInfo) {\n return this._RGBEReadPixelsRLE(uint8array, hdrInfo);\n }\n static _RGBEReadPixelsRLE(uint8array, hdrInfo) {\n let num_scanlines = hdrInfo.height;\n const scanline_width = hdrInfo.width;\n let a, b, c, d, count;\n let dataIndex = hdrInfo.dataPosition;\n let index = 0, endIndex = 0, i = 0;\n const scanLineArrayBuffer = new ArrayBuffer(scanline_width * 4); // four channel R G B E\n const scanLineArray = new Uint8Array(scanLineArrayBuffer);\n // 3 channels of 4 bytes per pixel in float.\n const resultBuffer = new ArrayBuffer(hdrInfo.width * hdrInfo.height * 4 * 3);\n const resultArray = new Float32Array(resultBuffer);\n // read in each successive scanline\n while (num_scanlines > 0) {\n a = uint8array[dataIndex++];\n b = uint8array[dataIndex++];\n c = uint8array[dataIndex++];\n d = uint8array[dataIndex++];\n if (a != 2 || b != 2 || c & 0x80 || hdrInfo.width < 8 || hdrInfo.width > 32767) {\n return this._RGBEReadPixelsNOTRLE(uint8array, hdrInfo);\n }\n if (((c << 8) | d) != scanline_width) {\n throw \"HDR Bad header format, wrong scan line width\";\n }\n index = 0;\n // read each of the four channels for the scanline into the buffer\n for (i = 0; i < 4; i++) {\n endIndex = (i + 1) * scanline_width;\n while (index < endIndex) {\n a = uint8array[dataIndex++];\n b = uint8array[dataIndex++];\n if (a > 128) {\n // a run of the same value\n count = a - 128;\n if (count == 0 || count > endIndex - index) {\n throw \"HDR Bad Format, bad scanline data (run)\";\n }\n while (count-- > 0) {\n scanLineArray[index++] = b;\n }\n }\n else {\n // a non-run\n count = a;\n if (count == 0 || count > endIndex - index) {\n throw \"HDR Bad Format, bad scanline data (non-run)\";\n }\n scanLineArray[index++] = b;\n if (--count > 0) {\n for (let j = 0; j < count; j++) {\n scanLineArray[index++] = uint8array[dataIndex++];\n }\n }\n }\n }\n }\n // now convert data from buffer into floats\n for (i = 0; i < scanline_width; i++) {\n a = scanLineArray[i];\n b = scanLineArray[i + scanline_width];\n c = scanLineArray[i + 2 * scanline_width];\n d = scanLineArray[i + 3 * scanline_width];\n this._Rgbe2float(resultArray, a, b, c, d, (hdrInfo.height - num_scanlines) * scanline_width * 3 + i * 3);\n }\n num_scanlines--;\n }\n return resultArray;\n }\n static _RGBEReadPixelsNOTRLE(uint8array, hdrInfo) {\n // this file is not run length encoded\n // read values sequentially\n let num_scanlines = hdrInfo.height;\n const scanline_width = hdrInfo.width;\n let a, b, c, d, i;\n let dataIndex = hdrInfo.dataPosition;\n // 3 channels of 4 bytes per pixel in float.\n const resultBuffer = new ArrayBuffer(hdrInfo.width * hdrInfo.height * 4 * 3);\n const resultArray = new Float32Array(resultBuffer);\n // read in each successive scanline\n while (num_scanlines > 0) {\n for (i = 0; i < hdrInfo.width; i++) {\n a = uint8array[dataIndex++];\n b = uint8array[dataIndex++];\n c = uint8array[dataIndex++];\n d = uint8array[dataIndex++];\n this._Rgbe2float(resultArray, a, b, c, d, (hdrInfo.height - num_scanlines) * scanline_width * 3 + i * 3);\n }\n num_scanlines--;\n }\n return resultArray;\n }\n }\n\n /**\n * This represents a texture coming from an HDR input.\n *\n * The only supported format is currently panorama picture stored in RGBE format.\n * Example of such files can be found on Poly Haven: https://polyhaven.com/hdris\n */\n class HDRCubeTexture extends BaseTexture {\n /**\n * Sets whether or not the texture is blocking during loading.\n */\n set isBlocking(value) {\n this._isBlocking = value;\n }\n /**\n * Gets whether or not the texture is blocking during loading.\n */\n get isBlocking() {\n return this._isBlocking;\n }\n /**\n * Sets texture matrix rotation angle around Y axis in radians.\n */\n set rotationY(value) {\n this._rotationY = value;\n this.setReflectionTextureMatrix(Matrix.RotationY(this._rotationY));\n }\n /**\n * Gets texture matrix rotation angle around Y axis radians.\n */\n get rotationY() {\n return this._rotationY;\n }\n /**\n * Gets or sets the size of the bounding box associated with the cube texture\n * When defined, the cubemap will switch to local mode\n * @see https://community.arm.com/graphics/b/blog/posts/reflections-based-on-local-cubemaps-in-unity\n * @example https://www.babylonjs-playground.com/#RNASML\n */\n set boundingBoxSize(value) {\n if (this._boundingBoxSize && this._boundingBoxSize.equals(value)) {\n return;\n }\n this._boundingBoxSize = value;\n const scene = this.getScene();\n if (scene) {\n scene.markAllMaterialsAsDirty(1);\n }\n }\n get boundingBoxSize() {\n return this._boundingBoxSize;\n }\n /**\n * Instantiates an HDRTexture from the following parameters.\n *\n * @param url The location of the HDR raw data (Panorama stored in RGBE format)\n * @param sceneOrEngine The scene or engine the texture will be used in\n * @param size The cubemap desired size (the more it increases the longer the generation will be)\n * @param noMipmap Forces to not generate the mipmap if true\n * @param generateHarmonics Specifies whether you want to extract the polynomial harmonics during the generation process\n * @param gammaSpace Specifies if the texture will be use in gamma or linear space (the PBR material requires those texture in linear space, but the standard material would require them in Gamma space)\n * @param prefilterOnLoad Prefilters HDR texture to allow use of this texture as a PBR reflection texture.\n * @param onLoad\n * @param onError\n */\n constructor(url, sceneOrEngine, size, noMipmap = false, generateHarmonics = true, gammaSpace = false, prefilterOnLoad = false, onLoad = null, onError = null, supersample = false) {\n var _a;\n super(sceneOrEngine);\n this._generateHarmonics = true;\n this._onError = null;\n this._isBlocking = true;\n this._rotationY = 0;\n /**\n * Gets or sets the center of the bounding box associated with the cube texture\n * It must define where the camera used to render the texture was set\n */\n this.boundingBoxPosition = Vector3.Zero();\n /**\n * Observable triggered once the texture has been loaded.\n */\n this.onLoadObservable = new Observable$1();\n if (!url) {\n return;\n }\n this._coordinatesMode = Texture.CUBIC_MODE;\n this.name = url;\n this.url = url;\n this.hasAlpha = false;\n this.isCube = true;\n this._textureMatrix = Matrix.Identity();\n this._prefilterOnLoad = prefilterOnLoad;\n this._onLoad = () => {\n this.onLoadObservable.notifyObservers(this);\n if (onLoad) {\n onLoad();\n }\n };\n this._onError = onError;\n this.gammaSpace = gammaSpace;\n this._noMipmap = noMipmap;\n this._size = size;\n this._supersample = supersample;\n this._generateHarmonics = generateHarmonics;\n this._texture = this._getFromCache(url, this._noMipmap, undefined, undefined, undefined, this.isCube);\n if (!this._texture) {\n if (!((_a = this.getScene()) === null || _a === void 0 ? void 0 : _a.useDelayedTextureLoading)) {\n this._loadTexture();\n }\n else {\n this.delayLoadState = 4;\n }\n }\n else {\n if (this._texture.isReady) {\n Tools.SetImmediate(() => this._onLoad());\n }\n else {\n this._texture.onLoadedObservable.add(this._onLoad);\n }\n }\n }\n /**\n * Get the current class name of the texture useful for serialization or dynamic coding.\n * @returns \"HDRCubeTexture\"\n */\n getClassName() {\n return \"HDRCubeTexture\";\n }\n /**\n * Occurs when the file is raw .hdr file.\n */\n _loadTexture() {\n const engine = this._getEngine();\n const caps = engine.getCaps();\n let textureType = 0;\n if (caps.textureFloat && caps.textureFloatLinearFiltering) {\n textureType = 1;\n }\n else if (caps.textureHalfFloat && caps.textureHalfFloatLinearFiltering) {\n textureType = 2;\n }\n const callback = (buffer) => {\n this.lodGenerationOffset = 0.0;\n this.lodGenerationScale = 0.8;\n // Extract the raw linear data.\n const data = HDRTools.GetCubeMapTextureData(buffer, this._size, this._supersample);\n // Generate harmonics if needed.\n if (this._generateHarmonics) {\n const sphericalPolynomial = CubeMapToSphericalPolynomialTools.ConvertCubeMapToSphericalPolynomial(data);\n this.sphericalPolynomial = sphericalPolynomial;\n }\n const results = [];\n let byteArray = null;\n let shortArray = null;\n // Push each faces.\n for (let j = 0; j < 6; j++) {\n // Create fallback array\n if (textureType === 2) {\n shortArray = new Uint16Array(this._size * this._size * 3);\n }\n else if (textureType === 0) {\n // 3 channels of 1 bytes per pixel in bytes.\n byteArray = new Uint8Array(this._size * this._size * 3);\n }\n const dataFace = data[HDRCubeTexture._FacesMapping[j]];\n // If special cases.\n if (this.gammaSpace || shortArray || byteArray) {\n for (let i = 0; i < this._size * this._size; i++) {\n // Put in gamma space if requested.\n if (this.gammaSpace) {\n dataFace[i * 3 + 0] = Math.pow(dataFace[i * 3 + 0], ToGammaSpace);\n dataFace[i * 3 + 1] = Math.pow(dataFace[i * 3 + 1], ToGammaSpace);\n dataFace[i * 3 + 2] = Math.pow(dataFace[i * 3 + 2], ToGammaSpace);\n }\n // Convert to half float texture for fallback.\n if (shortArray) {\n shortArray[i * 3 + 0] = ToHalfFloat(dataFace[i * 3 + 0]);\n shortArray[i * 3 + 1] = ToHalfFloat(dataFace[i * 3 + 1]);\n shortArray[i * 3 + 2] = ToHalfFloat(dataFace[i * 3 + 2]);\n }\n // Convert to int texture for fallback.\n if (byteArray) {\n let r = Math.max(dataFace[i * 3 + 0] * 255, 0);\n let g = Math.max(dataFace[i * 3 + 1] * 255, 0);\n let b = Math.max(dataFace[i * 3 + 2] * 255, 0);\n // May use luminance instead if the result is not accurate.\n const max = Math.max(Math.max(r, g), b);\n if (max > 255) {\n const scale = 255 / max;\n r *= scale;\n g *= scale;\n b *= scale;\n }\n byteArray[i * 3 + 0] = r;\n byteArray[i * 3 + 1] = g;\n byteArray[i * 3 + 2] = b;\n }\n }\n }\n if (shortArray) {\n results.push(shortArray);\n }\n else if (byteArray) {\n results.push(byteArray);\n }\n else {\n results.push(dataFace);\n }\n }\n return results;\n };\n if (engine._features.allowTexturePrefiltering && this._prefilterOnLoad) {\n const previousOnLoad = this._onLoad;\n const hdrFiltering = new HDRFiltering(engine);\n this._onLoad = () => {\n hdrFiltering.prefilter(this, previousOnLoad);\n };\n }\n this._texture = engine.createRawCubeTextureFromUrl(this.url, this.getScene(), this._size, 4, textureType, this._noMipmap, callback, null, this._onLoad, this._onError);\n }\n clone() {\n const newTexture = new HDRCubeTexture(this.url, this.getScene() || this._getEngine(), this._size, this._noMipmap, this._generateHarmonics, this.gammaSpace);\n // Base texture\n newTexture.level = this.level;\n newTexture.wrapU = this.wrapU;\n newTexture.wrapV = this.wrapV;\n newTexture.coordinatesIndex = this.coordinatesIndex;\n newTexture.coordinatesMode = this.coordinatesMode;\n return newTexture;\n }\n // Methods\n delayLoad() {\n if (this.delayLoadState !== 4) {\n return;\n }\n this.delayLoadState = 1;\n this._texture = this._getFromCache(this.url, this._noMipmap);\n if (!this._texture) {\n this._loadTexture();\n }\n }\n /**\n * Get the texture reflection matrix used to rotate/transform the reflection.\n * @returns the reflection matrix\n */\n getReflectionTextureMatrix() {\n return this._textureMatrix;\n }\n /**\n * Set the texture reflection matrix used to rotate/transform the reflection.\n * @param value Define the reflection matrix to set\n */\n setReflectionTextureMatrix(value) {\n var _a;\n this._textureMatrix = value;\n if (value.updateFlag === this._textureMatrix.updateFlag) {\n return;\n }\n if (value.isIdentity() !== this._textureMatrix.isIdentity()) {\n (_a = this.getScene()) === null || _a === void 0 ? void 0 : _a.markAllMaterialsAsDirty(1, (mat) => mat.getActiveTextures().indexOf(this) !== -1);\n }\n }\n /**\n * Dispose the texture and release its associated resources.\n */\n dispose() {\n this.onLoadObservable.clear();\n super.dispose();\n }\n /**\n * Parses a JSON representation of an HDR Texture in order to create the texture\n * @param parsedTexture Define the JSON representation\n * @param scene Define the scene the texture should be created in\n * @param rootUrl Define the root url in case we need to load relative dependencies\n * @returns the newly created texture after parsing\n */\n static Parse(parsedTexture, scene, rootUrl) {\n let texture = null;\n if (parsedTexture.name && !parsedTexture.isRenderTarget) {\n texture = new HDRCubeTexture(rootUrl + parsedTexture.name, scene, parsedTexture.size, parsedTexture.noMipmap, parsedTexture.generateHarmonics, parsedTexture.useInGammaSpace);\n texture.name = parsedTexture.name;\n texture.hasAlpha = parsedTexture.hasAlpha;\n texture.level = parsedTexture.level;\n texture.coordinatesMode = parsedTexture.coordinatesMode;\n texture.isBlocking = parsedTexture.isBlocking;\n }\n if (texture) {\n if (parsedTexture.boundingBoxPosition) {\n texture.boundingBoxPosition = Vector3.FromArray(parsedTexture.boundingBoxPosition);\n }\n if (parsedTexture.boundingBoxSize) {\n texture.boundingBoxSize = Vector3.FromArray(parsedTexture.boundingBoxSize);\n }\n if (parsedTexture.rotationY) {\n texture.rotationY = parsedTexture.rotationY;\n }\n }\n return texture;\n }\n serialize() {\n if (!this.name) {\n return null;\n }\n const serializationObject = {};\n serializationObject.name = this.name;\n serializationObject.hasAlpha = this.hasAlpha;\n serializationObject.isCube = true;\n serializationObject.level = this.level;\n serializationObject.size = this._size;\n serializationObject.coordinatesMode = this.coordinatesMode;\n serializationObject.useInGammaSpace = this.gammaSpace;\n serializationObject.generateHarmonics = this._generateHarmonics;\n serializationObject.customType = \"BABYLON.HDRCubeTexture\";\n serializationObject.noMipmap = this._noMipmap;\n serializationObject.isBlocking = this._isBlocking;\n serializationObject.rotationY = this._rotationY;\n return serializationObject;\n }\n }\n HDRCubeTexture._FacesMapping = [\"right\", \"left\", \"up\", \"down\", \"front\", \"back\"];\n RegisterClass(\"BABYLON.HDRCubeTexture\", HDRCubeTexture);\n\n ThinEngine.prototype.updateVideoTexture = function (texture, video, invertY) {\n if (!texture || texture._isDisabled) {\n return;\n }\n const glformat = this._getInternalFormat(texture.format);\n const internalFormat = this._getRGBABufferInternalSizedFormat(0, texture.format);\n const wasPreviouslyBound = this._bindTextureDirectly(this._gl.TEXTURE_2D, texture, true);\n this._unpackFlipY(!invertY); // Video are upside down by default\n try {\n // Testing video texture support\n if (this._videoTextureSupported === undefined) {\n // clear old errors just in case.\n this._gl.getError();\n this._gl.texImage2D(this._gl.TEXTURE_2D, 0, internalFormat, glformat, this._gl.UNSIGNED_BYTE, video);\n if (this._gl.getError() !== 0) {\n this._videoTextureSupported = false;\n }\n else {\n this._videoTextureSupported = true;\n }\n }\n // Copy video through the current working canvas if video texture is not supported\n if (!this._videoTextureSupported) {\n if (!texture._workingCanvas) {\n texture._workingCanvas = this.createCanvas(texture.width, texture.height);\n const context = texture._workingCanvas.getContext(\"2d\");\n if (!context) {\n throw new Error(\"Unable to get 2d context\");\n }\n texture._workingContext = context;\n texture._workingCanvas.width = texture.width;\n texture._workingCanvas.height = texture.height;\n }\n texture._workingContext.clearRect(0, 0, texture.width, texture.height);\n texture._workingContext.drawImage(video, 0, 0, video.videoWidth, video.videoHeight, 0, 0, texture.width, texture.height);\n this._gl.texImage2D(this._gl.TEXTURE_2D, 0, internalFormat, glformat, this._gl.UNSIGNED_BYTE, texture._workingCanvas);\n }\n else {\n this._gl.texImage2D(this._gl.TEXTURE_2D, 0, internalFormat, glformat, this._gl.UNSIGNED_BYTE, video);\n }\n if (texture.generateMipMaps) {\n this._gl.generateMipmap(this._gl.TEXTURE_2D);\n }\n if (!wasPreviouslyBound) {\n this._bindTextureDirectly(this._gl.TEXTURE_2D, null);\n }\n // this.resetTextureCache();\n texture.isReady = true;\n }\n catch (ex) {\n // Something unexpected\n // Let's disable the texture\n texture._isDisabled = true;\n }\n };\n\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n ThinEngine.prototype.createExternalTexture = function (video) {\n return null;\n };\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n ThinEngine.prototype.setExternalTexture = function (name, texture) {\n throw new Error(\"setExternalTexture: This engine does not support external textures!\");\n };\n\n //private static _TYPE_NO_DATA = 0;\n const _TYPE_INDEXED = 1;\n const _TYPE_RGB = 2;\n const _TYPE_GREY = 3;\n const _TYPE_RLE_INDEXED = 9;\n const _TYPE_RLE_RGB = 10;\n const _TYPE_RLE_GREY = 11;\n const _ORIGIN_MASK = 0x30;\n const _ORIGIN_SHIFT = 0x04;\n const _ORIGIN_BL = 0x00;\n const _ORIGIN_BR = 0x01;\n const _ORIGIN_UL = 0x02;\n const _ORIGIN_UR = 0x03;\n /**\n * Gets the header of a TGA file\n * @param data defines the TGA data\n * @returns the header\n */\n function GetTGAHeader(data) {\n let offset = 0;\n const header = {\n id_length: data[offset++],\n colormap_type: data[offset++],\n image_type: data[offset++],\n colormap_index: data[offset++] | (data[offset++] << 8),\n colormap_length: data[offset++] | (data[offset++] << 8),\n colormap_size: data[offset++],\n origin: [data[offset++] | (data[offset++] << 8), data[offset++] | (data[offset++] << 8)],\n width: data[offset++] | (data[offset++] << 8),\n height: data[offset++] | (data[offset++] << 8),\n pixel_size: data[offset++],\n flags: data[offset++],\n };\n return header;\n }\n /**\n * Uploads TGA content to a Babylon Texture\n * @internal\n */\n function UploadContent(texture, data) {\n // Not enough data to contain header ?\n if (data.length < 19) {\n Logger.Error(\"Unable to load TGA file - Not enough data to contain header\");\n return;\n }\n // Read Header\n let offset = 18;\n const header = GetTGAHeader(data);\n // Assume it's a valid Targa file.\n if (header.id_length + offset > data.length) {\n Logger.Error(\"Unable to load TGA file - Not enough data\");\n return;\n }\n // Skip not needed data\n offset += header.id_length;\n let use_rle = false;\n let use_pal = false;\n let use_grey = false;\n // Get some informations.\n switch (header.image_type) {\n case _TYPE_RLE_INDEXED:\n use_rle = true;\n // eslint-disable-next-line no-fallthrough\n case _TYPE_INDEXED:\n use_pal = true;\n break;\n case _TYPE_RLE_RGB:\n use_rle = true;\n // eslint-disable-next-line no-fallthrough\n case _TYPE_RGB:\n // use_rgb = true;\n break;\n case _TYPE_RLE_GREY:\n use_rle = true;\n // eslint-disable-next-line no-fallthrough\n case _TYPE_GREY:\n use_grey = true;\n break;\n }\n let pixel_data;\n // var numAlphaBits = header.flags & 0xf;\n const pixel_size = header.pixel_size >> 3;\n const pixel_total = header.width * header.height * pixel_size;\n // Read palettes\n let palettes;\n if (use_pal) {\n palettes = data.subarray(offset, (offset += header.colormap_length * (header.colormap_size >> 3)));\n }\n // Read LRE\n if (use_rle) {\n pixel_data = new Uint8Array(pixel_total);\n let c, count, i;\n let localOffset = 0;\n const pixels = new Uint8Array(pixel_size);\n while (offset < pixel_total && localOffset < pixel_total) {\n c = data[offset++];\n count = (c & 0x7f) + 1;\n // RLE pixels\n if (c & 0x80) {\n // Bind pixel tmp array\n for (i = 0; i < pixel_size; ++i) {\n pixels[i] = data[offset++];\n }\n // Copy pixel array\n for (i = 0; i < count; ++i) {\n pixel_data.set(pixels, localOffset + i * pixel_size);\n }\n localOffset += pixel_size * count;\n }\n // Raw pixels\n else {\n count *= pixel_size;\n for (i = 0; i < count; ++i) {\n pixel_data[localOffset + i] = data[offset++];\n }\n localOffset += count;\n }\n }\n }\n // RAW Pixels\n else {\n pixel_data = data.subarray(offset, (offset += use_pal ? header.width * header.height : pixel_total));\n }\n // Load to texture\n let x_start, y_start, x_step, y_step, y_end, x_end;\n switch ((header.flags & _ORIGIN_MASK) >> _ORIGIN_SHIFT) {\n default:\n case _ORIGIN_UL:\n x_start = 0;\n x_step = 1;\n x_end = header.width;\n y_start = 0;\n y_step = 1;\n y_end = header.height;\n break;\n case _ORIGIN_BL:\n x_start = 0;\n x_step = 1;\n x_end = header.width;\n y_start = header.height - 1;\n y_step = -1;\n y_end = -1;\n break;\n case _ORIGIN_UR:\n x_start = header.width - 1;\n x_step = -1;\n x_end = -1;\n y_start = 0;\n y_step = 1;\n y_end = header.height;\n break;\n case _ORIGIN_BR:\n x_start = header.width - 1;\n x_step = -1;\n x_end = -1;\n y_start = header.height - 1;\n y_step = -1;\n y_end = -1;\n break;\n }\n // Load the specify method\n const func = \"_getImageData\" + (use_grey ? \"Grey\" : \"\") + header.pixel_size + \"bits\";\n const imageData = TGATools[func](header, palettes, pixel_data, y_start, y_step, y_end, x_start, x_step, x_end);\n const engine = texture.getEngine();\n engine._uploadDataToTextureDirectly(texture, imageData);\n }\n /**\n * @internal\n */\n function _getImageData8bits(header, palettes, pixel_data, y_start, y_step, y_end, x_start, x_step, x_end) {\n const image = pixel_data, colormap = palettes;\n const width = header.width, height = header.height;\n let color, i = 0, x, y;\n const imageData = new Uint8Array(width * height * 4);\n for (y = y_start; y !== y_end; y += y_step) {\n for (x = x_start; x !== x_end; x += x_step, i++) {\n color = image[i];\n imageData[(x + width * y) * 4 + 3] = 255;\n imageData[(x + width * y) * 4 + 2] = colormap[color * 3 + 0];\n imageData[(x + width * y) * 4 + 1] = colormap[color * 3 + 1];\n imageData[(x + width * y) * 4 + 0] = colormap[color * 3 + 2];\n }\n }\n return imageData;\n }\n /**\n * @internal\n */\n function _getImageData16bits(header, palettes, pixel_data, y_start, y_step, y_end, x_start, x_step, x_end) {\n const image = pixel_data;\n const width = header.width, height = header.height;\n let color, i = 0, x, y;\n const imageData = new Uint8Array(width * height * 4);\n for (y = y_start; y !== y_end; y += y_step) {\n for (x = x_start; x !== x_end; x += x_step, i += 2) {\n color = image[i + 0] + (image[i + 1] << 8); // Inversed ?\n const r = ((((color & 0x7c00) >> 10) * 255) / 0x1f) | 0;\n const g = ((((color & 0x03e0) >> 5) * 255) / 0x1f) | 0;\n const b = (((color & 0x001f) * 255) / 0x1f) | 0;\n imageData[(x + width * y) * 4 + 0] = r;\n imageData[(x + width * y) * 4 + 1] = g;\n imageData[(x + width * y) * 4 + 2] = b;\n imageData[(x + width * y) * 4 + 3] = color & 0x8000 ? 0 : 255;\n }\n }\n return imageData;\n }\n /**\n * @internal\n */\n function _getImageData24bits(header, palettes, pixel_data, y_start, y_step, y_end, x_start, x_step, x_end) {\n const image = pixel_data;\n const width = header.width, height = header.height;\n let i = 0, x, y;\n const imageData = new Uint8Array(width * height * 4);\n for (y = y_start; y !== y_end; y += y_step) {\n for (x = x_start; x !== x_end; x += x_step, i += 3) {\n imageData[(x + width * y) * 4 + 3] = 255;\n imageData[(x + width * y) * 4 + 2] = image[i + 0];\n imageData[(x + width * y) * 4 + 1] = image[i + 1];\n imageData[(x + width * y) * 4 + 0] = image[i + 2];\n }\n }\n return imageData;\n }\n /**\n * @internal\n */\n function _getImageData32bits(header, palettes, pixel_data, y_start, y_step, y_end, x_start, x_step, x_end) {\n const image = pixel_data;\n const width = header.width, height = header.height;\n let i = 0, x, y;\n const imageData = new Uint8Array(width * height * 4);\n for (y = y_start; y !== y_end; y += y_step) {\n for (x = x_start; x !== x_end; x += x_step, i += 4) {\n imageData[(x + width * y) * 4 + 2] = image[i + 0];\n imageData[(x + width * y) * 4 + 1] = image[i + 1];\n imageData[(x + width * y) * 4 + 0] = image[i + 2];\n imageData[(x + width * y) * 4 + 3] = image[i + 3];\n }\n }\n return imageData;\n }\n /**\n * @internal\n */\n function _getImageDataGrey8bits(header, palettes, pixel_data, y_start, y_step, y_end, x_start, x_step, x_end) {\n const image = pixel_data;\n const width = header.width, height = header.height;\n let color, i = 0, x, y;\n const imageData = new Uint8Array(width * height * 4);\n for (y = y_start; y !== y_end; y += y_step) {\n for (x = x_start; x !== x_end; x += x_step, i++) {\n color = image[i];\n imageData[(x + width * y) * 4 + 0] = color;\n imageData[(x + width * y) * 4 + 1] = color;\n imageData[(x + width * y) * 4 + 2] = color;\n imageData[(x + width * y) * 4 + 3] = 255;\n }\n }\n return imageData;\n }\n /**\n * @internal\n */\n function _getImageDataGrey16bits(header, palettes, pixel_data, y_start, y_step, y_end, x_start, x_step, x_end) {\n const image = pixel_data;\n const width = header.width, height = header.height;\n let i = 0, x, y;\n const imageData = new Uint8Array(width * height * 4);\n for (y = y_start; y !== y_end; y += y_step) {\n for (x = x_start; x !== x_end; x += x_step, i += 2) {\n imageData[(x + width * y) * 4 + 0] = image[i + 0];\n imageData[(x + width * y) * 4 + 1] = image[i + 0];\n imageData[(x + width * y) * 4 + 2] = image[i + 0];\n imageData[(x + width * y) * 4 + 3] = image[i + 1];\n }\n }\n return imageData;\n }\n /**\n * Based on jsTGALoader - Javascript loader for TGA file\n * By Vincent Thibault\n * @see http://blog.robrowser.com/javascript-tga-loader.html\n */\n const TGATools = {\n /**\n * Gets the header of a TGA file\n * @param data defines the TGA data\n * @returns the header\n */\n GetTGAHeader,\n /**\n * Uploads TGA content to a Babylon Texture\n * @internal\n */\n UploadContent,\n /** @internal */\n _getImageData8bits,\n /** @internal */\n _getImageData16bits,\n /** @internal */\n _getImageData24bits,\n /** @internal */\n _getImageData32bits,\n /** @internal */\n _getImageDataGrey8bits,\n /** @internal */\n _getImageDataGrey16bits,\n };\n\n /**\n * Implementation of the TGA Texture Loader.\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class _TGATextureLoader {\n constructor() {\n /**\n * Defines whether the loader supports cascade loading the different faces.\n */\n this.supportCascades = false;\n }\n /**\n * This returns if the loader support the current file information.\n * @param extension defines the file extension of the file being loaded\n * @returns true if the loader can load the specified file\n */\n canLoad(extension) {\n return extension.endsWith(\".tga\");\n }\n /**\n * Uploads the cube texture data to the WebGL texture. It has already been bound.\n */\n loadCubeData() {\n throw \".env not supported in Cube.\";\n }\n /**\n * Uploads the 2D texture data to the WebGL texture. It has already been bound once in the callback.\n * @param data contains the texture data\n * @param texture defines the BabylonJS internal texture\n * @param callback defines the method to call once ready to upload\n */\n loadData(data, texture, callback) {\n const bytes = new Uint8Array(data.buffer, data.byteOffset, data.byteLength);\n const header = GetTGAHeader(bytes);\n callback(header.width, header.height, texture.generateMipMaps, false, () => {\n UploadContent(texture, bytes);\n });\n }\n }\n // Register the loader.\n Engine._TextureLoaders.push(new _TGATextureLoader());\n\n /**\n * Implementation of the HDR Texture Loader.\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class _HDRTextureLoader {\n constructor() {\n /**\n * Defines whether the loader supports cascade loading the different faces.\n */\n this.supportCascades = false;\n }\n /**\n * This returns if the loader support the current file information.\n * @param extension defines the file extension of the file being loaded\n * @returns true if the loader can load the specified file\n */\n canLoad(extension) {\n return extension.endsWith(\".hdr\");\n }\n /**\n * Uploads the cube texture data to the WebGL texture. It has already been bound.\n */\n loadCubeData() {\n throw \".env not supported in Cube.\";\n }\n /**\n * Uploads the 2D texture data to the WebGL texture. It has already been bound once in the callback.\n * @param data contains the texture data\n * @param texture defines the BabylonJS internal texture\n * @param callback defines the method to call once ready to upload\n */\n loadData(data, texture, callback) {\n const uint8array = new Uint8Array(data.buffer, data.byteOffset, data.byteLength);\n const hdrInfo = HDRTools.RGBE_ReadHeader(uint8array);\n const pixelsDataRGB32 = HDRTools.RGBE_ReadPixels(uint8array, hdrInfo);\n const pixels = hdrInfo.width * hdrInfo.height;\n const pixelsDataRGBA32 = new Float32Array(pixels * 4);\n for (let i = 0; i < pixels; i += 1) {\n pixelsDataRGBA32[i * 4] = pixelsDataRGB32[i * 3];\n pixelsDataRGBA32[i * 4 + 1] = pixelsDataRGB32[i * 3 + 1];\n pixelsDataRGBA32[i * 4 + 2] = pixelsDataRGB32[i * 3 + 2];\n pixelsDataRGBA32[i * 4 + 3] = 1;\n }\n callback(hdrInfo.width, hdrInfo.height, texture.generateMipMaps, false, () => {\n const engine = texture.getEngine();\n texture.type = 1;\n texture.format = 5;\n texture._gammaSpace = false;\n engine._uploadDataToTextureDirectly(texture, pixelsDataRGBA32);\n });\n }\n }\n // Register the loader.\n Engine._TextureLoaders.push(new _HDRTextureLoader());\n\n /**\n * @internal\n * Enum of basis transcoder formats\n */\n var BASIS_FORMATS;\n (function (BASIS_FORMATS) {\n BASIS_FORMATS[BASIS_FORMATS[\"cTFETC1\"] = 0] = \"cTFETC1\";\n BASIS_FORMATS[BASIS_FORMATS[\"cTFETC2\"] = 1] = \"cTFETC2\";\n BASIS_FORMATS[BASIS_FORMATS[\"cTFBC1\"] = 2] = \"cTFBC1\";\n BASIS_FORMATS[BASIS_FORMATS[\"cTFBC3\"] = 3] = \"cTFBC3\";\n BASIS_FORMATS[BASIS_FORMATS[\"cTFBC4\"] = 4] = \"cTFBC4\";\n BASIS_FORMATS[BASIS_FORMATS[\"cTFBC5\"] = 5] = \"cTFBC5\";\n BASIS_FORMATS[BASIS_FORMATS[\"cTFBC7\"] = 6] = \"cTFBC7\";\n BASIS_FORMATS[BASIS_FORMATS[\"cTFPVRTC1_4_RGB\"] = 8] = \"cTFPVRTC1_4_RGB\";\n BASIS_FORMATS[BASIS_FORMATS[\"cTFPVRTC1_4_RGBA\"] = 9] = \"cTFPVRTC1_4_RGBA\";\n BASIS_FORMATS[BASIS_FORMATS[\"cTFASTC_4x4\"] = 10] = \"cTFASTC_4x4\";\n BASIS_FORMATS[BASIS_FORMATS[\"cTFATC_RGB\"] = 11] = \"cTFATC_RGB\";\n BASIS_FORMATS[BASIS_FORMATS[\"cTFATC_RGBA_INTERPOLATED_ALPHA\"] = 12] = \"cTFATC_RGBA_INTERPOLATED_ALPHA\";\n BASIS_FORMATS[BASIS_FORMATS[\"cTFRGBA32\"] = 13] = \"cTFRGBA32\";\n BASIS_FORMATS[BASIS_FORMATS[\"cTFRGB565\"] = 14] = \"cTFRGB565\";\n BASIS_FORMATS[BASIS_FORMATS[\"cTFBGR565\"] = 15] = \"cTFBGR565\";\n BASIS_FORMATS[BASIS_FORMATS[\"cTFRGBA4444\"] = 16] = \"cTFRGBA4444\";\n BASIS_FORMATS[BASIS_FORMATS[\"cTFFXT1_RGB\"] = 17] = \"cTFFXT1_RGB\";\n BASIS_FORMATS[BASIS_FORMATS[\"cTFPVRTC2_4_RGB\"] = 18] = \"cTFPVRTC2_4_RGB\";\n BASIS_FORMATS[BASIS_FORMATS[\"cTFPVRTC2_4_RGBA\"] = 19] = \"cTFPVRTC2_4_RGBA\";\n BASIS_FORMATS[BASIS_FORMATS[\"cTFETC2_EAC_R11\"] = 20] = \"cTFETC2_EAC_R11\";\n BASIS_FORMATS[BASIS_FORMATS[\"cTFETC2_EAC_RG11\"] = 21] = \"cTFETC2_EAC_RG11\";\n })(BASIS_FORMATS || (BASIS_FORMATS = {}));\n /**\n * Used to load .Basis files\n * See https://github.com/BinomialLLC/basis_universal/tree/master/webgl\n */\n const BasisToolsOptions = {\n /**\n * URL to use when loading the basis transcoder\n */\n JSModuleURL: \"https://cdn.babylonjs.com/basisTranscoder/1/basis_transcoder.js\",\n /**\n * URL to use when loading the wasm module for the transcoder\n */\n WasmModuleURL: \"https://cdn.babylonjs.com/basisTranscoder/1/basis_transcoder.wasm\",\n };\n /**\n * Get the internal format to be passed to texImage2D corresponding to the .basis format value\n * @param basisFormat format chosen from GetSupportedTranscodeFormat\n * @param engine\n * @returns internal format corresponding to the Basis format\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n const GetInternalFormatFromBasisFormat = (basisFormat, engine) => {\n let format;\n switch (basisFormat) {\n case BASIS_FORMATS.cTFETC1:\n format = 36196;\n break;\n case BASIS_FORMATS.cTFBC1:\n format = 33776;\n break;\n case BASIS_FORMATS.cTFBC4:\n format = 33779;\n break;\n case BASIS_FORMATS.cTFASTC_4x4:\n format = 37808;\n break;\n case BASIS_FORMATS.cTFETC2:\n format = 37496;\n break;\n case BASIS_FORMATS.cTFBC7:\n format = 36492;\n break;\n }\n if (format === undefined) {\n throw \"The chosen Basis transcoder format is not currently supported\";\n }\n return format;\n };\n let _WorkerPromise = null;\n let _Worker = null;\n let _actionId = 0;\n const _IgnoreSupportedFormats = false;\n const _CreateWorkerAsync = () => {\n if (!_WorkerPromise) {\n _WorkerPromise = new Promise((res, reject) => {\n if (_Worker) {\n res(_Worker);\n }\n else {\n Tools.LoadFileAsync(BasisToolsOptions.WasmModuleURL)\n .then((wasmBinary) => {\n if (typeof URL !== \"function\") {\n return reject(\"Basis transcoder requires an environment with a URL constructor\");\n }\n const workerBlobUrl = URL.createObjectURL(new Blob([`(${workerFunc$1})()`], { type: \"application/javascript\" }));\n _Worker = new Worker(workerBlobUrl);\n const initHandler = (msg) => {\n if (msg.data.action === \"init\") {\n _Worker.removeEventListener(\"message\", initHandler);\n res(_Worker);\n }\n else if (msg.data.action === \"error\") {\n reject(msg.data.error || \"error initializing worker\");\n }\n };\n _Worker.addEventListener(\"message\", initHandler);\n _Worker.postMessage({ action: \"init\", url: BasisToolsOptions.JSModuleURL, wasmBinary: wasmBinary });\n })\n .catch(reject);\n }\n });\n }\n return _WorkerPromise;\n };\n /**\n * Transcodes a loaded image file to compressed pixel data\n * @param data image data to transcode\n * @param config configuration options for the transcoding\n * @returns a promise resulting in the transcoded image\n */\n const TranscodeAsync = (data, config) => {\n const dataView = data instanceof ArrayBuffer ? new Uint8Array(data) : data;\n return new Promise((res, rej) => {\n _CreateWorkerAsync().then(() => {\n const actionId = _actionId++;\n const messageHandler = (msg) => {\n if (msg.data.action === \"transcode\" && msg.data.id === actionId) {\n _Worker.removeEventListener(\"message\", messageHandler);\n if (!msg.data.success) {\n rej(\"Transcode is not supported on this device\");\n }\n else {\n res(msg.data);\n }\n }\n };\n _Worker.addEventListener(\"message\", messageHandler);\n const dataViewCopy = new Uint8Array(dataView.byteLength);\n dataViewCopy.set(new Uint8Array(dataView.buffer, dataView.byteOffset, dataView.byteLength));\n _Worker.postMessage({ action: \"transcode\", id: actionId, imageData: dataViewCopy, config: config, ignoreSupportedFormats: _IgnoreSupportedFormats }, [\n dataViewCopy.buffer,\n ]);\n }, (error) => {\n rej(error);\n });\n });\n };\n /**\n * Binds a texture according to its underlying target.\n * @param texture texture to bind\n * @param engine the engine to bind the texture in\n */\n const BindTexture = (texture, engine) => {\n var _a, _b;\n let target = (_a = engine._gl) === null || _a === void 0 ? void 0 : _a.TEXTURE_2D;\n if (texture.isCube) {\n target = (_b = engine._gl) === null || _b === void 0 ? void 0 : _b.TEXTURE_CUBE_MAP;\n }\n engine._bindTextureDirectly(target, texture, true);\n };\n /**\n * Loads a texture from the transcode result\n * @param texture texture load to\n * @param transcodeResult the result of transcoding the basis file to load from\n */\n const LoadTextureFromTranscodeResult = (texture, transcodeResult) => {\n const engine = texture.getEngine();\n for (let i = 0; i < transcodeResult.fileInfo.images.length; i++) {\n const rootImage = transcodeResult.fileInfo.images[i].levels[0];\n texture._invertVScale = texture.invertY;\n if (transcodeResult.format === -1 || transcodeResult.format === BASIS_FORMATS.cTFRGB565) {\n // No compatable compressed format found, fallback to RGB\n texture.type = 10;\n texture.format = 4;\n if (engine._features.basisNeedsPOT && (Scalar.Log2(rootImage.width) % 1 !== 0 || Scalar.Log2(rootImage.height) % 1 !== 0)) {\n // Create non power of two texture\n const source = new InternalTexture(engine, InternalTextureSource.Temp);\n texture._invertVScale = texture.invertY;\n source.type = 10;\n source.format = 4;\n // Fallback requires aligned width/height\n source.width = (rootImage.width + 3) & ~3;\n source.height = (rootImage.height + 3) & ~3;\n BindTexture(source, engine);\n engine._uploadDataToTextureDirectly(source, new Uint16Array(rootImage.transcodedPixels.buffer), i, 0, 4, true);\n // Resize to power of two\n engine._rescaleTexture(source, texture, engine.scenes[0], engine._getInternalFormat(4), () => {\n engine._releaseTexture(source);\n BindTexture(texture, engine);\n });\n }\n else {\n // Fallback is already inverted\n texture._invertVScale = !texture.invertY;\n // Upload directly\n texture.width = (rootImage.width + 3) & ~3;\n texture.height = (rootImage.height + 3) & ~3;\n texture.samplingMode = 2;\n BindTexture(texture, engine);\n engine._uploadDataToTextureDirectly(texture, new Uint16Array(rootImage.transcodedPixels.buffer), i, 0, 4, true);\n }\n }\n else {\n texture.width = rootImage.width;\n texture.height = rootImage.height;\n texture.generateMipMaps = transcodeResult.fileInfo.images[i].levels.length > 1;\n const format = BasisTools.GetInternalFormatFromBasisFormat(transcodeResult.format, engine);\n texture.format = format;\n BindTexture(texture, engine);\n // Upload all mip levels in the file\n transcodeResult.fileInfo.images[i].levels.forEach((level, index) => {\n engine._uploadCompressedDataToTextureDirectly(texture, format, level.width, level.height, level.transcodedPixels, i, index);\n });\n if (engine._features.basisNeedsPOT && (Scalar.Log2(texture.width) % 1 !== 0 || Scalar.Log2(texture.height) % 1 !== 0)) {\n Tools.Warn(\"Loaded .basis texture width and height are not a power of two. Texture wrapping will be set to Texture.CLAMP_ADDRESSMODE as other modes are not supported with non power of two dimensions in webGL 1.\");\n texture._cachedWrapU = Texture.CLAMP_ADDRESSMODE;\n texture._cachedWrapV = Texture.CLAMP_ADDRESSMODE;\n }\n }\n }\n };\n /**\n * Used to load .Basis files\n * See https://github.com/BinomialLLC/basis_universal/tree/master/webgl\n */\n const BasisTools = {\n /**\n * URL to use when loading the basis transcoder\n */\n JSModuleURL: BasisToolsOptions.JSModuleURL,\n /**\n * URL to use when loading the wasm module for the transcoder\n */\n WasmModuleURL: BasisToolsOptions.WasmModuleURL,\n /**\n * Get the internal format to be passed to texImage2D corresponding to the .basis format value\n * @param basisFormat format chosen from GetSupportedTranscodeFormat\n * @returns internal format corresponding to the Basis format\n */\n GetInternalFormatFromBasisFormat,\n /**\n * Transcodes a loaded image file to compressed pixel data\n * @param data image data to transcode\n * @param config configuration options for the transcoding\n * @returns a promise resulting in the transcoded image\n */\n TranscodeAsync,\n /**\n * Loads a texture from the transcode result\n * @param texture texture load to\n * @param transcodeResult the result of transcoding the basis file to load from\n */\n LoadTextureFromTranscodeResult,\n };\n function workerFunc$1() {\n const _BASIS_FORMAT = {\n cTFETC1: 0,\n cTFETC2: 1,\n cTFBC1: 2,\n cTFBC3: 3,\n cTFBC4: 4,\n cTFBC5: 5,\n cTFBC7: 6,\n cTFPVRTC1_4_RGB: 8,\n cTFPVRTC1_4_RGBA: 9,\n cTFASTC_4x4: 10,\n cTFATC_RGB: 11,\n cTFATC_RGBA_INTERPOLATED_ALPHA: 12,\n cTFRGBA32: 13,\n cTFRGB565: 14,\n cTFBGR565: 15,\n cTFRGBA4444: 16,\n cTFFXT1_RGB: 17,\n cTFPVRTC2_4_RGB: 18,\n cTFPVRTC2_4_RGBA: 19,\n cTFETC2_EAC_R11: 20,\n cTFETC2_EAC_RG11: 21,\n };\n let transcoderModulePromise = null;\n onmessage = (event) => {\n if (event.data.action === \"init\") {\n // Load the transcoder if it hasn't been yet\n if (!transcoderModulePromise) {\n // make sure we loaded the script correctly\n try {\n importScripts(event.data.url);\n }\n catch (e) {\n postMessage({ action: \"error\", error: e });\n }\n transcoderModulePromise = BASIS({\n // Override wasm binary\n wasmBinary: event.data.wasmBinary,\n });\n }\n if (transcoderModulePromise !== null) {\n transcoderModulePromise.then((m) => {\n BASIS = m;\n m.initializeBasis();\n postMessage({ action: \"init\" });\n });\n }\n }\n else if (event.data.action === \"transcode\") {\n // Transcode the basis image and return the resulting pixels\n const config = event.data.config;\n const imgData = event.data.imageData;\n const loadedFile = new BASIS.BasisFile(imgData);\n const fileInfo = GetFileInfo(loadedFile);\n let format = event.data.ignoreSupportedFormats ? null : GetSupportedTranscodeFormat(event.data.config, fileInfo);\n let needsConversion = false;\n if (format === null) {\n needsConversion = true;\n format = fileInfo.hasAlpha ? _BASIS_FORMAT.cTFBC3 : _BASIS_FORMAT.cTFBC1;\n }\n // Begin transcode\n let success = true;\n if (!loadedFile.startTranscoding()) {\n success = false;\n }\n const buffers = [];\n for (let imageIndex = 0; imageIndex < fileInfo.images.length; imageIndex++) {\n if (!success) {\n break;\n }\n const image = fileInfo.images[imageIndex];\n if (config.loadSingleImage === undefined || config.loadSingleImage === imageIndex) {\n let mipCount = image.levels.length;\n if (config.loadMipmapLevels === false) {\n mipCount = 1;\n }\n for (let levelIndex = 0; levelIndex < mipCount; levelIndex++) {\n const levelInfo = image.levels[levelIndex];\n const pixels = TranscodeLevel(loadedFile, imageIndex, levelIndex, format, needsConversion);\n if (!pixels) {\n success = false;\n break;\n }\n levelInfo.transcodedPixels = pixels;\n buffers.push(levelInfo.transcodedPixels.buffer);\n }\n }\n }\n // Close file\n loadedFile.close();\n loadedFile.delete();\n if (needsConversion) {\n format = -1;\n }\n if (!success) {\n postMessage({ action: \"transcode\", success: success, id: event.data.id });\n }\n else {\n postMessage({ action: \"transcode\", success: success, id: event.data.id, fileInfo: fileInfo, format: format }, buffers);\n }\n }\n };\n /**\n * Detects the supported transcode format for the file\n * @param config transcode config\n * @param fileInfo info about the file\n * @returns the chosed format or null if none are supported\n */\n function GetSupportedTranscodeFormat(config, fileInfo) {\n let format = null;\n if (config.supportedCompressionFormats) {\n if (config.supportedCompressionFormats.astc) {\n format = _BASIS_FORMAT.cTFASTC_4x4;\n }\n else if (config.supportedCompressionFormats.bc7) {\n format = _BASIS_FORMAT.cTFBC7;\n }\n else if (config.supportedCompressionFormats.s3tc) {\n format = fileInfo.hasAlpha ? _BASIS_FORMAT.cTFBC3 : _BASIS_FORMAT.cTFBC1;\n }\n else if (config.supportedCompressionFormats.pvrtc) {\n format = fileInfo.hasAlpha ? _BASIS_FORMAT.cTFPVRTC1_4_RGBA : _BASIS_FORMAT.cTFPVRTC1_4_RGB;\n }\n else if (config.supportedCompressionFormats.etc2) {\n format = _BASIS_FORMAT.cTFETC2;\n }\n else if (config.supportedCompressionFormats.etc1) {\n format = _BASIS_FORMAT.cTFETC1;\n }\n else {\n format = _BASIS_FORMAT.cTFRGB565;\n }\n }\n return format;\n }\n /**\n * Retrieves information about the basis file eg. dimensions\n * @param basisFile the basis file to get the info from\n * @returns information about the basis file\n */\n function GetFileInfo(basisFile) {\n const hasAlpha = basisFile.getHasAlpha();\n const imageCount = basisFile.getNumImages();\n const images = [];\n for (let i = 0; i < imageCount; i++) {\n const imageInfo = {\n levels: [],\n };\n const levelCount = basisFile.getNumLevels(i);\n for (let level = 0; level < levelCount; level++) {\n const levelInfo = {\n width: basisFile.getImageWidth(i, level),\n height: basisFile.getImageHeight(i, level),\n };\n imageInfo.levels.push(levelInfo);\n }\n images.push(imageInfo);\n }\n const info = { hasAlpha, images };\n return info;\n }\n function TranscodeLevel(loadedFile, imageIndex, levelIndex, format, convertToRgb565) {\n const dstSize = loadedFile.getImageTranscodedSizeInBytes(imageIndex, levelIndex, format);\n let dst = new Uint8Array(dstSize);\n if (!loadedFile.transcodeImage(dst, imageIndex, levelIndex, format, 1, 0)) {\n return null;\n }\n // If no supported format is found, load as dxt and convert to rgb565\n if (convertToRgb565) {\n const alignedWidth = (loadedFile.getImageWidth(imageIndex, levelIndex) + 3) & ~3;\n const alignedHeight = (loadedFile.getImageHeight(imageIndex, levelIndex) + 3) & ~3;\n dst = ConvertDxtToRgb565(dst, 0, alignedWidth, alignedHeight);\n }\n return dst;\n }\n /**\n * From https://github.com/BinomialLLC/basis_universal/blob/master/webgl/texture/dxt-to-rgb565.js\n * An unoptimized version of dxtToRgb565. Also, the floating\n * point math used to compute the colors actually results in\n * slightly different colors compared to hardware DXT decoders.\n * @param src dxt src pixels\n * @param srcByteOffset offset for the start of src\n * @param width aligned width of the image\n * @param height aligned height of the image\n * @returns the converted pixels\n */\n function ConvertDxtToRgb565(src, srcByteOffset, width, height) {\n const c = new Uint16Array(4);\n const dst = new Uint16Array(width * height);\n const blockWidth = width / 4;\n const blockHeight = height / 4;\n for (let blockY = 0; blockY < blockHeight; blockY++) {\n for (let blockX = 0; blockX < blockWidth; blockX++) {\n const i = srcByteOffset + 8 * (blockY * blockWidth + blockX);\n c[0] = src[i] | (src[i + 1] << 8);\n c[1] = src[i + 2] | (src[i + 3] << 8);\n c[2] =\n ((2 * (c[0] & 0x1f) + 1 * (c[1] & 0x1f)) / 3) |\n (((2 * (c[0] & 0x7e0) + 1 * (c[1] & 0x7e0)) / 3) & 0x7e0) |\n (((2 * (c[0] & 0xf800) + 1 * (c[1] & 0xf800)) / 3) & 0xf800);\n c[3] =\n ((2 * (c[1] & 0x1f) + 1 * (c[0] & 0x1f)) / 3) |\n (((2 * (c[1] & 0x7e0) + 1 * (c[0] & 0x7e0)) / 3) & 0x7e0) |\n (((2 * (c[1] & 0xf800) + 1 * (c[0] & 0xf800)) / 3) & 0xf800);\n for (let row = 0; row < 4; row++) {\n const m = src[i + 4 + row];\n let dstI = (blockY * 4 + row) * width + blockX * 4;\n dst[dstI++] = c[m & 0x3];\n dst[dstI++] = c[(m >> 2) & 0x3];\n dst[dstI++] = c[(m >> 4) & 0x3];\n dst[dstI++] = c[(m >> 6) & 0x3];\n }\n }\n }\n return dst;\n }\n }\n Object.defineProperty(BasisTools, \"JSModuleURL\", {\n get: function () {\n return BasisToolsOptions.JSModuleURL;\n },\n set: function (value) {\n BasisToolsOptions.JSModuleURL = value;\n },\n });\n Object.defineProperty(BasisTools, \"WasmModuleURL\", {\n get: function () {\n return BasisToolsOptions.WasmModuleURL;\n },\n set: function (value) {\n BasisToolsOptions.WasmModuleURL = value;\n },\n });\n\n /**\n * Loader for .basis file format\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n class _BasisTextureLoader {\n constructor() {\n /**\n * Defines whether the loader supports cascade loading the different faces.\n */\n this.supportCascades = false;\n }\n /**\n * This returns if the loader support the current file information.\n * @param extension defines the file extension of the file being loaded\n * @returns true if the loader can load the specified file\n */\n canLoad(extension) {\n return extension.endsWith(\".basis\");\n }\n /**\n * Uploads the cube texture data to the WebGL texture. It has already been bound.\n * @param data contains the texture data\n * @param texture defines the BabylonJS internal texture\n * @param createPolynomials will be true if polynomials have been requested\n * @param onLoad defines the callback to trigger once the texture is ready\n * @param onError defines the callback to trigger in case of error\n */\n loadCubeData(data, texture, createPolynomials, onLoad, onError) {\n if (Array.isArray(data)) {\n return;\n }\n const caps = texture.getEngine().getCaps();\n const transcodeConfig = {\n supportedCompressionFormats: {\n etc1: caps.etc1 ? true : false,\n s3tc: caps.s3tc ? true : false,\n pvrtc: caps.pvrtc ? true : false,\n etc2: caps.etc2 ? true : false,\n astc: caps.astc ? true : false,\n bc7: caps.bptc ? true : false,\n },\n };\n TranscodeAsync(data, transcodeConfig)\n .then((result) => {\n const hasMipmap = result.fileInfo.images[0].levels.length > 1 && texture.generateMipMaps;\n LoadTextureFromTranscodeResult(texture, result);\n texture.getEngine()._setCubeMapTextureParams(texture, hasMipmap);\n texture.isReady = true;\n texture.onLoadedObservable.notifyObservers(texture);\n texture.onLoadedObservable.clear();\n if (onLoad) {\n onLoad();\n }\n })\n .catch((err) => {\n const errorMessage = \"Failed to transcode Basis file, transcoding may not be supported on this device\";\n Tools.Warn(errorMessage);\n texture.isReady = true;\n if (onError) {\n onError(err);\n }\n });\n }\n /**\n * Uploads the 2D texture data to the WebGL texture. It has already been bound once in the callback.\n * @param data contains the texture data\n * @param texture defines the BabylonJS internal texture\n * @param callback defines the method to call once ready to upload\n */\n loadData(data, texture, callback) {\n const caps = texture.getEngine().getCaps();\n const transcodeConfig = {\n supportedCompressionFormats: {\n etc1: caps.etc1 ? true : false,\n s3tc: caps.s3tc ? true : false,\n pvrtc: caps.pvrtc ? true : false,\n etc2: caps.etc2 ? true : false,\n astc: caps.astc ? true : false,\n bc7: caps.bptc ? true : false,\n },\n };\n TranscodeAsync(data, transcodeConfig)\n .then((result) => {\n const rootImage = result.fileInfo.images[0].levels[0];\n const hasMipmap = result.fileInfo.images[0].levels.length > 1 && texture.generateMipMaps;\n callback(rootImage.width, rootImage.height, hasMipmap, result.format !== -1, () => {\n LoadTextureFromTranscodeResult(texture, result);\n });\n })\n .catch((err) => {\n Tools.Warn(\"Failed to transcode Basis file, transcoding may not be supported on this device\");\n Tools.Warn(`Failed to transcode Basis file: ${err}`);\n callback(0, 0, false, false, () => { }, true);\n });\n }\n }\n // Register the loader.\n Engine._TextureLoaders.push(new _BasisTextureLoader());\n\n // Do not edit.\n const name$2s = \"noisePixelShader\";\n const shader$2s = `uniform float brightness;uniform float persistence;uniform float timeScale;varying vec2 vUV;vec2 hash22(vec2 p)\n{p=p*mat2(127.1,311.7,269.5,183.3);p=-1.0+2.0*fract(sin(p)*43758.5453123);return sin(p*6.283+timeScale);}\nfloat interpolationNoise(vec2 p)\n{vec2 pi=floor(p);vec2 pf=p-pi;vec2 w=pf*pf*(3.-2.*pf);float f00=dot(hash22(pi+vec2(.0,.0)),pf-vec2(.0,.0));float f01=dot(hash22(pi+vec2(.0,1.)),pf-vec2(.0,1.));float f10=dot(hash22(pi+vec2(1.0,0.)),pf-vec2(1.0,0.));float f11=dot(hash22(pi+vec2(1.0,1.)),pf-vec2(1.0,1.));float xm1=mix(f00,f10,w.x);float xm2=mix(f01,f11,w.x);float ym=mix(xm1,xm2,w.y); \nreturn ym;}\nfloat perlinNoise2D(float x,float y)\n{float sum=0.0;float frequency=0.0;float amplitude=0.0;for(int i=0; i {\n this.getScene().clipPlane = this.refractionPlane;\n });\n this.onAfterRenderObservable.add(() => {\n this.getScene().clipPlane = null;\n });\n }\n /**\n * Clone the refraction texture.\n * @returns the cloned texture\n */\n clone() {\n const scene = this.getScene();\n if (!scene) {\n return this;\n }\n const textureSize = this.getSize();\n const newTexture = new RefractionTexture(this.name, textureSize.width, scene, this._generateMipMaps);\n // Base texture\n newTexture.hasAlpha = this.hasAlpha;\n newTexture.level = this.level;\n // Refraction Texture\n newTexture.refractionPlane = this.refractionPlane.clone();\n if (this.renderList) {\n newTexture.renderList = this.renderList.slice(0);\n }\n newTexture.depth = this.depth;\n return newTexture;\n }\n /**\n * Serialize the texture to a JSON representation you could use in Parse later on\n * @returns the serialized JSON representation\n */\n serialize() {\n if (!this.name) {\n return null;\n }\n const serializationObject = super.serialize();\n serializationObject.mirrorPlane = this.refractionPlane.asArray();\n serializationObject.depth = this.depth;\n return serializationObject;\n }\n }\n\n // DEFLATE is a complex format; to read this code, you should probably check the RFC first:\n // https://tools.ietf.org/html/rfc1951\n // You may also wish to take a look at the guide I made about this program:\n // https://gist.github.com/101arrowz/253f31eb5abc3d9275ab943003ffecad\n // Some of the following code is similar to that of UZIP.js:\n // https://github.com/photopea/UZIP.js\n // However, the vast majority of the codebase has diverged from UZIP.js to increase performance and reduce bundle size.\n // Sometimes 0 will appear where -1 would be more appropriate. This is because using a uint\n // is better for memory in most engines (I *think*).\n var ch2 = {};\n var wk = (function (c, id, msg, transfer, cb) {\n var w = new Worker(ch2[id] || (ch2[id] = URL.createObjectURL(new Blob([\n c + ';addEventListener(\"error\",function(e){e=e.error;postMessage({$e$:[e.message,e.code,e.stack]})})'\n ], { type: 'text/javascript' }))));\n w.onmessage = function (e) {\n var d = e.data, ed = d.$e$;\n if (ed) {\n var err = new Error(ed[0]);\n err['code'] = ed[1];\n err.stack = ed[2];\n cb(err, null);\n }\n else\n cb(null, d);\n };\n w.postMessage(msg, transfer);\n return w;\n });\n\n // aliases for shorter compressed code (most minifers don't do this)\n var u8 = Uint8Array, u16 = Uint16Array, u32 = Uint32Array;\n // fixed length extra bits\n var fleb = new u8([0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 2, 2, 2, 2, 3, 3, 3, 3, 4, 4, 4, 4, 5, 5, 5, 5, 0, /* unused */ 0, 0, /* impossible */ 0]);\n // fixed distance extra bits\n // see fleb note\n var fdeb = new u8([0, 0, 0, 0, 1, 1, 2, 2, 3, 3, 4, 4, 5, 5, 6, 6, 7, 7, 8, 8, 9, 9, 10, 10, 11, 11, 12, 12, 13, 13, /* unused */ 0, 0]);\n // code length index map\n var clim = new u8([16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15]);\n // get base, reverse index map from extra bits\n var freb = function (eb, start) {\n var b = new u16(31);\n for (var i = 0; i < 31; ++i) {\n b[i] = start += 1 << eb[i - 1];\n }\n // numbers here are at max 18 bits\n var r = new u32(b[30]);\n for (var i = 1; i < 30; ++i) {\n for (var j = b[i]; j < b[i + 1]; ++j) {\n r[j] = ((j - b[i]) << 5) | i;\n }\n }\n return [b, r];\n };\n var _a = freb(fleb, 2), fl = _a[0], revfl = _a[1];\n // we can ignore the fact that the other numbers are wrong; they never happen anyway\n fl[28] = 258, revfl[258] = 28;\n var _b = freb(fdeb, 0), revfd = _b[1];\n // map of value to reverse (assuming 16 bits)\n var rev = new u16(32768);\n for (var i = 0; i < 32768; ++i) {\n // reverse table algorithm from SO\n var x = ((i & 0xAAAA) >>> 1) | ((i & 0x5555) << 1);\n x = ((x & 0xCCCC) >>> 2) | ((x & 0x3333) << 2);\n x = ((x & 0xF0F0) >>> 4) | ((x & 0x0F0F) << 4);\n rev[i] = (((x & 0xFF00) >>> 8) | ((x & 0x00FF) << 8)) >>> 1;\n }\n // create huffman tree from u8 \"map\": index -> code length for code index\n // mb (max bits) must be at most 15\n // TODO: optimize/split up?\n var hMap = (function (cd, mb, r) {\n var s = cd.length;\n // index\n var i = 0;\n // u16 \"map\": index -> # of codes with bit length = index\n var l = new u16(mb);\n // length of cd must be 288 (total # of codes)\n for (; i < s; ++i) {\n if (cd[i])\n ++l[cd[i] - 1];\n }\n // u16 \"map\": index -> minimum code for bit length = index\n var le = new u16(mb);\n for (i = 0; i < mb; ++i) {\n le[i] = (le[i - 1] + l[i - 1]) << 1;\n }\n var co;\n if (r) {\n // u16 \"map\": index -> number of actual bits, symbol for code\n co = new u16(1 << mb);\n // bits to remove for reverser\n var rvb = 15 - mb;\n for (i = 0; i < s; ++i) {\n // ignore 0 lengths\n if (cd[i]) {\n // num encoding both symbol and bits read\n var sv = (i << 4) | cd[i];\n // free bits\n var r_1 = mb - cd[i];\n // start value\n var v = le[cd[i] - 1]++ << r_1;\n // m is end value\n for (var m = v | ((1 << r_1) - 1); v <= m; ++v) {\n // every 16 bit value starting with the code yields the same result\n co[rev[v] >>> rvb] = sv;\n }\n }\n }\n }\n else {\n co = new u16(s);\n for (i = 0; i < s; ++i) {\n if (cd[i]) {\n co[i] = rev[le[cd[i] - 1]++] >>> (15 - cd[i]);\n }\n }\n }\n return co;\n });\n // fixed length tree\n var flt = new u8(288);\n for (var i = 0; i < 144; ++i)\n flt[i] = 8;\n for (var i = 144; i < 256; ++i)\n flt[i] = 9;\n for (var i = 256; i < 280; ++i)\n flt[i] = 7;\n for (var i = 280; i < 288; ++i)\n flt[i] = 8;\n // fixed distance tree\n var fdt = new u8(32);\n for (var i = 0; i < 32; ++i)\n fdt[i] = 5;\n // fixed length map\n var flm = /*#__PURE__*/ hMap(flt, 9, 0);\n // fixed distance map\n var fdm = /*#__PURE__*/ hMap(fdt, 5, 0);\n // get end of byte\n var shft = function (p) { return ((p + 7) / 8) | 0; };\n // typed array slice - allows garbage collector to free original reference,\n // while being more compatible than .slice\n var slc = function (v, s, e) {\n if (s == null || s < 0)\n s = 0;\n if (e == null || e > v.length)\n e = v.length;\n // can't use .constructor in case user-supplied\n var n = new (v instanceof u16 ? u16 : v instanceof u32 ? u32 : u8)(e - s);\n n.set(v.subarray(s, e));\n return n;\n };\n // error codes\n var ec = [\n 'unexpected EOF',\n 'invalid block type',\n 'invalid length/literal',\n 'invalid distance',\n 'stream finished',\n 'no stream handler',\n ,\n 'no callback',\n 'invalid UTF-8 data',\n 'extra field too long',\n 'date not in range 1980-2099',\n 'filename too long',\n 'stream finishing',\n 'invalid zip data'\n // determined by unknown compression method\n ];\n var err = function (ind, msg, nt) {\n var e = new Error(msg || ec[ind]);\n e.code = ind;\n if (Error.captureStackTrace)\n Error.captureStackTrace(e, err);\n if (!nt)\n throw e;\n return e;\n };\n // starting at p, write the minimum number of bits that can hold v to d\n var wbits = function (d, p, v) {\n v <<= p & 7;\n var o = (p / 8) | 0;\n d[o] |= v;\n d[o + 1] |= v >>> 8;\n };\n // starting at p, write the minimum number of bits (>8) that can hold v to d\n var wbits16 = function (d, p, v) {\n v <<= p & 7;\n var o = (p / 8) | 0;\n d[o] |= v;\n d[o + 1] |= v >>> 8;\n d[o + 2] |= v >>> 16;\n };\n // creates code lengths from a frequency table\n var hTree = function (d, mb) {\n // Need extra info to make a tree\n var t = [];\n for (var i = 0; i < d.length; ++i) {\n if (d[i])\n t.push({ s: i, f: d[i] });\n }\n var s = t.length;\n var t2 = t.slice();\n if (!s)\n return [et, 0];\n if (s == 1) {\n var v = new u8(t[0].s + 1);\n v[t[0].s] = 1;\n return [v, 1];\n }\n t.sort(function (a, b) { return a.f - b.f; });\n // after i2 reaches last ind, will be stopped\n // freq must be greater than largest possible number of symbols\n t.push({ s: -1, f: 25001 });\n var l = t[0], r = t[1], i0 = 0, i1 = 1, i2 = 2;\n t[0] = { s: -1, f: l.f + r.f, l: l, r: r };\n // efficient algorithm from UZIP.js\n // i0 is lookbehind, i2 is lookahead - after processing two low-freq\n // symbols that combined have high freq, will start processing i2 (high-freq,\n // non-composite) symbols instead\n // see https://reddit.com/r/photopea/comments/ikekht/uzipjs_questions/\n while (i1 != s - 1) {\n l = t[t[i0].f < t[i2].f ? i0++ : i2++];\n r = t[i0 != i1 && t[i0].f < t[i2].f ? i0++ : i2++];\n t[i1++] = { s: -1, f: l.f + r.f, l: l, r: r };\n }\n var maxSym = t2[0].s;\n for (var i = 1; i < s; ++i) {\n if (t2[i].s > maxSym)\n maxSym = t2[i].s;\n }\n // code lengths\n var tr = new u16(maxSym + 1);\n // max bits in tree\n var mbt = ln(t[i1 - 1], tr, 0);\n if (mbt > mb) {\n // more algorithms from UZIP.js\n // TODO: find out how this code works (debt)\n // ind debt\n var i = 0, dt = 0;\n // left cost\n var lft = mbt - mb, cst = 1 << lft;\n t2.sort(function (a, b) { return tr[b.s] - tr[a.s] || a.f - b.f; });\n for (; i < s; ++i) {\n var i2_1 = t2[i].s;\n if (tr[i2_1] > mb) {\n dt += cst - (1 << (mbt - tr[i2_1]));\n tr[i2_1] = mb;\n }\n else\n break;\n }\n dt >>>= lft;\n while (dt > 0) {\n var i2_2 = t2[i].s;\n if (tr[i2_2] < mb)\n dt -= 1 << (mb - tr[i2_2]++ - 1);\n else\n ++i;\n }\n for (; i >= 0 && dt; --i) {\n var i2_3 = t2[i].s;\n if (tr[i2_3] == mb) {\n --tr[i2_3];\n ++dt;\n }\n }\n mbt = mb;\n }\n return [new u8(tr), mbt];\n };\n // get the max length and assign length codes\n var ln = function (n, l, d) {\n return n.s == -1\n ? Math.max(ln(n.l, l, d + 1), ln(n.r, l, d + 1))\n : (l[n.s] = d);\n };\n // length codes generation\n var lc = function (c) {\n var s = c.length;\n // Note that the semicolon was intentional\n while (s && !c[--s])\n ;\n var cl = new u16(++s);\n // ind num streak\n var cli = 0, cln = c[0], cls = 1;\n var w = function (v) { cl[cli++] = v; };\n for (var i = 1; i <= s; ++i) {\n if (c[i] == cln && i != s)\n ++cls;\n else {\n if (!cln && cls > 2) {\n for (; cls > 138; cls -= 138)\n w(32754);\n if (cls > 2) {\n w(cls > 10 ? ((cls - 11) << 5) | 28690 : ((cls - 3) << 5) | 12305);\n cls = 0;\n }\n }\n else if (cls > 3) {\n w(cln), --cls;\n for (; cls > 6; cls -= 6)\n w(8304);\n if (cls > 2)\n w(((cls - 3) << 5) | 8208), cls = 0;\n }\n while (cls--)\n w(cln);\n cls = 1;\n cln = c[i];\n }\n }\n return [cl.subarray(0, cli), s];\n };\n // calculate the length of output from tree, code lengths\n var clen = function (cf, cl) {\n var l = 0;\n for (var i = 0; i < cl.length; ++i)\n l += cf[i] * cl[i];\n return l;\n };\n // writes a fixed block\n // returns the new bit pos\n var wfblk = function (out, pos, dat) {\n // no need to write 00 as type: TypedArray defaults to 0\n var s = dat.length;\n var o = shft(pos + 2);\n out[o] = s & 255;\n out[o + 1] = s >>> 8;\n out[o + 2] = out[o] ^ 255;\n out[o + 3] = out[o + 1] ^ 255;\n for (var i = 0; i < s; ++i)\n out[o + i + 4] = dat[i];\n return (o + 4 + s) * 8;\n };\n // writes a block\n var wblk = function (dat, out, final, syms, lf, df, eb, li, bs, bl, p) {\n wbits(out, p++, final);\n ++lf[256];\n var _a = hTree(lf, 15), dlt = _a[0], mlb = _a[1];\n var _b = hTree(df, 15), ddt = _b[0], mdb = _b[1];\n var _c = lc(dlt), lclt = _c[0], nlc = _c[1];\n var _d = lc(ddt), lcdt = _d[0], ndc = _d[1];\n var lcfreq = new u16(19);\n for (var i = 0; i < lclt.length; ++i)\n lcfreq[lclt[i] & 31]++;\n for (var i = 0; i < lcdt.length; ++i)\n lcfreq[lcdt[i] & 31]++;\n var _e = hTree(lcfreq, 7), lct = _e[0], mlcb = _e[1];\n var nlcc = 19;\n for (; nlcc > 4 && !lct[clim[nlcc - 1]]; --nlcc)\n ;\n var flen = (bl + 5) << 3;\n var ftlen = clen(lf, flt) + clen(df, fdt) + eb;\n var dtlen = clen(lf, dlt) + clen(df, ddt) + eb + 14 + 3 * nlcc + clen(lcfreq, lct) + (2 * lcfreq[16] + 3 * lcfreq[17] + 7 * lcfreq[18]);\n if (flen <= ftlen && flen <= dtlen)\n return wfblk(out, p, dat.subarray(bs, bs + bl));\n var lm, ll, dm, dl;\n wbits(out, p, 1 + (dtlen < ftlen)), p += 2;\n if (dtlen < ftlen) {\n lm = hMap(dlt, mlb, 0), ll = dlt, dm = hMap(ddt, mdb, 0), dl = ddt;\n var llm = hMap(lct, mlcb, 0);\n wbits(out, p, nlc - 257);\n wbits(out, p + 5, ndc - 1);\n wbits(out, p + 10, nlcc - 4);\n p += 14;\n for (var i = 0; i < nlcc; ++i)\n wbits(out, p + 3 * i, lct[clim[i]]);\n p += 3 * nlcc;\n var lcts = [lclt, lcdt];\n for (var it = 0; it < 2; ++it) {\n var clct = lcts[it];\n for (var i = 0; i < clct.length; ++i) {\n var len = clct[i] & 31;\n wbits(out, p, llm[len]), p += lct[len];\n if (len > 15)\n wbits(out, p, (clct[i] >>> 5) & 127), p += clct[i] >>> 12;\n }\n }\n }\n else {\n lm = flm, ll = flt, dm = fdm, dl = fdt;\n }\n for (var i = 0; i < li; ++i) {\n if (syms[i] > 255) {\n var len = (syms[i] >>> 18) & 31;\n wbits16(out, p, lm[len + 257]), p += ll[len + 257];\n if (len > 7)\n wbits(out, p, (syms[i] >>> 23) & 31), p += fleb[len];\n var dst = syms[i] & 31;\n wbits16(out, p, dm[dst]), p += dl[dst];\n if (dst > 3)\n wbits16(out, p, (syms[i] >>> 5) & 8191), p += fdeb[dst];\n }\n else {\n wbits16(out, p, lm[syms[i]]), p += ll[syms[i]];\n }\n }\n wbits16(out, p, lm[256]);\n return p + ll[256];\n };\n // deflate options (nice << 13) | chain\n var deo = /*#__PURE__*/ new u32([65540, 131080, 131088, 131104, 262176, 1048704, 1048832, 2114560, 2117632]);\n // empty\n var et = /*#__PURE__*/ new u8(0);\n // compresses data into a raw DEFLATE buffer\n var dflt = function (dat, lvl, plvl, pre, post, lst) {\n var s = dat.length;\n var o = new u8(pre + s + 5 * (1 + Math.ceil(s / 7000)) + post);\n // writing to this writes to the output buffer\n var w = o.subarray(pre, o.length - post);\n var pos = 0;\n if (!lvl || s < 8) {\n for (var i = 0; i <= s; i += 65535) {\n // end\n var e = i + 65535;\n if (e >= s) {\n // write final block\n w[pos >> 3] = lst;\n }\n pos = wfblk(w, pos + 1, dat.subarray(i, e));\n }\n }\n else {\n var opt = deo[lvl - 1];\n var n = opt >>> 13, c = opt & 8191;\n var msk_1 = (1 << plvl) - 1;\n // prev 2-byte val map curr 2-byte val map\n var prev = new u16(32768), head = new u16(msk_1 + 1);\n var bs1_1 = Math.ceil(plvl / 3), bs2_1 = 2 * bs1_1;\n var hsh = function (i) { return (dat[i] ^ (dat[i + 1] << bs1_1) ^ (dat[i + 2] << bs2_1)) & msk_1; };\n // 24576 is an arbitrary number of maximum symbols per block\n // 424 buffer for last block\n var syms = new u32(25000);\n // length/literal freq distance freq\n var lf = new u16(288), df = new u16(32);\n // l/lcnt exbits index l/lind waitdx bitpos\n var lc_1 = 0, eb = 0, i = 0, li = 0, wi = 0, bs = 0;\n for (; i < s; ++i) {\n // hash value\n // deopt when i > s - 3 - at end, deopt acceptable\n var hv = hsh(i);\n // index mod 32768 previous index mod\n var imod = i & 32767, pimod = head[hv];\n prev[imod] = pimod;\n head[hv] = imod;\n // We always should modify head and prev, but only add symbols if\n // this data is not yet processed (\"wait\" for wait index)\n if (wi <= i) {\n // bytes remaining\n var rem = s - i;\n if ((lc_1 > 7000 || li > 24576) && rem > 423) {\n pos = wblk(dat, w, 0, syms, lf, df, eb, li, bs, i - bs, pos);\n li = lc_1 = eb = 0, bs = i;\n for (var j = 0; j < 286; ++j)\n lf[j] = 0;\n for (var j = 0; j < 30; ++j)\n df[j] = 0;\n }\n // len dist chain\n var l = 2, d = 0, ch_1 = c, dif = (imod - pimod) & 32767;\n if (rem > 2 && hv == hsh(i - dif)) {\n var maxn = Math.min(n, rem) - 1;\n var maxd = Math.min(32767, i);\n // max possible length\n // not capped at dif because decompressors implement \"rolling\" index population\n var ml = Math.min(258, rem);\n while (dif <= maxd && --ch_1 && imod != pimod) {\n if (dat[i + l] == dat[i + l - dif]) {\n var nl = 0;\n for (; nl < ml && dat[i + nl] == dat[i + nl - dif]; ++nl)\n ;\n if (nl > l) {\n l = nl, d = dif;\n // break out early when we reach \"nice\" (we are satisfied enough)\n if (nl > maxn)\n break;\n // now, find the rarest 2-byte sequence within this\n // length of literals and search for that instead.\n // Much faster than just using the start\n var mmd = Math.min(dif, nl - 2);\n var md = 0;\n for (var j = 0; j < mmd; ++j) {\n var ti = (i - dif + j + 32768) & 32767;\n var pti = prev[ti];\n var cd = (ti - pti + 32768) & 32767;\n if (cd > md)\n md = cd, pimod = ti;\n }\n }\n }\n // check the previous match\n imod = pimod, pimod = prev[imod];\n dif += (imod - pimod + 32768) & 32767;\n }\n }\n // d will be nonzero only when a match was found\n if (d) {\n // store both dist and len data in one Uint32\n // Make sure this is recognized as a len/dist with 28th bit (2^28)\n syms[li++] = 268435456 | (revfl[l] << 18) | revfd[d];\n var lin = revfl[l] & 31, din = revfd[d] & 31;\n eb += fleb[lin] + fdeb[din];\n ++lf[257 + lin];\n ++df[din];\n wi = i + l;\n ++lc_1;\n }\n else {\n syms[li++] = dat[i];\n ++lf[dat[i]];\n }\n }\n }\n pos = wblk(dat, w, lst, syms, lf, df, eb, li, bs, i - bs, pos);\n // this is the easiest way to avoid needing to maintain state\n if (!lst && pos & 7)\n pos = wfblk(w, pos + 1, et);\n }\n return slc(o, 0, pre + shft(pos) + post);\n };\n // CRC32 table\n var crct = /*#__PURE__*/ (function () {\n var t = new Int32Array(256);\n for (var i = 0; i < 256; ++i) {\n var c = i, k = 9;\n while (--k)\n c = ((c & 1) && -306674912) ^ (c >>> 1);\n t[i] = c;\n }\n return t;\n })();\n // CRC32\n var crc = function () {\n var c = -1;\n return {\n p: function (d) {\n // closures have awful performance\n var cr = c;\n for (var i = 0; i < d.length; ++i)\n cr = crct[(cr & 255) ^ d[i]] ^ (cr >>> 8);\n c = cr;\n },\n d: function () { return ~c; }\n };\n };\n // deflate with opts\n var dopt = function (dat, opt, pre, post, st) {\n return dflt(dat, opt.level == null ? 6 : opt.level, opt.mem == null ? Math.ceil(Math.max(8, Math.min(13, Math.log(dat.length))) * 1.5) : (12 + opt.mem), pre, post, !st);\n };\n // Walmart object spread\n var mrg = function (a, b) {\n var o = {};\n for (var k in a)\n o[k] = a[k];\n for (var k in b)\n o[k] = b[k];\n return o;\n };\n // worker clone\n // This is possibly the craziest part of the entire codebase, despite how simple it may seem.\n // The only parameter to this function is a closure that returns an array of variables outside of the function scope.\n // We're going to try to figure out the variable names used in the closure as strings because that is crucial for workerization.\n // We will return an object mapping of true variable name to value (basically, the current scope as a JS object).\n // The reason we can't just use the original variable names is minifiers mangling the toplevel scope.\n // This took me three weeks to figure out how to do.\n var wcln = function (fn, fnStr, td) {\n var dt = fn();\n var st = fn.toString();\n var ks = st.slice(st.indexOf('[') + 1, st.lastIndexOf(']')).replace(/ /g, '').split(',');\n for (var i = 0; i < dt.length; ++i) {\n var v = dt[i], k = ks[i];\n if (typeof v == 'function') {\n fnStr += ';' + k + '=';\n var st_1 = v.toString();\n if (v.prototype) {\n // for global objects\n if (st_1.indexOf('[native code]') != -1) {\n var spInd = st_1.indexOf(' ', 8) + 1;\n fnStr += st_1.slice(spInd, st_1.indexOf('(', spInd));\n }\n else {\n fnStr += st_1;\n for (var t in v.prototype)\n fnStr += ';' + k + '.prototype.' + t + '=' + v.prototype[t].toString();\n }\n }\n else\n fnStr += st_1;\n }\n else\n td[k] = v;\n }\n return [fnStr, td];\n };\n var ch = [];\n // clone bufs\n var cbfs = function (v) {\n var tl = [];\n for (var k in v) {\n if (v[k] instanceof u8 || v[k] instanceof u16 || v[k] instanceof u32)\n tl.push((v[k] = new v[k].constructor(v[k])).buffer);\n }\n return tl;\n };\n // use a worker to execute code\n var wrkr = function (fns, init, id, cb) {\n var _a;\n if (!ch[id]) {\n var fnStr = '', td_1 = {}, m = fns.length - 1;\n for (var i = 0; i < m; ++i)\n _a = wcln(fns[i], fnStr, td_1), fnStr = _a[0], td_1 = _a[1];\n ch[id] = wcln(fns[m], fnStr, td_1);\n }\n var td = mrg({}, ch[id][1]);\n return wk(ch[id][0] + ';onmessage=function(e){for(var k in e.data)self[k]=e.data[k];onmessage=' + init.toString() + '}', id, td, cbfs(td), cb);\n };\n var bDflt = function () { return [u8, u16, u32, fleb, fdeb, clim, revfl, revfd, flm, flt, fdm, fdt, rev, deo, et, hMap, wbits, wbits16, hTree, ln, lc, clen, wfblk, wblk, shft, slc, dflt, dopt, deflateSync, pbf]; };\n // post buf\n var pbf = function (msg) { return postMessage(msg, [msg.buffer]); };\n // async helper\n var cbify = function (dat, opts, fns, init, id, cb) {\n var w = wrkr(fns, init, id, function (err, dat) {\n w.terminate();\n cb(err, dat);\n });\n w.postMessage([dat, opts], opts.consume ? [dat.buffer] : []);\n return function () { w.terminate(); };\n };\n // write bytes\n var wbytes = function (d, b, v) {\n for (; v; ++b)\n d[b] = v, v >>>= 8;\n };\n function deflate(data, opts, cb) {\n if (!cb)\n cb = opts, opts = {};\n if (typeof cb != 'function')\n err(7);\n return cbify(data, opts, [\n bDflt,\n ], function (ev) { return pbf(deflateSync(ev.data[0], ev.data[1])); }, 0, cb);\n }\n /**\n * Compresses data with DEFLATE without any wrapper\n * @param data The data to compress\n * @param opts The compression options\n * @returns The deflated version of the data\n */\n function deflateSync(data, opts) {\n return dopt(data, opts || {}, 0, 0);\n }\n // flatten a directory structure\n var fltn = function (d, p, t, o) {\n for (var k in d) {\n var val = d[k], n = p + k;\n if (val instanceof u8)\n t[n] = [val, o];\n else if (Array.isArray(val))\n t[n] = [val[0], mrg(o, val[1])];\n else\n fltn(val, n + '/', t, o);\n }\n };\n // text encoder\n var te = typeof TextEncoder != 'undefined' && /*#__PURE__*/ new TextEncoder();\n // text decoder\n var td = typeof TextDecoder != 'undefined' && /*#__PURE__*/ new TextDecoder();\n // text decoder stream\n var tds = 0;\n try {\n td.decode(et, { stream: true });\n tds = 1;\n }\n catch (e) { }\n /**\n * Converts a string into a Uint8Array for use with compression/decompression methods\n * @param str The string to encode\n * @param latin1 Whether or not to interpret the data as Latin-1. This should\n * not need to be true unless decoding a binary string.\n * @returns The string encoded in UTF-8/Latin-1 binary\n */\n function strToU8(str, latin1) {\n if (latin1) {\n var ar_1 = new u8(str.length);\n for (var i = 0; i < str.length; ++i)\n ar_1[i] = str.charCodeAt(i);\n return ar_1;\n }\n if (te)\n return te.encode(str);\n var l = str.length;\n var ar = new u8(str.length + (str.length >> 1));\n var ai = 0;\n var w = function (v) { ar[ai++] = v; };\n for (var i = 0; i < l; ++i) {\n if (ai + 5 > ar.length) {\n var n = new u8(ai + 8 + ((l - i) << 1));\n n.set(ar);\n ar = n;\n }\n var c = str.charCodeAt(i);\n if (c < 128 || latin1)\n w(c);\n else if (c < 2048)\n w(192 | (c >> 6)), w(128 | (c & 63));\n else if (c > 55295 && c < 57344)\n c = 65536 + (c & 1023 << 10) | (str.charCodeAt(++i) & 1023),\n w(240 | (c >> 18)), w(128 | ((c >> 12) & 63)), w(128 | ((c >> 6) & 63)), w(128 | (c & 63));\n else\n w(224 | (c >> 12)), w(128 | ((c >> 6) & 63)), w(128 | (c & 63));\n }\n return slc(ar, 0, ai);\n }\n // extra field length\n var exfl = function (ex) {\n var le = 0;\n if (ex) {\n for (var k in ex) {\n var l = ex[k].length;\n if (l > 65535)\n err(9);\n le += l + 4;\n }\n }\n return le;\n };\n // write zip header\n var wzh = function (d, b, f, fn, u, c, ce, co) {\n var fl = fn.length, ex = f.extra, col = co && co.length;\n var exl = exfl(ex);\n wbytes(d, b, ce != null ? 0x2014B50 : 0x4034B50), b += 4;\n if (ce != null)\n d[b++] = 20, d[b++] = f.os;\n d[b] = 20, b += 2; // spec compliance? what's that?\n d[b++] = (f.flag << 1) | (c == null && 8), d[b++] = u && 8;\n d[b++] = f.compression & 255, d[b++] = f.compression >> 8;\n var dt = new Date(f.mtime == null ? Date.now() : f.mtime), y = dt.getFullYear() - 1980;\n if (y < 0 || y > 119)\n err(10);\n wbytes(d, b, (y << 25) | ((dt.getMonth() + 1) << 21) | (dt.getDate() << 16) | (dt.getHours() << 11) | (dt.getMinutes() << 5) | (dt.getSeconds() >>> 1)), b += 4;\n if (c != null) {\n wbytes(d, b, f.crc);\n wbytes(d, b + 4, c);\n wbytes(d, b + 8, f.size);\n }\n wbytes(d, b + 12, fl);\n wbytes(d, b + 14, exl), b += 16;\n if (ce != null) {\n wbytes(d, b, col);\n wbytes(d, b + 6, f.attrs);\n wbytes(d, b + 10, ce), b += 14;\n }\n d.set(fn, b);\n b += fl;\n if (exl) {\n for (var k in ex) {\n var exf = ex[k], l = exf.length;\n wbytes(d, b, +k);\n wbytes(d, b + 2, l);\n d.set(exf, b + 4), b += 4 + l;\n }\n }\n if (col)\n d.set(co, b), b += col;\n return b;\n };\n // write zip footer (end of central directory)\n var wzf = function (o, b, c, d, e) {\n wbytes(o, b, 0x6054B50); // skip disk\n wbytes(o, b + 8, c);\n wbytes(o, b + 10, c);\n wbytes(o, b + 12, d);\n wbytes(o, b + 16, e);\n };\n function zip(data, opts, cb) {\n if (!cb)\n cb = opts, opts = {};\n if (typeof cb != 'function')\n err(7);\n var r = {};\n fltn(data, '', r, opts);\n var k = Object.keys(r);\n var lft = k.length, o = 0, tot = 0;\n var slft = lft, files = new Array(lft);\n var term = [];\n var tAll = function () {\n for (var i = 0; i < term.length; ++i)\n term[i]();\n };\n var cbd = function (a, b) {\n mt(function () { cb(a, b); });\n };\n mt(function () { cbd = cb; });\n var cbf = function () {\n var out = new u8(tot + 22), oe = o, cdl = tot - o;\n tot = 0;\n for (var i = 0; i < slft; ++i) {\n var f = files[i];\n try {\n var l = f.c.length;\n wzh(out, tot, f, f.f, f.u, l);\n var badd = 30 + f.f.length + exfl(f.extra);\n var loc = tot + badd;\n out.set(f.c, loc);\n wzh(out, o, f, f.f, f.u, l, tot, f.m), o += 16 + badd + (f.m ? f.m.length : 0), tot = loc + l;\n }\n catch (e) {\n return cbd(e, null);\n }\n }\n wzf(out, o, files.length, cdl, oe);\n cbd(null, out);\n };\n if (!lft)\n cbf();\n var _loop_1 = function (i) {\n var fn = k[i];\n var _a = r[fn], file = _a[0], p = _a[1];\n var c = crc(), size = file.length;\n c.p(file);\n var f = strToU8(fn), s = f.length;\n var com = p.comment, m = com && strToU8(com), ms = m && m.length;\n var exl = exfl(p.extra);\n var compression = p.level == 0 ? 0 : 8;\n var cbl = function (e, d) {\n if (e) {\n tAll();\n cbd(e, null);\n }\n else {\n var l = d.length;\n files[i] = mrg(p, {\n size: size,\n crc: c.d(),\n c: d,\n f: f,\n m: m,\n u: s != fn.length || (m && (com.length != ms)),\n compression: compression\n });\n o += 30 + s + exl + l;\n tot += 76 + 2 * (s + exl) + (ms || 0) + l;\n if (!--lft)\n cbf();\n }\n };\n if (s > 65535)\n cbl(err(11, 0, 1), null);\n if (!compression)\n cbl(null, file);\n else if (size < 160000) {\n try {\n cbl(null, deflateSync(file, p));\n }\n catch (e) {\n cbl(e, null);\n }\n }\n else\n term.push(deflate(file, p, cbl));\n };\n // Cannot use lft because it can decrease\n for (var i = 0; i < slft; ++i) {\n _loop_1(i);\n }\n return tAll;\n }\n var mt = typeof queueMicrotask == 'function' ? queueMicrotask : typeof setTimeout == 'function' ? setTimeout : function (fn) { fn(); };\n\n class USDZConverter {\n parse(scene, sceneNode, useWorker = false) {\n return Promise.reject();\n }\n convertGeometry(payload) {\n const meshObject = this.buildGeometryFromBuffers(payload);\n return buildUSDAFile(payload.options, meshObject);\n }\n zipUSDZFiles(files) {\n return new Promise((resolve, reject) => {\n zip(files, {\n level: 0,\n }, (err, res) => {\n if (err) {\n reject(err);\n }\n else {\n const blob = new Blob([res.buffer], { type: 'model/vnd.usdz+zip' });\n resolve(blob);\n }\n });\n });\n }\n buildGeometryFromBuffers(payload) {\n const positions = payload.vertexBuffers[VertexBuffer.PositionKind];\n if (!positions) {\n throw 'No indices in geometry';\n }\n const vertexCount = Math.floor(positions.length / 3);\n return buildGeometryFromData(payload.indices, vertexCount, positions, payload.vertexBuffers[VertexBuffer.NormalKind], payload.vertexBuffers[VertexBuffer.UVKind]);\n }\n buildGeometryFromObject(geometry) {\n const indices = geometry.getIndices();\n const vertexCount = geometry.getTotalVertices();\n const positions = geometry.getVerticesData(VertexBuffer.PositionKind);\n if (!indices) {\n throw 'No indices in geometry';\n }\n if (!positions) {\n throw 'No indices in geometry';\n }\n return buildGeometryFromData(indices, vertexCount, positions, geometry.getVerticesData(VertexBuffer.NormalKind), geometry.getVerticesData(VertexBuffer.UVKind));\n }\n }\n const PRECISION = 7;\n function buildHeader(opts) {\n return `#usda 1.0\r\n(\r\n customLayerData = {\r\n string creator = \"KB3D USDZConverter\"\r\n }\r\n metersPerUnit = ${1 / opts.unitsPerMeter}\r\n upAxis = \"Y\"\r\n)\r\n`;\n }\n function buildUSDAFile(opts, dataToInsert) {\n let output = buildHeader(opts);\n output += dataToInsert;\n return strToU8(output);\n }\n // Xform\n function buildXform(object, geometry, material, appendFileName = '') {\n const name = 'Object__' + escapeGUID(object.id) + appendFileName;\n const worldMatrix = object.computeWorldMatrix(true);\n const transform = buildMatrix(worldMatrix);\n if (worldMatrix.determinant() < 0) {\n console.warn('USDZConverter: USDZ does not support negative scales', object);\n }\n return `def Xform \"${name}\" (\r\n prepend references = @./geometries/Geometry__${escapeGUID(geometry.id)}${appendFileName}.usd@\r\n)\r\n{\r\n matrix4d xformOp:transform = ${transform}\r\n uniform token[] xformOpOrder = [\"xformOp:transform\"]\r\n rel material:binding = \r\n}\r\n`;\n }\n function buildMatrix(matrix) {\n const array = matrix.toArray();\n return `( ${buildMatrixRow(array, 0)}, ${buildMatrixRow(array, 4)}, ${buildMatrixRow(array, 8)}, ${buildMatrixRow(array, 12)} )`;\n }\n function buildMatrixRow(array, offset) {\n return `(${array[offset + 0]}, ${array[offset + 1]}, ${array[offset + 2]}, ${array[offset + 3]})`;\n } // Mesh\n function buildGeometryFromData(indices, verticesCount, positions, normals, uvs) {\n const vertexCounts = Array(Math.floor(indices.length / 3))\n .fill(3)\n .join(', ');\n const name = 'Geometry';\n return `\r\ndef \"Geometry\"\r\n{\r\n def Mesh \"${name}\"\r\n {\r\n int[] faceVertexCounts = [${vertexCounts}]\r\n int[] faceVertexIndices = [${stringifyIntArray(indices)}]\r\n normal3f[] normals = [${buildVector3Array(normals, verticesCount)}] (\r\n interpolation = \"vertex\"\r\n )\r\n point3f[] points = [${buildVector3Array(positions, verticesCount)}]\r\n float2[] primvars:st = [${buildVector2Array(uvs, verticesCount)}] (\r\n interpolation = \"vertex\"\r\n )\r\n uniform token subdivisionScheme = \"none\"\r\n uniform token orientation = \"leftHanded\"\r\n }\r\n}\r\n`;\n }\n function buildVector3Array(data, count) {\n let res = '';\n if (!data) {\n console.warn('USDZConverter: Data missing.');\n for (let i = 0; i < count; i++) {\n res += '(0, 0, 0), ';\n }\n return res;\n }\n return stringifyFloatArray(data, count, 3);\n }\n function buildVector2Array(data, count) {\n let res = '';\n if (!data) {\n console.warn('USDZConverter: Data missing.');\n for (let i = 0; i < count; i++) {\n res += '(0, 0), ';\n }\n return res;\n }\n return stringifyFloatArray(data, count, 2);\n }\n function stringifyIntArray(data) {\n const count = data.length;\n let res = '';\n for (let i = 0; i < count; i++) {\n res += `${data[i]}, `;\n }\n return res.slice(0, -2);\n }\n function stringifyFloatArray(data, count, perStruc) {\n let res = '';\n if (perStruc === 2) {\n for (let i = 0; i < count; i++) {\n res += `(${data[i * 2].toPrecision(PRECISION)}, ${data[i * 2 + 1].toPrecision(PRECISION)}), `;\n }\n }\n else if (perStruc === 3) {\n for (let i = 0; i < count; i++) {\n res += `(${data[i * 3].toPrecision(PRECISION)}, ${data[i * 3 + 1].toPrecision(PRECISION)}, ${data[i * 3 + 2].toPrecision(PRECISION)}), `;\n }\n }\n else {\n throw Error('Not supported');\n }\n return res.slice(0, -2);\n }\n function escapeGUID(guid) {\n return guid.replace(/[-\\s]+/g, '_');\n }\n\n const canvasResize = document.createElement('canvas');\n class USDZBabylonConverter extends USDZConverter {\n parse(scene, sceneNode, useWorker = false) {\n const toCleanup = [];\n const files = {};\n const modelFileName = 'model.usda'; // model file should be first in USDZ archive so we init it here\n files[modelFileName] = new Uint8Array(0);\n let output = buildHeader(sceneNode);\n const materials = {};\n const textures = {};\n const conversionWorker = useWorker ? new Kb3dWorker() : null;\n const asyncGenerators = [];\n scene.meshes.forEach(mesh => {\n if (mesh instanceof Mesh &&\n mesh.isEnabled() &&\n mesh.visibility > 0 &&\n mesh.geometry &&\n !(mesh instanceof LinesMesh) &&\n // for filtering out skyboxes\n !mesh.infiniteDistance) {\n let subMeshes;\n let geometries;\n let meshMaterials;\n const baseGeometry = mesh.geometry;\n if (mesh.subMeshes && mesh.subMeshes.length > 1) {\n const splitGeometryByMaterialId = new Map();\n const baseVertexData = VertexData.ExtractFromGeometry(baseGeometry);\n mesh.subMeshes.forEach((subMesh, index) => {\n const subMeshMaterial = subMesh.getMaterial();\n if (subMeshMaterial) {\n let vertexData = splitGeometryByMaterialId.get(subMeshMaterial);\n if (!vertexData) {\n vertexData = { indices: [], positions: [], normals: [], uvs: [] };\n splitGeometryByMaterialId.set(subMeshMaterial, vertexData);\n }\n // Add polygons from this submesh to the mesh for this material\n let newIdx = vertexData.indices.length;\n const endIndex = subMesh.indexStart + subMesh.indexCount;\n for (let i = subMesh.indexStart; i < endIndex; i++) {\n if (baseVertexData.indices && baseVertexData.indices[i] !== undefined) {\n const oldIdx = baseVertexData.indices[i];\n vertexData.indices.push(newIdx);\n const i2 = oldIdx * 2;\n const i3 = oldIdx * 3;\n if (baseVertexData.positions &&\n typeof baseVertexData.positions[i3] !== 'undefined') {\n vertexData.positions.push(baseVertexData.positions[i3], baseVertexData.positions[i3 + 1], baseVertexData.positions[i3 + 2]);\n }\n if (baseVertexData.normals && typeof baseVertexData.normals[i3] !== 'undefined') {\n vertexData.normals.push(baseVertexData.normals[i3], baseVertexData.normals[i3 + 1], baseVertexData.normals[i3 + 2]);\n }\n if (baseVertexData.uvs && typeof baseVertexData.uvs[i2] !== 'undefined') {\n vertexData.uvs.push(baseVertexData.uvs[i2], baseVertexData.uvs[i2 + 1]);\n }\n newIdx++;\n }\n }\n }\n return null;\n });\n subMeshes = [];\n geometries = [];\n meshMaterials = [];\n splitGeometryByMaterialId.forEach((vertexData, material) => {\n if (material instanceof PBRMaterial) {\n const newMesh = mesh.clone();\n newMesh.id = mesh.id + '__' + material.id;\n newMesh.material = material;\n const newVertexData = new VertexData();\n newVertexData.indices = vertexData.indices;\n newVertexData.positions = vertexData.positions;\n newVertexData.normals = vertexData.normals;\n newVertexData.uvs = vertexData.uvs;\n const newGeometry = Geometry.CreateGeometryForMesh(newMesh);\n newGeometry.id = `sub-${newMesh.id}`;\n newVertexData.applyToGeometry(newGeometry);\n subMeshes.push(newMesh);\n geometries.push(newGeometry);\n meshMaterials.push(material);\n toCleanup.push(newGeometry, newMesh);\n }\n else {\n console.warn(`USDZExporter: No PBR material on submesh`);\n }\n });\n }\n else {\n subMeshes = [mesh];\n geometries = [baseGeometry];\n if (mesh.material instanceof PBRMaterial) {\n meshMaterials = [mesh.material];\n }\n else {\n meshMaterials = [];\n console.warn('USDZExporter: No PBR material on mesh');\n }\n }\n for (let i = 0; i < geometries.length; i++) {\n const subMesh = subMeshes[i];\n const geometry = geometries[i];\n const meshMaterial = meshMaterials[i];\n if (!subMesh || !geometry || !meshMaterial) {\n continue;\n }\n // const appendFileName = `__${i}`;\n const geometryFileName = 'geometries/Geometry__' + escapeGUID(geometry.id) + '.usd';\n if (!(geometryFileName in files)) {\n if (conversionWorker) {\n const geometryData = buildGeometryTransferrable({ unitsPerMeter: sceneNode.unitsPerMeter, xrAllowScaling: sceneNode.xrAllowScaling }, geometry);\n if (geometryData) {\n asyncGenerators.push(conversionWorker\n .startTask(exports.eWorkerTasks.usdConvertGeometry, geometryData.payload, geometryData.transfer)\n .pipe(take(1), map(res => {\n files[geometryFileName] = res.payload.file;\n }))\n .toPromise());\n }\n }\n else {\n const meshObject = this.buildGeometryFromObject(geometry);\n files[geometryFileName] = buildUSDAFile(sceneNode, meshObject);\n }\n }\n if (!(meshMaterial.id in meshMaterial)) {\n materials[meshMaterial.id] = meshMaterial;\n }\n output += buildXform(subMesh, geometry, meshMaterial);\n }\n }\n });\n output += buildMaterials(materials, textures);\n files[modelFileName] = strToU8(output);\n const textureGenerators = [];\n for (const id in textures) {\n const texture = textures[id];\n const color = id.match(/[0-9A-F]{6}$/);\n const color3 = color && color.length === 1 ? Color3.FromHexString(`#${color[0]}`) : undefined;\n const isRGBA = texture.hasAlpha;\n textureGenerators.push(() => processImageData(texture, isRGBA ? 'image/png' : 'image/jpeg', color3)\n .then(blob => {\n if (blob) {\n return blob.arrayBuffer();\n }\n })\n .then(buffer => {\n if (buffer) {\n files[`textures/Texture__${id}.${isRGBA ? 'png' : 'jpg'}`] = new Uint8Array(buffer);\n }\n }));\n } // 64 byte alignment\n // https://github.com/101arrowz/fflate/issues/39#issuecomment-777263109\n return Promise.all([processSequentially(asyncGenerators), processSequentially(textureGenerators)])\n .then(() => {\n let offset = 0;\n for (const filename in files) {\n const file = files[filename];\n const headerSize = 34 + filename.length;\n offset += headerSize;\n const offsetMod64 = offset & 63;\n if (offsetMod64 !== 4) {\n const padLength = 64 - offsetMod64;\n const padding = new Uint8Array(padLength);\n files[filename] = [\n file,\n {\n extra: {\n 12345: padding,\n },\n },\n ];\n }\n offset = file.length;\n }\n if (conversionWorker) {\n const transfer = [];\n for (const fileName in files) {\n const file = files[fileName];\n if (file instanceof Uint8Array) {\n transfer.push(file.buffer);\n }\n else {\n transfer.push(file[0].buffer);\n }\n }\n return conversionWorker\n .startTask(exports.eWorkerTasks.usdPackage, files, transfer)\n .pipe(take(1), map(res => {\n canvasResize.width = 0;\n canvasResize.height = 0;\n conversionWorker.dispose();\n return res.payload;\n }))\n .toPromise();\n }\n else {\n return this.zipUSDZFiles(files);\n }\n })\n .then(res => {\n toCleanup.forEach(item => item.dispose());\n return res;\n });\n }\n }\n async function processSequentially(cbs) {\n const result = [];\n for (const cb of cbs) {\n if (cb instanceof Promise) {\n result.push(await cb);\n }\n else {\n result.push(await cb());\n }\n }\n return result;\n }\n function processImageData(texture, mimeType, color) {\n if (texture instanceof DynamicTexture) {\n const context = texture.getContext();\n const targetTexureSize = getBinClampedSize(context.canvas.width, context.canvas.height);\n canvasResize.width = targetTexureSize;\n canvasResize.height = targetTexureSize;\n const resizeContext = canvasResize.getContext('2d');\n const canvas = context.canvas;\n if (!(canvas instanceof HTMLCanvasElement)) {\n throw Error('texture does not have a canvas');\n }\n resizeContext.drawImage(canvas, 0, 0, targetTexureSize, targetTexureSize);\n if (color) {\n const imagedata = resizeContext.getImageData(0, 0, targetTexureSize, targetTexureSize);\n const data = imagedata.data;\n for (let i = 0; i < data.length; i += 4) {\n data[i + 0] = data[i + 0] * color.r;\n data[i + 1] = data[i + 1] * color.g;\n data[i + 2] = data[i + 2] * color.b;\n }\n resizeContext.putImageData(imagedata, 0, 0);\n }\n return canvasToBlob(canvasResize, mimeType);\n }\n else {\n const size = texture.getSize();\n const dataPromise = texture.readPixels();\n if (!dataPromise) {\n return Promise.resolve(undefined);\n }\n return dataPromise.then(data => {\n if (data instanceof Float32Array || data instanceof Float64Array) {\n for (let i = 0; i < data.length; i += 4) {\n data[i + 0] = data[i + 0] * 255;\n data[i + 1] = data[i + 1] * 255;\n data[i + 2] = data[i + 2] * 255;\n data[i + 3] = data[i + 3] * 255;\n }\n }\n let targetTexureSize = getBinClampedSize(size.width, size.height);\n if (targetTexureSize > 1024) {\n targetTexureSize = 1024;\n }\n canvasResize.width = targetTexureSize;\n canvasResize.height = targetTexureSize;\n const pxLength = targetTexureSize * targetTexureSize;\n const resizedData = new Uint8ClampedArray(pxLength * 4);\n if (targetTexureSize !== size.width || targetTexureSize !== size.height) {\n const widthRatio = size.width / targetTexureSize;\n const heightRatio = size.height / targetTexureSize;\n // in the interest of speed, just downscale with nearest neighbor algorithm instead of averaging pixels.\n for (let i = 0; i < pxLength; i++) {\n const x = i % targetTexureSize;\n // Y must be flipped for USDZ output\n const y = Math.floor(targetTexureSize - i / targetTexureSize);\n const targetIndex = (Math.floor(x * widthRatio) + Math.floor(y * heightRatio) * size.width) * 4;\n const destIndex = i * 4;\n resizedData[destIndex + 0] = data[targetIndex + 0];\n resizedData[destIndex + 1] = data[targetIndex + 1];\n resizedData[destIndex + 2] = data[targetIndex + 2];\n resizedData[destIndex + 3] = data[targetIndex + 3];\n }\n }\n else {\n const lastLineOffset = (targetTexureSize - 1) * targetTexureSize;\n for (let y = 0; y < targetTexureSize / 2; y++) {\n const yOffset = y * targetTexureSize;\n const yOffsetTarget = lastLineOffset - yOffset;\n for (let x = 0; x < targetTexureSize; x++) {\n const offset = (yOffset + x) * 4;\n const offsetTarget = (yOffsetTarget + x) * 4;\n resizedData[offset] = data[offsetTarget];\n resizedData[offset + 1] = data[offsetTarget + 1];\n resizedData[offset + 2] = data[offsetTarget + 2];\n resizedData[offset + 3] = data[offsetTarget + 3];\n resizedData[offsetTarget] = data[offset];\n resizedData[offsetTarget + 1] = data[offset + 1];\n resizedData[offsetTarget + 2] = data[offset + 2];\n resizedData[offsetTarget + 3] = data[offset + 3];\n }\n }\n }\n if (color) {\n for (let i = 0; i < resizedData.length; i += 4) {\n resizedData[i + 0] = resizedData[i + 0] * color.r;\n resizedData[i + 1] = resizedData[i + 1] * color.g;\n resizedData[i + 2] = resizedData[i + 2] * color.b;\n }\n }\n const imageData = new ImageData(resizedData, targetTexureSize, targetTexureSize);\n const resizeContext = canvasResize.getContext('2d');\n resizeContext.putImageData(imageData, 0, 0);\n return canvasToBlob(canvasResize, mimeType);\n });\n }\n }\n function canvasToBlob(canvasResize, mimeType) {\n return new Promise((resolve, reject) => {\n canvasResize.toBlob(result => {\n if (result) {\n resolve(result);\n }\n reject('Could not generate texture');\n }, mimeType, 1);\n });\n }\n function buildMaterials(materials, textures) {\n const array = [];\n for (const uuid in materials) {\n const material = materials[uuid];\n array.push(buildMaterial(material, textures));\n }\n return `def \"Materials\"\r\n{\r\n${array.join('')}\r\n}\r\n`;\n }\n function buildMaterial(material, textures) {\n // https://graphics.pixar.com/usd/docs/UsdPreviewSurface-Proposal.html\n const pad = ' ';\n const inputs = [];\n const samplers = [];\n const materialId = escapeGUID(material.id);\n const newV = new Vector3();\n const tMatrix = new Matrix();\n const t2Matrix = new Matrix();\n const t3Matrix = new Matrix();\n function buildTexture(texture, mapType, color) {\n if (texture instanceof Texture) {\n const textureName = escapeGUID(texture.name);\n const parsedTextureId = textureName + (color ? '__' + color.toHexString().substring(1) : '');\n textures[parsedTextureId] = texture;\n const angleDeg = (texture.wAng * 180) / Math.PI;\n // USD doesn't seem to have a good way to apply a matrix to a texture. But we can use the rotation and scale straight,\n // and grab the translation off of the babylon texture matrix. The matrix includes the rotation pivot related translation\n // which is nasty to calculate without a matrix and USD doesn't support UV Pivot\n const textureMatrix = texture.getTextureMatrix().m;\n Matrix.RotationYawPitchRollToRef(texture.vAng, texture.uAng, texture.wAng, tMatrix);\n Matrix.ScalingToRef(texture.uScale, texture.vScale, 1, t2Matrix);\n tMatrix.multiplyToRef(t2Matrix, t3Matrix);\n t3Matrix.invert();\n Vector3.TransformCoordinatesFromFloatsToRef(textureMatrix[8], textureMatrix[9], 0, t3Matrix, newV);\n return `\r\n def Shader \"Transform2d_${mapType}\" (\r\n sdrMetadata = {\r\n string role = \"math\"\r\n }\r\n )\r\n {\r\n uniform token info:id = \"UsdTransform2d\"\r\n float2 inputs:in.connect = \r\n float2 inputs:scale = (${texture.uScale}, ${texture.vScale})\r\n float2 inputs:translation = (${newV.x}, ${newV.y})\r\n float inputs:rotation = ${angleDeg.toPrecision(PRECISION)}\r\n float2 outputs:result\r\n }\r\n def Shader \"Texture__${textureName}_${mapType}\"\r\n {\r\n uniform token info:id = \"UsdUVTexture\"\r\n asset inputs:file = @./textures/Texture__${parsedTextureId}.${texture.hasAlpha ? 'png' : 'jpg'}@\r\n float2 inputs:st.connect = \r\n token inputs:wrapS = \"repeat\"\r\n token inputs:wrapT = \"repeat\"\r\n float outputs:r\r\n float outputs:g\r\n float outputs:b\r\n ${texture.hasAlpha ? 'float outputs:a' : ''}\r\n float3 outputs:rgb\r\n }`;\n }\n return '';\n }\n if (material.albedoTexture) {\n inputs.push(`${pad}color3f inputs:diffuseColor.connect = `);\n samplers.push(buildTexture(material.albedoTexture, 'diffuse', material.albedoColor));\n }\n else {\n inputs.push(`${pad}color3f inputs:diffuseColor = ${buildColor(material.albedoColor)}`);\n }\n if (material.emissiveTexture) {\n inputs.push(`${pad}color3f inputs:emissiveColor.connect = `);\n samplers.push(buildTexture(material.emissiveTexture, 'emissive'));\n }\n else if (material.emissiveColor.toLuminance() > 0) {\n inputs.push(`${pad}color3f inputs:emissiveColor = ${buildColor(material.emissiveColor)}`);\n }\n if (material.bumpTexture) {\n inputs.push(`${pad}normal3f inputs:normal.connect = `);\n samplers.push(buildTexture(material.bumpTexture, 'normal'));\n }\n if (material.ambientTexture !== null) {\n inputs.push(`${pad}float inputs:occlusion.connect = `);\n samplers.push(buildTexture(material.ambientTexture, 'occlusion'));\n }\n if (material.microSurfaceTexture !== null && material.roughness === 1) {\n inputs.push(`${pad}float inputs:roughness.connect = `);\n samplers.push(buildTexture(material.microSurfaceTexture, 'roughness'));\n }\n else {\n inputs.push(`${pad}float inputs:roughness = ${material.roughness}`);\n }\n if (material.metallicTexture !== null && material.metallic === 1) {\n inputs.push(`${pad}float inputs:metallic.connect = `);\n samplers.push(buildTexture(material.metallicTexture, 'metallic'));\n }\n else {\n inputs.push(`${pad}float inputs:metallic = ${material.metallic}`);\n }\n if (material.opacityTexture !== null) {\n inputs.push(`${pad}float inputs:opacity.connect = `);\n inputs.push(`${pad}float inputs:opacityThreshold = 0.0001`);\n samplers.push(buildTexture(material.opacityTexture, 'opacity'));\n }\n else if (material.albedoTexture && material.albedoTexture.hasAlpha) {\n inputs.push(`${pad}float inputs:opacity.connect = `);\n inputs.push(`${pad}float inputs:opacityThreshold = 0.0001`);\n }\n else {\n inputs.push(`${pad}float inputs:opacity = ${material.alpha}`);\n }\n if (material.clearCoat && material.clearCoat.isEnabled) {\n inputs.push(`${pad}float inputs:clearcoat = ${material.clearCoat.intensity}`);\n inputs.push(`${pad}float inputs:clearcoatRoughness = ${material.clearCoat.roughness}`);\n // TODO More properties?\n }\n inputs.push(`${pad}float inputs:ior = ${material.indexOfRefraction}`);\n // TODO More properties?\n return `\r\n def Material \"Material__${materialId}\"\r\n {\r\n def Shader \"PreviewSurface\"\r\n {\r\n uniform token info:id = \"UsdPreviewSurface\"\r\n${inputs.join('\\n')}\r\n int inputs:useSpecularWorkflow = 0\r\n token outputs:surface\r\n }\r\n token outputs:surface.connect = \r\n token inputs:frame:stPrimvarName = \"st\"\r\n def Shader \"uvReader_st\"\r\n {\r\n uniform token info:id = \"UsdPrimvarReader_float2\"\r\n token inputs:varname.connect = \r\n float2 inputs:fallback = (0.0, 0.0)\r\n float2 outputs:result\r\n }\r\n${samplers.join('\\n')}\r\n }\r\n`;\n }\n function buildGeometryTransferrable(options, geometry) {\n const transfer = [];\n const geometryBuffers = geometry.getVertexBuffers();\n const indices = geometry.getIndices();\n if (geometryBuffers && indices) {\n const vertexBuffers = Object.keys(geometryBuffers).reduce((res, type) => {\n const data = geometryBuffers[type].getData();\n if ('buffer' in data) {\n res[type] = data;\n transfer.push(res[type].slice().buffer);\n }\n else if (data instanceof ArrayBuffer || 'length' in data) {\n res[type] = new Float32Array(data);\n transfer.push(res[type].buffer);\n }\n else {\n console.warn('Unsupported geometry');\n }\n return res;\n }, {});\n const indexBuffer = Array.isArray(indices) ? new Uint32Array(indices) : indices;\n transfer.push(indexBuffer.slice().buffer);\n const result = {\n payload: {\n options,\n id: geometry.id,\n indices: indexBuffer,\n vertexBuffers,\n },\n transfer,\n };\n return result;\n }\n }\n function buildColor(color) {\n return `(${color.r}, ${color.g}, ${color.b})`;\n }\n\n class XrHitTester {\n constructor(sessionMgr) {\n this.sessionMgr = sessionMgr;\n /** temp variables for saving memory while processing hit results */\n this._tmpPos = new Vector3();\n this._tmpMat = new Matrix();\n this._tmpQuat = new Quaternion();\n }\n initialize(offsetRay) {\n const session = this.sessionMgr.session;\n //setup initial hit testing source\n if (session) {\n return session.requestHitTestSource({\n space: this.sessionMgr.viewerReferenceSpace,\n offsetRay: offsetRay,\n }).then(hitSource => {\n return (this.source = hitSource);\n });\n }\n }\n getHitResult(frame) {\n if (!this.source)\n return;\n const hits = frame.getHitTestResults(this.source);\n if (!hits.length)\n return;\n const hit = hits[0]; //only concentrate on the first hit... might want to reconsider this later\n return XrHitTester.FromXRHitTestResult(hit, this.sessionMgr.referenceSpace);\n // // Check that the y-coordinate of the normal is large enough that the normal\n // // is pointing up.\n // return hitMatrix hitMatrix.elements[5] > 0.75 ?\n // hitPosition.setFromMatrixPosition(hitMatrix) :\n // null;\n }\n static FromXRHitTestResult(hit, refSpace) {\n const pose = hit.getPose(refSpace);\n if (pose == null)\n return;\n const xrpos = pose.transform.position;\n const xrquat = pose.transform.orientation;\n const mat = new Matrix();\n Matrix.FromFloat32ArrayToRefScaled(pose.transform.matrix, 0, 1, mat);\n return {\n position: new Vector3(xrpos.x, xrpos.y, xrpos.z),\n rotation: new Quaternion(xrquat.x, xrquat.y, xrquat.z, xrquat.w),\n matrix: mat,\n xrResult: hit,\n };\n }\n dispose() {\n if (this.source) {\n this.source.cancel();\n this.source = undefined;\n }\n }\n }\n\n /**\n * Manages shadow generators and their association with lights and meshes that participate\n */\n class ShadowManager {\n constructor(_bbCache) {\n this._bbCache = _bbCache;\n this.generatorMap = new Map();\n /** the meshes that are casting shadows. The mesh renderer adds and removes them from this list as their boolean flag is changed */\n this.casters = new Set();\n /** the meshes that are receiving shadows. The mesh renderer adds and removes them from this list as their boolean flag is changed */\n this.receivers = new Set();\n /**\n * Stores the calculated shadow projection matrix for directional lights. This cache is invalidated when\n * the root scene bounding box has changed or meshes change their casting / receiving flags.\n *\n * By default, babylon calculates the projection matrix of the shadows on every frame.\n * To be more efficient, we cache the projection matrix until something that warrants updating it happens.\n */\n this.projectionMatrixCache = new Map();\n this._bbCache.rootBoundingBoxChanged.subscribe(bbInfo => {\n //invalidate our projection matrix cache\n this.projectionMatrixCache.clear();\n });\n }\n /**\n * Add a shadow generator for the given light. Should only be called when the light.generatesShadows is set to true\n * @param light\n */\n add(light, shadowResolution) {\n let sg = this.get(light);\n if (!sg) {\n sg = new ShadowGenerator(ShadowManager.convertShadowResultion(shadowResolution), light);\n this.generatorMap.set(light, sg);\n this.addAllCastersToGenerator(sg);\n if (light instanceof DirectionalLight) {\n /**\n * Directional lights are tricky, as their light shadow fulstrum needs to be the optimal size for the shadows to look good.\n * The built-in calculation of the projection matrix does not account for meshes that are receiving shadows. So we override\n * the calculation here. It also gives us a way to cache and increase performance.\n */\n light.autoUpdateExtends = false;\n light.autoCalcShadowZBounds = false;\n light.customProjectionMatrixBuilder = (viewMatrix, renderList, matrix) => {\n return this.customProjectionMatrixBuilder(light, viewMatrix, renderList, matrix);\n };\n }\n }\n return sg;\n }\n /**\n * Get the shadow generator for a specific light (if it has one)\n * @param light\n */\n get(light) {\n return this.generatorMap.get(light);\n }\n /** Remove the shadow generator for the given light */\n remove(light) {\n const sg = this.get(light);\n light.customProjectionMatrixBuilder = undefined;\n if (sg) {\n this.generatorMap.delete(light);\n sg.dispose();\n }\n }\n /**\n * Adds a mesh as a shadow caster. Should only be called by the mesh renderer when the meshNode.castShadows flag is changed to true\n */\n addCaster(mesh) {\n if (!this.casters.has(mesh)) {\n this.casters.add(mesh);\n for (const sg of this.generatorMap.values()) {\n sg.addShadowCaster(mesh);\n }\n this.projectionMatrixCache.clear();\n }\n }\n /**\n * Removes a mesh as a shadow caster. Should only be called by the mesh renderer when the meshNode.castShadows flag is changed to false\n */\n removeCaster(mesh) {\n if (this.casters.has(mesh)) {\n this.casters.delete(mesh);\n for (const sg of this.generatorMap.values()) {\n sg.removeShadowCaster(mesh);\n }\n this.projectionMatrixCache.clear();\n }\n }\n /**\n * Adds/removes the mesh as a shadow caster\n * @param mesh\n * @param value\n */\n toggleCaster(mesh, value) {\n if (value) {\n this.addCaster(mesh);\n }\n else {\n this.removeCaster(mesh);\n }\n }\n addAllCastersToGenerator(sg) {\n for (const c of this.casters.values()) {\n sg.addShadowCaster(c);\n }\n }\n addReceiver(mesh) {\n if (!this.receivers.has(mesh)) {\n this.receivers.add(mesh);\n this.projectionMatrixCache.clear();\n }\n }\n removeReceiver(mesh) {\n if (this.receivers.has(mesh)) {\n this.receivers.delete(mesh);\n this.projectionMatrixCache.clear();\n }\n }\n toggleReceiver(mesh, value) {\n if (value) {\n this.addReceiver(mesh);\n }\n else {\n this.removeReceiver(mesh);\n }\n }\n dispose() {\n this.projectionMatrixCache.clear();\n for (const light of this.generatorMap.keys()) {\n this.remove(light);\n }\n }\n clearProjectMatrixCache() {\n this.projectionMatrixCache.clear();\n }\n /**\n * Directional lights are tricky, as their light shadow fulstrum needs to be the optimal size for the shadows to look good.\n * The built-in calculation of the projection matrix does not account for meshes that are receiving shadows. So we override\n * the calculation here. It also gives us a way to cache and increase performance.\n *\n * This is based on the built-in DirectionalLight.prototype._setDefaultAutoExtendShadowProjectionMatrix\n * but with these additions:\n * * caching of the projection matrix\n * * inclusion of shadow receivers in the box calculation so shadows are not chopped off with PCSS\n * * uses the shadowOrthoScale of the light to increase the minZ/maxZ range to avoid z-fighting on planes at one of the extents of the box\n */\n customProjectionMatrixBuilder(light, viewMatrix, renderList, matrix) {\n if (this.projectionMatrixCache.has(light)) {\n matrix.copyFrom(this.projectionMatrixCache.get(light));\n }\n else {\n const allMeshes = new Set([...this.casters, ...this.receivers]);\n const shadowOrthoScale = light.shadowOrthoScale;\n let orthoLeft = Number.MAX_VALUE;\n let orthoRight = Number.MIN_VALUE;\n let orthoTop = Number.MIN_VALUE;\n let orthoBottom = Number.MAX_VALUE;\n let shadowMinZ = Number.MAX_VALUE;\n let shadowMaxZ = Number.MIN_VALUE;\n const tempVector3 = Vector3.Zero();\n //loop through each caster/reciever to build up the fulstrum box\n for (const m of allMeshes) {\n if (m.isEnabled(true)) {\n const node = getByDynamicId(m.id);\n let bbInfo;\n if (node) {\n bbInfo = this._bbCache.get(node);\n }\n if (bbInfo) {\n const boundingBox = bbInfo.boundingInfo.boundingBox;\n for (let index = 0; index < boundingBox.vectorsWorld.length; index++) {\n Vector3.TransformCoordinatesToRef(boundingBox.vectorsWorld[index], viewMatrix, tempVector3);\n if (tempVector3.x < orthoLeft) {\n orthoLeft = tempVector3.x;\n }\n if (tempVector3.y < orthoBottom) {\n orthoBottom = tempVector3.y;\n }\n if (tempVector3.x > orthoRight) {\n orthoRight = tempVector3.x;\n }\n if (tempVector3.y > orthoTop) {\n orthoTop = tempVector3.y;\n }\n if (tempVector3.z < shadowMinZ) {\n shadowMinZ = tempVector3.z;\n }\n if (tempVector3.z > shadowMaxZ) {\n shadowMaxZ = tempVector3.z;\n }\n }\n }\n }\n }\n const xOffset = orthoRight - orthoLeft;\n const yOffset = orthoTop - orthoBottom;\n /** for exponential blur, it works way better when the max z is alot larger than the scene to get soft shadows */\n const sg = this.get(light);\n if (sg.filter == ShadowGenerator.FILTER_BLUREXPONENTIALSHADOWMAP) {\n shadowMaxZ = shadowMaxZ * 50;\n }\n const zOffset = shadowMaxZ - shadowMinZ;\n Matrix.OrthoOffCenterLHToRef(orthoLeft - xOffset * shadowOrthoScale, orthoRight + xOffset * shadowOrthoScale, orthoBottom - yOffset * shadowOrthoScale, orthoTop + yOffset * shadowOrthoScale, \n /**\n * Babylon does not account for the orthoScale when calculating minZ/maxZ, which leads to shadow z fighting on planes receiving shadows.\n * So here we change this behavior to offset by the scale just like left/right/top/bottom\n */\n shadowMinZ - zOffset * shadowOrthoScale, shadowMaxZ + zOffset * shadowOrthoScale, matrix);\n this.projectionMatrixCache.set(light, matrix);\n // console.count(\"shadow manager processed\");\n }\n }\n static convertShadowResultion(r) {\n switch (r) {\n case exports.eShadowResolution.low:\n return 512;\n case exports.eShadowResolution.medium:\n return 1024;\n case exports.eShadowResolution.high:\n return 2048;\n }\n }\n static convertShadowFilter(f) {\n switch (f) {\n case exports.eShadowFilter.none:\n return ShadowGenerator.FILTER_NONE;\n case exports.eShadowFilter.simple:\n return ShadowGenerator.FILTER_PCF;\n case exports.eShadowFilter.advanced:\n return ShadowGenerator.FILTER_PCSS;\n }\n }\n }\n\n class XrLight extends DirectionalLight {\n constructor(v, scene) {\n super('xrLight', new Vector3(0, -3, 0), scene);\n this.v = v;\n this.scene = scene;\n this._originalLightEnabledMap = new Map();\n //turn all existing lights off and save their setting\n for (const l of scene.lights) {\n if (l !== this) {\n this._originalLightEnabledMap.set(l, l.isEnabled(false));\n l.setEnabled(false);\n }\n }\n this.position = this.direction.negate();\n this.intensity = 0.5;\n this._shadowManager = new ShadowManager(this.v.bbCache);\n this._sg = this._shadowManager.add(this, exports.eShadowResolution.low);\n this._sg.setDarkness(0.3);\n this._sg.filter = ShadowGenerator.FILTER_BLUREXPONENTIALSHADOWMAP;\n // this._sg.blurBoxOffset = 10;\n // this._sg.blurScale = 2;\n this._sg.useKernelBlur = true;\n this._sg.blurKernel = 64; //64;\n this._sg.blurScale = 2;\n /**\n * Add all meshes from the scene to the shadow caster.\n * This includes the shadow plane so that extents are auto calculated to include it\n */\n const rootNode = this.v.rendererManager.sceneRenderer.get3d(this.v.sceneNode);\n for (const m of rootNode.getChildMeshes(false)) {\n if (m.name == 'xrShadowPlane') {\n this._shadowManager.addReceiver(m);\n }\n else {\n this._shadowManager.addCaster(m);\n }\n // console.log(`adding mesh '${m.name}' to shadow generator`);\n }\n }\n setDirection(x, y, z) {\n this.direction.set(x, y, z);\n this.position = this.direction.negate();\n this._shadowManager.clearProjectMatrixCache();\n }\n dispose() {\n for (const [l, e] of this._originalLightEnabledMap) {\n l.setEnabled(e);\n }\n this._originalLightEnabledMap.clear();\n if (this._shadowManager)\n this._shadowManager.dispose();\n this._shadowManager = undefined;\n this._sg = undefined;\n super.dispose();\n }\n }\n\n const SEGMENTS = 12;\n const RADIUS = 0.05;\n const DELTA_PHI = Math.PI / (2 * SEGMENTS);\n const LINE_WIDTH = 0.02;\n const FRAME_RATE = 1 / 0.3;\n class XrMarker extends Mesh {\n constructor(v, scene, naturalScale) {\n super('xrMarker', scene);\n this.v = v;\n this.scene = scene;\n this.naturalScale = naturalScale;\n this.subscription = new Subscription();\n if (!XrMarker.MarkerMaterial) {\n const m = (XrMarker.MarkerMaterial = new StandardMaterial('xrMarkerMaterial', this.scene));\n m.emissiveColor = Color3.White();\n m.disableLighting = true;\n m.backFaceCulling = false;\n m.alpha = 0.5;\n }\n this.material = XrMarker.MarkerMaterial;\n this.resize();\n }\n place(rootNode) {\n this.setParent(rootNode);\n this.rotation.set(0, 0, 0);\n this._placed = true;\n this.resize();\n this.hide();\n this.subscription.add(this.v.bbCache.rootBoundingBoxChanged.subscribe(bb => {\n this.resizeToBoundingBox(this.v.bbCache.getWorld(this.v.sceneNode));\n }));\n }\n show() {\n this.animateVisibility(1, EasingFunction.EASINGMODE_EASEIN);\n }\n hide() {\n this.animateVisibility(0, EasingFunction.EASINGMODE_EASEOUT);\n }\n animateVisibility(to, easingFunction) {\n const a = new Animation('hide', 'visibility', FRAME_RATE, Animation.ANIMATIONTYPE_FLOAT);\n a.setKeys([\n {\n frame: 0,\n value: this.visibility,\n },\n {\n frame: 1,\n value: to,\n },\n ]);\n const easing = new SineEase();\n easing.setEasingMode(easingFunction);\n a.setEasingFunction(easing);\n this.scene.stopAnimation(this);\n this.scene.beginDirectAnimation(this, [a], 0, 1, false);\n }\n resize() {\n const bb = this.v.bbCache.get(this.v.sceneNode);\n if (bb) {\n this.resizeToBoundingBox(bb);\n }\n }\n resizeToBoundingBox(bb) {\n let deltaZ;\n let deltaX;\n if (!this._placed) {\n deltaX = (bb.max.x - bb.min.x) * this.naturalScale;\n deltaZ = (bb.max.z - bb.min.z) * this.naturalScale;\n this._lineWidth = LINE_WIDTH;\n this._radius = RADIUS;\n }\n else {\n deltaX = bb.max.x - bb.min.x;\n deltaZ = bb.max.z - bb.min.z;\n this._lineWidth = LINE_WIDTH / this.naturalScale;\n this._radius = RADIUS / this.naturalScale;\n }\n const vertexData = this.getVertexData(deltaX, deltaZ);\n vertexData.applyToMesh(this, true);\n if (this._placed) {\n //it's in the root node now, so center to the root node local bb\n this.position.set(bb.center.x, 0, bb.center.z);\n }\n }\n getVertexData(xSize, zSize) {\n const vertices = [];\n const indices = [];\n const normals = [];\n const x = xSize / 2;\n const z = zSize / 2;\n this.addCorner(vertices, x, z);\n this.addCorner(vertices, -x, z);\n this.addCorner(vertices, -x, -z);\n this.addCorner(vertices, x, -z);\n const numVertices = vertices.length / 3;\n for (let i = 0; i < numVertices - 2; i += 2) {\n indices.push(i, i + 1, i + 3, i, i + 3, i + 2);\n }\n const i = numVertices - 2;\n indices.push(i, i + 1, 1, i, 1, 0);\n VertexData.ComputeNormals(vertices, indices, normals);\n const vdata = new VertexData();\n vdata.indices = indices;\n vdata.positions = vertices;\n vdata.normals = normals;\n return vdata;\n }\n addCorner(vertices, cornerX, cornerZ) {\n let phi = cornerX > 0 ? (cornerZ > 0 ? 0 : -Math.PI / 2) : cornerZ > 0 ? Math.PI / 2 : Math.PI;\n for (let i = 0; i <= SEGMENTS; ++i) {\n vertices.push(cornerX + (this._radius - this._lineWidth) * Math.cos(phi), 0, cornerZ + (this._radius - this._lineWidth) * Math.sin(phi), cornerX + this._radius * Math.cos(phi), 0, cornerZ + this._radius * Math.sin(phi));\n phi += DELTA_PHI;\n }\n }\n dispose() {\n this.subscription.unsubscribe();\n super.dispose();\n }\n }\n\n // Do not edit.\n const name$2t = \"imageProcessingCompatibility\";\n const shader$2t = `#ifdef IMAGEPROCESSINGPOSTPROCESS\ngl_FragColor.rgb=pow(gl_FragColor.rgb,vec3(2.2));\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$2t] = shader$2t;\n\n // Do not edit.\n const name$2u = \"shadowOnlyPixelShader\";\n const shader$2u = `precision highp float;uniform vec4 vEyePosition;uniform float alpha;uniform vec3 shadowColor;varying vec3 vPositionW;\n#ifdef NORMAL\nvarying vec3 vNormalW;\n#endif\n#include\n#include<__decl__lightFragment>[0..maxSimultaneousLights]\n#include\n#include\n#include\n#include\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_FRAGMENT_MAIN_BEGIN\n#include\nvec3 viewDirectionW=normalize(vEyePosition.xyz-vPositionW);\n#ifdef NORMAL\nvec3 normalW=normalize(vNormalW);\n#else\nvec3 normalW=vec3(1.0,1.0,1.0);\n#endif\nvec3 diffuseBase=vec3(0.,0.,0.);lightingInfo info;float shadow=1.;float glossiness=0.;\n#include[0..1]\nvec4 color=vec4(shadowColor,(1.0-clamp(shadow,0.,1.))*alpha);\n#include\ngl_FragColor=color;\n#include\n#define CUSTOM_FRAGMENT_MAIN_END\n}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2u] = shader$2u;\n\n // Do not edit.\n const name$2v = \"shadowOnlyVertexShader\";\n const shader$2v = `precision highp float;attribute vec3 position;\n#ifdef NORMAL\nattribute vec3 normal;\n#endif\n#include\n#include\n#include\nuniform mat4 view;uniform mat4 viewProjection;\n#ifdef POINTSIZE\nuniform float pointSize;\n#endif\nvarying vec3 vPositionW;\n#ifdef NORMAL\nvarying vec3 vNormalW;\n#endif\n#ifdef VERTEXCOLOR\nvarying vec4 vColor;\n#endif\n#include\n#include\n#include<__decl__lightFragment>[0..maxSimultaneousLights]\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\n#include\n#include\n#include\nvec4 worldPos=finalWorld*vec4(position,1.0);gl_Position=viewProjection*worldPos;vPositionW=vec3(worldPos);\n#ifdef NORMAL\nvNormalW=normalize(vec3(finalWorld*vec4(normal,0.0)));\n#endif\n#include\n#include\n#include[0..maxSimultaneousLights]\n#if defined(POINTSIZE) && !defined(WEBGPU)\ngl_PointSize=pointSize;\n#endif\n#define CUSTOM_VERTEX_MAIN_END\n}\n`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2v] = shader$2v;\n\n class ShadowOnlyMaterialDefines extends MaterialDefines {\n constructor() {\n super();\n this.CLIPPLANE = false;\n this.CLIPPLANE2 = false;\n this.CLIPPLANE3 = false;\n this.CLIPPLANE4 = false;\n this.CLIPPLANE5 = false;\n this.CLIPPLANE6 = false;\n this.POINTSIZE = false;\n this.FOG = false;\n this.NORMAL = false;\n this.NUM_BONE_INFLUENCERS = 0;\n this.BonesPerMesh = 0;\n this.INSTANCES = false;\n this.IMAGEPROCESSINGPOSTPROCESS = false;\n this.SKIPFINALCOLORCLAMP = false;\n this.rebuild();\n }\n }\n class ShadowOnlyMaterial extends PushMaterial {\n constructor(name, scene) {\n super(name, scene);\n this._needAlphaBlending = true;\n this.shadowColor = Color3.Black();\n }\n needAlphaBlending() {\n return this._needAlphaBlending;\n }\n needAlphaTesting() {\n return false;\n }\n getAlphaTestTexture() {\n return null;\n }\n get activeLight() {\n return this._activeLight;\n }\n set activeLight(light) {\n this._activeLight = light;\n }\n _getFirstShadowLightForMesh(mesh) {\n for (const light of mesh.lightSources) {\n if (light.shadowEnabled) {\n return light;\n }\n }\n return null;\n }\n // Methods\n isReadyForSubMesh(mesh, subMesh, useInstances) {\n var _a;\n if (this.isFrozen) {\n if (subMesh.effect && subMesh.effect._wasPreviouslyReady && subMesh.effect._wasPreviouslyUsingInstances === useInstances) {\n return true;\n }\n }\n if (!subMesh.materialDefines) {\n subMesh.materialDefines = new ShadowOnlyMaterialDefines();\n }\n const defines = subMesh.materialDefines;\n const scene = this.getScene();\n if (this._isReadyForSubMesh(subMesh)) {\n return true;\n }\n const engine = scene.getEngine();\n // Ensure that active light is the first shadow light\n if (this._activeLight) {\n for (const light of mesh.lightSources) {\n if (light.shadowEnabled) {\n if (this._activeLight === light) {\n break; // We are good\n }\n const lightPosition = mesh.lightSources.indexOf(this._activeLight);\n if (lightPosition !== -1) {\n mesh.lightSources.splice(lightPosition, 1);\n mesh.lightSources.splice(0, 0, this._activeLight);\n }\n break;\n }\n }\n }\n MaterialHelper.PrepareDefinesForFrameBoundValues(scene, engine, this, defines, useInstances ? true : false);\n MaterialHelper.PrepareDefinesForMisc(mesh, scene, false, this.pointsCloud, this.fogEnabled, this._shouldTurnAlphaTestOn(mesh), defines);\n defines._needNormals = MaterialHelper.PrepareDefinesForLights(scene, mesh, defines, false, 1);\n const shadowGenerator = (_a = this._getFirstShadowLightForMesh(mesh)) === null || _a === void 0 ? void 0 : _a.getShadowGenerator();\n this._needAlphaBlending = true;\n if (shadowGenerator && shadowGenerator.getClassName && shadowGenerator.getClassName() === \"CascadedShadowGenerator\") {\n const csg = shadowGenerator;\n this._needAlphaBlending = !csg.autoCalcDepthBounds;\n }\n // Attribs\n MaterialHelper.PrepareDefinesForAttributes(mesh, defines, false, true);\n // Get correct effect\n if (defines.isDirty) {\n defines.markAsProcessed();\n scene.resetCachedMaterial();\n // Fallbacks\n const fallbacks = new EffectFallbacks();\n if (defines.FOG) {\n fallbacks.addFallback(1, \"FOG\");\n }\n MaterialHelper.HandleFallbacksForShadows(defines, fallbacks, 1);\n if (defines.NUM_BONE_INFLUENCERS > 0) {\n fallbacks.addCPUSkinningFallback(0, mesh);\n }\n defines.IMAGEPROCESSINGPOSTPROCESS = scene.imageProcessingConfiguration.applyByPostProcess;\n //Attributes\n const attribs = [VertexBuffer.PositionKind];\n if (defines.NORMAL) {\n attribs.push(VertexBuffer.NormalKind);\n }\n MaterialHelper.PrepareAttributesForBones(attribs, mesh, defines, fallbacks);\n MaterialHelper.PrepareAttributesForInstances(attribs, defines);\n const shaderName = \"shadowOnly\";\n const join = defines.toString();\n const uniforms = [\"world\", \"view\", \"viewProjection\", \"vEyePosition\", \"vLightsType\", \"vFogInfos\", \"vFogColor\", \"pointSize\", \"alpha\", \"shadowColor\", \"mBones\"];\n const samplers = new Array();\n const uniformBuffers = new Array();\n addClipPlaneUniforms(uniforms);\n MaterialHelper.PrepareUniformsAndSamplersList({\n uniformsNames: uniforms,\n uniformBuffersNames: uniformBuffers,\n samplers: samplers,\n defines: defines,\n maxSimultaneousLights: 1,\n });\n subMesh.setEffect(scene.getEngine().createEffect(shaderName, {\n attributes: attribs,\n uniformsNames: uniforms,\n uniformBuffersNames: uniformBuffers,\n samplers: samplers,\n defines: join,\n fallbacks: fallbacks,\n onCompiled: this.onCompiled,\n onError: this.onError,\n indexParameters: { maxSimultaneousLights: 1 },\n }, engine), defines, this._materialContext);\n }\n if (!subMesh.effect || !subMesh.effect.isReady()) {\n return false;\n }\n defines._renderId = scene.getRenderId();\n subMesh.effect._wasPreviouslyReady = true;\n subMesh.effect._wasPreviouslyUsingInstances = !!useInstances;\n return true;\n }\n bindForSubMesh(world, mesh, subMesh) {\n const scene = this.getScene();\n const defines = subMesh.materialDefines;\n if (!defines) {\n return;\n }\n const effect = subMesh.effect;\n if (!effect) {\n return;\n }\n this._activeEffect = effect;\n // Matrices\n this.bindOnlyWorldMatrix(world);\n this._activeEffect.setMatrix(\"viewProjection\", scene.getTransformMatrix());\n // Bones\n MaterialHelper.BindBonesParameters(mesh, this._activeEffect);\n if (this._mustRebind(scene, effect)) {\n // Clip plane\n bindClipPlane(effect, this, scene);\n // Point size\n if (this.pointsCloud) {\n this._activeEffect.setFloat(\"pointSize\", this.pointSize);\n }\n this._activeEffect.setFloat(\"alpha\", this.alpha);\n this._activeEffect.setColor3(\"shadowColor\", this.shadowColor);\n scene.bindEyePosition(effect);\n }\n // Lights\n if (scene.lightsEnabled) {\n MaterialHelper.BindLights(scene, mesh, this._activeEffect, defines, 1);\n const light = this._getFirstShadowLightForMesh(mesh);\n if (light) {\n // Make sure the uniforms for this light will be rebound for other materials using this light when rendering the current frame.\n // Indeed, there is an optimization in Light that binds the light uniforms only once per frame for a given light (if using ubo).\n // Doing this way assumes that all uses of this light are the same, meaning all parameters passed to Light._bindLlight\n // are the same, notably useSpecular. However, isReadyForSubMesh (see above) is passing false for this parameter, which may not be\n // the value the other materials may pass.\n light._renderId = -1;\n }\n }\n // View\n if ((scene.fogEnabled && mesh.applyFog && scene.fogMode !== Scene.FOGMODE_NONE) || defines[\"SHADOWCSM0\"]) {\n this._activeEffect.setMatrix(\"view\", scene.getViewMatrix());\n }\n // Fog\n MaterialHelper.BindFogParameters(scene, mesh, this._activeEffect);\n this._afterBind(mesh, this._activeEffect);\n }\n clone(name) {\n return SerializationHelper.Clone(() => new ShadowOnlyMaterial(name, this.getScene()), this);\n }\n serialize() {\n const serializationObject = super.serialize();\n serializationObject.customType = \"BABYLON.ShadowOnlyMaterial\";\n return serializationObject;\n }\n getClassName() {\n return \"ShadowOnlyMaterial\";\n }\n // Statics\n static Parse(source, scene, rootUrl) {\n return SerializationHelper.Parse(() => new ShadowOnlyMaterial(source.name, scene), source, scene, rootUrl);\n }\n }\n RegisterClass(\"BABYLON.ShadowOnlyMaterial\", ShadowOnlyMaterial);\n\n /** padding in size to make enough space for shadow blur */\n const SIZE_RATIO = 4;\n class XrShadowPlane extends Mesh {\n constructor(v, scene) {\n super('xrShadowPlane', scene);\n this.v = v;\n this.scene = scene;\n this.subscription = new Subscription();\n this.receiveShadows = true;\n this.isPickable = false;\n this.shadowMaterial = new ShadowOnlyMaterial('xrShadowMaterial', scene);\n this.shadowMaterial.alpha = 0.4;\n this.material = this.shadowMaterial;\n // this.material = v.rendererManager.material.defaultMaterial;\n this.setEnabled(false);\n }\n place(rootNode) {\n this.setParent(rootNode);\n const bbCache = this.v.bbCache;\n const bb = bbCache.get(this.v.sceneNode);\n if (bb) {\n this.resize(bb);\n this.setEnabled(true);\n this.subscription.add(bbCache.rootBoundingBoxChanged.subscribe(newBb => {\n this.resize(bb);\n }));\n }\n }\n resize(bb) {\n const width = (bb.max.x - bb.min.x) * SIZE_RATIO;\n const height = (bb.max.z - bb.min.z) * SIZE_RATIO;\n const vertexData = VertexData.CreatePlane({\n width,\n height,\n sideOrientation: 0,\n });\n //bake in a 90 degree rotation so it's flat on the ground\n const rotMatrix = new Matrix();\n Quaternion.FromEulerAngles(Math.PI / 2, 0, 0).toRotationMatrix(rotMatrix);\n vertexData.transform(rotMatrix);\n vertexData.applyToMesh(this, true);\n this.position.set(bb.center.x, 0, bb.center.z);\n }\n dispose() {\n this.subscription.unsubscribe();\n super.dispose();\n if (this.shadowMaterial)\n this.shadowMaterial.dispose();\n }\n }\n\n class XrTransientHitTester {\n constructor(sessionMgr) {\n this.sessionMgr = sessionMgr;\n }\n initialize(offsetRay) {\n const session = this.sessionMgr.session;\n return session.requestHitTestSourceForTransientInput({\n profile: 'generic-touchscreen',\n offsetRay: offsetRay,\n }).then(hitSource => {\n return (this.source = hitSource);\n });\n }\n getFingers(frame) {\n if (!this.source)\n return [];\n const fingers = frame.getHitTestResultsForTransientInput(this.source);\n if (!fingers.length)\n return [];\n const convertedFingers = new Array();\n for (const f of fingers) {\n const r = {\n inputSource: f.inputSource,\n results: [],\n };\n convertedFingers.push(r);\n for (const hit of f.results) {\n const convertedHit = XrHitTester.FromXRHitTestResult(hit, this.sessionMgr.referenceSpace);\n if (convertedHit)\n r.results.push(convertedHit);\n }\n }\n return convertedFingers;\n }\n dispose() {\n if (this.source) {\n this.source.cancel();\n this.source = undefined;\n }\n }\n }\n\n // interface XRSystem extends EventTarget {\n // requestSession(mode: XRSessionMode, options?: XRSessionInit): Promise;\n // isSessionSupported(mode: XRSessionMode): Promise;\n // }\n // declare global {\n // interface Navigator {\n // xr?: XRSystem;\n // }\n // }\n var eXrState;\n (function (eXrState) {\n eXrState[eXrState[\"searchingForHit\"] = 0] = \"searchingForHit\";\n eXrState[eXrState[\"initialPlacement\"] = 1] = \"initialPlacement\";\n eXrState[eXrState[\"placed\"] = 2] = \"placed\";\n })(eXrState || (eXrState = {}));\n const ROTATION_RATE = 1.5;\n const SCALE_SNAP = 1.2;\n class Xr {\n constructor(v, scene) {\n this.v = v;\n this.scene = scene;\n this.onSessionEnd = new Subject();\n this._state = eXrState.searchingForHit;\n this._goalScale = 0;\n this._goalYaw = 0;\n this._lastScalar = 0;\n this._scaleSnapHigh = SCALE_SNAP;\n this._scaleSnapLow = 1 / SCALE_SNAP;\n this._naturalScale = 1;\n this._wasJustPlaced = false;\n this._lightingEstimationCount = 0;\n this.onSelectStart = (event) => {\n var _a;\n if (this._state == eXrState.initialPlacement) {\n this.placeModel();\n this._state = eXrState.placed;\n this._initialHitTester.dispose();\n this._initialHitTester = undefined;\n this._transientHitTester = new XrTransientHitTester(this._sessionMgr);\n this._transientHitTester.initialize();\n this._wasJustPlaced = true;\n }\n else if (this._state == eXrState.placed) {\n const fingers = (_a = this._transientHitTester) === null || _a === void 0 ? void 0 : _a.getFingers(this._lastFrame);\n if ((fingers === null || fingers === void 0 ? void 0 : fingers.length) === 1) {\n /** for one finger we do a hit test on kb3d to see if they pressed the model */\n this._selectStartScreenPos = this.screenNdcToPixels(event.inputSource.gamepad.axes); //get the screen coordinates of the finger press\n const pickInfo = this.pick(this._selectStartScreenPos);\n if (pickInfo) {\n // console.log({ pickInfo });\n this._isTranslating = true;\n this._dragNodeStartPos = this._rootNode.position.clone();\n // this._dragNodeStartScreenPos = toScreen(this._rootNode.absolutePosition, this.getRigCamera());\n // this._fingerStartScreenPos = this.screenNdcToPixels(fingers[0].inputSource.gamepad!.axes);\n const dragHitTester = new XrTransientHitTester(this._sessionMgr);\n const offset = this._dragNodeStartPos.subtract(pickInfo.pickedPoint.toVec3());\n // console.log({nodePos: this._dragNodeStartPos, pickedPoint: pickInfo.pickedPoint});\n const offsetRay = new XRRay(offset);\n dragHitTester.initialize(offsetRay).then(() => {\n this._dragHitTester = dragHitTester;\n });\n }\n else {\n this._isRotating = true;\n this._goalYaw = this._rootNode.rotation.y;\n this._lastScalar = event.inputSource.gamepad.axes[0];\n }\n //simulate pointer down observable\n const args = this.v.interaction.getKb3dPointerArgs({\n type: PointerEventTypes.POINTERDOWN,\n event: new PointerEvent('pointerdown', { pointerId: event.inputSource.gamepad.id }),\n pickInfo: pickInfo,\n });\n this.v.interaction.handlePointerDown(args);\n this._marker.show();\n }\n else if ((fingers === null || fingers === void 0 ? void 0 : fingers.length) === 2 && this.v.sceneNode.xrAllowScaling) {\n this._isScaling = true;\n this._lastScalar = this.getFingerSeparation(fingers) / this._rootNode.scaling.x;\n this._marker.show();\n }\n }\n };\n this.onSelectEnd = (event) => {\n var _a;\n this._isTranslating = false;\n this._isRotating = false;\n this._isScaling = false;\n if (this._dragHitTester) {\n this._dragHitTester.dispose();\n }\n this._dragHitTester = undefined;\n this._fingerStartPos = undefined;\n //simulate pointer up observable\n const fingers = (_a = this._transientHitTester) === null || _a === void 0 ? void 0 : _a.getFingers(this._lastFrame);\n const screenCoord = this.screenNdcToPixels(event.inputSource.gamepad.axes); //get the screen coordinates of the finger press\n const pickInfo = this.pick(screenCoord);\n const args = this.v.interaction.getKb3dPointerArgs({\n type: PointerEventTypes.POINTERUP,\n event: new PointerEvent('pointerup', { pointerId: event.inputSource.gamepad.id }),\n pickInfo: pickInfo,\n });\n this.v.interaction.handlePointerUp(args);\n //simulate the pointer tap observable if the input hasn't moved much\n if (this._state == eXrState.placed) {\n if ((fingers === null || fingers === void 0 ? void 0 : fingers.length) === 1 && !this._wasJustPlaced) {\n const deltaScreen = Vector2.Distance(this._selectStartScreenPos || screenCoord, screenCoord);\n if (deltaScreen < 5) {\n const args = this.v.interaction.getKb3dPointerArgs({\n type: PointerEventTypes.POINTERTAP,\n event: new PointerEvent('pointertap', { pointerId: event.inputSource.gamepad.id }),\n pickInfo: pickInfo,\n });\n this.v.interaction.handleClick(args);\n }\n }\n this._marker.hide();\n this._wasJustPlaced = false;\n }\n };\n }\n startSession(mode, refSpaceType, overlayElem) {\n var _a, _b;\n const xrInitOptions = {\n optionalFeatures: ['light-estimation'],\n requiredFeatures: ['hit-test'],\n };\n if (overlayElem) {\n overlayElem.addEventListener('beforexrselect', ev => {\n if (ev.target != overlayElem) {\n ev.preventDefault();\n }\n });\n (_a = xrInitOptions.optionalFeatures) === null || _a === void 0 ? void 0 : _a.push('dom-overlay');\n xrInitOptions.domOverlay = {\n root: overlayElem,\n };\n }\n if (refSpaceType !== 'viewer' && refSpaceType !== 'local') {\n (_b = xrInitOptions.optionalFeatures) === null || _b === void 0 ? void 0 : _b.push(refSpaceType);\n }\n this._naturalScale = 1 / this.v.sceneNode.unitsPerMeter;\n this._sessionMgr = new WebXRSessionManager(this.scene);\n this._sessionMgr.onXRSessionEnded.addOnce(() => this.sessionEnded());\n this._xrCamera = new WebXRCamera('', this.scene, this._sessionMgr);\n return this._sessionMgr\n .initializeAsync()\n .then(() => {\n this._renderTarget = this._sessionMgr.getWebXRRenderTarget();\n return this._sessionMgr.initializeSessionAsync(mode, xrInitOptions).then(() => {\n this._session = this._sessionMgr.session;\n });\n })\n .then(() => {\n return this._sessionMgr.setReferenceSpaceTypeAsync(refSpaceType).then(refSpace => {\n this._refSpace = refSpace;\n });\n })\n .then(() => {\n return this._renderTarget.initializeXRLayerAsync(this._sessionMgr.session);\n })\n .then(() => {\n return this._sessionMgr.updateRenderState({\n depthFar: this._xrCamera.maxZ,\n depthNear: this._xrCamera.minZ,\n baseLayer: this._renderTarget.xrLayer,\n });\n })\n .then(() => {\n this._rootNode = this.v.rendererManager.sceneRenderer.get3d(this.v.sceneNode); //get the root transform node for our scene... this is what we will move to position the scene in AR\n this._rootNode.setEnabled(false);\n //subscribe to render loop\n this._sessionMgr.onXRFrameObservable.add((frame) => this.onXRFrame(frame));\n this._sessionMgr.runXRRenderLoop(); //start the render loop\n this._originalSceneCamera = this.scene.activeCamera;\n this.scene.activeCamera = this._xrCamera; //this line is important\n this.scene.autoClear = false;\n this._xrCamera.compensateOnFirstFrame = false;\n this._originalBackground = this.v.sceneNode.background;\n this.v.sceneNode.background = exports.eBackground.transparent;\n //add xr input handlers\n this._session.addEventListener('selectstart', this.onSelectStart);\n this._session.addEventListener('selectend', this.onSelectEnd);\n this._initialHitTester = new XrHitTester(this._sessionMgr);\n return this._initialHitTester.initialize();\n });\n }\n endSession() {\n if (this._sessionMgr) {\n this.sessionEndingPromise = this._sessionMgr.exitXRAsync();\n return this.sessionEndingPromise;\n }\n else {\n return Promise.resolve();\n }\n }\n sessionEnded() {\n if (this.sessionEndingPromise) {\n this.sessionEndingPromise.then(() => {\n this.sessionEndingPromise = undefined;\n this.sessionEnded();\n });\n }\n else {\n this._session.removeEventListener('selectstart', this.onSelectStart);\n this._session.removeEventListener('selectend', this.onSelectEnd);\n if (this._xrCamera)\n this._xrCamera.dispose();\n if (this._sessionMgr)\n this._sessionMgr.dispose();\n if (this._renderTarget)\n this._renderTarget.dispose();\n if (this._initialHitTester)\n this._initialHitTester.dispose();\n if (this._marker)\n this._marker.dispose();\n if (this._light)\n this._light.dispose();\n if (this._shadow)\n this._shadow.dispose();\n if (this._originalSceneCamera) {\n this.scene.activeCamera = this._originalSceneCamera;\n }\n this.scene.autoClear = true;\n this.v.sceneNode.background = this._originalBackground || exports.eBackground.transparent;\n this._rootNode.scaling = Vector3.One();\n this._rootNode.rotation = Vector3.Zero();\n this._rootNode.position = Vector3.Zero();\n this._rootNode.setEnabled(true);\n this.onSessionEnd.next();\n }\n }\n onXRFrame(frame) {\n this._lastFrame = frame;\n //console.log(`user height: ${this._xrCamera.realWorldHeight}`);\n this._viewerPose = frame.getViewerPose(this._sessionMgr.referenceSpace);\n this.placeInitialMarker(frame);\n this.processInput(frame);\n this.setSceneTransform();\n this.runLightingEstimation(frame);\n }\n runLightingEstimation(frame) {\n if (this._lightProbe) {\n this._lightingEstimationCount++;\n if (this._lightingEstimationCount == 30) {\n this._lightingEstimationCount = 0;\n const estimate = frame.getLightEstimate(this._lightProbe);\n // const probePose = frame.getPose(this._lightProbe.probeSpace, this._refSpace);\n // console.log({probePose});\n // console.log({ lightEstimate: estimate });\n if (estimate && this._light) {\n const direction = new Vector3(estimate.primaryLightDirection.x, estimate.primaryLightDirection.y, estimate.primaryLightDirection.z);\n direction.negateInPlace();\n this._light.setDirection(direction.x, direction.y, direction.z);\n const intensity = Math.max(1.0, Math.max(estimate.primaryLightIntensity.x, Math.max(estimate.primaryLightIntensity.y, estimate.primaryLightIntensity.z)));\n this._light.specular.set(estimate.primaryLightIntensity.x / intensity, estimate.primaryLightIntensity.y / intensity, estimate.primaryLightIntensity.z / intensity);\n this._light.diffuse.set(estimate.primaryLightIntensity.x / intensity, estimate.primaryLightIntensity.y / intensity, estimate.primaryLightIntensity.z / intensity);\n this._light.intensity = intensity;\n }\n }\n }\n }\n placeInitialMarker(frame) {\n if (this._state == eXrState.searchingForHit || this._state == eXrState.initialPlacement) {\n const hit = this._initialHitTester.getHitResult(frame);\n if (!hit)\n return;\n if (this._state == eXrState.searchingForHit) {\n if (!this._marker) {\n this._marker = new XrMarker(this.v, this.scene, this._naturalScale);\n this._marker.setEnabled(true);\n }\n this._state = eXrState.initialPlacement; //we got our first hit so we change state\n }\n if (this._state == eXrState.initialPlacement) {\n const quat = new Quaternion();\n hit.matrix.decompose(this._marker.scaling, quat, this._marker.position);\n this._marker.rotation = quat.toEulerAngles();\n // console.log(JSON.stringify({ markerPosition: this._marker.position }));\n }\n }\n }\n processInput(frame) {\n if (this._state == eXrState.placed && (this._isTranslating || this._isRotating || this._isScaling)) {\n const fingers = this._transientHitTester.getFingers(frame);\n if (this._isScaling) {\n if (fingers.length < 2) {\n //the user removed their 2nd finger, so the scale is cancelled\n this._isScaling = false;\n }\n else {\n const separation = this.getFingerSeparation(fingers);\n const scale = separation / this._lastScalar;\n this._goalScale =\n scale < this._scaleSnapHigh && scale > this._scaleSnapLow ? this._naturalScale : scale;\n }\n }\n else if (fingers.length === 2 && this.v.sceneNode.xrAllowScaling) {\n // happens when translating or rotating and another finger was added. We switch to scaling in this instance\n this._isTranslating = false;\n this._isRotating = false;\n this._isScaling = true;\n const scale = this._rootNode.scaling.x;\n this._lastScalar = this.getFingerSeparation(fingers) / scale;\n }\n else if (this._isRotating) {\n const inputSource = fingers[0].inputSource;\n const dragX = inputSource.gamepad.axes[0];\n this._goalYaw += (dragX - this._lastScalar) * ROTATION_RATE;\n this._lastScalar = dragX;\n }\n else if (this._isTranslating) {\n if (fingers.length > 0 && this._dragHitTester) {\n const dragFingers = this._dragHitTester.getFingers(frame);\n if (dragFingers.length && dragFingers[0].results.length) {\n const dragFinger = dragFingers[0];\n const dragFingerResult = dragFinger.results[0];\n if (!this._fingerStartPos)\n this._fingerStartPos = dragFingerResult.position;\n const deltaPos = dragFingerResult.position.subtract(this._fingerStartPos);\n this._goalPosition = this._dragNodeStartPos.add(deltaPos);\n }\n }\n }\n }\n }\n placeModel() {\n this._goalPosition = this._marker.position.clone();\n this._goalYaw = this._marker.rotation.y;\n this._goalScale = this._naturalScale;\n this._scaleSnapHigh = this._naturalScale * SCALE_SNAP;\n this._scaleSnapLow = this._naturalScale / SCALE_SNAP;\n //we now add the marker to the root node so it can scale and move with the model\n this._marker.place(this._rootNode);\n if (this.v.sceneNode.xrAddShadow) {\n this._shadow = new XrShadowPlane(this.v, this.scene);\n this._shadow.place(this._rootNode);\n this._light = new XrLight(this.v, this.scene);\n if (this.v.sceneNode.xrLightingEstimation && this._session.requestLightProbe) {\n this._session.requestLightProbe().then(lp => (this._lightProbe = lp));\n }\n }\n this._rootNode.setEnabled(true);\n //this._marker.setEnabled(false);\n this._state = eXrState.placed;\n }\n /** does a pick of the kb3d scene (NOT a webxr hit test). Accepts x/y as screen pixel coordinates */\n pick(screenCoord) {\n /**\n * The WebXRCamera in babylon appears to be for reference only, and it's position/rotation\n * are updated on each frame to be the same as the viewer pose from XR.\n *\n * However, the forward ray from the WebXRCamera is not what we would expect, it's pointing\n * in exactly the opposite direction. However, inside the WebXRCamera are rig cameras (only 1\n * in the case of AR) who are responsible for actually doing the babylon rendering. The rig\n * camera direction is what we would expect. For now we are using the rig camera to do\n * the picking, but we should keep an eye on this part as new babylon/webxr releases come out,\n * as something seems amiss.\n */\n const hardwareScale = this.scene.getEngine().getHardwareScalingLevel();\n /** adjust by hardware scale as webxr does not support it, but the pick routine uses it */\n const pickInfo = this.v.interaction.pick(screenCoord.x * hardwareScale, screenCoord.y * hardwareScale, this.getRigCamera());\n return pickInfo;\n }\n getFingerSeparation(fingers) {\n const firstFinger = fingers[0].inputSource.gamepad.axes;\n const secondFinger = fingers[1].inputSource.gamepad.axes;\n const deltaX = secondFinger[0] - firstFinger[0];\n const deltaY = secondFinger[1] - firstFinger[1];\n return Math.sqrt(deltaX * deltaX + deltaY * deltaY);\n }\n /** set's the transforms of the scene to the goal values */\n setSceneTransform() {\n if (this._state == eXrState.placed) {\n let invalidateRootBoundingBox = false;\n if (this._rootNode.scaling.x != this._goalScale) {\n this._rootNode.scaling = new Vector3(this._goalScale, this._goalScale, this._goalScale);\n invalidateRootBoundingBox = true;\n }\n if (this._rootNode.rotation.y != this._goalYaw) {\n this._rootNode.rotation.y = this._goalYaw; // = Quaternion.FromEulerAngles(0, this._goalYaw, 0);\n // this._marker.rotation.y = this._goalYaw;\n invalidateRootBoundingBox = true;\n }\n if (this._goalPosition &&\n !this._goalPosition.equalsToFloats(this._rootNode.position.x, this._rootNode.position.y, this._rootNode.position.z)) {\n this._rootNode.position.set(this._goalPosition.x, this._goalPosition.y, this._goalPosition.z);\n //this._marker.position.set(this._goalPosition.x, this._goalPosition.y, this._goalPosition.z);\n invalidateRootBoundingBox = true;\n }\n if (invalidateRootBoundingBox) {\n this.v.bbCache.invalidate(this.v.sceneNode);\n }\n }\n }\n getRigCamera() {\n const rigCamera = this._xrCamera.rigCameras[0];\n return rigCamera;\n }\n screenNdcToPixels(axes) {\n const canvas = this.scene.getEngine().getRenderingCanvas();\n //need to convert NDC units to pixels\n const x = (axes[0] + 1) * (canvas.width / 2);\n const y = (axes[1] + 1) * (canvas.height / 2);\n return new Vector2(x, y);\n }\n screenPixelsToNdc(pos) {\n const canvas = this.scene.getEngine().getRenderingCanvas();\n return {\n x: pos.x / (canvas.width / 2) - 1,\n y: pos.y / (canvas.height / 2) - 2,\n };\n }\n }\n\n const IS_IOS = /iPad|iPhone|iPod/.test(navigator.platform) || (navigator.platform === 'MacIntel' && navigator.maxTouchPoints > 1); //https://stackoverflow.com/a/58065241\n const IS_ANDROID = /android/i.test(navigator.userAgent);\n class AR {\n constructor(v, scene) {\n this.v = v;\n this.scene = scene;\n this.isDebug = false;\n this.xrActive = false;\n this.xrLoading = false;\n this.onSessionEnd = new Subject();\n this.isDebug = !!v.options.debugMode;\n this.gltfExporter = new GLTFKb3dExporter(scene);\n }\n toggle() {\n if (this.xrActive) {\n return this.end();\n }\n else {\n return this.start();\n }\n }\n start(overlayElem, onEnterFunc) {\n // Have to use a callback because android should run this after entering XR, but ios must run it before entering XR\n if (!onEnterFunc) {\n onEnterFunc = () => Promise.resolve();\n }\n const _onEnterFunc = onEnterFunc;\n if (IS_ANDROID) {\n return this.startWebXR(overlayElem).then(() => _onEnterFunc());\n }\n else if (IS_IOS || this.isDebug) {\n const usdz = new USDZBabylonConverter();\n if (this.v.sceneNode) {\n const self = this;\n return this.v\n .whenAssetsLoaded()\n .then(() => _onEnterFunc())\n .then(() => this.v.forceAndWaitForNextRender())\n .then(() => usdz.parse(this.scene, this.v.sceneNode, true))\n .then(res => {\n const a = document.createElement('a');\n if (a.relList.supports('ar') || this.isDebug) {\n if (self.currentObjectUrl) {\n window.URL.revokeObjectURL(self.currentObjectUrl);\n self.currentObjectUrl = undefined;\n }\n self.currentObjectUrl = window.URL.createObjectURL(res);\n const img = document.createElement('img');\n img.src = '/images/ar_ios.png';\n a.appendChild(img);\n document.body.appendChild(a);\n a.href = self.currentObjectUrl;\n a.download = 'test.usdz';\n a.rel = 'ar';\n // Fire XR Exit right away on iOS since the USDZ is already generated and static now\n // So, whatever scene changes they want when the user exits AR should already be in the browser\n this.xrActive = false;\n this.xrLoading = false;\n this.onSessionEnd.next();\n setTimeout(() => {\n a.click();\n setTimeout(() => {\n document.body.removeChild(a);\n }, 0);\n }, 50);\n return undefined;\n }\n return Promise.reject('Your device does not support AR.');\n });\n }\n else {\n return Promise.reject('Scene not initialized');\n }\n }\n return Promise.reject('Your device does not support AR.');\n }\n end() {\n if (IS_ANDROID) {\n return this.endWebXr();\n }\n else if (IS_IOS) {\n if (this.currentObjectUrl) {\n URL.revokeObjectURL(this.currentObjectUrl);\n }\n //TODO usdz\n }\n return Promise.resolve();\n }\n startWebXR(overlayElem) {\n this._xr = new Xr(this.v, this.scene);\n this._xr.onSessionEnd.pipe(first()).subscribe(() => {\n this.xrActive = false;\n this.xrLoading = false;\n this.onSessionEnd.next();\n });\n this.xrLoading = true;\n return this._xr.startSession('immersive-ar', 'local-floor', overlayElem).then(() => {\n this.xrActive = true;\n this.xrLoading = false;\n });\n }\n endWebXr() {\n if (this._xr) {\n return this._xr.endSession();\n }\n else {\n return Promise.resolve();\n }\n }\n /**\n * Takes a URL to a USDZ file and sets the appropriate fields so that Safari\n * iOS can intent to their AR Quick Look.\n */\n startIosQuickLook(usdzSrc, canScale) {\n const anchor = document.createElement('a');\n anchor.setAttribute('rel', 'ar');\n anchor.appendChild(document.createElement('img'));\n const modelUrl = new URL(usdzSrc, self.location.toString());\n if (!canScale) {\n modelUrl.hash = 'allowsContentScaling=0';\n }\n anchor.setAttribute('href', modelUrl.toString());\n anchor.click();\n }\n }\n\n function toBoundingBoxInfo(binfo) {\n return {\n boundingInfo: binfo,\n min: exports.KbVector.FromVec3(binfo.minimum),\n max: exports.KbVector.FromVec3(binfo.maximum),\n center: exports.KbVector.FromVec3(binfo.boundingBox.center),\n };\n }\n const MAX_DIGESTS = 30;\n class BoundingBoxCache {\n constructor(v) {\n this.v = v;\n this.boundingBoxInvalidated = new Subject();\n this.rootBoundingBoxChanged = new Subject();\n this._digestCount = 0;\n this._db = new Map();\n this._watchedNodes = new Map();\n this._changedNodes = new Set(); //stored dynamic id's of changed nodes\n }\n watchForChanges(node) {\n let subject = this._watchedNodes.get(node._dynamicId);\n if (!subject) {\n subject = new BehaviorSubject(undefined);\n this._watchedNodes.set(node._dynamicId, subject);\n }\n const _subject = subject;\n // This is using an internal implementation of rxjs, hence the any. But this is the only way we can add teardown logic.\n const observable = new Observable(s => {\n s.next();\n return () => {\n setTimeout(() => {\n if (_subject.observers.length === 0) {\n _subject.complete();\n this._watchedNodes.delete(node._dynamicId);\n }\n });\n };\n });\n return observable.pipe(mergeMap(() => _subject));\n }\n refreshAndGet(node) {\n this._db.delete(node._dynamicId);\n return this.get(node);\n }\n get(node) {\n if (node && node.visible) {\n let bb = this._db.get(node._dynamicId);\n if (!bb) {\n let bbInfo;\n const bnode = this.v.rendererManager.mesh.get3d(node);\n if (node instanceof exports.MeshNode) {\n if (bnode) {\n bbInfo = bnode.getBoundingInfo();\n }\n }\n else {\n const childNodes = node.getSpaceChildren();\n const min = {};\n const max = {};\n let empty = true;\n for (const c of childNodes) {\n const cbb = this.get(c);\n // If cbb is undefined, that means the child node is not visible or doesn't exist\n if (cbb) {\n empty = false;\n const parentMin = cbb.min.toVec3();\n const parentMax = cbb.max.toVec3();\n const childBNode = this.v.rendererManager.mesh.get3d(c);\n if (childBNode) {\n const mat = Matrix.Compose(childBNode.scaling, childBNode.rotationQuaternion, childBNode.position);\n Vector3.TransformCoordinatesToRef(parentMin, mat, parentMin);\n Vector3.TransformCoordinatesToRef(parentMax, mat, parentMax);\n }\n if (min.x == null)\n min.x = parentMin.x;\n if (min.y == null)\n min.y = parentMin.y;\n if (min.z == null)\n min.z = parentMin.z;\n if (max.x == null)\n max.x = parentMax.x;\n if (max.y == null)\n max.y = parentMax.y;\n if (max.z == null)\n max.z = parentMax.z;\n min.x = Math.min(min.x, parentMin.x, parentMax.x);\n min.y = Math.min(min.y, parentMin.y, parentMax.y);\n min.z = Math.min(min.z, parentMin.z, parentMax.z);\n max.x = Math.max(max.x, parentMin.x, parentMax.x);\n max.y = Math.max(max.y, parentMin.y, parentMax.y);\n max.z = Math.max(max.z, parentMin.z, parentMax.z);\n }\n }\n if (empty) {\n return undefined; //don't consider empty groups\n }\n bbInfo = new BoundingInfo(new Vector3(min.x || 0, min.y || 0, min.z || 0), new Vector3(max.x || 0, max.y || 0, max.z || 0));\n }\n if (bbInfo) {\n bb = toBoundingBoxInfo(bbInfo);\n this._db.set(node._dynamicId, bb);\n }\n }\n return bb;\n }\n else {\n return undefined;\n }\n }\n invalidate(node) {\n if (node) {\n //when a node is invalidated, all of it's ancestors need to be invalidated\n let n = node;\n while (n && n instanceof exports.SpaceNode) {\n this._changedNodes.add(n._dynamicId);\n n = n._parent;\n }\n }\n }\n update() {\n var _a, _b;\n let boundingBoxesStable = true;\n if (this._changedNodes.size === 0) {\n return;\n }\n if (!this.v.userIsInteracting()) {\n this._digestCount++;\n }\n const watchersToUpdateAndClear = new Map();\n for (const dynamicId of this._changedNodes) {\n const n = getByDynamicId(dynamicId);\n if (!n) {\n continue;\n }\n const externalWatchers = (_a = n._manager) === null || _a === void 0 ? void 0 : _a.boundingboxChangeHandlers.getByKey(n);\n const hasRootWatchers = this.v.sceneNode._dynamicId === n._dynamicId && this.rootBoundingBoxChanged.observers.length > 0;\n const hasWatchers = this._watchedNodes.has(n._dynamicId) || (externalWatchers !== undefined && externalWatchers.length > 0);\n const oldBoundingBox = this._db.get(n._dynamicId);\n this._db.delete(n._dynamicId);\n this.boundingBoxInvalidated.next(n);\n if (!hasWatchers && !hasRootWatchers) {\n continue;\n }\n const bbInfo = this.get(n);\n const boundingBoxChanged = !boundingBoxesMatch(bbInfo, oldBoundingBox);\n const bbWatchersSubject = this._watchedNodes.get(n._dynamicId);\n if (boundingBoxChanged) {\n boundingBoxesStable = false;\n }\n if (bbWatchersSubject) {\n bbWatchersSubject.next(bbInfo);\n }\n if (externalWatchers && bbInfo) {\n for (const w of externalWatchers) {\n const once = w.mode === 'once';\n // Always call the handler if the mode is 'once', because it will be expecting an update regardless of whether it changed or not.\n // For persistent watchers, only call it if the bb actually changed.\n if (boundingBoxChanged && !once) {\n w.handler(bbInfo);\n }\n if (once) {\n watchersToUpdateAndClear.set(n, w);\n }\n }\n }\n if (hasRootWatchers && bbInfo && boundingBoxChanged) {\n this.rootBoundingBoxChanged.next(bbInfo);\n }\n }\n if (boundingBoxesStable && this._digestCount > 3) {\n this._changedNodes.clear();\n for (const [node, watcher] of watchersToUpdateAndClear.entries()) {\n const bbInfo = this.get(node);\n if (bbInfo) {\n watcher.handler(bbInfo);\n }\n (_b = node._manager) === null || _b === void 0 ? void 0 : _b.boundingboxChangeHandlers.deleteByKeyValue(node, watcher);\n }\n watchersToUpdateAndClear.clear();\n this._digestCount = 0;\n }\n else if (!boundingBoxesStable && this._digestCount > MAX_DIGESTS) {\n console.error('Bounding box digest cycle exceeded, but the meshes are not consistent');\n this._changedNodes.clear();\n watchersToUpdateAndClear.clear();\n this._digestCount = 0;\n }\n }\n /**\n * Gets the world bounding box for the specified node.\n * Be aware that this call is not cached, so use accordingly for performance reasons\n * @param node\n */\n getWorld(node) {\n if (node && node.visible) {\n const bb = this.get(node);\n if (!bb) {\n return null;\n }\n const bnode = this.v.rendererManager.mesh.get3d(node);\n const bbInfo = new BoundingInfo(new Vector3(bb.min.x || 0, bb.min.y || 0, bb.min.z || 0), new Vector3(bb.max.x || 0, bb.max.y || 0, bb.max.z || 0), bnode === null || bnode === void 0 ? void 0 : bnode.getWorldMatrix());\n return {\n boundingInfo: bbInfo,\n min: exports.KbVector.FromVec3(bbInfo.boundingBox.minimumWorld),\n max: exports.KbVector.FromVec3(bbInfo.boundingBox.maximumWorld),\n center: exports.KbVector.FromVec3(bbInfo.boundingBox.centerWorld),\n };\n }\n return null;\n }\n dispose() {\n this._db.clear();\n this._watchedNodes.forEach(subject => {\n subject.complete();\n });\n this._watchedNodes.clear();\n }\n }\n function boundingBoxesMatch(a, b) {\n if (!a && !b) {\n return true;\n }\n if (!a || !b) {\n return false;\n }\n return a.center.equal(b.center) && a.min.equal(b.min) && a.max.equal(b.max);\n }\n\n Object.defineProperty(exports.KbVector.prototype, 'toVec3', {\n enumerable: false,\n writable: true,\n value: function (ref) {\n // add ref option\n if (!ref) {\n ref = new Vector3();\n }\n return this ? ref.copyFromFloats(this.x, this.y, this.z) : ref.setAll(0);\n },\n });\n Object.defineProperty(exports.KbVector, 'FromVec3', {\n enumerable: false,\n writable: true,\n value: function (vector) {\n return new exports.KbVector(vector.x, vector.y, vector.z);\n },\n });\n Object.defineProperty(exports.KbQuaternion.prototype, 'toQuat', {\n enumerable: false,\n writable: true,\n value: function () {\n return this ? new Quaternion(this.x, this.y, this.z, this.w) : new Quaternion(0, 0, 0, 1);\n },\n });\n Object.defineProperty(exports.KbQuaternion, 'FromQuat', {\n enumerable: false,\n writable: true,\n value: function (q) {\n return new exports.KbQuaternion(q.x, q.y, q.z, q.w);\n },\n });\n Object.defineProperty(exports.KbColor.prototype, 'toColor3', {\n enumerable: false,\n writable: true,\n value: function () {\n return this ? new Color3(this.r, this.g, this.b) : Color3.Black();\n },\n });\n Object.defineProperty(exports.KbColor.prototype, 'toColor4', {\n enumerable: false,\n writable: true,\n value: function () {\n return this ? new Color4(this.r, this.g, this.b, this.a) : new Color4(0, 0, 0, 0);\n },\n });\n\n class BoundingBoxRelativeHelper {\n static convertRelativeToAbsolute(c, bb, position) {\n if (!position) {\n position = c.position;\n }\n if (isSketchControlPoint(c)) {\n return new Vector3(((position.x + 1) / 2) * bb.max.x + bb.min.x, ((position.y + 1) / 2) * bb.max.y + bb.min.y, ((position.z + 1) / 2) * bb.max.z + bb.min.z);\n }\n else {\n return new Vector3(bb.center.x + (position.x * Math.abs(bb.max.x - bb.min.x)) / 2, bb.center.y + (position.y * Math.abs(bb.max.y - bb.min.y)) / 2, bb.center.z + (position.z * Math.abs(bb.max.z - bb.min.z)) / 2);\n }\n }\n static convertAbsoluteToRelative(c, bb, position) {\n if (!position) {\n position = c.position;\n }\n if (isSketchControlPoint(c)) {\n return new Vector3(((position.x - bb.min.x) / bb.max.x) * 2 - 1, ((position.y - bb.min.y) / bb.max.y) * 2 - 1, ((position.z - bb.min.z) / bb.max.z) * 2 - 1);\n }\n else {\n return new Vector3(((position.x - bb.center.x) * 2) / Math.abs(bb.max.x - bb.min.x), ((position.y - bb.center.y) * 2) / Math.abs(bb.max.y - bb.min.y), ((position.z - bb.center.z) * 2) / Math.abs(bb.max.z - bb.min.z));\n }\n }\n static convertRelativeDirectionToAbsolute(c, direction, bb) {\n if (isSketchControlPoint(c)) {\n return new Vector3(direction.x * (bb.min.x - bb.max.x), direction.y * (bb.min.y - bb.max.y), direction.z * (bb.min.z - bb.max.z));\n }\n else {\n return new Vector3(direction.x, direction.y, direction.z);\n }\n }\n static convertAbsoluteDirectionToRelative(c, direction, bb) {\n if (isSketchControlPoint(c)) {\n return new Vector3(direction.x / (bb.min.x - bb.max.x), direction.y / (bb.min.y - bb.max.y), direction.z / (bb.min.z - bb.max.z));\n }\n else {\n return new Vector3(direction.x, direction.y, direction.z);\n }\n }\n }\n class BoundingBoxRelativeHelperKbVector {\n static convertRelativeToAbsolute(c, bb, position) {\n return exports.KbVector.FromVec3(BoundingBoxRelativeHelper.convertRelativeToAbsolute(c, bb, position));\n }\n static convertAbsoluteToRelative(c, bb, position) {\n return exports.KbVector.FromVec3(BoundingBoxRelativeHelper.convertAbsoluteToRelative(c, bb, position));\n }\n static convertRelativeDirectionToAbsolute(c, direction, bb) {\n return exports.KbVector.FromVec3(BoundingBoxRelativeHelper.convertRelativeDirectionToAbsolute(c, direction, bb));\n }\n static convertAbsoluteDirectionToRelative(c, direction, bb) {\n return exports.KbVector.FromVec3(BoundingBoxRelativeHelper.convertAbsoluteDirectionToRelative(c, direction, bb));\n }\n }\n\n (function (eSideOrientationBabylonMap) {\n eSideOrientationBabylonMap[eSideOrientationBabylonMap[\"frontSide\"] = 0] = \"frontSide\";\n eSideOrientationBabylonMap[eSideOrientationBabylonMap[\"backSide\"] = 1] = \"backSide\";\n eSideOrientationBabylonMap[eSideOrientationBabylonMap[\"doubleSide\"] = 2] = \"doubleSide\";\n })(exports.eSideOrientationBabylonMap || (exports.eSideOrientationBabylonMap = {}));\n (function (eUvMapAxisBabylonMap) {\n eUvMapAxisBabylonMap[eUvMapAxisBabylonMap[\"x\"] = 0] = \"x\";\n eUvMapAxisBabylonMap[eUvMapAxisBabylonMap[\"y\"] = 1] = \"y\";\n eUvMapAxisBabylonMap[eUvMapAxisBabylonMap[\"z\"] = 2] = \"z\";\n })(exports.eUvMapAxisBabylonMap || (exports.eUvMapAxisBabylonMap = {}));\n (function (eUvModeBabylonMap) {\n eUvModeBabylonMap[eUvModeBabylonMap[\"planar\"] = 0] = \"planar\";\n eUvModeBabylonMap[eUvModeBabylonMap[\"box\"] = 1] = \"box\";\n eUvModeBabylonMap[eUvModeBabylonMap[\"cylindrical\"] = 2] = \"cylindrical\";\n eUvModeBabylonMap[eUvModeBabylonMap[\"spherical\"] = 3] = \"spherical\";\n })(exports.eUvModeBabylonMap || (exports.eUvModeBabylonMap = {}));\n (function (eAlphaModeBabylonMap) {\n eAlphaModeBabylonMap[eAlphaModeBabylonMap[\"disable\"] = 0] = \"disable\";\n eAlphaModeBabylonMap[eAlphaModeBabylonMap[\"add\"] = 1] = \"add\";\n eAlphaModeBabylonMap[eAlphaModeBabylonMap[\"combine\"] = 2] = \"combine\";\n eAlphaModeBabylonMap[eAlphaModeBabylonMap[\"subtract\"] = 3] = \"subtract\";\n eAlphaModeBabylonMap[eAlphaModeBabylonMap[\"multiply\"] = 4] = \"multiply\";\n })(exports.eAlphaModeBabylonMap || (exports.eAlphaModeBabylonMap = {}));\n\n function convertHdrToEnv(hdrFile, hdrSize) {\n return new Observable(subject => {\n const dataUri = URL.createObjectURL(hdrFile);\n const canvas = document.createElement('canvas');\n const dummyEngine = new Engine(canvas, true, {\n premultipliedAlpha: false,\n });\n const env = new HDRCubeTexture(dataUri, dummyEngine, hdrSize, false, true, false, true, () => {\n EnvironmentTextureTools.CreateEnvTextureAsync(env).then(buffer => {\n URL.revokeObjectURL(dataUri);\n dummyEngine.dispose();\n subject.next(new Blob([buffer]));\n subject.complete();\n });\n });\n });\n // return new Promise((resolve, reject) => {\n // const env = new HDRCubeTexture(\n // \"textures/environment.hdr\",\n // scene,\n // hdrSize,\n // false,\n // true,\n // false,\n // true,\n // () => {\n // console.log(\"environment loaded\");\n // EnvironmentTextureTools\n // .CreateEnvTextureAsync(env)\n // .then((buffer) => {\n // console.log(buffer);\n // console.log('success');\n // resolve(new Blob([buffer]));\n // })\n // }\n // );\n // });\n }\n\n function createLightVisualization(light, scene, color) {\n let lightPositionGizmo;\n if (!color) {\n color = Color3.Yellow();\n }\n if (light instanceof HemisphericLight) {\n lightPositionGizmo = SphereBuilder.CreateSphere('lightPositionMarker', { diameter: 0.75, slice: 0.5, segments: 6 }, scene);\n const coloredMaterial = new StandardMaterial('', scene);\n coloredMaterial.wireframe = true;\n coloredMaterial.emissiveColor = color;\n lightPositionGizmo.material = coloredMaterial;\n }\n else if (light instanceof SpotLight) {\n lightPositionGizmo = CylinderBuilder.CreateCylinder('lightPositionCone', { diameterTop: 0.5, height: 0.5, diameterBottom: 0, tessellation: 12 }, scene);\n lightPositionGizmo.position.y = 0.25;\n lightPositionGizmo.bakeCurrentTransformIntoVertices();\n const coloredMaterial = new StandardMaterial('', scene);\n coloredMaterial.wireframe = true;\n coloredMaterial.emissiveColor = color;\n lightPositionGizmo.material = coloredMaterial;\n }\n else if (light instanceof DirectionalLight) {\n const gizmoOffset = 1;\n lightPositionGizmo = GroundBuilder.CreateGround('lightPositionPlane', {\n height: 1.5,\n width: 1.5,\n subdivisions: 1,\n }, scene);\n const cone = CylinderBuilder.CreateCylinder('cylinder', { diameterTop: 0, height: 0.075, diameterBottom: 0.0375, tessellation: 96 }, scene);\n const line = CylinderBuilder.CreateCylinder('cylinder', { diameterTop: 0.005, height: 0.5, diameterBottom: 0.005, tessellation: 96 }, scene);\n const coloredMaterial = new StandardMaterial('', scene);\n coloredMaterial.wireframe = true;\n coloredMaterial.emissiveColor = color;\n lightPositionGizmo.material = coloredMaterial;\n cone.material = coloredMaterial;\n line.material = coloredMaterial;\n //cone.rotation.x = Math.PI / 2;\n lightPositionGizmo.position.y = -gizmoOffset;\n lightPositionGizmo.bakeCurrentTransformIntoVertices();\n cone.parent = lightPositionGizmo;\n line.parent = lightPositionGizmo;\n cone.position.y = 0.275 - gizmoOffset;\n line.position.y = -gizmoOffset;\n //line.rotation.x = Math.PI / 2;\n }\n else {\n // PointLight\n lightPositionGizmo = SphereBuilder.CreateSphere('lightPositionMarker', { diameter: 0.1 }, scene);\n const coloredMaterial = new StandardMaterial('', scene);\n coloredMaterial.emissiveColor = color;\n lightPositionGizmo.material = coloredMaterial;\n }\n lightPositionGizmo.isPickable = false;\n lightPositionGizmo.setEnabled(true);\n return lightPositionGizmo;\n }\n\n class Quad {\n constructor(capacity = 4) {\n this.capacity = capacity;\n this.points = new Array();\n }\n }\n class Boundary {\n constructor(xmin, xmax, ymin, ymax) {\n this.xmin = xmin;\n this.xmax = xmax;\n this.ymin = ymin;\n this.ymax = ymax;\n }\n boundaryContains(point) {\n return this.xmin <= point.x && this.xmax >= point.x && this.ymin <= point.y && this.ymax >= point.y;\n }\n intersects(b) {\n if (this.xmin >= b.xmax || b.xmin >= this.xmax || this.ymin >= b.ymax || b.ymin >= this.ymax) {\n return false;\n }\n return true;\n }\n get halfPoint() {\n return {\n x: this.xmin + this.width / 2,\n y: this.ymin + this.height / 2,\n };\n }\n get width() {\n return this.xmax - this.xmin;\n }\n get height() {\n return this.ymax - this.ymin;\n }\n }\n class QuadTree extends Boundary {\n constructor(o) {\n super(o.xmin, o.xmax, o.ymin, o.ymax);\n this.o = o;\n this.items = new Array();\n this.size = 0;\n this.duplicatesMap = {};\n this._divided = false;\n this.xSelector = o.xSelector;\n this.ySelector = o.ySelector;\n this.capacity = o.capacity;\n if (o.items) {\n for (const item of o.items)\n this.add(item);\n }\n }\n get divided() {\n return this._divided;\n }\n get quads() {\n const arr = Array();\n if (this.nw)\n arr.push(this.nw);\n if (this.ne)\n arr.push(this.ne);\n if (this.sw)\n arr.push(this.sw);\n if (this.se)\n arr.push(this.se);\n return arr;\n }\n find(point, radius) {\n const b = new Boundary(point.x - radius, point.x + radius, point.y - radius, point.y + radius);\n const results = this.findByBoundary(b);\n //sort by proximity\n results.sortBy(r => r.proximity);\n return results;\n }\n findByBoundary(boundary) {\n const results = new Array();\n if (!this.intersects(boundary)) {\n return results;\n }\n if (this.divided) {\n for (const q of this.quads) {\n results.push(...q.findByBoundary(boundary));\n }\n }\n else {\n //check items at this level\n for (const i of this.items) {\n if (boundary.boundaryContains(this.itemPoint(i))) {\n results.push(this.getWithProximity(i, boundary.halfPoint));\n }\n }\n }\n return results;\n }\n getWithProximity(item, searchPoint) {\n const itemPoint = this.itemPoint(item);\n return {\n item: item,\n proximity: Math.sqrt(Math.pow(searchPoint.x - itemPoint.x, 2) + Math.pow(searchPoint.y - itemPoint.y, 2)),\n };\n }\n positionKey(point) {\n return point.x + ',' + point.y;\n }\n add(item) {\n const point = this.itemPoint(item);\n const key = this.positionKey(point);\n if (!this.boundaryContains(point)) {\n return false;\n }\n // If there is another point in this quad with the **exact** same position, add it to this quad as well.\n // This may result in overfilling the quad but there's no other way to handle exact duplicate points\n if (this.duplicatesMap[key]) {\n this.items.push(item);\n return true;\n }\n //if there is space for it, and no sub-divisions, add it to this quad\n if (this.size < this.o.capacity && !this.divided) {\n this.items.push(item);\n this.duplicatesMap[key] = item;\n this.size++;\n return true;\n }\n if (!this.divided) {\n this.subdivide();\n }\n for (const q of this.quads) {\n if (q.add(item))\n return true;\n }\n throw Error('unexpected error creating quadtree');\n }\n subdivide() {\n // No need to check y, if x is infinitesimally small then y is as well\n if (this.xmax - this.xmin <= Number.EPSILON * 2) {\n throw Error('unexpected error creating quadtree. Not enough granularity');\n }\n this.nw = this.createDivision({\n items: this.items,\n xmin: this.xmin,\n xmax: this.halfPoint.x,\n ymin: this.ymin,\n ymax: this.halfPoint.y,\n });\n this.ne = this.createDivision({\n items: this.items,\n xmin: this.halfPoint.x + Number.EPSILON,\n xmax: this.xmax,\n ymin: this.ymin,\n ymax: this.halfPoint.y,\n });\n this.sw = this.createDivision({\n items: this.items,\n xmin: this.xmin,\n xmax: this.halfPoint.x,\n ymin: this.halfPoint.y + Number.EPSILON,\n ymax: this.ymax,\n });\n this.se = this.createDivision({\n items: this.items,\n xmin: this.halfPoint.x + Number.EPSILON,\n xmax: this.xmax,\n ymin: this.halfPoint.y + Number.EPSILON,\n ymax: this.ymax,\n });\n this.items.clear();\n this.size = 0;\n this._divided = true;\n }\n itemPoint(item) {\n return {\n x: this.xSelector(item),\n y: this.ySelector(item),\n };\n }\n createDivision(options) {\n return new QuadTree({\n ...this.o,\n ...options,\n });\n }\n }\n\n // @ts-nocheck third party library\n /* eslint-disable */\n function earcut$1(data, holeIndices, dim) {\n dim = dim || 2;\n var hasHoles = holeIndices && holeIndices.length, outerLen = hasHoles ? holeIndices[0] * dim : data.length, outerNode = linkedList(data, 0, outerLen, dim, true), triangles = [];\n if (!outerNode || outerNode.next === outerNode.prev)\n return triangles;\n var minX, minY, maxX, maxY, x, y, invSize;\n if (hasHoles)\n outerNode = eliminateHoles(data, holeIndices, outerNode, dim);\n // if the shape is not too simple, we'll use z-order curve hash later; calculate polygon bbox\n if (data.length > 80 * dim) {\n minX = maxX = data[0];\n minY = maxY = data[1];\n for (var i = dim; i < outerLen; i += dim) {\n x = data[i];\n y = data[i + 1];\n if (x < minX)\n minX = x;\n if (y < minY)\n minY = y;\n if (x > maxX)\n maxX = x;\n if (y > maxY)\n maxY = y;\n }\n // minX, minY and invSize are later used to transform coords into integers for z-order calculation\n invSize = Math.max(maxX - minX, maxY - minY);\n invSize = invSize !== 0 ? 32767 / invSize : 0;\n }\n earcutLinked(outerNode, triangles, dim, minX, minY, invSize, 0);\n return triangles;\n }\n // create a circular doubly linked list from polygon points in the specified winding order\n function linkedList(data, start, end, dim, clockwise) {\n var i, last;\n if (clockwise === signedArea(data, start, end, dim) > 0) {\n for (i = start; i < end; i += dim)\n last = insertNode(i, data[i], data[i + 1], last);\n }\n else {\n for (i = end - dim; i >= start; i -= dim)\n last = insertNode(i, data[i], data[i + 1], last);\n }\n if (last && equals(last, last.next)) {\n removeNode(last);\n last = last.next;\n }\n return last;\n }\n // eliminate colinear or duplicate points\n function filterPoints(start, end) {\n if (!start)\n return start;\n if (!end)\n end = start;\n var p = start, again;\n do {\n again = false;\n if (!p.steiner && (equals(p, p.next) || area(p.prev, p, p.next) === 0)) {\n removeNode(p);\n p = end = p.prev;\n if (p === p.next)\n break;\n again = true;\n }\n else {\n p = p.next;\n }\n } while (again || p !== end);\n return end;\n }\n // main ear slicing loop which triangulates a polygon (given as a linked list)\n function earcutLinked(ear, triangles, dim, minX, minY, invSize, pass) {\n if (!ear)\n return;\n // interlink polygon nodes in z-order\n if (!pass && invSize)\n indexCurve(ear, minX, minY, invSize);\n var stop = ear, prev, next;\n // iterate through ears, slicing them one by one\n while (ear.prev !== ear.next) {\n prev = ear.prev;\n next = ear.next;\n if (invSize ? isEarHashed(ear, minX, minY, invSize) : isEar(ear)) {\n // cut off the triangle\n triangles.push((prev.i / dim) | 0);\n triangles.push((ear.i / dim) | 0);\n triangles.push((next.i / dim) | 0);\n removeNode(ear);\n // skipping the next vertex leads to less sliver triangles\n ear = next.next;\n stop = next.next;\n continue;\n }\n ear = next;\n // if we looped through the whole remaining polygon and can't find any more ears\n if (ear === stop) {\n // try filtering points and slicing again\n if (!pass) {\n earcutLinked(filterPoints(ear), triangles, dim, minX, minY, invSize, 1);\n // if this didn't work, try curing all small self-intersections locally\n }\n else if (pass === 1) {\n ear = cureLocalIntersections(filterPoints(ear), triangles, dim);\n earcutLinked(ear, triangles, dim, minX, minY, invSize, 2);\n // as a last resort, try splitting the remaining polygon into two\n }\n else if (pass === 2) {\n splitEarcut(ear, triangles, dim, minX, minY, invSize);\n }\n break;\n }\n }\n }\n // check whether a polygon node forms a valid ear with adjacent nodes\n function isEar(ear) {\n var a = ear.prev, b = ear, c = ear.next;\n if (area(a, b, c) >= 0)\n return false; // reflex, can't be an ear\n // now make sure we don't have other points inside the potential ear\n var ax = a.x, bx = b.x, cx = c.x, ay = a.y, by = b.y, cy = c.y;\n // triangle bbox; min & max are calculated like this for speed\n var x0 = ax < bx ? (ax < cx ? ax : cx) : bx < cx ? bx : cx, y0 = ay < by ? (ay < cy ? ay : cy) : by < cy ? by : cy, x1 = ax > bx ? (ax > cx ? ax : cx) : bx > cx ? bx : cx, y1 = ay > by ? (ay > cy ? ay : cy) : by > cy ? by : cy;\n var p = c.next;\n while (p !== a) {\n if (p.x >= x0 &&\n p.x <= x1 &&\n p.y >= y0 &&\n p.y <= y1 &&\n pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) &&\n area(p.prev, p, p.next) >= 0)\n return false;\n p = p.next;\n }\n return true;\n }\n function isEarHashed(ear, minX, minY, invSize) {\n var a = ear.prev, b = ear, c = ear.next;\n if (area(a, b, c) >= 0)\n return false; // reflex, can't be an ear\n var ax = a.x, bx = b.x, cx = c.x, ay = a.y, by = b.y, cy = c.y;\n // triangle bbox; min & max are calculated like this for speed\n var x0 = ax < bx ? (ax < cx ? ax : cx) : bx < cx ? bx : cx, y0 = ay < by ? (ay < cy ? ay : cy) : by < cy ? by : cy, x1 = ax > bx ? (ax > cx ? ax : cx) : bx > cx ? bx : cx, y1 = ay > by ? (ay > cy ? ay : cy) : by > cy ? by : cy;\n // z-order range for the current triangle bbox;\n var minZ = zOrder(x0, y0, minX, minY, invSize), maxZ = zOrder(x1, y1, minX, minY, invSize);\n var p = ear.prevZ, n = ear.nextZ;\n // look for points inside the triangle in both directions\n while (p && p.z >= minZ && n && n.z <= maxZ) {\n if (p.x >= x0 &&\n p.x <= x1 &&\n p.y >= y0 &&\n p.y <= y1 &&\n p !== a &&\n p !== c &&\n pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) &&\n area(p.prev, p, p.next) >= 0)\n return false;\n p = p.prevZ;\n if (n.x >= x0 &&\n n.x <= x1 &&\n n.y >= y0 &&\n n.y <= y1 &&\n n !== a &&\n n !== c &&\n pointInTriangle(ax, ay, bx, by, cx, cy, n.x, n.y) &&\n area(n.prev, n, n.next) >= 0)\n return false;\n n = n.nextZ;\n }\n // look for remaining points in decreasing z-order\n while (p && p.z >= minZ) {\n if (p.x >= x0 &&\n p.x <= x1 &&\n p.y >= y0 &&\n p.y <= y1 &&\n p !== a &&\n p !== c &&\n pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) &&\n area(p.prev, p, p.next) >= 0)\n return false;\n p = p.prevZ;\n }\n // look for remaining points in increasing z-order\n while (n && n.z <= maxZ) {\n if (n.x >= x0 &&\n n.x <= x1 &&\n n.y >= y0 &&\n n.y <= y1 &&\n n !== a &&\n n !== c &&\n pointInTriangle(ax, ay, bx, by, cx, cy, n.x, n.y) &&\n area(n.prev, n, n.next) >= 0)\n return false;\n n = n.nextZ;\n }\n return true;\n }\n // go through all polygon nodes and cure small local self-intersections\n function cureLocalIntersections(start, triangles, dim) {\n var p = start;\n do {\n var a = p.prev, b = p.next.next;\n if (!equals(a, b) && intersects(a, p, p.next, b) && locallyInside(a, b) && locallyInside(b, a)) {\n triangles.push((a.i / dim) | 0);\n triangles.push((p.i / dim) | 0);\n triangles.push((b.i / dim) | 0);\n // remove two nodes involved\n removeNode(p);\n removeNode(p.next);\n p = start = b;\n }\n p = p.next;\n } while (p !== start);\n return filterPoints(p);\n }\n // try splitting polygon into two and triangulate them independently\n function splitEarcut(start, triangles, dim, minX, minY, invSize) {\n // look for a valid diagonal that divides the polygon into two\n var a = start;\n do {\n var b = a.next.next;\n while (b !== a.prev) {\n if (a.i !== b.i && isValidDiagonal(a, b)) {\n // split the polygon in two by the diagonal\n var c = splitPolygon(a, b);\n // filter colinear points around the cuts\n a = filterPoints(a, a.next);\n c = filterPoints(c, c.next);\n // run earcut on each half\n earcutLinked(a, triangles, dim, minX, minY, invSize, 0);\n earcutLinked(c, triangles, dim, minX, minY, invSize, 0);\n return;\n }\n b = b.next;\n }\n a = a.next;\n } while (a !== start);\n }\n // link every hole into the outer loop, producing a single-ring polygon without holes\n function eliminateHoles(data, holeIndices, outerNode, dim) {\n var queue = [], i, len, start, end, list;\n for (i = 0, len = holeIndices.length; i < len; i++) {\n start = holeIndices[i] * dim;\n end = i < len - 1 ? holeIndices[i + 1] * dim : data.length;\n list = linkedList(data, start, end, dim, false);\n if (list === list.next)\n list.steiner = true;\n queue.push(getLeftmost(list));\n }\n queue.sort(compareX);\n // process holes from left to right\n for (i = 0; i < queue.length; i++) {\n outerNode = eliminateHole(queue[i], outerNode);\n }\n return outerNode;\n }\n function compareX(a, b) {\n return a.x - b.x;\n }\n // find a bridge between vertices that connects hole with an outer ring and and link it\n function eliminateHole(hole, outerNode) {\n var bridge = findHoleBridge(hole, outerNode);\n if (!bridge) {\n return outerNode;\n }\n var bridgeReverse = splitPolygon(bridge, hole);\n // filter collinear points around the cuts\n filterPoints(bridgeReverse, bridgeReverse.next);\n return filterPoints(bridge, bridge.next);\n }\n // David Eberly's algorithm for finding a bridge between hole and outer polygon\n function findHoleBridge(hole, outerNode) {\n var p = outerNode, hx = hole.x, hy = hole.y, qx = -Infinity, m;\n // find a segment intersected by a ray from the hole's leftmost point to the left;\n // segment's endpoint with lesser x will be potential connection point\n do {\n if (hy <= p.y && hy >= p.next.y && p.next.y !== p.y) {\n var x = p.x + ((hy - p.y) * (p.next.x - p.x)) / (p.next.y - p.y);\n if (x <= hx && x > qx) {\n qx = x;\n m = p.x < p.next.x ? p : p.next;\n if (x === hx)\n return m; // hole touches outer segment; pick leftmost endpoint\n }\n }\n p = p.next;\n } while (p !== outerNode);\n if (!m)\n return null;\n // look for points inside the triangle of hole point, segment intersection and endpoint;\n // if there are no points found, we have a valid connection;\n // otherwise choose the point of the minimum angle with the ray as connection point\n var stop = m, mx = m.x, my = m.y, tanMin = Infinity, tan;\n p = m;\n do {\n if (hx >= p.x &&\n p.x >= mx &&\n hx !== p.x &&\n pointInTriangle(hy < my ? hx : qx, hy, mx, my, hy < my ? qx : hx, hy, p.x, p.y)) {\n tan = Math.abs(hy - p.y) / (hx - p.x); // tangential\n if (locallyInside(p, hole) &&\n (tan < tanMin || (tan === tanMin && (p.x > m.x || (p.x === m.x && sectorContainsSector(m, p)))))) {\n m = p;\n tanMin = tan;\n }\n }\n p = p.next;\n } while (p !== stop);\n return m;\n }\n // whether sector in vertex m contains sector in vertex p in the same coordinates\n function sectorContainsSector(m, p) {\n return area(m.prev, m, p.prev) < 0 && area(p.next, m, m.next) < 0;\n }\n // interlink polygon nodes in z-order\n function indexCurve(start, minX, minY, invSize) {\n var p = start;\n do {\n if (p.z === 0)\n p.z = zOrder(p.x, p.y, minX, minY, invSize);\n p.prevZ = p.prev;\n p.nextZ = p.next;\n p = p.next;\n } while (p !== start);\n p.prevZ.nextZ = null;\n p.prevZ = null;\n sortLinked(p);\n }\n // Simon Tatham's linked list merge sort algorithm\n // http://www.chiark.greenend.org.uk/~sgtatham/algorithms/listsort.html\n function sortLinked(list) {\n var i, p, q, e, tail, numMerges, pSize, qSize, inSize = 1;\n do {\n p = list;\n list = null;\n tail = null;\n numMerges = 0;\n while (p) {\n numMerges++;\n q = p;\n pSize = 0;\n for (i = 0; i < inSize; i++) {\n pSize++;\n q = q.nextZ;\n if (!q)\n break;\n }\n qSize = inSize;\n while (pSize > 0 || (qSize > 0 && q)) {\n if (pSize !== 0 && (qSize === 0 || !q || p.z <= q.z)) {\n e = p;\n p = p.nextZ;\n pSize--;\n }\n else {\n e = q;\n q = q.nextZ;\n qSize--;\n }\n if (tail)\n tail.nextZ = e;\n else\n list = e;\n e.prevZ = tail;\n tail = e;\n }\n p = q;\n }\n tail.nextZ = null;\n inSize *= 2;\n } while (numMerges > 1);\n return list;\n }\n // z-order of a point given coords and inverse of the longer side of data bbox\n function zOrder(x, y, minX, minY, invSize) {\n // coords are transformed into non-negative 15-bit integer range\n x = ((x - minX) * invSize) | 0;\n y = ((y - minY) * invSize) | 0;\n x = (x | (x << 8)) & 0x00ff00ff;\n x = (x | (x << 4)) & 0x0f0f0f0f;\n x = (x | (x << 2)) & 0x33333333;\n x = (x | (x << 1)) & 0x55555555;\n y = (y | (y << 8)) & 0x00ff00ff;\n y = (y | (y << 4)) & 0x0f0f0f0f;\n y = (y | (y << 2)) & 0x33333333;\n y = (y | (y << 1)) & 0x55555555;\n return x | (y << 1);\n }\n // find the leftmost node of a polygon ring\n function getLeftmost(start) {\n var p = start, leftmost = start;\n do {\n if (p.x < leftmost.x || (p.x === leftmost.x && p.y < leftmost.y))\n leftmost = p;\n p = p.next;\n } while (p !== start);\n return leftmost;\n }\n // check if a point lies within a convex triangle\n function pointInTriangle(ax, ay, bx, by, cx, cy, px, py) {\n return ((cx - px) * (ay - py) >= (ax - px) * (cy - py) &&\n (ax - px) * (by - py) >= (bx - px) * (ay - py) &&\n (bx - px) * (cy - py) >= (cx - px) * (by - py));\n }\n // check if a diagonal between two polygon nodes is valid (lies in polygon interior)\n function isValidDiagonal(a, b) {\n return (a.next.i !== b.i &&\n a.prev.i !== b.i &&\n !intersectsPolygon(a, b) && // dones't intersect other edges\n ((locallyInside(a, b) &&\n locallyInside(b, a) &&\n middleInside(a, b) && // locally visible\n (area(a.prev, a, b.prev) || area(a, b.prev, b))) || // does not create opposite-facing sectors\n (equals(a, b) && area(a.prev, a, a.next) > 0 && area(b.prev, b, b.next) > 0))); // special zero-length case\n }\n // signed area of a triangle\n function area(p, q, r) {\n return (q.y - p.y) * (r.x - q.x) - (q.x - p.x) * (r.y - q.y);\n }\n // check if two points are equal\n function equals(p1, p2) {\n return p1.x === p2.x && p1.y === p2.y;\n }\n // check if two segments intersect\n function intersects(p1, q1, p2, q2) {\n var o1 = sign(area(p1, q1, p2));\n var o2 = sign(area(p1, q1, q2));\n var o3 = sign(area(p2, q2, p1));\n var o4 = sign(area(p2, q2, q1));\n if (o1 !== o2 && o3 !== o4)\n return true; // general case\n if (o1 === 0 && onSegment(p1, p2, q1))\n return true; // p1, q1 and p2 are collinear and p2 lies on p1q1\n if (o2 === 0 && onSegment(p1, q2, q1))\n return true; // p1, q1 and q2 are collinear and q2 lies on p1q1\n if (o3 === 0 && onSegment(p2, p1, q2))\n return true; // p2, q2 and p1 are collinear and p1 lies on p2q2\n if (o4 === 0 && onSegment(p2, q1, q2))\n return true; // p2, q2 and q1 are collinear and q1 lies on p2q2\n return false;\n }\n // for collinear points p, q, r, check if point q lies on segment pr\n function onSegment(p, q, r) {\n return (q.x <= Math.max(p.x, r.x) && q.x >= Math.min(p.x, r.x) && q.y <= Math.max(p.y, r.y) && q.y >= Math.min(p.y, r.y));\n }\n function sign(num) {\n return num > 0 ? 1 : num < 0 ? -1 : 0;\n }\n // check if a polygon diagonal intersects any polygon segments\n function intersectsPolygon(a, b) {\n var p = a;\n do {\n if (p.i !== a.i && p.next.i !== a.i && p.i !== b.i && p.next.i !== b.i && intersects(p, p.next, a, b))\n return true;\n p = p.next;\n } while (p !== a);\n return false;\n }\n // check if a polygon diagonal is locally inside the polygon\n function locallyInside(a, b) {\n return area(a.prev, a, a.next) < 0\n ? area(a, b, a.next) >= 0 && area(a, a.prev, b) >= 0\n : area(a, b, a.prev) < 0 || area(a, a.next, b) < 0;\n }\n // check if the middle point of a polygon diagonal is inside the polygon\n function middleInside(a, b) {\n var p = a, inside = false, px = (a.x + b.x) / 2, py = (a.y + b.y) / 2;\n do {\n if (p.y > py !== p.next.y > py &&\n p.next.y !== p.y &&\n px < ((p.next.x - p.x) * (py - p.y)) / (p.next.y - p.y) + p.x)\n inside = !inside;\n p = p.next;\n } while (p !== a);\n return inside;\n }\n // link two polygon vertices with a bridge; if the vertices belong to the same ring, it splits polygon into two;\n // if one belongs to the outer ring and another to a hole, it merges it into a single ring\n function splitPolygon(a, b) {\n var a2 = new Node$1(a.i, a.x, a.y), b2 = new Node$1(b.i, b.x, b.y), an = a.next, bp = b.prev;\n a.next = b;\n b.prev = a;\n a2.next = an;\n an.prev = a2;\n b2.next = a2;\n a2.prev = b2;\n bp.next = b2;\n b2.prev = bp;\n return b2;\n }\n // create a node and optionally link it with previous one (in a circular doubly linked list)\n function insertNode(i, x, y, last) {\n var p = new Node$1(i, x, y);\n if (!last) {\n p.prev = p;\n p.next = p;\n }\n else {\n p.next = last.next;\n p.prev = last;\n last.next.prev = p;\n last.next = p;\n }\n return p;\n }\n function removeNode(p) {\n p.next.prev = p.prev;\n p.prev.next = p.next;\n if (p.prevZ)\n p.prevZ.nextZ = p.nextZ;\n if (p.nextZ)\n p.nextZ.prevZ = p.prevZ;\n }\n function Node$1(i, x, y) {\n // vertex index in coordinates array\n this.i = i;\n // vertex coordinates\n this.x = x;\n this.y = y;\n // previous and next vertex nodes in a polygon ring\n this.prev = null;\n this.next = null;\n // z-order curve value\n this.z = 0;\n // previous and next nodes in z-order\n this.prevZ = null;\n this.nextZ = null;\n // indicates whether this is a steiner point\n this.steiner = false;\n }\n // return a percentage difference between the polygon area and its triangulation area;\n // used to verify correctness of triangulation\n earcut$1.deviation = function (data, holeIndices, dim, triangles) {\n var hasHoles = holeIndices && holeIndices.length;\n var outerLen = hasHoles ? holeIndices[0] * dim : data.length;\n var polygonArea = Math.abs(signedArea(data, 0, outerLen, dim));\n if (hasHoles) {\n for (var i = 0, len = holeIndices.length; i < len; i++) {\n var start = holeIndices[i] * dim;\n var end = i < len - 1 ? holeIndices[i + 1] * dim : data.length;\n polygonArea -= Math.abs(signedArea(data, start, end, dim));\n }\n }\n var trianglesArea = 0;\n for (i = 0; i < triangles.length; i += 3) {\n var a = triangles[i] * dim;\n var b = triangles[i + 1] * dim;\n var c = triangles[i + 2] * dim;\n trianglesArea += Math.abs((data[a] - data[c]) * (data[b + 1] - data[a + 1]) - (data[a] - data[b]) * (data[c + 1] - data[a + 1]));\n }\n return polygonArea === 0 && trianglesArea === 0 ? 0 : Math.abs((trianglesArea - polygonArea) / polygonArea);\n };\n function signedArea(data, start, end, dim) {\n var sum = 0;\n for (var i = start, j = end - dim; i < end; i += dim) {\n sum += (data[j] - data[i]) * (data[i + 1] + data[j + 1]);\n j = i;\n }\n return sum;\n }\n // turn a polygon in a multi-dimensional array form (e.g. as in GeoJSON) into a form Earcut accepts\n earcut$1.flatten = function (data) {\n var dim = data[0][0].length, result = { vertices: [], holes: [], dimensions: dim }, holeIndex = 0;\n for (var i = 0; i < data.length; i++) {\n for (var j = 0; j < data[i].length; j++) {\n for (var d = 0; d < dim; d++)\n result.vertices.push(data[i][j][d]);\n }\n if (i > 0) {\n holeIndex += data[i - 1].length;\n result.holes.push(holeIndex);\n }\n }\n return result;\n };\n\n class SubmeshVertexData extends VertexData {\n constructor() {\n super();\n this.indices = null;\n this.positions = null;\n this.normals = null;\n this.tangents = null;\n this.submeshData = null;\n }\n /**\n * @param kbGeometry the kbGeometry to copy the vertex data from.\n * Note that this copipes the vertex data by reference, so mutating the data WILL modify the kbGeometry.\n */\n static FromGeometry(kbGeometry) {\n const result = new SubmeshVertexData();\n result.indices = kbGeometry.indices || null;\n result.positions = kbGeometry.positions || null;\n result.normals = kbGeometry.normals || null;\n result.tangents = kbGeometry.tangents || null;\n result.uvs = kbGeometry.uvs || null;\n result.submeshData = [\n {\n materialIndex: 0,\n verticesStart: 0,\n verticesCount: kbGeometry.positions ? kbGeometry.positions.length / 3 : 0,\n indexStart: 0,\n indexCount: kbGeometry.indices ? kbGeometry.indices.length : 0,\n },\n ];\n return result;\n }\n static ExtractFromMesh(mesh, copyWhenShared, forceCopy) {\n const result = new SubmeshVertexData();\n const vertexData = super.ExtractFromMesh(mesh, copyWhenShared, forceCopy);\n SubmeshVertexData.CopyToRef(vertexData, result);\n if (mesh.subMeshes) {\n result.submeshData = extractSubmeshData(mesh.subMeshes);\n }\n if (mesh.metadata && mesh.metadata['paths']) {\n result.paths = copyWhenShared ? mesh.metadata['paths'].slice() : mesh.metadata['paths'];\n }\n if (mesh.metadata && mesh.metadata['holes']) {\n result.holes = copyWhenShared ? mesh.metadata['holes'].slice() : mesh.metadata['holes'];\n }\n return result;\n }\n static CopyToRef(source, target) {\n target.positions = source.positions || null;\n if ('paths' in source) {\n target.paths = source.paths || null;\n }\n if ('holes' in source) {\n target.holes = source.holes || null;\n }\n target.normals = source.normals || null;\n target.tangents = source.tangents || null;\n target.uvs = source.uvs || null;\n target.uvs2 = source.uvs2 || null;\n target.uvs3 = source.uvs3 || null;\n target.uvs4 = source.uvs4 || null;\n target.uvs5 = source.uvs5 || null;\n target.uvs6 = source.uvs6 || null;\n target.colors = source.colors || null;\n target.matricesIndices = source.matricesIndices || null;\n target.matricesWeights = source.matricesWeights || null;\n target.matricesIndicesExtra = source.matricesIndicesExtra || null;\n target.matricesWeightsExtra = source.matricesWeightsExtra || null;\n target.indices = source.indices || null;\n target.submeshData = source.submeshData;\n }\n static ExtendToRef(source, target) {\n if (source.positions !== undefined)\n target.positions = source.positions;\n if ('paths' in source && source.paths !== undefined)\n target.paths = source.paths;\n if ('holes' in source && source.holes !== undefined)\n target.holes = source.holes;\n if (source.normals !== undefined)\n target.normals = source.normals;\n if (source.tangents !== undefined)\n target.tangents = source.tangents;\n if (source.uvs !== undefined)\n target.uvs = source.uvs;\n if (source.uvs2 !== undefined)\n target.uvs2 = source.uvs2;\n if (source.uvs3 !== undefined)\n target.uvs3 = source.uvs3;\n if (source.uvs4 !== undefined)\n target.uvs4 = source.uvs4;\n if (source.uvs5 !== undefined)\n target.uvs5 = source.uvs5;\n if (source.uvs6 !== undefined)\n target.uvs6 = source.uvs6;\n if (source.colors !== undefined)\n target.colors = source.colors;\n if (source.matricesIndices !== undefined)\n target.matricesIndices = source.matricesIndices;\n if (source.matricesWeights !== undefined)\n target.matricesWeights = source.matricesWeights;\n if (source.matricesIndicesExtra !== undefined)\n target.matricesIndicesExtra = source.matricesIndicesExtra;\n if (source.matricesWeightsExtra !== undefined)\n target.matricesWeightsExtra = source.matricesWeightsExtra;\n if (source.indices !== undefined)\n target.indices = source.indices;\n if (source.submeshData !== undefined) {\n target.submeshData = source.submeshData;\n }\n }\n deepClone() {\n const result = new SubmeshVertexData();\n result.indices = this.indices ? this.indices.slice() : null;\n result.positions = this.positions ? this.positions.slice() : null;\n result.paths = this.paths ? this.paths.slice() : null;\n result.holes = this.holes ? this.holes.slice() : null;\n result.normals = this.normals ? this.normals.slice() : null;\n result.tangents = this.tangents ? this.tangents.slice() : null;\n result.uvs = this.uvs ? this.uvs.slice() : null;\n result.submeshData = this.submeshData ? JSON.parse(JSON.stringify(this.submeshData)) : undefined;\n return result;\n }\n shallowClone() {\n const result = new SubmeshVertexData();\n SubmeshVertexData.CopyToRef(this, result);\n return result;\n }\n copyFrom(source) {\n SubmeshVertexData.CopyToRef(source, this);\n }\n merge(others, use32BitsIndices, forceCloneIndices, mergeMaterialIds, enableCompletion) {\n if (!Array.isArray(others)) {\n others = [others];\n }\n let pathIdxOffset = this.paths ? this.paths.length : 0;\n let segmentIdxOffset = 0;\n let vertexIdxOffset = this.positions ? this.positions.length / 3 : 0;\n const newPaths = [];\n const newHoles = [];\n if (this.paths) {\n for (let i = 2; i < this.paths.length; i += 2) {\n if (this.paths[i] !== this.paths[i - 1]) {\n segmentIdxOffset++;\n }\n }\n if (this.holes) {\n segmentIdxOffset -= this.holes.length / 2;\n }\n segmentIdxOffset++;\n }\n others.map(other => {\n if (other.holes) {\n for (let i = 0; i < other.holes.length; i += 2) {\n newHoles.push(other.holes[i] + segmentIdxOffset, other.holes[i + 1] + pathIdxOffset);\n }\n segmentIdxOffset -= other.holes.length / 2;\n }\n if (other.paths) {\n for (let i = 0; i < other.paths.length; i++) {\n newPaths.push(other.paths[i] + vertexIdxOffset);\n }\n for (let i = 2; i < other.paths.length; i += 2) {\n if (other.paths[i] !== other.paths[i - 1]) {\n segmentIdxOffset++;\n }\n }\n segmentIdxOffset++;\n }\n vertexIdxOffset += other.positions ? other.positions.length / 3 : 0;\n pathIdxOffset += other.paths ? other.paths.length : 0;\n });\n super.merge(others, use32BitsIndices, true, mergeMaterialIds, enableCompletion);\n if (newHoles.length) {\n if (!this.holes) {\n this.holes = new Int32Array();\n }\n if ('push' in this.holes) {\n for (let i = 0; i < newHoles.length; i++) {\n this.holes.push(newHoles[i]);\n }\n }\n else {\n const newArr = new Int32Array(this.holes.length + newHoles.length);\n newArr.set(this.holes);\n newArr.set(newHoles, this.holes.length);\n this.holes = newArr;\n }\n }\n if (newPaths.length) {\n if (!this.paths) {\n this.paths = new Int32Array();\n }\n if ('push' in this.paths) {\n for (let i = 0; i < newPaths.length; i++) {\n this.paths.push(newPaths[i]);\n }\n }\n else {\n const newArr = new Int32Array(this.paths.length + newPaths.length);\n newArr.set(this.paths);\n newArr.set(newPaths, this.paths.length);\n this.paths = newArr;\n }\n }\n return this;\n }\n applyToMesh(mesh, updatable = true, writeSubmeshes = true) {\n if (!this.submeshData && writeSubmeshes && mesh.subMeshes && mesh.subMeshes.length > 1) {\n this.submeshData = extractSubmeshData(mesh.subMeshes);\n }\n const result = super.applyToMesh(mesh, updatable);\n if (this.paths) {\n if (!mesh.metadata) {\n mesh.metadata = {};\n }\n mesh.metadata['paths'] = this.paths.slice();\n }\n if (this.holes) {\n if (!mesh.metadata) {\n mesh.metadata = {};\n }\n mesh.metadata['holes'] = this.holes.slice();\n }\n if (mesh.geometry) {\n // Applying geometry doesn't remove vertex types if they are unset so remove them manually\n if (!this.normals) {\n mesh.geometry.removeVerticesData(VertexBuffer.NormalKind);\n }\n if (!this.uvs) {\n mesh.geometry.removeVerticesData(VertexBuffer.UVKind);\n }\n }\n if (writeSubmeshes) {\n if (this.submeshData) {\n mesh.subMeshes = [];\n this.submeshData.forEach(submesh => {\n // SubMeshes add themselves to the mesh\n new SubMesh(submesh.materialIndex, submesh.verticesStart, submesh.verticesCount, submesh.indexStart, submesh.indexCount, mesh);\n });\n }\n }\n else {\n this.submeshData = extractSubmeshData(mesh.subMeshes);\n }\n return result;\n }\n }\n function extractSubmeshData(subMeshes) {\n return subMeshes.map(submesh => {\n return {\n materialIndex: submesh.materialIndex,\n verticesStart: submesh.verticesStart,\n verticesCount: submesh.verticesCount,\n indexStart: submesh.indexStart,\n indexCount: submesh.indexCount,\n };\n });\n }\n /**\n * @param vertexData\n * @returns null if geometry is valid, string if invalid\n */\n function verifyGeometry(vertexData) {\n let outputError = null;\n if (vertexData.indices && vertexData.positions && vertexData.indices.length > 0) {\n const indicesCount = vertexData.indices.length;\n const vertexCount = vertexData.positions.length / 3;\n if (vertexData.submeshData) {\n vertexData.submeshData.forEach(submesh => {\n if (submesh.indexStart + submesh.indexCount > indicesCount ||\n submesh.verticesStart + submesh.verticesCount > vertexCount) {\n outputError = 'submeshes';\n }\n });\n }\n if (!outputError) {\n const c = vertexData.indices.length;\n let maxVertex = -1;\n for (let i = 0; i < c; i++) {\n maxVertex = Math.max(maxVertex, vertexData.indices[i]);\n }\n if (maxVertex > vertexData.positions.length / 3) {\n outputError = 'vertices';\n }\n }\n }\n return outputError;\n }\n function getPolygonNormal(_points) {\n const normal = Vector3.Zero();\n if (!_points) {\n return normal;\n }\n if (typeof _points[0] === 'number') {\n const points = _points;\n if (points.length < 9) {\n return normal;\n }\n const p1 = Vector3.Zero();\n const p2 = Vector3.Zero();\n let i = 0;\n while (p1.length() === 0 && points.length > i + 5) {\n p1.set(points[i + 3], points[i + 4], points[i + 5]);\n p1.subtractFromFloatsToRef(points[i], points[i + 1], points[i + 2], p1);\n i += 3;\n }\n p2.set(points[i], points[i + 1], points[i + 2]);\n while (normal.length() === 0 && points.length > i + 3) {\n p2.set(points[i + 3], points[i + 4], points[i + 5]);\n p2.subtractFromFloatsToRef(points[i], points[i + 1], points[i + 2], p2);\n Vector3.CrossToRef(p1, p2, normal);\n i += 3;\n }\n }\n else {\n const points = _points;\n if (points.length < 3) {\n return normal;\n }\n const p1 = Vector3.Zero();\n const p2 = Vector3.Zero();\n let i = 0;\n while (p1.length() === 0 && points.length > i + 1) {\n p1.copyFrom(points[i + 1]);\n p1.subtractInPlace(points[i]);\n i++;\n }\n while (normal.length() === 0 && points.length > i + 1) {\n p2.copyFrom(points[i + 1]);\n p2.subtractInPlace(points[i]);\n Vector3.CrossToRef(p1, p2, normal);\n i++;\n }\n }\n return normal.normalize();\n }\n\n const MAX_BEZIER_STEPS = 10;\n const BEZIER_STEP_SIZE = 20.0;\n const GLYPH_COORDS_SCALE = 0.0005;\n const IDENTITY_Q = Quaternion.Identity();\n // class for converting path commands into point data\n class TextMeshPolygon {\n constructor() {\n this.points = [];\n this.children = [];\n this.area = 0.0;\n }\n distance(p1, p2) {\n const dx = p1.x - p2.x, dy = p1.y - p2.y;\n return Math.sqrt(dx * dx + dy * dy);\n }\n lerp(p1, p2, t) {\n return { x: (1 - t) * p1.x + t * p2.x, y: (1 - t) * p1.y + t * p2.y };\n }\n cross(p1, p2) {\n return p1.x * p2.y - p1.y * p2.x;\n }\n moveTo(p) {\n this.points.push(p);\n }\n lineTo(p) {\n this.points.push(p);\n }\n close() {\n let cur = this.points[this.points.length - 1];\n this.points.forEach(next => {\n this.area += 0.5 * this.cross(cur, next);\n cur = next;\n });\n }\n conicTo(p, p1) {\n const p0 = this.points[this.points.length - 1];\n const dist = this.distance(p0, p1) + this.distance(p1, p);\n const steps = Math.max(2, Math.min(MAX_BEZIER_STEPS, dist / BEZIER_STEP_SIZE));\n for (let i = 1; i <= steps; ++i) {\n const t = i / steps;\n this.points.push(this.lerp(this.lerp(p0, p1, t), this.lerp(p1, p, t), t));\n }\n }\n cubicTo(p, p1, p2) {\n const p0 = this.points[this.points.length - 1];\n const dist = this.distance(p0, p1) + this.distance(p1, p2) + this.distance(p2, p);\n const steps = Math.max(2, Math.min(MAX_BEZIER_STEPS, dist / BEZIER_STEP_SIZE));\n for (let i = 1; i <= steps; ++i) {\n const t = i / steps;\n const a = this.lerp(this.lerp(p0, p1, t), this.lerp(p1, p2, t), t);\n const b = this.lerp(this.lerp(p1, p2, t), this.lerp(p2, p, t), t);\n this.points.push(this.lerp(a, b, t));\n }\n }\n inside(p) {\n const epsilon = 1e-6;\n let count = 0, cur = this.points[this.points.length - 1];\n this.points.forEach(next => {\n const p0 = cur.y < next.y ? cur : next;\n const p1 = cur.y < next.y ? next : cur;\n if (p0.y < p.y + epsilon && p1.y > p.y + epsilon) {\n if ((p1.x - p0.x) * (p.y - p0.y) > (p.x - p0.x) * (p1.y - p0.y)) {\n count++;\n }\n }\n cur = next;\n });\n return count % 2 !== 0;\n }\n }\n class TextMeshFont {\n constructor(name, font, scene) {\n this.font = font;\n this.scene = scene;\n this.glyphs = {};\n this.font = font;\n this.glyphsParent = new Mesh(name, scene);\n }\n createGlyph(ch) {\n if (this.glyphs[ch]) {\n return this.glyphs[ch];\n }\n const glyph = this.font.charToGlyph(ch);\n if (!glyph || !glyph.advanceWidth) {\n return undefined;\n }\n this.glyphs[ch] = {\n index: glyph.index,\n advanceWidth: glyph.advanceWidth,\n boundingBox: glyph.getBoundingBox(),\n };\n const glyphDef = this.glyphs[ch];\n if (!glyph.path || !glyph.path.commands || !glyph.path.commands.length) {\n return glyphDef;\n }\n const polys = [];\n glyph.path.commands.forEach(command => {\n const { type, x, y, x1, y1, x2, y2 } = command;\n switch (type) {\n case 'M':\n polys.push(new TextMeshPolygon());\n polys[polys.length - 1].moveTo({ x, y });\n break;\n case 'L':\n polys[polys.length - 1].moveTo({ x, y });\n break;\n case 'C':\n polys[polys.length - 1].cubicTo({ x, y }, { x: x1, y: y1 }, { x: x2, y: y2 });\n break;\n case 'Q':\n polys[polys.length - 1].conicTo({ x, y }, { x: x1, y: y1 });\n break;\n case 'Z':\n polys[polys.length - 1].close();\n break;\n }\n });\n // sort contours by descending area\n polys.sort((a, b) => Math.abs(b.area) - Math.abs(a.area));\n // classify contours to find holes and their 'parents'\n const root = [];\n for (let i = 0; i < polys.length; ++i) {\n let parent = null;\n for (let j = i - 1; j >= 0; --j) {\n // a contour is a hole if it is inside its parent and has different winding\n if (polys[j].inside(polys[i].points[0]) && polys[i].area * polys[j].area < 0) {\n parent = polys[j];\n break;\n }\n }\n if (parent) {\n parent.children.push(polys[i]);\n }\n else {\n root.push(polys[i]);\n }\n }\n const totalPoints = polys.reduce((sum, p) => sum + p.points.length, 0);\n const vertexData = new Float32Array(totalPoints * 2);\n let vertexCount = 0;\n let pathSegmentCount = 0;\n const indices = [];\n const paths = [];\n const holePaths = [];\n function process(poly) {\n // construct input for earcut\n const coords = [];\n const holes = [];\n const n = poly.points.length - 1;\n for (let i = 0; i < n; ++i) {\n const { x, y } = poly.points[n - i];\n if (i > 0) {\n const idx = coords.length / 2;\n paths.push(vertexCount + idx - 1, vertexCount + idx);\n }\n coords.push(x, y);\n }\n const parentSegmentIdx = pathSegmentCount;\n poly.children.forEach((child, i) => {\n // children's children are new, separate shapes\n child.children.forEach(process);\n // idx 0 of a hole is the segment index of the segment the hole is inside. each separate poly is a separate segment\n holePaths.push(parentSegmentIdx, paths.length);\n holes.push(coords.length / 2);\n const n = child.points.length - 1;\n for (let i = 0; i < n; ++i) {\n const { x, y } = child.points[i];\n if (i > 0) {\n const idx = coords.length / 2;\n paths.push(idx - 1, idx);\n }\n coords.push(x, y);\n }\n pathSegmentCount++;\n });\n // add vertex data\n vertexData.set(coords, vertexCount * 2);\n // add index data\n earcut$1(coords, holes).forEach(i => {\n indices.push(i + vertexCount);\n });\n vertexCount += coords.length / 2;\n pathSegmentCount++;\n }\n root.forEach(process);\n const meshdata = new SubmeshVertexData();\n const vertices = [];\n const normals = [];\n for (let i = 0; i < vertexCount; i++) {\n vertices.push(vertexData[i * 2] * GLYPH_COORDS_SCALE);\n vertices.push(vertexData[i * 2 + 1] * GLYPH_COORDS_SCALE);\n vertices.push(0);\n normals.push(0);\n normals.push(0);\n normals.push(1);\n }\n const flipIndices = [];\n for (let i = 0; i < indices.length; i += 3) {\n flipIndices.push(indices[i], indices[i + 2], indices[i + 1]);\n }\n meshdata.positions = vertices;\n meshdata.indices = flipIndices;\n meshdata.normals = normals;\n meshdata.paths = paths;\n meshdata.holes = holePaths;\n glyphDef.vertexData = meshdata;\n return glyphDef;\n }\n }\n class TextMeshVertexData {\n constructor(textMeshFont) {\n this.textMeshFont = textMeshFont;\n this.instances = {};\n }\n getText(text, fontSize = 1, charSpacing = 1, radius = 0) {\n radius = radius / fontSize;\n const scaleMatrix = Matrix.Compose(new Vector3(fontSize, fontSize, 1), IDENTITY_Q, Vector3.ZeroReadOnly);\n const vertexData = new SubmeshVertexData();\n const pos = Vector3.Zero();\n const transformMatrix = Matrix.Identity();\n const generatedVertices = [];\n for (let i = 0; i < text.length; i++) {\n const ch1 = text[i];\n if (ch1 === '\\n') {\n pos.x = 0;\n pos.y -= 1.1;\n continue;\n }\n transformMatrix.setTranslation(pos);\n const ch2 = text[i + 1];\n const g = this.textMeshFont.createGlyph(ch1);\n if (!g) {\n continue;\n }\n let advance = g.advanceWidth;\n if (g.vertexData) {\n generatedVertices.push({\n data: g.vertexData.deepClone(),\n transform: transformMatrix.clone(),\n center: (advance / 2 + g.boundingBox.x1) * GLYPH_COORDS_SCALE,\n });\n }\n if (advance) {\n if (ch2 && this.textMeshFont.glyphs[ch2]) {\n const kern = this.textMeshFont.font.getKerningValue(g.index, this.textMeshFont.glyphs[ch2].index);\n if (kern) {\n advance += kern;\n }\n }\n pos.x += (advance - g.boundingBox.x2 + g.boundingBox.x2 * charSpacing) * GLYPH_COORDS_SCALE;\n }\n }\n if (generatedVertices.length > 0) {\n const centerTransform = Matrix.Compose(Vector3.OneReadOnly, IDENTITY_Q, pos.scale(-0.5));\n for (const generatedVertexData of generatedVertices) {\n const transform = generatedVertexData.transform;\n transform.multiplyToRef(centerTransform, transform);\n // Add curve to text\n if (radius !== 0) {\n const baseOffset = transform.getTranslation();\n const angle = (baseOffset.x + generatedVertexData.center) / radius;\n const pivotMatrix = Matrix.Compose(Vector3.OneReadOnly, IDENTITY_Q, new Vector3(generatedVertexData.center, -radius, 0));\n const rotMatrix = Matrix.Compose(Vector3.OneReadOnly, Quaternion.RotationAxis(new Vector3(0, 0, -1), angle), Vector3.ZeroReadOnly);\n const curveTransform = pivotMatrix.clone().invert();\n curveTransform.multiplyToRef(rotMatrix, curveTransform);\n curveTransform.multiplyToRef(pivotMatrix, curveTransform);\n curveTransform.setTranslation(curveTransform.getTranslation().addInPlaceFromFloats(-generatedVertexData.center, 0, 0));\n transform.copyFrom(curveTransform);\n }\n transform.multiplyToRef(scaleMatrix, transform);\n generatedVertexData.data.transform(transform);\n }\n vertexData.copyFrom(generatedVertices[0].data);\n if (generatedVertices.length > 1) {\n vertexData.merge(generatedVertices.slice(1).map(generatedVertexData => generatedVertexData.data));\n }\n }\n // const calcW = this.textMeshFont.font.getAdvanceWidth(text) * GLYPH_COORDS_SCALE;\n // if (vertexData.positions) {\n // const max = Vector3.Zero();\n // for (let i = 0; i < vertexData.positions.length; i += 3) {\n // const p = vertexData.positions;\n // max.x = Math.max(max.x, p[i]);\n // max.y = Math.max(max.y, p[i + 1]);\n // max.z = Math.max(max.z, p[i + 2]);\n // }\n // const centerTransform = Matrix.Compose(Vector3.OneReadOnly, Quaternion.Identity(), max.scale(-0.5));\n // vertexData.transform(centerTransform);\n // }\n return vertexData;\n }\n }\n\n /**\n * Creates a hemisphere mesh\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param options.segments\n * @param options.diameter\n * @param options.sideOrientation\n * @param scene defines the hosting scene\n * @returns the hemisphere mesh\n */\n function CreateHemisphere(name, options = {}, scene) {\n if (!options.diameter) {\n options.diameter = 1;\n }\n if (!options.segments) {\n options.segments = 16;\n }\n const halfSphere = CreateSphere(\"\", { slice: 0.5, diameter: options.diameter, segments: options.segments }, scene);\n const disc = CreateDisc(\"\", { radius: options.diameter / 2, tessellation: options.segments * 3 + (4 - options.segments) }, scene);\n disc.rotation.x = -Math.PI / 2;\n disc.parent = halfSphere;\n const merged = Mesh.MergeMeshes([disc, halfSphere], true);\n merged.name = name;\n return merged;\n }\n /**\n * Creates a hemispheric light\n * @param name\n * @param segments\n * @param diameter\n * @param scene\n */\n Mesh.CreateHemisphere = (name, segments, diameter, scene) => {\n const options = {\n segments: segments,\n diameter: diameter,\n };\n return CreateHemisphere(name, options, scene);\n };\n\n /**\n * Material types for GreasedLine\n * {@link https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param/greased_line#materialtype}\n */\n var GreasedLineMeshMaterialType;\n (function (GreasedLineMeshMaterialType) {\n /**\n * StandardMaterial\n */\n GreasedLineMeshMaterialType[GreasedLineMeshMaterialType[\"MATERIAL_TYPE_STANDARD\"] = 0] = \"MATERIAL_TYPE_STANDARD\";\n /**\n * PBR Material\n */\n GreasedLineMeshMaterialType[GreasedLineMeshMaterialType[\"MATERIAL_TYPE_PBR\"] = 1] = \"MATERIAL_TYPE_PBR\";\n })(GreasedLineMeshMaterialType || (GreasedLineMeshMaterialType = {}));\n /**\n * Color blending mode of the @see GreasedLineMaterial and the base material\n * {@link https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param/greased_line#colormode}\n */\n var GreasedLineMeshColorMode;\n (function (GreasedLineMeshColorMode) {\n /**\n * Color blending mode SET\n */\n GreasedLineMeshColorMode[GreasedLineMeshColorMode[\"COLOR_MODE_SET\"] = 0] = \"COLOR_MODE_SET\";\n /**\n * Color blending mode ADD\n */\n GreasedLineMeshColorMode[GreasedLineMeshColorMode[\"COLOR_MODE_ADD\"] = 1] = \"COLOR_MODE_ADD\";\n /**\n * Color blending mode ADD\n */\n GreasedLineMeshColorMode[GreasedLineMeshColorMode[\"COLOR_MODE_MULTIPLY\"] = 2] = \"COLOR_MODE_MULTIPLY\";\n })(GreasedLineMeshColorMode || (GreasedLineMeshColorMode = {}));\n /**\n * Color distribution type of the @see colors.\n * {@link https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param/greased_line#colordistributiontype}\n *\n */\n var GreasedLineMeshColorDistributionType;\n (function (GreasedLineMeshColorDistributionType) {\n /**\n * Colors distributed between segments of the line\n */\n GreasedLineMeshColorDistributionType[GreasedLineMeshColorDistributionType[\"COLOR_DISTRIBUTION_TYPE_SEGMENT\"] = 0] = \"COLOR_DISTRIBUTION_TYPE_SEGMENT\";\n /**\n * Colors distributed along the line ingoring the segments\n */\n GreasedLineMeshColorDistributionType[GreasedLineMeshColorDistributionType[\"COLOR_DISTRIBUTION_TYPE_LINE\"] = 1] = \"COLOR_DISTRIBUTION_TYPE_LINE\";\n })(GreasedLineMeshColorDistributionType || (GreasedLineMeshColorDistributionType = {}));\n /**\n * @internal\n */\n class MaterialGreasedLineDefines extends MaterialDefines {\n constructor() {\n super(...arguments);\n /**\n * The material has a color option specified\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n this.GREASED_LINE_HAS_COLOR = false;\n /**\n * The material's size attenuation optiom\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n this.GREASED_LINE_SIZE_ATTENUATION = false;\n /**\n * The type of color distribution is set to line this value equals to true.\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n this.GREASED_LINE_COLOR_DISTRIBUTION_TYPE_LINE = false;\n /**\n * True if scene is in right handed coordinate system.\n */\n // eslint-disable-next-line @typescript-eslint/naming-convention\n this.GREASED_LNE_RIGHT_HANDED_COORDINATE_SYSTEM = false;\n }\n }\n /**\n * GreasedLinePluginMaterial for GreasedLineMesh\n */\n class GreasedLinePluginMaterial extends MaterialPluginBase {\n constructor(material, _scene, options) {\n var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m, _o, _p, _q, _r;\n options = options || {\n color: GreasedLinePluginMaterial.DEFAULT_COLOR,\n };\n const defines = new MaterialGreasedLineDefines();\n defines.GREASED_LINE_HAS_COLOR = !!options.color;\n defines.GREASED_LINE_SIZE_ATTENUATION = (_a = options.sizeAttenuation) !== null && _a !== void 0 ? _a : false;\n defines.GREASED_LINE_COLOR_DISTRIBUTION_TYPE_LINE = options.colorDistributionType === GreasedLineMeshColorDistributionType.COLOR_DISTRIBUTION_TYPE_LINE;\n defines.GREASED_LNE_RIGHT_HANDED_COORDINATE_SYSTEM = _scene.useRightHandedSystem;\n super(material, GreasedLinePluginMaterial.GREASED_LINE_MATERIAL_NAME, 200, defines);\n this._scene = _scene;\n this._scene = (_b = this._scene) !== null && _b !== void 0 ? _b : material.getScene();\n this._engine = this._scene.getEngine();\n this.visibility = (_c = options.visibility) !== null && _c !== void 0 ? _c : 1;\n this.useDash = (_d = options.useDash) !== null && _d !== void 0 ? _d : false;\n this.dashRatio = (_e = options.dashRatio) !== null && _e !== void 0 ? _e : 0.5;\n this.dashOffset = (_f = options.dashOffset) !== null && _f !== void 0 ? _f : 0;\n this.width = options.width ? options.width : options.sizeAttenuation ? GreasedLinePluginMaterial.DEFAULT_WIDTH_ATTENUATED : GreasedLinePluginMaterial.DEFAULT_WIDTH;\n this._sizeAttenuation = (_g = options.sizeAttenuation) !== null && _g !== void 0 ? _g : false;\n this.colorMode = (_h = options.colorMode) !== null && _h !== void 0 ? _h : GreasedLineMeshColorMode.COLOR_MODE_SET;\n this._color = (_j = options.color) !== null && _j !== void 0 ? _j : null;\n this.useColors = (_k = options.useColors) !== null && _k !== void 0 ? _k : false;\n this._colorsDistributionType = (_l = options.colorDistributionType) !== null && _l !== void 0 ? _l : GreasedLineMeshColorDistributionType.COLOR_DISTRIBUTION_TYPE_SEGMENT;\n this.colorsSampling = (_m = options.colorsSampling) !== null && _m !== void 0 ? _m : RawTexture.NEAREST_NEAREST;\n this._colors = (_o = options.colors) !== null && _o !== void 0 ? _o : null;\n this.dashCount = (_p = options.dashCount) !== null && _p !== void 0 ? _p : 1; // calculate the _dashArray value, call the setter\n this.resolution = (_q = options.resolution) !== null && _q !== void 0 ? _q : new Vector2(this._engine.getRenderWidth(), this._engine.getRenderHeight()); // calculate aspect call the setter\n if (this._colors) {\n this._createColorsTexture(`${material.name}-colors-texture`, this._colors);\n }\n else {\n this._color = (_r = this._color) !== null && _r !== void 0 ? _r : GreasedLinePluginMaterial.DEFAULT_COLOR;\n GreasedLinePluginMaterial._PrepareEmptyColorsTexture(_scene);\n }\n this._enable(true); // always enabled\n }\n /**\n * Get the shader attributes\n * @param attributes array which will be filled with the attributes\n */\n getAttributes(attributes) {\n attributes.push(\"grl_offsets\");\n attributes.push(\"grl_previousAndSide\");\n attributes.push(\"grl_nextAndCounters\");\n attributes.push(\"grl_widths\");\n attributes.push(\"grl_colorPointers\");\n }\n /**\n * Get the shader samplers\n * @param samplers\n */\n getSamplers(samplers) {\n samplers.push(\"grl_colors\");\n }\n /**\n * Get the shader textures\n * @param activeTextures\n */\n getActiveTextures(activeTextures) {\n if (this._colorsTexture) {\n activeTextures.push(this._colorsTexture);\n }\n }\n /**\n * Get the shader uniforms\n * @returns uniforms\n */\n getUniforms() {\n const ubo = [\n { name: \"grl_projection\", size: 16, type: \"mat4\" },\n { name: \"grl_singleColor\", size: 3, type: \"vec3\" },\n { name: \"grl_aspect_resolution_lineWidth\", size: 4, type: \"vec4\" },\n { name: \"grl_dashOptions\", size: 4, type: \"vec4\" },\n { name: \"grl_colorMode_visibility_colorsWidth_useColors\", size: 4, type: \"vec4\" },\n ];\n return {\n ubo,\n vertex: `\r\n uniform vec4 grl_aspect_resolution_lineWidth;\r\n uniform mat4 grl_projection;\r\n `,\n fragment: `\r\n uniform vec4 grl_dashOptions;\r\n uniform vec4 grl_colorMode_visibility_colorsWidth_useColors;\r\n uniform vec3 grl_singleColor;\r\n `,\n };\n }\n // only getter, it doesn't make sense to use this plugin on a mesh other than GreasedLineMesh\n // and it doesn't make sense to disable it on the mesh\n get isEnabled() {\n return true;\n }\n /**\n * Bind the uniform buffer\n * @param uniformBuffer\n */\n bindForSubMesh(uniformBuffer) {\n var _a;\n const activeCamera = this._scene.activeCamera;\n if (activeCamera) {\n const projection = activeCamera.getProjectionMatrix();\n uniformBuffer.updateMatrix(\"grl_projection\", projection);\n }\n else {\n throw Error(\"GreasedLinePluginMaterial requires an active camera.\");\n }\n const resolutionLineWidth = TmpVectors.Vector4[0];\n resolutionLineWidth.x = this._aspect;\n resolutionLineWidth.y = this._resolution.x;\n resolutionLineWidth.z = this._resolution.y;\n resolutionLineWidth.w = this.width;\n uniformBuffer.updateVector4(\"grl_aspect_resolution_lineWidth\", resolutionLineWidth);\n const dashOptions = TmpVectors.Vector4[0];\n dashOptions.x = GreasedLinePluginMaterial._BooleanToNumber(this.useDash);\n dashOptions.y = this._dashArray;\n dashOptions.z = this.dashOffset;\n dashOptions.w = this.dashRatio;\n uniformBuffer.updateVector4(\"grl_dashOptions\", dashOptions);\n const colorModeVisibilityColorsWidthUseColors = TmpVectors.Vector4[1];\n colorModeVisibilityColorsWidthUseColors.x = this.colorMode;\n colorModeVisibilityColorsWidthUseColors.y = this.visibility;\n colorModeVisibilityColorsWidthUseColors.z = this._colorsTexture ? this._colorsTexture.getSize().width : 0;\n colorModeVisibilityColorsWidthUseColors.w = GreasedLinePluginMaterial._BooleanToNumber(this.useColors);\n uniformBuffer.updateVector4(\"grl_colorMode_visibility_colorsWidth_useColors\", colorModeVisibilityColorsWidthUseColors);\n if (this._color) {\n uniformBuffer.updateColor3(\"grl_singleColor\", this._color);\n }\n uniformBuffer.setTexture(\"grl_colors\", (_a = this._colorsTexture) !== null && _a !== void 0 ? _a : GreasedLinePluginMaterial._EmptyColorsTexture);\n }\n /**\n * Prepare the defines\n * @param defines\n * @param _scene\n * @param _mesh\n */\n prepareDefines(defines, _scene, _mesh) {\n var _a;\n defines.GREASED_LINE_HAS_COLOR = !!this._color;\n defines.GREASED_LINE_SIZE_ATTENUATION = (_a = this._sizeAttenuation) !== null && _a !== void 0 ? _a : false;\n defines.GREASED_LINE_COLOR_DISTRIBUTION_TYPE_LINE = this._colorsDistributionType === GreasedLineMeshColorDistributionType.COLOR_DISTRIBUTION_TYPE_LINE;\n defines.GREASED_LNE_RIGHT_HANDED_COORDINATE_SYSTEM = _scene.useRightHandedSystem;\n }\n /**\n * Get the class name\n * @returns class name\n */\n getClassName() {\n return GreasedLinePluginMaterial.GREASED_LINE_MATERIAL_NAME;\n }\n /**\n * Get shader code\n * @param shaderType vertex/fragment\n * @returns shader code\n */\n getCustomCode(shaderType) {\n if (shaderType === \"vertex\") {\n return {\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CUSTOM_VERTEX_DEFINITIONS: `\r\n attribute vec4 grl_previousAndSide;\r\n attribute vec4 grl_nextAndCounters;\r\n attribute float grl_widths;\r\n attribute vec3 grl_offsets;\r\n attribute float grl_colorPointers;\r\n\r\n varying float grlCounters;\r\n varying float grlColorPointer;\r\n\r\n vec2 grlFix( vec4 i, float aspect ) {\r\n vec2 res = i.xy / i.w;\r\n res.x *= aspect;\r\n return res;\r\n }\r\n `,\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CUSTOM_VERTEX_UPDATE_POSITION: `\r\n vec3 grlPositionOffset = grl_offsets;\r\n positionUpdated += grlPositionOffset;\r\n `,\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CUSTOM_VERTEX_MAIN_END: `\r\n\r\n float grlAspect = grl_aspect_resolution_lineWidth.x;\r\n float grlBaseWidth = grl_aspect_resolution_lineWidth.w;\r\n\r\n grlColorPointer = grl_colorPointers;\r\n\r\n vec3 grlPrevious = grl_previousAndSide.xyz;\r\n float grlSide = grl_previousAndSide.w;\r\n\r\n vec3 grlNext = grl_nextAndCounters.xyz;\r\n grlCounters = grl_nextAndCounters.w;\r\n\r\n\r\n mat4 grlMatrix = viewProjection * world;\r\n vec4 grlFinalPosition = grlMatrix * vec4( positionUpdated , 1.0 );\r\n vec4 grlPrevPos = grlMatrix * vec4( grlPrevious + grlPositionOffset, 1.0 );\r\n vec4 grlNextPos = grlMatrix * vec4( grlNext + grlPositionOffset, 1.0 );\r\n\r\n vec2 grlCurrentP = grlFix( grlFinalPosition, grlAspect );\r\n vec2 grlPrevP = grlFix( grlPrevPos, grlAspect );\r\n vec2 grlNextP = grlFix( grlNextPos, grlAspect );\r\n\r\n float grlWidth = grlBaseWidth * grl_widths;\r\n\r\n vec2 grlDir;\r\n if( grlNextP == grlCurrentP ) grlDir = normalize( grlCurrentP - grlPrevP );\r\n else if( grlPrevP == grlCurrentP ) grlDir = normalize( grlNextP - grlCurrentP );\r\n else {\r\n vec2 grlDir1 = normalize( grlCurrentP - grlPrevP );\r\n vec2 grlDir2 = normalize( grlNextP - grlCurrentP );\r\n grlDir = normalize( grlDir1 + grlDir2 );\r\n }\r\n vec4 grlNormal = vec4( -grlDir.y, grlDir.x, 0., 1. );\r\n #ifdef GREASED_LNE_RIGHT_HANDED_COORDINATE_SYSTEM\r\n grlNormal.xy *= -.5 * grlWidth;\r\n #else\r\n grlNormal.xy *= .5 * grlWidth;\r\n #endif\r\n grlNormal *= grl_projection;\r\n #ifdef GREASED_LINE_SIZE_ATTENUATION\r\n grlNormal.xy *= grlFinalPosition.w;\r\n grlNormal.xy /= ( vec4( grl_aspect_resolution_lineWidth.yz, 0., 1. ) * grl_projection ).xy;\r\n #endif\r\n grlFinalPosition.xy += grlNormal.xy * grlSide;\r\n gl_Position = grlFinalPosition;\r\n\r\n vPositionW = vec3(grlFinalPosition);\r\n\r\n `,\n // eslint-disable-next-line @typescript-eslint/naming-convention\n \"!gl_Position\\\\=viewProjection\\\\*worldPos;\": \"//\", // remove\n };\n }\n if (shaderType === \"fragment\") {\n return {\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CUSTOM_FRAGMENT_DEFINITIONS: `\r\n varying float grlCounters;\r\n varying float grlColorPointer;\r\n uniform sampler2D grl_colors;\r\n `,\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CUSTOM_FRAGMENT_MAIN_END: `\r\n float grlColorMode = grl_colorMode_visibility_colorsWidth_useColors.x;\r\n float grlVisibility = grl_colorMode_visibility_colorsWidth_useColors.y;\r\n float grlColorsWidth = grl_colorMode_visibility_colorsWidth_useColors.z;\r\n float grlUseColors = grl_colorMode_visibility_colorsWidth_useColors.w;\r\n\r\n float grlUseDash = grl_dashOptions.x;\r\n float grlDashArray = grl_dashOptions.y;\r\n float grlDashOffset = grl_dashOptions.z;\r\n float grlDashRatio = grl_dashOptions.w;\r\n\r\n gl_FragColor.a *= step(grlCounters, grlVisibility);\r\n if( gl_FragColor.a == 0. ) discard;\r\n\r\n if(grlUseDash == 1.){\r\n gl_FragColor.a *= ceil(mod(grlCounters + grlDashOffset, grlDashArray) - (grlDashArray * grlDashRatio));\r\n if (gl_FragColor.a == 0.) discard;\r\n }\r\n\r\n #ifdef GREASED_LINE_HAS_COLOR\r\n if (grlColorMode == ${GreasedLineMeshColorMode.COLOR_MODE_SET}.) {\r\n gl_FragColor.rgb = grl_singleColor;\r\n } else if (grlColorMode == ${GreasedLineMeshColorMode.COLOR_MODE_ADD}.) {\r\n gl_FragColor.rgb += grl_singleColor;\r\n } else if (grlColorMode == ${GreasedLineMeshColorMode.COLOR_MODE_MULTIPLY}.) {\r\n gl_FragColor.rgb *= grl_singleColor;\r\n }\r\n #else\r\n if (grlUseColors == 1.) {\r\n #ifdef GREASED_LINE_COLOR_DISTRIBUTION_TYPE_LINE\r\n vec4 grlColor = texture2D(grl_colors, vec2(grlCounters, 0.), 0.);\r\n #else\r\n vec4 grlColor = texture2D(grl_colors, vec2(grlColorPointer/grlColorsWidth, 0.), 0.);\r\n #endif\r\n if (grlColorMode == ${GreasedLineMeshColorMode.COLOR_MODE_SET}.) {\r\n gl_FragColor = grlColor;\r\n } else if (grlColorMode == ${GreasedLineMeshColorMode.COLOR_MODE_ADD}.) {\r\n gl_FragColor += grlColor;\r\n } else if (grlColorMode == ${GreasedLineMeshColorMode.COLOR_MODE_MULTIPLY}.) {\r\n gl_FragColor *= grlColor;\r\n }\r\n }\r\n #endif\r\n `,\n };\n }\n return null;\n }\n /**\n * Converts boolean to number.\n * @param bool\n * @returns 1 if true, 0 if false.\n */\n static _BooleanToNumber(bool) {\n return bool ? 1 : 0;\n }\n /**\n * Converts an array of Color3 to Uint8Array\n * @param colors Arrray of Color3\n * @returns Uin8Array of colors [r, g, b, a, r, g, b, a, ...]\n */\n static _Color3toRGBAUint8(colors) {\n const colorTable = new Uint8Array(colors.length * 4);\n for (let i = 0, j = 0; i < colors.length; i++) {\n colorTable[j++] = colors[i].r * 255;\n colorTable[j++] = colors[i].g * 255;\n colorTable[j++] = colors[i].b * 255;\n colorTable[j++] = 255;\n }\n return colorTable;\n }\n /**\n * Creates a RawTexture from an RGBA color array and sets it on the plugin material instance.\n * @param name name of the texture\n * @param colors Uint8Array of colors\n */\n _createColorsTexture(name, colors) {\n const colorsArray = GreasedLinePluginMaterial._Color3toRGBAUint8(colors);\n this._colorsTexture = new RawTexture(colorsArray, colors.length, 1, Engine.TEXTUREFORMAT_RGBA, this._scene, false, true, this.colorsSampling);\n this._colorsTexture.name = name;\n }\n /**\n * Disposes the plugin material.\n */\n dispose() {\n var _a;\n (_a = this._colorsTexture) === null || _a === void 0 ? void 0 : _a.dispose();\n super.dispose();\n }\n /**\n * Returns the colors used to colorize the line\n */\n get colors() {\n return this._colors;\n }\n /**\n * Sets the colors used to colorize the line\n */\n set colors(value) {\n this.setColors(value);\n }\n /**\n * Creates or updates the colors texture\n * @param colors color table RGBA\n * @param lazy if lazy, the colors are not updated\n * @param forceUpdate force creation of a new texture\n * @returns\n */\n setColors(colors, lazy = false, forceUpdate = false) {\n var _a, _b, _c, _d;\n const origColorsCount = (_b = (_a = this._colors) === null || _a === void 0 ? void 0 : _a.length) !== null && _b !== void 0 ? _b : 0;\n this._colors = colors;\n if (colors === null || colors.length === 0) {\n (_c = this._colorsTexture) === null || _c === void 0 ? void 0 : _c.dispose();\n return;\n }\n if (lazy && !forceUpdate) {\n return;\n }\n if (this._colorsTexture && origColorsCount === colors.length && !forceUpdate) {\n const colorArray = GreasedLinePluginMaterial._Color3toRGBAUint8(colors);\n this._colorsTexture.update(colorArray);\n }\n else {\n (_d = this._colorsTexture) === null || _d === void 0 ? void 0 : _d.dispose();\n this._createColorsTexture(`${this._material.name}-colors-texture`, colors);\n }\n }\n /**\n * Updates the material. Use when material created in lazy mode.\n */\n updateLazy() {\n if (this._colors) {\n this.setColors(this._colors, false, true);\n }\n }\n /**\n * Gets the number of dashes in the line\n */\n get dashCount() {\n return this._dashCount;\n }\n /**\n * Sets the number of dashes in the line\n * @param value dash\n */\n set dashCount(value) {\n this._dashCount = value;\n this._dashArray = 1 / value;\n }\n /**\n * False means 1 unit in width = 1 unit on scene, true means 1 unit in width is reduced on the screen to make better looking lines\n */\n get sizeAttenuation() {\n return this._sizeAttenuation;\n }\n /**\n * Turn on/off attenuation of the width option and widths array.\n * @param value false means 1 unit in width = 1 unit on scene, true means 1 unit in width is reduced on the screen to make better looking lines\n */\n set sizeAttenuation(value) {\n this._sizeAttenuation = value;\n this.markAllDefinesAsDirty();\n }\n /**\n * Gets the color of the line\n */\n get color() {\n return this.color;\n }\n /**\n * Sets the color of the line\n * @param value Color3 or null to clear the color. You need to clear the color if you use colors and useColors = true\n */\n set color(value) {\n this.setColor(value);\n }\n /**\n * Sets the color of the line. If set the whole line will be mixed with this color according to the colorMode option.\n * @param value color\n */\n setColor(value, doNotMarkDirty = false) {\n if ((this._color === null && value !== null) || (this._color !== null && value === null)) {\n this._color = value;\n !doNotMarkDirty && this.markAllDefinesAsDirty();\n }\n else {\n this._color = value;\n }\n }\n /**\n * Gets the color distributiopn type\n */\n get colorsDistributionType() {\n return this._colorsDistributionType;\n }\n /**\n * Sets the color distribution type\n * @see GreasedLineMeshColorDistributionType\n * @param value color distribution type\n */\n set colorsDistributionType(value) {\n this._colorsDistributionType = value;\n this.markAllDefinesAsDirty();\n }\n /**\n * Gets the resolution\n */\n get resolution() {\n return this._resolution;\n }\n /**\n * Sets the resolution\n * @param value resolution of the screen for GreasedLine\n */\n set resolution(value) {\n this._aspect = value.x / value.y;\n this._resolution = value;\n }\n /**\n * Serializes this plugin material\n * @returns serializationObjec\n */\n serialize() {\n const serializationObject = super.serialize();\n const greasedLineMaterialOptions = {\n colorDistributionType: this._colorsDistributionType,\n colorsSampling: this.colorsSampling,\n colorMode: this.colorMode,\n dashCount: this._dashCount,\n dashOffset: this.dashOffset,\n dashRatio: this.dashRatio,\n resolution: this._resolution,\n sizeAttenuation: this._sizeAttenuation,\n useColors: this.useColors,\n useDash: this.useDash,\n visibility: this.visibility,\n width: this.width,\n };\n this._colors && (greasedLineMaterialOptions.colors = this._colors);\n this._color && (greasedLineMaterialOptions.color = this._color);\n serializationObject.greasedLineMaterialOptions = greasedLineMaterialOptions;\n return serializationObject;\n }\n /**\n * Parses a serialized objects\n * @param source serialized object\n * @param scene scene\n * @param rootUrl root url for textures\n */\n parse(source, scene, rootUrl) {\n var _a;\n super.parse(source, scene, rootUrl);\n const greasedLineMaterialOptions = source.greasedLineMaterialOptions;\n (_a = this._colorsTexture) === null || _a === void 0 ? void 0 : _a.dispose();\n if (greasedLineMaterialOptions.colors) {\n this._createColorsTexture(`${this._material.name}-colors-texture`, greasedLineMaterialOptions.colors);\n }\n else {\n GreasedLinePluginMaterial._PrepareEmptyColorsTexture(scene);\n }\n greasedLineMaterialOptions.color && this.setColor(greasedLineMaterialOptions.color, true);\n greasedLineMaterialOptions.colorDistributionType && (this.colorsDistributionType = greasedLineMaterialOptions.colorDistributionType);\n greasedLineMaterialOptions.colorsSampling && (this.colorsSampling = greasedLineMaterialOptions.colorsSampling);\n greasedLineMaterialOptions.colorMode && (this.colorMode = greasedLineMaterialOptions.colorMode);\n greasedLineMaterialOptions.useColors && (this.useColors = greasedLineMaterialOptions.useColors);\n greasedLineMaterialOptions.visibility && (this.visibility = greasedLineMaterialOptions.visibility);\n greasedLineMaterialOptions.useDash && (this.useDash = greasedLineMaterialOptions.useDash);\n greasedLineMaterialOptions.dashCount && (this.dashCount = greasedLineMaterialOptions.dashCount);\n greasedLineMaterialOptions.dashRatio && (this.dashRatio = greasedLineMaterialOptions.dashRatio);\n greasedLineMaterialOptions.dashOffset && (this.dashOffset = greasedLineMaterialOptions.dashOffset);\n greasedLineMaterialOptions.width && (this.width = greasedLineMaterialOptions.width);\n greasedLineMaterialOptions.sizeAttenuation && (this.sizeAttenuation = greasedLineMaterialOptions.sizeAttenuation);\n greasedLineMaterialOptions.resolution && (this.resolution = greasedLineMaterialOptions.resolution);\n this.markAllDefinesAsDirty();\n }\n /**\n * A minimum size texture for the colors sampler2D when there is no colors texture defined yet.\n * For fast switching using the useColors property without the need to use defines.\n * @param scene Scene\n */\n static _PrepareEmptyColorsTexture(scene) {\n if (!this._EmptyColorsTexture) {\n const colorsArray = new Uint8Array(4);\n this._EmptyColorsTexture = new RawTexture(colorsArray, 1, 1, Engine.TEXTUREFORMAT_RGBA, scene, false, false, RawTexture.NEAREST_NEAREST);\n this._EmptyColorsTexture.name = \"grlEmptyColorsTexture\";\n }\n }\n }\n /**\n * Plugin name\n */\n GreasedLinePluginMaterial.GREASED_LINE_MATERIAL_NAME = \"GreasedLinePluginMaterial\";\n /**\n * Default line color for newly created lines\n */\n GreasedLinePluginMaterial.DEFAULT_COLOR = Color3.White();\n /**\n * Default line width when sizeAttenuation is true\n */\n GreasedLinePluginMaterial.DEFAULT_WIDTH_ATTENUATED = 1;\n /**\n * Defaule line width\n */\n GreasedLinePluginMaterial.DEFAULT_WIDTH = 0.1;\n RegisterClass(`BABYLON.${GreasedLinePluginMaterial.GREASED_LINE_MATERIAL_NAME}`, GreasedLinePluginMaterial);\n\n /**\n * Tool functions for GreasedLine\n */\n class GreasedLineTools {\n /**\n * Omit zero length lines predicate for the MeshesToLines function\n * @param p1 point1 position of the face\n * @param p2 point2 position of the face\n * @param p3 point3 position of the face\n * @returns\n */\n static OmitZeroLengthPredicate(p1, p2, p3) {\n return p1.lengthSquared() === 0 && p2.lengthSquared() === 0 && p3.lengthSquared() === 0;\n }\n /**\n * Gets mesh triangles as line positions\n * @param meshes array of meshes\n * @param omitZeroLengthLines do not generate a line when the distance if the vertices in the triangle equals to zero\n * @returns array of arrays of points\n */\n static MeshesToLines(meshes, predicate) {\n const points = [];\n meshes.forEach((m, meshIndex) => {\n const vertices = m.getVerticesData(VertexBuffer.PositionKind);\n const indices = m.getIndices();\n if (vertices && indices) {\n for (let i = 0, ii = 0; i < indices.length; i++) {\n const vi1 = indices[ii++] * 3;\n const vi2 = indices[ii++] * 3;\n const vi3 = indices[ii++] * 3;\n const p1 = new Vector3(vertices[vi1], vertices[vi1 + 1], vertices[vi1 + 2]);\n const p2 = new Vector3(vertices[vi2], vertices[vi2 + 1], vertices[vi2 + 2]);\n const p3 = new Vector3(vertices[vi3], vertices[vi3 + 1], vertices[vi3 + 2]);\n if (predicate) {\n const pointsFromPredicate = predicate(p1, p2, p3, i, vi1, m, meshIndex, vertices, indices);\n pointsFromPredicate && points.push(pointsFromPredicate);\n }\n else {\n points.push([p1, p2, p3, p1]);\n }\n }\n }\n });\n return points;\n }\n /**\n * Converts number coordinates to Vector3s\n * @param points number array of x, y, z, x, y z, ... coordinates\n * @returns Vector3 array\n */\n static ToVector3Array(points) {\n const array = [];\n for (let i = 0; i < points.length; i += 3) {\n array.push(new Vector3(points[i], points[i + 1], points[i + 2]));\n }\n return array;\n }\n /**\n * Gets a number array from a Vector3 array.\n * You can you for example to convert your Vector3[] offsets to the required number[] for the offsets option.\n * @param points Vector3 array\n * @returns an array of x, y, z coordinates as numbers [x, y, z, x, y, z, x, y, z, ....]\n */\n static ToNumberArray(points) {\n return points.flatMap((v) => [v.x, v.y, v.z]);\n }\n /**\n * Calculates the sum of points of every line and the number of points in each line.\n * This function is useful when you are drawing multiple lines in one mesh and you want\n * to know the counts. For example for creating an offsets table.\n * @param points point array\n * @returns points count info\n */\n static GetPointsCountInfo(points) {\n const counts = new Array(points.length);\n let total = 0;\n for (let n = points.length; n--;) {\n counts[n] = points[n].length / 3;\n total += counts[n];\n }\n return { total, counts };\n }\n /**\n * Gets the length of the line counting all it's segments length\n * @param data array of line points\n * @returns length of the line\n */\n static GetLineLength(data) {\n if (data.length === 0) {\n return 0;\n }\n let points;\n if (typeof data[0] === \"number\") {\n points = GreasedLineTools.ToVector3Array(data);\n }\n else {\n points = data;\n }\n const tmp = TmpVectors.Vector3[0];\n let length = 0;\n for (let index = 0; index < points.length - 1; index++) {\n const point1 = points[index];\n const point2 = points[index + 1];\n length += point2.subtractToRef(point1, tmp).length();\n }\n return length;\n }\n /**\n * Divides a segment into smaller segments.\n * A segment is a part of the line between it's two points.\n * @param point1 first point of the line\n * @param point2 second point of the line\n * @param segmentCount number of segments we want to have in the divided line\n * @returns\n */\n static SegmentizeSegmentByCount(point1, point2, segmentCount) {\n const dividedLinePoints = [];\n const diff = point2.subtract(point1);\n const divisor = TmpVectors.Vector3[0];\n divisor.setAll(segmentCount);\n const segmentVector = TmpVectors.Vector3[1];\n diff.divideToRef(divisor, segmentVector);\n let nextPoint = point1.clone();\n dividedLinePoints.push(nextPoint);\n for (let index = 0; index < segmentCount; index++) {\n nextPoint = nextPoint.clone();\n dividedLinePoints.push(nextPoint.addInPlace(segmentVector));\n }\n return dividedLinePoints;\n }\n /**\n * Divides a line into segments.\n * A segment is a part of the line between it's two points.\n * @param what line points\n * @param segmentLength length of each segment of the resulting line (distance between two line points)\n * @returns line point\n */\n static SegmentizeLineBySegmentLength(what, segmentLength) {\n const subLines = what[0] instanceof Vector3 ? GreasedLineTools.GetLineSegments(what) : what;\n const points = [];\n subLines.forEach((s) => {\n if (s.length > segmentLength) {\n const segments = GreasedLineTools.SegmentizeSegmentByCount(s.point1, s.point2, Math.ceil(s.length / segmentLength));\n segments.forEach((seg) => {\n points.push(seg);\n });\n }\n else {\n points.push(s.point1);\n points.push(s.point2);\n }\n });\n return points;\n }\n /**\n * Divides a line into segments.\n * A segment is a part of the line between it's two points.\n * @param what line points\n * @param segmentCount number of segments\n * @returns line point\n */\n static SegmentizeLineBySegmentCount(what, segmentCount) {\n const segmentLength = GreasedLineTools.GetLineLength(what) / segmentCount;\n return GreasedLineTools.SegmentizeLineBySegmentLength(what, segmentLength);\n }\n /**\n * Gets line segments.\n * A segment is a part of the line between it's two points.\n * @param points line points\n * @returns segments information of the line segment including starting point, ending point and the distance between them\n */\n static GetLineSegments(points) {\n const segments = [];\n for (let index = 0; index < points.length - 1; index++) {\n const point1 = points[index];\n const point2 = points[index + 1];\n const length = point2.subtract(point1).length();\n segments.push({ point1, point2, length });\n }\n return segments;\n }\n /**\n * Gets the minimum and the maximum length of a line segment in the line.\n * A segment is a part of the line between it's two points.\n * @param points line points\n * @returns\n */\n static GetMinMaxSegmentLength(points) {\n const subLines = GreasedLineTools.GetLineSegments(points);\n const sorted = subLines.sort((s) => s.length);\n return {\n min: sorted[0].length,\n max: sorted[sorted.length - 1].length,\n };\n }\n /**\n * Finds the last visible position in world space of the line according to the visibility parameter\n * @param lineSegments segments of the line\n * @param lineLength total length of the line\n * @param visbility normalized value of visibility\n * @returns world space coordinate of the last visible piece of the line\n */\n static GetPositionOnLineByVisibility(lineSegments, lineLength, visbility, localSpace = false) {\n const lengthVisibilityRatio = lineLength * visbility;\n let sumSegmentLengths = 0;\n let segmentIndex = 0;\n const lineSegmentsLength = lineSegments.length;\n for (let i = 0; i < lineSegmentsLength; i++) {\n if (lengthVisibilityRatio <= sumSegmentLengths + lineSegments[i].length) {\n segmentIndex = i;\n break;\n }\n sumSegmentLengths += lineSegments[i].length;\n }\n const s = (lengthVisibilityRatio - sumSegmentLengths) / lineSegments[segmentIndex].length;\n lineSegments[segmentIndex].point2.subtractToRef(lineSegments[segmentIndex].point1, TmpVectors.Vector3[0]);\n TmpVectors.Vector3[1] = TmpVectors.Vector3[0].multiplyByFloats(s, s, s);\n if (!localSpace) {\n TmpVectors.Vector3[1].addInPlace(lineSegments[segmentIndex].point1);\n }\n return TmpVectors.Vector3[1].clone();\n }\n /**\n * Creates lines in a shape of circle/arc.\n * A segment is a part of the line between it's two points.\n * @param radiusX radiusX of the circle\n * @param segments number of segments in the circle\n * @param z z coordinate of the points. Defaults to 0.\n * @param radiusY radiusY of the circle - you can draw an oval if using different values\n * @param segmentAngle angle offset of the segments. Defaults to Math.PI * 2 / segments. Change this value to draw a part of the circle.\n * @returns line points\n */\n static GetCircleLinePoints(radiusX, segments, z = 0, radiusY = radiusX, segmentAngle = (Math.PI * 2) / segments) {\n const points = [];\n for (let i = 0; i <= segments; i++) {\n points.push(new Vector3(Math.cos(i * segmentAngle) * radiusX, Math.sin(i * segmentAngle) * radiusY, z));\n }\n return points;\n }\n /**\n * Gets line points in a shape of a bezier curve\n * @param p0 bezier point0\n * @param p1 bezier point1\n * @param p2 bezier point2\n * @param segments number of segments in the curve\n * @returns\n */\n static GetBezierLinePoints(p0, p1, p2, segments) {\n return Curve3.CreateQuadraticBezier(p0, p1, p2, segments)\n .getPoints()\n .flatMap((v) => [v.x, v.y, v.z]);\n }\n /**\n *\n * @param position position of the arrow cap (mainly you want to create a triangle, set widthUp and widthDown to the same value and omit widthStartUp and widthStartDown)\n * @param direction direction which the arrow points to\n * @param length length (size) of the arrow cap itself\n * @param widthUp the arrow width above the line\n * @param widthDown the arrow width belove the line\n * @param widthStartUp the arrow width at the start of the arrow above the line. In most scenarios this is 0.\n * @param widthStartDown the arrow width at the start of the arrow below the line. In most scenarios this is 0.\n * @returns\n */\n static GetArrowCap(position, direction, length, widthUp, widthDown, widthStartUp = 0, widthStartDown = 0) {\n const points = [position.clone(), position.add(direction.multiplyByFloats(length, length, length))];\n const widths = [widthUp, widthDown, widthStartUp, widthStartDown];\n return {\n points,\n widths,\n };\n }\n /**\n * Gets 3D positions of points from a text and font\n * @param text Text\n * @param size Size of the font\n * @param resolution Resolution of the font\n * @param fontData defines the font data (can be generated with http://gero3.github.io/facetype.js/)\n * @param z z coordinate\n * @param includeInner include the inner parts of the font in the result. Default true. If false, only the outlines will be returned.\n * @returns number[][] of 3D positions\n */\n static GetPointsFromText(text, size, resolution, fontData, z = 0, includeInner = true) {\n const allPoints = [];\n const shapePaths = CreateTextShapePaths(text, size, resolution, fontData);\n for (const sp of shapePaths) {\n for (const p of sp.paths) {\n const points = [];\n const points2d = p.getPoints();\n for (const p2d of points2d) {\n points.push(p2d.x, p2d.y, z);\n }\n allPoints.push(points);\n }\n if (includeInner) {\n for (const h of sp.holes) {\n const holes = [];\n const points2d = h.getPoints();\n for (const p2d of points2d) {\n holes.push(p2d.x, p2d.y, z);\n }\n allPoints.push(holes);\n }\n }\n }\n return allPoints;\n }\n }\n\n Mesh._GreasedLineMeshParser = (parsedMesh, scene) => {\n return GreasedLineMesh.Parse(parsedMesh, scene);\n };\n /**\n * GreasedLine\n */\n class GreasedLineMesh extends Mesh {\n constructor(name, scene, _options) {\n var _a, _b, _c, _d;\n super(name, scene, null, null, false, false);\n this.name = name;\n this._options = _options;\n this._lazy = false;\n this._updatable = false;\n /**\n * Treshold used to pick the mesh\n */\n this.intersectionThreshold = 0.1;\n this._lazy = (_a = _options.lazy) !== null && _a !== void 0 ? _a : false;\n this._updatable = (_b = _options.updatable) !== null && _b !== void 0 ? _b : false;\n this._vertexPositions = [];\n this._indices = [];\n this._uvs = [];\n this._points = [];\n this._colorPointers = (_c = _options.colorPointers) !== null && _c !== void 0 ? _c : [];\n this._widths = (_d = _options.widths) !== null && _d !== void 0 ? _d : new Array(_options.points.length).fill(1);\n this._previousAndSide = [];\n this._nextAndCounters = [];\n if (_options.points) {\n this.addPoints(GreasedLineMesh.ConvertPoints(_options.points));\n }\n }\n /**\n * \"GreasedLineMesh\"\n * @returns \"GreasedLineMesh\"\n */\n getClassName() {\n return \"GreasedLineMesh\";\n }\n /**\n * Converts GreasedLinePoints to number[][]\n * @param points GreasedLinePoints\n * @returns number[][] with x, y, z coordinates of the points, like [[x, y, z, x, y, z, ...], [x, y, z, ...]]\n */\n static ConvertPoints(points) {\n if (points.length && Array.isArray(points) && typeof points[0] === \"number\") {\n return [points];\n }\n else if (points.length && Array.isArray(points[0]) && typeof points[0][0] === \"number\") {\n return points;\n }\n else if (points.length && !Array.isArray(points[0]) && points[0] instanceof Vector3) {\n const positions = [];\n for (let j = 0; j < points.length; j++) {\n const p = points[j];\n positions.push(p.x, p.y, p.z);\n }\n return [positions];\n }\n else if (points.length > 0 && Array.isArray(points[0]) && points[0].length > 0 && points[0][0] instanceof Vector3) {\n const positions = [];\n const vectorPoints = points;\n vectorPoints.forEach((p) => {\n positions.push(p.flatMap((p2) => [p2.x, p2.y, p2.z]));\n });\n return positions;\n }\n else if (points instanceof Float32Array) {\n return [Array.from(points)];\n }\n else if (points.length && points[0] instanceof Float32Array) {\n const positions = [];\n points.forEach((p) => {\n positions.push(Array.from(p));\n });\n return positions;\n }\n return [];\n }\n /**\n * Updated a lazy line. Rerenders the line and updates boundinfo as well.\n */\n updateLazy() {\n var _a;\n this.setPoints(this._points);\n if (!this._options.colorPointers) {\n this._updateColorPointers();\n }\n this._createVertexBuffers();\n this.refreshBoundingInfo();\n (_a = this.greasedLineMaterial) === null || _a === void 0 ? void 0 : _a.updateLazy();\n }\n /**\n * Dispose the line and it's resources\n */\n dispose() {\n super.dispose();\n }\n /**\n *\n * @returns true if the mesh was created in lazy mode\n */\n isLazy() {\n return this._lazy;\n }\n /**\n * Return the the points offsets\n */\n get offsets() {\n return this._offsets;\n }\n /**\n * Sets point offests\n * @param offsets offset table [x,y,z, x,y,z, ....]\n */\n set offsets(offsets) {\n this._offsets = offsets;\n if (!this._offsetsBuffer) {\n this._createOffsetsBuffer(offsets);\n }\n else {\n this._offsetsBuffer && this._offsetsBuffer.update(offsets);\n }\n }\n /**\n * Gets widths at each line point like [widthLower, widthUpper, widthLower, widthUpper, ...]\n */\n get widths() {\n return this._widths;\n }\n /**\n * Sets widths at each line point\n * @param widths width table [widthLower, widthUpper, widthLower, widthUpper ...]\n */\n set widths(widths) {\n this._widths = widths;\n if (!this._lazy) {\n this._widthsBuffer && this._widthsBuffer.update(widths);\n }\n }\n /**\n * Gets the color pointer. Each vertex need a color pointer. These color pointers points to the colors in the color table @see colors\n */\n get colorPointers() {\n return this._colorPointers;\n }\n /**\n * Sets the color pointer\n * @param colorPointers array of color pointer in the colors array. One pointer for every vertex is needed.\n */\n set colorPointers(colorPointers) {\n this._colorPointers = colorPointers;\n if (!this._lazy) {\n this._colorPointersBuffer && this._colorPointersBuffer.update(colorPointers);\n }\n }\n /**\n * Gets the pluginMaterial associated with line\n */\n get greasedLineMaterial() {\n var _a, _b;\n return (_b = (_a = this.material) === null || _a === void 0 ? void 0 : _a.pluginManager) === null || _b === void 0 ? void 0 : _b.getPlugin(GreasedLinePluginMaterial.GREASED_LINE_MATERIAL_NAME);\n }\n /**\n * Return copy the points.\n */\n get points() {\n const pointsCopy = [];\n DeepCopier.DeepCopy(this._points, pointsCopy);\n return pointsCopy;\n }\n /**\n * Adds new points to the line. It doesn't rerenders the line if in lazy mode.\n * @param points points table\n */\n addPoints(points) {\n for (const p of points) {\n this._points.push(p);\n }\n if (!this._lazy) {\n this.setPoints(this._points);\n }\n }\n _updateColorPointers() {\n let colorPointer = 0;\n this._colorPointers = [];\n this._points.forEach((p) => {\n for (let jj = 0; jj < p.length; jj += 3) {\n this._colorPointers.push(colorPointer);\n this._colorPointers.push(colorPointer++);\n }\n });\n }\n /**\n * Sets line points and rerenders the line.\n * @param points points table\n */\n setPoints(points) {\n this._points = points;\n this._options.points = points;\n this._initGreasedLine();\n let indiceOffset = 0;\n points.forEach((p) => {\n var _a;\n const counters = [];\n const positions = [];\n const indices = [];\n const totalLength = GreasedLineTools.GetLineLength(p);\n for (let j = 0, jj = 0; jj < p.length; j++, jj += 3) {\n const partialLine = p.slice(0, jj + 3);\n const partialLineLength = GreasedLineTools.GetLineLength(partialLine);\n const c = partialLineLength / totalLength;\n positions.push(p[jj], p[jj + 1], p[jj + 2]);\n positions.push(p[jj], p[jj + 1], p[jj + 2]);\n counters.push(c);\n counters.push(c);\n if (jj < p.length - 3) {\n const n = j * 2 + indiceOffset;\n indices.push(n, n + 1, n + 2);\n indices.push(n + 2, n + 1, n + 3);\n }\n }\n indiceOffset += (p.length / 3) * 2;\n const previous = [];\n const next = [];\n const side = [];\n let uvs = [];\n this._preprocess(positions, previous, next, side, uvs);\n for (const vp of positions) {\n this._vertexPositions.push(vp);\n }\n for (const i of indices) {\n this._indices.push(i);\n }\n for (let i = 0; i < side.length; i++) {\n this._previousAndSide.push(previous[i * 3], previous[i * 3 + 1], previous[i * 3 + 2], side[i]);\n this._nextAndCounters.push(next[i * 3], next[i * 3 + 1], next[i * 3 + 2], counters[i]);\n }\n uvs = (_a = this._options.uvs) !== null && _a !== void 0 ? _a : uvs;\n for (const uv of uvs) {\n this._uvs.push(uv);\n }\n });\n if (!this._lazy) {\n if (!this._options.colorPointers) {\n this._updateColorPointers();\n }\n this._createVertexBuffers();\n this.refreshBoundingInfo();\n }\n }\n _createLineOptions() {\n const lineOptions = {\n points: this._points,\n colorPointers: this._colorPointers,\n lazy: this._lazy,\n updatable: this._updatable,\n uvs: this._uvs,\n widths: this._widths,\n };\n return lineOptions;\n }\n /**\n * Clones the GreasedLineMesh.\n * @param name new line name\n * @param newParent new parent node\n * @returns cloned line\n */\n clone(name = `${this.name}-cloned`, newParent) {\n const lineOptions = this._createLineOptions();\n const deepCopiedLineOptions = {};\n DeepCopier.DeepCopy(lineOptions, deepCopiedLineOptions, [\"instance\"]);\n const cloned = new GreasedLineMesh(name, this._scene, deepCopiedLineOptions);\n if (newParent) {\n cloned.parent = newParent;\n }\n cloned.material = this.material;\n return cloned;\n }\n /**\n * Serializes this GreasedLineMesh\n * @param serializationObject object to write serialization to\n */\n serialize(serializationObject) {\n super.serialize(serializationObject);\n serializationObject.type = this.getClassName();\n serializationObject.lineOptions = this._createLineOptions();\n }\n /**\n * Parses a serialized GreasedLineMesh\n * @param parsedMesh the serialized GreasedLineMesh\n * @param scene the scene to create the GreasedLineMesh in\n * @returns the created GreasedLineMesh\n */\n static Parse(parsedMesh, scene) {\n const lineOptions = parsedMesh.lineOptions;\n const name = parsedMesh.name;\n const result = new GreasedLineMesh(name, scene, lineOptions);\n return result;\n }\n /**\n * Checks whether a ray is intersecting this GreasedLineMesh\n * @param ray ray to check the intersection of this mesh with\n * @param fastCheck not supported\n * @param trianglePredicate not supported\n * @param onlyBoundingInfo defines a boolean indicating if picking should only happen using bounding info (false by default)\n * @param worldToUse not supported\n * @param skipBoundingInfo a boolean indicating if we should skip the bounding info check\n * @returns the picking info\n */\n intersects(ray, fastCheck, trianglePredicate, onlyBoundingInfo = false, worldToUse, skipBoundingInfo = false) {\n const pickingInfo = new PickingInfo();\n const intersections = this.findAllIntersections(ray, fastCheck, trianglePredicate, onlyBoundingInfo, worldToUse, skipBoundingInfo, true);\n if ((intersections === null || intersections === void 0 ? void 0 : intersections.length) === 1) {\n const intersection = intersections[0];\n pickingInfo.hit = true;\n pickingInfo.distance = intersection.distance;\n pickingInfo.ray = ray;\n pickingInfo.pickedMesh = this;\n pickingInfo.pickedPoint = intersection.point;\n }\n return pickingInfo;\n }\n /**\n * Gets all intersections of a ray and the line\n * @param ray Ray to check the intersection of this mesh with\n * @param _fastCheck not supported\n * @param _trianglePredicate not supported\n * @param onlyBoundingInfo defines a boolean indicating if picking should only happen using bounding info (false by default)\n * @param _worldToUse not supported\n * @param skipBoundingInfo a boolean indicating if we should skip the bounding info check\n * @param firstOnly If true, the first and only intersection is immediatelly returned if found\n * @returns intersection(s)\n */\n findAllIntersections(ray, _fastCheck, _trianglePredicate, onlyBoundingInfo = false, _worldToUse, skipBoundingInfo = false, firstOnly = false) {\n var _a, _b;\n if (onlyBoundingInfo && !skipBoundingInfo && ray.intersectsSphere(this._boundingSphere, this.intersectionThreshold) === false) {\n return;\n }\n const indices = this.getIndices();\n const positions = this.getVerticesData(VertexBuffer.PositionKind);\n const widths = this._widths;\n const lineWidth = (_b = (_a = this.greasedLineMaterial) === null || _a === void 0 ? void 0 : _a.width) !== null && _b !== void 0 ? _b : 1;\n const intersects = [];\n if (indices && positions && widths) {\n let i = 0, l = 0;\n for (i = 0, l = indices.length - 1; i < l; i += 3) {\n const a = indices[i];\n const b = indices[i + 1];\n GreasedLineMesh._V_START.fromArray(positions, a * 3);\n GreasedLineMesh._V_END.fromArray(positions, b * 3);\n if (this._offsets) {\n GreasedLineMesh._V_OFFSET_START.fromArray(this._offsets, a * 3);\n GreasedLineMesh._V_OFFSET_END.fromArray(this._offsets, b * 3);\n GreasedLineMesh._V_START.addInPlace(GreasedLineMesh._V_OFFSET_START);\n GreasedLineMesh._V_END.addInPlace(GreasedLineMesh._V_OFFSET_END);\n }\n const iFloored = Math.floor(i / 3);\n const width = widths[iFloored] !== undefined ? widths[iFloored] : 1;\n const precision = (this.intersectionThreshold * (lineWidth * width)) / 2;\n const distance = ray.intersectionSegment(GreasedLineMesh._V_START, GreasedLineMesh._V_END, precision);\n if (distance !== -1) {\n intersects.push({\n distance: distance,\n point: ray.direction.normalize().multiplyByFloats(distance, distance, distance).add(ray.origin),\n });\n if (firstOnly) {\n return intersects;\n }\n }\n }\n i = l;\n }\n return intersects;\n }\n _initGreasedLine() {\n this._vertexPositions = [];\n this._previousAndSide = [];\n this._nextAndCounters = [];\n this._indices = [];\n this._uvs = [];\n }\n get _boundingSphere() {\n return this.getBoundingInfo().boundingSphere;\n }\n static _CompareV3(positionIdx1, positionIdx2, positions) {\n const arrayIdx1 = positionIdx1 * 6;\n const arrayIdx2 = positionIdx2 * 6;\n return positions[arrayIdx1] === positions[arrayIdx2] && positions[arrayIdx1 + 1] === positions[arrayIdx2 + 1] && positions[arrayIdx1 + 2] === positions[arrayIdx2 + 2];\n }\n static _CopyV3(positionIdx, positions) {\n const arrayIdx = positionIdx * 6;\n return [positions[arrayIdx], positions[arrayIdx + 1], positions[arrayIdx + 2]];\n }\n _preprocess(positions, previous, next, side, uvs) {\n const l = positions.length / 6;\n let v = [];\n if (GreasedLineMesh._CompareV3(0, l - 1, positions)) {\n v = GreasedLineMesh._CopyV3(l - 2, positions);\n }\n else {\n v = GreasedLineMesh._CopyV3(0, positions);\n }\n previous.push(v[0], v[1], v[2]);\n previous.push(v[0], v[1], v[2]);\n for (let j = 0; j < l; j++) {\n side.push(1);\n side.push(-1);\n // uvs\n if (!this._options.uvs) {\n uvs.push(j / (l - 1), 0);\n uvs.push(j / (l - 1), 1);\n }\n if (j < l - 1) {\n v = GreasedLineMesh._CopyV3(j, positions);\n previous.push(v[0], v[1], v[2]);\n previous.push(v[0], v[1], v[2]);\n }\n if (j > 0) {\n v = GreasedLineMesh._CopyV3(j, positions);\n next.push(v[0], v[1], v[2]);\n next.push(v[0], v[1], v[2]);\n }\n }\n if (GreasedLineMesh._CompareV3(l - 1, 0, positions)) {\n v = GreasedLineMesh._CopyV3(1, positions);\n }\n else {\n v = GreasedLineMesh._CopyV3(l - 1, positions);\n }\n next.push(v[0], v[1], v[2]);\n next.push(v[0], v[1], v[2]);\n return {\n previous,\n next,\n uvs,\n side,\n };\n }\n _createVertexBuffers() {\n const vertexData = new VertexData();\n vertexData.positions = this._vertexPositions;\n vertexData.indices = this._indices;\n vertexData.uvs = this._uvs;\n vertexData.applyToMesh(this, this._options.updatable);\n const engine = this._scene.getEngine();\n const previousAndSideBuffer = new Buffer$1(engine, this._previousAndSide, false, 4);\n this.setVerticesBuffer(previousAndSideBuffer.createVertexBuffer(\"grl_previousAndSide\", 0, 4));\n const nextAndCountersBuffer = new Buffer$1(engine, this._nextAndCounters, false, 4);\n this.setVerticesBuffer(nextAndCountersBuffer.createVertexBuffer(\"grl_nextAndCounters\", 0, 4));\n const widthBuffer = new Buffer$1(engine, this._widths, this._updatable, 1);\n this.setVerticesBuffer(widthBuffer.createVertexBuffer(\"grl_widths\", 0, 1));\n this._widthsBuffer = widthBuffer;\n const colorPointersBuffer = new Buffer$1(engine, this._colorPointers, this._updatable, 1);\n this.setVerticesBuffer(colorPointersBuffer.createVertexBuffer(\"grl_colorPointers\", 0, 1));\n this._colorPointersBuffer = colorPointersBuffer;\n }\n _createOffsetsBuffer(offsets) {\n const engine = this._scene.getEngine();\n const offsetBuffer = new Buffer$1(engine, offsets, this._updatable, 3);\n this.setVerticesBuffer(offsetBuffer.createVertexBuffer(\"grl_offsets\", 0, 3));\n this._offsetsBuffer = offsetBuffer;\n }\n }\n GreasedLineMesh._V_START = new Vector3();\n GreasedLineMesh._V_END = new Vector3();\n GreasedLineMesh._V_OFFSET_START = new Vector3();\n GreasedLineMesh._V_OFFSET_END = new Vector3();\n\n /**\n * How are the colors distributed along the color table\n * {@link https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param/greased_line#colors-and-colordistribution}\n */\n var GreasedLineMeshColorDistribution;\n (function (GreasedLineMeshColorDistribution) {\n /**\n * Do no modify the color table\n */\n GreasedLineMeshColorDistribution[GreasedLineMeshColorDistribution[\"COLOR_DISTRIBUTION_NONE\"] = 0] = \"COLOR_DISTRIBUTION_NONE\";\n /**\n * Repeat the colors until the color table is full\n */\n GreasedLineMeshColorDistribution[GreasedLineMeshColorDistribution[\"COLOR_DISTRIBUTION_REPEAT\"] = 1] = \"COLOR_DISTRIBUTION_REPEAT\";\n /**\n * Distribute the colors evenly through the color table\n */\n GreasedLineMeshColorDistribution[GreasedLineMeshColorDistribution[\"COLOR_DISTRIBUTION_EVEN\"] = 2] = \"COLOR_DISTRIBUTION_EVEN\";\n /**\n * Put the colors to start of the color table a fill the rest with the default color\n */\n GreasedLineMeshColorDistribution[GreasedLineMeshColorDistribution[\"COLOR_DISTRIBUTION_START\"] = 3] = \"COLOR_DISTRIBUTION_START\";\n /**\n * Put the colors to the end of the color table and fill the rest with the default color\n */\n GreasedLineMeshColorDistribution[GreasedLineMeshColorDistribution[\"COLOR_DISTRIBUTION_END\"] = 4] = \"COLOR_DISTRIBUTION_END\";\n /**\n * Put the colors to start and to the end of the color table and fill the gap between with the default color\n */\n GreasedLineMeshColorDistribution[GreasedLineMeshColorDistribution[\"COLOR_DISTRIBUTION_START_END\"] = 5] = \"COLOR_DISTRIBUTION_START_END\";\n })(GreasedLineMeshColorDistribution || (GreasedLineMeshColorDistribution = {}));\n /**\n * How are the widths distributed along the width table\n * {@link https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param/greased_line#widths-and-widthdistribution}\n */\n var GreasedLineMeshWidthDistribution;\n (function (GreasedLineMeshWidthDistribution) {\n /**\n * Do no modify the width table\n */\n GreasedLineMeshWidthDistribution[GreasedLineMeshWidthDistribution[\"WIDTH_DISTRIBUTION_NONE\"] = 0] = \"WIDTH_DISTRIBUTION_NONE\";\n /**\n * Repeat the widths until the width table is full\n */\n GreasedLineMeshWidthDistribution[GreasedLineMeshWidthDistribution[\"WIDTH_DISTRIBUTION_REPEAT\"] = 1] = \"WIDTH_DISTRIBUTION_REPEAT\";\n /**\n * Distribute the widths evenly through the width table\n */\n GreasedLineMeshWidthDistribution[GreasedLineMeshWidthDistribution[\"WIDTH_DISTRIBUTION_EVEN\"] = 2] = \"WIDTH_DISTRIBUTION_EVEN\";\n /**\n * Put the widths to start of the width table a fill the rest with the default width\n */\n GreasedLineMeshWidthDistribution[GreasedLineMeshWidthDistribution[\"WIDTH_DISTRIBUTION_START\"] = 3] = \"WIDTH_DISTRIBUTION_START\";\n /**\n * Put the widths to the end of the width table and fill the rest with the default width\n */\n GreasedLineMeshWidthDistribution[GreasedLineMeshWidthDistribution[\"WIDTH_DISTRIBUTION_END\"] = 4] = \"WIDTH_DISTRIBUTION_END\";\n /**\n * Put the widths to start and to the end of the width table and fill the gap between with the default width\n */\n GreasedLineMeshWidthDistribution[GreasedLineMeshWidthDistribution[\"WIDTH_DISTRIBUTION_START_END\"] = 5] = \"WIDTH_DISTRIBUTION_START_END\";\n })(GreasedLineMeshWidthDistribution || (GreasedLineMeshWidthDistribution = {}));\n\n const pWorking = new Vector3();\n const p1 = new Vector3();\n const p2 = new Vector3();\n const p3 = new Vector3();\n const uv1 = new Vector2();\n const uv2 = new Vector2();\n const uv3 = new Vector2();\n const workingUv = new Vector2();\n const workingUv2 = new Vector2();\n const working1 = new Vector3();\n class UVMapHelper {\n generateUVMapInPlace(vertexData, feature, indexList) {\n const newControlPointMap = {};\n const axis = feature.projectionAxis.toVec3();\n const rotation = rotationToVector(axis, new Vector3(0, 1, 0));\n rotation.multiplyInPlace(Quaternion.RotationAxis(new Vector3(0, 1, 0), (feature.angle / 180) * Math.PI));\n // const matrixRotation = new Matrix();\n // Matrix.FromQuaternionToRef(rotation, matrixRotation);\n const transformMatrix = Matrix.Compose(feature.scale ? feature.scale.toVec3() : Vector3.One(), rotation, Vector3.Zero());\n const transformMatrixInv = Matrix.Invert(transformMatrix);\n if (!vertexData.positions) {\n throw Error('No vertex positions on mesh');\n }\n if (!vertexData.indices) {\n throw Error('No vertex indices on mesh');\n }\n const geometryPositions = vertexData.positions;\n const geometryIndices = vertexData.indices;\n const controlPointCount = geometryPositions.length / 3;\n const newUvs = vertexData.uvs && vertexData.uvs.length === controlPointCount * 2\n ? new Float32Array(vertexData.uvs)\n : new Float32Array(controlPointCount * 2);\n const appendVertexData = {\n positions: [],\n normals: [],\n uvs: [],\n };\n const nbFaces = indexList ? indexList.length / 3 : geometryIndices.length / 3;\n const touchedControlPoints = new Int8Array(controlPointCount);\n if (indexList) {\n indexList.forEach(index => {\n touchedControlPoints[geometryIndices[index]] = 1;\n });\n }\n const outputUvs = [uv1, uv2, uv3];\n for (let i = 0; i < nbFaces; i++) {\n /** here we generate the new UVs */\n let i1;\n let i2;\n let i3;\n if (indexList) {\n i1 = indexList[i * 3];\n i2 = indexList[i * 3 + 1];\n i3 = indexList[i * 3 + 2];\n }\n else {\n i1 = i * 3;\n i2 = i * 3 + 1;\n i3 = i * 3 + 2;\n }\n const v1 = geometryIndices[i1];\n const v2 = geometryIndices[i2];\n const v3 = geometryIndices[i3];\n if (feature.mode === exports.eUvMode.useExisting) {\n if (vertexData.uvs) {\n const angle = (feature.angle / 180) * Math.PI;\n const vertices = [v1, v2, v3];\n for (let i = 0; i < 3; i++) {\n const uvIdx = vertices[i] * 2;\n const u = vertexData.uvs[uvIdx] - 0.5;\n const v = vertexData.uvs[uvIdx + 1] - 0.5;\n let newU = u * Math.cos(angle) - v * Math.sin(angle) + 0.5;\n let newV = u * Math.sin(angle) + v * Math.cos(angle) + 0.5;\n newU = newU * feature.uTile + feature.uOffset;\n newV = newV * feature.vTile + feature.vOffset;\n outputUvs[i].copyFromFloats(newU, newV);\n }\n }\n }\n else {\n const start = v1 * 3;\n p1.copyFromFloats(geometryPositions[start], geometryPositions[start + 1], geometryPositions[start + 2]);\n const start2 = v2 * 3;\n p2.copyFromFloats(geometryPositions[start2], geometryPositions[start2 + 1], geometryPositions[start2 + 2]);\n const start3 = v3 * 3;\n p3.copyFromFloats(geometryPositions[start3], geometryPositions[start3 + 1], geometryPositions[start3 + 2]);\n const normal = computeNormal([p1, p2, p3]);\n applyMatrix4(normal, transformMatrixInv);\n if (feature.center) {\n const center = feature.center.toVec3();\n p1.subtractInPlace(center);\n p2.subtractInPlace(center);\n p3.subtractInPlace(center);\n }\n this.updateFaceUVMapToRef(normal, p1, feature, transformMatrixInv, uv1);\n this.updateFaceUVMapToRef(normal, p2, feature, transformMatrixInv, uv2);\n this.updateFaceUVMapToRef(normal, p3, feature, transformMatrixInv, uv3);\n /**\n * Find the UV at the midpoint position between each edge of the polygon.\n * This should closely match the midpoints of the generated vertex UVs.If it does not, we must wrap the UVs at that edge to avoid a UV seam.\n * The tolerance is an arbitrary number to avoid artifacts. If wrapping is occuring when it shouldn't, increase this number.\n * If wrapping is not occurring when it should, reduce it.\n * As you try to wrap a uv on polygons that are not arranged in a good way to accept them (i.e wrap a sphere onto a 8-vertex cube), the error will become significant.\n */\n const tolerance = 0.05;\n working1.copyFrom(p1);\n this.updateFaceUVMapToRef(normal, working1.addInPlace(p2).scaleInPlace(0.5), feature, transformMatrixInv, workingUv);\n workingUv2.copyFrom(uv1);\n workingUv2.addInPlace(uv2).scaleInPlace(0.5);\n workingUv.subtractInPlace(workingUv2);\n if (Math.abs(workingUv.x % 1) > tolerance) {\n if (uv1.x > uv2.x)\n uv1.x -= 1;\n else\n uv2.x -= 1;\n }\n if (Math.abs(workingUv.y % 1) > tolerance) {\n if (uv1.y > uv2.y)\n uv1.y -= 1;\n else\n uv2.y -= 1;\n }\n if (uv2.x >= 0 && uv2.y >= 0) {\n working1.copyFrom(p2);\n this.updateFaceUVMapToRef(normal, working1.addInPlace(p3).scaleInPlace(0.5), feature, transformMatrixInv, workingUv);\n workingUv2.copyFrom(uv2);\n workingUv2.addInPlace(uv3).scaleInPlace(0.5);\n workingUv.subtractInPlace(workingUv2);\n if (Math.abs(workingUv.x % 1) > tolerance) {\n if (uv2.x > uv3.x)\n uv2.x -= 1;\n else\n uv3.x -= 1;\n }\n if (Math.abs(workingUv.y % 1) > tolerance) {\n if (uv2.y > uv3.y)\n uv2.y -= 1;\n else\n uv3.y -= 1;\n }\n }\n if (uv1.x >= 0 && uv1.y >= 0 && uv3.x >= 0 && uv3.y >= 0) {\n working1.copyFrom(p1);\n this.updateFaceUVMapToRef(normal, working1.addInPlace(p3).scaleInPlace(0.5), feature, transformMatrixInv, workingUv);\n workingUv2.copyFrom(uv1);\n workingUv2.addInPlace(uv3).scaleInPlace(0.5);\n workingUv.subtractInPlace(workingUv2);\n if (Math.abs(workingUv.x % 1) > tolerance) {\n if (uv1.x > uv3.x)\n uv1.x -= 1;\n else\n uv3.x -= 1;\n }\n if (Math.abs(workingUv.y % 1) > tolerance) {\n if (uv1.y > uv3.y)\n uv1.y -= 1;\n else\n uv3.y -= 1;\n }\n }\n }\n /** Done generating the new UVs, stored in uv1, uv2 and uv3 for this polygon */\n if (touchedControlPoints[v1]) {\n expandControlPoints(i1, v1, uv1);\n }\n else {\n newUvs[v1 * 2] = uv1.x;\n newUvs[v1 * 2 + 1] = uv1.y;\n }\n if (touchedControlPoints[v2]) {\n expandControlPoints(i2, v2, uv2);\n }\n else {\n newUvs[v2 * 2] = uv2.x;\n newUvs[v2 * 2 + 1] = uv2.y;\n }\n if (touchedControlPoints[v3]) {\n expandControlPoints(i3, v3, uv3);\n }\n else {\n newUvs[v3 * 2] = uv3.x;\n newUvs[v3 * 2 + 1] = uv3.y;\n }\n touchedControlPoints[v1] = 1;\n touchedControlPoints[v2] = 1;\n touchedControlPoints[v3] = 1;\n }\n if (appendVertexData.positions.length > 0) {\n const newPositions = new Float32Array(vertexData.positions.length + appendVertexData.positions.length);\n newPositions.set(vertexData.positions);\n newPositions.set(appendVertexData.positions, vertexData.positions.length);\n vertexData.positions = newPositions;\n }\n if (vertexData.normals && appendVertexData.normals.length > 0) {\n const newNormals = new Float32Array(vertexData.normals.length + appendVertexData.normals.length);\n newNormals.set(vertexData.normals);\n newNormals.set(appendVertexData.normals, vertexData.normals.length);\n vertexData.normals = newNormals;\n }\n if (appendVertexData.uvs.length > 0) {\n const newUvsTyped = new Float32Array(newUvs.length + appendVertexData.uvs.length);\n newUvsTyped.set(newUvs);\n newUvsTyped.set(appendVertexData.uvs, newUvs.length);\n vertexData.uvs = newUvsTyped;\n }\n else {\n vertexData.uvs = new Float32Array(newUvs);\n }\n return vertexData;\n function expandControlPoints(index, vertexIndex, newUv) {\n if (newUvs[vertexIndex * 2] !== newUv.x || newUvs[vertexIndex * 2 + 1] !== newUv.y) {\n if (!vertexData.positions || !vertexData.indices) {\n throw Error('Invalid geometry: positions or indices are null');\n }\n const indices = vertexData.indices;\n const positions = vertexData.positions;\n const normals = vertexData.normals;\n const key = positions[vertexIndex * 3].toPrecision(6) +\n ',' +\n positions[vertexIndex * 3 + 1].toPrecision(6) +\n ',' +\n positions[vertexIndex * 3 + 2].toPrecision(6) +\n '_' +\n (normals\n ? normals[vertexIndex * 3].toPrecision(6) +\n ',' +\n normals[vertexIndex * 3 + 1].toPrecision(6) +\n ',' +\n normals[vertexIndex * 3 + 2].toPrecision(6) +\n '_'\n : '') +\n newUv.x.toPrecision(6) +\n ',' +\n newUv.y.toPrecision(6);\n if (newControlPointMap[key]) {\n indices[index] = newControlPointMap[key];\n }\n else {\n const newVertexIndex = (positions.length + appendVertexData.positions.length) / 3;\n indices[index] = newVertexIndex;\n appendVertexData.positions.push(...positions.slice(vertexIndex * 3, vertexIndex * 3 + 3));\n if (vertexData.normals) {\n appendVertexData.normals.push(...vertexData.normals.slice(vertexIndex * 3, vertexIndex * 3 + 3));\n }\n appendVertexData.uvs.push(newUv.x, newUv.y);\n if (vertexData.submeshData) {\n for (let i = 0; i < vertexData.submeshData.length; i++) {\n const submeshData = vertexData.submeshData[i];\n if (submeshData.verticesStart <= vertexIndex &&\n submeshData.verticesStart + submeshData.verticesCount > vertexIndex) {\n submeshData.verticesCount += 1;\n }\n else if (submeshData.verticesStart > vertexIndex) {\n submeshData.verticesStart += 1;\n }\n }\n }\n newControlPointMap[key] = newVertexIndex;\n }\n }\n }\n }\n computeAutoFitSize(mesh) {\n const boundingBox = mesh.getBoundingInfo().boundingBox;\n //const center = boundingBox.center;\n const max = boundingBox.maximum;\n const min = boundingBox.minimum;\n return new exports.KbVector(max.x - min.x, max.y - min.y, max.z - min.z);\n }\n updateFaceUVMapToRef(faceNormal, position, feat, transformMatrixInv, uv) {\n pWorking.copyFrom(position);\n applyMatrix4(pWorking, transformMatrixInv);\n switch (feat.mode) {\n case exports.eUvMode.planar:\n this.planarUVMap(pWorking, uv);\n break;\n case exports.eUvMode.box:\n this.boxUVMap(pWorking, faceNormal, uv);\n break;\n case exports.eUvMode.cylindrical:\n this.cylindricalUVMap(pWorking, uv);\n break;\n case exports.eUvMode.spherical:\n this.sphericalUVMap(pWorking, uv);\n break;\n case exports.eUvMode.useExisting:\n //don't do anything so that existing UVs of mesh are being used instead\n break;\n default:\n uv.set(0, 0);\n break;\n }\n if (feat.uTile && feat.uTile !== 1) {\n uv.x *= feat.uTile;\n }\n if (feat.vTile && feat.vTile !== 1) {\n uv.y *= feat.vTile;\n }\n return uv;\n }\n planarUVMap(vertex, uv) {\n let u = vertex.x;\n let v = -vertex.z;\n u = u + 0.5;\n v = v + 0.5;\n uv.set(u, v);\n return uv;\n }\n cylindricalUVMap(vertex, uv) {\n let lon;\n if (vertex.x === 0 && vertex.z === 0) {\n lon = 0;\n }\n else {\n lon = Math.atan2(vertex.x, vertex.z);\n if (lon < 0) {\n lon += Math.PI * 2;\n }\n }\n const t = vertex.y + 0.5;\n lon = lon / (Math.PI * 2);\n uv.set(lon, t);\n return uv;\n }\n boxUVMap(vertex, normal, uv) {\n let s = 0, t = 0;\n const x = vertex.x, y = vertex.y, z = vertex.z;\n const nx = Math.abs(normal.x), ny = Math.abs(normal.y), nz = Math.abs(normal.z);\n if (nx >= ny && nx >= nz) {\n s = -z + 0.5;\n t = y + 0.5;\n }\n if (ny >= nx && ny >= nz) {\n s = x + 0.5;\n t = -z + 0.5;\n }\n if (nz >= nx && nz >= ny) {\n s = x + 0.5;\n t = y + 0.5;\n }\n // not sure what this is for but it screws up the mapping and flips the uv's halfway through\n // if (normal.x < 0) {\n // s = -s + 1\n // }\n // if (normal.y < 0) {\n // s = -s + 1\n // }\n // if (normal.z < 0) {\n // s = -s + 1\n // }\n uv.set(s, t);\n return uv;\n }\n sphericalUVMap(vertex, uv) {\n const newVertex = vertex.clone();\n const lonlat = xyzToLongitudeLatitude(newVertex.x, -newVertex.y, newVertex.z);\n const lon = lonlat.lon / (Math.PI * 2);\n const lat = 0.5 - lonlat.lat / Math.PI;\n uv.set(lon, lat);\n return uv;\n function xyzToLongitudeLatitude(x, y, z) {\n let lo, la;\n const EPSILON = 1e-6;\n if (Math.abs(x) < EPSILON && Math.abs(z) < EPSILON) {\n lo = 0;\n if (Math.abs(y) < EPSILON) {\n la = 0;\n }\n else {\n la = y < 0 ? -(Math.PI / 2) : Math.PI / 2;\n }\n }\n else {\n lo = Math.atan2(x, z);\n if (lo < 0) {\n lo += Math.PI * 2;\n }\n const h = Math.sqrt(x * x + z * z);\n la = Math.atan2(y, h);\n }\n return {\n lon: lo,\n lat: la,\n };\n }\n }\n }\n function createUvMapVisualization(uvMap, scene, color) {\n let uvMapVisualization;\n if (!color) {\n color = Color3.Yellow();\n }\n const coloredMaterial = new StandardMaterial('', scene);\n coloredMaterial.wireframe = true;\n coloredMaterial.emissiveColor = color;\n if (uvMap.mode === exports.eUvMode.spherical) {\n uvMapVisualization = CreateSphere('UvMapSphere', { segments: 6 }, scene);\n }\n else if (uvMap.mode === exports.eUvMode.cylindrical) {\n uvMapVisualization = CreateCylinder('UvMapCylinder', { tessellation: 10 }, scene);\n }\n else if (uvMap.mode === exports.eUvMode.box) {\n uvMapVisualization = CreateBox('UvBox', {}, scene);\n }\n else if (uvMap.mode === exports.eUvMode.planar) {\n uvMapVisualization = CreatePlane('mirrorPlane', {}, scene);\n uvMapVisualization.rotation = new Vector3(Math.PI / 2, 0, 0);\n uvMapVisualization.bakeCurrentTransformIntoVertices();\n }\n else {\n uvMapVisualization = CreatePlane('mirrorPlane', {}, scene);\n uvMapVisualization.bakeCurrentTransformIntoVertices();\n uvMapVisualization.setEnabled(false);\n }\n uvMapVisualization.material = coloredMaterial;\n return uvMapVisualization;\n }\n\n function removeVerticesFromAll(vertexGroups, startIndex, endIndex) {\n for (let i = 0; i < vertexGroups.length; i++) {\n const vertexGroup = vertexGroups[i];\n if (vertexGroupsIntersect(vertexGroup, startIndex, endIndex)) {\n const result = excludeVertices(vertexGroup, startIndex, endIndex);\n vertexGroups.splice(i, 1, ...result);\n }\n }\n }\n function mergeVerticesIntoAll(vertexGroups, newVertex) {\n vertexGroups.push(newVertex);\n vertexGroups.sort((a, b) => a.startIndex - b.startIndex);\n for (let i = 0; i < vertexGroups.length; i++) {\n const vertexGroup = vertexGroups[i];\n const nextVertexGroup = vertexGroups[i + 1];\n if (nextVertexGroup && nextVertexGroup.startIndex < vertexGroup.endIndex) {\n const mergedVertexGroup = { ...vertexGroup, endIndex: nextVertexGroup.endIndex };\n vertexGroups.splice(i, 2, mergedVertexGroup);\n i--;\n }\n }\n }\n function excludeVertices(vertexGroup, startIndex, endIndex) {\n if (vertexGroupsIntersect(vertexGroup, startIndex, endIndex)) {\n const result = [];\n // Maintain the selections of either side of the current vertex\n if (startIndex > vertexGroup.startIndex) {\n result.push({\n ...vertexGroup,\n endIndex: startIndex,\n });\n }\n if (endIndex < vertexGroup.endIndex) {\n result.push({\n ...vertexGroup,\n startIndex: endIndex,\n });\n }\n return result;\n }\n else {\n return [vertexGroup];\n }\n }\n function mergeVertices(vertexGroup, newVertex) {\n const result = [];\n const newVertexGroup = { ...vertexGroup };\n const startIndex = newVertex.startIndex;\n const endIndex = newVertex.endIndex;\n if (vertexGroupsIntersect(vertexGroup, startIndex, endIndex)) {\n // Maintain the selections of either side of the current vertex\n if (startIndex < vertexGroup.startIndex) {\n newVertexGroup.startIndex = startIndex;\n }\n if (endIndex > vertexGroup.endIndex) {\n newVertexGroup.endIndex = endIndex;\n }\n result.push(newVertexGroup);\n }\n else {\n if (startIndex > vertexGroup.startIndex) {\n result.push(newVertexGroup, newVertex);\n }\n else {\n result.push(newVertex, newVertexGroup);\n }\n }\n return result;\n }\n function vertexGroupsIntersect(vertexGroup, startIndex, endIndex) {\n return startIndex <= vertexGroup.endIndex && endIndex >= vertexGroup.startIndex;\n }\n\n function createViewpointVisualization(viewpoint, scene, color) {\n if (!color) {\n color = Color3.Green();\n }\n let targetPositionMarker;\n const viewpointPositionGizmo = MeshBuilder.CreateCylinder('cameraGizmoLens', { diameterTop: 0.3, height: 0.2, diameterBottom: 0.2, tessellation: 8 }, scene);\n const coloredMaterial = new StandardMaterial('', scene);\n coloredMaterial.wireframe = true;\n coloredMaterial.emissiveColor = color;\n viewpointPositionGizmo.material = coloredMaterial;\n const lightPositionGizmoBox = MeshBuilder.CreateBox('cameraGizmoBody', { size: 0.4 }, scene);\n lightPositionGizmoBox.setParent(viewpointPositionGizmo);\n lightPositionGizmoBox.position = new Vector3(0, -0.3, 0);\n lightPositionGizmoBox.material = coloredMaterial;\n viewpointPositionGizmo.isPickable = false;\n viewpointPositionGizmo.setEnabled(true);\n if (viewpoint.type === exports.eViewpointType.orbit && viewpoint.targetMode === exports.eViewpointTargetMode.point) {\n targetPositionMarker = MeshBuilder.CreateSphere('viewpointTargetMarker', { diameter: 0.15 }, scene);\n const coloredMaterial = new StandardMaterial('', scene);\n coloredMaterial.emissiveColor = color;\n targetPositionMarker.material = coloredMaterial;\n targetPositionMarker.isPickable = false;\n targetPositionMarker.setEnabled(true);\n }\n return { viewpointPositionGizmo, targetPositionMarker };\n }\n\n /** The maxiumum amount of allowed time to spend in a single render cycle, to avoid locking the browser */\n const MAX_RENDER_TIME = 20000;\n class Renderer {\n constructor(viewer, scene) {\n this.viewer = viewer;\n this.scene = scene;\n this.babylonMap = new Map();\n }\n /**\n * Determines whether the given node should be handled by this renderer. Can be\n * overriden for special behavior\n */\n handlesNode(node) {\n return instanceOf(node, this.modelToken);\n }\n update3d(node) {\n this.update3dInternal(node);\n if (this.viewer.options.validator) {\n this.viewer.options.validator.validateNode(node);\n }\n }\n delete3d(node) {\n const o = this.babylonMap.get(node._dynamicId);\n if (o) {\n if (o instanceof Material) {\n o.dispose(true, true);\n }\n else {\n o.dispose();\n }\n }\n this.babylonMap.delete(node._dynamicId);\n SyncService.nodeDeletionProcessed(node);\n }\n get3d(node) {\n return this.babylonMap.get(node._dynamicId);\n }\n set3d(node, babylonObject) {\n this.babylonMap.set(node._dynamicId, babylonObject);\n }\n getChanges(node) {\n const isNew = this.get3d(node) == null;\n const changes = new Map(node.changes);\n if (isNew) {\n for (const prop in node) {\n changes.set(prop, node[prop]);\n }\n }\n return changes;\n }\n dispose() {\n this.babylonMap.clear();\n }\n }\n\n const TEXT_RESOLUTION = 128;\n class AnnotationRenderer extends Renderer {\n constructor(viewer, scene, rendererManager) {\n super(viewer, scene);\n this.viewer = viewer;\n this.scene = scene;\n this.rendererManager = rendererManager;\n this.modelToken = Token.AnnotationNode;\n this.annotationStore = new Map();\n this.billboards = new Map();\n this.utilityLayer = viewer.utilityLayer;\n scene.registerBeforeRender(() => {\n if (scene.activeCamera && this.billboards.size > 0) {\n this.billboards.forEach(meshInfo => {\n const camera = scene.activeCamera;\n meshInfo.rootMesh.lookAt(getCameraPositionWithOffset(camera), 0, Math.PI / 2);\n if (meshInfo.rootMesh.parent) {\n if (!meshInfo.rootMesh.rotationQuaternion) {\n meshInfo.rootMesh.rotationQuaternion = Quaternion.Identity();\n }\n const parentRotation = meshInfo.rootMesh.parent.computeWorldMatrix().getRotationMatrix();\n parentRotation.invert();\n const parentRotationQuat = Quaternion.FromRotationMatrix(parentRotation);\n meshInfo.rootMesh.rotationQuaternion = parentRotationQuat.multiplyInPlace(meshInfo.rootMesh.rotationQuaternion);\n }\n });\n }\n });\n }\n update3dInternal(node) {\n var _a, _b, _c, _d;\n let annotationInfo = this.annotationStore.get(node._dynamicId);\n if (!annotationInfo) {\n annotationInfo = {\n positionNeedsUpdate: false,\n textNeedsUpdate: false,\n lineNeedsUpdate: false,\n };\n this.annotationStore.set(node._dynamicId, annotationInfo);\n }\n if (annotationInfo.meshInfo && node.changes.has('keepOnTop')) {\n this.delete3d(node, true);\n annotationInfo.meshInfo = undefined;\n annotationInfo.positionNeedsUpdate = true;\n annotationInfo.textNeedsUpdate = true;\n annotationInfo.lineNeedsUpdate = true;\n }\n const scene = node.keepOnTop ? this.utilityLayer.utilityLayerScene : this.scene;\n if ((!annotationInfo.meshInfo && node.enabled && node.targetId) ||\n (node.changes.hasAny('isPickable') && node.enabled && node.targetId)) {\n if (annotationInfo.meshInfo) {\n annotationInfo.meshInfo = undefined;\n }\n annotationInfo.meshInfo = createAnnotationTextModels(node, scene);\n // Put the annotation on the parent scene so the scene visibility affects the node\n if (node._parentScene) {\n const parentSceneMesh = this.viewer.getRendererNode(node._parentScene);\n annotationInfo.meshInfo.rootMesh.setParent(parentSceneMesh);\n }\n this.set3d(node, annotationInfo.meshInfo.rootMesh);\n }\n const meshInfo = annotationInfo.meshInfo;\n // Set position of annotation\n if (node.changes.hasAny('targetId', 'offset', 'billboard', 'horizontalBoxAlign', 'verticalBoxAlign')) {\n annotationInfo.positionNeedsUpdate = true;\n }\n if (node.changes.hasAny('color', 'lineStyle')) {\n annotationInfo.lineNeedsUpdate = true;\n }\n if (node.changes.hasAny('fontFamily', 'fontSize', 'lineSpacing', 'label', 'color', 'preloadFontStylesheet', 'horizontalTextAlign')) {\n annotationInfo.textNeedsUpdate = true;\n }\n if (meshInfo) {\n if (node.changes.hasAny('enabled', 'billboard', 'keepOnTop')) {\n if (node.enabled && node.billboard) {\n this.billboards.set(node._dynamicId, meshInfo);\n }\n else {\n this.billboards.delete(node._dynamicId);\n }\n }\n let needsFont = false;\n if (node.preloadFontStylesheet && node.fontFamily) {\n const fontService = node._manager.getService(Token.FontService);\n if (!fontService.readyFonts.has(node.fontFamily)) {\n needsFont = true;\n fontService.loadFont(node.preloadFontStylesheet, [node.fontFamily]).then(resultFonts => {\n if (resultFonts.length > 0) {\n node.setPropertyIsModified('fontFamily', true);\n }\n });\n }\n }\n if (node.enabled && node.targetId && !needsFont) {\n const color = node.color.toColor4();\n if (annotationInfo.positionNeedsUpdate) {\n let targetNode;\n if (node.targetId && node._manager) {\n const manager = node._manager;\n targetNode = manager.getById(node.targetId);\n }\n if (annotationInfo.boundingInfo &&\n annotationInfo.boundingInfo.targetNode._dynamicId !== (targetNode === null || targetNode === void 0 ? void 0 : targetNode._dynamicId)) {\n annotationInfo.boundingInfo.subscription.unsubscribe();\n }\n if (targetNode) {\n if (targetNode instanceof exports.Connector) {\n // Only supporting connectors for now\n if (targetNode._parent) {\n annotationInfo.boundingInfo = {\n targetNode: targetNode,\n subscription: this.viewer.bbCache\n .watchForChanges(targetNode._parent)\n .subscribe(() => {\n node.setPropertyIsModified('targetId', true);\n }),\n };\n }\n }\n const transform = this.updatePositionFromAnnotation(node, targetNode, meshInfo);\n if (transform) {\n meshInfo.rootMesh.position = transform.position;\n // The annotation plane mesh position doesn't update in certain situations when the parent transform is moved,\n // so we have to manually update the world matrix\n meshInfo.rootMesh.getChildMeshes().forEach(mesh => mesh.computeWorldMatrix(true));\n meshInfo.textContainer.rotationQuaternion = transform.textRotation;\n if (!node.billboard) {\n meshInfo.rootMesh.rotationQuaternion = transform.rotation;\n }\n if (transform && transform.offset && transform.lineOffset && node.offset.length() > 0) {\n const basePoint = transform.position.subtract(transform.offset);\n meshInfo.line = {\n points: [basePoint, basePoint.add(transform.lineOffset)],\n updatable: true,\n colors: [color, color],\n instance: (_a = meshInfo.line) === null || _a === void 0 ? void 0 : _a.instance,\n };\n // If the line is not getting rebuilt, then just update the line position\n if (!annotationInfo.lineNeedsUpdate) {\n meshInfo.line.instance = drawLine(node, meshInfo.line, 1, scene);\n }\n }\n else {\n if ((_b = meshInfo.line) === null || _b === void 0 ? void 0 : _b.instance) {\n meshInfo.line.instance.dispose();\n }\n meshInfo.line = undefined;\n }\n }\n annotationInfo.positionNeedsUpdate = false;\n }\n }\n if (annotationInfo.lineNeedsUpdate) {\n if (meshInfo.line) {\n if (meshInfo.line.instance) {\n meshInfo.line.instance.dispose();\n meshInfo.line.instance = undefined;\n }\n meshInfo.line.colors = [];\n for (let i = 0; i < meshInfo.line.points.length; i++) {\n meshInfo.line.colors.push(color);\n }\n meshInfo.line.instance = drawLine(node, meshInfo.line, 1, scene);\n if (meshInfo.line.instance && node._parentScene) {\n const parentSceneMesh = this.viewer.getRendererNode(node._parentScene);\n meshInfo.line.instance.parent = parentSceneMesh;\n }\n annotationInfo.lineNeedsUpdate = false;\n }\n }\n if (annotationInfo.textNeedsUpdate) {\n updateAnnotationText(node, meshInfo.textMesh, meshInfo.textMaterial, this.scene.getEngine().webGLVersion < 2);\n annotationInfo.textNeedsUpdate = false;\n }\n }\n const enabled = node.enabled && !!node.targetId;\n if (meshInfo.rootMesh.isEnabled() !== enabled) {\n meshInfo.rootMesh.setEnabled(enabled);\n (_d = (_c = meshInfo.line) === null || _c === void 0 ? void 0 : _c.instance) === null || _d === void 0 ? void 0 : _d.setEnabled(enabled);\n }\n }\n }\n dispose() {\n this.billboards.clear();\n this.annotationStore.clear();\n super.dispose();\n }\n delete3d(node, keepInStore = false) {\n var _a;\n const annotationModels = this.annotationStore.get(node._dynamicId);\n if (annotationModels && annotationModels.meshInfo) {\n annotationModels.meshInfo.rootMesh.dispose();\n if ((_a = annotationModels.meshInfo.line) === null || _a === void 0 ? void 0 : _a.instance) {\n annotationModels.meshInfo.line.instance.dispose();\n }\n }\n if (!keepInStore) {\n this.annotationStore.delete(node._dynamicId);\n this.billboards.delete(node._dynamicId);\n }\n super.delete3d(node);\n }\n updatePositionFromAnnotation(node, targetNode, meshInfo) {\n let newPosition;\n let newDirection;\n let newAngle;\n // Only supporting connectors for now, but could support direct mesh targets in the future\n if (targetNode instanceof exports.Connector) {\n newPosition = getPositionFromConnnector(targetNode, this.rendererManager, node._parentScene);\n newDirection = targetNode.direction.toVec3();\n newAngle = targetNode.angle;\n }\n if (newPosition && newDirection && newAngle !== undefined) {\n const newOffset = node.offset.toVec3();\n if (meshInfo) {\n const newPivot = new Vector3(0, 0, 0);\n if (node.horizontalBoxAlign == 'left') {\n meshInfo.rootMesh.getPivotPoint();\n newOffset.x += meshInfo.textMesh.scaling.x / 2;\n newPivot.x -= meshInfo.textMesh.scaling.x / 2;\n }\n else if (node.horizontalBoxAlign == 'right') {\n newOffset.x -= meshInfo.textMesh.scaling.x / 2;\n newPivot.x += meshInfo.textMesh.scaling.x / 2;\n }\n if (node.verticalBoxAlign == 'top') {\n newOffset.z += meshInfo.textMesh.scaling.z / 2;\n newPivot.z -= meshInfo.textMesh.scaling.z / 2;\n }\n else if (node.verticalBoxAlign == 'bottom') {\n newOffset.z -= meshInfo.textMesh.scaling.z / 2;\n newPivot.z += meshInfo.textMesh.scaling.z / 2;\n }\n meshInfo.rootMesh.setPivotPoint(newPivot);\n }\n const transform = calculateTransform(newPosition, newDirection, newAngle, newOffset, node.offset.toVec3());\n return transform;\n }\n }\n }\n function createAnnotationTextModels(node, scene, isPickable = false) {\n const containerMesh = new TransformNode(node._dynamicId, scene);\n //Create plane\n const plane = PlaneBuilder.CreatePlane(`${node._dynamicId}_plane`, {\n width: 1,\n height: 1,\n frontUVs: new Vector4(1, 0, 0, 1),\n backUVs: new Vector4(0, 0, 1, 1),\n sideOrientation: Mesh.DOUBLESIDE,\n sourcePlane: new Plane(0, 1, 0, 0),\n }, scene);\n const planeContainer = new TransformNode(`${node._dynamicId}_planeContainer`);\n const dynamicTexture = new DynamicTexture(`${node._dynamicId}_planeTexture`, { width: TEXT_RESOLUTION, height: TEXT_RESOLUTION }, scene, true);\n dynamicTexture.isBlocking = false;\n dynamicTexture.hasAlpha = true;\n //create material\n const mat = new PBRMaterial(`${node._dynamicId}_mat`, scene);\n if (node.keepOnTop) {\n mat.emissiveColor = Color3.White();\n }\n mat.albedoTexture = dynamicTexture;\n mat.useAlphaFromAlbedoTexture = true;\n mat.backFaceCulling = true;\n mat.zOffset = -1;\n mat.albedoColor = Color3.White();\n mat.roughness = 1;\n mat.metallic = 0;\n //apply material\n plane.material = mat;\n plane.parent = planeContainer;\n planeContainer.parent = containerMesh;\n plane.isPickable = isPickable || node.isPickable;\n const annotationModels = {\n textMesh: plane,\n textContainer: planeContainer,\n textMaterial: mat,\n rootMesh: containerMesh,\n billboard: node.billboard,\n positionNeedsUpdate: false,\n textNeedsUpdate: false,\n };\n return annotationModels;\n }\n function updateAnnotationText(node, mesh, textMaterial, webgl1Context, defaultValue = ' ', values = {}) {\n const mat = textMaterial;\n const dynamicTexture = mat.albedoTexture;\n const fontFamily = node.fontFamily || 'Arial';\n const labelSize = node.fontSize;\n const labelText = evalHandlebarString(node.label || defaultValue, values);\n const textGap = node.lineSpacing;\n const textColor = node.color.toColor3().toLinearSpace();\n if (node.keepOnTop) {\n mat.emissiveColor = textColor;\n }\n // Determine the space needed for the text\n const ctx = dynamicTexture.getContext();\n ctx.font = TEXT_RESOLUTION + 'px ' + fontFamily;\n const labelStats = getMultiLineStats(labelText, ctx);\n const textWidth = labelStats.maximumWidth;\n const textHeight = labelStats.maximumHeight;\n const labelArray = labelStats.labelArray;\n const lineWidths = labelStats.lineWidths;\n labelArray.reverse();\n lineWidths.reverse();\n const totalCanvasHeight = (textHeight + textGap) * labelArray.length - textGap;\n ctx.clearRect(0 - textWidth / 2, 0, textWidth / 2, totalCanvasHeight);\n //ctx.clearRect(0, 0, textWidth, totalCanvasHeight);\n let heightRatioChange = 1;\n if (labelArray.length > 1) {\n heightRatioChange = totalCanvasHeight / textHeight;\n }\n mesh.scaling = new Vector3((textWidth / textHeight) * labelSize, labelSize * heightRatioChange, 1);\n let textBaseline = 'bottom';\n if (labelStats.maximumDescent > 0) {\n textBaseline = 'alphabetic';\n }\n let drawFont;\n if (webgl1Context) {\n let textureSize = getBinClampedSize(textWidth, totalCanvasHeight, 'ceil');\n if (textureSize > 2048) {\n textureSize = 2048;\n }\n dynamicTexture.scaleTo(textureSize, textureSize);\n ctx.scale(textureSize / textWidth, textureSize / totalCanvasHeight);\n drawFont = TEXT_RESOLUTION + 'px ' + fontFamily;\n }\n else {\n dynamicTexture.scaleTo(textWidth, totalCanvasHeight);\n drawFont = TEXT_RESOLUTION + 'px ' + fontFamily;\n }\n ctx.font = drawFont;\n if (ctx instanceof CanvasRenderingContext2D) {\n ctx.textBaseline = textBaseline;\n }\n ctx.fillStyle = textColor.toHexString();\n let drawHeight = totalCanvasHeight;\n for (let i = 0; i < labelArray.length; i++) {\n const lineWidth = lineWidths[i];\n //ctx.fillStyle = 'rgb(132,244,235, 0.3)';\n //ctx.fillRect(0, 0, textWidth, (textHeight + textGap) * labelArray.length - textGap);\n //rightAllign\n if (node.horizontalTextAlign === 'right') {\n ctx.fillText(labelArray[i], textWidth - lineWidth, drawHeight - labelStats.maximumDescent);\n }\n //leftAllign\n else if (node.horizontalTextAlign === 'left') {\n ctx.fillText(labelArray[i], 0, drawHeight - labelStats.maximumDescent);\n }\n else {\n //centered:\n ctx.fillText(labelArray[i], textWidth / 2 - lineWidth / 2, drawHeight - labelStats.maximumDescent);\n }\n drawHeight = drawHeight - textHeight - textGap;\n }\n // reset the context\n ctx.setTransform(1, 0, 0, 1, 0, 0);\n dynamicTexture.update(true);\n }\n function getPositionFromConnnector(targetNode, rendererManager, relativeToScene) {\n const targetMesh = rendererManager.connector.get3d(targetNode);\n if (targetMesh) {\n targetMesh.computeWorldMatrix(true);\n const result = targetMesh.absolutePosition.clone();\n if (relativeToScene) {\n const sceneMesh = rendererManager.viewer.getRendererNode(relativeToScene);\n if (sceneMesh) {\n // console.log('pre', newPosition, newDirection, newAngle);\n const transform = sceneMesh.getWorldMatrix().clone();\n transform.invert();\n return Vector3.TransformCoordinates(result, transform);\n }\n }\n return result;\n }\n return Vector3.Zero();\n }\n function calculateTransform(position, direction, angle, offset, lineOffset) {\n const newPosition = position.clone();\n const newRotation = rotationToVector(direction);\n const textRotation = Quaternion.RotationAxis(Vector3.Up(), (angle - 90) * (Math.PI / 180));\n let offsetWithTransform;\n let lineOffsetWithTransform;\n if (offset) {\n offsetWithTransform = transformVectorWithQuat(offset, newRotation);\n newPosition.addInPlace(offsetWithTransform);\n }\n if (lineOffset) {\n lineOffsetWithTransform = transformVectorWithQuat(lineOffset, newRotation);\n }\n return {\n position: newPosition,\n rotation: newRotation,\n textRotation: textRotation,\n offset: offsetWithTransform,\n lineOffset: lineOffsetWithTransform,\n };\n }\n function drawLine(node, lineInfo, lineLength, scene, distance) {\n if (lineInfo && node.lineStyle !== exports.eAnnotationLineStyle.none) {\n let linesMesh;\n if (node.lineStyle === exports.eAnnotationLineStyle.solid) {\n linesMesh = MeshBuilder.CreateLines(`${node._dynamicId}_line`, lineInfo, scene);\n }\n else {\n lineInfo.dashNb = lineLength * 30;\n if (node.lineStyle === exports.eAnnotationLineStyle.dashed) {\n lineInfo.gapSize = 4;\n lineInfo.dashSize = 6;\n if (distance) {\n // lineInfo.gapSize *= distance;\n // lineInfo.dashSize *= distance;\n lineInfo.dashNb = (lineInfo.dashNb / distance) * 1.5;\n }\n }\n else {\n lineInfo.gapSize = 6;\n lineInfo.dashSize = 1;\n }\n linesMesh = MeshBuilder.CreateDashedLines(`${node._dynamicId}_line`, lineInfo, scene);\n linesMesh.color = node.color.toColor3();\n }\n lineInfo.instance = linesMesh;\n lineInfo.instance.isPickable = false;\n return linesMesh;\n }\n }\n function evalHandlebarString(label, values) {\n if (!label.match(/\\{.*\\}/)) {\n return label;\n }\n const array = label.matchAll(/\\{([\\d\\w+\\-*/]+)\\}/g);\n let exp = array.next();\n while (!exp.done) {\n let mathExp = exp.value[1];\n for (const key in values) {\n mathExp = mathExp.replace(key, values[key]);\n }\n if (mathExp.match(/\\d/)) {\n // Strip all invalid math expression characters\n mathExp = mathExp.replace(/[^0-9.+*/-]/g, '');\n const numberVal = mathEval(mathExp);\n label = label.replace(exp.value[0], roundToPlaces(numberVal, 0.01).toString(10));\n }\n else {\n label = label.replace(exp.value[0], mathExp);\n }\n exp = array.next();\n }\n return label;\n }\n function getMultiLineStats(label, ctx) {\n // 402 is a florin and 96 is a backtick, generally the tallest and highest characters.\n const fullStringTextSize = ctx.measureText(label + String.fromCharCode(402, 71, 95, 96, 103));\n const substrings = label.split('\\\\n').map(label => label.trim());\n const lineWidths = [];\n let maximumWidth = 0;\n let maximumHeight = 0;\n let maximumDescent = 0;\n substrings.forEach(string => {\n if (string.length > maximumWidth) {\n maximumWidth = string.length;\n }\n const textSize = ctx.measureText(string);\n if (textSize.actualBoundingBoxLeft !== undefined && textSize.actualBoundingBoxRight !== undefined) {\n const textWidth = Math.abs(textSize.actualBoundingBoxLeft) + Math.abs(textSize.actualBoundingBoxRight);\n if (textWidth > maximumWidth) {\n maximumWidth = textWidth;\n }\n lineWidths.push(textWidth);\n }\n else {\n if (textSize.width > maximumWidth) {\n maximumWidth = textSize.width;\n }\n lineWidths.push(textSize.width);\n }\n });\n // The render puppetteer browser does not have fontBoundingBoxAscent or Descent.\n // That would be a better way to measure text for consistency. When upgrading puppeteer see if we can use fontBoundingBoxAscent\n if (fullStringTextSize.actualBoundingBoxAscent !== undefined &&\n fullStringTextSize.actualBoundingBoxDescent !== undefined) {\n maximumHeight =\n Math.abs(fullStringTextSize.actualBoundingBoxAscent) +\n Math.abs(fullStringTextSize.actualBoundingBoxDescent);\n maximumDescent = fullStringTextSize.actualBoundingBoxDescent;\n }\n if (maximumHeight === 0) {\n maximumHeight = TEXT_RESOLUTION;\n }\n const result = {\n labelArray: substrings,\n lineWidths,\n maximumHeight,\n maximumWidth,\n maximumDescent,\n };\n return result;\n }\n let loopCount = 0;\n function mathEval(exp) {\n loopCount++;\n if (loopCount > 1000) {\n console.warn('escaped infinite loop in math evaluator');\n return NaN;\n }\n if (!exp.match(/\\D/)) {\n return parseFloat(exp);\n }\n let result;\n if (exp.match(/\\+/)) {\n const parts = exp.split('+').reverse();\n result = mathEval(parts.pop());\n while (parts.length > 0) {\n result += mathEval(parts.pop());\n }\n }\n else if (exp.match(/-/)) {\n const parts = exp.split('-').reverse();\n result = mathEval(parts.pop());\n while (parts.length > 0) {\n result -= mathEval(parts.pop());\n }\n }\n else if (exp.match(/\\*/)) {\n const parts = exp.split('*').reverse();\n result = mathEval(parts.pop());\n while (parts.length > 0) {\n result *= mathEval(parts.pop());\n }\n }\n else if (exp.match(/\\//)) {\n const parts = exp.split('/').reverse();\n result = mathEval(parts.pop());\n while (parts.length > 0) {\n result /= mathEval(parts.pop());\n }\n }\n else {\n result = parseFloat(exp);\n }\n return result;\n }\n\n /**\n * Move line position calculation to private function so it can be called from bbSubscription\n * Move text font family creation stuff to common function and call it from here with midpoint of line\n * Store the line mesh in annotationStore, just recreate it from the private function when something changes\n */\n class DimensionRenderer extends Renderer {\n constructor(viewer, scene, rendererManager) {\n super(viewer, scene);\n this.viewer = viewer;\n this.scene = scene;\n this.rendererManager = rendererManager;\n this.modelToken = Token.DimensionNode;\n this.watchPosition = new Map();\n this.dimensionStore = new Map();\n this.billboards = new Map();\n this.utilityLayer = viewer.utilityLayer;\n scene.registerBeforeRender(() => {\n if (scene.activeCamera && this.dimensionStore.size > 0) {\n this.billboards.forEach(meshInfo => {\n const camera = scene.activeCamera;\n meshInfo.rootMesh.lookAt(getCameraPositionWithOffset(camera), 0, Math.PI / 2);\n });\n }\n });\n }\n update3dInternal(node) {\n var _a, _b, _c;\n let dimensionInfo = this.dimensionStore.get(node._dynamicId);\n if (!dimensionInfo) {\n dimensionInfo = {\n positionNeedsUpdate: false,\n textNeedsUpdate: false,\n lineNeedsUpdate: false,\n canCreate: false,\n };\n this.dimensionStore.set(node._dynamicId, dimensionInfo);\n }\n const rootMesh = (_a = dimensionInfo.meshInfo) === null || _a === void 0 ? void 0 : _a.rootMesh;\n if (rootMesh && node.scaleWithCamera) {\n if (!node.scaleWithCamera) {\n this.viewer.camera.clearMaintainedNodeScreenSize(rootMesh);\n }\n else {\n this.viewer.camera.maintainNodeScreenSize(rootMesh, (dist, mode) => {\n if (dimensionInfo && dimensionInfo.meshInfo && node.lineStyle === exports.eAnnotationLineStyle.dashed) {\n this.updateLineFromCamera(node, dimensionInfo.meshInfo, scene, dist);\n }\n });\n }\n }\n if (dimensionInfo.meshInfo && node.changes.has('keepOnTop')) {\n this.delete3d(node, true);\n dimensionInfo.meshInfo = undefined;\n dimensionInfo.positionNeedsUpdate = true;\n dimensionInfo.textNeedsUpdate = true;\n dimensionInfo.lineNeedsUpdate = true;\n }\n const scene = node.keepOnTop ? this.utilityLayer.utilityLayerScene : this.scene;\n if (!dimensionInfo.meshInfo && node.enabled && node.targetId) {\n dimensionInfo.meshInfo = {\n ...createAnnotationTextModels(node, scene, this.viewer.options.editorMode),\n line: {\n points: [],\n colors: [],\n updatable: true,\n },\n lineCap: {\n lines: [],\n colors: [],\n updatable: true,\n },\n distance: 0,\n };\n // Put the dimensions on the parent scene so the scene visibility affects the node\n if (node._parentScene) {\n const parentSceneMesh = this.viewer.getRendererNode(node._parentScene);\n dimensionInfo.meshInfo.rootMesh.setParent(parentSceneMesh);\n }\n this.set3d(node, dimensionInfo.meshInfo.rootMesh);\n }\n const meshInfo = dimensionInfo.meshInfo;\n // Set position of annotation\n if (node.changes.hasAny('enabled', 'targetId', 'target2Id', 'offset', 'billboard', 'labelOffset', 'axis', 'dimensionGap')) {\n dimensionInfo.positionNeedsUpdate = true;\n }\n if (node.changes.hasAny('color', 'lineStyle', 'lineCapStyle')) {\n dimensionInfo.lineNeedsUpdate = true;\n }\n if (node.changes.hasAny('fontFamily', 'fontSize', 'label', 'color', 'targetId', 'target2Id', 'dimensionUnitPrecision')) {\n dimensionInfo.textNeedsUpdate = true;\n }\n if (meshInfo) {\n if (node.changes.hasAny('enabled', 'billboard', 'keepOnTop')) {\n if (node.enabled && node.billboard) {\n this.billboards.set(node._dynamicId, meshInfo);\n }\n else {\n this.billboards.delete(node._dynamicId);\n }\n }\n if (node.enabled) {\n if (dimensionInfo.positionNeedsUpdate || dimensionInfo.lineNeedsUpdate) {\n dimensionInfo.canCreate = this.updateDimensionControlPoints(node, dimensionInfo);\n // If the line is not getting rebuilt, then just update the line position\n if (dimensionInfo.canCreate && !dimensionInfo.lineNeedsUpdate) {\n meshInfo.line.instance = drawLine(node, meshInfo.line, 1, scene);\n meshInfo.lineCap.instance = MeshBuilder.CreateLineSystem(`${node._dynamicId}_lineCap`, meshInfo.lineCap, scene);\n }\n dimensionInfo.positionNeedsUpdate = false;\n }\n if (dimensionInfo.canCreate) {\n if (dimensionInfo.lineNeedsUpdate) {\n if (meshInfo.line.instance) {\n meshInfo.line.instance.dispose();\n delete meshInfo.line.instance;\n }\n if (meshInfo.lineCap.instance) {\n meshInfo.lineCap.instance.dispose();\n delete meshInfo.lineCap.instance;\n }\n this.updateDimensionLineColor(node, meshInfo.line, meshInfo.lineCap);\n meshInfo.line.instance = drawLine(node, meshInfo.line, meshInfo.distance, scene);\n if (node.lineCapStyle !== exports.eDimensionLineCapStyle.none) {\n const lineCapsMesh = MeshBuilder.CreateLineSystem(`${node._dynamicId}_lineCap`, meshInfo.lineCap, scene);\n if (!meshInfo.lineCap.instance) {\n meshInfo.lineCap.instance = lineCapsMesh;\n }\n }\n // Put the lines on the parent scene so they get it's visibility properties\n if (node._parentScene) {\n const parentSceneMesh = this.viewer.getRendererNode(node._parentScene);\n if (meshInfo.line.instance) {\n meshInfo.line.instance.parent = parentSceneMesh;\n }\n if (meshInfo.lineCap.instance) {\n meshInfo.lineCap.instance.parent = parentSceneMesh;\n }\n }\n dimensionInfo.lineNeedsUpdate = false;\n }\n if (dimensionInfo.textNeedsUpdate) {\n let distanceDefaultVal;\n if (node._parentScene &&\n node._parentScene.dimensionUnitScalar &&\n node.dimensionUnitPrecision !== undefined) {\n let distance = node._parentScene.dimensionUnitScalar * meshInfo.distance;\n const precisionMultiplier = Math.pow(10, Math.abs(node.dimensionUnitPrecision));\n distance =\n Math.round((distance + Number.EPSILON) * precisionMultiplier) / precisionMultiplier;\n distanceDefaultVal = distance + (node._parentScene.dimensionUnitSymbol || '');\n }\n else {\n let distance = meshInfo.distance;\n const precisionMultiplier = Math.pow(10, Math.abs(node.dimensionUnitPrecision));\n distance =\n Math.round((distance + Number.EPSILON) * precisionMultiplier) / precisionMultiplier;\n distanceDefaultVal = distance.toString(10);\n }\n updateAnnotationText(node, meshInfo.textMesh, meshInfo.textMaterial, this.scene.getEngine().webGLVersion < 2, distanceDefaultVal, { v: meshInfo.distance });\n dimensionInfo.textNeedsUpdate = false;\n }\n }\n }\n if (meshInfo.rootMesh.isEnabled() !== node.enabled) {\n meshInfo.rootMesh.setEnabled(node.enabled);\n (_b = meshInfo.line.instance) === null || _b === void 0 ? void 0 : _b.setEnabled(node.enabled);\n (_c = meshInfo.lineCap.instance) === null || _c === void 0 ? void 0 : _c.setEnabled(node.enabled);\n }\n }\n }\n delete3d(node, keepInStore = false) {\n const dimensionModels = this.dimensionStore.get(node._dynamicId);\n if (dimensionModels === null || dimensionModels === void 0 ? void 0 : dimensionModels.meshInfo) {\n this.deleteLine(dimensionModels.meshInfo);\n }\n if (!keepInStore) {\n this.dimensionStore.delete(node._dynamicId);\n this.billboards.delete(node._dynamicId);\n }\n super.delete3d(node);\n }\n dispose() {\n this.billboards.clear();\n this.dimensionStore.clear();\n super.dispose();\n }\n deleteLine(dimensionModels) {\n if (dimensionModels.line.instance) {\n dimensionModels.line.instance.dispose();\n delete dimensionModels.line.instance;\n }\n if (dimensionModels.lineCap.instance) {\n dimensionModels.lineCap.instance.dispose();\n delete dimensionModels.lineCap.instance;\n }\n }\n updateLine(dimensionInfo, node, meshInfo, scene) {\n dimensionInfo.canCreate = this.updateDimensionControlPoints(node, dimensionInfo);\n // If the line is not getting rebuilt, then just update the line position\n if (dimensionInfo.canCreate && !dimensionInfo.lineNeedsUpdate) {\n meshInfo.line.instance = drawLine(node, meshInfo.line, 1, scene);\n meshInfo.lineCap.instance = MeshBuilder.CreateLineSystem(`${node._dynamicId}_lineCap`, meshInfo.lineCap, scene);\n }\n dimensionInfo.positionNeedsUpdate = false;\n if (dimensionInfo.canCreate) {\n if (dimensionInfo.lineNeedsUpdate) {\n if (meshInfo.line.instance) {\n meshInfo.line.instance.dispose();\n delete meshInfo.line.instance;\n }\n if (meshInfo.lineCap.instance) {\n meshInfo.lineCap.instance.dispose();\n delete meshInfo.lineCap.instance;\n }\n this.updateDimensionLineColor(node, meshInfo.line, meshInfo.lineCap);\n meshInfo.line.instance = drawLine(node, meshInfo.line, meshInfo.distance, scene);\n if (node.lineCapStyle !== exports.eDimensionLineCapStyle.none) {\n const lineCapsMesh = MeshBuilder.CreateLineSystem(`${node._dynamicId}_lineCap`, meshInfo.lineCap, scene);\n if (!meshInfo.lineCap.instance) {\n meshInfo.lineCap.instance = lineCapsMesh;\n }\n }\n // Put the lines on the parent scene so they get it's visibility properties\n if (node._parentScene) {\n const parentSceneMesh = this.viewer.getRendererNode(node._parentScene);\n if (meshInfo.line.instance) {\n meshInfo.line.instance.parent = parentSceneMesh;\n }\n if (meshInfo.lineCap.instance) {\n meshInfo.lineCap.instance.parent = parentSceneMesh;\n }\n }\n dimensionInfo.lineNeedsUpdate = false;\n }\n if (dimensionInfo.textNeedsUpdate) {\n let distanceDefaultVal;\n if (node._parentScene &&\n node._parentScene.dimensionUnitScalar &&\n node.dimensionUnitPrecision !== undefined) {\n let distance = node._parentScene.dimensionUnitScalar * meshInfo.distance;\n const precisionMultiplier = Math.pow(10, Math.abs(node.dimensionUnitPrecision));\n distance = Math.round((distance + Number.EPSILON) * precisionMultiplier) / precisionMultiplier;\n distanceDefaultVal = distance + (node._parentScene.dimensionUnitSymbol || '');\n }\n else {\n let distance = meshInfo.distance;\n const precisionMultiplier = Math.pow(10, Math.abs(node.dimensionUnitPrecision));\n distance = Math.round((distance + Number.EPSILON) * precisionMultiplier) / precisionMultiplier;\n distanceDefaultVal = distance.toString(10);\n }\n updateAnnotationText(node, meshInfo.textMesh, meshInfo.textMaterial, this.scene.getEngine().webGLVersion < 2, distanceDefaultVal, { v: meshInfo.distance });\n dimensionInfo.textNeedsUpdate = false;\n }\n }\n }\n updateLineFromCamera(node, meshInfo, scene, distance) {\n if (meshInfo.line.instance) {\n meshInfo.line.instance.dispose();\n delete meshInfo.line.instance;\n }\n this.updateDimensionLineColor(node, meshInfo.line, meshInfo.lineCap);\n meshInfo.line.instance = drawLine(node, meshInfo.line, 1, scene, distance); //CreateDashedLines('lines', meshInfo.line);\n if (node._parentScene) {\n const parentSceneMesh = this.viewer.getRendererNode(node._parentScene);\n if (meshInfo.line.instance) {\n meshInfo.line.instance.parent = parentSceneMesh;\n }\n if (meshInfo.lineCap.instance) {\n meshInfo.lineCap.instance.parent = parentSceneMesh;\n }\n }\n }\n updateDimensionControlPoints(node, dimensionInfo) {\n if (!dimensionInfo.meshInfo) {\n return false;\n }\n const dimensionModel = dimensionInfo.meshInfo;\n const lineOptions = dimensionModel.line;\n const annotationMesh = dimensionModel.rootMesh;\n let target1;\n let target2;\n if (node.targetId && node.target2Id && node._manager) {\n const manager = node._manager;\n target1 = manager.getById(node.targetId);\n target2 = manager.getById(node.target2Id);\n }\n if (dimensionInfo.boundingInfo &&\n (dimensionInfo.boundingInfo.targetNode._dynamicId !== (target1 === null || target1 === void 0 ? void 0 : target1._dynamicId) ||\n dimensionInfo.boundingInfo.targetNode2._dynamicId !== (target2 === null || target2 === void 0 ? void 0 : target2._dynamicId))) {\n dimensionInfo.boundingInfo.subscription.unsubscribe();\n dimensionInfo.boundingInfo = undefined;\n }\n if (target1 && target2) {\n let position1;\n let position2;\n let direction;\n let angleAdjustment = 0;\n // Only supporting connectors for now, but could support direct mesh targets in the future\n if (target1 instanceof exports.Connector && target2 instanceof exports.Connector) {\n position1 = getPositionFromConnnector(target1, this.rendererManager, node._parentScene);\n position2 = getPositionFromConnnector(target2, this.rendererManager, node._parentScene);\n direction = target1.direction.toVec3().normalize();\n angleAdjustment = target1.angle;\n if (target1._parent && target2._parent) {\n dimensionInfo.boundingInfo = {\n targetNode: target1,\n targetNode2: target2,\n subscription: merge(this.viewer.bbCache.watchForChanges(target1._parent), this.viewer.bbCache.watchForChanges(target2._parent)).subscribe(() => {\n node.setPropertyIsModified('targetId', true);\n node.setPropertyIsModified('target2Id', true);\n }),\n };\n }\n }\n if (position1 && position2 && direction) {\n /** Calculate the dimension line */\n const offset = node.offset.toVec3();\n const startConnector = calculateTransform(position1, direction, angleAdjustment, offset);\n const endConnector = calculateTransform(position2, direction, angleAdjustment, offset);\n const dimensionLine = endConnector.position.subtract(startConnector.position);\n const startPostion = startConnector.position;\n // Depending on the dimension axis, just use the\n let endPosition;\n if (node.axis === exports.eDimensionAxis.free) {\n endPosition = endConnector.position;\n }\n else {\n if (node.axis === exports.eDimensionAxis.x) {\n endPosition = startPostion.add(dimensionLine.multiply(new Vector3(1, 0, 0)));\n }\n else if (node.axis === exports.eDimensionAxis.y) {\n endPosition = startPostion.add(dimensionLine.multiply(new Vector3(0, 1, 0)));\n }\n else {\n endPosition = startPostion.add(dimensionLine.multiply(new Vector3(0, 0, 1)));\n }\n }\n const rectifiedDimensionLine = endPosition.subtract(startPostion);\n const lineDirectionCross = Vector3.Cross(rectifiedDimensionLine, direction).normalize();\n lineOptions.points = [startPostion, endPosition];\n dimensionModel.distance = rectifiedDimensionLine.length();\n annotationMesh.position = startPostion.add(endPosition).scaleInPlace(0.5);\n annotationMesh.position.addInPlace(transformVectorWithQuat(node.labelOffset.toVec3(), startConnector.rotation));\n dimensionModel.textContainer.rotationQuaternion = startConnector.textRotation;\n if (!node.billboard) {\n annotationMesh.rotationQuaternion = rotationToVector(lineDirectionCross);\n }\n /** Calculate the line caps */\n const capSize = direction.scale(node.fontSize / 3);\n const lineToStartConnector = direction.scale(Vector3.Dot(transformVectorWithQuat(offset, startConnector.rotation), direction) *\n (1 - node.dimensionGap));\n const endOffset = node.axis === exports.eDimensionAxis.free\n ? transformVectorWithQuat(offset, startConnector.rotation)\n : endPosition.subtract(position2);\n const lineToEndConnector = direction.scale(Vector3.Dot(endOffset, direction) * (1 - node.dimensionGap));\n dimensionModel.lineCap.lines = [];\n if (node.lineCapStyle !== exports.eDimensionLineCapStyle.none) {\n dimensionModel.lineCap.lines = [\n [startPostion.subtract(lineToStartConnector), startPostion, startPostion.add(capSize)],\n [endPosition.subtract(lineToEndConnector), endPosition, endPosition.add(capSize)],\n ];\n }\n if (node.lineCapStyle === exports.eDimensionLineCapStyle.arrow) {\n const capArrowOffset = rectifiedDimensionLine.normalizeToNew().scaleInPlace(capSize.length());\n dimensionModel.lineCap.lines.push([\n startPostion.subtract(capSize).addInPlace(capArrowOffset),\n startPostion,\n startPostion.add(capSize).addInPlace(capArrowOffset),\n ], [\n endPosition.subtract(capSize).subtractInPlace(capArrowOffset),\n endPosition,\n endPosition.add(capSize).subtractInPlace(capArrowOffset),\n ]);\n }\n return true;\n }\n }\n return false;\n }\n updateDimensionLineColor(node, lineOptions, lineCapOptions) {\n const color = node.color.toColor4();\n lineOptions.colors = [];\n for (let i = 0; i < lineOptions.points.length; i++) {\n lineOptions.colors.push(color);\n }\n lineCapOptions.colors = [];\n for (let i = 0; i < 4; i++) {\n const lineColor = [];\n lineCapOptions.colors.push(lineColor);\n for (let j = 0; j < 3; j++) {\n lineColor.push(color);\n }\n }\n }\n }\n function getDimensionLinePoints(node, rendererManager) {\n if (node.targetId && node.target2Id && node._manager) {\n const manager = node._manager;\n const target1 = manager.getById(node.targetId);\n const target2 = manager.getById(node.target2Id);\n let position1;\n let position2;\n let direction;\n if (target1 instanceof exports.Connector && target2 instanceof exports.Connector) {\n position1 = getPositionFromConnnector(target1, rendererManager, node._parentScene);\n position2 = getPositionFromConnnector(target2, rendererManager, node._parentScene);\n direction = target1.direction.toVec3();\n }\n if (position1 && position2 && direction) {\n const lineStart = calculateTransform(position1, direction, 0, node.offset.toVec3());\n const lineEnd = calculateTransform(position2, direction, 0, node.offset.toVec3());\n return [lineStart.position, lineEnd.position];\n }\n }\n return undefined;\n }\n\n const CONNECTOR_DIAM = 0.1;\n class ConnectorRenderer extends Renderer {\n constructor(viewer, scene, rendererManager) {\n super(viewer, scene);\n this.rendererManager = rendererManager;\n this.layers = new Map();\n this.subscriptionMap = new Map();\n this.controlPointScaleRatio = 0.24;\n this.modelToken = Token.Connector;\n this.layers.set('real', {\n template: null,\n map: this.babylonMap,\n scene,\n withArrows: !!viewer.options.editorMode,\n visibility: 1.0,\n pickable: !!viewer.options.editorMode,\n parts: new Map(),\n });\n this.layers.set('utility', {\n template: null,\n map: new Map(),\n scene: viewer.utilityLayer.utilityLayerScene,\n withArrows: !!viewer.options.editorMode,\n visibility: 0.3,\n pickable: false,\n parts: new Map(),\n });\n }\n handlesNode(node) {\n return instanceOf(node, Token.Connector);\n }\n update3dInternal(c, forceScreenScale = false) {\n /**\n * enabled, but not visible\n * enabled and visible\n * not enabled\n */\n const mesh = this.getMesh(c, this.layers.get('real'));\n const realLayer = this.layers.get('real');\n const utilityLayer = this.layers.get('utility');\n const utilityMesh = this.getMesh(c, this.layers.get('utility'));\n const realParts = realLayer.parts.get(c._dynamicId);\n const utilityParts = utilityLayer.parts.get(c._dynamicId);\n if (this.viewer.options.editorMode) {\n this.viewer.camera.maintainNodeScreenSize(mesh, dist => {\n var _a, _b, _c, _d, _e, _f;\n const scaledSize = dist * this.controlPointScaleRatio;\n realParts === null || realParts === void 0 ? void 0 : realParts.mesh.scaling.set(scaledSize, scaledSize, scaledSize);\n (_a = realParts === null || realParts === void 0 ? void 0 : realParts.handleContainer) === null || _a === void 0 ? void 0 : _a.scaling.set(1 / scaledSize, 1 / scaledSize, 1 / scaledSize);\n (_b = realParts === null || realParts === void 0 ? void 0 : realParts.directionHandle1) === null || _b === void 0 ? void 0 : _b.scaling.set(scaledSize, scaledSize, scaledSize);\n (_c = realParts === null || realParts === void 0 ? void 0 : realParts.directionHandle2) === null || _c === void 0 ? void 0 : _c.scaling.set(scaledSize, scaledSize, scaledSize);\n utilityParts === null || utilityParts === void 0 ? void 0 : utilityParts.mesh.scaling.set(scaledSize, scaledSize, scaledSize);\n (_d = utilityParts === null || utilityParts === void 0 ? void 0 : utilityParts.handleContainer) === null || _d === void 0 ? void 0 : _d.scaling.set(1 / scaledSize, 1 / scaledSize, 1 / scaledSize);\n (_e = utilityParts === null || utilityParts === void 0 ? void 0 : utilityParts.directionHandle1) === null || _e === void 0 ? void 0 : _e.scaling.set(scaledSize, scaledSize, scaledSize);\n (_f = utilityParts === null || utilityParts === void 0 ? void 0 : utilityParts.directionHandle2) === null || _f === void 0 ? void 0 : _f.scaling.set(scaledSize, scaledSize, scaledSize);\n });\n }\n const changes = this.getChanges(c);\n if (changes.has('_isPickable')) {\n mesh.isPickable = c._isPickable;\n }\n if (changes.has('positionZeroToOne') || changes.has('_sketchControlPoints')) {\n if (c.positionMode === exports.eConnectorPositionMode.moveAlongSketch) {\n if (c instanceof exports.Connector) {\n //this.updatePositionMoveAlongSketch(c);\n this.updatePosition(c);\n this.updateDirection(c);\n }\n }\n }\n if (changes.has('position') || changes.has('positionMode')) {\n const bb = c._parent instanceof exports.SpaceNode ? this.viewer.bbCache.get(c._parent) : undefined;\n this.updatePosition(c, bb);\n /**\n * subscribe to bounding box changes of the parent to know that\n * we need to invalidate the position of the connector if applicable\n */\n if (changes.has('positionMode')) {\n if (c.positionMode == exports.eConnectorPositionMode.boundingBox) {\n this.addSubscription(c);\n }\n else {\n this.removeSubscription(c);\n }\n }\n }\n if (changes.hasAny('direction', 'angle')) {\n this.updateDirection(c);\n this.updatePosition(c);\n }\n if (changes.hasAny('enabled', '_visible')) {\n if (c.enabled && c._visible) {\n mesh.setEnabled(true);\n utilityMesh.setEnabled(true);\n }\n else {\n mesh.setEnabled(false);\n utilityMesh.setEnabled(false);\n }\n }\n if (changes.has('_selected')) {\n if (c._selected) {\n mesh.material = this.getSelectedMaterial(this.layers.get('real').withArrows);\n }\n else {\n mesh.material = this.getConnectorMaterial(this.layers.get('real').withArrows);\n }\n utilityMesh.material = mesh.material;\n }\n return { mesh, utilityMesh };\n }\n addSubscription(c) {\n if (c._parent instanceof exports.SpaceNode) {\n const mesh = c._parent;\n if (!this.subscriptionMap.has(c)) {\n this.subscriptionMap.set(c, this.viewer.bbCache.watchForChanges(mesh).subscribe(bbInfo => {\n c.setPropertyIsModified('position', true); // Trigger changes on mates and everything using that connector\n this.updatePosition(c, bbInfo);\n this.updateScale(c);\n }));\n }\n }\n }\n removeSubscription(c) {\n const s = this.subscriptionMap.get(c);\n if (s) {\n s.unsubscribe();\n this.subscriptionMap.delete(c);\n }\n }\n updatePosition(c, bb) {\n const mesh = this.getMesh(c, this.layers.get('real'));\n const utilityMesh = this.getMesh(c, this.layers.get('utility'));\n if (c.positionMode == exports.eConnectorPositionMode.moveAlongSketch) {\n if (c instanceof exports.Connector) {\n const newPosition = this.updatePositionMoveAlongSketch(c);\n if (newPosition) {\n mesh.position = utilityMesh.position = newPosition;\n }\n }\n }\n else if (c.positionMode == exports.eConnectorPositionMode.absolute) {\n mesh.position = utilityMesh.position = c.position.toVec3();\n }\n else if (c.positionMode == exports.eConnectorPositionMode.boundingBox) {\n if (!bb) {\n bb = c._parent instanceof exports.SpaceNode ? this.viewer.bbCache.get(c._parent) : undefined;\n }\n if (bb) {\n mesh.position = BoundingBoxRelativeHelper.convertRelativeToAbsolute(c, bb);\n }\n else {\n mesh.position = new Vector3(c.position.x, c.position.y, c.position.z);\n }\n utilityMesh.position = mesh.position;\n }\n if (isTrackedClass(c)) {\n this.rendererManager.referenceProcessor.triggerChangesOnReferencers(c);\n }\n }\n updateDirection(c) {\n const mesh = this.getMesh(c, this.layers.get('real'));\n const utilityMesh = this.getMesh(c, this.layers.get('utility'));\n if (c.positionMode === exports.eConnectorPositionMode.moveAlongSketch) {\n if (c instanceof exports.Connector) {\n let newDirection = this.updateDirectionMoveAlongSketch(c);\n if (newDirection) {\n newDirection = newDirection.normalize();\n if (c._parent instanceof exports.SketchNode) {\n const normal = c._parent.normal;\n const normalVector = new Vector3(1, 0, 0);\n if (normal === exports.eAxis.y) {\n normalVector.set(0, 1, 0);\n }\n else if (normal === exports.eAxis.z) {\n normalVector.set(0, 0, 1);\n }\n if (mesh.parent instanceof TransformNode) {\n const connectorDirection = newDirection;\n const angleRot = Quaternion.RotationAxis(connectorDirection, toRadians(c.angle));\n const angleRotMatrix = new Matrix();\n angleRot.toRotationMatrix(angleRotMatrix);\n const parentDirection = Vector3.TransformCoordinates(normalVector, angleRotMatrix);\n const crossProduct = Vector3.Cross(connectorDirection, parentDirection);\n // prettier-ignore\n const matrix = Matrix.FromArray([connectorDirection.x, connectorDirection.y, connectorDirection.z, 0,\n parentDirection.x, parentDirection.y, parentDirection.z, 0,\n crossProduct.x, crossProduct.y, crossProduct.z, 0,\n 0, 0, 0, 1,\n ]);\n const newQuaternion = Quaternion.FromRotationMatrix(matrix);\n mesh.rotationQuaternion = utilityMesh.rotationQuaternion = newQuaternion;\n }\n }\n }\n }\n }\n else {\n /**\n * treat the positive x axis as the direction vector,\n * and the angle around that axis defined by the y axis.\n *\n * See Dragger.connect for a more in depth explanation\n */\n const deltaBtwXaxisAndDirection = getRotationBtwVectors(Axis.X, c.direction.toVec3().normalize());\n const angleRot = Quaternion.RotationAxis(Axis.X, toRadians(c.angle));\n mesh.rotationQuaternion = utilityMesh.rotationQuaternion = deltaBtwXaxisAndDirection.multiply(angleRot);\n }\n this.updateScale(c);\n }\n /**\n * connector meshes are children of their parent mesh, meaning they will scale\n * with any of their ancestor meshes or groups.\n * To counteract this we inverse their world scale.\n * WARNING! this technique does not work well when an ancestor has a non-uniform scale\n * @param c\n */\n updateScale(c) {\n if (this.layers.get('real').map.get(c._dynamicId)) {\n //this could get called at any time, make sure the mesh already exists\n const mesh = this.getMesh(c, this.layers.get('real'));\n const utilityMesh = this.getMesh(c, this.layers.get('utility'));\n mesh.scaling = Vector3.One();\n mesh.computeWorldMatrix(true);\n if (mesh.parent instanceof TransformNode) {\n const worldScale = mesh.parent.absoluteScaling;\n const numerator = (this.viewer.sceneNode.connectorSize || CONNECTOR_DIAM) / CONNECTOR_DIAM;\n const transformMatrix = Matrix.Scaling(numerator / worldScale.x, numerator / worldScale.y, numerator / worldScale.z);\n mesh.setPreTransformMatrix(transformMatrix);\n utilityMesh.setPreTransformMatrix(transformMatrix);\n if (isTrackedClass(c)) {\n this.rendererManager.referenceProcessor.triggerChangesOnReferencers(c);\n }\n }\n }\n }\n updatePositionMoveAlongSketch(connector) {\n const node = connector._parent;\n const result = new Vector3(0, 0, 0);\n if (node instanceof exports.SketchNode) {\n const sketchLength = node.getSketchLength();\n const posZeroToOne = connector.positionZeroToOne;\n let trackedLength = 0;\n let targetNotFound = true;\n node.paths.forEach(path => {\n let lastPosition = null;\n let lastLength = 0;\n path._points.forEach(point => {\n if (!lastPosition) {\n lastPosition = point;\n }\n else {\n const currentPos = point;\n const distance = Math.sqrt(Math.pow(lastPosition.x - currentPos.x, 2) +\n Math.pow(lastPosition.y - currentPos.y, 2) +\n Math.pow(lastPosition.z - currentPos.z, 2));\n lastLength = trackedLength;\n trackedLength += distance;\n const distanceBetweenPoints = trackedLength - lastLength;\n const distanceFromP1toTarget = posZeroToOne * sketchLength - lastLength;\n const percentageBetween2Points = distanceFromP1toTarget / distanceBetweenPoints;\n const vectorBetween2Points = new exports.KbVector(currentPos.x - lastPosition.x, currentPos.y - lastPosition.y, currentPos.z - lastPosition.z);\n if (trackedLength > posZeroToOne * sketchLength && targetNotFound) {\n const calculatedPosition = vectorBetween2Points\n .scale(percentageBetween2Points)\n .add(lastPosition);\n result.set(calculatedPosition.x, calculatedPosition.y, calculatedPosition.z);\n targetNotFound = false;\n }\n lastPosition = currentPos;\n }\n });\n if (posZeroToOne === 1) {\n const path = node.paths[node.paths.length - 1];\n const controlPoint = path._points[path._points.length - 1];\n result.set(controlPoint.x, controlPoint.y, controlPoint.z);\n }\n });\n }\n return result;\n }\n updateDirectionMoveAlongSketch(connector) {\n const node = connector._parent;\n const result = new Vector3(0, 0, 0);\n if (node instanceof exports.SketchNode) {\n const sketchLength = node.getSketchLength();\n const posZeroToOne = connector.positionZeroToOne;\n let trackedLength = 0;\n let targetNotFound = true;\n let lastPathVector = null;\n let nextPathVector = null;\n node.paths.forEach(path => {\n let lastPosition = null;\n let lastLength = 0;\n path._points.forEach(point => {\n if (!lastPosition) {\n lastPosition = point;\n }\n else {\n const currentPos = point;\n if (lastPosition) {\n const index = path._points.indexOf(lastPosition) + 1;\n const index2 = index + 1;\n if (index !== path._points.length) {\n const nextPosition = path._points[index];\n if (index2 !== path._points.length) {\n const nextNextPosition = path._points[index2];\n nextPathVector = new exports.KbVector(nextNextPosition.x - nextPosition.x, nextNextPosition.y - nextPosition.y, nextNextPosition.z - nextPosition.z).normalize();\n }\n }\n }\n const distance = Math.sqrt(Math.pow(lastPosition.x - currentPos.x, 2) +\n Math.pow(lastPosition.y - currentPos.y, 2) +\n Math.pow(lastPosition.z - currentPos.z, 2));\n lastLength = trackedLength;\n trackedLength += distance;\n const distanceBetweenPoints = trackedLength - lastLength;\n const distanceFromP1toTarget = posZeroToOne * sketchLength - lastLength;\n const percentageBetween2Points = distanceFromP1toTarget / distanceBetweenPoints;\n const vectorBetween2Points = new exports.KbVector(currentPos.x - lastPosition.x, currentPos.y - lastPosition.y, currentPos.z - lastPosition.z);\n if (trackedLength > posZeroToOne * sketchLength && targetNotFound) {\n let calculatedDirection = null;\n if (!connector.interpolateDirection) {\n calculatedDirection = vectorBetween2Points.normalize();\n }\n else {\n if (!lastPathVector) {\n if (nextPathVector) {\n calculatedDirection = vectorBetween2Points\n .normalize()\n .scale(1 - percentageBetween2Points)\n .add(nextPathVector\n .normalize()\n .add(vectorBetween2Points.normalize())\n .normalize()\n .scale(percentageBetween2Points))\n .normalize();\n }\n else {\n calculatedDirection = vectorBetween2Points.normalize();\n }\n }\n else if (!nextPathVector) {\n calculatedDirection = lastPathVector\n .scale(1 - percentageBetween2Points)\n .add(vectorBetween2Points.scale(1 + percentageBetween2Points))\n .scale(1 / 2)\n .normalize();\n }\n else {\n if (percentageBetween2Points <= 0.5) {\n calculatedDirection = lastPathVector\n .normalize()\n .add(vectorBetween2Points.normalize())\n .normalize()\n .scale(1 - percentageBetween2Points * 2)\n .add(vectorBetween2Points.scale(percentageBetween2Points * 2))\n .normalize();\n }\n else {\n calculatedDirection = vectorBetween2Points\n .normalize()\n .scale(1 - (percentageBetween2Points - 0.5) * 2)\n .add(nextPathVector\n .normalize()\n .add(vectorBetween2Points.normalize())\n .normalize()\n .scale((percentageBetween2Points - 0.5) * 2))\n .normalize();\n }\n }\n }\n targetNotFound = false;\n result.set(calculatedDirection.x, calculatedDirection.y, calculatedDirection.z);\n }\n lastPathVector = vectorBetween2Points;\n lastPosition = currentPos;\n }\n });\n });\n if (posZeroToOne === 1) {\n if (posZeroToOne === 1) {\n const path = node.paths[node.paths.length - 1];\n const currentPos = path._points[path._points.length - 1];\n const lastPosition = path._points[path._points.length - 2];\n const vectorBetween2Points = new exports.KbVector(currentPos.x - lastPosition.x, currentPos.y - lastPosition.y, currentPos.z - lastPosition.z).normalize();\n result.set(vectorBetween2Points.x, vectorBetween2Points.y, vectorBetween2Points.z);\n }\n }\n }\n return result;\n }\n getMesh(c, layer) {\n let mesh = layer.map.get(c._dynamicId);\n if (!mesh) {\n if (!layer.template) {\n layer.template = this.createMeshTemplate(layer);\n }\n mesh = layer.template.clone(c._dynamicId); // layer.template.createInstance(c._dynamicId);\n mesh.id = c._dynamicId;\n const parentNode = c._parent;\n if (parentNode && parentNode instanceof exports.SpaceNode) {\n const parent3d = this.viewer.getRendererNode(parentNode);\n if (parent3d) {\n mesh.setParent(parent3d);\n }\n else {\n console.warn(`mesh not found for ${parentNode.name}`);\n }\n }\n mesh.isPickable = layer.pickable;\n mesh.position = Vector3.Zero();\n mesh.rotationQuaternion = Quaternion.Identity();\n layer.map.set(c._dynamicId, mesh);\n layer.parts.set(c._dynamicId, {\n mesh,\n });\n }\n return mesh;\n }\n createMeshTemplate(layer) {\n const ball = CreateSphereVertexData({\n segments: 8,\n diameter: CONNECTOR_DIAM,\n });\n if (layer.withArrows) {\n const arrowHead = CreateCylinderVertexData({\n diameterTop: 0,\n height: CONNECTOR_DIAM / 2,\n diameterBottom: CONNECTOR_DIAM / 4,\n tessellation: 6,\n });\n arrowHead.transform(Matrix.Compose(Vector3.One(), Quaternion.FromEulerAngles(Math.PI / 2, 0, 0), new Vector3(0, 0, CONNECTOR_DIAM * 1.75)));\n const arrowShaft = CreateCylinderVertexData({\n diameterTop: CONNECTOR_DIAM / 10,\n height: CONNECTOR_DIAM,\n diameterBottom: CONNECTOR_DIAM / 10,\n tessellation: 6,\n });\n arrowShaft.transform(Matrix.Compose(Vector3.One(), Quaternion.FromEulerAngles(Math.PI / 2, 0, 0), new Vector3(0, 0, CONNECTOR_DIAM)));\n ball.merge(arrowHead).merge(arrowShaft);\n }\n const mesh = new Mesh('', layer.scene);\n ball.applyToMesh(mesh, false); //updateable false\n if (layer.withArrows) {\n //let ballHalf1 = new SubMesh(0, 0, 231, 0, 600, mesh);\n new SubMesh(1, 0, 231, 0, 600, mesh); // ball half 2\n }\n mesh.rotation.y = Math.PI / 2;\n mesh.bakeCurrentTransformIntoVertices();\n mesh.position = Vector3.Zero();\n mesh.material = this.getConnectorMaterial(layer.withArrows);\n mesh.visibility = layer.visibility;\n mesh.setEnabled(false);\n return mesh;\n }\n delete3d(c) {\n const utilityLayer = this.layers.get('utility');\n const utilityMesh = utilityLayer.map.get(c._dynamicId);\n if (utilityMesh) {\n utilityMesh.dispose();\n utilityLayer.map.delete(c._dynamicId);\n }\n this.removeSubscription(c);\n super.delete3d(c);\n }\n getConnectorMaterial(twoTone) {\n if (!this.connectorMaterial) {\n if (twoTone) {\n const m1 = new StandardMaterial('connectorMaterial0', this.scene);\n m1.emissiveColor = exports.KbColor.KbmaxOrange().toColor3();\n m1.disableLighting = true;\n const m2 = new StandardMaterial('connectorMaterial1', this.scene);\n m2.emissiveColor = new Color3(0, 101 / 255, 1); //new Color3(226/255, 101/255, 29/255);// KbColor.KbmaxBlue().toColor3();\n m2.disableLighting = true;\n const m = new MultiMaterial('connectorMaterial', this.scene);\n m.subMaterials.push(m1, m2);\n this.connectorMaterial = m;\n }\n else {\n const m = new StandardMaterial('connectorMaterial', this.scene);\n m.emissiveColor = this.viewer.highlightColor.toColor3();\n m.disableLighting = true;\n this.connectorMaterial = m;\n }\n }\n return this.connectorMaterial;\n }\n getSelectedMaterial(withArrows) {\n if (!this.selectedMaterial) {\n this.selectedMaterial = this.getConnectorMaterial(withArrows);\n if (withArrows) {\n const m1 = new StandardMaterial('connectorSelectedMaterial0', this.scene);\n m1.emissiveColor = new Color3(254 / 255, 233 / 255, 7 / 255);\n m1.disableLighting = true;\n const m2 = new StandardMaterial('connectorSelectedMaterial1', this.scene);\n m2.emissiveColor = new Color3(2 / 255, 251 / 255, 248 / 255);\n m2.disableLighting = true;\n const m = new MultiMaterial('connectorSelectedMaterial', this.scene);\n m.subMaterials.push(m1, m2);\n this.selectedMaterial = m;\n }\n else {\n const m = new StandardMaterial('connectorSelectedMaterial', this.scene);\n m.emissiveColor = this.viewer.options.highlightColor.toColor3();\n m.disableLighting = true;\n this.selectedMaterial = m;\n }\n }\n return this.selectedMaterial;\n }\n dispose() {\n super.dispose();\n for (const [, layer] of this.layers) {\n if (layer.template) {\n layer.template.dispose();\n }\n layer.map.clear();\n }\n if (this.connectorMaterial) {\n this.connectorMaterial.dispose();\n }\n }\n }\n\n class FeatureRenderer extends Renderer {\n constructor() {\n super(...arguments);\n this.vertexDataCache = new Map();\n }\n updateMeshGeometryCache(feature, vertexData, vertexDataChanges) {\n const self = this;\n const newVertexData = {};\n // Grab the properties in the mesh vertex that are needed for the feature\n // Only swap out the properties in the old vertex data with the stored copy if they have NOT been affected by another vertex\n if (feature.affectedBy) {\n const cachedData = this.vertexDataCache.get(feature._dynamicId);\n feature.affectedBy.forEach(affectedBy => {\n var _a, _b, _c, _d, _e, _f;\n if (affectedBy === exports.eMeshChangeTypes.VertexIndices) {\n if (vertexDataChanges.indices === null) {\n newVertexData.indices = null;\n clearVertexDataCache('indices');\n }\n else if (vertexDataChanges.indices) {\n newVertexData.indices = vertexDataChanges.indices.slice();\n updateVertexDataCache('indices', vertexDataChanges.indices);\n }\n else if (cachedData) {\n newVertexData.indices = (_a = cachedData.indices) === null || _a === void 0 ? void 0 : _a.slice();\n }\n else {\n newVertexData.indices = vertexData.indices.slice();\n updateVertexDataCache('indices', vertexData.indices);\n }\n if (vertexDataChanges.paths === null) {\n newVertexData.paths = null;\n clearVertexDataCache('paths');\n }\n else if (vertexDataChanges.paths) {\n newVertexData.paths = vertexDataChanges.paths.slice();\n updateVertexDataCache('paths', vertexDataChanges.paths);\n }\n else if (cachedData) {\n newVertexData.paths = (_b = cachedData.paths) === null || _b === void 0 ? void 0 : _b.slice();\n }\n else {\n if (vertexData.paths) {\n newVertexData.paths = vertexData.paths.slice();\n updateVertexDataCache('paths', vertexData.paths);\n }\n else {\n newVertexData.paths = null;\n }\n }\n if (vertexDataChanges.holes === null) {\n newVertexData.holes = null;\n clearVertexDataCache('holes');\n }\n else if (vertexDataChanges.holes) {\n newVertexData.holes = vertexDataChanges.holes.slice();\n updateVertexDataCache('holes', vertexDataChanges.holes);\n }\n else if (cachedData) {\n newVertexData.holes = (_c = cachedData.holes) === null || _c === void 0 ? void 0 : _c.slice();\n }\n else {\n if (vertexData.holes) {\n newVertexData.holes = vertexData.holes.slice();\n updateVertexDataCache('holes', vertexData.holes);\n }\n else {\n newVertexData.holes = null;\n }\n }\n }\n if (affectedBy === exports.eMeshChangeTypes.VertexPositions) {\n if (vertexDataChanges.positions === null) {\n newVertexData.positions = null;\n clearVertexDataCache('positions');\n }\n else if (vertexDataChanges.positions) {\n newVertexData.positions = vertexDataChanges.positions.slice();\n updateVertexDataCache('positions', vertexDataChanges.positions);\n }\n else if (cachedData) {\n newVertexData.positions = (_d = cachedData.positions) === null || _d === void 0 ? void 0 : _d.slice(); // compiler complains about the slice generic type for some reason\n }\n else {\n if (!vertexData.positions) {\n throw Error('Invalid geometry: positions is null');\n }\n newVertexData.positions = vertexData.positions.slice();\n updateVertexDataCache('positions', vertexData.positions);\n }\n }\n if (affectedBy === exports.eMeshChangeTypes.VertexNormals) {\n if (vertexDataChanges.normals === null) {\n newVertexData.normals = null;\n clearVertexDataCache('normals');\n }\n else if (vertexDataChanges.normals) {\n newVertexData.normals = vertexDataChanges.normals.slice();\n updateVertexDataCache('normals', vertexDataChanges.normals);\n }\n else if (cachedData) {\n newVertexData.normals = (_e = cachedData.normals) === null || _e === void 0 ? void 0 : _e.slice();\n }\n else {\n if (vertexData.normals) {\n newVertexData.normals = vertexData.normals.slice();\n updateVertexDataCache('normals', vertexData.normals);\n }\n else {\n newVertexData.normals = null;\n }\n }\n }\n if (affectedBy === exports.eMeshChangeTypes.VertexUvs) {\n if (vertexDataChanges.uvs === null) {\n newVertexData.uvs = null;\n clearVertexDataCache('uvs');\n }\n else if (vertexDataChanges.uvs) {\n newVertexData.uvs = vertexDataChanges.uvs.slice();\n updateVertexDataCache('uvs', vertexDataChanges.uvs);\n }\n else if (cachedData) {\n newVertexData.uvs = (_f = cachedData.uvs) === null || _f === void 0 ? void 0 : _f.slice();\n }\n else {\n if (vertexData.uvs) {\n newVertexData.uvs = vertexData.uvs.slice();\n updateVertexDataCache('uvs', vertexData.uvs);\n }\n else {\n newVertexData.uvs = null;\n }\n }\n }\n if (affectedBy === exports.eMeshChangeTypes.Submeshes) {\n if (vertexDataChanges.submeshData === null) {\n newVertexData.submeshData = null;\n clearVertexDataCache('submeshData');\n }\n else if (vertexDataChanges.submeshData) {\n newVertexData.submeshData = JSON.parse(JSON.stringify(vertexDataChanges.submeshData));\n updateVertexDataCache('submeshData', JSON.parse(JSON.stringify(vertexDataChanges.submeshData)));\n }\n else if (cachedData) {\n const cachedSubmeshes = cachedData.submeshData;\n newVertexData.submeshData = cachedSubmeshes\n ? JSON.parse(JSON.stringify(cachedSubmeshes))\n : undefined;\n }\n else {\n if (vertexData.submeshData) {\n newVertexData.submeshData = JSON.parse(JSON.stringify(vertexData.submeshData));\n updateVertexDataCache('submeshData', JSON.parse(JSON.stringify(vertexData.submeshData)));\n }\n else {\n newVertexData.submeshData = null;\n }\n }\n }\n });\n }\n function clearVertexDataCache(prop) {\n const vertexDataCache = self.vertexDataCache.get(feature._dynamicId);\n if (vertexDataCache) {\n delete vertexDataCache[prop];\n }\n }\n function updateVertexDataCache(prop, value) {\n if (!value) {\n throw Error('invalid geometry does not have ' + prop);\n }\n let vertexDataCache = self.vertexDataCache.get(feature._dynamicId);\n if (!vertexDataCache) {\n vertexDataCache = {};\n self.vertexDataCache.set(feature._dynamicId, vertexDataCache);\n }\n if (typeof value.slice !== 'function') {\n throw Error('fixme: how do I clone this property?');\n }\n vertexDataCache[prop] = value.slice(); // Need to make sure everything is cloned when inserted into the cache\n }\n return newVertexData;\n }\n updateMeshGeometry(feature, vertexData, vertexDataChanges) {\n return this.updateMeshGeometryCache(feature, vertexData, vertexDataChanges);\n }\n delete3d(node, preserveCache) {\n if (!preserveCache) {\n this.vertexDataCache.delete(node._dynamicId);\n }\n const t3d = this.get3d(node);\n if (t3d) {\n super.delete3d(node);\n }\n else {\n SyncService.nodeDeletionProcessed(node);\n }\n }\n clearCache(features) {\n if (features) {\n features.forEach(feature => {\n this.vertexDataCache.delete(feature._dynamicId);\n });\n }\n else {\n this.vertexDataCache.clear();\n }\n }\n dispose() {\n super.dispose();\n this.clearCache();\n }\n }\n\n class MoveVerticesRenderer extends FeatureRenderer {\n constructor() {\n super(...arguments);\n this.modelToken = Token.MoveVerticesFeature;\n }\n updateMeshGeometry(feature, vertexData, vertexDataChanges) {\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\n const parentMesh = feature._parent;\n if (parentMesh && newVertexData.positions && newVertexData.positions.length > 0) {\n const newMeshVertexPositions = newVertexData.positions;\n if (feature.vertexIndices && feature.vertexIndices.length > 0) {\n let pivot;\n let rotationQuaternion;\n let scaleVector;\n if (feature.rotationAngle !== 0 && !feature.rotationAxis.equal({ x: 0, y: 0, z: 0 })) {\n rotationQuaternion = Quaternion.RotationAxis(feature.rotationAxis.toVec3(), (feature.rotationAngle / 180) * Math.PI);\n }\n if (!feature.scale.equal({ x: 1, y: 1, z: 1 })) {\n scaleVector = feature.scale.toVec3();\n }\n // Calculate vertex center if needed\n if (rotationQuaternion || scaleVector) {\n if (feature.center === exports.eFeatureCenter.meshPivotPoint) {\n pivot = parentMesh.pivot.toVec3();\n }\n else {\n // Calculate vertices barycenter\n pivot = Vector3.Zero();\n const currentMeshVertexPositions = newVertexData.positions;\n if (currentMeshVertexPositions) {\n feature.vertexIndices.forEach(idx => {\n pivot.x += currentMeshVertexPositions[idx * 3];\n pivot.y += currentMeshVertexPositions[idx * 3 + 1];\n pivot.z += currentMeshVertexPositions[idx * 3 + 2];\n });\n pivot.scaleInPlace(1 / feature.vertexIndices.length);\n }\n }\n }\n const vertexPosition = new Vector3();\n feature.vertexIndices.forEach(index => {\n newMeshVertexPositions[index * 3] += feature.translation.x;\n newMeshVertexPositions[index * 3 + 1] += feature.translation.y;\n newMeshVertexPositions[index * 3 + 2] += feature.translation.z;\n if (pivot && (rotationQuaternion || scaleVector)) {\n vertexPosition.copyFromFloats(newMeshVertexPositions[index * 3] - pivot.x, newMeshVertexPositions[index * 3 + 1] - pivot.y, newMeshVertexPositions[index * 3 + 2] - pivot.z);\n if (rotationQuaternion) {\n vertexPosition.rotateByQuaternionToRef(rotationQuaternion, vertexPosition);\n }\n if (scaleVector) {\n vertexPosition.multiplyInPlace(scaleVector);\n }\n vertexPosition.addInPlace(pivot);\n newMeshVertexPositions[index * 3] = vertexPosition.x;\n newMeshVertexPositions[index * 3 + 1] = vertexPosition.y;\n newMeshVertexPositions[index * 3 + 2] = vertexPosition.z;\n }\n });\n }\n }\n return newVertexData;\n }\n update3dInternal() { }\n }\n\n class NormalSmoothingRenderer extends FeatureRenderer {\n constructor() {\n super(...arguments);\n this.modelToken = Token.NormalSmoothingFeature;\n this.vertexNeighbors = new Map();\n this.planeNormals = new Map();\n this.vectorA = new Vector3();\n this.vectorB = new Vector3();\n this.vectorC = new Vector3();\n this.normalVector = new Vector3();\n this.totalSelectedIndices = 0;\n }\n updateMeshGeometry(feature, vertexData, vertexDataChanges) {\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\n let vertexGroups = [];\n //if no faces are selected, select all faces to smooth entire mesh.\n if (feature.vertexGroups.length === 0 && newVertexData.indices) {\n for (let i = 0; i < newVertexData.indices.length; i += 6) {\n vertexGroups.push({ startIndex: i, endIndex: i + 6 });\n }\n }\n else {\n vertexGroups = feature.vertexGroups.sort((groupA, groupB) => groupA.startIndex - groupB.startIndex);\n }\n if (newVertexData.indices) {\n const planeNormal = new Vector3();\n let normals;\n let positions;\n let uvs;\n let indices;\n const newPositions = [];\n const newNormals = [];\n const newUvs = [];\n const submeshDataArray = [];\n if (newVertexData.normals) {\n normals = newVertexData.normals;\n }\n else if (vertexData.normals) {\n normals = Array.from(vertexData.normals);\n }\n else {\n normals = [];\n }\n if (newVertexData.positions) {\n positions = newVertexData.positions;\n }\n else if (vertexData.positions) {\n positions = Array.from(vertexData.positions);\n }\n else {\n positions = [];\n }\n if (newVertexData.uvs) {\n uvs = newVertexData.uvs;\n }\n else if (vertexData.uvs) {\n uvs = Array.from(vertexData.uvs);\n }\n else {\n uvs = [];\n }\n if (newVertexData.indices) {\n indices = newVertexData.indices;\n }\n else if (vertexData.indices) {\n indices = Array.from(vertexData.indices);\n }\n else {\n indices = [];\n }\n this.vertexNeighbors.clear();\n this.planeNormals.clear();\n this.totalSelectedIndices = 0;\n vertexGroups.forEach(vertexGroup => {\n this.totalSelectedIndices += vertexGroup.endIndex - vertexGroup.startIndex;\n });\n if (feature.smooth) {\n //looping through the vertex groups and finding all the neighboring faces\n vertexGroups.forEach(vertexGroup => {\n const planeCoordinates = [];\n const planePositions = [];\n for (let i = vertexGroup.startIndex; i < vertexGroup.endIndex; i++) {\n if (positions && indices) {\n let key = '';\n for (let j = 0; j < 3; j++) {\n const vertexIndex = indices[i];\n const coordinate = positions[vertexIndex * 3 + j];\n planePositions.push(coordinate);\n if (j == 2) {\n planeCoordinates.push([...planePositions]);\n planePositions.clear();\n }\n key += String(coordinate);\n }\n if (planeCoordinates.length == 3) {\n this.calculateNormalToRef(planeCoordinates, planeNormal);\n this.planeNormals.set(Math.floor(i / 3), planeNormal.asArray());\n planeCoordinates.clear();\n }\n let neighbor = this.vertexNeighbors.get(key);\n if (neighbor) {\n neighbor.push(i);\n this.vertexNeighbors.set(key, neighbor);\n }\n else {\n neighbor = [];\n neighbor.push(i);\n this.vertexNeighbors.set(key, neighbor);\n }\n }\n }\n });\n this.vertexNeighbors.forEach(position => {\n const newNormal = this.calculateSmoothedUnitVector(position);\n position.forEach(index => {\n if (normals && indices) {\n for (let i = 0; i < 3; i++) {\n normals[indices[index] * 3 + i] = newNormal[i];\n }\n }\n });\n });\n }\n else {\n //looping through the selected faces to get the face normals\n vertexGroups.forEach(vertexGroup => {\n const planeCoordinates = [];\n const planePositions = [];\n for (let i = vertexGroup.startIndex; i < vertexGroup.endIndex; i++) {\n if (positions && indices) {\n for (let j = 0; j < 3; j++) {\n const vertexIndex = indices[i];\n const coordinate = positions[vertexIndex * 3 + j];\n planePositions.push(coordinate);\n if (j == 2) {\n planeCoordinates.push([...planePositions]);\n planePositions.clear();\n }\n }\n if (planeCoordinates.length == 3) {\n this.calculateNormalToRef(planeCoordinates, planeNormal);\n const planeNormalArray = this.calculateUnitVector(planeNormal.asArray());\n this.planeNormals.set(i, planeNormalArray);\n this.planeNormals.set(i - 1, planeNormalArray);\n this.planeNormals.set(i - 2, planeNormalArray);\n planeCoordinates.clear();\n }\n }\n }\n });\n //looping through it again to change/add the normals\n let totalVertices = 0;\n if (positions) {\n totalVertices = positions.length / 3;\n }\n const visitedVertices = [];\n vertexGroups.forEach(vertexGroup => {\n for (let i = vertexGroup.startIndex; i < vertexGroup.endIndex; i++) {\n const vertex = indices[i];\n const planeNormal = this.planeNormals.get(i);\n const currentNormal = [];\n const currentPosition = [];\n const currentUv = [];\n for (let i = 0; i < 3; i++) {\n currentNormal.push(normals[vertex * 3 + i]);\n currentPosition.push(positions[vertex * 3 + i]);\n }\n for (let i = 0; i < 2; i++) {\n currentUv.push(uvs[vertex * 2 + i]);\n }\n //vertex is already visited, therefor we add a new one.\n //else: replace the existing vertex\n if (visitedVertices[vertex]) {\n for (let i = 0; i < 3; i++) {\n if (planeNormal) {\n newNormals.push(planeNormal[i]);\n newPositions.push(currentPosition[i]);\n }\n }\n for (let i = 0; i < 2; i++) {\n newUvs.push(currentUv[i]);\n }\n indices[i] = totalVertices;\n totalVertices += 1;\n //handling submesh data\n if (vertexData.submeshData) {\n for (let i = 0; i < vertexData.submeshData.length; i++) {\n const submeshData = vertexData.submeshData[i];\n if (submeshData.verticesStart <= vertex &&\n submeshData.verticesStart + submeshData.verticesCount > vertex) {\n submeshData.verticesCount += 1;\n }\n else if (submeshData.verticesStart > vertex) {\n submeshData.verticesStart += 1;\n }\n submeshData.indexCount = (positions.length + newPositions.length) / 3;\n submeshDataArray[i] = submeshData;\n }\n }\n }\n else {\n visitedVertices[vertex] = true;\n if (planeNormal) {\n for (let i = 0; i < 3; i++) {\n normals[vertex * 3 + i] = planeNormal[i];\n }\n }\n }\n }\n });\n const resultPositions = new Float32Array(positions.length + newPositions.length);\n resultPositions.set(positions);\n resultPositions.set(newPositions, positions.length);\n const resultNormals = new Float32Array(normals.length + newNormals.length);\n resultNormals.set(normals);\n resultNormals.set(newNormals, normals.length);\n const resultUvs = new Float32Array(uvs.length + newUvs.length);\n resultUvs.set(uvs);\n resultUvs.set(newUvs, uvs.length);\n newVertexData.positions = resultPositions;\n newVertexData.normals = resultNormals;\n newVertexData.uvs = resultUvs;\n newVertexData.indices = indices;\n newVertexData.submeshData = submeshDataArray;\n }\n }\n this.viewer.updateNormalSmoothingWarnings(feature, this.totalSelectedIndices / 3, false);\n return newVertexData;\n }\n update3dInternal() { }\n delete3d(node, preserveCache) {\n this.viewer.updateNormalSmoothingWarnings(node, 0, true);\n super.delete3d(node);\n }\n //Formula for normal of a plane: (vectorA-VectorB) x (VectorB-VectorC)\n calculateNormalToRef(coordinates, normalVector) {\n Vector3.FromArrayToRef(coordinates[0], 0, this.vectorA);\n Vector3.FromArrayToRef(coordinates[1], 0, this.vectorB);\n Vector3.FromArrayToRef(coordinates[2], 0, this.vectorC);\n this.vectorA.subtractInPlace(this.vectorB);\n this.vectorB.subtractInPlace(this.vectorC);\n return Vector3.CrossToRef(this.vectorB, this.vectorA, normalVector);\n }\n calculateSmoothedUnitVector(vertices) {\n let result = [];\n vertices.forEach(vertex => {\n const normalVector = this.planeNormals.get(Math.floor(vertex / 3));\n if (normalVector) {\n result = addArrays(result, normalVector);\n }\n });\n return this.calculateUnitVector(result);\n }\n calculateUnitVector(vector) {\n let divider = 0;\n const result = [];\n vector.forEach(element => {\n divider += element * element;\n });\n divider = Math.sqrt(divider);\n vector.forEach(element => {\n result.push(element / divider);\n });\n return result;\n }\n }\n\n /**\n * Represents one particle of a solid particle system.\n */\n class SolidParticle {\n /**\n * Particle BoundingInfo object\n * @returns a BoundingInfo\n */\n getBoundingInfo() {\n return this._boundingInfo;\n }\n /**\n * Returns true if there is already a bounding info\n */\n get hasBoundingInfo() {\n return this._boundingInfo !== null;\n }\n /**\n * Creates a Solid Particle object.\n * Don't create particles manually, use instead the Solid Particle System internal tools like _addParticle()\n * @param particleIndex (integer) is the particle index in the Solid Particle System pool.\n * @param particleId (integer) is the particle identifier. Unless some particles are removed from the SPS, it's the same value than the particle idx.\n * @param positionIndex (integer) is the starting index of the particle vertices in the SPS \"positions\" array.\n * @param indiceIndex (integer) is the starting index of the particle indices in the SPS \"indices\" array.\n * @param model (ModelShape) is a reference to the model shape on what the particle is designed.\n * @param shapeId (integer) is the model shape identifier in the SPS.\n * @param idxInShape (integer) is the index of the particle in the current model (ex: the 10th box of addShape(box, 30))\n * @param sps defines the sps it is associated to\n * @param modelBoundingInfo is the reference to the model BoundingInfo used for intersection computations.\n * @param materialIndex is the particle material identifier (integer) when the MultiMaterials are enabled in the SPS.\n */\n constructor(particleIndex, particleId, positionIndex, indiceIndex, model, shapeId, idxInShape, sps, modelBoundingInfo = null, materialIndex = null) {\n /**\n * particle global index\n */\n this.idx = 0;\n /**\n * particle identifier\n */\n this.id = 0;\n /**\n * The color of the particle\n */\n this.color = new Color4(1.0, 1.0, 1.0, 1.0);\n /**\n * The world space position of the particle.\n */\n this.position = Vector3.Zero();\n /**\n * The world space rotation of the particle. (Not use if rotationQuaternion is set)\n */\n this.rotation = Vector3.Zero();\n /**\n * The scaling of the particle.\n */\n this.scaling = Vector3.One();\n /**\n * The uvs of the particle.\n */\n this.uvs = new Vector4(0.0, 0.0, 1.0, 1.0);\n /**\n * The current speed of the particle.\n */\n this.velocity = Vector3.Zero();\n /**\n * The pivot point in the particle local space.\n */\n this.pivot = Vector3.Zero();\n /**\n * Must the particle be translated from its pivot point in its local space ?\n * In this case, the pivot point is set at the origin of the particle local space and the particle is translated.\n * Default : false\n */\n this.translateFromPivot = false;\n /**\n * Is the particle active or not ?\n */\n this.alive = true;\n /**\n * Is the particle visible or not ?\n */\n this.isVisible = true;\n /**\n * Index of this particle in the global \"positions\" array (Internal use)\n * @internal\n */\n this._pos = 0;\n /**\n * @internal Index of this particle in the global \"indices\" array (Internal use)\n */\n this._ind = 0;\n /**\n * ModelShape id of this particle\n */\n this.shapeId = 0;\n /**\n * Index of the particle in its shape id\n */\n this.idxInShape = 0;\n /**\n * @internal Still set as invisible in order to skip useless computations (Internal use)\n */\n this._stillInvisible = false;\n /**\n * @internal Last computed particle rotation matrix\n */\n this._rotationMatrix = [1.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 1.0];\n /**\n * Parent particle Id, if any.\n * Default null.\n */\n this.parentId = null;\n /**\n * The particle material identifier (integer) when MultiMaterials are enabled in the SPS.\n */\n this.materialIndex = null;\n /**\n * Custom object or properties.\n */\n this.props = null;\n /**\n * The culling strategy to use to check whether the solid particle must be culled or not when using isInFrustum().\n * The possible values are :\n * - AbstractMesh.CULLINGSTRATEGY_STANDARD\n * - AbstractMesh.CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY\n * - AbstractMesh.CULLINGSTRATEGY_OPTIMISTIC_INCLUSION\n * - AbstractMesh.CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY\n * The default value for solid particles is AbstractMesh.CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY\n * Please read each static variable documentation in the class AbstractMesh to get details about the culling process.\n * */\n this.cullingStrategy = AbstractMesh.CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY;\n /**\n * @internal Internal global position in the SPS.\n */\n this._globalPosition = Vector3.Zero();\n this.idx = particleIndex;\n this.id = particleId;\n this._pos = positionIndex;\n this._ind = indiceIndex;\n this._model = model;\n this.shapeId = shapeId;\n this.idxInShape = idxInShape;\n this._sps = sps;\n if (modelBoundingInfo) {\n this._modelBoundingInfo = modelBoundingInfo;\n this._boundingInfo = new BoundingInfo(modelBoundingInfo.minimum, modelBoundingInfo.maximum);\n }\n if (materialIndex !== null) {\n this.materialIndex = materialIndex;\n }\n }\n /**\n * Copies the particle property values into the existing target : position, rotation, scaling, uvs, colors, pivot, parent, visibility, alive\n * @param target the particle target\n * @returns the current particle\n */\n copyToRef(target) {\n target.position.copyFrom(this.position);\n target.rotation.copyFrom(this.rotation);\n if (this.rotationQuaternion) {\n if (target.rotationQuaternion) {\n target.rotationQuaternion.copyFrom(this.rotationQuaternion);\n }\n else {\n target.rotationQuaternion = this.rotationQuaternion.clone();\n }\n }\n target.scaling.copyFrom(this.scaling);\n if (this.color) {\n if (target.color) {\n target.color.copyFrom(this.color);\n }\n else {\n target.color = this.color.clone();\n }\n }\n target.uvs.copyFrom(this.uvs);\n target.velocity.copyFrom(this.velocity);\n target.pivot.copyFrom(this.pivot);\n target.translateFromPivot = this.translateFromPivot;\n target.alive = this.alive;\n target.isVisible = this.isVisible;\n target.parentId = this.parentId;\n target.cullingStrategy = this.cullingStrategy;\n if (this.materialIndex !== null) {\n target.materialIndex = this.materialIndex;\n }\n return this;\n }\n /**\n * Legacy support, changed scale to scaling\n */\n get scale() {\n return this.scaling;\n }\n /**\n * Legacy support, changed scale to scaling\n */\n set scale(scale) {\n this.scaling = scale;\n }\n /**\n * Legacy support, changed quaternion to rotationQuaternion\n */\n get quaternion() {\n return this.rotationQuaternion;\n }\n /**\n * Legacy support, changed quaternion to rotationQuaternion\n */\n set quaternion(q) {\n this.rotationQuaternion = q;\n }\n /**\n * Returns a boolean. True if the particle intersects another particle or another mesh, else false.\n * The intersection is computed on the particle bounding sphere and Axis Aligned Bounding Box (AABB)\n * @param target is the object (solid particle or mesh) what the intersection is computed against.\n * @returns true if it intersects\n */\n intersectsMesh(target) {\n if (!this._boundingInfo || !target.hasBoundingInfo) {\n return false;\n }\n if (this._sps._bSphereOnly) {\n return BoundingSphere.Intersects(this._boundingInfo.boundingSphere, target.getBoundingInfo().boundingSphere);\n }\n return this._boundingInfo.intersects(target.getBoundingInfo(), false);\n }\n /**\n * Returns `true` if the solid particle is within the frustum defined by the passed array of planes.\n * A particle is in the frustum if its bounding box intersects the frustum\n * @param frustumPlanes defines the frustum to test\n * @returns true if the particle is in the frustum planes\n */\n isInFrustum(frustumPlanes) {\n return this._boundingInfo !== null && this._boundingInfo.isInFrustum(frustumPlanes, this.cullingStrategy);\n }\n /**\n * get the rotation matrix of the particle\n * @internal\n */\n getRotationMatrix(m) {\n let quaternion;\n if (this.rotationQuaternion) {\n quaternion = this.rotationQuaternion;\n }\n else {\n quaternion = TmpVectors.Quaternion[0];\n const rotation = this.rotation;\n Quaternion.RotationYawPitchRollToRef(rotation.y, rotation.x, rotation.z, quaternion);\n }\n quaternion.toRotationMatrix(m);\n }\n }\n /**\n * Represents the shape of the model used by one particle of a solid particle system.\n * SPS internal tool, don't use it manually.\n */\n class ModelShape {\n /**\n * Get or set the shapeId\n * @deprecated Please use shapeId instead\n */\n get shapeID() {\n return this.shapeId;\n }\n set shapeID(shapeID) {\n this.shapeId = shapeID;\n }\n /**\n * Creates a ModelShape object. This is an internal simplified reference to a mesh used as for a model to replicate particles from by the SPS.\n * SPS internal tool, don't use it manually.\n * @internal\n */\n constructor(id, shape, indices, normals, colors, shapeUV, posFunction, vtxFunction, material) {\n /**\n * length of the shape in the model indices array (internal use)\n * @internal\n */\n this._indicesLength = 0;\n this.shapeId = id;\n this._shape = shape;\n this._indices = indices;\n this._indicesLength = indices.length;\n this._shapeUV = shapeUV;\n this._shapeColors = colors;\n this._normals = normals;\n this._positionFunction = posFunction;\n this._vertexFunction = vtxFunction;\n this._material = material;\n }\n }\n /**\n * Represents a Depth Sorted Particle in the solid particle system.\n * @internal\n */\n class DepthSortedParticle {\n /**\n * Creates a new sorted particle\n * @param idx\n * @param ind\n * @param indLength\n * @param materialIndex\n */\n constructor(idx, ind, indLength, materialIndex) {\n /**\n * Particle index\n */\n this.idx = 0;\n /**\n * Index of the particle in the \"indices\" array\n */\n this.ind = 0;\n /**\n * Length of the particle shape in the \"indices\" array\n */\n this.indicesLength = 0;\n /**\n * Squared distance from the particle to the camera\n */\n this.sqDistance = 0.0;\n /**\n * Material index when used with MultiMaterials\n */\n this.materialIndex = 0;\n this.idx = idx;\n this.ind = ind;\n this.indicesLength = indLength;\n this.materialIndex = materialIndex;\n }\n }\n /**\n * Represents a solid particle vertex\n */\n class SolidParticleVertex {\n /**\n * Creates a new solid particle vertex\n */\n constructor() {\n this.position = Vector3.Zero();\n this.color = new Color4(1.0, 1.0, 1.0, 1.0);\n this.uv = Vector2.Zero();\n }\n // Getters and Setters for back-compatibility\n /** Vertex x coordinate */\n get x() {\n return this.position.x;\n }\n set x(val) {\n this.position.x = val;\n }\n /** Vertex y coordinate */\n get y() {\n return this.position.y;\n }\n set y(val) {\n this.position.y = val;\n }\n /** Vertex z coordinate */\n get z() {\n return this.position.z;\n }\n set z(val) {\n this.position.z = val;\n }\n }\n\n /**\n * The SPS is a single updatable mesh. The solid particles are simply separate parts or faces of this big mesh.\n *As it is just a mesh, the SPS has all the same properties than any other BJS mesh : not more, not less. It can be scaled, rotated, translated, enlighted, textured, moved, etc.\n\n * The SPS is also a particle system. It provides some methods to manage the particles.\n * However it is behavior agnostic. This means it has no emitter, no particle physics, no particle recycler. You have to implement your own behavior.\n *\n * Full documentation here : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_intro\n */\n class SolidParticleSystem {\n /**\n * Creates a SPS (Solid Particle System) object.\n * @param name (String) is the SPS name, this will be the underlying mesh name.\n * @param scene (Scene) is the scene in which the SPS is added.\n * @param options defines the options of the sps e.g.\n * * updatable (optional boolean, default true) : if the SPS must be updatable or immutable.\n * * isPickable (optional boolean, default false) : if the solid particles must be pickable.\n * * enableDepthSort (optional boolean, default false) : if the solid particles must be sorted in the geometry according to their distance to the camera.\n * * useModelMaterial (optional boolean, default false) : if the model materials must be used to create the SPS multimaterial. This enables the multimaterial supports of the SPS.\n * * enableMultiMaterial (optional boolean, default false) : if the solid particles can be given different materials.\n * * expandable (optional boolean, default false) : if particles can still be added after the initial SPS mesh creation.\n * * particleIntersection (optional boolean, default false) : if the solid particle intersections must be computed.\n * * boundingSphereOnly (optional boolean, default false) : if the particle intersection must be computed only with the bounding sphere (no bounding box computation, so faster).\n * * bSphereRadiusFactor (optional float, default 1.0) : a number to multiply the bounding sphere radius by in order to reduce it for instance.\n * * computeBoundingBox (optional boolean, default false): if the bounding box of the entire SPS will be computed (for occlusion detection, for example). If it is false, the bounding box will be the bounding box of the first particle.\n * * autoFixFaceOrientation (optional boolean, default false): if the particle face orientations will be flipped for transformations that change orientation (scale (-1, 1, 1), for example)\n * @param options.updatable\n * @param options.isPickable\n * @param options.enableDepthSort\n * @param options.particleIntersection\n * @param options.boundingSphereOnly\n * @param options.bSphereRadiusFactor\n * @param options.expandable\n * @param options.useModelMaterial\n * @param options.enableMultiMaterial\n * @param options.computeBoundingBox\n * @param options.autoFixFaceOrientation\n * @example bSphereRadiusFactor = 1.0 / Math.sqrt(3.0) => the bounding sphere exactly matches a spherical mesh.\n */\n constructor(name, scene, options) {\n /**\n * The SPS array of Solid Particle objects. Just access each particle as with any classic array.\n * Example : var p = SPS.particles[i];\n */\n this.particles = new Array();\n /**\n * The SPS total number of particles. Read only. Use SPS.counter instead if you need to set your own value.\n */\n this.nbParticles = 0;\n /**\n * If the particles must ever face the camera (default false). Useful for planar particles.\n */\n this.billboard = false;\n /**\n * Recompute normals when adding a shape\n */\n this.recomputeNormals = false;\n /**\n * This a counter ofr your own usage. It's not set by any SPS functions.\n */\n this.counter = 0;\n /**\n * This empty object is intended to store some SPS specific or temporary values in order to lower the Garbage Collector activity.\n * Please read : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/optimize_sps#limit-garbage-collection\n */\n this.vars = {};\n /**\n * If the particle intersection must be computed only with the bounding sphere (no bounding box computation, so faster). (Internal use only)\n * @internal\n */\n this._bSphereOnly = false;\n /**\n * A number to multiply the bounding sphere radius by in order to reduce it for instance. (Internal use only)\n * @internal\n */\n this._bSphereRadiusFactor = 1.0;\n this._positions = new Array();\n this._indices = new Array();\n this._normals = new Array();\n this._colors = new Array();\n this._uvs = new Array();\n this._index = 0; // indices index\n this._updatable = true;\n this._pickable = false;\n this._isVisibilityBoxLocked = false;\n this._alwaysVisible = false;\n this._depthSort = false;\n this._expandable = false;\n this._shapeCounter = 0;\n this._copy = new SolidParticle(0, 0, 0, 0, null, 0, 0, this);\n this._color = new Color4(0, 0, 0, 0);\n this._computeParticleColor = true;\n this._computeParticleTexture = true;\n this._computeParticleRotation = true;\n this._computeParticleVertex = false;\n this._computeBoundingBox = false;\n this._autoFixFaceOrientation = false;\n this._depthSortParticles = true;\n this._mustUnrotateFixedNormals = false;\n this._particlesIntersect = false;\n this._needs32Bits = false;\n this._isNotBuilt = true;\n this._lastParticleId = 0;\n this._idxOfId = []; // array : key = particle.id / value = particle.idx\n this._multimaterialEnabled = false;\n this._useModelMaterial = false;\n this._depthSortFunction = (p1, p2) => p2.sqDistance - p1.sqDistance;\n this._materialSortFunction = (p1, p2) => p1.materialIndex - p2.materialIndex;\n this._autoUpdateSubMeshes = false;\n this._recomputeInvisibles = false;\n this.name = name;\n this._scene = scene || EngineStore.LastCreatedScene;\n this._camera = scene.activeCamera;\n this._pickable = options ? options.isPickable : false;\n this._depthSort = options ? options.enableDepthSort : false;\n this._multimaterialEnabled = options ? options.enableMultiMaterial : false;\n this._useModelMaterial = options ? options.useModelMaterial : false;\n this._multimaterialEnabled = this._useModelMaterial ? true : this._multimaterialEnabled;\n this._expandable = options ? options.expandable : false;\n this._particlesIntersect = options ? options.particleIntersection : false;\n this._bSphereOnly = options ? options.boundingSphereOnly : false;\n this._bSphereRadiusFactor = options && options.bSphereRadiusFactor ? options.bSphereRadiusFactor : 1.0;\n this._computeBoundingBox = (options === null || options === void 0 ? void 0 : options.computeBoundingBox) ? options.computeBoundingBox : false;\n this._autoFixFaceOrientation = (options === null || options === void 0 ? void 0 : options.autoFixFaceOrientation) ? options.autoFixFaceOrientation : false;\n if (options && options.updatable !== undefined) {\n this._updatable = options.updatable;\n }\n else {\n this._updatable = true;\n }\n if (this._pickable) {\n this.pickedBySubMesh = [[]];\n this.pickedParticles = this.pickedBySubMesh[0];\n }\n if (this._depthSort || this._multimaterialEnabled) {\n this.depthSortedParticles = [];\n }\n if (this._multimaterialEnabled) {\n this._multimaterial = new MultiMaterial(this.name + \"MultiMaterial\", this._scene);\n this._materials = [];\n this._materialIndexesById = {};\n }\n this._tmpVertex = new SolidParticleVertex();\n }\n /**\n * Builds the SPS underlying mesh. Returns a standard Mesh.\n * If no model shape was added to the SPS, the returned mesh is just a single triangular plane.\n * @returns the created mesh\n */\n buildMesh() {\n if (!this._isNotBuilt && this.mesh) {\n return this.mesh;\n }\n if (this.nbParticles === 0 && !this.mesh) {\n const triangle = CreateDisc(\"\", { radius: 1, tessellation: 3 }, this._scene);\n this.addShape(triangle, 1);\n triangle.dispose();\n }\n this._indices32 = this._needs32Bits ? new Uint32Array(this._indices) : new Uint16Array(this._indices);\n this._positions32 = new Float32Array(this._positions);\n this._uvs32 = new Float32Array(this._uvs);\n this._colors32 = new Float32Array(this._colors);\n if (!this.mesh) {\n // in case it's already expanded\n const mesh = new Mesh(this.name, this._scene);\n this.mesh = mesh;\n }\n if (!this._updatable && this._multimaterialEnabled) {\n this._sortParticlesByMaterial(); // this may reorder the indices32\n }\n if (this.recomputeNormals) {\n VertexData.ComputeNormals(this._positions32, this._indices32, this._normals);\n }\n this._normals32 = new Float32Array(this._normals);\n this._fixedNormal32 = new Float32Array(this._normals);\n if (this._mustUnrotateFixedNormals) {\n // the particles could be created already rotated in the mesh with a positionFunction\n this._unrotateFixedNormals();\n }\n const vertexData = new VertexData();\n vertexData.indices = this._depthSort ? this._indices : this._indices32;\n vertexData.set(this._positions32, VertexBuffer.PositionKind);\n vertexData.set(this._normals32, VertexBuffer.NormalKind);\n if (this._uvs32.length > 0) {\n vertexData.set(this._uvs32, VertexBuffer.UVKind);\n }\n if (this._colors32.length > 0) {\n vertexData.set(this._colors32, VertexBuffer.ColorKind);\n }\n vertexData.applyToMesh(this.mesh, this._updatable);\n this.mesh.isPickable = this._pickable;\n if (this._pickable) {\n let faceId = 0;\n for (let p = 0; p < this.nbParticles; p++) {\n const part = this.particles[p];\n const lind = part._model._indicesLength;\n for (let i = 0; i < lind; i++) {\n const f = i % 3;\n if (f == 0) {\n const pickedData = { idx: part.idx, faceId: faceId };\n this.pickedParticles[faceId] = pickedData;\n faceId++;\n }\n }\n }\n }\n if (this._multimaterialEnabled) {\n this.setMultiMaterial(this._materials);\n }\n if (!this._expandable) {\n // free memory\n if (!this._depthSort && !this._multimaterialEnabled && !this._autoFixFaceOrientation) {\n this._indices = null;\n }\n this._positions = null;\n this._normals = null;\n this._uvs = null;\n this._colors = null;\n if (!this._updatable) {\n this.particles.length = 0;\n }\n }\n this._isNotBuilt = false;\n this.recomputeNormals = false;\n this._recomputeInvisibles = true;\n return this.mesh;\n }\n /**\n * Digests the mesh and generates as many solid particles in the system as wanted. Returns the SPS.\n * These particles will have the same geometry than the mesh parts and will be positioned at the same localisation than the mesh original places.\n * Thus the particles generated from `digest()` have their property `position` set yet.\n * @param mesh ( Mesh ) is the mesh to be digested\n * @param options {facetNb} (optional integer, default 1) is the number of mesh facets per particle, this parameter is overridden by the parameter `number` if any\n * {delta} (optional integer, default 0) is the random extra number of facets per particle , each particle will have between `facetNb` and `facetNb + delta` facets\n * {number} (optional positive integer) is the wanted number of particles : each particle is built with `mesh_total_facets / number` facets\n * {storage} (optional existing array) is an array where the particles will be stored for a further use instead of being inserted in the SPS.\n * @param options.facetNb\n * @param options.number\n * @param options.delta\n * @param options.storage\n * @returns the current SPS\n */\n digest(mesh, options) {\n let size = (options && options.facetNb) || 1;\n let number = (options && options.number) || 0;\n let delta = (options && options.delta) || 0;\n const meshPos = mesh.getVerticesData(VertexBuffer.PositionKind);\n const meshInd = mesh.getIndices();\n const meshUV = mesh.getVerticesData(VertexBuffer.UVKind);\n const meshCol = mesh.getVerticesData(VertexBuffer.ColorKind);\n const meshNor = mesh.getVerticesData(VertexBuffer.NormalKind);\n const storage = options && options.storage ? options.storage : null;\n let f = 0; // facet counter\n const totalFacets = meshInd.length / 3; // a facet is a triangle, so 3 indices\n // compute size from number\n if (number) {\n number = number > totalFacets ? totalFacets : number;\n size = Math.round(totalFacets / number);\n delta = 0;\n }\n else {\n size = size > totalFacets ? totalFacets : size;\n }\n const facetPos = []; // submesh positions\n const facetNor = [];\n const facetInd = []; // submesh indices\n const facetUV = []; // submesh UV\n const facetCol = []; // submesh colors\n const barycenter = Vector3.Zero();\n const sizeO = size;\n while (f < totalFacets) {\n size = sizeO + Math.floor((1 + delta) * Math.random());\n if (f > totalFacets - size) {\n size = totalFacets - f;\n }\n // reset temp arrays\n facetPos.length = 0;\n facetNor.length = 0;\n facetInd.length = 0;\n facetUV.length = 0;\n facetCol.length = 0;\n // iterate over \"size\" facets\n let fi = 0;\n for (let j = f * 3; j < (f + size) * 3; j++) {\n facetInd.push(fi);\n const i = meshInd[j];\n const i3 = i * 3;\n facetPos.push(meshPos[i3], meshPos[i3 + 1], meshPos[i3 + 2]);\n facetNor.push(meshNor[i3], meshNor[i3 + 1], meshNor[i3 + 2]);\n if (meshUV) {\n const i2 = i * 2;\n facetUV.push(meshUV[i2], meshUV[i2 + 1]);\n }\n if (meshCol) {\n const i4 = i * 4;\n facetCol.push(meshCol[i4], meshCol[i4 + 1], meshCol[i4 + 2], meshCol[i4 + 3]);\n }\n fi++;\n }\n // create a model shape for each single particle\n let idx = this.nbParticles;\n const shape = this._posToShape(facetPos);\n const shapeUV = this._uvsToShapeUV(facetUV);\n const shapeInd = facetInd.slice();\n const shapeCol = facetCol.slice();\n const shapeNor = facetNor.slice();\n // compute the barycenter of the shape\n barycenter.copyFromFloats(0, 0, 0);\n let v;\n for (v = 0; v < shape.length; v++) {\n barycenter.addInPlace(shape[v]);\n }\n barycenter.scaleInPlace(1 / shape.length);\n // shift the shape from its barycenter to the origin\n // and compute the BBox required for intersection.\n const minimum = new Vector3(Infinity, Infinity, Infinity);\n const maximum = new Vector3(-Infinity, -Infinity, -Infinity);\n for (v = 0; v < shape.length; v++) {\n shape[v].subtractInPlace(barycenter);\n minimum.minimizeInPlaceFromFloats(shape[v].x, shape[v].y, shape[v].z);\n maximum.maximizeInPlaceFromFloats(shape[v].x, shape[v].y, shape[v].z);\n }\n let bInfo;\n if (this._particlesIntersect) {\n bInfo = new BoundingInfo(minimum, maximum);\n }\n let material = null;\n if (this._useModelMaterial) {\n material = mesh.material ? mesh.material : this._setDefaultMaterial();\n }\n const modelShape = new ModelShape(this._shapeCounter, shape, shapeInd, shapeNor, shapeCol, shapeUV, null, null, material);\n // add the particle in the SPS\n const currentPos = this._positions.length;\n const currentInd = this._indices.length;\n this._meshBuilder(this._index, currentInd, shape, this._positions, shapeInd, this._indices, facetUV, this._uvs, shapeCol, this._colors, shapeNor, this._normals, idx, 0, null, modelShape);\n this._addParticle(idx, this._lastParticleId, currentPos, currentInd, modelShape, this._shapeCounter, 0, bInfo, storage);\n // initialize the particle position\n this.particles[this.nbParticles].position.addInPlace(barycenter);\n if (!storage) {\n this._index += shape.length;\n idx++;\n this.nbParticles++;\n this._lastParticleId++;\n }\n this._shapeCounter++;\n f += size;\n }\n this._isNotBuilt = true; // buildMesh() is now expected for setParticles() to work\n return this;\n }\n /**\n * Unrotate the fixed normals in case the mesh was built with pre-rotated particles, ex : use of positionFunction in addShape()\n * @internal\n */\n _unrotateFixedNormals() {\n let index = 0;\n let idx = 0;\n const tmpNormal = TmpVectors.Vector3[0];\n const quaternion = TmpVectors.Quaternion[0];\n const invertedRotMatrix = TmpVectors.Matrix[0];\n for (let p = 0; p < this.particles.length; p++) {\n const particle = this.particles[p];\n const shape = particle._model._shape;\n // computing the inverse of the rotation matrix from the quaternion\n // is equivalent to computing the matrix of the inverse quaternion, i.e of the conjugate quaternion\n if (particle.rotationQuaternion) {\n particle.rotationQuaternion.conjugateToRef(quaternion);\n }\n else {\n const rotation = particle.rotation;\n Quaternion.RotationYawPitchRollToRef(rotation.y, rotation.x, rotation.z, quaternion);\n quaternion.conjugateInPlace();\n }\n quaternion.toRotationMatrix(invertedRotMatrix);\n for (let pt = 0; pt < shape.length; pt++) {\n idx = index + pt * 3;\n Vector3.TransformNormalFromFloatsToRef(this._normals32[idx], this._normals32[idx + 1], this._normals32[idx + 2], invertedRotMatrix, tmpNormal);\n tmpNormal.toArray(this._fixedNormal32, idx);\n }\n index = idx + 3;\n }\n }\n /**\n * Resets the temporary working copy particle\n * @internal\n */\n _resetCopy() {\n const copy = this._copy;\n copy.position.setAll(0);\n copy.rotation.setAll(0);\n copy.rotationQuaternion = null;\n copy.scaling.setAll(1);\n copy.uvs.copyFromFloats(0.0, 0.0, 1.0, 1.0);\n copy.color = null;\n copy.translateFromPivot = false;\n copy.shapeId = 0;\n copy.materialIndex = null;\n }\n /**\n * Inserts the shape model geometry in the global SPS mesh by updating the positions, indices, normals, colors, uvs arrays\n * @param p the current index in the positions array to be updated\n * @param ind the current index in the indices array\n * @param shape a Vector3 array, the shape geometry\n * @param positions the positions array to be updated\n * @param meshInd the shape indices array\n * @param indices the indices array to be updated\n * @param meshUV the shape uv array\n * @param uvs the uv array to be updated\n * @param meshCol the shape color array\n * @param colors the color array to be updated\n * @param meshNor the shape normals array\n * @param normals the normals array to be updated\n * @param idx the particle index\n * @param idxInShape the particle index in its shape\n * @param options the addShape() method passed options\n * @param model\n * @model the particle model\n * @internal\n */\n _meshBuilder(p, ind, shape, positions, meshInd, indices, meshUV, uvs, meshCol, colors, meshNor, normals, idx, idxInShape, options, model) {\n let i;\n let u = 0;\n let c = 0;\n let n = 0;\n this._resetCopy();\n const copy = this._copy;\n const storeApart = options && options.storage ? true : false;\n copy.idx = idx;\n copy.idxInShape = idxInShape;\n copy.shapeId = model.shapeId;\n if (this._useModelMaterial) {\n const materialId = model._material.uniqueId;\n const materialIndexesById = this._materialIndexesById;\n if (!Object.prototype.hasOwnProperty.call(materialIndexesById, materialId)) {\n materialIndexesById[materialId] = this._materials.length;\n this._materials.push(model._material);\n }\n const matIdx = materialIndexesById[materialId];\n copy.materialIndex = matIdx;\n }\n if (options && options.positionFunction) {\n // call to custom positionFunction\n options.positionFunction(copy, idx, idxInShape);\n this._mustUnrotateFixedNormals = true;\n }\n // in case the particle geometry must NOT be inserted in the SPS mesh geometry\n if (storeApart) {\n return copy;\n }\n const rotMatrix = TmpVectors.Matrix[0];\n const tmpVertex = this._tmpVertex;\n const tmpVector = tmpVertex.position;\n const tmpColor = tmpVertex.color;\n const tmpUV = tmpVertex.uv;\n const tmpRotated = TmpVectors.Vector3[1];\n const pivotBackTranslation = TmpVectors.Vector3[2];\n const scaledPivot = TmpVectors.Vector3[3];\n Matrix.IdentityToRef(rotMatrix);\n copy.getRotationMatrix(rotMatrix);\n copy.pivot.multiplyToRef(copy.scaling, scaledPivot);\n if (copy.translateFromPivot) {\n pivotBackTranslation.setAll(0.0);\n }\n else {\n pivotBackTranslation.copyFrom(scaledPivot);\n }\n const someVertexFunction = options && options.vertexFunction;\n for (i = 0; i < shape.length; i++) {\n tmpVector.copyFrom(shape[i]);\n if (copy.color) {\n tmpColor.copyFrom(copy.color);\n }\n if (meshUV) {\n tmpUV.copyFromFloats(meshUV[u], meshUV[u + 1]);\n }\n if (someVertexFunction) {\n options.vertexFunction(copy, tmpVertex, i);\n }\n tmpVector.multiplyInPlace(copy.scaling).subtractInPlace(scaledPivot);\n Vector3.TransformCoordinatesToRef(tmpVector, rotMatrix, tmpRotated);\n tmpRotated.addInPlace(pivotBackTranslation).addInPlace(copy.position);\n positions.push(tmpRotated.x, tmpRotated.y, tmpRotated.z);\n if (meshUV) {\n const copyUvs = copy.uvs;\n uvs.push((copyUvs.z - copyUvs.x) * tmpUV.x + copyUvs.x, (copyUvs.w - copyUvs.y) * tmpUV.y + copyUvs.y);\n u += 2;\n }\n if (copy.color) {\n this._color.copyFrom(tmpColor);\n }\n else {\n const color = this._color;\n if (meshCol && meshCol[c] !== undefined) {\n color.r = meshCol[c];\n color.g = meshCol[c + 1];\n color.b = meshCol[c + 2];\n color.a = meshCol[c + 3];\n }\n else {\n color.r = 1.0;\n color.g = 1.0;\n color.b = 1.0;\n color.a = 1.0;\n }\n }\n colors.push(this._color.r, this._color.g, this._color.b, this._color.a);\n c += 4;\n if (!this.recomputeNormals && meshNor) {\n Vector3.TransformNormalFromFloatsToRef(meshNor[n], meshNor[n + 1], meshNor[n + 2], rotMatrix, tmpVector);\n normals.push(tmpVector.x, tmpVector.y, tmpVector.z);\n n += 3;\n }\n }\n for (i = 0; i < meshInd.length; i++) {\n const current_ind = p + meshInd[i];\n indices.push(current_ind);\n if (current_ind > 65535) {\n this._needs32Bits = true;\n }\n }\n if (this._depthSort || this._multimaterialEnabled) {\n const matIndex = copy.materialIndex !== null ? copy.materialIndex : 0;\n this.depthSortedParticles.push(new DepthSortedParticle(idx, ind, meshInd.length, matIndex));\n }\n return copy;\n }\n /**\n * Returns a shape Vector3 array from positions float array\n * @param positions float array\n * @returns a vector3 array\n * @internal\n */\n _posToShape(positions) {\n const shape = [];\n for (let i = 0; i < positions.length; i += 3) {\n shape.push(Vector3.FromArray(positions, i));\n }\n return shape;\n }\n /**\n * Returns a shapeUV array from a float uvs (array deep copy)\n * @param uvs as a float array\n * @returns a shapeUV array\n * @internal\n */\n _uvsToShapeUV(uvs) {\n const shapeUV = [];\n if (uvs) {\n for (let i = 0; i < uvs.length; i++) {\n shapeUV.push(uvs[i]);\n }\n }\n return shapeUV;\n }\n /**\n * Adds a new particle object in the particles array\n * @param idx particle index in particles array\n * @param id particle id\n * @param idxpos positionIndex : the starting index of the particle vertices in the SPS \"positions\" array\n * @param idxind indiceIndex : he starting index of the particle indices in the SPS \"indices\" array\n * @param model particle ModelShape object\n * @param shapeId model shape identifier\n * @param idxInShape index of the particle in the current model\n * @param bInfo model bounding info object\n * @param storage target storage array, if any\n * @internal\n */\n _addParticle(idx, id, idxpos, idxind, model, shapeId, idxInShape, bInfo = null, storage = null) {\n const sp = new SolidParticle(idx, id, idxpos, idxind, model, shapeId, idxInShape, this, bInfo);\n const target = storage ? storage : this.particles;\n target.push(sp);\n return sp;\n }\n /**\n * Adds some particles to the SPS from the model shape. Returns the shape id.\n * Please read the doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/immutable_sps\n * @param mesh is any Mesh object that will be used as a model for the solid particles.\n * @param nb (positive integer) the number of particles to be created from this model\n * @param options {positionFunction} is an optional javascript function to called for each particle on SPS creation.\n * {vertexFunction} is an optional javascript function to called for each vertex of each particle on SPS creation\n * {storage} (optional existing array) is an array where the particles will be stored for a further use instead of being inserted in the SPS.\n * @param options.positionFunction\n * @param options.vertexFunction\n * @param options.storage\n * @returns the number of shapes in the system\n */\n addShape(mesh, nb, options) {\n const meshPos = mesh.getVerticesData(VertexBuffer.PositionKind);\n const meshInd = mesh.getIndices();\n const meshUV = mesh.getVerticesData(VertexBuffer.UVKind);\n const meshCol = mesh.getVerticesData(VertexBuffer.ColorKind);\n const meshNor = mesh.getVerticesData(VertexBuffer.NormalKind);\n this.recomputeNormals = meshNor ? false : true;\n const indices = Array.from(meshInd);\n const shapeNormals = Array.from(meshNor);\n const shapeColors = meshCol ? Array.from(meshCol) : [];\n const storage = options && options.storage ? options.storage : null;\n let bbInfo = null;\n if (this._particlesIntersect) {\n bbInfo = mesh.getBoundingInfo();\n }\n const shape = this._posToShape(meshPos);\n const shapeUV = this._uvsToShapeUV(meshUV);\n const posfunc = options ? options.positionFunction : null;\n const vtxfunc = options ? options.vertexFunction : null;\n let material = null;\n if (this._useModelMaterial) {\n material = mesh.material ? mesh.material : this._setDefaultMaterial();\n }\n const modelShape = new ModelShape(this._shapeCounter, shape, indices, shapeNormals, shapeColors, shapeUV, posfunc, vtxfunc, material);\n // particles\n for (let i = 0; i < nb; i++) {\n this._insertNewParticle(this.nbParticles, i, modelShape, shape, meshInd, meshUV, meshCol, meshNor, bbInfo, storage, options);\n }\n this._shapeCounter++;\n this._isNotBuilt = true; // buildMesh() call is now expected for setParticles() to work\n return this._shapeCounter - 1;\n }\n /**\n * Rebuilds a particle back to its just built status : if needed, recomputes the custom positions and vertices\n * @internal\n */\n _rebuildParticle(particle, reset = false) {\n this._resetCopy();\n const copy = this._copy;\n if (particle._model._positionFunction) {\n // recall to stored custom positionFunction\n particle._model._positionFunction(copy, particle.idx, particle.idxInShape);\n }\n const rotMatrix = TmpVectors.Matrix[0];\n const tmpVertex = TmpVectors.Vector3[0];\n const tmpRotated = TmpVectors.Vector3[1];\n const pivotBackTranslation = TmpVectors.Vector3[2];\n const scaledPivot = TmpVectors.Vector3[3];\n copy.getRotationMatrix(rotMatrix);\n particle.pivot.multiplyToRef(particle.scaling, scaledPivot);\n if (copy.translateFromPivot) {\n pivotBackTranslation.copyFromFloats(0.0, 0.0, 0.0);\n }\n else {\n pivotBackTranslation.copyFrom(scaledPivot);\n }\n const shape = particle._model._shape;\n for (let pt = 0; pt < shape.length; pt++) {\n tmpVertex.copyFrom(shape[pt]);\n if (particle._model._vertexFunction) {\n particle._model._vertexFunction(copy, tmpVertex, pt); // recall to stored vertexFunction\n }\n tmpVertex.multiplyInPlace(copy.scaling).subtractInPlace(scaledPivot);\n Vector3.TransformCoordinatesToRef(tmpVertex, rotMatrix, tmpRotated);\n tmpRotated\n .addInPlace(pivotBackTranslation)\n .addInPlace(copy.position)\n .toArray(this._positions32, particle._pos + pt * 3);\n }\n if (reset) {\n particle.position.setAll(0.0);\n particle.rotation.setAll(0.0);\n particle.rotationQuaternion = null;\n particle.scaling.setAll(1.0);\n particle.uvs.setAll(0.0);\n particle.pivot.setAll(0.0);\n particle.translateFromPivot = false;\n particle.parentId = null;\n }\n }\n /**\n * Rebuilds the whole mesh and updates the VBO : custom positions and vertices are recomputed if needed.\n * @param reset boolean, default false : if the particles must be reset at position and rotation zero, scaling 1, color white, initial UVs and not parented.\n * @returns the SPS.\n */\n rebuildMesh(reset = false) {\n for (let p = 0; p < this.particles.length; p++) {\n this._rebuildParticle(this.particles[p], reset);\n }\n this.mesh.updateVerticesData(VertexBuffer.PositionKind, this._positions32, false, false);\n return this;\n }\n /** Removes the particles from the start-th to the end-th included from an expandable SPS (required).\n * Returns an array with the removed particles.\n * If the number of particles to remove is lower than zero or greater than the global remaining particle number, then an empty array is returned.\n * The SPS can't be empty so at least one particle needs to remain in place.\n * Under the hood, the VertexData array, so the VBO buffer, is recreated each call.\n * @param start index of the first particle to remove\n * @param end index of the last particle to remove (included)\n * @returns an array populated with the removed particles\n */\n removeParticles(start, end) {\n const nb = end - start + 1;\n if (!this._expandable || nb <= 0 || nb >= this.nbParticles || !this._updatable) {\n return [];\n }\n const particles = this.particles;\n const currentNb = this.nbParticles;\n if (end < currentNb - 1) {\n // update the particle indexes in the positions array in case they're remaining particles after the last removed\n const firstRemaining = end + 1;\n const shiftPos = particles[firstRemaining]._pos - particles[start]._pos;\n const shifInd = particles[firstRemaining]._ind - particles[start]._ind;\n for (let i = firstRemaining; i < currentNb; i++) {\n const part = particles[i];\n part._pos -= shiftPos;\n part._ind -= shifInd;\n }\n }\n const removed = particles.splice(start, nb);\n this._positions.length = 0;\n this._indices.length = 0;\n this._colors.length = 0;\n this._uvs.length = 0;\n this._normals.length = 0;\n this._index = 0;\n this._idxOfId.length = 0;\n if (this._depthSort || this._multimaterialEnabled) {\n this.depthSortedParticles = [];\n }\n let ind = 0;\n const particlesLength = particles.length;\n for (let p = 0; p < particlesLength; p++) {\n const particle = particles[p];\n const model = particle._model;\n const shape = model._shape;\n const modelIndices = model._indices;\n const modelNormals = model._normals;\n const modelColors = model._shapeColors;\n const modelUVs = model._shapeUV;\n particle.idx = p;\n this._idxOfId[particle.id] = p;\n this._meshBuilder(this._index, ind, shape, this._positions, modelIndices, this._indices, modelUVs, this._uvs, modelColors, this._colors, modelNormals, this._normals, particle.idx, particle.idxInShape, null, model);\n this._index += shape.length;\n ind += modelIndices.length;\n }\n this.nbParticles -= nb;\n this._isNotBuilt = true; // buildMesh() call is now expected for setParticles() to work\n return removed;\n }\n /**\n * Inserts some pre-created particles in the solid particle system so that they can be managed by setParticles().\n * @param solidParticleArray an array populated with Solid Particles objects\n * @returns the SPS\n */\n insertParticlesFromArray(solidParticleArray) {\n if (!this._expandable) {\n return this;\n }\n let idxInShape = 0;\n let currentShapeId = solidParticleArray[0].shapeId;\n const nb = solidParticleArray.length;\n for (let i = 0; i < nb; i++) {\n const sp = solidParticleArray[i];\n const model = sp._model;\n const shape = model._shape;\n const meshInd = model._indices;\n const meshUV = model._shapeUV;\n const meshCol = model._shapeColors;\n const meshNor = model._normals;\n const noNor = meshNor ? false : true;\n this.recomputeNormals = noNor || this.recomputeNormals;\n const bbInfo = sp.getBoundingInfo();\n const newPart = this._insertNewParticle(this.nbParticles, idxInShape, model, shape, meshInd, meshUV, meshCol, meshNor, bbInfo, null, null);\n sp.copyToRef(newPart);\n idxInShape++;\n if (currentShapeId != sp.shapeId) {\n currentShapeId = sp.shapeId;\n idxInShape = 0;\n }\n }\n this._isNotBuilt = true; // buildMesh() call is now expected for setParticles() to work\n return this;\n }\n /**\n * Creates a new particle and modifies the SPS mesh geometry :\n * - calls _meshBuilder() to increase the SPS mesh geometry step by step\n * - calls _addParticle() to populate the particle array\n * factorized code from addShape() and insertParticlesFromArray()\n * @param idx particle index in the particles array\n * @param i particle index in its shape\n * @param modelShape particle ModelShape object\n * @param shape shape vertex array\n * @param meshInd shape indices array\n * @param meshUV shape uv array\n * @param meshCol shape color array\n * @param meshNor shape normals array\n * @param bbInfo shape bounding info\n * @param storage target particle storage\n * @param options\n * @options addShape() passed options\n * @internal\n */\n _insertNewParticle(idx, i, modelShape, shape, meshInd, meshUV, meshCol, meshNor, bbInfo, storage, options) {\n const currentPos = this._positions.length;\n const currentInd = this._indices.length;\n const currentCopy = this._meshBuilder(this._index, currentInd, shape, this._positions, meshInd, this._indices, meshUV, this._uvs, meshCol, this._colors, meshNor, this._normals, idx, i, options, modelShape);\n let sp = null;\n if (this._updatable) {\n sp = this._addParticle(this.nbParticles, this._lastParticleId, currentPos, currentInd, modelShape, this._shapeCounter, i, bbInfo, storage);\n sp.position.copyFrom(currentCopy.position);\n sp.rotation.copyFrom(currentCopy.rotation);\n if (currentCopy.rotationQuaternion) {\n if (sp.rotationQuaternion) {\n sp.rotationQuaternion.copyFrom(currentCopy.rotationQuaternion);\n }\n else {\n sp.rotationQuaternion = currentCopy.rotationQuaternion.clone();\n }\n }\n if (currentCopy.color) {\n if (sp.color) {\n sp.color.copyFrom(currentCopy.color);\n }\n else {\n sp.color = currentCopy.color.clone();\n }\n }\n sp.scaling.copyFrom(currentCopy.scaling);\n sp.uvs.copyFrom(currentCopy.uvs);\n if (currentCopy.materialIndex !== null) {\n sp.materialIndex = currentCopy.materialIndex;\n }\n if (this.expandable) {\n this._idxOfId[sp.id] = sp.idx;\n }\n }\n if (!storage) {\n this._index += shape.length;\n this.nbParticles++;\n this._lastParticleId++;\n }\n return sp;\n }\n /**\n * Sets all the particles : this method actually really updates the mesh according to the particle positions, rotations, colors, textures, etc.\n * This method calls `updateParticle()` for each particle of the SPS.\n * For an animated SPS, it is usually called within the render loop.\n * This methods does nothing if called on a non updatable or not yet built SPS. Example : buildMesh() not called after having added or removed particles from an expandable SPS.\n * @param start The particle index in the particle array where to start to compute the particle property values _(default 0)_\n * @param end The particle index in the particle array where to stop to compute the particle property values _(default nbParticle - 1)_\n * @param update If the mesh must be finally updated on this call after all the particle computations _(default true)_\n * @returns the SPS.\n */\n setParticles(start = 0, end = this.nbParticles - 1, update = true) {\n if (!this._updatable || this._isNotBuilt) {\n return this;\n }\n // custom beforeUpdate\n this.beforeUpdateParticles(start, end, update);\n const rotMatrix = TmpVectors.Matrix[0];\n const invertedMatrix = TmpVectors.Matrix[1];\n const mesh = this.mesh;\n const colors32 = this._colors32;\n const positions32 = this._positions32;\n const normals32 = this._normals32;\n const uvs32 = this._uvs32;\n const indices32 = this._indices32;\n const indices = this._indices;\n const fixedNormal32 = this._fixedNormal32;\n const depthSortParticles = this._depthSort && this._depthSortParticles;\n const tempVectors = TmpVectors.Vector3;\n const camAxisX = tempVectors[5].copyFromFloats(1.0, 0.0, 0.0);\n const camAxisY = tempVectors[6].copyFromFloats(0.0, 1.0, 0.0);\n const camAxisZ = tempVectors[7].copyFromFloats(0.0, 0.0, 1.0);\n const minimum = tempVectors[8].setAll(Number.MAX_VALUE);\n const maximum = tempVectors[9].setAll(-Number.MAX_VALUE);\n const camInvertedPosition = tempVectors[10].setAll(0);\n const tmpVertex = this._tmpVertex;\n const tmpVector = tmpVertex.position;\n const tmpColor = tmpVertex.color;\n const tmpUV = tmpVertex.uv;\n // cases when the World Matrix is to be computed first\n if (this.billboard || this._depthSort) {\n this.mesh.computeWorldMatrix(true);\n this.mesh._worldMatrix.invertToRef(invertedMatrix);\n }\n // if the particles will always face the camera\n if (this.billboard) {\n // compute the camera position and un-rotate it by the current mesh rotation\n const tmpVector0 = tempVectors[0];\n this._camera.getDirectionToRef(Axis.Z, tmpVector0);\n Vector3.TransformNormalToRef(tmpVector0, invertedMatrix, camAxisZ);\n camAxisZ.normalize();\n // same for camera up vector extracted from the cam view matrix\n const view = this._camera.getViewMatrix(true);\n Vector3.TransformNormalFromFloatsToRef(view.m[1], view.m[5], view.m[9], invertedMatrix, camAxisY);\n Vector3.CrossToRef(camAxisY, camAxisZ, camAxisX);\n camAxisY.normalize();\n camAxisX.normalize();\n }\n // if depthSort, compute the camera global position in the mesh local system\n if (this._depthSort) {\n Vector3.TransformCoordinatesToRef(this._camera.globalPosition, invertedMatrix, camInvertedPosition); // then un-rotate the camera\n }\n Matrix.IdentityToRef(rotMatrix);\n let idx = 0; // current position index in the global array positions32\n let index = 0; // position start index in the global array positions32 of the current particle\n let colidx = 0; // current color index in the global array colors32\n let colorIndex = 0; // color start index in the global array colors32 of the current particle\n let uvidx = 0; // current uv index in the global array uvs32\n let uvIndex = 0; // uv start index in the global array uvs32 of the current particle\n let pt = 0; // current index in the particle model shape\n if (this.mesh.isFacetDataEnabled) {\n this._computeBoundingBox = true;\n }\n end = end >= this.nbParticles ? this.nbParticles - 1 : end;\n if (this._computeBoundingBox) {\n if (start != 0 || end != this.nbParticles - 1) {\n // only some particles are updated, then use the current existing BBox basis. Note : it can only increase.\n const boundingInfo = this.mesh.getBoundingInfo();\n if (boundingInfo) {\n minimum.copyFrom(boundingInfo.minimum);\n maximum.copyFrom(boundingInfo.maximum);\n }\n }\n }\n // particle loop\n index = this.particles[start]._pos;\n const vpos = (index / 3) | 0;\n colorIndex = vpos * 4;\n uvIndex = vpos * 2;\n for (let p = start; p <= end; p++) {\n const particle = this.particles[p];\n // call to custom user function to update the particle properties\n this.updateParticle(particle);\n const shape = particle._model._shape;\n const shapeUV = particle._model._shapeUV;\n const particleRotationMatrix = particle._rotationMatrix;\n const particlePosition = particle.position;\n const particleRotation = particle.rotation;\n const particleScaling = particle.scaling;\n const particleGlobalPosition = particle._globalPosition;\n // camera-particle distance for depth sorting\n if (depthSortParticles) {\n const dsp = this.depthSortedParticles[p];\n dsp.idx = particle.idx;\n dsp.ind = particle._ind;\n dsp.indicesLength = particle._model._indicesLength;\n dsp.sqDistance = Vector3.DistanceSquared(particle.position, camInvertedPosition);\n }\n // skip the computations for inactive or already invisible particles\n if (!particle.alive || (particle._stillInvisible && !particle.isVisible && !this._recomputeInvisibles)) {\n // increment indexes for the next particle\n pt = shape.length;\n index += pt * 3;\n colorIndex += pt * 4;\n uvIndex += pt * 2;\n continue;\n }\n if (particle.isVisible) {\n particle._stillInvisible = false; // un-mark permanent invisibility\n const scaledPivot = tempVectors[12];\n particle.pivot.multiplyToRef(particleScaling, scaledPivot);\n // particle rotation matrix\n if (this.billboard) {\n particleRotation.x = 0.0;\n particleRotation.y = 0.0;\n }\n if (this._computeParticleRotation || this.billboard) {\n particle.getRotationMatrix(rotMatrix);\n }\n const particleHasParent = particle.parentId !== null;\n if (particleHasParent) {\n const parent = this.getParticleById(particle.parentId);\n if (parent) {\n const parentRotationMatrix = parent._rotationMatrix;\n const parentGlobalPosition = parent._globalPosition;\n const rotatedY = particlePosition.x * parentRotationMatrix[1] + particlePosition.y * parentRotationMatrix[4] + particlePosition.z * parentRotationMatrix[7];\n const rotatedX = particlePosition.x * parentRotationMatrix[0] + particlePosition.y * parentRotationMatrix[3] + particlePosition.z * parentRotationMatrix[6];\n const rotatedZ = particlePosition.x * parentRotationMatrix[2] + particlePosition.y * parentRotationMatrix[5] + particlePosition.z * parentRotationMatrix[8];\n particleGlobalPosition.x = parentGlobalPosition.x + rotatedX;\n particleGlobalPosition.y = parentGlobalPosition.y + rotatedY;\n particleGlobalPosition.z = parentGlobalPosition.z + rotatedZ;\n if (this._computeParticleRotation || this.billboard) {\n const rotMatrixValues = rotMatrix.m;\n particleRotationMatrix[0] =\n rotMatrixValues[0] * parentRotationMatrix[0] + rotMatrixValues[1] * parentRotationMatrix[3] + rotMatrixValues[2] * parentRotationMatrix[6];\n particleRotationMatrix[1] =\n rotMatrixValues[0] * parentRotationMatrix[1] + rotMatrixValues[1] * parentRotationMatrix[4] + rotMatrixValues[2] * parentRotationMatrix[7];\n particleRotationMatrix[2] =\n rotMatrixValues[0] * parentRotationMatrix[2] + rotMatrixValues[1] * parentRotationMatrix[5] + rotMatrixValues[2] * parentRotationMatrix[8];\n particleRotationMatrix[3] =\n rotMatrixValues[4] * parentRotationMatrix[0] + rotMatrixValues[5] * parentRotationMatrix[3] + rotMatrixValues[6] * parentRotationMatrix[6];\n particleRotationMatrix[4] =\n rotMatrixValues[4] * parentRotationMatrix[1] + rotMatrixValues[5] * parentRotationMatrix[4] + rotMatrixValues[6] * parentRotationMatrix[7];\n particleRotationMatrix[5] =\n rotMatrixValues[4] * parentRotationMatrix[2] + rotMatrixValues[5] * parentRotationMatrix[5] + rotMatrixValues[6] * parentRotationMatrix[8];\n particleRotationMatrix[6] =\n rotMatrixValues[8] * parentRotationMatrix[0] + rotMatrixValues[9] * parentRotationMatrix[3] + rotMatrixValues[10] * parentRotationMatrix[6];\n particleRotationMatrix[7] =\n rotMatrixValues[8] * parentRotationMatrix[1] + rotMatrixValues[9] * parentRotationMatrix[4] + rotMatrixValues[10] * parentRotationMatrix[7];\n particleRotationMatrix[8] =\n rotMatrixValues[8] * parentRotationMatrix[2] + rotMatrixValues[9] * parentRotationMatrix[5] + rotMatrixValues[10] * parentRotationMatrix[8];\n }\n }\n else {\n // in case the parent were removed at some moment\n particle.parentId = null;\n }\n }\n else {\n particleGlobalPosition.x = particlePosition.x;\n particleGlobalPosition.y = particlePosition.y;\n particleGlobalPosition.z = particlePosition.z;\n if (this._computeParticleRotation || this.billboard) {\n const rotMatrixValues = rotMatrix.m;\n particleRotationMatrix[0] = rotMatrixValues[0];\n particleRotationMatrix[1] = rotMatrixValues[1];\n particleRotationMatrix[2] = rotMatrixValues[2];\n particleRotationMatrix[3] = rotMatrixValues[4];\n particleRotationMatrix[4] = rotMatrixValues[5];\n particleRotationMatrix[5] = rotMatrixValues[6];\n particleRotationMatrix[6] = rotMatrixValues[8];\n particleRotationMatrix[7] = rotMatrixValues[9];\n particleRotationMatrix[8] = rotMatrixValues[10];\n }\n }\n const pivotBackTranslation = tempVectors[11];\n if (particle.translateFromPivot) {\n pivotBackTranslation.setAll(0.0);\n }\n else {\n pivotBackTranslation.copyFrom(scaledPivot);\n }\n // particle vertex loop\n for (pt = 0; pt < shape.length; pt++) {\n idx = index + pt * 3;\n colidx = colorIndex + pt * 4;\n uvidx = uvIndex + pt * 2;\n const iu = 2 * pt;\n const iv = iu + 1;\n tmpVector.copyFrom(shape[pt]);\n if (this._computeParticleColor && particle.color) {\n tmpColor.copyFrom(particle.color);\n }\n if (this._computeParticleTexture) {\n tmpUV.copyFromFloats(shapeUV[iu], shapeUV[iv]);\n }\n if (this._computeParticleVertex) {\n this.updateParticleVertex(particle, tmpVertex, pt);\n }\n // positions\n const vertexX = tmpVector.x * particleScaling.x - scaledPivot.x;\n const vertexY = tmpVector.y * particleScaling.y - scaledPivot.y;\n const vertexZ = tmpVector.z * particleScaling.z - scaledPivot.z;\n let rotatedX = vertexX * particleRotationMatrix[0] + vertexY * particleRotationMatrix[3] + vertexZ * particleRotationMatrix[6];\n let rotatedY = vertexX * particleRotationMatrix[1] + vertexY * particleRotationMatrix[4] + vertexZ * particleRotationMatrix[7];\n let rotatedZ = vertexX * particleRotationMatrix[2] + vertexY * particleRotationMatrix[5] + vertexZ * particleRotationMatrix[8];\n rotatedX += pivotBackTranslation.x;\n rotatedY += pivotBackTranslation.y;\n rotatedZ += pivotBackTranslation.z;\n const px = (positions32[idx] = particleGlobalPosition.x + camAxisX.x * rotatedX + camAxisY.x * rotatedY + camAxisZ.x * rotatedZ);\n const py = (positions32[idx + 1] = particleGlobalPosition.y + camAxisX.y * rotatedX + camAxisY.y * rotatedY + camAxisZ.y * rotatedZ);\n const pz = (positions32[idx + 2] = particleGlobalPosition.z + camAxisX.z * rotatedX + camAxisY.z * rotatedY + camAxisZ.z * rotatedZ);\n if (this._computeBoundingBox) {\n minimum.minimizeInPlaceFromFloats(px, py, pz);\n maximum.maximizeInPlaceFromFloats(px, py, pz);\n }\n // normals : if the particles can't be morphed then just rotate the normals, what is much more faster than ComputeNormals()\n if (!this._computeParticleVertex) {\n const normalx = fixedNormal32[idx];\n const normaly = fixedNormal32[idx + 1];\n const normalz = fixedNormal32[idx + 2];\n const rotatedx = normalx * particleRotationMatrix[0] + normaly * particleRotationMatrix[3] + normalz * particleRotationMatrix[6];\n const rotatedy = normalx * particleRotationMatrix[1] + normaly * particleRotationMatrix[4] + normalz * particleRotationMatrix[7];\n const rotatedz = normalx * particleRotationMatrix[2] + normaly * particleRotationMatrix[5] + normalz * particleRotationMatrix[8];\n normals32[idx] = camAxisX.x * rotatedx + camAxisY.x * rotatedy + camAxisZ.x * rotatedz;\n normals32[idx + 1] = camAxisX.y * rotatedx + camAxisY.y * rotatedy + camAxisZ.y * rotatedz;\n normals32[idx + 2] = camAxisX.z * rotatedx + camAxisY.z * rotatedy + camAxisZ.z * rotatedz;\n }\n if (this._computeParticleColor && particle.color) {\n const colors32 = this._colors32;\n colors32[colidx] = tmpColor.r;\n colors32[colidx + 1] = tmpColor.g;\n colors32[colidx + 2] = tmpColor.b;\n colors32[colidx + 3] = tmpColor.a;\n }\n if (this._computeParticleTexture) {\n const uvs = particle.uvs;\n uvs32[uvidx] = tmpUV.x * (uvs.z - uvs.x) + uvs.x;\n uvs32[uvidx + 1] = tmpUV.y * (uvs.w - uvs.y) + uvs.y;\n }\n }\n }\n // particle just set invisible : scaled to zero and positioned at the origin\n else {\n particle._stillInvisible = true; // mark the particle as invisible\n for (pt = 0; pt < shape.length; pt++) {\n idx = index + pt * 3;\n colidx = colorIndex + pt * 4;\n uvidx = uvIndex + pt * 2;\n positions32[idx] = positions32[idx + 1] = positions32[idx + 2] = 0;\n normals32[idx] = normals32[idx + 1] = normals32[idx + 2] = 0;\n if (this._computeParticleColor && particle.color) {\n const color = particle.color;\n colors32[colidx] = color.r;\n colors32[colidx + 1] = color.g;\n colors32[colidx + 2] = color.b;\n colors32[colidx + 3] = color.a;\n }\n if (this._computeParticleTexture) {\n const uvs = particle.uvs;\n uvs32[uvidx] = shapeUV[pt * 2] * (uvs.z - uvs.x) + uvs.x;\n uvs32[uvidx + 1] = shapeUV[pt * 2 + 1] * (uvs.w - uvs.y) + uvs.y;\n }\n }\n }\n // if the particle intersections must be computed : update the bbInfo\n if (this._particlesIntersect) {\n const bInfo = particle.getBoundingInfo();\n const bBox = bInfo.boundingBox;\n const bSphere = bInfo.boundingSphere;\n const modelBoundingInfo = particle._modelBoundingInfo;\n if (!this._bSphereOnly) {\n // place, scale and rotate the particle bbox within the SPS local system, then update it\n const modelBoundingInfoVectors = modelBoundingInfo.boundingBox.vectors;\n const tempMin = tempVectors[1];\n const tempMax = tempVectors[2];\n tempMin.setAll(Number.MAX_VALUE);\n tempMax.setAll(-Number.MAX_VALUE);\n for (let b = 0; b < 8; b++) {\n const scaledX = modelBoundingInfoVectors[b].x * particleScaling.x;\n const scaledY = modelBoundingInfoVectors[b].y * particleScaling.y;\n const scaledZ = modelBoundingInfoVectors[b].z * particleScaling.z;\n const rotatedX = scaledX * particleRotationMatrix[0] + scaledY * particleRotationMatrix[3] + scaledZ * particleRotationMatrix[6];\n const rotatedY = scaledX * particleRotationMatrix[1] + scaledY * particleRotationMatrix[4] + scaledZ * particleRotationMatrix[7];\n const rotatedZ = scaledX * particleRotationMatrix[2] + scaledY * particleRotationMatrix[5] + scaledZ * particleRotationMatrix[8];\n const x = particlePosition.x + camAxisX.x * rotatedX + camAxisY.x * rotatedY + camAxisZ.x * rotatedZ;\n const y = particlePosition.y + camAxisX.y * rotatedX + camAxisY.y * rotatedY + camAxisZ.y * rotatedZ;\n const z = particlePosition.z + camAxisX.z * rotatedX + camAxisY.z * rotatedY + camAxisZ.z * rotatedZ;\n tempMin.minimizeInPlaceFromFloats(x, y, z);\n tempMax.maximizeInPlaceFromFloats(x, y, z);\n }\n bBox.reConstruct(tempMin, tempMax, mesh._worldMatrix);\n }\n // place and scale the particle bouding sphere in the SPS local system, then update it\n const minBbox = modelBoundingInfo.minimum.multiplyToRef(particleScaling, tempVectors[1]);\n const maxBbox = modelBoundingInfo.maximum.multiplyToRef(particleScaling, tempVectors[2]);\n const bSphereCenter = maxBbox.addToRef(minBbox, tempVectors[3]).scaleInPlace(0.5).addInPlace(particleGlobalPosition);\n const halfDiag = maxBbox.subtractToRef(minBbox, tempVectors[4]).scaleInPlace(0.5 * this._bSphereRadiusFactor);\n const bSphereMinBbox = bSphereCenter.subtractToRef(halfDiag, tempVectors[1]);\n const bSphereMaxBbox = bSphereCenter.addToRef(halfDiag, tempVectors[2]);\n bSphere.reConstruct(bSphereMinBbox, bSphereMaxBbox, mesh._worldMatrix);\n }\n // increment indexes for the next particle\n index = idx + 3;\n colorIndex = colidx + 4;\n uvIndex = uvidx + 2;\n }\n // if the VBO must be updated\n if (update) {\n if (this._computeParticleColor) {\n const vb = mesh.getVertexBuffer(VertexBuffer.ColorKind);\n if (vb && !mesh.isPickable) {\n vb.updateDirectly(colors32, 0);\n }\n else {\n mesh.updateVerticesData(VertexBuffer.ColorKind, colors32, false, false);\n }\n }\n if (this._computeParticleTexture) {\n const vb = mesh.getVertexBuffer(VertexBuffer.UVKind);\n if (vb && !mesh.isPickable) {\n vb.updateDirectly(uvs32, 0);\n }\n else {\n mesh.updateVerticesData(VertexBuffer.UVKind, uvs32, false, false);\n }\n }\n const vbp = mesh.getVertexBuffer(VertexBuffer.PositionKind);\n if (vbp && !mesh.isPickable) {\n vbp.updateDirectly(positions32, 0);\n }\n else {\n mesh.updateVerticesData(VertexBuffer.PositionKind, positions32, false, false);\n }\n if (!mesh.areNormalsFrozen || mesh.isFacetDataEnabled) {\n if (this._computeParticleVertex || mesh.isFacetDataEnabled) {\n // recompute the normals only if the particles can be morphed, update then also the normal reference array _fixedNormal32[]\n const params = mesh.isFacetDataEnabled ? mesh.getFacetDataParameters() : null;\n VertexData.ComputeNormals(positions32, indices32, normals32, params);\n for (let i = 0; i < normals32.length; i++) {\n fixedNormal32[i] = normals32[i];\n }\n }\n if (!mesh.areNormalsFrozen) {\n const vb = mesh.getVertexBuffer(VertexBuffer.NormalKind);\n if (vb && !mesh.isPickable) {\n vb.updateDirectly(normals32, 0);\n }\n else {\n mesh.updateVerticesData(VertexBuffer.NormalKind, normals32, false, false);\n }\n }\n }\n if (depthSortParticles) {\n const depthSortedParticles = this.depthSortedParticles;\n depthSortedParticles.sort(this._depthSortFunction);\n const dspl = depthSortedParticles.length;\n let sid = 0;\n let faceId = 0;\n for (let sorted = 0; sorted < dspl; sorted++) {\n const sortedParticle = depthSortedParticles[sorted];\n const lind = sortedParticle.indicesLength;\n const sind = sortedParticle.ind;\n for (let i = 0; i < lind; i++) {\n indices32[sid] = indices[sind + i];\n sid++;\n if (this._pickable) {\n const f = i % 3;\n if (f == 0) {\n const pickedData = this.pickedParticles[faceId];\n pickedData.idx = sortedParticle.idx;\n pickedData.faceId = faceId;\n faceId++;\n }\n }\n }\n }\n }\n if (this._autoFixFaceOrientation) {\n let particleInd = 0;\n for (let particleIdx = 0; particleIdx < this.particles.length; particleIdx++) {\n const particle = depthSortParticles ? this.particles[this.depthSortedParticles[particleIdx].idx] : this.particles[particleIdx];\n const flipFaces = particle.scale.x * particle.scale.y * particle.scale.z < 0;\n if (flipFaces) {\n for (let faceInd = 0; faceInd < particle._model._indicesLength; faceInd += 3) {\n const tmp = indices[particle._ind + faceInd];\n indices32[particleInd + faceInd] = indices[particle._ind + faceInd + 1];\n indices32[particleInd + faceInd + 1] = tmp;\n }\n }\n particleInd += particle._model._indicesLength;\n }\n }\n if (depthSortParticles || this._autoFixFaceOrientation) {\n mesh.updateIndices(indices32);\n }\n }\n if (this._computeBoundingBox) {\n if (mesh.hasBoundingInfo) {\n mesh.getBoundingInfo().reConstruct(minimum, maximum, mesh._worldMatrix);\n }\n else {\n mesh.buildBoundingInfo(minimum, maximum, mesh._worldMatrix);\n }\n }\n if (this._autoUpdateSubMeshes) {\n this.computeSubMeshes();\n }\n this._recomputeInvisibles = false;\n this.afterUpdateParticles(start, end, update);\n return this;\n }\n /**\n * Disposes the SPS.\n */\n dispose() {\n this.mesh.dispose();\n this.vars = null;\n // drop references to internal big arrays for the GC\n this._positions = null;\n this._indices = null;\n this._normals = null;\n this._uvs = null;\n this._colors = null;\n this._indices32 = null;\n this._positions32 = null;\n this._normals32 = null;\n this._fixedNormal32 = null;\n this._uvs32 = null;\n this._colors32 = null;\n this.pickedParticles = null;\n this.pickedBySubMesh = null;\n this._materials = null;\n this._materialIndexes = null;\n this._indicesByMaterial = null;\n this._idxOfId = null;\n }\n /** Returns an object {idx: number faceId: number} for the picked particle from the passed pickingInfo object.\n * idx is the particle index in the SPS\n * faceId is the picked face index counted within this particle.\n * Returns null if the pickInfo can't identify a picked particle.\n * @param pickingInfo (PickingInfo object)\n * @returns {idx: number, faceId: number} or null\n */\n pickedParticle(pickingInfo) {\n if (pickingInfo.hit) {\n const subMesh = pickingInfo.subMeshId;\n const faceId = pickingInfo.faceId - this.mesh.subMeshes[subMesh].indexStart / 3;\n const picked = this.pickedBySubMesh;\n if (picked[subMesh] && picked[subMesh][faceId]) {\n return picked[subMesh][faceId];\n }\n }\n return null;\n }\n /**\n * Returns a SolidParticle object from its identifier : particle.id\n * @param id (integer) the particle Id\n * @returns the searched particle or null if not found in the SPS.\n */\n getParticleById(id) {\n const p = this.particles[id];\n if (p && p.id == id) {\n return p;\n }\n const particles = this.particles;\n const idx = this._idxOfId[id];\n if (idx !== undefined) {\n return particles[idx];\n }\n let i = 0;\n const nb = this.nbParticles;\n while (i < nb) {\n const particle = particles[i];\n if (particle.id == id) {\n return particle;\n }\n i++;\n }\n return null;\n }\n /**\n * Returns a new array populated with the particles having the passed shapeId.\n * @param shapeId (integer) the shape identifier\n * @returns a new solid particle array\n */\n getParticlesByShapeId(shapeId) {\n const ref = [];\n this.getParticlesByShapeIdToRef(shapeId, ref);\n return ref;\n }\n /**\n * Populates the passed array \"ref\" with the particles having the passed shapeId.\n * @param shapeId the shape identifier\n * @returns the SPS\n * @param ref\n */\n getParticlesByShapeIdToRef(shapeId, ref) {\n ref.length = 0;\n for (let i = 0; i < this.nbParticles; i++) {\n const p = this.particles[i];\n if (p.shapeId == shapeId) {\n ref.push(p);\n }\n }\n return this;\n }\n /**\n * Computes the required SubMeshes according the materials assigned to the particles.\n * @returns the solid particle system.\n * Does nothing if called before the SPS mesh is built.\n */\n computeSubMeshes() {\n if (!this.mesh || !this._multimaterialEnabled) {\n return this;\n }\n const depthSortedParticles = this.depthSortedParticles;\n if (this.particles.length > 0) {\n for (let p = 0; p < this.particles.length; p++) {\n const part = this.particles[p];\n if (!part.materialIndex) {\n part.materialIndex = 0;\n }\n const sortedPart = depthSortedParticles[p];\n sortedPart.materialIndex = part.materialIndex;\n sortedPart.ind = part._ind;\n sortedPart.indicesLength = part._model._indicesLength;\n sortedPart.idx = part.idx;\n }\n }\n this._sortParticlesByMaterial();\n const indicesByMaterial = this._indicesByMaterial;\n const materialIndexes = this._materialIndexes;\n const mesh = this.mesh;\n mesh.subMeshes = [];\n const vcount = mesh.getTotalVertices();\n for (let m = 0; m < materialIndexes.length; m++) {\n const start = indicesByMaterial[m];\n const count = indicesByMaterial[m + 1] - start;\n const matIndex = materialIndexes[m];\n new SubMesh(matIndex, 0, vcount, start, count, mesh);\n }\n return this;\n }\n /**\n * Sorts the solid particles by material when MultiMaterial is enabled.\n * Updates the indices32 array.\n * Updates the indicesByMaterial array.\n * Updates the mesh indices array.\n * @returns the SPS\n * @internal\n */\n _sortParticlesByMaterial() {\n const indicesByMaterial = [0];\n this._indicesByMaterial = indicesByMaterial;\n const materialIndexes = [];\n this._materialIndexes = materialIndexes;\n const depthSortedParticles = this.depthSortedParticles;\n depthSortedParticles.sort(this._materialSortFunction);\n const length = depthSortedParticles.length;\n const indices32 = this._indices32;\n const indices = this._indices;\n let subMeshIndex = 0;\n let subMeshFaceId = 0;\n let sid = 0;\n let lastMatIndex = depthSortedParticles[0].materialIndex;\n materialIndexes.push(lastMatIndex);\n if (this._pickable) {\n this.pickedBySubMesh = [[]];\n this.pickedParticles = this.pickedBySubMesh[0];\n }\n for (let sorted = 0; sorted < length; sorted++) {\n const sortedPart = depthSortedParticles[sorted];\n const lind = sortedPart.indicesLength;\n const sind = sortedPart.ind;\n if (sortedPart.materialIndex !== lastMatIndex) {\n lastMatIndex = sortedPart.materialIndex;\n indicesByMaterial.push(sid);\n materialIndexes.push(lastMatIndex);\n if (this._pickable) {\n subMeshIndex++;\n this.pickedBySubMesh[subMeshIndex] = [];\n subMeshFaceId = 0;\n }\n }\n let faceId = 0;\n for (let i = 0; i < lind; i++) {\n indices32[sid] = indices[sind + i];\n if (this._pickable) {\n const f = i % 3;\n if (f == 0) {\n const pickedData = this.pickedBySubMesh[subMeshIndex][subMeshFaceId];\n if (pickedData) {\n pickedData.idx = sortedPart.idx;\n pickedData.faceId = faceId;\n }\n else {\n this.pickedBySubMesh[subMeshIndex][subMeshFaceId] = { idx: sortedPart.idx, faceId: faceId };\n }\n subMeshFaceId++;\n faceId++;\n }\n }\n sid++;\n }\n }\n indicesByMaterial.push(indices32.length); // add the last number to ease the indices start/count values for subMeshes creation\n if (this._updatable) {\n this.mesh.updateIndices(indices32);\n }\n return this;\n }\n /**\n * Sets the material indexes by id materialIndexesById[id] = materialIndex\n * @internal\n */\n _setMaterialIndexesById() {\n this._materialIndexesById = {};\n for (let i = 0; i < this._materials.length; i++) {\n const id = this._materials[i].uniqueId;\n this._materialIndexesById[id] = i;\n }\n }\n /**\n * Returns an array with unique values of Materials from the passed array\n * @param array the material array to be checked and filtered\n * @internal\n */\n _filterUniqueMaterialId(array) {\n const filtered = array.filter(function (value, index, self) {\n return self.indexOf(value) === index;\n });\n return filtered;\n }\n /**\n * Sets a new Standard Material as _defaultMaterial if not already set.\n * @internal\n */\n _setDefaultMaterial() {\n if (!this._defaultMaterial) {\n this._defaultMaterial = new StandardMaterial(this.name + \"DefaultMaterial\", this._scene);\n }\n return this._defaultMaterial;\n }\n /**\n * Visibility helper : Recomputes the visible size according to the mesh bounding box\n * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_visibility\n * @returns the SPS.\n */\n refreshVisibleSize() {\n if (!this._isVisibilityBoxLocked) {\n this.mesh.refreshBoundingInfo();\n }\n return this;\n }\n /**\n * Visibility helper : Sets the size of a visibility box, this sets the underlying mesh bounding box.\n * @param size the size (float) of the visibility box\n * note : this doesn't lock the SPS mesh bounding box.\n * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_visibility\n */\n setVisibilityBox(size) {\n const vis = size / 2;\n this.mesh.buildBoundingInfo(new Vector3(-vis, -vis, -vis), new Vector3(vis, vis, vis));\n }\n /**\n * Gets whether the SPS as always visible or not\n * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_visibility\n */\n get isAlwaysVisible() {\n return this._alwaysVisible;\n }\n /**\n * Sets the SPS as always visible or not\n * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_visibility\n */\n set isAlwaysVisible(val) {\n this._alwaysVisible = val;\n this.mesh.alwaysSelectAsActiveMesh = val;\n }\n /**\n * Sets the SPS visibility box as locked or not. This enables/disables the underlying mesh bounding box updates.\n * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_visibility\n */\n set isVisibilityBoxLocked(val) {\n this._isVisibilityBoxLocked = val;\n const boundingInfo = this.mesh.getBoundingInfo();\n boundingInfo.isLocked = val;\n }\n /**\n * Gets if the SPS visibility box as locked or not. This enables/disables the underlying mesh bounding box updates.\n * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_visibility\n */\n get isVisibilityBoxLocked() {\n return this._isVisibilityBoxLocked;\n }\n /**\n * Tells to `setParticles()` to compute the particle rotations or not.\n * Default value : true. The SPS is faster when it's set to false.\n * Note : the particle rotations aren't stored values, so setting `computeParticleRotation` to false will prevents the particle to rotate.\n */\n set computeParticleRotation(val) {\n this._computeParticleRotation = val;\n }\n /**\n * Tells to `setParticles()` to compute the particle colors or not.\n * Default value : true. The SPS is faster when it's set to false.\n * Note : the particle colors are stored values, so setting `computeParticleColor` to false will keep yet the last colors set.\n */\n set computeParticleColor(val) {\n this._computeParticleColor = val;\n }\n set computeParticleTexture(val) {\n this._computeParticleTexture = val;\n }\n /**\n * Tells to `setParticles()` to call the vertex function for each vertex of each particle, or not.\n * Default value : false. The SPS is faster when it's set to false.\n * Note : the particle custom vertex positions aren't stored values.\n */\n set computeParticleVertex(val) {\n this._computeParticleVertex = val;\n }\n /**\n * Tells to `setParticles()` to compute or not the mesh bounding box when computing the particle positions.\n */\n set computeBoundingBox(val) {\n this._computeBoundingBox = val;\n }\n /**\n * Tells to `setParticles()` to sort or not the distance between each particle and the camera.\n * Skipped when `enableDepthSort` is set to `false` (default) at construction time.\n * Default : `true`\n */\n set depthSortParticles(val) {\n this._depthSortParticles = val;\n }\n /**\n * Gets if `setParticles()` computes the particle rotations or not.\n * Default value : true. The SPS is faster when it's set to false.\n * Note : the particle rotations aren't stored values, so setting `computeParticleRotation` to false will prevents the particle to rotate.\n */\n get computeParticleRotation() {\n return this._computeParticleRotation;\n }\n /**\n * Gets if `setParticles()` computes the particle colors or not.\n * Default value : true. The SPS is faster when it's set to false.\n * Note : the particle colors are stored values, so setting `computeParticleColor` to false will keep yet the last colors set.\n */\n get computeParticleColor() {\n return this._computeParticleColor;\n }\n /**\n * Gets if `setParticles()` computes the particle textures or not.\n * Default value : true. The SPS is faster when it's set to false.\n * Note : the particle textures are stored values, so setting `computeParticleTexture` to false will keep yet the last colors set.\n */\n get computeParticleTexture() {\n return this._computeParticleTexture;\n }\n /**\n * Gets if `setParticles()` calls the vertex function for each vertex of each particle, or not.\n * Default value : false. The SPS is faster when it's set to false.\n * Note : the particle custom vertex positions aren't stored values.\n */\n get computeParticleVertex() {\n return this._computeParticleVertex;\n }\n /**\n * Gets if `setParticles()` computes or not the mesh bounding box when computing the particle positions.\n */\n get computeBoundingBox() {\n return this._computeBoundingBox;\n }\n /**\n * Gets if `setParticles()` sorts or not the distance between each particle and the camera.\n * Skipped when `enableDepthSort` is set to `false` (default) at construction time.\n * Default : `true`\n */\n get depthSortParticles() {\n return this._depthSortParticles;\n }\n /**\n * Gets if the SPS is created as expandable at construction time.\n * Default : `false`\n */\n get expandable() {\n return this._expandable;\n }\n /**\n * Gets if the SPS supports the Multi Materials\n */\n get multimaterialEnabled() {\n return this._multimaterialEnabled;\n }\n /**\n * Gets if the SPS uses the model materials for its own multimaterial.\n */\n get useModelMaterial() {\n return this._useModelMaterial;\n }\n /**\n * The SPS used material array.\n */\n get materials() {\n return this._materials;\n }\n /**\n * Sets the SPS MultiMaterial from the passed materials.\n * Note : the passed array is internally copied and not used then by reference.\n * @param materials an array of material objects. This array indexes are the materialIndex values of the particles.\n */\n setMultiMaterial(materials) {\n this._materials = this._filterUniqueMaterialId(materials);\n this._setMaterialIndexesById();\n if (this._multimaterial) {\n this._multimaterial.dispose();\n }\n this._multimaterial = new MultiMaterial(this.name + \"MultiMaterial\", this._scene);\n for (let m = 0; m < this._materials.length; m++) {\n this._multimaterial.subMaterials.push(this._materials[m]);\n }\n this.computeSubMeshes();\n this.mesh.material = this._multimaterial;\n }\n /**\n * The SPS computed multimaterial object\n */\n get multimaterial() {\n return this._multimaterial;\n }\n set multimaterial(mm) {\n this._multimaterial = mm;\n }\n /**\n * If the subMeshes must be updated on the next call to setParticles()\n */\n get autoUpdateSubMeshes() {\n return this._autoUpdateSubMeshes;\n }\n set autoUpdateSubMeshes(val) {\n this._autoUpdateSubMeshes = val;\n }\n // =======================================================================\n // Particle behavior logic\n // these following methods may be overwritten by the user to fit his needs\n /**\n * This function does nothing. It may be overwritten to set all the particle first values.\n * The SPS doesn't call this function, you may have to call it by your own.\n * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/manage_sps_particles\n */\n initParticles() { }\n /**\n * This function does nothing. It may be overwritten to recycle a particle.\n * The SPS doesn't call this function, you may have to call it by your own.\n * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/manage_sps_particles\n * @param particle The particle to recycle\n * @returns the recycled particle\n */\n recycleParticle(particle) {\n return particle;\n }\n /**\n * Updates a particle : this function should be overwritten by the user.\n * It is called on each particle by `setParticles()`. This is the place to code each particle behavior.\n * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/manage_sps_particles\n * @example : just set a particle position or velocity and recycle conditions\n * @param particle The particle to update\n * @returns the updated particle\n */\n updateParticle(particle) {\n return particle;\n }\n /**\n * Updates a vertex of a particle : it can be overwritten by the user.\n * This will be called on each vertex particle by `setParticles()` if `computeParticleVertex` is set to true only.\n * @param particle the current particle\n * @param vertex the current vertex of the current particle : a SolidParticleVertex object\n * @param pt the index of the current vertex in the particle shape\n * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_vertices\n * @example : just set a vertex particle position or color\n * @returns the sps\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n updateParticleVertex(particle, vertex, pt) {\n return this;\n }\n /**\n * This will be called before any other treatment by `setParticles()` and will be passed three parameters.\n * This does nothing and may be overwritten by the user.\n * @param start the particle index in the particle array where to stop to iterate, same than the value passed to setParticle()\n * @param stop the particle index in the particle array where to stop to iterate, same than the value passed to setParticle()\n * @param update the boolean update value actually passed to setParticles()\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n beforeUpdateParticles(start, stop, update) { }\n /**\n * This will be called by `setParticles()` after all the other treatments and just before the actual mesh update.\n * This will be passed three parameters.\n * This does nothing and may be overwritten by the user.\n * @param start the particle index in the particle array where to stop to iterate, same than the value passed to setParticle()\n * @param stop the particle index in the particle array where to stop to iterate, same than the value passed to setParticle()\n * @param update the boolean update value actually passed to setParticles()\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n afterUpdateParticles(start, stop, update) { }\n }\n\n /** This will mutate the mesh geometry. Only use this on temp/intermediate mesh data */\n function ensureNormalsExist(mesh) {\n if (!mesh.getVerticesData(VertexBuffer.NormalKind)) {\n mesh.createNormals(true);\n mesh.convertToFlatShadedMesh();\n }\n }\n\n const patternStore = new Map();\n const identityRotation = Quaternion.Identity();\n class PatternRenderer extends FeatureRenderer {\n update3dInternal() { }\n updateMeshGeometry(feature, vertexData, vertexDataChanges, setParticlePositionFunc) {\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\n const meshNode = feature._parent;\n const mesh = this.viewer.getRendererNode(meshNode);\n if (mesh) {\n /** See line 156 */\n // let _pattern = patternStore.get(meshNode._dynamicId);\n let _pattern = null;\n // If there are no enabled pattern features before this one, update the pattern shape as some geometry may have changed\n if (!_pattern) {\n const meshUpdated = new Mesh('patternTempGeometry');\n meshUpdated.setEnabled(false);\n const tempVertexData = new SubmeshVertexData();\n tempVertexData.copyFrom(newVertexData);\n tempVertexData.applyToMesh(meshUpdated, false);\n ensureNormalsExist(meshUpdated);\n const sps = new SolidParticleSystem(`${mesh.id}_${feature._dynamicId}`, this.scene, {\n enableDepthSort: true,\n enableMultiMaterial: true,\n expandable: true,\n });\n if (feature) {\n if (feature.meshId) {\n const sketchNode = this.viewer.getNode(feature.meshId);\n if (sketchNode instanceof exports.SketchNode) {\n let distance = feature.distance;\n if (distance <= 0) {\n distance = 1;\n }\n const positionArray = this.getPositionArray(sketchNode, feature.distance, feature.addEndMesh);\n sps.addShape(meshUpdated, positionArray.length);\n sps.counter = positionArray.length;\n feature.count = positionArray.length;\n const sketchPosition = sketchNode.position.toVec3();\n sps.initParticles = () => {\n for (let p = 0; p < sps.counter; p++) {\n const particle = sps.particles[p];\n particle.position = positionArray[p].add(sketchPosition);\n }\n };\n }\n }\n else {\n sps.addShape(meshUpdated, feature.count + 1);\n sps.initParticles = () => {\n for (let p = 0; p < sps.nbParticles; p++) {\n const particle = sps.particles[p];\n if (particle.rotationQuaternion) {\n particle.rotationQuaternion.copyFrom(identityRotation);\n }\n else {\n particle.rotationQuaternion = Quaternion.Identity();\n }\n particle.position.setAll(0);\n }\n };\n }\n }\n const newPattern = {\n sps,\n size: 1,\n featureUpdaters: new Map(),\n sourceNode: meshNode,\n needsUpdate: true,\n needsRebuild: true,\n needsSubmeshRebuild: true,\n shape: meshUpdated,\n };\n sps.updateParticle = particle => {\n // Process the last update function as it will call the prior ones recursively\n const updates = newPattern.featureUpdaters.values();\n const update = updates.next().value;\n update(particle, updates);\n // Do we need this?\n // particle.scaling.copyFrom(mesh.scaling);\n return particle;\n };\n patternStore.set(meshNode._dynamicId, newPattern);\n _pattern = newPattern;\n }\n const pattern = _pattern;\n pattern.needsUpdate = true;\n pattern.featureUpdaters.set(feature._dynamicId, (particle, iterator, index) => {\n if (feature.enabled) {\n const particleIndex = particle.idx;\n if (particleIndex >= pattern.size) {\n particle.isVisible = false;\n return;\n }\n if (!particle.isVisible) {\n particle.isVisible = true;\n }\n const featureItemIndex = index !== undefined ? index : particleIndex;\n const i = featureItemIndex % feature.count;\n if (i > 0 && setParticlePositionFunc) {\n setParticlePositionFunc(i, particle);\n }\n const nextUpdateFunction = iterator.next().value;\n if (nextUpdateFunction) {\n nextUpdateFunction(particle, iterator, Math.floor(featureItemIndex / feature.count));\n }\n }\n });\n pattern.size = Array.from(pattern.featureUpdaters.keys()).reduce((result, featureId) => {\n const feature = meshNode.features.find(feature => feature._dynamicId === featureId);\n return result * feature.count;\n }, 1);\n if (!pattern.needsRebuild &&\n (vertexDataChanges.indices || vertexDataChanges.positions || vertexDataChanges.uvs)) {\n /**\n * If the shape of the mesh changed, then we have to remove and re-add every particle to recompute the geometry.\n * Potential optimization here is to only recompute vertex positions if the vertex count is the same,\n * and only do the following expensive operation if the vertex count (indices?) changes.\n */\n const currentSize = pattern.sps.nbParticles;\n pattern.sps.addShape(pattern.shape, Math.max(pattern.size, pattern.sps.nbParticles));\n pattern.sps.removeParticles(0, currentSize - 1);\n pattern.needsRebuild = true;\n }\n else if (pattern.sps.nbParticles < pattern.size) {\n // If we don't have enough particles to complete the pattern, expand the SPS as needed\n pattern.sps.addShape(pattern.shape, pattern.size - pattern.sps.nbParticles);\n pattern.needsRebuild = true;\n }\n if (vertexDataChanges.submeshData) {\n pattern.needsSubmeshRebuild = true;\n }\n /**\n * We are removing the pattern feature combination system because it was causing a lot of issues around the cache, with enabling/disabling and reordering features.\n * In the future we will just add multi-direction properties into the pattern feature instead.\n */\n // const nextFeature = meshNode.features[thisIndex + 1];\n // if (!nextFeature || !(nextFeature instanceof PatternFeature)) {\n this.processParticleSystem(pattern, newVertexData); // mutates newVertexData\n patternStore.delete(meshNode._dynamicId); // Delete the pattern since we are done processing it\n // }\n }\n return newVertexData;\n }\n processParticleSystem(pattern, resultVertexData) {\n if (!pattern.needsUpdate) {\n return;\n }\n // Shrink the particle system if there are way too many particles (more than 50% too many)\n // We don't want to shrink unless necessary beca use there's a performance hit to rebuilding the mesh\n if (pattern.sps.nbParticles - pattern.size > pattern.size / 2) {\n pattern.sps.removeParticles(pattern.size, pattern.sps.nbParticles - 1);\n pattern.needsRebuild = true;\n }\n pattern.sps.initParticles();\n if (!pattern.resultMesh || pattern.needsRebuild) {\n pattern.resultMesh = pattern.sps.buildMesh();\n pattern.needsRebuild = false;\n pattern.needsSubmeshRebuild = true;\n pattern.resultMesh.setEnabled(false);\n }\n pattern.sps.computeBoundingBox = true;\n pattern.sps.setParticles();\n const totalIndicesSourceCount = resultVertexData.indices.length;\n const totalVerticesSourceCount = resultVertexData.positions.length / 3;\n const newVertexData = VertexData.ExtractFromMesh(pattern.resultMesh);\n resultVertexData.indices = newVertexData.indices;\n resultVertexData.positions = newVertexData.positions;\n if (newVertexData.normals) {\n resultVertexData.normals = newVertexData.normals;\n }\n if (resultVertexData.uvs && newVertexData.uvs) {\n // Don't add UVs if UVs never existed\n resultVertexData.uvs = newVertexData.uvs;\n }\n // If the source mesh has submeshes, then we need to update them\n if (resultVertexData.submeshData) {\n const submeshesToClone = resultVertexData.submeshData.slice(1);\n const totalIndicesResultCount = resultVertexData.indices.length;\n const totalVerticesResultCount = resultVertexData.positions.length / 3;\n // resultMeshCsg.dispose();\n const baseMaterialIndex = resultVertexData.submeshData[0].materialIndex;\n resultVertexData.submeshData = [];\n resultVertexData.submeshData.push({\n materialIndex: baseMaterialIndex,\n verticesStart: 0,\n verticesCount: totalVerticesResultCount,\n indexStart: 0,\n indexCount: totalIndicesResultCount,\n });\n const spsPatternSize = totalVerticesResultCount / totalVerticesSourceCount;\n if (submeshesToClone.length > 0) {\n submeshesToClone.forEach(subMesh => {\n for (let i = 0; i < spsPatternSize; i++) {\n resultVertexData.submeshData.push({\n materialIndex: subMesh.materialIndex,\n verticesStart: subMesh.verticesStart + i * totalVerticesSourceCount,\n verticesCount: subMesh.verticesCount,\n indexStart: subMesh.indexStart + i * totalIndicesSourceCount,\n indexCount: subMesh.indexCount,\n });\n }\n });\n }\n }\n }\n getPositionArray(node, distance, addEndMesh) {\n const result = [];\n const sketchLength = node.getSketchLength();\n const numberOfMeshes = sketchLength / distance;\n const percentagePoints = [];\n for (let i = 0; i < numberOfMeshes; i++) {\n percentagePoints.push((1 / numberOfMeshes) * i);\n }\n percentagePoints.forEach(point => {\n result.push(this.getIntervalPosition(node, point));\n });\n if (addEndMesh) {\n const lastPath = node.paths[node.paths.length - 1];\n const lastPoint = lastPath._points[lastPath._points.length - 1];\n const lastPointVec3 = new Vector3(lastPoint.x, lastPoint.y, lastPoint.z);\n result.push(lastPointVec3);\n }\n return result;\n }\n getIntervalPosition(node, posZeroToOne) {\n const result = new Vector3(0, 0, 0);\n const sketchLength = node.getSketchLength();\n let trackedLength = 0;\n let targetNotFound = true;\n node.paths.forEach(path => {\n let lastPosition = null;\n let lastLength = 0;\n path._points.forEach(point => {\n if (!lastPosition) {\n lastPosition = point;\n }\n else {\n const currentPos = point;\n const distance = Math.sqrt(Math.pow(lastPosition.x - currentPos.x, 2) +\n Math.pow(lastPosition.y - currentPos.y, 2) +\n Math.pow(lastPosition.z - currentPos.z, 2));\n lastLength = trackedLength;\n trackedLength += distance;\n const distanceBetweenPoints = trackedLength - lastLength;\n const distanceFromP1toTarget = posZeroToOne * sketchLength - lastLength;\n const percentageBetween2Points = distanceFromP1toTarget / distanceBetweenPoints;\n const vectorBetween2Points = new exports.KbVector(currentPos.x - lastPosition.x, currentPos.y - lastPosition.y, currentPos.z - lastPosition.z);\n if (trackedLength > posZeroToOne * sketchLength && targetNotFound) {\n const calculatedPosition = vectorBetween2Points\n .scale(percentageBetween2Points)\n .add(lastPosition);\n result.set(calculatedPosition.x, calculatedPosition.y, calculatedPosition.z);\n targetNotFound = false;\n }\n lastPosition = currentPos;\n }\n });\n if (posZeroToOne === 1) {\n const path = node.paths[node.paths.length - 1];\n const controlPoint = path._points[path._points.length - 1];\n result.set(controlPoint.x, controlPoint.y, controlPoint.z);\n }\n });\n return result;\n }\n delete3d(feature, preserveCache) {\n patternStore.forEach((pattern, meshNode) => {\n if (pattern.featureUpdaters.has(feature._dynamicId)) {\n pattern.featureUpdaters.delete(feature._dynamicId);\n }\n if (pattern.featureUpdaters.size === 0) {\n patternStore.delete(meshNode);\n }\n });\n super.delete3d(feature, preserveCache);\n }\n dispose() {\n super.dispose();\n patternStore.clear();\n }\n }\n\n const uvHelper = new UVMapHelper();\n const multiMaterialUseCount = new Map();\n const featureMappings = new Map();\n class SubmeshRenderer extends FeatureRenderer {\n constructor(viewer, scene, rendererManager) {\n super(viewer, scene);\n this.rendererManager = rendererManager;\n this.modelToken = Token.SubmeshFeature;\n }\n updateMeshGeometry(feature, vertexData, vertexDataChanges) {\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\n const submeshFeatureMapping = featureMappings.get(feature._dynamicId);\n if (!submeshFeatureMapping) {\n console.warn(`INFO: Submesh mapping not found for ${feature.name}`);\n return newVertexData;\n }\n if (!newVertexData.submeshData) {\n newVertexData.submeshData = [];\n }\n if (submeshFeatureMapping.multiMaterialIndex) {\n const verticesCount = (vertexDataChanges.positions || newVertexData.positions).length / 3;\n const indicesCount = (vertexDataChanges.indices || newVertexData.indices).length;\n for (let i = 0; i < feature.vertexGroups.length; i++) {\n const vertexGroup = feature.vertexGroups[i];\n const newSubmesh = {\n materialIndex: submeshFeatureMapping.multiMaterialIndex,\n verticesStart: 0,\n verticesCount: verticesCount,\n indexStart: vertexGroup.startIndex,\n indexCount: Math.min(vertexGroup.endIndex, indicesCount) - vertexGroup.startIndex,\n };\n const existingIndex = newVertexData.submeshData.findIndex(existingSubmesh => existingSubmesh.materialIndex === newSubmesh.materialIndex &&\n existingSubmesh.verticesStart === newSubmesh.verticesStart &&\n existingSubmesh.verticesCount === newSubmesh.verticesCount &&\n existingSubmesh.indexStart === newSubmesh.indexStart &&\n existingSubmesh.indexCount === newSubmesh.indexCount);\n if (existingIndex >= 0) {\n newVertexData.submeshData.splice(existingIndex, 1);\n }\n newVertexData.submeshData.push(newSubmesh);\n }\n }\n // Calculate UV maps if that is enabled on this feature\n if (feature.uvMapping) {\n const uvMapVertexes = [];\n for (let i = 0; i < feature.vertexGroups.length; i++) {\n const vertexGroup = feature.vertexGroups[i];\n for (let i = vertexGroup.startIndex; i < vertexGroup.endIndex; i++) {\n uvMapVertexes.push(i);\n }\n }\n if (uvMapVertexes.length > 0) {\n let existingUvs = [];\n if (newVertexData.uvs) {\n existingUvs = [...newVertexData.uvs];\n }\n uvHelper.generateUVMapInPlace(newVertexData, feature, uvMapVertexes);\n // for projection mode \"use existing\", only apply the new uvs to the selected faces of the submesh\n if (feature.mode === exports.eUvMode.useExisting) {\n feature.vertexGroups.forEach(vertexGroup => {\n for (let i = vertexGroup.startIndex; i < vertexGroup.endIndex; i++) {\n if (newVertexData.indices && newVertexData.uvs) {\n existingUvs[newVertexData.indices[i] * 2] =\n newVertexData.uvs[newVertexData.indices[i] * 2];\n existingUvs[newVertexData.indices[i] * 2 + 1] =\n newVertexData.uvs[newVertexData.indices[i] * 2 + 1];\n }\n }\n });\n newVertexData.uvs = existingUvs;\n }\n }\n }\n return newVertexData;\n }\n update3dInternal(feature) {\n let _submeshFeatureMapping = featureMappings.get(feature._dynamicId);\n if (feature.changes.has('materialId') || !_submeshFeatureMapping) {\n const meshNode = feature._parent;\n const mesh = this.viewer.getRendererNode(meshNode);\n if (!mesh) {\n return;\n }\n if (!_submeshFeatureMapping) {\n _submeshFeatureMapping = {};\n featureMappings.set(feature._dynamicId, _submeshFeatureMapping);\n }\n const submeshFeatureMapping = _submeshFeatureMapping;\n if (feature.materialId) {\n if (!(mesh.material instanceof MultiMaterial)) {\n const multiMaterial = new MultiMaterial(meshNode._dynamicId, this.scene);\n if (mesh.material) {\n multiMaterial.subMaterials[0] = mesh.material;\n }\n else if (!mesh.material) {\n // If a submesh multimaterial has a null material at index 0, it causes shadows to not work\n multiMaterial.subMaterials[0] = this.rendererManager.material.defaultMaterial;\n }\n submeshFeatureMapping.multiMaterial = multiMaterial;\n multiMaterialUseCount.set(multiMaterial, 0);\n mesh.material = multiMaterial;\n }\n else {\n submeshFeatureMapping.multiMaterial = mesh.material;\n }\n this.rendererManager.mesh.fetchMaterialNode(meshNode, feature.materialId).then(materialNode => {\n // The material may have changed on the mesh, so we have to re-fetch the reference just in case.\n if (!(mesh.material instanceof MultiMaterial)) {\n throw Error('Another script has removed the multimaterial on a mesh');\n }\n const multiMaterial = mesh.material;\n if (multiMaterial !== submeshFeatureMapping.multiMaterial) {\n console.warn(`Submesh feature ${feature.name} lost reference to a multimaterial. Possible memory leak?`);\n submeshFeatureMapping.multiMaterial = multiMaterial;\n }\n // If the material is already part of the submaterials, do nothing\n const existingSubmaterialIndex = multiMaterial.subMaterials.findIndex(material => (material === null || material === void 0 ? void 0 : material.id) === materialNode._dynamicId);\n if ((submeshFeatureMapping === null || submeshFeatureMapping === void 0 ? void 0 : submeshFeatureMapping.multiMaterialIndex) === existingSubmaterialIndex) {\n return;\n }\n else if (existingSubmaterialIndex > -1) {\n submeshFeatureMapping.multiMaterialIndex = existingSubmaterialIndex;\n // Triggers change on this feature so the feature stack refreshes with the material index on this submesh\n feature.setPropertyIsModified('vertexGroups', true);\n }\n else {\n // init the material if the node is showing\n if (meshNode.visible) {\n materialNode.init();\n }\n const material = this.viewer.getRendererNode(materialNode);\n if (!material) {\n throw Error('Material could not be found in renderer. Has the material been created?');\n }\n submeshFeatureMapping.multiMaterialIndex = multiMaterial.subMaterials.length;\n multiMaterial.subMaterials.push(material);\n multiMaterialUseCount.set(multiMaterial, (multiMaterialUseCount.get(multiMaterial) || 0) + 1);\n // Triggers change on this feature so the feature stack refreshes with the material index on this submesh\n feature.setPropertyIsModified('vertexGroups', true);\n }\n }, err => {\n feature.materialId = '';\n });\n }\n }\n }\n delete3d(feature, preserveCache) {\n if (!preserveCache) {\n const submeshFeatureMapping = featureMappings.get(feature._dynamicId);\n if (submeshFeatureMapping && submeshFeatureMapping.multiMaterial) {\n const multiMaterial = submeshFeatureMapping.multiMaterial;\n const count = multiMaterialUseCount.get(multiMaterial);\n if (count !== undefined) {\n if (count <= 1) {\n const meshNode = feature._parent;\n const mesh = this.viewer.getRendererNode(meshNode);\n if (mesh) {\n mesh.material = multiMaterial.getSubMaterial(0);\n }\n multiMaterialUseCount.delete(multiMaterial);\n multiMaterial.dispose(true, true, false);\n }\n else {\n multiMaterialUseCount.set(multiMaterial, count - 1);\n }\n }\n }\n featureMappings.delete(feature._dynamicId);\n }\n super.delete3d(feature, preserveCache);\n }\n }\n\n const uvHelper$1 = new UVMapHelper();\n class UvMapRenderer extends FeatureRenderer {\n constructor() {\n super(...arguments);\n this.modelToken = Token.UvMapFeature;\n this.babylonUvPreviewsMap = new Map();\n this.previewLayer = this.viewer.utilityLayer;\n }\n updateMeshGeometry(feature, vertexData, vertexDataChanges) {\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\n uvHelper$1.generateUVMapInPlace(newVertexData, feature);\n return newVertexData;\n }\n updatePreviewMesh(feature) {\n let visualizationMesh = this.babylonUvPreviewsMap.get(feature.id);\n const scene = this.previewLayer.utilityLayerScene;\n if (!visualizationMesh) {\n visualizationMesh = createUvMapVisualization(feature, scene, Color3.Yellow());\n visualizationMesh.isPickable = false;\n this.babylonUvPreviewsMap.set(feature.id, visualizationMesh);\n }\n else {\n this.deletePreview3d(feature);\n visualizationMesh = createUvMapVisualization(feature, scene, Color3.Yellow());\n visualizationMesh.isPickable = false;\n this.babylonUvPreviewsMap.set(feature.id, visualizationMesh);\n }\n let meshPosition = visualizationMesh.getAbsolutePosition();\n if (feature._parent) {\n const featureParent = this.viewer.getRendererNode(feature._parent);\n meshPosition = featureParent.getAbsolutePosition();\n }\n meshPosition.add(feature.center.toVec3());\n meshPosition.addInPlace(feature.center.toVec3());\n visualizationMesh.position.set(meshPosition.x, meshPosition.y, meshPosition.z);\n //visualizationMesh.project;\n const newNormal = new Vector3();\n newNormal.set(feature.projectionAxis.x, feature.projectionAxis.y, feature.projectionAxis.z);\n visualizationMesh.rotationQuaternion = Quaternion.RotationAxis(newNormal, (feature.angle / 180) * Math.PI);\n visualizationMesh.rotationQuaternion = visualizationMesh.rotationQuaternion.multiply(rotationToVector(newNormal));\n visualizationMesh.scaling.set(feature.scale.x, feature.scale.y, feature.scale.z);\n }\n update3dInternal(feature) {\n const visualizationMesh = this.babylonUvPreviewsMap.get(feature.id);\n visualizationMesh === null || visualizationMesh === void 0 ? void 0 : visualizationMesh.setEnabled(feature._previewMaterialVisible);\n // change visibility here\n if (feature._previewMaterialVisible) {\n this.updatePreviewMesh(feature);\n }\n }\n delete3d(feature) {\n this.deletePreview3d(feature);\n super.delete3d(feature);\n }\n deletePreview3d(feature) {\n const previewMesh = this.babylonUvPreviewsMap.get(feature.id);\n if (previewMesh) {\n previewMesh.dispose();\n }\n this.babylonUvPreviewsMap.delete(feature.id);\n }\n }\n\n class LightRenderer extends Renderer {\n set showPreviews(show) {\n this.showAllLightPreviews = show;\n this.babylonMap.forEach(light => {\n this.updatePreviewMesh(light);\n });\n }\n constructor(viewer, scene, shadowManager) {\n super(viewer, scene);\n this.shadowManager = shadowManager;\n this.babylonLightPreviewsMap = new Map();\n this.modelToken = Token.LightNode;\n this.showAllLightPreviews = false;\n this.gizmoLayer = this.viewer.utilityLayer;\n }\n update3dInternal(node) {\n var _a;\n if (!node._parent && ((_a = node._parentScene) === null || _a === void 0 ? void 0 : _a.isNested())) {\n //don't render global lights in nested scenes\n return;\n }\n let light = this.get3d(node);\n const changes = this.getChanges(node);\n if (!light) {\n if (node instanceof exports.PointLightNode) {\n light = new PointLight(node._dynamicId, node.position.toVec3(), this.scene);\n }\n else if (node instanceof exports.HemisphericLightNode) {\n light = new HemisphericLight(node._dynamicId, node.direction.toVec3(), this.scene);\n }\n else if (node instanceof exports.SpotLightNode) {\n light = new SpotLight(node._dynamicId, node.position.toVec3(), node.direction.toVec3(), (node.coneAngle * Math.PI) / 180, 2, this.scene);\n light.exponent = 2;\n }\n else if (node instanceof exports.DirectionalLightNode) {\n light = new DirectionalLight(node._dynamicId, node.direction.toVec3(), this.scene);\n }\n this.set3d(node, light);\n }\n const pLight = light;\n if (changes.has('visible')) {\n // Can be falsy so need to check the map\n pLight.setEnabled(node.visible);\n }\n if (changes.has('intensity')) {\n // Can be falsy so need to check the map\n pLight.intensity = node.intensity;\n }\n if (changes.has('color')) {\n pLight.specular = node.color.toColor3();\n pLight.diffuse = node.color.toColor3();\n }\n if (changes.has('_parent')) {\n if (node._parent) {\n pLight.parent = this.viewer.getRendererNode(node._parent);\n }\n else {\n pLight.parent = null;\n }\n }\n let lightRotation;\n if (node.useEuler) {\n lightRotation = Quaternion.FromEulerAngles(toRadians(node.eulerRotation.x), toRadians(node.eulerRotation.y), toRadians(node.eulerRotation.z));\n }\n else {\n lightRotation = Quaternion.RotationAxis(node.rotationAxis.toVec3(), toRadians(node.rotationAngle));\n }\n const rotationChanged = changes.hasAny('useEuler', 'eulerRotation', 'rotationAxis', 'rotationAngle');\n if (node instanceof exports.PointLightNode) {\n const pLight = light;\n if (changes.has('position')) {\n pLight.position = node.position.toVec3();\n }\n }\n else if (node instanceof exports.HemisphericLightNode) {\n const pLight = light;\n if (changes.has('direction') || rotationChanged) {\n pLight.direction = this.rotateDirection(node.direction.toVec3(), lightRotation);\n }\n if (changes.has('groundColor')) {\n pLight.groundColor = node.groundColor.toColor3();\n }\n }\n else if (node instanceof exports.SpotLightNode) {\n const pLight = light;\n if (changes.has('position')) {\n pLight.position = node.position.toVec3();\n }\n if (changes.has('direction') || rotationChanged) {\n pLight.direction = this.rotateDirection(node.direction.toVec3(), lightRotation);\n }\n if (changes.has('coneAngle')) {\n pLight.angle = (node.coneAngle * Math.PI) / 180;\n }\n }\n else if (node instanceof exports.DirectionalLightNode) {\n const pLight = light;\n // pLight.position = new Vector3(0, 5, 0);\n if (changes.has('direction') || rotationChanged) {\n pLight.direction = this.rotateDirection(node.direction.toVec3(), lightRotation);\n //for directional lights, we hardcode the position to be negative direction for shadows\n pLight.position = pLight.direction.clone().negate();\n }\n }\n if (node instanceof ShadowLightNode) {\n const shadowLightNode = node;\n const shadowLight = light;\n let sg = this.shadowManager.get(shadowLight);\n let justCreated = false;\n if (changes.has('generatesShadows')) {\n if (shadowLightNode.generatesShadows) {\n sg = this.shadowManager.add(shadowLight, shadowLightNode.shadowResolution);\n shadowLight.shadowEnabled = true;\n justCreated = true;\n changes.set('shadowFilter', shadowLightNode.shadowFilter);\n changes.set('shadowQuality', shadowLightNode.shadowQuality);\n }\n else {\n if (sg) {\n shadowLight.shadowEnabled = false;\n this.shadowManager.remove(shadowLight);\n sg = undefined;\n }\n }\n }\n //if the resolution is changed, and the shadow generator wasn't just created, we need to recreate it\n if (changes.has('shadowResolution') && sg && !justCreated) {\n this.shadowManager.remove(shadowLight);\n sg = this.shadowManager.add(shadowLight, shadowLightNode.shadowResolution);\n changes.set('shadowFilter', shadowLightNode.shadowFilter);\n changes.set('shadowQuality', shadowLightNode.shadowQuality);\n changes.set('processTransparency', shadowLightNode.processTransparency);\n }\n if (sg) {\n if (changes.has('shadowFilter')) {\n sg.filter = ShadowManager.convertShadowFilter(shadowLightNode.shadowFilter);\n }\n if (changes.has('processTransparency')) {\n sg.transparencyShadow = shadowLightNode.processTransparency;\n sg.enableSoftTransparentShadow = shadowLightNode.processTransparency;\n }\n if (changes.has('shadowQuality')) {\n sg.filteringQuality = ShadowGenerator[`QUALITY_${shadowLightNode.shadowQuality.toUpperCase()}`];\n }\n //need to update projection matrix if direction has been changed\n if (changes.hasAny('direction', 'position')) {\n this.shadowManager.clearProjectMatrixCache();\n }\n }\n }\n if (changes.has('_selected')) {\n if (!pLight.metadata) {\n pLight.metadata = {};\n }\n pLight.metadata.selected = node._selected;\n }\n this.viewer.updateLightWarnings();\n this.updatePreviewMesh(pLight);\n }\n delete3d(node) {\n if (this.babylonLightPreviewsMap) {\n const lightPositionGizmo = this.babylonLightPreviewsMap.get(node._dynamicId);\n if (!lightPositionGizmo) {\n throw Error('Light gizmo not found for light ' + node.id);\n }\n lightPositionGizmo.dispose();\n this.babylonLightPreviewsMap.delete(node._dynamicId);\n }\n this.shadowManager.remove(this.get3d(node)); //remove also disposes the shadow generator if it exists\n super.delete3d(node);\n this.viewer.updateLightWarnings();\n }\n updatePreviewMesh(light) {\n let lightPositionGizmo = this.babylonLightPreviewsMap.get(light.id);\n const scene = this.gizmoLayer.utilityLayerScene;\n if (!lightPositionGizmo) {\n lightPositionGizmo = createLightVisualization(light, scene, Color3.Yellow());\n lightPositionGizmo.isPickable = false;\n this.babylonLightPreviewsMap.set(light.id, lightPositionGizmo);\n }\n if (this.showAllLightPreviews) {\n lightPositionGizmo.setEnabled(light.metadata.selected || light.isEnabled());\n }\n else {\n lightPositionGizmo.setEnabled(light.metadata.selected);\n }\n if (light instanceof HemisphericLight) {\n lightPositionGizmo.rotationQuaternion = rotationToVector(light.direction);\n }\n else if (light instanceof SpotLight) {\n lightPositionGizmo.position = this.getLightPosition(light);\n lightPositionGizmo.rotationQuaternion = rotationToVector(light.direction);\n lightPositionGizmo.scaling.set(Math.sin(light.angle / 2), Math.cos(light.angle / 2), Math.sin(light.angle / 2));\n }\n else if (light instanceof DirectionalLight) {\n lightPositionGizmo.rotationQuaternion = rotationToVector(light.direction);\n }\n else if (light instanceof PointLight) {\n lightPositionGizmo.position = this.getLightPosition(light);\n }\n }\n rotateDirection(direction, rotation) {\n direction.rotateByQuaternionToRef(rotation, direction);\n return direction;\n }\n getLightPosition(light) {\n return light.parent\n ? Vector3.TransformCoordinates(light.position, light.parent.computeWorldMatrix(true).clone())\n : light.position;\n }\n }\n\n class MateRenderer extends Renderer {\n constructor() {\n super(...arguments);\n this.modelToken = Token.MateNode;\n }\n update3dInternal(node) { }\n delete3d(node) {\n SyncService.nodeDeletionProcessed(node);\n }\n }\n\n AbstractScene.prototype.removeReflectionProbe = function (toRemove) {\n if (!this.reflectionProbes) {\n return -1;\n }\n const index = this.reflectionProbes.indexOf(toRemove);\n if (index !== -1) {\n this.reflectionProbes.splice(index, 1);\n }\n return index;\n };\n AbstractScene.prototype.addReflectionProbe = function (newReflectionProbe) {\n if (!this.reflectionProbes) {\n this.reflectionProbes = [];\n }\n this.reflectionProbes.push(newReflectionProbe);\n };\n /**\n * Class used to generate realtime reflection / refraction cube textures\n * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/reflectionProbes\n */\n class ReflectionProbe {\n /**\n * Creates a new reflection probe\n * @param name defines the name of the probe\n * @param size defines the texture resolution (for each face)\n * @param scene defines the hosting scene\n * @param generateMipMaps defines if mip maps should be generated automatically (true by default)\n * @param useFloat defines if HDR data (float data) should be used to store colors (false by default)\n * @param linearSpace defines if the probe should be generated in linear space or not (false by default)\n */\n constructor(\n /** defines the name of the probe */\n name, size, scene, generateMipMaps = true, useFloat = false, linearSpace = false) {\n this.name = name;\n this._viewMatrix = Matrix.Identity();\n this._target = Vector3.Zero();\n this._add = Vector3.Zero();\n this._invertYAxis = false;\n /** Gets or sets probe position (center of the cube map) */\n this.position = Vector3.Zero();\n /**\n * Gets or sets an object used to store user defined information for the reflection probe.\n */\n this.metadata = null;\n /** @internal */\n this._parentContainer = null;\n this._scene = scene;\n if (scene.getEngine().supportsUniformBuffers) {\n this._sceneUBOs = [];\n for (let i = 0; i < 6; ++i) {\n this._sceneUBOs.push(scene.createSceneUniformBuffer(`Scene for Reflection Probe (name \"${name}\") face #${i}`));\n }\n }\n // Create the scene field if not exist.\n if (!this._scene.reflectionProbes) {\n this._scene.reflectionProbes = new Array();\n }\n this._scene.reflectionProbes.push(this);\n let textureType = 0;\n if (useFloat) {\n const caps = this._scene.getEngine().getCaps();\n if (caps.textureHalfFloatRender) {\n textureType = 2;\n }\n else if (caps.textureFloatRender) {\n textureType = 1;\n }\n }\n this._renderTargetTexture = new RenderTargetTexture(name, size, scene, generateMipMaps, true, textureType, true);\n this._renderTargetTexture.gammaSpace = !linearSpace;\n this._renderTargetTexture.invertZ = scene.useRightHandedSystem;\n const useReverseDepthBuffer = scene.getEngine().useReverseDepthBuffer;\n this._renderTargetTexture.onBeforeRenderObservable.add((faceIndex) => {\n if (this._sceneUBOs) {\n scene.setSceneUniformBuffer(this._sceneUBOs[faceIndex]);\n scene.getSceneUniformBuffer().unbindEffect();\n }\n switch (faceIndex) {\n case 0:\n this._add.copyFromFloats(1, 0, 0);\n break;\n case 1:\n this._add.copyFromFloats(-1, 0, 0);\n break;\n case 2:\n this._add.copyFromFloats(0, this._invertYAxis ? 1 : -1, 0);\n break;\n case 3:\n this._add.copyFromFloats(0, this._invertYAxis ? -1 : 1, 0);\n break;\n case 4:\n this._add.copyFromFloats(0, 0, scene.useRightHandedSystem ? -1 : 1);\n break;\n case 5:\n this._add.copyFromFloats(0, 0, scene.useRightHandedSystem ? 1 : -1);\n break;\n }\n if (this._attachedMesh) {\n this.position.copyFrom(this._attachedMesh.getAbsolutePosition());\n }\n this.position.addToRef(this._add, this._target);\n const lookAtFunction = scene.useRightHandedSystem ? Matrix.LookAtRHToRef : Matrix.LookAtLHToRef;\n const perspectiveFunction = scene.useRightHandedSystem ? Matrix.PerspectiveFovRH : Matrix.PerspectiveFovLH;\n lookAtFunction(this.position, this._target, Vector3.Up(), this._viewMatrix);\n if (scene.activeCamera) {\n this._projectionMatrix = perspectiveFunction(Math.PI / 2, 1, useReverseDepthBuffer ? scene.activeCamera.maxZ : scene.activeCamera.minZ, useReverseDepthBuffer ? scene.activeCamera.minZ : scene.activeCamera.maxZ, this._scene.getEngine().isNDCHalfZRange);\n scene.setTransformMatrix(this._viewMatrix, this._projectionMatrix);\n if (scene.activeCamera.isRigCamera && !this._renderTargetTexture.activeCamera) {\n this._renderTargetTexture.activeCamera = scene.activeCamera.rigParent || null;\n }\n }\n scene._forcedViewPosition = this.position;\n });\n let currentApplyByPostProcess;\n this._renderTargetTexture.onBeforeBindObservable.add(() => {\n var _a, _b;\n this._currentSceneUBO = scene.getSceneUniformBuffer();\n (_b = (_a = scene.getEngine())._debugPushGroup) === null || _b === void 0 ? void 0 : _b.call(_a, `reflection probe generation for ${name}`, 1);\n currentApplyByPostProcess = this._scene.imageProcessingConfiguration.applyByPostProcess;\n if (linearSpace) {\n scene.imageProcessingConfiguration.applyByPostProcess = true;\n }\n });\n this._renderTargetTexture.onAfterUnbindObservable.add(() => {\n var _a, _b;\n scene.imageProcessingConfiguration.applyByPostProcess = currentApplyByPostProcess;\n scene._forcedViewPosition = null;\n if (this._sceneUBOs) {\n scene.setSceneUniformBuffer(this._currentSceneUBO);\n }\n scene.updateTransformMatrix(true);\n (_b = (_a = scene.getEngine())._debugPopGroup) === null || _b === void 0 ? void 0 : _b.call(_a, 1);\n });\n }\n /** Gets or sets the number of samples to use for multi-sampling (0 by default). Required WebGL2 */\n get samples() {\n return this._renderTargetTexture.samples;\n }\n set samples(value) {\n this._renderTargetTexture.samples = value;\n }\n /** Gets or sets the refresh rate to use (on every frame by default) */\n get refreshRate() {\n return this._renderTargetTexture.refreshRate;\n }\n set refreshRate(value) {\n this._renderTargetTexture.refreshRate = value;\n }\n /**\n * Gets the hosting scene\n * @returns a Scene\n */\n getScene() {\n return this._scene;\n }\n /** Gets the internal CubeTexture used to render to */\n get cubeTexture() {\n return this._renderTargetTexture;\n }\n /** Gets the list of meshes to render */\n get renderList() {\n return this._renderTargetTexture.renderList;\n }\n /**\n * Attach the probe to a specific mesh (Rendering will be done from attached mesh's position)\n * @param mesh defines the mesh to attach to\n */\n attachToMesh(mesh) {\n this._attachedMesh = mesh;\n }\n /**\n * Specifies whether or not the stencil and depth buffer are cleared between two rendering groups\n * @param renderingGroupId The rendering group id corresponding to its index\n * @param autoClearDepthStencil Automatically clears depth and stencil between groups if true.\n */\n setRenderingAutoClearDepthStencil(renderingGroupId, autoClearDepthStencil) {\n this._renderTargetTexture.setRenderingAutoClearDepthStencil(renderingGroupId, autoClearDepthStencil);\n }\n /**\n * Clean all associated resources\n */\n dispose() {\n const index = this._scene.reflectionProbes.indexOf(this);\n if (index !== -1) {\n // Remove from the scene if found\n this._scene.reflectionProbes.splice(index, 1);\n }\n if (this._parentContainer) {\n const index = this._parentContainer.reflectionProbes.indexOf(this);\n if (index > -1) {\n this._parentContainer.reflectionProbes.splice(index, 1);\n }\n this._parentContainer = null;\n }\n if (this._renderTargetTexture) {\n this._renderTargetTexture.dispose();\n this._renderTargetTexture = null;\n }\n if (this._sceneUBOs) {\n for (const ubo of this._sceneUBOs) {\n ubo.dispose();\n }\n this._sceneUBOs = [];\n }\n }\n /**\n * Converts the reflection probe information to a readable string for debug purpose.\n * @param fullDetails Supports for multiple levels of logging within scene loading\n * @returns the human readable reflection probe info\n */\n toString(fullDetails) {\n let ret = \"Name: \" + this.name;\n if (fullDetails) {\n ret += \", position: \" + this.position.toString();\n if (this._attachedMesh) {\n ret += \", attached mesh: \" + this._attachedMesh.name;\n }\n }\n return ret;\n }\n /**\n * Get the class name of the refection probe.\n * @returns \"ReflectionProbe\"\n */\n getClassName() {\n return \"ReflectionProbe\";\n }\n /**\n * Serialize the reflection probe to a JSON representation we can easily use in the respective Parse function.\n * @returns The JSON representation of the texture\n */\n serialize() {\n const serializationObject = SerializationHelper.Serialize(this, this._renderTargetTexture.serialize());\n serializationObject.isReflectionProbe = true;\n serializationObject.metadata = this.metadata;\n return serializationObject;\n }\n /**\n * Parse the JSON representation of a reflection probe in order to recreate the reflection probe in the given scene.\n * @param parsedReflectionProbe Define the JSON representation of the reflection probe\n * @param scene Define the scene the parsed reflection probe should be instantiated in\n * @param rootUrl Define the root url of the parsing sequence in the case of relative dependencies\n * @returns The parsed reflection probe if successful\n */\n static Parse(parsedReflectionProbe, scene, rootUrl) {\n let reflectionProbe = null;\n if (scene.reflectionProbes) {\n for (let index = 0; index < scene.reflectionProbes.length; index++) {\n const rp = scene.reflectionProbes[index];\n if (rp.name === parsedReflectionProbe.name) {\n reflectionProbe = rp;\n break;\n }\n }\n }\n reflectionProbe = SerializationHelper.Parse(() => reflectionProbe || new ReflectionProbe(parsedReflectionProbe.name, parsedReflectionProbe.renderTargetSize, scene, parsedReflectionProbe._generateMipMaps), parsedReflectionProbe, scene, rootUrl);\n reflectionProbe.cubeTexture._waitingRenderList = parsedReflectionProbe.renderList;\n if (parsedReflectionProbe._attachedMesh) {\n reflectionProbe.attachToMesh(scene.getMeshById(parsedReflectionProbe._attachedMesh));\n }\n if (parsedReflectionProbe.metadata) {\n reflectionProbe.metadata = parsedReflectionProbe.metadata;\n }\n return reflectionProbe;\n }\n }\n __decorate$1([\n serializeAsMeshReference()\n ], ReflectionProbe.prototype, \"_attachedMesh\", void 0);\n __decorate$1([\n serializeAsVector3()\n ], ReflectionProbe.prototype, \"position\", void 0);\n\n // Performance Code from Babylon 5.0. Remove once upgrade to Babylon 5.0 is merged\n Object.defineProperty(BaseTexture.prototype, 'hasAlpha', {\n get: function () {\n return this._hasAlpha;\n },\n set: function (value) {\n const self = this;\n if (this._hasAlpha === value) {\n return;\n }\n this._hasAlpha = value;\n if (this._scene) {\n this._scene.markAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag, function (mat) {\n return mat.hasTexture(self);\n });\n }\n },\n enumerable: false,\n configurable: true,\n });\n class MaterialRenderer extends Renderer {\n get defaultMaterial() {\n return this._defaultMaterial;\n }\n constructor(viewer, scene, textureRenderer) {\n super(viewer, scene);\n this.textureRenderer = textureRenderer;\n this._globalMaterialCache = new Map();\n this.orphanedMeshes = new Map();\n //keyed by the connector id of which the reflection probe is attached to\n this.reflectionProbeMap = new Map();\n this.reflectedMeshes = new Map();\n //key is the ConnectorId of the reflection probe, String[] is the ids of the meshes using the material;\n this.meshesUsingReflectionMaterial = new Map();\n this.modelToken = Token.MaterialNode;\n this._defaultMaterial = new PBRMaterial('default material', scene);\n this._defaultMaterial.albedoColor = new Color3(0.5, 0.5, 0.5);\n this._defaultMaterial.alphaMode = Engine.ALPHA_COMBINE;\n this._defaultMaterial.metallic = 0;\n this._defaultMaterial.roughness = 1;\n }\n get renderMode() {\n return this._renderMode;\n }\n set renderMode(mode) {\n this._renderMode = mode;\n this.babylonMap.forEach(material => {\n this.setRenderModeOnMaterial(material, mode);\n });\n this.setRenderModeOnMaterial(this._defaultMaterial, mode);\n }\n getAllDefinedMaterials() {\n return Array.from(this.babylonMap.values());\n }\n getGlobalMaterial(dynamicId) {\n if (!this._globalMaterialManager) {\n return undefined;\n }\n else {\n return this._globalMaterialManager.getById(dynamicId);\n }\n }\n delete3d(node) {\n this.textureRenderer.materialDeleted(node);\n const material = this.get3d(node);\n if (material instanceof PBRMaterial) {\n const orphanedMeshes = material.getBindedMeshes();\n this.orphanedMeshes.set(node.id, orphanedMeshes);\n }\n this.viewer.materialWarningsNeedsUpdate.next();\n return super.delete3d(node);\n }\n loadGlobalMaterial(materialId, managerServices) {\n if (!this._globalMaterialManager) {\n this._globalMaterialManager = new Kb3dManager({ services: managerServices });\n }\n const matService = this._globalMaterialManager.getService(Token.MaterialService);\n if (this._globalMaterialCache.has(materialId)) {\n return this._globalMaterialCache.get(materialId);\n }\n else {\n const obs = matService\n .getMaterialById(materialId)\n .pipe(map(imaterial => {\n if (imaterial) {\n let matNode = this._globalMaterialManager.getById(materialId);\n const newMatNode = this._globalMaterialManager.deserialize(imaterial.data);\n // If the material has already been added to the scene, just update it.\n if (matNode) {\n this._globalMaterialManager.assign(matNode, newMatNode);\n // We have to re-initialize the material to reload the texture\n if (matNode._initialized) {\n matNode._initialized = false;\n matNode.init();\n }\n }\n else {\n matNode = newMatNode;\n matNode._dynamicId = matNode.id; // Each global material is unique\n this._globalMaterialManager.register(matNode);\n //now create the material in babylon by rendering it\n this.update3d(matNode);\n }\n return matNode;\n }\n return undefined;\n }))\n .pipe(shareReplay(1));\n this._globalMaterialCache.set(materialId, obs);\n return obs;\n }\n }\n update3dInternal(node) {\n let material = this.get3d(node);\n let _changes = node.changes;\n // since the texture are nested in the material we have to refresh the texture in the material individually\n if (instanceOf(node, Token.AdvancedMaterialNode)) {\n let changes = _changes;\n let pbrAdvMaterial = material;\n if (!pbrAdvMaterial) {\n material = pbrAdvMaterial = new PBRMaterial(node._dynamicId, this.scene);\n material.name = node.name;\n material.id = node._dynamicId;\n const meshesToBind = this.orphanedMeshes.get(node.id);\n if (meshesToBind) {\n const _pbrAdvMaterial = pbrAdvMaterial;\n meshesToBind.forEach(mesh => {\n mesh.material = _pbrAdvMaterial;\n });\n }\n changes = _changes = new Map(Object.entries(node));\n changes.set('textures', node.textures);\n /** hardcode the alphaMode because the material always overrides the engine setting and we need proper alpha for dragging effects */\n material.alphaMode = Engine.ALPHA_COMBINE;\n material.useParallax = false;\n // Logarithmic depth is a good concept but is buggy. Maybe try again later.\n // material.useLogarithmicDepth = true;\n material.useLightmapAsShadowmap = true;\n material.enableSpecularAntiAliasing = true;\n material.forceDepthWrite = true;\n pbrAdvMaterial.transparencyMode = Material.MATERIAL_ALPHABLEND;\n material.useMetallnessFromMetallicTextureBlue = false;\n material.useMicroSurfaceFromReflectivityMapAlpha = false;\n material.useRoughnessFromMetallicTextureGreen = false;\n this.setRenderModeOnMaterial(material, this.renderMode);\n this.set3d(node, pbrAdvMaterial);\n }\n if (changes.has('probePositionTargetId')) {\n this.updateReflectionProbes();\n }\n if (changes.hasAny('probePositionTargetId', 'subsurfaceReflectionEnabled', 'reflectionType')) {\n //if (node.reflectionType === eReflectionType.probe) {\n if (!node.probePositionTargetId || !node.subsurfaceReflectionEnabled) {\n if (pbrAdvMaterial) {\n pbrAdvMaterial.reflectionTexture = null;\n }\n }\n else {\n if (node.reflectionType === exports.eReflectionType.probe) {\n if (node.probePositionTargetId && pbrAdvMaterial) {\n let reflectionProbe = this.reflectionProbeMap.get(node.probePositionTargetId);\n if (!reflectionProbe || reflectionProbe instanceof MirrorTexture) {\n reflectionProbe = new ReflectionProbe(node.probePositionTargetId, 512, this.scene);\n this.reflectionProbeMap.set(node.probePositionTargetId, reflectionProbe);\n const exceptions = this.meshesUsingReflectionMaterial.get(node.probePositionTargetId);\n this.reflectedMeshes.forEach(mesh => {\n var _a;\n if (reflectionProbe && reflectionProbe.renderList) {\n if (!(exceptions === null || exceptions === void 0 ? void 0 : exceptions.contains(mesh.id))) {\n (_a = reflectionProbe.renderList) === null || _a === void 0 ? void 0 : _a.push(mesh);\n }\n }\n });\n const sceneBackground = this.scene.meshes.find(obj => {\n return obj.id === 'hdrSkyBox';\n });\n if (reflectionProbe && reflectionProbe.renderList && sceneBackground) {\n reflectionProbe.renderList.push(sceneBackground);\n }\n }\n const connector = this.viewer.getNode(node.probePositionTargetId);\n if (connector instanceof exports.Connector) {\n const connectorMesh = this.viewer.getRendererNode(connector);\n reflectionProbe.attachToMesh(connectorMesh);\n }\n if (pbrAdvMaterial) {\n reflectionProbe.cubeTexture.invertZ = false;\n pbrAdvMaterial.reflectionTexture = reflectionProbe.cubeTexture;\n }\n }\n }\n else {\n if (node.probePositionTargetId && pbrAdvMaterial) {\n let reflectionProbe = this.reflectionProbeMap.get(node.probePositionTargetId);\n let noPlane = false;\n if (reflectionProbe && reflectionProbe instanceof MirrorTexture) {\n if (!reflectionProbe.mirrorPlane) {\n noPlane = true;\n }\n }\n if (!reflectionProbe || reflectionProbe instanceof MirrorTexture || noPlane) {\n reflectionProbe = new ReflectionProbe(node.probePositionTargetId, 512, this.scene);\n this.reflectionProbeMap.set(node.probePositionTargetId, reflectionProbe);\n const exceptions = this.meshesUsingReflectionMaterial.get(node.probePositionTargetId);\n this.reflectedMeshes.forEach(mesh => {\n var _a;\n if (reflectionProbe && reflectionProbe.renderList) {\n if (!(exceptions === null || exceptions === void 0 ? void 0 : exceptions.contains(mesh.id))) {\n (_a = reflectionProbe.renderList) === null || _a === void 0 ? void 0 : _a.push(mesh);\n }\n }\n });\n const sceneBackground = this.scene.meshes.find(obj => {\n return obj.id === 'hdrSkyBox';\n });\n if (reflectionProbe && reflectionProbe.renderList && sceneBackground) {\n reflectionProbe.renderList.push(sceneBackground);\n }\n }\n const mirrorTexture = new MirrorTexture('mirrorTexture', 1024, this.scene);\n const connector = this.viewer.getNode(node.probePositionTargetId);\n let position = new Vector3(0, 0, 0);\n if (connector &&\n connector instanceof exports.Connector &&\n connector._parent &&\n connector._parent instanceof exports.SpaceNode) {\n const connectorMesh = this.viewer.getRendererNode(connector._parent);\n if (connectorMesh) {\n const normal = Vector3.TransformCoordinates(connector.direction.toVec3().scale(-1), connectorMesh.getWorldMatrix().getRotationMatrix());\n position = connectorMesh.absolutePosition;\n const reflector = Plane.FromPositionAndNormal(position, normal);\n mirrorTexture.mirrorPlane = reflector;\n mirrorTexture.renderList = reflectionProbe.renderList;\n }\n }\n if (pbrAdvMaterial) {\n mirrorTexture.invertZ = false;\n mirrorTexture.adaptiveBlurKernel = node.roughness * 100;\n pbrAdvMaterial.reflectionTexture = mirrorTexture;\n }\n }\n }\n }\n }\n if (changes.has('hasTransparency')) {\n pbrAdvMaterial.useAlphaFromAlbedoTexture = node.hasTransparency;\n this.textureRenderer.setTransparency(node, node.hasTransparency);\n }\n if (changes.has('directIntensity'))\n pbrAdvMaterial.directIntensity = node.directIntensity;\n if (changes.has('emissiveIntensity'))\n pbrAdvMaterial.emissiveIntensity = node.emissiveIntensity;\n if (changes.has('environmentIntensity'))\n pbrAdvMaterial.environmentIntensity = node.environmentIntensity;\n if (changes.has('specularIntensity'))\n pbrAdvMaterial.specularIntensity = node.specularIntensity;\n if (changes.has('ambientTextureStrength')) {\n pbrAdvMaterial.ambientTextureStrength = node.ambientTextureStrength || 1;\n }\n if (changes.has('ambientColor')) {\n pbrAdvMaterial.ambientColor = new Color3(node.ambientColor.r, node.ambientColor.g, node.ambientColor.b).toLinearSpace();\n }\n if (changes.has('albedoColor')) {\n pbrAdvMaterial.albedoColor = new Color3(node.albedoColor.r, node.albedoColor.g, node.albedoColor.b).toLinearSpace();\n }\n if (changes.has('specularColor')) {\n pbrAdvMaterial.reflectivityColor = new Color3(node.specularColor.r, node.specularColor.g, node.specularColor.b).toLinearSpace();\n }\n if (changes.has('reflectionColor')) {\n pbrAdvMaterial.reflectionColor = new Color3(node.reflectionColor.r, node.reflectionColor.g, node.reflectionColor.b).toLinearSpace();\n }\n if (changes.has('microSurface'))\n pbrAdvMaterial.microSurface = node.microSurface;\n if (changes.has('useSpecularOverAlpha'))\n pbrAdvMaterial.useSpecularOverAlpha = node.useSpecularOverAlpha;\n if (changes.has('useRadianceOverAlpha'))\n pbrAdvMaterial.useRadianceOverAlpha = node.useRadianceOverAlpha;\n if (changes.has('textures') || changes.has('doubleSided') || changes.has('hasTransparency')) {\n // Only enable the material transparency handling if necessary\n if ((node.hasTransparency && node.doubleSided) ||\n node.textures.find(texture => texture.channel === exports.eTextureChannel.opacity)) {\n // These properties are necessary to ensure that partial transparency self-blending works correctly. There is a performace impact.\n pbrAdvMaterial.separateCullingPass = true;\n pbrAdvMaterial.transparencyMode = Material.MATERIAL_ALPHABLEND;\n // Refraction test\n // let renderTarget = new RefractionTexture(\"rt\", 1024, this.scene, true);\n // renderTarget.renderListPredicate = (mesh) => {\n // return mesh.name === 'hdrSkyBox'; // mesh.material === pbrAdvMaterial && mesh.isEnabled();\n // }\n // renderTarget.lodGenerationScale = 0.5;\n // renderTarget.depth = 1;\n // pbrAdvMaterial.refractionTexture = renderTarget;\n // renderTarget.refractionPlane = new Plane(0, 0, 0, 0);\n // pbrAdvMaterial.subSurface.useAlbedoToTintRefraction = true\n // pbrAdvMaterial.subSurface.indexOfRefraction = 1.5;\n // pbrAdvMaterial.subSurface.isTranslucencyEnabled = true;\n // pbrAdvMaterial.subSurface.linkRefractionWithTransparency = true;\n }\n else {\n pbrAdvMaterial.separateCullingPass = false;\n pbrAdvMaterial.transparencyMode = null;\n }\n }\n if (changes.has('anisotropicIntensity'))\n pbrAdvMaterial.anisotropy.intensity = node.anisotropicIntensity;\n if (changes.has('anisotropicDirection')) {\n pbrAdvMaterial.anisotropy.direction = new Vector2(node.anisotropicDirection.x, node.anisotropicDirection.y);\n }\n if (changes.has('anisotropyEnabled')) {\n pbrAdvMaterial.anisotropy.isEnabled = node.anisotropyEnabled;\n if (node.anisotropyEnabled) {\n const tangentTexture = node.textures.find(texture => texture.channel === exports.eTextureChannel.anisotropyTangent);\n if (tangentTexture && isTrackedClass(tangentTexture)) {\n tangentTexture.isModified = true;\n }\n }\n else {\n pbrAdvMaterial.anisotropy.texture = null;\n }\n }\n if (changes.has('clearcoatIntensity'))\n pbrAdvMaterial.clearCoat.intensity = node.clearcoatIntensity;\n if (changes.has('clearcoatTintEnabled'))\n pbrAdvMaterial.clearCoat.isTintEnabled = node.clearcoatTintEnabled;\n if (changes.has('clearcoatTintIntensity')) {\n pbrAdvMaterial.clearCoat.tintThickness = node.clearcoatTintIntensity;\n }\n if (changes.has('clearcoatTintColor')) {\n pbrAdvMaterial.clearCoat.tintColor = node.clearcoatTintColor.toColor3();\n }\n if (changes.has('clearcoatEnabled')) {\n pbrAdvMaterial.clearCoat.isEnabled = node.clearcoatEnabled;\n if (node.clearcoatEnabled) {\n const clearcoatNormalTexture = node.textures.find(texture => texture.channel === exports.eTextureChannel.clearCoatNormal);\n if (clearcoatNormalTexture && isTrackedClass(clearcoatNormalTexture)) {\n clearcoatNormalTexture.isModified = true;\n }\n }\n else {\n pbrAdvMaterial.clearCoat.bumpTexture = null;\n }\n }\n if (changes.has('subsurfaceThickness')) {\n pbrAdvMaterial.subSurface.minimumThickness = node.subsurfaceThickness;\n }\n if (changes.has('subsurfaceTintColor')) {\n pbrAdvMaterial.subSurface.tintColor = node.subsurfaceTintColor.toColor3();\n }\n if (changes.has('subsurfaceUseAlbedoForTintColor')) {\n pbrAdvMaterial.subSurface.useAlbedoToTintRefraction = node.subsurfaceUseAlbedoForTintColor;\n }\n if (changes.has('subsurfaceRefractionEnabled')) {\n pbrAdvMaterial.subSurface.isRefractionEnabled = node.subsurfaceRefractionEnabled;\n }\n if (changes.has('subsurfaceRefractionIntensity')) {\n pbrAdvMaterial.subSurface.refractionIntensity = node.subsurfaceRefractionIntensity;\n }\n if (changes.has('subsurfaceRefractionIndex')) {\n pbrAdvMaterial.subSurface.indexOfRefraction = node.subsurfaceRefractionIndex;\n }\n if (changes.has('subsurfaceRefractionLinkTransparency')) {\n pbrAdvMaterial.subSurface.linkRefractionWithTransparency = node.subsurfaceRefractionLinkTransparency;\n }\n if (changes.has('subsurfaceTranslucencyEnabled')) {\n pbrAdvMaterial.subSurface.isTranslucencyEnabled = node.subsurfaceTranslucencyEnabled;\n }\n if (changes.has('subsurfaceTranslucencyIntensity')) {\n pbrAdvMaterial.subSurface.translucencyIntensity = node.subsurfaceTranslucencyIntensity;\n }\n if (changes.hasAny('subsurfaceRefractionEnabled', 'subsurfaceTranslucencyEnabled', 'probePositionTargetId')) {\n if (!node.probePositionTargetId || !node.subsurfaceRefractionEnabled) {\n pbrAdvMaterial.refractionTexture = null;\n }\n else {\n let refractionProbe = this.reflectionProbeMap.get(node.probePositionTargetId);\n if (!refractionProbe) {\n refractionProbe = new ReflectionProbe(node.probePositionTargetId, 512, this.scene);\n const connector = this.viewer.getNode(node.probePositionTargetId);\n if (connector instanceof exports.Connector) {\n const connectorMesh = this.viewer.getRendererNode(connector);\n refractionProbe.attachToMesh(connectorMesh);\n this.reflectionProbeMap.set(node.probePositionTargetId, refractionProbe);\n }\n const exceptions = this.meshesUsingReflectionMaterial.get(node.probePositionTargetId);\n this.reflectedMeshes.forEach(mesh => {\n var _a;\n if (refractionProbe && refractionProbe.renderList) {\n if (!(exceptions === null || exceptions === void 0 ? void 0 : exceptions.contains(mesh.id))) {\n (_a = refractionProbe.renderList) === null || _a === void 0 ? void 0 : _a.push(mesh);\n }\n }\n });\n const sceneBackground = this.scene.meshes.find(obj => {\n return obj.id === 'hdrSkyBox';\n });\n if (refractionProbe && refractionProbe.renderList && sceneBackground) {\n refractionProbe.renderList.push(sceneBackground);\n }\n }\n const refractionTexture = new RefractionTexture('refractionTexture', 1024, this.scene);\n const connector = this.viewer.getNode(node.probePositionTargetId);\n let position = new Vector3(0, 0, 0);\n if (connector &&\n connector instanceof exports.Connector &&\n connector._parent &&\n connector._parent instanceof exports.SpaceNode) {\n const connectorMesh = this.viewer.getRendererNode(connector._parent);\n if (connectorMesh) {\n const normal = Vector3.TransformCoordinates(connector.direction.toVec3(), connectorMesh.getWorldMatrix().getRotationMatrix());\n position = connectorMesh.absolutePosition;\n const reflector = Plane.FromPositionAndNormal(position, normal);\n refractionTexture.refractionPlane = reflector;\n refractionTexture.renderList = refractionProbe.renderList;\n refractionTexture.depth = node.subsurfaceRefractionIndex;\n }\n }\n if (pbrAdvMaterial) {\n refractionTexture.invertZ = false;\n pbrAdvMaterial.refractionTexture = refractionTexture;\n }\n }\n if (node.subsurfaceRefractionEnabled || node.subsurfaceTranslucencyEnabled) {\n if (!pbrAdvMaterial.subSurface.thicknessTexture) {\n const subsurfaceThicknessTexture = node.textures.find(texture => texture.channel === exports.eTextureChannel.subsurfaceThickness);\n if (subsurfaceThicknessTexture && isTrackedClass(subsurfaceThicknessTexture)) {\n subsurfaceThicknessTexture.isModified = true;\n }\n }\n }\n else {\n pbrAdvMaterial.subSurface.thicknessTexture = null;\n }\n }\n if (changes.has('workflowMode')) {\n if (node.workflowMode === exports.eMaterialWorkflow.metallicRoughness) {\n pbrAdvMaterial.metallic = node.metallic;\n pbrAdvMaterial.roughness = node.roughness;\n }\n else {\n /* Remove the metallic texture if workflow is not metallic-roughness, otherwise metallic-roughness gets used */\n pbrAdvMaterial.metallicTexture = undefined;\n pbrAdvMaterial.metallic = null;\n pbrAdvMaterial.roughness = null;\n }\n if (pbrAdvMaterial.reflectionTexture && pbrAdvMaterial.reflectionTexture instanceof MirrorTexture) {\n if (node.roughness !== 0) {\n pbrAdvMaterial.reflectionTexture.adaptiveBlurKernel = node.roughness * 100;\n }\n else {\n //When setting the adaptiveBlurKernel to 0 we get very odd behavior.\n pbrAdvMaterial.reflectionTexture.adaptiveBlurKernel = 0.000001;\n }\n }\n }\n else if (node.workflowMode === exports.eMaterialWorkflow.metallicRoughness) {\n if (changes.has('metallic'))\n pbrAdvMaterial.metallic = node.metallic;\n if (changes.has('roughness')) {\n pbrAdvMaterial.roughness = node.roughness;\n if (pbrAdvMaterial.reflectionTexture && pbrAdvMaterial.reflectionTexture instanceof MirrorTexture) {\n if (node.roughness !== 0) {\n pbrAdvMaterial.reflectionTexture.adaptiveBlurKernel = node.roughness * 100;\n }\n else {\n //When setting the adaptiveBlurKernel to 0 we get very odd behavior.\n pbrAdvMaterial.reflectionTexture.adaptiveBlurKernel = 0.000001;\n }\n }\n }\n }\n }\n if (material) {\n const changes = _changes;\n if (changes.has('alpha'))\n material.alpha = node.alpha;\n if (changes.has('zOffset'))\n material.zOffset = node.zOffset || 0;\n if (changes.has('doubleSided')) {\n material.backFaceCulling = !node.doubleSided;\n material.twoSidedLighting = !!node.doubleSided;\n }\n if (changes.has('emissiveColor')) {\n if (node.emissiveColor) {\n material.emissiveColor = new Color3(node.emissiveColor.r, node.emissiveColor.g, node.emissiveColor.b);\n }\n else {\n material.emissiveColor = Color3.Black();\n }\n }\n // Wireframe mode for debugging geometry\n // if(true || node.viewMode === eMaterialViewMode.wireframe) {\n // material.wireframe = true;\n // material.emissiveColor = Color3.Red();\n // }\n }\n this.viewer.materialWarningsNeedsUpdate.next();\n }\n setRenderModeOnMaterial(material, mode) {\n if (mode === exports.eRenderMode.wireframe) {\n material.pointsCloud = false;\n material.wireframe = true;\n }\n else if (mode === exports.eRenderMode.hiddenEdges) {\n material.wireframe = false;\n material.pointsCloud = true;\n }\n else {\n material.pointsCloud = false;\n material.wireframe = false;\n }\n }\n getNumberOfMaterialRef(sceneNode, materialNode) {\n const allMeshes = sceneNode.filterByType(...Token.MeshTokens);\n let nrefs = 0;\n for (const m of allMeshes) {\n if (m.materialId == materialNode._dynamicId)\n nrefs++;\n }\n return nrefs;\n }\n updateReflectionProbes() {\n var _a, _b;\n this.meshesUsingReflectionMaterial.clear();\n (_a = this.viewer.sceneNode._manager) === null || _a === void 0 ? void 0 : _a.getAllNodes().forEach(mesh => {\n var _a;\n if (mesh instanceof exports.MeshNode) {\n if (mesh.materialId) {\n const material = (_a = mesh._manager) === null || _a === void 0 ? void 0 : _a.getById(mesh.materialId);\n if (material && material instanceof exports.MaterialNode && material.probePositionTargetId) {\n let exceptionList = this.meshesUsingReflectionMaterial.get(material.probePositionTargetId);\n if (!exceptionList) {\n exceptionList = [mesh._dynamicId];\n }\n else {\n exceptionList.push(mesh._dynamicId);\n }\n this.meshesUsingReflectionMaterial.set(material.probePositionTargetId, exceptionList);\n }\n }\n }\n });\n const sceneBackground = this.scene.meshes.filter(obj => {\n return obj.id === 'hdrSkyBox';\n });\n for (const keyValuePair of this.reflectionProbeMap) {\n const reflectionProbe = keyValuePair[1];\n const connectorId = keyValuePair[0];\n (_b = reflectionProbe.renderList) === null || _b === void 0 ? void 0 : _b.clear();\n const exceptions = this.meshesUsingReflectionMaterial.get(connectorId);\n this.reflectedMeshes.forEach(mesh => {\n var _a;\n if (!(exceptions === null || exceptions === void 0 ? void 0 : exceptions.contains(mesh.id))) {\n (_a = reflectionProbe.renderList) === null || _a === void 0 ? void 0 : _a.push(mesh);\n }\n });\n if (reflectionProbe && reflectionProbe.renderList) {\n reflectionProbe.renderList.push(sceneBackground[0]);\n }\n }\n }\n }\n\n /**\n * Class containing static functions to help procedurally build meshes\n */\n class CylinderBuilder$1 {\n /**\n * Creates a torus mesh\n * * The parameter `diameter` sets the diameter size (float) of the torus (default 1)\n * * The parameter `thickness` sets the diameter size of the tube of the torus (float, default 0.5)\n * * The parameter `tessellation` sets the number of torus sides (postive integer, default 16)\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/babylon101/Cylinderover_basic_elements#side-orientation\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param scene defines the hosting scene\n * @returns the torus mesh\n * @see https://doc.babylonjs.com/how_to/set_shapes#torus\n */\n static CreateCylinder(name, options, scene) {\n const Cylinder = new Mesh(name, scene);\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n Cylinder._originalBuilderSideOrientation = options.sideOrientation;\n const vertexData = createCylinder(options);\n vertexData.applyToMesh(Cylinder, options.updatable);\n return Cylinder;\n function createCylinder(options) {\n const height = options.height || 2;\n let diameterTop = options.diameterTop === 0 ? 0 : options.diameterTop || options.diameter || 1;\n let diameterBottom = options.diameterBottom === 0 ? 0 : options.diameterBottom || options.diameter || 1;\n diameterTop = diameterTop || 0.00001; // Prevent broken normals\n diameterBottom = diameterBottom || 0.00001; // Prevent broken normals\n const tessellation = options.tessellation || 24;\n const subdivisions = options.subdivisions || 1;\n const hasRings = options.hasRings ? true : false;\n const enclose = options.enclose ? true : false;\n const cap = options.cap === 0 ? 0 : options.cap || Mesh.CAP_ALL;\n const arc = options.arc && (options.arc <= 0 || options.arc > 1) ? 1.0 : options.arc || 1.0;\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\n const faceUV = options.faceUV || new Array(3);\n const faceColors = options.faceColors;\n // default face colors and UV if undefined\n const quadNb = arc !== 1 && enclose ? 2 : 0;\n const ringNb = hasRings ? subdivisions : 1;\n const surfaceNb = 2 + (1 + quadNb) * ringNb;\n let f;\n for (f = 0; f < surfaceNb; f++) {\n if (faceColors && faceColors[f] === undefined) {\n faceColors[f] = new Color4(1, 1, 1, 1);\n }\n }\n for (f = 0; f < surfaceNb; f++) {\n if (faceUV && faceUV[f] === undefined) {\n faceUV[f] = new Vector4(0, 0, 1, 1);\n }\n }\n const indices = new Array();\n const positions = new Array();\n const normals = new Array();\n const uvs = new Array();\n const colors = new Array();\n const angle_step = (Math.PI * 2 * arc) / tessellation;\n let angle;\n let h;\n let radius;\n const tan = (diameterBottom - diameterTop) / 2 / height;\n const ringVertex = Vector3.Zero();\n const ringNormal = Vector3.Zero();\n const ringFirstVertex = Vector3.Zero();\n const ringFirstNormal = Vector3.Zero();\n const quadNormal = Vector3.Zero();\n const Y = Axis.Y;\n // positions, normals, uvs\n let i;\n let j;\n let r;\n let ringIdx = 1;\n let s = 1; // surface index\n let cs = 0;\n let v = 0;\n for (i = 0; i <= subdivisions; i++) {\n h = i / subdivisions;\n radius = (h * (diameterTop - diameterBottom) + diameterBottom) / 2;\n ringIdx = hasRings && i !== 0 && i !== subdivisions ? 2 : 1;\n for (r = 0; r < ringIdx; r++) {\n if (hasRings) {\n s += r;\n }\n if (enclose) {\n s += 2 * r;\n }\n for (j = 0; j <= tessellation; j++) {\n angle = j * angle_step;\n // position\n ringVertex.x = Math.cos(-angle) * radius;\n ringVertex.y = -height / 2 + h * height;\n ringVertex.z = Math.sin(-angle) * radius;\n // normal\n if (diameterTop === 0 && i === subdivisions) {\n // if no top cap, reuse former normals\n ringNormal.x = normals[normals.length - (tessellation + 1) * 3];\n ringNormal.y = normals[normals.length - (tessellation + 1) * 3 + 1];\n ringNormal.z = normals[normals.length - (tessellation + 1) * 3 + 2];\n }\n else {\n ringNormal.x = ringVertex.x;\n ringNormal.z = ringVertex.z;\n ringNormal.y = Math.sqrt(ringNormal.x * ringNormal.x + ringNormal.z * ringNormal.z) * tan;\n ringNormal.normalize();\n }\n // keep first ring vertex values for enclose\n if (j === 0) {\n ringFirstVertex.copyFrom(ringVertex);\n ringFirstNormal.copyFrom(ringNormal);\n }\n positions.push(ringVertex.x, ringVertex.y, ringVertex.z);\n normals.push(ringNormal.x, ringNormal.y, ringNormal.z);\n if (hasRings) {\n v = cs !== s ? faceUV[s].y : faceUV[s].w;\n }\n else {\n v = faceUV[s].y + (faceUV[s].w - faceUV[s].y) * h;\n }\n uvs.push(faceUV[s].x + ((faceUV[s].z - faceUV[s].x) * j) / tessellation, v);\n if (faceColors) {\n colors.push(faceColors[s].r, faceColors[s].g, faceColors[s].b, faceColors[s].a);\n }\n }\n // if enclose, add four vertices and their dedicated normals\n if (arc !== 1 && enclose) {\n positions.push(ringVertex.x, ringVertex.y, ringVertex.z);\n positions.push(0, ringVertex.y, 0);\n positions.push(0, ringVertex.y, 0);\n positions.push(ringFirstVertex.x, ringFirstVertex.y, ringFirstVertex.z);\n Vector3.CrossToRef(Y, ringNormal, quadNormal);\n quadNormal.normalize();\n normals.push(quadNormal.x, quadNormal.y, quadNormal.z, quadNormal.x, quadNormal.y, quadNormal.z);\n Vector3.CrossToRef(ringFirstNormal, Y, quadNormal);\n quadNormal.normalize();\n normals.push(quadNormal.x, quadNormal.y, quadNormal.z, quadNormal.x, quadNormal.y, quadNormal.z);\n if (hasRings) {\n v = cs !== s ? faceUV[s + 1].y : faceUV[s + 1].w;\n }\n else {\n v = faceUV[s + 1].y + (faceUV[s + 1].w - faceUV[s + 1].y) * h;\n }\n uvs.push(faceUV[s + 1].x, v);\n uvs.push(faceUV[s + 1].z, v);\n if (hasRings) {\n v = cs !== s ? faceUV[s + 2].y : faceUV[s + 2].w;\n }\n else {\n v = faceUV[s + 2].y + (faceUV[s + 2].w - faceUV[s + 2].y) * h;\n }\n uvs.push(faceUV[s + 2].x, v);\n uvs.push(faceUV[s + 2].z, v);\n if (faceColors) {\n colors.push(faceColors[s + 1].r, faceColors[s + 1].g, faceColors[s + 1].b, faceColors[s + 1].a);\n colors.push(faceColors[s + 1].r, faceColors[s + 1].g, faceColors[s + 1].b, faceColors[s + 1].a);\n colors.push(faceColors[s + 2].r, faceColors[s + 2].g, faceColors[s + 2].b, faceColors[s + 2].a);\n colors.push(faceColors[s + 2].r, faceColors[s + 2].g, faceColors[s + 2].b, faceColors[s + 2].a);\n }\n }\n if (cs !== s) {\n cs = s;\n }\n }\n }\n // indices\n const e = arc !== 1 && enclose ? tessellation + 4 : tessellation; // correction of number of iteration if enclose\n let sI;\n i = 0;\n for (sI = 0; sI < subdivisions; sI++) {\n let i0 = 0;\n let i1 = 0;\n let i2 = 0;\n let i3 = 0;\n for (j = 0; j < tessellation; j++) {\n i0 = i * (e + 1) + j; // top left\n i1 = (i + 1) * (e + 1) + j; // bottom left\n i2 = i * (e + 1) + (j + 1); // top right\n i3 = (i + 1) * (e + 1) + (j + 1); // bottom right\n indices.push(i0, i1, i2);\n indices.push(i3, i2, i1);\n }\n if (arc !== 1 && enclose) {\n // if enclose, add two quads\n indices.push(i0 + 2, i1 + 2, i2 + 2);\n indices.push(i3 + 2, i2 + 2, i1 + 2);\n indices.push(i0 + 4, i1 + 4, i2 + 4);\n indices.push(i3 + 4, i2 + 4, i1 + 4);\n }\n else if (arc === 1) {\n indices.push(i2, i3, i * (e + 1));\n indices.push((i + 1) * (e + 1), i * (e + 1), i3);\n }\n i = hasRings ? i + 2 : i + 1;\n }\n // Caps\n const createCylinderCap = (isTop) => {\n const radius = isTop ? diameterTop / 2 : diameterBottom / 2;\n if (radius === 0) {\n return;\n }\n // Cap positions, normals & uvs\n let angle;\n let circleVector;\n let i;\n const u = isTop ? faceUV[surfaceNb - 1] : faceUV[0];\n let c = null;\n if (faceColors) {\n c = isTop ? faceColors[surfaceNb - 1] : faceColors[0];\n }\n // cap center\n const vbase = positions.length / 3;\n const offset = isTop ? height / 2 : -height / 2;\n const center = new Vector3(0, offset, 0);\n positions.push(center.x, center.y, center.z);\n normals.push(0, isTop ? 1 : -1, 0);\n uvs.push(u.x + (u.z - u.x) * 0.5, u.y + (u.w - u.y) * 0.5);\n if (c) {\n colors.push(c.r, c.g, c.b, c.a);\n }\n const textureScale = new Vector2(0.5, 0.5);\n for (i = 0; i <= tessellation; i++) {\n angle = i * angle_step;\n const cos = Math.cos(-angle);\n const sin = Math.sin(-angle);\n circleVector = new Vector3(cos * radius, offset, sin * radius);\n const textureCoordinate = new Vector2(cos * textureScale.x + 0.5, sin * textureScale.y + 0.5);\n positions.push(circleVector.x, circleVector.y, circleVector.z);\n normals.push(0, isTop ? 1 : -1, 0);\n uvs.push(u.x + (u.z - u.x) * textureCoordinate.x, u.y + (u.w - u.y) * textureCoordinate.y);\n if (c) {\n colors.push(c.r, c.g, c.b, c.a);\n }\n }\n // Cap indices\n for (i = 0; i < tessellation; i++) {\n if (!isTop) {\n indices.push(vbase);\n indices.push(vbase + (i + 1));\n indices.push(vbase + (i + 2));\n }\n else {\n indices.push(vbase);\n indices.push(vbase + (i + 2));\n indices.push(vbase + (i + 1));\n }\n }\n // Close the loop\n if (arc === 1) {\n if (!isTop) {\n indices.push(vbase, vbase + tessellation + 1, vbase + 1);\n }\n else {\n indices.push(vbase, vbase + 1, vbase + tessellation + 1);\n }\n }\n };\n // add caps to geometry based on cap parameter\n if (cap === Mesh.CAP_START || cap === Mesh.CAP_ALL) {\n createCylinderCap(false);\n }\n if (cap === Mesh.CAP_END || cap === Mesh.CAP_ALL) {\n createCylinderCap(true);\n }\n // Sides\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\n const vertexData = new VertexData();\n vertexData.indices = indices;\n vertexData.positions = positions;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n if (faceColors) {\n vertexData.colors = colors;\n }\n return vertexData;\n }\n }\n }\n\n /**\n * Class containing static functions to help procedurally build meshes\n */\n class DiscBuilder {\n /**\n * Creates a torus mesh\n * * The parameter `diameter` sets the diameter size (float) of the torus (default 1)\n * * The parameter `thickness` sets the diameter size of the tube of the torus (float, default 0.5)\n * * The parameter `tessellation` sets the number of torus sides (postive integer, default 16)\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/babylon101/discover_basic_elements#side-orientation\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param scene defines the hosting scene\n * @returns the torus mesh\n * @see https://doc.babylonjs.com/how_to/set_shapes#torus\n */\n static CreateDisc(name, options, scene) {\n const disc = new Mesh(name, scene);\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n disc._originalBuilderSideOrientation = options.sideOrientation;\n const vertexData = createDisc(options);\n vertexData.applyToMesh(disc, options.updatable);\n return disc;\n function createDisc(options) {\n const positions = new Array();\n const indices = new Array();\n const normals = new Array();\n const uvs = new Array();\n const radius = options.radius || 0.5;\n const tessellation = options.tessellation || 64;\n const arc = options.arc && (options.arc <= 0 || options.arc > 1) ? 1.0 : options.arc || 1.0;\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\n // positions and uvs\n positions.push(0, 0, 0); // disc center first\n uvs.push(0.5, 0.5);\n const theta = Math.PI * 2 * arc;\n const step = theta / tessellation;\n for (let a = 0; arc === 1 ? a < theta : a <= theta; a += step) {\n const x = Math.cos(a);\n const y = Math.sin(a);\n const u = (x + 1) / 2;\n const v = (1 - y) / 2;\n positions.push(radius * x, radius * y, 0);\n uvs.push(u, v);\n }\n //indices\n const vertexNb = positions.length / 3;\n for (let i = 1; i < vertexNb - 1; i++) {\n indices.push(i + 1, 0, i);\n }\n if (arc === 1) {\n indices.push(1, 0, vertexNb - 1); // close the circle\n }\n // result\n VertexData.ComputeNormals(positions, indices, normals);\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\n const vertexData = new VertexData();\n vertexData.indices = indices;\n vertexData.positions = positions;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n return vertexData;\n }\n }\n }\n\n /**\n * Class containing static functions to help procedurally build meshes\n */\n class PlaneBuilder$1 {\n /**\n * Creates a torus mesh\n * * The parameter `diameter` sets the diameter size (float) of the torus (default 1)\n * * The parameter `thickness` sets the diameter size of the tube of the torus (float, default 0.5)\n * * The parameter `tessellation` sets the number of torus sides (postive integer, default 16)\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/babylon101/discover_basic_elements#side-orientation\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param scene defines the hosting scene\n * @returns the torus mesh\n * @see https://doc.babylonjs.com/how_to/set_shapes#torus\n */\n static CreatePlane(name, options, scene) {\n const plane = new Mesh(name, scene);\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n const flipTile = options.pattern || Mesh.NO_FLIP;\n const tileWidth = options.tileWidth || options.tileSize || 1;\n const tileHeight = options.tileHeight || options.tileSize || 1;\n const alignH = options.alignHorizontal || 0;\n const alignV = options.alignVertical || 0;\n const width = options.width || options.size || 1;\n const tilesX = Math.floor(width / tileWidth);\n let offsetX = width - tilesX * tileWidth;\n const height = options.height || options.size || 1;\n const tilesY = Math.floor(height / tileHeight);\n let offsetY = height - tilesY * tileHeight;\n const halfWidth = (tileWidth * tilesX) / 2;\n const halfHeight = (tileHeight * tilesY) / 2;\n const indices = [];\n const positions = [];\n const normals = new Float32Array(tilesX * tilesY * 12);\n const colorSetArray = [1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1];\n const normalSetArray = [0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1];\n const colorSet = Float32Array.from(colorSetArray);\n const normalSet = Float32Array.from(normalSetArray);\n let adjustX = 0;\n let adjustY = 0;\n let startX = 0;\n let startY = 0;\n let endX = 0;\n let endY = 0;\n //Part Tiles\n if (offsetX > 0 || offsetY > 0) {\n startX = -halfWidth;\n startY = -halfHeight;\n endX = halfWidth;\n endY = halfHeight;\n switch (alignH) {\n case Mesh.CENTER:\n offsetX /= 2;\n startX -= offsetX;\n endX += offsetX;\n break;\n case Mesh.LEFT:\n endX += offsetX;\n adjustX = -offsetX / 2;\n break;\n case Mesh.RIGHT:\n startX -= offsetX;\n adjustX = offsetX / 2;\n break;\n }\n switch (alignV) {\n case Mesh.CENTER:\n offsetY /= 2;\n startY -= offsetY;\n endY += offsetY;\n break;\n case Mesh.BOTTOM:\n endY += offsetY;\n adjustY = -offsetY / 2;\n break;\n case Mesh.TOP:\n startY -= offsetY;\n adjustY = offsetY / 2;\n break;\n }\n }\n const uvBase = [];\n uvBase[0] = [0, 0, 1, 0, 1, 1, 0, 1];\n uvBase[1] = [0, 0, 1, 0, 1, 1, 0, 1];\n if (flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.ROTATE_ROW) {\n uvBase[1] = [1, 1, 0, 1, 0, 0, 1, 0];\n }\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.FLIP_ROW) {\n uvBase[1] = [1, 0, 0, 0, 0, 1, 1, 1];\n }\n if (flipTile === Mesh.FLIP_N_ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvBase[1] = [0, 1, 1, 1, 1, 0, 0, 0];\n }\n const uvs = new Float32Array(tilesY * tilesX * 8);\n const colors = new Float32Array(tilesY * tilesX * 16);\n let index = 0;\n for (let y = 0; y < tilesY; y++) {\n for (let x = 0; x < tilesX; x++) {\n positions.push(-halfWidth + x * tileWidth + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\n positions.push(-halfWidth + x * tileWidth + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\n if (flipTile === Mesh.FLIP_TILE ||\n flipTile === Mesh.ROTATE_TILE ||\n flipTile === Mesh.FLIP_N_ROTATE_TILE) {\n uvs.set(uvBase[((x % 2) + (y % 2)) % 2], index * 2);\n }\n else if (flipTile === Mesh.FLIP_ROW ||\n flipTile === Mesh.ROTATE_ROW ||\n flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvs.set(uvBase[y % 2], index * 2);\n }\n else {\n uvs.set(uvBase[0], index * 2);\n }\n colors.set(colorSet, index * 4);\n normals.set(normalSet, index * 3);\n index += 4;\n }\n }\n //Part Tiles\n if (offsetX > 0 || offsetY > 0) {\n const partialBottomRow = offsetY > 0 && (alignV === Mesh.CENTER || alignV === Mesh.TOP);\n const partialTopRow = offsetY > 0 && (alignV === Mesh.CENTER || alignV === Mesh.BOTTOM);\n const partialLeftCol = offsetX > 0 && (alignH === Mesh.CENTER || alignH === Mesh.RIGHT);\n const partialRightCol = offsetX > 0 && (alignH === Mesh.CENTER || alignH === Mesh.LEFT);\n let uvPart = [];\n let a, b, c, d;\n //corners\n if (partialBottomRow && partialLeftCol) {\n //bottom left corner\n positions.push(startX + adjustX, startY + adjustY, 0);\n positions.push(-halfWidth + adjustX, startY + adjustY, 0);\n positions.push(-halfWidth + adjustX, startY + offsetY + adjustY, 0);\n positions.push(startX + adjustX, startY + offsetY + adjustY, 0);\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\n index += 4;\n a = 1 - offsetX / tileWidth;\n b = 1 - offsetY / tileHeight;\n c = 1;\n d = 1;\n uvPart = [a, b, c, b, c, d, a, d];\n if (flipTile === Mesh.ROTATE_ROW) {\n uvPart = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\n }\n if (flipTile === Mesh.FLIP_ROW) {\n uvPart = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\n }\n if (flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvPart = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\n }\n uvs.set(uvPart, index * 2);\n uvs.set(uvPart, index);\n colors.set(colorSet, index * 4);\n normals.set(normalSet, index * 3);\n }\n if (partialBottomRow && partialRightCol) {\n //bottom right corner\n positions.push(halfWidth + adjustX, startY + adjustY, 0);\n positions.push(endX + adjustX, startY + adjustY, 0);\n positions.push(endX + adjustX, startY + offsetY + adjustY, 0);\n positions.push(halfWidth + adjustX, startY + offsetY + adjustY, 0);\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\n index += 4;\n a = 0;\n b = 1 - offsetY / tileHeight;\n c = offsetX / tileWidth;\n d = 1;\n uvPart = [a, b, c, b, c, d, a, d];\n if (flipTile === Mesh.ROTATE_ROW || (flipTile === Mesh.ROTATE_TILE && tilesX % 2 === 0)) {\n uvPart = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\n }\n if (flipTile === Mesh.FLIP_ROW || (flipTile === Mesh.FLIP_TILE && tilesX % 2 === 0)) {\n uvPart = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\n }\n if (flipTile === Mesh.FLIP_N_ROTATE_ROW || (flipTile === Mesh.FLIP_N_ROTATE_TILE && tilesX % 2 === 0)) {\n uvPart = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\n }\n uvs.set(uvPart, index * 2);\n colors.set(colorSet, index * 4);\n normals.set(normalSet, index * 3);\n }\n if (partialTopRow && partialLeftCol) {\n //top left corner\n positions.push(startX + adjustX, halfHeight + adjustY, 0);\n positions.push(-halfWidth + adjustX, halfHeight + adjustY, 0);\n positions.push(-halfWidth + adjustX, endY + adjustY, 0);\n positions.push(startX + adjustX, endY + adjustY, 0);\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\n index += 4;\n a = 1 - offsetX / tileWidth;\n b = 0;\n c = 1;\n d = offsetY / tileHeight;\n uvPart = [a, b, c, b, c, d, a, d];\n if ((flipTile === Mesh.ROTATE_ROW && tilesY % 2 === 1) ||\n (flipTile === Mesh.ROTATE_TILE && tilesY % 1 === 0)) {\n uvPart = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\n }\n if ((flipTile === Mesh.FLIP_ROW && tilesY % 2 === 1) ||\n (flipTile === Mesh.FLIP_TILE && tilesY % 2 === 0)) {\n uvPart = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\n }\n if ((flipTile === Mesh.FLIP_N_ROTATE_ROW && tilesY % 2 === 1) ||\n (flipTile === Mesh.FLIP_N_ROTATE_TILE && tilesY % 2 === 0)) {\n uvPart = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\n }\n uvs.set(uvPart, index * 2);\n colors.set(colorSet, index * 4);\n normals.set(normalSet, index * 3);\n }\n if (partialTopRow && partialRightCol) {\n //top right corner\n positions.push(halfWidth + adjustX, halfHeight + adjustY, 0);\n positions.push(endX + adjustX, halfHeight + adjustY, 0);\n positions.push(endX + adjustX, endY + adjustY, 0);\n positions.push(halfWidth + adjustX, endY + adjustY, 0);\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\n index += 4;\n a = 0;\n b = 0;\n c = offsetX / tileWidth;\n d = offsetY / tileHeight;\n uvPart = [a, b, c, b, c, d, a, d];\n if ((flipTile === Mesh.ROTATE_ROW && tilesY % 2 === 1) ||\n (flipTile === Mesh.ROTATE_TILE && (tilesY + tilesX) % 2 === 1)) {\n uvPart = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\n }\n if ((flipTile === Mesh.FLIP_ROW && tilesY % 2 === 1) ||\n (flipTile === Mesh.FLIP_TILE && (tilesY + tilesX) % 2 === 1)) {\n uvPart = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\n }\n if ((flipTile === Mesh.FLIP_N_ROTATE_ROW && tilesY % 2 === 1) ||\n (flipTile === Mesh.FLIP_N_ROTATE_TILE && (tilesY + tilesX) % 2 === 1)) {\n uvPart = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\n }\n uvs.set(uvPart, index * 2);\n colors.set(colorSet, index * 4);\n normals.set(normalSet, index * 3);\n }\n //part rows\n if (partialBottomRow) {\n const uvBaseBR = [];\n a = 0;\n b = 1 - offsetY / tileHeight;\n c = 1;\n d = 1;\n uvBaseBR[0] = [a, b, c, b, c, d, a, d];\n uvBaseBR[1] = [a, b, c, b, c, d, a, d];\n if (flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.ROTATE_ROW) {\n uvBaseBR[1] = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\n }\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.FLIP_ROW) {\n uvBaseBR[1] = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\n }\n if (flipTile === Mesh.FLIP_N_ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvBaseBR[1] = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\n }\n for (let x = 0; x < tilesX; x++) {\n positions.push(-halfWidth + x * tileWidth + adjustX, startY + adjustY, 0);\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, startY + adjustY, 0);\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, startY + offsetY + adjustY, 0);\n positions.push(-halfWidth + x * tileWidth + adjustX, startY + offsetY + adjustY, 0);\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\n index += 4;\n if (flipTile === Mesh.FLIP_TILE ||\n flipTile === Mesh.ROTATE_TILE ||\n flipTile === Mesh.FLIP_N_ROTATE_TILE) {\n uvs.set(uvBaseBR[(x + 1) % 2], index * 2);\n }\n else if (flipTile === Mesh.FLIP_ROW ||\n flipTile === Mesh.ROTATE_ROW ||\n flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvs.set(uvBaseBR[1], index * 2);\n }\n else {\n uvs.set(uvBaseBR[0], index * 2);\n }\n colors.set(colorSet, index * 4);\n normals.set(normalSet, index * 3);\n }\n }\n if (partialTopRow) {\n const uvBaseTR = [];\n a = 0;\n b = 0;\n c = 1;\n d = offsetY / tileHeight;\n uvBaseTR[0] = [a, b, c, b, c, d, a, d];\n uvBaseTR[1] = [a, b, c, b, c, d, a, d];\n if (flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.ROTATE_ROW) {\n uvBaseTR[1] = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\n }\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.FLIP_ROW) {\n uvBaseTR[1] = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\n }\n if (flipTile === Mesh.FLIP_N_ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvBaseTR[1] = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\n }\n for (let x = 0; x < tilesX; x++) {\n positions.push(-halfWidth + x * tileWidth + adjustX, endY - offsetY + adjustY, 0);\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, endY - offsetY + adjustY, 0);\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, endY + adjustY, 0);\n positions.push(-halfWidth + x * tileWidth + adjustX, endY + adjustY, 0);\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\n index += 4;\n if (flipTile === Mesh.FLIP_TILE ||\n flipTile === Mesh.ROTATE_TILE ||\n flipTile === Mesh.FLIP_N_ROTATE_TILE) {\n uvs.set(uvBaseTR[(x + tilesY) % 2], index * 2);\n }\n else if (flipTile === Mesh.FLIP_ROW ||\n flipTile === Mesh.ROTATE_ROW ||\n flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvs.set(uvBaseTR[tilesY % 2], index * 2);\n }\n else {\n uvs.set(uvBaseTR[0], index * 2);\n }\n colors.set(colorSet, index * 4);\n normals.set(normalSet, index * 3);\n }\n }\n if (partialLeftCol) {\n const uvBaseLC = [];\n a = 1 - offsetX / tileWidth;\n b = 0;\n c = 1;\n d = 1;\n uvBaseLC[0] = [a, b, c, b, c, d, a, d];\n uvBaseLC[1] = [a, b, c, b, c, d, a, d];\n if (flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.ROTATE_ROW) {\n uvBaseLC[1] = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\n }\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.FLIP_ROW) {\n uvBaseLC[1] = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\n }\n if (flipTile === Mesh.FLIP_N_ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvBaseLC[1] = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\n }\n for (let y = 0; y < tilesY; y++) {\n positions.push(startX + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\n positions.push(startX + offsetX + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\n positions.push(startX + offsetX + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\n positions.push(startX + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\n index += 4;\n if (flipTile === Mesh.FLIP_TILE ||\n flipTile === Mesh.ROTATE_TILE ||\n flipTile === Mesh.FLIP_N_ROTATE_TILE) {\n uvs.set(uvBaseLC[(y + 1) % 2], index * 2);\n }\n else if (flipTile === Mesh.FLIP_ROW ||\n flipTile === Mesh.ROTATE_ROW ||\n flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvs.set(uvBaseLC[y % 2], index * 2);\n }\n else {\n uvs.set(uvBaseLC[0], index * 2);\n }\n colors.set(colorSet, index * 4);\n normals.set(normalSet, index * 3);\n }\n }\n if (partialRightCol) {\n const uvBaseRC = [];\n a = 0;\n b = 0;\n c = offsetX / tileHeight;\n d = 1;\n uvBaseRC[0] = [a, b, c, b, c, d, a, d];\n uvBaseRC[1] = [a, b, c, b, c, d, a, d];\n if (flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.ROTATE_ROW) {\n uvBaseRC[1] = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\n }\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.FLIP_ROW) {\n uvBaseRC[1] = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\n }\n if (flipTile === Mesh.FLIP_N_ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvBaseRC[1] = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\n }\n for (let y = 0; y < tilesY; y++) {\n positions.push(endX - offsetX + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\n positions.push(endX + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\n positions.push(endX + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\n positions.push(endX - offsetX + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\n index += 4;\n if (flipTile === Mesh.FLIP_TILE ||\n flipTile === Mesh.ROTATE_TILE ||\n flipTile === Mesh.FLIP_N_ROTATE_TILE) {\n uvs.set(uvBaseRC[(y + tilesX) % 2], index * 2);\n }\n else if (flipTile === Mesh.FLIP_ROW ||\n flipTile === Mesh.ROTATE_ROW ||\n flipTile === Mesh.FLIP_N_ROTATE_ROW) {\n uvs.set(uvBaseRC[y % 2], index * 2);\n }\n else {\n uvs.set(uvBaseRC[0], index * 2);\n }\n colors.set(colorSet, index * 4);\n normals.set(normalSet, index * 3);\n }\n }\n }\n //reverse V values since old PlaneBuilder built them that way\n for (let i = 0; i < uvs.length / 2; i++) {\n uvs[i * 2 + 1] = 1 - uvs[i * 2 + 1];\n }\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\n // sides\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\n // Result\n const vertexData = new VertexData();\n vertexData.indices = indices;\n vertexData.positions = positions;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n vertexData.applyToMesh(plane, options.updatable);\n return plane;\n }\n }\n\n /**\n * Class containing static functions to help procedurally build meshes\n */\n class SphereBuilder$1 {\n /**\n * Creates a torus mesh\n * * The parameter `diameter` sets the diameter size (float) of the torus (default 1)\n * * The parameter `thickness` sets the diameter size of the tube of the torus (float, default 0.5)\n * * The parameter `tessellation` sets the number of torus sides (postive integer, default 16)\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/babylon101/discover_basic_elements#side-orientation\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param scene defines the hosting scene\n * @returns the torus mesh\n * @see https://doc.babylonjs.com/how_to/set_shapes#torus\n */\n static CreateSphere(name, options, scene) {\n const sphere = new Mesh(name, scene);\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n sphere._originalBuilderSideOrientation = options.sideOrientation;\n const vertexData = createSphere(options);\n vertexData.applyToMesh(sphere, options.updatable);\n return sphere;\n function createSphere(options) {\n const segments = options.segments || 32;\n const diameterX = options.diameterX || options.diameter || 1;\n const diameterY = options.diameterY || options.diameter || 1;\n const diameterZ = options.diameterZ || options.diameter || 1;\n const arc = options.arc && (options.arc <= 0 || options.arc > 1) ? 1.0 : options.arc || 1.0;\n const sliceStart = options.sliceStart && (options.sliceStart < 0 || options.sliceStart > 1) ? 0 : options.sliceStart || 0;\n let slice = options.slice && options.slice <= 0 ? 1.0 : options.slice || 1.0;\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\n if (slice + sliceStart > 1) {\n slice = 1 - sliceStart;\n }\n const radius = new Vector3(diameterX / 2, diameterY / 2, diameterZ / 2);\n const totalZRotationSteps = 2 + segments;\n const totalYRotationSegments = 2 * totalZRotationSteps;\n const totalYRotationSteps = totalYRotationSegments + 1;\n const indices = [];\n const positions = [];\n const normals = [];\n const uvs = [];\n for (let zRotationStep = 0; zRotationStep <= totalZRotationSteps; zRotationStep++) {\n const normalizedZ = zRotationStep / totalZRotationSteps;\n const angleZ = normalizedZ * Math.PI * slice + sliceStart * Math.PI;\n for (let yRotationStep = 0; yRotationStep < totalYRotationSteps; yRotationStep++) {\n const normalizedY = yRotationStep / totalYRotationSegments;\n const angleY = normalizedY * Math.PI * 2 * arc;\n const rotationZ = Matrix.RotationZ(-angleZ);\n const rotationY = Matrix.RotationY(angleY);\n const afterRotZ = Vector3.TransformCoordinates(Vector3.Up(), rotationZ);\n const complete = Vector3.TransformCoordinates(afterRotZ, rotationY);\n const vertex = complete.multiply(radius);\n const normal = complete.divide(radius).normalize();\n positions.push(vertex.x, vertex.y, vertex.z);\n normals.push(normal.x, normal.y, normal.z);\n uvs.push(normalizedY, 1 - normalizedZ);\n if (yRotationStep < totalYRotationSegments || arc === 1) {\n const nextYStep = (yRotationStep + 1) % totalYRotationSteps;\n if (sliceStart === 0 && zRotationStep === 0) {\n indices.push(0); // C\n indices.push(nextYStep); // B\n indices.push(yRotationStep); // A\n }\n else if (zRotationStep > 0) {\n indices.push((zRotationStep - 1) * totalYRotationSteps + yRotationStep); // C\n indices.push(zRotationStep * totalYRotationSteps + nextYStep); // B\n indices.push(zRotationStep * totalYRotationSteps + yRotationStep); // A\n indices.push((zRotationStep - 1) * totalYRotationSteps + yRotationStep); // C\n indices.push((zRotationStep - 1) * totalYRotationSteps + nextYStep); // D\n indices.push(zRotationStep * totalYRotationSteps + nextYStep); // B\n }\n }\n }\n }\n // Sides\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\n // Result\n const vertexData = new VertexData();\n vertexData.indices = indices;\n vertexData.positions = positions;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n return vertexData;\n }\n }\n }\n\n /**\n * Class containing static functions to help procedurally build meshes\n */\n class TorusBuilder {\n /**\n * Creates a torus mesh\n * * The parameter `diameter` sets the diameter size (float) of the torus (default 1)\n * * The parameter `thickness` sets the diameter size of the tube of the torus (float, default 0.5)\n * * The parameter `tessellation` sets the number of torus sides (postive integer, default 16)\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/babylon101/discover_basic_elements#side-orientation\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param scene defines the hosting scene\n * @returns the torus mesh\n * @see https://doc.babylonjs.com/how_to/set_shapes#torus\n */\n static CreateTorus(name, options, scene) {\n const torus = new Mesh(name, scene);\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n torus._originalBuilderSideOrientation = options.sideOrientation;\n const vertexData = createTorus(options);\n vertexData.applyToMesh(torus, options.updatable);\n return torus;\n function createTorus(options) {\n const indices = [];\n const positions = [];\n const normals = [];\n const uvs = [];\n const diameter = options.diameter || 1;\n const thickness = options.thickness || 0.5;\n const tessellation = options.tessellation || 12;\n const arc = options.arc && (options.arc <= 0 || options.arc > 1) ? 1.0 : options.arc || 1.0;\n let slice = options.slice && options.slice <= 0 ? 1.0 : options.slice || 1.0;\n const sliceStart = options.sliceStart && (options.sliceStart < 0 || options.sliceStart > 1) ? 0 : options.sliceStart || 0;\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\n if (slice + sliceStart > 1) {\n slice = 1 - sliceStart;\n }\n const outerSegments = Math.ceil(tessellation * arc * Math.PI); // Need more segments for the torus circumference\n const innerSegments = Math.ceil(tessellation * slice);\n const closedRing = arc === 1; // handle float errors\n const closedTube = slice === 1;\n const lastOuterRing = outerSegments + 1;\n const lastTubeSection = innerSegments + 1;\n for (let i = 0; i < lastOuterRing; i++) {\n const u = i / outerSegments;\n const outerAngle = u * (arc * Math.PI * 2) - Math.PI / 2;\n const transform = Matrix.Translation(diameter / 2, 0, 0).multiply(Matrix.RotationY(outerAngle));\n for (let j = 0; j < lastTubeSection; j++) {\n const v = 1 - j / innerSegments;\n const innerAngle = (j / innerSegments) * (Math.PI * 2 * slice) + Math.PI * 2 * sliceStart;\n const dx = Math.cos(innerAngle);\n const dy = Math.sin(innerAngle);\n // Create a vertex.\n let normal = new Vector3(dx, dy, 0);\n let position = normal.scale(thickness / 2);\n const textureCoordinate = new Vector2(u, v);\n position = Vector3.TransformCoordinates(position, transform);\n normal = Vector3.TransformNormal(normal, transform);\n positions.push(position.x, position.y, position.z);\n normals.push(normal.x, normal.y, normal.z);\n uvs.push(textureCoordinate.x, textureCoordinate.y);\n if ((i < outerSegments || closedRing) && (j < innerSegments || closedTube)) {\n // And create indices for two triangles.\n const nextI = (i + 1) % lastOuterRing;\n const nextJ = (j + 1) % lastTubeSection;\n indices.push(i * lastTubeSection + j); // A\n indices.push(i * lastTubeSection + nextJ); // B\n indices.push(nextI * lastTubeSection + j); // C\n indices.push(i * lastTubeSection + nextJ); // B\n indices.push(nextI * lastTubeSection + nextJ); // D\n indices.push(nextI * lastTubeSection + j); // C\n }\n }\n }\n // Sides\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\n // Result\n const vertexData = new VertexData();\n vertexData.indices = indices;\n vertexData.positions = positions;\n vertexData.normals = normals;\n vertexData.uvs = uvs;\n return vertexData;\n }\n }\n }\n\n function parseSketchGeometry(vertexData) {\n const segments = [];\n const holes = new Map();\n if (vertexData && vertexData.positions && vertexData.paths) {\n const paths = vertexData.paths;\n const holeIndices = new Map();\n if (vertexData.holes) {\n for (let i = 0; i < vertexData.holes.length; i += 2) {\n holeIndices.set(vertexData.holes[i + 1], vertexData.holes[i]);\n }\n }\n let i = 0;\n let currentHole = null;\n while (i < paths.length) {\n const positions = vertexData.positions;\n const segmentPoints = [];\n currentHole = holeIndices.has(i) ? holeIndices.get(i) : null;\n for (; i < paths.length; i += 2) {\n segmentPoints.push(Vector3.FromArray(positions, paths[i] * 3));\n if (paths[i + 2] !== paths[i + 1]) {\n segmentPoints.push(Vector3.FromArray(positions, paths[i + 1] * 3));\n i += 2;\n break;\n }\n }\n if (segmentPoints.length > 1) {\n if (currentHole !== null) {\n let currentHolePoints = holes.get(currentHole);\n if (!currentHolePoints) {\n currentHolePoints = [];\n holes.set(currentHole, currentHolePoints);\n }\n currentHolePoints.push(segmentPoints);\n }\n else {\n segments.push(segmentPoints);\n }\n }\n }\n }\n return { segments, holes };\n }\n function createSketchPointForEditor(manager, path, insertIndex) {\n if (!path._parent) {\n throw new Error('Cannot create a sketch point for a path that has no parent sketch');\n }\n const sketch = path._parent;\n const c = manager.create(Token.SketchControlPoint);\n c.name = 'Control Point';\n while (path.controlPoints.find(ec => ec.name === c.name)) {\n c.name = appendOrIncrementNumber(c.name);\n }\n let previousControlPoint;\n if (!insertIndex) {\n // If insert index is zero or undefined, determine the previous control point and use its position for the new control point\n if (path.controlPoints.length > 0 && insertIndex === undefined) {\n previousControlPoint = path.controlPoints[path.controlPoints.length - 1];\n }\n else {\n const currentPathIndex = sketch.paths.findIndex(p => p._dynamicId === path._dynamicId);\n if (currentPathIndex > 0) {\n const previousPath = sketch.paths[currentPathIndex - 1];\n if (previousPath.controlPoints.length > 0) {\n previousControlPoint = previousPath.controlPoints[previousPath.controlPoints.length - 1];\n }\n }\n }\n }\n else {\n previousControlPoint = path.controlPoints[insertIndex];\n }\n const sketchNormal = getSketchNormalVector3(sketch.normal);\n // determine the axis making up the sketch plane so we can make sure the direction vectors are on the sketch plane\n let axis1;\n if (sketchNormal.equalsWithEpsilon(Vector3.Forward()) || sketchNormal.equalsWithEpsilon(Vector3.Backward())) {\n axis1 = Vector3.Up();\n }\n else {\n axis1 = Vector3.Cross(sketchNormal, Vector3.Forward());\n }\n const axis2 = Vector3.Cross(sketchNormal, axis1);\n if (previousControlPoint) {\n // if therre is a previous control point, project its direction onto the sketch plane\n c.position = previousControlPoint.position.clone();\n const rawDirection = previousControlPoint.direction.toVec3();\n const projectedDirection = axis1\n .scale(Vector3.Dot(axis1, rawDirection))\n .add(axis2.scale(Vector3.Dot(axis2, rawDirection)));\n c.direction = exports.KbVector.FromVec3(projectedDirection);\n }\n else {\n // if this is the first control point, just pick a direction vector on the sketch plane\n c.direction = exports.KbVector.FromVec3(axis1);\n }\n c.direction2 = c.direction.scale(-1);\n if (insertIndex === undefined) {\n path.controlPoints.push(c);\n }\n else {\n path.controlPoints.splice(insertIndex, 0, c);\n }\n for (const eC of path.controlPoints) {\n eC._expanded = false;\n }\n c._visible = true;\n c._expanded = true;\n return c;\n }\n function getSketchNormalVector3(normal) {\n let planeNormal;\n switch (normal) {\n case exports.eAxis.x:\n planeNormal = Vector3.Right();\n break;\n case exports.eAxis.y:\n planeNormal = Vector3.Up();\n break;\n case exports.eAxis.z:\n planeNormal = Vector3.Forward();\n break;\n default:\n planeNormal = normal.toVec3();\n }\n planeNormal.normalize();\n return planeNormal;\n }\n function projectOnNodeAxis(node, x, y) {\n switch (node.normal) {\n case exports.eAxis.y:\n return new exports.KbVector(x, 0, -y);\n case exports.eAxis.z:\n return new exports.KbVector(-x, y, 0);\n default:\n return new exports.KbVector(0, y, x);\n }\n }\n function axisToVector3(axis) {\n switch (axis) {\n case exports.eAxis.y:\n return Vector3.Up();\n case exports.eAxis.z:\n return Vector3.Forward();\n default:\n return Vector3.Right();\n }\n }\n /**\n * returns the vertices in the order of positionVectors in sequence, followed by holePositions in sequence!\n * Indices will have to be recalculated if the vertex data is replaced with the results of this function.\n */\n function fillSketchPolygon(positionVectors, flipNormals, sketchNativeNormal, holePositions) {\n const sketchNormal = getPolygonNormal(positionVectors);\n if (flipNormals) {\n sketchNormal.scaleInPlace(-1);\n }\n // const geometryNormal = sketchNormal.scale(-1);\n const segmentPositions = [];\n let segmentIndices = [];\n for (let i = 0; i < positionVectors.length; i++) {\n const point = positionVectors[i];\n segmentPositions.push(point.x, point.y, point.z);\n }\n const holes = [];\n if (holePositions) {\n for (let i = 0; i < holePositions.length; i++) {\n const hole = holePositions[i];\n holes.push(segmentPositions.length / 3);\n for (let i = 0; i < hole.length; i++) {\n const point = hole[i];\n segmentPositions.push(point.x, point.y, point.z);\n }\n }\n }\n if (Math.abs(Vector3.Dot(sketchNormal, Vector3.UpReadOnly)) > 0.95) {\n const flatPositions = new Float32Array((segmentPositions.length * 2) / 3);\n const xAxis = Vector3.Cross(sketchNormal, new Vector3(0, 0, 1));\n const yAxis = Vector3.Cross(sketchNormal, xAxis);\n const working = new Vector3();\n for (let i = 0; i < segmentPositions.length; i += 3) {\n working.set(segmentPositions[i], segmentPositions[i + 1], segmentPositions[i + 2]);\n flatPositions.set([Vector3.Dot(working, xAxis), Vector3.Dot(working, yAxis)], (i * 2) / 3);\n }\n const faceNormal = calculateFaceNormalFromFloats(flatPositions, 2);\n if (Vector3.Dot(faceNormal, Vector3.RightHandedForwardReadOnly) < 0) {\n for (let i = 0; i < flatPositions.length; i += 2) {\n const tmp = flatPositions[i];\n flatPositions[i] = flatPositions[i + 1];\n flatPositions[i + 1] = tmp;\n }\n // geometryNormal.scaleInPlace(-1);\n }\n segmentIndices = earcut$1(flatPositions, holes, 2);\n }\n else {\n segmentIndices = earcut$1(segmentPositions, holes, 3);\n // Flip the geometry if needed so that it faces the same direction as the faces because earcut isn't aware of the 3d space\n const faceNormal = calculateFaceNormal(positionVectors);\n // If the rotation handedness is opposite the primary sketch vector, then flip the sketch path\n if (Vector3.Dot(faceNormal, sketchNativeNormal) < 0) {\n for (let i = 0; i < segmentIndices.length; i += 3) {\n const tmp = segmentIndices[i];\n segmentIndices[i] = segmentIndices[i + 2];\n segmentIndices[i + 2] = tmp;\n }\n // geometryNormal.scaleInPlace(-1);\n }\n }\n const segmentNormals = [];\n const geometryNormal = calculateFaceNormal(positionVectors);\n if (flipNormals) {\n geometryNormal.scaleInPlace(-1);\n }\n let totalVectorCount = positionVectors.length;\n if (holePositions) {\n totalVectorCount += holePositions.flat().length;\n }\n for (let i = 0; i < totalVectorCount; i++) {\n segmentNormals.push(geometryNormal.x, geometryNormal.y, geometryNormal.z);\n }\n const segmentUvs = [];\n for (let i = 0; i < totalVectorCount; i++) {\n segmentUvs.push(0, 0);\n }\n return {\n normals: segmentNormals,\n positions: segmentPositions,\n indices: segmentIndices,\n uvs: segmentUvs,\n };\n }\n function calculateFaceNormal(positionVectors) {\n const v1 = Vector3.Zero();\n const v2 = Vector3.Zero();\n const barycenter = Vector3.Zero();\n for (let i = 0; i < positionVectors.length; i++) {\n barycenter.addInPlace(positionVectors[i]);\n }\n barycenter.scaleInPlace(1 / positionVectors.length);\n const pathHandedness = Vector3.Zero();\n if (positionVectors.length > 1) {\n for (let iP = 0; iP < positionVectors.length - 1; iP++) {\n const point1 = positionVectors[iP];\n v1.copyFromFloats(point1.x, point1.y, point1.z);\n v1.subtractInPlace(barycenter);\n const point2 = positionVectors[iP + 1];\n v2.copyFromFloats(point2.x, point2.y, point2.z);\n v2.subtractInPlace(barycenter);\n pathHandedness.addInPlace(Vector3.Cross(v1, v2));\n }\n return pathHandedness.normalize();\n }\n return Vector3.Zero();\n }\n function calculateFaceNormalFromFloats(points, dim) {\n const v1 = Vector3.Zero();\n const v2 = Vector3.Zero();\n const barycenter = Vector3.Zero();\n for (let i = 0; i < points.length; i += dim) {\n barycenter.addInPlaceFromFloats(points[i], points[i + 1], dim === 3 ? points[i + 2] : 0);\n }\n barycenter.scaleInPlace(3 / points.length);\n const pathHandedness = Vector3.Zero();\n if (points.length > 3) {\n for (let i = 0; i < points.length - dim; i += dim) {\n v1.copyFromFloats(points[i], points[i + 1], dim === 3 ? points[i + 2] : 0);\n v1.subtractInPlace(barycenter);\n v2.copyFromFloats(points[i + dim], points[i + dim + 1], dim === 3 ? points[i + dim + 2] : 0);\n v2.subtractInPlace(barycenter);\n pathHandedness.addInPlace(Vector3.Cross(v1, v2));\n }\n return pathHandedness.normalize();\n }\n return Vector3.Zero();\n }\n\n /**\n * https://opentype.js.org v1.3.4 | (c) Frederik De Bleser and other contributors | MIT License | Uses tiny-inflate by Devon Govett and string.prototype.codepointat polyfill by Mathias Bynens\n */\n\n /*! https://mths.be/codepointat v0.2.0 by @mathias */\n if (!String.prototype.codePointAt) {\n \t(function() {\n \t\tvar defineProperty = (function() {\n \t\t\t// IE 8 only supports `Object.defineProperty` on DOM elements\n \t\t\ttry {\n \t\t\t\tvar object = {};\n \t\t\t\tvar $defineProperty = Object.defineProperty;\n \t\t\t\tvar result = $defineProperty(object, object, object) && $defineProperty;\n \t\t\t} catch(error) {}\n \t\t\treturn result;\n \t\t}());\n \t\tvar codePointAt = function(position) {\n \t\t\tif (this == null) {\n \t\t\t\tthrow TypeError();\n \t\t\t}\n \t\t\tvar string = String(this);\n \t\t\tvar size = string.length;\n \t\t\t// `ToInteger`\n \t\t\tvar index = position ? Number(position) : 0;\n \t\t\tif (index != index) { // better `isNaN`\n \t\t\t\tindex = 0;\n \t\t\t}\n \t\t\t// Account for out-of-bounds indices:\n \t\t\tif (index < 0 || index >= size) {\n \t\t\t\treturn undefined;\n \t\t\t}\n \t\t\t// Get the first code unit\n \t\t\tvar first = string.charCodeAt(index);\n \t\t\tvar second;\n \t\t\tif ( // check if it’s the start of a surrogate pair\n \t\t\t\tfirst >= 0xD800 && first <= 0xDBFF && // high surrogate\n \t\t\t\tsize > index + 1 // there is a next code unit\n \t\t\t) {\n \t\t\t\tsecond = string.charCodeAt(index + 1);\n \t\t\t\tif (second >= 0xDC00 && second <= 0xDFFF) { // low surrogate\n \t\t\t\t\t// https://mathiasbynens.be/notes/javascript-encoding#surrogate-formulae\n \t\t\t\t\treturn (first - 0xD800) * 0x400 + second - 0xDC00 + 0x10000;\n \t\t\t\t}\n \t\t\t}\n \t\t\treturn first;\n \t\t};\n \t\tif (defineProperty) {\n \t\t\tdefineProperty(String.prototype, 'codePointAt', {\n \t\t\t\t'value': codePointAt,\n \t\t\t\t'configurable': true,\n \t\t\t\t'writable': true\n \t\t\t});\n \t\t} else {\n \t\t\tString.prototype.codePointAt = codePointAt;\n \t\t}\n \t}());\n }\n\n var TINF_OK = 0;\n var TINF_DATA_ERROR = -3;\n\n function Tree() {\n this.table = new Uint16Array(16); /* table of code length counts */\n this.trans = new Uint16Array(288); /* code -> symbol translation table */\n }\n\n function Data(source, dest) {\n this.source = source;\n this.sourceIndex = 0;\n this.tag = 0;\n this.bitcount = 0;\n \n this.dest = dest;\n this.destLen = 0;\n \n this.ltree = new Tree(); /* dynamic length/symbol tree */\n this.dtree = new Tree(); /* dynamic distance tree */\n }\n\n /* --------------------------------------------------- *\n * -- uninitialized global data (static structures) -- *\n * --------------------------------------------------- */\n\n var sltree = new Tree();\n var sdtree = new Tree();\n\n /* extra bits and base tables for length codes */\n var length_bits = new Uint8Array(30);\n var length_base = new Uint16Array(30);\n\n /* extra bits and base tables for distance codes */\n var dist_bits = new Uint8Array(30);\n var dist_base = new Uint16Array(30);\n\n /* special ordering of code length codes */\n var clcidx = new Uint8Array([\n 16, 17, 18, 0, 8, 7, 9, 6,\n 10, 5, 11, 4, 12, 3, 13, 2,\n 14, 1, 15\n ]);\n\n /* used by tinf_decode_trees, avoids allocations every call */\n var code_tree = new Tree();\n var lengths = new Uint8Array(288 + 32);\n\n /* ----------------------- *\n * -- utility functions -- *\n * ----------------------- */\n\n /* build extra bits and base tables */\n function tinf_build_bits_base(bits, base, delta, first) {\n var i, sum;\n\n /* build bits table */\n for (i = 0; i < delta; ++i) { bits[i] = 0; }\n for (i = 0; i < 30 - delta; ++i) { bits[i + delta] = i / delta | 0; }\n\n /* build base table */\n for (sum = first, i = 0; i < 30; ++i) {\n base[i] = sum;\n sum += 1 << bits[i];\n }\n }\n\n /* build the fixed huffman trees */\n function tinf_build_fixed_trees(lt, dt) {\n var i;\n\n /* build fixed length tree */\n for (i = 0; i < 7; ++i) { lt.table[i] = 0; }\n\n lt.table[7] = 24;\n lt.table[8] = 152;\n lt.table[9] = 112;\n\n for (i = 0; i < 24; ++i) { lt.trans[i] = 256 + i; }\n for (i = 0; i < 144; ++i) { lt.trans[24 + i] = i; }\n for (i = 0; i < 8; ++i) { lt.trans[24 + 144 + i] = 280 + i; }\n for (i = 0; i < 112; ++i) { lt.trans[24 + 144 + 8 + i] = 144 + i; }\n\n /* build fixed distance tree */\n for (i = 0; i < 5; ++i) { dt.table[i] = 0; }\n\n dt.table[5] = 32;\n\n for (i = 0; i < 32; ++i) { dt.trans[i] = i; }\n }\n\n /* given an array of code lengths, build a tree */\n var offs = new Uint16Array(16);\n\n function tinf_build_tree(t, lengths, off, num) {\n var i, sum;\n\n /* clear code length count table */\n for (i = 0; i < 16; ++i) { t.table[i] = 0; }\n\n /* scan symbol lengths, and sum code length counts */\n for (i = 0; i < num; ++i) { t.table[lengths[off + i]]++; }\n\n t.table[0] = 0;\n\n /* compute offset table for distribution sort */\n for (sum = 0, i = 0; i < 16; ++i) {\n offs[i] = sum;\n sum += t.table[i];\n }\n\n /* create code->symbol translation table (symbols sorted by code) */\n for (i = 0; i < num; ++i) {\n if (lengths[off + i]) { t.trans[offs[lengths[off + i]]++] = i; }\n }\n }\n\n /* ---------------------- *\n * -- decode functions -- *\n * ---------------------- */\n\n /* get one bit from source stream */\n function tinf_getbit(d) {\n /* check if tag is empty */\n if (!d.bitcount--) {\n /* load next tag */\n d.tag = d.source[d.sourceIndex++];\n d.bitcount = 7;\n }\n\n /* shift bit out of tag */\n var bit = d.tag & 1;\n d.tag >>>= 1;\n\n return bit;\n }\n\n /* read a num bit value from a stream and add base */\n function tinf_read_bits(d, num, base) {\n if (!num)\n { return base; }\n\n while (d.bitcount < 24) {\n d.tag |= d.source[d.sourceIndex++] << d.bitcount;\n d.bitcount += 8;\n }\n\n var val = d.tag & (0xffff >>> (16 - num));\n d.tag >>>= num;\n d.bitcount -= num;\n return val + base;\n }\n\n /* given a data stream and a tree, decode a symbol */\n function tinf_decode_symbol(d, t) {\n while (d.bitcount < 24) {\n d.tag |= d.source[d.sourceIndex++] << d.bitcount;\n d.bitcount += 8;\n }\n \n var sum = 0, cur = 0, len = 0;\n var tag = d.tag;\n\n /* get more bits while code value is above sum */\n do {\n cur = 2 * cur + (tag & 1);\n tag >>>= 1;\n ++len;\n\n sum += t.table[len];\n cur -= t.table[len];\n } while (cur >= 0);\n \n d.tag = tag;\n d.bitcount -= len;\n\n return t.trans[sum + cur];\n }\n\n /* given a data stream, decode dynamic trees from it */\n function tinf_decode_trees(d, lt, dt) {\n var hlit, hdist, hclen;\n var i, num, length;\n\n /* get 5 bits HLIT (257-286) */\n hlit = tinf_read_bits(d, 5, 257);\n\n /* get 5 bits HDIST (1-32) */\n hdist = tinf_read_bits(d, 5, 1);\n\n /* get 4 bits HCLEN (4-19) */\n hclen = tinf_read_bits(d, 4, 4);\n\n for (i = 0; i < 19; ++i) { lengths[i] = 0; }\n\n /* read code lengths for code length alphabet */\n for (i = 0; i < hclen; ++i) {\n /* get 3 bits code length (0-7) */\n var clen = tinf_read_bits(d, 3, 0);\n lengths[clcidx[i]] = clen;\n }\n\n /* build code length tree */\n tinf_build_tree(code_tree, lengths, 0, 19);\n\n /* decode code lengths for the dynamic trees */\n for (num = 0; num < hlit + hdist;) {\n var sym = tinf_decode_symbol(d, code_tree);\n\n switch (sym) {\n case 16:\n /* copy previous code length 3-6 times (read 2 bits) */\n var prev = lengths[num - 1];\n for (length = tinf_read_bits(d, 2, 3); length; --length) {\n lengths[num++] = prev;\n }\n break;\n case 17:\n /* repeat code length 0 for 3-10 times (read 3 bits) */\n for (length = tinf_read_bits(d, 3, 3); length; --length) {\n lengths[num++] = 0;\n }\n break;\n case 18:\n /* repeat code length 0 for 11-138 times (read 7 bits) */\n for (length = tinf_read_bits(d, 7, 11); length; --length) {\n lengths[num++] = 0;\n }\n break;\n default:\n /* values 0-15 represent the actual code lengths */\n lengths[num++] = sym;\n break;\n }\n }\n\n /* build dynamic trees */\n tinf_build_tree(lt, lengths, 0, hlit);\n tinf_build_tree(dt, lengths, hlit, hdist);\n }\n\n /* ----------------------------- *\n * -- block inflate functions -- *\n * ----------------------------- */\n\n /* given a stream and two trees, inflate a block of data */\n function tinf_inflate_block_data(d, lt, dt) {\n while (1) {\n var sym = tinf_decode_symbol(d, lt);\n\n /* check for end of block */\n if (sym === 256) {\n return TINF_OK;\n }\n\n if (sym < 256) {\n d.dest[d.destLen++] = sym;\n } else {\n var length, dist, offs;\n var i;\n\n sym -= 257;\n\n /* possibly get more bits from length code */\n length = tinf_read_bits(d, length_bits[sym], length_base[sym]);\n\n dist = tinf_decode_symbol(d, dt);\n\n /* possibly get more bits from distance code */\n offs = d.destLen - tinf_read_bits(d, dist_bits[dist], dist_base[dist]);\n\n /* copy match */\n for (i = offs; i < offs + length; ++i) {\n d.dest[d.destLen++] = d.dest[i];\n }\n }\n }\n }\n\n /* inflate an uncompressed block of data */\n function tinf_inflate_uncompressed_block(d) {\n var length, invlength;\n var i;\n \n /* unread from bitbuffer */\n while (d.bitcount > 8) {\n d.sourceIndex--;\n d.bitcount -= 8;\n }\n\n /* get length */\n length = d.source[d.sourceIndex + 1];\n length = 256 * length + d.source[d.sourceIndex];\n\n /* get one's complement of length */\n invlength = d.source[d.sourceIndex + 3];\n invlength = 256 * invlength + d.source[d.sourceIndex + 2];\n\n /* check length */\n if (length !== (~invlength & 0x0000ffff))\n { return TINF_DATA_ERROR; }\n\n d.sourceIndex += 4;\n\n /* copy block */\n for (i = length; i; --i)\n { d.dest[d.destLen++] = d.source[d.sourceIndex++]; }\n\n /* make sure we start next block on a byte boundary */\n d.bitcount = 0;\n\n return TINF_OK;\n }\n\n /* inflate stream from source to dest */\n function tinf_uncompress(source, dest) {\n var d = new Data(source, dest);\n var bfinal, btype, res;\n\n do {\n /* read final block flag */\n bfinal = tinf_getbit(d);\n\n /* read block type (2 bits) */\n btype = tinf_read_bits(d, 2, 0);\n\n /* decompress block */\n switch (btype) {\n case 0:\n /* decompress uncompressed block */\n res = tinf_inflate_uncompressed_block(d);\n break;\n case 1:\n /* decompress block with fixed huffman trees */\n res = tinf_inflate_block_data(d, sltree, sdtree);\n break;\n case 2:\n /* decompress block with dynamic huffman trees */\n tinf_decode_trees(d, d.ltree, d.dtree);\n res = tinf_inflate_block_data(d, d.ltree, d.dtree);\n break;\n default:\n res = TINF_DATA_ERROR;\n }\n\n if (res !== TINF_OK)\n { throw new Error('Data error'); }\n\n } while (!bfinal);\n\n if (d.destLen < d.dest.length) {\n if (typeof d.dest.slice === 'function')\n { return d.dest.slice(0, d.destLen); }\n else\n { return d.dest.subarray(0, d.destLen); }\n }\n \n return d.dest;\n }\n\n /* -------------------- *\n * -- initialization -- *\n * -------------------- */\n\n /* build fixed huffman trees */\n tinf_build_fixed_trees(sltree, sdtree);\n\n /* build extra bits and base tables */\n tinf_build_bits_base(length_bits, length_base, 4, 3);\n tinf_build_bits_base(dist_bits, dist_base, 2, 1);\n\n /* fix a special case */\n length_bits[28] = 0;\n length_base[28] = 258;\n\n var tinyInflate = tinf_uncompress;\n\n // The Bounding Box object\n\n function derive(v0, v1, v2, v3, t) {\n return Math.pow(1 - t, 3) * v0 +\n 3 * Math.pow(1 - t, 2) * t * v1 +\n 3 * (1 - t) * Math.pow(t, 2) * v2 +\n Math.pow(t, 3) * v3;\n }\n /**\n * A bounding box is an enclosing box that describes the smallest measure within which all the points lie.\n * It is used to calculate the bounding box of a glyph or text path.\n *\n * On initialization, x1/y1/x2/y2 will be NaN. Check if the bounding box is empty using `isEmpty()`.\n *\n * @exports opentype.BoundingBox\n * @class\n * @constructor\n */\n function BoundingBox$1() {\n this.x1 = Number.NaN;\n this.y1 = Number.NaN;\n this.x2 = Number.NaN;\n this.y2 = Number.NaN;\n }\n\n /**\n * Returns true if the bounding box is empty, that is, no points have been added to the box yet.\n */\n BoundingBox$1.prototype.isEmpty = function() {\n return isNaN(this.x1) || isNaN(this.y1) || isNaN(this.x2) || isNaN(this.y2);\n };\n\n /**\n * Add the point to the bounding box.\n * The x1/y1/x2/y2 coordinates of the bounding box will now encompass the given point.\n * @param {number} x - The X coordinate of the point.\n * @param {number} y - The Y coordinate of the point.\n */\n BoundingBox$1.prototype.addPoint = function(x, y) {\n if (typeof x === 'number') {\n if (isNaN(this.x1) || isNaN(this.x2)) {\n this.x1 = x;\n this.x2 = x;\n }\n if (x < this.x1) {\n this.x1 = x;\n }\n if (x > this.x2) {\n this.x2 = x;\n }\n }\n if (typeof y === 'number') {\n if (isNaN(this.y1) || isNaN(this.y2)) {\n this.y1 = y;\n this.y2 = y;\n }\n if (y < this.y1) {\n this.y1 = y;\n }\n if (y > this.y2) {\n this.y2 = y;\n }\n }\n };\n\n /**\n * Add a X coordinate to the bounding box.\n * This extends the bounding box to include the X coordinate.\n * This function is used internally inside of addBezier.\n * @param {number} x - The X coordinate of the point.\n */\n BoundingBox$1.prototype.addX = function(x) {\n this.addPoint(x, null);\n };\n\n /**\n * Add a Y coordinate to the bounding box.\n * This extends the bounding box to include the Y coordinate.\n * This function is used internally inside of addBezier.\n * @param {number} y - The Y coordinate of the point.\n */\n BoundingBox$1.prototype.addY = function(y) {\n this.addPoint(null, y);\n };\n\n /**\n * Add a Bézier curve to the bounding box.\n * This extends the bounding box to include the entire Bézier.\n * @param {number} x0 - The starting X coordinate.\n * @param {number} y0 - The starting Y coordinate.\n * @param {number} x1 - The X coordinate of the first control point.\n * @param {number} y1 - The Y coordinate of the first control point.\n * @param {number} x2 - The X coordinate of the second control point.\n * @param {number} y2 - The Y coordinate of the second control point.\n * @param {number} x - The ending X coordinate.\n * @param {number} y - The ending Y coordinate.\n */\n BoundingBox$1.prototype.addBezier = function(x0, y0, x1, y1, x2, y2, x, y) {\n // This code is based on http://nishiohirokazu.blogspot.com/2009/06/how-to-calculate-bezier-curves-bounding.html\n // and https://github.com/icons8/svg-path-bounding-box\n\n var p0 = [x0, y0];\n var p1 = [x1, y1];\n var p2 = [x2, y2];\n var p3 = [x, y];\n\n this.addPoint(x0, y0);\n this.addPoint(x, y);\n\n for (var i = 0; i <= 1; i++) {\n var b = 6 * p0[i] - 12 * p1[i] + 6 * p2[i];\n var a = -3 * p0[i] + 9 * p1[i] - 9 * p2[i] + 3 * p3[i];\n var c = 3 * p1[i] - 3 * p0[i];\n\n if (a === 0) {\n if (b === 0) { continue; }\n var t = -c / b;\n if (0 < t && t < 1) {\n if (i === 0) { this.addX(derive(p0[i], p1[i], p2[i], p3[i], t)); }\n if (i === 1) { this.addY(derive(p0[i], p1[i], p2[i], p3[i], t)); }\n }\n continue;\n }\n\n var b2ac = Math.pow(b, 2) - 4 * c * a;\n if (b2ac < 0) { continue; }\n var t1 = (-b + Math.sqrt(b2ac)) / (2 * a);\n if (0 < t1 && t1 < 1) {\n if (i === 0) { this.addX(derive(p0[i], p1[i], p2[i], p3[i], t1)); }\n if (i === 1) { this.addY(derive(p0[i], p1[i], p2[i], p3[i], t1)); }\n }\n var t2 = (-b - Math.sqrt(b2ac)) / (2 * a);\n if (0 < t2 && t2 < 1) {\n if (i === 0) { this.addX(derive(p0[i], p1[i], p2[i], p3[i], t2)); }\n if (i === 1) { this.addY(derive(p0[i], p1[i], p2[i], p3[i], t2)); }\n }\n }\n };\n\n /**\n * Add a quadratic curve to the bounding box.\n * This extends the bounding box to include the entire quadratic curve.\n * @param {number} x0 - The starting X coordinate.\n * @param {number} y0 - The starting Y coordinate.\n * @param {number} x1 - The X coordinate of the control point.\n * @param {number} y1 - The Y coordinate of the control point.\n * @param {number} x - The ending X coordinate.\n * @param {number} y - The ending Y coordinate.\n */\n BoundingBox$1.prototype.addQuad = function(x0, y0, x1, y1, x, y) {\n var cp1x = x0 + 2 / 3 * (x1 - x0);\n var cp1y = y0 + 2 / 3 * (y1 - y0);\n var cp2x = cp1x + 1 / 3 * (x - x0);\n var cp2y = cp1y + 1 / 3 * (y - y0);\n this.addBezier(x0, y0, cp1x, cp1y, cp2x, cp2y, x, y);\n };\n\n // Geometric objects\n\n /**\n * A bézier path containing a set of path commands similar to a SVG path.\n * Paths can be drawn on a context using `draw`.\n * @exports opentype.Path\n * @class\n * @constructor\n */\n function Path() {\n this.commands = [];\n this.fill = 'black';\n this.stroke = null;\n this.strokeWidth = 1;\n }\n\n /**\n * @param {number} x\n * @param {number} y\n */\n Path.prototype.moveTo = function(x, y) {\n this.commands.push({\n type: 'M',\n x: x,\n y: y\n });\n };\n\n /**\n * @param {number} x\n * @param {number} y\n */\n Path.prototype.lineTo = function(x, y) {\n this.commands.push({\n type: 'L',\n x: x,\n y: y\n });\n };\n\n /**\n * Draws cubic curve\n * @function\n * curveTo\n * @memberof opentype.Path.prototype\n * @param {number} x1 - x of control 1\n * @param {number} y1 - y of control 1\n * @param {number} x2 - x of control 2\n * @param {number} y2 - y of control 2\n * @param {number} x - x of path point\n * @param {number} y - y of path point\n */\n\n /**\n * Draws cubic curve\n * @function\n * bezierCurveTo\n * @memberof opentype.Path.prototype\n * @param {number} x1 - x of control 1\n * @param {number} y1 - y of control 1\n * @param {number} x2 - x of control 2\n * @param {number} y2 - y of control 2\n * @param {number} x - x of path point\n * @param {number} y - y of path point\n * @see curveTo\n */\n Path.prototype.curveTo = Path.prototype.bezierCurveTo = function(x1, y1, x2, y2, x, y) {\n this.commands.push({\n type: 'C',\n x1: x1,\n y1: y1,\n x2: x2,\n y2: y2,\n x: x,\n y: y\n });\n };\n\n /**\n * Draws quadratic curve\n * @function\n * quadraticCurveTo\n * @memberof opentype.Path.prototype\n * @param {number} x1 - x of control\n * @param {number} y1 - y of control\n * @param {number} x - x of path point\n * @param {number} y - y of path point\n */\n\n /**\n * Draws quadratic curve\n * @function\n * quadTo\n * @memberof opentype.Path.prototype\n * @param {number} x1 - x of control\n * @param {number} y1 - y of control\n * @param {number} x - x of path point\n * @param {number} y - y of path point\n */\n Path.prototype.quadTo = Path.prototype.quadraticCurveTo = function(x1, y1, x, y) {\n this.commands.push({\n type: 'Q',\n x1: x1,\n y1: y1,\n x: x,\n y: y\n });\n };\n\n /**\n * Closes the path\n * @function closePath\n * @memberof opentype.Path.prototype\n */\n\n /**\n * Close the path\n * @function close\n * @memberof opentype.Path.prototype\n */\n Path.prototype.close = Path.prototype.closePath = function() {\n this.commands.push({\n type: 'Z'\n });\n };\n\n /**\n * Add the given path or list of commands to the commands of this path.\n * @param {Array} pathOrCommands - another opentype.Path, an opentype.BoundingBox, or an array of commands.\n */\n Path.prototype.extend = function(pathOrCommands) {\n if (pathOrCommands.commands) {\n pathOrCommands = pathOrCommands.commands;\n } else if (pathOrCommands instanceof BoundingBox$1) {\n var box = pathOrCommands;\n this.moveTo(box.x1, box.y1);\n this.lineTo(box.x2, box.y1);\n this.lineTo(box.x2, box.y2);\n this.lineTo(box.x1, box.y2);\n this.close();\n return;\n }\n\n Array.prototype.push.apply(this.commands, pathOrCommands);\n };\n\n /**\n * Calculate the bounding box of the path.\n * @returns {opentype.BoundingBox}\n */\n Path.prototype.getBoundingBox = function() {\n var box = new BoundingBox$1();\n\n var startX = 0;\n var startY = 0;\n var prevX = 0;\n var prevY = 0;\n for (var i = 0; i < this.commands.length; i++) {\n var cmd = this.commands[i];\n switch (cmd.type) {\n case 'M':\n box.addPoint(cmd.x, cmd.y);\n startX = prevX = cmd.x;\n startY = prevY = cmd.y;\n break;\n case 'L':\n box.addPoint(cmd.x, cmd.y);\n prevX = cmd.x;\n prevY = cmd.y;\n break;\n case 'Q':\n box.addQuad(prevX, prevY, cmd.x1, cmd.y1, cmd.x, cmd.y);\n prevX = cmd.x;\n prevY = cmd.y;\n break;\n case 'C':\n box.addBezier(prevX, prevY, cmd.x1, cmd.y1, cmd.x2, cmd.y2, cmd.x, cmd.y);\n prevX = cmd.x;\n prevY = cmd.y;\n break;\n case 'Z':\n prevX = startX;\n prevY = startY;\n break;\n default:\n throw new Error('Unexpected path command ' + cmd.type);\n }\n }\n if (box.isEmpty()) {\n box.addPoint(0, 0);\n }\n return box;\n };\n\n /**\n * Draw the path to a 2D context.\n * @param {CanvasRenderingContext2D} ctx - A 2D drawing context.\n */\n Path.prototype.draw = function(ctx) {\n ctx.beginPath();\n for (var i = 0; i < this.commands.length; i += 1) {\n var cmd = this.commands[i];\n if (cmd.type === 'M') {\n ctx.moveTo(cmd.x, cmd.y);\n } else if (cmd.type === 'L') {\n ctx.lineTo(cmd.x, cmd.y);\n } else if (cmd.type === 'C') {\n ctx.bezierCurveTo(cmd.x1, cmd.y1, cmd.x2, cmd.y2, cmd.x, cmd.y);\n } else if (cmd.type === 'Q') {\n ctx.quadraticCurveTo(cmd.x1, cmd.y1, cmd.x, cmd.y);\n } else if (cmd.type === 'Z') {\n ctx.closePath();\n }\n }\n\n if (this.fill) {\n ctx.fillStyle = this.fill;\n ctx.fill();\n }\n\n if (this.stroke) {\n ctx.strokeStyle = this.stroke;\n ctx.lineWidth = this.strokeWidth;\n ctx.stroke();\n }\n };\n\n /**\n * Convert the Path to a string of path data instructions\n * See http://www.w3.org/TR/SVG/paths.html#PathData\n * @param {number} [decimalPlaces=2] - The amount of decimal places for floating-point values\n * @return {string}\n */\n Path.prototype.toPathData = function(decimalPlaces) {\n decimalPlaces = decimalPlaces !== undefined ? decimalPlaces : 2;\n\n function floatToString(v) {\n if (Math.round(v) === v) {\n return '' + Math.round(v);\n } else {\n return v.toFixed(decimalPlaces);\n }\n }\n\n function packValues() {\n var arguments$1 = arguments;\n\n var s = '';\n for (var i = 0; i < arguments.length; i += 1) {\n var v = arguments$1[i];\n if (v >= 0 && i > 0) {\n s += ' ';\n }\n\n s += floatToString(v);\n }\n\n return s;\n }\n\n var d = '';\n for (var i = 0; i < this.commands.length; i += 1) {\n var cmd = this.commands[i];\n if (cmd.type === 'M') {\n d += 'M' + packValues(cmd.x, cmd.y);\n } else if (cmd.type === 'L') {\n d += 'L' + packValues(cmd.x, cmd.y);\n } else if (cmd.type === 'C') {\n d += 'C' + packValues(cmd.x1, cmd.y1, cmd.x2, cmd.y2, cmd.x, cmd.y);\n } else if (cmd.type === 'Q') {\n d += 'Q' + packValues(cmd.x1, cmd.y1, cmd.x, cmd.y);\n } else if (cmd.type === 'Z') {\n d += 'Z';\n }\n }\n\n return d;\n };\n\n /**\n * Convert the path to an SVG element, as a string.\n * @param {number} [decimalPlaces=2] - The amount of decimal places for floating-point values\n * @return {string}\n */\n Path.prototype.toSVG = function(decimalPlaces) {\n var svg = '= 0 && v <= 255, 'Byte value should be between 0 and 255.');\n return [v];\n };\n /**\n * @constant\n * @type {number}\n */\n sizeOf.BYTE = constant(1);\n\n /**\n * Convert a 8-bit signed integer to a list of 1 byte.\n * @param {string}\n * @returns {Array}\n */\n encode.CHAR = function(v) {\n return [v.charCodeAt(0)];\n };\n\n /**\n * @constant\n * @type {number}\n */\n sizeOf.CHAR = constant(1);\n\n /**\n * Convert an ASCII string to a list of bytes.\n * @param {string}\n * @returns {Array}\n */\n encode.CHARARRAY = function(v) {\n if (typeof v === 'undefined') {\n v = '';\n console.warn('Undefined CHARARRAY encountered and treated as an empty string. This is probably caused by a missing glyph name.');\n }\n var b = [];\n for (var i = 0; i < v.length; i += 1) {\n b[i] = v.charCodeAt(i);\n }\n\n return b;\n };\n\n /**\n * @param {Array}\n * @returns {number}\n */\n sizeOf.CHARARRAY = function(v) {\n if (typeof v === 'undefined') {\n return 0;\n }\n return v.length;\n };\n\n /**\n * Convert a 16-bit unsigned integer to a list of 2 bytes.\n * @param {number}\n * @returns {Array}\n */\n encode.USHORT = function(v) {\n return [(v >> 8) & 0xFF, v & 0xFF];\n };\n\n /**\n * @constant\n * @type {number}\n */\n sizeOf.USHORT = constant(2);\n\n /**\n * Convert a 16-bit signed integer to a list of 2 bytes.\n * @param {number}\n * @returns {Array}\n */\n encode.SHORT = function(v) {\n // Two's complement\n if (v >= LIMIT16) {\n v = -(2 * LIMIT16 - v);\n }\n\n return [(v >> 8) & 0xFF, v & 0xFF];\n };\n\n /**\n * @constant\n * @type {number}\n */\n sizeOf.SHORT = constant(2);\n\n /**\n * Convert a 24-bit unsigned integer to a list of 3 bytes.\n * @param {number}\n * @returns {Array}\n */\n encode.UINT24 = function(v) {\n return [(v >> 16) & 0xFF, (v >> 8) & 0xFF, v & 0xFF];\n };\n\n /**\n * @constant\n * @type {number}\n */\n sizeOf.UINT24 = constant(3);\n\n /**\n * Convert a 32-bit unsigned integer to a list of 4 bytes.\n * @param {number}\n * @returns {Array}\n */\n encode.ULONG = function(v) {\n return [(v >> 24) & 0xFF, (v >> 16) & 0xFF, (v >> 8) & 0xFF, v & 0xFF];\n };\n\n /**\n * @constant\n * @type {number}\n */\n sizeOf.ULONG = constant(4);\n\n /**\n * Convert a 32-bit unsigned integer to a list of 4 bytes.\n * @param {number}\n * @returns {Array}\n */\n encode.LONG = function(v) {\n // Two's complement\n if (v >= LIMIT32) {\n v = -(2 * LIMIT32 - v);\n }\n\n return [(v >> 24) & 0xFF, (v >> 16) & 0xFF, (v >> 8) & 0xFF, v & 0xFF];\n };\n\n /**\n * @constant\n * @type {number}\n */\n sizeOf.LONG = constant(4);\n\n encode.FIXED = encode.ULONG;\n sizeOf.FIXED = sizeOf.ULONG;\n\n encode.FWORD = encode.SHORT;\n sizeOf.FWORD = sizeOf.SHORT;\n\n encode.UFWORD = encode.USHORT;\n sizeOf.UFWORD = sizeOf.USHORT;\n\n /**\n * Convert a 32-bit Apple Mac timestamp integer to a list of 8 bytes, 64-bit timestamp.\n * @param {number}\n * @returns {Array}\n */\n encode.LONGDATETIME = function(v) {\n return [0, 0, 0, 0, (v >> 24) & 0xFF, (v >> 16) & 0xFF, (v >> 8) & 0xFF, v & 0xFF];\n };\n\n /**\n * @constant\n * @type {number}\n */\n sizeOf.LONGDATETIME = constant(8);\n\n /**\n * Convert a 4-char tag to a list of 4 bytes.\n * @param {string}\n * @returns {Array}\n */\n encode.TAG = function(v) {\n check.argument(v.length === 4, 'Tag should be exactly 4 ASCII characters.');\n return [v.charCodeAt(0),\n v.charCodeAt(1),\n v.charCodeAt(2),\n v.charCodeAt(3)];\n };\n\n /**\n * @constant\n * @type {number}\n */\n sizeOf.TAG = constant(4);\n\n // CFF data types ///////////////////////////////////////////////////////////\n\n encode.Card8 = encode.BYTE;\n sizeOf.Card8 = sizeOf.BYTE;\n\n encode.Card16 = encode.USHORT;\n sizeOf.Card16 = sizeOf.USHORT;\n\n encode.OffSize = encode.BYTE;\n sizeOf.OffSize = sizeOf.BYTE;\n\n encode.SID = encode.USHORT;\n sizeOf.SID = sizeOf.USHORT;\n\n // Convert a numeric operand or charstring number to a variable-size list of bytes.\n /**\n * Convert a numeric operand or charstring number to a variable-size list of bytes.\n * @param {number}\n * @returns {Array}\n */\n encode.NUMBER = function(v) {\n if (v >= -107 && v <= 107) {\n return [v + 139];\n } else if (v >= 108 && v <= 1131) {\n v = v - 108;\n return [(v >> 8) + 247, v & 0xFF];\n } else if (v >= -1131 && v <= -108) {\n v = -v - 108;\n return [(v >> 8) + 251, v & 0xFF];\n } else if (v >= -32768 && v <= 32767) {\n return encode.NUMBER16(v);\n } else {\n return encode.NUMBER32(v);\n }\n };\n\n /**\n * @param {number}\n * @returns {number}\n */\n sizeOf.NUMBER = function(v) {\n return encode.NUMBER(v).length;\n };\n\n /**\n * Convert a signed number between -32768 and +32767 to a three-byte value.\n * This ensures we always use three bytes, but is not the most compact format.\n * @param {number}\n * @returns {Array}\n */\n encode.NUMBER16 = function(v) {\n return [28, (v >> 8) & 0xFF, v & 0xFF];\n };\n\n /**\n * @constant\n * @type {number}\n */\n sizeOf.NUMBER16 = constant(3);\n\n /**\n * Convert a signed number between -(2^31) and +(2^31-1) to a five-byte value.\n * This is useful if you want to be sure you always use four bytes,\n * at the expense of wasting a few bytes for smaller numbers.\n * @param {number}\n * @returns {Array}\n */\n encode.NUMBER32 = function(v) {\n return [29, (v >> 24) & 0xFF, (v >> 16) & 0xFF, (v >> 8) & 0xFF, v & 0xFF];\n };\n\n /**\n * @constant\n * @type {number}\n */\n sizeOf.NUMBER32 = constant(5);\n\n /**\n * @param {number}\n * @returns {Array}\n */\n encode.REAL = function(v) {\n var value = v.toString();\n\n // Some numbers use an epsilon to encode the value. (e.g. JavaScript will store 0.0000001 as 1e-7)\n // This code converts it back to a number without the epsilon.\n var m = /\\.(\\d*?)(?:9{5,20}|0{5,20})\\d{0,2}(?:e(.+)|$)/.exec(value);\n if (m) {\n var epsilon = parseFloat('1e' + ((m[2] ? +m[2] : 0) + m[1].length));\n value = (Math.round(v * epsilon) / epsilon).toString();\n }\n\n var nibbles = '';\n for (var i = 0, ii = value.length; i < ii; i += 1) {\n var c = value[i];\n if (c === 'e') {\n nibbles += value[++i] === '-' ? 'c' : 'b';\n } else if (c === '.') {\n nibbles += 'a';\n } else if (c === '-') {\n nibbles += 'e';\n } else {\n nibbles += c;\n }\n }\n\n nibbles += (nibbles.length & 1) ? 'f' : 'ff';\n var out = [30];\n for (var i$1 = 0, ii$1 = nibbles.length; i$1 < ii$1; i$1 += 2) {\n out.push(parseInt(nibbles.substr(i$1, 2), 16));\n }\n\n return out;\n };\n\n /**\n * @param {number}\n * @returns {number}\n */\n sizeOf.REAL = function(v) {\n return encode.REAL(v).length;\n };\n\n encode.NAME = encode.CHARARRAY;\n sizeOf.NAME = sizeOf.CHARARRAY;\n\n encode.STRING = encode.CHARARRAY;\n sizeOf.STRING = sizeOf.CHARARRAY;\n\n /**\n * @param {DataView} data\n * @param {number} offset\n * @param {number} numBytes\n * @returns {string}\n */\n decode.UTF8 = function(data, offset, numBytes) {\n var codePoints = [];\n var numChars = numBytes;\n for (var j = 0; j < numChars; j++, offset += 1) {\n codePoints[j] = data.getUint8(offset);\n }\n\n return String.fromCharCode.apply(null, codePoints);\n };\n\n /**\n * @param {DataView} data\n * @param {number} offset\n * @param {number} numBytes\n * @returns {string}\n */\n decode.UTF16 = function(data, offset, numBytes) {\n var codePoints = [];\n var numChars = numBytes / 2;\n for (var j = 0; j < numChars; j++, offset += 2) {\n codePoints[j] = data.getUint16(offset);\n }\n\n return String.fromCharCode.apply(null, codePoints);\n };\n\n /**\n * Convert a JavaScript string to UTF16-BE.\n * @param {string}\n * @returns {Array}\n */\n encode.UTF16 = function(v) {\n var b = [];\n for (var i = 0; i < v.length; i += 1) {\n var codepoint = v.charCodeAt(i);\n b[b.length] = (codepoint >> 8) & 0xFF;\n b[b.length] = codepoint & 0xFF;\n }\n\n return b;\n };\n\n /**\n * @param {string}\n * @returns {number}\n */\n sizeOf.UTF16 = function(v) {\n return v.length * 2;\n };\n\n // Data for converting old eight-bit Macintosh encodings to Unicode.\n // This representation is optimized for decoding; encoding is slower\n // and needs more memory. The assumption is that all opentype.js users\n // want to open fonts, but saving a font will be comparatively rare\n // so it can be more expensive. Keyed by IANA character set name.\n //\n // Python script for generating these strings:\n //\n // s = u''.join([chr(c).decode('mac_greek') for c in range(128, 256)])\n // print(s.encode('utf-8'))\n /**\n * @private\n */\n var eightBitMacEncodings = {\n 'x-mac-croatian': // Python: 'mac_croatian'\n 'ÄÅÇÉÑÖÜáàâäãåçéèêëíìîïñóòôöõúùûü†°¢£§•¶ß®Š™´¨≠ŽØ∞±≤≥∆µ∂∑∏š∫ªºΩžø' +\n '¿¡¬√ƒ≈Ć«Č… ÀÃÕŒœĐ—“”‘’÷◊©⁄€‹›Æ»–·‚„‰ÂćÁčÈÍÎÏÌÓÔđÒÚÛÙıˆ˜¯πË˚¸Êæˇ',\n 'x-mac-cyrillic': // Python: 'mac_cyrillic'\n 'АБВГДЕЖЗИЙКЛМНОПРСТУФХЦЧШЩЪЫЬЭЮЯ†°Ґ£§•¶І®©™Ђђ≠Ѓѓ∞±≤≥іµґЈЄєЇїЉљЊњ' +\n 'јЅ¬√ƒ≈∆«»… ЋћЌќѕ–—“”‘’÷„ЎўЏџ№Ёёяабвгдежзийклмнопрстуфхцчшщъыьэю',\n 'x-mac-gaelic': // http://unicode.org/Public/MAPPINGS/VENDORS/APPLE/GAELIC.TXT\n 'ÄÅÇÉÑÖÜáàâäãåçéèêëíìîïñóòôöõúùûü†°¢£§•¶ß®©™´¨≠ÆØḂ±≤≥ḃĊċḊḋḞḟĠġṀæø' +\n 'ṁṖṗɼƒſṠ«»… ÀÃÕŒœ–—“”‘’ṡẛÿŸṪ€‹›Ŷŷṫ·Ỳỳ⁊ÂÊÁËÈÍÎÏÌÓÔ♣ÒÚÛÙıÝýŴŵẄẅẀẁẂẃ',\n 'x-mac-greek': // Python: 'mac_greek'\n 'Ĺ²É³ÖÜ΅àâä΄¨çéèê룙î‰ôö¦€ùûü†ΓΔΘΛΞΠß®©ΣΪ§≠°·Α±≤≥¥ΒΕΖΗΙΚΜΦΫΨΩ' +\n 'άΝ¬ΟΡ≈Τ«»… ΥΧΆΈœ–―“”‘’÷ΉΊΌΎέήίόΏύαβψδεφγηιξκλμνοπώρστθωςχυζϊϋΐΰ\\u00AD',\n 'x-mac-icelandic': // Python: 'mac_iceland'\n 'ÄÅÇÉÑÖÜáàâäãåçéèêëíìîïñóòôöõúùûüÝ°¢£§•¶ß®©™´¨≠ÆØ∞±≤≥¥µ∂∑∏π∫ªºΩæø' +\n '¿¡¬√ƒ≈∆«»… ÀÃÕŒœ–—“”‘’÷◊ÿŸ⁄€ÐðÞþý·‚„‰ÂÊÁËÈÍÎÏÌÓÔÒÚÛÙıˆ˜¯˘˙˚¸˝˛ˇ',\n 'x-mac-inuit': // http://unicode.org/Public/MAPPINGS/VENDORS/APPLE/INUIT.TXT\n 'ᐃᐄᐅᐆᐊᐋᐱᐲᐳᐴᐸᐹᑉᑎᑏᑐᑑᑕᑖᑦᑭᑮᑯᑰᑲᑳᒃᒋᒌᒍᒎᒐᒑ°ᒡᒥᒦ•¶ᒧ®©™ᒨᒪᒫᒻᓂᓃᓄᓅᓇᓈᓐᓯᓰᓱᓲᓴᓵᔅᓕᓖᓗ' +\n 'ᓘᓚᓛᓪᔨᔩᔪᔫᔭ… ᔮᔾᕕᕖᕗ–—“”‘’ᕘᕙᕚᕝᕆᕇᕈᕉᕋᕌᕐᕿᖀᖁᖂᖃᖄᖅᖏᖐᖑᖒᖓᖔᖕᙱᙲᙳᙴᙵᙶᖖᖠᖡᖢᖣᖤᖥᖦᕼŁł',\n 'x-mac-ce': // Python: 'mac_latin2'\n 'ÄĀāÉĄÖÜáąČäčĆć鏟ĎíďĒēĖóėôöõúĚěü†°Ę£§•¶ß®©™ę¨≠ģĮįĪ≤≥īĶ∂∑łĻļĽľĹĺŅ' +\n 'ņѬ√ńŇ∆«»… ňŐÕőŌ–—“”‘’÷◊ōŔŕŘ‹›řŖŗŠ‚„šŚśÁŤťÍŽžŪÓÔūŮÚůŰűŲųÝýķŻŁżĢˇ',\n macintosh: // Python: 'mac_roman'\n 'ÄÅÇÉÑÖÜáàâäãåçéèêëíìîïñóòôöõúùûü†°¢£§•¶ß®©™´¨≠ÆØ∞±≤≥¥µ∂∑∏π∫ªºΩæø' +\n '¿¡¬√ƒ≈∆«»… ÀÃÕŒœ–—“”‘’÷◊ÿŸ⁄€‹›fifl‡·‚„‰ÂÊÁËÈÍÎÏÌÓÔÒÚÛÙıˆ˜¯˘˙˚¸˝˛ˇ',\n 'x-mac-romanian': // Python: 'mac_romanian'\n 'ÄÅÇÉÑÖÜáàâäãåçéèêëíìîïñóòôöõúùûü†°¢£§•¶ß®©™´¨≠ĂȘ∞±≤≥¥µ∂∑∏π∫ªºΩăș' +\n '¿¡¬√ƒ≈∆«»… ÀÃÕŒœ–—“”‘’÷◊ÿŸ⁄€‹›Țț‡·‚„‰ÂÊÁËÈÍÎÏÌÓÔÒÚÛÙıˆ˜¯˘˙˚¸˝˛ˇ',\n 'x-mac-turkish': // Python: 'mac_turkish'\n 'ÄÅÇÉÑÖÜáàâäãåçéèêëíìîïñóòôöõúùûü†°¢£§•¶ß®©™´¨≠ÆØ∞±≤≥¥µ∂∑∏π∫ªºΩæø' +\n '¿¡¬√ƒ≈∆«»… ÀÃÕŒœ–—“”‘’÷◊ÿŸĞğİıŞş‡·‚„‰ÂÊÁËÈÍÎÏÌÓÔÒÚÛÙˆ˜¯˘˙˚¸˝˛ˇ'\n };\n\n /**\n * Decodes an old-style Macintosh string. Returns either a Unicode JavaScript\n * string, or 'undefined' if the encoding is unsupported. For example, we do\n * not support Chinese, Japanese or Korean because these would need large\n * mapping tables.\n * @param {DataView} dataView\n * @param {number} offset\n * @param {number} dataLength\n * @param {string} encoding\n * @returns {string}\n */\n decode.MACSTRING = function(dataView, offset, dataLength, encoding) {\n var table = eightBitMacEncodings[encoding];\n if (table === undefined) {\n return undefined;\n }\n\n var result = '';\n for (var i = 0; i < dataLength; i++) {\n var c = dataView.getUint8(offset + i);\n // In all eight-bit Mac encodings, the characters 0x00..0x7F are\n // mapped to U+0000..U+007F; we only need to look up the others.\n if (c <= 0x7F) {\n result += String.fromCharCode(c);\n } else {\n result += table[c & 0x7F];\n }\n }\n\n return result;\n };\n\n // Helper function for encode.MACSTRING. Returns a dictionary for mapping\n // Unicode character codes to their 8-bit MacOS equivalent. This table\n // is not exactly a super cheap data structure, but we do not care because\n // encoding Macintosh strings is only rarely needed in typical applications.\n var macEncodingTableCache = typeof WeakMap === 'function' && new WeakMap();\n var macEncodingCacheKeys;\n var getMacEncodingTable = function (encoding) {\n // Since we use encoding as a cache key for WeakMap, it has to be\n // a String object and not a literal. And at least on NodeJS 2.10.1,\n // WeakMap requires that the same String instance is passed for cache hits.\n if (!macEncodingCacheKeys) {\n macEncodingCacheKeys = {};\n for (var e in eightBitMacEncodings) {\n /*jshint -W053 */ // Suppress \"Do not use String as a constructor.\"\n macEncodingCacheKeys[e] = new String(e);\n }\n }\n\n var cacheKey = macEncodingCacheKeys[encoding];\n if (cacheKey === undefined) {\n return undefined;\n }\n\n // We can't do \"if (cache.has(key)) {return cache.get(key)}\" here:\n // since garbage collection may run at any time, it could also kick in\n // between the calls to cache.has() and cache.get(). In that case,\n // we would return 'undefined' even though we do support the encoding.\n if (macEncodingTableCache) {\n var cachedTable = macEncodingTableCache.get(cacheKey);\n if (cachedTable !== undefined) {\n return cachedTable;\n }\n }\n\n var decodingTable = eightBitMacEncodings[encoding];\n if (decodingTable === undefined) {\n return undefined;\n }\n\n var encodingTable = {};\n for (var i = 0; i < decodingTable.length; i++) {\n encodingTable[decodingTable.charCodeAt(i)] = i + 0x80;\n }\n\n if (macEncodingTableCache) {\n macEncodingTableCache.set(cacheKey, encodingTable);\n }\n\n return encodingTable;\n };\n\n /**\n * Encodes an old-style Macintosh string. Returns a byte array upon success.\n * If the requested encoding is unsupported, or if the input string contains\n * a character that cannot be expressed in the encoding, the function returns\n * 'undefined'.\n * @param {string} str\n * @param {string} encoding\n * @returns {Array}\n */\n encode.MACSTRING = function(str, encoding) {\n var table = getMacEncodingTable(encoding);\n if (table === undefined) {\n return undefined;\n }\n\n var result = [];\n for (var i = 0; i < str.length; i++) {\n var c = str.charCodeAt(i);\n\n // In all eight-bit Mac encodings, the characters 0x00..0x7F are\n // mapped to U+0000..U+007F; we only need to look up the others.\n if (c >= 0x80) {\n c = table[c];\n if (c === undefined) {\n // str contains a Unicode character that cannot be encoded\n // in the requested encoding.\n return undefined;\n }\n }\n result[i] = c;\n // result.push(c);\n }\n\n return result;\n };\n\n /**\n * @param {string} str\n * @param {string} encoding\n * @returns {number}\n */\n sizeOf.MACSTRING = function(str, encoding) {\n var b = encode.MACSTRING(str, encoding);\n if (b !== undefined) {\n return b.length;\n } else {\n return 0;\n }\n };\n\n // Helper for encode.VARDELTAS\n function isByteEncodable(value) {\n return value >= -128 && value <= 127;\n }\n\n // Helper for encode.VARDELTAS\n function encodeVarDeltaRunAsZeroes(deltas, pos, result) {\n var runLength = 0;\n var numDeltas = deltas.length;\n while (pos < numDeltas && runLength < 64 && deltas[pos] === 0) {\n ++pos;\n ++runLength;\n }\n result.push(0x80 | (runLength - 1));\n return pos;\n }\n\n // Helper for encode.VARDELTAS\n function encodeVarDeltaRunAsBytes(deltas, offset, result) {\n var runLength = 0;\n var numDeltas = deltas.length;\n var pos = offset;\n while (pos < numDeltas && runLength < 64) {\n var value = deltas[pos];\n if (!isByteEncodable(value)) {\n break;\n }\n\n // Within a byte-encoded run of deltas, a single zero is best\n // stored literally as 0x00 value. However, if we have two or\n // more zeroes in a sequence, it is better to start a new run.\n // Fore example, the sequence of deltas [15, 15, 0, 15, 15]\n // becomes 6 bytes (04 0F 0F 00 0F 0F) when storing the zero\n // within the current run, but 7 bytes (01 0F 0F 80 01 0F 0F)\n // when starting a new run.\n if (value === 0 && pos + 1 < numDeltas && deltas[pos + 1] === 0) {\n break;\n }\n\n ++pos;\n ++runLength;\n }\n result.push(runLength - 1);\n for (var i = offset; i < pos; ++i) {\n result.push((deltas[i] + 256) & 0xff);\n }\n return pos;\n }\n\n // Helper for encode.VARDELTAS\n function encodeVarDeltaRunAsWords(deltas, offset, result) {\n var runLength = 0;\n var numDeltas = deltas.length;\n var pos = offset;\n while (pos < numDeltas && runLength < 64) {\n var value = deltas[pos];\n\n // Within a word-encoded run of deltas, it is easiest to start\n // a new run (with a different encoding) whenever we encounter\n // a zero value. For example, the sequence [0x6666, 0, 0x7777]\n // needs 7 bytes when storing the zero inside the current run\n // (42 66 66 00 00 77 77), and equally 7 bytes when starting a\n // new run (40 66 66 80 40 77 77).\n if (value === 0) {\n break;\n }\n\n // Within a word-encoded run of deltas, a single value in the\n // range (-128..127) should be encoded within the current run\n // because it is more compact. For example, the sequence\n // [0x6666, 2, 0x7777] becomes 7 bytes when storing the value\n // literally (42 66 66 00 02 77 77), but 8 bytes when starting\n // a new run (40 66 66 00 02 40 77 77).\n if (isByteEncodable(value) && pos + 1 < numDeltas && isByteEncodable(deltas[pos + 1])) {\n break;\n }\n\n ++pos;\n ++runLength;\n }\n result.push(0x40 | (runLength - 1));\n for (var i = offset; i < pos; ++i) {\n var val = deltas[i];\n result.push(((val + 0x10000) >> 8) & 0xff, (val + 0x100) & 0xff);\n }\n return pos;\n }\n\n /**\n * Encode a list of variation adjustment deltas.\n *\n * Variation adjustment deltas are used in ‘gvar’ and ‘cvar’ tables.\n * They indicate how points (in ‘gvar’) or values (in ‘cvar’) get adjusted\n * when generating instances of variation fonts.\n *\n * @see https://www.microsoft.com/typography/otspec/gvar.htm\n * @see https://developer.apple.com/fonts/TrueType-Reference-Manual/RM06/Chap6gvar.html\n * @param {Array}\n * @return {Array}\n */\n encode.VARDELTAS = function(deltas) {\n var pos = 0;\n var result = [];\n while (pos < deltas.length) {\n var value = deltas[pos];\n if (value === 0) {\n pos = encodeVarDeltaRunAsZeroes(deltas, pos, result);\n } else if (value >= -128 && value <= 127) {\n pos = encodeVarDeltaRunAsBytes(deltas, pos, result);\n } else {\n pos = encodeVarDeltaRunAsWords(deltas, pos, result);\n }\n }\n return result;\n };\n\n // Convert a list of values to a CFF INDEX structure.\n // The values should be objects containing name / type / value.\n /**\n * @param {Array} l\n * @returns {Array}\n */\n encode.INDEX = function(l) {\n //var offset, offsets, offsetEncoder, encodedOffsets, encodedOffset, data,\n // i, v;\n // Because we have to know which data type to use to encode the offsets,\n // we have to go through the values twice: once to encode the data and\n // calculate the offsets, then again to encode the offsets using the fitting data type.\n var offset = 1; // First offset is always 1.\n var offsets = [offset];\n var data = [];\n for (var i = 0; i < l.length; i += 1) {\n var v = encode.OBJECT(l[i]);\n Array.prototype.push.apply(data, v);\n offset += v.length;\n offsets.push(offset);\n }\n\n if (data.length === 0) {\n return [0, 0];\n }\n\n var encodedOffsets = [];\n var offSize = (1 + Math.floor(Math.log(offset) / Math.log(2)) / 8) | 0;\n var offsetEncoder = [undefined, encode.BYTE, encode.USHORT, encode.UINT24, encode.ULONG][offSize];\n for (var i$1 = 0; i$1 < offsets.length; i$1 += 1) {\n var encodedOffset = offsetEncoder(offsets[i$1]);\n Array.prototype.push.apply(encodedOffsets, encodedOffset);\n }\n\n return Array.prototype.concat(encode.Card16(l.length),\n encode.OffSize(offSize),\n encodedOffsets,\n data);\n };\n\n /**\n * @param {Array}\n * @returns {number}\n */\n sizeOf.INDEX = function(v) {\n return encode.INDEX(v).length;\n };\n\n /**\n * Convert an object to a CFF DICT structure.\n * The keys should be numeric.\n * The values should be objects containing name / type / value.\n * @param {Object} m\n * @returns {Array}\n */\n encode.DICT = function(m) {\n var d = [];\n var keys = Object.keys(m);\n var length = keys.length;\n\n for (var i = 0; i < length; i += 1) {\n // Object.keys() return string keys, but our keys are always numeric.\n var k = parseInt(keys[i], 0);\n var v = m[k];\n // Value comes before the key.\n d = d.concat(encode.OPERAND(v.value, v.type));\n d = d.concat(encode.OPERATOR(k));\n }\n\n return d;\n };\n\n /**\n * @param {Object}\n * @returns {number}\n */\n sizeOf.DICT = function(m) {\n return encode.DICT(m).length;\n };\n\n /**\n * @param {number}\n * @returns {Array}\n */\n encode.OPERATOR = function(v) {\n if (v < 1200) {\n return [v];\n } else {\n return [12, v - 1200];\n }\n };\n\n /**\n * @param {Array} v\n * @param {string}\n * @returns {Array}\n */\n encode.OPERAND = function(v, type) {\n var d = [];\n if (Array.isArray(type)) {\n for (var i = 0; i < type.length; i += 1) {\n check.argument(v.length === type.length, 'Not enough arguments given for type' + type);\n d = d.concat(encode.OPERAND(v[i], type[i]));\n }\n } else {\n if (type === 'SID') {\n d = d.concat(encode.NUMBER(v));\n } else if (type === 'offset') {\n // We make it easy for ourselves and always encode offsets as\n // 4 bytes. This makes offset calculation for the top dict easier.\n d = d.concat(encode.NUMBER32(v));\n } else if (type === 'number') {\n d = d.concat(encode.NUMBER(v));\n } else if (type === 'real') {\n d = d.concat(encode.REAL(v));\n } else {\n throw new Error('Unknown operand type ' + type);\n // FIXME Add support for booleans\n }\n }\n\n return d;\n };\n\n encode.OP = encode.BYTE;\n sizeOf.OP = sizeOf.BYTE;\n\n // memoize charstring encoding using WeakMap if available\n var wmm = typeof WeakMap === 'function' && new WeakMap();\n\n /**\n * Convert a list of CharString operations to bytes.\n * @param {Array}\n * @returns {Array}\n */\n encode.CHARSTRING = function(ops) {\n // See encode.MACSTRING for why we don't do \"if (wmm && wmm.has(ops))\".\n if (wmm) {\n var cachedValue = wmm.get(ops);\n if (cachedValue !== undefined) {\n return cachedValue;\n }\n }\n\n var d = [];\n var length = ops.length;\n\n for (var i = 0; i < length; i += 1) {\n var op = ops[i];\n d = d.concat(encode[op.type](op.value));\n }\n\n if (wmm) {\n wmm.set(ops, d);\n }\n\n return d;\n };\n\n /**\n * @param {Array}\n * @returns {number}\n */\n sizeOf.CHARSTRING = function(ops) {\n return encode.CHARSTRING(ops).length;\n };\n\n // Utility functions ////////////////////////////////////////////////////////\n\n /**\n * Convert an object containing name / type / value to bytes.\n * @param {Object}\n * @returns {Array}\n */\n encode.OBJECT = function(v) {\n var encodingFunction = encode[v.type];\n check.argument(encodingFunction !== undefined, 'No encoding function for type ' + v.type);\n return encodingFunction(v.value);\n };\n\n /**\n * @param {Object}\n * @returns {number}\n */\n sizeOf.OBJECT = function(v) {\n var sizeOfFunction = sizeOf[v.type];\n check.argument(sizeOfFunction !== undefined, 'No sizeOf function for type ' + v.type);\n return sizeOfFunction(v.value);\n };\n\n /**\n * Convert a table object to bytes.\n * A table contains a list of fields containing the metadata (name, type and default value).\n * The table itself has the field values set as attributes.\n * @param {opentype.Table}\n * @returns {Array}\n */\n encode.TABLE = function(table) {\n var d = [];\n var length = table.fields.length;\n var subtables = [];\n var subtableOffsets = [];\n\n for (var i = 0; i < length; i += 1) {\n var field = table.fields[i];\n var encodingFunction = encode[field.type];\n check.argument(encodingFunction !== undefined, 'No encoding function for field type ' + field.type + ' (' + field.name + ')');\n var value = table[field.name];\n if (value === undefined) {\n value = field.value;\n }\n\n var bytes = encodingFunction(value);\n\n if (field.type === 'TABLE') {\n subtableOffsets.push(d.length);\n d = d.concat([0, 0]);\n subtables.push(bytes);\n } else {\n d = d.concat(bytes);\n }\n }\n\n for (var i$1 = 0; i$1 < subtables.length; i$1 += 1) {\n var o = subtableOffsets[i$1];\n var offset = d.length;\n check.argument(offset < 65536, 'Table ' + table.tableName + ' too big.');\n d[o] = offset >> 8;\n d[o + 1] = offset & 0xff;\n d = d.concat(subtables[i$1]);\n }\n\n return d;\n };\n\n /**\n * @param {opentype.Table}\n * @returns {number}\n */\n sizeOf.TABLE = function(table) {\n var numBytes = 0;\n var length = table.fields.length;\n\n for (var i = 0; i < length; i += 1) {\n var field = table.fields[i];\n var sizeOfFunction = sizeOf[field.type];\n check.argument(sizeOfFunction !== undefined, 'No sizeOf function for field type ' + field.type + ' (' + field.name + ')');\n var value = table[field.name];\n if (value === undefined) {\n value = field.value;\n }\n\n numBytes += sizeOfFunction(value);\n\n // Subtables take 2 more bytes for offsets.\n if (field.type === 'TABLE') {\n numBytes += 2;\n }\n }\n\n return numBytes;\n };\n\n encode.RECORD = encode.TABLE;\n sizeOf.RECORD = sizeOf.TABLE;\n\n // Merge in a list of bytes.\n encode.LITERAL = function(v) {\n return v;\n };\n\n sizeOf.LITERAL = function(v) {\n return v.length;\n };\n\n // Table metadata\n\n /**\n * @exports opentype.Table\n * @class\n * @param {string} tableName\n * @param {Array} fields\n * @param {Object} options\n * @constructor\n */\n function Table(tableName, fields, options) {\n // For coverage tables with coverage format 2, we do not want to add the coverage data directly to the table object,\n // as this will result in wrong encoding order of the coverage data on serialization to bytes.\n // The fallback of using the field values directly when not present on the table is handled in types.encode.TABLE() already.\n if (fields.length && (fields[0].name !== 'coverageFormat' || fields[0].value === 1)) {\n for (var i = 0; i < fields.length; i += 1) {\n var field = fields[i];\n this[field.name] = field.value;\n }\n }\n\n this.tableName = tableName;\n this.fields = fields;\n if (options) {\n var optionKeys = Object.keys(options);\n for (var i$1 = 0; i$1 < optionKeys.length; i$1 += 1) {\n var k = optionKeys[i$1];\n var v = options[k];\n if (this[k] !== undefined) {\n this[k] = v;\n }\n }\n }\n }\n\n /**\n * Encodes the table and returns an array of bytes\n * @return {Array}\n */\n Table.prototype.encode = function() {\n return encode.TABLE(this);\n };\n\n /**\n * Get the size of the table.\n * @return {number}\n */\n Table.prototype.sizeOf = function() {\n return sizeOf.TABLE(this);\n };\n\n /**\n * @private\n */\n function ushortList(itemName, list, count) {\n if (count === undefined) {\n count = list.length;\n }\n var fields = new Array(list.length + 1);\n fields[0] = {name: itemName + 'Count', type: 'USHORT', value: count};\n for (var i = 0; i < list.length; i++) {\n fields[i + 1] = {name: itemName + i, type: 'USHORT', value: list[i]};\n }\n return fields;\n }\n\n /**\n * @private\n */\n function tableList(itemName, records, itemCallback) {\n var count = records.length;\n var fields = new Array(count + 1);\n fields[0] = {name: itemName + 'Count', type: 'USHORT', value: count};\n for (var i = 0; i < count; i++) {\n fields[i + 1] = {name: itemName + i, type: 'TABLE', value: itemCallback(records[i], i)};\n }\n return fields;\n }\n\n /**\n * @private\n */\n function recordList(itemName, records, itemCallback) {\n var count = records.length;\n var fields = [];\n fields[0] = {name: itemName + 'Count', type: 'USHORT', value: count};\n for (var i = 0; i < count; i++) {\n fields = fields.concat(itemCallback(records[i], i));\n }\n return fields;\n }\n\n // Common Layout Tables\n\n /**\n * @exports opentype.Coverage\n * @class\n * @param {opentype.Table}\n * @constructor\n * @extends opentype.Table\n */\n function Coverage(coverageTable) {\n if (coverageTable.format === 1) {\n Table.call(this, 'coverageTable',\n [{name: 'coverageFormat', type: 'USHORT', value: 1}]\n .concat(ushortList('glyph', coverageTable.glyphs))\n );\n } else if (coverageTable.format === 2) {\n Table.call(this, 'coverageTable',\n [{name: 'coverageFormat', type: 'USHORT', value: 2}]\n .concat(recordList('rangeRecord', coverageTable.ranges, function(RangeRecord) {\n return [\n {name: 'startGlyphID', type: 'USHORT', value: RangeRecord.start},\n {name: 'endGlyphID', type: 'USHORT', value: RangeRecord.end},\n {name: 'startCoverageIndex', type: 'USHORT', value: RangeRecord.index} ];\n }))\n );\n } else {\n check.assert(false, 'Coverage format must be 1 or 2.');\n }\n }\n Coverage.prototype = Object.create(Table.prototype);\n Coverage.prototype.constructor = Coverage;\n\n function ScriptList(scriptListTable) {\n Table.call(this, 'scriptListTable',\n recordList('scriptRecord', scriptListTable, function(scriptRecord, i) {\n var script = scriptRecord.script;\n var defaultLangSys = script.defaultLangSys;\n check.assert(!!defaultLangSys, 'Unable to write GSUB: script ' + scriptRecord.tag + ' has no default language system.');\n return [\n {name: 'scriptTag' + i, type: 'TAG', value: scriptRecord.tag},\n {name: 'script' + i, type: 'TABLE', value: new Table('scriptTable', [\n {name: 'defaultLangSys', type: 'TABLE', value: new Table('defaultLangSys', [\n {name: 'lookupOrder', type: 'USHORT', value: 0},\n {name: 'reqFeatureIndex', type: 'USHORT', value: defaultLangSys.reqFeatureIndex}]\n .concat(ushortList('featureIndex', defaultLangSys.featureIndexes)))}\n ].concat(recordList('langSys', script.langSysRecords, function(langSysRecord, i) {\n var langSys = langSysRecord.langSys;\n return [\n {name: 'langSysTag' + i, type: 'TAG', value: langSysRecord.tag},\n {name: 'langSys' + i, type: 'TABLE', value: new Table('langSys', [\n {name: 'lookupOrder', type: 'USHORT', value: 0},\n {name: 'reqFeatureIndex', type: 'USHORT', value: langSys.reqFeatureIndex}\n ].concat(ushortList('featureIndex', langSys.featureIndexes)))}\n ];\n })))}\n ];\n })\n );\n }\n ScriptList.prototype = Object.create(Table.prototype);\n ScriptList.prototype.constructor = ScriptList;\n\n /**\n * @exports opentype.FeatureList\n * @class\n * @param {opentype.Table}\n * @constructor\n * @extends opentype.Table\n */\n function FeatureList(featureListTable) {\n Table.call(this, 'featureListTable',\n recordList('featureRecord', featureListTable, function(featureRecord, i) {\n var feature = featureRecord.feature;\n return [\n {name: 'featureTag' + i, type: 'TAG', value: featureRecord.tag},\n {name: 'feature' + i, type: 'TABLE', value: new Table('featureTable', [\n {name: 'featureParams', type: 'USHORT', value: feature.featureParams} ].concat(ushortList('lookupListIndex', feature.lookupListIndexes)))}\n ];\n })\n );\n }\n FeatureList.prototype = Object.create(Table.prototype);\n FeatureList.prototype.constructor = FeatureList;\n\n /**\n * @exports opentype.LookupList\n * @class\n * @param {opentype.Table}\n * @param {Object}\n * @constructor\n * @extends opentype.Table\n */\n function LookupList(lookupListTable, subtableMakers) {\n Table.call(this, 'lookupListTable', tableList('lookup', lookupListTable, function(lookupTable) {\n var subtableCallback = subtableMakers[lookupTable.lookupType];\n check.assert(!!subtableCallback, 'Unable to write GSUB lookup type ' + lookupTable.lookupType + ' tables.');\n return new Table('lookupTable', [\n {name: 'lookupType', type: 'USHORT', value: lookupTable.lookupType},\n {name: 'lookupFlag', type: 'USHORT', value: lookupTable.lookupFlag}\n ].concat(tableList('subtable', lookupTable.subtables, subtableCallback)));\n }));\n }\n LookupList.prototype = Object.create(Table.prototype);\n LookupList.prototype.constructor = LookupList;\n\n // Record = same as Table, but inlined (a Table has an offset and its data is further in the stream)\n // Don't use offsets inside Records (probable bug), only in Tables.\n var table = {\n Table: Table,\n Record: Table,\n Coverage: Coverage,\n ScriptList: ScriptList,\n FeatureList: FeatureList,\n LookupList: LookupList,\n ushortList: ushortList,\n tableList: tableList,\n recordList: recordList,\n };\n\n // Parsing utility functions\n\n // Retrieve an unsigned byte from the DataView.\n function getByte(dataView, offset) {\n return dataView.getUint8(offset);\n }\n\n // Retrieve an unsigned 16-bit short from the DataView.\n // The value is stored in big endian.\n function getUShort(dataView, offset) {\n return dataView.getUint16(offset, false);\n }\n\n // Retrieve a signed 16-bit short from the DataView.\n // The value is stored in big endian.\n function getShort(dataView, offset) {\n return dataView.getInt16(offset, false);\n }\n\n // Retrieve an unsigned 32-bit long from the DataView.\n // The value is stored in big endian.\n function getULong(dataView, offset) {\n return dataView.getUint32(offset, false);\n }\n\n // Retrieve a 32-bit signed fixed-point number (16.16) from the DataView.\n // The value is stored in big endian.\n function getFixed(dataView, offset) {\n var decimal = dataView.getInt16(offset, false);\n var fraction = dataView.getUint16(offset + 2, false);\n return decimal + fraction / 65535;\n }\n\n // Retrieve a 4-character tag from the DataView.\n // Tags are used to identify tables.\n function getTag(dataView, offset) {\n var tag = '';\n for (var i = offset; i < offset + 4; i += 1) {\n tag += String.fromCharCode(dataView.getInt8(i));\n }\n\n return tag;\n }\n\n // Retrieve an offset from the DataView.\n // Offsets are 1 to 4 bytes in length, depending on the offSize argument.\n function getOffset(dataView, offset, offSize) {\n var v = 0;\n for (var i = 0; i < offSize; i += 1) {\n v <<= 8;\n v += dataView.getUint8(offset + i);\n }\n\n return v;\n }\n\n // Retrieve a number of bytes from start offset to the end offset from the DataView.\n function getBytes(dataView, startOffset, endOffset) {\n var bytes = [];\n for (var i = startOffset; i < endOffset; i += 1) {\n bytes.push(dataView.getUint8(i));\n }\n\n return bytes;\n }\n\n // Convert the list of bytes to a string.\n function bytesToString(bytes) {\n var s = '';\n for (var i = 0; i < bytes.length; i += 1) {\n s += String.fromCharCode(bytes[i]);\n }\n\n return s;\n }\n\n var typeOffsets = {\n byte: 1,\n uShort: 2,\n short: 2,\n uLong: 4,\n fixed: 4,\n longDateTime: 8,\n tag: 4\n };\n\n // A stateful parser that changes the offset whenever a value is retrieved.\n // The data is a DataView.\n function Parser(data, offset) {\n this.data = data;\n this.offset = offset;\n this.relativeOffset = 0;\n }\n\n Parser.prototype.parseByte = function() {\n var v = this.data.getUint8(this.offset + this.relativeOffset);\n this.relativeOffset += 1;\n return v;\n };\n\n Parser.prototype.parseChar = function() {\n var v = this.data.getInt8(this.offset + this.relativeOffset);\n this.relativeOffset += 1;\n return v;\n };\n\n Parser.prototype.parseCard8 = Parser.prototype.parseByte;\n\n Parser.prototype.parseUShort = function() {\n var v = this.data.getUint16(this.offset + this.relativeOffset);\n this.relativeOffset += 2;\n return v;\n };\n\n Parser.prototype.parseCard16 = Parser.prototype.parseUShort;\n Parser.prototype.parseSID = Parser.prototype.parseUShort;\n Parser.prototype.parseOffset16 = Parser.prototype.parseUShort;\n\n Parser.prototype.parseShort = function() {\n var v = this.data.getInt16(this.offset + this.relativeOffset);\n this.relativeOffset += 2;\n return v;\n };\n\n Parser.prototype.parseF2Dot14 = function() {\n var v = this.data.getInt16(this.offset + this.relativeOffset) / 16384;\n this.relativeOffset += 2;\n return v;\n };\n\n Parser.prototype.parseULong = function() {\n var v = getULong(this.data, this.offset + this.relativeOffset);\n this.relativeOffset += 4;\n return v;\n };\n\n Parser.prototype.parseOffset32 = Parser.prototype.parseULong;\n\n Parser.prototype.parseFixed = function() {\n var v = getFixed(this.data, this.offset + this.relativeOffset);\n this.relativeOffset += 4;\n return v;\n };\n\n Parser.prototype.parseString = function(length) {\n var dataView = this.data;\n var offset = this.offset + this.relativeOffset;\n var string = '';\n this.relativeOffset += length;\n for (var i = 0; i < length; i++) {\n string += String.fromCharCode(dataView.getUint8(offset + i));\n }\n\n return string;\n };\n\n Parser.prototype.parseTag = function() {\n return this.parseString(4);\n };\n\n // LONGDATETIME is a 64-bit integer.\n // JavaScript and unix timestamps traditionally use 32 bits, so we\n // only take the last 32 bits.\n // + Since until 2038 those bits will be filled by zeros we can ignore them.\n Parser.prototype.parseLongDateTime = function() {\n var v = getULong(this.data, this.offset + this.relativeOffset + 4);\n // Subtract seconds between 01/01/1904 and 01/01/1970\n // to convert Apple Mac timestamp to Standard Unix timestamp\n v -= 2082844800;\n this.relativeOffset += 8;\n return v;\n };\n\n Parser.prototype.parseVersion = function(minorBase) {\n var major = getUShort(this.data, this.offset + this.relativeOffset);\n\n // How to interpret the minor version is very vague in the spec. 0x5000 is 5, 0x1000 is 1\n // Default returns the correct number if minor = 0xN000 where N is 0-9\n // Set minorBase to 1 for tables that use minor = N where N is 0-9\n var minor = getUShort(this.data, this.offset + this.relativeOffset + 2);\n this.relativeOffset += 4;\n if (minorBase === undefined) { minorBase = 0x1000; }\n return major + minor / minorBase / 10;\n };\n\n Parser.prototype.skip = function(type, amount) {\n if (amount === undefined) {\n amount = 1;\n }\n\n this.relativeOffset += typeOffsets[type] * amount;\n };\n\n ///// Parsing lists and records ///////////////////////////////\n\n // Parse a list of 32 bit unsigned integers.\n Parser.prototype.parseULongList = function(count) {\n if (count === undefined) { count = this.parseULong(); }\n var offsets = new Array(count);\n var dataView = this.data;\n var offset = this.offset + this.relativeOffset;\n for (var i = 0; i < count; i++) {\n offsets[i] = dataView.getUint32(offset);\n offset += 4;\n }\n\n this.relativeOffset += count * 4;\n return offsets;\n };\n\n // Parse a list of 16 bit unsigned integers. The length of the list can be read on the stream\n // or provided as an argument.\n Parser.prototype.parseOffset16List =\n Parser.prototype.parseUShortList = function(count) {\n if (count === undefined) { count = this.parseUShort(); }\n var offsets = new Array(count);\n var dataView = this.data;\n var offset = this.offset + this.relativeOffset;\n for (var i = 0; i < count; i++) {\n offsets[i] = dataView.getUint16(offset);\n offset += 2;\n }\n\n this.relativeOffset += count * 2;\n return offsets;\n };\n\n // Parses a list of 16 bit signed integers.\n Parser.prototype.parseShortList = function(count) {\n var list = new Array(count);\n var dataView = this.data;\n var offset = this.offset + this.relativeOffset;\n for (var i = 0; i < count; i++) {\n list[i] = dataView.getInt16(offset);\n offset += 2;\n }\n\n this.relativeOffset += count * 2;\n return list;\n };\n\n // Parses a list of bytes.\n Parser.prototype.parseByteList = function(count) {\n var list = new Array(count);\n var dataView = this.data;\n var offset = this.offset + this.relativeOffset;\n for (var i = 0; i < count; i++) {\n list[i] = dataView.getUint8(offset++);\n }\n\n this.relativeOffset += count;\n return list;\n };\n\n /**\n * Parse a list of items.\n * Record count is optional, if omitted it is read from the stream.\n * itemCallback is one of the Parser methods.\n */\n Parser.prototype.parseList = function(count, itemCallback) {\n if (!itemCallback) {\n itemCallback = count;\n count = this.parseUShort();\n }\n var list = new Array(count);\n for (var i = 0; i < count; i++) {\n list[i] = itemCallback.call(this);\n }\n return list;\n };\n\n Parser.prototype.parseList32 = function(count, itemCallback) {\n if (!itemCallback) {\n itemCallback = count;\n count = this.parseULong();\n }\n var list = new Array(count);\n for (var i = 0; i < count; i++) {\n list[i] = itemCallback.call(this);\n }\n return list;\n };\n\n /**\n * Parse a list of records.\n * Record count is optional, if omitted it is read from the stream.\n * Example of recordDescription: { sequenceIndex: Parser.uShort, lookupListIndex: Parser.uShort }\n */\n Parser.prototype.parseRecordList = function(count, recordDescription) {\n // If the count argument is absent, read it in the stream.\n if (!recordDescription) {\n recordDescription = count;\n count = this.parseUShort();\n }\n var records = new Array(count);\n var fields = Object.keys(recordDescription);\n for (var i = 0; i < count; i++) {\n var rec = {};\n for (var j = 0; j < fields.length; j++) {\n var fieldName = fields[j];\n var fieldType = recordDescription[fieldName];\n rec[fieldName] = fieldType.call(this);\n }\n records[i] = rec;\n }\n return records;\n };\n\n Parser.prototype.parseRecordList32 = function(count, recordDescription) {\n // If the count argument is absent, read it in the stream.\n if (!recordDescription) {\n recordDescription = count;\n count = this.parseULong();\n }\n var records = new Array(count);\n var fields = Object.keys(recordDescription);\n for (var i = 0; i < count; i++) {\n var rec = {};\n for (var j = 0; j < fields.length; j++) {\n var fieldName = fields[j];\n var fieldType = recordDescription[fieldName];\n rec[fieldName] = fieldType.call(this);\n }\n records[i] = rec;\n }\n return records;\n };\n\n // Parse a data structure into an object\n // Example of description: { sequenceIndex: Parser.uShort, lookupListIndex: Parser.uShort }\n Parser.prototype.parseStruct = function(description) {\n if (typeof description === 'function') {\n return description.call(this);\n } else {\n var fields = Object.keys(description);\n var struct = {};\n for (var j = 0; j < fields.length; j++) {\n var fieldName = fields[j];\n var fieldType = description[fieldName];\n struct[fieldName] = fieldType.call(this);\n }\n return struct;\n }\n };\n\n /**\n * Parse a GPOS valueRecord\n * https://docs.microsoft.com/en-us/typography/opentype/spec/gpos#value-record\n * valueFormat is optional, if omitted it is read from the stream.\n */\n Parser.prototype.parseValueRecord = function(valueFormat) {\n if (valueFormat === undefined) {\n valueFormat = this.parseUShort();\n }\n if (valueFormat === 0) {\n // valueFormat2 in kerning pairs is most often 0\n // in this case return undefined instead of an empty object, to save space\n return;\n }\n var valueRecord = {};\n\n if (valueFormat & 0x0001) { valueRecord.xPlacement = this.parseShort(); }\n if (valueFormat & 0x0002) { valueRecord.yPlacement = this.parseShort(); }\n if (valueFormat & 0x0004) { valueRecord.xAdvance = this.parseShort(); }\n if (valueFormat & 0x0008) { valueRecord.yAdvance = this.parseShort(); }\n\n // Device table (non-variable font) / VariationIndex table (variable font) not supported\n // https://docs.microsoft.com/fr-fr/typography/opentype/spec/chapter2#devVarIdxTbls\n if (valueFormat & 0x0010) { valueRecord.xPlaDevice = undefined; this.parseShort(); }\n if (valueFormat & 0x0020) { valueRecord.yPlaDevice = undefined; this.parseShort(); }\n if (valueFormat & 0x0040) { valueRecord.xAdvDevice = undefined; this.parseShort(); }\n if (valueFormat & 0x0080) { valueRecord.yAdvDevice = undefined; this.parseShort(); }\n\n return valueRecord;\n };\n\n /**\n * Parse a list of GPOS valueRecords\n * https://docs.microsoft.com/en-us/typography/opentype/spec/gpos#value-record\n * valueFormat and valueCount are read from the stream.\n */\n Parser.prototype.parseValueRecordList = function() {\n var valueFormat = this.parseUShort();\n var valueCount = this.parseUShort();\n var values = new Array(valueCount);\n for (var i = 0; i < valueCount; i++) {\n values[i] = this.parseValueRecord(valueFormat);\n }\n return values;\n };\n\n Parser.prototype.parsePointer = function(description) {\n var structOffset = this.parseOffset16();\n if (structOffset > 0) {\n // NULL offset => return undefined\n return new Parser(this.data, this.offset + structOffset).parseStruct(description);\n }\n return undefined;\n };\n\n Parser.prototype.parsePointer32 = function(description) {\n var structOffset = this.parseOffset32();\n if (structOffset > 0) {\n // NULL offset => return undefined\n return new Parser(this.data, this.offset + structOffset).parseStruct(description);\n }\n return undefined;\n };\n\n /**\n * Parse a list of offsets to lists of 16-bit integers,\n * or a list of offsets to lists of offsets to any kind of items.\n * If itemCallback is not provided, a list of list of UShort is assumed.\n * If provided, itemCallback is called on each item and must parse the item.\n * See examples in tables/gsub.js\n */\n Parser.prototype.parseListOfLists = function(itemCallback) {\n var offsets = this.parseOffset16List();\n var count = offsets.length;\n var relativeOffset = this.relativeOffset;\n var list = new Array(count);\n for (var i = 0; i < count; i++) {\n var start = offsets[i];\n if (start === 0) {\n // NULL offset\n // Add i as owned property to list. Convenient with assert.\n list[i] = undefined;\n continue;\n }\n this.relativeOffset = start;\n if (itemCallback) {\n var subOffsets = this.parseOffset16List();\n var subList = new Array(subOffsets.length);\n for (var j = 0; j < subOffsets.length; j++) {\n this.relativeOffset = start + subOffsets[j];\n subList[j] = itemCallback.call(this);\n }\n list[i] = subList;\n } else {\n list[i] = this.parseUShortList();\n }\n }\n this.relativeOffset = relativeOffset;\n return list;\n };\n\n ///// Complex tables parsing //////////////////////////////////\n\n // Parse a coverage table in a GSUB, GPOS or GDEF table.\n // https://www.microsoft.com/typography/OTSPEC/chapter2.htm\n // parser.offset must point to the start of the table containing the coverage.\n Parser.prototype.parseCoverage = function() {\n var startOffset = this.offset + this.relativeOffset;\n var format = this.parseUShort();\n var count = this.parseUShort();\n if (format === 1) {\n return {\n format: 1,\n glyphs: this.parseUShortList(count)\n };\n } else if (format === 2) {\n var ranges = new Array(count);\n for (var i = 0; i < count; i++) {\n ranges[i] = {\n start: this.parseUShort(),\n end: this.parseUShort(),\n index: this.parseUShort()\n };\n }\n return {\n format: 2,\n ranges: ranges\n };\n }\n throw new Error('0x' + startOffset.toString(16) + ': Coverage format must be 1 or 2.');\n };\n\n // Parse a Class Definition Table in a GSUB, GPOS or GDEF table.\n // https://www.microsoft.com/typography/OTSPEC/chapter2.htm\n Parser.prototype.parseClassDef = function() {\n var startOffset = this.offset + this.relativeOffset;\n var format = this.parseUShort();\n if (format === 1) {\n return {\n format: 1,\n startGlyph: this.parseUShort(),\n classes: this.parseUShortList()\n };\n } else if (format === 2) {\n return {\n format: 2,\n ranges: this.parseRecordList({\n start: Parser.uShort,\n end: Parser.uShort,\n classId: Parser.uShort\n })\n };\n }\n throw new Error('0x' + startOffset.toString(16) + ': ClassDef format must be 1 or 2.');\n };\n\n ///// Static methods ///////////////////////////////////\n // These convenience methods can be used as callbacks and should be called with \"this\" context set to a Parser instance.\n\n Parser.list = function(count, itemCallback) {\n return function() {\n return this.parseList(count, itemCallback);\n };\n };\n\n Parser.list32 = function(count, itemCallback) {\n return function() {\n return this.parseList32(count, itemCallback);\n };\n };\n\n Parser.recordList = function(count, recordDescription) {\n return function() {\n return this.parseRecordList(count, recordDescription);\n };\n };\n\n Parser.recordList32 = function(count, recordDescription) {\n return function() {\n return this.parseRecordList32(count, recordDescription);\n };\n };\n\n Parser.pointer = function(description) {\n return function() {\n return this.parsePointer(description);\n };\n };\n\n Parser.pointer32 = function(description) {\n return function() {\n return this.parsePointer32(description);\n };\n };\n\n Parser.tag = Parser.prototype.parseTag;\n Parser.byte = Parser.prototype.parseByte;\n Parser.uShort = Parser.offset16 = Parser.prototype.parseUShort;\n Parser.uShortList = Parser.prototype.parseUShortList;\n Parser.uLong = Parser.offset32 = Parser.prototype.parseULong;\n Parser.uLongList = Parser.prototype.parseULongList;\n Parser.struct = Parser.prototype.parseStruct;\n Parser.coverage = Parser.prototype.parseCoverage;\n Parser.classDef = Parser.prototype.parseClassDef;\n\n ///// Script, Feature, Lookup lists ///////////////////////////////////////////////\n // https://www.microsoft.com/typography/OTSPEC/chapter2.htm\n\n var langSysTable = {\n reserved: Parser.uShort,\n reqFeatureIndex: Parser.uShort,\n featureIndexes: Parser.uShortList\n };\n\n Parser.prototype.parseScriptList = function() {\n return this.parsePointer(Parser.recordList({\n tag: Parser.tag,\n script: Parser.pointer({\n defaultLangSys: Parser.pointer(langSysTable),\n langSysRecords: Parser.recordList({\n tag: Parser.tag,\n langSys: Parser.pointer(langSysTable)\n })\n })\n })) || [];\n };\n\n Parser.prototype.parseFeatureList = function() {\n return this.parsePointer(Parser.recordList({\n tag: Parser.tag,\n feature: Parser.pointer({\n featureParams: Parser.offset16,\n lookupListIndexes: Parser.uShortList\n })\n })) || [];\n };\n\n Parser.prototype.parseLookupList = function(lookupTableParsers) {\n return this.parsePointer(Parser.list(Parser.pointer(function() {\n var lookupType = this.parseUShort();\n check.argument(1 <= lookupType && lookupType <= 9, 'GPOS/GSUB lookup type ' + lookupType + ' unknown.');\n var lookupFlag = this.parseUShort();\n var useMarkFilteringSet = lookupFlag & 0x10;\n return {\n lookupType: lookupType,\n lookupFlag: lookupFlag,\n subtables: this.parseList(Parser.pointer(lookupTableParsers[lookupType])),\n markFilteringSet: useMarkFilteringSet ? this.parseUShort() : undefined\n };\n }))) || [];\n };\n\n Parser.prototype.parseFeatureVariationsList = function() {\n return this.parsePointer32(function() {\n var majorVersion = this.parseUShort();\n var minorVersion = this.parseUShort();\n check.argument(majorVersion === 1 && minorVersion < 1, 'GPOS/GSUB feature variations table unknown.');\n var featureVariations = this.parseRecordList32({\n conditionSetOffset: Parser.offset32,\n featureTableSubstitutionOffset: Parser.offset32\n });\n return featureVariations;\n }) || [];\n };\n\n var parse = {\n getByte: getByte,\n getCard8: getByte,\n getUShort: getUShort,\n getCard16: getUShort,\n getShort: getShort,\n getULong: getULong,\n getFixed: getFixed,\n getTag: getTag,\n getOffset: getOffset,\n getBytes: getBytes,\n bytesToString: bytesToString,\n Parser: Parser,\n };\n\n // The `cmap` table stores the mappings from characters to glyphs.\n\n function parseCmapTableFormat12(cmap, p) {\n //Skip reserved.\n p.parseUShort();\n\n // Length in bytes of the sub-tables.\n cmap.length = p.parseULong();\n cmap.language = p.parseULong();\n\n var groupCount;\n cmap.groupCount = groupCount = p.parseULong();\n cmap.glyphIndexMap = {};\n\n for (var i = 0; i < groupCount; i += 1) {\n var startCharCode = p.parseULong();\n var endCharCode = p.parseULong();\n var startGlyphId = p.parseULong();\n\n for (var c = startCharCode; c <= endCharCode; c += 1) {\n cmap.glyphIndexMap[c] = startGlyphId;\n startGlyphId++;\n }\n }\n }\n\n function parseCmapTableFormat4(cmap, p, data, start, offset) {\n // Length in bytes of the sub-tables.\n cmap.length = p.parseUShort();\n cmap.language = p.parseUShort();\n\n // segCount is stored x 2.\n var segCount;\n cmap.segCount = segCount = p.parseUShort() >> 1;\n\n // Skip searchRange, entrySelector, rangeShift.\n p.skip('uShort', 3);\n\n // The \"unrolled\" mapping from character codes to glyph indices.\n cmap.glyphIndexMap = {};\n var endCountParser = new parse.Parser(data, start + offset + 14);\n var startCountParser = new parse.Parser(data, start + offset + 16 + segCount * 2);\n var idDeltaParser = new parse.Parser(data, start + offset + 16 + segCount * 4);\n var idRangeOffsetParser = new parse.Parser(data, start + offset + 16 + segCount * 6);\n var glyphIndexOffset = start + offset + 16 + segCount * 8;\n for (var i = 0; i < segCount - 1; i += 1) {\n var glyphIndex = (void 0);\n var endCount = endCountParser.parseUShort();\n var startCount = startCountParser.parseUShort();\n var idDelta = idDeltaParser.parseShort();\n var idRangeOffset = idRangeOffsetParser.parseUShort();\n for (var c = startCount; c <= endCount; c += 1) {\n if (idRangeOffset !== 0) {\n // The idRangeOffset is relative to the current position in the idRangeOffset array.\n // Take the current offset in the idRangeOffset array.\n glyphIndexOffset = (idRangeOffsetParser.offset + idRangeOffsetParser.relativeOffset - 2);\n\n // Add the value of the idRangeOffset, which will move us into the glyphIndex array.\n glyphIndexOffset += idRangeOffset;\n\n // Then add the character index of the current segment, multiplied by 2 for USHORTs.\n glyphIndexOffset += (c - startCount) * 2;\n glyphIndex = parse.getUShort(data, glyphIndexOffset);\n if (glyphIndex !== 0) {\n glyphIndex = (glyphIndex + idDelta) & 0xFFFF;\n }\n } else {\n glyphIndex = (c + idDelta) & 0xFFFF;\n }\n\n cmap.glyphIndexMap[c] = glyphIndex;\n }\n }\n }\n\n // Parse the `cmap` table. This table stores the mappings from characters to glyphs.\n // There are many available formats, but we only support the Windows format 4 and 12.\n // This function returns a `CmapEncoding` object or null if no supported format could be found.\n function parseCmapTable(data, start) {\n var cmap = {};\n cmap.version = parse.getUShort(data, start);\n check.argument(cmap.version === 0, 'cmap table version should be 0.');\n\n // The cmap table can contain many sub-tables, each with their own format.\n // We're only interested in a \"platform 0\" (Unicode format) and \"platform 3\" (Windows format) table.\n cmap.numTables = parse.getUShort(data, start + 2);\n var offset = -1;\n for (var i = cmap.numTables - 1; i >= 0; i -= 1) {\n var platformId = parse.getUShort(data, start + 4 + (i * 8));\n var encodingId = parse.getUShort(data, start + 4 + (i * 8) + 2);\n if ((platformId === 3 && (encodingId === 0 || encodingId === 1 || encodingId === 10)) ||\n (platformId === 0 && (encodingId === 0 || encodingId === 1 || encodingId === 2 || encodingId === 3 || encodingId === 4))) {\n offset = parse.getULong(data, start + 4 + (i * 8) + 4);\n break;\n }\n }\n\n if (offset === -1) {\n // There is no cmap table in the font that we support.\n throw new Error('No valid cmap sub-tables found.');\n }\n\n var p = new parse.Parser(data, start + offset);\n cmap.format = p.parseUShort();\n\n if (cmap.format === 12) {\n parseCmapTableFormat12(cmap, p);\n } else if (cmap.format === 4) {\n parseCmapTableFormat4(cmap, p, data, start, offset);\n } else {\n throw new Error('Only format 4 and 12 cmap tables are supported (found format ' + cmap.format + ').');\n }\n\n return cmap;\n }\n\n function addSegment(t, code, glyphIndex) {\n t.segments.push({\n end: code,\n start: code,\n delta: -(code - glyphIndex),\n offset: 0,\n glyphIndex: glyphIndex\n });\n }\n\n function addTerminatorSegment(t) {\n t.segments.push({\n end: 0xFFFF,\n start: 0xFFFF,\n delta: 1,\n offset: 0\n });\n }\n\n // Make cmap table, format 4 by default, 12 if needed only\n function makeCmapTable(glyphs) {\n // Plan 0 is the base Unicode Plan but emojis, for example are on another plan, and needs cmap 12 format (with 32bit)\n var isPlan0Only = true;\n var i;\n\n // Check if we need to add cmap format 12 or if format 4 only is fine\n for (i = glyphs.length - 1; i > 0; i -= 1) {\n var g = glyphs.get(i);\n if (g.unicode > 65535) {\n console.log('Adding CMAP format 12 (needed!)');\n isPlan0Only = false;\n break;\n }\n }\n\n var cmapTable = [\n {name: 'version', type: 'USHORT', value: 0},\n {name: 'numTables', type: 'USHORT', value: isPlan0Only ? 1 : 2},\n\n // CMAP 4 header\n {name: 'platformID', type: 'USHORT', value: 3},\n {name: 'encodingID', type: 'USHORT', value: 1},\n {name: 'offset', type: 'ULONG', value: isPlan0Only ? 12 : (12 + 8)}\n ];\n\n if (!isPlan0Only)\n { cmapTable = cmapTable.concat([\n // CMAP 12 header\n {name: 'cmap12PlatformID', type: 'USHORT', value: 3}, // We encode only for PlatformID = 3 (Windows) because it is supported everywhere\n {name: 'cmap12EncodingID', type: 'USHORT', value: 10},\n {name: 'cmap12Offset', type: 'ULONG', value: 0}\n ]); }\n\n cmapTable = cmapTable.concat([\n // CMAP 4 Subtable\n {name: 'format', type: 'USHORT', value: 4},\n {name: 'cmap4Length', type: 'USHORT', value: 0},\n {name: 'language', type: 'USHORT', value: 0},\n {name: 'segCountX2', type: 'USHORT', value: 0},\n {name: 'searchRange', type: 'USHORT', value: 0},\n {name: 'entrySelector', type: 'USHORT', value: 0},\n {name: 'rangeShift', type: 'USHORT', value: 0}\n ]);\n\n var t = new table.Table('cmap', cmapTable);\n\n t.segments = [];\n for (i = 0; i < glyphs.length; i += 1) {\n var glyph = glyphs.get(i);\n for (var j = 0; j < glyph.unicodes.length; j += 1) {\n addSegment(t, glyph.unicodes[j], i);\n }\n\n t.segments = t.segments.sort(function (a, b) {\n return a.start - b.start;\n });\n }\n\n addTerminatorSegment(t);\n\n var segCount = t.segments.length;\n var segCountToRemove = 0;\n\n // CMAP 4\n // Set up parallel segment arrays.\n var endCounts = [];\n var startCounts = [];\n var idDeltas = [];\n var idRangeOffsets = [];\n var glyphIds = [];\n\n // CMAP 12\n var cmap12Groups = [];\n\n // Reminder this loop is not following the specification at 100%\n // The specification -> find suites of characters and make a group\n // Here we're doing one group for each letter\n // Doing as the spec can save 8 times (or more) space\n for (i = 0; i < segCount; i += 1) {\n var segment = t.segments[i];\n\n // CMAP 4\n if (segment.end <= 65535 && segment.start <= 65535) {\n endCounts = endCounts.concat({name: 'end_' + i, type: 'USHORT', value: segment.end});\n startCounts = startCounts.concat({name: 'start_' + i, type: 'USHORT', value: segment.start});\n idDeltas = idDeltas.concat({name: 'idDelta_' + i, type: 'SHORT', value: segment.delta});\n idRangeOffsets = idRangeOffsets.concat({name: 'idRangeOffset_' + i, type: 'USHORT', value: segment.offset});\n if (segment.glyphId !== undefined) {\n glyphIds = glyphIds.concat({name: 'glyph_' + i, type: 'USHORT', value: segment.glyphId});\n }\n } else {\n // Skip Unicode > 65535 (16bit unsigned max) for CMAP 4, will be added in CMAP 12\n segCountToRemove += 1;\n }\n\n // CMAP 12\n // Skip Terminator Segment\n if (!isPlan0Only && segment.glyphIndex !== undefined) {\n cmap12Groups = cmap12Groups.concat({name: 'cmap12Start_' + i, type: 'ULONG', value: segment.start});\n cmap12Groups = cmap12Groups.concat({name: 'cmap12End_' + i, type: 'ULONG', value: segment.end});\n cmap12Groups = cmap12Groups.concat({name: 'cmap12Glyph_' + i, type: 'ULONG', value: segment.glyphIndex});\n }\n }\n\n // CMAP 4 Subtable\n t.segCountX2 = (segCount - segCountToRemove) * 2;\n t.searchRange = Math.pow(2, Math.floor(Math.log((segCount - segCountToRemove)) / Math.log(2))) * 2;\n t.entrySelector = Math.log(t.searchRange / 2) / Math.log(2);\n t.rangeShift = t.segCountX2 - t.searchRange;\n\n t.fields = t.fields.concat(endCounts);\n t.fields.push({name: 'reservedPad', type: 'USHORT', value: 0});\n t.fields = t.fields.concat(startCounts);\n t.fields = t.fields.concat(idDeltas);\n t.fields = t.fields.concat(idRangeOffsets);\n t.fields = t.fields.concat(glyphIds);\n\n t.cmap4Length = 14 + // Subtable header\n endCounts.length * 2 +\n 2 + // reservedPad\n startCounts.length * 2 +\n idDeltas.length * 2 +\n idRangeOffsets.length * 2 +\n glyphIds.length * 2;\n\n if (!isPlan0Only) {\n // CMAP 12 Subtable\n var cmap12Length = 16 + // Subtable header\n cmap12Groups.length * 4;\n\n t.cmap12Offset = 12 + (2 * 2) + 4 + t.cmap4Length;\n t.fields = t.fields.concat([\n {name: 'cmap12Format', type: 'USHORT', value: 12},\n {name: 'cmap12Reserved', type: 'USHORT', value: 0},\n {name: 'cmap12Length', type: 'ULONG', value: cmap12Length},\n {name: 'cmap12Language', type: 'ULONG', value: 0},\n {name: 'cmap12nGroups', type: 'ULONG', value: cmap12Groups.length / 3}\n ]);\n\n t.fields = t.fields.concat(cmap12Groups);\n }\n\n return t;\n }\n\n var cmap = { parse: parseCmapTable, make: makeCmapTable };\n\n // Glyph encoding\n\n var cffStandardStrings = [\n '.notdef', 'space', 'exclam', 'quotedbl', 'numbersign', 'dollar', 'percent', 'ampersand', 'quoteright',\n 'parenleft', 'parenright', 'asterisk', 'plus', 'comma', 'hyphen', 'period', 'slash', 'zero', 'one', 'two',\n 'three', 'four', 'five', 'six', 'seven', 'eight', 'nine', 'colon', 'semicolon', 'less', 'equal', 'greater',\n 'question', 'at', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S',\n 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', 'bracketleft', 'backslash', 'bracketright', 'asciicircum', 'underscore',\n 'quoteleft', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't',\n 'u', 'v', 'w', 'x', 'y', 'z', 'braceleft', 'bar', 'braceright', 'asciitilde', 'exclamdown', 'cent', 'sterling',\n 'fraction', 'yen', 'florin', 'section', 'currency', 'quotesingle', 'quotedblleft', 'guillemotleft',\n 'guilsinglleft', 'guilsinglright', 'fi', 'fl', 'endash', 'dagger', 'daggerdbl', 'periodcentered', 'paragraph',\n 'bullet', 'quotesinglbase', 'quotedblbase', 'quotedblright', 'guillemotright', 'ellipsis', 'perthousand',\n 'questiondown', 'grave', 'acute', 'circumflex', 'tilde', 'macron', 'breve', 'dotaccent', 'dieresis', 'ring',\n 'cedilla', 'hungarumlaut', 'ogonek', 'caron', 'emdash', 'AE', 'ordfeminine', 'Lslash', 'Oslash', 'OE',\n 'ordmasculine', 'ae', 'dotlessi', 'lslash', 'oslash', 'oe', 'germandbls', 'onesuperior', 'logicalnot', 'mu',\n 'trademark', 'Eth', 'onehalf', 'plusminus', 'Thorn', 'onequarter', 'divide', 'brokenbar', 'degree', 'thorn',\n 'threequarters', 'twosuperior', 'registered', 'minus', 'eth', 'multiply', 'threesuperior', 'copyright',\n 'Aacute', 'Acircumflex', 'Adieresis', 'Agrave', 'Aring', 'Atilde', 'Ccedilla', 'Eacute', 'Ecircumflex',\n 'Edieresis', 'Egrave', 'Iacute', 'Icircumflex', 'Idieresis', 'Igrave', 'Ntilde', 'Oacute', 'Ocircumflex',\n 'Odieresis', 'Ograve', 'Otilde', 'Scaron', 'Uacute', 'Ucircumflex', 'Udieresis', 'Ugrave', 'Yacute',\n 'Ydieresis', 'Zcaron', 'aacute', 'acircumflex', 'adieresis', 'agrave', 'aring', 'atilde', 'ccedilla', 'eacute',\n 'ecircumflex', 'edieresis', 'egrave', 'iacute', 'icircumflex', 'idieresis', 'igrave', 'ntilde', 'oacute',\n 'ocircumflex', 'odieresis', 'ograve', 'otilde', 'scaron', 'uacute', 'ucircumflex', 'udieresis', 'ugrave',\n 'yacute', 'ydieresis', 'zcaron', 'exclamsmall', 'Hungarumlautsmall', 'dollaroldstyle', 'dollarsuperior',\n 'ampersandsmall', 'Acutesmall', 'parenleftsuperior', 'parenrightsuperior', '266 ff', 'onedotenleader',\n 'zerooldstyle', 'oneoldstyle', 'twooldstyle', 'threeoldstyle', 'fouroldstyle', 'fiveoldstyle', 'sixoldstyle',\n 'sevenoldstyle', 'eightoldstyle', 'nineoldstyle', 'commasuperior', 'threequartersemdash', 'periodsuperior',\n 'questionsmall', 'asuperior', 'bsuperior', 'centsuperior', 'dsuperior', 'esuperior', 'isuperior', 'lsuperior',\n 'msuperior', 'nsuperior', 'osuperior', 'rsuperior', 'ssuperior', 'tsuperior', 'ff', 'ffi', 'ffl',\n 'parenleftinferior', 'parenrightinferior', 'Circumflexsmall', 'hyphensuperior', 'Gravesmall', 'Asmall',\n 'Bsmall', 'Csmall', 'Dsmall', 'Esmall', 'Fsmall', 'Gsmall', 'Hsmall', 'Ismall', 'Jsmall', 'Ksmall', 'Lsmall',\n 'Msmall', 'Nsmall', 'Osmall', 'Psmall', 'Qsmall', 'Rsmall', 'Ssmall', 'Tsmall', 'Usmall', 'Vsmall', 'Wsmall',\n 'Xsmall', 'Ysmall', 'Zsmall', 'colonmonetary', 'onefitted', 'rupiah', 'Tildesmall', 'exclamdownsmall',\n 'centoldstyle', 'Lslashsmall', 'Scaronsmall', 'Zcaronsmall', 'Dieresissmall', 'Brevesmall', 'Caronsmall',\n 'Dotaccentsmall', 'Macronsmall', 'figuredash', 'hypheninferior', 'Ogoneksmall', 'Ringsmall', 'Cedillasmall',\n 'questiondownsmall', 'oneeighth', 'threeeighths', 'fiveeighths', 'seveneighths', 'onethird', 'twothirds',\n 'zerosuperior', 'foursuperior', 'fivesuperior', 'sixsuperior', 'sevensuperior', 'eightsuperior', 'ninesuperior',\n 'zeroinferior', 'oneinferior', 'twoinferior', 'threeinferior', 'fourinferior', 'fiveinferior', 'sixinferior',\n 'seveninferior', 'eightinferior', 'nineinferior', 'centinferior', 'dollarinferior', 'periodinferior',\n 'commainferior', 'Agravesmall', 'Aacutesmall', 'Acircumflexsmall', 'Atildesmall', 'Adieresissmall',\n 'Aringsmall', 'AEsmall', 'Ccedillasmall', 'Egravesmall', 'Eacutesmall', 'Ecircumflexsmall', 'Edieresissmall',\n 'Igravesmall', 'Iacutesmall', 'Icircumflexsmall', 'Idieresissmall', 'Ethsmall', 'Ntildesmall', 'Ogravesmall',\n 'Oacutesmall', 'Ocircumflexsmall', 'Otildesmall', 'Odieresissmall', 'OEsmall', 'Oslashsmall', 'Ugravesmall',\n 'Uacutesmall', 'Ucircumflexsmall', 'Udieresissmall', 'Yacutesmall', 'Thornsmall', 'Ydieresissmall', '001.000',\n '001.001', '001.002', '001.003', 'Black', 'Bold', 'Book', 'Light', 'Medium', 'Regular', 'Roman', 'Semibold'];\n\n var cffStandardEncoding = [\n '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '',\n '', '', '', '', 'space', 'exclam', 'quotedbl', 'numbersign', 'dollar', 'percent', 'ampersand', 'quoteright',\n 'parenleft', 'parenright', 'asterisk', 'plus', 'comma', 'hyphen', 'period', 'slash', 'zero', 'one', 'two',\n 'three', 'four', 'five', 'six', 'seven', 'eight', 'nine', 'colon', 'semicolon', 'less', 'equal', 'greater',\n 'question', 'at', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S',\n 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', 'bracketleft', 'backslash', 'bracketright', 'asciicircum', 'underscore',\n 'quoteleft', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't',\n 'u', 'v', 'w', 'x', 'y', 'z', 'braceleft', 'bar', 'braceright', 'asciitilde', '', '', '', '', '', '', '', '',\n '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '',\n 'exclamdown', 'cent', 'sterling', 'fraction', 'yen', 'florin', 'section', 'currency', 'quotesingle',\n 'quotedblleft', 'guillemotleft', 'guilsinglleft', 'guilsinglright', 'fi', 'fl', '', 'endash', 'dagger',\n 'daggerdbl', 'periodcentered', '', 'paragraph', 'bullet', 'quotesinglbase', 'quotedblbase', 'quotedblright',\n 'guillemotright', 'ellipsis', 'perthousand', '', 'questiondown', '', 'grave', 'acute', 'circumflex', 'tilde',\n 'macron', 'breve', 'dotaccent', 'dieresis', '', 'ring', 'cedilla', '', 'hungarumlaut', 'ogonek', 'caron',\n 'emdash', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', 'AE', '', 'ordfeminine', '', '', '',\n '', 'Lslash', 'Oslash', 'OE', 'ordmasculine', '', '', '', '', '', 'ae', '', '', '', 'dotlessi', '', '',\n 'lslash', 'oslash', 'oe', 'germandbls'];\n\n var cffExpertEncoding = [\n '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '',\n '', '', '', '', 'space', 'exclamsmall', 'Hungarumlautsmall', '', 'dollaroldstyle', 'dollarsuperior',\n 'ampersandsmall', 'Acutesmall', 'parenleftsuperior', 'parenrightsuperior', 'twodotenleader', 'onedotenleader',\n 'comma', 'hyphen', 'period', 'fraction', 'zerooldstyle', 'oneoldstyle', 'twooldstyle', 'threeoldstyle',\n 'fouroldstyle', 'fiveoldstyle', 'sixoldstyle', 'sevenoldstyle', 'eightoldstyle', 'nineoldstyle', 'colon',\n 'semicolon', 'commasuperior', 'threequartersemdash', 'periodsuperior', 'questionsmall', '', 'asuperior',\n 'bsuperior', 'centsuperior', 'dsuperior', 'esuperior', '', '', 'isuperior', '', '', 'lsuperior', 'msuperior',\n 'nsuperior', 'osuperior', '', '', 'rsuperior', 'ssuperior', 'tsuperior', '', 'ff', 'fi', 'fl', 'ffi', 'ffl',\n 'parenleftinferior', '', 'parenrightinferior', 'Circumflexsmall', 'hyphensuperior', 'Gravesmall', 'Asmall',\n 'Bsmall', 'Csmall', 'Dsmall', 'Esmall', 'Fsmall', 'Gsmall', 'Hsmall', 'Ismall', 'Jsmall', 'Ksmall', 'Lsmall',\n 'Msmall', 'Nsmall', 'Osmall', 'Psmall', 'Qsmall', 'Rsmall', 'Ssmall', 'Tsmall', 'Usmall', 'Vsmall', 'Wsmall',\n 'Xsmall', 'Ysmall', 'Zsmall', 'colonmonetary', 'onefitted', 'rupiah', 'Tildesmall', '', '', '', '', '', '', '',\n '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '',\n 'exclamdownsmall', 'centoldstyle', 'Lslashsmall', '', '', 'Scaronsmall', 'Zcaronsmall', 'Dieresissmall',\n 'Brevesmall', 'Caronsmall', '', 'Dotaccentsmall', '', '', 'Macronsmall', '', '', 'figuredash', 'hypheninferior',\n '', '', 'Ogoneksmall', 'Ringsmall', 'Cedillasmall', '', '', '', 'onequarter', 'onehalf', 'threequarters',\n 'questiondownsmall', 'oneeighth', 'threeeighths', 'fiveeighths', 'seveneighths', 'onethird', 'twothirds', '',\n '', 'zerosuperior', 'onesuperior', 'twosuperior', 'threesuperior', 'foursuperior', 'fivesuperior',\n 'sixsuperior', 'sevensuperior', 'eightsuperior', 'ninesuperior', 'zeroinferior', 'oneinferior', 'twoinferior',\n 'threeinferior', 'fourinferior', 'fiveinferior', 'sixinferior', 'seveninferior', 'eightinferior',\n 'nineinferior', 'centinferior', 'dollarinferior', 'periodinferior', 'commainferior', 'Agravesmall',\n 'Aacutesmall', 'Acircumflexsmall', 'Atildesmall', 'Adieresissmall', 'Aringsmall', 'AEsmall', 'Ccedillasmall',\n 'Egravesmall', 'Eacutesmall', 'Ecircumflexsmall', 'Edieresissmall', 'Igravesmall', 'Iacutesmall',\n 'Icircumflexsmall', 'Idieresissmall', 'Ethsmall', 'Ntildesmall', 'Ogravesmall', 'Oacutesmall',\n 'Ocircumflexsmall', 'Otildesmall', 'Odieresissmall', 'OEsmall', 'Oslashsmall', 'Ugravesmall', 'Uacutesmall',\n 'Ucircumflexsmall', 'Udieresissmall', 'Yacutesmall', 'Thornsmall', 'Ydieresissmall'];\n\n var standardNames = [\n '.notdef', '.null', 'nonmarkingreturn', 'space', 'exclam', 'quotedbl', 'numbersign', 'dollar', 'percent',\n 'ampersand', 'quotesingle', 'parenleft', 'parenright', 'asterisk', 'plus', 'comma', 'hyphen', 'period', 'slash',\n 'zero', 'one', 'two', 'three', 'four', 'five', 'six', 'seven', 'eight', 'nine', 'colon', 'semicolon', 'less',\n 'equal', 'greater', 'question', 'at', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O',\n 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', 'bracketleft', 'backslash', 'bracketright',\n 'asciicircum', 'underscore', 'grave', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o',\n 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', 'braceleft', 'bar', 'braceright', 'asciitilde',\n 'Adieresis', 'Aring', 'Ccedilla', 'Eacute', 'Ntilde', 'Odieresis', 'Udieresis', 'aacute', 'agrave',\n 'acircumflex', 'adieresis', 'atilde', 'aring', 'ccedilla', 'eacute', 'egrave', 'ecircumflex', 'edieresis',\n 'iacute', 'igrave', 'icircumflex', 'idieresis', 'ntilde', 'oacute', 'ograve', 'ocircumflex', 'odieresis',\n 'otilde', 'uacute', 'ugrave', 'ucircumflex', 'udieresis', 'dagger', 'degree', 'cent', 'sterling', 'section',\n 'bullet', 'paragraph', 'germandbls', 'registered', 'copyright', 'trademark', 'acute', 'dieresis', 'notequal',\n 'AE', 'Oslash', 'infinity', 'plusminus', 'lessequal', 'greaterequal', 'yen', 'mu', 'partialdiff', 'summation',\n 'product', 'pi', 'integral', 'ordfeminine', 'ordmasculine', 'Omega', 'ae', 'oslash', 'questiondown',\n 'exclamdown', 'logicalnot', 'radical', 'florin', 'approxequal', 'Delta', 'guillemotleft', 'guillemotright',\n 'ellipsis', 'nonbreakingspace', 'Agrave', 'Atilde', 'Otilde', 'OE', 'oe', 'endash', 'emdash', 'quotedblleft',\n 'quotedblright', 'quoteleft', 'quoteright', 'divide', 'lozenge', 'ydieresis', 'Ydieresis', 'fraction',\n 'currency', 'guilsinglleft', 'guilsinglright', 'fi', 'fl', 'daggerdbl', 'periodcentered', 'quotesinglbase',\n 'quotedblbase', 'perthousand', 'Acircumflex', 'Ecircumflex', 'Aacute', 'Edieresis', 'Egrave', 'Iacute',\n 'Icircumflex', 'Idieresis', 'Igrave', 'Oacute', 'Ocircumflex', 'apple', 'Ograve', 'Uacute', 'Ucircumflex',\n 'Ugrave', 'dotlessi', 'circumflex', 'tilde', 'macron', 'breve', 'dotaccent', 'ring', 'cedilla', 'hungarumlaut',\n 'ogonek', 'caron', 'Lslash', 'lslash', 'Scaron', 'scaron', 'Zcaron', 'zcaron', 'brokenbar', 'Eth', 'eth',\n 'Yacute', 'yacute', 'Thorn', 'thorn', 'minus', 'multiply', 'onesuperior', 'twosuperior', 'threesuperior',\n 'onehalf', 'onequarter', 'threequarters', 'franc', 'Gbreve', 'gbreve', 'Idotaccent', 'Scedilla', 'scedilla',\n 'Cacute', 'cacute', 'Ccaron', 'ccaron', 'dcroat'];\n\n /**\n * This is the encoding used for fonts created from scratch.\n * It loops through all glyphs and finds the appropriate unicode value.\n * Since it's linear time, other encodings will be faster.\n * @exports opentype.DefaultEncoding\n * @class\n * @constructor\n * @param {opentype.Font}\n */\n function DefaultEncoding(font) {\n this.font = font;\n }\n\n DefaultEncoding.prototype.charToGlyphIndex = function(c) {\n var code = c.codePointAt(0);\n var glyphs = this.font.glyphs;\n if (glyphs) {\n for (var i = 0; i < glyphs.length; i += 1) {\n var glyph = glyphs.get(i);\n for (var j = 0; j < glyph.unicodes.length; j += 1) {\n if (glyph.unicodes[j] === code) {\n return i;\n }\n }\n }\n }\n return null;\n };\n\n /**\n * @exports opentype.CmapEncoding\n * @class\n * @constructor\n * @param {Object} cmap - a object with the cmap encoded data\n */\n function CmapEncoding(cmap) {\n this.cmap = cmap;\n }\n\n /**\n * @param {string} c - the character\n * @return {number} The glyph index.\n */\n CmapEncoding.prototype.charToGlyphIndex = function(c) {\n return this.cmap.glyphIndexMap[c.codePointAt(0)] || 0;\n };\n\n /**\n * @exports opentype.CffEncoding\n * @class\n * @constructor\n * @param {string} encoding - The encoding\n * @param {Array} charset - The character set.\n */\n function CffEncoding(encoding, charset) {\n this.encoding = encoding;\n this.charset = charset;\n }\n\n /**\n * @param {string} s - The character\n * @return {number} The index.\n */\n CffEncoding.prototype.charToGlyphIndex = function(s) {\n var code = s.codePointAt(0);\n var charName = this.encoding[code];\n return this.charset.indexOf(charName);\n };\n\n /**\n * @exports opentype.GlyphNames\n * @class\n * @constructor\n * @param {Object} post\n */\n function GlyphNames(post) {\n switch (post.version) {\n case 1:\n this.names = standardNames.slice();\n break;\n case 2:\n this.names = new Array(post.numberOfGlyphs);\n for (var i = 0; i < post.numberOfGlyphs; i++) {\n if (post.glyphNameIndex[i] < standardNames.length) {\n this.names[i] = standardNames[post.glyphNameIndex[i]];\n } else {\n this.names[i] = post.names[post.glyphNameIndex[i] - standardNames.length];\n }\n }\n\n break;\n case 2.5:\n this.names = new Array(post.numberOfGlyphs);\n for (var i$1 = 0; i$1 < post.numberOfGlyphs; i$1++) {\n this.names[i$1] = standardNames[i$1 + post.glyphNameIndex[i$1]];\n }\n\n break;\n case 3:\n this.names = [];\n break;\n default:\n this.names = [];\n break;\n }\n }\n\n /**\n * Gets the index of a glyph by name.\n * @param {string} name - The glyph name\n * @return {number} The index\n */\n GlyphNames.prototype.nameToGlyphIndex = function(name) {\n return this.names.indexOf(name);\n };\n\n /**\n * @param {number} gid\n * @return {string}\n */\n GlyphNames.prototype.glyphIndexToName = function(gid) {\n return this.names[gid];\n };\n\n function addGlyphNamesAll(font) {\n var glyph;\n var glyphIndexMap = font.tables.cmap.glyphIndexMap;\n var charCodes = Object.keys(glyphIndexMap);\n\n for (var i = 0; i < charCodes.length; i += 1) {\n var c = charCodes[i];\n var glyphIndex = glyphIndexMap[c];\n glyph = font.glyphs.get(glyphIndex);\n glyph.addUnicode(parseInt(c));\n }\n\n for (var i$1 = 0; i$1 < font.glyphs.length; i$1 += 1) {\n glyph = font.glyphs.get(i$1);\n if (font.cffEncoding) {\n if (font.isCIDFont) {\n glyph.name = 'gid' + i$1;\n } else {\n glyph.name = font.cffEncoding.charset[i$1];\n }\n } else if (font.glyphNames.names) {\n glyph.name = font.glyphNames.glyphIndexToName(i$1);\n }\n }\n }\n\n function addGlyphNamesToUnicodeMap(font) {\n font._IndexToUnicodeMap = {};\n\n var glyphIndexMap = font.tables.cmap.glyphIndexMap;\n var charCodes = Object.keys(glyphIndexMap);\n\n for (var i = 0; i < charCodes.length; i += 1) {\n var c = charCodes[i];\n var glyphIndex = glyphIndexMap[c];\n if (font._IndexToUnicodeMap[glyphIndex] === undefined) {\n font._IndexToUnicodeMap[glyphIndex] = {\n unicodes: [parseInt(c)]\n };\n } else {\n font._IndexToUnicodeMap[glyphIndex].unicodes.push(parseInt(c));\n }\n }\n }\n\n /**\n * @alias opentype.addGlyphNames\n * @param {opentype.Font}\n * @param {Object}\n */\n function addGlyphNames(font, opt) {\n if (opt.lowMemory) {\n addGlyphNamesToUnicodeMap(font);\n } else {\n addGlyphNamesAll(font);\n }\n }\n\n // Drawing utility functions.\n\n // Draw a line on the given context from point `x1,y1` to point `x2,y2`.\n function line(ctx, x1, y1, x2, y2) {\n ctx.beginPath();\n ctx.moveTo(x1, y1);\n ctx.lineTo(x2, y2);\n ctx.stroke();\n }\n\n var draw = { line: line };\n\n // The Glyph object\n // import glyf from './tables/glyf' Can't be imported here, because it's a circular dependency\n\n function getPathDefinition(glyph, path) {\n var _path = path || new Path();\n return {\n configurable: true,\n\n get: function() {\n if (typeof _path === 'function') {\n _path = _path();\n }\n\n return _path;\n },\n\n set: function(p) {\n _path = p;\n }\n };\n }\n /**\n * @typedef GlyphOptions\n * @type Object\n * @property {string} [name] - The glyph name\n * @property {number} [unicode]\n * @property {Array} [unicodes]\n * @property {number} [xMin]\n * @property {number} [yMin]\n * @property {number} [xMax]\n * @property {number} [yMax]\n * @property {number} [advanceWidth]\n */\n\n // A Glyph is an individual mark that often corresponds to a character.\n // Some glyphs, such as ligatures, are a combination of many characters.\n // Glyphs are the basic building blocks of a font.\n //\n // The `Glyph` class contains utility methods for drawing the path and its points.\n /**\n * @exports opentype.Glyph\n * @class\n * @param {GlyphOptions}\n * @constructor\n */\n function Glyph(options) {\n // By putting all the code on a prototype function (which is only declared once)\n // we reduce the memory requirements for larger fonts by some 2%\n this.bindConstructorValues(options);\n }\n\n /**\n * @param {GlyphOptions}\n */\n Glyph.prototype.bindConstructorValues = function(options) {\n this.index = options.index || 0;\n\n // These three values cannot be deferred for memory optimization:\n this.name = options.name || null;\n this.unicode = options.unicode || undefined;\n this.unicodes = options.unicodes || options.unicode !== undefined ? [options.unicode] : [];\n\n // But by binding these values only when necessary, we reduce can\n // the memory requirements by almost 3% for larger fonts.\n if ('xMin' in options) {\n this.xMin = options.xMin;\n }\n\n if ('yMin' in options) {\n this.yMin = options.yMin;\n }\n\n if ('xMax' in options) {\n this.xMax = options.xMax;\n }\n\n if ('yMax' in options) {\n this.yMax = options.yMax;\n }\n\n if ('advanceWidth' in options) {\n this.advanceWidth = options.advanceWidth;\n }\n\n // The path for a glyph is the most memory intensive, and is bound as a value\n // with a getter/setter to ensure we actually do path parsing only once the\n // path is actually needed by anything.\n Object.defineProperty(this, 'path', getPathDefinition(this, options.path));\n };\n\n /**\n * @param {number}\n */\n Glyph.prototype.addUnicode = function(unicode) {\n if (this.unicodes.length === 0) {\n this.unicode = unicode;\n }\n\n this.unicodes.push(unicode);\n };\n\n /**\n * Calculate the minimum bounding box for this glyph.\n * @return {opentype.BoundingBox}\n */\n Glyph.prototype.getBoundingBox = function() {\n return this.path.getBoundingBox();\n };\n\n /**\n * Convert the glyph to a Path we can draw on a drawing context.\n * @param {number} [x=0] - Horizontal position of the beginning of the text.\n * @param {number} [y=0] - Vertical position of the *baseline* of the text.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n * @param {Object=} options - xScale, yScale to stretch the glyph.\n * @param {opentype.Font} if hinting is to be used, the font\n * @return {opentype.Path}\n */\n Glyph.prototype.getPath = function(x, y, fontSize, options, font) {\n x = x !== undefined ? x : 0;\n y = y !== undefined ? y : 0;\n fontSize = fontSize !== undefined ? fontSize : 72;\n var commands;\n var hPoints;\n if (!options) { options = { }; }\n var xScale = options.xScale;\n var yScale = options.yScale;\n\n if (options.hinting && font && font.hinting) {\n // in case of hinting, the hinting engine takes care\n // of scaling the points (not the path) before hinting.\n hPoints = this.path && font.hinting.exec(this, fontSize);\n // in case the hinting engine failed hPoints is undefined\n // and thus reverts to plain rending\n }\n\n if (hPoints) {\n // Call font.hinting.getCommands instead of `glyf.getPath(hPoints).commands` to avoid a circular dependency\n commands = font.hinting.getCommands(hPoints);\n x = Math.round(x);\n y = Math.round(y);\n // TODO in case of hinting xyScaling is not yet supported\n xScale = yScale = 1;\n } else {\n commands = this.path.commands;\n var scale = 1 / (this.path.unitsPerEm || 1000) * fontSize;\n if (xScale === undefined) { xScale = scale; }\n if (yScale === undefined) { yScale = scale; }\n }\n\n var p = new Path();\n for (var i = 0; i < commands.length; i += 1) {\n var cmd = commands[i];\n if (cmd.type === 'M') {\n p.moveTo(x + (cmd.x * xScale), y + (-cmd.y * yScale));\n } else if (cmd.type === 'L') {\n p.lineTo(x + (cmd.x * xScale), y + (-cmd.y * yScale));\n } else if (cmd.type === 'Q') {\n p.quadraticCurveTo(x + (cmd.x1 * xScale), y + (-cmd.y1 * yScale),\n x + (cmd.x * xScale), y + (-cmd.y * yScale));\n } else if (cmd.type === 'C') {\n p.curveTo(x + (cmd.x1 * xScale), y + (-cmd.y1 * yScale),\n x + (cmd.x2 * xScale), y + (-cmd.y2 * yScale),\n x + (cmd.x * xScale), y + (-cmd.y * yScale));\n } else if (cmd.type === 'Z') {\n p.closePath();\n }\n }\n\n return p;\n };\n\n /**\n * Split the glyph into contours.\n * This function is here for backwards compatibility, and to\n * provide raw access to the TrueType glyph outlines.\n * @return {Array}\n */\n Glyph.prototype.getContours = function() {\n if (this.points === undefined) {\n return [];\n }\n\n var contours = [];\n var currentContour = [];\n for (var i = 0; i < this.points.length; i += 1) {\n var pt = this.points[i];\n currentContour.push(pt);\n if (pt.lastPointOfContour) {\n contours.push(currentContour);\n currentContour = [];\n }\n }\n\n check.argument(currentContour.length === 0, 'There are still points left in the current contour.');\n return contours;\n };\n\n /**\n * Calculate the xMin/yMin/xMax/yMax/lsb/rsb for a Glyph.\n * @return {Object}\n */\n Glyph.prototype.getMetrics = function() {\n var commands = this.path.commands;\n var xCoords = [];\n var yCoords = [];\n for (var i = 0; i < commands.length; i += 1) {\n var cmd = commands[i];\n if (cmd.type !== 'Z') {\n xCoords.push(cmd.x);\n yCoords.push(cmd.y);\n }\n\n if (cmd.type === 'Q' || cmd.type === 'C') {\n xCoords.push(cmd.x1);\n yCoords.push(cmd.y1);\n }\n\n if (cmd.type === 'C') {\n xCoords.push(cmd.x2);\n yCoords.push(cmd.y2);\n }\n }\n\n var metrics = {\n xMin: Math.min.apply(null, xCoords),\n yMin: Math.min.apply(null, yCoords),\n xMax: Math.max.apply(null, xCoords),\n yMax: Math.max.apply(null, yCoords),\n leftSideBearing: this.leftSideBearing\n };\n\n if (!isFinite(metrics.xMin)) {\n metrics.xMin = 0;\n }\n\n if (!isFinite(metrics.xMax)) {\n metrics.xMax = this.advanceWidth;\n }\n\n if (!isFinite(metrics.yMin)) {\n metrics.yMin = 0;\n }\n\n if (!isFinite(metrics.yMax)) {\n metrics.yMax = 0;\n }\n\n metrics.rightSideBearing = this.advanceWidth - metrics.leftSideBearing - (metrics.xMax - metrics.xMin);\n return metrics;\n };\n\n /**\n * Draw the glyph on the given context.\n * @param {CanvasRenderingContext2D} ctx - A 2D drawing context, like Canvas.\n * @param {number} [x=0] - Horizontal position of the beginning of the text.\n * @param {number} [y=0] - Vertical position of the *baseline* of the text.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n * @param {Object=} options - xScale, yScale to stretch the glyph.\n */\n Glyph.prototype.draw = function(ctx, x, y, fontSize, options) {\n this.getPath(x, y, fontSize, options).draw(ctx);\n };\n\n /**\n * Draw the points of the glyph.\n * On-curve points will be drawn in blue, off-curve points will be drawn in red.\n * @param {CanvasRenderingContext2D} ctx - A 2D drawing context, like Canvas.\n * @param {number} [x=0] - Horizontal position of the beginning of the text.\n * @param {number} [y=0] - Vertical position of the *baseline* of the text.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n */\n Glyph.prototype.drawPoints = function(ctx, x, y, fontSize) {\n function drawCircles(l, x, y, scale) {\n ctx.beginPath();\n for (var j = 0; j < l.length; j += 1) {\n ctx.moveTo(x + (l[j].x * scale), y + (l[j].y * scale));\n ctx.arc(x + (l[j].x * scale), y + (l[j].y * scale), 2, 0, Math.PI * 2, false);\n }\n\n ctx.closePath();\n ctx.fill();\n }\n\n x = x !== undefined ? x : 0;\n y = y !== undefined ? y : 0;\n fontSize = fontSize !== undefined ? fontSize : 24;\n var scale = 1 / this.path.unitsPerEm * fontSize;\n\n var blueCircles = [];\n var redCircles = [];\n var path = this.path;\n for (var i = 0; i < path.commands.length; i += 1) {\n var cmd = path.commands[i];\n if (cmd.x !== undefined) {\n blueCircles.push({x: cmd.x, y: -cmd.y});\n }\n\n if (cmd.x1 !== undefined) {\n redCircles.push({x: cmd.x1, y: -cmd.y1});\n }\n\n if (cmd.x2 !== undefined) {\n redCircles.push({x: cmd.x2, y: -cmd.y2});\n }\n }\n\n ctx.fillStyle = 'blue';\n drawCircles(blueCircles, x, y, scale);\n ctx.fillStyle = 'red';\n drawCircles(redCircles, x, y, scale);\n };\n\n /**\n * Draw lines indicating important font measurements.\n * Black lines indicate the origin of the coordinate system (point 0,0).\n * Blue lines indicate the glyph bounding box.\n * Green line indicates the advance width of the glyph.\n * @param {CanvasRenderingContext2D} ctx - A 2D drawing context, like Canvas.\n * @param {number} [x=0] - Horizontal position of the beginning of the text.\n * @param {number} [y=0] - Vertical position of the *baseline* of the text.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n */\n Glyph.prototype.drawMetrics = function(ctx, x, y, fontSize) {\n var scale;\n x = x !== undefined ? x : 0;\n y = y !== undefined ? y : 0;\n fontSize = fontSize !== undefined ? fontSize : 24;\n scale = 1 / this.path.unitsPerEm * fontSize;\n ctx.lineWidth = 1;\n\n // Draw the origin\n ctx.strokeStyle = 'black';\n draw.line(ctx, x, -10000, x, 10000);\n draw.line(ctx, -10000, y, 10000, y);\n\n // This code is here due to memory optimization: by not using\n // defaults in the constructor, we save a notable amount of memory.\n var xMin = this.xMin || 0;\n var yMin = this.yMin || 0;\n var xMax = this.xMax || 0;\n var yMax = this.yMax || 0;\n var advanceWidth = this.advanceWidth || 0;\n\n // Draw the glyph box\n ctx.strokeStyle = 'blue';\n draw.line(ctx, x + (xMin * scale), -10000, x + (xMin * scale), 10000);\n draw.line(ctx, x + (xMax * scale), -10000, x + (xMax * scale), 10000);\n draw.line(ctx, -10000, y + (-yMin * scale), 10000, y + (-yMin * scale));\n draw.line(ctx, -10000, y + (-yMax * scale), 10000, y + (-yMax * scale));\n\n // Draw the advance width\n ctx.strokeStyle = 'green';\n draw.line(ctx, x + (advanceWidth * scale), -10000, x + (advanceWidth * scale), 10000);\n };\n\n // The GlyphSet object\n\n // Define a property on the glyph that depends on the path being loaded.\n function defineDependentProperty(glyph, externalName, internalName) {\n Object.defineProperty(glyph, externalName, {\n get: function() {\n // Request the path property to make sure the path is loaded.\n glyph.path; // jshint ignore:line\n return glyph[internalName];\n },\n set: function(newValue) {\n glyph[internalName] = newValue;\n },\n enumerable: true,\n configurable: true\n });\n }\n\n /**\n * A GlyphSet represents all glyphs available in the font, but modelled using\n * a deferred glyph loader, for retrieving glyphs only once they are absolutely\n * necessary, to keep the memory footprint down.\n * @exports opentype.GlyphSet\n * @class\n * @param {opentype.Font}\n * @param {Array}\n */\n function GlyphSet(font, glyphs) {\n this.font = font;\n this.glyphs = {};\n if (Array.isArray(glyphs)) {\n for (var i = 0; i < glyphs.length; i++) {\n var glyph = glyphs[i];\n glyph.path.unitsPerEm = font.unitsPerEm;\n this.glyphs[i] = glyph;\n }\n }\n\n this.length = (glyphs && glyphs.length) || 0;\n }\n\n /**\n * @param {number} index\n * @return {opentype.Glyph}\n */\n GlyphSet.prototype.get = function(index) {\n // this.glyphs[index] is 'undefined' when low memory mode is on. glyph is pushed on request only.\n if (this.glyphs[index] === undefined) {\n this.font._push(index);\n if (typeof this.glyphs[index] === 'function') {\n this.glyphs[index] = this.glyphs[index]();\n }\n\n var glyph = this.glyphs[index];\n var unicodeObj = this.font._IndexToUnicodeMap[index];\n\n if (unicodeObj) {\n for (var j = 0; j < unicodeObj.unicodes.length; j++)\n { glyph.addUnicode(unicodeObj.unicodes[j]); }\n }\n\n if (this.font.cffEncoding) {\n if (this.font.isCIDFont) {\n glyph.name = 'gid' + index;\n } else {\n glyph.name = this.font.cffEncoding.charset[index];\n }\n } else if (this.font.glyphNames.names) {\n glyph.name = this.font.glyphNames.glyphIndexToName(index);\n }\n\n this.glyphs[index].advanceWidth = this.font._hmtxTableData[index].advanceWidth;\n this.glyphs[index].leftSideBearing = this.font._hmtxTableData[index].leftSideBearing;\n } else {\n if (typeof this.glyphs[index] === 'function') {\n this.glyphs[index] = this.glyphs[index]();\n }\n }\n\n return this.glyphs[index];\n };\n\n /**\n * @param {number} index\n * @param {Object}\n */\n GlyphSet.prototype.push = function(index, loader) {\n this.glyphs[index] = loader;\n this.length++;\n };\n\n /**\n * @alias opentype.glyphLoader\n * @param {opentype.Font} font\n * @param {number} index\n * @return {opentype.Glyph}\n */\n function glyphLoader(font, index) {\n return new Glyph({index: index, font: font});\n }\n\n /**\n * Generate a stub glyph that can be filled with all metadata *except*\n * the \"points\" and \"path\" properties, which must be loaded only once\n * the glyph's path is actually requested for text shaping.\n * @alias opentype.ttfGlyphLoader\n * @param {opentype.Font} font\n * @param {number} index\n * @param {Function} parseGlyph\n * @param {Object} data\n * @param {number} position\n * @param {Function} buildPath\n * @return {opentype.Glyph}\n */\n function ttfGlyphLoader(font, index, parseGlyph, data, position, buildPath) {\n return function() {\n var glyph = new Glyph({index: index, font: font});\n\n glyph.path = function() {\n parseGlyph(glyph, data, position);\n var path = buildPath(font.glyphs, glyph);\n path.unitsPerEm = font.unitsPerEm;\n return path;\n };\n\n defineDependentProperty(glyph, 'xMin', '_xMin');\n defineDependentProperty(glyph, 'xMax', '_xMax');\n defineDependentProperty(glyph, 'yMin', '_yMin');\n defineDependentProperty(glyph, 'yMax', '_yMax');\n\n return glyph;\n };\n }\n /**\n * @alias opentype.cffGlyphLoader\n * @param {opentype.Font} font\n * @param {number} index\n * @param {Function} parseCFFCharstring\n * @param {string} charstring\n * @return {opentype.Glyph}\n */\n function cffGlyphLoader(font, index, parseCFFCharstring, charstring) {\n return function() {\n var glyph = new Glyph({index: index, font: font});\n\n glyph.path = function() {\n var path = parseCFFCharstring(font, glyph, charstring);\n path.unitsPerEm = font.unitsPerEm;\n return path;\n };\n\n return glyph;\n };\n }\n\n var glyphset = { GlyphSet: GlyphSet, glyphLoader: glyphLoader, ttfGlyphLoader: ttfGlyphLoader, cffGlyphLoader: cffGlyphLoader };\n\n // The `CFF` table contains the glyph outlines in PostScript format.\n\n // Custom equals function that can also check lists.\n function equals$1(a, b) {\n if (a === b) {\n return true;\n } else if (Array.isArray(a) && Array.isArray(b)) {\n if (a.length !== b.length) {\n return false;\n }\n\n for (var i = 0; i < a.length; i += 1) {\n if (!equals$1(a[i], b[i])) {\n return false;\n }\n }\n\n return true;\n } else {\n return false;\n }\n }\n\n // Subroutines are encoded using the negative half of the number space.\n // See type 2 chapter 4.7 \"Subroutine operators\".\n function calcCFFSubroutineBias(subrs) {\n var bias;\n if (subrs.length < 1240) {\n bias = 107;\n } else if (subrs.length < 33900) {\n bias = 1131;\n } else {\n bias = 32768;\n }\n\n return bias;\n }\n\n // Parse a `CFF` INDEX array.\n // An index array consists of a list of offsets, then a list of objects at those offsets.\n function parseCFFIndex(data, start, conversionFn) {\n var offsets = [];\n var objects = [];\n var count = parse.getCard16(data, start);\n var objectOffset;\n var endOffset;\n if (count !== 0) {\n var offsetSize = parse.getByte(data, start + 2);\n objectOffset = start + ((count + 1) * offsetSize) + 2;\n var pos = start + 3;\n for (var i = 0; i < count + 1; i += 1) {\n offsets.push(parse.getOffset(data, pos, offsetSize));\n pos += offsetSize;\n }\n\n // The total size of the index array is 4 header bytes + the value of the last offset.\n endOffset = objectOffset + offsets[count];\n } else {\n endOffset = start + 2;\n }\n\n for (var i$1 = 0; i$1 < offsets.length - 1; i$1 += 1) {\n var value = parse.getBytes(data, objectOffset + offsets[i$1], objectOffset + offsets[i$1 + 1]);\n if (conversionFn) {\n value = conversionFn(value);\n }\n\n objects.push(value);\n }\n\n return {objects: objects, startOffset: start, endOffset: endOffset};\n }\n\n function parseCFFIndexLowMemory(data, start) {\n var offsets = [];\n var count = parse.getCard16(data, start);\n var objectOffset;\n var endOffset;\n if (count !== 0) {\n var offsetSize = parse.getByte(data, start + 2);\n objectOffset = start + ((count + 1) * offsetSize) + 2;\n var pos = start + 3;\n for (var i = 0; i < count + 1; i += 1) {\n offsets.push(parse.getOffset(data, pos, offsetSize));\n pos += offsetSize;\n }\n\n // The total size of the index array is 4 header bytes + the value of the last offset.\n endOffset = objectOffset + offsets[count];\n } else {\n endOffset = start + 2;\n }\n\n return {offsets: offsets, startOffset: start, endOffset: endOffset};\n }\n function getCffIndexObject(i, offsets, data, start, conversionFn) {\n var count = parse.getCard16(data, start);\n var objectOffset = 0;\n if (count !== 0) {\n var offsetSize = parse.getByte(data, start + 2);\n objectOffset = start + ((count + 1) * offsetSize) + 2;\n }\n\n var value = parse.getBytes(data, objectOffset + offsets[i], objectOffset + offsets[i + 1]);\n if (conversionFn) {\n value = conversionFn(value);\n }\n return value;\n }\n\n // Parse a `CFF` DICT real value.\n function parseFloatOperand(parser) {\n var s = '';\n var eof = 15;\n var lookup = ['0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '.', 'E', 'E-', null, '-'];\n while (true) {\n var b = parser.parseByte();\n var n1 = b >> 4;\n var n2 = b & 15;\n\n if (n1 === eof) {\n break;\n }\n\n s += lookup[n1];\n\n if (n2 === eof) {\n break;\n }\n\n s += lookup[n2];\n }\n\n return parseFloat(s);\n }\n\n // Parse a `CFF` DICT operand.\n function parseOperand(parser, b0) {\n var b1;\n var b2;\n var b3;\n var b4;\n if (b0 === 28) {\n b1 = parser.parseByte();\n b2 = parser.parseByte();\n return b1 << 8 | b2;\n }\n\n if (b0 === 29) {\n b1 = parser.parseByte();\n b2 = parser.parseByte();\n b3 = parser.parseByte();\n b4 = parser.parseByte();\n return b1 << 24 | b2 << 16 | b3 << 8 | b4;\n }\n\n if (b0 === 30) {\n return parseFloatOperand(parser);\n }\n\n if (b0 >= 32 && b0 <= 246) {\n return b0 - 139;\n }\n\n if (b0 >= 247 && b0 <= 250) {\n b1 = parser.parseByte();\n return (b0 - 247) * 256 + b1 + 108;\n }\n\n if (b0 >= 251 && b0 <= 254) {\n b1 = parser.parseByte();\n return -(b0 - 251) * 256 - b1 - 108;\n }\n\n throw new Error('Invalid b0 ' + b0);\n }\n\n // Convert the entries returned by `parseDict` to a proper dictionary.\n // If a value is a list of one, it is unpacked.\n function entriesToObject(entries) {\n var o = {};\n for (var i = 0; i < entries.length; i += 1) {\n var key = entries[i][0];\n var values = entries[i][1];\n var value = (void 0);\n if (values.length === 1) {\n value = values[0];\n } else {\n value = values;\n }\n\n if (o.hasOwnProperty(key) && !isNaN(o[key])) {\n throw new Error('Object ' + o + ' already has key ' + key);\n }\n\n o[key] = value;\n }\n\n return o;\n }\n\n // Parse a `CFF` DICT object.\n // A dictionary contains key-value pairs in a compact tokenized format.\n function parseCFFDict(data, start, size) {\n start = start !== undefined ? start : 0;\n var parser = new parse.Parser(data, start);\n var entries = [];\n var operands = [];\n size = size !== undefined ? size : data.length;\n\n while (parser.relativeOffset < size) {\n var op = parser.parseByte();\n\n // The first byte for each dict item distinguishes between operator (key) and operand (value).\n // Values <= 21 are operators.\n if (op <= 21) {\n // Two-byte operators have an initial escape byte of 12.\n if (op === 12) {\n op = 1200 + parser.parseByte();\n }\n\n entries.push([op, operands]);\n operands = [];\n } else {\n // Since the operands (values) come before the operators (keys), we store all operands in a list\n // until we encounter an operator.\n operands.push(parseOperand(parser, op));\n }\n }\n\n return entriesToObject(entries);\n }\n\n // Given a String Index (SID), return the value of the string.\n // Strings below index 392 are standard CFF strings and are not encoded in the font.\n function getCFFString(strings, index) {\n if (index <= 390) {\n index = cffStandardStrings[index];\n } else {\n index = strings[index - 391];\n }\n\n return index;\n }\n\n // Interpret a dictionary and return a new dictionary with readable keys and values for missing entries.\n // This function takes `meta` which is a list of objects containing `operand`, `name` and `default`.\n function interpretDict(dict, meta, strings) {\n var newDict = {};\n var value;\n\n // Because we also want to include missing values, we start out from the meta list\n // and lookup values in the dict.\n for (var i = 0; i < meta.length; i += 1) {\n var m = meta[i];\n\n if (Array.isArray(m.type)) {\n var values = [];\n values.length = m.type.length;\n for (var j = 0; j < m.type.length; j++) {\n value = dict[m.op] !== undefined ? dict[m.op][j] : undefined;\n if (value === undefined) {\n value = m.value !== undefined && m.value[j] !== undefined ? m.value[j] : null;\n }\n if (m.type[j] === 'SID') {\n value = getCFFString(strings, value);\n }\n values[j] = value;\n }\n newDict[m.name] = values;\n } else {\n value = dict[m.op];\n if (value === undefined) {\n value = m.value !== undefined ? m.value : null;\n }\n\n if (m.type === 'SID') {\n value = getCFFString(strings, value);\n }\n newDict[m.name] = value;\n }\n }\n\n return newDict;\n }\n\n // Parse the CFF header.\n function parseCFFHeader(data, start) {\n var header = {};\n header.formatMajor = parse.getCard8(data, start);\n header.formatMinor = parse.getCard8(data, start + 1);\n header.size = parse.getCard8(data, start + 2);\n header.offsetSize = parse.getCard8(data, start + 3);\n header.startOffset = start;\n header.endOffset = start + 4;\n return header;\n }\n\n var TOP_DICT_META = [\n {name: 'version', op: 0, type: 'SID'},\n {name: 'notice', op: 1, type: 'SID'},\n {name: 'copyright', op: 1200, type: 'SID'},\n {name: 'fullName', op: 2, type: 'SID'},\n {name: 'familyName', op: 3, type: 'SID'},\n {name: 'weight', op: 4, type: 'SID'},\n {name: 'isFixedPitch', op: 1201, type: 'number', value: 0},\n {name: 'italicAngle', op: 1202, type: 'number', value: 0},\n {name: 'underlinePosition', op: 1203, type: 'number', value: -100},\n {name: 'underlineThickness', op: 1204, type: 'number', value: 50},\n {name: 'paintType', op: 1205, type: 'number', value: 0},\n {name: 'charstringType', op: 1206, type: 'number', value: 2},\n {\n name: 'fontMatrix',\n op: 1207,\n type: ['real', 'real', 'real', 'real', 'real', 'real'],\n value: [0.001, 0, 0, 0.001, 0, 0]\n },\n {name: 'uniqueId', op: 13, type: 'number'},\n {name: 'fontBBox', op: 5, type: ['number', 'number', 'number', 'number'], value: [0, 0, 0, 0]},\n {name: 'strokeWidth', op: 1208, type: 'number', value: 0},\n {name: 'xuid', op: 14, type: [], value: null},\n {name: 'charset', op: 15, type: 'offset', value: 0},\n {name: 'encoding', op: 16, type: 'offset', value: 0},\n {name: 'charStrings', op: 17, type: 'offset', value: 0},\n {name: 'private', op: 18, type: ['number', 'offset'], value: [0, 0]},\n {name: 'ros', op: 1230, type: ['SID', 'SID', 'number']},\n {name: 'cidFontVersion', op: 1231, type: 'number', value: 0},\n {name: 'cidFontRevision', op: 1232, type: 'number', value: 0},\n {name: 'cidFontType', op: 1233, type: 'number', value: 0},\n {name: 'cidCount', op: 1234, type: 'number', value: 8720},\n {name: 'uidBase', op: 1235, type: 'number'},\n {name: 'fdArray', op: 1236, type: 'offset'},\n {name: 'fdSelect', op: 1237, type: 'offset'},\n {name: 'fontName', op: 1238, type: 'SID'}\n ];\n\n var PRIVATE_DICT_META = [\n {name: 'subrs', op: 19, type: 'offset', value: 0},\n {name: 'defaultWidthX', op: 20, type: 'number', value: 0},\n {name: 'nominalWidthX', op: 21, type: 'number', value: 0}\n ];\n\n // Parse the CFF top dictionary. A CFF table can contain multiple fonts, each with their own top dictionary.\n // The top dictionary contains the essential metadata for the font, together with the private dictionary.\n function parseCFFTopDict(data, strings) {\n var dict = parseCFFDict(data, 0, data.byteLength);\n return interpretDict(dict, TOP_DICT_META, strings);\n }\n\n // Parse the CFF private dictionary. We don't fully parse out all the values, only the ones we need.\n function parseCFFPrivateDict(data, start, size, strings) {\n var dict = parseCFFDict(data, start, size);\n return interpretDict(dict, PRIVATE_DICT_META, strings);\n }\n\n // Returns a list of \"Top DICT\"s found using an INDEX list.\n // Used to read both the usual high-level Top DICTs and also the FDArray\n // discovered inside CID-keyed fonts. When a Top DICT has a reference to\n // a Private DICT that is read and saved into the Top DICT.\n //\n // In addition to the expected/optional values as outlined in TOP_DICT_META\n // the following values might be saved into the Top DICT.\n //\n // _subrs [] array of local CFF subroutines from Private DICT\n // _subrsBias bias value computed from number of subroutines\n // (see calcCFFSubroutineBias() and parseCFFCharstring())\n // _defaultWidthX default widths for CFF characters\n // _nominalWidthX bias added to width embedded within glyph description\n //\n // _privateDict saved copy of parsed Private DICT from Top DICT\n function gatherCFFTopDicts(data, start, cffIndex, strings) {\n var topDictArray = [];\n for (var iTopDict = 0; iTopDict < cffIndex.length; iTopDict += 1) {\n var topDictData = new DataView(new Uint8Array(cffIndex[iTopDict]).buffer);\n var topDict = parseCFFTopDict(topDictData, strings);\n topDict._subrs = [];\n topDict._subrsBias = 0;\n topDict._defaultWidthX = 0;\n topDict._nominalWidthX = 0;\n var privateSize = topDict.private[0];\n var privateOffset = topDict.private[1];\n if (privateSize !== 0 && privateOffset !== 0) {\n var privateDict = parseCFFPrivateDict(data, privateOffset + start, privateSize, strings);\n topDict._defaultWidthX = privateDict.defaultWidthX;\n topDict._nominalWidthX = privateDict.nominalWidthX;\n if (privateDict.subrs !== 0) {\n var subrOffset = privateOffset + privateDict.subrs;\n var subrIndex = parseCFFIndex(data, subrOffset + start);\n topDict._subrs = subrIndex.objects;\n topDict._subrsBias = calcCFFSubroutineBias(topDict._subrs);\n }\n topDict._privateDict = privateDict;\n }\n topDictArray.push(topDict);\n }\n return topDictArray;\n }\n\n // Parse the CFF charset table, which contains internal names for all the glyphs.\n // This function will return a list of glyph names.\n // See Adobe TN #5176 chapter 13, \"Charsets\".\n function parseCFFCharset(data, start, nGlyphs, strings) {\n var sid;\n var count;\n var parser = new parse.Parser(data, start);\n\n // The .notdef glyph is not included, so subtract 1.\n nGlyphs -= 1;\n var charset = ['.notdef'];\n\n var format = parser.parseCard8();\n if (format === 0) {\n for (var i = 0; i < nGlyphs; i += 1) {\n sid = parser.parseSID();\n charset.push(getCFFString(strings, sid));\n }\n } else if (format === 1) {\n while (charset.length <= nGlyphs) {\n sid = parser.parseSID();\n count = parser.parseCard8();\n for (var i$1 = 0; i$1 <= count; i$1 += 1) {\n charset.push(getCFFString(strings, sid));\n sid += 1;\n }\n }\n } else if (format === 2) {\n while (charset.length <= nGlyphs) {\n sid = parser.parseSID();\n count = parser.parseCard16();\n for (var i$2 = 0; i$2 <= count; i$2 += 1) {\n charset.push(getCFFString(strings, sid));\n sid += 1;\n }\n }\n } else {\n throw new Error('Unknown charset format ' + format);\n }\n\n return charset;\n }\n\n // Parse the CFF encoding data. Only one encoding can be specified per font.\n // See Adobe TN #5176 chapter 12, \"Encodings\".\n function parseCFFEncoding(data, start, charset) {\n var code;\n var enc = {};\n var parser = new parse.Parser(data, start);\n var format = parser.parseCard8();\n if (format === 0) {\n var nCodes = parser.parseCard8();\n for (var i = 0; i < nCodes; i += 1) {\n code = parser.parseCard8();\n enc[code] = i;\n }\n } else if (format === 1) {\n var nRanges = parser.parseCard8();\n code = 1;\n for (var i$1 = 0; i$1 < nRanges; i$1 += 1) {\n var first = parser.parseCard8();\n var nLeft = parser.parseCard8();\n for (var j = first; j <= first + nLeft; j += 1) {\n enc[j] = code;\n code += 1;\n }\n }\n } else {\n throw new Error('Unknown encoding format ' + format);\n }\n\n return new CffEncoding(enc, charset);\n }\n\n // Take in charstring code and return a Glyph object.\n // The encoding is described in the Type 2 Charstring Format\n // https://www.microsoft.com/typography/OTSPEC/charstr2.htm\n function parseCFFCharstring(font, glyph, code) {\n var c1x;\n var c1y;\n var c2x;\n var c2y;\n var p = new Path();\n var stack = [];\n var nStems = 0;\n var haveWidth = false;\n var open = false;\n var x = 0;\n var y = 0;\n var subrs;\n var subrsBias;\n var defaultWidthX;\n var nominalWidthX;\n if (font.isCIDFont) {\n var fdIndex = font.tables.cff.topDict._fdSelect[glyph.index];\n var fdDict = font.tables.cff.topDict._fdArray[fdIndex];\n subrs = fdDict._subrs;\n subrsBias = fdDict._subrsBias;\n defaultWidthX = fdDict._defaultWidthX;\n nominalWidthX = fdDict._nominalWidthX;\n } else {\n subrs = font.tables.cff.topDict._subrs;\n subrsBias = font.tables.cff.topDict._subrsBias;\n defaultWidthX = font.tables.cff.topDict._defaultWidthX;\n nominalWidthX = font.tables.cff.topDict._nominalWidthX;\n }\n var width = defaultWidthX;\n\n function newContour(x, y) {\n if (open) {\n p.closePath();\n }\n\n p.moveTo(x, y);\n open = true;\n }\n\n function parseStems() {\n var hasWidthArg;\n\n // The number of stem operators on the stack is always even.\n // If the value is uneven, that means a width is specified.\n hasWidthArg = stack.length % 2 !== 0;\n if (hasWidthArg && !haveWidth) {\n width = stack.shift() + nominalWidthX;\n }\n\n nStems += stack.length >> 1;\n stack.length = 0;\n haveWidth = true;\n }\n\n function parse(code) {\n var b1;\n var b2;\n var b3;\n var b4;\n var codeIndex;\n var subrCode;\n var jpx;\n var jpy;\n var c3x;\n var c3y;\n var c4x;\n var c4y;\n\n var i = 0;\n while (i < code.length) {\n var v = code[i];\n i += 1;\n switch (v) {\n case 1: // hstem\n parseStems();\n break;\n case 3: // vstem\n parseStems();\n break;\n case 4: // vmoveto\n if (stack.length > 1 && !haveWidth) {\n width = stack.shift() + nominalWidthX;\n haveWidth = true;\n }\n\n y += stack.pop();\n newContour(x, y);\n break;\n case 5: // rlineto\n while (stack.length > 0) {\n x += stack.shift();\n y += stack.shift();\n p.lineTo(x, y);\n }\n\n break;\n case 6: // hlineto\n while (stack.length > 0) {\n x += stack.shift();\n p.lineTo(x, y);\n if (stack.length === 0) {\n break;\n }\n\n y += stack.shift();\n p.lineTo(x, y);\n }\n\n break;\n case 7: // vlineto\n while (stack.length > 0) {\n y += stack.shift();\n p.lineTo(x, y);\n if (stack.length === 0) {\n break;\n }\n\n x += stack.shift();\n p.lineTo(x, y);\n }\n\n break;\n case 8: // rrcurveto\n while (stack.length > 0) {\n c1x = x + stack.shift();\n c1y = y + stack.shift();\n c2x = c1x + stack.shift();\n c2y = c1y + stack.shift();\n x = c2x + stack.shift();\n y = c2y + stack.shift();\n p.curveTo(c1x, c1y, c2x, c2y, x, y);\n }\n\n break;\n case 10: // callsubr\n codeIndex = stack.pop() + subrsBias;\n subrCode = subrs[codeIndex];\n if (subrCode) {\n parse(subrCode);\n }\n\n break;\n case 11: // return\n return;\n case 12: // flex operators\n v = code[i];\n i += 1;\n switch (v) {\n case 35: // flex\n // |- dx1 dy1 dx2 dy2 dx3 dy3 dx4 dy4 dx5 dy5 dx6 dy6 fd flex (12 35) |-\n c1x = x + stack.shift(); // dx1\n c1y = y + stack.shift(); // dy1\n c2x = c1x + stack.shift(); // dx2\n c2y = c1y + stack.shift(); // dy2\n jpx = c2x + stack.shift(); // dx3\n jpy = c2y + stack.shift(); // dy3\n c3x = jpx + stack.shift(); // dx4\n c3y = jpy + stack.shift(); // dy4\n c4x = c3x + stack.shift(); // dx5\n c4y = c3y + stack.shift(); // dy5\n x = c4x + stack.shift(); // dx6\n y = c4y + stack.shift(); // dy6\n stack.shift(); // flex depth\n p.curveTo(c1x, c1y, c2x, c2y, jpx, jpy);\n p.curveTo(c3x, c3y, c4x, c4y, x, y);\n break;\n case 34: // hflex\n // |- dx1 dx2 dy2 dx3 dx4 dx5 dx6 hflex (12 34) |-\n c1x = x + stack.shift(); // dx1\n c1y = y; // dy1\n c2x = c1x + stack.shift(); // dx2\n c2y = c1y + stack.shift(); // dy2\n jpx = c2x + stack.shift(); // dx3\n jpy = c2y; // dy3\n c3x = jpx + stack.shift(); // dx4\n c3y = c2y; // dy4\n c4x = c3x + stack.shift(); // dx5\n c4y = y; // dy5\n x = c4x + stack.shift(); // dx6\n p.curveTo(c1x, c1y, c2x, c2y, jpx, jpy);\n p.curveTo(c3x, c3y, c4x, c4y, x, y);\n break;\n case 36: // hflex1\n // |- dx1 dy1 dx2 dy2 dx3 dx4 dx5 dy5 dx6 hflex1 (12 36) |-\n c1x = x + stack.shift(); // dx1\n c1y = y + stack.shift(); // dy1\n c2x = c1x + stack.shift(); // dx2\n c2y = c1y + stack.shift(); // dy2\n jpx = c2x + stack.shift(); // dx3\n jpy = c2y; // dy3\n c3x = jpx + stack.shift(); // dx4\n c3y = c2y; // dy4\n c4x = c3x + stack.shift(); // dx5\n c4y = c3y + stack.shift(); // dy5\n x = c4x + stack.shift(); // dx6\n p.curveTo(c1x, c1y, c2x, c2y, jpx, jpy);\n p.curveTo(c3x, c3y, c4x, c4y, x, y);\n break;\n case 37: // flex1\n // |- dx1 dy1 dx2 dy2 dx3 dy3 dx4 dy4 dx5 dy5 d6 flex1 (12 37) |-\n c1x = x + stack.shift(); // dx1\n c1y = y + stack.shift(); // dy1\n c2x = c1x + stack.shift(); // dx2\n c2y = c1y + stack.shift(); // dy2\n jpx = c2x + stack.shift(); // dx3\n jpy = c2y + stack.shift(); // dy3\n c3x = jpx + stack.shift(); // dx4\n c3y = jpy + stack.shift(); // dy4\n c4x = c3x + stack.shift(); // dx5\n c4y = c3y + stack.shift(); // dy5\n if (Math.abs(c4x - x) > Math.abs(c4y - y)) {\n x = c4x + stack.shift();\n } else {\n y = c4y + stack.shift();\n }\n\n p.curveTo(c1x, c1y, c2x, c2y, jpx, jpy);\n p.curveTo(c3x, c3y, c4x, c4y, x, y);\n break;\n default:\n console.log('Glyph ' + glyph.index + ': unknown operator ' + 1200 + v);\n stack.length = 0;\n }\n break;\n case 14: // endchar\n if (stack.length > 0 && !haveWidth) {\n width = stack.shift() + nominalWidthX;\n haveWidth = true;\n }\n\n if (open) {\n p.closePath();\n open = false;\n }\n\n break;\n case 18: // hstemhm\n parseStems();\n break;\n case 19: // hintmask\n case 20: // cntrmask\n parseStems();\n i += (nStems + 7) >> 3;\n break;\n case 21: // rmoveto\n if (stack.length > 2 && !haveWidth) {\n width = stack.shift() + nominalWidthX;\n haveWidth = true;\n }\n\n y += stack.pop();\n x += stack.pop();\n newContour(x, y);\n break;\n case 22: // hmoveto\n if (stack.length > 1 && !haveWidth) {\n width = stack.shift() + nominalWidthX;\n haveWidth = true;\n }\n\n x += stack.pop();\n newContour(x, y);\n break;\n case 23: // vstemhm\n parseStems();\n break;\n case 24: // rcurveline\n while (stack.length > 2) {\n c1x = x + stack.shift();\n c1y = y + stack.shift();\n c2x = c1x + stack.shift();\n c2y = c1y + stack.shift();\n x = c2x + stack.shift();\n y = c2y + stack.shift();\n p.curveTo(c1x, c1y, c2x, c2y, x, y);\n }\n\n x += stack.shift();\n y += stack.shift();\n p.lineTo(x, y);\n break;\n case 25: // rlinecurve\n while (stack.length > 6) {\n x += stack.shift();\n y += stack.shift();\n p.lineTo(x, y);\n }\n\n c1x = x + stack.shift();\n c1y = y + stack.shift();\n c2x = c1x + stack.shift();\n c2y = c1y + stack.shift();\n x = c2x + stack.shift();\n y = c2y + stack.shift();\n p.curveTo(c1x, c1y, c2x, c2y, x, y);\n break;\n case 26: // vvcurveto\n if (stack.length % 2) {\n x += stack.shift();\n }\n\n while (stack.length > 0) {\n c1x = x;\n c1y = y + stack.shift();\n c2x = c1x + stack.shift();\n c2y = c1y + stack.shift();\n x = c2x;\n y = c2y + stack.shift();\n p.curveTo(c1x, c1y, c2x, c2y, x, y);\n }\n\n break;\n case 27: // hhcurveto\n if (stack.length % 2) {\n y += stack.shift();\n }\n\n while (stack.length > 0) {\n c1x = x + stack.shift();\n c1y = y;\n c2x = c1x + stack.shift();\n c2y = c1y + stack.shift();\n x = c2x + stack.shift();\n y = c2y;\n p.curveTo(c1x, c1y, c2x, c2y, x, y);\n }\n\n break;\n case 28: // shortint\n b1 = code[i];\n b2 = code[i + 1];\n stack.push(((b1 << 24) | (b2 << 16)) >> 16);\n i += 2;\n break;\n case 29: // callgsubr\n codeIndex = stack.pop() + font.gsubrsBias;\n subrCode = font.gsubrs[codeIndex];\n if (subrCode) {\n parse(subrCode);\n }\n\n break;\n case 30: // vhcurveto\n while (stack.length > 0) {\n c1x = x;\n c1y = y + stack.shift();\n c2x = c1x + stack.shift();\n c2y = c1y + stack.shift();\n x = c2x + stack.shift();\n y = c2y + (stack.length === 1 ? stack.shift() : 0);\n p.curveTo(c1x, c1y, c2x, c2y, x, y);\n if (stack.length === 0) {\n break;\n }\n\n c1x = x + stack.shift();\n c1y = y;\n c2x = c1x + stack.shift();\n c2y = c1y + stack.shift();\n y = c2y + stack.shift();\n x = c2x + (stack.length === 1 ? stack.shift() : 0);\n p.curveTo(c1x, c1y, c2x, c2y, x, y);\n }\n\n break;\n case 31: // hvcurveto\n while (stack.length > 0) {\n c1x = x + stack.shift();\n c1y = y;\n c2x = c1x + stack.shift();\n c2y = c1y + stack.shift();\n y = c2y + stack.shift();\n x = c2x + (stack.length === 1 ? stack.shift() : 0);\n p.curveTo(c1x, c1y, c2x, c2y, x, y);\n if (stack.length === 0) {\n break;\n }\n\n c1x = x;\n c1y = y + stack.shift();\n c2x = c1x + stack.shift();\n c2y = c1y + stack.shift();\n x = c2x + stack.shift();\n y = c2y + (stack.length === 1 ? stack.shift() : 0);\n p.curveTo(c1x, c1y, c2x, c2y, x, y);\n }\n\n break;\n default:\n if (v < 32) {\n console.log('Glyph ' + glyph.index + ': unknown operator ' + v);\n } else if (v < 247) {\n stack.push(v - 139);\n } else if (v < 251) {\n b1 = code[i];\n i += 1;\n stack.push((v - 247) * 256 + b1 + 108);\n } else if (v < 255) {\n b1 = code[i];\n i += 1;\n stack.push(-(v - 251) * 256 - b1 - 108);\n } else {\n b1 = code[i];\n b2 = code[i + 1];\n b3 = code[i + 2];\n b4 = code[i + 3];\n i += 4;\n stack.push(((b1 << 24) | (b2 << 16) | (b3 << 8) | b4) / 65536);\n }\n }\n }\n }\n\n parse(code);\n\n glyph.advanceWidth = width;\n return p;\n }\n\n function parseCFFFDSelect(data, start, nGlyphs, fdArrayCount) {\n var fdSelect = [];\n var fdIndex;\n var parser = new parse.Parser(data, start);\n var format = parser.parseCard8();\n if (format === 0) {\n // Simple list of nGlyphs elements\n for (var iGid = 0; iGid < nGlyphs; iGid++) {\n fdIndex = parser.parseCard8();\n if (fdIndex >= fdArrayCount) {\n throw new Error('CFF table CID Font FDSelect has bad FD index value ' + fdIndex + ' (FD count ' + fdArrayCount + ')');\n }\n fdSelect.push(fdIndex);\n }\n } else if (format === 3) {\n // Ranges\n var nRanges = parser.parseCard16();\n var first = parser.parseCard16();\n if (first !== 0) {\n throw new Error('CFF Table CID Font FDSelect format 3 range has bad initial GID ' + first);\n }\n var next;\n for (var iRange = 0; iRange < nRanges; iRange++) {\n fdIndex = parser.parseCard8();\n next = parser.parseCard16();\n if (fdIndex >= fdArrayCount) {\n throw new Error('CFF table CID Font FDSelect has bad FD index value ' + fdIndex + ' (FD count ' + fdArrayCount + ')');\n }\n if (next > nGlyphs) {\n throw new Error('CFF Table CID Font FDSelect format 3 range has bad GID ' + next);\n }\n for (; first < next; first++) {\n fdSelect.push(fdIndex);\n }\n first = next;\n }\n if (next !== nGlyphs) {\n throw new Error('CFF Table CID Font FDSelect format 3 range has bad final GID ' + next);\n }\n } else {\n throw new Error('CFF Table CID Font FDSelect table has unsupported format ' + format);\n }\n return fdSelect;\n }\n\n // Parse the `CFF` table, which contains the glyph outlines in PostScript format.\n function parseCFFTable(data, start, font, opt) {\n font.tables.cff = {};\n var header = parseCFFHeader(data, start);\n var nameIndex = parseCFFIndex(data, header.endOffset, parse.bytesToString);\n var topDictIndex = parseCFFIndex(data, nameIndex.endOffset);\n var stringIndex = parseCFFIndex(data, topDictIndex.endOffset, parse.bytesToString);\n var globalSubrIndex = parseCFFIndex(data, stringIndex.endOffset);\n font.gsubrs = globalSubrIndex.objects;\n font.gsubrsBias = calcCFFSubroutineBias(font.gsubrs);\n\n var topDictArray = gatherCFFTopDicts(data, start, topDictIndex.objects, stringIndex.objects);\n if (topDictArray.length !== 1) {\n throw new Error('CFF table has too many fonts in \\'FontSet\\' - count of fonts NameIndex.length = ' + topDictArray.length);\n }\n\n var topDict = topDictArray[0];\n font.tables.cff.topDict = topDict;\n\n if (topDict._privateDict) {\n font.defaultWidthX = topDict._privateDict.defaultWidthX;\n font.nominalWidthX = topDict._privateDict.nominalWidthX;\n }\n\n if (topDict.ros[0] !== undefined && topDict.ros[1] !== undefined) {\n font.isCIDFont = true;\n }\n\n if (font.isCIDFont) {\n var fdArrayOffset = topDict.fdArray;\n var fdSelectOffset = topDict.fdSelect;\n if (fdArrayOffset === 0 || fdSelectOffset === 0) {\n throw new Error('Font is marked as a CID font, but FDArray and/or FDSelect information is missing');\n }\n fdArrayOffset += start;\n var fdArrayIndex = parseCFFIndex(data, fdArrayOffset);\n var fdArray = gatherCFFTopDicts(data, start, fdArrayIndex.objects, stringIndex.objects);\n topDict._fdArray = fdArray;\n fdSelectOffset += start;\n topDict._fdSelect = parseCFFFDSelect(data, fdSelectOffset, font.numGlyphs, fdArray.length);\n }\n\n var privateDictOffset = start + topDict.private[1];\n var privateDict = parseCFFPrivateDict(data, privateDictOffset, topDict.private[0], stringIndex.objects);\n font.defaultWidthX = privateDict.defaultWidthX;\n font.nominalWidthX = privateDict.nominalWidthX;\n\n if (privateDict.subrs !== 0) {\n var subrOffset = privateDictOffset + privateDict.subrs;\n var subrIndex = parseCFFIndex(data, subrOffset);\n font.subrs = subrIndex.objects;\n font.subrsBias = calcCFFSubroutineBias(font.subrs);\n } else {\n font.subrs = [];\n font.subrsBias = 0;\n }\n\n // Offsets in the top dict are relative to the beginning of the CFF data, so add the CFF start offset.\n var charStringsIndex;\n if (opt.lowMemory) {\n charStringsIndex = parseCFFIndexLowMemory(data, start + topDict.charStrings);\n font.nGlyphs = charStringsIndex.offsets.length;\n } else {\n charStringsIndex = parseCFFIndex(data, start + topDict.charStrings);\n font.nGlyphs = charStringsIndex.objects.length;\n }\n\n var charset = parseCFFCharset(data, start + topDict.charset, font.nGlyphs, stringIndex.objects);\n if (topDict.encoding === 0) {\n // Standard encoding\n font.cffEncoding = new CffEncoding(cffStandardEncoding, charset);\n } else if (topDict.encoding === 1) {\n // Expert encoding\n font.cffEncoding = new CffEncoding(cffExpertEncoding, charset);\n } else {\n font.cffEncoding = parseCFFEncoding(data, start + topDict.encoding, charset);\n }\n\n // Prefer the CMAP encoding to the CFF encoding.\n font.encoding = font.encoding || font.cffEncoding;\n\n font.glyphs = new glyphset.GlyphSet(font);\n if (opt.lowMemory) {\n font._push = function(i) {\n var charString = getCffIndexObject(i, charStringsIndex.offsets, data, start + topDict.charStrings);\n font.glyphs.push(i, glyphset.cffGlyphLoader(font, i, parseCFFCharstring, charString));\n };\n } else {\n for (var i = 0; i < font.nGlyphs; i += 1) {\n var charString = charStringsIndex.objects[i];\n font.glyphs.push(i, glyphset.cffGlyphLoader(font, i, parseCFFCharstring, charString));\n }\n }\n }\n\n // Convert a string to a String ID (SID).\n // The list of strings is modified in place.\n function encodeString(s, strings) {\n var sid;\n\n // Is the string in the CFF standard strings?\n var i = cffStandardStrings.indexOf(s);\n if (i >= 0) {\n sid = i;\n }\n\n // Is the string already in the string index?\n i = strings.indexOf(s);\n if (i >= 0) {\n sid = i + cffStandardStrings.length;\n } else {\n sid = cffStandardStrings.length + strings.length;\n strings.push(s);\n }\n\n return sid;\n }\n\n function makeHeader() {\n return new table.Record('Header', [\n {name: 'major', type: 'Card8', value: 1},\n {name: 'minor', type: 'Card8', value: 0},\n {name: 'hdrSize', type: 'Card8', value: 4},\n {name: 'major', type: 'Card8', value: 1}\n ]);\n }\n\n function makeNameIndex(fontNames) {\n var t = new table.Record('Name INDEX', [\n {name: 'names', type: 'INDEX', value: []}\n ]);\n t.names = [];\n for (var i = 0; i < fontNames.length; i += 1) {\n t.names.push({name: 'name_' + i, type: 'NAME', value: fontNames[i]});\n }\n\n return t;\n }\n\n // Given a dictionary's metadata, create a DICT structure.\n function makeDict(meta, attrs, strings) {\n var m = {};\n for (var i = 0; i < meta.length; i += 1) {\n var entry = meta[i];\n var value = attrs[entry.name];\n if (value !== undefined && !equals$1(value, entry.value)) {\n if (entry.type === 'SID') {\n value = encodeString(value, strings);\n }\n\n m[entry.op] = {name: entry.name, type: entry.type, value: value};\n }\n }\n\n return m;\n }\n\n // The Top DICT houses the global font attributes.\n function makeTopDict(attrs, strings) {\n var t = new table.Record('Top DICT', [\n {name: 'dict', type: 'DICT', value: {}}\n ]);\n t.dict = makeDict(TOP_DICT_META, attrs, strings);\n return t;\n }\n\n function makeTopDictIndex(topDict) {\n var t = new table.Record('Top DICT INDEX', [\n {name: 'topDicts', type: 'INDEX', value: []}\n ]);\n t.topDicts = [{name: 'topDict_0', type: 'TABLE', value: topDict}];\n return t;\n }\n\n function makeStringIndex(strings) {\n var t = new table.Record('String INDEX', [\n {name: 'strings', type: 'INDEX', value: []}\n ]);\n t.strings = [];\n for (var i = 0; i < strings.length; i += 1) {\n t.strings.push({name: 'string_' + i, type: 'STRING', value: strings[i]});\n }\n\n return t;\n }\n\n function makeGlobalSubrIndex() {\n // Currently we don't use subroutines.\n return new table.Record('Global Subr INDEX', [\n {name: 'subrs', type: 'INDEX', value: []}\n ]);\n }\n\n function makeCharsets(glyphNames, strings) {\n var t = new table.Record('Charsets', [\n {name: 'format', type: 'Card8', value: 0}\n ]);\n for (var i = 0; i < glyphNames.length; i += 1) {\n var glyphName = glyphNames[i];\n var glyphSID = encodeString(glyphName, strings);\n t.fields.push({name: 'glyph_' + i, type: 'SID', value: glyphSID});\n }\n\n return t;\n }\n\n function glyphToOps(glyph) {\n var ops = [];\n var path = glyph.path;\n ops.push({name: 'width', type: 'NUMBER', value: glyph.advanceWidth});\n var x = 0;\n var y = 0;\n for (var i = 0; i < path.commands.length; i += 1) {\n var dx = (void 0);\n var dy = (void 0);\n var cmd = path.commands[i];\n if (cmd.type === 'Q') {\n // CFF only supports bézier curves, so convert the quad to a bézier.\n var _13 = 1 / 3;\n var _23 = 2 / 3;\n\n // We're going to create a new command so we don't change the original path.\n // Since all coordinates are relative, we round() them ASAP to avoid propagating errors.\n cmd = {\n type: 'C',\n x: cmd.x,\n y: cmd.y,\n x1: Math.round(_13 * x + _23 * cmd.x1),\n y1: Math.round(_13 * y + _23 * cmd.y1),\n x2: Math.round(_13 * cmd.x + _23 * cmd.x1),\n y2: Math.round(_13 * cmd.y + _23 * cmd.y1)\n };\n }\n\n if (cmd.type === 'M') {\n dx = Math.round(cmd.x - x);\n dy = Math.round(cmd.y - y);\n ops.push({name: 'dx', type: 'NUMBER', value: dx});\n ops.push({name: 'dy', type: 'NUMBER', value: dy});\n ops.push({name: 'rmoveto', type: 'OP', value: 21});\n x = Math.round(cmd.x);\n y = Math.round(cmd.y);\n } else if (cmd.type === 'L') {\n dx = Math.round(cmd.x - x);\n dy = Math.round(cmd.y - y);\n ops.push({name: 'dx', type: 'NUMBER', value: dx});\n ops.push({name: 'dy', type: 'NUMBER', value: dy});\n ops.push({name: 'rlineto', type: 'OP', value: 5});\n x = Math.round(cmd.x);\n y = Math.round(cmd.y);\n } else if (cmd.type === 'C') {\n var dx1 = Math.round(cmd.x1 - x);\n var dy1 = Math.round(cmd.y1 - y);\n var dx2 = Math.round(cmd.x2 - cmd.x1);\n var dy2 = Math.round(cmd.y2 - cmd.y1);\n dx = Math.round(cmd.x - cmd.x2);\n dy = Math.round(cmd.y - cmd.y2);\n ops.push({name: 'dx1', type: 'NUMBER', value: dx1});\n ops.push({name: 'dy1', type: 'NUMBER', value: dy1});\n ops.push({name: 'dx2', type: 'NUMBER', value: dx2});\n ops.push({name: 'dy2', type: 'NUMBER', value: dy2});\n ops.push({name: 'dx', type: 'NUMBER', value: dx});\n ops.push({name: 'dy', type: 'NUMBER', value: dy});\n ops.push({name: 'rrcurveto', type: 'OP', value: 8});\n x = Math.round(cmd.x);\n y = Math.round(cmd.y);\n }\n\n // Contours are closed automatically.\n }\n\n ops.push({name: 'endchar', type: 'OP', value: 14});\n return ops;\n }\n\n function makeCharStringsIndex(glyphs) {\n var t = new table.Record('CharStrings INDEX', [\n {name: 'charStrings', type: 'INDEX', value: []}\n ]);\n\n for (var i = 0; i < glyphs.length; i += 1) {\n var glyph = glyphs.get(i);\n var ops = glyphToOps(glyph);\n t.charStrings.push({name: glyph.name, type: 'CHARSTRING', value: ops});\n }\n\n return t;\n }\n\n function makePrivateDict(attrs, strings) {\n var t = new table.Record('Private DICT', [\n {name: 'dict', type: 'DICT', value: {}}\n ]);\n t.dict = makeDict(PRIVATE_DICT_META, attrs, strings);\n return t;\n }\n\n function makeCFFTable(glyphs, options) {\n var t = new table.Table('CFF ', [\n {name: 'header', type: 'RECORD'},\n {name: 'nameIndex', type: 'RECORD'},\n {name: 'topDictIndex', type: 'RECORD'},\n {name: 'stringIndex', type: 'RECORD'},\n {name: 'globalSubrIndex', type: 'RECORD'},\n {name: 'charsets', type: 'RECORD'},\n {name: 'charStringsIndex', type: 'RECORD'},\n {name: 'privateDict', type: 'RECORD'}\n ]);\n\n var fontScale = 1 / options.unitsPerEm;\n // We use non-zero values for the offsets so that the DICT encodes them.\n // This is important because the size of the Top DICT plays a role in offset calculation,\n // and the size shouldn't change after we've written correct offsets.\n var attrs = {\n version: options.version,\n fullName: options.fullName,\n familyName: options.familyName,\n weight: options.weightName,\n fontBBox: options.fontBBox || [0, 0, 0, 0],\n fontMatrix: [fontScale, 0, 0, fontScale, 0, 0],\n charset: 999,\n encoding: 0,\n charStrings: 999,\n private: [0, 999]\n };\n\n var privateAttrs = {};\n\n var glyphNames = [];\n var glyph;\n\n // Skip first glyph (.notdef)\n for (var i = 1; i < glyphs.length; i += 1) {\n glyph = glyphs.get(i);\n glyphNames.push(glyph.name);\n }\n\n var strings = [];\n\n t.header = makeHeader();\n t.nameIndex = makeNameIndex([options.postScriptName]);\n var topDict = makeTopDict(attrs, strings);\n t.topDictIndex = makeTopDictIndex(topDict);\n t.globalSubrIndex = makeGlobalSubrIndex();\n t.charsets = makeCharsets(glyphNames, strings);\n t.charStringsIndex = makeCharStringsIndex(glyphs);\n t.privateDict = makePrivateDict(privateAttrs, strings);\n\n // Needs to come at the end, to encode all custom strings used in the font.\n t.stringIndex = makeStringIndex(strings);\n\n var startOffset = t.header.sizeOf() +\n t.nameIndex.sizeOf() +\n t.topDictIndex.sizeOf() +\n t.stringIndex.sizeOf() +\n t.globalSubrIndex.sizeOf();\n attrs.charset = startOffset;\n\n // We use the CFF standard encoding; proper encoding will be handled in cmap.\n attrs.encoding = 0;\n attrs.charStrings = attrs.charset + t.charsets.sizeOf();\n attrs.private[1] = attrs.charStrings + t.charStringsIndex.sizeOf();\n\n // Recreate the Top DICT INDEX with the correct offsets.\n topDict = makeTopDict(attrs, strings);\n t.topDictIndex = makeTopDictIndex(topDict);\n\n return t;\n }\n\n var cff = { parse: parseCFFTable, make: makeCFFTable };\n\n // The `head` table contains global information about the font.\n\n // Parse the header `head` table\n function parseHeadTable(data, start) {\n var head = {};\n var p = new parse.Parser(data, start);\n head.version = p.parseVersion();\n head.fontRevision = Math.round(p.parseFixed() * 1000) / 1000;\n head.checkSumAdjustment = p.parseULong();\n head.magicNumber = p.parseULong();\n check.argument(head.magicNumber === 0x5F0F3CF5, 'Font header has wrong magic number.');\n head.flags = p.parseUShort();\n head.unitsPerEm = p.parseUShort();\n head.created = p.parseLongDateTime();\n head.modified = p.parseLongDateTime();\n head.xMin = p.parseShort();\n head.yMin = p.parseShort();\n head.xMax = p.parseShort();\n head.yMax = p.parseShort();\n head.macStyle = p.parseUShort();\n head.lowestRecPPEM = p.parseUShort();\n head.fontDirectionHint = p.parseShort();\n head.indexToLocFormat = p.parseShort();\n head.glyphDataFormat = p.parseShort();\n return head;\n }\n\n function makeHeadTable(options) {\n // Apple Mac timestamp epoch is 01/01/1904 not 01/01/1970\n var timestamp = Math.round(new Date().getTime() / 1000) + 2082844800;\n var createdTimestamp = timestamp;\n\n if (options.createdTimestamp) {\n createdTimestamp = options.createdTimestamp + 2082844800;\n }\n\n return new table.Table('head', [\n {name: 'version', type: 'FIXED', value: 0x00010000},\n {name: 'fontRevision', type: 'FIXED', value: 0x00010000},\n {name: 'checkSumAdjustment', type: 'ULONG', value: 0},\n {name: 'magicNumber', type: 'ULONG', value: 0x5F0F3CF5},\n {name: 'flags', type: 'USHORT', value: 0},\n {name: 'unitsPerEm', type: 'USHORT', value: 1000},\n {name: 'created', type: 'LONGDATETIME', value: createdTimestamp},\n {name: 'modified', type: 'LONGDATETIME', value: timestamp},\n {name: 'xMin', type: 'SHORT', value: 0},\n {name: 'yMin', type: 'SHORT', value: 0},\n {name: 'xMax', type: 'SHORT', value: 0},\n {name: 'yMax', type: 'SHORT', value: 0},\n {name: 'macStyle', type: 'USHORT', value: 0},\n {name: 'lowestRecPPEM', type: 'USHORT', value: 0},\n {name: 'fontDirectionHint', type: 'SHORT', value: 2},\n {name: 'indexToLocFormat', type: 'SHORT', value: 0},\n {name: 'glyphDataFormat', type: 'SHORT', value: 0}\n ], options);\n }\n\n var head = { parse: parseHeadTable, make: makeHeadTable };\n\n // The `hhea` table contains information for horizontal layout.\n\n // Parse the horizontal header `hhea` table\n function parseHheaTable(data, start) {\n var hhea = {};\n var p = new parse.Parser(data, start);\n hhea.version = p.parseVersion();\n hhea.ascender = p.parseShort();\n hhea.descender = p.parseShort();\n hhea.lineGap = p.parseShort();\n hhea.advanceWidthMax = p.parseUShort();\n hhea.minLeftSideBearing = p.parseShort();\n hhea.minRightSideBearing = p.parseShort();\n hhea.xMaxExtent = p.parseShort();\n hhea.caretSlopeRise = p.parseShort();\n hhea.caretSlopeRun = p.parseShort();\n hhea.caretOffset = p.parseShort();\n p.relativeOffset += 8;\n hhea.metricDataFormat = p.parseShort();\n hhea.numberOfHMetrics = p.parseUShort();\n return hhea;\n }\n\n function makeHheaTable(options) {\n return new table.Table('hhea', [\n {name: 'version', type: 'FIXED', value: 0x00010000},\n {name: 'ascender', type: 'FWORD', value: 0},\n {name: 'descender', type: 'FWORD', value: 0},\n {name: 'lineGap', type: 'FWORD', value: 0},\n {name: 'advanceWidthMax', type: 'UFWORD', value: 0},\n {name: 'minLeftSideBearing', type: 'FWORD', value: 0},\n {name: 'minRightSideBearing', type: 'FWORD', value: 0},\n {name: 'xMaxExtent', type: 'FWORD', value: 0},\n {name: 'caretSlopeRise', type: 'SHORT', value: 1},\n {name: 'caretSlopeRun', type: 'SHORT', value: 0},\n {name: 'caretOffset', type: 'SHORT', value: 0},\n {name: 'reserved1', type: 'SHORT', value: 0},\n {name: 'reserved2', type: 'SHORT', value: 0},\n {name: 'reserved3', type: 'SHORT', value: 0},\n {name: 'reserved4', type: 'SHORT', value: 0},\n {name: 'metricDataFormat', type: 'SHORT', value: 0},\n {name: 'numberOfHMetrics', type: 'USHORT', value: 0}\n ], options);\n }\n\n var hhea = { parse: parseHheaTable, make: makeHheaTable };\n\n // The `hmtx` table contains the horizontal metrics for all glyphs.\n\n function parseHmtxTableAll(data, start, numMetrics, numGlyphs, glyphs) {\n var advanceWidth;\n var leftSideBearing;\n var p = new parse.Parser(data, start);\n for (var i = 0; i < numGlyphs; i += 1) {\n // If the font is monospaced, only one entry is needed. This last entry applies to all subsequent glyphs.\n if (i < numMetrics) {\n advanceWidth = p.parseUShort();\n leftSideBearing = p.parseShort();\n }\n\n var glyph = glyphs.get(i);\n glyph.advanceWidth = advanceWidth;\n glyph.leftSideBearing = leftSideBearing;\n }\n }\n\n function parseHmtxTableOnLowMemory(font, data, start, numMetrics, numGlyphs) {\n font._hmtxTableData = {};\n\n var advanceWidth;\n var leftSideBearing;\n var p = new parse.Parser(data, start);\n for (var i = 0; i < numGlyphs; i += 1) {\n // If the font is monospaced, only one entry is needed. This last entry applies to all subsequent glyphs.\n if (i < numMetrics) {\n advanceWidth = p.parseUShort();\n leftSideBearing = p.parseShort();\n }\n\n font._hmtxTableData[i] = {\n advanceWidth: advanceWidth,\n leftSideBearing: leftSideBearing\n };\n }\n }\n\n // Parse the `hmtx` table, which contains the horizontal metrics for all glyphs.\n // This function augments the glyph array, adding the advanceWidth and leftSideBearing to each glyph.\n function parseHmtxTable(font, data, start, numMetrics, numGlyphs, glyphs, opt) {\n if (opt.lowMemory)\n { parseHmtxTableOnLowMemory(font, data, start, numMetrics, numGlyphs); }\n else\n { parseHmtxTableAll(data, start, numMetrics, numGlyphs, glyphs); }\n }\n\n function makeHmtxTable(glyphs) {\n var t = new table.Table('hmtx', []);\n for (var i = 0; i < glyphs.length; i += 1) {\n var glyph = glyphs.get(i);\n var advanceWidth = glyph.advanceWidth || 0;\n var leftSideBearing = glyph.leftSideBearing || 0;\n t.fields.push({name: 'advanceWidth_' + i, type: 'USHORT', value: advanceWidth});\n t.fields.push({name: 'leftSideBearing_' + i, type: 'SHORT', value: leftSideBearing});\n }\n\n return t;\n }\n\n var hmtx = { parse: parseHmtxTable, make: makeHmtxTable };\n\n // The `ltag` table stores IETF BCP-47 language tags. It allows supporting\n\n function makeLtagTable(tags) {\n var result = new table.Table('ltag', [\n {name: 'version', type: 'ULONG', value: 1},\n {name: 'flags', type: 'ULONG', value: 0},\n {name: 'numTags', type: 'ULONG', value: tags.length}\n ]);\n\n var stringPool = '';\n var stringPoolOffset = 12 + tags.length * 4;\n for (var i = 0; i < tags.length; ++i) {\n var pos = stringPool.indexOf(tags[i]);\n if (pos < 0) {\n pos = stringPool.length;\n stringPool += tags[i];\n }\n\n result.fields.push({name: 'offset ' + i, type: 'USHORT', value: stringPoolOffset + pos});\n result.fields.push({name: 'length ' + i, type: 'USHORT', value: tags[i].length});\n }\n\n result.fields.push({name: 'stringPool', type: 'CHARARRAY', value: stringPool});\n return result;\n }\n\n function parseLtagTable(data, start) {\n var p = new parse.Parser(data, start);\n var tableVersion = p.parseULong();\n check.argument(tableVersion === 1, 'Unsupported ltag table version.');\n // The 'ltag' specification does not define any flags; skip the field.\n p.skip('uLong', 1);\n var numTags = p.parseULong();\n\n var tags = [];\n for (var i = 0; i < numTags; i++) {\n var tag = '';\n var offset = start + p.parseUShort();\n var length = p.parseUShort();\n for (var j = offset; j < offset + length; ++j) {\n tag += String.fromCharCode(data.getInt8(j));\n }\n\n tags.push(tag);\n }\n\n return tags;\n }\n\n var ltag = { make: makeLtagTable, parse: parseLtagTable };\n\n // The `maxp` table establishes the memory requirements for the font.\n\n // Parse the maximum profile `maxp` table.\n function parseMaxpTable(data, start) {\n var maxp = {};\n var p = new parse.Parser(data, start);\n maxp.version = p.parseVersion();\n maxp.numGlyphs = p.parseUShort();\n if (maxp.version === 1.0) {\n maxp.maxPoints = p.parseUShort();\n maxp.maxContours = p.parseUShort();\n maxp.maxCompositePoints = p.parseUShort();\n maxp.maxCompositeContours = p.parseUShort();\n maxp.maxZones = p.parseUShort();\n maxp.maxTwilightPoints = p.parseUShort();\n maxp.maxStorage = p.parseUShort();\n maxp.maxFunctionDefs = p.parseUShort();\n maxp.maxInstructionDefs = p.parseUShort();\n maxp.maxStackElements = p.parseUShort();\n maxp.maxSizeOfInstructions = p.parseUShort();\n maxp.maxComponentElements = p.parseUShort();\n maxp.maxComponentDepth = p.parseUShort();\n }\n\n return maxp;\n }\n\n function makeMaxpTable(numGlyphs) {\n return new table.Table('maxp', [\n {name: 'version', type: 'FIXED', value: 0x00005000},\n {name: 'numGlyphs', type: 'USHORT', value: numGlyphs}\n ]);\n }\n\n var maxp = { parse: parseMaxpTable, make: makeMaxpTable };\n\n // The `name` naming table.\n\n // NameIDs for the name table.\n var nameTableNames = [\n 'copyright', // 0\n 'fontFamily', // 1\n 'fontSubfamily', // 2\n 'uniqueID', // 3\n 'fullName', // 4\n 'version', // 5\n 'postScriptName', // 6\n 'trademark', // 7\n 'manufacturer', // 8\n 'designer', // 9\n 'description', // 10\n 'manufacturerURL', // 11\n 'designerURL', // 12\n 'license', // 13\n 'licenseURL', // 14\n 'reserved', // 15\n 'preferredFamily', // 16\n 'preferredSubfamily', // 17\n 'compatibleFullName', // 18\n 'sampleText', // 19\n 'postScriptFindFontName', // 20\n 'wwsFamily', // 21\n 'wwsSubfamily' // 22\n ];\n\n var macLanguages = {\n 0: 'en',\n 1: 'fr',\n 2: 'de',\n 3: 'it',\n 4: 'nl',\n 5: 'sv',\n 6: 'es',\n 7: 'da',\n 8: 'pt',\n 9: 'no',\n 10: 'he',\n 11: 'ja',\n 12: 'ar',\n 13: 'fi',\n 14: 'el',\n 15: 'is',\n 16: 'mt',\n 17: 'tr',\n 18: 'hr',\n 19: 'zh-Hant',\n 20: 'ur',\n 21: 'hi',\n 22: 'th',\n 23: 'ko',\n 24: 'lt',\n 25: 'pl',\n 26: 'hu',\n 27: 'es',\n 28: 'lv',\n 29: 'se',\n 30: 'fo',\n 31: 'fa',\n 32: 'ru',\n 33: 'zh',\n 34: 'nl-BE',\n 35: 'ga',\n 36: 'sq',\n 37: 'ro',\n 38: 'cz',\n 39: 'sk',\n 40: 'si',\n 41: 'yi',\n 42: 'sr',\n 43: 'mk',\n 44: 'bg',\n 45: 'uk',\n 46: 'be',\n 47: 'uz',\n 48: 'kk',\n 49: 'az-Cyrl',\n 50: 'az-Arab',\n 51: 'hy',\n 52: 'ka',\n 53: 'mo',\n 54: 'ky',\n 55: 'tg',\n 56: 'tk',\n 57: 'mn-CN',\n 58: 'mn',\n 59: 'ps',\n 60: 'ks',\n 61: 'ku',\n 62: 'sd',\n 63: 'bo',\n 64: 'ne',\n 65: 'sa',\n 66: 'mr',\n 67: 'bn',\n 68: 'as',\n 69: 'gu',\n 70: 'pa',\n 71: 'or',\n 72: 'ml',\n 73: 'kn',\n 74: 'ta',\n 75: 'te',\n 76: 'si',\n 77: 'my',\n 78: 'km',\n 79: 'lo',\n 80: 'vi',\n 81: 'id',\n 82: 'tl',\n 83: 'ms',\n 84: 'ms-Arab',\n 85: 'am',\n 86: 'ti',\n 87: 'om',\n 88: 'so',\n 89: 'sw',\n 90: 'rw',\n 91: 'rn',\n 92: 'ny',\n 93: 'mg',\n 94: 'eo',\n 128: 'cy',\n 129: 'eu',\n 130: 'ca',\n 131: 'la',\n 132: 'qu',\n 133: 'gn',\n 134: 'ay',\n 135: 'tt',\n 136: 'ug',\n 137: 'dz',\n 138: 'jv',\n 139: 'su',\n 140: 'gl',\n 141: 'af',\n 142: 'br',\n 143: 'iu',\n 144: 'gd',\n 145: 'gv',\n 146: 'ga',\n 147: 'to',\n 148: 'el-polyton',\n 149: 'kl',\n 150: 'az',\n 151: 'nn'\n };\n\n // MacOS language ID → MacOS script ID\n //\n // Note that the script ID is not sufficient to determine what encoding\n // to use in TrueType files. For some languages, MacOS used a modification\n // of a mainstream script. For example, an Icelandic name would be stored\n // with smRoman in the TrueType naming table, but the actual encoding\n // is a special Icelandic version of the normal Macintosh Roman encoding.\n // As another example, Inuktitut uses an 8-bit encoding for Canadian Aboriginal\n // Syllables but MacOS had run out of available script codes, so this was\n // done as a (pretty radical) \"modification\" of Ethiopic.\n //\n // http://unicode.org/Public/MAPPINGS/VENDORS/APPLE/Readme.txt\n var macLanguageToScript = {\n 0: 0, // langEnglish → smRoman\n 1: 0, // langFrench → smRoman\n 2: 0, // langGerman → smRoman\n 3: 0, // langItalian → smRoman\n 4: 0, // langDutch → smRoman\n 5: 0, // langSwedish → smRoman\n 6: 0, // langSpanish → smRoman\n 7: 0, // langDanish → smRoman\n 8: 0, // langPortuguese → smRoman\n 9: 0, // langNorwegian → smRoman\n 10: 5, // langHebrew → smHebrew\n 11: 1, // langJapanese → smJapanese\n 12: 4, // langArabic → smArabic\n 13: 0, // langFinnish → smRoman\n 14: 6, // langGreek → smGreek\n 15: 0, // langIcelandic → smRoman (modified)\n 16: 0, // langMaltese → smRoman\n 17: 0, // langTurkish → smRoman (modified)\n 18: 0, // langCroatian → smRoman (modified)\n 19: 2, // langTradChinese → smTradChinese\n 20: 4, // langUrdu → smArabic\n 21: 9, // langHindi → smDevanagari\n 22: 21, // langThai → smThai\n 23: 3, // langKorean → smKorean\n 24: 29, // langLithuanian → smCentralEuroRoman\n 25: 29, // langPolish → smCentralEuroRoman\n 26: 29, // langHungarian → smCentralEuroRoman\n 27: 29, // langEstonian → smCentralEuroRoman\n 28: 29, // langLatvian → smCentralEuroRoman\n 29: 0, // langSami → smRoman\n 30: 0, // langFaroese → smRoman (modified)\n 31: 4, // langFarsi → smArabic (modified)\n 32: 7, // langRussian → smCyrillic\n 33: 25, // langSimpChinese → smSimpChinese\n 34: 0, // langFlemish → smRoman\n 35: 0, // langIrishGaelic → smRoman (modified)\n 36: 0, // langAlbanian → smRoman\n 37: 0, // langRomanian → smRoman (modified)\n 38: 29, // langCzech → smCentralEuroRoman\n 39: 29, // langSlovak → smCentralEuroRoman\n 40: 0, // langSlovenian → smRoman (modified)\n 41: 5, // langYiddish → smHebrew\n 42: 7, // langSerbian → smCyrillic\n 43: 7, // langMacedonian → smCyrillic\n 44: 7, // langBulgarian → smCyrillic\n 45: 7, // langUkrainian → smCyrillic (modified)\n 46: 7, // langByelorussian → smCyrillic\n 47: 7, // langUzbek → smCyrillic\n 48: 7, // langKazakh → smCyrillic\n 49: 7, // langAzerbaijani → smCyrillic\n 50: 4, // langAzerbaijanAr → smArabic\n 51: 24, // langArmenian → smArmenian\n 52: 23, // langGeorgian → smGeorgian\n 53: 7, // langMoldavian → smCyrillic\n 54: 7, // langKirghiz → smCyrillic\n 55: 7, // langTajiki → smCyrillic\n 56: 7, // langTurkmen → smCyrillic\n 57: 27, // langMongolian → smMongolian\n 58: 7, // langMongolianCyr → smCyrillic\n 59: 4, // langPashto → smArabic\n 60: 4, // langKurdish → smArabic\n 61: 4, // langKashmiri → smArabic\n 62: 4, // langSindhi → smArabic\n 63: 26, // langTibetan → smTibetan\n 64: 9, // langNepali → smDevanagari\n 65: 9, // langSanskrit → smDevanagari\n 66: 9, // langMarathi → smDevanagari\n 67: 13, // langBengali → smBengali\n 68: 13, // langAssamese → smBengali\n 69: 11, // langGujarati → smGujarati\n 70: 10, // langPunjabi → smGurmukhi\n 71: 12, // langOriya → smOriya\n 72: 17, // langMalayalam → smMalayalam\n 73: 16, // langKannada → smKannada\n 74: 14, // langTamil → smTamil\n 75: 15, // langTelugu → smTelugu\n 76: 18, // langSinhalese → smSinhalese\n 77: 19, // langBurmese → smBurmese\n 78: 20, // langKhmer → smKhmer\n 79: 22, // langLao → smLao\n 80: 30, // langVietnamese → smVietnamese\n 81: 0, // langIndonesian → smRoman\n 82: 0, // langTagalog → smRoman\n 83: 0, // langMalayRoman → smRoman\n 84: 4, // langMalayArabic → smArabic\n 85: 28, // langAmharic → smEthiopic\n 86: 28, // langTigrinya → smEthiopic\n 87: 28, // langOromo → smEthiopic\n 88: 0, // langSomali → smRoman\n 89: 0, // langSwahili → smRoman\n 90: 0, // langKinyarwanda → smRoman\n 91: 0, // langRundi → smRoman\n 92: 0, // langNyanja → smRoman\n 93: 0, // langMalagasy → smRoman\n 94: 0, // langEsperanto → smRoman\n 128: 0, // langWelsh → smRoman (modified)\n 129: 0, // langBasque → smRoman\n 130: 0, // langCatalan → smRoman\n 131: 0, // langLatin → smRoman\n 132: 0, // langQuechua → smRoman\n 133: 0, // langGuarani → smRoman\n 134: 0, // langAymara → smRoman\n 135: 7, // langTatar → smCyrillic\n 136: 4, // langUighur → smArabic\n 137: 26, // langDzongkha → smTibetan\n 138: 0, // langJavaneseRom → smRoman\n 139: 0, // langSundaneseRom → smRoman\n 140: 0, // langGalician → smRoman\n 141: 0, // langAfrikaans → smRoman\n 142: 0, // langBreton → smRoman (modified)\n 143: 28, // langInuktitut → smEthiopic (modified)\n 144: 0, // langScottishGaelic → smRoman (modified)\n 145: 0, // langManxGaelic → smRoman (modified)\n 146: 0, // langIrishGaelicScript → smRoman (modified)\n 147: 0, // langTongan → smRoman\n 148: 6, // langGreekAncient → smRoman\n 149: 0, // langGreenlandic → smRoman\n 150: 0, // langAzerbaijanRoman → smRoman\n 151: 0 // langNynorsk → smRoman\n };\n\n // While Microsoft indicates a region/country for all its language\n // IDs, we omit the region code if it's equal to the \"most likely\n // region subtag\" according to Unicode CLDR. For scripts, we omit\n // the subtag if it is equal to the Suppress-Script entry in the\n // IANA language subtag registry for IETF BCP 47.\n //\n // For example, Microsoft states that its language code 0x041A is\n // Croatian in Croatia. We transform this to the BCP 47 language code 'hr'\n // and not 'hr-HR' because Croatia is the default country for Croatian,\n // according to Unicode CLDR. As another example, Microsoft states\n // that 0x101A is Croatian (Latin) in Bosnia-Herzegovina. We transform\n // this to 'hr-BA' and not 'hr-Latn-BA' because Latin is the default script\n // for the Croatian language, according to IANA.\n //\n // http://www.unicode.org/cldr/charts/latest/supplemental/likely_subtags.html\n // http://www.iana.org/assignments/language-subtag-registry/language-subtag-registry\n var windowsLanguages = {\n 0x0436: 'af',\n 0x041C: 'sq',\n 0x0484: 'gsw',\n 0x045E: 'am',\n 0x1401: 'ar-DZ',\n 0x3C01: 'ar-BH',\n 0x0C01: 'ar',\n 0x0801: 'ar-IQ',\n 0x2C01: 'ar-JO',\n 0x3401: 'ar-KW',\n 0x3001: 'ar-LB',\n 0x1001: 'ar-LY',\n 0x1801: 'ary',\n 0x2001: 'ar-OM',\n 0x4001: 'ar-QA',\n 0x0401: 'ar-SA',\n 0x2801: 'ar-SY',\n 0x1C01: 'aeb',\n 0x3801: 'ar-AE',\n 0x2401: 'ar-YE',\n 0x042B: 'hy',\n 0x044D: 'as',\n 0x082C: 'az-Cyrl',\n 0x042C: 'az',\n 0x046D: 'ba',\n 0x042D: 'eu',\n 0x0423: 'be',\n 0x0845: 'bn',\n 0x0445: 'bn-IN',\n 0x201A: 'bs-Cyrl',\n 0x141A: 'bs',\n 0x047E: 'br',\n 0x0402: 'bg',\n 0x0403: 'ca',\n 0x0C04: 'zh-HK',\n 0x1404: 'zh-MO',\n 0x0804: 'zh',\n 0x1004: 'zh-SG',\n 0x0404: 'zh-TW',\n 0x0483: 'co',\n 0x041A: 'hr',\n 0x101A: 'hr-BA',\n 0x0405: 'cs',\n 0x0406: 'da',\n 0x048C: 'prs',\n 0x0465: 'dv',\n 0x0813: 'nl-BE',\n 0x0413: 'nl',\n 0x0C09: 'en-AU',\n 0x2809: 'en-BZ',\n 0x1009: 'en-CA',\n 0x2409: 'en-029',\n 0x4009: 'en-IN',\n 0x1809: 'en-IE',\n 0x2009: 'en-JM',\n 0x4409: 'en-MY',\n 0x1409: 'en-NZ',\n 0x3409: 'en-PH',\n 0x4809: 'en-SG',\n 0x1C09: 'en-ZA',\n 0x2C09: 'en-TT',\n 0x0809: 'en-GB',\n 0x0409: 'en',\n 0x3009: 'en-ZW',\n 0x0425: 'et',\n 0x0438: 'fo',\n 0x0464: 'fil',\n 0x040B: 'fi',\n 0x080C: 'fr-BE',\n 0x0C0C: 'fr-CA',\n 0x040C: 'fr',\n 0x140C: 'fr-LU',\n 0x180C: 'fr-MC',\n 0x100C: 'fr-CH',\n 0x0462: 'fy',\n 0x0456: 'gl',\n 0x0437: 'ka',\n 0x0C07: 'de-AT',\n 0x0407: 'de',\n 0x1407: 'de-LI',\n 0x1007: 'de-LU',\n 0x0807: 'de-CH',\n 0x0408: 'el',\n 0x046F: 'kl',\n 0x0447: 'gu',\n 0x0468: 'ha',\n 0x040D: 'he',\n 0x0439: 'hi',\n 0x040E: 'hu',\n 0x040F: 'is',\n 0x0470: 'ig',\n 0x0421: 'id',\n 0x045D: 'iu',\n 0x085D: 'iu-Latn',\n 0x083C: 'ga',\n 0x0434: 'xh',\n 0x0435: 'zu',\n 0x0410: 'it',\n 0x0810: 'it-CH',\n 0x0411: 'ja',\n 0x044B: 'kn',\n 0x043F: 'kk',\n 0x0453: 'km',\n 0x0486: 'quc',\n 0x0487: 'rw',\n 0x0441: 'sw',\n 0x0457: 'kok',\n 0x0412: 'ko',\n 0x0440: 'ky',\n 0x0454: 'lo',\n 0x0426: 'lv',\n 0x0427: 'lt',\n 0x082E: 'dsb',\n 0x046E: 'lb',\n 0x042F: 'mk',\n 0x083E: 'ms-BN',\n 0x043E: 'ms',\n 0x044C: 'ml',\n 0x043A: 'mt',\n 0x0481: 'mi',\n 0x047A: 'arn',\n 0x044E: 'mr',\n 0x047C: 'moh',\n 0x0450: 'mn',\n 0x0850: 'mn-CN',\n 0x0461: 'ne',\n 0x0414: 'nb',\n 0x0814: 'nn',\n 0x0482: 'oc',\n 0x0448: 'or',\n 0x0463: 'ps',\n 0x0415: 'pl',\n 0x0416: 'pt',\n 0x0816: 'pt-PT',\n 0x0446: 'pa',\n 0x046B: 'qu-BO',\n 0x086B: 'qu-EC',\n 0x0C6B: 'qu',\n 0x0418: 'ro',\n 0x0417: 'rm',\n 0x0419: 'ru',\n 0x243B: 'smn',\n 0x103B: 'smj-NO',\n 0x143B: 'smj',\n 0x0C3B: 'se-FI',\n 0x043B: 'se',\n 0x083B: 'se-SE',\n 0x203B: 'sms',\n 0x183B: 'sma-NO',\n 0x1C3B: 'sms',\n 0x044F: 'sa',\n 0x1C1A: 'sr-Cyrl-BA',\n 0x0C1A: 'sr',\n 0x181A: 'sr-Latn-BA',\n 0x081A: 'sr-Latn',\n 0x046C: 'nso',\n 0x0432: 'tn',\n 0x045B: 'si',\n 0x041B: 'sk',\n 0x0424: 'sl',\n 0x2C0A: 'es-AR',\n 0x400A: 'es-BO',\n 0x340A: 'es-CL',\n 0x240A: 'es-CO',\n 0x140A: 'es-CR',\n 0x1C0A: 'es-DO',\n 0x300A: 'es-EC',\n 0x440A: 'es-SV',\n 0x100A: 'es-GT',\n 0x480A: 'es-HN',\n 0x080A: 'es-MX',\n 0x4C0A: 'es-NI',\n 0x180A: 'es-PA',\n 0x3C0A: 'es-PY',\n 0x280A: 'es-PE',\n 0x500A: 'es-PR',\n\n // Microsoft has defined two different language codes for\n // “Spanish with modern sorting” and “Spanish with traditional\n // sorting”. This makes sense for collation APIs, and it would be\n // possible to express this in BCP 47 language tags via Unicode\n // extensions (eg., es-u-co-trad is Spanish with traditional\n // sorting). However, for storing names in fonts, the distinction\n // does not make sense, so we give “es” in both cases.\n 0x0C0A: 'es',\n 0x040A: 'es',\n\n 0x540A: 'es-US',\n 0x380A: 'es-UY',\n 0x200A: 'es-VE',\n 0x081D: 'sv-FI',\n 0x041D: 'sv',\n 0x045A: 'syr',\n 0x0428: 'tg',\n 0x085F: 'tzm',\n 0x0449: 'ta',\n 0x0444: 'tt',\n 0x044A: 'te',\n 0x041E: 'th',\n 0x0451: 'bo',\n 0x041F: 'tr',\n 0x0442: 'tk',\n 0x0480: 'ug',\n 0x0422: 'uk',\n 0x042E: 'hsb',\n 0x0420: 'ur',\n 0x0843: 'uz-Cyrl',\n 0x0443: 'uz',\n 0x042A: 'vi',\n 0x0452: 'cy',\n 0x0488: 'wo',\n 0x0485: 'sah',\n 0x0478: 'ii',\n 0x046A: 'yo'\n };\n\n // Returns a IETF BCP 47 language code, for example 'zh-Hant'\n // for 'Chinese in the traditional script'.\n function getLanguageCode(platformID, languageID, ltag) {\n switch (platformID) {\n case 0: // Unicode\n if (languageID === 0xFFFF) {\n return 'und';\n } else if (ltag) {\n return ltag[languageID];\n }\n\n break;\n\n case 1: // Macintosh\n return macLanguages[languageID];\n\n case 3: // Windows\n return windowsLanguages[languageID];\n }\n\n return undefined;\n }\n\n var utf16 = 'utf-16';\n\n // MacOS script ID → encoding. This table stores the default case,\n // which can be overridden by macLanguageEncodings.\n var macScriptEncodings = {\n 0: 'macintosh', // smRoman\n 1: 'x-mac-japanese', // smJapanese\n 2: 'x-mac-chinesetrad', // smTradChinese\n 3: 'x-mac-korean', // smKorean\n 6: 'x-mac-greek', // smGreek\n 7: 'x-mac-cyrillic', // smCyrillic\n 9: 'x-mac-devanagai', // smDevanagari\n 10: 'x-mac-gurmukhi', // smGurmukhi\n 11: 'x-mac-gujarati', // smGujarati\n 12: 'x-mac-oriya', // smOriya\n 13: 'x-mac-bengali', // smBengali\n 14: 'x-mac-tamil', // smTamil\n 15: 'x-mac-telugu', // smTelugu\n 16: 'x-mac-kannada', // smKannada\n 17: 'x-mac-malayalam', // smMalayalam\n 18: 'x-mac-sinhalese', // smSinhalese\n 19: 'x-mac-burmese', // smBurmese\n 20: 'x-mac-khmer', // smKhmer\n 21: 'x-mac-thai', // smThai\n 22: 'x-mac-lao', // smLao\n 23: 'x-mac-georgian', // smGeorgian\n 24: 'x-mac-armenian', // smArmenian\n 25: 'x-mac-chinesesimp', // smSimpChinese\n 26: 'x-mac-tibetan', // smTibetan\n 27: 'x-mac-mongolian', // smMongolian\n 28: 'x-mac-ethiopic', // smEthiopic\n 29: 'x-mac-ce', // smCentralEuroRoman\n 30: 'x-mac-vietnamese', // smVietnamese\n 31: 'x-mac-extarabic' // smExtArabic\n };\n\n // MacOS language ID → encoding. This table stores the exceptional\n // cases, which override macScriptEncodings. For writing MacOS naming\n // tables, we need to emit a MacOS script ID. Therefore, we cannot\n // merge macScriptEncodings into macLanguageEncodings.\n //\n // http://unicode.org/Public/MAPPINGS/VENDORS/APPLE/Readme.txt\n var macLanguageEncodings = {\n 15: 'x-mac-icelandic', // langIcelandic\n 17: 'x-mac-turkish', // langTurkish\n 18: 'x-mac-croatian', // langCroatian\n 24: 'x-mac-ce', // langLithuanian\n 25: 'x-mac-ce', // langPolish\n 26: 'x-mac-ce', // langHungarian\n 27: 'x-mac-ce', // langEstonian\n 28: 'x-mac-ce', // langLatvian\n 30: 'x-mac-icelandic', // langFaroese\n 37: 'x-mac-romanian', // langRomanian\n 38: 'x-mac-ce', // langCzech\n 39: 'x-mac-ce', // langSlovak\n 40: 'x-mac-ce', // langSlovenian\n 143: 'x-mac-inuit', // langInuktitut\n 146: 'x-mac-gaelic' // langIrishGaelicScript\n };\n\n function getEncoding(platformID, encodingID, languageID) {\n switch (platformID) {\n case 0: // Unicode\n return utf16;\n\n case 1: // Apple Macintosh\n return macLanguageEncodings[languageID] || macScriptEncodings[encodingID];\n\n case 3: // Microsoft Windows\n if (encodingID === 1 || encodingID === 10) {\n return utf16;\n }\n\n break;\n }\n\n return undefined;\n }\n\n // Parse the naming `name` table.\n // FIXME: Format 1 additional fields are not supported yet.\n // ltag is the content of the `ltag' table, such as ['en', 'zh-Hans', 'de-CH-1904'].\n function parseNameTable(data, start, ltag) {\n var name = {};\n var p = new parse.Parser(data, start);\n var format = p.parseUShort();\n var count = p.parseUShort();\n var stringOffset = p.offset + p.parseUShort();\n for (var i = 0; i < count; i++) {\n var platformID = p.parseUShort();\n var encodingID = p.parseUShort();\n var languageID = p.parseUShort();\n var nameID = p.parseUShort();\n var property = nameTableNames[nameID] || nameID;\n var byteLength = p.parseUShort();\n var offset = p.parseUShort();\n var language = getLanguageCode(platformID, languageID, ltag);\n var encoding = getEncoding(platformID, encodingID, languageID);\n if (encoding !== undefined && language !== undefined) {\n var text = (void 0);\n if (encoding === utf16) {\n text = decode.UTF16(data, stringOffset + offset, byteLength);\n } else {\n text = decode.MACSTRING(data, stringOffset + offset, byteLength, encoding);\n }\n\n if (text) {\n var translations = name[property];\n if (translations === undefined) {\n translations = name[property] = {};\n }\n\n translations[language] = text;\n }\n }\n }\n\n var langTagCount = 0;\n if (format === 1) {\n // FIXME: Also handle Microsoft's 'name' table 1.\n langTagCount = p.parseUShort();\n }\n\n return name;\n }\n\n // {23: 'foo'} → {'foo': 23}\n // ['bar', 'baz'] → {'bar': 0, 'baz': 1}\n function reverseDict(dict) {\n var result = {};\n for (var key in dict) {\n result[dict[key]] = parseInt(key);\n }\n\n return result;\n }\n\n function makeNameRecord(platformID, encodingID, languageID, nameID, length, offset) {\n return new table.Record('NameRecord', [\n {name: 'platformID', type: 'USHORT', value: platformID},\n {name: 'encodingID', type: 'USHORT', value: encodingID},\n {name: 'languageID', type: 'USHORT', value: languageID},\n {name: 'nameID', type: 'USHORT', value: nameID},\n {name: 'length', type: 'USHORT', value: length},\n {name: 'offset', type: 'USHORT', value: offset}\n ]);\n }\n\n // Finds the position of needle in haystack, or -1 if not there.\n // Like String.indexOf(), but for arrays.\n function findSubArray(needle, haystack) {\n var needleLength = needle.length;\n var limit = haystack.length - needleLength + 1;\n\n loop:\n for (var pos = 0; pos < limit; pos++) {\n for (; pos < limit; pos++) {\n for (var k = 0; k < needleLength; k++) {\n if (haystack[pos + k] !== needle[k]) {\n continue loop;\n }\n }\n\n return pos;\n }\n }\n\n return -1;\n }\n\n function addStringToPool(s, pool) {\n var offset = findSubArray(s, pool);\n if (offset < 0) {\n offset = pool.length;\n var i = 0;\n var len = s.length;\n for (; i < len; ++i) {\n pool.push(s[i]);\n }\n\n }\n\n return offset;\n }\n\n function makeNameTable(names, ltag) {\n var nameID;\n var nameIDs = [];\n\n var namesWithNumericKeys = {};\n var nameTableIds = reverseDict(nameTableNames);\n for (var key in names) {\n var id = nameTableIds[key];\n if (id === undefined) {\n id = key;\n }\n\n nameID = parseInt(id);\n\n if (isNaN(nameID)) {\n throw new Error('Name table entry \"' + key + '\" does not exist, see nameTableNames for complete list.');\n }\n\n namesWithNumericKeys[nameID] = names[key];\n nameIDs.push(nameID);\n }\n\n var macLanguageIds = reverseDict(macLanguages);\n var windowsLanguageIds = reverseDict(windowsLanguages);\n\n var nameRecords = [];\n var stringPool = [];\n\n for (var i = 0; i < nameIDs.length; i++) {\n nameID = nameIDs[i];\n var translations = namesWithNumericKeys[nameID];\n for (var lang in translations) {\n var text = translations[lang];\n\n // For MacOS, we try to emit the name in the form that was introduced\n // in the initial version of the TrueType spec (in the late 1980s).\n // However, this can fail for various reasons: the requested BCP 47\n // language code might not have an old-style Mac equivalent;\n // we might not have a codec for the needed character encoding;\n // or the name might contain characters that cannot be expressed\n // in the old-style Macintosh encoding. In case of failure, we emit\n // the name in a more modern fashion (Unicode encoding with BCP 47\n // language tags) that is recognized by MacOS 10.5, released in 2009.\n // If fonts were only read by operating systems, we could simply\n // emit all names in the modern form; this would be much easier.\n // However, there are many applications and libraries that read\n // 'name' tables directly, and these will usually only recognize\n // the ancient form (silently skipping the unrecognized names).\n var macPlatform = 1; // Macintosh\n var macLanguage = macLanguageIds[lang];\n var macScript = macLanguageToScript[macLanguage];\n var macEncoding = getEncoding(macPlatform, macScript, macLanguage);\n var macName = encode.MACSTRING(text, macEncoding);\n if (macName === undefined) {\n macPlatform = 0; // Unicode\n macLanguage = ltag.indexOf(lang);\n if (macLanguage < 0) {\n macLanguage = ltag.length;\n ltag.push(lang);\n }\n\n macScript = 4; // Unicode 2.0 and later\n macName = encode.UTF16(text);\n }\n\n var macNameOffset = addStringToPool(macName, stringPool);\n nameRecords.push(makeNameRecord(macPlatform, macScript, macLanguage,\n nameID, macName.length, macNameOffset));\n\n var winLanguage = windowsLanguageIds[lang];\n if (winLanguage !== undefined) {\n var winName = encode.UTF16(text);\n var winNameOffset = addStringToPool(winName, stringPool);\n nameRecords.push(makeNameRecord(3, 1, winLanguage,\n nameID, winName.length, winNameOffset));\n }\n }\n }\n\n nameRecords.sort(function(a, b) {\n return ((a.platformID - b.platformID) ||\n (a.encodingID - b.encodingID) ||\n (a.languageID - b.languageID) ||\n (a.nameID - b.nameID));\n });\n\n var t = new table.Table('name', [\n {name: 'format', type: 'USHORT', value: 0},\n {name: 'count', type: 'USHORT', value: nameRecords.length},\n {name: 'stringOffset', type: 'USHORT', value: 6 + nameRecords.length * 12}\n ]);\n\n for (var r = 0; r < nameRecords.length; r++) {\n t.fields.push({name: 'record_' + r, type: 'RECORD', value: nameRecords[r]});\n }\n\n t.fields.push({name: 'strings', type: 'LITERAL', value: stringPool});\n return t;\n }\n\n var _name = { parse: parseNameTable, make: makeNameTable };\n\n // The `OS/2` table contains metrics required in OpenType fonts.\n\n var unicodeRanges = [\n {begin: 0x0000, end: 0x007F}, // Basic Latin\n {begin: 0x0080, end: 0x00FF}, // Latin-1 Supplement\n {begin: 0x0100, end: 0x017F}, // Latin Extended-A\n {begin: 0x0180, end: 0x024F}, // Latin Extended-B\n {begin: 0x0250, end: 0x02AF}, // IPA Extensions\n {begin: 0x02B0, end: 0x02FF}, // Spacing Modifier Letters\n {begin: 0x0300, end: 0x036F}, // Combining Diacritical Marks\n {begin: 0x0370, end: 0x03FF}, // Greek and Coptic\n {begin: 0x2C80, end: 0x2CFF}, // Coptic\n {begin: 0x0400, end: 0x04FF}, // Cyrillic\n {begin: 0x0530, end: 0x058F}, // Armenian\n {begin: 0x0590, end: 0x05FF}, // Hebrew\n {begin: 0xA500, end: 0xA63F}, // Vai\n {begin: 0x0600, end: 0x06FF}, // Arabic\n {begin: 0x07C0, end: 0x07FF}, // NKo\n {begin: 0x0900, end: 0x097F}, // Devanagari\n {begin: 0x0980, end: 0x09FF}, // Bengali\n {begin: 0x0A00, end: 0x0A7F}, // Gurmukhi\n {begin: 0x0A80, end: 0x0AFF}, // Gujarati\n {begin: 0x0B00, end: 0x0B7F}, // Oriya\n {begin: 0x0B80, end: 0x0BFF}, // Tamil\n {begin: 0x0C00, end: 0x0C7F}, // Telugu\n {begin: 0x0C80, end: 0x0CFF}, // Kannada\n {begin: 0x0D00, end: 0x0D7F}, // Malayalam\n {begin: 0x0E00, end: 0x0E7F}, // Thai\n {begin: 0x0E80, end: 0x0EFF}, // Lao\n {begin: 0x10A0, end: 0x10FF}, // Georgian\n {begin: 0x1B00, end: 0x1B7F}, // Balinese\n {begin: 0x1100, end: 0x11FF}, // Hangul Jamo\n {begin: 0x1E00, end: 0x1EFF}, // Latin Extended Additional\n {begin: 0x1F00, end: 0x1FFF}, // Greek Extended\n {begin: 0x2000, end: 0x206F}, // General Punctuation\n {begin: 0x2070, end: 0x209F}, // Superscripts And Subscripts\n {begin: 0x20A0, end: 0x20CF}, // Currency Symbol\n {begin: 0x20D0, end: 0x20FF}, // Combining Diacritical Marks For Symbols\n {begin: 0x2100, end: 0x214F}, // Letterlike Symbols\n {begin: 0x2150, end: 0x218F}, // Number Forms\n {begin: 0x2190, end: 0x21FF}, // Arrows\n {begin: 0x2200, end: 0x22FF}, // Mathematical Operators\n {begin: 0x2300, end: 0x23FF}, // Miscellaneous Technical\n {begin: 0x2400, end: 0x243F}, // Control Pictures\n {begin: 0x2440, end: 0x245F}, // Optical Character Recognition\n {begin: 0x2460, end: 0x24FF}, // Enclosed Alphanumerics\n {begin: 0x2500, end: 0x257F}, // Box Drawing\n {begin: 0x2580, end: 0x259F}, // Block Elements\n {begin: 0x25A0, end: 0x25FF}, // Geometric Shapes\n {begin: 0x2600, end: 0x26FF}, // Miscellaneous Symbols\n {begin: 0x2700, end: 0x27BF}, // Dingbats\n {begin: 0x3000, end: 0x303F}, // CJK Symbols And Punctuation\n {begin: 0x3040, end: 0x309F}, // Hiragana\n {begin: 0x30A0, end: 0x30FF}, // Katakana\n {begin: 0x3100, end: 0x312F}, // Bopomofo\n {begin: 0x3130, end: 0x318F}, // Hangul Compatibility Jamo\n {begin: 0xA840, end: 0xA87F}, // Phags-pa\n {begin: 0x3200, end: 0x32FF}, // Enclosed CJK Letters And Months\n {begin: 0x3300, end: 0x33FF}, // CJK Compatibility\n {begin: 0xAC00, end: 0xD7AF}, // Hangul Syllables\n {begin: 0xD800, end: 0xDFFF}, // Non-Plane 0 *\n {begin: 0x10900, end: 0x1091F}, // Phoenicia\n {begin: 0x4E00, end: 0x9FFF}, // CJK Unified Ideographs\n {begin: 0xE000, end: 0xF8FF}, // Private Use Area (plane 0)\n {begin: 0x31C0, end: 0x31EF}, // CJK Strokes\n {begin: 0xFB00, end: 0xFB4F}, // Alphabetic Presentation Forms\n {begin: 0xFB50, end: 0xFDFF}, // Arabic Presentation Forms-A\n {begin: 0xFE20, end: 0xFE2F}, // Combining Half Marks\n {begin: 0xFE10, end: 0xFE1F}, // Vertical Forms\n {begin: 0xFE50, end: 0xFE6F}, // Small Form Variants\n {begin: 0xFE70, end: 0xFEFF}, // Arabic Presentation Forms-B\n {begin: 0xFF00, end: 0xFFEF}, // Halfwidth And Fullwidth Forms\n {begin: 0xFFF0, end: 0xFFFF}, // Specials\n {begin: 0x0F00, end: 0x0FFF}, // Tibetan\n {begin: 0x0700, end: 0x074F}, // Syriac\n {begin: 0x0780, end: 0x07BF}, // Thaana\n {begin: 0x0D80, end: 0x0DFF}, // Sinhala\n {begin: 0x1000, end: 0x109F}, // Myanmar\n {begin: 0x1200, end: 0x137F}, // Ethiopic\n {begin: 0x13A0, end: 0x13FF}, // Cherokee\n {begin: 0x1400, end: 0x167F}, // Unified Canadian Aboriginal Syllabics\n {begin: 0x1680, end: 0x169F}, // Ogham\n {begin: 0x16A0, end: 0x16FF}, // Runic\n {begin: 0x1780, end: 0x17FF}, // Khmer\n {begin: 0x1800, end: 0x18AF}, // Mongolian\n {begin: 0x2800, end: 0x28FF}, // Braille Patterns\n {begin: 0xA000, end: 0xA48F}, // Yi Syllables\n {begin: 0x1700, end: 0x171F}, // Tagalog\n {begin: 0x10300, end: 0x1032F}, // Old Italic\n {begin: 0x10330, end: 0x1034F}, // Gothic\n {begin: 0x10400, end: 0x1044F}, // Deseret\n {begin: 0x1D000, end: 0x1D0FF}, // Byzantine Musical Symbols\n {begin: 0x1D400, end: 0x1D7FF}, // Mathematical Alphanumeric Symbols\n {begin: 0xFF000, end: 0xFFFFD}, // Private Use (plane 15)\n {begin: 0xFE00, end: 0xFE0F}, // Variation Selectors\n {begin: 0xE0000, end: 0xE007F}, // Tags\n {begin: 0x1900, end: 0x194F}, // Limbu\n {begin: 0x1950, end: 0x197F}, // Tai Le\n {begin: 0x1980, end: 0x19DF}, // New Tai Lue\n {begin: 0x1A00, end: 0x1A1F}, // Buginese\n {begin: 0x2C00, end: 0x2C5F}, // Glagolitic\n {begin: 0x2D30, end: 0x2D7F}, // Tifinagh\n {begin: 0x4DC0, end: 0x4DFF}, // Yijing Hexagram Symbols\n {begin: 0xA800, end: 0xA82F}, // Syloti Nagri\n {begin: 0x10000, end: 0x1007F}, // Linear B Syllabary\n {begin: 0x10140, end: 0x1018F}, // Ancient Greek Numbers\n {begin: 0x10380, end: 0x1039F}, // Ugaritic\n {begin: 0x103A0, end: 0x103DF}, // Old Persian\n {begin: 0x10450, end: 0x1047F}, // Shavian\n {begin: 0x10480, end: 0x104AF}, // Osmanya\n {begin: 0x10800, end: 0x1083F}, // Cypriot Syllabary\n {begin: 0x10A00, end: 0x10A5F}, // Kharoshthi\n {begin: 0x1D300, end: 0x1D35F}, // Tai Xuan Jing Symbols\n {begin: 0x12000, end: 0x123FF}, // Cuneiform\n {begin: 0x1D360, end: 0x1D37F}, // Counting Rod Numerals\n {begin: 0x1B80, end: 0x1BBF}, // Sundanese\n {begin: 0x1C00, end: 0x1C4F}, // Lepcha\n {begin: 0x1C50, end: 0x1C7F}, // Ol Chiki\n {begin: 0xA880, end: 0xA8DF}, // Saurashtra\n {begin: 0xA900, end: 0xA92F}, // Kayah Li\n {begin: 0xA930, end: 0xA95F}, // Rejang\n {begin: 0xAA00, end: 0xAA5F}, // Cham\n {begin: 0x10190, end: 0x101CF}, // Ancient Symbols\n {begin: 0x101D0, end: 0x101FF}, // Phaistos Disc\n {begin: 0x102A0, end: 0x102DF}, // Carian\n {begin: 0x1F030, end: 0x1F09F} // Domino Tiles\n ];\n\n function getUnicodeRange(unicode) {\n for (var i = 0; i < unicodeRanges.length; i += 1) {\n var range = unicodeRanges[i];\n if (unicode >= range.begin && unicode < range.end) {\n return i;\n }\n }\n\n return -1;\n }\n\n // Parse the OS/2 and Windows metrics `OS/2` table\n function parseOS2Table(data, start) {\n var os2 = {};\n var p = new parse.Parser(data, start);\n os2.version = p.parseUShort();\n os2.xAvgCharWidth = p.parseShort();\n os2.usWeightClass = p.parseUShort();\n os2.usWidthClass = p.parseUShort();\n os2.fsType = p.parseUShort();\n os2.ySubscriptXSize = p.parseShort();\n os2.ySubscriptYSize = p.parseShort();\n os2.ySubscriptXOffset = p.parseShort();\n os2.ySubscriptYOffset = p.parseShort();\n os2.ySuperscriptXSize = p.parseShort();\n os2.ySuperscriptYSize = p.parseShort();\n os2.ySuperscriptXOffset = p.parseShort();\n os2.ySuperscriptYOffset = p.parseShort();\n os2.yStrikeoutSize = p.parseShort();\n os2.yStrikeoutPosition = p.parseShort();\n os2.sFamilyClass = p.parseShort();\n os2.panose = [];\n for (var i = 0; i < 10; i++) {\n os2.panose[i] = p.parseByte();\n }\n\n os2.ulUnicodeRange1 = p.parseULong();\n os2.ulUnicodeRange2 = p.parseULong();\n os2.ulUnicodeRange3 = p.parseULong();\n os2.ulUnicodeRange4 = p.parseULong();\n os2.achVendID = String.fromCharCode(p.parseByte(), p.parseByte(), p.parseByte(), p.parseByte());\n os2.fsSelection = p.parseUShort();\n os2.usFirstCharIndex = p.parseUShort();\n os2.usLastCharIndex = p.parseUShort();\n os2.sTypoAscender = p.parseShort();\n os2.sTypoDescender = p.parseShort();\n os2.sTypoLineGap = p.parseShort();\n os2.usWinAscent = p.parseUShort();\n os2.usWinDescent = p.parseUShort();\n if (os2.version >= 1) {\n os2.ulCodePageRange1 = p.parseULong();\n os2.ulCodePageRange2 = p.parseULong();\n }\n\n if (os2.version >= 2) {\n os2.sxHeight = p.parseShort();\n os2.sCapHeight = p.parseShort();\n os2.usDefaultChar = p.parseUShort();\n os2.usBreakChar = p.parseUShort();\n os2.usMaxContent = p.parseUShort();\n }\n\n return os2;\n }\n\n function makeOS2Table(options) {\n return new table.Table('OS/2', [\n {name: 'version', type: 'USHORT', value: 0x0003},\n {name: 'xAvgCharWidth', type: 'SHORT', value: 0},\n {name: 'usWeightClass', type: 'USHORT', value: 0},\n {name: 'usWidthClass', type: 'USHORT', value: 0},\n {name: 'fsType', type: 'USHORT', value: 0},\n {name: 'ySubscriptXSize', type: 'SHORT', value: 650},\n {name: 'ySubscriptYSize', type: 'SHORT', value: 699},\n {name: 'ySubscriptXOffset', type: 'SHORT', value: 0},\n {name: 'ySubscriptYOffset', type: 'SHORT', value: 140},\n {name: 'ySuperscriptXSize', type: 'SHORT', value: 650},\n {name: 'ySuperscriptYSize', type: 'SHORT', value: 699},\n {name: 'ySuperscriptXOffset', type: 'SHORT', value: 0},\n {name: 'ySuperscriptYOffset', type: 'SHORT', value: 479},\n {name: 'yStrikeoutSize', type: 'SHORT', value: 49},\n {name: 'yStrikeoutPosition', type: 'SHORT', value: 258},\n {name: 'sFamilyClass', type: 'SHORT', value: 0},\n {name: 'bFamilyType', type: 'BYTE', value: 0},\n {name: 'bSerifStyle', type: 'BYTE', value: 0},\n {name: 'bWeight', type: 'BYTE', value: 0},\n {name: 'bProportion', type: 'BYTE', value: 0},\n {name: 'bContrast', type: 'BYTE', value: 0},\n {name: 'bStrokeVariation', type: 'BYTE', value: 0},\n {name: 'bArmStyle', type: 'BYTE', value: 0},\n {name: 'bLetterform', type: 'BYTE', value: 0},\n {name: 'bMidline', type: 'BYTE', value: 0},\n {name: 'bXHeight', type: 'BYTE', value: 0},\n {name: 'ulUnicodeRange1', type: 'ULONG', value: 0},\n {name: 'ulUnicodeRange2', type: 'ULONG', value: 0},\n {name: 'ulUnicodeRange3', type: 'ULONG', value: 0},\n {name: 'ulUnicodeRange4', type: 'ULONG', value: 0},\n {name: 'achVendID', type: 'CHARARRAY', value: 'XXXX'},\n {name: 'fsSelection', type: 'USHORT', value: 0},\n {name: 'usFirstCharIndex', type: 'USHORT', value: 0},\n {name: 'usLastCharIndex', type: 'USHORT', value: 0},\n {name: 'sTypoAscender', type: 'SHORT', value: 0},\n {name: 'sTypoDescender', type: 'SHORT', value: 0},\n {name: 'sTypoLineGap', type: 'SHORT', value: 0},\n {name: 'usWinAscent', type: 'USHORT', value: 0},\n {name: 'usWinDescent', type: 'USHORT', value: 0},\n {name: 'ulCodePageRange1', type: 'ULONG', value: 0},\n {name: 'ulCodePageRange2', type: 'ULONG', value: 0},\n {name: 'sxHeight', type: 'SHORT', value: 0},\n {name: 'sCapHeight', type: 'SHORT', value: 0},\n {name: 'usDefaultChar', type: 'USHORT', value: 0},\n {name: 'usBreakChar', type: 'USHORT', value: 0},\n {name: 'usMaxContext', type: 'USHORT', value: 0}\n ], options);\n }\n\n var os2 = { parse: parseOS2Table, make: makeOS2Table, unicodeRanges: unicodeRanges, getUnicodeRange: getUnicodeRange };\n\n // The `post` table stores additional PostScript information, such as glyph names.\n\n // Parse the PostScript `post` table\n function parsePostTable(data, start) {\n var post = {};\n var p = new parse.Parser(data, start);\n post.version = p.parseVersion();\n post.italicAngle = p.parseFixed();\n post.underlinePosition = p.parseShort();\n post.underlineThickness = p.parseShort();\n post.isFixedPitch = p.parseULong();\n post.minMemType42 = p.parseULong();\n post.maxMemType42 = p.parseULong();\n post.minMemType1 = p.parseULong();\n post.maxMemType1 = p.parseULong();\n switch (post.version) {\n case 1:\n post.names = standardNames.slice();\n break;\n case 2:\n post.numberOfGlyphs = p.parseUShort();\n post.glyphNameIndex = new Array(post.numberOfGlyphs);\n for (var i = 0; i < post.numberOfGlyphs; i++) {\n post.glyphNameIndex[i] = p.parseUShort();\n }\n\n post.names = [];\n for (var i$1 = 0; i$1 < post.numberOfGlyphs; i$1++) {\n if (post.glyphNameIndex[i$1] >= standardNames.length) {\n var nameLength = p.parseChar();\n post.names.push(p.parseString(nameLength));\n }\n }\n\n break;\n case 2.5:\n post.numberOfGlyphs = p.parseUShort();\n post.offset = new Array(post.numberOfGlyphs);\n for (var i$2 = 0; i$2 < post.numberOfGlyphs; i$2++) {\n post.offset[i$2] = p.parseChar();\n }\n\n break;\n }\n return post;\n }\n\n function makePostTable() {\n return new table.Table('post', [\n {name: 'version', type: 'FIXED', value: 0x00030000},\n {name: 'italicAngle', type: 'FIXED', value: 0},\n {name: 'underlinePosition', type: 'FWORD', value: 0},\n {name: 'underlineThickness', type: 'FWORD', value: 0},\n {name: 'isFixedPitch', type: 'ULONG', value: 0},\n {name: 'minMemType42', type: 'ULONG', value: 0},\n {name: 'maxMemType42', type: 'ULONG', value: 0},\n {name: 'minMemType1', type: 'ULONG', value: 0},\n {name: 'maxMemType1', type: 'ULONG', value: 0}\n ]);\n }\n\n var post = { parse: parsePostTable, make: makePostTable };\n\n // The `GSUB` table contains ligatures, among other things.\n\n var subtableParsers = new Array(9); // subtableParsers[0] is unused\n\n // https://www.microsoft.com/typography/OTSPEC/GSUB.htm#SS\n subtableParsers[1] = function parseLookup1() {\n var start = this.offset + this.relativeOffset;\n var substFormat = this.parseUShort();\n if (substFormat === 1) {\n return {\n substFormat: 1,\n coverage: this.parsePointer(Parser.coverage),\n deltaGlyphId: this.parseUShort()\n };\n } else if (substFormat === 2) {\n return {\n substFormat: 2,\n coverage: this.parsePointer(Parser.coverage),\n substitute: this.parseOffset16List()\n };\n }\n check.assert(false, '0x' + start.toString(16) + ': lookup type 1 format must be 1 or 2.');\n };\n\n // https://www.microsoft.com/typography/OTSPEC/GSUB.htm#MS\n subtableParsers[2] = function parseLookup2() {\n var substFormat = this.parseUShort();\n check.argument(substFormat === 1, 'GSUB Multiple Substitution Subtable identifier-format must be 1');\n return {\n substFormat: substFormat,\n coverage: this.parsePointer(Parser.coverage),\n sequences: this.parseListOfLists()\n };\n };\n\n // https://www.microsoft.com/typography/OTSPEC/GSUB.htm#AS\n subtableParsers[3] = function parseLookup3() {\n var substFormat = this.parseUShort();\n check.argument(substFormat === 1, 'GSUB Alternate Substitution Subtable identifier-format must be 1');\n return {\n substFormat: substFormat,\n coverage: this.parsePointer(Parser.coverage),\n alternateSets: this.parseListOfLists()\n };\n };\n\n // https://www.microsoft.com/typography/OTSPEC/GSUB.htm#LS\n subtableParsers[4] = function parseLookup4() {\n var substFormat = this.parseUShort();\n check.argument(substFormat === 1, 'GSUB ligature table identifier-format must be 1');\n return {\n substFormat: substFormat,\n coverage: this.parsePointer(Parser.coverage),\n ligatureSets: this.parseListOfLists(function() {\n return {\n ligGlyph: this.parseUShort(),\n components: this.parseUShortList(this.parseUShort() - 1)\n };\n })\n };\n };\n\n var lookupRecordDesc = {\n sequenceIndex: Parser.uShort,\n lookupListIndex: Parser.uShort\n };\n\n // https://www.microsoft.com/typography/OTSPEC/GSUB.htm#CSF\n subtableParsers[5] = function parseLookup5() {\n var start = this.offset + this.relativeOffset;\n var substFormat = this.parseUShort();\n\n if (substFormat === 1) {\n return {\n substFormat: substFormat,\n coverage: this.parsePointer(Parser.coverage),\n ruleSets: this.parseListOfLists(function() {\n var glyphCount = this.parseUShort();\n var substCount = this.parseUShort();\n return {\n input: this.parseUShortList(glyphCount - 1),\n lookupRecords: this.parseRecordList(substCount, lookupRecordDesc)\n };\n })\n };\n } else if (substFormat === 2) {\n return {\n substFormat: substFormat,\n coverage: this.parsePointer(Parser.coverage),\n classDef: this.parsePointer(Parser.classDef),\n classSets: this.parseListOfLists(function() {\n var glyphCount = this.parseUShort();\n var substCount = this.parseUShort();\n return {\n classes: this.parseUShortList(glyphCount - 1),\n lookupRecords: this.parseRecordList(substCount, lookupRecordDesc)\n };\n })\n };\n } else if (substFormat === 3) {\n var glyphCount = this.parseUShort();\n var substCount = this.parseUShort();\n return {\n substFormat: substFormat,\n coverages: this.parseList(glyphCount, Parser.pointer(Parser.coverage)),\n lookupRecords: this.parseRecordList(substCount, lookupRecordDesc)\n };\n }\n check.assert(false, '0x' + start.toString(16) + ': lookup type 5 format must be 1, 2 or 3.');\n };\n\n // https://www.microsoft.com/typography/OTSPEC/GSUB.htm#CC\n subtableParsers[6] = function parseLookup6() {\n var start = this.offset + this.relativeOffset;\n var substFormat = this.parseUShort();\n if (substFormat === 1) {\n return {\n substFormat: 1,\n coverage: this.parsePointer(Parser.coverage),\n chainRuleSets: this.parseListOfLists(function() {\n return {\n backtrack: this.parseUShortList(),\n input: this.parseUShortList(this.parseShort() - 1),\n lookahead: this.parseUShortList(),\n lookupRecords: this.parseRecordList(lookupRecordDesc)\n };\n })\n };\n } else if (substFormat === 2) {\n return {\n substFormat: 2,\n coverage: this.parsePointer(Parser.coverage),\n backtrackClassDef: this.parsePointer(Parser.classDef),\n inputClassDef: this.parsePointer(Parser.classDef),\n lookaheadClassDef: this.parsePointer(Parser.classDef),\n chainClassSet: this.parseListOfLists(function() {\n return {\n backtrack: this.parseUShortList(),\n input: this.parseUShortList(this.parseShort() - 1),\n lookahead: this.parseUShortList(),\n lookupRecords: this.parseRecordList(lookupRecordDesc)\n };\n })\n };\n } else if (substFormat === 3) {\n return {\n substFormat: 3,\n backtrackCoverage: this.parseList(Parser.pointer(Parser.coverage)),\n inputCoverage: this.parseList(Parser.pointer(Parser.coverage)),\n lookaheadCoverage: this.parseList(Parser.pointer(Parser.coverage)),\n lookupRecords: this.parseRecordList(lookupRecordDesc)\n };\n }\n check.assert(false, '0x' + start.toString(16) + ': lookup type 6 format must be 1, 2 or 3.');\n };\n\n // https://www.microsoft.com/typography/OTSPEC/GSUB.htm#ES\n subtableParsers[7] = function parseLookup7() {\n // Extension Substitution subtable\n var substFormat = this.parseUShort();\n check.argument(substFormat === 1, 'GSUB Extension Substitution subtable identifier-format must be 1');\n var extensionLookupType = this.parseUShort();\n var extensionParser = new Parser(this.data, this.offset + this.parseULong());\n return {\n substFormat: 1,\n lookupType: extensionLookupType,\n extension: subtableParsers[extensionLookupType].call(extensionParser)\n };\n };\n\n // https://www.microsoft.com/typography/OTSPEC/GSUB.htm#RCCS\n subtableParsers[8] = function parseLookup8() {\n var substFormat = this.parseUShort();\n check.argument(substFormat === 1, 'GSUB Reverse Chaining Contextual Single Substitution Subtable identifier-format must be 1');\n return {\n substFormat: substFormat,\n coverage: this.parsePointer(Parser.coverage),\n backtrackCoverage: this.parseList(Parser.pointer(Parser.coverage)),\n lookaheadCoverage: this.parseList(Parser.pointer(Parser.coverage)),\n substitutes: this.parseUShortList()\n };\n };\n\n // https://www.microsoft.com/typography/OTSPEC/gsub.htm\n function parseGsubTable(data, start) {\n start = start || 0;\n var p = new Parser(data, start);\n var tableVersion = p.parseVersion(1);\n check.argument(tableVersion === 1 || tableVersion === 1.1, 'Unsupported GSUB table version.');\n if (tableVersion === 1) {\n return {\n version: tableVersion,\n scripts: p.parseScriptList(),\n features: p.parseFeatureList(),\n lookups: p.parseLookupList(subtableParsers)\n };\n } else {\n return {\n version: tableVersion,\n scripts: p.parseScriptList(),\n features: p.parseFeatureList(),\n lookups: p.parseLookupList(subtableParsers),\n variations: p.parseFeatureVariationsList()\n };\n }\n\n }\n\n // GSUB Writing //////////////////////////////////////////////\n var subtableMakers = new Array(9);\n\n subtableMakers[1] = function makeLookup1(subtable) {\n if (subtable.substFormat === 1) {\n return new table.Table('substitutionTable', [\n {name: 'substFormat', type: 'USHORT', value: 1},\n {name: 'coverage', type: 'TABLE', value: new table.Coverage(subtable.coverage)},\n {name: 'deltaGlyphID', type: 'USHORT', value: subtable.deltaGlyphId}\n ]);\n } else {\n return new table.Table('substitutionTable', [\n {name: 'substFormat', type: 'USHORT', value: 2},\n {name: 'coverage', type: 'TABLE', value: new table.Coverage(subtable.coverage)}\n ].concat(table.ushortList('substitute', subtable.substitute)));\n }\n };\n\n subtableMakers[2] = function makeLookup2(subtable) {\n check.assert(subtable.substFormat === 1, 'Lookup type 2 substFormat must be 1.');\n return new table.Table('substitutionTable', [\n {name: 'substFormat', type: 'USHORT', value: 1},\n {name: 'coverage', type: 'TABLE', value: new table.Coverage(subtable.coverage)}\n ].concat(table.tableList('seqSet', subtable.sequences, function(sequenceSet) {\n return new table.Table('sequenceSetTable', table.ushortList('sequence', sequenceSet));\n })));\n };\n\n subtableMakers[3] = function makeLookup3(subtable) {\n check.assert(subtable.substFormat === 1, 'Lookup type 3 substFormat must be 1.');\n return new table.Table('substitutionTable', [\n {name: 'substFormat', type: 'USHORT', value: 1},\n {name: 'coverage', type: 'TABLE', value: new table.Coverage(subtable.coverage)}\n ].concat(table.tableList('altSet', subtable.alternateSets, function(alternateSet) {\n return new table.Table('alternateSetTable', table.ushortList('alternate', alternateSet));\n })));\n };\n\n subtableMakers[4] = function makeLookup4(subtable) {\n check.assert(subtable.substFormat === 1, 'Lookup type 4 substFormat must be 1.');\n return new table.Table('substitutionTable', [\n {name: 'substFormat', type: 'USHORT', value: 1},\n {name: 'coverage', type: 'TABLE', value: new table.Coverage(subtable.coverage)}\n ].concat(table.tableList('ligSet', subtable.ligatureSets, function(ligatureSet) {\n return new table.Table('ligatureSetTable', table.tableList('ligature', ligatureSet, function(ligature) {\n return new table.Table('ligatureTable',\n [{name: 'ligGlyph', type: 'USHORT', value: ligature.ligGlyph}]\n .concat(table.ushortList('component', ligature.components, ligature.components.length + 1))\n );\n }));\n })));\n };\n\n subtableMakers[6] = function makeLookup6(subtable) {\n if (subtable.substFormat === 1) {\n var returnTable = new table.Table('chainContextTable', [\n {name: 'substFormat', type: 'USHORT', value: subtable.substFormat},\n {name: 'coverage', type: 'TABLE', value: new table.Coverage(subtable.coverage)}\n ].concat(table.tableList('chainRuleSet', subtable.chainRuleSets, function(chainRuleSet) {\n return new table.Table('chainRuleSetTable', table.tableList('chainRule', chainRuleSet, function(chainRule) {\n var tableData = table.ushortList('backtrackGlyph', chainRule.backtrack, chainRule.backtrack.length)\n .concat(table.ushortList('inputGlyph', chainRule.input, chainRule.input.length + 1))\n .concat(table.ushortList('lookaheadGlyph', chainRule.lookahead, chainRule.lookahead.length))\n .concat(table.ushortList('substitution', [], chainRule.lookupRecords.length));\n\n chainRule.lookupRecords.forEach(function (record, i) {\n tableData = tableData\n .concat({name: 'sequenceIndex' + i, type: 'USHORT', value: record.sequenceIndex})\n .concat({name: 'lookupListIndex' + i, type: 'USHORT', value: record.lookupListIndex});\n });\n return new table.Table('chainRuleTable', tableData);\n }));\n })));\n return returnTable;\n } else if (subtable.substFormat === 2) {\n check.assert(false, 'lookup type 6 format 2 is not yet supported.');\n } else if (subtable.substFormat === 3) {\n var tableData = [\n {name: 'substFormat', type: 'USHORT', value: subtable.substFormat} ];\n\n tableData.push({name: 'backtrackGlyphCount', type: 'USHORT', value: subtable.backtrackCoverage.length});\n subtable.backtrackCoverage.forEach(function (coverage, i) {\n tableData.push({name: 'backtrackCoverage' + i, type: 'TABLE', value: new table.Coverage(coverage)});\n });\n tableData.push({name: 'inputGlyphCount', type: 'USHORT', value: subtable.inputCoverage.length});\n subtable.inputCoverage.forEach(function (coverage, i) {\n tableData.push({name: 'inputCoverage' + i, type: 'TABLE', value: new table.Coverage(coverage)});\n });\n tableData.push({name: 'lookaheadGlyphCount', type: 'USHORT', value: subtable.lookaheadCoverage.length});\n subtable.lookaheadCoverage.forEach(function (coverage, i) {\n tableData.push({name: 'lookaheadCoverage' + i, type: 'TABLE', value: new table.Coverage(coverage)});\n });\n\n tableData.push({name: 'substitutionCount', type: 'USHORT', value: subtable.lookupRecords.length});\n subtable.lookupRecords.forEach(function (record, i) {\n tableData = tableData\n .concat({name: 'sequenceIndex' + i, type: 'USHORT', value: record.sequenceIndex})\n .concat({name: 'lookupListIndex' + i, type: 'USHORT', value: record.lookupListIndex});\n });\n\n var returnTable$1 = new table.Table('chainContextTable', tableData);\n\n return returnTable$1;\n }\n\n check.assert(false, 'lookup type 6 format must be 1, 2 or 3.');\n };\n\n function makeGsubTable(gsub) {\n return new table.Table('GSUB', [\n {name: 'version', type: 'ULONG', value: 0x10000},\n {name: 'scripts', type: 'TABLE', value: new table.ScriptList(gsub.scripts)},\n {name: 'features', type: 'TABLE', value: new table.FeatureList(gsub.features)},\n {name: 'lookups', type: 'TABLE', value: new table.LookupList(gsub.lookups, subtableMakers)}\n ]);\n }\n\n var gsub = { parse: parseGsubTable, make: makeGsubTable };\n\n // The `GPOS` table contains kerning pairs, among other things.\n\n // Parse the metadata `meta` table.\n // https://developer.apple.com/fonts/TrueType-Reference-Manual/RM06/Chap6meta.html\n function parseMetaTable(data, start) {\n var p = new parse.Parser(data, start);\n var tableVersion = p.parseULong();\n check.argument(tableVersion === 1, 'Unsupported META table version.');\n p.parseULong(); // flags - currently unused and set to 0\n p.parseULong(); // tableOffset\n var numDataMaps = p.parseULong();\n\n var tags = {};\n for (var i = 0; i < numDataMaps; i++) {\n var tag = p.parseTag();\n var dataOffset = p.parseULong();\n var dataLength = p.parseULong();\n var text = decode.UTF8(data, start + dataOffset, dataLength);\n\n tags[tag] = text;\n }\n return tags;\n }\n\n function makeMetaTable(tags) {\n var numTags = Object.keys(tags).length;\n var stringPool = '';\n var stringPoolOffset = 16 + numTags * 12;\n\n var result = new table.Table('meta', [\n {name: 'version', type: 'ULONG', value: 1},\n {name: 'flags', type: 'ULONG', value: 0},\n {name: 'offset', type: 'ULONG', value: stringPoolOffset},\n {name: 'numTags', type: 'ULONG', value: numTags}\n ]);\n\n for (var tag in tags) {\n var pos = stringPool.length;\n stringPool += tags[tag];\n\n result.fields.push({name: 'tag ' + tag, type: 'TAG', value: tag});\n result.fields.push({name: 'offset ' + tag, type: 'ULONG', value: stringPoolOffset + pos});\n result.fields.push({name: 'length ' + tag, type: 'ULONG', value: tags[tag].length});\n }\n\n result.fields.push({name: 'stringPool', type: 'CHARARRAY', value: stringPool});\n\n return result;\n }\n\n var meta = { parse: parseMetaTable, make: makeMetaTable };\n\n // The `sfnt` wrapper provides organization for the tables in the font.\n\n function log2(v) {\n return Math.log(v) / Math.log(2) | 0;\n }\n\n function computeCheckSum(bytes) {\n while (bytes.length % 4 !== 0) {\n bytes.push(0);\n }\n\n var sum = 0;\n for (var i = 0; i < bytes.length; i += 4) {\n sum += (bytes[i] << 24) +\n (bytes[i + 1] << 16) +\n (bytes[i + 2] << 8) +\n (bytes[i + 3]);\n }\n\n sum %= Math.pow(2, 32);\n return sum;\n }\n\n function makeTableRecord(tag, checkSum, offset, length) {\n return new table.Record('Table Record', [\n {name: 'tag', type: 'TAG', value: tag !== undefined ? tag : ''},\n {name: 'checkSum', type: 'ULONG', value: checkSum !== undefined ? checkSum : 0},\n {name: 'offset', type: 'ULONG', value: offset !== undefined ? offset : 0},\n {name: 'length', type: 'ULONG', value: length !== undefined ? length : 0}\n ]);\n }\n\n function makeSfntTable(tables) {\n var sfnt = new table.Table('sfnt', [\n {name: 'version', type: 'TAG', value: 'OTTO'},\n {name: 'numTables', type: 'USHORT', value: 0},\n {name: 'searchRange', type: 'USHORT', value: 0},\n {name: 'entrySelector', type: 'USHORT', value: 0},\n {name: 'rangeShift', type: 'USHORT', value: 0}\n ]);\n sfnt.tables = tables;\n sfnt.numTables = tables.length;\n var highestPowerOf2 = Math.pow(2, log2(sfnt.numTables));\n sfnt.searchRange = 16 * highestPowerOf2;\n sfnt.entrySelector = log2(highestPowerOf2);\n sfnt.rangeShift = sfnt.numTables * 16 - sfnt.searchRange;\n\n var recordFields = [];\n var tableFields = [];\n\n var offset = sfnt.sizeOf() + (makeTableRecord().sizeOf() * sfnt.numTables);\n while (offset % 4 !== 0) {\n offset += 1;\n tableFields.push({name: 'padding', type: 'BYTE', value: 0});\n }\n\n for (var i = 0; i < tables.length; i += 1) {\n var t = tables[i];\n check.argument(t.tableName.length === 4, 'Table name' + t.tableName + ' is invalid.');\n var tableLength = t.sizeOf();\n var tableRecord = makeTableRecord(t.tableName, computeCheckSum(t.encode()), offset, tableLength);\n recordFields.push({name: tableRecord.tag + ' Table Record', type: 'RECORD', value: tableRecord});\n tableFields.push({name: t.tableName + ' table', type: 'RECORD', value: t});\n offset += tableLength;\n check.argument(!isNaN(offset), 'Something went wrong calculating the offset.');\n while (offset % 4 !== 0) {\n offset += 1;\n tableFields.push({name: 'padding', type: 'BYTE', value: 0});\n }\n }\n\n // Table records need to be sorted alphabetically.\n recordFields.sort(function(r1, r2) {\n if (r1.value.tag > r2.value.tag) {\n return 1;\n } else {\n return -1;\n }\n });\n\n sfnt.fields = sfnt.fields.concat(recordFields);\n sfnt.fields = sfnt.fields.concat(tableFields);\n return sfnt;\n }\n\n // Get the metrics for a character. If the string has more than one character\n // this function returns metrics for the first available character.\n // You can provide optional fallback metrics if no characters are available.\n function metricsForChar(font, chars, notFoundMetrics) {\n for (var i = 0; i < chars.length; i += 1) {\n var glyphIndex = font.charToGlyphIndex(chars[i]);\n if (glyphIndex > 0) {\n var glyph = font.glyphs.get(glyphIndex);\n return glyph.getMetrics();\n }\n }\n\n return notFoundMetrics;\n }\n\n function average(vs) {\n var sum = 0;\n for (var i = 0; i < vs.length; i += 1) {\n sum += vs[i];\n }\n\n return sum / vs.length;\n }\n\n // Convert the font object to a SFNT data structure.\n // This structure contains all the necessary tables and metadata to create a binary OTF file.\n function fontToSfntTable(font) {\n var xMins = [];\n var yMins = [];\n var xMaxs = [];\n var yMaxs = [];\n var advanceWidths = [];\n var leftSideBearings = [];\n var rightSideBearings = [];\n var firstCharIndex;\n var lastCharIndex = 0;\n var ulUnicodeRange1 = 0;\n var ulUnicodeRange2 = 0;\n var ulUnicodeRange3 = 0;\n var ulUnicodeRange4 = 0;\n\n for (var i = 0; i < font.glyphs.length; i += 1) {\n var glyph = font.glyphs.get(i);\n var unicode = glyph.unicode | 0;\n\n if (isNaN(glyph.advanceWidth)) {\n throw new Error('Glyph ' + glyph.name + ' (' + i + '): advanceWidth is not a number.');\n }\n\n if (firstCharIndex > unicode || firstCharIndex === undefined) {\n // ignore .notdef char\n if (unicode > 0) {\n firstCharIndex = unicode;\n }\n }\n\n if (lastCharIndex < unicode) {\n lastCharIndex = unicode;\n }\n\n var position = os2.getUnicodeRange(unicode);\n if (position < 32) {\n ulUnicodeRange1 |= 1 << position;\n } else if (position < 64) {\n ulUnicodeRange2 |= 1 << position - 32;\n } else if (position < 96) {\n ulUnicodeRange3 |= 1 << position - 64;\n } else if (position < 123) {\n ulUnicodeRange4 |= 1 << position - 96;\n } else {\n throw new Error('Unicode ranges bits > 123 are reserved for internal usage');\n }\n // Skip non-important characters.\n if (glyph.name === '.notdef') { continue; }\n var metrics = glyph.getMetrics();\n xMins.push(metrics.xMin);\n yMins.push(metrics.yMin);\n xMaxs.push(metrics.xMax);\n yMaxs.push(metrics.yMax);\n leftSideBearings.push(metrics.leftSideBearing);\n rightSideBearings.push(metrics.rightSideBearing);\n advanceWidths.push(glyph.advanceWidth);\n }\n\n var globals = {\n xMin: Math.min.apply(null, xMins),\n yMin: Math.min.apply(null, yMins),\n xMax: Math.max.apply(null, xMaxs),\n yMax: Math.max.apply(null, yMaxs),\n advanceWidthMax: Math.max.apply(null, advanceWidths),\n advanceWidthAvg: average(advanceWidths),\n minLeftSideBearing: Math.min.apply(null, leftSideBearings),\n maxLeftSideBearing: Math.max.apply(null, leftSideBearings),\n minRightSideBearing: Math.min.apply(null, rightSideBearings)\n };\n globals.ascender = font.ascender;\n globals.descender = font.descender;\n\n var headTable = head.make({\n flags: 3, // 00000011 (baseline for font at y=0; left sidebearing point at x=0)\n unitsPerEm: font.unitsPerEm,\n xMin: globals.xMin,\n yMin: globals.yMin,\n xMax: globals.xMax,\n yMax: globals.yMax,\n lowestRecPPEM: 3,\n createdTimestamp: font.createdTimestamp\n });\n\n var hheaTable = hhea.make({\n ascender: globals.ascender,\n descender: globals.descender,\n advanceWidthMax: globals.advanceWidthMax,\n minLeftSideBearing: globals.minLeftSideBearing,\n minRightSideBearing: globals.minRightSideBearing,\n xMaxExtent: globals.maxLeftSideBearing + (globals.xMax - globals.xMin),\n numberOfHMetrics: font.glyphs.length\n });\n\n var maxpTable = maxp.make(font.glyphs.length);\n\n var os2Table = os2.make(Object.assign({\n xAvgCharWidth: Math.round(globals.advanceWidthAvg),\n usFirstCharIndex: firstCharIndex,\n usLastCharIndex: lastCharIndex,\n ulUnicodeRange1: ulUnicodeRange1,\n ulUnicodeRange2: ulUnicodeRange2,\n ulUnicodeRange3: ulUnicodeRange3,\n ulUnicodeRange4: ulUnicodeRange4,\n // See http://typophile.com/node/13081 for more info on vertical metrics.\n // We get metrics for typical characters (such as \"x\" for xHeight).\n // We provide some fallback characters if characters are unavailable: their\n // ordering was chosen experimentally.\n sTypoAscender: globals.ascender,\n sTypoDescender: globals.descender,\n sTypoLineGap: 0,\n usWinAscent: globals.yMax,\n usWinDescent: Math.abs(globals.yMin),\n ulCodePageRange1: 1, // FIXME: hard-code Latin 1 support for now\n sxHeight: metricsForChar(font, 'xyvw', {yMax: Math.round(globals.ascender / 2)}).yMax,\n sCapHeight: metricsForChar(font, 'HIKLEFJMNTZBDPRAGOQSUVWXY', globals).yMax,\n usDefaultChar: font.hasChar(' ') ? 32 : 0, // Use space as the default character, if available.\n usBreakChar: font.hasChar(' ') ? 32 : 0, // Use space as the break character, if available.\n }, font.tables.os2));\n\n var hmtxTable = hmtx.make(font.glyphs);\n var cmapTable = cmap.make(font.glyphs);\n\n var englishFamilyName = font.getEnglishName('fontFamily');\n var englishStyleName = font.getEnglishName('fontSubfamily');\n var englishFullName = englishFamilyName + ' ' + englishStyleName;\n var postScriptName = font.getEnglishName('postScriptName');\n if (!postScriptName) {\n postScriptName = englishFamilyName.replace(/\\s/g, '') + '-' + englishStyleName;\n }\n\n var names = {};\n for (var n in font.names) {\n names[n] = font.names[n];\n }\n\n if (!names.uniqueID) {\n names.uniqueID = {en: font.getEnglishName('manufacturer') + ':' + englishFullName};\n }\n\n if (!names.postScriptName) {\n names.postScriptName = {en: postScriptName};\n }\n\n if (!names.preferredFamily) {\n names.preferredFamily = font.names.fontFamily;\n }\n\n if (!names.preferredSubfamily) {\n names.preferredSubfamily = font.names.fontSubfamily;\n }\n\n var languageTags = [];\n var nameTable = _name.make(names, languageTags);\n var ltagTable = (languageTags.length > 0 ? ltag.make(languageTags) : undefined);\n\n var postTable = post.make();\n var cffTable = cff.make(font.glyphs, {\n version: font.getEnglishName('version'),\n fullName: englishFullName,\n familyName: englishFamilyName,\n weightName: englishStyleName,\n postScriptName: postScriptName,\n unitsPerEm: font.unitsPerEm,\n fontBBox: [0, globals.yMin, globals.ascender, globals.advanceWidthMax]\n });\n\n var metaTable = (font.metas && Object.keys(font.metas).length > 0) ? meta.make(font.metas) : undefined;\n\n // The order does not matter because makeSfntTable() will sort them.\n var tables = [headTable, hheaTable, maxpTable, os2Table, nameTable, cmapTable, postTable, cffTable, hmtxTable];\n if (ltagTable) {\n tables.push(ltagTable);\n }\n // Optional tables\n if (font.tables.gsub) {\n tables.push(gsub.make(font.tables.gsub));\n }\n if (metaTable) {\n tables.push(metaTable);\n }\n\n var sfntTable = makeSfntTable(tables);\n\n // Compute the font's checkSum and store it in head.checkSumAdjustment.\n var bytes = sfntTable.encode();\n var checkSum = computeCheckSum(bytes);\n var tableFields = sfntTable.fields;\n var checkSumAdjusted = false;\n for (var i$1 = 0; i$1 < tableFields.length; i$1 += 1) {\n if (tableFields[i$1].name === 'head table') {\n tableFields[i$1].value.checkSumAdjustment = 0xB1B0AFBA - checkSum;\n checkSumAdjusted = true;\n break;\n }\n }\n\n if (!checkSumAdjusted) {\n throw new Error('Could not find head table with checkSum to adjust.');\n }\n\n return sfntTable;\n }\n\n var sfnt = { make: makeSfntTable, fontToTable: fontToSfntTable, computeCheckSum: computeCheckSum };\n\n // The Layout object is the prototype of Substitution objects, and provides\n\n function searchTag(arr, tag) {\n /* jshint bitwise: false */\n var imin = 0;\n var imax = arr.length - 1;\n while (imin <= imax) {\n var imid = (imin + imax) >>> 1;\n var val = arr[imid].tag;\n if (val === tag) {\n return imid;\n } else if (val < tag) {\n imin = imid + 1;\n } else { imax = imid - 1; }\n }\n // Not found: return -1-insertion point\n return -imin - 1;\n }\n\n function binSearch(arr, value) {\n /* jshint bitwise: false */\n var imin = 0;\n var imax = arr.length - 1;\n while (imin <= imax) {\n var imid = (imin + imax) >>> 1;\n var val = arr[imid];\n if (val === value) {\n return imid;\n } else if (val < value) {\n imin = imid + 1;\n } else { imax = imid - 1; }\n }\n // Not found: return -1-insertion point\n return -imin - 1;\n }\n\n // binary search in a list of ranges (coverage, class definition)\n function searchRange(ranges, value) {\n // jshint bitwise: false\n var range;\n var imin = 0;\n var imax = ranges.length - 1;\n while (imin <= imax) {\n var imid = (imin + imax) >>> 1;\n range = ranges[imid];\n var start = range.start;\n if (start === value) {\n return range;\n } else if (start < value) {\n imin = imid + 1;\n } else { imax = imid - 1; }\n }\n if (imin > 0) {\n range = ranges[imin - 1];\n if (value > range.end) { return 0; }\n return range;\n }\n }\n\n /**\n * @exports opentype.Layout\n * @class\n */\n function Layout(font, tableName) {\n this.font = font;\n this.tableName = tableName;\n }\n\n Layout.prototype = {\n\n /**\n * Binary search an object by \"tag\" property\n * @instance\n * @function searchTag\n * @memberof opentype.Layout\n * @param {Array} arr\n * @param {string} tag\n * @return {number}\n */\n searchTag: searchTag,\n\n /**\n * Binary search in a list of numbers\n * @instance\n * @function binSearch\n * @memberof opentype.Layout\n * @param {Array} arr\n * @param {number} value\n * @return {number}\n */\n binSearch: binSearch,\n\n /**\n * Get or create the Layout table (GSUB, GPOS etc).\n * @param {boolean} create - Whether to create a new one.\n * @return {Object} The GSUB or GPOS table.\n */\n getTable: function(create) {\n var layout = this.font.tables[this.tableName];\n if (!layout && create) {\n layout = this.font.tables[this.tableName] = this.createDefaultTable();\n }\n return layout;\n },\n\n /**\n * Returns all scripts in the substitution table.\n * @instance\n * @return {Array}\n */\n getScriptNames: function() {\n var layout = this.getTable();\n if (!layout) { return []; }\n return layout.scripts.map(function(script) {\n return script.tag;\n });\n },\n\n /**\n * Returns the best bet for a script name.\n * Returns 'DFLT' if it exists.\n * If not, returns 'latn' if it exists.\n * If neither exist, returns undefined.\n */\n getDefaultScriptName: function() {\n var layout = this.getTable();\n if (!layout) { return; }\n var hasLatn = false;\n for (var i = 0; i < layout.scripts.length; i++) {\n var name = layout.scripts[i].tag;\n if (name === 'DFLT') { return name; }\n if (name === 'latn') { hasLatn = true; }\n }\n if (hasLatn) { return 'latn'; }\n },\n\n /**\n * Returns all LangSysRecords in the given script.\n * @instance\n * @param {string} [script='DFLT']\n * @param {boolean} create - forces the creation of this script table if it doesn't exist.\n * @return {Object} An object with tag and script properties.\n */\n getScriptTable: function(script, create) {\n var layout = this.getTable(create);\n if (layout) {\n script = script || 'DFLT';\n var scripts = layout.scripts;\n var pos = searchTag(layout.scripts, script);\n if (pos >= 0) {\n return scripts[pos].script;\n } else if (create) {\n var scr = {\n tag: script,\n script: {\n defaultLangSys: {reserved: 0, reqFeatureIndex: 0xffff, featureIndexes: []},\n langSysRecords: []\n }\n };\n scripts.splice(-1 - pos, 0, scr);\n return scr.script;\n }\n }\n },\n\n /**\n * Returns a language system table\n * @instance\n * @param {string} [script='DFLT']\n * @param {string} [language='dlft']\n * @param {boolean} create - forces the creation of this langSysTable if it doesn't exist.\n * @return {Object}\n */\n getLangSysTable: function(script, language, create) {\n var scriptTable = this.getScriptTable(script, create);\n if (scriptTable) {\n if (!language || language === 'dflt' || language === 'DFLT') {\n return scriptTable.defaultLangSys;\n }\n var pos = searchTag(scriptTable.langSysRecords, language);\n if (pos >= 0) {\n return scriptTable.langSysRecords[pos].langSys;\n } else if (create) {\n var langSysRecord = {\n tag: language,\n langSys: {reserved: 0, reqFeatureIndex: 0xffff, featureIndexes: []}\n };\n scriptTable.langSysRecords.splice(-1 - pos, 0, langSysRecord);\n return langSysRecord.langSys;\n }\n }\n },\n\n /**\n * Get a specific feature table.\n * @instance\n * @param {string} [script='DFLT']\n * @param {string} [language='dlft']\n * @param {string} feature - One of the codes listed at https://www.microsoft.com/typography/OTSPEC/featurelist.htm\n * @param {boolean} create - forces the creation of the feature table if it doesn't exist.\n * @return {Object}\n */\n getFeatureTable: function(script, language, feature, create) {\n var langSysTable = this.getLangSysTable(script, language, create);\n if (langSysTable) {\n var featureRecord;\n var featIndexes = langSysTable.featureIndexes;\n var allFeatures = this.font.tables[this.tableName].features;\n // The FeatureIndex array of indices is in arbitrary order,\n // even if allFeatures is sorted alphabetically by feature tag.\n for (var i = 0; i < featIndexes.length; i++) {\n featureRecord = allFeatures[featIndexes[i]];\n if (featureRecord.tag === feature) {\n return featureRecord.feature;\n }\n }\n if (create) {\n var index = allFeatures.length;\n // Automatic ordering of features would require to shift feature indexes in the script list.\n check.assert(index === 0 || feature >= allFeatures[index - 1].tag, 'Features must be added in alphabetical order.');\n featureRecord = {\n tag: feature,\n feature: { params: 0, lookupListIndexes: [] }\n };\n allFeatures.push(featureRecord);\n featIndexes.push(index);\n return featureRecord.feature;\n }\n }\n },\n\n /**\n * Get the lookup tables of a given type for a script/language/feature.\n * @instance\n * @param {string} [script='DFLT']\n * @param {string} [language='dlft']\n * @param {string} feature - 4-letter feature code\n * @param {number} lookupType - 1 to 9\n * @param {boolean} create - forces the creation of the lookup table if it doesn't exist, with no subtables.\n * @return {Object[]}\n */\n getLookupTables: function(script, language, feature, lookupType, create) {\n var featureTable = this.getFeatureTable(script, language, feature, create);\n var tables = [];\n if (featureTable) {\n var lookupTable;\n var lookupListIndexes = featureTable.lookupListIndexes;\n var allLookups = this.font.tables[this.tableName].lookups;\n // lookupListIndexes are in no particular order, so use naive search.\n for (var i = 0; i < lookupListIndexes.length; i++) {\n lookupTable = allLookups[lookupListIndexes[i]];\n if (lookupTable.lookupType === lookupType) {\n tables.push(lookupTable);\n }\n }\n if (tables.length === 0 && create) {\n lookupTable = {\n lookupType: lookupType,\n lookupFlag: 0,\n subtables: [],\n markFilteringSet: undefined\n };\n var index = allLookups.length;\n allLookups.push(lookupTable);\n lookupListIndexes.push(index);\n return [lookupTable];\n }\n }\n return tables;\n },\n\n /**\n * Find a glyph in a class definition table\n * https://docs.microsoft.com/en-us/typography/opentype/spec/chapter2#class-definition-table\n * @param {object} classDefTable - an OpenType Layout class definition table\n * @param {number} glyphIndex - the index of the glyph to find\n * @returns {number} -1 if not found\n */\n getGlyphClass: function(classDefTable, glyphIndex) {\n switch (classDefTable.format) {\n case 1:\n if (classDefTable.startGlyph <= glyphIndex && glyphIndex < classDefTable.startGlyph + classDefTable.classes.length) {\n return classDefTable.classes[glyphIndex - classDefTable.startGlyph];\n }\n return 0;\n case 2:\n var range = searchRange(classDefTable.ranges, glyphIndex);\n return range ? range.classId : 0;\n }\n },\n\n /**\n * Find a glyph in a coverage table\n * https://docs.microsoft.com/en-us/typography/opentype/spec/chapter2#coverage-table\n * @param {object} coverageTable - an OpenType Layout coverage table\n * @param {number} glyphIndex - the index of the glyph to find\n * @returns {number} -1 if not found\n */\n getCoverageIndex: function(coverageTable, glyphIndex) {\n switch (coverageTable.format) {\n case 1:\n var index = binSearch(coverageTable.glyphs, glyphIndex);\n return index >= 0 ? index : -1;\n case 2:\n var range = searchRange(coverageTable.ranges, glyphIndex);\n return range ? range.index + glyphIndex - range.start : -1;\n }\n },\n\n /**\n * Returns the list of glyph indexes of a coverage table.\n * Format 1: the list is stored raw\n * Format 2: compact list as range records.\n * @instance\n * @param {Object} coverageTable\n * @return {Array}\n */\n expandCoverage: function(coverageTable) {\n if (coverageTable.format === 1) {\n return coverageTable.glyphs;\n } else {\n var glyphs = [];\n var ranges = coverageTable.ranges;\n for (var i = 0; i < ranges.length; i++) {\n var range = ranges[i];\n var start = range.start;\n var end = range.end;\n for (var j = start; j <= end; j++) {\n glyphs.push(j);\n }\n }\n return glyphs;\n }\n }\n\n };\n\n // The Position object provides utility methods to manipulate\n\n /**\n * @exports opentype.Position\n * @class\n * @extends opentype.Layout\n * @param {opentype.Font}\n * @constructor\n */\n function Position(font) {\n Layout.call(this, font, 'gpos');\n }\n\n Position.prototype = Layout.prototype;\n\n /**\n * Init some data for faster and easier access later.\n */\n Position.prototype.init = function() {\n var script = this.getDefaultScriptName();\n this.defaultKerningTables = this.getKerningTables(script);\n };\n\n /**\n * Find a glyph pair in a list of lookup tables of type 2 and retrieve the xAdvance kerning value.\n *\n * @param {integer} leftIndex - left glyph index\n * @param {integer} rightIndex - right glyph index\n * @returns {integer}\n */\n Position.prototype.getKerningValue = function(kerningLookups, leftIndex, rightIndex) {\n for (var i = 0; i < kerningLookups.length; i++) {\n var subtables = kerningLookups[i].subtables;\n for (var j = 0; j < subtables.length; j++) {\n var subtable = subtables[j];\n var covIndex = this.getCoverageIndex(subtable.coverage, leftIndex);\n if (covIndex < 0) { continue; }\n switch (subtable.posFormat) {\n case 1:\n // Search Pair Adjustment Positioning Format 1\n var pairSet = subtable.pairSets[covIndex];\n for (var k = 0; k < pairSet.length; k++) {\n var pair = pairSet[k];\n if (pair.secondGlyph === rightIndex) {\n return pair.value1 && pair.value1.xAdvance || 0;\n }\n }\n break; // left glyph found, not right glyph - try next subtable\n case 2:\n // Search Pair Adjustment Positioning Format 2\n var class1 = this.getGlyphClass(subtable.classDef1, leftIndex);\n var class2 = this.getGlyphClass(subtable.classDef2, rightIndex);\n var pair$1 = subtable.classRecords[class1][class2];\n return pair$1.value1 && pair$1.value1.xAdvance || 0;\n }\n }\n }\n return 0;\n };\n\n /**\n * List all kerning lookup tables.\n *\n * @param {string} [script='DFLT'] - use font.position.getDefaultScriptName() for a better default value\n * @param {string} [language='dflt']\n * @return {object[]} The list of kerning lookup tables (may be empty), or undefined if there is no GPOS table (and we should use the kern table)\n */\n Position.prototype.getKerningTables = function(script, language) {\n if (this.font.tables.gpos) {\n return this.getLookupTables(script, language, 'kern', 2);\n }\n };\n\n // The Substitution object provides utility methods to manipulate\n\n /**\n * @exports opentype.Substitution\n * @class\n * @extends opentype.Layout\n * @param {opentype.Font}\n * @constructor\n */\n function Substitution(font) {\n Layout.call(this, font, 'gsub');\n }\n\n // Check if 2 arrays of primitives are equal.\n function arraysEqual(ar1, ar2) {\n var n = ar1.length;\n if (n !== ar2.length) { return false; }\n for (var i = 0; i < n; i++) {\n if (ar1[i] !== ar2[i]) { return false; }\n }\n return true;\n }\n\n // Find the first subtable of a lookup table in a particular format.\n function getSubstFormat(lookupTable, format, defaultSubtable) {\n var subtables = lookupTable.subtables;\n for (var i = 0; i < subtables.length; i++) {\n var subtable = subtables[i];\n if (subtable.substFormat === format) {\n return subtable;\n }\n }\n if (defaultSubtable) {\n subtables.push(defaultSubtable);\n return defaultSubtable;\n }\n return undefined;\n }\n\n Substitution.prototype = Layout.prototype;\n\n /**\n * Create a default GSUB table.\n * @return {Object} gsub - The GSUB table.\n */\n Substitution.prototype.createDefaultTable = function() {\n // Generate a default empty GSUB table with just a DFLT script and dflt lang sys.\n return {\n version: 1,\n scripts: [{\n tag: 'DFLT',\n script: {\n defaultLangSys: { reserved: 0, reqFeatureIndex: 0xffff, featureIndexes: [] },\n langSysRecords: []\n }\n }],\n features: [],\n lookups: []\n };\n };\n\n /**\n * List all single substitutions (lookup type 1) for a given script, language, and feature.\n * @param {string} [script='DFLT']\n * @param {string} [language='dflt']\n * @param {string} feature - 4-character feature name ('aalt', 'salt', 'ss01'...)\n * @return {Array} substitutions - The list of substitutions.\n */\n Substitution.prototype.getSingle = function(feature, script, language) {\n var substitutions = [];\n var lookupTables = this.getLookupTables(script, language, feature, 1);\n for (var idx = 0; idx < lookupTables.length; idx++) {\n var subtables = lookupTables[idx].subtables;\n for (var i = 0; i < subtables.length; i++) {\n var subtable = subtables[i];\n var glyphs = this.expandCoverage(subtable.coverage);\n var j = (void 0);\n if (subtable.substFormat === 1) {\n var delta = subtable.deltaGlyphId;\n for (j = 0; j < glyphs.length; j++) {\n var glyph = glyphs[j];\n substitutions.push({ sub: glyph, by: glyph + delta });\n }\n } else {\n var substitute = subtable.substitute;\n for (j = 0; j < glyphs.length; j++) {\n substitutions.push({ sub: glyphs[j], by: substitute[j] });\n }\n }\n }\n }\n return substitutions;\n };\n\n /**\n * List all multiple substitutions (lookup type 2) for a given script, language, and feature.\n * @param {string} [script='DFLT']\n * @param {string} [language='dflt']\n * @param {string} feature - 4-character feature name ('ccmp', 'stch')\n * @return {Array} substitutions - The list of substitutions.\n */\n Substitution.prototype.getMultiple = function(feature, script, language) {\n var substitutions = [];\n var lookupTables = this.getLookupTables(script, language, feature, 2);\n for (var idx = 0; idx < lookupTables.length; idx++) {\n var subtables = lookupTables[idx].subtables;\n for (var i = 0; i < subtables.length; i++) {\n var subtable = subtables[i];\n var glyphs = this.expandCoverage(subtable.coverage);\n var j = (void 0);\n\n for (j = 0; j < glyphs.length; j++) {\n var glyph = glyphs[j];\n var replacements = subtable.sequences[j];\n substitutions.push({ sub: glyph, by: replacements });\n }\n }\n }\n return substitutions;\n };\n\n /**\n * List all alternates (lookup type 3) for a given script, language, and feature.\n * @param {string} [script='DFLT']\n * @param {string} [language='dflt']\n * @param {string} feature - 4-character feature name ('aalt', 'salt'...)\n * @return {Array} alternates - The list of alternates\n */\n Substitution.prototype.getAlternates = function(feature, script, language) {\n var alternates = [];\n var lookupTables = this.getLookupTables(script, language, feature, 3);\n for (var idx = 0; idx < lookupTables.length; idx++) {\n var subtables = lookupTables[idx].subtables;\n for (var i = 0; i < subtables.length; i++) {\n var subtable = subtables[i];\n var glyphs = this.expandCoverage(subtable.coverage);\n var alternateSets = subtable.alternateSets;\n for (var j = 0; j < glyphs.length; j++) {\n alternates.push({ sub: glyphs[j], by: alternateSets[j] });\n }\n }\n }\n return alternates;\n };\n\n /**\n * List all ligatures (lookup type 4) for a given script, language, and feature.\n * The result is an array of ligature objects like { sub: [ids], by: id }\n * @param {string} feature - 4-letter feature name ('liga', 'rlig', 'dlig'...)\n * @param {string} [script='DFLT']\n * @param {string} [language='dflt']\n * @return {Array} ligatures - The list of ligatures.\n */\n Substitution.prototype.getLigatures = function(feature, script, language) {\n var ligatures = [];\n var lookupTables = this.getLookupTables(script, language, feature, 4);\n for (var idx = 0; idx < lookupTables.length; idx++) {\n var subtables = lookupTables[idx].subtables;\n for (var i = 0; i < subtables.length; i++) {\n var subtable = subtables[i];\n var glyphs = this.expandCoverage(subtable.coverage);\n var ligatureSets = subtable.ligatureSets;\n for (var j = 0; j < glyphs.length; j++) {\n var startGlyph = glyphs[j];\n var ligSet = ligatureSets[j];\n for (var k = 0; k < ligSet.length; k++) {\n var lig = ligSet[k];\n ligatures.push({\n sub: [startGlyph].concat(lig.components),\n by: lig.ligGlyph\n });\n }\n }\n }\n }\n return ligatures;\n };\n\n /**\n * Add or modify a single substitution (lookup type 1)\n * Format 2, more flexible, is always used.\n * @param {string} feature - 4-letter feature name ('liga', 'rlig', 'dlig'...)\n * @param {Object} substitution - { sub: id, by: id } (format 1 is not supported)\n * @param {string} [script='DFLT']\n * @param {string} [language='dflt']\n */\n Substitution.prototype.addSingle = function(feature, substitution, script, language) {\n var lookupTable = this.getLookupTables(script, language, feature, 1, true)[0];\n var subtable = getSubstFormat(lookupTable, 2, { // lookup type 1 subtable, format 2, coverage format 1\n substFormat: 2,\n coverage: {format: 1, glyphs: []},\n substitute: []\n });\n check.assert(subtable.coverage.format === 1, 'Single: unable to modify coverage table format ' + subtable.coverage.format);\n var coverageGlyph = substitution.sub;\n var pos = this.binSearch(subtable.coverage.glyphs, coverageGlyph);\n if (pos < 0) {\n pos = -1 - pos;\n subtable.coverage.glyphs.splice(pos, 0, coverageGlyph);\n subtable.substitute.splice(pos, 0, 0);\n }\n subtable.substitute[pos] = substitution.by;\n };\n\n /**\n * Add or modify a multiple substitution (lookup type 2)\n * @param {string} feature - 4-letter feature name ('ccmp', 'stch')\n * @param {Object} substitution - { sub: id, by: [id] } for format 2.\n * @param {string} [script='DFLT']\n * @param {string} [language='dflt']\n */\n Substitution.prototype.addMultiple = function(feature, substitution, script, language) {\n check.assert(substitution.by instanceof Array && substitution.by.length > 1, 'Multiple: \"by\" must be an array of two or more ids');\n var lookupTable = this.getLookupTables(script, language, feature, 2, true)[0];\n var subtable = getSubstFormat(lookupTable, 1, { // lookup type 2 subtable, format 1, coverage format 1\n substFormat: 1,\n coverage: {format: 1, glyphs: []},\n sequences: []\n });\n check.assert(subtable.coverage.format === 1, 'Multiple: unable to modify coverage table format ' + subtable.coverage.format);\n var coverageGlyph = substitution.sub;\n var pos = this.binSearch(subtable.coverage.glyphs, coverageGlyph);\n if (pos < 0) {\n pos = -1 - pos;\n subtable.coverage.glyphs.splice(pos, 0, coverageGlyph);\n subtable.sequences.splice(pos, 0, 0);\n }\n subtable.sequences[pos] = substitution.by;\n };\n\n /**\n * Add or modify an alternate substitution (lookup type 3)\n * @param {string} feature - 4-letter feature name ('liga', 'rlig', 'dlig'...)\n * @param {Object} substitution - { sub: id, by: [ids] }\n * @param {string} [script='DFLT']\n * @param {string} [language='dflt']\n */\n Substitution.prototype.addAlternate = function(feature, substitution, script, language) {\n var lookupTable = this.getLookupTables(script, language, feature, 3, true)[0];\n var subtable = getSubstFormat(lookupTable, 1, { // lookup type 3 subtable, format 1, coverage format 1\n substFormat: 1,\n coverage: {format: 1, glyphs: []},\n alternateSets: []\n });\n check.assert(subtable.coverage.format === 1, 'Alternate: unable to modify coverage table format ' + subtable.coverage.format);\n var coverageGlyph = substitution.sub;\n var pos = this.binSearch(subtable.coverage.glyphs, coverageGlyph);\n if (pos < 0) {\n pos = -1 - pos;\n subtable.coverage.glyphs.splice(pos, 0, coverageGlyph);\n subtable.alternateSets.splice(pos, 0, 0);\n }\n subtable.alternateSets[pos] = substitution.by;\n };\n\n /**\n * Add a ligature (lookup type 4)\n * Ligatures with more components must be stored ahead of those with fewer components in order to be found\n * @param {string} feature - 4-letter feature name ('liga', 'rlig', 'dlig'...)\n * @param {Object} ligature - { sub: [ids], by: id }\n * @param {string} [script='DFLT']\n * @param {string} [language='dflt']\n */\n Substitution.prototype.addLigature = function(feature, ligature, script, language) {\n var lookupTable = this.getLookupTables(script, language, feature, 4, true)[0];\n var subtable = lookupTable.subtables[0];\n if (!subtable) {\n subtable = { // lookup type 4 subtable, format 1, coverage format 1\n substFormat: 1,\n coverage: { format: 1, glyphs: [] },\n ligatureSets: []\n };\n lookupTable.subtables[0] = subtable;\n }\n check.assert(subtable.coverage.format === 1, 'Ligature: unable to modify coverage table format ' + subtable.coverage.format);\n var coverageGlyph = ligature.sub[0];\n var ligComponents = ligature.sub.slice(1);\n var ligatureTable = {\n ligGlyph: ligature.by,\n components: ligComponents\n };\n var pos = this.binSearch(subtable.coverage.glyphs, coverageGlyph);\n if (pos >= 0) {\n // ligatureSet already exists\n var ligatureSet = subtable.ligatureSets[pos];\n for (var i = 0; i < ligatureSet.length; i++) {\n // If ligature already exists, return.\n if (arraysEqual(ligatureSet[i].components, ligComponents)) {\n return;\n }\n }\n // ligature does not exist: add it.\n ligatureSet.push(ligatureTable);\n } else {\n // Create a new ligatureSet and add coverage for the first glyph.\n pos = -1 - pos;\n subtable.coverage.glyphs.splice(pos, 0, coverageGlyph);\n subtable.ligatureSets.splice(pos, 0, [ligatureTable]);\n }\n };\n\n /**\n * List all feature data for a given script and language.\n * @param {string} feature - 4-letter feature name\n * @param {string} [script='DFLT']\n * @param {string} [language='dflt']\n * @return {Array} substitutions - The list of substitutions.\n */\n Substitution.prototype.getFeature = function(feature, script, language) {\n if (/ss\\d\\d/.test(feature)) {\n // ss01 - ss20\n return this.getSingle(feature, script, language);\n }\n switch (feature) {\n case 'aalt':\n case 'salt':\n return this.getSingle(feature, script, language)\n .concat(this.getAlternates(feature, script, language));\n case 'dlig':\n case 'liga':\n case 'rlig':\n return this.getLigatures(feature, script, language);\n case 'ccmp':\n return this.getMultiple(feature, script, language)\n .concat(this.getLigatures(feature, script, language));\n case 'stch':\n return this.getMultiple(feature, script, language);\n }\n return undefined;\n };\n\n /**\n * Add a substitution to a feature for a given script and language.\n * @param {string} feature - 4-letter feature name\n * @param {Object} sub - the substitution to add (an object like { sub: id or [ids], by: id or [ids] })\n * @param {string} [script='DFLT']\n * @param {string} [language='dflt']\n */\n Substitution.prototype.add = function(feature, sub, script, language) {\n if (/ss\\d\\d/.test(feature)) {\n // ss01 - ss20\n return this.addSingle(feature, sub, script, language);\n }\n switch (feature) {\n case 'aalt':\n case 'salt':\n if (typeof sub.by === 'number') {\n return this.addSingle(feature, sub, script, language);\n }\n return this.addAlternate(feature, sub, script, language);\n case 'dlig':\n case 'liga':\n case 'rlig':\n return this.addLigature(feature, sub, script, language);\n case 'ccmp':\n if (sub.by instanceof Array) {\n return this.addMultiple(feature, sub, script, language);\n }\n return this.addLigature(feature, sub, script, language);\n }\n return undefined;\n };\n\n function isBrowser() {\n return typeof window !== 'undefined';\n }\n\n function arrayBufferToNodeBuffer(ab) {\n var buffer = new Buffer(ab.byteLength);\n var view = new Uint8Array(ab);\n for (var i = 0; i < buffer.length; ++i) {\n buffer[i] = view[i];\n }\n\n return buffer;\n }\n\n function checkArgument(expression, message) {\n if (!expression) {\n throw message;\n }\n }\n\n // The `glyf` table describes the glyphs in TrueType outline format.\n\n // Parse the coordinate data for a glyph.\n function parseGlyphCoordinate(p, flag, previousValue, shortVectorBitMask, sameBitMask) {\n var v;\n if ((flag & shortVectorBitMask) > 0) {\n // The coordinate is 1 byte long.\n v = p.parseByte();\n // The `same` bit is re-used for short values to signify the sign of the value.\n if ((flag & sameBitMask) === 0) {\n v = -v;\n }\n\n v = previousValue + v;\n } else {\n // The coordinate is 2 bytes long.\n // If the `same` bit is set, the coordinate is the same as the previous coordinate.\n if ((flag & sameBitMask) > 0) {\n v = previousValue;\n } else {\n // Parse the coordinate as a signed 16-bit delta value.\n v = previousValue + p.parseShort();\n }\n }\n\n return v;\n }\n\n // Parse a TrueType glyph.\n function parseGlyph(glyph, data, start) {\n var p = new parse.Parser(data, start);\n glyph.numberOfContours = p.parseShort();\n glyph._xMin = p.parseShort();\n glyph._yMin = p.parseShort();\n glyph._xMax = p.parseShort();\n glyph._yMax = p.parseShort();\n var flags;\n var flag;\n\n if (glyph.numberOfContours > 0) {\n // This glyph is not a composite.\n var endPointIndices = glyph.endPointIndices = [];\n for (var i = 0; i < glyph.numberOfContours; i += 1) {\n endPointIndices.push(p.parseUShort());\n }\n\n glyph.instructionLength = p.parseUShort();\n glyph.instructions = [];\n for (var i$1 = 0; i$1 < glyph.instructionLength; i$1 += 1) {\n glyph.instructions.push(p.parseByte());\n }\n\n var numberOfCoordinates = endPointIndices[endPointIndices.length - 1] + 1;\n flags = [];\n for (var i$2 = 0; i$2 < numberOfCoordinates; i$2 += 1) {\n flag = p.parseByte();\n flags.push(flag);\n // If bit 3 is set, we repeat this flag n times, where n is the next byte.\n if ((flag & 8) > 0) {\n var repeatCount = p.parseByte();\n for (var j = 0; j < repeatCount; j += 1) {\n flags.push(flag);\n i$2 += 1;\n }\n }\n }\n\n check.argument(flags.length === numberOfCoordinates, 'Bad flags.');\n\n if (endPointIndices.length > 0) {\n var points = [];\n var point;\n // X/Y coordinates are relative to the previous point, except for the first point which is relative to 0,0.\n if (numberOfCoordinates > 0) {\n for (var i$3 = 0; i$3 < numberOfCoordinates; i$3 += 1) {\n flag = flags[i$3];\n point = {};\n point.onCurve = !!(flag & 1);\n point.lastPointOfContour = endPointIndices.indexOf(i$3) >= 0;\n points.push(point);\n }\n\n var px = 0;\n for (var i$4 = 0; i$4 < numberOfCoordinates; i$4 += 1) {\n flag = flags[i$4];\n point = points[i$4];\n point.x = parseGlyphCoordinate(p, flag, px, 2, 16);\n px = point.x;\n }\n\n var py = 0;\n for (var i$5 = 0; i$5 < numberOfCoordinates; i$5 += 1) {\n flag = flags[i$5];\n point = points[i$5];\n point.y = parseGlyphCoordinate(p, flag, py, 4, 32);\n py = point.y;\n }\n }\n\n glyph.points = points;\n } else {\n glyph.points = [];\n }\n } else if (glyph.numberOfContours === 0) {\n glyph.points = [];\n } else {\n glyph.isComposite = true;\n glyph.points = [];\n glyph.components = [];\n var moreComponents = true;\n while (moreComponents) {\n flags = p.parseUShort();\n var component = {\n glyphIndex: p.parseUShort(),\n xScale: 1,\n scale01: 0,\n scale10: 0,\n yScale: 1,\n dx: 0,\n dy: 0\n };\n if ((flags & 1) > 0) {\n // The arguments are words\n if ((flags & 2) > 0) {\n // values are offset\n component.dx = p.parseShort();\n component.dy = p.parseShort();\n } else {\n // values are matched points\n component.matchedPoints = [p.parseUShort(), p.parseUShort()];\n }\n\n } else {\n // The arguments are bytes\n if ((flags & 2) > 0) {\n // values are offset\n component.dx = p.parseChar();\n component.dy = p.parseChar();\n } else {\n // values are matched points\n component.matchedPoints = [p.parseByte(), p.parseByte()];\n }\n }\n\n if ((flags & 8) > 0) {\n // We have a scale\n component.xScale = component.yScale = p.parseF2Dot14();\n } else if ((flags & 64) > 0) {\n // We have an X / Y scale\n component.xScale = p.parseF2Dot14();\n component.yScale = p.parseF2Dot14();\n } else if ((flags & 128) > 0) {\n // We have a 2x2 transformation\n component.xScale = p.parseF2Dot14();\n component.scale01 = p.parseF2Dot14();\n component.scale10 = p.parseF2Dot14();\n component.yScale = p.parseF2Dot14();\n }\n\n glyph.components.push(component);\n moreComponents = !!(flags & 32);\n }\n if (flags & 0x100) {\n // We have instructions\n glyph.instructionLength = p.parseUShort();\n glyph.instructions = [];\n for (var i$6 = 0; i$6 < glyph.instructionLength; i$6 += 1) {\n glyph.instructions.push(p.parseByte());\n }\n }\n }\n }\n\n // Transform an array of points and return a new array.\n function transformPoints(points, transform) {\n var newPoints = [];\n for (var i = 0; i < points.length; i += 1) {\n var pt = points[i];\n var newPt = {\n x: transform.xScale * pt.x + transform.scale01 * pt.y + transform.dx,\n y: transform.scale10 * pt.x + transform.yScale * pt.y + transform.dy,\n onCurve: pt.onCurve,\n lastPointOfContour: pt.lastPointOfContour\n };\n newPoints.push(newPt);\n }\n\n return newPoints;\n }\n\n function getContours(points) {\n var contours = [];\n var currentContour = [];\n for (var i = 0; i < points.length; i += 1) {\n var pt = points[i];\n currentContour.push(pt);\n if (pt.lastPointOfContour) {\n contours.push(currentContour);\n currentContour = [];\n }\n }\n\n check.argument(currentContour.length === 0, 'There are still points left in the current contour.');\n return contours;\n }\n\n // Convert the TrueType glyph outline to a Path.\n function getPath(points) {\n var p = new Path();\n if (!points) {\n return p;\n }\n\n var contours = getContours(points);\n\n for (var contourIndex = 0; contourIndex < contours.length; ++contourIndex) {\n var contour = contours[contourIndex];\n\n var prev = null;\n var curr = contour[contour.length - 1];\n var next = contour[0];\n\n if (curr.onCurve) {\n p.moveTo(curr.x, curr.y);\n } else {\n if (next.onCurve) {\n p.moveTo(next.x, next.y);\n } else {\n // If both first and last points are off-curve, start at their middle.\n var start = {x: (curr.x + next.x) * 0.5, y: (curr.y + next.y) * 0.5};\n p.moveTo(start.x, start.y);\n }\n }\n\n for (var i = 0; i < contour.length; ++i) {\n prev = curr;\n curr = next;\n next = contour[(i + 1) % contour.length];\n\n if (curr.onCurve) {\n // This is a straight line.\n p.lineTo(curr.x, curr.y);\n } else {\n var prev2 = prev;\n var next2 = next;\n\n if (!prev.onCurve) {\n prev2 = { x: (curr.x + prev.x) * 0.5, y: (curr.y + prev.y) * 0.5 };\n }\n\n if (!next.onCurve) {\n next2 = { x: (curr.x + next.x) * 0.5, y: (curr.y + next.y) * 0.5 };\n }\n\n p.quadraticCurveTo(curr.x, curr.y, next2.x, next2.y);\n }\n }\n\n p.closePath();\n }\n return p;\n }\n\n function buildPath(glyphs, glyph) {\n if (glyph.isComposite) {\n for (var j = 0; j < glyph.components.length; j += 1) {\n var component = glyph.components[j];\n var componentGlyph = glyphs.get(component.glyphIndex);\n // Force the ttfGlyphLoader to parse the glyph.\n componentGlyph.getPath();\n if (componentGlyph.points) {\n var transformedPoints = (void 0);\n if (component.matchedPoints === undefined) {\n // component positioned by offset\n transformedPoints = transformPoints(componentGlyph.points, component);\n } else {\n // component positioned by matched points\n if ((component.matchedPoints[0] > glyph.points.length - 1) ||\n (component.matchedPoints[1] > componentGlyph.points.length - 1)) {\n throw Error('Matched points out of range in ' + glyph.name);\n }\n var firstPt = glyph.points[component.matchedPoints[0]];\n var secondPt = componentGlyph.points[component.matchedPoints[1]];\n var transform = {\n xScale: component.xScale, scale01: component.scale01,\n scale10: component.scale10, yScale: component.yScale,\n dx: 0, dy: 0\n };\n secondPt = transformPoints([secondPt], transform)[0];\n transform.dx = firstPt.x - secondPt.x;\n transform.dy = firstPt.y - secondPt.y;\n transformedPoints = transformPoints(componentGlyph.points, transform);\n }\n glyph.points = glyph.points.concat(transformedPoints);\n }\n }\n }\n\n return getPath(glyph.points);\n }\n\n function parseGlyfTableAll(data, start, loca, font) {\n var glyphs = new glyphset.GlyphSet(font);\n\n // The last element of the loca table is invalid.\n for (var i = 0; i < loca.length - 1; i += 1) {\n var offset = loca[i];\n var nextOffset = loca[i + 1];\n if (offset !== nextOffset) {\n glyphs.push(i, glyphset.ttfGlyphLoader(font, i, parseGlyph, data, start + offset, buildPath));\n } else {\n glyphs.push(i, glyphset.glyphLoader(font, i));\n }\n }\n\n return glyphs;\n }\n\n function parseGlyfTableOnLowMemory(data, start, loca, font) {\n var glyphs = new glyphset.GlyphSet(font);\n\n font._push = function(i) {\n var offset = loca[i];\n var nextOffset = loca[i + 1];\n if (offset !== nextOffset) {\n glyphs.push(i, glyphset.ttfGlyphLoader(font, i, parseGlyph, data, start + offset, buildPath));\n } else {\n glyphs.push(i, glyphset.glyphLoader(font, i));\n }\n };\n\n return glyphs;\n }\n\n // Parse all the glyphs according to the offsets from the `loca` table.\n function parseGlyfTable(data, start, loca, font, opt) {\n if (opt.lowMemory)\n { return parseGlyfTableOnLowMemory(data, start, loca, font); }\n else\n { return parseGlyfTableAll(data, start, loca, font); }\n }\n\n var glyf = { getPath: getPath, parse: parseGlyfTable};\n\n /* A TrueType font hinting interpreter.\n *\n * (c) 2017 Axel Kittenberger\n *\n * This interpreter has been implemented according to this documentation:\n * https://developer.apple.com/fonts/TrueType-Reference-Manual/RM05/Chap5.html\n *\n * According to the documentation F24DOT6 values are used for pixels.\n * That means calculation is 1/64 pixel accurate and uses integer operations.\n * However, Javascript has floating point operations by default and only\n * those are available. One could make a case to simulate the 1/64 accuracy\n * exactly by truncating after every division operation\n * (for example with << 0) to get pixel exactly results as other TrueType\n * implementations. It may make sense since some fonts are pixel optimized\n * by hand using DELTAP instructions. The current implementation doesn't\n * and rather uses full floating point precision.\n *\n * xScale, yScale and rotation is currently ignored.\n *\n * A few non-trivial instructions are missing as I didn't encounter yet\n * a font that used them to test a possible implementation.\n *\n * Some fonts seem to use undocumented features regarding the twilight zone.\n * Only some of them are implemented as they were encountered.\n *\n * The exports.DEBUG statements are removed on the minified distribution file.\n */\n\n var instructionTable;\n var exec;\n var execGlyph;\n var execComponent;\n\n /*\n * Creates a hinting object.\n *\n * There ought to be exactly one\n * for each truetype font that is used for hinting.\n */\n function Hinting(font) {\n // the font this hinting object is for\n this.font = font;\n\n this.getCommands = function (hPoints) {\n return glyf.getPath(hPoints).commands;\n };\n\n // cached states\n this._fpgmState =\n this._prepState =\n undefined;\n\n // errorState\n // 0 ... all okay\n // 1 ... had an error in a glyf,\n // continue working but stop spamming\n // the console\n // 2 ... error at prep, stop hinting at this ppem\n // 3 ... error at fpeg, stop hinting for this font at all\n this._errorState = 0;\n }\n\n /*\n * Not rounding.\n */\n function roundOff(v) {\n return v;\n }\n\n /*\n * Rounding to grid.\n */\n function roundToGrid(v) {\n //Rounding in TT is supposed to \"symmetrical around zero\"\n return Math.sign(v) * Math.round(Math.abs(v));\n }\n\n /*\n * Rounding to double grid.\n */\n function roundToDoubleGrid(v) {\n return Math.sign(v) * Math.round(Math.abs(v * 2)) / 2;\n }\n\n /*\n * Rounding to half grid.\n */\n function roundToHalfGrid(v) {\n return Math.sign(v) * (Math.round(Math.abs(v) + 0.5) - 0.5);\n }\n\n /*\n * Rounding to up to grid.\n */\n function roundUpToGrid(v) {\n return Math.sign(v) * Math.ceil(Math.abs(v));\n }\n\n /*\n * Rounding to down to grid.\n */\n function roundDownToGrid(v) {\n return Math.sign(v) * Math.floor(Math.abs(v));\n }\n\n /*\n * Super rounding.\n */\n var roundSuper = function (v) {\n var period = this.srPeriod;\n var phase = this.srPhase;\n var threshold = this.srThreshold;\n var sign = 1;\n\n if (v < 0) {\n v = -v;\n sign = -1;\n }\n\n v += threshold - phase;\n\n v = Math.trunc(v / period) * period;\n\n v += phase;\n\n // according to http://xgridfit.sourceforge.net/round.html\n if (v < 0) { return phase * sign; }\n\n return v * sign;\n };\n\n /*\n * Unit vector of x-axis.\n */\n var xUnitVector = {\n x: 1,\n\n y: 0,\n\n axis: 'x',\n\n // Gets the projected distance between two points.\n // o1/o2 ... if true, respective original position is used.\n distance: function (p1, p2, o1, o2) {\n return (o1 ? p1.xo : p1.x) - (o2 ? p2.xo : p2.x);\n },\n\n // Moves point p so the moved position has the same relative\n // position to the moved positions of rp1 and rp2 than the\n // original positions had.\n //\n // See APPENDIX on INTERPOLATE at the bottom of this file.\n interpolate: function (p, rp1, rp2, pv) {\n var do1;\n var do2;\n var doa1;\n var doa2;\n var dm1;\n var dm2;\n var dt;\n\n if (!pv || pv === this) {\n do1 = p.xo - rp1.xo;\n do2 = p.xo - rp2.xo;\n dm1 = rp1.x - rp1.xo;\n dm2 = rp2.x - rp2.xo;\n doa1 = Math.abs(do1);\n doa2 = Math.abs(do2);\n dt = doa1 + doa2;\n\n if (dt === 0) {\n p.x = p.xo + (dm1 + dm2) / 2;\n return;\n }\n\n p.x = p.xo + (dm1 * doa2 + dm2 * doa1) / dt;\n return;\n }\n\n do1 = pv.distance(p, rp1, true, true);\n do2 = pv.distance(p, rp2, true, true);\n dm1 = pv.distance(rp1, rp1, false, true);\n dm2 = pv.distance(rp2, rp2, false, true);\n doa1 = Math.abs(do1);\n doa2 = Math.abs(do2);\n dt = doa1 + doa2;\n\n if (dt === 0) {\n xUnitVector.setRelative(p, p, (dm1 + dm2) / 2, pv, true);\n return;\n }\n\n xUnitVector.setRelative(p, p, (dm1 * doa2 + dm2 * doa1) / dt, pv, true);\n },\n\n // Slope of line normal to this\n normalSlope: Number.NEGATIVE_INFINITY,\n\n // Sets the point 'p' relative to point 'rp'\n // by the distance 'd'.\n //\n // See APPENDIX on SETRELATIVE at the bottom of this file.\n //\n // p ... point to set\n // rp ... reference point\n // d ... distance on projection vector\n // pv ... projection vector (undefined = this)\n // org ... if true, uses the original position of rp as reference.\n setRelative: function (p, rp, d, pv, org) {\n if (!pv || pv === this) {\n p.x = (org ? rp.xo : rp.x) + d;\n return;\n }\n\n var rpx = org ? rp.xo : rp.x;\n var rpy = org ? rp.yo : rp.y;\n var rpdx = rpx + d * pv.x;\n var rpdy = rpy + d * pv.y;\n\n p.x = rpdx + (p.y - rpdy) / pv.normalSlope;\n },\n\n // Slope of vector line.\n slope: 0,\n\n // Touches the point p.\n touch: function (p) {\n p.xTouched = true;\n },\n\n // Tests if a point p is touched.\n touched: function (p) {\n return p.xTouched;\n },\n\n // Untouches the point p.\n untouch: function (p) {\n p.xTouched = false;\n }\n };\n\n /*\n * Unit vector of y-axis.\n */\n var yUnitVector = {\n x: 0,\n\n y: 1,\n\n axis: 'y',\n\n // Gets the projected distance between two points.\n // o1/o2 ... if true, respective original position is used.\n distance: function (p1, p2, o1, o2) {\n return (o1 ? p1.yo : p1.y) - (o2 ? p2.yo : p2.y);\n },\n\n // Moves point p so the moved position has the same relative\n // position to the moved positions of rp1 and rp2 than the\n // original positions had.\n //\n // See APPENDIX on INTERPOLATE at the bottom of this file.\n interpolate: function (p, rp1, rp2, pv) {\n var do1;\n var do2;\n var doa1;\n var doa2;\n var dm1;\n var dm2;\n var dt;\n\n if (!pv || pv === this) {\n do1 = p.yo - rp1.yo;\n do2 = p.yo - rp2.yo;\n dm1 = rp1.y - rp1.yo;\n dm2 = rp2.y - rp2.yo;\n doa1 = Math.abs(do1);\n doa2 = Math.abs(do2);\n dt = doa1 + doa2;\n\n if (dt === 0) {\n p.y = p.yo + (dm1 + dm2) / 2;\n return;\n }\n\n p.y = p.yo + (dm1 * doa2 + dm2 * doa1) / dt;\n return;\n }\n\n do1 = pv.distance(p, rp1, true, true);\n do2 = pv.distance(p, rp2, true, true);\n dm1 = pv.distance(rp1, rp1, false, true);\n dm2 = pv.distance(rp2, rp2, false, true);\n doa1 = Math.abs(do1);\n doa2 = Math.abs(do2);\n dt = doa1 + doa2;\n\n if (dt === 0) {\n yUnitVector.setRelative(p, p, (dm1 + dm2) / 2, pv, true);\n return;\n }\n\n yUnitVector.setRelative(p, p, (dm1 * doa2 + dm2 * doa1) / dt, pv, true);\n },\n\n // Slope of line normal to this.\n normalSlope: 0,\n\n // Sets the point 'p' relative to point 'rp'\n // by the distance 'd'\n //\n // See APPENDIX on SETRELATIVE at the bottom of this file.\n //\n // p ... point to set\n // rp ... reference point\n // d ... distance on projection vector\n // pv ... projection vector (undefined = this)\n // org ... if true, uses the original position of rp as reference.\n setRelative: function (p, rp, d, pv, org) {\n if (!pv || pv === this) {\n p.y = (org ? rp.yo : rp.y) + d;\n return;\n }\n\n var rpx = org ? rp.xo : rp.x;\n var rpy = org ? rp.yo : rp.y;\n var rpdx = rpx + d * pv.x;\n var rpdy = rpy + d * pv.y;\n\n p.y = rpdy + pv.normalSlope * (p.x - rpdx);\n },\n\n // Slope of vector line.\n slope: Number.POSITIVE_INFINITY,\n\n // Touches the point p.\n touch: function (p) {\n p.yTouched = true;\n },\n\n // Tests if a point p is touched.\n touched: function (p) {\n return p.yTouched;\n },\n\n // Untouches the point p.\n untouch: function (p) {\n p.yTouched = false;\n }\n };\n\n Object.freeze(xUnitVector);\n Object.freeze(yUnitVector);\n\n /*\n * Creates a unit vector that is not x- or y-axis.\n */\n function UnitVector(x, y) {\n this.x = x;\n this.y = y;\n this.axis = undefined;\n this.slope = y / x;\n this.normalSlope = -x / y;\n Object.freeze(this);\n }\n\n /*\n * Gets the projected distance between two points.\n * o1/o2 ... if true, respective original position is used.\n */\n UnitVector.prototype.distance = function(p1, p2, o1, o2) {\n return (\n this.x * xUnitVector.distance(p1, p2, o1, o2) +\n this.y * yUnitVector.distance(p1, p2, o1, o2)\n );\n };\n\n /*\n * Moves point p so the moved position has the same relative\n * position to the moved positions of rp1 and rp2 than the\n * original positions had.\n *\n * See APPENDIX on INTERPOLATE at the bottom of this file.\n */\n UnitVector.prototype.interpolate = function(p, rp1, rp2, pv) {\n var dm1;\n var dm2;\n var do1;\n var do2;\n var doa1;\n var doa2;\n var dt;\n\n do1 = pv.distance(p, rp1, true, true);\n do2 = pv.distance(p, rp2, true, true);\n dm1 = pv.distance(rp1, rp1, false, true);\n dm2 = pv.distance(rp2, rp2, false, true);\n doa1 = Math.abs(do1);\n doa2 = Math.abs(do2);\n dt = doa1 + doa2;\n\n if (dt === 0) {\n this.setRelative(p, p, (dm1 + dm2) / 2, pv, true);\n return;\n }\n\n this.setRelative(p, p, (dm1 * doa2 + dm2 * doa1) / dt, pv, true);\n };\n\n /*\n * Sets the point 'p' relative to point 'rp'\n * by the distance 'd'\n *\n * See APPENDIX on SETRELATIVE at the bottom of this file.\n *\n * p ... point to set\n * rp ... reference point\n * d ... distance on projection vector\n * pv ... projection vector (undefined = this)\n * org ... if true, uses the original position of rp as reference.\n */\n UnitVector.prototype.setRelative = function(p, rp, d, pv, org) {\n pv = pv || this;\n\n var rpx = org ? rp.xo : rp.x;\n var rpy = org ? rp.yo : rp.y;\n var rpdx = rpx + d * pv.x;\n var rpdy = rpy + d * pv.y;\n\n var pvns = pv.normalSlope;\n var fvs = this.slope;\n\n var px = p.x;\n var py = p.y;\n\n p.x = (fvs * px - pvns * rpdx + rpdy - py) / (fvs - pvns);\n p.y = fvs * (p.x - px) + py;\n };\n\n /*\n * Touches the point p.\n */\n UnitVector.prototype.touch = function(p) {\n p.xTouched = true;\n p.yTouched = true;\n };\n\n /*\n * Returns a unit vector with x/y coordinates.\n */\n function getUnitVector(x, y) {\n var d = Math.sqrt(x * x + y * y);\n\n x /= d;\n y /= d;\n\n if (x === 1 && y === 0) { return xUnitVector; }\n else if (x === 0 && y === 1) { return yUnitVector; }\n else { return new UnitVector(x, y); }\n }\n\n /*\n * Creates a point in the hinting engine.\n */\n function HPoint(\n x,\n y,\n lastPointOfContour,\n onCurve\n ) {\n this.x = this.xo = Math.round(x * 64) / 64; // hinted x value and original x-value\n this.y = this.yo = Math.round(y * 64) / 64; // hinted y value and original y-value\n\n this.lastPointOfContour = lastPointOfContour;\n this.onCurve = onCurve;\n this.prevPointOnContour = undefined;\n this.nextPointOnContour = undefined;\n this.xTouched = false;\n this.yTouched = false;\n\n Object.preventExtensions(this);\n }\n\n /*\n * Returns the next touched point on the contour.\n *\n * v ... unit vector to test touch axis.\n */\n HPoint.prototype.nextTouched = function(v) {\n var p = this.nextPointOnContour;\n\n while (!v.touched(p) && p !== this) { p = p.nextPointOnContour; }\n\n return p;\n };\n\n /*\n * Returns the previous touched point on the contour\n *\n * v ... unit vector to test touch axis.\n */\n HPoint.prototype.prevTouched = function(v) {\n var p = this.prevPointOnContour;\n\n while (!v.touched(p) && p !== this) { p = p.prevPointOnContour; }\n\n return p;\n };\n\n /*\n * The zero point.\n */\n var HPZero = Object.freeze(new HPoint(0, 0));\n\n /*\n * The default state of the interpreter.\n *\n * Note: Freezing the defaultState and then deriving from it\n * makes the V8 Javascript engine going awkward,\n * so this is avoided, albeit the defaultState shouldn't\n * ever change.\n */\n var defaultState = {\n cvCutIn: 17 / 16, // control value cut in\n deltaBase: 9,\n deltaShift: 0.125,\n loop: 1, // loops some instructions\n minDis: 1, // minimum distance\n autoFlip: true\n };\n\n /*\n * The current state of the interpreter.\n *\n * env ... 'fpgm' or 'prep' or 'glyf'\n * prog ... the program\n */\n function State(env, prog) {\n this.env = env;\n this.stack = [];\n this.prog = prog;\n\n switch (env) {\n case 'glyf' :\n this.zp0 = this.zp1 = this.zp2 = 1;\n this.rp0 = this.rp1 = this.rp2 = 0;\n /* fall through */\n case 'prep' :\n this.fv = this.pv = this.dpv = xUnitVector;\n this.round = roundToGrid;\n }\n }\n\n /*\n * Executes a glyph program.\n *\n * This does the hinting for each glyph.\n *\n * Returns an array of moved points.\n *\n * glyph: the glyph to hint\n * ppem: the size the glyph is rendered for\n */\n Hinting.prototype.exec = function(glyph, ppem) {\n if (typeof ppem !== 'number') {\n throw new Error('Point size is not a number!');\n }\n\n // Received a fatal error, don't do any hinting anymore.\n if (this._errorState > 2) { return; }\n\n var font = this.font;\n var prepState = this._prepState;\n\n if (!prepState || prepState.ppem !== ppem) {\n var fpgmState = this._fpgmState;\n\n if (!fpgmState) {\n // Executes the fpgm state.\n // This is used by fonts to define functions.\n State.prototype = defaultState;\n\n fpgmState =\n this._fpgmState =\n new State('fpgm', font.tables.fpgm);\n\n fpgmState.funcs = [ ];\n fpgmState.font = font;\n\n if (exports.DEBUG) {\n console.log('---EXEC FPGM---');\n fpgmState.step = -1;\n }\n\n try {\n exec(fpgmState);\n } catch (e) {\n console.log('Hinting error in FPGM:' + e);\n this._errorState = 3;\n return;\n }\n }\n\n // Executes the prep program for this ppem setting.\n // This is used by fonts to set cvt values\n // depending on to be rendered font size.\n\n State.prototype = fpgmState;\n prepState =\n this._prepState =\n new State('prep', font.tables.prep);\n\n prepState.ppem = ppem;\n\n // Creates a copy of the cvt table\n // and scales it to the current ppem setting.\n var oCvt = font.tables.cvt;\n if (oCvt) {\n var cvt = prepState.cvt = new Array(oCvt.length);\n var scale = ppem / font.unitsPerEm;\n for (var c = 0; c < oCvt.length; c++) {\n cvt[c] = oCvt[c] * scale;\n }\n } else {\n prepState.cvt = [];\n }\n\n if (exports.DEBUG) {\n console.log('---EXEC PREP---');\n prepState.step = -1;\n }\n\n try {\n exec(prepState);\n } catch (e) {\n if (this._errorState < 2) {\n console.log('Hinting error in PREP:' + e);\n }\n this._errorState = 2;\n }\n }\n\n if (this._errorState > 1) { return; }\n\n try {\n return execGlyph(glyph, prepState);\n } catch (e) {\n if (this._errorState < 1) {\n console.log('Hinting error:' + e);\n console.log('Note: further hinting errors are silenced');\n }\n this._errorState = 1;\n return undefined;\n }\n };\n\n /*\n * Executes the hinting program for a glyph.\n */\n execGlyph = function(glyph, prepState) {\n // original point positions\n var xScale = prepState.ppem / prepState.font.unitsPerEm;\n var yScale = xScale;\n var components = glyph.components;\n var contours;\n var gZone;\n var state;\n\n State.prototype = prepState;\n if (!components) {\n state = new State('glyf', glyph.instructions);\n if (exports.DEBUG) {\n console.log('---EXEC GLYPH---');\n state.step = -1;\n }\n execComponent(glyph, state, xScale, yScale);\n gZone = state.gZone;\n } else {\n var font = prepState.font;\n gZone = [];\n contours = [];\n for (var i = 0; i < components.length; i++) {\n var c = components[i];\n var cg = font.glyphs.get(c.glyphIndex);\n\n state = new State('glyf', cg.instructions);\n\n if (exports.DEBUG) {\n console.log('---EXEC COMP ' + i + '---');\n state.step = -1;\n }\n\n execComponent(cg, state, xScale, yScale);\n // appends the computed points to the result array\n // post processes the component points\n var dx = Math.round(c.dx * xScale);\n var dy = Math.round(c.dy * yScale);\n var gz = state.gZone;\n var cc = state.contours;\n for (var pi = 0; pi < gz.length; pi++) {\n var p = gz[pi];\n p.xTouched = p.yTouched = false;\n p.xo = p.x = p.x + dx;\n p.yo = p.y = p.y + dy;\n }\n\n var gLen = gZone.length;\n gZone.push.apply(gZone, gz);\n for (var j = 0; j < cc.length; j++) {\n contours.push(cc[j] + gLen);\n }\n }\n\n if (glyph.instructions && !state.inhibitGridFit) {\n // the composite has instructions on its own\n state = new State('glyf', glyph.instructions);\n\n state.gZone = state.z0 = state.z1 = state.z2 = gZone;\n\n state.contours = contours;\n\n // note: HPZero cannot be used here, since\n // the point might be modified\n gZone.push(\n new HPoint(0, 0),\n new HPoint(Math.round(glyph.advanceWidth * xScale), 0)\n );\n\n if (exports.DEBUG) {\n console.log('---EXEC COMPOSITE---');\n state.step = -1;\n }\n\n exec(state);\n\n gZone.length -= 2;\n }\n }\n\n return gZone;\n };\n\n /*\n * Executes the hinting program for a component of a multi-component glyph\n * or of the glyph itself for a non-component glyph.\n */\n execComponent = function(glyph, state, xScale, yScale)\n {\n var points = glyph.points || [];\n var pLen = points.length;\n var gZone = state.gZone = state.z0 = state.z1 = state.z2 = [];\n var contours = state.contours = [];\n\n // Scales the original points and\n // makes copies for the hinted points.\n var cp; // current point\n for (var i = 0; i < pLen; i++) {\n cp = points[i];\n\n gZone[i] = new HPoint(\n cp.x * xScale,\n cp.y * yScale,\n cp.lastPointOfContour,\n cp.onCurve\n );\n }\n\n // Chain links the contours.\n var sp; // start point\n var np; // next point\n\n for (var i$1 = 0; i$1 < pLen; i$1++) {\n cp = gZone[i$1];\n\n if (!sp) {\n sp = cp;\n contours.push(i$1);\n }\n\n if (cp.lastPointOfContour) {\n cp.nextPointOnContour = sp;\n sp.prevPointOnContour = cp;\n sp = undefined;\n } else {\n np = gZone[i$1 + 1];\n cp.nextPointOnContour = np;\n np.prevPointOnContour = cp;\n }\n }\n\n if (state.inhibitGridFit) { return; }\n\n if (exports.DEBUG) {\n console.log('PROCESSING GLYPH', state.stack);\n for (var i$2 = 0; i$2 < pLen; i$2++) {\n console.log(i$2, gZone[i$2].x, gZone[i$2].y);\n }\n }\n\n gZone.push(\n new HPoint(0, 0),\n new HPoint(Math.round(glyph.advanceWidth * xScale), 0)\n );\n\n exec(state);\n\n // Removes the extra points.\n gZone.length -= 2;\n\n if (exports.DEBUG) {\n console.log('FINISHED GLYPH', state.stack);\n for (var i$3 = 0; i$3 < pLen; i$3++) {\n console.log(i$3, gZone[i$3].x, gZone[i$3].y);\n }\n }\n };\n\n /*\n * Executes the program loaded in state.\n */\n exec = function(state) {\n var prog = state.prog;\n\n if (!prog) { return; }\n\n var pLen = prog.length;\n var ins;\n\n for (state.ip = 0; state.ip < pLen; state.ip++) {\n if (exports.DEBUG) { state.step++; }\n ins = instructionTable[prog[state.ip]];\n\n if (!ins) {\n throw new Error(\n 'unknown instruction: 0x' +\n Number(prog[state.ip]).toString(16)\n );\n }\n\n ins(state);\n\n // very extensive debugging for each step\n /*\n if (exports.DEBUG) {\n var da;\n if (state.gZone) {\n da = [];\n for (let i = 0; i < state.gZone.length; i++)\n {\n da.push(i + ' ' +\n state.gZone[i].x * 64 + ' ' +\n state.gZone[i].y * 64 + ' ' +\n (state.gZone[i].xTouched ? 'x' : '') +\n (state.gZone[i].yTouched ? 'y' : '')\n );\n }\n console.log('GZ', da);\n }\n\n if (state.tZone) {\n da = [];\n for (let i = 0; i < state.tZone.length; i++) {\n da.push(i + ' ' +\n state.tZone[i].x * 64 + ' ' +\n state.tZone[i].y * 64 + ' ' +\n (state.tZone[i].xTouched ? 'x' : '') +\n (state.tZone[i].yTouched ? 'y' : '')\n );\n }\n console.log('TZ', da);\n }\n\n if (state.stack.length > 10) {\n console.log(\n state.stack.length,\n '...', state.stack.slice(state.stack.length - 10)\n );\n } else {\n console.log(state.stack.length, state.stack);\n }\n }\n */\n }\n };\n\n /*\n * Initializes the twilight zone.\n *\n * This is only done if a SZPx instruction\n * refers to the twilight zone.\n */\n function initTZone(state)\n {\n var tZone = state.tZone = new Array(state.gZone.length);\n\n // no idea if this is actually correct...\n for (var i = 0; i < tZone.length; i++)\n {\n tZone[i] = new HPoint(0, 0);\n }\n }\n\n /*\n * Skips the instruction pointer ahead over an IF/ELSE block.\n * handleElse .. if true breaks on matching ELSE\n */\n function skip(state, handleElse)\n {\n var prog = state.prog;\n var ip = state.ip;\n var nesting = 1;\n var ins;\n\n do {\n ins = prog[++ip];\n if (ins === 0x58) // IF\n { nesting++; }\n else if (ins === 0x59) // EIF\n { nesting--; }\n else if (ins === 0x40) // NPUSHB\n { ip += prog[ip + 1] + 1; }\n else if (ins === 0x41) // NPUSHW\n { ip += 2 * prog[ip + 1] + 1; }\n else if (ins >= 0xB0 && ins <= 0xB7) // PUSHB\n { ip += ins - 0xB0 + 1; }\n else if (ins >= 0xB8 && ins <= 0xBF) // PUSHW\n { ip += (ins - 0xB8 + 1) * 2; }\n else if (handleElse && nesting === 1 && ins === 0x1B) // ELSE\n { break; }\n } while (nesting > 0);\n\n state.ip = ip;\n }\n\n /*----------------------------------------------------------*\n * And then a lot of instructions... *\n *----------------------------------------------------------*/\n\n // SVTCA[a] Set freedom and projection Vectors To Coordinate Axis\n // 0x00-0x01\n function SVTCA(v, state) {\n if (exports.DEBUG) { console.log(state.step, 'SVTCA[' + v.axis + ']'); }\n\n state.fv = state.pv = state.dpv = v;\n }\n\n // SPVTCA[a] Set Projection Vector to Coordinate Axis\n // 0x02-0x03\n function SPVTCA(v, state) {\n if (exports.DEBUG) { console.log(state.step, 'SPVTCA[' + v.axis + ']'); }\n\n state.pv = state.dpv = v;\n }\n\n // SFVTCA[a] Set Freedom Vector to Coordinate Axis\n // 0x04-0x05\n function SFVTCA(v, state) {\n if (exports.DEBUG) { console.log(state.step, 'SFVTCA[' + v.axis + ']'); }\n\n state.fv = v;\n }\n\n // SPVTL[a] Set Projection Vector To Line\n // 0x06-0x07\n function SPVTL(a, state) {\n var stack = state.stack;\n var p2i = stack.pop();\n var p1i = stack.pop();\n var p2 = state.z2[p2i];\n var p1 = state.z1[p1i];\n\n if (exports.DEBUG) { console.log('SPVTL[' + a + ']', p2i, p1i); }\n\n var dx;\n var dy;\n\n if (!a) {\n dx = p1.x - p2.x;\n dy = p1.y - p2.y;\n } else {\n dx = p2.y - p1.y;\n dy = p1.x - p2.x;\n }\n\n state.pv = state.dpv = getUnitVector(dx, dy);\n }\n\n // SFVTL[a] Set Freedom Vector To Line\n // 0x08-0x09\n function SFVTL(a, state) {\n var stack = state.stack;\n var p2i = stack.pop();\n var p1i = stack.pop();\n var p2 = state.z2[p2i];\n var p1 = state.z1[p1i];\n\n if (exports.DEBUG) { console.log('SFVTL[' + a + ']', p2i, p1i); }\n\n var dx;\n var dy;\n\n if (!a) {\n dx = p1.x - p2.x;\n dy = p1.y - p2.y;\n } else {\n dx = p2.y - p1.y;\n dy = p1.x - p2.x;\n }\n\n state.fv = getUnitVector(dx, dy);\n }\n\n // SPVFS[] Set Projection Vector From Stack\n // 0x0A\n function SPVFS(state) {\n var stack = state.stack;\n var y = stack.pop();\n var x = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'SPVFS[]', y, x); }\n\n state.pv = state.dpv = getUnitVector(x, y);\n }\n\n // SFVFS[] Set Freedom Vector From Stack\n // 0x0B\n function SFVFS(state) {\n var stack = state.stack;\n var y = stack.pop();\n var x = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'SPVFS[]', y, x); }\n\n state.fv = getUnitVector(x, y);\n }\n\n // GPV[] Get Projection Vector\n // 0x0C\n function GPV(state) {\n var stack = state.stack;\n var pv = state.pv;\n\n if (exports.DEBUG) { console.log(state.step, 'GPV[]'); }\n\n stack.push(pv.x * 0x4000);\n stack.push(pv.y * 0x4000);\n }\n\n // GFV[] Get Freedom Vector\n // 0x0C\n function GFV(state) {\n var stack = state.stack;\n var fv = state.fv;\n\n if (exports.DEBUG) { console.log(state.step, 'GFV[]'); }\n\n stack.push(fv.x * 0x4000);\n stack.push(fv.y * 0x4000);\n }\n\n // SFVTPV[] Set Freedom Vector To Projection Vector\n // 0x0E\n function SFVTPV(state) {\n state.fv = state.pv;\n\n if (exports.DEBUG) { console.log(state.step, 'SFVTPV[]'); }\n }\n\n // ISECT[] moves point p to the InterSECTion of two lines\n // 0x0F\n function ISECT(state)\n {\n var stack = state.stack;\n var pa0i = stack.pop();\n var pa1i = stack.pop();\n var pb0i = stack.pop();\n var pb1i = stack.pop();\n var pi = stack.pop();\n var z0 = state.z0;\n var z1 = state.z1;\n var pa0 = z0[pa0i];\n var pa1 = z0[pa1i];\n var pb0 = z1[pb0i];\n var pb1 = z1[pb1i];\n var p = state.z2[pi];\n\n if (exports.DEBUG) { console.log('ISECT[], ', pa0i, pa1i, pb0i, pb1i, pi); }\n\n // math from\n // en.wikipedia.org/wiki/Line%E2%80%93line_intersection#Given_two_points_on_each_line\n\n var x1 = pa0.x;\n var y1 = pa0.y;\n var x2 = pa1.x;\n var y2 = pa1.y;\n var x3 = pb0.x;\n var y3 = pb0.y;\n var x4 = pb1.x;\n var y4 = pb1.y;\n\n var div = (x1 - x2) * (y3 - y4) - (y1 - y2) * (x3 - x4);\n var f1 = x1 * y2 - y1 * x2;\n var f2 = x3 * y4 - y3 * x4;\n\n p.x = (f1 * (x3 - x4) - f2 * (x1 - x2)) / div;\n p.y = (f1 * (y3 - y4) - f2 * (y1 - y2)) / div;\n }\n\n // SRP0[] Set Reference Point 0\n // 0x10\n function SRP0(state) {\n state.rp0 = state.stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'SRP0[]', state.rp0); }\n }\n\n // SRP1[] Set Reference Point 1\n // 0x11\n function SRP1(state) {\n state.rp1 = state.stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'SRP1[]', state.rp1); }\n }\n\n // SRP1[] Set Reference Point 2\n // 0x12\n function SRP2(state) {\n state.rp2 = state.stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'SRP2[]', state.rp2); }\n }\n\n // SZP0[] Set Zone Pointer 0\n // 0x13\n function SZP0(state) {\n var n = state.stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'SZP0[]', n); }\n\n state.zp0 = n;\n\n switch (n) {\n case 0:\n if (!state.tZone) { initTZone(state); }\n state.z0 = state.tZone;\n break;\n case 1 :\n state.z0 = state.gZone;\n break;\n default :\n throw new Error('Invalid zone pointer');\n }\n }\n\n // SZP1[] Set Zone Pointer 1\n // 0x14\n function SZP1(state) {\n var n = state.stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'SZP1[]', n); }\n\n state.zp1 = n;\n\n switch (n) {\n case 0:\n if (!state.tZone) { initTZone(state); }\n state.z1 = state.tZone;\n break;\n case 1 :\n state.z1 = state.gZone;\n break;\n default :\n throw new Error('Invalid zone pointer');\n }\n }\n\n // SZP2[] Set Zone Pointer 2\n // 0x15\n function SZP2(state) {\n var n = state.stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'SZP2[]', n); }\n\n state.zp2 = n;\n\n switch (n) {\n case 0:\n if (!state.tZone) { initTZone(state); }\n state.z2 = state.tZone;\n break;\n case 1 :\n state.z2 = state.gZone;\n break;\n default :\n throw new Error('Invalid zone pointer');\n }\n }\n\n // SZPS[] Set Zone PointerS\n // 0x16\n function SZPS(state) {\n var n = state.stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'SZPS[]', n); }\n\n state.zp0 = state.zp1 = state.zp2 = n;\n\n switch (n) {\n case 0:\n if (!state.tZone) { initTZone(state); }\n state.z0 = state.z1 = state.z2 = state.tZone;\n break;\n case 1 :\n state.z0 = state.z1 = state.z2 = state.gZone;\n break;\n default :\n throw new Error('Invalid zone pointer');\n }\n }\n\n // SLOOP[] Set LOOP variable\n // 0x17\n function SLOOP(state) {\n state.loop = state.stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'SLOOP[]', state.loop); }\n }\n\n // RTG[] Round To Grid\n // 0x18\n function RTG(state) {\n if (exports.DEBUG) { console.log(state.step, 'RTG[]'); }\n\n state.round = roundToGrid;\n }\n\n // RTHG[] Round To Half Grid\n // 0x19\n function RTHG(state) {\n if (exports.DEBUG) { console.log(state.step, 'RTHG[]'); }\n\n state.round = roundToHalfGrid;\n }\n\n // SMD[] Set Minimum Distance\n // 0x1A\n function SMD(state) {\n var d = state.stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'SMD[]', d); }\n\n state.minDis = d / 0x40;\n }\n\n // ELSE[] ELSE clause\n // 0x1B\n function ELSE(state) {\n // This instruction has been reached by executing a then branch\n // so it just skips ahead until matching EIF.\n //\n // In case the IF was negative the IF[] instruction already\n // skipped forward over the ELSE[]\n\n if (exports.DEBUG) { console.log(state.step, 'ELSE[]'); }\n\n skip(state, false);\n }\n\n // JMPR[] JuMP Relative\n // 0x1C\n function JMPR(state) {\n var o = state.stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'JMPR[]', o); }\n\n // A jump by 1 would do nothing.\n state.ip += o - 1;\n }\n\n // SCVTCI[] Set Control Value Table Cut-In\n // 0x1D\n function SCVTCI(state) {\n var n = state.stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'SCVTCI[]', n); }\n\n state.cvCutIn = n / 0x40;\n }\n\n // DUP[] DUPlicate top stack element\n // 0x20\n function DUP(state) {\n var stack = state.stack;\n\n if (exports.DEBUG) { console.log(state.step, 'DUP[]'); }\n\n stack.push(stack[stack.length - 1]);\n }\n\n // POP[] POP top stack element\n // 0x21\n function POP(state) {\n if (exports.DEBUG) { console.log(state.step, 'POP[]'); }\n\n state.stack.pop();\n }\n\n // CLEAR[] CLEAR the stack\n // 0x22\n function CLEAR(state) {\n if (exports.DEBUG) { console.log(state.step, 'CLEAR[]'); }\n\n state.stack.length = 0;\n }\n\n // SWAP[] SWAP the top two elements on the stack\n // 0x23\n function SWAP(state) {\n var stack = state.stack;\n\n var a = stack.pop();\n var b = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'SWAP[]'); }\n\n stack.push(a);\n stack.push(b);\n }\n\n // DEPTH[] DEPTH of the stack\n // 0x24\n function DEPTH(state) {\n var stack = state.stack;\n\n if (exports.DEBUG) { console.log(state.step, 'DEPTH[]'); }\n\n stack.push(stack.length);\n }\n\n // LOOPCALL[] LOOPCALL function\n // 0x2A\n function LOOPCALL(state) {\n var stack = state.stack;\n var fn = stack.pop();\n var c = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'LOOPCALL[]', fn, c); }\n\n // saves callers program\n var cip = state.ip;\n var cprog = state.prog;\n\n state.prog = state.funcs[fn];\n\n // executes the function\n for (var i = 0; i < c; i++) {\n exec(state);\n\n if (exports.DEBUG) { console.log(\n ++state.step,\n i + 1 < c ? 'next loopcall' : 'done loopcall',\n i\n ); }\n }\n\n // restores the callers program\n state.ip = cip;\n state.prog = cprog;\n }\n\n // CALL[] CALL function\n // 0x2B\n function CALL(state) {\n var fn = state.stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'CALL[]', fn); }\n\n // saves callers program\n var cip = state.ip;\n var cprog = state.prog;\n\n state.prog = state.funcs[fn];\n\n // executes the function\n exec(state);\n\n // restores the callers program\n state.ip = cip;\n state.prog = cprog;\n\n if (exports.DEBUG) { console.log(++state.step, 'returning from', fn); }\n }\n\n // CINDEX[] Copy the INDEXed element to the top of the stack\n // 0x25\n function CINDEX(state) {\n var stack = state.stack;\n var k = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'CINDEX[]', k); }\n\n // In case of k == 1, it copies the last element after popping\n // thus stack.length - k.\n stack.push(stack[stack.length - k]);\n }\n\n // MINDEX[] Move the INDEXed element to the top of the stack\n // 0x26\n function MINDEX(state) {\n var stack = state.stack;\n var k = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'MINDEX[]', k); }\n\n stack.push(stack.splice(stack.length - k, 1)[0]);\n }\n\n // FDEF[] Function DEFinition\n // 0x2C\n function FDEF(state) {\n if (state.env !== 'fpgm') { throw new Error('FDEF not allowed here'); }\n var stack = state.stack;\n var prog = state.prog;\n var ip = state.ip;\n\n var fn = stack.pop();\n var ipBegin = ip;\n\n if (exports.DEBUG) { console.log(state.step, 'FDEF[]', fn); }\n\n while (prog[++ip] !== 0x2D){ }\n\n state.ip = ip;\n state.funcs[fn] = prog.slice(ipBegin + 1, ip);\n }\n\n // MDAP[a] Move Direct Absolute Point\n // 0x2E-0x2F\n function MDAP(round, state) {\n var pi = state.stack.pop();\n var p = state.z0[pi];\n var fv = state.fv;\n var pv = state.pv;\n\n if (exports.DEBUG) { console.log(state.step, 'MDAP[' + round + ']', pi); }\n\n var d = pv.distance(p, HPZero);\n\n if (round) { d = state.round(d); }\n\n fv.setRelative(p, HPZero, d, pv);\n fv.touch(p);\n\n state.rp0 = state.rp1 = pi;\n }\n\n // IUP[a] Interpolate Untouched Points through the outline\n // 0x30\n function IUP(v, state) {\n var z2 = state.z2;\n var pLen = z2.length - 2;\n var cp;\n var pp;\n var np;\n\n if (exports.DEBUG) { console.log(state.step, 'IUP[' + v.axis + ']'); }\n\n for (var i = 0; i < pLen; i++) {\n cp = z2[i]; // current point\n\n // if this point has been touched go on\n if (v.touched(cp)) { continue; }\n\n pp = cp.prevTouched(v);\n\n // no point on the contour has been touched?\n if (pp === cp) { continue; }\n\n np = cp.nextTouched(v);\n\n if (pp === np) {\n // only one point on the contour has been touched\n // so simply moves the point like that\n\n v.setRelative(cp, cp, v.distance(pp, pp, false, true), v, true);\n }\n\n v.interpolate(cp, pp, np, v);\n }\n }\n\n // SHP[] SHift Point using reference point\n // 0x32-0x33\n function SHP(a, state) {\n var stack = state.stack;\n var rpi = a ? state.rp1 : state.rp2;\n var rp = (a ? state.z0 : state.z1)[rpi];\n var fv = state.fv;\n var pv = state.pv;\n var loop = state.loop;\n var z2 = state.z2;\n\n while (loop--)\n {\n var pi = stack.pop();\n var p = z2[pi];\n\n var d = pv.distance(rp, rp, false, true);\n fv.setRelative(p, p, d, pv);\n fv.touch(p);\n\n if (exports.DEBUG) {\n console.log(\n state.step,\n (state.loop > 1 ?\n 'loop ' + (state.loop - loop) + ': ' :\n ''\n ) +\n 'SHP[' + (a ? 'rp1' : 'rp2') + ']', pi\n );\n }\n }\n\n state.loop = 1;\n }\n\n // SHC[] SHift Contour using reference point\n // 0x36-0x37\n function SHC(a, state) {\n var stack = state.stack;\n var rpi = a ? state.rp1 : state.rp2;\n var rp = (a ? state.z0 : state.z1)[rpi];\n var fv = state.fv;\n var pv = state.pv;\n var ci = stack.pop();\n var sp = state.z2[state.contours[ci]];\n var p = sp;\n\n if (exports.DEBUG) { console.log(state.step, 'SHC[' + a + ']', ci); }\n\n var d = pv.distance(rp, rp, false, true);\n\n do {\n if (p !== rp) { fv.setRelative(p, p, d, pv); }\n p = p.nextPointOnContour;\n } while (p !== sp);\n }\n\n // SHZ[] SHift Zone using reference point\n // 0x36-0x37\n function SHZ(a, state) {\n var stack = state.stack;\n var rpi = a ? state.rp1 : state.rp2;\n var rp = (a ? state.z0 : state.z1)[rpi];\n var fv = state.fv;\n var pv = state.pv;\n\n var e = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'SHZ[' + a + ']', e); }\n\n var z;\n switch (e) {\n case 0 : z = state.tZone; break;\n case 1 : z = state.gZone; break;\n default : throw new Error('Invalid zone');\n }\n\n var p;\n var d = pv.distance(rp, rp, false, true);\n var pLen = z.length - 2;\n for (var i = 0; i < pLen; i++)\n {\n p = z[i];\n fv.setRelative(p, p, d, pv);\n //if (p !== rp) fv.setRelative(p, p, d, pv);\n }\n }\n\n // SHPIX[] SHift point by a PIXel amount\n // 0x38\n function SHPIX(state) {\n var stack = state.stack;\n var loop = state.loop;\n var fv = state.fv;\n var d = stack.pop() / 0x40;\n var z2 = state.z2;\n\n while (loop--) {\n var pi = stack.pop();\n var p = z2[pi];\n\n if (exports.DEBUG) {\n console.log(\n state.step,\n (state.loop > 1 ? 'loop ' + (state.loop - loop) + ': ' : '') +\n 'SHPIX[]', pi, d\n );\n }\n\n fv.setRelative(p, p, d);\n fv.touch(p);\n }\n\n state.loop = 1;\n }\n\n // IP[] Interpolate Point\n // 0x39\n function IP(state) {\n var stack = state.stack;\n var rp1i = state.rp1;\n var rp2i = state.rp2;\n var loop = state.loop;\n var rp1 = state.z0[rp1i];\n var rp2 = state.z1[rp2i];\n var fv = state.fv;\n var pv = state.dpv;\n var z2 = state.z2;\n\n while (loop--) {\n var pi = stack.pop();\n var p = z2[pi];\n\n if (exports.DEBUG) {\n console.log(\n state.step,\n (state.loop > 1 ? 'loop ' + (state.loop - loop) + ': ' : '') +\n 'IP[]', pi, rp1i, '<->', rp2i\n );\n }\n\n fv.interpolate(p, rp1, rp2, pv);\n\n fv.touch(p);\n }\n\n state.loop = 1;\n }\n\n // MSIRP[a] Move Stack Indirect Relative Point\n // 0x3A-0x3B\n function MSIRP(a, state) {\n var stack = state.stack;\n var d = stack.pop() / 64;\n var pi = stack.pop();\n var p = state.z1[pi];\n var rp0 = state.z0[state.rp0];\n var fv = state.fv;\n var pv = state.pv;\n\n fv.setRelative(p, rp0, d, pv);\n fv.touch(p);\n\n if (exports.DEBUG) { console.log(state.step, 'MSIRP[' + a + ']', d, pi); }\n\n state.rp1 = state.rp0;\n state.rp2 = pi;\n if (a) { state.rp0 = pi; }\n }\n\n // ALIGNRP[] Align to reference point.\n // 0x3C\n function ALIGNRP(state) {\n var stack = state.stack;\n var rp0i = state.rp0;\n var rp0 = state.z0[rp0i];\n var loop = state.loop;\n var fv = state.fv;\n var pv = state.pv;\n var z1 = state.z1;\n\n while (loop--) {\n var pi = stack.pop();\n var p = z1[pi];\n\n if (exports.DEBUG) {\n console.log(\n state.step,\n (state.loop > 1 ? 'loop ' + (state.loop - loop) + ': ' : '') +\n 'ALIGNRP[]', pi\n );\n }\n\n fv.setRelative(p, rp0, 0, pv);\n fv.touch(p);\n }\n\n state.loop = 1;\n }\n\n // RTG[] Round To Double Grid\n // 0x3D\n function RTDG(state) {\n if (exports.DEBUG) { console.log(state.step, 'RTDG[]'); }\n\n state.round = roundToDoubleGrid;\n }\n\n // MIAP[a] Move Indirect Absolute Point\n // 0x3E-0x3F\n function MIAP(round, state) {\n var stack = state.stack;\n var n = stack.pop();\n var pi = stack.pop();\n var p = state.z0[pi];\n var fv = state.fv;\n var pv = state.pv;\n var cv = state.cvt[n];\n\n if (exports.DEBUG) {\n console.log(\n state.step,\n 'MIAP[' + round + ']',\n n, '(', cv, ')', pi\n );\n }\n\n var d = pv.distance(p, HPZero);\n\n if (round) {\n if (Math.abs(d - cv) < state.cvCutIn) { d = cv; }\n\n d = state.round(d);\n }\n\n fv.setRelative(p, HPZero, d, pv);\n\n if (state.zp0 === 0) {\n p.xo = p.x;\n p.yo = p.y;\n }\n\n fv.touch(p);\n\n state.rp0 = state.rp1 = pi;\n }\n\n // NPUSB[] PUSH N Bytes\n // 0x40\n function NPUSHB(state) {\n var prog = state.prog;\n var ip = state.ip;\n var stack = state.stack;\n\n var n = prog[++ip];\n\n if (exports.DEBUG) { console.log(state.step, 'NPUSHB[]', n); }\n\n for (var i = 0; i < n; i++) { stack.push(prog[++ip]); }\n\n state.ip = ip;\n }\n\n // NPUSHW[] PUSH N Words\n // 0x41\n function NPUSHW(state) {\n var ip = state.ip;\n var prog = state.prog;\n var stack = state.stack;\n var n = prog[++ip];\n\n if (exports.DEBUG) { console.log(state.step, 'NPUSHW[]', n); }\n\n for (var i = 0; i < n; i++) {\n var w = (prog[++ip] << 8) | prog[++ip];\n if (w & 0x8000) { w = -((w ^ 0xffff) + 1); }\n stack.push(w);\n }\n\n state.ip = ip;\n }\n\n // WS[] Write Store\n // 0x42\n function WS(state) {\n var stack = state.stack;\n var store = state.store;\n\n if (!store) { store = state.store = []; }\n\n var v = stack.pop();\n var l = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'WS', v, l); }\n\n store[l] = v;\n }\n\n // RS[] Read Store\n // 0x43\n function RS(state) {\n var stack = state.stack;\n var store = state.store;\n\n var l = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'RS', l); }\n\n var v = (store && store[l]) || 0;\n\n stack.push(v);\n }\n\n // WCVTP[] Write Control Value Table in Pixel units\n // 0x44\n function WCVTP(state) {\n var stack = state.stack;\n\n var v = stack.pop();\n var l = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'WCVTP', v, l); }\n\n state.cvt[l] = v / 0x40;\n }\n\n // RCVT[] Read Control Value Table entry\n // 0x45\n function RCVT(state) {\n var stack = state.stack;\n var cvte = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'RCVT', cvte); }\n\n stack.push(state.cvt[cvte] * 0x40);\n }\n\n // GC[] Get Coordinate projected onto the projection vector\n // 0x46-0x47\n function GC(a, state) {\n var stack = state.stack;\n var pi = stack.pop();\n var p = state.z2[pi];\n\n if (exports.DEBUG) { console.log(state.step, 'GC[' + a + ']', pi); }\n\n stack.push(state.dpv.distance(p, HPZero, a, false) * 0x40);\n }\n\n // MD[a] Measure Distance\n // 0x49-0x4A\n function MD(a, state) {\n var stack = state.stack;\n var pi2 = stack.pop();\n var pi1 = stack.pop();\n var p2 = state.z1[pi2];\n var p1 = state.z0[pi1];\n var d = state.dpv.distance(p1, p2, a, a);\n\n if (exports.DEBUG) { console.log(state.step, 'MD[' + a + ']', pi2, pi1, '->', d); }\n\n state.stack.push(Math.round(d * 64));\n }\n\n // MPPEM[] Measure Pixels Per EM\n // 0x4B\n function MPPEM(state) {\n if (exports.DEBUG) { console.log(state.step, 'MPPEM[]'); }\n state.stack.push(state.ppem);\n }\n\n // FLIPON[] set the auto FLIP Boolean to ON\n // 0x4D\n function FLIPON(state) {\n if (exports.DEBUG) { console.log(state.step, 'FLIPON[]'); }\n state.autoFlip = true;\n }\n\n // LT[] Less Than\n // 0x50\n function LT(state) {\n var stack = state.stack;\n var e2 = stack.pop();\n var e1 = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'LT[]', e2, e1); }\n\n stack.push(e1 < e2 ? 1 : 0);\n }\n\n // LTEQ[] Less Than or EQual\n // 0x53\n function LTEQ(state) {\n var stack = state.stack;\n var e2 = stack.pop();\n var e1 = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'LTEQ[]', e2, e1); }\n\n stack.push(e1 <= e2 ? 1 : 0);\n }\n\n // GTEQ[] Greater Than\n // 0x52\n function GT(state) {\n var stack = state.stack;\n var e2 = stack.pop();\n var e1 = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'GT[]', e2, e1); }\n\n stack.push(e1 > e2 ? 1 : 0);\n }\n\n // GTEQ[] Greater Than or EQual\n // 0x53\n function GTEQ(state) {\n var stack = state.stack;\n var e2 = stack.pop();\n var e1 = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'GTEQ[]', e2, e1); }\n\n stack.push(e1 >= e2 ? 1 : 0);\n }\n\n // EQ[] EQual\n // 0x54\n function EQ(state) {\n var stack = state.stack;\n var e2 = stack.pop();\n var e1 = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'EQ[]', e2, e1); }\n\n stack.push(e2 === e1 ? 1 : 0);\n }\n\n // NEQ[] Not EQual\n // 0x55\n function NEQ(state) {\n var stack = state.stack;\n var e2 = stack.pop();\n var e1 = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'NEQ[]', e2, e1); }\n\n stack.push(e2 !== e1 ? 1 : 0);\n }\n\n // ODD[] ODD\n // 0x56\n function ODD(state) {\n var stack = state.stack;\n var n = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'ODD[]', n); }\n\n stack.push(Math.trunc(n) % 2 ? 1 : 0);\n }\n\n // EVEN[] EVEN\n // 0x57\n function EVEN(state) {\n var stack = state.stack;\n var n = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'EVEN[]', n); }\n\n stack.push(Math.trunc(n) % 2 ? 0 : 1);\n }\n\n // IF[] IF test\n // 0x58\n function IF(state) {\n var test = state.stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'IF[]', test); }\n\n // if test is true it just continues\n // if not the ip is skipped until matching ELSE or EIF\n if (!test) {\n skip(state, true);\n\n if (exports.DEBUG) { console.log(state.step, 'EIF[]'); }\n }\n }\n\n // EIF[] End IF\n // 0x59\n function EIF(state) {\n // this can be reached normally when\n // executing an else branch.\n // -> just ignore it\n\n if (exports.DEBUG) { console.log(state.step, 'EIF[]'); }\n }\n\n // AND[] logical AND\n // 0x5A\n function AND(state) {\n var stack = state.stack;\n var e2 = stack.pop();\n var e1 = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'AND[]', e2, e1); }\n\n stack.push(e2 && e1 ? 1 : 0);\n }\n\n // OR[] logical OR\n // 0x5B\n function OR(state) {\n var stack = state.stack;\n var e2 = stack.pop();\n var e1 = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'OR[]', e2, e1); }\n\n stack.push(e2 || e1 ? 1 : 0);\n }\n\n // NOT[] logical NOT\n // 0x5C\n function NOT(state) {\n var stack = state.stack;\n var e = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'NOT[]', e); }\n\n stack.push(e ? 0 : 1);\n }\n\n // DELTAP1[] DELTA exception P1\n // DELTAP2[] DELTA exception P2\n // DELTAP3[] DELTA exception P3\n // 0x5D, 0x71, 0x72\n function DELTAP123(b, state) {\n var stack = state.stack;\n var n = stack.pop();\n var fv = state.fv;\n var pv = state.pv;\n var ppem = state.ppem;\n var base = state.deltaBase + (b - 1) * 16;\n var ds = state.deltaShift;\n var z0 = state.z0;\n\n if (exports.DEBUG) { console.log(state.step, 'DELTAP[' + b + ']', n, stack); }\n\n for (var i = 0; i < n; i++) {\n var pi = stack.pop();\n var arg = stack.pop();\n var appem = base + ((arg & 0xF0) >> 4);\n if (appem !== ppem) { continue; }\n\n var mag = (arg & 0x0F) - 8;\n if (mag >= 0) { mag++; }\n if (exports.DEBUG) { console.log(state.step, 'DELTAPFIX', pi, 'by', mag * ds); }\n\n var p = z0[pi];\n fv.setRelative(p, p, mag * ds, pv);\n }\n }\n\n // SDB[] Set Delta Base in the graphics state\n // 0x5E\n function SDB(state) {\n var stack = state.stack;\n var n = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'SDB[]', n); }\n\n state.deltaBase = n;\n }\n\n // SDS[] Set Delta Shift in the graphics state\n // 0x5F\n function SDS(state) {\n var stack = state.stack;\n var n = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'SDS[]', n); }\n\n state.deltaShift = Math.pow(0.5, n);\n }\n\n // ADD[] ADD\n // 0x60\n function ADD(state) {\n var stack = state.stack;\n var n2 = stack.pop();\n var n1 = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'ADD[]', n2, n1); }\n\n stack.push(n1 + n2);\n }\n\n // SUB[] SUB\n // 0x61\n function SUB(state) {\n var stack = state.stack;\n var n2 = stack.pop();\n var n1 = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'SUB[]', n2, n1); }\n\n stack.push(n1 - n2);\n }\n\n // DIV[] DIV\n // 0x62\n function DIV(state) {\n var stack = state.stack;\n var n2 = stack.pop();\n var n1 = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'DIV[]', n2, n1); }\n\n stack.push(n1 * 64 / n2);\n }\n\n // MUL[] MUL\n // 0x63\n function MUL(state) {\n var stack = state.stack;\n var n2 = stack.pop();\n var n1 = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'MUL[]', n2, n1); }\n\n stack.push(n1 * n2 / 64);\n }\n\n // ABS[] ABSolute value\n // 0x64\n function ABS(state) {\n var stack = state.stack;\n var n = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'ABS[]', n); }\n\n stack.push(Math.abs(n));\n }\n\n // NEG[] NEGate\n // 0x65\n function NEG(state) {\n var stack = state.stack;\n var n = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'NEG[]', n); }\n\n stack.push(-n);\n }\n\n // FLOOR[] FLOOR\n // 0x66\n function FLOOR(state) {\n var stack = state.stack;\n var n = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'FLOOR[]', n); }\n\n stack.push(Math.floor(n / 0x40) * 0x40);\n }\n\n // CEILING[] CEILING\n // 0x67\n function CEILING(state) {\n var stack = state.stack;\n var n = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'CEILING[]', n); }\n\n stack.push(Math.ceil(n / 0x40) * 0x40);\n }\n\n // ROUND[ab] ROUND value\n // 0x68-0x6B\n function ROUND(dt, state) {\n var stack = state.stack;\n var n = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'ROUND[]'); }\n\n stack.push(state.round(n / 0x40) * 0x40);\n }\n\n // WCVTF[] Write Control Value Table in Funits\n // 0x70\n function WCVTF(state) {\n var stack = state.stack;\n var v = stack.pop();\n var l = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'WCVTF[]', v, l); }\n\n state.cvt[l] = v * state.ppem / state.font.unitsPerEm;\n }\n\n // DELTAC1[] DELTA exception C1\n // DELTAC2[] DELTA exception C2\n // DELTAC3[] DELTA exception C3\n // 0x73, 0x74, 0x75\n function DELTAC123(b, state) {\n var stack = state.stack;\n var n = stack.pop();\n var ppem = state.ppem;\n var base = state.deltaBase + (b - 1) * 16;\n var ds = state.deltaShift;\n\n if (exports.DEBUG) { console.log(state.step, 'DELTAC[' + b + ']', n, stack); }\n\n for (var i = 0; i < n; i++) {\n var c = stack.pop();\n var arg = stack.pop();\n var appem = base + ((arg & 0xF0) >> 4);\n if (appem !== ppem) { continue; }\n\n var mag = (arg & 0x0F) - 8;\n if (mag >= 0) { mag++; }\n\n var delta = mag * ds;\n\n if (exports.DEBUG) { console.log(state.step, 'DELTACFIX', c, 'by', delta); }\n\n state.cvt[c] += delta;\n }\n }\n\n // SROUND[] Super ROUND\n // 0x76\n function SROUND(state) {\n var n = state.stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'SROUND[]', n); }\n\n state.round = roundSuper;\n\n var period;\n\n switch (n & 0xC0) {\n case 0x00:\n period = 0.5;\n break;\n case 0x40:\n period = 1;\n break;\n case 0x80:\n period = 2;\n break;\n default:\n throw new Error('invalid SROUND value');\n }\n\n state.srPeriod = period;\n\n switch (n & 0x30) {\n case 0x00:\n state.srPhase = 0;\n break;\n case 0x10:\n state.srPhase = 0.25 * period;\n break;\n case 0x20:\n state.srPhase = 0.5 * period;\n break;\n case 0x30:\n state.srPhase = 0.75 * period;\n break;\n default: throw new Error('invalid SROUND value');\n }\n\n n &= 0x0F;\n\n if (n === 0) { state.srThreshold = 0; }\n else { state.srThreshold = (n / 8 - 0.5) * period; }\n }\n\n // S45ROUND[] Super ROUND 45 degrees\n // 0x77\n function S45ROUND(state) {\n var n = state.stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'S45ROUND[]', n); }\n\n state.round = roundSuper;\n\n var period;\n\n switch (n & 0xC0) {\n case 0x00:\n period = Math.sqrt(2) / 2;\n break;\n case 0x40:\n period = Math.sqrt(2);\n break;\n case 0x80:\n period = 2 * Math.sqrt(2);\n break;\n default:\n throw new Error('invalid S45ROUND value');\n }\n\n state.srPeriod = period;\n\n switch (n & 0x30) {\n case 0x00:\n state.srPhase = 0;\n break;\n case 0x10:\n state.srPhase = 0.25 * period;\n break;\n case 0x20:\n state.srPhase = 0.5 * period;\n break;\n case 0x30:\n state.srPhase = 0.75 * period;\n break;\n default:\n throw new Error('invalid S45ROUND value');\n }\n\n n &= 0x0F;\n\n if (n === 0) { state.srThreshold = 0; }\n else { state.srThreshold = (n / 8 - 0.5) * period; }\n }\n\n // ROFF[] Round Off\n // 0x7A\n function ROFF(state) {\n if (exports.DEBUG) { console.log(state.step, 'ROFF[]'); }\n\n state.round = roundOff;\n }\n\n // RUTG[] Round Up To Grid\n // 0x7C\n function RUTG(state) {\n if (exports.DEBUG) { console.log(state.step, 'RUTG[]'); }\n\n state.round = roundUpToGrid;\n }\n\n // RDTG[] Round Down To Grid\n // 0x7D\n function RDTG(state) {\n if (exports.DEBUG) { console.log(state.step, 'RDTG[]'); }\n\n state.round = roundDownToGrid;\n }\n\n // SCANCTRL[] SCAN conversion ConTRoL\n // 0x85\n function SCANCTRL(state) {\n var n = state.stack.pop();\n\n // ignored by opentype.js\n\n if (exports.DEBUG) { console.log(state.step, 'SCANCTRL[]', n); }\n }\n\n // SDPVTL[a] Set Dual Projection Vector To Line\n // 0x86-0x87\n function SDPVTL(a, state) {\n var stack = state.stack;\n var p2i = stack.pop();\n var p1i = stack.pop();\n var p2 = state.z2[p2i];\n var p1 = state.z1[p1i];\n\n if (exports.DEBUG) { console.log(state.step, 'SDPVTL[' + a + ']', p2i, p1i); }\n\n var dx;\n var dy;\n\n if (!a) {\n dx = p1.x - p2.x;\n dy = p1.y - p2.y;\n } else {\n dx = p2.y - p1.y;\n dy = p1.x - p2.x;\n }\n\n state.dpv = getUnitVector(dx, dy);\n }\n\n // GETINFO[] GET INFOrmation\n // 0x88\n function GETINFO(state) {\n var stack = state.stack;\n var sel = stack.pop();\n var r = 0;\n\n if (exports.DEBUG) { console.log(state.step, 'GETINFO[]', sel); }\n\n // v35 as in no subpixel hinting\n if (sel & 0x01) { r = 35; }\n\n // TODO rotation and stretch currently not supported\n // and thus those GETINFO are always 0.\n\n // opentype.js is always gray scaling\n if (sel & 0x20) { r |= 0x1000; }\n\n stack.push(r);\n }\n\n // ROLL[] ROLL the top three stack elements\n // 0x8A\n function ROLL(state) {\n var stack = state.stack;\n var a = stack.pop();\n var b = stack.pop();\n var c = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'ROLL[]'); }\n\n stack.push(b);\n stack.push(a);\n stack.push(c);\n }\n\n // MAX[] MAXimum of top two stack elements\n // 0x8B\n function MAX(state) {\n var stack = state.stack;\n var e2 = stack.pop();\n var e1 = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'MAX[]', e2, e1); }\n\n stack.push(Math.max(e1, e2));\n }\n\n // MIN[] MINimum of top two stack elements\n // 0x8C\n function MIN(state) {\n var stack = state.stack;\n var e2 = stack.pop();\n var e1 = stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'MIN[]', e2, e1); }\n\n stack.push(Math.min(e1, e2));\n }\n\n // SCANTYPE[] SCANTYPE\n // 0x8D\n function SCANTYPE(state) {\n var n = state.stack.pop();\n // ignored by opentype.js\n if (exports.DEBUG) { console.log(state.step, 'SCANTYPE[]', n); }\n }\n\n // INSTCTRL[] INSTCTRL\n // 0x8D\n function INSTCTRL(state) {\n var s = state.stack.pop();\n var v = state.stack.pop();\n\n if (exports.DEBUG) { console.log(state.step, 'INSTCTRL[]', s, v); }\n\n switch (s) {\n case 1 : state.inhibitGridFit = !!v; return;\n case 2 : state.ignoreCvt = !!v; return;\n default: throw new Error('invalid INSTCTRL[] selector');\n }\n }\n\n // PUSHB[abc] PUSH Bytes\n // 0xB0-0xB7\n function PUSHB(n, state) {\n var stack = state.stack;\n var prog = state.prog;\n var ip = state.ip;\n\n if (exports.DEBUG) { console.log(state.step, 'PUSHB[' + n + ']'); }\n\n for (var i = 0; i < n; i++) { stack.push(prog[++ip]); }\n\n state.ip = ip;\n }\n\n // PUSHW[abc] PUSH Words\n // 0xB8-0xBF\n function PUSHW(n, state) {\n var ip = state.ip;\n var prog = state.prog;\n var stack = state.stack;\n\n if (exports.DEBUG) { console.log(state.ip, 'PUSHW[' + n + ']'); }\n\n for (var i = 0; i < n; i++) {\n var w = (prog[++ip] << 8) | prog[++ip];\n if (w & 0x8000) { w = -((w ^ 0xffff) + 1); }\n stack.push(w);\n }\n\n state.ip = ip;\n }\n\n // MDRP[abcde] Move Direct Relative Point\n // 0xD0-0xEF\n // (if indirect is 0)\n //\n // and\n //\n // MIRP[abcde] Move Indirect Relative Point\n // 0xE0-0xFF\n // (if indirect is 1)\n\n function MDRP_MIRP(indirect, setRp0, keepD, ro, dt, state) {\n var stack = state.stack;\n var cvte = indirect && stack.pop();\n var pi = stack.pop();\n var rp0i = state.rp0;\n var rp = state.z0[rp0i];\n var p = state.z1[pi];\n\n var md = state.minDis;\n var fv = state.fv;\n var pv = state.dpv;\n var od; // original distance\n var d; // moving distance\n var sign; // sign of distance\n var cv;\n\n d = od = pv.distance(p, rp, true, true);\n sign = d >= 0 ? 1 : -1; // Math.sign would be 0 in case of 0\n\n // TODO consider autoFlip\n d = Math.abs(d);\n\n if (indirect) {\n cv = state.cvt[cvte];\n\n if (ro && Math.abs(d - cv) < state.cvCutIn) { d = cv; }\n }\n\n if (keepD && d < md) { d = md; }\n\n if (ro) { d = state.round(d); }\n\n fv.setRelative(p, rp, sign * d, pv);\n fv.touch(p);\n\n if (exports.DEBUG) {\n console.log(\n state.step,\n (indirect ? 'MIRP[' : 'MDRP[') +\n (setRp0 ? 'M' : 'm') +\n (keepD ? '>' : '_') +\n (ro ? 'R' : '_') +\n (dt === 0 ? 'Gr' : (dt === 1 ? 'Bl' : (dt === 2 ? 'Wh' : ''))) +\n ']',\n indirect ?\n cvte + '(' + state.cvt[cvte] + ',' + cv + ')' :\n '',\n pi,\n '(d =', od, '->', sign * d, ')'\n );\n }\n\n state.rp1 = state.rp0;\n state.rp2 = pi;\n if (setRp0) { state.rp0 = pi; }\n }\n\n /*\n * The instruction table.\n */\n instructionTable = [\n /* 0x00 */ SVTCA.bind(undefined, yUnitVector),\n /* 0x01 */ SVTCA.bind(undefined, xUnitVector),\n /* 0x02 */ SPVTCA.bind(undefined, yUnitVector),\n /* 0x03 */ SPVTCA.bind(undefined, xUnitVector),\n /* 0x04 */ SFVTCA.bind(undefined, yUnitVector),\n /* 0x05 */ SFVTCA.bind(undefined, xUnitVector),\n /* 0x06 */ SPVTL.bind(undefined, 0),\n /* 0x07 */ SPVTL.bind(undefined, 1),\n /* 0x08 */ SFVTL.bind(undefined, 0),\n /* 0x09 */ SFVTL.bind(undefined, 1),\n /* 0x0A */ SPVFS,\n /* 0x0B */ SFVFS,\n /* 0x0C */ GPV,\n /* 0x0D */ GFV,\n /* 0x0E */ SFVTPV,\n /* 0x0F */ ISECT,\n /* 0x10 */ SRP0,\n /* 0x11 */ SRP1,\n /* 0x12 */ SRP2,\n /* 0x13 */ SZP0,\n /* 0x14 */ SZP1,\n /* 0x15 */ SZP2,\n /* 0x16 */ SZPS,\n /* 0x17 */ SLOOP,\n /* 0x18 */ RTG,\n /* 0x19 */ RTHG,\n /* 0x1A */ SMD,\n /* 0x1B */ ELSE,\n /* 0x1C */ JMPR,\n /* 0x1D */ SCVTCI,\n /* 0x1E */ undefined, // TODO SSWCI\n /* 0x1F */ undefined, // TODO SSW\n /* 0x20 */ DUP,\n /* 0x21 */ POP,\n /* 0x22 */ CLEAR,\n /* 0x23 */ SWAP,\n /* 0x24 */ DEPTH,\n /* 0x25 */ CINDEX,\n /* 0x26 */ MINDEX,\n /* 0x27 */ undefined, // TODO ALIGNPTS\n /* 0x28 */ undefined,\n /* 0x29 */ undefined, // TODO UTP\n /* 0x2A */ LOOPCALL,\n /* 0x2B */ CALL,\n /* 0x2C */ FDEF,\n /* 0x2D */ undefined, // ENDF (eaten by FDEF)\n /* 0x2E */ MDAP.bind(undefined, 0),\n /* 0x2F */ MDAP.bind(undefined, 1),\n /* 0x30 */ IUP.bind(undefined, yUnitVector),\n /* 0x31 */ IUP.bind(undefined, xUnitVector),\n /* 0x32 */ SHP.bind(undefined, 0),\n /* 0x33 */ SHP.bind(undefined, 1),\n /* 0x34 */ SHC.bind(undefined, 0),\n /* 0x35 */ SHC.bind(undefined, 1),\n /* 0x36 */ SHZ.bind(undefined, 0),\n /* 0x37 */ SHZ.bind(undefined, 1),\n /* 0x38 */ SHPIX,\n /* 0x39 */ IP,\n /* 0x3A */ MSIRP.bind(undefined, 0),\n /* 0x3B */ MSIRP.bind(undefined, 1),\n /* 0x3C */ ALIGNRP,\n /* 0x3D */ RTDG,\n /* 0x3E */ MIAP.bind(undefined, 0),\n /* 0x3F */ MIAP.bind(undefined, 1),\n /* 0x40 */ NPUSHB,\n /* 0x41 */ NPUSHW,\n /* 0x42 */ WS,\n /* 0x43 */ RS,\n /* 0x44 */ WCVTP,\n /* 0x45 */ RCVT,\n /* 0x46 */ GC.bind(undefined, 0),\n /* 0x47 */ GC.bind(undefined, 1),\n /* 0x48 */ undefined, // TODO SCFS\n /* 0x49 */ MD.bind(undefined, 0),\n /* 0x4A */ MD.bind(undefined, 1),\n /* 0x4B */ MPPEM,\n /* 0x4C */ undefined, // TODO MPS\n /* 0x4D */ FLIPON,\n /* 0x4E */ undefined, // TODO FLIPOFF\n /* 0x4F */ undefined, // TODO DEBUG\n /* 0x50 */ LT,\n /* 0x51 */ LTEQ,\n /* 0x52 */ GT,\n /* 0x53 */ GTEQ,\n /* 0x54 */ EQ,\n /* 0x55 */ NEQ,\n /* 0x56 */ ODD,\n /* 0x57 */ EVEN,\n /* 0x58 */ IF,\n /* 0x59 */ EIF,\n /* 0x5A */ AND,\n /* 0x5B */ OR,\n /* 0x5C */ NOT,\n /* 0x5D */ DELTAP123.bind(undefined, 1),\n /* 0x5E */ SDB,\n /* 0x5F */ SDS,\n /* 0x60 */ ADD,\n /* 0x61 */ SUB,\n /* 0x62 */ DIV,\n /* 0x63 */ MUL,\n /* 0x64 */ ABS,\n /* 0x65 */ NEG,\n /* 0x66 */ FLOOR,\n /* 0x67 */ CEILING,\n /* 0x68 */ ROUND.bind(undefined, 0),\n /* 0x69 */ ROUND.bind(undefined, 1),\n /* 0x6A */ ROUND.bind(undefined, 2),\n /* 0x6B */ ROUND.bind(undefined, 3),\n /* 0x6C */ undefined, // TODO NROUND[ab]\n /* 0x6D */ undefined, // TODO NROUND[ab]\n /* 0x6E */ undefined, // TODO NROUND[ab]\n /* 0x6F */ undefined, // TODO NROUND[ab]\n /* 0x70 */ WCVTF,\n /* 0x71 */ DELTAP123.bind(undefined, 2),\n /* 0x72 */ DELTAP123.bind(undefined, 3),\n /* 0x73 */ DELTAC123.bind(undefined, 1),\n /* 0x74 */ DELTAC123.bind(undefined, 2),\n /* 0x75 */ DELTAC123.bind(undefined, 3),\n /* 0x76 */ SROUND,\n /* 0x77 */ S45ROUND,\n /* 0x78 */ undefined, // TODO JROT[]\n /* 0x79 */ undefined, // TODO JROF[]\n /* 0x7A */ ROFF,\n /* 0x7B */ undefined,\n /* 0x7C */ RUTG,\n /* 0x7D */ RDTG,\n /* 0x7E */ POP, // actually SANGW, supposed to do only a pop though\n /* 0x7F */ POP, // actually AA, supposed to do only a pop though\n /* 0x80 */ undefined, // TODO FLIPPT\n /* 0x81 */ undefined, // TODO FLIPRGON\n /* 0x82 */ undefined, // TODO FLIPRGOFF\n /* 0x83 */ undefined,\n /* 0x84 */ undefined,\n /* 0x85 */ SCANCTRL,\n /* 0x86 */ SDPVTL.bind(undefined, 0),\n /* 0x87 */ SDPVTL.bind(undefined, 1),\n /* 0x88 */ GETINFO,\n /* 0x89 */ undefined, // TODO IDEF\n /* 0x8A */ ROLL,\n /* 0x8B */ MAX,\n /* 0x8C */ MIN,\n /* 0x8D */ SCANTYPE,\n /* 0x8E */ INSTCTRL,\n /* 0x8F */ undefined,\n /* 0x90 */ undefined,\n /* 0x91 */ undefined,\n /* 0x92 */ undefined,\n /* 0x93 */ undefined,\n /* 0x94 */ undefined,\n /* 0x95 */ undefined,\n /* 0x96 */ undefined,\n /* 0x97 */ undefined,\n /* 0x98 */ undefined,\n /* 0x99 */ undefined,\n /* 0x9A */ undefined,\n /* 0x9B */ undefined,\n /* 0x9C */ undefined,\n /* 0x9D */ undefined,\n /* 0x9E */ undefined,\n /* 0x9F */ undefined,\n /* 0xA0 */ undefined,\n /* 0xA1 */ undefined,\n /* 0xA2 */ undefined,\n /* 0xA3 */ undefined,\n /* 0xA4 */ undefined,\n /* 0xA5 */ undefined,\n /* 0xA6 */ undefined,\n /* 0xA7 */ undefined,\n /* 0xA8 */ undefined,\n /* 0xA9 */ undefined,\n /* 0xAA */ undefined,\n /* 0xAB */ undefined,\n /* 0xAC */ undefined,\n /* 0xAD */ undefined,\n /* 0xAE */ undefined,\n /* 0xAF */ undefined,\n /* 0xB0 */ PUSHB.bind(undefined, 1),\n /* 0xB1 */ PUSHB.bind(undefined, 2),\n /* 0xB2 */ PUSHB.bind(undefined, 3),\n /* 0xB3 */ PUSHB.bind(undefined, 4),\n /* 0xB4 */ PUSHB.bind(undefined, 5),\n /* 0xB5 */ PUSHB.bind(undefined, 6),\n /* 0xB6 */ PUSHB.bind(undefined, 7),\n /* 0xB7 */ PUSHB.bind(undefined, 8),\n /* 0xB8 */ PUSHW.bind(undefined, 1),\n /* 0xB9 */ PUSHW.bind(undefined, 2),\n /* 0xBA */ PUSHW.bind(undefined, 3),\n /* 0xBB */ PUSHW.bind(undefined, 4),\n /* 0xBC */ PUSHW.bind(undefined, 5),\n /* 0xBD */ PUSHW.bind(undefined, 6),\n /* 0xBE */ PUSHW.bind(undefined, 7),\n /* 0xBF */ PUSHW.bind(undefined, 8),\n /* 0xC0 */ MDRP_MIRP.bind(undefined, 0, 0, 0, 0, 0),\n /* 0xC1 */ MDRP_MIRP.bind(undefined, 0, 0, 0, 0, 1),\n /* 0xC2 */ MDRP_MIRP.bind(undefined, 0, 0, 0, 0, 2),\n /* 0xC3 */ MDRP_MIRP.bind(undefined, 0, 0, 0, 0, 3),\n /* 0xC4 */ MDRP_MIRP.bind(undefined, 0, 0, 0, 1, 0),\n /* 0xC5 */ MDRP_MIRP.bind(undefined, 0, 0, 0, 1, 1),\n /* 0xC6 */ MDRP_MIRP.bind(undefined, 0, 0, 0, 1, 2),\n /* 0xC7 */ MDRP_MIRP.bind(undefined, 0, 0, 0, 1, 3),\n /* 0xC8 */ MDRP_MIRP.bind(undefined, 0, 0, 1, 0, 0),\n /* 0xC9 */ MDRP_MIRP.bind(undefined, 0, 0, 1, 0, 1),\n /* 0xCA */ MDRP_MIRP.bind(undefined, 0, 0, 1, 0, 2),\n /* 0xCB */ MDRP_MIRP.bind(undefined, 0, 0, 1, 0, 3),\n /* 0xCC */ MDRP_MIRP.bind(undefined, 0, 0, 1, 1, 0),\n /* 0xCD */ MDRP_MIRP.bind(undefined, 0, 0, 1, 1, 1),\n /* 0xCE */ MDRP_MIRP.bind(undefined, 0, 0, 1, 1, 2),\n /* 0xCF */ MDRP_MIRP.bind(undefined, 0, 0, 1, 1, 3),\n /* 0xD0 */ MDRP_MIRP.bind(undefined, 0, 1, 0, 0, 0),\n /* 0xD1 */ MDRP_MIRP.bind(undefined, 0, 1, 0, 0, 1),\n /* 0xD2 */ MDRP_MIRP.bind(undefined, 0, 1, 0, 0, 2),\n /* 0xD3 */ MDRP_MIRP.bind(undefined, 0, 1, 0, 0, 3),\n /* 0xD4 */ MDRP_MIRP.bind(undefined, 0, 1, 0, 1, 0),\n /* 0xD5 */ MDRP_MIRP.bind(undefined, 0, 1, 0, 1, 1),\n /* 0xD6 */ MDRP_MIRP.bind(undefined, 0, 1, 0, 1, 2),\n /* 0xD7 */ MDRP_MIRP.bind(undefined, 0, 1, 0, 1, 3),\n /* 0xD8 */ MDRP_MIRP.bind(undefined, 0, 1, 1, 0, 0),\n /* 0xD9 */ MDRP_MIRP.bind(undefined, 0, 1, 1, 0, 1),\n /* 0xDA */ MDRP_MIRP.bind(undefined, 0, 1, 1, 0, 2),\n /* 0xDB */ MDRP_MIRP.bind(undefined, 0, 1, 1, 0, 3),\n /* 0xDC */ MDRP_MIRP.bind(undefined, 0, 1, 1, 1, 0),\n /* 0xDD */ MDRP_MIRP.bind(undefined, 0, 1, 1, 1, 1),\n /* 0xDE */ MDRP_MIRP.bind(undefined, 0, 1, 1, 1, 2),\n /* 0xDF */ MDRP_MIRP.bind(undefined, 0, 1, 1, 1, 3),\n /* 0xE0 */ MDRP_MIRP.bind(undefined, 1, 0, 0, 0, 0),\n /* 0xE1 */ MDRP_MIRP.bind(undefined, 1, 0, 0, 0, 1),\n /* 0xE2 */ MDRP_MIRP.bind(undefined, 1, 0, 0, 0, 2),\n /* 0xE3 */ MDRP_MIRP.bind(undefined, 1, 0, 0, 0, 3),\n /* 0xE4 */ MDRP_MIRP.bind(undefined, 1, 0, 0, 1, 0),\n /* 0xE5 */ MDRP_MIRP.bind(undefined, 1, 0, 0, 1, 1),\n /* 0xE6 */ MDRP_MIRP.bind(undefined, 1, 0, 0, 1, 2),\n /* 0xE7 */ MDRP_MIRP.bind(undefined, 1, 0, 0, 1, 3),\n /* 0xE8 */ MDRP_MIRP.bind(undefined, 1, 0, 1, 0, 0),\n /* 0xE9 */ MDRP_MIRP.bind(undefined, 1, 0, 1, 0, 1),\n /* 0xEA */ MDRP_MIRP.bind(undefined, 1, 0, 1, 0, 2),\n /* 0xEB */ MDRP_MIRP.bind(undefined, 1, 0, 1, 0, 3),\n /* 0xEC */ MDRP_MIRP.bind(undefined, 1, 0, 1, 1, 0),\n /* 0xED */ MDRP_MIRP.bind(undefined, 1, 0, 1, 1, 1),\n /* 0xEE */ MDRP_MIRP.bind(undefined, 1, 0, 1, 1, 2),\n /* 0xEF */ MDRP_MIRP.bind(undefined, 1, 0, 1, 1, 3),\n /* 0xF0 */ MDRP_MIRP.bind(undefined, 1, 1, 0, 0, 0),\n /* 0xF1 */ MDRP_MIRP.bind(undefined, 1, 1, 0, 0, 1),\n /* 0xF2 */ MDRP_MIRP.bind(undefined, 1, 1, 0, 0, 2),\n /* 0xF3 */ MDRP_MIRP.bind(undefined, 1, 1, 0, 0, 3),\n /* 0xF4 */ MDRP_MIRP.bind(undefined, 1, 1, 0, 1, 0),\n /* 0xF5 */ MDRP_MIRP.bind(undefined, 1, 1, 0, 1, 1),\n /* 0xF6 */ MDRP_MIRP.bind(undefined, 1, 1, 0, 1, 2),\n /* 0xF7 */ MDRP_MIRP.bind(undefined, 1, 1, 0, 1, 3),\n /* 0xF8 */ MDRP_MIRP.bind(undefined, 1, 1, 1, 0, 0),\n /* 0xF9 */ MDRP_MIRP.bind(undefined, 1, 1, 1, 0, 1),\n /* 0xFA */ MDRP_MIRP.bind(undefined, 1, 1, 1, 0, 2),\n /* 0xFB */ MDRP_MIRP.bind(undefined, 1, 1, 1, 0, 3),\n /* 0xFC */ MDRP_MIRP.bind(undefined, 1, 1, 1, 1, 0),\n /* 0xFD */ MDRP_MIRP.bind(undefined, 1, 1, 1, 1, 1),\n /* 0xFE */ MDRP_MIRP.bind(undefined, 1, 1, 1, 1, 2),\n /* 0xFF */ MDRP_MIRP.bind(undefined, 1, 1, 1, 1, 3)\n ];\n\n /*****************************\n Mathematical Considerations\n ******************************\n\n fv ... refers to freedom vector\n pv ... refers to projection vector\n rp ... refers to reference point\n p ... refers to to point being operated on\n d ... refers to distance\n\n SETRELATIVE:\n ============\n\n case freedom vector == x-axis:\n ------------------------------\n\n (pv)\n .-'\n rpd .-'\n .-*\n d .-'90°'\n .-' '\n .-' '\n *-' ' b\n rp '\n '\n '\n p *----------*-------------- (fv)\n pm\n\n rpdx = rpx + d * pv.x\n rpdy = rpy + d * pv.y\n\n equation of line b\n\n y - rpdy = pvns * (x- rpdx)\n\n y = p.y\n\n x = rpdx + ( p.y - rpdy ) / pvns\n\n\n case freedom vector == y-axis:\n ------------------------------\n\n * pm\n |\\\n | \\\n | \\\n | \\\n | \\\n | \\\n | \\\n | \\\n | \\\n | \\ b\n | \\\n | \\\n | \\ .-' (pv)\n | 90° \\.-'\n | .-'* rpd\n | .-'\n * *-' d\n p rp\n\n rpdx = rpx + d * pv.x\n rpdy = rpy + d * pv.y\n\n equation of line b:\n pvns ... normal slope to pv\n\n y - rpdy = pvns * (x - rpdx)\n\n x = p.x\n\n y = rpdy + pvns * (p.x - rpdx)\n\n\n\n generic case:\n -------------\n\n\n .'(fv)\n .'\n .* pm\n .' !\n .' .\n .' !\n .' . b\n .' !\n * .\n p !\n 90° . ... (pv)\n ...-*-'''\n ...---''' rpd\n ...---''' d\n *--'''\n rp\n\n rpdx = rpx + d * pv.x\n rpdy = rpy + d * pv.y\n\n equation of line b:\n pvns... normal slope to pv\n\n y - rpdy = pvns * (x - rpdx)\n\n equation of freedom vector line:\n fvs ... slope of freedom vector (=fy/fx)\n\n y - py = fvs * (x - px)\n\n\n on pm both equations are true for same x/y\n\n y - rpdy = pvns * (x - rpdx)\n\n y - py = fvs * (x - px)\n\n form to y and set equal:\n\n pvns * (x - rpdx) + rpdy = fvs * (x - px) + py\n\n expand:\n\n pvns * x - pvns * rpdx + rpdy = fvs * x - fvs * px + py\n\n switch:\n\n fvs * x - fvs * px + py = pvns * x - pvns * rpdx + rpdy\n\n solve for x:\n\n fvs * x - pvns * x = fvs * px - pvns * rpdx - py + rpdy\n\n\n\n fvs * px - pvns * rpdx + rpdy - py\n x = -----------------------------------\n fvs - pvns\n\n and:\n\n y = fvs * (x - px) + py\n\n\n\n INTERPOLATE:\n ============\n\n Examples of point interpolation.\n\n The weight of the movement of the reference point gets bigger\n the further the other reference point is away, thus the safest\n option (that is avoiding 0/0 divisions) is to weight the\n original distance of the other point by the sum of both distances.\n\n If the sum of both distances is 0, then move the point by the\n arithmetic average of the movement of both reference points.\n\n\n\n\n (+6)\n rp1o *---->*rp1\n . . (+12)\n . . rp2o *---------->* rp2\n . . . .\n . . . .\n . 10 20 . .\n |.........|...................| .\n . . .\n . . (+8) .\n po *------>*p .\n . . .\n . 12 . 24 .\n |...........|.......................|\n 36\n\n\n -------\n\n\n\n (+10)\n rp1o *-------->*rp1\n . . (-10)\n . . rp2 *<---------* rpo2\n . . . .\n . . . .\n . 10 . 30 . .\n |.........|.............................|\n . .\n . (+5) .\n po *--->* p .\n . . .\n . . 20 .\n |....|..............|\n 5 15\n\n\n -------\n\n\n (+10)\n rp1o *-------->*rp1\n . .\n . .\n rp2o *-------->*rp2\n\n\n (+10)\n po *-------->* p\n\n -------\n\n\n (+10)\n rp1o *-------->*rp1\n . .\n . .(+30)\n rp2o *---------------------------->*rp2\n\n\n (+25)\n po *----------------------->* p\n\n\n\n vim: set ts=4 sw=4 expandtab:\n *****/\n\n /**\n * Converts a string into a list of tokens.\n */\n\n /**\n * Create a new token\n * @param {string} char a single char\n */\n function Token$1(char) {\n this.char = char;\n this.state = {};\n this.activeState = null;\n }\n\n /**\n * Create a new context range\n * @param {number} startIndex range start index\n * @param {number} endOffset range end index offset\n * @param {string} contextName owner context name\n */\n function ContextRange(startIndex, endOffset, contextName) {\n this.contextName = contextName;\n this.startIndex = startIndex;\n this.endOffset = endOffset;\n }\n\n /**\n * Check context start and end\n * @param {string} contextName a unique context name\n * @param {function} checkStart a predicate function the indicates a context's start\n * @param {function} checkEnd a predicate function the indicates a context's end\n */\n function ContextChecker(contextName, checkStart, checkEnd) {\n this.contextName = contextName;\n this.openRange = null;\n this.ranges = [];\n this.checkStart = checkStart;\n this.checkEnd = checkEnd;\n }\n\n /**\n * @typedef ContextParams\n * @type Object\n * @property {array} context context items\n * @property {number} currentIndex current item index\n */\n\n /**\n * Create a context params\n * @param {array} context a list of items\n * @param {number} currentIndex current item index\n */\n function ContextParams(context, currentIndex) {\n this.context = context;\n this.index = currentIndex;\n this.length = context.length;\n this.current = context[currentIndex];\n this.backtrack = context.slice(0, currentIndex);\n this.lookahead = context.slice(currentIndex + 1);\n }\n\n /**\n * Create an event instance\n * @param {string} eventId event unique id\n */\n function Event(eventId) {\n this.eventId = eventId;\n this.subscribers = [];\n }\n\n /**\n * Initialize a core events and auto subscribe required event handlers\n * @param {any} events an object that enlists core events handlers\n */\n function initializeCoreEvents(events) {\n var this$1 = this;\n\n var coreEvents = [\n 'start', 'end', 'next', 'newToken', 'contextStart',\n 'contextEnd', 'insertToken', 'removeToken', 'removeRange',\n 'replaceToken', 'replaceRange', 'composeRUD', 'updateContextsRanges'\n ];\n\n coreEvents.forEach(function (eventId) {\n Object.defineProperty(this$1.events, eventId, {\n value: new Event(eventId)\n });\n });\n\n if (!!events) {\n coreEvents.forEach(function (eventId) {\n var event = events[eventId];\n if (typeof event === 'function') {\n this$1.events[eventId].subscribe(event);\n }\n });\n }\n var requiresContextUpdate = [\n 'insertToken', 'removeToken', 'removeRange',\n 'replaceToken', 'replaceRange', 'composeRUD'\n ];\n requiresContextUpdate.forEach(function (eventId) {\n this$1.events[eventId].subscribe(\n this$1.updateContextsRanges\n );\n });\n }\n\n /**\n * Converts a string into a list of tokens\n * @param {any} events tokenizer core events\n */\n function Tokenizer(events) {\n this.tokens = [];\n this.registeredContexts = {};\n this.contextCheckers = [];\n this.events = {};\n this.registeredModifiers = [];\n\n initializeCoreEvents.call(this, events);\n }\n\n /**\n * Sets the state of a token, usually called by a state modifier.\n * @param {string} key state item key\n * @param {any} value state item value\n */\n Token$1.prototype.setState = function(key, value) {\n this.state[key] = value;\n this.activeState = { key: key, value: this.state[key] };\n return this.activeState;\n };\n\n Token$1.prototype.getState = function (stateId) {\n return this.state[stateId] || null;\n };\n\n /**\n * Checks if an index exists in the tokens list.\n * @param {number} index token index\n */\n Tokenizer.prototype.inboundIndex = function(index) {\n return index >= 0 && index < this.tokens.length;\n };\n\n /**\n * Compose and apply a list of operations (replace, update, delete)\n * @param {array} RUDs replace, update and delete operations\n * TODO: Perf. Optimization (lengthBefore === lengthAfter ? dispatch once)\n */\n Tokenizer.prototype.composeRUD = function (RUDs) {\n var this$1 = this;\n\n var silent = true;\n var state = RUDs.map(function (RUD) { return (\n this$1[RUD[0]].apply(this$1, RUD.slice(1).concat(silent))\n ); });\n var hasFAILObject = function (obj) { return (\n typeof obj === 'object' &&\n obj.hasOwnProperty('FAIL')\n ); };\n if (state.every(hasFAILObject)) {\n return {\n FAIL: \"composeRUD: one or more operations hasn't completed successfully\",\n report: state.filter(hasFAILObject)\n };\n }\n this.dispatch('composeRUD', [state.filter(function (op) { return !hasFAILObject(op); })]);\n };\n\n /**\n * Replace a range of tokens with a list of tokens\n * @param {number} startIndex range start index\n * @param {number} offset range offset\n * @param {token} tokens a list of tokens to replace\n * @param {boolean} silent dispatch events and update context ranges\n */\n Tokenizer.prototype.replaceRange = function (startIndex, offset, tokens, silent) {\n offset = offset !== null ? offset : this.tokens.length;\n var isTokenType = tokens.every(function (token) { return token instanceof Token$1; });\n if (!isNaN(startIndex) && this.inboundIndex(startIndex) && isTokenType) {\n var replaced = this.tokens.splice.apply(\n this.tokens, [startIndex, offset].concat(tokens)\n );\n if (!silent) { this.dispatch('replaceToken', [startIndex, offset, tokens]); }\n return [replaced, tokens];\n } else {\n return { FAIL: 'replaceRange: invalid tokens or startIndex.' };\n }\n };\n\n /**\n * Replace a token with another token\n * @param {number} index token index\n * @param {token} token a token to replace\n * @param {boolean} silent dispatch events and update context ranges\n */\n Tokenizer.prototype.replaceToken = function (index, token, silent) {\n if (!isNaN(index) && this.inboundIndex(index) && token instanceof Token$1) {\n var replaced = this.tokens.splice(index, 1, token);\n if (!silent) { this.dispatch('replaceToken', [index, token]); }\n return [replaced[0], token];\n } else {\n return { FAIL: 'replaceToken: invalid token or index.' };\n }\n };\n\n /**\n * Removes a range of tokens\n * @param {number} startIndex range start index\n * @param {number} offset range offset\n * @param {boolean} silent dispatch events and update context ranges\n */\n Tokenizer.prototype.removeRange = function(startIndex, offset, silent) {\n offset = !isNaN(offset) ? offset : this.tokens.length;\n var tokens = this.tokens.splice(startIndex, offset);\n if (!silent) { this.dispatch('removeRange', [tokens, startIndex, offset]); }\n return tokens;\n };\n\n /**\n * Remove a token at a certain index\n * @param {number} index token index\n * @param {boolean} silent dispatch events and update context ranges\n */\n Tokenizer.prototype.removeToken = function(index, silent) {\n if (!isNaN(index) && this.inboundIndex(index)) {\n var token = this.tokens.splice(index, 1);\n if (!silent) { this.dispatch('removeToken', [token, index]); }\n return token;\n } else {\n return { FAIL: 'removeToken: invalid token index.' };\n }\n };\n\n /**\n * Insert a list of tokens at a certain index\n * @param {array} tokens a list of tokens to insert\n * @param {number} index insert the list of tokens at index\n * @param {boolean} silent dispatch events and update context ranges\n */\n Tokenizer.prototype.insertToken = function (tokens, index, silent) {\n var tokenType = tokens.every(\n function (token) { return token instanceof Token$1; }\n );\n if (tokenType) {\n this.tokens.splice.apply(\n this.tokens, [index, 0].concat(tokens)\n );\n if (!silent) { this.dispatch('insertToken', [tokens, index]); }\n return tokens;\n } else {\n return { FAIL: 'insertToken: invalid token(s).' };\n }\n };\n\n /**\n * A state modifier that is called on 'newToken' event\n * @param {string} modifierId state modifier id\n * @param {function} condition a predicate function that returns true or false\n * @param {function} modifier a function to update token state\n */\n Tokenizer.prototype.registerModifier = function(modifierId, condition, modifier) {\n this.events.newToken.subscribe(function(token, contextParams) {\n var conditionParams = [token, contextParams];\n var canApplyModifier = (\n condition === null ||\n condition.apply(this, conditionParams) === true\n );\n var modifierParams = [token, contextParams];\n if (canApplyModifier) {\n var newStateValue = modifier.apply(this, modifierParams);\n token.setState(modifierId, newStateValue);\n }\n });\n this.registeredModifiers.push(modifierId);\n };\n\n /**\n * Subscribe a handler to an event\n * @param {function} eventHandler an event handler function\n */\n Event.prototype.subscribe = function (eventHandler) {\n if (typeof eventHandler === 'function') {\n return ((this.subscribers.push(eventHandler)) - 1);\n } else {\n return { FAIL: (\"invalid '\" + (this.eventId) + \"' event handler\")};\n }\n };\n\n /**\n * Unsubscribe an event handler\n * @param {string} subsId subscription id\n */\n Event.prototype.unsubscribe = function (subsId) {\n this.subscribers.splice(subsId, 1);\n };\n\n /**\n * Sets context params current value index\n * @param {number} index context params current value index\n */\n ContextParams.prototype.setCurrentIndex = function(index) {\n this.index = index;\n this.current = this.context[index];\n this.backtrack = this.context.slice(0, index);\n this.lookahead = this.context.slice(index + 1);\n };\n\n /**\n * Get an item at an offset from the current value\n * example (current value is 3):\n * 1 2 [3] 4 5 | items values\n * -2 -1 0 1 2 | offset values\n * @param {number} offset an offset from current value index\n */\n ContextParams.prototype.get = function (offset) {\n switch (true) {\n case (offset === 0):\n return this.current;\n case (offset < 0 && Math.abs(offset) <= this.backtrack.length):\n return this.backtrack.slice(offset)[0];\n case (offset > 0 && offset <= this.lookahead.length):\n return this.lookahead[offset - 1];\n default:\n return null;\n }\n };\n\n /**\n * Converts a context range into a string value\n * @param {contextRange} range a context range\n */\n Tokenizer.prototype.rangeToText = function (range) {\n if (range instanceof ContextRange) {\n return (\n this.getRangeTokens(range)\n .map(function (token) { return token.char; }).join('')\n );\n }\n };\n\n /**\n * Converts all tokens into a string\n */\n Tokenizer.prototype.getText = function () {\n return this.tokens.map(function (token) { return token.char; }).join('');\n };\n\n /**\n * Get a context by name\n * @param {string} contextName context name to get\n */\n Tokenizer.prototype.getContext = function (contextName) {\n var context = this.registeredContexts[contextName];\n return !!context ? context : null;\n };\n\n /**\n * Subscribes a new event handler to an event\n * @param {string} eventName event name to subscribe to\n * @param {function} eventHandler a function to be invoked on event\n */\n Tokenizer.prototype.on = function(eventName, eventHandler) {\n var event = this.events[eventName];\n if (!!event) {\n return event.subscribe(eventHandler);\n } else {\n return null;\n }\n };\n\n /**\n * Dispatches an event\n * @param {string} eventName event name\n * @param {any} args event handler arguments\n */\n Tokenizer.prototype.dispatch = function(eventName, args) {\n var this$1 = this;\n\n var event = this.events[eventName];\n if (event instanceof Event) {\n event.subscribers.forEach(function (subscriber) {\n subscriber.apply(this$1, args || []);\n });\n }\n };\n\n /**\n * Register a new context checker\n * @param {string} contextName a unique context name\n * @param {function} contextStartCheck a predicate function that returns true on context start\n * @param {function} contextEndCheck a predicate function that returns true on context end\n * TODO: call tokenize on registration to update context ranges with the new context.\n */\n Tokenizer.prototype.registerContextChecker = function(contextName, contextStartCheck, contextEndCheck) {\n if (!!this.getContext(contextName)) { return {\n FAIL:\n (\"context name '\" + contextName + \"' is already registered.\")\n }; }\n if (typeof contextStartCheck !== 'function') { return {\n FAIL:\n \"missing context start check.\"\n }; }\n if (typeof contextEndCheck !== 'function') { return {\n FAIL:\n \"missing context end check.\"\n }; }\n var contextCheckers = new ContextChecker(\n contextName, contextStartCheck, contextEndCheck\n );\n this.registeredContexts[contextName] = contextCheckers;\n this.contextCheckers.push(contextCheckers);\n return contextCheckers;\n };\n\n /**\n * Gets a context range tokens\n * @param {contextRange} range a context range\n */\n Tokenizer.prototype.getRangeTokens = function(range) {\n var endIndex = range.startIndex + range.endOffset;\n return [].concat(\n this.tokens\n .slice(range.startIndex, endIndex)\n );\n };\n\n /**\n * Gets the ranges of a context\n * @param {string} contextName context name\n */\n Tokenizer.prototype.getContextRanges = function(contextName) {\n var context = this.getContext(contextName);\n if (!!context) {\n return context.ranges;\n } else {\n return { FAIL: (\"context checker '\" + contextName + \"' is not registered.\") };\n }\n };\n\n /**\n * Resets context ranges to run context update\n */\n Tokenizer.prototype.resetContextsRanges = function () {\n var registeredContexts = this.registeredContexts;\n for (var contextName in registeredContexts) {\n if (registeredContexts.hasOwnProperty(contextName)) {\n var context = registeredContexts[contextName];\n context.ranges = [];\n }\n }\n };\n\n /**\n * Updates context ranges\n */\n Tokenizer.prototype.updateContextsRanges = function () {\n this.resetContextsRanges();\n var chars = this.tokens.map(function (token) { return token.char; });\n for (var i = 0; i < chars.length; i++) {\n var contextParams = new ContextParams(chars, i);\n this.runContextCheck(contextParams);\n }\n this.dispatch('updateContextsRanges', [this.registeredContexts]);\n };\n\n /**\n * Sets the end offset of an open range\n * @param {number} offset range end offset\n * @param {string} contextName context name\n */\n Tokenizer.prototype.setEndOffset = function (offset, contextName) {\n var startIndex = this.getContext(contextName).openRange.startIndex;\n var range = new ContextRange(startIndex, offset, contextName);\n var ranges = this.getContext(contextName).ranges;\n range.rangeId = contextName + \".\" + (ranges.length);\n ranges.push(range);\n this.getContext(contextName).openRange = null;\n return range;\n };\n\n /**\n * Runs a context check on the current context\n * @param {contextParams} contextParams current context params\n */\n Tokenizer.prototype.runContextCheck = function(contextParams) {\n var this$1 = this;\n\n var index = contextParams.index;\n this.contextCheckers.forEach(function (contextChecker) {\n var contextName = contextChecker.contextName;\n var openRange = this$1.getContext(contextName).openRange;\n if (!openRange && contextChecker.checkStart(contextParams)) {\n openRange = new ContextRange(index, null, contextName);\n this$1.getContext(contextName).openRange = openRange;\n this$1.dispatch('contextStart', [contextName, index]);\n }\n if (!!openRange && contextChecker.checkEnd(contextParams)) {\n var offset = (index - openRange.startIndex) + 1;\n var range = this$1.setEndOffset(offset, contextName);\n this$1.dispatch('contextEnd', [contextName, range]);\n }\n });\n };\n\n /**\n * Converts a text into a list of tokens\n * @param {string} text a text to tokenize\n */\n Tokenizer.prototype.tokenize = function (text) {\n this.tokens = [];\n this.resetContextsRanges();\n var chars = Array.from(text);\n this.dispatch('start');\n for (var i = 0; i < chars.length; i++) {\n var char = chars[i];\n var contextParams = new ContextParams(chars, i);\n this.dispatch('next', [contextParams]);\n this.runContextCheck(contextParams);\n var token = new Token$1(char);\n this.tokens.push(token);\n this.dispatch('newToken', [token, contextParams]);\n }\n this.dispatch('end', [this.tokens]);\n return this.tokens;\n };\n\n // ╭─┄┄┄────────────────────────┄─────────────────────────────────────────────╮\n // ┊ Character Class Assertions ┊ Checks if a char belongs to a certain class ┊\n // ╰─╾──────────────────────────┄─────────────────────────────────────────────╯\n // jscs:disable maximumLineLength\n /**\n * Check if a char is Arabic\n * @param {string} c a single char\n */\n function isArabicChar(c) {\n return /[\\u0600-\\u065F\\u066A-\\u06D2\\u06FA-\\u06FF]/.test(c);\n }\n\n /**\n * Check if a char is an isolated arabic char\n * @param {string} c a single char\n */\n function isIsolatedArabicChar(char) {\n return /[\\u0630\\u0690\\u0621\\u0631\\u0661\\u0671\\u0622\\u0632\\u0672\\u0692\\u06C2\\u0623\\u0673\\u0693\\u06C3\\u0624\\u0694\\u06C4\\u0625\\u0675\\u0695\\u06C5\\u06E5\\u0676\\u0696\\u06C6\\u0627\\u0677\\u0697\\u06C7\\u0648\\u0688\\u0698\\u06C8\\u0689\\u0699\\u06C9\\u068A\\u06CA\\u066B\\u068B\\u06CB\\u068C\\u068D\\u06CD\\u06FD\\u068E\\u06EE\\u06FE\\u062F\\u068F\\u06CF\\u06EF]/.test(char);\n }\n\n /**\n * Check if a char is an Arabic Tashkeel char\n * @param {string} c a single char\n */\n function isTashkeelArabicChar(char) {\n return /[\\u0600-\\u0605\\u060C-\\u060E\\u0610-\\u061B\\u061E\\u064B-\\u065F\\u0670\\u06D6-\\u06DC\\u06DF-\\u06E4\\u06E7\\u06E8\\u06EA-\\u06ED]/.test(char);\n }\n\n /**\n * Check if a char is Latin\n * @param {string} c a single char\n */\n function isLatinChar(c) {\n return /[A-z]/.test(c);\n }\n\n /**\n * Check if a char is whitespace char\n * @param {string} c a single char\n */\n function isWhiteSpace(c) {\n return /\\s/.test(c);\n }\n\n /**\n * Query a feature by some of it's properties to lookup a glyph substitution.\n */\n\n /**\n * Create feature query instance\n * @param {Font} font opentype font instance\n */\n function FeatureQuery(font) {\n this.font = font;\n this.features = {};\n }\n\n /**\n * @typedef SubstitutionAction\n * @type Object\n * @property {number} id substitution type\n * @property {string} tag feature tag\n * @property {any} substitution substitution value(s)\n */\n\n /**\n * Create a substitution action instance\n * @param {SubstitutionAction} action\n */\n function SubstitutionAction(action) {\n this.id = action.id;\n this.tag = action.tag;\n this.substitution = action.substitution;\n }\n\n /**\n * Lookup a coverage table\n * @param {number} glyphIndex glyph index\n * @param {CoverageTable} coverage coverage table\n */\n function lookupCoverage(glyphIndex, coverage) {\n if (!glyphIndex) { return -1; }\n switch (coverage.format) {\n case 1:\n return coverage.glyphs.indexOf(glyphIndex);\n\n case 2:\n var ranges = coverage.ranges;\n for (var i = 0; i < ranges.length; i++) {\n var range = ranges[i];\n if (glyphIndex >= range.start && glyphIndex <= range.end) {\n var offset = glyphIndex - range.start;\n return range.index + offset;\n }\n }\n break;\n default:\n return -1; // not found\n }\n return -1;\n }\n\n /**\n * Handle a single substitution - format 1\n * @param {ContextParams} contextParams context params to lookup\n */\n function singleSubstitutionFormat1(glyphIndex, subtable) {\n var substituteIndex = lookupCoverage(glyphIndex, subtable.coverage);\n if (substituteIndex === -1) { return null; }\n return glyphIndex + subtable.deltaGlyphId;\n }\n\n /**\n * Handle a single substitution - format 2\n * @param {ContextParams} contextParams context params to lookup\n */\n function singleSubstitutionFormat2(glyphIndex, subtable) {\n var substituteIndex = lookupCoverage(glyphIndex, subtable.coverage);\n if (substituteIndex === -1) { return null; }\n return subtable.substitute[substituteIndex];\n }\n\n /**\n * Lookup a list of coverage tables\n * @param {any} coverageList a list of coverage tables\n * @param {ContextParams} contextParams context params to lookup\n */\n function lookupCoverageList(coverageList, contextParams) {\n var lookupList = [];\n for (var i = 0; i < coverageList.length; i++) {\n var coverage = coverageList[i];\n var glyphIndex = contextParams.current;\n glyphIndex = Array.isArray(glyphIndex) ? glyphIndex[0] : glyphIndex;\n var lookupIndex = lookupCoverage(glyphIndex, coverage);\n if (lookupIndex !== -1) {\n lookupList.push(lookupIndex);\n }\n }\n if (lookupList.length !== coverageList.length) { return -1; }\n return lookupList;\n }\n\n /**\n * Handle chaining context substitution - format 3\n * @param {ContextParams} contextParams context params to lookup\n */\n function chainingSubstitutionFormat3(contextParams, subtable) {\n var lookupsCount = (\n subtable.inputCoverage.length +\n subtable.lookaheadCoverage.length +\n subtable.backtrackCoverage.length\n );\n if (contextParams.context.length < lookupsCount) { return []; }\n // INPUT LOOKUP //\n var inputLookups = lookupCoverageList(\n subtable.inputCoverage, contextParams\n );\n if (inputLookups === -1) { return []; }\n // LOOKAHEAD LOOKUP //\n var lookaheadOffset = subtable.inputCoverage.length - 1;\n if (contextParams.lookahead.length < subtable.lookaheadCoverage.length) { return []; }\n var lookaheadContext = contextParams.lookahead.slice(lookaheadOffset);\n while (lookaheadContext.length && isTashkeelArabicChar(lookaheadContext[0].char)) {\n lookaheadContext.shift();\n }\n var lookaheadParams = new ContextParams(lookaheadContext, 0);\n var lookaheadLookups = lookupCoverageList(\n subtable.lookaheadCoverage, lookaheadParams\n );\n // BACKTRACK LOOKUP //\n var backtrackContext = [].concat(contextParams.backtrack);\n backtrackContext.reverse();\n while (backtrackContext.length && isTashkeelArabicChar(backtrackContext[0].char)) {\n backtrackContext.shift();\n }\n if (backtrackContext.length < subtable.backtrackCoverage.length) { return []; }\n var backtrackParams = new ContextParams(backtrackContext, 0);\n var backtrackLookups = lookupCoverageList(\n subtable.backtrackCoverage, backtrackParams\n );\n var contextRulesMatch = (\n inputLookups.length === subtable.inputCoverage.length &&\n lookaheadLookups.length === subtable.lookaheadCoverage.length &&\n backtrackLookups.length === subtable.backtrackCoverage.length\n );\n var substitutions = [];\n if (contextRulesMatch) {\n for (var i = 0; i < subtable.lookupRecords.length; i++) {\n var lookupRecord = subtable.lookupRecords[i];\n var lookupListIndex = lookupRecord.lookupListIndex;\n var lookupTable = this.getLookupByIndex(lookupListIndex);\n for (var s = 0; s < lookupTable.subtables.length; s++) {\n var subtable$1 = lookupTable.subtables[s];\n var lookup = this.getLookupMethod(lookupTable, subtable$1);\n var substitutionType = this.getSubstitutionType(lookupTable, subtable$1);\n if (substitutionType === '12') {\n for (var n = 0; n < inputLookups.length; n++) {\n var glyphIndex = contextParams.get(n);\n var substitution = lookup(glyphIndex);\n if (substitution) { substitutions.push(substitution); }\n }\n }\n }\n }\n }\n return substitutions;\n }\n\n /**\n * Handle ligature substitution - format 1\n * @param {ContextParams} contextParams context params to lookup\n */\n function ligatureSubstitutionFormat1(contextParams, subtable) {\n // COVERAGE LOOKUP //\n var glyphIndex = contextParams.current;\n var ligSetIndex = lookupCoverage(glyphIndex, subtable.coverage);\n if (ligSetIndex === -1) { return null; }\n // COMPONENTS LOOKUP\n // (!) note, components are ordered in the written direction.\n var ligature;\n var ligatureSet = subtable.ligatureSets[ligSetIndex];\n for (var s = 0; s < ligatureSet.length; s++) {\n ligature = ligatureSet[s];\n for (var l = 0; l < ligature.components.length; l++) {\n var lookaheadItem = contextParams.lookahead[l];\n var component = ligature.components[l];\n if (lookaheadItem !== component) { break; }\n if (l === ligature.components.length - 1) { return ligature; }\n }\n }\n return null;\n }\n\n /**\n * Handle decomposition substitution - format 1\n * @param {number} glyphIndex glyph index\n * @param {any} subtable subtable\n */\n function decompositionSubstitutionFormat1(glyphIndex, subtable) {\n var substituteIndex = lookupCoverage(glyphIndex, subtable.coverage);\n if (substituteIndex === -1) { return null; }\n return subtable.sequences[substituteIndex];\n }\n\n /**\n * Get default script features indexes\n */\n FeatureQuery.prototype.getDefaultScriptFeaturesIndexes = function () {\n var scripts = this.font.tables.gsub.scripts;\n for (var s = 0; s < scripts.length; s++) {\n var script = scripts[s];\n if (script.tag === 'DFLT') { return (\n script.script.defaultLangSys.featureIndexes\n ); }\n }\n return [];\n };\n\n /**\n * Get feature indexes of a specific script\n * @param {string} scriptTag script tag\n */\n FeatureQuery.prototype.getScriptFeaturesIndexes = function(scriptTag) {\n var tables = this.font.tables;\n if (!tables.gsub) { return []; }\n if (!scriptTag) { return this.getDefaultScriptFeaturesIndexes(); }\n var scripts = this.font.tables.gsub.scripts;\n for (var i = 0; i < scripts.length; i++) {\n var script = scripts[i];\n if (script.tag === scriptTag && script.script.defaultLangSys) {\n return script.script.defaultLangSys.featureIndexes;\n } else {\n var langSysRecords = script.langSysRecords;\n if (!!langSysRecords) {\n for (var j = 0; j < langSysRecords.length; j++) {\n var langSysRecord = langSysRecords[j];\n if (langSysRecord.tag === scriptTag) {\n var langSys = langSysRecord.langSys;\n return langSys.featureIndexes;\n }\n }\n }\n }\n }\n return this.getDefaultScriptFeaturesIndexes();\n };\n\n /**\n * Map a feature tag to a gsub feature\n * @param {any} features gsub features\n * @param {string} scriptTag script tag\n */\n FeatureQuery.prototype.mapTagsToFeatures = function (features, scriptTag) {\n var tags = {};\n for (var i = 0; i < features.length; i++) {\n var tag = features[i].tag;\n var feature = features[i].feature;\n tags[tag] = feature;\n }\n this.features[scriptTag].tags = tags;\n };\n\n /**\n * Get features of a specific script\n * @param {string} scriptTag script tag\n */\n FeatureQuery.prototype.getScriptFeatures = function (scriptTag) {\n var features = this.features[scriptTag];\n if (this.features.hasOwnProperty(scriptTag)) { return features; }\n var featuresIndexes = this.getScriptFeaturesIndexes(scriptTag);\n if (!featuresIndexes) { return null; }\n var gsub = this.font.tables.gsub;\n features = featuresIndexes.map(function (index) { return gsub.features[index]; });\n this.features[scriptTag] = features;\n this.mapTagsToFeatures(features, scriptTag);\n return features;\n };\n\n /**\n * Get substitution type\n * @param {any} lookupTable lookup table\n * @param {any} subtable subtable\n */\n FeatureQuery.prototype.getSubstitutionType = function(lookupTable, subtable) {\n var lookupType = lookupTable.lookupType.toString();\n var substFormat = subtable.substFormat.toString();\n return lookupType + substFormat;\n };\n\n /**\n * Get lookup method\n * @param {any} lookupTable lookup table\n * @param {any} subtable subtable\n */\n FeatureQuery.prototype.getLookupMethod = function(lookupTable, subtable) {\n var this$1 = this;\n\n var substitutionType = this.getSubstitutionType(lookupTable, subtable);\n switch (substitutionType) {\n case '11':\n return function (glyphIndex) { return singleSubstitutionFormat1.apply(\n this$1, [glyphIndex, subtable]\n ); };\n case '12':\n return function (glyphIndex) { return singleSubstitutionFormat2.apply(\n this$1, [glyphIndex, subtable]\n ); };\n case '63':\n return function (contextParams) { return chainingSubstitutionFormat3.apply(\n this$1, [contextParams, subtable]\n ); };\n case '41':\n return function (contextParams) { return ligatureSubstitutionFormat1.apply(\n this$1, [contextParams, subtable]\n ); };\n case '21':\n return function (glyphIndex) { return decompositionSubstitutionFormat1.apply(\n this$1, [glyphIndex, subtable]\n ); };\n default:\n throw new Error(\n \"lookupType: \" + (lookupTable.lookupType) + \" - \" +\n \"substFormat: \" + (subtable.substFormat) + \" \" +\n \"is not yet supported\"\n );\n }\n };\n\n /**\n * [ LOOKUP TYPES ]\n * -------------------------------\n * Single 1;\n * Multiple 2;\n * Alternate 3;\n * Ligature 4;\n * Context 5;\n * ChainingContext 6;\n * ExtensionSubstitution 7;\n * ReverseChainingContext 8;\n * -------------------------------\n *\n */\n\n /**\n * @typedef FQuery\n * @type Object\n * @param {string} tag feature tag\n * @param {string} script feature script\n * @param {ContextParams} contextParams context params\n */\n\n /**\n * Lookup a feature using a query parameters\n * @param {FQuery} query feature query\n */\n FeatureQuery.prototype.lookupFeature = function (query) {\n var contextParams = query.contextParams;\n var currentIndex = contextParams.index;\n var feature = this.getFeature({\n tag: query.tag, script: query.script\n });\n if (!feature) { return new Error(\n \"font '\" + (this.font.names.fullName.en) + \"' \" +\n \"doesn't support feature '\" + (query.tag) + \"' \" +\n \"for script '\" + (query.script) + \"'.\"\n ); }\n var lookups = this.getFeatureLookups(feature);\n var substitutions = [].concat(contextParams.context);\n for (var l = 0; l < lookups.length; l++) {\n var lookupTable = lookups[l];\n var subtables = this.getLookupSubtables(lookupTable);\n for (var s = 0; s < subtables.length; s++) {\n var subtable = subtables[s];\n var substType = this.getSubstitutionType(lookupTable, subtable);\n var lookup = this.getLookupMethod(lookupTable, subtable);\n var substitution = (void 0);\n switch (substType) {\n case '11':\n substitution = lookup(contextParams.current);\n if (substitution) {\n substitutions.splice(currentIndex, 1, new SubstitutionAction({\n id: 11, tag: query.tag, substitution: substitution\n }));\n }\n break;\n case '12':\n substitution = lookup(contextParams.current);\n if (substitution) {\n substitutions.splice(currentIndex, 1, new SubstitutionAction({\n id: 12, tag: query.tag, substitution: substitution\n }));\n }\n break;\n case '63':\n substitution = lookup(contextParams);\n if (Array.isArray(substitution) && substitution.length) {\n substitutions.splice(currentIndex, 1, new SubstitutionAction({\n id: 63, tag: query.tag, substitution: substitution\n }));\n }\n break;\n case '41':\n substitution = lookup(contextParams);\n if (substitution) {\n substitutions.splice(currentIndex, 1, new SubstitutionAction({\n id: 41, tag: query.tag, substitution: substitution\n }));\n }\n break;\n case '21':\n substitution = lookup(contextParams.current);\n if (substitution) {\n substitutions.splice(currentIndex, 1, new SubstitutionAction({\n id: 21, tag: query.tag, substitution: substitution\n }));\n }\n break;\n }\n contextParams = new ContextParams(substitutions, currentIndex);\n if (Array.isArray(substitution) && !substitution.length) { continue; }\n substitution = null;\n }\n }\n return substitutions.length ? substitutions : null;\n };\n\n /**\n * Checks if a font supports a specific features\n * @param {FQuery} query feature query object\n */\n FeatureQuery.prototype.supports = function (query) {\n if (!query.script) { return false; }\n this.getScriptFeatures(query.script);\n var supportedScript = this.features.hasOwnProperty(query.script);\n if (!query.tag) { return supportedScript; }\n var supportedFeature = (\n this.features[query.script].some(function (feature) { return feature.tag === query.tag; })\n );\n return supportedScript && supportedFeature;\n };\n\n /**\n * Get lookup table subtables\n * @param {any} lookupTable lookup table\n */\n FeatureQuery.prototype.getLookupSubtables = function (lookupTable) {\n return lookupTable.subtables || null;\n };\n\n /**\n * Get lookup table by index\n * @param {number} index lookup table index\n */\n FeatureQuery.prototype.getLookupByIndex = function (index) {\n var lookups = this.font.tables.gsub.lookups;\n return lookups[index] || null;\n };\n\n /**\n * Get lookup tables for a feature\n * @param {string} feature\n */\n FeatureQuery.prototype.getFeatureLookups = function (feature) {\n // TODO: memoize\n return feature.lookupListIndexes.map(this.getLookupByIndex.bind(this));\n };\n\n /**\n * Query a feature by it's properties\n * @param {any} query an object that describes the properties of a query\n */\n FeatureQuery.prototype.getFeature = function getFeature(query) {\n if (!this.font) { return { FAIL: \"No font was found\"}; }\n if (!this.features.hasOwnProperty(query.script)) {\n this.getScriptFeatures(query.script);\n }\n var scriptFeatures = this.features[query.script];\n if (!scriptFeatures) { return (\n { FAIL: (\"No feature for script \" + (query.script))}\n ); }\n if (!scriptFeatures.tags[query.tag]) { return null; }\n return this.features[query.script].tags[query.tag];\n };\n\n /**\n * Arabic word context checkers\n */\n\n function arabicWordStartCheck(contextParams) {\n var char = contextParams.current;\n var prevChar = contextParams.get(-1);\n return (\n // ? arabic first char\n (prevChar === null && isArabicChar(char)) ||\n // ? arabic char preceded with a non arabic char\n (!isArabicChar(prevChar) && isArabicChar(char))\n );\n }\n\n function arabicWordEndCheck(contextParams) {\n var nextChar = contextParams.get(1);\n return (\n // ? last arabic char\n (nextChar === null) ||\n // ? next char is not arabic\n (!isArabicChar(nextChar))\n );\n }\n\n var arabicWordCheck = {\n startCheck: arabicWordStartCheck,\n endCheck: arabicWordEndCheck\n };\n\n /**\n * Arabic sentence context checkers\n */\n\n function arabicSentenceStartCheck(contextParams) {\n var char = contextParams.current;\n var prevChar = contextParams.get(-1);\n return (\n // ? an arabic char preceded with a non arabic char\n (isArabicChar(char) || isTashkeelArabicChar(char)) &&\n !isArabicChar(prevChar)\n );\n }\n\n function arabicSentenceEndCheck(contextParams) {\n var nextChar = contextParams.get(1);\n switch (true) {\n case nextChar === null:\n return true;\n case (!isArabicChar(nextChar) && !isTashkeelArabicChar(nextChar)):\n var nextIsWhitespace = isWhiteSpace(nextChar);\n if (!nextIsWhitespace) { return true; }\n if (nextIsWhitespace) {\n var arabicCharAhead = false;\n arabicCharAhead = (\n contextParams.lookahead.some(\n function (c) { return isArabicChar(c) || isTashkeelArabicChar(c); }\n )\n );\n if (!arabicCharAhead) { return true; }\n }\n break;\n default:\n return false;\n }\n }\n\n var arabicSentenceCheck = {\n startCheck: arabicSentenceStartCheck,\n endCheck: arabicSentenceEndCheck\n };\n\n /**\n * Apply single substitution format 1\n * @param {Array} substitutions substitutions\n * @param {any} tokens a list of tokens\n * @param {number} index token index\n */\n function singleSubstitutionFormat1$1(action, tokens, index) {\n tokens[index].setState(action.tag, action.substitution);\n }\n\n /**\n * Apply single substitution format 2\n * @param {Array} substitutions substitutions\n * @param {any} tokens a list of tokens\n * @param {number} index token index\n */\n function singleSubstitutionFormat2$1(action, tokens, index) {\n tokens[index].setState(action.tag, action.substitution);\n }\n\n /**\n * Apply chaining context substitution format 3\n * @param {Array} substitutions substitutions\n * @param {any} tokens a list of tokens\n * @param {number} index token index\n */\n function chainingSubstitutionFormat3$1(action, tokens, index) {\n action.substitution.forEach(function (subst, offset) {\n var token = tokens[index + offset];\n token.setState(action.tag, subst);\n });\n }\n\n /**\n * Apply ligature substitution format 1\n * @param {Array} substitutions substitutions\n * @param {any} tokens a list of tokens\n * @param {number} index token index\n */\n function ligatureSubstitutionFormat1$1(action, tokens, index) {\n var token = tokens[index];\n token.setState(action.tag, action.substitution.ligGlyph);\n var compsCount = action.substitution.components.length;\n for (var i = 0; i < compsCount; i++) {\n token = tokens[index + i + 1];\n token.setState('deleted', true);\n }\n }\n\n /**\n * Supported substitutions\n */\n var SUBSTITUTIONS = {\n 11: singleSubstitutionFormat1$1,\n 12: singleSubstitutionFormat2$1,\n 63: chainingSubstitutionFormat3$1,\n 41: ligatureSubstitutionFormat1$1\n };\n\n /**\n * Apply substitutions to a list of tokens\n * @param {Array} substitutions substitutions\n * @param {any} tokens a list of tokens\n * @param {number} index token index\n */\n function applySubstitution(action, tokens, index) {\n if (action instanceof SubstitutionAction && SUBSTITUTIONS[action.id]) {\n SUBSTITUTIONS[action.id](action, tokens, index);\n }\n }\n\n /**\n * Apply Arabic presentation forms to a range of tokens\n */\n\n /**\n * Check if a char can be connected to it's preceding char\n * @param {ContextParams} charContextParams context params of a char\n */\n function willConnectPrev(charContextParams) {\n var backtrack = [].concat(charContextParams.backtrack);\n for (var i = backtrack.length - 1; i >= 0; i--) {\n var prevChar = backtrack[i];\n var isolated = isIsolatedArabicChar(prevChar);\n var tashkeel = isTashkeelArabicChar(prevChar);\n if (!isolated && !tashkeel) { return true; }\n if (isolated) { return false; }\n }\n return false;\n }\n\n /**\n * Check if a char can be connected to it's proceeding char\n * @param {ContextParams} charContextParams context params of a char\n */\n function willConnectNext(charContextParams) {\n if (isIsolatedArabicChar(charContextParams.current)) { return false; }\n for (var i = 0; i < charContextParams.lookahead.length; i++) {\n var nextChar = charContextParams.lookahead[i];\n var tashkeel = isTashkeelArabicChar(nextChar);\n if (!tashkeel) { return true; }\n }\n return false;\n }\n\n /**\n * Apply arabic presentation forms to a list of tokens\n * @param {ContextRange} range a range of tokens\n */\n function arabicPresentationForms(range) {\n var this$1 = this;\n\n var script = 'arab';\n var tags = this.featuresTags[script];\n var tokens = this.tokenizer.getRangeTokens(range);\n if (tokens.length === 1) { return; }\n var contextParams = new ContextParams(\n tokens.map(function (token) { return token.getState('glyphIndex'); }\n ), 0);\n var charContextParams = new ContextParams(\n tokens.map(function (token) { return token.char; }\n ), 0);\n tokens.forEach(function (token, index) {\n if (isTashkeelArabicChar(token.char)) { return; }\n contextParams.setCurrentIndex(index);\n charContextParams.setCurrentIndex(index);\n var CONNECT = 0; // 2 bits 00 (10: can connect next) (01: can connect prev)\n if (willConnectPrev(charContextParams)) { CONNECT |= 1; }\n if (willConnectNext(charContextParams)) { CONNECT |= 2; }\n var tag;\n switch (CONNECT) {\n case 1: (tag = 'fina'); break;\n case 2: (tag = 'init'); break;\n case 3: (tag = 'medi'); break;\n }\n if (tags.indexOf(tag) === -1) { return; }\n var substitutions = this$1.query.lookupFeature({\n tag: tag, script: script, contextParams: contextParams\n });\n if (substitutions instanceof Error) { return console.info(substitutions.message); }\n substitutions.forEach(function (action, index) {\n if (action instanceof SubstitutionAction) {\n applySubstitution(action, tokens, index);\n contextParams.context[index] = action.substitution;\n }\n });\n });\n }\n\n /**\n * Apply Arabic required ligatures feature to a range of tokens\n */\n\n /**\n * Update context params\n * @param {any} tokens a list of tokens\n * @param {number} index current item index\n */\n function getContextParams(tokens, index) {\n var context = tokens.map(function (token) { return token.activeState.value; });\n return new ContextParams(context, index || 0);\n }\n\n /**\n * Apply Arabic required ligatures to a context range\n * @param {ContextRange} range a range of tokens\n */\n function arabicRequiredLigatures(range) {\n var this$1 = this;\n\n var script = 'arab';\n var tokens = this.tokenizer.getRangeTokens(range);\n var contextParams = getContextParams(tokens);\n contextParams.context.forEach(function (glyphIndex, index) {\n contextParams.setCurrentIndex(index);\n var substitutions = this$1.query.lookupFeature({\n tag: 'rlig', script: script, contextParams: contextParams\n });\n if (substitutions.length) {\n substitutions.forEach(\n function (action) { return applySubstitution(action, tokens, index); }\n );\n contextParams = getContextParams(tokens);\n }\n });\n }\n\n /**\n * Latin word context checkers\n */\n\n function latinWordStartCheck(contextParams) {\n var char = contextParams.current;\n var prevChar = contextParams.get(-1);\n return (\n // ? latin first char\n (prevChar === null && isLatinChar(char)) ||\n // ? latin char preceded with a non latin char\n (!isLatinChar(prevChar) && isLatinChar(char))\n );\n }\n\n function latinWordEndCheck(contextParams) {\n var nextChar = contextParams.get(1);\n return (\n // ? last latin char\n (nextChar === null) ||\n // ? next char is not latin\n (!isLatinChar(nextChar))\n );\n }\n\n var latinWordCheck = {\n startCheck: latinWordStartCheck,\n endCheck: latinWordEndCheck\n };\n\n /**\n * Apply Latin ligature feature to a range of tokens\n */\n\n /**\n * Update context params\n * @param {any} tokens a list of tokens\n * @param {number} index current item index\n */\n function getContextParams$1(tokens, index) {\n var context = tokens.map(function (token) { return token.activeState.value; });\n return new ContextParams(context, index || 0);\n }\n\n /**\n * Apply Arabic required ligatures to a context range\n * @param {ContextRange} range a range of tokens\n */\n function latinLigature(range) {\n var this$1 = this;\n\n var script = 'latn';\n var tokens = this.tokenizer.getRangeTokens(range);\n var contextParams = getContextParams$1(tokens);\n contextParams.context.forEach(function (glyphIndex, index) {\n contextParams.setCurrentIndex(index);\n var substitutions = this$1.query.lookupFeature({\n tag: 'liga', script: script, contextParams: contextParams\n });\n if (substitutions.length) {\n substitutions.forEach(\n function (action) { return applySubstitution(action, tokens, index); }\n );\n contextParams = getContextParams$1(tokens);\n }\n });\n }\n\n /**\n * Infer bidirectional properties for a given text and apply\n * the corresponding layout rules.\n */\n\n /**\n * Create Bidi. features\n * @param {string} baseDir text base direction. value either 'ltr' or 'rtl'\n */\n function Bidi(baseDir) {\n this.baseDir = baseDir || 'ltr';\n this.tokenizer = new Tokenizer();\n this.featuresTags = {};\n }\n\n /**\n * Sets Bidi text\n * @param {string} text a text input\n */\n Bidi.prototype.setText = function (text) {\n this.text = text;\n };\n\n /**\n * Store essential context checks:\n * arabic word check for applying gsub features\n * arabic sentence check for adjusting arabic layout\n */\n Bidi.prototype.contextChecks = ({\n latinWordCheck: latinWordCheck,\n arabicWordCheck: arabicWordCheck,\n arabicSentenceCheck: arabicSentenceCheck\n });\n\n /**\n * Register arabic word check\n */\n function registerContextChecker(checkId) {\n var check = this.contextChecks[(checkId + \"Check\")];\n return this.tokenizer.registerContextChecker(\n checkId, check.startCheck, check.endCheck\n );\n }\n\n /**\n * Perform pre tokenization procedure then\n * tokenize text input\n */\n function tokenizeText() {\n registerContextChecker.call(this, 'latinWord');\n registerContextChecker.call(this, 'arabicWord');\n registerContextChecker.call(this, 'arabicSentence');\n return this.tokenizer.tokenize(this.text);\n }\n\n /**\n * Reverse arabic sentence layout\n * TODO: check base dir before applying adjustments - priority low\n */\n function reverseArabicSentences() {\n var this$1 = this;\n\n var ranges = this.tokenizer.getContextRanges('arabicSentence');\n ranges.forEach(function (range) {\n var rangeTokens = this$1.tokenizer.getRangeTokens(range);\n this$1.tokenizer.replaceRange(\n range.startIndex,\n range.endOffset,\n rangeTokens.reverse()\n );\n });\n }\n\n /**\n * Register supported features tags\n * @param {script} script script tag\n * @param {Array} tags features tags list\n */\n Bidi.prototype.registerFeatures = function (script, tags) {\n var this$1 = this;\n\n var supportedTags = tags.filter(\n function (tag) { return this$1.query.supports({script: script, tag: tag}); }\n );\n if (!this.featuresTags.hasOwnProperty(script)) {\n this.featuresTags[script] = supportedTags;\n } else {\n this.featuresTags[script] =\n this.featuresTags[script].concat(supportedTags);\n }\n };\n\n /**\n * Apply GSUB features\n * @param {Array} tagsList a list of features tags\n * @param {string} script a script tag\n * @param {Font} font opentype font instance\n */\n Bidi.prototype.applyFeatures = function (font, features) {\n if (!font) { throw new Error(\n 'No valid font was provided to apply features'\n ); }\n if (!this.query) { this.query = new FeatureQuery(font); }\n for (var f = 0; f < features.length; f++) {\n var feature = features[f];\n if (!this.query.supports({script: feature.script})) { continue; }\n this.registerFeatures(feature.script, feature.tags);\n }\n };\n\n /**\n * Register a state modifier\n * @param {string} modifierId state modifier id\n * @param {function} condition a predicate function that returns true or false\n * @param {function} modifier a modifier function to set token state\n */\n Bidi.prototype.registerModifier = function (modifierId, condition, modifier) {\n this.tokenizer.registerModifier(modifierId, condition, modifier);\n };\n\n /**\n * Check if 'glyphIndex' is registered\n */\n function checkGlyphIndexStatus() {\n if (this.tokenizer.registeredModifiers.indexOf('glyphIndex') === -1) {\n throw new Error(\n 'glyphIndex modifier is required to apply ' +\n 'arabic presentation features.'\n );\n }\n }\n\n /**\n * Apply arabic presentation forms features\n */\n function applyArabicPresentationForms() {\n var this$1 = this;\n\n var script = 'arab';\n if (!this.featuresTags.hasOwnProperty(script)) { return; }\n checkGlyphIndexStatus.call(this);\n var ranges = this.tokenizer.getContextRanges('arabicWord');\n ranges.forEach(function (range) {\n arabicPresentationForms.call(this$1, range);\n });\n }\n\n /**\n * Apply required arabic ligatures\n */\n function applyArabicRequireLigatures() {\n var this$1 = this;\n\n var script = 'arab';\n if (!this.featuresTags.hasOwnProperty(script)) { return; }\n var tags = this.featuresTags[script];\n if (tags.indexOf('rlig') === -1) { return; }\n checkGlyphIndexStatus.call(this);\n var ranges = this.tokenizer.getContextRanges('arabicWord');\n ranges.forEach(function (range) {\n arabicRequiredLigatures.call(this$1, range);\n });\n }\n\n /**\n * Apply required arabic ligatures\n */\n function applyLatinLigatures() {\n var this$1 = this;\n\n var script = 'latn';\n if (!this.featuresTags.hasOwnProperty(script)) { return; }\n var tags = this.featuresTags[script];\n if (tags.indexOf('liga') === -1) { return; }\n checkGlyphIndexStatus.call(this);\n var ranges = this.tokenizer.getContextRanges('latinWord');\n ranges.forEach(function (range) {\n latinLigature.call(this$1, range);\n });\n }\n\n /**\n * Check if a context is registered\n * @param {string} contextId context id\n */\n Bidi.prototype.checkContextReady = function (contextId) {\n return !!this.tokenizer.getContext(contextId);\n };\n\n /**\n * Apply features to registered contexts\n */\n Bidi.prototype.applyFeaturesToContexts = function () {\n if (this.checkContextReady('arabicWord')) {\n applyArabicPresentationForms.call(this);\n applyArabicRequireLigatures.call(this);\n }\n if (this.checkContextReady('latinWord')) {\n applyLatinLigatures.call(this);\n }\n if (this.checkContextReady('arabicSentence')) {\n reverseArabicSentences.call(this);\n }\n };\n\n /**\n * process text input\n * @param {string} text an input text\n */\n Bidi.prototype.processText = function(text) {\n if (!this.text || this.text !== text) {\n this.setText(text);\n tokenizeText.call(this);\n this.applyFeaturesToContexts();\n }\n };\n\n /**\n * Process a string of text to identify and adjust\n * bidirectional text entities.\n * @param {string} text input text\n */\n Bidi.prototype.getBidiText = function (text) {\n this.processText(text);\n return this.tokenizer.getText();\n };\n\n /**\n * Get the current state index of each token\n * @param {text} text an input text\n */\n Bidi.prototype.getTextGlyphs = function (text) {\n this.processText(text);\n var indexes = [];\n for (var i = 0; i < this.tokenizer.tokens.length; i++) {\n var token = this.tokenizer.tokens[i];\n if (token.state.deleted) { continue; }\n var index = token.activeState.value;\n indexes.push(Array.isArray(index) ? index[0] : index);\n }\n return indexes;\n };\n\n // The Font object\n\n /**\n * @typedef FontOptions\n * @type Object\n * @property {Boolean} empty - whether to create a new empty font\n * @property {string} familyName\n * @property {string} styleName\n * @property {string=} fullName\n * @property {string=} postScriptName\n * @property {string=} designer\n * @property {string=} designerURL\n * @property {string=} manufacturer\n * @property {string=} manufacturerURL\n * @property {string=} license\n * @property {string=} licenseURL\n * @property {string=} version\n * @property {string=} description\n * @property {string=} copyright\n * @property {string=} trademark\n * @property {Number} unitsPerEm\n * @property {Number} ascender\n * @property {Number} descender\n * @property {Number} createdTimestamp\n * @property {string=} weightClass\n * @property {string=} widthClass\n * @property {string=} fsSelection\n */\n\n /**\n * A Font represents a loaded OpenType font file.\n * It contains a set of glyphs and methods to draw text on a drawing context,\n * or to get a path representing the text.\n * @exports opentype.Font\n * @class\n * @param {FontOptions}\n * @constructor\n */\n function Font(options) {\n options = options || {};\n options.tables = options.tables || {};\n\n if (!options.empty) {\n // Check that we've provided the minimum set of names.\n checkArgument(options.familyName, 'When creating a new Font object, familyName is required.');\n checkArgument(options.styleName, 'When creating a new Font object, styleName is required.');\n checkArgument(options.unitsPerEm, 'When creating a new Font object, unitsPerEm is required.');\n checkArgument(options.ascender, 'When creating a new Font object, ascender is required.');\n checkArgument(options.descender <= 0, 'When creating a new Font object, negative descender value is required.');\n\n // OS X will complain if the names are empty, so we put a single space everywhere by default.\n this.names = {\n fontFamily: {en: options.familyName || ' '},\n fontSubfamily: {en: options.styleName || ' '},\n fullName: {en: options.fullName || options.familyName + ' ' + options.styleName},\n // postScriptName may not contain any whitespace\n postScriptName: {en: options.postScriptName || (options.familyName + options.styleName).replace(/\\s/g, '')},\n designer: {en: options.designer || ' '},\n designerURL: {en: options.designerURL || ' '},\n manufacturer: {en: options.manufacturer || ' '},\n manufacturerURL: {en: options.manufacturerURL || ' '},\n license: {en: options.license || ' '},\n licenseURL: {en: options.licenseURL || ' '},\n version: {en: options.version || 'Version 0.1'},\n description: {en: options.description || ' '},\n copyright: {en: options.copyright || ' '},\n trademark: {en: options.trademark || ' '}\n };\n this.unitsPerEm = options.unitsPerEm || 1000;\n this.ascender = options.ascender;\n this.descender = options.descender;\n this.createdTimestamp = options.createdTimestamp;\n this.tables = Object.assign(options.tables, {\n os2: Object.assign({\n usWeightClass: options.weightClass || this.usWeightClasses.MEDIUM,\n usWidthClass: options.widthClass || this.usWidthClasses.MEDIUM,\n fsSelection: options.fsSelection || this.fsSelectionValues.REGULAR,\n }, options.tables.os2)\n });\n }\n\n this.supported = true; // Deprecated: parseBuffer will throw an error if font is not supported.\n this.glyphs = new glyphset.GlyphSet(this, options.glyphs || []);\n this.encoding = new DefaultEncoding(this);\n this.position = new Position(this);\n this.substitution = new Substitution(this);\n this.tables = this.tables || {};\n\n // needed for low memory mode only.\n this._push = null;\n this._hmtxTableData = {};\n\n Object.defineProperty(this, 'hinting', {\n get: function() {\n if (this._hinting) { return this._hinting; }\n if (this.outlinesFormat === 'truetype') {\n return (this._hinting = new Hinting(this));\n }\n }\n });\n }\n\n /**\n * Check if the font has a glyph for the given character.\n * @param {string}\n * @return {Boolean}\n */\n Font.prototype.hasChar = function(c) {\n return this.encoding.charToGlyphIndex(c) !== null;\n };\n\n /**\n * Convert the given character to a single glyph index.\n * Note that this function assumes that there is a one-to-one mapping between\n * the given character and a glyph; for complex scripts this might not be the case.\n * @param {string}\n * @return {Number}\n */\n Font.prototype.charToGlyphIndex = function(s) {\n return this.encoding.charToGlyphIndex(s);\n };\n\n /**\n * Convert the given character to a single Glyph object.\n * Note that this function assumes that there is a one-to-one mapping between\n * the given character and a glyph; for complex scripts this might not be the case.\n * @param {string}\n * @return {opentype.Glyph}\n */\n Font.prototype.charToGlyph = function(c) {\n var glyphIndex = this.charToGlyphIndex(c);\n var glyph = this.glyphs.get(glyphIndex);\n if (!glyph) {\n // .notdef\n glyph = this.glyphs.get(0);\n }\n\n return glyph;\n };\n\n /**\n * Update features\n * @param {any} options features options\n */\n Font.prototype.updateFeatures = function (options) {\n // TODO: update all features options not only 'latn'.\n return this.defaultRenderOptions.features.map(function (feature) {\n if (feature.script === 'latn') {\n return {\n script: 'latn',\n tags: feature.tags.filter(function (tag) { return options[tag]; })\n };\n } else {\n return feature;\n }\n });\n };\n\n /**\n * Convert the given text to a list of Glyph objects.\n * Note that there is no strict one-to-one mapping between characters and\n * glyphs, so the list of returned glyphs can be larger or smaller than the\n * length of the given string.\n * @param {string}\n * @param {GlyphRenderOptions} [options]\n * @return {opentype.Glyph[]}\n */\n Font.prototype.stringToGlyphs = function(s, options) {\n var this$1 = this;\n\n\n var bidi = new Bidi();\n\n // Create and register 'glyphIndex' state modifier\n var charToGlyphIndexMod = function (token) { return this$1.charToGlyphIndex(token.char); };\n bidi.registerModifier('glyphIndex', null, charToGlyphIndexMod);\n\n // roll-back to default features\n var features = options ?\n this.updateFeatures(options.features) :\n this.defaultRenderOptions.features;\n\n bidi.applyFeatures(this, features);\n\n var indexes = bidi.getTextGlyphs(s);\n\n var length = indexes.length;\n\n // convert glyph indexes to glyph objects\n var glyphs = new Array(length);\n var notdef = this.glyphs.get(0);\n for (var i = 0; i < length; i += 1) {\n glyphs[i] = this.glyphs.get(indexes[i]) || notdef;\n }\n return glyphs;\n };\n\n /**\n * @param {string}\n * @return {Number}\n */\n Font.prototype.nameToGlyphIndex = function(name) {\n return this.glyphNames.nameToGlyphIndex(name);\n };\n\n /**\n * @param {string}\n * @return {opentype.Glyph}\n */\n Font.prototype.nameToGlyph = function(name) {\n var glyphIndex = this.nameToGlyphIndex(name);\n var glyph = this.glyphs.get(glyphIndex);\n if (!glyph) {\n // .notdef\n glyph = this.glyphs.get(0);\n }\n\n return glyph;\n };\n\n /**\n * @param {Number}\n * @return {String}\n */\n Font.prototype.glyphIndexToName = function(gid) {\n if (!this.glyphNames.glyphIndexToName) {\n return '';\n }\n\n return this.glyphNames.glyphIndexToName(gid);\n };\n\n /**\n * Retrieve the value of the kerning pair between the left glyph (or its index)\n * and the right glyph (or its index). If no kerning pair is found, return 0.\n * The kerning value gets added to the advance width when calculating the spacing\n * between glyphs.\n * For GPOS kerning, this method uses the default script and language, which covers\n * most use cases. To have greater control, use font.position.getKerningValue .\n * @param {opentype.Glyph} leftGlyph\n * @param {opentype.Glyph} rightGlyph\n * @return {Number}\n */\n Font.prototype.getKerningValue = function(leftGlyph, rightGlyph) {\n leftGlyph = leftGlyph.index || leftGlyph;\n rightGlyph = rightGlyph.index || rightGlyph;\n var gposKerning = this.position.defaultKerningTables;\n if (gposKerning) {\n return this.position.getKerningValue(gposKerning, leftGlyph, rightGlyph);\n }\n // \"kern\" table\n return this.kerningPairs[leftGlyph + ',' + rightGlyph] || 0;\n };\n\n /**\n * @typedef GlyphRenderOptions\n * @type Object\n * @property {string} [script] - script used to determine which features to apply. By default, 'DFLT' or 'latn' is used.\n * See https://www.microsoft.com/typography/otspec/scripttags.htm\n * @property {string} [language='dflt'] - language system used to determine which features to apply.\n * See https://www.microsoft.com/typography/developers/opentype/languagetags.aspx\n * @property {boolean} [kerning=true] - whether to include kerning values\n * @property {object} [features] - OpenType Layout feature tags. Used to enable or disable the features of the given script/language system.\n * See https://www.microsoft.com/typography/otspec/featuretags.htm\n */\n Font.prototype.defaultRenderOptions = {\n kerning: true,\n features: [\n /**\n * these 4 features are required to render Arabic text properly\n * and shouldn't be turned off when rendering arabic text.\n */\n { script: 'arab', tags: ['init', 'medi', 'fina', 'rlig'] },\n { script: 'latn', tags: ['liga', 'rlig'] }\n ]\n };\n\n /**\n * Helper function that invokes the given callback for each glyph in the given text.\n * The callback gets `(glyph, x, y, fontSize, options)`.* @param {string} text\n * @param {string} text - The text to apply.\n * @param {number} [x=0] - Horizontal position of the beginning of the text.\n * @param {number} [y=0] - Vertical position of the *baseline* of the text.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n * @param {GlyphRenderOptions=} options\n * @param {Function} callback\n */\n Font.prototype.forEachGlyph = function(text, x, y, fontSize, options, callback) {\n x = x !== undefined ? x : 0;\n y = y !== undefined ? y : 0;\n fontSize = fontSize !== undefined ? fontSize : 72;\n options = Object.assign({}, this.defaultRenderOptions, options);\n var fontScale = 1 / this.unitsPerEm * fontSize;\n var glyphs = this.stringToGlyphs(text, options);\n var kerningLookups;\n if (options.kerning) {\n var script = options.script || this.position.getDefaultScriptName();\n kerningLookups = this.position.getKerningTables(script, options.language);\n }\n for (var i = 0; i < glyphs.length; i += 1) {\n var glyph = glyphs[i];\n callback.call(this, glyph, x, y, fontSize, options);\n if (glyph.advanceWidth) {\n x += glyph.advanceWidth * fontScale;\n }\n\n if (options.kerning && i < glyphs.length - 1) {\n // We should apply position adjustment lookups in a more generic way.\n // Here we only use the xAdvance value.\n var kerningValue = kerningLookups ?\n this.position.getKerningValue(kerningLookups, glyph.index, glyphs[i + 1].index) :\n this.getKerningValue(glyph, glyphs[i + 1]);\n x += kerningValue * fontScale;\n }\n\n if (options.letterSpacing) {\n x += options.letterSpacing * fontSize;\n } else if (options.tracking) {\n x += (options.tracking / 1000) * fontSize;\n }\n }\n return x;\n };\n\n /**\n * Create a Path object that represents the given text.\n * @param {string} text - The text to create.\n * @param {number} [x=0] - Horizontal position of the beginning of the text.\n * @param {number} [y=0] - Vertical position of the *baseline* of the text.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n * @param {GlyphRenderOptions=} options\n * @return {opentype.Path}\n */\n Font.prototype.getPath = function(text, x, y, fontSize, options) {\n var fullPath = new Path();\n this.forEachGlyph(text, x, y, fontSize, options, function(glyph, gX, gY, gFontSize) {\n var glyphPath = glyph.getPath(gX, gY, gFontSize, options, this);\n fullPath.extend(glyphPath);\n });\n return fullPath;\n };\n\n /**\n * Create an array of Path objects that represent the glyphs of a given text.\n * @param {string} text - The text to create.\n * @param {number} [x=0] - Horizontal position of the beginning of the text.\n * @param {number} [y=0] - Vertical position of the *baseline* of the text.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n * @param {GlyphRenderOptions=} options\n * @return {opentype.Path[]}\n */\n Font.prototype.getPaths = function(text, x, y, fontSize, options) {\n var glyphPaths = [];\n this.forEachGlyph(text, x, y, fontSize, options, function(glyph, gX, gY, gFontSize) {\n var glyphPath = glyph.getPath(gX, gY, gFontSize, options, this);\n glyphPaths.push(glyphPath);\n });\n\n return glyphPaths;\n };\n\n /**\n * Returns the advance width of a text.\n *\n * This is something different than Path.getBoundingBox() as for example a\n * suffixed whitespace increases the advanceWidth but not the bounding box\n * or an overhanging letter like a calligraphic 'f' might have a quite larger\n * bounding box than its advance width.\n *\n * This corresponds to canvas2dContext.measureText(text).width\n *\n * @param {string} text - The text to create.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n * @param {GlyphRenderOptions=} options\n * @return advance width\n */\n Font.prototype.getAdvanceWidth = function(text, fontSize, options) {\n return this.forEachGlyph(text, 0, 0, fontSize, options, function() {});\n };\n\n /**\n * Draw the text on the given drawing context.\n * @param {CanvasRenderingContext2D} ctx - A 2D drawing context, like Canvas.\n * @param {string} text - The text to create.\n * @param {number} [x=0] - Horizontal position of the beginning of the text.\n * @param {number} [y=0] - Vertical position of the *baseline* of the text.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n * @param {GlyphRenderOptions=} options\n */\n Font.prototype.draw = function(ctx, text, x, y, fontSize, options) {\n this.getPath(text, x, y, fontSize, options).draw(ctx);\n };\n\n /**\n * Draw the points of all glyphs in the text.\n * On-curve points will be drawn in blue, off-curve points will be drawn in red.\n * @param {CanvasRenderingContext2D} ctx - A 2D drawing context, like Canvas.\n * @param {string} text - The text to create.\n * @param {number} [x=0] - Horizontal position of the beginning of the text.\n * @param {number} [y=0] - Vertical position of the *baseline* of the text.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n * @param {GlyphRenderOptions=} options\n */\n Font.prototype.drawPoints = function(ctx, text, x, y, fontSize, options) {\n this.forEachGlyph(text, x, y, fontSize, options, function(glyph, gX, gY, gFontSize) {\n glyph.drawPoints(ctx, gX, gY, gFontSize);\n });\n };\n\n /**\n * Draw lines indicating important font measurements for all glyphs in the text.\n * Black lines indicate the origin of the coordinate system (point 0,0).\n * Blue lines indicate the glyph bounding box.\n * Green line indicates the advance width of the glyph.\n * @param {CanvasRenderingContext2D} ctx - A 2D drawing context, like Canvas.\n * @param {string} text - The text to create.\n * @param {number} [x=0] - Horizontal position of the beginning of the text.\n * @param {number} [y=0] - Vertical position of the *baseline* of the text.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n * @param {GlyphRenderOptions=} options\n */\n Font.prototype.drawMetrics = function(ctx, text, x, y, fontSize, options) {\n this.forEachGlyph(text, x, y, fontSize, options, function(glyph, gX, gY, gFontSize) {\n glyph.drawMetrics(ctx, gX, gY, gFontSize);\n });\n };\n\n /**\n * @param {string}\n * @return {string}\n */\n Font.prototype.getEnglishName = function(name) {\n var translations = this.names[name];\n if (translations) {\n return translations.en;\n }\n };\n\n /**\n * Validate\n */\n Font.prototype.validate = function() {\n var _this = this;\n\n function assert(predicate, message) {\n }\n\n function assertNamePresent(name) {\n var englishName = _this.getEnglishName(name);\n assert(englishName && englishName.trim().length > 0);\n }\n\n // Identification information\n assertNamePresent('fontFamily');\n assertNamePresent('weightName');\n assertNamePresent('manufacturer');\n assertNamePresent('copyright');\n assertNamePresent('version');\n\n // Dimension information\n assert(this.unitsPerEm > 0);\n };\n\n /**\n * Convert the font object to a SFNT data structure.\n * This structure contains all the necessary tables and metadata to create a binary OTF file.\n * @return {opentype.Table}\n */\n Font.prototype.toTables = function() {\n return sfnt.fontToTable(this);\n };\n /**\n * @deprecated Font.toBuffer is deprecated. Use Font.toArrayBuffer instead.\n */\n Font.prototype.toBuffer = function() {\n console.warn('Font.toBuffer is deprecated. Use Font.toArrayBuffer instead.');\n return this.toArrayBuffer();\n };\n /**\n * Converts a `opentype.Font` into an `ArrayBuffer`\n * @return {ArrayBuffer}\n */\n Font.prototype.toArrayBuffer = function() {\n var sfntTable = this.toTables();\n var bytes = sfntTable.encode();\n var buffer = new ArrayBuffer(bytes.length);\n var intArray = new Uint8Array(buffer);\n for (var i = 0; i < bytes.length; i++) {\n intArray[i] = bytes[i];\n }\n\n return buffer;\n };\n\n /**\n * Initiate a download of the OpenType font.\n */\n Font.prototype.download = function(fileName) {\n var familyName = this.getEnglishName('fontFamily');\n var styleName = this.getEnglishName('fontSubfamily');\n fileName = fileName || familyName.replace(/\\s/g, '') + '-' + styleName + '.otf';\n var arrayBuffer = this.toArrayBuffer();\n\n if (isBrowser()) {\n window.URL = window.URL || window.webkitURL;\n\n if (window.URL) {\n var dataView = new DataView(arrayBuffer);\n var blob = new Blob([dataView], {type: 'font/opentype'});\n\n var link = document.createElement('a');\n link.href = window.URL.createObjectURL(blob);\n link.download = fileName;\n\n var event = document.createEvent('MouseEvents');\n event.initEvent('click', true, false);\n link.dispatchEvent(event);\n } else {\n console.warn('Font file could not be downloaded. Try using a different browser.');\n }\n } else {\n var fs = require('fs');\n var buffer = arrayBufferToNodeBuffer(arrayBuffer);\n fs.writeFileSync(fileName, buffer);\n }\n };\n /**\n * @private\n */\n Font.prototype.fsSelectionValues = {\n ITALIC: 0x001, //1\n UNDERSCORE: 0x002, //2\n NEGATIVE: 0x004, //4\n OUTLINED: 0x008, //8\n STRIKEOUT: 0x010, //16\n BOLD: 0x020, //32\n REGULAR: 0x040, //64\n USER_TYPO_METRICS: 0x080, //128\n WWS: 0x100, //256\n OBLIQUE: 0x200 //512\n };\n\n /**\n * @private\n */\n Font.prototype.usWidthClasses = {\n ULTRA_CONDENSED: 1,\n EXTRA_CONDENSED: 2,\n CONDENSED: 3,\n SEMI_CONDENSED: 4,\n MEDIUM: 5,\n SEMI_EXPANDED: 6,\n EXPANDED: 7,\n EXTRA_EXPANDED: 8,\n ULTRA_EXPANDED: 9\n };\n\n /**\n * @private\n */\n Font.prototype.usWeightClasses = {\n THIN: 100,\n EXTRA_LIGHT: 200,\n LIGHT: 300,\n NORMAL: 400,\n MEDIUM: 500,\n SEMI_BOLD: 600,\n BOLD: 700,\n EXTRA_BOLD: 800,\n BLACK: 900\n };\n\n // The `fvar` table stores font variation axes and instances.\n\n function addName(name, names) {\n var nameString = JSON.stringify(name);\n var nameID = 256;\n for (var nameKey in names) {\n var n = parseInt(nameKey);\n if (!n || n < 256) {\n continue;\n }\n\n if (JSON.stringify(names[nameKey]) === nameString) {\n return n;\n }\n\n if (nameID <= n) {\n nameID = n + 1;\n }\n }\n\n names[nameID] = name;\n return nameID;\n }\n\n function makeFvarAxis(n, axis, names) {\n var nameID = addName(axis.name, names);\n return [\n {name: 'tag_' + n, type: 'TAG', value: axis.tag},\n {name: 'minValue_' + n, type: 'FIXED', value: axis.minValue << 16},\n {name: 'defaultValue_' + n, type: 'FIXED', value: axis.defaultValue << 16},\n {name: 'maxValue_' + n, type: 'FIXED', value: axis.maxValue << 16},\n {name: 'flags_' + n, type: 'USHORT', value: 0},\n {name: 'nameID_' + n, type: 'USHORT', value: nameID}\n ];\n }\n\n function parseFvarAxis(data, start, names) {\n var axis = {};\n var p = new parse.Parser(data, start);\n axis.tag = p.parseTag();\n axis.minValue = p.parseFixed();\n axis.defaultValue = p.parseFixed();\n axis.maxValue = p.parseFixed();\n p.skip('uShort', 1); // reserved for flags; no values defined\n axis.name = names[p.parseUShort()] || {};\n return axis;\n }\n\n function makeFvarInstance(n, inst, axes, names) {\n var nameID = addName(inst.name, names);\n var fields = [\n {name: 'nameID_' + n, type: 'USHORT', value: nameID},\n {name: 'flags_' + n, type: 'USHORT', value: 0}\n ];\n\n for (var i = 0; i < axes.length; ++i) {\n var axisTag = axes[i].tag;\n fields.push({\n name: 'axis_' + n + ' ' + axisTag,\n type: 'FIXED',\n value: inst.coordinates[axisTag] << 16\n });\n }\n\n return fields;\n }\n\n function parseFvarInstance(data, start, axes, names) {\n var inst = {};\n var p = new parse.Parser(data, start);\n inst.name = names[p.parseUShort()] || {};\n p.skip('uShort', 1); // reserved for flags; no values defined\n\n inst.coordinates = {};\n for (var i = 0; i < axes.length; ++i) {\n inst.coordinates[axes[i].tag] = p.parseFixed();\n }\n\n return inst;\n }\n\n function makeFvarTable(fvar, names) {\n var result = new table.Table('fvar', [\n {name: 'version', type: 'ULONG', value: 0x10000},\n {name: 'offsetToData', type: 'USHORT', value: 0},\n {name: 'countSizePairs', type: 'USHORT', value: 2},\n {name: 'axisCount', type: 'USHORT', value: fvar.axes.length},\n {name: 'axisSize', type: 'USHORT', value: 20},\n {name: 'instanceCount', type: 'USHORT', value: fvar.instances.length},\n {name: 'instanceSize', type: 'USHORT', value: 4 + fvar.axes.length * 4}\n ]);\n result.offsetToData = result.sizeOf();\n\n for (var i = 0; i < fvar.axes.length; i++) {\n result.fields = result.fields.concat(makeFvarAxis(i, fvar.axes[i], names));\n }\n\n for (var j = 0; j < fvar.instances.length; j++) {\n result.fields = result.fields.concat(makeFvarInstance(j, fvar.instances[j], fvar.axes, names));\n }\n\n return result;\n }\n\n function parseFvarTable(data, start, names) {\n var p = new parse.Parser(data, start);\n var tableVersion = p.parseULong();\n check.argument(tableVersion === 0x00010000, 'Unsupported fvar table version.');\n var offsetToData = p.parseOffset16();\n // Skip countSizePairs.\n p.skip('uShort', 1);\n var axisCount = p.parseUShort();\n var axisSize = p.parseUShort();\n var instanceCount = p.parseUShort();\n var instanceSize = p.parseUShort();\n\n var axes = [];\n for (var i = 0; i < axisCount; i++) {\n axes.push(parseFvarAxis(data, start + offsetToData + i * axisSize, names));\n }\n\n var instances = [];\n var instanceStart = start + offsetToData + axisCount * axisSize;\n for (var j = 0; j < instanceCount; j++) {\n instances.push(parseFvarInstance(data, instanceStart + j * instanceSize, axes, names));\n }\n\n return {axes: axes, instances: instances};\n }\n\n var fvar = { make: makeFvarTable, parse: parseFvarTable };\n\n // The `GDEF` table contains various glyph properties\n\n var attachList = function() {\n return {\n coverage: this.parsePointer(Parser.coverage),\n attachPoints: this.parseList(Parser.pointer(Parser.uShortList))\n };\n };\n\n var caretValue = function() {\n var format = this.parseUShort();\n check.argument(format === 1 || format === 2 || format === 3,\n 'Unsupported CaretValue table version.');\n if (format === 1) {\n return { coordinate: this.parseShort() };\n } else if (format === 2) {\n return { pointindex: this.parseShort() };\n } else if (format === 3) {\n // Device / Variation Index tables unsupported\n return { coordinate: this.parseShort() };\n }\n };\n\n var ligGlyph = function() {\n return this.parseList(Parser.pointer(caretValue));\n };\n\n var ligCaretList = function() {\n return {\n coverage: this.parsePointer(Parser.coverage),\n ligGlyphs: this.parseList(Parser.pointer(ligGlyph))\n };\n };\n\n var markGlyphSets = function() {\n this.parseUShort(); // Version\n return this.parseList(Parser.pointer(Parser.coverage));\n };\n\n function parseGDEFTable(data, start) {\n start = start || 0;\n var p = new Parser(data, start);\n var tableVersion = p.parseVersion(1);\n check.argument(tableVersion === 1 || tableVersion === 1.2 || tableVersion === 1.3,\n 'Unsupported GDEF table version.');\n var gdef = {\n version: tableVersion,\n classDef: p.parsePointer(Parser.classDef),\n attachList: p.parsePointer(attachList),\n ligCaretList: p.parsePointer(ligCaretList),\n markAttachClassDef: p.parsePointer(Parser.classDef)\n };\n if (tableVersion >= 1.2) {\n gdef.markGlyphSets = p.parsePointer(markGlyphSets);\n }\n return gdef;\n }\n var gdef = { parse: parseGDEFTable };\n\n // The `GPOS` table contains kerning pairs, among other things.\n\n var subtableParsers$1 = new Array(10); // subtableParsers[0] is unused\n\n // https://docs.microsoft.com/en-us/typography/opentype/spec/gpos#lookup-type-1-single-adjustment-positioning-subtable\n // this = Parser instance\n subtableParsers$1[1] = function parseLookup1() {\n var start = this.offset + this.relativeOffset;\n var posformat = this.parseUShort();\n if (posformat === 1) {\n return {\n posFormat: 1,\n coverage: this.parsePointer(Parser.coverage),\n value: this.parseValueRecord()\n };\n } else if (posformat === 2) {\n return {\n posFormat: 2,\n coverage: this.parsePointer(Parser.coverage),\n values: this.parseValueRecordList()\n };\n }\n check.assert(false, '0x' + start.toString(16) + ': GPOS lookup type 1 format must be 1 or 2.');\n };\n\n // https://docs.microsoft.com/en-us/typography/opentype/spec/gpos#lookup-type-2-pair-adjustment-positioning-subtable\n subtableParsers$1[2] = function parseLookup2() {\n var start = this.offset + this.relativeOffset;\n var posFormat = this.parseUShort();\n check.assert(posFormat === 1 || posFormat === 2, '0x' + start.toString(16) + ': GPOS lookup type 2 format must be 1 or 2.');\n var coverage = this.parsePointer(Parser.coverage);\n var valueFormat1 = this.parseUShort();\n var valueFormat2 = this.parseUShort();\n if (posFormat === 1) {\n // Adjustments for Glyph Pairs\n return {\n posFormat: posFormat,\n coverage: coverage,\n valueFormat1: valueFormat1,\n valueFormat2: valueFormat2,\n pairSets: this.parseList(Parser.pointer(Parser.list(function() {\n return { // pairValueRecord\n secondGlyph: this.parseUShort(),\n value1: this.parseValueRecord(valueFormat1),\n value2: this.parseValueRecord(valueFormat2)\n };\n })))\n };\n } else if (posFormat === 2) {\n var classDef1 = this.parsePointer(Parser.classDef);\n var classDef2 = this.parsePointer(Parser.classDef);\n var class1Count = this.parseUShort();\n var class2Count = this.parseUShort();\n return {\n // Class Pair Adjustment\n posFormat: posFormat,\n coverage: coverage,\n valueFormat1: valueFormat1,\n valueFormat2: valueFormat2,\n classDef1: classDef1,\n classDef2: classDef2,\n class1Count: class1Count,\n class2Count: class2Count,\n classRecords: this.parseList(class1Count, Parser.list(class2Count, function() {\n return {\n value1: this.parseValueRecord(valueFormat1),\n value2: this.parseValueRecord(valueFormat2)\n };\n }))\n };\n }\n };\n\n subtableParsers$1[3] = function parseLookup3() { return { error: 'GPOS Lookup 3 not supported' }; };\n subtableParsers$1[4] = function parseLookup4() { return { error: 'GPOS Lookup 4 not supported' }; };\n subtableParsers$1[5] = function parseLookup5() { return { error: 'GPOS Lookup 5 not supported' }; };\n subtableParsers$1[6] = function parseLookup6() { return { error: 'GPOS Lookup 6 not supported' }; };\n subtableParsers$1[7] = function parseLookup7() { return { error: 'GPOS Lookup 7 not supported' }; };\n subtableParsers$1[8] = function parseLookup8() { return { error: 'GPOS Lookup 8 not supported' }; };\n subtableParsers$1[9] = function parseLookup9() { return { error: 'GPOS Lookup 9 not supported' }; };\n\n // https://docs.microsoft.com/en-us/typography/opentype/spec/gpos\n function parseGposTable(data, start) {\n start = start || 0;\n var p = new Parser(data, start);\n var tableVersion = p.parseVersion(1);\n check.argument(tableVersion === 1 || tableVersion === 1.1, 'Unsupported GPOS table version ' + tableVersion);\n\n if (tableVersion === 1) {\n return {\n version: tableVersion,\n scripts: p.parseScriptList(),\n features: p.parseFeatureList(),\n lookups: p.parseLookupList(subtableParsers$1)\n };\n } else {\n return {\n version: tableVersion,\n scripts: p.parseScriptList(),\n features: p.parseFeatureList(),\n lookups: p.parseLookupList(subtableParsers$1),\n variations: p.parseFeatureVariationsList()\n };\n }\n\n }\n\n // GPOS Writing //////////////////////////////////////////////\n // NOT SUPPORTED\n var subtableMakers$1 = new Array(10);\n\n function makeGposTable(gpos) {\n return new table.Table('GPOS', [\n {name: 'version', type: 'ULONG', value: 0x10000},\n {name: 'scripts', type: 'TABLE', value: new table.ScriptList(gpos.scripts)},\n {name: 'features', type: 'TABLE', value: new table.FeatureList(gpos.features)},\n {name: 'lookups', type: 'TABLE', value: new table.LookupList(gpos.lookups, subtableMakers$1)}\n ]);\n }\n\n var gpos = { parse: parseGposTable, make: makeGposTable };\n\n // The `kern` table contains kerning pairs.\n\n function parseWindowsKernTable(p) {\n var pairs = {};\n // Skip nTables.\n p.skip('uShort');\n var subtableVersion = p.parseUShort();\n check.argument(subtableVersion === 0, 'Unsupported kern sub-table version.');\n // Skip subtableLength, subtableCoverage\n p.skip('uShort', 2);\n var nPairs = p.parseUShort();\n // Skip searchRange, entrySelector, rangeShift.\n p.skip('uShort', 3);\n for (var i = 0; i < nPairs; i += 1) {\n var leftIndex = p.parseUShort();\n var rightIndex = p.parseUShort();\n var value = p.parseShort();\n pairs[leftIndex + ',' + rightIndex] = value;\n }\n return pairs;\n }\n\n function parseMacKernTable(p) {\n var pairs = {};\n // The Mac kern table stores the version as a fixed (32 bits) but we only loaded the first 16 bits.\n // Skip the rest.\n p.skip('uShort');\n var nTables = p.parseULong();\n //check.argument(nTables === 1, 'Only 1 subtable is supported (got ' + nTables + ').');\n if (nTables > 1) {\n console.warn('Only the first kern subtable is supported.');\n }\n p.skip('uLong');\n var coverage = p.parseUShort();\n var subtableVersion = coverage & 0xFF;\n p.skip('uShort');\n if (subtableVersion === 0) {\n var nPairs = p.parseUShort();\n // Skip searchRange, entrySelector, rangeShift.\n p.skip('uShort', 3);\n for (var i = 0; i < nPairs; i += 1) {\n var leftIndex = p.parseUShort();\n var rightIndex = p.parseUShort();\n var value = p.parseShort();\n pairs[leftIndex + ',' + rightIndex] = value;\n }\n }\n return pairs;\n }\n\n // Parse the `kern` table which contains kerning pairs.\n function parseKernTable(data, start) {\n var p = new parse.Parser(data, start);\n var tableVersion = p.parseUShort();\n if (tableVersion === 0) {\n return parseWindowsKernTable(p);\n } else if (tableVersion === 1) {\n return parseMacKernTable(p);\n } else {\n throw new Error('Unsupported kern table version (' + tableVersion + ').');\n }\n }\n\n var kern = { parse: parseKernTable };\n\n // The `loca` table stores the offsets to the locations of the glyphs in the font.\n\n // Parse the `loca` table. This table stores the offsets to the locations of the glyphs in the font,\n // relative to the beginning of the glyphData table.\n // The number of glyphs stored in the `loca` table is specified in the `maxp` table (under numGlyphs)\n // The loca table has two versions: a short version where offsets are stored as uShorts, and a long\n // version where offsets are stored as uLongs. The `head` table specifies which version to use\n // (under indexToLocFormat).\n function parseLocaTable(data, start, numGlyphs, shortVersion) {\n var p = new parse.Parser(data, start);\n var parseFn = shortVersion ? p.parseUShort : p.parseULong;\n // There is an extra entry after the last index element to compute the length of the last glyph.\n // That's why we use numGlyphs + 1.\n var glyphOffsets = [];\n for (var i = 0; i < numGlyphs + 1; i += 1) {\n var glyphOffset = parseFn.call(p);\n if (shortVersion) {\n // The short table version stores the actual offset divided by 2.\n glyphOffset *= 2;\n }\n\n glyphOffsets.push(glyphOffset);\n }\n\n return glyphOffsets;\n }\n\n var loca = { parse: parseLocaTable };\n\n // Table Directory Entries //////////////////////////////////////////////\n /**\n * Parses OpenType table entries.\n * @param {DataView}\n * @param {Number}\n * @return {Object[]}\n */\n function parseOpenTypeTableEntries(data, numTables) {\n var tableEntries = [];\n var p = 12;\n for (var i = 0; i < numTables; i += 1) {\n var tag = parse.getTag(data, p);\n var checksum = parse.getULong(data, p + 4);\n var offset = parse.getULong(data, p + 8);\n var length = parse.getULong(data, p + 12);\n tableEntries.push({tag: tag, checksum: checksum, offset: offset, length: length, compression: false});\n p += 16;\n }\n\n return tableEntries;\n }\n\n /**\n * Parses WOFF table entries.\n * @param {DataView}\n * @param {Number}\n * @return {Object[]}\n */\n function parseWOFFTableEntries(data, numTables) {\n var tableEntries = [];\n var p = 44; // offset to the first table directory entry.\n for (var i = 0; i < numTables; i += 1) {\n var tag = parse.getTag(data, p);\n var offset = parse.getULong(data, p + 4);\n var compLength = parse.getULong(data, p + 8);\n var origLength = parse.getULong(data, p + 12);\n var compression = (void 0);\n if (compLength < origLength) {\n compression = 'WOFF';\n } else {\n compression = false;\n }\n\n tableEntries.push({tag: tag, offset: offset, compression: compression,\n compressedLength: compLength, length: origLength});\n p += 20;\n }\n\n return tableEntries;\n }\n\n /**\n * @typedef TableData\n * @type Object\n * @property {DataView} data - The DataView\n * @property {number} offset - The data offset.\n */\n\n /**\n * @param {DataView}\n * @param {Object}\n * @return {TableData}\n */\n function uncompressTable(data, tableEntry) {\n if (tableEntry.compression === 'WOFF') {\n var inBuffer = new Uint8Array(data.buffer, tableEntry.offset + 2, tableEntry.compressedLength - 2);\n var outBuffer = new Uint8Array(tableEntry.length);\n tinyInflate(inBuffer, outBuffer);\n if (outBuffer.byteLength !== tableEntry.length) {\n throw new Error('Decompression error: ' + tableEntry.tag + ' decompressed length doesn\\'t match recorded length');\n }\n\n var view = new DataView(outBuffer.buffer, 0);\n return {data: view, offset: 0};\n } else {\n return {data: data, offset: tableEntry.offset};\n }\n }\n\n // Public API ///////////////////////////////////////////////////////////\n\n /**\n * Parse the OpenType file data (as an ArrayBuffer) and return a Font object.\n * Throws an error if the font could not be parsed.\n * @param {ArrayBuffer}\n * @param {Object} opt - options for parsing\n * @return {opentype.Font}\n */\n function parseBuffer(buffer, opt) {\n opt = (opt === undefined || opt === null) ? {} : opt;\n\n var indexToLocFormat;\n var ltagTable;\n\n // Since the constructor can also be called to create new fonts from scratch, we indicate this\n // should be an empty font that we'll fill with our own data.\n var font = new Font({empty: true});\n\n // OpenType fonts use big endian byte ordering.\n // We can't rely on typed array view types, because they operate with the endianness of the host computer.\n // Instead we use DataViews where we can specify endianness.\n var data = new DataView(buffer, 0);\n var numTables;\n var tableEntries = [];\n var signature = parse.getTag(data, 0);\n if (signature === String.fromCharCode(0, 1, 0, 0) || signature === 'true' || signature === 'typ1') {\n font.outlinesFormat = 'truetype';\n numTables = parse.getUShort(data, 4);\n tableEntries = parseOpenTypeTableEntries(data, numTables);\n } else if (signature === 'OTTO') {\n font.outlinesFormat = 'cff';\n numTables = parse.getUShort(data, 4);\n tableEntries = parseOpenTypeTableEntries(data, numTables);\n } else if (signature === 'wOFF') {\n var flavor = parse.getTag(data, 4);\n if (flavor === String.fromCharCode(0, 1, 0, 0)) {\n font.outlinesFormat = 'truetype';\n } else if (flavor === 'OTTO') {\n font.outlinesFormat = 'cff';\n } else {\n throw new Error('Unsupported OpenType flavor ' + signature);\n }\n\n numTables = parse.getUShort(data, 12);\n tableEntries = parseWOFFTableEntries(data, numTables);\n } else {\n throw new Error('Unsupported OpenType signature ' + signature);\n }\n\n var cffTableEntry;\n var fvarTableEntry;\n var glyfTableEntry;\n var gdefTableEntry;\n var gposTableEntry;\n var gsubTableEntry;\n var hmtxTableEntry;\n var kernTableEntry;\n var locaTableEntry;\n var nameTableEntry;\n var metaTableEntry;\n var p;\n\n for (var i = 0; i < numTables; i += 1) {\n var tableEntry = tableEntries[i];\n var table = (void 0);\n switch (tableEntry.tag) {\n case 'cmap':\n table = uncompressTable(data, tableEntry);\n font.tables.cmap = cmap.parse(table.data, table.offset);\n font.encoding = new CmapEncoding(font.tables.cmap);\n break;\n case 'cvt ' :\n table = uncompressTable(data, tableEntry);\n p = new parse.Parser(table.data, table.offset);\n font.tables.cvt = p.parseShortList(tableEntry.length / 2);\n break;\n case 'fvar':\n fvarTableEntry = tableEntry;\n break;\n case 'fpgm' :\n table = uncompressTable(data, tableEntry);\n p = new parse.Parser(table.data, table.offset);\n font.tables.fpgm = p.parseByteList(tableEntry.length);\n break;\n case 'head':\n table = uncompressTable(data, tableEntry);\n font.tables.head = head.parse(table.data, table.offset);\n font.unitsPerEm = font.tables.head.unitsPerEm;\n indexToLocFormat = font.tables.head.indexToLocFormat;\n break;\n case 'hhea':\n table = uncompressTable(data, tableEntry);\n font.tables.hhea = hhea.parse(table.data, table.offset);\n font.ascender = font.tables.hhea.ascender;\n font.descender = font.tables.hhea.descender;\n font.numberOfHMetrics = font.tables.hhea.numberOfHMetrics;\n break;\n case 'hmtx':\n hmtxTableEntry = tableEntry;\n break;\n case 'ltag':\n table = uncompressTable(data, tableEntry);\n ltagTable = ltag.parse(table.data, table.offset);\n break;\n case 'maxp':\n table = uncompressTable(data, tableEntry);\n font.tables.maxp = maxp.parse(table.data, table.offset);\n font.numGlyphs = font.tables.maxp.numGlyphs;\n break;\n case 'name':\n nameTableEntry = tableEntry;\n break;\n case 'OS/2':\n table = uncompressTable(data, tableEntry);\n font.tables.os2 = os2.parse(table.data, table.offset);\n break;\n case 'post':\n table = uncompressTable(data, tableEntry);\n font.tables.post = post.parse(table.data, table.offset);\n font.glyphNames = new GlyphNames(font.tables.post);\n break;\n case 'prep' :\n table = uncompressTable(data, tableEntry);\n p = new parse.Parser(table.data, table.offset);\n font.tables.prep = p.parseByteList(tableEntry.length);\n break;\n case 'glyf':\n glyfTableEntry = tableEntry;\n break;\n case 'loca':\n locaTableEntry = tableEntry;\n break;\n case 'CFF ':\n cffTableEntry = tableEntry;\n break;\n case 'kern':\n kernTableEntry = tableEntry;\n break;\n case 'GDEF':\n gdefTableEntry = tableEntry;\n break;\n case 'GPOS':\n gposTableEntry = tableEntry;\n break;\n case 'GSUB':\n gsubTableEntry = tableEntry;\n break;\n case 'meta':\n metaTableEntry = tableEntry;\n break;\n }\n }\n\n var nameTable = uncompressTable(data, nameTableEntry);\n font.tables.name = _name.parse(nameTable.data, nameTable.offset, ltagTable);\n font.names = font.tables.name;\n\n if (glyfTableEntry && locaTableEntry) {\n var shortVersion = indexToLocFormat === 0;\n var locaTable = uncompressTable(data, locaTableEntry);\n var locaOffsets = loca.parse(locaTable.data, locaTable.offset, font.numGlyphs, shortVersion);\n var glyfTable = uncompressTable(data, glyfTableEntry);\n font.glyphs = glyf.parse(glyfTable.data, glyfTable.offset, locaOffsets, font, opt);\n } else if (cffTableEntry) {\n var cffTable = uncompressTable(data, cffTableEntry);\n cff.parse(cffTable.data, cffTable.offset, font, opt);\n } else {\n throw new Error('Font doesn\\'t contain TrueType or CFF outlines.');\n }\n\n var hmtxTable = uncompressTable(data, hmtxTableEntry);\n hmtx.parse(font, hmtxTable.data, hmtxTable.offset, font.numberOfHMetrics, font.numGlyphs, font.glyphs, opt);\n addGlyphNames(font, opt);\n\n if (kernTableEntry) {\n var kernTable = uncompressTable(data, kernTableEntry);\n font.kerningPairs = kern.parse(kernTable.data, kernTable.offset);\n } else {\n font.kerningPairs = {};\n }\n\n if (gdefTableEntry) {\n var gdefTable = uncompressTable(data, gdefTableEntry);\n font.tables.gdef = gdef.parse(gdefTable.data, gdefTable.offset);\n }\n\n if (gposTableEntry) {\n var gposTable = uncompressTable(data, gposTableEntry);\n font.tables.gpos = gpos.parse(gposTable.data, gposTable.offset);\n font.position.init();\n }\n\n if (gsubTableEntry) {\n var gsubTable = uncompressTable(data, gsubTableEntry);\n font.tables.gsub = gsub.parse(gsubTable.data, gsubTable.offset);\n }\n\n if (fvarTableEntry) {\n var fvarTable = uncompressTable(data, fvarTableEntry);\n font.tables.fvar = fvar.parse(fvarTable.data, fvarTable.offset, font.names);\n }\n\n if (metaTableEntry) {\n var metaTable = uncompressTable(data, metaTableEntry);\n font.tables.meta = meta.parse(metaTable.data, metaTable.offset);\n font.metas = font.tables.meta;\n }\n\n return font;\n }\n\n const primitiveShapeProperties = new Set([\n 'height',\n 'width',\n 'depth',\n 'radius',\n 'radiusTop',\n 'radiusBottom',\n 'thickness',\n 'arc',\n 'latitudeStart',\n 'latitudeSlice',\n 'tessellation',\n 'subdivisions',\n 'capMode',\n 'segments',\n ]);\n const PIVOT_GIZMO_SCALE = 0.33;\n const capModeMap = {\n [exports.eCylinderCap.both]: Mesh.CAP_ALL,\n [exports.eCylinderCap.start]: Mesh.CAP_START,\n [exports.eCylinderCap.end]: Mesh.CAP_END,\n [exports.eCylinderCap.none]: Mesh.NO_CAP,\n };\n const textMeshFonts = new Map();\n // const lineColor = new Color4(1, 1, 1);\n // const selectedLineColor = new Color4(1, 0, 0);\n class MeshRenderer extends Renderer {\n constructor(viewer, scene, rendererManager, shadowManager) {\n super(viewer, scene);\n this.rendererManager = rendererManager;\n this.shadowManager = shadowManager;\n this.disposed = new Subject();\n this.pivotGizmos = new Map();\n this.sketchPreviews = new Map();\n this.sketchTracings = new Map();\n this.sketchPlanePreviews = new Map();\n this.modelToken = Token.SpaceNode;\n this.utilityLayer = viewer.utilityLayer;\n const tempVector = new Vector3();\n viewer.afterRender.pipe(takeUntil(this.disposed)).subscribe(() => {\n let dist;\n const camera = this.utilityLayer.utilityLayerScene.activeCamera;\n this.pivotGizmos.forEach(gizmo => {\n if (camera.mode === Camera.ORTHOGRAPHIC_CAMERA) {\n dist = ((camera.orthoRight || 0) - (camera.orthoLeft || 0)) / 2;\n }\n else {\n const cameraPosition = camera.globalPosition;\n gizmo.position.subtractToRef(cameraPosition, tempVector);\n dist = tempVector.length() * PIVOT_GIZMO_SCALE;\n }\n gizmo.scaling.set(dist, dist, dist);\n });\n });\n }\n handlesNode(node) {\n return isMeshNode(node) || isMeshGroupNode(node) || (isSceneNode(node) && node.isNested());\n }\n setRenderMode(mode, sceneNode) {\n this.renderMode = mode;\n this.babylonMap.forEach(mesh => {\n if (mesh instanceof Mesh) {\n this.setRenderModeOnMesh(mesh, sceneNode);\n }\n });\n }\n dispose() {\n super.dispose();\n this.pivotGizmos.forEach(gizmo => {\n gizmo.dispose();\n });\n this.pivotGizmos.clear();\n this.disposed.next();\n }\n loadNewGeometry(node) {\n node._loading = node.geometryId;\n const meshService = node._manager.getService(Token.MeshService);\n meshService.hydrateGeometry(node).then(() => {\n node._loading = null;\n this.invalidateBoundingBox(node);\n });\n }\n parentsVisible(node) {\n while (node._parent && node._parent instanceof exports.SpaceNode) {\n if (!node._parent.visible) {\n return false;\n }\n node = node._parent;\n }\n return true;\n }\n update3dInternal(node) {\n const isNewMesh = !this.get3d(node);\n if (isCustomMeshNode(node)) {\n if (node.visible &&\n this.parentsVisible(node) &&\n node.hasGeometry &&\n !node._geometry &&\n !node.geometryLoaded &&\n !node._loading) {\n this.loadNewGeometry(node);\n }\n }\n const mesh = this.createOrUpdateMesh(node);\n const pivotGizmo = this.pivotGizmos.get(node._dynamicId);\n if (node._selected && mesh instanceof Mesh && !this.viewer.highlighter.has(node)) {\n this.viewer.highlighter.add(node, exports.KbColor.Red());\n if (pivotGizmo) {\n pivotGizmo.setEnabled(true);\n }\n }\n else if (!node._selected && mesh instanceof Mesh && this.viewer.highlighter.has(node)) {\n this.viewer.highlighter.remove(node);\n if (pivotGizmo) {\n pivotGizmo.setEnabled(false);\n }\n }\n if (node instanceof exports.SketchNode) {\n if (node._selected && node.visible) {\n const oldPreview = this.sketchPlanePreviews.get(node.id);\n if (oldPreview) {\n oldPreview.dispose();\n }\n const mesh = this.createSketchPlaneVisualization(node);\n if (node.visible && mesh) {\n this.sketchPlanePreviews.set(node.id, mesh);\n }\n else {\n mesh === null || mesh === void 0 ? void 0 : mesh.dispose();\n this.sketchPlanePreviews.delete(node.id);\n }\n }\n else if (!node._selected || !node.visible) {\n const SketchPlancePreviewMesh = this.sketchPlanePreviews.get(node.id);\n SketchPlancePreviewMesh === null || SketchPlancePreviewMesh === void 0 ? void 0 : SketchPlancePreviewMesh.dispose();\n this.sketchPlanePreviews.delete(node.id);\n }\n }\n if (!mesh) {\n throw Error('Could not create mesh');\n }\n if (isNewMesh || node.changes.has('_parent')) {\n this.updateParentMeshFromNode(node, mesh, isNewMesh);\n }\n }\n rebuildBaseGeometry(node, preTransformChanged = false) {\n const mesh = this.get3d(node);\n if (mesh && mesh instanceof Mesh) {\n let meshVertexData = null;\n if (isPrimitiveMeshNode(node)) {\n const newMesh = this.createPrimitiveMesh(node);\n meshVertexData = SubmeshVertexData.ExtractFromMesh(newMesh);\n newMesh.dispose();\n if (isDefined(node.uvOverrides)) {\n this.setBabylonUvData(node.uvOverrides, meshVertexData);\n }\n if (node instanceof exports.MeshNode && meshVertexData) {\n // Always pretransform primitives as they are always rebuilt from scratch\n this.preTransformVertices(meshVertexData, node);\n }\n }\n else if (isCustomMeshNode(node)) {\n // If custom geometry does not have a geometry, don't do anything. Return null.\n if (node._geometry && node._geometry.positions) {\n meshVertexData = this.getMeshVertexData(node);\n }\n }\n else if (isSketchNode(node)) {\n meshVertexData = this.buildSketchGeometry(node);\n }\n else if (isTextNode(node) && isTrackedClass(node)) {\n meshVertexData = this.buildTextGeometry(node);\n }\n else {\n throw Error('geometry needs to be recalculated but mesh is not a valid primitive or custom mesh');\n }\n return meshVertexData;\n }\n return null;\n }\n buildSketchGeometry(node) {\n const result = new SubmeshVertexData();\n const positionVectors = [];\n let pathSegmentIndices = [];\n // Keep track of the path groups so we can match the hole paths with the groups in the vertex data\n // The key is the path group key, and the value is the indices of the generated polygons that are part of that path\n // One connected path may have multiple sketch paths because it requires different types of curves\n const pathGroupKeys = new Map();\n const holePathsReferencingKeys = [];\n let currentPath;\n let lastPoint;\n for (let i = 0; i < node.paths.length; i++) {\n const path = node.paths[i];\n if (path._points && path._points.length > 0) {\n let thisPoint;\n for (let iP = 0; iP < path._points.length; iP++) {\n const { x, y, z } = path._points[iP];\n thisPoint = new Vector3(x, y, z);\n // Separate paths that don't have a shared start/end to different logical paths\n if (!lastPoint || !thisPoint.equals(lastPoint)) {\n if (iP === 0 || !currentPath) {\n if (!pathGroupKeys.has(path.pathGroupKey)) {\n pathGroupKeys.set(path.pathGroupKey, pathSegmentIndices.length);\n }\n if (path.holeForPathGroup) {\n holePathsReferencingKeys[pathSegmentIndices.length] = path.holeForPathGroup;\n }\n currentPath = [];\n pathSegmentIndices.push(currentPath);\n }\n currentPath.push(positionVectors.length);\n positionVectors.push(thisPoint);\n }\n lastPoint = thisPoint;\n }\n }\n }\n if (positionVectors.length === 0 || pathSegmentIndices.length === 0) {\n return null;\n }\n const positions = [];\n const indices = [];\n const normals = [];\n const paths = [];\n if (node.flipPath) {\n pathSegmentIndices.reverse();\n pathSegmentIndices.forEach(segment => segment.reverse());\n }\n const primaryNormal = axisToVector3(node.normal);\n // /**\n // * We want the sketch to always \"face\" in the positive direction of the axis for consistency\n // * Otherwise it's indeterminate which way the sketch will face and makes the sketch hard to manage\n // * because it depends on the handed-ness of the points (clockwise or counter-clockwise)\n // */\n pathSegmentIndices.forEach(segment => {\n const segmentPositions = [];\n for (let i = 0; i < segment.length; i++) {\n const point = positionVectors[segment[i]];\n segmentPositions.push(point);\n }\n // However, don't flip non-closed paths because it makes no sense and may be a drive curve\n const closedPath = (segmentPositions.length > 1 &&\n segmentPositions[0].equalsWithEpsilon(segmentPositions[segmentPositions.length - 1])) ||\n node.closePath;\n if (closedPath) {\n const faceNormal = calculateFaceNormal(segmentPositions);\n // If the rotation handedness is opposite the primary sketch vector, then flip the sketch path\n if (Vector3.Dot(faceNormal, primaryNormal) < 0) {\n segment.reverse();\n }\n }\n });\n // Parse the holes and match them to their parent polygons\n const holesByPathsWithIndices = [];\n pathSegmentIndices.forEach((segment, idx) => {\n if (!holePathsReferencingKeys[idx]) {\n return;\n }\n const parentPathIndex = pathGroupKeys.get(holePathsReferencingKeys[idx]);\n if (parentPathIndex === undefined) {\n return;\n }\n // This segment is a hole\n if (!holesByPathsWithIndices[parentPathIndex]) {\n holesByPathsWithIndices[parentPathIndex] = [];\n }\n holesByPathsWithIndices[parentPathIndex].push(segment);\n });\n if (node.addFace) {\n // We have to reorder the segment indices because the vertex position order gets modified by the face fill\n // Due to holes needing be be defined alongside the parent polygons\n const reorderedPathSegmentIndices = [];\n let positionPointer = 0;\n pathSegmentIndices.forEach((segment, idx) => {\n // Skip holes in the loop since they are processed with their parent polygons\n if (holePathsReferencingKeys[idx]) {\n return;\n }\n const segmentPositions = [];\n const segmentIndices = [];\n for (let i = 0; i < segment.length; i++) {\n const point = positionVectors[segment[i]];\n segmentPositions.push(point);\n segmentIndices.push(positionPointer);\n positionPointer++;\n }\n reorderedPathSegmentIndices.push(segmentIndices);\n const holesPositions = [];\n if (holesByPathsWithIndices[idx]) {\n for (let hi = 0; hi < holesByPathsWithIndices[idx].length; hi++) {\n const hole = holesByPathsWithIndices[idx][hi];\n const holeData = [];\n const holeIndices = [];\n for (let i = 0; i < hole.length; i++) {\n const point = positionVectors[hole[i]];\n holeData.push(point);\n holeIndices.push(positionPointer);\n positionPointer++;\n }\n holesPositions.push(holeData);\n reorderedPathSegmentIndices.push(holeIndices);\n }\n }\n const newMeshVertices = fillSketchPolygon(segmentPositions, node.flipPath, primaryNormal, holesPositions);\n // Fix the indices to point at the right vertices\n const startIndex = positions.length / 3;\n for (let i = 0; i < newMeshVertices.indices.length; i++) {\n newMeshVertices.indices[i] += startIndex;\n }\n indices.push(...newMeshVertices.indices);\n positions.push(...newMeshVertices.positions);\n normals.push(...newMeshVertices.normals);\n });\n //const normals = new Float32Array(positionVectors.length * 3);\n const uvs = new Float32Array(positionVectors.length * 2);\n uvs.fill(0);\n result.positions = new Float32Array(positions);\n result.indices = new Uint16Array(indices);\n result.normals = new Float32Array(normals);\n result.uvs = uvs;\n pathSegmentIndices = reorderedPathSegmentIndices;\n }\n else {\n const positions = new Float32Array(positionVectors.length * 3);\n for (let i = 0; i < positionVectors.length; i++) {\n const point = positionVectors[i];\n positions.set([point.x, point.y, point.z], i * 3);\n }\n result.positions = positions;\n }\n const indexMapToGrouped = [];\n let i = 0;\n pathSegmentIndices.forEach((segment, indexOfPath) => {\n // Skip holes since they are processed later\n if (holePathsReferencingKeys[indexOfPath]) {\n return;\n }\n indexMapToGrouped[indexOfPath] = i;\n for (let i = 1; i < segment.length; i++) {\n paths.push(segment[i - 1], segment[i]);\n }\n if (segment.length > 2 && node.closePath) {\n paths.push(segment[segment.length - 1], segment[0]);\n }\n i++;\n });\n const holes = [];\n pathSegmentIndices.forEach((segment, indexOfPath) => {\n // Add holes after the regular polys\n if (!holePathsReferencingKeys[indexOfPath]) {\n return;\n }\n const parentPathIndex = pathGroupKeys.get(holePathsReferencingKeys[indexOfPath]);\n if (parentPathIndex === undefined) {\n return;\n }\n holes.push(indexMapToGrouped[parentPathIndex], paths.length);\n for (let i = 1; i < segment.length; i++) {\n paths.push(segment[i - 1], segment[i]);\n }\n if (segment.length > 2 && node.closePath) {\n paths.push(segment[segment.length - 1], segment[0]);\n }\n });\n result.paths = new Uint16Array(paths);\n result.holes = new Uint16Array(holes);\n node._sketchPoints = new Float32Array(result.positions);\n let counter = -1;\n const startingPoints = [];\n //for loop building the starting points array which stores sketchControlPoints at the start of a line segment\n for (let i = 0; i < node.paths.length; i++) {\n const path = node.paths[i];\n let lastPoint;\n if (path._points && path._points.length > 0) {\n let thisPoint;\n for (let iP = 0; iP < path._points.length; iP++) {\n const { x, y, z } = path._points[iP];\n thisPoint = new Vector3(x, y, z);\n startingPoints.push(counter);\n if (!lastPoint || !thisPoint.equals(lastPoint)) {\n if (iP === 0 || !currentPath) {\n startingPoints.pop();\n }\n }\n lastPoint = thisPoint;\n counter++;\n }\n }\n }\n node._lineSegmentStartingPoints = startingPoints;\n node.calculatedBoundingBox = this.calculateSketchBoundingBox(node);\n return result;\n }\n updateSketchPreview(sketchNode, vertexData, mesh) {\n var _a;\n const previewMesh = this.sketchPreviews.get(sketchNode._dynamicId);\n const { segments } = parseSketchGeometry(vertexData);\n if (segments.length > 0) {\n const baseColor = ((_a = sketchNode._rootScene) === null || _a === void 0 ? void 0 : _a.edgeColor)\n ? sketchNode._rootScene.edgeColor.toColor4()\n : new Color4(1, 1, 1, 1);\n const colors = [];\n segments.forEach(segmentPoints => {\n const segmentColors = [];\n colors.push(segmentColors);\n segmentPoints.forEach(() => {\n segmentColors.push(baseColor);\n });\n });\n const lineColor = colors.length > 0 ? colors : undefined;\n if (previewMesh &&\n !sketchNode.changes.has('addFace') &&\n !sketchNode.changes.has('paths') &&\n !sketchNode.changes.has('closePath')) {\n MeshBuilder.CreateLineSystem(`${sketchNode._dynamicId}_preview`, {\n lines: segments,\n instance: previewMesh,\n colors: lineColor,\n updatable: true,\n }, this.scene);\n }\n else {\n if (previewMesh) {\n previewMesh.dispose();\n }\n const newPreviewMesh = MeshBuilder.CreateLineSystem(`${sketchNode._dynamicId}_preview`, {\n lines: segments,\n colors: lineColor,\n updatable: true,\n }, this.scene);\n this.sketchPreviews.set(sketchNode._dynamicId, newPreviewMesh);\n newPreviewMesh.parent = mesh;\n newPreviewMesh.enableEdgesRendering();\n newPreviewMesh.edgesColor = baseColor;\n this.viewer.camera.maintainNodeScreenSize(newPreviewMesh, (dist, mode) => {\n if (mode === Camera.PERSPECTIVE_CAMERA) {\n newPreviewMesh.edgesWidth = dist * 0.25;\n }\n else {\n newPreviewMesh.edgesWidth = 4;\n }\n });\n }\n }\n else if (previewMesh) {\n previewMesh.dispose();\n this.sketchPreviews.delete(sketchNode._dynamicId);\n }\n }\n updateSketchTracing(sketchNode, vertexData, mesh) {\n const previewMesh = this.sketchTracings.get(sketchNode._dynamicId);\n const sketchColor = sketchNode.pathColor.toColor4();\n if (sketchNode.tracePath) {\n if (previewMesh) {\n previewMesh.dispose();\n }\n //create/update drawings\n const linePoints = [];\n sketchNode.paths.forEach(path => {\n path._points.forEach(point => {\n linePoints.push(new Vector3(point.x, point.y, point.z));\n });\n });\n if (sketchNode.closePath) {\n const point = sketchNode.paths[0]._points[0];\n linePoints.push(new Vector3(point.x, point.y, point.z));\n }\n const lines = Mesh.CreateLines('lines', linePoints, this.scene, true);\n lines.enableEdgesRendering();\n lines.edgesWidth = sketchNode.lineThickness;\n lines.edgesColor = sketchColor;\n this.sketchTracings.set(sketchNode._dynamicId, lines);\n }\n else {\n if (previewMesh || (previewMesh && sketchNode.tracePath === false)) {\n previewMesh.dispose();\n this.sketchTracings.delete(sketchNode._dynamicId);\n }\n }\n }\n buildTextGeometry(node) {\n const fontService = node._manager.getService(Token.FontService);\n if (!node.fontFile || !node.text) {\n return null;\n }\n if (!fontService.readyFonts.has(node.fontFile)) {\n fontService.loadFontByFile(node.fontFile).then(resultData => {\n const fontBuffer = resultData;\n if (fontBuffer) {\n node.setPropertyIsModified('fontFile', true);\n }\n });\n return null;\n }\n else {\n let vertexDataBuilder = textMeshFonts.get(node.fontFile);\n if (!vertexDataBuilder) {\n const fontBuffer = fontService.readyFonts.get(node.fontFile);\n const openTypeFont = parseBuffer(fontBuffer);\n const font = new TextMeshFont(node.fontFile, openTypeFont, this.scene);\n vertexDataBuilder = new TextMeshVertexData(font);\n textMeshFonts.set(node.fontFile, vertexDataBuilder);\n }\n return vertexDataBuilder.getText(node.text, node.fontSize, node.charSpacing, node.curveRadius);\n }\n }\n getMeshGeometry(node) {\n const mesh = this.get3d(node);\n if (mesh && mesh instanceof Mesh && mesh._geometry) {\n const geometry = SubmeshVertexData.ExtractFromMesh(mesh);\n // // Make sure the geometry is valid and actually has some geometry in it so consumers of this function don't have to deep check.\n // if(!geometry.indices || !geometry.positions || geometry.indices.length === 0) {\n // return null;\n // }\n return geometry;\n }\n return null;\n }\n fetchMaterialNode(meshNode, materialId) {\n var _a;\n const localMaterial = (_a = meshNode._manager) === null || _a === void 0 ? void 0 : _a.getById(materialId); //find the material if it's local\n return new Promise((resolve, reject) => {\n if (localMaterial) {\n //it's a local material\n resolve(localMaterial);\n }\n else {\n // Not a local material, so it's a global or library material\n this.rendererManager.material\n .loadGlobalMaterial(materialId, meshNode._manager.services)\n .pipe(take(1))\n .subscribe(globalLibMaterial => {\n if (!globalLibMaterial) {\n console.warn(`Cannot find global material ${materialId} on node ${meshNode.name} (${meshNode.id})`);\n reject('could not load global material');\n }\n else {\n resolve(globalLibMaterial);\n }\n });\n }\n });\n }\n setRenderModeOnMesh(mesh, sceneNode) {\n const mode = this.renderMode;\n if (mesh.geometry) {\n if (mode === exports.eRenderMode.edge || mode === exports.eRenderMode.hiddenEdges) {\n mesh.enableEdgesRendering(sceneNode.edgeDisplayTolerance);\n mesh.renderOutline = true;\n mesh.edgesWidth = sceneNode.edgeWidth;\n mesh.edgesColor = sceneNode.edgeColor.toColor4();\n mesh.outlineWidth = sceneNode.edgeWidth / 100;\n mesh.outlineColor = sceneNode.edgeColor.toColor3();\n // Another option for the hidden edges renderer but results in render artifacts especially when meshes go on top of each other.\n // if (mode === eRenderMode.hiddenEdges) {\n // mesh.metadata = { ...mesh.metadata, visibility: mesh.visibility };\n // mesh.visibility = 0.001;\n // }\n }\n else {\n // mesh.visibility = mesh.metadata?.visibility || 1;\n mesh.renderOutline = false;\n mesh.disableEdgesRendering();\n }\n }\n }\n getMeshVertexData(node) {\n if (!node._geometry) {\n throw Error('Cannot get vertex data because mesh has no geometry');\n }\n let meshVertexData = SubmeshVertexData.FromGeometry(node._geometry);\n meshVertexData = this.preTransformVertices(meshVertexData, node, node._geometry);\n return meshVertexData;\n }\n preTransformVertices(meshVertexData, node, cachedGeometry) {\n if (node.preTransform && node.preTransform.length !== 16) {\n throw Error('invalid pretransform array ');\n }\n // If pretransform is not set on the node, the transform we want is the identity matrix (no change from original)\n const transform = node.preTransform ? Matrix.FromArray(node.preTransform) : Matrix.Identity();\n // Check if the geometry has a transform applied\n const existingTransform = cachedGeometry === null || cachedGeometry === void 0 ? void 0 : cachedGeometry._hasTransform;\n // If the geometry has already been transformed, then we have to undo that transform to get the correct transform\n // Starting with the invert matrix of the existing transform will cancel out the existing transform and result in the original mesh\n if (existingTransform) {\n Matrix.FromArray(existingTransform).invert().multiplyToRef(transform, transform);\n }\n // If the transform we want (taking the existing transform into account) is the identity matrix, do nothing and return the geometry\n if (!transform.isIdentity()) {\n if (existingTransform) {\n // If the geometry has already been transformed, don't overwrite the transform so we don't repeatedly transform it\n meshVertexData = meshVertexData.deepClone();\n }\n else if (cachedGeometry) {\n // If the geoemtry has no existing transform, set the hasTransform property since it will get mutated with the new transform.\n cachedGeometry._hasTransform = node.preTransform;\n }\n meshVertexData.transform(transform);\n }\n return meshVertexData;\n }\n updateParentMeshFromNode(node, targetNode, isNewMesh) {\n const parentNode = node._parent;\n if (parentNode && parentNode instanceof exports.SpaceNode) {\n const isRootScene = parentNode instanceof exports.SceneNode && !parentNode.isNested();\n const parent3d = isRootScene\n ? this.rendererManager.sceneRenderer.get3d(parentNode)\n : this.get3d(parentNode);\n if (parent3d) {\n if (targetNode instanceof Mesh) {\n targetNode.setParent(parent3d);\n }\n else {\n targetNode.parent = parent3d;\n }\n }\n else {\n console.error(`Could not find mesh by ID ${parentNode.id} in scene!`);\n }\n }\n else {\n return; // if the node has no parent, it's not part of the scene anymore\n }\n if (isNewMesh ||\n SettingsManagerService.get('parentChangeBehavior') === exports.eParentChangeBehavior.PRESERVE_PROPERTY) {\n this.updateMeshTransform(node, targetNode, null);\n }\n else {\n node.position = exports.KbVector.FromVec3(targetNode.position);\n if (targetNode.rotationQuaternion) {\n node.useEuler = false;\n node.setRotationQuaternion(new exports.KbQuaternion(targetNode.rotationQuaternion.x, targetNode.rotationQuaternion.y, targetNode.rotationQuaternion.z, targetNode.rotationQuaternion.w));\n }\n else if (node.eulerRotation) {\n node.useEuler = true;\n node.eulerRotation = new exports.KbVector(targetNode.rotation.x, targetNode.rotation.y, targetNode.rotation.z);\n }\n node.scaling = targetNode.scaling.x;\n }\n this.invalidateBoundingBox(parentNode); //invalidate parent node in the bounding box cache as now it has a new child\n }\n recursiveLoadChildMeshGeometry(node) {\n const children = node.getChildren();\n children.forEach(child => {\n if (child instanceof exports.SpaceNode && child.visible) {\n if (child instanceof exports.CustomMeshNode && isTrackedClass(child)) {\n if (child.hasGeometry && !child._geometry && !child.geometryLoaded && !child._loading) {\n this.loadNewGeometry(child);\n }\n }\n else {\n this.recursiveLoadChildMeshGeometry(child);\n }\n }\n });\n }\n createOrUpdateMesh(node, otherMesh) {\n let mesh;\n if (otherMesh) {\n mesh = otherMesh;\n }\n else {\n mesh = this.get3d(node);\n }\n const changes = node.changes;\n let allChanges = false;\n let meshVerticesNeedReset = false;\n if (!mesh) {\n // New Mesh\n allChanges = true;\n if (isMeshNode(node))\n changes.set('features', node.features);\n if (isMeshGroupNode(node) || isSceneNode(node)) {\n mesh = new TransformNode(node._dynamicId, this.scene);\n }\n else {\n const newMesh = (mesh = new Mesh(node._dynamicId, this.scene));\n if (isCustomMeshNode(node) && isTrackedClass(node) && node._geometry) {\n this.setBabylonVertexData(node, newMesh);\n }\n else {\n meshVerticesNeedReset = true;\n }\n }\n mesh.scaling = Vector3.One();\n mesh.name = node.name;\n mesh.id = node._dynamicId;\n this.set3d(node, mesh);\n }\n else if (isCustomMeshNode(node) && changes.has('geometryId') && node._loading !== node.geometryId) {\n // Only load new geometry if the geometry hasn't just changed\n this.loadNewGeometry(node);\n }\n else if (isCustomMeshNode(node) && isTrackedClass(node) && node._geometry && node.changes.has('_geometry')) {\n // If we're changing the geometry, we always need to (re)apply all mesh properties\n // to the mesh as the new geometry won't have the right properties\n allChanges = true;\n changes.set('features', node.features);\n }\n if (allChanges ||\n changes.hasAny('position', 'useEuler', 'eulerRotation', 'rotationAxis', 'rotationAngle', 'pivot', 'scaling', '_showPivot')) {\n this.updateMeshTransform(node, mesh, allChanges ? null : changes);\n if (isMeshNode(node)) {\n this.rendererManager.material.reflectedMeshes.set(node._dynamicId, mesh);\n }\n this.rendererManager.material.updateReflectionProbes();\n }\n if (changes.has('scaling')) {\n this.updateConnectorScaling(node);\n }\n const pivotGizmo = this.pivotGizmos.get(node._dynamicId);\n if (allChanges || changes.has('visible')) {\n mesh.setEnabled(!!node.visible);\n this.viewer.polygonTracker.next();\n this.viewer.warningTracker.next();\n if (pivotGizmo) {\n pivotGizmo.setEnabled(!!node.visible);\n }\n this.invalidateBoundingBox(node);\n // If a group node is made visible and it has a child node that has geometry load it now\n if (node.visible && node instanceof exports.MeshGroupNode) {\n let parentNode = node._parent;\n let allParentsVisible = true;\n while (parentNode) {\n if (!parentNode.visible) {\n allParentsVisible = false;\n break;\n }\n parentNode = parentNode._parent;\n }\n if (allParentsVisible) {\n this.recursiveLoadChildMeshGeometry(node);\n }\n }\n }\n if (mesh instanceof Mesh) {\n const meshNode = node;\n mesh.isBlocker = false;\n if (allChanges || changes.has('receiveShadows')) {\n mesh.receiveShadows = !!meshNode.receiveShadows;\n this.shadowManager.toggleReceiver(mesh, !!meshNode.receiveShadows);\n }\n if (changes.has('isPickable')) {\n if (meshNode.isPickable !== undefined) {\n mesh.isPickable = meshNode.isPickable;\n }\n else {\n mesh.isPickable = true;\n }\n }\n if (allChanges || changes.has('castShadows'))\n this.shadowManager.toggleCaster(mesh, !!meshNode.castShadows);\n if (allChanges || changes.has('checkCollisions'))\n mesh.checkCollisions = !!meshNode.checkCollisions;\n if (allChanges || changes.has('flipSurfaceOrientation')) {\n if (meshNode.flipSurfaceOrientation) {\n mesh.overrideMaterialSideOrientation = Mesh.BACKSIDE;\n }\n else {\n mesh.overrideMaterialSideOrientation = Mesh.FRONTSIDE;\n }\n }\n if (this.renderMode !== exports.eRenderMode.hiddenEdges) {\n if (allChanges || changes.has('opacity')) {\n mesh.visibility = meshNode.opacity == null ? 1 : meshNode.opacity;\n }\n }\n const warningTracker = {\n name: meshNode.name,\n isEnabled: meshNode.visible,\n castShadows: meshNode.castShadows || false,\n receiveShadows: meshNode.receiveShadows || false,\n polygonNumber: meshNode._polygonCount || 0,\n featureNumber: node.features.length,\n };\n this.viewer.MeshWarningTracker.set(mesh.id, warningTracker);\n // Material\n if (allChanges || changes.has('materialId') || changes.has('visible')) {\n const _mesh = mesh;\n // If the material is a multimaterial, make sure all the material nodes making it up are initialized\n if (meshNode.visible && mesh.material instanceof MultiMaterial) {\n mesh.material.subMaterials.forEach(material => {\n if (material) {\n const localMaterial = getByDynamicId(material.id, false);\n if (localMaterial) {\n localMaterial.init();\n }\n else {\n // Global material ?\n const globalMaterial = this.rendererManager.material.getGlobalMaterial(material.id);\n if (globalMaterial) {\n globalMaterial.init();\n }\n }\n }\n });\n }\n if (meshNode.materialId) {\n this.fetchMaterialNode(meshNode, meshNode.materialId).then(materialNode => {\n if (meshNode.visible) {\n this.setMaterialOnMesh(meshNode, _mesh, materialNode);\n }\n }, () => {\n meshNode.materialId = undefined;\n });\n }\n else {\n if (mesh.material instanceof MultiMaterial) {\n mesh.material.subMaterials[0] = this.rendererManager.material.defaultMaterial;\n mesh.material.forceCompilation(mesh); // If you don't do this webgl vomits errors because we invalidated the cache\n }\n else {\n mesh.material = this.rendererManager.material.defaultMaterial;\n }\n }\n if (isMeshNode(node)) {\n this.rendererManager.material.reflectedMeshes.set(node._dynamicId, mesh);\n }\n this.rendererManager.material.updateReflectionProbes();\n }\n let sketchNode;\n let updateSketchDrawing = false;\n // For primitives, recalculate the mesh if the shape properties changed\n if (isPrimitiveMeshNode(meshNode)) {\n if (allChanges) {\n meshVerticesNeedReset = true;\n }\n else {\n meshNode.changes.forEach((_, key) => {\n if (primitiveShapeProperties.has(key)) {\n meshVerticesNeedReset = true;\n }\n });\n }\n }\n else if (isCustomMeshNode(meshNode) && isTrackedClass(meshNode)) {\n meshVerticesNeedReset = meshNode.changes.has('_geometry') && !!meshNode._geometry;\n }\n else if (isSketchNode(meshNode) && isTrackedClass(meshNode)) {\n sketchNode = meshNode;\n meshVerticesNeedReset = sketchNode.changes.hasAny('paths', 'addFace', 'flipPath', 'clonePath', 'closePath', '_previewVisible');\n updateSketchDrawing = sketchNode.changes.hasAny('paths', 'addFace', 'flipPath', 'clonePath', 'closePath', '_previewVisible', 'lineThickness', 'pathColor', 'tracePath');\n if (sketchNode.changes.hasAny('boundingXOverride', 'boundingYOverride', 'boundingZOverride')) {\n sketchNode._boundingBoxChanges.next(sketchNode.getBoundingBox());\n }\n }\n else if (isTextNode(meshNode) && isTrackedClass(meshNode)) {\n meshVerticesNeedReset = meshNode.changes.hasAny('text', 'fontSize', 'curveRadius', 'charSpacing', 'fontFile');\n }\n let preTransformChanged = false;\n if (allChanges || changes.hasAny('features', 'preTransform')) {\n meshVerticesNeedReset = true;\n preTransformChanged = !(allChanges || changes.has('_geometry')); // If geometry has changed, then the geometry was just loaded\n }\n // Features or shape properties have been changed, we have to recalculate original geometry and reprocess features on top\n if (allChanges || changes.has('uvOverrides') || meshVerticesNeedReset || updateSketchDrawing) {\n const meshVertexData = this.rebuildBaseGeometry(node, preTransformChanged);\n if (meshVertexData) {\n // This call should also apply the new vertex data.\n this.rendererManager.features.resetMeshGeometry(meshNode, null, meshVertexData);\n }\n else {\n if (mesh.geometry) {\n // If mesh geometry is set, the mesh was previously initialized but now is being generated with no geometry so we should destroy the old geometry\n // Failing to do so will cause Babylon geometry issues\n mesh.dispose();\n this.delete3d(node);\n }\n }\n if (sketchNode && sketchNode._previewVisible) {\n this.updateSketchPreview(sketchNode, meshVertexData, mesh);\n }\n if (sketchNode && sketchNode.tracePath && updateSketchDrawing) {\n if (sketchNode) {\n this.updateSketchTracing(sketchNode, meshVertexData, mesh);\n }\n }\n }\n // If this is a sketch, see if we need to show a preview of it\n if (sketchNode && sketchNode.changes.has('_previewVisible')) {\n const previewMesh = this.sketchPreviews.get(meshNode._dynamicId);\n if (previewMesh) {\n previewMesh.setEnabled(sketchNode._previewVisible);\n }\n }\n this.viewer.meshWarningsNeedsUpdate.next();\n }\n return mesh;\n }\n updateMeshTransform(node, mesh, changes // null means all changes\n ) {\n const pivot = node.pivot.toVec3();\n const originOffset = pivot.scale(-1);\n const newPosition = node.position.toVec3();\n let newRotation;\n if (node.useEuler) {\n newRotation = Quaternion.FromEulerAngles(toRadians(node.eulerRotation.x), toRadians(node.eulerRotation.y), toRadians(node.eulerRotation.z));\n }\n else {\n newRotation = Quaternion.RotationAxis(node.rotationAxis.toVec3(), toRadians(node.rotationAngle));\n }\n const rotationMatrix = new Matrix();\n newRotation.toRotationMatrix(rotationMatrix);\n Vector3.TransformCoordinatesToRef(originOffset, rotationMatrix, originOffset);\n mesh.rotationQuaternion = newRotation;\n const scalingMatrix = Matrix.Scaling(node.scaling, node.scaling, node.scaling);\n Vector3.TransformCoordinatesToRef(originOffset, scalingMatrix, originOffset);\n mesh.scaling = new Vector3().setAll(node.scaling);\n newPosition.addInPlace(originOffset).addInPlace(pivot);\n if (!mesh.position.equals(newPosition)) {\n mesh.position = newPosition;\n }\n this.rendererManager.features.transformChanged(node);\n this.invalidateBoundingBox(!changes || changes.has('scaling') ? node : node._parent); //scaling, changes current node, but all others only change the parent\n if (node._showPivot) {\n let pivotGizmo = this.pivotGizmos.get(node._dynamicId);\n if (!pivotGizmo) {\n pivotGizmo = SphereBuilder.CreateSphere('pivotSphere', { diameter: 0.05 }, this.utilityLayer.utilityLayerScene);\n const coloredMaterial = new StandardMaterial('', this.utilityLayer.utilityLayerScene);\n coloredMaterial.emissiveColor = Color3.Red();\n pivotGizmo.material = coloredMaterial;\n this.pivotGizmos.set(node._dynamicId, pivotGizmo);\n }\n if (!pivotGizmo.isEnabled()) {\n pivotGizmo.setEnabled(true);\n }\n const worldPivot = pivot.add(node.position.toVec3());\n if (mesh.parent) {\n const transform = mesh.parent.computeWorldMatrix(true);\n Vector3.TransformCoordinatesToRef(worldPivot, transform, worldPivot);\n }\n pivotGizmo.position = worldPivot;\n }\n else {\n const pivotGizmo = this.pivotGizmos.get(node._dynamicId);\n if (pivotGizmo && pivotGizmo.isEnabled()) {\n pivotGizmo.setEnabled(false);\n }\n }\n return mesh;\n }\n createPrimitiveMesh(node) {\n if (!isPrimitiveMeshNode(node))\n throw Error('Primitive not set');\n if (isBoxMesh(node)) {\n if (node.tessellation <= 1) {\n return MeshBuilder.CreateBox(node._dynamicId, {\n height: node.height,\n width: node.width,\n depth: node.depth,\n sideOrientation: exports.eSideOrientationBabylonMap[node.sideOrientation],\n updatable: true,\n }, this.scene);\n }\n else {\n return MeshBuilder.CreateTiledBox(node._dynamicId, {\n height: node.height,\n width: node.width,\n depth: node.depth,\n sideOrientation: exports.eSideOrientationBabylonMap[node.sideOrientation],\n updatable: true,\n tileSize: 1 / node.tessellation,\n }, this.scene);\n }\n }\n else if (isSphereMesh(node)) {\n return SphereBuilder$1.CreateSphere(node._dynamicId, {\n segments: node.tessellation,\n diameter: node.radius * 2,\n arc: node.arc / 360,\n sliceStart: node.latitudeStart / 180,\n slice: node.latitudeSlice / 180,\n sideOrientation: exports.eSideOrientationBabylonMap[node.sideOrientation],\n updatable: true,\n }, this.scene);\n }\n else if (isCylinderMesh(node)) {\n return CylinderBuilder$1.CreateCylinder(node._dynamicId, {\n height: node.height,\n diameter: node.radius * 2,\n tessellation: node.tessellation,\n subdivisions: node.subdivisions,\n hasRings: node.subdivisions > 1,\n cap: capModeMap[node.capMode],\n sideOrientation: exports.eSideOrientationBabylonMap[node.sideOrientation],\n arc: node.arc / 360,\n updatable: true,\n }, this.scene);\n }\n else if (isConeMesh(node)) {\n return MeshBuilder.CreateCylinder(node._dynamicId, {\n height: node.height,\n diameterTop: node.radiusTop * 2,\n diameterBottom: node.radiusBottom * 2,\n tessellation: node.tessellation,\n sideOrientation: exports.eSideOrientationBabylonMap[node.sideOrientation],\n updatable: true,\n }, this.scene);\n }\n else if (isTorusMesh(node)) {\n return TorusBuilder.CreateTorus(node._dynamicId, {\n diameter: node.radius * 2,\n thickness: node.thickness,\n tessellation: node.tessellation,\n sideOrientation: exports.eSideOrientationBabylonMap[node.sideOrientation],\n arc: node.arc / 360,\n sliceStart: node.latitudeStart / 360,\n slice: node.latitudeSlice / 360,\n updatable: true,\n }, this.scene);\n }\n else if (isPlaneMesh(node)) {\n const mesh = PlaneBuilder$1.CreatePlane(node._dynamicId, {\n width: node.width,\n height: node.depth,\n tileHeight: node.depth / node.segments,\n tileWidth: node.width / node.segments,\n sideOrientation: exports.eSideOrientationBabylonMap[node.sideOrientation],\n updatable: true,\n }, this.scene);\n mesh.rotation.x = Math.PI / 2;\n mesh.bakeCurrentTransformIntoVertices();\n return mesh;\n }\n else if (isDiscMesh(node)) {\n return DiscBuilder.CreateDisc(node._dynamicId, {\n radius: node.radius,\n tessellation: node.tessellation,\n sideOrientation: exports.eSideOrientationBabylonMap[node.sideOrientation],\n updatable: true,\n }, this.scene);\n }\n else {\n throw Error('Mesh primitive not implemented');\n }\n }\n setBabylonVertexData(kbMesh, mesh, preserveSubmeshes = true) {\n if (!kbMesh._geometry) {\n return;\n }\n const vertexData = this.getMeshVertexData(kbMesh);\n const invalid = verifyGeometry(vertexData);\n if (invalid) {\n console.error(`Invalid or corrupted geometry in mesh ${kbMesh.name}. The ${invalid} are invalid.`);\n }\n else {\n vertexData.applyToMesh(mesh, true, preserveSubmeshes);\n kbMesh._polygonCount = mesh.getTotalIndices() / 3;\n kbMesh._verticesCount = mesh.getTotalVertices();\n this.viewer.polygonTracker.next();\n this.viewer.warningTracker.next();\n }\n this.setRenderModeOnMesh(mesh, kbMesh._rootScene);\n if (mesh.isFacetDataEnabled) {\n mesh.updateFacetData();\n }\n }\n setBabylonUvData(uvData, vertexData) {\n if (uvData.uvs) {\n vertexData.uvs = uvData.uvs;\n }\n if (uvData.uv2s) {\n vertexData.uvs2 = uvData.uv2s;\n }\n if (uvData.uv3s) {\n vertexData.uvs3 = uvData.uv3s;\n }\n if (uvData.uv4s) {\n vertexData.uvs4 = uvData.uv4s;\n }\n if (uvData.uv5s) {\n vertexData.uvs5 = uvData.uv5s;\n }\n if (uvData.uv6s) {\n vertexData.uvs6 = uvData.uv6s;\n }\n return vertexData;\n }\n setMaterialOnMesh(meshNode, babylonMesh, materialNode) {\n if (meshNode.visible) {\n materialNode.init();\n }\n const material = this.rendererManager.material.get3d(materialNode);\n if (!material) {\n throw Error('Material could not be found in renderer. Has the material been created?');\n }\n // If the mesh already has a material and it's a multimaterial, then update the material on index 0\n // Which is reserved for the \"base\" material.\n if (babylonMesh.material instanceof MultiMaterial) {\n babylonMesh.material.subMaterials[0] = material;\n babylonMesh.material.forceCompilation(babylonMesh);\n }\n else if (!babylonMesh.material || babylonMesh.material.id != materialNode._dynamicId) {\n babylonMesh.material = material;\n }\n }\n invalidateBoundingBox(node) {\n if (node) {\n this.viewer.bbCache.invalidate(node);\n }\n }\n delete3d(node) {\n if (isSketchNode(node)) {\n node._boundingBoxChanges.complete();\n }\n const bnode = this.get3d(node);\n if (bnode && bnode instanceof Mesh) {\n this.viewer.MeshWarningTracker.delete(bnode.id);\n this.shadowManager.removeCaster(bnode);\n this.shadowManager.removeReceiver(bnode);\n if (bnode.material instanceof MultiMaterial) {\n bnode.material.dispose(true, true, false); // multimaterials are 1:1 with meshes\n }\n }\n this.rendererManager.material.reflectedMeshes.delete(node._dynamicId);\n this.rendererManager.material.updateReflectionProbes();\n this.viewer.meshWarningsNeedsUpdate.next();\n super.delete3d(node);\n this.viewer.polygonTracker.next();\n this.viewer.warningTracker.next();\n }\n createSketchPlaneVisualization(sketch) {\n const color = Color3.Yellow();\n const boundingBox = sketch.calculatedBoundingBox;\n let squareSize = 3;\n let calculatedSize = 0;\n let boundingBoxCenter = new Vector3(0, 0, 0);\n if (boundingBox) {\n boundingBoxCenter = boundingBox.center.clone();\n calculatedSize = boundingBox.minimum.subtract(boundingBox.maximum).length();\n }\n if (calculatedSize > squareSize) {\n squareSize = calculatedSize * 1;\n }\n const coloredMaterial = new StandardMaterial('', this.scene);\n coloredMaterial.wireframe = true;\n coloredMaterial.emissiveColor = color;\n const sketchPlaneVisualization = CreatePlane('mirrorPlane', { size: squareSize }, this.scene);\n const sketchMesh = this.viewer.getRendererNode(sketch);\n if (sketchMesh) {\n const translation = sketchMesh.computeWorldMatrix();\n const position = Vector3.TransformCoordinates(boundingBoxCenter, translation);\n sketchPlaneVisualization.position = new Vector3(position.x, position.y, position.z);\n const sketchNormal = new Vector3(0, 0, 0);\n if (sketch.normal === exports.eAxis.x) {\n sketchNormal.set(1, 0, 0);\n }\n else if (sketch.normal === exports.eAxis.y) {\n sketchNormal.set(0, 1, 0);\n }\n else {\n sketchNormal.set(0, 0, 1);\n }\n const rotationVector = Vector3.TransformCoordinates(sketchNormal, sketchMesh.computeWorldMatrix().getRotationMatrix());\n rotationVector.z = rotationVector.z * -1;\n const resultQuaternion = new Quaternion();\n Quaternion.FromUnitVectorsToRef(rotationVector, new Vector3(0, 0, 1), resultQuaternion);\n sketchPlaneVisualization.rotationQuaternion = resultQuaternion;\n //uvMapVisualization.rotation = Vector3.RotationFromAxis()\n sketchPlaneVisualization.material = coloredMaterial;\n sketchPlaneVisualization.bakeCurrentTransformIntoVertices();\n return sketchPlaneVisualization;\n }\n else {\n console.warn('Sketch visualization plane cannot be generated, please check direction and position of sketchpoints.');\n return null;\n }\n }\n updateConnectorScaling(node) {\n if (node instanceof exports.SketchNode) {\n node.paths.forEach(path => {\n path.controlPoints.forEach(controlPoint => {\n this.viewer.rendererManager.controlPointRenderer.updateScale(controlPoint);\n });\n });\n }\n else {\n if (node.connectors) {\n node.connectors.forEach(connector => {\n this.viewer.rendererManager.connector.updateScale(connector);\n });\n }\n const childNodes = node.getChildren();\n if (childNodes) {\n childNodes.forEach(child => {\n if (child instanceof exports.SpaceNode) {\n this.updateConnectorScaling(child);\n }\n });\n }\n }\n }\n calculateSketchBoundingBox(node) {\n const firstPoint = node.paths[0]._points[0];\n const minVector = new Vector3(firstPoint.x, firstPoint.y, firstPoint.z);\n const maxVector = new Vector3(firstPoint.x, firstPoint.y, firstPoint.z);\n node.paths.forEach(path => {\n path._points.forEach(point => {\n if (point.x < minVector.x) {\n minVector.x = point.x;\n }\n if (point.y < minVector.y) {\n minVector.y = point.y;\n }\n if (point.z < minVector.z) {\n minVector.z = point.z;\n }\n if (point.x > maxVector.x) {\n maxVector.x = point.x;\n }\n if (point.y > maxVector.y) {\n maxVector.y = point.y;\n }\n if (point.z > maxVector.z) {\n maxVector.z = point.z;\n }\n });\n });\n return new BoundingBox(minVector, maxVector);\n }\n }\n\n class PathRenderer extends Renderer {\n constructor(viewer, scene) {\n super(viewer, scene);\n this.viewer = viewer;\n this.scene = scene;\n this.modelToken = Token.SketchPath;\n }\n update3dInternal(node) { }\n }\n\n // function limitFPE(val: number) {\n // return Math.round((val + Number.EPSILON) * 100000000) / 100000000;\n // }\n // function roundQuaterion(q: Quaternion) {\n // return new Quaternion(limitFPE(q.x), limitFPE(q.y), limitFPE(q.z), limitFPE(q.w));\n // }\n function mateControlPointToConnector(mate, c1Mesh, controlPointMesh) {\n let grandParentContainerWorldMatrixInv;\n if (controlPointMesh.parent) {\n grandParentContainerWorldMatrixInv = controlPointMesh.parent.computeWorldMatrix(true).clone().invert();\n }\n else {\n grandParentContainerWorldMatrixInv = Matrix.IdentityReadOnly;\n }\n const targetTransformMatrix = c1Mesh.computeWorldMatrix().clone();\n targetTransformMatrix.multiplyToRef(grandParentContainerWorldMatrixInv, targetTransformMatrix);\n const position = new Vector3();\n const direction = new Vector3();\n targetTransformMatrix.decompose(undefined, undefined, position);\n Vector3.TransformNormalFromFloatsToRef(-1.0, 0, 0, targetTransformMatrix, direction);\n if (!mate.flip) {\n direction.scaleInPlace(-1);\n }\n direction.normalize();\n return {\n position,\n direction,\n };\n }\n const flipMatrix = new Matrix();\n Quaternion.RotationAxis(Axis.Y, toRadians(180)).toRotationMatrix(flipMatrix);\n function mateConnectors(mate, followerMesh, c1Mesh, c2Mesh) {\n updateMatrixOfAncestorChain(followerMesh, c2Mesh);\n c1Mesh.computeWorldMatrix(true);\n let mateOffset = Vector3.Zero();\n let worldDelta = Vector3.Zero(); //the translation that c2 would need to make to get to it's new position\n if (mate.mateType == exports.eMateType.ball) {\n /**\n * Ball Mate\n * -move c2 until it's in the same position as c1\n * In the future this could be made more intelligent by rotating a node to satisfy this mate and another ball or slider mate\n */\n worldDelta = c1Mesh.getAbsolutePosition().subtract(c2Mesh.getAbsolutePosition());\n }\n else {\n /**\n * Aligning direction vectors\n * ------------------------------------------------\n * Easiest way to think about this:\n * Connectors have a direction vector, and an angle of rotation around the direction vector.\n * The connector meshes are rotated by the renderer such that the positive X axis is pointing\n * in the direction of the direction vector. And the angle vector will be pointing in the direction\n * of the y axis.\n *\n * We want the world transformation matrix of the second connector to match the world transform of the first connector (optionally flipped).\n * The math for calculating the world matrix of a connector can be written as:\n * * * \n * We are trying to solve for the local transform matrix of the connector parent so the full world transform matches the transform of the first connector,\n * so we can rewrite the equation as:\n * = * * \n *\n * Then we apply the result matrix to the connector parent.\n */\n const t = new Vector3();\n const r = new Quaternion();\n const s = new Vector3();\n c1Mesh.computeWorldMatrix(true).decompose(s, r, t);\n if (mate.offsetAngle && mate.offsetAngle !== 0) {\n const angleOffsetQ = Quaternion.RotationAxis(Axis.X, (mate.offsetAngle * Math.PI) / 180);\n r.multiplyInPlace(angleOffsetQ);\n }\n const targetWorldTransformMatrix = Matrix.Compose(followerMesh.scaling, r, t);\n const currentWorldTransformMatrix = c2Mesh.computeWorldMatrix(true).clone();\n if (!mate.flip) {\n flipMatrix.multiplyToRef(targetWorldTransformMatrix, targetWorldTransformMatrix);\n }\n const c2LocalTransform = Matrix.Compose(c2Mesh.scaling, c2Mesh.rotationQuaternion, c2Mesh.position);\n let grandParentContainerWorldMatrix;\n if (followerMesh.parent) {\n grandParentContainerWorldMatrix = followerMesh.parent.computeWorldMatrix(true).clone();\n }\n else {\n grandParentContainerWorldMatrix = Matrix.IdentityReadOnly;\n }\n const grandParentContainerWorldMatrixInv = grandParentContainerWorldMatrix.clone().invert();\n let calculatedTargetWorldTransform;\n if (mate.mateType.equalsAny(exports.eMateType.revolute, exports.eMateType.cylindrical)) {\n /**\n * For these types of mates, the objective is to match the direction vectors but allow the angle to float freely.\n * What this means in practice is that we have to split the quaternion into a swing and a twist component.\n * The swing component will rotate the direction vector to point the right way, along a rotation axis equal to the cross product of the X axis and the direction vector,\n * and the twist component will rotate the mesh around the direction vector to match the desired angles. We then throw away the \"new\" twist angle and use the original twist angle.\n * For this to work it's necessary to ensure that the same rotation axis is used for both the swing rotation of the target and the swing rotation of the current, if the directions match.\n * Otherwise the twist angles won't be referring to the same thing.\n */\n const targetQ = Quaternion.FromRotationMatrix(targetWorldTransformMatrix.getRotationMatrix());\n const currentQ = Quaternion.FromRotationMatrix(currentWorldTransformMatrix.getRotationMatrix());\n const targetDirection = Vector3.TransformNormal(Axis.X, targetWorldTransformMatrix).normalize();\n const currentDirection = c2Mesh.right;\n // At direction vectors close to or equal to the X axis, the twist angle becomes heavily affected by floating point errors, so we avoid loops by not updating the rotation if the direction vectors are close to each other.\n if (currentDirection.add(targetDirection).length() > 0.0001) {\n const targetST = decomposeSwingTwist(targetQ, targetDirection.scale(-1));\n const currentST = decomposeSwingTwist(currentQ, currentDirection);\n const targetRotationNew = targetST.swing.multiply(currentST.twist);\n calculatedTargetWorldTransform = Matrix.Compose(followerMesh.scaling, targetRotationNew, t);\n }\n }\n else if (mate.mateType.equalsAny(exports.eMateType.fastened, exports.eMateType.planar, exports.eMateType.slider)) {\n calculatedTargetWorldTransform = targetWorldTransformMatrix;\n }\n if (calculatedTargetWorldTransform) {\n const followerMeshTransformMatrix = c2LocalTransform.clone().invert();\n followerMeshTransformMatrix.multiplyToRef(calculatedTargetWorldTransform, followerMeshTransformMatrix);\n followerMeshTransformMatrix.multiplyToRef(grandParentContainerWorldMatrixInv, followerMeshTransformMatrix);\n const resultFollowerRotMatrix = followerMeshTransformMatrix.getRotationMatrix();\n {\n followerMesh.rotationQuaternion = Quaternion.FromRotationMatrix(resultFollowerRotMatrix);\n updateMatrixOfAncestorChain(followerMesh, c2Mesh); //update world matrix after having just rotated follower mesh\n }\n }\n if (mate.mateType == exports.eMateType.fastened || mate.mateType == exports.eMateType.revolute) {\n //translate so the connectors line up\n worldDelta = c1Mesh.getAbsolutePosition().subtract(c2Mesh.getAbsolutePosition());\n mateOffset = new Vector3(mate.offsetX, mate.offsetY, mate.offsetZ);\n }\n else if (mate.mateType == exports.eMateType.planar) {\n /**\n * Planar Mate\n * - define a plane at the position of c1, with a normal in the direction of c1\n * - make a ray from c2 in the direction of c2, and try to intersect the c1 plane\n * - if it doesn't intersect, flip it and try again\n * - the intersection of the ray and plane is the new position of c2 (not considering offset)\n */\n const c1PlaneNormal = Axis.X.rotateByQuaternionToRef(c1Mesh.absoluteRotationQuaternion, Vector3.Zero());\n const c1Plane = Plane.FromPositionAndNormal(c1Mesh.getAbsolutePosition(), c1PlaneNormal);\n const c2Direction = Axis.X.rotateByQuaternionToRef(c2Mesh.absoluteRotationQuaternion, Vector3.Zero());\n const ray = new Ray(c2Mesh.getAbsolutePosition(), c2Direction);\n let distance = ray.intersectsPlane(c1Plane);\n if (distance == null) {\n //if it doesn't intersect try flipping it\n ray.direction.negateInPlace();\n distance = ray.intersectsPlane(c1Plane);\n }\n worldDelta = ray.direction.scale(distance);\n mateOffset = new Vector3(mate.offsetX, 0, 0); //only use the x component for planar mates\n }\n else if (mate.mateType == exports.eMateType.slider || mate.mateType == exports.eMateType.cylindrical) {\n /**\n * Slider Mate\n * - define a plane at the position of c2, with a normal in the direction of c2\n * - make a ray from c1 in the direction of c1, and try to intersect the c2 plane\n * - if it doesn't intersect, flip it and try again\n * - the intersection of the ray and plane is the new position of c2 (not considering offset)\n */\n const c1Pos = c1Mesh.getAbsolutePosition();\n const c2Pos = c2Mesh.getAbsolutePosition();\n const c2PlaneNormal = c2Mesh.getDirection(Axis.X).normalize();\n const c2Plane = Plane.FromPositionAndNormal(c2Pos, c2PlaneNormal);\n const c1Direction = c1Mesh.getDirection(Axis.X).normalize();\n const ray = new Ray(c1Pos, c1Direction);\n let distance = ray.intersectsPlane(c2Plane);\n if (distance == null) {\n //if it doesn't intersect try flipping it\n ray.direction.negateInPlace();\n distance = ray.intersectsPlane(c2Plane);\n }\n worldDelta = ray.direction.scale(distance).add(c1Pos).subtract(c2Pos);\n mateOffset = new Vector3(0, mate.offsetY, mate.offsetZ); //don't use the x component for slider mates\n }\n else if (mate.mateType == exports.eMateType.parallel) ;\n }\n /**\n * now we process the offset and set the position of c2 parent node\n */\n if (!mate.flip)\n mateOffset.negateInPlace();\n if (mate._swap)\n mateOffset.y = -mateOffset.y;\n const worldOffset = mateOffset.rotateByQuaternionToRef(c1Mesh.absoluteRotationQuaternion, Vector3.Zero());\n const totalOffset = worldOffset.add(worldDelta);\n const newAbsPos = followerMesh.getAbsolutePosition().add(totalOffset);\n followerMesh.setAbsolutePosition(newAbsPos);\n followerMesh.computeWorldMatrix(true);\n }\n /**\n Decompose the rotation on to 2 parts.\n 1. Twist - rotation around the \"direction\" vector\n 2. Swing - rotation around axis that is perpendicular to \"direction\" vector\n The rotation can be composed back by\n rotation = swing * twist\n\n has singularity in case of swing_rotation close to 180 degrees rotation.\n if the input quaternion is of non-unit length, the outputs are non-unit as well\n otherwise, outputs are both unit\n */\n function decomposeSwingTwist(rotation, _direction) {\n // Ensure the direction vector is normalized\n const direction = _direction.clone().normalize();\n const twist = new Quaternion();\n const swing = new Quaternion();\n // Alternate code with indeterminate swing axis\n // const ra = new Vector3(rotation.x, rotation.y, rotation.z);\n // const p = direction.scale(Vector3.Dot(ra, direction));\n // twist.set(p.x, p.y, p.z, rotation.w);\n // twist.normalize();\n // rotation.multiplyToRef(twist.conjugate(), swing);\n rotationToVectorToRef(direction, Axis.X, swing);\n swing.conjugate().multiplyToRef(rotation, twist);\n return { twist, swing };\n }\n\n (function (ConstraintType) {\n ConstraintType[ConstraintType[\"Sphere\"] = 0] = \"Sphere\";\n ConstraintType[ConstraintType[\"Line\"] = 1] = \"Line\";\n ConstraintType[ConstraintType[\"Plane\"] = 2] = \"Plane\";\n ConstraintType[ConstraintType[\"Fixed\"] = 3] = \"Fixed\";\n ConstraintType[ConstraintType[\"None\"] = 4] = \"None\";\n })(exports.ConstraintType || (exports.ConstraintType = {}));\n function sortMates(validMates) {\n const fixed = new Set();\n const primes = new Set();\n const nodeMap = getNodeMap(validMates);\n //parent/child relationships are also a kind of edge in our graph, so add them\n for (const [node] of nodeMap) {\n if (node instanceof exports.SpaceNode) {\n if (node.fixed) {\n fixed.add(node);\n }\n else if (node._primeMover) {\n primes.add(node);\n }\n for (const ancestor of node.getAncestors()) {\n if (nodeMap.has(ancestor)) {\n nodeMap.add(node, {\n target: ancestor,\n constraint: { type: exports.ConstraintType.None, properties: undefined },\n });\n nodeMap.add(ancestor, {\n target: node,\n constraint: { type: exports.ConstraintType.Fixed, properties: undefined },\n });\n }\n }\n }\n }\n /**\n * * starting with primes, recurse through. Then move to non-primes, ignoring those already visited.\n */\n const visitedNodes = new Set();\n const visitedMates = new Set();\n const nodePath = new Set();\n const matePath = new Set();\n const sorted = new Set();\n const cycles = new Array();\n const overConstrained = new Array();\n for (const n of [...fixed, ...primes, ...nodeMap.keys()]) {\n recurseMateChain(nodeMap, visitedNodes, visitedMates, nodePath, matePath, sorted, cycles, overConstrained, n);\n }\n //remove overconstrained mates from sorted so they don't get processed\n for (const m of overConstrained) {\n sorted.delete(m);\n }\n return {\n sorted,\n cycles,\n overConstrained,\n };\n }\n /**\n * Gets the nodes that are connected to the given through a chain of mates\n */\n function getNodeChain(node, validMates) {\n const nodeMap = getNodeMap(validMates);\n const visitedNodes = new Set();\n const visitedMates = new Set();\n const nodePath = new Set();\n const matePath = new Set();\n const sorted = new Set();\n const cycles = new Array();\n const overConstrained = new Array();\n recurseMateChain(nodeMap, visitedNodes, visitedMates, nodePath, matePath, sorted, cycles, overConstrained, node);\n visitedNodes.delete(node);\n return visitedNodes;\n }\n function getNodeMap(validMates) {\n const nodeMap = new ArrayMap();\n for (const mate of validMates) {\n const c1 = mate.getC1();\n const c2 = mate.getC2();\n let c1Parent = c1 === null || c1 === void 0 ? void 0 : c1.getSpaceParent();\n let c2Parent = c2 === null || c2 === void 0 ? void 0 : c2.getSpaceParent();\n if (c1 && isSketchControlPoint(c1)) {\n c1Parent = c1;\n }\n if (c2 && isSketchControlPoint(c2)) {\n c2Parent = c2;\n }\n if (c1 && c2 && c1Parent && c2Parent) {\n const constraint = { type: exports.ConstraintType.Fixed, properties: undefined };\n // TODO [RO] - Later on we can add more constraint types here based on mate type (CPQ-3881)\n //if (isConstrainingConnector(c1) && isConstrainingConnector(c2)) {\n // constraint = { type: ConstraintType.Fixed, properties: undefined };\n // } else {\n // constraint = { type: ConstraintType.None, properties: undefined };\n // }\n nodeMap.add(c1Parent, { target: c2Parent, mate, constraint });\n nodeMap.add(c2Parent, { target: c1Parent, mate, constraint });\n }\n }\n return nodeMap;\n }\n function recurseMateChain(nodeMap, visitedNodes, visitedMates, nodePath, matePath, sorted, cycles, overConstrained, node, mate, sourceEdge, isOverconstrained = false) {\n if (!sourceEdge || sourceEdge.constraint.type === exports.ConstraintType.Fixed) {\n if (nodePath.has(node)) {\n const cycle = [...matePath];\n if (mate) {\n cycle.push(mate);\n }\n cycles.push(cycle);\n }\n if (visitedNodes.has(node)) {\n return;\n }\n }\n visitedNodes.add(node);\n nodePath.add(node);\n if (mate) {\n matePath.add(mate);\n visitedMates.add(mate);\n sorted.add(mate);\n const c1 = mate.getC1();\n const c2 = mate.getC2();\n /** Move along sketch connectors should always be the connector 1, because they can't be moved (they are read only) */\n if (c2.positionMode === exports.eConnectorPositionMode.moveAlongSketch) {\n mate._swap = true;\n }\n /** Control points are always follower meshes */\n else if (isConnector(c1) && isSketchControlPoint(c2)) {\n mate._swap = false;\n }\n else if (isConnector(c2) && isSketchControlPoint(c1)) {\n mate._swap = true;\n }\n else {\n /** mate needs to have c1/c2 swapped if we got here in the opposite direction */\n mate._swap = c1._parent === node;\n }\n }\n // If the source edge is fixed, we recurse through the node because all the node's connectors will be shifting.\n // If the source edge is not fixed, we stop here because only the mated connector will move\n // [RO] TODO eventually we will have to rethink this logic to support more constraint types (CPQ-3881)\n if (!sourceEdge || sourceEdge.constraint.type === exports.ConstraintType.Fixed) {\n for (const edge of nodeMap.get(node)) {\n if (!edge.mate || !visitedMates.has(edge.mate)) {\n if (edge.target instanceof exports.SpaceNode &&\n edge.target.fixed &&\n edge.constraint.type !== exports.ConstraintType.None) {\n isOverconstrained = true;\n }\n recurseMateChain(nodeMap, visitedNodes, visitedMates, nodePath, matePath, sorted, cycles, overConstrained, edge.target, edge.mate, edge, isOverconstrained);\n }\n }\n }\n nodePath.delete(node);\n if (mate) {\n matePath.delete(mate);\n if (isOverconstrained) {\n overConstrained.push(mate);\n }\n visitedMates.delete(mate);\n }\n }\n\n class Processor {\n constructor() {\n this.afterCleanupTasks = [];\n }\n beforeRender() { }\n beforeDeletions(deletions) { }\n afterDeletions(deletions) { }\n beforeUpdates(changes) { }\n afterUpdates(changes) { }\n afterCleanup(func) {\n this.afterCleanupTasks.push(func);\n }\n doAfterCleanup() {\n this.afterCleanupTasks.forEach(task => task());\n this.afterCleanupTasks.clear();\n }\n dispose() { }\n }\n\n const CYCLE_ERROR_START = 'Cycle detected: ';\n class MateProcessor extends Processor {\n constructor(viewer, scene, connectorRenderer, controlPointRenderer, referenceProcessor) {\n super();\n this.viewer = viewer;\n this.scene = scene;\n this.connectorRenderer = connectorRenderer;\n this.controlPointRenderer = controlPointRenderer;\n this.referenceProcessor = referenceProcessor;\n }\n beforeRender() {\n this.rebuildDependencies = false;\n }\n beforeDeletions(deletions) {\n const mateDeletions = deletions.get(Token.MateNode);\n const connectorDeletions = deletions.get(Token.Connector);\n this.rebuildDependencies =\n this.rebuildDependencies || mateDeletions.length > 0 || connectorDeletions.length > 0;\n }\n beforeUpdates(changes) {\n //determine if mate dependencies need to be rebuilt\n const mateChanges = changes.get(Token.MateNode);\n const connectorChanges = changes.get(Token.Connector);\n const meshChanges = changes.get(Token.SpaceNode);\n this.rebuildDependencies =\n this.rebuildDependencies ||\n mateChanges.some(m => m.changes.hasAny('connector1', 'connector2', 'enabled')) ||\n connectorChanges.some(c => c.changes.has('enabled')) ||\n meshChanges.some(c => c.changes.hasAny('fixed', '_primeMover'));\n /**\n * Mates need to be reprocessed if:\n * * the mate itself was changed\n * * connector1 of the node was changed\n * * parent node of connector1 was changed\n * * any ancestor space node of the parent node was changed\n * * any parent graph dependency node mate or it's parents were changed\n */\n }\n afterUpdates(changes) {\n if (this.rebuildDependencies) {\n this.buildDependencies();\n }\n if (this.sortedMates) {\n if (changes.has(Token.MateNode) ||\n changes.has(Token.SpaceNode) ||\n changes.has(Token.Connector) ||\n changes.has(Token.Feature)) {\n for (const mate of this.sortedMates) {\n //don't process the last mate in a cycle\n if (!this.cycles || !this.cycles.some(c => c.last() === mate)) {\n this.processMate(mate);\n }\n }\n }\n }\n }\n dispose() {\n if (this.sortedMates)\n this.sortedMates.clear();\n if (this.cycles)\n this.cycles.clear();\n }\n processMate(mate) {\n let c1 = mate.getC1();\n let c2 = mate.getC2();\n /**\n * If the sorter determined we need to swap the connectors.\n * */\n if (mate._swap || c2.positionMode === exports.eConnectorPositionMode.moveAlongSketch) {\n const temp = c1;\n c1 = c2;\n c2 = temp;\n }\n const c1Mesh = this.getConnectorMesh(c1);\n const c2Mesh = this.getConnectorMesh(c2);\n const followerNode = c2._parent;\n const followerMesh = this.viewer.rendererManager.mesh.get3d(followerNode);\n if (!c1Mesh || !c2Mesh)\n return;\n /**\n * Move along sketch connectors cannot be moved, because their position is being set by the sketch itself. They can only be used\n * to determine the position of other connectors. Moving them will cause an immediate overconstraint condition\n */\n if (c2.positionMode === exports.eConnectorPositionMode.moveAlongSketch) {\n return;\n }\n if (isSketchControlPoint(c2)) {\n const cp = c2;\n this.viewer.pauseRenderWhile(() => {\n const nodeWithTransform = mateControlPointToConnector(mate, c1Mesh, c2Mesh);\n const d1 = nodeWithTransform.direction.scale(cp.direction.length());\n const d2 = cp.direction2 ? nodeWithTransform.direction.scale(cp.direction2.length()) : undefined;\n if (!nodeWithTransform.position.equalsWithEpsilon(cp.position.toVec3()) ||\n !d1.equalsWithEpsilon(cp.direction.toVec3()) ||\n (cp.direction2 && d2 && !d2.equalsWithEpsilon(cp.direction2.toVec3()))) {\n this.afterCleanup(() => {\n cp.position = exports.KbVector.FromVec3(nodeWithTransform.position);\n cp.direction = exports.KbVector.FromVec3(d1);\n if (cp.direction2 && d2) {\n cp.direction2 = exports.KbVector.FromVec3(d2);\n }\n });\n }\n });\n }\n else if (followerMesh) {\n this.viewer.pauseRenderWhile(() => {\n const originalFollowerMeshAbsPos = followerMesh.getAbsolutePosition();\n const originalFollowerMeshAbsQuat = followerMesh.absoluteRotationQuaternion.clone();\n const originalFollowerMeshLocalPos = followerMesh.position.clone();\n const originalFollowerMeshLocalQuat = followerMesh.rotationQuaternion.clone();\n mateConnectors(mate, followerMesh, c1Mesh, c2Mesh);\n /** update real kb3d nodes with new transform info */\n if (!originalFollowerMeshLocalPos.equalsWithEpsilon(followerMesh.position) ||\n !originalFollowerMeshLocalQuat.equalsWithEpsilon(followerMesh.rotationQuaternion)) {\n followerNode.trackChanges(false);\n followerNode.setRotationQuaternion(exports.KbQuaternion.FromQuat(followerMesh.rotationQuaternion));\n followerNode.position = exports.KbVector.FromVec3(followerMesh.position);\n followerNode.trackChanges(true);\n this.referenceProcessor.triggerChangesOnReferencers(followerNode, new Map([['position', null]]));\n }\n /** invalidate bounding box of parent if follower mesh actually moved */\n if (!originalFollowerMeshAbsPos.equalsWithEpsilon(followerMesh.getAbsolutePosition()) ||\n !originalFollowerMeshAbsQuat.equalsWithEpsilon(followerMesh.absoluteRotationQuaternion)) {\n this.viewer.bbCache.invalidate(c2.getSpaceParent());\n }\n //console.log(`mate '${mate.name} processed`);\n });\n }\n }\n getConnectorMesh(c) {\n if (isSketchControlPoint(c)) {\n return this.controlPointRenderer.get3d(c);\n }\n else if (isConnector(c)) {\n return this.connectorRenderer.get3d(c);\n }\n }\n buildDependencies() {\n const mates = this.getValidMates();\n const sortResult = sortMates(mates);\n this.sortedMates = sortResult.sorted;\n this.cycles = sortResult.cycles;\n const validator = this.viewer.options.validator;\n for (const cycle of sortResult.cycles) {\n const err = `${CYCLE_ERROR_START}${cycle.map(n => n.name).join(' -> ')}`;\n if (validator) {\n for (const cm of cycle) {\n validator.addError(cm, '', err);\n }\n }\n else {\n console.warn(err);\n }\n }\n if (validator) {\n for (const om of sortResult.overConstrained) {\n validator.addError(om, '', `Cannot have more than one fixed node in the mate chain`);\n }\n }\n else {\n if (sortResult.overConstrained.length) {\n console.warn(`Mates overconstrained: [${sortResult.overConstrained.map(n => n.name).join(', ')}]`);\n }\n }\n }\n /**\n * gets all mates (including those in nested scenes)\n * that are enabled, and have valid references to connectors\n */\n getValidMates() {\n const rootScene = this.viewer.sceneNode;\n const mates = new Array();\n const validator = this.viewer.options.validator;\n rootScene.runOnAllScenes(s => {\n const sceneMates = s._manager.getByToken(Token.MateNode);\n for (const m of sceneMates) {\n if (m.isActive()) {\n mates.push(m);\n }\n //regardless, clear out cycle validation\n if (validator) {\n validator.clearProp(m, '');\n }\n }\n });\n return mates;\n }\n }\n\n class CircularPatternRenderer extends PatternRenderer {\n constructor() {\n super(...arguments);\n this.modelToken = Token.CircularPatternFeature;\n }\n updateMeshGeometry(feature, vertexData, vertexDataChanges) {\n const featurePivot = feature.pivot.toVec3();\n const featureAxis = feature.axis.toVec3();\n // Don't allow a zero vector for an axis because it will blow up the math\n if (featureAxis.equalsToFloats(0, 0, 0)) {\n featureAxis.set(0, 1, 0);\n }\n const featureArc = (feature.arc * Math.PI) / 180 / feature.count;\n const workingQuaternion = Quaternion.Identity();\n return super.updateMeshGeometry(feature, vertexData, vertexDataChanges, (idx, particle) => {\n particle.pivot = featurePivot;\n Quaternion.RotationAxisToRef(featureAxis, featureArc * idx, workingQuaternion);\n particle.rotationQuaternion.multiplyInPlace(workingQuaternion);\n });\n }\n }\n\n class DeleteFacesRenderer extends FeatureRenderer {\n constructor() {\n super(...arguments);\n this.modelToken = Token.DeleteFacesFeature;\n }\n updateMeshGeometry(feature, vertexData, vertexDataChanges) {\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\n if (newVertexData.indices && feature.vertexGroups.length > 0) {\n const vertexGroups = feature.vertexGroups.sort((groupA, groupB) => groupA.startIndex - groupB.startIndex);\n const newIndices = new Int32Array(newVertexData.indices);\n let currReadIndex = 0;\n vertexGroups.forEach(vertexGroup => {\n for (let i = Math.max(vertexGroup.startIndex, currReadIndex); i < Math.min(vertexGroup.endIndex, newIndices.length); i++) {\n newIndices[i] = -1;\n }\n currReadIndex = vertexGroup.endIndex;\n });\n // We need to remove submeshes targetting the indices that are being removed as well\n if (newVertexData.submeshData) {\n newVertexData.submeshData.forEach(submeshData => {\n const indexStart = submeshData.indexStart;\n for (let i = 0; i < indexStart; i++) {\n if (newIndices[i] === -1) {\n submeshData.indexStart--;\n }\n }\n const indexCount = submeshData.indexCount;\n for (let i = 0; i < indexCount; i++) {\n const index = indexStart + i;\n if (newIndices[index] === -1) {\n submeshData.indexCount--;\n }\n }\n });\n }\n newVertexData.indices = newIndices.filter(index => index !== -1);\n }\n return newVertexData;\n }\n update3dInternal() { }\n }\n\n class DisplacementMapRenderer extends FeatureRenderer {\n constructor() {\n super(...arguments);\n this.modelToken = Token.DisplacementMapFeature;\n this.textureCanvas = document.createElement('canvas');\n }\n updateMeshGeometry(feature, vertexData, vertexDataChanges) {\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\n const textureRenderer = this.viewer.rendererManager.textures;\n if (feature.textures && feature.textures.length > 0) {\n //create context\n const context = this.textureCanvas.getContext('2d', {\n willReadFrequently: true,\n });\n const strength = 1;\n const hasTransparency = false;\n const parameters = { strength, hasTransparency };\n const textureLayers = textureRenderer.getActiveLayers(feature, exports.eTextureChannel.albedo).filter(node => {\n const renderer = textureRenderer.getRenderer(node);\n return renderer.textureHasContent(node);\n });\n if (context) {\n const baseImageSize = textureRenderer\n .getRenderer(feature.textures[0])\n .getNativeSize(feature.textures[0]);\n if (this.scene.getEngine().webGLVersion < 2) {\n const clampedSize = getBinClampedSize(baseImageSize.width, baseImageSize.height);\n context.canvas.width = clampedSize;\n context.canvas.height = clampedSize;\n }\n else {\n if (feature.textures[0]) {\n context.canvas.width = baseImageSize.width;\n context.canvas.height = baseImageSize.height;\n }\n }\n context.clearRect(0, 0, context.canvas.width, context.canvas.height);\n const paintedImage = textureRenderer.paintTextureOnCanvas(textureLayers, context, parameters);\n const tempVertexData = new SubmeshVertexData();\n tempVertexData.copyFrom(newVertexData);\n if (tempVertexData.uvs) {\n const n = tempVertexData.uvs.length;\n for (let i = 1; i < n; i += 2) {\n tempVertexData.uvs[i] = 1 - tempVertexData.uvs[i];\n }\n const mesh = this.viewer.getRendererNode(feature._parent);\n const meshUpdated = new Mesh('ExistingMesh');\n tempVertexData.applyToMesh(meshUpdated, true);\n ensureNormalsExist(meshUpdated);\n const imageData = paintedImage === null || paintedImage === void 0 ? void 0 : paintedImage.getImageData(0, 0, this.textureCanvas.width, this.textureCanvas.height, mesh);\n if (imageData) {\n this.displace(imageData, feature.minDisplacement, feature.maxDisplacement, meshUpdated);\n }\n const meshVertexData = VertexData.ExtractFromMesh(meshUpdated);\n newVertexData.positions = meshVertexData.positions;\n newVertexData.normals = meshVertexData.normals;\n meshUpdated.dispose();\n }\n }\n }\n return newVertexData;\n }\n update3dInternal() { }\n displace(context, minDisplacement, maxDisplacement, parentMesh) {\n const uInt8Array = new Uint8Array(context.data);\n parentMesh.applyDisplacementMapFromBuffer(uInt8Array, context.width, context.height, minDisplacement, maxDisplacement);\n }\n }\n\n const xztoxy = Matrix.FromArray([1, 0, 0, 0, 0, 0, -1, 0, 0, 1, 0, 0, 0, 0, 0, 1]);\n class ExtrudeRenderer extends FeatureRenderer {\n constructor() {\n super(...arguments);\n this.modelToken = Token.ExtrudeFeature;\n }\n updateMeshGeometry(feature, vertexData, vertexDataChanges) {\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\n const parentMesh = feature._parent;\n if (parentMesh &&\n newVertexData.positions &&\n newVertexData.positions.length &&\n // paths are used by parseSketchGeometry\n newVertexData.paths &&\n newVertexData.paths.length > 0 &&\n feature.size !== 0) {\n const { segments, holes } = parseSketchGeometry(newVertexData);\n const extrudedMeshes = [];\n for (let i = 0; i < segments.length; i++) {\n const shape = segments[i];\n const shapeHoles = holes.get(i);\n const sketch = feature._parent && isSketchNode(feature._parent) ? feature._parent : undefined;\n const sketchNormal = sketch ? axisToVector3(sketch.normal) : Vector3.Zero();\n if (sketch && sketch.normal !== exports.eAxis.z) {\n sketchNormal.scaleInPlace(-1);\n }\n const rotToZ = rotationToVector(sketchNormal, new Vector3(0, 0, 1));\n const pathTransformToZ = new Matrix();\n const pathTransformToZInv = new Matrix();\n rotToZ.toRotationMatrix(pathTransformToZ);\n pathTransformToZ.invertToRef(pathTransformToZInv);\n for (let i = 0; i < shape.length; i++) {\n Vector3.TransformCoordinatesToRef(shape[i], pathTransformToZ, shape[i]);\n }\n const shape2d = shape.map(point => new Vector2(point.x, point.y));\n const builder = new PolygonMeshBuilder(feature._dynamicId, shape2d, this.scene, earcut$1);\n if (shapeHoles) {\n shapeHoles.forEach(holePoints => {\n for (let i = 0; i < holePoints.length; i++) {\n Vector3.TransformCoordinatesToRef(holePoints[i], pathTransformToZ, holePoints[i]);\n }\n const hole2d = holePoints.map(point => new Vector2(point.x, point.y));\n builder.addHole(hole2d);\n });\n }\n const extruded = builder.build(true, Math.abs(feature.size), 0.9);\n xztoxy.multiplyToRef(pathTransformToZInv, pathTransformToZInv);\n extruded.bakeTransformIntoVertices(pathTransformToZInv);\n extrudedMeshes.push(extruded);\n }\n const extrudedAll = Mesh.MergeMeshes(extrudedMeshes, true);\n if (extrudedAll) {\n applyExtrusion(newVertexData, extrudedAll, false);\n extrudedAll.dispose();\n }\n }\n return newVertexData;\n }\n getSketchShape(positions) {\n const shape = [];\n for (let i = 0; i < positions.length; i += 3) {\n const newPoint = new Vector3(positions[i], positions[i + 1], positions[i + 2]);\n if (shape.length === 0 || !shape[shape.length - 1].equals(newPoint)) {\n shape.push(newPoint);\n }\n }\n return shape;\n }\n update3dInternal() { }\n }\n function applyExtrusion(newVertexData, extruded, reverse = false) {\n if (extruded.geometry) {\n // Store the indices used for the extrude so that they can be merged into the new geometry\n const newIndices = extruded.geometry.getIndices(false, true);\n const newNormals = extruded.geometry.getVerticesData(VertexBuffer.NormalKind, false, true);\n const newUvs = extruded.geometry.getVerticesData(VertexBuffer.UVKind, false, true);\n const newPositions = extruded.geometry.getVerticesData(VertexBuffer.PositionKind, false, true);\n if (newPositions && reverse) {\n for (let i = 2; i < newPositions.length; i += 3) {\n newPositions[i] = -newPositions[i];\n }\n }\n normalizeGeometry(newVertexData);\n if (!newVertexData.indices) {\n newVertexData.indices = newIndices;\n }\n else if (newIndices) {\n const concatArr = new Int32Array(newVertexData.indices.length + newIndices.length);\n concatArr.set(newVertexData.indices);\n if (newVertexData.positions) {\n const pushAmount = newVertexData.positions.length / 3;\n for (let i = 0; i < newIndices.length; i++) {\n newIndices[i] += pushAmount;\n }\n }\n concatArr.set(newIndices, newVertexData.indices.length);\n newVertexData.indices = concatArr;\n }\n if (!newVertexData.positions) {\n newVertexData.positions = newPositions;\n }\n else if (newPositions) {\n const concatArr = new Float32Array(newVertexData.positions.length + newPositions.length);\n concatArr.set(newVertexData.positions);\n concatArr.set(newPositions, newVertexData.positions.length);\n newVertexData.positions = concatArr;\n }\n if (!newVertexData.normals) {\n newVertexData.normals = newNormals;\n }\n else if (newNormals) {\n const concatArr = new Float32Array(newVertexData.normals.length + newNormals.length);\n concatArr.set(newVertexData.normals);\n concatArr.set(newNormals, newVertexData.normals.length);\n newVertexData.normals = concatArr;\n }\n if (!newVertexData.uvs) {\n newVertexData.uvs = newUvs;\n }\n else if (newUvs) {\n const concatArr = new Float32Array(newVertexData.uvs.length + newUvs.length);\n concatArr.set(newVertexData.uvs);\n concatArr.set(newUvs, newVertexData.uvs.length);\n newVertexData.uvs = concatArr;\n }\n const baseMaterialIndex = newVertexData.submeshData ? newVertexData.submeshData[0].materialIndex : 0;\n const totalIndicesResultCount = newVertexData.indices.length;\n const totalVerticesResultCount = newVertexData.positions.length / 3;\n newVertexData.submeshData = [];\n newVertexData.submeshData.push({\n materialIndex: baseMaterialIndex,\n verticesStart: 0,\n verticesCount: totalVerticesResultCount,\n indexStart: 0,\n indexCount: totalIndicesResultCount,\n });\n }\n }\n function normalizeGeometry(newVertexData) {\n if (!newVertexData.indices) {\n // If there are no indices, throw away existing vertices because they won't be used\n newVertexData.positions = null;\n newVertexData.normals = null;\n newVertexData.uvs = null;\n }\n else {\n const indices = newVertexData.indices;\n let maxIdx = 0;\n for (let i = 0; i < indices.length; i++) {\n if (maxIdx < indices[i]) {\n maxIdx = indices[i];\n }\n }\n const usedVertexLength = maxIdx + 1;\n const positions = new Float32Array(usedVertexLength * 3);\n const normals = new Float32Array(usedVertexLength * 3);\n const uvs = new Float32Array(usedVertexLength * 2);\n if (newVertexData.positions) {\n if (Array.isArray(newVertexData.positions)) {\n positions.set(newVertexData.positions.slice(0, positions.length));\n }\n else {\n positions.set(newVertexData.positions.subarray(0, positions.length));\n }\n }\n if (newVertexData.normals) {\n if (Array.isArray(newVertexData.normals)) {\n normals.set(newVertexData.normals.slice(0, normals.length));\n }\n else {\n normals.set(newVertexData.normals.subarray(0, normals.length));\n }\n }\n if (newVertexData.uvs) {\n if (Array.isArray(newVertexData.uvs)) {\n uvs.set(newVertexData.uvs.slice(0, uvs.length));\n }\n else {\n uvs.set(newVertexData.uvs.subarray(0, uvs.length));\n }\n }\n newVertexData.positions = positions;\n newVertexData.normals = normals;\n newVertexData.uvs = uvs;\n }\n }\n\n /**\n * Unique ID when we import meshes from Babylon to CSG\n */\n let currentCSGMeshId = 0;\n /**\n * Represents a vertex of a polygon. Use your own vertex class instead of this\n * one to provide additional features like texture coordinates and vertex\n * colors. Custom vertex classes need to provide a `pos` property and `clone()`,\n * `flip()`, and `interpolate()` methods that behave analogous to the ones\n * defined by `BABYLON.CSG.Vertex`. This class provides `normal` so convenience\n * functions like `BABYLON.CSG.sphere()` can return a smooth vertex normal, but `normal`\n * is not used anywhere else.\n * Same goes for uv, it allows to keep the original vertex uv coordinates of the 2 meshes\n */\n class Vertex {\n /**\n * Initializes the vertex\n * @param pos The position of the vertex\n * @param normal The normal of the vertex\n * @param uv The texture coordinate of the vertex\n * @param vertColor The RGBA color of the vertex\n */\n constructor(\n /**\n * The position of the vertex\n */\n pos, \n /**\n * The normal of the vertex\n */\n normal, \n /**\n * The texture coordinate of the vertex\n */\n uv, \n /**\n * The texture coordinate of the vertex\n */\n vertColor) {\n this.pos = pos;\n this.normal = normal;\n this.uv = uv;\n this.vertColor = vertColor;\n }\n /**\n * Make a clone, or deep copy, of the vertex\n * @returns A new Vertex\n */\n clone() {\n var _a, _b;\n return new Vertex(this.pos.clone(), this.normal.clone(), (_a = this.uv) === null || _a === void 0 ? void 0 : _a.clone(), (_b = this.vertColor) === null || _b === void 0 ? void 0 : _b.clone());\n }\n /**\n * Invert all orientation-specific data (e.g. vertex normal). Called when the\n * orientation of a polygon is flipped.\n */\n flip() {\n this.normal = this.normal.scale(-1);\n }\n /**\n * Create a new vertex between this vertex and `other` by linearly\n * interpolating all properties using a parameter of `t`. Subclasses should\n * override this to interpolate additional properties.\n * @param other the vertex to interpolate against\n * @param t The factor used to linearly interpolate between the vertices\n */\n interpolate(other, t) {\n return new Vertex(Vector3.Lerp(this.pos, other.pos, t), Vector3.Lerp(this.normal, other.normal, t), this.uv && other.uv ? Vector2.Lerp(this.uv, other.uv, t) : undefined, this.vertColor && other.vertColor ? Color4.Lerp(this.vertColor, other.vertColor, t) : undefined);\n }\n }\n /**\n * Represents a plane in 3D space.\n */\n class Plane$1 {\n /**\n * Initializes the plane\n * @param normal The normal for the plane\n * @param w\n */\n constructor(normal, w) {\n this.normal = normal;\n this.w = w;\n }\n /**\n * Construct a plane from three points\n * @param a Point a\n * @param b Point b\n * @param c Point c\n */\n static FromPoints(a, b, c) {\n const v0 = c.subtract(a);\n const v1 = b.subtract(a);\n if (v0.lengthSquared() === 0 || v1.lengthSquared() === 0) {\n return null;\n }\n const n = Vector3.Normalize(Vector3.Cross(v0, v1));\n return new Plane$1(n, Vector3.Dot(n, a));\n }\n /**\n * Clone, or make a deep copy of the plane\n * @returns a new Plane\n */\n clone() {\n return new Plane$1(this.normal.clone(), this.w);\n }\n /**\n * Flip the face of the plane\n */\n flip() {\n this.normal.scaleInPlace(-1);\n this.w = -this.w;\n }\n /**\n * Split `polygon` by this plane if needed, then put the polygon or polygon\n * fragments in the appropriate lists. Coplanar polygons go into either\n `* coplanarFront` or `coplanarBack` depending on their orientation with\n * respect to this plane. Polygons in front or in back of this plane go into\n * either `front` or `back`\n * @param polygon The polygon to be split\n * @param coplanarFront Will contain polygons coplanar with the plane that are oriented to the front of the plane\n * @param coplanarBack Will contain polygons coplanar with the plane that are oriented to the back of the plane\n * @param front Will contain the polygons in front of the plane\n * @param back Will contain the polygons begind the plane\n */\n splitPolygon(polygon, coplanarFront, coplanarBack, front, back) {\n const COPLANAR = 0;\n const FRONT = 1;\n const BACK = 2;\n const SPANNING = 3;\n // Classify each point as well as the entire polygon into one of the above\n // four classes.\n let polygonType = 0;\n const types = [];\n let i;\n let t;\n for (i = 0; i < polygon.vertices.length; i++) {\n t = Vector3.Dot(this.normal, polygon.vertices[i].pos) - this.w;\n const type = t < -Plane$1.EPSILON ? BACK : t > Plane$1.EPSILON ? FRONT : COPLANAR;\n polygonType |= type;\n types.push(type);\n }\n // Put the polygon in the correct list, splitting it when necessary\n switch (polygonType) {\n case COPLANAR:\n (Vector3.Dot(this.normal, polygon.plane.normal) > 0 ? coplanarFront : coplanarBack).push(polygon);\n break;\n case FRONT:\n front.push(polygon);\n break;\n case BACK:\n back.push(polygon);\n break;\n case SPANNING: {\n const f = [], b = [];\n for (i = 0; i < polygon.vertices.length; i++) {\n const j = (i + 1) % polygon.vertices.length;\n const ti = types[i], tj = types[j];\n const vi = polygon.vertices[i], vj = polygon.vertices[j];\n if (ti !== BACK) {\n f.push(vi);\n }\n if (ti !== FRONT) {\n b.push(ti !== BACK ? vi.clone() : vi);\n }\n if ((ti | tj) === SPANNING) {\n t = (this.w - Vector3.Dot(this.normal, vi.pos)) / Vector3.Dot(this.normal, vj.pos.subtract(vi.pos));\n const v = vi.interpolate(vj, t);\n f.push(v);\n b.push(v.clone());\n }\n }\n let poly;\n if (f.length >= 3) {\n poly = new Polygon(f, polygon.shared);\n if (poly.plane) {\n front.push(poly);\n }\n }\n if (b.length >= 3) {\n poly = new Polygon(b, polygon.shared);\n if (poly.plane) {\n back.push(poly);\n }\n }\n break;\n }\n }\n }\n }\n /**\n * `CSG.Plane.EPSILON` is the tolerance used by `splitPolygon()` to decide if a\n * point is on the plane\n */\n Plane$1.EPSILON = 1e-5;\n /**\n * Represents a convex polygon. The vertices used to initialize a polygon must\n * be coplanar and form a convex loop.\n *\n * Each convex polygon has a `shared` property, which is shared between all\n * polygons that are clones of each other or were split from the same polygon.\n * This can be used to define per-polygon properties (such as surface color)\n */\n class Polygon {\n /**\n * Initializes the polygon\n * @param vertices The vertices of the polygon\n * @param shared The properties shared across all polygons\n */\n constructor(vertices, shared) {\n this.vertices = vertices;\n this.shared = shared;\n this.plane = Plane$1.FromPoints(vertices[0].pos, vertices[1].pos, vertices[2].pos);\n }\n /**\n * Clones, or makes a deep copy, or the polygon\n */\n clone() {\n const vertices = this.vertices.map((v) => v.clone());\n return new Polygon(vertices, this.shared);\n }\n /**\n * Flips the faces of the polygon\n */\n flip() {\n this.vertices.reverse().map((v) => {\n v.flip();\n });\n this.plane.flip();\n }\n }\n /**\n * Holds a node in a BSP tree. A BSP tree is built from a collection of polygons\n * by picking a polygon to split along. That polygon (and all other coplanar\n * polygons) are added directly to that node and the other polygons are added to\n * the front and/or back subtrees. This is not a leafy BSP tree since there is\n * no distinction between internal and leaf nodes\n */\n class Node$2 {\n /**\n * Initializes the node\n * @param polygons A collection of polygons held in the node\n */\n constructor(polygons) {\n this._plane = null;\n this._front = null;\n this._back = null;\n this._polygons = new Array();\n if (polygons) {\n this.build(polygons);\n }\n }\n /**\n * Clones, or makes a deep copy, of the node\n * @returns The cloned node\n */\n clone() {\n const node = new Node$2();\n node._plane = this._plane && this._plane.clone();\n node._front = this._front && this._front.clone();\n node._back = this._back && this._back.clone();\n node._polygons = this._polygons.map((p) => p.clone());\n return node;\n }\n /**\n * Convert solid space to empty space and empty space to solid space\n */\n invert() {\n for (let i = 0; i < this._polygons.length; i++) {\n this._polygons[i].flip();\n }\n if (this._plane) {\n this._plane.flip();\n }\n if (this._front) {\n this._front.invert();\n }\n if (this._back) {\n this._back.invert();\n }\n const temp = this._front;\n this._front = this._back;\n this._back = temp;\n }\n /**\n * Recursively remove all polygons in `polygons` that are inside this BSP\n * tree.\n * @param polygons Polygons to remove from the BSP\n * @returns Polygons clipped from the BSP\n */\n clipPolygons(polygons) {\n if (!this._plane) {\n return polygons.slice();\n }\n let front = new Array(), back = new Array();\n for (let i = 0; i < polygons.length; i++) {\n this._plane.splitPolygon(polygons[i], front, back, front, back);\n }\n if (this._front) {\n front = this._front.clipPolygons(front);\n }\n if (this._back) {\n back = this._back.clipPolygons(back);\n }\n else {\n back = [];\n }\n return front.concat(back);\n }\n /**\n * Remove all polygons in this BSP tree that are inside the other BSP tree\n * `bsp`.\n * @param bsp BSP containing polygons to remove from this BSP\n */\n clipTo(bsp) {\n this._polygons = bsp.clipPolygons(this._polygons);\n if (this._front) {\n this._front.clipTo(bsp);\n }\n if (this._back) {\n this._back.clipTo(bsp);\n }\n }\n /**\n * Return a list of all polygons in this BSP tree\n * @returns List of all polygons in this BSP tree\n */\n allPolygons() {\n let polygons = this._polygons.slice();\n if (this._front) {\n polygons = polygons.concat(this._front.allPolygons());\n }\n if (this._back) {\n polygons = polygons.concat(this._back.allPolygons());\n }\n return polygons;\n }\n /**\n * Build a BSP tree out of `polygons`. When called on an existing tree, the\n * new polygons are filtered down to the bottom of the tree and become new\n * nodes there. Each set of polygons is partitioned using the first polygon\n * (no heuristic is used to pick a good split)\n * @param polygons Polygons used to construct the BSP tree\n */\n build(polygons) {\n if (!polygons.length) {\n return;\n }\n if (!this._plane) {\n this._plane = polygons[0].plane.clone();\n }\n const front = new Array(), back = new Array();\n for (let i = 0; i < polygons.length; i++) {\n this._plane.splitPolygon(polygons[i], this._polygons, this._polygons, front, back);\n }\n if (front.length) {\n if (!this._front) {\n this._front = new Node$2();\n }\n this._front.build(front);\n }\n if (back.length) {\n if (!this._back) {\n this._back = new Node$2();\n }\n this._back.build(back);\n }\n }\n }\n /**\n * Class for building Constructive Solid Geometry\n */\n class CSG {\n constructor() {\n this._polygons = new Array();\n }\n /**\n * Convert the Mesh to CSG\n * @param mesh The Mesh to convert to CSG\n * @param absolute If true, the final (local) matrix transformation is set to the identity and not to that of `mesh`. It can help when dealing with right-handed meshes (default: false)\n * @returns A new CSG from the Mesh\n */\n static FromMesh(mesh, absolute = false) {\n let vertex, normal, uv = undefined, position, vertColor = undefined, polygon, vertices;\n const polygons = new Array();\n let matrix, meshPosition, meshRotation, meshRotationQuaternion = null, meshScaling;\n let invertWinding = false;\n if (mesh instanceof Mesh) {\n mesh.computeWorldMatrix(true);\n matrix = mesh.getWorldMatrix();\n meshPosition = mesh.position.clone();\n meshRotation = mesh.rotation.clone();\n if (mesh.rotationQuaternion) {\n meshRotationQuaternion = mesh.rotationQuaternion.clone();\n }\n meshScaling = mesh.scaling.clone();\n if (mesh.material && absolute) {\n invertWinding = mesh.material.sideOrientation === 0;\n }\n }\n else {\n throw \"BABYLON.CSG: Wrong Mesh type, must be BABYLON.Mesh\";\n }\n const indices = mesh.getIndices(), positions = mesh.getVerticesData(VertexBuffer.PositionKind), normals = mesh.getVerticesData(VertexBuffer.NormalKind), uvs = mesh.getVerticesData(VertexBuffer.UVKind), vertColors = mesh.getVerticesData(VertexBuffer.ColorKind);\n const subMeshes = mesh.subMeshes;\n for (let sm = 0, sml = subMeshes.length; sm < sml; sm++) {\n for (let i = subMeshes[sm].indexStart, il = subMeshes[sm].indexCount + subMeshes[sm].indexStart; i < il; i += 3) {\n vertices = [];\n for (let j = 0; j < 3; j++) {\n const indexIndices = j === 0 ? i + j : invertWinding ? i + 3 - j : i + j;\n const sourceNormal = new Vector3(normals[indices[indexIndices] * 3], normals[indices[indexIndices] * 3 + 1], normals[indices[indexIndices] * 3 + 2]);\n if (uvs) {\n uv = new Vector2(uvs[indices[indexIndices] * 2], uvs[indices[indexIndices] * 2 + 1]);\n }\n if (vertColors) {\n vertColor = new Color4(vertColors[indices[indexIndices] * 4], vertColors[indices[indexIndices] * 4 + 1], vertColors[indices[indexIndices] * 4 + 2], vertColors[indices[indexIndices] * 4 + 3]);\n }\n const sourcePosition = new Vector3(positions[indices[indexIndices] * 3], positions[indices[indexIndices] * 3 + 1], positions[indices[indexIndices] * 3 + 2]);\n position = Vector3.TransformCoordinates(sourcePosition, matrix);\n normal = Vector3.TransformNormal(sourceNormal, matrix);\n vertex = new Vertex(position, normal, uv, vertColor);\n vertices.push(vertex);\n }\n polygon = new Polygon(vertices, { subMeshId: sm, meshId: currentCSGMeshId, materialIndex: subMeshes[sm].materialIndex });\n // To handle the case of degenerated triangle\n // polygon.plane == null <=> the polygon does not represent 1 single plane <=> the triangle is degenerated\n if (polygon.plane) {\n polygons.push(polygon);\n }\n }\n }\n const csg = CSG._FromPolygons(polygons);\n csg.matrix = absolute ? Matrix.Identity() : matrix;\n csg.position = absolute ? Vector3.Zero() : meshPosition;\n csg.rotation = absolute ? Vector3.Zero() : meshRotation;\n csg.scaling = absolute ? Vector3.One() : meshScaling;\n csg.rotationQuaternion = absolute && meshRotationQuaternion ? Quaternion.Identity() : meshRotationQuaternion;\n currentCSGMeshId++;\n return csg;\n }\n /**\n * Construct a CSG solid from a list of `CSG.Polygon` instances.\n * @param polygons Polygons used to construct a CSG solid\n */\n static _FromPolygons(polygons) {\n const csg = new CSG();\n csg._polygons = polygons;\n return csg;\n }\n /**\n * Clones, or makes a deep copy, of the CSG\n * @returns A new CSG\n */\n clone() {\n const csg = new CSG();\n csg._polygons = this._polygons.map((p) => p.clone());\n csg.copyTransformAttributes(this);\n return csg;\n }\n /**\n * Unions this CSG with another CSG\n * @param csg The CSG to union against this CSG\n * @returns The unioned CSG\n */\n union(csg) {\n const a = new Node$2(this.clone()._polygons);\n const b = new Node$2(csg.clone()._polygons);\n a.clipTo(b);\n b.clipTo(a);\n b.invert();\n b.clipTo(a);\n b.invert();\n a.build(b.allPolygons());\n return CSG._FromPolygons(a.allPolygons()).copyTransformAttributes(this);\n }\n /**\n * Unions this CSG with another CSG in place\n * @param csg The CSG to union against this CSG\n */\n unionInPlace(csg) {\n const a = new Node$2(this._polygons);\n const b = new Node$2(csg._polygons);\n a.clipTo(b);\n b.clipTo(a);\n b.invert();\n b.clipTo(a);\n b.invert();\n a.build(b.allPolygons());\n this._polygons = a.allPolygons();\n }\n /**\n * Subtracts this CSG with another CSG\n * @param csg The CSG to subtract against this CSG\n * @returns A new CSG\n */\n subtract(csg) {\n const a = new Node$2(this.clone()._polygons);\n const b = new Node$2(csg.clone()._polygons);\n a.invert();\n a.clipTo(b);\n b.clipTo(a);\n b.invert();\n b.clipTo(a);\n b.invert();\n a.build(b.allPolygons());\n a.invert();\n return CSG._FromPolygons(a.allPolygons()).copyTransformAttributes(this);\n }\n /**\n * Subtracts this CSG with another CSG in place\n * @param csg The CSG to subtract against this CSG\n */\n subtractInPlace(csg) {\n const a = new Node$2(this._polygons);\n const b = new Node$2(csg._polygons);\n a.invert();\n a.clipTo(b);\n b.clipTo(a);\n b.invert();\n b.clipTo(a);\n b.invert();\n a.build(b.allPolygons());\n a.invert();\n this._polygons = a.allPolygons();\n }\n /**\n * Intersect this CSG with another CSG\n * @param csg The CSG to intersect against this CSG\n * @returns A new CSG\n */\n intersect(csg) {\n const a = new Node$2(this.clone()._polygons);\n const b = new Node$2(csg.clone()._polygons);\n a.invert();\n b.clipTo(a);\n b.invert();\n a.clipTo(b);\n b.clipTo(a);\n a.build(b.allPolygons());\n a.invert();\n return CSG._FromPolygons(a.allPolygons()).copyTransformAttributes(this);\n }\n /**\n * Intersects this CSG with another CSG in place\n * @param csg The CSG to intersect against this CSG\n */\n intersectInPlace(csg) {\n const a = new Node$2(this._polygons);\n const b = new Node$2(csg._polygons);\n a.invert();\n b.clipTo(a);\n b.invert();\n a.clipTo(b);\n b.clipTo(a);\n a.build(b.allPolygons());\n a.invert();\n this._polygons = a.allPolygons();\n }\n /**\n * Return a new CSG solid with solid and empty space switched. This solid is\n * not modified.\n * @returns A new CSG solid with solid and empty space switched\n */\n inverse() {\n const csg = this.clone();\n csg.inverseInPlace();\n return csg;\n }\n /**\n * Inverses the CSG in place\n */\n inverseInPlace() {\n this._polygons.map((p) => {\n p.flip();\n });\n }\n /**\n * This is used to keep meshes transformations so they can be restored\n * when we build back a Babylon Mesh\n * NB : All CSG operations are performed in world coordinates\n * @param csg The CSG to copy the transform attributes from\n * @returns This CSG\n */\n copyTransformAttributes(csg) {\n this.matrix = csg.matrix;\n this.position = csg.position;\n this.rotation = csg.rotation;\n this.scaling = csg.scaling;\n this.rotationQuaternion = csg.rotationQuaternion;\n return this;\n }\n /**\n * Build Raw mesh from CSG\n * Coordinates here are in world space\n * @param name The name of the mesh geometry\n * @param scene The Scene\n * @param keepSubMeshes Specifies if the submeshes should be kept\n * @returns A new Mesh\n */\n buildMeshGeometry(name, scene, keepSubMeshes) {\n const matrix = this.matrix.clone();\n matrix.invert();\n const mesh = new Mesh(name, scene);\n const vertices = [];\n const indices = [];\n const normals = [];\n let uvs = null;\n let vertColors = null;\n const vertex = Vector3.Zero();\n const normal = Vector3.Zero();\n const uv = Vector2.Zero();\n const vertColor = new Color4(0, 0, 0, 0);\n const polygons = this._polygons;\n const polygonIndices = [0, 0, 0];\n let polygon;\n const vertice_dict = {};\n let vertex_idx;\n let currentIndex = 0;\n const subMeshDict = {};\n let subMeshObj;\n if (keepSubMeshes) {\n // Sort Polygons, since subMeshes are indices range\n polygons.sort((a, b) => {\n if (a.shared.meshId === b.shared.meshId) {\n return a.shared.subMeshId - b.shared.subMeshId;\n }\n else {\n return a.shared.meshId - b.shared.meshId;\n }\n });\n }\n for (let i = 0, il = polygons.length; i < il; i++) {\n polygon = polygons[i];\n // Building SubMeshes\n if (!subMeshDict[polygon.shared.meshId]) {\n subMeshDict[polygon.shared.meshId] = {};\n }\n if (!subMeshDict[polygon.shared.meshId][polygon.shared.subMeshId]) {\n subMeshDict[polygon.shared.meshId][polygon.shared.subMeshId] = {\n indexStart: +Infinity,\n indexEnd: -Infinity,\n materialIndex: polygon.shared.materialIndex,\n };\n }\n subMeshObj = subMeshDict[polygon.shared.meshId][polygon.shared.subMeshId];\n for (let j = 2, jl = polygon.vertices.length; j < jl; j++) {\n polygonIndices[0] = 0;\n polygonIndices[1] = j - 1;\n polygonIndices[2] = j;\n for (let k = 0; k < 3; k++) {\n vertex.copyFrom(polygon.vertices[polygonIndices[k]].pos);\n normal.copyFrom(polygon.vertices[polygonIndices[k]].normal);\n if (polygon.vertices[polygonIndices[k]].uv) {\n if (!uvs) {\n uvs = [];\n }\n uv.copyFrom(polygon.vertices[polygonIndices[k]].uv);\n }\n if (polygon.vertices[polygonIndices[k]].vertColor) {\n if (!vertColors) {\n vertColors = [];\n }\n vertColor.copyFrom(polygon.vertices[polygonIndices[k]].vertColor);\n }\n const localVertex = Vector3.TransformCoordinates(vertex, matrix);\n const localNormal = Vector3.TransformNormal(normal, matrix);\n vertex_idx = vertice_dict[localVertex.x + \",\" + localVertex.y + \",\" + localVertex.z];\n let areUvsDifferent = false;\n if (uvs && !(uvs[vertex_idx * 2] === uv.x || uvs[vertex_idx * 2 + 1] === uv.y)) {\n areUvsDifferent = true;\n }\n let areColorsDifferent = false;\n if (vertColors &&\n !(vertColors[vertex_idx * 4] === vertColor.r ||\n vertColors[vertex_idx * 4 + 1] === vertColor.g ||\n vertColors[vertex_idx * 4 + 2] === vertColor.b ||\n vertColors[vertex_idx * 4 + 3] === vertColor.a)) {\n areColorsDifferent = true;\n }\n // Check if 2 points can be merged\n if (!(typeof vertex_idx !== \"undefined\" &&\n normals[vertex_idx * 3] === localNormal.x &&\n normals[vertex_idx * 3 + 1] === localNormal.y &&\n normals[vertex_idx * 3 + 2] === localNormal.z) ||\n areUvsDifferent ||\n areColorsDifferent) {\n vertices.push(localVertex.x, localVertex.y, localVertex.z);\n if (uvs) {\n uvs.push(uv.x, uv.y);\n }\n normals.push(normal.x, normal.y, normal.z);\n if (vertColors) {\n vertColors.push(vertColor.r, vertColor.g, vertColor.b, vertColor.a);\n }\n vertex_idx = vertice_dict[localVertex.x + \",\" + localVertex.y + \",\" + localVertex.z] = vertices.length / 3 - 1;\n }\n indices.push(vertex_idx);\n subMeshObj.indexStart = Math.min(currentIndex, subMeshObj.indexStart);\n subMeshObj.indexEnd = Math.max(currentIndex, subMeshObj.indexEnd);\n currentIndex++;\n }\n }\n }\n mesh.setVerticesData(VertexBuffer.PositionKind, vertices);\n mesh.setVerticesData(VertexBuffer.NormalKind, normals);\n if (uvs) {\n mesh.setVerticesData(VertexBuffer.UVKind, uvs);\n }\n if (vertColors) {\n mesh.setVerticesData(VertexBuffer.ColorKind, vertColors);\n }\n mesh.setIndices(indices, null);\n if (keepSubMeshes) {\n // We offset the materialIndex by the previous number of materials in the CSG mixed meshes\n let materialIndexOffset = 0, materialMaxIndex;\n mesh.subMeshes = new Array();\n for (const m in subMeshDict) {\n materialMaxIndex = -1;\n for (const sm in subMeshDict[m]) {\n subMeshObj = subMeshDict[m][sm];\n SubMesh.CreateFromIndices(subMeshObj.materialIndex + materialIndexOffset, subMeshObj.indexStart, subMeshObj.indexEnd - subMeshObj.indexStart + 1, mesh);\n materialMaxIndex = Math.max(subMeshObj.materialIndex, materialMaxIndex);\n }\n materialIndexOffset += ++materialMaxIndex;\n }\n }\n return mesh;\n }\n /**\n * Build Mesh from CSG taking material and transforms into account\n * @param name The name of the Mesh\n * @param material The material of the Mesh\n * @param scene The Scene\n * @param keepSubMeshes Specifies if submeshes should be kept\n * @returns The new Mesh\n */\n toMesh(name, material = null, scene, keepSubMeshes) {\n const mesh = this.buildMeshGeometry(name, scene, keepSubMeshes);\n mesh.material = material;\n mesh.position.copyFrom(this.position);\n mesh.rotation.copyFrom(this.rotation);\n if (this.rotationQuaternion) {\n mesh.rotationQuaternion = this.rotationQuaternion.clone();\n }\n mesh.scaling.copyFrom(this.scaling);\n mesh.computeWorldMatrix(true);\n return mesh;\n }\n }\n\n function createJoinGeometryVisualization(targetMesh, scene, color) {\n const visualization = targetMesh;\n if (!color) {\n color = Color3.Yellow();\n }\n const coloredMaterial = new StandardMaterial('', scene);\n //coloredMaterial.alpha = 0.5;\n coloredMaterial.wireframe = true;\n coloredMaterial.emissiveColor = color;\n visualization.material = coloredMaterial;\n return visualization;\n }\n\n const joinedMeshes = new Map();\n class JoinGeometryRenderer extends FeatureRenderer {\n constructor() {\n super(...arguments);\n this.modelToken = Token.JoinGeometryFeature;\n this.bbSubscriptions = new Map();\n this.targetMeshVisualizationMap = new Map();\n this.parentMeshVisualizationMap = new Map();\n this.previewLayer = this.viewer.utilityLayer;\n }\n updateMeshGeometry(feature, vertexData, vertexDataChanges) {\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\n const targetMesh = joinedMeshes.get(feature._dynamicId);\n const thisMesh = this.viewer.getRendererNode(feature._parent);\n if (targetMesh instanceof Mesh &&\n thisMesh instanceof Mesh &&\n targetMesh.geometry &&\n targetMesh.geometry.getTotalIndices() > 0 &&\n targetMesh != thisMesh) {\n const workingVertexData = vertexData.shallowClone();\n const tempMesh = new Mesh('csg-temp');\n tempMesh.doNotSerialize = true;\n SubmeshVertexData.CopyToRef(newVertexData, workingVertexData);\n if (!workingVertexData.positions || !workingVertexData.indices) {\n throw 'Invalid geometry';\n }\n const targetTransform = thisMesh.computeWorldMatrix(true);\n workingVertexData.applyToMesh(tempMesh);\n let previewMesh = this.parentMeshVisualizationMap.get(feature.id);\n if (!previewMesh) {\n const newMeshInstance = new Mesh('preview-mesh');\n newMeshInstance.doNotSerialize = true;\n workingVertexData.applyToMesh(newMeshInstance);\n this.parentMeshVisualizationMap.set(feature.id, newMeshInstance);\n if (newMeshInstance instanceof Mesh) {\n previewMesh = createJoinGeometryVisualization(newMeshInstance, this.previewLayer.utilityLayerScene, Color3.Yellow());\n this.parentMeshVisualizationMap.set(feature.id, previewMesh);\n previewMesh.setEnabled(false);\n if (feature._parent) {\n const featureParentMesh = this.viewer.getRendererNode(feature._parent);\n newMeshInstance.position.addInPlace(featureParentMesh.absolutePosition);\n }\n }\n }\n ensureNormalsExist(tempMesh);\n const resultCsg = CSG.FromMesh(tempMesh);\n const targetMeshMatchTransform = targetMesh.clone();\n const targetMeshTransform = targetMeshMatchTransform.computeWorldMatrix(true);\n targetMeshMatchTransform.parent = null;\n targetTransform.invert();\n targetMeshTransform.multiplyToRef(targetTransform, targetMeshTransform);\n targetMeshMatchTransform.rotationQuaternion = new Quaternion();\n targetMeshTransform.decompose(targetMeshMatchTransform.scaling, targetMeshMatchTransform.rotationQuaternion, targetMeshMatchTransform.position);\n try {\n const opCsg = CSG.FromMesh(targetMeshMatchTransform);\n if (feature.joinType === exports.eJoinType.cut) {\n resultCsg.subtractInPlace(opCsg);\n }\n else if (feature.joinType === exports.eJoinType.intersect) {\n resultCsg.intersectInPlace(opCsg);\n }\n else if (feature.joinType === exports.eJoinType.union) {\n resultCsg.unionInPlace(opCsg);\n }\n }\n catch (e) {\n console.warn('Could not merge geometry for join ' + feature.name, e);\n return newVertexData;\n }\n if (resultCsg) {\n const resultMeshCsg = resultCsg.toMesh('temp2', null, undefined, true);\n const resultVertexData = SubmeshVertexData.ExtractFromMesh(resultMeshCsg);\n newVertexData.positions = resultVertexData.positions;\n if (newVertexData.normals) {\n newVertexData.normals = resultVertexData.normals;\n }\n if (newVertexData.uvs) {\n newVertexData.uvs = resultVertexData.uvs;\n }\n newVertexData.indices = resultVertexData.indices;\n // Sometimes the CSG will add another material to the multiMaterial to represent the cut face. Change that material to be the base material, otherwise it will be black.\n const existingMaterials = new Set();\n // get a list of all the materials that are already in the multiMaterial\n if (newVertexData.submeshData) {\n newVertexData.submeshData.forEach(submeshData => {\n existingMaterials.add(submeshData.materialIndex);\n });\n }\n if (resultVertexData.submeshData) {\n resultVertexData.submeshData.forEach(submeshData => {\n if (!existingMaterials.has(submeshData.materialIndex)) {\n submeshData.materialIndex = 0;\n }\n });\n }\n newVertexData.submeshData = resultVertexData.submeshData;\n resultMeshCsg.dispose();\n }\n tempMesh.dispose();\n targetMeshMatchTransform.dispose();\n }\n return newVertexData;\n }\n updatePreviewMesh(node) {\n let featureInstance = this.targetMeshVisualizationMap.get(node.id);\n const scene = this.previewLayer.utilityLayerScene;\n const targetMesh = joinedMeshes.get(node._dynamicId);\n if (targetMesh && !featureInstance) {\n featureInstance = createJoinGeometryVisualization(targetMesh.clone(), scene, Color3.Yellow());\n featureInstance.isPickable = false;\n featureInstance.setEnabled(false);\n this.targetMeshVisualizationMap.set(node.id, featureInstance);\n return featureInstance;\n }\n else if (targetMesh) {\n this.deleteTargetMeshVisualization(node);\n featureInstance = createJoinGeometryVisualization(targetMesh.clone(), scene, Color3.Yellow());\n featureInstance.isPickable = false;\n featureInstance.setEnabled(false);\n this.targetMeshVisualizationMap.set(node.id, featureInstance);\n return featureInstance;\n }\n }\n update3dInternal(feature) {\n var _a;\n if (feature.changes.hasAny('meshId', 'enabled')) {\n const existingSubscription = this.bbSubscriptions.get(feature._dynamicId);\n if (existingSubscription) {\n existingSubscription.unsubscribe();\n }\n if (feature.enabled) {\n const targetMeshNode = (_a = feature._manager) === null || _a === void 0 ? void 0 : _a.getById(feature.meshId);\n if (targetMeshNode && targetMeshNode instanceof exports.MeshNode) {\n // if this is a custom mesh node and the geometry isn't loaded, we have to load it\n if (targetMeshNode instanceof exports.CustomMeshNode &&\n targetMeshNode.hasGeometry &&\n !targetMeshNode.geometryLoaded) {\n this.viewer.rendererManager.mesh.loadNewGeometry(targetMeshNode);\n }\n const targetMesh = this.viewer.getRendererNode(targetMeshNode);\n if (targetMesh instanceof Mesh && feature._parent) {\n joinedMeshes.set(feature._dynamicId, targetMesh);\n this.bbSubscriptions.set(feature._dynamicId, \n // TODO This should also watch the parent group for positional changes, otherwise the join won't update when the feature's parent mesh moves.\n // But watch out for infinite loops\n this.viewer.bbCache.watchForChanges(targetMeshNode).subscribe(() => {\n feature.setPropertyIsModified('meshId', true);\n }));\n }\n else {\n console.warn('Could not create CSG for join geometry');\n }\n }\n }\n }\n let featureInstance = this.targetMeshVisualizationMap.get(feature.id);\n const originalMesh = this.parentMeshVisualizationMap.get(feature.id);\n if (feature._previewVisible) {\n featureInstance = this.updatePreviewMesh(feature);\n }\n if (feature._previewVisible && feature.enabled) {\n if (feature.joinType === exports.eJoinType.cut) {\n originalMesh === null || originalMesh === void 0 ? void 0 : originalMesh.setEnabled(false);\n featureInstance === null || featureInstance === void 0 ? void 0 : featureInstance.setEnabled(true);\n }\n else if (feature.joinType === exports.eJoinType.intersect) {\n originalMesh === null || originalMesh === void 0 ? void 0 : originalMesh.setEnabled(true);\n featureInstance === null || featureInstance === void 0 ? void 0 : featureInstance.setEnabled(true);\n }\n else {\n originalMesh === null || originalMesh === void 0 ? void 0 : originalMesh.setEnabled(false);\n featureInstance === null || featureInstance === void 0 ? void 0 : featureInstance.setEnabled(false);\n }\n }\n else {\n originalMesh === null || originalMesh === void 0 ? void 0 : originalMesh.setEnabled(false);\n featureInstance === null || featureInstance === void 0 ? void 0 : featureInstance.setEnabled(false);\n }\n }\n delete3d(feature, preserveCache) {\n var _a;\n if (!preserveCache) {\n joinedMeshes.delete(feature._dynamicId);\n }\n (_a = this.bbSubscriptions.get(feature._dynamicId)) === null || _a === void 0 ? void 0 : _a.unsubscribe();\n this.bbSubscriptions.delete(feature._dynamicId);\n this.deleteTargetMeshVisualization(feature);\n const targetMeshCopy = this.parentMeshVisualizationMap.get(feature.id);\n targetMeshCopy === null || targetMeshCopy === void 0 ? void 0 : targetMeshCopy.dispose(false, true);\n this.parentMeshVisualizationMap.delete(feature.id);\n super.delete3d(feature);\n }\n deleteTargetMeshVisualization(feature) {\n const previewMesh = this.targetMeshVisualizationMap.get(feature.id);\n previewMesh === null || previewMesh === void 0 ? void 0 : previewMesh.dispose(false, true);\n this.targetMeshVisualizationMap.delete(feature.id);\n }\n dispose() {\n this.bbSubscriptions.forEach(s => {\n s.unsubscribe();\n });\n this.bbSubscriptions.clear();\n joinedMeshes.clear();\n super.dispose();\n }\n }\n\n class LinearPatternRenderer extends PatternRenderer {\n constructor() {\n super(...arguments);\n this.modelToken = Token.LinearPatternFeature;\n }\n updateMeshGeometry(feature, vertexData, vertexDataChanges) {\n const featureTranslation = feature.translation.toVec3();\n const workingVector = Vector3.Zero();\n return super.updateMeshGeometry(feature, vertexData, vertexDataChanges, (idx, particle) => {\n if (!feature.meshId) {\n workingVector.copyFrom(featureTranslation).scaleInPlace(idx);\n }\n particle.position.addInPlace(workingVector);\n });\n }\n }\n\n function createMirrorVisualization(mirror, scene, color) {\n if (!color) {\n color = Color3.Yellow();\n }\n const mirrorVisualization = MeshBuilder.CreatePlane('mirrorPlane', {}, scene);\n const coloredMaterial = new StandardMaterial('', scene);\n coloredMaterial.wireframe = true;\n coloredMaterial.emissiveColor = color;\n mirrorVisualization.material = coloredMaterial;\n mirrorVisualization.isPickable = false;\n mirrorVisualization.setEnabled(true);\n return mirrorVisualization;\n }\n\n class SliceFeatureRenderer extends FeatureRenderer {\n constructor() {\n super(...arguments);\n this.modelToken = Token.SliceFeature;\n this.previewLayer = this.viewer.utilityLayer;\n this.babylonMirrorPreviewsMap = new Map();\n }\n updateMeshGeometry(feature, vertexData, vertexDataChanges) {\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\n if (newVertexData.positions &&\n newVertexData.positions.length > 0 &&\n newVertexData.indices &&\n newVertexData.indices.length > 0) {\n const positions = newVertexData.positions;\n const normals = newVertexData.normals;\n const uvs = newVertexData.uvs;\n const indices = newVertexData.indices;\n const triCount = indices.length / 3;\n const positionsCount = positions.length / 3;\n const planeNormal = feature.normal.toVec3();\n const planePoint = feature.intersect.toVec3();\n const newPositions = [];\n const newNormals = [];\n const newUvs = [];\n const newIndices = [];\n const preSliceSubmeshes = newVertexData.submeshData;\n const newSubmeshes = preSliceSubmeshes.map(submesh => ({ ...submesh }));\n const edgeSplits = []; // which vertices are part of how many edges being split\n const edgeStart = Vector3.Zero();\n const edgeDir = Vector3.Zero();\n const workingVector = Vector3.Zero();\n let j;\n for (let i = 0; i < triCount; i++) {\n let hit = 0;\n let d = 0;\n let vertexIndex = -1;\n for (j = 0; j < 3; j++) {\n const iA = j, iB = (j + 1) % 3;\n const vA = indices[i * 3 + iA], vB = indices[i * 3 + iB];\n if (vertexIndex === -1) {\n vertexIndex = vA;\n }\n edgeStart.set(positions[vA * 3], positions[vA * 3 + 1], positions[vA * 3 + 2]);\n edgeStart.subtractFromFloatsToRef(positions[vB * 3], positions[vB * 3 + 1], positions[vB * 3 + 2], edgeDir);\n edgeDir.scaleInPlace(-1);\n const lnDot = Vector3.Dot(planeNormal, edgeDir);\n if (lnDot === 0) {\n continue;\n }\n planePoint.subtractToRef(edgeStart, workingVector);\n d = Vector3.Dot(workingVector, planeNormal) / lnDot;\n // Expect floating point errors. don't want unnecessary tiny triangles\n // If d is between 0 and 1, then the plane hit the edge\n if (d > 1e-12 && d < 1 - 1e-12) {\n hit++;\n edgeSplits[iA] = (edgeSplits[iA] || 0) + 1;\n edgeSplits[iB] = (edgeSplits[iB] || 0) + 1;\n edgeDir.scaleInPlace(d);\n edgeStart.addToRef(edgeDir, workingVector);\n newPositions.push(workingVector.x, workingVector.y, workingVector.z);\n if (normals) {\n workingVector.set(normals[vA * 3], normals[vA * 3 + 1], normals[vA * 3 + 2]);\n workingVector.scaleInPlace(1 - d);\n workingVector.addInPlaceFromFloats(normals[vB * 3] * d, normals[vB * 3 + 1] * d, normals[vB * 3 + 2] * d);\n newNormals.push(workingVector.x, workingVector.y, workingVector.z);\n }\n if (uvs) {\n workingVector.set(uvs[vA * 2], uvs[vA * 2 + 1], 0);\n workingVector.scaleInPlace(1 - d);\n workingVector.addInPlaceFromFloats(uvs[vB * 2] * d, uvs[vB * 2 + 1] * d, 0);\n newUvs.push(workingVector.x, workingVector.y);\n }\n }\n }\n if (hit) {\n // divide the triangle here\n if (hit === 1) {\n const split = positionsCount - 1 + newPositions.length / 3;\n if (!edgeSplits[0]) {\n newIndices.push(indices[i * 3], indices[i * 3 + 1], split, indices[i * 3], split, indices[i * 3 + 2]);\n }\n else if (!edgeSplits[1]) {\n newIndices.push(indices[i * 3], indices[i * 3 + 1], split, indices[i * 3 + 1], indices[i * 3 + 2], split);\n }\n else {\n newIndices.push(indices[i * 3], split, indices[i * 3 + 2], indices[i * 3 + 1], indices[i * 3 + 2], split);\n }\n newSubmeshes.forEach((submeshData, sI) => {\n const preSliceSubmesh = preSliceSubmeshes[sI];\n if (preSliceSubmesh.indexStart <= i * 3 &&\n preSliceSubmesh.indexStart + preSliceSubmesh.indexCount > i * 3) {\n submeshData.indexCount += 3;\n }\n else if (preSliceSubmesh.indexStart > i * 3) {\n submeshData.indexStart += 3;\n }\n });\n }\n if (hit === 2) {\n const aSplit = positionsCount - 1 + newPositions.length / 3 - 1, bSplit = positionsCount - 1 + newPositions.length / 3;\n newIndices.push(indices[i * 3]);\n newIndices.push(aSplit);\n newIndices.push(bSplit);\n if (edgeSplits[0] === 2) {\n newIndices.push(indices[i * 3 + 1], bSplit, aSplit, indices[i * 3 + 2], bSplit, indices[i * 3 + 1]);\n }\n else if (edgeSplits[1] === 2) {\n newIndices.push(indices[i * 3 + 2], indices[i * 3], bSplit, indices[i * 3 + 1], bSplit, aSplit);\n }\n else {\n newIndices.push(indices[i * 3 + 1], aSplit, indices[i * 3], indices[i * 3 + 2], bSplit, aSplit);\n }\n newSubmeshes.forEach((submeshData, sI) => {\n const preSliceSubmesh = preSliceSubmeshes[sI];\n if (preSliceSubmesh.indexStart <= i * 3 &&\n preSliceSubmesh.indexStart + preSliceSubmesh.indexCount > i * 3) {\n submeshData.indexCount += 6;\n }\n else if (preSliceSubmesh.indexStart > i * 3) {\n submeshData.indexStart += 6;\n }\n });\n }\n edgeSplits.clear();\n hit = 0;\n }\n else {\n // if(feature.removeGeometry && d > 1 - 1e-12) {\n // // If this face is in front of the face relative to the face normal, cull it\n // newSubmeshes.forEach(submeshData => {\n // if(submeshData.indexStart <= i && submeshData.indexStart + submeshData.indexCount > i) {\n // submeshData.indexCount -= 3;\n // } else if(submeshData.indexStart > i) {\n // submeshData.indexStart -= 3;\n // }\n // });\n // continue;\n // } else {\n // console.log(d);\n // }\n newIndices.push(indices[i * 3], indices[i * 3 + 1], indices[i * 3 + 2]);\n }\n // Update the vertices the submeshes are targetting\n if (vertexIndex !== -1 && hit > 0) {\n newSubmeshes.forEach(submeshData => {\n if (submeshData.verticesStart <= vertexIndex &&\n submeshData.verticesStart + submeshData.verticesCount > vertexIndex) {\n submeshData.verticesCount += hit;\n }\n else if (submeshData.verticesStart > vertexIndex) {\n submeshData.verticesStart += hit;\n }\n });\n }\n }\n newVertexData.indices = new Int32Array(newIndices);\n newVertexData.positions = new Float32Array(positions.length + newPositions.length);\n newVertexData.positions.set(positions);\n newVertexData.positions.set(newPositions, positions.length);\n if (normals) {\n newVertexData.normals = new Float32Array(normals.length + newNormals.length);\n newVertexData.normals.set(normals);\n newVertexData.normals.set(newNormals, normals.length);\n }\n if (uvs) {\n newVertexData.uvs = new Float32Array(uvs.length + newUvs.length);\n newVertexData.uvs.set(uvs);\n newVertexData.uvs.set(newUvs, uvs.length);\n }\n // Remove geometry on one side of the plane if the removeGeometry flag is true\n const baseGeometryVertexLength = positions.length / 3;\n if (feature.removeGeometry) {\n const indices = newVertexData.indices;\n const positions = newVertexData.positions;\n const normals = newVertexData.normals;\n const uvs = newVertexData.uvs;\n const newIndices = new Int32Array(indices.length);\n const newPositions = new Float32Array(positions.length);\n const newNormals = new Float32Array((normals === null || normals === void 0 ? void 0 : normals.length) || 0);\n const newUvs = new Float32Array((uvs === null || uvs === void 0 ? void 0 : uvs.length) || 0);\n const triCount = indices.length / 3;\n const vertexCount = positions.length / 3;\n const newPositionIndex = new Int32Array(vertexCount);\n // Keep track of how many positions we've used\n let position = 0;\n for (let i = 0; i < vertexCount; i++) {\n workingVector.set(positions[i * 3], positions[i * 3 + 1], positions[i * 3 + 2]);\n workingVector.subtractInPlace(planePoint);\n const d = Vector3.Dot(workingVector, planeNormal);\n // Don't remove new vertices that were created from the slice, those are guaranteed to be good\n if (i < baseGeometryVertexLength && d > 1e-12) {\n // Mark the original indices that have been removed so we can update the index array\n newPositionIndex[i] = -1;\n }\n else {\n // Copy over the vertex data we need and mark what new index they will have\n newPositionIndex[i] = position;\n newPositions[position * 3] = positions[i * 3];\n newPositions[position * 3 + 1] = positions[i * 3 + 1];\n newPositions[position * 3 + 2] = positions[i * 3 + 2];\n if (normals) {\n newNormals[position * 3] = normals[i * 3];\n newNormals[position * 3 + 1] = normals[i * 3 + 1];\n newNormals[position * 3 + 2] = normals[i * 3 + 2];\n }\n if (uvs) {\n newUvs[position * 2] = uvs[i * 2];\n newUvs[position * 2 + 1] = uvs[i * 2 + 1];\n }\n position++;\n }\n }\n // Trim off unused parts of the vertex arrays\n newVertexData.positions = newPositions.subarray(0, position * 3);\n if (normals) {\n newVertexData.normals = newNormals.subarray(0, position * 3);\n }\n if (uvs) {\n newVertexData.uvs = newUvs.subarray(0, position * 2);\n }\n position = 0;\n const originalSubmeshes = newSubmeshes.map(submesh => ({ ...submesh }));\n for (let i = 0; i < triCount; i++) {\n // If all of the positions in this triangle are being kept, add the triangle\n // The indices of the vertexes are changing so we need to use the value that we calculated prior\n if (newPositionIndex[indices[i * 3]] >= 0 &&\n newPositionIndex[indices[i * 3 + 1]] >= 0 &&\n newPositionIndex[indices[i * 3 + 2]] >= 0) {\n newIndices[position * 3] = newPositionIndex[indices[i * 3]];\n newIndices[position * 3 + 1] = newPositionIndex[indices[i * 3 + 1]];\n newIndices[position * 3 + 2] = newPositionIndex[indices[i * 3 + 2]];\n position++;\n }\n else {\n newSubmeshes.forEach((submeshData, sI) => {\n const originalSubmesh = originalSubmeshes[sI];\n if (originalSubmesh.indexStart <= i * 3 &&\n originalSubmesh.indexStart + originalSubmesh.indexCount > i * 3) {\n submeshData.indexCount -= 3;\n }\n else if (originalSubmesh.indexStart > i * 3) {\n submeshData.indexStart -= 3;\n }\n });\n }\n }\n // Update the vertex values on the submeshes as well\n for (let i = 0; i < newPositionIndex.length; i++) {\n if (newPositionIndex[i] === -1) {\n newSubmeshes.forEach(submeshData => {\n if (submeshData.verticesStart <= i &&\n submeshData.verticesStart + submeshData.verticesCount > i) {\n submeshData.verticesCount--;\n }\n else if (submeshData.verticesStart > i) {\n submeshData.verticesStart--;\n }\n });\n }\n }\n newVertexData.indices = newIndices.subarray(0, position * 3);\n }\n newVertexData.submeshData = newSubmeshes;\n }\n return newVertexData;\n }\n updatePreviewMesh(feature) {\n const scene = this.previewLayer.utilityLayerScene;\n let visualizationPlane = this.babylonMirrorPreviewsMap.get(feature.id);\n if (!visualizationPlane) {\n visualizationPlane = createMirrorVisualization(feature, scene);\n visualizationPlane.isPickable = false;\n this.babylonMirrorPreviewsMap.set(feature.id, visualizationPlane);\n }\n const currentNormal = new Vector3();\n currentNormal.set(0, 0, 1);\n const newNormal = new Vector3();\n newNormal.set(feature.normal.x, feature.normal.y, feature.normal.z);\n visualizationPlane.rotationQuaternion = rotationToVector(newNormal, currentNormal);\n let parentBoundingBox = undefined;\n if (feature._parent) {\n parentBoundingBox = this.viewer.getMeshBoundingBox(feature._parent);\n }\n if (parentBoundingBox) {\n const scale = parentBoundingBox.minWorld.subtract(parentBoundingBox.maxWorld).length();\n visualizationPlane.scaling.set(scale, scale, scale);\n }\n if (feature._parent) {\n const featureParent = this.viewer.getRendererNode(feature._parent);\n const meshPosition = featureParent.getAbsolutePosition();\n meshPosition.addInPlace(feature.intersect.toVec3());\n visualizationPlane.position.set(meshPosition.x, meshPosition.y, meshPosition.z);\n }\n }\n update3dInternal(node) {\n const visualizationPlane = this.babylonMirrorPreviewsMap.get(node.id);\n if (node._previewVisible) {\n this.updatePreviewMesh(node);\n if (node.enabled) {\n visualizationPlane === null || visualizationPlane === void 0 ? void 0 : visualizationPlane.setEnabled(true);\n }\n }\n else {\n visualizationPlane === null || visualizationPlane === void 0 ? void 0 : visualizationPlane.setEnabled(false);\n }\n }\n delete3d(node) {\n if (this.babylonMirrorPreviewsMap) {\n const featureInstance = this.babylonMirrorPreviewsMap.get(node.id);\n if (featureInstance) {\n featureInstance.dispose(false, true);\n }\n this.babylonMirrorPreviewsMap.delete(node.id);\n }\n super.delete3d(node);\n }\n }\n\n class MirrorFeatureRenderer extends SliceFeatureRenderer {\n constructor() {\n super(...arguments);\n this.modelToken = Token.MirrorFeature;\n }\n updateMeshGeometry(feature, vertexData, vertexDataChanges) {\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\n if (newVertexData.positions &&\n newVertexData.positions.length > 0 &&\n newVertexData.indices &&\n newVertexData.indices.length > 0) {\n const positions = newVertexData.positions;\n const normals = newVertexData.normals;\n const uvs = newVertexData.uvs;\n const indices = newVertexData.indices;\n const newIndices = new Int32Array(indices.length);\n const newPositions = new Float32Array(positions.length);\n const newNormals = new Float32Array((normals === null || normals === void 0 ? void 0 : normals.length) || 0);\n const workingVector = Vector3.Zero();\n const planeNormal = feature.normal.toVec3().normalize();\n const planeIntersect = feature.intersect.toVec3();\n const vertexCount = positions.length / 3;\n for (let i = 0; i < vertexCount; i++) {\n const idx = i * 3;\n workingVector.set(positions[idx], positions[idx + 1], positions[idx + 2]);\n workingVector.subtractInPlace(planeIntersect);\n let d = Vector3.Dot(workingVector, planeNormal);\n planeNormal.scaleAndAddToRef(-2 * d, workingVector);\n newPositions[idx] = workingVector.x + planeIntersect.x;\n newPositions[idx + 1] = workingVector.y + planeIntersect.y;\n newPositions[idx + 2] = workingVector.z + planeIntersect.z;\n if (normals) {\n workingVector.set(normals[idx], normals[idx + 1], normals[idx + 2]);\n d = Vector3.Dot(workingVector, planeNormal);\n planeNormal.scaleAndAddToRef(-2 * d, workingVector);\n newNormals[idx] = workingVector.x;\n newNormals[idx + 1] = workingVector.y;\n newNormals[idx + 2] = workingVector.z;\n }\n }\n // Flip the direction of the triangles so that the normals face the right way\n const triCount = indices.length / 3;\n for (let i = 0; i < triCount; i++) {\n const idx = i * 3;\n if (feature.cloneGeometry) {\n newIndices[idx] = vertexCount + indices[idx + 2];\n newIndices[idx + 1] = vertexCount + indices[idx + 1];\n newIndices[idx + 2] = vertexCount + indices[idx];\n }\n else {\n newIndices[idx] = indices[idx + 2];\n newIndices[idx + 1] = indices[idx + 1];\n newIndices[idx + 2] = indices[idx];\n }\n }\n if (feature.cloneGeometry) {\n const combinedIndices = new Int32Array(indices.length * 2);\n const combinedPositions = new Float32Array(positions.length * 2);\n const combinedNormals = new Float32Array(((normals === null || normals === void 0 ? void 0 : normals.length) || 0) * 2);\n const combinedUvs = new Float32Array(((uvs === null || uvs === void 0 ? void 0 : uvs.length) || 0) * 2);\n const combinedSubmeshes = [];\n combinedIndices.set(indices);\n combinedIndices.set(newIndices, indices.length);\n combinedPositions.set(positions);\n combinedPositions.set(newPositions, positions.length);\n if (normals) {\n combinedNormals.set(normals);\n combinedNormals.set(newNormals, normals.length);\n }\n if (uvs) {\n combinedUvs.set(uvs);\n combinedUvs.set(uvs, uvs.length);\n }\n if (newVertexData.submeshData) {\n newVertexData.submeshData.forEach(submesh => {\n const newSubmesh = {\n ...submesh,\n verticesStart: submesh.verticesStart + vertexCount,\n indexStart: submesh.indexStart + indices.length,\n };\n combinedSubmeshes.push(submesh, newSubmesh);\n });\n }\n newVertexData.indices = combinedIndices;\n newVertexData.positions = combinedPositions;\n if (normals) {\n newVertexData.normals = combinedNormals;\n }\n if (uvs) {\n newVertexData.uvs = combinedUvs;\n }\n newVertexData.submeshData = combinedSubmeshes;\n }\n else {\n newVertexData.indices = newIndices;\n newVertexData.positions = newPositions;\n if (normals) {\n newVertexData.normals = newNormals;\n }\n }\n }\n return newVertexData;\n }\n }\n\n /*! *****************************************************************************\n Copyright (c) Microsoft Corporation.\n\n Permission to use, copy, modify, and/or distribute this software for any\n purpose with or without fee is hereby granted.\n\n THE SOFTWARE IS PROVIDED \"AS IS\" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH\n REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY\n AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,\n INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM\n LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR\n OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR\n PERFORMANCE OF THIS SOFTWARE.\n ***************************************************************************** */\n var t=function(r,e){return (t=Object.setPrototypeOf||{__proto__:[]}instanceof Array&&function(t,r){t.__proto__=r;}||function(t,r){for(var e in r)Object.prototype.hasOwnProperty.call(r,e)&&(t[e]=r[e]);})(r,e)};function r(r,e){if(\"function\"!=typeof e&&null!==e)throw new TypeError(\"Class extends value \"+String(e)+\" is not a constructor or null\");function i(){this.constructor=r;}t(r,e),r.prototype=null===e?Object.create(e):(i.prototype=e.prototype,new i);}function e(t){var r=\"\";Array.isArray(t)||(t=[t]);for(var e=0;et.phi1&&(t.phi2-=2*n),1===t.sweepFlag&&t.phi2i)return [];if(0===i)return [[t*e/(t*t+r*r),r*e/(t*t+r*r)]];var n=Math.sqrt(i);return [[(t*e+r*n)/(t*t+r*r),(r*e-t*n)/(t*t+r*r)],[(t*e-r*n)/(t*t+r*r),(r*e+t*n)/(t*t+r*r)]]}var u,h=Math.PI/180;function c(t,r,e){return (1-e)*t+e*r}function y(t,r,e,i){return t+Math.cos(i/180*n)*r+Math.sin(i/180*n)*e}function p(t,r,e,i){var a=1e-6,n=r-t,o=e-r,s=3*n+3*(i-e)-6*o,u=6*(o-n),h=3*n;return Math.abs(s)T&&(a.sweepFlag=+!a.sweepFlag),a}))}function l(){return function(t){var r={};for(var e in t)r[e]=t[e];return r}}t.ROUND=function(t){function r(r){return Math.round(r*t)/t}return void 0===t&&(t=1e13),a(t),function(t){return void 0!==t.x1&&(t.x1=r(t.x1)),void 0!==t.y1&&(t.y1=r(t.y1)),void 0!==t.x2&&(t.x2=r(t.x2)),void 0!==t.y2&&(t.y2=r(t.y2)),void 0!==t.x&&(t.x=r(t.x)),void 0!==t.y&&(t.y=r(t.y)),void 0!==t.rX&&(t.rX=r(t.rX)),void 0!==t.rY&&(t.rY=r(t.rY)),t}},t.TO_ABS=r,t.TO_REL=function(){return u((function(t,r,e){return t.relative||(void 0!==t.x1&&(t.x1-=r),void 0!==t.y1&&(t.y1-=e),void 0!==t.x2&&(t.x2-=r),void 0!==t.y2&&(t.y2-=e),void 0!==t.x&&(t.x-=r),void 0!==t.y&&(t.y-=e),t.relative=!0),t}))},t.NORMALIZE_HVZ=function(t,r,e){return void 0===t&&(t=!0),void 0===r&&(r=!0),void 0===e&&(e=!0),u((function(i,a,n,o,s){if(isNaN(o)&&!(i.type&_.MOVE_TO))throw new Error(\"path must start with moveto\");return r&&i.type&_.HORIZ_LINE_TO&&(i.type=_.LINE_TO,i.y=i.relative?0:n),e&&i.type&_.VERT_LINE_TO&&(i.type=_.LINE_TO,i.x=i.relative?0:a),t&&i.type&_.CLOSE_PATH&&(i.type=_.LINE_TO,i.x=i.relative?o-a:o,i.y=i.relative?s-n:s),i.type&_.ARC&&(0===i.rX||0===i.rY)&&(i.type=_.LINE_TO,delete i.rX,delete i.rY,delete i.xRot,delete i.lArcFlag,delete i.sweepFlag),i}))},t.NORMALIZE_ST=e,t.QT_TO_C=n,t.INFO=u,t.SANITIZE=function(t){void 0===t&&(t=0),a(t);var r=NaN,e=NaN,i=NaN,n=NaN;return u((function(a,o,s,u,h){var c=Math.abs,y=!1,p=0,m=0;if(a.type&_.SMOOTH_CURVE_TO&&(p=isNaN(r)?0:o-r,m=isNaN(e)?0:s-e),a.type&(_.CURVE_TO|_.SMOOTH_CURVE_TO)?(r=a.relative?o+a.x2:a.x2,e=a.relative?s+a.y2:a.y2):(r=NaN,e=NaN),a.type&_.SMOOTH_QUAD_TO?(i=isNaN(i)?o:2*o-i,n=isNaN(n)?s:2*s-n):a.type&_.QUAD_TO?(i=a.relative?o+a.x1:a.x1,n=a.relative?s+a.y1:a.y2):(i=NaN,n=NaN),a.type&_.LINE_COMMANDS||a.type&_.ARC&&(0===a.rX||0===a.rY||!a.lArcFlag)||a.type&_.CURVE_TO||a.type&_.SMOOTH_CURVE_TO||a.type&_.QUAD_TO||a.type&_.SMOOTH_QUAD_TO){var O=void 0===a.x?0:a.relative?a.x:a.x-o,l=void 0===a.y?0:a.relative?a.y:a.y-s;p=isNaN(i)?void 0===a.x1?p:a.relative?a.x:a.x1-o:i-o,m=isNaN(n)?void 0===a.y1?m:a.relative?a.y:a.y1-s:n-s;var T=void 0===a.x2?0:a.relative?a.x:a.x2-o,v=void 0===a.y2?0:a.relative?a.y:a.y2-s;c(O)<=t&&c(l)<=t&&c(p)<=t&&c(m)<=t&&c(T)<=t&&c(v)<=t&&(y=!0);}return a.type&_.CLOSE_PATH&&c(o-u)<=t&&c(s-h)<=t&&(y=!0),y?[]:a}))},t.MATRIX=O,t.ROTATE=function(t,r,e){void 0===r&&(r=0),void 0===e&&(e=0),a(t,r,e);var i=Math.sin(t),n=Math.cos(t);return O(n,i,-i,n,r-r*n+e*i,e-r*i-e*n)},t.TRANSLATE=function(t,r){return void 0===r&&(r=0),a(t,r),O(1,0,0,1,t,r)},t.SCALE=function(t,r){return void 0===r&&(r=t),a(t,r),O(t,0,0,r,0,0)},t.SKEW_X=function(t){return a(t),O(1,0,Math.atan(t),1,0,0)},t.SKEW_Y=function(t){return a(t),O(1,Math.atan(t),0,1,0,0)},t.X_AXIS_SYMMETRY=function(t){return void 0===t&&(t=0),a(t),O(-1,0,0,1,t,0)},t.Y_AXIS_SYMMETRY=function(t){return void 0===t&&(t=0),a(t),O(1,0,0,-1,0,t)},t.A_TO_C=function(){return u((function(t,r,e){return _.ARC===t.type?function(t,r,e){var a,n,s,u;t.cX||o(t,r,e);for(var y=Math.min(t.phi1,t.phi2),p=Math.max(t.phi1,t.phi2)-y,m=Math.ceil(p/90),O=new Array(m),l=r,T=e,v=0;vc.maxX&&(c.maxX=t),tc.maxY&&(c.maxY=t),tw&&O(m(e,u.x1,u.x2,u.x,w));}for(var f=0,N=p(n,u.y1,u.y2,u.y);fw&&l(m(n,u.y1,u.y2,u.y,w));}}if(u.type&_.ARC){O(u.x),l(u.y),o(u,e,n);for(var x=u.xRot/180*Math.PI,d=Math.cos(x)*u.rX,E=Math.sin(x)*u.rX,A=-Math.sin(x)*u.rY,C=Math.cos(x)*u.rY,M=u.phi1u.phi2?[u.phi2+360,u.phi1+360]:[u.phi2,u.phi1],R=M[0],g=M[1],I=function(t){var r=t[0],e=t[1],i=180*Math.atan2(e,r)/Math.PI;return iR&&wR&&wu)throw new SyntaxError('Expected positive number, got \"'+u+'\" at index \"'+a+'\"')}else if((3===this.curArgs.length||4===this.curArgs.length)&&\"0\"!==this.curNumber&&\"1\"!==this.curNumber)throw new SyntaxError('Expected a flag, got \"'+this.curNumber+'\" at index \"'+a+'\"');this.curArgs.push(u),this.curArgs.length===N[this.curCommandType]&&(_.HORIZ_LINE_TO===this.curCommandType?i({type:_.HORIZ_LINE_TO,relative:this.curCommandRelative,x:u}):_.VERT_LINE_TO===this.curCommandType?i({type:_.VERT_LINE_TO,relative:this.curCommandRelative,y:u}):this.curCommandType===_.MOVE_TO||this.curCommandType===_.LINE_TO||this.curCommandType===_.SMOOTH_QUAD_TO?(i({type:this.curCommandType,relative:this.curCommandRelative,x:this.curArgs[0],y:this.curArgs[1]}),_.MOVE_TO===this.curCommandType&&(this.curCommandType=_.LINE_TO)):this.curCommandType===_.CURVE_TO?i({type:_.CURVE_TO,relative:this.curCommandRelative,x1:this.curArgs[0],y1:this.curArgs[1],x2:this.curArgs[2],y2:this.curArgs[3],x:this.curArgs[4],y:this.curArgs[5]}):this.curCommandType===_.SMOOTH_CURVE_TO?i({type:_.SMOOTH_CURVE_TO,relative:this.curCommandRelative,x2:this.curArgs[0],y2:this.curArgs[1],x:this.curArgs[2],y:this.curArgs[3]}):this.curCommandType===_.QUAD_TO?i({type:_.QUAD_TO,relative:this.curCommandRelative,x1:this.curArgs[0],y1:this.curArgs[1],x:this.curArgs[2],y:this.curArgs[3]}):this.curCommandType===_.ARC&&i({type:_.ARC,relative:this.curCommandRelative,rX:this.curArgs[0],rY:this.curArgs[1],xRot:this.curArgs[2],lArcFlag:this.curArgs[3],sweepFlag:this.curArgs[4],x:this.curArgs[5],y:this.curArgs[6]})),this.curNumber=\"\",this.curNumberHasExpDigits=!1,this.curNumberHasExp=!1,this.curNumberHasDecimal=!1,this.canParseCommandOrComma=!0;}if(!T(n))if(\",\"===n&&this.canParseCommandOrComma)this.canParseCommandOrComma=!1;else if(\"+\"!==n&&\"-\"!==n&&\".\"!==n)if(s)this.curNumber=n,this.curNumberHasDecimal=!1;else {if(0!==this.curArgs.length)throw new SyntaxError(\"Unterminated command at index \"+a+\".\");if(!this.canParseCommandOrComma)throw new SyntaxError('Unexpected character \"'+n+'\" at index '+a+\". Command cannot follow comma\");if(this.canParseCommandOrComma=!1,\"z\"!==n&&\"Z\"!==n)if(\"h\"===n||\"H\"===n)this.curCommandType=_.HORIZ_LINE_TO,this.curCommandRelative=\"h\"===n;else if(\"v\"===n||\"V\"===n)this.curCommandType=_.VERT_LINE_TO,this.curCommandRelative=\"v\"===n;else if(\"m\"===n||\"M\"===n)this.curCommandType=_.MOVE_TO,this.curCommandRelative=\"m\"===n;else if(\"l\"===n||\"L\"===n)this.curCommandType=_.LINE_TO,this.curCommandRelative=\"l\"===n;else if(\"c\"===n||\"C\"===n)this.curCommandType=_.CURVE_TO,this.curCommandRelative=\"c\"===n;else if(\"s\"===n||\"S\"===n)this.curCommandType=_.SMOOTH_CURVE_TO,this.curCommandRelative=\"s\"===n;else if(\"q\"===n||\"Q\"===n)this.curCommandType=_.QUAD_TO,this.curCommandRelative=\"q\"===n;else if(\"t\"===n||\"T\"===n)this.curCommandType=_.SMOOTH_QUAD_TO,this.curCommandRelative=\"t\"===n;else {if(\"a\"!==n&&\"A\"!==n)throw new SyntaxError('Unexpected character \"'+n+'\" at index '+a+\".\");this.curCommandType=_.ARC,this.curCommandRelative=\"a\"===n;}else r.push({type:_.CLOSE_PATH}),this.canParseCommandOrComma=!0,this.curCommandType=-1;}else this.curNumber=n,this.curNumberHasDecimal=\".\"===n;}else this.curNumber+=n,this.curNumberHasDecimal=!0;else this.curNumber+=n;else this.curNumber+=n,this.curNumberHasExp=!0;else this.curNumber+=n,this.curNumberHasExpDigits=this.curNumberHasExp;}return r},e.prototype.transform=function(t){return Object.create(this,{parse:{value:function(r,e){void 0===e&&(e=[]);for(var i=0,a=Object.getPrototypeOf(this).parse.call(this,r);i= commands.length - 1;\n }\n next() {\n const command = this.commands[++this.i];\n this.previousCommand = this.command;\n this.command = command;\n return command;\n }\n getPoint(xProp = 'x', yProp = 'y') {\n if (this.command) {\n const x = parseInt(xProp);\n const y = parseInt(yProp);\n const point = new Vector2(x, y);\n return this.makeAbsolute(point);\n }\n }\n getAsControlPoint(xProp, yProp) {\n const point = this.getPoint(xProp, yProp);\n if (point) {\n this.control = point;\n }\n return point;\n }\n getAsCurrentPoint(xProp, yProp) {\n const point = this.getPoint(xProp, yProp);\n if (point) {\n this.current = point;\n }\n return point;\n }\n getReflectedControlPoint() {\n let previousCommand = null;\n if (this.previousCommand) {\n previousCommand = this.previousCommand.type;\n }\n if (previousCommand !== _.CURVE_TO &&\n previousCommand !== _.SMOOTH_CURVE_TO &&\n previousCommand !== _.QUAD_TO &&\n previousCommand !== _.SMOOTH_QUAD_TO) {\n return this.current;\n }\n // reflect point\n const { current: { x: cx, y: cy }, control: { x: ox, y: oy }, } = this;\n const point = new Vector2(2 * cx - ox, 2 * cy - oy);\n return point;\n }\n makeAbsolute(point) {\n if (this.command && 'relative' in this.command && this.command.relative) {\n const { x, y } = this.current;\n point.x += x;\n point.y += y;\n }\n return point;\n }\n addMarker(point, from, priorTo) {\n const { points, angles } = this;\n // if the last angle isn't filled in because we didn't have this point yet ...\n if (priorTo && angles.length > 0 && !angles[angles.length - 1]) {\n angles[angles.length - 1] = this.angleBetweenPoints(points[points.length - 1], priorTo);\n }\n this.addMarkerAngle(point, from ? this.angleBetweenPoints(from, point) : null);\n }\n addMarkerAngle(point, angle) {\n this.points.push(point);\n this.angles.push(angle);\n }\n angleBetweenPoints(point1, point2) {\n return Math.atan2(point2.y - point1.y, point2.x - point1.x);\n }\n getMarkerPoints() {\n return this.points;\n }\n getMarkerAngles() {\n const { angles } = this;\n const len = angles.length;\n for (let i = 0; i < len; i++) {\n if (!angles[i]) {\n for (let j = i + 1; j < len; j++) {\n if (angles[j]) {\n angles[i] = angles[j];\n break;\n }\n }\n }\n }\n return angles;\n }\n }\n\n class PathHandler {\n get currentPosition() {\n return this._currentPosition || this.initialPosition;\n }\n constructor(sketchNode, scaleDivider, nodeManager) {\n this.sketchNode = sketchNode;\n this.scaleDivider = scaleDivider;\n this.initialPosition = new exports.KbVector(0, 0, 0);\n this.stubPath = { id: 'stub' };\n this.nodeManager = nodeManager;\n }\n parseTrace(commands) {\n const newPaths = [];\n if (!this.sketchNode || commands.length === 0) {\n return [];\n }\n const first = commands[0];\n if (first.type !== 'M') {\n this.pathM(newPaths, {\n type: PathParser.MOVE_TO,\n relative: false,\n x: first.x1,\n y: first.y1,\n });\n }\n for (let i = 0; i < commands.length; i++) {\n const command = commands[i];\n const commonPathCommand = {\n relative: false,\n x: command.x2,\n y: command.y2,\n x1: command.x3 || -1,\n y1: command.y3 || -1,\n };\n switch (command.type) {\n case 'M': {\n const pathCommand = {\n ...commonPathCommand,\n type: PathParser.MOVE_TO,\n };\n this.pathM(newPaths, pathCommand);\n break;\n }\n case 'L': {\n const pathCommand = {\n ...commonPathCommand,\n type: PathParser.LINE_TO,\n };\n this.pathL(newPaths, pathCommand);\n break;\n }\n case 'Q': {\n if (!command.x3 || !command.y3) {\n console.warn('Q command without x3 or y3');\n }\n else {\n const pathCommand = {\n ...commonPathCommand,\n type: PathParser.QUAD_TO,\n x: command.x3,\n y: command.y3,\n x1: command.x2,\n y1: command.y2,\n };\n this.pathQ(newPaths, pathCommand);\n }\n break;\n }\n /** imagetracer does not return C, S, or T paths\n *\n case 'C': {\n const pathCommand: CommandC = {\n ...commonPathCommand,\n type: PathParser.CURVE_TO,\n };\n this.pathC(newPaths, pathCommand);\n break;\n }\n case 'S': {\n const pathCommand: CommandS = {\n ...commonPathCommand,\n type: PathParser.SMOOTH_CURVE_TO,\n };\n this.pathS(newPaths, pathCommand);\n break;\n }\n case 'T': {\n const pathCommand: CommandT = {\n ...commonPathCommand,\n type: PathParser.SMOOTH_QUAD_TO,\n };\n this.pathT(newPaths, pathCommand);\n break;\n }\n */\n case 'Z': {\n this.pathZ(newPaths);\n }\n }\n }\n return this.reprojectPaths(newPaths.filter(path => path !== this.stubPath));\n }\n parsePath(path) {\n this.pathParser = new PathParser(path);\n this.pathParser.reset();\n if (!this.sketchNode) {\n return [];\n }\n const newPaths = [];\n while (!this.pathParser.isEnd()) {\n switch (this.pathParser.next().type) {\n case PathParser.MOVE_TO:\n this.pathM(newPaths);\n break;\n case PathParser.LINE_TO:\n this.pathL(newPaths);\n break;\n case PathParser.HORIZ_LINE_TO:\n this.pathL(newPaths);\n break;\n case PathParser.VERT_LINE_TO:\n this.pathL(newPaths);\n break;\n case PathParser.CURVE_TO:\n this.pathC(newPaths);\n break;\n case PathParser.SMOOTH_CURVE_TO:\n this.pathS(newPaths);\n break;\n case PathParser.QUAD_TO:\n this.pathQ(newPaths);\n break;\n case PathParser.SMOOTH_QUAD_TO:\n this.pathT(newPaths);\n break;\n case PathParser.ARC:\n console.warn('ARC not implemented');\n this.pathA(newPaths);\n break;\n case PathParser.CLOSE_PATH:\n this.pathZ(newPaths);\n break;\n }\n }\n return this.reprojectPaths(newPaths.filter(path => path !== this.stubPath));\n }\n reprojectPaths(paths) {\n paths.forEach(path => {\n path.controlPoints.forEach(cp => {\n cp.position = projectOnNodeAxis(this.sketchNode, cp.position.x, cp.position.y);\n cp.direction = projectOnNodeAxis(this.sketchNode, cp.direction.x, cp.direction.y);\n if (cp.direction2) {\n cp.direction2 = projectOnNodeAxis(this.sketchNode, cp.direction2.x, cp.direction2.y);\n }\n });\n });\n return paths;\n }\n checkIfNewPathNeeded(paths, desiredPathType) {\n let activePath = paths[paths.length - 1];\n if (!activePath || activePath === this.stubPath || activePath.type !== desiredPathType) {\n if (activePath === this.stubPath) {\n paths.pop();\n }\n activePath = this.addNewPath(paths, desiredPathType);\n const newControlPoint = this.nodeManager.create(Token.SketchControlPoint);\n if (!this._currentPosition) {\n console.warn('new path created with no moveto command');\n }\n newControlPoint.position = this.currentPosition;\n activePath.controlPoints.push(newControlPoint);\n newControlPoint.direction2 = new exports.KbVector(0, 0, 0);\n }\n return activePath;\n }\n pathM(paths, command) {\n const { pathParser } = this;\n let x = 0;\n let y = 0;\n if (!command && pathParser.command) {\n command = pathParser.command;\n }\n if (command) {\n x = command.x / this.scaleDivider;\n y = (command.y * -1) / this.scaleDivider;\n if (command.relative && this.currentPosition) {\n x = x + this.currentPosition.x;\n y = y + this.currentPosition.y;\n }\n this._currentPosition = new exports.KbVector(x, y, 0);\n this.initialPosition = this._currentPosition;\n paths.push(this.stubPath);\n }\n }\n pathL(paths, command) {\n const { pathParser } = this;\n if (!command && pathParser.command) {\n command = pathParser.command;\n }\n if (command) {\n let x;\n let y;\n //these 2 if checks need to happen because pathL could be a horizontal or vertical line and therefore not have one of the coordinates.\n if ('x' in command) {\n x = command.x / this.scaleDivider;\n }\n if ('y' in command) {\n y = (command.y * -1) / this.scaleDivider;\n }\n const activePath = this.checkIfNewPathNeeded(paths, exports.eSketchPathType.points);\n if (command.relative && activePath) {\n if (x !== undefined) {\n x = x + this.currentPosition.x;\n }\n if (y !== undefined) {\n y = y + this.currentPosition.y;\n }\n }\n if (x === undefined) {\n // V case\n x = this.currentPosition.x || 0;\n }\n if (y === undefined) {\n // H case\n y = this.currentPosition.y || 0;\n }\n const newControlPoint = this.nodeManager.create(Token.SketchControlPoint);\n newControlPoint.position = new exports.KbVector(x, y, 0);\n this._currentPosition = newControlPoint.position;\n activePath.controlPoints.push(newControlPoint);\n }\n }\n pathC(paths, command) {\n const { pathParser } = this;\n let x = 0;\n let y = 0;\n let x1 = 0;\n let y1 = 0;\n let x2 = 0;\n let y2 = 0;\n let lastControlPoint = undefined;\n if (!command && pathParser.command) {\n command = pathParser.command;\n }\n if (command) {\n x = command.x / this.scaleDivider;\n y = (command.y * -1) / this.scaleDivider;\n x1 = command.x1 / this.scaleDivider;\n y1 = (command.y1 * -1) / this.scaleDivider;\n x2 = command.x2 / this.scaleDivider;\n y2 = (command.y2 * -1) / this.scaleDivider;\n const activePath = this.checkIfNewPathNeeded(paths, exports.eSketchPathType.cubicBezier);\n lastControlPoint = activePath.controlPoints[activePath.controlPoints.length - 1];\n if (command.relative) {\n x = x + this.currentPosition.x;\n y = y + this.currentPosition.y;\n x1 = x1 + this.currentPosition.x;\n y1 = y1 + this.currentPosition.y;\n x2 = x2 + this.currentPosition.x;\n y2 = y2 + this.currentPosition.y;\n }\n if (lastControlPoint) {\n lastControlPoint.direction = new exports.KbVector(x1 - lastControlPoint.position.x, y1 - lastControlPoint.position.y, 0);\n }\n const newControlPoint = this.nodeManager.create(Token.SketchControlPoint);\n newControlPoint.position = new exports.KbVector(x, y, 0);\n this._currentPosition = newControlPoint.position;\n newControlPoint.direction2 = new exports.KbVector(x2 - x, y2 - y, 0);\n activePath.controlPoints.push(newControlPoint);\n }\n }\n pathS(paths, command) {\n var _a;\n const { pathParser } = this;\n let x = 0;\n let y = 0;\n let x2 = 0;\n let y2 = 0;\n let lastControlPoint = undefined;\n if (!command && pathParser.command) {\n command = pathParser.command;\n }\n if (command) {\n x = command.x / this.scaleDivider;\n y = (command.y * -1) / this.scaleDivider;\n x2 = command.x2 / this.scaleDivider;\n y2 = (command.y2 * -1) / this.scaleDivider;\n const activePath = this.checkIfNewPathNeeded(paths, exports.eSketchPathType.cubicBezier);\n lastControlPoint = activePath.controlPoints[activePath.controlPoints.length - 1];\n lastControlPoint.direction = new exports.KbVector(0, 0, 0);\n if (command.relative) {\n x = x + this.currentPosition.x;\n y = y + this.currentPosition.y;\n x2 = x2 + this.currentPosition.x;\n y2 = y2 + this.currentPosition.y;\n }\n const newControlPoint = this.nodeManager.create(Token.SketchControlPoint);\n newControlPoint.position = new exports.KbVector(x, y, 0);\n if (lastControlPoint && lastControlPoint.direction2) {\n lastControlPoint.direction = (_a = lastControlPoint.direction2) === null || _a === void 0 ? void 0 : _a.scale(-1);\n newControlPoint.direction2 = new exports.KbVector(x2 - x, y2 - y, 0);\n }\n this._currentPosition = newControlPoint.position;\n activePath.controlPoints.push(newControlPoint);\n return activePath;\n }\n }\n pathQ(paths, command) {\n const { pathParser } = this;\n let x = 0;\n let y = 0;\n let x1 = 0;\n let y1 = 0;\n let lastControlPoint = undefined;\n if (!command && pathParser.command) {\n command = pathParser.command;\n }\n if (command) {\n x = command.x / this.scaleDivider;\n y = (command.y * -1) / this.scaleDivider;\n x1 = command.x1 / this.scaleDivider;\n y1 = (command.y1 * -1) / this.scaleDivider;\n const activePath = this.checkIfNewPathNeeded(paths, exports.eSketchPathType.quadraticBezier);\n lastControlPoint = activePath.controlPoints[activePath.controlPoints.length - 1];\n if (command.relative) {\n x = x + this.currentPosition.x;\n y = y + this.currentPosition.y;\n x1 = x1 + this.currentPosition.x;\n y1 = y1 + this.currentPosition.y;\n }\n if (lastControlPoint) {\n lastControlPoint.direction = new exports.KbVector(x1 - lastControlPoint.position.x, y1 - lastControlPoint.position.y, 0);\n }\n const newControlPoint = this.nodeManager.create(Token.SketchControlPoint);\n newControlPoint.position = new exports.KbVector(x, y, 0);\n this._currentPosition = newControlPoint.position;\n activePath.controlPoints.push(newControlPoint);\n }\n }\n pathT(paths, command) {\n const { pathParser } = this;\n let x = 0;\n let y = 0;\n let lastControlPoint = undefined;\n if (!command && pathParser.command) {\n command = pathParser.command;\n }\n if (command) {\n x = command.x / this.scaleDivider;\n y = (command.y * -1) / this.scaleDivider;\n const activePath = this.checkIfNewPathNeeded(paths, exports.eSketchPathType.quadraticBezier);\n lastControlPoint = activePath.controlPoints[activePath.controlPoints.length - 1];\n lastControlPoint.direction = new exports.KbVector(0, 0, 0);\n if (command.relative) {\n x = x + this.currentPosition.x;\n y = y + this.currentPosition.y;\n }\n const secondLastControlPoint = activePath.controlPoints[activePath.controlPoints.length - 2];\n const newControlPoint = this.nodeManager.create(Token.SketchControlPoint);\n newControlPoint.position = new exports.KbVector(x, y, 0);\n this._currentPosition = newControlPoint.position;\n activePath.controlPoints.push(newControlPoint);\n if (lastControlPoint && secondLastControlPoint && secondLastControlPoint) {\n lastControlPoint.direction = this.calculateReflection(secondLastControlPoint.direction, secondLastControlPoint, lastControlPoint);\n }\n }\n }\n //Path A needs further implementing\n pathA(paths, command) {\n const { pathParser } = this;\n if (!command && pathParser.command) {\n command = pathParser.command;\n }\n if (command) {\n // const cX = command.cX;\n // const cY = command.cY;\n // const rX = command.rX;\n // const rY = command.rY;\n let x = command.x;\n let y = command.y;\n if (command.relative) {\n x = x + this.currentPosition.x;\n y = y + this.currentPosition.y;\n }\n // const xAxisRotation = command.xRot;\n // const largeArcFlag = command.lArcFlag;\n // const sweepFlag = command.sweepFlag;\n this._currentPosition = new exports.KbVector(x, y, 0);\n }\n }\n pathZ(paths) {\n const activePath = this.checkIfNewPathNeeded(paths, exports.eSketchPathType.points);\n const x = this.initialPosition.x;\n const y = this.initialPosition.y;\n const newControlPoint = this.nodeManager.create(Token.SketchControlPoint);\n newControlPoint.position = new exports.KbVector(x, y, 0);\n this._currentPosition = undefined;\n activePath.controlPoints.push(newControlPoint);\n return activePath;\n }\n addNewPath(paths, pathType) {\n const newPath = this.nodeManager.create(Token.SketchPath);\n newPath.name = 'Path';\n newPath._expanded = true;\n paths.push(newPath);\n if (pathType) {\n newPath.type = pathType;\n }\n return newPath;\n }\n calculateReflection(vector, controlPoint, controlPoint2) {\n const n = controlPoint2.position.subtract(controlPoint.position).normalize();\n return vector.subtract(n.scale(2 * vector.dotProduct(n))).scale(-1);\n }\n }\n\n const VERSION_NUMBER = \"0.2.1\";\n const PATHSCAN_COMBINED_LOOKUP = [\n [[-1, -1, -1, -1], [-1, -1, -1, -1], [-1, -1, -1, -1], [-1, -1, -1, -1]],\n [[0, 1, 0, -1], [-1, -1, -1, -1], [-1, -1, -1, -1], [0, 2, -1, 0]],\n [[-1, -1, -1, -1], [-1, -1, -1, -1], [0, 1, 0, -1], [0, 0, 1, 0]],\n [[0, 0, 1, 0], [-1, -1, -1, -1], [0, 2, -1, 0], [-1, -1, -1, -1]],\n [[-1, -1, -1, -1], [0, 0, 1, 0], [0, 3, 0, 1], [-1, -1, -1, -1]],\n [[13, 3, 0, 1], [13, 2, -1, 0], [7, 1, 0, -1], [7, 0, 1, 0]],\n [[-1, -1, -1, -1], [0, 1, 0, -1], [-1, -1, -1, -1], [0, 3, 0, 1]],\n [[0, 3, 0, 1], [0, 2, -1, 0], [-1, -1, -1, -1], [-1, -1, -1, -1]],\n [[0, 3, 0, 1], [0, 2, -1, 0], [-1, -1, -1, -1], [-1, -1, -1, -1]],\n [[-1, -1, -1, -1], [0, 1, 0, -1], [-1, -1, -1, -1], [0, 3, 0, 1]],\n [[11, 1, 0, -1], [14, 0, 1, 0], [14, 3, 0, 1], [11, 2, -1, 0]],\n [[-1, -1, -1, -1], [0, 0, 1, 0], [0, 3, 0, 1], [-1, -1, -1, -1]],\n [[0, 0, 1, 0], [-1, -1, -1, -1], [0, 2, -1, 0], [-1, -1, -1, -1]],\n [[-1, -1, -1, -1], [-1, -1, -1, -1], [0, 1, 0, -1], [0, 0, 1, 0]],\n [[0, 1, 0, -1], [-1, -1, -1, -1], [-1, -1, -1, -1], [0, 2, -1, 0]],\n [[-1, -1, -1, -1], [-1, -1, -1, -1], [-1, -1, -1, -1], [-1, -1, -1, -1]]\n ];\n const SPECPALETTE = [\n { r: 0, g: 0, b: 0, a: 255 },\n { r: 128, g: 128, b: 128, a: 255 },\n { r: 0, g: 0, b: 128, a: 255 },\n { r: 64, g: 64, b: 128, a: 255 },\n { r: 192, g: 192, b: 192, a: 255 },\n { r: 255, g: 255, b: 255, a: 255 },\n { r: 128, g: 128, b: 192, a: 255 },\n { r: 0, g: 0, b: 192, a: 255 },\n { r: 128, g: 0, b: 0, a: 255 },\n { r: 128, g: 64, b: 64, a: 255 },\n { r: 128, g: 0, b: 128, a: 255 },\n { r: 168, g: 168, b: 168, a: 255 },\n { r: 192, g: 128, b: 128, a: 255 },\n { r: 192, g: 0, b: 0, a: 255 },\n { r: 255, g: 255, b: 255, a: 255 },\n { r: 0, g: 128, b: 0, a: 255 }\n ];\n const GKS = [\n [0.27901, 0.44198, 0.27901],\n [0.135336, 0.228569, 0.272192, 0.228569, 0.135336],\n [0.086776, 0.136394, 0.178908, 0.195843, 0.178908, 0.136394, 0.086776],\n [0.063327, 0.093095, 0.122589, 0.144599, 0.152781, 0.144599, 0.122589, 0.093095, 0.063327],\n [0.049692, 0.069304, 0.089767, 0.107988, 0.120651, 0.125194, 0.120651, 0.107988, 0.089767, 0.069304, 0.049692]\n ];\n\n const OPTION_PRESETS = {\n default: {\n corsenabled: false,\n ltres: 1,\n qtres: 1,\n pathomit: 8,\n rightangleenhance: true,\n colorsampling: 2,\n numberofcolors: 16,\n mincolorratio: 0,\n colorquantcycles: 3,\n layering: 0,\n strokewidth: 1,\n linefilter: false,\n scale: 1,\n roundcoords: 1,\n viewbox: false,\n desc: false,\n lcpr: 0,\n qcpr: 0,\n blurradius: 0,\n blurdelta: 20\n },\n posterized1: { colorsampling: 0, numberofcolors: 2 },\n posterized2: { numberofcolors: 4, blurradius: 5 },\n curvy: { ltres: 0.01, linefilter: true, rightangleenhance: false },\n sharp: { qtres: 0.01, linefilter: false },\n detailed: { pathomit: 0, roundcoords: 2, ltres: 0.5, qtres: 0.5, numberofcolors: 64 },\n smoothed: { blurradius: 5, blurdelta: 64 },\n grayscale: { colorsampling: 0, colorquantcycles: 1, numberofcolors: 7 },\n fixedpalette: { colorsampling: 0, colorquantcycles: 1, numberofcolors: 27 },\n randomsampling1: { colorsampling: 1, numberofcolors: 8 },\n randomsampling2: { colorsampling: 1, numberofcolors: 64 },\n artistic1: { colorsampling: 0, colorquantcycles: 1, pathomit: 0, blurradius: 5, blurdelta: 64, ltres: 0.01, linefilter: true, numberofcolors: 16, strokewidth: 2 },\n artistic2: { qtres: 0.01, colorsampling: 0, colorquantcycles: 1, numberofcolors: 4, strokewidth: 0 },\n artistic3: { qtres: 10, ltres: 10, numberofcolors: 8 },\n artistic4: { qtres: 10, ltres: 10, numberofcolors: 64, blurradius: 5, blurdelta: 256, strokewidth: 2 },\n posterized3: {\n ltres: 1,\n qtres: 1,\n pathomit: 20,\n rightangleenhance: true,\n colorsampling: 0,\n numberofcolors: 3,\n mincolorratio: 0,\n colorquantcycles: 3,\n blurradius: 3,\n blurdelta: 20,\n strokewidth: 0,\n linefilter: false,\n roundcoords: 1,\n pal: [{ r: 0, g: 0, b: 100, a: 255 }, { r: 255, g: 255, b: 255, a: 255 }]\n }\n };\n\n function generatePalette(numberOfColors) {\n const palette = [];\n let rcnt;\n let gcnt;\n let bcnt;\n if (numberOfColors < 8) {\n const graystep = Math.floor(255 / (numberOfColors - 1));\n for (let i = 0; i < numberOfColors; i++)\n palette.push({ r: i * graystep, g: i * graystep, b: i * graystep, a: 255 });\n } else {\n const colorqnum = Math.floor(Math.pow(numberOfColors, 1 / 3));\n const colorstep = Math.floor(255 / (colorqnum - 1));\n const rndnum = numberOfColors - colorqnum * colorqnum * colorqnum;\n for (rcnt = 0; rcnt < colorqnum; rcnt++) {\n for (gcnt = 0; gcnt < colorqnum; gcnt++) {\n for (bcnt = 0; bcnt < colorqnum; bcnt++)\n palette.push({ r: rcnt * colorstep, g: gcnt * colorstep, b: bcnt * colorstep, a: 255 });\n }\n }\n for (rcnt = 0; rcnt < rndnum; rcnt++)\n palette.push({ r: Math.floor(Math.random() * 255), g: Math.floor(Math.random() * 255), b: Math.floor(Math.random() * 255), a: Math.floor(Math.random() * 255) });\n }\n return palette;\n }\n function samplePalette(numberOfColors, imageData) {\n let idx;\n const palette = [];\n for (let i = 0; i < numberOfColors; i++) {\n idx = Math.floor(Math.random() * imageData.data.length / 4) * 4;\n palette.push({ r: imageData.data[idx], g: imageData.data[idx + 1], b: imageData.data[idx + 2], a: imageData.data[idx + 3] });\n }\n return palette;\n }\n function samplePaletteByGrid(numberOfColors, imageData) {\n let idx;\n const palette = [];\n const ni = Math.ceil(Math.sqrt(numberOfColors));\n const nj = Math.ceil(numberOfColors / ni);\n const vx = imageData.width / (ni + 1);\n const vy = imageData.height / (nj + 1);\n for (let j = 0; j < nj; j++) {\n for (let i = 0; i < ni; i++) {\n if (palette.length === numberOfColors) {\n break;\n } else {\n idx = Math.floor((j + 1) * vy * imageData.width + (i + 1) * vx) * 4;\n palette.push({ r: imageData.data[idx], g: imageData.data[idx + 1], b: imageData.data[idx + 2], a: imageData.data[idx + 3] });\n }\n }\n }\n return palette;\n }\n\n function blur(imageData, radius, delta) {\n let i, j, k, d, idx, racc, gacc, bacc, aacc, wacc;\n const imageData2 = new ImageData(imageData.width, imageData.height);\n radius = Math.floor(radius);\n if (radius < 1)\n return imageData;\n if (radius > 5)\n radius = 5;\n delta = Math.abs(delta);\n if (delta > 1024)\n delta = 1024;\n const thisgk = GKS[radius - 1];\n for (j = 0; j < imageData.height; j++) {\n for (i = 0; i < imageData.width; i++) {\n racc = 0;\n gacc = 0;\n bacc = 0;\n aacc = 0;\n wacc = 0;\n for (k = -radius; k < radius + 1; k++) {\n if (i + k > 0 && i + k < imageData.width) {\n idx = (j * imageData.width + i + k) * 4;\n racc += imageData.data[idx] * thisgk[k + radius];\n gacc += imageData.data[idx + 1] * thisgk[k + radius];\n bacc += imageData.data[idx + 2] * thisgk[k + radius];\n aacc += imageData.data[idx + 3] * thisgk[k + radius];\n wacc += thisgk[k + radius];\n }\n }\n idx = (j * imageData.width + i) * 4;\n imageData2.data[idx] = Math.floor(racc / wacc);\n imageData2.data[idx + 1] = Math.floor(gacc / wacc);\n imageData2.data[idx + 2] = Math.floor(bacc / wacc);\n imageData2.data[idx + 3] = Math.floor(aacc / wacc);\n }\n }\n const himageData = new Uint8ClampedArray(imageData2.data);\n for (j = 0; j < imageData.height; j++) {\n for (i = 0; i < imageData.width; i++) {\n racc = 0;\n gacc = 0;\n bacc = 0;\n aacc = 0;\n wacc = 0;\n for (k = -radius; k < radius + 1; k++) {\n if (j + k > 0 && j + k < imageData.height) {\n idx = ((j + k) * imageData.width + i) * 4;\n racc += himageData[idx] * thisgk[k + radius];\n gacc += himageData[idx + 1] * thisgk[k + radius];\n bacc += himageData[idx + 2] * thisgk[k + radius];\n aacc += himageData[idx + 3] * thisgk[k + radius];\n wacc += thisgk[k + radius];\n }\n }\n idx = (j * imageData.width + i) * 4;\n imageData2.data[idx] = Math.floor(racc / wacc);\n imageData2.data[idx + 1] = Math.floor(gacc / wacc);\n imageData2.data[idx + 2] = Math.floor(bacc / wacc);\n imageData2.data[idx + 3] = Math.floor(aacc / wacc);\n }\n }\n for (j = 0; j < imageData.height; j++) {\n for (i = 0; i < imageData.width; i++) {\n idx = (j * imageData.width + i) * 4;\n d = Math.abs(imageData2.data[idx] - imageData.data[idx]) + Math.abs(imageData2.data[idx + 1] - imageData.data[idx + 1]) + Math.abs(imageData2.data[idx + 2] - imageData.data[idx + 2]) + Math.abs(imageData2.data[idx + 3] - imageData.data[idx + 3]);\n if (d > delta) {\n imageData2.data[idx] = imageData.data[idx];\n imageData2.data[idx + 1] = imageData.data[idx + 1];\n imageData2.data[idx + 2] = imageData.data[idx + 2];\n imageData2.data[idx + 3] = imageData.data[idx + 3];\n }\n }\n }\n return imageData2;\n }\n\n function getDirection(x1, y1, x2, y2) {\n let result = 8;\n if (x1 < x2) {\n if (y1 < y2)\n result = 1;\n else if (y1 > y2)\n result = 7;\n else\n result = 0;\n } else if (x1 > x2) {\n if (y1 < y2)\n result = 3;\n else if (y1 > y2)\n result = 5;\n else\n result = 4;\n } else {\n if (y1 < y2)\n result = 2;\n else if (y1 > y2)\n result = 6;\n else\n result = 8;\n }\n return result;\n }\n const roundToDec = (val, places = void 0) => +val.toFixed(places);\n function colorQuantization(imageData, options) {\n const arr = [];\n let idx = 0;\n let cd;\n let cdl;\n let ci;\n const paletteacc = [];\n const pixelnum = imageData.width * imageData.height;\n let i;\n let j;\n let k;\n let cnt;\n let palette;\n let imageDataCache = imageData.data;\n if (imageDataCache.length < pixelnum * 4) {\n const newImageData = new Uint8ClampedArray(pixelnum * 4);\n for (let pxcnt = 0; pxcnt < pixelnum; pxcnt++) {\n newImageData[pxcnt * 4] = imageDataCache[pxcnt * 3];\n newImageData[pxcnt * 4 + 1] = imageDataCache[pxcnt * 3 + 1];\n newImageData[pxcnt * 4 + 2] = imageDataCache[pxcnt * 3 + 2];\n newImageData[pxcnt * 4 + 3] = 255;\n }\n imageDataCache = newImageData;\n }\n for (j = 0; j < imageData.height + 2; j++) {\n arr[j] = [];\n for (i = 0; i < imageData.width + 2; i++)\n arr[j][i] = -1;\n }\n if (options.pal)\n palette = options.pal;\n else if (options.colorsampling === 0)\n palette = generatePalette(options.numberofcolors);\n else if (options.colorsampling === 1)\n palette = samplePalette(options.numberofcolors, imageData);\n else\n palette = samplePaletteByGrid(options.numberofcolors, imageData);\n if (options.blurradius > 0)\n imageData = blur(imageData, options.blurradius, options.blurdelta);\n for (cnt = 0; cnt < options.colorquantcycles; cnt++) {\n if (cnt > 0) {\n for (k = 0; k < palette.length; k++) {\n if (paletteacc[k].n > 0) {\n palette[k] = {\n r: Math.floor(paletteacc[k].r / paletteacc[k].n),\n g: Math.floor(paletteacc[k].g / paletteacc[k].n),\n b: Math.floor(paletteacc[k].b / paletteacc[k].n),\n a: Math.floor(paletteacc[k].a / paletteacc[k].n)\n };\n }\n if (paletteacc[k].n / pixelnum < options.mincolorratio && cnt < options.colorquantcycles - 1) {\n palette[k] = {\n r: Math.floor(Math.random() * 255),\n g: Math.floor(Math.random() * 255),\n b: Math.floor(Math.random() * 255),\n a: Math.floor(Math.random() * 255)\n };\n }\n }\n }\n for (i = 0; i < palette.length; i++)\n paletteacc[i] = { r: 0, g: 0, b: 0, a: 0, n: 0 };\n for (j = 0; j < imageData.height; j++) {\n for (i = 0; i < imageData.width; i++) {\n idx = (j * imageData.width + i) * 4;\n ci = 0;\n cdl = 1024;\n for (k = 0; k < palette.length; k++) {\n cd = (palette[k].r > imageDataCache[idx] ? palette[k].r - imageDataCache[idx] : imageDataCache[idx] - palette[k].r) + (palette[k].g > imageDataCache[idx + 1] ? palette[k].g - imageDataCache[idx + 1] : imageDataCache[idx + 1] - palette[k].g) + (palette[k].b > imageDataCache[idx + 2] ? palette[k].b - imageDataCache[idx + 2] : imageDataCache[idx + 2] - palette[k].b) + (palette[k].a > imageDataCache[idx + 3] ? palette[k].a - imageDataCache[idx + 3] : imageDataCache[idx + 3] - palette[k].a);\n if (cd < cdl) {\n cdl = cd;\n ci = k;\n }\n }\n paletteacc[ci].r += imageDataCache[idx];\n paletteacc[ci].g += imageDataCache[idx + 1];\n paletteacc[ci].b += imageDataCache[idx + 2];\n paletteacc[ci].a += imageDataCache[idx + 3];\n paletteacc[ci].n++;\n arr[j + 1][i + 1] = ci;\n }\n }\n }\n return { array: arr, palette };\n }\n const toRgbaStr = (c) => `rgba(${c.r},${c.g},${c.b},${c.a})`;\n\n function svgPathString(tracedata, lnum, pathnum, options) {\n const layer = tracedata.layers[lnum];\n const smp = layer[pathnum];\n let str = \"\";\n let pcnt;\n if (options.linefilter && smp.segments.length < 3)\n return str;\n str = `= 0; pcnt--) {\n str += `${hsmp.segments[pcnt].type} `;\n if (hsmp.segments[pcnt].hasOwnProperty(\"x3\"))\n str += `${hsmp.segments[pcnt].x2 * options.scale} ${hsmp.segments[pcnt].y2 * options.scale} `;\n str += `${hsmp.segments[pcnt].x1 * options.scale} ${hsmp.segments[pcnt].y1 * options.scale} `;\n }\n } else {\n if (hsmp.segments[hsmp.segments.length - 1].hasOwnProperty(\"x3\"))\n str += `M ${roundToDec(hsmp.segments[hsmp.segments.length - 1].x3 * options.scale)} ${roundToDec(hsmp.segments[hsmp.segments.length - 1].y3 * options.scale)} `;\n else\n str += `M ${roundToDec(hsmp.segments[hsmp.segments.length - 1].x2 * options.scale)} ${roundToDec(hsmp.segments[hsmp.segments.length - 1].y2 * options.scale)} `;\n for (pcnt = hsmp.segments.length - 1; pcnt >= 0; pcnt--) {\n str += `${hsmp.segments[pcnt].type} `;\n if (hsmp.segments[pcnt].hasOwnProperty(\"x3\"))\n str += `${roundToDec(hsmp.segments[pcnt].x2 * options.scale)} ${roundToDec(hsmp.segments[pcnt].y2 * options.scale)} `;\n str += `${roundToDec(hsmp.segments[pcnt].x1 * options.scale)} ${roundToDec(hsmp.segments[pcnt].y1 * options.scale)} `;\n }\n }\n str += \"Z \";\n }\n str += '\" />';\n if (options.lcpr || options.qcpr) {\n for (pcnt = 0; pcnt < smp.segments.length; pcnt++) {\n if (smp.segments[pcnt].hasOwnProperty(\"x3\") && options.qcpr) {\n str += ``;\n str += ``;\n str += ``;\n str += ``;\n }\n if (!smp.segments[pcnt].hasOwnProperty(\"x3\") && options.lcpr)\n str += ``;\n }\n for (let hcnt = 0; hcnt < smp.holechildren.length; hcnt++) {\n const hsmp = layer[smp.holechildren[hcnt]];\n for (pcnt = 0; pcnt < hsmp.segments.length; pcnt++) {\n if (hsmp.segments[pcnt].hasOwnProperty(\"x3\") && options.qcpr) {\n str += ``;\n str += ``;\n str += ``;\n str += ``;\n }\n if (!hsmp.segments[pcnt].hasOwnProperty(\"x3\") && options.lcpr)\n str += ``;\n }\n }\n }\n return str;\n }\n function toSvgColorStr(c, options) {\n return `fill=\"rgb(${c.r},${c.g},${c.b})\" stroke=\"rgb(${c.r},${c.g},${c.b})\" stroke-width=\"${options.strokewidth}\" opacity=\"${c.a / 255}\" `;\n }\n function fitSeq(path, ltres, qtres, seqstart, seqend) {\n if (seqend > path.points.length || seqend < 0)\n return [];\n let errorpoint = seqstart;\n let errorval = 0;\n let curvepass = true;\n let px;\n let py;\n let dist2;\n let tl = seqend - seqstart;\n if (tl < 0)\n tl += path.points.length;\n const vx = (path.points[seqend].x - path.points[seqstart].x) / tl;\n const vy = (path.points[seqend].y - path.points[seqstart].y) / tl;\n let pcnt = (seqstart + 1) % path.points.length;\n let pl;\n while (pcnt !== seqend) {\n pl = pcnt - seqstart;\n if (pl < 0)\n pl += path.points.length;\n px = path.points[seqstart].x + vx * pl;\n py = path.points[seqstart].y + vy * pl;\n dist2 = (path.points[pcnt].x - px) * (path.points[pcnt].x - px) + (path.points[pcnt].y - py) * (path.points[pcnt].y - py);\n if (dist2 > ltres)\n curvepass = false;\n if (dist2 > errorval) {\n errorpoint = pcnt;\n errorval = dist2;\n }\n pcnt = (pcnt + 1) % path.points.length;\n }\n if (curvepass)\n return [{ type: \"L\", x1: path.points[seqstart].x, y1: path.points[seqstart].y, x2: path.points[seqend].x, y2: path.points[seqend].y }];\n const fitpoint = errorpoint;\n curvepass = true;\n errorval = 0;\n let t = (fitpoint - seqstart) / tl;\n let t1 = (1 - t) * (1 - t);\n let t2 = 2 * (1 - t) * t;\n let t3 = t * t;\n const cpx = (t1 * path.points[seqstart].x + t3 * path.points[seqend].x - path.points[fitpoint].x) / -t2;\n const cpy = (t1 * path.points[seqstart].y + t3 * path.points[seqend].y - path.points[fitpoint].y) / -t2;\n pcnt = seqstart + 1;\n while (pcnt !== seqend) {\n t = (pcnt - seqstart) / tl;\n t1 = (1 - t) * (1 - t);\n t2 = 2 * (1 - t) * t;\n t3 = t * t;\n px = t1 * path.points[seqstart].x + t2 * cpx + t3 * path.points[seqend].x;\n py = t1 * path.points[seqstart].y + t2 * cpy + t3 * path.points[seqend].y;\n dist2 = (path.points[pcnt].x - px) * (path.points[pcnt].x - px) + (path.points[pcnt].y - py) * (path.points[pcnt].y - py);\n if (dist2 > qtres)\n curvepass = false;\n if (dist2 > errorval) {\n errorpoint = pcnt;\n errorval = dist2;\n }\n pcnt = (pcnt + 1) % path.points.length;\n }\n if (curvepass)\n return [{ type: \"Q\", x1: path.points[seqstart].x, y1: path.points[seqstart].y, x2: cpx, y2: cpy, x3: path.points[seqend].x, y3: path.points[seqend].y }];\n const splitpoint = fitpoint;\n return fitSeq(path, ltres, qtres, seqstart, splitpoint).concat(fitSeq(path, ltres, qtres, splitpoint, seqend));\n }\n function boundingBoxIncludes(parentbbox, childbbox) {\n return parentbbox[0] < childbbox[0] && parentbbox[1] < childbbox[1] && parentbbox[2] > childbbox[2] && parentbbox[3] > childbbox[3];\n }\n function pointInPoly(p, pa) {\n let isin = false;\n for (let i = 0, j = pa.length - 1; i < pa.length; j = i++) {\n isin = pa[i].y > p.y !== pa[j].y > p.y && p.x < (pa[j].x - pa[i].x) * (p.y - pa[i].y) / (pa[j].y - pa[i].y) + pa[i].x ? !isin : isin;\n }\n return isin;\n }\n function getSvgString(tracedata, options) {\n const w = tracedata.width * options.scale;\n const h = tracedata.height * options.scale;\n let svgstr = ``;\n for (let lcnt = 0; lcnt < tracedata.layers.length; lcnt++) {\n for (let pcnt = 0; pcnt < tracedata.layers[lcnt].length; pcnt++) {\n if (!tracedata.layers[lcnt][pcnt].isholepath)\n svgstr += svgPathString(tracedata, lcnt, pcnt, options);\n }\n }\n svgstr += \"\";\n return svgstr;\n }\n\n function testRightAngle(path, idx1, idx2, idx3, idx4, idx5) {\n return path.points[idx3].x === path.points[idx1].x && path.points[idx3].x === path.points[idx2].x && path.points[idx3].y === path.points[idx4].y && path.points[idx3].y === path.points[idx5].y || path.points[idx3].y === path.points[idx1].y && path.points[idx3].y === path.points[idx2].y && path.points[idx3].x === path.points[idx4].x && path.points[idx3].x === path.points[idx5].x;\n }\n function pathScan(arr, pathomit) {\n const paths = [];\n let pacnt = 0;\n let pcnt = 0;\n let px = 0;\n let py = 0;\n const w = arr[0].length;\n const h = arr.length;\n let dir = 0;\n let pathfinished = true;\n let holepath = false;\n let lookuprow;\n for (let j = 0; j < h; j++) {\n for (let i = 0; i < w; i++) {\n if (arr[j][i] === 4 || arr[j][i] === 11) {\n px = i;\n py = j;\n paths[pacnt] = {};\n paths[pacnt].points = [];\n paths[pacnt].boundingbox = [px, py, px, py];\n paths[pacnt].holechildren = [];\n pathfinished = false;\n pcnt = 0;\n holepath = arr[j][i] === 11;\n dir = 1;\n while (!pathfinished) {\n paths[pacnt].points[pcnt] = {};\n paths[pacnt].points[pcnt].x = px - 1;\n paths[pacnt].points[pcnt].y = py - 1;\n paths[pacnt].points[pcnt].t = arr[py][px];\n if (px - 1 < paths[pacnt].boundingbox[0])\n paths[pacnt].boundingbox[0] = px - 1;\n if (px - 1 > paths[pacnt].boundingbox[2])\n paths[pacnt].boundingbox[2] = px - 1;\n if (py - 1 < paths[pacnt].boundingbox[1])\n paths[pacnt].boundingbox[1] = py - 1;\n if (py - 1 > paths[pacnt].boundingbox[3])\n paths[pacnt].boundingbox[3] = py - 1;\n lookuprow = PATHSCAN_COMBINED_LOOKUP[arr[py][px]][dir];\n arr[py][px] = lookuprow[0];\n dir = lookuprow[1];\n px += lookuprow[2];\n py += lookuprow[3];\n if (px - 1 === paths[pacnt].points[0].x && py - 1 === paths[pacnt].points[0].y) {\n pathfinished = true;\n if (paths[pacnt].points.length < pathomit) {\n paths.pop();\n } else {\n paths[pacnt].isholepath = !!holepath;\n if (holepath) {\n let parentidx = 0;\n let parentbbox = [-1, -1, w + 1, h + 1];\n for (let parentcnt = 0; parentcnt < pacnt; parentcnt++) {\n if (!paths[parentcnt].isholepath && boundingBoxIncludes(paths[parentcnt].boundingbox, paths[pacnt].boundingbox) && boundingBoxIncludes(parentbbox, paths[parentcnt].boundingbox) && pointInPoly(paths[pacnt].points[0], paths[parentcnt].points)) {\n parentidx = parentcnt;\n parentbbox = paths[parentcnt].boundingbox;\n }\n }\n paths[parentidx].holechildren.push(pacnt);\n }\n pacnt++;\n }\n }\n pcnt++;\n }\n }\n }\n }\n return paths;\n }\n function batchPathScan(layers, pathomit) {\n const bpaths = [];\n for (let k = 0; k < layers.length; k++)\n bpaths[k] = pathScan(layers[k], pathomit);\n return bpaths;\n }\n function tracePath(path, ltres, qtres) {\n let pcnt = 0;\n let segtype1;\n let segtype2;\n let seqend;\n const smp = {\n segments: [],\n boundingbox: path.boundingbox,\n holechildren: path.holechildren,\n isholepath: path.isholepath\n };\n while (pcnt < path.points.length) {\n segtype1 = path.points[pcnt].linesegment;\n segtype2 = -1;\n seqend = pcnt + 1;\n while ((path.points[seqend].linesegment === segtype1 || path.points[seqend].linesegment === segtype2 || segtype2 === -1) && seqend < path.points.length - 1) {\n if (path.points[seqend].linesegment !== segtype1 && segtype2 === -1)\n segtype2 = path.points[seqend].linesegment;\n seqend++;\n }\n if (seqend === path.points.length - 1)\n seqend = 0;\n smp.segments = smp.segments.concat(fitSeq(path, ltres, qtres, pcnt, seqend));\n if (seqend > 0)\n pcnt = seqend;\n else\n pcnt = path.points.length;\n }\n return smp;\n }\n function batchTracePaths(internodepaths, ltres, qtres) {\n const btracedpaths = [];\n for (let k = 0; k < internodepaths.length; k++)\n btracedpaths.push(tracePath(internodepaths[k], ltres, qtres));\n return btracedpaths;\n }\n\n function batchInterNodes(bpaths, options) {\n const binternodes = [];\n for (let k = 0; k < bpaths.length; k++)\n binternodes[k] = interNodes(bpaths[k], options);\n return binternodes;\n }\n function interNodes(paths, options) {\n const ins = [];\n let palen = 0;\n let nextidx = 0;\n let nextidx2 = 0;\n let previdx = 0;\n let previdx2 = 0;\n let pacnt;\n let pcnt;\n for (pacnt = 0; pacnt < paths.length; pacnt++) {\n ins[pacnt] = {};\n ins[pacnt].points = [];\n ins[pacnt].boundingbox = paths[pacnt].boundingbox;\n ins[pacnt].holechildren = paths[pacnt].holechildren;\n ins[pacnt].isholepath = paths[pacnt].isholepath;\n palen = paths[pacnt].points.length;\n for (pcnt = 0; pcnt < palen; pcnt++) {\n nextidx = (pcnt + 1) % palen;\n nextidx2 = (pcnt + 2) % palen;\n previdx = (pcnt - 1 + palen) % palen;\n previdx2 = (pcnt - 2 + palen) % palen;\n if (options.rightangleenhance && testRightAngle(paths[pacnt], previdx2, previdx, pcnt, nextidx, nextidx2)) {\n if (ins[pacnt].points.length > 0) {\n ins[pacnt].points[ins[pacnt].points.length - 1].linesegment = getDirection(ins[pacnt].points[ins[pacnt].points.length - 1].x, ins[pacnt].points[ins[pacnt].points.length - 1].y, paths[pacnt].points[pcnt].x, paths[pacnt].points[pcnt].y);\n }\n ins[pacnt].points.push({\n x: paths[pacnt].points[pcnt].x,\n y: paths[pacnt].points[pcnt].y,\n linesegment: getDirection(paths[pacnt].points[pcnt].x, paths[pacnt].points[pcnt].y, (paths[pacnt].points[pcnt].x + paths[pacnt].points[nextidx].x) / 2, (paths[pacnt].points[pcnt].y + paths[pacnt].points[nextidx].y) / 2)\n });\n }\n ins[pacnt].points.push({\n x: (paths[pacnt].points[pcnt].x + paths[pacnt].points[nextidx].x) / 2,\n y: (paths[pacnt].points[pcnt].y + paths[pacnt].points[nextidx].y) / 2,\n linesegment: getDirection((paths[pacnt].points[pcnt].x + paths[pacnt].points[nextidx].x) / 2, (paths[pacnt].points[pcnt].y + paths[pacnt].points[nextidx].y) / 2, (paths[pacnt].points[nextidx].x + paths[pacnt].points[nextidx2].x) / 2, (paths[pacnt].points[nextidx].y + paths[pacnt].points[nextidx2].y) / 2)\n });\n }\n }\n return ins;\n }\n\n function batchTraceLayers(binternodes, ltres, qtres) {\n const btbis = [];\n for (let k = 0; k < binternodes.length; k++)\n btbis[k] = batchTracePaths(binternodes[k], ltres, qtres);\n return btbis;\n }\n function layering(ii) {\n const layers = [];\n let val = 0;\n const ah = ii.array.length;\n const aw = ii.array[0].length;\n let n1;\n let n2;\n let n3;\n let n4;\n let n5;\n let n6;\n let n7;\n let n8;\n let i;\n let j;\n let k;\n for (k = 0; k < ii.palette.length; k++) {\n layers[k] = [];\n for (j = 0; j < ah; j++) {\n layers[k][j] = [];\n for (i = 0; i < aw; i++)\n layers[k][j][i] = 0;\n }\n }\n for (j = 1; j < ah - 1; j++) {\n for (i = 1; i < aw - 1; i++) {\n val = ii.array[j][i];\n n1 = ii.array[j - 1][i - 1] === val ? 1 : 0;\n n2 = ii.array[j - 1][i] === val ? 1 : 0;\n n3 = ii.array[j - 1][i + 1] === val ? 1 : 0;\n n4 = ii.array[j][i - 1] === val ? 1 : 0;\n n5 = ii.array[j][i + 1] === val ? 1 : 0;\n n6 = ii.array[j + 1][i - 1] === val ? 1 : 0;\n n7 = ii.array[j + 1][i] === val ? 1 : 0;\n n8 = ii.array[j + 1][i + 1] === val ? 1 : 0;\n layers[val][j + 1][i + 1] = 1 + n5 * 2 + n8 * 4 + n7 * 8;\n if (!n4)\n layers[val][j + 1][i] = 0 + 2 + n7 * 4 + n6 * 8;\n if (!n2)\n layers[val][j][i + 1] = 0 + n3 * 2 + n5 * 4 + 8;\n if (!n1)\n layers[val][j][i] = 0 + n2 * 2 + 4 + n4 * 8;\n }\n }\n return layers;\n }\n function layeringStep(ii, cnum) {\n const layer = [];\n const ah = ii.array.length;\n const aw = ii.array[0].length;\n let i;\n let j;\n for (j = 0; j < ah; j++) {\n layer[j] = [];\n for (i = 0; i < aw; i++)\n layer[j][i] = 0;\n }\n for (j = 1; j < ah; j++) {\n for (i = 1; i < aw; i++) {\n layer[j][i] = (ii.array[j - 1][i - 1] === cnum ? 1 : 0) + (ii.array[j - 1][i] === cnum ? 2 : 0) + (ii.array[j][i - 1] === cnum ? 8 : 0) + (ii.array[j][i] === cnum ? 4 : 0);\n }\n }\n return layer;\n }\n function drawLayers(layers, palette, scale = 1, parentid) {\n let w, h, i, j;\n let div;\n if (parentid) {\n div = document.getElementById(parentid);\n if (!div) {\n div = document.createElement(\"div\");\n div.id = parentid;\n document.body.appendChild(div);\n }\n } else {\n div = document.createElement(\"div\");\n document.body.appendChild(div);\n }\n for (const layer of layers) {\n w = layer[0].length;\n h = layer.length;\n const canvas = document.createElement(\"canvas\");\n canvas.width = w * scale;\n canvas.height = h * scale;\n const context = canvas.getContext(\"2d\");\n for (j = 0; j < h; j++) {\n for (i = 0; i < w; i++) {\n context.fillStyle = toRgbaStr(palette[layer[j][i] % palette.length]);\n context.fillRect(i * scale, j * scale, scale, scale);\n }\n }\n div.appendChild(canvas);\n }\n }\n\n class ImageTracer {\n constructor() {\n this.versionnumber = VERSION_NUMBER;\n this.optionpresets = OPTION_PRESETS;\n }\n checkOptions(options) {\n if (options === void 0)\n return { ...this.optionpresets.default };\n if (typeof options === \"string\") {\n const presetName = options;\n const preset = this.optionpresets[presetName];\n return { ...this.optionpresets.default, ...preset || {} };\n }\n if (typeof options === \"object\") {\n return { ...this.optionpresets.default, ...options };\n }\n throw new Error(`Unknown Option Type: ${typeof options}`);\n }\n async imageToSVG(url, options) {\n options = this.checkOptions(options);\n const canvas = await this.loadImage(url, options);\n return this.imageDataToSVG(this.getImgdata(canvas), options);\n }\n getImgdata(canvas) {\n const context = canvas.getContext(\"2d\");\n return context.getImageData(0, 0, canvas.width, canvas.height);\n }\n loadImage(url, options) {\n return new Promise((resolve) => {\n const img = new Image();\n if (options && options.corsenabled)\n img.crossOrigin = \"Anonymous\";\n img.src = url;\n img.onload = function() {\n const canvas = document.createElement(\"canvas\");\n canvas.width = img.width;\n canvas.height = img.height;\n const context = canvas.getContext(\"2d\");\n context?.drawImage(img, 0, 0);\n resolve(canvas);\n };\n });\n }\n imageDataToSVG(imgd, options) {\n options = this.checkOptions(options);\n const td = this.imageDataToTracedata(imgd, options);\n return getSvgString(td, options);\n }\n async imageToTracedata(url, options) {\n options = this.checkOptions(options);\n const canvas = await this.loadImage(url, options);\n return this.imageDataToTracedata(this.getImgdata(canvas), options);\n }\n imageDataToTracedata(imgd, options) {\n options = this.checkOptions(options);\n let tracedata;\n const ii = colorQuantization(imgd, options);\n if (options.layering === 0) {\n tracedata = {\n layers: [],\n palette: ii.palette,\n width: ii.array[0].length - 2,\n height: ii.array.length - 2\n };\n for (let colornum = 0; colornum < ii.palette.length; colornum++) {\n const tracedlayer = batchTracePaths(interNodes(pathScan(layeringStep(ii, colornum), options.pathomit), options), options.ltres, options.qtres);\n tracedata.layers.push(tracedlayer);\n }\n } else {\n const ls = layering(ii);\n if (options.layercontainerid)\n drawLayers(ls, SPECPALETTE, options.scale, options.layercontainerid);\n const bps = batchPathScan(ls, options.pathomit);\n const bis = batchInterNodes(bps, options);\n tracedata = {\n layers: batchTraceLayers(bis, options.ltres, options.qtres),\n palette: ii.palette,\n width: imgd.width,\n height: imgd.height\n };\n }\n return tracedata;\n }\n appendSVGString(svgstr, parentid) {\n let div;\n if (parentid) {\n div = document.getElementById(parentid);\n if (!div) {\n div = document.createElement(\"div\");\n div.id = parentid;\n document.body.appendChild(div);\n }\n } else {\n div = document.createElement(\"div\");\n document.body.appendChild(div);\n }\n div.innerHTML += svgstr;\n }\n }\n const imageTracer = new ImageTracer();\n\n function getLocator(source, options) {\n if (options === void 0) { options = {}; }\n var offsetLine = options.offsetLine || 0;\n var offsetColumn = options.offsetColumn || 0;\n var originalLines = source.split('\\n');\n var start = 0;\n var lineRanges = originalLines.map(function (line, i) {\n var end = start + line.length + 1;\n var range = { start: start, end: end, line: i };\n start = end;\n return range;\n });\n var i = 0;\n function rangeContains(range, index) {\n return range.start <= index && index < range.end;\n }\n function getLocation(range, index) {\n return { line: offsetLine + range.line, column: offsetColumn + index - range.start, character: index };\n }\n function locate(search, startIndex) {\n if (typeof search === 'string') {\n search = source.indexOf(search, startIndex || 0);\n }\n var range = lineRanges[i];\n var d = search >= range.end ? 1 : -1;\n while (range) {\n if (rangeContains(range, search))\n return getLocation(range, search);\n i += d;\n range = lineRanges[i];\n }\n }\n return locate;\n }\n function locate(source, search, options) {\n if (typeof options === 'number') {\n throw new Error('locate takes a { startIndex, offsetLine, offsetColumn } object as the third argument');\n }\n return getLocator(source, options)(search, options && options.startIndex);\n }\n\n var validNameCharacters = /[a-zA-Z0-9:_-]/;\n var whitespace = /[\\s\\t\\r\\n]/;\n var quotemark = /['\"]/;\n\n function repeat(str, i) {\n \tvar result = '';\n \twhile (i--) { result += str; }\n \treturn result;\n }\n\n function parse$1(source) {\n \tvar header = '';\n \tvar stack = [];\n\n \tvar state = metadata;\n \tvar currentElement = null;\n \tvar root = null;\n\n \tfunction error(message) {\n \t\tvar ref = locate(source, i);\n \t\tvar line = ref.line;\n \t\tvar column = ref.column;\n \t\tvar before = source.slice(0, i);\n \t\tvar beforeLine = /(^|\\n).*$/.exec(before)[0].replace(/\\t/g, ' ');\n \t\tvar after = source.slice(i);\n \t\tvar afterLine = /.*(\\n|$)/.exec(after)[0];\n\n \t\tvar snippet = \"\" + beforeLine + afterLine + \"\\n\" + (repeat(' ', beforeLine.length)) + \"^\";\n\n \t\tthrow new Error(\n \t\t\t(message + \" (\" + line + \":\" + column + \"). If this is valid SVG, it's probably a bug in svg-parser. Please raise an issue at https://github.com/Rich-Harris/svg-parser/issues – thanks!\\n\\n\" + snippet)\n \t\t);\n \t}\n\n \tfunction metadata() {\n \t\twhile ((i < source.length && source[i] !== '<') || !validNameCharacters.test(source[i + 1])) {\n \t\t\theader += source[i++];\n \t\t}\n\n \t\treturn neutral();\n \t}\n\n \tfunction neutral() {\n \t\tvar text = '';\n \t\twhile (i < source.length && source[i] !== '<') { text += source[i++]; }\n\n \t\tif (/\\S/.test(text)) {\n \t\t\tcurrentElement.children.push({ type: 'text', value: text });\n \t\t}\n\n \t\tif (source[i] === '<') {\n \t\t\treturn tag;\n \t\t}\n\n \t\treturn neutral;\n \t}\n\n \tfunction tag() {\n \t\tvar char = source[i];\n\n \t\tif (char === '?') { return neutral; } // ') {\n \t\t\terror('Expected >');\n \t\t}\n\n \t\tif (!selfClosing) {\n \t\t\tcurrentElement = element;\n \t\t\tstack.push(element);\n \t\t}\n\n \t\treturn neutral;\n \t}\n\n \tfunction comment() {\n \t\tvar index = source.indexOf('-->', i);\n \t\tif (!~index) { error('expected -->'); }\n\n \t\ti = index + 2;\n \t\treturn neutral;\n \t}\n\n \tfunction cdata() {\n \t\tvar index = source.indexOf(']]>', i);\n \t\tif (!~index) { error('expected ]]>'); }\n\n \t\tcurrentElement.children.push(source.slice(i + 7, index));\n\n \t\ti = index + 2;\n \t\treturn neutral;\n \t}\n\n \tfunction closingTag() {\n \t\tvar tagName = getName();\n\n \t\tif (!tagName) { error('Expected tag name'); }\n\n \t\tif (tagName !== currentElement.tagName) {\n \t\t\terror((\"Expected closing tag to match opening tag <\" + (currentElement.tagName) + \">\"));\n \t\t}\n\n \t\tallowSpaces();\n\n \t\tif (source[i] !== '>') {\n \t\t\terror('Expected >');\n \t\t}\n\n \t\tstack.pop();\n \t\tcurrentElement = stack[stack.length - 1];\n\n \t\treturn neutral;\n \t}\n\n \tfunction getName() {\n \t\tvar name = '';\n \t\twhile (i < source.length && validNameCharacters.test(source[i])) { name += source[i++]; }\n\n \t\treturn name;\n \t}\n\n \tfunction getAttribute() {\n \t\tif (!whitespace.test(source[i])) { return null; }\n \t\tallowSpaces();\n\n \t\tvar name = getName();\n \t\tif (!name) { return null; }\n\n \t\tvar value = true;\n\n \t\tallowSpaces();\n \t\tif (source[i] === '=') {\n \t\t\ti += 1;\n \t\t\tallowSpaces();\n\n \t\t\tvalue = getAttributeValue();\n \t\t\tif (!isNaN(value) && value.trim() !== '') { value = +value; } // TODO whitelist numeric attributes?\n \t\t}\n\n \t\treturn { name: name, value: value };\n \t}\n\n \tfunction getAttributeValue() {\n \t\treturn quotemark.test(source[i]) ? getQuotedAttributeValue() : getUnquotedAttributeValue();\n \t}\n\n \tfunction getUnquotedAttributeValue() {\n \t\tvar value = '';\n \t\tdo {\n \t\t\tvar char = source[i];\n \t\t\tif (char === ' ' || char === '>' || char === '/') {\n \t\t\t\treturn value;\n \t\t\t}\n\n \t\t\tvalue += char;\n \t\t\ti += 1;\n \t\t} while (i < source.length);\n\n \t\treturn value;\n \t}\n\n \tfunction getQuotedAttributeValue() {\n \t\tvar quotemark = source[i++];\n\n \t\tvar value = '';\n \t\tvar escaped = false;\n\n \t\twhile (i < source.length) {\n \t\t\tvar char = source[i++];\n \t\t\tif (char === quotemark && !escaped) {\n \t\t\t\treturn value;\n \t\t\t}\n\n \t\t\tif (char === '\\\\' && !escaped) {\n \t\t\t\tescaped = true;\n \t\t\t}\n\n \t\t\tvalue += escaped ? (\"\\\\\" + char) : char;\n \t\t\tescaped = false;\n \t\t}\n \t}\n\n \tfunction allowSpaces() {\n \t\twhile (i < source.length && whitespace.test(source[i])) { i += 1; }\n \t}\n\n \tvar i = metadata.length;\n \twhile (i < source.length) {\n \t\tif (!state) { error('Unexpected character'); }\n \t\tstate = state();\n \t\ti += 1;\n \t}\n\n \tif (state !== neutral) {\n \t\terror('Unexpected end of input');\n \t}\n\n \tif (root.tagName === 'svg') { root.metadata = header; }\n \treturn {\n \t\ttype: 'root',\n \t\tchildren: [root]\n \t};\n }\n\n class SvgConverter {\n static overwriteSketchWithSvg(node, data) {\n const decover = new TextDecoder();\n node.paths.clear();\n const header = data.slice(0, 100);\n header.arrayBuffer().then(buffer => {\n const headerText = decover.decode(buffer);\n if (headerText.search(/JFIF/) !== -1 || headerText.search(/PNG/) !== -1) {\n // is raster\n const dataUrl = URL.createObjectURL(data);\n const options = {\n qtres: 0.1,\n pathomit: 16,\n numberofcolors: 2,\n };\n const nodeManager = node._manager;\n if (!nodeManager) {\n throw Error('no node manager');\n }\n imageTracer.imageToTracedata(dataUrl, options).then((trace) => {\n if (trace.palette && trace.palette.length > 1) {\n const pathParser = new PathHandler(node, SvgConverter.calculateScaleDivider([0, 0, trace.width, trace.height]), nodeManager);\n const paletteStrengths = trace.palette.map((color, idx) => ({\n idx,\n strength: getPaletteColorStrength(color),\n }));\n paletteStrengths.sort((a, b) => a.strength - b.strength);\n const bgIdx = paletteStrengths[0].idx;\n trace.layers.splice(bgIdx, 1);\n const primaryLayers = trace.layers.flat();\n const pathsCreated = [];\n // Convert layers to sketchPaths\n primaryLayers.forEach((layer, layerIndex) => {\n const newPaths = pathParser.parseTrace(layer.segments);\n const pathGroupKey = 'tracedPath' + layerIndex;\n pathsCreated[layerIndex] = newPaths;\n newPaths.forEach(path => {\n path.pathGroupKey = pathGroupKey;\n });\n node.paths.push(...newPaths);\n });\n // Convert layers to holes\n primaryLayers.forEach((layer, layerIndex) => {\n if (layer.holechildren) {\n layer.holechildren.forEach(holeLayerIdx => {\n const pathSet = pathsCreated[holeLayerIdx];\n if (pathSet) {\n pathSet.forEach(path => {\n path.holeForPathGroup = 'tracedPath' + layerIndex;\n });\n }\n });\n }\n });\n }\n });\n }\n else {\n data.text().then(svgText => {\n const parsedSVG = parse$1(svgText);\n SvgConverter.parseSvgRecursively(node, parsedSVG);\n });\n }\n });\n }\n static findSvgViewBox(svg) {\n for (let i = 0; i < svg.children.length; i++) {\n const child = svg.children[i];\n if (typeof child === 'string') {\n return undefined;\n }\n else if (child.tagName === 'svg' && 'properties' in child) {\n return getViewBoxFromSvgElement(child);\n }\n }\n function getViewBoxFromSvgElement(elem) {\n var _a, _b, _c;\n if (typeof ((_a = elem.properties) === null || _a === void 0 ? void 0 : _a.viewBox) === 'string' && elem.properties.viewBox) {\n return elem.properties.viewBox.split(' ').map(int => parseInt(int));\n }\n if (((_b = elem.properties) === null || _b === void 0 ? void 0 : _b.width) && ((_c = elem.properties) === null || _c === void 0 ? void 0 : _c.height)) {\n const width = elem.properties.width;\n const height = elem.properties.height;\n return [\n 0,\n 0,\n typeof width === 'string' ? parseInt(width) : width,\n typeof height === 'string' ? parseInt(height) : height,\n ];\n }\n }\n }\n static parseSvgRecursively(node, svg, viewBox) {\n if (!viewBox) {\n viewBox = SvgConverter.findSvgViewBox(svg) || [0, 0, 100, 100];\n }\n if (svg.children) {\n const nodeManager = node._manager;\n if (!nodeManager) {\n throw Error('no node manager');\n }\n for (let i = 0; i < svg.children.length; i++) {\n const child = svg.children[i];\n if (child.tagName === 'path' && child.properties) {\n const pathParser = new PathHandler(node, SvgConverter.calculateScaleDivider(viewBox), nodeManager);\n const newPaths = pathParser.parsePath(child.properties.d);\n node.paths.push(...newPaths);\n //call actual parser\n }\n else if (child.tagName === 'rect' && child.properties) {\n SvgConverter.createRectangle(node, child, viewBox);\n }\n else if (child.tagName === 'circle' && child.properties) {\n SvgConverter.createCircle(node, child, viewBox);\n }\n else if (child.tagName === 'polyline' && child.properties) {\n SvgConverter.createPolyLine(node, child, viewBox);\n }\n else if (child.tagName === 'line' && child.properties) {\n SvgConverter.createLine(node, child, viewBox);\n }\n else if (child.tagName === 'polygon' && child.properties) {\n SvgConverter.createPolyLine(node, child, viewBox, true);\n }\n SvgConverter.parseSvgRecursively(node, child, viewBox);\n nameGeneratedPaths(node);\n }\n }\n }\n static removeBackGroundOnConvertedSVG(children) {\n const parent = children[0];\n const newChildren = parent.children;\n const resultChildren = [];\n newChildren.forEach(child => {\n if (child && child.properties) {\n if ('opacity' in child.properties) {\n if (child.properties.opacity == 1) {\n resultChildren.push(child);\n }\n }\n }\n });\n newChildren.clear();\n resultChildren.forEach(child => {\n newChildren.push(child);\n });\n // if (svg.children) {\n // console.log(svg.children.length);\n // for (let i = 0; i < svg.children.length; i++) {\n // const child = svg.children[i] as ElementNode;\n // if (child.properties) {\n // if (child.properties.opacity) {\n // if (Number(child.properties.opacity) < 1) {\n // console.log('Not a valid path');\n // svg.children.remove(child);\n // } else {\n // console.log('valid path:', child);\n // }\n // }\n // }\n // this.removeBackGroundOnConvertedSVG(child);\n // }\n // }\n }\n // static projectSketchOnNormal(node: SketchNode) {\n // node.paths.forEach(path => {\n // path.controlPoints.forEach(point => {\n // if (node.normal === eAxis.y) {\n // point.position = new KbVector(point.position.x, 0, point.position.y);\n // point.direction = new KbVector(point.direction.x, 0, point.direction.y);\n // if (point.direction2) {\n // point.direction2 = new KbVector(point.direction2.x, 0, point.direction2.y);\n // }\n // } else if (node.normal === eAxis.x) {\n // point.position = new KbVector(0, point.position.y, point.position.x);\n // point.direction = new KbVector(0, point.direction.y, point.direction.x);\n // if (point.direction2) {\n // point.direction2 = new KbVector(0, point.direction2.y, point.direction2.x);\n // }\n // }\n // });\n // });\n // }\n static calculateScaleDivider(viewbox) {\n if (viewbox) {\n const width = viewbox[2];\n const height = viewbox[3];\n if (width > height) {\n return width;\n }\n else {\n return height;\n }\n }\n else {\n return 1;\n }\n }\n static addNewPath(node, pathType) {\n const nodeManager = node._manager;\n if (!nodeManager) {\n throw Error('no node manager');\n }\n const newPath = nodeManager.create(Token.SketchPath);\n if (newPath) {\n newPath.name = 'Path';\n newPath._expanded = true;\n if (pathType) {\n newPath.type = pathType;\n }\n node.paths.push(newPath);\n }\n return newPath;\n }\n static createRectangle(node, element, viewbox) {\n const nodeManager = node._manager;\n if (!nodeManager) {\n throw Error('no node manager');\n }\n const scaleDivider = SvgConverter.calculateScaleDivider(viewbox);\n if (!element.properties) {\n return;\n }\n const x = element.properties.x / scaleDivider;\n const y = (element.properties.y / scaleDivider) * -1;\n const rx = element.properties.rx / scaleDivider;\n const ry = (element.properties.ry / scaleDivider) * -1;\n const width = element.properties.width / scaleDivider;\n const height = (element.properties.height / scaleDivider) * -1;\n let newPath = SvgConverter.addNewPath(node, exports.eSketchPathType.points);\n if (rx && ry) {\n const newControlPoint = nodeManager.create(Token.SketchControlPoint);\n newControlPoint.position = projectOnNodeAxis(node, x, y + ry);\n newPath.controlPoints.push(newControlPoint);\n const newControlPoint2 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint2.position = projectOnNodeAxis(node, x, y + height - ry);\n newPath.controlPoints.push(newControlPoint2);\n newPath = SvgConverter.addNewPath(node, exports.eSketchPathType.quadraticBezier);\n const newControlPoint3 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint3.position = projectOnNodeAxis(node, x, y + height - ry);\n newControlPoint3.direction = projectOnNodeAxis(node, 0, ry);\n newPath.controlPoints.push(newControlPoint3);\n const newControlPoint4 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint4.position = projectOnNodeAxis(node, x + rx, y + height);\n newPath.controlPoints.push(newControlPoint4);\n newPath = SvgConverter.addNewPath(node, exports.eSketchPathType.points);\n const newControlPoint5 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint5.position = projectOnNodeAxis(node, x + rx, y + height);\n newPath.controlPoints.push(newControlPoint5);\n const newControlPoint6 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint6.position = projectOnNodeAxis(node, width + x - rx, y + height);\n newPath.controlPoints.push(newControlPoint6);\n newPath = SvgConverter.addNewPath(node, exports.eSketchPathType.quadraticBezier);\n const newControlPoint7 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint7.position = projectOnNodeAxis(node, width + x - rx, y + height);\n newControlPoint7.direction = projectOnNodeAxis(node, -ry, 0);\n newPath.controlPoints.push(newControlPoint7);\n const newControlPoint8 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint8.position = projectOnNodeAxis(node, width + x, y + height - ry);\n newPath.controlPoints.push(newControlPoint8);\n newPath = SvgConverter.addNewPath(node, exports.eSketchPathType.points);\n const newControlPoint9 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint9.position = projectOnNodeAxis(node, width + x, y + height - ry);\n newPath.controlPoints.push(newControlPoint9);\n const newControlPoint10 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint10.position = projectOnNodeAxis(node, width + x, y + ry);\n newPath.controlPoints.push(newControlPoint10);\n newPath = SvgConverter.addNewPath(node, exports.eSketchPathType.quadraticBezier);\n const newControlPoint11 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint11.position = projectOnNodeAxis(node, width + x, y + ry);\n newControlPoint11.direction = projectOnNodeAxis(node, 0, -ry);\n newPath.controlPoints.push(newControlPoint11);\n const newControlPoint12 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint12.position = projectOnNodeAxis(node, width + x - rx, y);\n newPath.controlPoints.push(newControlPoint12);\n newPath = SvgConverter.addNewPath(node, exports.eSketchPathType.points);\n const newControlPoint13 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint13.position = projectOnNodeAxis(node, width + x - rx, y);\n newPath.controlPoints.push(newControlPoint13);\n const newControlPoint14 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint14.position = projectOnNodeAxis(node, x + rx, y);\n newPath.controlPoints.push(newControlPoint14);\n newPath = SvgConverter.addNewPath(node, exports.eSketchPathType.quadraticBezier);\n const newControlPoint15 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint15.position = projectOnNodeAxis(node, x + rx, y);\n newControlPoint15.direction = projectOnNodeAxis(node, ry, 0);\n newPath.controlPoints.push(newControlPoint15);\n const newControlPoint16 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint16.position = projectOnNodeAxis(node, x, y + ry);\n newPath.controlPoints.push(newControlPoint16);\n }\n else {\n const newControlPoint = nodeManager.create(Token.SketchControlPoint);\n newControlPoint.position = projectOnNodeAxis(node, x, y);\n newPath.controlPoints.push(newControlPoint);\n const newControlPoint2 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint2.position = projectOnNodeAxis(node, x, y + height);\n newPath.controlPoints.push(newControlPoint2);\n const newControlPoint3 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint3.position = projectOnNodeAxis(node, x + width, y + height);\n newPath.controlPoints.push(newControlPoint3);\n const newControlPoint4 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint4.position = projectOnNodeAxis(node, x + width, y);\n newPath.controlPoints.push(newControlPoint4);\n const newControlPoint5 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint5.position = projectOnNodeAxis(node, x, y);\n newPath.controlPoints.push(newControlPoint5);\n }\n }\n static createCircle(node, element, viewbox) {\n const nodeManager = node._manager;\n if (!nodeManager) {\n throw Error('no node manager');\n }\n const scaleDivider = SvgConverter.calculateScaleDivider(viewbox);\n if (element.properties) {\n const cx = element.properties.cx / scaleDivider;\n const cy = (element.properties.cy / scaleDivider) * -1;\n const radius = element.properties.r / scaleDivider;\n const newPath = SvgConverter.addNewPath(node, exports.eSketchPathType.centerRadiusCircle);\n const newControlPoint = nodeManager.create(Token.SketchControlPoint);\n newControlPoint.position = projectOnNodeAxis(node, cx, cy);\n newControlPoint.direction = new exports.KbVector(0, 0, 1);\n newPath.controlPoints.push(newControlPoint);\n const newControlPoint2 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint2.position = projectOnNodeAxis(node, cx + radius, cy);\n newPath.controlPoints.push(newControlPoint2);\n }\n }\n static createLine(node, element, viewbox) {\n const nodeManager = node._manager;\n if (!nodeManager) {\n throw Error('no node manager');\n }\n const scaleDivider = SvgConverter.calculateScaleDivider(viewbox);\n if (element.properties) {\n const newPath = SvgConverter.addNewPath(node, exports.eSketchPathType.points);\n const x1 = element.properties.x1 / scaleDivider;\n const y1 = (element.properties.y1 / scaleDivider) * -1;\n const x2 = element.properties.x2 / scaleDivider;\n const y2 = (element.properties.y2 / scaleDivider) * -1;\n const newControlPoint = nodeManager.create(Token.SketchControlPoint);\n newControlPoint.position = projectOnNodeAxis(node, x1, y1);\n newPath.controlPoints.push(newControlPoint);\n const newControlPoint2 = nodeManager.create(Token.SketchControlPoint);\n newControlPoint2.position = projectOnNodeAxis(node, x2, y2);\n newPath.controlPoints.push(newControlPoint2);\n }\n }\n static createPolyLine(node, element, viewbox, closePath = false) {\n const nodeManager = node._manager;\n if (!nodeManager) {\n throw Error('no node manager');\n }\n const scaleDivider = SvgConverter.calculateScaleDivider(viewbox);\n if (element.properties) {\n const newPath = SvgConverter.addNewPath(node, exports.eSketchPathType.points);\n const points = element.properties.points.replace(',', ' ').split(' ');\n for (let i = 0; i < points.length; i = i + 2) {\n const newControlPoint = nodeManager.create(Token.SketchControlPoint);\n newControlPoint.position = new exports.KbVector(parseInt(points[i]) / scaleDivider, (parseInt(points[i + 1]) / scaleDivider) * -1, 0);\n newPath.controlPoints.push(newControlPoint);\n }\n if (closePath) {\n const newControlPoint = nodeManager.create(Token.SketchControlPoint);\n newControlPoint.position = new exports.KbVector(parseInt(points[0]) / scaleDivider, (parseInt(points[1]) / scaleDivider) * -1, 0);\n newPath.controlPoints.push(newControlPoint);\n }\n }\n }\n }\n function nameGeneratedPaths(node) {\n let counter = 0;\n node.paths.forEach(path => {\n if (counter > 0) {\n path.name = 'Path ' + counter.toString();\n }\n counter++;\n });\n }\n function getPaletteColorStrength(color) {\n const saturation = Math.max(color.r, color.g, color.b) - Math.min(color.r, color.g, color.b);\n return color.a * 255 + saturation;\n }\n\n class CompatibilityOptions$1 {\n }\n /**\n * Creates the VertexData for a Ribbon\n * @param options an object used to set the following optional parameters for the ribbon, required but can be empty\n * * pathArray array of paths, each of which an array of successive Vector3\n * * closeArray creates a seam between the first and the last paths of the pathArray, optional, default false\n * * closePath creates a seam between the first and the last points of each path of the path array, optional, default false\n * * offset a positive integer, only used when pathArray contains a single path (offset = 10 means the point 1 is joined to the point 11), default rounded half size of the pathArray length\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\n * * invertUV swaps in the U and V coordinates when applying a texture, optional, default false\n * * uvs a linear array, of length 2 * number of vertices, of custom UV values, optional\n * * colors a linear array, of length 4 * number of vertices, of custom color values, optional\n * @returns the VertexData of the ribbon\n */\n function CreateRibbonVertexData$1(options) {\n let pathArray = options.pathArray;\n const closeArray = options.closeArray || false;\n const closePath = options.closePath || false;\n const invertUV = options.invertUV || false;\n const defaultOffset = Math.floor(pathArray[0].length / 2);\n let offset = options.offset || defaultOffset;\n offset = offset > defaultOffset ? defaultOffset : Math.floor(offset); // offset max allowed : defaultOffset\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\n const customUV = options.uvs;\n const customColors = options.colors;\n const positions = [];\n const indices = [];\n const normals = [];\n const uvs = [];\n const us = []; // us[path_id] = [uDist1, uDist2, uDist3 ... ] distances between points on path path_id\n const vs = []; // vs[i] = [vDist1, vDist2, vDist3, ... ] distances between points i of consecutive paths from pathArray\n const uTotalDistance = []; // uTotalDistance[p] : total distance of path p\n const vTotalDistance = []; // vTotalDistance[i] : total distance between points i of first and last path from pathArray\n let minlg; // minimal length among all paths from pathArray\n const lg = []; // array of path lengths : nb of vertex per path\n const idx = []; // array of path indexes : index of each path (first vertex) in the total vertex number\n let p; // path iterator\n let i; // point iterator\n let j; // point iterator\n // if single path in pathArray\n if (pathArray.length < 2) {\n const ar1 = [];\n const ar2 = [];\n for (i = 0; i < pathArray[0].length - offset; i++) {\n ar1.push(pathArray[0][i]);\n ar2.push(pathArray[0][i + offset]);\n }\n pathArray = [ar1, ar2];\n }\n // positions and horizontal distances (u)\n let idc = 0;\n const closePathCorr = closePath ? 1 : 0; // the final index will be +1 if closePath\n const closeArrayCorr = closeArray ? 1 : 0;\n let path;\n let l;\n minlg = pathArray[0].length;\n let vectlg;\n let dist;\n for (p = 0; p < pathArray.length + closeArrayCorr; p++) {\n uTotalDistance[p] = 0;\n us[p] = [0];\n path = p === pathArray.length ? pathArray[0] : pathArray[p];\n l = path.length;\n minlg = minlg < l ? minlg : l;\n j = 0;\n while (j < l) {\n positions.push(path[j].x, path[j].y, path[j].z);\n if (j > 0) {\n vectlg = path[j].subtract(path[j - 1]).length();\n dist = vectlg + uTotalDistance[p];\n us[p].push(dist);\n uTotalDistance[p] = dist;\n }\n j++;\n }\n if (closePath) {\n // an extra hidden vertex is added in the \"positions\" array\n j--;\n positions.push(path[0].x, path[0].y, path[0].z);\n vectlg = path[j].subtract(path[0]).length();\n dist = vectlg + uTotalDistance[p];\n us[p].push(dist);\n uTotalDistance[p] = dist;\n }\n lg[p] = l + closePathCorr;\n idx[p] = idc;\n idc += l + closePathCorr;\n }\n // vertical distances (v)\n let path1;\n let path2;\n let vertex1 = null;\n let vertex2 = null;\n for (i = 0; i < minlg + closePathCorr; i++) {\n vTotalDistance[i] = 0;\n vs[i] = [0];\n for (p = 0; p < pathArray.length - 1 + closeArrayCorr; p++) {\n path1 = pathArray[p];\n path2 = p === pathArray.length - 1 ? pathArray[0] : pathArray[p + 1];\n if (i === minlg) {\n // closePath\n vertex1 = path1[0];\n vertex2 = path2[0];\n }\n else {\n vertex1 = path1[i];\n vertex2 = path2[i];\n }\n vectlg = vertex2.subtract(vertex1).length();\n dist = vectlg + vTotalDistance[i];\n vs[i].push(dist);\n vTotalDistance[i] = dist;\n }\n }\n // uvs\n let u;\n let v;\n if (customUV) {\n for (p = 0; p < customUV.length; p++) {\n uvs.push(customUV[p].x, CompatibilityOptions$1.useOpenGLOrientationForUV ? 1.0 - customUV[p].y : customUV[p].y);\n }\n }\n else {\n for (p = 0; p < pathArray.length + closeArrayCorr; p++) {\n for (i = 0; i < minlg + closePathCorr; i++) {\n u = uTotalDistance[p] != 0.0 ? us[p][i] / uTotalDistance[p] : 0.0;\n v = vTotalDistance[i] != 0.0 ? vs[i][p] / vTotalDistance[i] : 0.0;\n if (invertUV) {\n uvs.push(v, u);\n }\n else {\n uvs.push(u, CompatibilityOptions$1.useOpenGLOrientationForUV ? 1.0 - v : v);\n }\n }\n }\n }\n // indices\n p = 0; // path index\n let pi = 0; // positions array index\n let l1 = lg[p] - 1; // path1 length\n let l2 = lg[p + 1] - 1; // path2 length\n let min = l1 < l2 ? l1 : l2; // current path stop index\n let shft = idx[1] - idx[0]; // shift\n const path1nb = lg.length - 1; // number of path1 to iterate on\n while (pi <= min && p < path1nb) {\n // stay under min and don't go over next to last path\n // draw two triangles between path1 (p1) and path2 (p2) : (p1.pi, p2.pi, p1.pi+1) and (p2.pi+1, p1.pi+1, p2.pi) clockwise\n indices.push(pi, pi + shft, pi + 1);\n indices.push(pi + shft + 1, pi + 1, pi + shft);\n pi += 1;\n if (pi === min) {\n // if end of one of two consecutive paths reached, go to next existing path\n p++;\n shft = idx[p + 1] - idx[p];\n l1 = lg[p] - 1;\n l2 = lg[p + 1] - 1;\n pi = idx[p];\n min = l1 < l2 ? l1 + pi : l2 + pi;\n }\n }\n // normals\n VertexData.ComputeNormals(positions, indices, normals);\n if (closePath) {\n // update both the first and last vertex normals to their average value\n let indexFirst = 0;\n let indexLast = 0;\n for (p = 0; p < pathArray.length; p++) {\n indexFirst = idx[p] * 3;\n if (p + 1 < pathArray.length) {\n indexLast = (idx[p + 1] - 1) * 3;\n }\n else {\n indexLast = normals.length - 3;\n }\n normals[indexFirst] = (normals[indexFirst] + normals[indexLast]) * 0.5;\n normals[indexFirst + 1] = (normals[indexFirst + 1] + normals[indexLast + 1]) * 0.5;\n normals[indexFirst + 2] = (normals[indexFirst + 2] + normals[indexLast + 2]) * 0.5;\n const l = Math.sqrt(normals[indexFirst] * normals[indexFirst] +\n normals[indexFirst + 1] * normals[indexFirst + 1] +\n normals[indexFirst + 2] * normals[indexFirst + 2]);\n normals[indexFirst] /= l;\n normals[indexFirst + 1] /= l;\n normals[indexFirst + 2] /= l;\n normals[indexLast] = normals[indexFirst];\n normals[indexLast + 1] = normals[indexFirst + 1];\n normals[indexLast + 2] = normals[indexFirst + 2];\n }\n }\n if (closeArray) {\n let indexFirst = idx[0] * 3;\n let indexLast = idx[pathArray.length] * 3;\n for (i = 0; i < minlg + closePathCorr; i++) {\n normals[indexFirst] = (normals[indexFirst] + normals[indexLast]) * 0.5;\n normals[indexFirst + 1] = (normals[indexFirst + 1] + normals[indexLast + 1]) * 0.5;\n normals[indexFirst + 2] = (normals[indexFirst + 2] + normals[indexLast + 2]) * 0.5;\n const l = Math.sqrt(normals[indexFirst] * normals[indexFirst] +\n normals[indexFirst + 1] * normals[indexFirst + 1] +\n normals[indexFirst + 2] * normals[indexFirst + 2]);\n normals[indexFirst] /= l;\n normals[indexFirst + 1] /= l;\n normals[indexFirst + 2] /= l;\n normals[indexLast] = normals[indexFirst];\n normals[indexLast + 1] = normals[indexFirst + 1];\n normals[indexLast + 2] = normals[indexFirst + 2];\n indexFirst += 3;\n indexLast += 3;\n }\n }\n // sides\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\n // Colors\n let colors = null;\n if (customColors) {\n colors = new Float32Array(customColors.length * 4);\n for (let c = 0; c < customColors.length; c++) {\n colors[c * 4] = customColors[c].r;\n colors[c * 4 + 1] = customColors[c].g;\n colors[c * 4 + 2] = customColors[c].b;\n colors[c * 4 + 3] = customColors[c].a;\n }\n }\n // Result\n const vertexData = new VertexData();\n const positions32 = new Float32Array(positions);\n const normals32 = new Float32Array(normals);\n const uvs32 = new Float32Array(uvs);\n vertexData.indices = indices;\n vertexData.positions = positions32;\n vertexData.normals = normals32;\n vertexData.uvs = uvs32;\n if (colors) {\n vertexData.set(colors, VertexBuffer.ColorKind);\n }\n if (closePath) {\n vertexData._idx = idx;\n }\n return vertexData;\n }\n /**\n * Creates a ribbon mesh. The ribbon is a parametric shape. It has no predefined shape. Its final shape will depend on the input parameters\n * * The parameter `pathArray` is a required array of paths, what are each an array of successive Vector3. The pathArray parameter depicts the ribbon geometry\n * * The parameter `closeArray` (boolean, default false) creates a seam between the first and the last paths of the path array\n * * The parameter `closePath` (boolean, default false) creates a seam between the first and the last points of each path of the path array\n * * The parameter `offset` (positive integer, default : rounded half size of the pathArray length), is taken in account only if the `pathArray` is containing a single path\n * * It's the offset to join the points from the same path. Ex : offset = 10 means the point 1 is joined to the point 11\n * * The optional parameter `instance` is an instance of an existing Ribbon object to be updated with the passed `pathArray` parameter : https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#ribbon\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\n * * The optional parameter `invertUV` (boolean, default false) swaps in the geometry the U and V coordinates to apply a texture\n * * The parameter `uvs` is an optional flat array of `Vector2` to update/set each ribbon vertex with its own custom UV values instead of the computed ones\n * * The parameters `colors` is an optional flat array of `Color4` to set/update each ribbon vertex with its own custom color values\n * * Note that if you use the parameters `uvs` or `colors`, the passed arrays must be populated with the right number of elements, it is to say the number of ribbon vertices. Remember that if you set `closePath` to `true`, there's one extra vertex per path in the geometry\n * * Moreover, you can use the parameter `color` with `instance` (to update the ribbon), only if you previously used it at creation time\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\n * @param name defines the name of the mesh\n * @param options defines the options used to create the mesh\n * @param scene defines the hosting scene\n * @returns the ribbon mesh\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param/ribbon_extra\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param\n */\n function CreateMiteredRibbon(name, options, scene = null) {\n const pathArray = options.pathArray;\n const closeArray = options.closeArray;\n const closePath = options.closePath;\n const cap = options.cap === 0 ? 0 : options.cap || Mesh.NO_CAP;\n const sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\n const instance = options.instance;\n const updatable = options.updatable;\n if (instance) {\n // existing ribbon instance update\n // positionFunction : ribbon case\n // only pathArray and sideOrientation parameters are taken into account for positions update\n const minimum = TmpVectors.Vector3[0].setAll(Number.MAX_VALUE);\n const maximum = TmpVectors.Vector3[1].setAll(-Number.MAX_VALUE);\n const positionFunction = (positions) => {\n let minlg = pathArray[0].length;\n const mesh = instance;\n let i = 0;\n const ns = mesh._originalBuilderSideOrientation === Mesh.DOUBLESIDE ? 2 : 1;\n for (let si = 1; si <= ns; ++si) {\n for (let p = 0; p < pathArray.length; ++p) {\n const path = pathArray[p];\n const l = path.length;\n minlg = minlg < l ? minlg : l;\n for (let j = 0; j < minlg; ++j) {\n const pathPoint = path[j];\n positions[i] = pathPoint.x;\n positions[i + 1] = pathPoint.y;\n positions[i + 2] = pathPoint.z;\n minimum.minimizeInPlaceFromFloats(pathPoint.x, pathPoint.y, pathPoint.z);\n maximum.maximizeInPlaceFromFloats(pathPoint.x, pathPoint.y, pathPoint.z);\n i += 3;\n }\n if (mesh._creationDataStorage && mesh._creationDataStorage.closePath) {\n const pathPoint = path[0];\n positions[i] = pathPoint.x;\n positions[i + 1] = pathPoint.y;\n positions[i + 2] = pathPoint.z;\n i += 3;\n }\n }\n }\n };\n const positions = instance.getVerticesData(VertexBuffer.PositionKind);\n positionFunction(positions);\n if (instance.hasBoundingInfo) {\n instance.getBoundingInfo().reConstruct(minimum, maximum, instance._worldMatrix);\n }\n else {\n instance.buildBoundingInfo(minimum, maximum, instance._worldMatrix);\n }\n instance.updateVerticesData(VertexBuffer.PositionKind, positions, false, false);\n if (options.colors) {\n const colors = instance.getVerticesData(VertexBuffer.ColorKind);\n for (let c = 0, colorIndex = 0; c < options.colors.length; c++, colorIndex += 4) {\n const color = options.colors[c];\n colors[colorIndex] = color.r;\n colors[colorIndex + 1] = color.g;\n colors[colorIndex + 2] = color.b;\n colors[colorIndex + 3] = color.a;\n }\n instance.updateVerticesData(VertexBuffer.ColorKind, colors, false, false);\n }\n if (options.uvs) {\n const uvs = instance.getVerticesData(VertexBuffer.UVKind);\n for (let i = 0; i < options.uvs.length; i++) {\n uvs[i * 2] = options.uvs[i].x;\n uvs[i * 2 + 1] = CompatibilityOptions$1.useOpenGLOrientationForUV\n ? 1.0 - options.uvs[i].y\n : options.uvs[i].y;\n }\n instance.updateVerticesData(VertexBuffer.UVKind, uvs, false, false);\n }\n if (!instance.areNormalsFrozen || instance.isFacetDataEnabled) {\n const indices = instance.getIndices();\n const normals = instance.getVerticesData(VertexBuffer.NormalKind);\n const params = instance.isFacetDataEnabled ? instance.getFacetDataParameters() : null;\n VertexData.ComputeNormals(positions, indices, normals, params);\n if (instance._creationDataStorage && instance._creationDataStorage.closePath) {\n let indexFirst = 0;\n let indexLast = 0;\n for (let p = 0; p < pathArray.length; p++) {\n indexFirst = instance._creationDataStorage.idx[p] * 3;\n if (p + 1 < pathArray.length) {\n indexLast = (instance._creationDataStorage.idx[p + 1] - 1) * 3;\n }\n else {\n indexLast = normals.length - 3;\n }\n normals[indexFirst] = (normals[indexFirst] + normals[indexLast]) * 0.5;\n normals[indexFirst + 1] = (normals[indexFirst + 1] + normals[indexLast + 1]) * 0.5;\n normals[indexFirst + 2] = (normals[indexFirst + 2] + normals[indexLast + 2]) * 0.5;\n normals[indexLast] = normals[indexFirst];\n normals[indexLast + 1] = normals[indexFirst + 1];\n normals[indexLast + 2] = normals[indexFirst + 2];\n }\n }\n if (!instance.areNormalsFrozen) {\n instance.updateVerticesData(VertexBuffer.NormalKind, normals, false, false);\n }\n }\n return instance;\n }\n else {\n // new ribbon creation\n const ribbon = new Mesh(name, scene);\n ribbon._originalBuilderSideOrientation = sideOrientation;\n ribbon._creationDataStorage = new _CreationDataStorage();\n const vertexData = CreateRibbonVertexData$1(options);\n if (closePath) {\n ribbon._creationDataStorage.idx = vertexData._idx;\n }\n ribbon._creationDataStorage.closePath = closePath;\n ribbon._creationDataStorage.closeArray = closeArray;\n vertexData.applyToMesh(ribbon, updatable);\n createLineCaps(pathArray, cap, name, ribbon, vertexData);\n return ribbon;\n }\n }\n function createLineCaps(pathArray, cap, featureId, ribbon, vertexData) {\n const startCap = [];\n const endCap = [];\n let startCapVectors = [];\n let endCapVectors = [];\n const resultVectors = [];\n const resultCaps = [];\n pathArray.forEach(path => {\n if (path[0]) {\n const vector = path[0];\n startCapVectors.push(path[0]);\n startCap.push(vector.x, vector.y, vector.z);\n }\n if (path[path.length - 1]) {\n const vector = path[path.length - 1];\n endCapVectors.push(path[path.length - 1]);\n endCap.push(vector.x, vector.y, vector.z);\n }\n });\n //check to see if the handedness of the endCap is correct. Otherwise, flip it\n if (startCapVectors.length > 0) {\n const faceNormal = calculateFaceNormal(startCapVectors);\n const firstPath = pathArray[0];\n if (firstPath.length >= 2) {\n const firstPoint = firstPath[0];\n const secondPoint = firstPath[1];\n const sketchNativeNormal = secondPoint.subtract(firstPoint);\n // If the rotation handedness is opposite the primary sketch vector, then flip the sketch path\n if (Vector3.Dot(faceNormal, sketchNativeNormal) < 0) {\n startCapVectors = reverseHandednessOfPositions(startCapVectors);\n }\n }\n }\n //check to see if the handedness of the endCap is correct. Otherwise, flip it\n if (endCapVectors.length > 0) {\n const faceNormal = calculateFaceNormal(endCapVectors);\n const lastPath = pathArray[pathArray.length - 1];\n if (lastPath.length >= 2) {\n const lastPoint = lastPath[lastPath.length - 1];\n const secondLastPoint = lastPath[lastPath.length - 2];\n const sketchNativeNormal = lastPoint.subtract(secondLastPoint);\n // If the rotation handedness is opposite the primary sketch vector, then flip the sketch path\n if (Vector3.Dot(faceNormal, sketchNativeNormal) < 0) {\n endCapVectors = reverseHandednessOfPositions(endCapVectors);\n }\n }\n }\n if (cap === Mesh.CAP_START) {\n const startCapIndices = earcut$1(startCap, [], 3);\n resultCaps.push(startCapIndices);\n resultVectors.push(startCapVectors, []);\n }\n else if (cap === Mesh.CAP_END) {\n const endCapIndices = earcut$1(endCap, [], 3);\n resultCaps.push(endCapIndices);\n resultVectors.push([], endCapVectors);\n }\n else if (cap === Mesh.CAP_ALL) {\n const startCapIndices = earcut$1(startCap, [], 3);\n resultCaps.push(startCapIndices);\n const endCapIndices = earcut$1(endCap, [], 3);\n //offset the endCapIndices otherwise they are the same as the startCapIndices\n for (let i = 0; i < endCapIndices.length; i++) {\n endCapIndices[i] = endCapIndices[i] + startCapVectors.length;\n }\n resultCaps.push(endCapIndices);\n resultVectors.push(startCapVectors, endCapVectors);\n }\n const newVertexData = combineVertexDataWithCaps(vertexData, resultCaps, resultVectors);\n newVertexData.applyToMesh(ribbon, true);\n return resultCaps;\n }\n function combineVertexDataWithCaps(baseVertexData, capMeshIndices, capMeshPositions) {\n var _a, _b, _c, _d;\n const totalBasePositions = ((_a = baseVertexData.positions) === null || _a === void 0 ? void 0 : _a.length) || 0;\n const totalBaseNormals = ((_b = baseVertexData.normals) === null || _b === void 0 ? void 0 : _b.length) || 0;\n const totalBaseUvs = ((_c = baseVertexData.uvs) === null || _c === void 0 ? void 0 : _c.length) || 0;\n const totalBaseIndices = ((_d = baseVertexData.indices) === null || _d === void 0 ? void 0 : _d.length) || 0;\n const newPositions = [];\n const newNormals = [];\n const newUvs = [];\n const newIndices = [];\n const startCapPositions = capMeshPositions[0];\n const endCapPositions = capMeshPositions[1];\n capMeshIndices.forEach(meshIndices => {\n meshIndices.forEach(index => {\n const newIndex = index + totalBasePositions / 3;\n newIndices.push(newIndex);\n });\n });\n if (startCapPositions && startCapPositions.length !== 0) {\n const normal = calculateFaceNormal(startCapPositions);\n startCapPositions.forEach(position => {\n newPositions.push(position.x, position.y, position.z);\n newNormals.push(normal.x, normal.y, normal.z);\n });\n const generatedUvs = generateUvsForCaps(startCapPositions);\n newUvs.push(...generatedUvs);\n }\n if (endCapPositions && endCapPositions.length !== 0) {\n const normal = calculateFaceNormal(endCapPositions);\n endCapPositions.forEach(position => {\n newPositions.push(position.x, position.y, position.z);\n newNormals.push(normal.x, normal.y, normal.z);\n });\n const generatedUvs = generateUvsForCaps(endCapPositions);\n newUvs.push(...generatedUvs);\n }\n const newVertexData = new VertexData();\n newVertexData.positions = newPositions;\n newVertexData.normals = newNormals;\n newVertexData.indices = newIndices;\n const combinedPositionsArray = new Float32Array(totalBasePositions + newPositions.length);\n const combinedNormalsArray = new Float32Array(totalBaseNormals + newNormals.length);\n const combinedUvsArray = new Float32Array((combinedPositionsArray.length / 3) * 2);\n const combinedIndicesArray = new Uint16Array(totalBaseIndices + newIndices.length);\n if (baseVertexData.positions) {\n combinedPositionsArray.set(baseVertexData.positions);\n combinedPositionsArray.set(newPositions, totalBasePositions);\n baseVertexData.positions = combinedPositionsArray;\n }\n if (baseVertexData.normals) {\n combinedNormalsArray.set(baseVertexData.normals);\n combinedNormalsArray.set(newNormals, totalBaseNormals);\n baseVertexData.normals = combinedNormalsArray;\n }\n if (baseVertexData.uvs) {\n combinedUvsArray.set(baseVertexData.uvs);\n combinedUvsArray.set(newUvs, totalBaseUvs);\n baseVertexData.uvs = combinedUvsArray;\n }\n if (baseVertexData.indices) {\n combinedIndicesArray.set(baseVertexData.indices);\n combinedIndicesArray.set(newIndices, totalBaseIndices);\n baseVertexData.indices = combinedIndicesArray;\n }\n return baseVertexData;\n }\n function reverseHandednessOfPositions(positionsArray) {\n const reversedArray = [];\n positionsArray.forEach(position => {\n reversedArray.unshift(position);\n });\n return reversedArray;\n }\n function generateUvsForCaps(positions) {\n const newUvs = [];\n const operationVectors = findDistinctVectors(positions);\n if (operationVectors) {\n const vector1 = operationVectors[0];\n const vector2 = operationVectors[1];\n const vector3 = operationVectors[2];\n //xp1 is the capNormal\n if (vector1 && vector2 && vector3) {\n const xp1 = vector1.cross(vector2.subtract(vector3));\n const xp2 = vector1.subtract(vector2).cross(xp1);\n const normalizedXp3 = xp1.cross(xp2).normalize();\n const normalizedXp2 = xp2.normalize();\n positions.forEach(position => {\n const u = Vector3.Dot(position, normalizedXp2);\n const v = Vector3.Dot(position, normalizedXp3);\n newUvs.push(u, v);\n });\n }\n }\n else {\n positions.forEach(position => {\n newUvs.push(0, 0);\n });\n }\n return newUvs;\n }\n function findDistinctVectors(vectors) {\n let vector1 = Vector3.Zero();\n let vector2 = Vector3.Zero();\n let vector3 = Vector3.Zero();\n for (let i = 0; i < vectors.length; i++) {\n let isZeroVector = false;\n if (vectors[i].x === 0 && vectors[i].y === 0 && vectors[i].z === 0) {\n isZeroVector = true;\n }\n if (!isZeroVector) {\n vector1 = vectors[i];\n break;\n }\n }\n for (let i = 0; i < vectors.length; i++) {\n let isZeroVector = false;\n if (vectors[i].x === 0 && vectors[i].y === 0 && vectors[i].z === 0) {\n isZeroVector = true;\n }\n if (!isZeroVector && vectors[i] !== vector1) {\n vector2 = vectors[i];\n break;\n }\n }\n for (let i = 0; i < vectors.length; i++) {\n if (vectors[i] !== vector2 && vectors[i] !== vector1) {\n vector3 = vectors[i];\n break;\n }\n }\n return [vector1, vector2, vector3];\n }\n VertexData.CreateRibbon = CreateRibbonVertexData$1;\n Mesh.CreateRibbon = (name, pathArray, closeArray = false, closePath, offset, scene, updatable = false, sideOrientation, instance) => {\n return CreateMiteredRibbon(name, {\n pathArray: pathArray,\n closeArray: closeArray,\n closePath: closePath,\n offset: offset,\n updatable: updatable,\n sideOrientation: sideOrientation,\n instance: instance,\n }, scene);\n };\n\n //const capMap: Map = new Map();\n class SweepRenderer extends FeatureRenderer {\n constructor() {\n super(...arguments);\n this.modelToken = Token.SweepFeature;\n }\n updateMeshGeometry(feature, vertexData, vertexDataChanges) {\n var _a;\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\n let parentMesh;\n if (feature._parent) {\n parentMesh = this.viewer.getRendererNode(feature._parent);\n }\n let pathPoints = [];\n let targetPathNode;\n let pathMesh;\n if (feature.path) {\n targetPathNode = (_a = feature._manager) === null || _a === void 0 ? void 0 : _a.getById(feature.path);\n }\n if (targetPathNode) {\n pathMesh = this.viewer.getRendererNode(targetPathNode);\n const pathVertexData = SubmeshVertexData.ExtractFromMesh(pathMesh);\n const { segments } = parseSketchGeometry(pathVertexData);\n if (segments.length > 0) {\n pathPoints = segments[0];\n }\n }\n if (parentMesh &&\n targetPathNode &&\n pathMesh &&\n newVertexData.positions &&\n newVertexData.positions.length > 0 &&\n pathPoints.length > 1) {\n const { segments } = parseSketchGeometry(newVertexData);\n const extrudedMeshes = [];\n for (let i = 0; i < segments.length; i++) {\n const shape = segments[i];\n // Transform the path and the profile sketch into world space so we can combine them without worrying about local transforms\n const pathTransform = pathMesh.computeWorldMatrix();\n const sketchTransform = parentMesh.computeWorldMatrix();\n for (let i = 0; i < pathPoints.length; i++) {\n Vector3.TransformCoordinatesToRef(pathPoints[i], pathTransform, pathPoints[i]);\n }\n for (let i = 0; i < shape.length; i++) {\n Vector3.TransformCoordinatesToRef(shape[i], sketchTransform, shape[i]);\n }\n // Determine the vector to use as the primary direction of the extrusion\n const pathPrimaryVector = Vector3.Zero();\n let j = 0;\n while (pathPrimaryVector.length() === 0) {\n j++;\n pathPrimaryVector.copyFrom(pathPoints[j]).subtractInPlace(pathPoints[0]);\n }\n // Calculate the matrix to move the sketch path primary direction to be perpindicular to the XY plane for consistent behavior\n const rotToZ = rotationToVector(new Vector3(0, 0, 1), pathPrimaryVector);\n const pathTransformToZ = new Matrix();\n const pathTransformToZInv = new Matrix();\n rotToZ.toRotationMatrix(pathTransformToZ);\n Matrix.Compose(Vector3.One(), Quaternion.Identity(), pathPoints[0].scale(-1)).multiplyToRef(pathTransformToZ, pathTransformToZ);\n pathTransformToZ.invertToRef(pathTransformToZInv);\n // Transform both the path and the extrusion shape with the same matrix so the\n // shape and the path are positioned the same as before in relation to each other\n // but oriented to be extruded out of the XY plane, in the Z direction\n for (let i = 0; i < pathPoints.length; i++) {\n Vector3.TransformCoordinatesToRef(pathPoints[i], pathTransformToZ, pathPoints[i]);\n }\n for (let i = 0; i < shape.length; i++) {\n Vector3.TransformCoordinatesToRef(shape[i], pathTransformToZ, shape[i]);\n // Project the extrude shape onto the XY plane so that it's perfecly perpindicular to the sweep path\n shape[i].set(shape[i].x, shape[i].y, 0);\n }\n const extrudeOptions = {\n shape,\n path: pathPoints,\n updatable: false,\n cap: Mesh.NO_CAP,\n closeShape: false,\n // firstNormal: sketchNormal, this property is added in a future version of babylon\n };\n if (feature.capEnd && feature.capStart) {\n extrudeOptions.cap = Mesh.CAP_ALL;\n extrudeOptions.closeShape = true;\n }\n else if (feature.capStart) {\n extrudeOptions.cap = Mesh.CAP_START;\n extrudeOptions.closeShape = true;\n }\n else if (feature.capEnd) {\n extrudeOptions.cap = Mesh.CAP_END;\n extrudeOptions.closeShape = true;\n }\n // Perform the extrusion of the shape defined along the XY plane, to the path oriented to coincide with the Z+ axis\n let extruded;\n if (feature.mitredExtrusion) {\n extrudeOptions.closeShape = false;\n extruded = this.mitredExtrude(feature._dynamicId, extrudeOptions, this.scene);\n }\n else {\n extruded = MeshBuilder.ExtrudeShape(feature._dynamicId, extrudeOptions, this.scene);\n }\n // Rotate the resulting extruded shape to match the orientation of the original shape profile\n sketchTransform.invert();\n pathTransformToZInv.multiplyToRef(sketchTransform, pathTransformToZInv);\n extruded.bakeTransformIntoVertices(pathTransformToZInv);\n extrudedMeshes.push(extruded);\n }\n const extrudedAll = Mesh.MergeMeshes(extrudedMeshes, true);\n if (extrudedAll) {\n applyExtrusion(newVertexData, extrudedAll, false);\n extrudedAll.dispose();\n }\n }\n return newVertexData;\n }\n update3dInternal() { }\n mitredExtrude(featureId, options, scene) {\n const shape = options.shape;\n const path = options.path;\n const closed = options.closeShape || false;\n const nbPoints = path.length;\n const line = Vector3.Zero();\n const nextLine = Vector3.Zero();\n let axisX = Vector3.Zero();\n let axisY = Vector3.Zero();\n let axisZ = Vector3.Zero();\n let nextAxisX = Vector3.Zero();\n let nextAxisY = Vector3.Zero();\n let nextAxisZ = Vector3.Zero();\n let startPoint = Vector3.Zero();\n const nextStartPoint = Vector3.Zero();\n const bisector = Vector3.Zero();\n let distance = 0;\n let ray;\n const allPaths = [];\n // Note that the sketch will always be rotated such that the profile is in the XY plane\n const up = Vector3.Up();\n for (let s = 0; s < shape.length; s++) {\n path[1].subtractToRef(path[0], line);\n axisZ = line.clone().normalize();\n axisX = Vector3.Cross(Vector3.Up(), axisZ).normalize();\n axisY = Vector3.Cross(axisZ, axisX);\n startPoint = path[0].add(axisX.scale(shape[s].x)).add(axisY.scale(shape[s].y));\n const ribbonPath = [startPoint.clone()];\n for (let p = 0; p < nbPoints - 2; p++) {\n path[p + 2].subtractToRef(path[p + 1], nextLine);\n nextAxisZ = nextLine.clone().normalize();\n nextAxisX = Vector3.Cross(up, nextAxisZ).normalize();\n nextAxisY = Vector3.Cross(nextAxisZ, nextAxisX);\n nextAxisZ.subtractToRef(axisZ, bisector);\n const planeParallel = Vector3.Cross(nextAxisZ, axisZ);\n const planeNormal = Vector3.Cross(planeParallel, bisector);\n const plane = Plane.FromPositionAndNormal(path[p + 1], planeNormal);\n ray = new Ray(startPoint, axisZ);\n distance = ray.intersectsPlane(plane) || 0;\n startPoint.addToRef(axisZ.scale(distance), nextStartPoint);\n ribbonPath.push(nextStartPoint.clone());\n axisX = nextAxisX.clone();\n axisY = nextAxisY.clone();\n axisZ = nextAxisZ.clone();\n startPoint = nextStartPoint.clone();\n }\n // Last Point\n if (closed) {\n path[0].subtractToRef(path[nbPoints - 1], nextLine);\n nextAxisZ = nextLine.clone().normalize();\n nextAxisX = Vector3.Cross(up, nextAxisZ).normalize();\n nextAxisY = Vector3.Cross(nextAxisZ, nextAxisX);\n nextAxisZ.subtractToRef(axisZ, bisector);\n const closedPlaneParallel = Vector3.Cross(nextAxisZ, axisZ);\n const closedPlaneNormal = Vector3.Cross(closedPlaneParallel, bisector);\n const closedPlane = Plane.FromPositionAndNormal(path[nbPoints - 1], closedPlaneNormal);\n ray = new Ray(startPoint, axisZ);\n distance = ray.intersectsPlane(closedPlane) || 0;\n startPoint.addToRef(axisZ.scale(distance), nextStartPoint);\n ribbonPath.push(nextStartPoint.clone());\n axisX = nextAxisX.clone();\n axisY = nextAxisY.clone();\n axisZ = nextAxisZ.clone();\n startPoint = nextStartPoint.clone();\n path[1].subtractToRef(path[0], nextLine);\n nextAxisZ = nextLine.clone().normalize();\n nextAxisX = Vector3.Cross(up, nextAxisZ).normalize();\n nextAxisY = Vector3.Cross(nextAxisZ, nextAxisX);\n nextAxisZ.subtractToRef(axisZ, bisector);\n const planeParallel = Vector3.Cross(nextAxisZ, axisZ);\n const planeNormal = Vector3.Cross(planeParallel, bisector);\n const plane = Plane.FromPositionAndNormal(path[0], planeNormal);\n ray = new Ray(startPoint, axisZ);\n distance = ray.intersectsPlane(plane) || 0;\n startPoint.addToRef(axisZ.scale(distance), nextStartPoint);\n ribbonPath.shift();\n ribbonPath.unshift(nextStartPoint.clone());\n }\n else {\n const planeNormal = axisZ;\n const plane = Plane.FromPositionAndNormal(path[nbPoints - 1], planeNormal);\n ray = new Ray(startPoint, axisZ);\n const distance = ray.intersectsPlane(plane) || 0;\n startPoint.addToRef(axisZ.scale(distance), nextStartPoint);\n ribbonPath.push(nextStartPoint.clone());\n }\n allPaths.push(ribbonPath);\n }\n const ribbon = CreateMiteredRibbon(featureId, {\n pathArray: allPaths,\n sideOrientation: Mesh.DOUBLESIDE,\n closeArray: false,\n closePath: false,\n cap: options.cap,\n }, scene);\n return ribbon;\n }\n }\n\n class TransformRenderer extends FeatureRenderer {\n constructor() {\n super(...arguments);\n this.modelToken = Token.TransformFeature;\n }\n updateMeshGeometry(feature, vertexData, vertexDataChanges) {\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\n const parentMesh = feature._parent;\n if (parentMesh && newVertexData.positions && newVertexData.positions.length > 0) {\n const newMeshVertexPositions = newVertexData.positions;\n const indexCount = newMeshVertexPositions.length / 3;\n const pivot = parentMesh.pivot.toVec3();\n const vertexPosition = new Vector3();\n let scaleVector;\n if (!feature.scale.equal({ x: 1, y: 1, z: 1 })) {\n scaleVector = feature.scale.toVec3();\n }\n for (let i = 0; i < indexCount; i++) {\n if (pivot && scaleVector) {\n vertexPosition.copyFromFloats(newMeshVertexPositions[i * 3] - pivot.x, newMeshVertexPositions[i * 3 + 1] - pivot.y, newMeshVertexPositions[i * 3 + 2] - pivot.z);\n if (scaleVector) {\n vertexPosition.multiplyInPlace(scaleVector);\n }\n vertexPosition.addInPlace(pivot);\n newMeshVertexPositions[i * 3] = vertexPosition.x;\n newMeshVertexPositions[i * 3 + 1] = vertexPosition.y;\n newMeshVertexPositions[i * 3 + 2] = vertexPosition.z;\n }\n }\n }\n return newVertexData;\n }\n update3dInternal() { }\n }\n\n class WeldVerticesRenderer extends FeatureRenderer {\n constructor() {\n super(...arguments);\n this.modelToken = Token.WeldVerticesFeature;\n }\n updateMeshGeometry(feature, vertexData, vertexDataChanges) {\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\n const parentMesh = feature._parent;\n const indices = newVertexData.indices;\n let positions = newVertexData.positions;\n let normals = newVertexData.normals;\n let uvs = newVertexData.uvs;\n const positionAverage = [0, 0, 0];\n const normalsAverage = [0, 0, 0];\n const uvAverage = [0, 0];\n const selectedIndices = [];\n const selectedVertices = [];\n const selectedIndicesArray = [];\n const vertexVisited = [];\n const indicesResult = [];\n const submeshDataArray = [];\n if (parentMesh && newVertexData.positions && newVertexData.positions.length > 0) {\n if (newVertexData.positions) {\n positions = newVertexData.positions;\n }\n else if (vertexData.positions) {\n positions = Array.from(vertexData.positions);\n }\n else {\n positions = [];\n }\n if (newVertexData.normals) {\n normals = newVertexData.normals;\n }\n else if (vertexData.normals) {\n normals = Array.from(vertexData.normals);\n }\n if (newVertexData.uvs) {\n uvs = newVertexData.uvs;\n }\n else if (vertexData.uvs) {\n uvs = Array.from(vertexData.uvs);\n }\n if (feature.vertexIndices && feature.vertexIndices.length > 0) {\n //creating sparse array to see which vertices are selected\n for (let i = 0; i < feature.vertexIndices.length; i++) {\n selectedIndicesArray[feature.vertexIndices[i]] = true;\n }\n if (indices) {\n //Looping through indices and checking which indices are touched by the selected vertices\n for (let j = 0; j < (indices === null || indices === void 0 ? void 0 : indices.length); j++) {\n if (selectedIndicesArray[indices[j]]) {\n selectedIndices.push(j);\n if (!vertexVisited[indices[j]]) {\n selectedVertices.push(j);\n vertexVisited[indices[j]] = true;\n }\n }\n }\n }\n //averaging all selected positions of the vertex\n for (let i = 0; i < selectedVertices.length; i++) {\n if (indices) {\n const vertex = indices[selectedVertices[i]];\n for (let j = 0; j < 3; j++) {\n positionAverage[j] = positionAverage[j] + positions[vertex * 3 + j];\n if (normals) {\n normalsAverage[j] = normalsAverage[j] + normals[vertex * 3 + j];\n }\n if (j != 2 && uvs) {\n uvAverage[j] = uvAverage[j] + uvs[vertex * 2 + j];\n }\n }\n }\n }\n for (let i = 0; i < 3; i++) {\n positionAverage[i] = positionAverage[i] / selectedVertices.length;\n normalsAverage[i] = normalsAverage[i] / selectedVertices.length;\n if (i != 2) {\n uvAverage[i] = uvAverage[i] / selectedVertices.length;\n }\n }\n //setting one vertex to calculated average position\n if (indices) {\n const targetIndex = indices[selectedVertices[0]];\n for (let i = 0; i < 3; i++) {\n positions[targetIndex * 3 + i] = positionAverage[i];\n if (normals) {\n normals[targetIndex * 3 + i] = normalsAverage[i];\n }\n if (i != 2 && uvs) {\n uvs[targetIndex * 2 + i] = uvAverage[i];\n }\n }\n //setting all selectedIndices to targetIndex with new position\n for (let i = 0; i < selectedIndices.length; i++) {\n indices[selectedIndices[i]] = targetIndex;\n }\n //looping through indices and pruning out polygons that make up invalid geometry\n for (let i = 0; i < indices.length; i = i + 3) {\n if (indices[i] != indices[i + 1] &&\n indices[i] != indices[i + 2] &&\n indices[i + 1] != indices[i + 2]) {\n indicesResult.push(indices[i]);\n indicesResult.push(indices[i + 1]);\n indicesResult.push(indices[i + 2]);\n }\n }\n //handling submesh data\n if (vertexData.submeshData) {\n for (let i = 0; i < vertexData.submeshData.length; i++) {\n const submeshData = vertexData.submeshData[i];\n submeshData.indexCount = indicesResult.length;\n submeshDataArray[i] = submeshData;\n }\n }\n }\n }\n newVertexData.positions = positions;\n newVertexData.indices = indicesResult;\n newVertexData.uvs = uvs;\n newVertexData.normals = normals;\n newVertexData.submeshData = submeshDataArray;\n }\n return newVertexData;\n }\n update3dInternal() { }\n }\n\n class FeatureStackRenderer extends Renderer {\n constructor(viewer, scene, rendererManager) {\n super(viewer, scene);\n this.rendererManager = rendererManager;\n this.queuedGeometryToReset = new Map();\n this.queuedFeaturesToProcess = new Map(); // index of feature to start recalculation at\n this.processedFeatures = new Map();\n this.featureOrder = new Map();\n this.renderers = new Map();\n this.modelToken = Token.Feature;\n const moveVertices = new MoveVerticesRenderer(viewer, scene);\n this.renderers.set(Token.MoveVerticesFeature, moveVertices);\n const weldVertices = new WeldVerticesRenderer(viewer, scene);\n this.renderers.set(Token.WeldVerticesFeature, weldVertices);\n const displacementMap = new DisplacementMapRenderer(viewer, scene);\n this.renderers.set(Token.DisplacementMapFeature, displacementMap);\n const uvMaps = new UvMapRenderer(viewer, scene);\n this.renderers.set(Token.UvMapFeature, uvMaps);\n const submeshes = new SubmeshRenderer(viewer, scene, rendererManager);\n this.renderers.set(Token.SubmeshFeature, submeshes);\n const linearPatterns = new LinearPatternRenderer(viewer, scene);\n this.renderers.set(Token.LinearPatternFeature, linearPatterns);\n const circularPatterns = new CircularPatternRenderer(viewer, scene);\n this.renderers.set(Token.CircularPatternFeature, circularPatterns);\n const joinGeometry = new JoinGeometryRenderer(viewer, scene);\n this.renderers.set(Token.JoinGeometryFeature, joinGeometry);\n const deleteFaces = new DeleteFacesRenderer(viewer, scene);\n this.renderers.set(Token.DeleteFacesFeature, deleteFaces);\n const normalSmoothing = new NormalSmoothingRenderer(viewer, scene);\n this.renderers.set(Token.NormalSmoothingFeature, normalSmoothing);\n const slice = new SliceFeatureRenderer(viewer, scene);\n this.renderers.set(Token.SliceFeature, slice);\n const mirror = new MirrorFeatureRenderer(viewer, scene);\n this.renderers.set(Token.MirrorFeature, mirror);\n const transform = new TransformRenderer(viewer, scene);\n this.renderers.set(Token.TransformFeature, transform);\n const extrude = new ExtrudeRenderer(viewer, scene);\n this.renderers.set(Token.ExtrudeFeature, extrude);\n const sweep = new SweepRenderer(viewer, scene);\n this.renderers.set(Token.SweepFeature, sweep);\n }\n forceRebuildGeometry(meshNode, vertexData) {\n if (!this.processedFeatures.has(meshNode._dynamicId)) {\n if (vertexData) {\n this.applyNewVertexData(meshNode, {}, vertexData);\n }\n return;\n }\n if (!vertexData) {\n vertexData = this.rendererManager.mesh.rebuildBaseGeometry(meshNode);\n }\n this.queuedGeometryToReset.set(meshNode._dynamicId, vertexData);\n const allChanges = new Set(meshNode.features.map((_, i) => i));\n this.queuedFeaturesToProcess.set(meshNode, allChanges);\n const movedIndex = this.getMovedFeatureIndex(meshNode);\n this.updateFeatureOrder(meshNode);\n if (movedIndex !== undefined && movedIndex >= 0) {\n this.recursivelyUpdateFeatureVisibility([meshNode.features[movedIndex]], meshNode.features, []);\n }\n this.update3dGeometry(0, meshNode);\n this.renderers.forEach(renderer => {\n renderer.clearCache(meshNode.features);\n });\n }\n resetMeshGeometry(meshNode, dataTypes, vertexData, hasFeatureToUpdate = false) {\n var _a;\n if (!vertexData) {\n vertexData = this.rendererManager.mesh.rebuildBaseGeometry(meshNode);\n }\n if (vertexData) {\n let newGeometry;\n if (!dataTypes) {\n newGeometry = vertexData;\n }\n else {\n // If the feature that changed affects aspects of a geometry, reset those aspects so we can recalculate them with\n // this or any other features that affected that aspect\n newGeometry = this.queuedGeometryToReset.get(meshNode._dynamicId) || {};\n // Null means that the vertex data specifically doesn't have those geometries, so they will be cleared\n if (!newGeometry.positions && hasChange(dataTypes, exports.eMeshChangeTypes.VertexPositions)) {\n newGeometry.positions = vertexData.positions || null;\n }\n if (!newGeometry.normals && hasChange(dataTypes, exports.eMeshChangeTypes.VertexNormals)) {\n newGeometry.normals = vertexData.normals || null;\n }\n if (!newGeometry.uvs && hasChange(dataTypes, exports.eMeshChangeTypes.VertexUvs)) {\n newGeometry.uvs = vertexData.uvs || null;\n }\n if (!newGeometry.indices && hasChange(dataTypes, exports.eMeshChangeTypes.VertexIndices)) {\n newGeometry.indices = vertexData.indices || null;\n }\n if (!newGeometry.submeshData && hasChange(dataTypes, exports.eMeshChangeTypes.Submeshes)) {\n newGeometry.submeshData = vertexData.submeshData || null;\n }\n }\n this.queuedGeometryToReset.set(meshNode._dynamicId, newGeometry);\n // If a feature affects a geometry element that is being reset, then it will need to be recalculated\n for (let i = 0; i < meshNode.features.length; i++) {\n const feature = meshNode.features[i];\n // If this geometry hasn't been flagged for recalculation, see if it needs to be\n if (!((_a = this.queuedFeaturesToProcess.get(meshNode)) === null || _a === void 0 ? void 0 : _a.has(i)) &&\n feature.affects &&\n featureIsAffected(feature.affects, newGeometry)) {\n hasFeatureToUpdate = true;\n this.queueGeometryUpdate(meshNode, i);\n this.resetMeshGeometry(meshNode, feature.affects, vertexData, hasFeatureToUpdate);\n }\n }\n // If there are no features to be recalculated but we need to reset some mesh geometry, then do that directly.\n if (!hasFeatureToUpdate && Object.getOwnPropertyNames(newGeometry).length > 0) {\n const existingVertexData = this.rendererManager.mesh.getMeshGeometry(meshNode) || vertexData;\n this.queuedGeometryToReset.delete(meshNode._dynamicId);\n this.applyNewVertexData(meshNode, newGeometry, existingVertexData);\n }\n }\n this.updateFeatureOrder(meshNode);\n }\n transformChanged(meshNode) {\n // If the transform on a mesh changes, any features that depend on the transform need to be updated\n for (let i = 0; i < meshNode.features.length; i++) {\n const feature = meshNode.features[i];\n if (feature.affectedBy && feature.affectedBy.has(exports.eMeshChangeTypes.Transform)) {\n this.queueGeometryUpdate(meshNode, i);\n }\n }\n }\n queueGeometryUpdate(meshNode, index) {\n let currentChanges = this.queuedFeaturesToProcess.get(meshNode);\n if (!currentChanges) {\n currentChanges = new Set();\n this.queuedFeaturesToProcess.set(meshNode, currentChanges);\n }\n currentChanges.add(index === undefined ? 0 : index);\n }\n update3dInternal(feature) {\n if (isTrackedClass(feature)) {\n if (!feature.enabled || feature._suppressed) {\n if (feature.changes.has('enabled') || feature.changes.has('_suppressed')) {\n this.delete3d(feature, true);\n }\n return;\n }\n const index = this.getFeatureIndexOnParent(feature);\n if (index > -1) {\n this.queueGeometryUpdate(feature._parent, index);\n }\n // Process all of the feature's non-geometry effects\n const renderer = this.renderers.get(Token.withKey(feature.$type));\n if (!renderer) {\n throw Error('no renderer for feature');\n }\n renderer.update3d(feature);\n }\n }\n update3dGeometry(renderStartTime = 0, node) {\n if (!node) {\n for (const node of this.queuedFeaturesToProcess.keys()) {\n if (node instanceof exports.MeshNode) {\n this.update3dGeometry(renderStartTime || performance.now(), node);\n }\n }\n return;\n }\n if (this.queuedFeaturesToProcess.has(node) && (!(node instanceof exports.CustomMeshNode) || node.geometryLoaded)) {\n const queuedFeatureToProcess = this.queuedFeaturesToProcess.get(node);\n const startFeatureIndex = Math.min(...Array.from(queuedFeatureToProcess.values()));\n // There is a known issue here where the VertexData.ExtractFromMesh function does not contain the updated vertices from mesh.updateVerticesData yet.\n // There is probably some kind of cache that is not updated if VertexData.ExtractFromMesh runs in the same frame\n let vertexDataCurrent = this.rendererManager.mesh.getMeshGeometry(node);\n const vertexDataNew = this.queuedGeometryToReset.has(node._dynamicId)\n ? this.queuedGeometryToReset.get(node._dynamicId)\n : {};\n // If the whole mesh is being reset, it may be a new mesh without geometry, so we can use the entire vertexDataNew.\n // If the mesh renderer asks the feature renderer for updates, it will not apply the geometry on its own so we have to make sure it happens here.\n // This is done to prevent applying geometry to the mesh multiple times, which is expensive for very large geometry.\n if (!vertexDataCurrent && vertexDataNew instanceof SubmeshVertexData) {\n vertexDataCurrent = vertexDataNew.shallowClone();\n }\n if (!vertexDataCurrent) {\n this.queuedFeaturesToProcess.delete(node);\n this.queuedGeometryToReset.delete(node._dynamicId);\n return;\n }\n for (let i = startFeatureIndex; i < node.features.length; i++) {\n const renderTime = performance.now() - renderStartTime;\n if (renderTime > this.rendererManager.maxRenderTime) {\n // store the progress of this feature stack to continue processing next frame\n console.warn(`Feature processing time exceeded ${renderTime}ms, skipping ${this.queuedFeaturesToProcess.size} features this frame`);\n queuedFeatureToProcess.clear();\n queuedFeatureToProcess.add(i);\n this.queuedGeometryToReset.set(node._dynamicId, vertexDataNew);\n return;\n }\n const feature = node.features[i];\n const renderer = this.renderers.get(Token.withKey(feature.$type));\n if (!renderer) {\n throw Error('no renderer for feature');\n }\n if (!feature.enabled || feature._suppressed) {\n // If the feature is not enabled, then we don't process it, but we still want to update the feature's geometry cache\n // Otherwise when the feature is enabled later it might not have the right data.\n renderer.updateMeshGeometryCache(feature, vertexDataCurrent, vertexDataNew);\n continue;\n }\n // Always process the features that were modified\n let processThisFeature = queuedFeatureToProcess.has(i);\n // Figure out if a property this feature is affected by was modified this cycle\n if (!processThisFeature && feature.affectedBy) {\n processThisFeature = featureIsAffected(feature.affectedBy, vertexDataNew);\n }\n if (isTrackedClass(feature)) {\n if (processThisFeature) {\n // Process all of the features geometry effects\n const resultVertexData = renderer.updateMeshGeometry(feature, vertexDataCurrent, vertexDataNew);\n if (feature.affects) {\n feature.affects.forEach(affects => {\n if (affects === exports.eMeshChangeTypes.VertexPositions) {\n vertexDataNew.positions = resultVertexData.positions || null;\n }\n if (affects === exports.eMeshChangeTypes.VertexNormals) {\n vertexDataNew.normals = resultVertexData.normals || null;\n }\n if (affects === exports.eMeshChangeTypes.VertexUvs) {\n vertexDataNew.uvs = resultVertexData.uvs || null;\n }\n if (affects === exports.eMeshChangeTypes.VertexIndices) {\n vertexDataNew.indices = resultVertexData.indices || null;\n }\n if (affects === exports.eMeshChangeTypes.Submeshes) {\n vertexDataNew.submeshData = resultVertexData.submeshData || null;\n }\n });\n }\n }\n }\n }\n if (vertexDataNew.submeshData) {\n if (!vertexDataCurrent.positions) {\n throw Error('Invalid geometry: positions is null');\n }\n const submeshData = vertexDataNew.submeshData.filter(s => s.materialIndex !== 0);\n const baseSubmeshes = calculateBaseSubmeshes(vertexDataCurrent, vertexDataNew, submeshData);\n const verticesCount = (vertexDataNew.positions || vertexDataCurrent.positions).length / 3;\n submeshData.unshift(...baseSubmeshes.map(bs => ({\n materialIndex: 0,\n verticesStart: 0,\n verticesCount: verticesCount,\n indexStart: bs.indexStart,\n indexCount: bs.indexCount,\n })));\n vertexDataNew.submeshData = submeshData;\n }\n if (Object.getOwnPropertyNames(vertexDataNew).length > 0) {\n this.applyNewVertexData(node, vertexDataNew, vertexDataCurrent);\n }\n }\n this.queuedFeaturesToProcess.delete(node);\n this.queuedGeometryToReset.delete(node._dynamicId);\n this.processedFeatures.set(node._dynamicId, true);\n }\n getMovedFeatureIndex(mesh) {\n const oldArray = this.featureOrder.get(mesh._dynamicId);\n if (oldArray) {\n if (oldArray.length !== mesh.features.length) {\n return -1;\n }\n for (let i = 0; i < oldArray.length; i++) {\n const newArrayId = mesh.features[i]._dynamicId;\n if (oldArray[i] !== newArrayId) {\n return i;\n }\n }\n }\n else {\n return -1;\n }\n }\n updateFeatureOrder(mesh) {\n const features = [];\n mesh.features.forEach(feature => {\n features.push(feature._dynamicId);\n });\n this.featureOrder.set(mesh._dynamicId, features);\n }\n recursivelyUpdateFeatureVisibility(affectedFeatures, features, disabledFeatures) {\n if (affectedFeatures.length === 0) {\n disabledFeatures.forEach(feature => {\n console.warn('The feature named', feature.name, 'got disabled due to a possible breaking change in the feature stack.');\n });\n return;\n }\n const newAffectedFeatures = [];\n affectedFeatures.forEach(affectedFeature => {\n var _a;\n const featureIndex = getFeatureIndex(affectedFeature, features);\n if (featureIndex >= 0) {\n for (let i = featureIndex; i < features.length; i++) {\n const feature = features[i];\n if (feature.enabled) {\n (_a = affectedFeature.affects) === null || _a === void 0 ? void 0 : _a.forEach(affector => {\n if (feature.dependsOn && feature.dependsOn.has(affector)) {\n feature.enabled = false;\n disabledFeatures.push(feature);\n feature._disableFeatureStackChange = true;\n newAffectedFeatures.push(feature);\n }\n });\n }\n }\n }\n });\n this.recursivelyUpdateFeatureVisibility(newAffectedFeatures, features, disabledFeatures);\n function getFeatureIndex(feature, features) {\n for (let i = 0; i < features.length; i++) {\n if (features[i] === feature) {\n return i;\n }\n }\n return -1;\n }\n }\n applyNewVertexData(node, vertexDataNew, vertexDataCurrent) {\n const mesh = this.viewer.getRendererNode(node);\n SubmeshVertexData.ExtendToRef(vertexDataNew, vertexDataCurrent);\n const invalid = verifyGeometry(vertexDataCurrent);\n if (invalid) {\n console.error(`Invalid or corrupted geometry output after feature processing in mesh ${node.name}. The ${invalid} are invalid.`);\n }\n else {\n vertexDataCurrent.applyToMesh(mesh);\n node._polygonCount = mesh.getTotalIndices() / 3;\n node._verticesCount = mesh.getTotalVertices();\n this.viewer.polygonTracker.next();\n this.viewer.warningTracker.next();\n }\n this.rendererManager.mesh.setRenderModeOnMesh(mesh, node._rootScene);\n this.viewer.bbCache.invalidate(node);\n }\n delete3d(feature, preserveCache) {\n const meshNode = feature._parent;\n if (meshNode && feature.affects) {\n this.resetMeshGeometry(meshNode, feature.affects);\n }\n const renderer = this.renderers.get(Token.withKey(feature.$type));\n if (!renderer) {\n throw Error('no renderer for feature');\n }\n renderer.delete3d(feature, preserveCache);\n }\n dispose() {\n this.queuedFeaturesToProcess.clear();\n this.renderers.forEach(renderer => renderer.dispose());\n super.dispose();\n }\n getFeatureIndexOnParent(feature) {\n if (feature._parent) {\n const index = feature._parent.features.findIndex(f => f._dynamicId === feature._dynamicId);\n if (index > -1) {\n return index;\n }\n }\n return -1;\n }\n }\n function calculateBaseSubmeshes(vertexDataCurrent, vertexDataNew, submeshData) {\n submeshData.sort((a, b) => a.indexStart - b.indexStart);\n const indicesCount = (vertexDataNew.indices || vertexDataCurrent.indices).length;\n const baseSubmeshes = [\n {\n indexStart: 0,\n indexCount: indicesCount,\n },\n ];\n submeshData.forEach(submesh => {\n const lastSubmesh = baseSubmeshes[baseSubmeshes.length - 1];\n if (lastSubmesh) {\n if (submesh.indexStart <= lastSubmesh.indexStart &&\n submesh.indexStart + submesh.indexCount > lastSubmesh.indexStart) {\n lastSubmesh.indexStart = submesh.indexStart + submesh.indexCount;\n lastSubmesh.indexCount = indicesCount - lastSubmesh.indexStart;\n if (lastSubmesh.indexCount <= 0) {\n baseSubmeshes.splice(-1, 1);\n }\n }\n else if (submesh.indexStart > lastSubmesh.indexStart) {\n lastSubmesh.indexCount = submesh.indexStart - lastSubmesh.indexStart;\n const remainingIndexCount = indicesCount - (submesh.indexStart + submesh.indexCount);\n if (remainingIndexCount > 0) {\n baseSubmeshes.push({\n indexStart: submesh.indexStart + submesh.indexCount,\n indexCount: remainingIndexCount,\n });\n }\n }\n }\n });\n return baseSubmeshes;\n }\n function featureIsAffected(affected, vertexDataNew) {\n let result = false;\n affected.forEach(affectedBy => {\n if (affectedBy === exports.eMeshChangeTypes.VertexPositions) {\n if (vertexDataNew.positions) {\n result = true; // Process the feature if a property it's affected by was changed this cycle\n }\n }\n if (affectedBy === exports.eMeshChangeTypes.VertexNormals) {\n if (vertexDataNew.normals) {\n result = true; // Process the feature if a property it's affected by was changed this cycle\n }\n }\n if (affectedBy === exports.eMeshChangeTypes.VertexUvs) {\n if (vertexDataNew.uvs) {\n result = true; // See above\n }\n }\n if (affectedBy === exports.eMeshChangeTypes.VertexIndices) {\n if (vertexDataNew.indices) {\n result = true; // See above\n }\n }\n if (affectedBy === exports.eMeshChangeTypes.Submeshes) {\n if (vertexDataNew.submeshData) {\n result = true; // See above\n }\n }\n });\n return result;\n }\n function hasChange(set, changeType) {\n return set.has(changeType);\n }\n\n const getMeshName = {\n handles: (dynamicId) => `${dynamicId}_handles`,\n handleBar: (dynamicId) => `${dynamicId}_handleBar`,\n directionHandle1: (dynamicId) => `${dynamicId}_directionHandle1`,\n directionHandle2: (dynamicId) => `${dynamicId}_directionHandle2`,\n };\n class SketchControlPointRenderer extends ConnectorRenderer {\n constructor(viewer, scene, rendererManager) {\n super(viewer, scene, rendererManager);\n this.controlPointScaleRatio = 0.12;\n this.modelToken = Token.SketchControlPoint;\n this.layers.forEach(layer => {\n layer.parts = new Map();\n });\n }\n handlesNode(node) {\n return instanceOf(node, Token.SketchControlPoint);\n }\n update3dInternal(c) {\n const realLayer = this.layers.get('real');\n const utilityLayer = this.layers.get('utility');\n const { mesh, utilityMesh } = super.update3dInternal(c, true);\n if (c._parent instanceof exports.SketchPath) {\n const realParts = realLayer.parts.get(c._dynamicId);\n const utilityParts = utilityLayer.parts.get(c._dynamicId);\n if (c.changes.hasAny('direction', 'direction2')) {\n if (realParts) {\n updateHandles(realParts, c, c._parent);\n }\n if (utilityParts) {\n updateHandles(utilityParts, c, c._parent);\n }\n }\n }\n return { mesh, utilityMesh };\n function updateHandles(parts, controlPoint, path) {\n var _a, _b;\n let enableHandle1, enableHandle2;\n switch (path.type) {\n case exports.eSketchPathType.arcThrough3Points:\n case exports.eSketchPathType.points:\n case exports.eSketchPathType.centerRadiusCircle:\n case exports.eSketchPathType.catmullRom:\n enableHandle1 = false;\n enableHandle2 = false;\n break;\n case exports.eSketchPathType.quadraticBezier:\n case exports.eSketchPathType.hermite:\n // Don't show handles on the ends of the path since that handle does nothing\n enableHandle1 =\n path.controlPoints.findIndex(cp => cp._dynamicId === controlPoint._dynamicId) <\n path.controlPoints.length - 1;\n enableHandle2 = false;\n break;\n default:\n // Don't show handles on the ends of the path since that handle does nothing\n enableHandle1 =\n path.controlPoints.findIndex(cp => cp._dynamicId === controlPoint._dynamicId) <\n path.controlPoints.length - 1;\n enableHandle2 = path.controlPoints.findIndex(cp => cp._dynamicId === controlPoint._dynamicId) > 0;\n }\n if (parts.directionHandle1 && c.direction.length() > 0) {\n parts.directionHandle1.position = c.direction.toVec3();\n }\n if (parts.directionHandle2 && c.direction2 && c.direction2.length() > 0) {\n parts.directionHandle2.position = c.direction2.toVec3();\n }\n (_a = parts.directionHandle1) === null || _a === void 0 ? void 0 : _a.setEnabled(enableHandle1);\n (_b = parts.directionHandle2) === null || _b === void 0 ? void 0 : _b.setEnabled(enableHandle2);\n if (parts.handleBar) {\n parts.handleBar.setEnabled(true);\n parts.handleBar = MeshBuilder.CreateDashedLines(getMeshName.handleBar(c._dynamicId), {\n points: [\n enableHandle1 ? c.direction.toVec3() : Vector3.Zero(),\n Vector3.Zero(),\n enableHandle2 && c.direction2 ? c.direction2.toVec3() : Vector3.Zero(),\n ],\n updatable: true,\n instance: parts.handleBar,\n });\n }\n }\n }\n delete3d(c) {\n const mesh = this.getMesh(c, this.layers.get('real'));\n const utilityMesh = this.getMesh(c, this.layers.get('utility'));\n this.viewer.camera.clearMaintainedNodeScreenSize(mesh);\n if (utilityMesh) {\n this.viewer.camera.clearMaintainedNodeScreenSize(utilityMesh);\n }\n super.delete3d(c);\n const sketchPath = c._parent;\n if (sketchPath && isTrackedClass(sketchPath)) {\n sketchPath.setPropertyIsModified('controlPoints', true);\n }\n }\n updateDirection(c) {\n const realLayer = this.layers.get('real');\n const utilityLayer = this.layers.get('utility');\n const mesh = this.getMesh(c, realLayer);\n const utilityMesh = this.getMesh(c, utilityLayer);\n const deltaBtwXaxisAndDirection = getRotationBtwVectors(Axis.X, this.getRelativeDirection(c, c.direction).normalize());\n const angleRot = Quaternion.RotationAxis(Axis.X, toRadians(c.angle));\n mesh.rotationQuaternion = utilityMesh.rotationQuaternion = deltaBtwXaxisAndDirection.multiply(angleRot);\n const realParts = realLayer.parts.get(c._dynamicId);\n const utilityParts = utilityLayer.parts.get(c._dynamicId);\n if (realParts === null || realParts === void 0 ? void 0 : realParts.handleContainer) {\n const invTransform = Matrix.Compose(Vector3.One(), mesh.rotationQuaternion, Vector3.Zero());\n invTransform.invert();\n invTransform.decomposeToTransformNode(realParts.handleContainer);\n if (utilityParts === null || utilityParts === void 0 ? void 0 : utilityParts.handleContainer) {\n invTransform.decomposeToTransformNode(utilityParts.handleContainer);\n }\n }\n this.updateScale(c);\n }\n updatePosition(c, bb) {\n const mesh = this.getMesh(c, this.layers.get('real'));\n const utilityMesh = this.getMesh(c, this.layers.get('utility'));\n mesh.position = this.getControlPointAbsPosition(c, bb);\n if (utilityMesh) {\n utilityMesh.position = mesh.position;\n }\n const sketchPath = c._parent;\n if (sketchPath && isTrackedClass(sketchPath)) {\n sketchPath.setPropertyIsModified('controlPoints', true);\n }\n }\n getControlPointAbsPosition(c, bb) {\n var _a;\n if (c.positionMode == exports.eConnectorPositionMode.boundingBox) {\n if (!bb) {\n bb = (_a = c.getSketch()) === null || _a === void 0 ? void 0 : _a.getBoundingBox();\n }\n if (bb) {\n return BoundingBoxRelativeHelper.convertRelativeToAbsolute(c, bb);\n }\n else {\n return new Vector3(c.position.x, c.position.y, c.position.z);\n }\n }\n else {\n return c.position.toVec3();\n }\n }\n getRelativeDirection(c, direction, bb) {\n var _a;\n if (c.positionMode === exports.eConnectorPositionMode.absolute) {\n return direction.toVec3();\n }\n else {\n if (!bb) {\n bb = (_a = c.getSketch()) === null || _a === void 0 ? void 0 : _a.getBoundingBox();\n }\n if (!bb) {\n return Vector3.Zero();\n }\n return BoundingBoxRelativeHelper.convertRelativeDirectionToAbsolute(c, direction, bb);\n }\n }\n getMesh(c, layer) {\n var _a;\n let mesh = layer.map.get(c._dynamicId);\n if (!mesh) {\n if (!layer.template) {\n layer.template = this.createMeshTemplate(layer);\n }\n mesh = layer.template.clone(c._dynamicId); // layer.template.createInstance(c._dynamicId);\n mesh.id = c._dynamicId;\n const [handleContainer] = mesh.getChildren();\n const [directionHandle1, directionHandle2, _handleBar] = handleContainer.getChildMeshes(true);\n const handleBar = _handleBar;\n handleBar.makeGeometryUnique();\n handleContainer.id = getMeshName.handles(c._dynamicId);\n directionHandle1.id = getMeshName.directionHandle1(c._dynamicId);\n directionHandle2.id = getMeshName.directionHandle2(c._dynamicId);\n const parentNode = (_a = c._parent) === null || _a === void 0 ? void 0 : _a._parent;\n if (parentNode && parentNode instanceof exports.SpaceNode) {\n const parent3d = this.viewer.getRendererNode(parentNode);\n if (parent3d) {\n mesh.setParent(parent3d);\n }\n }\n mesh.isPickable = layer.pickable;\n mesh.position = Vector3.Zero();\n mesh.rotationQuaternion = Quaternion.Identity();\n layer.map.set(c._dynamicId, mesh);\n layer.parts.set(c._dynamicId, {\n mesh,\n handleContainer: handleContainer,\n directionHandle1,\n directionHandle2,\n handleBar: handleBar,\n });\n }\n return mesh;\n }\n createMeshTemplate(layer) {\n var _a;\n const mesh = super.createMeshTemplate(layer);\n const handleContainer = new TransformNode('', this.scene);\n const handleBall = CreateSphere('handle1', {\n segments: 4,\n diameter: CONNECTOR_DIAM / 2,\n });\n handleBall.setParent(handleContainer);\n const m = new StandardMaterial('controlPointMaterial', this.scene);\n m.emissiveColor = this.viewer.highlightColor.toColor3();\n m.disableLighting = true;\n handleBall.material = m;\n const handle1 = handleBall;\n const handle2 = handleBall.clone('handle2', handleContainer, true);\n handle1.setEnabled(false);\n handle2.setEnabled(false);\n const baseColor = ((_a = this.viewer.sceneNode) === null || _a === void 0 ? void 0 : _a.edgeColor)\n ? this.viewer.sceneNode.edgeColor.toColor3()\n : Color3.White();\n const handleBar = MeshBuilder.CreateDashedLines('handleBar', {\n points: [Vector3.Up(), Vector3.Zero(), Vector3.Down()],\n updatable: true,\n dashSize: 1,\n gapSize: 1,\n dashNb: 30,\n });\n handleBar.color = baseColor;\n handleBar.isPickable = false;\n // const handleBarMaterial = new StandardMaterial('controlPointHandleBarMaterial', this.scene);\n // handleBarMaterial.emissiveColor = this.viewer.sceneNode?.edgeColor ? this.viewer.sceneNode.edgeColor.toColor3() : Color3.White();\n // handleBar.material = handleBarMaterial;\n handleBar.setParent(handleContainer);\n handleContainer.setParent(mesh);\n return mesh;\n }\n addSubscription(c) {\n const sketch = c.getSketch();\n if (sketch && !this.subscriptionMap.has(c)) {\n this.subscriptionMap.set(c, sketch._boundingBoxChanges.subscribe(bbInfo => {\n this.updatePosition(c, bbInfo);\n this.updateScale(c);\n }));\n }\n }\n }\n\n class ReferenceProcessor extends Processor {\n constructor(viewer) {\n super();\n this.viewer = viewer;\n /**\n * referencedNodeId => ( referencedByNodeId => referencedByNode )\n *\n * Any node that is in a nested scene will be keyed by the nested scene ID.\n * If the node is not in a nested scene (or there is no nesting), then the node will be keyed by the bare node id.\n * This applies to the referencedNodeId and the referencedByNodeId.\n * */\n this.referencesTo = new Map();\n this.orphanedReferences = new Map();\n }\n beforeDeletions(deletions) {\n const rootScene = this.viewer.sceneNode;\n if (!rootScene || !rootScene._manager) {\n return;\n }\n const rootManager = rootScene._manager;\n deletions.all().forEach(node => {\n var _a;\n if (!node._manager) {\n return;\n }\n let nodeId = node.id;\n if ((_a = node._parentScene) === null || _a === void 0 ? void 0 : _a.isNested()) {\n nodeId = rootScene.getNestedPathTo(node);\n }\n const referenceTo = this.referencesTo.get(nodeId);\n if (referenceTo && referenceTo.size > 0) {\n referenceTo.forEach((opts, referencedById) => {\n const referencedBy = rootManager.getById(referencedById);\n if (referencedBy && referencedBy[opts.referencePropertyName] === nodeId) {\n this.orphanedReferences.set(nodeId, {\n referencedBy,\n propName: opts.referencePropertyName,\n referenceId: referencedBy[opts.referencePropertyName],\n });\n referencedBy[opts.referencePropertyName] = undefined;\n }\n });\n }\n });\n }\n getReferencesTo() {\n return this.referencesTo;\n }\n beforeUpdates(changes) {\n const rootScene = this.viewer.sceneNode;\n if (!rootScene || !rootScene._manager) {\n return;\n }\n changes.all().forEach(node => {\n var _a;\n let nodeId = node.id;\n if ((_a = node._parentScene) === null || _a === void 0 ? void 0 : _a.isNested()) {\n nodeId = rootScene.getNestedPathTo(node);\n }\n const orphanedReference = this.orphanedReferences.get(nodeId);\n if (orphanedReference) {\n orphanedReference.referencedBy[orphanedReference.propName] = orphanedReference.referenceId;\n }\n });\n }\n afterUpdates(changes) {\n const rootScene = this.viewer.sceneNode;\n if (!rootScene || !rootScene._manager) {\n return;\n }\n changes.all().forEach(node => {\n var _a, _b, _c;\n if (!node._manager) {\n return;\n }\n const changedNodeManager = node._manager;\n // This block is for managing/updating the node reference\n // First, check nodes referencing other nodes for updates\n if (node.nodeReferences) {\n const targettingProperties = node.nodeReferences.keys();\n for (const _prop of targettingProperties) {\n const property = _prop;\n const referenceSettings = node.nodeReferences.get(_prop);\n // If a node's references were changed, update all the references.\n if (node.changes.has(property)) {\n if (node[property] && typeof node[property] !== 'string') {\n console.error(`Non-string id reference value found on ${node.name} property ${property}`);\n }\n // First, remove the node from the previous target node's referencing nodes set\n if (node.originalValues.has(property)) {\n const originalReferencedNodeId = node.originalValues.get(property);\n if (originalReferencedNodeId) {\n // Figure out the original target node's key\n let originalReferencedNodeKey = originalReferencedNodeId;\n const originalReferencedNode = changedNodeManager.getById(originalReferencedNodeId);\n if ((_a = originalReferencedNode === null || originalReferencedNode === void 0 ? void 0 : originalReferencedNode._parentScene) === null || _a === void 0 ? void 0 : _a.isNested()) {\n originalReferencedNodeKey = rootScene.getNestedPathTo(originalReferencedNode);\n }\n const referencedBy = this.referencesTo.get(originalReferencedNodeKey);\n if (referencedBy) {\n let nestedPathIdSource = node.id;\n if ((_b = node._parentScene) === null || _b === void 0 ? void 0 : _b.isNested()) {\n nestedPathIdSource = rootScene.getNestedPathTo(node);\n }\n referencedBy.delete(nestedPathIdSource);\n }\n }\n }\n // Set up the new references or delete the reference from entry if the reference was cleared\n if (node[property]) {\n const targetNode = node._manager.getById(node[property]);\n if (targetNode) {\n if (targetNode instanceof referenceSettings.classRef) {\n let nestedPathIdTarget = node[property];\n let nestedPathIdSource = node.id;\n const isNestedSource = ((_c = node._parentScene) === null || _c === void 0 ? void 0 : _c.isNested()) || false;\n if (nestedPathIdTarget.startsWith('@')) {\n const nestedTargetNode = changedNodeManager.getById(nestedPathIdTarget);\n if (nestedTargetNode) {\n nestedPathIdTarget = rootScene.getNestedPathTo(nestedTargetNode);\n }\n }\n else if (isNestedSource) {\n // If source node is in a nested scene and the target node id is not a nested path, that means that\n // the nested node is targetting a node in it's own scene. But we need to use\n // a nested path because there could be multiple instances of the nested node's scene\n nestedPathIdTarget = rootScene.getNestedPathTo(targetNode);\n }\n if (isNestedSource) {\n nestedPathIdSource = rootScene.getNestedPathTo(node);\n }\n let referencedBy = this.referencesTo.get(nestedPathIdTarget);\n if (!referencedBy) {\n referencedBy = new Map();\n this.referencesTo.set(nestedPathIdTarget, referencedBy);\n }\n referencedBy.set(nestedPathIdSource, {\n limitKeys: referenceSettings.limitKeys,\n ref: node,\n referencePropertyName: property,\n });\n }\n else {\n console.warn(`Node ${node.name} property ${property} is not referencing a ${referenceSettings.classRef}`);\n }\n }\n }\n }\n }\n }\n });\n // Next, check nodes being referenced by other nodes to see if the other nodes will need to be updated\n changes.all().forEach(node => {\n this.triggerChangesOnReferencers(node);\n });\n }\n triggerChangesOnReferencers(node, changes) {\n const changeMap = changes || new Map(node.changes);\n if (changeMap.size > 0) {\n this.afterCleanup(() => {\n this.triggerChangesOnReferencersSync(node, changeMap);\n });\n }\n }\n triggerChangesOnReferencersSync(node, changes, transformationPropagation) {\n var _a;\n const rootScene = this.viewer.sceneNode;\n if (!rootScene || !rootScene._manager) {\n return;\n }\n const rootManager = rootScene._manager;\n let nodeId = node.id;\n if ((_a = node._parentScene) === null || _a === void 0 ? void 0 : _a.isNested()) {\n nodeId = rootScene.getNestedPathTo(node);\n }\n const referenceTo = this.referencesTo.get(nodeId);\n if (referenceTo) {\n referenceTo.forEach((referenceSettings, referencedById) => {\n const referencedBy = rootManager.getById(referencedById);\n if (referencedBy && isTrackedClass(referencedBy)) {\n if (!referenceSettings.limitKeys ||\n transformationPropagation ||\n changes.hasAny(...referenceSettings.limitKeys)) {\n referencedBy.setPropertyIsModified(referenceSettings.referencePropertyName, true);\n }\n }\n });\n }\n // For transformation changes, propagate to the children because the children will also get moved without passing through the change tracker\n // This avoids needing a secondary watcher for node position\n if (node instanceof exports.SpaceNode &&\n changes.hasAny('position', 'eulerRotation', 'rotationAngle', 'rotationAxis', 'pivot', 'scaling')) {\n const children = node.getChildren();\n for (let i = 0; i < children.length; i++) {\n // Always trigger changes on children because they may have different property names\n this.triggerChangesOnReferencersSync(children[i], changes, true);\n }\n }\n }\n dispose() {\n this.referencesTo.clear();\n }\n }\n\n /**\n * Processor to calculate the bounding box of a sketch, since it is determined by the path defined within.\n * Must run after the mate processor because mates can move sketch points and change the bounding box.\n */\n class SketchProcessor extends Processor {\n constructor(controlPointRenderer) {\n super();\n this.controlPointRenderer = controlPointRenderer;\n }\n beforeRender() { }\n beforeDeletions(deletions) {\n // Update paths that need to be updated\n if (deletions.has(Token.SketchPath)) {\n const sketchPaths = deletions.get(Token.SketchPath);\n sketchPaths.forEach(p => this.updatePath(p));\n }\n if (deletions.has(Token.SketchControlPoint) || deletions.has(Token.SketchPath)) {\n const sketchPathChanges = deletions.get(Token.SketchPath);\n const sketchControlPointChanges = deletions.get(Token.SketchControlPoint);\n const sketchesToUpdate = new Map();\n sketchPathChanges.forEach(node => {\n const sketch = node._parent;\n if (sketch) {\n if (!sketchesToUpdate.has(sketch._dynamicId)) {\n sketchesToUpdate.set(sketch._dynamicId, sketch);\n }\n }\n });\n sketchControlPointChanges.forEach(node => {\n const sketch = node.getSketch();\n if (sketch) {\n if (!sketchesToUpdate.has(sketch._dynamicId)) {\n sketchesToUpdate.set(sketch._dynamicId, sketch);\n }\n }\n });\n sketchesToUpdate.forEach(this.calculateSketchBounds);\n }\n }\n beforeUpdates(changes) {\n // Update paths that need to be updated\n if (changes.has(Token.SketchPath)) {\n const sketchPaths = changes.get(Token.SketchPath);\n sketchPaths.forEach(p => this.updatePath(p));\n }\n // Update the sketch bounding box with the path control points within it that are absolute positioned\n if (changes.has(Token.SketchControlPoint)) {\n const sketchControlPointChanges = changes.get(Token.SketchControlPoint);\n const sketchesToUpdate = new Map();\n sketchControlPointChanges.forEach(node => {\n if (node.changes.hasAny('position', 'positionMode')) {\n const sketch = node.getSketch();\n if (sketch) {\n if (!sketchesToUpdate.has(sketch._dynamicId)) {\n sketchesToUpdate.set(sketch._dynamicId, sketch);\n }\n }\n }\n });\n sketchesToUpdate.forEach(this.calculateSketchBounds);\n }\n }\n afterUpdates(changes) {\n if (changes.has(Token.SketchControlPoint)) {\n const sketchControlPointChanges = changes.get(Token.SketchControlPoint);\n sketchControlPointChanges.forEach(node => {\n const sketch = node.getSketch();\n if (sketch) {\n this.afterCleanup(() => {\n sketch.paths.forEach(path => {\n if (isTrackedClass(path)) {\n this.updatePath(path, true);\n }\n });\n sketch.connectors.forEach(connector => {\n if (connector.positionMode === exports.eConnectorPositionMode.moveAlongSketch) {\n if (connector._parent instanceof exports.SketchNode) {\n connector._sketchControlPoints = this.getControlPoints(connector._parent);\n }\n }\n });\n });\n }\n });\n }\n }\n calculateSketchBounds(sketch) {\n const controlPoints = sketch.paths.flatMap(path => path.controlPoints);\n let min;\n let max;\n controlPoints.forEach(controlPoint => {\n if (controlPoint.positionMode === exports.eConnectorPositionMode.absolute) {\n const position = controlPoint.position;\n if (!min) {\n min = position.toVec3();\n }\n max = position.toVec3();\n }\n });\n if (min && max) {\n sketch._boundingBoxMin = exports.KbVector.FromVec3(min);\n sketch._boundingBoxMax = exports.KbVector.FromVec3(max);\n }\n else {\n sketch._boundingBoxMin = exports.KbVector.Zero();\n sketch._boundingBoxMax = exports.KbVector.Zero();\n }\n sketch._boundingBoxChanges.next(sketch.getBoundingBox());\n }\n updatePath(path, force) {\n var _a;\n let points;\n if (force ||\n path.changes.hasAny('controlPoints', 'tessellation', 'type', 'holeForPathGroup', 'pathGroupKey') ||\n !((_a = path._points) === null || _a === void 0 ? void 0 : _a.length)) {\n points = this.buildSketchPoints(path);\n path._points = points;\n const parent = path._parent;\n if (parent instanceof exports.SketchNode && isTrackedClass(parent)) {\n parent.setPropertyIsModified('paths', true);\n }\n }\n else if (path.changes.has('_selected')) {\n const parent = path._parent;\n if (parent instanceof exports.SketchNode && isTrackedClass(parent)) {\n parent.setPropertyIsModified('paths', true);\n }\n }\n }\n buildSketchPoints(path) {\n const points = [];\n if (!path._parent) {\n return points;\n }\n const bb = path._parent.getBoundingBox();\n switch (path.type) {\n case exports.eSketchPathType.catmullRom: {\n const positions = path.controlPoints.map(v => this.getPointPosition(v));\n if (positions.length > 2) {\n const curve = Curve3.CreateCatmullRomSpline(positions, path.tessellation, false);\n points.push(...curve.getPoints());\n }\n else {\n points.push(...positions);\n }\n break;\n }\n case exports.eSketchPathType.hermite: {\n for (let i = 0; i + 1 < path.controlPoints.length; i++) {\n const start = path.controlPoints[i], end = path.controlPoints[i + 1];\n const p0 = this.getPointPosition(start);\n const p1 = this.getPointPosition(end);\n const d0 = this.getPointDirection(start, bb);\n const d1 = this.getPointDirection(end, bb);\n const curve = Curve3.CreateHermiteSpline(p0, d0, p1, d1, path.tessellation);\n points.push(...curve.getPoints());\n }\n break;\n }\n case exports.eSketchPathType.cubicBezier: {\n for (let i = 0; i + 1 < path.controlPoints.length; i++) {\n const start = path.controlPoints[i], end = path.controlPoints[i + 1];\n const direction2 = this.getPointDirection2(end, bb);\n if (direction2) {\n const p0 = this.getPointPosition(start);\n const p3 = this.getPointPosition(end);\n if (p0.equals(p3)) {\n continue;\n }\n const p1 = p0.add(this.getPointDirection(start, bb));\n const p2 = p3.add(direction2);\n const curve = Curve3.CreateCubicBezier(p0, p1, p2, p3, path.tessellation);\n points.push(...curve.getPoints());\n }\n }\n break;\n }\n case exports.eSketchPathType.quadraticBezier: {\n const controlPoint = new Vector3();\n for (let i = 0; i + 1 < path.controlPoints.length; i++) {\n const start = path.controlPoints[i], end = path.controlPoints[i + 1];\n const p0 = this.getPointPosition(start);\n const p2 = this.getPointPosition(end);\n if (p0.equals(p2)) {\n continue;\n }\n p0.addToRef(this.getPointDirection(start, bb), controlPoint);\n const curve = Curve3.CreateQuadraticBezier(p0, controlPoint, p2, path.tessellation);\n points.push(...curve.getPoints());\n }\n break;\n }\n case exports.eSketchPathType.centerRadiusCircle: {\n if (path.controlPoints.length >= 2) {\n const [p0, p1] = path.controlPoints.slice(0, 2).map(cp => this.getPointPosition(cp));\n const normal = this.getPointDirection(path.controlPoints[0], bb).normalize();\n const p2 = Vector3.Cross(p0.subtract(p1), normal).addInPlace(p0);\n const curve = Curve3.ArcThru3Points(p0.scale(2).subtractInPlace(p1), p1, p2, path.tessellation * 2, false, true);\n points.push(...curve.getPoints());\n }\n break;\n }\n case exports.eSketchPathType.arcThrough3Points: {\n if (path.controlPoints.length >= 3) {\n const [p0, p1, p2] = path.controlPoints.slice(0, 3).map(cp => this.getPointPosition(cp));\n const curve = Curve3.ArcThru3Points(p0, p1, p2, path.tessellation, false, false);\n points.push(...curve.getPoints());\n }\n break;\n }\n default: {\n path.controlPoints.forEach(point => {\n points.push(this.getPointPosition(point));\n });\n }\n }\n // /**\n // * We want the sketch to always \"face\" in the positive direction of the axis for consistency\n // * Otherwise it's indeterminate which way the sketch will face and makes the sketch hard to manage\n // * because it depends on the handed-ness of the points (clockwise or counter-clockwise)\n // */\n // const primaryNormal = axisToVector3(path._parent.normal);\n // const faceNormal = calculateFaceNormal(points);\n // if (Vector3.Dot(faceNormal, primaryNormal) < 0) {\n // points.reverse();\n // }\n return points;\n }\n getPointPosition(point) {\n return this.controlPointRenderer.getControlPointAbsPosition(point);\n }\n getPointDirection(point, bb) {\n return this.controlPointRenderer.getRelativeDirection(point, point.direction, bb);\n }\n getPointDirection2(point, bb) {\n if (!point.direction2) {\n return undefined;\n }\n else {\n return this.controlPointRenderer.getRelativeDirection(point, point.direction2, bb);\n }\n }\n getControlPoints(node) {\n const controlPoints = [];\n node.paths.forEach(path => {\n path._points.forEach(point => {\n controlPoints.push(point);\n });\n });\n return controlPoints;\n }\n }\n\n /**\n * PostProcessRenderPipeline\n * @see https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/postProcessRenderPipeline\n */\n class PostProcessRenderPipeline {\n /**\n * Gets pipeline name\n */\n get name() {\n return this._name;\n }\n /** Gets the list of attached cameras */\n get cameras() {\n return this._cameras;\n }\n /**\n * Initializes a PostProcessRenderPipeline\n * @param _engine engine to add the pipeline to\n * @param name name of the pipeline\n */\n constructor(_engine, name) {\n this._engine = _engine;\n this._name = name;\n this._renderEffects = {};\n this._renderEffectsForIsolatedPass = new Array();\n this._cameras = [];\n }\n /**\n * Gets the class name\n * @returns \"PostProcessRenderPipeline\"\n */\n getClassName() {\n return \"PostProcessRenderPipeline\";\n }\n /**\n * If all the render effects in the pipeline are supported\n */\n get isSupported() {\n for (const renderEffectName in this._renderEffects) {\n if (Object.prototype.hasOwnProperty.call(this._renderEffects, renderEffectName)) {\n if (!this._renderEffects[renderEffectName].isSupported) {\n return false;\n }\n }\n }\n return true;\n }\n /**\n * Adds an effect to the pipeline\n * @param renderEffect the effect to add\n */\n addEffect(renderEffect) {\n this._renderEffects[renderEffect._name] = renderEffect;\n }\n // private\n /** @internal */\n _rebuild() { }\n /**\n * @internal\n */\n _enableEffect(renderEffectName, cameras) {\n const renderEffects = this._renderEffects[renderEffectName];\n if (!renderEffects) {\n return;\n }\n renderEffects._enable(Tools.MakeArray(cameras || this._cameras));\n }\n /**\n * @internal\n */\n _disableEffect(renderEffectName, cameras) {\n const renderEffects = this._renderEffects[renderEffectName];\n if (!renderEffects) {\n return;\n }\n renderEffects._disable(Tools.MakeArray(cameras || this._cameras));\n }\n /**\n * @internal\n */\n _attachCameras(cameras, unique) {\n const cams = Tools.MakeArray(cameras || this._cameras);\n if (!cams) {\n return;\n }\n const indicesToDelete = [];\n let i;\n for (i = 0; i < cams.length; i++) {\n const camera = cams[i];\n if (!camera) {\n continue;\n }\n if (this._cameras.indexOf(camera) === -1) {\n this._cameras.push(camera);\n }\n else if (unique) {\n indicesToDelete.push(i);\n }\n }\n for (i = 0; i < indicesToDelete.length; i++) {\n cams.splice(indicesToDelete[i], 1);\n }\n for (const renderEffectName in this._renderEffects) {\n if (Object.prototype.hasOwnProperty.call(this._renderEffects, renderEffectName)) {\n this._renderEffects[renderEffectName]._attachCameras(cams);\n }\n }\n }\n /**\n * @internal\n */\n _detachCameras(cameras) {\n const cams = Tools.MakeArray(cameras || this._cameras);\n if (!cams) {\n return;\n }\n for (const renderEffectName in this._renderEffects) {\n if (Object.prototype.hasOwnProperty.call(this._renderEffects, renderEffectName)) {\n this._renderEffects[renderEffectName]._detachCameras(cams);\n }\n }\n for (let i = 0; i < cams.length; i++) {\n this._cameras.splice(this._cameras.indexOf(cams[i]), 1);\n }\n }\n /** @internal */\n _update() {\n for (const renderEffectName in this._renderEffects) {\n if (Object.prototype.hasOwnProperty.call(this._renderEffects, renderEffectName)) {\n this._renderEffects[renderEffectName]._update();\n }\n }\n for (let i = 0; i < this._cameras.length; i++) {\n if (!this._cameras[i]) {\n continue;\n }\n const cameraName = this._cameras[i].name;\n if (this._renderEffectsForIsolatedPass[cameraName]) {\n this._renderEffectsForIsolatedPass[cameraName]._update();\n }\n }\n }\n /** @internal */\n _reset() {\n this._renderEffects = {};\n this._renderEffectsForIsolatedPass = new Array();\n }\n _enableMSAAOnFirstPostProcess(sampleCount) {\n if (!this._engine._features.supportMSAA) {\n return false;\n }\n // Set samples of the very first post process to 4 to enable native anti-aliasing in browsers that support webGL 2.0 (See: https://github.com/BabylonJS/Babylon.js/issues/3754)\n const effectKeys = Object.keys(this._renderEffects);\n if (effectKeys.length > 0) {\n const postProcesses = this._renderEffects[effectKeys[0]].getPostProcesses();\n if (postProcesses) {\n postProcesses[0].samples = sampleCount;\n }\n }\n return true;\n }\n /**\n * Sets the required values to the prepass renderer.\n * @param prePassRenderer defines the prepass renderer to setup.\n * @returns true if the pre pass is needed.\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n setPrePassRenderer(prePassRenderer) {\n // Do Nothing by default\n return false;\n }\n /**\n * Disposes of the pipeline\n */\n dispose() {\n // Must be implemented by children\n }\n }\n __decorate$1([\n serialize()\n ], PostProcessRenderPipeline.prototype, \"_name\", void 0);\n\n /**\n * This represents a set of one or more post processes in Babylon.\n * A post process can be used to apply a shader to a texture after it is rendered.\n * @example https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/postProcessRenderPipeline\n */\n class PostProcessRenderEffect {\n /**\n * Instantiates a post process render effect.\n * A post process can be used to apply a shader to a texture after it is rendered.\n * @param engine The engine the effect is tied to\n * @param name The name of the effect\n * @param getPostProcesses A function that returns a set of post processes which the effect will run in order to be run.\n * @param singleInstance False if this post process can be run on multiple cameras. (default: true)\n */\n constructor(engine, name, getPostProcesses, singleInstance) {\n this._name = name;\n this._singleInstance = singleInstance || true;\n this._getPostProcesses = getPostProcesses;\n this._cameras = {};\n this._indicesForCamera = {};\n this._postProcesses = {};\n }\n /**\n * Checks if all the post processes in the effect are supported.\n */\n get isSupported() {\n for (const index in this._postProcesses) {\n if (Object.prototype.hasOwnProperty.call(this._postProcesses, index)) {\n const pps = this._postProcesses[index];\n for (let ppIndex = 0; ppIndex < pps.length; ppIndex++) {\n if (!pps[ppIndex].isSupported) {\n return false;\n }\n }\n }\n }\n return true;\n }\n /**\n * Updates the current state of the effect\n * @internal\n */\n _update() { }\n /**\n * Attaches the effect on cameras\n * @param cameras The camera to attach to.\n * @internal\n */\n _attachCameras(cameras) {\n let cameraKey;\n const cams = Tools.MakeArray(cameras || this._cameras);\n if (!cams) {\n return;\n }\n for (let i = 0; i < cams.length; i++) {\n const camera = cams[i];\n if (!camera) {\n continue;\n }\n const cameraName = camera.name;\n if (this._singleInstance) {\n cameraKey = 0;\n }\n else {\n cameraKey = cameraName;\n }\n if (!this._postProcesses[cameraKey]) {\n const postProcess = this._getPostProcesses();\n if (postProcess) {\n this._postProcesses[cameraKey] = Array.isArray(postProcess) ? postProcess : [postProcess];\n }\n }\n if (!this._indicesForCamera[cameraName]) {\n this._indicesForCamera[cameraName] = [];\n }\n this._postProcesses[cameraKey].forEach((postProcess) => {\n const index = camera.attachPostProcess(postProcess);\n this._indicesForCamera[cameraName].push(index);\n });\n if (!this._cameras[cameraName]) {\n this._cameras[cameraName] = camera;\n }\n }\n }\n /**\n * Detaches the effect on cameras\n * @param cameras The camera to detach from.\n * @internal\n */\n _detachCameras(cameras) {\n const cams = Tools.MakeArray(cameras || this._cameras);\n if (!cams) {\n return;\n }\n for (let i = 0; i < cams.length; i++) {\n const camera = cams[i];\n const cameraName = camera.name;\n const postProcesses = this._postProcesses[this._singleInstance ? 0 : cameraName];\n if (postProcesses) {\n postProcesses.forEach((postProcess) => {\n camera.detachPostProcess(postProcess);\n });\n }\n if (this._cameras[cameraName]) {\n this._cameras[cameraName] = null;\n }\n delete this._indicesForCamera[cameraName];\n }\n }\n /**\n * Enables the effect on given cameras\n * @param cameras The camera to enable.\n * @internal\n */\n _enable(cameras) {\n const cams = Tools.MakeArray(cameras || this._cameras);\n if (!cams) {\n return;\n }\n for (let i = 0; i < cams.length; i++) {\n const camera = cams[i];\n const cameraName = camera.name;\n const cameraKey = this._singleInstance ? 0 : cameraName;\n for (let j = 0; j < this._indicesForCamera[cameraName].length; j++) {\n const index = this._indicesForCamera[cameraName][j];\n const postProcess = camera._postProcesses[index];\n if (postProcess === undefined || postProcess === null) {\n cams[i].attachPostProcess(this._postProcesses[cameraKey][j], index);\n }\n }\n }\n }\n /**\n * Disables the effect on the given cameras\n * @param cameras The camera to disable.\n * @internal\n */\n _disable(cameras) {\n const cams = Tools.MakeArray(cameras || this._cameras);\n if (!cams) {\n return;\n }\n for (let i = 0; i < cams.length; i++) {\n const camera = cams[i];\n const cameraName = camera.name;\n this._postProcesses[this._singleInstance ? 0 : cameraName].forEach((postProcess) => {\n camera.detachPostProcess(postProcess);\n });\n }\n }\n /**\n * Gets a list of the post processes contained in the effect.\n * @param camera The camera to get the post processes on.\n * @returns The list of the post processes in the effect.\n */\n getPostProcesses(camera) {\n if (this._singleInstance) {\n return this._postProcesses[0];\n }\n else {\n if (!camera) {\n return null;\n }\n return this._postProcesses[camera.name];\n }\n }\n }\n\n /**\n * PostProcessRenderPipelineManager class\n * @see https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/postProcessRenderPipeline\n */\n class PostProcessRenderPipelineManager {\n /**\n * Initializes a PostProcessRenderPipelineManager\n * @see https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/postProcessRenderPipeline\n */\n constructor() {\n this._renderPipelines = {};\n }\n /**\n * Gets the list of supported render pipelines\n */\n get supportedPipelines() {\n const result = [];\n for (const renderPipelineName in this._renderPipelines) {\n if (Object.prototype.hasOwnProperty.call(this._renderPipelines, renderPipelineName)) {\n const pipeline = this._renderPipelines[renderPipelineName];\n if (pipeline.isSupported) {\n result.push(pipeline);\n }\n }\n }\n return result;\n }\n /**\n * Adds a pipeline to the manager\n * @param renderPipeline The pipeline to add\n */\n addPipeline(renderPipeline) {\n this._renderPipelines[renderPipeline._name] = renderPipeline;\n }\n /**\n * Remove the pipeline from the manager\n * @param renderPipelineName the name of the pipeline to remove\n */\n removePipeline(renderPipelineName) {\n delete this._renderPipelines[renderPipelineName];\n }\n /**\n * Attaches a camera to the pipeline\n * @param renderPipelineName The name of the pipeline to attach to\n * @param cameras the camera to attach\n * @param unique if the camera can be attached multiple times to the pipeline\n */\n attachCamerasToRenderPipeline(renderPipelineName, cameras, unique = false) {\n const renderPipeline = this._renderPipelines[renderPipelineName];\n if (!renderPipeline) {\n return;\n }\n renderPipeline._attachCameras(cameras, unique);\n }\n /**\n * Detaches a camera from the pipeline\n * @param renderPipelineName The name of the pipeline to detach from\n * @param cameras the camera to detach\n */\n detachCamerasFromRenderPipeline(renderPipelineName, cameras) {\n const renderPipeline = this._renderPipelines[renderPipelineName];\n if (!renderPipeline) {\n return;\n }\n renderPipeline._detachCameras(cameras);\n }\n /**\n * Enables an effect by name on a pipeline\n * @param renderPipelineName the name of the pipeline to enable the effect in\n * @param renderEffectName the name of the effect to enable\n * @param cameras the cameras that the effect should be enabled on\n */\n enableEffectInPipeline(renderPipelineName, renderEffectName, cameras) {\n const renderPipeline = this._renderPipelines[renderPipelineName];\n if (!renderPipeline) {\n return;\n }\n renderPipeline._enableEffect(renderEffectName, cameras);\n }\n /**\n * Disables an effect by name on a pipeline\n * @param renderPipelineName the name of the pipeline to disable the effect in\n * @param renderEffectName the name of the effect to disable\n * @param cameras the cameras that the effect should be disabled on\n */\n disableEffectInPipeline(renderPipelineName, renderEffectName, cameras) {\n const renderPipeline = this._renderPipelines[renderPipelineName];\n if (!renderPipeline) {\n return;\n }\n renderPipeline._disableEffect(renderEffectName, cameras);\n }\n /**\n * Updates the state of all contained render pipelines and disposes of any non supported pipelines\n */\n update() {\n for (const renderPipelineName in this._renderPipelines) {\n if (Object.prototype.hasOwnProperty.call(this._renderPipelines, renderPipelineName)) {\n const pipeline = this._renderPipelines[renderPipelineName];\n if (!pipeline.isSupported) {\n pipeline.dispose();\n delete this._renderPipelines[renderPipelineName];\n }\n else {\n pipeline._update();\n }\n }\n }\n }\n /** @internal */\n _rebuild() {\n for (const renderPipelineName in this._renderPipelines) {\n if (Object.prototype.hasOwnProperty.call(this._renderPipelines, renderPipelineName)) {\n const pipeline = this._renderPipelines[renderPipelineName];\n pipeline._rebuild();\n }\n }\n }\n /**\n * Disposes of the manager and pipelines\n */\n dispose() {\n for (const renderPipelineName in this._renderPipelines) {\n if (Object.prototype.hasOwnProperty.call(this._renderPipelines, renderPipelineName)) {\n const pipeline = this._renderPipelines[renderPipelineName];\n pipeline.dispose();\n }\n }\n }\n }\n\n Object.defineProperty(Scene.prototype, \"postProcessRenderPipelineManager\", {\n get: function () {\n if (!this._postProcessRenderPipelineManager) {\n // Register the G Buffer component to the scene.\n let component = this._getComponent(SceneComponentConstants.NAME_POSTPROCESSRENDERPIPELINEMANAGER);\n if (!component) {\n component = new PostProcessRenderPipelineManagerSceneComponent(this);\n this._addComponent(component);\n }\n this._postProcessRenderPipelineManager = new PostProcessRenderPipelineManager();\n }\n return this._postProcessRenderPipelineManager;\n },\n enumerable: true,\n configurable: true,\n });\n /**\n * Defines the Render Pipeline scene component responsible to rendering pipelines\n */\n class PostProcessRenderPipelineManagerSceneComponent {\n /**\n * Creates a new instance of the component for the given scene\n * @param scene Defines the scene to register the component in\n */\n constructor(scene) {\n /**\n * The component name helpful to identify the component in the list of scene components.\n */\n this.name = SceneComponentConstants.NAME_POSTPROCESSRENDERPIPELINEMANAGER;\n this.scene = scene;\n }\n /**\n * Registers the component in a given scene\n */\n register() {\n this.scene._gatherRenderTargetsStage.registerStep(SceneComponentConstants.STEP_GATHERRENDERTARGETS_POSTPROCESSRENDERPIPELINEMANAGER, this, this._gatherRenderTargets);\n }\n /**\n * Rebuilds the elements related to this component in case of\n * context lost for instance.\n */\n rebuild() {\n if (this.scene._postProcessRenderPipelineManager) {\n this.scene._postProcessRenderPipelineManager._rebuild();\n }\n }\n /**\n * Disposes the component and the associated resources\n */\n dispose() {\n if (this.scene._postProcessRenderPipelineManager) {\n this.scene._postProcessRenderPipelineManager.dispose();\n }\n }\n _gatherRenderTargets() {\n if (this.scene._postProcessRenderPipelineManager) {\n this.scene._postProcessRenderPipelineManager.update();\n }\n }\n }\n\n // Do not edit.\n const name$2w = \"chromaticAberrationPixelShader\";\n const shader$2w = `uniform sampler2D textureSampler; \nuniform float chromatic_aberration;uniform float radialIntensity;uniform vec2 direction;uniform vec2 centerPosition;uniform float screen_width;uniform float screen_height;varying vec2 vUV;\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{vec2 centered_screen_pos=vec2(vUV.x-centerPosition.x,vUV.y-centerPosition.y);vec2 directionOfEffect=direction;if(directionOfEffect.x==0. && directionOfEffect.y==0.){directionOfEffect=normalize(centered_screen_pos);}\nfloat radius2=centered_screen_pos.x*centered_screen_pos.x\n+ centered_screen_pos.y*centered_screen_pos.y;float radius=sqrt(radius2);vec4 original=texture2D(textureSampler,vUV);vec3 ref_indices=vec3(-0.3,0.0,0.3);float ref_shiftX=chromatic_aberration*pow(radius,radialIntensity)*directionOfEffect.x/screen_width;float ref_shiftY=chromatic_aberration*pow(radius,radialIntensity)*directionOfEffect.y/screen_height;vec2 ref_coords_r=vec2(vUV.x+ref_indices.r*ref_shiftX,vUV.y+ref_indices.r*ref_shiftY*0.5);vec2 ref_coords_g=vec2(vUV.x+ref_indices.g*ref_shiftX,vUV.y+ref_indices.g*ref_shiftY*0.5);vec2 ref_coords_b=vec2(vUV.x+ref_indices.b*ref_shiftX,vUV.y+ref_indices.b*ref_shiftY*0.5);original.r=texture2D(textureSampler,ref_coords_r).r;original.g=texture2D(textureSampler,ref_coords_g).g;original.b=texture2D(textureSampler,ref_coords_b).b;original.a=clamp(texture2D(textureSampler,ref_coords_r).a+texture2D(textureSampler,ref_coords_g).a+texture2D(textureSampler,ref_coords_b).a,0.,1.);gl_FragColor=original;}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2w] = shader$2w;\n\n // Do not edit.\n const name$2x = \"lensHighlightsPixelShader\";\n const shader$2x = `uniform sampler2D textureSampler; \nuniform float gain;uniform float threshold;uniform float screen_width;uniform float screen_height;varying vec2 vUV;vec4 highlightColor(vec4 color) {vec4 highlight=color;float luminance=dot(highlight.rgb,vec3(0.2125,0.7154,0.0721));float lum_threshold;if (threshold>1.0) { lum_threshold=0.94+0.01*threshold; }\nelse { lum_threshold=0.5+0.44*threshold; }\nluminance=clamp((luminance-lum_threshold)*(1.0/(1.0-lum_threshold)),0.0,1.0);highlight*=luminance*gain;highlight.a=1.0;return highlight;}\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{vec4 original=texture2D(textureSampler,vUV);if (gain==-1.0) {gl_FragColor=vec4(0.0,0.0,0.0,1.0);return;}\nfloat w=2.0/screen_width;float h=2.0/screen_height;float weight=1.0;vec4 blurred=vec4(0.0,0.0,0.0,0.0);\n#ifdef PENTAGON\nblurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.84*w,0.43*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.48*w,-1.29*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.61*w,1.51*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.55*w,-0.74*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.71*w,-0.52*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.94*w,1.59*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.40*w,-1.87*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.62*w,1.16*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.09*w,0.25*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.46*w,-1.71*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.08*w,2.42*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.85*w,-1.89*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.89*w,0.16*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.29*w,1.88*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.40*w,-2.81*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.54*w,2.26*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.60*w,-0.61*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.31*w,-1.30*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.83*w,2.53*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.12*w,-2.48*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.60*w,1.11*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.82*w,0.99*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.50*w,-2.81*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.85*w,3.33*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.94*w,-1.92*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(3.27*w,-0.53*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.95*w,2.48*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.23*w,-3.04*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.17*w,2.05*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.97*w,-0.04*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.25*w,-2.00*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.31*w,3.08*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.94*w,-2.59*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(3.37*w,0.64*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-3.13*w,1.93*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.03*w,-3.65*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.60*w,3.17*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-3.14*w,-1.19*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(3.00*w,-1.19*h)));\n#else\nblurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.85*w,0.36*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.52*w,-1.14*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.46*w,1.42*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.46*w,-0.83*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.79*w,-0.42*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.11*w,1.62*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.29*w,-2.07*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.69*w,1.39*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.28*w,0.12*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.65*w,-1.69*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.08*w,2.44*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.63*w,-1.90*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.55*w,0.31*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.13*w,1.52*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.56*w,-2.61*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.38*w,2.34*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.64*w,-0.81*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.53*w,-1.21*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.06*w,2.63*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.00*w,-2.69*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.59*w,1.32*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.82*w,0.78*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.57*w,-2.50*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.54*w,2.93*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.39*w,-1.81*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(3.01*w,-0.28*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.04*w,2.25*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.02*w,-3.05*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.09*w,2.25*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-3.07*w,-0.25*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.44*w,-1.90*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.52*w,3.05*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.68*w,-2.61*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(3.01*w,0.79*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.76*w,1.46*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.05*w,-2.94*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.21*w,2.88*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.84*w,-1.30*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.98*w,-0.96*h)));\n#endif\nblurred/=39.0;gl_FragColor=blurred;}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2x] = shader$2x;\n\n // Do not edit.\n const name$2y = \"depthOfFieldPixelShader\";\n const shader$2y = `uniform sampler2D textureSampler;uniform sampler2D highlightsSampler;uniform sampler2D depthSampler;uniform sampler2D grainSampler;uniform float grain_amount;uniform bool blur_noise;uniform float screen_width;uniform float screen_height;uniform float distortion;uniform bool dof_enabled;uniform float screen_distance; \nuniform float aperture;uniform float darken;uniform float edge_blur;uniform bool highlights;uniform float near;uniform float far;varying vec2 vUV;\n#define PI 3.14159265\n#define TWOPI 6.28318530\n#define inverse_focal_length 0.1 \nvec2 centered_screen_pos;vec2 distorted_coords;float radius2;float radius;vec2 rand(vec2 co)\n{float noise1=(fract(sin(dot(co,vec2(12.9898,78.233)))*43758.5453));float noise2=(fract(sin(dot(co,vec2(12.9898,78.233)*2.0))*43758.5453));return clamp(vec2(noise1,noise2),0.0,1.0);}\nvec2 getDistortedCoords(vec2 coords) {if (distortion==0.0) { return coords; }\nvec2 direction=1.0*normalize(centered_screen_pos);vec2 dist_coords=vec2(0.5,0.5);dist_coords.x=0.5+direction.x*radius2*1.0;dist_coords.y=0.5+direction.y*radius2*1.0;float dist_amount=clamp(distortion*0.23,0.0,1.0);dist_coords=mix(coords,dist_coords,dist_amount);return dist_coords;}\nfloat sampleScreen(inout vec4 color,in vec2 offset,in float weight) {vec2 coords=distorted_coords;float angle=rand(coords*100.0).x*TWOPI;coords+=vec2(offset.x*cos(angle)-offset.y*sin(angle),offset.x*sin(angle)+offset.y*cos(angle));color+=texture2D(textureSampler,coords)*weight;return weight;}\nfloat getBlurLevel(float size) {return min(3.0,ceil(size/1.0));}\nvec4 getBlurColor(float size) {vec4 col=texture2D(textureSampler,distorted_coords);float blur_level=getBlurLevel(size);float w=(size/screen_width);float h=(size/screen_height);float total_weight=1.0;vec2 sample_coords;total_weight+=sampleScreen(col,vec2(-0.50*w,0.24*h),0.93);total_weight+=sampleScreen(col,vec2(0.30*w,-0.75*h),0.90);total_weight+=sampleScreen(col,vec2(0.36*w,0.96*h),0.87);total_weight+=sampleScreen(col,vec2(-1.08*w,-0.55*h),0.85);total_weight+=sampleScreen(col,vec2(1.33*w,-0.37*h),0.83);total_weight+=sampleScreen(col,vec2(-0.82*w,1.31*h),0.80);total_weight+=sampleScreen(col,vec2(-0.31*w,-1.67*h),0.78);total_weight+=sampleScreen(col,vec2(1.47*w,1.11*h),0.76);total_weight+=sampleScreen(col,vec2(-1.97*w,0.19*h),0.74);total_weight+=sampleScreen(col,vec2(1.42*w,-1.57*h),0.72);if (blur_level>1.0) {total_weight+=sampleScreen(col,vec2(0.01*w,2.25*h),0.70);total_weight+=sampleScreen(col,vec2(-1.62*w,-1.74*h),0.67);total_weight+=sampleScreen(col,vec2(2.49*w,0.20*h),0.65);total_weight+=sampleScreen(col,vec2(-2.07*w,1.61*h),0.63);total_weight+=sampleScreen(col,vec2(0.46*w,-2.70*h),0.61);total_weight+=sampleScreen(col,vec2(1.55*w,2.40*h),0.59);total_weight+=sampleScreen(col,vec2(-2.88*w,-0.75*h),0.56);total_weight+=sampleScreen(col,vec2(2.73*w,-1.44*h),0.54);total_weight+=sampleScreen(col,vec2(-1.08*w,3.02*h),0.52);total_weight+=sampleScreen(col,vec2(-1.28*w,-3.05*h),0.49);}\nif (blur_level>2.0) {total_weight+=sampleScreen(col,vec2(3.11*w,1.43*h),0.46);total_weight+=sampleScreen(col,vec2(-3.36*w,1.08*h),0.44);total_weight+=sampleScreen(col,vec2(1.80*w,-3.16*h),0.41);total_weight+=sampleScreen(col,vec2(0.83*w,3.65*h),0.38);total_weight+=sampleScreen(col,vec2(-3.16*w,-2.19*h),0.34);total_weight+=sampleScreen(col,vec2(3.92*w,-0.53*h),0.31);total_weight+=sampleScreen(col,vec2(-2.59*w,3.12*h),0.26);total_weight+=sampleScreen(col,vec2(-0.20*w,-4.15*h),0.22);total_weight+=sampleScreen(col,vec2(3.02*w,3.00*h),0.15);}\ncol/=total_weight; \nif (darken>0.0) {col.rgb*=clamp(0.3,1.0,1.05-size*0.5*darken);}\nreturn col;}\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{centered_screen_pos=vec2(vUV.x-0.5,vUV.y-0.5);radius2=centered_screen_pos.x*centered_screen_pos.x+centered_screen_pos.y*centered_screen_pos.y;radius=sqrt(radius2);distorted_coords=getDistortedCoords(vUV); \nvec2 texels_coords=vec2(vUV.x*screen_width,vUV.y*screen_height); \nfloat depth=texture2D(depthSampler,distorted_coords).r; \nfloat distance=near+(far-near)*depth; \nvec4 color=texture2D(textureSampler,vUV); \nfloat coc=abs(aperture*(screen_distance*(inverse_focal_length-1.0/distance)-1.0));if (dof_enabled==false || coc<0.07) { coc=0.0; }\nfloat edge_blur_amount=0.0;if (edge_blur>0.0) {edge_blur_amount=clamp((radius*2.0-1.0+0.15*edge_blur)*1.5,0.0,1.0)*1.3;}\nfloat blur_amount=max(edge_blur_amount,coc);if (blur_amount==0.0) {gl_FragColor=texture2D(textureSampler,distorted_coords);}\nelse {gl_FragColor=getBlurColor(blur_amount*1.7);if (highlights) {gl_FragColor.rgb+=clamp(coc,0.0,1.0)*texture2D(highlightsSampler,distorted_coords).rgb;}\nif (blur_noise) {vec2 noise=rand(distorted_coords)*0.01*blur_amount;vec2 blurred_coord=vec2(distorted_coords.x+noise.x,distorted_coords.y+noise.y);gl_FragColor=0.04*texture2D(textureSampler,blurred_coord)+0.96*gl_FragColor;}}\nif (grain_amount>0.0) {vec4 grain_color=texture2D(grainSampler,texels_coords*0.003);gl_FragColor.rgb+=(-0.5+grain_color.rgb)*0.30*grain_amount;}}\n`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2y] = shader$2y;\n\n /**\n * BABYLON.JS Chromatic Aberration GLSL Shader\n * Author: Olivier Guyot\n * Separates very slightly R, G and B colors on the edges of the screen\n * Inspired by Francois Tarlier & Martins Upitis\n */\n class LensRenderingPipeline extends PostProcessRenderPipeline {\n /**\n * @constructor\n *\n * Effect parameters are as follow:\n * {\n * chromatic_aberration: number; // from 0 to x (1 for realism)\n * edge_blur: number; // from 0 to x (1 for realism)\n * distortion: number; // from 0 to x (1 for realism), note that this will effect the pointer position precision\n * grain_amount: number; // from 0 to 1\n * grain_texture: BABYLON.Texture; // texture to use for grain effect; if unset, use random B&W noise\n * dof_focus_distance: number; // depth-of-field: focus distance; unset to disable (disabled by default)\n * dof_aperture: number; // depth-of-field: focus blur bias (default: 1)\n * dof_darken: number; // depth-of-field: darken that which is out of focus (from 0 to 1, disabled by default)\n * dof_pentagon: boolean; // depth-of-field: makes a pentagon-like \"bokeh\" effect\n * dof_gain: number; // depth-of-field: highlights gain; unset to disable (disabled by default)\n * dof_threshold: number; // depth-of-field: highlights threshold (default: 1)\n * blur_noise: boolean; // add a little bit of noise to the blur (default: true)\n * }\n * Note: if an effect parameter is unset, effect is disabled\n *\n * @param name The rendering pipeline name\n * @param parameters - An object containing all parameters (see above)\n * @param scene The scene linked to this pipeline\n * @param ratio The size of the postprocesses (0.5 means that your postprocess will have a width = canvas.width 0.5 and a height = canvas.height 0.5)\n * @param cameras The array of cameras that the rendering pipeline will be attached to\n */\n constructor(name, parameters, scene, ratio = 1.0, cameras) {\n super(scene.getEngine(), name);\n // Lens effects can be of the following:\n // - chromatic aberration (slight shift of RGB colors)\n // - blur on the edge of the lens\n // - lens distortion\n // - depth-of-field blur & highlights enhancing\n // - depth-of-field 'bokeh' effect (shapes appearing in blurred areas)\n // - grain effect (noise or custom texture)\n // Two additional texture samplers are needed:\n // - depth map (for depth-of-field)\n // - grain texture\n /**\n * @ignore\n * The chromatic aberration PostProcess id in the pipeline\n */\n this.LensChromaticAberrationEffect = \"LensChromaticAberrationEffect\";\n /**\n * @ignore\n * The highlights enhancing PostProcess id in the pipeline\n */\n this.HighlightsEnhancingEffect = \"HighlightsEnhancingEffect\";\n /**\n * @ignore\n * The depth-of-field PostProcess id in the pipeline\n */\n this.LensDepthOfFieldEffect = \"LensDepthOfFieldEffect\";\n this._pentagonBokehIsEnabled = false;\n this._scene = scene;\n // Fetch texture samplers\n this._depthTexture = scene.enableDepthRenderer().getDepthMap(); // Force depth renderer \"on\"\n if (parameters.grain_texture) {\n this._grainTexture = parameters.grain_texture;\n }\n else {\n this._createGrainTexture();\n }\n // save parameters\n this._edgeBlur = parameters.edge_blur ? parameters.edge_blur : 0;\n this._grainAmount = parameters.grain_amount ? parameters.grain_amount : 0;\n this._chromaticAberration = parameters.chromatic_aberration ? parameters.chromatic_aberration : 0;\n this._distortion = parameters.distortion ? parameters.distortion : 0;\n this._highlightsGain = parameters.dof_gain !== undefined ? parameters.dof_gain : -1;\n this._highlightsThreshold = parameters.dof_threshold ? parameters.dof_threshold : 1;\n this._dofDistance = parameters.dof_focus_distance !== undefined ? parameters.dof_focus_distance : -1;\n this._dofAperture = parameters.dof_aperture ? parameters.dof_aperture : 1;\n this._dofDarken = parameters.dof_darken ? parameters.dof_darken : 0;\n this._dofPentagon = parameters.dof_pentagon !== undefined ? parameters.dof_pentagon : true;\n this._blurNoise = parameters.blur_noise !== undefined ? parameters.blur_noise : true;\n // Create effects\n this._createChromaticAberrationPostProcess(ratio);\n this._createHighlightsPostProcess(ratio);\n this._createDepthOfFieldPostProcess(ratio / 4);\n // Set up pipeline\n this.addEffect(new PostProcessRenderEffect(scene.getEngine(), this.LensChromaticAberrationEffect, () => {\n return this._chromaticAberrationPostProcess;\n }, true));\n this.addEffect(new PostProcessRenderEffect(scene.getEngine(), this.HighlightsEnhancingEffect, () => {\n return this._highlightsPostProcess;\n }, true));\n this.addEffect(new PostProcessRenderEffect(scene.getEngine(), this.LensDepthOfFieldEffect, () => {\n return this._depthOfFieldPostProcess;\n }, true));\n if (this._highlightsGain === -1) {\n this._disableEffect(this.HighlightsEnhancingEffect, null);\n }\n // Finish\n scene.postProcessRenderPipelineManager.addPipeline(this);\n if (cameras) {\n scene.postProcessRenderPipelineManager.attachCamerasToRenderPipeline(name, cameras);\n }\n }\n /**\n * Get the class name\n * @returns \"LensRenderingPipeline\"\n */\n getClassName() {\n return \"LensRenderingPipeline\";\n }\n // Properties\n /**\n * Gets associated scene\n */\n get scene() {\n return this._scene;\n }\n /**\n * Gets or sets the edge blur\n */\n get edgeBlur() {\n return this._edgeBlur;\n }\n set edgeBlur(value) {\n this.setEdgeBlur(value);\n }\n /**\n * Gets or sets the grain amount\n */\n get grainAmount() {\n return this._grainAmount;\n }\n set grainAmount(value) {\n this.setGrainAmount(value);\n }\n /**\n * Gets or sets the chromatic aberration amount\n */\n get chromaticAberration() {\n return this._chromaticAberration;\n }\n set chromaticAberration(value) {\n this.setChromaticAberration(value);\n }\n /**\n * Gets or sets the depth of field aperture\n */\n get dofAperture() {\n return this._dofAperture;\n }\n set dofAperture(value) {\n this.setAperture(value);\n }\n /**\n * Gets or sets the edge distortion\n */\n get edgeDistortion() {\n return this._distortion;\n }\n set edgeDistortion(value) {\n this.setEdgeDistortion(value);\n }\n /**\n * Gets or sets the depth of field distortion\n */\n get dofDistortion() {\n return this._dofDistance;\n }\n set dofDistortion(value) {\n this.setFocusDistance(value);\n }\n /**\n * Gets or sets the darken out of focus amount\n */\n get darkenOutOfFocus() {\n return this._dofDarken;\n }\n set darkenOutOfFocus(value) {\n this.setDarkenOutOfFocus(value);\n }\n /**\n * Gets or sets a boolean indicating if blur noise is enabled\n */\n get blurNoise() {\n return this._blurNoise;\n }\n set blurNoise(value) {\n this._blurNoise = value;\n }\n /**\n * Gets or sets a boolean indicating if pentagon bokeh is enabled\n */\n get pentagonBokeh() {\n return this._pentagonBokehIsEnabled;\n }\n set pentagonBokeh(value) {\n if (value) {\n this.enablePentagonBokeh();\n }\n else {\n this.disablePentagonBokeh();\n }\n }\n /**\n * Gets or sets the highlight grain amount\n */\n get highlightsGain() {\n return this._highlightsGain;\n }\n set highlightsGain(value) {\n this.setHighlightsGain(value);\n }\n /**\n * Gets or sets the highlight threshold\n */\n get highlightsThreshold() {\n return this._highlightsThreshold;\n }\n set highlightsThreshold(value) {\n this.setHighlightsThreshold(value);\n }\n // public methods (self explanatory)\n /**\n * Sets the amount of blur at the edges\n * @param amount blur amount\n */\n setEdgeBlur(amount) {\n this._edgeBlur = amount;\n }\n /**\n * Sets edge blur to 0\n */\n disableEdgeBlur() {\n this._edgeBlur = 0;\n }\n /**\n * Sets the amount of grain\n * @param amount Amount of grain\n */\n setGrainAmount(amount) {\n this._grainAmount = amount;\n }\n /**\n * Set grain amount to 0\n */\n disableGrain() {\n this._grainAmount = 0;\n }\n /**\n * Sets the chromatic aberration amount\n * @param amount amount of chromatic aberration\n */\n setChromaticAberration(amount) {\n this._chromaticAberration = amount;\n }\n /**\n * Sets chromatic aberration amount to 0\n */\n disableChromaticAberration() {\n this._chromaticAberration = 0;\n }\n /**\n * Sets the EdgeDistortion amount\n * @param amount amount of EdgeDistortion\n */\n setEdgeDistortion(amount) {\n this._distortion = amount;\n }\n /**\n * Sets edge distortion to 0\n */\n disableEdgeDistortion() {\n this._distortion = 0;\n }\n /**\n * Sets the FocusDistance amount\n * @param amount amount of FocusDistance\n */\n setFocusDistance(amount) {\n this._dofDistance = amount;\n }\n /**\n * Disables depth of field\n */\n disableDepthOfField() {\n this._dofDistance = -1;\n }\n /**\n * Sets the Aperture amount\n * @param amount amount of Aperture\n */\n setAperture(amount) {\n this._dofAperture = amount;\n }\n /**\n * Sets the DarkenOutOfFocus amount\n * @param amount amount of DarkenOutOfFocus\n */\n setDarkenOutOfFocus(amount) {\n this._dofDarken = amount;\n }\n /**\n * Creates a pentagon bokeh effect\n */\n enablePentagonBokeh() {\n this._highlightsPostProcess.updateEffect(\"#define PENTAGON\\n\");\n this._pentagonBokehIsEnabled = true;\n }\n /**\n * Disables the pentagon bokeh effect\n */\n disablePentagonBokeh() {\n this._pentagonBokehIsEnabled = false;\n this._highlightsPostProcess.updateEffect();\n }\n /**\n * Enables noise blur\n */\n enableNoiseBlur() {\n this._blurNoise = true;\n }\n /**\n * Disables noise blur\n */\n disableNoiseBlur() {\n this._blurNoise = false;\n }\n /**\n * Sets the HighlightsGain amount\n * @param amount amount of HighlightsGain\n */\n setHighlightsGain(amount) {\n this._highlightsGain = amount;\n }\n /**\n * Sets the HighlightsThreshold amount\n * @param amount amount of HighlightsThreshold\n */\n setHighlightsThreshold(amount) {\n if (this._highlightsGain === -1) {\n this._highlightsGain = 1.0;\n }\n this._highlightsThreshold = amount;\n }\n /**\n * Disables highlights\n */\n disableHighlights() {\n this._highlightsGain = -1;\n }\n /**\n * Removes the internal pipeline assets and detaches the pipeline from the scene cameras\n * @param disableDepthRender If the scene's depth rendering should be disabled (default: false)\n */\n dispose(disableDepthRender = false) {\n this._scene.postProcessRenderPipelineManager.detachCamerasFromRenderPipeline(this._name, this._scene.cameras);\n this._chromaticAberrationPostProcess = null;\n this._highlightsPostProcess = null;\n this._depthOfFieldPostProcess = null;\n this._grainTexture.dispose();\n if (disableDepthRender) {\n this._scene.disableDepthRenderer();\n }\n }\n // colors shifting and distortion\n _createChromaticAberrationPostProcess(ratio) {\n this._chromaticAberrationPostProcess = new PostProcess(\"LensChromaticAberration\", \"chromaticAberration\", [\"chromatic_aberration\", \"screen_width\", \"screen_height\", \"direction\", \"radialIntensity\", \"centerPosition\"], // uniforms\n [], // samplers\n ratio, null, Texture.TRILINEAR_SAMPLINGMODE, this._scene.getEngine(), false);\n this._chromaticAberrationPostProcess.onApply = (effect) => {\n effect.setFloat(\"chromatic_aberration\", this._chromaticAberration);\n effect.setFloat(\"screen_width\", this._scene.getEngine().getRenderWidth());\n effect.setFloat(\"screen_height\", this._scene.getEngine().getRenderHeight());\n effect.setFloat(\"radialIntensity\", 1);\n effect.setFloat2(\"direction\", 17, 17);\n effect.setFloat2(\"centerPosition\", 0.5, 0.5);\n };\n }\n // highlights enhancing\n _createHighlightsPostProcess(ratio) {\n this._highlightsPostProcess = new PostProcess(\"LensHighlights\", \"lensHighlights\", [\"gain\", \"threshold\", \"screen_width\", \"screen_height\"], // uniforms\n [], // samplers\n ratio, null, Texture.TRILINEAR_SAMPLINGMODE, this._scene.getEngine(), false, this._dofPentagon ? \"#define PENTAGON\\n\" : \"\");\n this._highlightsPostProcess.externalTextureSamplerBinding = true;\n this._highlightsPostProcess.onApply = (effect) => {\n effect.setFloat(\"gain\", this._highlightsGain);\n effect.setFloat(\"threshold\", this._highlightsThreshold);\n effect.setTextureFromPostProcess(\"textureSampler\", this._chromaticAberrationPostProcess);\n effect.setFloat(\"screen_width\", this._scene.getEngine().getRenderWidth());\n effect.setFloat(\"screen_height\", this._scene.getEngine().getRenderHeight());\n };\n }\n // colors shifting and distortion\n _createDepthOfFieldPostProcess(ratio) {\n this._depthOfFieldPostProcess = new PostProcess(\"LensDepthOfField\", \"depthOfField\", [\n \"grain_amount\",\n \"blur_noise\",\n \"screen_width\",\n \"screen_height\",\n \"distortion\",\n \"dof_enabled\",\n \"screen_distance\",\n \"aperture\",\n \"darken\",\n \"edge_blur\",\n \"highlights\",\n \"near\",\n \"far\",\n ], [\"depthSampler\", \"grainSampler\", \"highlightsSampler\"], ratio, null, Texture.TRILINEAR_SAMPLINGMODE, this._scene.getEngine(), false);\n this._depthOfFieldPostProcess.externalTextureSamplerBinding = true;\n this._depthOfFieldPostProcess.onApply = (effect) => {\n effect.setTexture(\"depthSampler\", this._depthTexture);\n effect.setTexture(\"grainSampler\", this._grainTexture);\n effect.setTextureFromPostProcess(\"textureSampler\", this._highlightsPostProcess);\n effect.setTextureFromPostProcess(\"highlightsSampler\", this._depthOfFieldPostProcess);\n effect.setFloat(\"grain_amount\", this._grainAmount);\n effect.setBool(\"blur_noise\", this._blurNoise);\n effect.setFloat(\"screen_width\", this._scene.getEngine().getRenderWidth());\n effect.setFloat(\"screen_height\", this._scene.getEngine().getRenderHeight());\n effect.setFloat(\"distortion\", this._distortion);\n effect.setBool(\"dof_enabled\", this._dofDistance !== -1);\n effect.setFloat(\"screen_distance\", 1.0 / (0.1 - 1.0 / this._dofDistance));\n effect.setFloat(\"aperture\", this._dofAperture);\n effect.setFloat(\"darken\", this._dofDarken);\n effect.setFloat(\"edge_blur\", this._edgeBlur);\n effect.setBool(\"highlights\", this._highlightsGain !== -1);\n if (this._scene.activeCamera) {\n effect.setFloat(\"near\", this._scene.activeCamera.minZ);\n effect.setFloat(\"far\", this._scene.activeCamera.maxZ);\n }\n };\n }\n // creates a black and white random noise texture, 512x512\n _createGrainTexture() {\n const size = 512;\n this._grainTexture = new DynamicTexture(\"LensNoiseTexture\", size, this._scene, false, Texture.BILINEAR_SAMPLINGMODE);\n this._grainTexture.wrapU = Texture.WRAP_ADDRESSMODE;\n this._grainTexture.wrapV = Texture.WRAP_ADDRESSMODE;\n const context = this._grainTexture.getContext();\n const rand = (min, max) => {\n return Math.random() * (max - min) + min;\n };\n let value;\n for (let x = 0; x < size; x++) {\n for (let y = 0; y < size; y++) {\n value = Math.floor(rand(0.42, 0.58) * 255);\n context.fillStyle = \"rgb(\" + value + \", \" + value + \", \" + value + \")\";\n context.fillRect(x, y, 1, 1);\n }\n }\n this._grainTexture.update(false);\n }\n }\n\n /**\n * Contains all parameters needed for the prepass to perform\n * screen space subsurface scattering\n */\n class SSAO2Configuration {\n constructor() {\n /**\n * Is subsurface enabled\n */\n this.enabled = false;\n /**\n * Name of the configuration\n */\n this.name = \"ssao2\";\n /**\n * Textures that should be present in the MRT for this effect to work\n */\n this.texturesRequired = [6, 5];\n }\n }\n\n // Do not edit.\n const name$2z = \"ssao2PixelShader\";\n const shader$2z = `precision highp float;uniform sampler2D textureSampler;varying vec2 vUV;\n#ifdef SSAO\nfloat scales[16]=float[16](\n0.1,\n0.11406250000000001,\n0.131640625,\n0.15625,\n0.187890625,\n0.2265625,\n0.272265625,\n0.325,\n0.384765625,\n0.4515625,\n0.525390625,\n0.60625,\n0.694140625,\n0.7890625,\n0.891015625,\n1.0\n);uniform float near;uniform float radius;uniform sampler2D depthSampler;uniform sampler2D randomSampler;uniform sampler2D normalSampler;uniform float randTextureTiles;uniform float samplesFactor;uniform vec3 sampleSphere[SAMPLES];uniform float totalStrength;uniform float base;uniform float xViewport;uniform float yViewport;uniform mat3 depthProjection;uniform float maxZ;uniform float minZAspect;uniform vec2 texelSize;uniform mat4 projection;void main()\n{vec3 random=textureLod(randomSampler,vUV*randTextureTiles,0.0).rgb;float depth=textureLod(depthSampler,vUV,0.0).r;float depthSign=depth/abs(depth);depth=depth*depthSign;vec3 normal=textureLod(normalSampler,vUV,0.0).rgb;float occlusion=0.0;float correctedRadius=min(radius,minZAspect*depth/near);vec3 vViewRay=vec3((vUV.x*2.0-1.0)*xViewport,(vUV.y*2.0-1.0)*yViewport,depthSign);vec3 vDepthFactor=depthProjection*vec3(1.0,1.0,depth);vec3 origin=vViewRay*vDepthFactor;vec3 rvec=random*2.0-1.0;rvec.z=0.0;float dotProduct=dot(rvec,normal);rvec=1.0-abs(dotProduct)>1e-2 ? rvec : vec3(-rvec.y,0.0,rvec.x);vec3 tangent=normalize(rvec-normal*dot(rvec,normal));vec3 bitangent=cross(normal,tangent);mat3 tbn=mat3(tangent,bitangent,normal);float difference;for (int i=0; i1.0 || offset.y>1.0) {continue;}\nfloat sampleDepth=abs(textureLod(depthSampler,offset.xy,0.0).r);difference=depthSign*samplePosition.z-sampleDepth;float rangeCheck=1.0-smoothstep(correctedRadius*0.5,correctedRadius,difference);occlusion+=step(EPSILON,difference)*rangeCheck;}\nocclusion=occlusion*(1.0-smoothstep(maxZ*0.75,maxZ,depth));float ao=1.0-totalStrength*occlusion*samplesFactor;float result=clamp(ao+base,0.0,1.0);gl_FragColor=vec4(vec3(result),1.0);}\n#endif\n#ifdef BLUR\nuniform float outSize;uniform float soften;uniform float tolerance;uniform int samples;\n#ifndef BLUR_BYPASS\nuniform sampler2D depthSampler;\n#ifdef BLUR_LEGACY\n#define inline\nfloat blur13Bilateral(sampler2D image,vec2 uv,vec2 step) {float result=0.0;vec2 off1=vec2(1.411764705882353)*step;vec2 off2=vec2(3.2941176470588234)*step;vec2 off3=vec2(5.176470588235294)*step;float compareDepth=abs(textureLod(depthSampler,uv,0.0).r);float sampleDepth;float weight;float weightSum=30.0;result+=textureLod(image,uv,0.0).r*30.0;sampleDepth=abs(textureLod(depthSampler,uv+off1,0.0).r);weight=clamp(1.0/( 0.003+abs(compareDepth-sampleDepth)),0.0,30.0);weightSum+= weight;result+=textureLod(image,uv+off1,0.0).r*weight;sampleDepth=abs(textureLod(depthSampler,uv-off1,0.0).r);weight=clamp(1.0/( 0.003+abs(compareDepth-sampleDepth)),0.0,30.0);weightSum+= weight;result+=textureLod(image,uv-off1,0.0).r*weight;sampleDepth=abs(textureLod(depthSampler,uv+off2,0.0).r);weight=clamp(1.0/( 0.003+abs(compareDepth-sampleDepth)),0.0,30.0);weightSum+=weight;result+=textureLod(image,uv+off2,0.0).r*weight;sampleDepth=abs(textureLod(depthSampler,uv-off2,0.0).r);weight=clamp(1.0/( 0.003+abs(compareDepth-sampleDepth)),0.0,30.0);weightSum+=weight;result+=textureLod(image,uv-off2,0.0).r*weight;sampleDepth=abs(textureLod(depthSampler,uv+off3,0.0).r);weight=clamp(1.0/( 0.003+abs(compareDepth-sampleDepth)),0.0,30.0);weightSum+=weight;result+=textureLod(image,uv+off3,0.0).r*weight;sampleDepth=abs(textureLod(depthSampler,uv-off3,0.0).r);weight=clamp(1.0/( 0.003+abs(compareDepth-sampleDepth)),0.0,30.0);weightSum+=weight;result+=textureLod(image,uv-off3,0.0).r*weight;return result/weightSum;}\n#endif\n#endif\nvoid main()\n{float result=0.0;\n#ifdef BLUR_BYPASS\nresult=textureLod(textureSampler,vUV,0.0).r;\n#else\n#ifdef BLUR_H\nvec2 step=vec2(1.0/outSize,0.0);\n#else\nvec2 step=vec2(0.0,1.0/outSize);\n#endif\n#ifdef BLUR_LEGACY\nresult=blur13Bilateral(textureSampler,vUV,step);\n#else\nfloat compareDepth=abs(textureLod(depthSampler,vUV,0.0).r);float weightSum=0.0;for (int i=-samples; i {\n return this._originalColorPostProcess;\n }, true));\n this.addEffect(new PostProcessRenderEffect(scene.getEngine(), this.SSAORenderEffect, () => {\n return this._ssaoPostProcess;\n }, true));\n this.addEffect(new PostProcessRenderEffect(scene.getEngine(), this.SSAOBlurHRenderEffect, () => {\n return this._blurHPostProcess;\n }, true));\n this.addEffect(new PostProcessRenderEffect(scene.getEngine(), this.SSAOBlurVRenderEffect, () => {\n return this._blurVPostProcess;\n }, true));\n this.addEffect(new PostProcessRenderEffect(scene.getEngine(), this.SSAOCombineRenderEffect, () => {\n return this._ssaoCombinePostProcess;\n }, true));\n // Finish\n scene.postProcessRenderPipelineManager.addPipeline(this);\n if (cameras) {\n scene.postProcessRenderPipelineManager.attachCamerasToRenderPipeline(name, cameras);\n }\n }\n // Public Methods\n /**\n * Get the class name\n * @returns \"SSAO2RenderingPipeline\"\n */\n getClassName() {\n return \"SSAO2RenderingPipeline\";\n }\n /**\n * Removes the internal pipeline assets and detaches the pipeline from the scene cameras\n * @param disableGeometryBufferRenderer\n */\n dispose(disableGeometryBufferRenderer = false) {\n for (let i = 0; i < this._scene.cameras.length; i++) {\n const camera = this._scene.cameras[i];\n this._originalColorPostProcess.dispose(camera);\n this._ssaoPostProcess.dispose(camera);\n this._blurHPostProcess.dispose(camera);\n this._blurVPostProcess.dispose(camera);\n this._ssaoCombinePostProcess.dispose(camera);\n }\n this._randomTexture.dispose();\n if (disableGeometryBufferRenderer) {\n this._scene.disableGeometryBufferRenderer();\n }\n this._scene.postProcessRenderPipelineManager.detachCamerasFromRenderPipeline(this._name, this._scene.cameras);\n super.dispose();\n }\n // Private Methods\n /** @internal */\n _rebuild() {\n super._rebuild();\n }\n _getSamplersForBlur(disabled) {\n return disabled ? [\"textureSampler\"] : [\"textureSampler\", \"depthSampler\"];\n }\n _getDefinesForBlur(bilateral, disabled) {\n let define = \"#define BLUR\\n\";\n if (disabled) {\n define += \"#define BLUR_BYPASS\\n\";\n }\n if (!bilateral) {\n define += \"#define BLUR_LEGACY\\n\";\n }\n return { h: define + \"#define BLUR_H\\n\", v: define };\n }\n _createBlurPostProcess(ssaoRatio, blurRatio, textureType) {\n const defines = this._getDefinesForBlur(this.expensiveBlur, this.bypassBlur);\n const samplers = this._getSamplersForBlur(this.bypassBlur);\n this._blurHPostProcess = this._createBlurFilter(\"BlurH\", samplers, ssaoRatio, defines.h, textureType, true);\n this._blurVPostProcess = this._createBlurFilter(\"BlurV\", samplers, blurRatio, defines.v, textureType, false);\n }\n _createBlurFilter(name, samplers, ratio, defines, textureType, horizontal) {\n const blurFilter = new PostProcess(name, \"ssao2\", [\"outSize\", \"samples\", \"soften\", \"tolerance\"], samplers, ratio, null, Texture.BILINEAR_SAMPLINGMODE, this._scene.getEngine(), false, defines, textureType);\n blurFilter.onApply = (effect) => {\n if (!this._scene.activeCamera) {\n return;\n }\n const ssaoCombineSize = horizontal ? this._ssaoCombinePostProcess.width : this._ssaoCombinePostProcess.height;\n const originalColorSize = horizontal ? this._originalColorPostProcess.width : this._originalColorPostProcess.height;\n effect.setFloat(\"outSize\", ssaoCombineSize > 0 ? ssaoCombineSize : originalColorSize);\n effect.setInt(\"samples\", this.bilateralSamples);\n effect.setFloat(\"soften\", this.bilateralSoften);\n effect.setFloat(\"tolerance\", this.bilateralTolerance);\n if (this._geometryBufferRenderer) {\n effect.setTexture(\"depthSampler\", this._geometryBufferRenderer.getGBuffer().textures[0]);\n }\n else if (this._prePassRenderer) {\n effect.setTexture(\"depthSampler\", this._prePassRenderer.getRenderTarget().textures[this._prePassRenderer.getIndex(5)]);\n }\n };\n blurFilter.samples = this.textureSamples;\n return blurFilter;\n }\n //Van der Corput radical inverse\n _radicalInverse_VdC(i) {\n this._bits[0] = i;\n this._bits[0] = ((this._bits[0] << 16) | (this._bits[0] >> 16)) >>> 0;\n this._bits[0] = ((this._bits[0] & 0x55555555) << 1) | (((this._bits[0] & 0xaaaaaaaa) >>> 1) >>> 0);\n this._bits[0] = ((this._bits[0] & 0x33333333) << 2) | (((this._bits[0] & 0xcccccccc) >>> 2) >>> 0);\n this._bits[0] = ((this._bits[0] & 0x0f0f0f0f) << 4) | (((this._bits[0] & 0xf0f0f0f0) >>> 4) >>> 0);\n this._bits[0] = ((this._bits[0] & 0x00ff00ff) << 8) | (((this._bits[0] & 0xff00ff00) >>> 8) >>> 0);\n return this._bits[0] * 2.3283064365386963e-10; // / 0x100000000 or / 4294967296\n }\n _hammersley(i, n) {\n return [i / n, this._radicalInverse_VdC(i)];\n }\n _hemisphereSample_uniform(u, v) {\n const phi = v * 2.0 * Math.PI;\n // rejecting samples that are close to tangent plane to avoid z-fighting artifacts\n const cosTheta = 1.0 - u * 0.85;\n const sinTheta = Math.sqrt(1.0 - cosTheta * cosTheta);\n return new Vector3(Math.cos(phi) * sinTheta, Math.sin(phi) * sinTheta, cosTheta);\n }\n _generateHemisphere() {\n const numSamples = this.samples;\n const result = [];\n let vector;\n let i = 0;\n while (i < numSamples) {\n if (numSamples < 16) {\n vector = this._hemisphereSample_uniform(Math.random(), Math.random());\n }\n else {\n const rand = this._hammersley(i, numSamples);\n vector = this._hemisphereSample_uniform(rand[0], rand[1]);\n }\n result.push(vector.x, vector.y, vector.z);\n i++;\n }\n return result;\n }\n _getDefinesForSSAO() {\n const defines = `#define SSAO\\n#define SAMPLES ${this.samples}\\n#define EPSILON ${this.epsilon.toFixed(4)}`;\n return defines;\n }\n _createSSAOPostProcess(ratio, textureType) {\n this._sampleSphere = this._generateHemisphere();\n const defines = this._getDefinesForSSAO();\n const samplers = [\"randomSampler\", \"depthSampler\", \"normalSampler\"];\n this._ssaoPostProcess = new PostProcess(\"ssao2\", \"ssao2\", [\n \"sampleSphere\",\n \"samplesFactor\",\n \"randTextureTiles\",\n \"totalStrength\",\n \"radius\",\n \"base\",\n \"range\",\n \"projection\",\n \"near\",\n \"texelSize\",\n \"xViewport\",\n \"yViewport\",\n \"maxZ\",\n \"minZAspect\",\n \"depthProjection\",\n ], samplers, ratio, null, Texture.BILINEAR_SAMPLINGMODE, this._scene.getEngine(), false, defines, textureType);\n this._ssaoPostProcess.onApply = (effect) => {\n var _a, _b, _c, _d;\n if (!this._scene.activeCamera) {\n return;\n }\n effect.setArray3(\"sampleSphere\", this._sampleSphere);\n effect.setFloat(\"randTextureTiles\", 32.0);\n effect.setFloat(\"samplesFactor\", 1 / this.samples);\n effect.setFloat(\"totalStrength\", this.totalStrength);\n effect.setFloat2(\"texelSize\", 1 / this._ssaoPostProcess.width, 1 / this._ssaoPostProcess.height);\n effect.setFloat(\"radius\", this.radius);\n effect.setFloat(\"maxZ\", this.maxZ);\n effect.setFloat(\"minZAspect\", this.minZAspect);\n effect.setFloat(\"base\", this.base);\n effect.setFloat(\"near\", this._scene.activeCamera.minZ);\n if (this._scene.activeCamera.mode === Camera.PERSPECTIVE_CAMERA) {\n effect.setMatrix3x3(\"depthProjection\", SSAO2RenderingPipeline.PERSPECTIVE_DEPTH_PROJECTION);\n effect.setFloat(\"xViewport\", Math.tan(this._scene.activeCamera.fov / 2) * this._scene.getEngine().getAspectRatio(this._scene.activeCamera, true));\n effect.setFloat(\"yViewport\", Math.tan(this._scene.activeCamera.fov / 2));\n }\n else {\n const halfWidth = this._scene.getEngine().getRenderWidth() / 2.0;\n const halfHeight = this._scene.getEngine().getRenderHeight() / 2.0;\n const orthoLeft = (_a = this._scene.activeCamera.orthoLeft) !== null && _a !== void 0 ? _a : -halfWidth;\n const orthoRight = (_b = this._scene.activeCamera.orthoRight) !== null && _b !== void 0 ? _b : halfWidth;\n const orthoBottom = (_c = this._scene.activeCamera.orthoBottom) !== null && _c !== void 0 ? _c : -halfHeight;\n const orthoTop = (_d = this._scene.activeCamera.orthoTop) !== null && _d !== void 0 ? _d : halfHeight;\n effect.setMatrix3x3(\"depthProjection\", SSAO2RenderingPipeline.ORTHO_DEPTH_PROJECTION);\n effect.setFloat(\"xViewport\", (orthoRight - orthoLeft) * 0.5);\n effect.setFloat(\"yViewport\", (orthoTop - orthoBottom) * 0.5);\n }\n effect.setMatrix(\"projection\", this._scene.getProjectionMatrix());\n if (this._geometryBufferRenderer) {\n effect.setTexture(\"depthSampler\", this._geometryBufferRenderer.getGBuffer().textures[0]);\n effect.setTexture(\"normalSampler\", this._geometryBufferRenderer.getGBuffer().textures[1]);\n }\n else if (this._prePassRenderer) {\n effect.setTexture(\"depthSampler\", this._prePassRenderer.getRenderTarget().textures[this._prePassRenderer.getIndex(5)]);\n effect.setTexture(\"normalSampler\", this._prePassRenderer.getRenderTarget().textures[this._prePassRenderer.getIndex(6)]);\n }\n effect.setTexture(\"randomSampler\", this._randomTexture);\n };\n this._ssaoPostProcess.samples = this.textureSamples;\n if (!this._forceGeometryBuffer) {\n this._ssaoPostProcess._prePassEffectConfiguration = new SSAO2Configuration();\n }\n }\n _createSSAOCombinePostProcess(ratio, textureType) {\n this._ssaoCombinePostProcess = new PostProcess(\"ssaoCombine\", \"ssaoCombine\", [], [\"originalColor\", \"viewport\"], ratio, null, Texture.BILINEAR_SAMPLINGMODE, this._scene.getEngine(), false, undefined, textureType);\n this._ssaoCombinePostProcess.onApply = (effect) => {\n const viewport = this._scene.activeCamera.viewport;\n effect.setVector4(\"viewport\", TmpVectors.Vector4[0].copyFromFloats(viewport.x, viewport.y, viewport.width, viewport.height));\n effect.setTextureFromPostProcessOutput(\"originalColor\", this._originalColorPostProcess);\n };\n this._ssaoCombinePostProcess.samples = this.textureSamples;\n }\n _createRandomTexture() {\n const size = 128;\n this._randomTexture = new DynamicTexture(\"SSAORandomTexture\", size, this._scene, false, Texture.BILINEAR_SAMPLINGMODE);\n this._randomTexture.wrapU = Texture.WRAP_ADDRESSMODE;\n this._randomTexture.wrapV = Texture.WRAP_ADDRESSMODE;\n const context = this._randomTexture.getContext();\n const rand = (min, max) => {\n return Math.random() * (max - min) + min;\n };\n const randVector = Vector3.Zero();\n for (let x = 0; x < size; x++) {\n for (let y = 0; y < size; y++) {\n randVector.x = rand(0.0, 1.0);\n randVector.y = rand(0.0, 1.0);\n randVector.z = 0.0;\n randVector.normalize();\n randVector.scaleInPlace(255);\n randVector.x = Math.floor(randVector.x);\n randVector.y = Math.floor(randVector.y);\n context.fillStyle = \"rgb(\" + randVector.x + \", \" + randVector.y + \", \" + randVector.z + \")\";\n context.fillRect(x, y, 1, 1);\n }\n }\n this._randomTexture.update(false);\n }\n /**\n * Serialize the rendering pipeline (Used when exporting)\n * @returns the serialized object\n */\n serialize() {\n const serializationObject = SerializationHelper.Serialize(this);\n serializationObject.customType = \"SSAO2RenderingPipeline\";\n return serializationObject;\n }\n /**\n * Parse the serialized pipeline\n * @param source Source pipeline.\n * @param scene The scene to load the pipeline to.\n * @param rootUrl The URL of the serialized pipeline.\n * @returns An instantiated pipeline from the serialized object.\n */\n static Parse(source, scene, rootUrl) {\n return SerializationHelper.Parse(() => new SSAO2RenderingPipeline(source._name, scene, source._ratio, undefined, source._forceGeometryBuffer, source._textureType), source, scene, rootUrl);\n }\n }\n SSAO2RenderingPipeline.ORTHO_DEPTH_PROJECTION = [1, 0, 0, 0, 1, 0, 0, 0, 1];\n SSAO2RenderingPipeline.PERSPECTIVE_DEPTH_PROJECTION = [0, 0, 0, 0, 0, 0, 1, 1, 1];\n __decorate$1([\n serialize()\n ], SSAO2RenderingPipeline.prototype, \"totalStrength\", void 0);\n __decorate$1([\n serialize()\n ], SSAO2RenderingPipeline.prototype, \"maxZ\", void 0);\n __decorate$1([\n serialize()\n ], SSAO2RenderingPipeline.prototype, \"minZAspect\", void 0);\n __decorate$1([\n serialize(\"epsilon\")\n ], SSAO2RenderingPipeline.prototype, \"_epsilon\", void 0);\n __decorate$1([\n serialize(\"samples\")\n ], SSAO2RenderingPipeline.prototype, \"_samples\", void 0);\n __decorate$1([\n serialize(\"textureSamples\")\n ], SSAO2RenderingPipeline.prototype, \"_textureSamples\", void 0);\n __decorate$1([\n serialize()\n ], SSAO2RenderingPipeline.prototype, \"_forceGeometryBuffer\", void 0);\n __decorate$1([\n serialize()\n ], SSAO2RenderingPipeline.prototype, \"_ratio\", void 0);\n __decorate$1([\n serialize()\n ], SSAO2RenderingPipeline.prototype, \"_textureType\", void 0);\n __decorate$1([\n serialize()\n ], SSAO2RenderingPipeline.prototype, \"radius\", void 0);\n __decorate$1([\n serialize()\n ], SSAO2RenderingPipeline.prototype, \"base\", void 0);\n __decorate$1([\n serialize(\"bypassBlur\")\n ], SSAO2RenderingPipeline.prototype, \"_bypassBlur\", void 0);\n __decorate$1([\n serialize(\"expensiveBlur\")\n ], SSAO2RenderingPipeline.prototype, \"_expensiveBlur\", void 0);\n __decorate$1([\n serialize()\n ], SSAO2RenderingPipeline.prototype, \"bilateralSamples\", void 0);\n __decorate$1([\n serialize()\n ], SSAO2RenderingPipeline.prototype, \"bilateralSoften\", void 0);\n __decorate$1([\n serialize()\n ], SSAO2RenderingPipeline.prototype, \"bilateralTolerance\", void 0);\n RegisterClass(\"BABYLON.SSAO2RenderingPipeline\", SSAO2RenderingPipeline);\n\n // Do not edit.\n const name$2B = \"fxaaPixelShader\";\n const shader$2B = `#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE)\n#define TEXTUREFUNC(s,c,l) texture2DLodEXT(s,c,l)\n#else\n#define TEXTUREFUNC(s,c,b) texture2D(s,c,b)\n#endif\nuniform sampler2D textureSampler;uniform vec2 texelSize;varying vec2 vUV;varying vec2 sampleCoordS;varying vec2 sampleCoordE;varying vec2 sampleCoordN;varying vec2 sampleCoordW;varying vec2 sampleCoordNW;varying vec2 sampleCoordSE;varying vec2 sampleCoordNE;varying vec2 sampleCoordSW;const float fxaaQualitySubpix=1.0;const float fxaaQualityEdgeThreshold=0.166;const float fxaaQualityEdgeThresholdMin=0.0833;const vec3 kLumaCoefficients=vec3(0.2126,0.7152,0.0722);\n#define FxaaLuma(rgba) dot(rgba.rgb,kLumaCoefficients)\nvoid main(){vec2 posM;posM.x=vUV.x;posM.y=vUV.y;vec4 rgbyM=TEXTUREFUNC(textureSampler,vUV,0.0);float lumaM=FxaaLuma(rgbyM);float lumaS=FxaaLuma(TEXTUREFUNC(textureSampler,sampleCoordS,0.0));float lumaE=FxaaLuma(TEXTUREFUNC(textureSampler,sampleCoordE,0.0));float lumaN=FxaaLuma(TEXTUREFUNC(textureSampler,sampleCoordN,0.0));float lumaW=FxaaLuma(TEXTUREFUNC(textureSampler,sampleCoordW,0.0));float maxSM=max(lumaS,lumaM);float minSM=min(lumaS,lumaM);float maxESM=max(lumaE,maxSM);float minESM=min(lumaE,minSM);float maxWN=max(lumaN,lumaW);float minWN=min(lumaN,lumaW);float rangeMax=max(maxWN,maxESM);float rangeMin=min(minWN,minESM);float rangeMaxScaled=rangeMax*fxaaQualityEdgeThreshold;float range=rangeMax-rangeMin;float rangeMaxClamped=max(fxaaQualityEdgeThresholdMin,rangeMaxScaled);\n#ifndef MALI\nif(range=edgeVert;float subpixA=subpixNSWE*2.0+subpixNWSWNESE;if (!horzSpan)\n{lumaN=lumaW;}\nif (!horzSpan) \n{lumaS=lumaE;}\nif (horzSpan) \n{lengthSign=texelSize.y;}\nfloat subpixB=(subpixA*(1.0/12.0))-lumaM;float gradientN=lumaN-lumaM;float gradientS=lumaS-lumaM;float lumaNN=lumaN+lumaM;float lumaSS=lumaS+lumaM;bool pairN=abs(gradientN)>=abs(gradientS);float gradient=max(abs(gradientN),abs(gradientS));if (pairN)\n{lengthSign=-lengthSign;}\nfloat subpixC=clamp(abs(subpixB)*subpixRcpRange,0.0,1.0);vec2 posB;posB.x=posM.x;posB.y=posM.y;vec2 offNP;offNP.x=(!horzSpan) ? 0.0 : texelSize.x;offNP.y=(horzSpan) ? 0.0 : texelSize.y;if (!horzSpan) \n{posB.x+=lengthSign*0.5;}\nif (horzSpan)\n{posB.y+=lengthSign*0.5;}\nvec2 posN;posN.x=posB.x-offNP.x*1.5;posN.y=posB.y-offNP.y*1.5;vec2 posP;posP.x=posB.x+offNP.x*1.5;posP.y=posB.y+offNP.y*1.5;float subpixD=((-2.0)*subpixC)+3.0;float lumaEndN=FxaaLuma(TEXTUREFUNC(textureSampler,posN,0.0));float subpixE=subpixC*subpixC;float lumaEndP=FxaaLuma(TEXTUREFUNC(textureSampler,posP,0.0));if (!pairN) \n{lumaNN=lumaSS;}\nfloat gradientScaled=gradient*1.0/4.0;float lumaMM=lumaM-lumaNN*0.5;float subpixF=subpixD*subpixE;bool lumaMLTZero=lumaMM<0.0;lumaEndN-=lumaNN*0.5;lumaEndP-=lumaNN*0.5;bool doneN=abs(lumaEndN)>=gradientScaled;bool doneP=abs(lumaEndP)>=gradientScaled;if (!doneN) \n{posN.x-=offNP.x*3.0;}\nif (!doneN) \n{posN.y-=offNP.y*3.0;}\nbool doneNP=(!doneN) || (!doneP);if (!doneP) \n{posP.x+=offNP.x*3.0;}\nif (!doneP)\n{posP.y+=offNP.y*3.0;}\nif (doneNP)\n{if (!doneN) lumaEndN=FxaaLuma(TEXTUREFUNC(textureSampler,posN.xy,0.0));if (!doneP) lumaEndP=FxaaLuma(TEXTUREFUNC(textureSampler,posP.xy,0.0));if (!doneN) lumaEndN=lumaEndN-lumaNN*0.5;if (!doneP) lumaEndP=lumaEndP-lumaNN*0.5;doneN=abs(lumaEndN)>=gradientScaled;doneP=abs(lumaEndP)>=gradientScaled;if (!doneN) posN.x-=offNP.x*12.0;if (!doneN) posN.y-=offNP.y*12.0;doneNP=(!doneN) || (!doneP);if (!doneP) posP.x+=offNP.x*12.0;if (!doneP) posP.y+=offNP.y*12.0;}\nfloat dstN=posM.x-posN.x;float dstP=posP.x-posM.x;if (!horzSpan)\n{dstN=posM.y-posN.y;}\nif (!horzSpan) \n{dstP=posP.y-posM.y;}\nbool goodSpanN=(lumaEndN<0.0) != lumaMLTZero;float spanLength=(dstP+dstN);bool goodSpanP=(lumaEndP<0.0) != lumaMLTZero;float spanLengthRcp=1.0/spanLength;bool directionN=dstN {\n const texelSize = this.texelSize;\n effect.setFloat2(\"texelSize\", texelSize.x, texelSize.y);\n });\n }\n _getDefines() {\n const engine = this.getEngine();\n if (!engine) {\n return null;\n }\n const glInfo = engine.getGlInfo();\n if (glInfo && glInfo.renderer && glInfo.renderer.toLowerCase().indexOf(\"mali\") > -1) {\n return \"#define MALI 1\\n\";\n }\n return null;\n }\n /**\n * @internal\n */\n static _Parse(parsedPostProcess, targetCamera, scene, rootUrl) {\n return SerializationHelper.Parse(() => {\n return new FxaaPostProcess(parsedPostProcess.name, parsedPostProcess.options, targetCamera, parsedPostProcess.renderTargetSamplingMode, scene.getEngine(), parsedPostProcess.reusable);\n }, parsedPostProcess, scene, rootUrl);\n }\n }\n RegisterClass(\"BABYLON.FxaaPostProcess\", FxaaPostProcess);\n\n const SKYBOX_SIZE = 1000;\n const SamplingModeMap = {\n [exports.eTextureSamplingMode.nearest]: Texture.NEAREST_SAMPLINGMODE,\n [exports.eTextureSamplingMode.bilinear]: Texture.BILINEAR_SAMPLINGMODE,\n [exports.eTextureSamplingMode.trilinear]: Texture.TRILINEAR_SAMPLINGMODE,\n };\n class SceneRenderer extends Renderer {\n constructor(viewer, scene, gui, renderManager) {\n super(viewer, scene);\n this.gui = gui;\n this.renderManager = renderManager;\n this.yVector = new Vector3(0, 1, 0);\n this.modelToken = Token.SceneNode;\n }\n handlesNode(node) {\n return isSceneNode(node) && !node.isNested();\n }\n update3dInternal(node) {\n var _a;\n this.processEnvironment(node);\n //we create a transform node that ends up holding all space nodes in the scene\n //this creates parity with nested scenes and gives us something to transform in WebXR without re-parenting anything\n let tnode = this.get3d(node);\n if (!tnode) {\n tnode = new TransformNode(node._dynamicId, this.scene);\n this.set3d(node, tnode);\n if (!this.viewer.options.forceDisablePostProcess && this.scene.activeCamera) {\n if (node.enableFxaaPostProcess) {\n this.fxaaPostProcess = new FxaaPostProcess('fxaa', 1.0, this.scene.activeCamera, Texture.BILINEAR_SAMPLINGMODE, this.scene.getEngine());\n }\n if (node.enableLensEffectsPostProcess) {\n this.lensEffectPostProcess = new LensRenderingPipeline('lens', {\n edge_blur: 0.5,\n chromatic_aberration: 0.5,\n distortion: 0,\n dof_focus_distance: 200,\n dof_aperture: 5.0,\n grain_amount: 0.1,\n dof_pentagon: true,\n dof_gain: 1.0,\n dof_threshold: 1.0,\n dof_darken: 0,\n }, this.scene, 1.0, [this.scene.activeCamera]);\n }\n if (node.enableSSAOPostProcess) {\n this.ssaoPostProcess = new SSAO2RenderingPipeline('ssaopipeline', this.scene, 0.75);\n }\n }\n }\n if (node.changes.has('enableDeferredSceneUpdates')) {\n this.renderManager.maxRenderTime =\n node.enableDeferredSceneUpdates && !this.viewer.options.editorMode ? 20 : 20000;\n }\n if (node.changes.has('textureSamplingMode')) {\n const samplingMode = SamplingModeMap[node.textureSamplingMode];\n const materials = this.renderManager.material.getAllDefinedMaterials();\n materials.forEach(material => {\n material.getActiveTextures().forEach(texture => {\n if (texture) {\n texture.updateSamplingMode(samplingMode);\n }\n });\n });\n }\n if (node.changes.has('backgroundColor') || node.changes.has('background')) {\n if (node.background == exports.eBackground.color) {\n this.scene.clearColor = node.backgroundColor.toColor4();\n }\n else {\n this.scene.clearColor = new Color4(0, 0, 0, 0);\n }\n }\n if (node.changes.has('ambientColor')) {\n this.scene.ambientColor = node.ambientColor.toColor3();\n }\n if (node.changes.hasAny('renderMode', 'edgeWidth', 'edgeColor', 'edgeDisplayTolerance')) {\n this.viewer.setRenderMode(node.renderMode);\n }\n if (node.changes.has('enableHoverEffects')) {\n this.scene.constantlyUpdateMeshUnderPointer = node.enableHoverEffects;\n }\n if (node.changes.has('connectorSize')) {\n for (const c of node.getAllConnectors()) {\n this.viewer.rendererManager.connector.updateScale(c);\n }\n for (const c of node.getAllSketchControlPoints()) {\n this.viewer.rendererManager.controlPointRenderer.updateScale(c);\n }\n }\n if (node.changes.hasAny('dimensionUnitScalar', 'dimensionUnitSymbol')) {\n node.annotations.forEach(annotationNode => {\n if (isTrackedClass(annotationNode)) {\n annotationNode.isModified = true;\n }\n });\n }\n let gridNeedsRebuild = false;\n if (node.changes.hasAny('_forceGrid', 'gridSize', 'gridNormal', 'gridStepSize', 'gridMajorInterval', 'backgroundColor')) {\n if (node._forceGrid || node.enableGrid) {\n this.gui.disableGrid();\n gridNeedsRebuild = true;\n }\n if (node._forceGrid || node.enableGridAxes) {\n this.gui.disableGridAxes();\n gridNeedsRebuild = true;\n }\n }\n if (gridNeedsRebuild || node.changes.has('enableGrid')) {\n if (node._forceGrid || node.enableGrid) {\n this.gui.enableGrid(node, node._forceGrid || node.enableGridAxes);\n }\n else {\n this.gui.disableGrid();\n }\n }\n if (gridNeedsRebuild || node.changes.has('enableGridAxes')) {\n if (node._forceGrid || node.enableGridAxes) {\n this.gui.enableGridAxes(node);\n }\n else {\n this.gui.disableGridAxes();\n }\n }\n if (((_a = node._rootScene) === null || _a === void 0 ? void 0 : _a._dynamicId) === node._dynamicId) {\n const eventPayload = Object.fromEntries(node.changes);\n if (this.viewer.canvas) {\n this.viewer.canvas.dispatchEvent(new CustomEvent('kb3d.scenePropertyChange', { detail: eventPayload, bubbles: true }));\n }\n }\n }\n processEnvironment(node) {\n var _a;\n if (node.changes.has('environment') || node.changes.has('background') || node.changes.has('environmentBlur')) {\n const matService = (_a = node._manager) === null || _a === void 0 ? void 0 : _a.getService(Token.MaterialService);\n if (!matService) {\n throw Error('Cannot fetch environment material');\n }\n let path;\n if (node.background === exports.eBackground.environment) {\n if (node.environmentBlur < 0.15) {\n path = matService.getAbsoluteEnvironmentPath(node.environment, 3);\n }\n else {\n path = matService.getAbsoluteEnvironmentPath(node.environment, 1);\n }\n }\n else {\n path = matService.getAbsoluteEnvironmentPath(node.environment, 0);\n }\n if (path !== '' && this._envTexturePath !== path) {\n this._envTexturePath = path;\n if (this._envTexture) {\n this._envTexture.dispose();\n }\n let texture = undefined;\n if (path.match(/\\.dds(\\?.*)?$/) || path.match(/\\.env(\\?.*)?$/)) {\n texture = CubeTexture.CreateFromPrefilteredData(path, this.scene);\n }\n else if (path.match(/\\.hdr(\\?.*)?$/)) {\n texture = new HDRCubeTexture(path, this.scene, 128);\n }\n if (texture != null) {\n texture.gammaSpace = false;\n this.scene.environmentTexture = texture;\n this._envTexture = texture;\n }\n else {\n this.scene.environmentTexture = null;\n }\n }\n }\n if ((node.changes.has('background') || node.changes.has('environment')) &&\n node.background == exports.eBackground.environment &&\n this._envTexture) {\n if (!this.skybox) {\n this.skybox = this.scene.createDefaultSkybox(this._envTexture, true, SKYBOX_SIZE, node.environmentBlur);\n }\n else if (node.changes.has('environment')) {\n const skyMat = this.skybox.material;\n skyMat.reflectionTexture = this._envTexture.clone();\n skyMat.reflectionTexture.coordinatesMode = Texture.SKYBOX_MODE;\n }\n }\n if (node.changes.has('background') && node.background != exports.eBackground.environment) {\n if (this.skybox) {\n this.skybox.dispose();\n this.skybox = null;\n }\n }\n if (node.changes.has('environmentBlur')) {\n if (this.skybox) {\n const skyMat = this.skybox.material;\n skyMat.microSurface = 1.0 - node.environmentBlur;\n }\n }\n if (node.changes.has('environmentIntensity')) {\n this.scene.environmentIntensity = node.environmentIntensity;\n }\n if (node.changes.has('environmentRotation') && this.scene.environmentTexture) {\n if (this.scene.environmentTexture instanceof CubeTexture ||\n this.scene.environmentTexture instanceof HDRCubeTexture) {\n this.scene.environmentTexture.rotationY = (node.environmentRotation * Math.PI) / 180;\n if (this.skybox) {\n this.skybox.rotationQuaternion = Quaternion.RotationAxis(this.yVector, (node.environmentRotation * Math.PI) / 180);\n }\n }\n }\n }\n dispose() {\n super.dispose();\n if (this._envTexture) {\n this._envTexture.dispose();\n }\n if (this.skybox) {\n this.skybox.dispose();\n }\n if (this.ssaoPostProcess) {\n this.ssaoPostProcess.dispose();\n }\n if (this.lensEffectPostProcess) {\n this.lensEffectPostProcess.dispose();\n }\n if (this.fxaaPostProcess) {\n this.fxaaPostProcess.dispose();\n }\n }\n getSkyBox() {\n return this.skybox;\n }\n }\n\n const blendModeMap = {\n [exports.eTextureBlendMode.normal]: 'source-over',\n [exports.eTextureBlendMode.multiply]: 'multiply',\n [exports.eTextureBlendMode.screen]: 'screen',\n [exports.eTextureBlendMode.overlay]: 'overlay',\n [exports.eTextureBlendMode.darken]: 'darken',\n [exports.eTextureBlendMode.lighten]: 'lighten',\n [exports.eTextureBlendMode.hue]: 'hue',\n [exports.eTextureBlendMode.saturation]: 'saturation',\n [exports.eTextureBlendMode.color]: 'color',\n [exports.eTextureBlendMode.luminosity]: 'luminosity',\n [exports.eTextureBlendMode.difference]: 'difference',\n [exports.eTextureBlendMode.mask]: 'destination-in',\n [exports.eTextureBlendMode.vector]: 'kb3d-vector', // special blending mode\n };\n const workingCanvases = [];\n /** Make sure to dispose the canvas after use! Otherwise you'll get a memory leak! */\n function getWorkingCanvas(width, height) {\n let canvas;\n for (let i = 0; i < workingCanvases.length; i++) {\n if (workingCanvases[i].width === 1) {\n canvas = workingCanvases[i];\n break;\n }\n }\n if (!canvas) {\n canvas = document.createElement('canvas');\n workingCanvases.push(canvas);\n }\n canvas.width = width;\n canvas.height = height;\n const context = canvas.getContext('2d', {\n willReadFrequently: true,\n });\n context.clearRect(0, 0, canvas.width, canvas.height);\n return canvas;\n }\n function disposeWorkingCanvas(canvas) {\n canvas.width = 1;\n canvas.height = 1;\n }\n class TextureLayerRenderer extends Renderer {\n constructor() {\n super(...arguments);\n this.modelToken = Token.TextureLayer;\n }\n paintTexture(texture, context, parameters, operations) { }\n getNativeSize(texture) {\n return { width: 1, height: 1 };\n }\n update3dInternal(node) {\n this.viewer.updateTextureWarnings();\n }\n textureHasContent(node) {\n return false;\n }\n calculateOffsetAndDimensions(texture, ctxWidth, ctxHeight, sourceWidth, sourceHeight) {\n let offsetX = texture.uOffset * ctxWidth, offsetY = texture.vOffset * ctxHeight, drawHeight = sourceHeight, drawWidth = sourceWidth;\n switch (texture.verticalFit) {\n case exports.eTextureFit.contain:\n if (sourceHeight < ctxHeight) {\n break;\n }\n // falls through\n case exports.eTextureFit.stretch:\n drawHeight = ctxHeight;\n }\n switch (texture.horizontalFit) {\n case exports.eTextureFit.contain:\n if (sourceWidth < ctxWidth) {\n break;\n }\n // falls through\n case exports.eTextureFit.stretch:\n drawWidth = ctxWidth;\n }\n if (texture.maintainAspect) {\n const minRatio = Math.min(drawWidth / sourceWidth, drawHeight / sourceHeight);\n drawWidth = minRatio * sourceWidth;\n drawHeight = minRatio * sourceHeight;\n }\n switch (texture.verticalAlign) {\n // left is offset with no adjustment\n case exports.eTextureVerticalAlign.bottom:\n offsetY += ctxHeight - drawHeight;\n break;\n case exports.eTextureVerticalAlign.middle:\n offsetY += (ctxHeight - drawHeight) / 2;\n break;\n }\n switch (texture.horizontalAlign) {\n // top is offset with no adjustment\n case exports.eTextureHorizontalAlign.right:\n offsetX += ctxWidth - drawWidth;\n break;\n case exports.eTextureHorizontalAlign.center:\n offsetX += (ctxWidth - drawWidth) / 2;\n break;\n }\n return {\n offsetX,\n offsetY,\n drawHeight,\n drawWidth,\n };\n }\n drawInBlendedContext(texture, context, parameters, cb, operations) {\n if (operations.mask) {\n const maskingCtx = operations.mask.context;\n delete operations.mask;\n // Transfer transform from context to masking context to mask what is drawn\n const transform = context.getTransform();\n maskingCtx.setTransform(transform);\n context.setTransform(1, 0, 0, 1, 0, 0);\n maskingCtx.globalCompositeOperation = 'source-in';\n cb(maskingCtx);\n maskingCtx.setTransform(1, 0, 0, 1, 0, 0);\n this.drawInBlendedContext(texture, context, parameters, ctx => {\n ctx.drawImage(maskingCtx.canvas, 0, 0, ctx.canvas.width, ctx.canvas.height);\n }, operations);\n }\n else if (Object.keys(operations).length > 0) {\n const tempCanvas = getWorkingCanvas(context.canvas.width, context.canvas.height);\n const tempContext = tempCanvas.getContext('2d');\n cb(tempContext);\n this.filterImageInPlace(tempCanvas, operations);\n this.drawInBlendedContext(texture, context, parameters, ctx => {\n ctx.drawImage(tempCanvas, 0, 0, ctx.canvas.width, ctx.canvas.height);\n disposeWorkingCanvas(tempCanvas);\n }, operations);\n }\n else {\n context.globalAlpha = texture.strength / parameters.strength;\n switch (texture.blendMode) {\n case exports.eTextureBlendMode.vector: {\n const tempCanvas = getWorkingCanvas(context.canvas.width, context.canvas.height);\n const tempContext = tempCanvas.getContext('2d');\n cb(tempContext);\n const srcData = tempContext.getImageData(0, 0, context.canvas.width, context.canvas.height).data;\n const dstImageData = context.getImageData(0, 0, context.canvas.width, context.canvas.height);\n const dstData = dstImageData.data;\n const v1 = Vector3.Zero();\n const v2 = Vector3.Zero();\n const alphaScale = context.globalAlpha / 255;\n for (let i = 0, n = srcData.length; i < n; i += 4) {\n v1.copyFromFloats(dstData[i] / 128 - 1, dstData[i + 1] / 128 - 1, dstData[i + 2] / 256 - 0.5);\n v2.copyFromFloats(srcData[i] / 128 - 1, srcData[i + 1] / 128 - 1, srcData[i + 2] / 256 - 0.5);\n v2.scaleInPlace(srcData[i + 3] * alphaScale);\n v1.addInPlace(v2);\n v1.normalize();\n dstData[i] = (v1.x + 1) * 128;\n dstData[i + 1] = (v1.y + 1) * 128;\n dstData[i + 2] = (v1.z + 1) * 128;\n dstData[i + 3] = 255;\n }\n tempContext.putImageData(dstImageData, 0, 0);\n context.drawImage(tempCanvas, 0, 0, context.canvas.width, context.canvas.height);\n disposeWorkingCanvas(tempCanvas);\n break;\n }\n default: {\n context.globalCompositeOperation = blendModeMap[texture.blendMode] || 'source-over';\n cb(context);\n context.globalCompositeOperation = 'source-over';\n }\n }\n context.globalAlpha = 1;\n }\n }\n filterImageInPlace(buffer, operations) {\n const ctx = buffer.getContext('2d');\n const width = buffer.width;\n const height = buffer.height;\n if (Object.keys(operations).length === 0) {\n return buffer;\n }\n if (operations.colorize) {\n const tempCanvas = getWorkingCanvas(width, height);\n const tmpCtx = tempCanvas.getContext('2d');\n const color = operations.colorize.color;\n delete operations.colorize;\n tmpCtx.globalCompositeOperation = 'source-over';\n tmpCtx.fillStyle = `rgb(${color.r * 255}, ${color.g * 255}, ${color.b * 255})`;\n tmpCtx.fillRect(0, 0, ctx.canvas.width, ctx.canvas.height);\n tmpCtx.globalCompositeOperation = 'luminosity';\n tmpCtx.drawImage(buffer, 0, 0);\n tmpCtx.globalCompositeOperation = 'destination-in';\n tmpCtx.drawImage(buffer, 0, 0);\n ctx.drawImage(tempCanvas, 0, 0);\n disposeWorkingCanvas(tempCanvas);\n }\n if (Object.keys(operations).length === 0) {\n return buffer;\n }\n const imgData = ctx.getImageData(0, 0, width, height);\n const data = imgData.data;\n if (operations.normalRotate) {\n const rotateAngle = -1 * operations.normalRotate.angle;\n delete operations.normalRotate;\n const cosA = Math.cos((rotateAngle * Math.PI) / 180);\n const sinA = Math.sin((rotateAngle * Math.PI) / 180);\n for (let i = 0, n = data.length; i < n; i += 4) {\n const x1 = data[i] / 128 - 1;\n const y1 = data[i + 1] / 128 - 1;\n const x2 = cosA * x1 - sinA * y1;\n const y2 = sinA * x1 + cosA * y1;\n data[i] = (x2 + 1) * 128;\n data[i + 1] = (y2 + 1) * 128;\n }\n }\n if (operations.flipNormalsRg) {\n delete operations.flipNormalsRg;\n for (let i = 0, n = data.length; i < n; i += 4) {\n const r = data[i];\n data[i] = data[i + 1];\n data[i + 1] = r;\n }\n }\n ctx.putImageData(imgData, 0, 0);\n return buffer;\n }\n drawSourceToCanvas(texture, source, // | CanvasImageSource,\n context, parameters, operations) {\n var _a, _b;\n let width = source.width, height = source.height;\n const ctxWidth = context.canvas.width, ctxHeight = context.canvas.height;\n if (texture.colorize) {\n operations.colorize = { color: texture.colorize };\n }\n const firstTextureLayer = ((_a = texture._parent) === null || _a === void 0 ? void 0 : _a.textures.filter(t => t.channel === texture.channel && t.enabled).findIndex(t => t === texture)) === 0;\n if (texture.wAngle && texture.channel === exports.eTextureChannel.normal && !firstTextureLayer) {\n operations.normalRotate = { angle: texture.wAngle };\n }\n if (texture instanceof exports.FileTextureLayer &&\n texture.flipNormalsRg &&\n texture.channel === exports.eTextureChannel.normal) {\n operations.flipNormalsRg = true;\n }\n if (!firstTextureLayer) {\n const drawAt = this.calculateOffsetAndDimensions(texture, ctxWidth, ctxHeight, width, height);\n const pivotX = texture.uPivot * ctxWidth, pivotY = texture.vPivot * ctxHeight;\n if (texture.tile) {\n let patternCanvas;\n let patternAttempt;\n // In the case of stretch modes, the draw image size will differ from the native size so we have to redraw it first\n // iOS is unable to draw ImageBitmaps to patterns, so we have to try/catch for iOS\n if (drawAt.drawWidth === width && drawAt.drawHeight === height) {\n try {\n patternAttempt = context.createPattern(source, 'repeat');\n }\n catch (e) {\n // Do nothing, fallback below\n }\n }\n if (!patternAttempt) {\n patternCanvas = getWorkingCanvas(drawAt.drawWidth, drawAt.drawHeight);\n const patternContext = patternCanvas.getContext('2d');\n patternContext.drawImage(source, 0, 0, drawAt.drawWidth, drawAt.drawHeight);\n width = drawAt.drawWidth;\n height = drawAt.drawHeight;\n patternAttempt = context.createPattern(patternCanvas, 'repeat');\n }\n const pattern = patternAttempt;\n // Initial canvas transformations for angle and scaling\n context.translate(pivotX, pivotY);\n //if legacy mode enabled\n if (texture._parent &&\n texture._parent instanceof exports.MaterialNode &&\n texture._parent.legacyTextureStacking === false) {\n context.scale(texture.uScale, texture.vScale);\n context.rotate((texture.wAngle * Math.PI) / 180);\n }\n else {\n context.rotate((texture.wAngle * Math.PI) / 180);\n context.scale(texture.uScale, texture.vScale);\n if ((texture.uScale !== 1 || texture.vScale !== 1) && texture.wAngle !== 0) {\n console.warn(`Texture scale and angle are both altered on the texture layer \"${texture.name}\" with legacy mode enabled on the material \"${(_b = texture._parent) === null || _b === void 0 ? void 0 : _b.name}\". The order of operations for scale and angle are now reversed on new materials. Please disable legacy mode on this material and adjust the texture stack properties as needed to preserve the desired behavior when legacy mode is deprecated.`);\n }\n }\n // Remove extra offset since the image will be tiled anyways\n drawAt.offsetX = drawAt.offsetX % width;\n drawAt.offsetY = drawAt.offsetY % height;\n // Initially set the draw size to cover the canvas\n let drawWidth = (ctxWidth - drawAt.offsetX) / texture.uScale;\n let drawHeight = (ctxHeight - drawAt.offsetY) / texture.vScale;\n // If the offset is greater than zero there will be a gap on the top or the left so we have to move the context over one tile\n if (drawAt.offsetX > 0) {\n context.translate(-width, 0);\n drawWidth += width;\n }\n if (drawAt.offsetY > 0) {\n context.translate(0, -height);\n drawHeight += height;\n }\n // add the offset and remove the pivot from earlier that was used to get the correct rotation and scaling matrix parts\n context.translate(drawAt.offsetX - pivotX, drawAt.offsetY - pivotY);\n // If the texture was scaled with a pivot, we need to draw the image larger to make sure there are no gaps\n if ((texture.uPivot !== 0 || texture.vPivot !== 0) && (texture.uScale !== 1 || texture.vScale !== 1)) {\n const scaleOffsetU = Math.abs(Math.ceil(texture.uPivot / texture.uScale)) * width;\n const scaleOffsetV = Math.abs(Math.ceil(texture.vPivot / texture.vScale)) * height;\n if (scaleOffsetU !== 0 || scaleOffsetV !== 0) {\n context.translate(-scaleOffsetU, -scaleOffsetV);\n drawWidth += scaleOffsetU * 2;\n drawHeight += scaleOffsetV * 2;\n }\n }\n // If the texture was rotated, we need to draw the image larger to make sure there are no gaps\n if (texture.wAngle !== 0) {\n const rotateOffsetU = Math.ceil(texture.uPivot / texture.uScale) * width;\n const rotateOffsetV = Math.ceil(texture.vPivot / texture.vScale) * height;\n if (rotateOffsetU > 0 || rotateOffsetV > 0) {\n context.translate(-rotateOffsetU, -rotateOffsetV);\n drawWidth += rotateOffsetU * 2;\n drawHeight += rotateOffsetV * 2;\n }\n }\n this.drawInBlendedContext(texture, context, parameters, innerContext => {\n innerContext.fillStyle = pattern;\n // Note that any rectangle start point outside of the box will not be rendered so all image transforms must be done on the canvas context\n innerContext.fillRect(0, 0, drawWidth, drawHeight);\n if (patternCanvas) {\n disposeWorkingCanvas(patternCanvas);\n }\n }, operations);\n }\n else {\n context.translate(pivotX, pivotY);\n context.scale(texture.uScale, texture.vScale);\n context.rotate((texture.wAngle * Math.PI) / 180);\n context.translate(-pivotX, -pivotY);\n this.drawInBlendedContext(texture, context, parameters, innerContext => {\n innerContext.drawImage(source, drawAt.offsetX, drawAt.offsetY, drawAt.drawWidth, drawAt.drawHeight);\n }, operations);\n if (texture._visualize) {\n context.strokeStyle = '#00FF00';\n context.strokeRect(texture.uOffset * ctxWidth, texture.vOffset * ctxHeight, ctxWidth, ctxHeight);\n }\n }\n // reset the context\n context.setTransform(1, 0, 0, 1, 0, 0);\n }\n else {\n // this block will never need to be masked because the layers using a clipping mask will never be the first layer\n if (Object.keys(operations).length > 0) {\n if (source instanceof HTMLCanvasElement) {\n source = this.filterImageInPlace(source, operations);\n }\n else {\n const tempCanvas = getWorkingCanvas(width, height);\n const tempCtx = tempCanvas.getContext('2d');\n tempCtx.drawImage(source, 0, 0);\n source = this.filterImageInPlace(tempCanvas, operations);\n }\n }\n context.drawImage(source, 0, 0, ctxWidth, ctxHeight);\n }\n return context;\n }\n }\n\n class ColorFillTextureLayerRenderer extends TextureLayerRenderer {\n constructor() {\n super(...arguments);\n this.modelToken = Token.ColorFillTextureLayer;\n }\n paintTexture(texture, context, parameters, operations) {\n this.drawInBlendedContext(texture, context, parameters, innerContext => {\n innerContext.fillStyle = texture.color.cssString();\n innerContext.fillRect(0, 0, innerContext.canvas.width, innerContext.canvas.height);\n }, operations);\n }\n textureHasContent() {\n return true;\n }\n getNativeSize(texture) {\n const width = texture.width, height = texture.height;\n if (texture.wAngle !== 0) {\n return getRotatedDimensions(width, height, texture.wAngle);\n }\n else {\n return { width, height };\n }\n }\n update3dInternal(node) {\n super.update3dInternal(node);\n }\n }\n\n class FileTextureLayerRenderer extends TextureLayerRenderer {\n constructor() {\n super(...arguments);\n this.modelToken = Token.FileTextureLayer;\n }\n paintTexture(texture, context, parameters, operations) {\n if (texture._buffer) {\n return this.drawSourceToCanvas(texture, texture._buffer, context, parameters, operations);\n }\n }\n textureHasContent(node) {\n return node.path !== '';\n }\n getNativeSize(texture) {\n if (texture._buffer) {\n const width = Math.max(texture._buffer.width, 1), height = Math.max(texture._buffer.height, 1);\n if (texture.wAngle !== 0) {\n return getRotatedDimensions(width, height, texture.wAngle);\n }\n else {\n return { width, height };\n }\n }\n return { width: 1, height: 1 };\n }\n update3dInternal(node) {\n if (((node.changes.has('height') || node.changes.has('width')) && node.path.endsWith('.svg')) ||\n node.changes.has('path') ||\n (node.changes.has('enabled') && !node._buffer) ||\n node.changes.has('_init')) {\n node._needsImageUpdate = true;\n }\n if (node._needsImageUpdate && node.enabled && node.path && node._init) {\n this.loadTextureBuffer(node);\n }\n super.update3dInternal(node);\n }\n loadTextureBuffer(node) {\n if (node._parent instanceof exports.MaterialNode || node._parent instanceof exports.DisplacementMapFeature) {\n node._needsImageUpdate = false;\n if (node._buffer) {\n node.trackChanges(false);\n node._buffer = undefined; // Remove the buffer so the texture stack doesn't think the image is already loaded if the image was changed\n node.trackChanges(true);\n }\n node._parent._materialService.getTextureImage(node).subscribe(resp => {\n node._buffer = resp.bitmap;\n node._bufferSource = resp.url;\n });\n }\n else {\n console.warn('Texure is not attached to a material node. Cannot load texture image bitmap.');\n }\n }\n }\n\n function base64ArrayBuffer(arrayBuffer) {\n let base64 = '';\n const encodings = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';\n const bytes = new Uint8Array(arrayBuffer);\n const byteLength = bytes.byteLength;\n const byteRemainder = byteLength % 3;\n const mainLength = byteLength - byteRemainder;\n let a, b, c, d;\n let chunk;\n // Main loop deals with bytes in chunks of 3\n for (let i = 0; i < mainLength; i = i + 3) {\n // Combine the three bytes into a single integer\n chunk = (bytes[i] << 16) | (bytes[i + 1] << 8) | bytes[i + 2];\n // Use bitmasks to extract 6-bit segments from the triplet\n a = (chunk & 16515072) >> 18; // 16515072 = (2^6 - 1) << 18\n b = (chunk & 258048) >> 12; // 258048 = (2^6 - 1) << 12\n c = (chunk & 4032) >> 6; // 4032 = (2^6 - 1) << 6\n d = chunk & 63; // 63 = 2^6 - 1\n // Convert the raw binary segments to the appropriate ASCII encoding\n base64 += encodings[a] + encodings[b] + encodings[c] + encodings[d];\n }\n // Deal with the remaining bytes and padding\n if (byteRemainder == 1) {\n chunk = bytes[mainLength];\n a = (chunk & 252) >> 2; // 252 = (2^6 - 1) << 2\n // Set the 4 least significant bits to zero\n b = (chunk & 3) << 4; // 3 = 2^2 - 1\n base64 += encodings[a] + encodings[b] + '==';\n }\n else if (byteRemainder == 2) {\n chunk = (bytes[mainLength] << 8) | bytes[mainLength + 1];\n a = (chunk & 64512) >> 10; // 64512 = (2^6 - 1) << 10\n b = (chunk & 1008) >> 4; // 1008 = (2^6 - 1) << 4\n // Set the 2 least significant bits to zero\n c = (chunk & 15) << 2; // 15 = 2^4 - 1\n base64 += encodings[a] + encodings[b] + encodings[c] + '=';\n }\n return base64;\n }\n\n class SvgTextureLayerRenderer extends TextureLayerRenderer {\n constructor() {\n super(...arguments);\n this.modelToken = Token.FileTextureLayer;\n this.imageElement = document.createElement('img');\n this.svgHelpers = new Map();\n }\n paintTexture(texture, context, parameters, operations) {\n const helper = this.svgHelpers.get(texture._dynamicId);\n if (texture._raster &&\n helper &&\n context.canvas.width === helper.renderedWidth &&\n context.canvas.height === helper.renderedHeight) {\n return this.drawSourceToCanvas(texture, texture._raster, context, parameters, operations);\n }\n else {\n let fontLoadPromise;\n if (texture.preloadFontStylesheet) {\n const fontFacesFound = [...texture.svgXml.matchAll(/font-family\\s*[=:]\\s*[\"'](.+?)[\"']/g)];\n const fontsToLoad = [];\n for (const match of fontFacesFound) {\n fontsToLoad.push(match[1]);\n }\n if (fontsToLoad.length > 0) {\n const manager = this.viewer.sceneNode._manager;\n const fontService = manager.getService(Token.FontService);\n fontLoadPromise = fontService.loadFontData(texture.preloadFontStylesheet, fontsToLoad);\n }\n else {\n fontLoadPromise = Promise.resolve();\n }\n }\n else {\n fontLoadPromise = Promise.resolve();\n }\n fontLoadPromise\n .then(result => this.generateSvg(texture, context.canvas.width, context.canvas.height, result || undefined))\n .then(bitmap => {\n if (bitmap) {\n texture._raster = bitmap;\n }\n })\n .catch(() => {\n // Don't need to do anything\n });\n }\n }\n textureHasContent(node) {\n return !!node.svgXml;\n }\n getNativeSize(texture) {\n const width = texture.width, height = texture.height;\n if (texture.wAngle !== 0) {\n return getRotatedDimensions(width, height, texture.wAngle);\n }\n else {\n return { width, height };\n }\n }\n update3dInternal(node) {\n if (node.changes.has('svgXml') ||\n node.changes.has('width') ||\n node.changes.has('height') ||\n node.changes.has('preloadFontStylesheet')) {\n node._raster = undefined;\n }\n super.update3dInternal(node);\n }\n generateSvg(texture, width, height, embedFonts) {\n if (texture.svgXml !== '' && width > 0 && height > 0) {\n let helper = this.svgHelpers.get(texture._dynamicId);\n if (!helper) {\n helper = {\n imgElement: document.createElement('img'),\n renderedHeight: height,\n renderedWidth: width,\n };\n this.svgHelpers.set(texture._dynamicId, helper);\n }\n else {\n helper.renderedHeight = height;\n helper.renderedWidth = width;\n }\n try {\n const imageElement = this.svgHelpers.get(texture._dynamicId).imgElement;\n let svgString = texture.svgXml;\n // Adding svg required tags\n if (!svgString.match(/` +\n svgString;\n }\n if (!svgString.match(/<\\?xml/)) {\n svgString = `` + svgString;\n }\n if (!svgString.match(/]*xmlns=[^>]*>/)) {\n svgString = svgString.replace(//m);\n if (!match) {\n console.warn('Invalid SVG');\n return Promise.reject();\n }\n else {\n if (embedFonts) {\n let fontStyles = '';\n for (const fontFamily in embedFonts) {\n const arrayBuffer = embedFonts[fontFamily];\n fontStyles += ` \r\n@font-face {\r\n font-family: \"${fontFamily}\";\r\n src: url(\"data:application/font-woff;charset=utf-8;base64,${base64ArrayBuffer(arrayBuffer)}\");\r\n}\r\n `;\n }\n const offset = (match.index || 0) + match[0].length;\n svgString =\n svgString.substring(0, offset) +\n `` +\n svgString.substring(offset);\n }\n const blobUrl = URL.createObjectURL(new Blob([svgString], { type: 'image/svg+xml;charset=utf-8' }));\n imageElement.src = blobUrl;\n const generateImagePromise = imageElement.decode().then(() => {\n // Don't use the img element from the closure so it can get garbage collected if this function isn't called!\n // There's no way to explicitly dispose of a DOM element\n const loadedImageElement = this.svgHelpers.get(texture._dynamicId).imgElement;\n // Figure out what resolution to actually draw the SVG at to fit within the envelope\n const scaled = scaleImagePreserveAspect(width, height, loadedImageElement.naturalWidth, loadedImageElement.naturalHeight);\n return createImageBitmap(loadedImageElement, {\n resizeWidth: scaled.drawWidth,\n resizeHeight: scaled.drawHeight,\n });\n }, err => {\n console.warn('Could not decode SVG XML: ', svgString);\n });\n this.viewer.deferSceneReady(generateImagePromise);\n return generateImagePromise;\n }\n }\n catch (e) {\n console.warn('Could not generate SVG Texture: ', e);\n Promise.reject(e);\n }\n }\n return Promise.reject();\n }\n delete3d(node) {\n const helper = this.svgHelpers.get(node._dynamicId);\n if (helper) {\n this.svgHelpers.delete(node._dynamicId);\n }\n super.delete3d(node);\n }\n }\n\n const textureChannelMap = {\n // General\n [exports.eTextureChannel.emissive]: 'emissiveTexture',\n [exports.eTextureChannel.ambientOcclusion]: 'lightmapTexture',\n // PBR\n [exports.eTextureChannel.albedo]: 'albedoTexture',\n [exports.eTextureChannel.normal]: 'bumpTexture',\n [exports.eTextureChannel.ambient]: 'ambientTexture',\n [exports.eTextureChannel.opacity]: 'opacityTexture',\n [exports.eTextureChannel.metallic]: 'metallicTexture',\n [exports.eTextureChannel.reflection]: 'reflectionTexture',\n [exports.eTextureChannel.specular]: 'reflectivityTexture',\n [exports.eTextureChannel.roughness]: 'microSurfaceTexture',\n [exports.eTextureChannel.glossiness]: 'microSurfaceTexture',\n [exports.eTextureChannel.anisotropyTangent]: ['anisotropy', 'texture'],\n [exports.eTextureChannel.clearCoatNormal]: ['clearCoat', 'bumpTexture'],\n [exports.eTextureChannel.subsurfaceThickness]: ['subSurface', 'thicknessTexture'],\n };\n const maskingCanvas = document.createElement('canvas');\n class TextureStackRenderer extends Renderer {\n constructor(viewer, scene, rendererManager) {\n super(viewer, scene);\n this.rendererManager = rendererManager;\n this.modelToken = Token.TextureLayer;\n this.queuedTexturesToProcess = new Map();\n this.queueSize = new Subject();\n this.renderers = new Map();\n this.noMipMapFallback = false;\n this.updateTextureStackRequest = new Subject();\n this.deferCalled = false;\n const fileTexture = new FileTextureLayerRenderer(viewer, scene);\n const svgTexture = new SvgTextureLayerRenderer(viewer, scene);\n const colorTexture = new ColorFillTextureLayerRenderer(viewer, scene);\n this.renderers.set(Token.FileTextureLayer, fileTexture);\n this.renderers.set(Token.SvgTextureLayer, svgTexture);\n this.renderers.set(Token.ColorFillTextureLayer, colorTexture);\n /**\n * See render.ts comment - this delay must be added to the render as a change can take up to this delay time to be displayed.\n * We don't want to repeatedly processes texture changes coming in quick succession, which often happens if there are many textures being loaded at once.\n * */\n const throttleDebounceTime = 800;\n merge(this.updateTextureStackRequest.pipe(throttleTime(throttleDebounceTime)), this.updateTextureStackRequest.pipe(debounceTime(throttleDebounceTime))).subscribe(() => {\n this.updateTextureStacksExec();\n });\n // This code works for forcing intel gpus to not use mipmaps, but some scenes don't have artifacting issues and need mipmaps. So\n // In the future we can introduce a scene setting to enable the intel fallback\n // if (this.viewer.glInfo.match(/Intel.*Iris.*Xe/)) {\n // this.noMipMapFallback = true;\n // }\n }\n materialDeleted(node) {\n for (const channel in exports.eTextureChannel) {\n const textureId = this.getTexture3dId(node, channel);\n const texture = this.babylonMap.get(textureId);\n if (texture) {\n this.disposeTexture(texture);\n this.babylonMap.delete(this.getTexture3dId(node, channel));\n }\n this.babylonMap.delete(textureId);\n }\n }\n /** This should only be called from the editor. Has a very heavy performance impact. */\n forceRebuildTexture(materialNode, channel) {\n if (channel) {\n this.resetTextureStack(materialNode, channel);\n }\n else {\n for (const channel in exports.eTextureChannel) {\n this.resetTextureStack(materialNode, channel);\n }\n }\n this.updateTextureStacksInternal(materialNode);\n }\n resetTextureStack(texturedNode, channel, setParameters) {\n let material;\n let materialNode;\n if (isMaterialNode(texturedNode)) {\n materialNode = texturedNode;\n if (!materialNode._initialized) {\n return;\n }\n material = this.rendererManager.material.get3d(materialNode);\n if (!material) {\n // There might not be a material in babylon if the material was deleted prior to this renderer running.\n // console.warn(`material not generated from ${materialNode.name}`);\n return;\n }\n }\n const hasTextureInChannel = !!texturedNode.textures.find(t => t.channel === channel);\n if (!hasTextureInChannel) {\n return;\n }\n let channelsToProcess = this.queuedTexturesToProcess.get(texturedNode);\n if (!channelsToProcess) {\n channelsToProcess = new Map();\n this.queuedTexturesToProcess.set(texturedNode, channelsToProcess);\n }\n // Here we don't want to filter out textures with no data because they cam still be used to set the canvas size\n const textureLayers = this.getActiveLayers(texturedNode, channel);\n // Register that the channel needs to be processed.\n // Figure out the highest strength among the texture layers so we can\n // calculate the strength of each one as a fraction of that.\n let channelStrength = 0;\n textureLayers.forEach(textureLayer => {\n channelStrength = Math.max(channelStrength, textureLayer.strength);\n });\n const existingProcess = channelsToProcess.get(channel);\n channelsToProcess.set(channel, {\n strength: channelStrength,\n hasTransparency: materialNode ? materialNode.hasTransparency : false,\n ...(existingProcess || {}),\n ...(setParameters || {}),\n });\n const propName3d = textureChannelMap[channel];\n if (material && materialNode) {\n // Reset the material to re-run the texture stack (or clear the texture if there are no textures anymore)\n if (textureLayers.length === 0) {\n // If no textures are using this channel anymore, then remove it from the material\n if (Array.isArray(propName3d)) {\n const res = getObjectPropertyAtPath(material, propName3d, false);\n if (res.obj && res.obj[res.prop]) {\n const texture = res.obj[res.prop];\n if (texture instanceof Texture) {\n this.disposeTexture(texture);\n res.obj[res.prop] = null;\n this.babylonMap.delete(this.getTexture3dId(materialNode, channel));\n }\n }\n }\n else {\n if (material && material[propName3d]) {\n material[propName3d].dispose();\n material[propName3d] = null;\n this.babylonMap.delete(this.getTexture3dId(materialNode, channel));\n }\n }\n }\n else {\n const texture = this.getTexture(materialNode, channel);\n if (texture instanceof DynamicTexture) {\n const baseImageSize = this.getRenderer(textureLayers[0]).getNativeSize(textureLayers[0]);\n const context = texture.getContext();\n if (this.scene.getEngine().webGLVersion < 2) {\n const clampedSize = getBinClampedSize(baseImageSize.width, baseImageSize.height);\n context.canvas.width = clampedSize;\n context.canvas.height = clampedSize;\n }\n else {\n context.canvas.width = baseImageSize.width;\n context.canvas.height = baseImageSize.height;\n }\n context.clearRect(0, 0, context.canvas.width, context.canvas.height);\n }\n }\n }\n }\n update3dInternal(feature) {\n if (isTrackedClass(feature)) {\n if (!feature.enabled || feature._suppressed) {\n if (feature.changes.has('enabled') || feature.changes.has('_suppressed')) {\n this.delete3d(feature);\n }\n return;\n }\n const renderer = this.renderers.get(Token.withKey(feature.$type));\n if (!renderer) {\n throw Error('no renderer for feature');\n }\n if (feature._parent && isFeatureNode(feature._parent)) {\n feature.init(feature._parent);\n }\n renderer.update3d(feature);\n if (feature._parent && isFeatureNode(feature._parent) && isTrackedClass(feature._parent)) {\n feature._parent.setPropertyIsModified('textures', true);\n }\n if (feature._parent) {\n this.resetTextureStack(feature._parent, feature.channel);\n }\n }\n }\n updateTextureStacks() {\n if (!this.deferCalled && this.queuedTexturesToProcess.size > 0) {\n const queueLive = this.queueSize.pipe(takeWhile(size => size > 0));\n const p = queueLive.toPromise();\n this.viewer.deferSceneReady(p);\n this.deferCalled = true;\n }\n for (const node of this.queuedTexturesToProcess.keys()) {\n if (isFeatureNode(node) || node._initialized) {\n for (const texture of node.textures) {\n if (!texture._init) {\n texture.init(node);\n }\n }\n }\n }\n this.updateTextureStackRequest.next();\n }\n updateTextureStacksExec() {\n this.queueSize.next(this.queuedTexturesToProcess.size);\n for (const node of this.queuedTexturesToProcess.keys()) {\n this.updateTextureStacksInternal(node);\n if (this.queuedTexturesToProcess.size > 0) {\n this.updateTextureStacks();\n }\n // Low perf code to prevent iOS from erroring out. Doesn't seem necessary for now\n // return;\n // End low perf code\n }\n this.queueSize.next(this.queuedTexturesToProcess.size);\n }\n updateTextureStacksInternal(node) {\n if (this.queuedTexturesToProcess.has(node)) {\n if (isMaterialNode(node)) {\n const channels = this.queuedTexturesToProcess.get(node);\n const material = this.rendererManager.material.get3d(node);\n if (!channels || !material) {\n return;\n }\n // Process texture stack\n channels.forEach((parameters, channel) => {\n // Put the texture in the right place on the material\n const propName3d = textureChannelMap[channel];\n if (instanceOf(node, Token.AdvancedMaterialNode)) {\n if (node.workflowMode === exports.eMaterialWorkflow.metallicRoughness &&\n channel === exports.eTextureChannel.glossiness) {\n return;\n }\n else if (node.workflowMode === exports.eMaterialWorkflow.specularGlossiness &&\n channel === exports.eTextureChannel.roughness) {\n return;\n }\n }\n const texture = this.getTexture(node, channel);\n // If texture is undefined, that means that there is a missing setting/layers that means we don't need a texture for this channel\n if (texture) {\n // Paint the image onto the texture\n // We don't want to count layers with no content in the stack, like file textures with no file or svg textures with no data\n const textureLayers = this.getActiveLayers(node, channel).filter(node => {\n const renderer = this.getRenderer(node);\n return renderer.textureHasContent(node);\n });\n if (textureLayers.length > 0) {\n // Put the texture in the right place on the material if it's not already\n if (Array.isArray(propName3d)) {\n const res = getObjectPropertyAtPath(material, propName3d);\n if (res.obj[res.prop] !== texture) {\n res.obj[res.prop] = texture;\n }\n }\n else {\n if (material[propName3d] !== texture) {\n material[propName3d] = texture;\n }\n }\n if (texture instanceof DynamicTexture) {\n const context = texture.getContext();\n if (!(context instanceof CanvasRenderingContext2D)) {\n throw Error('Unexpected rendering context');\n }\n if (channel !== exports.eTextureChannel.albedo) {\n // Fill texture with color that makes sense depending on the texture channel\n switch (channel) {\n case exports.eTextureChannel.normal: {\n context.fillStyle = 'rgb(128, 128, 255)';\n break;\n }\n case exports.eTextureChannel.opacity:\n case exports.eTextureChannel.ambient:\n case exports.eTextureChannel.ambientOcclusion: {\n context.fillStyle = 'rgb(255, 255, 255)';\n break;\n }\n default: {\n context.fillStyle = 'rgb(0, 0, 0)';\n break;\n }\n }\n context.fillRect(0, 0, context.canvas.width, context.canvas.height);\n }\n this.paintTextureOnCanvas(textureLayers, context, parameters);\n updateTextureProps(texture, parameters);\n texture.update(true); // invertY\n }\n else {\n // If texture is not a dynamic texture, then it's a basic texture and there is only one texture layer. We just have to make sure the path is correct\n const fileTextureLayer = textureLayers[0];\n if (!(fileTextureLayer instanceof exports.FileTextureLayer)) {\n throw Error('Unexpected texture type');\n }\n else {\n if (fileTextureLayer._parent instanceof exports.MaterialNode) {\n const path = fileTextureLayer._bufferSource || '';\n if (texture.url !== path && fileTextureLayer._buffer) {\n texture.updateURL(path, (fileTextureLayer._buffer || null));\n }\n }\n updateTextureProps(texture, parameters);\n }\n }\n }\n }\n });\n }\n else {\n if (isTrackedClass(node)) {\n node.setPropertyIsModified('textures', true);\n this.queuedTexturesToProcess.delete(node);\n return;\n }\n }\n }\n if (isMaterialNode(node) && isTrackedClass(node)) {\n this.applyUvProperties(node);\n }\n else {\n throw Error('Material is not being tracked');\n }\n this.queuedTexturesToProcess.delete(node);\n function updateTextureProps(texture, parameters) {\n texture.level = parameters.strength;\n texture.hasAlpha = parameters.hasTransparency;\n }\n }\n delete3d(feature) {\n if (feature._parent) {\n this.resetTextureStack(feature._parent, feature.channel);\n }\n const renderer = this.renderers.get(Token.withKey(feature.$type));\n if (!renderer) {\n throw Error('no renderer for feature');\n }\n renderer.delete3d(feature);\n }\n get3d(texture) {\n if (!texture._parent) {\n return undefined;\n }\n if (texture._parent instanceof exports.MaterialNode) {\n return this.babylonMap.get(this.getTexture3dId(texture._parent, texture.channel));\n }\n }\n set3d(texture, babylonObject) {\n if (texture._parent) {\n if (texture._parent instanceof exports.MaterialNode) {\n this.babylonMap.set(this.getTexture3dId(texture._parent, texture.channel), babylonObject);\n }\n }\n }\n dispose() {\n this.queuedTexturesToProcess.clear();\n this.renderers.forEach(renderer => renderer.dispose());\n this.updateTextureStackRequest.complete();\n super.dispose();\n }\n setTransparency(materialNode, hasTransparency) {\n // TODO: limit this to only albedo textures for now. Will need testing.\n this.resetTextureStack(materialNode, exports.eTextureChannel.albedo, { hasTransparency });\n }\n applyUvProperties(materialNode) {\n const material = this.rendererManager.material.get3d(materialNode);\n if (material) {\n for (const channel in textureChannelMap) {\n const firstTextureLayer = materialNode.textures.find(t => t.channel === channel);\n const propName3d = textureChannelMap[channel];\n let texture;\n if (Array.isArray(propName3d)) {\n const res = getObjectPropertyAtPath(material, propName3d);\n texture = res.obj[res.prop];\n }\n else {\n texture = material[propName3d];\n }\n if (texture && firstTextureLayer) {\n if (!(texture instanceof Texture)) {\n throw Error(propName3d.toString() + ' is not a texture');\n }\n texture.uOffset = firstTextureLayer.uOffset;\n texture.vOffset = firstTextureLayer.vOffset;\n texture.uScale = firstTextureLayer.uScale;\n texture.vScale = firstTextureLayer.vScale;\n texture.uAng = (firstTextureLayer.uAngle * Math.PI) / 180;\n texture.vAng = (firstTextureLayer.vAngle * Math.PI) / 180;\n texture.wAng = (firstTextureLayer.wAngle * Math.PI) / 180;\n texture.uRotationCenter = firstTextureLayer.uPivot;\n texture.vRotationCenter = firstTextureLayer.vPivot;\n texture.wRotationCenter = firstTextureLayer.wPivot;\n // if (firstTextureLayer.uWrap === eTextureWrapAddressMode.clamp) {\n // texture.wrapU = Texture.CLAMP_ADDRESSMODE;\n // } else {\n // texture.wrapU = Texture.WRAP_ADDRESSMODE;\n // }\n // if (firstTextureLayer.vWrap === eTextureWrapAddressMode.clamp) {\n // texture.wrapV = Texture.CLAMP_ADDRESSMODE;\n // } else {\n // texture.wrapV = Texture.WRAP_ADDRESSMODE;\n // }\n // For some reason wrap addressmode is not getting converted correctly. Repeat is the safer option so\n // Let's hardcode repeat for now\n texture.wrapU = Texture.WRAP_ADDRESSMODE;\n texture.wrapV = Texture.WRAP_ADDRESSMODE;\n if (!(texture instanceof DynamicTexture)) {\n texture.vScale *= -1;\n }\n }\n }\n }\n }\n paintTextureOnCanvas(textureLayers, context, parameters) {\n let clippingTexture = null;\n for (let i = 0; i < textureLayers.length; i++) {\n if (!textureLayers[i]._init) {\n continue;\n }\n const textureLayer = textureLayers[i];\n const renderer = this.getRenderer(textureLayer);\n // Continue clipping with the same texture as long as useClippingMask is true.\n // When there's a texture that's not clipping, that can potentially be a new clipping mask.\n if (!textureLayer.useClippingMask) {\n clippingTexture = null;\n }\n const maskingCtx = maskingCanvas.getContext('2d', {\n willReadFrequently: true,\n });\n if (textureLayers[i + 1] && textureLayers[i + 1].useClippingMask && !clippingTexture) {\n maskingCanvas.width = context.canvas.width;\n maskingCanvas.height = context.canvas.height;\n clippingTexture = textureLayer;\n }\n else if (textureLayer.useClippingMask && clippingTexture) {\n const clippingTextureRenderer = this.getRenderer(clippingTexture);\n maskingCtx.clearRect(0, 0, maskingCanvas.width, maskingCanvas.height);\n clippingTextureRenderer.paintTexture(clippingTexture, maskingCtx, { strength: 1, hasTransparency: true }, {});\n renderer.paintTexture(textureLayer, context, parameters, {\n mask: { context: maskingCtx },\n });\n }\n else {\n renderer.paintTexture(textureLayer, context, parameters, {});\n }\n }\n return context;\n }\n getActiveLayers(materialNode, channel, includeSupressed = false) {\n return materialNode.textures.filter(texture => texture.channel === channel && texture.enabled && (includeSupressed || !texture._suppressed));\n }\n getRenderer(textureLayer) {\n const renderer = this.renderers.get(Token.withKey(textureLayer.$type));\n if (!renderer) {\n throw Error('no renderer for texture layer');\n }\n return renderer;\n }\n canUseSimpleTexture(activeLayers) {\n if (activeLayers.length === 1) {\n const layer = activeLayers[0];\n return layer instanceof exports.FileTextureLayer;\n }\n return false;\n }\n disposeTexture(texture) {\n // Resizing textures before they are ready results in an error. Remove this code for now\n // if (texture instanceof DynamicTexture) {\n // texture.scaleTo(1, 1); // Prevent memory leaks in Safari\n // }\n texture.dispose();\n }\n getTexture3dId(node, channel) {\n return `${node._dynamicId}_${channel}`;\n }\n getTexture(node, channel) {\n const rootScene = this.viewer.sceneNode;\n const activeLayers = this.getActiveLayers(node, channel, true);\n const canUseSimpleTexture = this.canUseSimpleTexture(activeLayers);\n let texture = this.babylonMap.get(this.getTexture3dId(node, channel));\n const originalTexture = texture;\n if (originalTexture) {\n if (originalTexture instanceof DynamicTexture && canUseSimpleTexture) {\n this.disposeTexture(originalTexture);\n }\n else if (activeLayers.length > 1 && !(texture instanceof DynamicTexture)) {\n this.disposeTexture(originalTexture);\n }\n else {\n return originalTexture;\n }\n }\n if (canUseSimpleTexture) {\n const fileTexture = activeLayers[0];\n if (fileTexture.path && fileTexture._buffer && fileTexture._parent instanceof exports.MaterialNode) {\n texture = new Texture(fileTexture._bufferSource || '', this.scene, this.noMipMapFallback, false, SamplingModeMap[rootScene.textureSamplingMode], null, null, fileTexture._buffer);\n texture.name = this.getTexture3dId(node, channel);\n }\n else {\n return undefined;\n }\n }\n else {\n texture = new DynamicTexture(this.getTexture3dId(node, channel), 256, this.scene, !this.noMipMapFallback, SamplingModeMap[rootScene.textureSamplingMode]);\n }\n texture.homogeneousRotationInUVTransform = true;\n this.babylonMap.set(this.getTexture3dId(node, channel), texture);\n // set initial values\n if (originalTexture) {\n texture.hasAlpha = originalTexture.hasAlpha;\n texture.level = originalTexture.level;\n texture.getAlphaFromRGB = originalTexture.getAlphaFromRGB;\n }\n else {\n texture.hasAlpha = false;\n if (channel === exports.eTextureChannel.opacity) {\n texture.getAlphaFromRGB = true;\n }\n else {\n texture.getAlphaFromRGB = false;\n }\n texture.level = 1;\n }\n texture.coordinatesIndex = 0;\n texture.coordinatesMode = 0;\n texture.anisotropicFilteringLevel = 4;\n if (texture instanceof Texture) {\n texture.uOffset = 0; // Define an offset on the texture to offset the u coordinates of the UVs\n texture.vOffset = 0; // Define an offset on the texture to offset the v coordinates of the UVs\n texture.uScale = -1; // Define an offset on the texture to scale the u coordinates of the UVs\n texture.vScale = 1; // Define an offset on the texture to scale the v coordinates of the UVs\n texture.uAng = 0; // Define an offset on the texture to rotate around the u coordinates of the UVs\n texture.vAng = 0; // Define an offset on the texture to rotate around the v coordinates of the UVs\n texture.wAng = 0; // Define an offset on the texture to rotate around the w coordinates of the UVs (in case of 3d texture)\n texture.uRotationCenter = 0.5; // Defines the center of rotation (U)\n texture.vRotationCenter = 0.5; // Defines the center of rotation (V)\n texture.wRotationCenter = 0.5; // Defin\n texture.isBlocking = false;\n }\n return texture;\n }\n }\n function getObjectPropertyAtPath(_obj, path, throwOnNotFound = true) {\n let obj = _obj;\n const propPath = path.slice().reverse();\n while (propPath.length > 1 && obj) {\n const prop = propPath.pop();\n if (prop in obj) {\n obj = obj[prop];\n }\n else {\n obj = null;\n }\n }\n if (throwOnNotFound && !obj) {\n throw Error('Texture property not found for ' + path.toString());\n }\n return { obj, prop: propPath[0] };\n }\n\n class ViewpointRenderer extends Renderer {\n constructor() {\n super(...arguments);\n this.modelToken = Token.ViewpointNode;\n }\n update3dInternal(node) { }\n }\n\n let unprocessedChangesCount = 0;\n class ChangeMap extends ArrayMap {\n }\n class RendererManager {\n getRenderMode() {\n return this.material.renderMode;\n }\n /** Do not use this function directly. Call setRenderMode on the viewer instead so the correct sceneNode is used. */\n setRenderMode(mode, sceneNode, force) {\n if (force) {\n this.forcedRenderingMode = mode;\n }\n this.material.renderMode = this.forcedRenderingMode || mode;\n this.mesh.setRenderMode(this.forcedRenderingMode || mode, sceneNode);\n }\n constructor(viewer, scene) {\n this.viewer = viewer;\n this.scene = scene;\n this.deferredChanges = [];\n this.maxRenderTime = MAX_RENDER_TIME;\n this.processors = new Array();\n this.shadowManager = new ShadowManager(viewer.bbCache);\n this.sceneRenderer = new SceneRenderer(viewer, scene, viewer.gui, this);\n this.textures = new TextureStackRenderer(viewer, scene, this);\n this.material = new MaterialRenderer(viewer, scene, this.textures);\n this.light = new LightRenderer(viewer, scene, this.shadowManager);\n this.viewpoint = new ViewpointRenderer(viewer, scene);\n this.mesh = new MeshRenderer(viewer, scene, this, this.shadowManager);\n this.features = new FeatureStackRenderer(viewer, scene, this);\n this.connector = new ConnectorRenderer(viewer, scene, this);\n this.controlPointRenderer = new SketchControlPointRenderer(viewer, scene, this);\n this.annotation = new AnnotationRenderer(viewer, scene, this);\n this.dimension = new DimensionRenderer(viewer, scene, this);\n this.mate = new MateRenderer(viewer, scene);\n this.path = new PathRenderer(viewer, scene);\n // Order is important here, it defines the order in which renderers will be run\n this.renderers = new Set([\n this.sceneRenderer,\n this.material,\n this.textures,\n this.viewpoint,\n this.path,\n this.mesh,\n this.light,\n this.features,\n this.connector,\n this.controlPointRenderer,\n this.annotation,\n this.dimension,\n this.mate,\n ]);\n this.referenceProcessor = new ReferenceProcessor(viewer);\n this.mateProcessor = new MateProcessor(viewer, scene, this.connector, this.controlPointRenderer, this.referenceProcessor);\n this.sketchProcessor = new SketchProcessor(this.controlPointRenderer);\n this.processors.push(this.mateProcessor);\n this.processors.push(this.referenceProcessor);\n this.processors.push(this.sketchProcessor);\n }\n isStable() {\n return (!SyncService.hasChanges &&\n this.features.queuedFeaturesToProcess.size === 0 &&\n this.textures.queuedTexturesToProcess.size === 0);\n }\n render() {\n let changes;\n if (this.deferredChanges.length > 0) {\n changes = this.getChangeMap(this.deferredChanges);\n }\n else {\n for (const p of this.processors) {\n p.beforeRender();\n }\n // run deletions first\n const deletions = this.getChangeMap(SyncService.getDeletedNodes(), true);\n for (const p of this.processors) {\n p.beforeDeletions(deletions);\n }\n this.runChanges(deletions, (renderer, change) => renderer.delete3d(change));\n for (const p of this.processors) {\n p.afterDeletions(deletions);\n }\n //then run updates\n const newChanges = SyncService.getChangedNodes();\n changes = this.getChangeMap(newChanges);\n for (const p of this.processors) {\n p.beforeUpdates(changes);\n }\n changes = this.getChangeMap(SyncService.getChangedNodes()); //need to update changes in case processors added any\n }\n const allChangesByDid = new Map(changes.all().map(change => {\n return [change._dynamicId, change];\n }));\n const allChangeCount = allChangesByDid.size;\n const processedChangeMap = this.runChanges(changes, (renderer, change) => {\n renderer.update3d(change);\n });\n const processedChanges = processedChangeMap.all();\n processedChanges.forEach(change => {\n allChangesByDid.delete(change._dynamicId);\n });\n this.deferredChanges = Array.from(allChangesByDid.values());\n const skipCount = allChangeCount - processedChanges.length;\n // For now we are processing all changes becuase it causes unpredictable behavior.\n // The heaviest part of the render cycle is the geometry features processing.\n // if (processedChanges.length > 0 || skipCount > 0) {\n // console.log('processed ' + processedChanges.length + ' changes, skipped ' + skipCount + ' changes');\n // }\n // Don't run the geometry and texture stack processing until all changes have been processed\n if (skipCount === 0) {\n this.features.update3dGeometry();\n this.textures.updateTextureStacks();\n }\n for (const p of this.processors)\n p.afterUpdates(processedChangeMap);\n for (const n of processedChanges) {\n n.commitChanges();\n }\n for (const p of this.processors)\n p.doAfterCleanup();\n // Output a console warning if there are a lot of changes not being processed, as this could indicate a problem with changes not being processed.\n const unchangedNodes = SyncService.getChangedNodes();\n if (unchangedNodes.length > 0) {\n unprocessedChangesCount++;\n if (unprocessedChangesCount > 1000) {\n unprocessedChangesCount = 0;\n console.warn('Scene stabilization is taking longer than expected.');\n for (const n of unchangedNodes) {\n if (n._rootScene._dynamicId !== this.viewer.sceneNode._dynamicId) {\n n.commitChanges();\n console.warn(`node ${n.name} (${n.id}) is from a different, non-active scene instance. Clearing the change.`);\n }\n }\n }\n }\n }\n /** runs a change map in order */\n runChanges(changes, action) {\n const processedChanges = new ChangeMap();\n const renderStart = performance.now();\n for (const renderer of this.renderers) {\n const changeList = changes.get(renderer.modelToken);\n for (const change of changeList) {\n action(renderer, change);\n processedChanges.add(renderer.modelToken, change);\n }\n }\n const renderTime = performance.now() - renderStart;\n if (renderTime > 50) {\n console.warn(`Change cycle took ${renderTime}ms`);\n }\n return processedChanges;\n }\n getChangeMap(nodes, includeOrphaned = false) {\n var _a;\n const changes = new ChangeMap();\n for (const n of nodes) {\n if ((((_a = n._rootScene) === null || _a === void 0 ? void 0 : _a._dynamicId) === this.viewer.sceneNode._dynamicId && n._level > -1) ||\n includeOrphaned ||\n this.isGlobalMaterialOrTexture(n) //global/library material nodes held outside of graph\n ) {\n const renderer = this.renderers.find(r => r.handlesNode(n));\n if (!renderer)\n throw 'unknown node type';\n changes.add(renderer.modelToken, n);\n }\n else if (n._rootScene == null) {\n // Clear changes on nodes that are not in any scene tree\n n.commitChanges();\n }\n }\n changes.get(Token.SpaceNode).sortBy(m => m._level); //sort meshes by level first\n return changes;\n }\n disposeAll() {\n this.shadowManager.dispose();\n for (const processor of this.processors)\n processor.dispose();\n for (const renderer of this.renderers)\n renderer.dispose();\n }\n isGlobalMaterialOrTexture(n) {\n let mat = n;\n if (n instanceof exports.TextureLayer)\n mat = n._parent;\n return mat instanceof exports.MaterialNode && mat.location != exports.eMaterialLocation.local;\n }\n }\n\n /**\n * We are overriding the default mouse wheet input for the arc rotate camera here because\n * the default one does not act correctly when zoomed in close to the target: it\n * calculates the inertia based on the radius at the time the event happens. The result\n * is that zooming in is fast than zooming out. They had a misguided attempt at fixing\n * that here: https://github.com/BabylonJS/Babylon.js/issues/6215 but it is inadequate.\n * See comments below for where we changed the code\n */\n class KbArcRotateCameraMouseWheelInput {\n constructor() {\n this.wheelPrecision = 3.0;\n this.wheelDeltaPercentage = 0;\n }\n computeDeltaFromMouseWheelLegacyEvent(mouseWheelDelta, radius) {\n let delta = 0;\n const wheelDelta = mouseWheelDelta * 0.01 * this.wheelDeltaPercentage * radius;\n if (mouseWheelDelta > 0) {\n delta = wheelDelta / (1.0 + this.wheelDeltaPercentage);\n }\n else {\n delta = wheelDelta * (1.0 + this.wheelDeltaPercentage);\n }\n return delta;\n }\n attachControl(noPreventDefault) {\n this._wheel = (p, s) => {\n if (p.type !== PointerEventTypes.POINTERWHEEL) {\n return;\n }\n const event = p.event;\n let delta = 0;\n const mouseWheelLegacyEvent = event;\n let wheelDelta = 0;\n if (mouseWheelLegacyEvent.wheelDelta) {\n wheelDelta = mouseWheelLegacyEvent.wheelDelta;\n }\n else {\n wheelDelta = -(event.deltaY || event.detail || 0) * 60;\n }\n if (this.wheelDeltaPercentage) {\n delta = this.computeDeltaFromMouseWheelLegacyEvent(wheelDelta, this.camera.radius);\n // If zooming in, estimate the target radius and use that to compute the delta for inertia\n // this will stop multiple scroll events zooming in from adding too much inertia\n //if (delta > 0) { //KB Note: their misguided attempt at fixing the problem only affected zoom in, when it should affect both, so we commented this line out\n let estimatedTargetRadius = this.camera.radius;\n let targetInertia = this.camera.inertialRadiusOffset + delta;\n for (let i = 0; i < 20 && Math.abs(targetInertia) > this.camera.speed * Epsilon; i++) {\n //KB Note: fixed the lower limit where it opts out to match the camera's own intertial calculation\n estimatedTargetRadius -= targetInertia;\n targetInertia *= this.camera.inertia;\n }\n estimatedTargetRadius = Scalar.Clamp(estimatedTargetRadius, 0, Number.MAX_VALUE);\n delta = this.computeDeltaFromMouseWheelLegacyEvent(wheelDelta, estimatedTargetRadius);\n //}\n }\n else {\n delta = wheelDelta / (this.wheelPrecision * 40);\n }\n if (delta) {\n this.camera.inertialRadiusOffset += delta;\n }\n if (event.preventDefault) {\n if (!noPreventDefault) {\n event.preventDefault();\n }\n }\n };\n this._observer = this.camera.getScene().onPointerObservable.add(this._wheel, PointerEventTypes.POINTERWHEEL);\n }\n detachControl() {\n if (this._observer) {\n this.camera.getScene().onPointerObservable.remove(this._observer);\n this._observer = null;\n this._wheel = null;\n }\n }\n getClassName() {\n return 'KbArcRotateCameraMouseWheelInput';\n }\n getSimpleName() {\n return 'mousewheel';\n }\n }\n\n /** Arc Rotate Camera (orbit camera) modified to offset the screen position on pan and keep the orbit target in the same position */\n class KbArcRotateCamera extends ArcRotateCamera {\n constructor(name, alpha, beta, radius, target, scene, setActiveOnSceneIfNoneActive) {\n super(name, alpha, beta, radius, target, scene, setActiveOnSceneIfNoneActive);\n this.scene = scene;\n this.inertialOffsetPanningX = 0;\n this.inertialOffsetPanningY = 0;\n this.inertialOrthoZoom = 0;\n this.clientInfo = {\n mouseX: 0,\n mouseY: 0,\n screenLeft: 0,\n screenTop: 0,\n screenWidth: 0,\n screenHeight: 0,\n };\n this.lockRotation = false;\n this.lockPanning = false;\n this.lockZoom = false;\n this.limitMinimumTargetDistance = false;\n this._orthoWidth = 0;\n //replace the wheel input with our own\n this.inputs.removeByType('ArcRotateCameraMouseWheelInput');\n this.inputs.add(new KbArcRotateCameraMouseWheelInput());\n this.updateClient();\n }\n get orthoWidth() {\n return this._orthoWidth;\n }\n set orthoWidth(val) {\n this._orthoWidth = this.clampOrthoWidth(val);\n this.updateOrthographicPosition();\n }\n orthoFit(sizeX, sizeY) {\n if (this.mode === Camera.ORTHOGRAPHIC_CAMERA) {\n const ratio = this.clientInfo.screenHeight / this.clientInfo.screenWidth;\n if (sizeX * ratio < sizeY) {\n this.orthoWidth = sizeY / ratio;\n }\n else {\n this.orthoWidth = sizeX;\n }\n }\n }\n update() {\n if (this.scene.activeCamera !== this) {\n return;\n }\n let startZoomLevel;\n if (!this.targetScreenOffset) {\n this.targetScreenOffset = new Vector2(0, 0);\n }\n if (this.lockRotation && (this.inertialAlphaOffset !== 0 || this.inertialBetaOffset !== 0)) {\n this.inertialPanningX = this.inertialAlphaOffset;\n this.inertialPanningY = -this.inertialBetaOffset;\n this.inertialAlphaOffset = this.inertialBetaOffset = 0;\n }\n if (this.inertialPanningX !== 0 || this.inertialPanningY !== 0) {\n if (!this.lockPanning) {\n this.inertialOffsetPanningX = this.inertialPanningX;\n this.inertialOffsetPanningY = this.inertialPanningY;\n }\n this.inertialPanningX = 0;\n this.inertialPanningY = 0;\n }\n if (this.lockZoom && this.inertialRadiusOffset !== 0) {\n this.inertialRadiusOffset = 0;\n }\n if (this.inertialRadiusOffset !== 0) {\n if (this.mode === Camera.ORTHOGRAPHIC_CAMERA) {\n this.inertialOrthoZoom = this.inertialRadiusOffset;\n this.inertialRadiusOffset = 0;\n }\n else {\n startZoomLevel = this.radius;\n //this.inertialRadiusOffset = 0; //override so we have full control over the zoom\n //if (this.inertialRadiusOffset < .001) this.inertialRadiusOffset = .001;\n }\n }\n if (this.inertialOrthoZoom !== 0) {\n startZoomLevel = this.orthoWidth;\n }\n super.update();\n if (this.inertialOffsetPanningX !== 0 || this.inertialOffsetPanningY !== 0) {\n let offset;\n if (this.mode === Camera.ORTHOGRAPHIC_CAMERA) {\n offset = new Vector2(this.inertialOffsetPanningX * this.orthoWidth, this.inertialOffsetPanningY * this.orthoWidth);\n }\n else {\n offset = new Vector2(this.inertialOffsetPanningX * this.radius, this.inertialOffsetPanningY * this.radius);\n }\n this.targetScreenOffset.subtractInPlace(offset);\n this.inertialOffsetPanningX *= this.panningInertia;\n this.inertialOffsetPanningY *= this.panningInertia;\n }\n if (this.inertialOrthoZoom !== 0) {\n this.orthoWidth = this.orthoWidth - this.inertialOrthoZoom * this.orthoWidth;\n this.inertialOrthoZoom *= this.panningInertia;\n }\n if (Math.abs(this.inertialOffsetPanningX) < this.speed * Epsilon) {\n this.inertialOffsetPanningX = 0;\n }\n if (Math.abs(this.inertialOffsetPanningY) < this.speed * Epsilon) {\n this.inertialOffsetPanningY = 0;\n }\n if (Math.abs(this.inertialOrthoZoom) < this.speed * Epsilon) {\n this.inertialOrthoZoom = 0;\n }\n // When zooming, adjust the camera pan so you zoom into your mouse position, as long as pan is enabled\n if (!this.lockPanning && startZoomLevel !== undefined) {\n if (this.mode === Camera.ORTHOGRAPHIC_CAMERA) {\n const zoomChange = (startZoomLevel - this.orthoWidth) / 2;\n this.targetScreenOffset.x -= zoomChange * this.clientInfo.mouseX;\n this.targetScreenOffset.y += zoomChange * this.clientInfo.mouseY;\n }\n else {\n const gamma = Math.tan(this.fov / 2);\n const zoomChange = startZoomLevel - this.radius;\n const shiftX = zoomChange * gamma * this.clientInfo.mouseX;\n const shiftY = zoomChange * gamma * this.clientInfo.mouseY;\n this.targetScreenOffset.x -= shiftX;\n this.targetScreenOffset.y += shiftY;\n }\n }\n }\n updateClient() {\n const canvas = this.getEngine().getRenderingCanvas();\n if (!canvas) {\n return;\n }\n const clientRect = canvas.getBoundingClientRect();\n this.clientInfo.screenWidth = clientRect.width;\n this.clientInfo.screenHeight = clientRect.height;\n this.clientInfo.screenLeft = clientRect.left;\n this.clientInfo.screenTop = clientRect.top;\n if (this.mode === Camera.ORTHOGRAPHIC_CAMERA) {\n this.updateOrthographicPosition();\n this.panningSensibility = clientRect.width;\n }\n else {\n const focalLength = clientRect.height / (2 * Math.tan(this.fov / 2));\n this.panningSensibility = focalLength;\n }\n return canvas;\n }\n attachControl(noPreventDefault = false) {\n super.attachControl(noPreventDefault, true);\n if (this.disposeListeners) {\n this.disposeListeners();\n }\n const updateMousePosition = (event) => {\n this.clientInfo.mouseX =\n ((event.clientX - this.clientInfo.screenLeft) * 2) / this.clientInfo.screenWidth - 1;\n this.clientInfo.mouseY =\n ((event.clientY - this.clientInfo.screenTop) * 2) / this.clientInfo.screenHeight - 1;\n };\n const canvas = this.updateClient();\n const updateClientWrap = () => this.updateClient();\n if (canvas) {\n window.addEventListener('resize', updateClientWrap);\n canvas.addEventListener('mousemove', updateMousePosition);\n this.disposeListeners = () => {\n window.removeEventListener('resize', updateClientWrap);\n canvas.removeEventListener('mousemove', updateMousePosition);\n this.disposeListeners = undefined;\n };\n }\n }\n _checkLimits() {\n super._checkLimits();\n if (this.mode === Camera.ORTHOGRAPHIC_CAMERA) {\n this.orthoWidth = this.clampOrthoWidth(this.orthoWidth);\n }\n }\n clampOrthoWidth(val) {\n if (this.mode === Camera.ORTHOGRAPHIC_CAMERA) {\n if (this.lowerRadiusLimit != null && val < this.lowerRadiusLimit) {\n val = this.lowerRadiusLimit;\n }\n if (this.upperRadiusLimit != null && val > this.upperRadiusLimit) {\n val = this.upperRadiusLimit;\n }\n }\n return val;\n }\n dispose() {\n super.dispose();\n if (this.disposeListeners) {\n this.disposeListeners();\n }\n }\n updateOrthographicPosition() {\n if (this.clientInfo) {\n const ratio = this.clientInfo.screenHeight / this.clientInfo.screenWidth;\n this.orthoRight = this.orthoWidth / 2;\n this.orthoLeft = -this.orthoWidth / 2;\n this.orthoTop = (this.orthoWidth * ratio) / 2;\n this.orthoBottom = (-this.orthoWidth * ratio) / 2;\n }\n }\n }\n\n const ANIMATION_FPS = 30;\n var eCameraType;\n (function (eCameraType) {\n eCameraType[eCameraType[\"Universal\"] = 0] = \"Universal\";\n eCameraType[eCameraType[\"ArcRotate\"] = 1] = \"ArcRotate\";\n })(eCameraType || (eCameraType = {}));\n (function (eCameraOrientation) {\n eCameraOrientation[\"perspective\"] = \"perspective\";\n eCameraOrientation[\"front\"] = \"front\";\n eCameraOrientation[\"left\"] = \"left\";\n eCameraOrientation[\"back\"] = \"back\";\n eCameraOrientation[\"right\"] = \"right\";\n eCameraOrientation[\"top\"] = \"top\";\n eCameraOrientation[\"bottom\"] = \"bottom\";\n })(exports.eCameraOrientation || (exports.eCameraOrientation = {}));\n class CameraTools {\n get cameraMoving() {\n const camera = this.scene.activeCamera;\n if (this._activeAnimation) {\n return true;\n }\n if (camera instanceof KbArcRotateCamera) {\n return (Math.abs(camera.inertialRadiusOffset) > 0 ||\n Math.abs(camera.inertialAlphaOffset) > 0 ||\n Math.abs(camera.inertialBetaOffset) > 0 ||\n Math.abs(camera.inertialOffsetPanningX) > 0 ||\n Math.abs(camera.inertialOffsetPanningY) > 0 ||\n Math.abs(camera.inertialOrthoZoom) > 0);\n }\n if (camera instanceof UniversalCamera) {\n return (Math.abs(camera.cameraDirection.x) > 0 ||\n Math.abs(camera.cameraDirection.y) > 0 ||\n Math.abs(camera.cameraDirection.z) > 0 ||\n Math.abs(camera.cameraRotation.x) > 0 ||\n Math.abs(camera.cameraRotation.y) > 0);\n }\n return false;\n }\n constructor(scene, canvas, kbViewer) {\n this.scene = scene;\n this.canvas = canvas;\n this.kbViewer = kbViewer;\n this._lastOrientation = exports.eCameraOrientation.perspective;\n this.minVector = new Vector3(0, 0, 0);\n this.maxVector = new Vector3(0, 0, 0);\n this.objectsToScaleWithDistance = new Map();\n this._beforeRender = () => {\n if (this._cameraSettingsChange && scene.activeCamera && this.cameraMoving) {\n const camera = scene.activeCamera;\n const changeList = {};\n copyCameraStateIntoViewpoint(changeList, camera);\n this._cameraSettingsChange.next(changeList);\n }\n };\n scene.registerBeforeRender(this._beforeRender);\n kbViewer.viewChanged.subscribe(() => {\n const camera = scene.activeCamera;\n if (camera) {\n if (camera.mode === Camera.ORTHOGRAPHIC_CAMERA) {\n const dist = ((camera.orthoRight || 0) - (camera.orthoLeft || 0)) * 0.6;\n this.objectsToScaleWithDistance.forEach((param, item) => {\n if (item.isDisposed()) {\n this.objectsToScaleWithDistance.delete(item);\n return;\n }\n if (typeof param === 'function') {\n param(dist, camera.mode);\n }\n else {\n const scaledSize = param * dist;\n item.scaling.set(scaledSize, scaledSize, scaledSize);\n }\n });\n }\n else {\n let cameraPosition;\n if (camera instanceof WebXRCamera && camera.globalPosition) {\n cameraPosition = camera.globalPosition;\n }\n else if (camera instanceof ArcRotateCamera) {\n cameraPosition = getCameraPositionWithOffset(camera);\n }\n else {\n cameraPosition = camera.globalPosition;\n }\n this.objectsToScaleWithDistance.forEach((param, item) => {\n if (item.isDisposed()) {\n this.objectsToScaleWithDistance.delete(item);\n return;\n }\n const dist = item.absolutePosition.subtract(cameraPosition).length();\n if (typeof param === 'function') {\n param(dist, camera.mode);\n }\n else {\n const scaledSize = dist * param;\n item.scaling.set(scaledSize, scaledSize, scaledSize);\n }\n });\n }\n }\n });\n this.activeCamera = {\n get position() {\n return scene.activeCamera ? exports.KbVector.FromVec3(scene.activeCamera.position) : exports.KbVector.Zero();\n },\n set position(vector) {\n if (scene.activeCamera) {\n if (scene.activeCamera.mode === Camera.ORTHOGRAPHIC_CAMERA &&\n scene.activeCamera instanceof ArcRotateCamera) {\n const target = scene.activeCamera.target;\n scene.activeCamera.position = vector\n .toVec3()\n .subtract(target)\n .normalize()\n .scale(5000)\n .add(target);\n }\n else {\n scene.activeCamera.position = vector.toVec3();\n }\n }\n },\n get targetPoint() {\n return scene.activeCamera instanceof ArcRotateCamera\n ? exports.KbVector.FromVec3(scene.activeCamera.target)\n : exports.KbVector.Zero();\n },\n set targetPoint(vector) {\n scene.activeCamera instanceof ArcRotateCamera && (scene.activeCamera.target = vector.toVec3());\n },\n get rotation() {\n return scene.activeCamera instanceof UniversalCamera\n ? exports.KbVector.FromVec3(scene.activeCamera.rotation)\n : exports.KbVector.Zero();\n },\n set rotation(vector) {\n scene.activeCamera instanceof UniversalCamera && (scene.activeCamera.rotation = vector.toVec3());\n },\n get offsetX() {\n return scene.activeCamera instanceof ArcRotateCamera ? scene.activeCamera.targetScreenOffset.x : 0;\n },\n set offsetX(offset) {\n scene.activeCamera instanceof ArcRotateCamera && (scene.activeCamera.targetScreenOffset.x = offset);\n },\n get offsetY() {\n return scene.activeCamera instanceof ArcRotateCamera ? scene.activeCamera.targetScreenOffset.y : 0;\n },\n set offsetY(offset) {\n scene.activeCamera instanceof ArcRotateCamera && (scene.activeCamera.targetScreenOffset.y = offset);\n },\n get projection() {\n if (!scene.activeCamera) {\n return undefined;\n }\n return scene.activeCamera.mode === Camera.PERSPECTIVE_CAMERA\n ? exports.eViewpointProjection.perspective\n : exports.eViewpointProjection.orthographic;\n },\n set projection(projection) {\n scene.activeCamera &&\n (scene.activeCamera.mode =\n projection === exports.eViewpointProjection.perspective\n ? Camera.PERSPECTIVE_CAMERA\n : Camera.ORTHOGRAPHIC_CAMERA);\n },\n get fov() {\n if (!scene.activeCamera) {\n return 50;\n }\n return scene.activeCamera.mode === Camera.ORTHOGRAPHIC_CAMERA\n ? scene.activeCamera.orthoRight - scene.activeCamera.orthoLeft\n : (scene.activeCamera.fov * 180) / Math.PI;\n },\n set fov(fov) {\n if (!scene.activeCamera) {\n return;\n }\n if (scene.activeCamera.mode === Camera.ORTHOGRAPHIC_CAMERA &&\n scene.activeCamera instanceof KbArcRotateCamera) {\n scene.activeCamera.orthoWidth = fov;\n }\n else {\n scene.activeCamera.fov = (fov * Math.PI) / 180;\n }\n },\n get allowOrbit() {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n return !scene.activeCamera.lockRotation;\n }\n else if (scene.activeCamera instanceof UniversalCamera) {\n return !!scene.activeCamera.inputs.attached.mouse;\n }\n return false;\n },\n set allowOrbit(allow) {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n scene.activeCamera.lockRotation = !allow;\n }\n else if (scene.activeCamera instanceof UniversalCamera) {\n scene.activeCamera.inputs.clear();\n enableFreeLook(scene.activeCamera);\n }\n },\n get allowPan() {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n return !scene.activeCamera.lockPanning;\n }\n else if (scene.activeCamera instanceof UniversalCamera) {\n return !!scene.activeCamera.inputs.attached.keyboard;\n }\n return false;\n },\n set allowPan(allow) {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n scene.activeCamera.lockPanning = !allow;\n }\n else if (scene.activeCamera instanceof UniversalCamera) {\n scene.activeCamera.inputs.clear();\n enableFreeMove(scene.activeCamera);\n }\n },\n get allowZoom() {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n return !scene.activeCamera.lockZoom;\n }\n return false;\n },\n set allowZoom(allow) {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n scene.activeCamera.lockZoom = !allow;\n }\n },\n get limitMinimumTargetDistance() {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n return scene.activeCamera.limitMinimumTargetDistance;\n }\n return false;\n },\n set limitMinimumTargetDistance(allow) {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n scene.activeCamera.limitMinimumTargetDistance = !allow;\n }\n },\n get minTargetDistance() {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n return scene.activeCamera.lowerRadiusLimit || defaults.ViewpointNode.minTargetDistance;\n }\n return defaults.ViewpointNode.minTargetDistance;\n },\n set minTargetDistance(value) {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n scene.activeCamera.lowerRadiusLimit = value;\n }\n },\n get maxTargetDistance() {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n return scene.activeCamera.upperRadiusLimit || defaults.ViewpointNode.maxTargetDistance;\n }\n return defaults.ViewpointNode.maxTargetDistance;\n },\n set maxTargetDistance(value) {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n scene.activeCamera.upperRadiusLimit = value;\n }\n },\n get orbitAxis() {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n return exports.KbVector.FromVec3(scene.activeCamera.upVector);\n }\n return exports.KbVector.From(defaults.ViewpointNode.orbitAxis);\n },\n set orbitAxis(value) {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n scene.activeCamera.upVector = value.toVec3();\n }\n },\n get limitMinLatitude() {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n return ((scene.activeCamera.lowerBetaLimit || 0) * 180) / Math.PI;\n }\n return defaults.ViewpointNode.limitMinLatitude;\n },\n set limitMinLatitude(value) {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n scene.activeCamera.lowerBetaLimit = (value * Math.PI) / 180;\n }\n },\n get limitMaxLatitude() {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n return (((typeof scene.activeCamera.upperBetaLimit === 'number'\n ? scene.activeCamera.upperBetaLimit\n : Math.PI) *\n 180) /\n Math.PI);\n }\n return defaults.ViewpointNode.limitMaxLatitude;\n },\n set limitMaxLatitude(value) {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n scene.activeCamera.upperBetaLimit = (value * Math.PI) / 180;\n }\n },\n get limitMinLongitude() {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n return scene.activeCamera.lowerAlphaLimit\n ? (scene.activeCamera.lowerAlphaLimit * 180) / Math.PI\n : defaults.ViewpointNode.limitMinLongitude;\n }\n return defaults.ViewpointNode.limitMinLongitude;\n },\n set limitMinLongitude(value) {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n scene.activeCamera.lowerAlphaLimit = (value * Math.PI) / 180;\n }\n },\n get limitMaxLongitude() {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n return scene.activeCamera.upperAlphaLimit\n ? (scene.activeCamera.upperAlphaLimit * 180) / Math.PI\n : defaults.ViewpointNode.limitMinLongitude;\n }\n return defaults.ViewpointNode.limitMinLongitude;\n },\n set limitMaxLongitude(value) {\n if (scene.activeCamera instanceof KbArcRotateCamera) {\n scene.activeCamera.upperAlphaLimit = (value * Math.PI) / 180;\n }\n },\n };\n }\n dispose() {\n this.scene.unregisterBeforeRender(this._beforeRender);\n }\n setPipeline(pipeline) {\n this.activePipeline = pipeline;\n if (this.scene.activeCamera) {\n this.applyScenePostProcesses(this.scene.activeCamera);\n }\n }\n /** Frames the current active camera to fit the provided mesh node\n * @param node The node to fit on the screen\n * @param padding If provided, pads the node by a percentage of the node's size.\n */\n frameCamera(node, options, camera) {\n const _options = {\n rotate: false,\n pan: true,\n padding: 1,\n zoom: true,\n ...(options || {}),\n };\n const activeCamera = camera || this.scene.activeCamera;\n if (!activeCamera) {\n throw Error('No active camera');\n }\n const bounding = this.kbViewer.bbCache.getWorld(node);\n if (!bounding || bounding.boundingInfo.diagonalLength === 0) {\n return; // no op if the bounding box doesn't exist\n }\n else {\n const childrenBounding = this.kbViewer.sceneNode.getChildren();\n const maxVector = bounding.boundingInfo.boundingBox.maximum;\n const minVector = bounding.boundingInfo.boundingBox.minimum;\n childrenBounding.forEach(element => {\n var _a, _b;\n if ((element instanceof exports.AnnotationNode && element.enabled) ||\n (element instanceof exports.DimensionNode && element.enabled)) {\n let nodeInfo;\n if (element instanceof exports.AnnotationNode) {\n nodeInfo = this.kbViewer.rendererManager.annotation.annotationStore.get(element._dynamicId);\n }\n if (element instanceof exports.DimensionNode) {\n nodeInfo = this.kbViewer.rendererManager.dimension.dimensionStore.get(element._dynamicId);\n }\n const minBoundingBox = (_a = nodeInfo === null || nodeInfo === void 0 ? void 0 : nodeInfo.meshInfo) === null || _a === void 0 ? void 0 : _a.textContainer.getHierarchyBoundingVectors().min;\n const maxBoundingBox = (_b = nodeInfo === null || nodeInfo === void 0 ? void 0 : nodeInfo.meshInfo) === null || _b === void 0 ? void 0 : _b.textContainer.getHierarchyBoundingVectors().max;\n if (minBoundingBox && minBoundingBox.x < minVector.x) {\n minVector.x = minBoundingBox.x;\n }\n if (minBoundingBox && minBoundingBox.y < minVector.y) {\n minVector.y = minBoundingBox.y;\n }\n if (minBoundingBox && minBoundingBox.z < minVector.z) {\n minVector.z = minBoundingBox.z;\n }\n if (maxBoundingBox && maxBoundingBox.x > maxVector.x) {\n maxVector.x = maxBoundingBox.x;\n }\n if (maxBoundingBox && maxBoundingBox.y > maxVector.y) {\n maxVector.y = maxBoundingBox.y;\n }\n if (maxBoundingBox && maxBoundingBox.z > maxVector.z) {\n maxVector.z = maxBoundingBox.z;\n }\n }\n });\n this.minVector.set(minVector.x, minVector.y, minVector.z);\n this.maxVector.set(maxVector.x, maxVector.y, maxVector.z);\n bounding.boundingInfo.boundingBox.reConstruct(this.minVector, this.maxVector, bounding.boundingInfo.boundingBox.getWorldMatrix());\n const bbCenter = bounding.center.toVec3();\n const bbCorners = bounding.boundingInfo.boundingBox.vectorsWorld;\n if (activeCamera instanceof UniversalCamera) {\n activeCamera.setTarget(bbCenter);\n }\n else if (activeCamera instanceof KbArcRotateCamera) {\n if (_options.rotate) {\n activeCamera.setTarget(bbCenter);\n if (_options.pan) {\n activeCamera.targetScreenOffset.set(0, 0);\n }\n }\n else if (_options.pan) {\n const cameraRay = activeCamera.getForwardRay();\n const cameraProjectionTargetPlane = Plane.FromPositionAndNormal(bbCenter, cameraRay.direction);\n const cameraDistance = cameraRay.intersectsPlane(cameraProjectionTargetPlane);\n const cameraProjectionTargetPoint = cameraRay.origin.add(cameraRay.direction.scale(cameraDistance));\n const cameraPositionWorldOffset = cameraProjectionTargetPoint.subtract(bbCenter);\n const planeProjectionX = Vector3.Cross(cameraRay.direction, Vector3.Up()).normalize();\n const planeProjectionY = Vector3.Cross(cameraRay.direction, planeProjectionX).normalize();\n const bbCenterOffset = new Vector2(Vector3.Dot(cameraPositionWorldOffset, planeProjectionX), -Vector3.Dot(cameraPositionWorldOffset, planeProjectionY));\n activeCamera.targetScreenOffset.copyFrom(bbCenterOffset);\n }\n if (_options.zoom) {\n const clientRect = this.canvas.getBoundingClientRect();\n const ratio = clientRect.height / clientRect.width;\n const cameraRay = activeCamera.getForwardRay();\n const xAxis = Vector3.Cross(cameraRay.direction, Vector3.Up()).normalize();\n const yAxis = Vector3.Cross(cameraRay.direction, xAxis).normalize();\n const cameraTruePosition = cameraRay.origin\n .add(xAxis.scale(-activeCamera.targetScreenOffset.x))\n .add(yAxis.scale(activeCamera.targetScreenOffset.y));\n const cameraTrueTarget = activeCamera.target\n .add(xAxis.scale(-activeCamera.targetScreenOffset.x))\n .add(yAxis.scale(activeCamera.targetScreenOffset.y));\n const xPlane = Plane.FromPositionAndNormal(cameraTruePosition, yAxis);\n const yPlane = Plane.FromPositionAndNormal(cameraTruePosition, xAxis);\n const padding = 1 + _options.padding / 100;\n if (activeCamera.mode === Camera.ORTHOGRAPHIC_CAMERA) {\n let maxX = 0, maxY = 0;\n bbCorners.forEach(bbCorner => {\n const xO = xPlane.signedDistanceTo(bbCorner) * padding;\n maxX = Math.max(maxX, Math.abs(xO));\n const yO = yPlane.signedDistanceTo(bbCorner) * padding;\n maxY = Math.max(maxY, Math.abs(yO));\n });\n // The calculated distance is the maximum distance from the centerline to the bounding box corner. So double it to get the total width/height\n activeCamera.orthoFit(maxY * 2, maxX * 2);\n }\n else {\n let desiredDistance = 0;\n const fovH = 2 * Math.atan(Math.tan(activeCamera.fov / 2) / ratio);\n const thV = Math.tan(activeCamera.fov / 2);\n const thH = Math.tan(fovH / 2);\n bbCorners.forEach(bbCorner => {\n const xO = xPlane.signedDistanceTo(bbCorner) * padding;\n const xP = bbCorner.subtract(cameraTrueTarget).subtract(yAxis.scale(xO));\n const xA = Math.abs(xO) / thV;\n const xR = Vector3.Dot(xP, cameraRay.direction.scale(-1)) + xA;\n desiredDistance = Math.max(desiredDistance, xR);\n const yO = yPlane.signedDistanceTo(bbCorner) * padding;\n const yP = bbCorner.subtract(cameraTrueTarget).subtract(xAxis.scale(yO));\n const yA = Math.abs(yO) / thH;\n const yR = Vector3.Dot(yP, cameraRay.direction.scale(-1)) + yA;\n desiredDistance = Math.max(desiredDistance, yR);\n });\n activeCamera.radius = Math.max(desiredDistance, 0.001);\n }\n }\n }\n this.kbViewer.fireViewChangedOnNextRender(); //render loop won't pick up that the view changed, because no kb3d nodes changes, so we must update it explicitly\n }\n }\n getScreenPositionOfNode(node) {\n if (!this.scene.activeCamera) {\n throw Error('No active camera');\n }\n const bNode = this.kbViewer.getRendererNode(node);\n if (bNode) {\n return nodeToScreen(bNode, this.scene.activeCamera);\n }\n }\n getScreenBoundingBoxOfNode(node) {\n if (!this.scene.activeCamera) {\n throw Error('No active camera');\n }\n const bNode = this.kbViewer.getRendererNode(node);\n if (bNode) {\n return getScreenBoundingBox(bNode, this.scene.activeCamera, true);\n }\n }\n moveCameraToViewpoint(viewpoint) {\n const newCamera = this.createCameraFromViewpoint(viewpoint);\n this._activateCamera(newCamera);\n }\n animateToViewpoint(viewpoint, options) {\n if (!this.scene.activeCamera) {\n throw Error('No active camera');\n }\n let activeCamera = this.scene.activeCamera;\n const _options = {\n duration: 1000,\n ...(options || {}),\n };\n const endFrame = 30 * (_options.duration / 1000);\n const animations = [];\n const ease = new SineEase();\n ease.setEasingMode(SineEase.EASINGMODE_EASEINOUT);\n const circEaseOut = new CircleEase();\n circEaseOut.setEasingMode(CircleEase.EASINGMODE_EASEOUT);\n // Create the new viewpoint converted to an arc rotate camera so we can tween arc rotate properties\n // The arc rotate cameras are easier to animate between nicely.\n const newCameraTarget = this.createCameraFromViewpoint(viewpoint, eCameraType.ArcRotate);\n const newCamera = this.createCameraFromViewpoint(viewpoint);\n if (_options.frame && newCamera instanceof ArcRotateCamera) {\n const target = _options.frameTarget || viewpoint._parentScene;\n if (!target) {\n throw Error('Cannot frame camera, no target node to frame');\n }\n else {\n this.frameCamera(target, _options.frameOptions || {}, newCameraTarget);\n this.frameCamera(target, _options.frameOptions || {}, newCamera);\n }\n }\n if (activeCamera instanceof ArcRotateCamera) {\n if (Math.abs(activeCamera.alpha) > Math.PI * 2) {\n activeCamera.alpha = activeCamera.alpha % (Math.PI * 2);\n }\n }\n if (activeCamera instanceof UniversalCamera && viewpoint.type != exports.eViewpointType.firstPerson) {\n const orbitRotation = activeCamera.rotationQuaternion\n ? activeCamera.rotationQuaternion.toEulerAngles()\n : activeCamera.rotation;\n const ray = activeCamera.getForwardRay(newCameraTarget.radius);\n // If the current camera is a universal camera, convert it to an arc rotate camera as well/\n // The arc rotate target position is placed from the camera at the same distance as the destination viewpoint's target\n activeCamera = new KbArcRotateCamera('temporaryCamera', -(orbitRotation.y + Math.PI / 2), Math.PI / 2 - orbitRotation.x, ray.length, ray.direction.scale(ray.length).add(ray.origin), this.scene);\n activeCamera.fov = this.scene.activeCamera.fov;\n this._activateCamera(activeCamera);\n }\n if (!newCameraTarget) {\n throw Error('Could not create new camera');\n }\n if (activeCamera instanceof UniversalCamera && viewpoint.type === exports.eViewpointType.firstPerson) {\n const targetRotation = new Vector3(0, 0, 0);\n const targetPosition = new Vector3(viewpoint.position.x, viewpoint.position.y, viewpoint.position.z);\n targetRotation.y = (viewpoint.rotation.y * Math.PI) / 180;\n targetRotation.x = (viewpoint.rotation.x * Math.PI) / 180;\n targetRotation.z = (viewpoint.rotation.z * Math.PI) / 180;\n let activeAlpha = activeCamera.rotation.y;\n let targetAlpha = (viewpoint.rotation.y * Math.PI) / 180;\n let arc = 0;\n //Normalizing values to be between -PI and PI\n activeAlpha = normalizeRotation(activeAlpha);\n targetAlpha = normalizeRotation(targetAlpha);\n //calculating shortest arc and seeing if it crosses zero\n //if it does, change the target vector to account for expected direction of animation\n if (activeAlpha > targetAlpha) {\n arc = activeAlpha - targetAlpha;\n }\n else {\n arc = targetAlpha - activeAlpha;\n }\n if (arc > Math.PI) {\n if (activeAlpha < targetAlpha) {\n targetRotation.y = targetRotation.y - 2 * Math.PI;\n }\n else {\n targetRotation.y = targetRotation.y + 2 * Math.PI;\n }\n }\n const translateRotation = new Animation('translateCameraRotation', 'rotation', ANIMATION_FPS, Animation.ANIMATIONTYPE_VECTOR3, Animation.ANIMATIONLOOPMODE_CONSTANT);\n translateRotation.setKeys([\n { frame: 0, value: activeCamera.rotation },\n { frame: endFrame, value: targetRotation },\n ]);\n const translatePosition = new Animation('translateCameraPosition', 'position', ANIMATION_FPS, Animation.ANIMATIONTYPE_VECTOR3, Animation.ANIMATIONLOOPMODE_CONSTANT);\n translatePosition.setKeys([\n { frame: 0, value: activeCamera.position },\n { frame: endFrame, value: targetPosition },\n ]);\n const translateFov = new Animation('translateCameraFOV', 'fov', ANIMATION_FPS, Animation.ANIMATIONTYPE_FLOAT, Animation.ANIMATIONLOOPMODE_CONSTANT);\n translateFov.setKeys([\n { frame: 0, value: activeCamera.fov },\n { frame: endFrame, value: (viewpoint.fov / 180) * Math.PI },\n ]);\n translateRotation.setEasingFunction(ease);\n translatePosition.setEasingFunction(ease);\n translateFov.setEasingFunction(ease);\n animations.push(translateRotation);\n animations.push(translatePosition);\n animations.push(translateFov);\n }\n else if (newCameraTarget instanceof ArcRotateCamera && activeCamera instanceof ArcRotateCamera) {\n // When rotating between two arc rotate cameras, rotate around the target while animating the target\n if (!activeCamera.target.equals(newCameraTarget.target)) {\n const translateTarget = new Animation('translateCameraTarget', 'target', ANIMATION_FPS, Animation.ANIMATIONTYPE_VECTOR3, Animation.ANIMATIONLOOPMODE_CONSTANT);\n translateTarget.setKeys([\n { frame: 0, value: activeCamera.target },\n { frame: endFrame, value: newCameraTarget.target },\n ]);\n animations.push(translateTarget);\n }\n const translateA = new Animation('translateCameraA', 'alpha', ANIMATION_FPS, Animation.ANIMATIONTYPE_FLOAT, Animation.ANIMATIONLOOPMODE_CONSTANT);\n let crossesZero = false;\n let arc = 0;\n let shortestArc = 0;\n //Normalizing values to be between -PI and PI\n activeCamera.alpha = normalizeRotation(activeCamera.alpha);\n newCameraTarget.alpha = normalizeRotation(newCameraTarget.alpha);\n //calculating shortest arc between two cameras\n if (activeCamera.alpha > newCameraTarget.alpha) {\n arc = activeCamera.alpha - newCameraTarget.alpha;\n }\n else {\n arc = newCameraTarget.alpha - activeCamera.alpha;\n }\n if (arc > Math.PI) {\n shortestArc = 2 * Math.PI - arc;\n crossesZero = true;\n }\n //altering newCameraTarget to take shortest path\n if (newCameraTarget.alpha > 0 && crossesZero) {\n newCameraTarget.alpha = activeCamera.alpha - shortestArc;\n }\n else if (newCameraTarget.alpha < 0 && crossesZero) {\n newCameraTarget.alpha = activeCamera.alpha + shortestArc;\n }\n translateA.setKeys([\n { frame: 0, value: activeCamera.alpha },\n { frame: endFrame, value: newCameraTarget.alpha },\n ]);\n const translateB = new Animation('translateCameraB', 'beta', ANIMATION_FPS, Animation.ANIMATIONTYPE_FLOAT, Animation.ANIMATIONLOOPMODE_CONSTANT);\n translateB.setKeys([\n { frame: 0, value: activeCamera.beta },\n { frame: endFrame, value: newCameraTarget.beta },\n ]);\n const translateR = new Animation('translateCameraR', 'radius', ANIMATION_FPS, Animation.ANIMATIONTYPE_FLOAT, Animation.ANIMATIONLOOPMODE_CONSTANT);\n translateR.setKeys([\n { frame: 0, value: activeCamera.radius },\n { frame: endFrame, value: newCameraTarget.radius },\n ]);\n const translateOffset = new Animation('translateCameraOffset', 'targetScreenOffset', ANIMATION_FPS, Animation.ANIMATIONTYPE_VECTOR2, Animation.ANIMATIONLOOPMODE_CONSTANT);\n translateOffset.setKeys([\n { frame: 0, value: activeCamera.targetScreenOffset },\n { frame: endFrame, value: newCameraTarget.targetScreenOffset },\n ]);\n const translateFov = new Animation('translateCameraFOV', 'fov', ANIMATION_FPS, Animation.ANIMATIONTYPE_FLOAT, Animation.ANIMATIONLOOPMODE_CONSTANT);\n translateFov.setKeys([\n { frame: 0, value: activeCamera.fov },\n { frame: endFrame, value: newCameraTarget.fov },\n ]);\n if (activeCamera instanceof KbArcRotateCamera &&\n activeCamera.mode === Camera.ORTHOGRAPHIC_CAMERA &&\n newCameraTarget.mode === Camera.ORTHOGRAPHIC_CAMERA) {\n const translateOrthoWidth = new Animation('translateOrthoWidth', 'orthoWidth', ANIMATION_FPS, Animation.ANIMATIONTYPE_FLOAT, Animation.ANIMATIONLOOPMODE_CONSTANT);\n translateOrthoWidth.setKeys([\n { frame: 0, value: activeCamera.orthoWidth },\n { frame: endFrame, value: newCameraTarget.orthoWidth },\n ]);\n translateOrthoWidth.setEasingFunction(ease);\n animations.push(translateOrthoWidth);\n }\n translateA.setEasingFunction(ease);\n translateB.setEasingFunction(ease);\n translateR.setEasingFunction(ease);\n translateOffset.setEasingFunction(ease);\n translateFov.setEasingFunction(circEaseOut);\n animations.push(translateA);\n animations.push(translateB);\n animations.push(translateR);\n animations.push(translateOffset);\n animations.push(translateFov);\n }\n animations.forEach(animation => {\n activeCamera.animations.push(animation);\n });\n return new Promise(resolve => {\n this._activeAnimation = this.scene.beginAnimation(activeCamera, 0, endFrame, false, 1, () => {\n // on end of animation\n delete this._activeAnimation;\n this._activateCamera(newCamera);\n resolve();\n });\n });\n }\n storeCameraSettings() {\n if (!this.scene.activeCamera) {\n throw Error('No active camera');\n }\n this._storedViewpoint = new exports.ViewpointNode();\n const camera = this.scene.activeCamera;\n copyCameraStateIntoViewpoint(this._storedViewpoint, camera);\n }\n restoreCameraSettings() {\n if (this._storedViewpoint) {\n this.moveCameraToViewpoint(this._storedViewpoint);\n }\n else {\n console.warn('No stored camera settings to restore');\n }\n }\n watch() {\n this._cameraSettingsChange = new Subject();\n return this._cameraSettingsChange.pipe(finalize(() => {\n delete this._cameraSettingsChange;\n }), share());\n }\n maintainNodeScreenSize(renderedNode, relativeSize) {\n this.objectsToScaleWithDistance.set(renderedNode, relativeSize);\n }\n clearMaintainedNodeScreenSize(renderedNode) {\n this.objectsToScaleWithDistance.delete(renderedNode);\n }\n _activateCamera(newCamera) {\n this.scene.cameras.forEach(camera => {\n if (camera !== newCamera) {\n /**\n * Don't dispose the whole camera because that goes too hard and disposes all the render target textures and\n * post processes which messes things up for the next camera that's reusing those things. Instead, just dispose the\n * parts of the camera we don't need.\n */\n camera.onViewMatrixChangedObservable.clear();\n camera.onProjectionMatrixChangedObservable.clear();\n camera.onAfterCheckInputsObservable.clear();\n camera.onRestoreStateObservable.clear();\n // Inputs\n if (camera.inputs) {\n camera.inputs.clear();\n }\n // Animations\n this.scene.stopAnimation(camera);\n // Remove from scene\n this.scene.removeCamera(camera);\n this.scene.getEngine().releaseRenderPassId(camera.renderPassId);\n }\n });\n this.scene.customRenderTargets.forEach(rtt => {\n rtt.activeCamera = newCamera;\n });\n this.applyScenePostProcesses(newCamera);\n newCamera.attachControl(false);\n this.scene.activeCamera = newCamera;\n }\n applyScenePostProcesses(camera) {\n if (this.activePipeline) {\n this.scene.postProcessRenderPipelineManager.attachCamerasToRenderPipeline(this.activePipeline.name, camera);\n }\n }\n createCameraFromViewpoint(viewpoint, convertToCameraType) {\n const viewer = this.kbViewer;\n const bbCache = this.kbViewer.bbCache;\n const sceneNode = this.kbViewer.sceneNode;\n if (this.disposePreviousCamera && !this.disposePreviousCamera.isStopped) {\n this.disposePreviousCamera.next();\n this.disposePreviousCamera.complete();\n }\n const dispose = (this.disposePreviousCamera = new Subject());\n switch (viewpoint.type) {\n case exports.eViewpointType.orbit:\n case exports.eViewpointType.overhead:\n if (convertToCameraType === eCameraType.Universal) {\n const universalCamera = new UniversalCamera(viewpoint.id, viewpoint.position.toVec3(), this.scene);\n // this should be set in the setCommonViewpointProperties\n // const direction = viewpoint.targetPoint!.subtract(viewpoint.position).toVec3();\n // universalCamera.rotation = rotationToVector(direction).toEulerAngles();\n applyViewpointProperties(viewpoint, universalCamera);\n return universalCamera;\n }\n else {\n const arcRotateCamera = new KbArcRotateCamera(viewpoint.id, 0, 0, 0, viewpoint.targetPoint ? viewpoint.targetPoint.toVec3() : Vector3.Zero(), this.scene);\n applyViewpointProperties(viewpoint, arcRotateCamera);\n // overhead camera overrides target\n if (viewpoint.type === exports.eViewpointType.overhead) {\n arcRotateCamera.lockRotation = true;\n arcRotateCamera.target = new Vector3(viewpoint.position.x, 0, viewpoint.position.z);\n }\n return arcRotateCamera;\n }\n case exports.eViewpointType.firstPerson:\n //case eViewpointType.fixedPosition:\n if (convertToCameraType === eCameraType.ArcRotate) {\n const arcRotateCamera = new KbArcRotateCamera(viewpoint.id, 0, 0, 0, viewpoint.position.toVec3(), this.scene);\n const orbitRotation = viewpoint.rotation.toVec3().scale(Math.PI / 180);\n applyViewpointProperties(viewpoint, arcRotateCamera);\n arcRotateCamera.alpha = orbitRotation.y - Math.PI / 2;\n arcRotateCamera.beta = Math.PI / 2 - orbitRotation.x;\n arcRotateCamera.radius = 0.0001;\n return arcRotateCamera;\n }\n else {\n const universalCamera = new UniversalCamera(viewpoint.id, viewpoint.position.toVec3(), this.scene);\n applyViewpointProperties(viewpoint, universalCamera);\n return universalCamera;\n }\n }\n return null;\n function applyViewpointProperties(viewpoint, camera) {\n if (viewpoint.projection === exports.eViewpointProjection.orthographic) {\n camera.mode = Camera.ORTHOGRAPHIC_CAMERA;\n camera.maxZ = 200000;\n if (camera instanceof ArcRotateCamera) {\n camera.orthoWidth = viewpoint.fov || 8;\n camera.wheelPrecision = 60;\n camera.pinchPrecision = 60;\n camera.setPosition(viewpoint.position.toVec3().subtract(camera.target).normalize().scale(5000).add(camera.target));\n }\n }\n else if (viewpoint.projection === exports.eViewpointProjection.perspective) {\n camera.mode = Camera.PERSPECTIVE_CAMERA;\n camera.fov = (viewpoint.fov * Math.PI) / 180;\n if (camera instanceof ArcRotateCamera) {\n camera.wheelDeltaPercentage = 0.01;\n //camera.wheelPrecision = 10;\n camera.speed = 0.0001;\n camera.lowerRadiusLimit = 0.0001;\n camera.pinchDeltaPercentage = 0.001;\n camera.setPosition(viewpoint.position.toVec3());\n }\n }\n if (camera instanceof ArcRotateCamera) {\n if (!viewpoint.allowOrbit) {\n camera.lockRotation = true;\n }\n if (!viewpoint.allowPan) {\n camera.lockPanning = true;\n }\n if (!viewpoint.allowZoom) {\n camera.lockZoom = true;\n }\n camera.useFramingBehavior = false;\n camera.targetScreenOffset.x = viewpoint.offsetX;\n camera.targetScreenOffset.y = viewpoint.offsetY;\n if (viewpoint.limitMinimumTargetDistance) {\n viewer.bbCache.rootBoundingBoxChanged\n .pipe(takeUntil(dispose))\n .pipe(startWith(null))\n .subscribe(() => {\n const bounding = viewer.bbCache.getWorld(sceneNode);\n if (bounding) {\n const minVector = new Vector3(bounding.min.x, bounding.min.y, bounding.min.z);\n const maxVector = new Vector3(bounding.max.x, bounding.max.y, bounding.max.z);\n const boundingSphere = new BoundingSphere(minVector, maxVector);\n camera.lowerRadiusLimit = boundingSphere.radiusWorld;\n }\n else {\n camera.lowerRadiusLimit = viewpoint.minTargetDistance;\n }\n });\n }\n else {\n camera.lowerRadiusLimit = viewpoint.minTargetDistance;\n }\n camera.upperRadiusLimit = viewpoint.maxTargetDistance;\n if (viewpoint.orbitAxis) {\n const upVector = viewpoint.orbitAxis.toVec3();\n if (upVector.length() === 0) {\n upVector.set(0, 1, 0);\n }\n camera.upVector = upVector;\n }\n if (viewpoint.limitLngOrbit) {\n camera.lowerAlphaLimit = (viewpoint.limitMinLongitude * Math.PI) / 180;\n camera.upperAlphaLimit = (viewpoint.limitMaxLongitude * Math.PI) / 180;\n }\n else {\n camera.lowerAlphaLimit = null;\n camera.upperAlphaLimit = null;\n }\n if (viewpoint.limitLatOrbit) {\n camera.lowerBetaLimit = (viewpoint.limitMinLatitude * Math.PI) / 180;\n camera.upperBetaLimit = (viewpoint.limitMaxLatitude * Math.PI) / 180;\n }\n else {\n camera.lowerBetaLimit = 0.001;\n camera.upperBetaLimit = Math.PI - 0.001;\n }\n }\n else if (camera instanceof UniversalCamera) {\n // // For fixed rotation, only allow targeting the camera with the mouse and disable movement.\n // if (viewpoint.type === eViewpointType.fixedPosition) {\n // camera.inputs.clear();\n // camera.inputs.addMouse();\n // }\n camera.inputs.clear();\n if (viewpoint.allowPan) {\n enableFreeMove(camera);\n }\n if (viewpoint.allowOrbit) {\n enableFreeLook(camera);\n }\n }\n if (viewpoint.targetMode === exports.eViewpointTargetMode.point) {\n camera.setTarget(viewpoint.targetPoint.toVec3());\n }\n else if (viewpoint.targetMode === exports.eViewpointTargetMode.mesh) {\n const targetMeshNode = viewpoint._manager.getById(viewpoint.targetMeshId);\n if (targetMeshNode) {\n // TODO: When updating to babylon 5.0, use targetHost property for targetting mesh\n bbCache.rootBoundingBoxChanged.pipe(takeUntil(dispose)).subscribe(() => {\n const bbInfo = bbCache.getWorld(targetMeshNode);\n if (bbInfo) {\n camera.setTarget(bbInfo.center.toVec3());\n }\n });\n }\n }\n else {\n // rotation\n if (camera instanceof ArcRotateCamera) {\n camera.alpha = (viewpoint.rotation.y * Math.PI) / 180; // yaw = longitude\n camera.beta = (viewpoint.rotation.x * Math.PI) / 180; // pitch = latitude\n }\n else {\n camera.rotation = viewpoint.rotation.toVec3().scale(Math.PI / 180);\n }\n }\n bbCache\n .watchForChanges(sceneNode)\n .pipe(takeUntil(dispose))\n .subscribe(bbInfo => {\n var _a;\n if (bbInfo && bbInfo.boundingInfo.diagonalLength > 0) {\n const cameraDistance = bbInfo.boundingInfo.diagonalLength * 2;\n /**\n * This piece of code is very important. if the maxZ is too small, things will disappear as you zoom out.\n * If the maxZ is too large, the camera will not have enough buffer and\n * all kinds of things get messed up - clicking doesn't work, you start getting z-fighting.\n * Smallest the maxZ should be \"10\" due to the grid.\n * However ortho camera should be double that because calculations are made from half of the frame\n * And perspective camera should also take the camera distance into account (but have some sane multiplier based on the mesh size)\n */\n // should we set the minZ here as well, or just hard code it?\n if (viewpoint.projection === exports.eViewpointProjection.orthographic &&\n camera instanceof ArcRotateCamera) {\n camera.setPosition(camera.position\n .subtract(camera.target)\n .normalize()\n .scale(cameraDistance)\n .add(camera.target));\n camera.maxZ = Math.max(cameraDistance, 10) * 2;\n }\n else if (camera instanceof ArcRotateCamera && camera.upperRadiusLimit) {\n camera.maxZ = Math.max(Math.max(cameraDistance, 10) * 16, camera.upperRadiusLimit + cameraDistance);\n }\n else {\n camera.maxZ = Math.max(cameraDistance, 10) * 16;\n }\n // If maxZ is too high then it breaks clicking because the unprojected rays start returning NaN/Infinity\n if (camera.maxZ > 200000) {\n camera.maxZ = 200000;\n }\n if ((_a = viewer.rendererManager.sceneRenderer) === null || _a === void 0 ? void 0 : _a.skybox) {\n // need to resize skybox if it exists, otherwise it will get clipped\n const scale = camera.maxZ / 2 / SKYBOX_SIZE;\n viewer.rendererManager.sceneRenderer.skybox.scaling = new Vector3(scale, scale, scale);\n }\n }\n });\n camera.onDisposeObservable.add(() => {\n dispose.next();\n dispose.complete();\n });\n camera.minZ = 0.01; // hack the camera minz until we have a more automated way to set it\n camera.doNotSerialize = true;\n }\n }\n orient(orientation) {\n const bbCache = this.kbViewer.bbCache;\n if (this._lastOrientation === exports.eCameraOrientation.perspective) {\n this._lastPerspectivePosition = exports.KbVector.From(this.scene.activeCamera.position);\n }\n //get a viewpoint representing the current camera settings\n const viewpoint = new exports.ViewpointNode();\n const camera = this.scene.activeCamera;\n copyCameraStateIntoViewpoint(viewpoint, camera);\n let targetPoint = exports.KbVector.Zero();\n if (viewpoint.targetMode == exports.eViewpointTargetMode.point) {\n targetPoint = viewpoint.targetPoint;\n }\n else if (viewpoint.targetMode == exports.eViewpointTargetMode.mesh) {\n const targetMeshNode = this.kbViewer.sceneNode._manager.getById(viewpoint.targetMeshId);\n if (targetMeshNode) {\n const bbInfo = bbCache.getWorld(targetMeshNode);\n if (bbInfo) {\n targetPoint = bbInfo.center;\n }\n }\n }\n let distance = viewpoint.position.subtract(targetPoint).length();\n let newPos = targetPoint.toObject();\n if (orientation == exports.eCameraOrientation.top) {\n newPos.y += distance;\n }\n else if (orientation == exports.eCameraOrientation.front) {\n newPos.z += distance;\n }\n else if (orientation == exports.eCameraOrientation.left) {\n newPos.x -= distance;\n }\n else if (orientation == exports.eCameraOrientation.back) {\n newPos.z -= distance;\n }\n else if (orientation == exports.eCameraOrientation.right) {\n newPos.x += distance;\n }\n else if (orientation == exports.eCameraOrientation.bottom) {\n newPos.y -= distance;\n }\n else if (orientation == exports.eCameraOrientation.perspective) {\n const lastLength = this._lastPerspectivePosition.length();\n const ratio = distance / lastLength;\n newPos = this._lastPerspectivePosition.scale(ratio);\n }\n viewpoint.position = exports.KbVector.From(newPos);\n if (orientation != exports.eCameraOrientation.perspective) {\n viewpoint.projection = exports.eViewpointProjection.orthographic;\n viewpoint.allowOrbit = false;\n }\n else {\n viewpoint.projection = exports.eViewpointProjection.perspective;\n viewpoint.allowOrbit = true;\n }\n if (this._lastOrientation === exports.eCameraOrientation.perspective &&\n orientation !== exports.eCameraOrientation.perspective) {\n //switching to orthographic so we need to deal with fov\n viewpoint.fov = 2 * distance * Math.tan(toRadians(viewpoint.fov / 2));\n }\n else if (this._lastOrientation !== exports.eCameraOrientation.perspective &&\n orientation === exports.eCameraOrientation.perspective) {\n const orthoWidth = viewpoint.fov; //in orthographic, fov is the ortho width\n distance = orthoWidth / (2 * Math.tan(toRadians(defaults.ViewpointNode.fov / 2)));\n viewpoint.position = this._lastPerspectivePosition.scale(distance / this._lastPerspectivePosition.length());\n viewpoint.fov = defaults.ViewpointNode.fov;\n }\n viewpoint.offsetX = 0;\n viewpoint.offsetY = 0;\n this.moveCameraToViewpoint(viewpoint);\n this.frameCamera(this.kbViewer.sceneNode, {\n rotate: false,\n pan: true,\n zoom: false,\n });\n this._lastOrientation = orientation;\n }\n }\n function enableFreeLook(camera) {\n camera.inputs.addMouse();\n }\n function enableFreeMove(camera) {\n camera.inputs.addKeyboard();\n camera.inputs.addTouch();\n }\n function copyCameraStateIntoViewpoint(viewpoint, camera) {\n let normalizedPosition = camera.position;\n // If this is an orthographic camera, the real position is set very far away to prevent clipping. So normalize the position so the biggest dimension is 5.\n if (camera instanceof ArcRotateCamera && camera.mode === Camera.ORTHOGRAPHIC_CAMERA) {\n normalizedPosition = normalizedPosition.subtract(camera.position).normalize().add(camera.position);\n if (normalizedPosition.length() > 0) {\n const scale = Math.max(...normalizedPosition.asArray().map(dim => Math.abs(dim)));\n normalizedPosition = Vector3.FromArray(normalizedPosition.asArray().map(dim => Math.round((dim / scale) * 500) / 100));\n }\n }\n if (camera instanceof FreeCamera) {\n const rotDegrees = camera.rotation.scale(180 / Math.PI);\n viewpoint.rotation = exports.KbVector.FromVec3(rotDegrees);\n }\n viewpoint.position = exports.KbVector.FromVec3(normalizedPosition);\n viewpoint.offsetX = camera instanceof KbArcRotateCamera ? camera.targetScreenOffset.x : 0;\n viewpoint.offsetY = camera instanceof KbArcRotateCamera ? camera.targetScreenOffset.y : 0;\n viewpoint.fov =\n camera.mode === Camera.ORTHOGRAPHIC_CAMERA\n ? camera.orthoRight - camera.orthoLeft\n : (camera.fov * 180) / Math.PI;\n viewpoint.projection =\n camera.mode === Camera.ORTHOGRAPHIC_CAMERA\n ? exports.eViewpointProjection.orthographic\n : exports.eViewpointProjection.perspective;\n }\n function normalizeRotation(alpha) {\n const oneRotation = Math.PI * 2;\n const rotations = Math.floor(alpha / oneRotation);\n const radiansToRemove = rotations > 1 ? rotations * oneRotation : 0;\n const normalizedRotation = alpha - radiansToRemove;\n return normalizedRotation;\n }\n\n /**\n * Class used to store a cell in an octree\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimizeOctrees\n */\n class OctreeBlock {\n /**\n * Creates a new block\n * @param minPoint defines the minimum vector (in world space) of the block's bounding box\n * @param maxPoint defines the maximum vector (in world space) of the block's bounding box\n * @param capacity defines the maximum capacity of this block (if capacity is reached the block will be split into sub blocks)\n * @param depth defines the current depth of this block in the octree\n * @param maxDepth defines the maximal depth allowed (beyond this value, the capacity is ignored)\n * @param creationFunc defines a callback to call when an element is added to the block\n */\n constructor(minPoint, maxPoint, capacity, depth, maxDepth, creationFunc) {\n /**\n * Gets the content of the current block\n */\n this.entries = new Array();\n this._boundingVectors = new Array();\n this._capacity = capacity;\n this._depth = depth;\n this._maxDepth = maxDepth;\n this._creationFunc = creationFunc;\n this._minPoint = minPoint;\n this._maxPoint = maxPoint;\n this._boundingVectors.push(minPoint.clone());\n this._boundingVectors.push(maxPoint.clone());\n this._boundingVectors.push(minPoint.clone());\n this._boundingVectors[2].x = maxPoint.x;\n this._boundingVectors.push(minPoint.clone());\n this._boundingVectors[3].y = maxPoint.y;\n this._boundingVectors.push(minPoint.clone());\n this._boundingVectors[4].z = maxPoint.z;\n this._boundingVectors.push(maxPoint.clone());\n this._boundingVectors[5].z = minPoint.z;\n this._boundingVectors.push(maxPoint.clone());\n this._boundingVectors[6].x = minPoint.x;\n this._boundingVectors.push(maxPoint.clone());\n this._boundingVectors[7].y = minPoint.y;\n }\n // Property\n /**\n * Gets the maximum capacity of this block (if capacity is reached the block will be split into sub blocks)\n */\n get capacity() {\n return this._capacity;\n }\n /**\n * Gets the minimum vector (in world space) of the block's bounding box\n */\n get minPoint() {\n return this._minPoint;\n }\n /**\n * Gets the maximum vector (in world space) of the block's bounding box\n */\n get maxPoint() {\n return this._maxPoint;\n }\n // Methods\n /**\n * Add a new element to this block\n * @param entry defines the element to add\n */\n addEntry(entry) {\n if (this.blocks) {\n for (let index = 0; index < this.blocks.length; index++) {\n const block = this.blocks[index];\n block.addEntry(entry);\n }\n return;\n }\n this._creationFunc(entry, this);\n if (this.entries.length > this.capacity && this._depth < this._maxDepth) {\n this.createInnerBlocks();\n }\n }\n /**\n * Remove an element from this block\n * @param entry defines the element to remove\n */\n removeEntry(entry) {\n if (this.blocks) {\n for (let index = 0; index < this.blocks.length; index++) {\n const block = this.blocks[index];\n block.removeEntry(entry);\n }\n return;\n }\n const entryIndex = this.entries.indexOf(entry);\n if (entryIndex > -1) {\n this.entries.splice(entryIndex, 1);\n }\n }\n /**\n * Add an array of elements to this block\n * @param entries defines the array of elements to add\n */\n addEntries(entries) {\n for (let index = 0; index < entries.length; index++) {\n const mesh = entries[index];\n this.addEntry(mesh);\n }\n }\n /**\n * Test if the current block intersects the frustum planes and if yes, then add its content to the selection array\n * @param frustumPlanes defines the frustum planes to test\n * @param selection defines the array to store current content if selection is positive\n * @param allowDuplicate defines if the selection array can contains duplicated entries\n */\n select(frustumPlanes, selection, allowDuplicate) {\n if (BoundingBox.IsInFrustum(this._boundingVectors, frustumPlanes)) {\n if (this.blocks) {\n for (let index = 0; index < this.blocks.length; index++) {\n const block = this.blocks[index];\n block.select(frustumPlanes, selection, allowDuplicate);\n }\n return;\n }\n if (allowDuplicate) {\n selection.concat(this.entries);\n }\n else {\n selection.concatWithNoDuplicate(this.entries);\n }\n }\n }\n /**\n * Test if the current block intersect with the given bounding sphere and if yes, then add its content to the selection array\n * @param sphereCenter defines the bounding sphere center\n * @param sphereRadius defines the bounding sphere radius\n * @param selection defines the array to store current content if selection is positive\n * @param allowDuplicate defines if the selection array can contains duplicated entries\n */\n intersects(sphereCenter, sphereRadius, selection, allowDuplicate) {\n if (BoundingBox.IntersectsSphere(this._minPoint, this._maxPoint, sphereCenter, sphereRadius)) {\n if (this.blocks) {\n for (let index = 0; index < this.blocks.length; index++) {\n const block = this.blocks[index];\n block.intersects(sphereCenter, sphereRadius, selection, allowDuplicate);\n }\n return;\n }\n if (allowDuplicate) {\n selection.concat(this.entries);\n }\n else {\n selection.concatWithNoDuplicate(this.entries);\n }\n }\n }\n /**\n * Test if the current block intersect with the given ray and if yes, then add its content to the selection array\n * @param ray defines the ray to test with\n * @param selection defines the array to store current content if selection is positive\n */\n intersectsRay(ray, selection) {\n if (ray.intersectsBoxMinMax(this._minPoint, this._maxPoint)) {\n if (this.blocks) {\n for (let index = 0; index < this.blocks.length; index++) {\n const block = this.blocks[index];\n block.intersectsRay(ray, selection);\n }\n return;\n }\n selection.concatWithNoDuplicate(this.entries);\n }\n }\n /**\n * Subdivide the content into child blocks (this block will then be empty)\n */\n createInnerBlocks() {\n OctreeBlock._CreateBlocks(this._minPoint, this._maxPoint, this.entries, this._capacity, this._depth, this._maxDepth, this, this._creationFunc);\n this.entries.splice(0);\n }\n /**\n * @internal\n */\n static _CreateBlocks(worldMin, worldMax, entries, maxBlockCapacity, currentDepth, maxDepth, target, creationFunc) {\n target.blocks = new Array();\n const blockSize = new Vector3((worldMax.x - worldMin.x) / 2, (worldMax.y - worldMin.y) / 2, (worldMax.z - worldMin.z) / 2);\n // Segmenting space\n for (let x = 0; x < 2; x++) {\n for (let y = 0; y < 2; y++) {\n for (let z = 0; z < 2; z++) {\n const localMin = worldMin.add(blockSize.multiplyByFloats(x, y, z));\n const localMax = worldMin.add(blockSize.multiplyByFloats(x + 1, y + 1, z + 1));\n const block = new OctreeBlock(localMin, localMax, maxBlockCapacity, currentDepth + 1, maxDepth, creationFunc);\n block.addEntries(entries);\n target.blocks.push(block);\n }\n }\n }\n }\n }\n\n /**\n * Octrees are a really powerful data structure that can quickly select entities based on space coordinates.\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimizeOctrees\n */\n class Octree {\n /**\n * Creates a octree\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimizeOctrees\n * @param creationFunc function to be used to instantiate the octree\n * @param maxBlockCapacity defines the maximum number of meshes you want on your octree's leaves (default: 64)\n * @param maxDepth defines the maximum depth (sub-levels) for your octree. Default value is 2, which means 8 8 8 = 512 blocks :) (This parameter takes precedence over capacity.)\n */\n constructor(creationFunc, maxBlockCapacity, \n /** Defines the maximum depth (sub-levels) for your octree. Default value is 2, which means 8 8 8 = 512 blocks :) (This parameter takes precedence over capacity.) */\n maxDepth = 2) {\n this.maxDepth = maxDepth;\n /**\n * Content stored in the octree\n */\n this.dynamicContent = new Array();\n this._maxBlockCapacity = maxBlockCapacity || 64;\n this._selectionContent = new SmartArrayNoDuplicate(1024);\n this._creationFunc = creationFunc;\n }\n // Methods\n /**\n * Updates the octree by adding blocks for the passed in meshes within the min and max world parameters\n * @param worldMin worldMin for the octree blocks var blockSize = new Vector3((worldMax.x - worldMin.x) / 2, (worldMax.y - worldMin.y) / 2, (worldMax.z - worldMin.z) / 2);\n * @param worldMax worldMax for the octree blocks var blockSize = new Vector3((worldMax.x - worldMin.x) / 2, (worldMax.y - worldMin.y) / 2, (worldMax.z - worldMin.z) / 2);\n * @param entries meshes to be added to the octree blocks\n */\n update(worldMin, worldMax, entries) {\n OctreeBlock._CreateBlocks(worldMin, worldMax, entries, this._maxBlockCapacity, 0, this.maxDepth, this, this._creationFunc);\n }\n /**\n * Adds a mesh to the octree\n * @param entry Mesh to add to the octree\n */\n addMesh(entry) {\n for (let index = 0; index < this.blocks.length; index++) {\n const block = this.blocks[index];\n block.addEntry(entry);\n }\n }\n /**\n * Remove an element from the octree\n * @param entry defines the element to remove\n */\n removeMesh(entry) {\n for (let index = 0; index < this.blocks.length; index++) {\n const block = this.blocks[index];\n block.removeEntry(entry);\n }\n }\n /**\n * Selects an array of meshes within the frustum\n * @param frustumPlanes The frustum planes to use which will select all meshes within it\n * @param allowDuplicate If duplicate objects are allowed in the resulting object array\n * @returns array of meshes within the frustum\n */\n select(frustumPlanes, allowDuplicate) {\n this._selectionContent.reset();\n for (let index = 0; index < this.blocks.length; index++) {\n const block = this.blocks[index];\n block.select(frustumPlanes, this._selectionContent, allowDuplicate);\n }\n if (allowDuplicate) {\n this._selectionContent.concat(this.dynamicContent);\n }\n else {\n this._selectionContent.concatWithNoDuplicate(this.dynamicContent);\n }\n return this._selectionContent;\n }\n /**\n * Test if the octree intersect with the given bounding sphere and if yes, then add its content to the selection array\n * @param sphereCenter defines the bounding sphere center\n * @param sphereRadius defines the bounding sphere radius\n * @param allowDuplicate defines if the selection array can contains duplicated entries\n * @returns an array of objects that intersect the sphere\n */\n intersects(sphereCenter, sphereRadius, allowDuplicate) {\n this._selectionContent.reset();\n for (let index = 0; index < this.blocks.length; index++) {\n const block = this.blocks[index];\n block.intersects(sphereCenter, sphereRadius, this._selectionContent, allowDuplicate);\n }\n if (allowDuplicate) {\n this._selectionContent.concat(this.dynamicContent);\n }\n else {\n this._selectionContent.concatWithNoDuplicate(this.dynamicContent);\n }\n return this._selectionContent;\n }\n /**\n * Test if the octree intersect with the given ray and if yes, then add its content to resulting array\n * @param ray defines the ray to test with\n * @returns array of intersected objects\n */\n intersectsRay(ray) {\n this._selectionContent.reset();\n for (let index = 0; index < this.blocks.length; index++) {\n const block = this.blocks[index];\n block.intersectsRay(ray, this._selectionContent);\n }\n this._selectionContent.concatWithNoDuplicate(this.dynamicContent);\n return this._selectionContent;\n }\n }\n /**\n * Adds a mesh into the octree block if it intersects the block\n * @param entry\n * @param block\n */\n Octree.CreationFuncForMeshes = (entry, block) => {\n const boundingInfo = entry.getBoundingInfo();\n if (!entry.isBlocked && boundingInfo.boundingBox.intersectsMinMax(block.minPoint, block.maxPoint)) {\n block.entries.push(entry);\n }\n };\n /**\n * Adds a submesh into the octree block if it intersects the block\n * @param entry\n * @param block\n */\n Octree.CreationFuncForSubMeshes = (entry, block) => {\n const boundingInfo = entry.getBoundingInfo();\n if (boundingInfo.boundingBox.intersectsMinMax(block.minPoint, block.maxPoint)) {\n block.entries.push(entry);\n }\n };\n\n const CLICK_THRESHOLD = 8;\n class Dragger {\n constructor(viewer, scene) {\n this.viewer = viewer;\n this.scene = scene;\n this._externalConnectors = new Map(); //connectors out in the scene that are not part of what is being dragged\n this._dragConnectors = new Map(); //connectors that are children of what is being dragged\n this._dragConnectorsByDynamicId = new Map(); //the drag connectors stored by their dynamic id\n this._dropPreviewConnectors = new Map(); //the drop preview is a clone of the dragNode, this tracks the connector meshes in the drop preview\n this._dragFriends = new Set();\n this._dragFriendMeshes = new Array();\n this._dragFriendConnectors = new Set();\n this._matedConnectors = new Set();\n }\n startDrag(pointerArgs, draggable, onComplete) {\n var _a;\n let totalMoved = 0;\n let lastX;\n let lastY;\n this._pointerObserver = this.scene.onPointerObservable.add(p => {\n switch (p.type) {\n case PointerEventTypes.POINTERMOVE: {\n let movX = 0;\n let movY = 0;\n if (typeof lastX === 'undefined' || typeof lastY === 'undefined') {\n lastX = p.event.offsetX;\n lastY = p.event.offsetY;\n }\n else {\n movX = Math.abs(lastX - p.event.offsetX);\n movY = Math.abs(lastY - p.event.offsetY);\n }\n totalMoved += Math.sqrt(movX * movX + movY * movY);\n if (totalMoved > CLICK_THRESHOLD) {\n this.startDragCommit(pointerArgs, draggable, onComplete);\n this.dragging(p);\n }\n break;\n }\n case PointerEventTypes.POINTERUP: {\n if (totalMoved > CLICK_THRESHOLD) {\n this.dropped(p);\n }\n this.scene.onPointerObservable.remove(this._pointerObserver);\n this._pointerObserver = undefined;\n break;\n }\n }\n });\n (_a = this.scene.activeCamera) === null || _a === void 0 ? void 0 : _a.detachControl();\n }\n startDragCommit(pointerArgs, draggable, onComplete) {\n if (!this._draggable) {\n this._connectorRenderer = this.viewer.rendererManager.connector;\n this._dragNode = pointerArgs.sourceNode;\n this._dragNodeStartPosition = this._dragNode.position.clone();\n this._dragMesh = this.viewer.rendererManager.mesh.get3d(this._dragNode);\n if (this._dragMesh.parent) {\n this._dragNodeInverseTransform = this._dragMesh.parent\n .computeWorldMatrix(true)\n .clone()\n .setTranslation(Vector3.Zero())\n .invert();\n }\n this._draggable = draggable;\n if (this._draggable.connectionMode == null)\n this._draggable.connectionMode = exports.eConnectionMode.none;\n if (this._draggable.connectionType == null)\n this._draggable.connectionType = exports.eMateType.fastened;\n if (this._draggable.connectRadius == null) {\n if (this._draggable.connectionMode == exports.eConnectionMode.screen) {\n this._draggable.connectRadius = 5; //5 pixels\n }\n else {\n this._draggable.connectRadius = 0.3; //3d units\n }\n }\n if (this._draggable.dragOpacity == null)\n this._draggable.dragOpacity = 0.7;\n this._onComplete = onComplete;\n this.setupDragSurface(pointerArgs);\n if (this._draggable.dragStart) {\n this._draggable.dragStart(this.getDragArgs(pointerArgs, this._dragNodeStartPosition, this._dragMesh.rotationQuaternion));\n }\n this.setupDragFriends();\n this.setupConnectors();\n this.setOpacity(this._draggable.dragOpacity, this._dragMesh, ...this._dragFriendMeshes);\n }\n }\n dragging(p) {\n // this._dragNode._dragging = true;\n const newConnection = this.findConnection();\n //let connectionChanged = !this.connectionsAreEqual(newConnection, this._activeConnection);\n this._activeConnection = newConnection;\n if (this._activeConnection) {\n //if (connectionChanged) { //don't bother processing the connection if it's the same\n this.showDropPreview(this._activeConnection);\n //}\n }\n else {\n this.hideDropPreview();\n }\n this.processDragPosition(p, this._draggable.dragging);\n }\n applyDragResultToNode(node, dragArgs, applyEuler) {\n node.position = dragArgs.newPos;\n // If euler rotation was changed, apply the changed euler rotation. If not, then apply the quaternion rotation regardless of whether it was changed.\n if (applyEuler) {\n node.eulerRotation = dragArgs.newRotationEuler;\n const newKbQuat = exports.KbQuaternion.FromYawPitchRoll(dragArgs.newRotationEuler);\n dragArgs.newRotationAngle = newKbQuat.getAngle();\n dragArgs.newRotationAxis = newKbQuat.getAxis();\n node.rotationAxis = dragArgs.newRotationAxis;\n node.rotationAngle = dragArgs.newRotationAngle;\n }\n else {\n node.rotationAxis = dragArgs.newRotationAxis;\n node.rotationAngle = dragArgs.newRotationAngle;\n node.eulerRotation = exports.KbQuaternion.FromRotationAxis(dragArgs.newRotationAxis, dragArgs.newRotationAngle).toEuler();\n }\n }\n dropped(p) {\n this._dragNode._primeMover = false;\n const complete = new Promise(resolve => {\n if (this._activeConnection) {\n this._activeConnection.otherConnector._parent._primeMover = true;\n this.connect(this._activeConnection, this._dragMesh);\n // let newRotation = KbVector.FromVec3(this._dragMesh.rotation);\n const dragArgs = {\n ...this.getDragArgs(p, this._dragMesh.position, this._dragMesh.rotationQuaternion),\n connection: this._activeConnection,\n };\n if (this._draggable.withMates) {\n dragArgs.mate = this.viewer.sceneNode._manager.create(Token.MateNode, {\n connector1: this.viewer.sceneNode.getNestedPathTo(dragArgs.connection.otherConnector),\n connector2: this.viewer.sceneNode.getNestedPathTo(dragArgs.connection.dragConnector),\n mateType: this._draggable.connectionType,\n });\n }\n const previousEuler = dragArgs.newRotationEuler.clone();\n this.applyDragResultToNode(this._dragNode, dragArgs, !previousEuler.equal(dragArgs.newRotationEuler));\n if (dragArgs.mate) {\n this.viewer.sceneNode.mates.push(dragArgs.mate);\n }\n this.viewer.forceAndWaitForNextRender().then(() => {\n if (this._draggable.dropped) {\n this._draggable.dropped(dragArgs);\n if (!dragArgs.allow) {\n console.warn('Cannot disallow dropping when using connectors, use canConnect handler instead');\n }\n }\n resolve();\n });\n }\n else {\n this.processDragPosition(p, this._draggable.dragging);\n this.processDragPosition(p, this._draggable.dropped);\n resolve();\n }\n });\n complete.then(() => {\n var _a;\n this.toggleAllConnectors(false);\n this.resetOpacity(this._dragMesh, ...this._dragFriendMeshes);\n (_a = this.scene.activeCamera) === null || _a === void 0 ? void 0 : _a.attachControl(this.viewer.canvas);\n if (this._onComplete)\n this._onComplete();\n this.disposeDropPreview();\n this._onComplete = undefined;\n this._dragNode = undefined;\n this._dragMesh = undefined;\n this._draggable = undefined;\n this._dragPlane = undefined;\n this._dragSurfaces = undefined;\n this._onComplete = undefined;\n if (this._externalConnectors)\n this._externalConnectors.clear();\n if (this._dragConnectors)\n this._dragConnectors.clear();\n if (this._dragConnectorsByDynamicId)\n this._dragConnectorsByDynamicId.clear();\n if (this._dropPreviewConnectors)\n this._dropPreviewConnectors.clear();\n this._oct = undefined;\n this._quadTree = undefined;\n if (this._quadTree)\n this._quadTree.items;\n this._dropPreview = undefined;\n this._activeConnection = undefined;\n this._connectorRenderer = undefined;\n if (this._dragFriends)\n this._dragFriends.clear();\n if (this._dragFriendMeshes)\n this._dragFriendMeshes.clear();\n if (this._dragFriendConnectors)\n this._dragFriendConnectors.clear();\n if (this._matedConnectors)\n this._matedConnectors.clear();\n });\n }\n processDragPosition(p, event) {\n const newMouseIntersect = this.intersectPointerToDragSurface(p);\n if (newMouseIntersect) {\n let newPos;\n if (this._draggable.disableOffset || this._draggable.mode == exports.eDragMode.surface) {\n newPos = exports.KbVector.FromVec3(newMouseIntersect);\n }\n else {\n if (this._startMouseIntersect) {\n const delta = newMouseIntersect.subtract(this._startMouseIntersect);\n if (this._dragNodeInverseTransform) {\n Vector3.TransformCoordinatesToRef(delta, this._dragNodeInverseTransform, delta);\n }\n newPos = this._dragNodeStartPosition.add(delta);\n }\n else {\n newPos = this._dragNodeStartPosition;\n this._startMouseIntersect = newMouseIntersect;\n }\n }\n const dragArgs = this.getDragArgs(p, newPos, this._dragMesh.rotationQuaternion);\n const previousEuler = dragArgs.newRotationEuler.clone();\n if (event) {\n event(dragArgs);\n }\n if (dragArgs.allow) {\n this.applyDragResultToNode(this._dragNode, dragArgs, !previousEuler.equal(dragArgs.newRotationEuler));\n }\n return dragArgs;\n }\n return undefined;\n }\n setupDragSurface(p) {\n const d = this._draggable;\n const manager = this._dragNode._manager;\n if (d.mode == exports.eDragMode.plane) {\n // d.planeOrigin = p.pickedPoint!; // d.planeOrigin || this._dragNode.position;\n /** intersection plane is defined by the pickedPoint and the plane normal */\n d.planeNormal = d.planeNormal || new exports.KbVector(0, 1, 0);\n this._dragPlane = Plane.FromPositionAndNormal(p.pickInfo.pickedPoint, d.planeNormal.toVec3());\n }\n else if (d.mode == exports.eDragMode.free) {\n const cameraDirection = this.scene.activeCamera.position.subtract(this._dragMesh.getAbsolutePosition());\n this._dragPlane = Plane.FromPositionAndNormal(this._dragNode.position.toVec3(), cameraDirection);\n }\n else if (d.mode == exports.eDragMode.surface) {\n const surfaceIds = Array.isArray(d.surfaceNodes) ? d.surfaceNodes : [d.surfaceNodes];\n this._dragSurfaces =\n surfaceIds.map(sn => { var _a; return sn && ((_a = manager.getById(sn)) === null || _a === void 0 ? void 0 : _a._dynamicId); }).filter(sn => sn != null) ||\n [];\n }\n }\n intersectPointerToDragSurface(p) {\n return this.intersectDragSurface(p.pickInfo.ray);\n }\n intersectDragSurface(ray) {\n switch (this._draggable.mode) {\n case exports.eDragMode.plane:\n case exports.eDragMode.free: {\n const distance = ray.intersectsPlane(this._dragPlane);\n if (distance != null) {\n return ray.direction.scale(distance).add(ray.origin);\n }\n break;\n }\n case exports.eDragMode.surface: {\n const surfacePick = this.scene.pickWithRay(ray, mesh => {\n return this._dragSurfaces.some(sn => mesh.id == sn);\n });\n if (surfacePick && surfacePick.pickedPoint) {\n return surfacePick.pickedPoint;\n }\n break;\n }\n }\n return undefined;\n }\n getDragArgs(p, newPos, newRotation) {\n var _a, _b;\n const newKbRotation = exports.KbQuaternion.FromQuat(newRotation || Quaternion.Identity());\n return {\n dragNode: this._dragNode,\n startPos: this._dragNodeStartPosition,\n allow: true,\n u: ((_a = p.pickInfo) === null || _a === void 0 ? void 0 : _a.bu) || 0.5,\n v: ((_b = p.pickInfo) === null || _b === void 0 ? void 0 : _b.bv) || 0.5,\n newPos: newPos instanceof Vector3 ? exports.KbVector.FromVec3(newPos) : newPos,\n newRotationAxis: newKbRotation.getAxis(),\n newRotationAngle: newKbRotation.getAngle(),\n newRotationEuler: newKbRotation.toEuler(),\n };\n }\n findConnection() {\n if (this._draggable.connectionMode != exports.eConnectionMode.none) {\n const potentialConnectors = new Array();\n for (const [dragConnector, dragConnectorMesh] of this._dragConnectors) {\n const dragConPos = dragConnectorMesh.getAbsolutePosition();\n for (const p of this.intersectOctree(dragConPos)) {\n potentialConnectors.push({\n ...p,\n dragConnector,\n dragConnectorMesh,\n });\n }\n }\n if (this._draggable.connectionMode == exports.eConnectionMode.world) {\n potentialConnectors.sortBy(p => p.proximity);\n }\n else {\n potentialConnectors.sortBy(p => p.distanceToCamera);\n }\n for (const p of potentialConnectors) {\n const otherConnector = getByDynamicId(p.otherMesh.id);\n const a = {\n dragConnector: p.dragConnector,\n otherConnector: otherConnector,\n dragNode: p.dragConnector._parent,\n otherNode: otherConnector._parent,\n allow: true,\n // Private properties below, don't expose to Snap\n dragConnectorMesh: p.dragConnectorMesh,\n otherConnectorMesh: p.otherMesh,\n };\n if (this._draggable.canConnect) {\n this._draggable.canConnect(a);\n }\n if (a.allow) {\n return a;\n }\n }\n }\n }\n connectionsAreEqual(c1, c2) {\n return ((!c1 && !c2) || (c1 && c2 && c1.dragConnector == c2.dragConnector && c1.otherConnector == c2.otherConnector));\n }\n setupConnectors() {\n if (this._draggable.connectionMode != exports.eConnectionMode.none) {\n const externalConnectorMeshes = new Set();\n for (const c of this.viewer.sceneNode.getAllConnectors()) {\n if (!this._dragFriendConnectors.has(c) &&\n (!this._draggable.withMates || !this._matedConnectors.has(c)) &&\n this.parentsEnabled(c)) {\n let mesh = this.getConnectorMesh(c);\n if (!mesh) {\n this._connectorRenderer.update3d(c);\n mesh = this.getConnectorMesh(c);\n }\n externalConnectorMeshes.add(mesh);\n this._externalConnectors.set(c, mesh);\n }\n }\n const dragConnectors = this._dragNode\n .filterNodes(n => isSpaceNode(n) && n.visible, true, false)\n .selectMany(n => n.connectors);\n for (const c of dragConnectors) {\n //remove the drag connectors from the outside connectors\n if (this._externalConnectors.has(c)) {\n const mesh = this._externalConnectors.get(c);\n this._dragConnectors.set(c, mesh);\n this._dragConnectorsByDynamicId.set(c._dynamicId, c);\n this._externalConnectors.delete(c);\n externalConnectorMeshes.delete(mesh);\n }\n }\n this.toggleAllConnectors(true);\n if (this._draggable.connectionMode == exports.eConnectionMode.world) {\n //create an octree for holding the connectors\n this._oct = new Octree(Octree.CreationFuncForMeshes, 64, 2);\n //get the bounding extents of all the external connectors to contain the octree volume\n const boundingVectors = this.scene.getWorldExtends(m => externalConnectorMeshes.has(m));\n //update the octree with the connector meshes\n this._oct.update(boundingVectors.min, boundingVectors.max, [...externalConnectorMeshes]);\n }\n else {\n //screen mode\n const extConnectorPositions = new Array();\n externalConnectorMeshes.forEach(m => {\n extConnectorPositions.push({\n mesh: m,\n pos: toScreen(m.getAbsolutePosition(), this.scene.activeCamera),\n });\n });\n const engine = this.scene.getEngine();\n this._quadTree = new QuadTree({\n items: extConnectorPositions,\n xSelector: d => d.pos.x,\n ySelector: d => d.pos.y,\n capacity: 4,\n xmin: 0,\n xmax: engine.getRenderWidth(),\n ymin: 0,\n ymax: engine.getRenderHeight(),\n });\n }\n }\n }\n parentsEnabled(item) {\n while (item._parent && item._parent instanceof exports.SpaceNode) {\n if (!item._parent.visible) {\n return false;\n }\n item = item._parent;\n }\n return true;\n }\n setupDragFriends() {\n const activeMates = this.viewer.getActiveMates();\n for (const mate of activeMates) {\n const c1 = mate.getC1();\n if (c1 && isConnector(c1))\n this._matedConnectors.add(c1);\n const c2 = mate.getC2();\n if (c2 && isConnector(c2))\n this._matedConnectors.add(c2);\n }\n this._dragFriends = getNodeChain(this._dragNode, activeMates);\n this._dragFriends.forEach(friend => {\n if (!(friend instanceof exports.SpaceNode)) {\n this._dragFriends.delete(friend);\n }\n });\n for (const friendNode of this._dragFriends) {\n if (friendNode instanceof exports.SpaceNode) {\n friendNode._primeMover = false;\n const friendMesh = this.viewer.rendererManager.mesh.get3d(friendNode);\n if (friendMesh) {\n this._dragFriendMeshes.push(friendMesh);\n }\n const connectors = friendNode.filterByType(Token.Connector);\n for (const c of connectors)\n this._dragFriendConnectors.add(c);\n }\n }\n this._dragNode._primeMover = true;\n }\n getConnectorMesh(c) {\n return this._connectorRenderer.get3d(c);\n }\n toggleAllConnectors(show) {\n if (this._draggable.showConnectors) {\n for (const [c] of this._externalConnectors) {\n c._visible = show;\n }\n for (const [c] of this._dragConnectors) {\n c._visible = show;\n }\n }\n }\n intersectOctree(v) {\n let potentials = new Array();\n if (this._draggable.connectionMode == exports.eConnectionMode.world) {\n const bpotentials = this._oct.intersects(v, this._draggable.connectRadius, false);\n bpotentials.forEach(p => {\n const proximity = p.getAbsolutePosition().subtract(v).length();\n if (proximity < this._draggable.connectRadius) {\n potentials.push({ otherMesh: p, proximity });\n }\n });\n //potentials.sortBy(p => p.proximity);\n }\n else {\n //screen mode\n const screenPos = toScreen(v, this.scene.activeCamera);\n potentials = this._quadTree.find(screenPos, this._draggable.connectRadius).map(p => {\n return {\n otherMesh: p.item.mesh,\n proximity: p.proximity,\n distanceToCamera: Vector3.Distance(this.scene.activeCamera.position, p.item.mesh.getAbsolutePosition()),\n };\n });\n //potentials.sortBy(i => i.distanceToCamera);\n // let ray = this.scene.createPickingRay(screenPos.x, screenPos.y, Matrix.Identity(), this.scene.activeCamera);\n // let bpotentials = this._oct.intersectsRay(ray);\n // bpotentials.forEach(p => {\n // let potentialScreenPos = toScreen(p.getAbsolutePosition(), this.scene.activeCamera!);\n // let screenProximity = Math.sqrt(Math.pow(screenPos.x - potentialScreenPos.x, 2) + Math.pow(screenPos.y - potentialScreenPos.y, 2));\n // let realProximity = p.getAbsolutePosition().subtract(v).length();\n // console.log({ screenProximity, realProximity });\n // if (screenProximity < this._draggable.connectRadius!) {\n // potentials.push({ mesh: p, proximity: screenProximity });\n // }\n // });\n // potentials.sortBy(p => p.proximity);\n }\n return potentials;\n }\n showDropPreview(c) {\n if (!this._dropPreview) {\n /**\n * for the drop preview, we get the drag node and any other nodes that are connected to it through mates\n * and clone them, put them in a transformNode, and then that transform node becomes the drop preview\n */\n this._dropPreview = new TransformNode('_droppreview', this.viewer.scene);\n this._dropPreview.position = Vector3.Zero();\n this._dropPreview.rotationQuaternion = Quaternion.Identity();\n this._dropPreview.computeWorldMatrix(true);\n const dragMeshTransform = this._dragMesh.computeWorldMatrix(true).getRotationMatrix();\n dragMeshTransform.invert();\n for (const friendMesh of [this._dragMesh, ...this._dragFriendMeshes]) {\n createDragPreview(friendMesh, this._dropPreview, this);\n }\n this._dropPreview.setPreTransformMatrix(dragMeshTransform);\n }\n else {\n //line drop preview back up with the source models as they might have rotated\n if (this._draggable.connectionType != exports.eMateType.fastened) {\n this.connect({\n dragConnector: c.dragConnector,\n otherConnector: c.dragConnector,\n otherConnectorMesh: c.dragConnectorMesh,\n dragConnectorMesh: this._dropPreviewConnectors.get(c.dragConnector), //pass in the drop preview connector mesh\n }, this._dropPreview, exports.eMateType.fastened, true);\n }\n }\n this.connect({\n ...c,\n dragConnectorMesh: this._dropPreviewConnectors.get(c.dragConnector), //pass in the drop preview connector mesh\n }, this._dropPreview);\n this._dropPreview.setEnabled(true);\n //hide the drag node\n this._dragMesh.setEnabled(false);\n for (const friend of this._dragFriendMeshes)\n friend.setEnabled(false);\n function createDragPreview(mesh, parent, self) {\n const clonedMeshGroup = mesh.clone(mesh.id + '_dropPreview', mesh.parent, true);\n if (clonedMeshGroup) {\n clonedMeshGroup.setPivotMatrix(Matrix.Identity());\n // Cloning does not copy the pretransform matrix over so we have to do that manually\n // Otherwise the connector meshes scaling does not get applied to the preview meshes\n clonedMeshGroup.setEnabled(mesh.isEnabled());\n if (c instanceof Mesh && mesh instanceof Mesh) {\n c.setPreTransformMatrix(mesh.getPivotMatrix());\n }\n clonedMeshGroup.setParent(parent); // setting the parent with setParent will make the node stay in the same absolute position.\n const children = mesh.getChildren(child => child instanceof TransformNode, true);\n children.forEach(child => {\n createDragPreview(child, clonedMeshGroup, self);\n });\n clonedMeshGroup.metadata = {}; // babylon metadata refers to same instance of object after cloning so we clear it out\n const kbnode = getByDynamicId(mesh.id);\n if (kbnode && kbnode instanceof exports.Connector) {\n const connector = kbnode;\n self._dropPreviewConnectors.set(connector, clonedMeshGroup);\n clonedMeshGroup.setEnabled(connector._visible); //connector.enabled && !!this._draggable.showConnectors);\n }\n else {\n self.setOpacity(self._draggable.dragOpacity, clonedMeshGroup);\n }\n }\n }\n }\n connect(c, followerMesh, mateType = this._draggable.connectionType, flip = false) {\n this.viewer.pauseRenderWhile(() => {\n mateConnectors({\n mateType: mateType,\n flip: flip,\n offsetX: 0,\n offsetY: 0,\n offsetZ: 0,\n }, followerMesh, c.otherConnectorMesh, c.dragConnectorMesh);\n });\n }\n hideDropPreview() {\n if (this._dropPreview)\n this._dropPreview.setEnabled(false);\n this._dragMesh.setEnabled(true); //show the drag node again\n if (this._dragFriendMeshes) {\n for (const friend of this._dragFriendMeshes) {\n friend.setEnabled(getByDynamicId(friend.id).visible);\n }\n }\n }\n disposeDropPreview() {\n this.hideDropPreview();\n if (this._dropPreview)\n this._dropPreview.dispose();\n if (this._dragFriendMeshes)\n this._dragFriendMeshes.clear();\n if (this._dragFriends)\n this._dragFriends.clear();\n }\n getConnectorLocalPoint(connector) {\n if (connector.positionMode == exports.eConnectorPositionMode.absolute) {\n return connector.position;\n }\n }\n setOpacity(value, ...nodes) {\n if (this.shouldSetOpacity()) {\n for (const node of nodes) {\n for (const m of this.getAllMeshes(node, m => !(m instanceof InstancedMesh))) {\n //set opacity of all non-connector meshes being dragged\n if (!m.metadata) {\n m.metadata = {};\n }\n m.metadata.originalVisibility = m.visibility;\n m.visibility = value;\n }\n }\n }\n }\n resetOpacity(...nodes) {\n for (const node of nodes) {\n for (const m of this.getAllMeshes(node, m => !(m instanceof InstancedMesh))) {\n //set opacity of all non-connector meshes being dragged\n if (m.metadata && m.metadata.originalVisibility !== undefined) {\n m.visibility = m.metadata.originalVisibility;\n delete m.metadata.originalVisibility;\n }\n }\n }\n }\n shouldSetOpacity() {\n return this._draggable.connectionMode != exports.eConnectionMode.none && this._draggable.showConnectors; // && this._dragConnectors.size ;\n }\n /**get's all child meshes and includes the given node if it's a mesh */\n getAllMeshes(node, predicate) {\n const meshes = new Set(node.getChildMeshes(false, predicate));\n if (node instanceof AbstractMesh)\n meshes.add(node);\n return meshes;\n }\n }\n\n /**\n * Class for generating OBJ data from a Babylon scene.\n */\n class OBJExport {\n /**\n * Exports the geometry of a Mesh array in .OBJ file format (text)\n * @param mesh defines the list of meshes to serialize\n * @param materials defines if materials should be exported\n * @param matlibname defines the name of the associated mtl file\n * @param globalposition defines if the exported positions are globals or local to the exported mesh\n * @returns the OBJ content\n */\n static OBJ(mesh, materials, matlibname, globalposition) {\n const output = [];\n let v = 1;\n // keep track of uv index in case mixed meshes are passed in\n let textureV = 1;\n if (materials) {\n if (!matlibname) {\n matlibname = \"mat\";\n }\n output.push(\"mtllib \" + matlibname + \".mtl\");\n }\n for (let j = 0; j < mesh.length; j++) {\n output.push(\"g object\" + j);\n output.push(\"o object_\" + j);\n //Uses the position of the item in the scene, to the file (this back to normal in the end)\n let inverseTransform = null;\n if (globalposition) {\n const transform = mesh[j].computeWorldMatrix(true);\n inverseTransform = new Matrix();\n transform.invertToRef(inverseTransform);\n mesh[j].bakeTransformIntoVertices(transform);\n }\n //TODO: submeshes (groups)\n //TODO: smoothing groups (s 1, s off);\n if (materials) {\n const mat = mesh[j].material;\n if (mat) {\n output.push(\"usemtl \" + mat.id);\n }\n }\n const g = mesh[j].geometry;\n if (!g) {\n Tools.Warn(\"No geometry is present on the mesh\");\n continue;\n }\n const trunkVerts = g.getVerticesData(\"position\");\n const trunkNormals = g.getVerticesData(\"normal\");\n const trunkUV = g.getVerticesData(\"uv\");\n const trunkFaces = g.getIndices();\n let currentV = 0;\n let currentTextureV = 0;\n if (!trunkVerts || !trunkFaces) {\n Tools.Warn(\"There are no position vertices or indices on the mesh!\");\n continue;\n }\n for (let i = 0; i < trunkVerts.length; i += 3) {\n // Babylon.js default is left handed, while OBJ default is right handed\n // Need to invert Z vertices unless Babylon is set to use a right handed system\n if (mesh[0].getScene().useRightHandedSystem) {\n output.push(\"v \" + trunkVerts[i] + \" \" + trunkVerts[i + 1] + \" \" + trunkVerts[i + 2]);\n }\n else {\n output.push(\"v \" + trunkVerts[i] + \" \" + trunkVerts[i + 1] + \" \" + -trunkVerts[i + 2]);\n }\n currentV++;\n }\n if (trunkNormals != null) {\n for (let i = 0; i < trunkNormals.length; i += 3) {\n output.push(\"vn \" + trunkNormals[i] + \" \" + trunkNormals[i + 1] + \" \" + trunkNormals[i + 2]);\n }\n }\n if (trunkUV != null) {\n for (let i = 0; i < trunkUV.length; i += 2) {\n output.push(\"vt \" + trunkUV[i] + \" \" + trunkUV[i + 1]);\n currentTextureV++;\n }\n }\n for (let i = 0; i < trunkFaces.length; i += 3) {\n const indices = [String(trunkFaces[i + 2] + v), String(trunkFaces[i + 1] + v), String(trunkFaces[i] + v)];\n const textureIndices = [String(trunkFaces[i + 2] + textureV), String(trunkFaces[i + 1] + textureV), String(trunkFaces[i] + textureV)];\n const blanks = [\"\", \"\", \"\"];\n const facePositions = indices;\n const faceUVs = trunkUV != null ? textureIndices : blanks;\n const faceNormals = trunkNormals != null ? indices : blanks;\n output.push(\"f \" +\n facePositions[0] +\n \"/\" +\n faceUVs[0] +\n \"/\" +\n faceNormals[0] +\n \" \" +\n facePositions[1] +\n \"/\" +\n faceUVs[1] +\n \"/\" +\n faceNormals[1] +\n \" \" +\n facePositions[2] +\n \"/\" +\n faceUVs[2] +\n \"/\" +\n faceNormals[2]);\n }\n //back de previous matrix, to not change the original mesh in the scene\n if (globalposition && inverseTransform) {\n mesh[j].bakeTransformIntoVertices(inverseTransform);\n }\n v += currentV;\n textureV += currentTextureV;\n }\n const text = output.join(\"\\n\");\n return text;\n }\n /**\n * Exports the material(s) of a mesh in .MTL file format (text)\n * @param mesh defines the mesh to extract the material from\n * @returns the mtl content\n */\n //TODO: Export the materials of mesh array\n static MTL(mesh) {\n const output = [];\n const m = mesh.material;\n output.push(\"newmtl mat1\");\n output.push(\" Ns \" + m.specularPower.toFixed(4));\n output.push(\" Ni 1.5000\");\n output.push(\" d \" + m.alpha.toFixed(4));\n output.push(\" Tr 0.0000\");\n output.push(\" Tf 1.0000 1.0000 1.0000\");\n output.push(\" illum 2\");\n output.push(\" Ka \" + m.ambientColor.r.toFixed(4) + \" \" + m.ambientColor.g.toFixed(4) + \" \" + m.ambientColor.b.toFixed(4));\n output.push(\" Kd \" + m.diffuseColor.r.toFixed(4) + \" \" + m.diffuseColor.g.toFixed(4) + \" \" + m.diffuseColor.b.toFixed(4));\n output.push(\" Ks \" + m.specularColor.r.toFixed(4) + \" \" + m.specularColor.g.toFixed(4) + \" \" + m.specularColor.b.toFixed(4));\n output.push(\" Ke \" + m.emissiveColor.r.toFixed(4) + \" \" + m.emissiveColor.g.toFixed(4) + \" \" + m.emissiveColor.b.toFixed(4));\n //TODO: uv scale, offset, wrap\n //TODO: UV mirrored in Blender? second UV channel? lightMap? reflection textures?\n const uvscale = \"\";\n if (m.ambientTexture) {\n output.push(\" map_Ka \" + uvscale + m.ambientTexture.name);\n }\n if (m.diffuseTexture) {\n output.push(\" map_Kd \" + uvscale + m.diffuseTexture.name);\n //TODO: alpha testing, opacity in diffuse texture alpha channel (diffuseTexture.hasAlpha -> map_d)\n }\n if (m.specularTexture) {\n output.push(\" map_Ks \" + uvscale + m.specularTexture.name);\n /* TODO: glossiness = specular highlight component is in alpha channel of specularTexture. (???)\n if (m.useGlossinessFromSpecularMapAlpha) {\n output.push(\" map_Ns \"+uvscale + m.specularTexture.name);\n }\n */\n }\n /* TODO: emissive texture not in .MAT format (???)\n if (m.emissiveTexture) {\n output.push(\" map_d \"+uvscale+m.emissiveTexture.name);\n }\n */\n if (m.bumpTexture) {\n output.push(\" map_bump -imfchan z \" + uvscale + m.bumpTexture.name);\n }\n if (m.opacityTexture) {\n output.push(\" map_d \" + uvscale + m.opacityTexture.name);\n }\n const text = output.join(\"\\n\");\n return text;\n }\n }\n\n class ObjKb3dExporter {\n constructor(scene, rendererManager) {\n this.scene = scene;\n this.rendererManager = rendererManager;\n }\n /** generates the current scene into a glb file and returns an object url for it */\n generateObj(meshNode) {\n let meshes;\n if (!meshNode) {\n meshes = this.scene.meshes.filter(this.shouldExportMesh);\n }\n else {\n const mesh = this.rendererManager.mesh.get3d(meshNode);\n if (mesh && mesh instanceof Mesh) {\n meshes = [mesh];\n }\n else {\n return null;\n }\n }\n return OBJExport.OBJ(meshes, false, undefined, meshes.length > 1);\n }\n exportObj(filename = 'scene', meshNode) {\n const objData = this.generateObj(meshNode);\n if (objData) {\n const blob = this.createObjBlob(objData);\n const link = document.createElement('a');\n document.body.appendChild(link);\n link.setAttribute('type', 'hidden');\n link.download = `${filename}.obj`;\n link.href = window.URL.createObjectURL(blob);\n link.click();\n setTimeout(() => {\n window.URL.revokeObjectURL(link.href);\n link.remove();\n });\n }\n }\n shouldExportMesh(mesh) {\n return mesh instanceof Mesh && mesh.isVisible && mesh.isEnabled() && !!getByDynamicId(mesh.id);\n }\n createObjBlob(objData) {\n const mimeType = { type: 'text/plain' };\n return new Blob([objData], mimeType);\n }\n }\n\n // Do not edit.\n const name$2D = \"boundingBoxRendererFragmentDeclaration\";\n const shader$2D = `uniform vec4 color;\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$2D] = shader$2D;\n\n // Do not edit.\n const name$2E = \"boundingBoxRendererUboDeclaration\";\n const shader$2E = `#ifdef WEBGL2\nuniform vec4 color;uniform mat4 world;uniform mat4 viewProjection;\n#ifdef MULTIVIEW\nuniform mat4 viewProjectionR;\n#endif\n#else\nlayout(std140,column_major) uniform;uniform BoundingBoxRenderer {vec4 color;mat4 world;mat4 viewProjection;mat4 viewProjectionR;};\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$2E] = shader$2E;\n\n // Do not edit.\n const name$2F = \"boundingBoxRendererPixelShader\";\n const shader$2F = `#include<__decl__boundingBoxRendererFragment>\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_FRAGMENT_MAIN_BEGIN\ngl_FragColor=color;\n#define CUSTOM_FRAGMENT_MAIN_END\n}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2F] = shader$2F;\n\n // Do not edit.\n const name$2G = \"boundingBoxRendererVertexDeclaration\";\n const shader$2G = `uniform mat4 world;uniform mat4 viewProjection;\n#ifdef MULTIVIEW\nuniform mat4 viewProjectionR;\n#endif\n`;\n // Sideeffect\n ShaderStore.IncludesShadersStore[name$2G] = shader$2G;\n\n // Do not edit.\n const name$2H = \"boundingBoxRendererVertexShader\";\n const shader$2H = `attribute vec3 position;\n#include<__decl__boundingBoxRendererVertex>\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\nvec4 worldPos=world*vec4(position,1.0);\n#ifdef MULTIVIEW\nif (gl_ViewID_OVR==0u) {gl_Position=viewProjection*worldPos;} else {gl_Position=viewProjectionR*worldPos;}\n#else\ngl_Position=viewProjection*worldPos;\n#endif\n#define CUSTOM_VERTEX_MAIN_END\n}\n`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2H] = shader$2H;\n\n Object.defineProperty(Scene.prototype, \"forceShowBoundingBoxes\", {\n get: function () {\n return this._forceShowBoundingBoxes || false;\n },\n set: function (value) {\n this._forceShowBoundingBoxes = value;\n // Lazyly creates a BB renderer if needed.\n if (value) {\n this.getBoundingBoxRenderer();\n }\n },\n enumerable: true,\n configurable: true,\n });\n Scene.prototype.getBoundingBoxRenderer = function () {\n if (!this._boundingBoxRenderer) {\n this._boundingBoxRenderer = new BoundingBoxRenderer(this);\n }\n return this._boundingBoxRenderer;\n };\n Object.defineProperty(AbstractMesh.prototype, \"showBoundingBox\", {\n get: function () {\n return this._showBoundingBox || false;\n },\n set: function (value) {\n this._showBoundingBox = value;\n // Lazyly creates a BB renderer if needed.\n if (value) {\n this.getScene().getBoundingBoxRenderer();\n }\n },\n enumerable: true,\n configurable: true,\n });\n /**\n * Component responsible of rendering the bounding box of the meshes in a scene.\n * This is usually used through the mesh.showBoundingBox or the scene.forceShowBoundingBoxes properties\n */\n class BoundingBoxRenderer {\n /**\n * Instantiates a new bounding box renderer in a scene.\n * @param scene the scene the renderer renders in\n */\n constructor(scene) {\n /**\n * The component name helpful to identify the component in the list of scene components.\n */\n this.name = SceneComponentConstants.NAME_BOUNDINGBOXRENDERER;\n /**\n * Color of the bounding box lines placed in front of an object\n */\n this.frontColor = new Color3(1, 1, 1);\n /**\n * Color of the bounding box lines placed behind an object\n */\n this.backColor = new Color3(0.1, 0.1, 0.1);\n /**\n * Defines if the renderer should show the back lines or not\n */\n this.showBackLines = true;\n /**\n * Observable raised before rendering a bounding box\n */\n this.onBeforeBoxRenderingObservable = new Observable$1();\n /**\n * Observable raised after rendering a bounding box\n */\n this.onAfterBoxRenderingObservable = new Observable$1();\n /**\n * Observable raised after resources are created\n */\n this.onResourcesReadyObservable = new Observable$1();\n /**\n * When false, no bounding boxes will be rendered\n */\n this.enabled = true;\n /**\n * @internal\n */\n this.renderList = new SmartArray(32);\n this._vertexBuffers = {};\n this._fillIndexBuffer = null;\n this._fillIndexData = null;\n this.scene = scene;\n scene._addComponent(this);\n this._uniformBufferFront = new UniformBuffer(this.scene.getEngine(), undefined, undefined, \"BoundingBoxRendererFront\", !this.scene.getEngine().isWebGPU);\n this._buildUniformLayout(this._uniformBufferFront);\n this._uniformBufferBack = new UniformBuffer(this.scene.getEngine(), undefined, undefined, \"BoundingBoxRendererBack\", !this.scene.getEngine().isWebGPU);\n this._buildUniformLayout(this._uniformBufferBack);\n }\n _buildUniformLayout(ubo) {\n ubo.addUniform(\"color\", 4);\n ubo.addUniform(\"world\", 16);\n ubo.addUniform(\"viewProjection\", 16);\n ubo.addUniform(\"viewProjectionR\", 16);\n ubo.create();\n }\n /**\n * Registers the component in a given scene\n */\n register() {\n this.scene._beforeEvaluateActiveMeshStage.registerStep(SceneComponentConstants.STEP_BEFOREEVALUATEACTIVEMESH_BOUNDINGBOXRENDERER, this, this.reset);\n this.scene._preActiveMeshStage.registerStep(SceneComponentConstants.STEP_PREACTIVEMESH_BOUNDINGBOXRENDERER, this, this._preActiveMesh);\n this.scene._evaluateSubMeshStage.registerStep(SceneComponentConstants.STEP_EVALUATESUBMESH_BOUNDINGBOXRENDERER, this, this._evaluateSubMesh);\n this.scene._afterRenderingGroupDrawStage.registerStep(SceneComponentConstants.STEP_AFTERRENDERINGGROUPDRAW_BOUNDINGBOXRENDERER, this, this.render);\n }\n _evaluateSubMesh(mesh, subMesh) {\n if (mesh.showSubMeshesBoundingBox) {\n const boundingInfo = subMesh.getBoundingInfo();\n if (boundingInfo !== null && boundingInfo !== undefined) {\n boundingInfo.boundingBox._tag = mesh.renderingGroupId;\n this.renderList.push(boundingInfo.boundingBox);\n }\n }\n }\n _preActiveMesh(mesh) {\n if (mesh.showBoundingBox || this.scene.forceShowBoundingBoxes) {\n const boundingInfo = mesh.getBoundingInfo();\n boundingInfo.boundingBox._tag = mesh.renderingGroupId;\n this.renderList.push(boundingInfo.boundingBox);\n }\n }\n _prepareResources() {\n if (this._colorShader) {\n return;\n }\n this._colorShader = new ShaderMaterial(\"colorShader\", this.scene, \"boundingBoxRenderer\", {\n attributes: [VertexBuffer.PositionKind],\n uniforms: [\"world\", \"viewProjection\", \"viewProjectionR\", \"color\"],\n uniformBuffers: [\"BoundingBoxRenderer\"],\n }, false);\n this._colorShader.doNotSerialize = true;\n this._colorShader.reservedDataStore = {\n hidden: true,\n };\n this._colorShaderForOcclusionQuery = new ShaderMaterial(\"colorShaderOccQuery\", this.scene, \"boundingBoxRenderer\", {\n attributes: [VertexBuffer.PositionKind],\n uniforms: [\"world\", \"viewProjection\", \"viewProjectionR\", \"color\"],\n uniformBuffers: [\"BoundingBoxRenderer\"],\n }, true);\n this._colorShaderForOcclusionQuery.doNotSerialize = true;\n this._colorShaderForOcclusionQuery.reservedDataStore = {\n hidden: true,\n };\n const engine = this.scene.getEngine();\n const boxdata = CreateBoxVertexData({ size: 1.0 });\n this._vertexBuffers[VertexBuffer.PositionKind] = new VertexBuffer(engine, boxdata.positions, VertexBuffer.PositionKind, false);\n this._createIndexBuffer();\n this._fillIndexData = boxdata.indices;\n this.onResourcesReadyObservable.notifyObservers(this);\n }\n _createIndexBuffer() {\n const engine = this.scene.getEngine();\n this._indexBuffer = engine.createIndexBuffer([0, 1, 1, 2, 2, 3, 3, 0, 4, 5, 5, 6, 6, 7, 7, 4, 0, 7, 1, 6, 2, 5, 3, 4]);\n }\n /**\n * Rebuilds the elements related to this component in case of\n * context lost for instance.\n */\n rebuild() {\n const vb = this._vertexBuffers[VertexBuffer.PositionKind];\n if (vb) {\n vb._rebuild();\n }\n this._createIndexBuffer();\n }\n /**\n * @internal\n */\n reset() {\n this.renderList.reset();\n }\n /**\n * Render the bounding boxes of a specific rendering group\n * @param renderingGroupId defines the rendering group to render\n */\n render(renderingGroupId) {\n var _a, _b;\n if (this.renderList.length === 0 || !this.enabled) {\n return;\n }\n this._prepareResources();\n if (!this._colorShader.isReady()) {\n return;\n }\n const engine = this.scene.getEngine();\n engine.setDepthWrite(false);\n const transformMatrix = this.scene.getTransformMatrix();\n for (let boundingBoxIndex = 0; boundingBoxIndex < this.renderList.length; boundingBoxIndex++) {\n const boundingBox = this.renderList.data[boundingBoxIndex];\n if (boundingBox._tag !== renderingGroupId) {\n continue;\n }\n this._createWrappersForBoundingBox(boundingBox);\n this.onBeforeBoxRenderingObservable.notifyObservers(boundingBox);\n const min = boundingBox.minimum;\n const max = boundingBox.maximum;\n const diff = max.subtract(min);\n const median = min.add(diff.scale(0.5));\n const worldMatrix = Matrix.Scaling(diff.x, diff.y, diff.z).multiply(Matrix.Translation(median.x, median.y, median.z)).multiply(boundingBox.getWorldMatrix());\n const useReverseDepthBuffer = engine.useReverseDepthBuffer;\n if (this.showBackLines) {\n const drawWrapperBack = (_a = boundingBox._drawWrapperBack) !== null && _a !== void 0 ? _a : this._colorShader._getDrawWrapper();\n this._colorShader._preBind(drawWrapperBack);\n engine.bindBuffers(this._vertexBuffers, this._indexBuffer, this._colorShader.getEffect());\n // Back\n if (useReverseDepthBuffer) {\n engine.setDepthFunctionToLessOrEqual();\n }\n else {\n engine.setDepthFunctionToGreaterOrEqual();\n }\n this._uniformBufferBack.bindToEffect(drawWrapperBack.effect, \"BoundingBoxRenderer\");\n this._uniformBufferBack.updateColor4(\"color\", this.backColor, 1);\n this._uniformBufferBack.updateMatrix(\"world\", worldMatrix);\n this._uniformBufferBack.updateMatrix(\"viewProjection\", transformMatrix);\n this._uniformBufferBack.update();\n // Draw order\n engine.drawElementsType(Material.LineListDrawMode, 0, 24);\n }\n const drawWrapperFront = (_b = boundingBox._drawWrapperFront) !== null && _b !== void 0 ? _b : this._colorShader._getDrawWrapper();\n this._colorShader._preBind(drawWrapperFront);\n engine.bindBuffers(this._vertexBuffers, this._indexBuffer, this._colorShader.getEffect());\n // Front\n if (useReverseDepthBuffer) {\n engine.setDepthFunctionToGreater();\n }\n else {\n engine.setDepthFunctionToLess();\n }\n this._uniformBufferFront.bindToEffect(drawWrapperFront.effect, \"BoundingBoxRenderer\");\n this._uniformBufferFront.updateColor4(\"color\", this.frontColor, 1);\n this._uniformBufferFront.updateMatrix(\"world\", worldMatrix);\n this._uniformBufferFront.updateMatrix(\"viewProjection\", transformMatrix);\n this._uniformBufferFront.update();\n // Draw order\n engine.drawElementsType(Material.LineListDrawMode, 0, 24);\n this.onAfterBoxRenderingObservable.notifyObservers(boundingBox);\n }\n this._colorShader.unbind();\n engine.setDepthFunctionToLessOrEqual();\n engine.setDepthWrite(true);\n }\n _createWrappersForBoundingBox(boundingBox) {\n if (!boundingBox._drawWrapperFront) {\n const engine = this.scene.getEngine();\n boundingBox._drawWrapperFront = new DrawWrapper(engine);\n boundingBox._drawWrapperBack = new DrawWrapper(engine);\n boundingBox._drawWrapperFront.setEffect(this._colorShader.getEffect());\n boundingBox._drawWrapperBack.setEffect(this._colorShader.getEffect());\n }\n }\n /**\n * In case of occlusion queries, we can render the occlusion bounding box through this method\n * @param mesh Define the mesh to render the occlusion bounding box for\n */\n renderOcclusionBoundingBox(mesh) {\n const engine = this.scene.getEngine();\n if (this._renderPassIdForOcclusionQuery === undefined) {\n this._renderPassIdForOcclusionQuery = engine.createRenderPassId(`Render pass for occlusion query`);\n }\n const currentRenderPassId = engine.currentRenderPassId;\n engine.currentRenderPassId = this._renderPassIdForOcclusionQuery;\n this._prepareResources();\n const subMesh = mesh.subMeshes[0];\n if (!this._colorShaderForOcclusionQuery.isReady(mesh, undefined, subMesh) || !mesh.hasBoundingInfo) {\n engine.currentRenderPassId = currentRenderPassId;\n return;\n }\n if (!this._fillIndexBuffer) {\n this._fillIndexBuffer = engine.createIndexBuffer(this._fillIndexData);\n }\n const useReverseDepthBuffer = engine.useReverseDepthBuffer;\n engine.setDepthWrite(false);\n engine.setColorWrite(false);\n const boundingBox = mesh.getBoundingInfo().boundingBox;\n const min = boundingBox.minimum;\n const max = boundingBox.maximum;\n const diff = max.subtract(min);\n const median = min.add(diff.scale(0.5));\n const worldMatrix = Matrix.Scaling(diff.x, diff.y, diff.z).multiply(Matrix.Translation(median.x, median.y, median.z)).multiply(boundingBox.getWorldMatrix());\n const drawWrapper = subMesh._drawWrapper;\n this._colorShaderForOcclusionQuery._preBind(drawWrapper);\n engine.bindBuffers(this._vertexBuffers, this._fillIndexBuffer, drawWrapper.effect);\n if (useReverseDepthBuffer) {\n engine.setDepthFunctionToGreater();\n }\n else {\n engine.setDepthFunctionToLess();\n }\n this.scene.resetCachedMaterial();\n this._uniformBufferFront.bindToEffect(drawWrapper.effect, \"BoundingBoxRenderer\");\n this._uniformBufferFront.updateMatrix(\"world\", worldMatrix);\n this._uniformBufferFront.updateMatrix(\"viewProjection\", this.scene.getTransformMatrix());\n this._uniformBufferFront.update();\n engine.drawElementsType(Material.TriangleFillMode, 0, 36);\n this._colorShaderForOcclusionQuery.unbind();\n engine.setDepthFunctionToLessOrEqual();\n engine.setDepthWrite(true);\n engine.setColorWrite(true);\n engine.currentRenderPassId = currentRenderPassId;\n }\n /**\n * Dispose and release the resources attached to this renderer.\n */\n dispose() {\n if (this._renderPassIdForOcclusionQuery !== undefined) {\n this.scene.getEngine().releaseRenderPassId(this._renderPassIdForOcclusionQuery);\n this._renderPassIdForOcclusionQuery = undefined;\n }\n if (!this._colorShader) {\n return;\n }\n this.onBeforeBoxRenderingObservable.clear();\n this.onAfterBoxRenderingObservable.clear();\n this.onResourcesReadyObservable.clear();\n this.renderList.dispose();\n this._colorShader.dispose();\n this._colorShaderForOcclusionQuery.dispose();\n this._uniformBufferFront.dispose();\n this._uniformBufferBack.dispose();\n const buffer = this._vertexBuffers[VertexBuffer.PositionKind];\n if (buffer) {\n buffer.dispose();\n this._vertexBuffers[VertexBuffer.PositionKind] = null;\n }\n this.scene.getEngine()._releaseBuffer(this._indexBuffer);\n if (this._fillIndexBuffer) {\n this.scene.getEngine()._releaseBuffer(this._fillIndexBuffer);\n this._fillIndexBuffer = null;\n }\n }\n }\n\n /**\n * Class containing a set of static utilities functions for managing Pivots\n * @internal\n */\n class PivotTools {\n /**\n * @internal\n */\n static _RemoveAndStorePivotPoint(mesh) {\n if (mesh && PivotTools._PivotCached === 0) {\n // Save old pivot and set pivot to 0,0,0\n mesh.getPivotPointToRef(PivotTools._OldPivotPoint);\n PivotTools._PivotPostMultiplyPivotMatrix = mesh._postMultiplyPivotMatrix;\n if (!PivotTools._OldPivotPoint.equalsToFloats(0, 0, 0)) {\n mesh.setPivotMatrix(Matrix.IdentityReadOnly);\n PivotTools._OldPivotPoint.subtractToRef(mesh.getPivotPoint(), PivotTools._PivotTranslation);\n PivotTools._PivotTmpVector.copyFromFloats(1, 1, 1);\n PivotTools._PivotTmpVector.subtractInPlace(mesh.scaling);\n PivotTools._PivotTmpVector.multiplyInPlace(PivotTools._PivotTranslation);\n mesh.position.addInPlace(PivotTools._PivotTmpVector);\n }\n }\n PivotTools._PivotCached++;\n }\n /**\n * @internal\n */\n static _RestorePivotPoint(mesh) {\n if (mesh && !PivotTools._OldPivotPoint.equalsToFloats(0, 0, 0) && PivotTools._PivotCached === 1) {\n mesh.setPivotPoint(PivotTools._OldPivotPoint);\n mesh._postMultiplyPivotMatrix = PivotTools._PivotPostMultiplyPivotMatrix;\n PivotTools._PivotTmpVector.copyFromFloats(1, 1, 1);\n PivotTools._PivotTmpVector.subtractInPlace(mesh.scaling);\n PivotTools._PivotTmpVector.multiplyInPlace(PivotTools._PivotTranslation);\n mesh.position.subtractInPlace(PivotTools._PivotTmpVector);\n }\n this._PivotCached--;\n }\n }\n // Stores the state of the pivot cache (_oldPivotPoint, _pivotTranslation)\n // store/remove pivot point should only be applied during their outermost calls\n PivotTools._PivotCached = 0;\n PivotTools._OldPivotPoint = new Vector3();\n PivotTools._PivotTranslation = new Vector3();\n PivotTools._PivotTmpVector = new Vector3();\n PivotTools._PivotPostMultiplyPivotMatrix = false;\n\n /**\n * A behavior that when attached to a mesh will allow the mesh to be dragged around the screen based on pointer events\n */\n class PointerDragBehavior {\n /**\n * Get or set the currentDraggingPointerId\n * @deprecated Please use currentDraggingPointerId instead\n */\n get currentDraggingPointerID() {\n return this.currentDraggingPointerId;\n }\n set currentDraggingPointerID(currentDraggingPointerID) {\n this.currentDraggingPointerId = currentDraggingPointerID;\n }\n /**\n * If the drag behavior will react to drag events (Default: true)\n */\n set enabled(value) {\n if (value != this._enabled) {\n this.onEnabledObservable.notifyObservers(value);\n }\n this._enabled = value;\n }\n get enabled() {\n return this._enabled;\n }\n /**\n * Gets the options used by the behavior\n */\n get options() {\n return this._options;\n }\n /**\n * Sets the options used by the behavior\n */\n set options(options) {\n this._options = options;\n }\n /**\n * Creates a pointer drag behavior that can be attached to a mesh\n * @param options The drag axis or normal of the plane that will be dragged across. If no options are specified the drag plane will always face the ray's origin (eg. camera)\n * @param options.dragAxis\n * @param options.dragPlaneNormal\n */\n constructor(options) {\n this._useAlternatePickedPointAboveMaxDragAngleDragSpeed = -1.1;\n this._activeDragButton = -1;\n /**\n * The maximum tolerated angle between the drag plane and dragging pointer rays to trigger pointer events. Set to 0 to allow any angle (default: 0)\n */\n this.maxDragAngle = 0;\n /**\n * Butttons that can be used to initiate a drag\n */\n this.dragButtons = [0, 1, 2];\n /**\n * @internal\n */\n this._useAlternatePickedPointAboveMaxDragAngle = false;\n /**\n * The id of the pointer that is currently interacting with the behavior (-1 when no pointer is active)\n */\n this.currentDraggingPointerId = -1;\n /**\n * If the behavior is currently in a dragging state\n */\n this.dragging = false;\n /**\n * The distance towards the target drag position to move each frame. This can be useful to avoid jitter. Set this to 1 for no delay. (Default: 0.2)\n */\n this.dragDeltaRatio = 0.2;\n /**\n * If the drag plane orientation should be updated during the dragging (Default: true)\n */\n this.updateDragPlane = true;\n // Debug mode will display drag planes to help visualize behavior\n this._debugMode = false;\n this._moving = false;\n /**\n * Fires each time the attached mesh is dragged with the pointer\n * * delta between last drag position and current drag position in world space\n * * dragDistance along the drag axis\n * * dragPlaneNormal normal of the current drag plane used during the drag\n * * dragPlanePoint in world space where the drag intersects the drag plane\n *\n * (if validatedDrag is used, the position of the attached mesh might not equal dragPlanePoint)\n */\n this.onDragObservable = new Observable$1();\n /**\n * Fires each time a drag begins (eg. mouse down on mesh)\n * * dragPlanePoint in world space where the drag intersects the drag plane\n *\n * (if validatedDrag is used, the position of the attached mesh might not equal dragPlanePoint)\n */\n this.onDragStartObservable = new Observable$1();\n /**\n * Fires each time a drag ends (eg. mouse release after drag)\n * * dragPlanePoint in world space where the drag intersects the drag plane\n *\n * (if validatedDrag is used, the position of the attached mesh might not equal dragPlanePoint)\n */\n this.onDragEndObservable = new Observable$1();\n /**\n * Fires each time behavior enabled state changes\n */\n this.onEnabledObservable = new Observable$1();\n /**\n * If the attached mesh should be moved when dragged\n */\n this.moveAttached = true;\n this._enabled = true;\n /**\n * If pointer events should start and release the drag (Default: true)\n */\n this.startAndReleaseDragOnPointerEvents = true;\n /**\n * If camera controls should be detached during the drag\n */\n this.detachCameraControls = true;\n /**\n * If set, the drag plane/axis will be rotated based on the attached mesh's world rotation (Default: true)\n */\n this.useObjectOrientationForDragging = true;\n /**\n * Predicate to determine if it is valid to move the object to a new position when it is moved\n * @param targetPosition\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n this.validateDrag = (targetPosition) => {\n return true;\n };\n this._tmpVector = new Vector3(0, 0, 0);\n this._alternatePickedPoint = new Vector3(0, 0, 0);\n this._worldDragAxis = new Vector3(0, 0, 0);\n this._targetPosition = new Vector3(0, 0, 0);\n this._attachedToElement = false;\n this._startDragRay = new Ray(new Vector3(), new Vector3());\n this._lastPointerRay = {};\n this._dragDelta = new Vector3();\n // Variables to avoid instantiation in the below method\n this._pointA = new Vector3(0, 0, 0);\n this._pointC = new Vector3(0, 0, 0);\n this._localAxis = new Vector3(0, 0, 0);\n this._lookAt = new Vector3(0, 0, 0);\n this._options = options ? options : {};\n let optionCount = 0;\n if (this._options.dragAxis) {\n optionCount++;\n }\n if (this._options.dragPlaneNormal) {\n optionCount++;\n }\n if (optionCount > 1) {\n throw \"Multiple drag modes specified in dragBehavior options. Only one expected\";\n }\n }\n /**\n * The name of the behavior\n */\n get name() {\n return \"PointerDrag\";\n }\n /**\n * Initializes the behavior\n */\n init() { }\n /**\n * Attaches the drag behavior the passed in mesh\n * @param ownerNode The mesh that will be dragged around once attached\n * @param predicate Predicate to use for pick filtering\n */\n attach(ownerNode, predicate) {\n this._scene = ownerNode.getScene();\n ownerNode.isNearGrabbable = true;\n this.attachedNode = ownerNode;\n // Initialize drag plane to not interfere with existing scene\n if (!PointerDragBehavior._PlaneScene) {\n if (this._debugMode) {\n PointerDragBehavior._PlaneScene = this._scene;\n }\n else {\n PointerDragBehavior._PlaneScene = new Scene(this._scene.getEngine(), { virtual: true });\n PointerDragBehavior._PlaneScene.detachControl();\n this._scene.onDisposeObservable.addOnce(() => {\n PointerDragBehavior._PlaneScene.dispose();\n PointerDragBehavior._PlaneScene = null;\n });\n }\n }\n this._dragPlane = CreatePlane(\"pointerDragPlane\", { size: this._debugMode ? 1 : 10000, updatable: false, sideOrientation: Mesh.DOUBLESIDE }, PointerDragBehavior._PlaneScene);\n // State of the drag\n this.lastDragPosition = new Vector3(0, 0, 0);\n const pickPredicate = predicate\n ? predicate\n : (m) => {\n return this.attachedNode == m || m.isDescendantOf(this.attachedNode);\n };\n this._pointerObserver = this._scene.onPointerObservable.add((pointerInfo) => {\n if (!this.enabled) {\n // If behavior is disabled before releaseDrag is ever called, call it now.\n if (this._attachedToElement) {\n this.releaseDrag();\n }\n return;\n }\n if (pointerInfo.type == PointerEventTypes.POINTERDOWN) {\n if (this.startAndReleaseDragOnPointerEvents &&\n !this.dragging &&\n pointerInfo.pickInfo &&\n pointerInfo.pickInfo.hit &&\n pointerInfo.pickInfo.pickedMesh &&\n pointerInfo.pickInfo.pickedPoint &&\n pointerInfo.pickInfo.ray &&\n pickPredicate(pointerInfo.pickInfo.pickedMesh)) {\n if (this._activeDragButton === -1 && this.dragButtons.indexOf(pointerInfo.event.button) !== -1) {\n this._activeDragButton = pointerInfo.event.button;\n this._activePointerInfo = pointerInfo;\n this._startDrag(pointerInfo.event.pointerId, pointerInfo.pickInfo.ray, pointerInfo.pickInfo.pickedPoint);\n }\n }\n }\n else if (pointerInfo.type == PointerEventTypes.POINTERUP) {\n if (this.startAndReleaseDragOnPointerEvents &&\n this.currentDraggingPointerId == pointerInfo.event.pointerId &&\n (this._activeDragButton === pointerInfo.event.button || this._activeDragButton === -1)) {\n this.releaseDrag();\n }\n }\n else if (pointerInfo.type == PointerEventTypes.POINTERMOVE) {\n const pointerId = pointerInfo.event.pointerId;\n // If drag was started with anyMouseID specified, set pointerID to the next mouse that moved\n if (this.currentDraggingPointerId === PointerDragBehavior._AnyMouseId && pointerId !== PointerDragBehavior._AnyMouseId) {\n const evt = pointerInfo.event;\n const isMouseEvent = evt.pointerType === \"mouse\" || (!this._scene.getEngine().hostInformation.isMobile && evt instanceof MouseEvent);\n if (isMouseEvent) {\n if (this._lastPointerRay[this.currentDraggingPointerId]) {\n this._lastPointerRay[pointerId] = this._lastPointerRay[this.currentDraggingPointerId];\n delete this._lastPointerRay[this.currentDraggingPointerId];\n }\n this.currentDraggingPointerId = pointerId;\n }\n }\n // Keep track of last pointer ray, this is used simulating the start of a drag in startDrag()\n if (!this._lastPointerRay[pointerId]) {\n this._lastPointerRay[pointerId] = new Ray(new Vector3(), new Vector3());\n }\n if (pointerInfo.pickInfo && pointerInfo.pickInfo.ray) {\n this._lastPointerRay[pointerId].origin.copyFrom(pointerInfo.pickInfo.ray.origin);\n this._lastPointerRay[pointerId].direction.copyFrom(pointerInfo.pickInfo.ray.direction);\n if (this.currentDraggingPointerId == pointerId && this.dragging) {\n this._moveDrag(pointerInfo.pickInfo.ray);\n }\n }\n }\n });\n this._beforeRenderObserver = this._scene.onBeforeRenderObservable.add(() => {\n if (this._moving && this.moveAttached) {\n let needMatrixUpdate = false;\n PivotTools._RemoveAndStorePivotPoint(this.attachedNode);\n // Slowly move mesh to avoid jitter\n this._targetPosition.subtractToRef(this.attachedNode.absolutePosition, this._tmpVector);\n this._tmpVector.scaleInPlace(this.dragDeltaRatio);\n this.attachedNode.getAbsolutePosition().addToRef(this._tmpVector, this._tmpVector);\n if (this.validateDrag(this._tmpVector)) {\n this.attachedNode.setAbsolutePosition(this._tmpVector);\n needMatrixUpdate = true;\n }\n PivotTools._RestorePivotPoint(this.attachedNode);\n if (needMatrixUpdate) {\n this.attachedNode.computeWorldMatrix();\n }\n }\n });\n }\n /**\n * Force release the drag action by code.\n */\n releaseDrag() {\n if (this.dragging) {\n this.dragging = false;\n this.onDragEndObservable.notifyObservers({ dragPlanePoint: this.lastDragPosition, pointerId: this.currentDraggingPointerId, pointerInfo: this._activePointerInfo });\n }\n this.currentDraggingPointerId = -1;\n this._activeDragButton = -1;\n this._activePointerInfo = null;\n this._moving = false;\n // Reattach camera controls\n if (this.detachCameraControls && this._attachedToElement && this._scene.activeCamera && !this._scene.activeCamera.leftCamera) {\n if (this._scene.activeCamera.getClassName() === \"ArcRotateCamera\") {\n const arcRotateCamera = this._scene.activeCamera;\n arcRotateCamera.attachControl(arcRotateCamera.inputs ? arcRotateCamera.inputs.noPreventDefault : true, arcRotateCamera._useCtrlForPanning, arcRotateCamera._panningMouseButton);\n }\n else {\n this._scene.activeCamera.attachControl(this._scene.activeCamera.inputs ? this._scene.activeCamera.inputs.noPreventDefault : true);\n }\n this._attachedToElement = false;\n }\n }\n /**\n * Simulates the start of a pointer drag event on the behavior\n * @param pointerId pointerID of the pointer that should be simulated (Default: Any mouse pointer ID)\n * @param fromRay initial ray of the pointer to be simulated (Default: Ray from camera to attached mesh)\n * @param startPickedPoint picked point of the pointer to be simulated (Default: attached mesh position)\n */\n startDrag(pointerId = PointerDragBehavior._AnyMouseId, fromRay, startPickedPoint) {\n this._startDrag(pointerId, fromRay, startPickedPoint);\n let lastRay = this._lastPointerRay[pointerId];\n if (pointerId === PointerDragBehavior._AnyMouseId) {\n lastRay = this._lastPointerRay[Object.keys(this._lastPointerRay)[0]];\n }\n if (lastRay) {\n // if there was a last pointer ray drag the object there\n this._moveDrag(lastRay);\n }\n }\n _startDrag(pointerId, fromRay, startPickedPoint) {\n if (!this._scene.activeCamera || this.dragging || !this.attachedNode) {\n return;\n }\n PivotTools._RemoveAndStorePivotPoint(this.attachedNode);\n // Create start ray from the camera to the object\n if (fromRay) {\n this._startDragRay.direction.copyFrom(fromRay.direction);\n this._startDragRay.origin.copyFrom(fromRay.origin);\n }\n else {\n this._startDragRay.origin.copyFrom(this._scene.activeCamera.position);\n this.attachedNode.getWorldMatrix().getTranslationToRef(this._tmpVector);\n this._tmpVector.subtractToRef(this._scene.activeCamera.position, this._startDragRay.direction);\n }\n this._updateDragPlanePosition(this._startDragRay, startPickedPoint ? startPickedPoint : this._tmpVector);\n const pickedPoint = this._pickWithRayOnDragPlane(this._startDragRay);\n if (pickedPoint) {\n this.dragging = true;\n this.currentDraggingPointerId = pointerId;\n this.lastDragPosition.copyFrom(pickedPoint);\n this.onDragStartObservable.notifyObservers({ dragPlanePoint: pickedPoint, pointerId: this.currentDraggingPointerId, pointerInfo: this._activePointerInfo });\n this._targetPosition.copyFrom(this.attachedNode.getAbsolutePosition());\n // Detatch camera controls\n if (this.detachCameraControls && this._scene.activeCamera && this._scene.activeCamera.inputs && !this._scene.activeCamera.leftCamera) {\n if (this._scene.activeCamera.inputs.attachedToElement) {\n this._scene.activeCamera.detachControl();\n this._attachedToElement = true;\n }\n else {\n this._attachedToElement = false;\n }\n }\n }\n else {\n this.releaseDrag();\n }\n PivotTools._RestorePivotPoint(this.attachedNode);\n }\n _moveDrag(ray) {\n this._moving = true;\n const pickedPoint = this._pickWithRayOnDragPlane(ray);\n if (pickedPoint) {\n PivotTools._RemoveAndStorePivotPoint(this.attachedNode);\n if (this.updateDragPlane) {\n this._updateDragPlanePosition(ray, pickedPoint);\n }\n let dragLength = 0;\n // depending on the drag mode option drag accordingly\n if (this._options.dragAxis) {\n // Convert local drag axis to world if useObjectOrientationForDragging\n this.useObjectOrientationForDragging\n ? Vector3.TransformCoordinatesToRef(this._options.dragAxis, this.attachedNode.getWorldMatrix().getRotationMatrix(), this._worldDragAxis)\n : this._worldDragAxis.copyFrom(this._options.dragAxis);\n // Project delta drag from the drag plane onto the drag axis\n pickedPoint.subtractToRef(this.lastDragPosition, this._tmpVector);\n dragLength = Vector3.Dot(this._tmpVector, this._worldDragAxis);\n this._worldDragAxis.scaleToRef(dragLength, this._dragDelta);\n }\n else {\n dragLength = this._dragDelta.length();\n pickedPoint.subtractToRef(this.lastDragPosition, this._dragDelta);\n }\n this._targetPosition.addInPlace(this._dragDelta);\n this.onDragObservable.notifyObservers({\n dragDistance: dragLength,\n delta: this._dragDelta,\n dragPlanePoint: pickedPoint,\n dragPlaneNormal: this._dragPlane.forward,\n pointerId: this.currentDraggingPointerId,\n pointerInfo: this._activePointerInfo,\n });\n this.lastDragPosition.copyFrom(pickedPoint);\n PivotTools._RestorePivotPoint(this.attachedNode);\n }\n }\n _pickWithRayOnDragPlane(ray) {\n if (!ray) {\n return null;\n }\n // Calculate angle between plane normal and ray\n let angle = Math.acos(Vector3.Dot(this._dragPlane.forward, ray.direction));\n // Correct if ray is casted from oposite side\n if (angle > Math.PI / 2) {\n angle = Math.PI - angle;\n }\n // If the angle is too perpendicular to the plane pick another point on the plane where it is looking\n if (this.maxDragAngle > 0 && angle > this.maxDragAngle) {\n if (this._useAlternatePickedPointAboveMaxDragAngle) {\n // Invert ray direction along the towards object axis\n this._tmpVector.copyFrom(ray.direction);\n this.attachedNode.absolutePosition.subtractToRef(ray.origin, this._alternatePickedPoint);\n this._alternatePickedPoint.normalize();\n this._alternatePickedPoint.scaleInPlace(this._useAlternatePickedPointAboveMaxDragAngleDragSpeed * Vector3.Dot(this._alternatePickedPoint, this._tmpVector));\n this._tmpVector.addInPlace(this._alternatePickedPoint);\n // Project resulting vector onto the drag plane and add it to the attached nodes absolute position to get a picked point\n const dot = Vector3.Dot(this._dragPlane.forward, this._tmpVector);\n this._dragPlane.forward.scaleToRef(-dot, this._alternatePickedPoint);\n this._alternatePickedPoint.addInPlace(this._tmpVector);\n this._alternatePickedPoint.addInPlace(this.attachedNode.absolutePosition);\n return this._alternatePickedPoint;\n }\n else {\n return null;\n }\n }\n const pickResult = PointerDragBehavior._PlaneScene.pickWithRay(ray, (m) => {\n return m == this._dragPlane;\n });\n if (pickResult && pickResult.hit && pickResult.pickedMesh && pickResult.pickedPoint) {\n return pickResult.pickedPoint;\n }\n else {\n return null;\n }\n }\n // Position the drag plane based on the attached mesh position, for single axis rotate the plane along the axis to face the camera\n _updateDragPlanePosition(ray, dragPlanePosition) {\n this._pointA.copyFrom(dragPlanePosition);\n if (this._options.dragAxis) {\n this.useObjectOrientationForDragging\n ? Vector3.TransformCoordinatesToRef(this._options.dragAxis, this.attachedNode.getWorldMatrix().getRotationMatrix(), this._localAxis)\n : this._localAxis.copyFrom(this._options.dragAxis);\n // Calculate plane normal that is the cross product of local axis and (eye-dragPlanePosition)\n ray.origin.subtractToRef(this._pointA, this._pointC);\n this._pointC.normalize();\n if (Math.abs(Vector3.Dot(this._localAxis, this._pointC)) > 0.999) {\n // the drag axis is colinear with the (eye to position) ray. The cross product will give jittered values.\n // A new axis vector need to be computed\n if (Math.abs(Vector3.Dot(Vector3.UpReadOnly, this._pointC)) > 0.999) {\n this._lookAt.copyFrom(Vector3.Right());\n }\n else {\n this._lookAt.copyFrom(Vector3.UpReadOnly);\n }\n }\n else {\n Vector3.CrossToRef(this._localAxis, this._pointC, this._lookAt);\n // Get perpendicular line from previous result and drag axis to adjust lineB to be perpendicular to camera\n Vector3.CrossToRef(this._localAxis, this._lookAt, this._lookAt);\n this._lookAt.normalize();\n }\n this._dragPlane.position.copyFrom(this._pointA);\n this._pointA.addToRef(this._lookAt, this._lookAt);\n this._dragPlane.lookAt(this._lookAt);\n }\n else if (this._options.dragPlaneNormal) {\n this.useObjectOrientationForDragging\n ? Vector3.TransformCoordinatesToRef(this._options.dragPlaneNormal, this.attachedNode.getWorldMatrix().getRotationMatrix(), this._localAxis)\n : this._localAxis.copyFrom(this._options.dragPlaneNormal);\n this._dragPlane.position.copyFrom(this._pointA);\n this._pointA.addToRef(this._localAxis, this._lookAt);\n this._dragPlane.lookAt(this._lookAt);\n }\n else {\n this._dragPlane.position.copyFrom(this._pointA);\n this._dragPlane.lookAt(ray.origin);\n }\n // Update the position of the drag plane so it doesn't get out of sync with the node (eg. when moving back and forth quickly)\n this._dragPlane.position.copyFrom(this.attachedNode.getAbsolutePosition());\n this._dragPlane.computeWorldMatrix(true);\n }\n /**\n * Detaches the behavior from the mesh\n */\n detach() {\n this._lastPointerRay = {};\n if (this.attachedNode) {\n this.attachedNode.isNearGrabbable = false;\n }\n if (this._pointerObserver) {\n this._scene.onPointerObservable.remove(this._pointerObserver);\n }\n if (this._beforeRenderObserver) {\n this._scene.onBeforeRenderObservable.remove(this._beforeRenderObserver);\n }\n if (this._dragPlane) {\n this._dragPlane.dispose();\n }\n this.releaseDrag();\n }\n }\n PointerDragBehavior._AnyMouseId = -2;\n\n /**\n * Single plane rotation gizmo\n */\n class KbAxisMorphGizmo {\n /**\n * Ratio for the scale of the gizmo (Default: 1)\n */\n set scaleRatio(value) {\n this._scaleRatio = value;\n }\n get scaleRatio() {\n return this._scaleRatio;\n }\n set gizmoPosition(value) {\n this._rootMesh.position = value;\n }\n get gizmoPosition() {\n return this._rootMesh.position;\n }\n /**\n * Creates a PlaneRotationGizmo\n * @param gizmoLayer The utility layer the gizmo will be added to\n * @param color The color of the gizmo\n * @param tessellation Amount of tessellation to be used when creating rotation circles\n */\n constructor(color = Color3.Gray(), gizmoLayer = UtilityLayerRenderer.DefaultUtilityLayer, tessellation = 24) {\n this.gizmoLayer = gizmoLayer;\n this.dragActive = false;\n this.direction = new Vector3();\n this._pointerObserver = null;\n this._tempVector = new Vector3();\n this._updateQuat = new Quaternion();\n this._scaleRatio = 1;\n /**\n * Rotation distance in radians that the gizmo will snap to (Default: 0)\n */\n this.snapDistance = 0;\n /**\n * Event that fires each time the gizmo snaps to a new location.\n * * snapDistance is the the change in distance\n */\n this.updateObservable = new Observable$1();\n this._rootMesh = new Mesh('gizmoRootNode', gizmoLayer.utilityLayerScene);\n // Create Material\n const coloredMaterial = new StandardMaterial('', gizmoLayer.utilityLayerScene);\n coloredMaterial.emissiveColor = color;\n const wireframeMaterial = new StandardMaterial('', gizmoLayer.utilityLayerScene);\n wireframeMaterial.emissiveColor = color;\n wireframeMaterial.alpha = 0.4;\n // coloredMaterial.diffuseColor = color;\n // coloredMaterial.specularColor = color.subtract(new Color3(0.1, 0.1, 0.1));\n const hoverMaterial = new StandardMaterial('', gizmoLayer.utilityLayerScene);\n hoverMaterial.emissiveColor = color.add(new Color3(0.3, 0.3, 0.3));\n wireframeMaterial.alpha = 0.6;\n // hoverMaterial.diffuseColor = color.add(new Color3(0.3, 0.3, 0.3));\n // Build mesh to hold axis line\n this.axisMesh = new AbstractMesh('', gizmoLayer.utilityLayerScene);\n this.axisMesh.rotationQuaternion = Quaternion.Identity();\n const line = CylinderBuilder.CreateCylinder('', { diameterTop: 0.01, height: 0.3, diameterBottom: 0.01, tessellation }, gizmoLayer.utilityLayerScene);\n line.material = coloredMaterial;\n line.position.y = 0.375;\n const line2 = CylinderBuilder.CreateCylinder('', { diameterTop: 0.005, height: 0.9, diameterBottom: 0.005, tessellation }, gizmoLayer.utilityLayerScene);\n line2.material = coloredMaterial;\n line2.position.y = -0.225;\n const sphereMesh = SphereBuilder.CreateSphere('', { segments: tessellation, diameter: 0.45 }, gizmoLayer.utilityLayerScene);\n sphereMesh.material = wireframeMaterial;\n this.axisMesh.isPickable = false;\n this.axisMesh.addChild(line);\n this.axisMesh.addChild(line2);\n sphereMesh.addChild(this.axisMesh);\n sphereMesh.scaling.scaleInPlace(1 / 3);\n this._rootMesh.addChild(sphereMesh);\n let lastDragPosition = null;\n // Add drag behavior to handle events when the gizmo is dragged\n this.dragBehavior = new PointerDragBehavior();\n this.dragBehavior.moveAttached = false;\n this.dragBehavior.onDragStartObservable.add(() => {\n this.dragActive = true;\n });\n this.dragBehavior.onDragObservable.add(event => {\n // activeCamera.getForwardRayToRef(forwardRay, 1);\n // const cameraDirection = forwardRay.direction;\n // cameraDirection.rotateByQuaternionToRef(parentMesh.rotationQuaternion!, cameraDirection);\n // cameraDirection.addToRef(event.delta, targetDirection);\n // rotationToVectorToRef(targetDirection, cameraDirection, this._updateQuat);\n // parentMesh.rotationQuaternion!.multiplyInPlace(this._updateQuat);\n // this.updateObservable.notifyObservers({ delta: this._updateQuat });\n });\n this.dragBehavior.onDragEndObservable.add(() => {\n this.dragActive = false;\n });\n this._rootMesh.addBehavior(this.dragBehavior);\n this.dragBehavior.enabled = true;\n this._pointerObserver = gizmoLayer.utilityLayerScene.onPointerObservable.add(pointerInfo => {\n const isHovered = pointerInfo.pickInfo &&\n this._rootMesh.getChildMeshes().indexOf(pointerInfo.pickInfo.pickedMesh) != -1;\n sphereMesh.material = isHovered ? hoverMaterial : wireframeMaterial;\n if (pointerInfo.pickInfo && isHovered) {\n const point = pointerInfo.pickInfo.pickedPoint;\n if (point) {\n point.subtractInPlace(this._rootMesh.position);\n switch (pointerInfo.type) {\n case PointerEventTypes.POINTERDOWN:\n lastDragPosition = point.clone();\n break;\n case PointerEventTypes.POINTERMOVE:\n if (lastDragPosition && this.dragActive) {\n rotationToVectorToRef(point, lastDragPosition, this._updateQuat);\n lastDragPosition.copyFrom(point);\n // console.log(point, this._updateQuat);\n this.direction.rotateByQuaternionToRef(this._updateQuat, this.direction);\n this.updateObservable.notifyObservers({ delta: this._updateQuat });\n }\n break;\n }\n }\n }\n });\n // let light = gizmoLayer._getSharedGizmoLight();\n // light.includedOnlyMeshes = light.includedOnlyMeshes.concat(this._rootMesh.getChildMeshes(false));\n this._beforeRenderObserver = this.gizmoLayer.utilityLayerScene.onBeforeRenderObservable.add(() => {\n this._update();\n });\n }\n _update() {\n var _a;\n if (this.direction) {\n this.axisMesh.rotationQuaternion = rotationToVector(this.direction);\n }\n const activeCamera = this.gizmoLayer.utilityLayerScene.activeCamera;\n let cameraPosition = activeCamera.globalPosition;\n if (activeCamera.devicePosition) {\n cameraPosition = activeCamera.devicePosition;\n }\n this._rootMesh.position.subtractToRef(cameraPosition, this._tempVector);\n const dist = this._tempVector.length() * this.scaleRatio;\n this._rootMesh.scaling.set(dist, dist, dist);\n // Account for handedness, similar to Matrix.decompose\n if (((_a = this.gizmoLayer.originalScene.meshes[0]) === null || _a === void 0 ? void 0 : _a._getWorldMatrixDeterminant()) < 0) {\n this._rootMesh.scaling.y *= -1;\n }\n }\n /**\n * Disposes of the gizmo\n */\n dispose() {\n this.gizmoLayer.utilityLayerScene.onPointerObservable.remove(this._pointerObserver);\n this._rootMesh.dispose();\n this.dragBehavior.detach();\n if (this._beforeRenderObserver) {\n this.gizmoLayer.utilityLayerScene.onBeforeRenderObservable.remove(this._beforeRenderObserver);\n }\n }\n }\n\n /**\n * Anchor options where the Gizmo can be positioned in relation to its anchored node\n */\n var GizmoAnchorPoint;\n (function (GizmoAnchorPoint) {\n /** The origin of the attached node */\n GizmoAnchorPoint[GizmoAnchorPoint[\"Origin\"] = 0] = \"Origin\";\n /** The pivot point of the attached node*/\n GizmoAnchorPoint[GizmoAnchorPoint[\"Pivot\"] = 1] = \"Pivot\";\n })(GizmoAnchorPoint || (GizmoAnchorPoint = {}));\n /**\n * Coordinates mode: Local or World. Defines how axis is aligned: either on world axis or transform local axis\n */\n var GizmoCoordinatesMode;\n (function (GizmoCoordinatesMode) {\n GizmoCoordinatesMode[GizmoCoordinatesMode[\"World\"] = 0] = \"World\";\n GizmoCoordinatesMode[GizmoCoordinatesMode[\"Local\"] = 1] = \"Local\";\n })(GizmoCoordinatesMode || (GizmoCoordinatesMode = {}));\n /**\n * Renders gizmos on top of an existing scene which provide controls for position, rotation, etc.\n */\n class Gizmo {\n /**\n * Ratio for the scale of the gizmo (Default: 1)\n */\n set scaleRatio(value) {\n this._scaleRatio = value;\n }\n get scaleRatio() {\n return this._scaleRatio;\n }\n /**\n * True when the mouse pointer is hovered a gizmo mesh\n */\n get isHovered() {\n return this._isHovered;\n }\n /**\n * Mesh that the gizmo will be attached to. (eg. on a drag gizmo the mesh that will be dragged)\n * * When set, interactions will be enabled\n */\n get attachedMesh() {\n return this._attachedMesh;\n }\n set attachedMesh(value) {\n this._attachedMesh = value;\n if (value) {\n this._attachedNode = value;\n }\n this._rootMesh.setEnabled(value ? true : false);\n this._attachedNodeChanged(value);\n }\n /**\n * Node that the gizmo will be attached to. (eg. on a drag gizmo the mesh, bone or NodeTransform that will be dragged)\n * * When set, interactions will be enabled\n */\n get attachedNode() {\n return this._attachedNode;\n }\n set attachedNode(value) {\n this._attachedNode = value;\n this._attachedMesh = null;\n this._rootMesh.setEnabled(value ? true : false);\n this._attachedNodeChanged(value);\n }\n /**\n * Disposes and replaces the current meshes in the gizmo with the specified mesh\n * @param mesh The mesh to replace the default mesh of the gizmo\n */\n setCustomMesh(mesh) {\n if (mesh.getScene() != this.gizmoLayer.utilityLayerScene) {\n throw \"When setting a custom mesh on a gizmo, the custom meshes scene must be the same as the gizmos (eg. gizmo.gizmoLayer.utilityLayerScene)\";\n }\n this._rootMesh.getChildMeshes().forEach((c) => {\n c.dispose();\n });\n mesh.parent = this._rootMesh;\n this._customMeshSet = true;\n }\n /**\n * If set the gizmo's rotation will be updated to match the attached mesh each frame (Default: true)\n * NOTE: This is only possible for meshes with uniform scaling, as otherwise it's not possible to decompose the rotation\n */\n set updateGizmoRotationToMatchAttachedMesh(value) {\n this._updateGizmoRotationToMatchAttachedMesh = value;\n }\n get updateGizmoRotationToMatchAttachedMesh() {\n return this._updateGizmoRotationToMatchAttachedMesh;\n }\n /**\n * If set the gizmo's position will be updated to match the attached mesh each frame (Default: true)\n */\n set updateGizmoPositionToMatchAttachedMesh(value) {\n this._updateGizmoPositionToMatchAttachedMesh = value;\n }\n get updateGizmoPositionToMatchAttachedMesh() {\n return this._updateGizmoPositionToMatchAttachedMesh;\n }\n /**\n * Defines where the gizmo will be positioned if `updateGizmoPositionToMatchAttachedMesh` is enabled.\n * (Default: GizmoAnchorPoint.Origin)\n */\n set anchorPoint(value) {\n this._anchorPoint = value;\n }\n get anchorPoint() {\n return this._anchorPoint;\n }\n /**\n * Set the coordinate system to use. By default it's local.\n * But it's possible for a user to tweak so its local for translation and world for rotation.\n * In that case, setting the coordinate system will change `updateGizmoRotationToMatchAttachedMesh` and `updateGizmoPositionToMatchAttachedMesh`\n */\n set coordinatesMode(coordinatesMode) {\n this._coordinatesMode = coordinatesMode;\n const local = coordinatesMode == GizmoCoordinatesMode.Local;\n this.updateGizmoRotationToMatchAttachedMesh = local;\n this.updateGizmoPositionToMatchAttachedMesh = local;\n }\n get coordinatesMode() {\n return this._coordinatesMode;\n }\n /**\n * When set, the gizmo will always appear the same size no matter where the camera is (default: true)\n */\n set updateScale(value) {\n this._updateScale = value;\n }\n get updateScale() {\n return this._updateScale;\n }\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _attachedNodeChanged(value) { }\n /**\n * Creates a gizmo\n * @param gizmoLayer The utility layer the gizmo will be added to\n */\n constructor(\n /** The utility layer the gizmo will be added to */\n gizmoLayer = UtilityLayerRenderer.DefaultUtilityLayer) {\n this.gizmoLayer = gizmoLayer;\n this._attachedMesh = null;\n this._attachedNode = null;\n this._customRotationQuaternion = null;\n /**\n * Ratio for the scale of the gizmo (Default: 1)\n */\n this._scaleRatio = 1;\n /**\n * boolean updated by pointermove when a gizmo mesh is hovered\n */\n this._isHovered = false;\n /**\n * If a custom mesh has been set (Default: false)\n */\n this._customMeshSet = false;\n this._updateGizmoRotationToMatchAttachedMesh = true;\n this._updateGizmoPositionToMatchAttachedMesh = true;\n this._anchorPoint = GizmoAnchorPoint.Origin;\n this._updateScale = true;\n this._coordinatesMode = GizmoCoordinatesMode.Local;\n this._interactionsEnabled = true;\n this._rightHandtoLeftHandMatrix = Matrix.RotationY(Math.PI);\n this._rootMesh = new Mesh(\"gizmoRootNode\", gizmoLayer.utilityLayerScene);\n this._rootMesh.rotationQuaternion = Quaternion.Identity();\n this._beforeRenderObserver = this.gizmoLayer.utilityLayerScene.onBeforeRenderObservable.add(() => {\n this._update();\n });\n }\n /**\n * posture that the gizmo will be display\n * When set null, default value will be used (Quaternion(0, 0, 0, 1))\n */\n get customRotationQuaternion() {\n return this._customRotationQuaternion;\n }\n set customRotationQuaternion(customRotationQuaternion) {\n this._customRotationQuaternion = customRotationQuaternion;\n }\n /**\n * Updates the gizmo to match the attached mesh's position/rotation\n */\n _update() {\n if (this.attachedNode) {\n let effectiveNode = this.attachedNode;\n if (this.attachedMesh) {\n effectiveNode = this.attachedMesh || this.attachedNode;\n }\n // Position\n if (this.updateGizmoPositionToMatchAttachedMesh) {\n if (this.anchorPoint == GizmoAnchorPoint.Pivot && effectiveNode.getAbsolutePivotPoint) {\n const position = effectiveNode.getAbsolutePivotPoint();\n this._rootMesh.position.copyFrom(position);\n }\n else {\n const row = effectiveNode.getWorldMatrix().getRow(3);\n const position = row ? row.toVector3() : new Vector3(0, 0, 0);\n this._rootMesh.position.copyFrom(position);\n }\n }\n // Rotation\n if (this.updateGizmoRotationToMatchAttachedMesh) {\n const supportedNode = effectiveNode._isMesh ||\n effectiveNode.getClassName() === \"AbstractMesh\" ||\n effectiveNode.getClassName() === \"TransformNode\" ||\n effectiveNode.getClassName() === \"InstancedMesh\";\n const transformNode = supportedNode ? effectiveNode : undefined;\n effectiveNode.getWorldMatrix().decompose(undefined, this._rootMesh.rotationQuaternion, undefined, Gizmo.PreserveScaling ? transformNode : undefined);\n }\n else {\n if (this._customRotationQuaternion) {\n this._rootMesh.rotationQuaternion.copyFrom(this._customRotationQuaternion);\n }\n else {\n this._rootMesh.rotationQuaternion.set(0, 0, 0, 1);\n }\n }\n // Scale\n if (this.updateScale) {\n const activeCamera = this.gizmoLayer.utilityLayerScene.activeCamera;\n let cameraPosition = activeCamera.globalPosition;\n if (activeCamera.devicePosition) {\n cameraPosition = activeCamera.devicePosition;\n }\n this._rootMesh.position.subtractToRef(cameraPosition, TmpVectors.Vector3[0]);\n let scale = this.scaleRatio;\n if (activeCamera.mode == Camera.ORTHOGRAPHIC_CAMERA) {\n if (activeCamera.orthoTop && activeCamera.orthoBottom) {\n const orthoHeight = activeCamera.orthoTop - activeCamera.orthoBottom;\n scale *= orthoHeight;\n }\n }\n else {\n const camForward = activeCamera.getScene().useRightHandedSystem ? Vector3.RightHandedForwardReadOnly : Vector3.LeftHandedForwardReadOnly;\n const direction = activeCamera.getDirection(camForward);\n scale *= Vector3.Dot(TmpVectors.Vector3[0], direction);\n }\n this._rootMesh.scaling.setAll(scale);\n // Account for handedness, similar to Matrix.decompose\n if (effectiveNode._getWorldMatrixDeterminant() < 0 && !Gizmo.PreserveScaling) {\n this._rootMesh.scaling.y *= -1;\n }\n }\n else {\n this._rootMesh.scaling.setAll(this.scaleRatio);\n }\n }\n }\n /**\n * Handle position/translation when using an attached node using pivot\n */\n _handlePivot() {\n const attachedNodeTransform = this._attachedNode;\n // check there is an active pivot for the TransformNode attached\n if (attachedNodeTransform.isUsingPivotMatrix && attachedNodeTransform.isUsingPivotMatrix() && attachedNodeTransform.position) {\n // When a TransformNode has an active pivot, even without parenting,\n // translation from the world matrix is different from TransformNode.position.\n // Pivot works like a virtual parent that's using the node orientation.\n // As the world matrix is transformed by the gizmo and then decomposed to TRS\n // its translation part must be set to the Node's position.\n attachedNodeTransform.getWorldMatrix().setTranslation(attachedNodeTransform.position);\n }\n }\n /**\n * computes the rotation/scaling/position of the transform once the Node world matrix has changed.\n */\n _matrixChanged() {\n if (!this._attachedNode) {\n return;\n }\n if (this._attachedNode._isCamera) {\n const camera = this._attachedNode;\n let worldMatrix;\n let worldMatrixUC;\n if (camera.parent) {\n const parentInv = TmpVectors.Matrix[1];\n camera.parent._worldMatrix.invertToRef(parentInv);\n this._attachedNode._worldMatrix.multiplyToRef(parentInv, TmpVectors.Matrix[0]);\n worldMatrix = TmpVectors.Matrix[0];\n }\n else {\n worldMatrix = this._attachedNode._worldMatrix;\n }\n if (camera.getScene().useRightHandedSystem) {\n // avoid desync with RH matrix computation. Otherwise, rotation of PI around Y axis happens each frame resulting in axis flipped because worldMatrix is computed as inverse of viewMatrix.\n this._rightHandtoLeftHandMatrix.multiplyToRef(worldMatrix, TmpVectors.Matrix[1]);\n worldMatrixUC = TmpVectors.Matrix[1];\n }\n else {\n worldMatrixUC = worldMatrix;\n }\n worldMatrixUC.decompose(TmpVectors.Vector3[1], TmpVectors.Quaternion[0], TmpVectors.Vector3[0]);\n const inheritsTargetCamera = this._attachedNode.getClassName() === \"FreeCamera\" ||\n this._attachedNode.getClassName() === \"FlyCamera\" ||\n this._attachedNode.getClassName() === \"ArcFollowCamera\" ||\n this._attachedNode.getClassName() === \"TargetCamera\" ||\n this._attachedNode.getClassName() === \"TouchCamera\" ||\n this._attachedNode.getClassName() === \"UniversalCamera\";\n if (inheritsTargetCamera) {\n const targetCamera = this._attachedNode;\n targetCamera.rotation = TmpVectors.Quaternion[0].toEulerAngles();\n if (targetCamera.rotationQuaternion) {\n targetCamera.rotationQuaternion.copyFrom(TmpVectors.Quaternion[0]);\n targetCamera.rotationQuaternion.normalize();\n }\n }\n camera.position.copyFrom(TmpVectors.Vector3[0]);\n }\n else if (this._attachedNode._isMesh ||\n this._attachedNode.getClassName() === \"AbstractMesh\" ||\n this._attachedNode.getClassName() === \"TransformNode\" ||\n this._attachedNode.getClassName() === \"InstancedMesh\") {\n const transform = this._attachedNode;\n if (transform.parent) {\n const parentInv = TmpVectors.Matrix[0];\n const localMat = TmpVectors.Matrix[1];\n transform.parent.getWorldMatrix().invertToRef(parentInv);\n this._attachedNode.getWorldMatrix().multiplyToRef(parentInv, localMat);\n localMat.decompose(TmpVectors.Vector3[0], TmpVectors.Quaternion[0], transform.position, Gizmo.PreserveScaling ? transform : undefined);\n }\n else {\n this._attachedNode._worldMatrix.decompose(TmpVectors.Vector3[0], TmpVectors.Quaternion[0], transform.position, Gizmo.PreserveScaling ? transform : undefined);\n }\n TmpVectors.Vector3[0].scaleInPlace(1.0 / transform.scalingDeterminant);\n transform.scaling.copyFrom(TmpVectors.Vector3[0]);\n if (!transform.billboardMode) {\n if (transform.rotationQuaternion) {\n transform.rotationQuaternion.copyFrom(TmpVectors.Quaternion[0]);\n transform.rotationQuaternion.normalize();\n }\n else {\n transform.rotation = TmpVectors.Quaternion[0].toEulerAngles();\n }\n }\n }\n else if (this._attachedNode.getClassName() === \"Bone\") {\n const bone = this._attachedNode;\n const parent = bone.getParent();\n if (parent) {\n const invParent = TmpVectors.Matrix[0];\n const boneLocalMatrix = TmpVectors.Matrix[1];\n parent.getFinalMatrix().invertToRef(invParent);\n bone.getFinalMatrix().multiplyToRef(invParent, boneLocalMatrix);\n const lmat = bone.getLocalMatrix();\n lmat.copyFrom(boneLocalMatrix);\n }\n else {\n const lmat = bone.getLocalMatrix();\n lmat.copyFrom(bone.getFinalMatrix());\n }\n bone.markAsDirty();\n }\n else {\n const light = this._attachedNode;\n if (light.getTypeID) {\n const type = light.getTypeID();\n if (type === Light.LIGHTTYPEID_DIRECTIONALLIGHT || type === Light.LIGHTTYPEID_SPOTLIGHT || type === Light.LIGHTTYPEID_POINTLIGHT) {\n const parent = light.parent;\n if (parent) {\n const invParent = TmpVectors.Matrix[0];\n const nodeLocalMatrix = TmpVectors.Matrix[1];\n parent.getWorldMatrix().invertToRef(invParent);\n light.getWorldMatrix().multiplyToRef(invParent, nodeLocalMatrix);\n nodeLocalMatrix.decompose(undefined, TmpVectors.Quaternion[0], TmpVectors.Vector3[0]);\n }\n else {\n this._attachedNode._worldMatrix.decompose(undefined, TmpVectors.Quaternion[0], TmpVectors.Vector3[0]);\n }\n // setter doesn't copy values. Need a new Vector3\n light.position = new Vector3(TmpVectors.Vector3[0].x, TmpVectors.Vector3[0].y, TmpVectors.Vector3[0].z);\n if (light.direction) {\n light.direction = new Vector3(light.direction.x, light.direction.y, light.direction.z);\n }\n }\n }\n }\n }\n /**\n * refresh gizmo mesh material\n * @param gizmoMeshes\n * @param material material to apply\n */\n _setGizmoMeshMaterial(gizmoMeshes, material) {\n if (gizmoMeshes) {\n gizmoMeshes.forEach((m) => {\n m.material = material;\n if (m.color) {\n m.color = material.diffuseColor;\n }\n });\n }\n }\n /**\n * Subscribes to pointer up, down, and hover events. Used for responsive gizmos.\n * @param gizmoLayer The utility layer the gizmo will be added to\n * @param gizmoAxisCache Gizmo axis definition used for reactive gizmo UI\n * @returns {Observer} pointerObserver\n */\n static GizmoAxisPointerObserver(gizmoLayer, gizmoAxisCache) {\n let dragging = false;\n const pointerObserver = gizmoLayer.utilityLayerScene.onPointerObservable.add((pointerInfo) => {\n var _a, _b;\n if (pointerInfo.pickInfo) {\n // On Hover Logic\n if (pointerInfo.type === PointerEventTypes.POINTERMOVE) {\n if (dragging) {\n return;\n }\n gizmoAxisCache.forEach((cache) => {\n var _a, _b;\n if (cache.colliderMeshes && cache.gizmoMeshes) {\n const isHovered = ((_a = cache.colliderMeshes) === null || _a === void 0 ? void 0 : _a.indexOf((_b = pointerInfo === null || pointerInfo === void 0 ? void 0 : pointerInfo.pickInfo) === null || _b === void 0 ? void 0 : _b.pickedMesh)) != -1;\n const material = cache.dragBehavior.enabled ? (isHovered || cache.active ? cache.hoverMaterial : cache.material) : cache.disableMaterial;\n cache.gizmoMeshes.forEach((m) => {\n m.material = material;\n if (m.color) {\n m.color = material.diffuseColor;\n }\n });\n }\n });\n }\n // On Mouse Down\n if (pointerInfo.type === PointerEventTypes.POINTERDOWN) {\n // If user Clicked Gizmo\n if (gizmoAxisCache.has((_a = pointerInfo.pickInfo.pickedMesh) === null || _a === void 0 ? void 0 : _a.parent)) {\n dragging = true;\n const statusMap = gizmoAxisCache.get((_b = pointerInfo.pickInfo.pickedMesh) === null || _b === void 0 ? void 0 : _b.parent);\n statusMap.active = true;\n gizmoAxisCache.forEach((cache) => {\n var _a, _b;\n const isHovered = ((_a = cache.colliderMeshes) === null || _a === void 0 ? void 0 : _a.indexOf((_b = pointerInfo === null || pointerInfo === void 0 ? void 0 : pointerInfo.pickInfo) === null || _b === void 0 ? void 0 : _b.pickedMesh)) != -1;\n const material = (isHovered || cache.active) && cache.dragBehavior.enabled ? cache.hoverMaterial : cache.disableMaterial;\n cache.gizmoMeshes.forEach((m) => {\n m.material = material;\n if (m.color) {\n m.color = material.diffuseColor;\n }\n });\n });\n }\n }\n // On Mouse Up\n if (pointerInfo.type === PointerEventTypes.POINTERUP) {\n gizmoAxisCache.forEach((cache) => {\n cache.active = false;\n dragging = false;\n cache.gizmoMeshes.forEach((m) => {\n m.material = cache.dragBehavior.enabled ? cache.material : cache.disableMaterial;\n if (m.color) {\n m.color = cache.material.diffuseColor;\n }\n });\n });\n }\n }\n });\n return pointerObserver;\n }\n /**\n * Disposes of the gizmo\n */\n dispose() {\n this._rootMesh.dispose();\n if (this._beforeRenderObserver) {\n this.gizmoLayer.utilityLayerScene.onBeforeRenderObservable.remove(this._beforeRenderObserver);\n }\n }\n }\n /**\n * When enabled, any gizmo operation will perserve scaling sign. Default is off.\n * Only valid for TransformNode derived classes (Mesh, AbstractMesh, ...)\n */\n Gizmo.PreserveScaling = false;\n\n /**\n * Renders gizmos on top of an existing scene which provide controls for position, rotation, etc.\n */\n const AUTO_MESH_NAME = '_autoGizmoMesh';\n class KbGizmo extends Gizmo {\n updateGizmoPosition() {\n this._update();\n }\n attachTo(node) { }\n get attachedMesh() {\n return super.attachedMesh;\n }\n set attachedMesh(m) {\n if (m) {\n super.attachedMesh = m;\n }\n }\n get visibility() {\n return this._rootMesh.visibility;\n }\n set visibility(visibility) {\n this._rootMesh.setEnabled(visibility === 1);\n }\n attachMeshAtNodeSpace(scene, target, positionSpaceTransform) {\n if (!this.attachedMesh) {\n this.attachedMesh = new Mesh(AUTO_MESH_NAME, scene);\n }\n let translation;\n const rotation = Quaternion.Identity();\n const scale = Vector3.One();\n translation = target instanceof exports.SpaceNode ? target.position.toVec3() : target.toVec3();\n if (positionSpaceTransform) {\n translation = Vector3.TransformCoordinates(translation, positionSpaceTransform);\n positionSpaceTransform.decompose(scale, rotation);\n }\n this.attachedMesh.position = translation;\n this.attachedMesh.rotationQuaternion = rotation;\n this.attachedMesh.scaling = scale;\n return this.attachedMesh;\n }\n _update() {\n if (this.attachedMesh) {\n const _tempVector = new Vector3();\n const _tmpMatrix = new Matrix();\n if (this.updateGizmoRotationToMatchAttachedMesh) {\n if (!this._rootMesh.rotationQuaternion) {\n this._rootMesh.rotationQuaternion = Quaternion.RotationYawPitchRoll(this._rootMesh.rotation.y, this._rootMesh.rotation.x, this._rootMesh.rotation.z);\n }\n // Remove scaling before getting rotation matrix to get rotation matrix unmodified by scale\n _tempVector.copyFrom(this.attachedMesh.scaling);\n if (this.attachedMesh.scaling.x < 0) {\n this.attachedMesh.scaling.x *= -1;\n }\n if (this.attachedMesh.scaling.y < 0) {\n this.attachedMesh.scaling.y *= -1;\n }\n if (this.attachedMesh.scaling.z < 0) {\n this.attachedMesh.scaling.z *= -1;\n }\n this.attachedMesh.computeWorldMatrix().getRotationMatrixToRef(_tmpMatrix);\n this.attachedMesh.scaling.copyFrom(_tempVector);\n this.attachedMesh.computeWorldMatrix();\n Quaternion.FromRotationMatrixToRef(_tmpMatrix, this._rootMesh.rotationQuaternion);\n }\n // else if (this._rootMesh.rotationQuaternion) {\n // this._rootMesh.rotationQuaternion.set(0, 0, 0, 1);\n // }\n if (this.updateGizmoPositionToMatchAttachedMesh) {\n this._rootMesh.position.copyFrom(this.attachedMesh.absolutePosition);\n }\n if (this.updateScale && this.gizmoLayer.utilityLayerScene.activeCamera && this.attachedMesh) {\n let dist;\n const camera = this.gizmoLayer.utilityLayerScene.activeCamera;\n if (camera.mode === Camera.ORTHOGRAPHIC_CAMERA) {\n dist = ((camera.orthoRight || 0) - (camera.orthoLeft || 0)) / 2;\n }\n else {\n let cameraPosition;\n if (camera instanceof WebXRCamera && camera.globalPosition) {\n cameraPosition = camera.globalPosition;\n }\n else if (camera instanceof ArcRotateCamera) {\n cameraPosition = getCameraPositionWithOffset(camera);\n }\n else {\n cameraPosition = camera.globalPosition;\n }\n this._rootMesh.position.subtractToRef(cameraPosition, _tempVector);\n dist = _tempVector.length() * this.scaleRatio;\n }\n this._rootMesh.scaling.set(dist, dist, dist);\n }\n const pivot = this.attachedMesh.getPivotPoint();\n //let absolutePosition = this.attachedMesh.absolutePosition;\n // Absolute position is getting skewed because of the pivot point; need to get position without the rotation\n // if (pivot.length() > 0 && this.attachedMesh.rotationQuaternion) {\n // var m = new Matrix();\n // this.attachedMesh.rotationQuaternion.toRotationMatrix(m);\n // absolutePosition = Vector3.TransformCoordinates(absolutePosition, m)\n // }\n // if (pivot.length() > 0) {\n // absolutePosition = Vector3.Zero();\n // }\n this._rootMesh.position.copyFrom(Vector3.TransformCoordinates(pivot, this.attachedMesh.getWorldMatrix()));\n }\n }\n }\n\n /**\n * Single plane rotation gizmo\n */\n class KbPlaneRotationGizmo extends KbGizmo {\n /**\n * Creates a PlaneRotationGizmo\n * @param gizmoLayer The utility layer the gizmo will be added to\n * @param planeNormal The normal of the plane which the gizmo will be able to rotate on\n * @param color The color of the gizmo\n * @param tessellation Amount of tessellation to be used when creating rotation circles\n */\n constructor(planeNormal, color = Color3.Gray(), gizmoLayer = UtilityLayerRenderer.DefaultUtilityLayer, tessellation = 32) {\n super(gizmoLayer);\n this._pointerObserver = null;\n /**\n * Rotation distance in radians that the gizmo will snap to (Default: 0)\n */\n this.snapDistance = 0;\n /**\n * Event that fires each time the gizmo snaps to a new location.\n * * snapDistance is the the change in distance\n */\n this.onSnapObservable = new Observable$1();\n this.updateObservable = new Observable$1();\n // Create Material\n const coloredMaterial = new StandardMaterial('', gizmoLayer.utilityLayerScene);\n coloredMaterial.emissiveColor = color;\n // coloredMaterial.diffuseColor = color;\n // coloredMaterial.specularColor = color.subtract(new Color3(0.1, 0.1, 0.1));\n const hoverMaterial = new StandardMaterial('', gizmoLayer.utilityLayerScene);\n hoverMaterial.emissiveColor = color.add(new Color3(0.3, 0.3, 0.3));\n // hoverMaterial.diffuseColor = color.add(new Color3(0.3, 0.3, 0.3));\n // Build mesh on root node\n const parentMesh = new AbstractMesh('', gizmoLayer.utilityLayerScene);\n const drag = Mesh.CreateTorus('', 0.6, 0.045, tessellation, gizmoLayer.utilityLayerScene);\n drag.visibility = 0;\n const rotationMesh = Mesh.CreateTorus('', 0.6, 0.01, tessellation, gizmoLayer.utilityLayerScene);\n rotationMesh.material = coloredMaterial;\n // Position arrow pointing in its drag axis\n rotationMesh.rotation.x = Math.PI / 2;\n drag.rotation.x = Math.PI / 2;\n parentMesh.addChild(rotationMesh);\n parentMesh.addChild(drag);\n parentMesh.lookAt(this._rootMesh.position.add(planeNormal));\n this.rotationMesh = parentMesh;\n this._rootMesh.addChild(parentMesh);\n parentMesh.scaling.scaleInPlace(1 / 3);\n // Add drag behavior to handle events when the gizmo is dragged\n this.dragBehavior = new PointerDragBehavior({ dragPlaneNormal: planeNormal });\n this.dragBehavior.moveAttached = false;\n this.dragBehavior.maxDragAngle = (Math.PI * 9) / 20;\n this.dragBehavior._useAlternatePickedPointAboveMaxDragAngle = true;\n this._rootMesh.addBehavior(this.dragBehavior);\n const lastDragPosition = new Vector3();\n this.dragBehavior.onDragStartObservable.add(e => {\n if (this.attachedMesh) {\n lastDragPosition.copyFrom(e.dragPlanePoint);\n }\n });\n const rotationMatrix = new Matrix();\n const planeNormalTowardsCamera = new Vector3();\n let localPlaneNormalTowardsCamera = new Vector3();\n const tmpSnapEvent = { snapDistance: 0 };\n let currentSnapDragDistance = 0;\n const tmpMatrix = new Matrix();\n const tmpVector = new Vector3();\n const amountToRotate = new Quaternion();\n this.dragBehavior.onDragObservable.add(event => {\n if (this.attachedMesh) {\n if (!this.attachedMesh.rotationQuaternion) {\n this.attachedMesh.rotationQuaternion = Quaternion.RotationYawPitchRoll(this.attachedMesh.rotation.y, this.attachedMesh.rotation.x, this.attachedMesh.rotation.z);\n }\n // Remove parent priort to rotating\n const attachedMeshParent = this.attachedMesh.parent;\n if (attachedMeshParent) {\n this.attachedMesh.setParent(null);\n }\n // Calc angle over full 360 degree (https://stackoverflow.com/questions/43493711/the-angle-between-two-3d-vectors-with-a-result-range-0-360)\n const pivot = this.attachedMesh.getPivotPoint();\n const attachedMeshPosition = this.attachedMesh.absolutePosition.add(pivot);\n const newVector = event.dragPlanePoint.subtract(attachedMeshPosition).normalize();\n const originalVector = lastDragPosition.subtract(attachedMeshPosition).normalize();\n const cross = Vector3.Cross(newVector, originalVector);\n const dot = Vector3.Dot(newVector, originalVector);\n let angle = Math.atan2(cross.length(), dot);\n planeNormalTowardsCamera.copyFrom(planeNormal);\n localPlaneNormalTowardsCamera.copyFrom(planeNormal);\n if (this.updateGizmoRotationToMatchAttachedMesh) {\n this.attachedMesh.rotationQuaternion.toRotationMatrix(rotationMatrix);\n localPlaneNormalTowardsCamera = Vector3.TransformCoordinates(planeNormalTowardsCamera, rotationMatrix);\n }\n // Flip up vector depending on which side the camera is on\n if (gizmoLayer.utilityLayerScene.activeCamera) {\n const camVec = gizmoLayer.utilityLayerScene.activeCamera.position.subtract(this.attachedMesh.position);\n if (Vector3.Dot(camVec, localPlaneNormalTowardsCamera) > 0) {\n planeNormalTowardsCamera.scaleInPlace(-1);\n localPlaneNormalTowardsCamera.scaleInPlace(-1);\n }\n }\n const halfCircleSide = Vector3.Dot(localPlaneNormalTowardsCamera, cross) > 0.0;\n if (halfCircleSide) {\n angle = -angle;\n }\n // Snapping logic\n let snapped = false;\n if (this.snapDistance != 0) {\n currentSnapDragDistance += angle;\n if (Math.abs(currentSnapDragDistance) > this.snapDistance) {\n let dragSteps = Math.floor(Math.abs(currentSnapDragDistance) / this.snapDistance);\n if (currentSnapDragDistance < 0) {\n dragSteps *= -1;\n }\n currentSnapDragDistance = currentSnapDragDistance % this.snapDistance;\n angle = this.snapDistance * dragSteps;\n snapped = true;\n }\n else {\n angle = 0;\n }\n }\n // If the mesh has a parent, convert needed world rotation to local rotation\n tmpMatrix.reset();\n if (this.attachedMesh.parent) {\n this.attachedMesh.parent.computeWorldMatrix().invertToRef(tmpMatrix);\n tmpMatrix.getRotationMatrixToRef(tmpMatrix);\n Vector3.TransformCoordinatesToRef(planeNormalTowardsCamera, tmpMatrix, planeNormalTowardsCamera);\n }\n // Convert angle and axis to quaternion (http://www.euclideanspace.com/maths/geometry/rotations/conversions/angleToQuaternion/index.htm)\n const quaternionCoefficient = Math.sin(angle / 2);\n amountToRotate.set(planeNormalTowardsCamera.x * quaternionCoefficient, planeNormalTowardsCamera.y * quaternionCoefficient, planeNormalTowardsCamera.z * quaternionCoefficient, Math.cos(angle / 2));\n // If the meshes local scale is inverted (eg. loaded gltf file parent with z scale of -1) the rotation needs to be inverted on the y axis\n if (tmpMatrix.determinant() > 0) {\n amountToRotate.toEulerAnglesToRef(tmpVector);\n Quaternion.RotationYawPitchRollToRef(tmpVector.y, -tmpVector.x, -tmpVector.z, amountToRotate);\n }\n if (this.updateGizmoRotationToMatchAttachedMesh) {\n // Rotate selected mesh quaternion over fixed axis\n this.attachedMesh.rotationQuaternion.multiplyToRef(amountToRotate, this.attachedMesh.rotationQuaternion);\n }\n else {\n // Rotate selected mesh quaternion over rotated axis\n amountToRotate.multiplyToRef(this.attachedMesh.rotationQuaternion, this.attachedMesh.rotationQuaternion);\n }\n this.updateObservable.notifyObservers({ delta: amountToRotate, deltaAngle: angle });\n lastDragPosition.copyFrom(event.dragPlanePoint);\n if (snapped) {\n tmpSnapEvent.snapDistance = angle;\n this.onSnapObservable.notifyObservers(tmpSnapEvent);\n }\n // Restore parent\n if (attachedMeshParent) {\n this.attachedMesh.setParent(attachedMeshParent);\n }\n }\n });\n this._pointerObserver = gizmoLayer.utilityLayerScene.onPointerObservable.add(pointerInfo => {\n if (this._customMeshSet) {\n return;\n }\n const isHovered = pointerInfo.pickInfo &&\n this._rootMesh.getChildMeshes().indexOf(pointerInfo.pickInfo.pickedMesh) != -1;\n const material = isHovered ? hoverMaterial : coloredMaterial;\n this._rootMesh.getChildMeshes().forEach(m => {\n m.material = material;\n if (m.color) {\n m.color = material.diffuseColor;\n }\n });\n });\n // var light = gizmoLayer._getSharedGizmoLight();\n // light.includedOnlyMeshes = light.includedOnlyMeshes.concat(this._rootMesh.getChildMeshes(false));\n }\n _attachedMeshChanged(value) {\n if (this.dragBehavior) {\n this.dragBehavior.enabled = value ? true : false;\n }\n // if (value) {\n // this.rotationMesh.setPivotPoint(Vector3.Zero(), Space.LOCAL);\n // }\n }\n /**\n * Disposes of the gizmo\n */\n dispose() {\n this.onSnapObservable.clear();\n this.gizmoLayer.utilityLayerScene.onPointerObservable.remove(this._pointerObserver);\n this.dragBehavior.detach();\n super.dispose();\n }\n }\n\n const tessellation = 32;\n /**\n * TODO\n * Copy in plane rotation gizmo code here and change it to use a sphere and click projection on the sphere\n * Convert it to return an axis\n * Edit code in here to return an axis and angle\n * update gizmo tools code to hook to this gizmo\n * remove the vector option from the original rotation gizmo (now the euler rotation gizmo), simplify code\n */\n /**\n * Gizmo that enables rotating a mesh along 3 axis\n */\n class KbAxisAngleGizmo extends KbGizmo {\n attachTo(target, positionSpace, placementOffset) {\n if (positionSpace) {\n this._positionSpace = this.kbViewer.getRendererNode(positionSpace);\n }\n this._updateGizmoPosition(target, placementOffset);\n }\n updateGizmoPosition(changeTarget, placementOffset) {\n // Need to wait for the next frame because this can be called before the babylon object is actually updated\n this.kbViewer.onNextFrame(() => {\n this._updateGizmoPosition(changeTarget, placementOffset);\n });\n }\n _updateGizmoPosition(changeTarget, placementOffset) {\n var _a;\n // Maintain the placement offset if it is not provided later, so this can be called from other contexts to update based on external changes\n if (!placementOffset && this.placementOffset) {\n placementOffset = this.placementOffset;\n }\n else if (placementOffset) {\n this.placementOffset = placementOffset;\n }\n const positionSpaceTransform = (_a = this._positionSpace) === null || _a === void 0 ? void 0 : _a.computeWorldMatrix(true);\n if (changeTarget) {\n changeTarget.toVec3(this.targetAxis);\n // If target is a vector, then we want to rotate it as a direction, so set the gizmo position to the origin to calculations are done from there\n // But store the position as the offset because we want the gizmo itself to be visually placed on the position space origin\n this.attachMeshAtNodeSpace(this.gizmoLayer.utilityLayerScene, placementOffset || exports.KbVector.Zero(), positionSpaceTransform);\n if (this.attachedMesh) {\n this.gizmoOffset = exports.KbVector.FromVec3(this.attachedMesh.position);\n this.attachedMesh.position = Vector3.Zero();\n this.attachedMesh.computeWorldMatrix(true);\n }\n const targetAxisTransformed = this.targetAxis.clone();\n if (positionSpaceTransform) {\n Vector3.TransformCoordinatesToRef(this.targetAxis, positionSpaceTransform.getRotationMatrix(), targetAxisTransformed);\n }\n this.axisGizmo.direction.copyFrom(targetAxisTransformed);\n }\n if (this.gizmoOffset && this.attachedMesh) {\n this.attachedMesh.position.copyFrom(this.gizmoOffset.toVec3());\n this.axisGizmo.gizmoPosition.copyFrom(this.gizmoOffset.toVec3());\n // if (placementOffset) {\n // this.axisGizmo.gizmoPosition.addInPlace(placementOffset.toVec3());\n // }\n }\n // if (positionSpaceTransform) {\n // Vector3.TransformCoordinatesToRef(this.axisGizmo.gizmoPosition, positionSpaceTransform, this.axisGizmo.gizmoPosition);\n // }\n super.updateGizmoPosition();\n }\n get attachedMesh() {\n return super.attachedMesh;\n }\n set attachedMesh(mesh) {\n super.attachedMesh = mesh;\n if (mesh) {\n const angleMesh = new Mesh('', this.gizmoLayer.utilityLayerScene);\n mesh.addChild(angleMesh);\n if (this.angleGizmo) {\n this.angleGizmo.attachedMesh = angleMesh;\n }\n angleMesh.lookAt(this.targetAxis);\n angleMesh.setPivotPoint(mesh.getPivotPoint(), Space.WORLD);\n }\n else if (this.angleGizmo) {\n this.angleGizmo.attachedMesh = null;\n }\n }\n /**\n * Creates a RotationGizmo\n * @param gizmoLayer The utility layer the gizmo will be added to\n * @param tessellation Amount of tessellation to be used when creating rotation circles\n */\n constructor(kbViewer, gizmoLayer = UtilityLayerRenderer.DefaultUtilityLayer, disableAngle = false) {\n super(gizmoLayer);\n this.kbViewer = kbViewer;\n this.targetAxis = new Vector3();\n /** Fires an event when any of it's sub gizmos are dragged */\n this.onDragStartObservable = new Observable$1();\n /** Fires an event when any of it's sub gizmos are released from dragging */\n this.onDragObservable = new Observable$1();\n this.onDragEndObservable = new Observable$1();\n this.updateGizmoRotationToMatchAttachedMesh = true;\n this.axisGizmo = new KbAxisMorphGizmo(Color3.Green().scale(0.5), gizmoLayer, tessellation);\n if (!disableAngle) {\n this.angleGizmo = new KbPlaneRotationGizmo(new Vector3(0, 0, 1), Color3.Green().scale(0.5), gizmoLayer, tessellation);\n this.angleGizmo.updateGizmoPositionToMatchAttachedMesh = true;\n }\n const zeroVector = Vector3.Zero();\n const tempVector = Vector3.Zero();\n const tempVector2 = Vector3.Zero();\n let positionSpaceTransform;\n this.axisChange = Vector3.Zero();\n this.angleChange = 0;\n // Relay drag events\n [this.axisGizmo, this.angleGizmo].forEach(gizmo => {\n if (gizmo) {\n gizmo.dragBehavior.onDragStartObservable.add(() => {\n this.onDragStartObservable.notifyObservers();\n this.axisChange = Vector3.Zero();\n this.angleChange = 0;\n if (!positionSpaceTransform && this._positionSpace) {\n positionSpaceTransform = this._positionSpace\n .computeWorldMatrix(true)\n .getRotationMatrix()\n .clone()\n .invert();\n }\n });\n gizmo.dragBehavior.onDragEndObservable.add(() => {\n this.onDragEndObservable.notifyObservers({\n deltaAxis: this.axisChange,\n deltaAngle: this.angleChange,\n });\n });\n }\n });\n this.axisGizmo.updateObservable.add(event => {\n tempVector.copyFrom(this.targetAxis);\n // if (positionSpaceTransform) {\n // Vector3.TransformCoordinatesToRef(tempVector, positionSpaceTransform, tempVector);\n // }\n tempVector.rotateByQuaternionToRef(event.delta, tempVector2);\n // if (positionSpaceTransform) {\n // Vector3.TransformCoordinatesToRef(tempVector2, positionSpaceTransform.clone().invert(), tempVector2);\n // }\n tempVector2.subtractToRef(this.targetAxis, tempVector2);\n this.targetAxis.addInPlace(tempVector2);\n this.axisChange.addInPlace(tempVector2);\n this.onDragObservable.notifyObservers({ deltaAxis: tempVector2, deltaAngle: 0 });\n if (this.angleGizmo && this.angleGizmo.attachedMesh) {\n this.angleGizmo.attachedMesh.lookAt(positionSpaceTransform\n ? Vector3.TransformCoordinates(this.targetAxis, positionSpaceTransform)\n : this.targetAxis);\n }\n });\n if (this.angleGizmo) {\n this.angleGizmo.updateObservable.add(event => {\n this.angleChange -= event.deltaAngle;\n this.onDragObservable.notifyObservers({ deltaAxis: zeroVector, deltaAngle: -event.deltaAngle });\n });\n }\n this.attachedMesh = null;\n }\n /**\n * Drag distance in babylon units that the gizmo will snap to when dragged (Default: 0)\n */\n set snapDistance(value) {\n if (this.axisGizmo) {\n this.axisGizmo.snapDistance = value;\n }\n if (this.angleGizmo) {\n this.angleGizmo.snapDistance = value;\n }\n }\n get snapDistance() {\n return this.axisGizmo.snapDistance;\n }\n /**\n * Ratio for the scale of the gizmo (Default: 1)\n */\n set scaleRatio(value) {\n if (this.axisGizmo) {\n this.axisGizmo.scaleRatio = value;\n }\n if (this.angleGizmo) {\n this.angleGizmo.scaleRatio = value;\n }\n }\n get scaleRatio() {\n return this.axisGizmo.scaleRatio;\n }\n /**\n * Disposes of the gizmo\n */\n dispose() {\n if (this.axisGizmo) {\n this.axisGizmo.dispose();\n }\n if (this.angleGizmo) {\n this.angleGizmo.dispose();\n }\n this.onDragStartObservable.clear();\n this.onDragEndObservable.clear();\n }\n /**\n * CustomMeshes are not supported by this gizmo\n * @param mesh The mesh to replace the default mesh of the gizmo\n */\n setCustomMesh(mesh) {\n Logger.Error('Custom meshes are not supported on this gizmo, please set the custom meshes on the gizmos contained within this one (gizmo.xGizmo, gizmo.angleGizmo, gizmo.zGizmo)');\n }\n }\n\n const meshFaceInfo = new Map();\n class MeshFaceEngine {\n constructor(kbViewer) {\n this.kbViewer = kbViewer;\n this.bbCache = kbViewer.bbCache;\n }\n initialize(node) {\n let meshNode;\n let mesh;\n if (node instanceof exports.Kb3dObject) {\n mesh = this.kbViewer.getRendererNode(node);\n meshNode = node;\n }\n else {\n mesh = node;\n meshNode = getByDynamicId(node.id);\n }\n if (mesh instanceof Mesh && mesh.geometry) {\n this.bbCache\n .watchForChanges(meshNode)\n .pipe(take(1))\n .subscribe(() => {\n meshFaceInfo.delete(meshNode._dynamicId);\n });\n return this.updateNodeGeometry(mesh);\n }\n }\n getMeshFaces(node) {\n let result = meshFaceInfo.get(node instanceof exports.Kb3dObject ? node._dynamicId : node.id);\n if (!result) {\n result = this.initialize(node);\n }\n return result;\n }\n /**\n * @param selection array of selected faces to expand from\n * @param limitAngle angle to limit selection to. If undefined, function will expand by 1 polygon\n */\n expandSelection(selection, node, limitAngle) {\n const faceInfo = this.getMeshFaces(node);\n const faceNormals = faceInfo.faceNormals;\n const selectedFaceIds = [];\n const newSelection = [];\n if (limitAngle !== undefined) {\n limitAngle = (limitAngle * Math.PI) / 180;\n }\n const v1 = new Vector3(), v2 = new Vector3();\n if (!faceInfo) {\n return selection;\n }\n for (let i = 0; i < selection.length; i++) {\n selectedFaceIds[selection[i]] = true;\n }\n for (let i = 0; i < selection.length; i++) {\n const faceId = selection[i];\n v1.copyFromFloats(faceNormals[faceId * 3], faceNormals[faceId * 3 + 1], faceNormals[faceId * 3 + 2]);\n recursiveExpandSelection(faceId, 1, v1);\n }\n for (const faceId in selectedFaceIds) {\n newSelection.push(parseInt(faceId, 10));\n }\n return newSelection;\n function recursiveExpandSelection(faceId, iterCount, normal) {\n const neighbors = faceInfo.faceNeighbors[faceId];\n if (limitAngle === undefined && iterCount <= 0) {\n return;\n }\n for (let i = 0; i < neighbors.length; i++) {\n const adjacentFaceId = neighbors[i];\n if (!selectedFaceIds[adjacentFaceId]) {\n if (limitAngle !== undefined) {\n v2.copyFromFloats(faceNormals[adjacentFaceId * 3], faceNormals[adjacentFaceId * 3 + 1], faceNormals[adjacentFaceId * 3 + 2]);\n if (Math.abs(Math.acos(Math.round(Vector3.Dot(normal, v2) * 1000000) / 1000000)) <= limitAngle) {\n // deal with FP Errors\n selectedFaceIds[adjacentFaceId] = true;\n recursiveExpandSelection(adjacentFaceId, iterCount - 1, normal);\n }\n }\n else {\n selectedFaceIds[adjacentFaceId] = true;\n recursiveExpandSelection(adjacentFaceId, iterCount - 1, normal);\n }\n }\n }\n }\n }\n updateNodeGeometry(mesh) {\n if (mesh.geometry) {\n const indices = mesh.geometry.getIndices();\n const positions = mesh.geometry.getVerticesData(VertexBuffer.PositionKind);\n const v1 = new Vector3();\n const v2 = new Vector3();\n const v3 = new Vector3();\n const v4 = new Vector3();\n if (indices && positions) {\n const facesByEdges = {};\n /** Sets of three faces by index adjacent to each face by index */\n const faceNeighbors = [];\n /** Sets of xyz scalars making up the face normals */\n const faceNormals = [];\n const facePositions = [];\n const trianglesCount = indices.length / 3;\n for (let triId = 0; triId < trianglesCount; triId++) {\n const triOffset = triId * 3;\n // Get the triangle position vertices\n v1.copyFromFloats(positions[indices[triOffset] * 3], positions[indices[triOffset] * 3 + 1], positions[indices[triOffset] * 3 + 2]);\n v2.copyFromFloats(positions[indices[triOffset + 1] * 3], positions[indices[triOffset + 1] * 3 + 1], positions[indices[triOffset + 1] * 3 + 2]);\n v3.copyFromFloats(positions[indices[triOffset + 2] * 3], positions[indices[triOffset + 2] * 3 + 1], positions[indices[triOffset + 2] * 3 + 2]);\n // Calculate the centroid of each face\n v4.setAll(0);\n v4.addInPlace(v1).addInPlace(v2).addInPlace(v3);\n v4.scaleInPlace(1 / 3);\n facePositions.push(v4.x, v4.y, v4.z);\n // Get two of the triangle edges\n v1.subtractInPlace(v2);\n v2.subtractInPlace(v3);\n // Get the triangle face normal with the two edges.\n // Note that this is not the geometry normal, but the position-based face normal\n Vector3.CrossToRef(v1, v2, v4);\n const faceNormal = v4;\n faceNormal.normalize();\n faceNormals.push(faceNormal.x, faceNormal.y, faceNormal.z);\n // Now we need to consolidate what triangles are next to each other by organizing them by edges\n for (let i = 0; i < 3; i++) {\n const idx1 = indices[triOffset + i];\n const idx2 = indices[triOffset + ((i + 1) % 3)];\n // This is not going to work for geometry seams like on cylinders.\n // Possibly change from index ids to a representation of the vertex position\n const edgeId = idx1 > idx2 ? `${idx1}-${idx2}` : `${idx2}-${idx1}`;\n if (!facesByEdges[edgeId]) {\n facesByEdges[edgeId] = [];\n }\n facesByEdges[edgeId].push(triId);\n }\n }\n // Now we go through the edges and register the neighbors for each triangle\n for (const edgeId in facesByEdges) {\n const faces = facesByEdges[edgeId];\n if (!faceNeighbors[faces[0]]) {\n faceNeighbors[faces[0]] = [];\n }\n // Some edges may only have one face if the face is on the edge of the geometry\n if (faces.length > 1) {\n if (!faceNeighbors[faces[1]]) {\n faceNeighbors[faces[1]] = [];\n }\n faceNeighbors[faces[0]].push(faces[1]);\n faceNeighbors[faces[1]].push(faces[0]);\n }\n }\n const result = {\n faceCount: trianglesCount,\n faceNeighbors,\n facePositions,\n faceNormals,\n };\n meshFaceInfo.set(mesh.id, result);\n return result;\n }\n }\n }\n }\n\n (function (eMarqueeSelectMode) {\n eMarqueeSelectMode[eMarqueeSelectMode[\"Facets\"] = 0] = \"Facets\";\n eMarqueeSelectMode[eMarqueeSelectMode[\"Vertices\"] = 1] = \"Vertices\";\n })(exports.eMarqueeSelectMode || (exports.eMarqueeSelectMode = {}));\n class Interaction {\n constructor(kbViewer, scene, highlighter) {\n this.kbViewer = kbViewer;\n this.scene = scene;\n this.highlighter = highlighter;\n this.onClick$ = new Subject();\n this.onFacetClick$ = new Subject();\n this.onDoubleClick$ = new Subject();\n this.onPointerMove$ = new Subject();\n this.onPointerDown$ = new Subject();\n this.onPointerUp$ = new Subject();\n this.onKeyboard$ = new Subject();\n this.onMarqueeSelect = null;\n this.pointerIsDown = false;\n this.dragInProgress = false;\n this.selectionTolerance = 0;\n this.cameraFacingDirection = new Vector3();\n this.meshFaceEngine = new MeshFaceEngine(kbViewer);\n this.onClick = this.onClick$.asObservable();\n this.onFacetClick = this.onFacetClick$.asObservable();\n this.onDoubleClick = this.onDoubleClick$.asObservable();\n this.onPointerMove = this.onPointerMove$.asObservable();\n this.onPointerDown = this.onPointerDown$.asObservable();\n this.onPointerUp = this.onPointerUp$.asObservable();\n this.onKeyboard = this.onKeyboard$.asObservable();\n this.attachScene(scene);\n }\n attachScene(scene, utilityLayer = false) {\n scene.onPointerObservable.add(p => {\n const args = this.getKb3dPointerArgs(p);\n switch (p.type) {\n case PointerEventTypes.POINTERDOWN:\n this.handlePointerDown(args);\n break;\n case PointerEventTypes.POINTERUP:\n this.handlePointerUp(args);\n break;\n case PointerEventTypes.POINTERMOVE:\n this.handleMove(args);\n break;\n // case PointerEventTypes.POINTERWHEEL:\n // console.log(\"POINTER WHEEL\");\n // break;\n // case PointerEventTypes.POINTERPICK:\n // console.log(\"POINTER PICK\");\n // break;\n case PointerEventTypes.POINTERTAP:\n this.handleClick(args);\n break;\n case PointerEventTypes.POINTERDOUBLETAP:\n this.handleDoubleClick(args);\n break;\n }\n });\n scene.onKeyboardObservable.add(k => {\n const args = this.getKeyboardArgs(k);\n switch (k.type) {\n case KeyboardEventTypes.KEYDOWN:\n this.handleKeyboard(args);\n }\n });\n }\n setSelectionTolerance(tolerance) {\n this.selectionTolerance = tolerance;\n }\n setSelectionCulling(enabled) {\n if (this.marqueeSelectOptions) {\n this.marqueeSelectOptions.cullSelection = enabled;\n }\n }\n enableMarqueeSelect(meshNode, mode, options) {\n const mesh = this.kbViewer.getRendererNode(meshNode);\n if (mesh instanceof Mesh) {\n this.onMarqueeSelect = new Subject();\n this.marqueeSelectOptions = {\n cullSelection: false,\n onlyOnShiftPress: true,\n startX: 0,\n startY: 0,\n mode,\n mesh,\n ...(options || {}),\n };\n return this.onMarqueeSelect.asObservable().pipe(distinctUntilChanged((prev, curr) => {\n if (prev.selectedFaces.length !== curr.selectedFaces.length ||\n prev.vertexList.length !== curr.vertexList.length) {\n return false;\n }\n for (let i = 0; i < curr.selectedFaces.length; i++) {\n if (curr.selectedFaces[i] !== prev.selectedFaces[i]) {\n return false;\n }\n }\n for (let i = 0; i < curr.vertexList.length; i++) {\n if (curr.vertexList[i] !== prev.vertexList[i]) {\n return false;\n }\n }\n return true;\n }));\n }\n return EMPTY;\n }\n disableMarqueeSelect() {\n if (this.onMarqueeSelect) {\n this.onMarqueeSelect.complete();\n }\n this.onMarqueeSelect = null;\n }\n /** Do a hit test of the scene using x,y screen pixel coordinates */\n pick(x, y, camera) {\n const pickInfo = this.scene.pick(x, y, undefined, undefined, camera);\n if (pickInfo && pickInfo.hit) {\n const info = { ...pickInfo };\n this.fillKb3dPickInfo(pickInfo, info);\n return info;\n }\n return undefined;\n }\n handleClickTypeEvent(args, handlerType) {\n // For debugging\n // if (args.pickInfo?.ray) {\n // RayHelper.CreateAndShow(args.pickInfo.ray, this.scene, new Color3(1, 1, 0.1));\n // }\n if (args.targetNode) {\n //bubble up event, calling any handlers that have been applied to the nodes\n let h = this.getNextNodeHandlers(args.targetNode, handlerType);\n while (h) {\n let propagate = false;\n args.sourceNode = h.node;\n for (const handler of h.handlers) {\n handler(args);\n if (args.propagate)\n propagate = true;\n }\n if (!propagate)\n break;\n h = this.getNextNodeHandlers(h.node._parent, handlerType);\n }\n }\n else {\n //there was no clicked mesh, but if there is a handler on the root scene node, we trigger that\n const handlers = this.getHandlerMap(this.kbViewer.sceneNode, handlerType).getByKey(this.kbViewer.sceneNode);\n if (handlers) {\n args.targetNode = args.sourceNode = this.kbViewer.sceneNode;\n for (const handler of handlers) {\n handler(args);\n }\n }\n }\n }\n handleClick(args) {\n var _a;\n this.handleClickTypeEvent(args, exports.eHandler.click);\n // trigger the normal generic observable\n this.onClick$.next(args);\n // if there is an observer on the facet click, then send the facet data there\n if (this.onFacetClick$.observers.length > 0) {\n if (((_a = args.pickInfo) === null || _a === void 0 ? void 0 : _a.pickedMesh) && args.pickInfo.pickedMesh instanceof Mesh) {\n const pickedMesh = args.pickInfo.pickedMesh;\n const selectedTriangle = args.pickInfo.faceId;\n const faceInfo = this.meshFaceEngine.getMeshFaces(pickedMesh);\n if (faceInfo) {\n let selection = [];\n if (this.selectionTolerance >= 0) {\n selection = this.meshFaceEngine.expandSelection([selectedTriangle], pickedMesh, this.selectionTolerance);\n }\n else {\n selection = [selectedTriangle];\n }\n const facetArgs = {\n ...args,\n altKey: args.event.altKey,\n selectedFaces: selection,\n };\n this.onFacetClick$.next(facetArgs);\n }\n }\n }\n }\n handleDoubleClick(args) {\n this.handleClickTypeEvent(args, exports.eHandler.doubleClick);\n //trigger the normal generic observable\n this.onDoubleClick$.next(args);\n }\n handleMove(args) {\n if (!this.pointerIsDown && this.kbViewer.sceneNode && this.kbViewer.sceneNode.enableHoverEffects) {\n if (args.targetNode) {\n const clicks = this.getNextClickHandlers(args.targetNode);\n const draggables = this.getNextDraggables(args.targetNode);\n if (clicks || draggables) {\n const n = (clicks === null || clicks === void 0 ? void 0 : clicks.node) || (draggables === null || draggables === void 0 ? void 0 : draggables.node);\n this.highlighter.hover(n);\n }\n }\n else {\n this.highlighter.clearHover();\n }\n }\n if (this.scene.activeCamera instanceof KbArcRotateCamera && this.pointerIsDown && this.onMarqueeSelect) {\n if (!this.marqueeSelectOptions.onlyOnShiftPress || args.shiftKey) {\n args.event.preventDefault();\n args.event.stopPropagation();\n this.handleMarquee(args, this.scene.activeCamera);\n }\n else {\n this.unlockCamera();\n }\n }\n this.onPointerMove$.next(args);\n }\n handlePointerDown(args) {\n var _a;\n this.pointerIsDown = true;\n if (this.onMarqueeSelect) {\n if (!this.marqueeSelectOptions.onlyOnShiftPress || args.shiftKey) {\n this.marqueeSelectOptions.startX = args.canvasX || 0;\n this.marqueeSelectOptions.startY = args.canvasY || 0;\n this.marqueeSelectOptions.startRay =\n ((_a = this.scene.pick(args.canvasX, args.canvasY, undefined, undefined, this.scene.activeCamera)) === null || _a === void 0 ? void 0 : _a.ray) ||\n undefined;\n if (this.kbViewer.uiOverlay) {\n const selectBox = document.createElement('div');\n selectBox.style.position = 'absolute';\n selectBox.style.border = '1px dashed white';\n selectBox.style.background = 'rgba(128, 128, 128, 0.1)';\n selectBox.style.opacity = '0.5';\n this.kbViewer.uiOverlay.appendChild(selectBox);\n this.marqueeSelectOptions.selectBox = selectBox;\n }\n this.cameraFacingDirection = this.scene.activeCamera.getForwardRay().direction;\n if (!this.marqueeSelectOptions.mesh.isFacetDataEnabled &&\n this.marqueeSelectOptions.mode === exports.eMarqueeSelectMode.Facets) {\n const normals = this.marqueeSelectOptions.mesh.getVerticesData(VertexBuffer.NormalKind);\n if (!normals) {\n this.marqueeSelectOptions.mesh.setVerticesData(VertexBuffer.NormalKind, new Float32Array(this.marqueeSelectOptions.mesh.getTotalVertices() * 3));\n }\n }\n }\n }\n else if (args.targetNode) {\n const d = this.getNextDraggables(args.targetNode);\n if (d) {\n this.highlighter.clearHover();\n args.sourceNode = d.node;\n const dragger = new Dragger(this.kbViewer, this.scene);\n this.dragInProgress = true;\n dragger.startDrag(args, d.handlers[0], () => {\n this.dragInProgress = false;\n });\n }\n }\n this.onPointerDown$.next(args);\n }\n unlockCamera() {\n var _a;\n (_a = this.scene.activeCamera) === null || _a === void 0 ? void 0 : _a.attachControl(this.kbViewer.canvas);\n }\n handlePointerUp(args) {\n var _a;\n this.pointerIsDown = false;\n if (this.marqueeSelectOptions && this.marqueeSelectOptions.selectBox) {\n (_a = this.kbViewer.uiOverlay) === null || _a === void 0 ? void 0 : _a.removeChild(this.marqueeSelectOptions.selectBox);\n delete this.marqueeSelectOptions.selectBox;\n }\n this.unlockCamera();\n this.onPointerUp$.next(args);\n }\n handleKeyboard(args) {\n this.onKeyboard$.next(args);\n }\n handleMarquee(args, activeCamera) {\n var _a;\n const mouseX = args.canvasX || 0;\n const mouseY = args.canvasY || 0;\n activeCamera.detachControl();\n if (this.marqueeSelectOptions.selectBox) {\n this.marqueeSelectOptions.selectBox.style.left = Math.min(this.marqueeSelectOptions.startX, mouseX) + 'px';\n this.marqueeSelectOptions.selectBox.style.top = Math.min(this.marqueeSelectOptions.startY, mouseY) + 'px';\n this.marqueeSelectOptions.selectBox.style.width =\n Math.abs(this.marqueeSelectOptions.startX - mouseX) + 'px';\n this.marqueeSelectOptions.selectBox.style.height =\n Math.abs(this.marqueeSelectOptions.startY - mouseY) + 'px';\n }\n let topOrigin;\n let leftOrigin;\n let rightOrigin;\n let bottomOrigin;\n let topNormal;\n let leftNormal;\n let rightNormal;\n let bottomNormal;\n const m = Matrix.Identity();\n const boxL = Math.min(this.marqueeSelectOptions.startX, mouseX);\n const boxR = Math.max(this.marqueeSelectOptions.startX, mouseX);\n const boxT = Math.min(this.marqueeSelectOptions.startY, mouseY);\n const boxB = Math.max(this.marqueeSelectOptions.startY, mouseY);\n if (activeCamera.mode === Camera.ORTHOGRAPHIC_CAMERA) {\n // ORTHOGRAPHIC CAMERA\n const forwardDirection = activeCamera.getForwardRay().direction;\n const sideDirection = Vector3.Cross(forwardDirection, Vector3.Up());\n topNormal = Vector3.Cross(forwardDirection, sideDirection);\n leftNormal = sideDirection;\n rightNormal = sideDirection.scale(-1);\n bottomNormal = Vector3.Cross(sideDirection, forwardDirection);\n topOrigin = this.scene.createPickingRay(boxL, boxT, m, activeCamera).origin;\n leftOrigin = this.scene.createPickingRay(boxL, boxT, m, activeCamera).origin;\n rightOrigin = this.scene.createPickingRay(boxR, boxB, m, activeCamera).origin;\n bottomOrigin = this.scene.createPickingRay(boxR, boxB, m, activeCamera).origin;\n }\n else {\n // PERSPECTIVE CAMERA\n // Assemble the 4 planes of the marquee select\n const origin = ((_a = this.marqueeSelectOptions.startRay) === null || _a === void 0 ? void 0 : _a.origin) || activeCamera.position;\n topOrigin = origin;\n leftOrigin = origin;\n rightOrigin = origin;\n bottomOrigin = origin;\n const topLeft = this.scene.createPickingRay(boxL, boxT, m, activeCamera).direction;\n const topRight = this.scene.createPickingRay(boxR, boxT, m, activeCamera).direction;\n const botLeft = this.scene.createPickingRay(boxL, boxB, m, activeCamera).direction;\n const botRight = this.scene.createPickingRay(boxR, boxB, m, activeCamera).direction;\n if (topLeft && topRight && botLeft && botRight) {\n // Normals of the planes are equal to the cross product of the two vectors that define the plane\n topNormal = Vector3.Cross(topLeft, topRight);\n leftNormal = Vector3.Cross(botLeft, topLeft);\n rightNormal = Vector3.Cross(topRight, botRight);\n bottomNormal = Vector3.Cross(botRight, botLeft);\n }\n }\n if (topNormal && leftNormal && rightNormal && bottomNormal) {\n const mesh = this.marqueeSelectOptions.mesh;\n const fP = new Vector3();\n const fN = new Vector3();\n if (this.marqueeSelectOptions.mode === exports.eMarqueeSelectMode.Facets) {\n const faceInfo = this.meshFaceEngine.getMeshFaces(mesh);\n const transform = mesh.computeWorldMatrix();\n const n = faceInfo.faceCount;\n const res = [];\n for (let i = 0; i < n; i++) {\n Vector3.TransformCoordinatesFromFloatsToRef(faceInfo.facePositions[i * 3], faceInfo.facePositions[i * 3 + 1], faceInfo.facePositions[i * 3 + 2], transform, fP);\n if (this.marqueeSelectOptions.cullSelection) {\n Vector3.TransformCoordinatesFromFloatsToRef(faceInfo.faceNormals[i * 3], faceInfo.faceNormals[i * 3 + 1], faceInfo.faceNormals[i * 3 + 2], transform, fN);\n }\n if ((!this.marqueeSelectOptions.cullSelection || Vector3.Dot(this.cameraFacingDirection, fN) > 0) &&\n pointPlaneSizeToRef(topNormal, topOrigin, fP) >= 0 &&\n pointPlaneSizeToRef(leftNormal, leftOrigin, fP) >= 0 &&\n pointPlaneSizeToRef(rightNormal, rightOrigin, fP) >= 0 &&\n pointPlaneSizeToRef(bottomNormal, bottomOrigin, fP) >= 0) {\n res.push(i);\n }\n }\n this.onMarqueeSelect.next({\n ...args,\n altKey: args.event.altKey,\n vertexList: [],\n selectedFaces: res,\n });\n }\n else if (this.marqueeSelectOptions.mode === exports.eMarqueeSelectMode.Vertices) {\n const vertexPositions = mesh.getVerticesData(VertexBuffer.PositionKind);\n const transform = mesh.getWorldMatrix();\n const res = [];\n if (vertexPositions) {\n const n = vertexPositions.length / 3;\n for (let i = 0; i < n; i++) {\n fP.copyFromFloats(vertexPositions[i * 3], vertexPositions[i * 3 + 1], vertexPositions[i * 3 + 2]);\n Vector3.TransformCoordinatesToRef(fP, transform, fP);\n if ((!this.marqueeSelectOptions.cullSelection ||\n Vector3.Dot(this.cameraFacingDirection, mesh.getFacetNormal(i))) &&\n pointPlaneSizeToRef(topNormal, topOrigin, fP) >= 0 &&\n pointPlaneSizeToRef(leftNormal, leftOrigin, fP) >= 0 &&\n pointPlaneSizeToRef(rightNormal, rightOrigin, fP) >= 0 &&\n pointPlaneSizeToRef(bottomNormal, bottomOrigin, fP) >= 0) {\n res.push(i);\n }\n }\n mesh.getIndices();\n this.onMarqueeSelect.next({\n ...args,\n altKey: args.event.altKey,\n vertexList: res,\n selectedFaces: [],\n });\n }\n }\n }\n function pointPlaneSizeToRef(normal, origin, point) {\n return normal.x * (point.x - origin.x) + normal.y * (point.y - origin.y) + normal.z * (point.z - origin.z);\n }\n }\n getKb3dPointerArgs(p) {\n const args = {\n ...p,\n pickInfo: p.pickInfo,\n propagate: false,\n pointerType: p.event.pointerType,\n button: p.event.button,\n ctrlKey: p.event.ctrlKey || p.event.metaKey,\n shiftKey: p.event.shiftKey,\n canvasX: p.event.offsetX,\n canvasY: p.event.offsetY,\n getNormal: () => {\n if (p.pickInfo) {\n const normal = p.pickInfo.getNormal();\n return normal ? exports.KbVector.FromVec3(normal) : undefined;\n }\n return undefined;\n },\n };\n if (p.pickInfo) {\n this.fillKb3dPickInfo(p.pickInfo, args);\n }\n return args;\n }\n fillKb3dPickInfo(b, k) {\n k.u = b.bu;\n k.v = b.bv;\n if (b.pickedPoint)\n k.pickedPoint = exports.KbVector.FromVec3(b.pickedPoint);\n if (b.pickedMesh) {\n // recurse up the tree until we find something, or return undefined\n let pickedMesh = b.pickedMesh;\n k.targetNode = getByDynamicId(pickedMesh.id);\n while (!k.targetNode && pickedMesh.parent) {\n pickedMesh = pickedMesh.parent;\n k.targetNode = getByDynamicId(pickedMesh.id);\n }\n if (b.pickedPoint) {\n k.meshPoint = exports.KbVector.FromVec3(Vector3.TransformCoordinates(b.pickedPoint, b.pickedMesh.getWorldMatrix().clone().invert()));\n }\n }\n }\n getKeyboardArgs(k) {\n const event = k.event;\n const args = {\n ...k,\n key: event.key,\n keyCode: event.keyCode,\n altKey: event.altKey,\n ctrlKey: event.ctrlKey || event.metaKey,\n shiftKey: event.shiftKey,\n repeat: event.repeat,\n };\n return args;\n }\n /**\n * scans the given node and ancestors to find the first node with a click handler and returns\n * the node and it's handlers\n */\n getNextNodeHandlers(node, handlerType) {\n //bubble up event, calling any handlers that have been applied to the nodes\n let handlers = new Array();\n while (node && node._manager) {\n const hmap = this.getHandlerMap(node, handlerType);\n handlers = hmap.getByKey(node);\n //if this is a scene node, then we also have to look in the parent scene manager for click events on this item\n if (instanceOf(node, Token.SceneNode) && node.isNested() && node._parent && node._parent._manager) {\n handlers.push(...this.getHandlerMap(node._parent, handlerType).getByKey(node));\n }\n if (handlers && handlers.length) {\n break;\n }\n node = node._parent;\n }\n return node ? { node, handlers } : undefined;\n }\n getHandlerMap(node, type) {\n const m = node._manager;\n if (type == exports.eHandler.click)\n return m.clickHandlers;\n else if (type == exports.eHandler.draggable)\n return m.draggables;\n else if (type == exports.eHandler.doubleClick)\n return m.doubleClickHandlers;\n else\n throw 'unrecognized handler type';\n }\n getNextClickHandlers(node) {\n return this.getNextNodeHandlers(node, exports.eHandler.click);\n }\n getNextDoubleClickHandlers(node) {\n return this.getNextNodeHandlers(node, exports.eHandler.doubleClick);\n }\n getNextDraggables(node) {\n return this.getNextNodeHandlers(node, exports.eHandler.draggable);\n }\n }\n\n const dbSubmeshes = new Map();\n /**\n * Gizmo that enables moving vertices on a mesh.\n */\n class KbFacetSelectGizmo extends KbGizmo {\n get attachedMesh() {\n return this.__attachedMesh;\n }\n set attachedMesh(mesh) {\n if (mesh && !(mesh instanceof Mesh)) {\n throw Error('facet select gizmo must be attached to a mesh with geometry');\n }\n this.__attachedMesh = mesh;\n }\n /**\n * Creates a PositionGizmo\n * @param camera The camera being used for the scene. Used for positioning the particles so they line up with the vertices\n * @param gizmoLayer The utility layer the gizmo will be added to\n */\n constructor(kbViewer, scene) {\n super(kbViewer.utilityLayer);\n this.kbViewer = kbViewer;\n this.scene = scene;\n /** Fires an event when any of it's sub gizmos are dragged */\n this.onSelectionChange = new Observable$1();\n this.__attachedMesh = null;\n this.currentSelection = new Map();\n this.selectionsEnabled = false;\n this.highlightOpacity = 0.75;\n this.meshFaceEngine = new MeshFaceEngine(kbViewer);\n this.utilLayer = kbViewer.utilityLayer;\n this.attachedMesh = null;\n }\n attachTo(target) {\n this.__attachedNode = target;\n const aMesh = this.kbViewer.getRendererNode(target);\n if (!aMesh) {\n throw Error('Could not find mesh in scene to attach gizmo');\n }\n if (!(aMesh instanceof Mesh)) {\n throw Error('facet select gizmo must be attached to a mesh with geometry');\n }\n const mesh = aMesh;\n this.attachedMesh = mesh;\n // Clean up old marquee\n if (this.disposeSubject) {\n this.disposeSubject.next();\n this.disposeSubject.complete();\n }\n this.disposeSubject = new Subject();\n this.kbViewer.interaction.onFacetClick.pipe(takeUntil(this.disposeSubject)).subscribe(event => {\n if (!this.selectionsEnabled) {\n return;\n }\n event.selectedFaces.sort();\n if (event.ctrlKey) {\n // Add selections\n for (let i = 0; i < event.selectedFaces.length; i++) {\n this.currentSelection.set(event.selectedFaces[i], null);\n }\n }\n else if (event.altKey) {\n // Remove selections\n for (let i = 0; i < event.selectedFaces.length; i++) {\n this.currentSelection.delete(event.selectedFaces[i]);\n }\n }\n else {\n this.currentSelection.clear();\n for (let i = 0; i < event.selectedFaces.length; i++) {\n this.currentSelection.set(event.selectedFaces[i], null);\n }\n }\n const result = this.getVertexGroups(this.currentSelection.keys());\n this.visualizeSubmesh(mesh, result, exports.KbColor.Red());\n this.onSelectionChange.notifyObservers({ selectedFacets: result });\n });\n }\n updateGizmoPosition() {\n this.kbViewer.forceAndWaitForNextRender().then(() => {\n if (this.__attachedMesh) {\n const result = this.getVertexGroups(this.currentSelection.keys());\n this.visualizeSubmesh(this.__attachedMesh, result, exports.KbColor.Red(), true);\n }\n });\n }\n selectFacetGroups(selection) {\n selection.forEach(vertexGroup => {\n if (vertexGroup.startIndex === undefined || vertexGroup.endIndex === undefined) {\n throw Error('corrupt vertex group');\n }\n const startFacet = Math.round(vertexGroup.startIndex / 3);\n const endFacet = Math.ceil(vertexGroup.endIndex / 3);\n for (let i = startFacet; i < endFacet; i++) {\n this.currentSelection.set(i, null);\n }\n });\n this.updateGizmoPosition();\n }\n enableSelectionChange() {\n this.selectionsEnabled = true;\n if (this.__attachedNode) {\n this.kbViewer.interaction\n .enableMarqueeSelect(this.__attachedNode, exports.eMarqueeSelectMode.Facets)\n .pipe(takeUntil(this.disposeSubject))\n .subscribe(event => {\n if (this.selectionsEnabled &&\n this.__attachedMesh &&\n event.selectedFaces &&\n event.selectedFaces.length > 0) {\n if (event.ctrlKey) {\n // Add selections\n for (let i = 0; i < event.selectedFaces.length; i++) {\n this.currentSelection.set(event.selectedFaces[i], null);\n }\n }\n else if (event.altKey) {\n // Remove selections\n for (let i = 0; i < event.selectedFaces.length; i++) {\n this.currentSelection.delete(event.selectedFaces[i]);\n }\n }\n else {\n this.currentSelection.clear();\n for (let i = 0; i < event.selectedFaces.length; i++) {\n this.currentSelection.set(event.selectedFaces[i], null);\n }\n }\n const result = this.getVertexGroups(this.currentSelection.keys());\n this.visualizeSubmesh(this.__attachedMesh, result, exports.KbColor.Red());\n this.onSelectionChange.notifyObservers({ selectedFacets: result });\n }\n });\n }\n }\n disableSelectionChange() {\n this.selectionsEnabled = false;\n this.kbViewer.interaction.disableMarqueeSelect();\n }\n /**\n * Disposes of the gizmo\n */\n dispose() {\n if (this.__attachedNode) {\n this.kbViewer.highlighter.remove(this.__attachedNode);\n }\n if (this.__attachedMesh) {\n const submeshInfo = dbSubmeshes.get(this.__attachedMesh.id);\n submeshInfo === null || submeshInfo === void 0 ? void 0 : submeshInfo.visualizationMesh.setEnabled(false);\n }\n }\n /**\n * CustomMeshes are not supported by this gizmo\n * @param mesh The mesh to replace the default mesh of the gizmo\n */\n setCustomMesh(mesh) {\n Logger.Error('Custom meshes are not supported on this gizmo.');\n }\n setBackfaceCulling(enable) {\n this.kbViewer.interaction.setSelectionCulling(enable);\n }\n setTolerance(tolerance) {\n this.kbViewer.interaction.setSelectionTolerance(tolerance);\n }\n expandSelection(tolerance) {\n if (this.attachedMesh instanceof Mesh) {\n const currentSelection = Array.from(this.currentSelection.keys());\n const newSelection = this.meshFaceEngine.expandSelection(currentSelection, this.attachedMesh, tolerance);\n for (let i = 0; i < newSelection.length; i++) {\n this.currentSelection.set(newSelection[i], null);\n }\n const result = this.getVertexGroups(this.currentSelection.keys());\n this.visualizeSubmesh(this.attachedMesh, result, exports.KbColor.Red());\n this.onSelectionChange.notifyObservers({ selectedFacets: result });\n }\n }\n setVisualization(opacity) {\n const submeshInfo = dbSubmeshes.get(this.attachedMesh.id);\n if (submeshInfo && submeshInfo.visualizationMesh.material instanceof MultiMaterial) {\n submeshInfo.highlightMaterial.alpha = opacity;\n }\n this.highlightOpacity = opacity;\n }\n getVertexGroups(faceIds) {\n const result = [];\n for (const faceId of faceIds) {\n if (result.length > 0 && result[result.length - 1].endIndex === faceId * 3) {\n result[result.length - 1].endIndex += 3;\n }\n else {\n result.push({\n startIndex: faceId * 3,\n endIndex: (faceId + 1) * 3,\n });\n }\n }\n return result;\n }\n visualizeSubmesh(node, vertices, color, geometryNeedsUpdate) {\n if (node instanceof Mesh && node.geometry) {\n let submeshInfo = dbSubmeshes.get(node.id);\n let highlightMaterial;\n const utilityLayerScene = this.utilLayer.utilityLayerScene;\n if (submeshInfo) {\n highlightMaterial = submeshInfo.highlightMaterial;\n }\n if (submeshInfo && geometryNeedsUpdate) {\n submeshInfo.visualizationMesh.dispose();\n submeshInfo = undefined;\n }\n if (!submeshInfo) {\n const multiMaterial = new MultiMaterial(`submeshVisualizationMat_${node.id}`, utilityLayerScene);\n const transparantMaterial = new StandardMaterial(`submeshVisualizationMat1_${node.id}`, utilityLayerScene);\n transparantMaterial.alpha = 0;\n multiMaterial.subMaterials.push(transparantMaterial);\n if (!highlightMaterial) {\n highlightMaterial = new StandardMaterial(`submeshVisualizationMat2_${node.id}`, utilityLayerScene);\n highlightMaterial.emissiveColor = (color === null || color === void 0 ? void 0 : color.toColor3()) || exports.KbColor.KbmaxOrange().toColor3();\n highlightMaterial.alpha = this.highlightOpacity;\n }\n multiMaterial.subMaterials.push(highlightMaterial);\n submeshInfo = {\n visualizationMesh: new Mesh(`submeshVisualization_${node.id}`, utilityLayerScene, null, node, true),\n highlightMaterial,\n vertices,\n };\n submeshInfo.visualizationMesh.makeGeometryUnique();\n submeshInfo.visualizationMesh.material = multiMaterial;\n submeshInfo.visualizationMesh.isPickable = false;\n dbSubmeshes.set(node.id, submeshInfo);\n }\n else {\n submeshInfo.visualizationMesh.setEnabled(true);\n }\n const vMesh = submeshInfo.visualizationMesh;\n const totalIndices = node.geometry.getTotalIndices();\n const totalVertices = node.geometry.getTotalVertices();\n vMesh.subMeshes = [];\n new SubMesh(0, 0, totalVertices, 0, totalIndices, vMesh);\n vertices.forEach(indexGroup => {\n if (indexGroup.startIndex === undefined || indexGroup.endIndex === undefined) {\n throw Error('Missing submesh information');\n }\n new SubMesh(1, 0, totalVertices, indexGroup.startIndex, indexGroup.endIndex - indexGroup.startIndex, vMesh);\n });\n }\n }\n }\n\n /**\n * Gizmo that enables dragging and rotating lights\n */\n class KbLightGizmo extends KbGizmo {\n attachTo(lightNode) {\n const light = this.kbViewer.getRendererNode(lightNode);\n if (!light) {\n console.warn('could not find light in scene');\n return;\n }\n this._attachedLight = lightNode;\n if (this.posGizmo) {\n this.posGizmo.dispose();\n }\n if (this.rotGizmo) {\n this.rotGizmo.dispose();\n }\n const _helperMesh = (this._helperMesh = new AbstractMesh('', this.gizmoLayer.utilityLayerScene));\n _helperMesh.rotationQuaternion = Quaternion.Identity();\n const lightVisual = this.updatePreviewMesh(light);\n _helperMesh.addChild(lightVisual);\n }\n /**\n * Creates a PositionGizmo\n * @param gizmoLayer The utility layer the gizmo will be added to\n */\n constructor(kbViewer, gizmoLayer = UtilityLayerRenderer.DefaultUtilityLayer) {\n super(gizmoLayer);\n this.kbViewer = kbViewer;\n this.attachedMesh = null;\n }\n /**\n * Disposes of the gizmo\n */\n dispose() {\n if (this.posGizmo) {\n this.posGizmo.dispose();\n }\n if (this.rotGizmo) {\n this.rotGizmo.dispose();\n }\n if (this._helperMesh) {\n this._helperMesh.dispose();\n }\n if (this.lightPositionGizmo) {\n this.lightPositionGizmo.dispose();\n }\n delete this._attachedLight;\n }\n updateGizmoPosition(lightNode) {\n if (!lightNode) {\n return;\n }\n const light = this.kbViewer.getRendererNode(lightNode);\n if (light) {\n if (!this._attachedLight || lightNode._dynamicId !== this._attachedLight._dynamicId) {\n this.attachTo(lightNode);\n }\n else {\n this.updatePreviewMesh(light);\n }\n }\n }\n updatePreviewMesh(light) {\n const scene = this.gizmoLayer.utilityLayerScene;\n if (!this.lightPositionGizmo) {\n this.lightPositionGizmo = createLightVisualization(light, scene);\n }\n if (light instanceof SpotLight || light instanceof PointLight) {\n this.lightPositionGizmo.position = light.parent\n ? Vector3.TransformCoordinates(light.position, light.parent.computeWorldMatrix(true).clone().invert())\n : light.position;\n }\n return this.lightPositionGizmo;\n }\n }\n\n /**\n * Single axis drag gizmo\n */\n class KbAxisDragGizmo extends KbGizmo {\n /** @hidden */\n static _CreateArrow(scene, material) {\n const arrow = new TransformNode('arrow', scene);\n const cylinder = CylinderBuilder.CreateCylinder('cylinder', { diameterTop: 0, height: 0.075, diameterBottom: 0.0375, tessellation: 12 }, scene);\n const line = CylinderBuilder.CreateCylinder('cylinder', { diameterTop: 0.01, height: 0.275, diameterBottom: 0.01, tessellation: 6 }, scene);\n const drag = CylinderBuilder.CreateCylinder('cylinder', { diameterTop: 0.06, height: 0.4, diameterBottom: 0.06, tessellation: 6 }, scene);\n line.material = material;\n cylinder.parent = arrow;\n line.parent = arrow;\n drag.parent = arrow;\n // Position arrow pointing in its drag axis\n cylinder.material = material;\n cylinder.rotation.x = Math.PI / 2;\n cylinder.position.z += 0.3;\n line.position.z += 0.275 / 2;\n line.rotation.x = Math.PI / 2;\n drag.position.z += 0.4 / 2;\n drag.rotation = line.rotation;\n drag.visibility = 0;\n return arrow;\n }\n /** @hidden */\n static _CreateArrowInstance(scene, arrow) {\n const instance = new TransformNode('arrow', scene);\n for (const mesh of arrow.getChildMeshes()) {\n const childInstance = mesh.createInstance(mesh.name);\n childInstance.parent = instance;\n }\n return instance;\n }\n /**\n * Creates an AxisDragGizmo\n * @param gizmoLayer The utility layer the gizmo will be added to\n * @param dragAxis The axis which the gizmo will be able to drag on\n * @param color The color of the gizmo\n */\n constructor(dragAxis, color = Color3.Gray(), gizmoLayer = UtilityLayerRenderer.DefaultUtilityLayer) {\n super(gizmoLayer);\n this._pointerObserver = null;\n /**\n * Drag distance in babylon units that the gizmo will snap to when dragged (Default: 0)\n */\n this.snapDistance = 0;\n /**\n * Event that fires each time the gizmo snaps to a new location.\n * * snapDistance is the the change in distance\n */\n this.onSnapObservable = new Observable$1();\n this.updateObservable = new Observable$1();\n // Create Material\n const coloredMaterial = new StandardMaterial('', gizmoLayer.utilityLayerScene);\n coloredMaterial.emissiveColor = color;\n // coloredMaterial.diffuseColor = color;\n // coloredMaterial.specularColor = color.subtract(new Color3(0.1, 0.1, 0.1));\n const hoverMaterial = new StandardMaterial('', gizmoLayer.utilityLayerScene);\n hoverMaterial.emissiveColor = color.add(new Color3(0.3, 0.3, 0.3));\n // hoverMaterial.diffuseColor = color.add(new Color3(0.3, 0.3, 0.3));\n // Build mesh on root node\n const arrow = KbAxisDragGizmo._CreateArrow(gizmoLayer.utilityLayerScene, coloredMaterial);\n arrow.lookAt(this._rootMesh.position.add(dragAxis));\n arrow.scaling.scaleInPlace(1 / 3);\n arrow.parent = this._rootMesh;\n let currentSnapDragDistance = 0;\n const tmpVector = new Vector3();\n const tmpSnapEvent = { snapDistance: 0 };\n // Add drag behavior to handle events when the gizmo is dragged\n this.dragBehavior = new PointerDragBehavior({ dragAxis: dragAxis });\n this.dragBehavior.moveAttached = false;\n this._rootMesh.addBehavior(this.dragBehavior);\n const localDelta = new Vector3();\n const tmpMatrix = new Matrix();\n this.dragBehavior.onDragObservable.add(event => {\n if (this.attachedMesh) {\n // Convert delta to local translation if it has a parent\n if (this.attachedMesh.parent) {\n this.attachedMesh.parent.computeWorldMatrix().invertToRef(tmpMatrix);\n tmpMatrix.setTranslationFromFloats(0, 0, 0);\n Vector3.TransformCoordinatesToRef(event.delta, tmpMatrix, localDelta);\n }\n else {\n localDelta.copyFrom(event.delta);\n }\n // Snapping logic\n if (this.snapDistance == 0) {\n this.attachedMesh.position.addInPlace(localDelta);\n this.updateObservable.notifyObservers({ delta: localDelta });\n }\n else {\n currentSnapDragDistance += event.dragDistance;\n if (Math.abs(currentSnapDragDistance) > this.snapDistance) {\n const dragSteps = Math.floor(Math.abs(currentSnapDragDistance) / this.snapDistance);\n currentSnapDragDistance = currentSnapDragDistance % this.snapDistance;\n localDelta.normalizeToRef(tmpVector);\n tmpVector.scaleInPlace(this.snapDistance * dragSteps);\n this.attachedMesh.position.addInPlace(tmpVector);\n tmpSnapEvent.snapDistance = this.snapDistance * dragSteps;\n this.updateObservable.notifyObservers({ delta: tmpVector });\n this.onSnapObservable.notifyObservers(tmpSnapEvent);\n }\n }\n }\n });\n this._pointerObserver = gizmoLayer.utilityLayerScene.onPointerObservable.add(pointerInfo => {\n if (this._customMeshSet) {\n return;\n }\n const isHovered = pointerInfo.pickInfo &&\n this._rootMesh.getChildMeshes().indexOf(pointerInfo.pickInfo.pickedMesh) != -1;\n const material = isHovered ? hoverMaterial : coloredMaterial;\n this._rootMesh.getChildMeshes().forEach(m => {\n m.material = material;\n if (m.color) {\n m.color = material.diffuseColor;\n }\n });\n });\n }\n _attachedMeshChanged(value) {\n if (this.dragBehavior) {\n this.dragBehavior.enabled = value ? true : false;\n }\n }\n /**\n * Disposes of the gizmo\n */\n dispose() {\n this.onSnapObservable.clear();\n this.gizmoLayer.utilityLayerScene.onPointerObservable.remove(this._pointerObserver);\n this.dragBehavior.detach();\n super.dispose();\n }\n }\n\n /**\n * Gizmo that enables dragging a mesh along 3 axis\n */\n class KbPositionGizmo extends KbGizmo {\n attachTo(target, positionSpace, orientGizmo) {\n if (orientGizmo) {\n this.xGizmo.updateGizmoRotationToMatchAttachedMesh = false;\n this.yGizmo.updateGizmoRotationToMatchAttachedMesh = false;\n this.zGizmo.updateGizmoRotationToMatchAttachedMesh = false;\n this.xGizmo._rootMesh.rotationQuaternion = orientGizmo;\n this.yGizmo._rootMesh.rotationQuaternion = orientGizmo;\n this.zGizmo._rootMesh.rotationQuaternion = orientGizmo;\n }\n if (!positionSpace && target instanceof exports.SpaceNode && target._parent) {\n positionSpace = target._parent;\n }\n if (positionSpace) {\n this._positionSpace = this.kbViewer.getRendererNode(positionSpace);\n }\n if (target instanceof exports.SpaceNode) {\n this._attachedSpaceNode = target;\n this._updateGizmoPosition();\n }\n else {\n this._updateGizmoPosition(target);\n }\n }\n updateGizmoPosition(changeTarget) {\n // Need to wait for the next frame because this can be called before the babylon object is actually updated\n this.kbViewer.onNextFrame(() => {\n this._updateGizmoPosition(changeTarget);\n });\n }\n _updateGizmoPosition(changeTarget) {\n var _a;\n const positionSpaceTransform = (_a = this._positionSpace) === null || _a === void 0 ? void 0 : _a.computeWorldMatrix(true);\n if (changeTarget) {\n this.attachMeshAtNodeSpace(this.gizmoLayer.utilityLayerScene, changeTarget, positionSpaceTransform);\n }\n else if (this._attachedSpaceNode) {\n this.attachMeshAtNodeSpace(this.gizmoLayer.utilityLayerScene, this._attachedSpaceNode, positionSpaceTransform);\n }\n super.updateGizmoPosition();\n }\n get attachedMesh() {\n if (this.xGizmo) {\n return this.xGizmo.attachedMesh;\n }\n else if (this.yGizmo) {\n return this.yGizmo.attachedMesh;\n }\n else {\n return this.zGizmo.attachedMesh;\n }\n }\n set attachedMesh(mesh) {\n if (this.xGizmo) {\n this.xGizmo.attachedMesh = mesh;\n }\n if (this.yGizmo) {\n this.yGizmo.attachedMesh = mesh;\n }\n if (this.zGizmo) {\n this.zGizmo.attachedMesh = mesh;\n }\n if (mesh) {\n this._rootMesh.rotationQuaternion = mesh.rotationQuaternion\n ? mesh.rotationQuaternion\n : mesh.rotation.toQuaternion();\n this._rootMesh.position = mesh.position;\n this._rootMesh.setPivotPoint(mesh.getPivotPoint(), Space.WORLD);\n }\n }\n get visibility() {\n return this.xGizmo.visibility;\n }\n set visibility(visibility) {\n if (this.xGizmo) {\n this.xGizmo.visibility = visibility;\n this.yGizmo.visibility = visibility;\n this.zGizmo.visibility = visibility;\n }\n }\n /**\n * Creates a PositionGizmo\n * @param gizmoLayer The utility layer the gizmo will be added to\n */\n constructor(kbViewer, gizmoLayer = UtilityLayerRenderer.DefaultUtilityLayer, axisVisibility) {\n super(gizmoLayer);\n this.kbViewer = kbViewer;\n /** Fires an event when any of it's sub gizmos are dragged */\n this.onDragStartObservable = new Observable$1();\n /** Fires an event while any of it's sub gizmos are dragged */\n this.onDragObservable = new Observable$1();\n /** Fires an event when any of it's sub gizmos are released from dragging */\n this.onDragEndObservable = new Observable$1();\n this.positionChange = Vector3.Zero();\n let activeGizmos = [];\n if (axisVisibility) {\n if (axisVisibility[0]) {\n this.xGizmo = new KbAxisDragGizmo(new Vector3(1, 0, 0), Color3.Red().scale(0.5), gizmoLayer);\n this.xGizmo.updateGizmoRotationToMatchAttachedMesh = axisVisibility[0];\n this.xGizmo.updateGizmoPositionToMatchAttachedMesh = axisVisibility[0];\n activeGizmos.push(this.xGizmo);\n }\n if (axisVisibility[1]) {\n this.yGizmo = new KbAxisDragGizmo(new Vector3(0, 1, 0), Color3.Green().scale(0.5), gizmoLayer);\n this.yGizmo.updateGizmoRotationToMatchAttachedMesh = axisVisibility[1];\n this.yGizmo.updateGizmoPositionToMatchAttachedMesh = axisVisibility[1];\n activeGizmos.push(this.yGizmo);\n }\n if (axisVisibility[2]) {\n this.zGizmo = new KbAxisDragGizmo(new Vector3(0, 0, 1), Color3.Blue().scale(0.5), gizmoLayer);\n this.zGizmo.updateGizmoRotationToMatchAttachedMesh = axisVisibility[2];\n this.zGizmo.updateGizmoPositionToMatchAttachedMesh = axisVisibility[2];\n activeGizmos.push(this.zGizmo);\n }\n }\n else {\n this.xGizmo = new KbAxisDragGizmo(new Vector3(1, 0, 0), Color3.Red().scale(0.5), gizmoLayer);\n this.yGizmo = new KbAxisDragGizmo(new Vector3(0, 1, 0), Color3.Green().scale(0.5), gizmoLayer);\n this.zGizmo = new KbAxisDragGizmo(new Vector3(0, 0, 1), Color3.Blue().scale(0.5), gizmoLayer);\n this.xGizmo.updateGizmoRotationToMatchAttachedMesh = true;\n this.yGizmo.updateGizmoRotationToMatchAttachedMesh = true;\n this.zGizmo.updateGizmoRotationToMatchAttachedMesh = true;\n this.xGizmo.updateGizmoPositionToMatchAttachedMesh = true;\n this.yGizmo.updateGizmoPositionToMatchAttachedMesh = true;\n this.zGizmo.updateGizmoPositionToMatchAttachedMesh = true;\n activeGizmos = [this.xGizmo, this.yGizmo, this.zGizmo];\n }\n // Relay drag events\n activeGizmos.forEach(gizmo => {\n gizmo.dragBehavior.onDragStartObservable.add(() => {\n this.onDragStartObservable.notifyObservers();\n this.positionChange.setAll(0);\n });\n gizmo.updateObservable.add(event => {\n // transform the rotation to be in local space, the gizmos return global space movements\n const transform = new Matrix();\n this.attachedMesh.computeWorldMatrix(true).invertToRef(transform);\n transform.setTranslation(Vector3.Zero());\n const { delta } = event;\n const result = Vector3.TransformCoordinates(delta, transform);\n // squish floating point errors that look gross in the output\n if (Math.abs(result.x) < 0.00000001)\n result.x = 0;\n if (Math.abs(result.y) < 0.00000001)\n result.y = 0;\n if (Math.abs(result.z) < 0.00000001)\n result.z = 0;\n this.positionChange.addInPlace(result);\n this.onDragObservable.notifyObservers({ delta: result });\n });\n gizmo.dragBehavior.onDragEndObservable.add(() => {\n this.onDragEndObservable.notifyObservers({ delta: this.positionChange });\n });\n });\n this.attachedMesh = null;\n }\n /**\n * Drag distance in babylon units that the gizmo will snap to when dragged (Default: 0)\n */\n set snapDistance(value) {\n if (this.xGizmo) {\n this.xGizmo.snapDistance = value;\n }\n if (this.yGizmo) {\n this.yGizmo.snapDistance = value;\n }\n if (this.zGizmo) {\n this.zGizmo.snapDistance = value;\n }\n }\n get snapDistance() {\n if (this.xGizmo) {\n return this.xGizmo.snapDistance;\n }\n else if (this.yGizmo) {\n return this.yGizmo.snapDistance;\n }\n else {\n return this.zGizmo.snapDistance;\n }\n }\n /**\n * Ratio for the scale of the gizmo (Default: 1)\n */\n set scaleRatio(value) {\n if (this.xGizmo) {\n this.xGizmo.scaleRatio = value;\n }\n if (this.yGizmo) {\n this.yGizmo.scaleRatio = value;\n }\n if (this.zGizmo) {\n this.zGizmo.scaleRatio = value;\n }\n }\n get scaleRatio() {\n if (this.xGizmo) {\n return this.xGizmo.scaleRatio;\n }\n else if (this.yGizmo) {\n return this.yGizmo.scaleRatio;\n }\n else {\n return this.zGizmo.scaleRatio;\n }\n }\n /**\n * Disposes of the gizmo\n */\n dispose() {\n if (this.xGizmo) {\n this.xGizmo.dispose();\n }\n if (this.yGizmo) {\n this.yGizmo.dispose();\n }\n if (this.zGizmo) {\n this.zGizmo.dispose();\n }\n this.onDragStartObservable.clear();\n this.onDragEndObservable.clear();\n if (this.attachedMesh && this.attachedMesh.name === AUTO_MESH_NAME) {\n this.attachedMesh.dispose();\n }\n }\n /**\n * CustomMeshes are not supported by this gizmo\n * @param mesh The mesh to replace the default mesh of the gizmo\n */\n setCustomMesh(mesh) {\n Logger.Error('Custom meshes are not supported on this gizmo, please set the custom meshes on the gizmos contained within this one (gizmo.xGizmo, gizmo.yGizmo, gizmo.zGizmo)');\n }\n }\n\n const tessellation$1 = 32;\n /**\n * Gizmo that enables rotating a mesh along 3 axis\n */\n class KbRotationGizmo extends KbGizmo {\n attachTo(target, positionSpace, placementOffset) {\n if (!positionSpace && target instanceof exports.SpaceNode && target._parent) {\n positionSpace = target._parent;\n }\n if (positionSpace) {\n this._positionSpace = this.kbViewer.getRendererNode(positionSpace);\n }\n if (target instanceof exports.SpaceNode) {\n this._attachedSpaceNode = target;\n this._updateGizmoPosition();\n }\n else {\n this._updateGizmoPosition(target, placementOffset);\n }\n }\n updateGizmoPosition(changeTarget, placementOffset) {\n // Need to wait for the next frame because this can be called before the babylon object is actually updated\n this.kbViewer.onNextFrame(() => {\n this._updateGizmoPosition(changeTarget, placementOffset);\n });\n }\n _updateGizmoPosition(changeTarget, placementOffset) {\n var _a, _b;\n const positionSpaceTransform = (_a = this._positionSpace) === null || _a === void 0 ? void 0 : _a.computeWorldMatrix(true);\n if (changeTarget) {\n // If target is a vector, then we want to rotate it as a direction, so set the gizmo position to the origin to calculations are done from there\n // But store the position as the offset because we want the gizmo itself to be visually placed on the position space origin\n this.attachMeshAtNodeSpace(this.gizmoLayer.utilityLayerScene, placementOffset || exports.KbVector.Zero(), positionSpaceTransform);\n const corr = rotationToVector(changeTarget.toVec3(), Vector3.Up());\n this.gizmoOffset = exports.KbVector.FromVec3(this.attachedMesh.position);\n this.attachedMesh.position = Vector3.Zero();\n (_b = this.attachedMesh.rotationQuaternion) === null || _b === void 0 ? void 0 : _b.multiplyInPlace(corr);\n this.attachedMesh.computeWorldMatrix(true);\n }\n else if (this._attachedSpaceNode) {\n this.attachMeshAtNodeSpace(this.gizmoLayer.utilityLayerScene, this._attachedSpaceNode.position.add(this._attachedSpaceNode.pivot), positionSpaceTransform);\n let currentRotation;\n // Rotation gizmo should be rotated to match the current rotation if the target is a node\n if (this._attachedSpaceNode.useEuler) {\n currentRotation = this._attachedSpaceNode.eulerRotation\n .toVec3()\n .scale(Math.PI / 180)\n .toQuaternion();\n }\n else {\n currentRotation = rotationToVector(this._attachedSpaceNode.rotationAxis.toVec3());\n }\n this.attachedMesh.rotationQuaternion = (positionSpaceTransform\n ? Quaternion.FromRotationMatrix(positionSpaceTransform.getRotationMatrix())\n : Quaternion.Identity()).multiply(currentRotation);\n }\n this.gizmoOffset && this.attachedMesh.position.addInPlace(this.gizmoOffset.toVec3());\n super.updateGizmoPosition();\n }\n get attachedMesh() {\n return (this.xGizmo || this.yGizmo || this.zGizmo).attachedMesh;\n }\n set attachedMesh(mesh) {\n if (this.xGizmo) {\n this.xGizmo.attachedMesh = mesh;\n }\n if (this.yGizmo) {\n this.yGizmo.attachedMesh = mesh;\n }\n if (this.zGizmo) {\n this.zGizmo.attachedMesh = mesh;\n }\n if (mesh) {\n this._rootMesh.setPivotPoint(mesh.getPivotPoint(), Space.WORLD);\n }\n }\n /**\n * Creates a RotationGizmo\n * @param gizmoLayer The utility layer the gizmo will be added to\n * @param tessellation Amount of tessellation to be used when creating rotation circles\n */\n constructor(kbViewer, gizmoLayer = UtilityLayerRenderer.DefaultUtilityLayer, axes, axisVisibility) {\n super(gizmoLayer);\n this.kbViewer = kbViewer;\n /** Fires an event when any of it's sub gizmos are dragged */\n this.onDragStartObservable = new Observable$1();\n /** Fires an event when any of it's sub gizmos are released from dragging */\n this.onDragObservable = new Observable$1();\n this.onDragEndObservable = new Observable$1();\n let activeGizmos = [];\n if (axisVisibility) {\n if (axisVisibility[0]) {\n this.xGizmo = new KbPlaneRotationGizmo(new Vector3(1, 0, 0), Color3.Red().scale(0.5), gizmoLayer, tessellation$1);\n this.xGizmo.updateGizmoRotationToMatchAttachedMesh = axisVisibility[0];\n this.xGizmo.updateGizmoPositionToMatchAttachedMesh = axisVisibility[0];\n activeGizmos.push(this.xGizmo);\n }\n if (axisVisibility[1]) {\n this.yGizmo = new KbPlaneRotationGizmo(new Vector3(0, 1, 0), Color3.Green().scale(0.5), gizmoLayer, tessellation$1);\n this.yGizmo.updateGizmoRotationToMatchAttachedMesh = axisVisibility[1];\n this.yGizmo.updateGizmoPositionToMatchAttachedMesh = axisVisibility[1];\n activeGizmos.push(this.yGizmo);\n }\n if (axisVisibility[2]) {\n this.zGizmo = new KbPlaneRotationGizmo(new Vector3(0, 0, 1), Color3.Blue().scale(0.5), gizmoLayer, tessellation$1);\n this.zGizmo.updateGizmoRotationToMatchAttachedMesh = axisVisibility[2];\n this.zGizmo.updateGizmoPositionToMatchAttachedMesh = axisVisibility[2];\n activeGizmos.push(this.zGizmo);\n }\n }\n else {\n this.updateGizmoRotationToMatchAttachedMesh = true;\n if (!axes || axes.x) {\n this.xGizmo = new KbPlaneRotationGizmo(new Vector3(1, 0, 0), Color3.Red().scale(0.5), gizmoLayer, tessellation$1);\n this.xGizmo.updateGizmoRotationToMatchAttachedMesh = true;\n this.xGizmo.updateGizmoPositionToMatchAttachedMesh = true;\n }\n if (!axes || axes.y) {\n this.yGizmo = new KbPlaneRotationGizmo(new Vector3(0, 1, 0), Color3.Green().scale(0.5), gizmoLayer, tessellation$1);\n this.yGizmo.updateGizmoRotationToMatchAttachedMesh = true;\n this.yGizmo.updateGizmoPositionToMatchAttachedMesh = true;\n }\n if (!axes || axes.z) {\n this.zGizmo = new KbPlaneRotationGizmo(new Vector3(0, 0, 1), Color3.Blue().scale(0.5), gizmoLayer, tessellation$1);\n this.zGizmo.updateGizmoRotationToMatchAttachedMesh = true;\n this.zGizmo.updateGizmoPositionToMatchAttachedMesh = true;\n }\n activeGizmos = [this.xGizmo, this.yGizmo, this.zGizmo];\n }\n this.rotationChange = Quaternion.Identity();\n // Relay drag events\n activeGizmos.forEach(gizmo => {\n if (gizmo) {\n gizmo.dragBehavior.onDragStartObservable.add(() => {\n this.onDragStartObservable.notifyObservers();\n this.rotationChange = Quaternion.Identity();\n });\n gizmo.updateObservable.add(event => {\n this.rotationChange.multiplyInPlace(event.delta);\n this.onDragObservable.notifyObservers({ delta: event.delta });\n });\n gizmo.dragBehavior.onDragEndObservable.add(() => {\n this.onDragEndObservable.notifyObservers({ delta: this.rotationChange });\n });\n }\n });\n this.attachedMesh = null;\n }\n /**\n * Drag distance in babylon units that the gizmo will snap to when dragged (Default: 0)\n */\n set snapDistance(value) {\n if (this.xGizmo) {\n this.xGizmo.snapDistance = value;\n }\n if (this.yGizmo) {\n this.yGizmo.snapDistance = value;\n }\n if (this.zGizmo) {\n this.zGizmo.snapDistance = value;\n }\n }\n get snapDistance() {\n return (this.xGizmo || this.yGizmo || this.zGizmo).snapDistance;\n }\n /**\n * Ratio for the scale of the gizmo (Default: 1)\n */\n set scaleRatio(value) {\n if (this.xGizmo) {\n this.xGizmo.scaleRatio = value;\n }\n if (this.yGizmo) {\n this.yGizmo.scaleRatio = value;\n }\n if (this.zGizmo) {\n this.zGizmo.scaleRatio = value;\n }\n }\n get scaleRatio() {\n return (this.xGizmo || this.yGizmo || this.zGizmo).scaleRatio;\n }\n /**\n * Disposes of the gizmo\n */\n dispose() {\n if (this.xGizmo) {\n this.xGizmo.dispose();\n }\n if (this.yGizmo) {\n this.yGizmo.dispose();\n }\n if (this.zGizmo) {\n this.zGizmo.dispose();\n }\n this.onDragStartObservable.clear();\n this.onDragEndObservable.clear();\n }\n /**\n * CustomMeshes are not supported by this gizmo\n * @param mesh The mesh to replace the default mesh of the gizmo\n */\n setCustomMesh(mesh) {\n Logger.Error('Custom meshes are not supported on this gizmo, please set the custom meshes on the gizmos contained within this one (gizmo.xGizmo, gizmo.yGizmo, gizmo.zGizmo)');\n }\n }\n\n /**\n * Single axis scale gizmo\n */\n class KbAxisScaleGizmo extends KbGizmo {\n /**\n * Creates an AxisScaleGizmo\n * @param gizmoLayer The utility layer the gizmo will be added to\n * @param dragAxis The axis which the gizmo will be able to scale on\n * @param color The color of the gizmo\n */\n constructor(dragAxis, color = Color3.Gray(), gizmoLayer = UtilityLayerRenderer.DefaultUtilityLayer) {\n super(gizmoLayer);\n this._pointerObserver = null;\n /**\n * Scale distance in babylon units that the gizmo will snap to when dragged (Default: 0)\n */\n this.snapDistance = 0;\n /**\n * Event that fires each time the gizmo snaps to a new location.\n * * snapDistance is the the change in distance\n */\n this.onSnapObservable = new Observable$1();\n this.updateObservable = new Observable$1();\n /**\n * If the scaling operation should be done on all axis (default: false)\n */\n this.uniformScaling = false;\n // Create Material\n this._coloredMaterial = new StandardMaterial('', gizmoLayer.utilityLayerScene);\n this._coloredMaterial.disableLighting = true;\n this._coloredMaterial.emissiveColor = color;\n // this._coloredMaterial.diffuseColor = color;\n // this._coloredMaterial.specularColor = color.subtract(new Color3(0.1, 0.1, 0.1));\n const hoverMaterial = new StandardMaterial('', gizmoLayer.utilityLayerScene);\n hoverMaterial.disableLighting = true;\n hoverMaterial.emissiveColor = color.add(new Color3(0.3, 0.3, 0.3));\n // hoverMaterial.diffuseColor = color.add(new Color3(0.3, 0.3, 0.3));\n // Build mesh on root node\n const arrow = new AbstractMesh('', gizmoLayer.utilityLayerScene);\n const arrowMesh = BoxBuilder.CreateBox('yPosMesh', { size: 0.4 }, gizmoLayer.utilityLayerScene);\n const arrowTail = CylinderBuilder.CreateCylinder('cylinder', { diameterTop: 0.005, height: 0.275, diameterBottom: 0.005, tessellation: 96 }, gizmoLayer.utilityLayerScene);\n arrowTail.material = this._coloredMaterial;\n arrow.addChild(arrowMesh);\n arrow.addChild(arrowTail);\n // Position arrow pointing in its drag axis\n arrowMesh.scaling.scaleInPlace(0.1);\n arrowMesh.material = this._coloredMaterial;\n arrowMesh.rotation.x = Math.PI / 2;\n arrowMesh.position.z += 0.3;\n arrowTail.position.z += 0.275 / 2;\n arrowTail.rotation.x = Math.PI / 2;\n arrow.lookAt(this._rootMesh.position.add(dragAxis));\n this._rootMesh.addChild(arrow);\n arrow.scaling.scaleInPlace(1 / 3);\n // Add drag behavior to handle events when the gizmo is dragged\n this.dragBehavior = new PointerDragBehavior({ dragAxis: dragAxis });\n this.dragBehavior.moveAttached = false;\n this._rootMesh.addBehavior(this.dragBehavior);\n let currentSnapDragDistance = 0;\n const tmpVector = new Vector3();\n const tmpSnapEvent = { snapDistance: 0 };\n this.dragBehavior.onDragObservable.add(event => {\n if (this.attachedMesh) {\n // Drag strength is modified by the scale of the gizmo (eg. for small objects like boombox the strength will be increased to match the behavior of larger objects)\n const dragStrength = event.dragDistance * ((this.scaleRatio * 8) / this._rootMesh.scaling.length());\n // Snapping logic\n let snapped = false;\n let dragSteps = 0;\n if (this.uniformScaling) {\n this.attachedMesh.scaling.normalizeToRef(tmpVector);\n if (tmpVector.y < 0) {\n tmpVector.scaleInPlace(-1);\n }\n }\n else {\n tmpVector.copyFrom(dragAxis);\n }\n if (this.snapDistance == 0) {\n tmpVector.scaleToRef(dragStrength, tmpVector);\n }\n else {\n currentSnapDragDistance += dragStrength;\n if (Math.abs(currentSnapDragDistance) > this.snapDistance) {\n dragSteps = Math.floor(Math.abs(currentSnapDragDistance) / this.snapDistance);\n if (currentSnapDragDistance < 0) {\n dragSteps *= -1;\n }\n currentSnapDragDistance = currentSnapDragDistance % this.snapDistance;\n tmpVector.scaleToRef(this.snapDistance * dragSteps, tmpVector);\n snapped = true;\n }\n else {\n tmpVector.scaleInPlace(0);\n }\n }\n this.attachedMesh.scaling.addInPlace(tmpVector);\n this.updateObservable.notifyObservers({ delta: tmpVector });\n if (snapped) {\n tmpSnapEvent.snapDistance = this.snapDistance * dragSteps;\n this.onSnapObservable.notifyObservers(tmpSnapEvent);\n }\n }\n });\n this._pointerObserver = gizmoLayer.utilityLayerScene.onPointerObservable.add(pointerInfo => {\n if (this._customMeshSet) {\n return;\n }\n const isHovered = pointerInfo.pickInfo &&\n this._rootMesh.getChildMeshes().indexOf(pointerInfo.pickInfo.pickedMesh) != -1;\n const material = isHovered ? hoverMaterial : this._coloredMaterial;\n this._rootMesh.getChildMeshes().forEach(m => {\n m.material = material;\n if (m.color) {\n m.color = material.diffuseColor;\n }\n });\n });\n // var light = gizmoLayer._getSharedGizmoLight();\n // light.includedOnlyMeshes = light.includedOnlyMeshes.concat(this._rootMesh.getChildMeshes());\n }\n _attachedMeshChanged(value) {\n if (this.dragBehavior) {\n this.dragBehavior.enabled = value ? true : false;\n }\n }\n /**\n * Disposes of the gizmo\n */\n dispose() {\n this.onSnapObservable.clear();\n this.gizmoLayer.utilityLayerScene.onPointerObservable.remove(this._pointerObserver);\n this.dragBehavior.detach();\n super.dispose();\n }\n /**\n * Disposes and replaces the current meshes in the gizmo with the specified mesh\n * @param mesh The mesh to replace the default mesh of the gizmo\n * @param useGizmoMaterial If the gizmo's default material should be used (default: false)\n */\n setCustomMesh(mesh, useGizmoMaterial = false) {\n super.setCustomMesh(mesh);\n if (useGizmoMaterial) {\n this._rootMesh.getChildMeshes().forEach(m => {\n m.material = this._coloredMaterial;\n if (m.color) {\n m.color = this._coloredMaterial.diffuseColor;\n }\n });\n this._customMeshSet = false;\n }\n }\n }\n\n /**\n * Gizmo that enables scaling a mesh along 3 axis\n */\n class KbScaleGizmo extends KbGizmo {\n attachTo(target, positionSpace) {\n if (!positionSpace && target instanceof exports.SpaceNode && target._parent) {\n positionSpace = target._parent;\n }\n if (positionSpace) {\n this._positionSpace = this.kbViewer.getRendererNode(positionSpace);\n }\n if (target instanceof exports.SpaceNode) {\n this._attachedSpaceNode = target;\n this._updateGizmoPosition();\n }\n else {\n this._updateGizmoPosition(target);\n }\n }\n updateGizmoPosition(changeTarget) {\n // Need to wait for the next frame because this can be called before the babylon object is actually updated\n this.kbViewer.onNextFrame(() => {\n this._updateGizmoPosition(changeTarget);\n });\n }\n _updateGizmoPosition(changeTarget) {\n var _a;\n const positionSpaceTransform = (_a = this._positionSpace) === null || _a === void 0 ? void 0 : _a.computeWorldMatrix(true);\n if (changeTarget) {\n // If target is a vector, then we want to rotate it as a direction, so set the gizmo position to the origin to calculations are done from there\n // But store the position as the offset because we want the gizmo itself to be visually placed on the position space origin\n this.attachMeshAtNodeSpace(this.gizmoLayer.utilityLayerScene, changeTarget, positionSpaceTransform);\n this.gizmoOffset = exports.KbVector.FromVec3(this.attachedMesh.position);\n }\n else if (this._attachedSpaceNode) {\n this.attachMeshAtNodeSpace(this.gizmoLayer.utilityLayerScene, this._attachedSpaceNode, positionSpaceTransform);\n // Put the gizmo in the same position as the pivot, in world space\n this.attachedMesh.position.addInPlace(this._attachedSpaceNode.pivot.toVec3());\n }\n super.updateGizmoPosition();\n }\n get attachedMesh() {\n return this.xGizmo.attachedMesh;\n }\n set attachedMesh(mesh) {\n if (this.xGizmo) {\n this.xGizmo.attachedMesh = mesh;\n this.yGizmo.attachedMesh = mesh;\n this.zGizmo.attachedMesh = mesh;\n this.uniformScaleGizmo.attachedMesh = mesh;\n }\n }\n /**\n * Creates a ScaleGizmo\n * @param gizmoLayer The utility layer the gizmo will be added to\n */\n constructor(kbViewer, gizmoLayer = UtilityLayerRenderer.DefaultUtilityLayer) {\n super(gizmoLayer);\n this.kbViewer = kbViewer;\n /** Fires an event when any of it's sub gizmos are dragged */\n this.onDragStartObservable = new Observable$1();\n /** Fires an event while any of it's sub gizmos are dragged */\n this.onDragObservable = new Observable$1();\n /** Fires an event when any of it's sub gizmos are released from dragging */\n this.onDragEndObservable = new Observable$1();\n this.scaleChange = Vector3.Zero();\n this.xGizmo = new KbAxisScaleGizmo(new Vector3(1, 0, 0), Color3.Red().scale(0.5), gizmoLayer);\n this.yGizmo = new KbAxisScaleGizmo(new Vector3(0, 1, 0), Color3.Green().scale(0.5), gizmoLayer);\n this.zGizmo = new KbAxisScaleGizmo(new Vector3(0, 0, 1), Color3.Blue().scale(0.5), gizmoLayer);\n // Create uniform scale gizmo\n this.uniformScaleGizmo = new KbAxisScaleGizmo(new Vector3(0, 1, 0), Color3.Yellow().scale(0.5), gizmoLayer);\n this.uniformScaleGizmo.updateGizmoRotationToMatchAttachedMesh = false;\n this.uniformScaleGizmo.uniformScaling = true;\n const uniformScalingMesh = PolyhedronBuilder.CreatePolyhedron('', { type: 1 }, this.uniformScaleGizmo.gizmoLayer.utilityLayerScene);\n uniformScalingMesh.scaling.scaleInPlace(0.03);\n uniformScalingMesh.visibility = 0;\n const octahedron = PolyhedronBuilder.CreatePolyhedron('', { type: 1 }, this.uniformScaleGizmo.gizmoLayer.utilityLayerScene);\n octahedron.scaling.scaleInPlace(0.007);\n uniformScalingMesh.addChild(octahedron);\n this.uniformScaleGizmo.setCustomMesh(uniformScalingMesh, true);\n // var light = gizmoLayer._getSharedGizmoLight();\n // light.includedOnlyMeshes = light.includedOnlyMeshes.concat(octahedron);\n // Relay drag events\n [this.xGizmo, this.yGizmo, this.zGizmo, this.uniformScaleGizmo].forEach(gizmo => {\n gizmo.dragBehavior.onDragStartObservable.add(() => {\n this.onDragStartObservable.notifyObservers();\n this.scaleChange.setAll(0);\n });\n gizmo.updateObservable.add(event => {\n this.scaleChange.addInPlace(event.delta);\n this.onDragObservable.notifyObservers({ delta: event.delta });\n });\n gizmo.dragBehavior.onDragEndObservable.add(() => {\n this.onDragEndObservable.notifyObservers({ delta: this.scaleChange });\n });\n });\n this.attachedMesh = null;\n }\n set updateGizmoRotationToMatchAttachedMesh(value) {\n if (!value) {\n Logger.Warn('Setting updateGizmoRotationToMatchAttachedMesh = false on scaling gizmo is not supported.');\n }\n if (this.xGizmo) {\n this.xGizmo.updateGizmoRotationToMatchAttachedMesh = value;\n this.yGizmo.updateGizmoRotationToMatchAttachedMesh = value;\n this.zGizmo.updateGizmoRotationToMatchAttachedMesh = value;\n }\n }\n get updateGizmoRotationToMatchAttachedMesh() {\n return this.xGizmo.updateGizmoRotationToMatchAttachedMesh;\n }\n /**\n * Drag distance in babylon units that the gizmo will snap to when dragged (Default: 0)\n */\n set snapDistance(value) {\n if (this.xGizmo) {\n this.xGizmo.snapDistance = value;\n this.yGizmo.snapDistance = value;\n this.zGizmo.snapDistance = value;\n this.uniformScaleGizmo.snapDistance = value;\n }\n }\n get snapDistance() {\n return this.xGizmo.snapDistance;\n }\n /**\n * Ratio for the scale of the gizmo (Default: 1)\n */\n set scaleRatio(value) {\n if (this.xGizmo) {\n this.xGizmo.scaleRatio = value;\n this.yGizmo.scaleRatio = value;\n this.zGizmo.scaleRatio = value;\n this.uniformScaleGizmo.scaleRatio = value;\n }\n }\n get scaleRatio() {\n return this.xGizmo.scaleRatio;\n }\n /**\n * Disposes of the gizmo\n */\n dispose() {\n this.xGizmo.dispose();\n this.yGizmo.dispose();\n this.zGizmo.dispose();\n this.uniformScaleGizmo.dispose();\n this.onDragStartObservable.clear();\n this.onDragEndObservable.clear();\n }\n }\n\n /** Class used to store color4 gradient */\n class ColorGradient {\n /**\n * Creates a new color4 gradient\n * @param gradient gets or sets the gradient value (between 0 and 1)\n * @param color1 gets or sets first associated color\n * @param color2 gets or sets first second color\n */\n constructor(\n /**\n * Gets or sets the gradient value (between 0 and 1)\n */\n gradient, \n /**\n * Gets or sets first associated color\n */\n color1, \n /**\n * Gets or sets second associated color\n */\n color2) {\n this.gradient = gradient;\n this.color1 = color1;\n this.color2 = color2;\n }\n /**\n * Will get a color picked randomly between color1 and color2.\n * If color2 is undefined then color1 will be used\n * @param result defines the target Color4 to store the result in\n */\n getColorToRef(result) {\n if (!this.color2) {\n result.copyFrom(this.color1);\n return;\n }\n Color4.LerpToRef(this.color1, this.color2, Math.random(), result);\n }\n }\n /** Class used to store color 3 gradient */\n class Color3Gradient {\n /**\n * Creates a new color3 gradient\n * @param gradient gets or sets the gradient value (between 0 and 1)\n * @param color gets or sets associated color\n */\n constructor(\n /**\n * Gets or sets the gradient value (between 0 and 1)\n */\n gradient, \n /**\n * Gets or sets the associated color\n */\n color) {\n this.gradient = gradient;\n this.color = color;\n }\n }\n /** Class used to store factor gradient */\n class FactorGradient {\n /**\n * Creates a new factor gradient\n * @param gradient gets or sets the gradient value (between 0 and 1)\n * @param factor1 gets or sets first associated factor\n * @param factor2 gets or sets second associated factor\n */\n constructor(\n /**\n * Gets or sets the gradient value (between 0 and 1)\n */\n gradient, \n /**\n * Gets or sets first associated factor\n */\n factor1, \n /**\n * Gets or sets second associated factor\n */\n factor2) {\n this.gradient = gradient;\n this.factor1 = factor1;\n this.factor2 = factor2;\n }\n /**\n * Will get a number picked randomly between factor1 and factor2.\n * If factor2 is undefined then factor1 will be used\n * @returns the picked number\n */\n getFactor() {\n if (this.factor2 === undefined || this.factor2 === this.factor1) {\n return this.factor1;\n }\n return this.factor1 + (this.factor2 - this.factor1) * Math.random();\n }\n }\n /**\n * Helper used to simplify some generic gradient tasks\n */\n class GradientHelper {\n /**\n * Gets the current gradient from an array of IValueGradient\n * @param ratio defines the current ratio to get\n * @param gradients defines the array of IValueGradient\n * @param updateFunc defines the callback function used to get the final value from the selected gradients\n */\n static GetCurrentGradient(ratio, gradients, updateFunc) {\n // Use last index if over\n if (gradients[0].gradient > ratio) {\n updateFunc(gradients[0], gradients[0], 1.0);\n return;\n }\n for (let gradientIndex = 0; gradientIndex < gradients.length - 1; gradientIndex++) {\n const currentGradient = gradients[gradientIndex];\n const nextGradient = gradients[gradientIndex + 1];\n if (ratio >= currentGradient.gradient && ratio <= nextGradient.gradient) {\n const scale = (ratio - currentGradient.gradient) / (nextGradient.gradient - currentGradient.gradient);\n updateFunc(currentGradient, nextGradient, scale);\n return;\n }\n }\n // Use last index if over\n const lastIndex = gradients.length - 1;\n updateFunc(gradients[lastIndex], gradients[lastIndex], 1.0);\n }\n }\n\n /**\n * A particle represents one of the element emitted by a particle system.\n * This is mainly define by its coordinates, direction, velocity and age.\n */\n class Particle {\n /**\n * Creates a new instance Particle\n * @param particleSystem the particle system the particle belongs to\n */\n constructor(\n /**\n * The particle system the particle belongs to.\n */\n particleSystem) {\n this.particleSystem = particleSystem;\n /**\n * The world position of the particle in the scene.\n */\n this.position = Vector3.Zero();\n /**\n * The world direction of the particle in the scene.\n */\n this.direction = Vector3.Zero();\n /**\n * The color of the particle.\n */\n this.color = new Color4(0, 0, 0, 0);\n /**\n * The color change of the particle per step.\n */\n this.colorStep = new Color4(0, 0, 0, 0);\n /**\n * Defines how long will the life of the particle be.\n */\n this.lifeTime = 1.0;\n /**\n * The current age of the particle.\n */\n this.age = 0;\n /**\n * The current size of the particle.\n */\n this.size = 0;\n /**\n * The current scale of the particle.\n */\n this.scale = new Vector2(1, 1);\n /**\n * The current angle of the particle.\n */\n this.angle = 0;\n /**\n * Defines how fast is the angle changing.\n */\n this.angularSpeed = 0;\n /**\n * Defines the cell index used by the particle to be rendered from a sprite.\n */\n this.cellIndex = 0;\n /** @internal */\n this._attachedSubEmitters = null;\n /** @internal */\n this._currentColor1 = new Color4(0, 0, 0, 0);\n /** @internal */\n this._currentColor2 = new Color4(0, 0, 0, 0);\n /** @internal */\n this._currentSize1 = 0;\n /** @internal */\n this._currentSize2 = 0;\n /** @internal */\n this._currentAngularSpeed1 = 0;\n /** @internal */\n this._currentAngularSpeed2 = 0;\n /** @internal */\n this._currentVelocity1 = 0;\n /** @internal */\n this._currentVelocity2 = 0;\n /** @internal */\n this._currentLimitVelocity1 = 0;\n /** @internal */\n this._currentLimitVelocity2 = 0;\n /** @internal */\n this._currentDrag1 = 0;\n /** @internal */\n this._currentDrag2 = 0;\n this.id = Particle._Count++;\n if (!this.particleSystem.isAnimationSheetEnabled) {\n return;\n }\n this._updateCellInfoFromSystem();\n }\n _updateCellInfoFromSystem() {\n this.cellIndex = this.particleSystem.startSpriteCellID;\n }\n /**\n * Defines how the sprite cell index is updated for the particle\n */\n updateCellIndex() {\n let offsetAge = this.age;\n let changeSpeed = this.particleSystem.spriteCellChangeSpeed;\n if (this.particleSystem.spriteRandomStartCell) {\n if (this._randomCellOffset === undefined) {\n this._randomCellOffset = Math.random() * this.lifeTime;\n }\n if (changeSpeed === 0) {\n // Special case when speed = 0 meaning we want to stay on initial cell\n changeSpeed = 1;\n offsetAge = this._randomCellOffset;\n }\n else {\n offsetAge += this._randomCellOffset;\n }\n }\n const dist = this._initialEndSpriteCellID - this._initialStartSpriteCellID;\n let ratio;\n if (this._initialSpriteCellLoop) {\n ratio = Scalar.Clamp(((offsetAge * changeSpeed) % this.lifeTime) / this.lifeTime);\n }\n else {\n ratio = Scalar.Clamp((offsetAge * changeSpeed) / this.lifeTime);\n }\n this.cellIndex = (this._initialStartSpriteCellID + ratio * dist) | 0;\n }\n /**\n * @internal\n */\n _inheritParticleInfoToSubEmitter(subEmitter) {\n if (subEmitter.particleSystem.emitter.position) {\n const emitterMesh = subEmitter.particleSystem.emitter;\n emitterMesh.position.copyFrom(this.position);\n if (subEmitter.inheritDirection) {\n const temp = TmpVectors.Vector3[0];\n this.direction.normalizeToRef(temp);\n emitterMesh.setDirection(temp, 0, Math.PI / 2);\n }\n }\n else {\n const emitterPosition = subEmitter.particleSystem.emitter;\n emitterPosition.copyFrom(this.position);\n }\n // Set inheritedVelocityOffset to be used when new particles are created\n this.direction.scaleToRef(subEmitter.inheritedVelocityAmount / 2, TmpVectors.Vector3[0]);\n subEmitter.particleSystem._inheritedVelocityOffset.copyFrom(TmpVectors.Vector3[0]);\n }\n /** @internal */\n _inheritParticleInfoToSubEmitters() {\n if (this._attachedSubEmitters && this._attachedSubEmitters.length > 0) {\n this._attachedSubEmitters.forEach((subEmitter) => {\n this._inheritParticleInfoToSubEmitter(subEmitter);\n });\n }\n }\n /** @internal */\n _reset() {\n this.age = 0;\n this.id = Particle._Count++;\n this._currentColorGradient = null;\n this._currentSizeGradient = null;\n this._currentAngularSpeedGradient = null;\n this._currentVelocityGradient = null;\n this._currentLimitVelocityGradient = null;\n this._currentDragGradient = null;\n this.cellIndex = this.particleSystem.startSpriteCellID;\n this._randomCellOffset = undefined;\n }\n /**\n * Copy the properties of particle to another one.\n * @param other the particle to copy the information to.\n */\n copyTo(other) {\n other.position.copyFrom(this.position);\n if (this._initialDirection) {\n if (other._initialDirection) {\n other._initialDirection.copyFrom(this._initialDirection);\n }\n else {\n other._initialDirection = this._initialDirection.clone();\n }\n }\n else {\n other._initialDirection = null;\n }\n other.direction.copyFrom(this.direction);\n if (this._localPosition) {\n if (other._localPosition) {\n other._localPosition.copyFrom(this._localPosition);\n }\n else {\n other._localPosition = this._localPosition.clone();\n }\n }\n other.color.copyFrom(this.color);\n other.colorStep.copyFrom(this.colorStep);\n other.lifeTime = this.lifeTime;\n other.age = this.age;\n other._randomCellOffset = this._randomCellOffset;\n other.size = this.size;\n other.scale.copyFrom(this.scale);\n other.angle = this.angle;\n other.angularSpeed = this.angularSpeed;\n other.particleSystem = this.particleSystem;\n other.cellIndex = this.cellIndex;\n other.id = this.id;\n other._attachedSubEmitters = this._attachedSubEmitters;\n if (this._currentColorGradient) {\n other._currentColorGradient = this._currentColorGradient;\n other._currentColor1.copyFrom(this._currentColor1);\n other._currentColor2.copyFrom(this._currentColor2);\n }\n if (this._currentSizeGradient) {\n other._currentSizeGradient = this._currentSizeGradient;\n other._currentSize1 = this._currentSize1;\n other._currentSize2 = this._currentSize2;\n }\n if (this._currentAngularSpeedGradient) {\n other._currentAngularSpeedGradient = this._currentAngularSpeedGradient;\n other._currentAngularSpeed1 = this._currentAngularSpeed1;\n other._currentAngularSpeed2 = this._currentAngularSpeed2;\n }\n if (this._currentVelocityGradient) {\n other._currentVelocityGradient = this._currentVelocityGradient;\n other._currentVelocity1 = this._currentVelocity1;\n other._currentVelocity2 = this._currentVelocity2;\n }\n if (this._currentLimitVelocityGradient) {\n other._currentLimitVelocityGradient = this._currentLimitVelocityGradient;\n other._currentLimitVelocity1 = this._currentLimitVelocity1;\n other._currentLimitVelocity2 = this._currentLimitVelocity2;\n }\n if (this._currentDragGradient) {\n other._currentDragGradient = this._currentDragGradient;\n other._currentDrag1 = this._currentDrag1;\n other._currentDrag2 = this._currentDrag2;\n }\n if (this.particleSystem.isAnimationSheetEnabled) {\n other._initialStartSpriteCellID = this._initialStartSpriteCellID;\n other._initialEndSpriteCellID = this._initialEndSpriteCellID;\n other._initialSpriteCellLoop = this._initialSpriteCellLoop;\n }\n if (this.particleSystem.useRampGradients) {\n if (other.remapData && this.remapData) {\n other.remapData.copyFrom(this.remapData);\n }\n else {\n other.remapData = new Vector4(0, 0, 0, 0);\n }\n }\n if (this._randomNoiseCoordinates1) {\n if (other._randomNoiseCoordinates1) {\n other._randomNoiseCoordinates1.copyFrom(this._randomNoiseCoordinates1);\n other._randomNoiseCoordinates2.copyFrom(this._randomNoiseCoordinates2);\n }\n else {\n other._randomNoiseCoordinates1 = this._randomNoiseCoordinates1.clone();\n other._randomNoiseCoordinates2 = this._randomNoiseCoordinates2.clone();\n }\n }\n }\n }\n Particle._Count = 0;\n\n /**\n * Type of sub emitter\n */\n var SubEmitterType;\n (function (SubEmitterType) {\n /**\n * Attached to the particle over it's lifetime\n */\n SubEmitterType[SubEmitterType[\"ATTACHED\"] = 0] = \"ATTACHED\";\n /**\n * Created when the particle dies\n */\n SubEmitterType[SubEmitterType[\"END\"] = 1] = \"END\";\n })(SubEmitterType || (SubEmitterType = {}));\n /**\n * Sub emitter class used to emit particles from an existing particle\n */\n class SubEmitter {\n /**\n * Creates a sub emitter\n * @param particleSystem the particle system to be used by the sub emitter\n */\n constructor(\n /**\n * the particle system to be used by the sub emitter\n */\n particleSystem) {\n this.particleSystem = particleSystem;\n /**\n * Type of the submitter (Default: END)\n */\n this.type = SubEmitterType.END;\n /**\n * If the particle should inherit the direction from the particle it's attached to. (+Y will face the direction the particle is moving) (Default: false)\n * Note: This only is supported when using an emitter of type Mesh\n */\n this.inheritDirection = false;\n /**\n * How much of the attached particles speed should be added to the sub emitted particle (default: 0)\n */\n this.inheritedVelocityAmount = 0;\n // Create mesh as emitter to support rotation\n if (!particleSystem.emitter || !particleSystem.emitter.dispose) {\n const internalClass = GetClass(\"BABYLON.AbstractMesh\");\n particleSystem.emitter = new internalClass(\"SubemitterSystemEmitter\", particleSystem.getScene());\n particleSystem._disposeEmitterOnDispose = true;\n }\n }\n /**\n * Clones the sub emitter\n * @returns the cloned sub emitter\n */\n clone() {\n // Clone particle system\n let emitter = this.particleSystem.emitter;\n if (!emitter) {\n emitter = new Vector3();\n }\n else if (emitter instanceof Vector3) {\n emitter = emitter.clone();\n }\n else if (emitter.getClassName().indexOf(\"Mesh\") !== -1) {\n const internalClass = GetClass(\"BABYLON.Mesh\");\n emitter = new internalClass(\"\", emitter.getScene());\n emitter.isVisible = false;\n }\n const clone = new SubEmitter(this.particleSystem.clone(this.particleSystem.name, emitter));\n // Clone properties\n clone.particleSystem.name += \"Clone\";\n clone.type = this.type;\n clone.inheritDirection = this.inheritDirection;\n clone.inheritedVelocityAmount = this.inheritedVelocityAmount;\n clone.particleSystem._disposeEmitterOnDispose = true;\n clone.particleSystem.disposeOnStop = true;\n return clone;\n }\n /**\n * Serialize current object to a JSON object\n * @param serializeTexture defines if the texture must be serialized as well\n * @returns the serialized object\n */\n serialize(serializeTexture = false) {\n const serializationObject = {};\n serializationObject.type = this.type;\n serializationObject.inheritDirection = this.inheritDirection;\n serializationObject.inheritedVelocityAmount = this.inheritedVelocityAmount;\n serializationObject.particleSystem = this.particleSystem.serialize(serializeTexture);\n return serializationObject;\n }\n /**\n * @internal\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n static _ParseParticleSystem(system, sceneOrEngine, rootUrl, doNotStart = false) {\n throw _WarnImport(\"ParseParticle\");\n }\n /**\n * Creates a new SubEmitter from a serialized JSON version\n * @param serializationObject defines the JSON object to read from\n * @param sceneOrEngine defines the hosting scene or the hosting engine\n * @param rootUrl defines the rootUrl for data loading\n * @returns a new SubEmitter\n */\n static Parse(serializationObject, sceneOrEngine, rootUrl) {\n const system = serializationObject.particleSystem;\n const subEmitter = new SubEmitter(SubEmitter._ParseParticleSystem(system, sceneOrEngine, rootUrl, true));\n subEmitter.type = serializationObject.type;\n subEmitter.inheritDirection = serializationObject.inheritDirection;\n subEmitter.inheritedVelocityAmount = serializationObject.inheritedVelocityAmount;\n subEmitter.particleSystem._isSubEmitter = true;\n return subEmitter;\n }\n /** Release associated resources */\n dispose() {\n this.particleSystem.dispose();\n }\n }\n\n // Do not edit.\n const name$2I = \"particlesPixelShader\";\n const shader$2I = `#ifdef LOGARITHMICDEPTH\n#extension GL_EXT_frag_depth : enable\n#endif\nvarying vec2 vUV;varying vec4 vColor;uniform vec4 textureMask;uniform sampler2D diffuseSampler;\n#include\n#include\n#include\n#include\n#include\n#ifdef RAMPGRADIENT\nvarying vec4 remapRanges;uniform sampler2D rampSampler;\n#endif\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_FRAGMENT_MAIN_BEGIN\n#include\nvec4 textureColor=texture2D(diffuseSampler,vUV);vec4 baseColor=(textureColor*textureMask+(vec4(1.,1.,1.,1.)-textureMask))*vColor;\n#ifdef RAMPGRADIENT\nfloat alpha=baseColor.a;float remappedColorIndex=clamp((alpha-remapRanges.x)/remapRanges.y,0.0,1.0);vec4 rampColor=texture2D(rampSampler,vec2(1.0-remappedColorIndex,0.));baseColor.rgb*=rampColor.rgb;float finalAlpha=baseColor.a;baseColor.a=clamp((alpha*rampColor.a-remapRanges.z)/remapRanges.w,0.0,1.0);\n#endif\n#ifdef BLENDMULTIPLYMODE\nfloat sourceAlpha=vColor.a*textureColor.a;baseColor.rgb=baseColor.rgb*sourceAlpha+vec3(1.0)*(1.0-sourceAlpha);\n#endif\n#include\n#ifdef IMAGEPROCESSINGPOSTPROCESS\nbaseColor.rgb=toLinearSpace(baseColor.rgb);\n#else\n#ifdef IMAGEPROCESSING\nbaseColor.rgb=toLinearSpace(baseColor.rgb);baseColor=applyImageProcessing(baseColor);\n#endif\n#endif\ngl_FragColor=baseColor;\n#define CUSTOM_FRAGMENT_MAIN_END\n}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2I] = shader$2I;\n\n // Do not edit.\n const name$2J = \"particlesVertexShader\";\n const shader$2J = `attribute vec3 position;attribute vec4 color;attribute float angle;attribute vec2 size;\n#ifdef ANIMATESHEET\nattribute float cellIndex;\n#endif\n#ifndef BILLBOARD\nattribute vec3 direction;\n#endif\n#ifdef BILLBOARDSTRETCHED\nattribute vec3 direction;\n#endif\n#ifdef RAMPGRADIENT\nattribute vec4 remapData;\n#endif\nattribute vec2 offset;uniform mat4 view;uniform mat4 projection;uniform vec2 translationPivot;\n#ifdef ANIMATESHEET\nuniform vec3 particlesInfos; \n#endif\nvarying vec2 vUV;varying vec4 vColor;varying vec3 vPositionW;\n#ifdef RAMPGRADIENT\nvarying vec4 remapRanges;\n#endif\n#if defined(BILLBOARD) && !defined(BILLBOARDY) && !defined(BILLBOARDSTRETCHED)\nuniform mat4 invView;\n#endif\n#include\n#include\n#ifdef BILLBOARD\nuniform vec3 eyePosition;\n#endif\nvec3 rotate(vec3 yaxis,vec3 rotatedCorner) {vec3 xaxis=normalize(cross(vec3(0.,1.0,0.),yaxis));vec3 zaxis=normalize(cross(yaxis,xaxis));vec3 row0=vec3(xaxis.x,xaxis.y,xaxis.z);vec3 row1=vec3(yaxis.x,yaxis.y,yaxis.z);vec3 row2=vec3(zaxis.x,zaxis.y,zaxis.z);mat3 rotMatrix= mat3(row0,row1,row2);vec3 alignedCorner=rotMatrix*rotatedCorner;return position+alignedCorner;}\n#ifdef BILLBOARDSTRETCHED\nvec3 rotateAlign(vec3 toCamera,vec3 rotatedCorner) {vec3 normalizedToCamera=normalize(toCamera);vec3 normalizedCrossDirToCamera=normalize(cross(normalize(direction),normalizedToCamera));vec3 row0=vec3(normalizedCrossDirToCamera.x,normalizedCrossDirToCamera.y,normalizedCrossDirToCamera.z);vec3 row2=vec3(normalizedToCamera.x,normalizedToCamera.y,normalizedToCamera.z);\n#ifdef BILLBOARDSTRETCHED_LOCAL\nvec3 row1=direction;\n#else\nvec3 crossProduct=normalize(cross(normalizedToCamera,normalizedCrossDirToCamera));vec3 row1=vec3(crossProduct.x,crossProduct.y,crossProduct.z);\n#endif\nmat3 rotMatrix= mat3(row0,row1,row2);vec3 alignedCorner=rotMatrix*rotatedCorner;return position+alignedCorner;}\n#endif\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\nvec2 cornerPos;cornerPos=(vec2(offset.x-0.5,offset.y -0.5)-translationPivot)*size+translationPivot;\n#ifdef BILLBOARD\nvec3 rotatedCorner;\n#ifdef BILLBOARDY\nrotatedCorner.x=cornerPos.x*cos(angle)-cornerPos.y*sin(angle);rotatedCorner.z=cornerPos.x*sin(angle)+cornerPos.y*cos(angle);rotatedCorner.y=0.;vec3 yaxis=position-eyePosition;yaxis.y=0.;vPositionW=rotate(normalize(yaxis),rotatedCorner);vec3 viewPos=(view*vec4(vPositionW,1.0)).xyz;\n#elif defined(BILLBOARDSTRETCHED)\nrotatedCorner.x=cornerPos.x*cos(angle)-cornerPos.y*sin(angle);rotatedCorner.y=cornerPos.x*sin(angle)+cornerPos.y*cos(angle);rotatedCorner.z=0.;vec3 toCamera=position-eyePosition;vPositionW=rotateAlign(toCamera,rotatedCorner);vec3 viewPos=(view*vec4(vPositionW,1.0)).xyz;\n#else\nrotatedCorner.x=cornerPos.x*cos(angle)-cornerPos.y*sin(angle);rotatedCorner.y=cornerPos.x*sin(angle)+cornerPos.y*cos(angle);rotatedCorner.z=0.;vec3 viewPos=(view*vec4(position,1.0)).xyz+rotatedCorner;vPositionW=(invView*vec4(viewPos,1)).xyz;\n#endif\n#ifdef RAMPGRADIENT\nremapRanges=remapData;\n#endif\ngl_Position=projection*vec4(viewPos,1.0);\n#else\nvec3 rotatedCorner;rotatedCorner.x=cornerPos.x*cos(angle)-cornerPos.y*sin(angle);rotatedCorner.z=cornerPos.x*sin(angle)+cornerPos.y*cos(angle);rotatedCorner.y=0.;vec3 yaxis=normalize(direction);vPositionW=rotate(yaxis,rotatedCorner);gl_Position=projection*view*vec4(vPositionW,1.0);\n#endif\nvColor=color;\n#ifdef ANIMATESHEET\nfloat rowOffset=floor(cellIndex*particlesInfos.z);float columnOffset=cellIndex-rowOffset/particlesInfos.z;vec2 uvScale=particlesInfos.xy;vec2 uvOffset=vec2(offset.x ,1.0-offset.y);vUV=(uvOffset+vec2(columnOffset,rowOffset))*uvScale;\n#else\nvUV=offset;\n#endif\n#if defined(CLIPPLANE) || defined(CLIPPLANE2) || defined(CLIPPLANE3) || defined(CLIPPLANE4) || defined(CLIPPLANE5) || defined(CLIPPLANE6)\nvec4 worldPos=vec4(vPositionW,1.0);\n#endif\n#include\n#include\n#define CUSTOM_VERTEX_MAIN_END\n}`;\n // Sideeffect\n ShaderStore.ShadersStore[name$2J] = shader$2J;\n\n /**\n * This represents a particle system in Babylon.\n * Particles are often small sprites used to simulate hard-to-reproduce phenomena like fire, smoke, water, or abstract visual effects like magic glitter and faery dust.\n * Particles can take different shapes while emitted like box, sphere, cone or you can write your custom function.\n * @example https://doc.babylonjs.com/features/featuresDeepDive/particles/particle_system/particle_system_intro\n */\n class ParticleSystem extends BaseParticleSystem {\n /**\n * Sets a callback that will be triggered when the system is disposed\n */\n set onDispose(callback) {\n if (this._onDisposeObserver) {\n this.onDisposeObservable.remove(this._onDisposeObserver);\n }\n this._onDisposeObserver = this.onDisposeObservable.add(callback);\n }\n /** Gets or sets a boolean indicating that ramp gradients must be used\n * @see https://doc.babylonjs.com/features/featuresDeepDive/particles/particle_system/particle_system_intro#ramp-gradients\n */\n get useRampGradients() {\n return this._useRampGradients;\n }\n set useRampGradients(value) {\n if (this._useRampGradients === value) {\n return;\n }\n this._useRampGradients = value;\n this._resetEffect();\n }\n //end of Sub-emitter\n /**\n * Gets the current list of active particles\n */\n get particles() {\n return this._particles;\n }\n /**\n * Gets the number of particles active at the same time.\n * @returns The number of active particles.\n */\n getActiveCount() {\n return this._particles.length;\n }\n /**\n * Returns the string \"ParticleSystem\"\n * @returns a string containing the class name\n */\n getClassName() {\n return \"ParticleSystem\";\n }\n /**\n * Gets a boolean indicating that the system is stopping\n * @returns true if the system is currently stopping\n */\n isStopping() {\n return this._stopped && this.isAlive();\n }\n /**\n * Gets the custom effect used to render the particles\n * @param blendMode Blend mode for which the effect should be retrieved\n * @returns The effect\n */\n getCustomEffect(blendMode = 0) {\n var _a, _b;\n return (_b = (_a = this._customWrappers[blendMode]) === null || _a === void 0 ? void 0 : _a.effect) !== null && _b !== void 0 ? _b : this._customWrappers[0].effect;\n }\n _getCustomDrawWrapper(blendMode = 0) {\n var _a;\n return (_a = this._customWrappers[blendMode]) !== null && _a !== void 0 ? _a : this._customWrappers[0];\n }\n /**\n * Sets the custom effect used to render the particles\n * @param effect The effect to set\n * @param blendMode Blend mode for which the effect should be set\n */\n setCustomEffect(effect, blendMode = 0) {\n this._customWrappers[blendMode] = new DrawWrapper(this._engine);\n this._customWrappers[blendMode].effect = effect;\n if (this._customWrappers[blendMode].drawContext) {\n this._customWrappers[blendMode].drawContext.useInstancing = this._useInstancing;\n }\n }\n /**\n * Observable that will be called just before the particles are drawn\n */\n get onBeforeDrawParticlesObservable() {\n if (!this._onBeforeDrawParticlesObservable) {\n this._onBeforeDrawParticlesObservable = new Observable$1();\n }\n return this._onBeforeDrawParticlesObservable;\n }\n /**\n * Gets the name of the particle vertex shader\n */\n get vertexShaderName() {\n return \"particles\";\n }\n /**\n * Gets the vertex buffers used by the particle system\n */\n get vertexBuffers() {\n return this._vertexBuffers;\n }\n /**\n * Gets the index buffer used by the particle system (or null if no index buffer is used (if _useInstancing=true))\n */\n get indexBuffer() {\n return this._indexBuffer;\n }\n /**\n * Instantiates a particle system.\n * Particles are often small sprites used to simulate hard-to-reproduce phenomena like fire, smoke, water, or abstract visual effects like magic glitter and faery dust.\n * @param name The name of the particle system\n * @param capacity The max number of particles alive at the same time\n * @param sceneOrEngine The scene the particle system belongs to or the engine to use if no scene\n * @param customEffect a custom effect used to change the way particles are rendered by default\n * @param isAnimationSheetEnabled Must be true if using a spritesheet to animate the particles texture\n * @param epsilon Offset used to render the particles\n */\n constructor(name, capacity, sceneOrEngine, customEffect = null, isAnimationSheetEnabled = false, epsilon = 0.01) {\n super(name);\n this._emitterInverseWorldMatrix = Matrix.Identity();\n /**\n * @internal\n */\n this._inheritedVelocityOffset = new Vector3();\n /**\n * An event triggered when the system is disposed\n */\n this.onDisposeObservable = new Observable$1();\n /**\n * An event triggered when the system is stopped\n */\n this.onStoppedObservable = new Observable$1();\n this._particles = new Array();\n this._stockParticles = new Array();\n this._newPartsExcess = 0;\n this._vertexBuffers = {};\n this._scaledColorStep = new Color4(0, 0, 0, 0);\n this._colorDiff = new Color4(0, 0, 0, 0);\n this._scaledDirection = Vector3.Zero();\n this._scaledGravity = Vector3.Zero();\n this._currentRenderId = -1;\n this._useInstancing = false;\n this._started = false;\n this._stopped = false;\n this._actualFrame = 0;\n /** @internal */\n this._currentEmitRate1 = 0;\n /** @internal */\n this._currentEmitRate2 = 0;\n /** @internal */\n this._currentStartSize1 = 0;\n /** @internal */\n this._currentStartSize2 = 0;\n /** Indicates that the update of particles is done in the animate function */\n this.updateInAnimate = true;\n this._rawTextureWidth = 256;\n this._useRampGradients = false;\n /**\n * @internal\n * If the particle systems emitter should be disposed when the particle system is disposed\n */\n this._disposeEmitterOnDispose = false;\n /**\n * Specifies if the particles are updated in emitter local space or world space\n */\n this.isLocal = false;\n /** Indicates that the particle system is CPU based */\n this.isGPU = false;\n /** @internal */\n this._onBeforeDrawParticlesObservable = null;\n // start of sub system methods\n /**\n * \"Recycles\" one of the particle by copying it back to the \"stock\" of particles and removing it from the active list.\n * Its lifetime will start back at 0.\n * @param particle\n */\n this.recycleParticle = (particle) => {\n // move particle from activeParticle list to stock particles\n const lastParticle = this._particles.pop();\n if (lastParticle !== particle) {\n lastParticle.copyTo(particle);\n }\n this._stockParticles.push(lastParticle);\n };\n this._createParticle = () => {\n let particle;\n if (this._stockParticles.length !== 0) {\n particle = this._stockParticles.pop();\n particle._reset();\n }\n else {\n particle = new Particle(this);\n }\n // Attach emitters\n if (this._subEmitters && this._subEmitters.length > 0) {\n const subEmitters = this._subEmitters[Math.floor(Math.random() * this._subEmitters.length)];\n particle._attachedSubEmitters = [];\n subEmitters.forEach((subEmitter) => {\n if (subEmitter.type === SubEmitterType.ATTACHED) {\n const newEmitter = subEmitter.clone();\n particle._attachedSubEmitters.push(newEmitter);\n newEmitter.particleSystem.start();\n }\n });\n }\n return particle;\n };\n this._emitFromParticle = (particle) => {\n if (!this._subEmitters || this._subEmitters.length === 0) {\n return;\n }\n const templateIndex = Math.floor(Math.random() * this._subEmitters.length);\n this._subEmitters[templateIndex].forEach((subEmitter) => {\n if (subEmitter.type === SubEmitterType.END) {\n const subSystem = subEmitter.clone();\n particle._inheritParticleInfoToSubEmitter(subSystem);\n subSystem.particleSystem._rootParticleSystem = this;\n this.activeSubSystems.push(subSystem.particleSystem);\n subSystem.particleSystem.start();\n }\n });\n };\n this._capacity = capacity;\n this._epsilon = epsilon;\n this._isAnimationSheetEnabled = isAnimationSheetEnabled;\n if (!sceneOrEngine || sceneOrEngine.getClassName() === \"Scene\") {\n this._scene = sceneOrEngine || EngineStore.LastCreatedScene;\n this._engine = this._scene.getEngine();\n this.uniqueId = this._scene.getUniqueId();\n this._scene.particleSystems.push(this);\n }\n else {\n this._engine = sceneOrEngine;\n this.defaultProjectionMatrix = Matrix.PerspectiveFovLH(0.8, 1, 0.1, 100, this._engine.isNDCHalfZRange);\n }\n if (this._engine.getCaps().vertexArrayObject) {\n this._vertexArrayObject = null;\n }\n // Setup the default processing configuration to the scene.\n this._attachImageProcessingConfiguration(null);\n // eslint-disable-next-line @typescript-eslint/naming-convention\n this._customWrappers = { 0: new DrawWrapper(this._engine) };\n this._customWrappers[0].effect = customEffect;\n this._drawWrappers = [];\n this._useInstancing = this._engine.getCaps().instancedArrays;\n this._createIndexBuffer();\n this._createVertexBuffers();\n // Default emitter type\n this.particleEmitterType = new BoxParticleEmitter();\n let noiseTextureData = null;\n // Update\n this.updateFunction = (particles) => {\n var _a;\n let noiseTextureSize = null;\n if (this.noiseTexture) {\n // We need to get texture data back to CPU\n noiseTextureSize = this.noiseTexture.getSize();\n (_a = this.noiseTexture.getContent()) === null || _a === void 0 ? void 0 : _a.then((data) => {\n noiseTextureData = data;\n });\n }\n const sameParticleArray = particles === this._particles;\n for (let index = 0; index < particles.length; index++) {\n const particle = particles[index];\n let scaledUpdateSpeed = this._scaledUpdateSpeed;\n const previousAge = particle.age;\n particle.age += scaledUpdateSpeed;\n // Evaluate step to death\n if (particle.age > particle.lifeTime) {\n const diff = particle.age - previousAge;\n const oldDiff = particle.lifeTime - previousAge;\n scaledUpdateSpeed = (oldDiff * scaledUpdateSpeed) / diff;\n particle.age = particle.lifeTime;\n }\n const ratio = particle.age / particle.lifeTime;\n // Color\n if (this._colorGradients && this._colorGradients.length > 0) {\n GradientHelper.GetCurrentGradient(ratio, this._colorGradients, (currentGradient, nextGradient, scale) => {\n if (currentGradient !== particle._currentColorGradient) {\n particle._currentColor1.copyFrom(particle._currentColor2);\n nextGradient.getColorToRef(particle._currentColor2);\n particle._currentColorGradient = currentGradient;\n }\n Color4.LerpToRef(particle._currentColor1, particle._currentColor2, scale, particle.color);\n });\n }\n else {\n particle.colorStep.scaleToRef(scaledUpdateSpeed, this._scaledColorStep);\n particle.color.addInPlace(this._scaledColorStep);\n if (particle.color.a < 0) {\n particle.color.a = 0;\n }\n }\n // Angular speed\n if (this._angularSpeedGradients && this._angularSpeedGradients.length > 0) {\n GradientHelper.GetCurrentGradient(ratio, this._angularSpeedGradients, (currentGradient, nextGradient, scale) => {\n if (currentGradient !== particle._currentAngularSpeedGradient) {\n particle._currentAngularSpeed1 = particle._currentAngularSpeed2;\n particle._currentAngularSpeed2 = nextGradient.getFactor();\n particle._currentAngularSpeedGradient = currentGradient;\n }\n particle.angularSpeed = Scalar.Lerp(particle._currentAngularSpeed1, particle._currentAngularSpeed2, scale);\n });\n }\n particle.angle += particle.angularSpeed * scaledUpdateSpeed;\n // Direction\n let directionScale = scaledUpdateSpeed;\n /// Velocity\n if (this._velocityGradients && this._velocityGradients.length > 0) {\n GradientHelper.GetCurrentGradient(ratio, this._velocityGradients, (currentGradient, nextGradient, scale) => {\n if (currentGradient !== particle._currentVelocityGradient) {\n particle._currentVelocity1 = particle._currentVelocity2;\n particle._currentVelocity2 = nextGradient.getFactor();\n particle._currentVelocityGradient = currentGradient;\n }\n directionScale *= Scalar.Lerp(particle._currentVelocity1, particle._currentVelocity2, scale);\n });\n }\n particle.direction.scaleToRef(directionScale, this._scaledDirection);\n /// Limit velocity\n if (this._limitVelocityGradients && this._limitVelocityGradients.length > 0) {\n GradientHelper.GetCurrentGradient(ratio, this._limitVelocityGradients, (currentGradient, nextGradient, scale) => {\n if (currentGradient !== particle._currentLimitVelocityGradient) {\n particle._currentLimitVelocity1 = particle._currentLimitVelocity2;\n particle._currentLimitVelocity2 = nextGradient.getFactor();\n particle._currentLimitVelocityGradient = currentGradient;\n }\n const limitVelocity = Scalar.Lerp(particle._currentLimitVelocity1, particle._currentLimitVelocity2, scale);\n const currentVelocity = particle.direction.length();\n if (currentVelocity > limitVelocity) {\n particle.direction.scaleInPlace(this.limitVelocityDamping);\n }\n });\n }\n /// Drag\n if (this._dragGradients && this._dragGradients.length > 0) {\n GradientHelper.GetCurrentGradient(ratio, this._dragGradients, (currentGradient, nextGradient, scale) => {\n if (currentGradient !== particle._currentDragGradient) {\n particle._currentDrag1 = particle._currentDrag2;\n particle._currentDrag2 = nextGradient.getFactor();\n particle._currentDragGradient = currentGradient;\n }\n const drag = Scalar.Lerp(particle._currentDrag1, particle._currentDrag2, scale);\n this._scaledDirection.scaleInPlace(1.0 - drag);\n });\n }\n if (this.isLocal && particle._localPosition) {\n particle._localPosition.addInPlace(this._scaledDirection);\n Vector3.TransformCoordinatesToRef(particle._localPosition, this._emitterWorldMatrix, particle.position);\n }\n else {\n particle.position.addInPlace(this._scaledDirection);\n }\n // Noise\n if (noiseTextureData && noiseTextureSize && particle._randomNoiseCoordinates1) {\n const fetchedColorR = this._fetchR(particle._randomNoiseCoordinates1.x, particle._randomNoiseCoordinates1.y, noiseTextureSize.width, noiseTextureSize.height, noiseTextureData);\n const fetchedColorG = this._fetchR(particle._randomNoiseCoordinates1.z, particle._randomNoiseCoordinates2.x, noiseTextureSize.width, noiseTextureSize.height, noiseTextureData);\n const fetchedColorB = this._fetchR(particle._randomNoiseCoordinates2.y, particle._randomNoiseCoordinates2.z, noiseTextureSize.width, noiseTextureSize.height, noiseTextureData);\n const force = TmpVectors.Vector3[0];\n const scaledForce = TmpVectors.Vector3[1];\n force.copyFromFloats((2 * fetchedColorR - 1) * this.noiseStrength.x, (2 * fetchedColorG - 1) * this.noiseStrength.y, (2 * fetchedColorB - 1) * this.noiseStrength.z);\n force.scaleToRef(scaledUpdateSpeed, scaledForce);\n particle.direction.addInPlace(scaledForce);\n }\n // Gravity\n this.gravity.scaleToRef(scaledUpdateSpeed, this._scaledGravity);\n particle.direction.addInPlace(this._scaledGravity);\n // Size\n if (this._sizeGradients && this._sizeGradients.length > 0) {\n GradientHelper.GetCurrentGradient(ratio, this._sizeGradients, (currentGradient, nextGradient, scale) => {\n if (currentGradient !== particle._currentSizeGradient) {\n particle._currentSize1 = particle._currentSize2;\n particle._currentSize2 = nextGradient.getFactor();\n particle._currentSizeGradient = currentGradient;\n }\n particle.size = Scalar.Lerp(particle._currentSize1, particle._currentSize2, scale);\n });\n }\n // Remap data\n if (this._useRampGradients) {\n if (this._colorRemapGradients && this._colorRemapGradients.length > 0) {\n GradientHelper.GetCurrentGradient(ratio, this._colorRemapGradients, (currentGradient, nextGradient, scale) => {\n const min = Scalar.Lerp(currentGradient.factor1, nextGradient.factor1, scale);\n const max = Scalar.Lerp(currentGradient.factor2, nextGradient.factor2, scale);\n particle.remapData.x = min;\n particle.remapData.y = max - min;\n });\n }\n if (this._alphaRemapGradients && this._alphaRemapGradients.length > 0) {\n GradientHelper.GetCurrentGradient(ratio, this._alphaRemapGradients, (currentGradient, nextGradient, scale) => {\n const min = Scalar.Lerp(currentGradient.factor1, nextGradient.factor1, scale);\n const max = Scalar.Lerp(currentGradient.factor2, nextGradient.factor2, scale);\n particle.remapData.z = min;\n particle.remapData.w = max - min;\n });\n }\n }\n if (this._isAnimationSheetEnabled) {\n particle.updateCellIndex();\n }\n // Update the position of the attached sub-emitters to match their attached particle\n particle._inheritParticleInfoToSubEmitters();\n if (particle.age >= particle.lifeTime) {\n // Recycle by swapping with last particle\n this._emitFromParticle(particle);\n if (particle._attachedSubEmitters) {\n particle._attachedSubEmitters.forEach((subEmitter) => {\n subEmitter.particleSystem.disposeOnStop = true;\n subEmitter.particleSystem.stop();\n });\n particle._attachedSubEmitters = null;\n }\n this.recycleParticle(particle);\n if (sameParticleArray) {\n index--;\n }\n continue;\n }\n }\n };\n }\n _addFactorGradient(factorGradients, gradient, factor, factor2) {\n const newGradient = new FactorGradient(gradient, factor, factor2);\n factorGradients.push(newGradient);\n factorGradients.sort((a, b) => {\n if (a.gradient < b.gradient) {\n return -1;\n }\n else if (a.gradient > b.gradient) {\n return 1;\n }\n return 0;\n });\n }\n _removeFactorGradient(factorGradients, gradient) {\n if (!factorGradients) {\n return;\n }\n let index = 0;\n for (const factorGradient of factorGradients) {\n if (factorGradient.gradient === gradient) {\n factorGradients.splice(index, 1);\n break;\n }\n index++;\n }\n }\n /**\n * Adds a new life time gradient\n * @param gradient defines the gradient to use (between 0 and 1)\n * @param factor defines the life time factor to affect to the specified gradient\n * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from\n * @returns the current particle system\n */\n addLifeTimeGradient(gradient, factor, factor2) {\n if (!this._lifeTimeGradients) {\n this._lifeTimeGradients = [];\n }\n this._addFactorGradient(this._lifeTimeGradients, gradient, factor, factor2);\n return this;\n }\n /**\n * Remove a specific life time gradient\n * @param gradient defines the gradient to remove\n * @returns the current particle system\n */\n removeLifeTimeGradient(gradient) {\n this._removeFactorGradient(this._lifeTimeGradients, gradient);\n return this;\n }\n /**\n * Adds a new size gradient\n * @param gradient defines the gradient to use (between 0 and 1)\n * @param factor defines the size factor to affect to the specified gradient\n * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from\n * @returns the current particle system\n */\n addSizeGradient(gradient, factor, factor2) {\n if (!this._sizeGradients) {\n this._sizeGradients = [];\n }\n this._addFactorGradient(this._sizeGradients, gradient, factor, factor2);\n return this;\n }\n /**\n * Remove a specific size gradient\n * @param gradient defines the gradient to remove\n * @returns the current particle system\n */\n removeSizeGradient(gradient) {\n this._removeFactorGradient(this._sizeGradients, gradient);\n return this;\n }\n /**\n * Adds a new color remap gradient\n * @param gradient defines the gradient to use (between 0 and 1)\n * @param min defines the color remap minimal range\n * @param max defines the color remap maximal range\n * @returns the current particle system\n */\n addColorRemapGradient(gradient, min, max) {\n if (!this._colorRemapGradients) {\n this._colorRemapGradients = [];\n }\n this._addFactorGradient(this._colorRemapGradients, gradient, min, max);\n return this;\n }\n /**\n * Remove a specific color remap gradient\n * @param gradient defines the gradient to remove\n * @returns the current particle system\n */\n removeColorRemapGradient(gradient) {\n this._removeFactorGradient(this._colorRemapGradients, gradient);\n return this;\n }\n /**\n * Adds a new alpha remap gradient\n * @param gradient defines the gradient to use (between 0 and 1)\n * @param min defines the alpha remap minimal range\n * @param max defines the alpha remap maximal range\n * @returns the current particle system\n */\n addAlphaRemapGradient(gradient, min, max) {\n if (!this._alphaRemapGradients) {\n this._alphaRemapGradients = [];\n }\n this._addFactorGradient(this._alphaRemapGradients, gradient, min, max);\n return this;\n }\n /**\n * Remove a specific alpha remap gradient\n * @param gradient defines the gradient to remove\n * @returns the current particle system\n */\n removeAlphaRemapGradient(gradient) {\n this._removeFactorGradient(this._alphaRemapGradients, gradient);\n return this;\n }\n /**\n * Adds a new angular speed gradient\n * @param gradient defines the gradient to use (between 0 and 1)\n * @param factor defines the angular speed to affect to the specified gradient\n * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from\n * @returns the current particle system\n */\n addAngularSpeedGradient(gradient, factor, factor2) {\n if (!this._angularSpeedGradients) {\n this._angularSpeedGradients = [];\n }\n this._addFactorGradient(this._angularSpeedGradients, gradient, factor, factor2);\n return this;\n }\n /**\n * Remove a specific angular speed gradient\n * @param gradient defines the gradient to remove\n * @returns the current particle system\n */\n removeAngularSpeedGradient(gradient) {\n this._removeFactorGradient(this._angularSpeedGradients, gradient);\n return this;\n }\n /**\n * Adds a new velocity gradient\n * @param gradient defines the gradient to use (between 0 and 1)\n * @param factor defines the velocity to affect to the specified gradient\n * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from\n * @returns the current particle system\n */\n addVelocityGradient(gradient, factor, factor2) {\n if (!this._velocityGradients) {\n this._velocityGradients = [];\n }\n this._addFactorGradient(this._velocityGradients, gradient, factor, factor2);\n return this;\n }\n /**\n * Remove a specific velocity gradient\n * @param gradient defines the gradient to remove\n * @returns the current particle system\n */\n removeVelocityGradient(gradient) {\n this._removeFactorGradient(this._velocityGradients, gradient);\n return this;\n }\n /**\n * Adds a new limit velocity gradient\n * @param gradient defines the gradient to use (between 0 and 1)\n * @param factor defines the limit velocity value to affect to the specified gradient\n * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from\n * @returns the current particle system\n */\n addLimitVelocityGradient(gradient, factor, factor2) {\n if (!this._limitVelocityGradients) {\n this._limitVelocityGradients = [];\n }\n this._addFactorGradient(this._limitVelocityGradients, gradient, factor, factor2);\n return this;\n }\n /**\n * Remove a specific limit velocity gradient\n * @param gradient defines the gradient to remove\n * @returns the current particle system\n */\n removeLimitVelocityGradient(gradient) {\n this._removeFactorGradient(this._limitVelocityGradients, gradient);\n return this;\n }\n /**\n * Adds a new drag gradient\n * @param gradient defines the gradient to use (between 0 and 1)\n * @param factor defines the drag value to affect to the specified gradient\n * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from\n * @returns the current particle system\n */\n addDragGradient(gradient, factor, factor2) {\n if (!this._dragGradients) {\n this._dragGradients = [];\n }\n this._addFactorGradient(this._dragGradients, gradient, factor, factor2);\n return this;\n }\n /**\n * Remove a specific drag gradient\n * @param gradient defines the gradient to remove\n * @returns the current particle system\n */\n removeDragGradient(gradient) {\n this._removeFactorGradient(this._dragGradients, gradient);\n return this;\n }\n /**\n * Adds a new emit rate gradient (please note that this will only work if you set the targetStopDuration property)\n * @param gradient defines the gradient to use (between 0 and 1)\n * @param factor defines the emit rate value to affect to the specified gradient\n * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from\n * @returns the current particle system\n */\n addEmitRateGradient(gradient, factor, factor2) {\n if (!this._emitRateGradients) {\n this._emitRateGradients = [];\n }\n this._addFactorGradient(this._emitRateGradients, gradient, factor, factor2);\n return this;\n }\n /**\n * Remove a specific emit rate gradient\n * @param gradient defines the gradient to remove\n * @returns the current particle system\n */\n removeEmitRateGradient(gradient) {\n this._removeFactorGradient(this._emitRateGradients, gradient);\n return this;\n }\n /**\n * Adds a new start size gradient (please note that this will only work if you set the targetStopDuration property)\n * @param gradient defines the gradient to use (between 0 and 1)\n * @param factor defines the start size value to affect to the specified gradient\n * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from\n * @returns the current particle system\n */\n addStartSizeGradient(gradient, factor, factor2) {\n if (!this._startSizeGradients) {\n this._startSizeGradients = [];\n }\n this._addFactorGradient(this._startSizeGradients, gradient, factor, factor2);\n return this;\n }\n /**\n * Remove a specific start size gradient\n * @param gradient defines the gradient to remove\n * @returns the current particle system\n */\n removeStartSizeGradient(gradient) {\n this._removeFactorGradient(this._startSizeGradients, gradient);\n return this;\n }\n _createRampGradientTexture() {\n if (!this._rampGradients || !this._rampGradients.length || this._rampGradientsTexture || !this._scene) {\n return;\n }\n const data = new Uint8Array(this._rawTextureWidth * 4);\n const tmpColor = TmpColors.Color3[0];\n for (let x = 0; x < this._rawTextureWidth; x++) {\n const ratio = x / this._rawTextureWidth;\n GradientHelper.GetCurrentGradient(ratio, this._rampGradients, (currentGradient, nextGradient, scale) => {\n Color3.LerpToRef(currentGradient.color, nextGradient.color, scale, tmpColor);\n data[x * 4] = tmpColor.r * 255;\n data[x * 4 + 1] = tmpColor.g * 255;\n data[x * 4 + 2] = tmpColor.b * 255;\n data[x * 4 + 3] = 255;\n });\n }\n this._rampGradientsTexture = RawTexture.CreateRGBATexture(data, this._rawTextureWidth, 1, this._scene, false, false, 1);\n }\n /**\n * Gets the current list of ramp gradients.\n * You must use addRampGradient and removeRampGradient to update this list\n * @returns the list of ramp gradients\n */\n getRampGradients() {\n return this._rampGradients;\n }\n /** Force the system to rebuild all gradients that need to be resync */\n forceRefreshGradients() {\n this._syncRampGradientTexture();\n }\n _syncRampGradientTexture() {\n if (!this._rampGradients) {\n return;\n }\n this._rampGradients.sort((a, b) => {\n if (a.gradient < b.gradient) {\n return -1;\n }\n else if (a.gradient > b.gradient) {\n return 1;\n }\n return 0;\n });\n if (this._rampGradientsTexture) {\n this._rampGradientsTexture.dispose();\n this._rampGradientsTexture = null;\n }\n this._createRampGradientTexture();\n }\n /**\n * Adds a new ramp gradient used to remap particle colors\n * @param gradient defines the gradient to use (between 0 and 1)\n * @param color defines the color to affect to the specified gradient\n * @returns the current particle system\n */\n addRampGradient(gradient, color) {\n if (!this._rampGradients) {\n this._rampGradients = [];\n }\n const rampGradient = new Color3Gradient(gradient, color);\n this._rampGradients.push(rampGradient);\n this._syncRampGradientTexture();\n return this;\n }\n /**\n * Remove a specific ramp gradient\n * @param gradient defines the gradient to remove\n * @returns the current particle system\n */\n removeRampGradient(gradient) {\n this._removeGradientAndTexture(gradient, this._rampGradients, this._rampGradientsTexture);\n this._rampGradientsTexture = null;\n if (this._rampGradients && this._rampGradients.length > 0) {\n this._createRampGradientTexture();\n }\n return this;\n }\n /**\n * Adds a new color gradient\n * @param gradient defines the gradient to use (between 0 and 1)\n * @param color1 defines the color to affect to the specified gradient\n * @param color2 defines an additional color used to define a range ([color, color2]) with main color to pick the final color from\n * @returns this particle system\n */\n addColorGradient(gradient, color1, color2) {\n if (!this._colorGradients) {\n this._colorGradients = [];\n }\n const colorGradient = new ColorGradient(gradient, color1, color2);\n this._colorGradients.push(colorGradient);\n this._colorGradients.sort((a, b) => {\n if (a.gradient < b.gradient) {\n return -1;\n }\n else if (a.gradient > b.gradient) {\n return 1;\n }\n return 0;\n });\n return this;\n }\n /**\n * Remove a specific color gradient\n * @param gradient defines the gradient to remove\n * @returns this particle system\n */\n removeColorGradient(gradient) {\n if (!this._colorGradients) {\n return this;\n }\n let index = 0;\n for (const colorGradient of this._colorGradients) {\n if (colorGradient.gradient === gradient) {\n this._colorGradients.splice(index, 1);\n break;\n }\n index++;\n }\n return this;\n }\n /**\n * Resets the draw wrappers cache\n */\n resetDrawCache() {\n for (const drawWrappers of this._drawWrappers) {\n if (drawWrappers) {\n for (const drawWrapper of drawWrappers) {\n drawWrapper === null || drawWrapper === void 0 ? void 0 : drawWrapper.dispose();\n }\n }\n }\n this._drawWrappers = [];\n }\n _fetchR(u, v, width, height, pixels) {\n u = Math.abs(u) * 0.5 + 0.5;\n v = Math.abs(v) * 0.5 + 0.5;\n const wrappedU = (u * width) % width | 0;\n const wrappedV = (v * height) % height | 0;\n const position = (wrappedU + wrappedV * width) * 4;\n return pixels[position] / 255;\n }\n _reset() {\n this._resetEffect();\n }\n _resetEffect() {\n if (this._vertexBuffer) {\n this._vertexBuffer.dispose();\n this._vertexBuffer = null;\n }\n if (this._spriteBuffer) {\n this._spriteBuffer.dispose();\n this._spriteBuffer = null;\n }\n if (this._vertexArrayObject) {\n this._engine.releaseVertexArrayObject(this._vertexArrayObject);\n this._vertexArrayObject = null;\n }\n this._createVertexBuffers();\n }\n _createVertexBuffers() {\n this._vertexBufferSize = this._useInstancing ? 10 : 12;\n if (this._isAnimationSheetEnabled) {\n this._vertexBufferSize += 1;\n }\n if (!this._isBillboardBased || this.billboardMode === ParticleSystem.BILLBOARDMODE_STRETCHED || this.billboardMode === ParticleSystem.BILLBOARDMODE_STRETCHED_LOCAL) {\n this._vertexBufferSize += 3;\n }\n if (this._useRampGradients) {\n this._vertexBufferSize += 4;\n }\n const engine = this._engine;\n const vertexSize = this._vertexBufferSize * (this._useInstancing ? 1 : 4);\n this._vertexData = new Float32Array(this._capacity * vertexSize);\n this._vertexBuffer = new Buffer$1(engine, this._vertexData, true, vertexSize);\n let dataOffset = 0;\n const positions = this._vertexBuffer.createVertexBuffer(VertexBuffer.PositionKind, dataOffset, 3, this._vertexBufferSize, this._useInstancing);\n this._vertexBuffers[VertexBuffer.PositionKind] = positions;\n dataOffset += 3;\n const colors = this._vertexBuffer.createVertexBuffer(VertexBuffer.ColorKind, dataOffset, 4, this._vertexBufferSize, this._useInstancing);\n this._vertexBuffers[VertexBuffer.ColorKind] = colors;\n dataOffset += 4;\n const options = this._vertexBuffer.createVertexBuffer(\"angle\", dataOffset, 1, this._vertexBufferSize, this._useInstancing);\n this._vertexBuffers[\"angle\"] = options;\n dataOffset += 1;\n const size = this._vertexBuffer.createVertexBuffer(\"size\", dataOffset, 2, this._vertexBufferSize, this._useInstancing);\n this._vertexBuffers[\"size\"] = size;\n dataOffset += 2;\n if (this._isAnimationSheetEnabled) {\n const cellIndexBuffer = this._vertexBuffer.createVertexBuffer(\"cellIndex\", dataOffset, 1, this._vertexBufferSize, this._useInstancing);\n this._vertexBuffers[\"cellIndex\"] = cellIndexBuffer;\n dataOffset += 1;\n }\n if (!this._isBillboardBased || this.billboardMode === ParticleSystem.BILLBOARDMODE_STRETCHED || this.billboardMode === ParticleSystem.BILLBOARDMODE_STRETCHED_LOCAL) {\n const directionBuffer = this._vertexBuffer.createVertexBuffer(\"direction\", dataOffset, 3, this._vertexBufferSize, this._useInstancing);\n this._vertexBuffers[\"direction\"] = directionBuffer;\n dataOffset += 3;\n }\n if (this._useRampGradients) {\n const rampDataBuffer = this._vertexBuffer.createVertexBuffer(\"remapData\", dataOffset, 4, this._vertexBufferSize, this._useInstancing);\n this._vertexBuffers[\"remapData\"] = rampDataBuffer;\n dataOffset += 4;\n }\n let offsets;\n if (this._useInstancing) {\n const spriteData = new Float32Array([0, 0, 1, 0, 0, 1, 1, 1]);\n this._spriteBuffer = new Buffer$1(engine, spriteData, false, 2);\n offsets = this._spriteBuffer.createVertexBuffer(\"offset\", 0, 2);\n }\n else {\n offsets = this._vertexBuffer.createVertexBuffer(\"offset\", dataOffset, 2, this._vertexBufferSize, this._useInstancing);\n dataOffset += 2;\n }\n this._vertexBuffers[\"offset\"] = offsets;\n this.resetDrawCache();\n }\n _createIndexBuffer() {\n if (this._useInstancing) {\n return;\n }\n const indices = [];\n let index = 0;\n for (let count = 0; count < this._capacity; count++) {\n indices.push(index);\n indices.push(index + 1);\n indices.push(index + 2);\n indices.push(index);\n indices.push(index + 2);\n indices.push(index + 3);\n index += 4;\n }\n this._indexBuffer = this._engine.createIndexBuffer(indices);\n }\n /**\n * Gets the maximum number of particles active at the same time.\n * @returns The max number of active particles.\n */\n getCapacity() {\n return this._capacity;\n }\n /**\n * Gets whether there are still active particles in the system.\n * @returns True if it is alive, otherwise false.\n */\n isAlive() {\n return this._alive;\n }\n /**\n * Gets if the system has been started. (Note: this will still be true after stop is called)\n * @returns True if it has been started, otherwise false.\n */\n isStarted() {\n return this._started;\n }\n _prepareSubEmitterInternalArray() {\n this._subEmitters = new Array();\n if (this.subEmitters) {\n this.subEmitters.forEach((subEmitter) => {\n if (subEmitter instanceof ParticleSystem) {\n this._subEmitters.push([new SubEmitter(subEmitter)]);\n }\n else if (subEmitter instanceof SubEmitter) {\n this._subEmitters.push([subEmitter]);\n }\n else if (subEmitter instanceof Array) {\n this._subEmitters.push(subEmitter);\n }\n });\n }\n }\n /**\n * Starts the particle system and begins to emit\n * @param delay defines the delay in milliseconds before starting the system (this.startDelay by default)\n */\n start(delay = this.startDelay) {\n var _a;\n if (!this.targetStopDuration && this._hasTargetStopDurationDependantGradient()) {\n throw \"Particle system started with a targetStopDuration dependant gradient (eg. startSizeGradients) but no targetStopDuration set\";\n }\n if (delay) {\n setTimeout(() => {\n this.start(0);\n }, delay);\n return;\n }\n // Convert the subEmitters field to the constant type field _subEmitters\n this._prepareSubEmitterInternalArray();\n this._started = true;\n this._stopped = false;\n this._actualFrame = 0;\n if (this._subEmitters && this._subEmitters.length != 0) {\n this.activeSubSystems = new Array();\n }\n // Reset emit gradient so it acts the same on every start\n if (this._emitRateGradients) {\n if (this._emitRateGradients.length > 0) {\n this._currentEmitRateGradient = this._emitRateGradients[0];\n this._currentEmitRate1 = this._currentEmitRateGradient.getFactor();\n this._currentEmitRate2 = this._currentEmitRate1;\n }\n if (this._emitRateGradients.length > 1) {\n this._currentEmitRate2 = this._emitRateGradients[1].getFactor();\n }\n }\n // Reset start size gradient so it acts the same on every start\n if (this._startSizeGradients) {\n if (this._startSizeGradients.length > 0) {\n this._currentStartSizeGradient = this._startSizeGradients[0];\n this._currentStartSize1 = this._currentStartSizeGradient.getFactor();\n this._currentStartSize2 = this._currentStartSize1;\n }\n if (this._startSizeGradients.length > 1) {\n this._currentStartSize2 = this._startSizeGradients[1].getFactor();\n }\n }\n if (this.preWarmCycles) {\n if (((_a = this.emitter) === null || _a === void 0 ? void 0 : _a.getClassName().indexOf(\"Mesh\")) !== -1) {\n this.emitter.computeWorldMatrix(true);\n }\n const noiseTextureAsProcedural = this.noiseTexture;\n if (noiseTextureAsProcedural && noiseTextureAsProcedural.onGeneratedObservable) {\n noiseTextureAsProcedural.onGeneratedObservable.addOnce(() => {\n setTimeout(() => {\n for (let index = 0; index < this.preWarmCycles; index++) {\n this.animate(true);\n noiseTextureAsProcedural.render();\n }\n });\n });\n }\n else {\n for (let index = 0; index < this.preWarmCycles; index++) {\n this.animate(true);\n }\n }\n }\n // Animations\n if (this.beginAnimationOnStart && this.animations && this.animations.length > 0 && this._scene) {\n this._scene.beginAnimation(this, this.beginAnimationFrom, this.beginAnimationTo, this.beginAnimationLoop);\n }\n }\n /**\n * Stops the particle system.\n * @param stopSubEmitters if true it will stop the current system and all created sub-Systems if false it will stop the current root system only, this param is used by the root particle system only. the default value is true.\n */\n stop(stopSubEmitters = true) {\n if (this._stopped) {\n return;\n }\n this.onStoppedObservable.notifyObservers(this);\n this._stopped = true;\n if (stopSubEmitters) {\n this._stopSubEmitters();\n }\n }\n // animation sheet\n /**\n * Remove all active particles\n */\n reset() {\n this._stockParticles.length = 0;\n this._particles.length = 0;\n }\n /**\n * @internal (for internal use only)\n */\n _appendParticleVertex(index, particle, offsetX, offsetY) {\n let offset = index * this._vertexBufferSize;\n this._vertexData[offset++] = particle.position.x + this.worldOffset.x;\n this._vertexData[offset++] = particle.position.y + this.worldOffset.y;\n this._vertexData[offset++] = particle.position.z + this.worldOffset.z;\n this._vertexData[offset++] = particle.color.r;\n this._vertexData[offset++] = particle.color.g;\n this._vertexData[offset++] = particle.color.b;\n this._vertexData[offset++] = particle.color.a;\n this._vertexData[offset++] = particle.angle;\n this._vertexData[offset++] = particle.scale.x * particle.size;\n this._vertexData[offset++] = particle.scale.y * particle.size;\n if (this._isAnimationSheetEnabled) {\n this._vertexData[offset++] = particle.cellIndex;\n }\n if (!this._isBillboardBased) {\n if (particle._initialDirection) {\n let initialDirection = particle._initialDirection;\n if (this.isLocal) {\n Vector3.TransformNormalToRef(initialDirection, this._emitterWorldMatrix, TmpVectors.Vector3[0]);\n initialDirection = TmpVectors.Vector3[0];\n }\n if (initialDirection.x === 0 && initialDirection.z === 0) {\n initialDirection.x = 0.001;\n }\n this._vertexData[offset++] = initialDirection.x;\n this._vertexData[offset++] = initialDirection.y;\n this._vertexData[offset++] = initialDirection.z;\n }\n else {\n let direction = particle.direction;\n if (this.isLocal) {\n Vector3.TransformNormalToRef(direction, this._emitterWorldMatrix, TmpVectors.Vector3[0]);\n direction = TmpVectors.Vector3[0];\n }\n if (direction.x === 0 && direction.z === 0) {\n direction.x = 0.001;\n }\n this._vertexData[offset++] = direction.x;\n this._vertexData[offset++] = direction.y;\n this._vertexData[offset++] = direction.z;\n }\n }\n else if (this.billboardMode === ParticleSystem.BILLBOARDMODE_STRETCHED || this.billboardMode === ParticleSystem.BILLBOARDMODE_STRETCHED_LOCAL) {\n this._vertexData[offset++] = particle.direction.x;\n this._vertexData[offset++] = particle.direction.y;\n this._vertexData[offset++] = particle.direction.z;\n }\n if (this._useRampGradients && particle.remapData) {\n this._vertexData[offset++] = particle.remapData.x;\n this._vertexData[offset++] = particle.remapData.y;\n this._vertexData[offset++] = particle.remapData.z;\n this._vertexData[offset++] = particle.remapData.w;\n }\n if (!this._useInstancing) {\n if (this._isAnimationSheetEnabled) {\n if (offsetX === 0) {\n offsetX = this._epsilon;\n }\n else if (offsetX === 1) {\n offsetX = 1 - this._epsilon;\n }\n if (offsetY === 0) {\n offsetY = this._epsilon;\n }\n else if (offsetY === 1) {\n offsetY = 1 - this._epsilon;\n }\n }\n this._vertexData[offset++] = offsetX;\n this._vertexData[offset++] = offsetY;\n }\n }\n _stopSubEmitters() {\n if (!this.activeSubSystems) {\n return;\n }\n this.activeSubSystems.forEach((subSystem) => {\n subSystem.stop(true);\n });\n this.activeSubSystems = new Array();\n }\n _removeFromRoot() {\n if (!this._rootParticleSystem) {\n return;\n }\n const index = this._rootParticleSystem.activeSubSystems.indexOf(this);\n if (index !== -1) {\n this._rootParticleSystem.activeSubSystems.splice(index, 1);\n }\n this._rootParticleSystem = null;\n }\n // End of sub system methods\n _update(newParticles) {\n // Update current\n this._alive = this._particles.length > 0;\n if (this.emitter.position) {\n const emitterMesh = this.emitter;\n this._emitterWorldMatrix = emitterMesh.getWorldMatrix();\n }\n else {\n const emitterPosition = this.emitter;\n this._emitterWorldMatrix = Matrix.Translation(emitterPosition.x, emitterPosition.y, emitterPosition.z);\n }\n this._emitterWorldMatrix.invertToRef(this._emitterInverseWorldMatrix);\n this.updateFunction(this._particles);\n // Add new ones\n let particle;\n for (let index = 0; index < newParticles; index++) {\n if (this._particles.length === this._capacity) {\n break;\n }\n particle = this._createParticle();\n this._particles.push(particle);\n // Life time\n if (this.targetStopDuration && this._lifeTimeGradients && this._lifeTimeGradients.length > 0) {\n const ratio = Scalar.Clamp(this._actualFrame / this.targetStopDuration);\n GradientHelper.GetCurrentGradient(ratio, this._lifeTimeGradients, (currentGradient, nextGradient) => {\n const factorGradient1 = currentGradient;\n const factorGradient2 = nextGradient;\n const lifeTime1 = factorGradient1.getFactor();\n const lifeTime2 = factorGradient2.getFactor();\n const gradient = (ratio - factorGradient1.gradient) / (factorGradient2.gradient - factorGradient1.gradient);\n particle.lifeTime = Scalar.Lerp(lifeTime1, lifeTime2, gradient);\n });\n }\n else {\n particle.lifeTime = Scalar.RandomRange(this.minLifeTime, this.maxLifeTime);\n }\n // Emitter\n const emitPower = Scalar.RandomRange(this.minEmitPower, this.maxEmitPower);\n if (this.startPositionFunction) {\n this.startPositionFunction(this._emitterWorldMatrix, particle.position, particle, this.isLocal);\n }\n else {\n this.particleEmitterType.startPositionFunction(this._emitterWorldMatrix, particle.position, particle, this.isLocal);\n }\n if (this.isLocal) {\n if (!particle._localPosition) {\n particle._localPosition = particle.position.clone();\n }\n else {\n particle._localPosition.copyFrom(particle.position);\n }\n Vector3.TransformCoordinatesToRef(particle._localPosition, this._emitterWorldMatrix, particle.position);\n }\n if (this.startDirectionFunction) {\n this.startDirectionFunction(this._emitterWorldMatrix, particle.direction, particle, this.isLocal);\n }\n else {\n this.particleEmitterType.startDirectionFunction(this._emitterWorldMatrix, particle.direction, particle, this.isLocal, this._emitterInverseWorldMatrix);\n }\n if (emitPower === 0) {\n if (!particle._initialDirection) {\n particle._initialDirection = particle.direction.clone();\n }\n else {\n particle._initialDirection.copyFrom(particle.direction);\n }\n }\n else {\n particle._initialDirection = null;\n }\n particle.direction.scaleInPlace(emitPower);\n // Size\n if (!this._sizeGradients || this._sizeGradients.length === 0) {\n particle.size = Scalar.RandomRange(this.minSize, this.maxSize);\n }\n else {\n particle._currentSizeGradient = this._sizeGradients[0];\n particle._currentSize1 = particle._currentSizeGradient.getFactor();\n particle.size = particle._currentSize1;\n if (this._sizeGradients.length > 1) {\n particle._currentSize2 = this._sizeGradients[1].getFactor();\n }\n else {\n particle._currentSize2 = particle._currentSize1;\n }\n }\n // Size and scale\n particle.scale.copyFromFloats(Scalar.RandomRange(this.minScaleX, this.maxScaleX), Scalar.RandomRange(this.minScaleY, this.maxScaleY));\n // Adjust scale by start size\n if (this._startSizeGradients && this._startSizeGradients[0] && this.targetStopDuration) {\n const ratio = this._actualFrame / this.targetStopDuration;\n GradientHelper.GetCurrentGradient(ratio, this._startSizeGradients, (currentGradient, nextGradient, scale) => {\n if (currentGradient !== this._currentStartSizeGradient) {\n this._currentStartSize1 = this._currentStartSize2;\n this._currentStartSize2 = nextGradient.getFactor();\n this._currentStartSizeGradient = currentGradient;\n }\n const value = Scalar.Lerp(this._currentStartSize1, this._currentStartSize2, scale);\n particle.scale.scaleInPlace(value);\n });\n }\n // Angle\n if (!this._angularSpeedGradients || this._angularSpeedGradients.length === 0) {\n particle.angularSpeed = Scalar.RandomRange(this.minAngularSpeed, this.maxAngularSpeed);\n }\n else {\n particle._currentAngularSpeedGradient = this._angularSpeedGradients[0];\n particle.angularSpeed = particle._currentAngularSpeedGradient.getFactor();\n particle._currentAngularSpeed1 = particle.angularSpeed;\n if (this._angularSpeedGradients.length > 1) {\n particle._currentAngularSpeed2 = this._angularSpeedGradients[1].getFactor();\n }\n else {\n particle._currentAngularSpeed2 = particle._currentAngularSpeed1;\n }\n }\n particle.angle = Scalar.RandomRange(this.minInitialRotation, this.maxInitialRotation);\n // Velocity\n if (this._velocityGradients && this._velocityGradients.length > 0) {\n particle._currentVelocityGradient = this._velocityGradients[0];\n particle._currentVelocity1 = particle._currentVelocityGradient.getFactor();\n if (this._velocityGradients.length > 1) {\n particle._currentVelocity2 = this._velocityGradients[1].getFactor();\n }\n else {\n particle._currentVelocity2 = particle._currentVelocity1;\n }\n }\n // Limit velocity\n if (this._limitVelocityGradients && this._limitVelocityGradients.length > 0) {\n particle._currentLimitVelocityGradient = this._limitVelocityGradients[0];\n particle._currentLimitVelocity1 = particle._currentLimitVelocityGradient.getFactor();\n if (this._limitVelocityGradients.length > 1) {\n particle._currentLimitVelocity2 = this._limitVelocityGradients[1].getFactor();\n }\n else {\n particle._currentLimitVelocity2 = particle._currentLimitVelocity1;\n }\n }\n // Drag\n if (this._dragGradients && this._dragGradients.length > 0) {\n particle._currentDragGradient = this._dragGradients[0];\n particle._currentDrag1 = particle._currentDragGradient.getFactor();\n if (this._dragGradients.length > 1) {\n particle._currentDrag2 = this._dragGradients[1].getFactor();\n }\n else {\n particle._currentDrag2 = particle._currentDrag1;\n }\n }\n // Color\n if (!this._colorGradients || this._colorGradients.length === 0) {\n const step = Scalar.RandomRange(0, 1.0);\n Color4.LerpToRef(this.color1, this.color2, step, particle.color);\n this.colorDead.subtractToRef(particle.color, this._colorDiff);\n this._colorDiff.scaleToRef(1.0 / particle.lifeTime, particle.colorStep);\n }\n else {\n particle._currentColorGradient = this._colorGradients[0];\n particle._currentColorGradient.getColorToRef(particle.color);\n particle._currentColor1.copyFrom(particle.color);\n if (this._colorGradients.length > 1) {\n this._colorGradients[1].getColorToRef(particle._currentColor2);\n }\n else {\n particle._currentColor2.copyFrom(particle.color);\n }\n }\n // Sheet\n if (this._isAnimationSheetEnabled) {\n particle._initialStartSpriteCellID = this.startSpriteCellID;\n particle._initialEndSpriteCellID = this.endSpriteCellID;\n particle._initialSpriteCellLoop = this.spriteCellLoop;\n }\n // Inherited Velocity\n particle.direction.addInPlace(this._inheritedVelocityOffset);\n // Ramp\n if (this._useRampGradients) {\n particle.remapData = new Vector4(0, 1, 0, 1);\n }\n // Noise texture coordinates\n if (this.noiseTexture) {\n if (particle._randomNoiseCoordinates1) {\n particle._randomNoiseCoordinates1.copyFromFloats(Math.random(), Math.random(), Math.random());\n particle._randomNoiseCoordinates2.copyFromFloats(Math.random(), Math.random(), Math.random());\n }\n else {\n particle._randomNoiseCoordinates1 = new Vector3(Math.random(), Math.random(), Math.random());\n particle._randomNoiseCoordinates2 = new Vector3(Math.random(), Math.random(), Math.random());\n }\n }\n // Update the position of the attached sub-emitters to match their attached particle\n particle._inheritParticleInfoToSubEmitters();\n }\n }\n /**\n * @internal\n */\n static _GetAttributeNamesOrOptions(isAnimationSheetEnabled = false, isBillboardBased = false, useRampGradients = false) {\n const attributeNamesOrOptions = [VertexBuffer.PositionKind, VertexBuffer.ColorKind, \"angle\", \"offset\", \"size\"];\n if (isAnimationSheetEnabled) {\n attributeNamesOrOptions.push(\"cellIndex\");\n }\n if (!isBillboardBased) {\n attributeNamesOrOptions.push(\"direction\");\n }\n if (useRampGradients) {\n attributeNamesOrOptions.push(\"remapData\");\n }\n return attributeNamesOrOptions;\n }\n /**\n * @internal\n */\n static _GetEffectCreationOptions(isAnimationSheetEnabled = false, useLogarithmicDepth = false) {\n const effectCreationOption = [\"invView\", \"view\", \"projection\", \"textureMask\", \"translationPivot\", \"eyePosition\"];\n addClipPlaneUniforms(effectCreationOption);\n if (isAnimationSheetEnabled) {\n effectCreationOption.push(\"particlesInfos\");\n }\n if (useLogarithmicDepth) {\n effectCreationOption.push(\"logarithmicDepthConstant\");\n }\n return effectCreationOption;\n }\n /**\n * Fill the defines array according to the current settings of the particle system\n * @param defines Array to be updated\n * @param blendMode blend mode to take into account when updating the array\n */\n fillDefines(defines, blendMode) {\n if (this._scene) {\n prepareStringDefinesForClipPlanes(this, this._scene, defines);\n }\n if (this._isAnimationSheetEnabled) {\n defines.push(\"#define ANIMATESHEET\");\n }\n if (this.useLogarithmicDepth) {\n defines.push(\"#define LOGARITHMICDEPTH\");\n }\n if (blendMode === ParticleSystem.BLENDMODE_MULTIPLY) {\n defines.push(\"#define BLENDMULTIPLYMODE\");\n }\n if (this._useRampGradients) {\n defines.push(\"#define RAMPGRADIENT\");\n }\n if (this._isBillboardBased) {\n defines.push(\"#define BILLBOARD\");\n switch (this.billboardMode) {\n case ParticleSystem.BILLBOARDMODE_Y:\n defines.push(\"#define BILLBOARDY\");\n break;\n case ParticleSystem.BILLBOARDMODE_STRETCHED:\n case ParticleSystem.BILLBOARDMODE_STRETCHED_LOCAL:\n defines.push(\"#define BILLBOARDSTRETCHED\");\n if (this.billboardMode === ParticleSystem.BILLBOARDMODE_STRETCHED_LOCAL) {\n defines.push(\"#define BILLBOARDSTRETCHED_LOCAL\");\n }\n break;\n case ParticleSystem.BILLBOARDMODE_ALL:\n defines.push(\"#define BILLBOARDMODE_ALL\");\n break;\n }\n }\n if (this._imageProcessingConfiguration) {\n this._imageProcessingConfiguration.prepareDefines(this._imageProcessingConfigurationDefines);\n defines.push(this._imageProcessingConfigurationDefines.toString());\n }\n }\n /**\n * Fill the uniforms, attributes and samplers arrays according to the current settings of the particle system\n * @param uniforms Uniforms array to fill\n * @param attributes Attributes array to fill\n * @param samplers Samplers array to fill\n */\n fillUniformsAttributesAndSamplerNames(uniforms, attributes, samplers) {\n attributes.push(...ParticleSystem._GetAttributeNamesOrOptions(this._isAnimationSheetEnabled, this._isBillboardBased && this.billboardMode !== ParticleSystem.BILLBOARDMODE_STRETCHED && this.billboardMode !== ParticleSystem.BILLBOARDMODE_STRETCHED_LOCAL, this._useRampGradients));\n uniforms.push(...ParticleSystem._GetEffectCreationOptions(this._isAnimationSheetEnabled, this.useLogarithmicDepth));\n samplers.push(\"diffuseSampler\", \"rampSampler\");\n if (this._imageProcessingConfiguration) {\n ImageProcessingConfiguration.PrepareUniforms(uniforms, this._imageProcessingConfigurationDefines);\n ImageProcessingConfiguration.PrepareSamplers(samplers, this._imageProcessingConfigurationDefines);\n }\n }\n /**\n * @internal\n */\n _getWrapper(blendMode) {\n const customWrapper = this._getCustomDrawWrapper(blendMode);\n if (customWrapper === null || customWrapper === void 0 ? void 0 : customWrapper.effect) {\n return customWrapper;\n }\n const defines = [];\n this.fillDefines(defines, blendMode);\n // Effect\n const currentRenderPassId = this._engine._features.supportRenderPasses ? this._engine.currentRenderPassId : 0;\n let drawWrappers = this._drawWrappers[currentRenderPassId];\n if (!drawWrappers) {\n drawWrappers = this._drawWrappers[currentRenderPassId] = [];\n }\n let drawWrapper = drawWrappers[blendMode];\n if (!drawWrapper) {\n drawWrapper = new DrawWrapper(this._engine);\n if (drawWrapper.drawContext) {\n drawWrapper.drawContext.useInstancing = this._useInstancing;\n }\n drawWrappers[blendMode] = drawWrapper;\n }\n const join = defines.join(\"\\n\");\n if (drawWrapper.defines !== join) {\n const attributesNamesOrOptions = [];\n const effectCreationOption = [];\n const samplers = [];\n this.fillUniformsAttributesAndSamplerNames(effectCreationOption, attributesNamesOrOptions, samplers);\n drawWrapper.setEffect(this._engine.createEffect(\"particles\", attributesNamesOrOptions, effectCreationOption, samplers, join), join);\n }\n return drawWrapper;\n }\n /**\n * Animates the particle system for the current frame by emitting new particles and or animating the living ones.\n * @param preWarmOnly will prevent the system from updating the vertex buffer (default is false)\n */\n animate(preWarmOnly = false) {\n var _a;\n if (!this._started) {\n return;\n }\n if (!preWarmOnly && this._scene) {\n // Check\n if (!this.isReady()) {\n return;\n }\n if (this._currentRenderId === this._scene.getFrameId()) {\n return;\n }\n this._currentRenderId = this._scene.getFrameId();\n }\n this._scaledUpdateSpeed = this.updateSpeed * (preWarmOnly ? this.preWarmStepOffset : ((_a = this._scene) === null || _a === void 0 ? void 0 : _a.getAnimationRatio()) || 1);\n // Determine the number of particles we need to create\n let newParticles;\n if (this.manualEmitCount > -1) {\n newParticles = this.manualEmitCount;\n this._newPartsExcess = 0;\n this.manualEmitCount = 0;\n }\n else {\n let rate = this.emitRate;\n if (this._emitRateGradients && this._emitRateGradients.length > 0 && this.targetStopDuration) {\n const ratio = this._actualFrame / this.targetStopDuration;\n GradientHelper.GetCurrentGradient(ratio, this._emitRateGradients, (currentGradient, nextGradient, scale) => {\n if (currentGradient !== this._currentEmitRateGradient) {\n this._currentEmitRate1 = this._currentEmitRate2;\n this._currentEmitRate2 = nextGradient.getFactor();\n this._currentEmitRateGradient = currentGradient;\n }\n rate = Scalar.Lerp(this._currentEmitRate1, this._currentEmitRate2, scale);\n });\n }\n newParticles = (rate * this._scaledUpdateSpeed) >> 0;\n this._newPartsExcess += rate * this._scaledUpdateSpeed - newParticles;\n }\n if (this._newPartsExcess > 1.0) {\n newParticles += this._newPartsExcess >> 0;\n this._newPartsExcess -= this._newPartsExcess >> 0;\n }\n this._alive = false;\n if (!this._stopped) {\n this._actualFrame += this._scaledUpdateSpeed;\n if (this.targetStopDuration && this._actualFrame >= this.targetStopDuration) {\n this.stop();\n }\n }\n else {\n newParticles = 0;\n }\n this._update(newParticles);\n // Stopped?\n if (this._stopped) {\n if (!this._alive) {\n this._started = false;\n if (this.onAnimationEnd) {\n this.onAnimationEnd();\n }\n if (this.disposeOnStop && this._scene) {\n this._scene._toBeDisposed.push(this);\n }\n }\n }\n if (!preWarmOnly) {\n // Update VBO\n let offset = 0;\n for (let index = 0; index < this._particles.length; index++) {\n const particle = this._particles[index];\n this._appendParticleVertices(offset, particle);\n offset += this._useInstancing ? 1 : 4;\n }\n if (this._vertexBuffer) {\n this._vertexBuffer.updateDirectly(this._vertexData, 0, this._particles.length);\n }\n }\n if (this.manualEmitCount === 0 && this.disposeOnStop) {\n this.stop();\n }\n }\n _appendParticleVertices(offset, particle) {\n this._appendParticleVertex(offset++, particle, 0, 0);\n if (!this._useInstancing) {\n this._appendParticleVertex(offset++, particle, 1, 0);\n this._appendParticleVertex(offset++, particle, 1, 1);\n this._appendParticleVertex(offset++, particle, 0, 1);\n }\n }\n /**\n * Rebuilds the particle system.\n */\n rebuild() {\n var _a, _b;\n if (this._engine.getCaps().vertexArrayObject) {\n this._vertexArrayObject = null;\n }\n this._createIndexBuffer();\n (_a = this._spriteBuffer) === null || _a === void 0 ? void 0 : _a._rebuild();\n (_b = this._vertexBuffer) === null || _b === void 0 ? void 0 : _b._rebuild();\n for (const key in this._vertexBuffers) {\n this._vertexBuffers[key]._rebuild();\n }\n this.resetDrawCache();\n }\n /**\n * Is this system ready to be used/rendered\n * @returns true if the system is ready\n */\n isReady() {\n if (!this.emitter || (this._imageProcessingConfiguration && !this._imageProcessingConfiguration.isReady()) || !this.particleTexture || !this.particleTexture.isReady()) {\n return false;\n }\n if (this.blendMode !== ParticleSystem.BLENDMODE_MULTIPLYADD) {\n if (!this._getWrapper(this.blendMode).effect.isReady()) {\n return false;\n }\n }\n else {\n if (!this._getWrapper(ParticleSystem.BLENDMODE_MULTIPLY).effect.isReady()) {\n return false;\n }\n if (!this._getWrapper(ParticleSystem.BLENDMODE_ADD).effect.isReady()) {\n return false;\n }\n }\n return true;\n }\n _render(blendMode) {\n var _a, _b;\n const drawWrapper = this._getWrapper(blendMode);\n const effect = drawWrapper.effect;\n const engine = this._engine;\n // Render\n engine.enableEffect(drawWrapper);\n const viewMatrix = (_a = this.defaultViewMatrix) !== null && _a !== void 0 ? _a : this._scene.getViewMatrix();\n effect.setTexture(\"diffuseSampler\", this.particleTexture);\n effect.setMatrix(\"view\", viewMatrix);\n effect.setMatrix(\"projection\", (_b = this.defaultProjectionMatrix) !== null && _b !== void 0 ? _b : this._scene.getProjectionMatrix());\n if (this._isAnimationSheetEnabled && this.particleTexture) {\n const baseSize = this.particleTexture.getBaseSize();\n effect.setFloat3(\"particlesInfos\", this.spriteCellWidth / baseSize.width, this.spriteCellHeight / baseSize.height, this.spriteCellWidth / baseSize.width);\n }\n effect.setVector2(\"translationPivot\", this.translationPivot);\n effect.setFloat4(\"textureMask\", this.textureMask.r, this.textureMask.g, this.textureMask.b, this.textureMask.a);\n if (this._isBillboardBased && this._scene) {\n const camera = this._scene.activeCamera;\n effect.setVector3(\"eyePosition\", camera.globalPosition);\n }\n if (this._rampGradientsTexture) {\n if (!this._rampGradients || !this._rampGradients.length) {\n this._rampGradientsTexture.dispose();\n this._rampGradientsTexture = null;\n }\n effect.setTexture(\"rampSampler\", this._rampGradientsTexture);\n }\n const defines = effect.defines;\n if (this._scene) {\n bindClipPlane(effect, this, this._scene);\n }\n if (defines.indexOf(\"#define BILLBOARDMODE_ALL\") >= 0) {\n viewMatrix.invertToRef(TmpVectors.Matrix[0]);\n effect.setMatrix(\"invView\", TmpVectors.Matrix[0]);\n }\n if (this._vertexArrayObject !== undefined) {\n if (!this._vertexArrayObject) {\n this._vertexArrayObject = this._engine.recordVertexArrayObject(this._vertexBuffers, this._indexBuffer, effect);\n }\n this._engine.bindVertexArrayObject(this._vertexArrayObject, this._indexBuffer);\n }\n else {\n engine.bindBuffers(this._vertexBuffers, this._indexBuffer, effect);\n }\n // Log. depth\n if (this.useLogarithmicDepth && this._scene) {\n MaterialHelper.BindLogDepth(defines, effect, this._scene);\n }\n // image processing\n if (this._imageProcessingConfiguration && !this._imageProcessingConfiguration.applyByPostProcess) {\n this._imageProcessingConfiguration.bind(effect);\n }\n // Draw order\n switch (blendMode) {\n case ParticleSystem.BLENDMODE_ADD:\n engine.setAlphaMode(1);\n break;\n case ParticleSystem.BLENDMODE_ONEONE:\n engine.setAlphaMode(6);\n break;\n case ParticleSystem.BLENDMODE_STANDARD:\n engine.setAlphaMode(2);\n break;\n case ParticleSystem.BLENDMODE_MULTIPLY:\n engine.setAlphaMode(4);\n break;\n }\n if (this._onBeforeDrawParticlesObservable) {\n this._onBeforeDrawParticlesObservable.notifyObservers(effect);\n }\n if (this._useInstancing) {\n engine.drawArraysType(7, 0, 4, this._particles.length);\n }\n else {\n engine.drawElementsType(0, 0, this._particles.length * 6);\n }\n return this._particles.length;\n }\n /**\n * Renders the particle system in its current state.\n * @returns the current number of particles\n */\n render() {\n // Check\n if (!this.isReady() || !this._particles.length) {\n return 0;\n }\n const engine = this._engine;\n if (engine.setState) {\n engine.setState(false);\n if (this.forceDepthWrite) {\n engine.setDepthWrite(true);\n }\n }\n let outparticles = 0;\n if (this.blendMode === ParticleSystem.BLENDMODE_MULTIPLYADD) {\n outparticles = this._render(ParticleSystem.BLENDMODE_MULTIPLY) + this._render(ParticleSystem.BLENDMODE_ADD);\n }\n else {\n outparticles = this._render(this.blendMode);\n }\n this._engine.unbindInstanceAttributes();\n this._engine.setAlphaMode(0);\n return outparticles;\n }\n /**\n * Disposes the particle system and free the associated resources\n * @param disposeTexture defines if the particle texture must be disposed as well (true by default)\n */\n dispose(disposeTexture = true) {\n this.resetDrawCache();\n if (this._vertexBuffer) {\n this._vertexBuffer.dispose();\n this._vertexBuffer = null;\n }\n if (this._spriteBuffer) {\n this._spriteBuffer.dispose();\n this._spriteBuffer = null;\n }\n if (this._indexBuffer) {\n this._engine._releaseBuffer(this._indexBuffer);\n this._indexBuffer = null;\n }\n if (this._vertexArrayObject) {\n this._engine.releaseVertexArrayObject(this._vertexArrayObject);\n this._vertexArrayObject = null;\n }\n if (disposeTexture && this.particleTexture) {\n this.particleTexture.dispose();\n this.particleTexture = null;\n }\n if (disposeTexture && this.noiseTexture) {\n this.noiseTexture.dispose();\n this.noiseTexture = null;\n }\n if (this._rampGradientsTexture) {\n this._rampGradientsTexture.dispose();\n this._rampGradientsTexture = null;\n }\n this._removeFromRoot();\n if (this.subEmitters && !this._subEmitters) {\n this._prepareSubEmitterInternalArray();\n }\n if (this._subEmitters && this._subEmitters.length) {\n for (let index = 0; index < this._subEmitters.length; index++) {\n for (const subEmitter of this._subEmitters[index]) {\n subEmitter.dispose();\n }\n }\n this._subEmitters = [];\n this.subEmitters = [];\n }\n if (this._disposeEmitterOnDispose && this.emitter && this.emitter.dispose) {\n this.emitter.dispose(true);\n }\n if (this._onBeforeDrawParticlesObservable) {\n this._onBeforeDrawParticlesObservable.clear();\n }\n // Remove from scene\n if (this._scene) {\n const index = this._scene.particleSystems.indexOf(this);\n if (index > -1) {\n this._scene.particleSystems.splice(index, 1);\n }\n this._scene._activeParticleSystems.dispose();\n }\n // Callback\n this.onDisposeObservable.notifyObservers(this);\n this.onDisposeObservable.clear();\n this.onStoppedObservable.clear();\n this.reset();\n }\n // Clone\n /**\n * Clones the particle system.\n * @param name The name of the cloned object\n * @param newEmitter The new emitter to use\n * @param cloneTexture Also clone the textures if true\n * @returns the cloned particle system\n */\n clone(name, newEmitter, cloneTexture = false) {\n const custom = Object.assign({}, this._customWrappers);\n let program = null;\n const engine = this._engine;\n if (engine.createEffectForParticles) {\n if (this.customShader != null) {\n program = this.customShader;\n const defines = program.shaderOptions.defines.length > 0 ? program.shaderOptions.defines.join(\"\\n\") : \"\";\n const effect = engine.createEffectForParticles(program.shaderPath.fragmentElement, program.shaderOptions.uniforms, program.shaderOptions.samplers, defines);\n if (!custom[0]) {\n this.setCustomEffect(effect, 0);\n }\n else {\n custom[0].effect = effect;\n }\n }\n }\n const serialization = this.serialize(cloneTexture);\n const result = ParticleSystem.Parse(serialization, this._scene || this._engine, this._rootUrl);\n result.name = name;\n result.customShader = program;\n result._customWrappers = custom;\n if (newEmitter === undefined) {\n newEmitter = this.emitter;\n }\n if (this.noiseTexture) {\n result.noiseTexture = this.noiseTexture.clone();\n }\n result.emitter = newEmitter;\n if (!this.preventAutoStart) {\n result.start();\n }\n return result;\n }\n /**\n * Serializes the particle system to a JSON object\n * @param serializeTexture defines if the texture must be serialized as well\n * @returns the JSON object\n */\n serialize(serializeTexture = false) {\n const serializationObject = {};\n ParticleSystem._Serialize(serializationObject, this, serializeTexture);\n serializationObject.textureMask = this.textureMask.asArray();\n serializationObject.customShader = this.customShader;\n serializationObject.preventAutoStart = this.preventAutoStart;\n // SubEmitters\n if (this.subEmitters) {\n serializationObject.subEmitters = [];\n if (!this._subEmitters) {\n this._prepareSubEmitterInternalArray();\n }\n for (const subs of this._subEmitters) {\n const cell = [];\n for (const sub of subs) {\n cell.push(sub.serialize(serializeTexture));\n }\n serializationObject.subEmitters.push(cell);\n }\n }\n return serializationObject;\n }\n /**\n * @internal\n */\n static _Serialize(serializationObject, particleSystem, serializeTexture) {\n serializationObject.name = particleSystem.name;\n serializationObject.id = particleSystem.id;\n serializationObject.capacity = particleSystem.getCapacity();\n serializationObject.disposeOnStop = particleSystem.disposeOnStop;\n serializationObject.manualEmitCount = particleSystem.manualEmitCount;\n // Emitter\n if (particleSystem.emitter.position) {\n const emitterMesh = particleSystem.emitter;\n serializationObject.emitterId = emitterMesh.id;\n }\n else {\n const emitterPosition = particleSystem.emitter;\n serializationObject.emitter = emitterPosition.asArray();\n }\n // Emitter\n if (particleSystem.particleEmitterType) {\n serializationObject.particleEmitterType = particleSystem.particleEmitterType.serialize();\n }\n if (particleSystem.particleTexture) {\n if (serializeTexture) {\n serializationObject.texture = particleSystem.particleTexture.serialize();\n }\n else {\n serializationObject.textureName = particleSystem.particleTexture.name;\n serializationObject.invertY = !!particleSystem.particleTexture._invertY;\n }\n }\n serializationObject.isLocal = particleSystem.isLocal;\n // Animations\n SerializationHelper.AppendSerializedAnimations(particleSystem, serializationObject);\n serializationObject.beginAnimationOnStart = particleSystem.beginAnimationOnStart;\n serializationObject.beginAnimationFrom = particleSystem.beginAnimationFrom;\n serializationObject.beginAnimationTo = particleSystem.beginAnimationTo;\n serializationObject.beginAnimationLoop = particleSystem.beginAnimationLoop;\n // Particle system\n serializationObject.startDelay = particleSystem.startDelay;\n serializationObject.renderingGroupId = particleSystem.renderingGroupId;\n serializationObject.isBillboardBased = particleSystem.isBillboardBased;\n serializationObject.billboardMode = particleSystem.billboardMode;\n serializationObject.minAngularSpeed = particleSystem.minAngularSpeed;\n serializationObject.maxAngularSpeed = particleSystem.maxAngularSpeed;\n serializationObject.minSize = particleSystem.minSize;\n serializationObject.maxSize = particleSystem.maxSize;\n serializationObject.minScaleX = particleSystem.minScaleX;\n serializationObject.maxScaleX = particleSystem.maxScaleX;\n serializationObject.minScaleY = particleSystem.minScaleY;\n serializationObject.maxScaleY = particleSystem.maxScaleY;\n serializationObject.minEmitPower = particleSystem.minEmitPower;\n serializationObject.maxEmitPower = particleSystem.maxEmitPower;\n serializationObject.minLifeTime = particleSystem.minLifeTime;\n serializationObject.maxLifeTime = particleSystem.maxLifeTime;\n serializationObject.emitRate = particleSystem.emitRate;\n serializationObject.gravity = particleSystem.gravity.asArray();\n serializationObject.noiseStrength = particleSystem.noiseStrength.asArray();\n serializationObject.color1 = particleSystem.color1.asArray();\n serializationObject.color2 = particleSystem.color2.asArray();\n serializationObject.colorDead = particleSystem.colorDead.asArray();\n serializationObject.updateSpeed = particleSystem.updateSpeed;\n serializationObject.targetStopDuration = particleSystem.targetStopDuration;\n serializationObject.blendMode = particleSystem.blendMode;\n serializationObject.preWarmCycles = particleSystem.preWarmCycles;\n serializationObject.preWarmStepOffset = particleSystem.preWarmStepOffset;\n serializationObject.minInitialRotation = particleSystem.minInitialRotation;\n serializationObject.maxInitialRotation = particleSystem.maxInitialRotation;\n serializationObject.startSpriteCellID = particleSystem.startSpriteCellID;\n serializationObject.spriteCellLoop = particleSystem.spriteCellLoop;\n serializationObject.endSpriteCellID = particleSystem.endSpriteCellID;\n serializationObject.spriteCellChangeSpeed = particleSystem.spriteCellChangeSpeed;\n serializationObject.spriteCellWidth = particleSystem.spriteCellWidth;\n serializationObject.spriteCellHeight = particleSystem.spriteCellHeight;\n serializationObject.spriteRandomStartCell = particleSystem.spriteRandomStartCell;\n serializationObject.isAnimationSheetEnabled = particleSystem.isAnimationSheetEnabled;\n serializationObject.useLogarithmicDepth = particleSystem.useLogarithmicDepth;\n const colorGradients = particleSystem.getColorGradients();\n if (colorGradients) {\n serializationObject.colorGradients = [];\n for (const colorGradient of colorGradients) {\n const serializedGradient = {\n gradient: colorGradient.gradient,\n color1: colorGradient.color1.asArray(),\n };\n if (colorGradient.color2) {\n serializedGradient.color2 = colorGradient.color2.asArray();\n }\n else {\n serializedGradient.color2 = colorGradient.color1.asArray();\n }\n serializationObject.colorGradients.push(serializedGradient);\n }\n }\n const rampGradients = particleSystem.getRampGradients();\n if (rampGradients) {\n serializationObject.rampGradients = [];\n for (const rampGradient of rampGradients) {\n const serializedGradient = {\n gradient: rampGradient.gradient,\n color: rampGradient.color.asArray(),\n };\n serializationObject.rampGradients.push(serializedGradient);\n }\n serializationObject.useRampGradients = particleSystem.useRampGradients;\n }\n const colorRemapGradients = particleSystem.getColorRemapGradients();\n if (colorRemapGradients) {\n serializationObject.colorRemapGradients = [];\n for (const colorRemapGradient of colorRemapGradients) {\n const serializedGradient = {\n gradient: colorRemapGradient.gradient,\n factor1: colorRemapGradient.factor1,\n };\n if (colorRemapGradient.factor2 !== undefined) {\n serializedGradient.factor2 = colorRemapGradient.factor2;\n }\n else {\n serializedGradient.factor2 = colorRemapGradient.factor1;\n }\n serializationObject.colorRemapGradients.push(serializedGradient);\n }\n }\n const alphaRemapGradients = particleSystem.getAlphaRemapGradients();\n if (alphaRemapGradients) {\n serializationObject.alphaRemapGradients = [];\n for (const alphaRemapGradient of alphaRemapGradients) {\n const serializedGradient = {\n gradient: alphaRemapGradient.gradient,\n factor1: alphaRemapGradient.factor1,\n };\n if (alphaRemapGradient.factor2 !== undefined) {\n serializedGradient.factor2 = alphaRemapGradient.factor2;\n }\n else {\n serializedGradient.factor2 = alphaRemapGradient.factor1;\n }\n serializationObject.alphaRemapGradients.push(serializedGradient);\n }\n }\n const sizeGradients = particleSystem.getSizeGradients();\n if (sizeGradients) {\n serializationObject.sizeGradients = [];\n for (const sizeGradient of sizeGradients) {\n const serializedGradient = {\n gradient: sizeGradient.gradient,\n factor1: sizeGradient.factor1,\n };\n if (sizeGradient.factor2 !== undefined) {\n serializedGradient.factor2 = sizeGradient.factor2;\n }\n else {\n serializedGradient.factor2 = sizeGradient.factor1;\n }\n serializationObject.sizeGradients.push(serializedGradient);\n }\n }\n const angularSpeedGradients = particleSystem.getAngularSpeedGradients();\n if (angularSpeedGradients) {\n serializationObject.angularSpeedGradients = [];\n for (const angularSpeedGradient of angularSpeedGradients) {\n const serializedGradient = {\n gradient: angularSpeedGradient.gradient,\n factor1: angularSpeedGradient.factor1,\n };\n if (angularSpeedGradient.factor2 !== undefined) {\n serializedGradient.factor2 = angularSpeedGradient.factor2;\n }\n else {\n serializedGradient.factor2 = angularSpeedGradient.factor1;\n }\n serializationObject.angularSpeedGradients.push(serializedGradient);\n }\n }\n const velocityGradients = particleSystem.getVelocityGradients();\n if (velocityGradients) {\n serializationObject.velocityGradients = [];\n for (const velocityGradient of velocityGradients) {\n const serializedGradient = {\n gradient: velocityGradient.gradient,\n factor1: velocityGradient.factor1,\n };\n if (velocityGradient.factor2 !== undefined) {\n serializedGradient.factor2 = velocityGradient.factor2;\n }\n else {\n serializedGradient.factor2 = velocityGradient.factor1;\n }\n serializationObject.velocityGradients.push(serializedGradient);\n }\n }\n const dragGradients = particleSystem.getDragGradients();\n if (dragGradients) {\n serializationObject.dragGradients = [];\n for (const dragGradient of dragGradients) {\n const serializedGradient = {\n gradient: dragGradient.gradient,\n factor1: dragGradient.factor1,\n };\n if (dragGradient.factor2 !== undefined) {\n serializedGradient.factor2 = dragGradient.factor2;\n }\n else {\n serializedGradient.factor2 = dragGradient.factor1;\n }\n serializationObject.dragGradients.push(serializedGradient);\n }\n }\n const emitRateGradients = particleSystem.getEmitRateGradients();\n if (emitRateGradients) {\n serializationObject.emitRateGradients = [];\n for (const emitRateGradient of emitRateGradients) {\n const serializedGradient = {\n gradient: emitRateGradient.gradient,\n factor1: emitRateGradient.factor1,\n };\n if (emitRateGradient.factor2 !== undefined) {\n serializedGradient.factor2 = emitRateGradient.factor2;\n }\n else {\n serializedGradient.factor2 = emitRateGradient.factor1;\n }\n serializationObject.emitRateGradients.push(serializedGradient);\n }\n }\n const startSizeGradients = particleSystem.getStartSizeGradients();\n if (startSizeGradients) {\n serializationObject.startSizeGradients = [];\n for (const startSizeGradient of startSizeGradients) {\n const serializedGradient = {\n gradient: startSizeGradient.gradient,\n factor1: startSizeGradient.factor1,\n };\n if (startSizeGradient.factor2 !== undefined) {\n serializedGradient.factor2 = startSizeGradient.factor2;\n }\n else {\n serializedGradient.factor2 = startSizeGradient.factor1;\n }\n serializationObject.startSizeGradients.push(serializedGradient);\n }\n }\n const lifeTimeGradients = particleSystem.getLifeTimeGradients();\n if (lifeTimeGradients) {\n serializationObject.lifeTimeGradients = [];\n for (const lifeTimeGradient of lifeTimeGradients) {\n const serializedGradient = {\n gradient: lifeTimeGradient.gradient,\n factor1: lifeTimeGradient.factor1,\n };\n if (lifeTimeGradient.factor2 !== undefined) {\n serializedGradient.factor2 = lifeTimeGradient.factor2;\n }\n else {\n serializedGradient.factor2 = lifeTimeGradient.factor1;\n }\n serializationObject.lifeTimeGradients.push(serializedGradient);\n }\n }\n const limitVelocityGradients = particleSystem.getLimitVelocityGradients();\n if (limitVelocityGradients) {\n serializationObject.limitVelocityGradients = [];\n for (const limitVelocityGradient of limitVelocityGradients) {\n const serializedGradient = {\n gradient: limitVelocityGradient.gradient,\n factor1: limitVelocityGradient.factor1,\n };\n if (limitVelocityGradient.factor2 !== undefined) {\n serializedGradient.factor2 = limitVelocityGradient.factor2;\n }\n else {\n serializedGradient.factor2 = limitVelocityGradient.factor1;\n }\n serializationObject.limitVelocityGradients.push(serializedGradient);\n }\n serializationObject.limitVelocityDamping = particleSystem.limitVelocityDamping;\n }\n if (particleSystem.noiseTexture) {\n serializationObject.noiseTexture = particleSystem.noiseTexture.serialize();\n }\n }\n /**\n * @internal\n */\n static _Parse(parsedParticleSystem, particleSystem, sceneOrEngine, rootUrl) {\n var _a, _b, _c;\n let scene;\n if (sceneOrEngine instanceof ThinEngine) {\n scene = null;\n }\n else {\n scene = sceneOrEngine;\n }\n const internalClass = GetClass(\"BABYLON.Texture\");\n if (internalClass && scene) {\n // Texture\n if (parsedParticleSystem.texture) {\n particleSystem.particleTexture = internalClass.Parse(parsedParticleSystem.texture, scene, rootUrl);\n }\n else if (parsedParticleSystem.textureName) {\n particleSystem.particleTexture = new internalClass(rootUrl + parsedParticleSystem.textureName, scene, false, parsedParticleSystem.invertY !== undefined ? parsedParticleSystem.invertY : true);\n particleSystem.particleTexture.name = parsedParticleSystem.textureName;\n }\n }\n // Emitter\n if (!parsedParticleSystem.emitterId && parsedParticleSystem.emitterId !== 0 && parsedParticleSystem.emitter === undefined) {\n particleSystem.emitter = Vector3.Zero();\n }\n else if (parsedParticleSystem.emitterId && scene) {\n particleSystem.emitter = scene.getLastMeshById(parsedParticleSystem.emitterId);\n }\n else {\n particleSystem.emitter = Vector3.FromArray(parsedParticleSystem.emitter);\n }\n particleSystem.isLocal = !!parsedParticleSystem.isLocal;\n // Misc.\n if (parsedParticleSystem.renderingGroupId !== undefined) {\n particleSystem.renderingGroupId = parsedParticleSystem.renderingGroupId;\n }\n if (parsedParticleSystem.isBillboardBased !== undefined) {\n particleSystem.isBillboardBased = parsedParticleSystem.isBillboardBased;\n }\n if (parsedParticleSystem.billboardMode !== undefined) {\n particleSystem.billboardMode = parsedParticleSystem.billboardMode;\n }\n if (parsedParticleSystem.useLogarithmicDepth !== undefined) {\n particleSystem.useLogarithmicDepth = parsedParticleSystem.useLogarithmicDepth;\n }\n // Animations\n if (parsedParticleSystem.animations) {\n for (let animationIndex = 0; animationIndex < parsedParticleSystem.animations.length; animationIndex++) {\n const parsedAnimation = parsedParticleSystem.animations[animationIndex];\n const internalClass = GetClass(\"BABYLON.Animation\");\n if (internalClass) {\n particleSystem.animations.push(internalClass.Parse(parsedAnimation));\n }\n }\n particleSystem.beginAnimationOnStart = parsedParticleSystem.beginAnimationOnStart;\n particleSystem.beginAnimationFrom = parsedParticleSystem.beginAnimationFrom;\n particleSystem.beginAnimationTo = parsedParticleSystem.beginAnimationTo;\n particleSystem.beginAnimationLoop = parsedParticleSystem.beginAnimationLoop;\n }\n if (parsedParticleSystem.autoAnimate && scene) {\n scene.beginAnimation(particleSystem, parsedParticleSystem.autoAnimateFrom, parsedParticleSystem.autoAnimateTo, parsedParticleSystem.autoAnimateLoop, parsedParticleSystem.autoAnimateSpeed || 1.0);\n }\n // Particle system\n particleSystem.startDelay = parsedParticleSystem.startDelay | 0;\n particleSystem.minAngularSpeed = parsedParticleSystem.minAngularSpeed;\n particleSystem.maxAngularSpeed = parsedParticleSystem.maxAngularSpeed;\n particleSystem.minSize = parsedParticleSystem.minSize;\n particleSystem.maxSize = parsedParticleSystem.maxSize;\n if (parsedParticleSystem.minScaleX) {\n particleSystem.minScaleX = parsedParticleSystem.minScaleX;\n particleSystem.maxScaleX = parsedParticleSystem.maxScaleX;\n particleSystem.minScaleY = parsedParticleSystem.minScaleY;\n particleSystem.maxScaleY = parsedParticleSystem.maxScaleY;\n }\n if (parsedParticleSystem.preWarmCycles !== undefined) {\n particleSystem.preWarmCycles = parsedParticleSystem.preWarmCycles;\n particleSystem.preWarmStepOffset = parsedParticleSystem.preWarmStepOffset;\n }\n if (parsedParticleSystem.minInitialRotation !== undefined) {\n particleSystem.minInitialRotation = parsedParticleSystem.minInitialRotation;\n particleSystem.maxInitialRotation = parsedParticleSystem.maxInitialRotation;\n }\n particleSystem.minLifeTime = parsedParticleSystem.minLifeTime;\n particleSystem.maxLifeTime = parsedParticleSystem.maxLifeTime;\n particleSystem.minEmitPower = parsedParticleSystem.minEmitPower;\n particleSystem.maxEmitPower = parsedParticleSystem.maxEmitPower;\n particleSystem.emitRate = parsedParticleSystem.emitRate;\n particleSystem.gravity = Vector3.FromArray(parsedParticleSystem.gravity);\n if (parsedParticleSystem.noiseStrength) {\n particleSystem.noiseStrength = Vector3.FromArray(parsedParticleSystem.noiseStrength);\n }\n particleSystem.color1 = Color4.FromArray(parsedParticleSystem.color1);\n particleSystem.color2 = Color4.FromArray(parsedParticleSystem.color2);\n particleSystem.colorDead = Color4.FromArray(parsedParticleSystem.colorDead);\n particleSystem.updateSpeed = parsedParticleSystem.updateSpeed;\n particleSystem.targetStopDuration = parsedParticleSystem.targetStopDuration;\n particleSystem.blendMode = parsedParticleSystem.blendMode;\n if (parsedParticleSystem.colorGradients) {\n for (const colorGradient of parsedParticleSystem.colorGradients) {\n particleSystem.addColorGradient(colorGradient.gradient, Color4.FromArray(colorGradient.color1), colorGradient.color2 ? Color4.FromArray(colorGradient.color2) : undefined);\n }\n }\n if (parsedParticleSystem.rampGradients) {\n for (const rampGradient of parsedParticleSystem.rampGradients) {\n particleSystem.addRampGradient(rampGradient.gradient, Color3.FromArray(rampGradient.color));\n }\n particleSystem.useRampGradients = parsedParticleSystem.useRampGradients;\n }\n if (parsedParticleSystem.colorRemapGradients) {\n for (const colorRemapGradient of parsedParticleSystem.colorRemapGradients) {\n particleSystem.addColorRemapGradient(colorRemapGradient.gradient, colorRemapGradient.factor1 !== undefined ? colorRemapGradient.factor1 : colorRemapGradient.factor, colorRemapGradient.factor2);\n }\n }\n if (parsedParticleSystem.alphaRemapGradients) {\n for (const alphaRemapGradient of parsedParticleSystem.alphaRemapGradients) {\n particleSystem.addAlphaRemapGradient(alphaRemapGradient.gradient, alphaRemapGradient.factor1 !== undefined ? alphaRemapGradient.factor1 : alphaRemapGradient.factor, alphaRemapGradient.factor2);\n }\n }\n if (parsedParticleSystem.sizeGradients) {\n for (const sizeGradient of parsedParticleSystem.sizeGradients) {\n particleSystem.addSizeGradient(sizeGradient.gradient, sizeGradient.factor1 !== undefined ? sizeGradient.factor1 : sizeGradient.factor, sizeGradient.factor2);\n }\n }\n if (parsedParticleSystem.angularSpeedGradients) {\n for (const angularSpeedGradient of parsedParticleSystem.angularSpeedGradients) {\n particleSystem.addAngularSpeedGradient(angularSpeedGradient.gradient, angularSpeedGradient.factor1 !== undefined ? angularSpeedGradient.factor1 : angularSpeedGradient.factor, angularSpeedGradient.factor2);\n }\n }\n if (parsedParticleSystem.velocityGradients) {\n for (const velocityGradient of parsedParticleSystem.velocityGradients) {\n particleSystem.addVelocityGradient(velocityGradient.gradient, velocityGradient.factor1 !== undefined ? velocityGradient.factor1 : velocityGradient.factor, velocityGradient.factor2);\n }\n }\n if (parsedParticleSystem.dragGradients) {\n for (const dragGradient of parsedParticleSystem.dragGradients) {\n particleSystem.addDragGradient(dragGradient.gradient, dragGradient.factor1 !== undefined ? dragGradient.factor1 : dragGradient.factor, dragGradient.factor2);\n }\n }\n if (parsedParticleSystem.emitRateGradients) {\n for (const emitRateGradient of parsedParticleSystem.emitRateGradients) {\n particleSystem.addEmitRateGradient(emitRateGradient.gradient, emitRateGradient.factor1 !== undefined ? emitRateGradient.factor1 : emitRateGradient.factor, emitRateGradient.factor2);\n }\n }\n if (parsedParticleSystem.startSizeGradients) {\n for (const startSizeGradient of parsedParticleSystem.startSizeGradients) {\n particleSystem.addStartSizeGradient(startSizeGradient.gradient, startSizeGradient.factor1 !== undefined ? startSizeGradient.factor1 : startSizeGradient.factor, startSizeGradient.factor2);\n }\n }\n if (parsedParticleSystem.lifeTimeGradients) {\n for (const lifeTimeGradient of parsedParticleSystem.lifeTimeGradients) {\n particleSystem.addLifeTimeGradient(lifeTimeGradient.gradient, lifeTimeGradient.factor1 !== undefined ? lifeTimeGradient.factor1 : lifeTimeGradient.factor, lifeTimeGradient.factor2);\n }\n }\n if (parsedParticleSystem.limitVelocityGradients) {\n for (const limitVelocityGradient of parsedParticleSystem.limitVelocityGradients) {\n particleSystem.addLimitVelocityGradient(limitVelocityGradient.gradient, limitVelocityGradient.factor1 !== undefined ? limitVelocityGradient.factor1 : limitVelocityGradient.factor, limitVelocityGradient.factor2);\n }\n particleSystem.limitVelocityDamping = parsedParticleSystem.limitVelocityDamping;\n }\n if (parsedParticleSystem.noiseTexture && scene) {\n const internalClass = GetClass(\"BABYLON.ProceduralTexture\");\n particleSystem.noiseTexture = internalClass.Parse(parsedParticleSystem.noiseTexture, scene, rootUrl);\n }\n // Emitter\n let emitterType;\n if (parsedParticleSystem.particleEmitterType) {\n switch (parsedParticleSystem.particleEmitterType.type) {\n case \"SphereParticleEmitter\":\n emitterType = new SphereParticleEmitter();\n break;\n case \"SphereDirectedParticleEmitter\":\n emitterType = new SphereDirectedParticleEmitter();\n break;\n case \"ConeEmitter\":\n case \"ConeParticleEmitter\":\n emitterType = new ConeParticleEmitter();\n break;\n case \"CylinderParticleEmitter\":\n emitterType = new CylinderParticleEmitter();\n break;\n case \"CylinderDirectedParticleEmitter\":\n emitterType = new CylinderDirectedParticleEmitter();\n break;\n case \"HemisphericParticleEmitter\":\n emitterType = new HemisphericParticleEmitter();\n break;\n case \"PointParticleEmitter\":\n emitterType = new PointParticleEmitter();\n break;\n case \"MeshParticleEmitter\":\n emitterType = new MeshParticleEmitter();\n break;\n case \"BoxEmitter\":\n case \"BoxParticleEmitter\":\n default:\n emitterType = new BoxParticleEmitter();\n break;\n }\n emitterType.parse(parsedParticleSystem.particleEmitterType, scene);\n }\n else {\n emitterType = new BoxParticleEmitter();\n emitterType.parse(parsedParticleSystem, scene);\n }\n particleSystem.particleEmitterType = emitterType;\n // Animation sheet\n particleSystem.startSpriteCellID = parsedParticleSystem.startSpriteCellID;\n particleSystem.endSpriteCellID = parsedParticleSystem.endSpriteCellID;\n particleSystem.spriteCellLoop = (_a = parsedParticleSystem.spriteCellLoop) !== null && _a !== void 0 ? _a : true;\n particleSystem.spriteCellWidth = parsedParticleSystem.spriteCellWidth;\n particleSystem.spriteCellHeight = parsedParticleSystem.spriteCellHeight;\n particleSystem.spriteCellChangeSpeed = parsedParticleSystem.spriteCellChangeSpeed;\n particleSystem.spriteRandomStartCell = parsedParticleSystem.spriteRandomStartCell;\n particleSystem.disposeOnStop = (_b = parsedParticleSystem.disposeOnStop) !== null && _b !== void 0 ? _b : false;\n particleSystem.manualEmitCount = (_c = parsedParticleSystem.manualEmitCount) !== null && _c !== void 0 ? _c : -1;\n }\n /**\n * Parses a JSON object to create a particle system.\n * @param parsedParticleSystem The JSON object to parse\n * @param sceneOrEngine The scene or the engine to create the particle system in\n * @param rootUrl The root url to use to load external dependencies like texture\n * @param doNotStart Ignore the preventAutoStart attribute and does not start\n * @param capacity defines the system capacity (if null or undefined the sotred capacity will be used)\n * @returns the Parsed particle system\n */\n static Parse(parsedParticleSystem, sceneOrEngine, rootUrl, doNotStart = false, capacity) {\n const name = parsedParticleSystem.name;\n let custom = null;\n let program = null;\n let engine;\n let scene;\n if (sceneOrEngine instanceof ThinEngine) {\n engine = sceneOrEngine;\n }\n else {\n scene = sceneOrEngine;\n engine = scene.getEngine();\n }\n if (parsedParticleSystem.customShader && engine.createEffectForParticles) {\n program = parsedParticleSystem.customShader;\n const defines = program.shaderOptions.defines.length > 0 ? program.shaderOptions.defines.join(\"\\n\") : \"\";\n custom = engine.createEffectForParticles(program.shaderPath.fragmentElement, program.shaderOptions.uniforms, program.shaderOptions.samplers, defines);\n }\n const particleSystem = new ParticleSystem(name, capacity || parsedParticleSystem.capacity, sceneOrEngine, custom, parsedParticleSystem.isAnimationSheetEnabled);\n particleSystem.customShader = program;\n particleSystem._rootUrl = rootUrl;\n if (parsedParticleSystem.id) {\n particleSystem.id = parsedParticleSystem.id;\n }\n // SubEmitters\n if (parsedParticleSystem.subEmitters) {\n particleSystem.subEmitters = [];\n for (const cell of parsedParticleSystem.subEmitters) {\n const cellArray = [];\n for (const sub of cell) {\n cellArray.push(SubEmitter.Parse(sub, sceneOrEngine, rootUrl));\n }\n particleSystem.subEmitters.push(cellArray);\n }\n }\n ParticleSystem._Parse(parsedParticleSystem, particleSystem, sceneOrEngine, rootUrl);\n if (parsedParticleSystem.textureMask) {\n particleSystem.textureMask = Color4.FromArray(parsedParticleSystem.textureMask);\n }\n // Auto start\n if (parsedParticleSystem.preventAutoStart) {\n particleSystem.preventAutoStart = parsedParticleSystem.preventAutoStart;\n }\n if (!doNotStart && !particleSystem.preventAutoStart) {\n particleSystem.start();\n }\n return particleSystem;\n }\n }\n /**\n * Billboard mode will only apply to Y axis\n */\n ParticleSystem.BILLBOARDMODE_Y = 2;\n /**\n * Billboard mode will apply to all axes\n */\n ParticleSystem.BILLBOARDMODE_ALL = 7;\n /**\n * Special billboard mode where the particle will be biilboard to the camera but rotated to align with direction\n */\n ParticleSystem.BILLBOARDMODE_STRETCHED = 8;\n /**\n * Special billboard mode where the particle will be billboard to the camera but only around the axis of the direction of particle emission\n */\n ParticleSystem.BILLBOARDMODE_STRETCHED_LOCAL = 9;\n SubEmitter._ParseParticleSystem = ParticleSystem.Parse;\n\n /**\n * Gizmo that enables moving vertices on a mesh.\n */\n class KbVertexSelectGizmo extends KbGizmo {\n get attachedMesh() {\n return this.__attachedMesh;\n }\n set attachedMesh(mesh) {\n this.__attachedMesh = mesh;\n if (this.particleSystem) {\n this.particleSystem.dispose();\n this.particleSystem = undefined;\n }\n if (mesh) {\n this.drawVertices(mesh);\n }\n }\n constructor(kbViewer, scene) {\n super(kbViewer.utilityLayer);\n this.kbViewer = kbViewer;\n this.scene = scene;\n /** Fires an event when any of it's sub gizmos are dragged */\n this.onSelectionChange = new Observable$1();\n this.vertexTracker = new Subject();\n this.particleScale = 0.005;\n this.redrawVertices = false;\n this.__attachedMesh = null;\n this.selectedVertices = new Map();\n this.vertexSelectedColor = new Color4(0.4, 1, 0, 0.5);\n this.vertexUnSelectedColor = new Color4(0, 0.2, 1, 0.5);\n this.vertexTransparentColor = new Color4(0, 0, 0, 0);\n this.verticesUtilLayer = kbViewer.utilityLayer;\n this.attachedMesh = null;\n }\n attachTo(target) {\n const mesh = this.kbViewer.getRendererNode(target);\n if (!mesh) {\n throw Error('Could not find mesh in scene to attach gizmo');\n }\n this.attachedMesh = mesh;\n // Clean up old marquee\n if (this.disposeMarquee) {\n this.disposeMarquee.next();\n this.disposeMarquee.complete();\n }\n this.disposeMarquee = new Subject();\n let selectionMode = null;\n this.kbViewer.interaction\n .enableMarqueeSelect(target, exports.eMarqueeSelectMode.Vertices)\n .pipe(takeUntil(this.disposeMarquee))\n .subscribe(event => {\n if (selectionMode || event.vertexList.length > 0) {\n const vertices = event.vertexList;\n // If user is making a new selection, set the selection mode based on modifier keys\n if (selectionMode === null) {\n if (event.ctrlKey) {\n selectionMode = 'add';\n }\n else if (event.altKey) {\n selectionMode = 'remove';\n }\n else {\n selectionMode = 'new';\n }\n }\n if (selectionMode === 'new') {\n this.selectedVertices.clear();\n }\n if (selectionMode === 'remove') {\n vertices.forEach(vertex => this.selectedVertices.delete(vertex));\n }\n else {\n vertices.forEach(vertex => this.selectedVertices.set(vertex, null));\n }\n this.redraw();\n this.emitSelectionChange();\n }\n });\n this.kbViewer.interaction.onPointerUp.pipe(takeUntil(this.disposeMarquee)).subscribe(() => {\n selectionMode = null;\n });\n }\n updateParticleScale(scale) {\n this.particleScale = scale;\n this.redraw();\n }\n updateGizmoPosition() {\n this.kbViewer.forceAndWaitForNextRender().then(() => {\n var _a;\n if (this.__attachedMesh) {\n this.currentMeshVertexPositions = (_a = this.__attachedMesh\n .getVerticesData(VertexBuffer.PositionKind)) === null || _a === void 0 ? void 0 : _a.slice();\n if (!this.currentMeshVertexPositions) {\n return;\n }\n this.redraw();\n }\n });\n }\n dispose() {\n if (this.particleSystem) {\n this.particleSystem.dispose();\n this.particleSystem = undefined;\n }\n this.disposeMarquee.next();\n this.disposeMarquee.complete();\n this.kbViewer.interaction.disableMarqueeSelect();\n }\n setCustomMesh(mesh) {\n Logger.Error('Custom meshes are not supported on this gizmo.');\n }\n selectVertices(vertexIndices) {\n this.selectedVertices.clear();\n vertexIndices.forEach(vertex => this.selectedVertices.set(vertex, null));\n this.redraw();\n this.emitSelectionChange();\n }\n enableSelectionChange() {\n this.selectionsEnabled = true;\n this.redraw();\n }\n disableSelectionChange() {\n this.selectionsEnabled = false;\n this.redraw();\n this.vertexTracker.complete();\n }\n getBarycenter(indices) {\n let x = 0, y = 0, z = 0;\n const currentMeshVertexPositions = this.currentMeshVertexPositions;\n if (currentMeshVertexPositions) {\n indices.forEach(idx => {\n x += currentMeshVertexPositions[idx * 3];\n y += currentMeshVertexPositions[idx * 3 + 1];\n z += currentMeshVertexPositions[idx * 3 + 2];\n });\n }\n return new exports.KbVector(x / indices.length, y / indices.length, z / indices.length);\n }\n redraw() {\n this.vertexTracker.next(this.selectedVertices.size.toString());\n if (this.particleSystem && this.saveAnimate) {\n this.particleSystem.animate = this.saveAnimate;\n this.redrawVertices = true;\n }\n }\n emitSelectionChange() {\n this.onSelectionChange.notifyObservers({ selectedIndices: Array.from(this.selectedVertices.keys()) });\n }\n drawVertices(targetMesh) {\n var _a;\n if (!this.currentMeshVertexPositions) {\n this.currentMeshVertexPositions = (_a = targetMesh.getVerticesData(VertexBuffer.PositionKind)) === null || _a === void 0 ? void 0 : _a.slice();\n }\n if (!this.currentMeshVertexPositions) {\n return;\n }\n const vertexCount = this.currentMeshVertexPositions.length / 3;\n this.particleSystem = new ParticleSystem('vertexSelectParticleSystem', vertexCount, this.verticesUtilLayer.utilityLayerScene);\n const camera = this.scene.activeCamera instanceof ArcRotateCamera ? this.scene.activeCamera : null;\n const cameraDistance = Vector3.Zero();\n const cameraPosition = Vector3.Zero();\n this.saveAnimate = this.particleSystem.animate.bind(this.particleSystem);\n let particleIdx = 0;\n this.particleSystem.startPositionFunction = (worldMatrix, positionToUpdate, particle) => {\n particleIdx += 3;\n const vertexPositions = this.currentMeshVertexPositions;\n Vector3.TransformCoordinatesFromFloatsToRef(vertexPositions[particleIdx], vertexPositions[particleIdx + 1], vertexPositions[particleIdx + 2], worldMatrix, positionToUpdate);\n };\n const transform = targetMesh.computeWorldMatrix(true);\n this.particleSystem.updateFunction = particles => {\n let particlesToProcess = false;\n if (camera) {\n cameraPosition.copyFrom(getCameraPositionWithOffset(camera));\n }\n for (let i = 0; i < particles.length; i++) {\n const particle = particles[i];\n const vertexPositions = this.currentMeshVertexPositions;\n if (vertexPositions && (particle.age === 0 || this.redrawVertices)) {\n Vector3.TransformCoordinatesFromFloatsToRef(vertexPositions[i * 3], vertexPositions[i * 3 + 1], vertexPositions[i * 3 + 2], transform, particle.position);\n particle.age = 1;\n if (this.selectedVertices.has(i)) {\n particle.color = this.vertexSelectedColor;\n }\n else {\n if (this.selectionsEnabled) {\n particle.color = this.vertexUnSelectedColor;\n }\n else {\n particle.color = this.vertexTransparentColor;\n }\n }\n if (!particlesToProcess) {\n particlesToProcess = true;\n }\n }\n if (camera) {\n cameraPosition.subtractToRef(particle.position, cameraDistance);\n particle.size = (this.particleScale - 0.001) * cameraDistance.length();\n }\n else {\n particle.size = this.particleScale - 0.004;\n }\n }\n this.redrawVertices = false;\n // If there are a large number of particles, kill the updating function for performance\n if (particles.length && particles.length > 10000 && !particlesToProcess && this.particleSystem) {\n // stable\n this.particleSystem.stop();\n this.particleSystem.animate = () => void 0;\n }\n };\n const colorArray = new Uint8Array(3);\n for (let i = 0; i < colorArray.length; i += 3) {\n colorArray[i] = 225;\n colorArray[i + 1] = 225;\n colorArray[i + 2] = 225;\n }\n this.particleSystem.emitter = targetMesh;\n this.particleSystem.minSize = this.particleScale;\n this.particleSystem.maxSize = this.particleScale;\n this.particleSystem.disposeOnStop = false;\n this.particleSystem.particleTexture = RawTexture.CreateRGBTexture(colorArray, 1, 1, this.verticesUtilLayer.utilityLayerScene);\n this.particleSystem.manualEmitCount = vertexCount;\n // this.particleSystem.emitRate\n this.particleSystem.start();\n }\n }\n\n /**\n * Gizmo that enables dragging and rotating lights\n */\n class KbViewpointGizmo extends KbGizmo {\n attachTo(viewpoint) {\n var _a;\n // Get rid of any old gizmos\n if (this.posGizmo) {\n this.posGizmo.dispose();\n this.posGizmo = undefined;\n }\n if (this.targetGizmo) {\n this.targetGizmo.dispose();\n this.targetGizmo = undefined;\n }\n if (this.rotGizmo) {\n this.rotGizmo.dispose();\n this.rotGizmo = undefined;\n }\n // Create mesh to attach gizmos to, and the camera preview visualization attached to that mesh\n this._visualMesh = new AbstractMesh('', this.gizmoLayer.utilityLayerScene);\n const { targetPositionMarker, viewpointPositionGizmo } = createViewpointVisualization(viewpoint, this.gizmoLayer.utilityLayerScene);\n this._visualMesh.addChild(viewpointPositionGizmo);\n // Always create the position gizmo\n this.posGizmo = new KbPositionGizmo(this.kbViewer, this.gizmoLayer);\n this.previousTargetMode = viewpoint.targetMode;\n // Determine which gizmos to show. No extra gizmos for overhead camera (only position)\n // For the target point type viewpoint, make a separate mesh to hold the position gizmo\n if (viewpoint.type === exports.eViewpointType.overhead) {\n viewpointPositionGizmo.scaling = new Vector3(1, -1, 1);\n viewpointPositionGizmo.bakeCurrentTransformIntoVertices();\n }\n else {\n if (viewpoint.targetMode === exports.eViewpointTargetMode.rotation) {\n this.rotGizmo = new KbRotationGizmo(this.kbViewer, this.gizmoLayer, { x: true, y: true, z: false });\n // The first person camera is oriented pointing parallel to the +Z axis\n if (viewpoint.type == exports.eViewpointType.firstPerson) {\n viewpointPositionGizmo.rotate(Vector3.Right(), Math.PI / 2);\n viewpointPositionGizmo.bakeCurrentTransformIntoVertices();\n }\n }\n else if (viewpoint.targetMode === exports.eViewpointTargetMode.point) {\n this._gizmoMesh = new AbstractMesh('', this.gizmoLayer.utilityLayerScene);\n this._targetVisualMesh = new AbstractMesh('', this.gizmoLayer.utilityLayerScene);\n if (targetPositionMarker) {\n (_a = this._targetVisualMesh) === null || _a === void 0 ? void 0 : _a.addChild(targetPositionMarker);\n }\n this.targetGizmo = new KbPositionGizmo(this.kbViewer, this.gizmoLayer);\n }\n }\n if (!this._gizmoMesh) {\n this._gizmoMesh = this._visualMesh;\n }\n else {\n this._visualMesh.isPickable = false;\n }\n this.updateHelperMeshes(viewpoint);\n // Set up event handlers on the gizmos\n if (this.posGizmo) {\n this.posGizmo.attachedMesh = this._gizmoMesh;\n const positionChange = Vector3.Zero();\n this.posGizmo.onDragStartObservable.add(event => {\n positionChange.setAll(0);\n this.onDragStartObservable.notifyObservers();\n });\n this.posGizmo.onDragObservable.add(event => {\n var _a;\n if (this._gizmoMesh !== this._visualMesh) {\n (_a = this._visualMesh) === null || _a === void 0 ? void 0 : _a.position.addInPlace(event.delta);\n }\n positionChange.addInPlace(event.delta);\n this.onDragObservable.notifyObservers({ deltaPos: event.delta });\n this.updateHelperMeshDirectionToRotation(this._attachedViewpoint);\n });\n this.posGizmo.onDragEndObservable.add(event => {\n this.onDragEndObservable.notifyObservers({ deltaPos: positionChange });\n });\n }\n if (this.targetGizmo) {\n this.targetGizmo.attachedMesh = this._targetVisualMesh;\n const targetChange = Vector3.Zero();\n this.targetGizmo.onDragStartObservable.add(event => {\n targetChange.setAll(0);\n this.onDragStartObservable.notifyObservers();\n });\n this.targetGizmo.onDragObservable.add(event => {\n targetChange.addInPlace(event.delta);\n this.onDragObservable.notifyObservers({ deltaTargetPos: event.delta });\n this.updateHelperMeshDirectionToRotation(this._attachedViewpoint);\n });\n this.targetGizmo.onDragEndObservable.add(event => {\n this.onDragEndObservable.notifyObservers({ deltaTargetPos: targetChange });\n });\n }\n if (this.rotGizmo) {\n this.rotGizmo.attachedMesh = this._visualMesh;\n let rotationChange = this._visualMesh.rotationQuaternion || Quaternion.Identity();\n this.rotGizmo.onDragStartObservable.add(event => {\n this.onDragStartObservable.notifyObservers();\n rotationChange = Quaternion.Identity();\n });\n this.rotGizmo.onDragObservable.add(event => {\n const { delta } = event;\n rotationChange.multiplyInPlace(delta);\n this.onDragObservable.notifyObservers({ deltaRotation: delta });\n });\n this.rotGizmo.onDragEndObservable.add(() => {\n this.onDragEndObservable.notifyObservers({ deltaRotation: rotationChange });\n });\n }\n this._attachedViewpoint = viewpoint;\n }\n updateGizmoPosition(viewpointNode) {\n if (viewpointNode && viewpointNode !== this._attachedViewpoint) {\n this._attachedViewpoint = viewpointNode;\n }\n // If the target mode changed, rebuild the gizmo\n if (this._attachedViewpoint && this.previousTargetMode !== this._attachedViewpoint.targetMode) {\n const viewpoint = this._attachedViewpoint;\n this.dispose(true);\n this.attachTo(viewpoint);\n }\n else {\n if (this._attachedViewpoint) {\n this.updateHelperMeshes(this._attachedViewpoint);\n }\n super.updateGizmoPosition();\n }\n }\n updateHelperMeshes(viewpoint) {\n if (!this._visualMesh || !this._gizmoMesh) {\n return;\n }\n this._visualMesh.position = viewpoint.position.toVec3();\n this._gizmoMesh.position = viewpoint.position.toVec3();\n if (this._targetVisualMesh) {\n this._targetVisualMesh.position = viewpoint.targetPoint.toVec3();\n }\n if (viewpoint.targetMode === exports.eViewpointTargetMode.rotation) {\n const eulerRot = viewpoint.rotation.toVec3().scale(Math.PI / 180);\n this._visualMesh.rotationQuaternion = Quaternion.FromEulerVector(eulerRot);\n }\n else {\n this.updateHelperMeshDirectionToRotation(viewpoint);\n }\n }\n updateHelperMeshDirectionToRotation(viewpoint, delta) {\n if (!this._visualMesh) {\n return;\n }\n if (!delta) {\n delta = Vector3.Zero();\n }\n const _visualMesh = this._visualMesh;\n if (viewpoint.targetMode !== exports.eViewpointTargetMode.rotation) {\n let direction;\n if (this._targetVisualMesh) {\n direction = this._targetVisualMesh.position.subtract(_visualMesh.position);\n }\n else {\n // mesh\n const targetMeshNode = viewpoint._manager.getById(viewpoint.targetMeshId);\n if (targetMeshNode) {\n direction = targetMeshNode.position.subtract(viewpoint.position).toVec3();\n }\n else {\n direction = Vector3.Zero();\n }\n }\n const rotation = rotationToVector(direction);\n _visualMesh.rotationQuaternion = rotation;\n // This code is for fixing the rotation of the camera gizmo so that the x axis is always in the XZ plane (keeping the camera model level with the horizon.)\n // But, it is not working quite right.\n // _visualMesh.computeWorldMatrix(true);\n // let newXAxis = _visualMesh.getDirection(Vector3.Right());\n // const correction = rotationToVector(direction.cross(Vector3.Up()), newXAxis);\n // _visualMesh.rotationQuaternion = _visualMesh.rotationQuaternion.multiply(correction);\n }\n }\n /**\n * Creates a PositionGizmo\n * @param gizmoLayer The utility layer the gizmo will be added to\n */\n constructor(kbViewer, gizmoLayer = UtilityLayerRenderer.DefaultUtilityLayer) {\n super(gizmoLayer);\n this.kbViewer = kbViewer;\n /** Fires an event when any of it's sub gizmos are dragged */\n this.onDragStartObservable = new Observable$1();\n /** Fires an event while any of it's sub gizmos are dragged */\n this.onDragObservable = new Observable$1();\n /** Fires an event when any of it's sub gizmos are released from dragging */\n this.onDragEndObservable = new Observable$1();\n this.attachedMesh = null;\n }\n /**\n * Disposes of the gizmo\n */\n dispose(recreate) {\n if (this.posGizmo) {\n this.posGizmo.dispose();\n }\n if (this.rotGizmo) {\n this.rotGizmo.dispose();\n }\n if (this.targetGizmo) {\n this.targetGizmo.dispose();\n }\n if (this._visualMesh) {\n this._visualMesh.dispose();\n }\n if (this._targetVisualMesh) {\n this._targetVisualMesh.dispose();\n }\n if (this.viewpointPositionGizmo) {\n this.viewpointPositionGizmo.dispose();\n }\n if (!recreate) {\n this.onDragStartObservable.clear();\n this.onDragEndObservable.clear();\n delete this._attachedViewpoint;\n }\n }\n }\n\n /**\n * Manages the various gizmos and their connections to the scene meshes.\n * Keeps track of what gizmo is active and the selected meshes to put the gizmos on.\n * @param scene The root scene on which the utility/gizmo layer will be placed.\n * @param camera The camera being used. Required by some gizmos for displaying them correctly\n */\n const PREVIEW_NAME = '__UV_PREVIEW_MATERIAL__';\n class GizmoTools {\n constructor(scene, kbViewer) {\n this.kbViewer = kbViewer;\n this.activeGizmoSettings = [];\n this.dragging = false;\n this.uvMapStoredMaterials = new Map();\n this.scene = scene;\n this.gizmoLayer = kbViewer.utilityLayer;\n this.gizmoLayer.utilityLayerScene.useRightHandedSystem = true;\n }\n selectLight(lightNode) {\n const lightGizmo = new KbLightGizmo(this.kbViewer, this.gizmoLayer);\n const updatePosition = new Subject();\n const updateDirection = new Subject();\n const totalDeltaPosition = new Subject();\n this.activeGizmoSettings.push({\n subjects: [updatePosition, updateDirection, totalDeltaPosition],\n gizmo: lightGizmo,\n });\n lightGizmo.attachTo(lightNode);\n this.dragging = false;\n return {\n updateLightGizmo: (node) => {\n lightGizmo.updateGizmoPosition(node);\n },\n };\n }\n selectViewpoint(viewpoint) {\n const viewpointGizmo = new KbViewpointGizmo(this.kbViewer, this.gizmoLayer);\n viewpointGizmo.attachTo(viewpoint);\n const updatePosition = new Subject();\n const updateTargetPosition = new Subject();\n const updateRotation = new Subject();\n const totalDeltaPosition = new Subject();\n const totalDeltaTargetPosition = new Subject();\n const totalDeltaRotation = new Subject();\n this.activeGizmoSettings.push({\n subjects: [\n updatePosition,\n updateTargetPosition,\n updateRotation,\n totalDeltaPosition,\n totalDeltaTargetPosition,\n totalDeltaRotation,\n ],\n gizmo: viewpointGizmo,\n });\n this.dragging = false;\n const totalDeltaPositionQueue = Vector3.Zero();\n const totalDeltaTargetPositionQueue = Vector3.Zero();\n const totalDeltaRotationQueue = Quaternion.Identity();\n viewpointGizmo.onDragStartObservable.add(() => {\n this.dragging = true;\n });\n viewpointGizmo.onDragObservable.add(event => {\n if (this.dragging) {\n if (event.deltaPos) {\n updatePosition.next(exports.KbVector.FromVec3(event.deltaPos));\n totalDeltaPositionQueue.addInPlace(event.deltaPos);\n }\n if (event.deltaTargetPos) {\n updateTargetPosition.next(exports.KbVector.FromVec3(event.deltaTargetPos));\n totalDeltaTargetPositionQueue.addInPlace(event.deltaTargetPos);\n }\n if (event.deltaRotation) {\n updateRotation.next(exports.KbVector.FromVec3(event.deltaRotation.toEulerAngles().scale(180 / Math.PI)));\n totalDeltaRotationQueue.multiplyInPlace(event.deltaRotation);\n }\n }\n });\n viewpointGizmo.onDragEndObservable.add(() => {\n if (this.dragging) {\n this.dragging = false;\n if (!totalDeltaPositionQueue.equalsToFloats(0, 0, 0)) {\n totalDeltaPosition.next(exports.KbVector.FromVec3(totalDeltaPositionQueue));\n totalDeltaPositionQueue.setAll(0);\n }\n else if (!totalDeltaTargetPositionQueue.equalsToFloats(0, 0, 0)) {\n totalDeltaTargetPosition.next(exports.KbVector.FromVec3(totalDeltaTargetPositionQueue));\n totalDeltaTargetPositionQueue.setAll(0);\n }\n else if (!totalDeltaRotationQueue.equals(Quaternion.Identity())) {\n totalDeltaRotation.next(exports.KbVector.FromVec3(totalDeltaRotationQueue.toEulerAngles().scale(180 / Math.PI)));\n totalDeltaRotationQueue.copyFrom(Quaternion.Identity());\n }\n }\n });\n return {\n updatePosition: updatePosition.asObservable(),\n updateTargetPosition: updateTargetPosition.asObservable(),\n updateRotation: updateRotation.asObservable(),\n totalDeltaPosition: totalDeltaPosition.asObservable(),\n totalDeltaTargetPosition: totalDeltaTargetPosition.asObservable(),\n totalDeltaRotation: totalDeltaRotation.asObservable(),\n updateViewpointGizmo: (node) => {\n viewpointGizmo.updateGizmoPosition(node);\n },\n };\n }\n position(target, positionSpace, axisVisibility, orientGizmo) {\n const positionGizmo = new KbPositionGizmo(this.kbViewer, this.gizmoLayer, axisVisibility);\n positionGizmo.attachTo(target, positionSpace, orientGizmo);\n const updatePosition = new Subject();\n const totalDelta = new Subject();\n this.activeGizmoSettings.push({\n subjects: [updatePosition, totalDelta],\n gizmo: positionGizmo,\n });\n this.dragging = false;\n let totalDeltaQueue = Vector3.Zero();\n positionGizmo.onDragStartObservable.add(() => {\n this.dragging = true;\n });\n positionGizmo.onDragObservable.add(event => {\n if (this.dragging) {\n totalDeltaQueue.addInPlace(event.delta);\n updatePosition.next(exports.KbVector.FromVec3(event.delta));\n }\n });\n positionGizmo.onDragEndObservable.add(() => {\n if (this.dragging) {\n totalDelta.next(exports.KbVector.FromVec3(totalDeltaQueue));\n totalDeltaQueue = Vector3.Zero();\n this.dragging = false;\n }\n });\n return {\n update: updatePosition.asObservable(),\n totalDelta: totalDelta.asObservable(),\n updateGizmoPosition: (update) => positionGizmo.updateGizmoPosition(update),\n dispose: () => this.clearGizmo(positionGizmo),\n };\n }\n rotateAxisAngle(target, positionSpace, options) {\n const updateAxisAngle = new Subject();\n const totalDelta = new Subject();\n const totalDeltaQueue = { axis: Vector3.Zero(), angle: 0 };\n const axisAngleGizmo = new KbAxisAngleGizmo(this.kbViewer, this.gizmoLayer, options === null || options === void 0 ? void 0 : options.disableAngle);\n axisAngleGizmo.attachTo(target, positionSpace, options === null || options === void 0 ? void 0 : options.placementOffset);\n this.activeGizmoSettings.push({\n subjects: [updateAxisAngle, totalDelta],\n gizmo: axisAngleGizmo,\n });\n this.dragging = false;\n axisAngleGizmo.onDragStartObservable.add(() => {\n this.dragging = true;\n });\n axisAngleGizmo.onDragObservable.add(event => {\n if (this.dragging) {\n const angle = (event.deltaAngle * 180) / Math.PI;\n totalDeltaQueue.axis.addInPlace(event.deltaAxis);\n totalDeltaQueue.angle += angle;\n updateAxisAngle.next({ axis: exports.KbVector.FromVec3(event.deltaAxis), angle });\n }\n });\n axisAngleGizmo.onDragEndObservable.add(() => {\n if (this.dragging) {\n totalDelta.next({ axis: exports.KbVector.FromVec3(totalDeltaQueue.axis), angle: totalDeltaQueue.angle });\n totalDeltaQueue.axis = Vector3.Zero();\n totalDeltaQueue.angle = 0;\n this.dragging = false;\n }\n });\n return {\n update: updateAxisAngle.asObservable(),\n totalDelta: totalDelta.asObservable(),\n updateGizmoPosition: (updateAxis, updatePosition) => axisAngleGizmo.updateGizmoPosition(updateAxis, updatePosition),\n dispose: () => this.clearGizmo(axisAngleGizmo),\n };\n }\n /**\n * Creates a rotation gizmo that can rotate many items\n * nodes will be rotated and repositioned in space if the center of rotation is not equal to the position\n * vectors will be repositioned in space as if each vector is the position of a point\n */\n rotate(targets, positionSpace, placementOffset, axisVisibility) {\n var _a, _b;\n let gizmoOffset = Vector3.Zero();\n // Get the transform matrix to get the world space rotation and coordinates of the node we are rotating\n let positionTransform;\n if (positionSpace) {\n positionTransform = (_a = this.kbViewer.getRendererNode(positionSpace)) === null || _a === void 0 ? void 0 : _a.computeWorldMatrix(true);\n }\n else if (targets[0] instanceof exports.SpaceNode && targets[0]._parent) {\n positionTransform = (_b = this.kbViewer.getRendererNode(targets[0]._parent)) === null || _b === void 0 ? void 0 : _b.computeWorldMatrix(true);\n }\n const rotationGizmo = new KbRotationGizmo(this.kbViewer, this.gizmoLayer, undefined, axisVisibility);\n if (targets.length === 1) {\n const target = targets[0];\n rotationGizmo.attachTo(target, positionSpace, placementOffset);\n }\n else {\n // If multiple targets are selected, put the pivot in the average of the positions\n // Maybe enhance this to use bounding box later, if that's important...\n const pivot = targets\n .reduce((result, current) => {\n return result.add(current instanceof exports.SpaceNode ? current.position : current);\n }, exports.KbVector.Zero())\n .scale(1 / targets.length);\n rotationGizmo.attachTo(exports.KbVector.Zero(), undefined, pivot);\n }\n const update = new Subject();\n const totalDelta = new Subject();\n this.activeGizmoSettings.push({\n subjects: [update, totalDelta],\n gizmo: rotationGizmo,\n });\n this.dragging = false;\n let totalDeltaQueue = {\n rotation: Quaternion.Identity(),\n position: Vector3.Zero(),\n axisTranslate: Vector3.Zero(),\n };\n // Set up caches for the positions and rotations of the nodes because it's not guaranteed that they will be updated by reference\n // And as you drag the nodes these values will change and the gizmos won't be dragging in the right direction anymore\n const positions = targets.map(target => target instanceof exports.SpaceNode ? target.position.toVec3() : target.toVec3());\n const axes = targets.map(target => (target instanceof exports.SpaceNode ? target.rotationAxis.toVec3() : null));\n const quaternionCache = targets.map(target => target instanceof exports.SpaceNode\n ? Quaternion.FromEulerVector(target.eulerRotation.toVec3().scale(Math.PI / 180))\n : Quaternion.Identity());\n // Get the current pivot which will always be where the gizmo is located (minus the offset which just visually offsets the gizmo position)\n let gizmoPosition = rotationGizmo.attachedMesh.position;\n rotationGizmo.onDragStartObservable.add(() => {\n this.dragging = true;\n if (rotationGizmo.gizmoOffset) {\n gizmoOffset = rotationGizmo.gizmoOffset.toVec3();\n }\n if (gizmoOffset) {\n gizmoPosition = gizmoPosition.subtract(gizmoOffset);\n }\n });\n rotationGizmo.onDragObservable.add(event => {\n if (this.dragging) {\n const result = [];\n const rotationMatrix = new Matrix();\n const gizmoRotation = rotationGizmo.attachedMesh.rotationQuaternion || Quaternion.Identity();\n // Get the true world space rotation in the case that the gizmo is rotated\n // The gizmos output their rotation deltas in local space\n const meshSpaceRotation = gizmoRotation.multiply(event.delta).multiply(gizmoRotation.conjugate());\n meshSpaceRotation.toRotationMatrix(rotationMatrix);\n const worldTransformMatrix = new Matrix();\n event.delta.toRotationMatrix(worldTransformMatrix);\n worldTransformMatrix.multiplyToRef(rotationGizmo.attachedMesh.computeWorldMatrix(), worldTransformMatrix);\n // Inverse the position transform to rotate the delta back into local \"positionSpace\" space\n const positionTransformInv = positionTransform === null || positionTransform === void 0 ? void 0 : positionTransform.clone().invert();\n positions.forEach((position, index) => {\n var _a;\n let positionDelta;\n let axisDelta;\n const target = targets[index];\n const axis = axes[index];\n // If the node is a space node, then rotate it around the pivot to get the updated position\n if (target instanceof exports.SpaceNode) {\n // Treat the pivot point as the center of the node so the node rotates around the pivot point\n let worldNodePivot = target.pivot.toVec3();\n const nodeTransform = (_a = this.kbViewer.getRendererNode(target)) === null || _a === void 0 ? void 0 : _a.computeWorldMatrix(true);\n if (nodeTransform) {\n worldNodePivot = Vector3.TransformCoordinates(worldNodePivot, nodeTransform);\n }\n positionDelta = Vector3.TransformCoordinates(worldNodePivot.subtract(gizmoPosition), rotationMatrix)\n .add(gizmoPosition)\n .subtract(worldNodePivot);\n // If the node is a vector, then it's a direction vector, so we want to rotate it as if the vector started from the origin.\n // So don't rotate around the pivot point. The updated position will actually be the updated direction.\n }\n else {\n positionDelta = Vector3.TransformCoordinates(position, rotationMatrix).subtract(position);\n }\n if (axis) {\n axisDelta = Vector3.TransformCoordinates(axis.length() === 0 ? new Vector3(0, 1, 0) : axis, rotationMatrix).subtract(axis);\n axis.addInPlace(axisDelta);\n }\n // Rotate the delta vector into local \"positionSpace\" space\n if (positionTransformInv) {\n positionDelta = Vector3.TransformCoordinates(positionDelta, positionTransformInv.getRotationMatrix());\n }\n // Update the position cache by reference (make sure this position variable isn't shadowed)\n position.addInPlace(positionDelta);\n totalDeltaQueue.position.addInPlace(positionDelta);\n if (axisDelta) {\n totalDeltaQueue.axisTranslate.addInPlace(axisDelta);\n }\n // Get the rotation delta by converting the previous quaternion rotation to euler, rotating to the new position, and getting euler angles of that.\n // This is necessary because it we output the plain quaternion delta euler, the euler rotation order could result\n // In an incorrect rotation if there are other euler angles being rotated first\n const previousAngle = quaternionCache[index].toEulerAngles();\n quaternionCache[index].multiplyInPlace(event.delta);\n const newAngle = quaternionCache[index].toEulerAngles();\n totalDeltaQueue.rotation.multiplyInPlace(event.delta);\n result.push({\n position: exports.KbVector.FromVec3(positionDelta),\n rotation: exports.KbVector.FromVec3(newAngle.subtract(previousAngle).scale(180 / Math.PI)),\n worldTransformMatrix: worldTransformMatrix,\n axisTranslate: axisDelta ? exports.KbVector.FromVec3(axisDelta) : exports.KbVector.Zero(),\n });\n });\n update.next(result);\n }\n });\n rotationGizmo.onDragEndObservable.add(() => {\n if (this.dragging) {\n totalDelta.next({\n rotation: exports.KbVector.FromVec3(totalDeltaQueue.rotation.toEulerAngles().scale(180 / Math.PI)),\n position: exports.KbVector.FromVec3(totalDeltaQueue.position),\n axisTranslate: exports.KbVector.FromVec3(totalDeltaQueue.axisTranslate),\n });\n totalDeltaQueue = {\n rotation: Quaternion.Identity(),\n position: Vector3.Zero(),\n axisTranslate: Vector3.Zero(),\n };\n this.dragging = false;\n }\n });\n return {\n update: update.asObservable(),\n totalDelta: totalDelta.asObservable(),\n updateGizmoPosition: (update, offset) => rotationGizmo.updateGizmoPosition(update, offset),\n dispose: () => this.clearGizmo(rotationGizmo),\n };\n }\n scale(targets, positionSpace) {\n // if (targets.length === 0) {\n // return {\n // update: empty(),\n // totalDelta: empty(),\n // };\n // }\n var _a, _b;\n const scaleGizmo = new KbScaleGizmo(this.kbViewer, this.gizmoLayer);\n if (targets.length === 1) {\n const target = targets[0];\n scaleGizmo.attachTo(target, positionSpace);\n }\n else {\n // If multiple targets are selected, put the pivot in the average of the positions\n // Maybe enhance this to use bounding box later, if that's important...\n const pivot = targets\n .reduce((result, current) => {\n return result.add(current instanceof exports.SpaceNode ? current.position : current);\n }, exports.KbVector.Zero())\n .scale(1 / targets.length);\n scaleGizmo.attachTo(pivot, positionSpace);\n }\n const update = new Subject();\n const totalDelta = new Subject();\n this.activeGizmoSettings.push({\n subjects: [update, totalDelta],\n gizmo: scaleGizmo,\n });\n this.dragging = false;\n let totalDeltaQueue = Vector3.Zero();\n let positionTransform;\n // Get the transform matrix to get the world space rotation and coordinates of the node we are scaling\n if (positionSpace) {\n positionTransform = (_a = this.kbViewer.getRendererNode(positionSpace)) === null || _a === void 0 ? void 0 : _a.computeWorldMatrix(true);\n }\n else if (targets[0] instanceof exports.SpaceNode && targets[0]._parent) {\n positionTransform = (_b = this.kbViewer.getRendererNode(targets[0]._parent)) === null || _b === void 0 ? void 0 : _b.computeWorldMatrix(true);\n }\n // Get the current pivot which will always be where the gizmo is located\n let gizmoPosition = scaleGizmo.attachedMesh.position;\n scaleGizmo.onDragStartObservable.add(() => {\n this.dragging = true;\n if (scaleGizmo.gizmoOffset) {\n gizmoPosition = gizmoPosition.subtract(scaleGizmo.gizmoOffset.toVec3());\n }\n });\n scaleGizmo.onDragObservable.add(event => {\n if (this.dragging) {\n const result = [];\n targets.forEach(target => {\n var _a;\n let positionDelta;\n // If the node is a space node, then rotate it around the pivot to get the updated position\n if (target instanceof exports.SpaceNode) {\n // Treat the pivot point as the center of the node so the node rotates around the pivot point\n let worldNodePivot = target.pivot.toVec3();\n const nodeTransform = (_a = this.kbViewer.getRendererNode(target)) === null || _a === void 0 ? void 0 : _a.computeWorldMatrix(true);\n if (nodeTransform) {\n worldNodePivot = Vector3.TransformCoordinates(worldNodePivot, nodeTransform);\n }\n positionDelta = worldNodePivot\n .subtract(gizmoPosition)\n .multiply(event.delta)\n .add(gizmoPosition)\n .subtract(worldNodePivot);\n // If the node is a vector, then it's a direction vector, so we want to rotate it as if the vector started from the origin.\n // So don't rotate around the pivot point. The updated position will actually be the updated direction.\n }\n else {\n // Transform the node position to world space because the pivot and gizmo will be in world space as well\n let worldPosition = target.toVec3();\n if (positionTransform) {\n worldPosition = Vector3.TransformCoordinates(worldPosition, positionTransform);\n }\n positionDelta = worldPosition.multiply(event.delta).subtract(worldPosition);\n }\n totalDeltaQueue.addInPlace(event.delta);\n result.push({\n position: exports.KbVector.FromVec3(positionDelta),\n scaling: exports.KbVector.FromVec3(event.delta),\n });\n });\n update.next(result);\n }\n });\n scaleGizmo.onDragEndObservable.add(() => {\n if (this.dragging) {\n totalDelta.next(exports.KbVector.FromVec3(totalDeltaQueue));\n totalDeltaQueue = Vector3.Zero();\n this.dragging = false;\n }\n });\n return {\n update: update.asObservable(),\n totalDelta: totalDelta.asObservable(),\n updateGizmoPosition: (update) => scaleGizmo.updateGizmoPosition(update),\n dispose: () => this.clearGizmo(scaleGizmo),\n };\n }\n selectVertices(target, selectedVertices) {\n const vertexSelectionGizmo = new KbVertexSelectGizmo(this.kbViewer, this.scene);\n vertexSelectionGizmo.attachTo(target);\n const update = new Subject();\n this.activeGizmoSettings.push({\n subjects: [update],\n gizmo: vertexSelectionGizmo,\n });\n if (selectedVertices) {\n vertexSelectionGizmo.selectVertices(selectedVertices);\n }\n this.dragging = false;\n vertexSelectionGizmo.onSelectionChange.add(result => {\n if (result) {\n update.next({\n vertexIndices: result.selectedIndices,\n });\n }\n });\n return {\n update: update.asObservable(),\n vertexTracker: vertexSelectionGizmo.vertexTracker,\n updateGizmoPositions: () => vertexSelectionGizmo.updateGizmoPosition(),\n updateParticleScale: (scale) => vertexSelectionGizmo.updateParticleScale(scale),\n disableSelectionChange: () => vertexSelectionGizmo.disableSelectionChange(),\n enableSelectionChange: () => vertexSelectionGizmo.enableSelectionChange(),\n getBarycenter: (indices) => vertexSelectionGizmo.getBarycenter(indices),\n dispose: () => this.clearGizmo(vertexSelectionGizmo),\n };\n }\n selectFacets(target, selectedVertices) {\n const facetSelectionGizmo = new KbFacetSelectGizmo(this.kbViewer, this.scene);\n facetSelectionGizmo.attachTo(target);\n const update = new Subject();\n this.activeGizmoSettings.push({\n subjects: [update],\n gizmo: facetSelectionGizmo,\n });\n if (selectedVertices) {\n facetSelectionGizmo.selectFacetGroups(selectedVertices);\n }\n this.dragging = false;\n facetSelectionGizmo.onSelectionChange.add(result => {\n if (result) {\n update.next({\n selectedFacets: result.selectedFacets,\n });\n }\n });\n return {\n update: update.asObservable(),\n updateGizmoPositions: () => facetSelectionGizmo.updateGizmoPosition(),\n disableSelectionChange: () => facetSelectionGizmo.disableSelectionChange(),\n enableSelectionChange: () => facetSelectionGizmo.enableSelectionChange(),\n dispose: () => this.clearGizmo(facetSelectionGizmo),\n setBackfaceCulling: (enabled) => facetSelectionGizmo.setBackfaceCulling(enabled),\n setTolerance: (tolerance) => facetSelectionGizmo.setTolerance(tolerance),\n setVisualization: (opacity) => facetSelectionGizmo.setVisualization(opacity),\n expandSelection: (tolerance) => facetSelectionGizmo.expandSelection(tolerance),\n };\n }\n clearGizmo(gizmo) {\n if (gizmo) {\n const index = this.activeGizmoSettings.findIndex(g => g.gizmo === gizmo);\n if (index >= 0) {\n const gizmoSettings = this.activeGizmoSettings[index];\n gizmoSettings.subjects.forEach(subject => subject.complete());\n gizmoSettings.gizmo.dispose();\n this.activeGizmoSettings.splice(index, 1);\n }\n }\n else {\n this.activeGizmoSettings.forEach(gizmoSettings => {\n gizmoSettings.subjects.forEach(subject => subject.complete());\n gizmoSettings.gizmo.dispose();\n });\n this.activeGizmoSettings = [];\n }\n }\n updateGizmoPositions() {\n if (!this.dragging) {\n this.activeGizmoSettings.forEach(gizmoSettings => {\n gizmoSettings.gizmo.updateGizmoPosition();\n });\n }\n }\n showUvMapMaterial(target) {\n if (target._showMaterial && target._previewMaterialVisible) {\n if (target._parent) {\n const mesh = this.kbViewer.getRendererNode(target._parent);\n if (mesh instanceof Mesh) {\n if (mesh.material != null && mesh.material.name != PREVIEW_NAME) {\n const previousMaterial = mesh.material;\n this.uvMapStoredMaterials.set(target.id, previousMaterial);\n }\n if (!this.previewMaterial) {\n this.previewMaterial = new PBRMaterial(PREVIEW_NAME, this.kbViewer.scene);\n const previewTexture = new Texture('/assets/images/uv_colorgrid.png', this.kbViewer.scene);\n this.previewMaterial.albedoTexture = previewTexture;\n this.previewMaterial.albedoColor = Color3.White();\n this.previewMaterial.metallic = 0;\n this.previewMaterial.roughness = 1;\n }\n mesh.material = this.previewMaterial;\n }\n }\n }\n else {\n if (target._parent) {\n const mesh = this.kbViewer.getRendererNode(target._parent);\n if (mesh instanceof Mesh) {\n const storedMaterial = this.uvMapStoredMaterials.get(target.id);\n if (storedMaterial) {\n mesh.material = storedMaterial;\n }\n }\n }\n }\n }\n // Bounding mesh code is not used right now, but it may be useful when we are doing mates\n // private calculateBoundingBox(nodes: TransformNode[]) {\n // const meshes = nodes.filter((node): node is Mesh => node instanceof Mesh);\n // meshes.forEach(mesh => mesh.computeWorldMatrix(true));\n // if (meshes.length === 0) {\n // return {\n // pivot: Vector3.Zero(),\n // boundingInfo: new BoundingInfo(Vector3.Zero(), Vector3.Zero())\n // };\n // }\n // let boundingInfo = meshes[0].getBoundingInfo();\n // let minimumWorld = boundingInfo.boundingBox.minimumWorld;\n // let maximumWorld = boundingInfo.boundingBox.maximumWorld;\n // for (let i = 1; i < meshes.length; i++) {\n // let boundingInfo = meshes[i].getBoundingInfo();\n // minimumWorld = Vector3.Minimize(minimumWorld, boundingInfo.boundingBox.minimumWorld);\n // maximumWorld = Vector3.Maximize(maximumWorld, boundingInfo.boundingBox.maximumWorld);\n // }\n // let bounding = new BoundingInfo(minimumWorld, maximumWorld);\n // let middlePivot = new Vector3(bounding.boundingBox.center.x, bounding.boundingBox.center.y, bounding.boundingBox.center.z);\n // return {\n // pivot: middlePivot,\n // boundingInfo: bounding\n // };\n // }\n // private displayKbBoundingBox(mesh: Mesh) {\n // let oMesh: Mesh & { kbBoundingBox?: boundingBoxGeometry };\n // oMesh = mesh;\n // var bbNode: boundingBoxGeometry;\n // if (this.currentSelectedMeshes.length === 0) {\n // if (oMesh.kbBoundingBox) {\n // oMesh.kbBoundingBox.wireFrame.dispose();\n // delete oMesh.kbBoundingBox;\n // }\n // return;\n // }\n // // if the mesh is resized/rotated/something we need to recalculate the bounding box\n // mesh.computeWorldMatrix(true);\n // mesh.refreshBoundingInfo();\n // mesh.refreshBoundingInfo = (applySkeleton?: boolean) => {\n // return mesh;\n // };\n // const center = mesh.getBoundingInfo().boundingBox.centerWorld.clone();\n // const diagonalLen = mesh.getBoundingInfo().diagonalLength;\n // const min = mesh.getBoundingInfo().boundingBox.minimumWorld;\n // const max = mesh.getBoundingInfo().boundingBox.maximumWorld;\n // const d = max.subtract(min);\n // const bOffset = 0.02;\n // d.x += bOffset;\n // d.y += bOffset;\n // d.z += bOffset;\n // // calculate the position of the bounding box vertices\n // const v1 = new Vector3(center.x - d.x / 2, center.y + d.y / 2, center.z + d.z / 2);\n // const v2 = new Vector3(center.x + d.x / 2, center.y + d.y / 2, center.z + d.z / 2);\n // const v3 = new Vector3(center.x - d.x / 2, center.y - d.y / 2, center.z + d.z / 2);\n // const v4 = new Vector3(center.x + d.x / 2, center.y - d.y / 2, center.z + d.z / 2);\n // const v5 = new Vector3(center.x - d.x / 2, center.y + d.y / 2, center.z - d.z / 2);\n // const v6 = new Vector3(center.x + d.x / 2, center.y + d.y / 2, center.z - d.z / 2);\n // const v7 = new Vector3(center.x - d.x / 2, center.y - d.y / 2, center.z - d.z / 2);\n // const v8 = new Vector3(center.x + d.x / 2, center.y - d.y / 2, center.z - d.z / 2);\n // const lines = Mesh.CreateLines(KBMAXPREFIX + \"boundingBox\", [\n // v1, v2,\n // v1, v3,\n // v3, v4,\n // v4, v2,\n // v2, v6,\n // v6, v5,\n // v5, v7,\n // v7, v8,\n // v8, v6,\n // v8, v4,\n // v3, v7,\n // v5, v1\n // ], this.gizmoLayer.utilityLayerScene);\n // lines.color = new Color3(1, 1, 1);\n // lines.isPickable = false;\n // if (oMesh.kbBoundingBox) {\n // // the bounding box exists so it must be recalculated and repositioned\n // bbNode = oMesh.kbBoundingBox;\n // bbNode.wireFrame.dispose(); // it is easier to recreate it then reposition it\n // bbNode.wireFrame = lines;\n // }\n // else {\n // // the bounding box does not exist, so it must be created\n // bbNode = new boundingBoxGeometry(lines);\n // }\n // oMesh.kbBoundingBox = bbNode; // force to attach the current boundingBox\n // }\n filterToChangedValues(vector) {\n return {\n x: vector.x ? vector.x : undefined,\n y: vector.y ? vector.y : undefined,\n z: vector.z ? vector.z : undefined,\n w: 'w' in vector ? vector.w : undefined,\n };\n }\n }\n // class boundingBoxGeometry {\n // mateHelper1?: Mesh;\n // mateHelper2?: Mesh;\n // mateHelper3?: Mesh;\n // mateHelper4?: Mesh;\n // mateHelper5?: Mesh;\n // mateHelper6?: Mesh;\n // wireFrame: Mesh;\n // constructor(wireFrame: Mesh) {\n // this.wireFrame = wireFrame;\n // }\n // }\n\n function bakeMeshTransform(node) {\n const newTransform = computeLocalTransform(node);\n if (node.preTransform) {\n if (node.preTransform.length !== 16) {\n throw Error('invalid pretransform array ');\n }\n const preTransform = Matrix.FromArray(node.preTransform);\n preTransform.multiplyToRef(newTransform, newTransform);\n }\n node.preTransform = Array.from(newTransform.toArray());\n node.pivot = exports.KbVector.Zero();\n node.position = exports.KbVector.Zero();\n node.scaling = 1;\n node.rotationAngle = 0;\n node.rotationAxis = new exports.KbVector(0, 1, 0);\n node.eulerRotation = exports.KbVector.Zero();\n return node;\n }\n\n const DefaultSketchOptions = {\n faceCamera: true,\n snapSize: 0.5,\n };\n let activeHitPoint;\n let sketchPlaneIntersect;\n class SketchEditor {\n constructor(kbViewer, scene, defaultHighlightColor) {\n this.kbViewer = kbViewer;\n this.scene = scene;\n this.defaultHighlightColor = defaultHighlightColor;\n this.mouseUp = new Subject();\n this.end = new Subject();\n this.sketchPoints = [];\n }\n updateActiveOptions(options) {\n if (this.currentOptions) {\n for (const prop in options) {\n // @ts-expect-error typing not smart enough to know that option objects are the same\n this.currentOptions[prop] = options[prop];\n }\n }\n }\n start(sketch) {\n let origin = exports.KbVector.Zero();\n for (let i = 0; i < sketch.paths.length; i++) {\n const path = sketch.paths[i];\n if (path.controlPoints.length > 0) {\n origin = path.controlPoints[0].position;\n break;\n }\n }\n return this.startOnPlane(sketch, {\n normal: sketch.normal,\n intersect: origin,\n }, undefined, true);\n }\n startOnPlane(sketch, plane, userOptions, useSketchNormal = false) {\n this.activeSketch = sketch;\n sketchPlaneIntersect = plane.intersect;\n const currentCanvas = this.kbViewer.canvas;\n const options = { ...DefaultSketchOptions, ...userOptions };\n this.currentOptions = options;\n this.sketchPoints = sketch.paths.reduce((result, path) => {\n for (const point of path.controlPoints) {\n point._visible = true;\n result.push(point);\n }\n return result;\n }, []);\n const previewVisibleOriginal = sketch._previewVisible;\n sketch._previewVisible = true;\n let planeNormal = getSketchNormalVector3(plane.normal);\n if (useSketchNormal) {\n const sketchNode = this.kbViewer.getRendererNode(sketch);\n if (sketchNode) {\n const matrix = sketchNode.getWorldMatrix().getRotationMatrix();\n planeNormal = Vector3.TransformCoordinates(planeNormal, matrix);\n }\n }\n const planeIntersect = plane.intersect.toVec3();\n const sketchPlane = Plane.FromPositionAndNormal(planeIntersect, planeNormal);\n const originalCamera = this.scene.activeCamera;\n let sketchCamera;\n if (options.faceCamera) {\n originalCamera.detachControl();\n // camera.storeCameraSettings();\n // const viewpoint = new ViewpointNode()\n // camera.moveCameraToViewpoint(viewpoint);\n sketchCamera = new KbArcRotateCamera('sketchEditorView', 0, 0, 10, planeIntersect, this.scene);\n sketchCamera.mode = Camera.ORTHOGRAPHIC_CAMERA;\n sketchCamera.lockRotation = true;\n sketchCamera.wheelPrecision = 60;\n sketchCamera.pinchPrecision = 60;\n switch (plane.normal) {\n case exports.eAxis.x:\n case exports.eAxis.z:\n sketchCamera.upVector = Vector3.Up();\n break;\n case exports.eAxis.y:\n sketchCamera.upVector = Vector3.Backward();\n break;\n default:\n sketchCamera.upVector = Vector3.Cross(planeNormal, planeNormal.equalsToFloats(0, 1, 0) ? Vector3.Right() : Vector3.Up());\n }\n sketchCamera.setPosition(planeNormal.scale(5000).add(planeIntersect));\n sketchCamera.orthoFit(10, 10);\n this.scene.activeCamera = sketchCamera;\n currentCanvas.focus();\n sketchCamera.attachControl();\n this.kbViewer.viewChanged.next();\n }\n const currentCamera = this.scene.activeCamera;\n let dragObjects = [];\n let lastMousePosition = null;\n const sketchParent = sketch._parent ? this.kbViewer.getRendererNode(sketch._parent) : undefined;\n this.kbViewer.interaction.onPointerDown.pipe(takeUntil(this.end)).subscribe(e => {\n if (e.button === 0 && e.pickInfo && e.pickInfo.ray) {\n const deletedPoints = [];\n const ray = e.pickInfo.ray;\n lastMousePosition = getPointOnPlane(ray);\n if (lastMousePosition) {\n roundVectorToPlaces(lastMousePosition, options.snapSize);\n }\n const v1 = new Vector3(-1, -1, -1), v2 = new Vector3(1, 1, 1), collisionSphere = new BoundingSphere(v1, v2);\n const r = CONNECTOR_DIAM / 2;\n for (let i = 0; i < this.sketchPoints.length; i++) {\n const point = this.sketchPoints[i];\n const pointRendered = this.kbViewer.getRendererNode(point);\n if (!pointRendered || pointRendered.isDisposed()) {\n this.sketchPoints.splice(i, 1);\n i--;\n continue;\n }\n const cpCenter = pointRendered.position.clone();\n const cpHandle1 = cpCenter.add(point.direction.toVec3());\n const cpHandle2 = point.direction2 ? cpCenter.add(point.direction2.toVec3()) : undefined;\n if (sketchParent) {\n const matrix = sketchParent.computeWorldMatrix();\n Vector3.TransformCoordinatesToRef(cpCenter, matrix, cpCenter);\n Vector3.TransformCoordinatesToRef(cpHandle1, matrix, cpHandle1);\n if (cpHandle2) {\n Vector3.TransformCoordinatesToRef(cpHandle2, matrix, cpHandle2);\n }\n }\n v1.copyFrom(cpCenter).addInPlaceFromFloats(-r, -r, -r);\n v2.copyFrom(cpCenter).addInPlaceFromFloats(r, r, r);\n collisionSphere.reConstruct(v1, v2);\n let hit = ray.intersectsSphere(collisionSphere);\n let handle = 'position';\n if (!hit) {\n v1.copyFrom(cpHandle1).addInPlaceFromFloats(-r, -r, -r);\n v2.copyFrom(cpHandle1).addInPlaceFromFloats(r, r, r);\n collisionSphere.reConstruct(v1, v2);\n hit = ray.intersectsSphere(collisionSphere);\n handle = 'direction';\n }\n if (!hit && cpHandle2) {\n v1.copyFrom(cpHandle2).addInPlaceFromFloats(-r, -r, -r);\n v2.copyFrom(cpHandle2).addInPlaceFromFloats(r, r, r);\n collisionSphere.reConstruct(v1, v2);\n hit = ray.intersectsSphere(collisionSphere);\n handle = 'direction2';\n }\n if (hit) {\n dragObjects.push({ point, handle });\n // If ctrl key is held down, allow dragging just one point so you can separate paths\n if (e.ctrlKey) {\n break;\n }\n }\n }\n deletedPoints.forEach(point => {\n const index = this.sketchPoints.findIndex(ec => ec.id === point.id);\n this.sketchPoints.splice(index, 1);\n });\n // If both direction handles and positions are hit, only drag the positions and leave the handles alone\n if (dragObjects.find(d => d.handle === 'position')) {\n dragObjects = dragObjects.filter(d => d.handle === 'position');\n }\n if (dragObjects.length === 0) {\n activeHitPoint = undefined;\n }\n else {\n activeHitPoint = dragObjects[0].point;\n currentCamera.detachControl();\n }\n }\n });\n this.kbViewer.interaction.onPointerUp.pipe(takeUntil(this.end)).subscribe(e => {\n dragObjects = [];\n this.mouseUp.next();\n currentCamera.attachControl(currentCanvas);\n });\n this.kbViewer.interaction.onPointerMove.pipe(takeUntil(this.end)).subscribe(e => {\n if (dragObjects.length > 0 && e.pickInfo && e.pickInfo.ray) {\n const ray = e.pickInfo.ray;\n const intersect = getPointOnPlane(ray);\n if (intersect) {\n if (sketchParent) {\n const transformInv = sketchParent.computeWorldMatrix().clone().invert();\n Vector3.TransformCoordinatesToRef(intersect, transformInv, intersect);\n }\n roundVectorToPlaces(intersect, options.snapSize);\n dragObjects.forEach(dragObject => {\n if (dragObject.handle === 'position') {\n roundVectorToPlaces(intersect, options.snapSize);\n dragObject.point.position = exports.KbVector.FromVec3(intersect);\n }\n else {\n const movement = intersect.subtract(dragObject.point.position.toVec3());\n roundVectorToPlaces(intersect, options.snapSize);\n if (dragObject.handle === 'direction') {\n dragObject.point.direction = exports.KbVector.FromVec3(movement);\n }\n else if (dragObject.point.direction2) {\n dragObject.point.direction2 = exports.KbVector.FromVec3(movement);\n }\n }\n });\n }\n }\n });\n this.kbViewer.interaction.onClick.pipe(takeUntil(this.end)).subscribe(e => {\n if (dragObjects.length === 0 && e.pickInfo) {\n if (e.button === 2 && e.canvasX && e.canvasY) {\n if (e.pickInfo.pickedMesh instanceof LinesMesh && e.pickInfo.pickedPoint) {\n this.kbViewer.rightClickMenu.showContextMenu(e.canvasX, e.canvasY, this.getRightClickButtonsForLines(sketch, e.pickInfo.faceId, e.pickInfo.pickedPoint));\n }\n else if (e.pickInfo.pickedMesh instanceof Mesh) {\n const kb3dNode = getByDynamicId(e.pickInfo.pickedMesh.id);\n if (kb3dNode instanceof exports.SketchControlPoint) {\n activeHitPoint = kb3dNode;\n this.kbViewer.rightClickMenu.showContextMenu(e.canvasX, e.canvasY, this.getRightClickButtons());\n }\n }\n }\n else if (e.button === 0 && e.pickInfo.ray) {\n if (this.kbViewer.rightClickMenu.isOpen()) {\n this.kbViewer.rightClickMenu.hideContextMenu();\n return;\n }\n const ray = e.pickInfo.ray;\n const intersect = getPointOnPlane(ray);\n if (intersect) {\n const manager = sketch._manager;\n roundVectorToPlaces(intersect, options.snapSize);\n let lastPath = sketch.paths.length > 0 ? sketch.paths[sketch.paths.length - 1] : undefined;\n if (!lastPath) {\n lastPath = manager.create(Token.SketchPath);\n lastPath.name = 'Path';\n sketch.addChildren(lastPath);\n }\n const c = createSketchPointForEditor(manager, lastPath);\n if (sketch._parent instanceof exports.SpaceNode) {\n const rendererNode = this.kbViewer.getRendererNode(sketch);\n if (rendererNode) {\n Vector3.TransformCoordinatesToRef(intersect, rendererNode.computeWorldMatrix().clone().invert(), intersect);\n }\n c.position = exports.KbVector.FromVec3(intersect);\n }\n this.sketchPoints.push(c);\n }\n }\n }\n });\n this.end.pipe(take(1)).subscribe(() => {\n if (!previewVisibleOriginal) {\n sketch.paths.forEach(path => {\n for (const point of path.controlPoints) {\n point._visible = false;\n }\n });\n }\n originalCamera.attachControl(currentCanvas);\n this.scene.activeCamera = originalCamera;\n if (sketchCamera) {\n sketchCamera.dispose();\n }\n sketch._previewVisible = previewVisibleOriginal;\n });\n function getPointOnPlane(ray) {\n const d = ray.intersectsPlane(sketchPlane);\n if (typeof d === 'number') {\n return ray.origin.add(ray.direction.scale(d));\n }\n return null;\n }\n }\n clear() {\n this.end.next();\n this.kbViewer.rightClickMenu.hideContextMenu();\n this.currentOptions = undefined;\n this.activeSketch = undefined;\n }\n dispose() {\n this.kbViewer.rightClickMenu.hideContextMenu();\n this.end.next();\n this.end.complete();\n this.mouseUp.complete();\n this.activeSketch = undefined;\n }\n deleteSketchPoint() {\n if (this.kbViewer.sceneNode._manager && activeHitPoint) {\n const path = activeHitPoint._parent;\n const targetPoint = activeHitPoint;\n if (path instanceof exports.SketchPath) {\n const index = path.controlPoints.findIndex(ec => ec.id === targetPoint.id);\n path.controlPoints.splice(index, 1);\n }\n }\n }\n deleteSketchPath() {\n if (this.kbViewer.sceneNode._manager && activeHitPoint) {\n if (this.activeSketch) {\n const sketch = this.activeSketch;\n const targetPoint = activeHitPoint;\n sketch.paths.forEach(path => {\n if (path instanceof exports.SketchPath) {\n const index = path.controlPoints.findIndex(ec => { var _a; return ec._dynamicId === ((_a = targetPoint._parent) === null || _a === void 0 ? void 0 : _a._dynamicId); });\n sketch.paths.splice(index, 1);\n }\n });\n }\n }\n }\n separateSketchNode(sketchpathType) {\n var _a, _b;\n if (this.kbViewer.sceneNode._manager && activeHitPoint) {\n // let targetControlPoint: SketchControlPoint | undefined;\n const manager = this.kbViewer.sceneNode._manager;\n if (this.activeSketch) {\n const sketch = this.activeSketch;\n const targetPoint = activeHitPoint;\n for (let pathIndex = 0; pathIndex < sketch.paths.length; pathIndex++) {\n const path = sketch.paths[pathIndex];\n const cpIndex = path.controlPoints.findIndex(ec => ec._dynamicId === targetPoint._dynamicId);\n if (cpIndex === -1) {\n continue;\n }\n if (path.type === sketchpathType) {\n return;\n }\n if (path.controlPoints.length === 1 && sketchpathType === exports.eSketchPathType.arcThrough3Points) {\n const newControlPoint2 = createSketchPointForEditor(manager, path);\n const newControlPoint3 = createSketchPointForEditor(manager, path);\n let sketchNormal = new Vector3(1, 0, 0);\n if (sketch.normal === exports.eAxis.y) {\n sketchNormal = new Vector3(0, 1, 0);\n }\n else if (sketch.normal === exports.eAxis.z) {\n sketchNormal = new Vector3(0, 0, 1);\n }\n const controlPointMesh = this.kbViewer.getRendererNode(sketch.paths[pathIndex].controlPoints[0]);\n const pointOnePosition = sketchNormal.cross(sketchPlaneIntersect.toVec3().subtract(controlPointMesh.getAbsolutePosition()));\n newControlPoint2.position = new exports.KbVector(pointOnePosition.x, pointOnePosition.y, pointOnePosition.z);\n const pointTwoPosition = sketchNormal.cross(pointOnePosition.subtract(controlPointMesh.getAbsolutePosition()));\n newControlPoint3.position = new exports.KbVector(pointTwoPosition.x, pointTwoPosition.y, pointTwoPosition.z);\n path._expanded = true;\n path.type = sketchpathType;\n this.sketchPoints.push(newControlPoint2);\n this.sketchPoints.push(newControlPoint3);\n break;\n }\n else if (cpIndex === 0 && ((_a = sketch.paths[pathIndex - 1]) === null || _a === void 0 ? void 0 : _a.type) === sketchpathType) {\n // If an adjacent path has the same path type as the one we're making, we'll just add the control point to it\n // Previous path matching case\n const cp = path.controlPoints.shift();\n sketch.paths[pathIndex - 1].controlPoints.push(cp.clone());\n break;\n }\n else if (\n // Next path matching case\n cpIndex === path.controlPoints.length - 1 &&\n ((_b = sketch.paths[pathIndex + 1]) === null || _b === void 0 ? void 0 : _b.type) === sketchpathType) {\n const cp = path.controlPoints.pop();\n sketch.paths[pathIndex + 1].controlPoints.push(cp.clone());\n break;\n }\n else if (path.controlPoints.length > 1) {\n let selectedControlPoints;\n let previousPath;\n let selectedPath;\n let nextPath;\n if (cpIndex === 0) {\n selectedPath = path;\n }\n else {\n previousPath = path;\n selectedControlPoints = Array.from(previousPath.controlPoints.splice(cpIndex, path.controlPoints.length - cpIndex));\n selectedPath = manager.create(Token.SketchPath);\n sketch.paths.splice(pathIndex + 1, 0, selectedPath);\n selectedPath.name = 'Path';\n selectedControlPoints.reverse();\n while (selectedControlPoints.length > 0) {\n const cp = manager.clone(selectedControlPoints.pop());\n cp._visible = true;\n selectedPath.controlPoints.push(cp);\n this.sketchPoints.push(cp);\n }\n }\n if (selectedPath.controlPoints.length > 2) {\n let nextPathPoints;\n if (sketchpathType == exports.eSketchPathType.arcThrough3Points &&\n selectedPath.controlPoints.length > 3) {\n nextPathPoints = Array.from(selectedPath.controlPoints.splice(3, selectedPath.controlPoints.length - 3));\n }\n else {\n nextPathPoints = Array.from(selectedPath.controlPoints.splice(1, selectedPath.controlPoints.length - 1));\n }\n nextPath = manager.create(Token.SketchPath);\n sketch.paths.splice(pathIndex + 2, 0, nextPath);\n nextPath.name = 'Path';\n nextPathPoints.reverse();\n while (nextPathPoints.length > 0) {\n const cp = manager.clone(nextPathPoints.pop());\n cp._visible = true;\n nextPath.controlPoints.push(cp);\n this.sketchPoints.push(cp);\n }\n }\n selectedPath._expanded = true;\n selectedPath.type = sketchpathType;\n if (previousPath) {\n const match = manager.clone(selectedPath.controlPoints[0]);\n match._visible = true;\n previousPath.controlPoints.push(match);\n this.sketchPoints.push(match);\n }\n if (nextPath && nextPath.controlPoints.length > 0) {\n const match = manager.clone(nextPath.controlPoints[0]);\n match._visible = true;\n selectedPath.controlPoints.push(match);\n this.sketchPoints.push(match);\n }\n break;\n }\n }\n }\n }\n }\n addSketchNodeOnPath(node, faceId, pickedPoint) {\n const index = node._lineSegmentStartingPoints[faceId];\n let counter = 0;\n const manager = node._manager;\n for (let pathIndex = 0; pathIndex < node.paths.length; pathIndex++) {\n const path = node.paths[pathIndex];\n for (let cpIndex = 0; cpIndex < node.paths.length; cpIndex++) {\n const controlPoint = path.controlPoints[cpIndex];\n if (counter === index) {\n const spliceIndex = path.controlPoints.indexOf(controlPoint);\n const sketchControlPoint = manager.create(Token.SketchControlPoint);\n sketchControlPoint.position = exports.KbVector.FromVec3(pickedPoint);\n path.controlPoints.splice(spliceIndex + 1, 0, sketchControlPoint);\n sketchControlPoint._visible = true;\n sketchControlPoint._expanded = true;\n this.sketchPoints.push(sketchControlPoint);\n return;\n }\n counter++;\n }\n }\n }\n getRightClickButtons() {\n return [\n { label: 'delete node', callback: () => this.deleteSketchPoint() },\n { label: 'delete path', callback: () => this.deleteSketchPath() },\n { label: 'convert to cubic bezier', callback: () => this.separateSketchNode(exports.eSketchPathType.cubicBezier) },\n {\n label: 'convert to quadratic bezier',\n callback: () => this.separateSketchNode(exports.eSketchPathType.quadraticBezier),\n },\n { label: 'convert to hermite', callback: () => this.separateSketchNode(exports.eSketchPathType.hermite) },\n { label: 'convert to catmullrom', callback: () => this.separateSketchNode(exports.eSketchPathType.catmullRom) },\n { label: 'convert to points', callback: () => this.separateSketchNode(exports.eSketchPathType.points) },\n {\n label: 'arc through 3 points',\n callback: () => this.separateSketchNode(exports.eSketchPathType.arcThrough3Points),\n },\n ];\n }\n getRightClickButtonsForLines(node, faceId, pickedPoint) {\n return [\n {\n label: 'add point on line',\n callback: () => this.addSketchNodeOnPath(node, faceId, pickedPoint),\n },\n ];\n }\n }\n\n function downloadBlob(blob, filename = 'download') {\n const url = URL.createObjectURL(blob);\n const a = document.createElement('a');\n a.href = url;\n a.download = filename || 'download';\n const clickHandler = () => {\n setTimeout(() => {\n URL.revokeObjectURL(url);\n a.removeEventListener('click', clickHandler);\n }, 150);\n };\n a.addEventListener('click', clickHandler, false);\n a.click();\n }\n function dataURLtoBlob(dataurl) {\n const arr = dataurl.split(',');\n const mime = arr[0].match(/:(.*?);/)[1];\n const bstr = atob(arr[1]);\n let n = bstr.length;\n const u8arr = new Uint8Array(n);\n while (n--) {\n u8arr[n] = bstr.charCodeAt(n);\n }\n return new Blob([u8arr], { type: mime });\n }\n\n (function (eSnapshotType) {\n eSnapshotType[\"image\"] = \"image\";\n eSnapshotType[\"blob\"] = \"blob\";\n eSnapshotType[\"dataUrl\"] = \"dataUrl\";\n })(exports.eSnapshotType || (exports.eSnapshotType = {}));\n class SnapshotModule {\n constructor(v, scene) {\n this.v = v;\n this.scene = scene;\n }\n take(args) {\n if (args.dataType == exports.eSnapshotType.image) {\n return this.download(args);\n }\n else if (args.dataType == exports.eSnapshotType.blob) {\n return this.takeAsBlob(args);\n }\n else {\n return this.takeAsDataUrl(args);\n }\n }\n takeAsDataUrl(args) {\n //set the viewpoint\n let viewpointSet = false;\n if (args.viewpoint) {\n const viewpoint = this.v.sceneNode._manager.getNodeByIdOrName(args.viewpoint);\n if (!viewpoint) {\n throw `Could not find viewpoint ${args.viewpoint} for snapshot`;\n }\n else {\n this.v.camera.storeCameraSettings();\n this.v.camera.moveCameraToViewpoint(viewpoint);\n viewpointSet = true;\n }\n }\n return this.v\n .whenAssetsLoaded()\n .then(() => {\n if (args.frameCamera) {\n this.v.camera.frameCamera(this.v.sceneNode, {\n padding: args.framePadding,\n });\n }\n return this.v.forceAndWaitForNextRender().then(() => {\n return this.v.forceAndWaitForNextRender();\n });\n })\n .then(() => {\n const engine = this.v.scene.getEngine();\n const mimeType = args.mimeType || 'image/png';\n const snapshotCanvas = document.createElement('canvas');\n snapshotCanvas.width = args.width;\n snapshotCanvas.height = args.height;\n const renderContext = snapshotCanvas.getContext('2d');\n const ratio = engine.getRenderWidth() / engine.getRenderHeight();\n let newWidth = args.width;\n let newHeight = newWidth / ratio;\n if (newHeight > args.height) {\n newHeight = args.height;\n newWidth = newHeight * ratio;\n }\n const offsetX = Math.max(0, args.width - newWidth) / 2;\n const offsetY = Math.max(0, args.height - newHeight) / 2;\n const renderingCanvas = engine.getRenderingCanvas();\n if (renderContext && renderingCanvas) {\n renderContext.drawImage(renderingCanvas, offsetX, offsetY, newWidth, newHeight);\n }\n const dataUrl = snapshotCanvas.toDataURL(mimeType);\n if (viewpointSet) {\n this.v.camera.restoreCameraSettings();\n }\n snapshotCanvas.width = 0;\n snapshotCanvas.height = 0;\n return dataUrl;\n });\n }\n takeAsBlob(args) {\n return this.takeAsDataUrl(args).then(dataUrl => dataURLtoBlob(dataUrl));\n }\n download(args) {\n return this.takeAsBlob(args).then(blob => downloadBlob(blob, args.filename));\n }\n }\n\n var maskFragmentShader = \"precision highp float;\\r\\n \\r\\nuniform vec4 lineColor;\\r\\n\\r\\nvoid main(void) {\\r\\n gl_FragColor = lineColor;\\r\\n}\";\n\n var maskVertexShader = \"precision highp float;\\r\\n // Attributes\\r\\nattribute vec3 position;\\r\\n// Uniforms\\r\\nuniform mat4 worldViewProjection;\\r\\nvoid main(void) {\\r\\n gl_Position = worldViewProjection * vec4(position, 1.0);\\r\\n}\\r\\n\\r\\n\";\n\n var outlineFragmentShader = \"#ifdef GL_ES\\r\\nprecision highp float;\\r\\n#endif\\r\\n\\r\\n// Samplers\\r\\nvarying vec2 vUV;\\r\\nuniform sampler2D textureSampler;\\r\\nuniform sampler2D maskSampler;\\r\\n\\r\\n// Parameters\\r\\nuniform vec2 screenSize;\\r\\nuniform float threshold;\\r\\nuniform float lineWidth;\\r\\nuniform bool debug;\\r\\n\\r\\nvoid main(void) \\r\\n{\\r\\n // Calculating how far away to check pixels for different colors to determine whether to draw an outline pixel\\r\\n float dx = (1.0 / screenSize.x) * lineWidth;\\r\\n float dy = (1.0 / screenSize.y) * lineWidth;\\r\\n\\r\\n float xMin = dx * 0.5;\\r\\n float xMax = 1. - dx * 0.5;\\r\\n float yMin = dy * 0.5;\\r\\n float yMax = 1. - dy * 0.5;\\r\\n\\r\\n float right = clamp(vUV.x + dx, xMin, xMax);\\r\\n float left = clamp(vUV.x - dx, xMin, xMax);\\r\\n float top = clamp(vUV.y + dy, yMin, yMax);\\r\\n float bottom = clamp(vUV.y - dy, yMin, yMax);\\r\\n\\r\\n // calculating the uvs corresponsing to the pixels from the step above\\r\\n vec2 uvCenter = vUV;\\r\\n vec2 uvLeft = vec2(left, uvCenter.y);\\r\\n vec2 uvRight = vec2(right, uvCenter.y);\\r\\n vec2 uvTop = vec2(uvCenter.x, top);\\r\\n vec2 uvBottom = vec2(uvCenter.x, bottom);\\r\\n vec2 uvTopLeft = vec2(left, top);\\r\\n vec2 uvTopRight = vec2(right, top);\\r\\n vec2 uvBottomLeft = vec2(left, bottom);\\r\\n vec2 uvBottomRight = vec2(right, bottom);\\r\\n\\r\\n // getting the colors of the pixels at the 4 corners of our check box\\r\\n vec4 mCenter = texture2D(maskSampler, uvCenter);\\r\\n vec4 mleft = texture2D(maskSampler, uvLeft);\\r\\n vec4 mRight = texture2D(maskSampler, uvRight);\\r\\n vec4 mTop = texture2D(maskSampler, uvTop);\\r\\n vec4 mBottom = texture2D(maskSampler, uvBottom);\\r\\n vec4 mTopleft = texture2D(maskSampler, uvTopLeft);\\r\\n vec4 mTopRight = texture2D(maskSampler, uvTopRight);\\r\\n vec4 mBottomLeft = texture2D(maskSampler, uvBottomLeft);\\r\\n vec4 mBottomRight = texture2D(maskSampler, uvBottomRight);\\r\\n\\r\\n // getting the difference in colors between the center pixel and the other pixels\\r\\n vec4 dL = abs(mCenter - mleft);\\r\\n vec4 dR = abs(mCenter - mRight);\\r\\n vec4 dT = abs(mCenter - mTop);\\r\\n vec4 dB = abs(mCenter - mBottom);\\r\\n vec4 dTL = abs(mCenter - mTopleft);\\r\\n vec4 dTR = abs(mCenter - mTopRight);\\r\\n vec4 dBL = abs(mCenter - mBottomLeft);\\r\\n vec4 dBR = abs(mCenter - mBottomRight);\\r\\n\\r\\n // getting the maximum difference from the center pixel to any of the other pixels\\r\\n vec4 totalD = max(dT, dR);\\r\\n totalD = max(totalD, dB);\\r\\n totalD = max(totalD, dL);\\r\\n totalD = max(totalD, dTL);\\r\\n totalD = max(totalD, dTR);\\r\\n totalD = max(totalD, dBL);\\r\\n totalD = max(totalD, dBR);\\r\\n\\r\\n float delta = length(totalD) * mCenter.a;\\r\\n if(delta < threshold) {\\r\\n delta = 0.0;\\r\\n }\\r\\n\\r\\n if (debug == true) {\\r\\n gl_FragColor = texture2D(maskSampler, vUV);\\r\\n } else {\\r\\n // if(delta > threshold) {\\r\\n // gl_FragColor = texture2D(maskSampler, vUV);\\r\\n // } else {\\r\\n // gl_FragColor = texture2D(textureSampler, vUV);\\r\\n // }\\r\\n // Merge the real scene output with the outline\\r\\n vec4 albedoColor = texture2D(textureSampler, vUV);\\r\\n gl_FragColor = mCenter.rgba * delta + albedoColor * (1.0 - delta);\\r\\n // gl_FragColor = mCenter.rgba * delta;\\r\\n }\\r\\n}\";\n\n Effect.ShadersStore['customMaskVertexShader'] = maskVertexShader;\n Effect.ShadersStore['customMaskFragmentShader'] = maskFragmentShader;\n Effect.ShadersStore['simpleOutlineFragmentShader'] = outlineFragmentShader;\n class Highlighter {\n constructor(kbViewer, scene, defaultHighlightColor, cameraManager) {\n this.kbViewer = kbViewer;\n this.scene = scene;\n this.defaultHighlightColor = defaultHighlightColor;\n this.colorMap = new Map();\n this.nodeColors = new Map();\n this.highlighterDb = new Map();\n this.engine = scene.getEngine();\n this.highlightPipeline = new PostProcessRenderPipeline(this.engine, 'highlightPipeline');\n this.renderTarget = new RenderTargetTexture('Highlight render target', 1024, scene, { samples: 2 });\n this.renderTarget.renderList = [];\n this.renderTarget.clearColor = new Color4(0, 0, 0, 0);\n scene.customRenderTargets.push(this.renderTarget);\n this.postProcess = new PostProcess('Highlighter post process', 'simpleOutline', ['screenSize', 'threshold', 'lineWidth', 'debug'], ['maskSampler'], 1.0, null, undefined, this.engine, true);\n // This may help with the highlight separating from the scene meshes, if we can ge the highlight to combine with the scene\n // this.postProcess.autoClear = false;\n // this.postProcess.alphaMode = Engine.ALPHA_COMBINE;\n this.postProcess.samples = 4;\n this.postProcess.onApply = effect => {\n effect.setFloat2('screenSize', this.postProcess.width, this.postProcess.height);\n effect.setFloat('threshold', 0.3);\n effect.setFloat('lineWidth', 4.0);\n effect.setBool('debug', false);\n effect.setTexture('maskSampler', this.renderTarget);\n window.enableHighlightDebug = () => {\n enableHighlightDebug(effect);\n };\n };\n const highlightEffect = new PostProcessRenderEffect(this.engine, 'highlightEffect', () => [this.postProcess], false);\n this.highlightPipeline.addEffect(highlightEffect);\n this.scene.postProcessRenderPipelineManager.addPipeline(this.highlightPipeline);\n cameraManager.setPipeline(this.highlightPipeline);\n function enableHighlightDebug(effect) {\n effect.setBool('debug', true);\n }\n }\n has(node) {\n return this.highlighterDb.has(node._dynamicId);\n }\n add(node, color = this.defaultHighlightColor) {\n this.startPostProcess();\n const renderNode = this.kbViewer.getRendererNode(node);\n this.highlighterDb.set(node._dynamicId, color);\n this.addHighlightToRendered(renderNode, color);\n }\n remove(node) {\n var _a;\n if (this.highlighterDb.has(node._dynamicId)) {\n const renderNode = this.kbViewer.getRendererNode(node);\n if (((_a = this.activeHover) === null || _a === void 0 ? void 0 : _a._dynamicId) !== node._dynamicId) {\n // Don't remove the highlight from a hover. Once the hover clears the highlight will be cleared.\n this.removeHighlightFromRendered(renderNode);\n this.highlighterDb.delete(node._dynamicId);\n }\n }\n }\n hover(node) {\n if (node != this.activeHover) {\n this.startPostProcess();\n this.clearHover();\n this.activeHover = node;\n const renderNode = this.kbViewer.getRendererNode(node);\n this.addHighlightToRendered(renderNode, this.defaultHighlightColor);\n this.kbViewer.canvas.style.cursor = 'pointer';\n this.scene.defaultCursor = 'pointer';\n this.kbViewer.forceRenderOnNextFrame = true;\n }\n }\n clearHover() {\n if (this.activeHover) {\n const renderNode = this.kbViewer.getRendererNode(this.activeHover);\n this.removeHighlightFromRendered(renderNode);\n //if the node had a highlight previously, restore it\n if (this.highlighterDb.has(this.activeHover._dynamicId)) {\n const color = this.highlighterDb.get(this.activeHover._dynamicId);\n this.addHighlightToRendered(renderNode, color);\n }\n // const targets = renderNode instanceof AbstractMesh ? [renderNode] : renderNode.getChildMeshes();\n // for (const target of targets) {\n // if (this.highlighterDb.has(target)) {\n // const color = this.highlighterDb.get(target)!;\n // this.addHighlightToRendered(target, color);\n // }\n // }\n this.activeHover = undefined;\n this.kbViewer.canvas.style.cursor = 'default';\n this.scene.defaultCursor = 'default';\n this.kbViewer.forceRenderOnNextFrame = true;\n }\n }\n clearAll() {\n this.colorMap.forEach(entry => {\n this.renderTarget.setMaterialForRendering(Array.from(entry.meshes), undefined);\n });\n if (this.renderTarget.renderList) {\n while (this.renderTarget.renderList.length > 0) {\n this.renderTarget.renderList.pop();\n }\n }\n this.nodeColors.clear();\n this.highlighterDb.clear();\n this.colorMap.clear();\n this.activeHover = undefined;\n }\n addHighlightToRendered(node, color) {\n var _a;\n const targets = node instanceof AbstractMesh ? [node] : node.getChildMeshes();\n for (const renderNode of targets) {\n if (((_a = this.renderTarget.renderList) === null || _a === void 0 ? void 0 : _a.contains(renderNode)) === false) {\n if (!this.renderTarget.renderList) {\n this.renderTarget.renderList = [];\n }\n this.renderTarget.renderList.push(renderNode);\n }\n let colorMapKey = '';\n if (color) {\n colorMapKey = this.colorToString(color);\n }\n else {\n colorMapKey = this.colorToString(this.defaultHighlightColor);\n }\n this.addNodeToColorMap(colorMapKey, renderNode, color);\n this.nodeColors.set(renderNode, colorMapKey);\n }\n }\n removeHighlightFromRendered(node) {\n var _a;\n const targets = node instanceof AbstractMesh ? [node] : node.getChildMeshes();\n for (const renderNode of targets) {\n const colorKey = this.nodeColors.get(renderNode);\n (_a = this.renderTarget.renderList) === null || _a === void 0 ? void 0 : _a.remove(renderNode);\n if (colorKey) {\n this.removeNodeFromColorMap(colorKey, renderNode);\n this.nodeColors.delete(renderNode);\n }\n }\n }\n addNodeToColorMap(colorMapKey, node, color) {\n let colorMapEntry = this.colorMap.get(colorMapKey);\n if (!colorMapEntry) {\n const newShader = new ShaderMaterial('customMaskMat_' + colorMapKey, this.scene, 'customMask', {\n attributes: ['position', 'uv'],\n uniforms: ['worldViewProjection', 'color', 'lineColor'],\n });\n if (color) {\n newShader.setColor4('lineColor', color.toColor4());\n }\n else {\n newShader.setColor4('lineColor', this.defaultHighlightColor.toColor4());\n }\n colorMapEntry = {\n material: newShader,\n meshes: new Set(),\n };\n }\n colorMapEntry.meshes.add(node);\n this.renderTarget.setMaterialForRendering(Array.from(colorMapEntry.meshes), colorMapEntry.material);\n }\n removeNodeFromColorMap(colorMapKey, node) {\n const colorMapEntry = this.colorMap.get(colorMapKey);\n if (colorMapEntry) {\n colorMapEntry.meshes.delete(node);\n this.renderTarget.setMaterialForRendering(node, undefined);\n }\n }\n colorToString(color) {\n return color.cssString();\n }\n startPostProcess() { }\n }\n\n class RightClickMenu {\n constructor(viewer, scene) {\n this.viewer = viewer;\n this.scene = scene;\n this.contextMenuOpen = false;\n }\n isOpen() {\n return this.contextMenuOpen;\n }\n showContextMenu(xPosition, yPosition, buttons) {\n this.createContextMenu(xPosition, yPosition, buttons);\n if (this.viewer.uiOverlay && this.contextmenu) {\n this.viewer.container.appendChild(this.contextmenu);\n this.contextMenuOpen = true;\n }\n }\n hideContextMenu() {\n if (this.contextmenu) {\n this.contextmenu.style.display = 'none';\n this.contextMenuOpen = false;\n }\n }\n createContextMenu(xPosition, yPosition, buttons) {\n var _a;\n //const buttons = [];\n if (this.contextmenu && this.viewer.container.contains(this.contextmenu)) {\n (_a = this.viewer.container) === null || _a === void 0 ? void 0 : _a.removeChild(this.contextmenu);\n }\n const contextMenu = document.createElement('div');\n contextMenu.style.position = 'absolute';\n contextMenu.style.top = yPosition.toString() + 'px';\n contextMenu.style.left = (xPosition + 10).toString() + 'px';\n contextMenu.style.width = 'auto';\n contextMenu.style.height = 'auto';\n contextMenu.style.zIndex = '2';\n contextMenu.addEventListener('click', e => {\n if (e.button === 2) {\n e.preventDefault();\n e.stopImmediatePropagation();\n }\n });\n document.addEventListener('click', e => {\n if (e.button != 2) {\n this.hideContextMenu();\n }\n });\n document.addEventListener('wheel', e => {\n this.hideContextMenu();\n });\n contextMenu.addEventListener('scroll', e => {\n if (this.contextmenu) {\n this.contextmenu.style.display = 'none';\n }\n });\n this.contextmenu = contextMenu;\n buttons.forEach(button => {\n var _a;\n const newButton = document.createElement('div');\n newButton.innerText = button.label;\n newButton.addEventListener('click', e => {\n button.callback();\n this.hideContextMenu();\n });\n (_a = this.contextmenu) === null || _a === void 0 ? void 0 : _a.appendChild(newButton);\n newButton.style.zIndex = '100000';\n newButton.style.color = 'white';\n newButton.style.background = 'rgba(36,36,36,255)';\n newButton.style.cursor = 'pointer';\n newButton.style.padding = '10px';\n newButton.addEventListener('mouseover', e => {\n newButton.style.background = 'rgba(61,61,61,255)';\n });\n newButton.addEventListener('mouseout', e => {\n newButton.style.background = 'rgba(36,36,36,255)';\n });\n });\n }\n }\n\n const FADE_TIME = 300;\n /**\n * Applcation class for the viewer, to contain all of the initializers and hooks and connect the modules and 3d engine.\n * @param canvasElement An HTML DOM element that the viewer will be attached to\n */\n class KbViewer {\n constructor(options) {\n this.forceRenderOnNextFrame = false;\n this.pauseRender = false;\n this.MeshWarningTracker = new Map();\n this.normalSmoothingWarnings = new Map();\n /** Events / Observables */\n /** observable for tracking when something changed that could affect the position of a node on the screen */\n this.viewChanged = new Subject();\n this.afterRender = new Subject();\n this.fpsTracker = new Subject();\n this.polygonTracker = new Subject();\n this.warningTracker = new Subject();\n this.meshWarningsNeedsUpdate = new Subject();\n this.materialWarningsNeedsUpdate = new Subject();\n this.animationWarningsNeedsUpdate = new Subject();\n this.disposed = false;\n this.forceRunRenderer = false;\n this.isDevEnvironment = false;\n /** tracks babylon nodes who are being tracked for positions (hotspots) */\n this.positionTrackedNodes = new Map();\n this.nextFrameCallbacks = [];\n this._renderLoopRunning = false;\n this.allAssetsLoaded = false;\n /** tracks any additional promises that renders and load spinners should wait for */\n this.additionalLoadBlockers = [];\n this.options = {\n highlightColor: exports.KbColor.KbmaxOrange(),\n editorMode: false,\n forceDisablePostProcess: false,\n enableScreenshots: false,\n debugMode: false,\n webworkerPath: '/worker3d.js',\n ...options,\n };\n if (typeof window !== 'undefined' && options.webworkerPath && !options.webworkerPath.startsWith('http')) {\n this.options.webworkerPath = `${window.location.protocol}//${window.location.host}${this.options.webworkerPath}`;\n }\n Kb3dWorker.workerPath = this.options.webworkerPath;\n // Create canvas and engine.\n this.container = options.container;\n this.canvas = this.createCanvas();\n // Anti-aliasing=true (second parameter) causes the canvas to not get cleared on iOS 15.4. Warrants testing in future iOS updates.\n const iOS = /(iPad|iPhone|iPod)/g.test(navigator.userAgent);\n this.engine = new Engine(this.canvas, !iOS, { stencil: true, preserveDrawingBuffer: this.options.enableScreenshots }, true); //set adaptToDeviceRatio to true, and let the hardware scaling optimizer take over from there\n this.engine.setAlphaMode(Engine.ALPHA_COMBINE);\n this.glInfo = this.engine.getGlInfo();\n // eslint-disable-next-line no-console\n console.log('WebGL Renderer', this.glInfo.vendor, this.glInfo.renderer, this.glInfo.version);\n // Create the scene.\n const scene = this.createScene();\n this.scene = scene;\n this.scene.hoverCursor = 'pointer';\n this.utilityLayer = new UtilityLayerRenderer(this.scene);\n this.utilityLayer.utilityLayerScene.useRightHandedSystem = true;\n this.utilityLayer.processAllEvents = true;\n this.camera = new CameraTools(this.scene, this.canvas, this);\n this.bbCache = new BoundingBoxCache(this);\n // var hdrTexture = CubeTexture.CreateFromPrefilteredData('/assets/environments/environment.dds', scene);\n // hdrTexture.gammaSpace = false;\n // scene.environmentTexture = hdrTexture;\n this.highlightColor = options.highlightColor || exports.KbColor.KbmaxOrange();\n this.gizmos = new GizmoTools(scene, this);\n this.rightClickMenu = new RightClickMenu(this, this.scene);\n this.animation = new AnimationModule(this, this.scene);\n this.highlighter = new Highlighter(this, this.scene, this.highlightColor, this.camera);\n this.interaction = new Interaction(this, this.scene, this.highlighter);\n this.sketchEditor = new SketchEditor(this, this.scene, this.highlightColor);\n this.gui = new GuiHelper(scene, this.highlighter);\n this.ar = new AR(this, scene);\n this.snapshot = new SnapshotModule(this, scene);\n this.rendererManager = new RendererManager(this, scene);\n this.interaction.attachScene(this.utilityLayer.utilityLayerScene, true);\n this.gltfExporter = new GLTFKb3dExporter(scene);\n this.objExporter = new ObjKb3dExporter(scene, this.rendererManager);\n }\n get settings() {\n return SettingsManagerService;\n }\n setRenderMode(mode, force) {\n this.rendererManager.setRenderMode(mode, this.sceneNode, force);\n }\n loadScene(o) {\n if (!o.sceneNode._manager) {\n throw 'sceneNode must be constructed with a manager';\n }\n this.sceneNode = o.sceneNode;\n this.manager = o.sceneNode._manager;\n this.isDevEnvironment = this.manager.isDevEnvironment;\n const loadingExperience = o.loadingExperience || this.sceneNode.loadingExperience;\n this.createLoadingExperience(loadingExperience, this.sceneNode.loadingImagePath);\n let viewpoint = o.viewpoint ||\n (o.sceneNode.viewpointId ? this.manager.getById(o.sceneNode.viewpointId) : undefined);\n if (!viewpoint) {\n viewpoint = this.manager.create(Token.ViewpointNode);\n viewpoint.type = exports.eViewpointType.orbit;\n viewpoint.targetMode = exports.eViewpointTargetMode.point;\n viewpoint.position = new exports.KbVector(3, 3, 3);\n viewpoint.targetPoint = exports.KbVector.Zero();\n }\n this.camera.moveCameraToViewpoint(viewpoint);\n // Start render loop.\n this.startRender();\n // The canvas/window resize event handler.\n // window.addEventListener('resize', () => {\n // if (!this.ar.xrLoading && !this.ar.xrActive) { //don't mess with the AR canvas resize\n // this.engine.resize();\n // this.viewChanged.next();\n // }\n // });\n const promises = [this.whenAssetsLoaded()];\n if (o.deferLoadingPromise)\n promises.push(o.deferLoadingPromise);\n return Promise.all(promises).then(() => {\n var _a;\n const http = (_a = this.sceneNode._manager) === null || _a === void 0 ? void 0 : _a.getService(Token.HttpService);\n let frame = true;\n if (this.sceneNode.viewpointId)\n frame = this.sceneNode.frameViewpoint;\n if (o.frameViewpoint != null)\n frame = o.frameViewpoint;\n if (frame) {\n this.camera.frameCamera(this.sceneNode);\n }\n this.hideLoadingExperience();\n // Wait for everything to be loaded and processed before trying to optimize scene because the scene optimizer watches the FPS and we want the state to be stable\n // Disable the graphics optimization (for example when rendering)\n if (!this.options.forceDisableGraphicsOptimization) {\n this.waitForNextKb3dRender().then(() => {\n this.optimizeScene(this.scene);\n });\n }\n this.meshWarningsNeedsUpdate\n .pipe(throttleTime(1000, undefined, { leading: true, trailing: true }))\n .subscribe(() => {\n this.updateMeshWarnings();\n this.updateTextureWarnings();\n });\n this.materialWarningsNeedsUpdate\n .pipe(throttleTime(15000, undefined, { leading: true, trailing: false }))\n .subscribe(() => {\n this.updateMaterialWarnings();\n });\n this.animationWarningsNeedsUpdate.subscribe(value => {\n this.updateAnimationWarnings(value);\n });\n this.updateTextureWarnings();\n this.updateLightWarnings();\n this.updateMaterialWarnings();\n if (http) {\n http.executeDeferredCalls();\n }\n });\n }\n getNode(id) {\n return this.manager.getById(id);\n }\n getRendererNode(n) {\n if (isSceneNode(n) && n.isNested() === false) {\n return this.rendererManager.sceneRenderer.get3d(n);\n }\n else if (isSceneNode(n) && n.isNested()) {\n return this.rendererManager.mesh.get3d(n);\n }\n else if (isLightNode(n)) {\n return this.rendererManager.light.get3d(n);\n }\n else if (isSpaceNode(n)) {\n return this.rendererManager.mesh.get3d(n);\n }\n else if (isMaterialNode(n)) {\n return this.rendererManager.material.get3d(n);\n }\n else if (isConnector(n)) {\n return this.rendererManager.connector.get3d(n);\n }\n else if (isSketchControlPoint(n)) {\n return this.rendererManager.controlPointRenderer.get3d(n);\n }\n else if (isTextureLayer(n)) {\n return this.rendererManager.textures.get3d(n);\n }\n return undefined;\n }\n getFPS() {\n return this.engine.getFps().toFixed();\n }\n getWarnings() {\n return this.warningMessages;\n }\n getMeshBoundingBox(node, vertexGroups) {\n let mesh = this.rendererManager.mesh.get3d(node);\n let newMesh;\n if (vertexGroups) {\n const newIndices = [];\n const newPositions = [];\n let indices;\n let positions;\n if (mesh instanceof Mesh && mesh.geometry) {\n positions = mesh.geometry.getVerticesData(VertexBuffer.PositionKind);\n indices = mesh.geometry.getIndices();\n }\n else {\n indices = [];\n positions = [];\n }\n for (let i = 0; i < vertexGroups.length; i++) {\n const vertexGroup = vertexGroups[i];\n if (vertexGroup && vertexGroup.startIndex && vertexGroup.endIndex) {\n for (let j = vertexGroup.startIndex; j < vertexGroup.endIndex; j++) {\n newIndices.push(indices[j]);\n for (let k = 0; k < 3; k++) {\n newPositions[indices[j] * 3 + k] = positions[indices[j] * 3 + k];\n }\n }\n }\n }\n newMesh = new Mesh('temp mesh', this.scene, mesh === null || mesh === void 0 ? void 0 : mesh.parent);\n const newVertexData = new VertexData();\n newVertexData.indices = newIndices;\n newVertexData.positions = newPositions;\n newVertexData.applyToMesh(newMesh, true);\n newMesh.visibility = 0;\n }\n if (newMesh) {\n mesh = newMesh;\n }\n if (mesh && mesh instanceof AbstractMesh) {\n const boundingBox = mesh.getBoundingInfo().boundingBox;\n newMesh === null || newMesh === void 0 ? void 0 : newMesh.dispose();\n return {\n maximum: boundingBox.maximum,\n minimum: boundingBox.minimum,\n center: boundingBox.center,\n minWorld: boundingBox.minimumWorld,\n maxWorld: boundingBox.maximumWorld,\n };\n }\n }\n updateLightWarnings() {\n let totalSpotLights = 0;\n let totalPointLights = 0;\n let shadowCastingLights = 0;\n let tempWarning = '';\n this.lightsWarningMessages = [];\n this.scene.lights.forEach(light => {\n if (light.isEnabled()) {\n if (light instanceof SpotLight) {\n totalSpotLights += 1;\n }\n else if (light instanceof PointLight) {\n totalPointLights += 1;\n }\n if (light.shadowEnabled) {\n shadowCastingLights += 1;\n }\n }\n });\n if (totalSpotLights + totalPointLights > 10) {\n tempWarning = `Performance Warning: The number of spotlights (${totalSpotLights}) and/or pointlights (${totalPointLights}) is higher than recommended for optimal performance.`;\n this.lightsWarningMessages.push(tempWarning);\n if (this.isDevEnvironment) {\n console.warn(tempWarning);\n }\n }\n if (shadowCastingLights > 4) {\n tempWarning = `Performance Warning: The number of shadow casting lights (${shadowCastingLights}) is higher than recommended for optimal performance.`;\n this.lightsWarningMessages.push(tempWarning);\n if (this.isDevEnvironment) {\n console.warn(tempWarning);\n }\n }\n this.updateWarnings();\n }\n updateTextureWarnings() {\n const bigTextures2048 = new Map();\n const bigTextures4096 = new Map();\n let tempWarning = '';\n this.textureWarningMessages = [];\n this.sceneNode.meshes.forEach(node => {\n const mesh = this.getRendererNode(node);\n let textures = [];\n if (mesh instanceof AbstractMesh && mesh.isEnabled()) {\n if (mesh.material instanceof MultiMaterial) {\n mesh.material.subMaterials.forEach(subMaterial => {\n const subMaterialTextures = subMaterial === null || subMaterial === void 0 ? void 0 : subMaterial.getActiveTextures();\n if (subMaterialTextures) {\n subMaterialTextures.forEach(texture => {\n if (texture instanceof Texture) {\n textures.push(texture);\n }\n });\n }\n });\n }\n else if (mesh.material instanceof Material) {\n textures = mesh.material.getActiveTextures();\n }\n }\n if (textures) {\n textures.forEach(texture => {\n const textureSize = Math.max(texture.getSize().height, texture.getSize().width);\n if (texture instanceof Texture) {\n if (textureSize > 4096) {\n bigTextures4096.set(texture.uniqueId, texture);\n bigTextures2048.set(texture.uniqueId, texture);\n }\n else if (textureSize > 2048) {\n bigTextures2048.set(texture.uniqueId, texture);\n }\n }\n });\n }\n });\n if (bigTextures4096.size > 4) {\n tempWarning = `Warning, number of textures with resolutions over 4096 (${bigTextures4096.size}) is higher than recommended for optimal performance.`;\n this.textureWarningMessages.push(tempWarning);\n if (this.isDevEnvironment) {\n console.warn(tempWarning);\n }\n }\n else if (bigTextures2048.size > 20) {\n tempWarning = `Warning, number of textures with resolutions over 2048 (${bigTextures2048.size}) is higher than recommended for optimal performance.`;\n this.textureWarningMessages.push(tempWarning);\n if (this.isDevEnvironment) {\n console.warn(tempWarning);\n }\n }\n this.updateWarnings();\n }\n updateMeshWarnings() {\n this.totalPolygons = this.recursiveGetPolygons(this.sceneNode).polygons;\n let meshCastAndReceiveShadows = 0;\n let tempWarning = '';\n this.meshWarningMessages = [];\n this.MeshWarningTracker.forEach(mesh => {\n if (mesh.isEnabled) {\n if (mesh.featureNumber > 15) {\n tempWarning = `Performance Warning: The number of features on ${mesh.name} is higher than recommended for optimal performance.`;\n this.meshWarningMessages.push(tempWarning);\n if (this.isDevEnvironment) {\n console.warn(tempWarning);\n }\n }\n if (mesh.castShadows && mesh.receiveShadows) {\n meshCastAndReceiveShadows += 1;\n }\n }\n });\n if (this.totalPolygons > 500000) {\n tempWarning = `Performance Warning: Polygon count of ${this.totalPolygons} is higher than recommended for optimal performance.`;\n this.meshWarningMessages.push(tempWarning);\n if (this.isDevEnvironment) {\n console.warn(tempWarning);\n }\n }\n if (meshCastAndReceiveShadows > 10) {\n tempWarning = `Performance Warning: The number of meshes that cast and receive shadows (${meshCastAndReceiveShadows}) is higher than recommended for optimal performance.`;\n this.meshWarningMessages.push(tempWarning);\n if (this.isDevEnvironment) {\n console.warn(tempWarning);\n }\n }\n this.updateWarnings();\n }\n updateNormalSmoothingWarnings(feature, polygonCount, removed) {\n var _a;\n if (removed) {\n this.normalSmoothingWarnings.delete(feature.id);\n }\n else {\n if (polygonCount > 100000 && feature.enabled) {\n const warning = `Performance Warning: Too many polygons (${polygonCount}) are selected inside \"${feature.name}\" on \"${(_a = feature._parent) === null || _a === void 0 ? void 0 : _a.name}\".`;\n this.normalSmoothingWarnings.set(feature.id, warning);\n if (this.isDevEnvironment) {\n console.warn(warning);\n }\n }\n else {\n this.normalSmoothingWarnings.delete(feature.id);\n }\n }\n this.updateWarnings;\n }\n updateMaterialWarnings() {\n let tempWarning = '';\n const localMaterialCount = this.getLocalMaterialsRecursive(this.sceneNode);\n this.materialWarningMessages = [];\n if (localMaterialCount > 20) {\n tempWarning = `Performance Warning: The number of local materials (${localMaterialCount}) is higher than recommended for optimal performance.`;\n this.materialWarningMessages.push(tempWarning);\n if (this.isDevEnvironment) {\n console.warn(tempWarning);\n }\n }\n }\n updateAnimationWarnings(activeAnimations) {\n let tempWarning = '';\n if (activeAnimations > 1) {\n tempWarning = `Performance Warning: More than 1 animation (${activeAnimations}) is active at the same instance.`;\n if (this.isDevEnvironment) {\n console.warn(tempWarning);\n }\n }\n }\n updateWarnings() {\n const tempArray = [];\n if (this.lightsWarningMessages && this.lightsWarningMessages.length > 0) {\n this.lightsWarningMessages.forEach(warning => {\n tempArray.push(warning);\n });\n }\n if (this.textureWarningMessages && this.textureWarningMessages.length > 0) {\n this.textureWarningMessages.forEach(warning => {\n tempArray.push(warning);\n });\n }\n if (this.meshWarningMessages && this.meshWarningMessages.length > 0) {\n this.meshWarningMessages.forEach(warning => {\n tempArray.push(warning);\n });\n }\n if (this.materialWarningMessages && this.materialWarningMessages.length > 0) {\n this.materialWarningMessages.forEach(warning => {\n tempArray.push(warning);\n });\n }\n if (this.normalSmoothingWarnings && this.normalSmoothingWarnings.size > 0) {\n this.normalSmoothingWarnings.forEach(warning => {\n tempArray.push(warning);\n });\n }\n if (tempArray.length === 0) {\n tempArray.push('No warnings to display.');\n }\n this.warningMessages = tempArray;\n this.warningTracker.next();\n }\n extractMeshGeometry(node) {\n const mesh = this.rendererManager.mesh.get3d(node);\n if (mesh && mesh instanceof Mesh && mesh.geometry) {\n const newGeometry = new exports.KbGeometry();\n newGeometry.indices = new Uint32Array(mesh.geometry.getIndices() || []);\n newGeometry.positions = new Float32Array(mesh.geometry.getVerticesData(VertexBuffer.PositionKind) || []);\n if (newGeometry.indices.length === 0 || newGeometry.positions.length === 0) {\n return null;\n }\n const normals = mesh.geometry.getVerticesData(VertexBuffer.NormalKind);\n if (normals) {\n newGeometry.normals = new Float32Array(normals);\n }\n const tangents = mesh.geometry.getVerticesData(VertexBuffer.TangentKind);\n if (tangents) {\n newGeometry.tangents = new Float32Array(tangents);\n }\n const uvs = mesh.geometry.getVerticesData(VertexBuffer.UVKind);\n if (uvs) {\n newGeometry.uvs = new Float32Array(uvs);\n }\n return newGeometry;\n }\n }\n applySvgToSketch(node, data) {\n return SvgConverter.overwriteSketchWithSvg(node, data);\n }\n draggingEvent(mousePosition) {\n let lastPickedMesh;\n const dragOver = new Subject();\n mousePosition.subscribe(position => {\n const pickInfo = this.interaction.pick(position.x, position.y);\n const mesh = pickInfo === null || pickInfo === void 0 ? void 0 : pickInfo.targetNode;\n if (lastPickedMesh !== mesh) {\n if (lastPickedMesh) {\n this.highlighter.remove(lastPickedMesh);\n }\n if (mesh) {\n this.highlighter.add(mesh, exports.KbColor.Green());\n }\n dragOver.next(mesh);\n lastPickedMesh = mesh;\n }\n }, () => { }, () => {\n if (lastPickedMesh) {\n this.highlighter.remove(lastPickedMesh);\n dragOver.complete();\n }\n });\n return dragOver.asObservable();\n }\n dispose() {\n this.disposed = true;\n this.positionTrackedNodes.clear();\n this.rendererManager.disposeAll();\n this.scene.dispose();\n this.engine.dispose();\n this.camera.dispose();\n this.manager.dispose();\n this.bbCache.dispose();\n this.sketchEditor.dispose();\n if (this.canvas) {\n this.container.removeChild(this.canvas);\n delete this.canvas;\n }\n if (this.uiOverlay) {\n this.container.removeChild(this.uiOverlay);\n delete this.uiOverlay;\n }\n if (this.resizeObserver) {\n this.resizeObserver.disconnect();\n delete this.resizeObserver;\n }\n this.fpsTracker.complete();\n this.polygonTracker.complete();\n this.warningTracker.complete();\n this.meshWarningsNeedsUpdate.complete();\n this.materialWarningsNeedsUpdate.complete();\n this.animationWarningsNeedsUpdate.complete();\n }\n getScreenPositionOfNode(node) {\n const bNode = this.rendererManager.mesh.get3d(node);\n if (bNode) {\n return nodeToScreen(bNode, this.scene.activeCamera);\n }\n }\n getScreenBoundingBoxOfNode(node) {\n const bNode = this.rendererManager.mesh.get3d(node);\n if (bNode) {\n return getScreenBoundingBox(bNode, this.scene.activeCamera, true);\n }\n }\n getBabylonJson() {\n return JSON.stringify(SceneSerializer.Serialize(this.scene), null, 2);\n }\n exportGlb() {\n var _a;\n return this.gltfExporter.exportGlb((_a = this.sceneNode) === null || _a === void 0 ? void 0 : _a.name);\n }\n exportObj(meshNode) {\n var _a;\n const fileName = meshNode ? meshNode.name : (_a = this.sceneNode) === null || _a === void 0 ? void 0 : _a.name;\n return this.objExporter.exportObj(fileName, meshNode);\n }\n //returns an object containing the polygons and vertices withing the selected node\n recursiveGetPolygons(node) {\n if (node.visible == false) {\n return { polygons: 0, vertices: 0 };\n }\n if (node instanceof exports.MeshNode) {\n return { polygons: node._polygonCount || 0, vertices: node._verticesCount || 0 };\n }\n const result = { polygons: 0, vertices: 0 };\n const children = node.getChildren();\n children.forEach(child => {\n if (child instanceof exports.MeshNode || child instanceof exports.MeshGroupNode) {\n const childData = this.recursiveGetPolygons(child);\n result.polygons += childData.polygons;\n result.vertices += childData.vertices;\n }\n });\n return result;\n }\n //call with null\n onNextFrame(func) {\n this.nextFrameCallbacks.push(func);\n }\n userIsInteracting() {\n return this.interaction.pointerIsDown || this.gizmos.dragging;\n }\n /** runs the kb3d renderers. Called before the babylon render */\n preRender() {\n // If this kb viewer instance is disposed, exit the render loop\n if (this.disposed) {\n return;\n }\n this.bbCache.update(); //update watchers of the bbcache of everything that changed since the last frame\n this.animation.update(); //update all animations. This can result in changes, so should be called before the check for changes\n if (!this.rendererManager.isStable()) {\n /** nodes must be processed in a specific order. For example, materials must be processed before meshes,\n * or a new material wouldn't exist yet in babylon when the mesh renderer tries to apply it.\n */\n const hasChanges = SyncService.hasChanges;\n this.rendererManager.render();\n this.gizmos.updateGizmoPositions();\n //if there were changes, force a babylon render asap to keep responsive\n if (hasChanges) {\n this.forceRenderOnNextFrame = true;\n }\n this.viewChanged.next();\n while (this.nextFrameCallbacks.length > 0) {\n this.nextFrameCallbacks.pop()();\n }\n nextTick();\n }\n }\n startRender() {\n if (!this._renderLoopRunning) {\n this._renderLoopRunning = true;\n let frameCount = 0;\n // Run the render loop.\n this.engine.runRenderLoop(() => {\n this.preRender();\n let skipped = true;\n if (!this.pauseRender) {\n frameCount++;\n if (this.forceRunRenderer ||\n this.forceRenderOnNextFrame ||\n this.userIsInteracting() ||\n this.camera.cameraMoving ||\n this.animation.animationsAreRunning ||\n this.ar.xrActive) {\n this.scene.render();\n this.forceRenderOnNextFrame = false;\n this.viewChanged.next();\n skipped = false;\n }\n else if (frameCount > 10) {\n this.scene.render();\n skipped = false;\n }\n if (frameCount > 10) {\n frameCount = 0;\n this.fpsTracker.next(this.getFPS());\n }\n this.afterRender.next({ skipped });\n }\n });\n }\n }\n stopRender() {\n this.engine.stopRenderLoop();\n this._renderLoopRunning = false;\n }\n createCanvas() {\n const canvas = document.createElement('canvas');\n canvas.style.width = '100%';\n canvas.style.height = '100%';\n canvas.style.outline = 'none';\n canvas.style.opacity = '0';\n canvas.style.transition = `opacity ${FADE_TIME}ms linear`;\n this.container.appendChild(canvas);\n if (!this.container.style.position) {\n this.container.style.position = 'relative';\n }\n this.uiOverlay = this.createOverlay();\n this.container.appendChild(this.uiOverlay);\n this.resizeObserver = new ResizeObserver(() => this.resize());\n this.resizeObserver.observe(canvas);\n return canvas;\n }\n createOverlay() {\n const overlay = document.createElement('div');\n overlay.style.position = 'absolute';\n overlay.style.top = '0';\n overlay.style.left = '0';\n overlay.style.width = '100%';\n overlay.style.height = '100%';\n overlay.style.pointerEvents = 'none';\n return overlay;\n }\n createLoadingExperience(loadingExperience, loadingImagePath) {\n if (loadingExperience == exports.eLoadingExperience.image) {\n const matService = this.manager.getService(Token.MaterialService);\n this.loadingOverlay = this.createOverlay();\n this.loadingOverlay.style.opacity = '0';\n this.loadingOverlay.style.transition = `opacity ${FADE_TIME}ms linear`;\n const img = document.createElement('img');\n img.style.position = 'absolute';\n img.style.width = '100%';\n img.style.height = '100%';\n img.style.backgroundSize = 'auto 100%';\n img.style.backgroundRepeat = 'no-repeat';\n img.style.backgroundPosition = 'center';\n img.style.backgroundColor = 'transparent';\n img.style.backgroundImage = `url('${matService.getAbsoluteTexturePath(loadingImagePath)}')`;\n this.loadingOverlay.appendChild(img);\n this.container.appendChild(this.loadingOverlay);\n this.loadingOverlay.style.opacity = '1';\n }\n else if (loadingExperience == exports.eLoadingExperience.fade) ;\n else if (loadingExperience == exports.eLoadingExperience.spinner) {\n this.loadingOverlay = this.createOverlay();\n this.loadingOverlay.style.opacity = '0';\n this.loadingOverlay.style.transition = `opacity ${FADE_TIME}ms linear`;\n this.loadingOverlay.style.display = 'flex';\n this.loadingOverlay.style.alignItems = 'center';\n this.loadingOverlay.style.justifyContent = 'center';\n this.loadingOverlay.innerHTML = `\r\n \r\n \r\n \r\n \r\n `;\n this.container.appendChild(this.loadingOverlay);\n this.loadingOverlay.style.opacity = '1';\n }\n else {\n this.canvas.style.opacity = '1';\n }\n }\n hideLoadingExperience() {\n if (this.loadingOverlay) {\n this.loadingOverlay.style.opacity = '0';\n setTimeout(() => {\n var _a;\n (_a = this.loadingOverlay) === null || _a === void 0 ? void 0 : _a.remove();\n this.loadingOverlay = undefined;\n }, FADE_TIME);\n }\n this.canvas.style.opacity = '1';\n }\n resize() {\n if (!this.ar || (!this.ar.xrLoading && !this.ar.xrActive)) {\n //don't mess with the AR canvas resize\n this.engine.resize();\n this.fireViewChangedOnNextRender();\n }\n }\n optimizeScene(scene) {\n if (this.sceneNode.enableSceneOptimization) {\n this.forceRunRenderer = true; // Run renderer until optimizer is done optimizing\n const sceneOptimizerOptions = new SceneOptimizerOptions(30, 2000);\n let priority = 0;\n sceneOptimizerOptions.addCustomOptimization(function () {\n console.warn('Low scene performance detected - disabling 3d features to improve performance.');\n return true;\n }, function () {\n return 'custom optimization to throw console warning when optimizer needs to run.';\n }, priority);\n sceneOptimizerOptions.optimizations.push(new ShadowsOptimization(priority));\n sceneOptimizerOptions.optimizations.push(new LensFlaresOptimization(priority));\n // Next priority\n priority++;\n sceneOptimizerOptions.optimizations.push(new PostProcessesOptimization(priority));\n sceneOptimizerOptions.optimizations.push(new ParticlesOptimization(priority));\n priority++;\n sceneOptimizerOptions.optimizations.push(new TextureOptimization(priority, 256));\n priority++;\n sceneOptimizerOptions.optimizations.push(new RenderTargetsOptimization(priority));\n // Don't scale the resolution when in editor mode because it affects usability\n if (!this.options.editorMode) {\n priority++;\n sceneOptimizerOptions.optimizations.push(new HardwareScalingOptimization(priority, 4));\n }\n const sceneOptimizer = new SceneOptimizer(scene, sceneOptimizerOptions, false);\n sceneOptimizer.onSuccessObservable.add(() => {\n this.startSceneRenderAutoStop(scene, this.engine);\n // Run optimizer in improvement mode in case the scene speeds up later\n new SceneOptimizer(scene, sceneOptimizerOptions, false, true);\n });\n sceneOptimizer.onFailureObservable.add(() => this.startSceneRenderAutoStop(scene, this.engine));\n sceneOptimizer.start();\n }\n }\n createScene() {\n // Create a basic BJS Scene object.\n const scene = new Scene(this.engine);\n scene.useRightHandedSystem = true;\n scene.clearColor = new Color4(0, 0, 0, 0); //default background to transparent\n // Create a FreeCamera, and set its position to (x:0, y:5, z:-10).\n //this.m_camera = new FreeCamera('camera1', new Vector3(0, 5, -10), this.m_scene);\n //scene.createDefaultCameraOrLight(true, true, true);\n this.createDefaultEnv();\n return scene;\n }\n startSceneRenderAutoStop(scene, engine) {\n this.forceRunRenderer = false;\n }\n createDefaultEnv() {\n if (!this.scene) {\n return;\n }\n const helper = this.scene.createDefaultEnvironment({\n skyboxSize: 500,\n groundShadowLevel: 0.5,\n });\n if (helper !== null && helper.skybox !== null && helper.ground !== null) {\n helper.setMainColor(Color3.Gray());\n helper.rootMesh.position.y = -5;\n helper.skybox.isPickable = false;\n helper.rootMesh.isPickable = false;\n helper.ground.isPickable = false;\n //helper.rootMesh.name = KbViewer.KBMAXPREFIX + helper.rootMesh.name;\n //helper.skybox.name = KbViewer.KBMAXPREFIX + helper.skybox.name;\n for (let i = 0; i < this.scene.meshes.length; i++) {\n this.scene.meshes[i].name = KBMAXPREFIX + this.scene.meshes[i].name;\n this.scene.meshes[i].id = KBMAXPREFIX + this.scene.meshes[i].id;\n }\n }\n }\n pauseRenderWhile(action) {\n this.pauseRender = true;\n action();\n this.pauseRender = false;\n }\n /**\n * returns a promise that is resolved when the following has happened:\n * * the scene graph is fully loaded\n * * at least one pass through the render loop since the scene has been loaded has completed\n * * any server assets in flight (meshes, textures) have been received and processed\n */\n whenAssetsLoaded(waitForMaterials = true) {\n if (!this.sceneNode) {\n throw Error('a scene needs to be loaded into the viewer to wait for assets');\n }\n const promise = new Promise((resolve, reject) => {\n var _a, _b;\n const http = (_a = this.sceneNode._manager) === null || _a === void 0 ? void 0 : _a.getService(Token.HttpService);\n const materialService = (_b = this.sceneNode._manager) === null || _b === void 0 ? void 0 : _b.getService(Token.MaterialService);\n if (!http) {\n throw Error('HTTP API service could not be initialized');\n }\n if (!materialService) {\n throw Error('Material service could not be initialized');\n }\n this.forceAndWaitForNextRender().then(() => {\n const promises = [\n ...http.trackedAssetCalls,\n ...this.additionalLoadBlockers,\n this.scene.whenReadyAsync(),\n ];\n if (waitForMaterials) {\n promises.push(...materialService.trackedAssetCalls);\n }\n Promise.all(promises).finally(() => {\n // now wait for a render loop and make sure no new loading blockers were added since the last batch\n this.forceAndWaitForNextRender().then(() => {\n if (http.trackedAssetCalls.length ||\n this.additionalLoadBlockers.length ||\n (waitForMaterials && materialService.trackedAssetCalls.length) ||\n !this.scene.isReady() ||\n !this.rendererManager.isStable()) {\n return this.whenAssetsLoaded(waitForMaterials).finally(() => {\n resolve();\n });\n }\n else {\n resolve();\n }\n });\n });\n });\n });\n return promise;\n }\n deferSceneReady(promise) {\n promise.then(() => {\n this.additionalLoadBlockers = this.additionalLoadBlockers.filter(somePromise => somePromise !== promise);\n });\n this.additionalLoadBlockers.push(promise);\n }\n /**\n * Returns a promise that resolves when the next kb3d render processing has happened.\n * Note that this does not necessarily mean that the babylon render has happened!\n * If you want to wait for both, see forceAndWaitForNextRender\n */\n waitForNextKb3dRender() {\n return this.afterRender.pipe(take(1)).toPromise();\n }\n forceAndWaitForNextRender() {\n this.forceRenderOnNextFrame = true;\n return this.waitForNextKb3dRender();\n }\n async waitForStableRender(timeout = 5000) {\n const start = performance.now();\n await this.forceAndWaitForNextRender();\n while (!this.rendererManager.isStable()) {\n await this.forceAndWaitForNextRender();\n if (performance.now() - start > timeout) {\n console.error(`waiting for stable render took longer than timeout of ${timeout}ms, unblocking loading`);\n return false;\n }\n }\n return true;\n }\n fireViewChangedOnNextRender() {\n this.forceAndWaitForNextRender().then(() => {\n this.viewChanged.next();\n });\n }\n getActiveMates() {\n const mates = new Array();\n this.sceneNode.runOnAllScenes(s => {\n mates.push(...s._manager.getByToken(Token.MateNode).filter(m => m.isActive()));\n });\n return mates;\n }\n nodesIntersect(node1, node2) {\n const bb1 = this.bbCache.getWorld(node1);\n const bb2 = this.bbCache.getWorld(node2);\n if (bb1 && bb2) {\n return bb1.boundingInfo.intersects(bb2.boundingInfo, false);\n }\n return false;\n }\n getReferenceNodes(dynamicId) {\n var _a;\n const result = [];\n const references = this.rendererManager.referenceProcessor.getReferencesTo();\n if (references.has(dynamicId)) {\n const ids = (_a = references.get(dynamicId)) === null || _a === void 0 ? void 0 : _a.keys();\n if (ids) {\n for (const key of ids) {\n result.push(this.manager.getById(key));\n }\n }\n return result;\n }\n else {\n return [];\n }\n }\n getLocalMaterialsRecursive(node) {\n let result = 0;\n if (!node.scenes) {\n return node.materials.length | 0;\n }\n else {\n node.scenes.forEach(scene => {\n result += this.getLocalMaterialsRecursive(scene);\n });\n result += node.materials.length;\n }\n return result;\n }\n getTransformSpaceDifference(originNode, destinationNode) {\n const originMesh = this.getRendererNode(originNode);\n const destinationMesh = this.getRendererNode(destinationNode);\n const result = {\n scale: Vector3.One(),\n rotationQuaternion: new Quaternion(),\n translation: Vector3.Zero(),\n };\n if (!destinationMesh) {\n const resultMatrix = originMesh.getWorldMatrix();\n resultMatrix.decompose(result.scale, result.rotationQuaternion, result.translation);\n }\n else if (originMesh && destinationMesh) {\n const originMatrix = originMesh.getWorldMatrix();\n const destinationMatrix = destinationMesh.getWorldMatrix().clone();\n destinationMatrix.invert();\n const resultMatrix = originMatrix.multiply(destinationMatrix);\n resultMatrix.decompose(result.scale, result.rotationQuaternion, result.translation);\n }\n else {\n return null;\n }\n return {\n scale: result.scale.x,\n rotationQuaternion: exports.KbQuaternion.FromQuat(result.rotationQuaternion),\n translation: exports.KbVector.FromVec3(result.translation),\n };\n }\n centerMeshToOrigin(node) {\n const boundingBox = this.getMeshBoundingBox(node);\n const center = boundingBox === null || boundingBox === void 0 ? void 0 : boundingBox.maxWorld.add(boundingBox.minWorld);\n if (center) {\n center === null || center === void 0 ? void 0 : center.scaleInPlace(-0.5);\n const result = new exports.KbVector(center.x, center.y, center.z);\n return result;\n }\n else {\n return new exports.KbVector(0, 0, 0);\n }\n }\n }\n\n function kb3dTest() {\n let map = new SetMap();\n map.add('A', 'B', 'C');\n map.add('B', 'C', 'D');\n map.add('C', 'E');\n map.add('D', 'E', 'F');\n map.add('E');\n map.add('F');\n map.add('G', 'F', 'E');\n topologicalSort(map, s => s);\n map = new SetMap();\n map.add('0', '1', '2');\n map.add('1', '2');\n map.add('2', '0', '3');\n map.add('3');\n topologicalSort(map, s => s);\n map = new SetMap();\n map.add('2');\n map.add('3', '8', '10');\n map.add('5', '11');\n map.add('7', '8', '11');\n map.add('8', '9');\n map.add('9');\n map.add('10');\n map.add('11', '2', '9', '10');\n topologicalSort(map, s => s);\n const kbViewer = new KbViewer({ container: document.getElementById('viewer-div') });\n let manager = new Kb3dManager({ assetRoot: '' }); //default services automatically used\n let sceneNode = manager.create(Token.SceneNode);\n sceneNode.backgroundColor = new exports.KbColor(0.9, 0.9, 0.9, 1);\n sceneNode.enableHoverEffects = true;\n kbViewer.loadScene({ sceneNode });\n const group = manager.create(Token.MeshGroupNode);\n const box = manager.create(Token.BoxNode);\n const boxId = box.id;\n group.meshes.push(box);\n // let blah = manager.getById(boxId);\n // let db = manager.getAll().map(n => { return { id: n.id, name: n.name, type: n.$type }; });\n // alert(JSON.stringify(db, null, 2));\n // sceneNode.meshes.push(group);\n // db = manager.getAll().map(n => { return { id: n.id, name: n.name, type: n.$type }; });\n // alert(JSON.stringify(db, null, 2));\n // setTimeout(() => {\n // sceneNode.meshes.remove(group);\n // db = manager.getAll().map(n => { return { id: n.id, name: n.name, type: n.$type }; });\n // alert(JSON.stringify(db, null, 2));\n // }, 500);\n //sceneNode.meshes.push(box);\n // function animate(time: number) {\n // requestAnimationFrame(animate);\n // TWEEN.update(time);\n // }\n // requestAnimationFrame(animate);\n // let pos = { x: box.position.x, y: box.position.y, z: box.position.z };\n // const anim = new TWEEN.Tween(pos)\n // .to({ x: 3, y: 0, z: 0 }, 5000)\n // .onUpdate(() => {\n // box.position = KbVector.From(pos);\n // })\n // .start();\n // let group = manager.create(Token.MeshGroupNode);\n // group.rotation = new KbVector(0, 0, 0);\n // sceneNode.meshes.push(group);\n // let box = manager.create(Token.BoxNode);\n // let boxId = box.id;\n // box.materialId = \"bricks01\";\n // group.meshes.push(box);\n // let conn1 = manager.create(Token.Connector, {\n // positionMode: eConnectorPositionMode.boundingBox,\n // position: new KbVector(0, 1, 1),\n // direction: new KbVector(0, 0, 1)\n // });\n // box.features.push(conn1);\n // let mat = manager.create(Token.AdvancedMaterialNode, {\n // albedoColor: new KbColor(1, 0, 0),\n // backFaceCulling: false\n // });\n // sceneNode.materials.push(mat);\n // box.materialId = mat.id;\n // let sphere = manager.create(Token.SphereNode, {\n // diameter: 1,\n // position: new KbVector(2, 0,0)\n // });\n // group.meshes.push(sphere);\n // let box2 = manager.create(Token.BoxNode);\n // sceneNode.meshes.push(box2);\n // box2.position = new KbVector(0, 0, 3);\n // box2.width = 2;\n // let conn3 = manager.create(Token.Connector, {\n // positionMode: eConnectorPositionMode.absolute,\n // position: new KbVector(0, 0, -0.5),\n // direction: new KbVector(0, 0, -1),\n // angle: 0\n // });\n // let conn4 = manager.create(Token.Connector, {\n // positionMode: eConnectorPositionMode.absolute,\n // position: new KbVector(1, 0, 0),\n // direction: new KbVector(1, 0, 0),\n // angle: 30\n // });\n // let conn5 = manager.create(Token.Connector, {\n // positionMode: eConnectorPositionMode.absolute,\n // position: new KbVector(0, .5, 0),\n // direction: new KbVector(0, 1, 0),\n // angle: 0\n // });\n // box2.features.push(conn3);\n // box2.features.push(conn4);\n // box2.features.push(conn5);\n // manager.draggables.add(group.id, \"blah\", {\n // mode: eDragMode.free,\n // //surfaceNodes: [sphere.id, sphere2.id],\n // planeNormal: new KbVector(0, 1, 0),\n // // planeOrigin: new KbVector(0,0,0),\n // //disableOffset: false,\n // connectionMode: eConnectionMode.screen,\n // showConnectors: true,\n // connectRadius: 10,\n // dragOpacity: .3,\n // dragStart: (a) => {\n // },\n // dragging: (a) => {\n // },\n // dropped: (a) => {\n // },\n // // canConnect: (a) => {\n // // a.\n // // }\n // });\n // kbViewer.interaction.onPointerUp.subscribe(p => {\n // console.log(kbViewer.getScreenPositionOfNode(box));\n // console.log(kbViewer.getScreenBoundingBoxOfNode(box));\n // });\n const light = manager.create(Token.DirectionalLightNode);\n sceneNode.lights.push(light);\n // let mat = manager.create(tok.AdvancedMaterialNode, {\n // albedoColor: new KbColor(252, 0, 0)\n // });\n // sceneNode.materials.push(mat);\n // box.materialId = mat.id;\n //box.materialId = \"4454a0e3-e1e4-42d4-aea5-6915681bfc12\";\n // for (let row = 0; row < 10; row++){\n // for (let col = 0; col < 10; col++) {\n // let nested = manager.nestScene(sceneNode);\n // nested.name = \"nested\" + row;\n // let nestedManager = nested._manager as Kb3dManager;\n // let nestedBox = nestedManager.create(tok.BoxNode);\n // nestedBox.name = \"nestedBox\" + row;\n // if (row % 2 > 0) {\n // nestedBox.materialId = \"rock01\";\n // } else {\n // nestedBox.materialId = \"bricks01\";\n // }\n // nested.meshes.push(nestedBox);\n // nested.position = new KbVector(row * 1.5, col * 1.5, 0);\n // }\n // }\n kbViewer.interaction.onClick.subscribe(p => {\n console.log(p);\n });\n kbViewer.interaction.onKeyboard.subscribe(k => {\n console.log(k);\n });\n const btnAnimate = document.getElementById('btnAnimate');\n if (btnAnimate) {\n btnAnimate.onclick = () => {\n kbViewer.animation\n .queue({\n name: 'a1',\n node: box,\n prop: 'position',\n toValue: new exports.KbVector(10, 0, 0),\n duration: 1000,\n easing: exports.eEasing.quadraticIn,\n })\n .then(() => {\n return kbViewer.animation.queue({\n name: 'a2',\n node: box,\n prop: 'rotation',\n toValue: new exports.KbVector(box.eulerRotation.x + 720, 0, 0),\n duration: 1000,\n });\n })\n .then(() => {\n return kbViewer.animation.queue({\n name: 'a3',\n node: box,\n prop: 'position',\n toValue: new exports.KbVector(0, 0, 0),\n duration: 1000,\n easing: exports.eEasing.quadraticOut,\n });\n });\n };\n }\n const btnStop = document.getElementById('btnStop');\n if (btnStop) {\n btnStop.onclick = () => {\n kbViewer.animation.stop('a1');\n };\n }\n const txtJson = document.getElementById('txtJson');\n const txtCounter = document.getElementById('txtCounter');\n const btnToJson = document.getElementById('btnToJson');\n if (btnToJson) {\n btnToJson.onclick = () => {\n txtJson.value = manager.serializeToJson(sceneNode, { prettyPrint: true });\n updateJsonLength();\n };\n }\n const btnToFullJson = document.getElementById('btnToFullJson');\n if (btnToFullJson) {\n btnToFullJson.onclick = () => {\n txtJson.value = manager.serializeToJson(sceneNode, { omitDefaultValues: false, prettyPrint: true });\n updateJsonLength();\n };\n }\n const btnFromJson = document.getElementById('btnFromJson');\n if (btnFromJson) {\n btnFromJson.onclick = () => {\n manager = new Kb3dManager();\n const jsonObject = JSON.parse(txtJson.value);\n sceneNode = manager.create(Token.SceneNode, jsonObject);\n kbViewer.loadScene({ sceneNode });\n };\n }\n const btnBabylonJson = document.getElementById('btnBabylonJson');\n if (btnBabylonJson) {\n btnBabylonJson.onclick = () => {\n txtJson.value = kbViewer.getBabylonJson();\n };\n }\n const btnWidth = document.getElementById('btnWidth');\n if (btnWidth) {\n btnWidth.onclick = () => {\n box.depth = box.depth + 1;\n };\n }\n const btnMaterialColor = document.getElementById('btnMaterialColor');\n if (btnMaterialColor) {\n btnMaterialColor.onclick = () => {\n //mat.albedoColor = new KbColor(0, 0, 252);\n };\n }\n // let btnToggleVisible = document.getElementById(\"btnToggleVisible\");\n // if (btnToggleVisible) {\n // btnToggleVisible.onclick = () => {\n // box.visible = !box.visible;\n // };\n // }\n const btnSaveMesh = document.getElementById('btnSaveMesh');\n if (btnSaveMesh) {\n btnSaveMesh.onclick = () => {\n const myArray = new ArrayBuffer(512);\n const longInt8View = new Uint8Array(myArray);\n // generate some data\n for (let i = 0; i < longInt8View.length; i++) {\n longInt8View[i] = i % 256;\n }\n const xhr = new XMLHttpRequest();\n xhr.open('POST', '/api/meshes/72b7487f-2ab2-4faf-b034-a1178feeb191', false);\n xhr.send(myArray);\n };\n }\n const btnGetSavedMesh = document.getElementById('btnGetSavedMesh');\n if (btnGetSavedMesh) {\n btnGetSavedMesh.onclick = () => {\n const oReq = new XMLHttpRequest();\n oReq.open('GET', '/api/meshes/72b7487f-2ab2-4faf-b034-a1178feeb191', true);\n oReq.responseType = 'arraybuffer';\n oReq.onload = function (oEvent) {\n const arrayBuffer = oReq.response; // Note: not oReq.responseText\n if (arrayBuffer) {\n const byteArray = new Uint8Array(arrayBuffer);\n for (let i = 0; i < byteArray.byteLength; i++) {\n // do something with each byte in the array\n console.log(byteArray[i]);\n }\n }\n };\n oReq.send(null);\n };\n }\n const updateJsonLength = () => {\n txtCounter.innerHTML = new TextEncoder().encode(txtJson.value).length.toString() + ' bytes';\n };\n }\n\n const kb3dEnums = _kb3dEnums;\n\n exports.ABORTED = ABORTED;\n exports.ALL_CHANGES = ALL_CHANGES;\n exports.ALL_GEOMETRY = ALL_GEOMETRY;\n exports.AR = AR;\n exports.AUTO_MESH_NAME = AUTO_MESH_NAME;\n exports.AnimationModule = AnimationModule;\n exports.AnnotationRenderer = AnnotationRenderer;\n exports.ArrayMap = ArrayMap;\n exports.BaseKb3dModel = BaseKb3dModel;\n exports.Boundary = Boundary;\n exports.BoundingBoxCache = BoundingBoxCache;\n exports.BoundingBoxRelativeHelper = BoundingBoxRelativeHelperKbVector;\n exports.CONNECTOR_DIAM = CONNECTOR_DIAM;\n exports.CameraTools = CameraTools;\n exports.ChangeMap = ChangeMap;\n exports.ColorFillTextureLayerRenderer = ColorFillTextureLayerRenderer;\n exports.ConnectorRenderer = ConnectorRenderer;\n exports.DimensionRenderer = DimensionRenderer;\n exports.Dragger = Dragger;\n exports.Enumerable = Enumerable;\n exports.FileTextureLayerRenderer = FileTextureLayerRenderer;\n exports.FontService = FontService;\n exports.GLTFKb3dExporter = GLTFKb3dExporter;\n exports.GizmoTools = GizmoTools;\n exports.GuiHelper = GuiHelper;\n exports.HandlerMap = HandlerMap;\n exports.HttpService = HttpService;\n exports.Ignore = Ignore;\n exports.IgnoreChanges = IgnoreChanges;\n exports.IgnoreHistory = IgnoreHistory;\n exports.Inject = Inject;\n exports.Interaction = Interaction;\n exports.KBMAXPREFIX = KBMAXPREFIX;\n exports.Kb3dManager = Kb3dManager;\n exports.Kb3dModel = Kb3dModel;\n exports.Kb3dModels = Kb3dModels;\n exports.Kb3dService = Kb3dService;\n exports.Kb3dWorker = Kb3dWorker;\n exports.KbAxisDragGizmo = KbAxisDragGizmo;\n exports.KbAxisScaleGizmo = KbAxisScaleGizmo;\n exports.KbGizmo = KbGizmo;\n exports.KbLightGizmo = KbLightGizmo;\n exports.KbPlaneRotationGizmo = KbPlaneRotationGizmo;\n exports.KbPositionGizmo = KbPositionGizmo;\n exports.KbRotationGizmo = KbRotationGizmo;\n exports.KbScaleGizmo = KbScaleGizmo;\n exports.KbVertexSelectGizmo = KbVertexSelectGizmo;\n exports.KbViewer = KbViewer;\n exports.KbViewpointGizmo = KbViewpointGizmo;\n exports.LOCAL_ENVIRONMENT_DOMAIN = LOCAL_ENVIRONMENT_DOMAIN;\n exports.LibraryEnvironments = LibraryEnvironments;\n exports.LibraryEnvironmentsMap = LibraryEnvironmentsMap;\n exports.LibraryMaterials = LibraryMaterials;\n exports.LibraryMaterialsByCategory = LibraryMaterialsByCategory;\n exports.LightRenderer = LightRenderer;\n exports.MateProcessor = MateProcessor;\n exports.MateRenderer = MateRenderer;\n exports.MaterialRenderer = MaterialRenderer;\n exports.MeshRenderer = MeshRenderer;\n exports.MeshService = MeshService;\n exports.MoveVerticesRenderer = MoveVerticesRenderer;\n exports.NEW_GEOMETRY = NEW_GEOMETRY;\n exports.NodeReference = NodeReference;\n exports.NormalSmoothingRenderer = NormalSmoothingRenderer;\n exports.ObjKb3dExporter = ObjKb3dExporter;\n exports.PRECISION = PRECISION;\n exports.PathHandler = PathHandler;\n exports.PathParser = PathParser;\n exports.PathRenderer = PathRenderer;\n exports.PatternFeature = PatternFeature;\n exports.PatternRenderer = PatternRenderer;\n exports.Processor = Processor;\n exports.Quad = Quad;\n exports.QuadTree = QuadTree;\n exports.RendererManager = RendererManager;\n exports.SKYBOX_SIZE = SKYBOX_SIZE;\n exports.SamplingModeMap = SamplingModeMap;\n exports.SceneRenderer = SceneRenderer;\n exports.SetMap = SetMap;\n exports.ShadowLightNode = ShadowLightNode;\n exports.SketchEditor = SketchEditor;\n exports.SnapshotModule = SnapshotModule;\n exports.SubmeshRenderer = SubmeshRenderer;\n exports.SvgConverter = SvgConverter;\n exports.SvgTextureLayerRenderer = SvgTextureLayerRenderer;\n exports.SyncService = SyncService;\n exports.TagArrayMap = TagArrayMap;\n exports.TextMeshFont = TextMeshFont;\n exports.TextMeshVertexData = TextMeshVertexData;\n exports.TextureLayerRenderer = TextureLayerRenderer;\n exports.Token = Token;\n exports.TrackChanges = TrackChanges;\n exports.TrackChangesNode = TrackChangesNode;\n exports.USDZConverter = USDZConverter;\n exports.UUID = UUID;\n exports.UVMapHelper = UVMapHelper;\n exports.UvMapRenderer = UvMapRenderer;\n exports.Validator = Validator;\n exports.ViewpointRenderer = ViewpointRenderer;\n exports.absoluteUrlRegex = absoluteUrlRegex;\n exports.addArrays = addArrays;\n exports.appendOrIncrementNumber = appendOrIncrementNumber;\n exports.applyMatrix4 = applyMatrix4;\n exports.applyTransformToSpace = applyTransformToSpace;\n exports.applyTransformToVector = applyTransformToVector;\n exports.axisToVector3 = axisToVector3;\n exports.bakeMeshTransform = bakeMeshTransform;\n exports.boundingVectorsToScreen = boundingVectorsToScreen;\n exports.buildHeader = buildHeader;\n exports.buildUSDAFile = buildUSDAFile;\n exports.buildXform = buildXform;\n exports.calculateFaceNormal = calculateFaceNormal;\n exports.calculateFaceNormalFromFloats = calculateFaceNormalFromFloats;\n exports.calculateTransform = calculateTransform;\n exports.changesProcessed = changesProcessed;\n exports.computeLocalTransform = computeLocalTransform;\n exports.computeNormal = computeNormal;\n exports.convertHdrToEnv = convertHdrToEnv;\n exports.createAnnotationTextModels = createAnnotationTextModels;\n exports.createLightVisualization = createLightVisualization;\n exports.createSketchPointForEditor = createSketchPointForEditor;\n exports.createUvMapVisualization = createUvMapVisualization;\n exports.createViewpointVisualization = createViewpointVisualization;\n exports.defaults = defaults;\n exports.degreeToRadian = degreeToRadian;\n exports.drawLine = drawLine;\n exports.escapeGUID = escapeGUID;\n exports.excludeVertices = excludeVertices;\n exports.fillSketchPolygon = fillSketchPolygon;\n exports.getByDynamicId = getByDynamicId;\n exports.getCameraPositionWithOffset = getCameraPositionWithOffset;\n exports.getConstructorParamsForModel = getConstructorParamsForModel;\n exports.getDimensionLinePoints = getDimensionLinePoints;\n exports.getHierarchyBoundingVectors = getHierarchyBoundingVectors;\n exports.getNodeChain = getNodeChain;\n exports.getNodeMap = getNodeMap;\n exports.getPositionFromConnnector = getPositionFromConnnector;\n exports.getRotationBtwVectors = getRotationBtwVectors;\n exports.getScreenBoundingBox = getScreenBoundingBox;\n exports.getSketchNormalVector3 = getSketchNormalVector3;\n exports.getToken = getToken;\n exports.guidRegex = guidRegex;\n exports.history = history;\n exports.injectParameters = injectParameters;\n exports.instanceOf = instanceOf;\n exports.instanceOfAny = instanceOfAny;\n exports.isAbsoluteUrl = isAbsoluteUrl;\n exports.isAnnotation = isAnnotation;\n exports.isBoxMesh = isBoxMesh;\n exports.isCircularPattern = isCircularPattern;\n exports.isConeMesh = isConeMesh;\n exports.isConnector = isConnector;\n exports.isCustomMeshNode = isCustomMeshNode;\n exports.isCylinderMesh = isCylinderMesh;\n exports.isDefined = isDefined;\n exports.isDiscMesh = isDiscMesh;\n exports.isFeatureNode = isFeatureNode;\n exports.isGuid = isGuid;\n exports.isInstanceOf = isInstanceOf;\n exports.isLightNode = isLightNode;\n exports.isLinearPattern = isLinearPattern;\n exports.isMate = isMate;\n exports.isMaterialNode = isMaterialNode;\n exports.isMeshGroupNode = isMeshGroupNode;\n exports.isMeshNode = isMeshNode;\n exports.isObject = isObject$1;\n exports.isPattern = isPattern;\n exports.isPickable = isPickable;\n exports.isPlaneMesh = isPlaneMesh;\n exports.isPrimitiveMeshNode = isPrimitiveMeshNode;\n exports.isSceneNode = isSceneNode;\n exports.isSketchControlPoint = isSketchControlPoint;\n exports.isSketchNode = isSketchNode;\n exports.isSpaceNode = isSpaceNode;\n exports.isSphereMesh = isSphereMesh;\n exports.isSubmesh = isSubmesh;\n exports.isTextNode = isTextNode;\n exports.isTextureLayer = isTextureLayer;\n exports.isTorusMesh = isTorusMesh;\n exports.isTrackedArray = isTrackedArray;\n exports.isTrackedClass = isTrackedClass;\n exports.isTrackedInstance = isTrackedInstance;\n exports.isTrackedInstanceOfAny = isTrackedInstanceOfAny;\n exports.isTrackedMeshNode = isTrackedMeshNode;\n exports.isViewpointNode = isViewpointNode;\n exports.kb3dEnums = kb3dEnums;\n exports.kb3dTest = kb3dTest;\n exports.makeRotationAxis = makeRotationAxis;\n exports.mathClamp = mathClamp;\n exports.mergeVertices = mergeVertices;\n exports.mergeVerticesIntoAll = mergeVerticesIntoAll;\n exports.multiplyScalar = multiplyScalar;\n exports.nextTick = nextTick;\n exports.nodeToScreen = nodeToScreen;\n exports.ofToken = ofToken;\n exports.omitUndefinedElements = omitUndefinedElements;\n exports.parseSketchGeometry = parseSketchGeometry;\n exports.projectOnNodeAxis = projectOnNodeAxis;\n exports.removeVerticesFromAll = removeVerticesFromAll;\n exports.rotationToVector = rotationToVector;\n exports.rotationToVectorToRef = rotationToVectorToRef;\n exports.roundToPlaces = roundToPlaces;\n exports.roundVectorToPlaces = roundVectorToPlaces;\n exports.sortMates = sortMates;\n exports.toBoundingBoxInfo = toBoundingBoxInfo;\n exports.toNDC = toNDC;\n exports.toPixels = toPixels;\n exports.toRadians = toRadians;\n exports.toScreen = toScreen;\n exports.transformVectorWithQuat = transformVectorWithQuat;\n exports.updateAnnotationText = updateAnnotationText;\n exports.updateMatrixOfAncestorChain = updateMatrixOfAncestorChain;\n\n return exports;\n\n}({}));\n//# sourceMappingURL=viewer.js.map\n","declare global {\r\n interface String {\r\n capitalize(): string;\r\n splitWords(): string[];\r\n isEqual(str: string | undefined): boolean;\r\n equalsAny(...strings: string[]): boolean;\r\n toSentence(): string;\r\n }\r\n interface Array {\r\n sortBy(predicate: (element: T) => any): void;\r\n /**\r\n * Conditionally removes items from an array and returns a new array with the removed items.\r\n */\r\n removeWhere(predicate: (item: T) => boolean): T[];\r\n\r\n remove(item: T): T[];\r\n\r\n selectMany(selector: (item: T) => TChild[]): TChild[];\r\n first(): T;\r\n last(): T;\r\n\r\n clear(): void;\r\n\r\n reorder(fromIndex: number, toIndex: number, undo?: boolean): void;\r\n\r\n contains(item: T): boolean;\r\n /**\r\n * Given a predicate to reduce each item in the array to a value,\r\n * will return that value if all items in the array have the same value.\r\n * If not, will return null.\r\n */\r\n getCommonOrNull(predicate: (item: T) => any): any;\r\n\r\n toMap(keySelector: (item: T) => TKey): Map;\r\n }\r\n interface Set {\r\n find(predicate: (item: T) => boolean): T | undefined;\r\n }\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n interface Map {\r\n hasAny(...keys: K[]): boolean;\r\n }\r\n}\r\n\r\nString.prototype.capitalize = function () {\r\n return this.charAt(0).toUpperCase() + this.slice(1);\r\n};\r\n\r\nconst splitWordsRegex = /[A-Z]{2,}(?=[A-Z][a-z]+[0-9]*|\\b)|[A-Z]?[a-z]+[0-9]*|[A-Z]|[0-9]+/g;\r\nString.prototype.splitWords = function () {\r\n return this.match(splitWordsRegex) || [];\r\n};\r\n\r\nString.prototype.isEqual = function (str: string) {\r\n return (this ? this.toUpperCase() : undefined) === (str ? str.toUpperCase() : undefined);\r\n};\r\n\r\nString.prototype.equalsAny = function (...strings: string[]) {\r\n return strings && strings.some(s => s.isEqual(this as string));\r\n};\r\n\r\nString.prototype.toSentence = function () {\r\n return this.replace(/([A-Z])/g, ' $1') // insert a space before all caps\r\n .replace(/^./, function (str) {\r\n return str.toUpperCase();\r\n }) // uppercase the first character\r\n .trim(); //trip leading and trainling whitespace\r\n};\r\n\r\nObject.defineProperty(Array.prototype, 'sortBy', {\r\n enumerable: false,\r\n writable: true,\r\n value: function (predicate: (item: any) => any) {\r\n return (this as any[]).sort((a, b) => {\r\n if (predicate(a) < predicate(b)) {\r\n return -1;\r\n } else if (predicate(a) > predicate(b)) {\r\n return 1;\r\n } else {\r\n return 0;\r\n }\r\n });\r\n },\r\n});\r\n\r\nObject.defineProperty(Array.prototype, 'remove', {\r\n enumerable: false,\r\n writable: true,\r\n value: function (item: any): any[] | undefined {\r\n const index = this.indexOf(item);\r\n if (index != -1) {\r\n return this.splice(this.indexOf(item), 1);\r\n }\r\n },\r\n});\r\n\r\nObject.defineProperty(Array.prototype, 'removeWhere', {\r\n enumerable: false,\r\n writable: true,\r\n value: function (predicate: (item: any) => boolean) {\r\n const removed = new Array();\r\n for (let i = this.length - 1; i >= 0; i--) {\r\n const item = this[i];\r\n if (predicate(item)) {\r\n removed.push(...this.splice(i, 1));\r\n }\r\n }\r\n return removed;\r\n },\r\n});\r\n\r\nObject.defineProperty(Array.prototype, 'contains', {\r\n enumerable: false,\r\n writable: true,\r\n value: function (item: any): boolean {\r\n const index = this.indexOf(item);\r\n return index != -1;\r\n },\r\n});\r\n\r\nObject.defineProperty(Array.prototype, 'selectMany', {\r\n enumerable: false,\r\n writable: true,\r\n value: function (selector: (item: any) => any[]) {\r\n return this.reduce((out: any, v: any) => {\r\n let vres = selector(v);\r\n if (vres != null) {\r\n if (!Array.isArray(vres)) vres = [vres];\r\n out.push(...vres);\r\n }\r\n\r\n return out;\r\n }, []);\r\n },\r\n});\r\n\r\nObject.defineProperty(Array.prototype, 'first', {\r\n enumerable: false,\r\n writable: true,\r\n value: function () {\r\n let item;\r\n if (this.length) {\r\n item = this[0];\r\n }\r\n return item;\r\n },\r\n});\r\n\r\nObject.defineProperty(Array.prototype, 'last', {\r\n enumerable: false,\r\n writable: true,\r\n value: function () {\r\n if (this.length) {\r\n return this[this.length - 1];\r\n }\r\n return undefined;\r\n },\r\n});\r\n\r\nObject.defineProperty(Array.prototype, 'getCommonOrNull', {\r\n enumerable: false,\r\n writable: true,\r\n value: function (valueDelegate: (item: any) => any) {\r\n const arr = this as any[];\r\n if (this.length && arr.every(i => valueDelegate(i) == valueDelegate(arr[0]))) {\r\n return valueDelegate(arr[0]);\r\n }\r\n return null;\r\n },\r\n});\r\n\r\nObject.defineProperty(Array.prototype, 'toMap', {\r\n enumerable: false,\r\n writable: true,\r\n value: function (keySelector: (item: any) => any) {\r\n const m = new Map();\r\n for (const i of this) {\r\n m.set(keySelector(i), i);\r\n }\r\n return m;\r\n },\r\n});\r\n\r\nObject.defineProperty(Array.prototype, 'clear', {\r\n enumerable: false,\r\n writable: true,\r\n value: function () {\r\n if (this.length > 0) {\r\n this.splice(0, this.length);\r\n }\r\n },\r\n});\r\n\r\nObject.defineProperty(Array.prototype, 'reorder', {\r\n enumerable: false,\r\n writable: true,\r\n value: function (fromIndex: number, toIndex: number, undo?: boolean) {\r\n if (undo) {\r\n if (fromIndex > toIndex) {\r\n fromIndex--;\r\n } else {\r\n toIndex++;\r\n }\r\n }\r\n const itemToMove = this[fromIndex];\r\n for (let i = fromIndex; i < this.length - 1; i++) {\r\n this[i] = this[i + 1];\r\n }\r\n if (toIndex > fromIndex) {\r\n toIndex--;\r\n }\r\n for (let i = this.length - 1; i > toIndex; i--) {\r\n this[i] = this[i - 1];\r\n }\r\n this[toIndex] = itemToMove;\r\n },\r\n});\r\n\r\nObject.defineProperty(Set.prototype, 'find', {\r\n enumerable: false,\r\n writable: true,\r\n value: function (predicate: (item: any) => boolean) {\r\n for (const i of this) {\r\n if (predicate(i)) return i;\r\n }\r\n return undefined;\r\n },\r\n});\r\n\r\nObject.defineProperty(Map.prototype, 'hasAny', {\r\n enumerable: false,\r\n writable: true,\r\n value: function (...keys: any[]) {\r\n return keys.some(k => this.has(k));\r\n },\r\n});\r\n\r\nexport {};\r\n","/******************************************************************************\r\nCopyright (c) Microsoft Corporation.\r\n\r\nPermission to use, copy, modify, and/or distribute this software for any\r\npurpose with or without fee is hereby granted.\r\n\r\nTHE SOFTWARE IS PROVIDED \"AS IS\" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH\r\nREGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY\r\nAND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,\r\nINDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM\r\nLOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR\r\nOTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR\r\nPERFORMANCE OF THIS SOFTWARE.\r\n***************************************************************************** */\r\n/* global Reflect, Promise, SuppressedError, Symbol */\r\n\r\nvar extendStatics = function(d, b) {\r\n extendStatics = Object.setPrototypeOf ||\r\n ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||\r\n function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; };\r\n return extendStatics(d, b);\r\n};\r\n\r\nexport function __extends(d, b) {\r\n if (typeof b !== \"function\" && b !== null)\r\n throw new TypeError(\"Class extends value \" + String(b) + \" is not a constructor or null\");\r\n extendStatics(d, b);\r\n function __() { this.constructor = d; }\r\n d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());\r\n}\r\n\r\nexport var __assign = function() {\r\n __assign = Object.assign || function __assign(t) {\r\n for (var s, i = 1, n = arguments.length; i < n; i++) {\r\n s = arguments[i];\r\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p];\r\n }\r\n return t;\r\n }\r\n return __assign.apply(this, arguments);\r\n}\r\n\r\nexport function __rest(s, e) {\r\n var t = {};\r\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)\r\n t[p] = s[p];\r\n if (s != null && typeof Object.getOwnPropertySymbols === \"function\")\r\n for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {\r\n if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))\r\n t[p[i]] = s[p[i]];\r\n }\r\n return t;\r\n}\r\n\r\nexport function __decorate(decorators, target, key, desc) {\r\n var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;\r\n if (typeof Reflect === \"object\" && typeof Reflect.decorate === \"function\") r = Reflect.decorate(decorators, target, key, desc);\r\n else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;\r\n return c > 3 && r && Object.defineProperty(target, key, r), r;\r\n}\r\n\r\nexport function __param(paramIndex, decorator) {\r\n return function (target, key) { decorator(target, key, paramIndex); }\r\n}\r\n\r\nexport function __esDecorate(ctor, descriptorIn, decorators, contextIn, initializers, extraInitializers) {\r\n function accept(f) { if (f !== void 0 && typeof f !== \"function\") throw new TypeError(\"Function expected\"); return f; }\r\n var kind = contextIn.kind, key = kind === \"getter\" ? \"get\" : kind === \"setter\" ? \"set\" : \"value\";\r\n var target = !descriptorIn && ctor ? contextIn[\"static\"] ? ctor : ctor.prototype : null;\r\n var descriptor = descriptorIn || (target ? Object.getOwnPropertyDescriptor(target, contextIn.name) : {});\r\n var _, done = false;\r\n for (var i = decorators.length - 1; i >= 0; i--) {\r\n var context = {};\r\n for (var p in contextIn) context[p] = p === \"access\" ? {} : contextIn[p];\r\n for (var p in contextIn.access) context.access[p] = contextIn.access[p];\r\n context.addInitializer = function (f) { if (done) throw new TypeError(\"Cannot add initializers after decoration has completed\"); extraInitializers.push(accept(f || null)); };\r\n var result = (0, decorators[i])(kind === \"accessor\" ? { get: descriptor.get, set: descriptor.set } : descriptor[key], context);\r\n if (kind === \"accessor\") {\r\n if (result === void 0) continue;\r\n if (result === null || typeof result !== \"object\") throw new TypeError(\"Object expected\");\r\n if (_ = accept(result.get)) descriptor.get = _;\r\n if (_ = accept(result.set)) descriptor.set = _;\r\n if (_ = accept(result.init)) initializers.unshift(_);\r\n }\r\n else if (_ = accept(result)) {\r\n if (kind === \"field\") initializers.unshift(_);\r\n else descriptor[key] = _;\r\n }\r\n }\r\n if (target) Object.defineProperty(target, contextIn.name, descriptor);\r\n done = true;\r\n};\r\n\r\nexport function __runInitializers(thisArg, initializers, value) {\r\n var useValue = arguments.length > 2;\r\n for (var i = 0; i < initializers.length; i++) {\r\n value = useValue ? initializers[i].call(thisArg, value) : initializers[i].call(thisArg);\r\n }\r\n return useValue ? value : void 0;\r\n};\r\n\r\nexport function __propKey(x) {\r\n return typeof x === \"symbol\" ? x : \"\".concat(x);\r\n};\r\n\r\nexport function __setFunctionName(f, name, prefix) {\r\n if (typeof name === \"symbol\") name = name.description ? \"[\".concat(name.description, \"]\") : \"\";\r\n return Object.defineProperty(f, \"name\", { configurable: true, value: prefix ? \"\".concat(prefix, \" \", name) : name });\r\n};\r\n\r\nexport function __metadata(metadataKey, metadataValue) {\r\n if (typeof Reflect === \"object\" && typeof Reflect.metadata === \"function\") return Reflect.metadata(metadataKey, metadataValue);\r\n}\r\n\r\nexport function __awaiter(thisArg, _arguments, P, generator) {\r\n function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\r\n return new (P || (P = Promise))(function (resolve, reject) {\r\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\r\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\r\n function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\r\n step((generator = generator.apply(thisArg, _arguments || [])).next());\r\n });\r\n}\r\n\r\nexport function __generator(thisArg, body) {\r\n var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;\r\n return g = { next: verb(0), \"throw\": verb(1), \"return\": verb(2) }, typeof Symbol === \"function\" && (g[Symbol.iterator] = function() { return this; }), g;\r\n function verb(n) { return function (v) { return step([n, v]); }; }\r\n function step(op) {\r\n if (f) throw new TypeError(\"Generator is already executing.\");\r\n while (g && (g = 0, op[0] && (_ = 0)), _) try {\r\n if (f = 1, y && (t = op[0] & 2 ? y[\"return\"] : op[0] ? y[\"throw\"] || ((t = y[\"return\"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;\r\n if (y = 0, t) op = [op[0] & 2, t.value];\r\n switch (op[0]) {\r\n case 0: case 1: t = op; break;\r\n case 4: _.label++; return { value: op[1], done: false };\r\n case 5: _.label++; y = op[1]; op = [0]; continue;\r\n case 7: op = _.ops.pop(); _.trys.pop(); continue;\r\n default:\r\n if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }\r\n if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }\r\n if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }\r\n if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }\r\n if (t[2]) _.ops.pop();\r\n _.trys.pop(); continue;\r\n }\r\n op = body.call(thisArg, _);\r\n } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }\r\n if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };\r\n }\r\n}\r\n\r\nexport var __createBinding = Object.create ? (function(o, m, k, k2) {\r\n if (k2 === undefined) k2 = k;\r\n var desc = Object.getOwnPropertyDescriptor(m, k);\r\n if (!desc || (\"get\" in desc ? !m.__esModule : desc.writable || desc.configurable)) {\r\n desc = { enumerable: true, get: function() { return m[k]; } };\r\n }\r\n Object.defineProperty(o, k2, desc);\r\n}) : (function(o, m, k, k2) {\r\n if (k2 === undefined) k2 = k;\r\n o[k2] = m[k];\r\n});\r\n\r\nexport function __exportStar(m, o) {\r\n for (var p in m) if (p !== \"default\" && !Object.prototype.hasOwnProperty.call(o, p)) __createBinding(o, m, p);\r\n}\r\n\r\nexport function __values(o) {\r\n var s = typeof Symbol === \"function\" && Symbol.iterator, m = s && o[s], i = 0;\r\n if (m) return m.call(o);\r\n if (o && typeof o.length === \"number\") return {\r\n next: function () {\r\n if (o && i >= o.length) o = void 0;\r\n return { value: o && o[i++], done: !o };\r\n }\r\n };\r\n throw new TypeError(s ? \"Object is not iterable.\" : \"Symbol.iterator is not defined.\");\r\n}\r\n\r\nexport function __read(o, n) {\r\n var m = typeof Symbol === \"function\" && o[Symbol.iterator];\r\n if (!m) return o;\r\n var i = m.call(o), r, ar = [], e;\r\n try {\r\n while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value);\r\n }\r\n catch (error) { e = { error: error }; }\r\n finally {\r\n try {\r\n if (r && !r.done && (m = i[\"return\"])) m.call(i);\r\n }\r\n finally { if (e) throw e.error; }\r\n }\r\n return ar;\r\n}\r\n\r\n/** @deprecated */\r\nexport function __spread() {\r\n for (var ar = [], i = 0; i < arguments.length; i++)\r\n ar = ar.concat(__read(arguments[i]));\r\n return ar;\r\n}\r\n\r\n/** @deprecated */\r\nexport function __spreadArrays() {\r\n for (var s = 0, i = 0, il = arguments.length; i < il; i++) s += arguments[i].length;\r\n for (var r = Array(s), k = 0, i = 0; i < il; i++)\r\n for (var a = arguments[i], j = 0, jl = a.length; j < jl; j++, k++)\r\n r[k] = a[j];\r\n return r;\r\n}\r\n\r\nexport function __spreadArray(to, from, pack) {\r\n if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) {\r\n if (ar || !(i in from)) {\r\n if (!ar) ar = Array.prototype.slice.call(from, 0, i);\r\n ar[i] = from[i];\r\n }\r\n }\r\n return to.concat(ar || Array.prototype.slice.call(from));\r\n}\r\n\r\nexport function __await(v) {\r\n return this instanceof __await ? (this.v = v, this) : new __await(v);\r\n}\r\n\r\nexport function __asyncGenerator(thisArg, _arguments, generator) {\r\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\r\n var g = generator.apply(thisArg, _arguments || []), i, q = [];\r\n return i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i;\r\n function verb(n) { if (g[n]) i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; }\r\n function resume(n, v) { try { step(g[n](v)); } catch (e) { settle(q[0][3], e); } }\r\n function step(r) { r.value instanceof __await ? Promise.resolve(r.value.v).then(fulfill, reject) : settle(q[0][2], r); }\r\n function fulfill(value) { resume(\"next\", value); }\r\n function reject(value) { resume(\"throw\", value); }\r\n function settle(f, v) { if (f(v), q.shift(), q.length) resume(q[0][0], q[0][1]); }\r\n}\r\n\r\nexport function __asyncDelegator(o) {\r\n var i, p;\r\n return i = {}, verb(\"next\"), verb(\"throw\", function (e) { throw e; }), verb(\"return\"), i[Symbol.iterator] = function () { return this; }, i;\r\n function verb(n, f) { i[n] = o[n] ? function (v) { return (p = !p) ? { value: __await(o[n](v)), done: false } : f ? f(v) : v; } : f; }\r\n}\r\n\r\nexport function __asyncValues(o) {\r\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\r\n var m = o[Symbol.asyncIterator], i;\r\n return m ? m.call(o) : (o = typeof __values === \"function\" ? __values(o) : o[Symbol.iterator](), i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i);\r\n function verb(n) { i[n] = o[n] && function (v) { return new Promise(function (resolve, reject) { v = o[n](v), settle(resolve, reject, v.done, v.value); }); }; }\r\n function settle(resolve, reject, d, v) { Promise.resolve(v).then(function(v) { resolve({ value: v, done: d }); }, reject); }\r\n}\r\n\r\nexport function __makeTemplateObject(cooked, raw) {\r\n if (Object.defineProperty) { Object.defineProperty(cooked, \"raw\", { value: raw }); } else { cooked.raw = raw; }\r\n return cooked;\r\n};\r\n\r\nvar __setModuleDefault = Object.create ? (function(o, v) {\r\n Object.defineProperty(o, \"default\", { enumerable: true, value: v });\r\n}) : function(o, v) {\r\n o[\"default\"] = v;\r\n};\r\n\r\nexport function __importStar(mod) {\r\n if (mod && mod.__esModule) return mod;\r\n var result = {};\r\n if (mod != null) for (var k in mod) if (k !== \"default\" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);\r\n __setModuleDefault(result, mod);\r\n return result;\r\n}\r\n\r\nexport function __importDefault(mod) {\r\n return (mod && mod.__esModule) ? mod : { default: mod };\r\n}\r\n\r\nexport function __classPrivateFieldGet(receiver, state, kind, f) {\r\n if (kind === \"a\" && !f) throw new TypeError(\"Private accessor was defined without a getter\");\r\n if (typeof state === \"function\" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError(\"Cannot read private member from an object whose class did not declare it\");\r\n return kind === \"m\" ? f : kind === \"a\" ? f.call(receiver) : f ? f.value : state.get(receiver);\r\n}\r\n\r\nexport function __classPrivateFieldSet(receiver, state, value, kind, f) {\r\n if (kind === \"m\") throw new TypeError(\"Private method is not writable\");\r\n if (kind === \"a\" && !f) throw new TypeError(\"Private accessor was defined without a setter\");\r\n if (typeof state === \"function\" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError(\"Cannot write private member to an object whose class did not declare it\");\r\n return (kind === \"a\" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;\r\n}\r\n\r\nexport function __classPrivateFieldIn(state, receiver) {\r\n if (receiver === null || (typeof receiver !== \"object\" && typeof receiver !== \"function\")) throw new TypeError(\"Cannot use 'in' operator on non-object\");\r\n return typeof state === \"function\" ? receiver === state : state.has(receiver);\r\n}\r\n\r\nexport function __addDisposableResource(env, value, async) {\r\n if (value !== null && value !== void 0) {\r\n if (typeof value !== \"object\" && typeof value !== \"function\") throw new TypeError(\"Object expected.\");\r\n var dispose;\r\n if (async) {\r\n if (!Symbol.asyncDispose) throw new TypeError(\"Symbol.asyncDispose is not defined.\");\r\n dispose = value[Symbol.asyncDispose];\r\n }\r\n if (dispose === void 0) {\r\n if (!Symbol.dispose) throw new TypeError(\"Symbol.dispose is not defined.\");\r\n dispose = value[Symbol.dispose];\r\n }\r\n if (typeof dispose !== \"function\") throw new TypeError(\"Object not disposable.\");\r\n env.stack.push({ value: value, dispose: dispose, async: async });\r\n }\r\n else if (async) {\r\n env.stack.push({ async: true });\r\n }\r\n return value;\r\n}\r\n\r\nvar _SuppressedError = typeof SuppressedError === \"function\" ? SuppressedError : function (error, suppressed, message) {\r\n var e = new Error(message);\r\n return e.name = \"SuppressedError\", e.error = error, e.suppressed = suppressed, e;\r\n};\r\n\r\nexport function __disposeResources(env) {\r\n function fail(e) {\r\n env.error = env.hasError ? new _SuppressedError(e, env.error, \"An error was suppressed during disposal.\") : e;\r\n env.hasError = true;\r\n }\r\n function next() {\r\n while (env.stack.length) {\r\n var rec = env.stack.pop();\r\n try {\r\n var result = rec.dispose && rec.dispose.call(rec.value);\r\n if (rec.async) return Promise.resolve(result).then(next, function(e) { fail(e); return next(); });\r\n }\r\n catch (e) {\r\n fail(e);\r\n }\r\n }\r\n if (env.hasError) throw env.error;\r\n }\r\n return next();\r\n}\r\n\r\nexport default {\r\n __extends: __extends,\r\n __assign: __assign,\r\n __rest: __rest,\r\n __decorate: __decorate,\r\n __param: __param,\r\n __metadata: __metadata,\r\n __awaiter: __awaiter,\r\n __generator: __generator,\r\n __createBinding: __createBinding,\r\n __exportStar: __exportStar,\r\n __values: __values,\r\n __read: __read,\r\n __spread: __spread,\r\n __spreadArrays: __spreadArrays,\r\n __spreadArray: __spreadArray,\r\n __await: __await,\r\n __asyncGenerator: __asyncGenerator,\r\n __asyncDelegator: __asyncDelegator,\r\n __asyncValues: __asyncValues,\r\n __makeTemplateObject: __makeTemplateObject,\r\n __importStar: __importStar,\r\n __importDefault: __importDefault,\r\n __classPrivateFieldGet: __classPrivateFieldGet,\r\n __classPrivateFieldSet: __classPrivateFieldSet,\r\n __classPrivateFieldIn: __classPrivateFieldIn,\r\n __addDisposableResource: __addDisposableResource,\r\n __disposeResources: __disposeResources,\r\n};\r\n","import { FontService, HttpService, MeshService } from '@models/services';\r\nimport { MaterialService } from '../services/material-service';\r\nimport { AnnotationNode, DimensionNode } from './annotations';\r\nimport {\r\n BaseConnector,\r\n CircularPatternFeature,\r\n DeleteFacesFeature,\r\n DisplacementMapFeature,\r\n ExtrudeFeature,\r\n Feature,\r\n JoinGeometryFeature,\r\n MoveVerticesFeature,\r\n NormalSmoothingFeature,\r\n SweepFeature,\r\n TransformFeature,\r\n UvMapFeature,\r\n WeldVerticesFeature,\r\n} from './features';\r\nimport { Connector } from './features/connector';\r\nimport { MirrorFeature } from './features/mirror-feature';\r\nimport { PatternFeature } from './features/pattern-feature';\r\nimport { SketchControlPoint } from './features/sketch-control-point';\r\nimport { SliceFeature } from './features/slice-feature';\r\nimport { SubmeshFeature } from './features/submesh-feature';\r\nimport { IImportInfo } from './kb3d-models';\r\nimport { Kb3dObject } from './kb3d-object';\r\nimport { DirectionalLightNode, HemisphericLightNode, LightNode, PointLightNode, SpotLightNode } from './lights';\r\nimport {\r\n AdvancedMaterialNode,\r\n FileTextureLayer,\r\n MaterialNode,\r\n SvgTextureLayer,\r\n TextTextureLayer,\r\n TextureLayer,\r\n} from './materials';\r\nimport { ColorFillTextureLayer } from './materials/color-fill-texture-layer';\r\nimport { MateNode } from './mates/mate-node';\r\nimport {\r\n BoxNode,\r\n CustomMeshNode,\r\n CylinderNode,\r\n DiscNode,\r\n MeshGroupNode,\r\n MeshNode,\r\n PlaneNode,\r\n PrimitiveMeshNode,\r\n SpaceNode,\r\n SphereNode,\r\n TorusNode,\r\n} from './meshes';\r\nimport { ConeNode } from './meshes/cone-node';\r\nimport { SketchNode } from './meshes/sketch-node';\r\nimport { SketchPath } from './meshes/sketch-node-path';\r\nimport { TextNode } from './meshes/text-node';\r\nimport { KbColor, KbGeometry, KbQuaternion, KbVector } from './primitives';\r\nimport { SceneNode } from './scene-node';\r\nimport { ViewpointNode } from './viewpoints/viewpoint-node';\r\n\r\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\r\nexport class Token {\r\n protected constructor(public key: string) {}\r\n\r\n public static withKey(key: keyof typeof Token): Token {\r\n const token = this[key] as Token;\r\n if (!token) throw new Error(`Could not find token for key '${key}'`);\r\n return token;\r\n }\r\n\r\n public static tryWithKey(key: keyof typeof Token): Token | undefined {\r\n const token = this[key] as Token;\r\n return token;\r\n }\r\n\r\n /** Services */\r\n public static HttpService = new Token('HttpService');\r\n public static MaterialService = new Token('MaterialService');\r\n public static MeshService = new Token('MeshService');\r\n public static JobService = new Token('JobService');\r\n public static SceneService = new Token('SceneService');\r\n public static FontService = new Token('FontService');\r\n public static Ignore = new Token('Ignore');\r\n\r\n /** Roots */\r\n public static SceneNode = new Token('SceneNode');\r\n public static Kb3dObject = new Token('Kb3dObject');\r\n public static ImportInfo = new Token('ImportInfo');\r\n\r\n /** Lights */\r\n public static LightNode = new Token('LightNode');\r\n public static DirectionalLightNode = new Token('DirectionalLightNode');\r\n public static HemisphericLightNode = new Token('HemisphericLightNode');\r\n public static PointLightNode = new Token('PointLightNode');\r\n public static SpotLightNode = new Token('SpotLightNode');\r\n\r\n /** Meshes */\r\n public static BoxNode = new Token('BoxNode');\r\n public static CustomMeshNode = new Token('CustomMeshNode');\r\n public static CylinderNode = new Token('CylinderNode');\r\n public static ConeNode = new Token('ConeNode');\r\n public static DiscNode = new Token('DiscNode');\r\n public static PlaneNode = new Token('PlaneNode');\r\n public static SphereNode = new Token('SphereNode');\r\n public static TorusNode = new Token('TorusNode');\r\n public static MeshNode = new Token('MeshNode');\r\n public static PrimitiveMeshNode = new Token('PrimitiveMeshNode');\r\n public static MeshGroupNode = new Token('MeshGroupNode');\r\n public static SpaceNode = new Token('SpaceNode');\r\n public static TextNode = new Token('TextNode');\r\n public static SketchNode = new Token('SketchNode');\r\n public static SketchPath = new Token('SketchPath');\r\n\r\n /** Dimensions */\r\n public static AnnotationNode = new Token('AnnotationNode');\r\n public static DimensionNode = new Token('DimensionNode');\r\n\r\n /** Primitives */\r\n public static KbColor = new Token('KbColor');\r\n // public static KbColor4 = new Token(\"KbColor4\");\r\n public static KbGeometry = new Token('KbGeometry');\r\n public static KbQuaternion = new Token('KbQuaternion');\r\n public static KbVector = new Token('KbVector');\r\n\r\n /** Materials */\r\n public static MaterialNode = new Token('MaterialNode');\r\n public static AdvancedMaterialNode = new Token('AdvancedMaterialNode');\r\n public static MaterialMapping = new Token('MaterialMapping');\r\n\r\n /** Textures */\r\n public static TextureLayer = new Token('TextureLayer');\r\n public static FileTextureLayer = new Token('FileTextureLayer');\r\n public static SvgTextureLayer = new Token('SvgTextureLayer');\r\n public static TextTextureLayer = new Token('TextTextureLayer');\r\n public static ColorFillTextureLayer = new Token('ColorFillTextureLayer');\r\n\r\n /** Viewpoints */\r\n public static ViewpointNode = new Token('ViewpointNode');\r\n\r\n /** Features */\r\n public static Feature = new Token('Feature');\r\n public static BaseConnector = new Token('BaseConnector');\r\n public static Connector = new Token('Connector');\r\n public static SketchControlPoint = new Token('SketchControlPoint');\r\n public static SubmeshFeature = new Token('SubmeshFeature');\r\n public static MoveVerticesFeature = new Token('MoveVerticesFeature');\r\n public static WeldVerticesFeature = new Token('WeldVerticesFeature');\r\n public static DisplacementMapFeature = new Token('DisplacementMapFeature');\r\n public static UvMapFeature = new Token('UvMapFeature');\r\n public static LinearPatternFeature = new Token('LinearPatternFeature');\r\n public static CircularPatternFeature = new Token('CircularPatternFeature');\r\n public static JoinGeometryFeature = new Token('JoinGeometryFeature');\r\n public static DeleteFacesFeature = new Token('DeleteFacesFeature');\r\n public static NormalSmoothingFeature = new Token('NormalSmoothingFeature');\r\n public static SliceFeature = new Token('SliceFeature');\r\n public static MirrorFeature = new Token('MirrorFeature');\r\n public static TransformFeature = new Token('TransformFeature');\r\n public static ExtrudeFeature = new Token('ExtrudeFeature');\r\n public static SweepFeature = new Token('SweepFeature');\r\n\r\n public static MateNode = new Token('MateNode');\r\n\r\n protected static MapFrom(...tokens: Token[]) {\r\n return new Map(tokens.map(t => [t, null]));\r\n }\r\n\r\n public static PrimitiveTokens = [Token.KbVector, Token.KbQuaternion, Token.KbColor];\r\n public static PrimitiveTokenMap = Token.MapFrom(...Token.PrimitiveTokens);\r\n\r\n public static PrimitiveMeshTokens = [\r\n Token.BoxNode,\r\n Token.CylinderNode,\r\n Token.ConeNode,\r\n Token.DiscNode,\r\n Token.PlaneNode,\r\n Token.SphereNode,\r\n Token.TorusNode,\r\n ];\r\n public static PrimitiveMeshTokenMap = Token.MapFrom(...Token.PrimitiveMeshTokens);\r\n\r\n public static MeshTokens = Token.PrimitiveMeshTokens.concat(Token.CustomMeshNode, Token.SketchNode, Token.TextNode);\r\n public static MeshTokenMap = Token.MapFrom(...Token.MeshTokens);\r\n\r\n public static MeshAndGroupTokens = Token.MeshTokens.concat(Token.MeshGroupNode);\r\n public static MeshAndGroupTokenMap = Token.MapFrom(...Token.MeshAndGroupTokens);\r\n\r\n public static SpaceTokens = Token.MeshAndGroupTokens.concat(Token.SceneNode);\r\n public static SpaceTokenMap = Token.MapFrom(...Token.SpaceTokens);\r\n\r\n public static MaterialTokens = [Token.AdvancedMaterialNode];\r\n\r\n public static MaterialTokenMap = Token.MapFrom(...Token.MaterialTokens);\r\n\r\n public static AnnotationTokens = [Token.AnnotationNode, Token.DimensionNode];\r\n\r\n public static AnnotationTokenMap = Token.MapFrom(...Token.AnnotationTokens);\r\n\r\n public static PositionalLightTokens = [Token.PointLightNode, Token.SpotLightNode];\r\n public static GlobalLightTokens = [Token.DirectionalLightNode, Token.HemisphericLightNode];\r\n public static LightTokens = [...Token.GlobalLightTokens, ...Token.PositionalLightTokens];\r\n public static LightTokenMap = Token.MapFrom(...Token.LightTokens);\r\n\r\n public static FeatureTokens = [\r\n Token.MoveVerticesFeature,\r\n Token.WeldVerticesFeature,\r\n Token.DisplacementMapFeature,\r\n Token.UvMapFeature,\r\n Token.SubmeshFeature,\r\n Token.LinearPatternFeature,\r\n Token.CircularPatternFeature,\r\n Token.JoinGeometryFeature,\r\n Token.DeleteFacesFeature,\r\n Token.NormalSmoothingFeature,\r\n Token.SliceFeature,\r\n Token.MirrorFeature,\r\n Token.TransformFeature,\r\n Token.ExtrudeFeature,\r\n Token.SweepFeature,\r\n ];\r\n public static FeatureTokenMap = Token.MapFrom(...Token.FeatureTokens);\r\n\r\n public static ConnectorTokens = [Token.Connector, Token.SketchControlPoint];\r\n\r\n public static TextureLayerTokens = [\r\n Token.TextureLayer,\r\n Token.FileTextureLayer,\r\n Token.SvgTextureLayer,\r\n Token.TextTextureLayer,\r\n Token.ColorFillTextureLayer,\r\n ];\r\n public static TextureLayerTokenMap = Token.MapFrom(...Token.TextureLayerTokens);\r\n\r\n public static NodeTokens = [\r\n ...Token.SpaceTokens,\r\n ...Token.MaterialTokens,\r\n ...Token.LightTokens,\r\n ...Token.AnnotationTokens,\r\n Token.ViewpointNode,\r\n Token.MateNode,\r\n ];\r\n public static NodeTokenMap = Token.MapFrom(...Token.NodeTokens);\r\n public static BaseNodeTokens = [\r\n Token.SpaceNode,\r\n Token.MaterialNode,\r\n Token.LightNode,\r\n Token.ViewpointNode,\r\n Token.MateNode,\r\n Token.AnnotationNode,\r\n ];\r\n\r\n public static AllTokens = Token.NodeTokens.concat(\r\n ...Token.PrimitiveTokens,\r\n Token.FeatureTokens,\r\n Token.TextureLayerTokens,\r\n Token.ConnectorTokens,\r\n [Token.SketchPath]\r\n );\r\n public static AllTokenMap = Token.MapFrom(...Token.AllTokens);\r\n\r\n public isPrimitive(): boolean {\r\n return Token.PrimitiveTokenMap.has(this);\r\n }\r\n public isPrimitiveMesh(): boolean {\r\n return Token.PrimitiveMeshTokenMap.has(this);\r\n }\r\n public isMesh(): boolean {\r\n return Token.MeshTokenMap.has(this);\r\n }\r\n public isMeshOrGroup(): boolean {\r\n return Token.MeshAndGroupTokenMap.has(this);\r\n }\r\n public isMaterial(): boolean {\r\n return Token.MaterialTokenMap.has(this);\r\n }\r\n public isFeature(): boolean {\r\n return Token.FeatureTokenMap.has(this);\r\n }\r\n public isTexture(): boolean {\r\n return Token.TextureLayerTokenMap.has(this);\r\n }\r\n public isLight(): boolean {\r\n return Token.LightTokenMap.has(this);\r\n }\r\n public isViewpoint(): boolean {\r\n return Token.ViewpointNode === this;\r\n }\r\n public isMate(): boolean {\r\n return Token.MateNode === this;\r\n }\r\n public isConnector(): boolean {\r\n return Token.Connector === this || Token.SketchControlPoint === this;\r\n }\r\n public isAnnotation(): boolean {\r\n return Token.AnnotationTokenMap.has(this);\r\n }\r\n public isNode(): boolean {\r\n return Token.NodeTokenMap.has(this);\r\n }\r\n public isSceneNode(): boolean {\r\n return Token.SceneNode === this;\r\n }\r\n public isSketchPath(): boolean {\r\n return Token.SketchPath === this;\r\n }\r\n}\r\n","import { ITrackedClass } from '@models/interfaces';\r\nimport { Token } from '../token';\r\nimport { Validator } from '../validator';\r\nimport { TrackChangesOptions } from './change-tracker';\r\nimport { IDecoratedModel } from './kb3d-decorator';\r\n\r\nexport type Constructor = new (...args: any[]) => T;\r\n// eslint-disable-next-line @typescript-eslint/ban-types\r\nexport type AbstractConstructor = Function & { prototype: T };\r\n\r\nexport interface IKb3dModel extends TrackChangesOptions {\r\n token: Token;\r\n classConstructor: Constructor | AbstractConstructor;\r\n /**\r\n * the function to create the class used by the manager.\r\n * If provided, you will be passed an anonymous object in the shape\r\n * of the class that can be used to create the instance.\r\n * If not provided, the constructor of the class will be called by\r\n * the manager with no arguments.\r\n */\r\n create?: (args: Partial) => T;\r\n afterCreate?: (instance: T) => void;\r\n /**\r\n * Hook that happens while serializing this model to JSON.\r\n * You are passed the instance and the default serialization of the model,\r\n * which you can modify.\r\n * Return true to include the instance in the serialization, or false to omit it\r\n */\r\n onSerializing?: (instance: T, serializedObject: Partial) => boolean;\r\n\r\n /** if trackChanges is true, the model will be wrapped in the TrackChanges decorator */\r\n trackChanges: boolean;\r\n\r\n validate?: (node: ITrackedClass, v: Validator) => void;\r\n}\r\n\r\nexport const Kb3dModels = new Map>();\r\n\r\n/**\r\n * Curry function to filter items in an array to items of a token type\r\n * @param classKey String key of the class you want to see if the instance is an instance of\r\n */\r\nexport function ofToken(token: Token) {\r\n return (instance: any): instance is T => instanceOf(instance, token);\r\n}\r\n\r\n/**\r\n * Checks if the instance of a class given is an instance of the class represented by it's\r\n * injectable class key. If the instance is an instance of a class that was not\r\n * decorated, or if the class represented by the key was not decorated, then this will\r\n * return false!\r\n * @param instance An instance of a class that was decorated\r\n * @param classKey String key of the class you want to see if the instance is an instance of\r\n */\r\nexport function instanceOf(instance: any, token: Token): instance is T {\r\n if (!instance) return false;\r\n\r\n const model = Kb3dModels.get(token);\r\n return model ? instance instanceof model.classConstructor : false;\r\n}\r\n\r\n/** Curry-able version of instanceOf */\r\nexport function isInstanceOf(token: Token) {\r\n return (instance: any): instance is T => instanceOf(instance, token);\r\n}\r\n\r\nexport function getToken(object: T): Token | undefined {\r\n const node = object as unknown as IDecoratedModel;\r\n if (node.$type) {\r\n return Token.tryWithKey(node.$type);\r\n }\r\n}\r\n\r\n/**\r\n * Checks if the instance of a class given is an instance of the class represented by it's\r\n * injectable class key. If the instance is an instance of a class that was not\r\n * decorated, or if the class represented by the key was not decorated, then this will\r\n * return false!\r\n * @param instance An instance of a class that was decorated\r\n * @param classKey String key of the class you want to see if the instance is an instance of\r\n */\r\nexport function instanceOfAny(instance: any, ...tokens: Token[]): instance is T {\r\n if (!instance) return false;\r\n\r\n return tokens.some(t => instanceOf(instance, t));\r\n}\r\n","import { Kb3dModels } from '@models/classes/decorators/model-utils';\r\nimport { Token } from '@models/classes/token';\r\n\r\nexport type ITrackedClass = T & {\r\n isModified: boolean;\r\n /** properties that have been changed since the last render processing */\r\n changes: Map, any>;\r\n /** the values of properties before they were changed */\r\n originalValues: Map;\r\n destroy: () => void;\r\n removedFromChildArray: (parent: ITrackedClass, item: any, arrayPropertyName: string) => void;\r\n addedToChildArray: (parent: ITrackedClass, item: any, arrayPropertyName: string) => void;\r\n commitChanges: () => void;\r\n setPropertyIsModified: (prop: string | symbol, modified: boolean) => void;\r\n trackChanges: (active: boolean) => void;\r\n nodeReferences:\r\n | Map<\r\n string | symbol,\r\n {\r\n classRef: new (...args: any[]) => any;\r\n limitKeys: Set | undefined;\r\n }\r\n >\r\n | undefined;\r\n rawNode: T;\r\n};\r\n\r\nexport type ITrackedArray = T[] & {\r\n pop: (destroy: boolean) => T | undefined;\r\n};\r\n\r\nexport enum EChangeOperation {\r\n // CreateAndPush and Remove correspond to functions in the custom KBArray class.\r\n // They use Push and Splice internally but also signify we need a\r\n // changeTracker destroy on undo and redo respectively and create on redo and undo respectively.\r\n Push,\r\n Pop,\r\n Shift,\r\n Unshift,\r\n Splice,\r\n CreateAndPush,\r\n Remove,\r\n}\r\n\r\nexport interface IHistoryStateChange {\r\n node: G & ITrackedClass;\r\n properties: Array>;\r\n}\r\nexport interface IHistoryStateChangeProperty {\r\n name: keyof G;\r\n operation?: EChangeOperation;\r\n previousValue: T;\r\n newValue: T;\r\n}\r\n\r\nexport function isTrackedClass(object: T): object is ITrackedClass {\r\n return object && typeof object === 'object' && '$changesTracked' in object;\r\n}\r\n\r\nexport function isTrackedArray(object: T[]): object is ITrackedArray {\r\n return !!(object as any).$isTracked;\r\n}\r\n\r\nexport function isTrackedInstance(instance: T, token: Token): instance is ITrackedClass {\r\n if (!instance) return false;\r\n\r\n const model = Kb3dModels.get(token);\r\n return model ? instance instanceof model.classConstructor && isTrackedClass(instance) : false;\r\n}\r\n\r\nexport function isTrackedInstanceOfAny(instance: T, ...tokens: Token[]): instance is T {\r\n if (!instance) return false;\r\n\r\n return tokens.some(t => isTrackedInstance(instance, t));\r\n}\r\n","import { ITrackedClass } from '@models/interfaces/tracked-class';\r\n\r\nconst changedItems = new Set();\r\nconst deletedItems = new Set();\r\n\r\n/** This class syncs changes to the renderer to update the scene view */\r\nexport class SyncService {\r\n static registerChangedState(tracker: ITrackedClass, modified: boolean) {\r\n if (modified) {\r\n // If node was deleted, ignore further changes, unless the item was moved to a different parent\r\n if (!deletedItems.has(tracker) || tracker.changes.has('_parent')) {\r\n changedItems.add(tracker);\r\n }\r\n } else {\r\n changedItems.delete(tracker);\r\n }\r\n }\r\n // When a new object is being created, the proxy being used to set the values will be the proxy of the parent of the class being created.\r\n // So, we need to make sure the changed items map is always pointing to the correct proxy (the proxy with the highest specificity)\r\n static registerNewObject(tracker: ITrackedClass) {\r\n changedItems.add(tracker);\r\n }\r\n static registerDeleted(tracker: ITrackedClass) {\r\n deletedItems.add(tracker);\r\n }\r\n\r\n static get hasChanges() {\r\n return changedItems.size > 0 || deletedItems.size > 0;\r\n }\r\n static getChangedNodes() {\r\n return [...changedItems.values()];\r\n }\r\n static getDeletedNodes() {\r\n return [...deletedItems.values()];\r\n }\r\n\r\n static nodeDeletionProcessed(tracker: ITrackedClass) {\r\n //when a node is deleted, we remove it from all the maps\r\n changedItems.delete(tracker);\r\n deletedItems.delete(tracker);\r\n }\r\n\r\n static nodeChangesProcessed(tracker: ITrackedClass) {\r\n //when node changes have been processed, we only remove it from the changedItems map\r\n changedItems.delete(tracker);\r\n }\r\n}\r\n","/*! *****************************************************************************\r\nCopyright (c) Microsoft Corporation.\r\n\r\nPermission to use, copy, modify, and/or distribute this software for any\r\npurpose with or without fee is hereby granted.\r\n\r\nTHE SOFTWARE IS PROVIDED \"AS IS\" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH\r\nREGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY\r\nAND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,\r\nINDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM\r\nLOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR\r\nOTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR\r\nPERFORMANCE OF THIS SOFTWARE.\r\n***************************************************************************** */\r\n/* global Reflect, Promise */\r\n\r\nvar extendStatics = function(d, b) {\r\n extendStatics = Object.setPrototypeOf ||\r\n ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||\r\n function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; };\r\n return extendStatics(d, b);\r\n};\r\n\r\nexport function __extends(d, b) {\r\n extendStatics(d, b);\r\n function __() { this.constructor = d; }\r\n d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());\r\n}\r\n\r\nexport var __assign = function() {\r\n __assign = Object.assign || function __assign(t) {\r\n for (var s, i = 1, n = arguments.length; i < n; i++) {\r\n s = arguments[i];\r\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p];\r\n }\r\n return t;\r\n }\r\n return __assign.apply(this, arguments);\r\n}\r\n\r\nexport function __rest(s, e) {\r\n var t = {};\r\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)\r\n t[p] = s[p];\r\n if (s != null && typeof Object.getOwnPropertySymbols === \"function\")\r\n for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {\r\n if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))\r\n t[p[i]] = s[p[i]];\r\n }\r\n return t;\r\n}\r\n\r\nexport function __decorate(decorators, target, key, desc) {\r\n var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;\r\n if (typeof Reflect === \"object\" && typeof Reflect.decorate === \"function\") r = Reflect.decorate(decorators, target, key, desc);\r\n else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;\r\n return c > 3 && r && Object.defineProperty(target, key, r), r;\r\n}\r\n\r\nexport function __param(paramIndex, decorator) {\r\n return function (target, key) { decorator(target, key, paramIndex); }\r\n}\r\n\r\nexport function __metadata(metadataKey, metadataValue) {\r\n if (typeof Reflect === \"object\" && typeof Reflect.metadata === \"function\") return Reflect.metadata(metadataKey, metadataValue);\r\n}\r\n\r\nexport function __awaiter(thisArg, _arguments, P, generator) {\r\n function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\r\n return new (P || (P = Promise))(function (resolve, reject) {\r\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\r\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\r\n function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\r\n step((generator = generator.apply(thisArg, _arguments || [])).next());\r\n });\r\n}\r\n\r\nexport function __generator(thisArg, body) {\r\n var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;\r\n return g = { next: verb(0), \"throw\": verb(1), \"return\": verb(2) }, typeof Symbol === \"function\" && (g[Symbol.iterator] = function() { return this; }), g;\r\n function verb(n) { return function (v) { return step([n, v]); }; }\r\n function step(op) {\r\n if (f) throw new TypeError(\"Generator is already executing.\");\r\n while (_) try {\r\n if (f = 1, y && (t = op[0] & 2 ? y[\"return\"] : op[0] ? y[\"throw\"] || ((t = y[\"return\"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;\r\n if (y = 0, t) op = [op[0] & 2, t.value];\r\n switch (op[0]) {\r\n case 0: case 1: t = op; break;\r\n case 4: _.label++; return { value: op[1], done: false };\r\n case 5: _.label++; y = op[1]; op = [0]; continue;\r\n case 7: op = _.ops.pop(); _.trys.pop(); continue;\r\n default:\r\n if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }\r\n if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }\r\n if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }\r\n if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }\r\n if (t[2]) _.ops.pop();\r\n _.trys.pop(); continue;\r\n }\r\n op = body.call(thisArg, _);\r\n } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }\r\n if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };\r\n }\r\n}\r\n\r\nexport function __createBinding(o, m, k, k2) {\r\n if (k2 === undefined) k2 = k;\r\n o[k2] = m[k];\r\n}\r\n\r\nexport function __exportStar(m, exports) {\r\n for (var p in m) if (p !== \"default\" && !exports.hasOwnProperty(p)) exports[p] = m[p];\r\n}\r\n\r\nexport function __values(o) {\r\n var s = typeof Symbol === \"function\" && Symbol.iterator, m = s && o[s], i = 0;\r\n if (m) return m.call(o);\r\n if (o && typeof o.length === \"number\") return {\r\n next: function () {\r\n if (o && i >= o.length) o = void 0;\r\n return { value: o && o[i++], done: !o };\r\n }\r\n };\r\n throw new TypeError(s ? \"Object is not iterable.\" : \"Symbol.iterator is not defined.\");\r\n}\r\n\r\nexport function __read(o, n) {\r\n var m = typeof Symbol === \"function\" && o[Symbol.iterator];\r\n if (!m) return o;\r\n var i = m.call(o), r, ar = [], e;\r\n try {\r\n while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value);\r\n }\r\n catch (error) { e = { error: error }; }\r\n finally {\r\n try {\r\n if (r && !r.done && (m = i[\"return\"])) m.call(i);\r\n }\r\n finally { if (e) throw e.error; }\r\n }\r\n return ar;\r\n}\r\n\r\nexport function __spread() {\r\n for (var ar = [], i = 0; i < arguments.length; i++)\r\n ar = ar.concat(__read(arguments[i]));\r\n return ar;\r\n}\r\n\r\nexport function __spreadArrays() {\r\n for (var s = 0, i = 0, il = arguments.length; i < il; i++) s += arguments[i].length;\r\n for (var r = Array(s), k = 0, i = 0; i < il; i++)\r\n for (var a = arguments[i], j = 0, jl = a.length; j < jl; j++, k++)\r\n r[k] = a[j];\r\n return r;\r\n};\r\n\r\nexport function __await(v) {\r\n return this instanceof __await ? (this.v = v, this) : new __await(v);\r\n}\r\n\r\nexport function __asyncGenerator(thisArg, _arguments, generator) {\r\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\r\n var g = generator.apply(thisArg, _arguments || []), i, q = [];\r\n return i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i;\r\n function verb(n) { if (g[n]) i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; }\r\n function resume(n, v) { try { step(g[n](v)); } catch (e) { settle(q[0][3], e); } }\r\n function step(r) { r.value instanceof __await ? Promise.resolve(r.value.v).then(fulfill, reject) : settle(q[0][2], r); }\r\n function fulfill(value) { resume(\"next\", value); }\r\n function reject(value) { resume(\"throw\", value); }\r\n function settle(f, v) { if (f(v), q.shift(), q.length) resume(q[0][0], q[0][1]); }\r\n}\r\n\r\nexport function __asyncDelegator(o) {\r\n var i, p;\r\n return i = {}, verb(\"next\"), verb(\"throw\", function (e) { throw e; }), verb(\"return\"), i[Symbol.iterator] = function () { return this; }, i;\r\n function verb(n, f) { i[n] = o[n] ? function (v) { return (p = !p) ? { value: __await(o[n](v)), done: n === \"return\" } : f ? f(v) : v; } : f; }\r\n}\r\n\r\nexport function __asyncValues(o) {\r\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\r\n var m = o[Symbol.asyncIterator], i;\r\n return m ? m.call(o) : (o = typeof __values === \"function\" ? __values(o) : o[Symbol.iterator](), i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i);\r\n function verb(n) { i[n] = o[n] && function (v) { return new Promise(function (resolve, reject) { v = o[n](v), settle(resolve, reject, v.done, v.value); }); }; }\r\n function settle(resolve, reject, d, v) { Promise.resolve(v).then(function(v) { resolve({ value: v, done: d }); }, reject); }\r\n}\r\n\r\nexport function __makeTemplateObject(cooked, raw) {\r\n if (Object.defineProperty) { Object.defineProperty(cooked, \"raw\", { value: raw }); } else { cooked.raw = raw; }\r\n return cooked;\r\n};\r\n\r\nexport function __importStar(mod) {\r\n if (mod && mod.__esModule) return mod;\r\n var result = {};\r\n if (mod != null) for (var k in mod) if (Object.hasOwnProperty.call(mod, k)) result[k] = mod[k];\r\n result.default = mod;\r\n return result;\r\n}\r\n\r\nexport function __importDefault(mod) {\r\n return (mod && mod.__esModule) ? mod : { default: mod };\r\n}\r\n\r\nexport function __classPrivateFieldGet(receiver, privateMap) {\r\n if (!privateMap.has(receiver)) {\r\n throw new TypeError(\"attempted to get private field on non-instance\");\r\n }\r\n return privateMap.get(receiver);\r\n}\r\n\r\nexport function __classPrivateFieldSet(receiver, privateMap, value) {\r\n if (!privateMap.has(receiver)) {\r\n throw new TypeError(\"attempted to set private field on non-instance\");\r\n }\r\n privateMap.set(receiver, value);\r\n return value;\r\n}\r\n","export function isFunction(x: any): x is Function {\n return typeof x === 'function';\n}\n","let _enable_super_gross_mode_that_will_cause_bad_things = false;\n\n/**\n * The global configuration object for RxJS, used to configure things\n * like what Promise contructor should used to create Promises\n */\nexport const config = {\n /**\n * The promise constructor used by default for methods such as\n * {@link toPromise} and {@link forEach}\n */\n Promise: undefined as PromiseConstructorLike,\n\n /**\n * If true, turns on synchronous error rethrowing, which is a deprecated behavior\n * in v6 and higher. This behavior enables bad patterns like wrapping a subscribe\n * call in a try/catch block. It also enables producer interference, a nasty bug\n * where a multicast can be broken for all observers by a downstream consumer with\n * an unhandled error. DO NOT USE THIS FLAG UNLESS IT'S NEEDED TO BY TIME\n * FOR MIGRATION REASONS.\n */\n set useDeprecatedSynchronousErrorHandling(value: boolean) {\n if (value) {\n const error = new Error();\n console.warn('DEPRECATED! RxJS was set to use deprecated synchronous error handling behavior by code at: \\n' + error.stack);\n } else if (_enable_super_gross_mode_that_will_cause_bad_things) {\n console.log('RxJS: Back to a better error behavior. Thank you. <3');\n }\n _enable_super_gross_mode_that_will_cause_bad_things = value;\n },\n\n get useDeprecatedSynchronousErrorHandling() {\n return _enable_super_gross_mode_that_will_cause_bad_things;\n },\n};\n","/**\n * Throws an error on another job so that it's picked up by the runtime's\n * uncaught error handling mechanism.\n * @param err the error to throw\n */\nexport function hostReportError(err: any) {\n setTimeout(() => { throw err; }, 0);\n}","import { Observer } from './types';\nimport { config } from './config';\nimport { hostReportError } from './util/hostReportError';\n\nexport const empty: Observer = {\n closed: true,\n next(value: any): void { /* noop */},\n error(err: any): void {\n if (config.useDeprecatedSynchronousErrorHandling) {\n throw err;\n } else {\n hostReportError(err);\n }\n },\n complete(): void { /*noop*/ }\n};\n","export function isObject(x: any): x is Object {\n return x !== null && typeof x === 'object';\n}\n","export interface UnsubscriptionError extends Error {\n readonly errors: any[];\n}\n\nexport interface UnsubscriptionErrorCtor {\n new(errors: any[]): UnsubscriptionError;\n}\n\nconst UnsubscriptionErrorImpl = (() => {\n function UnsubscriptionErrorImpl(this: any, errors: any[]) {\n Error.call(this);\n this.message = errors ?\n `${errors.length} errors occurred during unsubscription:\n${errors.map((err, i) => `${i + 1}) ${err.toString()}`).join('\\n ')}` : '';\n this.name = 'UnsubscriptionError';\n this.errors = errors;\n return this;\n }\n\n UnsubscriptionErrorImpl.prototype = Object.create(Error.prototype);\n\n return UnsubscriptionErrorImpl;\n})();\n\n/**\n * An error thrown when one or more errors have occurred during the\n * `unsubscribe` of a {@link Subscription}.\n */\nexport const UnsubscriptionError: UnsubscriptionErrorCtor = UnsubscriptionErrorImpl as any;","import { isArray } from './util/isArray';\nimport { isObject } from './util/isObject';\nimport { isFunction } from './util/isFunction';\nimport { UnsubscriptionError } from './util/UnsubscriptionError';\nimport { SubscriptionLike, TeardownLogic } from './types';\n\n/**\n * Represents a disposable resource, such as the execution of an Observable. A\n * Subscription has one important method, `unsubscribe`, that takes no argument\n * and just disposes the resource held by the subscription.\n *\n * Additionally, subscriptions may be grouped together through the `add()`\n * method, which will attach a child Subscription to the current Subscription.\n * When a Subscription is unsubscribed, all its children (and its grandchildren)\n * will be unsubscribed as well.\n *\n * @class Subscription\n */\nexport class Subscription implements SubscriptionLike {\n /** @nocollapse */\n public static EMPTY: Subscription = (function(empty: any) {\n empty.closed = true;\n return empty;\n }(new Subscription()));\n\n /**\n * A flag to indicate whether this Subscription has already been unsubscribed.\n * @type {boolean}\n */\n public closed: boolean = false;\n\n /** @internal */\n protected _parentOrParents: Subscription | Subscription[] = null;\n /** @internal */\n private _subscriptions: SubscriptionLike[] = null;\n\n /**\n * @param {function(): void} [unsubscribe] A function describing how to\n * perform the disposal of resources when the `unsubscribe` method is called.\n */\n constructor(unsubscribe?: () => void) {\n if (unsubscribe) {\n (this as any)._ctorUnsubscribe = true;\n (this as any)._unsubscribe = unsubscribe;\n }\n }\n\n /**\n * Disposes the resources held by the subscription. May, for instance, cancel\n * an ongoing Observable execution or cancel any other type of work that\n * started when the Subscription was created.\n * @return {void}\n */\n unsubscribe(): void {\n let errors: any[];\n\n if (this.closed) {\n return;\n }\n\n let { _parentOrParents, _ctorUnsubscribe, _unsubscribe, _subscriptions } = (this as any);\n\n this.closed = true;\n this._parentOrParents = null;\n // null out _subscriptions first so any child subscriptions that attempt\n // to remove themselves from this subscription will noop\n this._subscriptions = null;\n\n if (_parentOrParents instanceof Subscription) {\n _parentOrParents.remove(this);\n } else if (_parentOrParents !== null) {\n for (let index = 0; index < _parentOrParents.length; ++index) {\n const parent = _parentOrParents[index];\n parent.remove(this);\n }\n }\n\n if (isFunction(_unsubscribe)) {\n // It's only possible to null _unsubscribe - to release the reference to\n // any teardown function passed in the constructor - if the property was\n // actually assigned in the constructor, as there are some classes that\n // are derived from Subscriber (which derives from Subscription) that\n // implement an _unsubscribe method as a mechanism for obtaining\n // unsubscription notifications and some of those subscribers are\n // recycled. Also, in some of those subscribers, _unsubscribe switches\n // from a prototype method to an instance property - see notifyNext in\n // RetryWhenSubscriber.\n if (_ctorUnsubscribe) {\n (this as any)._unsubscribe = undefined;\n }\n try {\n _unsubscribe.call(this);\n } catch (e) {\n errors = e instanceof UnsubscriptionError ? flattenUnsubscriptionErrors(e.errors) : [e];\n }\n }\n\n if (isArray(_subscriptions)) {\n let index = -1;\n let len = _subscriptions.length;\n\n while (++index < len) {\n const sub = _subscriptions[index];\n if (isObject(sub)) {\n try {\n sub.unsubscribe();\n } catch (e) {\n errors = errors || [];\n if (e instanceof UnsubscriptionError) {\n errors = errors.concat(flattenUnsubscriptionErrors(e.errors));\n } else {\n errors.push(e);\n }\n }\n }\n }\n }\n\n if (errors) {\n throw new UnsubscriptionError(errors);\n }\n }\n\n /**\n * Adds a tear down to be called during the unsubscribe() of this\n * Subscription. Can also be used to add a child subscription.\n *\n * If the tear down being added is a subscription that is already\n * unsubscribed, is the same reference `add` is being called on, or is\n * `Subscription.EMPTY`, it will not be added.\n *\n * If this subscription is already in an `closed` state, the passed\n * tear down logic will be executed immediately.\n *\n * When a parent subscription is unsubscribed, any child subscriptions that were added to it are also unsubscribed.\n *\n * @param {TeardownLogic} teardown The additional logic to execute on\n * teardown.\n * @return {Subscription} Returns the Subscription used or created to be\n * added to the inner subscriptions list. This Subscription can be used with\n * `remove()` to remove the passed teardown logic from the inner subscriptions\n * list.\n */\n add(teardown: TeardownLogic): Subscription {\n let subscription = (teardown);\n\n if (!teardown) {\n return Subscription.EMPTY;\n }\n\n switch (typeof teardown) {\n case 'function':\n subscription = new Subscription(<(() => void)>teardown);\n case 'object':\n if (subscription === this || subscription.closed || typeof subscription.unsubscribe !== 'function') {\n // This also covers the case where `subscription` is `Subscription.EMPTY`, which is always in `closed` state.\n return subscription;\n } else if (this.closed) {\n subscription.unsubscribe();\n return subscription;\n } else if (!(subscription instanceof Subscription)) {\n const tmp = subscription;\n subscription = new Subscription();\n subscription._subscriptions = [tmp];\n }\n break;\n default: {\n throw new Error('unrecognized teardown ' + teardown + ' added to Subscription.');\n }\n }\n\n // Add `this` as parent of `subscription` if that's not already the case.\n let { _parentOrParents } = subscription;\n if (_parentOrParents === null) {\n // If we don't have a parent, then set `subscription._parents` to\n // the `this`, which is the common case that we optimize for.\n subscription._parentOrParents = this;\n } else if (_parentOrParents instanceof Subscription) {\n if (_parentOrParents === this) {\n // The `subscription` already has `this` as a parent.\n return subscription;\n }\n // If there's already one parent, but not multiple, allocate an\n // Array to store the rest of the parent Subscriptions.\n subscription._parentOrParents = [_parentOrParents, this];\n } else if (_parentOrParents.indexOf(this) === -1) {\n // Only add `this` to the _parentOrParents list if it's not already there.\n _parentOrParents.push(this);\n } else {\n // The `subscription` already has `this` as a parent.\n return subscription;\n }\n\n // Optimize for the common case when adding the first subscription.\n const subscriptions = this._subscriptions;\n if (subscriptions === null) {\n this._subscriptions = [subscription];\n } else {\n subscriptions.push(subscription);\n }\n\n return subscription;\n }\n\n /**\n * Removes a Subscription from the internal list of subscriptions that will\n * unsubscribe during the unsubscribe process of this Subscription.\n * @param {Subscription} subscription The subscription to remove.\n * @return {void}\n */\n remove(subscription: Subscription): void {\n const subscriptions = this._subscriptions;\n if (subscriptions) {\n const subscriptionIndex = subscriptions.indexOf(subscription);\n if (subscriptionIndex !== -1) {\n subscriptions.splice(subscriptionIndex, 1);\n }\n }\n }\n}\n\nfunction flattenUnsubscriptionErrors(errors: any[]) {\n return errors.reduce((errs, err) => errs.concat((err instanceof UnsubscriptionError) ? err.errors : err), []);\n}\n","/** @deprecated do not use, this is no longer checked by RxJS internals */\nexport const rxSubscriber = (() =>\n typeof Symbol === 'function'\n ? Symbol('rxSubscriber')\n : '@@rxSubscriber_' + Math.random())();\n\n/**\n * @deprecated use rxSubscriber instead\n */\nexport const $$rxSubscriber = rxSubscriber;\n","import { isFunction } from './util/isFunction';\nimport { empty as emptyObserver } from './Observer';\nimport { Observer, PartialObserver, TeardownLogic } from './types';\nimport { Subscription } from './Subscription';\nimport { rxSubscriber as rxSubscriberSymbol } from '../internal/symbol/rxSubscriber';\nimport { config } from './config';\nimport { hostReportError } from './util/hostReportError';\n\n/**\n * Implements the {@link Observer} interface and extends the\n * {@link Subscription} class. While the {@link Observer} is the public API for\n * consuming the values of an {@link Observable}, all Observers get converted to\n * a Subscriber, in order to provide Subscription-like capabilities such as\n * `unsubscribe`. Subscriber is a common type in RxJS, and crucial for\n * implementing operators, but it is rarely used as a public API.\n *\n * @class Subscriber\n */\nexport class Subscriber extends Subscription implements Observer {\n\n [rxSubscriberSymbol]() { return this; }\n\n /**\n * A static factory for a Subscriber, given a (potentially partial) definition\n * of an Observer.\n * @param {function(x: ?T): void} [next] The `next` callback of an Observer.\n * @param {function(e: ?any): void} [error] The `error` callback of an\n * Observer.\n * @param {function(): void} [complete] The `complete` callback of an\n * Observer.\n * @return {Subscriber} A Subscriber wrapping the (partially defined)\n * Observer represented by the given arguments.\n * @nocollapse\n */\n static create(next?: (x?: T) => void,\n error?: (e?: any) => void,\n complete?: () => void): Subscriber {\n const subscriber = new Subscriber(next, error, complete);\n subscriber.syncErrorThrowable = false;\n return subscriber;\n }\n\n /** @internal */ syncErrorValue: any = null;\n /** @internal */ syncErrorThrown: boolean = false;\n /** @internal */ syncErrorThrowable: boolean = false;\n\n protected isStopped: boolean = false;\n protected destination: PartialObserver | Subscriber; // this `any` is the escape hatch to erase extra type param (e.g. R)\n\n /**\n * @param {Observer|function(value: T): void} [destinationOrNext] A partially\n * defined Observer or a `next` callback function.\n * @param {function(e: ?any): void} [error] The `error` callback of an\n * Observer.\n * @param {function(): void} [complete] The `complete` callback of an\n * Observer.\n */\n constructor(destinationOrNext?: PartialObserver | ((value: T) => void),\n error?: (e?: any) => void,\n complete?: () => void) {\n super();\n\n switch (arguments.length) {\n case 0:\n this.destination = emptyObserver;\n break;\n case 1:\n if (!destinationOrNext) {\n this.destination = emptyObserver;\n break;\n }\n if (typeof destinationOrNext === 'object') {\n if (destinationOrNext instanceof Subscriber) {\n this.syncErrorThrowable = destinationOrNext.syncErrorThrowable;\n this.destination = destinationOrNext;\n destinationOrNext.add(this);\n } else {\n this.syncErrorThrowable = true;\n this.destination = new SafeSubscriber(this, > destinationOrNext);\n }\n break;\n }\n default:\n this.syncErrorThrowable = true;\n this.destination = new SafeSubscriber(this, <((value: T) => void)> destinationOrNext, error, complete);\n break;\n }\n }\n\n /**\n * The {@link Observer} callback to receive notifications of type `next` from\n * the Observable, with a value. The Observable may call this method 0 or more\n * times.\n * @param {T} [value] The `next` value.\n * @return {void}\n */\n next(value?: T): void {\n if (!this.isStopped) {\n this._next(value);\n }\n }\n\n /**\n * The {@link Observer} callback to receive notifications of type `error` from\n * the Observable, with an attached `Error`. Notifies the Observer that\n * the Observable has experienced an error condition.\n * @param {any} [err] The `error` exception.\n * @return {void}\n */\n error(err?: any): void {\n if (!this.isStopped) {\n this.isStopped = true;\n this._error(err);\n }\n }\n\n /**\n * The {@link Observer} callback to receive a valueless notification of type\n * `complete` from the Observable. Notifies the Observer that the Observable\n * has finished sending push-based notifications.\n * @return {void}\n */\n complete(): void {\n if (!this.isStopped) {\n this.isStopped = true;\n this._complete();\n }\n }\n\n unsubscribe(): void {\n if (this.closed) {\n return;\n }\n this.isStopped = true;\n super.unsubscribe();\n }\n\n protected _next(value: T): void {\n this.destination.next(value);\n }\n\n protected _error(err: any): void {\n this.destination.error(err);\n this.unsubscribe();\n }\n\n protected _complete(): void {\n this.destination.complete();\n this.unsubscribe();\n }\n\n /** @deprecated This is an internal implementation detail, do not use. */\n _unsubscribeAndRecycle(): Subscriber {\n const { _parentOrParents } = this;\n this._parentOrParents = null;\n this.unsubscribe();\n this.closed = false;\n this.isStopped = false;\n this._parentOrParents = _parentOrParents;\n return this;\n }\n}\n\n/**\n * We need this JSDoc comment for affecting ESDoc.\n * @ignore\n * @extends {Ignored}\n */\nexport class SafeSubscriber extends Subscriber {\n\n private _context: any;\n\n constructor(private _parentSubscriber: Subscriber,\n observerOrNext?: PartialObserver | ((value: T) => void),\n error?: (e?: any) => void,\n complete?: () => void) {\n super();\n\n let next: ((value: T) => void);\n let context: any = this;\n\n if (isFunction(observerOrNext)) {\n next = (<((value: T) => void)> observerOrNext);\n } else if (observerOrNext) {\n next = (> observerOrNext).next;\n error = (> observerOrNext).error;\n complete = (> observerOrNext).complete;\n if (observerOrNext !== emptyObserver) {\n context = Object.create(observerOrNext);\n if (isFunction(context.unsubscribe)) {\n this.add(<() => void> context.unsubscribe.bind(context));\n }\n context.unsubscribe = this.unsubscribe.bind(this);\n }\n }\n\n this._context = context;\n this._next = next;\n this._error = error;\n this._complete = complete;\n }\n\n next(value?: T): void {\n if (!this.isStopped && this._next) {\n const { _parentSubscriber } = this;\n if (!config.useDeprecatedSynchronousErrorHandling || !_parentSubscriber.syncErrorThrowable) {\n this.__tryOrUnsub(this._next, value);\n } else if (this.__tryOrSetError(_parentSubscriber, this._next, value)) {\n this.unsubscribe();\n }\n }\n }\n\n error(err?: any): void {\n if (!this.isStopped) {\n const { _parentSubscriber } = this;\n const { useDeprecatedSynchronousErrorHandling } = config;\n if (this._error) {\n if (!useDeprecatedSynchronousErrorHandling || !_parentSubscriber.syncErrorThrowable) {\n this.__tryOrUnsub(this._error, err);\n this.unsubscribe();\n } else {\n this.__tryOrSetError(_parentSubscriber, this._error, err);\n this.unsubscribe();\n }\n } else if (!_parentSubscriber.syncErrorThrowable) {\n this.unsubscribe();\n if (useDeprecatedSynchronousErrorHandling) {\n throw err;\n }\n hostReportError(err);\n } else {\n if (useDeprecatedSynchronousErrorHandling) {\n _parentSubscriber.syncErrorValue = err;\n _parentSubscriber.syncErrorThrown = true;\n } else {\n hostReportError(err);\n }\n this.unsubscribe();\n }\n }\n }\n\n complete(): void {\n if (!this.isStopped) {\n const { _parentSubscriber } = this;\n if (this._complete) {\n const wrappedComplete = () => this._complete.call(this._context);\n\n if (!config.useDeprecatedSynchronousErrorHandling || !_parentSubscriber.syncErrorThrowable) {\n this.__tryOrUnsub(wrappedComplete);\n this.unsubscribe();\n } else {\n this.__tryOrSetError(_parentSubscriber, wrappedComplete);\n this.unsubscribe();\n }\n } else {\n this.unsubscribe();\n }\n }\n }\n\n private __tryOrUnsub(fn: Function, value?: any): void {\n try {\n fn.call(this._context, value);\n } catch (err) {\n this.unsubscribe();\n if (config.useDeprecatedSynchronousErrorHandling) {\n throw err;\n } else {\n hostReportError(err);\n }\n }\n }\n\n private __tryOrSetError(parent: Subscriber, fn: Function, value?: any): boolean {\n if (!config.useDeprecatedSynchronousErrorHandling) {\n throw new Error('bad call');\n }\n try {\n fn.call(this._context, value);\n } catch (err) {\n if (config.useDeprecatedSynchronousErrorHandling) {\n parent.syncErrorValue = err;\n parent.syncErrorThrown = true;\n return true;\n } else {\n hostReportError(err);\n return true;\n }\n }\n return false;\n }\n\n /** @internal This is an internal implementation detail, do not use. */\n _unsubscribe(): void {\n const { _parentSubscriber } = this;\n this._context = null;\n this._parentSubscriber = null;\n _parentSubscriber.unsubscribe();\n }\n}\n","export function identity(x: T): T {\n return x;\n}\n","import { noop } from './noop';\nimport { identity } from './identity';\nimport { UnaryFunction } from '../types';\n\n/* tslint:disable:max-line-length */\nexport function pipe(): UnaryFunction;\nexport function pipe(fn1: UnaryFunction): UnaryFunction;\nexport function pipe(fn1: UnaryFunction, fn2: UnaryFunction): UnaryFunction;\nexport function pipe(fn1: UnaryFunction, fn2: UnaryFunction, fn3: UnaryFunction): UnaryFunction;\nexport function pipe(fn1: UnaryFunction, fn2: UnaryFunction, fn3: UnaryFunction, fn4: UnaryFunction): UnaryFunction;\nexport function pipe(fn1: UnaryFunction, fn2: UnaryFunction, fn3: UnaryFunction, fn4: UnaryFunction, fn5: UnaryFunction): UnaryFunction;\nexport function pipe(fn1: UnaryFunction, fn2: UnaryFunction, fn3: UnaryFunction, fn4: UnaryFunction, fn5: UnaryFunction, fn6: UnaryFunction): UnaryFunction;\nexport function pipe(fn1: UnaryFunction, fn2: UnaryFunction, fn3: UnaryFunction, fn4: UnaryFunction, fn5: UnaryFunction, fn6: UnaryFunction, fn7: UnaryFunction): UnaryFunction;\nexport function pipe(fn1: UnaryFunction, fn2: UnaryFunction, fn3: UnaryFunction, fn4: UnaryFunction, fn5: UnaryFunction, fn6: UnaryFunction, fn7: UnaryFunction, fn8: UnaryFunction): UnaryFunction;\nexport function pipe(fn1: UnaryFunction, fn2: UnaryFunction, fn3: UnaryFunction, fn4: UnaryFunction, fn5: UnaryFunction, fn6: UnaryFunction, fn7: UnaryFunction, fn8: UnaryFunction, fn9: UnaryFunction): UnaryFunction;\nexport function pipe(fn1: UnaryFunction, fn2: UnaryFunction, fn3: UnaryFunction, fn4: UnaryFunction, fn5: UnaryFunction, fn6: UnaryFunction, fn7: UnaryFunction, fn8: UnaryFunction, fn9: UnaryFunction, ...fns: UnaryFunction[]): UnaryFunction;\n/* tslint:enable:max-line-length */\n\nexport function pipe(...fns: Array>): UnaryFunction {\n return pipeFromArray(fns);\n}\n\n/** @internal */\nexport function pipeFromArray(fns: Array>): UnaryFunction {\n if (fns.length === 0) {\n return identity as UnaryFunction;\n }\n\n if (fns.length === 1) {\n return fns[0];\n }\n\n return function piped(input: T): R {\n return fns.reduce((prev: any, fn: UnaryFunction) => fn(prev), input as any);\n };\n}\n","import { Operator } from './Operator';\nimport { Subscriber } from './Subscriber';\nimport { Subscription } from './Subscription';\nimport { TeardownLogic, OperatorFunction, PartialObserver, Subscribable } from './types';\nimport { canReportError } from './util/canReportError';\nimport { toSubscriber } from './util/toSubscriber';\nimport { iif } from './observable/iif';\nimport { throwError } from './observable/throwError';\nimport { observable as Symbol_observable } from './symbol/observable';\nimport { pipeFromArray } from './util/pipe';\nimport { config } from './config';\n\n/**\n * A representation of any set of values over any amount of time. This is the most basic building block\n * of RxJS.\n *\n * @class Observable\n */\nexport class Observable implements Subscribable {\n\n /** Internal implementation detail, do not use directly. */\n public _isScalar: boolean = false;\n\n /** @deprecated This is an internal implementation detail, do not use. */\n source: Observable;\n\n /** @deprecated This is an internal implementation detail, do not use. */\n operator: Operator;\n\n /**\n * @constructor\n * @param {Function} subscribe the function that is called when the Observable is\n * initially subscribed to. This function is given a Subscriber, to which new values\n * can be `next`ed, or an `error` method can be called to raise an error, or\n * `complete` can be called to notify of a successful completion.\n */\n constructor(subscribe?: (this: Observable, subscriber: Subscriber) => TeardownLogic) {\n if (subscribe) {\n this._subscribe = subscribe;\n }\n }\n\n // HACK: Since TypeScript inherits static properties too, we have to\n // fight against TypeScript here so Subject can have a different static create signature\n /**\n * Creates a new cold Observable by calling the Observable constructor\n * @static true\n * @owner Observable\n * @method create\n * @param {Function} subscribe? the subscriber function to be passed to the Observable constructor\n * @return {Observable} a new cold observable\n * @nocollapse\n * @deprecated use new Observable() instead\n */\n static create: Function = (subscribe?: (subscriber: Subscriber) => TeardownLogic) => {\n return new Observable(subscribe);\n }\n\n /**\n * Creates a new Observable, with this Observable as the source, and the passed\n * operator defined as the new observable's operator.\n * @method lift\n * @param {Operator} operator the operator defining the operation to take on the observable\n * @return {Observable} a new observable with the Operator applied\n */\n lift(operator: Operator): Observable {\n const observable = new Observable();\n observable.source = this;\n observable.operator = operator;\n return observable;\n }\n\n subscribe(observer?: PartialObserver): Subscription;\n /** @deprecated Use an observer instead of a complete callback */\n subscribe(next: null | undefined, error: null | undefined, complete: () => void): Subscription;\n /** @deprecated Use an observer instead of an error callback */\n subscribe(next: null | undefined, error: (error: any) => void, complete?: () => void): Subscription;\n /** @deprecated Use an observer instead of a complete callback */\n subscribe(next: (value: T) => void, error: null | undefined, complete: () => void): Subscription;\n subscribe(next?: (value: T) => void, error?: (error: any) => void, complete?: () => void): Subscription;\n /**\n * Invokes an execution of an Observable and registers Observer handlers for notifications it will emit.\n *\n * Use it when you have all these Observables, but still nothing is happening.\n *\n * `subscribe` is not a regular operator, but a method that calls Observable's internal `subscribe` function. It\n * might be for example a function that you passed to Observable's constructor, but most of the time it is\n * a library implementation, which defines what will be emitted by an Observable, and when it be will emitted. This means\n * that calling `subscribe` is actually the moment when Observable starts its work, not when it is created, as it is often\n * the thought.\n *\n * Apart from starting the execution of an Observable, this method allows you to listen for values\n * that an Observable emits, as well as for when it completes or errors. You can achieve this in two\n * of the following ways.\n *\n * The first way is creating an object that implements {@link Observer} interface. It should have methods\n * defined by that interface, but note that it should be just a regular JavaScript object, which you can create\n * yourself in any way you want (ES6 class, classic function constructor, object literal etc.). In particular do\n * not attempt to use any RxJS implementation details to create Observers - you don't need them. Remember also\n * that your object does not have to implement all methods. If you find yourself creating a method that doesn't\n * do anything, you can simply omit it. Note however, if the `error` method is not provided, all errors will\n * be left uncaught.\n *\n * The second way is to give up on Observer object altogether and simply provide callback functions in place of its methods.\n * This means you can provide three functions as arguments to `subscribe`, where the first function is equivalent\n * of a `next` method, the second of an `error` method and the third of a `complete` method. Just as in case of Observer,\n * if you do not need to listen for something, you can omit a function, preferably by passing `undefined` or `null`,\n * since `subscribe` recognizes these functions by where they were placed in function call. When it comes\n * to `error` function, just as before, if not provided, errors emitted by an Observable will be thrown.\n *\n * Whichever style of calling `subscribe` you use, in both cases it returns a Subscription object.\n * This object allows you to call `unsubscribe` on it, which in turn will stop the work that an Observable does and will clean\n * up all resources that an Observable used. Note that cancelling a subscription will not call `complete` callback\n * provided to `subscribe` function, which is reserved for a regular completion signal that comes from an Observable.\n *\n * Remember that callbacks provided to `subscribe` are not guaranteed to be called asynchronously.\n * It is an Observable itself that decides when these functions will be called. For example {@link of}\n * by default emits all its values synchronously. Always check documentation for how given Observable\n * will behave when subscribed and if its default behavior can be modified with a `scheduler`.\n *\n * ## Example\n * ### Subscribe with an Observer\n * ```ts\n * import { of } from 'rxjs';\n *\n * const sumObserver = {\n * sum: 0,\n * next(value) {\n * console.log('Adding: ' + value);\n * this.sum = this.sum + value;\n * },\n * error() {\n * // We actually could just remove this method,\n * // since we do not really care about errors right now.\n * },\n * complete() {\n * console.log('Sum equals: ' + this.sum);\n * }\n * };\n *\n * of(1, 2, 3) // Synchronously emits 1, 2, 3 and then completes.\n * .subscribe(sumObserver);\n *\n * // Logs:\n * // \"Adding: 1\"\n * // \"Adding: 2\"\n * // \"Adding: 3\"\n * // \"Sum equals: 6\"\n * ```\n *\n * ### Subscribe with functions\n * ```ts\n * import { of } from 'rxjs'\n *\n * let sum = 0;\n *\n * of(1, 2, 3).subscribe(\n * value => {\n * console.log('Adding: ' + value);\n * sum = sum + value;\n * },\n * undefined,\n * () => console.log('Sum equals: ' + sum)\n * );\n *\n * // Logs:\n * // \"Adding: 1\"\n * // \"Adding: 2\"\n * // \"Adding: 3\"\n * // \"Sum equals: 6\"\n * ```\n *\n * ### Cancel a subscription\n * ```ts\n * import { interval } from 'rxjs';\n *\n * const subscription = interval(1000).subscribe(\n * num => console.log(num),\n * undefined,\n * () => {\n * // Will not be called, even when cancelling subscription.\n * console.log('completed!');\n * }\n * );\n *\n * setTimeout(() => {\n * subscription.unsubscribe();\n * console.log('unsubscribed!');\n * }, 2500);\n *\n * // Logs:\n * // 0 after 1s\n * // 1 after 2s\n * // \"unsubscribed!\" after 2.5s\n * ```\n *\n * @param {Observer|Function} observerOrNext (optional) Either an observer with methods to be called,\n * or the first of three possible handlers, which is the handler for each value emitted from the subscribed\n * Observable.\n * @param {Function} error (optional) A handler for a terminal event resulting from an error. If no error handler is provided,\n * the error will be thrown as unhandled.\n * @param {Function} complete (optional) A handler for a terminal event resulting from successful completion.\n * @return {ISubscription} a subscription reference to the registered handlers\n * @method subscribe\n */\n subscribe(observerOrNext?: PartialObserver | ((value: T) => void),\n error?: (error: any) => void,\n complete?: () => void): Subscription {\n\n const { operator } = this;\n const sink = toSubscriber(observerOrNext, error, complete);\n\n if (operator) {\n sink.add(operator.call(sink, this.source));\n } else {\n sink.add(\n this.source || (config.useDeprecatedSynchronousErrorHandling && !sink.syncErrorThrowable) ?\n this._subscribe(sink) :\n this._trySubscribe(sink)\n );\n }\n\n if (config.useDeprecatedSynchronousErrorHandling) {\n if (sink.syncErrorThrowable) {\n sink.syncErrorThrowable = false;\n if (sink.syncErrorThrown) {\n throw sink.syncErrorValue;\n }\n }\n }\n\n return sink;\n }\n\n /** @deprecated This is an internal implementation detail, do not use. */\n _trySubscribe(sink: Subscriber): TeardownLogic {\n try {\n return this._subscribe(sink);\n } catch (err) {\n if (config.useDeprecatedSynchronousErrorHandling) {\n sink.syncErrorThrown = true;\n sink.syncErrorValue = err;\n }\n if (canReportError(sink)) {\n sink.error(err);\n } else {\n console.warn(err);\n }\n }\n }\n\n /**\n * @method forEach\n * @param {Function} next a handler for each value emitted by the observable\n * @param {PromiseConstructor} [promiseCtor] a constructor function used to instantiate the Promise\n * @return {Promise} a promise that either resolves on observable completion or\n * rejects with the handled error\n */\n forEach(next: (value: T) => void, promiseCtor?: PromiseConstructorLike): Promise {\n promiseCtor = getPromiseCtor(promiseCtor);\n\n return new promiseCtor((resolve, reject) => {\n // Must be declared in a separate statement to avoid a ReferenceError when\n // accessing subscription below in the closure due to Temporal Dead Zone.\n let subscription: Subscription;\n subscription = this.subscribe((value) => {\n try {\n next(value);\n } catch (err) {\n reject(err);\n if (subscription) {\n subscription.unsubscribe();\n }\n }\n }, reject, resolve);\n }) as Promise;\n }\n\n /** @internal This is an internal implementation detail, do not use. */\n _subscribe(subscriber: Subscriber): TeardownLogic {\n const { source } = this;\n return source && source.subscribe(subscriber);\n }\n\n // `if` and `throw` are special snow flakes, the compiler sees them as reserved words. Deprecated in\n // favor of iif and throwError functions.\n /**\n * @nocollapse\n * @deprecated In favor of iif creation function: import { iif } from 'rxjs';\n */\n static if: typeof iif;\n /**\n * @nocollapse\n * @deprecated In favor of throwError creation function: import { throwError } from 'rxjs';\n */\n static throw: typeof throwError;\n\n /**\n * An interop point defined by the es7-observable spec https://github.com/zenparsing/es-observable\n * @method Symbol.observable\n * @return {Observable} this instance of the observable\n */\n [Symbol_observable]() {\n return this;\n }\n\n /* tslint:disable:max-line-length */\n pipe(): Observable;\n pipe(op1: OperatorFunction): Observable;\n pipe(op1: OperatorFunction, op2: OperatorFunction): Observable;\n pipe(op1: OperatorFunction, op2: OperatorFunction, op3: OperatorFunction): Observable;\n pipe(op1: OperatorFunction, op2: OperatorFunction, op3: OperatorFunction, op4: OperatorFunction): Observable;\n pipe(op1: OperatorFunction, op2: OperatorFunction, op3: OperatorFunction, op4: OperatorFunction, op5: OperatorFunction): Observable;\n pipe(op1: OperatorFunction, op2: OperatorFunction, op3: OperatorFunction, op4: OperatorFunction, op5: OperatorFunction, op6: OperatorFunction): Observable;\n pipe(op1: OperatorFunction, op2: OperatorFunction, op3: OperatorFunction, op4: OperatorFunction, op5: OperatorFunction, op6: OperatorFunction, op7: OperatorFunction): Observable;\n pipe(op1: OperatorFunction, op2: OperatorFunction, op3: OperatorFunction, op4: OperatorFunction, op5: OperatorFunction, op6: OperatorFunction, op7: OperatorFunction, op8: OperatorFunction): Observable;\n pipe(op1: OperatorFunction, op2: OperatorFunction, op3: OperatorFunction, op4: OperatorFunction, op5: OperatorFunction, op6: OperatorFunction, op7: OperatorFunction, op8: OperatorFunction, op9: OperatorFunction): Observable;\n pipe(op1: OperatorFunction, op2: OperatorFunction, op3: OperatorFunction, op4: OperatorFunction, op5: OperatorFunction, op6: OperatorFunction, op7: OperatorFunction, op8: OperatorFunction, op9: OperatorFunction, ...operations: OperatorFunction[]): Observable<{}>;\n /* tslint:enable:max-line-length */\n\n /**\n * Used to stitch together functional operators into a chain.\n * @method pipe\n * @return {Observable} the Observable result of all of the operators having\n * been called in the order they were passed in.\n *\n * ### Example\n * ```ts\n * import { interval } from 'rxjs';\n * import { map, filter, scan } from 'rxjs/operators';\n *\n * interval(1000)\n * .pipe(\n * filter(x => x % 2 === 0),\n * map(x => x + x),\n * scan((acc, x) => acc + x)\n * )\n * .subscribe(x => console.log(x))\n * ```\n */\n pipe(...operations: OperatorFunction[]): Observable {\n if (operations.length === 0) {\n return this as any;\n }\n\n return pipeFromArray(operations)(this);\n }\n\n /* tslint:disable:max-line-length */\n toPromise(this: Observable): Promise;\n toPromise(this: Observable, PromiseCtor: typeof Promise): Promise;\n toPromise(this: Observable, PromiseCtor: PromiseConstructorLike): Promise;\n /* tslint:enable:max-line-length */\n\n toPromise(promiseCtor?: PromiseConstructorLike): Promise {\n promiseCtor = getPromiseCtor(promiseCtor);\n\n return new promiseCtor((resolve, reject) => {\n let value: any;\n this.subscribe((x: T) => value = x, (err: any) => reject(err), () => resolve(value));\n }) as Promise;\n }\n}\n\n/**\n * Decides between a passed promise constructor from consuming code,\n * A default configured promise constructor, and the native promise\n * constructor and returns it. If nothing can be found, it will throw\n * an error.\n * @param promiseCtor The optional promise constructor to passed by consuming code\n */\nfunction getPromiseCtor(promiseCtor: PromiseConstructorLike | undefined) {\n if (!promiseCtor) {\n promiseCtor = config.Promise || Promise;\n }\n\n if (!promiseCtor) {\n throw new Error('no Promise impl found');\n }\n\n return promiseCtor;\n}\n","import { Subscriber } from '../Subscriber';\nimport { rxSubscriber as rxSubscriberSymbol } from '../symbol/rxSubscriber';\nimport { empty as emptyObserver } from '../Observer';\nimport { PartialObserver } from '../types';\n\nexport function toSubscriber(\n nextOrObserver?: PartialObserver | ((value: T) => void),\n error?: (error: any) => void,\n complete?: () => void): Subscriber {\n\n if (nextOrObserver) {\n if (nextOrObserver instanceof Subscriber) {\n return (> nextOrObserver);\n }\n\n if (nextOrObserver[rxSubscriberSymbol]) {\n return nextOrObserver[rxSubscriberSymbol]();\n }\n }\n\n if (!nextOrObserver && !error && !complete) {\n return new Subscriber(emptyObserver);\n }\n\n return new Subscriber(nextOrObserver, error, complete);\n}\n","import { Subscriber } from '../Subscriber';\nimport { Subject } from '../Subject';\n\n/**\n * Determines whether the ErrorObserver is closed or stopped or has a\n * destination that is closed or stopped - in which case errors will\n * need to be reported via a different mechanism.\n * @param observer the observer\n */\nexport function canReportError(observer: Subscriber | Subject): boolean {\n while (observer) {\n const { closed, destination, isStopped } = observer as any;\n if (closed || isStopped) {\n return false;\n } else if (destination && destination instanceof Subscriber) {\n observer = destination;\n } else {\n observer = null;\n }\n }\n return true;\n}\n","export interface ObjectUnsubscribedError extends Error {\n}\n\nexport interface ObjectUnsubscribedErrorCtor {\n new(): ObjectUnsubscribedError;\n}\n\nconst ObjectUnsubscribedErrorImpl = (() => {\n function ObjectUnsubscribedErrorImpl(this: any) {\n Error.call(this);\n this.message = 'object unsubscribed';\n this.name = 'ObjectUnsubscribedError';\n return this;\n }\n\n ObjectUnsubscribedErrorImpl.prototype = Object.create(Error.prototype);\n\n return ObjectUnsubscribedErrorImpl;\n})();\n\n/**\n * An error thrown when an action is invalid because the object has been\n * unsubscribed.\n *\n * @see {@link Subject}\n * @see {@link BehaviorSubject}\n *\n * @class ObjectUnsubscribedError\n */\nexport const ObjectUnsubscribedError: ObjectUnsubscribedErrorCtor = ObjectUnsubscribedErrorImpl as any;","import { Subject } from './Subject';\nimport { Observer } from './types';\nimport { Subscription } from './Subscription';\n\n/**\n * We need this JSDoc comment for affecting ESDoc.\n * @ignore\n * @extends {Ignored}\n */\nexport class SubjectSubscription extends Subscription {\n closed: boolean = false;\n\n constructor(public subject: Subject, public subscriber: Observer) {\n super();\n }\n\n unsubscribe() {\n if (this.closed) {\n return;\n }\n\n this.closed = true;\n\n const subject = this.subject;\n const observers = subject.observers;\n\n this.subject = null;\n\n if (!observers || observers.length === 0 || subject.isStopped || subject.closed) {\n return;\n }\n\n const subscriberIndex = observers.indexOf(this.subscriber);\n\n if (subscriberIndex !== -1) {\n observers.splice(subscriberIndex, 1);\n }\n }\n}\n","import { Operator } from './Operator';\nimport { Observable } from './Observable';\nimport { Subscriber } from './Subscriber';\nimport { Subscription } from './Subscription';\nimport { Observer, SubscriptionLike, TeardownLogic } from './types';\nimport { ObjectUnsubscribedError } from './util/ObjectUnsubscribedError';\nimport { SubjectSubscription } from './SubjectSubscription';\nimport { rxSubscriber as rxSubscriberSymbol } from '../internal/symbol/rxSubscriber';\n\n/**\n * @class SubjectSubscriber\n */\nexport class SubjectSubscriber extends Subscriber {\n constructor(protected destination: Subject) {\n super(destination);\n }\n}\n\n/**\n * A Subject is a special type of Observable that allows values to be\n * multicasted to many Observers. Subjects are like EventEmitters.\n *\n * Every Subject is an Observable and an Observer. You can subscribe to a\n * Subject, and you can call next to feed values as well as error and complete.\n *\n * @class Subject\n */\nexport class Subject extends Observable implements SubscriptionLike {\n\n [rxSubscriberSymbol]() {\n return new SubjectSubscriber(this);\n }\n\n observers: Observer[] = [];\n\n closed = false;\n\n isStopped = false;\n\n hasError = false;\n\n thrownError: any = null;\n\n constructor() {\n super();\n }\n\n /**@nocollapse\n * @deprecated use new Subject() instead\n */\n static create: Function = (destination: Observer, source: Observable): AnonymousSubject => {\n return new AnonymousSubject(destination, source);\n }\n\n lift(operator: Operator): Observable {\n const subject = new AnonymousSubject(this, this);\n subject.operator = operator;\n return subject;\n }\n\n next(value?: T) {\n if (this.closed) {\n throw new ObjectUnsubscribedError();\n }\n if (!this.isStopped) {\n const { observers } = this;\n const len = observers.length;\n const copy = observers.slice();\n for (let i = 0; i < len; i++) {\n copy[i].next(value);\n }\n }\n }\n\n error(err: any) {\n if (this.closed) {\n throw new ObjectUnsubscribedError();\n }\n this.hasError = true;\n this.thrownError = err;\n this.isStopped = true;\n const { observers } = this;\n const len = observers.length;\n const copy = observers.slice();\n for (let i = 0; i < len; i++) {\n copy[i].error(err);\n }\n this.observers.length = 0;\n }\n\n complete() {\n if (this.closed) {\n throw new ObjectUnsubscribedError();\n }\n this.isStopped = true;\n const { observers } = this;\n const len = observers.length;\n const copy = observers.slice();\n for (let i = 0; i < len; i++) {\n copy[i].complete();\n }\n this.observers.length = 0;\n }\n\n unsubscribe() {\n this.isStopped = true;\n this.closed = true;\n this.observers = null;\n }\n\n /** @deprecated This is an internal implementation detail, do not use. */\n _trySubscribe(subscriber: Subscriber): TeardownLogic {\n if (this.closed) {\n throw new ObjectUnsubscribedError();\n } else {\n return super._trySubscribe(subscriber);\n }\n }\n\n /** @deprecated This is an internal implementation detail, do not use. */\n _subscribe(subscriber: Subscriber): Subscription {\n if (this.closed) {\n throw new ObjectUnsubscribedError();\n } else if (this.hasError) {\n subscriber.error(this.thrownError);\n return Subscription.EMPTY;\n } else if (this.isStopped) {\n subscriber.complete();\n return Subscription.EMPTY;\n } else {\n this.observers.push(subscriber);\n return new SubjectSubscription(this, subscriber);\n }\n }\n\n /**\n * Creates a new Observable with this Subject as the source. You can do this\n * to create customize Observer-side logic of the Subject and conceal it from\n * code that uses the Observable.\n * @return {Observable} Observable that the Subject casts to\n */\n asObservable(): Observable {\n const observable = new Observable();\n (observable).source = this;\n return observable;\n }\n}\n\n/**\n * @class AnonymousSubject\n */\nexport class AnonymousSubject extends Subject {\n constructor(protected destination?: Observer, source?: Observable) {\n super();\n this.source = source;\n }\n\n next(value: T) {\n const { destination } = this;\n if (destination && destination.next) {\n destination.next(value);\n }\n }\n\n error(err: any) {\n const { destination } = this;\n if (destination && destination.error) {\n this.destination.error(err);\n }\n }\n\n complete() {\n const { destination } = this;\n if (destination && destination.complete) {\n this.destination.complete();\n }\n }\n\n /** @deprecated This is an internal implementation detail, do not use. */\n _subscribe(subscriber: Subscriber): Subscription {\n const { source } = this;\n if (source) {\n return this.source.subscribe(subscriber);\n } else {\n return Subscription.EMPTY;\n }\n }\n}\n","import { Operator } from '../Operator';\nimport { Subscriber } from '../Subscriber';\nimport { Subscription } from '../Subscription';\nimport { MonoTypeOperatorFunction, TeardownLogic } from '../types';\nimport { ConnectableObservable } from '../observable/ConnectableObservable';\nimport { Observable } from '../Observable';\n\n/**\n * Make a {@link ConnectableObservable} behave like a ordinary observable and automates the way\n * you can connect to it.\n *\n * Internally it counts the subscriptions to the observable and subscribes (only once) to the source if\n * the number of subscriptions is larger than 0. If the number of subscriptions is smaller than 1, it\n * unsubscribes from the source. This way you can make sure that everything before the *published*\n * refCount has only a single subscription independently of the number of subscribers to the target\n * observable.\n *\n * Note that using the {@link share} operator is exactly the same as using the *publish* operator\n * (making the observable hot) and the *refCount* operator in a sequence.\n *\n * ![](refCount.png)\n *\n * ## Example\n *\n * In the following example there are two intervals turned into connectable observables\n * by using the *publish* operator. The first one uses the *refCount* operator, the\n * second one does not use it. You will notice that a connectable observable does nothing\n * until you call its connect function.\n *\n * ```ts\n * import { interval } from 'rxjs';\n * import { tap, publish, refCount } from 'rxjs/operators';\n *\n * // Turn the interval observable into a ConnectableObservable (hot)\n * const refCountInterval = interval(400).pipe(\n * tap((num) => console.log(`refCount ${num}`)),\n * publish(),\n * refCount()\n * );\n *\n * const publishedInterval = interval(400).pipe(\n * tap((num) => console.log(`publish ${num}`)),\n * publish()\n * );\n *\n * refCountInterval.subscribe();\n * refCountInterval.subscribe();\n * // 'refCount 0' -----> 'refCount 1' -----> etc\n * // All subscriptions will receive the same value and the tap (and\n * // every other operator) before the publish operator will be executed\n * // only once per event independently of the number of subscriptions.\n *\n * publishedInterval.subscribe();\n * // Nothing happens until you call .connect() on the observable.\n * ```\n *\n * @see {@link ConnectableObservable}\n * @see {@link share}\n * @see {@link publish}\n */\nexport function refCount(): MonoTypeOperatorFunction {\n return function refCountOperatorFunction(source: ConnectableObservable): Observable {\n return source.lift(new RefCountOperator(source));\n } as MonoTypeOperatorFunction;\n}\n\nclass RefCountOperator implements Operator {\n constructor(private connectable: ConnectableObservable) {\n }\n call(subscriber: Subscriber, source: any): TeardownLogic {\n\n const { connectable } = this;\n ( connectable)._refCount++;\n\n const refCounter = new RefCountSubscriber(subscriber, connectable);\n const subscription = source.subscribe(refCounter);\n\n if (!refCounter.closed) {\n ( refCounter).connection = connectable.connect();\n }\n\n return subscription;\n }\n}\n\nclass RefCountSubscriber extends Subscriber {\n\n private connection: Subscription;\n\n constructor(destination: Subscriber,\n private connectable: ConnectableObservable) {\n super(destination);\n }\n\n protected _unsubscribe() {\n\n const { connectable } = this;\n if (!connectable) {\n this.connection = null;\n return;\n }\n\n this.connectable = null;\n const refCount = ( connectable)._refCount;\n if (refCount <= 0) {\n this.connection = null;\n return;\n }\n\n ( connectable)._refCount = refCount - 1;\n if (refCount > 1) {\n this.connection = null;\n return;\n }\n\n ///\n // Compare the local RefCountSubscriber's connection Subscription to the\n // connection Subscription on the shared ConnectableObservable. In cases\n // where the ConnectableObservable source synchronously emits values, and\n // the RefCountSubscriber's downstream Observers synchronously unsubscribe,\n // execution continues to here before the RefCountOperator has a chance to\n // supply the RefCountSubscriber with the shared connection Subscription.\n // For example:\n // ```\n // range(0, 10).pipe(\n // publish(),\n // refCount(),\n // take(5),\n // )\n // .subscribe();\n // ```\n // In order to account for this case, RefCountSubscriber should only dispose\n // the ConnectableObservable's shared connection Subscription if the\n // connection Subscription exists, *and* either:\n // a. RefCountSubscriber doesn't have a reference to the shared connection\n // Subscription yet, or,\n // b. RefCountSubscriber's connection Subscription reference is identical\n // to the shared connection Subscription\n ///\n const { connection } = this;\n const sharedConnection = ( connectable)._connection;\n this.connection = null;\n\n if (sharedConnection && (!connection || sharedConnection === connection)) {\n sharedConnection.unsubscribe();\n }\n }\n}\n","import { Subject, SubjectSubscriber } from '../Subject';\nimport { Operator } from '../Operator';\nimport { Observable } from '../Observable';\nimport { Subscriber } from '../Subscriber';\nimport { Subscription } from '../Subscription';\nimport { TeardownLogic } from '../types';\nimport { refCount as higherOrderRefCount } from '../operators/refCount';\n\n/**\n * @class ConnectableObservable\n */\nexport class ConnectableObservable extends Observable {\n\n protected _subject: Subject;\n protected _refCount: number = 0;\n protected _connection: Subscription;\n /** @internal */\n _isComplete = false;\n\n constructor(public source: Observable,\n protected subjectFactory: () => Subject) {\n super();\n }\n\n /** @deprecated This is an internal implementation detail, do not use. */\n _subscribe(subscriber: Subscriber) {\n return this.getSubject().subscribe(subscriber);\n }\n\n protected getSubject(): Subject {\n const subject = this._subject;\n if (!subject || subject.isStopped) {\n this._subject = this.subjectFactory();\n }\n return this._subject;\n }\n\n connect(): Subscription {\n let connection = this._connection;\n if (!connection) {\n this._isComplete = false;\n connection = this._connection = new Subscription();\n connection.add(this.source\n .subscribe(new ConnectableSubscriber(this.getSubject(), this)));\n if (connection.closed) {\n this._connection = null;\n connection = Subscription.EMPTY;\n }\n }\n return connection;\n }\n\n refCount(): Observable {\n return higherOrderRefCount()(this) as Observable;\n }\n}\n\nexport const connectableObservableDescriptor: PropertyDescriptorMap = (() => {\n const connectableProto = ConnectableObservable.prototype;\n return {\n operator: { value: null as null },\n _refCount: { value: 0, writable: true },\n _subject: { value: null as null, writable: true },\n _connection: { value: null as null, writable: true },\n _subscribe: { value: connectableProto._subscribe },\n _isComplete: { value: connectableProto._isComplete, writable: true },\n getSubject: { value: connectableProto.getSubject },\n connect: { value: connectableProto.connect },\n refCount: { value: connectableProto.refCount }\n };\n})();\n\nclass ConnectableSubscriber extends SubjectSubscriber {\n constructor(destination: Subject,\n private connectable: ConnectableObservable) {\n super(destination);\n }\n protected _error(err: any): void {\n this._unsubscribe();\n super._error(err);\n }\n protected _complete(): void {\n this.connectable._isComplete = true;\n this._unsubscribe();\n super._complete();\n }\n protected _unsubscribe() {\n const connectable = this.connectable;\n if (connectable) {\n this.connectable = null;\n const connection = connectable._connection;\n connectable._refCount = 0;\n connectable._subject = null;\n connectable._connection = null;\n if (connection) {\n connection.unsubscribe();\n }\n }\n }\n}\n\nclass RefCountOperator implements Operator {\n constructor(private connectable: ConnectableObservable) {\n }\n call(subscriber: Subscriber, source: any): TeardownLogic {\n\n const { connectable } = this;\n ( connectable)._refCount++;\n\n const refCounter = new RefCountSubscriber(subscriber, connectable);\n const subscription = source.subscribe(refCounter);\n\n if (!refCounter.closed) {\n ( refCounter).connection = connectable.connect();\n }\n\n return subscription;\n }\n}\n\nclass RefCountSubscriber extends Subscriber {\n\n private connection: Subscription;\n\n constructor(destination: Subscriber,\n private connectable: ConnectableObservable) {\n super(destination);\n }\n\n protected _unsubscribe() {\n\n const { connectable } = this;\n if (!connectable) {\n this.connection = null;\n return;\n }\n\n this.connectable = null;\n const refCount = ( connectable)._refCount;\n if (refCount <= 0) {\n this.connection = null;\n return;\n }\n\n ( connectable)._refCount = refCount - 1;\n if (refCount > 1) {\n this.connection = null;\n return;\n }\n\n ///\n // Compare the local RefCountSubscriber's connection Subscription to the\n // connection Subscription on the shared ConnectableObservable. In cases\n // where the ConnectableObservable source synchronously emits values, and\n // the RefCountSubscriber's downstream Observers synchronously unsubscribe,\n // execution continues to here before the RefCountOperator has a chance to\n // supply the RefCountSubscriber with the shared connection Subscription.\n // For example:\n // ```\n // range(0, 10).pipe(\n // publish(),\n // refCount(),\n // take(5),\n // ).subscribe();\n // ```\n // In order to account for this case, RefCountSubscriber should only dispose\n // the ConnectableObservable's shared connection Subscription if the\n // connection Subscription exists, *and* either:\n // a. RefCountSubscriber doesn't have a reference to the shared connection\n // Subscription yet, or,\n // b. RefCountSubscriber's connection Subscription reference is identical\n // to the shared connection Subscription\n ///\n const { connection } = this;\n const sharedConnection = ( connectable)._connection;\n this.connection = null;\n\n if (sharedConnection && (!connection || sharedConnection === connection)) {\n sharedConnection.unsubscribe();\n }\n }\n}\n","import { Subject } from './Subject';\nimport { Subscriber } from './Subscriber';\nimport { Subscription } from './Subscription';\nimport { SubscriptionLike } from './types';\nimport { ObjectUnsubscribedError } from './util/ObjectUnsubscribedError';\n\n/**\n * A variant of Subject that requires an initial value and emits its current\n * value whenever it is subscribed to.\n *\n * @class BehaviorSubject\n */\nexport class BehaviorSubject extends Subject {\n\n constructor(private _value: T) {\n super();\n }\n\n get value(): T {\n return this.getValue();\n }\n\n /** @deprecated This is an internal implementation detail, do not use. */\n _subscribe(subscriber: Subscriber): Subscription {\n const subscription = super._subscribe(subscriber);\n if (subscription && !(subscription).closed) {\n subscriber.next(this._value);\n }\n return subscription;\n }\n\n getValue(): T {\n if (this.hasError) {\n throw this.thrownError;\n } else if (this.closed) {\n throw new ObjectUnsubscribedError();\n } else {\n return this._value;\n }\n }\n\n next(value: T): void {\n super.next(this._value = value);\n }\n}\n","import { Action } from './Action';\nimport { SchedulerAction } from '../types';\nimport { Subscription } from '../Subscription';\nimport { AsyncScheduler } from './AsyncScheduler';\n\n/**\n * We need this JSDoc comment for affecting ESDoc.\n * @ignore\n * @extends {Ignored}\n */\nexport class AsyncAction extends Action {\n\n public id: any;\n public state: T;\n public delay: number;\n protected pending: boolean = false;\n\n constructor(protected scheduler: AsyncScheduler,\n protected work: (this: SchedulerAction, state?: T) => void) {\n super(scheduler, work);\n }\n\n public schedule(state?: T, delay: number = 0): Subscription {\n\n if (this.closed) {\n return this;\n }\n\n // Always replace the current state with the new state.\n this.state = state;\n\n const id = this.id;\n const scheduler = this.scheduler;\n\n //\n // Important implementation note:\n //\n // Actions only execute once by default, unless rescheduled from within the\n // scheduled callback. This allows us to implement single and repeat\n // actions via the same code path, without adding API surface area, as well\n // as mimic traditional recursion but across asynchronous boundaries.\n //\n // However, JS runtimes and timers distinguish between intervals achieved by\n // serial `setTimeout` calls vs. a single `setInterval` call. An interval of\n // serial `setTimeout` calls can be individually delayed, which delays\n // scheduling the next `setTimeout`, and so on. `setInterval` attempts to\n // guarantee the interval callback will be invoked more precisely to the\n // interval period, regardless of load.\n //\n // Therefore, we use `setInterval` to schedule single and repeat actions.\n // If the action reschedules itself with the same delay, the interval is not\n // canceled. If the action doesn't reschedule, or reschedules with a\n // different delay, the interval will be canceled after scheduled callback\n // execution.\n //\n if (id != null) {\n this.id = this.recycleAsyncId(scheduler, id, delay);\n }\n\n // Set the pending flag indicating that this action has been scheduled, or\n // has recursively rescheduled itself.\n this.pending = true;\n\n this.delay = delay;\n // If this action has already an async Id, don't request a new one.\n this.id = this.id || this.requestAsyncId(scheduler, this.id, delay);\n\n return this;\n }\n\n protected requestAsyncId(scheduler: AsyncScheduler, id?: any, delay: number = 0): any {\n return setInterval(scheduler.flush.bind(scheduler, this), delay);\n }\n\n protected recycleAsyncId(scheduler: AsyncScheduler, id: any, delay: number = 0): any {\n // If this action is rescheduled with the same delay time, don't clear the interval id.\n if (delay !== null && this.delay === delay && this.pending === false) {\n return id;\n }\n // Otherwise, if the action's delay time is different from the current delay,\n // or the action has been rescheduled before it's executed, clear the interval id\n clearInterval(id);\n return undefined;\n }\n\n /**\n * Immediately executes this action and the `work` it contains.\n * @return {any}\n */\n public execute(state: T, delay: number): any {\n\n if (this.closed) {\n return new Error('executing a cancelled action');\n }\n\n this.pending = false;\n const error = this._execute(state, delay);\n if (error) {\n return error;\n } else if (this.pending === false && this.id != null) {\n // Dequeue if the action didn't reschedule itself. Don't call\n // unsubscribe(), because the action could reschedule later.\n // For example:\n // ```\n // scheduler.schedule(function doWork(counter) {\n // /* ... I'm a busy worker bee ... */\n // var originalAction = this;\n // /* wait 100ms before rescheduling the action */\n // setTimeout(function () {\n // originalAction.schedule(counter + 1);\n // }, 100);\n // }, 1000);\n // ```\n this.id = this.recycleAsyncId(this.scheduler, this.id, null);\n }\n }\n\n protected _execute(state: T, delay: number): any {\n let errored: boolean = false;\n let errorValue: any = undefined;\n try {\n this.work(state);\n } catch (e) {\n errored = true;\n errorValue = !!e && e || new Error(e);\n }\n if (errored) {\n this.unsubscribe();\n return errorValue;\n }\n }\n\n /** @deprecated This is an internal implementation detail, do not use. */\n _unsubscribe() {\n\n const id = this.id;\n const scheduler = this.scheduler;\n const actions = scheduler.actions;\n const index = actions.indexOf(this);\n\n this.work = null;\n this.state = null;\n this.pending = false;\n this.scheduler = null;\n\n if (index !== -1) {\n actions.splice(index, 1);\n }\n\n if (id != null) {\n this.id = this.recycleAsyncId(scheduler, id, null);\n }\n\n this.delay = null;\n }\n}\n","import { Scheduler } from '../Scheduler';\nimport { Subscription } from '../Subscription';\nimport { SchedulerAction } from '../types';\n\n/**\n * A unit of work to be executed in a `scheduler`. An action is typically\n * created from within a {@link SchedulerLike} and an RxJS user does not need to concern\n * themselves about creating and manipulating an Action.\n *\n * ```ts\n * class Action extends Subscription {\n * new (scheduler: Scheduler, work: (state?: T) => void);\n * schedule(state?: T, delay: number = 0): Subscription;\n * }\n * ```\n *\n * @class Action\n */\nexport class Action extends Subscription {\n constructor(scheduler: Scheduler, work: (this: SchedulerAction, state?: T) => void) {\n super();\n }\n /**\n * Schedules this action on its parent {@link SchedulerLike} for execution. May be passed\n * some context object, `state`. May happen at some point in the future,\n * according to the `delay` parameter, if specified.\n * @param {T} [state] Some contextual data that the `work` function uses when\n * called by the Scheduler.\n * @param {number} [delay] Time to wait before executing the work, where the\n * time unit is implicit and defined by the Scheduler.\n * @return {void}\n */\n public schedule(state?: T, delay: number = 0): Subscription {\n return this;\n }\n}\n","import { AsyncAction } from './AsyncAction';\nimport { Subscription } from '../Subscription';\nimport { QueueScheduler } from './QueueScheduler';\nimport { SchedulerAction } from '../types';\n\n/**\n * We need this JSDoc comment for affecting ESDoc.\n * @ignore\n * @extends {Ignored}\n */\nexport class QueueAction extends AsyncAction {\n\n constructor(protected scheduler: QueueScheduler,\n protected work: (this: SchedulerAction, state?: T) => void) {\n super(scheduler, work);\n }\n\n public schedule(state?: T, delay: number = 0): Subscription {\n if (delay > 0) {\n return super.schedule(state, delay);\n }\n this.delay = delay;\n this.state = state;\n this.scheduler.flush(this);\n return this;\n }\n\n public execute(state: T, delay: number): any {\n return (delay > 0 || this.closed) ?\n super.execute(state, delay) :\n this._execute(state, delay) ;\n }\n\n protected requestAsyncId(scheduler: QueueScheduler, id?: any, delay: number = 0): any {\n // If delay exists and is greater than 0, or if the delay is null (the\n // action wasn't rescheduled) but was originally scheduled as an async\n // action, then recycle as an async action.\n if ((delay !== null && delay > 0) || (delay === null && this.delay > 0)) {\n return super.requestAsyncId(scheduler, id, delay);\n }\n // Otherwise flush the scheduler starting with this action.\n return scheduler.flush(this);\n }\n}\n","import { Action } from './scheduler/Action';\nimport { Subscription } from './Subscription';\nimport { SchedulerLike, SchedulerAction } from './types';\n\n/**\n * An execution context and a data structure to order tasks and schedule their\n * execution. Provides a notion of (potentially virtual) time, through the\n * `now()` getter method.\n *\n * Each unit of work in a Scheduler is called an `Action`.\n *\n * ```ts\n * class Scheduler {\n * now(): number;\n * schedule(work, delay?, state?): Subscription;\n * }\n * ```\n *\n * @class Scheduler\n * @deprecated Scheduler is an internal implementation detail of RxJS, and\n * should not be used directly. Rather, create your own class and implement\n * {@link SchedulerLike}\n */\nexport class Scheduler implements SchedulerLike {\n\n /**\n * Note: the extra arrow function wrapper is to make testing by overriding\n * Date.now easier.\n * @nocollapse\n */\n public static now: () => number = () => Date.now();\n\n constructor(private SchedulerAction: typeof Action,\n now: () => number = Scheduler.now) {\n this.now = now;\n }\n\n /**\n * A getter method that returns a number representing the current time\n * (at the time this function was called) according to the scheduler's own\n * internal clock.\n * @return {number} A number that represents the current time. May or may not\n * have a relation to wall-clock time. May or may not refer to a time unit\n * (e.g. milliseconds).\n */\n public now: () => number;\n\n /**\n * Schedules a function, `work`, for execution. May happen at some point in\n * the future, according to the `delay` parameter, if specified. May be passed\n * some context object, `state`, which will be passed to the `work` function.\n *\n * The given arguments will be processed an stored as an Action object in a\n * queue of actions.\n *\n * @param {function(state: ?T): ?Subscription} work A function representing a\n * task, or some unit of work to be executed by the Scheduler.\n * @param {number} [delay] Time to wait before executing the work, where the\n * time unit is implicit and defined by the Scheduler itself.\n * @param {T} [state] Some contextual data that the `work` function uses when\n * called by the Scheduler.\n * @return {Subscription} A subscription in order to be able to unsubscribe\n * the scheduled work.\n */\n public schedule(work: (this: SchedulerAction, state?: T) => void, delay: number = 0, state?: T): Subscription {\n return new this.SchedulerAction(this, work).schedule(state, delay);\n }\n}\n","import { Scheduler } from '../Scheduler';\nimport { Action } from './Action';\nimport { AsyncAction } from './AsyncAction';\nimport { SchedulerAction } from '../types';\nimport { Subscription } from '../Subscription';\n\nexport class AsyncScheduler extends Scheduler {\n public static delegate?: Scheduler;\n public actions: Array> = [];\n /**\n * A flag to indicate whether the Scheduler is currently executing a batch of\n * queued actions.\n * @type {boolean}\n * @deprecated internal use only\n */\n public active: boolean = false;\n /**\n * An internal ID used to track the latest asynchronous task such as those\n * coming from `setTimeout`, `setInterval`, `requestAnimationFrame`, and\n * others.\n * @type {any}\n * @deprecated internal use only\n */\n public scheduled: any = undefined;\n\n constructor(SchedulerAction: typeof Action,\n now: () => number = Scheduler.now) {\n super(SchedulerAction, () => {\n if (AsyncScheduler.delegate && AsyncScheduler.delegate !== this) {\n return AsyncScheduler.delegate.now();\n } else {\n return now();\n }\n });\n }\n\n public schedule(work: (this: SchedulerAction, state?: T) => void, delay: number = 0, state?: T): Subscription {\n if (AsyncScheduler.delegate && AsyncScheduler.delegate !== this) {\n return AsyncScheduler.delegate.schedule(work, delay, state);\n } else {\n return super.schedule(work, delay, state);\n }\n }\n\n public flush(action: AsyncAction): void {\n\n const {actions} = this;\n\n if (this.active) {\n actions.push(action);\n return;\n }\n\n let error: any;\n this.active = true;\n\n do {\n if (error = action.execute(action.state, action.delay)) {\n break;\n }\n } while (action = actions.shift()); // exhaust the scheduler queue\n\n this.active = false;\n\n if (error) {\n while (action = actions.shift()) {\n action.unsubscribe();\n }\n throw error;\n }\n }\n}\n","import { AsyncScheduler } from './AsyncScheduler';\n\nexport class QueueScheduler extends AsyncScheduler {\n}\n","import { QueueAction } from './QueueAction';\nimport { QueueScheduler } from './QueueScheduler';\n\n/**\n *\n * Queue Scheduler\n *\n * Put every next task on a queue, instead of executing it immediately\n *\n * `queue` scheduler, when used with delay, behaves the same as {@link asyncScheduler} scheduler.\n *\n * When used without delay, it schedules given task synchronously - executes it right when\n * it is scheduled. However when called recursively, that is when inside the scheduled task,\n * another task is scheduled with queue scheduler, instead of executing immediately as well,\n * that task will be put on a queue and wait for current one to finish.\n *\n * This means that when you execute task with `queue` scheduler, you are sure it will end\n * before any other task scheduled with that scheduler will start.\n *\n * ## Examples\n * Schedule recursively first, then do something\n * ```ts\n * import { queueScheduler } from 'rxjs';\n *\n * queueScheduler.schedule(() => {\n * queueScheduler.schedule(() => console.log('second')); // will not happen now, but will be put on a queue\n *\n * console.log('first');\n * });\n *\n * // Logs:\n * // \"first\"\n * // \"second\"\n * ```\n *\n * Reschedule itself recursively\n * ```ts\n * import { queueScheduler } from 'rxjs';\n *\n * queueScheduler.schedule(function(state) {\n * if (state !== 0) {\n * console.log('before', state);\n * this.schedule(state - 1); // `this` references currently executing Action,\n * // which we reschedule with new state\n * console.log('after', state);\n * }\n * }, 0, 3);\n *\n * // In scheduler that runs recursively, you would expect:\n * // \"before\", 3\n * // \"before\", 2\n * // \"before\", 1\n * // \"after\", 1\n * // \"after\", 2\n * // \"after\", 3\n *\n * // But with queue it logs:\n * // \"before\", 3\n * // \"after\", 3\n * // \"before\", 2\n * // \"after\", 2\n * // \"before\", 1\n * // \"after\", 1\n * ```\n */\nexport const queueScheduler = new QueueScheduler(QueueAction);\n\n/**\n * @deprecated renamed. Use {@link queueScheduler}\n */\nexport const queue = queueScheduler;\n","import { Observable } from '../Observable';\nimport { SchedulerLike } from '../types';\n\n/**\n * The same Observable instance returned by any call to {@link empty} without a\n * `scheduler`. It is preferrable to use this over `empty()`.\n */\nexport const EMPTY = new Observable(subscriber => subscriber.complete());\n\n/**\n * Creates an Observable that emits no items to the Observer and immediately\n * emits a complete notification.\n *\n * Just emits 'complete', and nothing else.\n * \n *\n * ![](empty.png)\n *\n * This static operator is useful for creating a simple Observable that only\n * emits the complete notification. It can be used for composing with other\n * Observables, such as in a {@link mergeMap}.\n *\n * ## Examples\n * ### Emit the number 7, then complete\n * ```ts\n * import { empty } from 'rxjs';\n * import { startWith } from 'rxjs/operators';\n *\n * const result = empty().pipe(startWith(7));\n * result.subscribe(x => console.log(x));\n * ```\n *\n * ### Map and flatten only odd numbers to the sequence 'a', 'b', 'c'\n * ```ts\n * import { empty, interval, of } from 'rxjs';\n * import { mergeMap } from 'rxjs/operators';\n *\n * const interval$ = interval(1000);\n * const result = interval$.pipe(\n * mergeMap(x => x % 2 === 1 ? of('a', 'b', 'c') : empty()),\n * );\n * result.subscribe(x => console.log(x));\n *\n * // Results in the following to the console:\n * // x is equal to the count on the interval eg(0,1,2,3,...)\n * // x will occur every 1000ms\n * // if x % 2 is equal to 1 print abc\n * // if x % 2 is not equal to 1 nothing will be output\n * ```\n *\n * @see {@link Observable}\n * @see {@link never}\n * @see {@link of}\n * @see {@link throwError}\n *\n * @param scheduler A {@link SchedulerLike} to use for scheduling\n * the emission of the complete notification.\n * @return An \"empty\" Observable: emits only the complete\n * notification.\n * @deprecated Deprecated in favor of using {@link EMPTY} constant, or {@link scheduled} (e.g. `scheduled([], scheduler)`)\n */\nexport function empty(scheduler?: SchedulerLike) {\n return scheduler ? emptyScheduled(scheduler) : EMPTY;\n}\n\nfunction emptyScheduled(scheduler: SchedulerLike) {\n return new Observable(subscriber => scheduler.schedule(() => subscriber.complete()));\n}\n","import { SchedulerLike } from '../types';\n\nexport function isScheduler(value: any): value is SchedulerLike {\n return value && typeof (value).schedule === 'function';\n}\n","import { Subscriber } from '../Subscriber';\n\n/**\n * Subscribes to an ArrayLike with a subscriber\n * @param array The array or array-like to subscribe to\n */\nexport const subscribeToArray = (array: ArrayLike) => (subscriber: Subscriber) => {\n for (let i = 0, len = array.length; i < len && !subscriber.closed; i++) {\n subscriber.next(array[i]);\n }\n subscriber.complete();\n};\n","import { Observable } from '../Observable';\nimport { SchedulerLike } from '../types';\nimport { Subscription } from '../Subscription';\n\nexport function scheduleArray(input: ArrayLike, scheduler: SchedulerLike) {\n return new Observable(subscriber => {\n const sub = new Subscription();\n let i = 0;\n sub.add(scheduler.schedule(function () {\n if (i === input.length) {\n subscriber.complete();\n return;\n }\n subscriber.next(input[i++]);\n if (!subscriber.closed) {\n sub.add(this.schedule());\n }\n }));\n return sub;\n });\n}\n","import { Observable } from '../Observable';\nimport { SchedulerLike } from '../types';\nimport { subscribeToArray } from '../util/subscribeToArray';\nimport { scheduleArray } from '../scheduled/scheduleArray';\n\nexport function fromArray(input: ArrayLike, scheduler?: SchedulerLike) {\n if (!scheduler) {\n return new Observable(subscribeToArray(input));\n } else {\n return scheduleArray(input, scheduler);\n }\n}\n","import { SchedulerLike } from '../types';\nimport { isScheduler } from '../util/isScheduler';\nimport { fromArray } from './fromArray';\nimport { Observable } from '../Observable';\nimport { scheduleArray } from '../scheduled/scheduleArray';\n\n/* tslint:disable:max-line-length */\n/** @deprecated use {@link scheduled} instead `scheduled([a, b, c], scheduler)` */\nexport function of(a: T, scheduler: SchedulerLike): Observable;\n/** @deprecated use {@link scheduled} instead `scheduled([a, b, c], scheduler)` */\nexport function of(a: T, b: T2, scheduler: SchedulerLike): Observable;\n/** @deprecated use {@link scheduled} instead `scheduled([a, b, c], scheduler)` */\nexport function of(a: T, b: T2, c: T3, scheduler: SchedulerLike): Observable;\n/** @deprecated use {@link scheduled} instead `scheduled([a, b, c], scheduler)` */\nexport function of(a: T, b: T2, c: T3, d: T4, scheduler: SchedulerLike): Observable;\n/** @deprecated use {@link scheduled} instead `scheduled([a, b, c], scheduler)` */\nexport function of(a: T, b: T2, c: T3, d: T4, e: T5, scheduler: SchedulerLike): Observable;\n/** @deprecated use {@link scheduled} instead `scheduled([a, b, c], scheduler)` */\nexport function of(a: T, b: T2, c: T3, d: T4, e: T5, f: T6, scheduler: SchedulerLike): Observable;\n/** @deprecated use {@link scheduled} instead `scheduled([a, b, c], scheduler)` */\nexport function of(a: T, b: T2, c: T3, d: T4, e: T5, f: T6, g: T7, scheduler: SchedulerLike):\n Observable;\n/** @deprecated use {@link scheduled} instead `scheduled([a, b, c], scheduler)` */\nexport function of(a: T, b: T2, c: T3, d: T4, e: T5, f: T6, g: T7, h: T8, scheduler: SchedulerLike):\n Observable;\n/** @deprecated use {@link scheduled} instead `scheduled([a, b, c], scheduler)` */\nexport function of(a: T, b: T2, c: T3, d: T4, e: T5, f: T6, g: T7, h: T8, i: T9, scheduler: SchedulerLike):\n Observable;\nexport function of(...args: (T | SchedulerLike)[]): Observable;\n\n// TODO(benlesh): Update the typings for this when we can switch to TS 3.x\nexport function of(a: T): Observable;\nexport function of(a: T, b: T2): Observable;\nexport function of(a: T, b: T2, c: T3): Observable;\nexport function of(a: T, b: T2, c: T3, d: T4): Observable;\nexport function of(a: T, b: T2, c: T3, d: T4, e: T5): Observable;\nexport function of(a: T, b: T2, c: T3, d: T4, e: T5, f: T6): Observable;\nexport function of(a: T, b: T2, c: T3, d: T4, e: T5, f: T6, g: T7):\n Observable;\nexport function of(a: T, b: T2, c: T3, d: T4, e: T5, f: T6, g: T7, h: T8):\n Observable;\nexport function of(a: T, b: T2, c: T3, d: T4, e: T5, f: T6, g: T7, h: T8, i: T9):\n Observable;\nexport function of(...args: T[]): Observable;\n/* tslint:enable:max-line-length */\n\n/**\n * Converts the arguments to an observable sequence.\n *\n * Each argument becomes a `next` notification.\n *\n * ![](of.png)\n *\n * Unlike {@link from}, it does not do any flattening and emits each argument in whole\n * as a separate `next` notification.\n *\n * ## Examples\n *\n * Emit the values `10, 20, 30`\n *\n * ```ts\n * import { of } from 'rxjs';\n *\n * of(10, 20, 30)\n * .subscribe(\n * next => console.log('next:', next),\n * err => console.log('error:', err),\n * () => console.log('the end'),\n * );\n * // result:\n * // 'next: 10'\n * // 'next: 20'\n * // 'next: 30'\n *\n * ```\n *\n * Emit the array `[1,2,3]`\n *\n * ```ts\n * import { of } from 'rxjs';\n *\n * of([1,2,3])\n * .subscribe(\n * next => console.log('next:', next),\n * err => console.log('error:', err),\n * () => console.log('the end'),\n * );\n * // result:\n * // 'next: [1,2,3]'\n * ```\n *\n * @see {@link from}\n * @see {@link range}\n *\n * @param {...T} values A comma separated list of arguments you want to be emitted\n * @return {Observable} An Observable that emits the arguments\n * described above and then completes.\n * @method of\n * @owner Observable\n */\n\nexport function of(...args: Array): Observable {\n let scheduler = args[args.length - 1] as SchedulerLike;\n if (isScheduler(scheduler)) {\n args.pop();\n return scheduleArray(args as T[], scheduler);\n } else {\n return fromArray(args as T[]);\n }\n}\n","import { Observable } from '../Observable';\nimport { SchedulerLike } from '../types';\nimport { Subscriber } from '../Subscriber';\n\n/**\n * Creates an Observable that emits no items to the Observer and immediately\n * emits an error notification.\n *\n * Just emits 'error', and nothing else.\n * \n *\n * ![](throw.png)\n *\n * This static operator is useful for creating a simple Observable that only\n * emits the error notification. It can be used for composing with other\n * Observables, such as in a {@link mergeMap}.\n *\n * ## Examples\n * ### Emit the number 7, then emit an error\n * ```ts\n * import { throwError, concat, of } from 'rxjs';\n *\n * const result = concat(of(7), throwError(new Error('oops!')));\n * result.subscribe(x => console.log(x), e => console.error(e));\n *\n * // Logs:\n * // 7\n * // Error: oops!\n * ```\n *\n * ---\n *\n * ### Map and flatten numbers to the sequence 'a', 'b', 'c', but throw an error for 2\n * ```ts\n * import { throwError, interval, of } from 'rxjs';\n * import { mergeMap } from 'rxjs/operators';\n *\n * interval(1000).pipe(\n * mergeMap(x => x === 2\n * ? throwError('Twos are bad')\n * : of('a', 'b', 'c')\n * ),\n * ).subscribe(x => console.log(x), e => console.error(e));\n *\n * // Logs:\n * // a\n * // b\n * // c\n * // a\n * // b\n * // c\n * // Twos are bad\n * ```\n *\n * @see {@link Observable}\n * @see {@link empty}\n * @see {@link never}\n * @see {@link of}\n *\n * @param {any} error The particular Error to pass to the error notification.\n * @param {SchedulerLike} [scheduler] A {@link SchedulerLike} to use for scheduling\n * the emission of the error notification.\n * @return {Observable} An error Observable: emits only the error notification\n * using the given error argument.\n * @static true\n * @name throwError\n * @owner Observable\n */\nexport function throwError(error: any, scheduler?: SchedulerLike): Observable {\n if (!scheduler) {\n return new Observable(subscriber => subscriber.error(error));\n } else {\n return new Observable(subscriber => scheduler.schedule(dispatch, 0, { error, subscriber }));\n }\n}\n\ninterface DispatchArg {\n error: any;\n subscriber: Subscriber;\n}\n\nfunction dispatch({ error, subscriber }: DispatchArg) {\n subscriber.error(error);\n}\n","import { PartialObserver } from './types';\nimport { Observable } from './Observable';\nimport { empty } from './observable/empty';\nimport { of } from './observable/of';\nimport { throwError } from './observable/throwError';\nimport { deprecate } from 'util';\n\n// TODO: When this enum is removed, replace it with a type alias. See #4556.\n/**\n * @deprecated NotificationKind is deprecated as const enums are not compatible with isolated modules. Use a string literal instead.\n */\nexport enum NotificationKind {\n NEXT = 'N',\n ERROR = 'E',\n COMPLETE = 'C',\n}\n\n/**\n * Represents a push-based event or value that an {@link Observable} can emit.\n * This class is particularly useful for operators that manage notifications,\n * like {@link materialize}, {@link dematerialize}, {@link observeOn}, and\n * others. Besides wrapping the actual delivered value, it also annotates it\n * with metadata of, for instance, what type of push message it is (`next`,\n * `error`, or `complete`).\n *\n * @see {@link materialize}\n * @see {@link dematerialize}\n * @see {@link observeOn}\n *\n * @class Notification\n */\nexport class Notification {\n hasValue: boolean;\n\n constructor(public kind: 'N' | 'E' | 'C', public value?: T, public error?: any) {\n this.hasValue = kind === 'N';\n }\n\n /**\n * Delivers to the given `observer` the value wrapped by this Notification.\n * @param {Observer} observer\n * @return\n */\n observe(observer: PartialObserver): any {\n switch (this.kind) {\n case 'N':\n return observer.next && observer.next(this.value);\n case 'E':\n return observer.error && observer.error(this.error);\n case 'C':\n return observer.complete && observer.complete();\n }\n }\n\n /**\n * Given some {@link Observer} callbacks, deliver the value represented by the\n * current Notification to the correctly corresponding callback.\n * @param {function(value: T): void} next An Observer `next` callback.\n * @param {function(err: any): void} [error] An Observer `error` callback.\n * @param {function(): void} [complete] An Observer `complete` callback.\n * @return {any}\n */\n do(next: (value: T) => void, error?: (err: any) => void, complete?: () => void): any {\n const kind = this.kind;\n switch (kind) {\n case 'N':\n return next && next(this.value);\n case 'E':\n return error && error(this.error);\n case 'C':\n return complete && complete();\n }\n }\n\n /**\n * Takes an Observer or its individual callback functions, and calls `observe`\n * or `do` methods accordingly.\n * @param {Observer|function(value: T): void} nextOrObserver An Observer or\n * the `next` callback.\n * @param {function(err: any): void} [error] An Observer `error` callback.\n * @param {function(): void} [complete] An Observer `complete` callback.\n * @return {any}\n */\n accept(nextOrObserver: PartialObserver | ((value: T) => void), error?: (err: any) => void, complete?: () => void) {\n if (nextOrObserver && typeof (>nextOrObserver).next === 'function') {\n return this.observe(>nextOrObserver);\n } else {\n return this.do(<(value: T) => void>nextOrObserver, error, complete);\n }\n }\n\n /**\n * Returns a simple Observable that just delivers the notification represented\n * by this Notification instance.\n * @return {any}\n */\n toObservable(): Observable {\n const kind = this.kind;\n switch (kind) {\n case 'N':\n return of(this.value);\n case 'E':\n return throwError(this.error);\n case 'C':\n return empty();\n }\n throw new Error('unexpected notification kind value');\n }\n\n private static completeNotification: Notification = new Notification('C');\n private static undefinedValueNotification: Notification = new Notification('N', undefined);\n\n /**\n * A shortcut to create a Notification instance of the type `next` from a\n * given value.\n * @param {T} value The `next` value.\n * @return {Notification} The \"next\" Notification representing the\n * argument.\n * @nocollapse\n */\n static createNext(value: T): Notification {\n if (typeof value !== 'undefined') {\n return new Notification('N', value);\n }\n return Notification.undefinedValueNotification;\n }\n\n /**\n * A shortcut to create a Notification instance of the type `error` from a\n * given error.\n * @param {any} [err] The `error` error.\n * @return {Notification} The \"error\" Notification representing the\n * argument.\n * @nocollapse\n */\n static createError(err?: any): Notification {\n return new Notification('E', undefined, err);\n }\n\n /**\n * A shortcut to create a Notification instance of the type `complete`.\n * @return {Notification} The valueless \"complete\" Notification.\n * @nocollapse\n */\n static createComplete(): Notification {\n return Notification.completeNotification;\n }\n}\n","import { Observable } from '../Observable';\nimport { Operator } from '../Operator';\nimport { Subscriber } from '../Subscriber';\nimport { Subscription } from '../Subscription';\nimport { Notification } from '../Notification';\nimport { MonoTypeOperatorFunction, PartialObserver, SchedulerAction, SchedulerLike, TeardownLogic } from '../types';\n\n/**\n *\n * Re-emits all notifications from source Observable with specified scheduler.\n *\n * Ensure a specific scheduler is used, from outside of an Observable.\n *\n * `observeOn` is an operator that accepts a scheduler as a first parameter, which will be used to reschedule\n * notifications emitted by the source Observable. It might be useful, if you do not have control over\n * internal scheduler of a given Observable, but want to control when its values are emitted nevertheless.\n *\n * Returned Observable emits the same notifications (nexted values, complete and error events) as the source Observable,\n * but rescheduled with provided scheduler. Note that this doesn't mean that source Observables internal\n * scheduler will be replaced in any way. Original scheduler still will be used, but when the source Observable emits\n * notification, it will be immediately scheduled again - this time with scheduler passed to `observeOn`.\n * An anti-pattern would be calling `observeOn` on Observable that emits lots of values synchronously, to split\n * that emissions into asynchronous chunks. For this to happen, scheduler would have to be passed into the source\n * Observable directly (usually into the operator that creates it). `observeOn` simply delays notifications a\n * little bit more, to ensure that they are emitted at expected moments.\n *\n * As a matter of fact, `observeOn` accepts second parameter, which specifies in milliseconds with what delay notifications\n * will be emitted. The main difference between {@link delay} operator and `observeOn` is that `observeOn`\n * will delay all notifications - including error notifications - while `delay` will pass through error\n * from source Observable immediately when it is emitted. In general it is highly recommended to use `delay` operator\n * for any kind of delaying of values in the stream, while using `observeOn` to specify which scheduler should be used\n * for notification emissions in general.\n *\n * ## Example\n * Ensure values in subscribe are called just before browser repaint.\n * ```ts\n * import { interval } from 'rxjs';\n * import { observeOn } from 'rxjs/operators';\n *\n * const intervals = interval(10); // Intervals are scheduled\n * // with async scheduler by default...\n * intervals.pipe(\n * observeOn(animationFrameScheduler), // ...but we will observe on animationFrame\n * ) // scheduler to ensure smooth animation.\n * .subscribe(val => {\n * someDiv.style.height = val + 'px';\n * });\n * ```\n *\n * @see {@link delay}\n *\n * @param {SchedulerLike} scheduler Scheduler that will be used to reschedule notifications from source Observable.\n * @param {number} [delay] Number of milliseconds that states with what delay every notification should be rescheduled.\n * @return {Observable} Observable that emits the same notifications as the source Observable,\n * but with provided scheduler.\n *\n * @method observeOn\n * @owner Observable\n */\nexport function observeOn(scheduler: SchedulerLike, delay: number = 0): MonoTypeOperatorFunction {\n return function observeOnOperatorFunction(source: Observable): Observable {\n return source.lift(new ObserveOnOperator(scheduler, delay));\n };\n}\n\nexport class ObserveOnOperator implements Operator {\n constructor(private scheduler: SchedulerLike, private delay: number = 0) {\n }\n\n call(subscriber: Subscriber, source: any): TeardownLogic {\n return source.subscribe(new ObserveOnSubscriber(subscriber, this.scheduler, this.delay));\n }\n}\n\n/**\n * We need this JSDoc comment for affecting ESDoc.\n * @ignore\n * @extends {Ignored}\n */\nexport class ObserveOnSubscriber extends Subscriber {\n /** @nocollapse */\n static dispatch(this: SchedulerAction, arg: ObserveOnMessage) {\n const { notification, destination } = arg;\n notification.observe(destination);\n this.unsubscribe();\n }\n\n constructor(destination: Subscriber,\n private scheduler: SchedulerLike,\n private delay: number = 0) {\n super(destination);\n }\n\n private scheduleMessage(notification: Notification): void {\n const destination = this.destination as Subscription;\n destination.add(this.scheduler.schedule(\n ObserveOnSubscriber.dispatch,\n this.delay,\n new ObserveOnMessage(notification, this.destination)\n ));\n }\n\n protected _next(value: T): void {\n this.scheduleMessage(Notification.createNext(value));\n }\n\n protected _error(err: any): void {\n this.scheduleMessage(Notification.createError(err));\n this.unsubscribe();\n }\n\n protected _complete(): void {\n this.scheduleMessage(Notification.createComplete());\n this.unsubscribe();\n }\n}\n\nexport class ObserveOnMessage {\n constructor(public notification: Notification,\n public destination: PartialObserver) {\n }\n}\n","import { Subject } from './Subject';\nimport { SchedulerLike } from './types';\nimport { queue } from './scheduler/queue';\nimport { Subscriber } from './Subscriber';\nimport { Subscription } from './Subscription';\nimport { ObserveOnSubscriber } from './operators/observeOn';\nimport { ObjectUnsubscribedError } from './util/ObjectUnsubscribedError';\nimport { SubjectSubscription } from './SubjectSubscription';\n/**\n * A variant of Subject that \"replays\" or emits old values to new subscribers.\n * It buffers a set number of values and will emit those values immediately to\n * any new subscribers in addition to emitting new values to existing subscribers.\n *\n * @class ReplaySubject\n */\nexport class ReplaySubject extends Subject {\n private _events: (ReplayEvent | T)[] = [];\n private _bufferSize: number;\n private _windowTime: number;\n private _infiniteTimeWindow: boolean = false;\n\n constructor(bufferSize: number = Number.POSITIVE_INFINITY,\n windowTime: number = Number.POSITIVE_INFINITY,\n private scheduler?: SchedulerLike) {\n super();\n this._bufferSize = bufferSize < 1 ? 1 : bufferSize;\n this._windowTime = windowTime < 1 ? 1 : windowTime;\n\n if (windowTime === Number.POSITIVE_INFINITY) {\n this._infiniteTimeWindow = true;\n this.next = this.nextInfiniteTimeWindow;\n } else {\n this.next = this.nextTimeWindow;\n }\n }\n\n private nextInfiniteTimeWindow(value: T): void {\n if (!this.isStopped) {\n const _events = this._events;\n _events.push(value);\n // Since this method is invoked in every next() call than the buffer\n // can overgrow the max size only by one item\n if (_events.length > this._bufferSize) {\n _events.shift();\n }\n }\n super.next(value);\n }\n\n private nextTimeWindow(value: T): void {\n if (!this.isStopped) {\n this._events.push(new ReplayEvent(this._getNow(), value));\n this._trimBufferThenGetEvents();\n }\n super.next(value);\n }\n\n /** @deprecated This is an internal implementation detail, do not use. */\n _subscribe(subscriber: Subscriber): Subscription {\n // When `_infiniteTimeWindow === true` then the buffer is already trimmed\n const _infiniteTimeWindow = this._infiniteTimeWindow;\n const _events = _infiniteTimeWindow ? this._events : this._trimBufferThenGetEvents();\n const scheduler = this.scheduler;\n const len = _events.length;\n let subscription: Subscription;\n\n if (this.closed) {\n throw new ObjectUnsubscribedError();\n } else if (this.isStopped || this.hasError) {\n subscription = Subscription.EMPTY;\n } else {\n this.observers.push(subscriber);\n subscription = new SubjectSubscription(this, subscriber);\n }\n\n if (scheduler) {\n subscriber.add(subscriber = new ObserveOnSubscriber(subscriber, scheduler));\n }\n\n if (_infiniteTimeWindow) {\n for (let i = 0; i < len && !subscriber.closed; i++) {\n subscriber.next(_events[i]);\n }\n } else {\n for (let i = 0; i < len && !subscriber.closed; i++) {\n subscriber.next((>_events[i]).value);\n }\n }\n\n if (this.hasError) {\n subscriber.error(this.thrownError);\n } else if (this.isStopped) {\n subscriber.complete();\n }\n\n return subscription;\n }\n\n _getNow(): number {\n return (this.scheduler || queue).now();\n }\n\n private _trimBufferThenGetEvents(): ReplayEvent[] {\n const now = this._getNow();\n const _bufferSize = this._bufferSize;\n const _windowTime = this._windowTime;\n const _events = []>this._events;\n\n const eventsCount = _events.length;\n let spliceCount = 0;\n\n // Trim events that fall out of the time window.\n // Start at the front of the list. Break early once\n // we encounter an event that falls within the window.\n while (spliceCount < eventsCount) {\n if ((now - _events[spliceCount].time) < _windowTime) {\n break;\n }\n spliceCount++;\n }\n\n if (eventsCount > _bufferSize) {\n spliceCount = Math.max(spliceCount, eventsCount - _bufferSize);\n }\n\n if (spliceCount > 0) {\n _events.splice(0, spliceCount);\n }\n\n return _events;\n }\n\n}\n\nclass ReplayEvent {\n constructor(public time: number, public value: T) {\n }\n}\n","import { AsyncAction } from './AsyncAction';\nimport { AsyncScheduler } from './AsyncScheduler';\n\n/**\n *\n * Async Scheduler\n *\n * Schedule task as if you used setTimeout(task, duration)\n *\n * `async` scheduler schedules tasks asynchronously, by putting them on the JavaScript\n * event loop queue. It is best used to delay tasks in time or to schedule tasks repeating\n * in intervals.\n *\n * If you just want to \"defer\" task, that is to perform it right after currently\n * executing synchronous code ends (commonly achieved by `setTimeout(deferredTask, 0)`),\n * better choice will be the {@link asapScheduler} scheduler.\n *\n * ## Examples\n * Use async scheduler to delay task\n * ```ts\n * import { asyncScheduler } from 'rxjs';\n *\n * const task = () => console.log('it works!');\n *\n * asyncScheduler.schedule(task, 2000);\n *\n * // After 2 seconds logs:\n * // \"it works!\"\n * ```\n *\n * Use async scheduler to repeat task in intervals\n * ```ts\n * import { asyncScheduler } from 'rxjs';\n *\n * function task(state) {\n * console.log(state);\n * this.schedule(state + 1, 1000); // `this` references currently executing Action,\n * // which we reschedule with new state and delay\n * }\n *\n * asyncScheduler.schedule(task, 3000, 0);\n *\n * // Logs:\n * // 0 after 3s\n * // 1 after 4s\n * // 2 after 5s\n * // 3 after 6s\n * ```\n */\nexport const asyncScheduler = new AsyncScheduler(AsyncAction);\n\n/**\n * @deprecated renamed. Use {@link asyncScheduler}\n */\nexport const async = asyncScheduler;","export interface ArgumentOutOfRangeError extends Error {\n}\n\nexport interface ArgumentOutOfRangeErrorCtor {\n new(): ArgumentOutOfRangeError;\n}\n\nconst ArgumentOutOfRangeErrorImpl = (() => {\n function ArgumentOutOfRangeErrorImpl(this: any) {\n Error.call(this);\n this.message = 'argument out of range';\n this.name = 'ArgumentOutOfRangeError';\n return this;\n }\n\n ArgumentOutOfRangeErrorImpl.prototype = Object.create(Error.prototype);\n\n return ArgumentOutOfRangeErrorImpl;\n})();\n\n/**\n * An error thrown when an element was queried at a certain index of an\n * Observable, but no such index or position exists in that sequence.\n *\n * @see {@link elementAt}\n * @see {@link take}\n * @see {@link takeLast}\n *\n * @class ArgumentOutOfRangeError\n */\nexport const ArgumentOutOfRangeError: ArgumentOutOfRangeErrorCtor = ArgumentOutOfRangeErrorImpl as any;","export interface EmptyError extends Error {\n}\n\nexport interface EmptyErrorCtor {\n new(): EmptyError;\n}\n\nconst EmptyErrorImpl = (() => {\n function EmptyErrorImpl(this: any) {\n Error.call(this);\n this.message = 'no elements in sequence';\n this.name = 'EmptyError';\n return this;\n }\n\n EmptyErrorImpl.prototype = Object.create(Error.prototype);\n\n return EmptyErrorImpl;\n})();\n\n/**\n * An error thrown when an Observable or a sequence was queried but has no\n * elements.\n *\n * @see {@link first}\n * @see {@link last}\n * @see {@link single}\n *\n * @class EmptyError\n */\nexport const EmptyError: EmptyErrorCtor = EmptyErrorImpl as any;","import { Operator } from '../Operator';\nimport { Subscriber } from '../Subscriber';\nimport { Observable } from '../Observable';\nimport { OperatorFunction } from '../types';\n\n/**\n * Applies a given `project` function to each value emitted by the source\n * Observable, and emits the resulting values as an Observable.\n *\n * Like [Array.prototype.map()](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/map),\n * it passes each source value through a transformation function to get\n * corresponding output values.\n *\n * ![](map.png)\n *\n * Similar to the well known `Array.prototype.map` function, this operator\n * applies a projection to each value and emits that projection in the output\n * Observable.\n *\n * ## Example\n * Map every click to the clientX position of that click\n * ```ts\n * import { fromEvent } from 'rxjs';\n * import { map } from 'rxjs/operators';\n *\n * const clicks = fromEvent(document, 'click');\n * const positions = clicks.pipe(map(ev => ev.clientX));\n * positions.subscribe(x => console.log(x));\n * ```\n *\n * @see {@link mapTo}\n * @see {@link pluck}\n *\n * @param {function(value: T, index: number): R} project The function to apply\n * to each `value` emitted by the source Observable. The `index` parameter is\n * the number `i` for the i-th emission that has happened since the\n * subscription, starting from the number `0`.\n * @param {any} [thisArg] An optional argument to define what `this` is in the\n * `project` function.\n * @return {Observable} An Observable that emits the values from the source\n * Observable transformed by the given `project` function.\n * @method map\n * @owner Observable\n */\nexport function map(project: (value: T, index: number) => R, thisArg?: any): OperatorFunction {\n return function mapOperation(source: Observable): Observable {\n if (typeof project !== 'function') {\n throw new TypeError('argument is not a function. Are you looking for `mapTo()`?');\n }\n return source.lift(new MapOperator(project, thisArg));\n };\n}\n\nexport class MapOperator implements Operator {\n constructor(private project: (value: T, index: number) => R, private thisArg: any) {\n }\n\n call(subscriber: Subscriber, source: any): any {\n return source.subscribe(new MapSubscriber(subscriber, this.project, this.thisArg));\n }\n}\n\n/**\n * We need this JSDoc comment for affecting ESDoc.\n * @ignore\n * @extends {Ignored}\n */\nclass MapSubscriber extends Subscriber {\n count: number = 0;\n private thisArg: any;\n\n constructor(destination: Subscriber,\n private project: (value: T, index: number) => R,\n thisArg: any) {\n super(destination);\n this.thisArg = thisArg || this;\n }\n\n // NOTE: This looks unoptimized, but it's actually purposefully NOT\n // using try/catch optimizations.\n protected _next(value: T) {\n let result: R;\n try {\n result = this.project.call(this.thisArg, value, this.count++);\n } catch (err) {\n this.destination.error(err);\n return;\n }\n this.destination.next(result);\n }\n}\n","import { Subscriber } from './Subscriber';\nimport { InnerSubscriber } from './InnerSubscriber';\n\n/**\n * We need this JSDoc comment for affecting ESDoc.\n * @ignore\n * @extends {Ignored}\n */\nexport class OuterSubscriber extends Subscriber {\n notifyNext(outerValue: T, innerValue: R,\n outerIndex: number, innerIndex: number,\n innerSub: InnerSubscriber): void {\n this.destination.next(innerValue);\n }\n\n notifyError(error: any, innerSub: InnerSubscriber): void {\n this.destination.error(error);\n }\n\n notifyComplete(innerSub: InnerSubscriber): void {\n this.destination.complete();\n }\n}\n","import { Subscriber } from './Subscriber';\nimport { OuterSubscriber } from './OuterSubscriber';\n\n/**\n * We need this JSDoc comment for affecting ESDoc.\n * @ignore\n * @extends {Ignored}\n */\nexport class InnerSubscriber extends Subscriber {\n private index = 0;\n\n constructor(private parent: OuterSubscriber, public outerValue: T, public outerIndex: number) {\n super();\n }\n\n protected _next(value: R): void {\n this.parent.notifyNext(this.outerValue, value, this.outerIndex, this.index++, this);\n }\n\n protected _error(error: any): void {\n this.parent.notifyError(error, this);\n this.unsubscribe();\n }\n\n protected _complete(): void {\n this.parent.notifyComplete(this);\n this.unsubscribe();\n }\n}\n","export function getSymbolIterator(): symbol {\n if (typeof Symbol !== 'function' || !Symbol.iterator) {\n return '@@iterator' as any;\n }\n\n return Symbol.iterator;\n}\n\nexport const iterator = getSymbolIterator();\n\n/**\n * @deprecated use {@link iterator} instead\n */\nexport const $$iterator = iterator;\n","/**\n * Tests to see if the object is an ES2015 (ES6) Promise\n * @see {@link https://www.ecma-international.org/ecma-262/6.0/#sec-promise-objects}\n * @param value the object to test\n */\nexport function isPromise(value: any): value is PromiseLike {\n return !!value && typeof (value).subscribe !== 'function' && typeof (value as any).then === 'function';\n}\n","import { ObservableInput } from '../types';\nimport { subscribeToArray } from './subscribeToArray';\nimport { subscribeToPromise } from './subscribeToPromise';\nimport { subscribeToIterable } from './subscribeToIterable';\nimport { subscribeToObservable } from './subscribeToObservable';\nimport { isArrayLike } from './isArrayLike';\nimport { isPromise } from './isPromise';\nimport { isObject } from './isObject';\nimport { iterator as Symbol_iterator } from '../symbol/iterator';\nimport { observable as Symbol_observable } from '../symbol/observable';\nimport { Subscription } from '../Subscription';\nimport { Subscriber } from '../Subscriber';\n\nexport const subscribeTo = (result: ObservableInput): (subscriber: Subscriber) => Subscription | void => {\n if (!!result && typeof result[Symbol_observable] === 'function') {\n return subscribeToObservable(result as any);\n } else if (isArrayLike(result)) {\n return subscribeToArray(result);\n } else if (isPromise(result)) {\n return subscribeToPromise(result as Promise);\n } else if (!!result && typeof result[Symbol_iterator] === 'function') {\n return subscribeToIterable(result as any);\n } else {\n const value = isObject(result) ? 'an invalid object' : `'${result}'`;\n const msg = `You provided ${value} where a stream was expected.`\n + ' You can provide an Observable, Promise, Array, or Iterable.';\n throw new TypeError(msg);\n }\n};\n","import { Subscriber } from '../Subscriber';\nimport { observable as Symbol_observable } from '../symbol/observable';\n\n/**\n * Subscribes to an object that implements Symbol.observable with the given\n * Subscriber.\n * @param obj An object that implements Symbol.observable\n */\nexport const subscribeToObservable = (obj: any) => (subscriber: Subscriber) => {\n const obs = obj[Symbol_observable]();\n if (typeof obs.subscribe !== 'function') {\n // Should be caught by observable subscribe function error handling.\n throw new TypeError('Provided object does not correctly implement Symbol.observable');\n } else {\n return obs.subscribe(subscriber);\n }\n};\n","import { Subscriber } from '../Subscriber';\nimport { hostReportError } from './hostReportError';\n\nexport const subscribeToPromise = (promise: PromiseLike) => (subscriber: Subscriber) => {\n promise.then(\n (value) => {\n if (!subscriber.closed) {\n subscriber.next(value);\n subscriber.complete();\n }\n },\n (err: any) => subscriber.error(err)\n )\n .then(null, hostReportError);\n return subscriber;\n};\n","import { Subscriber } from '../Subscriber';\nimport { iterator as Symbol_iterator } from '../symbol/iterator';\n\nexport const subscribeToIterable = (iterable: Iterable) => (subscriber: Subscriber) => {\n const iterator = (iterable as any)[Symbol_iterator]();\n\n do {\n let item: IteratorResult;\n try {\n item = iterator.next();\n } catch (err) {\n subscriber.error(err);\n return subscriber;\n }\n if (item.done) {\n subscriber.complete();\n break;\n }\n subscriber.next(item.value);\n if (subscriber.closed) {\n break;\n }\n } while (true);\n\n // Finalize the iterator if it happens to be a Generator\n if (typeof iterator.return === 'function') {\n subscriber.add(() => {\n if (iterator.return) {\n iterator.return();\n }\n });\n }\n\n return subscriber;\n};\n","import { Subscription } from '../Subscription';\nimport { InnerSubscriber } from '../InnerSubscriber';\nimport { OuterSubscriber } from '../OuterSubscriber';\nimport { Subscriber } from '../Subscriber';\nimport { subscribeTo } from './subscribeTo';\nimport { Observable } from '../Observable';\n\nexport function subscribeToResult(\n outerSubscriber: OuterSubscriber,\n result: any,\n outerValue: undefined,\n outerIndex: undefined,\n innerSubscriber: InnerSubscriber\n): Subscription | undefined;\n\nexport function subscribeToResult(\n outerSubscriber: OuterSubscriber,\n result: any,\n outerValue?: T,\n outerIndex?: number\n): Subscription | undefined;\n\nexport function subscribeToResult(\n outerSubscriber: OuterSubscriber,\n result: any,\n outerValue?: T,\n outerIndex?: number,\n innerSubscriber: Subscriber = new InnerSubscriber(outerSubscriber, outerValue, outerIndex)\n): Subscription | undefined {\n if (innerSubscriber.closed) {\n return undefined;\n }\n if (result instanceof Observable) {\n return result.subscribe(innerSubscriber);\n }\n return subscribeTo(result)(innerSubscriber) as Subscription;\n}\n","import { Observable } from '../Observable';\nimport { ObservableInput, SchedulerLike, ObservedValueOf } from '../types';\nimport { isScheduler } from '../util/isScheduler';\nimport { isArray } from '../util/isArray';\nimport { Subscriber } from '../Subscriber';\nimport { OuterSubscriber } from '../OuterSubscriber';\nimport { Operator } from '../Operator';\nimport { InnerSubscriber } from '../InnerSubscriber';\nimport { subscribeToResult } from '../util/subscribeToResult';\nimport { fromArray } from './fromArray';\n\nconst NONE = {};\n\n/* tslint:disable:max-line-length */\n\n// If called with a single array, it \"auto-spreads\" the array, with result selector\n/** @deprecated resultSelector no longer supported, pipe to map instead */\nexport function combineLatest, R>(sources: [O1], resultSelector: (v1: ObservedValueOf) => R, scheduler?: SchedulerLike): Observable;\n/** @deprecated resultSelector no longer supported, pipe to map instead */\nexport function combineLatest, O2 extends ObservableInput, R>(sources: [O1, O2], resultSelector: (v1: ObservedValueOf, v2: ObservedValueOf) => R, scheduler?: SchedulerLike): Observable;\n/** @deprecated resultSelector no longer supported, pipe to map instead */\nexport function combineLatest, O2 extends ObservableInput, O3 extends ObservableInput, R>(sources: [O1, O2, O3], resultSelector: (v1: ObservedValueOf, v2: ObservedValueOf, v3: ObservedValueOf) => R, scheduler?: SchedulerLike): Observable;\n/** @deprecated resultSelector no longer supported, pipe to map instead */\nexport function combineLatest, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput, R>(sources: [O1, O2, O3, O4], resultSelector: (v1: ObservedValueOf, v2: ObservedValueOf, v3: ObservedValueOf, v4: ObservedValueOf) => R, scheduler?: SchedulerLike): Observable;\n/** @deprecated resultSelector no longer supported, pipe to map instead */\nexport function combineLatest, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput, O5 extends ObservableInput, R>(sources: [O1, O2, O3, O4, O5], resultSelector: (v1: ObservedValueOf, v2: ObservedValueOf, v3: ObservedValueOf, v4: ObservedValueOf, v5: ObservedValueOf) => R, scheduler?: SchedulerLike): Observable;\n/** @deprecated resultSelector no longer supported, pipe to map instead */\nexport function combineLatest, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput, O5 extends ObservableInput, O6 extends ObservableInput, R>(sources: [O1, O2, O3, O4, O5, O6], resultSelector: (v1: ObservedValueOf, v2: ObservedValueOf, v3: ObservedValueOf, v4: ObservedValueOf, v5: ObservedValueOf, v6: ObservedValueOf) => R, scheduler?: SchedulerLike): Observable;\n/** @deprecated resultSelector no longer supported, pipe to map instead */\nexport function combineLatest, R>(sources: O[], resultSelector: (...args: ObservedValueOf[]) => R, scheduler?: SchedulerLike): Observable;\n\n// standard call, but with a result selector\n/** @deprecated resultSelector no longer supported, pipe to map instead */\nexport function combineLatest, R>(v1: O1, resultSelector: (v1: ObservedValueOf) => R, scheduler?: SchedulerLike): Observable;\n/** @deprecated resultSelector no longer supported, pipe to map instead */\nexport function combineLatest, O2 extends ObservableInput, R>(v1: O1, v2: O2, resultSelector: (v1: ObservedValueOf, v2: ObservedValueOf) => R, scheduler?: SchedulerLike): Observable;\n/** @deprecated resultSelector no longer supported, pipe to map instead */\nexport function combineLatest, O2 extends ObservableInput, O3 extends ObservableInput, R>(v1: O1, v2: O2, v3: O3, resultSelector: (v1: ObservedValueOf, v2: ObservedValueOf, v3: ObservedValueOf) => R, scheduler?: SchedulerLike): Observable;\n/** @deprecated resultSelector no longer supported, pipe to map instead */\nexport function combineLatest, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput, R>(v1: O1, v2: O2, v3: O3, v4: O4, resultSelector: (v1: ObservedValueOf, v2: ObservedValueOf, v3: ObservedValueOf, v4: ObservedValueOf) => R, scheduler?: SchedulerLike): Observable;\n/** @deprecated resultSelector no longer supported, pipe to map instead */\nexport function combineLatest, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput, O5 extends ObservableInput, R>(v1: O1, v2: O2, v3: O3, v4: O4, v5: O5, resultSelector: (v1: ObservedValueOf, v2: ObservedValueOf, v3: ObservedValueOf, v4: ObservedValueOf, v5: ObservedValueOf) => R, scheduler?: SchedulerLike): Observable;\n/** @deprecated resultSelector no longer supported, pipe to map instead */\nexport function combineLatest, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput, O5 extends ObservableInput, O6 extends ObservableInput, R>(v1: O1, v2: O2, v3: O3, v4: O4, v5: O5, v6: O6, resultSelector: (v1: ObservedValueOf, v2: ObservedValueOf, v3: ObservedValueOf, v4: ObservedValueOf, v5: ObservedValueOf, v6: ObservedValueOf) => R, scheduler?: SchedulerLike): Observable;\n\n// With a scheduler (deprecated)\n/** @deprecated Passing a scheduler here is deprecated, use {@link subscribeOn} and/or {@link observeOn} instead */\nexport function combineLatest>(sources: [O1], scheduler: SchedulerLike): Observable<[ObservedValueOf]>;\n/** @deprecated Passing a scheduler here is deprecated, use {@link subscribeOn} and/or {@link observeOn} instead */\nexport function combineLatest, O2 extends ObservableInput>(sources: [O1, O2], scheduler: SchedulerLike): Observable<[ObservedValueOf, ObservedValueOf]>;\n/** @deprecated Passing a scheduler here is deprecated, use {@link subscribeOn} and/or {@link observeOn} instead */\nexport function combineLatest, O2 extends ObservableInput, O3 extends ObservableInput>(sources: [O1, O2, O3], scheduler: SchedulerLike): Observable<[ObservedValueOf, ObservedValueOf, ObservedValueOf]>;\n/** @deprecated Passing a scheduler here is deprecated, use {@link subscribeOn} and/or {@link observeOn} instead */\nexport function combineLatest, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput>(sources: [O1, O2, O3, O4], scheduler: SchedulerLike): Observable<[ObservedValueOf, ObservedValueOf, ObservedValueOf, ObservedValueOf]>;\n/** @deprecated Passing a scheduler here is deprecated, use {@link subscribeOn} and/or {@link observeOn} instead */\nexport function combineLatest, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput, O5 extends ObservableInput>(sources: [O1, O2, O3, O4, O5], scheduler: SchedulerLike): Observable<[ObservedValueOf, ObservedValueOf, ObservedValueOf, ObservedValueOf, ObservedValueOf]>;\n/** @deprecated Passing a scheduler here is deprecated, use {@link subscribeOn} and/or {@link observeOn} instead */\nexport function combineLatest, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput, O5 extends ObservableInput, O6 extends ObservableInput>(sources: [O1, O2, O3, O4, O5, O6], scheduler: SchedulerLike): Observable<[ObservedValueOf, ObservedValueOf, ObservedValueOf, ObservedValueOf, ObservedValueOf, ObservedValueOf]>;\n/** @deprecated Passing a scheduler here is deprecated, use {@link subscribeOn} and/or {@link observeOn} instead */\nexport function combineLatest>(sources: O[], scheduler: SchedulerLike): Observable[]>;\n\n// Best case\nexport function combineLatest>(sources: [O1]): Observable<[ObservedValueOf]>;\nexport function combineLatest, O2 extends ObservableInput>(sources: [O1, O2]): Observable<[ObservedValueOf, ObservedValueOf]>;\nexport function combineLatest, O2 extends ObservableInput, O3 extends ObservableInput>(sources: [O1, O2, O3]): Observable<[ObservedValueOf, ObservedValueOf, ObservedValueOf]>;\nexport function combineLatest, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput>(sources: [O1, O2, O3, O4]): Observable<[ObservedValueOf, ObservedValueOf, ObservedValueOf, ObservedValueOf]>;\nexport function combineLatest, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput, O5 extends ObservableInput>(sources: [O1, O2, O3, O4, O5]): Observable<[ObservedValueOf, ObservedValueOf, ObservedValueOf, ObservedValueOf, ObservedValueOf]>;\nexport function combineLatest, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput, O5 extends ObservableInput, O6 extends ObservableInput>(sources: [O1, O2, O3, O4, O5, O6]): Observable<[ObservedValueOf, ObservedValueOf, ObservedValueOf, ObservedValueOf, ObservedValueOf, ObservedValueOf]>;\nexport function combineLatest>(sources: O[]): Observable[]>;\n\n// Standard calls\n/** @deprecated Pass arguments in a single array instead `combineLatest([a, b, c])` */\nexport function combineLatest>(v1: O1, scheduler?: SchedulerLike): Observable<[ObservedValueOf]>;\n/** @deprecated Pass arguments in a single array instead `combineLatest([a, b, c])` */\nexport function combineLatest, O2 extends ObservableInput>(v1: O1, v2: O2, scheduler?: SchedulerLike): Observable<[ObservedValueOf, ObservedValueOf]>;\n/** @deprecated Pass arguments in a single array instead `combineLatest([a, b, c])` */\nexport function combineLatest, O2 extends ObservableInput, O3 extends ObservableInput>(v1: O1, v2: O2, v3: O3, scheduler?: SchedulerLike): Observable<[ObservedValueOf, ObservedValueOf, ObservedValueOf]>;\n/** @deprecated Pass arguments in a single array instead `combineLatest([a, b, c])` */\nexport function combineLatest, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput>(v1: O1, v2: O2, v3: O3, v4: O4, scheduler?: SchedulerLike): Observable<[ObservedValueOf, ObservedValueOf, ObservedValueOf, ObservedValueOf]>;\n/** @deprecated Pass arguments in a single array instead `combineLatest([a, b, c])` */\nexport function combineLatest, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput, O5 extends ObservableInput>(v1: O1, v2: O2, v3: O3, v4: O4, v5: O5, scheduler?: SchedulerLike): Observable<[ObservedValueOf, ObservedValueOf, ObservedValueOf, ObservedValueOf, ObservedValueOf]>;\n/** @deprecated Pass arguments in a single array instead `combineLatest([a, b, c])` */\nexport function combineLatest, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput, O5 extends ObservableInput, O6 extends ObservableInput>(v1: O1, v2: O2, v3: O3, v4: O4, v5: O5, v6: O6, scheduler?: SchedulerLike): Observable<[ObservedValueOf, ObservedValueOf, ObservedValueOf, ObservedValueOf, ObservedValueOf, ObservedValueOf]>;\n\n/** @deprecated Pass arguments in a single array instead `combineLatest([a, b, c])` */\nexport function combineLatest>(...observables: O[]): Observable;\n\n/** @deprecated Pass arguments in a single array instead `combineLatest([a, b, c])` */\nexport function combineLatest, R>(...observables: Array | ((...values: Array) => R)>): Observable;\n\n/** @deprecated resultSelector no longer supported, pipe to map instead */\nexport function combineLatest, R>(array: O[], resultSelector: (...values: ObservedValueOf[]) => R, scheduler?: SchedulerLike): Observable;\n\n/** @deprecated Passing a scheduler here is deprecated, use {@link subscribeOn} and/or {@link observeOn} instead */\nexport function combineLatest>(...observables: Array): Observable;\n\n/** @deprecated Passing a scheduler here is deprecated, use {@link subscribeOn} and/or {@link observeOn} instead */\nexport function combineLatest, R>(...observables: Array[]) => R) | SchedulerLike>): Observable;\n\n/** @deprecated Passing a scheduler here is deprecated, use {@link subscribeOn} and/or {@link observeOn} instead */\nexport function combineLatest(...observables: Array | ((...values: Array) => R) | SchedulerLike>): Observable;\n/* tslint:enable:max-line-length */\n\n/**\n * Combines multiple Observables to create an Observable whose values are\n * calculated from the latest values of each of its input Observables.\n *\n * Whenever any input Observable emits a value, it\n * computes a formula using the latest values from all the inputs, then emits\n * the output of that formula.\n *\n * ![](combineLatest.png)\n *\n * `combineLatest` combines the values from all the Observables passed as\n * arguments. This is done by subscribing to each Observable in order and,\n * whenever any Observable emits, collecting an array of the most recent\n * values from each Observable. So if you pass `n` Observables to operator,\n * returned Observable will always emit an array of `n` values, in order\n * corresponding to order of passed Observables (value from the first Observable\n * on the first place and so on).\n *\n * Static version of `combineLatest` accepts either an array of Observables\n * or each Observable can be put directly as an argument. Note that array of\n * Observables is good choice, if you don't know beforehand how many Observables\n * you will combine. Passing empty array will result in Observable that\n * completes immediately.\n *\n * To ensure output array has always the same length, `combineLatest` will\n * actually wait for all input Observables to emit at least once,\n * before it starts emitting results. This means if some Observable emits\n * values before other Observables started emitting, all these values but the last\n * will be lost. On the other hand, if some Observable does not emit a value but\n * completes, resulting Observable will complete at the same moment without\n * emitting anything, since it will be now impossible to include value from\n * completed Observable in resulting array. Also, if some input Observable does\n * not emit any value and never completes, `combineLatest` will also never emit\n * and never complete, since, again, it will wait for all streams to emit some\n * value.\n *\n * If at least one Observable was passed to `combineLatest` and all passed Observables\n * emitted something, resulting Observable will complete when all combined\n * streams complete. So even if some Observable completes, result of\n * `combineLatest` will still emit values when other Observables do. In case\n * of completed Observable, its value from now on will always be the last\n * emitted value. On the other hand, if any Observable errors, `combineLatest`\n * will error immediately as well, and all other Observables will be unsubscribed.\n *\n * `combineLatest` accepts as optional parameter `project` function, which takes\n * as arguments all values that would normally be emitted by resulting Observable.\n * `project` can return any kind of value, which will be then emitted by Observable\n * instead of default array. Note that `project` does not take as argument that array\n * of values, but values themselves. That means default `project` can be imagined\n * as function that takes all its arguments and puts them into an array.\n *\n * ## Examples\n * ### Combine two timer Observables\n * ```ts\n * import { combineLatest, timer } from 'rxjs';\n *\n * const firstTimer = timer(0, 1000); // emit 0, 1, 2... after every second, starting from now\n * const secondTimer = timer(500, 1000); // emit 0, 1, 2... after every second, starting 0,5s from now\n * const combinedTimers = combineLatest(firstTimer, secondTimer);\n * combinedTimers.subscribe(value => console.log(value));\n * // Logs\n * // [0, 0] after 0.5s\n * // [1, 0] after 1s\n * // [1, 1] after 1.5s\n * // [2, 1] after 2s\n * ```\n *\n * ### Combine an array of Observables\n * ```ts\n * import { combineLatest, of } from 'rxjs';\n * import { delay, starWith } from 'rxjs/operators';\n *\n * const observables = [1, 5, 10].map(\n * n => of(n).pipe(\n * delay(n * 1000), // emit 0 and then emit n after n seconds\n * startWith(0),\n * )\n * );\n * const combined = combineLatest(observables);\n * combined.subscribe(value => console.log(value));\n * // Logs\n * // [0, 0, 0] immediately\n * // [1, 0, 0] after 1s\n * // [1, 5, 0] after 5s\n * // [1, 5, 10] after 10s\n * ```\n *\n *\n * ### Use project function to dynamically calculate the Body-Mass Index\n * ```ts\n * import { combineLatest, of } from 'rxjs';\n * import { map } from 'rxjs/operators';\n *\n * const weight = of(70, 72, 76, 79, 75);\n * const height = of(1.76, 1.77, 1.78);\n * const bmi = combineLatest(weight, height).pipe(\n * map(([w, h]) => w / (h * h)),\n * );\n * bmi.subscribe(x => console.log('BMI is ' + x));\n *\n * // With output to console:\n * // BMI is 24.212293388429753\n * // BMI is 23.93948099205209\n * // BMI is 23.671253629592222\n * ```\n *\n * @see {@link combineAll}\n * @see {@link merge}\n * @see {@link withLatestFrom}\n *\n * @param {ObservableInput} observable1 An input Observable to combine with other Observables.\n * @param {ObservableInput} observable2 An input Observable to combine with other Observables.\n * More than one input Observables may be given as arguments\n * or an array of Observables may be given as the first argument.\n * @param {function} [project] An optional function to project the values from\n * the combined latest values into a new value on the output Observable.\n * @param {SchedulerLike} [scheduler=null] The {@link SchedulerLike} to use for subscribing to\n * each input Observable.\n * @return {Observable} An Observable of projected values from the most recent\n * values from each input Observable, or an array of the most recent values from\n * each input Observable.\n */\nexport function combineLatest, R>(\n ...observables: (O | ((...values: ObservedValueOf[]) => R) | SchedulerLike)[]\n): Observable {\n let resultSelector: ((...values: Array) => R) | undefined = undefined;\n let scheduler: SchedulerLike|undefined = undefined;\n\n if (isScheduler(observables[observables.length - 1])) {\n scheduler = observables.pop() as SchedulerLike;\n }\n\n if (typeof observables[observables.length - 1] === 'function') {\n resultSelector = observables.pop() as (...values: Array) => R;\n }\n\n // if the first and only other argument besides the resultSelector is an array\n // assume it's been called with `combineLatest([obs1, obs2, obs3], resultSelector)`\n if (observables.length === 1 && isArray(observables[0])) {\n observables = observables[0] as any;\n }\n\n return fromArray(observables, scheduler).lift(new CombineLatestOperator(resultSelector));\n}\n\nexport class CombineLatestOperator implements Operator {\n constructor(private resultSelector?: (...values: Array) => R) {\n }\n\n call(subscriber: Subscriber, source: any): any {\n return source.subscribe(new CombineLatestSubscriber(subscriber, this.resultSelector));\n }\n}\n\n/**\n * We need this JSDoc comment for affecting ESDoc.\n * @ignore\n * @extends {Ignored}\n */\nexport class CombineLatestSubscriber extends OuterSubscriber {\n private active: number = 0;\n private values: any[] = [];\n private observables: any[] = [];\n private toRespond?: number;\n\n constructor(destination: Subscriber, private resultSelector?: (...values: Array) => R) {\n super(destination);\n }\n\n protected _next(observable: any) {\n this.values.push(NONE);\n this.observables.push(observable);\n }\n\n protected _complete() {\n const observables = this.observables;\n const len = observables.length;\n if (len === 0) {\n this.destination.complete!();\n } else {\n this.active = len;\n this.toRespond = len;\n for (let i = 0; i < len; i++) {\n const observable = observables[i];\n this.add(subscribeToResult(this, observable, undefined, i));\n }\n }\n }\n\n notifyComplete(unused: Subscriber): void {\n if ((this.active -= 1) === 0) {\n this.destination.complete!();\n }\n }\n\n notifyNext(_outerValue: T, innerValue: R,\n outerIndex: number): void {\n const values = this.values;\n const oldVal = values[outerIndex];\n const toRespond = !this.toRespond\n ? 0\n : oldVal === NONE ? --this.toRespond : this.toRespond;\n values[outerIndex] = innerValue;\n\n if (toRespond === 0) {\n if (this.resultSelector) {\n this._tryResultSelector(values);\n } else {\n this.destination.next!(values.slice());\n }\n }\n }\n\n private _tryResultSelector(values: any[]) {\n let result: any;\n try {\n result = this.resultSelector!.apply(this, values);\n } catch (err) {\n this.destination.error!(err);\n return;\n }\n this.destination.next!(result);\n }\n}\n","import { scheduleObservable } from './scheduleObservable';\nimport { schedulePromise } from './schedulePromise';\nimport { scheduleArray } from './scheduleArray';\nimport { scheduleIterable } from './scheduleIterable';\nimport { ObservableInput, SchedulerLike, Observable } from 'rxjs';\nimport { isInteropObservable } from '../util/isInteropObservable';\nimport { isPromise } from '../util/isPromise';\nimport { isArrayLike } from '../util/isArrayLike';\nimport { isIterable } from '../util/isIterable';\n\n/**\n * Converts from a common {@link ObservableInput} type to an observable where subscription and emissions\n * are scheduled on the provided scheduler.\n *\n * @see from\n * @see of\n *\n * @param input The observable, array, promise, iterable, etc you would like to schedule\n * @param scheduler The scheduler to use to schedule the subscription and emissions from\n * the returned observable.\n */\nexport function scheduled(input: ObservableInput, scheduler: SchedulerLike): Observable {\n if (input != null) {\n if (isInteropObservable(input)) {\n return scheduleObservable(input, scheduler);\n } else if (isPromise(input)) {\n return schedulePromise(input, scheduler);\n } else if (isArrayLike(input)) {\n return scheduleArray(input, scheduler);\n } else if (isIterable(input) || typeof input === 'string') {\n return scheduleIterable(input, scheduler);\n }\n }\n\n throw new TypeError((input !== null && typeof input || input) + ' is not observable');\n}\n","import { InteropObservable } from '../types';\nimport { observable as Symbol_observable } from '../symbol/observable';\n\n/** Identifies an input as being Observable (but not necessary an Rx Observable) */\nexport function isInteropObservable(input: any): input is InteropObservable {\n return input && typeof input[Symbol_observable] === 'function';\n}\n","import { Observable } from '../Observable';\nimport { Subscription } from '../Subscription';\nimport { observable as Symbol_observable } from '../symbol/observable';\nimport { InteropObservable, SchedulerLike, Subscribable } from '../types';\n\nexport function scheduleObservable(input: InteropObservable, scheduler: SchedulerLike) {\n return new Observable(subscriber => {\n const sub = new Subscription();\n sub.add(scheduler.schedule(() => {\n const observable: Subscribable = input[Symbol_observable]();\n sub.add(observable.subscribe({\n next(value) { sub.add(scheduler.schedule(() => subscriber.next(value))); },\n error(err) { sub.add(scheduler.schedule(() => subscriber.error(err))); },\n complete() { sub.add(scheduler.schedule(() => subscriber.complete())); },\n }));\n }));\n return sub;\n });\n}\n","import { Observable } from '../Observable';\nimport { SchedulerLike } from '../types';\nimport { Subscription } from '../Subscription';\n\nexport function schedulePromise(input: PromiseLike, scheduler: SchedulerLike) {\n return new Observable(subscriber => {\n const sub = new Subscription();\n sub.add(scheduler.schedule(() => input.then(\n value => {\n sub.add(scheduler.schedule(() => {\n subscriber.next(value);\n sub.add(scheduler.schedule(() => subscriber.complete()));\n }));\n },\n err => {\n sub.add(scheduler.schedule(() => subscriber.error(err)));\n }\n )));\n return sub;\n });\n}\n","import { iterator as Symbol_iterator } from '../symbol/iterator';\n\n/** Identifies an input as being an Iterable */\nexport function isIterable(input: any): input is Iterable {\n return input && typeof input[Symbol_iterator] === 'function';\n}\n","import { Observable } from '../Observable';\nimport { SchedulerLike } from '../types';\nimport { Subscription } from '../Subscription';\nimport { iterator as Symbol_iterator } from '../symbol/iterator';\n\nexport function scheduleIterable(input: Iterable, scheduler: SchedulerLike) {\n if (!input) {\n throw new Error('Iterable cannot be null');\n }\n return new Observable(subscriber => {\n const sub = new Subscription();\n let iterator: Iterator;\n sub.add(() => {\n // Finalize generators\n if (iterator && typeof iterator.return === 'function') {\n iterator.return();\n }\n });\n sub.add(scheduler.schedule(() => {\n iterator = input[Symbol_iterator]();\n sub.add(scheduler.schedule(function () {\n if (subscriber.closed) {\n return;\n }\n let value: T;\n let done: boolean;\n try {\n const result = iterator.next();\n value = result.value;\n done = result.done;\n } catch (err) {\n subscriber.error(err);\n return;\n }\n if (done) {\n subscriber.complete();\n } else {\n subscriber.next(value);\n this.schedule();\n }\n }));\n }));\n return sub;\n });\n}\n","import { Observable } from '../Observable';\nimport { subscribeTo } from '../util/subscribeTo';\nimport { ObservableInput, SchedulerLike, ObservedValueOf } from '../types';\nimport { scheduled } from '../scheduled/scheduled';\n\nexport function from>(input: O): Observable>;\n/** @deprecated use {@link scheduled} instead. */\nexport function from>(input: O, scheduler: SchedulerLike): Observable>;\n\n/**\n * Creates an Observable from an Array, an array-like object, a Promise, an iterable object, or an Observable-like object.\n *\n * Converts almost anything to an Observable.\n *\n * ![](from.png)\n *\n * `from` converts various other objects and data types into Observables. It also converts a Promise, an array-like, or an\n * iterable\n * object into an Observable that emits the items in that promise, array, or iterable. A String, in this context, is treated\n * as an array of characters. Observable-like objects (contains a function named with the ES2015 Symbol for Observable) can also be\n * converted through this operator.\n *\n * ## Examples\n *\n * ### Converts an array to an Observable\n *\n * ```ts\n * import { from } from 'rxjs';\n *\n * const array = [10, 20, 30];\n * const result = from(array);\n *\n * result.subscribe(x => console.log(x));\n *\n * // Logs:\n * // 10\n * // 20\n * // 30\n * ```\n *\n * ---\n *\n * ### Convert an infinite iterable (from a generator) to an Observable\n *\n * ```ts\n * import { from } from 'rxjs';\n * import { take } from 'rxjs/operators';\n *\n * function* generateDoubles(seed) {\n * let i = seed;\n * while (true) {\n * yield i;\n * i = 2 * i; // double it\n * }\n * }\n *\n * const iterator = generateDoubles(3);\n * const result = from(iterator).pipe(take(10));\n *\n * result.subscribe(x => console.log(x));\n *\n * // Logs:\n * // 3\n * // 6\n * // 12\n * // 24\n * // 48\n * // 96\n * // 192\n * // 384\n * // 768\n * // 1536\n * ```\n *\n * ---\n *\n * ### With async scheduler\n *\n * ```ts\n * import { from, asyncScheduler } from 'rxjs';\n *\n * console.log('start');\n *\n * const array = [10, 20, 30];\n * const result = from(array, asyncScheduler);\n *\n * result.subscribe(x => console.log(x));\n *\n * console.log('end');\n *\n * // Logs:\n * // start\n * // end\n * // 10\n * // 20\n * // 30\n * ```\n *\n * @see {@link fromEvent}\n * @see {@link fromEventPattern}\n *\n * @param {ObservableInput} A subscription object, a Promise, an Observable-like,\n * an Array, an iterable, or an array-like object to be converted.\n * @param {SchedulerLike} An optional {@link SchedulerLike} on which to schedule the emission of values.\n * @return {Observable}\n * @name from\n * @owner Observable\n */\nexport function from(input: ObservableInput, scheduler?: SchedulerLike): Observable {\n if (!scheduler) {\n if (input instanceof Observable) {\n return input;\n }\n return new Observable(subscribeTo(input));\n } else {\n return scheduled(input, scheduler);\n }\n}\n","/** @prettier */\nimport { Subscription } from './Subscription';\nimport { Subscriber } from './Subscriber';\nimport { Observable } from './Observable';\nimport { subscribeTo } from './util/subscribeTo';\n\ninterface SimpleOuterSubscriberLike {\n /**\n * A handler for inner next notifications from the inner subscription\n * @param innerValue the value nexted by the inner producer\n */\n notifyNext(innerValue: T): void;\n /**\n * A handler for inner error notifications from the inner subscription\n * @param err the error from the inner producer\n */\n notifyError(err: any): void;\n /**\n * A handler for inner complete notifications from the inner subscription.\n */\n notifyComplete(): void;\n}\n\nexport class SimpleInnerSubscriber extends Subscriber {\n constructor(private parent: SimpleOuterSubscriberLike) {\n super();\n }\n\n protected _next(value: T): void {\n this.parent.notifyNext(value);\n }\n\n protected _error(error: any): void {\n this.parent.notifyError(error);\n this.unsubscribe();\n }\n\n protected _complete(): void {\n this.parent.notifyComplete();\n this.unsubscribe();\n }\n}\n\nexport class ComplexInnerSubscriber extends Subscriber {\n constructor(private parent: ComplexOuterSubscriber, public outerValue: T, public outerIndex: number) {\n super();\n }\n\n protected _next(value: R): void {\n this.parent.notifyNext(this.outerValue, value, this.outerIndex, this);\n }\n\n protected _error(error: any): void {\n this.parent.notifyError(error);\n this.unsubscribe();\n }\n\n protected _complete(): void {\n this.parent.notifyComplete(this);\n this.unsubscribe();\n }\n}\n\nexport class SimpleOuterSubscriber extends Subscriber implements SimpleOuterSubscriberLike {\n notifyNext(innerValue: R): void {\n this.destination.next(innerValue);\n }\n\n notifyError(err: any): void {\n this.destination.error(err);\n }\n\n notifyComplete(): void {\n this.destination.complete();\n }\n}\n\n/**\n * DO NOT USE (formerly \"OuterSubscriber\")\n * TODO: We want to refactor this and remove it. It is retaining values it shouldn't for long\n * periods of time.\n */\nexport class ComplexOuterSubscriber extends Subscriber {\n /**\n * @param _outerValue Used by: bufferToggle, delayWhen, windowToggle\n * @param innerValue Used by: subclass default, combineLatest, race, bufferToggle, windowToggle, withLatestFrom\n * @param _outerIndex Used by: combineLatest, race, withLatestFrom\n * @param _innerSub Used by: delayWhen\n */\n notifyNext(_outerValue: T, innerValue: R, _outerIndex: number, _innerSub: ComplexInnerSubscriber): void {\n this.destination.next(innerValue);\n }\n\n notifyError(error: any): void {\n this.destination.error(error);\n }\n\n /**\n * @param _innerSub Used by: race, bufferToggle, delayWhen, windowToggle, windowWhen\n */\n notifyComplete(_innerSub: ComplexInnerSubscriber): void {\n this.destination.complete();\n }\n}\n\nexport function innerSubscribe(result: any, innerSubscriber: Subscriber): Subscription | undefined {\n if (innerSubscriber.closed) {\n return undefined;\n }\n if (result instanceof Observable) {\n return result.subscribe(innerSubscriber);\n }\n let subscription: Subscription;\n try {\n subscription = subscribeTo(result)(innerSubscriber) as Subscription;\n } catch (error) {\n innerSubscriber.error(error);\n }\n return subscription;\n}\n","import { Observable } from '../Observable';\nimport { Operator } from '../Operator';\nimport { Subscriber } from '../Subscriber';\nimport { Subscription } from '../Subscription';\nimport { ObservableInput, OperatorFunction, ObservedValueOf } from '../types';\nimport { map } from './map';\nimport { from } from '../observable/from';\nimport { SimpleOuterSubscriber, SimpleInnerSubscriber, innerSubscribe } from '../innerSubscribe';\n\n/* tslint:disable:max-line-length */\nexport function mergeMap>(project: (value: T, index: number) => O, concurrent?: number): OperatorFunction>;\n/** @deprecated resultSelector no longer supported, use inner map instead */\nexport function mergeMap>(project: (value: T, index: number) => O, resultSelector: undefined, concurrent?: number): OperatorFunction>;\n/** @deprecated resultSelector no longer supported, use inner map instead */\nexport function mergeMap>(project: (value: T, index: number) => O, resultSelector: (outerValue: T, innerValue: ObservedValueOf, outerIndex: number, innerIndex: number) => R, concurrent?: number): OperatorFunction;\n/* tslint:enable:max-line-length */\n\n/**\n * Projects each source value to an Observable which is merged in the output\n * Observable.\n *\n * Maps each value to an Observable, then flattens all of\n * these inner Observables using {@link mergeAll}.\n *\n * ![](mergeMap.png)\n *\n * Returns an Observable that emits items based on applying a function that you\n * supply to each item emitted by the source Observable, where that function\n * returns an Observable, and then merging those resulting Observables and\n * emitting the results of this merger.\n *\n * ## Example\n * Map and flatten each letter to an Observable ticking every 1 second\n * ```ts\n * import { of, interval } from 'rxjs';\n * import { mergeMap, map } from 'rxjs/operators';\n *\n * const letters = of('a', 'b', 'c');\n * const result = letters.pipe(\n * mergeMap(x => interval(1000).pipe(map(i => x+i))),\n * );\n * result.subscribe(x => console.log(x));\n *\n * // Results in the following:\n * // a0\n * // b0\n * // c0\n * // a1\n * // b1\n * // c1\n * // continues to list a,b,c with respective ascending integers\n * ```\n *\n * @see {@link concatMap}\n * @see {@link exhaustMap}\n * @see {@link merge}\n * @see {@link mergeAll}\n * @see {@link mergeMapTo}\n * @see {@link mergeScan}\n * @see {@link switchMap}\n *\n * @param {function(value: T, ?index: number): ObservableInput} project A function\n * that, when applied to an item emitted by the source Observable, returns an\n * Observable.\n * @param {number} [concurrent=Number.POSITIVE_INFINITY] Maximum number of input\n * Observables being subscribed to concurrently.\n * @return {Observable} An Observable that emits the result of applying the\n * projection function (and the optional deprecated `resultSelector`) to each item\n * emitted by the source Observable and merging the results of the Observables\n * obtained from this transformation.\n */\nexport function mergeMap>(\n project: (value: T, index: number) => O,\n resultSelector?: ((outerValue: T, innerValue: ObservedValueOf, outerIndex: number, innerIndex: number) => R) | number,\n concurrent: number = Number.POSITIVE_INFINITY\n): OperatorFunction|R> {\n if (typeof resultSelector === 'function') {\n // DEPRECATED PATH\n return (source: Observable) => source.pipe(\n mergeMap((a, i) => from(project(a, i)).pipe(\n map((b: any, ii: number) => resultSelector(a, b, i, ii)),\n ), concurrent)\n );\n } else if (typeof resultSelector === 'number') {\n concurrent = resultSelector;\n }\n return (source: Observable) => source.lift(new MergeMapOperator(project, concurrent));\n}\n\nexport class MergeMapOperator implements Operator {\n constructor(private project: (value: T, index: number) => ObservableInput,\n private concurrent: number = Number.POSITIVE_INFINITY) {\n }\n\n call(observer: Subscriber, source: any): any {\n return source.subscribe(new MergeMapSubscriber(\n observer, this.project, this.concurrent\n ));\n }\n}\n\n/**\n * We need this JSDoc comment for affecting ESDoc.\n * @ignore\n * @extends {Ignored}\n */\nexport class MergeMapSubscriber extends SimpleOuterSubscriber {\n private hasCompleted: boolean = false;\n private buffer: T[] = [];\n private active: number = 0;\n protected index: number = 0;\n\n constructor(destination: Subscriber,\n private project: (value: T, index: number) => ObservableInput,\n private concurrent: number = Number.POSITIVE_INFINITY) {\n super(destination);\n }\n\n protected _next(value: T): void {\n if (this.active < this.concurrent) {\n this._tryNext(value);\n } else {\n this.buffer.push(value);\n }\n }\n\n protected _tryNext(value: T) {\n let result: ObservableInput;\n const index = this.index++;\n try {\n result = this.project(value, index);\n } catch (err) {\n this.destination.error!(err);\n return;\n }\n this.active++;\n this._innerSub(result);\n }\n\n private _innerSub(ish: ObservableInput): void {\n const innerSubscriber = new SimpleInnerSubscriber(this);\n const destination = this.destination as Subscription;\n destination.add(innerSubscriber);\n const innerSubscription = innerSubscribe(ish, innerSubscriber);\n // The returned subscription will usually be the subscriber that was\n // passed. However, interop subscribers will be wrapped and for\n // unsubscriptions to chain correctly, the wrapper needs to be added, too.\n if (innerSubscription !== innerSubscriber) {\n destination.add(innerSubscription);\n }\n }\n\n protected _complete(): void {\n this.hasCompleted = true;\n if (this.active === 0 && this.buffer.length === 0) {\n this.destination.complete!();\n }\n this.unsubscribe();\n }\n\n notifyNext(innerValue: R): void {\n this.destination.next!(innerValue);\n }\n\n notifyComplete(): void {\n const buffer = this.buffer;\n this.active--;\n if (buffer.length > 0) {\n this._next(buffer.shift()!);\n } else if (this.active === 0 && this.hasCompleted) {\n this.destination.complete!();\n }\n }\n}\n\n/**\n * @deprecated renamed. Use {@link mergeMap}\n */\nexport const flatMap = mergeMap;","\nimport { mergeMap } from './mergeMap';\nimport { identity } from '../util/identity';\nimport { OperatorFunction, ObservableInput } from '../types';\n\n/**\n * Converts a higher-order Observable into a first-order Observable which\n * concurrently delivers all values that are emitted on the inner Observables.\n *\n * Flattens an Observable-of-Observables.\n *\n * ![](mergeAll.png)\n *\n * `mergeAll` subscribes to an Observable that emits Observables, also known as\n * a higher-order Observable. Each time it observes one of these emitted inner\n * Observables, it subscribes to that and delivers all the values from the\n * inner Observable on the output Observable. The output Observable only\n * completes once all inner Observables have completed. Any error delivered by\n * a inner Observable will be immediately emitted on the output Observable.\n *\n * ## Examples\n * Spawn a new interval Observable for each click event, and blend their outputs as one Observable\n * ```ts\n * import { fromEvent, interval } from 'rxjs';\n * import { map, mergeAll } from 'rxjs/operators';\n *\n * const clicks = fromEvent(document, 'click');\n * const higherOrder = clicks.pipe(map((ev) => interval(1000)));\n * const firstOrder = higherOrder.pipe(mergeAll());\n * firstOrder.subscribe(x => console.log(x));\n * ```\n *\n * Count from 0 to 9 every second for each click, but only allow 2 concurrent timers\n * ```ts\n * import { fromEvent, interval } from 'rxjs';\n * import { take, map, mergeAll } from 'rxjs/operators';\n *\n * const clicks = fromEvent(document, 'click');\n * const higherOrder = clicks.pipe(\n * map((ev) => interval(1000).pipe(take(10))),\n * );\n * const firstOrder = higherOrder.pipe(mergeAll(2));\n * firstOrder.subscribe(x => console.log(x));\n * ```\n *\n * @see {@link combineAll}\n * @see {@link concatAll}\n * @see {@link exhaust}\n * @see {@link merge}\n * @see {@link mergeMap}\n * @see {@link mergeMapTo}\n * @see {@link mergeScan}\n * @see {@link switchAll}\n * @see {@link switchMap}\n * @see {@link zipAll}\n *\n * @param {number} [concurrent=Number.POSITIVE_INFINITY] Maximum number of inner\n * Observables being subscribed to concurrently.\n * @return {Observable} An Observable that emits values coming from all the\n * inner Observables emitted by the source Observable.\n * @method mergeAll\n * @owner Observable\n */\nexport function mergeAll(concurrent: number = Number.POSITIVE_INFINITY): OperatorFunction, T> {\n return mergeMap(identity, concurrent);\n}\n","\nimport { mergeAll } from './mergeAll';\nimport { OperatorFunction, ObservableInput } from '../types';\n\nexport function concatAll(): OperatorFunction, T>;\nexport function concatAll(): OperatorFunction;\n\n/**\n * Converts a higher-order Observable into a first-order Observable by\n * concatenating the inner Observables in order.\n *\n * Flattens an Observable-of-Observables by putting one\n * inner Observable after the other.\n *\n * ![](concatAll.png)\n *\n * Joins every Observable emitted by the source (a higher-order Observable), in\n * a serial fashion. It subscribes to each inner Observable only after the\n * previous inner Observable has completed, and merges all of their values into\n * the returned observable.\n *\n * __Warning:__ If the source Observable emits Observables quickly and\n * endlessly, and the inner Observables it emits generally complete slower than\n * the source emits, you can run into memory issues as the incoming Observables\n * collect in an unbounded buffer.\n *\n * Note: `concatAll` is equivalent to `mergeAll` with concurrency parameter set\n * to `1`.\n *\n * ## Example\n *\n * For each click event, tick every second from 0 to 3, with no concurrency\n * ```ts\n * import { fromEvent, interval } from 'rxjs';\n * import { map, take, concatAll } from 'rxjs/operators';\n *\n * const clicks = fromEvent(document, 'click');\n * const higherOrder = clicks.pipe(\n * map(ev => interval(1000).pipe(take(4))),\n * );\n * const firstOrder = higherOrder.pipe(concatAll());\n * firstOrder.subscribe(x => console.log(x));\n *\n * // Results in the following:\n * // (results are not concurrent)\n * // For every click on the \"document\" it will emit values 0 to 3 spaced\n * // on a 1000ms interval\n * // one click = 1000ms-> 0 -1000ms-> 1 -1000ms-> 2 -1000ms-> 3\n * ```\n *\n * @see {@link combineAll}\n * @see {@link concat}\n * @see {@link concatMap}\n * @see {@link concatMapTo}\n * @see {@link exhaust}\n * @see {@link mergeAll}\n * @see {@link switchAll}\n * @see {@link switchMap}\n * @see {@link zipAll}\n *\n * @return {Observable} An Observable emitting values from all the inner\n * Observables concatenated.\n * @method concatAll\n * @owner Observable\n */\nexport function concatAll(): OperatorFunction, T> {\n return mergeAll(1);\n}\n","import { Observable } from '../Observable';\nimport { ObservableInput, SchedulerLike, ObservedValueOf } from '../types';\nimport { isScheduler } from '../util/isScheduler';\nimport { of } from './of';\nimport { from } from './from';\nimport { concatAll } from '../operators/concatAll';\n\n/* tslint:disable:max-line-length */\n/** @deprecated Use {@link scheduled} and {@link concatAll} (e.g. `scheduled([o1, o2, o3], scheduler).pipe(concatAll())`) */\nexport function concat>(v1: O1, scheduler: SchedulerLike): Observable>;\n/** @deprecated Use {@link scheduled} and {@link concatAll} (e.g. `scheduled([o1, o2, o3], scheduler).pipe(concatAll())`) */\nexport function concat, O2 extends ObservableInput>(v1: O1, v2: O2, scheduler: SchedulerLike): Observable | ObservedValueOf>;\n/** @deprecated Use {@link scheduled} and {@link concatAll} (e.g. `scheduled([o1, o2, o3], scheduler).pipe(concatAll())`) */\nexport function concat, O2 extends ObservableInput, O3 extends ObservableInput>(v1: O1, v2: O2, v3: O3, scheduler: SchedulerLike): Observable | ObservedValueOf | ObservedValueOf>;\n/** @deprecated Use {@link scheduled} and {@link concatAll} (e.g. `scheduled([o1, o2, o3], scheduler).pipe(concatAll())`) */\nexport function concat, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput>(v1: O1, v2: O2, v3: O3, v4: O4, scheduler: SchedulerLike): Observable | ObservedValueOf | ObservedValueOf | ObservedValueOf>;\n/** @deprecated Use {@link scheduled} and {@link concatAll} (e.g. `scheduled([o1, o2, o3], scheduler).pipe(concatAll())`) */\nexport function concat, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput, O5 extends ObservableInput>(v1: O1, v2: O2, v3: O3, v4: O4, v5: O5, scheduler: SchedulerLike): Observable | ObservedValueOf | ObservedValueOf | ObservedValueOf | ObservedValueOf>;\n/** @deprecated Use {@link scheduled} and {@link concatAll} (e.g. `scheduled([o1, o2, o3], scheduler).pipe(concatAll())`) */\nexport function concat, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput, O5 extends ObservableInput, O6 extends ObservableInput>(v1: O1, v2: O2, v3: O3, v4: O4, v5: O5, v6: O6, scheduler: SchedulerLike): Observable | ObservedValueOf | ObservedValueOf | ObservedValueOf | ObservedValueOf | ObservedValueOf>;\n\nexport function concat>(v1: O1): Observable>;\nexport function concat, O2 extends ObservableInput>(v1: O1, v2: O2): Observable | ObservedValueOf>;\nexport function concat, O2 extends ObservableInput, O3 extends ObservableInput>(v1: O1, v2: O2, v3: O3): Observable | ObservedValueOf | ObservedValueOf>;\nexport function concat, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput>(v1: O1, v2: O2, v3: O3, v4: O4): Observable | ObservedValueOf | ObservedValueOf | ObservedValueOf>;\nexport function concat, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput, O5 extends ObservableInput>(v1: O1, v2: O2, v3: O3, v4: O4, v5: O5): Observable | ObservedValueOf | ObservedValueOf | ObservedValueOf | ObservedValueOf>;\nexport function concat, O2 extends ObservableInput, O3 extends ObservableInput, O4 extends ObservableInput, O5 extends ObservableInput, O6 extends ObservableInput>(v1: O1, v2: O2, v3: O3, v4: O4, v5: O5, v6: O6): Observable | ObservedValueOf | ObservedValueOf | ObservedValueOf | ObservedValueOf | ObservedValueOf>;\nexport function concat>(...observables: O[]): Observable>;\n/** @deprecated Use {@link scheduled} and {@link concatAll} (e.g. `scheduled([o1, o2, o3], scheduler).pipe(concatAll())`) */\nexport function concat>(...observables: (O | SchedulerLike)[]): Observable>;\nexport function concat(...observables: ObservableInput[]): Observable;\n/** @deprecated Use {@link scheduled} and {@link concatAll} (e.g. `scheduled([o1, o2, o3], scheduler).pipe(concatAll())`) */\nexport function concat(...observables: (ObservableInput | SchedulerLike)[]): Observable;\n/* tslint:enable:max-line-length */\n/**\n * Creates an output Observable which sequentially emits all values from given\n * Observable and then moves on to the next.\n *\n * Concatenates multiple Observables together by\n * sequentially emitting their values, one Observable after the other.\n *\n * ![](concat.png)\n *\n * `concat` joins multiple Observables together, by subscribing to them one at a time and\n * merging their results into the output Observable. You can pass either an array of\n * Observables, or put them directly as arguments. Passing an empty array will result\n * in Observable that completes immediately.\n *\n * `concat` will subscribe to first input Observable and emit all its values, without\n * changing or affecting them in any way. When that Observable completes, it will\n * subscribe to then next Observable passed and, again, emit its values. This will be\n * repeated, until the operator runs out of Observables. When last input Observable completes,\n * `concat` will complete as well. At any given moment only one Observable passed to operator\n * emits values. If you would like to emit values from passed Observables concurrently, check out\n * {@link merge} instead, especially with optional `concurrent` parameter. As a matter of fact,\n * `concat` is an equivalent of `merge` operator with `concurrent` parameter set to `1`.\n *\n * Note that if some input Observable never completes, `concat` will also never complete\n * and Observables following the one that did not complete will never be subscribed. On the other\n * hand, if some Observable simply completes immediately after it is subscribed, it will be\n * invisible for `concat`, which will just move on to the next Observable.\n *\n * If any Observable in chain errors, instead of passing control to the next Observable,\n * `concat` will error immediately as well. Observables that would be subscribed after\n * the one that emitted error, never will.\n *\n * If you pass to `concat` the same Observable many times, its stream of values\n * will be \"replayed\" on every subscription, which means you can repeat given Observable\n * as many times as you like. If passing the same Observable to `concat` 1000 times becomes tedious,\n * you can always use {@link repeat}.\n *\n * ## Examples\n * ### Concatenate a timer counting from 0 to 3 with a synchronous sequence from 1 to 10\n * ```ts\n * import { concat, interval, range } from 'rxjs';\n * import { take } from 'rxjs/operators';\n *\n * const timer = interval(1000).pipe(take(4));\n * const sequence = range(1, 10);\n * const result = concat(timer, sequence);\n * result.subscribe(x => console.log(x));\n *\n * // results in:\n * // 0 -1000ms-> 1 -1000ms-> 2 -1000ms-> 3 -immediate-> 1 ... 10\n * ```\n *\n * ### Concatenate 3 Observables\n * ```ts\n * import { concat, interval } from 'rxjs';\n * import { take } from 'rxjs/operators';\n *\n * const timer1 = interval(1000).pipe(take(10));\n * const timer2 = interval(2000).pipe(take(6));\n * const timer3 = interval(500).pipe(take(10));\n *\n * const result = concat(timer1, timer2, timer3);\n * result.subscribe(x => console.log(x));\n *\n * // results in the following:\n * // (Prints to console sequentially)\n * // -1000ms-> 0 -1000ms-> 1 -1000ms-> ... 9\n * // -2000ms-> 0 -2000ms-> 1 -2000ms-> ... 5\n * // -500ms-> 0 -500ms-> 1 -500ms-> ... 9\n * ```\n *\n * ### Concatenate the same Observable to repeat it\n * ```ts\n * import { concat, interval } from 'rxjs';\n * import { take } from 'rxjs/operators';\n *\n * const timer = interval(1000).pipe(take(2));\n *\n * concat(timer, timer) // concatenating the same Observable!\n * .subscribe(\n * value => console.log(value),\n * err => {},\n * () => console.log('...and it is done!')\n * );\n *\n * // Logs:\n * // 0 after 1s\n * // 1 after 2s\n * // 0 after 3s\n * // 1 after 4s\n * // \"...and it is done!\" also after 4s\n * ```\n *\n * @see {@link concatAll}\n * @see {@link concatMap}\n * @see {@link concatMapTo}\n * @see {@link startWith}\n * @see {@link endWith}\n *\n * @param {ObservableInput} input1 An input Observable to concatenate with others.\n * @param {ObservableInput} input2 An input Observable to concatenate with others.\n * More than one input Observables may be given as argument.\n * @param {SchedulerLike} [scheduler=null] An optional {@link SchedulerLike} to schedule each\n * Observable subscription on.\n * @return {Observable} All values of each passed Observable merged into a\n * single Observable, in order, in serial fashion.\n * @static true\n * @name concat\n * @owner Observable\n */\nexport function concat, R>(...observables: Array): Observable | R> {\n return concatAll()(of(...observables));\n}\n","import { Observable } from '../Observable';\nimport { ObservableInput, SchedulerLike} from '../types';\nimport { isScheduler } from '../util/isScheduler';\nimport { mergeAll } from '../operators/mergeAll';\nimport { fromArray } from './fromArray';\n\n/* tslint:disable:max-line-length */\n/** @deprecated use {@link scheduled} and {@link mergeAll} (e.g. `scheduled([ob1, ob2, ob3], scheduled).pipe(mergeAll())*/\nexport function merge(v1: ObservableInput, scheduler: SchedulerLike): Observable;\n/** @deprecated use {@link scheduled} and {@link mergeAll} (e.g. `scheduled([ob1, ob2, ob3], scheduled).pipe(mergeAll())*/\nexport function merge(v1: ObservableInput, concurrent: number, scheduler: SchedulerLike): Observable;\n/** @deprecated use {@link scheduled} and {@link mergeAll} (e.g. `scheduled([ob1, ob2, ob3], scheduled).pipe(mergeAll())*/\nexport function merge(v1: ObservableInput, v2: ObservableInput, scheduler: SchedulerLike): Observable;\n/** @deprecated use {@link scheduled} and {@link mergeAll} (e.g. `scheduled([ob1, ob2, ob3], scheduled).pipe(mergeAll())*/\nexport function merge(v1: ObservableInput, v2: ObservableInput, concurrent: number, scheduler: SchedulerLike): Observable;\n/** @deprecated use {@link scheduled} and {@link mergeAll} (e.g. `scheduled([ob1, ob2, ob3], scheduled).pipe(mergeAll())*/\nexport function merge(v1: ObservableInput, v2: ObservableInput, v3: ObservableInput, scheduler: SchedulerLike): Observable;\n/** @deprecated use {@link scheduled} and {@link mergeAll} (e.g. `scheduled([ob1, ob2, ob3], scheduled).pipe(mergeAll())*/\nexport function merge(v1: ObservableInput, v2: ObservableInput, v3: ObservableInput, concurrent: number, scheduler: SchedulerLike): Observable;\n/** @deprecated use {@link scheduled} and {@link mergeAll} (e.g. `scheduled([ob1, ob2, ob3], scheduled).pipe(mergeAll())*/\nexport function merge(v1: ObservableInput, v2: ObservableInput, v3: ObservableInput, v4: ObservableInput, scheduler: SchedulerLike): Observable;\n/** @deprecated use {@link scheduled} and {@link mergeAll} (e.g. `scheduled([ob1, ob2, ob3], scheduled).pipe(mergeAll())*/\nexport function merge(v1: ObservableInput, v2: ObservableInput, v3: ObservableInput, v4: ObservableInput, concurrent: number, scheduler: SchedulerLike): Observable;\n/** @deprecated use {@link scheduled} and {@link mergeAll} (e.g. `scheduled([ob1, ob2, ob3], scheduled).pipe(mergeAll())*/\nexport function merge(v1: ObservableInput, v2: ObservableInput, v3: ObservableInput, v4: ObservableInput, v5: ObservableInput, scheduler: SchedulerLike): Observable;\n/** @deprecated use {@link scheduled} and {@link mergeAll} (e.g. `scheduled([ob1, ob2, ob3], scheduled).pipe(mergeAll())*/\nexport function merge(v1: ObservableInput, v2: ObservableInput, v3: ObservableInput, v4: ObservableInput, v5: ObservableInput, concurrent: number, scheduler: SchedulerLike): Observable;\n/** @deprecated use {@link scheduled} and {@link mergeAll} (e.g. `scheduled([ob1, ob2, ob3], scheduled).pipe(mergeAll())*/\nexport function merge(v1: ObservableInput, v2: ObservableInput, v3: ObservableInput, v4: ObservableInput, v5: ObservableInput, v6: ObservableInput, scheduler: SchedulerLike): Observable;\n/** @deprecated use {@link scheduled} and {@link mergeAll} (e.g. `scheduled([ob1, ob2, ob3], scheduled).pipe(mergeAll())*/\nexport function merge(v1: ObservableInput, v2: ObservableInput, v3: ObservableInput, v4: ObservableInput, v5: ObservableInput, v6: ObservableInput, concurrent: number, scheduler: SchedulerLike): Observable;\n\nexport function merge(v1: ObservableInput): Observable;\nexport function merge(v1: ObservableInput, concurrent?: number): Observable;\nexport function merge(v1: ObservableInput, v2: ObservableInput): Observable;\nexport function merge(v1: ObservableInput, v2: ObservableInput, concurrent?: number): Observable;\nexport function merge(v1: ObservableInput, v2: ObservableInput, v3: ObservableInput): Observable;\nexport function merge(v1: ObservableInput, v2: ObservableInput, v3: ObservableInput, concurrent?: number): Observable;\nexport function merge(v1: ObservableInput, v2: ObservableInput, v3: ObservableInput, v4: ObservableInput): Observable;\nexport function merge(v1: ObservableInput, v2: ObservableInput, v3: ObservableInput, v4: ObservableInput, concurrent?: number): Observable;\nexport function merge(v1: ObservableInput, v2: ObservableInput, v3: ObservableInput, v4: ObservableInput, v5: ObservableInput): Observable;\nexport function merge(v1: ObservableInput, v2: ObservableInput, v3: ObservableInput, v4: ObservableInput, v5: ObservableInput, concurrent?: number): Observable;\nexport function merge(v1: ObservableInput, v2: ObservableInput, v3: ObservableInput, v4: ObservableInput, v5: ObservableInput, v6: ObservableInput): Observable;\nexport function merge(v1: ObservableInput, v2: ObservableInput, v3: ObservableInput, v4: ObservableInput, v5: ObservableInput, v6: ObservableInput, concurrent?: number): Observable;\nexport function merge(...observables: (ObservableInput | number)[]): Observable;\n/** @deprecated use {@link scheduled} and {@link mergeAll} (e.g. `scheduled([ob1, ob2, ob3], scheduled).pipe(mergeAll())*/\nexport function merge(...observables: (ObservableInput | SchedulerLike | number)[]): Observable;\nexport function merge(...observables: (ObservableInput | number)[]): Observable;\n/** @deprecated use {@link scheduled} and {@link mergeAll} (e.g. `scheduled([ob1, ob2, ob3], scheduled).pipe(mergeAll())*/\nexport function merge(...observables: (ObservableInput | SchedulerLike | number)[]): Observable;\n/* tslint:enable:max-line-length */\n/**\n * Creates an output Observable which concurrently emits all values from every\n * given input Observable.\n *\n * Flattens multiple Observables together by blending\n * their values into one Observable.\n *\n * ![](merge.png)\n *\n * `merge` subscribes to each given input Observable (as arguments), and simply\n * forwards (without doing any transformation) all the values from all the input\n * Observables to the output Observable. The output Observable only completes\n * once all input Observables have completed. Any error delivered by an input\n * Observable will be immediately emitted on the output Observable.\n *\n * ## Examples\n * ### Merge together two Observables: 1s interval and clicks\n * ```ts\n * import { merge, fromEvent, interval } from 'rxjs';\n *\n * const clicks = fromEvent(document, 'click');\n * const timer = interval(1000);\n * const clicksOrTimer = merge(clicks, timer);\n * clicksOrTimer.subscribe(x => console.log(x));\n *\n * // Results in the following:\n * // timer will emit ascending values, one every second(1000ms) to console\n * // clicks logs MouseEvents to console everytime the \"document\" is clicked\n * // Since the two streams are merged you see these happening\n * // as they occur.\n * ```\n *\n * ### Merge together 3 Observables, but only 2 run concurrently\n * ```ts\n * import { merge, interval } from 'rxjs';\n * import { take } from 'rxjs/operators';\n *\n * const timer1 = interval(1000).pipe(take(10));\n * const timer2 = interval(2000).pipe(take(6));\n * const timer3 = interval(500).pipe(take(10));\n * const concurrent = 2; // the argument\n * const merged = merge(timer1, timer2, timer3, concurrent);\n * merged.subscribe(x => console.log(x));\n *\n * // Results in the following:\n * // - First timer1 and timer2 will run concurrently\n * // - timer1 will emit a value every 1000ms for 10 iterations\n * // - timer2 will emit a value every 2000ms for 6 iterations\n * // - after timer1 hits its max iteration, timer2 will\n * // continue, and timer3 will start to run concurrently with timer2\n * // - when timer2 hits its max iteration it terminates, and\n * // timer3 will continue to emit a value every 500ms until it is complete\n * ```\n *\n * @see {@link mergeAll}\n * @see {@link mergeMap}\n * @see {@link mergeMapTo}\n * @see {@link mergeScan}\n *\n * @param {...ObservableInput} observables Input Observables to merge together.\n * @param {number} [concurrent=Number.POSITIVE_INFINITY] Maximum number of input\n * Observables being subscribed to concurrently.\n * @param {SchedulerLike} [scheduler=null] The {@link SchedulerLike} to use for managing\n * concurrency of input Observables.\n * @return {Observable} an Observable that emits items that are the result of\n * every input Observable.\n * @static true\n * @name merge\n * @owner Observable\n */\nexport function merge(...observables: Array | SchedulerLike | number>): Observable {\n let concurrent = Number.POSITIVE_INFINITY;\n let scheduler: SchedulerLike = null;\n let last: any = observables[observables.length - 1];\n if (isScheduler(last)) {\n scheduler = observables.pop();\n if (observables.length > 1 && typeof observables[observables.length - 1] === 'number') {\n concurrent = observables.pop();\n }\n } else if (typeof last === 'number') {\n concurrent = observables.pop();\n }\n\n if (scheduler === null && observables.length === 1 && observables[0] instanceof Observable) {\n return >observables[0];\n }\n\n return mergeAll(concurrent)(fromArray(observables, scheduler));\n}\n","import { Operator } from '../Operator';\nimport { Subscriber } from '../Subscriber';\nimport { Observable } from '../Observable';\nimport { OperatorFunction, MonoTypeOperatorFunction, TeardownLogic } from '../types';\n\n/* tslint:disable:max-line-length */\nexport function filter(predicate: (value: T, index: number) => value is S,\n thisArg?: any): OperatorFunction;\nexport function filter(predicate: (value: T, index: number) => boolean,\n thisArg?: any): MonoTypeOperatorFunction;\n/* tslint:enable:max-line-length */\n\n/**\n * Filter items emitted by the source Observable by only emitting those that\n * satisfy a specified predicate.\n *\n * Like\n * [Array.prototype.filter()](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/filter),\n * it only emits a value from the source if it passes a criterion function.\n *\n * ![](filter.png)\n *\n * Similar to the well-known `Array.prototype.filter` method, this operator\n * takes values from the source Observable, passes them through a `predicate`\n * function and only emits those values that yielded `true`.\n *\n * ## Example\n * Emit only click events whose target was a DIV element\n * ```ts\n * import { fromEvent } from 'rxjs';\n * import { filter } from 'rxjs/operators';\n *\n * const clicks = fromEvent(document, 'click');\n * const clicksOnDivs = clicks.pipe(filter(ev => ev.target.tagName === 'DIV'));\n * clicksOnDivs.subscribe(x => console.log(x));\n * ```\n *\n * @see {@link distinct}\n * @see {@link distinctUntilChanged}\n * @see {@link distinctUntilKeyChanged}\n * @see {@link ignoreElements}\n * @see {@link partition}\n * @see {@link skip}\n *\n * @param {function(value: T, index: number): boolean} predicate A function that\n * evaluates each value emitted by the source Observable. If it returns `true`,\n * the value is emitted, if `false` the value is not passed to the output\n * Observable. The `index` parameter is the number `i` for the i-th source\n * emission that has happened since the subscription, starting from the number\n * `0`.\n * @param {any} [thisArg] An optional argument to determine the value of `this`\n * in the `predicate` function.\n * @return {Observable} An Observable of values from the source that were\n * allowed by the `predicate` function.\n * @method filter\n * @owner Observable\n */\nexport function filter(predicate: (value: T, index: number) => boolean,\n thisArg?: any): MonoTypeOperatorFunction {\n return function filterOperatorFunction(source: Observable): Observable {\n return source.lift(new FilterOperator(predicate, thisArg));\n };\n}\n\nclass FilterOperator implements Operator {\n constructor(private predicate: (value: T, index: number) => boolean,\n private thisArg?: any) {\n }\n\n call(subscriber: Subscriber, source: any): TeardownLogic {\n return source.subscribe(new FilterSubscriber(subscriber, this.predicate, this.thisArg));\n }\n}\n\n/**\n * We need this JSDoc comment for affecting ESDoc.\n * @ignore\n * @extends {Ignored}\n */\nclass FilterSubscriber extends Subscriber {\n\n count: number = 0;\n\n constructor(destination: Subscriber,\n private predicate: (value: T, index: number) => boolean,\n private thisArg: any) {\n super(destination);\n }\n\n // the try catch block below is left specifically for\n // optimization and perf reasons. a tryCatcher is not necessary here.\n protected _next(value: T) {\n let result: any;\n try {\n result = this.predicate.call(this.thisArg, value, this.count++);\n } catch (err) {\n this.destination.error(err);\n return;\n }\n if (result) {\n this.destination.next(value);\n }\n }\n}\n","import { Operator } from '../Operator';\nimport { Observable } from '../Observable';\nimport { Subscriber } from '../Subscriber';\nimport { Subscription } from '../Subscription';\nimport { async } from '../scheduler/async';\nimport { MonoTypeOperatorFunction, SchedulerLike, TeardownLogic } from '../types';\n\n/**\n * Emits a value from the source Observable only after a particular time span\n * has passed without another source emission.\n *\n * It's like {@link delay}, but passes only the most\n * recent value from each burst of emissions.\n *\n * ![](debounceTime.png)\n *\n * `debounceTime` delays values emitted by the source Observable, but drops\n * previous pending delayed emissions if a new value arrives on the source\n * Observable. This operator keeps track of the most recent value from the\n * source Observable, and emits that only when `dueTime` enough time has passed\n * without any other value appearing on the source Observable. If a new value\n * appears before `dueTime` silence occurs, the previous value will be dropped\n * and will not be emitted on the output Observable.\n *\n * This is a rate-limiting operator, because it is impossible for more than one\n * value to be emitted in any time window of duration `dueTime`, but it is also\n * a delay-like operator since output emissions do not occur at the same time as\n * they did on the source Observable. Optionally takes a {@link SchedulerLike} for\n * managing timers.\n *\n * ## Example\n * Emit the most recent click after a burst of clicks\n * ```ts\n * import { fromEvent } from 'rxjs';\n * import { debounceTime } from 'rxjs/operators';\n *\n * const clicks = fromEvent(document, 'click');\n * const result = clicks.pipe(debounceTime(1000));\n * result.subscribe(x => console.log(x));\n * ```\n *\n * @see {@link auditTime}\n * @see {@link debounce}\n * @see {@link delay}\n * @see {@link sampleTime}\n * @see {@link throttleTime}\n *\n * @param {number} dueTime The timeout duration in milliseconds (or the time\n * unit determined internally by the optional `scheduler`) for the window of\n * time required to wait for emission silence before emitting the most recent\n * source value.\n * @param {SchedulerLike} [scheduler=async] The {@link SchedulerLike} to use for\n * managing the timers that handle the timeout for each value.\n * @return {Observable} An Observable that delays the emissions of the source\n * Observable by the specified `dueTime`, and may drop some values if they occur\n * too frequently.\n * @method debounceTime\n * @owner Observable\n */\nexport function debounceTime(dueTime: number, scheduler: SchedulerLike = async): MonoTypeOperatorFunction {\n return (source: Observable) => source.lift(new DebounceTimeOperator(dueTime, scheduler));\n}\n\nclass DebounceTimeOperator implements Operator {\n constructor(private dueTime: number, private scheduler: SchedulerLike) {\n }\n\n call(subscriber: Subscriber, source: any): TeardownLogic {\n return source.subscribe(new DebounceTimeSubscriber(subscriber, this.dueTime, this.scheduler));\n }\n}\n\n/**\n * We need this JSDoc comment for affecting ESDoc.\n * @ignore\n * @extends {Ignored}\n */\nclass DebounceTimeSubscriber extends Subscriber {\n private debouncedSubscription: Subscription = null;\n private lastValue: T = null;\n private hasValue: boolean = false;\n\n constructor(destination: Subscriber,\n private dueTime: number,\n private scheduler: SchedulerLike) {\n super(destination);\n }\n\n protected _next(value: T) {\n this.clearDebounce();\n this.lastValue = value;\n this.hasValue = true;\n this.add(this.debouncedSubscription = this.scheduler.schedule(dispatchNext, this.dueTime, this));\n }\n\n protected _complete() {\n this.debouncedNext();\n this.destination.complete();\n }\n\n debouncedNext(): void {\n this.clearDebounce();\n\n if (this.hasValue) {\n const { lastValue } = this;\n // This must be done *before* passing the value\n // along to the destination because it's possible for\n // the value to synchronously re-enter this operator\n // recursively when scheduled with things like\n // VirtualScheduler/TestScheduler.\n this.lastValue = null;\n this.hasValue = false;\n this.destination.next(lastValue);\n }\n }\n\n private clearDebounce(): void {\n const debouncedSubscription = this.debouncedSubscription;\n\n if (debouncedSubscription !== null) {\n this.remove(debouncedSubscription);\n debouncedSubscription.unsubscribe();\n this.debouncedSubscription = null;\n }\n }\n}\n\nfunction dispatchNext(subscriber: DebounceTimeSubscriber) {\n subscriber.debouncedNext();\n}\n","import { Operator } from '../Operator';\nimport { Observable } from '../Observable';\nimport { Subscriber } from '../Subscriber';\nimport { OperatorFunction, MonoTypeOperatorFunction } from '../types';\n\n/* tslint:disable:max-line-length */\nexport function defaultIfEmpty(defaultValue?: T): MonoTypeOperatorFunction;\nexport function defaultIfEmpty(defaultValue?: R): OperatorFunction;\n/* tslint:enable:max-line-length */\n\n/**\n * Emits a given value if the source Observable completes without emitting any\n * `next` value, otherwise mirrors the source Observable.\n *\n * If the source Observable turns out to be empty, then\n * this operator will emit a default value.\n *\n * ![](defaultIfEmpty.png)\n *\n * `defaultIfEmpty` emits the values emitted by the source Observable or a\n * specified default value if the source Observable is empty (completes without\n * having emitted any `next` value).\n *\n * ## Example\n * If no clicks happen in 5 seconds, then emit \"no clicks\"\n * ```ts\n * import { fromEvent } from 'rxjs';\n * import { defaultIfEmpty, takeUntil } from 'rxjs/operators';\n *\n * const clicks = fromEvent(document, 'click');\n * const clicksBeforeFive = clicks.pipe(takeUntil(interval(5000)));\n * const result = clicksBeforeFive.pipe(defaultIfEmpty('no clicks'));\n * result.subscribe(x => console.log(x));\n * ```\n *\n * @see {@link empty}\n * @see {@link last}\n *\n * @param {any} [defaultValue=null] The default value used if the source\n * Observable is empty.\n * @return {Observable} An Observable that emits either the specified\n * `defaultValue` if the source Observable emits no items, or the values emitted\n * by the source Observable.\n * @method defaultIfEmpty\n * @owner Observable\n */\nexport function defaultIfEmpty(defaultValue: R = null): OperatorFunction {\n return (source: Observable) => source.lift(new DefaultIfEmptyOperator(defaultValue)) as Observable;\n}\n\nclass DefaultIfEmptyOperator implements Operator {\n\n constructor(private defaultValue: R) {\n }\n\n call(subscriber: Subscriber, source: any): any {\n return source.subscribe(new DefaultIfEmptySubscriber(subscriber, this.defaultValue));\n }\n}\n\n/**\n * We need this JSDoc comment for affecting ESDoc.\n * @ignore\n * @extends {Ignored}\n */\nclass DefaultIfEmptySubscriber extends Subscriber {\n private isEmpty: boolean = true;\n\n constructor(destination: Subscriber, private defaultValue: R) {\n super(destination);\n }\n\n protected _next(value: T): void {\n this.isEmpty = false;\n this.destination.next(value);\n }\n\n protected _complete(): void {\n if (this.isEmpty) {\n this.destination.next(this.defaultValue);\n }\n this.destination.complete();\n }\n}\n","import { Operator } from '../Operator';\nimport { Subscriber } from '../Subscriber';\nimport { Observable } from '../Observable';\nimport { MonoTypeOperatorFunction, TeardownLogic } from '../types';\n\n/* tslint:disable:max-line-length */\nexport function distinctUntilChanged(compare?: (x: T, y: T) => boolean): MonoTypeOperatorFunction;\nexport function distinctUntilChanged(compare: (x: K, y: K) => boolean, keySelector: (x: T) => K): MonoTypeOperatorFunction;\n/* tslint:enable:max-line-length */\n\n/**\n * Returns an Observable that emits all items emitted by the source Observable that are distinct by comparison from the previous item.\n *\n * If a comparator function is provided, then it will be called for each item to test for whether or not that value should be emitted.\n *\n * If a comparator function is not provided, an equality check is used by default.\n *\n * ## Example\n * A simple example with numbers\n * ```ts\n * import { of } from 'rxjs';\n * import { distinctUntilChanged } from 'rxjs/operators';\n *\n * of(1, 1, 2, 2, 2, 1, 1, 2, 3, 3, 4).pipe(\n * distinctUntilChanged(),\n * )\n * .subscribe(x => console.log(x)); // 1, 2, 1, 2, 3, 4\n * ```\n *\n * An example using a compare function\n * ```typescript\n * import { of } from 'rxjs';\n * import { distinctUntilChanged } from 'rxjs/operators';\n *\n * interface Person {\n * age: number,\n * name: string\n * }\n *\n * of(\n * { age: 4, name: 'Foo'},\n * { age: 7, name: 'Bar'},\n * { age: 5, name: 'Foo'},\n * { age: 6, name: 'Foo'},\n * ).pipe(\n * distinctUntilChanged((p: Person, q: Person) => p.name === q.name),\n * )\n * .subscribe(x => console.log(x));\n *\n * // displays:\n * // { age: 4, name: 'Foo' }\n * // { age: 7, name: 'Bar' }\n * // { age: 5, name: 'Foo' }\n * ```\n *\n * @see {@link distinct}\n * @see {@link distinctUntilKeyChanged}\n *\n * @param {function} [compare] Optional comparison function called to test if an item is distinct from the previous item in the source.\n * @return {Observable} An Observable that emits items from the source Observable with distinct values.\n * @method distinctUntilChanged\n * @owner Observable\n */\nexport function distinctUntilChanged(compare?: (x: K, y: K) => boolean, keySelector?: (x: T) => K): MonoTypeOperatorFunction {\n return (source: Observable) => source.lift(new DistinctUntilChangedOperator(compare, keySelector));\n}\n\nclass DistinctUntilChangedOperator implements Operator {\n constructor(private compare: (x: K, y: K) => boolean,\n private keySelector: (x: T) => K) {\n }\n\n call(subscriber: Subscriber, source: any): TeardownLogic {\n return source.subscribe(new DistinctUntilChangedSubscriber(subscriber, this.compare, this.keySelector));\n }\n}\n\n/**\n * We need this JSDoc comment for affecting ESDoc.\n * @ignore\n * @extends {Ignored}\n */\nclass DistinctUntilChangedSubscriber extends Subscriber {\n private key: K;\n private hasKey: boolean = false;\n\n constructor(destination: Subscriber,\n compare: (x: K, y: K) => boolean,\n private keySelector: (x: T) => K) {\n super(destination);\n if (typeof compare === 'function') {\n this.compare = compare;\n }\n }\n\n private compare(x: any, y: any): boolean {\n return x === y;\n }\n\n protected _next(value: T): void {\n let key: any;\n try {\n const { keySelector } = this;\n key = keySelector ? keySelector(value) : value;\n } catch (err) {\n return this.destination.error(err);\n }\n let result = false;\n if (this.hasKey) {\n try {\n const { compare } = this;\n result = compare(this.key, key);\n } catch (err) {\n return this.destination.error(err);\n }\n } else {\n this.hasKey = true;\n }\n if (!result) {\n this.key = key;\n this.destination.next(value);\n }\n }\n}\n","import { EmptyError } from '../util/EmptyError';\nimport { Observable } from '../Observable';\nimport { Operator } from '../Operator';\nimport { Subscriber } from '../Subscriber';\nimport { TeardownLogic, MonoTypeOperatorFunction } from '../types';\n\n/**\n * If the source observable completes without emitting a value, it will emit\n * an error. The error will be created at that time by the optional\n * `errorFactory` argument, otherwise, the error will be {@link EmptyError}.\n *\n * ![](throwIfEmpty.png)\n *\n * ## Example\n * ```ts\n * import { fromEvent, timer } from 'rxjs';\n * import { throwIfEmpty, takeUntil } from 'rxjs/operators';\n *\n * const click$ = fromEvent(document, 'click');\n *\n * click$.pipe(\n * takeUntil(timer(1000)),\n * throwIfEmpty(\n * () => new Error('the document was not clicked within 1 second')\n * ),\n * )\n * .subscribe({\n * next() { console.log('The button was clicked'); },\n * error(err) { console.error(err); }\n * });\n * ```\n *\n * @param errorFactory A factory function called to produce the\n * error to be thrown when the source observable completes without emitting a\n * value.\n */\nexport function throwIfEmpty (errorFactory: (() => any) = defaultErrorFactory): MonoTypeOperatorFunction {\n return (source: Observable) => {\n return source.lift(new ThrowIfEmptyOperator(errorFactory));\n };\n}\n\nclass ThrowIfEmptyOperator implements Operator {\n constructor(private errorFactory: () => any) {\n }\n\n call(subscriber: Subscriber, source: any): TeardownLogic {\n return source.subscribe(new ThrowIfEmptySubscriber(subscriber, this.errorFactory));\n }\n}\n\nclass ThrowIfEmptySubscriber extends Subscriber {\n private hasValue: boolean = false;\n\n constructor(destination: Subscriber, private errorFactory: () => any) {\n super(destination);\n }\n\n protected _next(value: T): void {\n this.hasValue = true;\n this.destination.next(value);\n }\n\n protected _complete() {\n if (!this.hasValue) {\n let err: any;\n try {\n err = this.errorFactory();\n } catch (e) {\n err = e;\n }\n this.destination.error(err);\n } else {\n return this.destination.complete();\n }\n }\n}\n\nfunction defaultErrorFactory() {\n return new EmptyError();\n}\n","import { Operator } from '../Operator';\nimport { Subscriber } from '../Subscriber';\nimport { ArgumentOutOfRangeError } from '../util/ArgumentOutOfRangeError';\nimport { empty } from '../observable/empty';\nimport { Observable } from '../Observable';\nimport { MonoTypeOperatorFunction, TeardownLogic } from '../types';\n\n/**\n * Emits only the first `count` values emitted by the source Observable.\n *\n * Takes the first `count` values from the source, then\n * completes.\n *\n * ![](take.png)\n *\n * `take` returns an Observable that emits only the first `count` values emitted\n * by the source Observable. If the source emits fewer than `count` values then\n * all of its values are emitted. After that, it completes, regardless if the\n * source completes.\n *\n * ## Example\n * Take the first 5 seconds of an infinite 1-second interval Observable\n * ```ts\n * import { interval } from 'rxjs';\n * import { take } from 'rxjs/operators';\n *\n * const intervalCount = interval(1000);\n * const takeFive = intervalCount.pipe(take(5));\n * takeFive.subscribe(x => console.log(x));\n *\n * // Logs:\n * // 0\n * // 1\n * // 2\n * // 3\n * // 4\n * ```\n *\n * @see {@link takeLast}\n * @see {@link takeUntil}\n * @see {@link takeWhile}\n * @see {@link skip}\n *\n * @throws {ArgumentOutOfRangeError} When using `take(i)`, it delivers an\n * ArgumentOutOrRangeError to the Observer's `error` callback if `i < 0`.\n *\n * @param {number} count The maximum number of `next` values to emit.\n * @return {Observable} An Observable that emits only the first `count`\n * values emitted by the source Observable, or all of the values from the source\n * if the source emits fewer than `count` values.\n * @method take\n * @owner Observable\n */\nexport function take(count: number): MonoTypeOperatorFunction {\n return (source: Observable) => {\n if (count === 0) {\n return empty();\n } else {\n return source.lift(new TakeOperator(count));\n }\n };\n}\n\nclass TakeOperator implements Operator {\n constructor(private total: number) {\n if (this.total < 0) {\n throw new ArgumentOutOfRangeError;\n }\n }\n\n call(subscriber: Subscriber, source: any): TeardownLogic {\n return source.subscribe(new TakeSubscriber(subscriber, this.total));\n }\n}\n\n/**\n * We need this JSDoc comment for affecting ESDoc.\n * @ignore\n * @extends {Ignored}\n */\nclass TakeSubscriber extends Subscriber {\n private count: number = 0;\n\n constructor(destination: Subscriber, private total: number) {\n super(destination);\n }\n\n protected _next(value: T): void {\n const total = this.total;\n const count = ++this.count;\n if (count <= total) {\n this.destination.next(value);\n if (count === total) {\n this.destination.complete();\n this.unsubscribe();\n }\n }\n }\n}\n","import { Operator } from '../Operator';\nimport { Subscriber } from '../Subscriber';\nimport { Subscription } from '../Subscription';\nimport { Observable } from '../Observable';\nimport { MonoTypeOperatorFunction, TeardownLogic } from '../types';\n\n/**\n * Returns an Observable that mirrors the source Observable, but will call a specified function when\n * the source terminates on complete or error.\n * @param {function} callback Function to be called when source terminates.\n * @return {Observable} An Observable that mirrors the source, but will call the specified function on termination.\n * @method finally\n * @owner Observable\n */\nexport function finalize(callback: () => void): MonoTypeOperatorFunction {\n return (source: Observable) => source.lift(new FinallyOperator(callback));\n}\n\nclass FinallyOperator implements Operator {\n constructor(private callback: () => void) {\n }\n\n call(subscriber: Subscriber, source: any): TeardownLogic {\n return source.subscribe(new FinallySubscriber(subscriber, this.callback));\n }\n}\n\n/**\n * We need this JSDoc comment for affecting ESDoc.\n * @ignore\n * @extends {Ignored}\n */\nclass FinallySubscriber extends Subscriber {\n constructor(destination: Subscriber, callback: () => void) {\n super(destination);\n this.add(new Subscription(callback));\n }\n}\n","import { Subject } from '../Subject';\nimport { Operator } from '../Operator';\nimport { Subscriber } from '../Subscriber';\nimport { Observable } from '../Observable';\nimport { ConnectableObservable, connectableObservableDescriptor } from '../observable/ConnectableObservable';\nimport { MonoTypeOperatorFunction, OperatorFunction, UnaryFunction, ObservedValueOf, ObservableInput } from '../types';\n\n/* tslint:disable:max-line-length */\nexport function multicast(subject: Subject): UnaryFunction, ConnectableObservable>;\nexport function multicast>(subject: Subject, selector: (shared: Observable) => O): UnaryFunction, ConnectableObservable>>;\nexport function multicast(subjectFactory: (this: Observable) => Subject): UnaryFunction, ConnectableObservable>;\nexport function multicast>(SubjectFactory: (this: Observable) => Subject, selector: (shared: Observable) => O): OperatorFunction>;\n/* tslint:enable:max-line-length */\n\n/**\n * Returns an Observable that emits the results of invoking a specified selector on items\n * emitted by a ConnectableObservable that shares a single subscription to the underlying stream.\n *\n * ![](multicast.png)\n *\n * @param {Function|Subject} subjectOrSubjectFactory - Factory function to create an intermediate subject through\n * which the source sequence's elements will be multicast to the selector function\n * or Subject to push source elements into.\n * @param {Function} [selector] - Optional selector function that can use the multicasted source stream\n * as many times as needed, without causing multiple subscriptions to the source stream.\n * Subscribers to the given source will receive all notifications of the source from the\n * time of the subscription forward.\n * @return {Observable} An Observable that emits the results of invoking the selector\n * on the items emitted by a `ConnectableObservable` that shares a single subscription to\n * the underlying stream.\n * @method multicast\n * @owner Observable\n */\nexport function multicast(subjectOrSubjectFactory: Subject | (() => Subject),\n selector?: (source: Observable) => Observable): OperatorFunction {\n return function multicastOperatorFunction(source: Observable): Observable {\n let subjectFactory: () => Subject;\n if (typeof subjectOrSubjectFactory === 'function') {\n subjectFactory = <() => Subject>subjectOrSubjectFactory;\n } else {\n subjectFactory = function subjectFactory() {\n return >subjectOrSubjectFactory;\n };\n }\n\n if (typeof selector === 'function') {\n return source.lift(new MulticastOperator(subjectFactory, selector));\n }\n\n const connectable: any = Object.create(source, connectableObservableDescriptor);\n connectable.source = source;\n connectable.subjectFactory = subjectFactory;\n\n return > connectable;\n };\n}\n\nexport class MulticastOperator implements Operator {\n constructor(private subjectFactory: () => Subject,\n private selector: (source: Observable) => Observable) {\n }\n call(subscriber: Subscriber, source: any): any {\n const { selector } = this;\n const subject = this.subjectFactory();\n const subscription = selector(subject).subscribe(subscriber);\n subscription.add(source.subscribe(subject));\n return subscription;\n }\n}\n","import { Observable } from '../Observable';\nimport { multicast } from './multicast';\nimport { refCount } from './refCount';\nimport { Subject } from '../Subject';\n\nimport { MonoTypeOperatorFunction } from '../types';\n\nfunction shareSubjectFactory() {\n return new Subject();\n}\n\n/**\n * Returns a new Observable that multicasts (shares) the original Observable. As long as there is at least one\n * Subscriber this Observable will be subscribed and emitting data. When all subscribers have unsubscribed it will\n * unsubscribe from the source Observable. Because the Observable is multicasting it makes the stream `hot`.\n * This is an alias for `multicast(() => new Subject()), refCount()`.\n *\n * ![](share.png)\n *\n * @return {Observable} An Observable that upon connection causes the source Observable to emit items to its Observers.\n * @method share\n * @owner Observable\n */\nexport function share(): MonoTypeOperatorFunction {\n return (source: Observable) => refCount()(multicast(shareSubjectFactory)(source)) as Observable;\n}\n","import { Observable } from '../Observable';\nimport { ReplaySubject } from '../ReplaySubject';\nimport { Subscription } from '../Subscription';\nimport { MonoTypeOperatorFunction, SchedulerLike } from '../types';\nimport { Subscriber } from '../Subscriber';\n\nexport interface ShareReplayConfig {\n bufferSize?: number;\n windowTime?: number;\n refCount: boolean;\n scheduler?: SchedulerLike;\n}\n\n/**\n * Share source and replay specified number of emissions on subscription.\n *\n * This operator is a specialization of `replay` that connects to a source observable\n * and multicasts through a `ReplaySubject` constructed with the specified arguments.\n * A successfully completed source will stay cached in the `shareReplayed observable` forever,\n * but an errored source can be retried.\n *\n * ## Why use shareReplay?\n * You generally want to use `shareReplay` when you have side-effects or taxing computations\n * that you do not wish to be executed amongst multiple subscribers.\n * It may also be valuable in situations where you know you will have late subscribers to\n * a stream that need access to previously emitted values.\n * This ability to replay values on subscription is what differentiates {@link share} and `shareReplay`.\n *\n * ![](shareReplay.png)\n *\n * ## Example\n * ```ts\n * import { interval } from 'rxjs';\n * import { shareReplay, take } from 'rxjs/operators';\n *\n * const obs$ = interval(1000);\n * const shared$ = obs$.pipe(\n * take(4),\n * shareReplay(3)\n * );\n * shared$.subscribe(x => console.log('source A: ', x));\n * shared$.subscribe(y => console.log('source B: ', y));\n *\n * ```\n *\n * @see {@link publish}\n * @see {@link share}\n * @see {@link publishReplay}\n *\n * @param {Number} [bufferSize=Number.POSITIVE_INFINITY] Maximum element count of the replay buffer.\n * @param {Number} [windowTime=Number.POSITIVE_INFINITY] Maximum time length of the replay buffer in milliseconds.\n * @param {Scheduler} [scheduler] Scheduler where connected observers within the selector function\n * will be invoked on.\n * @return {Observable} An observable sequence that contains the elements of a sequence produced\n * by multicasting the source sequence within a selector function.\n * @method shareReplay\n * @owner Observable\n */\nexport function shareReplay(\n config: ShareReplayConfig\n): MonoTypeOperatorFunction;\nexport function shareReplay(\n bufferSize?: number,\n windowTime?: number,\n scheduler?: SchedulerLike\n): MonoTypeOperatorFunction;\nexport function shareReplay(\n configOrBufferSize?: ShareReplayConfig | number,\n windowTime?: number,\n scheduler?: SchedulerLike\n): MonoTypeOperatorFunction {\n let config: ShareReplayConfig;\n if (configOrBufferSize && typeof configOrBufferSize === 'object') {\n config = configOrBufferSize as ShareReplayConfig;\n } else {\n config = {\n bufferSize: configOrBufferSize as number | undefined,\n windowTime,\n refCount: false,\n scheduler,\n };\n }\n return (source: Observable) => source.lift(shareReplayOperator(config));\n}\n\nfunction shareReplayOperator({\n bufferSize = Number.POSITIVE_INFINITY,\n windowTime = Number.POSITIVE_INFINITY,\n refCount: useRefCount,\n scheduler,\n}: ShareReplayConfig) {\n let subject: ReplaySubject | undefined;\n let refCount = 0;\n let subscription: Subscription | undefined;\n let hasError = false;\n let isComplete = false;\n\n return function shareReplayOperation(\n this: Subscriber,\n source: Observable\n ) {\n refCount++;\n let innerSub: Subscription;\n if (!subject || hasError) {\n hasError = false;\n subject = new ReplaySubject(bufferSize, windowTime, scheduler);\n innerSub = subject.subscribe(this);\n subscription = source.subscribe({\n next(value) {\n subject.next(value);\n },\n error(err) {\n hasError = true;\n subject.error(err);\n },\n complete() {\n isComplete = true;\n subscription = undefined;\n subject.complete();\n },\n });\n\n // Here we need to check to see if the source synchronously completed. Although\n // we're setting `subscription = undefined` in the completion handler, if the source\n // is synchronous, that will happen *before* subscription is set by the return of\n // the `subscribe` call.\n if (isComplete) {\n subscription = undefined;\n }\n } else {\n innerSub = subject.subscribe(this);\n }\n\n this.add(() => {\n refCount--;\n innerSub.unsubscribe();\n innerSub = undefined;\n if (subscription && !isComplete && useRefCount && refCount === 0) {\n subscription.unsubscribe();\n subscription = undefined;\n subject = undefined;\n }\n });\n };\n}\n","import { Operator } from '../Operator';\nimport { Observable } from '../Observable';\nimport { Subscriber } from '../Subscriber';\nimport { Subscription } from '../Subscription';\nimport { ObservableInput, OperatorFunction, ObservedValueOf } from '../types';\nimport { map } from './map';\nimport { from } from '../observable/from';\nimport { SimpleOuterSubscriber, SimpleInnerSubscriber, innerSubscribe } from '../innerSubscribe';\n\n/* tslint:disable:max-line-length */\nexport function switchMap>(project: (value: T, index: number) => O): OperatorFunction>;\n/** @deprecated resultSelector is no longer supported, use inner map instead */\nexport function switchMap>(project: (value: T, index: number) => O, resultSelector: undefined): OperatorFunction>;\n/** @deprecated resultSelector is no longer supported, use inner map instead */\nexport function switchMap>(project: (value: T, index: number) => O, resultSelector: (outerValue: T, innerValue: ObservedValueOf, outerIndex: number, innerIndex: number) => R): OperatorFunction;\n/* tslint:enable:max-line-length */\n\n/**\n * Projects each source value to an Observable which is merged in the output\n * Observable, emitting values only from the most recently projected Observable.\n *\n * Maps each value to an Observable, then flattens all of\n * these inner Observables.\n *\n * ![](switchMap.png)\n *\n * Returns an Observable that emits items based on applying a function that you\n * supply to each item emitted by the source Observable, where that function\n * returns an (so-called \"inner\") Observable. Each time it observes one of these\n * inner Observables, the output Observable begins emitting the items emitted by\n * that inner Observable. When a new inner Observable is emitted, `switchMap`\n * stops emitting items from the earlier-emitted inner Observable and begins\n * emitting items from the new one. It continues to behave like this for\n * subsequent inner Observables.\n *\n * ## Example\n * Generate new Observable according to source Observable values\n * ```typescript\n * import { of } from 'rxjs';\n * import { switchMap } from 'rxjs/operators';\n *\n * const switched = of(1, 2, 3).pipe(switchMap((x: number) => of(x, x ** 2, x ** 3)));\n * switched.subscribe(x => console.log(x));\n * // outputs\n * // 1\n * // 1\n * // 1\n * // 2\n * // 4\n * // 8\n * // ... and so on\n * ```\n *\n * Rerun an interval Observable on every click event\n * ```ts\n * import { fromEvent, interval } from 'rxjs';\n * import { switchMap } from 'rxjs/operators';\n *\n * const clicks = fromEvent(document, 'click');\n * const result = clicks.pipe(switchMap((ev) => interval(1000)));\n * result.subscribe(x => console.log(x));\n * ```\n *\n * @see {@link concatMap}\n * @see {@link exhaustMap}\n * @see {@link mergeMap}\n * @see {@link switchAll}\n * @see {@link switchMapTo}\n *\n * @param {function(value: T, ?index: number): ObservableInput} project A function\n * that, when applied to an item emitted by the source Observable, returns an\n * Observable.\n * @return {Observable} An Observable that emits the result of applying the\n * projection function (and the optional deprecated `resultSelector`) to each item\n * emitted by the source Observable and taking only the values from the most recently\n * projected inner Observable.\n * @method switchMap\n * @owner Observable\n */\nexport function switchMap>(\n project: (value: T, index: number) => O,\n resultSelector?: (outerValue: T, innerValue: ObservedValueOf, outerIndex: number, innerIndex: number) => R,\n): OperatorFunction|R> {\n if (typeof resultSelector === 'function') {\n return (source: Observable) => source.pipe(\n switchMap((a, i) => from(project(a, i)).pipe(\n map((b, ii) => resultSelector(a, b, i, ii))\n ))\n );\n }\n return (source: Observable) => source.lift(new SwitchMapOperator(project));\n}\n\nclass SwitchMapOperator implements Operator {\n constructor(private project: (value: T, index: number) => ObservableInput) {\n }\n\n call(subscriber: Subscriber, source: any): any {\n return source.subscribe(new SwitchMapSubscriber(subscriber, this.project));\n }\n}\n\n/**\n * We need this JSDoc comment for affecting ESDoc.\n * @ignore\n * @extends {Ignored}\n */\nclass SwitchMapSubscriber extends SimpleOuterSubscriber {\n private index = 0;\n private innerSubscription?: Subscription;\n\n constructor(destination: Subscriber,\n private project: (value: T, index: number) => ObservableInput) {\n super(destination);\n }\n\n protected _next(value: T) {\n let result: ObservableInput;\n const index = this.index++;\n try {\n result = this.project(value, index);\n } catch (error) {\n this.destination.error!(error);\n return;\n }\n this._innerSub(result);\n }\n\n private _innerSub(result: ObservableInput) {\n const innerSubscription = this.innerSubscription;\n if (innerSubscription) {\n innerSubscription.unsubscribe();\n }\n const innerSubscriber = new SimpleInnerSubscriber(this);\n const destination = this.destination as Subscription;\n destination.add(innerSubscriber);\n this.innerSubscription = innerSubscribe(result, innerSubscriber);\n // The returned subscription will usually be the subscriber that was\n // passed. However, interop subscribers will be wrapped and for\n // unsubscriptions to chain correctly, the wrapper needs to be added, too.\n if (this.innerSubscription !== innerSubscriber) {\n destination.add(this.innerSubscription);\n }\n }\n\n protected _complete(): void {\n const {innerSubscription} = this;\n if (!innerSubscription || innerSubscription.closed) {\n super._complete();\n }\n this.unsubscribe();\n }\n\n protected _unsubscribe() {\n this.innerSubscription = undefined;\n }\n\n notifyComplete(): void {\n this.innerSubscription = undefined;\n if (this.isStopped) {\n super._complete();\n }\n }\n\n notifyNext(innerValue: R): void {\n this.destination.next!(innerValue);\n }\n}\n","import { Operator } from '../Operator';\nimport { Observable } from '../Observable';\nimport { Subscriber } from '../Subscriber';\n\nimport { MonoTypeOperatorFunction, TeardownLogic } from '../types';\nimport { innerSubscribe, SimpleInnerSubscriber, SimpleOuterSubscriber } from '../innerSubscribe';\n\n/**\n * Emits the values emitted by the source Observable until a `notifier`\n * Observable emits a value.\n *\n * Lets values pass until a second Observable,\n * `notifier`, emits a value. Then, it completes.\n *\n * ![](takeUntil.png)\n *\n * `takeUntil` subscribes and begins mirroring the source Observable. It also\n * monitors a second Observable, `notifier` that you provide. If the `notifier`\n * emits a value, the output Observable stops mirroring the source Observable\n * and completes. If the `notifier` doesn't emit any value and completes\n * then `takeUntil` will pass all values.\n *\n * ## Example\n * Tick every second until the first click happens\n * ```ts\n * import { fromEvent, interval } from 'rxjs';\n * import { takeUntil } from 'rxjs/operators';\n *\n * const source = interval(1000);\n * const clicks = fromEvent(document, 'click');\n * const result = source.pipe(takeUntil(clicks));\n * result.subscribe(x => console.log(x));\n * ```\n *\n * @see {@link take}\n * @see {@link takeLast}\n * @see {@link takeWhile}\n * @see {@link skip}\n *\n * @param {Observable} notifier The Observable whose first emitted value will\n * cause the output Observable of `takeUntil` to stop emitting values from the\n * source Observable.\n * @return {Observable} An Observable that emits the values from the source\n * Observable until such time as `notifier` emits its first value.\n * @method takeUntil\n * @owner Observable\n */\nexport function takeUntil(notifier: Observable): MonoTypeOperatorFunction {\n return (source: Observable) => source.lift(new TakeUntilOperator(notifier));\n}\n\nclass TakeUntilOperator implements Operator {\n constructor(private notifier: Observable) {\n }\n\n call(subscriber: Subscriber, source: any): TeardownLogic {\n const takeUntilSubscriber = new TakeUntilSubscriber(subscriber);\n const notifierSubscription = innerSubscribe(this.notifier, new SimpleInnerSubscriber(takeUntilSubscriber));\n if (notifierSubscription && !takeUntilSubscriber.seenValue) {\n takeUntilSubscriber.add(notifierSubscription);\n return source.subscribe(takeUntilSubscriber);\n }\n return takeUntilSubscriber;\n }\n}\n\n/**\n * We need this JSDoc comment for affecting ESDoc.\n * @ignore\n * @extends {Ignored}\n */\nclass TakeUntilSubscriber extends SimpleOuterSubscriber {\n seenValue = false;\n\n constructor(destination: Subscriber, ) {\n super(destination);\n }\n\n notifyNext(): void {\n this.seenValue = true;\n this.complete();\n }\n\n notifyComplete(): void {\n // noop\n }\n}\n","import { Operator } from '../Operator';\nimport { Observable } from '../Observable';\nimport { Subscriber } from '../Subscriber';\nimport { OperatorFunction, MonoTypeOperatorFunction, TeardownLogic } from '../types';\n\nexport function takeWhile(predicate: (value: T, index: number) => value is S): OperatorFunction;\nexport function takeWhile(predicate: (value: T, index: number) => value is S, inclusive: false): OperatorFunction;\nexport function takeWhile(predicate: (value: T, index: number) => boolean, inclusive?: boolean): MonoTypeOperatorFunction;\n\n/**\n * Emits values emitted by the source Observable so long as each value satisfies\n * the given `predicate`, and then completes as soon as this `predicate` is not\n * satisfied.\n *\n * Takes values from the source only while they pass the\n * condition given. When the first value does not satisfy, it completes.\n *\n * ![](takeWhile.png)\n *\n * `takeWhile` subscribes and begins mirroring the source Observable. Each value\n * emitted on the source is given to the `predicate` function which returns a\n * boolean, representing a condition to be satisfied by the source values. The\n * output Observable emits the source values until such time as the `predicate`\n * returns false, at which point `takeWhile` stops mirroring the source\n * Observable and completes the output Observable.\n *\n * ## Example\n * Emit click events only while the clientX property is greater than 200\n * ```ts\n * import { fromEvent } from 'rxjs';\n * import { takeWhile } from 'rxjs/operators';\n *\n * const clicks = fromEvent(document, 'click');\n * const result = clicks.pipe(takeWhile(ev => ev.clientX > 200));\n * result.subscribe(x => console.log(x));\n * ```\n *\n * @see {@link take}\n * @see {@link takeLast}\n * @see {@link takeUntil}\n * @see {@link skip}\n *\n * @param {function(value: T, index: number): boolean} predicate A function that\n * evaluates a value emitted by the source Observable and returns a boolean.\n * Also takes the (zero-based) index as the second argument.\n * @param {boolean} inclusive When set to `true` the value that caused\n * `predicate` to return `false` will also be emitted.\n * @return {Observable} An Observable that emits the values from the source\n * Observable so long as each value satisfies the condition defined by the\n * `predicate`, then completes.\n * @method takeWhile\n * @owner Observable\n */\nexport function takeWhile(\n predicate: (value: T, index: number) => boolean,\n inclusive = false): MonoTypeOperatorFunction {\n return (source: Observable) =>\n source.lift(new TakeWhileOperator(predicate, inclusive));\n}\n\nclass TakeWhileOperator implements Operator {\n constructor(\n private predicate: (value: T, index: number) => boolean,\n private inclusive: boolean) {}\n\n call(subscriber: Subscriber, source: any): TeardownLogic {\n return source.subscribe(\n new TakeWhileSubscriber(subscriber, this.predicate, this.inclusive));\n }\n}\n\n/**\n * We need this JSDoc comment for affecting ESDoc.\n * @ignore\n * @extends {Ignored}\n */\nclass TakeWhileSubscriber extends Subscriber {\n private index: number = 0;\n\n constructor(\n destination: Subscriber,\n private predicate: (value: T, index: number) => boolean,\n private inclusive: boolean) {\n super(destination);\n }\n\n protected _next(value: T): void {\n const destination = this.destination;\n let result: boolean;\n try {\n result = this.predicate(value, this.index++);\n } catch (err) {\n destination.error(err);\n return;\n }\n this.nextOrComplete(value, result);\n }\n\n private nextOrComplete(value: T, predicateResult: boolean): void {\n const destination = this.destination;\n if (Boolean(predicateResult)) {\n destination.next(value);\n } else {\n if (this.inclusive) {\n destination.next(value);\n }\n destination.complete();\n }\n }\n}\n","import { Operator } from '../Operator';\nimport { Observable } from '../Observable';\nimport { Subscriber } from '../Subscriber';\nimport { Subscription } from '../Subscription';\n\nimport { MonoTypeOperatorFunction, SubscribableOrPromise, TeardownLogic } from '../types';\nimport { SimpleOuterSubscriber, innerSubscribe, SimpleInnerSubscriber } from '../innerSubscribe';\n\nexport interface ThrottleConfig {\n leading?: boolean;\n trailing?: boolean;\n}\n\nexport const defaultThrottleConfig: ThrottleConfig = {\n leading: true,\n trailing: false\n};\n\n/**\n * Emits a value from the source Observable, then ignores subsequent source\n * values for a duration determined by another Observable, then repeats this\n * process.\n *\n * It's like {@link throttleTime}, but the silencing\n * duration is determined by a second Observable.\n *\n * ![](throttle.png)\n *\n * `throttle` emits the source Observable values on the output Observable\n * when its internal timer is disabled, and ignores source values when the timer\n * is enabled. Initially, the timer is disabled. As soon as the first source\n * value arrives, it is forwarded to the output Observable, and then the timer\n * is enabled by calling the `durationSelector` function with the source value,\n * which returns the \"duration\" Observable. When the duration Observable emits a\n * value or completes, the timer is disabled, and this process repeats for the\n * next source value.\n *\n * ## Example\n * Emit clicks at a rate of at most one click per second\n * ```ts\n * import { fromEvent } from 'rxjs';\n * import { throttle } from 'rxjs/operators';\n *\n * const clicks = fromEvent(document, 'click');\n * const result = clicks.pipe(throttle(ev => interval(1000)));\n * result.subscribe(x => console.log(x));\n * ```\n *\n * @see {@link audit}\n * @see {@link debounce}\n * @see {@link delayWhen}\n * @see {@link sample}\n * @see {@link throttleTime}\n *\n * @param {function(value: T): SubscribableOrPromise} durationSelector A function\n * that receives a value from the source Observable, for computing the silencing\n * duration for each source value, returned as an Observable or a Promise.\n * @param {Object} config a configuration object to define `leading` and `trailing` behavior. Defaults\n * to `{ leading: true, trailing: false }`.\n * @return {Observable} An Observable that performs the throttle operation to\n * limit the rate of emissions from the source.\n * @method throttle\n * @owner Observable\n */\nexport function throttle(durationSelector: (value: T) => SubscribableOrPromise,\n config: ThrottleConfig = defaultThrottleConfig): MonoTypeOperatorFunction {\n return (source: Observable) => source.lift(new ThrottleOperator(durationSelector, !!config.leading, !!config.trailing));\n}\n\nclass ThrottleOperator implements Operator {\n constructor(private durationSelector: (value: T) => SubscribableOrPromise,\n private leading: boolean,\n private trailing: boolean) {\n }\n\n call(subscriber: Subscriber, source: any): TeardownLogic {\n return source.subscribe(\n new ThrottleSubscriber(subscriber, this.durationSelector, this.leading, this.trailing)\n );\n }\n}\n\n/**\n * We need this JSDoc comment for affecting ESDoc\n * @ignore\n * @extends {Ignored}\n */\nclass ThrottleSubscriber extends SimpleOuterSubscriber {\n private _throttled?: Subscription;\n private _sendValue?: T;\n private _hasValue = false;\n\n constructor(protected destination: Subscriber,\n private durationSelector: (value: T) => SubscribableOrPromise,\n private _leading: boolean,\n private _trailing: boolean) {\n super(destination);\n }\n\n protected _next(value: T): void {\n this._hasValue = true;\n this._sendValue = value;\n\n if (!this._throttled) {\n if (this._leading) {\n this.send();\n } else {\n this.throttle(value);\n }\n }\n }\n\n private send() {\n const { _hasValue, _sendValue } = this;\n if (_hasValue) {\n this.destination.next(_sendValue);\n this.throttle(_sendValue!);\n }\n this._hasValue = false;\n this._sendValue = undefined;\n }\n\n private throttle(value: T): void {\n const duration = this.tryDurationSelector(value);\n if (!!duration) {\n this.add(this._throttled = innerSubscribe(duration, new SimpleInnerSubscriber(this)));\n }\n }\n\n private tryDurationSelector(value: T): SubscribableOrPromise | null {\n try {\n return this.durationSelector(value);\n } catch (err) {\n this.destination.error(err);\n return null;\n }\n }\n\n private throttlingDone() {\n const { _throttled, _trailing } = this;\n if (_throttled) {\n _throttled.unsubscribe();\n }\n this._throttled = undefined;\n\n if (_trailing) {\n this.send();\n }\n }\n\n notifyNext(): void {\n this.throttlingDone();\n }\n\n notifyComplete(): void {\n this.throttlingDone();\n }\n}\n","import { Operator } from '../Operator';\nimport { Subscriber } from '../Subscriber';\nimport { Subscription } from '../Subscription';\nimport { async } from '../scheduler/async';\nimport { Observable } from '../Observable';\nimport { ThrottleConfig, defaultThrottleConfig } from './throttle';\nimport { MonoTypeOperatorFunction, SchedulerLike, TeardownLogic } from '../types';\n\n/**\n * Emits a value from the source Observable, then ignores subsequent source\n * values for `duration` milliseconds, then repeats this process.\n *\n * Lets a value pass, then ignores source values for the\n * next `duration` milliseconds.\n *\n * ![](throttleTime.png)\n *\n * `throttleTime` emits the source Observable values on the output Observable\n * when its internal timer is disabled, and ignores source values when the timer\n * is enabled. Initially, the timer is disabled. As soon as the first source\n * value arrives, it is forwarded to the output Observable, and then the timer\n * is enabled. After `duration` milliseconds (or the time unit determined\n * internally by the optional `scheduler`) has passed, the timer is disabled,\n * and this process repeats for the next source value. Optionally takes a\n * {@link SchedulerLike} for managing timers.\n *\n * ## Examples\n *\n * #### Limit click rate\n *\n * Emit clicks at a rate of at most one click per second\n * ```ts\n * import { fromEvent } from 'rxjs';\n * import { throttleTime } from 'rxjs/operators';\n *\n * const clicks = fromEvent(document, 'click');\n * const result = clicks.pipe(throttleTime(1000));\n * result.subscribe(x => console.log(x));\n * ```\n *\n * #### Double Click\n *\n * The following example only emits clicks which happen within a subsequent\n * delay of 400ms of the previous click. This for example can emulate a double\n * click. It makes use of the `trailing` parameter of the throttle configuration.\n *\n * ```ts\n * import { fromEvent, asyncScheduler } from 'rxjs';\n * import { throttleTime, withLatestFrom } from 'rxjs/operators';\n *\n * // defaultThottleConfig = { leading: true, trailing: false }\n * const throttleConfig = {\n * leading: false,\n * trailing: true\n * }\n *\n * const click = fromEvent(document, 'click');\n * const doubleClick = click.pipe(\n * throttleTime(400, asyncScheduler, throttleConfig)\n * );\n *\n * doubleClick.subscribe((throttleValue: Event) => {\n * console.log(`Double-clicked! Timestamp: ${throttleValue.timeStamp}`);\n * });\n * ```\n *\n * If you enable the `leading` parameter in this example, the output would be the primary click and\n * the double click, but restricts additional clicks within 400ms.\n *\n * @see {@link auditTime}\n * @see {@link debounceTime}\n * @see {@link delay}\n * @see {@link sampleTime}\n * @see {@link throttle}\n *\n * @param {number} duration Time to wait before emitting another value after\n * emitting the last value, measured in milliseconds or the time unit determined\n * internally by the optional `scheduler`.\n * @param {SchedulerLike} [scheduler=async] The {@link SchedulerLike} to use for\n * managing the timers that handle the throttling.\n * @param {Object} config a configuration object to define `leading` and\n * `trailing` behavior. Defaults to `{ leading: true, trailing: false }`.\n * @return {Observable} An Observable that performs the throttle operation to\n * limit the rate of emissions from the source.\n * @method throttleTime\n * @owner Observable\n */\nexport function throttleTime(duration: number,\n scheduler: SchedulerLike = async,\n config: ThrottleConfig = defaultThrottleConfig): MonoTypeOperatorFunction {\n return (source: Observable) => source.lift(new ThrottleTimeOperator(duration, scheduler, config.leading, config.trailing));\n}\n\nclass ThrottleTimeOperator implements Operator {\n constructor(private duration: number,\n private scheduler: SchedulerLike,\n private leading: boolean,\n private trailing: boolean) {\n }\n\n call(subscriber: Subscriber, source: any): TeardownLogic {\n return source.subscribe(\n new ThrottleTimeSubscriber(subscriber, this.duration, this.scheduler, this.leading, this.trailing)\n );\n }\n}\n\n/**\n * We need this JSDoc comment for affecting ESDoc.\n * @ignore\n * @extends {Ignored}\n */\nclass ThrottleTimeSubscriber extends Subscriber {\n private throttled: Subscription;\n private _hasTrailingValue: boolean = false;\n private _trailingValue: T = null;\n\n constructor(destination: Subscriber,\n private duration: number,\n private scheduler: SchedulerLike,\n private leading: boolean,\n private trailing: boolean) {\n super(destination);\n }\n\n protected _next(value: T) {\n if (this.throttled) {\n if (this.trailing) {\n this._trailingValue = value;\n this._hasTrailingValue = true;\n }\n } else {\n this.add(this.throttled = this.scheduler.schedule>(dispatchNext, this.duration, { subscriber: this }));\n if (this.leading) {\n this.destination.next(value);\n } else if (this.trailing) {\n this._trailingValue = value;\n this._hasTrailingValue = true;\n }\n }\n }\n\n protected _complete() {\n if (this._hasTrailingValue) {\n this.destination.next(this._trailingValue);\n this.destination.complete();\n } else {\n this.destination.complete();\n }\n }\n\n clearThrottle() {\n const throttled = this.throttled;\n if (throttled) {\n if (this.trailing && this._hasTrailingValue) {\n this.destination.next(this._trailingValue);\n this._trailingValue = null;\n this._hasTrailingValue = false;\n }\n throttled.unsubscribe();\n this.remove(throttled);\n this.throttled = null;\n }\n }\n}\n\ninterface DispatchArg {\n subscriber: ThrottleTimeSubscriber;\n}\n\nfunction dispatchNext(arg: DispatchArg) {\n const { subscriber } = arg;\n subscriber.clearThrottle();\n}\n","import { Subject } from 'rxjs';\r\n\r\nexport function isDefined(item: T | undefined | null): item is T {\r\n return typeof item !== 'undefined' && item !== null;\r\n}\r\n\r\nexport function isObject(o: any) {\r\n return typeof o === 'object' && o != null;\r\n}\r\n\r\n/** Type guard for filtering out undefined */\r\nexport function omitUndefinedElements(item: T): item is NonNullable {\r\n return item !== null && typeof item !== 'undefined';\r\n}\r\n\r\nexport const changesProcessed = new Subject();\r\n\r\nexport const guidRegex = /^[0-9a-f]{8}-[0-9a-f]{4}-[1-5][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$/i;\r\n\r\nexport function isGuid(str: string) {\r\n return guidRegex.test(str);\r\n}\r\n\r\nexport const absoluteUrlRegex = /^(?:[a-z]+:)?\\/\\//i;\r\nexport function isAbsoluteUrl(url: string): boolean {\r\n return absoluteUrlRegex.test(url);\r\n}\r\n\r\nexport function appendOrIncrementNumber(name: string) {\r\n const lastNumber = name.replace(/\\D*(\\d*)\\s*$/, '$1');\r\n if (lastNumber === '') {\r\n return name + ' 2';\r\n } else {\r\n return name.replace(/\\d*\\s*$/, (parseInt(lastNumber, 10) + 1).toString());\r\n }\r\n}\r\n\r\n/** UUID generation based on: https://github.com/jchook/uuid-random/blob/master/index.js */\r\nexport class UUID {\r\n private static buf: Uint8Array;\r\n private static bufIdx = 0;\r\n private static hexBytes = new Array();\r\n private static i: number;\r\n private static bufferSize = 4096;\r\n\r\n public static initialize() {\r\n // Pre-calculate toString(16) for speed\r\n for (UUID.i = 0; UUID.i < 256; UUID.i++) {\r\n UUID.hexBytes[UUID.i] = (UUID.i + 0x100).toString(16).substr(1);\r\n }\r\n }\r\n\r\n // Buffer some random bytes for speed\r\n protected static randomBytesBuffered(n: number) {\r\n if (!UUID.buf || UUID.bufIdx + n > UUID.bufferSize) {\r\n UUID.bufIdx = 0;\r\n UUID.buf = crypto.getRandomValues(new Uint8Array(UUID.bufferSize));\r\n }\r\n return UUID.buf.slice(UUID.bufIdx, (UUID.bufIdx += n));\r\n }\r\n\r\n // uuid.bin\r\n protected static uuidBin() {\r\n const b = UUID.randomBytesBuffered(16);\r\n b[6] = (b[6] & 0x0f) | 0x40;\r\n b[8] = (b[8] & 0x3f) | 0x80;\r\n return b;\r\n }\r\n\r\n public static generate() {\r\n const b = UUID.uuidBin();\r\n return (\r\n UUID.hexBytes[b[0]] +\r\n UUID.hexBytes[b[1]] +\r\n UUID.hexBytes[b[2]] +\r\n UUID.hexBytes[b[3]] +\r\n '-' +\r\n UUID.hexBytes[b[4]] +\r\n UUID.hexBytes[b[5]] +\r\n '-' +\r\n UUID.hexBytes[b[6]] +\r\n UUID.hexBytes[b[7]] +\r\n '-' +\r\n UUID.hexBytes[b[8]] +\r\n UUID.hexBytes[b[9]] +\r\n '-' +\r\n UUID.hexBytes[b[10]] +\r\n UUID.hexBytes[b[11]] +\r\n UUID.hexBytes[b[12]] +\r\n UUID.hexBytes[b[13]] +\r\n UUID.hexBytes[b[14]] +\r\n UUID.hexBytes[b[15]]\r\n );\r\n }\r\n}\r\nUUID.initialize();\r\n","import { BehaviorSubject } from 'rxjs';\r\nimport { take } from 'rxjs/operators';\r\nimport { EChangeOperation, IHistoryStateChange, isTrackedClass, ITrackedClass } from '../interfaces/tracked-class';\r\nimport { changesProcessed } from './utilities';\r\n\r\ntype TrackedObject = ITrackedClass | any[];\r\ntype ChangeGroup = {\r\n change: Map;\r\n undoFunction?: () => void;\r\n redoFunction?: () => void;\r\n description: string;\r\n};\r\n\r\nexport interface ChangeInfo {\r\n index: number;\r\n description: string;\r\n nodes: any[];\r\n properties: IterableIterator;\r\n}\r\n\r\nclass SceneHistory {\r\n private active = false;\r\n private changes: ChangeGroup[] = [];\r\n private undoneChanges: ChangeGroup[] = [];\r\n\r\n private ignoreChangesFromObjects = new Set();\r\n\r\n openChange: ChangeGroup | undefined;\r\n state = new BehaviorSubject([]);\r\n undoneState = new BehaviorSubject([]);\r\n\r\n private updateState() {\r\n return this.state.next(this.getMergedChanges(this.changes));\r\n }\r\n private updateUndoneState() {\r\n return this.undoneState.next(this.getMergedChanges(this.undoneChanges));\r\n }\r\n\r\n private getMergedChanges(changes: ChangeGroup[]) {\r\n return changes.map((c, index) => {\r\n const changePart = Array.from(c.change.values());\r\n return {\r\n description: c.description,\r\n index,\r\n nodes: changePart.map(node => node.node),\r\n // Return a set of properties to drop duplicate array elements\r\n properties: new Set(\r\n changePart.flatMap(node => node.properties.flatMap(property => property.name))\r\n ).values(),\r\n };\r\n });\r\n }\r\n\r\n ignoreUntilStable(object: TrackedObject) {\r\n if (this.ignoreChangesFromObjects.size === 0) {\r\n changesProcessed.pipe(take(1)).subscribe(() => {\r\n this.ignoreChangesFromObjects.clear();\r\n });\r\n }\r\n this.ignoreChangesFromObjects.add(object);\r\n }\r\n\r\n start() {\r\n this.active = true;\r\n }\r\n\r\n stop() {\r\n this.active = false;\r\n }\r\n\r\n /**\r\n * Adds a change to the history. If you added an open-ended change\r\n * @param description\r\n * @param processFunction\r\n * @param undoFunction\r\n */\r\n addChange(description: string, processFunction: (redo: boolean) => void, undoFunction?: () => void) {\r\n // if (this.openChange) {\r\n // console.warn(\r\n // 'Starting a new history change with a previous change still open. You may get a strange history state.',\r\n // `Change still open: ${this.openChange.description}`,\r\n // `New change: ${description}`\r\n // );\r\n // }\r\n\r\n let newChange: ChangeGroup | undefined;\r\n if (!this.openChange) {\r\n newChange = this.openChange = {\r\n change: new Map(),\r\n description,\r\n undoFunction,\r\n };\r\n }\r\n processFunction(false);\r\n\r\n const done = (description: string | void) => {\r\n if (this.openChange) {\r\n if (this.openChange.change.size > 0 || this.openChange.undoFunction !== undefined) {\r\n if (description) {\r\n this.openChange.description = description;\r\n }\r\n this.changes.push(this.openChange);\r\n }\r\n\r\n delete this.openChange;\r\n this.updateState();\r\n\r\n if (this.undoneChanges.length > 0) {\r\n this.undoneChanges.clear();\r\n this.updateUndoneState();\r\n }\r\n }\r\n };\r\n\r\n if (undoFunction) {\r\n this.openChange.redoFunction = () => processFunction(true);\r\n done();\r\n } else if (newChange) {\r\n changesProcessed.pipe(take(1)).subscribe(done);\r\n }\r\n }\r\n\r\n undo() {\r\n if (this.openChange) {\r\n throw Error(\"Can't undo while there are open changes\");\r\n }\r\n if (this.changes.length < 1) {\r\n return;\r\n }\r\n const lastChangeSet = this.changes.pop();\r\n if (lastChangeSet) {\r\n this.undoneChanges.push(lastChangeSet);\r\n if (lastChangeSet.undoFunction) {\r\n lastChangeSet.undoFunction();\r\n } else {\r\n this.processChangeSet(lastChangeSet, change => {\r\n // Undo changes in reverse so they are consistent with how they were added.\r\n change.properties\r\n .slice()\r\n .reverse()\r\n .forEach(property => {\r\n const restore = property.previousValue;\r\n switch (property.operation) {\r\n case EChangeOperation.CreateAndPush:\r\n change.node.pop(true);\r\n break;\r\n case EChangeOperation.Push:\r\n change.node.pop();\r\n break;\r\n case EChangeOperation.Unshift:\r\n change.node.shift();\r\n break;\r\n case EChangeOperation.Shift:\r\n change.node.unshift(property.previousValue);\r\n if (isTrackedClass(restore)) {\r\n (restore as ITrackedClass).isModified = true;\r\n }\r\n break;\r\n case EChangeOperation.Pop:\r\n change.node.push(property.previousValue);\r\n if (isTrackedClass(restore)) {\r\n (restore as ITrackedClass).isModified = true;\r\n }\r\n break;\r\n case EChangeOperation.Remove:\r\n case EChangeOperation.Splice:\r\n // Property name is the index of the splice\r\n // Remove all the added values, add back in the old ones\r\n change.node.splice(property.name, property.newValue.length, ...restore);\r\n if (Array.isArray(restore) && restore.length > 0) {\r\n restore.forEach(restoreItem => {\r\n if (isTrackedClass(restoreItem)) {\r\n (restoreItem as ITrackedClass).isModified = true;\r\n }\r\n });\r\n }\r\n break;\r\n default:\r\n change.node[property.name] = property.previousValue;\r\n }\r\n });\r\n });\r\n }\r\n\r\n this.updateState();\r\n this.updateUndoneState();\r\n }\r\n }\r\n\r\n redo() {\r\n if (this.openChange) {\r\n throw Error(\"Can't redo while there are open changes\");\r\n }\r\n if (this.undoneChanges.length < 1) {\r\n return;\r\n }\r\n const lastUndoneChangeSet = this.undoneChanges.pop();\r\n if (lastUndoneChangeSet) {\r\n this.changes.push(lastUndoneChangeSet);\r\n if (lastUndoneChangeSet.redoFunction) {\r\n lastUndoneChangeSet.redoFunction();\r\n } else {\r\n this.processChangeSet(lastUndoneChangeSet, change => {\r\n // Redo changes in the initial order\r\n change.properties.forEach(property => {\r\n const restore = property.newValue;\r\n switch (property.operation) {\r\n case EChangeOperation.CreateAndPush:\r\n change.node.createAndPush(property.newValue);\r\n if (isTrackedClass(restore)) {\r\n (restore as ITrackedClass).isModified = true;\r\n }\r\n break;\r\n case EChangeOperation.Push:\r\n change.node.push(property.newValue);\r\n break;\r\n case EChangeOperation.Unshift:\r\n change.node.unshift(property.newValue);\r\n if (isTrackedClass(restore)) {\r\n (restore as ITrackedClass).isModified = true;\r\n }\r\n break;\r\n case EChangeOperation.Shift:\r\n change.node.shift();\r\n break;\r\n case EChangeOperation.Pop:\r\n change.node.pop();\r\n break;\r\n case EChangeOperation.Remove:\r\n property.previousValue.length === 1 && change.node.remove(property.previousValue[0]);\r\n break;\r\n case EChangeOperation.Splice:\r\n change.node.splice(property.name, property.previousValue.length, ...restore);\r\n if (Array.isArray(restore) && restore.length > 0) {\r\n restore.forEach(restoreItem => {\r\n if (isTrackedClass(restoreItem)) {\r\n (restoreItem as ITrackedClass).isModified = true;\r\n }\r\n });\r\n }\r\n break;\r\n default:\r\n change.node[property.name] = property.newValue;\r\n }\r\n });\r\n });\r\n }\r\n\r\n this.updateState();\r\n this.updateUndoneState();\r\n }\r\n }\r\n\r\n undoTo(index: number) {\r\n while (this.changes.length > index + 1) {\r\n this.undo();\r\n }\r\n }\r\n\r\n redoTo(index: number) {\r\n while (this.undoneChanges.length > index) {\r\n this.redo();\r\n }\r\n }\r\n\r\n // startGroupingAllChanges() {\r\n // this.groupAllIncomingChanges = true;\r\n // this.changes.push(new Map());\r\n // }\r\n\r\n // stopGroupingAllChanges() {\r\n // this.groupAllIncomingChanges = false;\r\n // this.removeLastChangeIfEmpty();\r\n // this.updateState();\r\n // }\r\n\r\n // startGroupingObjectsChanges(objects: Array) {\r\n // this.groupChangesInObjects = new Set(objects.filter(isTrackedClass));\r\n // this.changes.push(new Map());\r\n // }\r\n\r\n // stopGroupingObjectsChanges() {\r\n // this.groupChangesInObjects = undefined;\r\n // this.removeLastChangeIfEmpty();\r\n // }\r\n\r\n changeEvent(\r\n object: T,\r\n property: keyof T | symbol,\r\n value: any,\r\n originalValue: any,\r\n operation?: EChangeOperation\r\n ) {\r\n if (\r\n this.active &&\r\n (isTrackedClass(object) || Array.isArray(object)) &&\r\n value !== originalValue &&\r\n !this.ignoreChangesFromObjects.has(object)\r\n ) {\r\n if (!this.openChange) {\r\n return;\r\n }\r\n const openChange = this.openChange;\r\n\r\n let lastChangeOnObject = openChange.change.get(object);\r\n if (!lastChangeOnObject) {\r\n lastChangeOnObject = { node: object, properties: [] };\r\n openChange.change.set(object, lastChangeOnObject);\r\n }\r\n\r\n const lastChangeOnObjectProperty = lastChangeOnObject.properties.find(\r\n openChange => openChange.name === property\r\n );\r\n if (!lastChangeOnObjectProperty) {\r\n lastChangeOnObject.properties.push({\r\n name: property,\r\n previousValue: originalValue,\r\n newValue: value,\r\n operation,\r\n });\r\n } else {\r\n lastChangeOnObjectProperty.newValue = value;\r\n }\r\n }\r\n }\r\n\r\n changeEvents(\r\n object: T,\r\n changes: Array<{ property: keyof T | symbol; value: any; originalValue: any; operation: EChangeOperation }>\r\n ) {\r\n if (\r\n this.active &&\r\n (isTrackedClass(object) || Array.isArray(object)) &&\r\n !this.ignoreChangesFromObjects.has(object)\r\n ) {\r\n changes.forEach(change => {\r\n this.changeEvent(object, change.property, change.value, change.originalValue, change.operation);\r\n });\r\n }\r\n }\r\n\r\n removeLastChange(object: T, property: keyof T) {\r\n // todo\r\n }\r\n\r\n private processChangeSet(changeSet: ChangeGroup, process: (change: IHistoryStateChange) => void) {\r\n changeSet.change.forEach(change => process(change));\r\n }\r\n}\r\n\r\nexport const history = new SceneHistory();\r\n","import { EChangeOperation, isTrackedClass, ITrackedArray, ITrackedClass } from '../../interfaces/tracked-class';\r\nimport { SyncService } from '../../services/sync-service';\r\nimport { history } from '../history';\r\nimport { changesProcessed } from '../utilities';\r\n\r\nexport function nextTick() {\r\n changesProcessed.next();\r\n}\r\n\r\nconst __arrayProtoFunctions: any = {};\r\n['push', 'unshift', 'shift', 'pop', 'splice', 'reorder', 'clear'].forEach((arrayFunc: any) => {\r\n __arrayProtoFunctions[arrayFunc] = Array.prototype[arrayFunc];\r\n\r\n // We shouldn't need this code after the Vue update to 3.2. Was needed to fix reactivity on arrays for Vue\r\n // Delete this when no confirmed that there are no issues with reactivity in creator\r\n // Array.prototype[arrayFunc] = function(...args: any) {\r\n // if(isTrackedArray(this)) {\r\n // return TrackedArray.prototype[arrayFunc].apply(this, args);\r\n // } else {\r\n // return __arrayProtoFunctions[arrayFunc].apply(this, args);\r\n // }\r\n // }\r\n});\r\n\r\nclass TrackedArray extends Array implements ITrackedArray {\r\n $isTracked = true;\r\n\r\n constructor(\r\n private parentObj: T & ITrackedClass,\r\n private propertyName: string,\r\n private changesIgnored: boolean,\r\n private historyIgnored: boolean,\r\n source?: A[]\r\n ) {\r\n super();\r\n\r\n if (source) {\r\n for (let i = 0; i < source.length; i++) {\r\n __arrayProtoFunctions.push.call(this, source[i]);\r\n }\r\n }\r\n }\r\n\r\n push(...items: A[]) {\r\n if (!this.changesIgnored) {\r\n this.parentObj.setPropertyIsModified(this.propertyName, true);\r\n if (!this.historyIgnored) {\r\n history.changeEvents(\r\n this,\r\n items.map(item => ({\r\n property: this.length,\r\n operation: EChangeOperation.Push,\r\n value: item,\r\n originalValue: undefined,\r\n }))\r\n );\r\n }\r\n }\r\n\r\n const res = __arrayProtoFunctions.push.apply(this, items); // super.push(...items);\r\n for (const item of items) {\r\n this.parentObj.addedToChildArray(this.parentObj, item, this.propertyName);\r\n }\r\n return res;\r\n }\r\n\r\n unshift(...items: A[]) {\r\n if (!this.changesIgnored) {\r\n this.parentObj.setPropertyIsModified(this.propertyName, true);\r\n if (!this.historyIgnored) {\r\n history.changeEvents(\r\n this,\r\n items.map(item => ({\r\n property: 0,\r\n operation: EChangeOperation.Push,\r\n value: item,\r\n originalValue: undefined,\r\n }))\r\n );\r\n }\r\n }\r\n const res = __arrayProtoFunctions.unshift.apply(this, items); // super.unshift(...items);\r\n\r\n for (const item of items) {\r\n this.parentObj.addedToChildArray(this.parentObj, item, this.propertyName);\r\n }\r\n\r\n return res;\r\n }\r\n\r\n shift(destroy?: boolean): A {\r\n const result = __arrayProtoFunctions.shift.apply(this); // super.shift();\r\n if (!this.changesIgnored) {\r\n this.parentObj.setPropertyIsModified(this.propertyName, true);\r\n if (!this.historyIgnored) {\r\n history.changeEvent(this, 0, undefined, result, EChangeOperation.Shift);\r\n }\r\n }\r\n if (destroy && isTrackedClass(result)) {\r\n result.destroy();\r\n }\r\n this.parentObj.addedToChildArray(this.parentObj, result, this.propertyName);\r\n\r\n return result;\r\n }\r\n\r\n pop(destroy?: boolean): A {\r\n const result = __arrayProtoFunctions.pop.apply(this); // super.pop();\r\n if (!this.changesIgnored) {\r\n this.parentObj.setPropertyIsModified(this.propertyName, true);\r\n if (!this.historyIgnored) {\r\n history.changeEvent(this, this.length, undefined, result, EChangeOperation.Pop);\r\n }\r\n }\r\n if (destroy && isTrackedClass(result)) {\r\n result.destroy();\r\n }\r\n this.parentObj.addedToChildArray(this.parentObj, result, this.propertyName);\r\n return result;\r\n }\r\n\r\n splice(start: number, deleteCount: number | undefined, ...insert: A[]): A[] {\r\n // const result = Array.prototype.splice.call(array, start, deleteCount || 0, ...insert);\r\n let result: A[];\r\n if (insert.length === 0) {\r\n result = __arrayProtoFunctions.splice.apply(this, [start, deleteCount]); // super.splice(start, deleteCount);\r\n } else {\r\n result = __arrayProtoFunctions.splice.apply(this, [start, deleteCount || 0, ...insert]); // super.splice(start, deleteCount || 0, ...insert);\r\n }\r\n\r\n result.forEach(resultItem => {\r\n if (isTrackedClass(resultItem)) {\r\n resultItem.destroy();\r\n }\r\n });\r\n for (const item of result) {\r\n this.parentObj.removedFromChildArray(this.parentObj, item, this.propertyName);\r\n }\r\n for (const item of insert) {\r\n this.parentObj.addedToChildArray(this.parentObj, item, this.propertyName);\r\n }\r\n\r\n if (!this.changesIgnored) {\r\n this.parentObj.setPropertyIsModified(this.propertyName, true);\r\n if (!this.historyIgnored) {\r\n history.changeEvent(this, start, insert, result, EChangeOperation.Splice);\r\n }\r\n }\r\n return result;\r\n }\r\n\r\n remove(item: A): A[] {\r\n const index = this.indexOf(item);\r\n if (index > -1) {\r\n return this.splice(index, 1);\r\n }\r\n return [];\r\n }\r\n\r\n reorder(fromIndex: number, toIndex: number, undo: boolean) {\r\n __arrayProtoFunctions.reorder.apply(this, [fromIndex, toIndex, undo]);\r\n\r\n if (!this.changesIgnored) {\r\n this.parentObj.setPropertyIsModified(this.propertyName, true);\r\n }\r\n }\r\n}\r\n\r\n/** Decorator to track the properties of a class. Used in conjunction with the change tracker class.\r\n * Properties with this decorator will be watched by the change tracker\r\n * @param {Object} options decorator options object\r\n * @param {Function} options.onCreate function that is called when a new class instance is created. Called with the proxy object that was created.\r\n */\r\nexport function TrackChanges(options?: TrackChangesOptions) {\r\n const onCreateCallback = options ? options.onCreate : undefined;\r\n const isPropertyEqualOverride = options ? options.isPropertyEqual : undefined;\r\n const onDestroy = options ? options.onDestroy : undefined;\r\n const addedToChildArray = options ? options.addedToChildArray : undefined;\r\n const removedFromChildArray = options ? options.removedFromChildArray : undefined;\r\n\r\n return (targetClass: C) => {\r\n const TrackedClass = class extends targetClass implements ITrackedClass {\r\n get $changesTracked() {\r\n return true;\r\n }\r\n\r\n setPropertyIsModified() {}\r\n destroy() {}\r\n\r\n constructor(...args: any[]) {\r\n super(...args);\r\n const trackingState: { isModified: boolean; baseObject: any } = {\r\n baseObject: this,\r\n isModified: false,\r\n };\r\n\r\n if (typeof this.originalValues === 'undefined') {\r\n Object.defineProperties(this, {\r\n changes: {\r\n value: new Map(),\r\n enumerable: false,\r\n writable: false,\r\n configurable: false,\r\n },\r\n originalValues: {\r\n value: new Map(),\r\n enumerable: false,\r\n writable: false,\r\n configurable: false,\r\n },\r\n _trackChanges: {\r\n value: false,\r\n enumerable: false,\r\n writable: true,\r\n configurable: false,\r\n },\r\n isModified: {\r\n configurable: false,\r\n enumerable: false,\r\n set: () => {},\r\n get: () => trackingState.isModified,\r\n },\r\n commitChanges: {\r\n configurable: false,\r\n enumerable: false,\r\n writable: false,\r\n value: (inherited = false) => commitChanges(this, inherited),\r\n },\r\n setPropertyIsModified: {\r\n configurable: false,\r\n enumerable: false,\r\n writable: false,\r\n value: setPropertyIsModified,\r\n },\r\n destroy: {\r\n configurable: false,\r\n enumerable: false,\r\n writable: false,\r\n value: () => destroy(this),\r\n },\r\n trackChanges: {\r\n configurable: false,\r\n enumerable: false,\r\n writable: false,\r\n value: (active: boolean) => (this._trackChanges = active),\r\n },\r\n rawNode: {\r\n configurable: false,\r\n enumerable: false,\r\n writable: false,\r\n value: trackingState.baseObject,\r\n },\r\n });\r\n }\r\n\r\n const changedValuesMap: Map = this.changes;\r\n const originalValuesMap: Map = this.originalValues;\r\n\r\n const proxy = new Proxy(this, {\r\n // get: (target, property) => {\r\n // if (property === 'isModified') {\r\n // return trackingState.isModified;\r\n // } else if (property === 'commitChanges') {\r\n // return (inherited: boolean = false) => commitChanges(target, inherited);\r\n // } else if (property === 'setPropertyIsModified') {\r\n // return setPropertyIsModified;\r\n // } else if (property === 'addedToChildArray') {\r\n // return addedToChildArray ? addedToChildArray : () => { }\r\n // } else if (property === 'removedFromChildArray') {\r\n // return removedFromChildArray ? removedFromChildArray : () => { }\r\n // } else if (property === 'destroy') {\r\n // return () => destroy(target);\r\n // } else if (property === 'trackChanges') {\r\n // return (active: boolean) => this._trackChanges = active;\r\n // } else if (property === 'rawNode') {\r\n // return trackingState.baseObject;\r\n // }\r\n // return target[property as string];\r\n // },\r\n set: (target, property, newValue) => {\r\n // Allow clearing changes by setting is modified to false.\r\n if (property === 'isModified') {\r\n if (newValue === false && newValue !== trackingState.isModified) {\r\n originalValuesMap.clear();\r\n changedValuesMap.clear();\r\n updateIsModified();\r\n } else if (newValue === true) {\r\n for (const key in target) {\r\n setPropertyIsModified(key, true);\r\n }\r\n }\r\n return true;\r\n // Throw type error on trying to redefine decorator functions\r\n } else if (property === 'setPropertyIsModified') {\r\n return false;\r\n } else {\r\n // If object is not being tracked, just set the value without tracking the changes\r\n if (!this._trackChanges) {\r\n return Reflect.set(trackingState.baseObject, property, newValue);\r\n }\r\n\r\n const changesIgnored =\r\n Reflect.has(target, '_changesIgnoredProperties') &&\r\n Reflect.get(target, '_changesIgnoredProperties').has(property);\r\n const historyIgnored =\r\n Reflect.has(target, '_historyIgnoredProperties') &&\r\n Reflect.get(target, '_historyIgnoredProperties').has(property);\r\n\r\n // If the value is an array, then we have to proxy the array to pick up new array elements\r\n // Important that this occurs even if the object is not being tracked because\r\n // parent node management will break without the array proxy\r\n if (Array.isArray(newValue)) {\r\n newValue = new TrackedArray(\r\n proxy,\r\n property.toString(),\r\n changesIgnored,\r\n historyIgnored,\r\n newValue\r\n );\r\n }\r\n\r\n if (!changesIgnored) {\r\n const originalValue = originalValuesMap.get(property.toString());\r\n const currentValue = Reflect.get(target, property);\r\n\r\n // If the new value is the same as previous, do nothing.\r\n if (\r\n isPropertyEqualOverride &&\r\n isPropertyEqualOverride(currentValue, newValue, property as any)\r\n ) {\r\n return true;\r\n } else if (currentValue === newValue) {\r\n return true;\r\n }\r\n\r\n // Skip checks if originalValue is null. It means isModified is forced to be true until node is processed.\r\n if (originalValue !== null) {\r\n if (!originalValue) {\r\n // If the value is changed and was never changed before, register the change\r\n originalValuesMap.set(property.toString(), currentValue);\r\n } else {\r\n let isPropertyModified: boolean;\r\n // If the value was changed before, see if it's the same as the original value.\r\n let isEqualToOriginal: boolean | undefined;\r\n if (isPropertyEqualOverride) {\r\n isEqualToOriginal = isPropertyEqualOverride(\r\n originalValue,\r\n newValue,\r\n property as any\r\n );\r\n }\r\n if (typeof isEqualToOriginal !== 'undefined') {\r\n isPropertyModified = !isEqualToOriginal;\r\n } else {\r\n isPropertyModified = originalValue !== newValue;\r\n }\r\n\r\n // If the value is the same as the original, de-register the change\r\n if (!isPropertyModified) {\r\n originalValuesMap.delete(property.toString());\r\n }\r\n }\r\n\r\n updateIsModified();\r\n }\r\n }\r\n if (originalValuesMap.has(property.toString())) {\r\n changedValuesMap.set(property.toString(), newValue);\r\n } else {\r\n changedValuesMap.delete(property.toString());\r\n }\r\n\r\n return Reflect.set(trackingState.baseObject, property, newValue);\r\n }\r\n },\r\n });\r\n // Don't register changes in the history when creating new objects as they have to be registered as new objects instead.\r\n history.ignoreUntilStable(proxy);\r\n\r\n // The sync service won't have the right proxy object when a new object is created\r\n SyncService.registerNewObject(proxy);\r\n\r\n // The properties set on the object in the constructor won't be tracked because the original constructor runs before\r\n // the object is proxied.\r\n let trackedNewProperty = false;\r\n for (const property in this) {\r\n if (!this._changesIgnoredProperties || !this._changesIgnoredProperties.has(property)) {\r\n if (Array.isArray(this[property])) {\r\n const arrayHistoryIgnored =\r\n this._historyIgnoredProperties && this._historyIgnoredProperties.has(property);\r\n // We've already filtered out change ignored properties above\r\n this[property] = new TrackedArray(\r\n proxy,\r\n property.toString(),\r\n false,\r\n arrayHistoryIgnored,\r\n this[property]\r\n ) as any;\r\n }\r\n originalValuesMap.set(property, this[property]);\r\n changedValuesMap.set(property, this[property]);\r\n trackedNewProperty = true;\r\n }\r\n }\r\n if (trackedNewProperty) {\r\n updateIsModified();\r\n }\r\n\r\n if (onCreateCallback) {\r\n onCreateCallback(proxy as unknown as T & ITrackedClass);\r\n }\r\n\r\n return proxy;\r\n\r\n function updateIsModified() {\r\n // If the original value map is not empty, we have some pending changes so this object is considered modified\r\n if (trackingState.isModified !== originalValuesMap.size > 0) {\r\n trackingState.isModified = originalValuesMap.size > 0;\r\n SyncService.registerChangedState(proxy, trackingState.isModified);\r\n }\r\n }\r\n\r\n function commitChanges(target: any, inherited: boolean) {\r\n if (trackingState.isModified) {\r\n if (!inherited) {\r\n // Register the change on the history\r\n originalValuesMap.forEach((oldValue, property) => {\r\n const historyIgnored =\r\n Reflect.has(target, '_historyIgnoredProperties') &&\r\n Reflect.get(target, '_historyIgnoredProperties').has(property);\r\n\r\n if (!historyIgnored) {\r\n history.changeEvent(proxy, property, trackingState.baseObject[property], oldValue);\r\n }\r\n });\r\n\r\n originalValuesMap.clear();\r\n changedValuesMap.clear();\r\n }\r\n updateIsModified();\r\n }\r\n SyncService.nodeChangesProcessed(target);\r\n }\r\n\r\n function setPropertyIsModified(prop: string, modified: boolean) {\r\n if (modified) {\r\n // When forcing a property as modified, set the original value to null so it stays modified until processed (never matches)\r\n originalValuesMap.set(prop, null);\r\n changedValuesMap.set(prop, proxy[prop]);\r\n } else {\r\n originalValuesMap.delete(prop);\r\n changedValuesMap.delete(prop);\r\n }\r\n updateIsModified();\r\n }\r\n\r\n function destroy(target: any) {\r\n if (onDestroy) {\r\n onDestroy(proxy as unknown as T & ITrackedClass);\r\n }\r\n SyncService.registerDeleted(proxy);\r\n const inheritedDestroyFunc = Object.getPrototypeOf(target).destroy;\r\n if (typeof inheritedDestroyFunc === 'function') {\r\n inheritedDestroyFunc();\r\n }\r\n }\r\n }\r\n };\r\n\r\n Object.defineProperties(TrackedClass.prototype, {\r\n addedToChildArray: {\r\n configurable: false,\r\n enumerable: false,\r\n writable: false,\r\n value: addedToChildArray ? addedToChildArray : () => {},\r\n },\r\n removedFromChildArray: {\r\n configurable: false,\r\n enumerable: false,\r\n writable: false,\r\n value: removedFromChildArray ? removedFromChildArray : () => {},\r\n },\r\n });\r\n\r\n return TrackedClass;\r\n };\r\n}\r\n\r\n// eslint-disable-next-line @typescript-eslint/ban-types\r\nexport function NodeReference(classRef: Function & { prototype: T }, ...keys: Array) {\r\n return (target: any, propertyKey: string, parameterIndex?: number) => {\r\n if (typeof target.nodeReferences === 'undefined') {\r\n Object.defineProperty(target, 'nodeReferences', {\r\n value: new Map(),\r\n enumerable: false,\r\n writable: false,\r\n configurable: false,\r\n });\r\n }\r\n\r\n target.nodeReferences.set(propertyKey, {\r\n classRef,\r\n limitKeys: keys.length > 0 ? new Set(keys) : undefined,\r\n });\r\n };\r\n}\r\n\r\n/** Property decorator to ignore changes on some properties. The node will not be processed if only ignored properties change. */\r\nexport function IgnoreChanges(target: any, propertyKey: string, parameterIndex?: number) {\r\n if (typeof target._changesIgnoredProperties === 'undefined') {\r\n Object.defineProperty(target, '_changesIgnoredProperties', {\r\n value: new Map(),\r\n enumerable: false,\r\n writable: false,\r\n configurable: false,\r\n });\r\n }\r\n\r\n target._changesIgnoredProperties.set(propertyKey, true);\r\n}\r\n\r\n/** Property decorator to ignore history on some properties. The property changing will not trigger history additions. */\r\nexport function IgnoreHistory(target: any, propertyKey: string, parameterIndex?: number) {\r\n if (typeof target._historyIgnoredProperties === 'undefined') {\r\n Object.defineProperty(target, '_historyIgnoredProperties', {\r\n value: new Map(),\r\n enumerable: false,\r\n writable: false,\r\n configurable: false,\r\n });\r\n }\r\n\r\n target._historyIgnoredProperties.set(propertyKey, true);\r\n}\r\n\r\nexport interface TrackChangesOptions {\r\n /** Function to run on creation of class instance.\r\n * @param proxyObj the constructed class instance proxy object.\r\n */\r\n onCreate?: (proxyObj: T & ITrackedClass) => void;\r\n /** Function to run on removal of class instance - called manually, or when removed from an array.\r\n * @param proxyObj the constructed class instance proxy object.\r\n */\r\n onDestroy?: (proxyObj: T & ITrackedClass) => void;\r\n /** Function to run on undoing removing a class instance from the scene. If we need to undo anything that was destroyed in the onDestroy hook.\r\n * @param proxyObj the constructed class instance proxy object.\r\n */\r\n onUndestroy?: (proxyObj: T & ITrackedClass) => void;\r\n /** Function to override the changed check behavior for certain properties on this class.\r\n * If property has not changed compared to the original value, then change queue will be cleared.\r\n * @param originalValue the old value A to compare against\r\n * @param newValue the new value B to compare against\r\n * @param name the property name being changed\r\n * @returns boolean or undefined. If undefined, will fall back to strict equality check for property.\r\n */\r\n isPropertyEqual?: (originalValue: any, newValue: any, name: keyof T) => boolean | undefined;\r\n /**\r\n * Function called when an item is added to a property array of the tracked class\r\n */\r\n addedToChildArray?: (parent: T & ITrackedClass, item: any, arrayPropertyName: string) => void;\r\n /**\r\n * Function called when an item is removed from a property array of the tracked class\r\n */\r\n removedFromChildArray?: (parent: T & ITrackedClass, item: any, arrayPropertyName: string) => void;\r\n}\r\n","import { isTrackedClass, ITrackedClass } from '../../interfaces/tracked-class';\r\nimport { isPrimitive } from '../primitives/primitive-utilities';\r\nimport { Token } from '../token';\r\nimport { TrackChanges, TrackChangesOptions } from './change-tracker';\r\nimport { instanceOf } from './model-utils';\r\n\r\n/** Convenience decorator that sets up default options for Kb3dObjects with the change tracker */\r\nexport function TrackChangesNode(options: TrackChangesOptions = {}) {\r\n const origAddedToChildArray = options.addedToChildArray;\r\n options.addedToChildArray = (parent: T, item: any, arrayPropName: string) => {\r\n if (origAddedToChildArray) origAddedToChildArray(parent, item, arrayPropName);\r\n if (instanceOf(parent, Token.Kb3dObject) && instanceOf(item, Token.Kb3dObject)) {\r\n item._parent = parent;\r\n if (isTrackedClass(parent) && (parent as any)._trackChanges) {\r\n item.calculateParents();\r\n }\r\n }\r\n };\r\n\r\n const origDestroy = options.onDestroy;\r\n options.onDestroy = (item: T & ITrackedClass) => {\r\n if (origDestroy) origDestroy(item);\r\n\r\n const children = item.getChildren();\r\n for (let i = 0; i < children.length; i++) {\r\n const child = children[i];\r\n if (isTrackedClass(child)) {\r\n child.destroy();\r\n }\r\n }\r\n\r\n if (item._parent && instanceOf(item._parent, Token.Kb3dObject) && instanceOf(item, Token.Kb3dObject)) {\r\n if (item._parent._manager) {\r\n item._parent._manager.unregister(item);\r\n }\r\n }\r\n\r\n item._parent = undefined;\r\n item.isModified = false;\r\n };\r\n\r\n if (!options.isPropertyEqual) {\r\n options.isPropertyEqual = (oldValue, newValue, propertyKey) => {\r\n let result: boolean | undefined;\r\n\r\n // Check equality on node properties by handling arrays and KB primitives. Fall back to strict equality checks for other values.\r\n if (Array.isArray(oldValue) && Array.isArray(newValue)) {\r\n if (oldValue.length === newValue.length) {\r\n let isEqual = true;\r\n\r\n for (let i = 0; i < newValue.length; i++) {\r\n if (newValue[i] !== oldValue[i]) {\r\n isEqual = false;\r\n break;\r\n }\r\n }\r\n result = isEqual;\r\n } else {\r\n result = false;\r\n }\r\n } else if (isPrimitive(newValue)) {\r\n result = typeof oldValue !== 'undefined' && newValue.equal(oldValue);\r\n }\r\n // console.log('checking equality', oldValue, newValue, result);\r\n return result;\r\n };\r\n }\r\n\r\n return TrackChanges(options);\r\n}\r\n","import { KbColor } from './kb-color';\r\nimport { KbQuaternion } from './kb-quaternion';\r\nimport { IKbVectorLike, KbVector } from './kb-vector';\r\n\r\nexport function isKbVector(object: any): object is KbVector {\r\n return object && object.x !== undefined && object.y !== undefined && object.z !== undefined;\r\n}\r\n\r\nexport function isKbColor(object: any): object is KbColor {\r\n return (\r\n object && object.r !== undefined && object.g !== undefined && object.b !== undefined && object.a !== undefined\r\n );\r\n}\r\n\r\nexport function isKbQuaternion(object: any): object is KbQuaternion {\r\n return (\r\n object && object.x !== undefined && object.y !== undefined && object.z !== undefined && object.w !== undefined\r\n );\r\n}\r\n\r\nexport function isPrimitive(object: any) {\r\n return isKbVector(object) || isKbColor(object) || isKbQuaternion(object);\r\n}\r\n\r\nexport interface IBoundingBox {\r\n min: IKbVectorLike;\r\n max: IKbVectorLike;\r\n center: IKbVectorLike;\r\n}\r\n","import { Token } from '../token';\r\nimport { TrackChangesNode } from './change-tracker-node';\r\nimport { AbstractConstructor, IKb3dModel, Kb3dModels } from './model-utils';\r\n\r\n/** represents an instance of a class decorated by kb3dModel */\r\nexport interface IDecoratedModel {\r\n $type: keyof typeof Token;\r\n [key: string]: any;\r\n}\r\n\r\nexport function Kb3dModel(options: Partial> & Pick, 'token'>) {\r\n if (!('trackChanges' in options)) options.trackChanges = true; //if trackChanges was not provided, default to true\r\n\r\n const $type = options.token.key;\r\n\r\n return (constructor: C) => {\r\n //set the $type on the prototype so we can always retrieve it from an instance\r\n Object.defineProperty(constructor.prototype, '$type', {\r\n enumerable: false,\r\n writable: true,\r\n value: $type,\r\n });\r\n\r\n /** we bundle up the track changes decorator here as well, as we want our map storing the constructors of\r\n * our models to actually be storing the constructor of the track changes proxy... so that when a class\r\n * is constructed using \"create\" that it will actually create the proxy class.\r\n */\r\n if (options.trackChanges) {\r\n const trackChangesFactory = TrackChangesNode(options);\r\n const combinedConstructor = trackChangesFactory(constructor);\r\n options.classConstructor = combinedConstructor;\r\n } else {\r\n options.classConstructor = constructor;\r\n }\r\n Kb3dModels.set(options.token, options as IKb3dModel);\r\n };\r\n}\r\n\r\nexport function BaseKb3dModel(options: Partial> & Pick, 'token'>) {\r\n return >(constructor: C) => {\r\n options.classConstructor = constructor;\r\n Kb3dModels.set(options.token, options as IKb3dModel);\r\n };\r\n}\r\n","/**\r\n * Applied to a getter or setter property, will make it enumerable\r\n */\r\nexport function Enumerable() {\r\n return function (target: any, propertyKey: string, descriptor: PropertyDescriptor) {\r\n descriptor.enumerable = true;\r\n };\r\n}\r\n","import { Token } from '../token';\r\nimport { Constructor, IKb3dModel } from './model-utils';\r\n\r\nexport interface IInjectParameter {\r\n classConstructor: Constructor;\r\n token: Token;\r\n parameterIndex: number;\r\n ignore: boolean;\r\n}\r\n\r\nexport const injectParameters: any[] = [];\r\n\r\nexport function Inject(token: Token) {\r\n return (target: Constructor, propertyKey: undefined, parameterIndex: number) => {\r\n injectParameters.push({\r\n token: token,\r\n classConstructor: target,\r\n parameterIndex,\r\n ignore: false,\r\n });\r\n };\r\n}\r\n\r\nexport function Ignore() {\r\n return (target: Constructor, propertyKey: undefined, parameterIndex: number) => {\r\n injectParameters.push({\r\n token: Token.Ignore,\r\n classConstructor: target,\r\n parameterIndex,\r\n ignore: true,\r\n });\r\n };\r\n}\r\n\r\nexport function getConstructorParamsForModel(modelDef: IKb3dModel): IInjectParameter[] {\r\n let constructor: any;\r\n if (modelDef.trackChanges) {\r\n constructor = Object.getPrototypeOf(modelDef.classConstructor);\r\n } else {\r\n constructor = modelDef.classConstructor;\r\n }\r\n const paramsForType = injectParameters.filter(i => i.classConstructor == constructor);\r\n\r\n paramsForType.sortBy(p => p.index);\r\n if (modelDef.classConstructor && paramsForType.length < modelDef.classConstructor.length) {\r\n throw new Error(\r\n 'You must decorate every constructor parameter with an @Inject or @Ignore decorator for type ' +\r\n modelDef.token.key\r\n );\r\n }\r\n return paramsForType;\r\n}\r\n","import { IKb3dModel, Kb3dModels } from './model-utils';\r\n\r\nexport function Kb3dService(options: Partial> & Pick, 'token'>) {\r\n const $type = options.token.key;\r\n\r\n return (constructor: C) => {\r\n constructor.prototype.$type = $type;\r\n options.classConstructor = constructor;\r\n Kb3dModels.set(options.token, options as IKb3dModel);\r\n };\r\n}\r\n","export enum eParentChangeBehavior {\r\n PRESERVE_POSITION,\r\n PRESERVE_PROPERTY,\r\n}\r\nexport interface KbViewerSettings {\r\n parentChangeBehavior: eParentChangeBehavior;\r\n}\r\n","import { isTrackedClass, ITrackedClass } from '@models/interfaces';\r\nimport { IgnoreChanges } from './decorators/change-tracker';\r\nimport { Enumerable } from './decorators/enumerable';\r\nimport { BaseKb3dModel } from './decorators/kb3d-decorator';\r\nimport { instanceOf, instanceOfAny } from './decorators/model-utils';\r\nimport { Kb3dManager } from './kb3d-manager';\r\nimport { IKb3dObject } from './kb3d-models';\r\nimport { SceneNode } from './scene-node';\r\nimport { Token } from './token';\r\nimport { UUID } from './utilities';\r\n\r\nexport enum eNodeError {\r\n 'FEATURE_REFERENCE',\r\n 'GEOMETRY_SAVE',\r\n}\r\n\r\nexport interface INodeError {\r\n error: eNodeError;\r\n}\r\n\r\nexport type ValidationObject = {\r\n [K in keyof T]: string[];\r\n};\r\n\r\n/**\r\n * Base class for kb3d models. Any model that has an id and $type should derive from this class\r\n */\r\n@BaseKb3dModel({\r\n token: Token.Kb3dObject,\r\n})\r\nexport abstract class Kb3dObject implements IKb3dObject {\r\n constructor() {\r\n this.id = UUID.generate();\r\n this._dynamicId = UUID.generate();\r\n }\r\n\r\n public id: string;\r\n\r\n /**\r\n * Dynamically created id that is used to track assets in the 3d engine. Ensures there is\r\n * no duplicate id's when there are nested scenes.\r\n */\r\n public _dynamicId: string;\r\n\r\n @IgnoreChanges\r\n public name: string;\r\n\r\n @IgnoreChanges\r\n public $type: keyof typeof Token;\r\n\r\n get _token() {\r\n return Token.withKey(this.$type);\r\n }\r\n\r\n @IgnoreChanges\r\n public _manager: Kb3dManager | undefined;\r\n\r\n /**\r\n * The parent kb3dObject. This should not need to be set manually as adding nodes to arrays will set this automatically.\r\n * If this is a nested sceneNode, then _parent will be it's parent scene node\r\n * */\r\n public _parent: Kb3dObject | undefined;\r\n\r\n /** the closest ancestor this is a sceneNode. If this is a nested scene node, then _parentScene will be equal to itself */\r\n @IgnoreChanges\r\n public _parentScene: SceneNode | undefined;\r\n\r\n @IgnoreChanges\r\n private __rootScene: SceneNode | undefined;\r\n get _rootScene() {\r\n return this.__rootScene;\r\n }\r\n\r\n protected setRootScene(scene: SceneNode | undefined) {\r\n if (this.__rootScene !== scene) {\r\n this.__rootScene = scene;\r\n if (isTrackedClass(this)) {\r\n (this as ITrackedClass).trackChanges(!!scene);\r\n }\r\n }\r\n }\r\n\r\n @IgnoreChanges\r\n public _level = -1;\r\n\r\n @IgnoreChanges\r\n public _errors: INodeError[] = [];\r\n\r\n _selected = false;\r\n\r\n @IgnoreChanges\r\n _expanded = false;\r\n\r\n private _metadata: any;\r\n @Enumerable()\r\n public get metadata(): any {\r\n if (!this._metadata) this._metadata = {};\r\n return this._metadata;\r\n }\r\n public set metadata(val) {\r\n this._metadata = val;\r\n }\r\n\r\n public clone(this: T) {\r\n if (this._manager) {\r\n return this._manager.clone(this);\r\n }\r\n }\r\n\r\n public serialize(this: T): Partial | undefined {\r\n if (this._manager) {\r\n return this._manager.serialize(this);\r\n }\r\n }\r\n\r\n public serializeToJson(prettyPrint = false): string {\r\n if (this._manager) {\r\n return this._manager.serializeToJson(this, { prettyPrint });\r\n }\r\n return '';\r\n }\r\n\r\n public calculateParents(deletion = false, parent?: Kb3dObject) {\r\n // When a node is deleted and recreated, the children won't have parents because they were not re-added to the parent's arrays.\r\n // This solves that case by making sure the hierarchy is consistent from the top down\r\n if (parent && this._parent !== parent) {\r\n this._parent = parent;\r\n }\r\n if (this._parent) {\r\n this._level = this._parent._level == -1 ? -1 : this._parent._level + 1;\r\n this._parentScene = this._parent._parentScene;\r\n this.setRootScene(this._parent._rootScene);\r\n if (instanceOf(this, Token.SceneNode)) {\r\n //it's a nested scene\r\n this._parentScene = this;\r\n } else {\r\n this._manager = this._parent._manager; // a nested scene uses it's own manager, but everything else should have the same as it's parent\r\n }\r\n if (this._parentScene && !deletion) {\r\n this._manager?.register(this);\r\n }\r\n } else if (instanceOf(this, Token.SceneNode) && !deletion) {\r\n this._level = 0;\r\n this._parentScene = this;\r\n this.setRootScene(this);\r\n this._manager?.register(this);\r\n } else {\r\n this._level = -1; // -1 means this is not part of a scene node hierarchy\r\n\r\n // TODO add this change after release on Jan 13 2023, test to ensure it works\r\n // if(isTrackedClass(this)) {\r\n // this.trackChanges(false);\r\n // }\r\n }\r\n\r\n for (const c of this.getChildren()) {\r\n c.calculateParents(deletion, this);\r\n }\r\n }\r\n\r\n /**\r\n * Derived classes that have children should override this\r\n */\r\n public getChildren(): Kb3dObject[] {\r\n return [];\r\n }\r\n\r\n /**\r\n * Derived classes with children should override this\r\n */\r\n public getChildArrayForType(token: Token): Kb3dObject[] | undefined {\r\n return undefined;\r\n }\r\n\r\n public getChildArray(child: Kb3dObject) {\r\n return this.getChildArrayForType(Token.withKey(child.$type));\r\n }\r\n\r\n public addChildren(...children: Kb3dObject[]) {\r\n for (const c of children) {\r\n this.getChildArray(c)?.push(c);\r\n }\r\n }\r\n\r\n public removeChildren(...children: Kb3dObject[]) {\r\n for (const c of children) {\r\n this.getChildArray(c)?.remove(c);\r\n }\r\n }\r\n\r\n public insertChildAfter(afterChild: Kb3dObject, newChild: Kb3dObject) {\r\n const arr1 = this.getChildArray(afterChild);\r\n const arr2 = this.getChildArray(newChild);\r\n if (!arr1 || !arr2) throw 'incompatible child';\r\n if (arr1 !== arr2) throw 'children mismatch';\r\n const index = arr1.indexOf(afterChild);\r\n if (index === -1) throw 'could not find given child';\r\n arr1.splice(index + 1, 0, newChild);\r\n }\r\n\r\n public getAncestors() {\r\n const ancestors = new Array();\r\n let parent = this._parent;\r\n while (parent != null) {\r\n ancestors.push(parent);\r\n parent = parent._parent;\r\n }\r\n return ancestors;\r\n }\r\n\r\n public getNamePath() {\r\n let path = '';\r\n let self: Kb3dObject | undefined = this;\r\n while (self && !instanceOf(self, Token.SceneNode)) {\r\n path = self.name + (path ? '/' + path : '');\r\n self = self._parent;\r\n }\r\n return path;\r\n }\r\n\r\n /**\r\n * Filters children recursively based on a condition. This is an O(n) operation, so only use when performance is not paramount\r\n * @param predicate expression to test whether the object should be included\r\n * @param includeSelf whether to include this kb3dObject in the search\r\n * @param alwaysIncludeChildren if true, will always search children even if the parent resolves to false\r\n */\r\n public filter(\r\n predicate: (node: Kb3dObject) => boolean,\r\n includeSelf = false,\r\n alwaysIncludeChildren = true\r\n ): T[] {\r\n const res: Kb3dObject[] = [];\r\n let included = true;\r\n if (includeSelf && predicate(this)) {\r\n res.push(this);\r\n } else {\r\n included = false;\r\n }\r\n if (included || alwaysIncludeChildren) {\r\n const children = this.getChildren();\r\n for (const child of children) {\r\n res.push(...child.filter(predicate, true, alwaysIncludeChildren));\r\n }\r\n }\r\n\r\n return res as T[];\r\n }\r\n\r\n /**\r\n * filters children by type given a list of type tokens\r\n * @param tokens\r\n */\r\n public filterByType(...tokens: Array>): T[] {\r\n return this.filter(node => instanceOfAny(node, ...tokens)) as T[];\r\n }\r\n\r\n /**\r\n * filters children, but only those that qualify as nodes\r\n * @param predicate\r\n * @param includeSelf\r\n * @param alwaysIncludeChildren\r\n */\r\n public filterNodes(\r\n predicate: (node: Kb3dObject) => boolean,\r\n includeSelf = false,\r\n alwaysIncludeChildren = true\r\n ): T[] {\r\n const nodePredicate = (node: Kb3dObject) => node.isNode() && predicate(node);\r\n return this.filter(nodePredicate, includeSelf, alwaysIncludeChildren);\r\n }\r\n\r\n public isNode() {\r\n return this._token.isNode();\r\n }\r\n}\r\n","import { BaseKb3dModel, instanceOf } from '../decorators';\r\nimport { IFeature } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { TextureLayer } from '../materials';\r\nimport { SpaceNode } from '../meshes';\r\nimport { Token } from '../token';\r\n\r\nexport enum eMeshChangeTypes {\r\n VertexIndices,\r\n VertexPositions,\r\n VertexNormals,\r\n VertexUvs,\r\n Submeshes,\r\n Transform,\r\n}\r\n\r\nexport const ALL_GEOMETRY = [\r\n eMeshChangeTypes.VertexIndices,\r\n eMeshChangeTypes.VertexPositions,\r\n eMeshChangeTypes.VertexNormals,\r\n eMeshChangeTypes.VertexUvs,\r\n eMeshChangeTypes.Submeshes,\r\n];\r\n\r\nexport const ALL_CHANGES = [...ALL_GEOMETRY, eMeshChangeTypes.Transform];\r\n\r\nexport type FeatureWithTextures = Feature & { textures: TextureLayer[] };\r\n\r\n@BaseKb3dModel({\r\n token: Token.Feature,\r\n})\r\nexport abstract class Feature extends Kb3dObject implements IFeature {\r\n /**\r\n * Defines what kinds of geometry this feature depends on to function; if earlier features\r\n * that affect this geometry kind are modified, added or removed, then this feature gets disabled.\r\n * */\r\n static _dependsOn?: Set;\r\n\r\n /**\r\n * Defines what kinds of geometry this feature will affect; if this geometry kind is reset on the mesh,\r\n * then this feature will be re-processed.\r\n */\r\n static _affects?: Set;\r\n\r\n /**\r\n * Defines what geometry changes will cause this feature to require an update\r\n */\r\n static _affectedBy?: Set;\r\n\r\n get dependsOn() {\r\n return (this.constructor as typeof Feature)._dependsOn;\r\n }\r\n\r\n get affects() {\r\n return (this.constructor as typeof Feature)._affects;\r\n }\r\n\r\n get affectedBy() {\r\n return (this.constructor as typeof Feature)._affectedBy;\r\n }\r\n\r\n constructor() {\r\n super();\r\n }\r\n\r\n public enabled: boolean;\r\n public _suppressed = false;\r\n\r\n //this gets set to true when a change in the feature stack causes this feature to break\r\n public _disableFeatureStackChange: boolean;\r\n\r\n /** override base parent for more specific typing*/\r\n public _parent: SpaceNode | undefined;\r\n\r\n public getAncestors() {\r\n return super.getAncestors() as SpaceNode[];\r\n }\r\n}\r\n\r\nexport function isFeatureNode(node: Kb3dObject): node is Feature {\r\n return instanceOf(node, Token.Feature);\r\n}\r\n","import { BaseKb3dModel } from '../decorators';\r\nimport { eConnectorPositionMode } from '../enums';\r\nimport { IBaseConnector } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { SpaceNode } from '../meshes/space-node';\r\nimport { IKbVectorLike, KbVector } from '../primitives';\r\nimport { Token } from '../token';\r\n\r\n@BaseKb3dModel({\r\n token: Token.BaseConnector,\r\n})\r\nexport abstract class BaseConnector extends Kb3dObject implements IBaseConnector {\r\n public enabled: boolean;\r\n public positionMode: eConnectorPositionMode;\r\n public position: KbVector;\r\n public direction: KbVector;\r\n public angle: number;\r\n public tag: string;\r\n public positionZeroToOne: number;\r\n public interpolateDirection: boolean;\r\n\r\n public _visible: boolean;\r\n public _isPickable: boolean;\r\n public _sketchControlPoints: IKbVectorLike[] = [];\r\n\r\n /** Extended by child classes to return the space node parent of this connector */\r\n public getSpaceParent(): SpaceNode | undefined {\r\n return undefined;\r\n }\r\n}\r\n","let defaults = {\r\n AnnotationNode: {\r\n enabled: true,\r\n label: null,\r\n fontFamily: null,\r\n fontSize: 0.1,\r\n lineSpacing: 0,\r\n preloadFontStylesheet: \"\",\r\n color: {\"$type\":\"KbColor\",\"r\":1.0,\"g\":1.0,\"b\":1.0,\"a\":1.0},\r\n targetId: null,\r\n billboard: false,\r\n isPickable: false,\r\n keepOnTop: false,\r\n lineStyle: \"solid\",\r\n horizontalTextAlign: \"left\",\r\n verticalBoxAlign: \"middle\",\r\n horizontalBoxAlign: \"center\",\r\n offset: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n name: null,\r\n metadata: {},\r\n },\r\n DimensionNode: {\r\n target2Id: null,\r\n lineCapStyle: \"line\",\r\n axis: \"free\",\r\n lineThickness: 0.05,\r\n dimensionGap: 0.1,\r\n dimensionUnitPrecision: 2,\r\n labelOffset: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n scaleWithCamera: false,\r\n enabled: true,\r\n label: null,\r\n fontFamily: null,\r\n fontSize: 0.1,\r\n lineSpacing: 0,\r\n preloadFontStylesheet: \"\",\r\n color: {\"$type\":\"KbColor\",\"r\":1.0,\"g\":1.0,\"b\":1.0,\"a\":1.0},\r\n targetId: null,\r\n billboard: false,\r\n isPickable: false,\r\n keepOnTop: false,\r\n lineStyle: \"solid\",\r\n horizontalTextAlign: \"left\",\r\n verticalBoxAlign: \"middle\",\r\n horizontalBoxAlign: \"center\",\r\n offset: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n name: null,\r\n metadata: {},\r\n },\r\n BaseConnector: {\r\n positionMode: \"boundingBox\",\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n direction: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":1.0,\"z\":0.0},\r\n angle: 0,\r\n positionZeroToOne: 0,\r\n interpolateDirection: true,\r\n name: null,\r\n metadata: {},\r\n },\r\n Connector: {\r\n enabled: true,\r\n positionMode: \"boundingBox\",\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n direction: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":1.0,\"z\":0.0},\r\n angle: 0,\r\n positionZeroToOne: 0,\r\n interpolateDirection: true,\r\n name: null,\r\n metadata: {},\r\n },\r\n CircularPatternFeature: {\r\n arc: 360,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n axis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":1.0,\"z\":0.0},\r\n count: 1,\r\n meshId: null,\r\n distance: 1,\r\n addEndMesh: false,\r\n enabled: true,\r\n name: null,\r\n metadata: {},\r\n },\r\n DeleteFacesFeature: {\r\n enabled: true,\r\n name: null,\r\n metadata: {},\r\n },\r\n DisplacementMapFeature: {\r\n minDisplacement: 0,\r\n maxDisplacement: 1,\r\n enabled: true,\r\n name: null,\r\n metadata: {},\r\n },\r\n ExtrudeFeature: {\r\n size: 0,\r\n capEnd: false,\r\n capStart: false,\r\n enabled: true,\r\n name: null,\r\n metadata: {},\r\n },\r\n Feature: {\r\n enabled: true,\r\n name: null,\r\n metadata: {},\r\n },\r\n JoinGeometryFeature: {\r\n joinType: \"cut\",\r\n meshId: null,\r\n enabled: true,\r\n name: null,\r\n metadata: {},\r\n },\r\n LinearPatternFeature: {\r\n translation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n count: 1,\r\n meshId: null,\r\n distance: 1,\r\n addEndMesh: false,\r\n enabled: true,\r\n name: null,\r\n metadata: {},\r\n },\r\n MirrorFeature: {\r\n cloneGeometry: false,\r\n intersect: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n normal: {\"$type\":\"KbVector\",\"x\":1.0,\"y\":1.0,\"z\":1.0},\r\n removeGeometry: false,\r\n enabled: true,\r\n name: null,\r\n metadata: {},\r\n },\r\n MoveVerticesFeature: {\r\n translation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scale: {\"$type\":\"KbVector\",\"x\":1.0,\"y\":1.0,\"z\":1.0},\r\n center: \"barycenter\",\r\n enabled: true,\r\n name: null,\r\n metadata: {},\r\n },\r\n NormalSmoothingFeature: {\r\n smooth: true,\r\n enabled: true,\r\n name: null,\r\n metadata: {},\r\n },\r\n PatternFeature: {\r\n count: 1,\r\n meshId: null,\r\n distance: 1,\r\n addEndMesh: false,\r\n enabled: true,\r\n name: null,\r\n metadata: {},\r\n },\r\n SliceFeature: {\r\n intersect: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n normal: {\"$type\":\"KbVector\",\"x\":1.0,\"y\":1.0,\"z\":1.0},\r\n removeGeometry: false,\r\n enabled: true,\r\n name: null,\r\n metadata: {},\r\n },\r\n SubmeshFeature: {\r\n uvMapping: false,\r\n materialId: null,\r\n center: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n projectionAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":1.0,\"z\":0.0},\r\n angle: 0,\r\n scale: {\"$type\":\"KbVector\",\"x\":1.0,\"y\":1.0,\"z\":1.0},\r\n uTile: 1,\r\n vTile: 1,\r\n uOffset: 0,\r\n vOffset: 0,\r\n mode: \"planar\",\r\n enabled: true,\r\n name: null,\r\n metadata: {},\r\n },\r\n SweepFeature: {\r\n start: null,\r\n end: null,\r\n path: null,\r\n mitredExtrusion: false,\r\n size: 0,\r\n capEnd: false,\r\n capStart: false,\r\n enabled: true,\r\n name: null,\r\n metadata: {},\r\n },\r\n TransformFeature: {\r\n scale: {\"$type\":\"KbVector\",\"x\":1.0,\"y\":1.0,\"z\":1.0},\r\n enabled: true,\r\n name: null,\r\n metadata: {},\r\n },\r\n UvMapFeature: {\r\n center: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n projectionAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":1.0,\"z\":0.0},\r\n angle: 0,\r\n scale: {\"$type\":\"KbVector\",\"x\":1.0,\"y\":1.0,\"z\":1.0},\r\n uTile: 1,\r\n vTile: 1,\r\n uOffset: 0,\r\n vOffset: 0,\r\n mode: \"planar\",\r\n enabled: true,\r\n name: null,\r\n metadata: {},\r\n },\r\n WeldVerticesFeature: {\r\n enabled: true,\r\n name: null,\r\n metadata: {},\r\n },\r\n Kb3dObject: {\r\n name: null,\r\n metadata: {},\r\n },\r\n DirectionalLightNode: {\r\n direction: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":1.0,\"z\":0.0},\r\n generatesShadows: false,\r\n shadowQuality: \"medium\",\r\n shadowResolution: \"low\",\r\n shadowFilter: \"simple\",\r\n processTransparency: false,\r\n intensity: 1,\r\n color: {\"$type\":\"KbColor\",\"r\":0.0,\"g\":0.0,\"b\":0.0,\"a\":1.0},\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n HemisphericLightNode: {\r\n direction: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":1.0,\"z\":0.0},\r\n groundColor: {\"$type\":\"KbColor\",\"r\":0.0,\"g\":0.0,\"b\":0.0,\"a\":1.0},\r\n intensity: 1,\r\n color: {\"$type\":\"KbColor\",\"r\":0.0,\"g\":0.0,\"b\":0.0,\"a\":1.0},\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n LightNode: {\r\n intensity: 1,\r\n color: {\"$type\":\"KbColor\",\"r\":0.0,\"g\":0.0,\"b\":0.0,\"a\":1.0},\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n PointLightNode: {\r\n generatesShadows: false,\r\n shadowQuality: \"medium\",\r\n shadowResolution: \"low\",\r\n shadowFilter: \"simple\",\r\n processTransparency: false,\r\n intensity: 1,\r\n color: {\"$type\":\"KbColor\",\"r\":0.0,\"g\":0.0,\"b\":0.0,\"a\":1.0},\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n ShadowLightNode: {\r\n generatesShadows: false,\r\n shadowQuality: \"medium\",\r\n shadowResolution: \"low\",\r\n shadowFilter: \"simple\",\r\n processTransparency: false,\r\n intensity: 1,\r\n color: {\"$type\":\"KbColor\",\"r\":0.0,\"g\":0.0,\"b\":0.0,\"a\":1.0},\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n SpotLightNode: {\r\n coneAngle: 60,\r\n direction: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":1.0,\"z\":0.0},\r\n generatesShadows: false,\r\n shadowQuality: \"medium\",\r\n shadowResolution: \"low\",\r\n shadowFilter: \"simple\",\r\n processTransparency: false,\r\n intensity: 1,\r\n color: {\"$type\":\"KbColor\",\"r\":0.0,\"g\":0.0,\"b\":0.0,\"a\":1.0},\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n AdvancedMaterialNode: {\r\n workflowMode: \"metallicRoughness\",\r\n backFaceCulling: false,\r\n directIntensity: 1,\r\n emissiveIntensity: 1,\r\n environmentIntensity: 1,\r\n specularIntensity: 1,\r\n metallic: 0,\r\n roughness: 1,\r\n microSurface: 1,\r\n useSpecularOverAlpha: true,\r\n useRadianceOverAlpha: true,\r\n ambientColor: {\"$type\":\"KbColor\",\"r\":1.0,\"g\":1.0,\"b\":1.0,\"a\":1.0},\r\n albedoColor: {\"$type\":\"KbColor\",\"r\":1.0,\"g\":1.0,\"b\":1.0,\"a\":1.0},\r\n specularColor: {\"$type\":\"KbColor\",\"r\":1.0,\"g\":1.0,\"b\":1.0,\"a\":1.0},\r\n reflectionColor: {\"$type\":\"KbColor\",\"r\":1.0,\"g\":1.0,\"b\":1.0,\"a\":1.0},\r\n anisotropyEnabled: false,\r\n anisotropicDirection: {\"$type\":\"KbVector\",\"x\":1.0,\"y\":1.0,\"z\":1.0},\r\n anisotropicIntensity: 1,\r\n subsurfaceThickness: 1,\r\n subsurfaceRefractionEnabled: false,\r\n subsurfaceReflectionEnabled: false,\r\n reflectionType: \"plane\",\r\n probePositionTargetId: null,\r\n subsurfaceRefractionIndex: 1,\r\n subsurfaceRefractionIntensity: 1,\r\n subsurfaceRefractionLinkTransparency: false,\r\n subsurfaceTranslucencyEnabled: false,\r\n subsurfaceTranslucencyIntensity: 1,\r\n subsurfaceUseAlbedoForTintColor: true,\r\n subsurfaceTintColor: {\"$type\":\"KbColor\",\"r\":1.0,\"g\":1.0,\"b\":1.0,\"a\":1.0},\r\n clearcoatEnabled: false,\r\n clearcoatIntensity: 1,\r\n clearcoatTintEnabled: false,\r\n clearcoatTintColor: {\"$type\":\"KbColor\",\"r\":1.0,\"g\":1.0,\"b\":1.0,\"a\":1.0},\r\n clearcoatTintIntensity: 0.1,\r\n archived: false,\r\n location: \"local\",\r\n alpha: 1,\r\n zOffset: 0,\r\n doubleSided: false,\r\n legacyTextureStacking: true,\r\n viewMode: \"shaded\",\r\n emissiveColor: {\"$type\":\"KbColor\",\"r\":0.0,\"g\":0.0,\"b\":0.0,\"a\":1.0},\r\n hasTransparency: false,\r\n uOffset: 0,\r\n vOffset: 0,\r\n uScale: 1,\r\n vScale: 1,\r\n uAngle: 0,\r\n vAngle: 0,\r\n wAngle: 0,\r\n uPivot: 0.5,\r\n vPivot: 0.5,\r\n wPivot: 0.5,\r\n name: null,\r\n metadata: {},\r\n },\r\n ColorFillTextureLayer: {\r\n color: {\"$type\":\"KbColor\",\"r\":0.0,\"g\":0.0,\"b\":0.0,\"a\":1.0},\r\n width: 256,\r\n height: 256,\r\n enabled: true,\r\n strength: 1,\r\n tile: false,\r\n maintainAspect: false,\r\n useClippingMask: false,\r\n blendMode: \"normal\",\r\n horizontalAlign: \"center\",\r\n verticalAlign: \"middle\",\r\n horizontalFit: \"none\",\r\n verticalFit: \"none\",\r\n uOffset: 0,\r\n vOffset: 0,\r\n uScale: 1,\r\n vScale: 1,\r\n uAngle: 0,\r\n vAngle: 0,\r\n wAngle: 0,\r\n uPivot: 0.5,\r\n vPivot: 0.5,\r\n wPivot: 0.5,\r\n name: null,\r\n metadata: {},\r\n },\r\n FileTextureLayer: {\r\n width: 256,\r\n height: 256,\r\n flipNormalsRg: false,\r\n enabled: true,\r\n strength: 1,\r\n tile: false,\r\n maintainAspect: false,\r\n useClippingMask: false,\r\n blendMode: \"normal\",\r\n horizontalAlign: \"center\",\r\n verticalAlign: \"middle\",\r\n horizontalFit: \"none\",\r\n verticalFit: \"none\",\r\n uOffset: 0,\r\n vOffset: 0,\r\n uScale: 1,\r\n vScale: 1,\r\n uAngle: 0,\r\n vAngle: 0,\r\n wAngle: 0,\r\n uPivot: 0.5,\r\n vPivot: 0.5,\r\n wPivot: 0.5,\r\n name: null,\r\n metadata: {},\r\n },\r\n MaterialMapping: {\r\n uOffset: 0,\r\n vOffset: 0,\r\n uScale: 1,\r\n vScale: 1,\r\n uAngle: 0,\r\n vAngle: 0,\r\n wAngle: 0,\r\n uPivot: 0.5,\r\n vPivot: 0.5,\r\n wPivot: 0.5,\r\n name: null,\r\n metadata: {},\r\n },\r\n MaterialNode: {\r\n location: \"local\",\r\n alpha: 1,\r\n zOffset: 0,\r\n reflectionType: \"plane\",\r\n doubleSided: false,\r\n legacyTextureStacking: true,\r\n viewMode: \"shaded\",\r\n emissiveColor: {\"$type\":\"KbColor\",\"r\":0.0,\"g\":0.0,\"b\":0.0,\"a\":1.0},\r\n hasTransparency: false,\r\n uOffset: 0,\r\n vOffset: 0,\r\n uScale: 1,\r\n vScale: 1,\r\n uAngle: 0,\r\n vAngle: 0,\r\n wAngle: 0,\r\n uPivot: 0.5,\r\n vPivot: 0.5,\r\n wPivot: 0.5,\r\n name: null,\r\n metadata: {},\r\n },\r\n SvgTextureLayer: {\r\n width: 256,\r\n height: 256,\r\n preloadFontStylesheet: \"\",\r\n enabled: true,\r\n strength: 1,\r\n tile: false,\r\n maintainAspect: false,\r\n useClippingMask: false,\r\n blendMode: \"normal\",\r\n horizontalAlign: \"center\",\r\n verticalAlign: \"middle\",\r\n horizontalFit: \"none\",\r\n verticalFit: \"none\",\r\n uOffset: 0,\r\n vOffset: 0,\r\n uScale: 1,\r\n vScale: 1,\r\n uAngle: 0,\r\n vAngle: 0,\r\n wAngle: 0,\r\n uPivot: 0.5,\r\n vPivot: 0.5,\r\n wPivot: 0.5,\r\n name: null,\r\n metadata: {},\r\n },\r\n TextTextureLayer: {\r\n text: \"\",\r\n color: {\"$type\":\"KbColor\",\"r\":1.0,\"g\":1.0,\"b\":1.0,\"a\":1.0},\r\n enabled: true,\r\n strength: 1,\r\n tile: false,\r\n maintainAspect: false,\r\n useClippingMask: false,\r\n blendMode: \"normal\",\r\n horizontalAlign: \"center\",\r\n verticalAlign: \"middle\",\r\n horizontalFit: \"none\",\r\n verticalFit: \"none\",\r\n uOffset: 0,\r\n vOffset: 0,\r\n uScale: 1,\r\n vScale: 1,\r\n uAngle: 0,\r\n vAngle: 0,\r\n wAngle: 0,\r\n uPivot: 0.5,\r\n vPivot: 0.5,\r\n wPivot: 0.5,\r\n name: null,\r\n metadata: {},\r\n },\r\n TextureLayer: {\r\n enabled: true,\r\n strength: 1,\r\n tile: false,\r\n maintainAspect: false,\r\n useClippingMask: false,\r\n blendMode: \"normal\",\r\n horizontalAlign: \"center\",\r\n verticalAlign: \"middle\",\r\n horizontalFit: \"none\",\r\n verticalFit: \"none\",\r\n uOffset: 0,\r\n vOffset: 0,\r\n uScale: 1,\r\n vScale: 1,\r\n uAngle: 0,\r\n vAngle: 0,\r\n wAngle: 0,\r\n uPivot: 0.5,\r\n vPivot: 0.5,\r\n wPivot: 0.5,\r\n name: null,\r\n metadata: {},\r\n },\r\n MateNode: {\r\n enabled: true,\r\n connector1: null,\r\n connector2: null,\r\n mateType: \"fastened\",\r\n flip: false,\r\n offsetX: 0,\r\n offsetY: 0,\r\n offsetZ: 0,\r\n offsetAngle: 0,\r\n name: null,\r\n metadata: {},\r\n },\r\n BoxNode: {\r\n height: 1,\r\n width: 1,\r\n depth: 1,\r\n tessellation: 1,\r\n sideOrientation: \"frontSide\",\r\n materialId: null,\r\n flipSurfaceOrientation: false,\r\n receiveShadows: false,\r\n castShadows: false,\r\n isPickable: true,\r\n checkCollisions: false,\r\n opacity: 1,\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n ConeNode: {\r\n height: 1,\r\n radiusTop: 0,\r\n radiusBottom: 0.5,\r\n tessellation: 24,\r\n sideOrientation: \"frontSide\",\r\n materialId: null,\r\n flipSurfaceOrientation: false,\r\n receiveShadows: false,\r\n castShadows: false,\r\n isPickable: true,\r\n checkCollisions: false,\r\n opacity: 1,\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n CustomMeshNode: {\r\n materialId: null,\r\n flipSurfaceOrientation: false,\r\n receiveShadows: false,\r\n castShadows: false,\r\n isPickable: true,\r\n checkCollisions: false,\r\n opacity: 1,\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n CylinderNode: {\r\n height: 1,\r\n radius: 0.5,\r\n arc: 360,\r\n tessellation: 24,\r\n subdivisions: 1,\r\n capMode: \"both\",\r\n sideOrientation: \"frontSide\",\r\n materialId: null,\r\n flipSurfaceOrientation: false,\r\n receiveShadows: false,\r\n castShadows: false,\r\n isPickable: true,\r\n checkCollisions: false,\r\n opacity: 1,\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n DiscNode: {\r\n radius: 0.5,\r\n tessellation: 64,\r\n sideOrientation: \"frontSide\",\r\n materialId: null,\r\n flipSurfaceOrientation: false,\r\n receiveShadows: false,\r\n castShadows: false,\r\n isPickable: true,\r\n checkCollisions: false,\r\n opacity: 1,\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n GroundNode: {\r\n height: 1,\r\n width: 1,\r\n subdivisions: 1,\r\n sideOrientation: \"frontSide\",\r\n materialId: null,\r\n flipSurfaceOrientation: false,\r\n receiveShadows: false,\r\n castShadows: false,\r\n isPickable: true,\r\n checkCollisions: false,\r\n opacity: 1,\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n MeshGroupNode: {\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n MeshNode: {\r\n materialId: null,\r\n flipSurfaceOrientation: false,\r\n receiveShadows: false,\r\n castShadows: false,\r\n isPickable: true,\r\n checkCollisions: false,\r\n opacity: 1,\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n PlaneNode: {\r\n depth: 4,\r\n width: 4,\r\n segments: 1,\r\n sideOrientation: \"frontSide\",\r\n materialId: null,\r\n flipSurfaceOrientation: false,\r\n receiveShadows: false,\r\n castShadows: false,\r\n isPickable: true,\r\n checkCollisions: false,\r\n opacity: 1,\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n PrimitiveMeshNode: {\r\n sideOrientation: \"frontSide\",\r\n materialId: null,\r\n flipSurfaceOrientation: false,\r\n receiveShadows: false,\r\n castShadows: false,\r\n isPickable: true,\r\n checkCollisions: false,\r\n opacity: 1,\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n SketchControlPoint: {\r\n positionMode: \"absolute\",\r\n direction2: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":-1.0,\"z\":0.0},\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n direction: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":1.0,\"z\":0.0},\r\n angle: 0,\r\n positionZeroToOne: 0,\r\n interpolateDirection: true,\r\n name: null,\r\n metadata: {},\r\n },\r\n SketchNode: {\r\n normal: \"y\",\r\n addFace: false,\r\n closePath: false,\r\n tracePath: false,\r\n lineThickness: 1,\r\n pathColor: {\"$type\":\"KbColor\",\"r\":1.0,\"g\":1.0,\"b\":1.0,\"a\":1.0},\r\n flipPath: false,\r\n clonePath: null,\r\n boundingXOverride: null,\r\n boundingYOverride: null,\r\n boundingZOverride: null,\r\n materialId: null,\r\n flipSurfaceOrientation: false,\r\n receiveShadows: false,\r\n castShadows: false,\r\n isPickable: true,\r\n checkCollisions: false,\r\n opacity: 1,\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n SketchPath: {\r\n pathGroupKey: \"default\",\r\n holeForPathGroup: null,\r\n type: \"points\",\r\n tessellation: 16,\r\n name: null,\r\n metadata: {},\r\n },\r\n SpaceNode: {\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n SphereNode: {\r\n tessellation: 32,\r\n radius: 0.5,\r\n latitudeStart: 0,\r\n latitudeSlice: 180,\r\n arc: 360,\r\n sideOrientation: \"frontSide\",\r\n materialId: null,\r\n flipSurfaceOrientation: false,\r\n receiveShadows: false,\r\n castShadows: false,\r\n isPickable: true,\r\n checkCollisions: false,\r\n opacity: 1,\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n TextNode: {\r\n curveRadius: 0,\r\n charSpacing: 1,\r\n text: \"\",\r\n fontFile: \"\",\r\n fontSize: 1,\r\n materialId: null,\r\n flipSurfaceOrientation: false,\r\n receiveShadows: false,\r\n castShadows: false,\r\n isPickable: true,\r\n checkCollisions: false,\r\n opacity: 1,\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n TorusNode: {\r\n radius: 0.5,\r\n thickness: 0.25,\r\n arc: 360,\r\n latitudeStart: 0,\r\n latitudeSlice: 360,\r\n tessellation: 16,\r\n sideOrientation: \"frontSide\",\r\n materialId: null,\r\n flipSurfaceOrientation: false,\r\n receiveShadows: false,\r\n castShadows: false,\r\n isPickable: true,\r\n checkCollisions: false,\r\n opacity: 1,\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n SceneNode: {\r\n background: \"transparent\",\r\n environment: \"library://adams_place_bridge\",\r\n environmentBlur: 0.27,\r\n environmentIntensity: 1,\r\n backgroundColor: {\"$type\":\"KbColor\",\"r\":0.0,\"g\":0.0,\"b\":0.0,\"a\":0.0},\r\n ambientColor: {\"$type\":\"KbColor\",\"r\":0.0,\"g\":0.0,\"b\":0.0,\"a\":1.0},\r\n renderMode: \"shaded\",\r\n edgeColor: {\"$type\":\"KbColor\",\"r\":1.0,\"g\":1.0,\"b\":1.0,\"a\":1.0},\r\n edgeWidth: 1,\r\n edgeDisplayTolerance: 0.95,\r\n viewpointId: null,\r\n frameViewpoint: false,\r\n progressiveTextureLoad: true,\r\n enableDeferredSceneUpdates: true,\r\n enableGrid: false,\r\n enableGridAxes: false,\r\n gridSize: 10,\r\n gridStepSize: 1,\r\n gridMajorInterval: 1,\r\n gridNormal: \"y\",\r\n environmentRotation: 0,\r\n connectorSize: 0.1,\r\n enableFxaaPostProcess: false,\r\n enableLensEffectsPostProcess: false,\r\n enableSSAOPostProcess: false,\r\n textureSamplingMode: \"trilinear\",\r\n fullscreenEnabled: true,\r\n arEnabled: true,\r\n xrAllowScaling: false,\r\n unitsPerMeter: 1,\r\n xrAddShadow: true,\r\n xrLightingEstimation: false,\r\n enableHoverEffects: false,\r\n enableSceneOptimization: true,\r\n loadingExperience: \"spinner\",\r\n loadingImagePath: null,\r\n dimensionUnitScalar: 1,\r\n dimensionUnitSymbol: \"m\",\r\n conversionMap: {},\r\n visible: true,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n useEuler: false,\r\n eulerRotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n rotationAngle: 0,\r\n scaling: 1,\r\n pivot: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fixed: false,\r\n name: null,\r\n metadata: {},\r\n },\r\n ViewpointNode: {\r\n type: \"orbit\",\r\n targetMode: \"point\",\r\n projection: \"perspective\",\r\n offsetX: 0,\r\n offsetY: 0,\r\n position: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n targetPoint: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n targetMeshId: null,\r\n rotation: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":0.0,\"z\":0.0},\r\n fov: 50,\r\n allowOrbit: true,\r\n allowPan: true,\r\n allowZoom: true,\r\n limitMinimumTargetDistance: false,\r\n minTargetDistance: 0.001,\r\n maxTargetDistance: 10000,\r\n orbitAxis: {\"$type\":\"KbVector\",\"x\":0.0,\"y\":1.0,\"z\":0.0},\r\n limitLatOrbit: false,\r\n limitMinLatitude: 0,\r\n limitMaxLatitude: 180,\r\n limitLngOrbit: false,\r\n limitMinLongitude: -180,\r\n limitMaxLongitude: 180,\r\n name: null,\r\n metadata: {},\r\n },\r\n};\r\nexport { defaults };","import { BaseKb3dModel } from '../decorators/kb3d-decorator';\r\nimport { IMaterialMapping } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { Token } from '../token';\r\n\r\n@BaseKb3dModel({\r\n token: Token.MaterialMapping,\r\n})\r\nexport abstract class MaterialMapping extends Kb3dObject implements IMaterialMapping {\r\n uOffset = 0; // Define an offset on the texture to offset the u coordinates of the UVs\r\n vOffset = 0; // Define an offset on the texture to offset the v coordinates of the UVs\r\n uScale = 1; // Define an offset on the texture to scale the u coordinates of the UVs\r\n vScale = 1; // Define an offset on the texture to scale the v coordinates of the UVs\r\n uAngle = 0; // Define an offset on the texture to rotate around the u coordinates of the UVs\r\n vAngle = 0; // Define an offset on the texture to rotate around the v coordinates of the UVs\r\n wAngle = 0; // Define an offset on the texture to rotate around the w coordinates of the UVs\r\n uPivot = 0.5; // Defines the center of rotation (U)\r\n vPivot = 0.5; // Defines the center of rotation (V)\r\n wPivot = 0.5; // Defines the center of rotation (W)\r\n}\r\n","import { isTrackedClass } from '../../interfaces/tracked-class';\r\nimport { IMaterialService } from '../../services/material-service';\r\nimport { IgnoreChanges } from '../decorators/change-tracker';\r\nimport { Enumerable } from '../decorators/enumerable';\r\nimport { BaseKb3dModel } from '../decorators/kb3d-decorator';\r\nimport { instanceOf, instanceOfAny } from '../decorators/model-utils';\r\nimport { defaults } from '../default-values';\r\nimport { eMaterialLocation, eMaterialViewMode, eReflectionType } from '../enums';\r\nimport { IMaterialNode, IReference, ITextureLayer } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { KbColor } from '../primitives/kb-color';\r\nimport { Token } from '../token';\r\nimport { isAbsoluteUrl } from '../utilities';\r\nimport { AdvancedMaterialNode } from './advanced-material-node';\r\nimport { MaterialMapping } from './material-mapping';\r\nimport { TextureLayer } from './texture-layer';\r\n\r\nconst NEVER_INHERIT = new Set([\r\n '$type',\r\n 'overriddenProperties',\r\n '_parentMaterial',\r\n 'parentMaterialId',\r\n 'location',\r\n 'children',\r\n '_selected',\r\n '_expanded',\r\n 'id',\r\n 'name',\r\n 'visible',\r\n '_parent',\r\n '_parentScene',\r\n '_rootScene',\r\n '_level',\r\n]);\r\n\r\n@BaseKb3dModel({\r\n token: Token.MaterialNode,\r\n})\r\nexport abstract class MaterialNode extends MaterialMapping implements IMaterialNode {\r\n constructor(public _materialService: IMaterialService) {\r\n super();\r\n\r\n this._initialized = false;\r\n }\r\n\r\n protected _materials: MaterialNode[];\r\n @Enumerable()\r\n public get materials(): MaterialNode[] {\r\n if (!this._materials) this._materials = [];\r\n return this._materials;\r\n }\r\n public set materials(val) {\r\n this._materials = val;\r\n }\r\n\r\n @IgnoreChanges\r\n _initialized: boolean;\r\n\r\n // Location is where this material is actually defined/saved. Here we also assign the icon this material uses in the tree.\r\n // Local is unique to this scene\r\n // Global is saved to this tenant's account and can be used in any of this tenant's scenes\r\n // Library is defined by kbmax and can be used on any scene by any tenant.\r\n @IgnoreChanges\r\n location: eMaterialLocation;\r\n\r\n // Id of material to inherit from. This material will be loaded and the user can copy properties from this material.\r\n @IgnoreChanges\r\n parentMaterialId?: string;\r\n _parentMaterial?: IMaterialNode;\r\n\r\n public _overriddenPropertiesSet = new Set();\r\n\r\n /** Implementation detail for persisting to database, do not use this property in the client code */\r\n private _overriddenProperties: string[];\r\n @Enumerable()\r\n public get overriddenProperties(): string[] {\r\n if (!this._overriddenProperties) this._overriddenProperties = [];\r\n return this._overriddenProperties;\r\n }\r\n public set overriddenProperties(val) {\r\n this._overriddenProperties = val;\r\n }\r\n\r\n /**\r\n * Archived: When this is set to true, material will be archived and\r\n * will not automatically be displayed when materials get loaded.\r\n */\r\n archived: boolean;\r\n\r\n private _textures: TextureLayer[];\r\n @Enumerable()\r\n public get textures(): TextureLayer[] {\r\n if (!this._textures) this._textures = [];\r\n return this._textures;\r\n }\r\n public set textures(val) {\r\n this._textures = val;\r\n }\r\n\r\n alpha: number;\r\n emissiveColor?: KbColor;\r\n viewMode: eMaterialViewMode;\r\n zOffset?: number;\r\n doubleSided: boolean;\r\n hasTransparency: boolean;\r\n probePositionTargetId?: string;\r\n reflectionType?: eReflectionType;\r\n @IgnoreChanges\r\n legacyTextureStacking: boolean;\r\n\r\n init() {\r\n this.refreshParentMaterial();\r\n\r\n if (!this._initialized) {\r\n if (isTrackedClass(this)) {\r\n // Always track changes on initialized materials because they may be global materials\r\n this.trackChanges(true);\r\n this.isModified = true;\r\n }\r\n this._initialized = true;\r\n\r\n this.textures.forEach(texture => {\r\n texture.init(this);\r\n });\r\n }\r\n }\r\n\r\n getChildren() {\r\n return [...this.textures, ...this.materials];\r\n }\r\n\r\n getChildArrayForType(token: Token): Kb3dObject[] | undefined {\r\n if (token.isMaterial()) {\r\n return this.materials;\r\n } else if (token.isTexture()) {\r\n return this.textures;\r\n } else {\r\n throw 'unexpected token';\r\n }\r\n }\r\n\r\n overrideProperty(property: string | symbol | number) {\r\n if (this.parentMaterialId) {\r\n this._overriddenPropertiesSet.add(property);\r\n }\r\n }\r\n\r\n inheritProperty(property: string | symbol | number) {\r\n if (this.parentMaterialId) {\r\n this._overriddenPropertiesSet.delete(property);\r\n\r\n if (this._parentMaterial) {\r\n (this as any)[property] =\r\n (this._parentMaterial as any)[property] == null\r\n ? (defaults as any)[this.$type][property]\r\n : (this._parentMaterial as any)[property];\r\n }\r\n }\r\n }\r\n\r\n getTextureByName(name: string) {\r\n return [...this.textures].find(f => f.name.isEqual(name));\r\n }\r\n\r\n protected inheritFromMaterial(parentMat: IMaterialNode) {\r\n this.parentMaterialId = parentMat.id;\r\n this._parentMaterial = parentMat;\r\n if (!this._manager) {\r\n return;\r\n }\r\n const manager = this._manager;\r\n\r\n NEVER_INHERIT.forEach(prop => {\r\n this._overriddenPropertiesSet.add(prop);\r\n });\r\n\r\n for (const _key in parentMat) {\r\n const key = _key as keyof IMaterialNode;\r\n if (!this._overriddenPropertiesSet.has(key)) {\r\n //if this is a serialization of a texture node, then we need to deserialize it\r\n if (Array.isArray(parentMat[key])) {\r\n const currentTextures = (this as any)[key] as TextureLayer[];\r\n (parentMat[key] as ITextureLayer[]).forEach((tex, indexOnParent) => {\r\n const texDef = { ...tex, name: `Base ${tex.name || tex.channel}` };\r\n if (!texDef.id) {\r\n texDef.id = `${parentMat.id}_${tex.channel}`;\r\n }\r\n\r\n const currentIndex = currentTextures.findIndex(\r\n currentTexture => currentTexture.id === texDef.id\r\n );\r\n if (currentIndex < 0) {\r\n const textureNode = manager.deserialize(texDef) as TextureLayer;\r\n textureNode._external = true;\r\n // If this material is already initialized, make sure the new texture is also initialized.\r\n if (this._initialized) {\r\n textureNode.init(this);\r\n }\r\n currentTextures.splice(indexOnParent, 0, textureNode);\r\n } else {\r\n // currentTextures.splice(currentIndex, 1, textureNode);\r\n }\r\n });\r\n } else {\r\n (this as any)[key] =\r\n parentMat[key] instanceof Kb3dObject ? parentMat[key] : manager.deserialize(parentMat[key]);\r\n }\r\n }\r\n }\r\n }\r\n\r\n public refreshParentMaterial() {\r\n if (this.parentMaterialId) {\r\n this._materialService.getMaterialById(this.parentMaterialId).subscribe(parentMaterial => {\r\n if (parentMaterial && parentMaterial.data) {\r\n this.inheritFromMaterial(parentMaterial.data as any);\r\n }\r\n });\r\n }\r\n }\r\n\r\n public getMediaReferences() {\r\n const refs = new Array();\r\n for (const texture of this.textures) {\r\n if (instanceOf(texture, Token.FileTextureLayer) && texture.path && !isAbsoluteUrl(texture.path)) {\r\n refs.push({\r\n type: 'Media',\r\n id: 'media/' + texture.path,\r\n });\r\n }\r\n }\r\n return refs;\r\n }\r\n}\r\n\r\nexport function isMaterialNode(node: Kb3dObject): node is MaterialNode {\r\n return instanceOfAny(node, ...Token.MaterialTokens);\r\n}\r\n","\n/******************************************************************************\nCopyright (c) Microsoft Corporation.\nPermission to use, copy, modify, and/or distribute this software for any\npurpose with or without fee is hereby granted.\nTHE SOFTWARE IS PROVIDED \"AS IS\" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH\nREGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY\nAND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,\nINDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM\nLOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR\nOTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR\nPERFORMANCE OF THIS SOFTWARE.\n***************************************************************************** */\n/* global Reflect, Promise */\n\nvar extendStatics = function(d, b) {\n extendStatics = Object.setPrototypeOf ||\n ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||\n function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; };\n return extendStatics(d, b);\n};\n\nexport function __extends(d, b) {\n if (typeof b !== \"function\" && b !== null)\n throw new TypeError(\"Class extends value \" + String(b) + \" is not a constructor or null\");\n extendStatics(d, b);\n function __() { this.constructor = d; }\n d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());\n}\n\nexport var __assign = function() {\n __assign = Object.assign || function __assign(t) {\n for (var s, i = 1, n = arguments.length; i < n; i++) {\n s = arguments[i];\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p];\n }\n return t;\n }\n return __assign.apply(this, arguments);\n}\n\nexport function __rest(s, e) {\n var t = {};\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)\n t[p] = s[p];\n if (s != null && typeof Object.getOwnPropertySymbols === \"function\")\n for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {\n if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))\n t[p[i]] = s[p[i]];\n }\n return t;\n}\n\nexport function __decorate(decorators, target, key, desc) {\n var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;\n if (typeof Reflect === \"object\" && typeof Reflect.decorate === \"function\") r = Reflect.decorate(decorators, target, key, desc);\n else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;\n return c > 3 && r && Object.defineProperty(target, key, r), r;\n}\n\nexport function __param(paramIndex, decorator) {\n return function (target, key) { decorator(target, key, paramIndex); }\n}\n\nexport function __metadata(metadataKey, metadataValue) {\n if (typeof Reflect === \"object\" && typeof Reflect.metadata === \"function\") return Reflect.metadata(metadataKey, metadataValue);\n}\n\nexport function __awaiter(thisArg, _arguments, P, generator) {\n function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n return new (P || (P = Promise))(function (resolve, reject) {\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n step((generator = generator.apply(thisArg, _arguments || [])).next());\n });\n}\n\nexport function __generator(thisArg, body) {\n var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;\n return g = { next: verb(0), \"throw\": verb(1), \"return\": verb(2) }, typeof Symbol === \"function\" && (g[Symbol.iterator] = function() { return this; }), g;\n function verb(n) { return function (v) { return step([n, v]); }; }\n function step(op) {\n if (f) throw new TypeError(\"Generator is already executing.\");\n while (_) try {\n if (f = 1, y && (t = op[0] & 2 ? y[\"return\"] : op[0] ? y[\"throw\"] || ((t = y[\"return\"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;\n if (y = 0, t) op = [op[0] & 2, t.value];\n switch (op[0]) {\n case 0: case 1: t = op; break;\n case 4: _.label++; return { value: op[1], done: false };\n case 5: _.label++; y = op[1]; op = [0]; continue;\n case 7: op = _.ops.pop(); _.trys.pop(); continue;\n default:\n if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }\n if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }\n if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }\n if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }\n if (t[2]) _.ops.pop();\n _.trys.pop(); continue;\n }\n op = body.call(thisArg, _);\n } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }\n if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };\n }\n}\n\nexport var __createBinding = Object.create ? (function(o, m, k, k2) {\n if (k2 === undefined) k2 = k;\n var desc = Object.getOwnPropertyDescriptor(m, k);\n if (!desc || (\"get\" in desc ? !m.__esModule : desc.writable || desc.configurable)) {\n desc = { enumerable: true, get: function() { return m[k]; } };\n }\n Object.defineProperty(o, k2, desc);\n}) : (function(o, m, k, k2) {\n if (k2 === undefined) k2 = k;\n o[k2] = m[k];\n});\n\nexport function __exportStar(m, o) {\n for (var p in m) if (p !== \"default\" && !Object.prototype.hasOwnProperty.call(o, p)) __createBinding(o, m, p);\n}\n\nexport function __values(o) {\n var s = typeof Symbol === \"function\" && Symbol.iterator, m = s && o[s], i = 0;\n if (m) return m.call(o);\n if (o && typeof o.length === \"number\") return {\n next: function () {\n if (o && i >= o.length) o = void 0;\n return { value: o && o[i++], done: !o };\n }\n };\n throw new TypeError(s ? \"Object is not iterable.\" : \"Symbol.iterator is not defined.\");\n}\n\nexport function __read(o, n) {\n var m = typeof Symbol === \"function\" && o[Symbol.iterator];\n if (!m) return o;\n var i = m.call(o), r, ar = [], e;\n try {\n while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value);\n }\n catch (error) { e = { error: error }; }\n finally {\n try {\n if (r && !r.done && (m = i[\"return\"])) m.call(i);\n }\n finally { if (e) throw e.error; }\n }\n return ar;\n}\n\n/** @deprecated */\nexport function __spread() {\n for (var ar = [], i = 0; i < arguments.length; i++)\n ar = ar.concat(__read(arguments[i]));\n return ar;\n}\n\n/** @deprecated */\nexport function __spreadArrays() {\n for (var s = 0, i = 0, il = arguments.length; i < il; i++) s += arguments[i].length;\n for (var r = Array(s), k = 0, i = 0; i < il; i++)\n for (var a = arguments[i], j = 0, jl = a.length; j < jl; j++, k++)\n r[k] = a[j];\n return r;\n}\n\nexport function __spreadArray(to, from, pack) {\n if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) {\n if (ar || !(i in from)) {\n if (!ar) ar = Array.prototype.slice.call(from, 0, i);\n ar[i] = from[i];\n }\n }\n return to.concat(ar || Array.prototype.slice.call(from));\n}\n\nexport function __await(v) {\n return this instanceof __await ? (this.v = v, this) : new __await(v);\n}\n\nexport function __asyncGenerator(thisArg, _arguments, generator) {\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\n var g = generator.apply(thisArg, _arguments || []), i, q = [];\n return i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i;\n function verb(n) { if (g[n]) i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; }\n function resume(n, v) { try { step(g[n](v)); } catch (e) { settle(q[0][3], e); } }\n function step(r) { r.value instanceof __await ? Promise.resolve(r.value.v).then(fulfill, reject) : settle(q[0][2], r); }\n function fulfill(value) { resume(\"next\", value); }\n function reject(value) { resume(\"throw\", value); }\n function settle(f, v) { if (f(v), q.shift(), q.length) resume(q[0][0], q[0][1]); }\n}\n\nexport function __asyncDelegator(o) {\n var i, p;\n return i = {}, verb(\"next\"), verb(\"throw\", function (e) { throw e; }), verb(\"return\"), i[Symbol.iterator] = function () { return this; }, i;\n function verb(n, f) { i[n] = o[n] ? function (v) { return (p = !p) ? { value: __await(o[n](v)), done: n === \"return\" } : f ? f(v) : v; } : f; }\n}\n\nexport function __asyncValues(o) {\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\n var m = o[Symbol.asyncIterator], i;\n return m ? m.call(o) : (o = typeof __values === \"function\" ? __values(o) : o[Symbol.iterator](), i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i);\n function verb(n) { i[n] = o[n] && function (v) { return new Promise(function (resolve, reject) { v = o[n](v), settle(resolve, reject, v.done, v.value); }); }; }\n function settle(resolve, reject, d, v) { Promise.resolve(v).then(function(v) { resolve({ value: v, done: d }); }, reject); }\n}\n\nexport function __makeTemplateObject(cooked, raw) {\n if (Object.defineProperty) { Object.defineProperty(cooked, \"raw\", { value: raw }); } else { cooked.raw = raw; }\n return cooked;\n};\n\nvar __setModuleDefault = Object.create ? (function(o, v) {\n Object.defineProperty(o, \"default\", { enumerable: true, value: v });\n}) : function(o, v) {\n o[\"default\"] = v;\n};\n\nexport function __importStar(mod) {\n if (mod && mod.__esModule) return mod;\n var result = {};\n if (mod != null) for (var k in mod) if (k !== \"default\" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);\n __setModuleDefault(result, mod);\n return result;\n}\n\nexport function __importDefault(mod) {\n return (mod && mod.__esModule) ? mod : { default: mod };\n}\n\nexport function __classPrivateFieldGet(receiver, state, kind, f) {\n if (kind === \"a\" && !f) throw new TypeError(\"Private accessor was defined without a getter\");\n if (typeof state === \"function\" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError(\"Cannot read private member from an object whose class did not declare it\");\n return kind === \"m\" ? f : kind === \"a\" ? f.call(receiver) : f ? f.value : state.get(receiver);\n}\n\nexport function __classPrivateFieldSet(receiver, state, value, kind, f) {\n if (kind === \"m\") throw new TypeError(\"Private method is not writable\");\n if (kind === \"a\" && !f) throw new TypeError(\"Private accessor was defined without a setter\");\n if (typeof state === \"function\" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError(\"Cannot write private member to an object whose class did not declare it\");\n return (kind === \"a\" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;\n}\n\nexport function __classPrivateFieldIn(state, receiver) {\n if (receiver === null || (typeof receiver !== \"object\" && typeof receiver !== \"function\")) throw new TypeError(\"Cannot use 'in' operator on non-object\");\n return typeof state === \"function\" ? receiver === state : state.has(receiver);\n}\n","/**\r\n * Class used to evaluate queries containing `and` and `or` operators\r\n */\r\nexport class AndOrNotEvaluator {\r\n /**\r\n * Evaluate a query\r\n * @param query defines the query to evaluate\r\n * @param evaluateCallback defines the callback used to filter result\r\n * @returns true if the query matches\r\n */\r\n public static Eval(query: string, evaluateCallback: (val: any) => boolean): boolean {\r\n if (!query.match(/\\([^()]*\\)/g)) {\r\n query = AndOrNotEvaluator._HandleParenthesisContent(query, evaluateCallback);\r\n } else {\r\n query = query.replace(/\\([^()]*\\)/g, (r) => {\r\n // remove parenthesis\r\n r = r.slice(1, r.length - 1);\r\n return AndOrNotEvaluator._HandleParenthesisContent(r, evaluateCallback);\r\n });\r\n }\r\n\r\n if (query === \"true\") {\r\n return true;\r\n }\r\n\r\n if (query === \"false\") {\r\n return false;\r\n }\r\n\r\n return AndOrNotEvaluator.Eval(query, evaluateCallback);\r\n }\r\n\r\n private static _HandleParenthesisContent(parenthesisContent: string, evaluateCallback: (val: string) => boolean): string {\r\n evaluateCallback =\r\n evaluateCallback ||\r\n ((r) => {\r\n return r === \"true\" ? true : false;\r\n });\r\n\r\n let result;\r\n const or = parenthesisContent.split(\"||\");\r\n\r\n for (const i in or) {\r\n if (Object.prototype.hasOwnProperty.call(or, i)) {\r\n let ori = AndOrNotEvaluator._SimplifyNegation(or[i].trim());\r\n const and = ori.split(\"&&\");\r\n\r\n if (and.length > 1) {\r\n for (let j = 0; j < and.length; ++j) {\r\n const andj = AndOrNotEvaluator._SimplifyNegation(and[j].trim());\r\n if (andj !== \"true\" && andj !== \"false\") {\r\n if (andj[0] === \"!\") {\r\n result = !evaluateCallback(andj.substring(1));\r\n } else {\r\n result = evaluateCallback(andj);\r\n }\r\n } else {\r\n result = andj === \"true\" ? true : false;\r\n }\r\n if (!result) {\r\n // no need to continue since 'false && ... && ...' will always return false\r\n ori = \"false\";\r\n break;\r\n }\r\n }\r\n }\r\n\r\n if (result || ori === \"true\") {\r\n // no need to continue since 'true || ... || ...' will always return true\r\n result = true;\r\n break;\r\n }\r\n\r\n // result equals false (or undefined)\r\n\r\n if (ori !== \"true\" && ori !== \"false\") {\r\n if (ori[0] === \"!\") {\r\n result = !evaluateCallback(ori.substring(1));\r\n } else {\r\n result = evaluateCallback(ori);\r\n }\r\n } else {\r\n result = ori === \"true\" ? true : false;\r\n }\r\n }\r\n }\r\n\r\n // the whole parenthesis scope is replaced by 'true' or 'false'\r\n return result ? \"true\" : \"false\";\r\n }\r\n\r\n private static _SimplifyNegation(booleanString: string): string {\r\n booleanString = booleanString.replace(/^[\\s!]+/, (r) => {\r\n // remove whitespaces\r\n r = r.replace(/[\\s]/g, () => \"\");\r\n return r.length % 2 ? \"!\" : \"\";\r\n });\r\n\r\n booleanString = booleanString.trim();\r\n\r\n if (booleanString === \"!true\") {\r\n booleanString = \"false\";\r\n } else if (booleanString === \"!false\") {\r\n booleanString = \"true\";\r\n }\r\n\r\n return booleanString;\r\n }\r\n}\r\n","import { AndOrNotEvaluator } from \"./andOrNotEvaluator\";\r\n\r\n/**\r\n * Class used to store custom tags\r\n */\r\nexport class Tags {\r\n /**\r\n * Adds support for tags on the given object\r\n * @param obj defines the object to use\r\n */\r\n public static EnableFor(obj: any): void {\r\n obj._tags = obj._tags || {};\r\n\r\n obj.hasTags = () => {\r\n return Tags.HasTags(obj);\r\n };\r\n\r\n obj.addTags = (tagsString: string) => {\r\n return Tags.AddTagsTo(obj, tagsString);\r\n };\r\n\r\n obj.removeTags = (tagsString: string) => {\r\n return Tags.RemoveTagsFrom(obj, tagsString);\r\n };\r\n\r\n obj.matchesTagsQuery = (tagsQuery: string) => {\r\n return Tags.MatchesQuery(obj, tagsQuery);\r\n };\r\n }\r\n\r\n /**\r\n * Removes tags support\r\n * @param obj defines the object to use\r\n */\r\n public static DisableFor(obj: any): void {\r\n delete obj._tags;\r\n delete obj.hasTags;\r\n delete obj.addTags;\r\n delete obj.removeTags;\r\n delete obj.matchesTagsQuery;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if the given object has tags\r\n * @param obj defines the object to use\r\n * @returns a boolean\r\n */\r\n public static HasTags(obj: any): boolean {\r\n if (!obj._tags) {\r\n return false;\r\n }\r\n\r\n const tags = obj._tags;\r\n for (const i in tags) {\r\n if (Object.prototype.hasOwnProperty.call(tags, i)) {\r\n return true;\r\n }\r\n }\r\n return false;\r\n }\r\n\r\n /**\r\n * Gets the tags available on a given object\r\n * @param obj defines the object to use\r\n * @param asString defines if the tags must be returned as a string instead of an array of strings\r\n * @returns the tags\r\n */\r\n public static GetTags(obj: any, asString: boolean = true): any {\r\n if (!obj._tags) {\r\n return null;\r\n }\r\n if (asString) {\r\n const tagsArray = [];\r\n for (const tag in obj._tags) {\r\n if (Object.prototype.hasOwnProperty.call(obj._tags, tag) && obj._tags[tag] === true) {\r\n tagsArray.push(tag);\r\n }\r\n }\r\n return tagsArray.join(\" \");\r\n } else {\r\n return obj._tags;\r\n }\r\n }\r\n\r\n /**\r\n * Adds tags to an object\r\n * @param obj defines the object to use\r\n * @param tagsString defines the tag string. The tags 'true' and 'false' are reserved and cannot be used as tags.\r\n * A tag cannot start with '||', '&&', and '!'. It cannot contain whitespaces\r\n */\r\n public static AddTagsTo(obj: any, tagsString: string): void {\r\n if (!tagsString) {\r\n return;\r\n }\r\n\r\n if (typeof tagsString !== \"string\") {\r\n return;\r\n }\r\n\r\n const tags = tagsString.split(\" \");\r\n tags.forEach(function (tag) {\r\n Tags._AddTagTo(obj, tag);\r\n });\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _AddTagTo(obj: any, tag: string): void {\r\n tag = tag.trim();\r\n\r\n if (tag === \"\" || tag === \"true\" || tag === \"false\") {\r\n return;\r\n }\r\n\r\n if (tag.match(/[\\s]/) || tag.match(/^([!]|([|]|[&]){2})/)) {\r\n return;\r\n }\r\n\r\n Tags.EnableFor(obj);\r\n obj._tags[tag] = true;\r\n }\r\n\r\n /**\r\n * Removes specific tags from a specific object\r\n * @param obj defines the object to use\r\n * @param tagsString defines the tags to remove\r\n */\r\n public static RemoveTagsFrom(obj: any, tagsString: string) {\r\n if (!Tags.HasTags(obj)) {\r\n return;\r\n }\r\n const tags = tagsString.split(\" \");\r\n for (const t in tags) {\r\n Tags._RemoveTagFrom(obj, tags[t]);\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _RemoveTagFrom(obj: any, tag: string): void {\r\n delete obj._tags[tag];\r\n }\r\n\r\n /**\r\n * Defines if tags hosted on an object match a given query\r\n * @param obj defines the object to use\r\n * @param tagsQuery defines the tag query\r\n * @returns a boolean\r\n */\r\n public static MatchesQuery(obj: any, tagsQuery: string): boolean {\r\n if (tagsQuery === undefined) {\r\n return true;\r\n }\r\n\r\n if (tagsQuery === \"\") {\r\n return Tags.HasTags(obj);\r\n }\r\n\r\n return AndOrNotEvaluator.Eval(tagsQuery, (r) => Tags.HasTags(obj) && obj._tags[r]);\r\n }\r\n}\r\n","/**\r\n * Scalar computation library\r\n */\r\nexport class Scalar {\r\n /**\r\n * Two pi constants convenient for computation.\r\n */\r\n public static TwoPi: number = Math.PI * 2;\r\n\r\n /**\r\n * Boolean : true if the absolute difference between a and b is lower than epsilon (default = 1.401298E-45)\r\n * @param a number\r\n * @param b number\r\n * @param epsilon (default = 1.401298E-45)\r\n * @returns true if the absolute difference between a and b is lower than epsilon (default = 1.401298E-45)\r\n */\r\n public static WithinEpsilon(a: number, b: number, epsilon: number = 1.401298e-45): boolean {\r\n return Math.abs(a - b) <= epsilon;\r\n }\r\n\r\n /**\r\n * Returns a string : the upper case translation of the number i to hexadecimal.\r\n * @param i number\r\n * @returns the upper case translation of the number i to hexadecimal.\r\n */\r\n public static ToHex(i: number): string {\r\n const str = i.toString(16);\r\n\r\n if (i <= 15) {\r\n return (\"0\" + str).toUpperCase();\r\n }\r\n\r\n return str.toUpperCase();\r\n }\r\n\r\n /**\r\n * Returns -1 if value is negative and +1 is value is positive.\r\n * @param value the value\r\n * @returns the value itself if it's equal to zero.\r\n */\r\n public static Sign(value: number): number {\r\n value = +value; // convert to a number\r\n\r\n if (value === 0 || isNaN(value)) {\r\n return value;\r\n }\r\n\r\n return value > 0 ? 1 : -1;\r\n }\r\n\r\n /**\r\n * Returns the value itself if it's between min and max.\r\n * Returns min if the value is lower than min.\r\n * Returns max if the value is greater than max.\r\n * @param value the value to clmap\r\n * @param min the min value to clamp to (default: 0)\r\n * @param max the max value to clamp to (default: 1)\r\n * @returns the clamped value\r\n */\r\n public static Clamp(value: number, min = 0, max = 1): number {\r\n return Math.min(max, Math.max(min, value));\r\n }\r\n\r\n /**\r\n * the log2 of value.\r\n * @param value the value to compute log2 of\r\n * @returns the log2 of value.\r\n */\r\n public static Log2(value: number): number {\r\n return Math.log(value) * Math.LOG2E;\r\n }\r\n\r\n /**\r\n * the floor part of a log2 value.\r\n * @param value the value to compute log2 of\r\n * @returns the log2 of value.\r\n */\r\n public static ILog2(value: number): number {\r\n if (Math.log2) {\r\n return Math.floor(Math.log2(value));\r\n }\r\n\r\n if (value < 0) {\r\n return NaN;\r\n } else if (value === 0) {\r\n return -Infinity;\r\n }\r\n\r\n let n = 0;\r\n if (value < 1) {\r\n while (value < 1) {\r\n n++;\r\n value = value * 2;\r\n }\r\n n = -n;\r\n } else if (value > 1) {\r\n while (value > 1) {\r\n n++;\r\n value = Math.floor(value / 2);\r\n }\r\n }\r\n\r\n return n;\r\n }\r\n\r\n /**\r\n * Loops the value, so that it is never larger than length and never smaller than 0.\r\n *\r\n * This is similar to the modulo operator but it works with floating point numbers.\r\n * For example, using 3.0 for t and 2.5 for length, the result would be 0.5.\r\n * With t = 5 and length = 2.5, the result would be 0.0.\r\n * Note, however, that the behaviour is not defined for negative numbers as it is for the modulo operator\r\n * @param value the value\r\n * @param length the length\r\n * @returns the looped value\r\n */\r\n public static Repeat(value: number, length: number): number {\r\n return value - Math.floor(value / length) * length;\r\n }\r\n\r\n /**\r\n * Normalize the value between 0.0 and 1.0 using min and max values\r\n * @param value value to normalize\r\n * @param min max to normalize between\r\n * @param max min to normalize between\r\n * @returns the normalized value\r\n */\r\n public static Normalize(value: number, min: number, max: number): number {\r\n return (value - min) / (max - min);\r\n }\r\n\r\n /**\r\n * Denormalize the value from 0.0 and 1.0 using min and max values\r\n * @param normalized value to denormalize\r\n * @param min max to denormalize between\r\n * @param max min to denormalize between\r\n * @returns the denormalized value\r\n */\r\n public static Denormalize(normalized: number, min: number, max: number): number {\r\n return normalized * (max - min) + min;\r\n }\r\n\r\n /**\r\n * Calculates the shortest difference between two given angles given in degrees.\r\n * @param current current angle in degrees\r\n * @param target target angle in degrees\r\n * @returns the delta\r\n */\r\n public static DeltaAngle(current: number, target: number): number {\r\n let num: number = Scalar.Repeat(target - current, 360.0);\r\n if (num > 180.0) {\r\n num -= 360.0;\r\n }\r\n return num;\r\n }\r\n\r\n /**\r\n * PingPongs the value t, so that it is never larger than length and never smaller than 0.\r\n * @param tx value\r\n * @param length length\r\n * @returns The returned value will move back and forth between 0 and length\r\n */\r\n public static PingPong(tx: number, length: number): number {\r\n const t: number = Scalar.Repeat(tx, length * 2.0);\r\n return length - Math.abs(t - length);\r\n }\r\n\r\n /**\r\n * Interpolates between min and max with smoothing at the limits.\r\n *\r\n * This function interpolates between min and max in a similar way to Lerp. However, the interpolation will gradually speed up\r\n * from the start and slow down toward the end. This is useful for creating natural-looking animation, fading and other transitions.\r\n * @param from from\r\n * @param to to\r\n * @param tx value\r\n * @returns the smooth stepped value\r\n */\r\n public static SmoothStep(from: number, to: number, tx: number): number {\r\n let t: number = Scalar.Clamp(tx);\r\n t = -2.0 * t * t * t + 3.0 * t * t;\r\n return to * t + from * (1.0 - t);\r\n }\r\n\r\n /**\r\n * Moves a value current towards target.\r\n *\r\n * This is essentially the same as Mathf.Lerp but instead the function will ensure that the speed never exceeds maxDelta.\r\n * Negative values of maxDelta pushes the value away from target.\r\n * @param current current value\r\n * @param target target value\r\n * @param maxDelta max distance to move\r\n * @returns resulting value\r\n */\r\n public static MoveTowards(current: number, target: number, maxDelta: number): number {\r\n let result: number = 0;\r\n if (Math.abs(target - current) <= maxDelta) {\r\n result = target;\r\n } else {\r\n result = current + Scalar.Sign(target - current) * maxDelta;\r\n }\r\n return result;\r\n }\r\n\r\n /**\r\n * Same as MoveTowards but makes sure the values interpolate correctly when they wrap around 360 degrees.\r\n *\r\n * Variables current and target are assumed to be in degrees. For optimization reasons, negative values of maxDelta\r\n * are not supported and may cause oscillation. To push current away from a target angle, add 180 to that angle instead.\r\n * @param current current value\r\n * @param target target value\r\n * @param maxDelta max distance to move\r\n * @returns resulting angle\r\n */\r\n public static MoveTowardsAngle(current: number, target: number, maxDelta: number): number {\r\n const num: number = Scalar.DeltaAngle(current, target);\r\n let result: number = 0;\r\n if (-maxDelta < num && num < maxDelta) {\r\n result = target;\r\n } else {\r\n target = current + num;\r\n result = Scalar.MoveTowards(current, target, maxDelta);\r\n }\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new scalar with values linearly interpolated of \"amount\" between the start scalar and the end scalar.\r\n * @param start start value\r\n * @param end target value\r\n * @param amount amount to lerp between\r\n * @returns the lerped value\r\n */\r\n public static Lerp(start: number, end: number, amount: number): number {\r\n return start + (end - start) * amount;\r\n }\r\n\r\n /**\r\n * Same as Lerp but makes sure the values interpolate correctly when they wrap around 360 degrees.\r\n * The parameter t is clamped to the range [0, 1]. Variables a and b are assumed to be in degrees.\r\n * @param start start value\r\n * @param end target value\r\n * @param amount amount to lerp between\r\n * @returns the lerped value\r\n */\r\n public static LerpAngle(start: number, end: number, amount: number): number {\r\n let num: number = Scalar.Repeat(end - start, 360.0);\r\n if (num > 180.0) {\r\n num -= 360.0;\r\n }\r\n return start + num * Scalar.Clamp(amount);\r\n }\r\n\r\n /**\r\n * Calculates the linear parameter t that produces the interpolant value within the range [a, b].\r\n * @param a start value\r\n * @param b target value\r\n * @param value value between a and b\r\n * @returns the inverseLerp value\r\n */\r\n public static InverseLerp(a: number, b: number, value: number): number {\r\n let result: number = 0;\r\n if (a != b) {\r\n result = Scalar.Clamp((value - a) / (b - a));\r\n } else {\r\n result = 0.0;\r\n }\r\n return result;\r\n }\r\n\r\n /**\r\n * Returns a new scalar located for \"amount\" (float) on the Hermite spline defined by the scalars \"value1\", \"value3\", \"tangent1\", \"tangent2\".\r\n * @see http://mathworld.wolfram.com/HermitePolynomial.html\r\n * @param value1 defines the first control point\r\n * @param tangent1 defines the first tangent\r\n * @param value2 defines the second control point\r\n * @param tangent2 defines the second tangent\r\n * @param amount defines the amount on the interpolation spline (between 0 and 1)\r\n * @returns hermite result\r\n */\r\n public static Hermite(value1: number, tangent1: number, value2: number, tangent2: number, amount: number): number {\r\n const squared = amount * amount;\r\n const cubed = amount * squared;\r\n const part1 = 2.0 * cubed - 3.0 * squared + 1.0;\r\n const part2 = -2.0 * cubed + 3.0 * squared;\r\n const part3 = cubed - 2.0 * squared + amount;\r\n const part4 = cubed - squared;\r\n\r\n return value1 * part1 + value2 * part2 + tangent1 * part3 + tangent2 * part4;\r\n }\r\n\r\n /**\r\n * Returns a new scalar which is the 1st derivative of the Hermite spline defined by the scalars \"value1\", \"value2\", \"tangent1\", \"tangent2\".\r\n * @param value1 defines the first control point\r\n * @param tangent1 defines the first tangent\r\n * @param value2 defines the second control point\r\n * @param tangent2 defines the second tangent\r\n * @param time define where the derivative must be done\r\n * @returns 1st derivative\r\n */\r\n public static Hermite1stDerivative(value1: number, tangent1: number, value2: number, tangent2: number, time: number): number {\r\n const t2 = time * time;\r\n return (t2 - time) * 6 * value1 + (3 * t2 - 4 * time + 1) * tangent1 + (-t2 + time) * 6 * value2 + (3 * t2 - 2 * time) * tangent2;\r\n }\r\n\r\n /**\r\n * Returns a random float number between and min and max values\r\n * @param min min value of random\r\n * @param max max value of random\r\n * @returns random value\r\n */\r\n public static RandomRange(min: number, max: number): number {\r\n if (min === max) {\r\n return min;\r\n }\r\n return Math.random() * (max - min) + min;\r\n }\r\n\r\n /**\r\n * This function returns percentage of a number in a given range.\r\n *\r\n * RangeToPercent(40,20,60) will return 0.5 (50%)\r\n * RangeToPercent(34,0,100) will return 0.34 (34%)\r\n * @param number to convert to percentage\r\n * @param min min range\r\n * @param max max range\r\n * @returns the percentage\r\n */\r\n public static RangeToPercent(number: number, min: number, max: number): number {\r\n return (number - min) / (max - min);\r\n }\r\n\r\n /**\r\n * This function returns number that corresponds to the percentage in a given range.\r\n *\r\n * PercentToRange(0.34,0,100) will return 34.\r\n * @param percent to convert to number\r\n * @param min min range\r\n * @param max max range\r\n * @returns the number\r\n */\r\n public static PercentToRange(percent: number, min: number, max: number): number {\r\n return (max - min) * percent + min;\r\n }\r\n\r\n /**\r\n * Returns the angle converted to equivalent value between -Math.PI and Math.PI radians.\r\n * @param angle The angle to normalize in radian.\r\n * @returns The converted angle.\r\n */\r\n public static NormalizeRadians(angle: number): number {\r\n // More precise but slower version kept for reference.\r\n // angle = angle % Tools.TwoPi;\r\n // angle = (angle + Tools.TwoPi) % Tools.TwoPi;\r\n\r\n //if (angle > Math.PI) {\r\n //\tangle -= Tools.TwoPi;\r\n //}\r\n\r\n angle -= Scalar.TwoPi * Math.floor((angle + Math.PI) / Scalar.TwoPi);\r\n\r\n return angle;\r\n }\r\n\r\n /**\r\n * Returns the highest common factor of two integers.\r\n * @param a first parameter\r\n * @param b second parameter\r\n * @returns HCF of a and b\r\n */\r\n public static HCF(a: number, b: number): number {\r\n const r: number = a % b;\r\n if (r === 0) {\r\n return b;\r\n }\r\n return Scalar.HCF(b, r);\r\n }\r\n}\r\n","/**\r\n * Constant used to convert a value to gamma space\r\n * @ignorenaming\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport const ToGammaSpace = 1 / 2.2;\r\n\r\n/**\r\n * Constant used to convert a value to linear space\r\n * @ignorenaming\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport const ToLinearSpace = 2.2;\r\n\r\n/**\r\n * Constant Golden Ratio value in Babylon.js\r\n * @ignorenaming\r\n */\r\nexport const PHI = (1 + Math.sqrt(5)) / 2;\r\n\r\n/**\r\n * Constant used to define the minimal number value in Babylon.js\r\n * @ignorenaming\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nconst Epsilon = 0.001;\r\nexport { Epsilon };\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\n\r\nimport type { Nullable } from \"../types\";\r\n\r\n/** @internal */\r\ninterface TupleTypes {\r\n 2: [T, T];\r\n 3: [T, T, T];\r\n 4: [T, T, T, T];\r\n 5: [T, T, T, T, T];\r\n 6: [T, T, T, T, T, T];\r\n 7: [T, T, T, T, T, T, T];\r\n 8: [T, T, T, T, T, T, T, T];\r\n 9: [T, T, T, T, T, T, T, T, T];\r\n 10: [T, T, T, T, T, T, T, T, T, T];\r\n 11: [T, T, T, T, T, T, T, T, T, T, T];\r\n 12: [T, T, T, T, T, T, T, T, T, T, T, T];\r\n 13: [T, T, T, T, T, T, T, T, T, T, T, T, T];\r\n 14: [T, T, T, T, T, T, T, T, T, T, T, T, T, T];\r\n 15: [T, T, T, T, T, T, T, T, T, T, T, T, T, T, T];\r\n}\r\n\r\n/**\r\n * Class containing a set of static utilities functions for arrays.\r\n */\r\nexport class ArrayTools {\r\n /**\r\n * Returns an array of the given size filled with elements built from the given constructor and the parameters.\r\n * @param size the number of element to construct and put in the array.\r\n * @param itemBuilder a callback responsible for creating new instance of item. Called once per array entry.\r\n * @returns a new array filled with new objects.\r\n */\r\n public static BuildArray(size: number, itemBuilder: () => T): Array {\r\n const a: T[] = [];\r\n for (let i = 0; i < size; ++i) {\r\n a.push(itemBuilder());\r\n }\r\n return a;\r\n }\r\n\r\n /**\r\n * Returns a tuple of the given size filled with elements built from the given constructor and the parameters.\r\n * @param size he number of element to construct and put in the tuple.\r\n * @param itemBuilder a callback responsible for creating new instance of item. Called once per tuple entry.\r\n * @returns a new tuple filled with new objects.\r\n */\r\n public static BuildTuple>(size: N, itemBuilder: () => T): TupleTypes[N] {\r\n return ArrayTools.BuildArray(size, itemBuilder) as any;\r\n }\r\n}\r\n\r\n/**\r\n * Defines the callback type used when an observed array function is triggered.\r\n * @internal\r\n */\r\nexport type _ObserveCallback = (functionName: string, previousLength: number) => void;\r\n\r\n/**\r\n * Observes a function and calls the given callback when it is called.\r\n * @param object Defines the object the function to observe belongs to.\r\n * @param functionName Defines the name of the function to observe.\r\n * @param callback Defines the callback to call when the function is called.\r\n * @returns A function to call to stop observing\r\n */\r\nfunction _observeArrayfunction(object: { [key: string]: any }, functionName: string, callback: _ObserveCallback): Nullable<() => void> {\r\n // Finds the function to observe\r\n const oldFunction = object[functionName];\r\n if (typeof oldFunction !== \"function\") {\r\n return null;\r\n }\r\n\r\n // Creates a new function that calls the callback and the old function\r\n const newFunction = function () {\r\n const previousLength = object.length;\r\n const returnValue = newFunction.previous.apply(object, arguments);\r\n callback(functionName, previousLength);\r\n return returnValue;\r\n } as any;\r\n\r\n // Doublishly links the new function and the old function\r\n oldFunction.next = newFunction;\r\n newFunction.previous = oldFunction;\r\n\r\n // Replaces the old function with the new function\r\n object[functionName] = newFunction;\r\n\r\n // Returns a function to disable the hook\r\n return () => {\r\n // Only unhook if the function is still hooked\r\n const previous = newFunction.previous;\r\n if (!previous) {\r\n return;\r\n }\r\n\r\n // Finds the ref to the next function in the chain\r\n const next = newFunction.next;\r\n\r\n // If in the middle of the chain, link the previous and next functions\r\n if (next) {\r\n previous.next = next;\r\n next.previous = previous;\r\n }\r\n // If at the end of the chain, remove the reference to the previous function\r\n // and restore the previous function\r\n else {\r\n previous.next = undefined;\r\n object[functionName] = previous;\r\n }\r\n\r\n // Lose reference to the previous and next functions\r\n newFunction.next = undefined;\r\n newFunction.previous = undefined;\r\n };\r\n}\r\n\r\n/**\r\n * Defines the list of functions to proxy when observing an array.\r\n * The scope is currently reduced to the common functions used in the render target render list and the scene cameras.\r\n */\r\nconst observedArrayFunctions = [\"push\", \"splice\", \"pop\", \"shift\", \"unshift\"];\r\n\r\n/**\r\n * Observes an array and notifies the given observer when the array is modified.\r\n * @param array Defines the array to observe\r\n * @param callback Defines the function to call when the array is modified (in the limit of the observed array functions)\r\n * @returns A function to call to stop observing the array\r\n * @internal\r\n */\r\nexport function _ObserveArray(array: T[], callback: _ObserveCallback) {\r\n // Observes all the required array functions and stores the unhook functions\r\n const unObserveFunctions = observedArrayFunctions.map((name) => {\r\n return _observeArrayfunction(array, name, callback);\r\n });\r\n\r\n // Returns a function that unhook all the observed functions\r\n return () => {\r\n unObserveFunctions.forEach((unObserveFunction) => {\r\n unObserveFunction?.();\r\n });\r\n };\r\n}\r\n","/** @internal */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nconst _RegisteredTypes: { [key: string]: Object } = {};\r\n\r\n/**\r\n * @internal\r\n */\r\nexport function RegisterClass(className: string, type: Object) {\r\n _RegisteredTypes[className] = type;\r\n}\r\n\r\n/**\r\n * @internal\r\n */\r\nexport function GetClass(fqdn: string): any {\r\n return _RegisteredTypes[fqdn];\r\n}\r\n","/** @internal */\r\nexport class PerformanceConfigurator {\r\n /** @internal */\r\n public static MatrixUse64Bits = false;\r\n /** @internal */\r\n public static MatrixTrackPrecisionChange = true;\r\n /** @internal */\r\n public static MatrixCurrentType: any = Float32Array;\r\n /** @internal */\r\n public static MatrixTrackedMatrices: Array | null = [];\r\n\r\n /**\r\n * @internal\r\n */\r\n public static SetMatrixPrecision(use64bits: boolean) {\r\n PerformanceConfigurator.MatrixTrackPrecisionChange = false;\r\n\r\n if (use64bits && !PerformanceConfigurator.MatrixUse64Bits) {\r\n if (PerformanceConfigurator.MatrixTrackedMatrices) {\r\n for (let m = 0; m < PerformanceConfigurator.MatrixTrackedMatrices.length; ++m) {\r\n const matrix = PerformanceConfigurator.MatrixTrackedMatrices[m];\r\n const values = matrix._m;\r\n\r\n matrix._m = new Array(16);\r\n\r\n for (let i = 0; i < 16; ++i) {\r\n matrix._m[i] = values[i];\r\n }\r\n }\r\n }\r\n }\r\n\r\n PerformanceConfigurator.MatrixUse64Bits = use64bits;\r\n PerformanceConfigurator.MatrixCurrentType = PerformanceConfigurator.MatrixUse64Bits ? Array : Float32Array;\r\n PerformanceConfigurator.MatrixTrackedMatrices = null; // reclaim some memory, as we don't need _TrackedMatrices anymore\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\n\r\n/**\r\n * A class serves as a medium between the observable and its observers\r\n */\r\nexport class EventState {\r\n /**\r\n * Create a new EventState\r\n * @param mask defines the mask associated with this state\r\n * @param skipNextObservers defines a flag which will instruct the observable to skip following observers when set to true\r\n * @param target defines the original target of the state\r\n * @param currentTarget defines the current target of the state\r\n */\r\n constructor(mask: number, skipNextObservers = false, target?: any, currentTarget?: any) {\r\n this.initialize(mask, skipNextObservers, target, currentTarget);\r\n }\r\n\r\n /**\r\n * Initialize the current event state\r\n * @param mask defines the mask associated with this state\r\n * @param skipNextObservers defines a flag which will instruct the observable to skip following observers when set to true\r\n * @param target defines the original target of the state\r\n * @param currentTarget defines the current target of the state\r\n * @returns the current event state\r\n */\r\n public initialize(mask: number, skipNextObservers = false, target?: any, currentTarget?: any): EventState {\r\n this.mask = mask;\r\n this.skipNextObservers = skipNextObservers;\r\n this.target = target;\r\n this.currentTarget = currentTarget;\r\n return this;\r\n }\r\n\r\n /**\r\n * An Observer can set this property to true to prevent subsequent observers of being notified\r\n */\r\n public skipNextObservers: boolean;\r\n\r\n /**\r\n * Get the mask value that were used to trigger the event corresponding to this EventState object\r\n */\r\n public mask: number;\r\n\r\n /**\r\n * The object that originally notified the event\r\n */\r\n public target?: any;\r\n\r\n /**\r\n * The current object in the bubbling phase\r\n */\r\n public currentTarget?: any;\r\n\r\n /**\r\n * This will be populated with the return value of the last function that was executed.\r\n * If it is the first function in the callback chain it will be the event data.\r\n */\r\n public lastReturnValue?: any;\r\n\r\n /**\r\n * User defined information that will be sent to observers\r\n */\r\n public userInfo?: any;\r\n}\r\n\r\n/**\r\n * Represent an Observer registered to a given Observable object.\r\n */\r\nexport class Observer {\r\n /** @internal */\r\n public _willBeUnregistered = false;\r\n /**\r\n * Gets or sets a property defining that the observer as to be unregistered after the next notification\r\n */\r\n public unregisterOnNextCall = false;\r\n\r\n /**\r\n * this function can be used to remove the observer from the observable.\r\n * It will be set by the observable that the observer belongs to.\r\n * @internal\r\n */\r\n public _remove: Nullable<() => void> = null;\r\n\r\n /**\r\n * Creates a new observer\r\n * @param callback defines the callback to call when the observer is notified\r\n * @param mask defines the mask of the observer (used to filter notifications)\r\n * @param scope defines the current scope used to restore the JS context\r\n */\r\n constructor(\r\n /**\r\n * Defines the callback to call when the observer is notified\r\n */\r\n public callback: (eventData: T, eventState: EventState) => void,\r\n /**\r\n * Defines the mask of the observer (used to filter notifications)\r\n */\r\n public mask: number,\r\n /**\r\n * Defines the current scope used to restore the JS context\r\n */\r\n public scope: any = null\r\n ) {}\r\n\r\n /**\r\n * Remove the observer from its observable\r\n * This can be used instead of using the observable's remove function.\r\n */\r\n public remove() {\r\n if (this._remove) {\r\n this._remove();\r\n }\r\n }\r\n}\r\n\r\n/**\r\n * The Observable class is a simple implementation of the Observable pattern.\r\n *\r\n * There's one slight particularity though: a given Observable can notify its observer using a particular mask value, only the Observers registered with this mask value will be notified.\r\n * This enable a more fine grained execution without having to rely on multiple different Observable objects.\r\n * For instance you may have a given Observable that have four different types of notifications: Move (mask = 0x01), Stop (mask = 0x02), Turn Right (mask = 0X04), Turn Left (mask = 0X08).\r\n * A given observer can register itself with only Move and Stop (mask = 0x03), then it will only be notified when one of these two occurs and will never be for Turn Left/Right.\r\n */\r\nexport class Observable {\r\n private _observers = new Array>();\r\n private _numObserversMarkedAsDeleted = 0;\r\n private _hasNotified = false;\r\n private _lastNotifiedValue?: T;\r\n\r\n /**\r\n * @internal\r\n */\r\n public _eventState: EventState;\r\n\r\n private _onObserverAdded: Nullable<(observer: Observer) => void>;\r\n\r\n /**\r\n * Create an observable from a Promise.\r\n * @param promise a promise to observe for fulfillment.\r\n * @param onErrorObservable an observable to notify if a promise was rejected.\r\n * @returns the new Observable\r\n */\r\n public static FromPromise(promise: Promise, onErrorObservable?: Observable): Observable {\r\n const observable = new Observable();\r\n\r\n promise\r\n .then((ret: T) => {\r\n observable.notifyObservers(ret);\r\n })\r\n .catch((err) => {\r\n if (onErrorObservable) {\r\n onErrorObservable.notifyObservers(err as E);\r\n } else {\r\n throw err;\r\n }\r\n });\r\n\r\n return observable;\r\n }\r\n\r\n /**\r\n * Gets the list of observers\r\n * Note that observers that were recently deleted may still be present in the list because they are only really deleted on the next javascript tick!\r\n */\r\n public get observers(): Array> {\r\n return this._observers;\r\n }\r\n\r\n /**\r\n * Creates a new observable\r\n * @param onObserverAdded defines a callback to call when a new observer is added\r\n * @param notifyIfTriggered If set to true the observable will notify when an observer was added if the observable was already triggered.\r\n */\r\n constructor(\r\n onObserverAdded?: (observer: Observer) => void,\r\n /**\r\n * If set to true the observable will notify when an observer was added if the observable was already triggered.\r\n * This is helpful to single-state observables like the scene onReady or the dispose observable.\r\n */\r\n public notifyIfTriggered = false\r\n ) {\r\n this._eventState = new EventState(0);\r\n\r\n if (onObserverAdded) {\r\n this._onObserverAdded = onObserverAdded;\r\n }\r\n }\r\n\r\n /**\r\n * Create a new Observer with the specified callback\r\n * @param callback the callback that will be executed for that Observer\r\n * @param mask the mask used to filter observers\r\n * @param insertFirst if true the callback will be inserted at the first position, hence executed before the others ones. If false (default behavior) the callback will be inserted at the last position, executed after all the others already present.\r\n * @param scope optional scope for the callback to be called from\r\n * @param unregisterOnFirstCall defines if the observer as to be unregistered after the next notification\r\n * @returns the new observer created for the callback\r\n */\r\n public add(\r\n callback: (eventData: T, eventState: EventState) => void,\r\n mask: number = -1,\r\n insertFirst = false,\r\n scope: any = null,\r\n unregisterOnFirstCall = false\r\n ): Nullable> {\r\n if (!callback) {\r\n return null;\r\n }\r\n\r\n const observer = new Observer(callback, mask, scope);\r\n observer.unregisterOnNextCall = unregisterOnFirstCall;\r\n\r\n if (insertFirst) {\r\n this._observers.unshift(observer);\r\n } else {\r\n this._observers.push(observer);\r\n }\r\n\r\n if (this._onObserverAdded) {\r\n this._onObserverAdded(observer);\r\n }\r\n\r\n // If the observable was already triggered and the observable is set to notify if triggered, notify the new observer\r\n if (this._hasNotified && this.notifyIfTriggered) {\r\n if (this._lastNotifiedValue !== undefined) {\r\n this.notifyObserver(observer, this._lastNotifiedValue);\r\n }\r\n }\r\n // attach the remove function to the observer\r\n observer._remove = () => {\r\n this.remove(observer);\r\n };\r\n\r\n return observer;\r\n }\r\n\r\n /**\r\n * Create a new Observer with the specified callback and unregisters after the next notification\r\n * @param callback the callback that will be executed for that Observer\r\n * @returns the new observer created for the callback\r\n */\r\n public addOnce(callback: (eventData: T, eventState: EventState) => void): Nullable> {\r\n return this.add(callback, undefined, undefined, undefined, true);\r\n }\r\n\r\n /**\r\n * Remove an Observer from the Observable object\r\n * @param observer the instance of the Observer to remove\r\n * @returns false if it doesn't belong to this Observable\r\n */\r\n public remove(observer: Nullable>): boolean {\r\n if (!observer) {\r\n return false;\r\n }\r\n\r\n observer._remove = null;\r\n const index = this._observers.indexOf(observer);\r\n\r\n if (index !== -1) {\r\n this._deferUnregister(observer);\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Remove a callback from the Observable object\r\n * @param callback the callback to remove\r\n * @param scope optional scope. If used only the callbacks with this scope will be removed\r\n * @returns false if it doesn't belong to this Observable\r\n */\r\n public removeCallback(callback: (eventData: T, eventState: EventState) => void, scope?: any): boolean {\r\n for (let index = 0; index < this._observers.length; index++) {\r\n const observer = this._observers[index];\r\n if (observer._willBeUnregistered) {\r\n continue;\r\n }\r\n if (observer.callback === callback && (!scope || scope === observer.scope)) {\r\n this._deferUnregister(observer);\r\n return true;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _deferUnregister(observer: Observer): void {\r\n if (observer._willBeUnregistered) {\r\n return;\r\n }\r\n this._numObserversMarkedAsDeleted++;\r\n observer.unregisterOnNextCall = false;\r\n observer._willBeUnregistered = true;\r\n setTimeout(() => {\r\n this._remove(observer);\r\n }, 0);\r\n }\r\n\r\n // This should only be called when not iterating over _observers to avoid callback skipping.\r\n // Removes an observer from the _observer Array.\r\n private _remove(observer: Nullable>, updateCounter = true): boolean {\r\n if (!observer) {\r\n return false;\r\n }\r\n\r\n const index = this._observers.indexOf(observer);\r\n\r\n if (index !== -1) {\r\n if (updateCounter) {\r\n this._numObserversMarkedAsDeleted--;\r\n }\r\n this._observers.splice(index, 1);\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Moves the observable to the top of the observer list making it get called first when notified\r\n * @param observer the observer to move\r\n */\r\n public makeObserverTopPriority(observer: Observer) {\r\n this._remove(observer, false);\r\n this._observers.unshift(observer);\r\n }\r\n\r\n /**\r\n * Moves the observable to the bottom of the observer list making it get called last when notified\r\n * @param observer the observer to move\r\n */\r\n public makeObserverBottomPriority(observer: Observer) {\r\n this._remove(observer, false);\r\n this._observers.push(observer);\r\n }\r\n\r\n /**\r\n * Notify all Observers by calling their respective callback with the given data\r\n * Will return true if all observers were executed, false if an observer set skipNextObservers to true, then prevent the subsequent ones to execute\r\n * @param eventData defines the data to send to all observers\r\n * @param mask defines the mask of the current notification (observers with incompatible mask (ie mask & observer.mask === 0) will not be notified)\r\n * @param target defines the original target of the state\r\n * @param currentTarget defines the current target of the state\r\n * @param userInfo defines any user info to send to observers\r\n * @returns false if the complete observer chain was not processed (because one observer set the skipNextObservers to true)\r\n */\r\n public notifyObservers(eventData: T, mask: number = -1, target?: any, currentTarget?: any, userInfo?: any): boolean {\r\n // this prevents potential memory leaks - if an object is disposed but the observable doesn't get cleared.\r\n if (this.notifyIfTriggered) {\r\n this._hasNotified = true;\r\n this._lastNotifiedValue = eventData;\r\n }\r\n if (!this._observers.length) {\r\n return true;\r\n }\r\n\r\n const state = this._eventState;\r\n state.mask = mask;\r\n state.target = target;\r\n state.currentTarget = currentTarget;\r\n state.skipNextObservers = false;\r\n state.lastReturnValue = eventData;\r\n state.userInfo = userInfo;\r\n\r\n for (const obs of this._observers) {\r\n if (obs._willBeUnregistered) {\r\n continue;\r\n }\r\n\r\n if (obs.mask & mask) {\r\n if (obs.unregisterOnNextCall) {\r\n this._deferUnregister(obs);\r\n }\r\n\r\n if (obs.scope) {\r\n state.lastReturnValue = obs.callback.apply(obs.scope, [eventData, state]);\r\n } else {\r\n state.lastReturnValue = obs.callback(eventData, state);\r\n }\r\n }\r\n if (state.skipNextObservers) {\r\n return false;\r\n }\r\n }\r\n return true;\r\n }\r\n\r\n /**\r\n * Notify a specific observer\r\n * @param observer defines the observer to notify\r\n * @param eventData defines the data to be sent to each callback\r\n * @param mask is used to filter observers defaults to -1\r\n */\r\n public notifyObserver(observer: Observer, eventData: T, mask: number = -1): void {\r\n // this prevents potential memory leaks - if an object is disposed but the observable doesn't get cleared.\r\n if (this.notifyIfTriggered) {\r\n this._hasNotified = true;\r\n this._lastNotifiedValue = eventData;\r\n }\r\n if (observer._willBeUnregistered) {\r\n return;\r\n }\r\n\r\n const state = this._eventState;\r\n state.mask = mask;\r\n state.skipNextObservers = false;\r\n\r\n if (observer.unregisterOnNextCall) {\r\n this._deferUnregister(observer);\r\n }\r\n\r\n observer.callback(eventData, state);\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if the observable has at least one observer\r\n * @returns true is the Observable has at least one Observer registered\r\n */\r\n public hasObservers(): boolean {\r\n return this._observers.length - this._numObserversMarkedAsDeleted > 0;\r\n }\r\n\r\n /**\r\n * Clear the list of observers\r\n */\r\n public clear(): void {\r\n while (this._observers.length) {\r\n const o = this._observers.pop();\r\n if (o) {\r\n o._remove = null;\r\n }\r\n }\r\n this._onObserverAdded = null;\r\n this._numObserversMarkedAsDeleted = 0;\r\n this.cleanLastNotifiedState();\r\n }\r\n\r\n /**\r\n * Clean the last notified state - both the internal last value and the has-notified flag\r\n */\r\n public cleanLastNotifiedState(): void {\r\n this._hasNotified = false;\r\n this._lastNotifiedValue = undefined;\r\n }\r\n\r\n /**\r\n * Clone the current observable\r\n * @returns a new observable\r\n */\r\n public clone(): Observable {\r\n const result = new Observable();\r\n\r\n result._observers = this._observers.slice(0);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Does this observable handles observer registered with a given mask\r\n * @param mask defines the mask to be tested\r\n * @returns whether or not one observer registered with the given mask is handled\r\n **/\r\n public hasSpecificMask(mask: number = -1): boolean {\r\n for (const obs of this._observers) {\r\n if (obs.mask & mask || obs.mask === mask) {\r\n return true;\r\n }\r\n }\r\n return false;\r\n }\r\n}\r\n","import { Observable } from \"../Misc/observable\";\r\nimport type { Nullable } from \"../types\";\r\n\r\nimport type { Engine } from \"./engine\";\r\nimport type { Scene } from \"../scene\";\r\n\r\n/**\r\n * The engine store class is responsible to hold all the instances of Engine and Scene created\r\n * during the life time of the application.\r\n */\r\nexport class EngineStore {\r\n /** Gets the list of created engines */\r\n public static Instances = new Array();\r\n\r\n /**\r\n * Notifies when an engine was disposed.\r\n * Mainly used for static/cache cleanup\r\n */\r\n public static OnEnginesDisposedObservable = new Observable();\r\n\r\n /** @internal */\r\n public static _LastCreatedScene: Nullable = null;\r\n\r\n /**\r\n * Gets the latest created engine\r\n */\r\n public static get LastCreatedEngine(): Nullable {\r\n if (this.Instances.length === 0) {\r\n return null;\r\n }\r\n\r\n return this.Instances[this.Instances.length - 1];\r\n }\r\n\r\n /**\r\n * Gets the latest created scene\r\n */\r\n public static get LastCreatedScene(): Nullable {\r\n return this._LastCreatedScene;\r\n }\r\n\r\n /**\r\n * Gets or sets a global variable indicating if fallback texture must be used when a texture cannot be loaded\r\n * @ignorenaming\r\n */\r\n public static UseFallbackTexture = true;\r\n\r\n /**\r\n * Texture content used if a texture cannot loaded\r\n * @ignorenaming\r\n */\r\n public static FallbackTexture = \"\";\r\n}\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport { Scalar } from \"./math.scalar\";\r\nimport { Epsilon } from \"./math.constants\";\r\nimport type { Viewport } from \"./math.viewport\";\r\nimport type { DeepImmutable, Nullable, FloatArray, float } from \"../types\";\r\nimport { ArrayTools } from \"../Misc/arrayTools\";\r\nimport type { IPlaneLike } from \"./math.like\";\r\nimport { RegisterClass } from \"../Misc/typeStore\";\r\nimport type { Plane } from \"./math.plane\";\r\nimport { PerformanceConfigurator } from \"../Engines/performanceConfigurator\";\r\nimport { EngineStore } from \"../Engines/engineStore\";\r\nimport type { TransformNode } from \"../Meshes/transformNode\";\r\n\r\nexport type Vector2Constructor = new (...args: ConstructorParameters) => T;\r\nexport type Vector3Constructor = new (...args: ConstructorParameters) => T;\r\nexport type Vector4Constructor = new (...args: ConstructorParameters) => T;\r\nexport type QuaternionConstructor = new (...args: ConstructorParameters) => T;\r\nexport type MatrixConstructor = new () => T;\r\n\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nconst _ExtractAsInt = (value: number) => {\r\n return parseInt(value.toString().replace(/\\W/g, \"\"));\r\n};\r\n\r\n/**\r\n * Class representing a vector containing 2 coordinates\r\n * Example Playground - Overview - https://playground.babylonjs.com/#QYBWV4#9\r\n */\r\nexport class Vector2 {\r\n private static _ZeroReadOnly = Vector2.Zero() as DeepImmutable;\r\n\r\n /**\r\n * Creates a new Vector2 from the given x and y coordinates\r\n * @param x defines the first coordinate\r\n * @param y defines the second coordinate\r\n */\r\n constructor(\r\n /** defines the first coordinate */\r\n public x: number = 0,\r\n /** defines the second coordinate */\r\n public y: number = 0\r\n ) {}\r\n\r\n /**\r\n * Gets a string with the Vector2 coordinates\r\n * @returns a string with the Vector2 coordinates\r\n */\r\n public toString(): string {\r\n return `{X: ${this.x} Y: ${this.y}}`;\r\n }\r\n\r\n /**\r\n * Gets class name\r\n * @returns the string \"Vector2\"\r\n */\r\n public getClassName(): string {\r\n return \"Vector2\";\r\n }\r\n\r\n /**\r\n * Gets current vector hash code\r\n * @returns the Vector2 hash code as a number\r\n */\r\n public getHashCode(): number {\r\n const x = _ExtractAsInt(this.x);\r\n const y = _ExtractAsInt(this.y);\r\n let hash = x;\r\n hash = (hash * 397) ^ y;\r\n return hash;\r\n }\r\n\r\n // Operators\r\n\r\n /**\r\n * Sets the Vector2 coordinates in the given array or Float32Array from the given index.\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#15\r\n * @param array defines the source array\r\n * @param index defines the offset in source array\r\n * @returns the current Vector2\r\n */\r\n public toArray(array: FloatArray, index: number = 0): this {\r\n array[index] = this.x;\r\n array[index + 1] = this.y;\r\n return this;\r\n }\r\n\r\n /**\r\n * Update the current vector from an array\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#39\r\n * @param array defines the destination array\r\n * @param index defines the offset in the destination array\r\n * @returns the current Vector2\r\n */\r\n public fromArray(array: FloatArray, index: number = 0): this {\r\n Vector2.FromArrayToRef(array, index, this);\r\n return this;\r\n }\r\n\r\n /**\r\n * Copy the current vector to an array\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#40\r\n * @returns a new array with 2 elements: the Vector2 coordinates.\r\n */\r\n public asArray(): number[] {\r\n const result = new Array();\r\n this.toArray(result, 0);\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the Vector2 coordinates with the given Vector2 coordinates\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#24\r\n * @param source defines the source Vector2\r\n * @returns the current updated Vector2\r\n */\r\n public copyFrom(source: DeepImmutable): this {\r\n this.x = source.x;\r\n this.y = source.y;\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets the Vector2 coordinates with the given floats\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#25\r\n * @param x defines the first coordinate\r\n * @param y defines the second coordinate\r\n * @returns the current updated Vector2\r\n */\r\n public copyFromFloats(x: number, y: number): this {\r\n this.x = x;\r\n this.y = y;\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets the Vector2 coordinates with the given floats\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#62\r\n * @param x defines the first coordinate\r\n * @param y defines the second coordinate\r\n * @returns the current updated Vector2\r\n */\r\n public set(x: number, y: number): this {\r\n return this.copyFromFloats(x, y);\r\n }\r\n /**\r\n * Add another vector with the current one\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#11\r\n * @param otherVector defines the other vector\r\n * @returns a new Vector2 set with the addition of the current Vector2 and the given one coordinates\r\n */\r\n public add(otherVector: DeepImmutable): this {\r\n return new (this.constructor as Vector2Constructor)(this.x + otherVector.x, this.y + otherVector.y);\r\n }\r\n\r\n /**\r\n * Sets the \"result\" coordinates with the addition of the current Vector2 and the given one coordinates\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#12\r\n * @param otherVector defines the other vector\r\n * @param result defines the target vector\r\n * @returns result input\r\n */\r\n public addToRef(otherVector: DeepImmutable, result: T): T {\r\n result.x = this.x + otherVector.x;\r\n result.y = this.y + otherVector.y;\r\n return result;\r\n }\r\n\r\n /**\r\n * Set the Vector2 coordinates by adding the given Vector2 coordinates\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#13\r\n * @param otherVector defines the other vector\r\n * @returns the current updated Vector2\r\n */\r\n public addInPlace(otherVector: DeepImmutable): this {\r\n this.x += otherVector.x;\r\n this.y += otherVector.y;\r\n return this;\r\n }\r\n\r\n /**\r\n * Gets a new Vector2 by adding the current Vector2 coordinates to the given Vector3 x, y coordinates\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#14\r\n * @param otherVector defines the other vector\r\n * @returns a new Vector2\r\n */\r\n public addVector3(otherVector: Vector3): this {\r\n return new (this.constructor as Vector2Constructor)(this.x + otherVector.x, this.y + otherVector.y);\r\n }\r\n\r\n /**\r\n * Gets a new Vector2 set with the subtracted coordinates of the given one from the current Vector2\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#61\r\n * @param otherVector defines the other vector\r\n * @returns a new Vector2\r\n */\r\n public subtract(otherVector: Vector2): this {\r\n return new (this.constructor as Vector2Constructor)(this.x - otherVector.x, this.y - otherVector.y);\r\n }\r\n\r\n /**\r\n * Sets the \"result\" coordinates with the subtraction of the given one from the current Vector2 coordinates.\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#63\r\n * @param otherVector defines the other vector\r\n * @param result defines the target vector\r\n * @returns result input\r\n */\r\n public subtractToRef(otherVector: DeepImmutable, result: T): T {\r\n result.x = this.x - otherVector.x;\r\n result.y = this.y - otherVector.y;\r\n return result;\r\n }\r\n /**\r\n * Sets the current Vector2 coordinates by subtracting from it the given one coordinates\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#88\r\n * @param otherVector defines the other vector\r\n * @returns the current updated Vector2\r\n */\r\n public subtractInPlace(otherVector: DeepImmutable): this {\r\n this.x -= otherVector.x;\r\n this.y -= otherVector.y;\r\n return this;\r\n }\r\n\r\n /**\r\n * Multiplies in place the current Vector2 coordinates by the given ones\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#43\r\n * @param otherVector defines the other vector\r\n * @returns the current updated Vector2\r\n */\r\n public multiplyInPlace(otherVector: DeepImmutable): this {\r\n this.x *= otherVector.x;\r\n this.y *= otherVector.y;\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns a new Vector2 set with the multiplication of the current Vector2 and the given one coordinates\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#42\r\n * @param otherVector defines the other vector\r\n * @returns a new Vector2\r\n */\r\n public multiply(otherVector: DeepImmutable): this {\r\n return new (this.constructor as Vector2Constructor)(this.x * otherVector.x, this.y * otherVector.y);\r\n }\r\n\r\n /**\r\n * Sets \"result\" coordinates with the multiplication of the current Vector2 and the given one coordinates\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#44\r\n * @param otherVector defines the other vector\r\n * @param result defines the target vector\r\n * @returns result input\r\n */\r\n public multiplyToRef(otherVector: DeepImmutable, result: T): T {\r\n result.x = this.x * otherVector.x;\r\n result.y = this.y * otherVector.y;\r\n return result;\r\n }\r\n\r\n /**\r\n * Gets a new Vector2 set with the Vector2 coordinates multiplied by the given floats\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#89\r\n * @param x defines the first coordinate\r\n * @param y defines the second coordinate\r\n * @returns a new Vector2\r\n */\r\n public multiplyByFloats(x: number, y: number): this {\r\n return new (this.constructor as Vector2Constructor)(this.x * x, this.y * y);\r\n }\r\n\r\n /**\r\n * Returns a new Vector2 set with the Vector2 coordinates divided by the given one coordinates\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#27\r\n * @param otherVector defines the other vector\r\n * @returns a new Vector2\r\n */\r\n public divide(otherVector: Vector2): this {\r\n return new (this.constructor as Vector2Constructor)(this.x / otherVector.x, this.y / otherVector.y);\r\n }\r\n\r\n /**\r\n * Sets the \"result\" coordinates with the Vector2 divided by the given one coordinates\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#30\r\n * @param otherVector defines the other vector\r\n * @param result defines the target vector\r\n * @returns result input\r\n */\r\n public divideToRef(otherVector: DeepImmutable, result: T): T {\r\n result.x = this.x / otherVector.x;\r\n result.y = this.y / otherVector.y;\r\n return result;\r\n }\r\n\r\n /**\r\n * Divides the current Vector2 coordinates by the given ones\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#28\r\n * @param otherVector defines the other vector\r\n * @returns the current updated Vector2\r\n */\r\n public divideInPlace(otherVector: DeepImmutable): this {\r\n return this.divideToRef(otherVector, this);\r\n }\r\n\r\n /**\r\n * Gets a new Vector2 with current Vector2 negated coordinates\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#22\r\n * @returns a new Vector2\r\n */\r\n public negate(): this {\r\n return new (this.constructor as Vector2Constructor)(-this.x, -this.y);\r\n }\r\n\r\n /**\r\n * Negate this vector in place\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#23\r\n * @returns this\r\n */\r\n public negateInPlace(): this {\r\n this.x *= -1;\r\n this.y *= -1;\r\n return this;\r\n }\r\n\r\n /**\r\n * Negate the current Vector2 and stores the result in the given vector \"result\" coordinates\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#41\r\n * @param result defines the Vector3 object where to store the result\r\n * @returns the result\r\n */\r\n public negateToRef(result: T): T {\r\n return result.copyFromFloats(this.x * -1, this.y * -1);\r\n }\r\n\r\n /**\r\n * Multiply the Vector2 coordinates by\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#59\r\n * @param scale defines the scaling factor\r\n * @returns the current updated Vector2\r\n */\r\n public scaleInPlace(scale: number): this {\r\n this.x *= scale;\r\n this.y *= scale;\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns a new Vector2 scaled by \"scale\" from the current Vector2\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#52\r\n * @param scale defines the scaling factor\r\n * @returns a new Vector2\r\n */\r\n public scale(scale: number): this {\r\n const result = new (this.constructor as Vector2Constructor)(0, 0);\r\n this.scaleToRef(scale, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Scale the current Vector2 values by a factor to a given Vector2\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#57\r\n * @param scale defines the scale factor\r\n * @param result defines the Vector2 object where to store the result\r\n * @returns result input\r\n */\r\n public scaleToRef(scale: number, result: T): T {\r\n result.x = this.x * scale;\r\n result.y = this.y * scale;\r\n return result;\r\n }\r\n\r\n /**\r\n * Scale the current Vector2 values by a factor and add the result to a given Vector2\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#58\r\n * @param scale defines the scale factor\r\n * @param result defines the Vector2 object where to store the result\r\n * @returns result input\r\n */\r\n public scaleAndAddToRef(scale: number, result: T): T {\r\n result.x += this.x * scale;\r\n result.y += this.y * scale;\r\n return result;\r\n }\r\n\r\n /**\r\n * Gets a boolean if two vectors are equals\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#31\r\n * @param otherVector defines the other vector\r\n * @returns true if the given vector coordinates strictly equal the current Vector2 ones\r\n */\r\n public equals(otherVector: DeepImmutable): boolean {\r\n return otherVector && this.x === otherVector.x && this.y === otherVector.y;\r\n }\r\n\r\n /**\r\n * Gets a boolean if two vectors are equals (using an epsilon value)\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#32\r\n * @param otherVector defines the other vector\r\n * @param epsilon defines the minimal distance to consider equality\r\n * @returns true if the given vector coordinates are close to the current ones by a distance of epsilon.\r\n */\r\n public equalsWithEpsilon(otherVector: DeepImmutable, epsilon: number = Epsilon): boolean {\r\n return otherVector && Scalar.WithinEpsilon(this.x, otherVector.x, epsilon) && Scalar.WithinEpsilon(this.y, otherVector.y, epsilon);\r\n }\r\n\r\n /**\r\n * Gets a new Vector2 from current Vector2 floored values\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#35\r\n * eg (1.2, 2.31) returns (1, 2)\r\n * @returns a new Vector2\r\n */\r\n public floor(): this {\r\n return new (this.constructor as Vector2Constructor)(Math.floor(this.x), Math.floor(this.y));\r\n }\r\n\r\n /**\r\n * Gets a new Vector2 from current Vector2 fractional values\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#34\r\n * eg (1.2, 2.31) returns (0.2, 0.31)\r\n * @returns a new Vector2\r\n */\r\n public fract(): this {\r\n return new (this.constructor as Vector2Constructor)(this.x - Math.floor(this.x), this.y - Math.floor(this.y));\r\n }\r\n\r\n /**\r\n * Rotate the current vector into a given result vector\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#49\r\n * @param angle defines the rotation angle\r\n * @param result defines the result vector where to store the rotated vector\r\n * @returns result input\r\n */\r\n public rotateToRef(angle: number, result: T): T {\r\n const cos = Math.cos(angle);\r\n const sin = Math.sin(angle);\r\n const x = cos * this.x - sin * this.y;\r\n const y = sin * this.x + cos * this.y;\r\n result.x = x;\r\n result.y = y;\r\n return result;\r\n }\r\n\r\n // Properties\r\n\r\n /**\r\n * Gets the length of the vector\r\n * @returns the vector length (float)\r\n */\r\n public length(): number {\r\n return Math.sqrt(this.x * this.x + this.y * this.y);\r\n }\r\n\r\n /**\r\n * Gets the vector squared length\r\n * @returns the vector squared length (float)\r\n */\r\n public lengthSquared(): number {\r\n return this.x * this.x + this.y * this.y;\r\n }\r\n\r\n // Methods\r\n\r\n /**\r\n * Normalize the vector\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#48\r\n * @returns the current updated Vector2\r\n */\r\n public normalize(): this {\r\n Vector2.NormalizeToRef(this, this);\r\n return this;\r\n }\r\n\r\n /**\r\n * Gets a new Vector2 copied from the Vector2\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#20\r\n * @returns a new Vector2\r\n */\r\n public clone(): this {\r\n return new (this.constructor as Vector2Constructor)(this.x, this.y);\r\n }\r\n\r\n // Statics\r\n\r\n /**\r\n * Gets a new Vector2(0, 0)\r\n * @returns a new Vector2\r\n */\r\n public static Zero(): Vector2 {\r\n return new Vector2(0, 0);\r\n }\r\n\r\n /**\r\n * Gets a new Vector2(1, 1)\r\n * @returns a new Vector2\r\n */\r\n public static One(): Vector2 {\r\n return new Vector2(1, 1);\r\n }\r\n\r\n /**\r\n * Returns a new Vector2 with random values between min and max\r\n * @param min the minimum random value\r\n * @param max the maximum random value\r\n * @returns a Vector2 with random values between min and max\r\n */\r\n public static Random(min: number = 0, max: number = 1): Vector2 {\r\n return new Vector2(Scalar.RandomRange(min, max), Scalar.RandomRange(min, max));\r\n }\r\n\r\n /**\r\n * Gets a zero Vector2 that must not be updated\r\n */\r\n public static get ZeroReadOnly(): DeepImmutable {\r\n return Vector2._ZeroReadOnly;\r\n }\r\n\r\n /**\r\n * Gets a new Vector2 set from the given index element of the given array\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#79\r\n * @param array defines the data source\r\n * @param offset defines the offset in the data source\r\n * @returns a new Vector2\r\n */\r\n public static FromArray(array: DeepImmutable>, offset: number = 0): Vector2 {\r\n return new Vector2(array[offset], array[offset + 1]);\r\n }\r\n\r\n /**\r\n * Sets \"result\" from the given index element of the given array\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#80\r\n * @param array defines the data source\r\n * @param offset defines the offset in the data source\r\n * @param result defines the target vector\r\n * @returns result input\r\n */\r\n public static FromArrayToRef(array: DeepImmutable>, offset: number, result: T): T {\r\n result.x = array[offset];\r\n result.y = array[offset + 1];\r\n return result;\r\n }\r\n\r\n /**\r\n * Gets a new Vector2 located for \"amount\" (float) on the CatmullRom spline defined by the given four Vector2\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#65\r\n * @param value1 defines 1st point of control\r\n * @param value2 defines 2nd point of control\r\n * @param value3 defines 3rd point of control\r\n * @param value4 defines 4th point of control\r\n * @param amount defines the interpolation factor\r\n * @returns a new Vector2\r\n */\r\n public static CatmullRom(\r\n value1: DeepImmutable,\r\n value2: DeepImmutable,\r\n value3: DeepImmutable,\r\n value4: DeepImmutable,\r\n amount: number\r\n ): T {\r\n const squared = amount * amount;\r\n const cubed = amount * squared;\r\n\r\n const x =\r\n 0.5 *\r\n (2.0 * value2.x +\r\n (-value1.x + value3.x) * amount +\r\n (2.0 * value1.x - 5.0 * value2.x + 4.0 * value3.x - value4.x) * squared +\r\n (-value1.x + 3.0 * value2.x - 3.0 * value3.x + value4.x) * cubed);\r\n\r\n const y =\r\n 0.5 *\r\n (2.0 * value2.y +\r\n (-value1.y + value3.y) * amount +\r\n (2.0 * value1.y - 5.0 * value2.y + 4.0 * value3.y - value4.y) * squared +\r\n (-value1.y + 3.0 * value2.y - 3.0 * value3.y + value4.y) * cubed);\r\n\r\n return new (value1.constructor as Vector2Constructor)(x, y);\r\n }\r\n\r\n /**\r\n * Returns a new Vector2 set with same the coordinates than \"value\" ones if the vector \"value\" is in the square defined by \"min\" and \"max\".\r\n * If a coordinate of \"value\" is lower than \"min\" coordinates, the returned Vector2 is given this \"min\" coordinate.\r\n * If a coordinate of \"value\" is greater than \"max\" coordinates, the returned Vector2 is given this \"max\" coordinate\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#76\r\n * @param value defines the value to clamp\r\n * @param min defines the lower limit\r\n * @param max defines the upper limit\r\n * @returns a new Vector2\r\n */\r\n public static Clamp(value: DeepImmutable, min: DeepImmutable, max: DeepImmutable): T {\r\n let x = value.x;\r\n x = x > max.x ? max.x : x;\r\n x = x < min.x ? min.x : x;\r\n\r\n let y = value.y;\r\n y = y > max.y ? max.y : y;\r\n y = y < min.y ? min.y : y;\r\n\r\n return new (value.constructor as Vector2Constructor)(x, y);\r\n }\r\n\r\n /**\r\n * Returns a new Vector2 located for \"amount\" (float) on the Hermite spline defined by the vectors \"value1\", \"value2\", \"tangent1\", \"tangent2\"\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#81\r\n * @param value1 defines the 1st control point\r\n * @param tangent1 defines the outgoing tangent\r\n * @param value2 defines the 2nd control point\r\n * @param tangent2 defines the incoming tangent\r\n * @param amount defines the interpolation factor\r\n * @returns a new Vector2\r\n */\r\n public static Hermite(\r\n value1: DeepImmutable,\r\n tangent1: DeepImmutable,\r\n value2: DeepImmutable,\r\n tangent2: DeepImmutable,\r\n amount: number\r\n ): T {\r\n const squared = amount * amount;\r\n const cubed = amount * squared;\r\n const part1 = 2.0 * cubed - 3.0 * squared + 1.0;\r\n const part2 = -2.0 * cubed + 3.0 * squared;\r\n const part3 = cubed - 2.0 * squared + amount;\r\n const part4 = cubed - squared;\r\n\r\n const x = value1.x * part1 + value2.x * part2 + tangent1.x * part3 + tangent2.x * part4;\r\n const y = value1.y * part1 + value2.y * part2 + tangent1.y * part3 + tangent2.y * part4;\r\n\r\n return new (value1.constructor as Vector2Constructor)(x, y);\r\n }\r\n\r\n /**\r\n * Returns a new Vector2 which is the 1st derivative of the Hermite spline defined by the vectors \"value1\", \"value2\", \"tangent1\", \"tangent2\".\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#82\r\n * @param value1 defines the first control point\r\n * @param tangent1 defines the first tangent\r\n * @param value2 defines the second control point\r\n * @param tangent2 defines the second tangent\r\n * @param time define where the derivative must be done\r\n * @returns 1st derivative\r\n */\r\n public static Hermite1stDerivative(\r\n value1: DeepImmutable,\r\n tangent1: DeepImmutable,\r\n value2: DeepImmutable,\r\n tangent2: DeepImmutable,\r\n time: number\r\n ): T {\r\n const result = new (value1.constructor as Vector2Constructor)();\r\n\r\n this.Hermite1stDerivativeToRef(value1, tangent1, value2, tangent2, time, result);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Returns a new Vector2 which is the 1st derivative of the Hermite spline defined by the vectors \"value1\", \"value2\", \"tangent1\", \"tangent2\".\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#83\r\n * @param value1 defines the first control point\r\n * @param tangent1 defines the first tangent\r\n * @param value2 defines the second control point\r\n * @param tangent2 defines the second tangent\r\n * @param time define where the derivative must be done\r\n * @param result define where the derivative will be stored\r\n * @returns result input\r\n */\r\n public static Hermite1stDerivativeToRef(\r\n value1: DeepImmutable,\r\n tangent1: DeepImmutable,\r\n value2: DeepImmutable,\r\n tangent2: DeepImmutable,\r\n time: number,\r\n result: T\r\n ): T {\r\n const t2 = time * time;\r\n\r\n result.x = (t2 - time) * 6 * value1.x + (3 * t2 - 4 * time + 1) * tangent1.x + (-t2 + time) * 6 * value2.x + (3 * t2 - 2 * time) * tangent2.x;\r\n result.y = (t2 - time) * 6 * value1.y + (3 * t2 - 4 * time + 1) * tangent1.y + (-t2 + time) * 6 * value2.y + (3 * t2 - 2 * time) * tangent2.y;\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Returns a new Vector2 located for \"amount\" (float) on the linear interpolation between the vector \"start\" adn the vector \"end\".\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#84\r\n * @param start defines the start vector\r\n * @param end defines the end vector\r\n * @param amount defines the interpolation factor\r\n * @returns a new Vector2\r\n */\r\n public static Lerp(start: DeepImmutable, end: DeepImmutable, amount: number): Vector2 {\r\n const x = start.x + (end.x - start.x) * amount;\r\n const y = start.y + (end.y - start.y) * amount;\r\n return new (start.constructor as Vector2Constructor)(x, y);\r\n }\r\n\r\n /**\r\n * Gets the dot product of the vector \"left\" and the vector \"right\"\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#90\r\n * @param left defines first vector\r\n * @param right defines second vector\r\n * @returns the dot product (float)\r\n */\r\n public static Dot(left: DeepImmutable, right: DeepImmutable): number {\r\n return left.x * right.x + left.y * right.y;\r\n }\r\n\r\n /**\r\n * Returns a new Vector2 equal to the normalized given vector\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#46\r\n * @param vector defines the vector to normalize\r\n * @returns a new Vector2\r\n */\r\n public static Normalize(vector: DeepImmutable): T {\r\n const newVector = new (vector.constructor as Vector2Constructor)();\r\n this.NormalizeToRef(vector, newVector);\r\n return newVector;\r\n }\r\n\r\n /**\r\n * Normalize a given vector into a second one\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#50\r\n * @param vector defines the vector to normalize\r\n * @param result defines the vector where to store the result\r\n * @returns result input\r\n */\r\n public static NormalizeToRef(vector: DeepImmutable, result: T): T {\r\n const len = vector.length();\r\n\r\n if (len === 0) {\r\n return result;\r\n }\r\n\r\n result.x = vector.x / len;\r\n result.y = vector.y / len;\r\n return result;\r\n }\r\n\r\n /**\r\n * Gets a new Vector2 set with the minimal coordinate values from the \"left\" and \"right\" vectors\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#86\r\n * @param left defines 1st vector\r\n * @param right defines 2nd vector\r\n * @returns a new Vector2\r\n */\r\n public static Minimize(left: DeepImmutable, right: DeepImmutable): T {\r\n const x = left.x < right.x ? left.x : right.x;\r\n const y = left.y < right.y ? left.y : right.y;\r\n return new (left.constructor as Vector2Constructor)(x, y);\r\n }\r\n\r\n /**\r\n * Gets a new Vector2 set with the maximal coordinate values from the \"left\" and \"right\" vectors\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#86\r\n * @param left defines 1st vector\r\n * @param right defines 2nd vector\r\n * @returns a new Vector2\r\n */\r\n public static Maximize(left: DeepImmutable, right: DeepImmutable): T {\r\n const x = left.x > right.x ? left.x : right.x;\r\n const y = left.y > right.y ? left.y : right.y;\r\n return new (left.constructor as Vector2Constructor)(x, y);\r\n }\r\n\r\n /**\r\n * Gets a new Vector2 set with the transformed coordinates of the given vector by the given transformation matrix\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#17\r\n * @param vector defines the vector to transform\r\n * @param transformation defines the matrix to apply\r\n * @returns a new Vector2\r\n */\r\n public static Transform(vector: DeepImmutable, transformation: DeepImmutable): T {\r\n const result = new (vector.constructor as Vector2Constructor)();\r\n Vector2.TransformToRef(vector, transformation, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Transforms the given vector coordinates by the given transformation matrix and stores the result in the vector \"result\" coordinates\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#19\r\n * @param vector defines the vector to transform\r\n * @param transformation defines the matrix to apply\r\n * @param result defines the target vector\r\n * @returns result input\r\n */\r\n public static TransformToRef(vector: DeepImmutable, transformation: DeepImmutable, result: T): T {\r\n const m = transformation.m;\r\n const x = vector.x * m[0] + vector.y * m[4] + m[12];\r\n const y = vector.x * m[1] + vector.y * m[5] + m[13];\r\n result.x = x;\r\n result.y = y;\r\n return result;\r\n }\r\n\r\n /**\r\n * Determines if a given vector is included in a triangle\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#87\r\n * @param p defines the vector to test\r\n * @param p0 defines 1st triangle point\r\n * @param p1 defines 2nd triangle point\r\n * @param p2 defines 3rd triangle point\r\n * @returns true if the point \"p\" is in the triangle defined by the vectors \"p0\", \"p1\", \"p2\"\r\n */\r\n public static PointInTriangle(p: DeepImmutable, p0: DeepImmutable, p1: DeepImmutable, p2: DeepImmutable): boolean {\r\n const a = (1 / 2) * (-p1.y * p2.x + p0.y * (-p1.x + p2.x) + p0.x * (p1.y - p2.y) + p1.x * p2.y);\r\n const sign = a < 0 ? -1 : 1;\r\n const s = (p0.y * p2.x - p0.x * p2.y + (p2.y - p0.y) * p.x + (p0.x - p2.x) * p.y) * sign;\r\n const t = (p0.x * p1.y - p0.y * p1.x + (p0.y - p1.y) * p.x + (p1.x - p0.x) * p.y) * sign;\r\n\r\n return s > 0 && t > 0 && s + t < 2 * a * sign;\r\n }\r\n\r\n /**\r\n * Gets the distance between the vectors \"value1\" and \"value2\"\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#71\r\n * @param value1 defines first vector\r\n * @param value2 defines second vector\r\n * @returns the distance between vectors\r\n */\r\n public static Distance(value1: DeepImmutable, value2: DeepImmutable): number {\r\n return Math.sqrt(Vector2.DistanceSquared(value1, value2));\r\n }\r\n\r\n /**\r\n * Returns the squared distance between the vectors \"value1\" and \"value2\"\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#72\r\n * @param value1 defines first vector\r\n * @param value2 defines second vector\r\n * @returns the squared distance between vectors\r\n */\r\n public static DistanceSquared(value1: DeepImmutable, value2: DeepImmutable): number {\r\n const x = value1.x - value2.x;\r\n const y = value1.y - value2.y;\r\n return x * x + y * y;\r\n }\r\n\r\n /**\r\n * Gets a new Vector2 located at the center of the vectors \"value1\" and \"value2\"\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#86\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#66\r\n * @param value1 defines first vector\r\n * @param value2 defines second vector\r\n * @returns a new Vector2\r\n */\r\n public static Center(value1: DeepImmutable, value2: DeepImmutable): T {\r\n const result = new (value1.constructor as Vector2Constructor)();\r\n return Vector2.CenterToRef(value1, value2, result);\r\n }\r\n\r\n /**\r\n * Gets the center of the vectors \"value1\" and \"value2\" and stores the result in the vector \"ref\"\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#66\r\n * @param value1 defines first vector\r\n * @param value2 defines second vector\r\n * @param ref defines third vector\r\n * @returns ref\r\n */\r\n public static CenterToRef(value1: DeepImmutable, value2: DeepImmutable, ref: T): T {\r\n return ref.copyFromFloats((value1.x + value2.x) / 2, (value1.y + value2.y) / 2);\r\n }\r\n\r\n /**\r\n * Gets the shortest distance (float) between the point \"p\" and the segment defined by the two points \"segA\" and \"segB\".\r\n * Example Playground https://playground.babylonjs.com/#QYBWV4#77\r\n * @param p defines the middle point\r\n * @param segA defines one point of the segment\r\n * @param segB defines the other point of the segment\r\n * @returns the shortest distance\r\n */\r\n public static DistanceOfPointFromSegment(p: DeepImmutable, segA: DeepImmutable, segB: DeepImmutable): number {\r\n const l2 = Vector2.DistanceSquared(segA, segB);\r\n if (l2 === 0.0) {\r\n return Vector2.Distance(p, segA);\r\n }\r\n const v = segB.subtract(segA);\r\n const t = Math.max(0, Math.min(1, Vector2.Dot(p.subtract(segA), v) / l2));\r\n const proj = segA.add(v.multiplyByFloats(t, t));\r\n return Vector2.Distance(p, proj);\r\n }\r\n}\r\n\r\n/**\r\n * Class used to store (x,y,z) vector representation\r\n * A Vector3 is the main object used in 3D geometry\r\n * It can represent either the coordinates of a point the space, either a direction\r\n * Reminder: js uses a left handed forward facing system\r\n * Example Playground - Overview - https://playground.babylonjs.com/#R1F8YU\r\n */\r\nexport class Vector3 {\r\n private static _UpReadOnly = Vector3.Up() as DeepImmutable;\r\n private static _DownReadOnly = Vector3.Down() as DeepImmutable;\r\n private static _LeftHandedForwardReadOnly = Vector3.Forward(false) as DeepImmutable;\r\n private static _RightHandedForwardReadOnly = Vector3.Forward(true) as DeepImmutable;\r\n private static _LeftHandedBackwardReadOnly = Vector3.Backward(false) as DeepImmutable;\r\n private static _RightHandedBackwardReadOnly = Vector3.Backward(true) as DeepImmutable;\r\n private static _RightReadOnly = Vector3.Right() as DeepImmutable;\r\n private static _LeftReadOnly = Vector3.Left() as DeepImmutable;\r\n private static _ZeroReadOnly = Vector3.Zero() as DeepImmutable;\r\n private static _OneReadOnly = Vector3.One() as DeepImmutable;\r\n\r\n /** @internal */\r\n public _x: number;\r\n\r\n /** @internal */\r\n public _y: number;\r\n\r\n /** @internal */\r\n public _z: number;\r\n\r\n /** @internal */\r\n public _isDirty = true;\r\n\r\n /** Gets or sets the x coordinate */\r\n public get x() {\r\n return this._x;\r\n }\r\n\r\n public set x(value: number) {\r\n this._x = value;\r\n this._isDirty = true;\r\n }\r\n\r\n /** Gets or sets the y coordinate */\r\n public get y() {\r\n return this._y;\r\n }\r\n\r\n public set y(value: number) {\r\n this._y = value;\r\n this._isDirty = true;\r\n }\r\n\r\n /** Gets or sets the z coordinate */\r\n public get z() {\r\n return this._z;\r\n }\r\n\r\n public set z(value: number) {\r\n this._z = value;\r\n this._isDirty = true;\r\n }\r\n\r\n /**\r\n * Creates a new Vector3 object from the given x, y, z (floats) coordinates.\r\n * @param x defines the first coordinates (on X axis)\r\n * @param y defines the second coordinates (on Y axis)\r\n * @param z defines the third coordinates (on Z axis)\r\n */\r\n constructor(x: number = 0, y: number = 0, z: number = 0) {\r\n this._x = x;\r\n this._y = y;\r\n this._z = z;\r\n }\r\n\r\n /**\r\n * Creates a string representation of the Vector3\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#67\r\n * @returns a string with the Vector3 coordinates.\r\n */\r\n public toString(): string {\r\n return `{X: ${this._x} Y: ${this._y} Z: ${this._z}}`;\r\n }\r\n\r\n /**\r\n * Gets the class name\r\n * @returns the string \"Vector3\"\r\n */\r\n public getClassName(): string {\r\n return \"Vector3\";\r\n }\r\n\r\n /**\r\n * Creates the Vector3 hash code\r\n * @returns a number which tends to be unique between Vector3 instances\r\n */\r\n public getHashCode(): number {\r\n const x = _ExtractAsInt(this._x);\r\n const y = _ExtractAsInt(this._y);\r\n const z = _ExtractAsInt(this._z);\r\n\r\n let hash = x;\r\n hash = (hash * 397) ^ y;\r\n hash = (hash * 397) ^ z;\r\n return hash;\r\n }\r\n\r\n // Operators\r\n\r\n /**\r\n * Creates an array containing three elements : the coordinates of the Vector3\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#10\r\n * @returns a new array of numbers\r\n */\r\n public asArray(): number[] {\r\n const result: number[] = [];\r\n this.toArray(result, 0);\r\n return result;\r\n }\r\n\r\n /**\r\n * Populates the given array or Float32Array from the given index with the successive coordinates of the Vector3\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#65\r\n * @param array defines the destination array\r\n * @param index defines the offset in the destination array\r\n * @returns the current Vector3\r\n */\r\n public toArray(array: FloatArray, index: number = 0): this {\r\n array[index] = this._x;\r\n array[index + 1] = this._y;\r\n array[index + 2] = this._z;\r\n return this;\r\n }\r\n\r\n /**\r\n * Update the current vector from an array\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#24\r\n * @param array defines the destination array\r\n * @param index defines the offset in the destination array\r\n * @returns the current Vector3\r\n */\r\n public fromArray(array: FloatArray, index: number = 0): this {\r\n Vector3.FromArrayToRef(array, index, this);\r\n return this;\r\n }\r\n\r\n /**\r\n * Converts the current Vector3 into a quaternion (considering that the Vector3 contains Euler angles representation of a rotation)\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#66\r\n * @returns a new Quaternion object, computed from the Vector3 coordinates\r\n */\r\n public toQuaternion(): Quaternion {\r\n return Quaternion.RotationYawPitchRoll(this._y, this._x, this._z);\r\n }\r\n\r\n /**\r\n * Adds the given vector to the current Vector3\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#4\r\n * @param otherVector defines the second operand\r\n * @returns the current updated Vector3\r\n */\r\n public addInPlace(otherVector: DeepImmutable): this {\r\n return this.addInPlaceFromFloats(otherVector._x, otherVector._y, otherVector._z);\r\n }\r\n\r\n /**\r\n * Adds the given coordinates to the current Vector3\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#5\r\n * @param x defines the x coordinate of the operand\r\n * @param y defines the y coordinate of the operand\r\n * @param z defines the z coordinate of the operand\r\n * @returns the current updated Vector3\r\n */\r\n public addInPlaceFromFloats(x: number, y: number, z: number): this {\r\n this._x += x;\r\n this._y += y;\r\n this._z += z;\r\n this._isDirty = true;\r\n return this;\r\n }\r\n\r\n /**\r\n * Gets a new Vector3, result of the addition the current Vector3 and the given vector\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#3\r\n * @param otherVector defines the second operand\r\n * @returns the resulting Vector3\r\n */\r\n public add(otherVector: DeepImmutable): this {\r\n return new (this.constructor as Vector3Constructor)(this._x + otherVector._x, this._y + otherVector._y, this._z + otherVector._z);\r\n }\r\n\r\n /**\r\n * Adds the current Vector3 to the given one and stores the result in the vector \"result\"\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#6\r\n * @param otherVector defines the second operand\r\n * @param result defines the Vector3 object where to store the result\r\n * @returns the result\r\n */\r\n public addToRef(otherVector: DeepImmutable, result: T): T {\r\n return result.copyFromFloats(this._x + otherVector._x, this._y + otherVector._y, this._z + otherVector._z);\r\n }\r\n\r\n /**\r\n * Subtract the given vector from the current Vector3\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#61\r\n * @param otherVector defines the second operand\r\n * @returns the current updated Vector3\r\n */\r\n public subtractInPlace(otherVector: DeepImmutable): this {\r\n this._x -= otherVector._x;\r\n this._y -= otherVector._y;\r\n this._z -= otherVector._z;\r\n this._isDirty = true;\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns a new Vector3, result of the subtraction of the given vector from the current Vector3\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#60\r\n * @param otherVector defines the second operand\r\n * @returns the resulting Vector3\r\n */\r\n public subtract(otherVector: DeepImmutable): this {\r\n return new (this.constructor as Vector3Constructor)(this._x - otherVector._x, this._y - otherVector._y, this._z - otherVector._z);\r\n }\r\n\r\n /**\r\n * Subtracts the given vector from the current Vector3 and stores the result in the vector \"result\".\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#63\r\n * @param otherVector defines the second operand\r\n * @param result defines the Vector3 object where to store the result\r\n * @returns the result\r\n */\r\n public subtractToRef(otherVector: DeepImmutable, result: T): T {\r\n return this.subtractFromFloatsToRef(otherVector._x, otherVector._y, otherVector._z, result);\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 set with the subtraction of the given floats from the current Vector3 coordinates\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#62\r\n * @param x defines the x coordinate of the operand\r\n * @param y defines the y coordinate of the operand\r\n * @param z defines the z coordinate of the operand\r\n * @returns the resulting Vector3\r\n */\r\n public subtractFromFloats(x: number, y: number, z: number): this {\r\n return new (this.constructor as Vector3Constructor)(this._x - x, this._y - y, this._z - z);\r\n }\r\n\r\n /**\r\n * Subtracts the given floats from the current Vector3 coordinates and set the given vector \"result\" with this result\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#64\r\n * @param x defines the x coordinate of the operand\r\n * @param y defines the y coordinate of the operand\r\n * @param z defines the z coordinate of the operand\r\n * @param result defines the Vector3 object where to store the result\r\n * @returns the result\r\n */\r\n public subtractFromFloatsToRef(x: number, y: number, z: number, result: T): T {\r\n return result.copyFromFloats(this._x - x, this._y - y, this._z - z);\r\n }\r\n\r\n /**\r\n * Gets a new Vector3 set with the current Vector3 negated coordinates\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#35\r\n * @returns a new Vector3\r\n */\r\n public negate(): this {\r\n return new (this.constructor as Vector3Constructor)(-this._x, -this._y, -this._z);\r\n }\r\n\r\n /**\r\n * Negate this vector in place\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#36\r\n * @returns this\r\n */\r\n public negateInPlace(): this {\r\n this._x *= -1;\r\n this._y *= -1;\r\n this._z *= -1;\r\n this._isDirty = true;\r\n return this;\r\n }\r\n\r\n /**\r\n * Negate the current Vector3 and stores the result in the given vector \"result\" coordinates\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#37\r\n * @param result defines the Vector3 object where to store the result\r\n * @returns the result\r\n */\r\n public negateToRef(result: T): T {\r\n return result.copyFromFloats(this._x * -1, this._y * -1, this._z * -1);\r\n }\r\n\r\n /**\r\n * Multiplies the Vector3 coordinates by the float \"scale\"\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#56\r\n * @param scale defines the multiplier factor\r\n * @returns the current updated Vector3\r\n */\r\n public scaleInPlace(scale: number): this {\r\n this._x *= scale;\r\n this._y *= scale;\r\n this._z *= scale;\r\n this._isDirty = true;\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 set with the current Vector3 coordinates multiplied by the float \"scale\"\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#53\r\n * @param scale defines the multiplier factor\r\n * @returns a new Vector3\r\n */\r\n public scale(scale: number): this {\r\n return new (this.constructor as Vector3Constructor)(this._x * scale, this._y * scale, this._z * scale);\r\n }\r\n\r\n /**\r\n * Multiplies the current Vector3 coordinates by the float \"scale\" and stores the result in the given vector \"result\" coordinates\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#57\r\n * @param scale defines the multiplier factor\r\n * @param result defines the Vector3 object where to store the result\r\n * @returns the result\r\n */\r\n public scaleToRef(scale: number, result: T): T {\r\n return result.copyFromFloats(this._x * scale, this._y * scale, this._z * scale);\r\n }\r\n\r\n /**\r\n * Creates a vector normal (perpendicular) to the current Vector3 and stores the result in the given vector\r\n * Out of the infinite possibilities the normal chosen is the one formed by rotating the current vector\r\n * 90 degrees about an axis which lies perpendicular to the current vector\r\n * and its projection on the xz plane. In the case of a current vector in the xz plane\r\n * the normal is calculated to be along the y axis.\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#230\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#231\r\n * @param result defines the Vector3 object where to store the resultant normal\r\n * returns the result\r\n */\r\n public getNormalToRef(result: DeepImmutable): Vector3 {\r\n /**\r\n * Calculates the spherical coordinates of the current vector\r\n * so saves on memory rather than importing whole Spherical Class\r\n */\r\n const radius: number = this.length();\r\n let theta: number = Math.acos(this.y / radius);\r\n const phi = Math.atan2(this.z, this.x);\r\n //makes angle 90 degs to current vector\r\n if (theta > Math.PI / 2) {\r\n theta -= Math.PI / 2;\r\n } else {\r\n theta += Math.PI / 2;\r\n }\r\n //Calculates resutant normal vector from spherical coordinate of perpendicular vector\r\n const x = radius * Math.sin(theta) * Math.cos(phi);\r\n const y = radius * Math.cos(theta);\r\n const z = radius * Math.sin(theta) * Math.sin(phi);\r\n result.set(x, y, z);\r\n return result;\r\n }\r\n\r\n /**\r\n * Rotates the vector using the given unit quaternion and stores the new vector in result\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#9\r\n * @param q the unit quaternion representing the rotation\r\n * @param result the output vector\r\n * @returns the result\r\n */\r\n public applyRotationQuaternionToRef(q: Quaternion, result: T): T {\r\n // Derived from https://raw.org/proof/vector-rotation-using-quaternions/\r\n\r\n const vx = this._x,\r\n vy = this._y,\r\n vz = this._z;\r\n const qx = q._x,\r\n qy = q._y,\r\n qz = q._z,\r\n qw = q._w;\r\n\r\n // t = 2q x v\r\n const tx = 2 * (qy * vz - qz * vy);\r\n const ty = 2 * (qz * vx - qx * vz);\r\n const tz = 2 * (qx * vy - qy * vx);\r\n\r\n // v + w t + q x t\r\n result._x = vx + qw * tx + qy * tz - qz * ty;\r\n result._y = vy + qw * ty + qz * tx - qx * tz;\r\n result._z = vz + qw * tz + qx * ty - qy * tx;\r\n\r\n result._isDirty = true;\r\n return result;\r\n }\r\n\r\n /**\r\n * Rotates the vector in place using the given unit quaternion\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#8\r\n * @param q the unit quaternion representing the rotation\r\n * @returns the current updated Vector3\r\n */\r\n public applyRotationQuaternionInPlace(q: Quaternion): this {\r\n return this.applyRotationQuaternionToRef(q, this);\r\n }\r\n\r\n /**\r\n * Rotates the vector using the given unit quaternion and returns the new vector\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#7\r\n * @param q the unit quaternion representing the rotation\r\n * @returns a new Vector3\r\n */\r\n public applyRotationQuaternion(q: Quaternion): this {\r\n return this.applyRotationQuaternionToRef(q, new (this.constructor as Vector3Constructor)());\r\n }\r\n\r\n /**\r\n * Scale the current Vector3 values by a factor and add the result to a given Vector3\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#55\r\n * @param scale defines the scale factor\r\n * @param result defines the Vector3 object where to store the result\r\n * @returns result input\r\n */\r\n public scaleAndAddToRef(scale: number, result: T): T {\r\n return result.addInPlaceFromFloats(this._x * scale, this._y * scale, this._z * scale);\r\n }\r\n\r\n /**\r\n * Projects the current point Vector3 to a plane along a ray starting from a specified origin and passing through the current point Vector3.\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#48\r\n * @param plane defines the plane to project to\r\n * @param origin defines the origin of the projection ray\r\n * @returns the projected vector3\r\n */\r\n public projectOnPlane(plane: Plane, origin: Vector3): T {\r\n const result = new (this.constructor as Vector3Constructor)();\r\n this.projectOnPlaneToRef(plane, origin, result);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Projects the current point Vector3 to a plane along a ray starting from a specified origin and passing through the current point Vector3.\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#49\r\n * @param plane defines the plane to project to\r\n * @param origin defines the origin of the projection ray\r\n * @param result defines the Vector3 where to store the result\r\n * @returns result input\r\n */\r\n public projectOnPlaneToRef(plane: Plane, origin: Vector3, result: T): T {\r\n const n = plane.normal;\r\n const d = plane.d;\r\n\r\n const V = MathTmp.Vector3[0];\r\n\r\n // ray direction\r\n this.subtractToRef(origin, V);\r\n\r\n V.normalize();\r\n\r\n const denom = Vector3.Dot(V, n);\r\n\r\n //When the ray is close to parallel to the plane return infinity vector\r\n if (Math.abs(denom) < Math.pow(10, -10)) {\r\n result.setAll(Infinity);\r\n } else {\r\n const t = -(Vector3.Dot(origin, n) + d) / denom;\r\n\r\n // P = P0 + t*V\r\n const scaledV = V.scaleInPlace(t);\r\n origin.addToRef(scaledV, result);\r\n }\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Returns true if the current Vector3 and the given vector coordinates are strictly equal\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#19\r\n * @param otherVector defines the second operand\r\n * @returns true if both vectors are equals\r\n */\r\n public equals(otherVector: DeepImmutable): boolean {\r\n return otherVector && this._x === otherVector._x && this._y === otherVector._y && this._z === otherVector._z;\r\n }\r\n\r\n /**\r\n * Returns true if the current Vector3 and the given vector coordinates are distant less than epsilon\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#21\r\n * @param otherVector defines the second operand\r\n * @param epsilon defines the minimal distance to define values as equals\r\n * @returns true if both vectors are distant less than epsilon\r\n */\r\n public equalsWithEpsilon(otherVector: DeepImmutable, epsilon: number = Epsilon): boolean {\r\n return (\r\n otherVector &&\r\n Scalar.WithinEpsilon(this._x, otherVector._x, epsilon) &&\r\n Scalar.WithinEpsilon(this._y, otherVector._y, epsilon) &&\r\n Scalar.WithinEpsilon(this._z, otherVector._z, epsilon)\r\n );\r\n }\r\n\r\n /**\r\n * Returns true if the current Vector3 coordinates equals the given floats\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#20\r\n * @param x defines the x coordinate of the operand\r\n * @param y defines the y coordinate of the operand\r\n * @param z defines the z coordinate of the operand\r\n * @returns true if both vectors are equal\r\n */\r\n public equalsToFloats(x: number, y: number, z: number): boolean {\r\n return this._x === x && this._y === y && this._z === z;\r\n }\r\n\r\n /**\r\n * Multiplies the current Vector3 coordinates by the given ones\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#32\r\n * @param otherVector defines the second operand\r\n * @returns the current updated Vector3\r\n */\r\n public multiplyInPlace(otherVector: DeepImmutable): this {\r\n this._x *= otherVector._x;\r\n this._y *= otherVector._y;\r\n this._z *= otherVector._z;\r\n this._isDirty = true;\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns a new Vector3, result of the multiplication of the current Vector3 by the given vector\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#31\r\n * @param otherVector defines the second operand\r\n * @returns the new Vector3\r\n */\r\n public multiply(otherVector: DeepImmutable): this {\r\n return this.multiplyByFloats(otherVector._x, otherVector._y, otherVector._z);\r\n }\r\n\r\n /**\r\n * Multiplies the current Vector3 by the given one and stores the result in the given vector \"result\"\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#33\r\n * @param otherVector defines the second operand\r\n * @param result defines the Vector3 object where to store the result\r\n * @returns the result\r\n */\r\n public multiplyToRef(otherVector: DeepImmutable, result: T): T {\r\n return result.copyFromFloats(this._x * otherVector._x, this._y * otherVector._y, this._z * otherVector._z);\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 set with the result of the multiplication of the current Vector3 coordinates by the given floats\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#34\r\n * @param x defines the x coordinate of the operand\r\n * @param y defines the y coordinate of the operand\r\n * @param z defines the z coordinate of the operand\r\n * @returns the new Vector3\r\n */\r\n public multiplyByFloats(x: number, y: number, z: number): this {\r\n return new (this.constructor as Vector3Constructor)(this._x * x, this._y * y, this._z * z);\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 set with the result of the division of the current Vector3 coordinates by the given ones\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#16\r\n * @param otherVector defines the second operand\r\n * @returns the new Vector3\r\n */\r\n public divide(otherVector: DeepImmutable): this {\r\n return new (this.constructor as Vector3Constructor)(this._x / otherVector._x, this._y / otherVector._y, this._z / otherVector._z);\r\n }\r\n\r\n /**\r\n * Divides the current Vector3 coordinates by the given ones and stores the result in the given vector \"result\"\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#18\r\n * @param otherVector defines the second operand\r\n * @param result defines the Vector3 object where to store the result\r\n * @returns the result\r\n */\r\n public divideToRef(otherVector: DeepImmutable, result: T): T {\r\n return result.copyFromFloats(this._x / otherVector._x, this._y / otherVector._y, this._z / otherVector._z);\r\n }\r\n\r\n /**\r\n * Divides the current Vector3 coordinates by the given ones.\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#17\r\n * @param otherVector defines the second operand\r\n * @returns the current updated Vector3\r\n */\r\n public divideInPlace(otherVector: Vector3): this {\r\n return this.divideToRef(otherVector, this);\r\n }\r\n\r\n /**\r\n * Updates the current Vector3 with the minimal coordinate values between its and the given vector ones\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#29\r\n * @param other defines the second operand\r\n * @returns the current updated Vector3\r\n */\r\n public minimizeInPlace(other: DeepImmutable): this {\r\n return this.minimizeInPlaceFromFloats(other._x, other._y, other._z);\r\n }\r\n\r\n /**\r\n * Updates the current Vector3 with the maximal coordinate values between its and the given vector ones.\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#27\r\n * @param other defines the second operand\r\n * @returns the current updated Vector3\r\n */\r\n public maximizeInPlace(other: DeepImmutable): this {\r\n return this.maximizeInPlaceFromFloats(other._x, other._y, other._z);\r\n }\r\n\r\n /**\r\n * Updates the current Vector3 with the minimal coordinate values between its and the given coordinates\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#30\r\n * @param x defines the x coordinate of the operand\r\n * @param y defines the y coordinate of the operand\r\n * @param z defines the z coordinate of the operand\r\n * @returns the current updated Vector3\r\n */\r\n public minimizeInPlaceFromFloats(x: number, y: number, z: number): this {\r\n if (x < this._x) {\r\n this.x = x;\r\n }\r\n if (y < this._y) {\r\n this.y = y;\r\n }\r\n if (z < this._z) {\r\n this.z = z;\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Updates the current Vector3 with the maximal coordinate values between its and the given coordinates.\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#28\r\n * @param x defines the x coordinate of the operand\r\n * @param y defines the y coordinate of the operand\r\n * @param z defines the z coordinate of the operand\r\n * @returns the current updated Vector3\r\n */\r\n public maximizeInPlaceFromFloats(x: number, y: number, z: number): this {\r\n if (x > this._x) {\r\n this.x = x;\r\n }\r\n if (y > this._y) {\r\n this.y = y;\r\n }\r\n if (z > this._z) {\r\n this.z = z;\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Due to float precision, scale of a mesh could be uniform but float values are off by a small fraction\r\n * Check if is non uniform within a certain amount of decimal places to account for this\r\n * @param epsilon the amount the values can differ\r\n * @returns if the the vector is non uniform to a certain number of decimal places\r\n */\r\n public isNonUniformWithinEpsilon(epsilon: number) {\r\n const absX = Math.abs(this._x);\r\n const absY = Math.abs(this._y);\r\n if (!Scalar.WithinEpsilon(absX, absY, epsilon)) {\r\n return true;\r\n }\r\n\r\n const absZ = Math.abs(this._z);\r\n if (!Scalar.WithinEpsilon(absX, absZ, epsilon)) {\r\n return true;\r\n }\r\n\r\n if (!Scalar.WithinEpsilon(absY, absZ, epsilon)) {\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating that the vector is non uniform meaning x, y or z are not all the same\r\n */\r\n public get isNonUniform(): boolean {\r\n const absX = Math.abs(this._x);\r\n const absY = Math.abs(this._y);\r\n if (absX !== absY) {\r\n return true;\r\n }\r\n\r\n const absZ = Math.abs(this._z);\r\n if (absX !== absZ) {\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Gets a new Vector3 from current Vector3 floored values\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#22\r\n * @returns a new Vector3\r\n */\r\n public floor(): this {\r\n return new (this.constructor as Vector3Constructor)(Math.floor(this._x), Math.floor(this._y), Math.floor(this._z));\r\n }\r\n\r\n /**\r\n * Gets a new Vector3 from current Vector3 fractional values\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#23\r\n * @returns a new Vector3\r\n */\r\n public fract(): this {\r\n return new (this.constructor as Vector3Constructor)(this._x - Math.floor(this._x), this._y - Math.floor(this._y), this._z - Math.floor(this._z));\r\n }\r\n\r\n // Properties\r\n /**\r\n * Gets the length of the Vector3\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#25\r\n * @returns the length of the Vector3\r\n */\r\n public length(): number {\r\n return Math.sqrt(this._x * this._x + this._y * this._y + this._z * this._z);\r\n }\r\n\r\n /**\r\n * Gets the squared length of the Vector3\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#26\r\n * @returns squared length of the Vector3\r\n */\r\n public lengthSquared(): number {\r\n return this._x * this._x + this._y * this._y + this._z * this._z;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if the vector contains a zero in one of its components\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#1\r\n */\r\n public get hasAZeroComponent(): boolean {\r\n return this._x * this._y * this._z === 0;\r\n }\r\n\r\n /**\r\n * Normalize the current Vector3.\r\n * Please note that this is an in place operation.\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#122\r\n * @returns the current updated Vector3\r\n */\r\n public normalize(): this {\r\n return this.normalizeFromLength(this.length());\r\n }\r\n\r\n /**\r\n * Reorders the x y z properties of the vector in place\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#44\r\n * @param order new ordering of the properties (eg. for vector 1,2,3 with \"ZYX\" will produce 3,2,1)\r\n * @returns the current updated vector\r\n */\r\n public reorderInPlace(order: string) {\r\n order = order.toLowerCase();\r\n if (order === \"xyz\") {\r\n return this;\r\n }\r\n MathTmp.Vector3[0].copyFrom(this);\r\n [\"x\", \"y\", \"z\"].forEach((val, i) => {\r\n (this)[val] = (MathTmp.Vector3[0])[order[i]];\r\n });\r\n return this;\r\n }\r\n\r\n /**\r\n * Rotates the vector around 0,0,0 by a quaternion\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#47\r\n * @param quaternion the rotation quaternion\r\n * @param result vector to store the result\r\n * @returns the resulting vector\r\n */\r\n public rotateByQuaternionToRef(quaternion: Quaternion, result: T): T {\r\n quaternion.toRotationMatrix(MathTmp.Matrix[0]);\r\n Vector3.TransformCoordinatesToRef(this, MathTmp.Matrix[0], result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Rotates a vector around a given point\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#46\r\n * @param quaternion the rotation quaternion\r\n * @param point the point to rotate around\r\n * @param result vector to store the result\r\n * @returns the resulting vector\r\n */\r\n public rotateByQuaternionAroundPointToRef(quaternion: Quaternion, point: Vector3, result: T): T {\r\n this.subtractToRef(point, MathTmp.Vector3[0]);\r\n MathTmp.Vector3[0].rotateByQuaternionToRef(quaternion, MathTmp.Vector3[0]);\r\n point.addToRef(MathTmp.Vector3[0], result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 as the cross product of the current vector and the \"other\" one\r\n * The cross product is then orthogonal to both current and \"other\"\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#14\r\n * @param other defines the right operand\r\n * @returns the cross product\r\n */\r\n public cross(other: Vector3): this {\r\n const result = new (this.constructor as Vector3Constructor)();\r\n return Vector3.CrossToRef(this, other, result);\r\n }\r\n\r\n /**\r\n * Normalize the current Vector3 with the given input length.\r\n * Please note that this is an in place operation.\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#123\r\n * @param len the length of the vector\r\n * @returns the current updated Vector3\r\n */\r\n public normalizeFromLength(len: number): this {\r\n if (len === 0 || len === 1.0) {\r\n return this;\r\n }\r\n\r\n return this.scaleInPlace(1.0 / len);\r\n }\r\n\r\n /**\r\n * Normalize the current Vector3 to a new vector\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#124\r\n * @returns the new Vector3\r\n */\r\n public normalizeToNew(): this {\r\n const normalized = new (this.constructor as Vector3Constructor)(0, 0, 0);\r\n this.normalizeToRef(normalized);\r\n return normalized;\r\n }\r\n\r\n /**\r\n * Normalize the current Vector3 to the reference\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#125\r\n * @param reference define the Vector3 to update\r\n * @returns the updated Vector3\r\n */\r\n public normalizeToRef(reference: T): T {\r\n const len = this.length();\r\n if (len === 0 || len === 1.0) {\r\n return reference.copyFromFloats(this._x, this._y, this._z);\r\n }\r\n\r\n return this.scaleToRef(1.0 / len, reference);\r\n }\r\n\r\n /**\r\n * Creates a new Vector3 copied from the current Vector3\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#11\r\n * @returns the new Vector3\r\n */\r\n public clone(): this {\r\n return new (this.constructor as Vector3Constructor)(this._x, this._y, this._z);\r\n }\r\n\r\n /**\r\n * Copies the given vector coordinates to the current Vector3 ones\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#12\r\n * @param source defines the source Vector3\r\n * @returns the current updated Vector3\r\n */\r\n public copyFrom(source: DeepImmutable): this {\r\n return this.copyFromFloats(source._x, source._y, source._z);\r\n }\r\n\r\n /**\r\n * Copies the given floats to the current Vector3 coordinates\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#13\r\n * @param x defines the x coordinate of the operand\r\n * @param y defines the y coordinate of the operand\r\n * @param z defines the z coordinate of the operand\r\n * @returns the current updated Vector3\r\n */\r\n public copyFromFloats(x: number, y: number, z: number): this {\r\n this._x = x;\r\n this._y = y;\r\n this._z = z;\r\n this._isDirty = true;\r\n return this;\r\n }\r\n\r\n /**\r\n * Copies the given floats to the current Vector3 coordinates\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#58\r\n * @param x defines the x coordinate of the operand\r\n * @param y defines the y coordinate of the operand\r\n * @param z defines the z coordinate of the operand\r\n * @returns the current updated Vector3\r\n */\r\n public set(x: number, y: number, z: number): this {\r\n return this.copyFromFloats(x, y, z);\r\n }\r\n\r\n /**\r\n * Copies the given float to the current Vector3 coordinates\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#59\r\n * @param v defines the x, y and z coordinates of the operand\r\n * @returns the current updated Vector3\r\n */\r\n public setAll(v: number): this {\r\n this._x = this._y = this._z = v;\r\n this._isDirty = true;\r\n return this;\r\n }\r\n\r\n // Statics\r\n\r\n /**\r\n * Get the clip factor between two vectors\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#126\r\n * @param vector0 defines the first operand\r\n * @param vector1 defines the second operand\r\n * @param axis defines the axis to use\r\n * @param size defines the size along the axis\r\n * @returns the clip factor\r\n */\r\n public static GetClipFactor(vector0: DeepImmutable, vector1: DeepImmutable, axis: DeepImmutable, size: number): number {\r\n const d0 = Vector3.Dot(vector0, axis) - size;\r\n const d1 = Vector3.Dot(vector1, axis) - size;\r\n\r\n const s = d0 / (d0 - d1);\r\n\r\n return s;\r\n }\r\n\r\n /**\r\n * Get angle between two vectors\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#86\r\n * @param vector0 the starting point\r\n * @param vector1 the ending point\r\n * @param normal direction of the normal\r\n * @returns the angle between vector0 and vector1\r\n */\r\n public static GetAngleBetweenVectors(vector0: DeepImmutable, vector1: DeepImmutable, normal: DeepImmutable): number {\r\n const v0: Vector3 = vector0.normalizeToRef(MathTmp.Vector3[1]);\r\n const v1: Vector3 = vector1.normalizeToRef(MathTmp.Vector3[2]);\r\n let dot: number = Vector3.Dot(v0, v1);\r\n // Vectors are normalized so dot will be in [-1, 1] (aside precision issues enough to break the result which explains the below clamp)\r\n dot = Scalar.Clamp(dot, -1, 1);\r\n\r\n const angle = Math.acos(dot);\r\n const n = MathTmp.Vector3[3];\r\n Vector3.CrossToRef(v0, v1, n);\r\n if (Vector3.Dot(n, normal) > 0) {\r\n return isNaN(angle) ? 0 : angle;\r\n }\r\n return isNaN(angle) ? -Math.PI : -Math.acos(dot);\r\n }\r\n\r\n /**\r\n * Get angle between two vectors projected on a plane\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#87\r\n * Expectation compute time: 0.01 ms (median) and 0.02 ms (percentile 95%)\r\n * @param vector0 angle between vector0 and vector1\r\n * @param vector1 angle between vector0 and vector1\r\n * @param normal Normal of the projection plane\r\n * @returns the angle in radians (float) between vector0 and vector1 projected on the plane with the specified normal\r\n */\r\n public static GetAngleBetweenVectorsOnPlane(vector0: Vector3, vector1: Vector3, normal: Vector3): number {\r\n MathTmp.Vector3[0].copyFrom(vector0);\r\n const v0 = MathTmp.Vector3[0];\r\n MathTmp.Vector3[1].copyFrom(vector1);\r\n const v1 = MathTmp.Vector3[1];\r\n MathTmp.Vector3[2].copyFrom(normal);\r\n const vNormal = MathTmp.Vector3[2];\r\n const right = MathTmp.Vector3[3];\r\n const forward = MathTmp.Vector3[4];\r\n\r\n v0.normalize();\r\n v1.normalize();\r\n vNormal.normalize();\r\n\r\n Vector3.CrossToRef(vNormal, v0, right);\r\n Vector3.CrossToRef(right, vNormal, forward);\r\n\r\n const angle = Math.atan2(Vector3.Dot(v1, right), Vector3.Dot(v1, forward));\r\n\r\n return Scalar.NormalizeRadians(angle);\r\n }\r\n\r\n /**\r\n * Gets the rotation that aligns the roll axis (Y) to the line joining the start point to the target point and stores it in the ref Vector3\r\n * Example PG https://playground.babylonjs.com/#R1F8YU#189\r\n * @param start the starting point\r\n * @param target the target point\r\n * @param ref the vector3 to store the result\r\n * @returns ref in the form (pitch, yaw, 0)\r\n */\r\n public static PitchYawRollToMoveBetweenPointsToRef(start: Vector3, target: Vector3, ref: T): T {\r\n const diff = TmpVectors.Vector3[0];\r\n target.subtractToRef(start, diff);\r\n ref._y = Math.atan2(diff.x, diff.z) || 0;\r\n ref._x = Math.atan2(Math.sqrt(diff.x ** 2 + diff.z ** 2), diff.y) || 0;\r\n ref._z = 0;\r\n ref._isDirty = true;\r\n return ref;\r\n }\r\n\r\n /**\r\n * Gets the rotation that aligns the roll axis (Y) to the line joining the start point to the target point\r\n * Example PG https://playground.babylonjs.com/#R1F8YU#188\r\n * @param start the starting point\r\n * @param target the target point\r\n * @returns the rotation in the form (pitch, yaw, 0)\r\n */\r\n public static PitchYawRollToMoveBetweenPoints(start: Vector3, target: Vector3): Vector3 {\r\n const ref = Vector3.Zero();\r\n return Vector3.PitchYawRollToMoveBetweenPointsToRef(start, target, ref);\r\n }\r\n\r\n /**\r\n * Slerp between two vectors. See also `SmoothToRef`\r\n * Slerp is a spherical linear interpolation\r\n * giving a slow in and out effect\r\n * Example Playground 1 https://playground.babylonjs.com/#R1F8YU#108\r\n * Example Playground 2 https://playground.babylonjs.com/#R1F8YU#109\r\n * @param vector0 Start vector\r\n * @param vector1 End vector\r\n * @param slerp amount (will be clamped between 0 and 1)\r\n * @param result The slerped vector\r\n */\r\n public static SlerpToRef(vector0: Vector3, vector1: Vector3, slerp: number, result: T): T {\r\n slerp = Scalar.Clamp(slerp, 0, 1);\r\n const vector0Dir = MathTmp.Vector3[0];\r\n const vector1Dir = MathTmp.Vector3[1];\r\n\r\n vector0Dir.copyFrom(vector0);\r\n const vector0Length = vector0Dir.length();\r\n vector0Dir.normalizeFromLength(vector0Length);\r\n\r\n vector1Dir.copyFrom(vector1);\r\n const vector1Length = vector1Dir.length();\r\n vector1Dir.normalizeFromLength(vector1Length);\r\n\r\n const dot = Vector3.Dot(vector0Dir, vector1Dir);\r\n\r\n let scale0;\r\n let scale1;\r\n\r\n if (dot < 1 - Epsilon) {\r\n const omega = Math.acos(dot);\r\n const invSin = 1 / Math.sin(omega);\r\n scale0 = Math.sin((1 - slerp) * omega) * invSin;\r\n scale1 = Math.sin(slerp * omega) * invSin;\r\n } else {\r\n // Use linear interpolation\r\n scale0 = 1 - slerp;\r\n scale1 = slerp;\r\n }\r\n\r\n vector0Dir.scaleInPlace(scale0);\r\n vector1Dir.scaleInPlace(scale1);\r\n result.copyFrom(vector0Dir).addInPlace(vector1Dir);\r\n result.scaleInPlace(Scalar.Lerp(vector0Length, vector1Length, slerp));\r\n return result;\r\n }\r\n\r\n /**\r\n * Smooth interpolation between two vectors using Slerp\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#110\r\n * @param source source vector\r\n * @param goal goal vector\r\n * @param deltaTime current interpolation frame\r\n * @param lerpTime total interpolation time\r\n * @param result the smoothed vector\r\n */\r\n public static SmoothToRef(source: Vector3, goal: Vector3, deltaTime: number, lerpTime: number, result: T): T {\r\n Vector3.SlerpToRef(source, goal, lerpTime === 0 ? 1 : deltaTime / lerpTime, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 set from the index \"offset\" of the given array\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#83\r\n * @param array defines the source array\r\n * @param offset defines the offset in the source array\r\n * @returns the new Vector3\r\n */\r\n public static FromArray(array: DeepImmutable>, offset: number = 0): Vector3 {\r\n return new Vector3(array[offset], array[offset + 1], array[offset + 2]);\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 set from the index \"offset\" of the given Float32Array\r\n * @param array defines the source array\r\n * @param offset defines the offset in the source array\r\n * @returns the new Vector3\r\n * @deprecated Please use FromArray instead.\r\n */\r\n public static FromFloatArray(array: DeepImmutable, offset?: number): Vector3 {\r\n return Vector3.FromArray(array, offset);\r\n }\r\n\r\n /**\r\n * Sets the given vector \"result\" with the element values from the index \"offset\" of the given array\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#84\r\n * @param array defines the source array\r\n * @param offset defines the offset in the source array\r\n * @param result defines the Vector3 where to store the result\r\n * @returns result input\r\n */\r\n public static FromArrayToRef(array: DeepImmutable>, offset: number, result: T): T {\r\n result._x = array[offset];\r\n result._y = array[offset + 1];\r\n result._z = array[offset + 2];\r\n result._isDirty = true;\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the given vector \"result\" with the element values from the index \"offset\" of the given Float32Array\r\n * @param array defines the source array\r\n * @param offset defines the offset in the source array\r\n * @param result defines the Vector3 where to store the result\r\n * @deprecated Please use FromArrayToRef instead.\r\n */\r\n public static FromFloatArrayToRef(array: DeepImmutable, offset: number, result: T): T {\r\n return Vector3.FromArrayToRef(array, offset, result);\r\n }\r\n\r\n /**\r\n * Sets the given vector \"result\" with the given floats.\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#85\r\n * @param x defines the x coordinate of the source\r\n * @param y defines the y coordinate of the source\r\n * @param z defines the z coordinate of the source\r\n * @param result defines the Vector3 where to store the result\r\n */\r\n public static FromFloatsToRef(x: number, y: number, z: number, result: T): T {\r\n result.copyFromFloats(x, y, z);\r\n return result;\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 set to (0.0, 0.0, 0.0)\r\n * @returns a new empty Vector3\r\n */\r\n public static Zero(): Vector3 {\r\n return new Vector3(0.0, 0.0, 0.0);\r\n }\r\n /**\r\n * Returns a new Vector3 set to (1.0, 1.0, 1.0)\r\n * @returns a new Vector3\r\n */\r\n public static One(): Vector3 {\r\n return new Vector3(1.0, 1.0, 1.0);\r\n }\r\n /**\r\n * Returns a new Vector3 set to (0.0, 1.0, 0.0)\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#71\r\n * @returns a new up Vector3\r\n */\r\n public static Up(): Vector3 {\r\n return new Vector3(0.0, 1.0, 0.0);\r\n }\r\n\r\n /**\r\n * Gets an up Vector3 that must not be updated\r\n */\r\n public static get UpReadOnly(): DeepImmutable {\r\n return Vector3._UpReadOnly;\r\n }\r\n\r\n /**\r\n * Gets a down Vector3 that must not be updated\r\n */\r\n public static get DownReadOnly(): DeepImmutable {\r\n return Vector3._DownReadOnly;\r\n }\r\n\r\n /**\r\n * Gets a right Vector3 that must not be updated\r\n */\r\n public static get RightReadOnly(): DeepImmutable {\r\n return Vector3._RightReadOnly;\r\n }\r\n\r\n /**\r\n * Gets a left Vector3 that must not be updated\r\n */\r\n public static get LeftReadOnly(): DeepImmutable {\r\n return Vector3._LeftReadOnly;\r\n }\r\n\r\n /**\r\n * Gets a forward Vector3 that must not be updated\r\n */\r\n public static get LeftHandedForwardReadOnly(): DeepImmutable {\r\n return Vector3._LeftHandedForwardReadOnly;\r\n }\r\n\r\n /**\r\n * Gets a forward Vector3 that must not be updated\r\n */\r\n public static get RightHandedForwardReadOnly(): DeepImmutable {\r\n return Vector3._RightHandedForwardReadOnly;\r\n }\r\n\r\n /**\r\n * Gets a backward Vector3 that must not be updated\r\n */\r\n public static get LeftHandedBackwardReadOnly(): DeepImmutable {\r\n return Vector3._LeftHandedBackwardReadOnly;\r\n }\r\n\r\n /**\r\n * Gets a backward Vector3 that must not be updated\r\n */\r\n public static get RightHandedBackwardReadOnly(): DeepImmutable {\r\n return Vector3._RightHandedBackwardReadOnly;\r\n }\r\n\r\n /**\r\n * Gets a zero Vector3 that must not be updated\r\n */\r\n public static get ZeroReadOnly(): DeepImmutable {\r\n return Vector3._ZeroReadOnly;\r\n }\r\n\r\n /**\r\n * Gets a one Vector3 that must not be updated\r\n */\r\n public static get OneReadOnly(): DeepImmutable {\r\n return Vector3._OneReadOnly;\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 set to (0.0, -1.0, 0.0)\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#71\r\n * @returns a new down Vector3\r\n */\r\n public static Down(): Vector3 {\r\n return new Vector3(0.0, -1.0, 0.0);\r\n }\r\n /**\r\n * Returns a new Vector3 set to (0.0, 0.0, 1.0)\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#71\r\n * @param rightHandedSystem is the scene right-handed (negative z)\r\n * @returns a new forward Vector3\r\n */\r\n public static Forward(rightHandedSystem: boolean = false): Vector3 {\r\n return new Vector3(0.0, 0.0, rightHandedSystem ? -1.0 : 1.0);\r\n }\r\n /**\r\n * Returns a new Vector3 set to (0.0, 0.0, -1.0)\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#71\r\n * @param rightHandedSystem is the scene right-handed (negative-z)\r\n * @returns a new Backward Vector3\r\n */\r\n public static Backward(rightHandedSystem: boolean = false): Vector3 {\r\n return new Vector3(0.0, 0.0, rightHandedSystem ? 1.0 : -1.0);\r\n }\r\n /**\r\n * Returns a new Vector3 set to (1.0, 0.0, 0.0)\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#71\r\n * @returns a new right Vector3\r\n */\r\n public static Right(): Vector3 {\r\n return new Vector3(1.0, 0.0, 0.0);\r\n }\r\n /**\r\n * Returns a new Vector3 set to (-1.0, 0.0, 0.0)\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#71\r\n * @returns a new left Vector3\r\n */\r\n public static Left(): Vector3 {\r\n return new Vector3(-1.0, 0.0, 0.0);\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 with random values between min and max\r\n * @param min the minimum random value\r\n * @param max the maximum random value\r\n * @returns a Vector3 with random values between min and max\r\n */\r\n public static Random(min: number = 0, max: number = 1): Vector3 {\r\n return new Vector3(Scalar.RandomRange(min, max), Scalar.RandomRange(min, max), Scalar.RandomRange(min, max));\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 set with the result of the transformation by the given matrix of the given vector.\r\n * This method computes transformed coordinates only, not transformed direction vectors (ie. it takes translation in account)\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#111\r\n * @param vector defines the Vector3 to transform\r\n * @param transformation defines the transformation matrix\r\n * @returns the transformed Vector3\r\n */\r\n public static TransformCoordinates(vector: DeepImmutable, transformation: DeepImmutable): Vector3 {\r\n const result = Vector3.Zero();\r\n Vector3.TransformCoordinatesToRef(vector, transformation, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the given vector \"result\" coordinates with the result of the transformation by the given matrix of the given vector\r\n * This method computes transformed coordinates only, not transformed direction vectors (ie. it takes translation in account)\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#113\r\n * @param vector defines the Vector3 to transform\r\n * @param transformation defines the transformation matrix\r\n * @param result defines the Vector3 where to store the result\r\n * @returns result input\r\n */\r\n public static TransformCoordinatesToRef(vector: DeepImmutable, transformation: DeepImmutable, result: T): T {\r\n Vector3.TransformCoordinatesFromFloatsToRef(vector._x, vector._y, vector._z, transformation, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the given vector \"result\" coordinates with the result of the transformation by the given matrix of the given floats (x, y, z)\r\n * This method computes transformed coordinates only, not transformed direction vectors\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#115\r\n * @param x define the x coordinate of the source vector\r\n * @param y define the y coordinate of the source vector\r\n * @param z define the z coordinate of the source vector\r\n * @param transformation defines the transformation matrix\r\n * @param result defines the Vector3 where to store the result\r\n * @returns result input\r\n */\r\n public static TransformCoordinatesFromFloatsToRef(x: number, y: number, z: number, transformation: DeepImmutable, result: T): T {\r\n const m = transformation.m;\r\n const rx = x * m[0] + y * m[4] + z * m[8] + m[12];\r\n const ry = x * m[1] + y * m[5] + z * m[9] + m[13];\r\n const rz = x * m[2] + y * m[6] + z * m[10] + m[14];\r\n const rw = 1 / (x * m[3] + y * m[7] + z * m[11] + m[15]);\r\n\r\n result._x = rx * rw;\r\n result._y = ry * rw;\r\n result._z = rz * rw;\r\n result._isDirty = true;\r\n return result;\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 set with the result of the normal transformation by the given matrix of the given vector\r\n * This methods computes transformed normalized direction vectors only (ie. it does not apply translation)\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#112\r\n * @param vector defines the Vector3 to transform\r\n * @param transformation defines the transformation matrix\r\n * @returns the new Vector3\r\n */\r\n public static TransformNormal(vector: DeepImmutable, transformation: DeepImmutable): Vector3 {\r\n const result = Vector3.Zero();\r\n Vector3.TransformNormalToRef(vector, transformation, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the given vector \"result\" with the result of the normal transformation by the given matrix of the given vector\r\n * This methods computes transformed normalized direction vectors only (ie. it does not apply translation)\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#114\r\n * @param vector defines the Vector3 to transform\r\n * @param transformation defines the transformation matrix\r\n * @param result defines the Vector3 where to store the result\r\n * @returns result input\r\n */\r\n public static TransformNormalToRef(vector: DeepImmutable, transformation: DeepImmutable, result: T): T {\r\n this.TransformNormalFromFloatsToRef(vector._x, vector._y, vector._z, transformation, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the given vector \"result\" with the result of the normal transformation by the given matrix of the given floats (x, y, z)\r\n * This methods computes transformed normalized direction vectors only (ie. it does not apply translation)\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#116\r\n * @param x define the x coordinate of the source vector\r\n * @param y define the y coordinate of the source vector\r\n * @param z define the z coordinate of the source vector\r\n * @param transformation defines the transformation matrix\r\n * @param result defines the Vector3 where to store the result\r\n * @returns result input\r\n */\r\n public static TransformNormalFromFloatsToRef(x: number, y: number, z: number, transformation: DeepImmutable, result: T): T {\r\n const m = transformation.m;\r\n result._x = x * m[0] + y * m[4] + z * m[8];\r\n result._y = x * m[1] + y * m[5] + z * m[9];\r\n result._z = x * m[2] + y * m[6] + z * m[10];\r\n result._isDirty = true;\r\n return result;\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 located for \"amount\" on the CatmullRom interpolation spline defined by the vectors \"value1\", \"value2\", \"value3\", \"value4\"\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#69\r\n * @param value1 defines the first control point\r\n * @param value2 defines the second control point\r\n * @param value3 defines the third control point\r\n * @param value4 defines the fourth control point\r\n * @param amount defines the amount on the spline to use\r\n * @returns the new Vector3\r\n */\r\n public static CatmullRom(\r\n value1: DeepImmutable,\r\n value2: DeepImmutable,\r\n value3: DeepImmutable,\r\n value4: DeepImmutable,\r\n amount: number\r\n ): T {\r\n const squared = amount * amount;\r\n const cubed = amount * squared;\r\n\r\n const x =\r\n 0.5 *\r\n (2.0 * value2._x +\r\n (-value1._x + value3._x) * amount +\r\n (2.0 * value1._x - 5.0 * value2._x + 4.0 * value3._x - value4._x) * squared +\r\n (-value1._x + 3.0 * value2._x - 3.0 * value3._x + value4._x) * cubed);\r\n\r\n const y =\r\n 0.5 *\r\n (2.0 * value2._y +\r\n (-value1._y + value3._y) * amount +\r\n (2.0 * value1._y - 5.0 * value2._y + 4.0 * value3._y - value4._y) * squared +\r\n (-value1._y + 3.0 * value2._y - 3.0 * value3._y + value4._y) * cubed);\r\n\r\n const z =\r\n 0.5 *\r\n (2.0 * value2._z +\r\n (-value1._z + value3._z) * amount +\r\n (2.0 * value1._z - 5.0 * value2._z + 4.0 * value3._z - value4._z) * squared +\r\n (-value1._z + 3.0 * value2._z - 3.0 * value3._z + value4._z) * cubed);\r\n\r\n return new (value1.constructor as Vector3Constructor)(x, y, z);\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 set with the coordinates of \"value\", if the vector \"value\" is in the cube defined by the vectors \"min\" and \"max\"\r\n * If a coordinate value of \"value\" is lower than one of the \"min\" coordinate, then this \"value\" coordinate is set with the \"min\" one\r\n * If a coordinate value of \"value\" is greater than one of the \"max\" coordinate, then this \"value\" coordinate is set with the \"max\" one\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#76\r\n * @param value defines the current value\r\n * @param min defines the lower range value\r\n * @param max defines the upper range value\r\n * @returns the new Vector3\r\n */\r\n public static Clamp(value: DeepImmutable, min: DeepImmutable, max: DeepImmutable): T {\r\n const result = new (value.constructor as Vector3Constructor)();\r\n Vector3.ClampToRef(value, min, max, result);\r\n return result;\r\n }\r\n /**\r\n * Sets the given vector \"result\" with the coordinates of \"value\", if the vector \"value\" is in the cube defined by the vectors \"min\" and \"max\"\r\n * If a coordinate value of \"value\" is lower than one of the \"min\" coordinate, then this \"value\" coordinate is set with the \"min\" one\r\n * If a coordinate value of \"value\" is greater than one of the \"max\" coordinate, then this \"value\" coordinate is set with the \"max\" one\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#77\r\n * @param value defines the current value\r\n * @param min defines the lower range value\r\n * @param max defines the upper range value\r\n * @param result defines the Vector3 where to store the result\r\n * @returns result input\r\n */\r\n public static ClampToRef(value: DeepImmutable, min: DeepImmutable, max: DeepImmutable, result: T): T {\r\n let x = value._x;\r\n x = x > max._x ? max._x : x;\r\n x = x < min._x ? min._x : x;\r\n\r\n let y = value._y;\r\n y = y > max._y ? max._y : y;\r\n y = y < min._y ? min._y : y;\r\n\r\n let z = value._z;\r\n z = z > max._z ? max._z : z;\r\n z = z < min._z ? min._z : z;\r\n\r\n result.copyFromFloats(x, y, z);\r\n return result;\r\n }\r\n\r\n /**\r\n * Checks if a given vector is inside a specific range\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#75\r\n * @param v defines the vector to test\r\n * @param min defines the minimum range\r\n * @param max defines the maximum range\r\n */\r\n public static CheckExtends(v: Vector3, min: Vector3, max: Vector3): void {\r\n min.minimizeInPlace(v);\r\n max.maximizeInPlace(v);\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 located for \"amount\" (float) on the Hermite interpolation spline defined by the vectors \"value1\", \"tangent1\", \"value2\", \"tangent2\"\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#89\r\n * @param value1 defines the first control point\r\n * @param tangent1 defines the first tangent vector\r\n * @param value2 defines the second control point\r\n * @param tangent2 defines the second tangent vector\r\n * @param amount defines the amount on the interpolation spline (between 0 and 1)\r\n * @returns the new Vector3\r\n */\r\n public static Hermite(\r\n value1: DeepImmutable,\r\n tangent1: DeepImmutable,\r\n value2: DeepImmutable,\r\n tangent2: DeepImmutable,\r\n amount: number\r\n ): T {\r\n const squared = amount * amount;\r\n const cubed = amount * squared;\r\n const part1 = 2.0 * cubed - 3.0 * squared + 1.0;\r\n const part2 = -2.0 * cubed + 3.0 * squared;\r\n const part3 = cubed - 2.0 * squared + amount;\r\n const part4 = cubed - squared;\r\n\r\n const x = value1._x * part1 + value2._x * part2 + tangent1._x * part3 + tangent2._x * part4;\r\n const y = value1._y * part1 + value2._y * part2 + tangent1._y * part3 + tangent2._y * part4;\r\n const z = value1._z * part1 + value2._z * part2 + tangent1._z * part3 + tangent2._z * part4;\r\n return new (value1.constructor as Vector3Constructor)(x, y, z);\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 which is the 1st derivative of the Hermite spline defined by the vectors \"value1\", \"value2\", \"tangent1\", \"tangent2\".\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#90\r\n * @param value1 defines the first control point\r\n * @param tangent1 defines the first tangent\r\n * @param value2 defines the second control point\r\n * @param tangent2 defines the second tangent\r\n * @param time define where the derivative must be done\r\n * @returns 1st derivative\r\n */\r\n public static Hermite1stDerivative(\r\n value1: DeepImmutable,\r\n tangent1: DeepImmutable,\r\n value2: DeepImmutable,\r\n tangent2: DeepImmutable,\r\n time: number\r\n ): T {\r\n const result = new (value1.constructor as Vector3Constructor)();\r\n\r\n this.Hermite1stDerivativeToRef(value1, tangent1, value2, tangent2, time, result);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Update a Vector3 with the 1st derivative of the Hermite spline defined by the vectors \"value1\", \"value2\", \"tangent1\", \"tangent2\".\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#91\r\n * @param value1 defines the first control point\r\n * @param tangent1 defines the first tangent\r\n * @param value2 defines the second control point\r\n * @param tangent2 defines the second tangent\r\n * @param time define where the derivative must be done\r\n * @param result define where to store the derivative\r\n * @returns result input\r\n */\r\n public static Hermite1stDerivativeToRef(\r\n value1: DeepImmutable,\r\n tangent1: DeepImmutable,\r\n value2: DeepImmutable,\r\n tangent2: DeepImmutable,\r\n time: number,\r\n result: T\r\n ): T {\r\n const t2 = time * time;\r\n\r\n result._x = (t2 - time) * 6 * value1._x + (3 * t2 - 4 * time + 1) * tangent1._x + (-t2 + time) * 6 * value2._x + (3 * t2 - 2 * time) * tangent2._x;\r\n result._y = (t2 - time) * 6 * value1._y + (3 * t2 - 4 * time + 1) * tangent1._y + (-t2 + time) * 6 * value2._y + (3 * t2 - 2 * time) * tangent2._y;\r\n result._z = (t2 - time) * 6 * value1._z + (3 * t2 - 4 * time + 1) * tangent1._z + (-t2 + time) * 6 * value2._z + (3 * t2 - 2 * time) * tangent2._z;\r\n result._isDirty = true;\r\n return result;\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 located for \"amount\" (float) on the linear interpolation between the vectors \"start\" and \"end\"\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#95\r\n * @param start defines the start value\r\n * @param end defines the end value\r\n * @param amount max defines amount between both (between 0 and 1)\r\n * @returns the new Vector3\r\n */\r\n public static Lerp(start: DeepImmutable, end: DeepImmutable, amount: number): T {\r\n const result = new (start.constructor as Vector3Constructor)(0, 0, 0);\r\n Vector3.LerpToRef(start, end, amount, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the given vector \"result\" with the result of the linear interpolation from the vector \"start\" for \"amount\" to the vector \"end\"\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#93\r\n * @param start defines the start value\r\n * @param end defines the end value\r\n * @param amount max defines amount between both (between 0 and 1)\r\n * @param result defines the Vector3 where to store the result\r\n * @returns result input\r\n */\r\n public static LerpToRef(start: DeepImmutable, end: DeepImmutable, amount: number, result: T): T {\r\n result._x = start._x + (end._x - start._x) * amount;\r\n result._y = start._y + (end._y - start._y) * amount;\r\n result._z = start._z + (end._z - start._z) * amount;\r\n result._isDirty = true;\r\n return result;\r\n }\r\n\r\n /**\r\n * Returns the dot product (float) between the vectors \"left\" and \"right\"\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#82\r\n * @param left defines the left operand\r\n * @param right defines the right operand\r\n * @returns the dot product\r\n */\r\n public static Dot(left: DeepImmutable, right: DeepImmutable): number {\r\n return left._x * right._x + left._y * right._y + left._z * right._z;\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 as the cross product of the vectors \"left\" and \"right\"\r\n * The cross product is then orthogonal to both \"left\" and \"right\"\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#15\r\n * @param left defines the left operand\r\n * @param right defines the right operand\r\n * @returns the cross product\r\n */\r\n public static Cross(left: DeepImmutable, right: DeepImmutable): T {\r\n const result = new (left.constructor as Vector3Constructor)();\r\n Vector3.CrossToRef(left, right, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the given vector \"result\" with the cross product of \"left\" and \"right\"\r\n * The cross product is then orthogonal to both \"left\" and \"right\"\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#78\r\n * @param left defines the left operand\r\n * @param right defines the right operand\r\n * @param result defines the Vector3 where to store the result\r\n * @returns result input\r\n */\r\n public static CrossToRef(left: DeepImmutable, right: DeepImmutable, result: T): T {\r\n const x = left._y * right._z - left._z * right._y;\r\n const y = left._z * right._x - left._x * right._z;\r\n const z = left._x * right._y - left._y * right._x;\r\n result.copyFromFloats(x, y, z);\r\n return result;\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 as the normalization of the given vector\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#98\r\n * @param vector defines the Vector3 to normalize\r\n * @returns the new Vector3\r\n */\r\n public static Normalize(vector: DeepImmutable): Vector3 {\r\n const result = Vector3.Zero();\r\n Vector3.NormalizeToRef(vector, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the given vector \"result\" with the normalization of the given first vector\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#98\r\n * @param vector defines the Vector3 to normalize\r\n * @param result defines the Vector3 where to store the result\r\n * @returns result input\r\n */\r\n public static NormalizeToRef(vector: DeepImmutable, result: T): T {\r\n vector.normalizeToRef(result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Project a Vector3 onto screen space\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#101\r\n * @param vector defines the Vector3 to project\r\n * @param world defines the world matrix to use\r\n * @param transform defines the transform (view x projection) matrix to use\r\n * @param viewport defines the screen viewport to use\r\n * @returns the new Vector3\r\n */\r\n public static Project(vector: DeepImmutable, world: DeepImmutable, transform: DeepImmutable, viewport: DeepImmutable): T {\r\n const result = new (vector.constructor as Vector3Constructor)();\r\n Vector3.ProjectToRef(vector, world, transform, viewport, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Project a Vector3 onto screen space to reference\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#102\r\n * @param vector defines the Vector3 to project\r\n * @param world defines the world matrix to use\r\n * @param transform defines the transform (view x projection) matrix to use\r\n * @param viewport defines the screen viewport to use\r\n * @param result the vector in which the screen space will be stored\r\n * @returns result input\r\n */\r\n public static ProjectToRef(\r\n vector: DeepImmutable,\r\n world: DeepImmutable,\r\n transform: DeepImmutable,\r\n viewport: DeepImmutable,\r\n result: T\r\n ): T {\r\n const cw = viewport.width;\r\n const ch = viewport.height;\r\n const cx = viewport.x;\r\n const cy = viewport.y;\r\n\r\n const viewportMatrix = MathTmp.Matrix[1];\r\n\r\n Matrix.FromValuesToRef(cw / 2.0, 0, 0, 0, 0, -ch / 2.0, 0, 0, 0, 0, 0.5, 0, cx + cw / 2.0, ch / 2.0 + cy, 0.5, 1, viewportMatrix);\r\n\r\n const matrix = MathTmp.Matrix[0];\r\n world.multiplyToRef(transform, matrix);\r\n matrix.multiplyToRef(viewportMatrix, matrix);\r\n\r\n Vector3.TransformCoordinatesToRef(vector, matrix, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Reflects a vector off the plane defined by a normalized normal\r\n * @param inDirection defines the vector direction\r\n * @param normal defines the normal - Must be normalized\r\n * @returns the resulting vector\r\n */\r\n public static Reflect(inDirection: DeepImmutable, normal: DeepImmutable): Vector3 {\r\n return this.ReflectToRef(inDirection, normal, new Vector3());\r\n }\r\n\r\n /**\r\n * Reflects a vector off the plane defined by a normalized normal to reference\r\n * @param inDirection defines the vector direction\r\n * @param normal defines the normal - Must be normalized\r\n * @param result defines the Vector3 where to store the result\r\n * @returns the resulting vector\r\n */\r\n public static ReflectToRef(inDirection: DeepImmutable, normal: DeepImmutable, ref: T): T {\r\n const tmp = TmpVectors.Vector3[0];\r\n tmp.copyFrom(normal).scaleInPlace(2 * Vector3.Dot(inDirection, normal));\r\n\r\n return ref.copyFrom(inDirection).subtractInPlace(tmp);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _UnprojectFromInvertedMatrixToRef(source: DeepImmutable, matrix: DeepImmutable, result: T): T {\r\n Vector3.TransformCoordinatesToRef(source, matrix, result);\r\n const m = matrix.m;\r\n const num = source._x * m[3] + source._y * m[7] + source._z * m[11] + m[15];\r\n if (Scalar.WithinEpsilon(num, 1.0)) {\r\n result.scaleInPlace(1.0 / num);\r\n }\r\n return result;\r\n }\r\n\r\n /**\r\n * Unproject from screen space to object space\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#121\r\n * @param source defines the screen space Vector3 to use\r\n * @param viewportWidth defines the current width of the viewport\r\n * @param viewportHeight defines the current height of the viewport\r\n * @param world defines the world matrix to use (can be set to Identity to go to world space)\r\n * @param transform defines the transform (view x projection) matrix to use\r\n * @returns the new Vector3\r\n */\r\n public static UnprojectFromTransform(\r\n source: DeepImmutable,\r\n viewportWidth: number,\r\n viewportHeight: number,\r\n world: DeepImmutable,\r\n transform: DeepImmutable\r\n ): T {\r\n return this.Unproject(source, viewportWidth, viewportHeight, world, transform, Matrix.IdentityReadOnly);\r\n }\r\n\r\n /**\r\n * Unproject from screen space to object space\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#117\r\n * @param source defines the screen space Vector3 to use\r\n * @param viewportWidth defines the current width of the viewport\r\n * @param viewportHeight defines the current height of the viewport\r\n * @param world defines the world matrix to use (can be set to Identity to go to world space)\r\n * @param view defines the view matrix to use\r\n * @param projection defines the projection matrix to use\r\n * @returns the new Vector3\r\n */\r\n public static Unproject(\r\n source: DeepImmutable,\r\n viewportWidth: number,\r\n viewportHeight: number,\r\n world: DeepImmutable,\r\n view: DeepImmutable,\r\n projection: DeepImmutable\r\n ): T {\r\n const result = new (source.constructor as Vector3Constructor)();\r\n\r\n Vector3.UnprojectToRef(source, viewportWidth, viewportHeight, world, view, projection, result);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Unproject from screen space to object space\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#119\r\n * @param source defines the screen space Vector3 to use\r\n * @param viewportWidth defines the current width of the viewport\r\n * @param viewportHeight defines the current height of the viewport\r\n * @param world defines the world matrix to use (can be set to Identity to go to world space)\r\n * @param view defines the view matrix to use\r\n * @param projection defines the projection matrix to use\r\n * @param result defines the Vector3 where to store the result\r\n * @returns result input\r\n */\r\n public static UnprojectToRef(\r\n source: DeepImmutable,\r\n viewportWidth: number,\r\n viewportHeight: number,\r\n world: DeepImmutable,\r\n view: DeepImmutable,\r\n projection: DeepImmutable,\r\n result: T\r\n ): T {\r\n Vector3.UnprojectFloatsToRef(source._x, source._y, source._z, viewportWidth, viewportHeight, world, view, projection, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Unproject from screen space to object space\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#120\r\n * @param sourceX defines the screen space x coordinate to use\r\n * @param sourceY defines the screen space y coordinate to use\r\n * @param sourceZ defines the screen space z coordinate to use\r\n * @param viewportWidth defines the current width of the viewport\r\n * @param viewportHeight defines the current height of the viewport\r\n * @param world defines the world matrix to use (can be set to Identity to go to world space)\r\n * @param view defines the view matrix to use\r\n * @param projection defines the projection matrix to use\r\n * @param result defines the Vector3 where to store the result\r\n * @returns result input\r\n */\r\n public static UnprojectFloatsToRef(\r\n sourceX: float,\r\n sourceY: float,\r\n sourceZ: float,\r\n viewportWidth: number,\r\n viewportHeight: number,\r\n world: DeepImmutable,\r\n view: DeepImmutable,\r\n projection: DeepImmutable,\r\n result: T\r\n ): T {\r\n const matrix = MathTmp.Matrix[0];\r\n world.multiplyToRef(view, matrix);\r\n matrix.multiplyToRef(projection, matrix);\r\n matrix.invert();\r\n\r\n const screenSource = MathTmp.Vector3[0];\r\n screenSource.x = (sourceX / viewportWidth) * 2 - 1;\r\n screenSource.y = -((sourceY / viewportHeight) * 2 - 1);\r\n if (EngineStore.LastCreatedEngine?.isNDCHalfZRange) {\r\n screenSource.z = sourceZ;\r\n } else {\r\n screenSource.z = 2 * sourceZ - 1.0;\r\n }\r\n\r\n Vector3._UnprojectFromInvertedMatrixToRef(screenSource, matrix, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Gets the minimal coordinate values between two Vector3\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#97\r\n * @param left defines the first operand\r\n * @param right defines the second operand\r\n * @returns the new Vector3\r\n */\r\n public static Minimize(left: DeepImmutable, right: DeepImmutable): T {\r\n const min = new (left.constructor as Vector3Constructor)();\r\n min.copyFrom(left);\r\n min.minimizeInPlace(right);\r\n return min;\r\n }\r\n\r\n /**\r\n * Gets the maximal coordinate values between two Vector3\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#96\r\n * @param left defines the first operand\r\n * @param right defines the second operand\r\n * @returns the new Vector3\r\n */\r\n public static Maximize(left: DeepImmutable, right: DeepImmutable): T {\r\n const max = new (left.constructor as Vector3Constructor)();\r\n max.copyFrom(left);\r\n max.maximizeInPlace(right);\r\n return max;\r\n }\r\n\r\n /**\r\n * Returns the distance between the vectors \"value1\" and \"value2\"\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#81\r\n * @param value1 defines the first operand\r\n * @param value2 defines the second operand\r\n * @returns the distance\r\n */\r\n public static Distance(value1: DeepImmutable, value2: DeepImmutable): number {\r\n return Math.sqrt(Vector3.DistanceSquared(value1, value2));\r\n }\r\n\r\n /**\r\n * Returns the squared distance between the vectors \"value1\" and \"value2\"\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#80\r\n * @param value1 defines the first operand\r\n * @param value2 defines the second operand\r\n * @returns the squared distance\r\n */\r\n public static DistanceSquared(value1: DeepImmutable, value2: DeepImmutable): number {\r\n const x = value1._x - value2._x;\r\n const y = value1._y - value2._y;\r\n const z = value1._z - value2._z;\r\n\r\n return x * x + y * y + z * z;\r\n }\r\n\r\n /**\r\n * Projects \"vector\" on the triangle determined by its extremities \"p0\", \"p1\" and \"p2\", stores the result in \"ref\"\r\n * and returns the distance to the projected point.\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#104\r\n * From http://citeseerx.ist.psu.edu/viewdoc/download?doi=10.1.1.104.4264&rep=rep1&type=pdf\r\n *\r\n * @param vector the vector to get distance from\r\n * @param p0 extremity of the triangle\r\n * @param p1 extremity of the triangle\r\n * @param p2 extremity of the triangle\r\n * @param ref variable to store the result to\r\n * @returns The distance between \"ref\" and \"vector\"\r\n */\r\n public static ProjectOnTriangleToRef(vector: DeepImmutable, p0: DeepImmutable, p1: DeepImmutable, p2: DeepImmutable, ref: Vector3): number {\r\n const p1p0 = MathTmp.Vector3[0];\r\n const p2p0 = MathTmp.Vector3[1];\r\n const p2p1 = MathTmp.Vector3[2];\r\n const normal = MathTmp.Vector3[3];\r\n const vectorp0 = MathTmp.Vector3[4];\r\n\r\n // Triangle vectors\r\n p1.subtractToRef(p0, p1p0);\r\n p2.subtractToRef(p0, p2p0);\r\n p2.subtractToRef(p1, p2p1);\r\n\r\n const p1p0L = p1p0.length();\r\n const p2p0L = p2p0.length();\r\n const p2p1L = p2p1.length();\r\n\r\n if (p1p0L < Epsilon || p2p0L < Epsilon || p2p1L < Epsilon) {\r\n // This is a degenerate triangle. As we assume this is part of a non-degenerate mesh,\r\n // we will find a better intersection later.\r\n // Let's just return one of the extremities\r\n ref.copyFrom(p0);\r\n return Vector3.Distance(vector, p0);\r\n }\r\n\r\n // Compute normal and vector to p0\r\n vector.subtractToRef(p0, vectorp0);\r\n Vector3.CrossToRef(p1p0, p2p0, normal);\r\n const nl = normal.length();\r\n if (nl < Epsilon) {\r\n // Extremities are aligned, we are back on the case of a degenerate triangle\r\n ref.copyFrom(p0);\r\n return Vector3.Distance(vector, p0);\r\n }\r\n normal.normalizeFromLength(nl);\r\n let l = vectorp0.length();\r\n if (l < Epsilon) {\r\n // Vector is p0\r\n ref.copyFrom(p0);\r\n return 0;\r\n }\r\n vectorp0.normalizeFromLength(l);\r\n\r\n // Project to \"proj\" that lies on the triangle plane\r\n const cosA = Vector3.Dot(normal, vectorp0);\r\n const projVector = MathTmp.Vector3[5];\r\n const proj = MathTmp.Vector3[6];\r\n projVector.copyFrom(normal).scaleInPlace(-l * cosA);\r\n proj.copyFrom(vector).addInPlace(projVector);\r\n\r\n // Compute barycentric coordinates (v0, v1 and v2 are axis from barycenter to extremities)\r\n const v0 = MathTmp.Vector3[4];\r\n const v1 = MathTmp.Vector3[5];\r\n const v2 = MathTmp.Vector3[7];\r\n const tmp = MathTmp.Vector3[8];\r\n\r\n v0.copyFrom(p1p0).scaleInPlace(1 / p1p0L);\r\n tmp.copyFrom(p2p0).scaleInPlace(1 / p2p0L);\r\n v0.addInPlace(tmp).scaleInPlace(-1);\r\n\r\n v1.copyFrom(p1p0).scaleInPlace(-1 / p1p0L);\r\n tmp.copyFrom(p2p1).scaleInPlace(1 / p2p1L);\r\n v1.addInPlace(tmp).scaleInPlace(-1);\r\n\r\n v2.copyFrom(p2p1).scaleInPlace(-1 / p2p1L);\r\n tmp.copyFrom(p2p0).scaleInPlace(-1 / p2p0L);\r\n v2.addInPlace(tmp).scaleInPlace(-1);\r\n\r\n // Determines which edge of the triangle is closest to \"proj\"\r\n const projP = MathTmp.Vector3[9];\r\n let dot;\r\n projP.copyFrom(proj).subtractInPlace(p0);\r\n Vector3.CrossToRef(v0, projP, tmp);\r\n dot = Vector3.Dot(tmp, normal);\r\n const s0 = dot;\r\n\r\n projP.copyFrom(proj).subtractInPlace(p1);\r\n Vector3.CrossToRef(v1, projP, tmp);\r\n dot = Vector3.Dot(tmp, normal);\r\n const s1 = dot;\r\n\r\n projP.copyFrom(proj).subtractInPlace(p2);\r\n Vector3.CrossToRef(v2, projP, tmp);\r\n dot = Vector3.Dot(tmp, normal);\r\n const s2 = dot;\r\n\r\n const edge = MathTmp.Vector3[10];\r\n let e0, e1;\r\n if (s0 > 0 && s1 < 0) {\r\n edge.copyFrom(p1p0);\r\n e0 = p0;\r\n e1 = p1;\r\n } else if (s1 > 0 && s2 < 0) {\r\n edge.copyFrom(p2p1);\r\n e0 = p1;\r\n e1 = p2;\r\n } else {\r\n edge.copyFrom(p2p0).scaleInPlace(-1);\r\n e0 = p2;\r\n e1 = p0;\r\n }\r\n\r\n // Determines if \"proj\" lies inside the triangle\r\n const tmp2 = MathTmp.Vector3[9];\r\n const tmp3 = MathTmp.Vector3[4];\r\n e0.subtractToRef(proj, tmp);\r\n e1.subtractToRef(proj, tmp2);\r\n Vector3.CrossToRef(tmp, tmp2, tmp3);\r\n const isOutside = Vector3.Dot(tmp3, normal) < 0;\r\n\r\n // If inside, we already found the projected point, \"proj\"\r\n if (!isOutside) {\r\n ref.copyFrom(proj);\r\n return Math.abs(l * cosA);\r\n }\r\n\r\n // If outside, we find \"triProj\", the closest point from \"proj\" on the closest edge\r\n const r = MathTmp.Vector3[5];\r\n Vector3.CrossToRef(edge, tmp3, r);\r\n r.normalize();\r\n const e0proj = MathTmp.Vector3[9];\r\n e0proj.copyFrom(e0).subtractInPlace(proj);\r\n const e0projL = e0proj.length();\r\n if (e0projL < Epsilon) {\r\n // Proj is e0\r\n ref.copyFrom(e0);\r\n return Vector3.Distance(vector, e0);\r\n }\r\n e0proj.normalizeFromLength(e0projL);\r\n const cosG = Vector3.Dot(r, e0proj);\r\n const triProj = MathTmp.Vector3[7];\r\n triProj.copyFrom(proj).addInPlace(r.scaleInPlace(e0projL * cosG));\r\n\r\n // Now we clamp \"triProj\" so it lies between e0 and e1\r\n tmp.copyFrom(triProj).subtractInPlace(e0);\r\n l = edge.length();\r\n edge.normalizeFromLength(l);\r\n let t = Vector3.Dot(tmp, edge) / Math.max(l, Epsilon);\r\n t = Scalar.Clamp(t, 0, 1);\r\n triProj.copyFrom(e0).addInPlace(edge.scaleInPlace(t * l));\r\n ref.copyFrom(triProj);\r\n\r\n return Vector3.Distance(vector, triProj);\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 located at the center between \"value1\" and \"value2\"\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#72\r\n * @param value1 defines the first operand\r\n * @param value2 defines the second operand\r\n * @returns the new Vector3\r\n */\r\n public static Center(value1: DeepImmutable, value2: DeepImmutable): Vector3 {\r\n return Vector3.CenterToRef(value1, value2, Vector3.Zero());\r\n }\r\n\r\n /**\r\n * Gets the center of the vectors \"value1\" and \"value2\" and stores the result in the vector \"ref\"\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#73\r\n * @param value1 defines first vector\r\n * @param value2 defines second vector\r\n * @param ref defines third vector\r\n * @returns ref\r\n */\r\n public static CenterToRef(value1: DeepImmutable, value2: DeepImmutable, ref: T): T {\r\n return ref.copyFromFloats((value1._x + value2._x) / 2, (value1._y + value2._y) / 2, (value1._z + value2._z) / 2);\r\n }\r\n\r\n /**\r\n * Given three orthogonal normalized left-handed oriented Vector3 axis in space (target system),\r\n * RotationFromAxis() returns the rotation Euler angles (ex : rotation.x, rotation.y, rotation.z) to apply\r\n * to something in order to rotate it from its local system to the given target system\r\n * Note: axis1, axis2 and axis3 are normalized during this operation\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#106\r\n * @param axis1 defines the first axis\r\n * @param axis2 defines the second axis\r\n * @param axis3 defines the third axis\r\n * @returns a new Vector3\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/center_origin/target_align\r\n */\r\n public static RotationFromAxis(axis1: DeepImmutable, axis2: DeepImmutable, axis3: DeepImmutable): T {\r\n const rotation = new (axis1.constructor as Vector3Constructor)();\r\n Vector3.RotationFromAxisToRef(axis1, axis2, axis3, rotation);\r\n return rotation;\r\n }\r\n\r\n /**\r\n * The same than RotationFromAxis but updates the given ref Vector3 parameter instead of returning a new Vector3\r\n * Example Playground https://playground.babylonjs.com/#R1F8YU#107\r\n * @param axis1 defines the first axis\r\n * @param axis2 defines the second axis\r\n * @param axis3 defines the third axis\r\n * @param ref defines the Vector3 where to store the result\r\n * @returns result input\r\n */\r\n public static RotationFromAxisToRef(axis1: DeepImmutable, axis2: DeepImmutable, axis3: DeepImmutable, ref: T): T {\r\n const quat = MathTmp.Quaternion[0];\r\n Quaternion.RotationQuaternionFromAxisToRef(axis1, axis2, axis3, quat);\r\n quat.toEulerAnglesToRef(ref);\r\n return ref;\r\n }\r\n}\r\n\r\n/**\r\n * Vector4 class created for EulerAngle class conversion to Quaternion\r\n */\r\nexport class Vector4 {\r\n private static _ZeroReadOnly = Vector4.Zero() as DeepImmutable;\r\n\r\n /**\r\n * Creates a Vector4 object from the given floats.\r\n * @param x x value of the vector\r\n * @param y y value of the vector\r\n * @param z z value of the vector\r\n * @param w w value of the vector\r\n */\r\n constructor(\r\n /** x value of the vector */\r\n public x: number = 0,\r\n /** y value of the vector */\r\n public y: number = 0,\r\n /** z value of the vector */\r\n public z: number = 0,\r\n /** w value of the vector */\r\n public w: number = 0\r\n ) {}\r\n\r\n /**\r\n * Returns the string with the Vector4 coordinates.\r\n * @returns a string containing all the vector values\r\n */\r\n public toString(): string {\r\n return `{X: ${this.x} Y: ${this.y} Z: ${this.z} W: ${this.w}}`;\r\n }\r\n\r\n /**\r\n * Returns the string \"Vector4\".\r\n * @returns \"Vector4\"\r\n */\r\n public getClassName(): string {\r\n return \"Vector4\";\r\n }\r\n\r\n /**\r\n * Returns the Vector4 hash code.\r\n * @returns a unique hash code\r\n */\r\n public getHashCode(): number {\r\n const x = _ExtractAsInt(this.x);\r\n const y = _ExtractAsInt(this.y);\r\n const z = _ExtractAsInt(this.z);\r\n const w = _ExtractAsInt(this.w);\r\n\r\n let hash = x;\r\n hash = (hash * 397) ^ y;\r\n hash = (hash * 397) ^ z;\r\n hash = (hash * 397) ^ w;\r\n return hash;\r\n }\r\n\r\n // Operators\r\n /**\r\n * Returns a new array populated with 4 elements : the Vector4 coordinates.\r\n * @returns the resulting array\r\n */\r\n public asArray(): number[] {\r\n const result = new Array();\r\n\r\n this.toArray(result, 0);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Populates the given array from the given index with the Vector4 coordinates.\r\n * @param array array to populate\r\n * @param index index of the array to start at (default: 0)\r\n * @returns the Vector4.\r\n */\r\n public toArray(array: FloatArray, index?: number): this {\r\n if (index === undefined) {\r\n index = 0;\r\n }\r\n array[index] = this.x;\r\n array[index + 1] = this.y;\r\n array[index + 2] = this.z;\r\n array[index + 3] = this.w;\r\n return this;\r\n }\r\n\r\n /**\r\n * Update the current vector from an array\r\n * @param array defines the destination array\r\n * @param index defines the offset in the destination array\r\n * @returns the current Vector3\r\n */\r\n public fromArray(array: FloatArray, index: number = 0): this {\r\n Vector4.FromArrayToRef(array, index, this);\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds the given vector to the current Vector4.\r\n * @param otherVector the vector to add\r\n * @returns the updated Vector4.\r\n */\r\n public addInPlace(otherVector: DeepImmutable): this {\r\n this.x += otherVector.x;\r\n this.y += otherVector.y;\r\n this.z += otherVector.z;\r\n this.w += otherVector.w;\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns a new Vector4 as the result of the addition of the current Vector4 and the given one.\r\n * @param otherVector the vector to add\r\n * @returns the resulting vector\r\n */\r\n public add(otherVector: DeepImmutable): this {\r\n return new (this.constructor as Vector4Constructor)(this.x + otherVector.x, this.y + otherVector.y, this.z + otherVector.z, this.w + otherVector.w);\r\n }\r\n\r\n /**\r\n * Updates the given vector \"result\" with the result of the addition of the current Vector4 and the given one.\r\n * @param otherVector the vector to add\r\n * @param result the vector to store the result\r\n * @returns result input\r\n */\r\n public addToRef(otherVector: DeepImmutable, result: T): T {\r\n result.x = this.x + otherVector.x;\r\n result.y = this.y + otherVector.y;\r\n result.z = this.z + otherVector.z;\r\n result.w = this.w + otherVector.w;\r\n return result;\r\n }\r\n\r\n /**\r\n * Subtract in place the given vector from the current Vector4.\r\n * @param otherVector the vector to subtract\r\n * @returns the updated Vector4.\r\n */\r\n public subtractInPlace(otherVector: DeepImmutable): this {\r\n this.x -= otherVector.x;\r\n this.y -= otherVector.y;\r\n this.z -= otherVector.z;\r\n this.w -= otherVector.w;\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns a new Vector4 with the result of the subtraction of the given vector from the current Vector4.\r\n * @param otherVector the vector to add\r\n * @returns the new vector with the result\r\n */\r\n public subtract(otherVector: DeepImmutable): this {\r\n return new (this.constructor as Vector4Constructor)(this.x - otherVector.x, this.y - otherVector.y, this.z - otherVector.z, this.w - otherVector.w);\r\n }\r\n\r\n /**\r\n * Sets the given vector \"result\" with the result of the subtraction of the given vector from the current Vector4.\r\n * @param otherVector the vector to subtract\r\n * @param result the vector to store the result\r\n * @returns result input\r\n */\r\n public subtractToRef(otherVector: DeepImmutable, result: T): T {\r\n result.x = this.x - otherVector.x;\r\n result.y = this.y - otherVector.y;\r\n result.z = this.z - otherVector.z;\r\n result.w = this.w - otherVector.w;\r\n return result;\r\n }\r\n\r\n /**\r\n * Returns a new Vector4 set with the result of the subtraction of the given floats from the current Vector4 coordinates.\r\n */\r\n /**\r\n * Returns a new Vector4 set with the result of the subtraction of the given floats from the current Vector4 coordinates.\r\n * @param x value to subtract\r\n * @param y value to subtract\r\n * @param z value to subtract\r\n * @param w value to subtract\r\n * @returns new vector containing the result\r\n */\r\n public subtractFromFloats(x: number, y: number, z: number, w: number): this {\r\n return new (this.constructor as Vector4Constructor)(this.x - x, this.y - y, this.z - z, this.w - w);\r\n }\r\n\r\n /**\r\n * Sets the given vector \"result\" set with the result of the subtraction of the given floats from the current Vector4 coordinates.\r\n * @param x value to subtract\r\n * @param y value to subtract\r\n * @param z value to subtract\r\n * @param w value to subtract\r\n * @param result the vector to store the result in\r\n * @returns result input\r\n */\r\n public subtractFromFloatsToRef(x: number, y: number, z: number, w: number, result: T): T {\r\n result.x = this.x - x;\r\n result.y = this.y - y;\r\n result.z = this.z - z;\r\n result.w = this.w - w;\r\n return result;\r\n }\r\n\r\n /**\r\n * Returns a new Vector4 set with the current Vector4 negated coordinates.\r\n * @returns a new vector with the negated values\r\n */\r\n public negate(): this {\r\n return new (this.constructor as Vector4Constructor)(-this.x, -this.y, -this.z, -this.w);\r\n }\r\n\r\n /**\r\n * Negate this vector in place\r\n * @returns this\r\n */\r\n public negateInPlace(): this {\r\n this.x *= -1;\r\n this.y *= -1;\r\n this.z *= -1;\r\n this.w *= -1;\r\n return this;\r\n }\r\n\r\n /**\r\n * Negate the current Vector4 and stores the result in the given vector \"result\" coordinates\r\n * @param result defines the Vector3 object where to store the result\r\n * @returns the result\r\n */\r\n public negateToRef(result: T): T {\r\n return result.copyFromFloats(this.x * -1, this.y * -1, this.z * -1, this.w * -1);\r\n }\r\n\r\n /**\r\n * Multiplies the current Vector4 coordinates by scale (float).\r\n * @param scale the number to scale with\r\n * @returns the updated Vector4.\r\n */\r\n public scaleInPlace(scale: number): this {\r\n this.x *= scale;\r\n this.y *= scale;\r\n this.z *= scale;\r\n this.w *= scale;\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns a new Vector4 set with the current Vector4 coordinates multiplied by scale (float).\r\n * @param scale the number to scale with\r\n * @returns a new vector with the result\r\n */\r\n public scale(scale: number): this {\r\n return new (this.constructor as Vector4Constructor)(this.x * scale, this.y * scale, this.z * scale, this.w * scale);\r\n }\r\n\r\n /**\r\n * Sets the given vector \"result\" with the current Vector4 coordinates multiplied by scale (float).\r\n * @param scale the number to scale with\r\n * @param result a vector to store the result in\r\n * @returns result input\r\n */\r\n public scaleToRef(scale: number, result: T): T {\r\n result.x = this.x * scale;\r\n result.y = this.y * scale;\r\n result.z = this.z * scale;\r\n result.w = this.w * scale;\r\n return result;\r\n }\r\n\r\n /**\r\n * Scale the current Vector4 values by a factor and add the result to a given Vector4\r\n * @param scale defines the scale factor\r\n * @param result defines the Vector4 object where to store the result\r\n * @returns result input\r\n */\r\n public scaleAndAddToRef(scale: number, result: T): T {\r\n result.x += this.x * scale;\r\n result.y += this.y * scale;\r\n result.z += this.z * scale;\r\n result.w += this.w * scale;\r\n return result;\r\n }\r\n\r\n /**\r\n * Boolean : True if the current Vector4 coordinates are stricly equal to the given ones.\r\n * @param otherVector the vector to compare against\r\n * @returns true if they are equal\r\n */\r\n public equals(otherVector: DeepImmutable): boolean {\r\n return otherVector && this.x === otherVector.x && this.y === otherVector.y && this.z === otherVector.z && this.w === otherVector.w;\r\n }\r\n\r\n /**\r\n * Boolean : True if the current Vector4 coordinates are each beneath the distance \"epsilon\" from the given vector ones.\r\n * @param otherVector vector to compare against\r\n * @param epsilon (Default: very small number)\r\n * @returns true if they are equal\r\n */\r\n public equalsWithEpsilon(otherVector: DeepImmutable, epsilon: number = Epsilon): boolean {\r\n return (\r\n otherVector &&\r\n Scalar.WithinEpsilon(this.x, otherVector.x, epsilon) &&\r\n Scalar.WithinEpsilon(this.y, otherVector.y, epsilon) &&\r\n Scalar.WithinEpsilon(this.z, otherVector.z, epsilon) &&\r\n Scalar.WithinEpsilon(this.w, otherVector.w, epsilon)\r\n );\r\n }\r\n\r\n /**\r\n * Boolean : True if the given floats are strictly equal to the current Vector4 coordinates.\r\n * @param x x value to compare against\r\n * @param y y value to compare against\r\n * @param z z value to compare against\r\n * @param w w value to compare against\r\n * @returns true if equal\r\n */\r\n public equalsToFloats(x: number, y: number, z: number, w: number): boolean {\r\n return this.x === x && this.y === y && this.z === z && this.w === w;\r\n }\r\n\r\n /**\r\n * Multiplies in place the current Vector4 by the given one.\r\n * @param otherVector vector to multiple with\r\n * @returns the updated Vector4.\r\n */\r\n public multiplyInPlace(otherVector: Vector4): this {\r\n this.x *= otherVector.x;\r\n this.y *= otherVector.y;\r\n this.z *= otherVector.z;\r\n this.w *= otherVector.w;\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns a new Vector4 set with the multiplication result of the current Vector4 and the given one.\r\n * @param otherVector vector to multiple with\r\n * @returns resulting new vector\r\n */\r\n public multiply(otherVector: DeepImmutable): this {\r\n return new (this.constructor as Vector4Constructor)(this.x * otherVector.x, this.y * otherVector.y, this.z * otherVector.z, this.w * otherVector.w);\r\n }\r\n /**\r\n * Updates the given vector \"result\" with the multiplication result of the current Vector4 and the given one.\r\n * @param otherVector vector to multiple with\r\n * @param result vector to store the result\r\n * @returns result input\r\n */\r\n public multiplyToRef(otherVector: DeepImmutable, result: T): T {\r\n result.x = this.x * otherVector.x;\r\n result.y = this.y * otherVector.y;\r\n result.z = this.z * otherVector.z;\r\n result.w = this.w * otherVector.w;\r\n return result;\r\n }\r\n /**\r\n * Returns a new Vector4 set with the multiplication result of the given floats and the current Vector4 coordinates.\r\n * @param x x value multiply with\r\n * @param y y value multiply with\r\n * @param z z value multiply with\r\n * @param w w value multiply with\r\n * @returns resulting new vector\r\n */\r\n public multiplyByFloats(x: number, y: number, z: number, w: number): this {\r\n return new (this.constructor as Vector4Constructor)(this.x * x, this.y * y, this.z * z, this.w * w);\r\n }\r\n /**\r\n * Returns a new Vector4 set with the division result of the current Vector4 by the given one.\r\n * @param otherVector vector to devide with\r\n * @returns resulting new vector\r\n */\r\n public divide(otherVector: DeepImmutable): this {\r\n return new (this.constructor as Vector4Constructor)(this.x / otherVector.x, this.y / otherVector.y, this.z / otherVector.z, this.w / otherVector.w);\r\n }\r\n /**\r\n * Updates the given vector \"result\" with the division result of the current Vector4 by the given one.\r\n * @param otherVector vector to devide with\r\n * @param result vector to store the result\r\n * @returns result input\r\n */\r\n public divideToRef(otherVector: DeepImmutable, result: T): T {\r\n result.x = this.x / otherVector.x;\r\n result.y = this.y / otherVector.y;\r\n result.z = this.z / otherVector.z;\r\n result.w = this.w / otherVector.w;\r\n return result;\r\n }\r\n\r\n /**\r\n * Divides the current Vector3 coordinates by the given ones.\r\n * @param otherVector vector to devide with\r\n * @returns the updated Vector3.\r\n */\r\n public divideInPlace(otherVector: DeepImmutable): this {\r\n return this.divideToRef(otherVector, this);\r\n }\r\n\r\n /**\r\n * Updates the Vector4 coordinates with the minimum values between its own and the given vector ones\r\n * @param other defines the second operand\r\n * @returns the current updated Vector4\r\n */\r\n public minimizeInPlace(other: DeepImmutable): this {\r\n if (other.x < this.x) {\r\n this.x = other.x;\r\n }\r\n if (other.y < this.y) {\r\n this.y = other.y;\r\n }\r\n if (other.z < this.z) {\r\n this.z = other.z;\r\n }\r\n if (other.w < this.w) {\r\n this.w = other.w;\r\n }\r\n return this;\r\n }\r\n /**\r\n * Updates the Vector4 coordinates with the maximum values between its own and the given vector ones\r\n * @param other defines the second operand\r\n * @returns the current updated Vector4\r\n */\r\n public maximizeInPlace(other: DeepImmutable): this {\r\n if (other.x > this.x) {\r\n this.x = other.x;\r\n }\r\n if (other.y > this.y) {\r\n this.y = other.y;\r\n }\r\n if (other.z > this.z) {\r\n this.z = other.z;\r\n }\r\n if (other.w > this.w) {\r\n this.w = other.w;\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Gets a new Vector4 from current Vector4 floored values\r\n * @returns a new Vector4\r\n */\r\n public floor(): this {\r\n return new (this.constructor as Vector4Constructor)(Math.floor(this.x), Math.floor(this.y), Math.floor(this.z), Math.floor(this.w));\r\n }\r\n\r\n /**\r\n * Gets a new Vector4 from current Vector4 fractional values\r\n * @returns a new Vector4\r\n */\r\n public fract(): this {\r\n return new (this.constructor as Vector4Constructor)(\r\n this.x - Math.floor(this.x),\r\n this.y - Math.floor(this.y),\r\n this.z - Math.floor(this.z),\r\n this.w - Math.floor(this.w)\r\n );\r\n }\r\n\r\n // Properties\r\n /**\r\n * Returns the Vector4 length (float).\r\n * @returns the length\r\n */\r\n public length(): number {\r\n return Math.sqrt(this.x * this.x + this.y * this.y + this.z * this.z + this.w * this.w);\r\n }\r\n /**\r\n * Returns the Vector4 squared length (float).\r\n * @returns the length squared\r\n */\r\n public lengthSquared(): number {\r\n return this.x * this.x + this.y * this.y + this.z * this.z + this.w * this.w;\r\n }\r\n\r\n // Methods\r\n /**\r\n * Normalizes in place the Vector4.\r\n * @returns the updated Vector4.\r\n */\r\n public normalize(): this {\r\n const len = this.length();\r\n\r\n if (len === 0) {\r\n return this;\r\n }\r\n\r\n return this.scaleInPlace(1.0 / len);\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 from the Vector4 (x, y, z) coordinates.\r\n * @returns this converted to a new vector3\r\n */\r\n public toVector3(): Vector3 {\r\n return new Vector3(this.x, this.y, this.z);\r\n }\r\n /**\r\n * Returns a new Vector4 copied from the current one.\r\n * @returns the new cloned vector\r\n */\r\n public clone(): this {\r\n return new (this.constructor as Vector4Constructor)(this.x, this.y, this.z, this.w);\r\n }\r\n /**\r\n * Updates the current Vector4 with the given one coordinates.\r\n * @param source the source vector to copy from\r\n * @returns the updated Vector4.\r\n */\r\n public copyFrom(source: DeepImmutable): this {\r\n this.x = source.x;\r\n this.y = source.y;\r\n this.z = source.z;\r\n this.w = source.w;\r\n return this;\r\n }\r\n /**\r\n * Updates the current Vector4 coordinates with the given floats.\r\n * @param x float to copy from\r\n * @param y float to copy from\r\n * @param z float to copy from\r\n * @param w float to copy from\r\n * @returns the updated Vector4.\r\n */\r\n public copyFromFloats(x: number, y: number, z: number, w: number): this {\r\n this.x = x;\r\n this.y = y;\r\n this.z = z;\r\n this.w = w;\r\n return this;\r\n }\r\n /**\r\n * Updates the current Vector4 coordinates with the given floats.\r\n * @param x float to set from\r\n * @param y float to set from\r\n * @param z float to set from\r\n * @param w float to set from\r\n * @returns the updated Vector4.\r\n */\r\n public set(x: number, y: number, z: number, w: number): this {\r\n return this.copyFromFloats(x, y, z, w);\r\n }\r\n\r\n /**\r\n * Copies the given float to the current Vector3 coordinates\r\n * @param v defines the x, y, z and w coordinates of the operand\r\n * @returns the current updated Vector3\r\n */\r\n public setAll(v: number): this {\r\n this.x = this.y = this.z = this.w = v;\r\n return this;\r\n }\r\n\r\n // Statics\r\n /**\r\n * Returns a new Vector4 set from the starting index of the given array.\r\n * @param array the array to pull values from\r\n * @param offset the offset into the array to start at\r\n * @returns the new vector\r\n */\r\n public static FromArray(array: DeepImmutable>, offset?: number): Vector4 {\r\n if (!offset) {\r\n offset = 0;\r\n }\r\n return new Vector4(array[offset], array[offset + 1], array[offset + 2], array[offset + 3]);\r\n }\r\n /**\r\n * Updates the given vector \"result\" from the starting index of the given array.\r\n * @param array the array to pull values from\r\n * @param offset the offset into the array to start at\r\n * @param result the vector to store the result in\r\n * @returns result input\r\n */\r\n public static FromArrayToRef(array: DeepImmutable>, offset: number, result: T): T {\r\n result.x = array[offset];\r\n result.y = array[offset + 1];\r\n result.z = array[offset + 2];\r\n result.w = array[offset + 3];\r\n return result;\r\n }\r\n /**\r\n * Updates the given vector \"result\" from the starting index of the given Float32Array.\r\n * @param array the array to pull values from\r\n * @param offset the offset into the array to start at\r\n * @param result the vector to store the result in\r\n * @returns result input\r\n */\r\n public static FromFloatArrayToRef(array: DeepImmutable, offset: number, result: T): T {\r\n Vector4.FromArrayToRef(array, offset, result);\r\n return result;\r\n }\r\n /**\r\n * Updates the given vector \"result\" coordinates from the given floats.\r\n * @param x float to set from\r\n * @param y float to set from\r\n * @param z float to set from\r\n * @param w float to set from\r\n * @param result the vector to the floats in\r\n * @returns result input\r\n */\r\n public static FromFloatsToRef(x: number, y: number, z: number, w: number, result: T): T {\r\n result.x = x;\r\n result.y = y;\r\n result.z = z;\r\n result.w = w;\r\n return result;\r\n }\r\n /**\r\n * Returns a new Vector4 set to (0.0, 0.0, 0.0, 0.0)\r\n * @returns the new vector\r\n */\r\n public static Zero(): Vector4 {\r\n return new Vector4(0.0, 0.0, 0.0, 0.0);\r\n }\r\n /**\r\n * Returns a new Vector4 set to (1.0, 1.0, 1.0, 1.0)\r\n * @returns the new vector\r\n */\r\n public static One(): Vector4 {\r\n return new Vector4(1.0, 1.0, 1.0, 1.0);\r\n }\r\n\r\n /**\r\n * Returns a new Vector4 with random values between min and max\r\n * @param min the minimum random value\r\n * @param max the maximum random value\r\n * @returns a Vector4 with random values between min and max\r\n */\r\n public static Random(min: number = 0, max: number = 1): Vector4 {\r\n return new Vector4(Scalar.RandomRange(min, max), Scalar.RandomRange(min, max), Scalar.RandomRange(min, max), Scalar.RandomRange(min, max));\r\n }\r\n\r\n /**\r\n * Gets a zero Vector4 that must not be updated\r\n */\r\n public static get ZeroReadOnly(): DeepImmutable {\r\n return Vector4._ZeroReadOnly;\r\n }\r\n /**\r\n * Returns a new normalized Vector4 from the given one.\r\n * @param vector the vector to normalize\r\n * @returns the vector\r\n */\r\n public static Normalize(vector: DeepImmutable): Vector4 {\r\n const result = Vector4.Zero();\r\n Vector4.NormalizeToRef(vector, result);\r\n return result;\r\n }\r\n /**\r\n * Updates the given vector \"result\" from the normalization of the given one.\r\n * @param vector the vector to normalize\r\n * @param result the vector to store the result in\r\n * @returns result input\r\n */\r\n public static NormalizeToRef(vector: DeepImmutable, result: T): T {\r\n result.copyFrom(vector);\r\n result.normalize();\r\n return result;\r\n }\r\n\r\n /**\r\n * Returns a vector with the minimum values from the left and right vectors\r\n * @param left left vector to minimize\r\n * @param right right vector to minimize\r\n * @returns a new vector with the minimum of the left and right vector values\r\n */\r\n public static Minimize(left: DeepImmutable, right: DeepImmutable): T {\r\n const min = new (left.constructor as Vector4Constructor)();\r\n min.copyFrom(left);\r\n min.minimizeInPlace(right);\r\n return min;\r\n }\r\n\r\n /**\r\n * Returns a vector with the maximum values from the left and right vectors\r\n * @param left left vector to maximize\r\n * @param right right vector to maximize\r\n * @returns a new vector with the maximum of the left and right vector values\r\n */\r\n public static Maximize(left: DeepImmutable, right: DeepImmutable): T {\r\n const max = new (left.constructor as Vector4Constructor)();\r\n max.copyFrom(left);\r\n max.maximizeInPlace(right);\r\n return max;\r\n }\r\n /**\r\n * Returns the distance (float) between the vectors \"value1\" and \"value2\".\r\n * @param value1 value to calulate the distance between\r\n * @param value2 value to calulate the distance between\r\n * @returns the distance between the two vectors\r\n */\r\n public static Distance(value1: DeepImmutable, value2: DeepImmutable): number {\r\n return Math.sqrt(Vector4.DistanceSquared(value1, value2));\r\n }\r\n /**\r\n * Returns the squared distance (float) between the vectors \"value1\" and \"value2\".\r\n * @param value1 value to calulate the distance between\r\n * @param value2 value to calulate the distance between\r\n * @returns the distance between the two vectors squared\r\n */\r\n public static DistanceSquared(value1: DeepImmutable, value2: DeepImmutable): number {\r\n const x = value1.x - value2.x;\r\n const y = value1.y - value2.y;\r\n const z = value1.z - value2.z;\r\n const w = value1.w - value2.w;\r\n\r\n return x * x + y * y + z * z + w * w;\r\n }\r\n /**\r\n * Returns a new Vector4 located at the center between the vectors \"value1\" and \"value2\".\r\n * @param value1 value to calulate the center between\r\n * @param value2 value to calulate the center between\r\n * @returns the center between the two vectors\r\n */\r\n public static Center(value1: DeepImmutable, value2: DeepImmutable): Vector4 {\r\n return Vector4.CenterToRef(value1, value2, Vector4.Zero());\r\n }\r\n\r\n /**\r\n * Gets the center of the vectors \"value1\" and \"value2\" and stores the result in the vector \"ref\"\r\n * @param value1 defines first vector\r\n * @param value2 defines second vector\r\n * @param ref defines third vector\r\n * @returns ref\r\n */\r\n public static CenterToRef(value1: DeepImmutable, value2: DeepImmutable, ref: T): T {\r\n return ref.copyFromFloats((value1.x + value2.x) / 2, (value1.y + value2.y) / 2, (value1.z + value2.z) / 2, (value1.w + value2.w) / 2);\r\n }\r\n\r\n /**\r\n * Returns a new Vector4 set with the result of the transformation by the given matrix of the given vector.\r\n * This method computes tranformed coordinates only, not transformed direction vectors (ie. it takes translation in account)\r\n * The difference with Vector3.TransformCoordinates is that the w component is not used to divide the other coordinates but is returned in the w coordinate instead\r\n * @param vector defines the Vector3 to transform\r\n * @param transformation defines the transformation matrix\r\n * @returns the transformed Vector4\r\n */\r\n public static TransformCoordinates(vector: DeepImmutable, transformation: DeepImmutable): Vector4 {\r\n const result = Vector4.Zero();\r\n Vector4.TransformCoordinatesToRef(vector, transformation, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the given vector \"result\" coordinates with the result of the transformation by the given matrix of the given vector\r\n * This method computes tranformed coordinates only, not transformed direction vectors (ie. it takes translation in account)\r\n * The difference with Vector3.TransformCoordinatesToRef is that the w component is not used to divide the other coordinates but is returned in the w coordinate instead\r\n * @param vector defines the Vector3 to transform\r\n * @param transformation defines the transformation matrix\r\n * @param result defines the Vector4 where to store the result\r\n * @returns result input\r\n */\r\n public static TransformCoordinatesToRef(vector: DeepImmutable, transformation: DeepImmutable, result: T): T {\r\n Vector4.TransformCoordinatesFromFloatsToRef(vector._x, vector._y, vector._z, transformation, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the given vector \"result\" coordinates with the result of the transformation by the given matrix of the given floats (x, y, z)\r\n * This method computes tranformed coordinates only, not transformed direction vectors\r\n * The difference with Vector3.TransformCoordinatesFromFloatsToRef is that the w component is not used to divide the other coordinates but is returned in the w coordinate instead\r\n * @param x define the x coordinate of the source vector\r\n * @param y define the y coordinate of the source vector\r\n * @param z define the z coordinate of the source vector\r\n * @param transformation defines the transformation matrix\r\n * @param result defines the Vector4 where to store the result\r\n * @returns result input\r\n */\r\n public static TransformCoordinatesFromFloatsToRef(x: number, y: number, z: number, transformation: DeepImmutable, result: T): T {\r\n const m = transformation.m;\r\n const rx = x * m[0] + y * m[4] + z * m[8] + m[12];\r\n const ry = x * m[1] + y * m[5] + z * m[9] + m[13];\r\n const rz = x * m[2] + y * m[6] + z * m[10] + m[14];\r\n const rw = x * m[3] + y * m[7] + z * m[11] + m[15];\r\n\r\n result.x = rx;\r\n result.y = ry;\r\n result.z = rz;\r\n result.w = rw;\r\n return result;\r\n }\r\n\r\n /**\r\n * Returns a new Vector4 set with the result of the normal transformation by the given matrix of the given vector.\r\n * This methods computes transformed normalized direction vectors only.\r\n * @param vector the vector to transform\r\n * @param transformation the transformation matrix to apply\r\n * @returns the new vector\r\n */\r\n public static TransformNormal(vector: DeepImmutable, transformation: DeepImmutable): T {\r\n const result = new (vector.constructor as Vector4Constructor)();\r\n Vector4.TransformNormalToRef(vector, transformation, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the given vector \"result\" with the result of the normal transformation by the given matrix of the given vector.\r\n * This methods computes transformed normalized direction vectors only.\r\n * @param vector the vector to transform\r\n * @param transformation the transformation matrix to apply\r\n * @param result the vector to store the result in\r\n * @returns result input\r\n */\r\n public static TransformNormalToRef(vector: DeepImmutable, transformation: DeepImmutable, result: T): T {\r\n const m = transformation.m;\r\n const x = vector.x * m[0] + vector.y * m[4] + vector.z * m[8];\r\n const y = vector.x * m[1] + vector.y * m[5] + vector.z * m[9];\r\n const z = vector.x * m[2] + vector.y * m[6] + vector.z * m[10];\r\n result.x = x;\r\n result.y = y;\r\n result.z = z;\r\n result.w = vector.w;\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the given vector \"result\" with the result of the normal transformation by the given matrix of the given floats (x, y, z, w).\r\n * This methods computes transformed normalized direction vectors only.\r\n * @param x value to transform\r\n * @param y value to transform\r\n * @param z value to transform\r\n * @param w value to transform\r\n * @param transformation the transformation matrix to apply\r\n * @param result the vector to store the results in\r\n * @returns result input\r\n */\r\n public static TransformNormalFromFloatsToRef(x: number, y: number, z: number, w: number, transformation: DeepImmutable, result: T): T {\r\n const m = transformation.m;\r\n result.x = x * m[0] + y * m[4] + z * m[8];\r\n result.y = x * m[1] + y * m[5] + z * m[9];\r\n result.z = x * m[2] + y * m[6] + z * m[10];\r\n result.w = w;\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new Vector4 from a Vector3\r\n * @param source defines the source data\r\n * @param w defines the 4th component (default is 0)\r\n * @returns a new Vector4\r\n */\r\n public static FromVector3(source: Vector3, w: number = 0) {\r\n return new Vector4(source._x, source._y, source._z, w);\r\n }\r\n}\r\n\r\n/**\r\n * Class used to store quaternion data\r\n * Example Playground - Overview - https://playground.babylonjs.com/#L49EJ7#100\r\n * @see https://en.wikipedia.org/wiki/Quaternion\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms\r\n */\r\nexport class Quaternion {\r\n /** @internal */\r\n public _x: number;\r\n\r\n /** @internal */\r\n public _y: number;\r\n\r\n /** @internal */\r\n public _z: number;\r\n\r\n /** @internal */\r\n public _w: number;\r\n\r\n /** @internal */\r\n public _isDirty = true;\r\n\r\n /** Gets or sets the x coordinate */\r\n public get x() {\r\n return this._x;\r\n }\r\n\r\n public set x(value: number) {\r\n this._x = value;\r\n this._isDirty = true;\r\n }\r\n\r\n /** Gets or sets the y coordinate */\r\n public get y() {\r\n return this._y;\r\n }\r\n\r\n public set y(value: number) {\r\n this._y = value;\r\n this._isDirty = true;\r\n }\r\n\r\n /** Gets or sets the z coordinate */\r\n public get z() {\r\n return this._z;\r\n }\r\n\r\n public set z(value: number) {\r\n this._z = value;\r\n this._isDirty = true;\r\n }\r\n\r\n /** Gets or sets the w coordinate */\r\n public get w() {\r\n return this._w;\r\n }\r\n\r\n public set w(value: number) {\r\n this._w = value;\r\n this._isDirty = true;\r\n }\r\n /**\r\n * Creates a new Quaternion from the given floats\r\n * @param x defines the first component (0 by default)\r\n * @param y defines the second component (0 by default)\r\n * @param z defines the third component (0 by default)\r\n * @param w defines the fourth component (1.0 by default)\r\n */\r\n constructor(x: number = 0.0, y: number = 0.0, z: number = 0.0, w: number = 1.0) {\r\n this._x = x;\r\n this._y = y;\r\n this._z = z;\r\n this._w = w;\r\n }\r\n\r\n /**\r\n * Gets a string representation for the current quaternion\r\n * @returns a string with the Quaternion coordinates\r\n */\r\n public toString(): string {\r\n return `{X: ${this._x} Y: ${this._y} Z: ${this._z} W: ${this._w}}`;\r\n }\r\n\r\n /**\r\n * Gets the class name of the quaternion\r\n * @returns the string \"Quaternion\"\r\n */\r\n public getClassName(): string {\r\n return \"Quaternion\";\r\n }\r\n\r\n /**\r\n * Gets a hash code for this quaternion\r\n * @returns the quaternion hash code\r\n */\r\n public getHashCode(): number {\r\n const x = _ExtractAsInt(this._x);\r\n const y = _ExtractAsInt(this._y);\r\n const z = _ExtractAsInt(this._z);\r\n const w = _ExtractAsInt(this._w);\r\n\r\n let hash = x;\r\n hash = (hash * 397) ^ y;\r\n hash = (hash * 397) ^ z;\r\n hash = (hash * 397) ^ w;\r\n return hash;\r\n }\r\n\r\n /**\r\n * Copy the quaternion to an array\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#13\r\n * @returns a new array populated with 4 elements from the quaternion coordinates\r\n */\r\n public asArray(): number[] {\r\n return [this._x, this._y, this._z, this._w];\r\n }\r\n\r\n /**\r\n * Stores from the starting index in the given array the Quaternion successive values\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#59\r\n * @param array defines the array where to store the x,y,z,w components\r\n * @param index defines an optional index in the target array to define where to start storing values\r\n * @returns the current Quaternion object\r\n */\r\n public toArray(array: FloatArray, index: number = 0): Quaternion {\r\n array[index] = this._x;\r\n array[index + 1] = this._y;\r\n array[index + 2] = this._z;\r\n array[index + 3] = this._w;\r\n return this;\r\n }\r\n\r\n /**\r\n * Check if two quaternions are equals\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#38\r\n * @param otherQuaternion defines the second operand\r\n * @returns true if the current quaternion and the given one coordinates are strictly equals\r\n */\r\n public equals(otherQuaternion: DeepImmutable): boolean {\r\n return otherQuaternion && this._x === otherQuaternion._x && this._y === otherQuaternion._y && this._z === otherQuaternion._z && this._w === otherQuaternion._w;\r\n }\r\n\r\n /**\r\n * Gets a boolean if two quaternions are equals (using an epsilon value)\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#37\r\n * @param otherQuaternion defines the other quaternion\r\n * @param epsilon defines the minimal distance to consider equality\r\n * @returns true if the given quaternion coordinates are close to the current ones by a distance of epsilon.\r\n */\r\n public equalsWithEpsilon(otherQuaternion: DeepImmutable, epsilon: number = Epsilon): boolean {\r\n return (\r\n otherQuaternion &&\r\n Scalar.WithinEpsilon(this._x, otherQuaternion._x, epsilon) &&\r\n Scalar.WithinEpsilon(this._y, otherQuaternion._y, epsilon) &&\r\n Scalar.WithinEpsilon(this._z, otherQuaternion._z, epsilon) &&\r\n Scalar.WithinEpsilon(this._w, otherQuaternion._w, epsilon)\r\n );\r\n }\r\n\r\n /**\r\n * Clone the current quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#12\r\n * @returns a new quaternion copied from the current one\r\n */\r\n public clone(): this {\r\n return new (this.constructor as QuaternionConstructor)(this._x, this._y, this._z, this._w);\r\n }\r\n\r\n /**\r\n * Copy a quaternion to the current one\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#86\r\n * @param other defines the other quaternion\r\n * @returns the updated current quaternion\r\n */\r\n public copyFrom(other: DeepImmutable): this {\r\n this._x = other._x;\r\n this._y = other._y;\r\n this._z = other._z;\r\n this._w = other._w;\r\n this._isDirty = true;\r\n return this;\r\n }\r\n\r\n /**\r\n * Updates the current quaternion with the given float coordinates\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#87\r\n * @param x defines the x coordinate\r\n * @param y defines the y coordinate\r\n * @param z defines the z coordinate\r\n * @param w defines the w coordinate\r\n * @returns the updated current quaternion\r\n */\r\n public copyFromFloats(x: number, y: number, z: number, w: number): this {\r\n this._x = x;\r\n this._y = y;\r\n this._z = z;\r\n this._w = w;\r\n this._isDirty = true;\r\n return this;\r\n }\r\n\r\n /**\r\n * Updates the current quaternion from the given float coordinates\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#56\r\n * @param x defines the x coordinate\r\n * @param y defines the y coordinate\r\n * @param z defines the z coordinate\r\n * @param w defines the w coordinate\r\n * @returns the updated current quaternion\r\n */\r\n public set(x: number, y: number, z: number, w: number): this {\r\n return this.copyFromFloats(x, y, z, w);\r\n }\r\n\r\n /**\r\n * Adds two quaternions\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#10\r\n * @param other defines the second operand\r\n * @returns a new quaternion as the addition result of the given one and the current quaternion\r\n */\r\n public add(other: DeepImmutable): this {\r\n return new (this.constructor as QuaternionConstructor)(this._x + other._x, this._y + other._y, this._z + other._z, this._w + other._w);\r\n }\r\n\r\n /**\r\n * Add a quaternion to the current one\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#11\r\n * @param other defines the quaternion to add\r\n * @returns the current quaternion\r\n */\r\n public addInPlace(other: DeepImmutable): this {\r\n this._x += other._x;\r\n this._y += other._y;\r\n this._z += other._z;\r\n this._w += other._w;\r\n this._isDirty = true;\r\n return this;\r\n }\r\n\r\n /**\r\n * Subtract two quaternions\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#57\r\n * @param other defines the second operand\r\n * @returns a new quaternion as the subtraction result of the given one from the current one\r\n */\r\n public subtract(other: Quaternion): this {\r\n return new (this.constructor as QuaternionConstructor)(this._x - other._x, this._y - other._y, this._z - other._z, this._w - other._w);\r\n }\r\n\r\n /**\r\n * Subtract a quaternion to the current one\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#58\r\n * @param other defines the quaternion to subtract\r\n * @returns the current quaternion\r\n */\r\n public subtractInPlace(other: DeepImmutable): this {\r\n this._x -= other._x;\r\n this._y -= other._y;\r\n this._z -= other._z;\r\n this._w -= other._w;\r\n this._isDirty = true;\r\n return this;\r\n }\r\n\r\n /**\r\n * Multiplies the current quaternion by a scale factor\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#88\r\n * @param value defines the scale factor\r\n * @returns a new quaternion set by multiplying the current quaternion coordinates by the float \"scale\"\r\n */\r\n public scale(value: number): this {\r\n return new (this.constructor as QuaternionConstructor)(this._x * value, this._y * value, this._z * value, this._w * value);\r\n }\r\n\r\n /**\r\n * Scale the current quaternion values by a factor and stores the result to a given quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#89\r\n * @param scale defines the scale factor\r\n * @param result defines the Quaternion object where to store the result\r\n * @returns result input\r\n */\r\n public scaleToRef(scale: number, result: T): T {\r\n result._x = this._x * scale;\r\n result._y = this._y * scale;\r\n result._z = this._z * scale;\r\n result._w = this._w * scale;\r\n result._isDirty = true;\r\n return result;\r\n }\r\n\r\n /**\r\n * Multiplies in place the current quaternion by a scale factor\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#90\r\n * @param value defines the scale factor\r\n * @returns the current modified quaternion\r\n */\r\n public scaleInPlace(value: number): this {\r\n this._x *= value;\r\n this._y *= value;\r\n this._z *= value;\r\n this._w *= value;\r\n this._isDirty = true;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Scale the current quaternion values by a factor and add the result to a given quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#91\r\n * @param scale defines the scale factor\r\n * @param result defines the Quaternion object where to store the result\r\n * @returns result input\r\n */\r\n public scaleAndAddToRef(scale: number, result: T): T {\r\n result._x += this._x * scale;\r\n result._y += this._y * scale;\r\n result._z += this._z * scale;\r\n result._w += this._w * scale;\r\n result._isDirty = true;\r\n return result;\r\n }\r\n\r\n /**\r\n * Multiplies two quaternions\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#43\r\n * @param q1 defines the second operand\r\n * @returns a new quaternion set as the multiplication result of the current one with the given one \"q1\"\r\n */\r\n public multiply(q1: DeepImmutable): this {\r\n const result = new (this.constructor as QuaternionConstructor)(0, 0, 0, 1.0);\r\n this.multiplyToRef(q1, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the given \"result\" as the the multiplication result of the current one with the given one \"q1\"\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#45\r\n * @param q1 defines the second operand\r\n * @param result defines the target quaternion\r\n * @returns the current quaternion\r\n */\r\n public multiplyToRef(q1: DeepImmutable, result: T): T {\r\n const x = this._x * q1._w + this._y * q1._z - this._z * q1._y + this._w * q1._x;\r\n const y = -this._x * q1._z + this._y * q1._w + this._z * q1._x + this._w * q1._y;\r\n const z = this._x * q1._y - this._y * q1._x + this._z * q1._w + this._w * q1._z;\r\n const w = -this._x * q1._x - this._y * q1._y - this._z * q1._z + this._w * q1._w;\r\n result.copyFromFloats(x, y, z, w);\r\n return result;\r\n }\r\n\r\n /**\r\n * Updates the current quaternion with the multiplication of itself with the given one \"q1\"\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#46\r\n * @param q1 defines the second operand\r\n * @returns the currentupdated quaternion\r\n */\r\n public multiplyInPlace(q1: DeepImmutable): this {\r\n this.multiplyToRef(q1, this);\r\n return this;\r\n }\r\n\r\n /**\r\n * Conjugates the current quaternion and stores the result in the given quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#81\r\n * @param ref defines the target quaternion\r\n * @returns result input\r\n */\r\n public conjugateToRef(ref: T): T {\r\n ref.copyFromFloats(-this._x, -this._y, -this._z, this._w);\r\n return ref;\r\n }\r\n\r\n /**\r\n * Conjugates in place the current quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#82\r\n * @returns the current updated quaternion\r\n */\r\n public conjugateInPlace(): this {\r\n this._x *= -1;\r\n this._y *= -1;\r\n this._z *= -1;\r\n this._isDirty = true;\r\n return this;\r\n }\r\n\r\n /**\r\n * Conjugates (1-q) the current quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#83\r\n * @returns a new quaternion\r\n */\r\n public conjugate(): this {\r\n return new (this.constructor as QuaternionConstructor)(-this._x, -this._y, -this._z, this._w);\r\n }\r\n\r\n /**\r\n * Returns the inverse of the current quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#84\r\n * @returns a new quaternion\r\n */\r\n public invert(): this {\r\n const conjugate = this.conjugate();\r\n const lengthSquared = this.lengthSquared();\r\n if (lengthSquared == 0 || lengthSquared == 1) {\r\n return conjugate;\r\n }\r\n conjugate.scaleInPlace(1 / lengthSquared);\r\n return conjugate;\r\n }\r\n\r\n /**\r\n * Invert in place the current quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#85\r\n * @returns this quaternion\r\n */\r\n public invertInPlace(): this {\r\n this.conjugateInPlace();\r\n const lengthSquared = this.lengthSquared();\r\n if (lengthSquared == 0 || lengthSquared == 1) {\r\n return this;\r\n }\r\n this.scaleInPlace(1 / lengthSquared);\r\n return this;\r\n }\r\n\r\n /**\r\n * Gets squared length of current quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#29\r\n * @returns the quaternion length (float)\r\n */\r\n public lengthSquared(): number {\r\n return this._x * this._x + this._y * this._y + this._z * this._z + this._w * this._w;\r\n }\r\n\r\n /**\r\n * Gets length of current quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#28\r\n * @returns the quaternion length (float)\r\n */\r\n public length(): number {\r\n return Math.sqrt(this.lengthSquared());\r\n }\r\n\r\n /**\r\n * Normalize in place the current quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#54\r\n * @returns the current updated quaternion\r\n */\r\n public normalize(): this {\r\n const len = this.length();\r\n if (len === 0) {\r\n return this;\r\n }\r\n\r\n const inv = 1.0 / len;\r\n this.scaleInPlace(inv);\r\n return this;\r\n }\r\n\r\n /**\r\n * Normalize a copy of the current quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#55\r\n * @returns the normalized quaternion\r\n */\r\n public normalizeToNew(): this {\r\n const len = this.length();\r\n if (len === 0) {\r\n return this.clone();\r\n }\r\n\r\n const inv = 1.0 / len;\r\n return this.scale(inv);\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 set with the Euler angles translated from the current quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#32\r\n * @returns a new Vector3 containing the Euler angles\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/center_origin/rotation_conventions\r\n */\r\n public toEulerAngles(): Vector3 {\r\n const result = Vector3.Zero();\r\n this.toEulerAnglesToRef(result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the given vector3 \"result\" with the Euler angles translated from the current quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#31\r\n * @param result defines the vector which will be filled with the Euler angles\r\n * @returns result input\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/center_origin/rotation_conventions\r\n */\r\n public toEulerAnglesToRef(result: T): T {\r\n const qz = this._z;\r\n const qx = this._x;\r\n const qy = this._y;\r\n const qw = this._w;\r\n\r\n const zAxisY = qy * qz - qx * qw;\r\n const limit = 0.4999999;\r\n\r\n if (zAxisY < -limit) {\r\n result._y = 2 * Math.atan2(qy, qw);\r\n result._x = Math.PI / 2;\r\n result._z = 0;\r\n result._isDirty = true;\r\n } else if (zAxisY > limit) {\r\n result._y = 2 * Math.atan2(qy, qw);\r\n result._x = -Math.PI / 2;\r\n result._z = 0;\r\n result._isDirty = true;\r\n } else {\r\n const sqw = qw * qw;\r\n const sqz = qz * qz;\r\n const sqx = qx * qx;\r\n const sqy = qy * qy;\r\n result._z = Math.atan2(2.0 * (qx * qy + qz * qw), -sqz - sqx + sqy + sqw);\r\n result._x = Math.asin(-2.0 * zAxisY);\r\n result._y = Math.atan2(2.0 * (qz * qx + qy * qw), sqz - sqx - sqy + sqw);\r\n result._isDirty = true;\r\n }\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Updates the given rotation matrix with the current quaternion values\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#67\r\n * @param result defines the target matrix\r\n * @returns the current unchanged quaternion\r\n */\r\n public toRotationMatrix(result: T): T {\r\n Matrix.FromQuaternionToRef(this, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Updates the current quaternion from the given rotation matrix values\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#41\r\n * @param matrix defines the source matrix\r\n * @returns the current updated quaternion\r\n */\r\n public fromRotationMatrix(matrix: DeepImmutable): this {\r\n Quaternion.FromRotationMatrixToRef(matrix, this);\r\n return this;\r\n }\r\n\r\n // Statics\r\n\r\n /**\r\n * Creates a new quaternion from a rotation matrix\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#101\r\n * @param matrix defines the source matrix\r\n * @returns a new quaternion created from the given rotation matrix values\r\n */\r\n public static FromRotationMatrix(matrix: DeepImmutable): Quaternion {\r\n const result = new Quaternion();\r\n Quaternion.FromRotationMatrixToRef(matrix, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Updates the given quaternion with the given rotation matrix values\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#102\r\n * @param matrix defines the source matrix\r\n * @param result defines the target quaternion\r\n * @returns result input\r\n */\r\n public static FromRotationMatrixToRef(matrix: DeepImmutable, result: T): T {\r\n const data = matrix.m;\r\n const m11 = data[0],\r\n m12 = data[4],\r\n m13 = data[8];\r\n const m21 = data[1],\r\n m22 = data[5],\r\n m23 = data[9];\r\n const m31 = data[2],\r\n m32 = data[6],\r\n m33 = data[10];\r\n const trace = m11 + m22 + m33;\r\n let s;\r\n\r\n if (trace > 0) {\r\n s = 0.5 / Math.sqrt(trace + 1.0);\r\n\r\n result._w = 0.25 / s;\r\n result._x = (m32 - m23) * s;\r\n result._y = (m13 - m31) * s;\r\n result._z = (m21 - m12) * s;\r\n result._isDirty = true;\r\n } else if (m11 > m22 && m11 > m33) {\r\n s = 2.0 * Math.sqrt(1.0 + m11 - m22 - m33);\r\n\r\n result._w = (m32 - m23) / s;\r\n result._x = 0.25 * s;\r\n result._y = (m12 + m21) / s;\r\n result._z = (m13 + m31) / s;\r\n result._isDirty = true;\r\n } else if (m22 > m33) {\r\n s = 2.0 * Math.sqrt(1.0 + m22 - m11 - m33);\r\n\r\n result._w = (m13 - m31) / s;\r\n result._x = (m12 + m21) / s;\r\n result._y = 0.25 * s;\r\n result._z = (m23 + m32) / s;\r\n result._isDirty = true;\r\n } else {\r\n s = 2.0 * Math.sqrt(1.0 + m33 - m11 - m22);\r\n\r\n result._w = (m21 - m12) / s;\r\n result._x = (m13 + m31) / s;\r\n result._y = (m23 + m32) / s;\r\n result._z = 0.25 * s;\r\n result._isDirty = true;\r\n }\r\n return result;\r\n }\r\n\r\n /**\r\n * Returns the dot product (float) between the quaternions \"left\" and \"right\"\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#61\r\n * @param left defines the left operand\r\n * @param right defines the right operand\r\n * @returns the dot product\r\n */\r\n public static Dot(left: DeepImmutable, right: DeepImmutable): number {\r\n return left._x * right._x + left._y * right._y + left._z * right._z + left._w * right._w;\r\n }\r\n\r\n /**\r\n * Checks if the orientations of two rotation quaternions are close to each other\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#60\r\n * @param quat0 defines the first quaternion to check\r\n * @param quat1 defines the second quaternion to check\r\n * @param epsilon defines closeness, 0 same orientation, 1 PI apart, default 0.1\r\n * @returns true if the two quaternions are close to each other within epsilon\r\n */\r\n public static AreClose(quat0: DeepImmutable, quat1: DeepImmutable, epsilon: number = 0.1): boolean {\r\n const dot = Quaternion.Dot(quat0, quat1);\r\n\r\n return 1 - dot * dot <= epsilon;\r\n }\r\n\r\n /**\r\n * Smooth interpolation between two quaternions using Slerp\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#93\r\n * @param source source quaternion\r\n * @param goal goal quaternion\r\n * @param deltaTime current interpolation frame\r\n * @param lerpTime total interpolation time\r\n * @param result the smoothed quaternion\r\n */\r\n public static SmoothToRef(source: Quaternion, goal: Quaternion, deltaTime: number, lerpTime: number, result: T): T {\r\n let slerp = lerpTime === 0 ? 1 : deltaTime / lerpTime;\r\n slerp = Scalar.Clamp(slerp, 0, 1);\r\n\r\n Quaternion.SlerpToRef(source, goal, slerp, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates an empty quaternion\r\n * @returns a new quaternion set to (0.0, 0.0, 0.0)\r\n */\r\n public static Zero(): Quaternion {\r\n return new Quaternion(0.0, 0.0, 0.0, 0.0);\r\n }\r\n\r\n /**\r\n * Inverse a given quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#103\r\n * @param q defines the source quaternion\r\n * @returns a new quaternion as the inverted current quaternion\r\n */\r\n public static Inverse(q: DeepImmutable): T {\r\n return new (q.constructor as QuaternionConstructor)(-q._x, -q._y, -q._z, q._w);\r\n }\r\n\r\n /**\r\n * Inverse a given quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#104\r\n * @param q defines the source quaternion\r\n * @param result the quaternion the result will be stored in\r\n * @returns the result quaternion\r\n */\r\n public static InverseToRef(q: Quaternion, result: T): T {\r\n result.set(-q._x, -q._y, -q._z, q._w);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates an identity quaternion\r\n * @returns the identity quaternion\r\n */\r\n public static Identity(): Quaternion {\r\n return new Quaternion(0.0, 0.0, 0.0, 1.0);\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if the given quaternion is identity\r\n * @param quaternion defines the quaternion to check\r\n * @returns true if the quaternion is identity\r\n */\r\n public static IsIdentity(quaternion: DeepImmutable): boolean {\r\n return quaternion && quaternion._x === 0 && quaternion._y === 0 && quaternion._z === 0 && quaternion._w === 1;\r\n }\r\n\r\n /**\r\n * Creates a quaternion from a rotation around an axis\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#72\r\n * @param axis defines the axis to use\r\n * @param angle defines the angle to use\r\n * @returns a new quaternion created from the given axis (Vector3) and angle in radians (float)\r\n */\r\n public static RotationAxis(axis: DeepImmutable, angle: number): Quaternion {\r\n return Quaternion.RotationAxisToRef(axis, angle, new Quaternion());\r\n }\r\n\r\n /**\r\n * Creates a rotation around an axis and stores it into the given quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#73\r\n * @param axis defines the axis to use\r\n * @param angle defines the angle to use\r\n * @param result defines the target quaternion\r\n * @returns the target quaternion\r\n */\r\n public static RotationAxisToRef(axis: DeepImmutable, angle: number, result: T): T {\r\n const sin = Math.sin(angle / 2);\r\n axis.normalize();\r\n result._w = Math.cos(angle / 2);\r\n result._x = axis._x * sin;\r\n result._y = axis._y * sin;\r\n result._z = axis._z * sin;\r\n result._isDirty = true;\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new quaternion from data stored into an array\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#63\r\n * @param array defines the data source\r\n * @param offset defines the offset in the source array where the data starts\r\n * @returns a new quaternion\r\n */\r\n public static FromArray(array: DeepImmutable>, offset?: number): Quaternion {\r\n if (!offset) {\r\n offset = 0;\r\n }\r\n return new Quaternion(array[offset], array[offset + 1], array[offset + 2], array[offset + 3]);\r\n }\r\n\r\n /**\r\n * Updates the given quaternion \"result\" from the starting index of the given array.\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#64\r\n * @param array the array to pull values from\r\n * @param offset the offset into the array to start at\r\n * @param result the quaternion to store the result in\r\n * @returns result input\r\n */\r\n public static FromArrayToRef(array: DeepImmutable>, offset: number, result: T): T {\r\n result._x = array[offset];\r\n result._y = array[offset + 1];\r\n result._z = array[offset + 2];\r\n result._w = array[offset + 3];\r\n result._isDirty = true;\r\n return result;\r\n }\r\n\r\n /**\r\n * Create a quaternion from Euler rotation angles\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#33\r\n * @param x Pitch\r\n * @param y Yaw\r\n * @param z Roll\r\n * @returns the new Quaternion\r\n */\r\n public static FromEulerAngles(x: number, y: number, z: number): Quaternion {\r\n const q = new Quaternion();\r\n Quaternion.RotationYawPitchRollToRef(y, x, z, q);\r\n return q;\r\n }\r\n\r\n /**\r\n * Updates a quaternion from Euler rotation angles\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#34\r\n * @param x Pitch\r\n * @param y Yaw\r\n * @param z Roll\r\n * @param result the quaternion to store the result\r\n * @returns the updated quaternion\r\n */\r\n public static FromEulerAnglesToRef(x: number, y: number, z: number, result: T): T {\r\n Quaternion.RotationYawPitchRollToRef(y, x, z, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Create a quaternion from Euler rotation vector\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#35\r\n * @param vec the Euler vector (x Pitch, y Yaw, z Roll)\r\n * @returns the new Quaternion\r\n */\r\n public static FromEulerVector(vec: DeepImmutable): Quaternion {\r\n const q = new Quaternion();\r\n Quaternion.RotationYawPitchRollToRef(vec._y, vec._x, vec._z, q);\r\n return q;\r\n }\r\n\r\n /**\r\n * Updates a quaternion from Euler rotation vector\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#36\r\n * @param vec the Euler vector (x Pitch, y Yaw, z Roll)\r\n * @param result the quaternion to store the result\r\n * @returns the updated quaternion\r\n */\r\n public static FromEulerVectorToRef(vec: DeepImmutable, result: T): T {\r\n Quaternion.RotationYawPitchRollToRef(vec._y, vec._x, vec._z, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Updates a quaternion so that it rotates vector vecFrom to vector vecTo\r\n * Example Playground - https://playground.babylonjs.com/#L49EJ7#70\r\n * @param vecFrom defines the direction vector from which to rotate\r\n * @param vecTo defines the direction vector to which to rotate\r\n * @param result the quaternion to store the result\r\n * @param epsilon defines the minimal dot value to define vecs as opposite. Default: `BABYLON.Epsilon`\r\n * @returns the updated quaternion\r\n */\r\n public static FromUnitVectorsToRef(vecFrom: DeepImmutable, vecTo: DeepImmutable, result: T, epsilon = Epsilon): T {\r\n const r = Vector3.Dot(vecFrom, vecTo) + 1;\r\n\r\n if (r < epsilon) {\r\n if (Math.abs(vecFrom.x) > Math.abs(vecFrom.z)) {\r\n result.set(-vecFrom.y, vecFrom.x, 0, 0);\r\n } else {\r\n result.set(0, -vecFrom.z, vecFrom.y, 0);\r\n }\r\n } else {\r\n Vector3.CrossToRef(vecFrom, vecTo, TmpVectors.Vector3[0]);\r\n result.set(TmpVectors.Vector3[0].x, TmpVectors.Vector3[0].y, TmpVectors.Vector3[0].z, r);\r\n }\r\n\r\n return result.normalize();\r\n }\r\n\r\n /**\r\n * Creates a new quaternion from the given Euler float angles (y, x, z)\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#77\r\n * @param yaw defines the rotation around Y axis\r\n * @param pitch defines the rotation around X axis\r\n * @param roll defines the rotation around Z axis\r\n * @returns the new quaternion\r\n */\r\n public static RotationYawPitchRoll(yaw: number, pitch: number, roll: number): Quaternion {\r\n const q = new Quaternion();\r\n Quaternion.RotationYawPitchRollToRef(yaw, pitch, roll, q);\r\n return q;\r\n }\r\n\r\n /**\r\n * Creates a new rotation from the given Euler float angles (y, x, z) and stores it in the target quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#78\r\n * @param yaw defines the rotation around Y axis\r\n * @param pitch defines the rotation around X axis\r\n * @param roll defines the rotation around Z axis\r\n * @param result defines the target quaternion\r\n * @returns result input\r\n */\r\n public static RotationYawPitchRollToRef(yaw: number, pitch: number, roll: number, result: T): T {\r\n // Produces a quaternion from Euler angles in the z-y-x orientation (Tait-Bryan angles)\r\n const halfRoll = roll * 0.5;\r\n const halfPitch = pitch * 0.5;\r\n const halfYaw = yaw * 0.5;\r\n\r\n const sinRoll = Math.sin(halfRoll);\r\n const cosRoll = Math.cos(halfRoll);\r\n const sinPitch = Math.sin(halfPitch);\r\n const cosPitch = Math.cos(halfPitch);\r\n const sinYaw = Math.sin(halfYaw);\r\n const cosYaw = Math.cos(halfYaw);\r\n\r\n result._x = cosYaw * sinPitch * cosRoll + sinYaw * cosPitch * sinRoll;\r\n result._y = sinYaw * cosPitch * cosRoll - cosYaw * sinPitch * sinRoll;\r\n result._z = cosYaw * cosPitch * sinRoll - sinYaw * sinPitch * cosRoll;\r\n result._w = cosYaw * cosPitch * cosRoll + sinYaw * sinPitch * sinRoll;\r\n result._isDirty = true;\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new quaternion from the given Euler float angles expressed in z-x-z orientation\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#68\r\n * @param alpha defines the rotation around first axis\r\n * @param beta defines the rotation around second axis\r\n * @param gamma defines the rotation around third axis\r\n * @returns the new quaternion\r\n */\r\n public static RotationAlphaBetaGamma(alpha: number, beta: number, gamma: number): Quaternion {\r\n const result = new Quaternion();\r\n Quaternion.RotationAlphaBetaGammaToRef(alpha, beta, gamma, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new quaternion from the given Euler float angles expressed in z-x-z orientation and stores it in the target quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#69\r\n * @param alpha defines the rotation around first axis\r\n * @param beta defines the rotation around second axis\r\n * @param gamma defines the rotation around third axis\r\n * @param result defines the target quaternion\r\n * @returns result input\r\n */\r\n public static RotationAlphaBetaGammaToRef(alpha: number, beta: number, gamma: number, result: T): T {\r\n // Produces a quaternion from Euler angles in the z-x-z orientation\r\n const halfGammaPlusAlpha = (gamma + alpha) * 0.5;\r\n const halfGammaMinusAlpha = (gamma - alpha) * 0.5;\r\n const halfBeta = beta * 0.5;\r\n\r\n result._x = Math.cos(halfGammaMinusAlpha) * Math.sin(halfBeta);\r\n result._y = Math.sin(halfGammaMinusAlpha) * Math.sin(halfBeta);\r\n result._z = Math.sin(halfGammaPlusAlpha) * Math.cos(halfBeta);\r\n result._w = Math.cos(halfGammaPlusAlpha) * Math.cos(halfBeta);\r\n result._isDirty = true;\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new quaternion containing the rotation value to reach the target (axis1, axis2, axis3) orientation as a rotated XYZ system (axis1, axis2 and axis3 are normalized during this operation)\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#75\r\n * @param axis1 defines the first axis\r\n * @param axis2 defines the second axis\r\n * @param axis3 defines the third axis\r\n * @returns the new quaternion\r\n */\r\n public static RotationQuaternionFromAxis(axis1: DeepImmutable, axis2: DeepImmutable, axis3: DeepImmutable): Quaternion {\r\n const quat = new Quaternion(0.0, 0.0, 0.0, 0.0);\r\n Quaternion.RotationQuaternionFromAxisToRef(axis1, axis2, axis3, quat);\r\n return quat;\r\n }\r\n\r\n /**\r\n * Creates a rotation value to reach the target (axis1, axis2, axis3) orientation as a rotated XYZ system (axis1, axis2 and axis3 are normalized during this operation) and stores it in the target quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#76\r\n * @param axis1 defines the first axis\r\n * @param axis2 defines the second axis\r\n * @param axis3 defines the third axis\r\n * @param ref defines the target quaternion\r\n * @returns result input\r\n */\r\n public static RotationQuaternionFromAxisToRef(axis1: DeepImmutable, axis2: DeepImmutable, axis3: DeepImmutable, ref: T): T {\r\n const rotMat = MathTmp.Matrix[0];\r\n Matrix.FromXYZAxesToRef(axis1.normalize(), axis2.normalize(), axis3.normalize(), rotMat);\r\n Quaternion.FromRotationMatrixToRef(rotMat, ref);\r\n return ref;\r\n }\r\n\r\n /**\r\n * Creates a new rotation value to orient an object to look towards the given forward direction, the up direction being oriented like \"up\".\r\n * This function works in left handed mode\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#96\r\n * @param forward defines the forward direction - Must be normalized and orthogonal to up.\r\n * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward.\r\n * @returns A new quaternion oriented toward the specified forward and up.\r\n */\r\n public static FromLookDirectionLH(forward: DeepImmutable, up: DeepImmutable): Quaternion {\r\n const quat = new Quaternion();\r\n Quaternion.FromLookDirectionLHToRef(forward, up, quat);\r\n return quat;\r\n }\r\n\r\n /**\r\n * Creates a new rotation value to orient an object to look towards the given forward direction with the up direction being oriented like \"up\", and stores it in the target quaternion.\r\n * This function works in left handed mode\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#97\r\n * @param forward defines the forward direction - Must be normalized and orthogonal to up.\r\n * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward.\r\n * @param ref defines the target quaternion.\r\n * @returns result input\r\n */\r\n public static FromLookDirectionLHToRef(forward: DeepImmutable, up: DeepImmutable, ref: T): T {\r\n const rotMat = MathTmp.Matrix[0];\r\n Matrix.LookDirectionLHToRef(forward, up, rotMat);\r\n Quaternion.FromRotationMatrixToRef(rotMat, ref);\r\n return ref;\r\n }\r\n\r\n /**\r\n * Creates a new rotation value to orient an object to look towards the given forward direction, the up direction being oriented like \"up\".\r\n * This function works in right handed mode\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#98\r\n * @param forward defines the forward direction - Must be normalized and orthogonal to up.\r\n * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward.\r\n * @returns A new quaternion oriented toward the specified forward and up.\r\n */\r\n public static FromLookDirectionRH(forward: DeepImmutable, up: DeepImmutable): Quaternion {\r\n const quat = new Quaternion();\r\n Quaternion.FromLookDirectionRHToRef(forward, up, quat);\r\n return quat;\r\n }\r\n\r\n /**\r\n * Creates a new rotation value to orient an object to look towards the given forward direction with the up direction being oriented like \"up\", and stores it in the target quaternion.\r\n * This function works in right handed mode\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#105\r\n * @param forward defines the forward direction - Must be normalized and orthogonal to up.\r\n * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward.\r\n * @param ref defines the target quaternion.\r\n * @returns result input\r\n */\r\n public static FromLookDirectionRHToRef(forward: DeepImmutable, up: DeepImmutable, ref: T): T {\r\n const rotMat = MathTmp.Matrix[0];\r\n Matrix.LookDirectionRHToRef(forward, up, rotMat);\r\n return Quaternion.FromRotationMatrixToRef(rotMat, ref);\r\n }\r\n\r\n /**\r\n * Interpolates between two quaternions\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#79\r\n * @param left defines first quaternion\r\n * @param right defines second quaternion\r\n * @param amount defines the gradient to use\r\n * @returns the new interpolated quaternion\r\n */\r\n public static Slerp(left: DeepImmutable, right: DeepImmutable, amount: number): Quaternion {\r\n const result = Quaternion.Identity();\r\n\r\n Quaternion.SlerpToRef(left, right, amount, result);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Interpolates between two quaternions and stores it into a target quaternion\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#92\r\n * @param left defines first quaternion\r\n * @param right defines second quaternion\r\n * @param amount defines the gradient to use\r\n * @param result defines the target quaternion\r\n * @returns result input\r\n */\r\n public static SlerpToRef(left: DeepImmutable, right: DeepImmutable, amount: number, result: T): T {\r\n let num2;\r\n let num3;\r\n let num4 = left._x * right._x + left._y * right._y + left._z * right._z + left._w * right._w;\r\n let flag = false;\r\n\r\n if (num4 < 0) {\r\n flag = true;\r\n num4 = -num4;\r\n }\r\n\r\n if (num4 > 0.999999) {\r\n num3 = 1 - amount;\r\n num2 = flag ? -amount : amount;\r\n } else {\r\n const num5 = Math.acos(num4);\r\n const num6 = 1.0 / Math.sin(num5);\r\n num3 = Math.sin((1.0 - amount) * num5) * num6;\r\n num2 = flag ? -Math.sin(amount * num5) * num6 : Math.sin(amount * num5) * num6;\r\n }\r\n\r\n result._x = num3 * left._x + num2 * right._x;\r\n result._y = num3 * left._y + num2 * right._y;\r\n result._z = num3 * left._z + num2 * right._z;\r\n result._w = num3 * left._w + num2 * right._w;\r\n result._isDirty = true;\r\n return result;\r\n }\r\n\r\n /**\r\n * Interpolate between two quaternions using Hermite interpolation\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#47\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/drawCurves#hermite-quaternion-spline\r\n * @param value1 defines first quaternion\r\n * @param tangent1 defines the incoming tangent\r\n * @param value2 defines second quaternion\r\n * @param tangent2 defines the outgoing tangent\r\n * @param amount defines the target quaternion\r\n * @returns the new interpolated quaternion\r\n */\r\n public static Hermite(\r\n value1: DeepImmutable,\r\n tangent1: DeepImmutable,\r\n value2: DeepImmutable,\r\n tangent2: DeepImmutable,\r\n amount: number\r\n ): T {\r\n const squared = amount * amount;\r\n const cubed = amount * squared;\r\n const part1 = 2.0 * cubed - 3.0 * squared + 1.0;\r\n const part2 = -2.0 * cubed + 3.0 * squared;\r\n const part3 = cubed - 2.0 * squared + amount;\r\n const part4 = cubed - squared;\r\n\r\n const x = value1._x * part1 + value2._x * part2 + tangent1._x * part3 + tangent2._x * part4;\r\n const y = value1._y * part1 + value2._y * part2 + tangent1._y * part3 + tangent2._y * part4;\r\n const z = value1._z * part1 + value2._z * part2 + tangent1._z * part3 + tangent2._z * part4;\r\n const w = value1._w * part1 + value2._w * part2 + tangent1._w * part3 + tangent2._w * part4;\r\n return new (value1.constructor as QuaternionConstructor)(x, y, z, w);\r\n }\r\n\r\n /**\r\n * Returns a new Quaternion which is the 1st derivative of the Hermite spline defined by the quaternions \"value1\", \"value2\", \"tangent1\", \"tangent2\".\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#48\r\n * @param value1 defines the first control point\r\n * @param tangent1 defines the first tangent\r\n * @param value2 defines the second control point\r\n * @param tangent2 defines the second tangent\r\n * @param time define where the derivative must be done\r\n * @returns 1st derivative\r\n */\r\n public static Hermite1stDerivative(\r\n value1: DeepImmutable,\r\n tangent1: DeepImmutable,\r\n value2: DeepImmutable,\r\n tangent2: DeepImmutable,\r\n time: number\r\n ): T {\r\n const result = new (value1.constructor as QuaternionConstructor)();\r\n\r\n this.Hermite1stDerivativeToRef(value1, tangent1, value2, tangent2, time, result);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Update a Quaternion with the 1st derivative of the Hermite spline defined by the quaternions \"value1\", \"value2\", \"tangent1\", \"tangent2\".\r\n * Example Playground https://playground.babylonjs.com/#L49EJ7#49\r\n * @param value1 defines the first control point\r\n * @param tangent1 defines the first tangent\r\n * @param value2 defines the second control point\r\n * @param tangent2 defines the second tangent\r\n * @param time define where the derivative must be done\r\n * @param result define where to store the derivative\r\n * @returns result input\r\n */\r\n public static Hermite1stDerivativeToRef(\r\n value1: DeepImmutable,\r\n tangent1: DeepImmutable,\r\n value2: DeepImmutable,\r\n tangent2: DeepImmutable,\r\n time: number,\r\n result: T\r\n ): T {\r\n const t2 = time * time;\r\n\r\n result._x = (t2 - time) * 6 * value1._x + (3 * t2 - 4 * time + 1) * tangent1._x + (-t2 + time) * 6 * value2._x + (3 * t2 - 2 * time) * tangent2._x;\r\n result._y = (t2 - time) * 6 * value1._y + (3 * t2 - 4 * time + 1) * tangent1._y + (-t2 + time) * 6 * value2._y + (3 * t2 - 2 * time) * tangent2._y;\r\n result._z = (t2 - time) * 6 * value1._z + (3 * t2 - 4 * time + 1) * tangent1._z + (-t2 + time) * 6 * value2._z + (3 * t2 - 2 * time) * tangent2._z;\r\n result._w = (t2 - time) * 6 * value1._w + (3 * t2 - 4 * time + 1) * tangent1._w + (-t2 + time) * 6 * value2._w + (3 * t2 - 2 * time) * tangent2._w;\r\n result._isDirty = true;\r\n return result;\r\n }\r\n}\r\n\r\n/**\r\n * Class used to store matrix data (4x4)\r\n * Note on matrix definitions in Babylon.js for setting values directly\r\n * rather than using one of the methods available.\r\n * Matrix size is given by rows x columns.\r\n * A Vector3 is a 1 X 3 matrix [x, y, z].\r\n *\r\n * In Babylon.js multiplying a 1 x 3 matrix by a 4 x 4 matrix\r\n * is done using BABYLON.Vector4.TransformCoordinates(Vector3, Matrix).\r\n * and extending the passed Vector3 to a Vector4, V = [x, y, z, 1].\r\n * Let M be a matrix with elements m(row, column), so that\r\n * m(2, 3) is the element in row 2 column 3 of M.\r\n *\r\n * Multiplication is of the form VM and has the resulting Vector4\r\n * VM = [xm(0, 0) + ym(1, 0) + zm(2, 0) + m(3, 0), xm(0, 1) + ym(1, 1) + zm(2, 1) + m(3, 1), xm(0, 2) + ym(1, 2) + zm(2, 2) + m(3, 2), xm(0, 3) + ym(1, 3) + zm(2, 3) + m(3, 3)].\r\n * On the web you will find many examples that use the opposite convention of MV,\r\n * in which case to make use of the examples you will need to transpose the matrix.\r\n *\r\n * Example Playground - Overview Linear Algebra - https://playground.babylonjs.com/#AV9X17\r\n * Example Playground - Overview Transformation - https://playground.babylonjs.com/#AV9X17#1\r\n * Example Playground - Overview Projection - https://playground.babylonjs.com/#AV9X17#2\r\n */\r\nexport class Matrix {\r\n /**\r\n * Gets the precision of matrix computations\r\n */\r\n public static get Use64Bits(): boolean {\r\n return PerformanceConfigurator.MatrixUse64Bits;\r\n }\r\n\r\n private static _UpdateFlagSeed = 0;\r\n private static _IdentityReadOnly = Matrix.Identity() as DeepImmutable;\r\n\r\n private _isIdentity = false;\r\n private _isIdentityDirty = true;\r\n private _isIdentity3x2 = true;\r\n private _isIdentity3x2Dirty = true;\r\n /**\r\n * Gets the update flag of the matrix which is an unique number for the matrix.\r\n * It will be incremented every time the matrix data change.\r\n * You can use it to speed the comparison between two versions of the same matrix.\r\n */\r\n public updateFlag: number = -1;\r\n\r\n private readonly _m: Float32Array | Array;\r\n\r\n /**\r\n * Gets the internal data of the matrix\r\n */\r\n public get m(): DeepImmutable> {\r\n return this._m;\r\n }\r\n\r\n /**\r\n * Update the updateFlag to indicate that the matrix has been updated\r\n */\r\n public markAsUpdated() {\r\n this.updateFlag = Matrix._UpdateFlagSeed++;\r\n this._isIdentity = false;\r\n this._isIdentity3x2 = false;\r\n this._isIdentityDirty = true;\r\n this._isIdentity3x2Dirty = true;\r\n }\r\n\r\n private _updateIdentityStatus(isIdentity: boolean, isIdentityDirty: boolean = false, isIdentity3x2: boolean = false, isIdentity3x2Dirty: boolean = true) {\r\n this._isIdentity = isIdentity;\r\n this._isIdentity3x2 = isIdentity || isIdentity3x2;\r\n this._isIdentityDirty = this._isIdentity ? false : isIdentityDirty;\r\n this._isIdentity3x2Dirty = this._isIdentity3x2 ? false : isIdentity3x2Dirty;\r\n }\r\n\r\n /**\r\n * Creates an empty matrix (filled with zeros)\r\n */\r\n public constructor() {\r\n if (PerformanceConfigurator.MatrixTrackPrecisionChange) {\r\n PerformanceConfigurator.MatrixTrackedMatrices!.push(this);\r\n }\r\n\r\n this._m = new PerformanceConfigurator.MatrixCurrentType(16);\r\n\r\n this.markAsUpdated();\r\n }\r\n\r\n // Properties\r\n\r\n /**\r\n * Check if the current matrix is identity\r\n * @returns true is the matrix is the identity matrix\r\n */\r\n public isIdentity(): boolean {\r\n if (this._isIdentityDirty) {\r\n this._isIdentityDirty = false;\r\n const m = this._m;\r\n this._isIdentity =\r\n m[0] === 1.0 &&\r\n m[1] === 0.0 &&\r\n m[2] === 0.0 &&\r\n m[3] === 0.0 &&\r\n m[4] === 0.0 &&\r\n m[5] === 1.0 &&\r\n m[6] === 0.0 &&\r\n m[7] === 0.0 &&\r\n m[8] === 0.0 &&\r\n m[9] === 0.0 &&\r\n m[10] === 1.0 &&\r\n m[11] === 0.0 &&\r\n m[12] === 0.0 &&\r\n m[13] === 0.0 &&\r\n m[14] === 0.0 &&\r\n m[15] === 1.0;\r\n }\r\n\r\n return this._isIdentity;\r\n }\r\n\r\n /**\r\n * Check if the current matrix is identity as a texture matrix (3x2 store in 4x4)\r\n * @returns true is the matrix is the identity matrix\r\n */\r\n public isIdentityAs3x2(): boolean {\r\n if (this._isIdentity3x2Dirty) {\r\n this._isIdentity3x2Dirty = false;\r\n if (this._m[0] !== 1.0 || this._m[5] !== 1.0 || this._m[15] !== 1.0) {\r\n this._isIdentity3x2 = false;\r\n } else if (\r\n this._m[1] !== 0.0 ||\r\n this._m[2] !== 0.0 ||\r\n this._m[3] !== 0.0 ||\r\n this._m[4] !== 0.0 ||\r\n this._m[6] !== 0.0 ||\r\n this._m[7] !== 0.0 ||\r\n this._m[8] !== 0.0 ||\r\n this._m[9] !== 0.0 ||\r\n this._m[10] !== 0.0 ||\r\n this._m[11] !== 0.0 ||\r\n this._m[12] !== 0.0 ||\r\n this._m[13] !== 0.0 ||\r\n this._m[14] !== 0.0\r\n ) {\r\n this._isIdentity3x2 = false;\r\n } else {\r\n this._isIdentity3x2 = true;\r\n }\r\n }\r\n\r\n return this._isIdentity3x2;\r\n }\r\n\r\n /**\r\n * Gets the determinant of the matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#34\r\n * @returns the matrix determinant\r\n */\r\n public determinant(): number {\r\n if (this._isIdentity === true) {\r\n return 1;\r\n }\r\n\r\n const m = this._m;\r\n const m00 = m[0],\r\n m01 = m[1],\r\n m02 = m[2],\r\n m03 = m[3];\r\n const m10 = m[4],\r\n m11 = m[5],\r\n m12 = m[6],\r\n m13 = m[7];\r\n const m20 = m[8],\r\n m21 = m[9],\r\n m22 = m[10],\r\n m23 = m[11];\r\n const m30 = m[12],\r\n m31 = m[13],\r\n m32 = m[14],\r\n m33 = m[15];\r\n // https://en.wikipedia.org/wiki/Laplace_expansion\r\n // to compute the deterrminant of a 4x4 Matrix we compute the cofactors of any row or column,\r\n // then we multiply each Cofactor by its corresponding matrix value and sum them all to get the determinant\r\n // Cofactor(i, j) = sign(i,j) * det(Minor(i, j))\r\n // where\r\n // - sign(i,j) = (i+j) % 2 === 0 ? 1 : -1\r\n // - Minor(i, j) is the 3x3 matrix we get by removing row i and column j from current Matrix\r\n //\r\n // Here we do that for the 1st row.\r\n\r\n const det_22_33 = m22 * m33 - m32 * m23;\r\n const det_21_33 = m21 * m33 - m31 * m23;\r\n const det_21_32 = m21 * m32 - m31 * m22;\r\n const det_20_33 = m20 * m33 - m30 * m23;\r\n const det_20_32 = m20 * m32 - m22 * m30;\r\n const det_20_31 = m20 * m31 - m30 * m21;\r\n const cofact_00 = +(m11 * det_22_33 - m12 * det_21_33 + m13 * det_21_32);\r\n const cofact_01 = -(m10 * det_22_33 - m12 * det_20_33 + m13 * det_20_32);\r\n const cofact_02 = +(m10 * det_21_33 - m11 * det_20_33 + m13 * det_20_31);\r\n const cofact_03 = -(m10 * det_21_32 - m11 * det_20_32 + m12 * det_20_31);\r\n return m00 * cofact_00 + m01 * cofact_01 + m02 * cofact_02 + m03 * cofact_03;\r\n }\r\n\r\n // Methods\r\n\r\n /**\r\n * Returns the matrix as a Float32Array or Array\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#49\r\n * @returns the matrix underlying array\r\n */\r\n public toArray(): DeepImmutable> {\r\n return this._m;\r\n }\r\n /**\r\n * Returns the matrix as a Float32Array or Array\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#114\r\n * @returns the matrix underlying array.\r\n */\r\n public asArray(): DeepImmutable> {\r\n return this._m;\r\n }\r\n\r\n /**\r\n * Inverts the current matrix in place\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#118\r\n * @returns the current inverted matrix\r\n */\r\n public invert(): this {\r\n this.invertToRef(this);\r\n return this;\r\n }\r\n /**\r\n * Sets all the matrix elements to zero\r\n * @returns the current matrix\r\n */\r\n public reset(): this {\r\n Matrix.FromValuesToRef(0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, this);\r\n this._updateIdentityStatus(false);\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds the current matrix with a second one\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#44\r\n * @param other defines the matrix to add\r\n * @returns a new matrix as the addition of the current matrix and the given one\r\n */\r\n public add(other: DeepImmutable): this {\r\n const result = new (this.constructor as MatrixConstructor)();\r\n this.addToRef(other, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the given matrix \"result\" to the addition of the current matrix and the given one\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#45\r\n * @param other defines the matrix to add\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public addToRef(other: DeepImmutable, result: T): T {\r\n const m = this._m;\r\n const resultM = result._m;\r\n const otherM = other.m;\r\n for (let index = 0; index < 16; index++) {\r\n resultM[index] = m[index] + otherM[index];\r\n }\r\n result.markAsUpdated();\r\n return result;\r\n }\r\n\r\n /**\r\n * Adds in place the given matrix to the current matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#46\r\n * @param other defines the second operand\r\n * @returns the current updated matrix\r\n */\r\n public addToSelf(other: DeepImmutable): this {\r\n const m = this._m;\r\n const otherM = other.m;\r\n for (let index = 0; index < 16; index++) {\r\n m[index] += otherM[index];\r\n }\r\n this.markAsUpdated();\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets the given matrix to the current inverted Matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#119\r\n * @param other defines the target matrix\r\n * @returns result input\r\n */\r\n public invertToRef(other: T): T {\r\n if (this._isIdentity === true) {\r\n Matrix.IdentityToRef(other);\r\n return other;\r\n }\r\n\r\n // the inverse of a Matrix is the transpose of cofactor matrix divided by the determinant\r\n const m = this._m;\r\n const m00 = m[0],\r\n m01 = m[1],\r\n m02 = m[2],\r\n m03 = m[3];\r\n const m10 = m[4],\r\n m11 = m[5],\r\n m12 = m[6],\r\n m13 = m[7];\r\n const m20 = m[8],\r\n m21 = m[9],\r\n m22 = m[10],\r\n m23 = m[11];\r\n const m30 = m[12],\r\n m31 = m[13],\r\n m32 = m[14],\r\n m33 = m[15];\r\n\r\n const det_22_33 = m22 * m33 - m32 * m23;\r\n const det_21_33 = m21 * m33 - m31 * m23;\r\n const det_21_32 = m21 * m32 - m31 * m22;\r\n const det_20_33 = m20 * m33 - m30 * m23;\r\n const det_20_32 = m20 * m32 - m22 * m30;\r\n const det_20_31 = m20 * m31 - m30 * m21;\r\n\r\n const cofact_00 = +(m11 * det_22_33 - m12 * det_21_33 + m13 * det_21_32);\r\n const cofact_01 = -(m10 * det_22_33 - m12 * det_20_33 + m13 * det_20_32);\r\n const cofact_02 = +(m10 * det_21_33 - m11 * det_20_33 + m13 * det_20_31);\r\n const cofact_03 = -(m10 * det_21_32 - m11 * det_20_32 + m12 * det_20_31);\r\n\r\n const det = m00 * cofact_00 + m01 * cofact_01 + m02 * cofact_02 + m03 * cofact_03;\r\n\r\n if (det === 0) {\r\n // not invertible\r\n other.copyFrom(this);\r\n return other;\r\n }\r\n\r\n const detInv = 1 / det;\r\n const det_12_33 = m12 * m33 - m32 * m13;\r\n const det_11_33 = m11 * m33 - m31 * m13;\r\n const det_11_32 = m11 * m32 - m31 * m12;\r\n const det_10_33 = m10 * m33 - m30 * m13;\r\n const det_10_32 = m10 * m32 - m30 * m12;\r\n const det_10_31 = m10 * m31 - m30 * m11;\r\n const det_12_23 = m12 * m23 - m22 * m13;\r\n const det_11_23 = m11 * m23 - m21 * m13;\r\n const det_11_22 = m11 * m22 - m21 * m12;\r\n const det_10_23 = m10 * m23 - m20 * m13;\r\n const det_10_22 = m10 * m22 - m20 * m12;\r\n const det_10_21 = m10 * m21 - m20 * m11;\r\n\r\n const cofact_10 = -(m01 * det_22_33 - m02 * det_21_33 + m03 * det_21_32);\r\n const cofact_11 = +(m00 * det_22_33 - m02 * det_20_33 + m03 * det_20_32);\r\n const cofact_12 = -(m00 * det_21_33 - m01 * det_20_33 + m03 * det_20_31);\r\n const cofact_13 = +(m00 * det_21_32 - m01 * det_20_32 + m02 * det_20_31);\r\n\r\n const cofact_20 = +(m01 * det_12_33 - m02 * det_11_33 + m03 * det_11_32);\r\n const cofact_21 = -(m00 * det_12_33 - m02 * det_10_33 + m03 * det_10_32);\r\n const cofact_22 = +(m00 * det_11_33 - m01 * det_10_33 + m03 * det_10_31);\r\n const cofact_23 = -(m00 * det_11_32 - m01 * det_10_32 + m02 * det_10_31);\r\n\r\n const cofact_30 = -(m01 * det_12_23 - m02 * det_11_23 + m03 * det_11_22);\r\n const cofact_31 = +(m00 * det_12_23 - m02 * det_10_23 + m03 * det_10_22);\r\n const cofact_32 = -(m00 * det_11_23 - m01 * det_10_23 + m03 * det_10_21);\r\n const cofact_33 = +(m00 * det_11_22 - m01 * det_10_22 + m02 * det_10_21);\r\n\r\n Matrix.FromValuesToRef(\r\n cofact_00 * detInv,\r\n cofact_10 * detInv,\r\n cofact_20 * detInv,\r\n cofact_30 * detInv,\r\n cofact_01 * detInv,\r\n cofact_11 * detInv,\r\n cofact_21 * detInv,\r\n cofact_31 * detInv,\r\n cofact_02 * detInv,\r\n cofact_12 * detInv,\r\n cofact_22 * detInv,\r\n cofact_32 * detInv,\r\n cofact_03 * detInv,\r\n cofact_13 * detInv,\r\n cofact_23 * detInv,\r\n cofact_33 * detInv,\r\n other\r\n );\r\n\r\n return other;\r\n }\r\n\r\n /**\r\n * add a value at the specified position in the current Matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#47\r\n * @param index the index of the value within the matrix. between 0 and 15.\r\n * @param value the value to be added\r\n * @returns the current updated matrix\r\n */\r\n public addAtIndex(index: number, value: number): this {\r\n this._m[index] += value;\r\n this.markAsUpdated();\r\n return this;\r\n }\r\n\r\n /**\r\n * mutiply the specified position in the current Matrix by a value\r\n * @param index the index of the value within the matrix. between 0 and 15.\r\n * @param value the value to be added\r\n * @returns the current updated matrix\r\n */\r\n public multiplyAtIndex(index: number, value: number): this {\r\n this._m[index] *= value;\r\n this.markAsUpdated();\r\n return this;\r\n }\r\n\r\n /**\r\n * Inserts the translation vector (using 3 floats) in the current matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#120\r\n * @param x defines the 1st component of the translation\r\n * @param y defines the 2nd component of the translation\r\n * @param z defines the 3rd component of the translation\r\n * @returns the current updated matrix\r\n */\r\n public setTranslationFromFloats(x: number, y: number, z: number): this {\r\n this._m[12] = x;\r\n this._m[13] = y;\r\n this._m[14] = z;\r\n this.markAsUpdated();\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds the translation vector (using 3 floats) in the current matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#20\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#48\r\n * @param x defines the 1st component of the translation\r\n * @param y defines the 2nd component of the translation\r\n * @param z defines the 3rd component of the translation\r\n * @returns the current updated matrix\r\n */\r\n public addTranslationFromFloats(x: number, y: number, z: number): this {\r\n this._m[12] += x;\r\n this._m[13] += y;\r\n this._m[14] += z;\r\n this.markAsUpdated();\r\n return this;\r\n }\r\n\r\n /**\r\n * Inserts the translation vector in the current matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#121\r\n * @param vector3 defines the translation to insert\r\n * @returns the current updated matrix\r\n */\r\n public setTranslation(vector3: DeepImmutable): this {\r\n return this.setTranslationFromFloats(vector3._x, vector3._y, vector3._z);\r\n }\r\n\r\n /**\r\n * Gets the translation value of the current matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#122\r\n * @returns a new Vector3 as the extracted translation from the matrix\r\n */\r\n public getTranslation(): Vector3 {\r\n return new Vector3(this._m[12], this._m[13], this._m[14]);\r\n }\r\n\r\n /**\r\n * Fill a Vector3 with the extracted translation from the matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#123\r\n * @param result defines the Vector3 where to store the translation\r\n * @returns the current matrix\r\n */\r\n public getTranslationToRef(result: T): T {\r\n result.x = this._m[12];\r\n result.y = this._m[13];\r\n result.z = this._m[14];\r\n return result;\r\n }\r\n\r\n /**\r\n * Remove rotation and scaling part from the matrix\r\n * @returns the updated matrix\r\n */\r\n public removeRotationAndScaling(): this {\r\n const m = this.m;\r\n Matrix.FromValuesToRef(1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, m[12], m[13], m[14], m[15], this);\r\n this._updateIdentityStatus(m[12] === 0 && m[13] === 0 && m[14] === 0 && m[15] === 1);\r\n return this;\r\n }\r\n\r\n /**\r\n * Multiply two matrices\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#15\r\n * A.multiply(B) means apply B to A so result is B x A\r\n * @param other defines the second operand\r\n * @returns a new matrix set with the multiplication result of the current Matrix and the given one\r\n */\r\n public multiply(other: DeepImmutable): this {\r\n const result = new (this.constructor as MatrixConstructor)();\r\n this.multiplyToRef(other, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Copy the current matrix from the given one\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#21\r\n * @param other defines the source matrix\r\n * @returns the current updated matrix\r\n */\r\n public copyFrom(other: DeepImmutable): this {\r\n other.copyToArray(this._m);\r\n const o = other as Matrix;\r\n this.updateFlag = o.updateFlag;\r\n this._updateIdentityStatus(o._isIdentity, o._isIdentityDirty, o._isIdentity3x2, o._isIdentity3x2Dirty);\r\n return this;\r\n }\r\n\r\n /**\r\n * Populates the given array from the starting index with the current matrix values\r\n * @param array defines the target array\r\n * @param offset defines the offset in the target array where to start storing values\r\n * @returns the current matrix\r\n */\r\n public copyToArray(array: Float32Array | Array, offset: number = 0): this {\r\n const source = this._m;\r\n array[offset] = source[0];\r\n array[offset + 1] = source[1];\r\n array[offset + 2] = source[2];\r\n array[offset + 3] = source[3];\r\n array[offset + 4] = source[4];\r\n array[offset + 5] = source[5];\r\n array[offset + 6] = source[6];\r\n array[offset + 7] = source[7];\r\n array[offset + 8] = source[8];\r\n array[offset + 9] = source[9];\r\n array[offset + 10] = source[10];\r\n array[offset + 11] = source[11];\r\n array[offset + 12] = source[12];\r\n array[offset + 13] = source[13];\r\n array[offset + 14] = source[14];\r\n array[offset + 15] = source[15];\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets the given matrix \"result\" with the multiplication result of the current Matrix and the given one\r\n * A.multiplyToRef(B, R) means apply B to A and store in R and R = B x A\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#16\r\n * @param other defines the second operand\r\n * @param result defines the matrix where to store the multiplication\r\n * @returns result input\r\n */\r\n public multiplyToRef(other: DeepImmutable, result: T): T {\r\n if (this._isIdentity) {\r\n result.copyFrom(other);\r\n return result;\r\n }\r\n if ((other as Matrix)._isIdentity) {\r\n result.copyFrom(this);\r\n return result;\r\n }\r\n\r\n this.multiplyToArray(other, result._m, 0);\r\n result.markAsUpdated();\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the Float32Array \"result\" from the given index \"offset\" with the multiplication of the current matrix and the given one\r\n * @param other defines the second operand\r\n * @param result defines the array where to store the multiplication\r\n * @param offset defines the offset in the target array where to start storing values\r\n * @returns the current matrix\r\n */\r\n public multiplyToArray(other: DeepImmutable, result: Float32Array | Array, offset: number): this {\r\n const m = this._m;\r\n const otherM = other.m;\r\n const tm0 = m[0],\r\n tm1 = m[1],\r\n tm2 = m[2],\r\n tm3 = m[3];\r\n const tm4 = m[4],\r\n tm5 = m[5],\r\n tm6 = m[6],\r\n tm7 = m[7];\r\n const tm8 = m[8],\r\n tm9 = m[9],\r\n tm10 = m[10],\r\n tm11 = m[11];\r\n const tm12 = m[12],\r\n tm13 = m[13],\r\n tm14 = m[14],\r\n tm15 = m[15];\r\n\r\n const om0 = otherM[0],\r\n om1 = otherM[1],\r\n om2 = otherM[2],\r\n om3 = otherM[3];\r\n const om4 = otherM[4],\r\n om5 = otherM[5],\r\n om6 = otherM[6],\r\n om7 = otherM[7];\r\n const om8 = otherM[8],\r\n om9 = otherM[9],\r\n om10 = otherM[10],\r\n om11 = otherM[11];\r\n const om12 = otherM[12],\r\n om13 = otherM[13],\r\n om14 = otherM[14],\r\n om15 = otherM[15];\r\n\r\n result[offset] = tm0 * om0 + tm1 * om4 + tm2 * om8 + tm3 * om12;\r\n result[offset + 1] = tm0 * om1 + tm1 * om5 + tm2 * om9 + tm3 * om13;\r\n result[offset + 2] = tm0 * om2 + tm1 * om6 + tm2 * om10 + tm3 * om14;\r\n result[offset + 3] = tm0 * om3 + tm1 * om7 + tm2 * om11 + tm3 * om15;\r\n\r\n result[offset + 4] = tm4 * om0 + tm5 * om4 + tm6 * om8 + tm7 * om12;\r\n result[offset + 5] = tm4 * om1 + tm5 * om5 + tm6 * om9 + tm7 * om13;\r\n result[offset + 6] = tm4 * om2 + tm5 * om6 + tm6 * om10 + tm7 * om14;\r\n result[offset + 7] = tm4 * om3 + tm5 * om7 + tm6 * om11 + tm7 * om15;\r\n\r\n result[offset + 8] = tm8 * om0 + tm9 * om4 + tm10 * om8 + tm11 * om12;\r\n result[offset + 9] = tm8 * om1 + tm9 * om5 + tm10 * om9 + tm11 * om13;\r\n result[offset + 10] = tm8 * om2 + tm9 * om6 + tm10 * om10 + tm11 * om14;\r\n result[offset + 11] = tm8 * om3 + tm9 * om7 + tm10 * om11 + tm11 * om15;\r\n\r\n result[offset + 12] = tm12 * om0 + tm13 * om4 + tm14 * om8 + tm15 * om12;\r\n result[offset + 13] = tm12 * om1 + tm13 * om5 + tm14 * om9 + tm15 * om13;\r\n result[offset + 14] = tm12 * om2 + tm13 * om6 + tm14 * om10 + tm15 * om14;\r\n result[offset + 15] = tm12 * om3 + tm13 * om7 + tm14 * om11 + tm15 * om15;\r\n return this;\r\n }\r\n\r\n /**\r\n * Check equality between this matrix and a second one\r\n * @param value defines the second matrix to compare\r\n * @returns true is the current matrix and the given one values are strictly equal\r\n */\r\n public equals(value: DeepImmutable): boolean {\r\n const other = value as Matrix;\r\n if (!other) {\r\n return false;\r\n }\r\n\r\n if (this._isIdentity || other._isIdentity) {\r\n if (!this._isIdentityDirty && !other._isIdentityDirty) {\r\n return this._isIdentity && other._isIdentity;\r\n }\r\n }\r\n\r\n const m = this.m;\r\n const om = other.m;\r\n return (\r\n m[0] === om[0] &&\r\n m[1] === om[1] &&\r\n m[2] === om[2] &&\r\n m[3] === om[3] &&\r\n m[4] === om[4] &&\r\n m[5] === om[5] &&\r\n m[6] === om[6] &&\r\n m[7] === om[7] &&\r\n m[8] === om[8] &&\r\n m[9] === om[9] &&\r\n m[10] === om[10] &&\r\n m[11] === om[11] &&\r\n m[12] === om[12] &&\r\n m[13] === om[13] &&\r\n m[14] === om[14] &&\r\n m[15] === om[15]\r\n );\r\n }\r\n\r\n /**\r\n * Clone the current matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#18\r\n * @returns a new matrix from the current matrix\r\n */\r\n public clone(): this {\r\n const matrix = new (this.constructor as MatrixConstructor)();\r\n matrix.copyFrom(this);\r\n return matrix;\r\n }\r\n\r\n /**\r\n * Returns the name of the current matrix class\r\n * @returns the string \"Matrix\"\r\n */\r\n public getClassName(): string {\r\n return \"Matrix\";\r\n }\r\n\r\n /**\r\n * Gets the hash code of the current matrix\r\n * @returns the hash code\r\n */\r\n public getHashCode(): number {\r\n let hash = _ExtractAsInt(this._m[0]);\r\n for (let i = 1; i < 16; i++) {\r\n hash = (hash * 397) ^ _ExtractAsInt(this._m[i]);\r\n }\r\n return hash;\r\n }\r\n\r\n /**\r\n * Decomposes the current Matrix into a translation, rotation and scaling components of the provided node\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#13\r\n * @param node the node to decompose the matrix to\r\n * @returns true if operation was successful\r\n */\r\n public decomposeToTransformNode(node: TransformNode): boolean {\r\n node.rotationQuaternion = node.rotationQuaternion || new Quaternion();\r\n return this.decompose(node.scaling, node.rotationQuaternion, node.position);\r\n }\r\n /**\r\n * Decomposes the current Matrix into a translation, rotation and scaling components\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#12\r\n * @param scale defines the scale vector3 given as a reference to update\r\n * @param rotation defines the rotation quaternion given as a reference to update\r\n * @param translation defines the translation vector3 given as a reference to update\r\n * @param preserveScalingNode Use scaling sign coming from this node. Otherwise scaling sign might change.\r\n * @returns true if operation was successful\r\n */\r\n public decompose(scale?: Vector3, rotation?: Quaternion, translation?: Vector3, preserveScalingNode?: TransformNode): boolean {\r\n if (this._isIdentity) {\r\n if (translation) {\r\n translation.setAll(0);\r\n }\r\n if (scale) {\r\n scale.setAll(1);\r\n }\r\n if (rotation) {\r\n rotation.copyFromFloats(0, 0, 0, 1);\r\n }\r\n return true;\r\n }\r\n\r\n const m = this._m;\r\n if (translation) {\r\n translation.copyFromFloats(m[12], m[13], m[14]);\r\n }\r\n\r\n scale = scale || MathTmp.Vector3[0];\r\n\r\n scale.x = Math.sqrt(m[0] * m[0] + m[1] * m[1] + m[2] * m[2]);\r\n scale.y = Math.sqrt(m[4] * m[4] + m[5] * m[5] + m[6] * m[6]);\r\n scale.z = Math.sqrt(m[8] * m[8] + m[9] * m[9] + m[10] * m[10]);\r\n\r\n if (preserveScalingNode) {\r\n const signX = preserveScalingNode.scaling.x < 0 ? -1 : 1;\r\n const signY = preserveScalingNode.scaling.y < 0 ? -1 : 1;\r\n const signZ = preserveScalingNode.scaling.z < 0 ? -1 : 1;\r\n\r\n scale.x *= signX;\r\n scale.y *= signY;\r\n scale.z *= signZ;\r\n } else {\r\n if (this.determinant() <= 0) {\r\n scale.y *= -1;\r\n }\r\n }\r\n\r\n if (scale._x === 0 || scale._y === 0 || scale._z === 0) {\r\n if (rotation) {\r\n rotation.copyFromFloats(0.0, 0.0, 0.0, 1.0);\r\n }\r\n return false;\r\n }\r\n\r\n if (rotation) {\r\n const sx = 1 / scale._x,\r\n sy = 1 / scale._y,\r\n sz = 1 / scale._z;\r\n Matrix.FromValuesToRef(\r\n m[0] * sx,\r\n m[1] * sx,\r\n m[2] * sx,\r\n 0.0,\r\n m[4] * sy,\r\n m[5] * sy,\r\n m[6] * sy,\r\n 0.0,\r\n m[8] * sz,\r\n m[9] * sz,\r\n m[10] * sz,\r\n 0.0,\r\n 0.0,\r\n 0.0,\r\n 0.0,\r\n 1.0,\r\n MathTmp.Matrix[0]\r\n );\r\n\r\n Quaternion.FromRotationMatrixToRef(MathTmp.Matrix[0], rotation);\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Gets specific row of the matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#36\r\n * @param index defines the number of the row to get\r\n * @returns the index-th row of the current matrix as a new Vector4\r\n */\r\n public getRow(index: number): Nullable {\r\n if (index < 0 || index > 3) {\r\n return null;\r\n }\r\n const i = index * 4;\r\n return new Vector4(this._m[i + 0], this._m[i + 1], this._m[i + 2], this._m[i + 3]);\r\n }\r\n\r\n /**\r\n * Gets specific row of the matrix to ref\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#36\r\n * @param index defines the number of the row to get\r\n * @param rowVector vector to store the index-th row of the current matrix\r\n * @returns result input\r\n */\r\n public getRowToRef(index: number, rowVector: T): T {\r\n if (index >= 0 && index < 3) {\r\n const i = index * 4;\r\n rowVector.x = this._m[i + 0];\r\n rowVector.y = this._m[i + 1];\r\n rowVector.z = this._m[i + 2];\r\n rowVector.w = this._m[i + 3];\r\n }\r\n return rowVector;\r\n }\r\n\r\n /**\r\n * Sets the index-th row of the current matrix to the vector4 values\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#36\r\n * @param index defines the number of the row to set\r\n * @param row defines the target vector4\r\n * @returns the updated current matrix\r\n */\r\n public setRow(index: number, row: Vector4): this {\r\n return this.setRowFromFloats(index, row.x, row.y, row.z, row.w);\r\n }\r\n\r\n /**\r\n * Compute the transpose of the matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#40\r\n * @returns the new transposed matrix\r\n */\r\n public transpose(): this {\r\n const result = new (this.constructor as MatrixConstructor)();\r\n Matrix.TransposeToRef(this, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Compute the transpose of the matrix and store it in a given matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#41\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public transposeToRef(result: T): T {\r\n Matrix.TransposeToRef(this, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the index-th row of the current matrix with the given 4 x float values\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#36\r\n * @param index defines the row index\r\n * @param x defines the x component to set\r\n * @param y defines the y component to set\r\n * @param z defines the z component to set\r\n * @param w defines the w component to set\r\n * @returns the updated current matrix\r\n */\r\n public setRowFromFloats(index: number, x: number, y: number, z: number, w: number): this {\r\n if (index < 0 || index > 3) {\r\n return this;\r\n }\r\n const i = index * 4;\r\n this._m[i + 0] = x;\r\n this._m[i + 1] = y;\r\n this._m[i + 2] = z;\r\n this._m[i + 3] = w;\r\n\r\n this.markAsUpdated();\r\n return this;\r\n }\r\n\r\n /**\r\n * Compute a new matrix set with the current matrix values multiplied by scale (float)\r\n * @param scale defines the scale factor\r\n * @returns a new matrix\r\n */\r\n public scale(scale: number): this {\r\n const result = new (this.constructor as MatrixConstructor)();\r\n this.scaleToRef(scale, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Scale the current matrix values by a factor to a given result matrix\r\n * @param scale defines the scale factor\r\n * @param result defines the matrix to store the result\r\n * @returns result input\r\n */\r\n public scaleToRef(scale: number, result: T): T {\r\n for (let index = 0; index < 16; index++) {\r\n result._m[index] = this._m[index] * scale;\r\n }\r\n result.markAsUpdated();\r\n return result;\r\n }\r\n\r\n /**\r\n * Scale the current matrix values by a factor and add the result to a given matrix\r\n * @param scale defines the scale factor\r\n * @param result defines the Matrix to store the result\r\n * @returns result input\r\n */\r\n public scaleAndAddToRef(scale: number, result: T): T {\r\n for (let index = 0; index < 16; index++) {\r\n result._m[index] += this._m[index] * scale;\r\n }\r\n result.markAsUpdated();\r\n return result;\r\n }\r\n\r\n /**\r\n * Writes to the given matrix a normal matrix, computed from this one (using values from identity matrix for fourth row and column).\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#17\r\n * @param ref matrix to store the result\r\n */\r\n public toNormalMatrix(ref: T): T {\r\n const tmp = MathTmp.Matrix[0];\r\n this.invertToRef(tmp);\r\n tmp.transposeToRef(ref);\r\n const m = ref._m;\r\n Matrix.FromValuesToRef(m[0], m[1], m[2], 0.0, m[4], m[5], m[6], 0.0, m[8], m[9], m[10], 0.0, 0.0, 0.0, 0.0, 1.0, ref);\r\n return ref;\r\n }\r\n\r\n /**\r\n * Gets only rotation part of the current matrix\r\n * @returns a new matrix sets to the extracted rotation matrix from the current one\r\n */\r\n public getRotationMatrix(): this {\r\n const result = new (this.constructor as MatrixConstructor)();\r\n this.getRotationMatrixToRef(result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Extracts the rotation matrix from the current one and sets it as the given \"result\"\r\n * @param result defines the target matrix to store data to\r\n * @returns result input\r\n */\r\n public getRotationMatrixToRef(result: T): T {\r\n const scale = MathTmp.Vector3[0];\r\n if (!this.decompose(scale)) {\r\n Matrix.IdentityToRef(result);\r\n return result;\r\n }\r\n\r\n const m = this._m;\r\n const sx = 1 / scale._x,\r\n sy = 1 / scale._y,\r\n sz = 1 / scale._z;\r\n Matrix.FromValuesToRef(m[0] * sx, m[1] * sx, m[2] * sx, 0.0, m[4] * sy, m[5] * sy, m[6] * sy, 0.0, m[8] * sz, m[9] * sz, m[10] * sz, 0.0, 0.0, 0.0, 0.0, 1.0, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Toggles model matrix from being right handed to left handed in place and vice versa\r\n */\r\n public toggleModelMatrixHandInPlace(): this {\r\n const m = this._m;\r\n m[2] *= -1;\r\n m[6] *= -1;\r\n m[8] *= -1;\r\n m[9] *= -1;\r\n m[14] *= -1;\r\n this.markAsUpdated();\r\n return this;\r\n }\r\n\r\n /**\r\n * Toggles projection matrix from being right handed to left handed in place and vice versa\r\n */\r\n public toggleProjectionMatrixHandInPlace(): this {\r\n const m = this._m;\r\n m[8] *= -1;\r\n m[9] *= -1;\r\n m[10] *= -1;\r\n m[11] *= -1;\r\n this.markAsUpdated();\r\n return this;\r\n }\r\n\r\n // Statics\r\n /**\r\n * Creates a matrix from an array\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#42\r\n * @param array defines the source array\r\n * @param offset defines an offset in the source array\r\n * @returns a new Matrix set from the starting index of the given array\r\n */\r\n public static FromArray(array: DeepImmutable>, offset: number = 0): Matrix {\r\n const result = new Matrix();\r\n Matrix.FromArrayToRef(array, offset, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Copy the content of an array into a given matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#43\r\n * @param array defines the source array\r\n * @param offset defines an offset in the source array\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static FromArrayToRef(array: DeepImmutable>, offset: number, result: T): T {\r\n for (let index = 0; index < 16; index++) {\r\n result._m[index] = array[index + offset];\r\n }\r\n result.markAsUpdated();\r\n return result;\r\n }\r\n\r\n /**\r\n * Stores an array into a matrix after having multiplied each component by a given factor\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#50\r\n * @param array defines the source array\r\n * @param offset defines the offset in the source array\r\n * @param scale defines the scaling factor\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static FromFloat32ArrayToRefScaled(array: DeepImmutable>, offset: number, scale: number, result: T): T {\r\n for (let index = 0; index < 16; index++) {\r\n result._m[index] = array[index + offset] * scale;\r\n }\r\n result.markAsUpdated();\r\n return result;\r\n }\r\n\r\n /**\r\n * Gets an identity matrix that must not be updated\r\n */\r\n public static get IdentityReadOnly(): DeepImmutable {\r\n return Matrix._IdentityReadOnly;\r\n }\r\n\r\n /**\r\n * Stores a list of values (16) inside a given matrix\r\n * @param initialM11 defines 1st value of 1st row\r\n * @param initialM12 defines 2nd value of 1st row\r\n * @param initialM13 defines 3rd value of 1st row\r\n * @param initialM14 defines 4th value of 1st row\r\n * @param initialM21 defines 1st value of 2nd row\r\n * @param initialM22 defines 2nd value of 2nd row\r\n * @param initialM23 defines 3rd value of 2nd row\r\n * @param initialM24 defines 4th value of 2nd row\r\n * @param initialM31 defines 1st value of 3rd row\r\n * @param initialM32 defines 2nd value of 3rd row\r\n * @param initialM33 defines 3rd value of 3rd row\r\n * @param initialM34 defines 4th value of 3rd row\r\n * @param initialM41 defines 1st value of 4th row\r\n * @param initialM42 defines 2nd value of 4th row\r\n * @param initialM43 defines 3rd value of 4th row\r\n * @param initialM44 defines 4th value of 4th row\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static FromValuesToRef(\r\n initialM11: number,\r\n initialM12: number,\r\n initialM13: number,\r\n initialM14: number,\r\n initialM21: number,\r\n initialM22: number,\r\n initialM23: number,\r\n initialM24: number,\r\n initialM31: number,\r\n initialM32: number,\r\n initialM33: number,\r\n initialM34: number,\r\n initialM41: number,\r\n initialM42: number,\r\n initialM43: number,\r\n initialM44: number,\r\n result: Matrix\r\n ): void {\r\n const m = result._m;\r\n m[0] = initialM11;\r\n m[1] = initialM12;\r\n m[2] = initialM13;\r\n m[3] = initialM14;\r\n m[4] = initialM21;\r\n m[5] = initialM22;\r\n m[6] = initialM23;\r\n m[7] = initialM24;\r\n m[8] = initialM31;\r\n m[9] = initialM32;\r\n m[10] = initialM33;\r\n m[11] = initialM34;\r\n m[12] = initialM41;\r\n m[13] = initialM42;\r\n m[14] = initialM43;\r\n m[15] = initialM44;\r\n\r\n result.markAsUpdated();\r\n }\r\n\r\n /**\r\n * Creates new matrix from a list of values (16)\r\n * @param initialM11 defines 1st value of 1st row\r\n * @param initialM12 defines 2nd value of 1st row\r\n * @param initialM13 defines 3rd value of 1st row\r\n * @param initialM14 defines 4th value of 1st row\r\n * @param initialM21 defines 1st value of 2nd row\r\n * @param initialM22 defines 2nd value of 2nd row\r\n * @param initialM23 defines 3rd value of 2nd row\r\n * @param initialM24 defines 4th value of 2nd row\r\n * @param initialM31 defines 1st value of 3rd row\r\n * @param initialM32 defines 2nd value of 3rd row\r\n * @param initialM33 defines 3rd value of 3rd row\r\n * @param initialM34 defines 4th value of 3rd row\r\n * @param initialM41 defines 1st value of 4th row\r\n * @param initialM42 defines 2nd value of 4th row\r\n * @param initialM43 defines 3rd value of 4th row\r\n * @param initialM44 defines 4th value of 4th row\r\n * @returns the new matrix\r\n */\r\n public static FromValues(\r\n initialM11: number,\r\n initialM12: number,\r\n initialM13: number,\r\n initialM14: number,\r\n initialM21: number,\r\n initialM22: number,\r\n initialM23: number,\r\n initialM24: number,\r\n initialM31: number,\r\n initialM32: number,\r\n initialM33: number,\r\n initialM34: number,\r\n initialM41: number,\r\n initialM42: number,\r\n initialM43: number,\r\n initialM44: number\r\n ): Matrix {\r\n const result = new Matrix();\r\n const m = result._m;\r\n m[0] = initialM11;\r\n m[1] = initialM12;\r\n m[2] = initialM13;\r\n m[3] = initialM14;\r\n m[4] = initialM21;\r\n m[5] = initialM22;\r\n m[6] = initialM23;\r\n m[7] = initialM24;\r\n m[8] = initialM31;\r\n m[9] = initialM32;\r\n m[10] = initialM33;\r\n m[11] = initialM34;\r\n m[12] = initialM41;\r\n m[13] = initialM42;\r\n m[14] = initialM43;\r\n m[15] = initialM44;\r\n result.markAsUpdated();\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new matrix composed by merging scale (vector3), rotation (quaternion) and translation (vector3)\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#24\r\n * @param scale defines the scale vector3\r\n * @param rotation defines the rotation quaternion\r\n * @param translation defines the translation vector3\r\n * @returns a new matrix\r\n */\r\n public static Compose(scale: DeepImmutable, rotation: DeepImmutable, translation: DeepImmutable): Matrix {\r\n const result = new Matrix();\r\n Matrix.ComposeToRef(scale, rotation, translation, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets a matrix to a value composed by merging scale (vector3), rotation (quaternion) and translation (vector3)\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#25\r\n * @param scale defines the scale vector3\r\n * @param rotation defines the rotation quaternion\r\n * @param translation defines the translation vector3\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static ComposeToRef(scale: DeepImmutable, rotation: DeepImmutable, translation: DeepImmutable, result: T): T {\r\n const m = result._m;\r\n const x = rotation._x,\r\n y = rotation._y,\r\n z = rotation._z,\r\n w = rotation._w;\r\n const x2 = x + x,\r\n y2 = y + y,\r\n z2 = z + z;\r\n const xx = x * x2,\r\n xy = x * y2,\r\n xz = x * z2;\r\n const yy = y * y2,\r\n yz = y * z2,\r\n zz = z * z2;\r\n const wx = w * x2,\r\n wy = w * y2,\r\n wz = w * z2;\r\n\r\n const sx = scale._x,\r\n sy = scale._y,\r\n sz = scale._z;\r\n\r\n m[0] = (1 - (yy + zz)) * sx;\r\n m[1] = (xy + wz) * sx;\r\n m[2] = (xz - wy) * sx;\r\n m[3] = 0;\r\n\r\n m[4] = (xy - wz) * sy;\r\n m[5] = (1 - (xx + zz)) * sy;\r\n m[6] = (yz + wx) * sy;\r\n m[7] = 0;\r\n\r\n m[8] = (xz + wy) * sz;\r\n m[9] = (yz - wx) * sz;\r\n m[10] = (1 - (xx + yy)) * sz;\r\n m[11] = 0;\r\n\r\n m[12] = translation._x;\r\n m[13] = translation._y;\r\n m[14] = translation._z;\r\n m[15] = 1;\r\n\r\n result.markAsUpdated();\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new identity matrix\r\n * @returns a new identity matrix\r\n */\r\n public static Identity(): Matrix {\r\n const identity = Matrix.FromValues(1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 0.0, 1.0);\r\n identity._updateIdentityStatus(true);\r\n return identity;\r\n }\r\n\r\n /**\r\n * Creates a new identity matrix and stores the result in a given matrix\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static IdentityToRef(result: T): T {\r\n Matrix.FromValuesToRef(1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 0.0, 1.0, result);\r\n result._updateIdentityStatus(true);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new zero matrix\r\n * @returns a new zero matrix\r\n */\r\n public static Zero(): Matrix {\r\n const zero = Matrix.FromValues(0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0);\r\n zero._updateIdentityStatus(false);\r\n return zero;\r\n }\r\n\r\n /**\r\n * Creates a new rotation matrix for \"angle\" radians around the X axis\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#97\r\n * @param angle defines the angle (in radians) to use\r\n * @returns the new matrix\r\n */\r\n public static RotationX(angle: number): Matrix {\r\n const result = new Matrix();\r\n Matrix.RotationXToRef(angle, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new matrix as the invert of a given matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#124\r\n * @param source defines the source matrix\r\n * @returns the new matrix\r\n */\r\n public static Invert(source: DeepImmutable): T {\r\n const result = new (source.constructor as MatrixConstructor)();\r\n source.invertToRef(result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new rotation matrix for \"angle\" radians around the X axis and stores it in a given matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#98\r\n * @param angle defines the angle (in radians) to use\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static RotationXToRef(angle: number, result: T): T {\r\n const s = Math.sin(angle);\r\n const c = Math.cos(angle);\r\n Matrix.FromValuesToRef(1.0, 0.0, 0.0, 0.0, 0.0, c, s, 0.0, 0.0, -s, c, 0.0, 0.0, 0.0, 0.0, 1.0, result);\r\n\r\n result._updateIdentityStatus(c === 1 && s === 0);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new rotation matrix for \"angle\" radians around the Y axis\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#99\r\n * @param angle defines the angle (in radians) to use\r\n * @returns the new matrix\r\n */\r\n public static RotationY(angle: number): Matrix {\r\n const result = new Matrix();\r\n Matrix.RotationYToRef(angle, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new rotation matrix for \"angle\" radians around the Y axis and stores it in a given matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#100\r\n * @param angle defines the angle (in radians) to use\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static RotationYToRef(angle: number, result: T): T {\r\n const s = Math.sin(angle);\r\n const c = Math.cos(angle);\r\n Matrix.FromValuesToRef(c, 0.0, -s, 0.0, 0.0, 1.0, 0.0, 0.0, s, 0.0, c, 0.0, 0.0, 0.0, 0.0, 1.0, result);\r\n\r\n result._updateIdentityStatus(c === 1 && s === 0);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new rotation matrix for \"angle\" radians around the Z axis\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#101\r\n * @param angle defines the angle (in radians) to use\r\n * @returns the new matrix\r\n */\r\n public static RotationZ(angle: number): Matrix {\r\n const result = new Matrix();\r\n Matrix.RotationZToRef(angle, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new rotation matrix for \"angle\" radians around the Z axis and stores it in a given matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#102\r\n * @param angle defines the angle (in radians) to use\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static RotationZToRef(angle: number, result: T): T {\r\n const s = Math.sin(angle);\r\n const c = Math.cos(angle);\r\n Matrix.FromValuesToRef(c, s, 0.0, 0.0, -s, c, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 0.0, 1.0, result);\r\n\r\n result._updateIdentityStatus(c === 1 && s === 0);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new rotation matrix for \"angle\" radians around the given axis\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#96\r\n * @param axis defines the axis to use\r\n * @param angle defines the angle (in radians) to use\r\n * @returns the new matrix\r\n */\r\n public static RotationAxis(axis: DeepImmutable, angle: number): Matrix {\r\n const result = new Matrix();\r\n Matrix.RotationAxisToRef(axis, angle, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new rotation matrix for \"angle\" radians around the given axis and stores it in a given matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#94\r\n * @param axis defines the axis to use\r\n * @param angle defines the angle (in radians) to use\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static RotationAxisToRef(axis: DeepImmutable, angle: number, result: T): T {\r\n const s = Math.sin(-angle);\r\n const c = Math.cos(-angle);\r\n const c1 = 1 - c;\r\n\r\n axis.normalize();\r\n const m = result._m;\r\n m[0] = axis._x * axis._x * c1 + c;\r\n m[1] = axis._x * axis._y * c1 - axis._z * s;\r\n m[2] = axis._x * axis._z * c1 + axis._y * s;\r\n m[3] = 0.0;\r\n\r\n m[4] = axis._y * axis._x * c1 + axis._z * s;\r\n m[5] = axis._y * axis._y * c1 + c;\r\n m[6] = axis._y * axis._z * c1 - axis._x * s;\r\n m[7] = 0.0;\r\n\r\n m[8] = axis._z * axis._x * c1 - axis._y * s;\r\n m[9] = axis._z * axis._y * c1 + axis._x * s;\r\n m[10] = axis._z * axis._z * c1 + c;\r\n m[11] = 0.0;\r\n\r\n m[12] = 0.0;\r\n m[13] = 0.0;\r\n m[14] = 0.0;\r\n m[15] = 1.0;\r\n\r\n result.markAsUpdated();\r\n return result;\r\n }\r\n\r\n /**\r\n * Takes normalised vectors and returns a rotation matrix to align \"from\" with \"to\".\r\n * Taken from http://www.iquilezles.org/www/articles/noacos/noacos.htm\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#93\r\n * @param from defines the vector to align\r\n * @param to defines the vector to align to\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static RotationAlignToRef(from: DeepImmutable, to: DeepImmutable, result: T): T {\r\n const c = Vector3.Dot(to, from);\r\n const m = result._m;\r\n if (c < -1 + Epsilon) {\r\n // from and to are colinear and opposite direction.\r\n // compute a PI rotation on Z axis\r\n m[0] = -1;\r\n m[1] = 0;\r\n m[2] = 0;\r\n m[3] = 0;\r\n m[4] = 0;\r\n m[5] = -1;\r\n m[6] = 0;\r\n m[7] = 0;\r\n m[8] = 0;\r\n m[9] = 0;\r\n m[10] = 1;\r\n m[11] = 0;\r\n } else {\r\n const v = Vector3.Cross(to, from);\r\n const k = 1 / (1 + c);\r\n\r\n m[0] = v._x * v._x * k + c;\r\n m[1] = v._y * v._x * k - v._z;\r\n m[2] = v._z * v._x * k + v._y;\r\n m[3] = 0;\r\n m[4] = v._x * v._y * k + v._z;\r\n m[5] = v._y * v._y * k + c;\r\n m[6] = v._z * v._y * k - v._x;\r\n m[7] = 0;\r\n m[8] = v._x * v._z * k - v._y;\r\n m[9] = v._y * v._z * k + v._x;\r\n m[10] = v._z * v._z * k + c;\r\n m[11] = 0;\r\n }\r\n m[12] = 0;\r\n m[13] = 0;\r\n m[14] = 0;\r\n m[15] = 1;\r\n result.markAsUpdated();\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a rotation matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#103\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#105\r\n * @param yaw defines the yaw angle in radians (Y axis)\r\n * @param pitch defines the pitch angle in radians (X axis)\r\n * @param roll defines the roll angle in radians (Z axis)\r\n * @returns the new rotation matrix\r\n */\r\n public static RotationYawPitchRoll(yaw: number, pitch: number, roll: number): Matrix {\r\n const result = new Matrix();\r\n Matrix.RotationYawPitchRollToRef(yaw, pitch, roll, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a rotation matrix and stores it in a given matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#104\r\n * @param yaw defines the yaw angle in radians (Y axis)\r\n * @param pitch defines the pitch angle in radians (X axis)\r\n * @param roll defines the roll angle in radians (Z axis)\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static RotationYawPitchRollToRef(yaw: number, pitch: number, roll: number, result: T): T {\r\n Quaternion.RotationYawPitchRollToRef(yaw, pitch, roll, MathTmp.Quaternion[0]);\r\n MathTmp.Quaternion[0].toRotationMatrix(result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a scaling matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#107\r\n * @param x defines the scale factor on X axis\r\n * @param y defines the scale factor on Y axis\r\n * @param z defines the scale factor on Z axis\r\n * @returns the new matrix\r\n */\r\n public static Scaling(x: number, y: number, z: number): Matrix {\r\n const result = new Matrix();\r\n Matrix.ScalingToRef(x, y, z, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a scaling matrix and stores it in a given matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#108\r\n * @param x defines the scale factor on X axis\r\n * @param y defines the scale factor on Y axis\r\n * @param z defines the scale factor on Z axis\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static ScalingToRef(x: number, y: number, z: number, result: T): T {\r\n Matrix.FromValuesToRef(x, 0.0, 0.0, 0.0, 0.0, y, 0.0, 0.0, 0.0, 0.0, z, 0.0, 0.0, 0.0, 0.0, 1.0, result);\r\n\r\n result._updateIdentityStatus(x === 1 && y === 1 && z === 1);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a translation matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#109\r\n * @param x defines the translation on X axis\r\n * @param y defines the translation on Y axis\r\n * @param z defines the translationon Z axis\r\n * @returns the new matrix\r\n */\r\n public static Translation(x: number, y: number, z: number): Matrix {\r\n const result = new Matrix();\r\n Matrix.TranslationToRef(x, y, z, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a translation matrix and stores it in a given matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#110\r\n * @param x defines the translation on X axis\r\n * @param y defines the translation on Y axis\r\n * @param z defines the translationon Z axis\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static TranslationToRef(x: number, y: number, z: number, result: T): T {\r\n Matrix.FromValuesToRef(1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, x, y, z, 1.0, result);\r\n result._updateIdentityStatus(x === 0 && y === 0 && z === 0);\r\n return result;\r\n }\r\n\r\n /**\r\n * Returns a new Matrix whose values are the interpolated values for \"gradient\" (float) between the ones of the matrices \"startValue\" and \"endValue\".\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#55\r\n * @param startValue defines the start value\r\n * @param endValue defines the end value\r\n * @param gradient defines the gradient factor\r\n * @returns the new matrix\r\n */\r\n public static Lerp(startValue: DeepImmutable, endValue: DeepImmutable, gradient: number): T {\r\n const result = new (startValue.constructor as MatrixConstructor)();\r\n Matrix.LerpToRef(startValue, endValue, gradient, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Set the given matrix \"result\" as the interpolated values for \"gradient\" (float) between the ones of the matrices \"startValue\" and \"endValue\".\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#54\r\n * @param startValue defines the start value\r\n * @param endValue defines the end value\r\n * @param gradient defines the gradient factor\r\n * @param result defines the Matrix object where to store data\r\n * @returns result input\r\n */\r\n public static LerpToRef(startValue: DeepImmutable, endValue: DeepImmutable, gradient: number, result: T): T {\r\n const resultM = result._m;\r\n const startM = startValue.m;\r\n const endM = endValue.m;\r\n for (let index = 0; index < 16; index++) {\r\n resultM[index] = startM[index] * (1.0 - gradient) + endM[index] * gradient;\r\n }\r\n result.markAsUpdated();\r\n return result;\r\n }\r\n\r\n /**\r\n * Builds a new matrix whose values are computed by:\r\n * * decomposing the the \"startValue\" and \"endValue\" matrices into their respective scale, rotation and translation matrices\r\n * * interpolating for \"gradient\" (float) the values between each of these decomposed matrices between the start and the end\r\n * * recomposing a new matrix from these 3 interpolated scale, rotation and translation matrices\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#22\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#51\r\n * @param startValue defines the first matrix\r\n * @param endValue defines the second matrix\r\n * @param gradient defines the gradient between the two matrices\r\n * @returns the new matrix\r\n */\r\n public static DecomposeLerp(startValue: DeepImmutable, endValue: DeepImmutable, gradient: number): T {\r\n const result = new (startValue.constructor as MatrixConstructor)();\r\n Matrix.DecomposeLerpToRef(startValue, endValue, gradient, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Update a matrix to values which are computed by:\r\n * * decomposing the the \"startValue\" and \"endValue\" matrices into their respective scale, rotation and translation matrices\r\n * * interpolating for \"gradient\" (float) the values between each of these decomposed matrices between the start and the end\r\n * * recomposing a new matrix from these 3 interpolated scale, rotation and translation matrices\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#23\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#53\r\n * @param startValue defines the first matrix\r\n * @param endValue defines the second matrix\r\n * @param gradient defines the gradient between the two matrices\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static DecomposeLerpToRef(startValue: DeepImmutable, endValue: DeepImmutable, gradient: number, result: T): T {\r\n const startScale = MathTmp.Vector3[0];\r\n const startRotation = MathTmp.Quaternion[0];\r\n const startTranslation = MathTmp.Vector3[1];\r\n startValue.decompose(startScale, startRotation, startTranslation);\r\n\r\n const endScale = MathTmp.Vector3[2];\r\n const endRotation = MathTmp.Quaternion[1];\r\n const endTranslation = MathTmp.Vector3[3];\r\n endValue.decompose(endScale, endRotation, endTranslation);\r\n\r\n const resultScale = MathTmp.Vector3[4];\r\n Vector3.LerpToRef(startScale, endScale, gradient, resultScale);\r\n const resultRotation = MathTmp.Quaternion[2];\r\n Quaternion.SlerpToRef(startRotation, endRotation, gradient, resultRotation);\r\n\r\n const resultTranslation = MathTmp.Vector3[5];\r\n Vector3.LerpToRef(startTranslation, endTranslation, gradient, resultTranslation);\r\n\r\n Matrix.ComposeToRef(resultScale, resultRotation, resultTranslation, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new matrix that transforms vertices from world space to camera space. It takes three vectors as arguments that together describe the position and orientation of the camera.\r\n * This function generates a matrix suitable for a left handed coordinate system\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#58\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#59\r\n * @param eye defines the final position of the entity\r\n * @param target defines where the entity should look at\r\n * @param up defines the up vector for the entity\r\n * @returns the new matrix\r\n */\r\n public static LookAtLH(eye: DeepImmutable, target: DeepImmutable, up: DeepImmutable): Matrix {\r\n const result = new Matrix();\r\n Matrix.LookAtLHToRef(eye, target, up, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the given \"result\" Matrix to a matrix that transforms vertices from world space to camera space. It takes three vectors as arguments that together describe the position and orientation of the camera.\r\n * This function generates a matrix suitable for a left handed coordinate system\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#60\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#61\r\n * @param eye defines the final position of the entity\r\n * @param target defines where the entity should look at\r\n * @param up defines the up vector for the entity\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static LookAtLHToRef(eye: DeepImmutable, target: DeepImmutable, up: DeepImmutable, result: Matrix): void {\r\n const xAxis = MathTmp.Vector3[0];\r\n const yAxis = MathTmp.Vector3[1];\r\n const zAxis = MathTmp.Vector3[2];\r\n\r\n // Z axis\r\n target.subtractToRef(eye, zAxis);\r\n zAxis.normalize();\r\n\r\n // X axis\r\n Vector3.CrossToRef(up, zAxis, xAxis);\r\n\r\n const xSquareLength = xAxis.lengthSquared();\r\n if (xSquareLength === 0) {\r\n xAxis.x = 1.0;\r\n } else {\r\n xAxis.normalizeFromLength(Math.sqrt(xSquareLength));\r\n }\r\n\r\n // Y axis\r\n Vector3.CrossToRef(zAxis, xAxis, yAxis);\r\n yAxis.normalize();\r\n\r\n // Eye angles\r\n const ex = -Vector3.Dot(xAxis, eye);\r\n const ey = -Vector3.Dot(yAxis, eye);\r\n const ez = -Vector3.Dot(zAxis, eye);\r\n\r\n Matrix.FromValuesToRef(xAxis._x, yAxis._x, zAxis._x, 0.0, xAxis._y, yAxis._y, zAxis._y, 0.0, xAxis._z, yAxis._z, zAxis._z, 0.0, ex, ey, ez, 1.0, result);\r\n }\r\n\r\n /**\r\n * Creates a new matrix that transforms vertices from world space to camera space. It takes three vectors as arguments that together describe the position and orientation of the camera.\r\n * This function generates a matrix suitable for a right handed coordinate system\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#62\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#63\r\n * @param eye defines the final position of the entity\r\n * @param target defines where the entity should look at\r\n * @param up defines the up vector for the entity\r\n * @returns the new matrix\r\n */\r\n public static LookAtRH(eye: DeepImmutable, target: DeepImmutable, up: DeepImmutable): Matrix {\r\n const result = new Matrix();\r\n Matrix.LookAtRHToRef(eye, target, up, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the given \"result\" Matrix to a matrix that transforms vertices from world space to camera space. It takes three vectors as arguments that together describe the position and orientation of the camera.\r\n * This function generates a matrix suitable for a right handed coordinate system\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#64\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#65\r\n * @param eye defines the final position of the entity\r\n * @param target defines where the entity should look at\r\n * @param up defines the up vector for the entity\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static LookAtRHToRef(eye: DeepImmutable, target: DeepImmutable, up: DeepImmutable, result: T): T {\r\n const xAxis = MathTmp.Vector3[0];\r\n const yAxis = MathTmp.Vector3[1];\r\n const zAxis = MathTmp.Vector3[2];\r\n\r\n // Z axis\r\n eye.subtractToRef(target, zAxis);\r\n zAxis.normalize();\r\n\r\n // X axis\r\n Vector3.CrossToRef(up, zAxis, xAxis);\r\n\r\n const xSquareLength = xAxis.lengthSquared();\r\n if (xSquareLength === 0) {\r\n xAxis.x = 1.0;\r\n } else {\r\n xAxis.normalizeFromLength(Math.sqrt(xSquareLength));\r\n }\r\n\r\n // Y axis\r\n Vector3.CrossToRef(zAxis, xAxis, yAxis);\r\n yAxis.normalize();\r\n\r\n // Eye angles\r\n const ex = -Vector3.Dot(xAxis, eye);\r\n const ey = -Vector3.Dot(yAxis, eye);\r\n const ez = -Vector3.Dot(zAxis, eye);\r\n\r\n Matrix.FromValuesToRef(xAxis._x, yAxis._x, zAxis._x, 0.0, xAxis._y, yAxis._y, zAxis._y, 0.0, xAxis._z, yAxis._z, zAxis._z, 0.0, ex, ey, ez, 1.0, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new matrix that transforms vertices from world space to camera space. It takes two vectors as arguments that together describe the orientation of the camera. The position is assumed to be at the origin (0,0,0)\r\n * This function generates a matrix suitable for a left handed coordinate system\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#66\r\n * @param forward defines the forward direction - Must be normalized and orthogonal to up.\r\n * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward.\r\n * @returns the new matrix\r\n */\r\n public static LookDirectionLH(forward: DeepImmutable, up: DeepImmutable): Matrix {\r\n const result = new Matrix();\r\n Matrix.LookDirectionLHToRef(forward, up, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the given \"result\" Matrix to a matrix that transforms vertices from world space to camera space. It takes two vectors as arguments that together describe the orientation of the camera. The position is assumed to be at the origin (0,0,0)\r\n * This function generates a matrix suitable for a left handed coordinate system\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#67\r\n * @param forward defines the forward direction - Must be normalized and orthogonal to up.\r\n * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward.\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static LookDirectionLHToRef(forward: DeepImmutable, up: DeepImmutable, result: T): T {\r\n const back = MathTmp.Vector3[0];\r\n back.copyFrom(forward);\r\n back.scaleInPlace(-1);\r\n const left = MathTmp.Vector3[1];\r\n Vector3.CrossToRef(up, back, left);\r\n\r\n // Generate the rotation matrix.\r\n Matrix.FromValuesToRef(left._x, left._y, left._z, 0.0, up._x, up._y, up._z, 0.0, back._x, back._y, back._z, 0.0, 0, 0, 0, 1.0, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new matrix that transforms vertices from world space to camera space. It takes two vectors as arguments that together describe the orientation of the camera. The position is assumed to be at the origin (0,0,0)\r\n * This function generates a matrix suitable for a right handed coordinate system\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#68\r\n * @param forward defines the forward direction - Must be normalized and orthogonal to up.\r\n * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward.\r\n * @returns the new matrix\r\n */\r\n public static LookDirectionRH(forward: DeepImmutable, up: DeepImmutable): Matrix {\r\n const result = new Matrix();\r\n Matrix.LookDirectionRHToRef(forward, up, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the given \"result\" Matrix to a matrix that transforms vertices from world space to camera space. It takes two vectors as arguments that together describe the orientation of the camera. The position is assumed to be at the origin (0,0,0)\r\n * This function generates a matrix suitable for a right handed coordinate system\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#69\r\n * @param forward defines the forward direction - Must be normalized and orthogonal to up.\r\n * @param up defines the up vector for the entity - Must be normalized and orthogonal to forward.\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static LookDirectionRHToRef(forward: DeepImmutable, up: DeepImmutable, result: T): T {\r\n const right = MathTmp.Vector3[2];\r\n Vector3.CrossToRef(up, forward, right);\r\n\r\n // Generate the rotation matrix.\r\n Matrix.FromValuesToRef(right._x, right._y, right._z, 0.0, up._x, up._y, up._z, 0.0, forward._x, forward._y, forward._z, 0.0, 0, 0, 0, 1.0, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Create a left-handed orthographic projection matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#70\r\n * @param width defines the viewport width\r\n * @param height defines the viewport height\r\n * @param znear defines the near clip plane\r\n * @param zfar defines the far clip plane\r\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\r\n * @returns a new matrix as a left-handed orthographic projection matrix\r\n */\r\n public static OrthoLH(width: number, height: number, znear: number, zfar: number, halfZRange?: boolean): Matrix {\r\n const matrix = new Matrix();\r\n Matrix.OrthoLHToRef(width, height, znear, zfar, matrix, halfZRange);\r\n return matrix;\r\n }\r\n\r\n /**\r\n * Store a left-handed orthographic projection to a given matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#71\r\n * @param width defines the viewport width\r\n * @param height defines the viewport height\r\n * @param znear defines the near clip plane\r\n * @param zfar defines the far clip plane\r\n * @param result defines the target matrix\r\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\r\n * @returns result input\r\n */\r\n public static OrthoLHToRef(width: number, height: number, znear: number, zfar: number, result: T, halfZRange?: boolean): T {\r\n const n = znear;\r\n const f = zfar;\r\n\r\n const a = 2.0 / width;\r\n const b = 2.0 / height;\r\n const c = 2.0 / (f - n);\r\n const d = -(f + n) / (f - n);\r\n\r\n Matrix.FromValuesToRef(a, 0.0, 0.0, 0.0, 0.0, b, 0.0, 0.0, 0.0, 0.0, c, 0.0, 0.0, 0.0, d, 1.0, result);\r\n\r\n if (halfZRange) {\r\n result.multiplyToRef(mtxConvertNDCToHalfZRange, result);\r\n }\r\n\r\n result._updateIdentityStatus(a === 1 && b === 1 && c === 1 && d === 0);\r\n return result;\r\n }\r\n\r\n /**\r\n * Create a left-handed orthographic projection matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#72\r\n * @param left defines the viewport left coordinate\r\n * @param right defines the viewport right coordinate\r\n * @param bottom defines the viewport bottom coordinate\r\n * @param top defines the viewport top coordinate\r\n * @param znear defines the near clip plane\r\n * @param zfar defines the far clip plane\r\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\r\n * @returns a new matrix as a left-handed orthographic projection matrix\r\n */\r\n public static OrthoOffCenterLH(left: number, right: number, bottom: number, top: number, znear: number, zfar: number, halfZRange?: boolean): Matrix {\r\n const matrix = new Matrix();\r\n Matrix.OrthoOffCenterLHToRef(left, right, bottom, top, znear, zfar, matrix, halfZRange);\r\n return matrix;\r\n }\r\n\r\n /**\r\n * Stores a left-handed orthographic projection into a given matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#73\r\n * @param left defines the viewport left coordinate\r\n * @param right defines the viewport right coordinate\r\n * @param bottom defines the viewport bottom coordinate\r\n * @param top defines the viewport top coordinate\r\n * @param znear defines the near clip plane\r\n * @param zfar defines the far clip plane\r\n * @param result defines the target matrix\r\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\r\n * @returns result input\r\n */\r\n public static OrthoOffCenterLHToRef(\r\n left: number,\r\n right: number,\r\n bottom: number,\r\n top: number,\r\n znear: number,\r\n zfar: number,\r\n result: T,\r\n halfZRange?: boolean\r\n ): T {\r\n const n = znear;\r\n const f = zfar;\r\n\r\n const a = 2.0 / (right - left);\r\n const b = 2.0 / (top - bottom);\r\n const c = 2.0 / (f - n);\r\n const d = -(f + n) / (f - n);\r\n const i0 = (left + right) / (left - right);\r\n const i1 = (top + bottom) / (bottom - top);\r\n\r\n Matrix.FromValuesToRef(a, 0.0, 0.0, 0.0, 0.0, b, 0.0, 0.0, 0.0, 0.0, c, 0.0, i0, i1, d, 1.0, result);\r\n\r\n if (halfZRange) {\r\n result.multiplyToRef(mtxConvertNDCToHalfZRange, result);\r\n }\r\n\r\n result.markAsUpdated();\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a right-handed orthographic projection matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#76\r\n * @param left defines the viewport left coordinate\r\n * @param right defines the viewport right coordinate\r\n * @param bottom defines the viewport bottom coordinate\r\n * @param top defines the viewport top coordinate\r\n * @param znear defines the near clip plane\r\n * @param zfar defines the far clip plane\r\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\r\n * @returns a new matrix as a right-handed orthographic projection matrix\r\n */\r\n public static OrthoOffCenterRH(left: number, right: number, bottom: number, top: number, znear: number, zfar: number, halfZRange?: boolean): Matrix {\r\n const matrix = new Matrix();\r\n Matrix.OrthoOffCenterRHToRef(left, right, bottom, top, znear, zfar, matrix, halfZRange);\r\n return matrix;\r\n }\r\n\r\n /**\r\n * Stores a right-handed orthographic projection into a given matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#77\r\n * @param left defines the viewport left coordinate\r\n * @param right defines the viewport right coordinate\r\n * @param bottom defines the viewport bottom coordinate\r\n * @param top defines the viewport top coordinate\r\n * @param znear defines the near clip plane\r\n * @param zfar defines the far clip plane\r\n * @param result defines the target matrix\r\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\r\n * @returns result input\r\n */\r\n public static OrthoOffCenterRHToRef(\r\n left: number,\r\n right: number,\r\n bottom: number,\r\n top: number,\r\n znear: number,\r\n zfar: number,\r\n result: T,\r\n halfZRange?: boolean\r\n ): T {\r\n Matrix.OrthoOffCenterLHToRef(left, right, bottom, top, znear, zfar, result, halfZRange);\r\n result._m[10] *= -1; // No need to call markAsUpdated as previous function already called it and let _isIdentityDirty to true\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a left-handed perspective projection matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#85\r\n * @param width defines the viewport width\r\n * @param height defines the viewport height\r\n * @param znear defines the near clip plane\r\n * @param zfar defines the far clip plane\r\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\r\n * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal)\r\n * @returns a new matrix as a left-handed perspective projection matrix\r\n */\r\n public static PerspectiveLH(width: number, height: number, znear: number, zfar: number, halfZRange?: boolean, projectionPlaneTilt: number = 0): Matrix {\r\n const matrix = new Matrix();\r\n\r\n const n = znear;\r\n const f = zfar;\r\n\r\n const a = (2.0 * n) / width;\r\n const b = (2.0 * n) / height;\r\n const c = (f + n) / (f - n);\r\n const d = (-2.0 * f * n) / (f - n);\r\n const rot = Math.tan(projectionPlaneTilt);\r\n\r\n Matrix.FromValuesToRef(a, 0.0, 0.0, 0.0, 0.0, b, 0.0, rot, 0.0, 0.0, c, 1.0, 0.0, 0.0, d, 0.0, matrix);\r\n\r\n if (halfZRange) {\r\n matrix.multiplyToRef(mtxConvertNDCToHalfZRange, matrix);\r\n }\r\n\r\n matrix._updateIdentityStatus(false);\r\n return matrix;\r\n }\r\n\r\n /**\r\n * Creates a left-handed perspective projection matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#78\r\n * @param fov defines the horizontal field of view\r\n * @param aspect defines the aspect ratio\r\n * @param znear defines the near clip plane\r\n * @param zfar defines the far clip plane. If 0, assume we are in \"infinite zfar\" mode\r\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\r\n * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal)\r\n * @param reverseDepthBufferMode true to indicate that we are in a reverse depth buffer mode (meaning znear and zfar have been inverted when calling the function)\r\n * @returns a new matrix as a left-handed perspective projection matrix\r\n */\r\n public static PerspectiveFovLH(\r\n fov: number,\r\n aspect: number,\r\n znear: number,\r\n zfar: number,\r\n halfZRange?: boolean,\r\n projectionPlaneTilt: number = 0,\r\n reverseDepthBufferMode: boolean = false\r\n ): Matrix {\r\n const matrix = new Matrix();\r\n Matrix.PerspectiveFovLHToRef(fov, aspect, znear, zfar, matrix, true, halfZRange, projectionPlaneTilt, reverseDepthBufferMode);\r\n return matrix;\r\n }\r\n\r\n /**\r\n * Stores a left-handed perspective projection into a given matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#81\r\n * @param fov defines the horizontal field of view\r\n * @param aspect defines the aspect ratio\r\n * @param znear defines the near clip plane\r\n * @param zfar defines the far clip plane. If 0, assume we are in \"infinite zfar\" mode\r\n * @param result defines the target matrix\r\n * @param isVerticalFovFixed defines it the fov is vertically fixed (default) or horizontally\r\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\r\n * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal)\r\n * @param reverseDepthBufferMode true to indicate that we are in a reverse depth buffer mode (meaning znear and zfar have been inverted when calling the function)\r\n * @returns result input\r\n */\r\n public static PerspectiveFovLHToRef(\r\n fov: number,\r\n aspect: number,\r\n znear: number,\r\n zfar: number,\r\n result: T,\r\n isVerticalFovFixed = true,\r\n halfZRange?: boolean,\r\n projectionPlaneTilt: number = 0,\r\n reverseDepthBufferMode: boolean = false\r\n ): T {\r\n const n = znear;\r\n const f = zfar;\r\n\r\n const t = 1.0 / Math.tan(fov * 0.5);\r\n const a = isVerticalFovFixed ? t / aspect : t;\r\n const b = isVerticalFovFixed ? t : t * aspect;\r\n const c = reverseDepthBufferMode && n === 0 ? -1 : f !== 0 ? (f + n) / (f - n) : 1;\r\n const d = reverseDepthBufferMode && n === 0 ? 2 * f : f !== 0 ? (-2.0 * f * n) / (f - n) : -2 * n;\r\n const rot = Math.tan(projectionPlaneTilt);\r\n\r\n Matrix.FromValuesToRef(a, 0.0, 0.0, 0.0, 0.0, b, 0.0, rot, 0.0, 0.0, c, 1.0, 0.0, 0.0, d, 0.0, result);\r\n\r\n if (halfZRange) {\r\n result.multiplyToRef(mtxConvertNDCToHalfZRange, result);\r\n }\r\n\r\n result._updateIdentityStatus(false);\r\n return result;\r\n }\r\n\r\n /**\r\n * Stores a left-handed perspective projection into a given matrix with depth reversed\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#89\r\n * @param fov defines the horizontal field of view\r\n * @param aspect defines the aspect ratio\r\n * @param znear defines the near clip plane\r\n * @param zfar not used as infinity is used as far clip\r\n * @param result defines the target matrix\r\n * @param isVerticalFovFixed defines it the fov is vertically fixed (default) or horizontally\r\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\r\n * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal)\r\n * @returns result input\r\n */\r\n public static PerspectiveFovReverseLHToRef(\r\n fov: number,\r\n aspect: number,\r\n znear: number,\r\n zfar: number,\r\n result: T,\r\n isVerticalFovFixed = true,\r\n halfZRange?: boolean,\r\n projectionPlaneTilt: number = 0\r\n ): T {\r\n const t = 1.0 / Math.tan(fov * 0.5);\r\n const a = isVerticalFovFixed ? t / aspect : t;\r\n const b = isVerticalFovFixed ? t : t * aspect;\r\n const rot = Math.tan(projectionPlaneTilt);\r\n\r\n Matrix.FromValuesToRef(a, 0.0, 0.0, 0.0, 0.0, b, 0.0, rot, 0.0, 0.0, -znear, 1.0, 0.0, 0.0, 1.0, 0.0, result);\r\n if (halfZRange) {\r\n result.multiplyToRef(mtxConvertNDCToHalfZRange, result);\r\n }\r\n result._updateIdentityStatus(false);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a right-handed perspective projection matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#83\r\n * @param fov defines the horizontal field of view\r\n * @param aspect defines the aspect ratio\r\n * @param znear defines the near clip plane\r\n * @param zfar defines the far clip plane. If 0, assume we are in \"infinite zfar\" mode\r\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\r\n * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal)\r\n * @param reverseDepthBufferMode true to indicate that we are in a reverse depth buffer mode (meaning znear and zfar have been inverted when calling the function)\r\n * @returns a new matrix as a right-handed perspective projection matrix\r\n */\r\n public static PerspectiveFovRH(\r\n fov: number,\r\n aspect: number,\r\n znear: number,\r\n zfar: number,\r\n halfZRange?: boolean,\r\n projectionPlaneTilt: number = 0,\r\n reverseDepthBufferMode: boolean = false\r\n ): Matrix {\r\n const matrix = new Matrix();\r\n Matrix.PerspectiveFovRHToRef(fov, aspect, znear, zfar, matrix, true, halfZRange, projectionPlaneTilt, reverseDepthBufferMode);\r\n return matrix;\r\n }\r\n\r\n /**\r\n * Stores a right-handed perspective projection into a given matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#84\r\n * @param fov defines the horizontal field of view\r\n * @param aspect defines the aspect ratio\r\n * @param znear defines the near clip plane\r\n * @param zfar defines the far clip plane. If 0, assume we are in \"infinite zfar\" mode\r\n * @param result defines the target matrix\r\n * @param isVerticalFovFixed defines it the fov is vertically fixed (default) or horizontally\r\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\r\n * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal)\r\n * @param reverseDepthBufferMode true to indicate that we are in a reverse depth buffer mode (meaning znear and zfar have been inverted when calling the function)\r\n * @returns result input\r\n */\r\n public static PerspectiveFovRHToRef(\r\n fov: number,\r\n aspect: number,\r\n znear: number,\r\n zfar: number,\r\n result: T,\r\n isVerticalFovFixed = true,\r\n halfZRange?: boolean,\r\n projectionPlaneTilt: number = 0,\r\n reverseDepthBufferMode: boolean = false\r\n ): T {\r\n //alternatively this could be expressed as:\r\n // m = PerspectiveFovLHToRef\r\n // m[10] *= -1.0;\r\n // m[11] *= -1.0;\r\n\r\n const n = znear;\r\n const f = zfar;\r\n\r\n const t = 1.0 / Math.tan(fov * 0.5);\r\n const a = isVerticalFovFixed ? t / aspect : t;\r\n const b = isVerticalFovFixed ? t : t * aspect;\r\n const c = reverseDepthBufferMode && n === 0 ? 1 : f !== 0 ? -(f + n) / (f - n) : -1;\r\n const d = reverseDepthBufferMode && n === 0 ? 2 * f : f !== 0 ? (-2 * f * n) / (f - n) : -2 * n;\r\n const rot = Math.tan(projectionPlaneTilt);\r\n\r\n Matrix.FromValuesToRef(a, 0.0, 0.0, 0.0, 0.0, b, 0.0, rot, 0.0, 0.0, c, -1.0, 0.0, 0.0, d, 0.0, result);\r\n\r\n if (halfZRange) {\r\n result.multiplyToRef(mtxConvertNDCToHalfZRange, result);\r\n }\r\n\r\n result._updateIdentityStatus(false);\r\n return result;\r\n }\r\n\r\n /**\r\n * Stores a right-handed perspective projection into a given matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#90\r\n * @param fov defines the horizontal field of view\r\n * @param aspect defines the aspect ratio\r\n * @param znear defines the near clip plane\r\n * @param zfar not used as infinity is used as far clip\r\n * @param result defines the target matrix\r\n * @param isVerticalFovFixed defines it the fov is vertically fixed (default) or horizontally\r\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\r\n * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal)\r\n * @returns result input\r\n */\r\n public static PerspectiveFovReverseRHToRef(\r\n fov: number,\r\n aspect: number,\r\n znear: number,\r\n zfar: number,\r\n result: T,\r\n isVerticalFovFixed = true,\r\n halfZRange?: boolean,\r\n projectionPlaneTilt: number = 0\r\n ): T {\r\n const t = 1.0 / Math.tan(fov * 0.5);\r\n const a = isVerticalFovFixed ? t / aspect : t;\r\n const b = isVerticalFovFixed ? t : t * aspect;\r\n const rot = Math.tan(projectionPlaneTilt);\r\n\r\n Matrix.FromValuesToRef(a, 0.0, 0.0, 0.0, 0.0, b, 0.0, rot, 0.0, 0.0, -znear, -1.0, 0.0, 0.0, -1.0, 0.0, result);\r\n\r\n if (halfZRange) {\r\n result.multiplyToRef(mtxConvertNDCToHalfZRange, result);\r\n }\r\n\r\n result._updateIdentityStatus(false);\r\n return result;\r\n }\r\n\r\n /**\r\n * Stores a perspective projection for WebVR info a given matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#92\r\n * @param fov defines the field of view\r\n * @param fov.upDegrees\r\n * @param fov.downDegrees\r\n * @param fov.leftDegrees\r\n * @param fov.rightDegrees\r\n * @param znear defines the near clip plane\r\n * @param zfar defines the far clip plane\r\n * @param result defines the target matrix\r\n * @param rightHanded defines if the matrix must be in right-handed mode (false by default)\r\n * @param halfZRange true to generate NDC coordinates between 0 and 1 instead of -1 and 1 (default: false)\r\n * @param projectionPlaneTilt optional tilt angle of the projection plane around the X axis (horizontal)\r\n * @returns result input\r\n */\r\n public static PerspectiveFovWebVRToRef(\r\n fov: { upDegrees: number; downDegrees: number; leftDegrees: number; rightDegrees: number },\r\n znear: number,\r\n zfar: number,\r\n result: T,\r\n rightHanded = false,\r\n halfZRange?: boolean,\r\n projectionPlaneTilt: number = 0\r\n ): T {\r\n const rightHandedFactor = rightHanded ? -1 : 1;\r\n\r\n const upTan = Math.tan((fov.upDegrees * Math.PI) / 180.0);\r\n const downTan = Math.tan((fov.downDegrees * Math.PI) / 180.0);\r\n const leftTan = Math.tan((fov.leftDegrees * Math.PI) / 180.0);\r\n const rightTan = Math.tan((fov.rightDegrees * Math.PI) / 180.0);\r\n const xScale = 2.0 / (leftTan + rightTan);\r\n const yScale = 2.0 / (upTan + downTan);\r\n const rot = Math.tan(projectionPlaneTilt);\r\n\r\n const m = result._m;\r\n m[0] = xScale;\r\n m[1] = m[2] = m[3] = m[4] = 0.0;\r\n m[5] = yScale;\r\n m[6] = 0.0;\r\n m[7] = rot;\r\n m[8] = (leftTan - rightTan) * xScale * 0.5;\r\n m[9] = -((upTan - downTan) * yScale * 0.5);\r\n m[10] = -zfar / (znear - zfar);\r\n m[11] = 1.0 * rightHandedFactor;\r\n m[12] = m[13] = m[15] = 0.0;\r\n m[14] = -(2.0 * zfar * znear) / (zfar - znear);\r\n\r\n if (halfZRange) {\r\n result.multiplyToRef(mtxConvertNDCToHalfZRange, result);\r\n }\r\n\r\n result.markAsUpdated();\r\n return result;\r\n }\r\n\r\n /**\r\n * Computes a complete transformation matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#113\r\n * @param viewport defines the viewport to use\r\n * @param world defines the world matrix\r\n * @param view defines the view matrix\r\n * @param projection defines the projection matrix\r\n * @param zmin defines the near clip plane\r\n * @param zmax defines the far clip plane\r\n * @returns the transformation matrix\r\n */\r\n public static GetFinalMatrix(\r\n viewport: DeepImmutable,\r\n world: DeepImmutable,\r\n view: DeepImmutable,\r\n projection: DeepImmutable,\r\n zmin: number,\r\n zmax: number\r\n ): T {\r\n const cw = viewport.width;\r\n const ch = viewport.height;\r\n const cx = viewport.x;\r\n const cy = viewport.y;\r\n\r\n const viewportMatrix = Matrix.FromValues(cw / 2.0, 0.0, 0.0, 0.0, 0.0, -ch / 2.0, 0.0, 0.0, 0.0, 0.0, zmax - zmin, 0.0, cx + cw / 2.0, ch / 2.0 + cy, zmin, 1.0);\r\n\r\n const matrix = new (world.constructor as MatrixConstructor)();\r\n world.multiplyToRef(view, matrix);\r\n matrix.multiplyToRef(projection, matrix);\r\n return matrix.multiplyToRef(viewportMatrix, matrix);\r\n }\r\n\r\n /**\r\n * Extracts a 2x2 matrix from a given matrix and store the result in a Float32Array\r\n * @param matrix defines the matrix to use\r\n * @returns a new Float32Array array with 4 elements : the 2x2 matrix extracted from the given matrix\r\n */\r\n public static GetAsMatrix2x2(matrix: DeepImmutable): Float32Array | Array {\r\n const m = matrix.m;\r\n const arr = [m[0], m[1], m[4], m[5]];\r\n return PerformanceConfigurator.MatrixUse64Bits ? arr : new Float32Array(arr);\r\n }\r\n /**\r\n * Extracts a 3x3 matrix from a given matrix and store the result in a Float32Array\r\n * @param matrix defines the matrix to use\r\n * @returns a new Float32Array array with 9 elements : the 3x3 matrix extracted from the given matrix\r\n */\r\n public static GetAsMatrix3x3(matrix: DeepImmutable): Float32Array | Array {\r\n const m = matrix.m;\r\n const arr = [m[0], m[1], m[2], m[4], m[5], m[6], m[8], m[9], m[10]];\r\n return PerformanceConfigurator.MatrixUse64Bits ? arr : new Float32Array(arr);\r\n }\r\n\r\n /**\r\n * Compute the transpose of a given matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#111\r\n * @param matrix defines the matrix to transpose\r\n * @returns the new matrix\r\n */\r\n public static Transpose(matrix: DeepImmutable): T {\r\n const result = new (matrix.constructor as MatrixConstructor)();\r\n Matrix.TransposeToRef(matrix, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Compute the transpose of a matrix and store it in a target matrix\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#112\r\n * @param matrix defines the matrix to transpose\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static TransposeToRef(matrix: DeepImmutable, result: T): T {\r\n const rm = result._m;\r\n const mm = matrix.m;\r\n rm[0] = mm[0];\r\n rm[1] = mm[4];\r\n rm[2] = mm[8];\r\n rm[3] = mm[12];\r\n\r\n rm[4] = mm[1];\r\n rm[5] = mm[5];\r\n rm[6] = mm[9];\r\n rm[7] = mm[13];\r\n\r\n rm[8] = mm[2];\r\n rm[9] = mm[6];\r\n rm[10] = mm[10];\r\n rm[11] = mm[14];\r\n\r\n rm[12] = mm[3];\r\n rm[13] = mm[7];\r\n rm[14] = mm[11];\r\n rm[15] = mm[15];\r\n\r\n result.markAsUpdated();\r\n\r\n // identity-ness does not change when transposing\r\n result._updateIdentityStatus((matrix as Matrix)._isIdentity, (matrix as Matrix)._isIdentityDirty);\r\n return result;\r\n }\r\n\r\n /**\r\n * Computes a reflection matrix from a plane\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#87\r\n * @param plane defines the reflection plane\r\n * @returns a new matrix\r\n */\r\n public static Reflection(plane: DeepImmutable): Matrix {\r\n const matrix = new Matrix();\r\n Matrix.ReflectionToRef(plane, matrix);\r\n return matrix;\r\n }\r\n\r\n /**\r\n * Computes a reflection matrix from a plane\r\n * Example Playground - https://playground.babylonjs.com/#AV9X17#88\r\n * @param plane defines the reflection plane\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static ReflectionToRef(plane: DeepImmutable, result: T): T {\r\n plane.normalize();\r\n const x = plane.normal.x;\r\n const y = plane.normal.y;\r\n const z = plane.normal.z;\r\n const temp = -2 * x;\r\n const temp2 = -2 * y;\r\n const temp3 = -2 * z;\r\n Matrix.FromValuesToRef(\r\n temp * x + 1,\r\n temp2 * x,\r\n temp3 * x,\r\n 0.0,\r\n temp * y,\r\n temp2 * y + 1,\r\n temp3 * y,\r\n 0.0,\r\n temp * z,\r\n temp2 * z,\r\n temp3 * z + 1,\r\n 0.0,\r\n temp * plane.d,\r\n temp2 * plane.d,\r\n temp3 * plane.d,\r\n 1.0,\r\n result\r\n );\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the given matrix as a rotation matrix composed from the 3 left handed axes\r\n * @param xaxis defines the value of the 1st axis\r\n * @param yaxis defines the value of the 2nd axis\r\n * @param zaxis defines the value of the 3rd axis\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static FromXYZAxesToRef(xaxis: DeepImmutable, yaxis: DeepImmutable, zaxis: DeepImmutable, result: T): T {\r\n Matrix.FromValuesToRef(xaxis._x, xaxis._y, xaxis._z, 0.0, yaxis._x, yaxis._y, yaxis._z, 0.0, zaxis._x, zaxis._y, zaxis._z, 0.0, 0.0, 0.0, 0.0, 1.0, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a rotation matrix from a quaternion and stores it in a target matrix\r\n * @param quat defines the quaternion to use\r\n * @param result defines the target matrix\r\n * @returns result input\r\n */\r\n public static FromQuaternionToRef(quat: DeepImmutable, result: T): T {\r\n const xx = quat._x * quat._x;\r\n const yy = quat._y * quat._y;\r\n const zz = quat._z * quat._z;\r\n const xy = quat._x * quat._y;\r\n const zw = quat._z * quat._w;\r\n const zx = quat._z * quat._x;\r\n const yw = quat._y * quat._w;\r\n const yz = quat._y * quat._z;\r\n const xw = quat._x * quat._w;\r\n\r\n result._m[0] = 1.0 - 2.0 * (yy + zz);\r\n result._m[1] = 2.0 * (xy + zw);\r\n result._m[2] = 2.0 * (zx - yw);\r\n result._m[3] = 0.0;\r\n\r\n result._m[4] = 2.0 * (xy - zw);\r\n result._m[5] = 1.0 - 2.0 * (zz + xx);\r\n result._m[6] = 2.0 * (yz + xw);\r\n result._m[7] = 0.0;\r\n\r\n result._m[8] = 2.0 * (zx + yw);\r\n result._m[9] = 2.0 * (yz - xw);\r\n result._m[10] = 1.0 - 2.0 * (yy + xx);\r\n result._m[11] = 0.0;\r\n\r\n result._m[12] = 0.0;\r\n result._m[13] = 0.0;\r\n result._m[14] = 0.0;\r\n result._m[15] = 1.0;\r\n\r\n result.markAsUpdated();\r\n return result;\r\n }\r\n}\r\n\r\n/**\r\n * @internal\r\n * Same as Tmp but not exported to keep it only for math functions to avoid conflicts\r\n */\r\nclass MathTmp {\r\n public static Vector3 = ArrayTools.BuildTuple(11, Vector3.Zero);\r\n public static Matrix = ArrayTools.BuildTuple(2, Matrix.Identity);\r\n public static Quaternion = ArrayTools.BuildTuple(3, Quaternion.Zero);\r\n}\r\n\r\n/**\r\n * @internal\r\n */\r\nexport class TmpVectors {\r\n public static Vector2 = ArrayTools.BuildTuple(3, Vector2.Zero); // 3 temp Vector2 at once should be enough\r\n public static Vector3 = ArrayTools.BuildTuple(13, Vector3.Zero); // 13 temp Vector3 at once should be enough\r\n public static Vector4 = ArrayTools.BuildTuple(3, Vector4.Zero); // 3 temp Vector4 at once should be enough\r\n public static Quaternion = ArrayTools.BuildTuple(2, Quaternion.Zero); // 2 temp Quaternion at once should be enough\r\n public static Matrix = ArrayTools.BuildTuple(8, Matrix.Identity); // 8 temp Matrices at once should be enough\r\n}\r\n\r\nRegisterClass(\"BABYLON.Vector2\", Vector2);\r\nRegisterClass(\"BABYLON.Vector3\", Vector3);\r\nRegisterClass(\"BABYLON.Vector4\", Vector4);\r\nRegisterClass(\"BABYLON.Matrix\", Matrix);\r\n\r\nconst mtxConvertNDCToHalfZRange = Matrix.FromValues(1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0.5, 0, 0, 0, 0.5, 1);\r\n","/**\r\n * @internal\r\n */\r\nexport function _WarnImport(name: string) {\r\n return `${name} needs to be imported before as it contains a side-effect required by your code.`;\r\n}\r\n","import type { DeepImmutable, FloatArray } from \"../types\";\r\nimport { Scalar } from \"./math.scalar\";\r\nimport { ToLinearSpace, ToGammaSpace } from \"./math.constants\";\r\nimport { ArrayTools } from \"../Misc/arrayTools\";\r\nimport { RegisterClass } from \"../Misc/typeStore\";\r\n\r\nfunction colorChannelToLinearSpace(color: number): number {\r\n return Math.pow(color, ToLinearSpace);\r\n}\r\n\r\nfunction colorChannelToLinearSpaceExact(color: number): number {\r\n if (color <= 0.04045) {\r\n return 0.0773993808 * color;\r\n }\r\n return Math.pow(0.947867299 * (color + 0.055), 2.4);\r\n}\r\n\r\nfunction colorChannelToGammaSpace(color: number): number {\r\n return Math.pow(color, ToGammaSpace);\r\n}\r\n\r\nfunction colorChannelToGammaSpaceExact(color: number): number {\r\n if (color <= 0.0031308) {\r\n return 12.92 * color;\r\n }\r\n return 1.055 * Math.pow(color, 0.41666) - 0.055;\r\n}\r\n\r\n/**\r\n * Class used to hold a RGB color\r\n */\r\nexport class Color3 {\r\n /**\r\n * Creates a new Color3 object from red, green, blue values, all between 0 and 1\r\n * @param r defines the red component (between 0 and 1, default is 0)\r\n * @param g defines the green component (between 0 and 1, default is 0)\r\n * @param b defines the blue component (between 0 and 1, default is 0)\r\n */\r\n constructor(\r\n /**\r\n * Defines the red component (between 0 and 1, default is 0)\r\n */\r\n public r: number = 0,\r\n /**\r\n * Defines the green component (between 0 and 1, default is 0)\r\n */\r\n public g: number = 0,\r\n /**\r\n * Defines the blue component (between 0 and 1, default is 0)\r\n */\r\n public b: number = 0\r\n ) {}\r\n\r\n /**\r\n * Creates a string with the Color3 current values\r\n * @returns the string representation of the Color3 object\r\n */\r\n public toString(): string {\r\n return \"{R: \" + this.r + \" G:\" + this.g + \" B:\" + this.b + \"}\";\r\n }\r\n\r\n /**\r\n * Returns the string \"Color3\"\r\n * @returns \"Color3\"\r\n */\r\n public getClassName(): string {\r\n return \"Color3\";\r\n }\r\n\r\n /**\r\n * Compute the Color3 hash code\r\n * @returns an unique number that can be used to hash Color3 objects\r\n */\r\n public getHashCode(): number {\r\n let hash = (this.r * 255) | 0;\r\n hash = (hash * 397) ^ ((this.g * 255) | 0);\r\n hash = (hash * 397) ^ ((this.b * 255) | 0);\r\n return hash;\r\n }\r\n\r\n // Operators\r\n\r\n /**\r\n * Stores in the given array from the given starting index the red, green, blue values as successive elements\r\n * @param array defines the array where to store the r,g,b components\r\n * @param index defines an optional index in the target array to define where to start storing values\r\n * @returns the current Color3 object\r\n */\r\n public toArray(array: FloatArray, index: number = 0): Color3 {\r\n array[index] = this.r;\r\n array[index + 1] = this.g;\r\n array[index + 2] = this.b;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Update the current color with values stored in an array from the starting index of the given array\r\n * @param array defines the source array\r\n * @param offset defines an offset in the source array\r\n * @returns the current Color3 object\r\n */\r\n public fromArray(array: DeepImmutable>, offset: number = 0): Color3 {\r\n Color3.FromArrayToRef(array, offset, this);\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns a new Color4 object from the current Color3 and the given alpha\r\n * @param alpha defines the alpha component on the new Color4 object (default is 1)\r\n * @returns a new Color4 object\r\n */\r\n public toColor4(alpha: number = 1): Color4 {\r\n return new Color4(this.r, this.g, this.b, alpha);\r\n }\r\n\r\n /**\r\n * Returns a new array populated with 3 numeric elements : red, green and blue values\r\n * @returns the new array\r\n */\r\n public asArray(): number[] {\r\n return [this.r, this.g, this.b];\r\n }\r\n\r\n /**\r\n * Returns the luminance value\r\n * @returns a float value\r\n */\r\n public toLuminance(): number {\r\n return this.r * 0.3 + this.g * 0.59 + this.b * 0.11;\r\n }\r\n\r\n /**\r\n * Multiply each Color3 rgb values by the given Color3 rgb values in a new Color3 object\r\n * @param otherColor defines the second operand\r\n * @returns the new Color3 object\r\n */\r\n public multiply(otherColor: DeepImmutable): Color3 {\r\n return new Color3(this.r * otherColor.r, this.g * otherColor.g, this.b * otherColor.b);\r\n }\r\n\r\n /**\r\n * Multiply the rgb values of the Color3 and the given Color3 and stores the result in the object \"result\"\r\n * @param otherColor defines the second operand\r\n * @param result defines the Color3 object where to store the result\r\n * @returns the current Color3\r\n */\r\n public multiplyToRef(otherColor: DeepImmutable, result: Color3): Color3 {\r\n result.r = this.r * otherColor.r;\r\n result.g = this.g * otherColor.g;\r\n result.b = this.b * otherColor.b;\r\n return this;\r\n }\r\n\r\n /**\r\n * Determines equality between Color3 objects\r\n * @param otherColor defines the second operand\r\n * @returns true if the rgb values are equal to the given ones\r\n */\r\n public equals(otherColor: DeepImmutable): boolean {\r\n return otherColor && this.r === otherColor.r && this.g === otherColor.g && this.b === otherColor.b;\r\n }\r\n\r\n /**\r\n * Determines equality between the current Color3 object and a set of r,b,g values\r\n * @param r defines the red component to check\r\n * @param g defines the green component to check\r\n * @param b defines the blue component to check\r\n * @returns true if the rgb values are equal to the given ones\r\n */\r\n public equalsFloats(r: number, g: number, b: number): boolean {\r\n return this.r === r && this.g === g && this.b === b;\r\n }\r\n\r\n /**\r\n * Creates a new Color3 with the current Color3 values multiplied by scale\r\n * @param scale defines the scaling factor to apply\r\n * @returns a new Color3 object\r\n */\r\n public scale(scale: number): Color3 {\r\n return new Color3(this.r * scale, this.g * scale, this.b * scale);\r\n }\r\n\r\n /**\r\n * Multiplies the Color3 values by the float \"scale\"\r\n * @param scale defines the scaling factor to apply\r\n * @returns the current updated Color3\r\n */\r\n public scaleInPlace(scale: number): Color3 {\r\n this.r *= scale;\r\n this.g *= scale;\r\n this.b *= scale;\r\n return this;\r\n }\r\n\r\n /**\r\n * Multiplies the rgb values by scale and stores the result into \"result\"\r\n * @param scale defines the scaling factor\r\n * @param result defines the Color3 object where to store the result\r\n * @returns the unmodified current Color3\r\n */\r\n public scaleToRef(scale: number, result: Color3): Color3 {\r\n result.r = this.r * scale;\r\n result.g = this.g * scale;\r\n result.b = this.b * scale;\r\n return this;\r\n }\r\n\r\n /**\r\n * Scale the current Color3 values by a factor and add the result to a given Color3\r\n * @param scale defines the scale factor\r\n * @param result defines color to store the result into\r\n * @returns the unmodified current Color3\r\n */\r\n public scaleAndAddToRef(scale: number, result: Color3): Color3 {\r\n result.r += this.r * scale;\r\n result.g += this.g * scale;\r\n result.b += this.b * scale;\r\n return this;\r\n }\r\n\r\n /**\r\n * Clamps the rgb values by the min and max values and stores the result into \"result\"\r\n * @param min defines minimum clamping value (default is 0)\r\n * @param max defines maximum clamping value (default is 1)\r\n * @param result defines color to store the result into\r\n * @returns the original Color3\r\n */\r\n public clampToRef(min: number = 0, max: number = 1, result: Color3): Color3 {\r\n result.r = Scalar.Clamp(this.r, min, max);\r\n result.g = Scalar.Clamp(this.g, min, max);\r\n result.b = Scalar.Clamp(this.b, min, max);\r\n return this;\r\n }\r\n\r\n /**\r\n * Creates a new Color3 set with the added values of the current Color3 and of the given one\r\n * @param otherColor defines the second operand\r\n * @returns the new Color3\r\n */\r\n public add(otherColor: DeepImmutable): Color3 {\r\n return new Color3(this.r + otherColor.r, this.g + otherColor.g, this.b + otherColor.b);\r\n }\r\n\r\n /**\r\n * Stores the result of the addition of the current Color3 and given one rgb values into \"result\"\r\n * @param otherColor defines the second operand\r\n * @param result defines Color3 object to store the result into\r\n * @returns the unmodified current Color3\r\n */\r\n public addToRef(otherColor: DeepImmutable, result: Color3): Color3 {\r\n result.r = this.r + otherColor.r;\r\n result.g = this.g + otherColor.g;\r\n result.b = this.b + otherColor.b;\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns a new Color3 set with the subtracted values of the given one from the current Color3\r\n * @param otherColor defines the second operand\r\n * @returns the new Color3\r\n */\r\n public subtract(otherColor: DeepImmutable): Color3 {\r\n return new Color3(this.r - otherColor.r, this.g - otherColor.g, this.b - otherColor.b);\r\n }\r\n\r\n /**\r\n * Stores the result of the subtraction of given one from the current Color3 rgb values into \"result\"\r\n * @param otherColor defines the second operand\r\n * @param result defines Color3 object to store the result into\r\n * @returns the unmodified current Color3\r\n */\r\n public subtractToRef(otherColor: DeepImmutable, result: Color3): Color3 {\r\n result.r = this.r - otherColor.r;\r\n result.g = this.g - otherColor.g;\r\n result.b = this.b - otherColor.b;\r\n return this;\r\n }\r\n\r\n /**\r\n * Copy the current object\r\n * @returns a new Color3 copied the current one\r\n */\r\n public clone(): Color3 {\r\n return new Color3(this.r, this.g, this.b);\r\n }\r\n\r\n /**\r\n * Copies the rgb values from the source in the current Color3\r\n * @param source defines the source Color3 object\r\n * @returns the updated Color3 object\r\n */\r\n public copyFrom(source: DeepImmutable): Color3 {\r\n this.r = source.r;\r\n this.g = source.g;\r\n this.b = source.b;\r\n return this;\r\n }\r\n\r\n /**\r\n * Updates the Color3 rgb values from the given floats\r\n * @param r defines the red component to read from\r\n * @param g defines the green component to read from\r\n * @param b defines the blue component to read from\r\n * @returns the current Color3 object\r\n */\r\n public copyFromFloats(r: number, g: number, b: number): Color3 {\r\n this.r = r;\r\n this.g = g;\r\n this.b = b;\r\n return this;\r\n }\r\n\r\n /**\r\n * Updates the Color3 rgb values from the given floats\r\n * @param r defines the red component to read from\r\n * @param g defines the green component to read from\r\n * @param b defines the blue component to read from\r\n * @returns the current Color3 object\r\n */\r\n public set(r: number, g: number, b: number): Color3 {\r\n return this.copyFromFloats(r, g, b);\r\n }\r\n\r\n /**\r\n * Compute the Color3 hexadecimal code as a string\r\n * @returns a string containing the hexadecimal representation of the Color3 object\r\n */\r\n public toHexString(): string {\r\n const intR = Math.round(this.r * 255);\r\n const intG = Math.round(this.g * 255);\r\n const intB = Math.round(this.b * 255);\r\n return \"#\" + Scalar.ToHex(intR) + Scalar.ToHex(intG) + Scalar.ToHex(intB);\r\n }\r\n\r\n /**\r\n * Converts current color in rgb space to HSV values\r\n * @returns a new color3 representing the HSV values\r\n */\r\n public toHSV(): Color3 {\r\n const result = new Color3();\r\n\r\n this.toHSVToRef(result);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Converts current color in rgb space to HSV values\r\n * @param result defines the Color3 where to store the HSV values\r\n */\r\n public toHSVToRef(result: Color3) {\r\n const r = this.r;\r\n const g = this.g;\r\n const b = this.b;\r\n\r\n const max = Math.max(r, g, b);\r\n const min = Math.min(r, g, b);\r\n let h = 0;\r\n let s = 0;\r\n const v = max;\r\n\r\n const dm = max - min;\r\n\r\n if (max !== 0) {\r\n s = dm / max;\r\n }\r\n\r\n if (max != min) {\r\n if (max == r) {\r\n h = (g - b) / dm;\r\n if (g < b) {\r\n h += 6;\r\n }\r\n } else if (max == g) {\r\n h = (b - r) / dm + 2;\r\n } else if (max == b) {\r\n h = (r - g) / dm + 4;\r\n }\r\n h *= 60;\r\n }\r\n\r\n result.r = h;\r\n result.g = s;\r\n result.b = v;\r\n }\r\n\r\n /**\r\n * Computes a new Color3 converted from the current one to linear space\r\n * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false)\r\n * @returns a new Color3 object\r\n */\r\n public toLinearSpace(exact = false): Color3 {\r\n const convertedColor = new Color3();\r\n this.toLinearSpaceToRef(convertedColor, exact);\r\n return convertedColor;\r\n }\r\n\r\n /**\r\n * Converts the Color3 values to linear space and stores the result in \"convertedColor\"\r\n * @param convertedColor defines the Color3 object where to store the linear space version\r\n * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false)\r\n * @returns the unmodified Color3\r\n */\r\n public toLinearSpaceToRef(convertedColor: Color3, exact = false): Color3 {\r\n if (exact) {\r\n convertedColor.r = colorChannelToLinearSpaceExact(this.r);\r\n convertedColor.g = colorChannelToLinearSpaceExact(this.g);\r\n convertedColor.b = colorChannelToLinearSpaceExact(this.b);\r\n } else {\r\n convertedColor.r = colorChannelToLinearSpace(this.r);\r\n convertedColor.g = colorChannelToLinearSpace(this.g);\r\n convertedColor.b = colorChannelToLinearSpace(this.b);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Computes a new Color3 converted from the current one to gamma space\r\n * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false)\r\n * @returns a new Color3 object\r\n */\r\n public toGammaSpace(exact = false): Color3 {\r\n const convertedColor = new Color3();\r\n this.toGammaSpaceToRef(convertedColor, exact);\r\n return convertedColor;\r\n }\r\n\r\n /**\r\n * Converts the Color3 values to gamma space and stores the result in \"convertedColor\"\r\n * @param convertedColor defines the Color3 object where to store the gamma space version\r\n * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false)\r\n * @returns the unmodified Color3\r\n */\r\n public toGammaSpaceToRef(convertedColor: Color3, exact = false): Color3 {\r\n if (exact) {\r\n convertedColor.r = colorChannelToGammaSpaceExact(this.r);\r\n convertedColor.g = colorChannelToGammaSpaceExact(this.g);\r\n convertedColor.b = colorChannelToGammaSpaceExact(this.b);\r\n } else {\r\n convertedColor.r = colorChannelToGammaSpace(this.r);\r\n convertedColor.g = colorChannelToGammaSpace(this.g);\r\n convertedColor.b = colorChannelToGammaSpace(this.b);\r\n }\r\n return this;\r\n }\r\n\r\n // Statics\r\n\r\n private static _BlackReadOnly = Color3.Black() as DeepImmutable;\r\n\r\n /**\r\n * Converts Hue, saturation and value to a Color3 (RGB)\r\n * @param hue defines the hue (value between 0 and 360)\r\n * @param saturation defines the saturation (value between 0 and 1)\r\n * @param value defines the value (value between 0 and 1)\r\n * @param result defines the Color3 where to store the RGB values\r\n */\r\n public static HSVtoRGBToRef(hue: number, saturation: number, value: number, result: Color3) {\r\n const chroma = value * saturation;\r\n const h = hue / 60;\r\n const x = chroma * (1 - Math.abs((h % 2) - 1));\r\n let r = 0;\r\n let g = 0;\r\n let b = 0;\r\n\r\n if (h >= 0 && h <= 1) {\r\n r = chroma;\r\n g = x;\r\n } else if (h >= 1 && h <= 2) {\r\n r = x;\r\n g = chroma;\r\n } else if (h >= 2 && h <= 3) {\r\n g = chroma;\r\n b = x;\r\n } else if (h >= 3 && h <= 4) {\r\n g = x;\r\n b = chroma;\r\n } else if (h >= 4 && h <= 5) {\r\n r = x;\r\n b = chroma;\r\n } else if (h >= 5 && h <= 6) {\r\n r = chroma;\r\n b = x;\r\n }\r\n\r\n const m = value - chroma;\r\n result.set(r + m, g + m, b + m);\r\n }\r\n\r\n /**\r\n * Converts Hue, saturation and value to a new Color3 (RGB)\r\n * @param hue defines the hue (value between 0 and 360)\r\n * @param saturation defines the saturation (value between 0 and 1)\r\n * @param value defines the value (value between 0 and 1)\r\n * @returns a new Color3 object\r\n */\r\n public static FromHSV(hue: number, saturation: number, value: number): Color3 {\r\n const result = new Color3(0, 0, 0);\r\n Color3.HSVtoRGBToRef(hue, saturation, value, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new Color3 from the string containing valid hexadecimal values\r\n * @param hex defines a string containing valid hexadecimal values\r\n * @returns a new Color3 object\r\n */\r\n public static FromHexString(hex: string): Color3 {\r\n if (hex.substring(0, 1) !== \"#\" || hex.length !== 7) {\r\n return new Color3(0, 0, 0);\r\n }\r\n\r\n const r = parseInt(hex.substring(1, 3), 16);\r\n const g = parseInt(hex.substring(3, 5), 16);\r\n const b = parseInt(hex.substring(5, 7), 16);\r\n\r\n return Color3.FromInts(r, g, b);\r\n }\r\n\r\n /**\r\n * Creates a new Color3 from the starting index of the given array\r\n * @param array defines the source array\r\n * @param offset defines an offset in the source array\r\n * @returns a new Color3 object\r\n */\r\n public static FromArray(array: DeepImmutable>, offset: number = 0): Color3 {\r\n return new Color3(array[offset], array[offset + 1], array[offset + 2]);\r\n }\r\n\r\n /**\r\n * Creates a new Color3 from the starting index element of the given array\r\n * @param array defines the source array to read from\r\n * @param offset defines the offset in the source array\r\n * @param result defines the target Color3 object\r\n */\r\n public static FromArrayToRef(array: DeepImmutable>, offset: number = 0, result: Color3) {\r\n result.r = array[offset];\r\n result.g = array[offset + 1];\r\n result.b = array[offset + 2];\r\n }\r\n\r\n /**\r\n * Creates a new Color3 from integer values (< 256)\r\n * @param r defines the red component to read from (value between 0 and 255)\r\n * @param g defines the green component to read from (value between 0 and 255)\r\n * @param b defines the blue component to read from (value between 0 and 255)\r\n * @returns a new Color3 object\r\n */\r\n public static FromInts(r: number, g: number, b: number): Color3 {\r\n return new Color3(r / 255.0, g / 255.0, b / 255.0);\r\n }\r\n\r\n /**\r\n * Creates a new Color3 with values linearly interpolated of \"amount\" between the start Color3 and the end Color3\r\n * @param start defines the start Color3 value\r\n * @param end defines the end Color3 value\r\n * @param amount defines the gradient value between start and end\r\n * @returns a new Color3 object\r\n */\r\n public static Lerp(start: DeepImmutable, end: DeepImmutable, amount: number): Color3 {\r\n const result = new Color3(0.0, 0.0, 0.0);\r\n Color3.LerpToRef(start, end, amount, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a new Color3 with values linearly interpolated of \"amount\" between the start Color3 and the end Color3\r\n * @param left defines the start value\r\n * @param right defines the end value\r\n * @param amount defines the gradient factor\r\n * @param result defines the Color3 object where to store the result\r\n */\r\n public static LerpToRef(left: DeepImmutable, right: DeepImmutable, amount: number, result: Color3): void {\r\n result.r = left.r + (right.r - left.r) * amount;\r\n result.g = left.g + (right.g - left.g) * amount;\r\n result.b = left.b + (right.b - left.b) * amount;\r\n }\r\n\r\n /**\r\n * Returns a new Color3 located for \"amount\" (float) on the Hermite interpolation spline defined by the vectors \"value1\", \"tangent1\", \"value2\", \"tangent2\"\r\n * @param value1 defines the first control point\r\n * @param tangent1 defines the first tangent Color3\r\n * @param value2 defines the second control point\r\n * @param tangent2 defines the second tangent Color3\r\n * @param amount defines the amount on the interpolation spline (between 0 and 1)\r\n * @returns the new Color3\r\n */\r\n public static Hermite(value1: DeepImmutable, tangent1: DeepImmutable, value2: DeepImmutable, tangent2: DeepImmutable, amount: number): Color3 {\r\n const squared = amount * amount;\r\n const cubed = amount * squared;\r\n const part1 = 2.0 * cubed - 3.0 * squared + 1.0;\r\n const part2 = -2.0 * cubed + 3.0 * squared;\r\n const part3 = cubed - 2.0 * squared + amount;\r\n const part4 = cubed - squared;\r\n\r\n const r = value1.r * part1 + value2.r * part2 + tangent1.r * part3 + tangent2.r * part4;\r\n const g = value1.g * part1 + value2.g * part2 + tangent1.g * part3 + tangent2.g * part4;\r\n const b = value1.b * part1 + value2.b * part2 + tangent1.b * part3 + tangent2.b * part4;\r\n return new Color3(r, g, b);\r\n }\r\n\r\n /**\r\n * Returns a new Color3 which is the 1st derivative of the Hermite spline defined by the colors \"value1\", \"value2\", \"tangent1\", \"tangent2\".\r\n * @param value1 defines the first control point\r\n * @param tangent1 defines the first tangent\r\n * @param value2 defines the second control point\r\n * @param tangent2 defines the second tangent\r\n * @param time define where the derivative must be done\r\n * @returns 1st derivative\r\n */\r\n public static Hermite1stDerivative(\r\n value1: DeepImmutable,\r\n tangent1: DeepImmutable,\r\n value2: DeepImmutable,\r\n tangent2: DeepImmutable,\r\n time: number\r\n ): Color3 {\r\n const result = Color3.Black();\r\n\r\n this.Hermite1stDerivativeToRef(value1, tangent1, value2, tangent2, time, result);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Returns a new Color3 which is the 1st derivative of the Hermite spline defined by the colors \"value1\", \"value2\", \"tangent1\", \"tangent2\".\r\n * @param value1 defines the first control point\r\n * @param tangent1 defines the first tangent\r\n * @param value2 defines the second control point\r\n * @param tangent2 defines the second tangent\r\n * @param time define where the derivative must be done\r\n * @param result define where to store the derivative\r\n */\r\n public static Hermite1stDerivativeToRef(\r\n value1: DeepImmutable,\r\n tangent1: DeepImmutable,\r\n value2: DeepImmutable,\r\n tangent2: DeepImmutable,\r\n time: number,\r\n result: Color3\r\n ) {\r\n const t2 = time * time;\r\n\r\n result.r = (t2 - time) * 6 * value1.r + (3 * t2 - 4 * time + 1) * tangent1.r + (-t2 + time) * 6 * value2.r + (3 * t2 - 2 * time) * tangent2.r;\r\n result.g = (t2 - time) * 6 * value1.g + (3 * t2 - 4 * time + 1) * tangent1.g + (-t2 + time) * 6 * value2.g + (3 * t2 - 2 * time) * tangent2.g;\r\n result.b = (t2 - time) * 6 * value1.b + (3 * t2 - 4 * time + 1) * tangent1.b + (-t2 + time) * 6 * value2.b + (3 * t2 - 2 * time) * tangent2.b;\r\n }\r\n\r\n /**\r\n * Returns a Color3 value containing a red color\r\n * @returns a new Color3 object\r\n */\r\n public static Red(): Color3 {\r\n return new Color3(1, 0, 0);\r\n }\r\n /**\r\n * Returns a Color3 value containing a green color\r\n * @returns a new Color3 object\r\n */\r\n public static Green(): Color3 {\r\n return new Color3(0, 1, 0);\r\n }\r\n /**\r\n * Returns a Color3 value containing a blue color\r\n * @returns a new Color3 object\r\n */\r\n public static Blue(): Color3 {\r\n return new Color3(0, 0, 1);\r\n }\r\n /**\r\n * Returns a Color3 value containing a black color\r\n * @returns a new Color3 object\r\n */\r\n public static Black(): Color3 {\r\n return new Color3(0, 0, 0);\r\n }\r\n\r\n /**\r\n * Gets a Color3 value containing a black color that must not be updated\r\n */\r\n public static get BlackReadOnly(): DeepImmutable {\r\n return Color3._BlackReadOnly;\r\n }\r\n\r\n /**\r\n * Returns a Color3 value containing a white color\r\n * @returns a new Color3 object\r\n */\r\n public static White(): Color3 {\r\n return new Color3(1, 1, 1);\r\n }\r\n /**\r\n * Returns a Color3 value containing a purple color\r\n * @returns a new Color3 object\r\n */\r\n public static Purple(): Color3 {\r\n return new Color3(0.5, 0, 0.5);\r\n }\r\n /**\r\n * Returns a Color3 value containing a magenta color\r\n * @returns a new Color3 object\r\n */\r\n public static Magenta(): Color3 {\r\n return new Color3(1, 0, 1);\r\n }\r\n /**\r\n * Returns a Color3 value containing a yellow color\r\n * @returns a new Color3 object\r\n */\r\n public static Yellow(): Color3 {\r\n return new Color3(1, 1, 0);\r\n }\r\n /**\r\n * Returns a Color3 value containing a gray color\r\n * @returns a new Color3 object\r\n */\r\n public static Gray(): Color3 {\r\n return new Color3(0.5, 0.5, 0.5);\r\n }\r\n /**\r\n * Returns a Color3 value containing a teal color\r\n * @returns a new Color3 object\r\n */\r\n public static Teal(): Color3 {\r\n return new Color3(0, 1.0, 1.0);\r\n }\r\n /**\r\n * Returns a Color3 value containing a random color\r\n * @returns a new Color3 object\r\n */\r\n public static Random(): Color3 {\r\n return new Color3(Math.random(), Math.random(), Math.random());\r\n }\r\n}\r\n\r\n/**\r\n * Class used to hold a RBGA color\r\n */\r\nexport class Color4 {\r\n /**\r\n * Creates a new Color4 object from red, green, blue values, all between 0 and 1\r\n * @param r defines the red component (between 0 and 1, default is 0)\r\n * @param g defines the green component (between 0 and 1, default is 0)\r\n * @param b defines the blue component (between 0 and 1, default is 0)\r\n * @param a defines the alpha component (between 0 and 1, default is 1)\r\n */\r\n constructor(\r\n /**\r\n * Defines the red component (between 0 and 1, default is 0)\r\n */\r\n public r: number = 0,\r\n /**\r\n * Defines the green component (between 0 and 1, default is 0)\r\n */\r\n public g: number = 0,\r\n /**\r\n * Defines the blue component (between 0 and 1, default is 0)\r\n */\r\n public b: number = 0,\r\n /**\r\n * Defines the alpha component (between 0 and 1, default is 1)\r\n */\r\n public a: number = 1\r\n ) {}\r\n\r\n // Operators\r\n\r\n /**\r\n * Adds in place the given Color4 values to the current Color4 object\r\n * @param right defines the second operand\r\n * @returns the current updated Color4 object\r\n */\r\n public addInPlace(right: DeepImmutable): Color4 {\r\n this.r += right.r;\r\n this.g += right.g;\r\n this.b += right.b;\r\n this.a += right.a;\r\n return this;\r\n }\r\n\r\n /**\r\n * Creates a new array populated with 4 numeric elements : red, green, blue, alpha values\r\n * @returns the new array\r\n */\r\n public asArray(): number[] {\r\n return [this.r, this.g, this.b, this.a];\r\n }\r\n\r\n /**\r\n * Stores from the starting index in the given array the Color4 successive values\r\n * @param array defines the array where to store the r,g,b components\r\n * @param index defines an optional index in the target array to define where to start storing values\r\n * @returns the current Color4 object\r\n */\r\n public toArray(array: FloatArray, index: number = 0): Color4 {\r\n array[index] = this.r;\r\n array[index + 1] = this.g;\r\n array[index + 2] = this.b;\r\n array[index + 3] = this.a;\r\n return this;\r\n }\r\n\r\n /**\r\n * Update the current color with values stored in an array from the starting index of the given array\r\n * @param array defines the source array\r\n * @param offset defines an offset in the source array\r\n * @returns the current Color4 object\r\n */\r\n public fromArray(array: DeepImmutable>, offset: number = 0): Color4 {\r\n Color4.FromArrayToRef(array, offset, this);\r\n return this;\r\n }\r\n\r\n /**\r\n * Determines equality between Color4 objects\r\n * @param otherColor defines the second operand\r\n * @returns true if the rgba values are equal to the given ones\r\n */\r\n public equals(otherColor: DeepImmutable): boolean {\r\n return otherColor && this.r === otherColor.r && this.g === otherColor.g && this.b === otherColor.b && this.a === otherColor.a;\r\n }\r\n\r\n /**\r\n * Creates a new Color4 set with the added values of the current Color4 and of the given one\r\n * @param right defines the second operand\r\n * @returns a new Color4 object\r\n */\r\n public add(right: DeepImmutable): Color4 {\r\n return new Color4(this.r + right.r, this.g + right.g, this.b + right.b, this.a + right.a);\r\n }\r\n\r\n /**\r\n * Creates a new Color4 set with the subtracted values of the given one from the current Color4\r\n * @param right defines the second operand\r\n * @returns a new Color4 object\r\n */\r\n public subtract(right: DeepImmutable): Color4 {\r\n return new Color4(this.r - right.r, this.g - right.g, this.b - right.b, this.a - right.a);\r\n }\r\n\r\n /**\r\n * Subtracts the given ones from the current Color4 values and stores the results in \"result\"\r\n * @param right defines the second operand\r\n * @param result defines the Color4 object where to store the result\r\n * @returns the current Color4 object\r\n */\r\n public subtractToRef(right: DeepImmutable, result: Color4): Color4 {\r\n result.r = this.r - right.r;\r\n result.g = this.g - right.g;\r\n result.b = this.b - right.b;\r\n result.a = this.a - right.a;\r\n return this;\r\n }\r\n\r\n /**\r\n * Creates a new Color4 with the current Color4 values multiplied by scale\r\n * @param scale defines the scaling factor to apply\r\n * @returns a new Color4 object\r\n */\r\n public scale(scale: number): Color4 {\r\n return new Color4(this.r * scale, this.g * scale, this.b * scale, this.a * scale);\r\n }\r\n\r\n /**\r\n * Multiplies the Color4 values by the float \"scale\"\r\n * @param scale defines the scaling factor to apply\r\n * @returns the current updated Color4\r\n */\r\n public scaleInPlace(scale: number): Color4 {\r\n this.r *= scale;\r\n this.g *= scale;\r\n this.b *= scale;\r\n this.a *= scale;\r\n return this;\r\n }\r\n\r\n /**\r\n * Multiplies the current Color4 values by scale and stores the result in \"result\"\r\n * @param scale defines the scaling factor to apply\r\n * @param result defines the Color4 object where to store the result\r\n * @returns the current unmodified Color4\r\n */\r\n public scaleToRef(scale: number, result: Color4): Color4 {\r\n result.r = this.r * scale;\r\n result.g = this.g * scale;\r\n result.b = this.b * scale;\r\n result.a = this.a * scale;\r\n return this;\r\n }\r\n\r\n /**\r\n * Scale the current Color4 values by a factor and add the result to a given Color4\r\n * @param scale defines the scale factor\r\n * @param result defines the Color4 object where to store the result\r\n * @returns the unmodified current Color4\r\n */\r\n public scaleAndAddToRef(scale: number, result: Color4): Color4 {\r\n result.r += this.r * scale;\r\n result.g += this.g * scale;\r\n result.b += this.b * scale;\r\n result.a += this.a * scale;\r\n return this;\r\n }\r\n\r\n /**\r\n * Clamps the rgb values by the min and max values and stores the result into \"result\"\r\n * @param min defines minimum clamping value (default is 0)\r\n * @param max defines maximum clamping value (default is 1)\r\n * @param result defines color to store the result into.\r\n * @returns the current Color4\r\n */\r\n public clampToRef(min: number = 0, max: number = 1, result: Color4): Color4 {\r\n result.r = Scalar.Clamp(this.r, min, max);\r\n result.g = Scalar.Clamp(this.g, min, max);\r\n result.b = Scalar.Clamp(this.b, min, max);\r\n result.a = Scalar.Clamp(this.a, min, max);\r\n return this;\r\n }\r\n\r\n /**\r\n * Multiply an Color4 value by another and return a new Color4 object\r\n * @param color defines the Color4 value to multiply by\r\n * @returns a new Color4 object\r\n */\r\n public multiply(color: Color4): Color4 {\r\n return new Color4(this.r * color.r, this.g * color.g, this.b * color.b, this.a * color.a);\r\n }\r\n\r\n /**\r\n * Multiply a Color4 value by another and push the result in a reference value\r\n * @param color defines the Color4 value to multiply by\r\n * @param result defines the Color4 to fill the result in\r\n * @returns the result Color4\r\n */\r\n public multiplyToRef(color: Color4, result: Color4): Color4 {\r\n result.r = this.r * color.r;\r\n result.g = this.g * color.g;\r\n result.b = this.b * color.b;\r\n result.a = this.a * color.a;\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a string with the Color4 current values\r\n * @returns the string representation of the Color4 object\r\n */\r\n public toString(): string {\r\n return \"{R: \" + this.r + \" G:\" + this.g + \" B:\" + this.b + \" A:\" + this.a + \"}\";\r\n }\r\n\r\n /**\r\n * Returns the string \"Color4\"\r\n * @returns \"Color4\"\r\n */\r\n public getClassName(): string {\r\n return \"Color4\";\r\n }\r\n\r\n /**\r\n * Compute the Color4 hash code\r\n * @returns an unique number that can be used to hash Color4 objects\r\n */\r\n public getHashCode(): number {\r\n let hash = (this.r * 255) | 0;\r\n hash = (hash * 397) ^ ((this.g * 255) | 0);\r\n hash = (hash * 397) ^ ((this.b * 255) | 0);\r\n hash = (hash * 397) ^ ((this.a * 255) | 0);\r\n return hash;\r\n }\r\n\r\n /**\r\n * Creates a new Color4 copied from the current one\r\n * @returns a new Color4 object\r\n */\r\n public clone(): Color4 {\r\n return new Color4(this.r, this.g, this.b, this.a);\r\n }\r\n\r\n /**\r\n * Copies the given Color4 values into the current one\r\n * @param source defines the source Color4 object\r\n * @returns the current updated Color4 object\r\n */\r\n public copyFrom(source: Color4): Color4 {\r\n this.r = source.r;\r\n this.g = source.g;\r\n this.b = source.b;\r\n this.a = source.a;\r\n return this;\r\n }\r\n\r\n /**\r\n * Copies the given float values into the current one\r\n * @param r defines the red component to read from\r\n * @param g defines the green component to read from\r\n * @param b defines the blue component to read from\r\n * @param a defines the alpha component to read from\r\n * @returns the current updated Color4 object\r\n */\r\n public copyFromFloats(r: number, g: number, b: number, a: number): Color4 {\r\n this.r = r;\r\n this.g = g;\r\n this.b = b;\r\n this.a = a;\r\n return this;\r\n }\r\n\r\n /**\r\n * Copies the given float values into the current one\r\n * @param r defines the red component to read from\r\n * @param g defines the green component to read from\r\n * @param b defines the blue component to read from\r\n * @param a defines the alpha component to read from\r\n * @returns the current updated Color4 object\r\n */\r\n public set(r: number, g: number, b: number, a: number): Color4 {\r\n return this.copyFromFloats(r, g, b, a);\r\n }\r\n\r\n /**\r\n * Compute the Color4 hexadecimal code as a string\r\n * @param returnAsColor3 defines if the string should only contains RGB values (off by default)\r\n * @returns a string containing the hexadecimal representation of the Color4 object\r\n */\r\n public toHexString(returnAsColor3 = false): string {\r\n const intR = Math.round(this.r * 255);\r\n const intG = Math.round(this.g * 255);\r\n const intB = Math.round(this.b * 255);\r\n\r\n if (returnAsColor3) {\r\n return \"#\" + Scalar.ToHex(intR) + Scalar.ToHex(intG) + Scalar.ToHex(intB);\r\n }\r\n\r\n const intA = Math.round(this.a * 255);\r\n return \"#\" + Scalar.ToHex(intR) + Scalar.ToHex(intG) + Scalar.ToHex(intB) + Scalar.ToHex(intA);\r\n }\r\n\r\n /**\r\n * Computes a new Color4 converted from the current one to linear space\r\n * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false)\r\n * @returns a new Color4 object\r\n */\r\n public toLinearSpace(exact = false): Color4 {\r\n const convertedColor = new Color4();\r\n this.toLinearSpaceToRef(convertedColor, exact);\r\n return convertedColor;\r\n }\r\n\r\n /**\r\n * Converts the Color4 values to linear space and stores the result in \"convertedColor\"\r\n * @param convertedColor defines the Color4 object where to store the linear space version\r\n * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false)\r\n * @returns the unmodified Color4\r\n */\r\n public toLinearSpaceToRef(convertedColor: Color4, exact = false): Color4 {\r\n if (exact) {\r\n convertedColor.r = colorChannelToLinearSpaceExact(this.r);\r\n convertedColor.g = colorChannelToLinearSpaceExact(this.g);\r\n convertedColor.b = colorChannelToLinearSpaceExact(this.b);\r\n } else {\r\n convertedColor.r = colorChannelToLinearSpace(this.r);\r\n convertedColor.g = colorChannelToLinearSpace(this.g);\r\n convertedColor.b = colorChannelToLinearSpace(this.b);\r\n }\r\n convertedColor.a = this.a;\r\n return this;\r\n }\r\n\r\n /**\r\n * Computes a new Color4 converted from the current one to gamma space\r\n * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false)\r\n * @returns a new Color4 object\r\n */\r\n public toGammaSpace(exact = false): Color4 {\r\n const convertedColor = new Color4();\r\n this.toGammaSpaceToRef(convertedColor, exact);\r\n return convertedColor;\r\n }\r\n\r\n /**\r\n * Converts the Color4 values to gamma space and stores the result in \"convertedColor\"\r\n * @param convertedColor defines the Color4 object where to store the gamma space version\r\n * @param exact defines if the conversion will be done in an exact way which is slower but more accurate (default is false)\r\n * @returns the unmodified Color4\r\n */\r\n public toGammaSpaceToRef(convertedColor: Color4, exact = false): Color4 {\r\n if (exact) {\r\n convertedColor.r = colorChannelToGammaSpaceExact(this.r);\r\n convertedColor.g = colorChannelToGammaSpaceExact(this.g);\r\n convertedColor.b = colorChannelToGammaSpaceExact(this.b);\r\n } else {\r\n convertedColor.r = colorChannelToGammaSpace(this.r);\r\n convertedColor.g = colorChannelToGammaSpace(this.g);\r\n convertedColor.b = colorChannelToGammaSpace(this.b);\r\n }\r\n convertedColor.a = this.a;\r\n return this;\r\n }\r\n\r\n // Statics\r\n\r\n /**\r\n * Creates a new Color4 from the string containing valid hexadecimal values.\r\n *\r\n * A valid hex string is either in the format #RRGGBB or #RRGGBBAA.\r\n *\r\n * When a hex string without alpha is passed, the resulting Color4 has\r\n * its alpha value set to 1.0.\r\n *\r\n * An invalid string results in a Color with all its channels set to 0.0,\r\n * i.e. \"transparent black\".\r\n *\r\n * @param hex defines a string containing valid hexadecimal values\r\n * @returns a new Color4 object\r\n */\r\n public static FromHexString(hex: string): Color4 {\r\n if (hex.substring(0, 1) !== \"#\" || (hex.length !== 9 && hex.length !== 7)) {\r\n return new Color4(0.0, 0.0, 0.0, 0.0);\r\n }\r\n\r\n const r = parseInt(hex.substring(1, 3), 16);\r\n const g = parseInt(hex.substring(3, 5), 16);\r\n const b = parseInt(hex.substring(5, 7), 16);\r\n const a = hex.length === 9 ? parseInt(hex.substring(7, 9), 16) : 255;\r\n\r\n return Color4.FromInts(r, g, b, a);\r\n }\r\n\r\n /**\r\n * Creates a new Color4 object set with the linearly interpolated values of \"amount\" between the left Color4 object and the right Color4 object\r\n * @param left defines the start value\r\n * @param right defines the end value\r\n * @param amount defines the gradient factor\r\n * @returns a new Color4 object\r\n */\r\n public static Lerp(left: DeepImmutable, right: DeepImmutable, amount: number): Color4 {\r\n const result = new Color4(0.0, 0.0, 0.0, 0.0);\r\n Color4.LerpToRef(left, right, amount, result);\r\n return result;\r\n }\r\n\r\n /**\r\n * Set the given \"result\" with the linearly interpolated values of \"amount\" between the left Color4 object and the right Color4 object\r\n * @param left defines the start value\r\n * @param right defines the end value\r\n * @param amount defines the gradient factor\r\n * @param result defines the Color4 object where to store data\r\n */\r\n public static LerpToRef(left: DeepImmutable, right: DeepImmutable, amount: number, result: Color4): void {\r\n result.r = left.r + (right.r - left.r) * amount;\r\n result.g = left.g + (right.g - left.g) * amount;\r\n result.b = left.b + (right.b - left.b) * amount;\r\n result.a = left.a + (right.a - left.a) * amount;\r\n }\r\n\r\n /**\r\n * Interpolate between two Color4 using Hermite interpolation\r\n * @param value1 defines first Color4\r\n * @param tangent1 defines the incoming tangent\r\n * @param value2 defines second Color4\r\n * @param tangent2 defines the outgoing tangent\r\n * @param amount defines the target Color4\r\n * @returns the new interpolated Color4\r\n */\r\n public static Hermite(value1: DeepImmutable, tangent1: DeepImmutable, value2: DeepImmutable, tangent2: DeepImmutable, amount: number): Color4 {\r\n const squared = amount * amount;\r\n const cubed = amount * squared;\r\n const part1 = 2.0 * cubed - 3.0 * squared + 1.0;\r\n const part2 = -2.0 * cubed + 3.0 * squared;\r\n const part3 = cubed - 2.0 * squared + amount;\r\n const part4 = cubed - squared;\r\n\r\n const r = value1.r * part1 + value2.r * part2 + tangent1.r * part3 + tangent2.r * part4;\r\n const g = value1.g * part1 + value2.g * part2 + tangent1.g * part3 + tangent2.g * part4;\r\n const b = value1.b * part1 + value2.b * part2 + tangent1.b * part3 + tangent2.b * part4;\r\n const a = value1.a * part1 + value2.a * part2 + tangent1.a * part3 + tangent2.a * part4;\r\n return new Color4(r, g, b, a);\r\n }\r\n\r\n /**\r\n * Returns a new Color4 which is the 1st derivative of the Hermite spline defined by the colors \"value1\", \"value2\", \"tangent1\", \"tangent2\".\r\n * @param value1 defines the first control point\r\n * @param tangent1 defines the first tangent\r\n * @param value2 defines the second control point\r\n * @param tangent2 defines the second tangent\r\n * @param time define where the derivative must be done\r\n * @returns 1st derivative\r\n */\r\n public static Hermite1stDerivative(\r\n value1: DeepImmutable,\r\n tangent1: DeepImmutable,\r\n value2: DeepImmutable,\r\n tangent2: DeepImmutable,\r\n time: number\r\n ): Color4 {\r\n const result = new Color4();\r\n\r\n this.Hermite1stDerivativeToRef(value1, tangent1, value2, tangent2, time, result);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Update a Color4 with the 1st derivative of the Hermite spline defined by the colors \"value1\", \"value2\", \"tangent1\", \"tangent2\".\r\n * @param value1 defines the first control point\r\n * @param tangent1 defines the first tangent\r\n * @param value2 defines the second control point\r\n * @param tangent2 defines the second tangent\r\n * @param time define where the derivative must be done\r\n * @param result define where to store the derivative\r\n */\r\n public static Hermite1stDerivativeToRef(\r\n value1: DeepImmutable,\r\n tangent1: DeepImmutable,\r\n value2: DeepImmutable,\r\n tangent2: DeepImmutable,\r\n time: number,\r\n result: Color4\r\n ) {\r\n const t2 = time * time;\r\n\r\n result.r = (t2 - time) * 6 * value1.r + (3 * t2 - 4 * time + 1) * tangent1.r + (-t2 + time) * 6 * value2.r + (3 * t2 - 2 * time) * tangent2.r;\r\n result.g = (t2 - time) * 6 * value1.g + (3 * t2 - 4 * time + 1) * tangent1.g + (-t2 + time) * 6 * value2.g + (3 * t2 - 2 * time) * tangent2.g;\r\n result.b = (t2 - time) * 6 * value1.b + (3 * t2 - 4 * time + 1) * tangent1.b + (-t2 + time) * 6 * value2.b + (3 * t2 - 2 * time) * tangent2.b;\r\n result.a = (t2 - time) * 6 * value1.a + (3 * t2 - 4 * time + 1) * tangent1.a + (-t2 + time) * 6 * value2.a + (3 * t2 - 2 * time) * tangent2.a;\r\n }\r\n\r\n /**\r\n * Creates a new Color4 from a Color3 and an alpha value\r\n * @param color3 defines the source Color3 to read from\r\n * @param alpha defines the alpha component (1.0 by default)\r\n * @returns a new Color4 object\r\n */\r\n public static FromColor3(color3: DeepImmutable, alpha: number = 1.0): Color4 {\r\n return new Color4(color3.r, color3.g, color3.b, alpha);\r\n }\r\n\r\n /**\r\n * Creates a new Color4 from the starting index element of the given array\r\n * @param array defines the source array to read from\r\n * @param offset defines the offset in the source array\r\n * @returns a new Color4 object\r\n */\r\n public static FromArray(array: DeepImmutable>, offset: number = 0): Color4 {\r\n return new Color4(array[offset], array[offset + 1], array[offset + 2], array[offset + 3]);\r\n }\r\n\r\n /**\r\n * Creates a new Color4 from the starting index element of the given array\r\n * @param array defines the source array to read from\r\n * @param offset defines the offset in the source array\r\n * @param result defines the target Color4 object\r\n */\r\n public static FromArrayToRef(array: DeepImmutable>, offset: number = 0, result: Color4) {\r\n result.r = array[offset];\r\n result.g = array[offset + 1];\r\n result.b = array[offset + 2];\r\n result.a = array[offset + 3];\r\n }\r\n\r\n /**\r\n * Creates a new Color3 from integer values (< 256)\r\n * @param r defines the red component to read from (value between 0 and 255)\r\n * @param g defines the green component to read from (value between 0 and 255)\r\n * @param b defines the blue component to read from (value between 0 and 255)\r\n * @param a defines the alpha component to read from (value between 0 and 255)\r\n * @returns a new Color3 object\r\n */\r\n public static FromInts(r: number, g: number, b: number, a: number): Color4 {\r\n return new Color4(r / 255.0, g / 255.0, b / 255.0, a / 255.0);\r\n }\r\n\r\n /**\r\n * Check the content of a given array and convert it to an array containing RGBA data\r\n * If the original array was already containing count * 4 values then it is returned directly\r\n * @param colors defines the array to check\r\n * @param count defines the number of RGBA data to expect\r\n * @returns an array containing count * 4 values (RGBA)\r\n */\r\n public static CheckColors4(colors: number[], count: number): number[] {\r\n // Check if color3 was used\r\n if (colors.length === count * 3) {\r\n const colors4 = [];\r\n for (let index = 0; index < colors.length; index += 3) {\r\n const newIndex = (index / 3) * 4;\r\n colors4[newIndex] = colors[index];\r\n colors4[newIndex + 1] = colors[index + 1];\r\n colors4[newIndex + 2] = colors[index + 2];\r\n colors4[newIndex + 3] = 1.0;\r\n }\r\n\r\n return colors4;\r\n }\r\n\r\n return colors;\r\n }\r\n}\r\n\r\n/**\r\n * @internal\r\n */\r\nexport class TmpColors {\r\n public static Color3: Color3[] = ArrayTools.BuildArray(3, Color3.Black);\r\n public static Color4: Color4[] = ArrayTools.BuildArray(3, () => new Color4(0, 0, 0, 0));\r\n}\r\n\r\nRegisterClass(\"BABYLON.Color3\", Color3);\r\nRegisterClass(\"BABYLON.Color4\", Color4);\r\n","/* eslint-disable @typescript-eslint/no-unused-vars */\r\n/* eslint-disable @typescript-eslint/naming-convention */\r\nimport { Tags } from \"../Misc/tags\";\r\nimport type { Nullable } from \"../types\";\r\nimport { Quaternion, Vector2, Vector3, Matrix } from \"../Maths/math.vector\";\r\nimport { _WarnImport } from \"./devTools\";\r\nimport type { IAnimatable } from \"../Animations/animatable.interface\";\r\nimport { Color4, Color3 } from \"../Maths/math.color\";\r\n\r\nimport type { Scene } from \"../scene\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\n\r\nimport type { ImageProcessingConfiguration } from \"../Materials/imageProcessingConfiguration\";\r\nimport type { FresnelParameters } from \"../Materials/fresnelParameters\";\r\nimport type { ColorCurves } from \"../Materials/colorCurves\";\r\nimport type { BaseTexture } from \"../Materials/Textures/baseTexture\";\r\n\r\nconst __decoratorInitialStore = {};\r\nconst __mergedStore = {};\r\n\r\n/** @internal */\r\nexport interface CopySourceOptions {\r\n /*\r\n * if a texture is used in more than one channel (e.g diffuse and opacity),\r\n * only clone it once and reuse it on the other channels. Default false\r\n */\r\n cloneTexturesOnlyOnce?: boolean;\r\n}\r\n\r\nconst _copySource = function (creationFunction: () => T, source: T, instanciate: boolean, options: CopySourceOptions = {}): T {\r\n const destination = creationFunction();\r\n\r\n // Tags\r\n if (Tags && Tags.HasTags(source)) {\r\n Tags.AddTagsTo(destination, Tags.GetTags(source, true));\r\n }\r\n\r\n const classStore = getMergedStore(destination);\r\n\r\n // Map from source texture uniqueId to destination texture\r\n const textureMap: Record = {};\r\n\r\n // Properties\r\n for (const property in classStore) {\r\n const propertyDescriptor = classStore[property];\r\n const sourceProperty = (source)[property];\r\n const propertyType = propertyDescriptor.type;\r\n\r\n if (sourceProperty !== undefined && sourceProperty !== null && (property !== \"uniqueId\" || SerializationHelper.AllowLoadingUniqueId)) {\r\n switch (propertyType) {\r\n case 0: // Value\r\n case 6: // Mesh reference\r\n case 11: // Camera reference\r\n (destination)[property] = sourceProperty;\r\n break;\r\n case 1: // Texture\r\n if (options.cloneTexturesOnlyOnce && textureMap[sourceProperty.uniqueId]) {\r\n (destination)[property] = textureMap[sourceProperty.uniqueId];\r\n } else {\r\n (destination)[property] = instanciate || sourceProperty.isRenderTarget ? sourceProperty : sourceProperty.clone();\r\n textureMap[sourceProperty.uniqueId] = (destination)[property];\r\n }\r\n break;\r\n case 2: // Color3\r\n case 3: // FresnelParameters\r\n case 4: // Vector2\r\n case 5: // Vector3\r\n case 7: // Color Curves\r\n case 10: // Quaternion\r\n case 12: // Matrix\r\n (destination)[property] = instanciate ? sourceProperty : sourceProperty.clone();\r\n break;\r\n }\r\n }\r\n }\r\n\r\n return destination;\r\n};\r\n\r\nfunction getDirectStore(target: any): any {\r\n const classKey = target.getClassName();\r\n\r\n if (!(__decoratorInitialStore)[classKey]) {\r\n (__decoratorInitialStore)[classKey] = {};\r\n }\r\n\r\n return (__decoratorInitialStore)[classKey];\r\n}\r\n\r\n/**\r\n * Return the list of properties flagged as serializable\r\n * @param target host object\r\n */\r\nfunction getMergedStore(target: any): any {\r\n const classKey = target.getClassName();\r\n\r\n if ((__mergedStore)[classKey]) {\r\n return (__mergedStore)[classKey];\r\n }\r\n\r\n (__mergedStore)[classKey] = {};\r\n\r\n const store = (__mergedStore)[classKey];\r\n let currentTarget = target;\r\n let currentKey = classKey;\r\n while (currentKey) {\r\n const initialStore = (__decoratorInitialStore)[currentKey];\r\n for (const property in initialStore) {\r\n store[property] = initialStore[property];\r\n }\r\n\r\n let parent: any;\r\n let done = false;\r\n\r\n do {\r\n parent = Object.getPrototypeOf(currentTarget);\r\n if (!parent.getClassName) {\r\n done = true;\r\n break;\r\n }\r\n\r\n if (parent.getClassName() !== currentKey) {\r\n break;\r\n }\r\n\r\n currentTarget = parent;\r\n } while (parent);\r\n\r\n if (done) {\r\n break;\r\n }\r\n\r\n currentKey = parent.getClassName();\r\n currentTarget = parent;\r\n }\r\n\r\n return store;\r\n}\r\n\r\nfunction generateSerializableMember(type: number, sourceName?: string) {\r\n return (target: any, propertyKey: string | symbol) => {\r\n const classStore = getDirectStore(target);\r\n\r\n if (!classStore[propertyKey]) {\r\n classStore[propertyKey] = { type: type, sourceName: sourceName };\r\n }\r\n };\r\n}\r\n\r\nfunction generateExpandMember(setCallback: string, targetKey: Nullable = null) {\r\n return (target: any, propertyKey: string) => {\r\n const key = targetKey || \"_\" + propertyKey;\r\n Object.defineProperty(target, propertyKey, {\r\n get: function (this: any) {\r\n return this[key];\r\n },\r\n set: function (this: any, value) {\r\n // does this object (i.e. vector3) has an equals function? use it!\r\n // Note - not using \"with epsilon\" here, it is expected te behave like the internal cache does.\r\n if (typeof this.equals === \"function\") {\r\n if (this.equals(value)) {\r\n return;\r\n }\r\n }\r\n if (this[key] === value) {\r\n return;\r\n }\r\n this[key] = value;\r\n\r\n target[setCallback].apply(this);\r\n },\r\n enumerable: true,\r\n configurable: true,\r\n });\r\n };\r\n}\r\n\r\nexport function expandToProperty(callback: string, targetKey: Nullable = null) {\r\n return generateExpandMember(callback, targetKey);\r\n}\r\n\r\nexport function serialize(sourceName?: string) {\r\n return generateSerializableMember(0, sourceName); // value member\r\n}\r\n\r\nexport function serializeAsTexture(sourceName?: string) {\r\n return generateSerializableMember(1, sourceName); // texture member\r\n}\r\n\r\nexport function serializeAsColor3(sourceName?: string) {\r\n return generateSerializableMember(2, sourceName); // color3 member\r\n}\r\n\r\nexport function serializeAsFresnelParameters(sourceName?: string) {\r\n return generateSerializableMember(3, sourceName); // fresnel parameters member\r\n}\r\n\r\nexport function serializeAsVector2(sourceName?: string) {\r\n return generateSerializableMember(4, sourceName); // vector2 member\r\n}\r\n\r\nexport function serializeAsVector3(sourceName?: string) {\r\n return generateSerializableMember(5, sourceName); // vector3 member\r\n}\r\n\r\nexport function serializeAsMeshReference(sourceName?: string) {\r\n return generateSerializableMember(6, sourceName); // mesh reference member\r\n}\r\n\r\nexport function serializeAsColorCurves(sourceName?: string) {\r\n return generateSerializableMember(7, sourceName); // color curves\r\n}\r\n\r\nexport function serializeAsColor4(sourceName?: string) {\r\n return generateSerializableMember(8, sourceName); // color 4\r\n}\r\n\r\nexport function serializeAsImageProcessingConfiguration(sourceName?: string) {\r\n return generateSerializableMember(9, sourceName); // image processing\r\n}\r\n\r\nexport function serializeAsQuaternion(sourceName?: string) {\r\n return generateSerializableMember(10, sourceName); // quaternion member\r\n}\r\n\r\nexport function serializeAsMatrix(sourceName?: string) {\r\n return generateSerializableMember(12, sourceName); // matrix member\r\n}\r\n\r\n/**\r\n * Decorator used to define property that can be serialized as reference to a camera\r\n * @param sourceName defines the name of the property to decorate\r\n */\r\nexport function serializeAsCameraReference(sourceName?: string) {\r\n return generateSerializableMember(11, sourceName); // camera reference member\r\n}\r\n\r\n/**\r\n * Class used to help serialization objects\r\n */\r\nexport class SerializationHelper {\r\n /**\r\n * Gets or sets a boolean to indicate if the UniqueId property should be serialized\r\n */\r\n public static AllowLoadingUniqueId = false;\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _ImageProcessingConfigurationParser = (sourceProperty: any): ImageProcessingConfiguration => {\r\n throw _WarnImport(\"ImageProcessingConfiguration\");\r\n };\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _FresnelParametersParser = (sourceProperty: any): FresnelParameters => {\r\n throw _WarnImport(\"FresnelParameters\");\r\n };\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _ColorCurvesParser = (sourceProperty: any): ColorCurves => {\r\n throw _WarnImport(\"ColorCurves\");\r\n };\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _TextureParser = (sourceProperty: any, scene: Scene, rootUrl: string): Nullable => {\r\n throw _WarnImport(\"Texture\");\r\n };\r\n\r\n /**\r\n * Appends the serialized animations from the source animations\r\n * @param source Source containing the animations\r\n * @param destination Target to store the animations\r\n */\r\n public static AppendSerializedAnimations(source: IAnimatable, destination: any): void {\r\n if (source.animations) {\r\n destination.animations = [];\r\n for (let animationIndex = 0; animationIndex < source.animations.length; animationIndex++) {\r\n const animation = source.animations[animationIndex];\r\n\r\n destination.animations.push(animation.serialize());\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Static function used to serialized a specific entity\r\n * @param entity defines the entity to serialize\r\n * @param serializationObject defines the optional target object where serialization data will be stored\r\n * @returns a JSON compatible object representing the serialization of the entity\r\n */\r\n public static Serialize(entity: T, serializationObject?: any): any {\r\n if (!serializationObject) {\r\n serializationObject = {};\r\n }\r\n\r\n // Tags\r\n if (Tags) {\r\n serializationObject.tags = Tags.GetTags(entity);\r\n }\r\n\r\n const serializedProperties = getMergedStore(entity);\r\n\r\n // Properties\r\n for (const property in serializedProperties) {\r\n const propertyDescriptor = serializedProperties[property];\r\n const targetPropertyName = propertyDescriptor.sourceName || property;\r\n const propertyType = propertyDescriptor.type;\r\n const sourceProperty = (entity)[property];\r\n\r\n if (sourceProperty !== undefined && sourceProperty !== null && (property !== \"uniqueId\" || SerializationHelper.AllowLoadingUniqueId)) {\r\n switch (propertyType) {\r\n case 0: // Value\r\n serializationObject[targetPropertyName] = sourceProperty;\r\n break;\r\n case 1: // Texture\r\n serializationObject[targetPropertyName] = sourceProperty.serialize();\r\n break;\r\n case 2: // Color3\r\n serializationObject[targetPropertyName] = sourceProperty.asArray();\r\n break;\r\n case 3: // FresnelParameters\r\n serializationObject[targetPropertyName] = sourceProperty.serialize();\r\n break;\r\n case 4: // Vector2\r\n serializationObject[targetPropertyName] = sourceProperty.asArray();\r\n break;\r\n case 5: // Vector3\r\n serializationObject[targetPropertyName] = sourceProperty.asArray();\r\n break;\r\n case 6: // Mesh reference\r\n serializationObject[targetPropertyName] = sourceProperty.id;\r\n break;\r\n case 7: // Color Curves\r\n serializationObject[targetPropertyName] = sourceProperty.serialize();\r\n break;\r\n case 8: // Color 4\r\n serializationObject[targetPropertyName] = (sourceProperty).asArray();\r\n break;\r\n case 9: // Image Processing\r\n serializationObject[targetPropertyName] = (sourceProperty).serialize();\r\n break;\r\n case 10: // Quaternion\r\n serializationObject[targetPropertyName] = (sourceProperty).asArray();\r\n break;\r\n case 11: // Camera reference\r\n serializationObject[targetPropertyName] = (sourceProperty).id;\r\n break;\r\n case 12: // Matrix\r\n serializationObject[targetPropertyName] = (sourceProperty).asArray();\r\n break;\r\n }\r\n }\r\n }\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Given a source json and a destination object in a scene, this function will parse the source and will try to apply its content to the destination object\r\n * @param source the source json data\r\n * @param destination the destination object\r\n * @param scene the scene where the object is\r\n * @param rootUrl root url to use to load assets\r\n */\r\n public static ParseProperties(source: any, destination: any, scene: Nullable, rootUrl: Nullable) {\r\n if (!rootUrl) {\r\n rootUrl = \"\";\r\n }\r\n\r\n const classStore = getMergedStore(destination);\r\n\r\n // Properties\r\n for (const property in classStore) {\r\n const propertyDescriptor = classStore[property];\r\n const sourceProperty = source[propertyDescriptor.sourceName || property];\r\n const propertyType = propertyDescriptor.type;\r\n\r\n if (sourceProperty !== undefined && sourceProperty !== null && (property !== \"uniqueId\" || SerializationHelper.AllowLoadingUniqueId)) {\r\n const dest = destination;\r\n switch (propertyType) {\r\n case 0: // Value\r\n dest[property] = sourceProperty;\r\n break;\r\n case 1: // Texture\r\n if (scene) {\r\n dest[property] = SerializationHelper._TextureParser(sourceProperty, scene, rootUrl);\r\n }\r\n break;\r\n case 2: // Color3\r\n dest[property] = Color3.FromArray(sourceProperty);\r\n break;\r\n case 3: // FresnelParameters\r\n dest[property] = SerializationHelper._FresnelParametersParser(sourceProperty);\r\n break;\r\n case 4: // Vector2\r\n dest[property] = Vector2.FromArray(sourceProperty);\r\n break;\r\n case 5: // Vector3\r\n dest[property] = Vector3.FromArray(sourceProperty);\r\n break;\r\n case 6: // Mesh reference\r\n if (scene) {\r\n dest[property] = scene.getLastMeshById(sourceProperty);\r\n }\r\n break;\r\n case 7: // Color Curves\r\n dest[property] = SerializationHelper._ColorCurvesParser(sourceProperty);\r\n break;\r\n case 8: // Color 4\r\n dest[property] = Color4.FromArray(sourceProperty);\r\n break;\r\n case 9: // Image Processing\r\n dest[property] = SerializationHelper._ImageProcessingConfigurationParser(sourceProperty);\r\n break;\r\n case 10: // Quaternion\r\n dest[property] = Quaternion.FromArray(sourceProperty);\r\n break;\r\n case 11: // Camera reference\r\n if (scene) {\r\n dest[property] = scene.getCameraById(sourceProperty);\r\n }\r\n break;\r\n case 12: // Matrix\r\n dest[property] = Matrix.FromArray(sourceProperty);\r\n break;\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Creates a new entity from a serialization data object\r\n * @param creationFunction defines a function used to instanciated the new entity\r\n * @param source defines the source serialization data\r\n * @param scene defines the hosting scene\r\n * @param rootUrl defines the root url for resources\r\n * @returns a new entity\r\n */\r\n public static Parse(creationFunction: () => T, source: any, scene: Nullable, rootUrl: Nullable = null): T {\r\n const destination = creationFunction();\r\n\r\n // Tags\r\n if (Tags) {\r\n Tags.AddTagsTo(destination, source.tags);\r\n }\r\n\r\n SerializationHelper.ParseProperties(source, destination, scene, rootUrl);\r\n\r\n return destination;\r\n }\r\n\r\n /**\r\n * Clones an object\r\n * @param creationFunction defines the function used to instanciate the new object\r\n * @param source defines the source object\r\n * @returns the cloned object\r\n */\r\n public static Clone(creationFunction: () => T, source: T, options: CopySourceOptions = {}): T {\r\n return _copySource(creationFunction, source, false, options);\r\n }\r\n\r\n /**\r\n * Instanciates a new object based on a source one (some data will be shared between both object)\r\n * @param creationFunction defines the function used to instanciate the new object\r\n * @param source defines the source object\r\n * @returns the new object\r\n */\r\n public static Instanciate(creationFunction: () => T, source: T): T {\r\n return _copySource(creationFunction, source, true);\r\n }\r\n}\r\n\r\n/** @internal */\r\ndeclare const _native: any;\r\n\r\n/**\r\n * Decorator used to redirect a function to a native implementation if available.\r\n * @internal\r\n */\r\nexport function nativeOverride boolean>(\r\n target: any,\r\n propertyKey: string,\r\n descriptor: TypedPropertyDescriptor<(...params: Parameters) => unknown>,\r\n predicate?: T\r\n) {\r\n // Cache the original JS function for later.\r\n const jsFunc = descriptor.value!;\r\n\r\n // Override the JS function to check for a native override on first invocation. Setting descriptor.value overrides the function at the early stage of code being loaded/imported.\r\n descriptor.value = (...params: Parameters): unknown => {\r\n // Assume the resolved function will be the original JS function, then we will check for the Babylon Native context.\r\n let func = jsFunc;\r\n\r\n // Check if we are executing in a Babylon Native context (e.g. check the presence of the _native global property) and if so also check if a function override is available.\r\n if (typeof _native !== \"undefined\" && _native[propertyKey]) {\r\n const nativeFunc = _native[propertyKey] as (...params: Parameters) => unknown;\r\n // If a predicate was provided, then we'll need to invoke the predicate on each invocation of the underlying function to determine whether to call the native function or the JS function.\r\n if (predicate) {\r\n // The resolved function will execute the predicate and then either execute the native function or the JS function.\r\n func = (...params: Parameters) => (predicate(...params) ? nativeFunc(...params) : jsFunc(...params));\r\n } else {\r\n // The resolved function will directly execute the native function.\r\n func = nativeFunc;\r\n }\r\n }\r\n\r\n // Override the JS function again with the final resolved target function.\r\n target[propertyKey] = func;\r\n\r\n // The JS function has now been overridden based on whether we're executing in the context of Babylon Native, but we still need to invoke that function.\r\n // Future invocations of the function will just directly invoke the final overridden function, not any of the decorator setup logic above.\r\n return func(...params);\r\n };\r\n}\r\n\r\n/**\r\n * Decorator factory that applies the nativeOverride decorator, but determines whether to redirect to the native implementation based on a filter function that evaluates the function arguments.\r\n * @param predicate\r\n * @example @nativeOverride.filter((...[arg1]: Parameters) => arg1.length > 20)\r\n * public someMethod(arg1: string, arg2: number): string {\r\n * @internal\r\n */\r\nnativeOverride.filter = function boolean>(predicate: T) {\r\n return (target: any, propertyKey: string, descriptor: TypedPropertyDescriptor<(...params: Parameters) => unknown>) =>\r\n nativeOverride(target, propertyKey, descriptor, predicate);\r\n};\r\n","/* eslint-disable @typescript-eslint/no-unused-vars */\r\nimport type { Scene } from \"./scene\";\r\nimport type { Nullable } from \"./types\";\r\nimport { Matrix, Vector3 } from \"./Maths/math.vector\";\r\nimport type { Engine } from \"./Engines/engine\";\r\nimport type { IBehaviorAware, Behavior } from \"./Behaviors/behavior\";\r\nimport { SerializationHelper, serialize } from \"./Misc/decorators\";\r\nimport type { Observer } from \"./Misc/observable\";\r\nimport { Observable } from \"./Misc/observable\";\r\nimport { EngineStore } from \"./Engines/engineStore\";\r\nimport { _WarnImport } from \"./Misc/devTools\";\r\nimport type { AbstractActionManager } from \"./Actions/abstractActionManager\";\r\nimport type { IInspectable } from \"./Misc/iInspectable\";\r\nimport type { AbstractScene } from \"./abstractScene\";\r\nimport type { IAccessibilityTag } from \"./IAccessibilityTag\";\r\nimport type { AnimationRange } from \"./Animations/animationRange\";\r\nimport type { AnimationPropertiesOverride } from \"./Animations/animationPropertiesOverride\";\r\nimport type { AbstractMesh } from \"./Meshes/abstractMesh\";\r\nimport type { Animation } from \"./Animations/animation\";\r\nimport type { Animatable } from \"./Animations/animatable\";\r\n\r\n/**\r\n * Defines how a node can be built from a string name.\r\n */\r\nexport type NodeConstructor = (name: string, scene: Scene, options?: any) => () => Node;\r\n\r\n/** @internal */\r\nclass _InternalNodeDataInfo {\r\n public _doNotSerialize = false;\r\n public _isDisposed = false;\r\n public _sceneRootNodesIndex = -1;\r\n public _isEnabled = true;\r\n public _isParentEnabled = true;\r\n public _isReady = true;\r\n public _onEnabledStateChangedObservable = new Observable();\r\n public _onClonedObservable = new Observable();\r\n}\r\n\r\n/**\r\n * Node is the basic class for all scene objects (Mesh, Light, Camera.)\r\n */\r\nexport class Node implements IBehaviorAware {\r\n protected _isDirty = false;\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _AnimationRangeFactory = (_name: string, _from: number, _to: number): AnimationRange => {\r\n throw _WarnImport(\"AnimationRange\");\r\n };\r\n\r\n private static _NodeConstructors: { [key: string]: any } = {};\r\n\r\n /**\r\n * Add a new node constructor\r\n * @param type defines the type name of the node to construct\r\n * @param constructorFunc defines the constructor function\r\n */\r\n public static AddNodeConstructor(type: string, constructorFunc: NodeConstructor) {\r\n this._NodeConstructors[type] = constructorFunc;\r\n }\r\n\r\n /**\r\n * Returns a node constructor based on type name\r\n * @param type defines the type name\r\n * @param name defines the new node name\r\n * @param scene defines the hosting scene\r\n * @param options defines optional options to transmit to constructors\r\n * @returns the new constructor or null\r\n */\r\n public static Construct(type: string, name: string, scene: Scene, options?: any): Nullable<() => Node> {\r\n const constructorFunc = this._NodeConstructors[type];\r\n\r\n if (!constructorFunc) {\r\n return null;\r\n }\r\n\r\n return constructorFunc(name, scene, options);\r\n }\r\n\r\n private _nodeDataStorage = new _InternalNodeDataInfo();\r\n\r\n /**\r\n * Gets or sets the name of the node\r\n */\r\n @serialize()\r\n public name: string;\r\n\r\n /**\r\n * Gets or sets the id of the node\r\n */\r\n @serialize()\r\n public id: string;\r\n\r\n /**\r\n * Gets or sets the unique id of the node\r\n */\r\n @serialize()\r\n public uniqueId: number;\r\n\r\n /**\r\n * Gets or sets a string used to store user defined state for the node\r\n */\r\n @serialize()\r\n public state = \"\";\r\n\r\n /**\r\n * Gets or sets an object used to store user defined information for the node\r\n */\r\n @serialize()\r\n public metadata: any = null;\r\n\r\n /** @internal */\r\n public _internalMetadata: any;\r\n\r\n /**\r\n * For internal use only. Please do not use.\r\n */\r\n public reservedDataStore: any = null;\r\n\r\n /**\r\n * List of inspectable custom properties (used by the Inspector)\r\n * @see https://doc.babylonjs.com/toolsAndResources/inspector#extensibility\r\n */\r\n public inspectableCustomProperties: IInspectable[];\r\n\r\n /**\r\n * Gets or sets the accessibility tag to describe the node for accessibility purpose.\r\n */\r\n public set accessibilityTag(value: Nullable) {\r\n this._accessibilityTag = value;\r\n this.onAccessibilityTagChangedObservable.notifyObservers(value);\r\n }\r\n\r\n public get accessibilityTag() {\r\n return this._accessibilityTag;\r\n }\r\n\r\n protected _accessibilityTag: Nullable = null;\r\n\r\n public onAccessibilityTagChangedObservable = new Observable>();\r\n\r\n /**\r\n * Gets or sets a boolean used to define if the node must be serialized\r\n */\r\n public get doNotSerialize() {\r\n if (this._nodeDataStorage._doNotSerialize) {\r\n return true;\r\n }\r\n\r\n if (this._parentNode) {\r\n return this._parentNode.doNotSerialize;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n public set doNotSerialize(value: boolean) {\r\n this._nodeDataStorage._doNotSerialize = value;\r\n }\r\n\r\n /** @internal */\r\n public _parentContainer: Nullable = null;\r\n\r\n /**\r\n * Gets a list of Animations associated with the node\r\n */\r\n public animations = new Array();\r\n protected _ranges: { [name: string]: Nullable } = {};\r\n\r\n /**\r\n * Callback raised when the node is ready to be used\r\n */\r\n public onReady: Nullable<(node: Node) => void> = null;\r\n\r\n /** @internal */\r\n public _currentRenderId = -1;\r\n private _parentUpdateId = -1;\r\n /** @internal */\r\n public _childUpdateId = -1;\r\n\r\n /** @internal */\r\n public _waitingParentId: Nullable = null;\r\n /** @internal */\r\n public _waitingParentInstanceIndex: Nullable = null;\r\n /** @internal */\r\n public _waitingParsedUniqueId: Nullable = null;\r\n /** @internal */\r\n public _scene: Scene;\r\n /** @internal */\r\n public _cache: any = {};\r\n\r\n protected _parentNode: Nullable = null;\r\n\r\n /** @internal */\r\n protected _children: Nullable = null;\r\n\r\n /** @internal */\r\n public _worldMatrix = Matrix.Identity();\r\n /** @internal */\r\n public _worldMatrixDeterminant = 0;\r\n /** @internal */\r\n public _worldMatrixDeterminantIsDirty = true;\r\n\r\n /**\r\n * Gets a boolean indicating if the node has been disposed\r\n * @returns true if the node was disposed\r\n */\r\n public isDisposed(): boolean {\r\n return this._nodeDataStorage._isDisposed;\r\n }\r\n\r\n /**\r\n * Gets or sets the parent of the node (without keeping the current position in the scene)\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/parent_pivot/parent\r\n */\r\n public set parent(parent: Nullable) {\r\n if (this._parentNode === parent) {\r\n return;\r\n }\r\n\r\n const previousParentNode = this._parentNode;\r\n\r\n // Remove self from list of children of parent\r\n if (this._parentNode && this._parentNode._children !== undefined && this._parentNode._children !== null) {\r\n const index = this._parentNode._children.indexOf(this);\r\n if (index !== -1) {\r\n this._parentNode._children.splice(index, 1);\r\n }\r\n\r\n if (!parent && !this._nodeDataStorage._isDisposed) {\r\n this._addToSceneRootNodes();\r\n }\r\n }\r\n\r\n // Store new parent\r\n this._parentNode = parent;\r\n\r\n // Add as child to new parent\r\n if (this._parentNode) {\r\n if (this._parentNode._children === undefined || this._parentNode._children === null) {\r\n this._parentNode._children = new Array();\r\n }\r\n this._parentNode._children.push(this);\r\n\r\n if (!previousParentNode) {\r\n this._removeFromSceneRootNodes();\r\n }\r\n }\r\n\r\n // Enabled state\r\n this._syncParentEnabledState();\r\n }\r\n\r\n public get parent(): Nullable {\r\n return this._parentNode;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _serializeAsParent(serializationObject: any): void {\r\n serializationObject.parentId = this.uniqueId;\r\n }\r\n\r\n /** @internal */\r\n public _addToSceneRootNodes() {\r\n if (this._nodeDataStorage._sceneRootNodesIndex === -1) {\r\n this._nodeDataStorage._sceneRootNodesIndex = this._scene.rootNodes.length;\r\n this._scene.rootNodes.push(this);\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _removeFromSceneRootNodes() {\r\n if (this._nodeDataStorage._sceneRootNodesIndex !== -1) {\r\n const rootNodes = this._scene.rootNodes;\r\n const lastIdx = rootNodes.length - 1;\r\n rootNodes[this._nodeDataStorage._sceneRootNodesIndex] = rootNodes[lastIdx];\r\n rootNodes[this._nodeDataStorage._sceneRootNodesIndex]._nodeDataStorage._sceneRootNodesIndex = this._nodeDataStorage._sceneRootNodesIndex;\r\n this._scene.rootNodes.pop();\r\n this._nodeDataStorage._sceneRootNodesIndex = -1;\r\n }\r\n }\r\n\r\n private _animationPropertiesOverride: Nullable = null;\r\n\r\n /**\r\n * Gets or sets the animation properties override\r\n */\r\n public get animationPropertiesOverride(): Nullable {\r\n if (!this._animationPropertiesOverride) {\r\n return this._scene.animationPropertiesOverride;\r\n }\r\n return this._animationPropertiesOverride;\r\n }\r\n\r\n public set animationPropertiesOverride(value: Nullable) {\r\n this._animationPropertiesOverride = value;\r\n }\r\n\r\n /**\r\n * Gets a string identifying the name of the class\r\n * @returns \"Node\" string\r\n */\r\n public getClassName(): string {\r\n return \"Node\";\r\n }\r\n\r\n /** @internal */\r\n public readonly _isNode = true;\r\n\r\n /**\r\n * An event triggered when the mesh is disposed\r\n */\r\n public onDisposeObservable = new Observable();\r\n\r\n private _onDisposeObserver: Nullable> = null;\r\n /**\r\n * Sets a callback that will be raised when the node will be disposed\r\n */\r\n public set onDispose(callback: () => void) {\r\n if (this._onDisposeObserver) {\r\n this.onDisposeObservable.remove(this._onDisposeObserver);\r\n }\r\n this._onDisposeObserver = this.onDisposeObservable.add(callback);\r\n }\r\n\r\n /**\r\n * An event triggered when the enabled state of the node changes\r\n */\r\n public get onEnabledStateChangedObservable(): Observable {\r\n return this._nodeDataStorage._onEnabledStateChangedObservable;\r\n }\r\n\r\n /**\r\n * An event triggered when the node is cloned\r\n */\r\n public get onClonedObservable(): Observable {\r\n return this._nodeDataStorage._onClonedObservable;\r\n }\r\n\r\n /**\r\n * Creates a new Node\r\n * @param name the name and id to be given to this node\r\n * @param scene the scene this node will be added to\r\n */\r\n constructor(name: string, scene: Nullable = null) {\r\n this.name = name;\r\n this.id = name;\r\n this._scene = (scene || EngineStore.LastCreatedScene);\r\n this.uniqueId = this._scene.getUniqueId();\r\n this._initCache();\r\n }\r\n\r\n /**\r\n * Gets the scene of the node\r\n * @returns a scene\r\n */\r\n public getScene(): Scene {\r\n return this._scene;\r\n }\r\n\r\n /**\r\n * Gets the engine of the node\r\n * @returns a Engine\r\n */\r\n public getEngine(): Engine {\r\n return this._scene.getEngine();\r\n }\r\n\r\n // Behaviors\r\n private _behaviors = new Array>();\r\n\r\n /**\r\n * Attach a behavior to the node\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors\r\n * @param behavior defines the behavior to attach\r\n * @param attachImmediately defines that the behavior must be attached even if the scene is still loading\r\n * @returns the current Node\r\n */\r\n public addBehavior(behavior: Behavior, attachImmediately = false): Node {\r\n const index = this._behaviors.indexOf(behavior);\r\n\r\n if (index !== -1) {\r\n return this;\r\n }\r\n\r\n behavior.init();\r\n if (this._scene.isLoading && !attachImmediately) {\r\n // We defer the attach when the scene will be loaded\r\n this._scene.onDataLoadedObservable.addOnce(() => {\r\n behavior.attach(this);\r\n });\r\n } else {\r\n behavior.attach(this);\r\n }\r\n this._behaviors.push(behavior);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Remove an attached behavior\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors\r\n * @param behavior defines the behavior to attach\r\n * @returns the current Node\r\n */\r\n public removeBehavior(behavior: Behavior): Node {\r\n const index = this._behaviors.indexOf(behavior);\r\n\r\n if (index === -1) {\r\n return this;\r\n }\r\n\r\n this._behaviors[index].detach();\r\n this._behaviors.splice(index, 1);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Gets the list of attached behaviors\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors\r\n */\r\n public get behaviors(): Behavior[] {\r\n return this._behaviors;\r\n }\r\n\r\n /**\r\n * Gets an attached behavior by name\r\n * @param name defines the name of the behavior to look for\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors\r\n * @returns null if behavior was not found else the requested behavior\r\n */\r\n public getBehaviorByName(name: string): Nullable> {\r\n for (const behavior of this._behaviors) {\r\n if (behavior.name === name) {\r\n return behavior;\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Returns the latest update of the World matrix\r\n * @returns a Matrix\r\n */\r\n public getWorldMatrix(): Matrix {\r\n if (this._currentRenderId !== this._scene.getRenderId()) {\r\n this.computeWorldMatrix();\r\n }\r\n return this._worldMatrix;\r\n }\r\n\r\n /** @internal */\r\n public _getWorldMatrixDeterminant(): number {\r\n if (this._worldMatrixDeterminantIsDirty) {\r\n this._worldMatrixDeterminantIsDirty = false;\r\n this._worldMatrixDeterminant = this._worldMatrix.determinant();\r\n }\r\n return this._worldMatrixDeterminant;\r\n }\r\n\r\n /**\r\n * Returns directly the latest state of the mesh World matrix.\r\n * A Matrix is returned.\r\n */\r\n public get worldMatrixFromCache(): Matrix {\r\n return this._worldMatrix;\r\n }\r\n\r\n // override it in derived class if you add new variables to the cache\r\n // and call the parent class method\r\n /** @internal */\r\n public _initCache() {\r\n this._cache = {};\r\n this._cache.parent = undefined;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public updateCache(force?: boolean): void {\r\n if (!force && this.isSynchronized()) {\r\n return;\r\n }\r\n\r\n this._cache.parent = this.parent;\r\n\r\n this._updateCache();\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getActionManagerForTrigger(trigger?: number, _initialCall = true): Nullable {\r\n if (!this.parent) {\r\n return null;\r\n }\r\n\r\n return this.parent._getActionManagerForTrigger(trigger, false);\r\n }\r\n\r\n // override it in derived class if you add new variables to the cache\r\n // and call the parent class method if !ignoreParentClass\r\n /**\r\n * @internal\r\n */\r\n public _updateCache(_ignoreParentClass?: boolean): void {}\r\n\r\n // override it in derived class if you add new variables to the cache\r\n /** @internal */\r\n public _isSynchronized(): boolean {\r\n return true;\r\n }\r\n\r\n /** @internal */\r\n public _markSyncedWithParent() {\r\n if (this._parentNode) {\r\n this._parentUpdateId = this._parentNode._childUpdateId;\r\n }\r\n }\r\n\r\n /** @internal */\r\n public isSynchronizedWithParent(): boolean {\r\n if (!this._parentNode) {\r\n return true;\r\n }\r\n\r\n if (this._parentNode._isDirty || this._parentUpdateId !== this._parentNode._childUpdateId) {\r\n return false;\r\n }\r\n\r\n return this._parentNode.isSynchronized();\r\n }\r\n\r\n /** @internal */\r\n public isSynchronized(): boolean {\r\n if (this._cache.parent !== this._parentNode) {\r\n this._cache.parent = this._parentNode;\r\n return false;\r\n }\r\n\r\n if (this._parentNode && !this.isSynchronizedWithParent()) {\r\n return false;\r\n }\r\n\r\n return this._isSynchronized();\r\n }\r\n\r\n /**\r\n * Is this node ready to be used/rendered\r\n * @param _completeCheck defines if a complete check (including materials and lights) has to be done (false by default)\r\n * @returns true if the node is ready\r\n */\r\n public isReady(_completeCheck = false): boolean {\r\n return this._nodeDataStorage._isReady;\r\n }\r\n\r\n /**\r\n * Flag the node as dirty (Forcing it to update everything)\r\n * @param _property helps children apply precise \"dirtyfication\"\r\n * @returns this node\r\n */\r\n public markAsDirty(_property?: string): Node {\r\n this._currentRenderId = Number.MAX_VALUE;\r\n this._isDirty = true;\r\n return this;\r\n }\r\n\r\n /**\r\n * Is this node enabled?\r\n * If the node has a parent, all ancestors will be checked and false will be returned if any are false (not enabled), otherwise will return true\r\n * @param checkAncestors indicates if this method should check the ancestors. The default is to check the ancestors. If set to false, the method will return the value of this node without checking ancestors\r\n * @returns whether this node (and its parent) is enabled\r\n */\r\n public isEnabled(checkAncestors: boolean = true): boolean {\r\n if (checkAncestors === false) {\r\n return this._nodeDataStorage._isEnabled;\r\n }\r\n\r\n if (!this._nodeDataStorage._isEnabled) {\r\n return false;\r\n }\r\n\r\n return this._nodeDataStorage._isParentEnabled;\r\n }\r\n\r\n /** @internal */\r\n protected _syncParentEnabledState() {\r\n this._nodeDataStorage._isParentEnabled = this._parentNode ? this._parentNode.isEnabled() : true;\r\n\r\n if (this._children) {\r\n this._children.forEach((c) => {\r\n c._syncParentEnabledState(); // Force children to update accordingly\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Set the enabled state of this node\r\n * @param value defines the new enabled state\r\n */\r\n public setEnabled(value: boolean): void {\r\n if (this._nodeDataStorage._isEnabled === value) {\r\n return;\r\n }\r\n this._nodeDataStorage._isEnabled = value;\r\n this._syncParentEnabledState();\r\n this._nodeDataStorage._onEnabledStateChangedObservable.notifyObservers(value);\r\n }\r\n\r\n /**\r\n * Is this node a descendant of the given node?\r\n * The function will iterate up the hierarchy until the ancestor was found or no more parents defined\r\n * @param ancestor defines the parent node to inspect\r\n * @returns a boolean indicating if this node is a descendant of the given node\r\n */\r\n public isDescendantOf(ancestor: Node): boolean {\r\n if (this.parent) {\r\n if (this.parent === ancestor) {\r\n return true;\r\n }\r\n\r\n return this.parent.isDescendantOf(ancestor);\r\n }\r\n return false;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getDescendants(results: Node[], directDescendantsOnly: boolean = false, predicate?: (node: Node) => boolean): void {\r\n if (!this._children) {\r\n return;\r\n }\r\n\r\n for (let index = 0; index < this._children.length; index++) {\r\n const item = this._children[index];\r\n\r\n if (!predicate || predicate(item)) {\r\n results.push(item);\r\n }\r\n\r\n if (!directDescendantsOnly) {\r\n item._getDescendants(results, false, predicate);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Will return all nodes that have this node as ascendant\r\n * @param directDescendantsOnly defines if true only direct descendants of 'this' will be considered, if false direct and also indirect (children of children, an so on in a recursive manner) descendants of 'this' will be considered\r\n * @param predicate defines an optional predicate that will be called on every evaluated child, the predicate must return true for a given child to be part of the result, otherwise it will be ignored\r\n * @returns all children nodes of all types\r\n */\r\n public getDescendants(directDescendantsOnly?: boolean, predicate?: (node: Node) => node is T): T[];\r\n\r\n /**\r\n * Will return all nodes that have this node as ascendant\r\n * @param directDescendantsOnly defines if true only direct descendants of 'this' will be considered, if false direct and also indirect (children of children, an so on in a recursive manner) descendants of 'this' will be considered\r\n * @param predicate defines an optional predicate that will be called on every evaluated child, the predicate must return true for a given child to be part of the result, otherwise it will be ignored\r\n * @returns all children nodes of all types\r\n */\r\n public getDescendants(directDescendantsOnly?: boolean, predicate?: (node: Node) => boolean): Node[];\r\n\r\n /**\r\n * Will return all nodes that have this node as ascendant\r\n * @param directDescendantsOnly defines if true only direct descendants of 'this' will be considered, if false direct and also indirect (children of children, an so on in a recursive manner) descendants of 'this' will be considered\r\n * @param predicate defines an optional predicate that will be called on every evaluated child, the predicate must return true for a given child to be part of the result, otherwise it will be ignored\r\n * @returns all children nodes of all types\r\n */\r\n public getDescendants(directDescendantsOnly?: boolean, predicate?: (node: Node) => boolean): Node[] {\r\n const results = new Array();\r\n\r\n this._getDescendants(results, directDescendantsOnly, predicate);\r\n\r\n return results;\r\n }\r\n\r\n /**\r\n * Get all child-meshes of this node\r\n * @param directDescendantsOnly defines if true only direct descendants of 'this' will be considered, if false direct and also indirect (children of children, an so on in a recursive manner) descendants of 'this' will be considered (Default: false)\r\n * @param predicate defines an optional predicate that will be called on every evaluated child, the predicate must return true for a given child to be part of the result, otherwise it will be ignored\r\n * @returns an array of AbstractMesh\r\n */\r\n public getChildMeshes(directDescendantsOnly?: boolean, predicate?: (node: Node) => node is T): T[];\r\n\r\n /**\r\n * Get all child-meshes of this node\r\n * @param directDescendantsOnly defines if true only direct descendants of 'this' will be considered, if false direct and also indirect (children of children, an so on in a recursive manner) descendants of 'this' will be considered (Default: false)\r\n * @param predicate defines an optional predicate that will be called on every evaluated child, the predicate must return true for a given child to be part of the result, otherwise it will be ignored\r\n * @returns an array of AbstractMesh\r\n */\r\n public getChildMeshes(directDescendantsOnly?: boolean, predicate?: (node: Node) => boolean): AbstractMesh[];\r\n\r\n /**\r\n * Get all child-meshes of this node\r\n * @param directDescendantsOnly defines if true only direct descendants of 'this' will be considered, if false direct and also indirect (children of children, an so on in a recursive manner) descendants of 'this' will be considered (Default: false)\r\n * @param predicate defines an optional predicate that will be called on every evaluated child, the predicate must return true for a given child to be part of the result, otherwise it will be ignored\r\n * @returns an array of AbstractMesh\r\n */\r\n public getChildMeshes(directDescendantsOnly?: boolean, predicate?: (node: Node) => boolean): AbstractMesh[] {\r\n const results: Array = [];\r\n this._getDescendants(results, directDescendantsOnly, (node: Node) => {\r\n return (!predicate || predicate(node)) && (node).cullingStrategy !== undefined;\r\n });\r\n return results;\r\n }\r\n\r\n /**\r\n * Get all direct children of this node\r\n * @param predicate defines an optional predicate that will be called on every evaluated child, the predicate must return true for a given child to be part of the result, otherwise it will be ignored\r\n * @param directDescendantsOnly defines if true only direct descendants of 'this' will be considered, if false direct and also indirect (children of children, an so on in a recursive manner) descendants of 'this' will be considered (Default: true)\r\n * @returns an array of Node\r\n */\r\n public getChildren(predicate?: (node: Node) => node is T, directDescendantsOnly?: boolean): T[];\r\n\r\n /**\r\n * Get all direct children of this node\r\n * @param predicate defines an optional predicate that will be called on every evaluated child, the predicate must return true for a given child to be part of the result, otherwise it will be ignored\r\n * @param directDescendantsOnly defines if true only direct descendants of 'this' will be considered, if false direct and also indirect (children of children, an so on in a recursive manner) descendants of 'this' will be considered (Default: true)\r\n * @returns an array of Node\r\n */\r\n public getChildren(predicate?: (node: Node) => boolean, directDescendantsOnly?: boolean): Node[];\r\n\r\n /**\r\n * Get all direct children of this node\r\n * @param predicate defines an optional predicate that will be called on every evaluated child, the predicate must return true for a given child to be part of the result, otherwise it will be ignored\r\n * @param directDescendantsOnly defines if true only direct descendants of 'this' will be considered, if false direct and also indirect (children of children, an so on in a recursive manner) descendants of 'this' will be considered (Default: true)\r\n * @returns an array of Node\r\n */\r\n public getChildren(predicate?: (node: Node) => boolean, directDescendantsOnly = true): Node[] {\r\n return this.getDescendants(directDescendantsOnly, predicate);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _setReady(state: boolean): void {\r\n if (state === this._nodeDataStorage._isReady) {\r\n return;\r\n }\r\n\r\n if (!state) {\r\n this._nodeDataStorage._isReady = false;\r\n return;\r\n }\r\n\r\n if (this.onReady) {\r\n this.onReady(this);\r\n }\r\n this._nodeDataStorage._isReady = true;\r\n }\r\n\r\n /**\r\n * Get an animation by name\r\n * @param name defines the name of the animation to look for\r\n * @returns null if not found else the requested animation\r\n */\r\n public getAnimationByName(name: string): Nullable {\r\n for (let i = 0; i < this.animations.length; i++) {\r\n const animation = this.animations[i];\r\n\r\n if (animation.name === name) {\r\n return animation;\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Creates an animation range for this node\r\n * @param name defines the name of the range\r\n * @param from defines the starting key\r\n * @param to defines the end key\r\n */\r\n public createAnimationRange(name: string, from: number, to: number): void {\r\n // check name not already in use\r\n if (!this._ranges[name]) {\r\n this._ranges[name] = Node._AnimationRangeFactory(name, from, to);\r\n for (let i = 0, nAnimations = this.animations.length; i < nAnimations; i++) {\r\n if (this.animations[i]) {\r\n this.animations[i].createRange(name, from, to);\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Delete a specific animation range\r\n * @param name defines the name of the range to delete\r\n * @param deleteFrames defines if animation frames from the range must be deleted as well\r\n */\r\n public deleteAnimationRange(name: string, deleteFrames = true): void {\r\n for (let i = 0, nAnimations = this.animations.length; i < nAnimations; i++) {\r\n if (this.animations[i]) {\r\n this.animations[i].deleteRange(name, deleteFrames);\r\n }\r\n }\r\n this._ranges[name] = null; // said much faster than 'delete this._range[name]'\r\n }\r\n\r\n /**\r\n * Get an animation range by name\r\n * @param name defines the name of the animation range to look for\r\n * @returns null if not found else the requested animation range\r\n */\r\n public getAnimationRange(name: string): Nullable {\r\n return this._ranges[name] || null;\r\n }\r\n\r\n /**\r\n * Clone the current node\r\n * @param name Name of the new clone\r\n * @param newParent New parent for the clone\r\n * @param doNotCloneChildren Do not clone children hierarchy\r\n * @returns the new transform node\r\n */\r\n public clone(name: string, newParent: Nullable, doNotCloneChildren?: boolean): Nullable {\r\n const result = SerializationHelper.Clone(() => new Node(name, this.getScene()), this);\r\n\r\n if (newParent) {\r\n result.parent = newParent;\r\n }\r\n\r\n if (!doNotCloneChildren) {\r\n // Children\r\n const directDescendants = this.getDescendants(true);\r\n for (let index = 0; index < directDescendants.length; index++) {\r\n const child = directDescendants[index];\r\n\r\n child.clone(name + \".\" + child.name, result);\r\n }\r\n }\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Gets the list of all animation ranges defined on this node\r\n * @returns an array\r\n */\r\n public getAnimationRanges(): Nullable[] {\r\n const animationRanges: Nullable[] = [];\r\n let name: string;\r\n for (name in this._ranges) {\r\n animationRanges.push(this._ranges[name]);\r\n }\r\n return animationRanges;\r\n }\r\n\r\n /**\r\n * Will start the animation sequence\r\n * @param name defines the range frames for animation sequence\r\n * @param loop defines if the animation should loop (false by default)\r\n * @param speedRatio defines the speed factor in which to run the animation (1 by default)\r\n * @param onAnimationEnd defines a function to be executed when the animation ended (undefined by default)\r\n * @returns the object created for this animation. If range does not exist, it will return null\r\n */\r\n public beginAnimation(name: string, loop?: boolean, speedRatio?: number, onAnimationEnd?: () => void): Nullable {\r\n const range = this.getAnimationRange(name);\r\n\r\n if (!range) {\r\n return null;\r\n }\r\n\r\n return this._scene.beginAnimation(this, range.from, range.to, loop, speedRatio, onAnimationEnd);\r\n }\r\n\r\n /**\r\n * Serialize animation ranges into a JSON compatible object\r\n * @returns serialization object\r\n */\r\n public serializeAnimationRanges(): any {\r\n const serializationRanges = [];\r\n for (const name in this._ranges) {\r\n const localRange = this._ranges[name];\r\n if (!localRange) {\r\n continue;\r\n }\r\n const range: any = {};\r\n range.name = name;\r\n range.from = localRange.from;\r\n range.to = localRange.to;\r\n serializationRanges.push(range);\r\n }\r\n return serializationRanges;\r\n }\r\n\r\n /**\r\n * Computes the world matrix of the node\r\n * @param _force defines if the cache version should be invalidated forcing the world matrix to be created from scratch\r\n * @returns the world matrix\r\n */\r\n public computeWorldMatrix(_force?: boolean): Matrix {\r\n if (!this._worldMatrix) {\r\n this._worldMatrix = Matrix.Identity();\r\n }\r\n return this._worldMatrix;\r\n }\r\n\r\n /**\r\n * Releases resources associated with this node.\r\n * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default)\r\n * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default)\r\n */\r\n public dispose(doNotRecurse?: boolean, disposeMaterialAndTextures = false): void {\r\n this._nodeDataStorage._isDisposed = true;\r\n\r\n if (!doNotRecurse) {\r\n const nodes = this.getDescendants(true);\r\n for (const node of nodes) {\r\n node.dispose(doNotRecurse, disposeMaterialAndTextures);\r\n }\r\n }\r\n\r\n if (!this.parent) {\r\n this._removeFromSceneRootNodes();\r\n } else {\r\n this.parent = null;\r\n }\r\n\r\n // Callback\r\n this.onDisposeObservable.notifyObservers(this);\r\n this.onDisposeObservable.clear();\r\n\r\n this.onEnabledStateChangedObservable.clear();\r\n this.onClonedObservable.clear();\r\n\r\n // Behaviors\r\n for (const behavior of this._behaviors) {\r\n behavior.detach();\r\n }\r\n\r\n this._behaviors.length = 0;\r\n\r\n this.metadata = null;\r\n }\r\n\r\n /**\r\n * Parse animation range data from a serialization object and store them into a given node\r\n * @param node defines where to store the animation ranges\r\n * @param parsedNode defines the serialization object to read data from\r\n * @param _scene defines the hosting scene\r\n */\r\n public static ParseAnimationRanges(node: Node, parsedNode: any, _scene: Scene): void {\r\n if (parsedNode.ranges) {\r\n for (let index = 0; index < parsedNode.ranges.length; index++) {\r\n const data = parsedNode.ranges[index];\r\n node.createAnimationRange(data.name, data.from, data.to);\r\n }\r\n }\r\n }\r\n /**\r\n * Return the minimum and maximum world vectors of the entire hierarchy under current node\r\n * @param includeDescendants Include bounding info from descendants as well (true by default)\r\n * @param predicate defines a callback function that can be customize to filter what meshes should be included in the list used to compute the bounding vectors\r\n * @returns the new bounding vectors\r\n */\r\n public getHierarchyBoundingVectors(includeDescendants = true, predicate: Nullable<(abstractMesh: AbstractMesh) => boolean> = null): { min: Vector3; max: Vector3 } {\r\n // Ensures that all world matrix will be recomputed.\r\n this.getScene().incrementRenderId();\r\n\r\n this.computeWorldMatrix(true);\r\n\r\n let min: Vector3;\r\n let max: Vector3;\r\n\r\n const thisAbstractMesh = this as Node as AbstractMesh;\r\n if (thisAbstractMesh.getBoundingInfo && thisAbstractMesh.subMeshes) {\r\n // If this is an abstract mesh get its bounding info\r\n const boundingInfo = thisAbstractMesh.getBoundingInfo();\r\n min = boundingInfo.boundingBox.minimumWorld.clone();\r\n max = boundingInfo.boundingBox.maximumWorld.clone();\r\n } else {\r\n min = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\r\n max = new Vector3(-Number.MAX_VALUE, -Number.MAX_VALUE, -Number.MAX_VALUE);\r\n }\r\n\r\n if (includeDescendants) {\r\n const descendants = this.getDescendants(false);\r\n\r\n for (const descendant of descendants) {\r\n const childMesh = descendant;\r\n childMesh.computeWorldMatrix(true);\r\n\r\n // Filters meshes based on custom predicate function.\r\n if (predicate && !predicate(childMesh)) {\r\n continue;\r\n }\r\n\r\n //make sure we have the needed params to get mix and max\r\n if (!childMesh.getBoundingInfo || childMesh.getTotalVertices() === 0) {\r\n continue;\r\n }\r\n\r\n const childBoundingInfo = childMesh.getBoundingInfo();\r\n const boundingBox = childBoundingInfo.boundingBox;\r\n\r\n const minBox = boundingBox.minimumWorld;\r\n const maxBox = boundingBox.maximumWorld;\r\n\r\n Vector3.CheckExtends(minBox, min, max);\r\n Vector3.CheckExtends(maxBox, min, max);\r\n }\r\n }\r\n\r\n return {\r\n min: min,\r\n max: max,\r\n };\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\n/**\r\n * Checks if the window object exists\r\n * @returns true if the window object exists\r\n */\r\nexport function IsWindowObjectExist(): boolean {\r\n return typeof window !== \"undefined\";\r\n}\r\n\r\n/**\r\n * Checks if the navigator object exists\r\n * @returns true if the navigator object exists\r\n */\r\nexport function IsNavigatorAvailable(): boolean {\r\n return typeof navigator !== \"undefined\";\r\n}\r\n\r\n/**\r\n * Check if the document object exists\r\n * @returns true if the document object exists\r\n */\r\nexport function IsDocumentAvailable(): boolean {\r\n return typeof document !== \"undefined\";\r\n}\r\n\r\n/**\r\n * Extracts text content from a DOM element hierarchy\r\n * @param element defines the root element\r\n * @returns a string\r\n */\r\nexport function GetDOMTextContent(element: HTMLElement): string {\r\n let result = \"\";\r\n let child = element.firstChild;\r\n\r\n while (child) {\r\n if (child.nodeType === 3) {\r\n result += child.textContent;\r\n }\r\n child = child.nextSibling;\r\n }\r\n\r\n return result;\r\n}\r\n\r\n/**\r\n * Sets of helpers dealing with the DOM and some of the recurrent functions needed in\r\n * Babylon.js\r\n */\r\nexport const DomManagement = {\r\n /**\r\n * Checks if the window object exists\r\n * @returns true if the window object exists\r\n */\r\n IsWindowObjectExist,\r\n\r\n /**\r\n * Checks if the navigator object exists\r\n * @returns true if the navigator object exists\r\n */\r\n IsNavigatorAvailable,\r\n\r\n /**\r\n * Check if the document object exists\r\n * @returns true if the document object exists\r\n */\r\n IsDocumentAvailable,\r\n /**\r\n * Extracts text content from a DOM element hierarchy\r\n * @param element defines the root element\r\n * @returns a string\r\n */\r\n GetDOMTextContent,\r\n};\r\n","/**\r\n * Logger used throughout the application to allow configuration of\r\n * the log level required for the messages.\r\n */\r\nexport class Logger {\r\n /**\r\n * No log\r\n */\r\n public static readonly NoneLogLevel = 0;\r\n /**\r\n * Only message logs\r\n */\r\n public static readonly MessageLogLevel = 1;\r\n /**\r\n * Only warning logs\r\n */\r\n public static readonly WarningLogLevel = 2;\r\n /**\r\n * Only error logs\r\n */\r\n public static readonly ErrorLogLevel = 4;\r\n /**\r\n * All logs\r\n */\r\n public static readonly AllLogLevel = 7;\r\n\r\n /**\r\n * Message to display when a message has been logged too many times\r\n */\r\n public static MessageLimitReached = \"Too many %TYPE%s (%LIMIT%), no more %TYPE%s will be reported for this message.\";\r\n\r\n private static _LogCache = \"\";\r\n private static _LogLimitOutputs: { [message: string]: { limit: number; current: number } } = {};\r\n // levels according to the (binary) numbering.\r\n private static _Levels = [\r\n {},\r\n { color: \"white\", logFunc: console.log, name: \"Log\" },\r\n { color: \"orange\", logFunc: console.warn, name: \"Warn\" },\r\n {},\r\n { color: \"red\", logFunc: console.error, name: \"Error\" },\r\n ];\r\n\r\n /**\r\n * Gets a value indicating the number of loading errors\r\n * @ignorenaming\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public static errorsCount = 0;\r\n\r\n /**\r\n * Callback called when a new log is added\r\n */\r\n public static OnNewCacheEntry: (entry: string) => void;\r\n\r\n private static _CheckLimit(message: string, limit: number): boolean {\r\n let entry = Logger._LogLimitOutputs[message];\r\n if (!entry) {\r\n entry = { limit, current: 1 };\r\n Logger._LogLimitOutputs[message] = entry;\r\n } else {\r\n entry.current++;\r\n }\r\n return entry.current <= entry.limit;\r\n }\r\n\r\n private static _GenerateLimitMessage(message: string, level: number = 1): void {\r\n const entry = Logger._LogLimitOutputs[message];\r\n if (!entry || !Logger.MessageLimitReached) {\r\n return;\r\n }\r\n const type = this._Levels[level];\r\n if (entry.current === entry.limit) {\r\n Logger[type.name as \"Log\" | \"Warn\" | \"Error\"](Logger.MessageLimitReached.replace(/%LIMIT%/g, \"\" + entry.limit).replace(/%TYPE%/g, type.name ?? \"\"));\r\n }\r\n }\r\n\r\n private static _AddLogEntry(entry: string) {\r\n Logger._LogCache = entry + Logger._LogCache;\r\n\r\n if (Logger.OnNewCacheEntry) {\r\n Logger.OnNewCacheEntry(entry);\r\n }\r\n }\r\n\r\n private static _FormatMessage(message: string): string {\r\n const padStr = (i: number) => (i < 10 ? \"0\" + i : \"\" + i);\r\n\r\n const date = new Date();\r\n return \"[\" + padStr(date.getHours()) + \":\" + padStr(date.getMinutes()) + \":\" + padStr(date.getSeconds()) + \"]: \" + message;\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n private static _LogDisabled(message: string, limit?: number): void {\r\n // nothing to do\r\n }\r\n private static _LogEnabled(level: number = 1, message: string, limit?: number): void {\r\n if (limit !== undefined && !Logger._CheckLimit(message, limit)) {\r\n return;\r\n }\r\n\r\n const formattedMessage = Logger._FormatMessage(message);\r\n const type = this._Levels[level];\r\n type.logFunc && type.logFunc(\"BJS - \" + formattedMessage);\r\n\r\n const entry = `
${formattedMessage}

`;\r\n Logger._AddLogEntry(entry);\r\n Logger._GenerateLimitMessage(message, level);\r\n }\r\n\r\n /**\r\n * Log a message to the console\r\n */\r\n public static Log: (message: string, limit?: number) => void = Logger._LogEnabled.bind(Logger, Logger.MessageLogLevel);\r\n\r\n /**\r\n * Write a warning message to the console\r\n */\r\n public static Warn: (message: string, limit?: number) => void = Logger._LogEnabled.bind(Logger, Logger.WarningLogLevel);\r\n\r\n /**\r\n * Write an error message to the console\r\n */\r\n public static Error: (message: string, limit?: number) => void = Logger._LogEnabled.bind(Logger, Logger.ErrorLogLevel);\r\n\r\n /**\r\n * Gets current log cache (list of logs)\r\n */\r\n public static get LogCache(): string {\r\n return Logger._LogCache;\r\n }\r\n\r\n /**\r\n * Clears the log cache\r\n */\r\n public static ClearLogCache(): void {\r\n Logger._LogCache = \"\";\r\n Logger._LogLimitOutputs = {};\r\n Logger.errorsCount = 0;\r\n }\r\n\r\n /**\r\n * Sets the current log level (MessageLogLevel / WarningLogLevel / ErrorLogLevel)\r\n */\r\n public static set LogLevels(level: number) {\r\n Logger.Log = Logger._LogDisabled;\r\n Logger.Warn = Logger._LogDisabled;\r\n Logger.Error = Logger._LogDisabled;\r\n [Logger.MessageLogLevel, Logger.WarningLogLevel, Logger.ErrorLogLevel].forEach((l) => {\r\n if ((level & l) === l) {\r\n const type = this._Levels[l];\r\n Logger[type.name as \"Log\" | \"Warn\" | \"Error\"] = Logger._LogEnabled.bind(Logger, l);\r\n }\r\n });\r\n }\r\n}\r\n","import { Logger } from \"./logger\";\r\n\r\nconst CloneValue = (source: any, destinationObject: any) => {\r\n if (!source) {\r\n return null;\r\n }\r\n\r\n if (source.getClassName && source.getClassName() === \"Mesh\") {\r\n return null;\r\n }\r\n\r\n if (source.getClassName && (source.getClassName() === \"SubMesh\" || source.getClassName() === \"PhysicsBody\")) {\r\n return source.clone(destinationObject);\r\n } else if (source.clone) {\r\n return source.clone();\r\n } else if (Array.isArray(source)) {\r\n return source.slice();\r\n }\r\n return null;\r\n};\r\n\r\nfunction GetAllPropertyNames(obj: any): string[] {\r\n const props: string[] = [];\r\n\r\n do {\r\n Object.getOwnPropertyNames(obj).forEach(function (prop) {\r\n if (props.indexOf(prop) === -1) {\r\n props.push(prop);\r\n }\r\n });\r\n } while ((obj = Object.getPrototypeOf(obj)));\r\n\r\n return props;\r\n}\r\n\r\n/**\r\n * Class containing a set of static utilities functions for deep copy.\r\n */\r\nexport class DeepCopier {\r\n /**\r\n * Tries to copy an object by duplicating every property\r\n * @param source defines the source object\r\n * @param destination defines the target object\r\n * @param doNotCopyList defines a list of properties to avoid\r\n * @param mustCopyList defines a list of properties to copy (even if they start with _)\r\n */\r\n public static DeepCopy(source: any, destination: any, doNotCopyList?: string[], mustCopyList?: string[]): void {\r\n const properties = GetAllPropertyNames(source);\r\n for (const prop of properties) {\r\n if (prop[0] === \"_\" && (!mustCopyList || mustCopyList.indexOf(prop) === -1)) {\r\n continue;\r\n }\r\n\r\n if (prop.endsWith(\"Observable\")) {\r\n continue;\r\n }\r\n\r\n if (doNotCopyList && doNotCopyList.indexOf(prop) !== -1) {\r\n continue;\r\n }\r\n\r\n const sourceValue = source[prop];\r\n const typeOfSourceValue = typeof sourceValue;\r\n\r\n if (typeOfSourceValue === \"function\") {\r\n continue;\r\n }\r\n\r\n try {\r\n if (typeOfSourceValue === \"object\") {\r\n if (sourceValue instanceof Uint8Array) {\r\n destination[prop] = Uint8Array.from(sourceValue);\r\n } else if (sourceValue instanceof Array) {\r\n destination[prop] = [];\r\n\r\n if (sourceValue.length > 0) {\r\n if (typeof sourceValue[0] == \"object\") {\r\n for (let index = 0; index < sourceValue.length; index++) {\r\n const clonedValue = CloneValue(sourceValue[index], destination);\r\n\r\n if (destination[prop].indexOf(clonedValue) === -1) {\r\n // Test if auto inject was not done\r\n destination[prop].push(clonedValue);\r\n }\r\n }\r\n } else {\r\n destination[prop] = sourceValue.slice(0);\r\n }\r\n }\r\n } else {\r\n destination[prop] = CloneValue(sourceValue, destination);\r\n }\r\n } else {\r\n destination[prop] = sourceValue;\r\n }\r\n } catch (e) {\r\n // Log a warning (it could be because of a read-only property)\r\n Logger.Warn(e.message);\r\n }\r\n }\r\n }\r\n}\r\n","import { IsWindowObjectExist } from \"./domManagement\";\r\n\r\n/**\r\n * Class containing a set of static utilities functions for precision date\r\n */\r\nexport class PrecisionDate {\r\n /**\r\n * Gets either window.performance.now() if supported or Date.now() else\r\n */\r\n public static get Now(): number {\r\n if (IsWindowObjectExist() && window.performance && window.performance.now) {\r\n return window.performance.now();\r\n }\r\n\r\n return Date.now();\r\n }\r\n}\r\n","import type { IWebRequest } from \"./interfaces/iWebRequest\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { INative } from \"../Engines/Native/nativeInterfaces\";\r\n\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\ndeclare const _native: INative;\r\n\r\n/** @internal */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nfunction createXMLHttpRequest(): XMLHttpRequest {\r\n // If running in Babylon Native, then defer to the native XMLHttpRequest, which has the same public contract\r\n if (typeof _native !== \"undefined\" && _native.XMLHttpRequest) {\r\n return new _native.XMLHttpRequest();\r\n } else {\r\n return new XMLHttpRequest();\r\n }\r\n}\r\n\r\n/**\r\n * Extended version of XMLHttpRequest with support for customizations (headers, ...)\r\n */\r\nexport class WebRequest implements IWebRequest {\r\n private readonly _xhr = createXMLHttpRequest();\r\n\r\n /**\r\n * Custom HTTP Request Headers to be sent with XMLHttpRequests\r\n * i.e. when loading files, where the server/service expects an Authorization header\r\n */\r\n public static CustomRequestHeaders: { [key: string]: string } = {};\r\n\r\n /**\r\n * Add callback functions in this array to update all the requests before they get sent to the network\r\n */\r\n public static CustomRequestModifiers = new Array<(request: XMLHttpRequest, url: string) => void>();\r\n\r\n public static SkipRequestModificationForBabylonCDN = true;\r\n\r\n private _requestURL: string = \"\";\r\n\r\n private _injectCustomRequestHeaders(): void {\r\n if (this._shouldSkipRequestModifications(this._requestURL)) {\r\n return;\r\n }\r\n for (const key in WebRequest.CustomRequestHeaders) {\r\n const val = WebRequest.CustomRequestHeaders[key];\r\n if (val) {\r\n this._xhr.setRequestHeader(key, val);\r\n }\r\n }\r\n }\r\n\r\n private _shouldSkipRequestModifications(url: string): boolean {\r\n return WebRequest.SkipRequestModificationForBabylonCDN && (url.includes(\"preview.babylonjs.com\") || url.includes(\"cdn.babylonjs.com\"));\r\n }\r\n\r\n /**\r\n * Gets or sets a function to be called when loading progress changes\r\n */\r\n public get onprogress(): ((this: XMLHttpRequest, ev: ProgressEvent) => any) | null {\r\n return this._xhr.onprogress;\r\n }\r\n\r\n public set onprogress(value: ((this: XMLHttpRequest, ev: ProgressEvent) => any) | null) {\r\n this._xhr.onprogress = value;\r\n }\r\n\r\n /**\r\n * Returns client's state\r\n */\r\n public get readyState(): number {\r\n return this._xhr.readyState;\r\n }\r\n\r\n /**\r\n * Returns client's status\r\n */\r\n public get status(): number {\r\n return this._xhr.status;\r\n }\r\n\r\n /**\r\n * Returns client's status as a text\r\n */\r\n public get statusText(): string {\r\n return this._xhr.statusText;\r\n }\r\n\r\n /**\r\n * Returns client's response\r\n */\r\n public get response(): any {\r\n return this._xhr.response;\r\n }\r\n\r\n /**\r\n * Returns client's response url\r\n */\r\n public get responseURL(): string {\r\n return this._xhr.responseURL;\r\n }\r\n\r\n /**\r\n * Returns client's response as text\r\n */\r\n public get responseText(): string {\r\n return this._xhr.responseText;\r\n }\r\n\r\n /**\r\n * Gets or sets the expected response type\r\n */\r\n public get responseType(): XMLHttpRequestResponseType {\r\n return this._xhr.responseType;\r\n }\r\n\r\n public set responseType(value: XMLHttpRequestResponseType) {\r\n this._xhr.responseType = value;\r\n }\r\n\r\n /**\r\n * Gets or sets the timeout value in milliseconds\r\n */\r\n public get timeout(): number {\r\n return this._xhr.timeout;\r\n }\r\n\r\n public set timeout(value: number) {\r\n this._xhr.timeout = value;\r\n }\r\n\r\n /** @internal */\r\n public addEventListener(\r\n type: K,\r\n listener: (this: XMLHttpRequest, ev: XMLHttpRequestEventMap[K]) => any,\r\n options?: boolean | AddEventListenerOptions\r\n ): void;\r\n public addEventListener(type: string, listener: EventListenerOrEventListenerObject, options?: boolean | AddEventListenerOptions): void {\r\n this._xhr.addEventListener(type, listener, options);\r\n }\r\n\r\n /** @internal */\r\n public removeEventListener(\r\n type: K,\r\n listener: (this: XMLHttpRequest, ev: XMLHttpRequestEventMap[K]) => any,\r\n options?: boolean | EventListenerOptions\r\n ): void;\r\n public removeEventListener(type: string, listener: EventListenerOrEventListenerObject, options?: boolean | EventListenerOptions): void {\r\n this._xhr.removeEventListener(type, listener, options);\r\n }\r\n\r\n /**\r\n * Cancels any network activity\r\n */\r\n public abort() {\r\n this._xhr.abort();\r\n }\r\n\r\n /**\r\n * Initiates the request. The optional argument provides the request body. The argument is ignored if request method is GET or HEAD\r\n * @param body defines an optional request body\r\n */\r\n public send(body?: Document | XMLHttpRequestBodyInit | null): void {\r\n if (WebRequest.CustomRequestHeaders) {\r\n this._injectCustomRequestHeaders();\r\n }\r\n\r\n this._xhr.send(body);\r\n }\r\n\r\n /**\r\n * Sets the request method, request URL\r\n * @param method defines the method to use (GET, POST, etc..)\r\n * @param url defines the url to connect with\r\n */\r\n public open(method: string, url: string): void {\r\n for (const update of WebRequest.CustomRequestModifiers) {\r\n if (this._shouldSkipRequestModifications(url)) {\r\n return;\r\n }\r\n update(this._xhr, url);\r\n }\r\n\r\n // Clean url\r\n url = url.replace(\"file:http:\", \"http:\");\r\n url = url.replace(\"file:https:\", \"https:\");\r\n\r\n this._requestURL = url;\r\n\r\n return this._xhr.open(method, url, true);\r\n }\r\n\r\n /**\r\n * Sets the value of a request header.\r\n * @param name The name of the header whose value is to be set\r\n * @param value The value to set as the body of the header\r\n */\r\n setRequestHeader(name: string, value: string): void {\r\n this._xhr.setRequestHeader(name, value);\r\n }\r\n\r\n /**\r\n * Get the string containing the text of a particular header's value.\r\n * @param name The name of the header\r\n * @returns The string containing the text of the given header name\r\n */\r\n getResponseHeader(name: string): Nullable {\r\n return this._xhr.getResponseHeader(name);\r\n }\r\n}\r\n","/**\r\n * Class used to help managing file picking and drag'n'drop\r\n * File Storage\r\n */\r\nexport class FilesInputStore {\r\n /**\r\n * List of files ready to be loaded\r\n */\r\n public static FilesToLoad: { [key: string]: File } = {};\r\n}\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\n/**\r\n * Base error. Due to limitations of typedoc-check and missing documentation\r\n * in lib.es5.d.ts, cannot extend Error directly for RuntimeError.\r\n * @ignore\r\n */\r\nexport abstract class BaseError extends Error {\r\n // See https://stackoverflow.com/questions/12915412/how-do-i-extend-a-host-object-e-g-error-in-typescript\r\n // and https://github.com/Microsoft/TypeScript/wiki/Breaking-Changes#extending-built-ins-like-error-array-and-map-may-no-longer-work\r\n\r\n // Polyfill for Object.setPrototypeOf if necessary.\r\n protected static _setPrototypeOf: (o: any, proto: object | null) => any =\r\n (Object as any).setPrototypeOf ||\r\n ((o, proto) => {\r\n o.__proto__ = proto;\r\n return o;\r\n });\r\n}\r\n\r\n/* IMP! DO NOT CHANGE THE NUMBERING OF EXISTING ERROR CODES */\r\n/**\r\n * Error codes for BaseError\r\n */\r\nexport const ErrorCodes = {\r\n // Mesh errors 0-999\r\n /** Invalid or empty mesh vertex positions. */\r\n MeshInvalidPositionsError: 0,\r\n\r\n // Texture errors 1000-1999\r\n /** Unsupported texture found. */\r\n UnsupportedTextureError: 1000,\r\n\r\n // GLTFLoader errors 2000-2999\r\n /** Unexpected magic number found in GLTF file header. */\r\n GLTFLoaderUnexpectedMagicError: 2000,\r\n\r\n // SceneLoader errors 3000-3999\r\n /** SceneLoader generic error code. Ideally wraps the inner exception. */\r\n SceneLoaderError: 3000,\r\n\r\n // File related errors 4000-4999\r\n /** Load file error */\r\n LoadFileError: 4000,\r\n /** Request file error */\r\n RequestFileError: 4001,\r\n /** Read file error */\r\n ReadFileError: 4002,\r\n} as const;\r\n\r\n/**\r\n * Error code type\r\n */\r\nexport type ErrorCodesType = (typeof ErrorCodes)[keyof typeof ErrorCodes];\r\n\r\n/**\r\n * Application runtime error\r\n */\r\nexport class RuntimeError extends BaseError {\r\n /**\r\n * The error code\r\n */\r\n public errorCode: ErrorCodesType;\r\n\r\n /**\r\n * The error that caused this outer error\r\n */\r\n public innerError?: Error;\r\n\r\n /**\r\n * Creates a new RuntimeError\r\n * @param message defines the message of the error\r\n * @param errorCode the error code\r\n * @param innerError the error that caused the outer error\r\n */\r\n public constructor(message: string, errorCode: ErrorCodesType, innerError?: Error) {\r\n super(message);\r\n\r\n this.errorCode = errorCode;\r\n this.innerError = innerError;\r\n\r\n this.name = \"RuntimeError\";\r\n BaseError._setPrototypeOf(this, RuntimeError.prototype);\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\n\r\n/**\r\n * Checks for a matching suffix at the end of a string (for ES5 and lower)\r\n * @param str Source string\r\n * @param suffix Suffix to search for in the source string\r\n * @returns Boolean indicating whether the suffix was found (true) or not (false)\r\n * @deprecated Please use native string function instead\r\n */\r\nexport const EndsWith = (str: string, suffix: string): boolean => {\r\n return str.endsWith(suffix);\r\n};\r\n\r\n/**\r\n * Checks for a matching suffix at the beginning of a string (for ES5 and lower)\r\n * @param str Source string\r\n * @param suffix Suffix to search for in the source string\r\n * @returns Boolean indicating whether the suffix was found (true) or not (false)\r\n * @deprecated Please use native string function instead\r\n */\r\nexport const StartsWith = (str: string, suffix: string): boolean => {\r\n if (!str) {\r\n return false;\r\n }\r\n return str.startsWith(suffix);\r\n};\r\n\r\n/**\r\n * Decodes a buffer into a string\r\n * @param buffer The buffer to decode\r\n * @returns The decoded string\r\n */\r\nexport const Decode = (buffer: Uint8Array | Uint16Array): string => {\r\n if (typeof TextDecoder !== \"undefined\") {\r\n return new TextDecoder().decode(buffer);\r\n }\r\n\r\n let result = \"\";\r\n for (let i = 0; i < buffer.byteLength; i++) {\r\n result += String.fromCharCode(buffer[i]);\r\n }\r\n\r\n return result;\r\n};\r\n\r\n/**\r\n * Encode a buffer to a base64 string\r\n * @param buffer defines the buffer to encode\r\n * @returns the encoded string\r\n */\r\nexport const EncodeArrayBufferToBase64 = (buffer: ArrayBuffer | ArrayBufferView): string => {\r\n const keyStr = \"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=\";\r\n let output = \"\";\r\n let chr1, chr2, chr3, enc1, enc2, enc3, enc4;\r\n let i = 0;\r\n const bytes = ArrayBuffer.isView(buffer) ? new Uint8Array(buffer.buffer, buffer.byteOffset, buffer.byteLength) : new Uint8Array(buffer);\r\n\r\n while (i < bytes.length) {\r\n chr1 = bytes[i++];\r\n chr2 = i < bytes.length ? bytes[i++] : Number.NaN;\r\n chr3 = i < bytes.length ? bytes[i++] : Number.NaN;\r\n\r\n enc1 = chr1 >> 2;\r\n enc2 = ((chr1 & 3) << 4) | (chr2 >> 4);\r\n enc3 = ((chr2 & 15) << 2) | (chr3 >> 6);\r\n enc4 = chr3 & 63;\r\n\r\n if (isNaN(chr2)) {\r\n enc3 = enc4 = 64;\r\n } else if (isNaN(chr3)) {\r\n enc4 = 64;\r\n }\r\n output += keyStr.charAt(enc1) + keyStr.charAt(enc2) + keyStr.charAt(enc3) + keyStr.charAt(enc4);\r\n }\r\n\r\n return output;\r\n};\r\n\r\n/**\r\n * Converts a given base64 string as an ASCII encoded stream of data\r\n * @param base64Data The base64 encoded string to decode\r\n * @returns Decoded ASCII string\r\n */\r\nexport const DecodeBase64ToString = (base64Data: string): string => {\r\n return atob(base64Data);\r\n};\r\n\r\n/**\r\n * Converts a given base64 string into an ArrayBuffer of raw byte data\r\n * @param base64Data The base64 encoded string to decode\r\n * @returns ArrayBuffer of byte data\r\n */\r\nexport const DecodeBase64ToBinary = (base64Data: string): ArrayBuffer => {\r\n const decodedString = DecodeBase64ToString(base64Data);\r\n const bufferLength = decodedString.length;\r\n const bufferView = new Uint8Array(new ArrayBuffer(bufferLength));\r\n\r\n for (let i = 0; i < bufferLength; i++) {\r\n bufferView[i] = decodedString.charCodeAt(i);\r\n }\r\n\r\n return bufferView.buffer;\r\n};\r\n\r\n/**\r\n * Converts a number to string and pads with preceding zeroes until it is of specified length.\r\n * @param num the number to convert and pad\r\n * @param length the expected length of the string\r\n * @returns the padded string\r\n */\r\nexport const PadNumber = (num: number, length: number): string => {\r\n let str = String(num);\r\n while (str.length < length) {\r\n str = \"0\" + str;\r\n }\r\n return str;\r\n};\r\n/**\r\n * Helper to manipulate strings\r\n */\r\nexport const StringTools = {\r\n EndsWith,\r\n StartsWith,\r\n Decode,\r\n EncodeArrayBufferToBase64,\r\n DecodeBase64ToString,\r\n DecodeBase64ToBinary,\r\n PadNumber,\r\n};\r\n","import type { ProcessingOptions } from \"./shaderProcessingOptions\";\r\n\r\nconst defaultAttributeKeywordName = \"attribute\";\r\nconst defaultVaryingKeywordName = \"varying\";\r\n\r\n/** @internal */\r\nexport class ShaderCodeNode {\r\n line: string;\r\n children: ShaderCodeNode[] = [];\r\n additionalDefineKey?: string;\r\n additionalDefineValue?: string;\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n isValid(preprocessors: { [key: string]: string }): boolean {\r\n return true;\r\n }\r\n\r\n process(preprocessors: { [key: string]: string }, options: ProcessingOptions): string {\r\n let result = \"\";\r\n if (this.line) {\r\n let value: string = this.line;\r\n const processor = options.processor;\r\n if (processor) {\r\n // This must be done before other replacements to avoid mistakenly changing something that was already changed.\r\n if (processor.lineProcessor) {\r\n value = processor.lineProcessor(value, options.isFragment, options.processingContext);\r\n }\r\n\r\n const attributeKeyword = options.processor?.attributeKeywordName ?? defaultAttributeKeywordName;\r\n const varyingKeyword =\r\n options.isFragment && options.processor?.varyingFragmentKeywordName\r\n ? options.processor?.varyingFragmentKeywordName\r\n : !options.isFragment && options.processor?.varyingVertexKeywordName\r\n ? options.processor?.varyingVertexKeywordName\r\n : defaultVaryingKeywordName;\r\n\r\n if (!options.isFragment && processor.attributeProcessor && this.line.startsWith(attributeKeyword)) {\r\n value = processor.attributeProcessor(this.line, preprocessors, options.processingContext);\r\n } else if (\r\n processor.varyingProcessor &&\r\n (processor.varyingCheck?.(this.line, options.isFragment) || (!processor.varyingCheck && this.line.startsWith(varyingKeyword)))\r\n ) {\r\n value = processor.varyingProcessor(this.line, options.isFragment, preprocessors, options.processingContext);\r\n } else if (processor.uniformProcessor && processor.uniformRegexp && processor.uniformRegexp.test(this.line)) {\r\n if (!options.lookForClosingBracketForUniformBuffer) {\r\n value = processor.uniformProcessor(this.line, options.isFragment, preprocessors, options.processingContext);\r\n }\r\n } else if (processor.uniformBufferProcessor && processor.uniformBufferRegexp && processor.uniformBufferRegexp.test(this.line)) {\r\n if (!options.lookForClosingBracketForUniformBuffer) {\r\n value = processor.uniformBufferProcessor(this.line, options.isFragment, options.processingContext);\r\n options.lookForClosingBracketForUniformBuffer = true;\r\n }\r\n } else if (processor.textureProcessor && processor.textureRegexp && processor.textureRegexp.test(this.line)) {\r\n value = processor.textureProcessor(this.line, options.isFragment, preprocessors, options.processingContext);\r\n } else if ((processor.uniformProcessor || processor.uniformBufferProcessor) && this.line.startsWith(\"uniform\") && !options.lookForClosingBracketForUniformBuffer) {\r\n const regex = /uniform\\s+(?:(?:highp)?|(?:lowp)?)\\s*(\\S+)\\s+(\\S+)\\s*;/;\r\n\r\n if (regex.test(this.line)) {\r\n // uniform\r\n if (processor.uniformProcessor) {\r\n value = processor.uniformProcessor(this.line, options.isFragment, preprocessors, options.processingContext);\r\n }\r\n } else {\r\n // Uniform buffer\r\n if (processor.uniformBufferProcessor) {\r\n value = processor.uniformBufferProcessor(this.line, options.isFragment, options.processingContext);\r\n options.lookForClosingBracketForUniformBuffer = true;\r\n }\r\n }\r\n }\r\n\r\n if (options.lookForClosingBracketForUniformBuffer && this.line.indexOf(\"}\") !== -1) {\r\n options.lookForClosingBracketForUniformBuffer = false;\r\n if (processor.endOfUniformBufferProcessor) {\r\n value = processor.endOfUniformBufferProcessor(this.line, options.isFragment, options.processingContext);\r\n }\r\n }\r\n }\r\n\r\n result += value + \"\\n\";\r\n }\r\n\r\n this.children.forEach((child) => {\r\n result += child.process(preprocessors, options);\r\n });\r\n\r\n if (this.additionalDefineKey) {\r\n preprocessors[this.additionalDefineKey] = this.additionalDefineValue || \"true\";\r\n }\r\n\r\n return result;\r\n }\r\n}\r\n","/** @internal */\r\nexport class ShaderCodeCursor {\r\n private _lines: string[] = [];\r\n lineIndex: number;\r\n\r\n get currentLine(): string {\r\n return this._lines[this.lineIndex];\r\n }\r\n\r\n get canRead(): boolean {\r\n return this.lineIndex < this._lines.length - 1;\r\n }\r\n\r\n set lines(value: string[]) {\r\n this._lines.length = 0;\r\n\r\n for (const line of value) {\r\n // Skip empty lines\r\n if (!line || line === \"\\r\") {\r\n continue;\r\n }\r\n\r\n // Prevent removing line break in macros.\r\n if (line[0] === \"#\") {\r\n this._lines.push(line);\r\n continue;\r\n }\r\n\r\n // Do not split single line comments\r\n const trimmedLine = line.trim();\r\n\r\n if (!trimmedLine) {\r\n continue;\r\n }\r\n\r\n if (trimmedLine.startsWith(\"//\")) {\r\n this._lines.push(line);\r\n continue;\r\n }\r\n\r\n // Work with semicolon in the line\r\n const semicolonIndex = trimmedLine.indexOf(\";\");\r\n\r\n if (semicolonIndex === -1) {\r\n // No semicolon in the line\r\n this._lines.push(trimmedLine);\r\n } else if (semicolonIndex === trimmedLine.length - 1) {\r\n // Single semicolon at the end of the line\r\n // If trimmedLine == \";\", we must not push, to be backward compatible with the old code!\r\n if (trimmedLine.length > 1) {\r\n this._lines.push(trimmedLine);\r\n }\r\n } else {\r\n // Semicolon in the middle of the line\r\n const split = line.split(\";\");\r\n\r\n for (let index = 0; index < split.length; index++) {\r\n let subLine = split[index];\r\n\r\n if (!subLine) {\r\n continue;\r\n }\r\n\r\n subLine = subLine.trim();\r\n\r\n if (!subLine) {\r\n continue;\r\n }\r\n\r\n this._lines.push(subLine + (index !== split.length - 1 ? \";\" : \"\"));\r\n }\r\n }\r\n }\r\n }\r\n}\r\n","import { ShaderCodeNode } from \"./shaderCodeNode\";\r\nimport type { ProcessingOptions } from \"./shaderProcessingOptions\";\r\n\r\n/** @internal */\r\nexport class ShaderCodeConditionNode extends ShaderCodeNode {\r\n process(preprocessors: { [key: string]: string }, options: ProcessingOptions) {\r\n for (let index = 0; index < this.children.length; index++) {\r\n const node = this.children[index];\r\n\r\n if (node.isValid(preprocessors)) {\r\n return node.process(preprocessors, options);\r\n }\r\n }\r\n\r\n return \"\";\r\n }\r\n}\r\n","import { ShaderCodeNode } from \"./shaderCodeNode\";\r\nimport type { ShaderDefineExpression } from \"./Expressions/shaderDefineExpression\";\r\n\r\n/** @internal */\r\nexport class ShaderCodeTestNode extends ShaderCodeNode {\r\n public testExpression: ShaderDefineExpression;\r\n\r\n public isValid(preprocessors: { [key: string]: string }) {\r\n return this.testExpression.isTrue(preprocessors);\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\n/** @internal */\r\nexport class ShaderDefineExpression {\r\n /**\r\n * Cache items count limit for the InfixToPostfix cache.\r\n * It uses to improve the performance of the shader compilation.\r\n * For details see PR: https://github.com/BabylonJS/Babylon.js/pull/13936\r\n */\r\n static InfixToPostfixCacheLimitSize = 50000;\r\n\r\n /**\r\n * When the cache size is exceeded, a cache cleanup will be triggered\r\n * and the cache will be reduced by the size specified\r\n * in the InfixToPostfixCacheCleanupSize variable, removing entries\r\n * that have not been accessed the longest.\r\n */\r\n static InfixToPostfixCacheCleanupSize = 25000;\r\n\r\n protected static _InfixToPostfixCache: Map<\r\n string,\r\n {\r\n accessTime: number;\r\n result: string[];\r\n }\r\n > = new Map();\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public isTrue(preprocessors: { [key: string]: string }): boolean {\r\n return true;\r\n }\r\n\r\n private static _OperatorPriority: { [name: string]: number } = {\r\n \")\": 0,\r\n \"(\": 1,\r\n \"||\": 2,\r\n \"&&\": 3,\r\n };\r\n\r\n private static _Stack = [\"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\", \"\"];\r\n\r\n public static postfixToInfix(postfix: string[]): string {\r\n const stack: string[] = [];\r\n\r\n for (const c of postfix) {\r\n if (ShaderDefineExpression._OperatorPriority[c] === undefined) {\r\n stack.push(c);\r\n } else {\r\n const v1 = stack[stack.length - 1],\r\n v2 = stack[stack.length - 2];\r\n\r\n stack.length -= 2;\r\n stack.push(`(${v2}${c}${v1})`);\r\n }\r\n }\r\n\r\n return stack[stack.length - 1];\r\n }\r\n\r\n /**\r\n * Converts an infix expression to a postfix expression.\r\n *\r\n * This method is used to transform infix expressions, which are more human-readable,\r\n * into postfix expressions, also known as Reverse Polish Notation (RPN), that can be\r\n * evaluated more efficiently by a computer. The conversion is based on the operator\r\n * priority defined in _OperatorPriority.\r\n *\r\n * The function employs a stack-based algorithm for the conversion and caches the result\r\n * to improve performance. The cache keeps track of each converted expression's access time\r\n * to manage the cache size and optimize memory usage. When the cache size exceeds a specified\r\n * limit, the least recently accessed items in the cache are deleted.\r\n *\r\n * The cache mechanism is particularly helpful for shader compilation, where the same infix\r\n * expressions might be encountered repeatedly, hence the caching can speed up the process.\r\n *\r\n * @param infix - The infix expression to be converted.\r\n * @returns The postfix expression as an array of strings.\r\n */\r\n public static infixToPostfix(infix: string): string[] {\r\n // Is infix already in cache\r\n const cacheItem = ShaderDefineExpression._InfixToPostfixCache.get(infix);\r\n if (cacheItem) {\r\n cacheItem.accessTime = Date.now();\r\n return cacheItem.result;\r\n }\r\n\r\n // Is infix contain any operator\r\n if (!infix.includes(\"&&\") && !infix.includes(\"||\") && !infix.includes(\")\") && !infix.includes(\"(\")) {\r\n return [infix];\r\n }\r\n\r\n const result: string[] = [];\r\n\r\n let stackIdx = -1;\r\n\r\n const pushOperand = () => {\r\n operand = operand.trim();\r\n if (operand !== \"\") {\r\n result.push(operand);\r\n operand = \"\";\r\n }\r\n };\r\n\r\n const push = (s: string) => {\r\n if (stackIdx < ShaderDefineExpression._Stack.length - 1) {\r\n ShaderDefineExpression._Stack[++stackIdx] = s;\r\n }\r\n };\r\n\r\n const peek = () => ShaderDefineExpression._Stack[stackIdx];\r\n\r\n const pop = () => (stackIdx === -1 ? \"!!INVALID EXPRESSION!!\" : ShaderDefineExpression._Stack[stackIdx--]);\r\n\r\n let idx = 0,\r\n operand = \"\";\r\n\r\n while (idx < infix.length) {\r\n const c = infix.charAt(idx),\r\n token = idx < infix.length - 1 ? infix.substr(idx, 2) : \"\";\r\n\r\n if (c === \"(\") {\r\n operand = \"\";\r\n push(c);\r\n } else if (c === \")\") {\r\n pushOperand();\r\n while (stackIdx !== -1 && peek() !== \"(\") {\r\n result.push(pop());\r\n }\r\n pop();\r\n } else if (ShaderDefineExpression._OperatorPriority[token] > 1) {\r\n pushOperand();\r\n while (stackIdx !== -1 && ShaderDefineExpression._OperatorPriority[peek()] >= ShaderDefineExpression._OperatorPriority[token]) {\r\n result.push(pop());\r\n }\r\n push(token);\r\n idx++;\r\n } else {\r\n operand += c;\r\n }\r\n idx++;\r\n }\r\n\r\n pushOperand();\r\n\r\n while (stackIdx !== -1) {\r\n if (peek() === \"(\") {\r\n pop();\r\n } else {\r\n result.push(pop());\r\n }\r\n }\r\n\r\n // If the cache is at capacity, clear it before adding a new item\r\n if (ShaderDefineExpression._InfixToPostfixCache.size >= ShaderDefineExpression.InfixToPostfixCacheLimitSize) {\r\n ShaderDefineExpression.ClearCache();\r\n }\r\n\r\n // Add the new item to the cache, including the current time as the last access time\r\n ShaderDefineExpression._InfixToPostfixCache.set(infix, { result, accessTime: Date.now() });\r\n\r\n return result;\r\n }\r\n\r\n private static ClearCache(): void {\r\n // Convert the cache to an array and sort by last access time\r\n const sortedCache = Array.from(ShaderDefineExpression._InfixToPostfixCache.entries()).sort((a, b) => a[1].accessTime - b[1].accessTime);\r\n\r\n // Remove the least recently accessed half of the cache\r\n for (let i = 0; i < ShaderDefineExpression.InfixToPostfixCacheCleanupSize; i++) {\r\n ShaderDefineExpression._InfixToPostfixCache.delete(sortedCache[i][0]);\r\n }\r\n }\r\n}\r\n","import { ShaderDefineExpression } from \"../shaderDefineExpression\";\r\n\r\n/** @internal */\r\nexport class ShaderDefineIsDefinedOperator extends ShaderDefineExpression {\r\n public constructor(public define: string, public not: boolean = false) {\r\n super();\r\n }\r\n\r\n public isTrue(preprocessors: { [key: string]: string }) {\r\n let condition = preprocessors[this.define] !== undefined;\r\n\r\n if (this.not) {\r\n condition = !condition;\r\n }\r\n\r\n return condition;\r\n }\r\n}\r\n","import { ShaderDefineExpression } from \"../shaderDefineExpression\";\r\n\r\n/** @internal */\r\nexport class ShaderDefineOrOperator extends ShaderDefineExpression {\r\n public leftOperand: ShaderDefineExpression;\r\n public rightOperand: ShaderDefineExpression;\r\n\r\n public isTrue(preprocessors: { [key: string]: string }): boolean {\r\n return this.leftOperand.isTrue(preprocessors) || this.rightOperand.isTrue(preprocessors);\r\n }\r\n}\r\n","import { ShaderDefineExpression } from \"../shaderDefineExpression\";\r\n\r\n/** @internal */\r\nexport class ShaderDefineAndOperator extends ShaderDefineExpression {\r\n public leftOperand: ShaderDefineExpression;\r\n public rightOperand: ShaderDefineExpression;\r\n\r\n public isTrue(preprocessors: { [key: string]: string }): boolean {\r\n return this.leftOperand.isTrue(preprocessors) && this.rightOperand.isTrue(preprocessors);\r\n }\r\n}\r\n","import { ShaderDefineExpression } from \"../shaderDefineExpression\";\r\n\r\n/** @internal */\r\nexport class ShaderDefineArithmeticOperator extends ShaderDefineExpression {\r\n public constructor(public define: string, public operand: string, public testValue: string) {\r\n super();\r\n }\r\n\r\n public isTrue(preprocessors: { [key: string]: string }) {\r\n let value = preprocessors[this.define];\r\n\r\n if (value === undefined) {\r\n value = this.define;\r\n }\r\n\r\n let condition = false;\r\n const left = parseInt(value);\r\n const right = parseInt(this.testValue);\r\n\r\n switch (this.operand) {\r\n case \">\":\r\n condition = left > right;\r\n break;\r\n case \"<\":\r\n condition = left < right;\r\n break;\r\n case \"<=\":\r\n condition = left <= right;\r\n break;\r\n case \">=\":\r\n condition = left >= right;\r\n break;\r\n case \"==\":\r\n condition = left === right;\r\n break;\r\n case \"!=\":\r\n condition = left !== right;\r\n break;\r\n }\r\n\r\n return condition;\r\n }\r\n}\r\n","/**\r\n * Language of the shader code\r\n */\r\nexport enum ShaderLanguage {\r\n /** language is GLSL (used by WebGL) */\r\n GLSL,\r\n /** language is WGSL (used by WebGPU) */\r\n WGSL,\r\n}\r\n","/* eslint-disable @typescript-eslint/no-unused-vars */\r\nimport { ShaderCodeNode } from \"./shaderCodeNode\";\r\nimport { ShaderCodeCursor } from \"./shaderCodeCursor\";\r\nimport { ShaderCodeConditionNode } from \"./shaderCodeConditionNode\";\r\nimport { ShaderCodeTestNode } from \"./shaderCodeTestNode\";\r\nimport { ShaderDefineIsDefinedOperator } from \"./Expressions/Operators/shaderDefineIsDefinedOperator\";\r\nimport { ShaderDefineOrOperator } from \"./Expressions/Operators/shaderDefineOrOperator\";\r\nimport { ShaderDefineAndOperator } from \"./Expressions/Operators/shaderDefineAndOperator\";\r\nimport { ShaderDefineExpression } from \"./Expressions/shaderDefineExpression\";\r\nimport { ShaderDefineArithmeticOperator } from \"./Expressions/Operators/shaderDefineArithmeticOperator\";\r\nimport type { ProcessingOptions } from \"./shaderProcessingOptions\";\r\nimport { _WarnImport } from \"../../Misc/devTools\";\r\nimport { ShaderLanguage } from \"../../Materials/shaderLanguage\";\r\n\r\nimport type { WebRequest } from \"../../Misc/webRequest\";\r\nimport type { LoadFileError } from \"../../Misc/fileTools\";\r\nimport type { IOfflineProvider } from \"../../Offline/IOfflineProvider\";\r\nimport type { IFileRequest } from \"../../Misc/fileRequest\";\r\nimport type { ThinEngine } from \"../thinEngine\";\r\n\r\nconst regexSE = /defined\\s*?\\((.+?)\\)/g;\r\nconst regexSERevert = /defined\\s*?\\[(.+?)\\]/g;\r\nconst regexShaderInclude = /#include\\s?<(.+)>(\\((.*)\\))*(\\[(.*)\\])*/g;\r\nconst regexShaderDecl = /__decl__/;\r\nconst regexLightX = /light\\{X\\}.(\\w*)/g;\r\nconst regexX = /\\{X\\}/g;\r\nconst reusableMatches: RegExpMatchArray[] = [];\r\n\r\n/** @internal */\r\nexport class ShaderProcessor {\r\n private static _MoveCursorRegex = /(#ifdef)|(#else)|(#elif)|(#endif)|(#ifndef)|(#if)/;\r\n\r\n public static Initialize(options: ProcessingOptions): void {\r\n if (options.processor && options.processor.initializeShaders) {\r\n options.processor.initializeShaders(options.processingContext);\r\n }\r\n }\r\n\r\n public static Process(sourceCode: string, options: ProcessingOptions, callback: (migratedCode: string, codeBeforeMigration: string) => void, engine: ThinEngine) {\r\n if (options.processor?.preProcessShaderCode) {\r\n sourceCode = options.processor.preProcessShaderCode(sourceCode, options.isFragment);\r\n }\r\n this._ProcessIncludes(sourceCode, options, (codeWithIncludes) => {\r\n if (options.processCodeAfterIncludes) {\r\n codeWithIncludes = options.processCodeAfterIncludes(options.isFragment ? \"fragment\" : \"vertex\", codeWithIncludes);\r\n }\r\n const migratedCode = this._ProcessShaderConversion(codeWithIncludes, options, engine);\r\n callback(migratedCode, codeWithIncludes);\r\n });\r\n }\r\n\r\n public static PreProcess(sourceCode: string, options: ProcessingOptions, callback: (migratedCode: string, codeBeforeMigration: string) => void, engine: ThinEngine) {\r\n if (options.processor?.preProcessShaderCode) {\r\n sourceCode = options.processor.preProcessShaderCode(sourceCode, options.isFragment);\r\n }\r\n this._ProcessIncludes(sourceCode, options, (codeWithIncludes) => {\r\n if (options.processCodeAfterIncludes) {\r\n codeWithIncludes = options.processCodeAfterIncludes(options.isFragment ? \"fragment\" : \"vertex\", codeWithIncludes);\r\n }\r\n const migratedCode = this._ApplyPreProcessing(codeWithIncludes, options, engine);\r\n callback(migratedCode, codeWithIncludes);\r\n });\r\n }\r\n\r\n public static Finalize(vertexCode: string, fragmentCode: string, options: ProcessingOptions): { vertexCode: string; fragmentCode: string } {\r\n if (!options.processor || !options.processor.finalizeShaders) {\r\n return { vertexCode, fragmentCode };\r\n }\r\n\r\n return options.processor.finalizeShaders(vertexCode, fragmentCode, options.processingContext);\r\n }\r\n\r\n private static _ProcessPrecision(source: string, options: ProcessingOptions): string {\r\n if (options.processor?.noPrecision) {\r\n return source;\r\n }\r\n\r\n const shouldUseHighPrecisionShader = options.shouldUseHighPrecisionShader;\r\n\r\n if (source.indexOf(\"precision highp float\") === -1) {\r\n if (!shouldUseHighPrecisionShader) {\r\n source = \"precision mediump float;\\n\" + source;\r\n } else {\r\n source = \"precision highp float;\\n\" + source;\r\n }\r\n } else {\r\n if (!shouldUseHighPrecisionShader) {\r\n // Moving highp to mediump\r\n source = source.replace(\"precision highp float\", \"precision mediump float\");\r\n }\r\n }\r\n\r\n return source;\r\n }\r\n\r\n private static _ExtractOperation(expression: string) {\r\n const regex = /defined\\((.+)\\)/;\r\n\r\n const match = regex.exec(expression);\r\n\r\n if (match && match.length) {\r\n return new ShaderDefineIsDefinedOperator(match[1].trim(), expression[0] === \"!\");\r\n }\r\n\r\n const operators = [\"==\", \"!=\", \">=\", \"<=\", \"<\", \">\"];\r\n let operator = \"\";\r\n let indexOperator = 0;\r\n\r\n for (operator of operators) {\r\n indexOperator = expression.indexOf(operator);\r\n\r\n if (indexOperator > -1) {\r\n break;\r\n }\r\n }\r\n\r\n if (indexOperator === -1) {\r\n return new ShaderDefineIsDefinedOperator(expression);\r\n }\r\n\r\n const define = expression.substring(0, indexOperator).trim();\r\n const value = expression.substring(indexOperator + operator.length).trim();\r\n\r\n return new ShaderDefineArithmeticOperator(define, operator, value);\r\n }\r\n\r\n private static _BuildSubExpression(expression: string): ShaderDefineExpression {\r\n expression = expression.replace(regexSE, \"defined[$1]\");\r\n\r\n const postfix = ShaderDefineExpression.infixToPostfix(expression);\r\n\r\n const stack: (string | ShaderDefineExpression)[] = [];\r\n\r\n for (const c of postfix) {\r\n if (c !== \"||\" && c !== \"&&\") {\r\n stack.push(c);\r\n } else if (stack.length >= 2) {\r\n let v1 = stack[stack.length - 1],\r\n v2 = stack[stack.length - 2];\r\n\r\n stack.length -= 2;\r\n\r\n const operator = c == \"&&\" ? new ShaderDefineAndOperator() : new ShaderDefineOrOperator();\r\n\r\n if (typeof v1 === \"string\") {\r\n v1 = v1.replace(regexSERevert, \"defined($1)\");\r\n }\r\n\r\n if (typeof v2 === \"string\") {\r\n v2 = v2.replace(regexSERevert, \"defined($1)\");\r\n }\r\n\r\n operator.leftOperand = typeof v2 === \"string\" ? this._ExtractOperation(v2) : v2;\r\n operator.rightOperand = typeof v1 === \"string\" ? this._ExtractOperation(v1) : v1;\r\n\r\n stack.push(operator);\r\n }\r\n }\r\n\r\n let result = stack[stack.length - 1];\r\n\r\n if (typeof result === \"string\") {\r\n result = result.replace(regexSERevert, \"defined($1)\");\r\n }\r\n\r\n // note: stack.length !== 1 if there was an error in the parsing\r\n\r\n return typeof result === \"string\" ? this._ExtractOperation(result) : result;\r\n }\r\n\r\n private static _BuildExpression(line: string, start: number): ShaderCodeTestNode {\r\n const node = new ShaderCodeTestNode();\r\n const command = line.substring(0, start);\r\n let expression = line.substring(start);\r\n\r\n expression = expression.substring(0, (expression.indexOf(\"//\") + 1 || expression.length + 1) - 1).trim();\r\n\r\n if (command === \"#ifdef\") {\r\n node.testExpression = new ShaderDefineIsDefinedOperator(expression);\r\n } else if (command === \"#ifndef\") {\r\n node.testExpression = new ShaderDefineIsDefinedOperator(expression, true);\r\n } else {\r\n node.testExpression = this._BuildSubExpression(expression);\r\n }\r\n\r\n return node;\r\n }\r\n\r\n private static _MoveCursorWithinIf(cursor: ShaderCodeCursor, rootNode: ShaderCodeConditionNode, ifNode: ShaderCodeNode) {\r\n let line = cursor.currentLine;\r\n while (this._MoveCursor(cursor, ifNode)) {\r\n line = cursor.currentLine;\r\n const first5 = line.substring(0, 5).toLowerCase();\r\n\r\n if (first5 === \"#else\") {\r\n const elseNode = new ShaderCodeNode();\r\n rootNode.children.push(elseNode);\r\n this._MoveCursor(cursor, elseNode);\r\n return;\r\n } else if (first5 === \"#elif\") {\r\n const elifNode = this._BuildExpression(line, 5);\r\n\r\n rootNode.children.push(elifNode);\r\n ifNode = elifNode;\r\n }\r\n }\r\n }\r\n\r\n private static _MoveCursor(cursor: ShaderCodeCursor, rootNode: ShaderCodeNode): boolean {\r\n while (cursor.canRead) {\r\n cursor.lineIndex++;\r\n const line = cursor.currentLine;\r\n\r\n if (line.indexOf(\"#\") >= 0) {\r\n const matches = ShaderProcessor._MoveCursorRegex.exec(line);\r\n\r\n if (matches && matches.length) {\r\n const keyword = matches[0];\r\n\r\n switch (keyword) {\r\n case \"#ifdef\": {\r\n const newRootNode = new ShaderCodeConditionNode();\r\n rootNode.children.push(newRootNode);\r\n\r\n const ifNode = this._BuildExpression(line, 6);\r\n newRootNode.children.push(ifNode);\r\n this._MoveCursorWithinIf(cursor, newRootNode, ifNode);\r\n break;\r\n }\r\n case \"#else\":\r\n case \"#elif\":\r\n return true;\r\n case \"#endif\":\r\n return false;\r\n case \"#ifndef\": {\r\n const newRootNode = new ShaderCodeConditionNode();\r\n rootNode.children.push(newRootNode);\r\n\r\n const ifNode = this._BuildExpression(line, 7);\r\n newRootNode.children.push(ifNode);\r\n this._MoveCursorWithinIf(cursor, newRootNode, ifNode);\r\n break;\r\n }\r\n case \"#if\": {\r\n const newRootNode = new ShaderCodeConditionNode();\r\n const ifNode = this._BuildExpression(line, 3);\r\n rootNode.children.push(newRootNode);\r\n\r\n newRootNode.children.push(ifNode);\r\n this._MoveCursorWithinIf(cursor, newRootNode, ifNode);\r\n break;\r\n }\r\n }\r\n continue;\r\n }\r\n }\r\n\r\n const newNode = new ShaderCodeNode();\r\n newNode.line = line;\r\n rootNode.children.push(newNode);\r\n\r\n // Detect additional defines\r\n if (line[0] === \"#\" && line[1] === \"d\") {\r\n const split = line.replace(\";\", \"\").split(\" \");\r\n newNode.additionalDefineKey = split[1];\r\n\r\n if (split.length === 3) {\r\n newNode.additionalDefineValue = split[2];\r\n }\r\n }\r\n }\r\n return false;\r\n }\r\n\r\n private static _EvaluatePreProcessors(sourceCode: string, preprocessors: { [key: string]: string }, options: ProcessingOptions): string {\r\n const rootNode = new ShaderCodeNode();\r\n const cursor = new ShaderCodeCursor();\r\n\r\n cursor.lineIndex = -1;\r\n cursor.lines = sourceCode.split(\"\\n\");\r\n\r\n // Decompose (We keep it in 2 steps so it is easier to maintain and perf hit is insignificant)\r\n this._MoveCursor(cursor, rootNode);\r\n\r\n // Recompose\r\n return rootNode.process(preprocessors, options);\r\n }\r\n\r\n private static _PreparePreProcessors(options: ProcessingOptions, engine: ThinEngine): { [key: string]: string } {\r\n const defines = options.defines;\r\n const preprocessors: { [key: string]: string } = {};\r\n\r\n for (const define of defines) {\r\n const keyValue = define.replace(\"#define\", \"\").replace(\";\", \"\").trim();\r\n const split = keyValue.split(\" \");\r\n preprocessors[split[0]] = split.length > 1 ? split[1] : \"\";\r\n }\r\n\r\n if (options.processor?.shaderLanguage === ShaderLanguage.GLSL) {\r\n preprocessors[\"GL_ES\"] = \"true\";\r\n }\r\n preprocessors[\"__VERSION__\"] = options.version;\r\n preprocessors[options.platformName] = \"true\";\r\n\r\n engine._getGlobalDefines(preprocessors);\r\n\r\n return preprocessors;\r\n }\r\n\r\n private static _ProcessShaderConversion(sourceCode: string, options: ProcessingOptions, engine: ThinEngine): string {\r\n let preparedSourceCode = this._ProcessPrecision(sourceCode, options);\r\n\r\n if (!options.processor) {\r\n return preparedSourceCode;\r\n }\r\n\r\n // Already converted\r\n if (options.processor.shaderLanguage === ShaderLanguage.GLSL && preparedSourceCode.indexOf(\"#version 3\") !== -1) {\r\n preparedSourceCode = preparedSourceCode.replace(\"#version 300 es\", \"\");\r\n if (!options.processor.parseGLES3) {\r\n return preparedSourceCode;\r\n }\r\n }\r\n\r\n const defines = options.defines;\r\n\r\n const preprocessors = this._PreparePreProcessors(options, engine);\r\n\r\n // General pre processing\r\n if (options.processor.preProcessor) {\r\n preparedSourceCode = options.processor.preProcessor(preparedSourceCode, defines, options.isFragment, options.processingContext);\r\n }\r\n\r\n preparedSourceCode = this._EvaluatePreProcessors(preparedSourceCode, preprocessors, options);\r\n\r\n // Post processing\r\n if (options.processor.postProcessor) {\r\n preparedSourceCode = options.processor.postProcessor(preparedSourceCode, defines, options.isFragment, options.processingContext, engine);\r\n }\r\n\r\n // Inline functions tagged with #define inline\r\n if (engine._features.needShaderCodeInlining) {\r\n preparedSourceCode = engine.inlineShaderCode(preparedSourceCode);\r\n }\r\n\r\n return preparedSourceCode;\r\n }\r\n\r\n private static _ApplyPreProcessing(sourceCode: string, options: ProcessingOptions, engine: ThinEngine): string {\r\n let preparedSourceCode = sourceCode;\r\n\r\n const defines = options.defines;\r\n\r\n const preprocessors = this._PreparePreProcessors(options, engine);\r\n\r\n // General pre processing\r\n if (options.processor?.preProcessor) {\r\n preparedSourceCode = options.processor.preProcessor(preparedSourceCode, defines, options.isFragment, options.processingContext);\r\n }\r\n\r\n preparedSourceCode = this._EvaluatePreProcessors(preparedSourceCode, preprocessors, options);\r\n\r\n // Post processing\r\n if (options.processor?.postProcessor) {\r\n preparedSourceCode = options.processor.postProcessor(preparedSourceCode, defines, options.isFragment, options.processingContext, engine);\r\n }\r\n\r\n // Inline functions tagged with #define inline\r\n if (engine._features.needShaderCodeInlining) {\r\n preparedSourceCode = engine.inlineShaderCode(preparedSourceCode);\r\n }\r\n\r\n return preparedSourceCode;\r\n }\r\n\r\n /** @internal */\r\n public static _ProcessIncludes(sourceCode: string, options: ProcessingOptions, callback: (data: any) => void): void {\r\n reusableMatches.length = 0;\r\n let match: RegExpMatchArray | null;\r\n // stay back-compat to the old matchAll syntax\r\n while ((match = regexShaderInclude.exec(sourceCode)) !== null) {\r\n reusableMatches.push(match);\r\n }\r\n\r\n let returnValue = String(sourceCode);\r\n let parts = [sourceCode];\r\n\r\n let keepProcessing = false;\r\n\r\n for (const match of reusableMatches) {\r\n let includeFile = match[1];\r\n\r\n // Uniform declaration\r\n if (includeFile.indexOf(\"__decl__\") !== -1) {\r\n includeFile = includeFile.replace(regexShaderDecl, \"\");\r\n if (options.supportsUniformBuffers) {\r\n includeFile = includeFile.replace(\"Vertex\", \"Ubo\").replace(\"Fragment\", \"Ubo\");\r\n }\r\n includeFile = includeFile + \"Declaration\";\r\n }\r\n\r\n if (options.includesShadersStore[includeFile]) {\r\n // Substitution\r\n let includeContent = options.includesShadersStore[includeFile];\r\n if (match[2]) {\r\n const splits = match[3].split(\",\");\r\n\r\n for (let index = 0; index < splits.length; index += 2) {\r\n const source = new RegExp(splits[index], \"g\");\r\n const dest = splits[index + 1];\r\n\r\n includeContent = includeContent.replace(source, dest);\r\n }\r\n }\r\n\r\n if (match[4]) {\r\n const indexString = match[5];\r\n\r\n if (indexString.indexOf(\"..\") !== -1) {\r\n const indexSplits = indexString.split(\"..\");\r\n const minIndex = parseInt(indexSplits[0]);\r\n let maxIndex = parseInt(indexSplits[1]);\r\n let sourceIncludeContent = includeContent.slice(0);\r\n includeContent = \"\";\r\n\r\n if (isNaN(maxIndex)) {\r\n maxIndex = options.indexParameters[indexSplits[1]];\r\n }\r\n\r\n for (let i = minIndex; i < maxIndex; i++) {\r\n if (!options.supportsUniformBuffers) {\r\n // Ubo replacement\r\n sourceIncludeContent = sourceIncludeContent.replace(regexLightX, (str: string, p1: string) => {\r\n return p1 + \"{X}\";\r\n });\r\n }\r\n includeContent += sourceIncludeContent.replace(regexX, i.toString()) + \"\\n\";\r\n }\r\n } else {\r\n if (!options.supportsUniformBuffers) {\r\n // Ubo replacement\r\n includeContent = includeContent.replace(regexLightX, (str: string, p1: string) => {\r\n return p1 + \"{X}\";\r\n });\r\n }\r\n includeContent = includeContent.replace(regexX, indexString);\r\n }\r\n }\r\n\r\n // Replace\r\n // Split all parts on match[0] and intersperse the parts with the include content\r\n const newParts = [];\r\n for (const part of parts) {\r\n const splitPart = part.split(match[0]);\r\n for (let i = 0; i < splitPart.length - 1; i++) {\r\n newParts.push(splitPart[i]);\r\n newParts.push(includeContent);\r\n }\r\n newParts.push(splitPart[splitPart.length - 1]);\r\n }\r\n parts = newParts;\r\n\r\n keepProcessing = keepProcessing || includeContent.indexOf(\"#include<\") >= 0 || includeContent.indexOf(\"#include <\") >= 0;\r\n } else {\r\n const includeShaderUrl = options.shadersRepository + \"ShadersInclude/\" + includeFile + \".fx\";\r\n\r\n ShaderProcessor._FileToolsLoadFile(includeShaderUrl, (fileContent) => {\r\n options.includesShadersStore[includeFile] = fileContent as string;\r\n this._ProcessIncludes(parts.join(\"\"), options, callback);\r\n });\r\n return;\r\n }\r\n }\r\n reusableMatches.length = 0;\r\n\r\n returnValue = parts.join(\"\");\r\n\r\n if (keepProcessing) {\r\n this._ProcessIncludes(returnValue.toString(), options, callback);\r\n } else {\r\n callback(returnValue);\r\n }\r\n }\r\n\r\n /**\r\n * Loads a file from a url\r\n * @param url url to load\r\n * @param onSuccess callback called when the file successfully loads\r\n * @param onProgress callback called while file is loading (if the server supports this mode)\r\n * @param offlineProvider defines the offline provider for caching\r\n * @param useArrayBuffer defines a boolean indicating that date must be returned as ArrayBuffer\r\n * @param onError callback called when the file fails to load\r\n * @returns a file request object\r\n * @internal\r\n */\r\n public static _FileToolsLoadFile(\r\n url: string,\r\n onSuccess: (data: string | ArrayBuffer, responseURL?: string) => void,\r\n onProgress?: (ev: ProgressEvent) => void,\r\n offlineProvider?: IOfflineProvider,\r\n useArrayBuffer?: boolean,\r\n onError?: (request?: WebRequest, exception?: LoadFileError) => void\r\n ): IFileRequest {\r\n throw _WarnImport(\"FileTools\");\r\n }\r\n}\r\n","import { ShaderLanguage } from \"../Materials/shaderLanguage\";\r\n\r\n/**\r\n * Defines the shader related stores and directory\r\n */\r\nexport class ShaderStore {\r\n /**\r\n * Gets or sets the relative url used to load shaders if using the engine in non-minified mode\r\n */\r\n public static ShadersRepository = \"src/Shaders/\";\r\n /**\r\n * Store of each shader (The can be looked up using effect.key)\r\n */\r\n public static ShadersStore: { [key: string]: string } = {};\r\n /**\r\n * Store of each included file for a shader (The can be looked up using effect.key)\r\n */\r\n public static IncludesShadersStore: { [key: string]: string } = {};\r\n\r\n /**\r\n * Gets or sets the relative url used to load shaders (WGSL) if using the engine in non-minified mode\r\n */\r\n public static ShadersRepositoryWGSL = \"src/ShadersWGSL/\";\r\n /**\r\n * Store of each shader (WGSL)\r\n */\r\n public static ShadersStoreWGSL: { [key: string]: string } = {};\r\n /**\r\n * Store of each included file for a shader (WGSL)\r\n */\r\n public static IncludesShadersStoreWGSL: { [key: string]: string } = {};\r\n\r\n /**\r\n * Gets the shaders repository path for a given shader language\r\n * @param shaderLanguage the shader language\r\n * @returns the path to the shaders repository\r\n */\r\n public static GetShadersRepository(shaderLanguage = ShaderLanguage.GLSL): string {\r\n return shaderLanguage === ShaderLanguage.GLSL ? ShaderStore.ShadersRepository : ShaderStore.ShadersRepositoryWGSL;\r\n }\r\n\r\n /**\r\n * Gets the shaders store of a given shader language\r\n * @param shaderLanguage the shader language\r\n * @returns the shaders store\r\n */\r\n public static GetShadersStore(shaderLanguage = ShaderLanguage.GLSL): { [key: string]: string } {\r\n return shaderLanguage === ShaderLanguage.GLSL ? ShaderStore.ShadersStore : ShaderStore.ShadersStoreWGSL;\r\n }\r\n\r\n /**\r\n * Gets the include shaders store of a given shader language\r\n * @param shaderLanguage the shader language\r\n * @returns the include shaders store\r\n */\r\n public static GetIncludesShadersStore(shaderLanguage = ShaderLanguage.GLSL): { [key: string]: string } {\r\n return shaderLanguage === ShaderLanguage.GLSL ? ShaderStore.IncludesShadersStore : ShaderStore.IncludesShadersStoreWGSL;\r\n }\r\n}\r\n","import { Observable } from \"../Misc/observable\";\r\nimport type { FloatArray, Nullable } from \"../types\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport { GetDOMTextContent, IsWindowObjectExist } from \"../Misc/domManagement\";\r\nimport { Logger } from \"../Misc/logger\";\r\nimport type { IDisposable } from \"../scene\";\r\nimport type { IPipelineContext } from \"../Engines/IPipelineContext\";\r\nimport type { DataBuffer } from \"../Buffers/dataBuffer\";\r\nimport { ShaderProcessor } from \"../Engines/Processors/shaderProcessor\";\r\nimport type { ProcessingOptions, ShaderCustomProcessingFunction, ShaderProcessingContext } from \"../Engines/Processors/shaderProcessingOptions\";\r\nimport type { IMatrixLike, IVector2Like, IVector3Like, IVector4Like, IColor3Like, IColor4Like, IQuaternionLike } from \"../Maths/math.like\";\r\nimport type { ThinEngine } from \"../Engines/thinEngine\";\r\nimport type { IEffectFallbacks } from \"./iEffectFallbacks\";\r\nimport { ShaderStore as EngineShaderStore } from \"../Engines/shaderStore\";\r\nimport { ShaderLanguage } from \"./shaderLanguage\";\r\n\r\nimport type { Engine } from \"../Engines/engine\";\r\nimport type { InternalTexture } from \"../Materials/Textures/internalTexture\";\r\nimport type { ThinTexture } from \"../Materials/Textures/thinTexture\";\r\nimport type { RenderTargetTexture } from \"../Materials/Textures/renderTargetTexture\";\r\nimport type { PostProcess } from \"../PostProcesses/postProcess\";\r\n\r\n/**\r\n * Options to be used when creating an effect.\r\n */\r\nexport interface IEffectCreationOptions {\r\n /**\r\n * Attributes that will be used in the shader.\r\n */\r\n attributes: string[];\r\n /**\r\n * Uniform variable names that will be set in the shader.\r\n */\r\n uniformsNames: string[];\r\n /**\r\n * Uniform buffer variable names that will be set in the shader.\r\n */\r\n uniformBuffersNames: string[];\r\n /**\r\n * Sampler texture variable names that will be set in the shader.\r\n */\r\n samplers: string[];\r\n /**\r\n * Define statements that will be set in the shader.\r\n */\r\n defines: any;\r\n /**\r\n * Possible fallbacks for this effect to improve performance when needed.\r\n */\r\n fallbacks: Nullable;\r\n /**\r\n * Callback that will be called when the shader is compiled.\r\n */\r\n onCompiled: Nullable<(effect: Effect) => void>;\r\n /**\r\n * Callback that will be called if an error occurs during shader compilation.\r\n */\r\n onError: Nullable<(effect: Effect, errors: string) => void>;\r\n /**\r\n * Parameters to be used with Babylons include syntax to iterate over an array (eg. {lights: 10})\r\n */\r\n indexParameters?: any;\r\n /**\r\n * Max number of lights that can be used in the shader.\r\n */\r\n maxSimultaneousLights?: number;\r\n /**\r\n * See https://developer.mozilla.org/en-US/docs/Web/API/WebGL2RenderingContext/transformFeedbackVaryings\r\n */\r\n transformFeedbackVaryings?: Nullable;\r\n /**\r\n * If provided, will be called two times with the vertex and fragment code so that this code can be updated before it is compiled by the GPU\r\n */\r\n processFinalCode?: Nullable;\r\n /**\r\n * If provided, will be called two times with the vertex and fragment code so that this code can be updated after the #include have been processed\r\n */\r\n processCodeAfterIncludes?: Nullable;\r\n /**\r\n * Is this effect rendering to several color attachments ?\r\n */\r\n multiTarget?: boolean;\r\n /**\r\n * The language the shader is written in (default: GLSL)\r\n */\r\n shaderLanguage?: ShaderLanguage;\r\n}\r\n\r\n/**\r\n * Effect containing vertex and fragment shader that can be executed on an object.\r\n */\r\nexport class Effect implements IDisposable {\r\n /**\r\n * Gets or sets the relative url used to load shaders if using the engine in non-minified mode\r\n */\r\n public static get ShadersRepository(): string {\r\n return EngineShaderStore.ShadersRepository;\r\n }\r\n public static set ShadersRepository(repo: string) {\r\n EngineShaderStore.ShadersRepository = repo;\r\n }\r\n /**\r\n * Enable logging of the shader code when a compilation error occurs\r\n */\r\n public static LogShaderCodeOnCompilationError = true;\r\n /**\r\n * Name of the effect.\r\n */\r\n public name: any = null;\r\n /**\r\n * String container all the define statements that should be set on the shader.\r\n */\r\n public defines: string = \"\";\r\n /**\r\n * Callback that will be called when the shader is compiled.\r\n */\r\n public onCompiled: Nullable<(effect: Effect) => void> = null;\r\n /**\r\n * Callback that will be called if an error occurs during shader compilation.\r\n */\r\n public onError: Nullable<(effect: Effect, errors: string) => void> = null;\r\n /**\r\n * Callback that will be called when effect is bound.\r\n */\r\n public onBind: Nullable<(effect: Effect) => void> = null;\r\n /**\r\n * Unique ID of the effect.\r\n */\r\n public uniqueId = 0;\r\n /**\r\n * Observable that will be called when the shader is compiled.\r\n * It is recommended to use executeWhenCompile() or to make sure that scene.isReady() is called to get this observable raised.\r\n */\r\n public onCompileObservable = new Observable();\r\n /**\r\n * Observable that will be called if an error occurs during shader compilation.\r\n */\r\n public onErrorObservable = new Observable();\r\n\r\n /** @internal */\r\n public _onBindObservable: Nullable> = null;\r\n\r\n /**\r\n * @internal\r\n * Specifies if the effect was previously ready\r\n */\r\n public _wasPreviouslyReady = false;\r\n\r\n /**\r\n * @internal\r\n * Forces the code from bindForSubMesh to be fully run the next time it is called\r\n * It is used in frozen mode to make sure the effect is properly rebound when a new effect is created\r\n */\r\n public _forceRebindOnNextCall = false;\r\n\r\n /**\r\n * @internal\r\n * Specifies if the effect was previously using instances\r\n */\r\n public _wasPreviouslyUsingInstances: Nullable = null;\r\n\r\n private _isDisposed = false;\r\n\r\n /**\r\n * Observable that will be called when effect is bound.\r\n */\r\n public get onBindObservable(): Observable {\r\n if (!this._onBindObservable) {\r\n this._onBindObservable = new Observable();\r\n }\r\n\r\n return this._onBindObservable;\r\n }\r\n\r\n /** @internal */\r\n public _bonesComputationForcedToCPU = false;\r\n /** @internal */\r\n public _uniformBuffersNames: { [key: string]: number } = {};\r\n /** @internal */\r\n public _samplerList: string[];\r\n /** @internal */\r\n public _multiTarget: boolean = false;\r\n\r\n private static _UniqueIdSeed = 0;\r\n /** @internal */\r\n public _engine: Engine;\r\n private _uniformBuffersNamesList: string[];\r\n private _uniformsNames: string[];\r\n private _samplers: { [key: string]: number } = {};\r\n private _isReady = false;\r\n private _compilationError = \"\";\r\n private _allFallbacksProcessed = false;\r\n private _attributesNames: string[];\r\n private _attributes: number[];\r\n private _attributeLocationByName: { [name: string]: number };\r\n private _uniforms: { [key: string]: Nullable } = {};\r\n /**\r\n * Key for the effect.\r\n * @internal\r\n */\r\n public _key: string = \"\";\r\n private _indexParameters: any;\r\n private _fallbacks: Nullable = null;\r\n private _vertexSourceCodeOverride: string = \"\";\r\n private _fragmentSourceCodeOverride: string = \"\";\r\n private _transformFeedbackVaryings: Nullable = null;\r\n private _shaderLanguage: ShaderLanguage;\r\n /**\r\n * Compiled shader to webGL program.\r\n * @internal\r\n */\r\n public _pipelineContext: Nullable = null;\r\n /** @internal */\r\n public _vertexSourceCode: string = \"\";\r\n /** @internal */\r\n public _fragmentSourceCode: string = \"\";\r\n\r\n /** @internal */\r\n private _vertexSourceCodeBeforeMigration: string = \"\";\r\n /** @internal */\r\n private _fragmentSourceCodeBeforeMigration: string = \"\";\r\n\r\n /** @internal */\r\n private _rawVertexSourceCode: string = \"\";\r\n /** @internal */\r\n private _rawFragmentSourceCode: string = \"\";\r\n\r\n private static _BaseCache: { [key: number]: DataBuffer } = {};\r\n private _processingContext: Nullable;\r\n\r\n /**\r\n * Instantiates an effect.\r\n * An effect can be used to create/manage/execute vertex and fragment shaders.\r\n * @param baseName Name of the effect.\r\n * @param attributesNamesOrOptions List of attribute names that will be passed to the shader or set of all options to create the effect.\r\n * @param uniformsNamesOrEngine List of uniform variable names that will be passed to the shader or the engine that will be used to render effect.\r\n * @param samplers List of sampler variables that will be passed to the shader.\r\n * @param engine Engine to be used to render the effect\r\n * @param defines Define statements to be added to the shader.\r\n * @param fallbacks Possible fallbacks for this effect to improve performance when needed.\r\n * @param onCompiled Callback that will be called when the shader is compiled.\r\n * @param onError Callback that will be called if an error occurs during shader compilation.\r\n * @param indexParameters Parameters to be used with Babylons include syntax to iterate over an array (eg. {lights: 10})\r\n * @param key Effect Key identifying uniquely compiled shader variants\r\n * @param shaderLanguage the language the shader is written in (default: GLSL)\r\n */\r\n constructor(\r\n baseName: any,\r\n attributesNamesOrOptions: string[] | IEffectCreationOptions,\r\n uniformsNamesOrEngine: string[] | ThinEngine,\r\n samplers: Nullable = null,\r\n engine?: ThinEngine,\r\n defines: Nullable = null,\r\n fallbacks: Nullable = null,\r\n onCompiled: Nullable<(effect: Effect) => void> = null,\r\n onError: Nullable<(effect: Effect, errors: string) => void> = null,\r\n indexParameters?: any,\r\n key: string = \"\",\r\n shaderLanguage = ShaderLanguage.GLSL\r\n ) {\r\n this.name = baseName;\r\n this._key = key;\r\n\r\n let processCodeAfterIncludes: ShaderCustomProcessingFunction | undefined = undefined;\r\n let processFinalCode: Nullable = null;\r\n\r\n if ((attributesNamesOrOptions).attributes) {\r\n const options = attributesNamesOrOptions;\r\n this._engine = uniformsNamesOrEngine;\r\n\r\n this._attributesNames = options.attributes;\r\n this._uniformsNames = options.uniformsNames.concat(options.samplers);\r\n this._samplerList = options.samplers.slice();\r\n this.defines = options.defines;\r\n this.onError = options.onError;\r\n this.onCompiled = options.onCompiled;\r\n this._fallbacks = options.fallbacks;\r\n this._indexParameters = options.indexParameters;\r\n this._transformFeedbackVaryings = options.transformFeedbackVaryings || null;\r\n this._multiTarget = !!options.multiTarget;\r\n this._shaderLanguage = options.shaderLanguage ?? ShaderLanguage.GLSL;\r\n\r\n if (options.uniformBuffersNames) {\r\n this._uniformBuffersNamesList = options.uniformBuffersNames.slice();\r\n for (let i = 0; i < options.uniformBuffersNames.length; i++) {\r\n this._uniformBuffersNames[options.uniformBuffersNames[i]] = i;\r\n }\r\n }\r\n\r\n processFinalCode = options.processFinalCode ?? null;\r\n processCodeAfterIncludes = options.processCodeAfterIncludes ?? undefined;\r\n } else {\r\n this._engine = engine;\r\n this.defines = defines == null ? \"\" : defines;\r\n this._uniformsNames = (uniformsNamesOrEngine).concat(samplers);\r\n this._samplerList = samplers ? samplers.slice() : [];\r\n this._attributesNames = attributesNamesOrOptions;\r\n this._uniformBuffersNamesList = [];\r\n this._shaderLanguage = shaderLanguage;\r\n\r\n this.onError = onError;\r\n this.onCompiled = onCompiled;\r\n\r\n this._indexParameters = indexParameters;\r\n this._fallbacks = fallbacks;\r\n }\r\n\r\n this._attributeLocationByName = {};\r\n\r\n this.uniqueId = Effect._UniqueIdSeed++;\r\n\r\n let vertexSource: any;\r\n let fragmentSource: any;\r\n\r\n const hostDocument = IsWindowObjectExist() ? this._engine.getHostDocument() : null;\r\n\r\n if (baseName.vertexSource) {\r\n vertexSource = \"source:\" + baseName.vertexSource;\r\n } else if (baseName.vertexElement) {\r\n vertexSource = hostDocument ? hostDocument.getElementById(baseName.vertexElement) : null;\r\n\r\n if (!vertexSource) {\r\n vertexSource = baseName.vertexElement;\r\n }\r\n } else {\r\n vertexSource = baseName.vertex || baseName;\r\n }\r\n\r\n if (baseName.fragmentSource) {\r\n fragmentSource = \"source:\" + baseName.fragmentSource;\r\n } else if (baseName.fragmentElement) {\r\n fragmentSource = hostDocument ? hostDocument.getElementById(baseName.fragmentElement) : null;\r\n\r\n if (!fragmentSource) {\r\n fragmentSource = baseName.fragmentElement;\r\n }\r\n } else {\r\n fragmentSource = baseName.fragment || baseName;\r\n }\r\n\r\n this._processingContext = this._engine._getShaderProcessingContext(this._shaderLanguage);\r\n\r\n let processorOptions: ProcessingOptions = {\r\n defines: this.defines.split(\"\\n\"),\r\n indexParameters: this._indexParameters,\r\n isFragment: false,\r\n shouldUseHighPrecisionShader: this._engine._shouldUseHighPrecisionShader,\r\n processor: this._engine._getShaderProcessor(this._shaderLanguage),\r\n supportsUniformBuffers: this._engine.supportsUniformBuffers,\r\n shadersRepository: EngineShaderStore.GetShadersRepository(this._shaderLanguage),\r\n includesShadersStore: EngineShaderStore.GetIncludesShadersStore(this._shaderLanguage),\r\n version: (this._engine.version * 100).toString(),\r\n platformName: this._engine.shaderPlatformName,\r\n processingContext: this._processingContext,\r\n isNDCHalfZRange: this._engine.isNDCHalfZRange,\r\n useReverseDepthBuffer: this._engine.useReverseDepthBuffer,\r\n processCodeAfterIncludes,\r\n };\r\n\r\n const shaderCodes: [string | undefined, string | undefined] = [undefined, undefined];\r\n const shadersLoaded = () => {\r\n if (shaderCodes[0] && shaderCodes[1]) {\r\n processorOptions.isFragment = true;\r\n const [migratedVertexCode, fragmentCode] = shaderCodes;\r\n ShaderProcessor.Process(\r\n fragmentCode,\r\n processorOptions,\r\n (migratedFragmentCode, codeBeforeMigration) => {\r\n this._fragmentSourceCodeBeforeMigration = codeBeforeMigration;\r\n if (processFinalCode) {\r\n migratedFragmentCode = processFinalCode(\"fragment\", migratedFragmentCode);\r\n }\r\n const finalShaders = ShaderProcessor.Finalize(migratedVertexCode, migratedFragmentCode, processorOptions);\r\n processorOptions = null as any;\r\n this._useFinalCode(finalShaders.vertexCode, finalShaders.fragmentCode, baseName);\r\n },\r\n this._engine\r\n );\r\n }\r\n };\r\n this._loadShader(vertexSource, \"Vertex\", \"\", (vertexCode) => {\r\n ShaderProcessor.Initialize(processorOptions);\r\n ShaderProcessor.Process(\r\n vertexCode,\r\n processorOptions,\r\n (migratedVertexCode, codeBeforeMigration) => {\r\n this._rawVertexSourceCode = vertexCode;\r\n this._vertexSourceCodeBeforeMigration = codeBeforeMigration;\r\n if (processFinalCode) {\r\n migratedVertexCode = processFinalCode(\"vertex\", migratedVertexCode);\r\n }\r\n shaderCodes[0] = migratedVertexCode;\r\n shadersLoaded();\r\n },\r\n this._engine\r\n );\r\n });\r\n this._loadShader(fragmentSource, \"Fragment\", \"Pixel\", (fragmentCode) => {\r\n this._rawFragmentSourceCode = fragmentCode;\r\n shaderCodes[1] = fragmentCode;\r\n shadersLoaded();\r\n });\r\n }\r\n\r\n private _useFinalCode(migratedVertexCode: string, migratedFragmentCode: string, baseName: any) {\r\n if (baseName) {\r\n const vertex = baseName.vertexElement || baseName.vertex || baseName.spectorName || baseName;\r\n const fragment = baseName.fragmentElement || baseName.fragment || baseName.spectorName || baseName;\r\n\r\n this._vertexSourceCode = (this._shaderLanguage === ShaderLanguage.WGSL ? \"//\" : \"\") + \"#define SHADER_NAME vertex:\" + vertex + \"\\n\" + migratedVertexCode;\r\n this._fragmentSourceCode = (this._shaderLanguage === ShaderLanguage.WGSL ? \"//\" : \"\") + \"#define SHADER_NAME fragment:\" + fragment + \"\\n\" + migratedFragmentCode;\r\n } else {\r\n this._vertexSourceCode = migratedVertexCode;\r\n this._fragmentSourceCode = migratedFragmentCode;\r\n }\r\n this._prepareEffect();\r\n }\r\n\r\n /**\r\n * Unique key for this effect\r\n */\r\n public get key(): string {\r\n return this._key;\r\n }\r\n\r\n /**\r\n * If the effect has been compiled and prepared.\r\n * @returns if the effect is compiled and prepared.\r\n */\r\n public isReady(): boolean {\r\n try {\r\n return this._isReadyInternal();\r\n } catch {\r\n return false;\r\n }\r\n }\r\n\r\n private _isReadyInternal(): boolean {\r\n if (this._isReady) {\r\n return true;\r\n }\r\n if (this._pipelineContext) {\r\n return this._pipelineContext.isReady;\r\n }\r\n return false;\r\n }\r\n\r\n /**\r\n * The engine the effect was initialized with.\r\n * @returns the engine.\r\n */\r\n public getEngine(): Engine {\r\n return this._engine;\r\n }\r\n\r\n /**\r\n * The pipeline context for this effect\r\n * @returns the associated pipeline context\r\n */\r\n public getPipelineContext(): Nullable {\r\n return this._pipelineContext;\r\n }\r\n\r\n /**\r\n * The set of names of attribute variables for the shader.\r\n * @returns An array of attribute names.\r\n */\r\n public getAttributesNames(): string[] {\r\n return this._attributesNames;\r\n }\r\n\r\n /**\r\n * Returns the attribute at the given index.\r\n * @param index The index of the attribute.\r\n * @returns The location of the attribute.\r\n */\r\n public getAttributeLocation(index: number): number {\r\n return this._attributes[index];\r\n }\r\n\r\n /**\r\n * Returns the attribute based on the name of the variable.\r\n * @param name of the attribute to look up.\r\n * @returns the attribute location.\r\n */\r\n public getAttributeLocationByName(name: string): number {\r\n return this._attributeLocationByName[name];\r\n }\r\n\r\n /**\r\n * The number of attributes.\r\n * @returns the number of attributes.\r\n */\r\n public getAttributesCount(): number {\r\n return this._attributes.length;\r\n }\r\n\r\n /**\r\n * Gets the index of a uniform variable.\r\n * @param uniformName of the uniform to look up.\r\n * @returns the index.\r\n */\r\n public getUniformIndex(uniformName: string): number {\r\n return this._uniformsNames.indexOf(uniformName);\r\n }\r\n\r\n /**\r\n * Returns the attribute based on the name of the variable.\r\n * @param uniformName of the uniform to look up.\r\n * @returns the location of the uniform.\r\n */\r\n public getUniform(uniformName: string): Nullable {\r\n return this._uniforms[uniformName];\r\n }\r\n\r\n /**\r\n * Returns an array of sampler variable names\r\n * @returns The array of sampler variable names.\r\n */\r\n public getSamplers(): string[] {\r\n return this._samplerList;\r\n }\r\n\r\n /**\r\n * Returns an array of uniform variable names\r\n * @returns The array of uniform variable names.\r\n */\r\n public getUniformNames(): string[] {\r\n return this._uniformsNames;\r\n }\r\n\r\n /**\r\n * Returns an array of uniform buffer variable names\r\n * @returns The array of uniform buffer variable names.\r\n */\r\n public getUniformBuffersNames(): string[] {\r\n return this._uniformBuffersNamesList;\r\n }\r\n\r\n /**\r\n * Returns the index parameters used to create the effect\r\n * @returns The index parameters object\r\n */\r\n public getIndexParameters(): any {\r\n return this._indexParameters;\r\n }\r\n\r\n /**\r\n * The error from the last compilation.\r\n * @returns the error string.\r\n */\r\n public getCompilationError(): string {\r\n return this._compilationError;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating that all fallbacks were used during compilation\r\n * @returns true if all fallbacks were used\r\n */\r\n public allFallbacksProcessed(): boolean {\r\n return this._allFallbacksProcessed;\r\n }\r\n\r\n /**\r\n * Adds a callback to the onCompiled observable and call the callback immediately if already ready.\r\n * @param func The callback to be used.\r\n */\r\n public executeWhenCompiled(func: (effect: Effect) => void): void {\r\n if (this.isReady()) {\r\n func(this);\r\n return;\r\n }\r\n\r\n this.onCompileObservable.add((effect) => {\r\n func(effect);\r\n });\r\n\r\n if (!this._pipelineContext || this._pipelineContext.isAsync) {\r\n setTimeout(() => {\r\n this._checkIsReady(null);\r\n }, 16);\r\n }\r\n }\r\n\r\n private _checkIsReady(previousPipelineContext: Nullable) {\r\n try {\r\n if (this._isReadyInternal()) {\r\n return;\r\n }\r\n } catch (e) {\r\n this._processCompilationErrors(e, previousPipelineContext);\r\n return;\r\n }\r\n\r\n if (this._isDisposed) {\r\n return;\r\n }\r\n\r\n setTimeout(() => {\r\n this._checkIsReady(previousPipelineContext);\r\n }, 16);\r\n }\r\n\r\n private _loadShader(shader: any, key: string, optionalKey: string, callback: (data: any) => void): void {\r\n if (typeof HTMLElement !== \"undefined\") {\r\n // DOM element ?\r\n if (shader instanceof HTMLElement) {\r\n const shaderCode = GetDOMTextContent(shader);\r\n callback(shaderCode);\r\n return;\r\n }\r\n }\r\n\r\n // Direct source ?\r\n if (shader.substr(0, 7) === \"source:\") {\r\n callback(shader.substr(7));\r\n return;\r\n }\r\n\r\n // Base64 encoded ?\r\n if (shader.substr(0, 7) === \"base64:\") {\r\n const shaderBinary = window.atob(shader.substr(7));\r\n callback(shaderBinary);\r\n return;\r\n }\r\n\r\n const shaderStore = EngineShaderStore.GetShadersStore(this._shaderLanguage);\r\n\r\n // Is in local store ?\r\n if (shaderStore[shader + key + \"Shader\"]) {\r\n callback(shaderStore[shader + key + \"Shader\"]);\r\n return;\r\n }\r\n\r\n if (optionalKey && shaderStore[shader + optionalKey + \"Shader\"]) {\r\n callback(shaderStore[shader + optionalKey + \"Shader\"]);\r\n return;\r\n }\r\n\r\n let shaderUrl;\r\n\r\n if (shader[0] === \".\" || shader[0] === \"/\" || shader.indexOf(\"http\") > -1) {\r\n shaderUrl = shader;\r\n } else {\r\n shaderUrl = EngineShaderStore.GetShadersRepository(this._shaderLanguage) + shader;\r\n }\r\n\r\n // Vertex shader\r\n this._engine._loadFile(shaderUrl + \".\" + key.toLowerCase() + \".fx\", callback);\r\n }\r\n\r\n /**\r\n * Gets the vertex shader source code of this effect\r\n * This is the final source code that will be compiled, after all the processing has been done (pre-processing applied, code injection/replacement, etc)\r\n */\r\n public get vertexSourceCode(): string {\r\n return this._vertexSourceCodeOverride && this._fragmentSourceCodeOverride\r\n ? this._vertexSourceCodeOverride\r\n : this._pipelineContext?._getVertexShaderCode() ?? this._vertexSourceCode;\r\n }\r\n\r\n /**\r\n * Gets the fragment shader source code of this effect\r\n * This is the final source code that will be compiled, after all the processing has been done (pre-processing applied, code injection/replacement, etc)\r\n */\r\n public get fragmentSourceCode(): string {\r\n return this._vertexSourceCodeOverride && this._fragmentSourceCodeOverride\r\n ? this._fragmentSourceCodeOverride\r\n : this._pipelineContext?._getFragmentShaderCode() ?? this._fragmentSourceCode;\r\n }\r\n\r\n /**\r\n * Gets the vertex shader source code before migration.\r\n * This is the source code after the include directives have been replaced by their contents but before the code is migrated, i.e. before ShaderProcess._ProcessShaderConversion is executed.\r\n * This method is, among other things, responsible for parsing #if/#define directives as well as converting GLES2 syntax to GLES3 (in the case of WebGL).\r\n */\r\n public get vertexSourceCodeBeforeMigration(): string {\r\n return this._vertexSourceCodeBeforeMigration;\r\n }\r\n\r\n /**\r\n * Gets the fragment shader source code before migration.\r\n * This is the source code after the include directives have been replaced by their contents but before the code is migrated, i.e. before ShaderProcess._ProcessShaderConversion is executed.\r\n * This method is, among other things, responsible for parsing #if/#define directives as well as converting GLES2 syntax to GLES3 (in the case of WebGL).\r\n */\r\n public get fragmentSourceCodeBeforeMigration(): string {\r\n return this._fragmentSourceCodeBeforeMigration;\r\n }\r\n\r\n /**\r\n * Gets the vertex shader source code before it has been modified by any processing\r\n */\r\n public get rawVertexSourceCode(): string {\r\n return this._rawVertexSourceCode;\r\n }\r\n\r\n /**\r\n * Gets the fragment shader source code before it has been modified by any processing\r\n */\r\n public get rawFragmentSourceCode(): string {\r\n return this._rawFragmentSourceCode;\r\n }\r\n\r\n /**\r\n * Recompiles the webGL program\r\n * @param vertexSourceCode The source code for the vertex shader.\r\n * @param fragmentSourceCode The source code for the fragment shader.\r\n * @param onCompiled Callback called when completed.\r\n * @param onError Callback called on error.\r\n * @internal\r\n */\r\n public _rebuildProgram(vertexSourceCode: string, fragmentSourceCode: string, onCompiled: (pipelineContext: IPipelineContext) => void, onError: (message: string) => void) {\r\n this._isReady = false;\r\n\r\n this._vertexSourceCodeOverride = vertexSourceCode;\r\n this._fragmentSourceCodeOverride = fragmentSourceCode;\r\n this.onError = (effect, error) => {\r\n if (onError) {\r\n onError(error);\r\n }\r\n };\r\n this.onCompiled = () => {\r\n const scenes = this.getEngine().scenes;\r\n if (scenes) {\r\n for (let i = 0; i < scenes.length; i++) {\r\n scenes[i].markAllMaterialsAsDirty(Constants.MATERIAL_AllDirtyFlag);\r\n }\r\n }\r\n\r\n this._pipelineContext!._handlesSpectorRebuildCallback(onCompiled);\r\n };\r\n this._fallbacks = null;\r\n this._prepareEffect();\r\n }\r\n\r\n /**\r\n * Prepares the effect\r\n * @internal\r\n */\r\n public _prepareEffect() {\r\n const attributesNames = this._attributesNames;\r\n const defines = this.defines;\r\n\r\n const previousPipelineContext = this._pipelineContext;\r\n\r\n this._isReady = false;\r\n\r\n try {\r\n const engine = this._engine;\r\n\r\n this._pipelineContext = engine.createPipelineContext(this._processingContext);\r\n this._pipelineContext._name = this._key;\r\n\r\n const rebuildRebind = this._rebuildProgram.bind(this);\r\n if (this._vertexSourceCodeOverride && this._fragmentSourceCodeOverride) {\r\n engine._preparePipelineContext(\r\n this._pipelineContext,\r\n this._vertexSourceCodeOverride,\r\n this._fragmentSourceCodeOverride,\r\n true,\r\n this._rawVertexSourceCode,\r\n this._rawFragmentSourceCode,\r\n rebuildRebind,\r\n null,\r\n this._transformFeedbackVaryings,\r\n this._key\r\n );\r\n } else {\r\n engine._preparePipelineContext(\r\n this._pipelineContext,\r\n this._vertexSourceCode,\r\n this._fragmentSourceCode,\r\n false,\r\n this._rawVertexSourceCode,\r\n this._rawFragmentSourceCode,\r\n rebuildRebind,\r\n defines,\r\n this._transformFeedbackVaryings,\r\n this._key\r\n );\r\n }\r\n\r\n engine._executeWhenRenderingStateIsCompiled(this._pipelineContext, () => {\r\n this._attributes = [];\r\n this._pipelineContext!._fillEffectInformation(\r\n this,\r\n this._uniformBuffersNames,\r\n this._uniformsNames,\r\n this._uniforms,\r\n this._samplerList,\r\n this._samplers,\r\n attributesNames,\r\n this._attributes\r\n );\r\n\r\n // Caches attribute locations.\r\n if (attributesNames) {\r\n for (let i = 0; i < attributesNames.length; i++) {\r\n const name = attributesNames[i];\r\n this._attributeLocationByName[name] = this._attributes[i];\r\n }\r\n }\r\n\r\n engine.bindSamplers(this);\r\n\r\n this._compilationError = \"\";\r\n this._isReady = true;\r\n if (this.onCompiled) {\r\n this.onCompiled(this);\r\n }\r\n this.onCompileObservable.notifyObservers(this);\r\n this.onCompileObservable.clear();\r\n\r\n // Unbind mesh reference in fallbacks\r\n if (this._fallbacks) {\r\n this._fallbacks.unBindMesh();\r\n }\r\n\r\n if (previousPipelineContext) {\r\n this.getEngine()._deletePipelineContext(previousPipelineContext);\r\n }\r\n });\r\n\r\n if (this._pipelineContext.isAsync) {\r\n this._checkIsReady(previousPipelineContext);\r\n }\r\n } catch (e) {\r\n this._processCompilationErrors(e, previousPipelineContext);\r\n }\r\n }\r\n\r\n private _getShaderCodeAndErrorLine(code: Nullable, error: Nullable, isFragment: boolean): [Nullable, Nullable] {\r\n const regexp = isFragment ? /FRAGMENT SHADER ERROR: 0:(\\d+?):/ : /VERTEX SHADER ERROR: 0:(\\d+?):/;\r\n\r\n let errorLine = null;\r\n\r\n if (error && code) {\r\n const res = error.match(regexp);\r\n if (res && res.length === 2) {\r\n const lineNumber = parseInt(res[1]);\r\n const lines = code.split(\"\\n\", -1);\r\n if (lines.length >= lineNumber) {\r\n errorLine = `Offending line [${lineNumber}] in ${isFragment ? \"fragment\" : \"vertex\"} code: ${lines[lineNumber - 1]}`;\r\n }\r\n }\r\n }\r\n\r\n return [code, errorLine];\r\n }\r\n\r\n private _processCompilationErrors(e: any, previousPipelineContext: Nullable = null) {\r\n this._compilationError = e.message;\r\n const attributesNames = this._attributesNames;\r\n const fallbacks = this._fallbacks;\r\n\r\n // Let's go through fallbacks then\r\n Logger.Error(\"Unable to compile effect:\");\r\n Logger.Error(\r\n \"Uniforms: \" +\r\n this._uniformsNames.map(function (uniform) {\r\n return \" \" + uniform;\r\n })\r\n );\r\n Logger.Error(\r\n \"Attributes: \" +\r\n attributesNames.map(function (attribute) {\r\n return \" \" + attribute;\r\n })\r\n );\r\n Logger.Error(\"Defines:\\n\" + this.defines);\r\n if (Effect.LogShaderCodeOnCompilationError) {\r\n let lineErrorVertex = null,\r\n lineErrorFragment = null,\r\n code = null;\r\n if (this._pipelineContext?._getVertexShaderCode()) {\r\n [code, lineErrorVertex] = this._getShaderCodeAndErrorLine(this._pipelineContext._getVertexShaderCode(), this._compilationError, false);\r\n if (code) {\r\n Logger.Error(\"Vertex code:\");\r\n Logger.Error(code);\r\n }\r\n }\r\n if (this._pipelineContext?._getFragmentShaderCode()) {\r\n [code, lineErrorFragment] = this._getShaderCodeAndErrorLine(this._pipelineContext?._getFragmentShaderCode(), this._compilationError, true);\r\n if (code) {\r\n Logger.Error(\"Fragment code:\");\r\n Logger.Error(code);\r\n }\r\n }\r\n if (lineErrorVertex) {\r\n Logger.Error(lineErrorVertex);\r\n }\r\n if (lineErrorFragment) {\r\n Logger.Error(lineErrorFragment);\r\n }\r\n }\r\n Logger.Error(\"Error: \" + this._compilationError);\r\n\r\n const notifyErrors = () => {\r\n if (this.onError) {\r\n this.onError(this, this._compilationError);\r\n }\r\n this.onErrorObservable.notifyObservers(this);\r\n };\r\n\r\n // In case a previous compilation was successful, we need to restore the previous pipeline context\r\n if (previousPipelineContext) {\r\n this._pipelineContext = previousPipelineContext;\r\n this._isReady = true;\r\n notifyErrors();\r\n }\r\n\r\n // Lets try to compile fallbacks as long as we have some.\r\n if (fallbacks) {\r\n this._pipelineContext = null;\r\n if (fallbacks.hasMoreFallbacks) {\r\n this._allFallbacksProcessed = false;\r\n Logger.Error(\"Trying next fallback.\");\r\n this.defines = fallbacks.reduce(this.defines, this);\r\n this._prepareEffect();\r\n } else {\r\n // Sorry we did everything we can\r\n this._allFallbacksProcessed = true;\r\n notifyErrors();\r\n this.onErrorObservable.clear();\r\n\r\n // Unbind mesh reference in fallbacks\r\n if (this._fallbacks) {\r\n this._fallbacks.unBindMesh();\r\n }\r\n }\r\n } else {\r\n this._allFallbacksProcessed = true;\r\n\r\n // In case of error, without any prior successful compilation, let s notify observers\r\n if (!previousPipelineContext) {\r\n notifyErrors();\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Checks if the effect is supported. (Must be called after compilation)\r\n */\r\n public get isSupported(): boolean {\r\n return this._compilationError === \"\";\r\n }\r\n\r\n /**\r\n * Binds a texture to the engine to be used as output of the shader.\r\n * @param channel Name of the output variable.\r\n * @param texture Texture to bind.\r\n * @internal\r\n */\r\n public _bindTexture(channel: string, texture: Nullable): void {\r\n this._engine._bindTexture(this._samplers[channel], texture, channel);\r\n }\r\n\r\n /**\r\n * Sets a texture on the engine to be used in the shader.\r\n * @param channel Name of the sampler variable.\r\n * @param texture Texture to set.\r\n */\r\n public setTexture(channel: string, texture: Nullable): void {\r\n this._engine.setTexture(this._samplers[channel], this._uniforms[channel], texture, channel);\r\n }\r\n\r\n /**\r\n * Sets a depth stencil texture from a render target on the engine to be used in the shader.\r\n * @param channel Name of the sampler variable.\r\n * @param texture Texture to set.\r\n */\r\n public setDepthStencilTexture(channel: string, texture: Nullable): void {\r\n this._engine.setDepthStencilTexture(this._samplers[channel], this._uniforms[channel], texture, channel);\r\n }\r\n\r\n /**\r\n * Sets an array of textures on the engine to be used in the shader.\r\n * @param channel Name of the variable.\r\n * @param textures Textures to set.\r\n */\r\n public setTextureArray(channel: string, textures: ThinTexture[]): void {\r\n const exName = channel + \"Ex\";\r\n if (this._samplerList.indexOf(exName + \"0\") === -1) {\r\n const initialPos = this._samplerList.indexOf(channel);\r\n for (let index = 1; index < textures.length; index++) {\r\n const currentExName = exName + (index - 1).toString();\r\n this._samplerList.splice(initialPos + index, 0, currentExName);\r\n }\r\n\r\n // Reset every channels\r\n let channelIndex = 0;\r\n for (const key of this._samplerList) {\r\n this._samplers[key] = channelIndex;\r\n channelIndex += 1;\r\n }\r\n }\r\n\r\n this._engine.setTextureArray(this._samplers[channel], this._uniforms[channel], textures, channel);\r\n }\r\n\r\n /**\r\n * Sets a texture to be the input of the specified post process. (To use the output, pass in the next post process in the pipeline)\r\n * @param channel Name of the sampler variable.\r\n * @param postProcess Post process to get the input texture from.\r\n */\r\n public setTextureFromPostProcess(channel: string, postProcess: Nullable): void {\r\n this._engine.setTextureFromPostProcess(this._samplers[channel], postProcess, channel);\r\n }\r\n\r\n /**\r\n * (Warning! setTextureFromPostProcessOutput may be desired instead)\r\n * Sets the input texture of the passed in post process to be input of this effect. (To use the output of the passed in post process use setTextureFromPostProcessOutput)\r\n * @param channel Name of the sampler variable.\r\n * @param postProcess Post process to get the output texture from.\r\n */\r\n public setTextureFromPostProcessOutput(channel: string, postProcess: Nullable): void {\r\n this._engine.setTextureFromPostProcessOutput(this._samplers[channel], postProcess, channel);\r\n }\r\n\r\n /**\r\n * Binds a buffer to a uniform.\r\n * @param buffer Buffer to bind.\r\n * @param name Name of the uniform variable to bind to.\r\n */\r\n public bindUniformBuffer(buffer: DataBuffer, name: string): void {\r\n const bufferName = this._uniformBuffersNames[name];\r\n if (bufferName === undefined || (Effect._BaseCache[bufferName] === buffer && this._engine._features.useUBOBindingCache)) {\r\n return;\r\n }\r\n Effect._BaseCache[bufferName] = buffer;\r\n this._engine.bindUniformBufferBase(buffer, bufferName, name);\r\n }\r\n\r\n /**\r\n * Binds block to a uniform.\r\n * @param blockName Name of the block to bind.\r\n * @param index Index to bind.\r\n */\r\n public bindUniformBlock(blockName: string, index: number): void {\r\n this._engine.bindUniformBlock(this._pipelineContext!, blockName, index);\r\n }\r\n\r\n /**\r\n * Sets an integer value on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param value Value to be set.\r\n * @returns this effect.\r\n */\r\n public setInt(uniformName: string, value: number): Effect {\r\n this._pipelineContext!.setInt(uniformName, value);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an int2 value on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param x First int in int2.\r\n * @param y Second int in int2.\r\n * @returns this effect.\r\n */\r\n public setInt2(uniformName: string, x: number, y: number): Effect {\r\n this._pipelineContext!.setInt2(uniformName, x, y);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an int3 value on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param x First int in int3.\r\n * @param y Second int in int3.\r\n * @param z Third int in int3.\r\n * @returns this effect.\r\n */\r\n public setInt3(uniformName: string, x: number, y: number, z: number): Effect {\r\n this._pipelineContext!.setInt3(uniformName, x, y, z);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an int4 value on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param x First int in int4.\r\n * @param y Second int in int4.\r\n * @param z Third int in int4.\r\n * @param w Fourth int in int4.\r\n * @returns this effect.\r\n */\r\n public setInt4(uniformName: string, x: number, y: number, z: number, w: number): Effect {\r\n this._pipelineContext!.setInt4(uniformName, x, y, z, w);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an int array on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n * @returns this effect.\r\n */\r\n public setIntArray(uniformName: string, array: Int32Array): Effect {\r\n this._pipelineContext!.setIntArray(uniformName, array);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an int array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n * @returns this effect.\r\n */\r\n public setIntArray2(uniformName: string, array: Int32Array): Effect {\r\n this._pipelineContext!.setIntArray2(uniformName, array);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an int array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n * @returns this effect.\r\n */\r\n public setIntArray3(uniformName: string, array: Int32Array): Effect {\r\n this._pipelineContext!.setIntArray3(uniformName, array);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an int array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n * @returns this effect.\r\n */\r\n public setIntArray4(uniformName: string, array: Int32Array): Effect {\r\n this._pipelineContext!.setIntArray4(uniformName, array);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an unsigned integer value on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param value Value to be set.\r\n * @returns this effect.\r\n */\r\n public setUInt(uniformName: string, value: number): Effect {\r\n this._pipelineContext!.setInt(uniformName, value);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an unsigned int2 value on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param x First unsigned int in uint2.\r\n * @param y Second unsigned int in uint2.\r\n * @returns this effect.\r\n */\r\n public setUInt2(uniformName: string, x: number, y: number): Effect {\r\n this._pipelineContext!.setInt2(uniformName, x, y);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an unsigned int3 value on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param x First unsigned int in uint3.\r\n * @param y Second unsigned int in uint3.\r\n * @param z Third unsigned int in uint3.\r\n * @returns this effect.\r\n */\r\n public setUInt3(uniformName: string, x: number, y: number, z: number): Effect {\r\n this._pipelineContext!.setInt3(uniformName, x, y, z);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an unsigned int4 value on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param x First unsigned int in uint4.\r\n * @param y Second unsigned int in uint4.\r\n * @param z Third unsigned int in uint4.\r\n * @param w Fourth unsigned int in uint4.\r\n * @returns this effect.\r\n */\r\n public setUInt4(uniformName: string, x: number, y: number, z: number, w: number): Effect {\r\n this._pipelineContext!.setInt4(uniformName, x, y, z, w);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an unsigned int array on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n * @returns this effect.\r\n */\r\n public setUIntArray(uniformName: string, array: Uint32Array): Effect {\r\n this._pipelineContext!.setUIntArray(uniformName, array);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an unsigned int array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n * @returns this effect.\r\n */\r\n public setUIntArray2(uniformName: string, array: Uint32Array): Effect {\r\n this._pipelineContext!.setUIntArray2(uniformName, array);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an unsigned int array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n * @returns this effect.\r\n */\r\n public setUIntArray3(uniformName: string, array: Uint32Array): Effect {\r\n this._pipelineContext!.setUIntArray3(uniformName, array);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an unsigned int array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n * @returns this effect.\r\n */\r\n public setUIntArray4(uniformName: string, array: Uint32Array): Effect {\r\n this._pipelineContext!.setUIntArray4(uniformName, array);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an float array on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n * @returns this effect.\r\n */\r\n public setFloatArray(uniformName: string, array: FloatArray): Effect {\r\n this._pipelineContext!.setArray(uniformName, array);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an float array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n * @returns this effect.\r\n */\r\n public setFloatArray2(uniformName: string, array: FloatArray): Effect {\r\n this._pipelineContext!.setArray2(uniformName, array);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an float array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n * @returns this effect.\r\n */\r\n public setFloatArray3(uniformName: string, array: FloatArray): Effect {\r\n this._pipelineContext!.setArray3(uniformName, array);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an float array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n * @returns this effect.\r\n */\r\n public setFloatArray4(uniformName: string, array: FloatArray): Effect {\r\n this._pipelineContext!.setArray4(uniformName, array);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an array on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n * @returns this effect.\r\n */\r\n public setArray(uniformName: string, array: number[]): Effect {\r\n this._pipelineContext!.setArray(uniformName, array);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n * @returns this effect.\r\n */\r\n public setArray2(uniformName: string, array: number[]): Effect {\r\n this._pipelineContext!.setArray2(uniformName, array);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n * @returns this effect.\r\n */\r\n public setArray3(uniformName: string, array: number[]): Effect {\r\n this._pipelineContext!.setArray3(uniformName, array);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets an array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n * @returns this effect.\r\n */\r\n public setArray4(uniformName: string, array: number[]): Effect {\r\n this._pipelineContext!.setArray4(uniformName, array);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets matrices on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param matrices matrices to be set.\r\n * @returns this effect.\r\n */\r\n public setMatrices(uniformName: string, matrices: Float32Array | Array): Effect {\r\n this._pipelineContext!.setMatrices(uniformName, matrices as Float32Array);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets matrix on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param matrix matrix to be set.\r\n * @returns this effect.\r\n */\r\n public setMatrix(uniformName: string, matrix: IMatrixLike): Effect {\r\n this._pipelineContext!.setMatrix(uniformName, matrix);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets a 3x3 matrix on a uniform variable. (Specified as [1,2,3,4,5,6,7,8,9] will result in [1,2,3][4,5,6][7,8,9] matrix)\r\n * @param uniformName Name of the variable.\r\n * @param matrix matrix to be set.\r\n * @returns this effect.\r\n */\r\n public setMatrix3x3(uniformName: string, matrix: Float32Array | Array): Effect {\r\n // the cast is ok because it is gl.uniformMatrix3fv() which is called at the end, and this function accepts Float32Array and Array\r\n this._pipelineContext!.setMatrix3x3(uniformName, matrix as Float32Array);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets a 2x2 matrix on a uniform variable. (Specified as [1,2,3,4] will result in [1,2][3,4] matrix)\r\n * @param uniformName Name of the variable.\r\n * @param matrix matrix to be set.\r\n * @returns this effect.\r\n */\r\n public setMatrix2x2(uniformName: string, matrix: Float32Array | Array): Effect {\r\n // the cast is ok because it is gl.uniformMatrix3fv() which is called at the end, and this function accepts Float32Array and Array\r\n this._pipelineContext!.setMatrix2x2(uniformName, matrix as Float32Array);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets a float on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param value value to be set.\r\n * @returns this effect.\r\n */\r\n public setFloat(uniformName: string, value: number): Effect {\r\n this._pipelineContext!.setFloat(uniformName, value);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets a boolean on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param bool value to be set.\r\n * @returns this effect.\r\n */\r\n public setBool(uniformName: string, bool: boolean): Effect {\r\n this._pipelineContext!.setInt(uniformName, bool ? 1 : 0);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets a Vector2 on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param vector2 vector2 to be set.\r\n * @returns this effect.\r\n */\r\n public setVector2(uniformName: string, vector2: IVector2Like): Effect {\r\n this._pipelineContext!.setVector2(uniformName, vector2);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets a float2 on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param x First float in float2.\r\n * @param y Second float in float2.\r\n * @returns this effect.\r\n */\r\n public setFloat2(uniformName: string, x: number, y: number): Effect {\r\n this._pipelineContext!.setFloat2(uniformName, x, y);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets a Vector3 on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param vector3 Value to be set.\r\n * @returns this effect.\r\n */\r\n public setVector3(uniformName: string, vector3: IVector3Like): Effect {\r\n this._pipelineContext!.setVector3(uniformName, vector3);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets a float3 on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param x First float in float3.\r\n * @param y Second float in float3.\r\n * @param z Third float in float3.\r\n * @returns this effect.\r\n */\r\n public setFloat3(uniformName: string, x: number, y: number, z: number): Effect {\r\n this._pipelineContext!.setFloat3(uniformName, x, y, z);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets a Vector4 on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param vector4 Value to be set.\r\n * @returns this effect.\r\n */\r\n public setVector4(uniformName: string, vector4: IVector4Like): Effect {\r\n this._pipelineContext!.setVector4(uniformName, vector4);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets a Quaternion on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param quaternion Value to be set.\r\n * @returns this effect.\r\n */\r\n public setQuaternion(uniformName: string, quaternion: IQuaternionLike): Effect {\r\n this._pipelineContext!.setQuaternion(uniformName, quaternion);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets a float4 on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param x First float in float4.\r\n * @param y Second float in float4.\r\n * @param z Third float in float4.\r\n * @param w Fourth float in float4.\r\n * @returns this effect.\r\n */\r\n public setFloat4(uniformName: string, x: number, y: number, z: number, w: number): Effect {\r\n this._pipelineContext!.setFloat4(uniformName, x, y, z, w);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets a Color3 on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param color3 Value to be set.\r\n * @returns this effect.\r\n */\r\n public setColor3(uniformName: string, color3: IColor3Like): Effect {\r\n this._pipelineContext!.setColor3(uniformName, color3);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets a Color4 on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param color3 Value to be set.\r\n * @param alpha Alpha value to be set.\r\n * @returns this effect.\r\n */\r\n public setColor4(uniformName: string, color3: IColor3Like, alpha: number): Effect {\r\n this._pipelineContext!.setColor4(uniformName, color3, alpha);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets a Color4 on a uniform variable\r\n * @param uniformName defines the name of the variable\r\n * @param color4 defines the value to be set\r\n * @returns this effect.\r\n */\r\n public setDirectColor4(uniformName: string, color4: IColor4Like): Effect {\r\n this._pipelineContext!.setDirectColor4(uniformName, color4);\r\n return this;\r\n }\r\n\r\n /**\r\n * Release all associated resources.\r\n **/\r\n public dispose() {\r\n if (this._pipelineContext) {\r\n this._pipelineContext.dispose();\r\n }\r\n this._engine._releaseEffect(this);\r\n\r\n this._isDisposed = true;\r\n }\r\n\r\n /**\r\n * This function will add a new shader to the shader store\r\n * @param name the name of the shader\r\n * @param pixelShader optional pixel shader content\r\n * @param vertexShader optional vertex shader content\r\n * @param shaderLanguage the language the shader is written in (default: GLSL)\r\n */\r\n public static RegisterShader(name: string, pixelShader?: string, vertexShader?: string, shaderLanguage = ShaderLanguage.GLSL) {\r\n if (pixelShader) {\r\n EngineShaderStore.GetShadersStore(shaderLanguage)[`${name}PixelShader`] = pixelShader;\r\n }\r\n\r\n if (vertexShader) {\r\n EngineShaderStore.GetShadersStore(shaderLanguage)[`${name}VertexShader`] = vertexShader;\r\n }\r\n }\r\n\r\n /**\r\n * Store of each shader (The can be looked up using effect.key)\r\n */\r\n public static ShadersStore: { [key: string]: string } = EngineShaderStore.ShadersStore;\r\n /**\r\n * Store of each included file for a shader (The can be looked up using effect.key)\r\n */\r\n public static IncludesShadersStore: { [key: string]: string } = EngineShaderStore.IncludesShadersStore;\r\n\r\n /**\r\n * Resets the cache of effects.\r\n */\r\n public static ResetCache() {\r\n Effect._BaseCache = {};\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\n\r\n/**\r\n * @internal\r\n **/\r\nexport class DepthCullingState {\r\n protected _isDepthTestDirty = false;\r\n protected _isDepthMaskDirty = false;\r\n protected _isDepthFuncDirty = false;\r\n protected _isCullFaceDirty = false;\r\n protected _isCullDirty = false;\r\n protected _isZOffsetDirty = false;\r\n protected _isFrontFaceDirty = false;\r\n\r\n protected _depthTest: boolean;\r\n protected _depthMask: boolean;\r\n protected _depthFunc: Nullable;\r\n protected _cull: Nullable;\r\n protected _cullFace: Nullable;\r\n protected _zOffset: number;\r\n protected _zOffsetUnits: number;\r\n protected _frontFace: Nullable;\r\n\r\n /**\r\n * Initializes the state.\r\n * @param reset\r\n */\r\n public constructor(reset = true) {\r\n if (reset) {\r\n this.reset();\r\n }\r\n }\r\n\r\n public get isDirty(): boolean {\r\n return (\r\n this._isDepthFuncDirty ||\r\n this._isDepthTestDirty ||\r\n this._isDepthMaskDirty ||\r\n this._isCullFaceDirty ||\r\n this._isCullDirty ||\r\n this._isZOffsetDirty ||\r\n this._isFrontFaceDirty\r\n );\r\n }\r\n\r\n public get zOffset(): number {\r\n return this._zOffset;\r\n }\r\n\r\n public set zOffset(value: number) {\r\n if (this._zOffset === value) {\r\n return;\r\n }\r\n\r\n this._zOffset = value;\r\n this._isZOffsetDirty = true;\r\n }\r\n\r\n public get zOffsetUnits(): number {\r\n return this._zOffsetUnits;\r\n }\r\n\r\n public set zOffsetUnits(value: number) {\r\n if (this._zOffsetUnits === value) {\r\n return;\r\n }\r\n\r\n this._zOffsetUnits = value;\r\n this._isZOffsetDirty = true;\r\n }\r\n\r\n public get cullFace(): Nullable {\r\n return this._cullFace;\r\n }\r\n\r\n public set cullFace(value: Nullable) {\r\n if (this._cullFace === value) {\r\n return;\r\n }\r\n\r\n this._cullFace = value;\r\n this._isCullFaceDirty = true;\r\n }\r\n\r\n public get cull(): Nullable {\r\n return this._cull;\r\n }\r\n\r\n public set cull(value: Nullable) {\r\n if (this._cull === value) {\r\n return;\r\n }\r\n\r\n this._cull = value;\r\n this._isCullDirty = true;\r\n }\r\n\r\n public get depthFunc(): Nullable {\r\n return this._depthFunc;\r\n }\r\n\r\n public set depthFunc(value: Nullable) {\r\n if (this._depthFunc === value) {\r\n return;\r\n }\r\n\r\n this._depthFunc = value;\r\n this._isDepthFuncDirty = true;\r\n }\r\n\r\n public get depthMask(): boolean {\r\n return this._depthMask;\r\n }\r\n\r\n public set depthMask(value: boolean) {\r\n if (this._depthMask === value) {\r\n return;\r\n }\r\n\r\n this._depthMask = value;\r\n this._isDepthMaskDirty = true;\r\n }\r\n\r\n public get depthTest(): boolean {\r\n return this._depthTest;\r\n }\r\n\r\n public set depthTest(value: boolean) {\r\n if (this._depthTest === value) {\r\n return;\r\n }\r\n\r\n this._depthTest = value;\r\n this._isDepthTestDirty = true;\r\n }\r\n\r\n public get frontFace(): Nullable {\r\n return this._frontFace;\r\n }\r\n\r\n public set frontFace(value: Nullable) {\r\n if (this._frontFace === value) {\r\n return;\r\n }\r\n\r\n this._frontFace = value;\r\n this._isFrontFaceDirty = true;\r\n }\r\n\r\n public reset() {\r\n this._depthMask = true;\r\n this._depthTest = true;\r\n this._depthFunc = null;\r\n this._cullFace = null;\r\n this._cull = null;\r\n this._zOffset = 0;\r\n this._zOffsetUnits = 0;\r\n this._frontFace = null;\r\n\r\n this._isDepthTestDirty = true;\r\n this._isDepthMaskDirty = true;\r\n this._isDepthFuncDirty = false;\r\n this._isCullFaceDirty = false;\r\n this._isCullDirty = false;\r\n this._isZOffsetDirty = true;\r\n this._isFrontFaceDirty = false;\r\n }\r\n\r\n public apply(gl: WebGLRenderingContext) {\r\n if (!this.isDirty) {\r\n return;\r\n }\r\n\r\n // Cull\r\n if (this._isCullDirty) {\r\n if (this.cull) {\r\n gl.enable(gl.CULL_FACE);\r\n } else {\r\n gl.disable(gl.CULL_FACE);\r\n }\r\n\r\n this._isCullDirty = false;\r\n }\r\n\r\n // Cull face\r\n if (this._isCullFaceDirty) {\r\n gl.cullFace(this.cullFace);\r\n this._isCullFaceDirty = false;\r\n }\r\n\r\n // Depth mask\r\n if (this._isDepthMaskDirty) {\r\n gl.depthMask(this.depthMask);\r\n this._isDepthMaskDirty = false;\r\n }\r\n\r\n // Depth test\r\n if (this._isDepthTestDirty) {\r\n if (this.depthTest) {\r\n gl.enable(gl.DEPTH_TEST);\r\n } else {\r\n gl.disable(gl.DEPTH_TEST);\r\n }\r\n this._isDepthTestDirty = false;\r\n }\r\n\r\n // Depth func\r\n if (this._isDepthFuncDirty) {\r\n gl.depthFunc(this.depthFunc);\r\n this._isDepthFuncDirty = false;\r\n }\r\n\r\n // zOffset\r\n if (this._isZOffsetDirty) {\r\n if (this.zOffset || this.zOffsetUnits) {\r\n gl.enable(gl.POLYGON_OFFSET_FILL);\r\n gl.polygonOffset(this.zOffset, this.zOffsetUnits);\r\n } else {\r\n gl.disable(gl.POLYGON_OFFSET_FILL);\r\n }\r\n\r\n this._isZOffsetDirty = false;\r\n }\r\n\r\n // Front face\r\n if (this._isFrontFaceDirty) {\r\n gl.frontFace(this.frontFace);\r\n this._isFrontFaceDirty = false;\r\n }\r\n }\r\n}\r\n","import { Constants } from \"../Engines/constants\";\r\nimport type { IStencilState } from \"./IStencilState\";\r\n\r\n/**\r\n * @internal\r\n **/\r\nexport class StencilState implements IStencilState {\r\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will always pass. i.e. Pixels will be drawn in the order they are drawn */\r\n public static readonly ALWAYS = Constants.ALWAYS;\r\n /** Passed to stencilOperation to specify that stencil value must be kept */\r\n public static readonly KEEP = Constants.KEEP;\r\n /** Passed to stencilOperation to specify that stencil value must be replaced */\r\n public static readonly REPLACE = Constants.REPLACE;\r\n\r\n public constructor() {\r\n this.reset();\r\n }\r\n\r\n public reset() {\r\n this.enabled = false;\r\n this.mask = 0xff;\r\n\r\n this.func = StencilState.ALWAYS;\r\n this.funcRef = 1;\r\n this.funcMask = 0xff;\r\n\r\n this.opStencilFail = StencilState.KEEP;\r\n this.opDepthFail = StencilState.KEEP;\r\n this.opStencilDepthPass = StencilState.REPLACE;\r\n }\r\n\r\n public func: number;\r\n public get stencilFunc(): number {\r\n return this.func;\r\n }\r\n\r\n public set stencilFunc(value: number) {\r\n this.func = value;\r\n }\r\n\r\n public funcRef: number;\r\n public get stencilFuncRef(): number {\r\n return this.funcRef;\r\n }\r\n\r\n public set stencilFuncRef(value: number) {\r\n this.funcRef = value;\r\n }\r\n\r\n public funcMask: number;\r\n public get stencilFuncMask(): number {\r\n return this.funcMask;\r\n }\r\n\r\n public set stencilFuncMask(value: number) {\r\n this.funcMask = value;\r\n }\r\n\r\n public opStencilFail: number;\r\n public get stencilOpStencilFail(): number {\r\n return this.opStencilFail;\r\n }\r\n\r\n public set stencilOpStencilFail(value: number) {\r\n this.opStencilFail = value;\r\n }\r\n\r\n public opDepthFail: number;\r\n public get stencilOpDepthFail(): number {\r\n return this.opDepthFail;\r\n }\r\n\r\n public set stencilOpDepthFail(value: number) {\r\n this.opDepthFail = value;\r\n }\r\n\r\n public opStencilDepthPass: number;\r\n public get stencilOpStencilDepthPass(): number {\r\n return this.opStencilDepthPass;\r\n }\r\n\r\n public set stencilOpStencilDepthPass(value: number) {\r\n this.opStencilDepthPass = value;\r\n }\r\n\r\n public mask: number;\r\n public get stencilMask(): number {\r\n return this.mask;\r\n }\r\n\r\n public set stencilMask(value: number) {\r\n this.mask = value;\r\n }\r\n\r\n public enabled: boolean;\r\n public get stencilTest(): boolean {\r\n return this.enabled;\r\n }\r\n\r\n public set stencilTest(value: boolean) {\r\n this.enabled = value;\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\n\r\n/**\r\n * @internal\r\n **/\r\nexport class AlphaState {\r\n public _blendFunctionParameters = new Array>(4);\r\n public _blendEquationParameters = new Array>(2);\r\n public _blendConstants = new Array>(4);\r\n public _isBlendConstantsDirty = false;\r\n\r\n private _alphaBlend = false;\r\n private _isAlphaBlendDirty = false;\r\n private _isBlendFunctionParametersDirty = false;\r\n private _isBlendEquationParametersDirty = false;\r\n\r\n /**\r\n * Initializes the state.\r\n */\r\n public constructor() {\r\n this.reset();\r\n }\r\n\r\n public get isDirty(): boolean {\r\n return this._isAlphaBlendDirty || this._isBlendFunctionParametersDirty || this._isBlendEquationParametersDirty;\r\n }\r\n\r\n public get alphaBlend(): boolean {\r\n return this._alphaBlend;\r\n }\r\n\r\n public set alphaBlend(value: boolean) {\r\n if (this._alphaBlend === value) {\r\n return;\r\n }\r\n\r\n this._alphaBlend = value;\r\n this._isAlphaBlendDirty = true;\r\n }\r\n\r\n public setAlphaBlendConstants(r: number, g: number, b: number, a: number): void {\r\n if (this._blendConstants[0] === r && this._blendConstants[1] === g && this._blendConstants[2] === b && this._blendConstants[3] === a) {\r\n return;\r\n }\r\n\r\n this._blendConstants[0] = r;\r\n this._blendConstants[1] = g;\r\n this._blendConstants[2] = b;\r\n this._blendConstants[3] = a;\r\n\r\n this._isBlendConstantsDirty = true;\r\n }\r\n\r\n public setAlphaBlendFunctionParameters(value0: number, value1: number, value2: number, value3: number): void {\r\n if (\r\n this._blendFunctionParameters[0] === value0 &&\r\n this._blendFunctionParameters[1] === value1 &&\r\n this._blendFunctionParameters[2] === value2 &&\r\n this._blendFunctionParameters[3] === value3\r\n ) {\r\n return;\r\n }\r\n\r\n this._blendFunctionParameters[0] = value0;\r\n this._blendFunctionParameters[1] = value1;\r\n this._blendFunctionParameters[2] = value2;\r\n this._blendFunctionParameters[3] = value3;\r\n\r\n this._isBlendFunctionParametersDirty = true;\r\n }\r\n\r\n public setAlphaEquationParameters(rgb: number, alpha: number): void {\r\n if (this._blendEquationParameters[0] === rgb && this._blendEquationParameters[1] === alpha) {\r\n return;\r\n }\r\n\r\n this._blendEquationParameters[0] = rgb;\r\n this._blendEquationParameters[1] = alpha;\r\n\r\n this._isBlendEquationParametersDirty = true;\r\n }\r\n\r\n public reset() {\r\n this._alphaBlend = false;\r\n this._blendFunctionParameters[0] = null;\r\n this._blendFunctionParameters[1] = null;\r\n this._blendFunctionParameters[2] = null;\r\n this._blendFunctionParameters[3] = null;\r\n\r\n this._blendEquationParameters[0] = null;\r\n this._blendEquationParameters[1] = null;\r\n\r\n this._blendConstants[0] = null;\r\n this._blendConstants[1] = null;\r\n this._blendConstants[2] = null;\r\n this._blendConstants[3] = null;\r\n\r\n this._isAlphaBlendDirty = true;\r\n this._isBlendFunctionParametersDirty = false;\r\n this._isBlendEquationParametersDirty = false;\r\n this._isBlendConstantsDirty = false;\r\n }\r\n\r\n public apply(gl: WebGLRenderingContext) {\r\n if (!this.isDirty) {\r\n return;\r\n }\r\n\r\n // Alpha blend\r\n if (this._isAlphaBlendDirty) {\r\n if (this._alphaBlend) {\r\n gl.enable(gl.BLEND);\r\n } else {\r\n gl.disable(gl.BLEND);\r\n }\r\n\r\n this._isAlphaBlendDirty = false;\r\n }\r\n\r\n // Alpha function\r\n if (this._isBlendFunctionParametersDirty) {\r\n gl.blendFuncSeparate(\r\n this._blendFunctionParameters[0],\r\n this._blendFunctionParameters[1],\r\n this._blendFunctionParameters[2],\r\n this._blendFunctionParameters[3]\r\n );\r\n this._isBlendFunctionParametersDirty = false;\r\n }\r\n\r\n // Alpha equation\r\n if (this._isBlendEquationParametersDirty) {\r\n gl.blendEquationSeparate(this._blendEquationParameters[0]!, this._blendEquationParameters[1]!);\r\n this._isBlendEquationParametersDirty = false;\r\n }\r\n\r\n // Constants\r\n if (this._isBlendConstantsDirty) {\r\n gl.blendColor(this._blendConstants[0], this._blendConstants[1], this._blendConstants[2], this._blendConstants[3]);\r\n this._isBlendConstantsDirty = false;\r\n }\r\n }\r\n}\r\n","import { Observable } from \"../../Misc/observable\";\r\nimport type { Nullable, int } from \"../../types\";\r\nimport type { ICanvas, ICanvasRenderingContext } from \"../../Engines/ICanvas\";\r\nimport type { HardwareTextureWrapper } from \"./hardwareTextureWrapper\";\r\nimport { TextureSampler } from \"./textureSampler\";\r\n\r\nimport type { ThinEngine } from \"../../Engines/thinEngine\";\r\nimport type { BaseTexture } from \"../../Materials/Textures/baseTexture\";\r\nimport type { SphericalPolynomial } from \"../../Maths/sphericalPolynomial\";\r\n\r\n/**\r\n * Defines the source of the internal texture\r\n */\r\nexport enum InternalTextureSource {\r\n /**\r\n * The source of the texture data is unknown\r\n */\r\n Unknown,\r\n /**\r\n * Texture data comes from an URL\r\n */\r\n Url,\r\n /**\r\n * Texture data is only used for temporary storage\r\n */\r\n Temp,\r\n /**\r\n * Texture data comes from raw data (ArrayBuffer)\r\n */\r\n Raw,\r\n /**\r\n * Texture content is dynamic (video or dynamic texture)\r\n */\r\n Dynamic,\r\n /**\r\n * Texture content is generated by rendering to it\r\n */\r\n RenderTarget,\r\n /**\r\n * Texture content is part of a multi render target process\r\n */\r\n MultiRenderTarget,\r\n /**\r\n * Texture data comes from a cube data file\r\n */\r\n Cube,\r\n /**\r\n * Texture data comes from a raw cube data\r\n */\r\n CubeRaw,\r\n /**\r\n * Texture data come from a prefiltered cube data file\r\n */\r\n CubePrefiltered,\r\n /**\r\n * Texture content is raw 3D data\r\n */\r\n Raw3D,\r\n /**\r\n * Texture content is raw 2D array data\r\n */\r\n Raw2DArray,\r\n /**\r\n * Texture content is a depth/stencil texture\r\n */\r\n DepthStencil,\r\n /**\r\n * Texture data comes from a raw cube data encoded with RGBD\r\n */\r\n CubeRawRGBD,\r\n /**\r\n * Texture content is a depth texture\r\n */\r\n Depth,\r\n}\r\n\r\n/**\r\n * Class used to store data associated with WebGL texture data for the engine\r\n * This class should not be used directly\r\n */\r\nexport class InternalTexture extends TextureSampler {\r\n /**\r\n * Defines if the texture is ready\r\n */\r\n public isReady: boolean = false;\r\n /**\r\n * Defines if the texture is a cube texture\r\n */\r\n public isCube: boolean = false;\r\n /**\r\n * Defines if the texture contains 3D data\r\n */\r\n public is3D: boolean = false;\r\n /**\r\n * Defines if the texture contains 2D array data\r\n */\r\n public is2DArray: boolean = false;\r\n /**\r\n * Defines if the texture contains multiview data\r\n */\r\n public isMultiview: boolean = false;\r\n /**\r\n * Gets the URL used to load this texture\r\n */\r\n public url: string = \"\";\r\n /** @internal */\r\n public _originalUrl: string; // not empty only if different from url\r\n /**\r\n * Gets a boolean indicating if the texture needs mipmaps generation\r\n */\r\n public generateMipMaps: boolean = false;\r\n /**\r\n * Gets a boolean indicating if the texture uses mipmaps\r\n * TODO implements useMipMaps as a separate setting from generateMipMaps\r\n */\r\n public get useMipMaps() {\r\n return this.generateMipMaps;\r\n }\r\n public set useMipMaps(value: boolean) {\r\n this.generateMipMaps = value;\r\n }\r\n /**\r\n * Gets the number of samples used by the texture (WebGL2+ only)\r\n */\r\n public samples: number = 0;\r\n /**\r\n * Gets the type of the texture (int, float...)\r\n */\r\n public type: number = -1;\r\n /**\r\n * Gets the format of the texture (RGB, RGBA...)\r\n */\r\n public format: number = -1;\r\n /**\r\n * Observable called when the texture is loaded\r\n */\r\n public onLoadedObservable = new Observable();\r\n /**\r\n * Observable called when the texture load is raising an error\r\n */\r\n public onErrorObservable = new Observable>();\r\n /**\r\n * If this callback is defined it will be called instead of the default _rebuild function\r\n */\r\n public onRebuildCallback: Nullable<\r\n (internalTexture: InternalTexture) => {\r\n proxy: Nullable>;\r\n isReady: boolean;\r\n isAsync: boolean;\r\n }\r\n > = null;\r\n /**\r\n * Gets the width of the texture\r\n */\r\n public width: number = 0;\r\n /**\r\n * Gets the height of the texture\r\n */\r\n public height: number = 0;\r\n /**\r\n * Gets the depth of the texture\r\n */\r\n public depth: number = 0;\r\n /**\r\n * Gets the initial width of the texture (It could be rescaled if the current system does not support non power of two textures)\r\n */\r\n public baseWidth: number = 0;\r\n /**\r\n * Gets the initial height of the texture (It could be rescaled if the current system does not support non power of two textures)\r\n */\r\n public baseHeight: number = 0;\r\n /**\r\n * Gets the initial depth of the texture (It could be rescaled if the current system does not support non power of two textures)\r\n */\r\n public baseDepth: number = 0;\r\n /**\r\n * Gets a boolean indicating if the texture is inverted on Y axis\r\n */\r\n public invertY: boolean = false;\r\n /**\r\n * Used for debugging purpose only\r\n */\r\n public label?: string;\r\n\r\n // Private\r\n /** @internal */\r\n public _invertVScale = false;\r\n /** @internal */\r\n public _associatedChannel = -1;\r\n /** @internal */\r\n public _source = InternalTextureSource.Unknown;\r\n /** @internal */\r\n public _buffer: Nullable = null;\r\n /** @internal */\r\n public _bufferView: Nullable = null;\r\n /** @internal */\r\n public _bufferViewArray: Nullable = null;\r\n /** @internal */\r\n public _bufferViewArrayArray: Nullable = null;\r\n /** @internal */\r\n public _size: number = 0;\r\n /** @internal */\r\n public _extension: string = \"\";\r\n /** @internal */\r\n public _files: Nullable = null;\r\n /** @internal */\r\n public _workingCanvas: Nullable = null;\r\n /** @internal */\r\n public _workingContext: Nullable = null;\r\n /** @internal */\r\n public _cachedCoordinatesMode: Nullable = null;\r\n /** @internal */\r\n public _isDisabled: boolean = false;\r\n /** @internal */\r\n public _compression: Nullable = null;\r\n /** @internal */\r\n public _sphericalPolynomial: Nullable = null;\r\n /** @internal */\r\n public _sphericalPolynomialPromise: Nullable> = null;\r\n /** @internal */\r\n public _sphericalPolynomialComputed = false;\r\n /** @internal */\r\n public _lodGenerationScale: number = 0;\r\n /** @internal */\r\n public _lodGenerationOffset: number = 0;\r\n /** @internal */\r\n public _useSRGBBuffer: boolean = false;\r\n\r\n // The following three fields helps sharing generated fixed LODs for texture filtering\r\n // In environment not supporting the textureLOD extension like EDGE. They are for internal use only.\r\n // They are at the level of the gl texture to benefit from the cache.\r\n /** @internal */\r\n public _lodTextureHigh: Nullable = null;\r\n /** @internal */\r\n public _lodTextureMid: Nullable = null;\r\n /** @internal */\r\n public _lodTextureLow: Nullable = null;\r\n /** @internal */\r\n public _isRGBD: boolean = false;\r\n\r\n /** @internal */\r\n public _linearSpecularLOD: boolean = false;\r\n /** @internal */\r\n public _irradianceTexture: Nullable = null;\r\n\r\n /** @internal */\r\n public _hardwareTexture: Nullable = null;\r\n\r\n /** @internal */\r\n public _maxLodLevel: Nullable = null;\r\n\r\n /** @internal */\r\n public _references: number = 1;\r\n\r\n /** @internal */\r\n public _gammaSpace: Nullable = null;\r\n\r\n private _engine: ThinEngine;\r\n private _uniqueId: number;\r\n\r\n /** @internal */\r\n public static _Counter = 0;\r\n\r\n /** Gets the unique id of the internal texture */\r\n public get uniqueId() {\r\n return this._uniqueId;\r\n }\r\n\r\n /** @internal */\r\n public _setUniqueId(id: number) {\r\n this._uniqueId = id;\r\n }\r\n\r\n /**\r\n * Gets the Engine the texture belongs to.\r\n * @returns The babylon engine\r\n */\r\n public getEngine(): ThinEngine {\r\n return this._engine;\r\n }\r\n\r\n /**\r\n * Gets the data source type of the texture\r\n */\r\n public get source(): InternalTextureSource {\r\n return this._source;\r\n }\r\n\r\n /**\r\n * Creates a new InternalTexture\r\n * @param engine defines the engine to use\r\n * @param source defines the type of data that will be used\r\n * @param delayAllocation if the texture allocation should be delayed (default: false)\r\n */\r\n constructor(engine: ThinEngine, source: InternalTextureSource, delayAllocation = false) {\r\n super();\r\n\r\n this._engine = engine;\r\n this._source = source;\r\n this._uniqueId = InternalTexture._Counter++;\r\n\r\n if (!delayAllocation) {\r\n this._hardwareTexture = engine._createHardwareTexture();\r\n }\r\n }\r\n\r\n /**\r\n * Increments the number of references (ie. the number of Texture that point to it)\r\n */\r\n public incrementReferences(): void {\r\n this._references++;\r\n }\r\n\r\n /**\r\n * Change the size of the texture (not the size of the content)\r\n * @param width defines the new width\r\n * @param height defines the new height\r\n * @param depth defines the new depth (1 by default)\r\n */\r\n public updateSize(width: int, height: int, depth: int = 1): void {\r\n this._engine.updateTextureDimensions(this, width, height, depth);\r\n\r\n this.width = width;\r\n this.height = height;\r\n this.depth = depth;\r\n\r\n this.baseWidth = width;\r\n this.baseHeight = height;\r\n this.baseDepth = depth;\r\n\r\n this._size = width * height * depth;\r\n }\r\n\r\n /** @internal */\r\n public _rebuild(): void {\r\n this.isReady = false;\r\n this._cachedCoordinatesMode = null;\r\n this._cachedWrapU = null;\r\n this._cachedWrapV = null;\r\n this._cachedWrapR = null;\r\n this._cachedAnisotropicFilteringLevel = null;\r\n if (this.onRebuildCallback) {\r\n const data = this.onRebuildCallback(this);\r\n const swapAndSetIsReady = (proxyInternalTexture: InternalTexture) => {\r\n proxyInternalTexture._swapAndDie(this, false);\r\n this.isReady = data.isReady;\r\n };\r\n if (data.isAsync) {\r\n (data.proxy as Promise).then(swapAndSetIsReady);\r\n } else {\r\n swapAndSetIsReady(data.proxy as InternalTexture);\r\n }\r\n return;\r\n }\r\n\r\n let proxy: InternalTexture;\r\n switch (this.source) {\r\n case InternalTextureSource.Temp:\r\n break;\r\n\r\n case InternalTextureSource.Url:\r\n proxy = this._engine.createTexture(\r\n this._originalUrl ?? this.url,\r\n !this.generateMipMaps,\r\n this.invertY,\r\n null,\r\n this.samplingMode,\r\n // Do not use Proxy here as it could be fully synchronous\r\n // and proxy would be undefined.\r\n (temp) => {\r\n temp._swapAndDie(this, false);\r\n this.isReady = true;\r\n },\r\n null,\r\n this._buffer,\r\n undefined,\r\n this.format,\r\n this._extension,\r\n undefined,\r\n undefined,\r\n undefined,\r\n this._useSRGBBuffer\r\n );\r\n return;\r\n\r\n case InternalTextureSource.Raw:\r\n proxy = this._engine.createRawTexture(\r\n this._bufferView,\r\n this.baseWidth,\r\n this.baseHeight,\r\n this.format,\r\n this.generateMipMaps,\r\n this.invertY,\r\n this.samplingMode,\r\n this._compression,\r\n this.type,\r\n undefined,\r\n this._useSRGBBuffer\r\n );\r\n proxy._swapAndDie(this, false);\r\n\r\n this.isReady = true;\r\n break;\r\n\r\n case InternalTextureSource.Raw3D:\r\n proxy = this._engine.createRawTexture3D(\r\n this._bufferView,\r\n this.baseWidth,\r\n this.baseHeight,\r\n this.baseDepth,\r\n this.format,\r\n this.generateMipMaps,\r\n this.invertY,\r\n this.samplingMode,\r\n this._compression,\r\n this.type\r\n );\r\n proxy._swapAndDie(this, false);\r\n\r\n this.isReady = true;\r\n break;\r\n\r\n case InternalTextureSource.Raw2DArray:\r\n proxy = this._engine.createRawTexture2DArray(\r\n this._bufferView,\r\n this.baseWidth,\r\n this.baseHeight,\r\n this.baseDepth,\r\n this.format,\r\n this.generateMipMaps,\r\n this.invertY,\r\n this.samplingMode,\r\n this._compression,\r\n this.type\r\n );\r\n proxy._swapAndDie(this, false);\r\n\r\n this.isReady = true;\r\n break;\r\n\r\n case InternalTextureSource.Dynamic:\r\n proxy = this._engine.createDynamicTexture(this.baseWidth, this.baseHeight, this.generateMipMaps, this.samplingMode);\r\n proxy._swapAndDie(this, false);\r\n this._engine.updateDynamicTexture(this, this._engine.getRenderingCanvas()!, this.invertY, undefined, undefined, true);\r\n\r\n // The engine will make sure to update content so no need to flag it as isReady = true\r\n break;\r\n\r\n case InternalTextureSource.Cube:\r\n proxy = this._engine.createCubeTexture(\r\n this.url,\r\n null,\r\n this._files,\r\n !this.generateMipMaps,\r\n () => {\r\n proxy._swapAndDie(this, false);\r\n this.isReady = true;\r\n },\r\n null,\r\n this.format,\r\n this._extension,\r\n false,\r\n 0,\r\n 0,\r\n null,\r\n undefined,\r\n this._useSRGBBuffer\r\n );\r\n return;\r\n\r\n case InternalTextureSource.CubeRaw:\r\n proxy = this._engine.createRawCubeTexture(\r\n this._bufferViewArray!,\r\n this.width,\r\n this.format,\r\n this.type,\r\n this.generateMipMaps,\r\n this.invertY,\r\n this.samplingMode,\r\n this._compression\r\n );\r\n proxy._swapAndDie(this, false);\r\n this.isReady = true;\r\n break;\r\n\r\n case InternalTextureSource.CubeRawRGBD:\r\n // This case is being handeled by the environment texture tools and is not a part of the rebuild process.\r\n // To use CubeRawRGBD use updateRGBDAsync on the cube texture.\r\n return;\r\n\r\n case InternalTextureSource.CubePrefiltered:\r\n proxy = this._engine.createPrefilteredCubeTexture(\r\n this.url,\r\n null,\r\n this._lodGenerationScale,\r\n this._lodGenerationOffset,\r\n (proxy) => {\r\n if (proxy) {\r\n proxy._swapAndDie(this, false);\r\n }\r\n this.isReady = true;\r\n },\r\n null,\r\n this.format,\r\n this._extension\r\n );\r\n proxy._sphericalPolynomial = this._sphericalPolynomial;\r\n return;\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _swapAndDie(target: InternalTexture, swapAll = true): void {\r\n // TODO what about refcount on target?\r\n\r\n this._hardwareTexture?.setUsage(target._source, this.generateMipMaps, this.isCube, this.width, this.height);\r\n\r\n target._hardwareTexture = this._hardwareTexture;\r\n if (swapAll) {\r\n target._isRGBD = this._isRGBD;\r\n }\r\n\r\n if (this._lodTextureHigh) {\r\n if (target._lodTextureHigh) {\r\n target._lodTextureHigh.dispose();\r\n }\r\n target._lodTextureHigh = this._lodTextureHigh;\r\n }\r\n\r\n if (this._lodTextureMid) {\r\n if (target._lodTextureMid) {\r\n target._lodTextureMid.dispose();\r\n }\r\n target._lodTextureMid = this._lodTextureMid;\r\n }\r\n\r\n if (this._lodTextureLow) {\r\n if (target._lodTextureLow) {\r\n target._lodTextureLow.dispose();\r\n }\r\n target._lodTextureLow = this._lodTextureLow;\r\n }\r\n\r\n if (this._irradianceTexture) {\r\n if (target._irradianceTexture) {\r\n target._irradianceTexture.dispose();\r\n }\r\n target._irradianceTexture = this._irradianceTexture;\r\n }\r\n\r\n const cache = this._engine.getLoadedTexturesCache();\r\n let index = cache.indexOf(this);\r\n if (index !== -1) {\r\n cache.splice(index, 1);\r\n }\r\n\r\n index = cache.indexOf(target);\r\n if (index === -1) {\r\n cache.push(target);\r\n }\r\n }\r\n\r\n /**\r\n * Dispose the current allocated resources\r\n */\r\n public dispose(): void {\r\n this._references--;\r\n this.onLoadedObservable.clear();\r\n this.onErrorObservable.clear();\r\n if (this._references === 0) {\r\n this._engine._releaseTexture(this);\r\n this._hardwareTexture = null;\r\n }\r\n }\r\n}\r\n","import { Constants } from \"../../Engines/constants\";\r\nimport type { Nullable } from \"../../types\";\r\n\r\n/**\r\n * Class used to store a texture sampler data\r\n */\r\nexport class TextureSampler {\r\n /**\r\n * Gets the sampling mode of the texture\r\n */\r\n public samplingMode: number = -1;\r\n\r\n /**\r\n * | Value | Type | Description |\r\n * | ----- | ------------------ | ----------- |\r\n * | 0 | CLAMP_ADDRESSMODE | |\r\n * | 1 | WRAP_ADDRESSMODE | |\r\n * | 2 | MIRROR_ADDRESSMODE | |\r\n */\r\n public get wrapU() {\r\n return this._cachedWrapU;\r\n }\r\n\r\n public set wrapU(value: Nullable) {\r\n this._cachedWrapU = value;\r\n }\r\n\r\n /**\r\n * | Value | Type | Description |\r\n * | ----- | ------------------ | ----------- |\r\n * | 0 | CLAMP_ADDRESSMODE | |\r\n * | 1 | WRAP_ADDRESSMODE | |\r\n * | 2 | MIRROR_ADDRESSMODE | |\r\n */\r\n public get wrapV() {\r\n return this._cachedWrapV;\r\n }\r\n\r\n public set wrapV(value: Nullable) {\r\n this._cachedWrapV = value;\r\n }\r\n\r\n /**\r\n * | Value | Type | Description |\r\n * | ----- | ------------------ | ----------- |\r\n * | 0 | CLAMP_ADDRESSMODE | |\r\n * | 1 | WRAP_ADDRESSMODE | |\r\n * | 2 | MIRROR_ADDRESSMODE | |\r\n */\r\n public get wrapR() {\r\n return this._cachedWrapR;\r\n }\r\n\r\n public set wrapR(value: Nullable) {\r\n this._cachedWrapR = value;\r\n }\r\n\r\n /**\r\n * With compliant hardware and browser (supporting anisotropic filtering)\r\n * this defines the level of anisotropic filtering in the texture.\r\n * The higher the better but the slower.\r\n */\r\n public get anisotropicFilteringLevel() {\r\n return this._cachedAnisotropicFilteringLevel;\r\n }\r\n\r\n public set anisotropicFilteringLevel(value: Nullable) {\r\n this._cachedAnisotropicFilteringLevel = value;\r\n }\r\n\r\n /**\r\n * Gets or sets the comparison function (Constants.LESS, Constants.EQUAL, etc). Set 0 to not use a comparison function\r\n */\r\n public get comparisonFunction() {\r\n return this._comparisonFunction;\r\n }\r\n\r\n public set comparisonFunction(value: number) {\r\n this._comparisonFunction = value;\r\n }\r\n\r\n private _useMipMaps = true;\r\n /**\r\n * Indicates to use the mip maps (if available on the texture).\r\n * Thanks to this flag, you can instruct the sampler to not sample the mipmaps even if they exist (and if the sampling mode is set to a value that normally samples the mipmaps!)\r\n */\r\n public get useMipMaps() {\r\n return this._useMipMaps;\r\n }\r\n\r\n public set useMipMaps(value: boolean) {\r\n this._useMipMaps = value;\r\n }\r\n\r\n /** @internal */\r\n public _cachedWrapU: Nullable = null;\r\n\r\n /** @internal */\r\n public _cachedWrapV: Nullable = null;\r\n\r\n /** @internal */\r\n public _cachedWrapR: Nullable = null;\r\n\r\n /** @internal */\r\n public _cachedAnisotropicFilteringLevel: Nullable = null;\r\n\r\n /** @internal */\r\n public _comparisonFunction: number = 0;\r\n\r\n /**\r\n * Creates a Sampler instance\r\n */\r\n constructor() {}\r\n\r\n /**\r\n * Sets all the parameters of the sampler\r\n * @param wrapU u address mode (default: TEXTURE_WRAP_ADDRESSMODE)\r\n * @param wrapV v address mode (default: TEXTURE_WRAP_ADDRESSMODE)\r\n * @param wrapR r address mode (default: TEXTURE_WRAP_ADDRESSMODE)\r\n * @param anisotropicFilteringLevel anisotropic level (default: 1)\r\n * @param samplingMode sampling mode (default: Constants.TEXTURE_BILINEAR_SAMPLINGMODE)\r\n * @param comparisonFunction comparison function (default: 0 - no comparison function)\r\n * @returns the current sampler instance\r\n */\r\n public setParameters(\r\n wrapU = Constants.TEXTURE_WRAP_ADDRESSMODE,\r\n wrapV = Constants.TEXTURE_WRAP_ADDRESSMODE,\r\n wrapR = Constants.TEXTURE_WRAP_ADDRESSMODE,\r\n anisotropicFilteringLevel = 1,\r\n samplingMode = Constants.TEXTURE_BILINEAR_SAMPLINGMODE,\r\n comparisonFunction = 0\r\n ): TextureSampler {\r\n this._cachedWrapU = wrapU;\r\n this._cachedWrapV = wrapV;\r\n this._cachedWrapR = wrapR;\r\n this._cachedAnisotropicFilteringLevel = anisotropicFilteringLevel;\r\n this.samplingMode = samplingMode;\r\n this._comparisonFunction = comparisonFunction;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Compares this sampler with another one\r\n * @param other sampler to compare with\r\n * @returns true if the samplers have the same parametres, else false\r\n */\r\n public compareSampler(other: TextureSampler): boolean {\r\n return (\r\n this._cachedWrapU === other._cachedWrapU &&\r\n this._cachedWrapV === other._cachedWrapV &&\r\n this._cachedWrapR === other._cachedWrapR &&\r\n this._cachedAnisotropicFilteringLevel === other._cachedAnisotropicFilteringLevel &&\r\n this.samplingMode === other.samplingMode &&\r\n this._comparisonFunction === other._comparisonFunction &&\r\n this._useMipMaps === other._useMipMaps\r\n );\r\n }\r\n}\r\n","import { ShaderLanguage } from \"../../Materials/shaderLanguage\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { IShaderProcessor } from \"../Processors/iShaderProcessor\";\r\nimport type { ShaderProcessingContext } from \"../Processors/shaderProcessingOptions\";\r\n\r\nimport type { ThinEngine } from \"../thinEngine\";\r\n\r\n/** @internal */\r\nexport class WebGLShaderProcessor implements IShaderProcessor {\r\n public shaderLanguage = ShaderLanguage.GLSL;\r\n\r\n public postProcessor(code: string, defines: string[], isFragment: boolean, processingContext: Nullable, engine: ThinEngine) {\r\n // Remove extensions\r\n if (!engine.getCaps().drawBuffersExtension) {\r\n // even if enclosed in #if/#endif, IE11 does parse the #extension declaration, so we need to remove it altogether\r\n const regex = /#extension.+GL_EXT_draw_buffers.+(enable|require)/g;\r\n code = code.replace(regex, \"\");\r\n }\r\n\r\n return code;\r\n }\r\n}\r\n","import { ShaderLanguage } from \"../../Materials/shaderLanguage\";\r\nimport type { IShaderProcessor } from \"../Processors/iShaderProcessor\";\r\n\r\nconst varyingRegex = /(flat\\s)?\\s*varying\\s*.*/;\r\n\r\n/** @internal */\r\nexport class WebGL2ShaderProcessor implements IShaderProcessor {\r\n public shaderLanguage = ShaderLanguage.GLSL;\r\n\r\n public attributeProcessor(attribute: string) {\r\n return attribute.replace(\"attribute\", \"in\");\r\n }\r\n\r\n public varyingCheck(varying: string, _isFragment: boolean) {\r\n return varyingRegex.test(varying);\r\n }\r\n\r\n public varyingProcessor(varying: string, isFragment: boolean) {\r\n return varying.replace(\"varying\", isFragment ? \"in\" : \"out\");\r\n }\r\n\r\n public postProcessor(code: string, defines: string[], isFragment: boolean) {\r\n const hasDrawBuffersExtension = code.search(/#extension.+GL_EXT_draw_buffers.+require/) !== -1;\r\n\r\n // Remove extensions\r\n const regex = /#extension.+(GL_OVR_multiview2|GL_OES_standard_derivatives|GL_EXT_shader_texture_lod|GL_EXT_frag_depth|GL_EXT_draw_buffers).+(enable|require)/g;\r\n code = code.replace(regex, \"\");\r\n\r\n // Replace instructions\r\n code = code.replace(/texture2D\\s*\\(/g, \"texture(\");\r\n if (isFragment) {\r\n const hasOutput = code.search(/layout *\\(location *= *0\\) *out/g) !== -1;\r\n\r\n code = code.replace(/texture2DLodEXT\\s*\\(/g, \"textureLod(\");\r\n code = code.replace(/textureCubeLodEXT\\s*\\(/g, \"textureLod(\");\r\n code = code.replace(/textureCube\\s*\\(/g, \"texture(\");\r\n code = code.replace(/gl_FragDepthEXT/g, \"gl_FragDepth\");\r\n code = code.replace(/gl_FragColor/g, \"glFragColor\");\r\n code = code.replace(/gl_FragData/g, \"glFragData\");\r\n code = code.replace(/void\\s+?main\\s*\\(/g, (hasDrawBuffersExtension || hasOutput ? \"\" : \"layout(location = 0) out vec4 glFragColor;\\n\") + \"void main(\");\r\n } else {\r\n const hasMultiviewExtension = defines.indexOf(\"#define MULTIVIEW\") !== -1;\r\n if (hasMultiviewExtension) {\r\n return \"#extension GL_OVR_multiview2 : require\\nlayout (num_views = 2) in;\\n\" + code;\r\n }\r\n }\r\n\r\n return code;\r\n }\r\n}\r\n","/**\r\n * Class used to store gfx data (like WebGLBuffer)\r\n */\r\nexport class DataBuffer {\r\n private static _Counter = 0;\r\n\r\n /**\r\n * Gets or sets the number of objects referencing this buffer\r\n */\r\n public references: number = 0;\r\n /** Gets or sets the size of the underlying buffer */\r\n public capacity: number = 0;\r\n /**\r\n * Gets or sets a boolean indicating if the buffer contains 32bits indices\r\n */\r\n public is32Bits: boolean = false;\r\n\r\n /**\r\n * Gets the underlying buffer\r\n */\r\n public get underlyingResource(): any {\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets the unique id of this buffer\r\n */\r\n public readonly uniqueId: number;\r\n\r\n /**\r\n * Constructs the buffer\r\n */\r\n constructor() {\r\n this.uniqueId = DataBuffer._Counter++;\r\n }\r\n}\r\n","import { DataBuffer } from \"../../Buffers/dataBuffer\";\r\nimport type { Nullable } from \"../../types\";\r\n\r\n/** @internal */\r\nexport class WebGLDataBuffer extends DataBuffer {\r\n private _buffer: Nullable;\r\n\r\n public constructor(resource: WebGLBuffer) {\r\n super();\r\n this._buffer = resource;\r\n }\r\n\r\n public get underlyingResource(): any {\r\n return this._buffer;\r\n }\r\n}\r\n","import type { IPipelineContext } from \"../IPipelineContext\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { Effect } from \"../../Materials/effect\";\r\nimport type { IMatrixLike, IVector2Like, IVector3Like, IVector4Like, IColor3Like, IColor4Like, IQuaternionLike } from \"../../Maths/math.like\";\r\nimport type { ThinEngine } from \"../thinEngine\";\r\n\r\n/** @internal */\r\nexport class WebGLPipelineContext implements IPipelineContext {\r\n private _valueCache: { [key: string]: any } = {};\r\n private _uniforms: { [key: string]: Nullable };\r\n\r\n public engine: ThinEngine;\r\n public program: Nullable;\r\n public context?: WebGLRenderingContext;\r\n public vertexShader?: WebGLShader;\r\n public fragmentShader?: WebGLShader;\r\n public isParallelCompiled: boolean;\r\n public onCompiled?: () => void;\r\n public transformFeedback?: WebGLTransformFeedback | null;\r\n\r\n public vertexCompilationError: Nullable = null;\r\n public fragmentCompilationError: Nullable = null;\r\n public programLinkError: Nullable = null;\r\n public programValidationError: Nullable = null;\r\n\r\n /** @internal */\r\n public _isDisposed = false;\r\n\r\n public get isAsync() {\r\n return this.isParallelCompiled;\r\n }\r\n\r\n public get isReady(): boolean {\r\n if (this.program) {\r\n if (this.isParallelCompiled) {\r\n return this.engine._isRenderingStateCompiled(this);\r\n }\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n public _handlesSpectorRebuildCallback(onCompiled: (program: WebGLProgram) => void): void {\r\n if (onCompiled && this.program) {\r\n onCompiled(this.program);\r\n }\r\n }\r\n\r\n public _fillEffectInformation(\r\n effect: Effect,\r\n uniformBuffersNames: { [key: string]: number },\r\n uniformsNames: string[],\r\n uniforms: { [key: string]: Nullable },\r\n samplerList: string[],\r\n samplers: { [key: string]: number },\r\n attributesNames: string[],\r\n attributes: number[]\r\n ) {\r\n const engine = this.engine;\r\n if (engine.supportsUniformBuffers) {\r\n for (const name in uniformBuffersNames) {\r\n effect.bindUniformBlock(name, uniformBuffersNames[name]);\r\n }\r\n }\r\n\r\n const effectAvailableUniforms = this.engine.getUniforms(this, uniformsNames);\r\n effectAvailableUniforms.forEach((uniform, index) => {\r\n uniforms[uniformsNames[index]] = uniform;\r\n });\r\n this._uniforms = uniforms;\r\n\r\n let index: number;\r\n for (index = 0; index < samplerList.length; index++) {\r\n const sampler = effect.getUniform(samplerList[index]);\r\n if (sampler == null) {\r\n samplerList.splice(index, 1);\r\n index--;\r\n }\r\n }\r\n\r\n samplerList.forEach((name, index) => {\r\n samplers[name] = index;\r\n });\r\n\r\n for (const attr of engine.getAttributes(this, attributesNames)) {\r\n attributes.push(attr);\r\n }\r\n }\r\n\r\n /**\r\n * Release all associated resources.\r\n **/\r\n public dispose() {\r\n this._uniforms = {};\r\n this._isDisposed = true;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _cacheMatrix(uniformName: string, matrix: IMatrixLike): boolean {\r\n const cache = this._valueCache[uniformName];\r\n const flag = matrix.updateFlag;\r\n if (cache !== undefined && cache === flag) {\r\n return false;\r\n }\r\n\r\n this._valueCache[uniformName] = flag;\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _cacheFloat2(uniformName: string, x: number, y: number): boolean {\r\n let cache = this._valueCache[uniformName];\r\n if (!cache || cache.length !== 2) {\r\n cache = [x, y];\r\n this._valueCache[uniformName] = cache;\r\n return true;\r\n }\r\n\r\n let changed = false;\r\n if (cache[0] !== x) {\r\n cache[0] = x;\r\n changed = true;\r\n }\r\n if (cache[1] !== y) {\r\n cache[1] = y;\r\n changed = true;\r\n }\r\n\r\n return changed;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _cacheFloat3(uniformName: string, x: number, y: number, z: number): boolean {\r\n let cache = this._valueCache[uniformName];\r\n if (!cache || cache.length !== 3) {\r\n cache = [x, y, z];\r\n this._valueCache[uniformName] = cache;\r\n return true;\r\n }\r\n\r\n let changed = false;\r\n if (cache[0] !== x) {\r\n cache[0] = x;\r\n changed = true;\r\n }\r\n if (cache[1] !== y) {\r\n cache[1] = y;\r\n changed = true;\r\n }\r\n if (cache[2] !== z) {\r\n cache[2] = z;\r\n changed = true;\r\n }\r\n\r\n return changed;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _cacheFloat4(uniformName: string, x: number, y: number, z: number, w: number): boolean {\r\n let cache = this._valueCache[uniformName];\r\n if (!cache || cache.length !== 4) {\r\n cache = [x, y, z, w];\r\n this._valueCache[uniformName] = cache;\r\n return true;\r\n }\r\n\r\n let changed = false;\r\n if (cache[0] !== x) {\r\n cache[0] = x;\r\n changed = true;\r\n }\r\n if (cache[1] !== y) {\r\n cache[1] = y;\r\n changed = true;\r\n }\r\n if (cache[2] !== z) {\r\n cache[2] = z;\r\n changed = true;\r\n }\r\n if (cache[3] !== w) {\r\n cache[3] = w;\r\n changed = true;\r\n }\r\n\r\n return changed;\r\n }\r\n\r\n /**\r\n * Sets an integer value on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param value Value to be set.\r\n */\r\n public setInt(uniformName: string, value: number): void {\r\n const cache = this._valueCache[uniformName];\r\n if (cache !== undefined && cache === value) {\r\n return;\r\n }\r\n\r\n if (this.engine.setInt(this._uniforms[uniformName], value)) {\r\n this._valueCache[uniformName] = value;\r\n }\r\n }\r\n\r\n /**\r\n * Sets a int2 on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param x First int in int2.\r\n * @param y Second int in int2.\r\n */\r\n public setInt2(uniformName: string, x: number, y: number): void {\r\n if (this._cacheFloat2(uniformName, x, y)) {\r\n if (!this.engine.setInt2(this._uniforms[uniformName], x, y)) {\r\n this._valueCache[uniformName] = null;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Sets a int3 on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param x First int in int3.\r\n * @param y Second int in int3.\r\n * @param z Third int in int3.\r\n */\r\n public setInt3(uniformName: string, x: number, y: number, z: number): void {\r\n if (this._cacheFloat3(uniformName, x, y, z)) {\r\n if (!this.engine.setInt3(this._uniforms[uniformName], x, y, z)) {\r\n this._valueCache[uniformName] = null;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Sets a int4 on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param x First int in int4.\r\n * @param y Second int in int4.\r\n * @param z Third int in int4.\r\n * @param w Fourth int in int4.\r\n */\r\n public setInt4(uniformName: string, x: number, y: number, z: number, w: number): void {\r\n if (this._cacheFloat4(uniformName, x, y, z, w)) {\r\n if (!this.engine.setInt4(this._uniforms[uniformName], x, y, z, w)) {\r\n this._valueCache[uniformName] = null;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Sets an int array on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n */\r\n public setIntArray(uniformName: string, array: Int32Array): void {\r\n this._valueCache[uniformName] = null;\r\n this.engine.setIntArray(this._uniforms[uniformName], array);\r\n }\r\n\r\n /**\r\n * Sets an int array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n */\r\n public setIntArray2(uniformName: string, array: Int32Array): void {\r\n this._valueCache[uniformName] = null;\r\n this.engine.setIntArray2(this._uniforms[uniformName], array);\r\n }\r\n\r\n /**\r\n * Sets an int array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n */\r\n public setIntArray3(uniformName: string, array: Int32Array): void {\r\n this._valueCache[uniformName] = null;\r\n this.engine.setIntArray3(this._uniforms[uniformName], array);\r\n }\r\n\r\n /**\r\n * Sets an int array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n */\r\n public setIntArray4(uniformName: string, array: Int32Array): void {\r\n this._valueCache[uniformName] = null;\r\n this.engine.setIntArray4(this._uniforms[uniformName], array);\r\n }\r\n\r\n /**\r\n * Sets an unsigned integer value on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param value Value to be set.\r\n */\r\n public setUInt(uniformName: string, value: number): void {\r\n const cache = this._valueCache[uniformName];\r\n if (cache !== undefined && cache === value) {\r\n return;\r\n }\r\n\r\n if (this.engine.setUInt(this._uniforms[uniformName], value)) {\r\n this._valueCache[uniformName] = value;\r\n }\r\n }\r\n\r\n /**\r\n * Sets an unsigned int2 value on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param x First unsigned int in uint2.\r\n * @param y Second unsigned int in uint2.\r\n */\r\n public setUInt2(uniformName: string, x: number, y: number): void {\r\n if (this._cacheFloat2(uniformName, x, y)) {\r\n if (!this.engine.setUInt2(this._uniforms[uniformName], x, y)) {\r\n this._valueCache[uniformName] = null;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Sets an unsigned int3 value on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param x First unsigned int in uint3.\r\n * @param y Second unsigned int in uint3.\r\n * @param z Third unsigned int in uint3.\r\n */\r\n public setUInt3(uniformName: string, x: number, y: number, z: number): void {\r\n if (this._cacheFloat3(uniformName, x, y, z)) {\r\n if (!this.engine.setUInt3(this._uniforms[uniformName], x, y, z)) {\r\n this._valueCache[uniformName] = null;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Sets an unsigned int4 value on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param x First unsigned int in uint4.\r\n * @param y Second unsigned int in uint4.\r\n * @param z Third unsigned int in uint4.\r\n * @param w Fourth unsigned int in uint4.\r\n */\r\n public setUInt4(uniformName: string, x: number, y: number, z: number, w: number): void {\r\n if (this._cacheFloat4(uniformName, x, y, z, w)) {\r\n if (!this.engine.setUInt4(this._uniforms[uniformName], x, y, z, w)) {\r\n this._valueCache[uniformName] = null;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Sets an unsigned int array on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n */\r\n public setUIntArray(uniformName: string, array: Uint32Array): void {\r\n this._valueCache[uniformName] = null;\r\n this.engine.setUIntArray(this._uniforms[uniformName], array);\r\n }\r\n\r\n /**\r\n * Sets an unsigned int array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n */\r\n public setUIntArray2(uniformName: string, array: Uint32Array): void {\r\n this._valueCache[uniformName] = null;\r\n this.engine.setUIntArray2(this._uniforms[uniformName], array);\r\n }\r\n\r\n /**\r\n * Sets an unsigned int array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n */\r\n public setUIntArray3(uniformName: string, array: Uint32Array): void {\r\n this._valueCache[uniformName] = null;\r\n this.engine.setUIntArray3(this._uniforms[uniformName], array);\r\n }\r\n\r\n /**\r\n * Sets an unsigned int array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n */\r\n public setUIntArray4(uniformName: string, array: Uint32Array): void {\r\n this._valueCache[uniformName] = null;\r\n this.engine.setUIntArray4(this._uniforms[uniformName], array);\r\n }\r\n\r\n /**\r\n * Sets an array on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n */\r\n public setArray(uniformName: string, array: number[]): void {\r\n this._valueCache[uniformName] = null;\r\n this.engine.setArray(this._uniforms[uniformName], array);\r\n }\r\n\r\n /**\r\n * Sets an array 2 on a uniform variable. (Array is specified as single array eg. [1,2,3,4] will result in [[1,2],[3,4]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n */\r\n public setArray2(uniformName: string, array: number[]): void {\r\n this._valueCache[uniformName] = null;\r\n this.engine.setArray2(this._uniforms[uniformName], array);\r\n }\r\n\r\n /**\r\n * Sets an array 3 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6] will result in [[1,2,3],[4,5,6]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n * @returns this effect.\r\n */\r\n public setArray3(uniformName: string, array: number[]): void {\r\n this._valueCache[uniformName] = null;\r\n this.engine.setArray3(this._uniforms[uniformName], array);\r\n }\r\n\r\n /**\r\n * Sets an array 4 on a uniform variable. (Array is specified as single array eg. [1,2,3,4,5,6,7,8] will result in [[1,2,3,4],[5,6,7,8]] in the shader)\r\n * @param uniformName Name of the variable.\r\n * @param array array to be set.\r\n */\r\n public setArray4(uniformName: string, array: number[]): void {\r\n this._valueCache[uniformName] = null;\r\n this.engine.setArray4(this._uniforms[uniformName], array);\r\n }\r\n\r\n /**\r\n * Sets matrices on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param matrices matrices to be set.\r\n */\r\n public setMatrices(uniformName: string, matrices: Float32Array): void {\r\n if (!matrices) {\r\n return;\r\n }\r\n\r\n this._valueCache[uniformName] = null;\r\n this.engine.setMatrices(this._uniforms[uniformName], matrices);\r\n }\r\n\r\n /**\r\n * Sets matrix on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param matrix matrix to be set.\r\n */\r\n public setMatrix(uniformName: string, matrix: IMatrixLike): void {\r\n if (this._cacheMatrix(uniformName, matrix)) {\r\n if (!this.engine.setMatrices(this._uniforms[uniformName], matrix.toArray() as Float32Array)) {\r\n this._valueCache[uniformName] = null;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Sets a 3x3 matrix on a uniform variable. (Specified as [1,2,3,4,5,6,7,8,9] will result in [1,2,3][4,5,6][7,8,9] matrix)\r\n * @param uniformName Name of the variable.\r\n * @param matrix matrix to be set.\r\n */\r\n public setMatrix3x3(uniformName: string, matrix: Float32Array): void {\r\n this._valueCache[uniformName] = null;\r\n this.engine.setMatrix3x3(this._uniforms[uniformName], matrix);\r\n }\r\n\r\n /**\r\n * Sets a 2x2 matrix on a uniform variable. (Specified as [1,2,3,4] will result in [1,2][3,4] matrix)\r\n * @param uniformName Name of the variable.\r\n * @param matrix matrix to be set.\r\n */\r\n public setMatrix2x2(uniformName: string, matrix: Float32Array): void {\r\n this._valueCache[uniformName] = null;\r\n this.engine.setMatrix2x2(this._uniforms[uniformName], matrix);\r\n }\r\n\r\n /**\r\n * Sets a float on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param value value to be set.\r\n * @returns this effect.\r\n */\r\n public setFloat(uniformName: string, value: number): void {\r\n const cache = this._valueCache[uniformName];\r\n if (cache !== undefined && cache === value) {\r\n return;\r\n }\r\n\r\n if (this.engine.setFloat(this._uniforms[uniformName], value)) {\r\n this._valueCache[uniformName] = value;\r\n }\r\n }\r\n\r\n /**\r\n * Sets a Vector2 on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param vector2 vector2 to be set.\r\n */\r\n public setVector2(uniformName: string, vector2: IVector2Like): void {\r\n if (this._cacheFloat2(uniformName, vector2.x, vector2.y)) {\r\n if (!this.engine.setFloat2(this._uniforms[uniformName], vector2.x, vector2.y)) {\r\n this._valueCache[uniformName] = null;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Sets a float2 on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param x First float in float2.\r\n * @param y Second float in float2.\r\n */\r\n public setFloat2(uniformName: string, x: number, y: number): void {\r\n if (this._cacheFloat2(uniformName, x, y)) {\r\n if (!this.engine.setFloat2(this._uniforms[uniformName], x, y)) {\r\n this._valueCache[uniformName] = null;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Sets a Vector3 on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param vector3 Value to be set.\r\n */\r\n public setVector3(uniformName: string, vector3: IVector3Like): void {\r\n if (this._cacheFloat3(uniformName, vector3.x, vector3.y, vector3.z)) {\r\n if (!this.engine.setFloat3(this._uniforms[uniformName], vector3.x, vector3.y, vector3.z)) {\r\n this._valueCache[uniformName] = null;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Sets a float3 on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param x First float in float3.\r\n * @param y Second float in float3.\r\n * @param z Third float in float3.\r\n */\r\n public setFloat3(uniformName: string, x: number, y: number, z: number): void {\r\n if (this._cacheFloat3(uniformName, x, y, z)) {\r\n if (!this.engine.setFloat3(this._uniforms[uniformName], x, y, z)) {\r\n this._valueCache[uniformName] = null;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Sets a Vector4 on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param vector4 Value to be set.\r\n */\r\n public setVector4(uniformName: string, vector4: IVector4Like): void {\r\n if (this._cacheFloat4(uniformName, vector4.x, vector4.y, vector4.z, vector4.w)) {\r\n if (!this.engine.setFloat4(this._uniforms[uniformName], vector4.x, vector4.y, vector4.z, vector4.w)) {\r\n this._valueCache[uniformName] = null;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Sets a Quaternion on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param quaternion Value to be set.\r\n */\r\n public setQuaternion(uniformName: string, quaternion: IQuaternionLike): void {\r\n if (this._cacheFloat4(uniformName, quaternion.x, quaternion.y, quaternion.z, quaternion.w)) {\r\n if (!this.engine.setFloat4(this._uniforms[uniformName], quaternion.x, quaternion.y, quaternion.z, quaternion.w)) {\r\n this._valueCache[uniformName] = null;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Sets a float4 on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param x First float in float4.\r\n * @param y Second float in float4.\r\n * @param z Third float in float4.\r\n * @param w Fourth float in float4.\r\n * @returns this effect.\r\n */\r\n public setFloat4(uniformName: string, x: number, y: number, z: number, w: number): void {\r\n if (this._cacheFloat4(uniformName, x, y, z, w)) {\r\n if (!this.engine.setFloat4(this._uniforms[uniformName], x, y, z, w)) {\r\n this._valueCache[uniformName] = null;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Sets a Color3 on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param color3 Value to be set.\r\n */\r\n public setColor3(uniformName: string, color3: IColor3Like): void {\r\n if (this._cacheFloat3(uniformName, color3.r, color3.g, color3.b)) {\r\n if (!this.engine.setFloat3(this._uniforms[uniformName], color3.r, color3.g, color3.b)) {\r\n this._valueCache[uniformName] = null;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Sets a Color4 on a uniform variable.\r\n * @param uniformName Name of the variable.\r\n * @param color3 Value to be set.\r\n * @param alpha Alpha value to be set.\r\n */\r\n public setColor4(uniformName: string, color3: IColor3Like, alpha: number): void {\r\n if (this._cacheFloat4(uniformName, color3.r, color3.g, color3.b, alpha)) {\r\n if (!this.engine.setFloat4(this._uniforms[uniformName], color3.r, color3.g, color3.b, alpha)) {\r\n this._valueCache[uniformName] = null;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Sets a Color4 on a uniform variable\r\n * @param uniformName defines the name of the variable\r\n * @param color4 defines the value to be set\r\n */\r\n public setDirectColor4(uniformName: string, color4: IColor4Like): void {\r\n if (this._cacheFloat4(uniformName, color4.r, color4.g, color4.b, color4.a)) {\r\n if (!this.engine.setFloat4(this._uniforms[uniformName], color4.r, color4.g, color4.b, color4.a)) {\r\n this._valueCache[uniformName] = null;\r\n }\r\n }\r\n }\r\n\r\n public _getVertexShaderCode(): string | null {\r\n return this.vertexShader ? this.engine._getShaderSource(this.vertexShader) : null;\r\n }\r\n\r\n public _getFragmentShaderCode(): string | null {\r\n return this.fragmentShader ? this.engine._getShaderSource(this.fragmentShader) : null;\r\n }\r\n}\r\n","import type { HardwareTextureWrapper } from \"../../Materials/Textures/hardwareTextureWrapper\";\r\nimport type { Nullable } from \"../../types\";\r\n\r\n/** @internal */\r\nexport class WebGLHardwareTexture implements HardwareTextureWrapper {\r\n private _webGLTexture: WebGLTexture;\r\n private _context: WebGLRenderingContext;\r\n\r\n // There can be multiple buffers for a single WebGL texture because different layers of a 2DArrayTexture / 3DTexture\r\n // or different faces of a cube texture can be bound to different render targets at the same time.\r\n private _MSAARenderBuffers: Nullable = null;\r\n\r\n public get underlyingResource(): Nullable {\r\n return this._webGLTexture;\r\n }\r\n\r\n constructor(existingTexture: Nullable = null, context: WebGLRenderingContext) {\r\n this._context = context as WebGLRenderingContext;\r\n if (!existingTexture) {\r\n existingTexture = context.createTexture();\r\n if (!existingTexture) {\r\n throw new Error(\"Unable to create webGL texture\");\r\n }\r\n }\r\n this.set(existingTexture);\r\n }\r\n\r\n public setUsage(): void {}\r\n\r\n public set(hardwareTexture: WebGLTexture) {\r\n this._webGLTexture = hardwareTexture;\r\n }\r\n\r\n public reset() {\r\n this._webGLTexture = null as any;\r\n this._MSAARenderBuffers = null;\r\n }\r\n\r\n public addMSAARenderBuffer(buffer: WebGLRenderbuffer) {\r\n if (!this._MSAARenderBuffers) {\r\n this._MSAARenderBuffers = [];\r\n }\r\n this._MSAARenderBuffers.push(buffer);\r\n }\r\n\r\n public releaseMSAARenderBuffers() {\r\n if (this._MSAARenderBuffers) {\r\n for (const buffer of this._MSAARenderBuffers) {\r\n this._context.deleteRenderbuffer(buffer);\r\n }\r\n this._MSAARenderBuffers = null;\r\n }\r\n }\r\n\r\n public release() {\r\n this.releaseMSAARenderBuffers();\r\n\r\n if (this._webGLTexture) {\r\n this._context.deleteTexture(this._webGLTexture);\r\n }\r\n this.reset();\r\n }\r\n}\r\n","import type { IDrawContext } from \"../Engines/IDrawContext\";\r\nimport type { IMaterialContext } from \"../Engines/IMaterialContext\";\r\nimport type { Nullable } from \"../types\";\r\n\r\nimport type { ThinEngine } from \"../Engines/thinEngine\";\r\nimport type { Effect } from \"./effect\";\r\nimport type { MaterialDefines } from \"./materialDefines\";\r\n\r\n/** @internal */\r\nexport class DrawWrapper {\r\n public effect: Nullable;\r\n public defines: Nullable;\r\n public materialContext?: IMaterialContext;\r\n public drawContext?: IDrawContext;\r\n\r\n public static IsWrapper(effect: Effect | DrawWrapper): effect is DrawWrapper {\r\n return (effect as Effect).getPipelineContext === undefined;\r\n }\r\n\r\n public static GetEffect(effect: Effect | DrawWrapper): Nullable {\r\n return (effect as Effect).getPipelineContext === undefined ? (effect as DrawWrapper).effect : (effect as Effect);\r\n }\r\n\r\n constructor(engine: ThinEngine, createMaterialContext = true) {\r\n this.effect = null;\r\n this.defines = null;\r\n this.drawContext = engine.createDrawContext();\r\n if (createMaterialContext) {\r\n this.materialContext = engine.createMaterialContext();\r\n }\r\n }\r\n\r\n public setEffect(effect: Nullable, defines?: Nullable, resetContext = true): void {\r\n this.effect = effect;\r\n if (defines !== undefined) {\r\n this.defines = defines;\r\n }\r\n if (resetContext) {\r\n this.drawContext?.reset();\r\n }\r\n }\r\n\r\n public dispose(): void {\r\n this.drawContext?.dispose();\r\n }\r\n}\r\n","import type { IStencilState } from \"./IStencilState\";\r\n\r\n/**\r\n * @internal\r\n **/\r\nexport class StencilStateComposer {\r\n protected _isStencilTestDirty = false;\r\n protected _isStencilMaskDirty = false;\r\n protected _isStencilFuncDirty = false;\r\n protected _isStencilOpDirty = false;\r\n\r\n protected _enabled: boolean;\r\n\r\n protected _mask: number;\r\n\r\n protected _func: number;\r\n protected _funcRef: number;\r\n protected _funcMask: number;\r\n\r\n protected _opStencilFail: number;\r\n protected _opDepthFail: number;\r\n protected _opStencilDepthPass: number;\r\n\r\n public stencilGlobal: IStencilState;\r\n public stencilMaterial: IStencilState | undefined;\r\n\r\n public useStencilGlobalOnly = false;\r\n\r\n public get isDirty(): boolean {\r\n return this._isStencilTestDirty || this._isStencilMaskDirty || this._isStencilFuncDirty || this._isStencilOpDirty;\r\n }\r\n\r\n public get func(): number {\r\n return this._func;\r\n }\r\n\r\n public set func(value: number) {\r\n if (this._func === value) {\r\n return;\r\n }\r\n\r\n this._func = value;\r\n this._isStencilFuncDirty = true;\r\n }\r\n\r\n public get funcRef(): number {\r\n return this._funcRef;\r\n }\r\n\r\n public set funcRef(value: number) {\r\n if (this._funcRef === value) {\r\n return;\r\n }\r\n\r\n this._funcRef = value;\r\n this._isStencilFuncDirty = true;\r\n }\r\n\r\n public get funcMask(): number {\r\n return this._funcMask;\r\n }\r\n\r\n public set funcMask(value: number) {\r\n if (this._funcMask === value) {\r\n return;\r\n }\r\n\r\n this._funcMask = value;\r\n this._isStencilFuncDirty = true;\r\n }\r\n\r\n public get opStencilFail(): number {\r\n return this._opStencilFail;\r\n }\r\n\r\n public set opStencilFail(value: number) {\r\n if (this._opStencilFail === value) {\r\n return;\r\n }\r\n\r\n this._opStencilFail = value;\r\n this._isStencilOpDirty = true;\r\n }\r\n\r\n public get opDepthFail(): number {\r\n return this._opDepthFail;\r\n }\r\n\r\n public set opDepthFail(value: number) {\r\n if (this._opDepthFail === value) {\r\n return;\r\n }\r\n\r\n this._opDepthFail = value;\r\n this._isStencilOpDirty = true;\r\n }\r\n\r\n public get opStencilDepthPass(): number {\r\n return this._opStencilDepthPass;\r\n }\r\n\r\n public set opStencilDepthPass(value: number) {\r\n if (this._opStencilDepthPass === value) {\r\n return;\r\n }\r\n\r\n this._opStencilDepthPass = value;\r\n this._isStencilOpDirty = true;\r\n }\r\n\r\n public get mask(): number {\r\n return this._mask;\r\n }\r\n\r\n public set mask(value: number) {\r\n if (this._mask === value) {\r\n return;\r\n }\r\n\r\n this._mask = value;\r\n this._isStencilMaskDirty = true;\r\n }\r\n\r\n public get enabled(): boolean {\r\n return this._enabled;\r\n }\r\n\r\n public set enabled(value: boolean) {\r\n if (this._enabled === value) {\r\n return;\r\n }\r\n\r\n this._enabled = value;\r\n this._isStencilTestDirty = true;\r\n }\r\n\r\n public constructor(reset = true) {\r\n if (reset) {\r\n this.reset();\r\n }\r\n }\r\n\r\n public reset() {\r\n this.stencilMaterial = undefined;\r\n\r\n this.stencilGlobal?.reset();\r\n\r\n this._isStencilTestDirty = true;\r\n this._isStencilMaskDirty = true;\r\n this._isStencilFuncDirty = true;\r\n this._isStencilOpDirty = true;\r\n }\r\n\r\n public apply(gl?: WebGLRenderingContext) {\r\n if (!gl) {\r\n return;\r\n }\r\n\r\n const stencilMaterialEnabled = !this.useStencilGlobalOnly && !!this.stencilMaterial?.enabled;\r\n\r\n this.enabled = stencilMaterialEnabled ? this.stencilMaterial!.enabled : this.stencilGlobal.enabled;\r\n this.func = stencilMaterialEnabled ? this.stencilMaterial!.func : this.stencilGlobal.func;\r\n this.funcRef = stencilMaterialEnabled ? this.stencilMaterial!.funcRef : this.stencilGlobal.funcRef;\r\n this.funcMask = stencilMaterialEnabled ? this.stencilMaterial!.funcMask : this.stencilGlobal.funcMask;\r\n this.opStencilFail = stencilMaterialEnabled ? this.stencilMaterial!.opStencilFail : this.stencilGlobal.opStencilFail;\r\n this.opDepthFail = stencilMaterialEnabled ? this.stencilMaterial!.opDepthFail : this.stencilGlobal.opDepthFail;\r\n this.opStencilDepthPass = stencilMaterialEnabled ? this.stencilMaterial!.opStencilDepthPass : this.stencilGlobal.opStencilDepthPass;\r\n this.mask = stencilMaterialEnabled ? this.stencilMaterial!.mask : this.stencilGlobal.mask;\r\n\r\n if (!this.isDirty) {\r\n return;\r\n }\r\n\r\n // Stencil test\r\n if (this._isStencilTestDirty) {\r\n if (this.enabled) {\r\n gl.enable(gl.STENCIL_TEST);\r\n } else {\r\n gl.disable(gl.STENCIL_TEST);\r\n }\r\n this._isStencilTestDirty = false;\r\n }\r\n\r\n // Stencil mask\r\n if (this._isStencilMaskDirty) {\r\n gl.stencilMask(this.mask);\r\n this._isStencilMaskDirty = false;\r\n }\r\n\r\n // Stencil func\r\n if (this._isStencilFuncDirty) {\r\n gl.stencilFunc(this.func, this.funcRef, this.funcMask);\r\n this._isStencilFuncDirty = false;\r\n }\r\n\r\n // Stencil op\r\n if (this._isStencilOpDirty) {\r\n gl.stencilOp(this.opStencilFail, this.opDepthFail, this.opStencilDepthPass);\r\n this._isStencilOpDirty = false;\r\n }\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/no-unused-vars */\r\nimport { EngineStore } from \"./engineStore\";\r\nimport type { IInternalTextureLoader } from \"../Materials/Textures/internalTextureLoader\";\r\nimport type { IEffectCreationOptions } from \"../Materials/effect\";\r\nimport { Effect } from \"../Materials/effect\";\r\nimport { _WarnImport } from \"../Misc/devTools\";\r\nimport type { IShaderProcessor } from \"./Processors/iShaderProcessor\";\r\nimport type { ShaderProcessingContext } from \"./Processors/shaderProcessingOptions\";\r\nimport type { UniformBuffer } from \"../Materials/uniformBuffer\";\r\nimport type { Nullable, DataArray, IndicesArray } from \"../types\";\r\nimport type { EngineCapabilities } from \"./engineCapabilities\";\r\nimport type { Observer } from \"../Misc/observable\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport { DepthCullingState } from \"../States/depthCullingState\";\r\nimport { StencilState } from \"../States/stencilState\";\r\nimport { AlphaState } from \"../States/alphaCullingState\";\r\nimport { Constants } from \"./constants\";\r\nimport { InternalTexture, InternalTextureSource } from \"../Materials/Textures/internalTexture\";\r\nimport type { IViewportLike, IColor4Like } from \"../Maths/math.like\";\r\nimport type { DataBuffer } from \"../Buffers/dataBuffer\";\r\nimport type { IFileRequest } from \"../Misc/fileRequest\";\r\nimport { Logger } from \"../Misc/logger\";\r\nimport { IsDocumentAvailable, IsWindowObjectExist } from \"../Misc/domManagement\";\r\nimport { WebGLShaderProcessor } from \"./WebGL/webGLShaderProcessors\";\r\nimport { WebGL2ShaderProcessor } from \"./WebGL/webGL2ShaderProcessors\";\r\nimport { WebGLDataBuffer } from \"../Meshes/WebGL/webGLDataBuffer\";\r\nimport type { IPipelineContext } from \"./IPipelineContext\";\r\nimport { WebGLPipelineContext } from \"./WebGL/webGLPipelineContext\";\r\nimport type { VertexBuffer } from \"../Buffers/buffer\";\r\nimport type { InstancingAttributeInfo } from \"./instancingAttributeInfo\";\r\nimport type { ThinTexture } from \"../Materials/Textures/thinTexture\";\r\nimport type { IOfflineProvider } from \"../Offline/IOfflineProvider\";\r\nimport type { IEffectFallbacks } from \"../Materials/iEffectFallbacks\";\r\nimport type { IWebRequest } from \"../Misc/interfaces/iWebRequest\";\r\nimport { PerformanceConfigurator } from \"./performanceConfigurator\";\r\nimport type { EngineFeatures } from \"./engineFeatures\";\r\nimport type { HardwareTextureWrapper } from \"../Materials/Textures/hardwareTextureWrapper\";\r\nimport { WebGLHardwareTexture } from \"./WebGL/webGLHardwareTexture\";\r\nimport { DrawWrapper } from \"../Materials/drawWrapper\";\r\nimport type { IMaterialContext } from \"./IMaterialContext\";\r\nimport type { IDrawContext } from \"./IDrawContext\";\r\nimport type { ICanvas, ICanvasRenderingContext, IImage } from \"./ICanvas\";\r\nimport { StencilStateComposer } from \"../States/stencilStateComposer\";\r\nimport type { StorageBuffer } from \"../Buffers/storageBuffer\";\r\nimport type { IAudioEngineOptions } from \"../Audio/Interfaces/IAudioEngineOptions\";\r\nimport type { IStencilState } from \"../States/IStencilState\";\r\nimport type { InternalTextureCreationOptions, TextureSize } from \"../Materials/Textures/textureCreationOptions\";\r\nimport { ShaderLanguage } from \"../Materials/shaderLanguage\";\r\nimport type { RenderTargetWrapper } from \"./renderTargetWrapper\";\r\nimport type { WebGLRenderTargetWrapper } from \"./WebGL/webGLRenderTargetWrapper\";\r\nimport type { VideoTexture } from \"../Materials/Textures/videoTexture\";\r\nimport type { RenderTargetTexture } from \"../Materials/Textures/renderTargetTexture\";\r\nimport type { WebRequest } from \"../Misc/webRequest\";\r\nimport type { LoadFileError } from \"../Misc/fileTools\";\r\nimport type { Texture } from \"../Materials/Textures/texture\";\r\nimport { PrecisionDate } from \"core/Misc/precisionDate\";\r\n\r\n/**\r\n * Defines the interface used by objects working like Scene\r\n * @internal\r\n */\r\nexport interface ISceneLike {\r\n addPendingData(data: any): void;\r\n removePendingData(data: any): void;\r\n offlineProvider: IOfflineProvider;\r\n}\r\n\r\n/**\r\n * Keeps track of all the buffer info used in engine.\r\n */\r\nclass BufferPointer {\r\n public active: boolean;\r\n public index: number;\r\n public size: number;\r\n public type: number;\r\n public normalized: boolean;\r\n public stride: number;\r\n public offset: number;\r\n public buffer: WebGLBuffer;\r\n}\r\n\r\n/**\r\n * Information about the current host\r\n */\r\nexport interface HostInformation {\r\n /**\r\n * Defines if the current host is a mobile\r\n */\r\n isMobile: boolean;\r\n}\r\n\r\n/** Interface defining initialization parameters for ThinEngine class */\r\nexport interface ThinEngineOptions {\r\n /**\r\n * Defines if the engine should no exceed a specified device ratio\r\n * @see https://developer.mozilla.org/en-US/docs/Web/API/Window/devicePixelRatio\r\n */\r\n limitDeviceRatio?: number;\r\n /**\r\n * Defines if webaudio should be initialized as well\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/audio/playingSoundsMusic\r\n */\r\n audioEngine?: boolean;\r\n /**\r\n * Specifies options for the audio engine\r\n */\r\n audioEngineOptions?: IAudioEngineOptions;\r\n\r\n /**\r\n * Defines if animations should run using a deterministic lock step\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep\r\n */\r\n deterministicLockstep?: boolean;\r\n /** Defines the maximum steps to use with deterministic lock step mode */\r\n lockstepMaxSteps?: number;\r\n /** Defines the seconds between each deterministic lock step */\r\n timeStep?: number;\r\n /**\r\n * Defines that engine should ignore context lost events\r\n * If this event happens when this parameter is true, you will have to reload the page to restore rendering\r\n */\r\n doNotHandleContextLost?: boolean;\r\n /**\r\n * Defines that engine should ignore modifying touch action attribute and style\r\n * If not handle, you might need to set it up on your side for expected touch devices behavior.\r\n */\r\n doNotHandleTouchAction?: boolean;\r\n\r\n /**\r\n * Make the matrix computations to be performed in 64 bits instead of 32 bits. False by default\r\n */\r\n useHighPrecisionMatrix?: boolean;\r\n\r\n /**\r\n * Defines whether to adapt to the device's viewport characteristics (default: false)\r\n */\r\n adaptToDeviceRatio?: boolean;\r\n\r\n /**\r\n * True if the more expensive but exact conversions should be used for transforming colors to and from linear space within shaders.\r\n * Otherwise, the default is to use a cheaper approximation.\r\n */\r\n useExactSrgbConversions?: boolean;\r\n\r\n /**\r\n * Defines whether MSAA is enabled on the canvas.\r\n */\r\n antialias?: boolean;\r\n\r\n /**\r\n * Defines whether the stencil buffer should be enabled.\r\n */\r\n stencil?: boolean;\r\n\r\n /**\r\n * Defines whether the canvas should be created in \"premultiplied\" mode (if false, the canvas is created in the \"opaque\" mode) (true by default)\r\n */\r\n premultipliedAlpha?: boolean;\r\n}\r\n\r\n/** Interface defining initialization parameters for Engine class */\r\nexport interface EngineOptions extends ThinEngineOptions, WebGLContextAttributes {\r\n /**\r\n * Defines if webvr should be enabled automatically\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/webVRCamera\r\n */\r\n autoEnableWebVR?: boolean;\r\n /**\r\n * Defines if webgl2 should be turned off even if supported\r\n * @see https://doc.babylonjs.com/setup/support/webGL2\r\n */\r\n disableWebGL2Support?: boolean;\r\n\r\n /**\r\n * Defines that engine should compile shaders with high precision floats (if supported). True by default\r\n */\r\n useHighPrecisionFloats?: boolean;\r\n /**\r\n * Make the canvas XR Compatible for XR sessions\r\n */\r\n xrCompatible?: boolean;\r\n\r\n /**\r\n * Will prevent the system from falling back to software implementation if a hardware device cannot be created\r\n */\r\n failIfMajorPerformanceCaveat?: boolean;\r\n\r\n /**\r\n * If sRGB Buffer support is not set during construction, use this value to force a specific state\r\n * This is added due to an issue when processing textures in chrome/edge/firefox\r\n * This will not influence NativeEngine and WebGPUEngine which set the behavior to true during construction.\r\n */\r\n forceSRGBBufferSupportState?: boolean;\r\n}\r\n\r\n/**\r\n * The base engine class (root of all engines)\r\n */\r\nexport class ThinEngine {\r\n private static _TempClearColorUint32 = new Uint32Array(4);\r\n private static _TempClearColorInt32 = new Int32Array(4);\r\n\r\n /** Use this array to turn off some WebGL2 features on known buggy browsers version */\r\n public static ExceptionList = [\r\n { key: \"Chrome/63.0\", capture: \"63\\\\.0\\\\.3239\\\\.(\\\\d+)\", captureConstraint: 108, targets: [\"uniformBuffer\"] },\r\n { key: \"Firefox/58\", capture: null, captureConstraint: null, targets: [\"uniformBuffer\"] },\r\n { key: \"Firefox/59\", capture: null, captureConstraint: null, targets: [\"uniformBuffer\"] },\r\n { key: \"Chrome/72.+?Mobile\", capture: null, captureConstraint: null, targets: [\"vao\"] },\r\n { key: \"Chrome/73.+?Mobile\", capture: null, captureConstraint: null, targets: [\"vao\"] },\r\n { key: \"Chrome/74.+?Mobile\", capture: null, captureConstraint: null, targets: [\"vao\"] },\r\n { key: \"Mac OS.+Chrome/71\", capture: null, captureConstraint: null, targets: [\"vao\"] },\r\n { key: \"Mac OS.+Chrome/72\", capture: null, captureConstraint: null, targets: [\"vao\"] },\r\n { key: \"Mac OS.+Chrome\", capture: null, captureConstraint: null, targets: [\"uniformBuffer\"] },\r\n // desktop osx safari 15.4\r\n { key: \".*AppleWebKit.*(15.4).*Safari\", capture: null, captureConstraint: null, targets: [\"antialias\", \"maxMSAASamples\"] },\r\n // mobile browsers using safari 15.4 on ios\r\n { key: \".*(15.4).*AppleWebKit.*Safari\", capture: null, captureConstraint: null, targets: [\"antialias\", \"maxMSAASamples\"] },\r\n ];\r\n\r\n /** @internal */\r\n public static _TextureLoaders: IInternalTextureLoader[] = [];\r\n\r\n /**\r\n * Returns the current npm package of the sdk\r\n */\r\n // Not mixed with Version for tooling purpose.\r\n public static get NpmPackage(): string {\r\n return \"babylonjs@6.15.0\";\r\n }\r\n\r\n /**\r\n * Returns the current version of the framework\r\n */\r\n public static get Version(): string {\r\n return \"6.15.0\";\r\n }\r\n\r\n /**\r\n * Returns a string describing the current engine\r\n */\r\n public get description(): string {\r\n let description = this.name + this.webGLVersion;\r\n\r\n if (this._caps.parallelShaderCompile) {\r\n description += \" - Parallel shader compilation\";\r\n }\r\n\r\n return description;\r\n }\r\n\r\n /** @internal */\r\n protected _name = \"WebGL\";\r\n\r\n /**\r\n * Gets or sets the name of the engine\r\n */\r\n public get name(): string {\r\n return this._name;\r\n }\r\n\r\n public set name(value: string) {\r\n this._name = value;\r\n }\r\n\r\n /**\r\n * Returns the version of the engine\r\n */\r\n public get version(): number {\r\n return this._webGLVersion;\r\n }\r\n\r\n protected _isDisposed = false;\r\n\r\n public get isDisposed(): boolean {\r\n return this._isDisposed;\r\n }\r\n\r\n // Updatable statics so stick with vars here\r\n\r\n /**\r\n * Gets or sets the epsilon value used by collision engine\r\n */\r\n public static CollisionsEpsilon = 0.001;\r\n\r\n /**\r\n * Gets or sets the relative url used to load shaders if using the engine in non-minified mode\r\n */\r\n public static get ShadersRepository(): string {\r\n return Effect.ShadersRepository;\r\n }\r\n public static set ShadersRepository(value: string) {\r\n Effect.ShadersRepository = value;\r\n }\r\n\r\n protected _shaderProcessor: Nullable;\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getShaderProcessor(shaderLanguage: ShaderLanguage): Nullable {\r\n return this._shaderProcessor;\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean that indicates if textures must be forced to power of 2 size even if not required\r\n */\r\n public forcePOTTextures = false;\r\n\r\n /**\r\n * Gets a boolean indicating if the engine is currently rendering in fullscreen mode\r\n */\r\n public isFullscreen = false;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if back faces must be culled. If false, front faces are culled instead (true by default)\r\n * If non null, this takes precedence over the value from the material\r\n */\r\n public cullBackFaces: Nullable = null;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if the engine must keep rendering even if the window is not in foreground\r\n */\r\n public renderEvenInBackground = true;\r\n\r\n /**\r\n * Gets or sets a boolean indicating that cache can be kept between frames\r\n */\r\n public preventCacheWipeBetweenFrames = false;\r\n\r\n /** Gets or sets a boolean indicating if the engine should validate programs after compilation */\r\n public validateShaderPrograms = false;\r\n\r\n private _useReverseDepthBuffer = false;\r\n /**\r\n * Gets or sets a boolean indicating if depth buffer should be reverse, going from far to near.\r\n * This can provide greater z depth for distant objects.\r\n */\r\n public get useReverseDepthBuffer(): boolean {\r\n return this._useReverseDepthBuffer;\r\n }\r\n\r\n public set useReverseDepthBuffer(useReverse) {\r\n if (useReverse === this._useReverseDepthBuffer) {\r\n return;\r\n }\r\n\r\n this._useReverseDepthBuffer = useReverse;\r\n\r\n if (useReverse) {\r\n this._depthCullingState.depthFunc = Constants.GEQUAL;\r\n } else {\r\n this._depthCullingState.depthFunc = Constants.LEQUAL;\r\n }\r\n }\r\n\r\n /**\r\n * Indicates if the z range in NDC space is 0..1 (value: true) or -1..1 (value: false)\r\n */\r\n public readonly isNDCHalfZRange: boolean = false;\r\n\r\n /**\r\n * Indicates that the origin of the texture/framebuffer space is the bottom left corner. If false, the origin is top left\r\n */\r\n public readonly hasOriginBottomLeft: boolean = true;\r\n\r\n /**\r\n * Gets or sets a boolean indicating that uniform buffers must be disabled even if they are supported\r\n */\r\n public disableUniformBuffers = false;\r\n\r\n /**\r\n * An event triggered when the engine is disposed.\r\n */\r\n public readonly onDisposeObservable = new Observable();\r\n\r\n private _frameId = 0;\r\n /**\r\n * Gets the current frame id\r\n */\r\n public get frameId(): number {\r\n return this._frameId;\r\n }\r\n\r\n /**\r\n * The time (in milliseconds elapsed since the current page has been loaded) when the engine was initialized\r\n */\r\n public readonly startTime: number;\r\n\r\n /** @internal */\r\n public _uniformBuffers = new Array();\r\n /** @internal */\r\n public _storageBuffers = new Array();\r\n\r\n /**\r\n * Gets a boolean indicating that the engine supports uniform buffers\r\n * @see https://doc.babylonjs.com/setup/support/webGL2#uniform-buffer-objets\r\n */\r\n public get supportsUniformBuffers(): boolean {\r\n return this.webGLVersion > 1 && !this.disableUniformBuffers;\r\n }\r\n\r\n // Private Members\r\n\r\n /** @internal */\r\n public _gl: WebGL2RenderingContext;\r\n /** @internal */\r\n public _webGLVersion = 1.0;\r\n protected _renderingCanvas: Nullable;\r\n protected _windowIsBackground = false;\r\n protected _creationOptions: EngineOptions;\r\n protected _audioContext: Nullable;\r\n protected _audioDestination: Nullable;\r\n /** @internal */\r\n public _glSRGBExtensionValues: {\r\n SRGB: typeof WebGL2RenderingContext.SRGB;\r\n SRGB8: typeof WebGL2RenderingContext.SRGB8 | EXT_sRGB[\"SRGB_ALPHA_EXT\"];\r\n SRGB8_ALPHA8: typeof WebGL2RenderingContext.SRGB8_ALPHA8 | EXT_sRGB[\"SRGB_ALPHA_EXT\"];\r\n };\r\n /**\r\n * Gets the options used for engine creation\r\n * @returns EngineOptions object\r\n */\r\n public getCreationOptions() {\r\n return this._creationOptions;\r\n }\r\n\r\n protected _highPrecisionShadersAllowed = true;\r\n /** @internal */\r\n public get _shouldUseHighPrecisionShader(): boolean {\r\n return !!(this._caps.highPrecisionShaderSupported && this._highPrecisionShadersAllowed);\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating that only power of 2 textures are supported\r\n * Please note that you can still use non power of 2 textures but in this case the engine will forcefully convert them\r\n */\r\n public get needPOTTextures(): boolean {\r\n return this._webGLVersion < 2 || this.forcePOTTextures;\r\n }\r\n\r\n /** @internal */\r\n public _badOS = false;\r\n\r\n /** @internal */\r\n public _badDesktopOS = false;\r\n\r\n /** @internal */\r\n public _hardwareScalingLevel: number;\r\n /** @internal */\r\n public _caps: EngineCapabilities;\r\n /** @internal */\r\n public _features: EngineFeatures;\r\n protected _isStencilEnable: boolean;\r\n\r\n private _glVersion: string;\r\n private _glRenderer: string;\r\n private _glVendor: string;\r\n\r\n /** @internal */\r\n public _videoTextureSupported: boolean;\r\n\r\n protected _renderingQueueLaunched = false;\r\n protected _activeRenderLoops = new Array<() => void>();\r\n\r\n /**\r\n * Gets the list of current active render loop functions\r\n * @returns an array with the current render loop functions\r\n */\r\n public get activeRenderLoops(): Array<() => void> {\r\n return this._activeRenderLoops;\r\n }\r\n\r\n // Lost context\r\n /**\r\n * Observable signaled when a context lost event is raised\r\n */\r\n public onContextLostObservable = new Observable();\r\n /**\r\n * Observable signaled when a context restored event is raised\r\n */\r\n public onContextRestoredObservable = new Observable();\r\n private _onContextLost: (evt: Event) => void;\r\n private _onContextRestored: (evt: Event) => void;\r\n protected _contextWasLost = false;\r\n\r\n /** @internal */\r\n public _doNotHandleContextLost = false;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if resources should be retained to be able to handle context lost events\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimize_your_scene#handling-webgl-context-lost\r\n */\r\n public get doNotHandleContextLost(): boolean {\r\n return this._doNotHandleContextLost;\r\n }\r\n\r\n public set doNotHandleContextLost(value: boolean) {\r\n this._doNotHandleContextLost = value;\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating that vertex array object must be disabled even if they are supported\r\n */\r\n public disableVertexArrayObjects = false;\r\n\r\n // States\r\n /** @internal */\r\n protected _colorWrite = true;\r\n /** @internal */\r\n protected _colorWriteChanged = true;\r\n /** @internal */\r\n protected _depthCullingState = new DepthCullingState();\r\n /** @internal */\r\n protected _stencilStateComposer = new StencilStateComposer();\r\n /** @internal */\r\n protected _stencilState = new StencilState();\r\n /** @internal */\r\n public _alphaState = new AlphaState();\r\n /** @internal */\r\n public _alphaMode = Constants.ALPHA_ADD;\r\n /** @internal */\r\n public _alphaEquation = Constants.ALPHA_DISABLE;\r\n\r\n // Cache\r\n /** @internal */\r\n public _internalTexturesCache = new Array();\r\n /** @internal */\r\n public _renderTargetWrapperCache = new Array();\r\n /** @internal */\r\n protected _activeChannel = 0;\r\n private _currentTextureChannel = -1;\r\n /** @internal */\r\n protected _boundTexturesCache: { [key: string]: Nullable } = {};\r\n protected _currentEffect: Nullable;\r\n /** @internal */\r\n public _currentDrawContext: IDrawContext;\r\n /** @internal */\r\n public _currentMaterialContext: IMaterialContext;\r\n /** @internal */\r\n protected _currentProgram: Nullable;\r\n protected _compiledEffects: { [key: string]: Effect } = {};\r\n private _vertexAttribArraysEnabled: boolean[] = [];\r\n /** @internal */\r\n protected _cachedViewport: Nullable;\r\n private _cachedVertexArrayObject: Nullable;\r\n /** @internal */\r\n protected _cachedVertexBuffers: any;\r\n /** @internal */\r\n protected _cachedIndexBuffer: Nullable;\r\n /** @internal */\r\n protected _cachedEffectForVertexBuffers: Nullable;\r\n\r\n /** @internal */\r\n public _currentRenderTarget: Nullable;\r\n private _uintIndicesCurrentlySet = false;\r\n protected _currentBoundBuffer = new Array>();\r\n /** @internal */\r\n public _currentFramebuffer: Nullable = null;\r\n /** @internal */\r\n public _dummyFramebuffer: Nullable = null;\r\n private _currentBufferPointers = new Array();\r\n private _currentInstanceLocations = new Array();\r\n private _currentInstanceBuffers = new Array();\r\n private _textureUnits: Int32Array;\r\n\r\n /** @internal */\r\n public _workingCanvas: Nullable;\r\n /** @internal */\r\n public _workingContext: Nullable;\r\n\r\n /** @internal */\r\n public _boundRenderFunction: any;\r\n\r\n private _vaoRecordInProgress = false;\r\n private _mustWipeVertexAttributes = false;\r\n\r\n private _emptyTexture: Nullable;\r\n private _emptyCubeTexture: Nullable;\r\n private _emptyTexture3D: Nullable;\r\n private _emptyTexture2DArray: Nullable;\r\n\r\n /** @internal */\r\n public _frameHandler: number;\r\n\r\n private _nextFreeTextureSlots = new Array();\r\n private _maxSimultaneousTextures = 0;\r\n private _maxMSAASamplesOverride: Nullable = null;\r\n\r\n private _activeRequests = new Array();\r\n\r\n /**\r\n * If set to true zooming in and out in the browser will rescale the hardware-scaling correctly.\r\n */\r\n public adaptToDeviceRatio: boolean = false;\r\n /** @internal */\r\n protected _lastDevicePixelRatio: number = 1.0;\r\n\r\n /** @internal */\r\n public _transformTextureUrl: Nullable<(url: string) => string> = null;\r\n\r\n /**\r\n * Gets information about the current host\r\n */\r\n public hostInformation: HostInformation = {\r\n isMobile: false,\r\n };\r\n\r\n protected get _supportsHardwareTextureRescaling() {\r\n return false;\r\n }\r\n\r\n private _framebufferDimensionsObject: Nullable<{ framebufferWidth: number; framebufferHeight: number }>;\r\n\r\n /**\r\n * sets the object from which width and height will be taken from when getting render width and height\r\n * Will fallback to the gl object\r\n * @param dimensions the framebuffer width and height that will be used.\r\n */\r\n public set framebufferDimensionsObject(dimensions: Nullable<{ framebufferWidth: number; framebufferHeight: number }>) {\r\n this._framebufferDimensionsObject = dimensions;\r\n }\r\n\r\n /**\r\n * Gets the current viewport\r\n */\r\n public get currentViewport(): Nullable {\r\n return this._cachedViewport;\r\n }\r\n\r\n /**\r\n * Gets the default empty texture\r\n */\r\n public get emptyTexture(): InternalTexture {\r\n if (!this._emptyTexture) {\r\n this._emptyTexture = this.createRawTexture(new Uint8Array(4), 1, 1, Constants.TEXTUREFORMAT_RGBA, false, false, Constants.TEXTURE_NEAREST_SAMPLINGMODE);\r\n }\r\n\r\n return this._emptyTexture;\r\n }\r\n\r\n /**\r\n * Gets the default empty 3D texture\r\n */\r\n public get emptyTexture3D(): InternalTexture {\r\n if (!this._emptyTexture3D) {\r\n this._emptyTexture3D = this.createRawTexture3D(new Uint8Array(4), 1, 1, 1, Constants.TEXTUREFORMAT_RGBA, false, false, Constants.TEXTURE_NEAREST_SAMPLINGMODE);\r\n }\r\n\r\n return this._emptyTexture3D;\r\n }\r\n\r\n /**\r\n * Gets the default empty 2D array texture\r\n */\r\n public get emptyTexture2DArray(): InternalTexture {\r\n if (!this._emptyTexture2DArray) {\r\n this._emptyTexture2DArray = this.createRawTexture2DArray(\r\n new Uint8Array(4),\r\n 1,\r\n 1,\r\n 1,\r\n Constants.TEXTUREFORMAT_RGBA,\r\n false,\r\n false,\r\n Constants.TEXTURE_NEAREST_SAMPLINGMODE\r\n );\r\n }\r\n\r\n return this._emptyTexture2DArray;\r\n }\r\n\r\n /**\r\n * Gets the default empty cube texture\r\n */\r\n public get emptyCubeTexture(): InternalTexture {\r\n if (!this._emptyCubeTexture) {\r\n const faceData = new Uint8Array(4);\r\n const cubeData = [faceData, faceData, faceData, faceData, faceData, faceData];\r\n this._emptyCubeTexture = this.createRawCubeTexture(\r\n cubeData,\r\n 1,\r\n Constants.TEXTUREFORMAT_RGBA,\r\n Constants.TEXTURETYPE_UNSIGNED_INT,\r\n false,\r\n false,\r\n Constants.TEXTURE_NEAREST_SAMPLINGMODE\r\n );\r\n }\r\n\r\n return this._emptyCubeTexture;\r\n }\r\n\r\n /**\r\n * Defines whether the engine has been created with the premultipliedAlpha option on or not.\r\n */\r\n public premultipliedAlpha: boolean = true;\r\n\r\n /**\r\n * Observable event triggered before each texture is initialized\r\n */\r\n public onBeforeTextureInitObservable = new Observable();\r\n\r\n /** @internal */\r\n protected _isWebGPU: boolean = false;\r\n /**\r\n * Gets a boolean indicating if the engine runs in WebGPU or not.\r\n */\r\n public get isWebGPU(): boolean {\r\n return this._isWebGPU;\r\n }\r\n\r\n /** @internal */\r\n protected _shaderPlatformName: string;\r\n /**\r\n * Gets the shader platform name used by the effects.\r\n */\r\n public get shaderPlatformName(): string {\r\n return this._shaderPlatformName;\r\n }\r\n\r\n /**\r\n * Enables or disables the snapshot rendering mode\r\n * Note that the WebGL engine does not support snapshot rendering so setting the value won't have any effect for this engine\r\n */\r\n public get snapshotRendering(): boolean {\r\n return false;\r\n }\r\n\r\n public set snapshotRendering(activate) {\r\n // WebGL engine does not support snapshot rendering\r\n }\r\n\r\n protected _snapshotRenderingMode = Constants.SNAPSHOTRENDERING_STANDARD;\r\n /**\r\n * Gets or sets the snapshot rendering mode\r\n */\r\n public get snapshotRenderingMode(): number {\r\n return this._snapshotRenderingMode;\r\n }\r\n\r\n public set snapshotRenderingMode(mode: number) {\r\n this._snapshotRenderingMode = mode;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if the exact sRGB conversions or faster approximations are used for converting to and from linear space.\r\n */\r\n public readonly useExactSrgbConversions: boolean;\r\n\r\n /**\r\n * Creates a new snapshot at the next frame using the current snapshotRenderingMode\r\n */\r\n public snapshotRenderingReset(): void {\r\n this.snapshotRendering = false;\r\n }\r\n\r\n private _checkForMobile: () => void;\r\n\r\n private static _CreateCanvas(width: number, height: number): ICanvas {\r\n if (typeof document === \"undefined\") {\r\n return (new OffscreenCanvas(width, height));\r\n }\r\n const canvas = (document.createElement(\"canvas\"));\r\n canvas.width = width;\r\n canvas.height = height;\r\n return canvas;\r\n }\r\n\r\n /**\r\n * Create a canvas. This method is overridden by other engines\r\n * @param width width\r\n * @param height height\r\n * @returns ICanvas interface\r\n */\r\n public createCanvas(width: number, height: number): ICanvas {\r\n return ThinEngine._CreateCanvas(width, height);\r\n }\r\n\r\n /**\r\n * Create an image to use with canvas\r\n * @returns IImage interface\r\n */\r\n public createCanvasImage(): IImage {\r\n return document.createElement(\"img\");\r\n }\r\n\r\n /**\r\n * Creates a new engine\r\n * @param canvasOrContext defines the canvas or WebGL context to use for rendering. If you provide a WebGL context, Babylon.js will not hook events on the canvas (like pointers, keyboards, etc...) so no event observables will be available. This is mostly used when Babylon.js is used as a plugin on a system which already used the WebGL context\r\n * @param antialias defines enable antialiasing (default: false)\r\n * @param options defines further options to be sent to the getContext() function\r\n * @param adaptToDeviceRatio defines whether to adapt to the device's viewport characteristics (default: false)\r\n */\r\n constructor(\r\n canvasOrContext: Nullable,\r\n antialias?: boolean,\r\n options?: EngineOptions,\r\n adaptToDeviceRatio?: boolean\r\n ) {\r\n this.startTime = PrecisionDate.Now;\r\n\r\n let canvas: Nullable = null;\r\n\r\n options = options || {};\r\n\r\n this._creationOptions = options;\r\n\r\n // Save this off for use in resize().\r\n this.adaptToDeviceRatio = adaptToDeviceRatio ?? false;\r\n\r\n this._stencilStateComposer.stencilGlobal = this._stencilState;\r\n\r\n PerformanceConfigurator.SetMatrixPrecision(!!options.useHighPrecisionMatrix);\r\n\r\n options.antialias = antialias ?? options.antialias;\r\n options.deterministicLockstep = options.deterministicLockstep ?? false;\r\n options.lockstepMaxSteps = options.lockstepMaxSteps ?? 4;\r\n options.timeStep = options.timeStep ?? 1 / 60;\r\n options.audioEngine = options.audioEngine ?? true;\r\n options.stencil = options.stencil ?? true;\r\n\r\n this._audioContext = options.audioEngineOptions?.audioContext ?? null;\r\n this._audioDestination = options.audioEngineOptions?.audioDestination ?? null;\r\n this.premultipliedAlpha = options.premultipliedAlpha ?? true;\r\n this.useExactSrgbConversions = options.useExactSrgbConversions ?? false;\r\n this._doNotHandleContextLost = !!options.doNotHandleContextLost;\r\n this._isStencilEnable = options.stencil ? true : false;\r\n\r\n // Viewport\r\n adaptToDeviceRatio = adaptToDeviceRatio || options.adaptToDeviceRatio || false;\r\n\r\n const devicePixelRatio = IsWindowObjectExist() ? window.devicePixelRatio || 1.0 : 1.0;\r\n\r\n const limitDeviceRatio = options.limitDeviceRatio || devicePixelRatio;\r\n this._hardwareScalingLevel = adaptToDeviceRatio ? 1.0 / Math.min(limitDeviceRatio, devicePixelRatio) : 1.0;\r\n this._lastDevicePixelRatio = devicePixelRatio;\r\n\r\n if (!canvasOrContext) {\r\n return;\r\n }\r\n\r\n if ((canvasOrContext as any).getContext) {\r\n canvas = canvasOrContext;\r\n this._renderingCanvas = canvas;\r\n\r\n if (options.preserveDrawingBuffer === undefined) {\r\n options.preserveDrawingBuffer = false;\r\n }\r\n\r\n if (options.xrCompatible === undefined) {\r\n options.xrCompatible = true;\r\n }\r\n\r\n // Exceptions\r\n if (navigator && navigator.userAgent) {\r\n this._setupMobileChecks();\r\n\r\n const ua = navigator.userAgent;\r\n for (const exception of ThinEngine.ExceptionList) {\r\n const key = exception.key;\r\n const targets = exception.targets;\r\n const check = new RegExp(key);\r\n\r\n if (check.test(ua)) {\r\n if (exception.capture && exception.captureConstraint) {\r\n const capture = exception.capture;\r\n const constraint = exception.captureConstraint;\r\n\r\n const regex = new RegExp(capture);\r\n const matches = regex.exec(ua);\r\n\r\n if (matches && matches.length > 0) {\r\n const capturedValue = parseInt(matches[matches.length - 1]);\r\n if (capturedValue >= constraint) {\r\n continue;\r\n }\r\n }\r\n }\r\n\r\n for (const target of targets) {\r\n switch (target) {\r\n case \"uniformBuffer\":\r\n this.disableUniformBuffers = true;\r\n break;\r\n case \"vao\":\r\n this.disableVertexArrayObjects = true;\r\n break;\r\n case \"antialias\":\r\n options.antialias = false;\r\n break;\r\n case \"maxMSAASamples\":\r\n this._maxMSAASamplesOverride = 1;\r\n break;\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Context lost\r\n if (!this._doNotHandleContextLost) {\r\n this._onContextLost = (evt: Event) => {\r\n evt.preventDefault();\r\n this._contextWasLost = true;\r\n Logger.Warn(\"WebGL context lost.\");\r\n\r\n this.onContextLostObservable.notifyObservers(this);\r\n };\r\n\r\n this._onContextRestored = () => {\r\n this._restoreEngineAfterContextLost(this._initGLContext.bind(this));\r\n };\r\n\r\n canvas.addEventListener(\"webglcontextlost\", this._onContextLost, false);\r\n canvas.addEventListener(\"webglcontextrestored\", this._onContextRestored, false);\r\n\r\n options.powerPreference = options.powerPreference || \"high-performance\";\r\n }\r\n\r\n // Detect if we are running on a faulty buggy desktop OS.\r\n this._badDesktopOS = /^((?!chrome|android).)*safari/i.test(navigator.userAgent);\r\n if (this._badDesktopOS) {\r\n options.xrCompatible = false;\r\n }\r\n\r\n // GL\r\n if (!options.disableWebGL2Support) {\r\n try {\r\n this._gl = (canvas.getContext(\"webgl2\", options) || canvas.getContext(\"experimental-webgl2\", options));\r\n if (this._gl) {\r\n this._webGLVersion = 2.0;\r\n this._shaderPlatformName = \"WEBGL2\";\r\n\r\n // Prevent weird browsers to lie (yeah that happens!)\r\n if (!this._gl.deleteQuery) {\r\n this._webGLVersion = 1.0;\r\n this._shaderPlatformName = \"WEBGL1\";\r\n }\r\n }\r\n } catch (e) {\r\n // Do nothing\r\n }\r\n }\r\n\r\n if (!this._gl) {\r\n if (!canvas) {\r\n throw new Error(\"The provided canvas is null or undefined.\");\r\n }\r\n try {\r\n this._gl = (canvas.getContext(\"webgl\", options) || canvas.getContext(\"experimental-webgl\", options));\r\n } catch (e) {\r\n throw new Error(\"WebGL not supported\");\r\n }\r\n }\r\n\r\n if (!this._gl) {\r\n throw new Error(\"WebGL not supported\");\r\n }\r\n } else {\r\n this._gl = canvasOrContext;\r\n this._renderingCanvas = this._gl.canvas as HTMLCanvasElement;\r\n\r\n if ((this._gl as any).renderbufferStorageMultisample) {\r\n this._webGLVersion = 2.0;\r\n this._shaderPlatformName = \"WEBGL2\";\r\n } else {\r\n this._shaderPlatformName = \"WEBGL1\";\r\n }\r\n\r\n const attributes = this._gl.getContextAttributes();\r\n if (attributes) {\r\n options.stencil = attributes.stencil;\r\n }\r\n }\r\n\r\n // Ensures a consistent color space unpacking of textures cross browser.\r\n this._gl.pixelStorei(this._gl.UNPACK_COLORSPACE_CONVERSION_WEBGL, this._gl.NONE);\r\n\r\n if (options.useHighPrecisionFloats !== undefined) {\r\n this._highPrecisionShadersAllowed = options.useHighPrecisionFloats;\r\n }\r\n\r\n this.resize();\r\n\r\n this._initGLContext();\r\n this._initFeatures();\r\n\r\n // Prepare buffer pointers\r\n for (let i = 0; i < this._caps.maxVertexAttribs; i++) {\r\n this._currentBufferPointers[i] = new BufferPointer();\r\n }\r\n\r\n // Shader processor\r\n this._shaderProcessor = this.webGLVersion > 1 ? new WebGL2ShaderProcessor() : new WebGLShaderProcessor();\r\n\r\n // Detect if we are running on a faulty buggy OS.\r\n this._badOS = /iPad/i.test(navigator.userAgent) || /iPhone/i.test(navigator.userAgent);\r\n\r\n // Starting with iOS 14, we can trust the browser\r\n // let matches = navigator.userAgent.match(/Version\\/(\\d+)/);\r\n\r\n // if (matches && matches.length === 2) {\r\n // if (parseInt(matches[1]) >= 14) {\r\n // this._badOS = false;\r\n // }\r\n // }\r\n\r\n const versionToLog = `Babylon.js v${ThinEngine.Version}`;\r\n console.log(versionToLog + ` - ${this.description}`);\r\n\r\n // Check setAttribute in case of workers\r\n if (this._renderingCanvas && this._renderingCanvas.setAttribute) {\r\n this._renderingCanvas.setAttribute(\"data-engine\", versionToLog);\r\n }\r\n }\r\n\r\n protected _setupMobileChecks(): void {\r\n if (!(navigator && navigator.userAgent)) {\r\n return;\r\n }\r\n\r\n // Function to check if running on mobile device\r\n this._checkForMobile = () => {\r\n const currentUA = navigator.userAgent;\r\n this.hostInformation.isMobile =\r\n currentUA.indexOf(\"Mobile\") !== -1 ||\r\n // Needed for iOS 13+ detection on iPad (inspired by solution from https://stackoverflow.com/questions/9038625/detect-if-device-is-ios)\r\n (currentUA.indexOf(\"Mac\") !== -1 && IsDocumentAvailable() && \"ontouchend\" in document);\r\n };\r\n\r\n // Set initial isMobile value\r\n this._checkForMobile();\r\n\r\n // Set up event listener to check when window is resized (used to get emulator activation to work properly)\r\n if (IsWindowObjectExist()) {\r\n window.addEventListener(\"resize\", this._checkForMobile);\r\n }\r\n }\r\n\r\n protected _restoreEngineAfterContextLost(initEngine: () => void): void {\r\n // Adding a timeout to avoid race condition at browser level\r\n setTimeout(async () => {\r\n this._dummyFramebuffer = null;\r\n\r\n const depthTest = this._depthCullingState.depthTest; // backup those values because the call to initEngine / wipeCaches will reset them\r\n const depthFunc = this._depthCullingState.depthFunc;\r\n const depthMask = this._depthCullingState.depthMask;\r\n const stencilTest = this._stencilState.stencilTest;\r\n\r\n // Rebuild context\r\n await initEngine();\r\n\r\n // Ensure webgl and engine states are matching\r\n this.wipeCaches(true);\r\n\r\n // Rebuild effects\r\n this._rebuildEffects();\r\n this._rebuildComputeEffects?.();\r\n\r\n // Note:\r\n // The call to _rebuildBuffers must be made before the call to _rebuildInternalTextures because in the process of _rebuildBuffers the buffers used by the post process managers will be rebuilt\r\n // and we may need to use the post process manager of the scene during _rebuildInternalTextures (in WebGL1, non-POT textures are rescaled using a post process + post process manager of the scene)\r\n\r\n // Rebuild buffers\r\n this._rebuildBuffers();\r\n // Rebuild textures\r\n this._rebuildInternalTextures();\r\n // Rebuild textures\r\n this._rebuildRenderTargetWrappers();\r\n\r\n // Reset engine states after all the buffer/textures/... have been rebuilt\r\n this.wipeCaches(true);\r\n\r\n this._depthCullingState.depthTest = depthTest;\r\n this._depthCullingState.depthFunc = depthFunc;\r\n this._depthCullingState.depthMask = depthMask;\r\n this._stencilState.stencilTest = stencilTest;\r\n\r\n Logger.Warn(this.name + \" context successfully restored.\");\r\n this.onContextRestoredObservable.notifyObservers(this);\r\n this._contextWasLost = false;\r\n }, 0);\r\n }\r\n\r\n /**\r\n * Shared initialization across engines types.\r\n * @param canvas The canvas associated with this instance of the engine.\r\n */\r\n protected _sharedInit(canvas: HTMLCanvasElement) {\r\n this._renderingCanvas = canvas;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getShaderProcessingContext(shaderLanguage: ShaderLanguage): Nullable {\r\n return null;\r\n }\r\n\r\n private _rebuildInternalTextures(): void {\r\n const currentState = this._internalTexturesCache.slice(); // Do a copy because the rebuild will add proxies\r\n\r\n for (const internalTexture of currentState) {\r\n internalTexture._rebuild();\r\n }\r\n }\r\n\r\n private _rebuildRenderTargetWrappers(): void {\r\n const currentState = this._renderTargetWrapperCache.slice(); // Do a copy because the rebuild will add proxies\r\n\r\n for (const renderTargetWrapper of currentState) {\r\n renderTargetWrapper._rebuild();\r\n }\r\n }\r\n\r\n private _rebuildEffects(): void {\r\n for (const key in this._compiledEffects) {\r\n const effect = this._compiledEffects[key];\r\n\r\n effect._pipelineContext = null; // because _prepareEffect will try to dispose this pipeline before recreating it and that would lead to webgl errors\r\n effect._wasPreviouslyReady = false;\r\n effect._prepareEffect();\r\n }\r\n\r\n Effect.ResetCache();\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if all created effects are ready\r\n * @returns true if all effects are ready\r\n */\r\n public areAllEffectsReady(): boolean {\r\n for (const key in this._compiledEffects) {\r\n const effect = this._compiledEffects[key];\r\n\r\n if (!effect.isReady()) {\r\n return false;\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n protected _rebuildBuffers(): void {\r\n // Uniforms\r\n for (const uniformBuffer of this._uniformBuffers) {\r\n uniformBuffer._rebuild();\r\n }\r\n // Storage buffers\r\n for (const storageBuffer of this._storageBuffers) {\r\n storageBuffer._rebuild();\r\n }\r\n }\r\n\r\n protected _initGLContext(): void {\r\n // Caps\r\n this._caps = {\r\n maxTexturesImageUnits: this._gl.getParameter(this._gl.MAX_TEXTURE_IMAGE_UNITS),\r\n maxCombinedTexturesImageUnits: this._gl.getParameter(this._gl.MAX_COMBINED_TEXTURE_IMAGE_UNITS),\r\n maxVertexTextureImageUnits: this._gl.getParameter(this._gl.MAX_VERTEX_TEXTURE_IMAGE_UNITS),\r\n maxTextureSize: this._gl.getParameter(this._gl.MAX_TEXTURE_SIZE),\r\n maxSamples: this._webGLVersion > 1 ? this._gl.getParameter(this._gl.MAX_SAMPLES) : 1,\r\n maxCubemapTextureSize: this._gl.getParameter(this._gl.MAX_CUBE_MAP_TEXTURE_SIZE),\r\n maxRenderTextureSize: this._gl.getParameter(this._gl.MAX_RENDERBUFFER_SIZE),\r\n maxVertexAttribs: this._gl.getParameter(this._gl.MAX_VERTEX_ATTRIBS),\r\n maxVaryingVectors: this._gl.getParameter(this._gl.MAX_VARYING_VECTORS),\r\n maxFragmentUniformVectors: this._gl.getParameter(this._gl.MAX_FRAGMENT_UNIFORM_VECTORS),\r\n maxVertexUniformVectors: this._gl.getParameter(this._gl.MAX_VERTEX_UNIFORM_VECTORS),\r\n parallelShaderCompile: this._gl.getExtension(\"KHR_parallel_shader_compile\") || undefined,\r\n standardDerivatives: this._webGLVersion > 1 || this._gl.getExtension(\"OES_standard_derivatives\") !== null,\r\n maxAnisotropy: 1,\r\n astc: this._gl.getExtension(\"WEBGL_compressed_texture_astc\") || this._gl.getExtension(\"WEBKIT_WEBGL_compressed_texture_astc\"),\r\n bptc: this._gl.getExtension(\"EXT_texture_compression_bptc\") || this._gl.getExtension(\"WEBKIT_EXT_texture_compression_bptc\"),\r\n s3tc: this._gl.getExtension(\"WEBGL_compressed_texture_s3tc\") || this._gl.getExtension(\"WEBKIT_WEBGL_compressed_texture_s3tc\"),\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n s3tc_srgb: this._gl.getExtension(\"WEBGL_compressed_texture_s3tc_srgb\") || this._gl.getExtension(\"WEBKIT_WEBGL_compressed_texture_s3tc_srgb\"),\r\n pvrtc: this._gl.getExtension(\"WEBGL_compressed_texture_pvrtc\") || this._gl.getExtension(\"WEBKIT_WEBGL_compressed_texture_pvrtc\"),\r\n etc1: this._gl.getExtension(\"WEBGL_compressed_texture_etc1\") || this._gl.getExtension(\"WEBKIT_WEBGL_compressed_texture_etc1\"),\r\n etc2:\r\n this._gl.getExtension(\"WEBGL_compressed_texture_etc\") ||\r\n this._gl.getExtension(\"WEBKIT_WEBGL_compressed_texture_etc\") ||\r\n this._gl.getExtension(\"WEBGL_compressed_texture_es3_0\"), // also a requirement of OpenGL ES 3\r\n textureAnisotropicFilterExtension:\r\n this._gl.getExtension(\"EXT_texture_filter_anisotropic\") ||\r\n this._gl.getExtension(\"WEBKIT_EXT_texture_filter_anisotropic\") ||\r\n this._gl.getExtension(\"MOZ_EXT_texture_filter_anisotropic\"),\r\n uintIndices: this._webGLVersion > 1 || this._gl.getExtension(\"OES_element_index_uint\") !== null,\r\n fragmentDepthSupported: this._webGLVersion > 1 || this._gl.getExtension(\"EXT_frag_depth\") !== null,\r\n highPrecisionShaderSupported: false,\r\n timerQuery: this._gl.getExtension(\"EXT_disjoint_timer_query_webgl2\") || this._gl.getExtension(\"EXT_disjoint_timer_query\"),\r\n supportOcclusionQuery: this._webGLVersion > 1,\r\n canUseTimestampForTimerQuery: false,\r\n drawBuffersExtension: false,\r\n maxMSAASamples: 1,\r\n colorBufferFloat: !!(this._webGLVersion > 1 && this._gl.getExtension(\"EXT_color_buffer_float\")),\r\n colorBufferHalfFloat: !!(this._webGLVersion > 1 && this._gl.getExtension(\"EXT_color_buffer_half_float\")),\r\n textureFloat: this._webGLVersion > 1 || this._gl.getExtension(\"OES_texture_float\") ? true : false,\r\n textureHalfFloat: this._webGLVersion > 1 || this._gl.getExtension(\"OES_texture_half_float\") ? true : false,\r\n textureHalfFloatRender: false,\r\n textureFloatLinearFiltering: false,\r\n textureFloatRender: false,\r\n textureHalfFloatLinearFiltering: false,\r\n vertexArrayObject: false,\r\n instancedArrays: false,\r\n textureLOD: this._webGLVersion > 1 || this._gl.getExtension(\"EXT_shader_texture_lod\") ? true : false,\r\n texelFetch: this._webGLVersion !== 1,\r\n blendMinMax: false,\r\n multiview: this._gl.getExtension(\"OVR_multiview2\"),\r\n oculusMultiview: this._gl.getExtension(\"OCULUS_multiview\"),\r\n depthTextureExtension: false,\r\n canUseGLInstanceID: this._webGLVersion > 1,\r\n canUseGLVertexID: this._webGLVersion > 1,\r\n supportComputeShaders: false,\r\n supportSRGBBuffers: false,\r\n supportTransformFeedbacks: this._webGLVersion > 1,\r\n textureMaxLevel: this._webGLVersion > 1,\r\n texture2DArrayMaxLayerCount: this._webGLVersion > 1 ? this._gl.getParameter(this._gl.MAX_ARRAY_TEXTURE_LAYERS) : 128,\r\n disableMorphTargetTexture: false,\r\n };\r\n\r\n // Infos\r\n this._glVersion = this._gl.getParameter(this._gl.VERSION);\r\n\r\n const rendererInfo: any = this._gl.getExtension(\"WEBGL_debug_renderer_info\");\r\n if (rendererInfo != null) {\r\n this._glRenderer = this._gl.getParameter(rendererInfo.UNMASKED_RENDERER_WEBGL);\r\n this._glVendor = this._gl.getParameter(rendererInfo.UNMASKED_VENDOR_WEBGL);\r\n }\r\n\r\n if (!this._glVendor) {\r\n this._glVendor = this._gl.getParameter(this._gl.VENDOR) || \"Unknown vendor\";\r\n }\r\n\r\n if (!this._glRenderer) {\r\n this._glRenderer = this._gl.getParameter(this._gl.RENDERER) || \"Unknown renderer\";\r\n }\r\n\r\n // Constants\r\n if (this._gl.HALF_FLOAT_OES !== 0x8d61) {\r\n this._gl.HALF_FLOAT_OES = 0x8d61; // Half floating-point type (16-bit).\r\n }\r\n if (this._gl.RGBA16F !== 0x881a) {\r\n this._gl.RGBA16F = 0x881a; // RGBA 16-bit floating-point color-renderable internal sized format.\r\n }\r\n if (this._gl.RGBA32F !== 0x8814) {\r\n this._gl.RGBA32F = 0x8814; // RGBA 32-bit floating-point color-renderable internal sized format.\r\n }\r\n if (this._gl.DEPTH24_STENCIL8 !== 35056) {\r\n this._gl.DEPTH24_STENCIL8 = 35056;\r\n }\r\n\r\n // Extensions\r\n if (this._caps.timerQuery) {\r\n if (this._webGLVersion === 1) {\r\n this._gl.getQuery = (this._caps.timerQuery).getQueryEXT.bind(this._caps.timerQuery);\r\n }\r\n // WebGLQuery casted to number to avoid TS error\r\n this._caps.canUseTimestampForTimerQuery = ((this._gl.getQuery(this._caps.timerQuery.TIMESTAMP_EXT, this._caps.timerQuery.QUERY_COUNTER_BITS_EXT) as number) ?? 0) > 0;\r\n }\r\n\r\n this._caps.maxAnisotropy = this._caps.textureAnisotropicFilterExtension\r\n ? this._gl.getParameter(this._caps.textureAnisotropicFilterExtension.MAX_TEXTURE_MAX_ANISOTROPY_EXT)\r\n : 0;\r\n this._caps.textureFloatLinearFiltering = this._caps.textureFloat && this._gl.getExtension(\"OES_texture_float_linear\") ? true : false;\r\n this._caps.textureFloatRender = this._caps.textureFloat && this._canRenderToFloatFramebuffer() ? true : false;\r\n this._caps.textureHalfFloatLinearFiltering =\r\n this._webGLVersion > 1 || (this._caps.textureHalfFloat && this._gl.getExtension(\"OES_texture_half_float_linear\")) ? true : false;\r\n\r\n // Compressed formats\r\n if (this._caps.astc) {\r\n this._gl.COMPRESSED_SRGB8_ALPHA8_ASTC_4x4_KHR = this._caps.astc.COMPRESSED_SRGB8_ALPHA8_ASTC_4x4_KHR;\r\n }\r\n if (this._caps.bptc) {\r\n this._gl.COMPRESSED_SRGB_ALPHA_BPTC_UNORM_EXT = this._caps.bptc.COMPRESSED_SRGB_ALPHA_BPTC_UNORM_EXT;\r\n }\r\n if (this._caps.s3tc_srgb) {\r\n this._gl.COMPRESSED_SRGB_S3TC_DXT1_EXT = this._caps.s3tc_srgb.COMPRESSED_SRGB_S3TC_DXT1_EXT;\r\n this._gl.COMPRESSED_SRGB_ALPHA_S3TC_DXT1_EXT = this._caps.s3tc_srgb.COMPRESSED_SRGB_ALPHA_S3TC_DXT1_EXT;\r\n this._gl.COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT = this._caps.s3tc_srgb.COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT;\r\n }\r\n if (this._caps.etc2) {\r\n this._gl.COMPRESSED_SRGB8_ETC2 = this._caps.etc2.COMPRESSED_SRGB8_ETC2;\r\n this._gl.COMPRESSED_SRGB8_ALPHA8_ETC2_EAC = this._caps.etc2.COMPRESSED_SRGB8_ALPHA8_ETC2_EAC;\r\n }\r\n\r\n // Checks if some of the format renders first to allow the use of webgl inspector.\r\n if (this._webGLVersion > 1) {\r\n if (this._gl.HALF_FLOAT_OES !== 0x140b) {\r\n this._gl.HALF_FLOAT_OES = 0x140b;\r\n }\r\n }\r\n this._caps.textureHalfFloatRender = this._caps.textureHalfFloat && this._canRenderToHalfFloatFramebuffer();\r\n // Draw buffers\r\n if (this._webGLVersion > 1) {\r\n this._caps.drawBuffersExtension = true;\r\n this._caps.maxMSAASamples = this._maxMSAASamplesOverride !== null ? this._maxMSAASamplesOverride : this._gl.getParameter(this._gl.MAX_SAMPLES);\r\n } else {\r\n const drawBuffersExtension = this._gl.getExtension(\"WEBGL_draw_buffers\");\r\n\r\n if (drawBuffersExtension !== null) {\r\n this._caps.drawBuffersExtension = true;\r\n this._gl.drawBuffers = drawBuffersExtension.drawBuffersWEBGL.bind(drawBuffersExtension);\r\n (this._gl.DRAW_FRAMEBUFFER as any) = this._gl.FRAMEBUFFER;\r\n\r\n for (let i = 0; i < 16; i++) {\r\n (this._gl)[\"COLOR_ATTACHMENT\" + i + \"_WEBGL\"] = (drawBuffersExtension)[\"COLOR_ATTACHMENT\" + i + \"_WEBGL\"];\r\n }\r\n }\r\n }\r\n\r\n // Depth Texture\r\n if (this._webGLVersion > 1) {\r\n this._caps.depthTextureExtension = true;\r\n } else {\r\n const depthTextureExtension = this._gl.getExtension(\"WEBGL_depth_texture\");\r\n\r\n if (depthTextureExtension != null) {\r\n this._caps.depthTextureExtension = true;\r\n this._gl.UNSIGNED_INT_24_8 = depthTextureExtension.UNSIGNED_INT_24_8_WEBGL;\r\n }\r\n }\r\n\r\n // Vertex array object\r\n if (this.disableVertexArrayObjects) {\r\n this._caps.vertexArrayObject = false;\r\n } else if (this._webGLVersion > 1) {\r\n this._caps.vertexArrayObject = true;\r\n } else {\r\n const vertexArrayObjectExtension = this._gl.getExtension(\"OES_vertex_array_object\");\r\n\r\n if (vertexArrayObjectExtension != null) {\r\n this._caps.vertexArrayObject = true;\r\n this._gl.createVertexArray = vertexArrayObjectExtension.createVertexArrayOES.bind(vertexArrayObjectExtension);\r\n this._gl.bindVertexArray = vertexArrayObjectExtension.bindVertexArrayOES.bind(vertexArrayObjectExtension);\r\n this._gl.deleteVertexArray = vertexArrayObjectExtension.deleteVertexArrayOES.bind(vertexArrayObjectExtension);\r\n }\r\n }\r\n\r\n // Instances count\r\n if (this._webGLVersion > 1) {\r\n this._caps.instancedArrays = true;\r\n } else {\r\n const instanceExtension = this._gl.getExtension(\"ANGLE_instanced_arrays\");\r\n\r\n if (instanceExtension != null) {\r\n this._caps.instancedArrays = true;\r\n this._gl.drawArraysInstanced = instanceExtension.drawArraysInstancedANGLE.bind(instanceExtension);\r\n this._gl.drawElementsInstanced = instanceExtension.drawElementsInstancedANGLE.bind(instanceExtension);\r\n this._gl.vertexAttribDivisor = instanceExtension.vertexAttribDivisorANGLE.bind(instanceExtension);\r\n } else {\r\n this._caps.instancedArrays = false;\r\n }\r\n }\r\n\r\n if (this._gl.getShaderPrecisionFormat) {\r\n const vertexhighp = this._gl.getShaderPrecisionFormat(this._gl.VERTEX_SHADER, this._gl.HIGH_FLOAT);\r\n const fragmenthighp = this._gl.getShaderPrecisionFormat(this._gl.FRAGMENT_SHADER, this._gl.HIGH_FLOAT);\r\n\r\n if (vertexhighp && fragmenthighp) {\r\n this._caps.highPrecisionShaderSupported = vertexhighp.precision !== 0 && fragmenthighp.precision !== 0;\r\n }\r\n }\r\n\r\n if (this._webGLVersion > 1) {\r\n this._caps.blendMinMax = true;\r\n } else {\r\n const blendMinMaxExtension = this._gl.getExtension(\"EXT_blend_minmax\");\r\n if (blendMinMaxExtension != null) {\r\n this._caps.blendMinMax = true;\r\n this._gl.MAX = blendMinMaxExtension.MAX_EXT as typeof WebGL2RenderingContext.MAX;\r\n this._gl.MIN = blendMinMaxExtension.MIN_EXT as typeof WebGL2RenderingContext.MIN;\r\n }\r\n }\r\n\r\n // sRGB buffers\r\n // only run this if not already set to true (in the constructor, for example)\r\n if (!this._caps.supportSRGBBuffers) {\r\n if (this._webGLVersion > 1) {\r\n this._caps.supportSRGBBuffers = true;\r\n this._glSRGBExtensionValues = {\r\n SRGB: WebGL2RenderingContext.SRGB,\r\n SRGB8: WebGL2RenderingContext.SRGB8,\r\n SRGB8_ALPHA8: WebGL2RenderingContext.SRGB8_ALPHA8,\r\n };\r\n } else {\r\n const sRGBExtension = this._gl.getExtension(\"EXT_sRGB\");\r\n\r\n if (sRGBExtension != null) {\r\n this._caps.supportSRGBBuffers = true;\r\n this._glSRGBExtensionValues = {\r\n SRGB: sRGBExtension.SRGB_EXT as typeof WebGL2RenderingContext.SRGB | EXT_sRGB[\"SRGB_EXT\"],\r\n SRGB8: sRGBExtension.SRGB_ALPHA_EXT as typeof WebGL2RenderingContext.SRGB8 | EXT_sRGB[\"SRGB_ALPHA_EXT\"],\r\n SRGB8_ALPHA8: sRGBExtension.SRGB_ALPHA_EXT as typeof WebGL2RenderingContext.SRGB8_ALPHA8 | EXT_sRGB[\"SRGB8_ALPHA8_EXT\"],\r\n };\r\n }\r\n }\r\n // take into account the forced state that was provided in options\r\n // When the issue in angle/chrome is fixed the flag should be taken into account only when it is explicitly defined\r\n this._caps.supportSRGBBuffers = this._caps.supportSRGBBuffers && !!(this._creationOptions && this._creationOptions.forceSRGBBufferSupportState);\r\n }\r\n\r\n // Depth buffer\r\n this._depthCullingState.depthTest = true;\r\n this._depthCullingState.depthFunc = this._gl.LEQUAL;\r\n this._depthCullingState.depthMask = true;\r\n\r\n // Texture maps\r\n this._maxSimultaneousTextures = this._caps.maxCombinedTexturesImageUnits;\r\n for (let slot = 0; slot < this._maxSimultaneousTextures; slot++) {\r\n this._nextFreeTextureSlots.push(slot);\r\n }\r\n\r\n if (this._glRenderer === \"Mali-G72\") {\r\n // Overcome a bug when using a texture to store morph targets on Mali-G72\r\n this._caps.disableMorphTargetTexture = true;\r\n }\r\n }\r\n\r\n protected _initFeatures(): void {\r\n this._features = {\r\n forceBitmapOverHTMLImageElement: false,\r\n supportRenderAndCopyToLodForFloatTextures: this._webGLVersion !== 1,\r\n supportDepthStencilTexture: this._webGLVersion !== 1,\r\n supportShadowSamplers: this._webGLVersion !== 1,\r\n uniformBufferHardCheckMatrix: false,\r\n allowTexturePrefiltering: this._webGLVersion !== 1,\r\n trackUbosInFrame: false,\r\n checkUbosContentBeforeUpload: false,\r\n supportCSM: this._webGLVersion !== 1,\r\n basisNeedsPOT: this._webGLVersion === 1,\r\n support3DTextures: this._webGLVersion !== 1,\r\n needTypeSuffixInShaderConstants: this._webGLVersion !== 1,\r\n supportMSAA: this._webGLVersion !== 1,\r\n supportSSAO2: this._webGLVersion !== 1,\r\n supportExtendedTextureFormats: this._webGLVersion !== 1,\r\n supportSwitchCaseInShader: this._webGLVersion !== 1,\r\n supportSyncTextureRead: true,\r\n needsInvertingBitmap: true,\r\n useUBOBindingCache: true,\r\n needShaderCodeInlining: false,\r\n needToAlwaysBindUniformBuffers: false,\r\n supportRenderPasses: false,\r\n supportSpriteInstancing: true,\r\n _collectUbosUpdatedInFrame: false,\r\n };\r\n }\r\n\r\n /**\r\n * Gets version of the current webGL context\r\n * Keep it for back compat - use version instead\r\n */\r\n public get webGLVersion(): number {\r\n return this._webGLVersion;\r\n }\r\n\r\n /**\r\n * Gets a string identifying the name of the class\r\n * @returns \"Engine\" string\r\n */\r\n public getClassName(): string {\r\n return \"ThinEngine\";\r\n }\r\n\r\n /**\r\n * Returns true if the stencil buffer has been enabled through the creation option of the context.\r\n */\r\n public get isStencilEnable(): boolean {\r\n return this._isStencilEnable;\r\n }\r\n\r\n /** @internal */\r\n public _prepareWorkingCanvas(): void {\r\n if (this._workingCanvas) {\r\n return;\r\n }\r\n\r\n this._workingCanvas = this.createCanvas(1, 1);\r\n const context = this._workingCanvas.getContext(\"2d\");\r\n\r\n if (context) {\r\n this._workingContext = context;\r\n }\r\n }\r\n\r\n /**\r\n * Reset the texture cache to empty state\r\n */\r\n public resetTextureCache() {\r\n for (const key in this._boundTexturesCache) {\r\n if (!Object.prototype.hasOwnProperty.call(this._boundTexturesCache, key)) {\r\n continue;\r\n }\r\n this._boundTexturesCache[key] = null;\r\n }\r\n\r\n this._currentTextureChannel = -1;\r\n }\r\n\r\n /**\r\n * Gets an object containing information about the current engine context\r\n * @returns an object containing the vendor, the renderer and the version of the current engine context\r\n */\r\n public getInfo() {\r\n return this.getGlInfo();\r\n }\r\n\r\n /**\r\n * Gets an object containing information about the current webGL context\r\n * @returns an object containing the vendor, the renderer and the version of the current webGL context\r\n */\r\n public getGlInfo() {\r\n return {\r\n vendor: this._glVendor,\r\n renderer: this._glRenderer,\r\n version: this._glVersion,\r\n };\r\n }\r\n\r\n /**\r\n * Defines the hardware scaling level.\r\n * By default the hardware scaling level is computed from the window device ratio.\r\n * if level = 1 then the engine will render at the exact resolution of the canvas. If level = 0.5 then the engine will render at twice the size of the canvas.\r\n * @param level defines the level to use\r\n */\r\n public setHardwareScalingLevel(level: number): void {\r\n this._hardwareScalingLevel = level;\r\n this.resize();\r\n }\r\n\r\n /**\r\n * Gets the current hardware scaling level.\r\n * By default the hardware scaling level is computed from the window device ratio.\r\n * if level = 1 then the engine will render at the exact resolution of the canvas. If level = 0.5 then the engine will render at twice the size of the canvas.\r\n * @returns a number indicating the current hardware scaling level\r\n */\r\n public getHardwareScalingLevel(): number {\r\n return this._hardwareScalingLevel;\r\n }\r\n\r\n /**\r\n * Gets the list of loaded textures\r\n * @returns an array containing all loaded textures\r\n */\r\n public getLoadedTexturesCache(): InternalTexture[] {\r\n return this._internalTexturesCache;\r\n }\r\n\r\n /**\r\n * Gets the object containing all engine capabilities\r\n * @returns the EngineCapabilities object\r\n */\r\n public getCaps(): EngineCapabilities {\r\n return this._caps;\r\n }\r\n\r\n /**\r\n * stop executing a render loop function and remove it from the execution array\r\n * @param renderFunction defines the function to be removed. If not provided all functions will be removed.\r\n */\r\n public stopRenderLoop(renderFunction?: () => void): void {\r\n if (!renderFunction) {\r\n this._activeRenderLoops.length = 0;\r\n this._cancelFrame();\r\n return;\r\n }\r\n\r\n const index = this._activeRenderLoops.indexOf(renderFunction);\r\n\r\n if (index >= 0) {\r\n this._activeRenderLoops.splice(index, 1);\r\n if (this._activeRenderLoops.length == 0) {\r\n this._cancelFrame();\r\n }\r\n }\r\n }\r\n\r\n protected _cancelFrame() {\r\n if (this._renderingQueueLaunched && this._frameHandler) {\r\n this._renderingQueueLaunched = false;\r\n if (!IsWindowObjectExist()) {\r\n if (typeof cancelAnimationFrame === \"function\") {\r\n return cancelAnimationFrame(this._frameHandler);\r\n }\r\n } else {\r\n const { cancelAnimationFrame } = this.getHostWindow() || window;\r\n if (typeof cancelAnimationFrame === \"function\") {\r\n return cancelAnimationFrame(this._frameHandler);\r\n }\r\n }\r\n return clearTimeout(this._frameHandler);\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _renderLoop(): void {\r\n if (!this._contextWasLost) {\r\n let shouldRender = true;\r\n if (this._isDisposed || (!this.renderEvenInBackground && this._windowIsBackground)) {\r\n shouldRender = false;\r\n }\r\n\r\n if (shouldRender) {\r\n // Start new frame\r\n this.beginFrame();\r\n\r\n for (let index = 0; index < this._activeRenderLoops.length; index++) {\r\n const renderFunction = this._activeRenderLoops[index];\r\n\r\n renderFunction();\r\n }\r\n\r\n // Present\r\n this.endFrame();\r\n }\r\n }\r\n\r\n if (this._activeRenderLoops.length > 0) {\r\n this._frameHandler = this._queueNewFrame(this._boundRenderFunction, this.getHostWindow());\r\n } else {\r\n this._renderingQueueLaunched = false;\r\n }\r\n }\r\n\r\n /**\r\n * Gets the HTML canvas attached with the current webGL context\r\n * @returns a HTML canvas\r\n */\r\n public getRenderingCanvas(): Nullable {\r\n return this._renderingCanvas;\r\n }\r\n\r\n /**\r\n * Gets the audio context specified in engine initialization options\r\n * @returns an Audio Context\r\n */\r\n public getAudioContext(): Nullable {\r\n return this._audioContext;\r\n }\r\n\r\n /**\r\n * Gets the audio destination specified in engine initialization options\r\n * @returns an audio destination node\r\n */\r\n public getAudioDestination(): Nullable {\r\n return this._audioDestination;\r\n }\r\n\r\n /**\r\n * Gets host window\r\n * @returns the host window object\r\n */\r\n public getHostWindow(): Nullable {\r\n if (!IsWindowObjectExist()) {\r\n return null;\r\n }\r\n\r\n if (this._renderingCanvas && this._renderingCanvas.ownerDocument && this._renderingCanvas.ownerDocument.defaultView) {\r\n return this._renderingCanvas.ownerDocument.defaultView;\r\n }\r\n\r\n return window;\r\n }\r\n\r\n /**\r\n * Gets the current render width\r\n * @param useScreen defines if screen size must be used (or the current render target if any)\r\n * @returns a number defining the current render width\r\n */\r\n public getRenderWidth(useScreen = false): number {\r\n if (!useScreen && this._currentRenderTarget) {\r\n return this._currentRenderTarget.width;\r\n }\r\n\r\n return this._framebufferDimensionsObject ? this._framebufferDimensionsObject.framebufferWidth : this._gl.drawingBufferWidth;\r\n }\r\n\r\n /**\r\n * Gets the current render height\r\n * @param useScreen defines if screen size must be used (or the current render target if any)\r\n * @returns a number defining the current render height\r\n */\r\n public getRenderHeight(useScreen = false): number {\r\n if (!useScreen && this._currentRenderTarget) {\r\n return this._currentRenderTarget.height;\r\n }\r\n\r\n return this._framebufferDimensionsObject ? this._framebufferDimensionsObject.framebufferHeight : this._gl.drawingBufferHeight;\r\n }\r\n\r\n /**\r\n * Can be used to override the current requestAnimationFrame requester.\r\n * @internal\r\n */\r\n protected _queueNewFrame(bindedRenderFunction: any, requester?: any): number {\r\n return ThinEngine.QueueNewFrame(bindedRenderFunction, requester);\r\n }\r\n\r\n /**\r\n * Register and execute a render loop. The engine can have more than one render function\r\n * @param renderFunction defines the function to continuously execute\r\n */\r\n public runRenderLoop(renderFunction: () => void): void {\r\n if (this._activeRenderLoops.indexOf(renderFunction) !== -1) {\r\n return;\r\n }\r\n\r\n this._activeRenderLoops.push(renderFunction);\r\n\r\n if (!this._renderingQueueLaunched) {\r\n this._renderingQueueLaunched = true;\r\n this._boundRenderFunction = this._renderLoop.bind(this);\r\n this._frameHandler = this._queueNewFrame(this._boundRenderFunction, this.getHostWindow());\r\n }\r\n }\r\n\r\n /**\r\n * Clear the current render buffer or the current render target (if any is set up)\r\n * @param color defines the color to use\r\n * @param backBuffer defines if the back buffer must be cleared\r\n * @param depth defines if the depth buffer must be cleared\r\n * @param stencil defines if the stencil buffer must be cleared\r\n */\r\n public clear(color: Nullable, backBuffer: boolean, depth: boolean, stencil: boolean = false): void {\r\n const useStencilGlobalOnly = this.stencilStateComposer.useStencilGlobalOnly;\r\n this.stencilStateComposer.useStencilGlobalOnly = true; // make sure the stencil mask is coming from the global stencil and not from a material (effect) which would currently be in effect\r\n\r\n this.applyStates();\r\n\r\n this.stencilStateComposer.useStencilGlobalOnly = useStencilGlobalOnly;\r\n\r\n let mode = 0;\r\n if (backBuffer && color) {\r\n let setBackBufferColor = true;\r\n if (this._currentRenderTarget) {\r\n const textureFormat = this._currentRenderTarget.texture?.format;\r\n if (\r\n textureFormat === Constants.TEXTUREFORMAT_RED_INTEGER ||\r\n textureFormat === Constants.TEXTUREFORMAT_RG_INTEGER ||\r\n textureFormat === Constants.TEXTUREFORMAT_RGB_INTEGER ||\r\n textureFormat === Constants.TEXTUREFORMAT_RGBA_INTEGER\r\n ) {\r\n const textureType = this._currentRenderTarget.texture?.type;\r\n if (textureType === Constants.TEXTURETYPE_UNSIGNED_INTEGER || textureType === Constants.TEXTURETYPE_UNSIGNED_SHORT) {\r\n ThinEngine._TempClearColorUint32[0] = color.r * 255;\r\n ThinEngine._TempClearColorUint32[1] = color.g * 255;\r\n ThinEngine._TempClearColorUint32[2] = color.b * 255;\r\n ThinEngine._TempClearColorUint32[3] = color.a * 255;\r\n this._gl.clearBufferuiv(this._gl.COLOR, 0, ThinEngine._TempClearColorUint32);\r\n setBackBufferColor = false;\r\n } else {\r\n ThinEngine._TempClearColorInt32[0] = color.r * 255;\r\n ThinEngine._TempClearColorInt32[1] = color.g * 255;\r\n ThinEngine._TempClearColorInt32[2] = color.b * 255;\r\n ThinEngine._TempClearColorInt32[3] = color.a * 255;\r\n this._gl.clearBufferiv(this._gl.COLOR, 0, ThinEngine._TempClearColorInt32);\r\n setBackBufferColor = false;\r\n }\r\n }\r\n }\r\n\r\n if (setBackBufferColor) {\r\n this._gl.clearColor(color.r, color.g, color.b, color.a !== undefined ? color.a : 1.0);\r\n mode |= this._gl.COLOR_BUFFER_BIT;\r\n }\r\n }\r\n\r\n if (depth) {\r\n if (this.useReverseDepthBuffer) {\r\n this._depthCullingState.depthFunc = this._gl.GEQUAL;\r\n this._gl.clearDepth(0.0);\r\n } else {\r\n this._gl.clearDepth(1.0);\r\n }\r\n mode |= this._gl.DEPTH_BUFFER_BIT;\r\n }\r\n if (stencil) {\r\n this._gl.clearStencil(0);\r\n mode |= this._gl.STENCIL_BUFFER_BIT;\r\n }\r\n this._gl.clear(mode);\r\n }\r\n\r\n protected _viewportCached = { x: 0, y: 0, z: 0, w: 0 };\r\n\r\n /**\r\n * @internal\r\n */\r\n public _viewport(x: number, y: number, width: number, height: number): void {\r\n if (x !== this._viewportCached.x || y !== this._viewportCached.y || width !== this._viewportCached.z || height !== this._viewportCached.w) {\r\n this._viewportCached.x = x;\r\n this._viewportCached.y = y;\r\n this._viewportCached.z = width;\r\n this._viewportCached.w = height;\r\n\r\n this._gl.viewport(x, y, width, height);\r\n }\r\n }\r\n\r\n /**\r\n * Set the WebGL's viewport\r\n * @param viewport defines the viewport element to be used\r\n * @param requiredWidth defines the width required for rendering. If not provided the rendering canvas' width is used\r\n * @param requiredHeight defines the height required for rendering. If not provided the rendering canvas' height is used\r\n */\r\n public setViewport(viewport: IViewportLike, requiredWidth?: number, requiredHeight?: number): void {\r\n const width = requiredWidth || this.getRenderWidth();\r\n const height = requiredHeight || this.getRenderHeight();\r\n const x = viewport.x || 0;\r\n const y = viewport.y || 0;\r\n\r\n this._cachedViewport = viewport;\r\n\r\n this._viewport(x * width, y * height, width * viewport.width, height * viewport.height);\r\n }\r\n\r\n /**\r\n * Begin a new frame\r\n */\r\n public beginFrame(): void {}\r\n\r\n /**\r\n * Enf the current frame\r\n */\r\n public endFrame(): void {\r\n // Force a flush in case we are using a bad OS.\r\n if (this._badOS) {\r\n this.flushFramebuffer();\r\n }\r\n this._frameId++;\r\n }\r\n\r\n /**\r\n * Resize the view according to the canvas' size\r\n * @param forceSetSize true to force setting the sizes of the underlying canvas\r\n */\r\n public resize(forceSetSize = false): void {\r\n let width: number;\r\n let height: number;\r\n\r\n // Re-query hardware scaling level to handle zoomed-in resizing.\r\n if (this.adaptToDeviceRatio) {\r\n const devicePixelRatio = IsWindowObjectExist() ? window.devicePixelRatio || 1.0 : 1.0;\r\n const changeRatio = this._lastDevicePixelRatio / devicePixelRatio;\r\n this._lastDevicePixelRatio = devicePixelRatio;\r\n this._hardwareScalingLevel *= changeRatio;\r\n }\r\n\r\n if (IsWindowObjectExist() && IsDocumentAvailable()) {\r\n // make sure it is a Node object, and is a part of the document.\r\n if (this._renderingCanvas) {\r\n const boundingRect = this._renderingCanvas.getBoundingClientRect\r\n ? this._renderingCanvas.getBoundingClientRect()\r\n : {\r\n // fallback to last solution in case the function doesn't exist\r\n width: this._renderingCanvas.width * this._hardwareScalingLevel,\r\n height: this._renderingCanvas.height * this._hardwareScalingLevel,\r\n };\r\n width = this._renderingCanvas.clientWidth || boundingRect.width || this._renderingCanvas.width || 100;\r\n height = this._renderingCanvas.clientHeight || boundingRect.height || this._renderingCanvas.height || 100;\r\n } else {\r\n width = window.innerWidth;\r\n height = window.innerHeight;\r\n }\r\n } else {\r\n width = this._renderingCanvas ? this._renderingCanvas.width : 100;\r\n height = this._renderingCanvas ? this._renderingCanvas.height : 100;\r\n }\r\n\r\n this.setSize(width / this._hardwareScalingLevel, height / this._hardwareScalingLevel, forceSetSize);\r\n }\r\n\r\n /**\r\n * Force a specific size of the canvas\r\n * @param width defines the new canvas' width\r\n * @param height defines the new canvas' height\r\n * @param forceSetSize true to force setting the sizes of the underlying canvas\r\n * @returns true if the size was changed\r\n */\r\n public setSize(width: number, height: number, forceSetSize = false): boolean {\r\n if (!this._renderingCanvas) {\r\n return false;\r\n }\r\n\r\n width = width | 0;\r\n height = height | 0;\r\n\r\n if (!forceSetSize && this._renderingCanvas.width === width && this._renderingCanvas.height === height) {\r\n return false;\r\n }\r\n\r\n this._renderingCanvas.width = width;\r\n this._renderingCanvas.height = height;\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Binds the frame buffer to the specified texture.\r\n * @param rtWrapper The render target wrapper to render to\r\n * @param faceIndex The face of the texture to render to in case of cube texture and if the render target wrapper is not a multi render target\r\n * @param requiredWidth The width of the target to render to\r\n * @param requiredHeight The height of the target to render to\r\n * @param forceFullscreenViewport Forces the viewport to be the entire texture/screen if true\r\n * @param lodLevel Defines the lod level to bind to the frame buffer\r\n * @param layer Defines the 2d array index to bind to the frame buffer if the render target wrapper is not a multi render target\r\n */\r\n public bindFramebuffer(\r\n rtWrapper: RenderTargetWrapper,\r\n faceIndex: number = 0,\r\n requiredWidth?: number,\r\n requiredHeight?: number,\r\n forceFullscreenViewport?: boolean,\r\n lodLevel = 0,\r\n layer = 0\r\n ): void {\r\n const webglRTWrapper = rtWrapper as WebGLRenderTargetWrapper;\r\n\r\n if (this._currentRenderTarget) {\r\n this.unBindFramebuffer(this._currentRenderTarget);\r\n }\r\n this._currentRenderTarget = rtWrapper;\r\n this._bindUnboundFramebuffer(webglRTWrapper._MSAAFramebuffer ? webglRTWrapper._MSAAFramebuffer : webglRTWrapper._framebuffer);\r\n\r\n const gl = this._gl;\r\n if (!rtWrapper.isMulti) {\r\n if (rtWrapper.is2DArray) {\r\n gl.framebufferTextureLayer(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, rtWrapper.texture!._hardwareTexture?.underlyingResource, lodLevel, layer);\r\n } else if (rtWrapper.isCube) {\r\n gl.framebufferTexture2D(\r\n gl.FRAMEBUFFER,\r\n gl.COLOR_ATTACHMENT0,\r\n gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex,\r\n rtWrapper.texture!._hardwareTexture?.underlyingResource,\r\n lodLevel\r\n );\r\n }\r\n }\r\n\r\n const depthStencilTexture = rtWrapper._depthStencilTexture;\r\n if (depthStencilTexture) {\r\n const attachment = rtWrapper._depthStencilTextureWithStencil ? gl.DEPTH_STENCIL_ATTACHMENT : gl.DEPTH_ATTACHMENT;\r\n if (rtWrapper.is2DArray) {\r\n gl.framebufferTextureLayer(gl.FRAMEBUFFER, attachment, depthStencilTexture._hardwareTexture?.underlyingResource, lodLevel, layer);\r\n } else if (rtWrapper.isCube) {\r\n gl.framebufferTexture2D(gl.FRAMEBUFFER, attachment, gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex, depthStencilTexture._hardwareTexture?.underlyingResource, lodLevel);\r\n } else {\r\n gl.framebufferTexture2D(gl.FRAMEBUFFER, attachment, gl.TEXTURE_2D, depthStencilTexture._hardwareTexture?.underlyingResource, lodLevel);\r\n }\r\n }\r\n\r\n if (this._cachedViewport && !forceFullscreenViewport) {\r\n this.setViewport(this._cachedViewport, requiredWidth, requiredHeight);\r\n } else {\r\n if (!requiredWidth) {\r\n requiredWidth = rtWrapper.width;\r\n if (lodLevel) {\r\n requiredWidth = requiredWidth / Math.pow(2, lodLevel);\r\n }\r\n }\r\n if (!requiredHeight) {\r\n requiredHeight = rtWrapper.height;\r\n if (lodLevel) {\r\n requiredHeight = requiredHeight / Math.pow(2, lodLevel);\r\n }\r\n }\r\n\r\n this._viewport(0, 0, requiredWidth, requiredHeight);\r\n }\r\n\r\n this.wipeCaches();\r\n }\r\n\r\n /**\r\n * Set various states to the webGL context\r\n * @param culling defines culling state: true to enable culling, false to disable it\r\n * @param zOffset defines the value to apply to zOffset (0 by default)\r\n * @param force defines if states must be applied even if cache is up to date\r\n * @param reverseSide defines if culling must be reversed (CCW if false, CW if true)\r\n * @param cullBackFaces true to cull back faces, false to cull front faces (if culling is enabled)\r\n * @param stencil stencil states to set\r\n * @param zOffsetUnits defines the value to apply to zOffsetUnits (0 by default)\r\n */\r\n public setState(culling: boolean, zOffset: number = 0, force?: boolean, reverseSide = false, cullBackFaces?: boolean, stencil?: IStencilState, zOffsetUnits: number = 0): void {\r\n // Culling\r\n if (this._depthCullingState.cull !== culling || force) {\r\n this._depthCullingState.cull = culling;\r\n }\r\n\r\n // Cull face\r\n const cullFace = this.cullBackFaces ?? cullBackFaces ?? true ? this._gl.BACK : this._gl.FRONT;\r\n if (this._depthCullingState.cullFace !== cullFace || force) {\r\n this._depthCullingState.cullFace = cullFace;\r\n }\r\n\r\n // Z offset\r\n this.setZOffset(zOffset);\r\n this.setZOffsetUnits(zOffsetUnits);\r\n\r\n // Front face\r\n const frontFace = reverseSide ? this._gl.CW : this._gl.CCW;\r\n if (this._depthCullingState.frontFace !== frontFace || force) {\r\n this._depthCullingState.frontFace = frontFace;\r\n }\r\n\r\n this._stencilStateComposer.stencilMaterial = stencil;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if depth testing is enabled\r\n * @returns the current state\r\n */\r\n public getDepthBuffer(): boolean {\r\n return this._depthCullingState.depthTest;\r\n }\r\n\r\n /**\r\n * Enable or disable depth buffering\r\n * @param enable defines the state to set\r\n */\r\n public setDepthBuffer(enable: boolean): void {\r\n this._depthCullingState.depthTest = enable;\r\n }\r\n\r\n /**\r\n * Set the z offset Factor to apply to current rendering\r\n * @param value defines the offset to apply\r\n */\r\n public setZOffset(value: number): void {\r\n this._depthCullingState.zOffset = this.useReverseDepthBuffer ? -value : value;\r\n }\r\n\r\n /**\r\n * Gets the current value of the zOffset Factor\r\n * @returns the current zOffset Factor state\r\n */\r\n public getZOffset(): number {\r\n const zOffset = this._depthCullingState.zOffset;\r\n return this.useReverseDepthBuffer ? -zOffset : zOffset;\r\n }\r\n\r\n /**\r\n * Set the z offset Units to apply to current rendering\r\n * @param value defines the offset to apply\r\n */\r\n public setZOffsetUnits(value: number): void {\r\n this._depthCullingState.zOffsetUnits = this.useReverseDepthBuffer ? -value : value;\r\n }\r\n\r\n /**\r\n * Gets the current value of the zOffset Units\r\n * @returns the current zOffset Units state\r\n */\r\n public getZOffsetUnits(): number {\r\n const zOffsetUnits = this._depthCullingState.zOffsetUnits;\r\n return this.useReverseDepthBuffer ? -zOffsetUnits : zOffsetUnits;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _bindUnboundFramebuffer(framebuffer: Nullable) {\r\n if (this._currentFramebuffer !== framebuffer) {\r\n this._gl.bindFramebuffer(this._gl.FRAMEBUFFER, framebuffer);\r\n this._currentFramebuffer = framebuffer;\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _currentFrameBufferIsDefaultFrameBuffer() {\r\n return this._currentFramebuffer === null;\r\n }\r\n\r\n /**\r\n * Generates the mipmaps for a texture\r\n * @param texture texture to generate the mipmaps for\r\n */\r\n public generateMipmaps(texture: InternalTexture): void {\r\n this._bindTextureDirectly(this._gl.TEXTURE_2D, texture, true);\r\n this._gl.generateMipmap(this._gl.TEXTURE_2D);\r\n this._bindTextureDirectly(this._gl.TEXTURE_2D, null);\r\n }\r\n\r\n /**\r\n * Unbind the current render target texture from the webGL context\r\n * @param texture defines the render target wrapper to unbind\r\n * @param disableGenerateMipMaps defines a boolean indicating that mipmaps must not be generated\r\n * @param onBeforeUnbind defines a function which will be called before the effective unbind\r\n */\r\n public unBindFramebuffer(texture: RenderTargetWrapper, disableGenerateMipMaps = false, onBeforeUnbind?: () => void): void {\r\n const webglRTWrapper = texture as WebGLRenderTargetWrapper;\r\n\r\n this._currentRenderTarget = null;\r\n\r\n // If MSAA, we need to bitblt back to main texture\r\n const gl = this._gl;\r\n if (webglRTWrapper._MSAAFramebuffer) {\r\n if (texture.isMulti) {\r\n // This texture is part of a MRT texture, we need to treat all attachments\r\n this.unBindMultiColorAttachmentFramebuffer(texture, disableGenerateMipMaps, onBeforeUnbind);\r\n return;\r\n }\r\n gl.bindFramebuffer(gl.READ_FRAMEBUFFER, webglRTWrapper._MSAAFramebuffer);\r\n gl.bindFramebuffer(gl.DRAW_FRAMEBUFFER, webglRTWrapper._framebuffer);\r\n gl.blitFramebuffer(0, 0, texture.width, texture.height, 0, 0, texture.width, texture.height, gl.COLOR_BUFFER_BIT, gl.NEAREST);\r\n }\r\n\r\n if (texture.texture?.generateMipMaps && !disableGenerateMipMaps && !texture.isCube) {\r\n this.generateMipmaps(texture.texture);\r\n }\r\n\r\n if (onBeforeUnbind) {\r\n if (webglRTWrapper._MSAAFramebuffer) {\r\n // Bind the correct framebuffer\r\n this._bindUnboundFramebuffer(webglRTWrapper._framebuffer);\r\n }\r\n onBeforeUnbind();\r\n }\r\n\r\n this._bindUnboundFramebuffer(null);\r\n }\r\n\r\n /**\r\n * Force a webGL flush (ie. a flush of all waiting webGL commands)\r\n */\r\n public flushFramebuffer(): void {\r\n this._gl.flush();\r\n }\r\n\r\n /**\r\n * Unbind the current render target and bind the default framebuffer\r\n */\r\n public restoreDefaultFramebuffer(): void {\r\n if (this._currentRenderTarget) {\r\n this.unBindFramebuffer(this._currentRenderTarget);\r\n } else {\r\n this._bindUnboundFramebuffer(null);\r\n }\r\n if (this._cachedViewport) {\r\n this.setViewport(this._cachedViewport);\r\n }\r\n\r\n this.wipeCaches();\r\n }\r\n\r\n // VBOs\r\n\r\n /** @internal */\r\n protected _resetVertexBufferBinding(): void {\r\n this.bindArrayBuffer(null);\r\n this._cachedVertexBuffers = null;\r\n }\r\n\r\n /**\r\n * Creates a vertex buffer\r\n * @param data the data for the vertex buffer\r\n * @returns the new WebGL static buffer\r\n */\r\n public createVertexBuffer(data: DataArray): DataBuffer {\r\n return this._createVertexBuffer(data, this._gl.STATIC_DRAW);\r\n }\r\n\r\n private _createVertexBuffer(data: DataArray, usage: number): DataBuffer {\r\n const vbo = this._gl.createBuffer();\r\n\r\n if (!vbo) {\r\n throw new Error(\"Unable to create vertex buffer\");\r\n }\r\n\r\n const dataBuffer = new WebGLDataBuffer(vbo);\r\n this.bindArrayBuffer(dataBuffer);\r\n\r\n if (data instanceof Array) {\r\n this._gl.bufferData(this._gl.ARRAY_BUFFER, new Float32Array(data), usage);\r\n } else {\r\n this._gl.bufferData(this._gl.ARRAY_BUFFER, data, usage);\r\n }\r\n\r\n this._resetVertexBufferBinding();\r\n\r\n dataBuffer.references = 1;\r\n return dataBuffer;\r\n }\r\n\r\n /**\r\n * Creates a dynamic vertex buffer\r\n * @param data the data for the dynamic vertex buffer\r\n * @returns the new WebGL dynamic buffer\r\n */\r\n public createDynamicVertexBuffer(data: DataArray): DataBuffer {\r\n return this._createVertexBuffer(data, this._gl.DYNAMIC_DRAW);\r\n }\r\n\r\n protected _resetIndexBufferBinding(): void {\r\n this.bindIndexBuffer(null);\r\n this._cachedIndexBuffer = null;\r\n }\r\n\r\n /**\r\n * Creates a new index buffer\r\n * @param indices defines the content of the index buffer\r\n * @param updatable defines if the index buffer must be updatable\r\n * @returns a new webGL buffer\r\n */\r\n public createIndexBuffer(indices: IndicesArray, updatable?: boolean): DataBuffer {\r\n const vbo = this._gl.createBuffer();\r\n const dataBuffer = new WebGLDataBuffer(vbo!);\r\n\r\n if (!vbo) {\r\n throw new Error(\"Unable to create index buffer\");\r\n }\r\n\r\n this.bindIndexBuffer(dataBuffer);\r\n\r\n const data = this._normalizeIndexData(indices);\r\n this._gl.bufferData(this._gl.ELEMENT_ARRAY_BUFFER, data, updatable ? this._gl.DYNAMIC_DRAW : this._gl.STATIC_DRAW);\r\n this._resetIndexBufferBinding();\r\n dataBuffer.references = 1;\r\n dataBuffer.is32Bits = data.BYTES_PER_ELEMENT === 4;\r\n return dataBuffer;\r\n }\r\n\r\n protected _normalizeIndexData(indices: IndicesArray): Uint16Array | Uint32Array {\r\n const bytesPerElement = (indices as Exclude).BYTES_PER_ELEMENT;\r\n if (bytesPerElement === 2) {\r\n return indices as Uint16Array;\r\n }\r\n\r\n // Check 32 bit support\r\n if (this._caps.uintIndices) {\r\n if (indices instanceof Uint32Array) {\r\n return indices;\r\n } else {\r\n // number[] or Int32Array, check if 32 bit is necessary\r\n for (let index = 0; index < indices.length; index++) {\r\n if (indices[index] >= 65535) {\r\n return new Uint32Array(indices);\r\n }\r\n }\r\n\r\n return new Uint16Array(indices);\r\n }\r\n }\r\n\r\n // No 32 bit support, force conversion to 16 bit (values greater 16 bit are lost)\r\n return new Uint16Array(indices);\r\n }\r\n\r\n /**\r\n * Bind a webGL buffer to the webGL context\r\n * @param buffer defines the buffer to bind\r\n */\r\n public bindArrayBuffer(buffer: Nullable): void {\r\n if (!this._vaoRecordInProgress) {\r\n this._unbindVertexArrayObject();\r\n }\r\n this._bindBuffer(buffer, this._gl.ARRAY_BUFFER);\r\n }\r\n\r\n /**\r\n * Bind a specific block at a given index in a specific shader program\r\n * @param pipelineContext defines the pipeline context to use\r\n * @param blockName defines the block name\r\n * @param index defines the index where to bind the block\r\n */\r\n public bindUniformBlock(pipelineContext: IPipelineContext, blockName: string, index: number): void {\r\n const program = (pipelineContext as WebGLPipelineContext).program!;\r\n\r\n const uniformLocation = this._gl.getUniformBlockIndex(program, blockName);\r\n\r\n this._gl.uniformBlockBinding(program, uniformLocation, index);\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n protected bindIndexBuffer(buffer: Nullable): void {\r\n if (!this._vaoRecordInProgress) {\r\n this._unbindVertexArrayObject();\r\n }\r\n this._bindBuffer(buffer, this._gl.ELEMENT_ARRAY_BUFFER);\r\n }\r\n\r\n private _bindBuffer(buffer: Nullable, target: number): void {\r\n if (this._vaoRecordInProgress || this._currentBoundBuffer[target] !== buffer) {\r\n this._gl.bindBuffer(target, buffer ? buffer.underlyingResource : null);\r\n this._currentBoundBuffer[target] = buffer;\r\n }\r\n }\r\n\r\n /**\r\n * update the bound buffer with the given data\r\n * @param data defines the data to update\r\n */\r\n public updateArrayBuffer(data: Float32Array): void {\r\n this._gl.bufferSubData(this._gl.ARRAY_BUFFER, 0, data);\r\n }\r\n\r\n private _vertexAttribPointer(buffer: DataBuffer, indx: number, size: number, type: number, normalized: boolean, stride: number, offset: number): void {\r\n const pointer = this._currentBufferPointers[indx];\r\n if (!pointer) {\r\n return;\r\n }\r\n\r\n let changed = false;\r\n if (!pointer.active) {\r\n changed = true;\r\n pointer.active = true;\r\n pointer.index = indx;\r\n pointer.size = size;\r\n pointer.type = type;\r\n pointer.normalized = normalized;\r\n pointer.stride = stride;\r\n pointer.offset = offset;\r\n pointer.buffer = buffer;\r\n } else {\r\n if (pointer.buffer !== buffer) {\r\n pointer.buffer = buffer;\r\n changed = true;\r\n }\r\n if (pointer.size !== size) {\r\n pointer.size = size;\r\n changed = true;\r\n }\r\n if (pointer.type !== type) {\r\n pointer.type = type;\r\n changed = true;\r\n }\r\n if (pointer.normalized !== normalized) {\r\n pointer.normalized = normalized;\r\n changed = true;\r\n }\r\n if (pointer.stride !== stride) {\r\n pointer.stride = stride;\r\n changed = true;\r\n }\r\n if (pointer.offset !== offset) {\r\n pointer.offset = offset;\r\n changed = true;\r\n }\r\n }\r\n\r\n if (changed || this._vaoRecordInProgress) {\r\n this.bindArrayBuffer(buffer);\r\n if (type === this._gl.UNSIGNED_INT || type === this._gl.INT) {\r\n this._gl.vertexAttribIPointer(indx, size, type, stride, offset);\r\n } else {\r\n this._gl.vertexAttribPointer(indx, size, type, normalized, stride, offset);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _bindIndexBufferWithCache(indexBuffer: Nullable): void {\r\n if (indexBuffer == null) {\r\n return;\r\n }\r\n if (this._cachedIndexBuffer !== indexBuffer) {\r\n this._cachedIndexBuffer = indexBuffer;\r\n this.bindIndexBuffer(indexBuffer);\r\n this._uintIndicesCurrentlySet = indexBuffer.is32Bits;\r\n }\r\n }\r\n\r\n private _bindVertexBuffersAttributes(\r\n vertexBuffers: { [key: string]: Nullable },\r\n effect: Effect,\r\n overrideVertexBuffers?: { [kind: string]: Nullable }\r\n ): void {\r\n const attributes = effect.getAttributesNames();\r\n\r\n if (!this._vaoRecordInProgress) {\r\n this._unbindVertexArrayObject();\r\n }\r\n\r\n this.unbindAllAttributes();\r\n\r\n for (let index = 0; index < attributes.length; index++) {\r\n const order = effect.getAttributeLocation(index);\r\n\r\n if (order >= 0) {\r\n const ai = attributes[index];\r\n let vertexBuffer: Nullable = null;\r\n\r\n if (overrideVertexBuffers) {\r\n vertexBuffer = overrideVertexBuffers[ai];\r\n }\r\n\r\n if (!vertexBuffer) {\r\n vertexBuffer = vertexBuffers[ai];\r\n }\r\n\r\n if (!vertexBuffer) {\r\n continue;\r\n }\r\n\r\n this._gl.enableVertexAttribArray(order);\r\n if (!this._vaoRecordInProgress) {\r\n this._vertexAttribArraysEnabled[order] = true;\r\n }\r\n\r\n const buffer = vertexBuffer.getBuffer();\r\n if (buffer) {\r\n this._vertexAttribPointer(buffer, order, vertexBuffer.getSize(), vertexBuffer.type, vertexBuffer.normalized, vertexBuffer.byteStride, vertexBuffer.byteOffset);\r\n\r\n if (vertexBuffer.getIsInstanced()) {\r\n this._gl.vertexAttribDivisor(order, vertexBuffer.getInstanceDivisor());\r\n if (!this._vaoRecordInProgress) {\r\n this._currentInstanceLocations.push(order);\r\n this._currentInstanceBuffers.push(buffer);\r\n }\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Records a vertex array object\r\n * @see https://doc.babylonjs.com/setup/support/webGL2#vertex-array-objects\r\n * @param vertexBuffers defines the list of vertex buffers to store\r\n * @param indexBuffer defines the index buffer to store\r\n * @param effect defines the effect to store\r\n * @param overrideVertexBuffers defines optional list of avertex buffers that overrides the entries in vertexBuffers\r\n * @returns the new vertex array object\r\n */\r\n public recordVertexArrayObject(\r\n vertexBuffers: { [key: string]: VertexBuffer },\r\n indexBuffer: Nullable,\r\n effect: Effect,\r\n overrideVertexBuffers?: { [kind: string]: Nullable }\r\n ): WebGLVertexArrayObject {\r\n const vao = this._gl.createVertexArray();\r\n\r\n if (!vao) {\r\n throw new Error(\"Unable to create VAO\");\r\n }\r\n\r\n this._vaoRecordInProgress = true;\r\n\r\n this._gl.bindVertexArray(vao);\r\n\r\n this._mustWipeVertexAttributes = true;\r\n this._bindVertexBuffersAttributes(vertexBuffers, effect, overrideVertexBuffers);\r\n\r\n this.bindIndexBuffer(indexBuffer);\r\n\r\n this._vaoRecordInProgress = false;\r\n this._gl.bindVertexArray(null);\r\n\r\n return vao;\r\n }\r\n\r\n /**\r\n * Bind a specific vertex array object\r\n * @see https://doc.babylonjs.com/setup/support/webGL2#vertex-array-objects\r\n * @param vertexArrayObject defines the vertex array object to bind\r\n * @param indexBuffer defines the index buffer to bind\r\n */\r\n public bindVertexArrayObject(vertexArrayObject: WebGLVertexArrayObject, indexBuffer: Nullable): void {\r\n if (this._cachedVertexArrayObject !== vertexArrayObject) {\r\n this._cachedVertexArrayObject = vertexArrayObject;\r\n\r\n this._gl.bindVertexArray(vertexArrayObject);\r\n this._cachedVertexBuffers = null;\r\n this._cachedIndexBuffer = null;\r\n\r\n this._uintIndicesCurrentlySet = indexBuffer != null && indexBuffer.is32Bits;\r\n this._mustWipeVertexAttributes = true;\r\n }\r\n }\r\n\r\n /**\r\n * Bind webGl buffers directly to the webGL context\r\n * @param vertexBuffer defines the vertex buffer to bind\r\n * @param indexBuffer defines the index buffer to bind\r\n * @param vertexDeclaration defines the vertex declaration to use with the vertex buffer\r\n * @param vertexStrideSize defines the vertex stride of the vertex buffer\r\n * @param effect defines the effect associated with the vertex buffer\r\n */\r\n public bindBuffersDirectly(vertexBuffer: DataBuffer, indexBuffer: DataBuffer, vertexDeclaration: number[], vertexStrideSize: number, effect: Effect): void {\r\n if (this._cachedVertexBuffers !== vertexBuffer || this._cachedEffectForVertexBuffers !== effect) {\r\n this._cachedVertexBuffers = vertexBuffer;\r\n this._cachedEffectForVertexBuffers = effect;\r\n\r\n const attributesCount = effect.getAttributesCount();\r\n\r\n this._unbindVertexArrayObject();\r\n this.unbindAllAttributes();\r\n\r\n let offset = 0;\r\n for (let index = 0; index < attributesCount; index++) {\r\n if (index < vertexDeclaration.length) {\r\n const order = effect.getAttributeLocation(index);\r\n\r\n if (order >= 0) {\r\n this._gl.enableVertexAttribArray(order);\r\n this._vertexAttribArraysEnabled[order] = true;\r\n this._vertexAttribPointer(vertexBuffer, order, vertexDeclaration[index], this._gl.FLOAT, false, vertexStrideSize, offset);\r\n }\r\n\r\n offset += vertexDeclaration[index] * 4;\r\n }\r\n }\r\n }\r\n\r\n this._bindIndexBufferWithCache(indexBuffer);\r\n }\r\n\r\n private _unbindVertexArrayObject(): void {\r\n if (!this._cachedVertexArrayObject) {\r\n return;\r\n }\r\n\r\n this._cachedVertexArrayObject = null;\r\n this._gl.bindVertexArray(null);\r\n }\r\n\r\n /**\r\n * Bind a list of vertex buffers to the webGL context\r\n * @param vertexBuffers defines the list of vertex buffers to bind\r\n * @param indexBuffer defines the index buffer to bind\r\n * @param effect defines the effect associated with the vertex buffers\r\n * @param overrideVertexBuffers defines optional list of avertex buffers that overrides the entries in vertexBuffers\r\n */\r\n public bindBuffers(\r\n vertexBuffers: { [key: string]: Nullable },\r\n indexBuffer: Nullable,\r\n effect: Effect,\r\n overrideVertexBuffers?: { [kind: string]: Nullable }\r\n ): void {\r\n if (this._cachedVertexBuffers !== vertexBuffers || this._cachedEffectForVertexBuffers !== effect) {\r\n this._cachedVertexBuffers = vertexBuffers;\r\n this._cachedEffectForVertexBuffers = effect;\r\n\r\n this._bindVertexBuffersAttributes(vertexBuffers, effect, overrideVertexBuffers);\r\n }\r\n\r\n this._bindIndexBufferWithCache(indexBuffer);\r\n }\r\n\r\n /**\r\n * Unbind all instance attributes\r\n */\r\n public unbindInstanceAttributes() {\r\n let boundBuffer;\r\n for (let i = 0, ul = this._currentInstanceLocations.length; i < ul; i++) {\r\n const instancesBuffer = this._currentInstanceBuffers[i];\r\n if (boundBuffer != instancesBuffer && instancesBuffer.references) {\r\n boundBuffer = instancesBuffer;\r\n this.bindArrayBuffer(instancesBuffer);\r\n }\r\n const offsetLocation = this._currentInstanceLocations[i];\r\n this._gl.vertexAttribDivisor(offsetLocation, 0);\r\n }\r\n this._currentInstanceBuffers.length = 0;\r\n this._currentInstanceLocations.length = 0;\r\n }\r\n\r\n /**\r\n * Release and free the memory of a vertex array object\r\n * @param vao defines the vertex array object to delete\r\n */\r\n public releaseVertexArrayObject(vao: WebGLVertexArrayObject) {\r\n this._gl.deleteVertexArray(vao);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _releaseBuffer(buffer: DataBuffer): boolean {\r\n buffer.references--;\r\n\r\n if (buffer.references === 0) {\r\n this._deleteBuffer(buffer);\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n protected _deleteBuffer(buffer: DataBuffer): void {\r\n this._gl.deleteBuffer(buffer.underlyingResource);\r\n }\r\n\r\n /**\r\n * Update the content of a webGL buffer used with instantiation and bind it to the webGL context\r\n * @param instancesBuffer defines the webGL buffer to update and bind\r\n * @param data defines the data to store in the buffer\r\n * @param offsetLocations defines the offsets or attributes information used to determine where data must be stored in the buffer\r\n */\r\n public updateAndBindInstancesBuffer(instancesBuffer: DataBuffer, data: Float32Array, offsetLocations: number[] | InstancingAttributeInfo[]): void {\r\n this.bindArrayBuffer(instancesBuffer);\r\n if (data) {\r\n this._gl.bufferSubData(this._gl.ARRAY_BUFFER, 0, data);\r\n }\r\n\r\n if ((offsetLocations[0]).index !== undefined) {\r\n this.bindInstancesBuffer(instancesBuffer, offsetLocations as any, true);\r\n } else {\r\n for (let index = 0; index < 4; index++) {\r\n const offsetLocation = offsetLocations[index];\r\n\r\n if (!this._vertexAttribArraysEnabled[offsetLocation]) {\r\n this._gl.enableVertexAttribArray(offsetLocation);\r\n this._vertexAttribArraysEnabled[offsetLocation] = true;\r\n }\r\n\r\n this._vertexAttribPointer(instancesBuffer, offsetLocation, 4, this._gl.FLOAT, false, 64, index * 16);\r\n this._gl.vertexAttribDivisor(offsetLocation, 1);\r\n this._currentInstanceLocations.push(offsetLocation);\r\n this._currentInstanceBuffers.push(instancesBuffer);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Bind the content of a webGL buffer used with instantiation\r\n * @param instancesBuffer defines the webGL buffer to bind\r\n * @param attributesInfo defines the offsets or attributes information used to determine where data must be stored in the buffer\r\n * @param computeStride defines Whether to compute the strides from the info or use the default 0\r\n */\r\n public bindInstancesBuffer(instancesBuffer: DataBuffer, attributesInfo: InstancingAttributeInfo[], computeStride = true): void {\r\n this.bindArrayBuffer(instancesBuffer);\r\n\r\n let stride = 0;\r\n if (computeStride) {\r\n for (let i = 0; i < attributesInfo.length; i++) {\r\n const ai = attributesInfo[i];\r\n stride += ai.attributeSize * 4;\r\n }\r\n }\r\n\r\n for (let i = 0; i < attributesInfo.length; i++) {\r\n const ai = attributesInfo[i];\r\n if (ai.index === undefined) {\r\n ai.index = this._currentEffect!.getAttributeLocationByName(ai.attributeName);\r\n }\r\n\r\n if (ai.index < 0) {\r\n continue;\r\n }\r\n\r\n if (!this._vertexAttribArraysEnabled[ai.index]) {\r\n this._gl.enableVertexAttribArray(ai.index);\r\n this._vertexAttribArraysEnabled[ai.index] = true;\r\n }\r\n\r\n this._vertexAttribPointer(instancesBuffer, ai.index, ai.attributeSize, ai.attributeType || this._gl.FLOAT, ai.normalized || false, stride, ai.offset);\r\n this._gl.vertexAttribDivisor(ai.index, ai.divisor === undefined ? 1 : ai.divisor);\r\n this._currentInstanceLocations.push(ai.index);\r\n this._currentInstanceBuffers.push(instancesBuffer);\r\n }\r\n }\r\n\r\n /**\r\n * Disable the instance attribute corresponding to the name in parameter\r\n * @param name defines the name of the attribute to disable\r\n */\r\n public disableInstanceAttributeByName(name: string) {\r\n if (!this._currentEffect) {\r\n return;\r\n }\r\n\r\n const attributeLocation = this._currentEffect.getAttributeLocationByName(name);\r\n this.disableInstanceAttribute(attributeLocation);\r\n }\r\n\r\n /**\r\n * Disable the instance attribute corresponding to the location in parameter\r\n * @param attributeLocation defines the attribute location of the attribute to disable\r\n */\r\n public disableInstanceAttribute(attributeLocation: number) {\r\n let shouldClean = false;\r\n let index: number;\r\n while ((index = this._currentInstanceLocations.indexOf(attributeLocation)) !== -1) {\r\n this._currentInstanceLocations.splice(index, 1);\r\n this._currentInstanceBuffers.splice(index, 1);\r\n\r\n shouldClean = true;\r\n index = this._currentInstanceLocations.indexOf(attributeLocation);\r\n }\r\n\r\n if (shouldClean) {\r\n this._gl.vertexAttribDivisor(attributeLocation, 0);\r\n this.disableAttributeByIndex(attributeLocation);\r\n }\r\n }\r\n\r\n /**\r\n * Disable the attribute corresponding to the location in parameter\r\n * @param attributeLocation defines the attribute location of the attribute to disable\r\n */\r\n public disableAttributeByIndex(attributeLocation: number) {\r\n this._gl.disableVertexAttribArray(attributeLocation);\r\n this._vertexAttribArraysEnabled[attributeLocation] = false;\r\n this._currentBufferPointers[attributeLocation].active = false;\r\n }\r\n\r\n /**\r\n * Send a draw order\r\n * @param useTriangles defines if triangles must be used to draw (else wireframe will be used)\r\n * @param indexStart defines the starting index\r\n * @param indexCount defines the number of index to draw\r\n * @param instancesCount defines the number of instances to draw (if instantiation is enabled)\r\n */\r\n public draw(useTriangles: boolean, indexStart: number, indexCount: number, instancesCount?: number): void {\r\n this.drawElementsType(useTriangles ? Constants.MATERIAL_TriangleFillMode : Constants.MATERIAL_WireFrameFillMode, indexStart, indexCount, instancesCount);\r\n }\r\n\r\n /**\r\n * Draw a list of points\r\n * @param verticesStart defines the index of first vertex to draw\r\n * @param verticesCount defines the count of vertices to draw\r\n * @param instancesCount defines the number of instances to draw (if instantiation is enabled)\r\n */\r\n public drawPointClouds(verticesStart: number, verticesCount: number, instancesCount?: number): void {\r\n this.drawArraysType(Constants.MATERIAL_PointFillMode, verticesStart, verticesCount, instancesCount);\r\n }\r\n\r\n /**\r\n * Draw a list of unindexed primitives\r\n * @param useTriangles defines if triangles must be used to draw (else wireframe will be used)\r\n * @param verticesStart defines the index of first vertex to draw\r\n * @param verticesCount defines the count of vertices to draw\r\n * @param instancesCount defines the number of instances to draw (if instantiation is enabled)\r\n */\r\n public drawUnIndexed(useTriangles: boolean, verticesStart: number, verticesCount: number, instancesCount?: number): void {\r\n this.drawArraysType(useTriangles ? Constants.MATERIAL_TriangleFillMode : Constants.MATERIAL_WireFrameFillMode, verticesStart, verticesCount, instancesCount);\r\n }\r\n\r\n /**\r\n * Draw a list of indexed primitives\r\n * @param fillMode defines the primitive to use\r\n * @param indexStart defines the starting index\r\n * @param indexCount defines the number of index to draw\r\n * @param instancesCount defines the number of instances to draw (if instantiation is enabled)\r\n */\r\n public drawElementsType(fillMode: number, indexStart: number, indexCount: number, instancesCount?: number): void {\r\n // Apply states\r\n this.applyStates();\r\n\r\n this._reportDrawCall();\r\n\r\n // Render\r\n\r\n const drawMode = this._drawMode(fillMode);\r\n const indexFormat = this._uintIndicesCurrentlySet ? this._gl.UNSIGNED_INT : this._gl.UNSIGNED_SHORT;\r\n const mult = this._uintIndicesCurrentlySet ? 4 : 2;\r\n if (instancesCount) {\r\n this._gl.drawElementsInstanced(drawMode, indexCount, indexFormat, indexStart * mult, instancesCount);\r\n } else {\r\n this._gl.drawElements(drawMode, indexCount, indexFormat, indexStart * mult);\r\n }\r\n }\r\n\r\n /**\r\n * Draw a list of unindexed primitives\r\n * @param fillMode defines the primitive to use\r\n * @param verticesStart defines the index of first vertex to draw\r\n * @param verticesCount defines the count of vertices to draw\r\n * @param instancesCount defines the number of instances to draw (if instantiation is enabled)\r\n */\r\n public drawArraysType(fillMode: number, verticesStart: number, verticesCount: number, instancesCount?: number): void {\r\n // Apply states\r\n this.applyStates();\r\n\r\n this._reportDrawCall();\r\n\r\n const drawMode = this._drawMode(fillMode);\r\n if (instancesCount) {\r\n this._gl.drawArraysInstanced(drawMode, verticesStart, verticesCount, instancesCount);\r\n } else {\r\n this._gl.drawArrays(drawMode, verticesStart, verticesCount);\r\n }\r\n }\r\n\r\n private _drawMode(fillMode: number): number {\r\n switch (fillMode) {\r\n // Triangle views\r\n case Constants.MATERIAL_TriangleFillMode:\r\n return this._gl.TRIANGLES;\r\n case Constants.MATERIAL_PointFillMode:\r\n return this._gl.POINTS;\r\n case Constants.MATERIAL_WireFrameFillMode:\r\n return this._gl.LINES;\r\n // Draw modes\r\n case Constants.MATERIAL_PointListDrawMode:\r\n return this._gl.POINTS;\r\n case Constants.MATERIAL_LineListDrawMode:\r\n return this._gl.LINES;\r\n case Constants.MATERIAL_LineLoopDrawMode:\r\n return this._gl.LINE_LOOP;\r\n case Constants.MATERIAL_LineStripDrawMode:\r\n return this._gl.LINE_STRIP;\r\n case Constants.MATERIAL_TriangleStripDrawMode:\r\n return this._gl.TRIANGLE_STRIP;\r\n case Constants.MATERIAL_TriangleFanDrawMode:\r\n return this._gl.TRIANGLE_FAN;\r\n default:\r\n return this._gl.TRIANGLES;\r\n }\r\n }\r\n\r\n /** @internal */\r\n protected _reportDrawCall() {\r\n // Will be implemented by children\r\n }\r\n\r\n // Shaders\r\n\r\n /**\r\n * @internal\r\n */\r\n public _releaseEffect(effect: Effect): void {\r\n if (this._compiledEffects[effect._key]) {\r\n delete this._compiledEffects[effect._key];\r\n }\r\n const pipelineContext = effect.getPipelineContext();\r\n if (pipelineContext) {\r\n this._deletePipelineContext(pipelineContext);\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _deletePipelineContext(pipelineContext: IPipelineContext): void {\r\n const webGLPipelineContext = pipelineContext as WebGLPipelineContext;\r\n if (webGLPipelineContext && webGLPipelineContext.program) {\r\n webGLPipelineContext.program.__SPECTOR_rebuildProgram = null;\r\n\r\n this._gl.deleteProgram(webGLPipelineContext.program);\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _getGlobalDefines(defines?: { [key: string]: string }): string | undefined {\r\n if (defines) {\r\n if (this.isNDCHalfZRange) {\r\n defines[\"IS_NDC_HALF_ZRANGE\"] = \"\";\r\n } else {\r\n delete defines[\"IS_NDC_HALF_ZRANGE\"];\r\n }\r\n if (this.useReverseDepthBuffer) {\r\n defines[\"USE_REVERSE_DEPTHBUFFER\"] = \"\";\r\n } else {\r\n delete defines[\"USE_REVERSE_DEPTHBUFFER\"];\r\n }\r\n if (this.useExactSrgbConversions) {\r\n defines[\"USE_EXACT_SRGB_CONVERSIONS\"] = \"\";\r\n } else {\r\n delete defines[\"USE_EXACT_SRGB_CONVERSIONS\"];\r\n }\r\n return;\r\n } else {\r\n let s = \"\";\r\n if (this.isNDCHalfZRange) {\r\n s += \"#define IS_NDC_HALF_ZRANGE\";\r\n }\r\n if (this.useReverseDepthBuffer) {\r\n if (s) {\r\n s += \"\\n\";\r\n }\r\n s += \"#define USE_REVERSE_DEPTHBUFFER\";\r\n }\r\n if (this.useExactSrgbConversions) {\r\n if (s) {\r\n s += \"\\n\";\r\n }\r\n s += \"#define USE_EXACT_SRGB_CONVERSIONS\";\r\n }\r\n return s;\r\n }\r\n }\r\n\r\n /**\r\n * Create a new effect (used to store vertex/fragment shaders)\r\n * @param baseName defines the base name of the effect (The name of file without .fragment.fx or .vertex.fx)\r\n * @param attributesNamesOrOptions defines either a list of attribute names or an IEffectCreationOptions object\r\n * @param uniformsNamesOrEngine defines either a list of uniform names or the engine to use\r\n * @param samplers defines an array of string used to represent textures\r\n * @param defines defines the string containing the defines to use to compile the shaders\r\n * @param fallbacks defines the list of potential fallbacks to use if shader compilation fails\r\n * @param onCompiled defines a function to call when the effect creation is successful\r\n * @param onError defines a function to call when the effect creation has failed\r\n * @param indexParameters defines an object containing the index values to use to compile shaders (like the maximum number of simultaneous lights)\r\n * @param shaderLanguage the language the shader is written in (default: GLSL)\r\n * @returns the new Effect\r\n */\r\n public createEffect(\r\n baseName: any,\r\n attributesNamesOrOptions: string[] | IEffectCreationOptions,\r\n uniformsNamesOrEngine: string[] | ThinEngine,\r\n samplers?: string[],\r\n defines?: string,\r\n fallbacks?: IEffectFallbacks,\r\n onCompiled?: Nullable<(effect: Effect) => void>,\r\n onError?: Nullable<(effect: Effect, errors: string) => void>,\r\n indexParameters?: any,\r\n shaderLanguage = ShaderLanguage.GLSL\r\n ): Effect {\r\n const vertex = baseName.vertexElement || baseName.vertex || baseName.vertexToken || baseName.vertexSource || baseName;\r\n const fragment = baseName.fragmentElement || baseName.fragment || baseName.fragmentToken || baseName.fragmentSource || baseName;\r\n const globalDefines = this._getGlobalDefines()!;\r\n\r\n let fullDefines = defines ?? (attributesNamesOrOptions).defines ?? \"\";\r\n\r\n if (globalDefines) {\r\n fullDefines += globalDefines;\r\n }\r\n\r\n const name = vertex + \"+\" + fragment + \"@\" + fullDefines;\r\n if (this._compiledEffects[name]) {\r\n const compiledEffect = this._compiledEffects[name];\r\n if (onCompiled && compiledEffect.isReady()) {\r\n onCompiled(compiledEffect);\r\n }\r\n\r\n return compiledEffect;\r\n }\r\n const effect = new Effect(\r\n baseName,\r\n attributesNamesOrOptions,\r\n uniformsNamesOrEngine,\r\n samplers,\r\n this,\r\n defines,\r\n fallbacks,\r\n onCompiled,\r\n onError,\r\n indexParameters,\r\n name,\r\n shaderLanguage\r\n );\r\n this._compiledEffects[name] = effect;\r\n\r\n return effect;\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n protected static _ConcatenateShader(source: string, defines: Nullable, shaderVersion: string = \"\"): string {\r\n return shaderVersion + (defines ? defines + \"\\n\" : \"\") + source;\r\n }\r\n\r\n private _compileShader(source: string, type: string, defines: Nullable, shaderVersion: string): WebGLShader {\r\n return this._compileRawShader(ThinEngine._ConcatenateShader(source, defines, shaderVersion), type);\r\n }\r\n\r\n private _compileRawShader(source: string, type: string): WebGLShader {\r\n const gl = this._gl;\r\n\r\n const shader = gl.createShader(type === \"vertex\" ? gl.VERTEX_SHADER : gl.FRAGMENT_SHADER);\r\n\r\n if (!shader) {\r\n let error: GLenum = gl.NO_ERROR;\r\n let tempError: GLenum = gl.NO_ERROR;\r\n while ((tempError = gl.getError()) !== gl.NO_ERROR) {\r\n error = tempError;\r\n }\r\n\r\n throw new Error(\r\n `Something went wrong while creating a gl ${type} shader object. gl error=${error}, gl isContextLost=${gl.isContextLost()}, _contextWasLost=${this._contextWasLost}`\r\n );\r\n }\r\n\r\n gl.shaderSource(shader, source);\r\n gl.compileShader(shader);\r\n\r\n return shader;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getShaderSource(shader: WebGLShader): Nullable {\r\n return this._gl.getShaderSource(shader);\r\n }\r\n\r\n /**\r\n * Directly creates a webGL program\r\n * @param pipelineContext defines the pipeline context to attach to\r\n * @param vertexCode defines the vertex shader code to use\r\n * @param fragmentCode defines the fragment shader code to use\r\n * @param context defines the webGL context to use (if not set, the current one will be used)\r\n * @param transformFeedbackVaryings defines the list of transform feedback varyings to use\r\n * @returns the new webGL program\r\n */\r\n public createRawShaderProgram(\r\n pipelineContext: IPipelineContext,\r\n vertexCode: string,\r\n fragmentCode: string,\r\n context?: WebGLRenderingContext,\r\n transformFeedbackVaryings: Nullable = null\r\n ): WebGLProgram {\r\n context = context || this._gl;\r\n\r\n const vertexShader = this._compileRawShader(vertexCode, \"vertex\");\r\n const fragmentShader = this._compileRawShader(fragmentCode, \"fragment\");\r\n\r\n return this._createShaderProgram(pipelineContext as WebGLPipelineContext, vertexShader, fragmentShader, context, transformFeedbackVaryings);\r\n }\r\n\r\n /**\r\n * Creates a webGL program\r\n * @param pipelineContext defines the pipeline context to attach to\r\n * @param vertexCode defines the vertex shader code to use\r\n * @param fragmentCode defines the fragment shader code to use\r\n * @param defines defines the string containing the defines to use to compile the shaders\r\n * @param context defines the webGL context to use (if not set, the current one will be used)\r\n * @param transformFeedbackVaryings defines the list of transform feedback varyings to use\r\n * @returns the new webGL program\r\n */\r\n public createShaderProgram(\r\n pipelineContext: IPipelineContext,\r\n vertexCode: string,\r\n fragmentCode: string,\r\n defines: Nullable,\r\n context?: WebGLRenderingContext,\r\n transformFeedbackVaryings: Nullable = null\r\n ): WebGLProgram {\r\n context = context || this._gl;\r\n\r\n const shaderVersion = this._webGLVersion > 1 ? \"#version 300 es\\n#define WEBGL2 \\n\" : \"\";\r\n const vertexShader = this._compileShader(vertexCode, \"vertex\", defines, shaderVersion);\r\n const fragmentShader = this._compileShader(fragmentCode, \"fragment\", defines, shaderVersion);\r\n\r\n return this._createShaderProgram(pipelineContext as WebGLPipelineContext, vertexShader, fragmentShader, context, transformFeedbackVaryings);\r\n }\r\n\r\n /**\r\n * Inline functions in shader code that are marked to be inlined\r\n * @param code code to inline\r\n * @returns inlined code\r\n */\r\n public inlineShaderCode(code: string): string {\r\n // no inlining needed in the WebGL engine\r\n return code;\r\n }\r\n\r\n /**\r\n * Creates a new pipeline context\r\n * @param shaderProcessingContext defines the shader processing context used during the processing if available\r\n * @returns the new pipeline\r\n */\r\n public createPipelineContext(shaderProcessingContext: Nullable): IPipelineContext {\r\n const pipelineContext = new WebGLPipelineContext();\r\n pipelineContext.engine = this;\r\n\r\n if (this._caps.parallelShaderCompile) {\r\n pipelineContext.isParallelCompiled = true;\r\n }\r\n\r\n return pipelineContext;\r\n }\r\n\r\n /**\r\n * Creates a new material context\r\n * @returns the new context\r\n */\r\n public createMaterialContext(): IMaterialContext | undefined {\r\n return undefined;\r\n }\r\n\r\n /**\r\n * Creates a new draw context\r\n * @returns the new context\r\n */\r\n public createDrawContext(): IDrawContext | undefined {\r\n return undefined;\r\n }\r\n\r\n protected _createShaderProgram(\r\n pipelineContext: WebGLPipelineContext,\r\n vertexShader: WebGLShader,\r\n fragmentShader: WebGLShader,\r\n context: WebGLRenderingContext,\r\n transformFeedbackVaryings: Nullable = null\r\n ): WebGLProgram {\r\n const shaderProgram = context.createProgram();\r\n pipelineContext.program = shaderProgram;\r\n\r\n if (!shaderProgram) {\r\n throw new Error(\"Unable to create program\");\r\n }\r\n\r\n context.attachShader(shaderProgram, vertexShader);\r\n context.attachShader(shaderProgram, fragmentShader);\r\n\r\n context.linkProgram(shaderProgram);\r\n\r\n pipelineContext.context = context;\r\n pipelineContext.vertexShader = vertexShader;\r\n pipelineContext.fragmentShader = fragmentShader;\r\n\r\n if (!pipelineContext.isParallelCompiled) {\r\n this._finalizePipelineContext(pipelineContext);\r\n }\r\n\r\n return shaderProgram;\r\n }\r\n\r\n protected _finalizePipelineContext(pipelineContext: WebGLPipelineContext) {\r\n const context = pipelineContext.context!;\r\n const vertexShader = pipelineContext.vertexShader!;\r\n const fragmentShader = pipelineContext.fragmentShader!;\r\n const program = pipelineContext.program!;\r\n\r\n const linked = context.getProgramParameter(program, context.LINK_STATUS);\r\n if (!linked) {\r\n // Get more info\r\n // Vertex\r\n if (!this._gl.getShaderParameter(vertexShader, this._gl.COMPILE_STATUS)) {\r\n const log = this._gl.getShaderInfoLog(vertexShader);\r\n if (log) {\r\n pipelineContext.vertexCompilationError = log;\r\n throw new Error(\"VERTEX SHADER \" + log);\r\n }\r\n }\r\n\r\n // Fragment\r\n if (!this._gl.getShaderParameter(fragmentShader, this._gl.COMPILE_STATUS)) {\r\n const log = this._gl.getShaderInfoLog(fragmentShader);\r\n if (log) {\r\n pipelineContext.fragmentCompilationError = log;\r\n throw new Error(\"FRAGMENT SHADER \" + log);\r\n }\r\n }\r\n\r\n const error = context.getProgramInfoLog(program);\r\n if (error) {\r\n pipelineContext.programLinkError = error;\r\n throw new Error(error);\r\n }\r\n }\r\n\r\n if (this.validateShaderPrograms) {\r\n context.validateProgram(program);\r\n const validated = context.getProgramParameter(program, context.VALIDATE_STATUS);\r\n\r\n if (!validated) {\r\n const error = context.getProgramInfoLog(program);\r\n if (error) {\r\n pipelineContext.programValidationError = error;\r\n throw new Error(error);\r\n }\r\n }\r\n }\r\n\r\n context.deleteShader(vertexShader);\r\n context.deleteShader(fragmentShader);\r\n\r\n pipelineContext.vertexShader = undefined;\r\n pipelineContext.fragmentShader = undefined;\r\n\r\n if (pipelineContext.onCompiled) {\r\n pipelineContext.onCompiled();\r\n pipelineContext.onCompiled = undefined;\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _preparePipelineContext(\r\n pipelineContext: IPipelineContext,\r\n vertexSourceCode: string,\r\n fragmentSourceCode: string,\r\n createAsRaw: boolean,\r\n rawVertexSourceCode: string,\r\n rawFragmentSourceCode: string,\r\n rebuildRebind: any,\r\n defines: Nullable,\r\n transformFeedbackVaryings: Nullable,\r\n key: string\r\n ) {\r\n const webGLRenderingState = pipelineContext as WebGLPipelineContext;\r\n\r\n if (createAsRaw) {\r\n webGLRenderingState.program = this.createRawShaderProgram(webGLRenderingState, vertexSourceCode, fragmentSourceCode, undefined, transformFeedbackVaryings);\r\n } else {\r\n webGLRenderingState.program = this.createShaderProgram(webGLRenderingState, vertexSourceCode, fragmentSourceCode, defines, undefined, transformFeedbackVaryings);\r\n }\r\n webGLRenderingState.program.__SPECTOR_rebuildProgram = rebuildRebind;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _isRenderingStateCompiled(pipelineContext: IPipelineContext): boolean {\r\n const webGLPipelineContext = pipelineContext as WebGLPipelineContext;\r\n if (this._isDisposed || webGLPipelineContext._isDisposed) {\r\n return false;\r\n }\r\n if (this._gl.getProgramParameter(webGLPipelineContext.program!, this._caps.parallelShaderCompile!.COMPLETION_STATUS_KHR)) {\r\n this._finalizePipelineContext(webGLPipelineContext);\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _executeWhenRenderingStateIsCompiled(pipelineContext: IPipelineContext, action: () => void) {\r\n const webGLPipelineContext = pipelineContext as WebGLPipelineContext;\r\n\r\n if (!webGLPipelineContext.isParallelCompiled) {\r\n action();\r\n return;\r\n }\r\n\r\n const oldHandler = webGLPipelineContext.onCompiled;\r\n\r\n if (oldHandler) {\r\n webGLPipelineContext.onCompiled = () => {\r\n oldHandler!();\r\n action();\r\n };\r\n } else {\r\n webGLPipelineContext.onCompiled = action;\r\n }\r\n }\r\n\r\n /**\r\n * Gets the list of webGL uniform locations associated with a specific program based on a list of uniform names\r\n * @param pipelineContext defines the pipeline context to use\r\n * @param uniformsNames defines the list of uniform names\r\n * @returns an array of webGL uniform locations\r\n */\r\n public getUniforms(pipelineContext: IPipelineContext, uniformsNames: string[]): Nullable[] {\r\n const results = new Array>();\r\n const webGLPipelineContext = pipelineContext as WebGLPipelineContext;\r\n\r\n for (let index = 0; index < uniformsNames.length; index++) {\r\n results.push(this._gl.getUniformLocation(webGLPipelineContext.program!, uniformsNames[index]));\r\n }\r\n\r\n return results;\r\n }\r\n\r\n /**\r\n * Gets the list of active attributes for a given webGL program\r\n * @param pipelineContext defines the pipeline context to use\r\n * @param attributesNames defines the list of attribute names to get\r\n * @returns an array of indices indicating the offset of each attribute\r\n */\r\n public getAttributes(pipelineContext: IPipelineContext, attributesNames: string[]): number[] {\r\n const results = [];\r\n const webGLPipelineContext = pipelineContext as WebGLPipelineContext;\r\n\r\n for (let index = 0; index < attributesNames.length; index++) {\r\n try {\r\n results.push(this._gl.getAttribLocation(webGLPipelineContext.program!, attributesNames[index]));\r\n } catch (e) {\r\n results.push(-1);\r\n }\r\n }\r\n\r\n return results;\r\n }\r\n\r\n /**\r\n * Activates an effect, making it the current one (ie. the one used for rendering)\r\n * @param effect defines the effect to activate\r\n */\r\n public enableEffect(effect: Nullable): void {\r\n effect = effect !== null && DrawWrapper.IsWrapper(effect) ? effect.effect : effect; // get only the effect, we don't need a Wrapper in the WebGL engine\r\n\r\n if (!effect || effect === this._currentEffect) {\r\n return;\r\n }\r\n\r\n this._stencilStateComposer.stencilMaterial = undefined;\r\n\r\n effect = effect as Effect;\r\n\r\n // Use program\r\n this.bindSamplers(effect);\r\n\r\n this._currentEffect = effect;\r\n\r\n if (effect.onBind) {\r\n effect.onBind(effect);\r\n }\r\n if (effect._onBindObservable) {\r\n effect._onBindObservable.notifyObservers(effect);\r\n }\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to a number (int)\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param value defines the int number to store\r\n * @returns true if the value was set\r\n */\r\n public setInt(uniform: Nullable, value: number): boolean {\r\n if (!uniform) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform1i(uniform, value);\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to a int2\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param x defines the 1st component of the value\r\n * @param y defines the 2nd component of the value\r\n * @returns true if the value was set\r\n */\r\n public setInt2(uniform: Nullable, x: number, y: number): boolean {\r\n if (!uniform) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform2i(uniform, x, y);\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to a int3\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param x defines the 1st component of the value\r\n * @param y defines the 2nd component of the value\r\n * @param z defines the 3rd component of the value\r\n * @returns true if the value was set\r\n */\r\n public setInt3(uniform: Nullable, x: number, y: number, z: number): boolean {\r\n if (!uniform) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform3i(uniform, x, y, z);\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to a int4\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param x defines the 1st component of the value\r\n * @param y defines the 2nd component of the value\r\n * @param z defines the 3rd component of the value\r\n * @param w defines the 4th component of the value\r\n * @returns true if the value was set\r\n */\r\n public setInt4(uniform: Nullable, x: number, y: number, z: number, w: number): boolean {\r\n if (!uniform) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform4i(uniform, x, y, z, w);\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to an array of int32\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param array defines the array of int32 to store\r\n * @returns true if the value was set\r\n */\r\n public setIntArray(uniform: Nullable, array: Int32Array): boolean {\r\n if (!uniform) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform1iv(uniform, array);\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to an array of int32 (stored as vec2)\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param array defines the array of int32 to store\r\n * @returns true if the value was set\r\n */\r\n public setIntArray2(uniform: Nullable, array: Int32Array): boolean {\r\n if (!uniform || array.length % 2 !== 0) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform2iv(uniform, array);\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to an array of int32 (stored as vec3)\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param array defines the array of int32 to store\r\n * @returns true if the value was set\r\n */\r\n public setIntArray3(uniform: Nullable, array: Int32Array): boolean {\r\n if (!uniform || array.length % 3 !== 0) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform3iv(uniform, array);\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to an array of int32 (stored as vec4)\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param array defines the array of int32 to store\r\n * @returns true if the value was set\r\n */\r\n public setIntArray4(uniform: Nullable, array: Int32Array): boolean {\r\n if (!uniform || array.length % 4 !== 0) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform4iv(uniform, array);\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to a number (unsigned int)\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param value defines the unsigned int number to store\r\n * @returns true if the value was set\r\n */\r\n public setUInt(uniform: Nullable, value: number): boolean {\r\n if (!uniform) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform1ui(uniform, value);\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to a unsigned int2\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param x defines the 1st component of the value\r\n * @param y defines the 2nd component of the value\r\n * @returns true if the value was set\r\n */\r\n public setUInt2(uniform: Nullable, x: number, y: number): boolean {\r\n if (!uniform) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform2ui(uniform, x, y);\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to a unsigned int3\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param x defines the 1st component of the value\r\n * @param y defines the 2nd component of the value\r\n * @param z defines the 3rd component of the value\r\n * @returns true if the value was set\r\n */\r\n public setUInt3(uniform: Nullable, x: number, y: number, z: number): boolean {\r\n if (!uniform) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform3ui(uniform, x, y, z);\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to a unsigned int4\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param x defines the 1st component of the value\r\n * @param y defines the 2nd component of the value\r\n * @param z defines the 3rd component of the value\r\n * @param w defines the 4th component of the value\r\n * @returns true if the value was set\r\n */\r\n public setUInt4(uniform: Nullable, x: number, y: number, z: number, w: number): boolean {\r\n if (!uniform) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform4ui(uniform, x, y, z, w);\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to an array of unsigned int32\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param array defines the array of unsigned int32 to store\r\n * @returns true if the value was set\r\n */\r\n public setUIntArray(uniform: Nullable, array: Uint32Array): boolean {\r\n if (!uniform) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform1uiv(uniform, array);\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to an array of unsigned int32 (stored as vec2)\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param array defines the array of unsigned int32 to store\r\n * @returns true if the value was set\r\n */\r\n public setUIntArray2(uniform: Nullable, array: Uint32Array): boolean {\r\n if (!uniform || array.length % 2 !== 0) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform2uiv(uniform, array);\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to an array of unsigned int32 (stored as vec3)\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param array defines the array of unsigned int32 to store\r\n * @returns true if the value was set\r\n */\r\n public setUIntArray3(uniform: Nullable, array: Uint32Array): boolean {\r\n if (!uniform || array.length % 3 !== 0) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform3uiv(uniform, array);\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to an array of unsigned int32 (stored as vec4)\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param array defines the array of unsigned int32 to store\r\n * @returns true if the value was set\r\n */\r\n public setUIntArray4(uniform: Nullable, array: Uint32Array): boolean {\r\n if (!uniform || array.length % 4 !== 0) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform4uiv(uniform, array);\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to an array of number\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param array defines the array of number to store\r\n * @returns true if the value was set\r\n */\r\n public setArray(uniform: Nullable, array: number[] | Float32Array): boolean {\r\n if (!uniform) {\r\n return false;\r\n }\r\n\r\n if (array.length < 1) {\r\n return false;\r\n }\r\n this._gl.uniform1fv(uniform, array);\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to an array of number (stored as vec2)\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param array defines the array of number to store\r\n * @returns true if the value was set\r\n */\r\n public setArray2(uniform: Nullable, array: number[] | Float32Array): boolean {\r\n if (!uniform || array.length % 2 !== 0) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform2fv(uniform, array);\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to an array of number (stored as vec3)\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param array defines the array of number to store\r\n * @returns true if the value was set\r\n */\r\n public setArray3(uniform: Nullable, array: number[] | Float32Array): boolean {\r\n if (!uniform || array.length % 3 !== 0) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform3fv(uniform, array);\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to an array of number (stored as vec4)\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param array defines the array of number to store\r\n * @returns true if the value was set\r\n */\r\n public setArray4(uniform: Nullable, array: number[] | Float32Array): boolean {\r\n if (!uniform || array.length % 4 !== 0) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform4fv(uniform, array);\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to an array of float32 (stored as matrices)\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param matrices defines the array of float32 to store\r\n * @returns true if the value was set\r\n */\r\n public setMatrices(uniform: Nullable, matrices: Float32Array): boolean {\r\n if (!uniform) {\r\n return false;\r\n }\r\n\r\n this._gl.uniformMatrix4fv(uniform, false, matrices);\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to a matrix (3x3)\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param matrix defines the Float32Array representing the 3x3 matrix to store\r\n * @returns true if the value was set\r\n */\r\n public setMatrix3x3(uniform: Nullable, matrix: Float32Array): boolean {\r\n if (!uniform) {\r\n return false;\r\n }\r\n\r\n this._gl.uniformMatrix3fv(uniform, false, matrix);\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to a matrix (2x2)\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param matrix defines the Float32Array representing the 2x2 matrix to store\r\n * @returns true if the value was set\r\n */\r\n public setMatrix2x2(uniform: Nullable, matrix: Float32Array): boolean {\r\n if (!uniform) {\r\n return false;\r\n }\r\n\r\n this._gl.uniformMatrix2fv(uniform, false, matrix);\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to a number (float)\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param value defines the float number to store\r\n * @returns true if the value was transferred\r\n */\r\n public setFloat(uniform: Nullable, value: number): boolean {\r\n if (!uniform) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform1f(uniform, value);\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to a vec2\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param x defines the 1st component of the value\r\n * @param y defines the 2nd component of the value\r\n * @returns true if the value was set\r\n */\r\n public setFloat2(uniform: Nullable, x: number, y: number): boolean {\r\n if (!uniform) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform2f(uniform, x, y);\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to a vec3\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param x defines the 1st component of the value\r\n * @param y defines the 2nd component of the value\r\n * @param z defines the 3rd component of the value\r\n * @returns true if the value was set\r\n */\r\n public setFloat3(uniform: Nullable, x: number, y: number, z: number): boolean {\r\n if (!uniform) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform3f(uniform, x, y, z);\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Set the value of an uniform to a vec4\r\n * @param uniform defines the webGL uniform location where to store the value\r\n * @param x defines the 1st component of the value\r\n * @param y defines the 2nd component of the value\r\n * @param z defines the 3rd component of the value\r\n * @param w defines the 4th component of the value\r\n * @returns true if the value was set\r\n */\r\n public setFloat4(uniform: Nullable, x: number, y: number, z: number, w: number): boolean {\r\n if (!uniform) {\r\n return false;\r\n }\r\n\r\n this._gl.uniform4f(uniform, x, y, z, w);\r\n\r\n return true;\r\n }\r\n\r\n // States\r\n\r\n /**\r\n * Apply all cached states (depth, culling, stencil and alpha)\r\n */\r\n public applyStates() {\r\n this._depthCullingState.apply(this._gl);\r\n this._stencilStateComposer.apply(this._gl);\r\n this._alphaState.apply(this._gl);\r\n\r\n if (this._colorWriteChanged) {\r\n this._colorWriteChanged = false;\r\n const enable = this._colorWrite;\r\n this._gl.colorMask(enable, enable, enable, enable);\r\n }\r\n }\r\n\r\n /**\r\n * Enable or disable color writing\r\n * @param enable defines the state to set\r\n */\r\n public setColorWrite(enable: boolean): void {\r\n if (enable !== this._colorWrite) {\r\n this._colorWriteChanged = true;\r\n this._colorWrite = enable;\r\n }\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if color writing is enabled\r\n * @returns the current color writing state\r\n */\r\n public getColorWrite(): boolean {\r\n return this._colorWrite;\r\n }\r\n\r\n /**\r\n * Gets the depth culling state manager\r\n */\r\n public get depthCullingState(): DepthCullingState {\r\n return this._depthCullingState;\r\n }\r\n\r\n /**\r\n * Gets the alpha state manager\r\n */\r\n public get alphaState(): AlphaState {\r\n return this._alphaState;\r\n }\r\n\r\n /**\r\n * Gets the stencil state manager\r\n */\r\n public get stencilState(): StencilState {\r\n return this._stencilState;\r\n }\r\n\r\n /**\r\n * Gets the stencil state composer\r\n */\r\n public get stencilStateComposer(): StencilStateComposer {\r\n return this._stencilStateComposer;\r\n }\r\n\r\n // Textures\r\n\r\n /**\r\n * Clears the list of texture accessible through engine.\r\n * This can help preventing texture load conflict due to name collision.\r\n */\r\n public clearInternalTexturesCache() {\r\n this._internalTexturesCache.length = 0;\r\n }\r\n\r\n /**\r\n * Force the entire cache to be cleared\r\n * You should not have to use this function unless your engine needs to share the webGL context with another engine\r\n * @param bruteForce defines a boolean to force clearing ALL caches (including stencil, detoh and alpha states)\r\n */\r\n public wipeCaches(bruteForce?: boolean): void {\r\n if (this.preventCacheWipeBetweenFrames && !bruteForce) {\r\n return;\r\n }\r\n this._currentEffect = null;\r\n this._viewportCached.x = 0;\r\n this._viewportCached.y = 0;\r\n this._viewportCached.z = 0;\r\n this._viewportCached.w = 0;\r\n\r\n // Done before in case we clean the attributes\r\n this._unbindVertexArrayObject();\r\n\r\n if (bruteForce) {\r\n this._currentProgram = null;\r\n this.resetTextureCache();\r\n\r\n this._stencilStateComposer.reset();\r\n\r\n this._depthCullingState.reset();\r\n this._depthCullingState.depthFunc = this._gl.LEQUAL;\r\n\r\n this._alphaState.reset();\r\n this._alphaMode = Constants.ALPHA_ADD;\r\n this._alphaEquation = Constants.ALPHA_DISABLE;\r\n\r\n this._colorWrite = true;\r\n this._colorWriteChanged = true;\r\n\r\n this._unpackFlipYCached = null;\r\n\r\n this._gl.pixelStorei(this._gl.UNPACK_COLORSPACE_CONVERSION_WEBGL, this._gl.NONE);\r\n this._gl.pixelStorei(this._gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, 0);\r\n\r\n this._mustWipeVertexAttributes = true;\r\n this.unbindAllAttributes();\r\n }\r\n\r\n this._resetVertexBufferBinding();\r\n this._cachedIndexBuffer = null;\r\n this._cachedEffectForVertexBuffers = null;\r\n this.bindIndexBuffer(null);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getSamplingParameters(samplingMode: number, generateMipMaps: boolean): { min: number; mag: number } {\r\n const gl = this._gl;\r\n let magFilter: GLenum = gl.NEAREST;\r\n let minFilter: GLenum = gl.NEAREST;\r\n\r\n switch (samplingMode) {\r\n case Constants.TEXTURE_LINEAR_LINEAR_MIPNEAREST:\r\n magFilter = gl.LINEAR;\r\n if (generateMipMaps) {\r\n minFilter = gl.LINEAR_MIPMAP_NEAREST;\r\n } else {\r\n minFilter = gl.LINEAR;\r\n }\r\n break;\r\n case Constants.TEXTURE_LINEAR_LINEAR_MIPLINEAR:\r\n magFilter = gl.LINEAR;\r\n if (generateMipMaps) {\r\n minFilter = gl.LINEAR_MIPMAP_LINEAR;\r\n } else {\r\n minFilter = gl.LINEAR;\r\n }\r\n break;\r\n case Constants.TEXTURE_NEAREST_NEAREST_MIPLINEAR:\r\n magFilter = gl.NEAREST;\r\n if (generateMipMaps) {\r\n minFilter = gl.NEAREST_MIPMAP_LINEAR;\r\n } else {\r\n minFilter = gl.NEAREST;\r\n }\r\n break;\r\n case Constants.TEXTURE_NEAREST_NEAREST_MIPNEAREST:\r\n magFilter = gl.NEAREST;\r\n if (generateMipMaps) {\r\n minFilter = gl.NEAREST_MIPMAP_NEAREST;\r\n } else {\r\n minFilter = gl.NEAREST;\r\n }\r\n break;\r\n case Constants.TEXTURE_NEAREST_LINEAR_MIPNEAREST:\r\n magFilter = gl.NEAREST;\r\n if (generateMipMaps) {\r\n minFilter = gl.LINEAR_MIPMAP_NEAREST;\r\n } else {\r\n minFilter = gl.LINEAR;\r\n }\r\n break;\r\n case Constants.TEXTURE_NEAREST_LINEAR_MIPLINEAR:\r\n magFilter = gl.NEAREST;\r\n if (generateMipMaps) {\r\n minFilter = gl.LINEAR_MIPMAP_LINEAR;\r\n } else {\r\n minFilter = gl.LINEAR;\r\n }\r\n break;\r\n case Constants.TEXTURE_NEAREST_LINEAR:\r\n magFilter = gl.NEAREST;\r\n minFilter = gl.LINEAR;\r\n break;\r\n case Constants.TEXTURE_NEAREST_NEAREST:\r\n magFilter = gl.NEAREST;\r\n minFilter = gl.NEAREST;\r\n break;\r\n case Constants.TEXTURE_LINEAR_NEAREST_MIPNEAREST:\r\n magFilter = gl.LINEAR;\r\n if (generateMipMaps) {\r\n minFilter = gl.NEAREST_MIPMAP_NEAREST;\r\n } else {\r\n minFilter = gl.NEAREST;\r\n }\r\n break;\r\n case Constants.TEXTURE_LINEAR_NEAREST_MIPLINEAR:\r\n magFilter = gl.LINEAR;\r\n if (generateMipMaps) {\r\n minFilter = gl.NEAREST_MIPMAP_LINEAR;\r\n } else {\r\n minFilter = gl.NEAREST;\r\n }\r\n break;\r\n case Constants.TEXTURE_LINEAR_LINEAR:\r\n magFilter = gl.LINEAR;\r\n minFilter = gl.LINEAR;\r\n break;\r\n case Constants.TEXTURE_LINEAR_NEAREST:\r\n magFilter = gl.LINEAR;\r\n minFilter = gl.NEAREST;\r\n break;\r\n }\r\n\r\n return {\r\n min: minFilter,\r\n mag: magFilter,\r\n };\r\n }\r\n\r\n /** @internal */\r\n protected _createTexture(): WebGLTexture {\r\n const texture = this._gl.createTexture();\r\n\r\n if (!texture) {\r\n throw new Error(\"Unable to create texture\");\r\n }\r\n\r\n return texture;\r\n }\r\n\r\n /** @internal */\r\n public _createHardwareTexture(): HardwareTextureWrapper {\r\n return new WebGLHardwareTexture(this._createTexture(), this._gl);\r\n }\r\n\r\n /**\r\n * Creates an internal texture without binding it to a framebuffer\r\n * @internal\r\n * @param size defines the size of the texture\r\n * @param options defines the options used to create the texture\r\n * @param delayGPUTextureCreation true to delay the texture creation the first time it is really needed. false to create it right away\r\n * @param source source type of the texture\r\n * @returns a new internal texture\r\n */\r\n public _createInternalTexture(\r\n size: TextureSize,\r\n options: boolean | InternalTextureCreationOptions,\r\n delayGPUTextureCreation = true,\r\n source = InternalTextureSource.Unknown\r\n ): InternalTexture {\r\n let generateMipMaps = false;\r\n let type = Constants.TEXTURETYPE_UNSIGNED_INT;\r\n let samplingMode = Constants.TEXTURE_TRILINEAR_SAMPLINGMODE;\r\n let format = Constants.TEXTUREFORMAT_RGBA;\r\n let useSRGBBuffer = false;\r\n let samples = 1;\r\n let label: string | undefined;\r\n if (options !== undefined && typeof options === \"object\") {\r\n generateMipMaps = !!options.generateMipMaps;\r\n type = options.type === undefined ? Constants.TEXTURETYPE_UNSIGNED_INT : options.type;\r\n samplingMode = options.samplingMode === undefined ? Constants.TEXTURE_TRILINEAR_SAMPLINGMODE : options.samplingMode;\r\n format = options.format === undefined ? Constants.TEXTUREFORMAT_RGBA : options.format;\r\n useSRGBBuffer = options.useSRGBBuffer === undefined ? false : options.useSRGBBuffer;\r\n samples = options.samples ?? 1;\r\n label = options.label;\r\n } else {\r\n generateMipMaps = !!options;\r\n }\r\n\r\n useSRGBBuffer &&= this._caps.supportSRGBBuffers && (this.webGLVersion > 1 || this.isWebGPU);\r\n\r\n if (type === Constants.TEXTURETYPE_FLOAT && !this._caps.textureFloatLinearFiltering) {\r\n // if floating point linear (gl.FLOAT) then force to NEAREST_SAMPLINGMODE\r\n samplingMode = Constants.TEXTURE_NEAREST_SAMPLINGMODE;\r\n } else if (type === Constants.TEXTURETYPE_HALF_FLOAT && !this._caps.textureHalfFloatLinearFiltering) {\r\n // if floating point linear (HALF_FLOAT) then force to NEAREST_SAMPLINGMODE\r\n samplingMode = Constants.TEXTURE_NEAREST_SAMPLINGMODE;\r\n }\r\n if (type === Constants.TEXTURETYPE_FLOAT && !this._caps.textureFloat) {\r\n type = Constants.TEXTURETYPE_UNSIGNED_INT;\r\n Logger.Warn(\"Float textures are not supported. Type forced to TEXTURETYPE_UNSIGNED_BYTE\");\r\n }\r\n\r\n const gl = this._gl;\r\n const texture = new InternalTexture(this, source);\r\n const width = (<{ width: number; height: number; layers?: number }>size).width || size;\r\n const height = (<{ width: number; height: number; layers?: number }>size).height || size;\r\n const layers = (<{ width: number; height: number; layers?: number }>size).layers || 0;\r\n const filters = this._getSamplingParameters(samplingMode, generateMipMaps);\r\n const target = layers !== 0 ? gl.TEXTURE_2D_ARRAY : gl.TEXTURE_2D;\r\n const sizedFormat = this._getRGBABufferInternalSizedFormat(type, format, useSRGBBuffer);\r\n const internalFormat = this._getInternalFormat(format);\r\n const textureType = this._getWebGLTextureType(type);\r\n\r\n // Bind\r\n this._bindTextureDirectly(target, texture);\r\n\r\n if (layers !== 0) {\r\n texture.is2DArray = true;\r\n gl.texImage3D(target, 0, sizedFormat, width, height, layers, 0, internalFormat, textureType, null);\r\n } else {\r\n gl.texImage2D(target, 0, sizedFormat, width, height, 0, internalFormat, textureType, null);\r\n }\r\n\r\n gl.texParameteri(target, gl.TEXTURE_MAG_FILTER, filters.mag);\r\n gl.texParameteri(target, gl.TEXTURE_MIN_FILTER, filters.min);\r\n gl.texParameteri(target, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\r\n gl.texParameteri(target, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\r\n\r\n // MipMaps\r\n if (generateMipMaps) {\r\n this._gl.generateMipmap(target);\r\n }\r\n\r\n this._bindTextureDirectly(target, null);\r\n\r\n texture._useSRGBBuffer = useSRGBBuffer;\r\n texture.baseWidth = width;\r\n texture.baseHeight = height;\r\n texture.width = width;\r\n texture.height = height;\r\n texture.depth = layers;\r\n texture.isReady = true;\r\n texture.samples = samples;\r\n texture.generateMipMaps = generateMipMaps;\r\n texture.samplingMode = samplingMode;\r\n texture.type = type;\r\n texture.format = format;\r\n texture.label = label;\r\n\r\n this._internalTexturesCache.push(texture);\r\n\r\n return texture;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getUseSRGBBuffer(useSRGBBuffer: boolean, noMipmap: boolean): boolean {\r\n // Generating mipmaps for sRGB textures is not supported in WebGL1 so we must disable the support if mipmaps is enabled\r\n return useSRGBBuffer && this._caps.supportSRGBBuffers && (this.webGLVersion > 1 || this.isWebGPU || noMipmap);\r\n }\r\n\r\n protected _createTextureBase(\r\n url: Nullable,\r\n noMipmap: boolean,\r\n invertY: boolean,\r\n scene: Nullable,\r\n samplingMode: number = Constants.TEXTURE_TRILINEAR_SAMPLINGMODE,\r\n onLoad: Nullable<(texture: InternalTexture) => void> = null,\r\n onError: Nullable<(message: string, exception: any) => void> = null,\r\n prepareTexture: (\r\n texture: InternalTexture,\r\n extension: string,\r\n scene: Nullable,\r\n img: HTMLImageElement | ImageBitmap | { width: number; height: number },\r\n invertY: boolean,\r\n noMipmap: boolean,\r\n isCompressed: boolean,\r\n processFunction: (\r\n width: number,\r\n height: number,\r\n img: HTMLImageElement | ImageBitmap | { width: number; height: number },\r\n extension: string,\r\n texture: InternalTexture,\r\n continuationCallback: () => void\r\n ) => boolean,\r\n samplingMode: number\r\n ) => void,\r\n prepareTextureProcessFunction: (\r\n width: number,\r\n height: number,\r\n img: HTMLImageElement | ImageBitmap | { width: number; height: number },\r\n extension: string,\r\n texture: InternalTexture,\r\n continuationCallback: () => void\r\n ) => boolean,\r\n buffer: Nullable = null,\r\n fallback: Nullable = null,\r\n format: Nullable = null,\r\n forcedExtension: Nullable = null,\r\n mimeType?: string,\r\n loaderOptions?: any,\r\n useSRGBBuffer?: boolean\r\n ): InternalTexture {\r\n url = url || \"\";\r\n const fromData = url.substr(0, 5) === \"data:\";\r\n const fromBlob = url.substr(0, 5) === \"blob:\";\r\n const isBase64 = fromData && url.indexOf(\";base64,\") !== -1;\r\n\r\n const texture = fallback ? fallback : new InternalTexture(this, InternalTextureSource.Url);\r\n\r\n if (texture !== fallback) {\r\n texture.label = url.substring(0, 60); // default label, can be overriden by the caller\r\n }\r\n\r\n const originalUrl = url;\r\n if (this._transformTextureUrl && !isBase64 && !fallback && !buffer) {\r\n url = this._transformTextureUrl(url);\r\n }\r\n\r\n if (originalUrl !== url) {\r\n texture._originalUrl = originalUrl;\r\n }\r\n\r\n // establish the file extension, if possible\r\n const lastDot = url.lastIndexOf(\".\");\r\n let extension = forcedExtension ? forcedExtension : lastDot > -1 ? url.substring(lastDot).toLowerCase() : \"\";\r\n let loader: Nullable = null;\r\n\r\n // Remove query string\r\n const queryStringIndex = extension.indexOf(\"?\");\r\n\r\n if (queryStringIndex > -1) {\r\n extension = extension.split(\"?\")[0];\r\n }\r\n\r\n for (const availableLoader of ThinEngine._TextureLoaders) {\r\n if (availableLoader.canLoad(extension, mimeType)) {\r\n loader = availableLoader;\r\n break;\r\n }\r\n }\r\n\r\n if (scene) {\r\n scene.addPendingData(texture);\r\n }\r\n texture.url = url;\r\n texture.generateMipMaps = !noMipmap;\r\n texture.samplingMode = samplingMode;\r\n texture.invertY = invertY;\r\n texture._useSRGBBuffer = this._getUseSRGBBuffer(!!useSRGBBuffer, noMipmap);\r\n\r\n if (!this._doNotHandleContextLost) {\r\n // Keep a link to the buffer only if we plan to handle context lost\r\n texture._buffer = buffer;\r\n }\r\n\r\n let onLoadObserver: Nullable> = null;\r\n if (onLoad && !fallback) {\r\n onLoadObserver = texture.onLoadedObservable.add(onLoad);\r\n }\r\n\r\n if (!fallback) {\r\n this._internalTexturesCache.push(texture);\r\n }\r\n\r\n const onInternalError = (message?: string, exception?: any) => {\r\n if (scene) {\r\n scene.removePendingData(texture);\r\n }\r\n\r\n if (url === originalUrl) {\r\n if (onLoadObserver) {\r\n texture.onLoadedObservable.remove(onLoadObserver);\r\n }\r\n\r\n if (EngineStore.UseFallbackTexture) {\r\n this._createTextureBase(\r\n EngineStore.FallbackTexture,\r\n noMipmap,\r\n texture.invertY,\r\n scene,\r\n samplingMode,\r\n null,\r\n onError,\r\n prepareTexture,\r\n prepareTextureProcessFunction,\r\n buffer,\r\n texture\r\n );\r\n }\r\n\r\n message = (message || \"Unknown error\") + (EngineStore.UseFallbackTexture ? \" - Fallback texture was used\" : \"\");\r\n texture.onErrorObservable.notifyObservers({ message, exception });\r\n if (onError) {\r\n onError(message, exception);\r\n }\r\n } else {\r\n // fall back to the original url if the transformed url fails to load\r\n Logger.Warn(`Failed to load ${url}, falling back to ${originalUrl}`);\r\n this._createTextureBase(\r\n originalUrl,\r\n noMipmap,\r\n texture.invertY,\r\n scene,\r\n samplingMode,\r\n onLoad,\r\n onError,\r\n prepareTexture,\r\n prepareTextureProcessFunction,\r\n buffer,\r\n texture,\r\n format,\r\n forcedExtension,\r\n mimeType,\r\n loaderOptions,\r\n useSRGBBuffer\r\n );\r\n }\r\n };\r\n\r\n // processing for non-image formats\r\n if (loader) {\r\n const callback = (data: ArrayBufferView) => {\r\n loader!.loadData(\r\n data,\r\n texture,\r\n (width: number, height: number, loadMipmap: boolean, isCompressed: boolean, done: () => void, loadFailed) => {\r\n if (loadFailed) {\r\n onInternalError(\"TextureLoader failed to load data\");\r\n } else {\r\n prepareTexture(\r\n texture,\r\n extension,\r\n scene,\r\n { width, height },\r\n texture.invertY,\r\n !loadMipmap,\r\n isCompressed,\r\n () => {\r\n done();\r\n return false;\r\n },\r\n samplingMode\r\n );\r\n }\r\n },\r\n loaderOptions\r\n );\r\n };\r\n\r\n if (!buffer) {\r\n this._loadFile(\r\n url,\r\n (data) => callback(new Uint8Array(data as ArrayBuffer)),\r\n undefined,\r\n scene ? scene.offlineProvider : undefined,\r\n true,\r\n (request?: IWebRequest, exception?: any) => {\r\n onInternalError(\"Unable to load \" + (request ? request.responseURL : url, exception));\r\n }\r\n );\r\n } else {\r\n if (buffer instanceof ArrayBuffer) {\r\n callback(new Uint8Array(buffer));\r\n } else if (ArrayBuffer.isView(buffer)) {\r\n callback(buffer);\r\n } else {\r\n if (onError) {\r\n onError(\"Unable to load: only ArrayBuffer or ArrayBufferView is supported\", null);\r\n }\r\n }\r\n }\r\n } else {\r\n const onload = (img: HTMLImageElement | ImageBitmap) => {\r\n if (fromBlob && !this._doNotHandleContextLost) {\r\n // We need to store the image if we need to rebuild the texture\r\n // in case of a webgl context lost\r\n texture._buffer = img;\r\n }\r\n\r\n prepareTexture(texture, extension, scene, img, texture.invertY, noMipmap, false, prepareTextureProcessFunction, samplingMode);\r\n };\r\n // According to the WebGL spec section 6.10, ImageBitmaps must be inverted on creation.\r\n // So, we pass imageOrientation to _FileToolsLoadImage() as it may create an ImageBitmap.\r\n\r\n if (!fromData || isBase64) {\r\n if (buffer && (typeof (buffer).decoding === \"string\" || (buffer).close)) {\r\n onload(buffer);\r\n } else {\r\n ThinEngine._FileToolsLoadImage(\r\n url,\r\n onload,\r\n onInternalError,\r\n scene ? scene.offlineProvider : null,\r\n mimeType,\r\n texture.invertY && this._features.needsInvertingBitmap ? { imageOrientation: \"flipY\" } : undefined\r\n );\r\n }\r\n } else if (typeof buffer === \"string\" || buffer instanceof ArrayBuffer || ArrayBuffer.isView(buffer) || buffer instanceof Blob) {\r\n ThinEngine._FileToolsLoadImage(\r\n buffer,\r\n onload,\r\n onInternalError,\r\n scene ? scene.offlineProvider : null,\r\n mimeType,\r\n texture.invertY && this._features.needsInvertingBitmap ? { imageOrientation: \"flipY\" } : undefined\r\n );\r\n } else if (buffer) {\r\n onload(buffer);\r\n }\r\n }\r\n\r\n return texture;\r\n }\r\n\r\n /**\r\n * Usually called from Texture.ts.\r\n * Passed information to create a WebGLTexture\r\n * @param url defines a value which contains one of the following:\r\n * * A conventional http URL, e.g. 'http://...' or 'file://...'\r\n * * A base64 string of in-line texture data, e.g. '...'\r\n * * An indicator that data being passed using the buffer parameter, e.g. 'data:mytexture.jpg'\r\n * @param noMipmap defines a boolean indicating that no mipmaps shall be generated. Ignored for compressed textures. They must be in the file\r\n * @param invertY when true, image is flipped when loaded. You probably want true. Certain compressed textures may invert this if their default is inverted (eg. ktx)\r\n * @param scene needed for loading to the correct scene\r\n * @param samplingMode mode with should be used sample / access the texture (Default: Texture.TRILINEAR_SAMPLINGMODE)\r\n * @param onLoad optional callback to be called upon successful completion\r\n * @param onError optional callback to be called upon failure\r\n * @param buffer a source of a file previously fetched as either a base64 string, an ArrayBuffer (compressed or image format), HTMLImageElement (image format), or a Blob\r\n * @param fallback an internal argument in case the function must be called again, due to etc1 not having alpha capabilities\r\n * @param format internal format. Default: RGB when extension is '.jpg' else RGBA. Ignored for compressed textures\r\n * @param forcedExtension defines the extension to use to pick the right loader\r\n * @param mimeType defines an optional mime type\r\n * @param loaderOptions options to be passed to the loader\r\n * @param creationFlags specific flags to use when creating the texture (Constants.TEXTURE_CREATIONFLAG_STORAGE for storage textures, for eg)\r\n * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU).\r\n * @returns a InternalTexture for assignment back into BABYLON.Texture\r\n */\r\n public createTexture(\r\n url: Nullable,\r\n noMipmap: boolean,\r\n invertY: boolean,\r\n scene: Nullable,\r\n samplingMode: number = Constants.TEXTURE_TRILINEAR_SAMPLINGMODE,\r\n onLoad: Nullable<(texture: InternalTexture) => void> = null,\r\n onError: Nullable<(message: string, exception: any) => void> = null,\r\n buffer: Nullable = null,\r\n fallback: Nullable = null,\r\n format: Nullable = null,\r\n forcedExtension: Nullable = null,\r\n mimeType?: string,\r\n loaderOptions?: any,\r\n creationFlags?: number,\r\n useSRGBBuffer?: boolean\r\n ): InternalTexture {\r\n return this._createTextureBase(\r\n url,\r\n noMipmap,\r\n invertY,\r\n scene,\r\n samplingMode,\r\n onLoad,\r\n onError,\r\n this._prepareWebGLTexture.bind(this),\r\n (potWidth, potHeight, img, extension, texture, continuationCallback) => {\r\n const gl = this._gl;\r\n const isPot = img.width === potWidth && img.height === potHeight;\r\n\r\n const internalFormat = format\r\n ? this._getInternalFormat(format, texture._useSRGBBuffer)\r\n : extension === \".jpg\" && !texture._useSRGBBuffer\r\n ? gl.RGB\r\n : texture._useSRGBBuffer\r\n ? this._glSRGBExtensionValues.SRGB8_ALPHA8\r\n : gl.RGBA;\r\n let texelFormat = format ? this._getInternalFormat(format) : extension === \".jpg\" && !texture._useSRGBBuffer ? gl.RGB : gl.RGBA;\r\n\r\n if (texture._useSRGBBuffer && this.webGLVersion === 1) {\r\n texelFormat = internalFormat;\r\n }\r\n\r\n if (isPot) {\r\n gl.texImage2D(gl.TEXTURE_2D, 0, internalFormat, texelFormat, gl.UNSIGNED_BYTE, img as any);\r\n return false;\r\n }\r\n\r\n const maxTextureSize = this._caps.maxTextureSize;\r\n\r\n if (img.width > maxTextureSize || img.height > maxTextureSize || !this._supportsHardwareTextureRescaling) {\r\n this._prepareWorkingCanvas();\r\n if (!this._workingCanvas || !this._workingContext) {\r\n return false;\r\n }\r\n\r\n this._workingCanvas.width = potWidth;\r\n this._workingCanvas.height = potHeight;\r\n\r\n this._workingContext.drawImage(img as any, 0, 0, img.width, img.height, 0, 0, potWidth, potHeight);\r\n gl.texImage2D(gl.TEXTURE_2D, 0, internalFormat, texelFormat, gl.UNSIGNED_BYTE, this._workingCanvas as TexImageSource);\r\n\r\n texture.width = potWidth;\r\n texture.height = potHeight;\r\n\r\n return false;\r\n } else {\r\n // Using shaders when possible to rescale because canvas.drawImage is lossy\r\n const source = new InternalTexture(this, InternalTextureSource.Temp);\r\n this._bindTextureDirectly(gl.TEXTURE_2D, source, true);\r\n gl.texImage2D(gl.TEXTURE_2D, 0, internalFormat, texelFormat, gl.UNSIGNED_BYTE, img as any);\r\n\r\n this._rescaleTexture(source, texture, scene, internalFormat, () => {\r\n this._releaseTexture(source);\r\n this._bindTextureDirectly(gl.TEXTURE_2D, texture, true);\r\n\r\n continuationCallback();\r\n });\r\n }\r\n\r\n return true;\r\n },\r\n buffer,\r\n fallback,\r\n format,\r\n forcedExtension,\r\n mimeType,\r\n loaderOptions,\r\n useSRGBBuffer\r\n );\r\n }\r\n\r\n /**\r\n * Loads an image as an HTMLImageElement.\r\n * @param input url string, ArrayBuffer, or Blob to load\r\n * @param onLoad callback called when the image successfully loads\r\n * @param onError callback called when the image fails to load\r\n * @param offlineProvider offline provider for caching\r\n * @param mimeType optional mime type\r\n * @param imageBitmapOptions optional the options to use when creating an ImageBitmap\r\n * @returns the HTMLImageElement of the loaded image\r\n * @internal\r\n */\r\n public static _FileToolsLoadImage(\r\n input: string | ArrayBuffer | ArrayBufferView | Blob,\r\n onLoad: (img: HTMLImageElement | ImageBitmap) => void,\r\n onError: (message?: string, exception?: any) => void,\r\n offlineProvider: Nullable,\r\n mimeType?: string,\r\n imageBitmapOptions?: ImageBitmapOptions\r\n ): Nullable {\r\n throw _WarnImport(\"FileTools\");\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _rescaleTexture(source: InternalTexture, destination: InternalTexture, scene: Nullable, internalFormat: number, onComplete: () => void): void {}\r\n\r\n /**\r\n * Creates a raw texture\r\n * @param data defines the data to store in the texture\r\n * @param width defines the width of the texture\r\n * @param height defines the height of the texture\r\n * @param format defines the format of the data\r\n * @param generateMipMaps defines if the engine should generate the mip levels\r\n * @param invertY defines if data must be stored with Y axis inverted\r\n * @param samplingMode defines the required sampling mode (Texture.NEAREST_SAMPLINGMODE by default)\r\n * @param compression defines the compression used (null by default)\r\n * @param type defines the type fo the data (Engine.TEXTURETYPE_UNSIGNED_INT by default)\r\n * @param creationFlags specific flags to use when creating the texture (Constants.TEXTURE_CREATIONFLAG_STORAGE for storage textures, for eg)\r\n * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU).\r\n * @returns the raw texture inside an InternalTexture\r\n */\r\n public createRawTexture(\r\n data: Nullable,\r\n width: number,\r\n height: number,\r\n format: number,\r\n generateMipMaps: boolean,\r\n invertY: boolean,\r\n samplingMode: number,\r\n compression: Nullable = null,\r\n type: number = Constants.TEXTURETYPE_UNSIGNED_INT,\r\n creationFlags = 0,\r\n useSRGBBuffer: boolean = false\r\n ): InternalTexture {\r\n throw _WarnImport(\"Engine.RawTexture\");\r\n }\r\n\r\n /**\r\n * Creates a new raw cube texture\r\n * @param data defines the array of data to use to create each face\r\n * @param size defines the size of the textures\r\n * @param format defines the format of the data\r\n * @param type defines the type of the data (like Engine.TEXTURETYPE_UNSIGNED_INT)\r\n * @param generateMipMaps defines if the engine should generate the mip levels\r\n * @param invertY defines if data must be stored with Y axis inverted\r\n * @param samplingMode defines the required sampling mode (like Texture.NEAREST_SAMPLINGMODE)\r\n * @param compression defines the compression used (null by default)\r\n * @returns the cube texture as an InternalTexture\r\n */\r\n public createRawCubeTexture(\r\n data: Nullable,\r\n size: number,\r\n format: number,\r\n type: number,\r\n generateMipMaps: boolean,\r\n invertY: boolean,\r\n samplingMode: number,\r\n compression: Nullable = null\r\n ): InternalTexture {\r\n throw _WarnImport(\"Engine.RawTexture\");\r\n }\r\n\r\n /**\r\n * Creates a new raw 3D texture\r\n * @param data defines the data used to create the texture\r\n * @param width defines the width of the texture\r\n * @param height defines the height of the texture\r\n * @param depth defines the depth of the texture\r\n * @param format defines the format of the texture\r\n * @param generateMipMaps defines if the engine must generate mip levels\r\n * @param invertY defines if data must be stored with Y axis inverted\r\n * @param samplingMode defines the required sampling mode (like Texture.NEAREST_SAMPLINGMODE)\r\n * @param compression defines the compressed used (can be null)\r\n * @param textureType defines the compressed used (can be null)\r\n * @returns a new raw 3D texture (stored in an InternalTexture)\r\n */\r\n public createRawTexture3D(\r\n data: Nullable,\r\n width: number,\r\n height: number,\r\n depth: number,\r\n format: number,\r\n generateMipMaps: boolean,\r\n invertY: boolean,\r\n samplingMode: number,\r\n compression: Nullable = null,\r\n textureType = Constants.TEXTURETYPE_UNSIGNED_INT\r\n ): InternalTexture {\r\n throw _WarnImport(\"Engine.RawTexture\");\r\n }\r\n\r\n /**\r\n * Creates a new raw 2D array texture\r\n * @param data defines the data used to create the texture\r\n * @param width defines the width of the texture\r\n * @param height defines the height of the texture\r\n * @param depth defines the number of layers of the texture\r\n * @param format defines the format of the texture\r\n * @param generateMipMaps defines if the engine must generate mip levels\r\n * @param invertY defines if data must be stored with Y axis inverted\r\n * @param samplingMode defines the required sampling mode (like Texture.NEAREST_SAMPLINGMODE)\r\n * @param compression defines the compressed used (can be null)\r\n * @param textureType defines the compressed used (can be null)\r\n * @returns a new raw 2D array texture (stored in an InternalTexture)\r\n */\r\n public createRawTexture2DArray(\r\n data: Nullable,\r\n width: number,\r\n height: number,\r\n depth: number,\r\n format: number,\r\n generateMipMaps: boolean,\r\n invertY: boolean,\r\n samplingMode: number,\r\n compression: Nullable = null,\r\n textureType = Constants.TEXTURETYPE_UNSIGNED_INT\r\n ): InternalTexture {\r\n throw _WarnImport(\"Engine.RawTexture\");\r\n }\r\n\r\n private _unpackFlipYCached: Nullable = null;\r\n\r\n /**\r\n * In case you are sharing the context with other applications, it might\r\n * be interested to not cache the unpack flip y state to ensure a consistent\r\n * value would be set.\r\n */\r\n public enableUnpackFlipYCached = true;\r\n\r\n /**\r\n * @internal\r\n */\r\n public _unpackFlipY(value: boolean): void {\r\n if (this._unpackFlipYCached !== value) {\r\n this._gl.pixelStorei(this._gl.UNPACK_FLIP_Y_WEBGL, value ? 1 : 0);\r\n\r\n if (this.enableUnpackFlipYCached) {\r\n this._unpackFlipYCached = value;\r\n }\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _getUnpackAlignement(): number {\r\n return this._gl.getParameter(this._gl.UNPACK_ALIGNMENT);\r\n }\r\n\r\n private _getTextureTarget(texture: InternalTexture): number {\r\n if (texture.isCube) {\r\n return this._gl.TEXTURE_CUBE_MAP;\r\n } else if (texture.is3D) {\r\n return this._gl.TEXTURE_3D;\r\n } else if (texture.is2DArray || texture.isMultiview) {\r\n return this._gl.TEXTURE_2D_ARRAY;\r\n }\r\n return this._gl.TEXTURE_2D;\r\n }\r\n\r\n /**\r\n * Update the sampling mode of a given texture\r\n * @param samplingMode defines the required sampling mode\r\n * @param texture defines the texture to update\r\n * @param generateMipMaps defines whether to generate mipmaps for the texture\r\n */\r\n public updateTextureSamplingMode(samplingMode: number, texture: InternalTexture, generateMipMaps: boolean = false): void {\r\n const target = this._getTextureTarget(texture);\r\n const filters = this._getSamplingParameters(samplingMode, texture.useMipMaps || generateMipMaps);\r\n\r\n this._setTextureParameterInteger(target, this._gl.TEXTURE_MAG_FILTER, filters.mag, texture);\r\n this._setTextureParameterInteger(target, this._gl.TEXTURE_MIN_FILTER, filters.min);\r\n\r\n if (generateMipMaps) {\r\n texture.generateMipMaps = true;\r\n this._gl.generateMipmap(target);\r\n }\r\n\r\n this._bindTextureDirectly(target, null);\r\n\r\n texture.samplingMode = samplingMode;\r\n }\r\n\r\n /**\r\n * Update the dimensions of a texture\r\n * @param texture texture to update\r\n * @param width new width of the texture\r\n * @param height new height of the texture\r\n * @param depth new depth of the texture\r\n */\r\n public updateTextureDimensions(texture: InternalTexture, width: number, height: number, depth: number = 1): void {}\r\n\r\n /**\r\n * Update the sampling mode of a given texture\r\n * @param texture defines the texture to update\r\n * @param wrapU defines the texture wrap mode of the u coordinates\r\n * @param wrapV defines the texture wrap mode of the v coordinates\r\n * @param wrapR defines the texture wrap mode of the r coordinates\r\n */\r\n public updateTextureWrappingMode(texture: InternalTexture, wrapU: Nullable, wrapV: Nullable = null, wrapR: Nullable = null): void {\r\n const target = this._getTextureTarget(texture);\r\n\r\n if (wrapU !== null) {\r\n this._setTextureParameterInteger(target, this._gl.TEXTURE_WRAP_S, this._getTextureWrapMode(wrapU), texture);\r\n texture._cachedWrapU = wrapU;\r\n }\r\n if (wrapV !== null) {\r\n this._setTextureParameterInteger(target, this._gl.TEXTURE_WRAP_T, this._getTextureWrapMode(wrapV), texture);\r\n texture._cachedWrapV = wrapV;\r\n }\r\n if ((texture.is2DArray || texture.is3D) && wrapR !== null) {\r\n this._setTextureParameterInteger(target, this._gl.TEXTURE_WRAP_R, this._getTextureWrapMode(wrapR), texture);\r\n texture._cachedWrapR = wrapR;\r\n }\r\n\r\n this._bindTextureDirectly(target, null);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _setupDepthStencilTexture(\r\n internalTexture: InternalTexture,\r\n size: number | { width: number; height: number; layers?: number },\r\n generateStencil: boolean,\r\n bilinearFiltering: boolean,\r\n comparisonFunction: number,\r\n samples = 1\r\n ): void {\r\n const width = (<{ width: number; height: number; layers?: number }>size).width || size;\r\n const height = (<{ width: number; height: number; layers?: number }>size).height || size;\r\n const layers = (<{ width: number; height: number; layers?: number }>size).layers || 0;\r\n\r\n internalTexture.baseWidth = width;\r\n internalTexture.baseHeight = height;\r\n internalTexture.width = width;\r\n internalTexture.height = height;\r\n internalTexture.is2DArray = layers > 0;\r\n internalTexture.depth = layers;\r\n internalTexture.isReady = true;\r\n internalTexture.samples = samples;\r\n internalTexture.generateMipMaps = false;\r\n internalTexture.samplingMode = bilinearFiltering ? Constants.TEXTURE_BILINEAR_SAMPLINGMODE : Constants.TEXTURE_NEAREST_SAMPLINGMODE;\r\n internalTexture.type = Constants.TEXTURETYPE_UNSIGNED_INT;\r\n internalTexture._comparisonFunction = comparisonFunction;\r\n\r\n const gl = this._gl;\r\n const target = this._getTextureTarget(internalTexture);\r\n const samplingParameters = this._getSamplingParameters(internalTexture.samplingMode, false);\r\n gl.texParameteri(target, gl.TEXTURE_MAG_FILTER, samplingParameters.mag);\r\n gl.texParameteri(target, gl.TEXTURE_MIN_FILTER, samplingParameters.min);\r\n gl.texParameteri(target, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\r\n gl.texParameteri(target, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\r\n\r\n // TEXTURE_COMPARE_FUNC/MODE are only availble in WebGL2.\r\n if (this.webGLVersion > 1) {\r\n if (comparisonFunction === 0) {\r\n gl.texParameteri(target, gl.TEXTURE_COMPARE_FUNC, Constants.LEQUAL);\r\n gl.texParameteri(target, gl.TEXTURE_COMPARE_MODE, gl.NONE);\r\n } else {\r\n gl.texParameteri(target, gl.TEXTURE_COMPARE_FUNC, comparisonFunction);\r\n gl.texParameteri(target, gl.TEXTURE_COMPARE_MODE, gl.COMPARE_REF_TO_TEXTURE);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _uploadCompressedDataToTextureDirectly(\r\n texture: InternalTexture,\r\n internalFormat: number,\r\n width: number,\r\n height: number,\r\n data: ArrayBufferView,\r\n faceIndex: number = 0,\r\n lod: number = 0\r\n ) {\r\n const gl = this._gl;\r\n\r\n let target: GLenum = gl.TEXTURE_2D;\r\n if (texture.isCube) {\r\n target = gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex;\r\n }\r\n\r\n if (texture._useSRGBBuffer) {\r\n switch (internalFormat) {\r\n case Constants.TEXTUREFORMAT_COMPRESSED_RGB8_ETC2:\r\n case Constants.TEXTUREFORMAT_COMPRESSED_RGB_ETC1_WEBGL:\r\n // Note, if using ETC1 and sRGB is requested, this will use ETC2 if available.\r\n if (this._caps.etc2) {\r\n internalFormat = gl.COMPRESSED_SRGB8_ETC2;\r\n } else {\r\n texture._useSRGBBuffer = false;\r\n }\r\n break;\r\n case Constants.TEXTUREFORMAT_COMPRESSED_RGBA8_ETC2_EAC:\r\n if (this._caps.etc2) {\r\n internalFormat = gl.COMPRESSED_SRGB8_ALPHA8_ETC2_EAC;\r\n } else {\r\n texture._useSRGBBuffer = false;\r\n }\r\n break;\r\n case Constants.TEXTUREFORMAT_COMPRESSED_RGBA_BPTC_UNORM:\r\n internalFormat = gl.COMPRESSED_SRGB_ALPHA_BPTC_UNORM_EXT;\r\n break;\r\n case Constants.TEXTUREFORMAT_COMPRESSED_RGBA_ASTC_4x4:\r\n internalFormat = gl.COMPRESSED_SRGB8_ALPHA8_ASTC_4x4_KHR;\r\n break;\r\n case Constants.TEXTUREFORMAT_COMPRESSED_RGB_S3TC_DXT1:\r\n if (this._caps.s3tc_srgb) {\r\n internalFormat = gl.COMPRESSED_SRGB_S3TC_DXT1_EXT;\r\n } else {\r\n // S3TC sRGB extension not supported\r\n texture._useSRGBBuffer = false;\r\n }\r\n break;\r\n case Constants.TEXTUREFORMAT_COMPRESSED_RGBA_S3TC_DXT1:\r\n if (this._caps.s3tc_srgb) {\r\n internalFormat = gl.COMPRESSED_SRGB_ALPHA_S3TC_DXT1_EXT;\r\n } else {\r\n // S3TC sRGB extension not supported\r\n texture._useSRGBBuffer = false;\r\n }\r\n break;\r\n case Constants.TEXTUREFORMAT_COMPRESSED_RGBA_S3TC_DXT5:\r\n if (this._caps.s3tc_srgb) {\r\n internalFormat = gl.COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT;\r\n } else {\r\n // S3TC sRGB extension not supported\r\n texture._useSRGBBuffer = false;\r\n }\r\n break;\r\n default:\r\n // We don't support a sRGB format corresponding to internalFormat, so revert to non sRGB format\r\n texture._useSRGBBuffer = false;\r\n break;\r\n }\r\n }\r\n\r\n this._gl.compressedTexImage2D(target, lod, internalFormat, width, height, 0, data);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _uploadDataToTextureDirectly(\r\n texture: InternalTexture,\r\n imageData: ArrayBufferView,\r\n faceIndex: number = 0,\r\n lod: number = 0,\r\n babylonInternalFormat?: number,\r\n useTextureWidthAndHeight = false\r\n ): void {\r\n const gl = this._gl;\r\n\r\n const textureType = this._getWebGLTextureType(texture.type);\r\n const format = this._getInternalFormat(texture.format);\r\n const internalFormat =\r\n babylonInternalFormat === undefined\r\n ? this._getRGBABufferInternalSizedFormat(texture.type, texture.format, texture._useSRGBBuffer)\r\n : this._getInternalFormat(babylonInternalFormat, texture._useSRGBBuffer);\r\n\r\n this._unpackFlipY(texture.invertY);\r\n\r\n let target: GLenum = gl.TEXTURE_2D;\r\n if (texture.isCube) {\r\n target = gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex;\r\n }\r\n\r\n const lodMaxWidth = Math.round(Math.log(texture.width) * Math.LOG2E);\r\n const lodMaxHeight = Math.round(Math.log(texture.height) * Math.LOG2E);\r\n const width = useTextureWidthAndHeight ? texture.width : Math.pow(2, Math.max(lodMaxWidth - lod, 0));\r\n const height = useTextureWidthAndHeight ? texture.height : Math.pow(2, Math.max(lodMaxHeight - lod, 0));\r\n\r\n gl.texImage2D(target, lod, internalFormat, width, height, 0, format, textureType, imageData);\r\n }\r\n\r\n /**\r\n * Update a portion of an internal texture\r\n * @param texture defines the texture to update\r\n * @param imageData defines the data to store into the texture\r\n * @param xOffset defines the x coordinates of the update rectangle\r\n * @param yOffset defines the y coordinates of the update rectangle\r\n * @param width defines the width of the update rectangle\r\n * @param height defines the height of the update rectangle\r\n * @param faceIndex defines the face index if texture is a cube (0 by default)\r\n * @param lod defines the lod level to update (0 by default)\r\n * @param generateMipMaps defines whether to generate mipmaps or not\r\n */\r\n public updateTextureData(\r\n texture: InternalTexture,\r\n imageData: ArrayBufferView,\r\n xOffset: number,\r\n yOffset: number,\r\n width: number,\r\n height: number,\r\n faceIndex: number = 0,\r\n lod: number = 0,\r\n generateMipMaps = false\r\n ): void {\r\n const gl = this._gl;\r\n\r\n const textureType = this._getWebGLTextureType(texture.type);\r\n const format = this._getInternalFormat(texture.format);\r\n\r\n this._unpackFlipY(texture.invertY);\r\n\r\n let targetForBinding: GLenum = gl.TEXTURE_2D;\r\n let target: GLenum = gl.TEXTURE_2D;\r\n if (texture.isCube) {\r\n target = gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex;\r\n targetForBinding = gl.TEXTURE_CUBE_MAP;\r\n }\r\n\r\n this._bindTextureDirectly(targetForBinding, texture, true);\r\n\r\n gl.texSubImage2D(target, lod, xOffset, yOffset, width, height, format, textureType, imageData);\r\n\r\n if (generateMipMaps) {\r\n this._gl.generateMipmap(target);\r\n }\r\n\r\n this._bindTextureDirectly(targetForBinding, null);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _uploadArrayBufferViewToTexture(texture: InternalTexture, imageData: ArrayBufferView, faceIndex: number = 0, lod: number = 0): void {\r\n const gl = this._gl;\r\n const bindTarget = texture.isCube ? gl.TEXTURE_CUBE_MAP : gl.TEXTURE_2D;\r\n\r\n this._bindTextureDirectly(bindTarget, texture, true);\r\n\r\n this._uploadDataToTextureDirectly(texture, imageData, faceIndex, lod);\r\n\r\n this._bindTextureDirectly(bindTarget, null, true);\r\n }\r\n\r\n protected _prepareWebGLTextureContinuation(texture: InternalTexture, scene: Nullable, noMipmap: boolean, isCompressed: boolean, samplingMode: number): void {\r\n const gl = this._gl;\r\n if (!gl) {\r\n return;\r\n }\r\n\r\n const filters = this._getSamplingParameters(samplingMode, !noMipmap);\r\n\r\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, filters.mag);\r\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, filters.min);\r\n\r\n if (!noMipmap && !isCompressed) {\r\n gl.generateMipmap(gl.TEXTURE_2D);\r\n }\r\n\r\n this._bindTextureDirectly(gl.TEXTURE_2D, null);\r\n\r\n // this.resetTextureCache();\r\n if (scene) {\r\n scene.removePendingData(texture);\r\n }\r\n\r\n texture.onLoadedObservable.notifyObservers(texture);\r\n texture.onLoadedObservable.clear();\r\n }\r\n\r\n private _prepareWebGLTexture(\r\n texture: InternalTexture,\r\n extension: string,\r\n scene: Nullable,\r\n img: HTMLImageElement | ImageBitmap | { width: number; height: number },\r\n invertY: boolean,\r\n noMipmap: boolean,\r\n isCompressed: boolean,\r\n processFunction: (\r\n width: number,\r\n height: number,\r\n img: HTMLImageElement | ImageBitmap | { width: number; height: number },\r\n extension: string,\r\n texture: InternalTexture,\r\n continuationCallback: () => void\r\n ) => boolean,\r\n samplingMode: number = Constants.TEXTURE_TRILINEAR_SAMPLINGMODE\r\n ): void {\r\n const maxTextureSize = this.getCaps().maxTextureSize;\r\n const potWidth = Math.min(maxTextureSize, this.needPOTTextures ? ThinEngine.GetExponentOfTwo(img.width, maxTextureSize) : img.width);\r\n const potHeight = Math.min(maxTextureSize, this.needPOTTextures ? ThinEngine.GetExponentOfTwo(img.height, maxTextureSize) : img.height);\r\n\r\n const gl = this._gl;\r\n if (!gl) {\r\n return;\r\n }\r\n\r\n if (!texture._hardwareTexture) {\r\n // this.resetTextureCache();\r\n if (scene) {\r\n scene.removePendingData(texture);\r\n }\r\n\r\n return;\r\n }\r\n\r\n this._bindTextureDirectly(gl.TEXTURE_2D, texture, true);\r\n this._unpackFlipY(invertY === undefined ? true : invertY ? true : false);\r\n\r\n texture.baseWidth = img.width;\r\n texture.baseHeight = img.height;\r\n texture.width = potWidth;\r\n texture.height = potHeight;\r\n texture.isReady = true;\r\n\r\n if (\r\n processFunction(potWidth, potHeight, img, extension, texture, () => {\r\n this._prepareWebGLTextureContinuation(texture, scene, noMipmap, isCompressed, samplingMode);\r\n })\r\n ) {\r\n // Returning as texture needs extra async steps\r\n return;\r\n }\r\n\r\n this._prepareWebGLTextureContinuation(texture, scene, noMipmap, isCompressed, samplingMode);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _setupFramebufferDepthAttachments(\r\n generateStencilBuffer: boolean,\r\n generateDepthBuffer: boolean,\r\n width: number,\r\n height: number,\r\n samples = 1\r\n ): Nullable {\r\n const gl = this._gl;\r\n\r\n // Create the depth/stencil buffer\r\n if (generateStencilBuffer && generateDepthBuffer) {\r\n return this._createRenderBuffer(width, height, samples, gl.DEPTH_STENCIL, gl.DEPTH24_STENCIL8, gl.DEPTH_STENCIL_ATTACHMENT);\r\n }\r\n if (generateDepthBuffer) {\r\n let depthFormat: GLenum = gl.DEPTH_COMPONENT16;\r\n if (this._webGLVersion > 1) {\r\n depthFormat = gl.DEPTH_COMPONENT32F;\r\n }\r\n\r\n return this._createRenderBuffer(width, height, samples, depthFormat, depthFormat, gl.DEPTH_ATTACHMENT);\r\n }\r\n if (generateStencilBuffer) {\r\n return this._createRenderBuffer(width, height, samples, gl.STENCIL_INDEX8, gl.STENCIL_INDEX8, gl.STENCIL_ATTACHMENT);\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _createRenderBuffer(\r\n width: number,\r\n height: number,\r\n samples: number,\r\n internalFormat: number,\r\n msInternalFormat: number,\r\n attachment: number,\r\n unbindBuffer = true\r\n ): Nullable {\r\n const gl = this._gl;\r\n const renderBuffer = gl.createRenderbuffer();\r\n return this._updateRenderBuffer(renderBuffer, width, height, samples, internalFormat, msInternalFormat, attachment, unbindBuffer);\r\n }\r\n\r\n public _updateRenderBuffer(\r\n renderBuffer: Nullable,\r\n width: number,\r\n height: number,\r\n samples: number,\r\n internalFormat: number,\r\n msInternalFormat: number,\r\n attachment: number,\r\n unbindBuffer = true\r\n ): Nullable {\r\n const gl = this._gl;\r\n\r\n gl.bindRenderbuffer(gl.RENDERBUFFER, renderBuffer);\r\n\r\n if (samples > 1 && gl.renderbufferStorageMultisample) {\r\n gl.renderbufferStorageMultisample(gl.RENDERBUFFER, samples, msInternalFormat, width, height);\r\n } else {\r\n gl.renderbufferStorage(gl.RENDERBUFFER, internalFormat, width, height);\r\n }\r\n\r\n gl.framebufferRenderbuffer(gl.FRAMEBUFFER, attachment, gl.RENDERBUFFER, renderBuffer);\r\n\r\n if (unbindBuffer) {\r\n gl.bindRenderbuffer(gl.RENDERBUFFER, null);\r\n }\r\n\r\n return renderBuffer;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _releaseTexture(texture: InternalTexture): void {\r\n this._deleteTexture(texture._hardwareTexture?.underlyingResource);\r\n\r\n // Unbind channels\r\n this.unbindAllTextures();\r\n\r\n const index = this._internalTexturesCache.indexOf(texture);\r\n if (index !== -1) {\r\n this._internalTexturesCache.splice(index, 1);\r\n }\r\n\r\n // Integrated fixed lod samplers.\r\n if (texture._lodTextureHigh) {\r\n texture._lodTextureHigh.dispose();\r\n }\r\n if (texture._lodTextureMid) {\r\n texture._lodTextureMid.dispose();\r\n }\r\n if (texture._lodTextureLow) {\r\n texture._lodTextureLow.dispose();\r\n }\r\n\r\n // Integrated irradiance map.\r\n if (texture._irradianceTexture) {\r\n texture._irradianceTexture.dispose();\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _releaseRenderTargetWrapper(rtWrapper: RenderTargetWrapper): void {\r\n const index = this._renderTargetWrapperCache.indexOf(rtWrapper);\r\n if (index !== -1) {\r\n this._renderTargetWrapperCache.splice(index, 1);\r\n }\r\n }\r\n\r\n protected _deleteTexture(texture: Nullable): void {\r\n if (texture) {\r\n this._gl.deleteTexture(texture);\r\n }\r\n }\r\n\r\n protected _setProgram(program: WebGLProgram): void {\r\n if (this._currentProgram !== program) {\r\n this._gl.useProgram(program);\r\n this._currentProgram = program;\r\n }\r\n }\r\n\r\n protected _boundUniforms: { [key: number]: WebGLUniformLocation } = {};\r\n\r\n /**\r\n * Binds an effect to the webGL context\r\n * @param effect defines the effect to bind\r\n */\r\n public bindSamplers(effect: Effect): void {\r\n const webGLPipelineContext = effect.getPipelineContext() as WebGLPipelineContext;\r\n this._setProgram(webGLPipelineContext.program!);\r\n const samplers = effect.getSamplers();\r\n for (let index = 0; index < samplers.length; index++) {\r\n const uniform = effect.getUniform(samplers[index]);\r\n\r\n if (uniform) {\r\n this._boundUniforms[index] = uniform;\r\n }\r\n }\r\n this._currentEffect = null;\r\n }\r\n\r\n private _activateCurrentTexture() {\r\n if (this._currentTextureChannel !== this._activeChannel) {\r\n this._gl.activeTexture(this._gl.TEXTURE0 + this._activeChannel);\r\n this._currentTextureChannel = this._activeChannel;\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _bindTextureDirectly(target: number, texture: Nullable, forTextureDataUpdate = false, force = false): boolean {\r\n let wasPreviouslyBound = false;\r\n const isTextureForRendering = texture && texture._associatedChannel > -1;\r\n if (forTextureDataUpdate && isTextureForRendering) {\r\n this._activeChannel = texture!._associatedChannel;\r\n }\r\n\r\n const currentTextureBound = this._boundTexturesCache[this._activeChannel];\r\n\r\n if (currentTextureBound !== texture || force) {\r\n this._activateCurrentTexture();\r\n\r\n if (texture && texture.isMultiview) {\r\n //this._gl.bindTexture(target, texture ? texture._colorTextureArray : null);\r\n console.error(target, texture);\r\n throw \"_bindTextureDirectly called with a multiview texture!\";\r\n } else {\r\n this._gl.bindTexture(target, texture?._hardwareTexture?.underlyingResource ?? null);\r\n }\r\n\r\n this._boundTexturesCache[this._activeChannel] = texture;\r\n\r\n if (texture) {\r\n texture._associatedChannel = this._activeChannel;\r\n }\r\n } else if (forTextureDataUpdate) {\r\n wasPreviouslyBound = true;\r\n this._activateCurrentTexture();\r\n }\r\n\r\n if (isTextureForRendering && !forTextureDataUpdate) {\r\n this._bindSamplerUniformToChannel(texture!._associatedChannel, this._activeChannel);\r\n }\r\n\r\n return wasPreviouslyBound;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _bindTexture(channel: number, texture: Nullable, name: string): void {\r\n if (channel === undefined) {\r\n return;\r\n }\r\n\r\n if (texture) {\r\n texture._associatedChannel = channel;\r\n }\r\n\r\n this._activeChannel = channel;\r\n const target = texture ? this._getTextureTarget(texture) : this._gl.TEXTURE_2D;\r\n this._bindTextureDirectly(target, texture);\r\n }\r\n\r\n /**\r\n * Unbind all textures from the webGL context\r\n */\r\n public unbindAllTextures(): void {\r\n for (let channel = 0; channel < this._maxSimultaneousTextures; channel++) {\r\n this._activeChannel = channel;\r\n this._bindTextureDirectly(this._gl.TEXTURE_2D, null);\r\n this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, null);\r\n if (this.webGLVersion > 1) {\r\n this._bindTextureDirectly(this._gl.TEXTURE_3D, null);\r\n this._bindTextureDirectly(this._gl.TEXTURE_2D_ARRAY, null);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Sets a texture to the according uniform.\r\n * @param channel The texture channel\r\n * @param uniform The uniform to set\r\n * @param texture The texture to apply\r\n * @param name The name of the uniform in the effect\r\n */\r\n public setTexture(channel: number, uniform: Nullable, texture: Nullable, name: string): void {\r\n if (channel === undefined) {\r\n return;\r\n }\r\n\r\n if (uniform) {\r\n this._boundUniforms[channel] = uniform;\r\n }\r\n\r\n this._setTexture(channel, texture);\r\n }\r\n\r\n private _bindSamplerUniformToChannel(sourceSlot: number, destination: number) {\r\n const uniform = this._boundUniforms[sourceSlot];\r\n if (!uniform || uniform._currentState === destination) {\r\n return;\r\n }\r\n this._gl.uniform1i(uniform, destination);\r\n uniform._currentState = destination;\r\n }\r\n\r\n private _getTextureWrapMode(mode: number): number {\r\n switch (mode) {\r\n case Constants.TEXTURE_WRAP_ADDRESSMODE:\r\n return this._gl.REPEAT;\r\n case Constants.TEXTURE_CLAMP_ADDRESSMODE:\r\n return this._gl.CLAMP_TO_EDGE;\r\n case Constants.TEXTURE_MIRROR_ADDRESSMODE:\r\n return this._gl.MIRRORED_REPEAT;\r\n }\r\n return this._gl.REPEAT;\r\n }\r\n\r\n protected _setTexture(channel: number, texture: Nullable, isPartOfTextureArray = false, depthStencilTexture = false, name = \"\"): boolean {\r\n // Not ready?\r\n if (!texture) {\r\n if (this._boundTexturesCache[channel] != null) {\r\n this._activeChannel = channel;\r\n this._bindTextureDirectly(this._gl.TEXTURE_2D, null);\r\n this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, null);\r\n if (this.webGLVersion > 1) {\r\n this._bindTextureDirectly(this._gl.TEXTURE_3D, null);\r\n this._bindTextureDirectly(this._gl.TEXTURE_2D_ARRAY, null);\r\n }\r\n }\r\n return false;\r\n }\r\n\r\n // Video\r\n if ((texture).video) {\r\n this._activeChannel = channel;\r\n const videoInternalTexture = (texture).getInternalTexture();\r\n if (videoInternalTexture) {\r\n videoInternalTexture._associatedChannel = channel;\r\n }\r\n (texture).update();\r\n } else if (texture.delayLoadState === Constants.DELAYLOADSTATE_NOTLOADED) {\r\n // Delay loading\r\n texture.delayLoad();\r\n return false;\r\n }\r\n\r\n let internalTexture: InternalTexture;\r\n if (depthStencilTexture) {\r\n internalTexture = (texture).depthStencilTexture!;\r\n } else if (texture.isReady()) {\r\n internalTexture = texture.getInternalTexture();\r\n } else if (texture.isCube) {\r\n internalTexture = this.emptyCubeTexture;\r\n } else if (texture.is3D) {\r\n internalTexture = this.emptyTexture3D;\r\n } else if (texture.is2DArray) {\r\n internalTexture = this.emptyTexture2DArray;\r\n } else {\r\n internalTexture = this.emptyTexture;\r\n }\r\n\r\n if (!isPartOfTextureArray && internalTexture) {\r\n internalTexture._associatedChannel = channel;\r\n }\r\n\r\n let needToBind = true;\r\n if (this._boundTexturesCache[channel] === internalTexture) {\r\n if (!isPartOfTextureArray) {\r\n this._bindSamplerUniformToChannel(internalTexture._associatedChannel, channel);\r\n }\r\n\r\n needToBind = false;\r\n }\r\n\r\n this._activeChannel = channel;\r\n const target = this._getTextureTarget(internalTexture);\r\n if (needToBind) {\r\n this._bindTextureDirectly(target, internalTexture, isPartOfTextureArray);\r\n }\r\n\r\n if (internalTexture && !internalTexture.isMultiview) {\r\n // CUBIC_MODE and SKYBOX_MODE both require CLAMP_TO_EDGE. All other modes use REPEAT.\r\n if (internalTexture.isCube && internalTexture._cachedCoordinatesMode !== texture.coordinatesMode) {\r\n internalTexture._cachedCoordinatesMode = texture.coordinatesMode;\r\n\r\n const textureWrapMode =\r\n texture.coordinatesMode !== Constants.TEXTURE_CUBIC_MODE && texture.coordinatesMode !== Constants.TEXTURE_SKYBOX_MODE\r\n ? Constants.TEXTURE_WRAP_ADDRESSMODE\r\n : Constants.TEXTURE_CLAMP_ADDRESSMODE;\r\n texture.wrapU = textureWrapMode;\r\n texture.wrapV = textureWrapMode;\r\n }\r\n\r\n if (internalTexture._cachedWrapU !== texture.wrapU) {\r\n internalTexture._cachedWrapU = texture.wrapU;\r\n this._setTextureParameterInteger(target, this._gl.TEXTURE_WRAP_S, this._getTextureWrapMode(texture.wrapU), internalTexture);\r\n }\r\n\r\n if (internalTexture._cachedWrapV !== texture.wrapV) {\r\n internalTexture._cachedWrapV = texture.wrapV;\r\n this._setTextureParameterInteger(target, this._gl.TEXTURE_WRAP_T, this._getTextureWrapMode(texture.wrapV), internalTexture);\r\n }\r\n\r\n if (internalTexture.is3D && internalTexture._cachedWrapR !== texture.wrapR) {\r\n internalTexture._cachedWrapR = texture.wrapR;\r\n this._setTextureParameterInteger(target, this._gl.TEXTURE_WRAP_R, this._getTextureWrapMode(texture.wrapR), internalTexture);\r\n }\r\n\r\n this._setAnisotropicLevel(target, internalTexture, texture.anisotropicFilteringLevel);\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Sets an array of texture to the webGL context\r\n * @param channel defines the channel where the texture array must be set\r\n * @param uniform defines the associated uniform location\r\n * @param textures defines the array of textures to bind\r\n * @param name name of the channel\r\n */\r\n public setTextureArray(channel: number, uniform: Nullable, textures: ThinTexture[], name: string): void {\r\n if (channel === undefined || !uniform) {\r\n return;\r\n }\r\n\r\n if (!this._textureUnits || this._textureUnits.length !== textures.length) {\r\n this._textureUnits = new Int32Array(textures.length);\r\n }\r\n for (let i = 0; i < textures.length; i++) {\r\n const texture = textures[i].getInternalTexture();\r\n\r\n if (texture) {\r\n this._textureUnits[i] = channel + i;\r\n texture._associatedChannel = channel + i;\r\n } else {\r\n this._textureUnits[i] = -1;\r\n }\r\n }\r\n this._gl.uniform1iv(uniform, this._textureUnits);\r\n\r\n for (let index = 0; index < textures.length; index++) {\r\n this._setTexture(this._textureUnits[index], textures[index], true);\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _setAnisotropicLevel(target: number, internalTexture: InternalTexture, anisotropicFilteringLevel: number) {\r\n const anisotropicFilterExtension = this._caps.textureAnisotropicFilterExtension;\r\n if (\r\n internalTexture.samplingMode !== Constants.TEXTURE_LINEAR_LINEAR_MIPNEAREST &&\r\n internalTexture.samplingMode !== Constants.TEXTURE_LINEAR_LINEAR_MIPLINEAR &&\r\n internalTexture.samplingMode !== Constants.TEXTURE_LINEAR_LINEAR\r\n ) {\r\n anisotropicFilteringLevel = 1; // Forcing the anisotropic to 1 because else webgl will force filters to linear\r\n }\r\n\r\n if (anisotropicFilterExtension && internalTexture._cachedAnisotropicFilteringLevel !== anisotropicFilteringLevel) {\r\n this._setTextureParameterFloat(\r\n target,\r\n anisotropicFilterExtension.TEXTURE_MAX_ANISOTROPY_EXT,\r\n Math.min(anisotropicFilteringLevel, this._caps.maxAnisotropy),\r\n internalTexture\r\n );\r\n internalTexture._cachedAnisotropicFilteringLevel = anisotropicFilteringLevel;\r\n }\r\n }\r\n\r\n private _setTextureParameterFloat(target: number, parameter: number, value: number, texture: InternalTexture): void {\r\n this._bindTextureDirectly(target, texture, true, true);\r\n this._gl.texParameterf(target, parameter, value);\r\n }\r\n\r\n private _setTextureParameterInteger(target: number, parameter: number, value: number, texture?: InternalTexture) {\r\n if (texture) {\r\n this._bindTextureDirectly(target, texture, true, true);\r\n }\r\n this._gl.texParameteri(target, parameter, value);\r\n }\r\n\r\n /**\r\n * Unbind all vertex attributes from the webGL context\r\n */\r\n public unbindAllAttributes() {\r\n if (this._mustWipeVertexAttributes) {\r\n this._mustWipeVertexAttributes = false;\r\n\r\n for (let i = 0; i < this._caps.maxVertexAttribs; i++) {\r\n this.disableAttributeByIndex(i);\r\n }\r\n return;\r\n }\r\n\r\n for (let i = 0, ul = this._vertexAttribArraysEnabled.length; i < ul; i++) {\r\n if (i >= this._caps.maxVertexAttribs || !this._vertexAttribArraysEnabled[i]) {\r\n continue;\r\n }\r\n\r\n this.disableAttributeByIndex(i);\r\n }\r\n }\r\n\r\n /**\r\n * Force the engine to release all cached effects. This means that next effect compilation will have to be done completely even if a similar effect was already compiled\r\n */\r\n public releaseEffects() {\r\n for (const name in this._compiledEffects) {\r\n const webGLPipelineContext = this._compiledEffects[name].getPipelineContext() as WebGLPipelineContext;\r\n this._deletePipelineContext(webGLPipelineContext);\r\n }\r\n\r\n this._compiledEffects = {};\r\n }\r\n\r\n /**\r\n * Dispose and release all associated resources\r\n */\r\n public dispose(): void {\r\n this._isDisposed = true;\r\n this.stopRenderLoop();\r\n\r\n // Clear observables\r\n if (this.onBeforeTextureInitObservable) {\r\n this.onBeforeTextureInitObservable.clear();\r\n }\r\n\r\n // Empty texture\r\n if (this._emptyTexture) {\r\n this._releaseTexture(this._emptyTexture);\r\n this._emptyTexture = null;\r\n }\r\n if (this._emptyCubeTexture) {\r\n this._releaseTexture(this._emptyCubeTexture);\r\n this._emptyCubeTexture = null;\r\n }\r\n\r\n if (this._dummyFramebuffer) {\r\n this._gl.deleteFramebuffer(this._dummyFramebuffer);\r\n }\r\n\r\n // Release effects\r\n this.releaseEffects();\r\n this.releaseComputeEffects?.();\r\n\r\n // Unbind\r\n this.unbindAllAttributes();\r\n this._boundUniforms = {};\r\n\r\n // Events\r\n if (IsWindowObjectExist()) {\r\n if (this._renderingCanvas) {\r\n if (!this._doNotHandleContextLost) {\r\n this._renderingCanvas.removeEventListener(\"webglcontextlost\", this._onContextLost);\r\n this._renderingCanvas.removeEventListener(\"webglcontextrestored\", this._onContextRestored);\r\n }\r\n\r\n window.removeEventListener(\"resize\", this._checkForMobile);\r\n }\r\n }\r\n\r\n this._workingCanvas = null;\r\n this._workingContext = null;\r\n this._currentBufferPointers.length = 0;\r\n this._renderingCanvas = null;\r\n this._currentProgram = null;\r\n this._boundRenderFunction = null;\r\n\r\n Effect.ResetCache();\r\n\r\n // Abort active requests\r\n for (const request of this._activeRequests) {\r\n request.abort();\r\n }\r\n\r\n this.onDisposeObservable.notifyObservers(this);\r\n this.onDisposeObservable.clear();\r\n }\r\n\r\n /**\r\n * Attach a new callback raised when context lost event is fired\r\n * @param callback defines the callback to call\r\n */\r\n public attachContextLostEvent(callback: (event: WebGLContextEvent) => void): void {\r\n if (this._renderingCanvas) {\r\n this._renderingCanvas.addEventListener(\"webglcontextlost\", callback, false);\r\n }\r\n }\r\n\r\n /**\r\n * Attach a new callback raised when context restored event is fired\r\n * @param callback defines the callback to call\r\n */\r\n public attachContextRestoredEvent(callback: (event: WebGLContextEvent) => void): void {\r\n if (this._renderingCanvas) {\r\n this._renderingCanvas.addEventListener(\"webglcontextrestored\", callback, false);\r\n }\r\n }\r\n\r\n /**\r\n * Get the current error code of the webGL context\r\n * @returns the error code\r\n * @see https://developer.mozilla.org/en-US/docs/Web/API/WebGLRenderingContext/getError\r\n */\r\n public getError(): number {\r\n return this._gl.getError();\r\n }\r\n\r\n private _canRenderToFloatFramebuffer(): boolean {\r\n if (this._webGLVersion > 1) {\r\n return this._caps.colorBufferFloat;\r\n }\r\n return this._canRenderToFramebuffer(Constants.TEXTURETYPE_FLOAT);\r\n }\r\n\r\n private _canRenderToHalfFloatFramebuffer(): boolean {\r\n if (this._webGLVersion > 1) {\r\n return this._caps.colorBufferFloat;\r\n }\r\n return this._canRenderToFramebuffer(Constants.TEXTURETYPE_HALF_FLOAT);\r\n }\r\n\r\n // Thank you : http://stackoverflow.com/questions/28827511/webgl-ios-render-to-floating-point-texture\r\n private _canRenderToFramebuffer(type: number): boolean {\r\n const gl = this._gl;\r\n\r\n //clear existing errors\r\n // eslint-disable-next-line no-empty\r\n while (gl.getError() !== gl.NO_ERROR) {}\r\n\r\n let successful = true;\r\n\r\n const texture = gl.createTexture();\r\n gl.bindTexture(gl.TEXTURE_2D, texture);\r\n gl.texImage2D(gl.TEXTURE_2D, 0, this._getRGBABufferInternalSizedFormat(type), 1, 1, 0, gl.RGBA, this._getWebGLTextureType(type), null);\r\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST);\r\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST);\r\n\r\n const fb = gl.createFramebuffer();\r\n gl.bindFramebuffer(gl.FRAMEBUFFER, fb);\r\n gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, texture, 0);\r\n const status = gl.checkFramebufferStatus(gl.FRAMEBUFFER);\r\n\r\n successful = successful && status === gl.FRAMEBUFFER_COMPLETE;\r\n successful = successful && gl.getError() === gl.NO_ERROR;\r\n\r\n //try render by clearing frame buffer's color buffer\r\n if (successful) {\r\n gl.clear(gl.COLOR_BUFFER_BIT);\r\n successful = successful && gl.getError() === gl.NO_ERROR;\r\n }\r\n\r\n //try reading from frame to ensure render occurs (just creating the FBO is not sufficient to determine if rendering is supported)\r\n if (successful) {\r\n //in practice it's sufficient to just read from the backbuffer rather than handle potentially issues reading from the texture\r\n gl.bindFramebuffer(gl.FRAMEBUFFER, null);\r\n const readFormat = gl.RGBA;\r\n const readType = gl.UNSIGNED_BYTE;\r\n const buffer = new Uint8Array(4);\r\n gl.readPixels(0, 0, 1, 1, readFormat, readType, buffer);\r\n successful = successful && gl.getError() === gl.NO_ERROR;\r\n }\r\n\r\n //clean up\r\n gl.deleteTexture(texture);\r\n gl.deleteFramebuffer(fb);\r\n gl.bindFramebuffer(gl.FRAMEBUFFER, null);\r\n\r\n //clear accumulated errors\r\n // eslint-disable-next-line no-empty\r\n while (!successful && gl.getError() !== gl.NO_ERROR) {}\r\n\r\n return successful;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getWebGLTextureType(type: number): number {\r\n if (this._webGLVersion === 1) {\r\n switch (type) {\r\n case Constants.TEXTURETYPE_FLOAT:\r\n return this._gl.FLOAT;\r\n case Constants.TEXTURETYPE_HALF_FLOAT:\r\n return this._gl.HALF_FLOAT_OES;\r\n case Constants.TEXTURETYPE_UNSIGNED_BYTE:\r\n return this._gl.UNSIGNED_BYTE;\r\n case Constants.TEXTURETYPE_UNSIGNED_SHORT_4_4_4_4:\r\n return this._gl.UNSIGNED_SHORT_4_4_4_4;\r\n case Constants.TEXTURETYPE_UNSIGNED_SHORT_5_5_5_1:\r\n return this._gl.UNSIGNED_SHORT_5_5_5_1;\r\n case Constants.TEXTURETYPE_UNSIGNED_SHORT_5_6_5:\r\n return this._gl.UNSIGNED_SHORT_5_6_5;\r\n }\r\n return this._gl.UNSIGNED_BYTE;\r\n }\r\n\r\n switch (type) {\r\n case Constants.TEXTURETYPE_BYTE:\r\n return this._gl.BYTE;\r\n case Constants.TEXTURETYPE_UNSIGNED_BYTE:\r\n return this._gl.UNSIGNED_BYTE;\r\n case Constants.TEXTURETYPE_SHORT:\r\n return this._gl.SHORT;\r\n case Constants.TEXTURETYPE_UNSIGNED_SHORT:\r\n return this._gl.UNSIGNED_SHORT;\r\n case Constants.TEXTURETYPE_INT:\r\n return this._gl.INT;\r\n case Constants.TEXTURETYPE_UNSIGNED_INTEGER: // Refers to UNSIGNED_INT\r\n return this._gl.UNSIGNED_INT;\r\n case Constants.TEXTURETYPE_FLOAT:\r\n return this._gl.FLOAT;\r\n case Constants.TEXTURETYPE_HALF_FLOAT:\r\n return this._gl.HALF_FLOAT;\r\n case Constants.TEXTURETYPE_UNSIGNED_SHORT_4_4_4_4:\r\n return this._gl.UNSIGNED_SHORT_4_4_4_4;\r\n case Constants.TEXTURETYPE_UNSIGNED_SHORT_5_5_5_1:\r\n return this._gl.UNSIGNED_SHORT_5_5_5_1;\r\n case Constants.TEXTURETYPE_UNSIGNED_SHORT_5_6_5:\r\n return this._gl.UNSIGNED_SHORT_5_6_5;\r\n case Constants.TEXTURETYPE_UNSIGNED_INT_2_10_10_10_REV:\r\n return this._gl.UNSIGNED_INT_2_10_10_10_REV;\r\n case Constants.TEXTURETYPE_UNSIGNED_INT_24_8:\r\n return this._gl.UNSIGNED_INT_24_8;\r\n case Constants.TEXTURETYPE_UNSIGNED_INT_10F_11F_11F_REV:\r\n return this._gl.UNSIGNED_INT_10F_11F_11F_REV;\r\n case Constants.TEXTURETYPE_UNSIGNED_INT_5_9_9_9_REV:\r\n return this._gl.UNSIGNED_INT_5_9_9_9_REV;\r\n case Constants.TEXTURETYPE_FLOAT_32_UNSIGNED_INT_24_8_REV:\r\n return this._gl.FLOAT_32_UNSIGNED_INT_24_8_REV;\r\n }\r\n\r\n return this._gl.UNSIGNED_BYTE;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getInternalFormat(format: number, useSRGBBuffer = false): number {\r\n let internalFormat: GLenum = useSRGBBuffer ? this._glSRGBExtensionValues.SRGB8_ALPHA8 : this._gl.RGBA;\r\n\r\n switch (format) {\r\n case Constants.TEXTUREFORMAT_ALPHA:\r\n internalFormat = this._gl.ALPHA;\r\n break;\r\n case Constants.TEXTUREFORMAT_LUMINANCE:\r\n internalFormat = this._gl.LUMINANCE;\r\n break;\r\n case Constants.TEXTUREFORMAT_LUMINANCE_ALPHA:\r\n internalFormat = this._gl.LUMINANCE_ALPHA;\r\n break;\r\n case Constants.TEXTUREFORMAT_RED:\r\n internalFormat = this._gl.RED;\r\n break;\r\n case Constants.TEXTUREFORMAT_RG:\r\n internalFormat = this._gl.RG;\r\n break;\r\n case Constants.TEXTUREFORMAT_RGB:\r\n internalFormat = useSRGBBuffer ? this._glSRGBExtensionValues.SRGB : this._gl.RGB;\r\n break;\r\n case Constants.TEXTUREFORMAT_RGBA:\r\n internalFormat = useSRGBBuffer ? this._glSRGBExtensionValues.SRGB8_ALPHA8 : this._gl.RGBA;\r\n break;\r\n }\r\n\r\n if (this._webGLVersion > 1) {\r\n switch (format) {\r\n case Constants.TEXTUREFORMAT_RED_INTEGER:\r\n internalFormat = this._gl.RED_INTEGER;\r\n break;\r\n case Constants.TEXTUREFORMAT_RG_INTEGER:\r\n internalFormat = this._gl.RG_INTEGER;\r\n break;\r\n case Constants.TEXTUREFORMAT_RGB_INTEGER:\r\n internalFormat = this._gl.RGB_INTEGER;\r\n break;\r\n case Constants.TEXTUREFORMAT_RGBA_INTEGER:\r\n internalFormat = this._gl.RGBA_INTEGER;\r\n break;\r\n }\r\n }\r\n\r\n return internalFormat;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getRGBABufferInternalSizedFormat(type: number, format?: number, useSRGBBuffer = false): number {\r\n if (this._webGLVersion === 1) {\r\n if (format !== undefined) {\r\n switch (format) {\r\n case Constants.TEXTUREFORMAT_ALPHA:\r\n return this._gl.ALPHA;\r\n case Constants.TEXTUREFORMAT_LUMINANCE:\r\n return this._gl.LUMINANCE;\r\n case Constants.TEXTUREFORMAT_LUMINANCE_ALPHA:\r\n return this._gl.LUMINANCE_ALPHA;\r\n case Constants.TEXTUREFORMAT_RGB:\r\n return useSRGBBuffer ? this._glSRGBExtensionValues.SRGB : this._gl.RGB;\r\n }\r\n }\r\n return this._gl.RGBA;\r\n }\r\n\r\n switch (type) {\r\n case Constants.TEXTURETYPE_BYTE:\r\n switch (format) {\r\n case Constants.TEXTUREFORMAT_RED:\r\n return this._gl.R8_SNORM;\r\n case Constants.TEXTUREFORMAT_RG:\r\n return this._gl.RG8_SNORM;\r\n case Constants.TEXTUREFORMAT_RGB:\r\n return this._gl.RGB8_SNORM;\r\n case Constants.TEXTUREFORMAT_RED_INTEGER:\r\n return this._gl.R8I;\r\n case Constants.TEXTUREFORMAT_RG_INTEGER:\r\n return this._gl.RG8I;\r\n case Constants.TEXTUREFORMAT_RGB_INTEGER:\r\n return this._gl.RGB8I;\r\n case Constants.TEXTUREFORMAT_RGBA_INTEGER:\r\n return this._gl.RGBA8I;\r\n default:\r\n return this._gl.RGBA8_SNORM;\r\n }\r\n case Constants.TEXTURETYPE_UNSIGNED_BYTE:\r\n switch (format) {\r\n case Constants.TEXTUREFORMAT_RED:\r\n return this._gl.R8;\r\n case Constants.TEXTUREFORMAT_RG:\r\n return this._gl.RG8;\r\n case Constants.TEXTUREFORMAT_RGB:\r\n return useSRGBBuffer ? this._glSRGBExtensionValues.SRGB8 : this._gl.RGB8; // By default. Other possibilities are RGB565, SRGB8.\r\n case Constants.TEXTUREFORMAT_RGBA:\r\n return useSRGBBuffer ? this._glSRGBExtensionValues.SRGB8_ALPHA8 : this._gl.RGBA8; // By default. Other possibilities are RGB5_A1, RGBA4, SRGB8_ALPHA8.\r\n case Constants.TEXTUREFORMAT_RED_INTEGER:\r\n return this._gl.R8UI;\r\n case Constants.TEXTUREFORMAT_RG_INTEGER:\r\n return this._gl.RG8UI;\r\n case Constants.TEXTUREFORMAT_RGB_INTEGER:\r\n return this._gl.RGB8UI;\r\n case Constants.TEXTUREFORMAT_RGBA_INTEGER:\r\n return this._gl.RGBA8UI;\r\n case Constants.TEXTUREFORMAT_ALPHA:\r\n return this._gl.ALPHA;\r\n case Constants.TEXTUREFORMAT_LUMINANCE:\r\n return this._gl.LUMINANCE;\r\n case Constants.TEXTUREFORMAT_LUMINANCE_ALPHA:\r\n return this._gl.LUMINANCE_ALPHA;\r\n default:\r\n return this._gl.RGBA8;\r\n }\r\n case Constants.TEXTURETYPE_SHORT:\r\n switch (format) {\r\n case Constants.TEXTUREFORMAT_RED_INTEGER:\r\n return this._gl.R16I;\r\n case Constants.TEXTUREFORMAT_RG_INTEGER:\r\n return this._gl.RG16I;\r\n case Constants.TEXTUREFORMAT_RGB_INTEGER:\r\n return this._gl.RGB16I;\r\n case Constants.TEXTUREFORMAT_RGBA_INTEGER:\r\n return this._gl.RGBA16I;\r\n default:\r\n return this._gl.RGBA16I;\r\n }\r\n case Constants.TEXTURETYPE_UNSIGNED_SHORT:\r\n switch (format) {\r\n case Constants.TEXTUREFORMAT_RED_INTEGER:\r\n return this._gl.R16UI;\r\n case Constants.TEXTUREFORMAT_RG_INTEGER:\r\n return this._gl.RG16UI;\r\n case Constants.TEXTUREFORMAT_RGB_INTEGER:\r\n return this._gl.RGB16UI;\r\n case Constants.TEXTUREFORMAT_RGBA_INTEGER:\r\n return this._gl.RGBA16UI;\r\n default:\r\n return this._gl.RGBA16UI;\r\n }\r\n case Constants.TEXTURETYPE_INT:\r\n switch (format) {\r\n case Constants.TEXTUREFORMAT_RED_INTEGER:\r\n return this._gl.R32I;\r\n case Constants.TEXTUREFORMAT_RG_INTEGER:\r\n return this._gl.RG32I;\r\n case Constants.TEXTUREFORMAT_RGB_INTEGER:\r\n return this._gl.RGB32I;\r\n case Constants.TEXTUREFORMAT_RGBA_INTEGER:\r\n return this._gl.RGBA32I;\r\n default:\r\n return this._gl.RGBA32I;\r\n }\r\n case Constants.TEXTURETYPE_UNSIGNED_INTEGER: // Refers to UNSIGNED_INT\r\n switch (format) {\r\n case Constants.TEXTUREFORMAT_RED_INTEGER:\r\n return this._gl.R32UI;\r\n case Constants.TEXTUREFORMAT_RG_INTEGER:\r\n return this._gl.RG32UI;\r\n case Constants.TEXTUREFORMAT_RGB_INTEGER:\r\n return this._gl.RGB32UI;\r\n case Constants.TEXTUREFORMAT_RGBA_INTEGER:\r\n return this._gl.RGBA32UI;\r\n default:\r\n return this._gl.RGBA32UI;\r\n }\r\n case Constants.TEXTURETYPE_FLOAT:\r\n switch (format) {\r\n case Constants.TEXTUREFORMAT_RED:\r\n return this._gl.R32F; // By default. Other possibility is R16F.\r\n case Constants.TEXTUREFORMAT_RG:\r\n return this._gl.RG32F; // By default. Other possibility is RG16F.\r\n case Constants.TEXTUREFORMAT_RGB:\r\n return this._gl.RGB32F; // By default. Other possibilities are RGB16F, R11F_G11F_B10F, RGB9_E5.\r\n case Constants.TEXTUREFORMAT_RGBA:\r\n return this._gl.RGBA32F; // By default. Other possibility is RGBA16F.\r\n default:\r\n return this._gl.RGBA32F;\r\n }\r\n case Constants.TEXTURETYPE_HALF_FLOAT:\r\n switch (format) {\r\n case Constants.TEXTUREFORMAT_RED:\r\n return this._gl.R16F;\r\n case Constants.TEXTUREFORMAT_RG:\r\n return this._gl.RG16F;\r\n case Constants.TEXTUREFORMAT_RGB:\r\n return this._gl.RGB16F; // By default. Other possibilities are R11F_G11F_B10F, RGB9_E5.\r\n case Constants.TEXTUREFORMAT_RGBA:\r\n return this._gl.RGBA16F;\r\n default:\r\n return this._gl.RGBA16F;\r\n }\r\n case Constants.TEXTURETYPE_UNSIGNED_SHORT_5_6_5:\r\n return this._gl.RGB565;\r\n case Constants.TEXTURETYPE_UNSIGNED_INT_10F_11F_11F_REV:\r\n return this._gl.R11F_G11F_B10F;\r\n case Constants.TEXTURETYPE_UNSIGNED_INT_5_9_9_9_REV:\r\n return this._gl.RGB9_E5;\r\n case Constants.TEXTURETYPE_UNSIGNED_SHORT_4_4_4_4:\r\n return this._gl.RGBA4;\r\n case Constants.TEXTURETYPE_UNSIGNED_SHORT_5_5_5_1:\r\n return this._gl.RGB5_A1;\r\n case Constants.TEXTURETYPE_UNSIGNED_INT_2_10_10_10_REV:\r\n switch (format) {\r\n case Constants.TEXTUREFORMAT_RGBA:\r\n return this._gl.RGB10_A2; // By default. Other possibility is RGB5_A1.\r\n case Constants.TEXTUREFORMAT_RGBA_INTEGER:\r\n return this._gl.RGB10_A2UI;\r\n default:\r\n return this._gl.RGB10_A2;\r\n }\r\n }\r\n\r\n return useSRGBBuffer ? this._glSRGBExtensionValues.SRGB8_ALPHA8 : this._gl.RGBA8;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getRGBAMultiSampleBufferFormat(type: number, format = Constants.TEXTUREFORMAT_RGBA): number {\r\n switch (type) {\r\n case Constants.TEXTURETYPE_FLOAT:\r\n switch (format) {\r\n case Constants.TEXTUREFORMAT_R:\r\n return this._gl.R32F;\r\n default:\r\n return this._gl.RGBA32F;\r\n }\r\n case Constants.TEXTURETYPE_HALF_FLOAT:\r\n switch (format) {\r\n case Constants.TEXTUREFORMAT_R:\r\n return this._gl.R16F;\r\n default:\r\n return this._gl.RGBA16F;\r\n }\r\n }\r\n\r\n return this._gl.RGBA8;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _loadFile(\r\n url: string,\r\n onSuccess: (data: string | ArrayBuffer, responseURL?: string) => void,\r\n onProgress?: (data: any) => void,\r\n offlineProvider?: IOfflineProvider,\r\n useArrayBuffer?: boolean,\r\n onError?: (request?: IWebRequest, exception?: any) => void\r\n ): IFileRequest {\r\n const request = ThinEngine._FileToolsLoadFile(url, onSuccess, onProgress, offlineProvider, useArrayBuffer, onError);\r\n this._activeRequests.push(request);\r\n request.onCompleteObservable.add((request) => {\r\n this._activeRequests.splice(this._activeRequests.indexOf(request), 1);\r\n });\r\n return request;\r\n }\r\n\r\n /**\r\n * Loads a file from a url\r\n * @param url url to load\r\n * @param onSuccess callback called when the file successfully loads\r\n * @param onProgress callback called while file is loading (if the server supports this mode)\r\n * @param offlineProvider defines the offline provider for caching\r\n * @param useArrayBuffer defines a boolean indicating that date must be returned as ArrayBuffer\r\n * @param onError callback called when the file fails to load\r\n * @returns a file request object\r\n * @internal\r\n */\r\n public static _FileToolsLoadFile(\r\n url: string,\r\n onSuccess: (data: string | ArrayBuffer, responseURL?: string) => void,\r\n onProgress?: (ev: ProgressEvent) => void,\r\n offlineProvider?: IOfflineProvider,\r\n useArrayBuffer?: boolean,\r\n onError?: (request?: WebRequest, exception?: LoadFileError) => void\r\n ): IFileRequest {\r\n throw _WarnImport(\"FileTools\");\r\n }\r\n\r\n /**\r\n * Reads pixels from the current frame buffer. Please note that this function can be slow\r\n * @param x defines the x coordinate of the rectangle where pixels must be read\r\n * @param y defines the y coordinate of the rectangle where pixels must be read\r\n * @param width defines the width of the rectangle where pixels must be read\r\n * @param height defines the height of the rectangle where pixels must be read\r\n * @param hasAlpha defines whether the output should have alpha or not (defaults to true)\r\n * @param flushRenderer true to flush the renderer from the pending commands before reading the pixels\r\n * @returns a ArrayBufferView promise (Uint8Array) containing RGBA colors\r\n */\r\n public readPixels(x: number, y: number, width: number, height: number, hasAlpha = true, flushRenderer = true): Promise {\r\n const numChannels = hasAlpha ? 4 : 3;\r\n const format = hasAlpha ? this._gl.RGBA : this._gl.RGB;\r\n const data = new Uint8Array(height * width * numChannels);\r\n if (flushRenderer) {\r\n this.flushFramebuffer();\r\n }\r\n this._gl.readPixels(x, y, width, height, format, this._gl.UNSIGNED_BYTE, data);\r\n return Promise.resolve(data);\r\n }\r\n\r\n // Statics\r\n\r\n private static _IsSupported: Nullable = null;\r\n private static _HasMajorPerformanceCaveat: Nullable = null;\r\n\r\n /**\r\n * Gets a Promise indicating if the engine can be instantiated (ie. if a webGL context can be found)\r\n */\r\n public static get IsSupportedAsync(): Promise {\r\n return Promise.resolve(this.isSupported());\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if the engine can be instantiated (ie. if a webGL context can be found)\r\n */\r\n public static get IsSupported(): boolean {\r\n return this.isSupported(); // Backward compat\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if the engine can be instantiated (ie. if a webGL context can be found)\r\n * @returns true if the engine can be created\r\n * @ignorenaming\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public static isSupported(): boolean {\r\n if (this._HasMajorPerformanceCaveat !== null) {\r\n return !this._HasMajorPerformanceCaveat; // We know it is performant so WebGL is supported\r\n }\r\n\r\n if (this._IsSupported === null) {\r\n try {\r\n const tempcanvas = this._CreateCanvas(1, 1);\r\n const gl = tempcanvas.getContext(\"webgl\") || (tempcanvas as any).getContext(\"experimental-webgl\");\r\n\r\n this._IsSupported = gl != null && !!window.WebGLRenderingContext;\r\n } catch (e) {\r\n this._IsSupported = false;\r\n }\r\n }\r\n\r\n return this._IsSupported;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if the engine can be instantiated on a performant device (ie. if a webGL context can be found and it does not use a slow implementation)\r\n */\r\n public static get HasMajorPerformanceCaveat(): boolean {\r\n if (this._HasMajorPerformanceCaveat === null) {\r\n try {\r\n const tempcanvas = this._CreateCanvas(1, 1);\r\n const gl =\r\n tempcanvas.getContext(\"webgl\", { failIfMajorPerformanceCaveat: true }) ||\r\n (tempcanvas as any).getContext(\"experimental-webgl\", { failIfMajorPerformanceCaveat: true });\r\n\r\n this._HasMajorPerformanceCaveat = !gl;\r\n } catch (e) {\r\n this._HasMajorPerformanceCaveat = false;\r\n }\r\n }\r\n\r\n return this._HasMajorPerformanceCaveat;\r\n }\r\n\r\n /**\r\n * Find the next highest power of two.\r\n * @param x Number to start search from.\r\n * @returns Next highest power of two.\r\n */\r\n public static CeilingPOT(x: number): number {\r\n x--;\r\n x |= x >> 1;\r\n x |= x >> 2;\r\n x |= x >> 4;\r\n x |= x >> 8;\r\n x |= x >> 16;\r\n x++;\r\n return x;\r\n }\r\n\r\n /**\r\n * Find the next lowest power of two.\r\n * @param x Number to start search from.\r\n * @returns Next lowest power of two.\r\n */\r\n public static FloorPOT(x: number): number {\r\n x = x | (x >> 1);\r\n x = x | (x >> 2);\r\n x = x | (x >> 4);\r\n x = x | (x >> 8);\r\n x = x | (x >> 16);\r\n return x - (x >> 1);\r\n }\r\n\r\n /**\r\n * Find the nearest power of two.\r\n * @param x Number to start search from.\r\n * @returns Next nearest power of two.\r\n */\r\n public static NearestPOT(x: number): number {\r\n const c = ThinEngine.CeilingPOT(x);\r\n const f = ThinEngine.FloorPOT(x);\r\n return c - x > x - f ? f : c;\r\n }\r\n\r\n /**\r\n * Get the closest exponent of two\r\n * @param value defines the value to approximate\r\n * @param max defines the maximum value to return\r\n * @param mode defines how to define the closest value\r\n * @returns closest exponent of two of the given value\r\n */\r\n public static GetExponentOfTwo(value: number, max: number, mode = Constants.SCALEMODE_NEAREST): number {\r\n let pot;\r\n\r\n switch (mode) {\r\n case Constants.SCALEMODE_FLOOR:\r\n pot = ThinEngine.FloorPOT(value);\r\n break;\r\n case Constants.SCALEMODE_NEAREST:\r\n pot = ThinEngine.NearestPOT(value);\r\n break;\r\n case Constants.SCALEMODE_CEILING:\r\n default:\r\n pot = ThinEngine.CeilingPOT(value);\r\n break;\r\n }\r\n\r\n return Math.min(pot, max);\r\n }\r\n\r\n /**\r\n * Queue a new function into the requested animation frame pool (ie. this function will be executed by the browser (or the javascript engine) for the next frame)\r\n * @param func - the function to be called\r\n * @param requester - the object that will request the next frame. Falls back to window.\r\n * @returns frame number\r\n */\r\n public static QueueNewFrame(func: () => void, requester?: any): number {\r\n // Note that there is kind of a typing issue here, as `setTimeout` might return something else than a number (NodeJs returns a NodeJS.Timeout object).\r\n // Also if the global `requestAnimationFrame`'s returnType is number, `requester.requestPostAnimationFrame` and `requester.requestAnimationFrame` types\r\n // are `any`.\r\n\r\n if (!IsWindowObjectExist()) {\r\n if (typeof requestAnimationFrame === \"function\") {\r\n return requestAnimationFrame(func);\r\n }\r\n } else {\r\n const { requestAnimationFrame } = requester || window;\r\n if (typeof requestAnimationFrame === \"function\") {\r\n return requestAnimationFrame(func);\r\n }\r\n }\r\n\r\n // fallback to the global `setTimeout`.\r\n // In most cases (aka in the browser), `window` is the global object, so instead of calling `window.setTimeout` we could call the global `setTimeout`.\r\n return setTimeout(func, 16) as unknown as number;\r\n }\r\n\r\n /**\r\n * Gets host document\r\n * @returns the host document object\r\n */\r\n public getHostDocument(): Nullable {\r\n if (this._renderingCanvas && this._renderingCanvas.ownerDocument) {\r\n return this._renderingCanvas.ownerDocument;\r\n }\r\n\r\n return IsDocumentAvailable() ? document : null;\r\n }\r\n}\r\n","import { IsWindowObjectExist } from \"./domManagement\";\r\n\r\n/**\r\n * Class used to provide helper for timing\r\n */\r\nexport class TimingTools {\r\n /**\r\n * Polyfill for setImmediate\r\n * @param action defines the action to execute after the current execution block\r\n */\r\n public static SetImmediate(action: () => void) {\r\n if (IsWindowObjectExist() && window.setImmediate) {\r\n window.setImmediate(action);\r\n } else {\r\n setTimeout(action, 1);\r\n }\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport { WebRequest } from \"./webRequest\";\r\nimport { IsWindowObjectExist } from \"./domManagement\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { IOfflineProvider } from \"../Offline/IOfflineProvider\";\r\nimport type { IFileRequest } from \"./fileRequest\";\r\nimport { Observable } from \"./observable\";\r\nimport { FilesInputStore } from \"./filesInputStore\";\r\nimport { RetryStrategy } from \"./retryStrategy\";\r\nimport { BaseError, ErrorCodes, RuntimeError } from \"./error\";\r\nimport { DecodeBase64ToBinary, DecodeBase64ToString, EncodeArrayBufferToBase64 } from \"./stringTools\";\r\nimport { ShaderProcessor } from \"../Engines/Processors/shaderProcessor\";\r\nimport { ThinEngine } from \"../Engines/thinEngine\";\r\nimport { EngineStore } from \"../Engines/engineStore\";\r\nimport { Logger } from \"./logger\";\r\nimport { TimingTools } from \"./timingTools\";\r\n\r\nconst Base64DataUrlRegEx = new RegExp(/^data:([^,]+\\/[^,]+)?;base64,/i);\r\n\r\n/** @ignore */\r\nexport class LoadFileError extends RuntimeError {\r\n public request?: WebRequest;\r\n public file?: File;\r\n\r\n /**\r\n * Creates a new LoadFileError\r\n * @param message defines the message of the error\r\n * @param object defines the optional web request\r\n */\r\n constructor(message: string, object?: WebRequest | File) {\r\n super(message, ErrorCodes.LoadFileError);\r\n\r\n this.name = \"LoadFileError\";\r\n BaseError._setPrototypeOf(this, LoadFileError.prototype);\r\n\r\n if (object instanceof WebRequest) {\r\n this.request = object;\r\n } else {\r\n this.file = object;\r\n }\r\n }\r\n}\r\n\r\n/** @ignore */\r\nexport class RequestFileError extends RuntimeError {\r\n /**\r\n * Creates a new LoadFileError\r\n * @param message defines the message of the error\r\n * @param request defines the optional web request\r\n */\r\n constructor(message: string, public request: WebRequest) {\r\n super(message, ErrorCodes.RequestFileError);\r\n this.name = \"RequestFileError\";\r\n BaseError._setPrototypeOf(this, RequestFileError.prototype);\r\n }\r\n}\r\n\r\n/** @ignore */\r\nexport class ReadFileError extends RuntimeError {\r\n /**\r\n * Creates a new ReadFileError\r\n * @param message defines the message of the error\r\n * @param file defines the optional file\r\n */\r\n constructor(message: string, public file: File) {\r\n super(message, ErrorCodes.ReadFileError);\r\n this.name = \"ReadFileError\";\r\n BaseError._setPrototypeOf(this, ReadFileError.prototype);\r\n }\r\n}\r\n/**\r\n * @internal\r\n */\r\nexport const FileToolsOptions: {\r\n DefaultRetryStrategy: (url: string, request: WebRequest, retryIndex: number) => number;\r\n BaseUrl: string;\r\n CorsBehavior: string | ((url: string | string[]) => string);\r\n PreprocessUrl: (url: string) => string;\r\n} = {\r\n /**\r\n * Gets or sets the retry strategy to apply when an error happens while loading an asset.\r\n * When defining this function, return the wait time before trying again or return -1 to\r\n * stop retrying and error out.\r\n */\r\n DefaultRetryStrategy: RetryStrategy.ExponentialBackoff(),\r\n\r\n /**\r\n * Gets or sets the base URL to use to load assets\r\n */\r\n BaseUrl: \"\",\r\n\r\n /**\r\n * Default behaviour for cors in the application.\r\n * It can be a string if the expected behavior is identical in the entire app.\r\n * Or a callback to be able to set it per url or on a group of them (in case of Video source for instance)\r\n */\r\n CorsBehavior: \"anonymous\",\r\n\r\n /**\r\n * Gets or sets a function used to pre-process url before using them to load assets\r\n * @param url\r\n */\r\n PreprocessUrl: (url: string) => {\r\n return url;\r\n },\r\n};\r\n\r\n/**\r\n * Removes unwanted characters from an url\r\n * @param url defines the url to clean\r\n * @returns the cleaned url\r\n */\r\nconst _CleanUrl = (url: string): string => {\r\n url = url.replace(/#/gm, \"%23\");\r\n return url;\r\n};\r\n\r\n/**\r\n * Sets the cors behavior on a dom element. This will add the required Tools.CorsBehavior to the element.\r\n * @param url define the url we are trying\r\n * @param element define the dom element where to configure the cors policy\r\n * @internal\r\n */\r\nexport const SetCorsBehavior = (url: string | string[], element: { crossOrigin: string | null }): void => {\r\n if (url && url.indexOf(\"data:\") === 0) {\r\n return;\r\n }\r\n\r\n if (FileToolsOptions.CorsBehavior) {\r\n if (typeof FileToolsOptions.CorsBehavior === \"string\" || FileToolsOptions.CorsBehavior instanceof String) {\r\n element.crossOrigin = FileToolsOptions.CorsBehavior;\r\n } else {\r\n const result = FileToolsOptions.CorsBehavior(url);\r\n if (result) {\r\n element.crossOrigin = result;\r\n }\r\n }\r\n }\r\n};\r\n\r\n/**\r\n * Loads an image as an HTMLImageElement.\r\n * @param input url string, ArrayBuffer, or Blob to load\r\n * @param onLoad callback called when the image successfully loads\r\n * @param onError callback called when the image fails to load\r\n * @param offlineProvider offline provider for caching\r\n * @param mimeType optional mime type\r\n * @param imageBitmapOptions\r\n * @returns the HTMLImageElement of the loaded image\r\n * @internal\r\n */\r\nexport const LoadImage = (\r\n input: string | ArrayBuffer | ArrayBufferView | Blob,\r\n onLoad: (img: HTMLImageElement | ImageBitmap) => void,\r\n onError: (message?: string, exception?: any) => void,\r\n offlineProvider: Nullable,\r\n mimeType: string = \"\",\r\n imageBitmapOptions?: ImageBitmapOptions\r\n): Nullable => {\r\n let url: string;\r\n let usingObjectURL = false;\r\n\r\n if (input instanceof ArrayBuffer || ArrayBuffer.isView(input)) {\r\n if (typeof Blob !== \"undefined\" && typeof URL !== \"undefined\") {\r\n url = URL.createObjectURL(new Blob([input], { type: mimeType }));\r\n usingObjectURL = true;\r\n } else {\r\n url = `data:${mimeType};base64,` + EncodeArrayBufferToBase64(input);\r\n }\r\n } else if (input instanceof Blob) {\r\n url = URL.createObjectURL(input);\r\n usingObjectURL = true;\r\n } else {\r\n url = _CleanUrl(input);\r\n url = FileToolsOptions.PreprocessUrl(input);\r\n }\r\n\r\n const engine = EngineStore.LastCreatedEngine;\r\n\r\n const onErrorHandler = (exception: any) => {\r\n if (onError) {\r\n const inputText = url || input.toString();\r\n onError(`Error while trying to load image: ${inputText.indexOf(\"http\") === 0 || inputText.length <= 128 ? inputText : inputText.slice(0, 128) + \"...\"}`, exception);\r\n }\r\n };\r\n\r\n if (typeof Image === \"undefined\" || (engine?._features.forceBitmapOverHTMLImageElement ?? false)) {\r\n LoadFile(\r\n url,\r\n (data) => {\r\n engine!\r\n .createImageBitmap(new Blob([data], { type: mimeType }), { premultiplyAlpha: \"none\", ...imageBitmapOptions })\r\n .then((imgBmp) => {\r\n onLoad(imgBmp);\r\n if (usingObjectURL) {\r\n URL.revokeObjectURL(url);\r\n }\r\n })\r\n .catch((reason) => {\r\n if (onError) {\r\n onError(\"Error while trying to load image: \" + input, reason);\r\n }\r\n });\r\n },\r\n undefined,\r\n offlineProvider || undefined,\r\n true,\r\n (request, exception) => {\r\n onErrorHandler(exception);\r\n }\r\n );\r\n\r\n return null;\r\n }\r\n\r\n const img = new Image();\r\n SetCorsBehavior(url, img);\r\n\r\n const handlersList: { target: any; name: string; handler: any }[] = [];\r\n\r\n const loadHandlersList = () => {\r\n handlersList.forEach((handler) => {\r\n handler.target.addEventListener(handler.name, handler.handler);\r\n });\r\n };\r\n\r\n const unloadHandlersList = () => {\r\n handlersList.forEach((handler) => {\r\n handler.target.removeEventListener(handler.name, handler.handler);\r\n });\r\n handlersList.length = 0;\r\n };\r\n\r\n const loadHandler = () => {\r\n unloadHandlersList();\r\n\r\n onLoad(img);\r\n\r\n // Must revoke the URL after calling onLoad to avoid security exceptions in\r\n // certain scenarios (e.g. when hosted in vscode).\r\n if (usingObjectURL && img.src) {\r\n URL.revokeObjectURL(img.src);\r\n }\r\n };\r\n\r\n const errorHandler = (err: any) => {\r\n unloadHandlersList();\r\n\r\n onErrorHandler(err);\r\n\r\n if (usingObjectURL && img.src) {\r\n URL.revokeObjectURL(img.src);\r\n }\r\n };\r\n\r\n const cspHandler = (err: any) => {\r\n if (err.blockedURI !== img.src) {\r\n return;\r\n }\r\n\r\n unloadHandlersList();\r\n const cspException = new Error(`CSP violation of policy ${err.effectiveDirective} ${err.blockedURI}. Current policy is ${err.originalPolicy}`);\r\n\r\n EngineStore.UseFallbackTexture = false;\r\n onErrorHandler(cspException);\r\n if (usingObjectURL && img.src) {\r\n URL.revokeObjectURL(img.src);\r\n }\r\n img.src = \"\";\r\n };\r\n\r\n handlersList.push({ target: img, name: \"load\", handler: loadHandler });\r\n handlersList.push({ target: img, name: \"error\", handler: errorHandler });\r\n handlersList.push({ target: document, name: \"securitypolicyviolation\", handler: cspHandler });\r\n\r\n loadHandlersList();\r\n\r\n const fromBlob = url.substring(0, 5) === \"blob:\";\r\n const fromData = url.substring(0, 5) === \"data:\";\r\n const noOfflineSupport = () => {\r\n if (fromBlob || fromData) {\r\n img.src = url;\r\n } else {\r\n LoadFile(\r\n url,\r\n (data, _, contentType) => {\r\n const type = !mimeType && contentType ? contentType : mimeType;\r\n const blob = new Blob([data], { type });\r\n const url = URL.createObjectURL(blob);\r\n usingObjectURL = true;\r\n img.src = url;\r\n },\r\n undefined,\r\n offlineProvider || undefined,\r\n true,\r\n (request, exception) => {\r\n onErrorHandler(exception);\r\n }\r\n );\r\n }\r\n };\r\n\r\n const loadFromOfflineSupport = () => {\r\n if (offlineProvider) {\r\n offlineProvider.loadImage(url, img);\r\n }\r\n };\r\n\r\n if (!fromBlob && !fromData && offlineProvider && offlineProvider.enableTexturesOffline) {\r\n offlineProvider.open(loadFromOfflineSupport, noOfflineSupport);\r\n } else {\r\n if (url.indexOf(\"file:\") !== -1) {\r\n const textureName = decodeURIComponent(url.substring(5).toLowerCase());\r\n if (FilesInputStore.FilesToLoad[textureName] && typeof URL !== \"undefined\") {\r\n try {\r\n let blobURL;\r\n try {\r\n blobURL = URL.createObjectURL(FilesInputStore.FilesToLoad[textureName]);\r\n } catch (ex) {\r\n // Chrome doesn't support oneTimeOnly parameter\r\n blobURL = URL.createObjectURL(FilesInputStore.FilesToLoad[textureName]);\r\n }\r\n img.src = blobURL;\r\n usingObjectURL = true;\r\n } catch (e) {\r\n img.src = \"\";\r\n }\r\n return img;\r\n }\r\n }\r\n\r\n noOfflineSupport();\r\n }\r\n\r\n return img;\r\n};\r\n\r\n/**\r\n * Reads a file from a File object\r\n * @param file defines the file to load\r\n * @param onSuccess defines the callback to call when data is loaded\r\n * @param onProgress defines the callback to call during loading process\r\n * @param useArrayBuffer defines a boolean indicating that data must be returned as an ArrayBuffer\r\n * @param onError defines the callback to call when an error occurs\r\n * @returns a file request object\r\n * @internal\r\n */\r\nexport const ReadFile = (\r\n file: File,\r\n onSuccess: (data: any) => void,\r\n onProgress?: (ev: ProgressEvent) => any,\r\n useArrayBuffer?: boolean,\r\n onError?: (error: ReadFileError) => void\r\n): IFileRequest => {\r\n const reader = new FileReader();\r\n const fileRequest: IFileRequest = {\r\n onCompleteObservable: new Observable(),\r\n abort: () => reader.abort(),\r\n };\r\n\r\n reader.onloadend = () => fileRequest.onCompleteObservable.notifyObservers(fileRequest);\r\n if (onError) {\r\n reader.onerror = () => {\r\n onError(new ReadFileError(`Unable to read ${file.name}`, file));\r\n };\r\n }\r\n reader.onload = (e) => {\r\n //target doesn't have result from ts 1.3\r\n onSuccess((e.target)[\"result\"]);\r\n };\r\n if (onProgress) {\r\n reader.onprogress = onProgress;\r\n }\r\n if (!useArrayBuffer) {\r\n // Asynchronous read\r\n reader.readAsText(file);\r\n } else {\r\n reader.readAsArrayBuffer(file);\r\n }\r\n\r\n return fileRequest;\r\n};\r\n\r\n/**\r\n * Loads a file from a url, a data url, or a file url\r\n * @param fileOrUrl file, url, data url, or file url to load\r\n * @param onSuccess callback called when the file successfully loads\r\n * @param onProgress callback called while file is loading (if the server supports this mode)\r\n * @param offlineProvider defines the offline provider for caching\r\n * @param useArrayBuffer defines a boolean indicating that date must be returned as ArrayBuffer\r\n * @param onError callback called when the file fails to load\r\n * @param onOpened\r\n * @returns a file request object\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport const LoadFile = (\r\n fileOrUrl: File | string,\r\n onSuccess: (data: string | ArrayBuffer, responseURL?: string, contentType?: Nullable) => void,\r\n onProgress?: (ev: ProgressEvent) => void,\r\n offlineProvider?: IOfflineProvider,\r\n useArrayBuffer?: boolean,\r\n onError?: (request?: WebRequest, exception?: LoadFileError) => void,\r\n onOpened?: (request: WebRequest) => void\r\n): IFileRequest => {\r\n if ((fileOrUrl as File).name) {\r\n return ReadFile(\r\n fileOrUrl as File,\r\n onSuccess,\r\n onProgress,\r\n useArrayBuffer,\r\n onError\r\n ? (error: ReadFileError) => {\r\n onError(undefined, error);\r\n }\r\n : undefined\r\n );\r\n }\r\n\r\n const url = fileOrUrl as string;\r\n\r\n // If file and file input are set\r\n if (url.indexOf(\"file:\") !== -1) {\r\n let fileName = decodeURIComponent(url.substring(5).toLowerCase());\r\n if (fileName.indexOf(\"./\") === 0) {\r\n fileName = fileName.substring(2);\r\n }\r\n const file = FilesInputStore.FilesToLoad[fileName];\r\n if (file) {\r\n return ReadFile(file, onSuccess, onProgress, useArrayBuffer, onError ? (error) => onError(undefined, new LoadFileError(error.message, error.file)) : undefined);\r\n }\r\n }\r\n\r\n // For a Base64 Data URL\r\n const { match, type } = TestBase64DataUrl(url);\r\n if (match) {\r\n const fileRequest: IFileRequest = {\r\n onCompleteObservable: new Observable(),\r\n abort: () => () => {},\r\n };\r\n\r\n try {\r\n const data = useArrayBuffer ? DecodeBase64UrlToBinary(url) : DecodeBase64UrlToString(url);\r\n onSuccess(data, undefined, type);\r\n } catch (error) {\r\n if (onError) {\r\n onError(undefined, error);\r\n } else {\r\n Logger.Error(error.message || \"Failed to parse the Data URL\");\r\n }\r\n }\r\n\r\n TimingTools.SetImmediate(() => {\r\n fileRequest.onCompleteObservable.notifyObservers(fileRequest);\r\n });\r\n\r\n return fileRequest;\r\n }\r\n\r\n return RequestFile(\r\n url,\r\n (data, request) => {\r\n onSuccess(data, request?.responseURL, request?.getResponseHeader(\"content-type\"));\r\n },\r\n onProgress,\r\n offlineProvider,\r\n useArrayBuffer,\r\n onError\r\n ? (error) => {\r\n onError(error.request, new LoadFileError(error.message, error.request));\r\n }\r\n : undefined,\r\n onOpened\r\n );\r\n};\r\n\r\n/**\r\n * Loads a file from a url\r\n * @param url url to load\r\n * @param onSuccess callback called when the file successfully loads\r\n * @param onProgress callback called while file is loading (if the server supports this mode)\r\n * @param offlineProvider defines the offline provider for caching\r\n * @param useArrayBuffer defines a boolean indicating that date must be returned as ArrayBuffer\r\n * @param onError callback called when the file fails to load\r\n * @param onOpened callback called when the web request is opened\r\n * @returns a file request object\r\n * @internal\r\n */\r\nexport const RequestFile = (\r\n url: string,\r\n onSuccess?: (data: string | ArrayBuffer, request?: WebRequest) => void,\r\n onProgress?: (event: ProgressEvent) => void,\r\n offlineProvider?: IOfflineProvider,\r\n useArrayBuffer?: boolean,\r\n onError?: (error: RequestFileError) => void,\r\n onOpened?: (request: WebRequest) => void\r\n): IFileRequest => {\r\n url = _CleanUrl(url);\r\n url = FileToolsOptions.PreprocessUrl(url);\r\n\r\n const loadUrl = FileToolsOptions.BaseUrl + url;\r\n\r\n let aborted = false;\r\n const fileRequest: IFileRequest = {\r\n onCompleteObservable: new Observable(),\r\n abort: () => (aborted = true),\r\n };\r\n\r\n const requestFile = () => {\r\n let request: Nullable = new WebRequest();\r\n let retryHandle: Nullable> = null;\r\n let onReadyStateChange: Nullable<() => void>;\r\n\r\n const unbindEvents = () => {\r\n if (!request) {\r\n return;\r\n }\r\n\r\n if (onProgress) {\r\n request.removeEventListener(\"progress\", onProgress);\r\n }\r\n if (onReadyStateChange) {\r\n request.removeEventListener(\"readystatechange\", onReadyStateChange);\r\n }\r\n request.removeEventListener(\"loadend\", onLoadEnd!);\r\n };\r\n\r\n let onLoadEnd: Nullable<() => void> = () => {\r\n unbindEvents();\r\n\r\n fileRequest.onCompleteObservable.notifyObservers(fileRequest);\r\n fileRequest.onCompleteObservable.clear();\r\n\r\n onProgress = undefined;\r\n onReadyStateChange = null;\r\n onLoadEnd = null;\r\n onError = undefined;\r\n onOpened = undefined;\r\n onSuccess = undefined;\r\n };\r\n\r\n fileRequest.abort = () => {\r\n aborted = true;\r\n\r\n if (onLoadEnd) {\r\n onLoadEnd();\r\n }\r\n\r\n if (request && request.readyState !== (XMLHttpRequest.DONE || 4)) {\r\n request.abort();\r\n }\r\n\r\n if (retryHandle !== null) {\r\n clearTimeout(retryHandle);\r\n retryHandle = null;\r\n }\r\n\r\n request = null;\r\n };\r\n\r\n const handleError = (error: any) => {\r\n const message = error.message || \"Unknown error\";\r\n if (onError && request) {\r\n onError(new RequestFileError(message, request));\r\n } else {\r\n Logger.Error(message);\r\n }\r\n };\r\n\r\n const retryLoop = (retryIndex: number) => {\r\n if (!request) {\r\n return;\r\n }\r\n request.open(\"GET\", loadUrl);\r\n\r\n if (onOpened) {\r\n try {\r\n onOpened(request);\r\n } catch (e) {\r\n handleError(e);\r\n return;\r\n }\r\n }\r\n\r\n if (useArrayBuffer) {\r\n request.responseType = \"arraybuffer\";\r\n }\r\n\r\n if (onProgress) {\r\n request.addEventListener(\"progress\", onProgress);\r\n }\r\n\r\n if (onLoadEnd) {\r\n request.addEventListener(\"loadend\", onLoadEnd);\r\n }\r\n\r\n onReadyStateChange = () => {\r\n if (aborted || !request) {\r\n return;\r\n }\r\n\r\n // In case of undefined state in some browsers.\r\n if (request.readyState === (XMLHttpRequest.DONE || 4)) {\r\n // Some browsers have issues where onreadystatechange can be called multiple times with the same value.\r\n if (onReadyStateChange) {\r\n request.removeEventListener(\"readystatechange\", onReadyStateChange);\r\n }\r\n\r\n if ((request.status >= 200 && request.status < 300) || (request.status === 0 && (!IsWindowObjectExist() || IsFileURL()))) {\r\n try {\r\n if (onSuccess) {\r\n onSuccess(useArrayBuffer ? request.response : request.responseText, request);\r\n }\r\n } catch (e) {\r\n handleError(e);\r\n }\r\n return;\r\n }\r\n\r\n const retryStrategy = FileToolsOptions.DefaultRetryStrategy;\r\n if (retryStrategy) {\r\n const waitTime = retryStrategy(loadUrl, request, retryIndex);\r\n if (waitTime !== -1) {\r\n // Prevent the request from completing for retry.\r\n unbindEvents();\r\n\r\n request = new WebRequest();\r\n retryHandle = setTimeout(() => retryLoop(retryIndex + 1), waitTime);\r\n return;\r\n }\r\n }\r\n\r\n const error = new RequestFileError(\"Error status: \" + request.status + \" \" + request.statusText + \" - Unable to load \" + loadUrl, request);\r\n if (onError) {\r\n onError(error);\r\n }\r\n }\r\n };\r\n\r\n request.addEventListener(\"readystatechange\", onReadyStateChange);\r\n\r\n request.send();\r\n };\r\n\r\n retryLoop(0);\r\n };\r\n\r\n // Caching all files\r\n if (offlineProvider && offlineProvider.enableSceneOffline) {\r\n const noOfflineSupport = (request?: any) => {\r\n if (request && request.status > 400) {\r\n if (onError) {\r\n onError(request);\r\n }\r\n } else {\r\n requestFile();\r\n }\r\n };\r\n\r\n const loadFromOfflineSupport = () => {\r\n // TODO: database needs to support aborting and should return a IFileRequest\r\n\r\n if (offlineProvider) {\r\n offlineProvider.loadFile(\r\n FileToolsOptions.BaseUrl + url,\r\n (data) => {\r\n if (!aborted && onSuccess) {\r\n onSuccess(data);\r\n }\r\n\r\n fileRequest.onCompleteObservable.notifyObservers(fileRequest);\r\n },\r\n onProgress\r\n ? (event) => {\r\n if (!aborted && onProgress) {\r\n onProgress(event);\r\n }\r\n }\r\n : undefined,\r\n noOfflineSupport,\r\n useArrayBuffer\r\n );\r\n }\r\n };\r\n\r\n offlineProvider.open(loadFromOfflineSupport, noOfflineSupport);\r\n } else {\r\n requestFile();\r\n }\r\n\r\n return fileRequest;\r\n};\r\n\r\n/**\r\n * Checks if the loaded document was accessed via `file:`-Protocol.\r\n * @returns boolean\r\n * @internal\r\n */\r\nexport const IsFileURL = (): boolean => {\r\n return typeof location !== \"undefined\" && location.protocol === \"file:\";\r\n};\r\n\r\n/**\r\n * Test if the given uri is a valid base64 data url\r\n * @param uri The uri to test\r\n * @returns True if the uri is a base64 data url or false otherwise\r\n * @internal\r\n */\r\nexport const IsBase64DataUrl = (uri: string): boolean => {\r\n return Base64DataUrlRegEx.test(uri);\r\n};\r\n\r\nexport const TestBase64DataUrl = (uri: string): { match: boolean; type: string } => {\r\n const results = Base64DataUrlRegEx.exec(uri);\r\n if (results === null || results.length === 0) {\r\n return { match: false, type: \"\" };\r\n } else {\r\n const type = results[0].replace(\"data:\", \"\").replace(\"base64,\", \"\");\r\n return { match: true, type };\r\n }\r\n};\r\n\r\n/**\r\n * Decode the given base64 uri.\r\n * @param uri The uri to decode\r\n * @returns The decoded base64 data.\r\n * @internal\r\n */\r\nexport function DecodeBase64UrlToBinary(uri: string): ArrayBuffer {\r\n return DecodeBase64ToBinary(uri.split(\",\")[1]);\r\n}\r\n\r\n/**\r\n * Decode the given base64 uri into a UTF-8 encoded string.\r\n * @param uri The uri to decode\r\n * @returns The decoded base64 data.\r\n * @internal\r\n */\r\nexport const DecodeBase64UrlToString = (uri: string): string => {\r\n return DecodeBase64ToString(uri.split(\",\")[1]);\r\n};\r\n\r\n/**\r\n * This will be executed automatically for UMD and es5.\r\n * If esm dev wants the side effects to execute they will have to run it manually\r\n * Once we build native modules those need to be exported.\r\n * @internal\r\n */\r\nconst initSideEffects = () => {\r\n ThinEngine._FileToolsLoadImage = LoadImage;\r\n ThinEngine._FileToolsLoadFile = LoadFile;\r\n ShaderProcessor._FileToolsLoadFile = LoadFile;\r\n};\r\n\r\ninitSideEffects();\r\n\r\n// deprecated\r\n\r\n/**\r\n * FileTools defined as any.\r\n * This should not be imported or used in future releases or in any module in the framework\r\n * @internal\r\n * @deprecated import the needed function from fileTools.ts\r\n */\r\nexport let FileTools: {\r\n DecodeBase64UrlToBinary: (uri: string) => ArrayBuffer;\r\n DecodeBase64UrlToString: (uri: string) => string;\r\n DefaultRetryStrategy: any;\r\n BaseUrl: any;\r\n CorsBehavior: any;\r\n PreprocessUrl: any;\r\n IsBase64DataUrl: (uri: string) => boolean;\r\n IsFileURL: () => boolean;\r\n LoadFile: (\r\n fileOrUrl: string | File,\r\n onSuccess: (data: string | ArrayBuffer, responseURL?: string | undefined) => void,\r\n onProgress?: ((ev: ProgressEvent) => void) | undefined,\r\n offlineProvider?: IOfflineProvider | undefined,\r\n useArrayBuffer?: boolean | undefined,\r\n onError?: ((request?: WebRequest | undefined, exception?: LoadFileError | undefined) => void) | undefined,\r\n onOpened?: ((request: WebRequest) => void) | undefined\r\n ) => IFileRequest;\r\n LoadImage: (\r\n input: string | ArrayBuffer | Blob | ArrayBufferView,\r\n onLoad: (img: HTMLImageElement | ImageBitmap) => void,\r\n onError: (message?: string | undefined, exception?: any) => void,\r\n offlineProvider: Nullable,\r\n mimeType?: string | undefined,\r\n imageBitmapOptions?: ImageBitmapOptions | undefined\r\n ) => Nullable;\r\n ReadFile: (\r\n file: File,\r\n onSuccess: (data: any) => void,\r\n onProgress?: ((ev: ProgressEvent) => any) | undefined,\r\n useArrayBuffer?: boolean | undefined,\r\n onError?: ((error: ReadFileError) => void) | undefined\r\n ) => IFileRequest;\r\n RequestFile: (\r\n url: string,\r\n onSuccess: (data: string | ArrayBuffer, request?: WebRequest | undefined) => void,\r\n onProgress?: ((event: ProgressEvent) => void) | undefined,\r\n offlineProvider?: IOfflineProvider | undefined,\r\n useArrayBuffer?: boolean | undefined,\r\n onError?: ((error: RequestFileError) => void) | undefined,\r\n onOpened?: ((request: WebRequest) => void) | undefined\r\n ) => IFileRequest;\r\n SetCorsBehavior: (url: string | string[], element: { crossOrigin: string | null }) => void;\r\n};\r\n/**\r\n * @param DecodeBase64UrlToBinary\r\n * @param DecodeBase64UrlToString\r\n * @param FileToolsOptions\r\n * @internal\r\n */\r\nexport const _injectLTSFileTools = (\r\n DecodeBase64UrlToBinary: (uri: string) => ArrayBuffer,\r\n DecodeBase64UrlToString: (uri: string) => string,\r\n FileToolsOptions: { DefaultRetryStrategy: any; BaseUrl: any; CorsBehavior: any; PreprocessUrl: any },\r\n IsBase64DataUrl: (uri: string) => boolean,\r\n IsFileURL: () => boolean,\r\n LoadFile: (\r\n fileOrUrl: string | File,\r\n onSuccess: (data: string | ArrayBuffer, responseURL?: string | undefined) => void,\r\n onProgress?: ((ev: ProgressEvent) => void) | undefined,\r\n offlineProvider?: IOfflineProvider | undefined,\r\n useArrayBuffer?: boolean | undefined,\r\n onError?: ((request?: WebRequest | undefined, exception?: LoadFileError | undefined) => void) | undefined,\r\n onOpened?: ((request: WebRequest) => void) | undefined\r\n ) => IFileRequest,\r\n LoadImage: (\r\n input: string | ArrayBuffer | ArrayBufferView | Blob,\r\n onLoad: (img: HTMLImageElement | ImageBitmap) => void,\r\n onError: (message?: string | undefined, exception?: any) => void,\r\n offlineProvider: Nullable,\r\n mimeType?: string,\r\n imageBitmapOptions?: ImageBitmapOptions | undefined\r\n ) => Nullable,\r\n ReadFile: (\r\n file: File,\r\n onSuccess: (data: any) => void,\r\n onProgress?: ((ev: ProgressEvent) => any) | undefined,\r\n useArrayBuffer?: boolean | undefined,\r\n onError?: ((error: ReadFileError) => void) | undefined\r\n ) => IFileRequest,\r\n RequestFile: (\r\n url: string,\r\n onSuccess: (data: string | ArrayBuffer, request?: WebRequest | undefined) => void,\r\n onProgress?: ((event: ProgressEvent) => void) | undefined,\r\n offlineProvider?: IOfflineProvider | undefined,\r\n useArrayBuffer?: boolean | undefined,\r\n onError?: ((error: RequestFileError) => void) | undefined,\r\n onOpened?: ((request: WebRequest) => void) | undefined\r\n ) => IFileRequest,\r\n SetCorsBehavior: (url: string | string[], element: { crossOrigin: string | null }) => void\r\n) => {\r\n /**\r\n * Backwards compatibility.\r\n * @internal\r\n * @deprecated\r\n */\r\n FileTools = {\r\n DecodeBase64UrlToBinary,\r\n DecodeBase64UrlToString,\r\n DefaultRetryStrategy: FileToolsOptions.DefaultRetryStrategy,\r\n BaseUrl: FileToolsOptions.BaseUrl,\r\n CorsBehavior: FileToolsOptions.CorsBehavior,\r\n PreprocessUrl: FileToolsOptions.PreprocessUrl,\r\n IsBase64DataUrl,\r\n IsFileURL,\r\n LoadFile,\r\n LoadImage,\r\n ReadFile,\r\n RequestFile,\r\n SetCorsBehavior,\r\n };\r\n\r\n Object.defineProperty(FileTools, \"DefaultRetryStrategy\", {\r\n get: function (this: null) {\r\n return FileToolsOptions.DefaultRetryStrategy;\r\n },\r\n set: function (this: null, value: (url: string, request: WebRequest, retryIndex: number) => number) {\r\n FileToolsOptions.DefaultRetryStrategy = value;\r\n },\r\n });\r\n\r\n Object.defineProperty(FileTools, \"BaseUrl\", {\r\n get: function (this: null) {\r\n return FileToolsOptions.BaseUrl;\r\n },\r\n set: function (this: null, value: string) {\r\n FileToolsOptions.BaseUrl = value;\r\n },\r\n });\r\n\r\n Object.defineProperty(FileTools, \"PreprocessUrl\", {\r\n get: function (this: null) {\r\n return FileToolsOptions.PreprocessUrl;\r\n },\r\n set: function (this: null, value: (url: string) => string) {\r\n FileToolsOptions.PreprocessUrl = value;\r\n },\r\n });\r\n\r\n Object.defineProperty(FileTools, \"CorsBehavior\", {\r\n get: function (this: null) {\r\n return FileToolsOptions.CorsBehavior;\r\n },\r\n set: function (this: null, value: string | ((url: string | string[]) => string)) {\r\n FileToolsOptions.CorsBehavior = value;\r\n },\r\n });\r\n};\r\n\r\n_injectLTSFileTools(DecodeBase64UrlToBinary, DecodeBase64UrlToString, FileToolsOptions, IsBase64DataUrl, IsFileURL, LoadFile, LoadImage, ReadFile, RequestFile, SetCorsBehavior);\r\n","import type { WebRequest } from \"./webRequest\";\r\n\r\n/**\r\n * Class used to define a retry strategy when error happens while loading assets\r\n */\r\nexport class RetryStrategy {\r\n /**\r\n * Function used to defines an exponential back off strategy\r\n * @param maxRetries defines the maximum number of retries (3 by default)\r\n * @param baseInterval defines the interval between retries\r\n * @returns the strategy function to use\r\n */\r\n public static ExponentialBackoff(maxRetries = 3, baseInterval = 500) {\r\n return (url: string, request: WebRequest, retryIndex: number): number => {\r\n if (request.status !== 0 || retryIndex >= maxRetries || url.indexOf(\"file:\") !== -1) {\r\n return -1;\r\n }\r\n\r\n return Math.pow(2, retryIndex) * baseInterval;\r\n };\r\n }\r\n}\r\n","import { Logger } from \"./logger\";\r\nimport { GetClass } from \"./typeStore\";\r\n\r\n/**\r\n * Class used to enable instantiation of objects by class name\r\n */\r\nexport class InstantiationTools {\r\n /**\r\n * Use this object to register external classes like custom textures or material\r\n * to allow the loaders to instantiate them\r\n */\r\n public static RegisteredExternalClasses: { [key: string]: Object } = {};\r\n\r\n /**\r\n * Tries to instantiate a new object from a given class name\r\n * @param className defines the class name to instantiate\r\n * @returns the new object or null if the system was not able to do the instantiation\r\n */\r\n public static Instantiate(className: string): any {\r\n if (this.RegisteredExternalClasses && this.RegisteredExternalClasses[className]) {\r\n return this.RegisteredExternalClasses[className];\r\n }\r\n\r\n const internalClass = GetClass(className);\r\n if (internalClass) {\r\n return internalClass;\r\n }\r\n\r\n Logger.Warn(className + \" not found, you may have missed an import.\");\r\n\r\n const arr = className.split(\".\");\r\n\r\n let fn: any = window || this;\r\n for (let i = 0, len = arr.length; i < len; i++) {\r\n fn = fn[arr[i]];\r\n }\r\n\r\n if (typeof fn !== \"function\") {\r\n return null;\r\n }\r\n\r\n return fn;\r\n }\r\n}\r\n","/**\r\n * Implementation from http://stackoverflow.com/questions/105034/how-to-create-a-guid-uuid-in-javascript/2117523#answer-2117523\r\n * Be aware Math.random() could cause collisions, but:\r\n * \"All but 6 of the 128 bits of the ID are randomly generated, which means that for any two ids, there's a 1 in 2^^122 (or 5.3x10^^36) chance they'll collide\"\r\n * @returns a pseudo random id\r\n */\r\nexport function RandomGUID(): string {\r\n return \"xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx\".replace(/[xy]/g, (c) => {\r\n const r = (Math.random() * 16) | 0,\r\n v = c === \"x\" ? r : (r & 0x3) | 0x8;\r\n return v.toString(16);\r\n });\r\n}\r\n/**\r\n * Class used to manipulate GUIDs\r\n */\r\nexport const GUID = {\r\n /**\r\n * Implementation from http://stackoverflow.com/questions/105034/how-to-create-a-guid-uuid-in-javascript/2117523#answer-2117523\r\n * Be aware Math.random() could cause collisions, but:\r\n * \"All but 6 of the 128 bits of the ID are randomly generated, which means that for any two ids, there's a 1 in 2^^122 (or 5.3x10^^36) chance they'll collide\"\r\n * @returns a pseudo random id\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n RandomId: RandomGUID,\r\n};\r\n","/* eslint-disable @typescript-eslint/no-unused-vars */\r\nimport type { Nullable } from \"../types\";\r\nimport { Observable } from \"./observable\";\r\nimport { GetDOMTextContent, IsNavigatorAvailable, IsWindowObjectExist } from \"./domManagement\";\r\nimport { Logger } from \"./logger\";\r\nimport { DeepCopier } from \"./deepCopier\";\r\nimport { PrecisionDate } from \"./precisionDate\";\r\nimport { _WarnImport } from \"./devTools\";\r\nimport { WebRequest } from \"./webRequest\";\r\nimport type { IFileRequest } from \"./fileRequest\";\r\nimport { EngineStore } from \"../Engines/engineStore\";\r\nimport type { ReadFileError } from \"./fileTools\";\r\nimport {\r\n FileToolsOptions,\r\n DecodeBase64UrlToBinary,\r\n IsBase64DataUrl,\r\n LoadFile as FileToolsLoadFile,\r\n LoadImage as FileToolLoadImage,\r\n ReadFile as FileToolsReadFile,\r\n SetCorsBehavior,\r\n} from \"./fileTools\";\r\nimport type { IOfflineProvider } from \"../Offline/IOfflineProvider\";\r\nimport { TimingTools } from \"./timingTools\";\r\nimport { InstantiationTools } from \"./instantiationTools\";\r\nimport { RandomGUID } from \"./guid\";\r\nimport type { IScreenshotSize } from \"./interfaces/screenshotSize\";\r\nimport type { Engine } from \"../Engines/engine\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\nimport type { IColor4Like } from \"../Maths/math.like\";\r\n\r\ndeclare function importScripts(...urls: string[]): void;\r\n\r\n/**\r\n * Class containing a set of static utilities functions\r\n */\r\nexport class Tools {\r\n /**\r\n * Gets or sets the base URL to use to load assets\r\n */\r\n public static get BaseUrl() {\r\n return FileToolsOptions.BaseUrl;\r\n }\r\n\r\n public static set BaseUrl(value: string) {\r\n FileToolsOptions.BaseUrl = value;\r\n }\r\n\r\n /**\r\n * Enable/Disable Custom HTTP Request Headers globally.\r\n * default = false\r\n * @see CustomRequestHeaders\r\n */\r\n public static UseCustomRequestHeaders = false;\r\n\r\n /**\r\n * Custom HTTP Request Headers to be sent with XMLHttpRequests\r\n * i.e. when loading files, where the server/service expects an Authorization header\r\n */\r\n public static CustomRequestHeaders = WebRequest.CustomRequestHeaders;\r\n\r\n /**\r\n * Gets or sets the retry strategy to apply when an error happens while loading an asset\r\n */\r\n public static get DefaultRetryStrategy() {\r\n return FileToolsOptions.DefaultRetryStrategy;\r\n }\r\n\r\n public static set DefaultRetryStrategy(strategy: (url: string, request: WebRequest, retryIndex: number) => number) {\r\n FileToolsOptions.DefaultRetryStrategy = strategy;\r\n }\r\n\r\n /**\r\n * Default behaviour for cors in the application.\r\n * It can be a string if the expected behavior is identical in the entire app.\r\n * Or a callback to be able to set it per url or on a group of them (in case of Video source for instance)\r\n */\r\n public static get CorsBehavior(): string | ((url: string | string[]) => string) {\r\n return FileToolsOptions.CorsBehavior;\r\n }\r\n\r\n public static set CorsBehavior(value: string | ((url: string | string[]) => string)) {\r\n FileToolsOptions.CorsBehavior = value;\r\n }\r\n\r\n /**\r\n * Gets or sets a global variable indicating if fallback texture must be used when a texture cannot be loaded\r\n * @ignorenaming\r\n */\r\n public static get UseFallbackTexture() {\r\n return EngineStore.UseFallbackTexture;\r\n }\r\n\r\n public static set UseFallbackTexture(value: boolean) {\r\n EngineStore.UseFallbackTexture = value;\r\n }\r\n\r\n /**\r\n * Use this object to register external classes like custom textures or material\r\n * to allow the loaders to instantiate them\r\n */\r\n public static get RegisteredExternalClasses() {\r\n return InstantiationTools.RegisteredExternalClasses;\r\n }\r\n\r\n public static set RegisteredExternalClasses(classes: { [key: string]: Object }) {\r\n InstantiationTools.RegisteredExternalClasses = classes;\r\n }\r\n\r\n /**\r\n * Texture content used if a texture cannot loaded\r\n * @ignorenaming\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public static get fallbackTexture() {\r\n return EngineStore.FallbackTexture;\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public static set fallbackTexture(value: string) {\r\n EngineStore.FallbackTexture = value;\r\n }\r\n\r\n /**\r\n * Read the content of a byte array at a specified coordinates (taking in account wrapping)\r\n * @param u defines the coordinate on X axis\r\n * @param v defines the coordinate on Y axis\r\n * @param width defines the width of the source data\r\n * @param height defines the height of the source data\r\n * @param pixels defines the source byte array\r\n * @param color defines the output color\r\n */\r\n public static FetchToRef(u: number, v: number, width: number, height: number, pixels: Uint8Array, color: IColor4Like): void {\r\n const wrappedU = (Math.abs(u) * width) % width | 0;\r\n const wrappedV = (Math.abs(v) * height) % height | 0;\r\n\r\n const position = (wrappedU + wrappedV * width) * 4;\r\n color.r = pixels[position] / 255;\r\n color.g = pixels[position + 1] / 255;\r\n color.b = pixels[position + 2] / 255;\r\n color.a = pixels[position + 3] / 255;\r\n }\r\n\r\n /**\r\n * Interpolates between a and b via alpha\r\n * @param a The lower value (returned when alpha = 0)\r\n * @param b The upper value (returned when alpha = 1)\r\n * @param alpha The interpolation-factor\r\n * @returns The mixed value\r\n */\r\n public static Mix(a: number, b: number, alpha: number): number {\r\n return a * (1 - alpha) + b * alpha;\r\n }\r\n\r\n /**\r\n * Tries to instantiate a new object from a given class name\r\n * @param className defines the class name to instantiate\r\n * @returns the new object or null if the system was not able to do the instantiation\r\n */\r\n public static Instantiate(className: string): any {\r\n return InstantiationTools.Instantiate(className);\r\n }\r\n\r\n /**\r\n * Polyfill for setImmediate\r\n * @param action defines the action to execute after the current execution block\r\n */\r\n public static SetImmediate(action: () => void) {\r\n TimingTools.SetImmediate(action);\r\n }\r\n\r\n /**\r\n * Function indicating if a number is an exponent of 2\r\n * @param value defines the value to test\r\n * @returns true if the value is an exponent of 2\r\n */\r\n public static IsExponentOfTwo(value: number): boolean {\r\n let count = 1;\r\n\r\n do {\r\n count *= 2;\r\n } while (count < value);\r\n\r\n return count === value;\r\n }\r\n\r\n private static _TmpFloatArray = new Float32Array(1);\r\n\r\n /**\r\n * Returns the nearest 32-bit single precision float representation of a Number\r\n * @param value A Number. If the parameter is of a different type, it will get converted\r\n * to a number or to NaN if it cannot be converted\r\n * @returns number\r\n */\r\n public static FloatRound(value: number): number {\r\n if (Math.fround) {\r\n return Math.fround(value);\r\n }\r\n\r\n return (Tools._TmpFloatArray[0] = value), Tools._TmpFloatArray[0];\r\n }\r\n\r\n /**\r\n * Extracts the filename from a path\r\n * @param path defines the path to use\r\n * @returns the filename\r\n */\r\n public static GetFilename(path: string): string {\r\n const index = path.lastIndexOf(\"/\");\r\n if (index < 0) {\r\n return path;\r\n }\r\n\r\n return path.substring(index + 1);\r\n }\r\n\r\n /**\r\n * Extracts the \"folder\" part of a path (everything before the filename).\r\n * @param uri The URI to extract the info from\r\n * @param returnUnchangedIfNoSlash Do not touch the URI if no slashes are present\r\n * @returns The \"folder\" part of the path\r\n */\r\n public static GetFolderPath(uri: string, returnUnchangedIfNoSlash = false): string {\r\n const index = uri.lastIndexOf(\"/\");\r\n if (index < 0) {\r\n if (returnUnchangedIfNoSlash) {\r\n return uri;\r\n }\r\n return \"\";\r\n }\r\n\r\n return uri.substring(0, index + 1);\r\n }\r\n\r\n /**\r\n * Extracts text content from a DOM element hierarchy\r\n * Back Compat only, please use GetDOMTextContent instead.\r\n */\r\n public static GetDOMTextContent = GetDOMTextContent;\r\n\r\n /**\r\n * Convert an angle in radians to degrees\r\n * @param angle defines the angle to convert\r\n * @returns the angle in degrees\r\n */\r\n public static ToDegrees(angle: number): number {\r\n return (angle * 180) / Math.PI;\r\n }\r\n\r\n /**\r\n * Convert an angle in degrees to radians\r\n * @param angle defines the angle to convert\r\n * @returns the angle in radians\r\n */\r\n public static ToRadians(angle: number): number {\r\n return (angle * Math.PI) / 180;\r\n }\r\n\r\n /**\r\n * Smooth angle changes (kind of low-pass filter), in particular for device orientation \"shaking\"\r\n * Use trigonometric functions to avoid discontinuity (0/360, -180/180)\r\n * @param previousAngle defines last angle value, in degrees\r\n * @param newAngle defines new angle value, in degrees\r\n * @param smoothFactor defines smoothing sensitivity; min 0: no smoothing, max 1: new data ignored\r\n * @returns the angle in degrees\r\n */\r\n public static SmoothAngleChange(previousAngle: number, newAngle: number, smoothFactor = 0.9): number {\r\n const previousAngleRad = this.ToRadians(previousAngle);\r\n const newAngleRad = this.ToRadians(newAngle);\r\n return this.ToDegrees(\r\n Math.atan2(\r\n (1 - smoothFactor) * Math.sin(newAngleRad) + smoothFactor * Math.sin(previousAngleRad),\r\n (1 - smoothFactor) * Math.cos(newAngleRad) + smoothFactor * Math.cos(previousAngleRad)\r\n )\r\n );\r\n }\r\n\r\n /**\r\n * Returns an array if obj is not an array\r\n * @param obj defines the object to evaluate as an array\r\n * @param allowsNullUndefined defines a boolean indicating if obj is allowed to be null or undefined\r\n * @returns either obj directly if obj is an array or a new array containing obj\r\n */\r\n public static MakeArray(obj: any, allowsNullUndefined?: boolean): Nullable> {\r\n if (allowsNullUndefined !== true && (obj === undefined || obj == null)) {\r\n return null;\r\n }\r\n\r\n return Array.isArray(obj) ? obj : [obj];\r\n }\r\n\r\n /**\r\n * Gets the pointer prefix to use\r\n * @param engine defines the engine we are finding the prefix for\r\n * @returns \"pointer\" if touch is enabled. Else returns \"mouse\"\r\n */\r\n public static GetPointerPrefix(engine: Engine): string {\r\n let eventPrefix = \"pointer\";\r\n\r\n // Check if pointer events are supported\r\n if (IsWindowObjectExist() && !window.PointerEvent) {\r\n eventPrefix = \"mouse\";\r\n }\r\n\r\n // Special Fallback MacOS Safari...\r\n if (\r\n engine._badDesktopOS &&\r\n !engine._badOS &&\r\n // And not ipad pros who claim to be macs...\r\n !(document && \"ontouchend\" in document)\r\n ) {\r\n eventPrefix = \"mouse\";\r\n }\r\n\r\n return eventPrefix;\r\n }\r\n\r\n /**\r\n * Sets the cors behavior on a dom element. This will add the required Tools.CorsBehavior to the element.\r\n * @param url define the url we are trying\r\n * @param element define the dom element where to configure the cors policy\r\n * @param element.crossOrigin\r\n */\r\n public static SetCorsBehavior(url: string | string[], element: { crossOrigin: string | null }): void {\r\n SetCorsBehavior(url, element);\r\n }\r\n\r\n /**\r\n * Sets the referrerPolicy behavior on a dom element.\r\n * @param referrerPolicy define the referrer policy to use\r\n * @param element define the dom element where to configure the referrer policy\r\n * @param element.referrerPolicy\r\n */\r\n public static SetReferrerPolicyBehavior(referrerPolicy: Nullable, element: { referrerPolicy: string | null }): void {\r\n element.referrerPolicy = referrerPolicy;\r\n }\r\n\r\n // External files\r\n\r\n /**\r\n * Removes unwanted characters from an url\r\n * @param url defines the url to clean\r\n * @returns the cleaned url\r\n */\r\n public static CleanUrl(url: string): string {\r\n url = url.replace(/#/gm, \"%23\");\r\n return url;\r\n }\r\n\r\n /**\r\n * Gets or sets a function used to pre-process url before using them to load assets\r\n */\r\n public static get PreprocessUrl() {\r\n return FileToolsOptions.PreprocessUrl;\r\n }\r\n\r\n public static set PreprocessUrl(processor: (url: string) => string) {\r\n FileToolsOptions.PreprocessUrl = processor;\r\n }\r\n\r\n /**\r\n * Loads an image as an HTMLImageElement.\r\n * @param input url string, ArrayBuffer, or Blob to load\r\n * @param onLoad callback called when the image successfully loads\r\n * @param onError callback called when the image fails to load\r\n * @param offlineProvider offline provider for caching\r\n * @param mimeType optional mime type\r\n * @param imageBitmapOptions optional the options to use when creating an ImageBitmap\r\n * @returns the HTMLImageElement of the loaded image\r\n */\r\n public static LoadImage(\r\n input: string | ArrayBuffer | Blob,\r\n onLoad: (img: HTMLImageElement | ImageBitmap) => void,\r\n onError: (message?: string, exception?: any) => void,\r\n offlineProvider: Nullable,\r\n mimeType?: string,\r\n imageBitmapOptions?: ImageBitmapOptions\r\n ): Nullable {\r\n return FileToolLoadImage(input, onLoad, onError, offlineProvider, mimeType, imageBitmapOptions);\r\n }\r\n\r\n /**\r\n * Loads a file from a url\r\n * @param url url string, ArrayBuffer, or Blob to load\r\n * @param onSuccess callback called when the file successfully loads\r\n * @param onProgress callback called while file is loading (if the server supports this mode)\r\n * @param offlineProvider defines the offline provider for caching\r\n * @param useArrayBuffer defines a boolean indicating that date must be returned as ArrayBuffer\r\n * @param onError callback called when the file fails to load\r\n * @returns a file request object\r\n */\r\n public static LoadFile(\r\n url: string,\r\n onSuccess: (data: string | ArrayBuffer, responseURL?: string) => void,\r\n onProgress?: (data: any) => void,\r\n offlineProvider?: IOfflineProvider,\r\n useArrayBuffer?: boolean,\r\n onError?: (request?: WebRequest, exception?: any) => void\r\n ): IFileRequest {\r\n return FileToolsLoadFile(url, onSuccess, onProgress, offlineProvider, useArrayBuffer, onError);\r\n }\r\n\r\n /**\r\n * Loads a file from a url\r\n * @param url the file url to load\r\n * @param useArrayBuffer defines a boolean indicating that date must be returned as ArrayBuffer\r\n * @returns a promise containing an ArrayBuffer corresponding to the loaded file\r\n */\r\n public static LoadFileAsync(url: string, useArrayBuffer = true): Promise {\r\n return new Promise((resolve, reject) => {\r\n FileToolsLoadFile(\r\n url,\r\n (data) => {\r\n resolve(data);\r\n },\r\n undefined,\r\n undefined,\r\n useArrayBuffer,\r\n (request, exception) => {\r\n reject(exception);\r\n }\r\n );\r\n });\r\n }\r\n\r\n /**\r\n * Load a script (identified by an url). When the url returns, the\r\n * content of this file is added into a new script element, attached to the DOM (body element)\r\n * @param scriptUrl defines the url of the script to laod\r\n * @param onSuccess defines the callback called when the script is loaded\r\n * @param onError defines the callback to call if an error occurs\r\n * @param scriptId defines the id of the script element\r\n */\r\n public static LoadScript(scriptUrl: string, onSuccess: () => void, onError?: (message?: string, exception?: any) => void, scriptId?: string) {\r\n if (typeof importScripts === \"function\") {\r\n try {\r\n importScripts(scriptUrl);\r\n onSuccess();\r\n } catch (e) {\r\n onError?.(`Unable to load script '${scriptUrl}' in worker`, e);\r\n }\r\n return;\r\n } else if (!IsWindowObjectExist()) {\r\n onError?.(`Cannot load script '${scriptUrl}' outside of a window or a worker`);\r\n return;\r\n }\r\n const head = document.getElementsByTagName(\"head\")[0];\r\n const script = document.createElement(\"script\");\r\n script.setAttribute(\"type\", \"text/javascript\");\r\n script.setAttribute(\"src\", scriptUrl);\r\n if (scriptId) {\r\n script.id = scriptId;\r\n }\r\n\r\n script.onload = () => {\r\n if (onSuccess) {\r\n onSuccess();\r\n }\r\n };\r\n\r\n script.onerror = (e) => {\r\n if (onError) {\r\n onError(`Unable to load script '${scriptUrl}'`, e);\r\n }\r\n };\r\n\r\n head.appendChild(script);\r\n }\r\n\r\n /**\r\n * Load an asynchronous script (identified by an url). When the url returns, the\r\n * content of this file is added into a new script element, attached to the DOM (body element)\r\n * @param scriptUrl defines the url of the script to laod\r\n * @returns a promise request object\r\n */\r\n public static LoadScriptAsync(scriptUrl: string): Promise {\r\n return new Promise((resolve, reject) => {\r\n this.LoadScript(\r\n scriptUrl,\r\n () => {\r\n resolve();\r\n },\r\n (message, exception) => {\r\n reject(exception || new Error(message));\r\n }\r\n );\r\n });\r\n }\r\n\r\n /**\r\n * Loads a file from a blob\r\n * @param fileToLoad defines the blob to use\r\n * @param callback defines the callback to call when data is loaded\r\n * @param progressCallback defines the callback to call during loading process\r\n * @returns a file request object\r\n */\r\n public static ReadFileAsDataURL(fileToLoad: Blob, callback: (data: any) => void, progressCallback: (ev: ProgressEvent) => any): IFileRequest {\r\n const reader = new FileReader();\r\n\r\n const request: IFileRequest = {\r\n onCompleteObservable: new Observable(),\r\n abort: () => reader.abort(),\r\n };\r\n\r\n reader.onloadend = () => {\r\n request.onCompleteObservable.notifyObservers(request);\r\n };\r\n\r\n reader.onload = (e) => {\r\n //target doesn't have result from ts 1.3\r\n callback((e.target)[\"result\"]);\r\n };\r\n\r\n reader.onprogress = progressCallback;\r\n\r\n reader.readAsDataURL(fileToLoad);\r\n\r\n return request;\r\n }\r\n\r\n /**\r\n * Reads a file from a File object\r\n * @param file defines the file to load\r\n * @param onSuccess defines the callback to call when data is loaded\r\n * @param onProgress defines the callback to call during loading process\r\n * @param useArrayBuffer defines a boolean indicating that data must be returned as an ArrayBuffer\r\n * @param onError defines the callback to call when an error occurs\r\n * @returns a file request object\r\n */\r\n public static ReadFile(\r\n file: File,\r\n onSuccess: (data: any) => void,\r\n onProgress?: (ev: ProgressEvent) => any,\r\n useArrayBuffer?: boolean,\r\n onError?: (error: ReadFileError) => void\r\n ): IFileRequest {\r\n return FileToolsReadFile(file, onSuccess, onProgress, useArrayBuffer, onError);\r\n }\r\n\r\n /**\r\n * Creates a data url from a given string content\r\n * @param content defines the content to convert\r\n * @returns the new data url link\r\n */\r\n public static FileAsURL(content: string): string {\r\n const fileBlob = new Blob([content]);\r\n const url = window.URL;\r\n const link: string = url.createObjectURL(fileBlob);\r\n return link;\r\n }\r\n\r\n /**\r\n * Format the given number to a specific decimal format\r\n * @param value defines the number to format\r\n * @param decimals defines the number of decimals to use\r\n * @returns the formatted string\r\n */\r\n public static Format(value: number, decimals = 2): string {\r\n return value.toFixed(decimals);\r\n }\r\n\r\n /**\r\n * Tries to copy an object by duplicating every property\r\n * @param source defines the source object\r\n * @param destination defines the target object\r\n * @param doNotCopyList defines a list of properties to avoid\r\n * @param mustCopyList defines a list of properties to copy (even if they start with _)\r\n */\r\n public static DeepCopy(source: any, destination: any, doNotCopyList?: string[], mustCopyList?: string[]): void {\r\n DeepCopier.DeepCopy(source, destination, doNotCopyList, mustCopyList);\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if the given object has no own property\r\n * @param obj defines the object to test\r\n * @returns true if object has no own property\r\n */\r\n public static IsEmpty(obj: any): boolean {\r\n for (const i in obj) {\r\n if (Object.prototype.hasOwnProperty.call(obj, i)) {\r\n return false;\r\n }\r\n }\r\n return true;\r\n }\r\n\r\n /**\r\n * Function used to register events at window level\r\n * @param windowElement defines the Window object to use\r\n * @param events defines the events to register\r\n */\r\n public static RegisterTopRootEvents(windowElement: Window, events: { name: string; handler: Nullable<(e: FocusEvent) => any> }[]): void {\r\n for (let index = 0; index < events.length; index++) {\r\n const event = events[index];\r\n windowElement.addEventListener(event.name, event.handler, false);\r\n\r\n try {\r\n if (window.parent) {\r\n window.parent.addEventListener(event.name, event.handler, false);\r\n }\r\n } catch (e) {\r\n // Silently fails...\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Function used to unregister events from window level\r\n * @param windowElement defines the Window object to use\r\n * @param events defines the events to unregister\r\n */\r\n public static UnregisterTopRootEvents(windowElement: Window, events: { name: string; handler: Nullable<(e: FocusEvent) => any> }[]): void {\r\n for (let index = 0; index < events.length; index++) {\r\n const event = events[index];\r\n windowElement.removeEventListener(event.name, event.handler);\r\n\r\n try {\r\n if (windowElement.parent) {\r\n windowElement.parent.removeEventListener(event.name, event.handler);\r\n }\r\n } catch (e) {\r\n // Silently fails...\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Dumps the current bound framebuffer\r\n * @param width defines the rendering width\r\n * @param height defines the rendering height\r\n * @param engine defines the hosting engine\r\n * @param successCallback defines the callback triggered once the data are available\r\n * @param mimeType defines the mime type of the result\r\n * @param fileName defines the filename to download. If present, the result will automatically be downloaded\r\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\r\n * @returns a void promise\r\n */\r\n public static async DumpFramebuffer(\r\n width: number,\r\n height: number,\r\n engine: Engine,\r\n successCallback?: (data: string) => void,\r\n mimeType = \"image/png\",\r\n fileName?: string,\r\n quality?: number\r\n ) {\r\n throw _WarnImport(\"DumpTools\");\r\n }\r\n\r\n /**\r\n * Dumps an array buffer\r\n * @param width defines the rendering width\r\n * @param height defines the rendering height\r\n * @param data the data array\r\n * @param successCallback defines the callback triggered once the data are available\r\n * @param mimeType defines the mime type of the result\r\n * @param fileName defines the filename to download. If present, the result will automatically be downloaded\r\n * @param invertY true to invert the picture in the Y dimension\r\n * @param toArrayBuffer true to convert the data to an ArrayBuffer (encoded as `mimeType`) instead of a base64 string\r\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\r\n */\r\n public static DumpData(\r\n width: number,\r\n height: number,\r\n data: ArrayBufferView,\r\n successCallback?: (data: string | ArrayBuffer) => void,\r\n mimeType = \"image/png\",\r\n fileName?: string,\r\n invertY = false,\r\n toArrayBuffer = false,\r\n quality?: number\r\n ) {\r\n throw _WarnImport(\"DumpTools\");\r\n }\r\n\r\n /**\r\n * Dumps an array buffer\r\n * @param width defines the rendering width\r\n * @param height defines the rendering height\r\n * @param data the data array\r\n * @param mimeType defines the mime type of the result\r\n * @param fileName defines the filename to download. If present, the result will automatically be downloaded\r\n * @param invertY true to invert the picture in the Y dimension\r\n * @param toArrayBuffer true to convert the data to an ArrayBuffer (encoded as `mimeType`) instead of a base64 string\r\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\r\n * @returns a promise that resolve to the final data\r\n */\r\n public static DumpDataAsync(\r\n width: number,\r\n height: number,\r\n data: ArrayBufferView,\r\n mimeType = \"image/png\",\r\n fileName?: string,\r\n invertY = false,\r\n toArrayBuffer = false,\r\n quality?: number\r\n ): Promise {\r\n throw _WarnImport(\"DumpTools\");\r\n }\r\n\r\n private static _IsOffScreenCanvas(canvas: HTMLCanvasElement | OffscreenCanvas): canvas is OffscreenCanvas {\r\n return (canvas as OffscreenCanvas).convertToBlob !== undefined;\r\n }\r\n\r\n /**\r\n * Converts the canvas data to blob.\r\n * This acts as a polyfill for browsers not supporting the to blob function.\r\n * @param canvas Defines the canvas to extract the data from (can be an offscreen canvas)\r\n * @param successCallback Defines the callback triggered once the data are available\r\n * @param mimeType Defines the mime type of the result\r\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\r\n */\r\n static ToBlob(canvas: HTMLCanvasElement | OffscreenCanvas, successCallback: (blob: Nullable) => void, mimeType = \"image/png\", quality?: number): void {\r\n // We need HTMLCanvasElement.toBlob for HD screenshots\r\n if (!Tools._IsOffScreenCanvas(canvas) && !canvas.toBlob) {\r\n // low performance polyfill based on toDataURL (https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob)\r\n canvas.toBlob = function (callback, type, quality) {\r\n setTimeout(() => {\r\n const binStr = atob(this.toDataURL(type, quality).split(\",\")[1]),\r\n len = binStr.length,\r\n arr = new Uint8Array(len);\r\n\r\n for (let i = 0; i < len; i++) {\r\n arr[i] = binStr.charCodeAt(i);\r\n }\r\n callback(new Blob([arr]));\r\n });\r\n };\r\n }\r\n if (Tools._IsOffScreenCanvas(canvas)) {\r\n canvas\r\n .convertToBlob({\r\n type: mimeType,\r\n quality,\r\n })\r\n .then((blob) => successCallback(blob));\r\n } else {\r\n canvas.toBlob(\r\n function (blob) {\r\n successCallback(blob);\r\n },\r\n mimeType,\r\n quality\r\n );\r\n }\r\n }\r\n\r\n /**\r\n * Download a Blob object\r\n * @param blob the Blob object\r\n * @param fileName the file name to download\r\n * @returns\r\n */\r\n static DownloadBlob(blob: Blob, fileName?: string) {\r\n //Creating a link if the browser have the download attribute on the a tag, to automatically start download generated image.\r\n if (\"download\" in document.createElement(\"a\")) {\r\n if (!fileName) {\r\n const date = new Date();\r\n const stringDate =\r\n (date.getFullYear() + \"-\" + (date.getMonth() + 1)).slice(2) + \"-\" + date.getDate() + \"_\" + date.getHours() + \"-\" + (\"0\" + date.getMinutes()).slice(-2);\r\n fileName = \"screenshot_\" + stringDate + \".png\";\r\n }\r\n Tools.Download(blob, fileName);\r\n } else {\r\n if (blob && typeof URL !== \"undefined\") {\r\n const url = URL.createObjectURL(blob);\r\n\r\n const newWindow = window.open(\"\");\r\n if (!newWindow) {\r\n return;\r\n }\r\n const img = newWindow.document.createElement(\"img\");\r\n img.onload = function () {\r\n // no longer need to read the blob so it's revoked\r\n URL.revokeObjectURL(url);\r\n };\r\n img.src = url;\r\n newWindow.document.body.appendChild(img);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Encodes the canvas data to base 64, or automatically downloads the result if `fileName` is defined.\r\n * @param canvas The canvas to get the data from, which can be an offscreen canvas.\r\n * @param successCallback The callback which is triggered once the data is available. If `fileName` is defined, the callback will be invoked after the download occurs, and the `data` argument will be an empty string.\r\n * @param mimeType The mime type of the result.\r\n * @param fileName The name of the file to download. If defined, the result will automatically be downloaded. If not defined, and `successCallback` is also not defined, the result will automatically be downloaded with an auto-generated file name.\r\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\r\n */\r\n static EncodeScreenshotCanvasData(\r\n canvas: HTMLCanvasElement | OffscreenCanvas,\r\n successCallback?: (data: string) => void,\r\n mimeType = \"image/png\",\r\n fileName?: string,\r\n quality?: number\r\n ): void {\r\n if (typeof fileName === \"string\" || !successCallback) {\r\n this.ToBlob(\r\n canvas,\r\n function (blob) {\r\n if (blob) {\r\n Tools.DownloadBlob(blob, fileName);\r\n }\r\n if (successCallback) {\r\n successCallback(\"\");\r\n }\r\n },\r\n mimeType,\r\n quality\r\n );\r\n } else if (successCallback) {\r\n if (Tools._IsOffScreenCanvas(canvas)) {\r\n canvas\r\n .convertToBlob({\r\n type: mimeType,\r\n quality,\r\n })\r\n .then((blob) => {\r\n const reader = new FileReader();\r\n reader.readAsDataURL(blob);\r\n reader.onloadend = () => {\r\n const base64data = reader.result;\r\n successCallback(base64data as string);\r\n };\r\n });\r\n return;\r\n }\r\n const base64Image = canvas.toDataURL(mimeType, quality);\r\n successCallback(base64Image);\r\n }\r\n }\r\n\r\n /**\r\n * Downloads a blob in the browser\r\n * @param blob defines the blob to download\r\n * @param fileName defines the name of the downloaded file\r\n */\r\n public static Download(blob: Blob, fileName: string): void {\r\n if (typeof URL === \"undefined\") {\r\n return;\r\n }\r\n\r\n const url = window.URL.createObjectURL(blob);\r\n const a = document.createElement(\"a\");\r\n document.body.appendChild(a);\r\n a.style.display = \"none\";\r\n a.href = url;\r\n a.download = fileName;\r\n a.addEventListener(\"click\", () => {\r\n if (a.parentElement) {\r\n a.parentElement.removeChild(a);\r\n }\r\n });\r\n a.click();\r\n window.URL.revokeObjectURL(url);\r\n }\r\n\r\n /**\r\n * Will return the right value of the noPreventDefault variable\r\n * Needed to keep backwards compatibility to the old API.\r\n *\r\n * @param args arguments passed to the attachControl function\r\n * @returns the correct value for noPreventDefault\r\n */\r\n public static BackCompatCameraNoPreventDefault(args: IArguments): boolean {\r\n // is it used correctly?\r\n if (typeof args[0] === \"boolean\") {\r\n return args[0];\r\n } else if (typeof args[1] === \"boolean\") {\r\n return args[1];\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Captures a screenshot of the current rendering\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/renderToPNG\r\n * @param engine defines the rendering engine\r\n * @param camera defines the source camera\r\n * @param size This parameter can be set to a single number or to an object with the\r\n * following (optional) properties: precision, width, height. If a single number is passed,\r\n * it will be used for both width and height. If an object is passed, the screenshot size\r\n * will be derived from the parameters. The precision property is a multiplier allowing\r\n * rendering at a higher or lower resolution\r\n * @param successCallback defines the callback receives a single parameter which contains the\r\n * screenshot as a string of base64-encoded characters. This string can be assigned to the\r\n * src parameter of an to display it\r\n * @param mimeType defines the MIME type of the screenshot image (default: image/png).\r\n * Check your browser for supported MIME types\r\n * @param forceDownload force the system to download the image even if a successCallback is provided\r\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public static CreateScreenshot(\r\n engine: Engine,\r\n camera: Camera,\r\n size: IScreenshotSize | number,\r\n successCallback?: (data: string) => void,\r\n mimeType = \"image/png\",\r\n forceDownload = false,\r\n quality?: number\r\n ): void {\r\n throw _WarnImport(\"ScreenshotTools\");\r\n }\r\n\r\n /**\r\n * Captures a screenshot of the current rendering\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/renderToPNG\r\n * @param engine defines the rendering engine\r\n * @param camera defines the source camera\r\n * @param size This parameter can be set to a single number or to an object with the\r\n * following (optional) properties: precision, width, height. If a single number is passed,\r\n * it will be used for both width and height. If an object is passed, the screenshot size\r\n * will be derived from the parameters. The precision property is a multiplier allowing\r\n * rendering at a higher or lower resolution\r\n * @param mimeType defines the MIME type of the screenshot image (default: image/png).\r\n * Check your browser for supported MIME types\r\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\r\n * @returns screenshot as a string of base64-encoded characters. This string can be assigned\r\n * to the src parameter of an to display it\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public static CreateScreenshotAsync(engine: Engine, camera: Camera, size: IScreenshotSize | number, mimeType = \"image/png\", quality?: number): Promise {\r\n throw _WarnImport(\"ScreenshotTools\");\r\n }\r\n\r\n /**\r\n * Generates an image screenshot from the specified camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/renderToPNG\r\n * @param engine The engine to use for rendering\r\n * @param camera The camera to use for rendering\r\n * @param size This parameter can be set to a single number or to an object with the\r\n * following (optional) properties: precision, width, height. If a single number is passed,\r\n * it will be used for both width and height. If an object is passed, the screenshot size\r\n * will be derived from the parameters. The precision property is a multiplier allowing\r\n * rendering at a higher or lower resolution\r\n * @param successCallback The callback receives a single parameter which contains the\r\n * screenshot as a string of base64-encoded characters. This string can be assigned to the\r\n * src parameter of an to display it\r\n * @param mimeType The MIME type of the screenshot image (default: image/png).\r\n * Check your browser for supported MIME types\r\n * @param samples Texture samples (default: 1)\r\n * @param antialiasing Whether antialiasing should be turned on or not (default: false)\r\n * @param fileName A name for for the downloaded file.\r\n * @param renderSprites Whether the sprites should be rendered or not (default: false)\r\n * @param enableStencilBuffer Whether the stencil buffer should be enabled or not (default: false)\r\n * @param useLayerMask if the camera's layer mask should be used to filter what should be rendered (default: true)\r\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public static CreateScreenshotUsingRenderTarget(\r\n engine: Engine,\r\n camera: Camera,\r\n size: IScreenshotSize | number,\r\n successCallback?: (data: string) => void,\r\n mimeType = \"image/png\",\r\n samples = 1,\r\n antialiasing = false,\r\n fileName?: string,\r\n renderSprites = false,\r\n enableStencilBuffer = false,\r\n useLayerMask = true,\r\n quality?: number\r\n ): void {\r\n throw _WarnImport(\"ScreenshotTools\");\r\n }\r\n\r\n /**\r\n * Generates an image screenshot from the specified camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/renderToPNG\r\n * @param engine The engine to use for rendering\r\n * @param camera The camera to use for rendering\r\n * @param size This parameter can be set to a single number or to an object with the\r\n * following (optional) properties: precision, width, height. If a single number is passed,\r\n * it will be used for both width and height. If an object is passed, the screenshot size\r\n * will be derived from the parameters. The precision property is a multiplier allowing\r\n * rendering at a higher or lower resolution\r\n * @param mimeType The MIME type of the screenshot image (default: image/png).\r\n * Check your browser for supported MIME types\r\n * @param samples Texture samples (default: 1)\r\n * @param antialiasing Whether antialiasing should be turned on or not (default: false)\r\n * @param fileName A name for for the downloaded file.\r\n * @returns screenshot as a string of base64-encoded characters. This string can be assigned\r\n * @param renderSprites Whether the sprites should be rendered or not (default: false)\r\n * @param enableStencilBuffer Whether the stencil buffer should be enabled or not (default: false)\r\n * @param useLayerMask if the camera's layer mask should be used to filter what should be rendered (default: true)\r\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\r\n * to the src parameter of an to display it\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public static CreateScreenshotUsingRenderTargetAsync(\r\n engine: Engine,\r\n camera: Camera,\r\n size: IScreenshotSize | number,\r\n mimeType = \"image/png\",\r\n samples = 1,\r\n antialiasing = false,\r\n fileName?: string,\r\n renderSprites = false,\r\n enableStencilBuffer = false,\r\n useLayerMask = true,\r\n quality?: number\r\n ): Promise {\r\n throw _WarnImport(\"ScreenshotTools\");\r\n }\r\n\r\n /**\r\n * Implementation from http://stackoverflow.com/questions/105034/how-to-create-a-guid-uuid-in-javascript/2117523#answer-2117523\r\n * Be aware Math.random() could cause collisions, but:\r\n * \"All but 6 of the 128 bits of the ID are randomly generated, which means that for any two ids, there's a 1 in 2^^122 (or 5.3x10^^36) chance they'll collide\"\r\n * @returns a pseudo random id\r\n */\r\n public static RandomId(): string {\r\n return RandomGUID();\r\n }\r\n\r\n /**\r\n * Test if the given uri is a base64 string\r\n * @deprecated Please use FileTools.IsBase64DataUrl instead.\r\n * @param uri The uri to test\r\n * @returns True if the uri is a base64 string or false otherwise\r\n */\r\n public static IsBase64(uri: string): boolean {\r\n return IsBase64DataUrl(uri);\r\n }\r\n\r\n /**\r\n * Decode the given base64 uri.\r\n * @deprecated Please use FileTools.DecodeBase64UrlToBinary instead.\r\n * @param uri The uri to decode\r\n * @returns The decoded base64 data.\r\n */\r\n public static DecodeBase64(uri: string): ArrayBuffer {\r\n return DecodeBase64UrlToBinary(uri);\r\n }\r\n\r\n public static GetAbsoluteUrl: (url: string) => string =\r\n typeof document === \"object\"\r\n ? (url) => {\r\n const a = document.createElement(\"a\");\r\n a.href = url;\r\n return a.href;\r\n }\r\n : typeof URL === \"function\" && typeof location === \"object\"\r\n ? (url) => new URL(url, location.origin).href\r\n : () => {\r\n throw new Error(\"Unable to get absolute URL. Override BABYLON.Tools.GetAbsoluteUrl to a custom implementation for the current context.\");\r\n };\r\n\r\n // Logs\r\n /**\r\n * No log\r\n */\r\n public static readonly NoneLogLevel = Logger.NoneLogLevel;\r\n /**\r\n * Only message logs\r\n */\r\n public static readonly MessageLogLevel = Logger.MessageLogLevel;\r\n /**\r\n * Only warning logs\r\n */\r\n public static readonly WarningLogLevel = Logger.WarningLogLevel;\r\n /**\r\n * Only error logs\r\n */\r\n public static readonly ErrorLogLevel = Logger.ErrorLogLevel;\r\n /**\r\n * All logs\r\n */\r\n public static readonly AllLogLevel = Logger.AllLogLevel;\r\n\r\n /**\r\n * Gets a value indicating the number of loading errors\r\n * @ignorenaming\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public static get errorsCount(): number {\r\n return Logger.errorsCount;\r\n }\r\n\r\n /**\r\n * Callback called when a new log is added\r\n */\r\n public static OnNewCacheEntry: (entry: string) => void;\r\n\r\n /**\r\n * Log a message to the console\r\n * @param message defines the message to log\r\n */\r\n public static Log(message: string): void {\r\n Logger.Log(message);\r\n }\r\n\r\n /**\r\n * Write a warning message to the console\r\n * @param message defines the message to log\r\n */\r\n public static Warn(message: string): void {\r\n Logger.Warn(message);\r\n }\r\n\r\n /**\r\n * Write an error message to the console\r\n * @param message defines the message to log\r\n */\r\n public static Error(message: string): void {\r\n Logger.Error(message);\r\n }\r\n\r\n /**\r\n * Gets current log cache (list of logs)\r\n */\r\n public static get LogCache(): string {\r\n return Logger.LogCache;\r\n }\r\n\r\n /**\r\n * Clears the log cache\r\n */\r\n public static ClearLogCache(): void {\r\n Logger.ClearLogCache();\r\n }\r\n\r\n /**\r\n * Sets the current log level (MessageLogLevel / WarningLogLevel / ErrorLogLevel)\r\n */\r\n public static set LogLevels(level: number) {\r\n Logger.LogLevels = level;\r\n }\r\n\r\n /**\r\n * Checks if the window object exists\r\n * Back Compat only, please use IsWindowObjectExist instead.\r\n */\r\n public static IsWindowObjectExist = IsWindowObjectExist;\r\n\r\n // Performances\r\n\r\n /**\r\n * No performance log\r\n */\r\n public static readonly PerformanceNoneLogLevel = 0;\r\n /**\r\n * Use user marks to log performance\r\n */\r\n public static readonly PerformanceUserMarkLogLevel = 1;\r\n /**\r\n * Log performance to the console\r\n */\r\n public static readonly PerformanceConsoleLogLevel = 2;\r\n\r\n private static _Performance: Performance;\r\n\r\n /**\r\n * Sets the current performance log level\r\n */\r\n public static set PerformanceLogLevel(level: number) {\r\n if ((level & Tools.PerformanceUserMarkLogLevel) === Tools.PerformanceUserMarkLogLevel) {\r\n Tools.StartPerformanceCounter = Tools._StartUserMark;\r\n Tools.EndPerformanceCounter = Tools._EndUserMark;\r\n return;\r\n }\r\n\r\n if ((level & Tools.PerformanceConsoleLogLevel) === Tools.PerformanceConsoleLogLevel) {\r\n Tools.StartPerformanceCounter = Tools._StartPerformanceConsole;\r\n Tools.EndPerformanceCounter = Tools._EndPerformanceConsole;\r\n return;\r\n }\r\n\r\n Tools.StartPerformanceCounter = Tools._StartPerformanceCounterDisabled;\r\n Tools.EndPerformanceCounter = Tools._EndPerformanceCounterDisabled;\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n private static _StartPerformanceCounterDisabled(counterName: string, condition?: boolean): void {}\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n private static _EndPerformanceCounterDisabled(counterName: string, condition?: boolean): void {}\r\n\r\n private static _StartUserMark(counterName: string, condition = true): void {\r\n if (!Tools._Performance) {\r\n if (!IsWindowObjectExist()) {\r\n return;\r\n }\r\n Tools._Performance = window.performance;\r\n }\r\n\r\n if (!condition || !Tools._Performance.mark) {\r\n return;\r\n }\r\n Tools._Performance.mark(counterName + \"-Begin\");\r\n }\r\n\r\n private static _EndUserMark(counterName: string, condition = true): void {\r\n if (!condition || !Tools._Performance.mark) {\r\n return;\r\n }\r\n Tools._Performance.mark(counterName + \"-End\");\r\n Tools._Performance.measure(counterName, counterName + \"-Begin\", counterName + \"-End\");\r\n }\r\n\r\n private static _StartPerformanceConsole(counterName: string, condition = true): void {\r\n if (!condition) {\r\n return;\r\n }\r\n\r\n Tools._StartUserMark(counterName, condition);\r\n\r\n if (console.time) {\r\n console.time(counterName);\r\n }\r\n }\r\n\r\n private static _EndPerformanceConsole(counterName: string, condition = true): void {\r\n if (!condition) {\r\n return;\r\n }\r\n\r\n Tools._EndUserMark(counterName, condition);\r\n\r\n console.timeEnd(counterName);\r\n }\r\n\r\n /**\r\n * Starts a performance counter\r\n */\r\n public static StartPerformanceCounter: (counterName: string, condition?: boolean) => void = Tools._StartPerformanceCounterDisabled;\r\n\r\n /**\r\n * Ends a specific performance counter\r\n */\r\n public static EndPerformanceCounter: (counterName: string, condition?: boolean) => void = Tools._EndPerformanceCounterDisabled;\r\n\r\n /**\r\n * Gets either window.performance.now() if supported or Date.now() else\r\n */\r\n public static get Now(): number {\r\n return PrecisionDate.Now;\r\n }\r\n\r\n /**\r\n * This method will return the name of the class used to create the instance of the given object.\r\n * It will works only on Javascript basic data types (number, string, ...) and instance of class declared with the @className decorator.\r\n * @param object the object to get the class name from\r\n * @param isType defines if the object is actually a type\r\n * @returns the name of the class, will be \"object\" for a custom data type not using the @className decorator\r\n */\r\n public static GetClassName(object: any, isType = false): string {\r\n let name = null;\r\n\r\n if (!isType && object.getClassName) {\r\n name = object.getClassName();\r\n } else {\r\n if (object instanceof Object) {\r\n const classObj = isType ? object : Object.getPrototypeOf(object);\r\n name = classObj.constructor[\"__bjsclassName__\"];\r\n }\r\n if (!name) {\r\n name = typeof object;\r\n }\r\n }\r\n return name;\r\n }\r\n\r\n /**\r\n * Gets the first element of an array satisfying a given predicate\r\n * @param array defines the array to browse\r\n * @param predicate defines the predicate to use\r\n * @returns null if not found or the element\r\n */\r\n public static First(array: Array, predicate: (item: T) => boolean): Nullable {\r\n for (const el of array) {\r\n if (predicate(el)) {\r\n return el;\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * This method will return the name of the full name of the class, including its owning module (if any).\r\n * It will works only on Javascript basic data types (number, string, ...) and instance of class declared with the @className decorator or implementing a method getClassName():string (in which case the module won't be specified).\r\n * @param object the object to get the class name from\r\n * @param isType defines if the object is actually a type\r\n * @returns a string that can have two forms: \"moduleName.className\" if module was specified when the class' Name was registered or \"className\" if there was not module specified.\r\n * @ignorenaming\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public static getFullClassName(object: any, isType = false): Nullable {\r\n let className = null;\r\n let moduleName = null;\r\n\r\n if (!isType && object.getClassName) {\r\n className = object.getClassName();\r\n } else {\r\n if (object instanceof Object) {\r\n const classObj = isType ? object : Object.getPrototypeOf(object);\r\n className = classObj.constructor[\"__bjsclassName__\"];\r\n moduleName = classObj.constructor[\"__bjsmoduleName__\"];\r\n }\r\n if (!className) {\r\n className = typeof object;\r\n }\r\n }\r\n\r\n if (!className) {\r\n return null;\r\n }\r\n\r\n return (moduleName != null ? moduleName + \".\" : \"\") + className;\r\n }\r\n\r\n /**\r\n * Returns a promise that resolves after the given amount of time.\r\n * @param delay Number of milliseconds to delay\r\n * @returns Promise that resolves after the given amount of time\r\n */\r\n public static DelayAsync(delay: number): Promise {\r\n return new Promise((resolve) => {\r\n setTimeout(() => {\r\n resolve();\r\n }, delay);\r\n });\r\n }\r\n\r\n /**\r\n * Utility function to detect if the current user agent is Safari\r\n * @returns whether or not the current user agent is safari\r\n */\r\n public static IsSafari(): boolean {\r\n if (!IsNavigatorAvailable()) {\r\n return false;\r\n }\r\n\r\n return /^((?!chrome|android).)*safari/i.test(navigator.userAgent);\r\n }\r\n}\r\n\r\n/**\r\n * Use this className as a decorator on a given class definition to add it a name and optionally its module.\r\n * You can then use the Tools.getClassName(obj) on an instance to retrieve its class name.\r\n * This method is the only way to get it done in all cases, even if the .js file declaring the class is minified\r\n * @param name The name of the class, case should be preserved\r\n * @param module The name of the Module hosting the class, optional, but strongly recommended to specify if possible. Case should be preserved.\r\n */\r\nexport function className(name: string, module?: string): (target: Object) => void {\r\n return (target: Object) => {\r\n (target)[\"__bjsclassName__\"] = name;\r\n (target)[\"__bjsmoduleName__\"] = module != null ? module : null;\r\n };\r\n}\r\n\r\n/**\r\n * An implementation of a loop for asynchronous functions.\r\n */\r\nexport class AsyncLoop {\r\n /**\r\n * Defines the current index of the loop.\r\n */\r\n public index: number;\r\n private _done: boolean;\r\n private _fn: (asyncLoop: AsyncLoop) => void;\r\n private _successCallback: () => void;\r\n\r\n /**\r\n * Constructor.\r\n * @param iterations the number of iterations.\r\n * @param func the function to run each iteration\r\n * @param successCallback the callback that will be called upon successful execution\r\n * @param offset starting offset.\r\n */\r\n constructor(\r\n /**\r\n * Defines the number of iterations for the loop\r\n */\r\n public iterations: number,\r\n func: (asyncLoop: AsyncLoop) => void,\r\n successCallback: () => void,\r\n offset = 0\r\n ) {\r\n this.index = offset - 1;\r\n this._done = false;\r\n this._fn = func;\r\n this._successCallback = successCallback;\r\n }\r\n\r\n /**\r\n * Execute the next iteration. Must be called after the last iteration was finished.\r\n */\r\n public executeNext(): void {\r\n if (!this._done) {\r\n if (this.index + 1 < this.iterations) {\r\n ++this.index;\r\n this._fn(this);\r\n } else {\r\n this.breakLoop();\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Break the loop and run the success callback.\r\n */\r\n public breakLoop(): void {\r\n this._done = true;\r\n this._successCallback();\r\n }\r\n\r\n /**\r\n * Create and run an async loop.\r\n * @param iterations the number of iterations.\r\n * @param fn the function to run each iteration\r\n * @param successCallback the callback that will be called upon successful execution\r\n * @param offset starting offset.\r\n * @returns the created async loop object\r\n */\r\n public static Run(iterations: number, fn: (asyncLoop: AsyncLoop) => void, successCallback: () => void, offset = 0): AsyncLoop {\r\n const loop = new AsyncLoop(iterations, fn, successCallback, offset);\r\n\r\n loop.executeNext();\r\n\r\n return loop;\r\n }\r\n\r\n /**\r\n * A for-loop that will run a given number of iterations synchronous and the rest async.\r\n * @param iterations total number of iterations\r\n * @param syncedIterations number of synchronous iterations in each async iteration.\r\n * @param fn the function to call each iteration.\r\n * @param callback a success call back that will be called when iterating stops.\r\n * @param breakFunction a break condition (optional)\r\n * @param timeout timeout settings for the setTimeout function. default - 0.\r\n * @returns the created async loop object\r\n */\r\n public static SyncAsyncForLoop(\r\n iterations: number,\r\n syncedIterations: number,\r\n fn: (iteration: number) => void,\r\n callback: () => void,\r\n breakFunction?: () => boolean,\r\n timeout = 0\r\n ): AsyncLoop {\r\n return AsyncLoop.Run(\r\n Math.ceil(iterations / syncedIterations),\r\n (loop: AsyncLoop) => {\r\n if (breakFunction && breakFunction()) {\r\n loop.breakLoop();\r\n } else {\r\n setTimeout(() => {\r\n for (let i = 0; i < syncedIterations; ++i) {\r\n const iteration = loop.index * syncedIterations + i;\r\n if (iteration >= iterations) {\r\n break;\r\n }\r\n fn(iteration);\r\n if (breakFunction && breakFunction()) {\r\n loop.breakLoop();\r\n break;\r\n }\r\n }\r\n loop.executeNext();\r\n }, timeout);\r\n }\r\n },\r\n callback\r\n );\r\n }\r\n}\r\n\r\n// Will only be define if Tools is imported freeing up some space when only engine is required\r\nEngineStore.FallbackTexture =\r\n \"\";\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\n// \"Coroutines are computer program components that generalize subroutines for non-preemptive multitasking, by allowing execution to be suspended and resumed.\"\r\n// https://en.wikipedia.org/wiki/Coroutine\r\n\r\n// In this implementation, coroutines are typically created via generator functions (function* with yield statements).\r\n// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/function*\r\n\r\n// In this implementation, the generator function (possibly parameterized) is referred to as a coroutine factory, and the returned iterator is referred to as the coroutine.\r\n// Technically yielding generator functions are not required - anything that implements the contract of Coroutine can be run as a coroutine.\r\n\r\n// The coroutine is started with the first call to next on the iterator, it is suspended with yield statements, and it is resumed with additional calls to next on the iterator.\r\n// To create an object satisfying the Coroutine contract with a generator function, it must not yield values, but rather only void via a plain \"yield;\" statement.\r\n// Coroutines can call other coroutines via:\r\n// 1. yield* someOtherCoroutine(); // If the called coroutine does not return a value\r\n// 2. const result = yield* someOtherCoroutine(); // If the called coroutine returns a value\r\n\r\n// Coroutines are run with the runCoroutine function, which takes a Coroutine, a CoroutineScheduler, and a success and error callback.\r\n// A scheduler is responsible for scheduling the next step of a coroutine, either synchronously or asynchronously.\r\n\r\n/**\r\n * A Coroutine is the intersection of:\r\n * 1. An Iterator that yields void, returns a T, and is not passed values with calls to next.\r\n * 2. An IterableIterator of void (since it only yields void).\r\n */\r\ntype CoroutineBase = Iterator & IterableIterator;\r\n/** @internal */\r\nexport type Coroutine = CoroutineBase;\r\n/** @internal */\r\nexport type AsyncCoroutine = CoroutineBase, T>;\r\n\r\n// A CoroutineStep represents a single step of a coroutine, and is an IteratorResult as returned from Coroutine.next().\r\n/** @internal */\r\nexport type CoroutineStep = IteratorResult;\r\n\r\n// A CoroutineScheduler is responsible for scheduling the call to Coroutine.next and invokes the success or error callback after next is called.\r\n/** @internal */\r\nexport type CoroutineScheduler = (coroutine: AsyncCoroutine, onStep: (stepResult: CoroutineStep) => void, onError: (stepError: any) => void) => void;\r\n\r\n// The inline scheduler simply steps the coroutine synchronously. This is useful for running a coroutine synchronously, and also as a helper function for other schedulers.\r\n/**\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport function inlineScheduler(coroutine: AsyncCoroutine, onStep: (stepResult: CoroutineStep) => void, onError: (stepError: any) => void) {\r\n try {\r\n const step = coroutine.next();\r\n\r\n if (step.done) {\r\n onStep(step);\r\n } else if (!step.value) {\r\n // NOTE: The properties of step have been narrowed, but the type of step itself is not narrowed, so the cast below is the most type safe way to deal with this without instantiating a new object to hold the values.\r\n onStep(step as { done: typeof step.done; value: typeof step.value });\r\n } else {\r\n step.value.then(() => {\r\n step.value = undefined;\r\n onStep(step as { done: typeof step.done; value: typeof step.value });\r\n }, onError);\r\n }\r\n } catch (error) {\r\n onError(error);\r\n }\r\n}\r\n\r\n// The yielding scheduler steps the coroutine synchronously until the specified time interval has elapsed, then yields control so other operations can be performed.\r\n// A single instance of a yielding scheduler could be shared across multiple coroutines to yield when their collective work exceeds the threshold.\r\n/**\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport function createYieldingScheduler(yieldAfterMS = 25) {\r\n let startTime: number | undefined;\r\n return (coroutine: AsyncCoroutine, onStep: (stepResult: CoroutineStep) => void, onError: (stepError: any) => void) => {\r\n const currentTime = performance.now();\r\n\r\n if (startTime === undefined || currentTime - startTime > yieldAfterMS) {\r\n // If this is the first coroutine step, or if the time interval has elapsed, record a new start time, and schedule the coroutine step to happen later, effectively yielding control of the execution context.\r\n startTime = currentTime;\r\n setTimeout(() => {\r\n inlineScheduler(coroutine, onStep, onError);\r\n }, 0);\r\n } else {\r\n // Otherwise it is not time to yield yet, so step the coroutine synchronously.\r\n inlineScheduler(coroutine, onStep, onError);\r\n }\r\n };\r\n}\r\n\r\n// Runs the specified coroutine with the specified scheduler. The success or error callback will be invoked when the coroutine finishes.\r\n/**\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport function runCoroutine(\r\n coroutine: AsyncCoroutine,\r\n scheduler: CoroutineScheduler,\r\n onSuccess: (result: T) => void,\r\n onError: (error: any) => void,\r\n abortSignal?: AbortSignal\r\n) {\r\n const resume = () => {\r\n let reschedule: boolean | undefined;\r\n\r\n const onStep = (stepResult: CoroutineStep) => {\r\n if (stepResult.done) {\r\n // If the coroutine is done, report success.\r\n onSuccess(stepResult.value);\r\n } else {\r\n // If the coroutine is not done, resume the coroutine (via the scheduler).\r\n if (reschedule === undefined) {\r\n // If reschedule is undefined at this point, then the coroutine must have stepped synchronously, so just flag another loop iteration.\r\n reschedule = true;\r\n } else {\r\n // If reschedule is defined at this point, then the coroutine must have stepped asynchronously, so call resume to restart the step loop.\r\n resume();\r\n }\r\n }\r\n };\r\n\r\n do {\r\n reschedule = undefined;\r\n\r\n if (!abortSignal || !abortSignal.aborted) {\r\n scheduler(coroutine, onStep, onError);\r\n } else {\r\n onError(new Error(\"Aborted\"));\r\n }\r\n\r\n if (reschedule === undefined) {\r\n // If reschedule is undefined at this point, then the coroutine must have stepped asynchronously, so stop looping and let the coroutine be resumed later.\r\n reschedule = false;\r\n }\r\n } while (reschedule);\r\n };\r\n\r\n resume();\r\n}\r\n\r\n// Runs the specified coroutine synchronously.\r\n/**\r\n * @internal\r\n */\r\nexport function runCoroutineSync(coroutine: Coroutine, abortSignal?: AbortSignal): T {\r\n // Run the coroutine with the inline scheduler, storing the returned value, or re-throwing the error (since the error callback will be called synchronously by the inline scheduler).\r\n let result: T | undefined;\r\n runCoroutine(\r\n coroutine,\r\n inlineScheduler,\r\n (r: T) => (result = r),\r\n (e: any) => {\r\n throw e;\r\n },\r\n abortSignal\r\n );\r\n\r\n // Synchronously return the result of the coroutine.\r\n return result!;\r\n}\r\n\r\n// Runs the specified coroutine asynchronously with the specified scheduler.\r\n/**\r\n * @internal\r\n */\r\nexport function runCoroutineAsync(coroutine: AsyncCoroutine, scheduler: CoroutineScheduler, abortSignal?: AbortSignal): Promise {\r\n // Run the coroutine with a yielding scheduler, resolving or rejecting the result promise when the coroutine finishes.\r\n return new Promise((resolve, reject) => {\r\n runCoroutine(coroutine, scheduler, resolve, reject, abortSignal);\r\n });\r\n}\r\n\r\n/**\r\n * Given a function that returns a Coroutine, produce a function with the same parameters that returns a T.\r\n * The returned function runs the coroutine synchronously.\r\n * @param coroutineFactory A function that returns a Coroutine.\r\n * @param abortSignal\r\n * @returns A function that runs the coroutine synchronously.\r\n * @internal\r\n */\r\nexport function makeSyncFunction(\r\n coroutineFactory: (...params: TParams) => Coroutine,\r\n abortSignal?: AbortSignal\r\n): (...params: TParams) => TReturn {\r\n return (...params: TParams) => {\r\n // Run the coroutine synchronously.\r\n return runCoroutineSync(coroutineFactory(...params), abortSignal);\r\n };\r\n}\r\n\r\n/**\r\n * Given a function that returns a Coroutine, product a function with the same parameters that returns a Promise.\r\n * The returned function runs the coroutine asynchronously, yield control of the execution context occasionally to enable a more responsive experience.\r\n * @param coroutineFactory A function that returns a Coroutine.\r\n * @param scheduler\r\n * @param abortSignal\r\n * @returns A function that runs the coroutine asynchronously.\r\n * @internal\r\n */\r\nexport function makeAsyncFunction(\r\n coroutineFactory: (...params: TParams) => AsyncCoroutine,\r\n scheduler: CoroutineScheduler,\r\n abortSignal?: AbortSignal\r\n): (...params: TParams) => Promise {\r\n return (...params: TParams) => {\r\n // Run the coroutine asynchronously.\r\n return runCoroutineAsync(coroutineFactory(...params), scheduler, abortSignal);\r\n };\r\n}\r\n","/**\r\n * Defines an array and its length.\r\n * It can be helpful to group result from both Arrays and smart arrays in one structure.\r\n */\r\nexport interface ISmartArrayLike {\r\n /**\r\n * The data of the array.\r\n */\r\n data: Array;\r\n /**\r\n * The active length of the array.\r\n */\r\n length: number;\r\n}\r\n\r\n/**\r\n * Defines an GC Friendly array where the backfield array do not shrink to prevent over allocations.\r\n */\r\nexport class SmartArray implements ISmartArrayLike {\r\n /**\r\n * The full set of data from the array.\r\n */\r\n public data: Array;\r\n\r\n /**\r\n * The active length of the array.\r\n */\r\n public length: number = 0;\r\n\r\n protected _id: number;\r\n\r\n /**\r\n * Instantiates a Smart Array.\r\n * @param capacity defines the default capacity of the array.\r\n */\r\n constructor(capacity: number) {\r\n this.data = new Array(capacity);\r\n this._id = SmartArray._GlobalId++;\r\n }\r\n\r\n /**\r\n * Pushes a value at the end of the active data.\r\n * @param value defines the object to push in the array.\r\n */\r\n public push(value: T): void {\r\n this.data[this.length++] = value;\r\n\r\n if (this.length > this.data.length) {\r\n this.data.length *= 2;\r\n }\r\n }\r\n\r\n /**\r\n * Iterates over the active data and apply the lambda to them.\r\n * @param func defines the action to apply on each value.\r\n */\r\n public forEach(func: (content: T) => void): void {\r\n for (let index = 0; index < this.length; index++) {\r\n func(this.data[index]);\r\n }\r\n }\r\n\r\n /**\r\n * Sorts the full sets of data.\r\n * @param compareFn defines the comparison function to apply.\r\n */\r\n public sort(compareFn: (a: T, b: T) => number): void {\r\n this.data.sort(compareFn);\r\n }\r\n\r\n /**\r\n * Resets the active data to an empty array.\r\n */\r\n public reset(): void {\r\n this.length = 0;\r\n }\r\n\r\n /**\r\n * Releases all the data from the array as well as the array.\r\n */\r\n public dispose(): void {\r\n this.reset();\r\n\r\n if (this.data) {\r\n this.data.length = 0;\r\n }\r\n }\r\n\r\n /**\r\n * Concats the active data with a given array.\r\n * @param array defines the data to concatenate with.\r\n */\r\n public concat(array: any): void {\r\n if (array.length === 0) {\r\n return;\r\n }\r\n if (this.length + array.length > this.data.length) {\r\n this.data.length = (this.length + array.length) * 2;\r\n }\r\n\r\n for (let index = 0; index < array.length; index++) {\r\n this.data[this.length++] = (array.data || array)[index];\r\n }\r\n }\r\n\r\n /**\r\n * Returns the position of a value in the active data.\r\n * @param value defines the value to find the index for\r\n * @returns the index if found in the active data otherwise -1\r\n */\r\n public indexOf(value: T): number {\r\n const position = this.data.indexOf(value);\r\n\r\n if (position >= this.length) {\r\n return -1;\r\n }\r\n\r\n return position;\r\n }\r\n\r\n /**\r\n * Returns whether an element is part of the active data.\r\n * @param value defines the value to look for\r\n * @returns true if found in the active data otherwise false\r\n */\r\n public contains(value: T): boolean {\r\n return this.indexOf(value) !== -1;\r\n }\r\n\r\n // Statics\r\n private static _GlobalId = 0;\r\n}\r\n\r\n/**\r\n * Defines an GC Friendly array where the backfield array do not shrink to prevent over allocations.\r\n * The data in this array can only be present once\r\n */\r\nexport class SmartArrayNoDuplicate extends SmartArray {\r\n private _duplicateId = 0;\r\n\r\n /**\r\n * Pushes a value at the end of the active data.\r\n * THIS DOES NOT PREVENT DUPPLICATE DATA\r\n * @param value defines the object to push in the array.\r\n */\r\n public push(value: T): void {\r\n super.push(value);\r\n\r\n if (!(value).__smartArrayFlags) {\r\n (value).__smartArrayFlags = {};\r\n }\r\n\r\n (value).__smartArrayFlags[this._id] = this._duplicateId;\r\n }\r\n\r\n /**\r\n * Pushes a value at the end of the active data.\r\n * If the data is already present, it won t be added again\r\n * @param value defines the object to push in the array.\r\n * @returns true if added false if it was already present\r\n */\r\n public pushNoDuplicate(value: T): boolean {\r\n if ((value).__smartArrayFlags && (value).__smartArrayFlags[this._id] === this._duplicateId) {\r\n return false;\r\n }\r\n this.push(value);\r\n return true;\r\n }\r\n\r\n /**\r\n * Resets the active data to an empty array.\r\n */\r\n public reset(): void {\r\n super.reset();\r\n this._duplicateId++;\r\n }\r\n\r\n /**\r\n * Concats the active data with a given array.\r\n * This ensures no duplicate will be present in the result.\r\n * @param array defines the data to concatenate with.\r\n */\r\n public concatWithNoDuplicate(array: any): void {\r\n if (array.length === 0) {\r\n return;\r\n }\r\n if (this.length + array.length > this.data.length) {\r\n this.data.length = (this.length + array.length) * 2;\r\n }\r\n\r\n for (let index = 0; index < array.length; index++) {\r\n const item = (array.data || array)[index];\r\n this.pushNoDuplicate(item);\r\n }\r\n }\r\n}\r\n","/**\r\n * Class used to represent a viewport on screen\r\n */\r\nexport class Viewport {\r\n /**\r\n * Creates a Viewport object located at (x, y) and sized (width, height)\r\n * @param x defines viewport left coordinate\r\n * @param y defines viewport top coordinate\r\n * @param width defines the viewport width\r\n * @param height defines the viewport height\r\n */\r\n constructor(\r\n /** viewport left coordinate */\r\n public x: number,\r\n /** viewport top coordinate */\r\n public y: number,\r\n /**viewport width */\r\n public width: number,\r\n /** viewport height */\r\n public height: number\r\n ) {}\r\n\r\n /**\r\n * Creates a new viewport using absolute sizing (from 0-> width, 0-> height instead of 0->1)\r\n * @param renderWidth defines the rendering width\r\n * @param renderHeight defines the rendering height\r\n * @returns a new Viewport\r\n */\r\n public toGlobal(renderWidth: number, renderHeight: number): Viewport {\r\n return new Viewport(this.x * renderWidth, this.y * renderHeight, this.width * renderWidth, this.height * renderHeight);\r\n }\r\n\r\n /**\r\n * Stores absolute viewport value into a target viewport (from 0-> width, 0-> height instead of 0->1)\r\n * @param renderWidth defines the rendering width\r\n * @param renderHeight defines the rendering height\r\n * @param ref defines the target viewport\r\n * @returns the current viewport\r\n */\r\n public toGlobalToRef(renderWidth: number, renderHeight: number, ref: Viewport): Viewport {\r\n ref.x = this.x * renderWidth;\r\n ref.y = this.y * renderHeight;\r\n ref.width = this.width * renderWidth;\r\n ref.height = this.height * renderHeight;\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns a new Viewport copied from the current one\r\n * @returns a new Viewport\r\n */\r\n public clone(): Viewport {\r\n return new Viewport(this.x, this.y, this.width, this.height);\r\n }\r\n}\r\n","import type { DeepImmutable } from \"../types\";\r\nimport { Vector3, Matrix } from \"./math.vector\";\r\n\r\n/**\r\n * Represents a plane by the equation ax + by + cz + d = 0\r\n */\r\nexport class Plane {\r\n private static _TmpMatrix = Matrix.Identity();\r\n\r\n /**\r\n * Normal of the plane (a,b,c)\r\n */\r\n public normal: Vector3;\r\n /**\r\n * d component of the plane\r\n */\r\n public d: number;\r\n /**\r\n * Creates a Plane object according to the given floats a, b, c, d and the plane equation : ax + by + cz + d = 0\r\n * @param a a component of the plane\r\n * @param b b component of the plane\r\n * @param c c component of the plane\r\n * @param d d component of the plane\r\n */\r\n constructor(a: number, b: number, c: number, d: number) {\r\n this.normal = new Vector3(a, b, c);\r\n this.d = d;\r\n }\r\n\r\n /**\r\n * @returns the plane coordinates as a new array of 4 elements [a, b, c, d].\r\n */\r\n public asArray(): number[] {\r\n return [this.normal.x, this.normal.y, this.normal.z, this.d];\r\n }\r\n\r\n // Methods\r\n /**\r\n * @returns a new plane copied from the current Plane.\r\n */\r\n public clone(): Plane {\r\n return new Plane(this.normal.x, this.normal.y, this.normal.z, this.d);\r\n }\r\n /**\r\n * @returns the string \"Plane\".\r\n */\r\n public getClassName(): string {\r\n return \"Plane\";\r\n }\r\n /**\r\n * @returns the Plane hash code.\r\n */\r\n public getHashCode(): number {\r\n let hash = this.normal.getHashCode();\r\n hash = (hash * 397) ^ (this.d | 0);\r\n return hash;\r\n }\r\n /**\r\n * Normalize the current Plane in place.\r\n * @returns the updated Plane.\r\n */\r\n public normalize(): Plane {\r\n const norm = Math.sqrt(this.normal.x * this.normal.x + this.normal.y * this.normal.y + this.normal.z * this.normal.z);\r\n let magnitude = 0.0;\r\n\r\n if (norm !== 0) {\r\n magnitude = 1.0 / norm;\r\n }\r\n this.normal.x *= magnitude;\r\n this.normal.y *= magnitude;\r\n this.normal.z *= magnitude;\r\n this.d *= magnitude;\r\n return this;\r\n }\r\n /**\r\n * Applies a transformation the plane and returns the result\r\n * @param transformation the transformation matrix to be applied to the plane\r\n * @returns a new Plane as the result of the transformation of the current Plane by the given matrix.\r\n */\r\n public transform(transformation: DeepImmutable): Plane {\r\n const invertedMatrix = Plane._TmpMatrix;\r\n transformation.invertToRef(invertedMatrix);\r\n const m = invertedMatrix.m;\r\n const x = this.normal.x;\r\n const y = this.normal.y;\r\n const z = this.normal.z;\r\n const d = this.d;\r\n\r\n const normalX = x * m[0] + y * m[1] + z * m[2] + d * m[3];\r\n const normalY = x * m[4] + y * m[5] + z * m[6] + d * m[7];\r\n const normalZ = x * m[8] + y * m[9] + z * m[10] + d * m[11];\r\n const finalD = x * m[12] + y * m[13] + z * m[14] + d * m[15];\r\n\r\n return new Plane(normalX, normalY, normalZ, finalD);\r\n }\r\n\r\n /**\r\n * Compute the dot product between the point and the plane normal\r\n * @param point point to calculate the dot product with\r\n * @returns the dot product (float) of the point coordinates and the plane normal.\r\n */\r\n public dotCoordinate(point: DeepImmutable): number {\r\n return this.normal.x * point.x + this.normal.y * point.y + this.normal.z * point.z + this.d;\r\n }\r\n\r\n /**\r\n * Updates the current Plane from the plane defined by the three given points.\r\n * @param point1 one of the points used to construct the plane\r\n * @param point2 one of the points used to construct the plane\r\n * @param point3 one of the points used to construct the plane\r\n * @returns the updated Plane.\r\n */\r\n public copyFromPoints(point1: DeepImmutable, point2: DeepImmutable, point3: DeepImmutable): Plane {\r\n const x1 = point2.x - point1.x;\r\n const y1 = point2.y - point1.y;\r\n const z1 = point2.z - point1.z;\r\n const x2 = point3.x - point1.x;\r\n const y2 = point3.y - point1.y;\r\n const z2 = point3.z - point1.z;\r\n const yz = y1 * z2 - z1 * y2;\r\n const xz = z1 * x2 - x1 * z2;\r\n const xy = x1 * y2 - y1 * x2;\r\n const pyth = Math.sqrt(yz * yz + xz * xz + xy * xy);\r\n let invPyth;\r\n\r\n if (pyth !== 0) {\r\n invPyth = 1.0 / pyth;\r\n } else {\r\n invPyth = 0.0;\r\n }\r\n\r\n this.normal.x = yz * invPyth;\r\n this.normal.y = xz * invPyth;\r\n this.normal.z = xy * invPyth;\r\n this.d = -(this.normal.x * point1.x + this.normal.y * point1.y + this.normal.z * point1.z);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Checks if the plane is facing a given direction (meaning if the plane's normal is pointing in the opposite direction of the given vector).\r\n * Note that for this function to work as expected you should make sure that:\r\n * - direction and the plane normal are normalized\r\n * - epsilon is a number just bigger than -1, something like -0.99 for eg\r\n * @param direction the direction to check if the plane is facing\r\n * @param epsilon value the dot product is compared against (returns true if dot <= epsilon)\r\n * @returns True if the plane is facing the given direction\r\n */\r\n public isFrontFacingTo(direction: DeepImmutable, epsilon: number): boolean {\r\n const dot = Vector3.Dot(this.normal, direction);\r\n return dot <= epsilon;\r\n }\r\n\r\n /**\r\n * Calculates the distance to a point\r\n * @param point point to calculate distance to\r\n * @returns the signed distance (float) from the given point to the Plane.\r\n */\r\n public signedDistanceTo(point: DeepImmutable): number {\r\n return Vector3.Dot(point, this.normal) + this.d;\r\n }\r\n\r\n // Statics\r\n /**\r\n * Creates a plane from an array\r\n * @param array the array to create a plane from\r\n * @returns a new Plane from the given array.\r\n */\r\n static FromArray(array: DeepImmutable>): Plane {\r\n return new Plane(array[0], array[1], array[2], array[3]);\r\n }\r\n /**\r\n * Creates a plane from three points\r\n * @param point1 point used to create the plane\r\n * @param point2 point used to create the plane\r\n * @param point3 point used to create the plane\r\n * @returns a new Plane defined by the three given points.\r\n */\r\n static FromPoints(point1: DeepImmutable, point2: DeepImmutable, point3: DeepImmutable): Plane {\r\n const result = new Plane(0.0, 0.0, 0.0, 0.0);\r\n result.copyFromPoints(point1, point2, point3);\r\n return result;\r\n }\r\n /**\r\n * Creates a plane from an origin point and a normal\r\n * @param origin origin of the plane to be constructed\r\n * @param normal normal of the plane to be constructed\r\n * @returns a new Plane the normal vector to this plane at the given origin point.\r\n * Note : the vector \"normal\" is updated because normalized.\r\n */\r\n static FromPositionAndNormal(origin: DeepImmutable, normal: Vector3): Plane {\r\n const result = new Plane(0.0, 0.0, 0.0, 0.0);\r\n normal.normalize();\r\n result.normal = normal;\r\n result.d = -(normal.x * origin.x + normal.y * origin.y + normal.z * origin.z);\r\n return result;\r\n }\r\n\r\n /**\r\n * Calculates the distance from a plane and a point\r\n * @param origin origin of the plane to be constructed\r\n * @param normal normal of the plane to be constructed\r\n * @param point point to calculate distance to\r\n * @returns the signed distance between the plane defined by the normal vector at the \"origin\"\" point and the given other point.\r\n */\r\n static SignedDistanceToPlaneFromPositionAndNormal(origin: DeepImmutable, normal: DeepImmutable, point: DeepImmutable): number {\r\n const d = -(normal.x * origin.x + normal.y * origin.y + normal.z * origin.z);\r\n return Vector3.Dot(point, normal) + d;\r\n }\r\n}\r\n","import type { Matrix, Vector3 } from \"./math.vector\";\r\nimport type { DeepImmutable } from \"../types\";\r\nimport { Plane } from \"./math.plane\";\r\n\r\n/**\r\n * Represents a camera frustum\r\n */\r\nexport class Frustum {\r\n /**\r\n * Gets the planes representing the frustum\r\n * @param transform matrix to be applied to the returned planes\r\n * @returns a new array of 6 Frustum planes computed by the given transformation matrix.\r\n */\r\n public static GetPlanes(transform: DeepImmutable): Plane[] {\r\n const frustumPlanes = [];\r\n for (let index = 0; index < 6; index++) {\r\n frustumPlanes.push(new Plane(0.0, 0.0, 0.0, 0.0));\r\n }\r\n Frustum.GetPlanesToRef(transform, frustumPlanes);\r\n return frustumPlanes;\r\n }\r\n\r\n /**\r\n * Gets the near frustum plane transformed by the transform matrix\r\n * @param transform transformation matrix to be applied to the resulting frustum plane\r\n * @param frustumPlane the resulting frustum plane\r\n */\r\n public static GetNearPlaneToRef(transform: DeepImmutable, frustumPlane: Plane): void {\r\n const m = transform.m;\r\n frustumPlane.normal.x = m[3] + m[2];\r\n frustumPlane.normal.y = m[7] + m[6];\r\n frustumPlane.normal.z = m[11] + m[10];\r\n frustumPlane.d = m[15] + m[14];\r\n frustumPlane.normalize();\r\n }\r\n\r\n /**\r\n * Gets the far frustum plane transformed by the transform matrix\r\n * @param transform transformation matrix to be applied to the resulting frustum plane\r\n * @param frustumPlane the resulting frustum plane\r\n */\r\n public static GetFarPlaneToRef(transform: DeepImmutable, frustumPlane: Plane): void {\r\n const m = transform.m;\r\n frustumPlane.normal.x = m[3] - m[2];\r\n frustumPlane.normal.y = m[7] - m[6];\r\n frustumPlane.normal.z = m[11] - m[10];\r\n frustumPlane.d = m[15] - m[14];\r\n frustumPlane.normalize();\r\n }\r\n\r\n /**\r\n * Gets the left frustum plane transformed by the transform matrix\r\n * @param transform transformation matrix to be applied to the resulting frustum plane\r\n * @param frustumPlane the resulting frustum plane\r\n */\r\n public static GetLeftPlaneToRef(transform: DeepImmutable, frustumPlane: Plane): void {\r\n const m = transform.m;\r\n frustumPlane.normal.x = m[3] + m[0];\r\n frustumPlane.normal.y = m[7] + m[4];\r\n frustumPlane.normal.z = m[11] + m[8];\r\n frustumPlane.d = m[15] + m[12];\r\n frustumPlane.normalize();\r\n }\r\n\r\n /**\r\n * Gets the right frustum plane transformed by the transform matrix\r\n * @param transform transformation matrix to be applied to the resulting frustum plane\r\n * @param frustumPlane the resulting frustum plane\r\n */\r\n public static GetRightPlaneToRef(transform: DeepImmutable, frustumPlane: Plane): void {\r\n const m = transform.m;\r\n frustumPlane.normal.x = m[3] - m[0];\r\n frustumPlane.normal.y = m[7] - m[4];\r\n frustumPlane.normal.z = m[11] - m[8];\r\n frustumPlane.d = m[15] - m[12];\r\n frustumPlane.normalize();\r\n }\r\n\r\n /**\r\n * Gets the top frustum plane transformed by the transform matrix\r\n * @param transform transformation matrix to be applied to the resulting frustum plane\r\n * @param frustumPlane the resulting frustum plane\r\n */\r\n public static GetTopPlaneToRef(transform: DeepImmutable, frustumPlane: Plane): void {\r\n const m = transform.m;\r\n frustumPlane.normal.x = m[3] - m[1];\r\n frustumPlane.normal.y = m[7] - m[5];\r\n frustumPlane.normal.z = m[11] - m[9];\r\n frustumPlane.d = m[15] - m[13];\r\n frustumPlane.normalize();\r\n }\r\n\r\n /**\r\n * Gets the bottom frustum plane transformed by the transform matrix\r\n * @param transform transformation matrix to be applied to the resulting frustum plane\r\n * @param frustumPlane the resulting frustum plane\r\n */\r\n public static GetBottomPlaneToRef(transform: DeepImmutable, frustumPlane: Plane): void {\r\n const m = transform.m;\r\n frustumPlane.normal.x = m[3] + m[1];\r\n frustumPlane.normal.y = m[7] + m[5];\r\n frustumPlane.normal.z = m[11] + m[9];\r\n frustumPlane.d = m[15] + m[13];\r\n frustumPlane.normalize();\r\n }\r\n\r\n /**\r\n * Sets the given array \"frustumPlanes\" with the 6 Frustum planes computed by the given transformation matrix.\r\n * @param transform transformation matrix to be applied to the resulting frustum planes\r\n * @param frustumPlanes the resulting frustum planes\r\n */\r\n public static GetPlanesToRef(transform: DeepImmutable, frustumPlanes: Plane[]): void {\r\n // Near\r\n Frustum.GetNearPlaneToRef(transform, frustumPlanes[0]);\r\n\r\n // Far\r\n Frustum.GetFarPlaneToRef(transform, frustumPlanes[1]);\r\n\r\n // Left\r\n Frustum.GetLeftPlaneToRef(transform, frustumPlanes[2]);\r\n\r\n // Right\r\n Frustum.GetRightPlaneToRef(transform, frustumPlanes[3]);\r\n\r\n // Top\r\n Frustum.GetTopPlaneToRef(transform, frustumPlanes[4]);\r\n\r\n // Bottom\r\n Frustum.GetBottomPlaneToRef(transform, frustumPlanes[5]);\r\n }\r\n\r\n /**\r\n * Tests if a point is located between the frustum planes.\r\n * @param point defines the point to test\r\n * @param frustumPlanes defines the frustum planes to test\r\n * @returns true if the point is located between the frustum planes\r\n */\r\n public static IsPointInFrustum(point: Vector3, frustumPlanes: Array>): boolean {\r\n for (let i = 0; i < 6; i++) {\r\n if (frustumPlanes[i].dotCoordinate(point) < 0) {\r\n return false;\r\n }\r\n }\r\n return true;\r\n }\r\n}\r\n","import { serialize, SerializationHelper, serializeAsVector3 } from \"../Misc/decorators\";\r\nimport { SmartArray } from \"../Misc/smartArray\";\r\nimport { Tools } from \"../Misc/tools\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { CameraInputsManager } from \"./cameraInputsManager\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Matrix, Vector3, Quaternion } from \"../Maths/math.vector\";\r\nimport { Node } from \"../node\";\r\nimport type { Mesh } from \"../Meshes/mesh\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { ICullable } from \"../Culling/boundingInfo\";\r\nimport { Logger } from \"../Misc/logger\";\r\nimport { GetClass } from \"../Misc/typeStore\";\r\nimport { _WarnImport } from \"../Misc/devTools\";\r\nimport { Viewport } from \"../Maths/math.viewport\";\r\nimport { Frustum } from \"../Maths/math.frustum\";\r\nimport type { Plane } from \"../Maths/math.plane\";\r\nimport { Constants } from \"../Engines/constants\";\r\n\r\nimport type { PostProcess } from \"../PostProcesses/postProcess\";\r\nimport type { RenderTargetTexture } from \"../Materials/Textures/renderTargetTexture\";\r\nimport type { FreeCamera } from \"./freeCamera\";\r\nimport type { TargetCamera } from \"./targetCamera\";\r\nimport type { Ray } from \"../Culling/ray\";\r\n\r\n/**\r\n * This is the base class of all the camera used in the application.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras\r\n */\r\nexport class Camera extends Node {\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public static _CreateDefaultParsedCamera = (name: string, scene: Scene): Camera => {\r\n throw _WarnImport(\"UniversalCamera\");\r\n };\r\n\r\n /**\r\n * This is the default projection mode used by the cameras.\r\n * It helps recreating a feeling of perspective and better appreciate depth.\r\n * This is the best way to simulate real life cameras.\r\n */\r\n public static readonly PERSPECTIVE_CAMERA = Constants.PERSPECTIVE_CAMERA;\r\n /**\r\n * This helps creating camera with an orthographic mode.\r\n * Orthographic is commonly used in engineering as a means to produce object specifications that communicate dimensions unambiguously, each line of 1 unit length (cm, meter..whatever) will appear to have the same length everywhere on the drawing. This allows the drafter to dimension only a subset of lines and let the reader know that other lines of that length on the drawing are also that length in reality. Every parallel line in the drawing is also parallel in the object.\r\n */\r\n public static readonly ORTHOGRAPHIC_CAMERA = Constants.ORTHOGRAPHIC_CAMERA;\r\n\r\n /**\r\n * This is the default FOV mode for perspective cameras.\r\n * This setting aligns the upper and lower bounds of the viewport to the upper and lower bounds of the camera frustum.\r\n */\r\n public static readonly FOVMODE_VERTICAL_FIXED = Constants.FOVMODE_VERTICAL_FIXED;\r\n /**\r\n * This setting aligns the left and right bounds of the viewport to the left and right bounds of the camera frustum.\r\n */\r\n public static readonly FOVMODE_HORIZONTAL_FIXED = Constants.FOVMODE_HORIZONTAL_FIXED;\r\n\r\n /**\r\n * This specifies there is no need for a camera rig.\r\n * Basically only one eye is rendered corresponding to the camera.\r\n */\r\n public static readonly RIG_MODE_NONE = Constants.RIG_MODE_NONE;\r\n /**\r\n * Simulates a camera Rig with one blue eye and one red eye.\r\n * This can be use with 3d blue and red glasses.\r\n */\r\n public static readonly RIG_MODE_STEREOSCOPIC_ANAGLYPH = Constants.RIG_MODE_STEREOSCOPIC_ANAGLYPH;\r\n /**\r\n * Defines that both eyes of the camera will be rendered side by side with a parallel target.\r\n */\r\n public static readonly RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL = Constants.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL;\r\n /**\r\n * Defines that both eyes of the camera will be rendered side by side with a none parallel target.\r\n */\r\n public static readonly RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED = Constants.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED;\r\n /**\r\n * Defines that both eyes of the camera will be rendered over under each other.\r\n */\r\n public static readonly RIG_MODE_STEREOSCOPIC_OVERUNDER = Constants.RIG_MODE_STEREOSCOPIC_OVERUNDER;\r\n /**\r\n * Defines that both eyes of the camera will be rendered on successive lines interlaced for passive 3d monitors.\r\n */\r\n public static readonly RIG_MODE_STEREOSCOPIC_INTERLACED = Constants.RIG_MODE_STEREOSCOPIC_INTERLACED;\r\n /**\r\n * Defines that both eyes of the camera should be renderered in a VR mode (carbox).\r\n */\r\n public static readonly RIG_MODE_VR = Constants.RIG_MODE_VR;\r\n /**\r\n * Defines that both eyes of the camera should be renderered in a VR mode (webVR).\r\n */\r\n public static readonly RIG_MODE_WEBVR = Constants.RIG_MODE_WEBVR;\r\n /**\r\n * Custom rig mode allowing rig cameras to be populated manually with any number of cameras\r\n */\r\n public static readonly RIG_MODE_CUSTOM = Constants.RIG_MODE_CUSTOM;\r\n\r\n /**\r\n * Defines if by default attaching controls should prevent the default javascript event to continue.\r\n */\r\n public static ForceAttachControlToAlwaysPreventDefault = false;\r\n\r\n /**\r\n * Define the input manager associated with the camera.\r\n */\r\n public inputs: CameraInputsManager;\r\n\r\n /** @internal */\r\n @serializeAsVector3(\"position\")\r\n public _position = Vector3.Zero();\r\n\r\n /**\r\n * Define the current local position of the camera in the scene\r\n */\r\n public get position(): Vector3 {\r\n return this._position;\r\n }\r\n\r\n public set position(newPosition: Vector3) {\r\n this._position = newPosition;\r\n }\r\n\r\n @serializeAsVector3(\"upVector\")\r\n protected _upVector = Vector3.Up();\r\n\r\n /**\r\n * The vector the camera should consider as up.\r\n * (default is Vector3(0, 1, 0) aka Vector3.Up())\r\n */\r\n public set upVector(vec: Vector3) {\r\n this._upVector = vec;\r\n }\r\n\r\n public get upVector() {\r\n return this._upVector;\r\n }\r\n\r\n /**\r\n * The screen area in scene units squared\r\n */\r\n public get screenArea(): number {\r\n let x = 0;\r\n let y = 0;\r\n if (this.mode === Camera.PERSPECTIVE_CAMERA) {\r\n if (this.fovMode === Camera.FOVMODE_VERTICAL_FIXED) {\r\n y = this.minZ * 2 * Math.tan(this.fov / 2);\r\n x = this.getEngine().getAspectRatio(this) * y;\r\n } else {\r\n x = this.minZ * 2 * Math.tan(this.fov / 2);\r\n y = x / this.getEngine().getAspectRatio(this);\r\n }\r\n } else {\r\n const halfWidth = this.getEngine().getRenderWidth() / 2.0;\r\n const halfHeight = this.getEngine().getRenderHeight() / 2.0;\r\n\r\n x = (this.orthoRight ?? halfWidth) - (this.orthoLeft ?? -halfWidth);\r\n y = (this.orthoTop ?? halfHeight) - (this.orthoBottom ?? -halfHeight);\r\n }\r\n\r\n return x * y;\r\n }\r\n\r\n /**\r\n * Define the current limit on the left side for an orthographic camera\r\n * In scene unit\r\n */\r\n private _orthoLeft: Nullable = null;\r\n\r\n public set orthoLeft(value: Nullable) {\r\n this._orthoLeft = value;\r\n\r\n for (const rigCamera of this._rigCameras) {\r\n rigCamera.orthoLeft = value;\r\n }\r\n }\r\n\r\n @serialize()\r\n public get orthoLeft(): Nullable {\r\n return this._orthoLeft;\r\n }\r\n\r\n /**\r\n * Define the current limit on the right side for an orthographic camera\r\n * In scene unit\r\n */\r\n private _orthoRight: Nullable = null;\r\n\r\n public set orthoRight(value: Nullable) {\r\n this._orthoRight = value;\r\n\r\n for (const rigCamera of this._rigCameras) {\r\n rigCamera.orthoRight = value;\r\n }\r\n }\r\n\r\n @serialize()\r\n public get orthoRight(): Nullable {\r\n return this._orthoRight;\r\n }\r\n\r\n /**\r\n * Define the current limit on the bottom side for an orthographic camera\r\n * In scene unit\r\n */\r\n private _orthoBottom: Nullable = null;\r\n\r\n public set orthoBottom(value: Nullable) {\r\n this._orthoBottom = value;\r\n\r\n for (const rigCamera of this._rigCameras) {\r\n rigCamera.orthoBottom = value;\r\n }\r\n }\r\n\r\n @serialize()\r\n public get orthoBottom(): Nullable {\r\n return this._orthoBottom;\r\n }\r\n\r\n /**\r\n * Define the current limit on the top side for an orthographic camera\r\n * In scene unit\r\n */\r\n private _orthoTop: Nullable = null;\r\n\r\n public set orthoTop(value: Nullable) {\r\n this._orthoTop = value;\r\n\r\n for (const rigCamera of this._rigCameras) {\r\n rigCamera.orthoTop = value;\r\n }\r\n }\r\n\r\n @serialize()\r\n public get orthoTop(): Nullable {\r\n return this._orthoTop;\r\n }\r\n\r\n /**\r\n * Field Of View is set in Radians. (default is 0.8)\r\n */\r\n @serialize()\r\n public fov = 0.8;\r\n\r\n /**\r\n * Projection plane tilt around the X axis (horizontal), set in Radians. (default is 0)\r\n * Can be used to make vertical lines in world space actually vertical on the screen.\r\n * See https://forum.babylonjs.com/t/add-vertical-shift-to-3ds-max-exporter-babylon-cameras/17480\r\n */\r\n @serialize()\r\n public projectionPlaneTilt = 0;\r\n\r\n /**\r\n * Define the minimum distance the camera can see from.\r\n * This is important to note that the depth buffer are not infinite and the closer it starts\r\n * the more your scene might encounter depth fighting issue.\r\n */\r\n @serialize()\r\n public minZ = 1;\r\n\r\n /**\r\n * Define the maximum distance the camera can see to.\r\n * This is important to note that the depth buffer are not infinite and the further it end\r\n * the more your scene might encounter depth fighting issue.\r\n */\r\n @serialize()\r\n public maxZ = 10000.0;\r\n\r\n /**\r\n * Define the default inertia of the camera.\r\n * This helps giving a smooth feeling to the camera movement.\r\n */\r\n @serialize()\r\n public inertia = 0.9;\r\n\r\n /**\r\n * Define the mode of the camera (Camera.PERSPECTIVE_CAMERA or Camera.ORTHOGRAPHIC_CAMERA)\r\n */\r\n private _mode = Camera.PERSPECTIVE_CAMERA;\r\n set mode(mode: number) {\r\n this._mode = mode;\r\n\r\n // Pass the mode down to the rig cameras\r\n for (const rigCamera of this._rigCameras) {\r\n rigCamera.mode = mode;\r\n }\r\n }\r\n\r\n @serialize()\r\n get mode(): number {\r\n return this._mode;\r\n }\r\n\r\n /**\r\n * Define whether the camera is intermediate.\r\n * This is useful to not present the output directly to the screen in case of rig without post process for instance\r\n */\r\n public isIntermediate = false;\r\n\r\n /**\r\n * Define the viewport of the camera.\r\n * This correspond to the portion of the screen the camera will render to in normalized 0 to 1 unit.\r\n */\r\n public viewport = new Viewport(0, 0, 1.0, 1.0);\r\n\r\n /**\r\n * Restricts the camera to viewing objects with the same layerMask.\r\n * A camera with a layerMask of 1 will render mesh.layerMask & camera.layerMask!== 0\r\n */\r\n @serialize()\r\n public layerMask: number = 0x0fffffff;\r\n\r\n /**\r\n * fovMode sets the camera frustum bounds to the viewport bounds. (default is FOVMODE_VERTICAL_FIXED)\r\n */\r\n @serialize()\r\n public fovMode: number = Camera.FOVMODE_VERTICAL_FIXED;\r\n\r\n /**\r\n * Rig mode of the camera.\r\n * This is useful to create the camera with two \"eyes\" instead of one to create VR or stereoscopic scenes.\r\n * This is normally controlled byt the camera themselves as internal use.\r\n */\r\n @serialize()\r\n public cameraRigMode = Camera.RIG_MODE_NONE;\r\n\r\n /**\r\n * Defines the distance between both \"eyes\" in case of a RIG\r\n */\r\n @serialize()\r\n public interaxialDistance: number;\r\n\r\n /**\r\n * Defines if stereoscopic rendering is done side by side or over under.\r\n */\r\n @serialize()\r\n public isStereoscopicSideBySide: boolean;\r\n\r\n /**\r\n * Defines the list of custom render target which are rendered to and then used as the input to this camera's render. Eg. display another camera view on a TV in the main scene\r\n * This is pretty helpful if you wish to make a camera render to a texture you could reuse somewhere\r\n * else in the scene. (Eg. security camera)\r\n *\r\n * To change the final output target of the camera, camera.outputRenderTarget should be used instead (eg. webXR renders to a render target corresponding to an HMD)\r\n */\r\n public customRenderTargets = new Array();\r\n /**\r\n * When set, the camera will render to this render target instead of the default canvas\r\n *\r\n * If the desire is to use the output of a camera as a texture in the scene consider using camera.customRenderTargets instead\r\n */\r\n public outputRenderTarget: Nullable = null;\r\n\r\n /**\r\n * Observable triggered when the camera view matrix has changed.\r\n */\r\n public onViewMatrixChangedObservable = new Observable();\r\n /**\r\n * Observable triggered when the camera Projection matrix has changed.\r\n */\r\n public onProjectionMatrixChangedObservable = new Observable();\r\n /**\r\n * Observable triggered when the inputs have been processed.\r\n */\r\n public onAfterCheckInputsObservable = new Observable();\r\n /**\r\n * Observable triggered when reset has been called and applied to the camera.\r\n */\r\n public onRestoreStateObservable = new Observable();\r\n\r\n /**\r\n * Is this camera a part of a rig system?\r\n */\r\n public isRigCamera: boolean = false;\r\n\r\n /**\r\n * If isRigCamera set to true this will be set with the parent camera.\r\n * The parent camera is not (!) necessarily the .parent of this camera (like in the case of XR)\r\n */\r\n public rigParent?: Camera;\r\n\r\n /**\r\n * Render pass id used by the camera to render into the main framebuffer\r\n */\r\n public renderPassId: number;\r\n\r\n /** @internal */\r\n public _cameraRigParams: any;\r\n /** @internal */\r\n public _rigCameras = new Array();\r\n /** @internal */\r\n public _rigPostProcess: Nullable;\r\n\r\n protected _webvrViewMatrix = Matrix.Identity();\r\n /** @internal */\r\n public _skipRendering = false;\r\n\r\n /** @internal */\r\n public _projectionMatrix = new Matrix();\r\n\r\n /** @internal */\r\n public _postProcesses = new Array>();\r\n\r\n /** @internal */\r\n public _activeMeshes = new SmartArray(256);\r\n\r\n protected _globalPosition = Vector3.Zero();\r\n\r\n /** @internal */\r\n public _computedViewMatrix = Matrix.Identity();\r\n private _doNotComputeProjectionMatrix = false;\r\n private _transformMatrix = Matrix.Zero();\r\n private _frustumPlanes: Plane[];\r\n private _refreshFrustumPlanes = true;\r\n private _storedFov: number;\r\n private _stateStored: boolean;\r\n private _absoluteRotation: Quaternion = Quaternion.Identity();\r\n\r\n /**\r\n * Instantiates a new camera object.\r\n * This should not be used directly but through the inherited cameras: ArcRotate, Free...\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras\r\n * @param name Defines the name of the camera in the scene\r\n * @param position Defines the position of the camera\r\n * @param scene Defines the scene the camera belongs too\r\n * @param setActiveOnSceneIfNoneActive Defines if the camera should be set as active after creation if no other camera have been defined in the scene\r\n */\r\n constructor(name: string, position: Vector3, scene?: Scene, setActiveOnSceneIfNoneActive = true) {\r\n super(name, scene);\r\n\r\n this.getScene().addCamera(this);\r\n\r\n if (setActiveOnSceneIfNoneActive && !this.getScene().activeCamera) {\r\n this.getScene().activeCamera = this;\r\n }\r\n\r\n this.position = position;\r\n this.renderPassId = this.getScene().getEngine().createRenderPassId(`Camera ${name}`);\r\n }\r\n\r\n /**\r\n * Store current camera state (fov, position, etc..)\r\n * @returns the camera\r\n */\r\n public storeState(): Camera {\r\n this._stateStored = true;\r\n this._storedFov = this.fov;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Restores the camera state values if it has been stored. You must call storeState() first\r\n */\r\n protected _restoreStateValues(): boolean {\r\n if (!this._stateStored) {\r\n return false;\r\n }\r\n\r\n this.fov = this._storedFov;\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Restored camera state. You must call storeState() first.\r\n * @returns true if restored and false otherwise\r\n */\r\n public restoreState(): boolean {\r\n if (this._restoreStateValues()) {\r\n this.onRestoreStateObservable.notifyObservers(this);\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Gets the class name of the camera.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"Camera\";\r\n }\r\n\r\n /** @internal */\r\n public readonly _isCamera = true;\r\n\r\n /**\r\n * Gets a string representation of the camera useful for debug purpose.\r\n * @param fullDetails Defines that a more verbose level of logging is required\r\n * @returns the string representation\r\n */\r\n public toString(fullDetails?: boolean): string {\r\n let ret = \"Name: \" + this.name;\r\n ret += \", type: \" + this.getClassName();\r\n if (this.animations) {\r\n for (let i = 0; i < this.animations.length; i++) {\r\n ret += \", animation[0]: \" + this.animations[i].toString(fullDetails);\r\n }\r\n }\r\n return ret;\r\n }\r\n\r\n /**\r\n * Automatically tilts the projection plane, using `projectionPlaneTilt`, to correct the perspective effect on vertical lines.\r\n */\r\n public applyVerticalCorrection() {\r\n const rot = this.absoluteRotation.toEulerAngles();\r\n\r\n this.projectionPlaneTilt = this._scene.useRightHandedSystem ? -rot.x : rot.x;\r\n }\r\n\r\n /**\r\n * Gets the current world space position of the camera.\r\n */\r\n public get globalPosition(): Vector3 {\r\n return this._globalPosition;\r\n }\r\n\r\n /**\r\n * Gets the list of active meshes this frame (meshes no culled or excluded by lod s in the frame)\r\n * @returns the active meshe list\r\n */\r\n public getActiveMeshes(): SmartArray {\r\n return this._activeMeshes;\r\n }\r\n\r\n /**\r\n * Check whether a mesh is part of the current active mesh list of the camera\r\n * @param mesh Defines the mesh to check\r\n * @returns true if active, false otherwise\r\n */\r\n public isActiveMesh(mesh: Mesh): boolean {\r\n return this._activeMeshes.indexOf(mesh) !== -1;\r\n }\r\n\r\n /**\r\n * Is this camera ready to be used/rendered\r\n * @param completeCheck defines if a complete check (including post processes) has to be done (false by default)\r\n * @returns true if the camera is ready\r\n */\r\n public isReady(completeCheck = false): boolean {\r\n if (completeCheck) {\r\n for (const pp of this._postProcesses) {\r\n if (pp && !pp.isReady()) {\r\n return false;\r\n }\r\n }\r\n }\r\n return super.isReady(completeCheck);\r\n }\r\n\r\n /** @internal */\r\n public _initCache() {\r\n super._initCache();\r\n\r\n this._cache.position = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\r\n this._cache.upVector = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\r\n\r\n this._cache.mode = undefined;\r\n this._cache.minZ = undefined;\r\n this._cache.maxZ = undefined;\r\n\r\n this._cache.fov = undefined;\r\n this._cache.fovMode = undefined;\r\n this._cache.aspectRatio = undefined;\r\n\r\n this._cache.orthoLeft = undefined;\r\n this._cache.orthoRight = undefined;\r\n this._cache.orthoBottom = undefined;\r\n this._cache.orthoTop = undefined;\r\n this._cache.renderWidth = undefined;\r\n this._cache.renderHeight = undefined;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _updateCache(ignoreParentClass?: boolean): void {\r\n if (!ignoreParentClass) {\r\n super._updateCache();\r\n }\r\n\r\n this._cache.position.copyFrom(this.position);\r\n this._cache.upVector.copyFrom(this.upVector);\r\n }\r\n\r\n /** @internal */\r\n public _isSynchronized(): boolean {\r\n return this._isSynchronizedViewMatrix() && this._isSynchronizedProjectionMatrix();\r\n }\r\n\r\n /** @internal */\r\n public _isSynchronizedViewMatrix(): boolean {\r\n if (!super._isSynchronized()) {\r\n return false;\r\n }\r\n\r\n return this._cache.position.equals(this.position) && this._cache.upVector.equals(this.upVector) && this.isSynchronizedWithParent();\r\n }\r\n\r\n /** @internal */\r\n public _isSynchronizedProjectionMatrix(): boolean {\r\n let check = this._cache.mode === this.mode && this._cache.minZ === this.minZ && this._cache.maxZ === this.maxZ;\r\n\r\n if (!check) {\r\n return false;\r\n }\r\n\r\n const engine = this.getEngine();\r\n\r\n if (this.mode === Camera.PERSPECTIVE_CAMERA) {\r\n check =\r\n this._cache.fov === this.fov &&\r\n this._cache.fovMode === this.fovMode &&\r\n this._cache.aspectRatio === engine.getAspectRatio(this) &&\r\n this._cache.projectionPlaneTilt === this.projectionPlaneTilt;\r\n } else {\r\n check =\r\n this._cache.orthoLeft === this.orthoLeft &&\r\n this._cache.orthoRight === this.orthoRight &&\r\n this._cache.orthoBottom === this.orthoBottom &&\r\n this._cache.orthoTop === this.orthoTop &&\r\n this._cache.renderWidth === engine.getRenderWidth() &&\r\n this._cache.renderHeight === engine.getRenderHeight();\r\n }\r\n\r\n return check;\r\n }\r\n\r\n /**\r\n * Attach the input controls to a specific dom element to get the input from.\r\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n */\r\n public attachControl(noPreventDefault?: boolean): void;\r\n /**\r\n * Attach the input controls to a specific dom element to get the input from.\r\n * @param ignored defines an ignored parameter kept for backward compatibility.\r\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n * BACK COMPAT SIGNATURE ONLY.\r\n */\r\n public attachControl(ignored: any, noPreventDefault?: boolean): void;\r\n /**\r\n * Attach the input controls to a specific dom element to get the input from.\r\n * This function is here because typescript removes the typing of the last function.\r\n * @param _ignored defines an ignored parameter kept for backward compatibility.\r\n * @param _noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n */\r\n public attachControl(_ignored?: any, _noPreventDefault?: boolean): void {}\r\n\r\n /**\r\n * Detach the current controls from the specified dom element.\r\n */\r\n public detachControl(): void;\r\n /**\r\n * Detach the current controls from the specified dom element.\r\n * @param ignored defines an ignored parameter kept for backward compatibility.\r\n */\r\n public detachControl(ignored?: any): void;\r\n /**\r\n * Detach the current controls from the specified dom element.\r\n * This function is here because typescript removes the typing of the last function.\r\n * @param _ignored defines an ignored parameter kept for backward compatibility.\r\n */\r\n public detachControl(_ignored?: any): void {}\r\n\r\n /**\r\n * Update the camera state according to the different inputs gathered during the frame.\r\n */\r\n public update(): void {\r\n this._checkInputs();\r\n if (this.cameraRigMode !== Camera.RIG_MODE_NONE) {\r\n this._updateRigCameras();\r\n }\r\n\r\n // Attempt to update the camera's view and projection matrices.\r\n // This call is being made because these matrices are no longer being updated\r\n // as a part of the picking ray process (in addition to scene.render).\r\n this.getViewMatrix();\r\n this.getProjectionMatrix();\r\n }\r\n\r\n /** @internal */\r\n public _checkInputs(): void {\r\n this.onAfterCheckInputsObservable.notifyObservers(this);\r\n }\r\n\r\n /** @internal */\r\n public get rigCameras(): Camera[] {\r\n return this._rigCameras;\r\n }\r\n\r\n /**\r\n * Gets the post process used by the rig cameras\r\n */\r\n public get rigPostProcess(): Nullable {\r\n return this._rigPostProcess;\r\n }\r\n\r\n /**\r\n * Internal, gets the first post process.\r\n * @returns the first post process to be run on this camera.\r\n */\r\n public _getFirstPostProcess(): Nullable {\r\n for (let ppIndex = 0; ppIndex < this._postProcesses.length; ppIndex++) {\r\n if (this._postProcesses[ppIndex] !== null) {\r\n return this._postProcesses[ppIndex];\r\n }\r\n }\r\n return null;\r\n }\r\n\r\n private _cascadePostProcessesToRigCams(): void {\r\n // invalidate framebuffer\r\n const firstPostProcess = this._getFirstPostProcess();\r\n if (firstPostProcess) {\r\n firstPostProcess.markTextureDirty();\r\n }\r\n\r\n // glue the rigPostProcess to the end of the user postprocesses & assign to each sub-camera\r\n for (let i = 0, len = this._rigCameras.length; i < len; i++) {\r\n const cam = this._rigCameras[i];\r\n const rigPostProcess = cam._rigPostProcess;\r\n\r\n // for VR rig, there does not have to be a post process\r\n if (rigPostProcess) {\r\n const isPass = rigPostProcess.getEffectName() === \"pass\";\r\n if (isPass) {\r\n // any rig which has a PassPostProcess for rig[0], cannot be isIntermediate when there are also user postProcesses\r\n cam.isIntermediate = this._postProcesses.length === 0;\r\n }\r\n cam._postProcesses = this._postProcesses.slice(0).concat(rigPostProcess);\r\n rigPostProcess.markTextureDirty();\r\n } else {\r\n cam._postProcesses = this._postProcesses.slice(0);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Attach a post process to the camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/usePostProcesses#attach-postprocess\r\n * @param postProcess The post process to attach to the camera\r\n * @param insertAt The position of the post process in case several of them are in use in the scene\r\n * @returns the position the post process has been inserted at\r\n */\r\n public attachPostProcess(postProcess: PostProcess, insertAt: Nullable = null): number {\r\n if (!postProcess.isReusable() && this._postProcesses.indexOf(postProcess) > -1) {\r\n Logger.Error(\"You're trying to reuse a post process not defined as reusable.\");\r\n return 0;\r\n }\r\n\r\n if (insertAt == null || insertAt < 0) {\r\n this._postProcesses.push(postProcess);\r\n } else if (this._postProcesses[insertAt] === null) {\r\n this._postProcesses[insertAt] = postProcess;\r\n } else {\r\n this._postProcesses.splice(insertAt, 0, postProcess);\r\n }\r\n this._cascadePostProcessesToRigCams(); // also ensures framebuffer invalidated\r\n\r\n // Update prePass\r\n if (this._scene.prePassRenderer) {\r\n this._scene.prePassRenderer.markAsDirty();\r\n }\r\n\r\n return this._postProcesses.indexOf(postProcess);\r\n }\r\n\r\n /**\r\n * Detach a post process to the camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/usePostProcesses#attach-postprocess\r\n * @param postProcess The post process to detach from the camera\r\n */\r\n public detachPostProcess(postProcess: PostProcess): void {\r\n const idx = this._postProcesses.indexOf(postProcess);\r\n if (idx !== -1) {\r\n this._postProcesses[idx] = null;\r\n }\r\n\r\n // Update prePass\r\n if (this._scene.prePassRenderer) {\r\n this._scene.prePassRenderer.markAsDirty();\r\n }\r\n\r\n this._cascadePostProcessesToRigCams(); // also ensures framebuffer invalidated\r\n }\r\n\r\n /**\r\n * Gets the current world matrix of the camera\r\n */\r\n public getWorldMatrix(): Matrix {\r\n if (this._isSynchronizedViewMatrix()) {\r\n return this._worldMatrix;\r\n }\r\n\r\n // Getting the the view matrix will also compute the world matrix.\r\n this.getViewMatrix();\r\n\r\n return this._worldMatrix;\r\n }\r\n\r\n /** @internal */\r\n public _getViewMatrix(): Matrix {\r\n return Matrix.Identity();\r\n }\r\n\r\n /**\r\n * Gets the current view matrix of the camera.\r\n * @param force forces the camera to recompute the matrix without looking at the cached state\r\n * @returns the view matrix\r\n */\r\n public getViewMatrix(force?: boolean): Matrix {\r\n if (!force && this._isSynchronizedViewMatrix()) {\r\n return this._computedViewMatrix;\r\n }\r\n\r\n this.updateCache();\r\n this._computedViewMatrix = this._getViewMatrix();\r\n this._currentRenderId = this.getScene().getRenderId();\r\n this._childUpdateId++;\r\n\r\n this._refreshFrustumPlanes = true;\r\n\r\n if (this._cameraRigParams && this._cameraRigParams.vrPreViewMatrix) {\r\n this._computedViewMatrix.multiplyToRef(this._cameraRigParams.vrPreViewMatrix, this._computedViewMatrix);\r\n }\r\n\r\n // Notify parent camera if rig camera is changed\r\n if (this.parent && (this.parent as Camera).onViewMatrixChangedObservable) {\r\n (this.parent as Camera).onViewMatrixChangedObservable.notifyObservers(this.parent as Camera);\r\n }\r\n\r\n this.onViewMatrixChangedObservable.notifyObservers(this);\r\n\r\n this._computedViewMatrix.invertToRef(this._worldMatrix);\r\n\r\n return this._computedViewMatrix;\r\n }\r\n\r\n /**\r\n * Freeze the projection matrix.\r\n * It will prevent the cache check of the camera projection compute and can speed up perf\r\n * if no parameter of the camera are meant to change\r\n * @param projection Defines manually a projection if necessary\r\n */\r\n public freezeProjectionMatrix(projection?: Matrix): void {\r\n this._doNotComputeProjectionMatrix = true;\r\n if (projection !== undefined) {\r\n this._projectionMatrix = projection;\r\n }\r\n }\r\n\r\n /**\r\n * Unfreeze the projection matrix if it has previously been freezed by freezeProjectionMatrix.\r\n */\r\n public unfreezeProjectionMatrix(): void {\r\n this._doNotComputeProjectionMatrix = false;\r\n }\r\n\r\n /**\r\n * Gets the current projection matrix of the camera.\r\n * @param force forces the camera to recompute the matrix without looking at the cached state\r\n * @returns the projection matrix\r\n */\r\n public getProjectionMatrix(force?: boolean): Matrix {\r\n if (this._doNotComputeProjectionMatrix || (!force && this._isSynchronizedProjectionMatrix())) {\r\n return this._projectionMatrix;\r\n }\r\n\r\n // Cache\r\n this._cache.mode = this.mode;\r\n this._cache.minZ = this.minZ;\r\n this._cache.maxZ = this.maxZ;\r\n\r\n // Matrix\r\n this._refreshFrustumPlanes = true;\r\n\r\n const engine = this.getEngine();\r\n const scene = this.getScene();\r\n const reverseDepth = engine.useReverseDepthBuffer;\r\n if (this.mode === Camera.PERSPECTIVE_CAMERA) {\r\n this._cache.fov = this.fov;\r\n this._cache.fovMode = this.fovMode;\r\n this._cache.aspectRatio = engine.getAspectRatio(this);\r\n this._cache.projectionPlaneTilt = this.projectionPlaneTilt;\r\n\r\n if (this.minZ <= 0) {\r\n this.minZ = 0.1;\r\n }\r\n\r\n let getProjectionMatrix: (\r\n fov: number,\r\n aspect: number,\r\n znear: number,\r\n zfar: number,\r\n result: Matrix,\r\n isVerticalFovFixed: boolean,\r\n halfZRange: boolean,\r\n projectionPlaneTilt: number,\r\n reverseDepthBufferMode: boolean\r\n ) => void;\r\n if (scene.useRightHandedSystem) {\r\n getProjectionMatrix = Matrix.PerspectiveFovRHToRef;\r\n } else {\r\n getProjectionMatrix = Matrix.PerspectiveFovLHToRef;\r\n }\r\n\r\n getProjectionMatrix(\r\n this.fov,\r\n engine.getAspectRatio(this),\r\n reverseDepth ? this.maxZ : this.minZ,\r\n reverseDepth ? this.minZ : this.maxZ,\r\n this._projectionMatrix,\r\n this.fovMode === Camera.FOVMODE_VERTICAL_FIXED,\r\n engine.isNDCHalfZRange,\r\n this.projectionPlaneTilt,\r\n reverseDepth\r\n );\r\n } else {\r\n const halfWidth = engine.getRenderWidth() / 2.0;\r\n const halfHeight = engine.getRenderHeight() / 2.0;\r\n if (scene.useRightHandedSystem) {\r\n Matrix.OrthoOffCenterRHToRef(\r\n this.orthoLeft ?? -halfWidth,\r\n this.orthoRight ?? halfWidth,\r\n this.orthoBottom ?? -halfHeight,\r\n this.orthoTop ?? halfHeight,\r\n reverseDepth ? this.maxZ : this.minZ,\r\n reverseDepth ? this.minZ : this.maxZ,\r\n this._projectionMatrix,\r\n engine.isNDCHalfZRange\r\n );\r\n } else {\r\n Matrix.OrthoOffCenterLHToRef(\r\n this.orthoLeft ?? -halfWidth,\r\n this.orthoRight ?? halfWidth,\r\n this.orthoBottom ?? -halfHeight,\r\n this.orthoTop ?? halfHeight,\r\n reverseDepth ? this.maxZ : this.minZ,\r\n reverseDepth ? this.minZ : this.maxZ,\r\n this._projectionMatrix,\r\n engine.isNDCHalfZRange\r\n );\r\n }\r\n\r\n this._cache.orthoLeft = this.orthoLeft;\r\n this._cache.orthoRight = this.orthoRight;\r\n this._cache.orthoBottom = this.orthoBottom;\r\n this._cache.orthoTop = this.orthoTop;\r\n this._cache.renderWidth = engine.getRenderWidth();\r\n this._cache.renderHeight = engine.getRenderHeight();\r\n }\r\n\r\n this.onProjectionMatrixChangedObservable.notifyObservers(this);\r\n\r\n return this._projectionMatrix;\r\n }\r\n\r\n /**\r\n * Gets the transformation matrix (ie. the multiplication of view by projection matrices)\r\n * @returns a Matrix\r\n */\r\n public getTransformationMatrix(): Matrix {\r\n this._computedViewMatrix.multiplyToRef(this._projectionMatrix, this._transformMatrix);\r\n return this._transformMatrix;\r\n }\r\n\r\n private _updateFrustumPlanes(): void {\r\n if (!this._refreshFrustumPlanes) {\r\n return;\r\n }\r\n\r\n this.getTransformationMatrix();\r\n\r\n if (!this._frustumPlanes) {\r\n this._frustumPlanes = Frustum.GetPlanes(this._transformMatrix);\r\n } else {\r\n Frustum.GetPlanesToRef(this._transformMatrix, this._frustumPlanes);\r\n }\r\n\r\n this._refreshFrustumPlanes = false;\r\n }\r\n\r\n /**\r\n * Checks if a cullable object (mesh...) is in the camera frustum\r\n * This checks the bounding box center. See isCompletelyInFrustum for a full bounding check\r\n * @param target The object to check\r\n * @param checkRigCameras If the rig cameras should be checked (eg. with webVR camera both eyes should be checked) (Default: false)\r\n * @returns true if the object is in frustum otherwise false\r\n */\r\n public isInFrustum(target: ICullable, checkRigCameras = false): boolean {\r\n this._updateFrustumPlanes();\r\n\r\n if (checkRigCameras && this.rigCameras.length > 0) {\r\n let result = false;\r\n this.rigCameras.forEach((cam) => {\r\n cam._updateFrustumPlanes();\r\n result = result || target.isInFrustum(cam._frustumPlanes);\r\n });\r\n return result;\r\n } else {\r\n return target.isInFrustum(this._frustumPlanes);\r\n }\r\n }\r\n\r\n /**\r\n * Checks if a cullable object (mesh...) is in the camera frustum\r\n * Unlike isInFrustum this checks the full bounding box\r\n * @param target The object to check\r\n * @returns true if the object is in frustum otherwise false\r\n */\r\n public isCompletelyInFrustum(target: ICullable): boolean {\r\n this._updateFrustumPlanes();\r\n\r\n return target.isCompletelyInFrustum(this._frustumPlanes);\r\n }\r\n\r\n /**\r\n * Gets a ray in the forward direction from the camera.\r\n * @param length Defines the length of the ray to create\r\n * @param transform Defines the transform to apply to the ray, by default the world matrix is used to create a workd space ray\r\n * @param origin Defines the start point of the ray which defaults to the camera position\r\n * @returns the forward ray\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public getForwardRay(length = 100, transform?: Matrix, origin?: Vector3): Ray {\r\n throw _WarnImport(\"Ray\");\r\n }\r\n\r\n /**\r\n * Gets a ray in the forward direction from the camera.\r\n * @param refRay the ray to (re)use when setting the values\r\n * @param length Defines the length of the ray to create\r\n * @param transform Defines the transform to apply to the ray, by default the world matrx is used to create a workd space ray\r\n * @param origin Defines the start point of the ray which defaults to the camera position\r\n * @returns the forward ray\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public getForwardRayToRef(refRay: Ray, length = 100, transform?: Matrix, origin?: Vector3): Ray {\r\n throw _WarnImport(\"Ray\");\r\n }\r\n\r\n /**\r\n * Releases resources associated with this node.\r\n * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default)\r\n * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default)\r\n */\r\n public dispose(doNotRecurse?: boolean, disposeMaterialAndTextures = false): void {\r\n // Observables\r\n this.onViewMatrixChangedObservable.clear();\r\n this.onProjectionMatrixChangedObservable.clear();\r\n this.onAfterCheckInputsObservable.clear();\r\n this.onRestoreStateObservable.clear();\r\n\r\n // Inputs\r\n if (this.inputs) {\r\n this.inputs.clear();\r\n }\r\n\r\n // Animations\r\n this.getScene().stopAnimation(this);\r\n\r\n // Remove from scene\r\n this.getScene().removeCamera(this);\r\n while (this._rigCameras.length > 0) {\r\n const camera = this._rigCameras.pop();\r\n if (camera) {\r\n camera.dispose();\r\n }\r\n }\r\n\r\n if (this._parentContainer) {\r\n const index = this._parentContainer.cameras.indexOf(this);\r\n if (index > -1) {\r\n this._parentContainer.cameras.splice(index, 1);\r\n }\r\n this._parentContainer = null;\r\n }\r\n\r\n // Postprocesses\r\n if (this._rigPostProcess) {\r\n this._rigPostProcess.dispose(this);\r\n this._rigPostProcess = null;\r\n this._postProcesses.length = 0;\r\n } else if (this.cameraRigMode !== Camera.RIG_MODE_NONE) {\r\n this._rigPostProcess = null;\r\n this._postProcesses.length = 0;\r\n } else {\r\n let i = this._postProcesses.length;\r\n while (--i >= 0) {\r\n const postProcess = this._postProcesses[i];\r\n if (postProcess) {\r\n postProcess.dispose(this);\r\n }\r\n }\r\n }\r\n\r\n // Render targets\r\n let i = this.customRenderTargets.length;\r\n while (--i >= 0) {\r\n this.customRenderTargets[i].dispose();\r\n }\r\n this.customRenderTargets.length = 0;\r\n\r\n // Active Meshes\r\n this._activeMeshes.dispose();\r\n\r\n this.getScene().getEngine().releaseRenderPassId(this.renderPassId);\r\n\r\n super.dispose(doNotRecurse, disposeMaterialAndTextures);\r\n }\r\n\r\n /** @internal */\r\n public _isLeftCamera = false;\r\n /**\r\n * Gets the left camera of a rig setup in case of Rigged Camera\r\n */\r\n public get isLeftCamera(): boolean {\r\n return this._isLeftCamera;\r\n }\r\n\r\n /** @internal */\r\n public _isRightCamera = false;\r\n /**\r\n * Gets the right camera of a rig setup in case of Rigged Camera\r\n */\r\n public get isRightCamera(): boolean {\r\n return this._isRightCamera;\r\n }\r\n\r\n /**\r\n * Gets the left camera of a rig setup in case of Rigged Camera\r\n */\r\n public get leftCamera(): Nullable {\r\n if (this._rigCameras.length < 1) {\r\n return null;\r\n }\r\n return this._rigCameras[0];\r\n }\r\n\r\n /**\r\n * Gets the right camera of a rig setup in case of Rigged Camera\r\n */\r\n public get rightCamera(): Nullable {\r\n if (this._rigCameras.length < 2) {\r\n return null;\r\n }\r\n return this._rigCameras[1];\r\n }\r\n\r\n /**\r\n * Gets the left camera target of a rig setup in case of Rigged Camera\r\n * @returns the target position\r\n */\r\n public getLeftTarget(): Nullable {\r\n if (this._rigCameras.length < 1) {\r\n return null;\r\n }\r\n return (this._rigCameras[0]).getTarget();\r\n }\r\n\r\n /**\r\n * Gets the right camera target of a rig setup in case of Rigged Camera\r\n * @returns the target position\r\n */\r\n public getRightTarget(): Nullable {\r\n if (this._rigCameras.length < 2) {\r\n return null;\r\n }\r\n return (this._rigCameras[1]).getTarget();\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public setCameraRigMode(mode: number, rigParams: any): void {\r\n if (this.cameraRigMode === mode) {\r\n return;\r\n }\r\n\r\n while (this._rigCameras.length > 0) {\r\n const camera = this._rigCameras.pop();\r\n\r\n if (camera) {\r\n camera.dispose();\r\n }\r\n }\r\n this.cameraRigMode = mode;\r\n this._cameraRigParams = {};\r\n //we have to implement stereo camera calcultating left and right viewpoints from interaxialDistance and target,\r\n //not from a given angle as it is now, but until that complete code rewriting provisional stereoHalfAngle value is introduced\r\n this._cameraRigParams.interaxialDistance = rigParams.interaxialDistance || 0.0637;\r\n this._cameraRigParams.stereoHalfAngle = Tools.ToRadians(this._cameraRigParams.interaxialDistance / 0.0637);\r\n\r\n // create the rig cameras, unless none\r\n if (this.cameraRigMode !== Camera.RIG_MODE_NONE) {\r\n const leftCamera = this.createRigCamera(this.name + \"_L\", 0);\r\n if (leftCamera) {\r\n leftCamera._isLeftCamera = true;\r\n }\r\n const rightCamera = this.createRigCamera(this.name + \"_R\", 1);\r\n if (rightCamera) {\r\n rightCamera._isRightCamera = true;\r\n }\r\n if (leftCamera && rightCamera) {\r\n this._rigCameras.push(leftCamera);\r\n this._rigCameras.push(rightCamera);\r\n }\r\n }\r\n\r\n this._setRigMode(rigParams);\r\n\r\n this._cascadePostProcessesToRigCams();\r\n this.update();\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n protected _setRigMode(rigParams: any) {\r\n // no-op\r\n }\r\n\r\n /** @internal */\r\n public _getVRProjectionMatrix(): Matrix {\r\n Matrix.PerspectiveFovLHToRef(\r\n this._cameraRigParams.vrMetrics.aspectRatioFov,\r\n this._cameraRigParams.vrMetrics.aspectRatio,\r\n this.minZ,\r\n this.maxZ,\r\n this._cameraRigParams.vrWorkMatrix,\r\n true,\r\n this.getEngine().isNDCHalfZRange\r\n );\r\n this._cameraRigParams.vrWorkMatrix.multiplyToRef(this._cameraRigParams.vrHMatrix, this._projectionMatrix);\r\n return this._projectionMatrix;\r\n }\r\n\r\n protected _updateCameraRotationMatrix() {\r\n //Here for WebVR\r\n }\r\n\r\n protected _updateWebVRCameraRotationMatrix() {\r\n //Here for WebVR\r\n }\r\n\r\n /**\r\n * This function MUST be overwritten by the different WebVR cameras available.\r\n * The context in which it is running is the RIG camera. So 'this' is the TargetCamera, left or right.\r\n * @internal\r\n */\r\n public _getWebVRProjectionMatrix(): Matrix {\r\n return Matrix.Identity();\r\n }\r\n\r\n /**\r\n * This function MUST be overwritten by the different WebVR cameras available.\r\n * The context in which it is running is the RIG camera. So 'this' is the TargetCamera, left or right.\r\n * @internal\r\n */\r\n public _getWebVRViewMatrix(): Matrix {\r\n return Matrix.Identity();\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public setCameraRigParameter(name: string, value: any) {\r\n if (!this._cameraRigParams) {\r\n this._cameraRigParams = {};\r\n }\r\n this._cameraRigParams[name] = value;\r\n //provisionnally:\r\n if (name === \"interaxialDistance\") {\r\n this._cameraRigParams.stereoHalfAngle = Tools.ToRadians(value / 0.0637);\r\n }\r\n }\r\n\r\n /**\r\n * needs to be overridden by children so sub has required properties to be copied\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public createRigCamera(name: string, cameraIndex: number): Nullable {\r\n return null;\r\n }\r\n\r\n /**\r\n * May need to be overridden by children\r\n * @internal\r\n */\r\n public _updateRigCameras() {\r\n for (let i = 0; i < this._rigCameras.length; i++) {\r\n this._rigCameras[i].minZ = this.minZ;\r\n this._rigCameras[i].maxZ = this.maxZ;\r\n this._rigCameras[i].fov = this.fov;\r\n this._rigCameras[i].upVector.copyFrom(this.upVector);\r\n }\r\n\r\n // only update viewport when ANAGLYPH\r\n if (this.cameraRigMode === Camera.RIG_MODE_STEREOSCOPIC_ANAGLYPH) {\r\n this._rigCameras[0].viewport = this._rigCameras[1].viewport = this.viewport;\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _setupInputs() {}\r\n\r\n /**\r\n * Serialiaze the camera setup to a json representation\r\n * @returns the JSON representation\r\n */\r\n public serialize(): any {\r\n const serializationObject = SerializationHelper.Serialize(this);\r\n serializationObject.uniqueId = this.uniqueId;\r\n\r\n // Type\r\n serializationObject.type = this.getClassName();\r\n\r\n // Parent\r\n if (this.parent) {\r\n this.parent._serializeAsParent(serializationObject);\r\n }\r\n\r\n if (this.inputs) {\r\n this.inputs.serialize(serializationObject);\r\n }\r\n // Animations\r\n SerializationHelper.AppendSerializedAnimations(this, serializationObject);\r\n serializationObject.ranges = this.serializeAnimationRanges();\r\n\r\n serializationObject.isEnabled = this.isEnabled();\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Clones the current camera.\r\n * @param name The cloned camera name\r\n * @param newParent The cloned camera's new parent (none by default)\r\n * @returns the cloned camera\r\n */\r\n public clone(name: string, newParent: Nullable = null): Camera {\r\n const camera = SerializationHelper.Clone(\r\n Camera.GetConstructorFromName(this.getClassName(), name, this.getScene(), this.interaxialDistance, this.isStereoscopicSideBySide),\r\n this\r\n );\r\n camera.name = name;\r\n camera.parent = newParent;\r\n\r\n this.onClonedObservable.notifyObservers(camera);\r\n\r\n return camera;\r\n }\r\n\r\n /**\r\n * Gets the direction of the camera relative to a given local axis.\r\n * @param localAxis Defines the reference axis to provide a relative direction.\r\n * @returns the direction\r\n */\r\n public getDirection(localAxis: Vector3): Vector3 {\r\n const result = Vector3.Zero();\r\n\r\n this.getDirectionToRef(localAxis, result);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Returns the current camera absolute rotation\r\n */\r\n public get absoluteRotation(): Quaternion {\r\n this.getWorldMatrix().decompose(undefined, this._absoluteRotation);\r\n\r\n return this._absoluteRotation;\r\n }\r\n\r\n /**\r\n * Gets the direction of the camera relative to a given local axis into a passed vector.\r\n * @param localAxis Defines the reference axis to provide a relative direction.\r\n * @param result Defines the vector to store the result in\r\n */\r\n public getDirectionToRef(localAxis: Vector3, result: Vector3): void {\r\n Vector3.TransformNormalToRef(localAxis, this.getWorldMatrix(), result);\r\n }\r\n\r\n /**\r\n * Gets a camera constructor for a given camera type\r\n * @param type The type of the camera to construct (should be equal to one of the camera class name)\r\n * @param name The name of the camera the result will be able to instantiate\r\n * @param scene The scene the result will construct the camera in\r\n * @param interaxial_distance In case of stereoscopic setup, the distance between both eyes\r\n * @param isStereoscopicSideBySide In case of stereoscopic setup, should the sereo be side b side\r\n * @returns a factory method to construct the camera\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n static GetConstructorFromName(type: string, name: string, scene: Scene, interaxial_distance: number = 0, isStereoscopicSideBySide: boolean = true): () => Camera {\r\n const constructorFunc = Node.Construct(type, name, scene, {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n interaxial_distance: interaxial_distance,\r\n isStereoscopicSideBySide: isStereoscopicSideBySide,\r\n });\r\n\r\n if (constructorFunc) {\r\n return <() => Camera>constructorFunc;\r\n }\r\n\r\n // Default to universal camera\r\n return () => Camera._CreateDefaultParsedCamera(name, scene);\r\n }\r\n\r\n /**\r\n * Compute the world matrix of the camera.\r\n * @returns the camera world matrix\r\n */\r\n public computeWorldMatrix(): Matrix {\r\n return this.getWorldMatrix();\r\n }\r\n\r\n /**\r\n * Parse a JSON and creates the camera from the parsed information\r\n * @param parsedCamera The JSON to parse\r\n * @param scene The scene to instantiate the camera in\r\n * @returns the newly constructed camera\r\n */\r\n public static Parse(parsedCamera: any, scene: Scene): Camera {\r\n const type = parsedCamera.type;\r\n const construct = Camera.GetConstructorFromName(type, parsedCamera.name, scene, parsedCamera.interaxial_distance, parsedCamera.isStereoscopicSideBySide);\r\n\r\n const camera = SerializationHelper.Parse(construct, parsedCamera, scene);\r\n\r\n // Parent\r\n if (parsedCamera.parentId !== undefined) {\r\n camera._waitingParentId = parsedCamera.parentId;\r\n }\r\n\r\n // Parent instance index\r\n if (parsedCamera.parentInstanceIndex !== undefined) {\r\n camera._waitingParentInstanceIndex = parsedCamera.parentInstanceIndex;\r\n }\r\n\r\n //If camera has an input manager, let it parse inputs settings\r\n if (camera.inputs) {\r\n camera.inputs.parse(parsedCamera);\r\n\r\n camera._setupInputs();\r\n }\r\n\r\n if (parsedCamera.upVector) {\r\n camera.upVector = Vector3.FromArray(parsedCamera.upVector); // need to force the upVector\r\n }\r\n\r\n if ((camera).setPosition) {\r\n // need to force position\r\n camera.position.copyFromFloats(0, 0, 0);\r\n (camera).setPosition(Vector3.FromArray(parsedCamera.position));\r\n }\r\n\r\n // Target\r\n if (parsedCamera.target) {\r\n if ((camera).setTarget) {\r\n (camera).setTarget(Vector3.FromArray(parsedCamera.target));\r\n }\r\n }\r\n\r\n // Apply 3d rig, when found\r\n if (parsedCamera.cameraRigMode) {\r\n const rigParams = parsedCamera.interaxial_distance ? { interaxialDistance: parsedCamera.interaxial_distance } : {};\r\n camera.setCameraRigMode(parsedCamera.cameraRigMode, rigParams);\r\n }\r\n\r\n // Animations\r\n if (parsedCamera.animations) {\r\n for (let animationIndex = 0; animationIndex < parsedCamera.animations.length; animationIndex++) {\r\n const parsedAnimation = parsedCamera.animations[animationIndex];\r\n const internalClass = GetClass(\"BABYLON.Animation\");\r\n if (internalClass) {\r\n camera.animations.push(internalClass.Parse(parsedAnimation));\r\n }\r\n }\r\n Node.ParseAnimationRanges(camera, parsedCamera, scene);\r\n }\r\n\r\n if (parsedCamera.autoAnimate) {\r\n scene.beginAnimation(camera, parsedCamera.autoAnimateFrom, parsedCamera.autoAnimateTo, parsedCamera.autoAnimateLoop, parsedCamera.autoAnimateSpeed || 1.0);\r\n }\r\n\r\n // Check if isEnabled is defined to be back compatible with prior serialized versions.\r\n if (parsedCamera.isEnabled !== undefined) {\r\n camera.setEnabled(parsedCamera.isEnabled);\r\n }\r\n\r\n return camera;\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\n\r\n/**\r\n * This class implement a typical dictionary using a string as key and the generic type T as value.\r\n * The underlying implementation relies on an associative array to ensure the best performances.\r\n * The value can be anything including 'null' but except 'undefined'\r\n */\r\nexport class StringDictionary {\r\n /**\r\n * This will clear this dictionary and copy the content from the 'source' one.\r\n * If the T value is a custom object, it won't be copied/cloned, the same object will be used\r\n * @param source the dictionary to take the content from and copy to this dictionary\r\n */\r\n public copyFrom(source: StringDictionary) {\r\n this.clear();\r\n source.forEach((t, v) => this.add(t, v));\r\n }\r\n\r\n /**\r\n * Get a value based from its key\r\n * @param key the given key to get the matching value from\r\n * @returns the value if found, otherwise undefined is returned\r\n */\r\n public get(key: string): T | undefined {\r\n const val = this._data[key];\r\n if (val !== undefined) {\r\n return val;\r\n }\r\n return undefined;\r\n }\r\n\r\n /**\r\n * Get a value from its key or add it if it doesn't exist.\r\n * This method will ensure you that a given key/data will be present in the dictionary.\r\n * @param key the given key to get the matching value from\r\n * @param factory the factory that will create the value if the key is not present in the dictionary.\r\n * The factory will only be invoked if there's no data for the given key.\r\n * @returns the value corresponding to the key.\r\n */\r\n public getOrAddWithFactory(key: string, factory: (key: string) => T): T {\r\n let val = this.get(key);\r\n if (val !== undefined) {\r\n return val;\r\n }\r\n\r\n val = factory(key);\r\n if (val) {\r\n this.add(key, val);\r\n }\r\n\r\n return val;\r\n }\r\n\r\n /**\r\n * Get a value from its key if present in the dictionary otherwise add it\r\n * @param key the key to get the value from\r\n * @param val if there's no such key/value pair in the dictionary add it with this value\r\n * @returns the value corresponding to the key\r\n */\r\n public getOrAdd(key: string, val: T): T {\r\n const curVal = this.get(key);\r\n if (curVal !== undefined) {\r\n return curVal;\r\n }\r\n\r\n this.add(key, val);\r\n return val;\r\n }\r\n\r\n /**\r\n * Check if there's a given key in the dictionary\r\n * @param key the key to check for\r\n * @returns true if the key is present, false otherwise\r\n */\r\n public contains(key: string): boolean {\r\n return this._data[key] !== undefined;\r\n }\r\n\r\n /**\r\n * Add a new key and its corresponding value\r\n * @param key the key to add\r\n * @param value the value corresponding to the key\r\n * @returns true if the operation completed successfully, false if we couldn't insert the key/value because there was already this key in the dictionary\r\n */\r\n public add(key: string, value: T): boolean {\r\n if (this._data[key] !== undefined) {\r\n return false;\r\n }\r\n this._data[key] = value;\r\n ++this._count;\r\n return true;\r\n }\r\n\r\n /**\r\n * Update a specific value associated to a key\r\n * @param key defines the key to use\r\n * @param value defines the value to store\r\n * @returns true if the value was updated (or false if the key was not found)\r\n */\r\n public set(key: string, value: T): boolean {\r\n if (this._data[key] === undefined) {\r\n return false;\r\n }\r\n this._data[key] = value;\r\n return true;\r\n }\r\n\r\n /**\r\n * Get the element of the given key and remove it from the dictionary\r\n * @param key defines the key to search\r\n * @returns the value associated with the key or null if not found\r\n */\r\n public getAndRemove(key: string): Nullable {\r\n const val = this.get(key);\r\n if (val !== undefined) {\r\n delete this._data[key];\r\n --this._count;\r\n return val;\r\n }\r\n return null;\r\n }\r\n\r\n /**\r\n * Remove a key/value from the dictionary.\r\n * @param key the key to remove\r\n * @returns true if the item was successfully deleted, false if no item with such key exist in the dictionary\r\n */\r\n public remove(key: string): boolean {\r\n if (this.contains(key)) {\r\n delete this._data[key];\r\n --this._count;\r\n return true;\r\n }\r\n return false;\r\n }\r\n\r\n /**\r\n * Clear the whole content of the dictionary\r\n */\r\n public clear() {\r\n this._data = {};\r\n this._count = 0;\r\n }\r\n\r\n /**\r\n * Gets the current count\r\n */\r\n public get count() {\r\n return this._count;\r\n }\r\n\r\n /**\r\n * Execute a callback on each key/val of the dictionary.\r\n * Note that you can remove any element in this dictionary in the callback implementation\r\n * @param callback the callback to execute on a given key/value pair\r\n */\r\n public forEach(callback: (key: string, val: T) => void) {\r\n for (const cur in this._data) {\r\n const val = this._data[cur];\r\n callback(cur, val);\r\n }\r\n }\r\n\r\n /**\r\n * Execute a callback on every occurrence of the dictionary until it returns a valid TRes object.\r\n * If the callback returns null or undefined the method will iterate to the next key/value pair\r\n * Note that you can remove any element in this dictionary in the callback implementation\r\n * @param callback the callback to execute, if it return a valid T instanced object the enumeration will stop and the object will be returned\r\n * @returns the first item\r\n */\r\n public first(callback: (key: string, val: T) => TRes) {\r\n for (const cur in this._data) {\r\n const val = this._data[cur];\r\n const res = callback(cur, val);\r\n if (res) {\r\n return res;\r\n }\r\n }\r\n return null;\r\n }\r\n\r\n private _count = 0;\r\n private _data: { [key: string]: T } = {};\r\n}\r\n","import type { Scene } from \"./scene\";\r\nimport type { Nullable } from \"./types\";\r\nimport type { AbstractMesh } from \"./Meshes/abstractMesh\";\r\nimport type { TransformNode } from \"./Meshes/transformNode\";\r\nimport type { Geometry } from \"./Meshes/geometry\";\r\nimport type { Skeleton } from \"./Bones/skeleton\";\r\nimport type { MorphTargetManager } from \"./Morph/morphTargetManager\";\r\nimport type { AssetContainer } from \"./assetContainer\";\r\nimport type { IParticleSystem } from \"./Particles/IParticleSystem\";\r\nimport type { AnimationGroup } from \"./Animations/animationGroup\";\r\nimport type { BaseTexture } from \"./Materials/Textures/baseTexture\";\r\nimport type { Material } from \"./Materials/material\";\r\nimport type { MultiMaterial } from \"./Materials/multiMaterial\";\r\nimport type { AbstractActionManager } from \"./Actions/abstractActionManager\";\r\nimport type { Camera } from \"./Cameras/camera\";\r\nimport type { Light } from \"./Lights/light\";\r\nimport type { Node } from \"./node\";\r\nimport type { PostProcess } from \"./PostProcesses/postProcess\";\r\nimport type { Animation } from \"./Animations/animation\";\r\n\r\n/**\r\n * Defines how the parser contract is defined.\r\n * These parsers are used to parse a list of specific assets (like particle systems, etc..)\r\n */\r\nexport type BabylonFileParser = (parsedData: any, scene: Scene, container: AssetContainer, rootUrl: string) => void;\r\n\r\n/**\r\n * Defines how the individual parser contract is defined.\r\n * These parser can parse an individual asset\r\n */\r\nexport type IndividualBabylonFileParser = (parsedData: any, scene: Scene, rootUrl: string) => any;\r\n\r\n/**\r\n * Base class of the scene acting as a container for the different elements composing a scene.\r\n * This class is dynamically extended by the different components of the scene increasing\r\n * flexibility and reducing coupling\r\n */\r\nexport abstract class AbstractScene {\r\n /**\r\n * Stores the list of available parsers in the application.\r\n */\r\n private static _BabylonFileParsers: { [key: string]: BabylonFileParser } = {};\r\n\r\n /**\r\n * Stores the list of available individual parsers in the application.\r\n */\r\n private static _IndividualBabylonFileParsers: { [key: string]: IndividualBabylonFileParser } = {};\r\n\r\n /**\r\n * Adds a parser in the list of available ones\r\n * @param name Defines the name of the parser\r\n * @param parser Defines the parser to add\r\n */\r\n public static AddParser(name: string, parser: BabylonFileParser): void {\r\n this._BabylonFileParsers[name] = parser;\r\n }\r\n\r\n /**\r\n * Gets a general parser from the list of available ones\r\n * @param name Defines the name of the parser\r\n * @returns the requested parser or null\r\n */\r\n public static GetParser(name: string): Nullable {\r\n if (this._BabylonFileParsers[name]) {\r\n return this._BabylonFileParsers[name];\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Adds n individual parser in the list of available ones\r\n * @param name Defines the name of the parser\r\n * @param parser Defines the parser to add\r\n */\r\n public static AddIndividualParser(name: string, parser: IndividualBabylonFileParser): void {\r\n this._IndividualBabylonFileParsers[name] = parser;\r\n }\r\n\r\n /**\r\n * Gets an individual parser from the list of available ones\r\n * @param name Defines the name of the parser\r\n * @returns the requested parser or null\r\n */\r\n public static GetIndividualParser(name: string): Nullable {\r\n if (this._IndividualBabylonFileParsers[name]) {\r\n return this._IndividualBabylonFileParsers[name];\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Parser json data and populate both a scene and its associated container object\r\n * @param jsonData Defines the data to parse\r\n * @param scene Defines the scene to parse the data for\r\n * @param container Defines the container attached to the parsing sequence\r\n * @param rootUrl Defines the root url of the data\r\n */\r\n public static Parse(jsonData: any, scene: Scene, container: AssetContainer, rootUrl: string): void {\r\n for (const parserName in this._BabylonFileParsers) {\r\n if (Object.prototype.hasOwnProperty.call(this._BabylonFileParsers, parserName)) {\r\n this._BabylonFileParsers[parserName](jsonData, scene, container, rootUrl);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Gets the list of root nodes (ie. nodes with no parent)\r\n */\r\n public rootNodes = new Array();\r\n\r\n /** All of the cameras added to this scene\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras\r\n */\r\n public cameras = new Array();\r\n\r\n /**\r\n * All of the lights added to this scene\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction\r\n */\r\n public lights = new Array();\r\n\r\n /**\r\n * All of the (abstract) meshes added to this scene\r\n */\r\n public meshes = new Array();\r\n\r\n /**\r\n * The list of skeletons added to the scene\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/bonesSkeletons\r\n */\r\n public skeletons = new Array();\r\n\r\n /**\r\n * All of the particle systems added to this scene\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/particles/particle_system/particle_system_intro\r\n */\r\n public particleSystems = new Array();\r\n\r\n /**\r\n * Gets a list of Animations associated with the scene\r\n */\r\n public animations: Animation[] = [];\r\n\r\n /**\r\n * All of the animation groups added to this scene\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/groupAnimations\r\n */\r\n public animationGroups = new Array();\r\n\r\n /**\r\n * All of the multi-materials added to this scene\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/multiMaterials\r\n */\r\n public multiMaterials = new Array();\r\n\r\n /**\r\n * All of the materials added to this scene\r\n * In the context of a Scene, it is not supposed to be modified manually.\r\n * Any addition or removal should be done using the addMaterial and removeMaterial Scene methods.\r\n * Note also that the order of the Material within the array is not significant and might change.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/materials_introduction\r\n */\r\n public materials = new Array();\r\n\r\n /**\r\n * The list of morph target managers added to the scene\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph\r\n */\r\n public morphTargetManagers = new Array();\r\n\r\n /**\r\n * The list of geometries used in the scene.\r\n */\r\n public geometries = new Array();\r\n\r\n /**\r\n * All of the transform nodes added to this scene\r\n * In the context of a Scene, it is not supposed to be modified manually.\r\n * Any addition or removal should be done using the addTransformNode and removeTransformNode Scene methods.\r\n * Note also that the order of the TransformNode within the array is not significant and might change.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/parent_pivot/transform_node\r\n */\r\n public transformNodes = new Array();\r\n\r\n /**\r\n * ActionManagers available on the scene.\r\n * @deprecated\r\n */\r\n public actionManagers = new Array();\r\n\r\n /**\r\n * Textures to keep.\r\n */\r\n public textures = new Array();\r\n\r\n /** @internal */\r\n protected _environmentTexture: Nullable = null;\r\n /**\r\n * Texture used in all pbr material as the reflection texture.\r\n * As in the majority of the scene they are the same (exception for multi room and so on),\r\n * this is easier to reference from here than from all the materials.\r\n */\r\n public get environmentTexture(): Nullable {\r\n return this._environmentTexture;\r\n }\r\n\r\n public set environmentTexture(value: Nullable) {\r\n this._environmentTexture = value;\r\n }\r\n\r\n /**\r\n * The list of postprocesses added to the scene\r\n */\r\n public postProcesses = new Array();\r\n\r\n /**\r\n * @returns all meshes, lights, cameras, transformNodes and bones\r\n */\r\n public getNodes(): Array {\r\n let nodes = new Array();\r\n nodes = nodes.concat(this.meshes);\r\n nodes = nodes.concat(this.lights);\r\n nodes = nodes.concat(this.cameras);\r\n nodes = nodes.concat(this.transformNodes); // dummies\r\n this.skeletons.forEach((skeleton) => (nodes = nodes.concat(skeleton.bones)));\r\n return nodes;\r\n }\r\n}\r\n","/**\r\n * Manages the defines for the Material\r\n */\r\nexport class MaterialDefines {\r\n /** @internal */\r\n protected _keys: string[] = [];\r\n private _isDirty = true;\r\n /** @internal */\r\n public _renderId: number;\r\n\r\n /** @internal */\r\n public _areLightsDirty = true;\r\n /** @internal */\r\n public _areLightsDisposed = false;\r\n /** @internal */\r\n public _areAttributesDirty = true;\r\n /** @internal */\r\n public _areTexturesDirty = true;\r\n /** @internal */\r\n public _areFresnelDirty = true;\r\n /** @internal */\r\n public _areMiscDirty = true;\r\n /** @internal */\r\n public _arePrePassDirty = true;\r\n /** @internal */\r\n public _areImageProcessingDirty = true;\r\n\r\n /** @internal */\r\n public _normals = false;\r\n /** @internal */\r\n public _uvs = false;\r\n\r\n /** @internal */\r\n public _needNormals = false;\r\n /** @internal */\r\n public _needUVs = false;\r\n\r\n protected _externalProperties?: { [name: string]: { type: string; default: any } };\r\n\r\n [id: string]: any;\r\n\r\n /**\r\n * Creates a new instance\r\n * @param externalProperties list of external properties to inject into the object\r\n */\r\n constructor(externalProperties?: { [name: string]: { type: string; default: any } }) {\r\n this._externalProperties = externalProperties;\r\n\r\n // Initialize External Properties\r\n if (externalProperties) {\r\n for (const prop in externalProperties) {\r\n if (Object.prototype.hasOwnProperty.call(externalProperties, prop)) {\r\n this._setDefaultValue(prop);\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Specifies if the material needs to be re-calculated\r\n */\r\n public get isDirty(): boolean {\r\n return this._isDirty;\r\n }\r\n\r\n /**\r\n * Marks the material to indicate that it has been re-calculated\r\n */\r\n public markAsProcessed() {\r\n this._isDirty = false;\r\n this._areAttributesDirty = false;\r\n this._areTexturesDirty = false;\r\n this._areFresnelDirty = false;\r\n this._areLightsDirty = false;\r\n this._areLightsDisposed = false;\r\n this._areMiscDirty = false;\r\n this._arePrePassDirty = false;\r\n this._areImageProcessingDirty = false;\r\n }\r\n\r\n /**\r\n * Marks the material to indicate that it needs to be re-calculated\r\n */\r\n public markAsUnprocessed() {\r\n this._isDirty = true;\r\n }\r\n\r\n /**\r\n * Marks the material to indicate all of its defines need to be re-calculated\r\n */\r\n public markAllAsDirty() {\r\n this._areTexturesDirty = true;\r\n this._areAttributesDirty = true;\r\n this._areLightsDirty = true;\r\n this._areFresnelDirty = true;\r\n this._areMiscDirty = true;\r\n this._arePrePassDirty = false;\r\n this._areImageProcessingDirty = true;\r\n this._isDirty = true;\r\n }\r\n\r\n /**\r\n * Marks the material to indicate that image processing needs to be re-calculated\r\n */\r\n public markAsImageProcessingDirty() {\r\n this._areImageProcessingDirty = true;\r\n this._isDirty = true;\r\n }\r\n\r\n /**\r\n * Marks the material to indicate the lights need to be re-calculated\r\n * @param disposed Defines whether the light is dirty due to dispose or not\r\n */\r\n public markAsLightDirty(disposed = false) {\r\n this._areLightsDirty = true;\r\n this._areLightsDisposed = this._areLightsDisposed || disposed;\r\n this._isDirty = true;\r\n }\r\n\r\n /**\r\n * Marks the attribute state as changed\r\n */\r\n public markAsAttributesDirty() {\r\n this._areAttributesDirty = true;\r\n this._isDirty = true;\r\n }\r\n\r\n /**\r\n * Marks the texture state as changed\r\n */\r\n public markAsTexturesDirty() {\r\n this._areTexturesDirty = true;\r\n this._isDirty = true;\r\n }\r\n\r\n /**\r\n * Marks the fresnel state as changed\r\n */\r\n public markAsFresnelDirty() {\r\n this._areFresnelDirty = true;\r\n this._isDirty = true;\r\n }\r\n\r\n /**\r\n * Marks the misc state as changed\r\n */\r\n public markAsMiscDirty() {\r\n this._areMiscDirty = true;\r\n this._isDirty = true;\r\n }\r\n\r\n /**\r\n * Marks the prepass state as changed\r\n */\r\n public markAsPrePassDirty() {\r\n this._arePrePassDirty = true;\r\n this._isDirty = true;\r\n }\r\n\r\n /**\r\n * Rebuilds the material defines\r\n */\r\n public rebuild() {\r\n this._keys.length = 0;\r\n\r\n for (const key of Object.keys(this)) {\r\n if (key[0] === \"_\") {\r\n continue;\r\n }\r\n\r\n this._keys.push(key);\r\n }\r\n\r\n if (this._externalProperties) {\r\n for (const name in this._externalProperties) {\r\n if (this._keys.indexOf(name) === -1) {\r\n this._keys.push(name);\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Specifies if two material defines are equal\r\n * @param other - A material define instance to compare to\r\n * @returns - Boolean indicating if the material defines are equal (true) or not (false)\r\n */\r\n public isEqual(other: MaterialDefines): boolean {\r\n if (this._keys.length !== other._keys.length) {\r\n return false;\r\n }\r\n\r\n for (let index = 0; index < this._keys.length; index++) {\r\n const prop = this._keys[index];\r\n\r\n if ((this)[prop] !== (other)[prop]) {\r\n return false;\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Clones this instance's defines to another instance\r\n * @param other - material defines to clone values to\r\n */\r\n public cloneTo(other: MaterialDefines): void {\r\n if (this._keys.length !== other._keys.length) {\r\n other._keys = this._keys.slice(0);\r\n }\r\n\r\n for (let index = 0; index < this._keys.length; index++) {\r\n const prop = this._keys[index];\r\n\r\n (other)[prop] = (this)[prop];\r\n }\r\n }\r\n\r\n /**\r\n * Resets the material define values\r\n */\r\n public reset(): void {\r\n this._keys.forEach((prop) => this._setDefaultValue(prop));\r\n }\r\n\r\n private _setDefaultValue(prop: string): void {\r\n const type = this._externalProperties?.[prop]?.type ?? typeof (this)[prop];\r\n const defValue = this._externalProperties?.[prop]?.default;\r\n\r\n switch (type) {\r\n case \"number\":\r\n (this)[prop] = defValue ?? 0;\r\n break;\r\n case \"string\":\r\n (this)[prop] = defValue ?? \"\";\r\n break;\r\n default:\r\n (this)[prop] = defValue ?? false;\r\n break;\r\n }\r\n }\r\n\r\n /**\r\n * Converts the material define values to a string\r\n * @returns - String of material define information\r\n */\r\n public toString(): string {\r\n let result = \"\";\r\n for (let index = 0; index < this._keys.length; index++) {\r\n const prop = this._keys[index];\r\n const value = (this)[prop];\r\n const type = typeof value;\r\n\r\n switch (type) {\r\n case \"number\":\r\n case \"string\":\r\n result += \"#define \" + prop + \" \" + value + \"\\n\";\r\n break;\r\n default:\r\n if (value) {\r\n result += \"#define \" + prop + \"\\n\";\r\n }\r\n break;\r\n }\r\n }\r\n\r\n return result;\r\n }\r\n}\r\n","import { SerializationHelper, serialize } from \"../Misc/decorators\";\r\nimport { Color4 } from \"../Maths/math.color\";\r\nimport type { Effect } from \"../Materials/effect\";\r\n\r\n/**\r\n * The color grading curves provide additional color adjustment that is applied after any color grading transform (3D LUT).\r\n * They allow basic adjustment of saturation and small exposure adjustments, along with color filter tinting to provide white balance adjustment or more stylistic effects.\r\n * These are similar to controls found in many professional imaging or colorist software. The global controls are applied to the entire image. For advanced tuning, extra controls are provided to adjust the shadow, midtone and highlight areas of the image;\r\n * corresponding to low luminance, medium luminance, and high luminance areas respectively.\r\n */\r\nexport class ColorCurves {\r\n private _dirty = true;\r\n\r\n private _tempColor = new Color4(0, 0, 0, 0);\r\n\r\n private _globalCurve = new Color4(0, 0, 0, 0);\r\n private _highlightsCurve = new Color4(0, 0, 0, 0);\r\n private _midtonesCurve = new Color4(0, 0, 0, 0);\r\n private _shadowsCurve = new Color4(0, 0, 0, 0);\r\n\r\n private _positiveCurve = new Color4(0, 0, 0, 0);\r\n private _negativeCurve = new Color4(0, 0, 0, 0);\r\n\r\n @serialize()\r\n private _globalHue = 30;\r\n\r\n @serialize()\r\n private _globalDensity = 0;\r\n\r\n @serialize()\r\n private _globalSaturation = 0;\r\n\r\n @serialize()\r\n private _globalExposure = 0;\r\n\r\n /**\r\n * Gets the global Hue value.\r\n * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange).\r\n */\r\n public get globalHue(): number {\r\n return this._globalHue;\r\n }\r\n /**\r\n * Sets the global Hue value.\r\n * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange).\r\n */\r\n public set globalHue(value: number) {\r\n this._globalHue = value;\r\n this._dirty = true;\r\n }\r\n /**\r\n * Gets the global Density value.\r\n * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect.\r\n * Values less than zero provide a filter of opposite hue.\r\n */\r\n public get globalDensity(): number {\r\n return this._globalDensity;\r\n }\r\n /**\r\n * Sets the global Density value.\r\n * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect.\r\n * Values less than zero provide a filter of opposite hue.\r\n */\r\n public set globalDensity(value: number) {\r\n this._globalDensity = value;\r\n this._dirty = true;\r\n }\r\n /**\r\n * Gets the global Saturation value.\r\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation.\r\n */\r\n public get globalSaturation(): number {\r\n return this._globalSaturation;\r\n }\r\n /**\r\n * Sets the global Saturation value.\r\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation.\r\n */\r\n public set globalSaturation(value: number) {\r\n this._globalSaturation = value;\r\n this._dirty = true;\r\n }\r\n\r\n /**\r\n * Gets the global Exposure value.\r\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure.\r\n */\r\n public get globalExposure(): number {\r\n return this._globalExposure;\r\n }\r\n /**\r\n * Sets the global Exposure value.\r\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure.\r\n */\r\n public set globalExposure(value: number) {\r\n this._globalExposure = value;\r\n this._dirty = true;\r\n }\r\n\r\n @serialize()\r\n private _highlightsHue = 30;\r\n\r\n @serialize()\r\n private _highlightsDensity = 0;\r\n\r\n @serialize()\r\n private _highlightsSaturation = 0;\r\n\r\n @serialize()\r\n private _highlightsExposure = 0;\r\n\r\n /**\r\n * Gets the highlights Hue value.\r\n * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange).\r\n */\r\n public get highlightsHue(): number {\r\n return this._highlightsHue;\r\n }\r\n /**\r\n * Sets the highlights Hue value.\r\n * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange).\r\n */\r\n public set highlightsHue(value: number) {\r\n this._highlightsHue = value;\r\n this._dirty = true;\r\n }\r\n /**\r\n * Gets the highlights Density value.\r\n * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect.\r\n * Values less than zero provide a filter of opposite hue.\r\n */\r\n public get highlightsDensity(): number {\r\n return this._highlightsDensity;\r\n }\r\n /**\r\n * Sets the highlights Density value.\r\n * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect.\r\n * Values less than zero provide a filter of opposite hue.\r\n */\r\n public set highlightsDensity(value: number) {\r\n this._highlightsDensity = value;\r\n this._dirty = true;\r\n }\r\n /**\r\n * Gets the highlights Saturation value.\r\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation.\r\n */\r\n public get highlightsSaturation(): number {\r\n return this._highlightsSaturation;\r\n }\r\n /**\r\n * Sets the highlights Saturation value.\r\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation.\r\n */\r\n public set highlightsSaturation(value: number) {\r\n this._highlightsSaturation = value;\r\n this._dirty = true;\r\n }\r\n /**\r\n * Gets the highlights Exposure value.\r\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure.\r\n */\r\n public get highlightsExposure(): number {\r\n return this._highlightsExposure;\r\n }\r\n /**\r\n * Sets the highlights Exposure value.\r\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure.\r\n */\r\n public set highlightsExposure(value: number) {\r\n this._highlightsExposure = value;\r\n this._dirty = true;\r\n }\r\n\r\n @serialize()\r\n private _midtonesHue = 30;\r\n\r\n @serialize()\r\n private _midtonesDensity = 0;\r\n\r\n @serialize()\r\n private _midtonesSaturation = 0;\r\n\r\n @serialize()\r\n private _midtonesExposure = 0;\r\n\r\n /**\r\n * Gets the midtones Hue value.\r\n * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange).\r\n */\r\n public get midtonesHue(): number {\r\n return this._midtonesHue;\r\n }\r\n /**\r\n * Sets the midtones Hue value.\r\n * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange).\r\n */\r\n public set midtonesHue(value: number) {\r\n this._midtonesHue = value;\r\n this._dirty = true;\r\n }\r\n /**\r\n * Gets the midtones Density value.\r\n * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect.\r\n * Values less than zero provide a filter of opposite hue.\r\n */\r\n public get midtonesDensity(): number {\r\n return this._midtonesDensity;\r\n }\r\n /**\r\n * Sets the midtones Density value.\r\n * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect.\r\n * Values less than zero provide a filter of opposite hue.\r\n */\r\n public set midtonesDensity(value: number) {\r\n this._midtonesDensity = value;\r\n this._dirty = true;\r\n }\r\n /**\r\n * Gets the midtones Saturation value.\r\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation.\r\n */\r\n public get midtonesSaturation(): number {\r\n return this._midtonesSaturation;\r\n }\r\n /**\r\n * Sets the midtones Saturation value.\r\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation.\r\n */\r\n public set midtonesSaturation(value: number) {\r\n this._midtonesSaturation = value;\r\n this._dirty = true;\r\n }\r\n /**\r\n * Gets the midtones Exposure value.\r\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure.\r\n */\r\n public get midtonesExposure(): number {\r\n return this._midtonesExposure;\r\n }\r\n /**\r\n * Sets the midtones Exposure value.\r\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure.\r\n */\r\n public set midtonesExposure(value: number) {\r\n this._midtonesExposure = value;\r\n this._dirty = true;\r\n }\r\n\r\n private _shadowsHue = 30;\r\n private _shadowsDensity = 0;\r\n private _shadowsSaturation = 0;\r\n private _shadowsExposure = 0;\r\n\r\n /**\r\n * Gets the shadows Hue value.\r\n * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange).\r\n */\r\n public get shadowsHue(): number {\r\n return this._shadowsHue;\r\n }\r\n /**\r\n * Sets the shadows Hue value.\r\n * The hue value is a standard HSB hue in the range [0,360] where 0=red, 120=green and 240=blue. The default value is 30 degrees (orange).\r\n */\r\n public set shadowsHue(value: number) {\r\n this._shadowsHue = value;\r\n this._dirty = true;\r\n }\r\n /**\r\n * Gets the shadows Density value.\r\n * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect.\r\n * Values less than zero provide a filter of opposite hue.\r\n */\r\n public get shadowsDensity(): number {\r\n return this._shadowsDensity;\r\n }\r\n /**\r\n * Sets the shadows Density value.\r\n * The density value is in range [-100,+100] where 0 means the color filter has no effect and +100 means the color filter has maximum effect.\r\n * Values less than zero provide a filter of opposite hue.\r\n */\r\n public set shadowsDensity(value: number) {\r\n this._shadowsDensity = value;\r\n this._dirty = true;\r\n }\r\n /**\r\n * Gets the shadows Saturation value.\r\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation.\r\n */\r\n public get shadowsSaturation(): number {\r\n return this._shadowsSaturation;\r\n }\r\n /**\r\n * Sets the shadows Saturation value.\r\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase saturation and negative values decrease saturation.\r\n */\r\n public set shadowsSaturation(value: number) {\r\n this._shadowsSaturation = value;\r\n this._dirty = true;\r\n }\r\n /**\r\n * Gets the shadows Exposure value.\r\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure.\r\n */\r\n public get shadowsExposure(): number {\r\n return this._shadowsExposure;\r\n }\r\n /**\r\n * Sets the shadows Exposure value.\r\n * This is an adjustment value in the range [-100,+100], where the default value of 0.0 makes no adjustment, positive values increase exposure and negative values decrease exposure.\r\n */\r\n public set shadowsExposure(value: number) {\r\n this._shadowsExposure = value;\r\n this._dirty = true;\r\n }\r\n\r\n /**\r\n * Returns the class name\r\n * @returns The class name\r\n */\r\n public getClassName(): string {\r\n return \"ColorCurves\";\r\n }\r\n\r\n /**\r\n * Binds the color curves to the shader.\r\n * @param colorCurves The color curve to bind\r\n * @param effect The effect to bind to\r\n * @param positiveUniform The positive uniform shader parameter\r\n * @param neutralUniform The neutral uniform shader parameter\r\n * @param negativeUniform The negative uniform shader parameter\r\n */\r\n public static Bind(\r\n colorCurves: ColorCurves,\r\n effect: Effect,\r\n positiveUniform = \"vCameraColorCurvePositive\",\r\n neutralUniform = \"vCameraColorCurveNeutral\",\r\n negativeUniform = \"vCameraColorCurveNegative\"\r\n ): void {\r\n if (colorCurves._dirty) {\r\n colorCurves._dirty = false;\r\n\r\n // Fill in global info.\r\n colorCurves._getColorGradingDataToRef(\r\n colorCurves._globalHue,\r\n colorCurves._globalDensity,\r\n colorCurves._globalSaturation,\r\n colorCurves._globalExposure,\r\n colorCurves._globalCurve\r\n );\r\n\r\n // Compute highlights info.\r\n colorCurves._getColorGradingDataToRef(\r\n colorCurves._highlightsHue,\r\n colorCurves._highlightsDensity,\r\n colorCurves._highlightsSaturation,\r\n colorCurves._highlightsExposure,\r\n colorCurves._tempColor\r\n );\r\n colorCurves._tempColor.multiplyToRef(colorCurves._globalCurve, colorCurves._highlightsCurve);\r\n\r\n // Compute midtones info.\r\n colorCurves._getColorGradingDataToRef(\r\n colorCurves._midtonesHue,\r\n colorCurves._midtonesDensity,\r\n colorCurves._midtonesSaturation,\r\n colorCurves._midtonesExposure,\r\n colorCurves._tempColor\r\n );\r\n colorCurves._tempColor.multiplyToRef(colorCurves._globalCurve, colorCurves._midtonesCurve);\r\n\r\n // Compute shadows info.\r\n colorCurves._getColorGradingDataToRef(\r\n colorCurves._shadowsHue,\r\n colorCurves._shadowsDensity,\r\n colorCurves._shadowsSaturation,\r\n colorCurves._shadowsExposure,\r\n colorCurves._tempColor\r\n );\r\n colorCurves._tempColor.multiplyToRef(colorCurves._globalCurve, colorCurves._shadowsCurve);\r\n\r\n // Compute deltas (neutral is midtones).\r\n colorCurves._highlightsCurve.subtractToRef(colorCurves._midtonesCurve, colorCurves._positiveCurve);\r\n colorCurves._midtonesCurve.subtractToRef(colorCurves._shadowsCurve, colorCurves._negativeCurve);\r\n }\r\n\r\n if (effect) {\r\n effect.setFloat4(positiveUniform, colorCurves._positiveCurve.r, colorCurves._positiveCurve.g, colorCurves._positiveCurve.b, colorCurves._positiveCurve.a);\r\n effect.setFloat4(neutralUniform, colorCurves._midtonesCurve.r, colorCurves._midtonesCurve.g, colorCurves._midtonesCurve.b, colorCurves._midtonesCurve.a);\r\n effect.setFloat4(negativeUniform, colorCurves._negativeCurve.r, colorCurves._negativeCurve.g, colorCurves._negativeCurve.b, colorCurves._negativeCurve.a);\r\n }\r\n }\r\n\r\n /**\r\n * Prepare the list of uniforms associated with the ColorCurves effects.\r\n * @param uniformsList The list of uniforms used in the effect\r\n */\r\n public static PrepareUniforms(uniformsList: string[]): void {\r\n uniformsList.push(\"vCameraColorCurveNeutral\", \"vCameraColorCurvePositive\", \"vCameraColorCurveNegative\");\r\n }\r\n\r\n /**\r\n * Returns color grading data based on a hue, density, saturation and exposure value.\r\n * @param hue\r\n * @param density\r\n * @param saturation The saturation.\r\n * @param exposure The exposure.\r\n * @param result The result data container.\r\n */\r\n private _getColorGradingDataToRef(hue: number, density: number, saturation: number, exposure: number, result: Color4): void {\r\n if (hue == null) {\r\n return;\r\n }\r\n\r\n hue = ColorCurves._Clamp(hue, 0, 360);\r\n density = ColorCurves._Clamp(density, -100, 100);\r\n saturation = ColorCurves._Clamp(saturation, -100, 100);\r\n exposure = ColorCurves._Clamp(exposure, -100, 100);\r\n\r\n // Remap the slider/config filter density with non-linear mapping and also scale by half\r\n // so that the maximum filter density is only 50% control. This provides fine control\r\n // for small values and reasonable range.\r\n density = ColorCurves._ApplyColorGradingSliderNonlinear(density);\r\n density *= 0.5;\r\n\r\n exposure = ColorCurves._ApplyColorGradingSliderNonlinear(exposure);\r\n\r\n if (density < 0) {\r\n density *= -1;\r\n hue = (hue + 180) % 360;\r\n }\r\n\r\n ColorCurves._FromHSBToRef(hue, density, 50 + 0.25 * exposure, result);\r\n result.scaleToRef(2, result);\r\n result.a = 1 + 0.01 * saturation;\r\n }\r\n\r\n /**\r\n * Takes an input slider value and returns an adjusted value that provides extra control near the centre.\r\n * @param value The input slider value in range [-100,100].\r\n * @returns Adjusted value.\r\n */\r\n private static _ApplyColorGradingSliderNonlinear(value: number): number {\r\n value /= 100;\r\n\r\n let x: number = Math.abs(value);\r\n x = Math.pow(x, 2);\r\n\r\n if (value < 0) {\r\n x *= -1;\r\n }\r\n\r\n x *= 100;\r\n\r\n return x;\r\n }\r\n\r\n /**\r\n * Returns an RGBA Color4 based on Hue, Saturation and Brightness (also referred to as value, HSV).\r\n * @param hue The hue (H) input.\r\n * @param saturation The saturation (S) input.\r\n * @param brightness The brightness (B) input.\r\n * @param result\r\n * @result An RGBA color represented as Vector4.\r\n */\r\n private static _FromHSBToRef(hue: number, saturation: number, brightness: number, result: Color4): void {\r\n let h: number = ColorCurves._Clamp(hue, 0, 360);\r\n const s: number = ColorCurves._Clamp(saturation / 100, 0, 1);\r\n const v: number = ColorCurves._Clamp(brightness / 100, 0, 1);\r\n\r\n if (s === 0) {\r\n result.r = v;\r\n result.g = v;\r\n result.b = v;\r\n } else {\r\n // sector 0 to 5\r\n h /= 60;\r\n const i = Math.floor(h);\r\n\r\n // fractional part of h\r\n const f = h - i;\r\n const p = v * (1 - s);\r\n const q = v * (1 - s * f);\r\n const t = v * (1 - s * (1 - f));\r\n\r\n switch (i) {\r\n case 0:\r\n result.r = v;\r\n result.g = t;\r\n result.b = p;\r\n break;\r\n case 1:\r\n result.r = q;\r\n result.g = v;\r\n result.b = p;\r\n break;\r\n case 2:\r\n result.r = p;\r\n result.g = v;\r\n result.b = t;\r\n break;\r\n case 3:\r\n result.r = p;\r\n result.g = q;\r\n result.b = v;\r\n break;\r\n case 4:\r\n result.r = t;\r\n result.g = p;\r\n result.b = v;\r\n break;\r\n default:\r\n // case 5:\r\n result.r = v;\r\n result.g = p;\r\n result.b = q;\r\n break;\r\n }\r\n }\r\n\r\n result.a = 1;\r\n }\r\n\r\n /**\r\n * Returns a value clamped between min and max\r\n * @param value The value to clamp\r\n * @param min The minimum of value\r\n * @param max The maximum of value\r\n * @returns The clamped value.\r\n */\r\n private static _Clamp(value: number, min: number, max: number): number {\r\n return Math.min(Math.max(value, min), max);\r\n }\r\n\r\n /**\r\n * Clones the current color curve instance.\r\n * @returns The cloned curves\r\n */\r\n public clone(): ColorCurves {\r\n return SerializationHelper.Clone(() => new ColorCurves(), this);\r\n }\r\n\r\n /**\r\n * Serializes the current color curve instance to a json representation.\r\n * @returns a JSON representation\r\n */\r\n public serialize(): any {\r\n return SerializationHelper.Serialize(this);\r\n }\r\n\r\n /**\r\n * Parses the color curve from a json representation.\r\n * @param source the JSON source to parse\r\n * @returns The parsed curves\r\n */\r\n public static Parse(source: any): ColorCurves {\r\n return SerializationHelper.Parse(() => new ColorCurves(), source, null, null);\r\n }\r\n}\r\n\r\n// References the dependencies.\r\nSerializationHelper._ColorCurvesParser = ColorCurves.Parse;\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport { serialize, SerializationHelper, serializeAsTexture, serializeAsColorCurves, serializeAsColor4 } from \"../Misc/decorators\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport { Tools } from \"../Misc/tools\";\r\nimport type { Nullable } from \"../types\";\r\nimport { Color4 } from \"../Maths/math.color\";\r\nimport { MaterialDefines } from \"../Materials/materialDefines\";\r\nimport { ColorCurves } from \"../Materials/colorCurves\";\r\n\r\nimport type { BaseTexture } from \"../Materials/Textures/baseTexture\";\r\nimport type { Effect } from \"../Materials/effect\";\r\n\r\n/**\r\n * Interface to follow in your material defines to integrate easily the\r\n * Image processing functions.\r\n * @internal\r\n */\r\nexport interface IImageProcessingConfigurationDefines {\r\n IMAGEPROCESSING: boolean;\r\n VIGNETTE: boolean;\r\n VIGNETTEBLENDMODEMULTIPLY: boolean;\r\n VIGNETTEBLENDMODEOPAQUE: boolean;\r\n TONEMAPPING: boolean;\r\n TONEMAPPING_ACES: boolean;\r\n CONTRAST: boolean;\r\n EXPOSURE: boolean;\r\n COLORCURVES: boolean;\r\n COLORGRADING: boolean;\r\n COLORGRADING3D: boolean;\r\n SAMPLER3DGREENDEPTH: boolean;\r\n SAMPLER3DBGRMAP: boolean;\r\n DITHER: boolean;\r\n IMAGEPROCESSINGPOSTPROCESS: boolean;\r\n SKIPFINALCOLORCLAMP: boolean;\r\n}\r\n\r\n/**\r\n * @internal\r\n */\r\nexport class ImageProcessingConfigurationDefines extends MaterialDefines implements IImageProcessingConfigurationDefines {\r\n public IMAGEPROCESSING = false;\r\n public VIGNETTE = false;\r\n public VIGNETTEBLENDMODEMULTIPLY = false;\r\n public VIGNETTEBLENDMODEOPAQUE = false;\r\n public TONEMAPPING = false;\r\n public TONEMAPPING_ACES = false;\r\n public CONTRAST = false;\r\n public COLORCURVES = false;\r\n public COLORGRADING = false;\r\n public COLORGRADING3D = false;\r\n public SAMPLER3DGREENDEPTH = false;\r\n public SAMPLER3DBGRMAP = false;\r\n public DITHER = false;\r\n public IMAGEPROCESSINGPOSTPROCESS = false;\r\n public EXPOSURE = false;\r\n public SKIPFINALCOLORCLAMP = false;\r\n\r\n constructor() {\r\n super();\r\n this.rebuild();\r\n }\r\n}\r\n\r\n/**\r\n * This groups together the common properties used for image processing either in direct forward pass\r\n * or through post processing effect depending on the use of the image processing pipeline in your scene\r\n * or not.\r\n */\r\nexport class ImageProcessingConfiguration {\r\n /**\r\n * Default tone mapping applied in BabylonJS.\r\n */\r\n public static readonly TONEMAPPING_STANDARD = 0;\r\n\r\n /**\r\n * ACES Tone mapping (used by default in unreal and unity). This can help getting closer\r\n * to other engines rendering to increase portability.\r\n */\r\n public static readonly TONEMAPPING_ACES = 1;\r\n\r\n /**\r\n * Color curves setup used in the effect if colorCurvesEnabled is set to true\r\n */\r\n @serializeAsColorCurves()\r\n public colorCurves: Nullable = new ColorCurves();\r\n\r\n @serialize()\r\n private _colorCurvesEnabled = false;\r\n /**\r\n * Gets whether the color curves effect is enabled.\r\n */\r\n public get colorCurvesEnabled(): boolean {\r\n return this._colorCurvesEnabled;\r\n }\r\n /**\r\n * Sets whether the color curves effect is enabled.\r\n */\r\n public set colorCurvesEnabled(value: boolean) {\r\n if (this._colorCurvesEnabled === value) {\r\n return;\r\n }\r\n\r\n this._colorCurvesEnabled = value;\r\n this._updateParameters();\r\n }\r\n\r\n @serializeAsTexture(\"colorGradingTexture\")\r\n private _colorGradingTexture: Nullable;\r\n /**\r\n * Color grading LUT texture used in the effect if colorGradingEnabled is set to true\r\n */\r\n public get colorGradingTexture(): Nullable {\r\n return this._colorGradingTexture;\r\n }\r\n /**\r\n * Color grading LUT texture used in the effect if colorGradingEnabled is set to true\r\n */\r\n public set colorGradingTexture(value: Nullable) {\r\n if (this._colorGradingTexture === value) {\r\n return;\r\n }\r\n\r\n this._colorGradingTexture = value;\r\n this._updateParameters();\r\n }\r\n\r\n @serialize()\r\n private _colorGradingEnabled = false;\r\n /**\r\n * Gets whether the color grading effect is enabled.\r\n */\r\n public get colorGradingEnabled(): boolean {\r\n return this._colorGradingEnabled;\r\n }\r\n /**\r\n * Sets whether the color grading effect is enabled.\r\n */\r\n public set colorGradingEnabled(value: boolean) {\r\n if (this._colorGradingEnabled === value) {\r\n return;\r\n }\r\n\r\n this._colorGradingEnabled = value;\r\n this._updateParameters();\r\n }\r\n\r\n @serialize()\r\n private _colorGradingWithGreenDepth = true;\r\n /**\r\n * Gets whether the color grading effect is using a green depth for the 3d Texture.\r\n */\r\n public get colorGradingWithGreenDepth(): boolean {\r\n return this._colorGradingWithGreenDepth;\r\n }\r\n /**\r\n * Sets whether the color grading effect is using a green depth for the 3d Texture.\r\n */\r\n public set colorGradingWithGreenDepth(value: boolean) {\r\n if (this._colorGradingWithGreenDepth === value) {\r\n return;\r\n }\r\n\r\n this._colorGradingWithGreenDepth = value;\r\n this._updateParameters();\r\n }\r\n\r\n @serialize()\r\n private _colorGradingBGR = true;\r\n /**\r\n * Gets whether the color grading texture contains BGR values.\r\n */\r\n public get colorGradingBGR(): boolean {\r\n return this._colorGradingBGR;\r\n }\r\n /**\r\n * Sets whether the color grading texture contains BGR values.\r\n */\r\n public set colorGradingBGR(value: boolean) {\r\n if (this._colorGradingBGR === value) {\r\n return;\r\n }\r\n\r\n this._colorGradingBGR = value;\r\n this._updateParameters();\r\n }\r\n\r\n /** @internal */\r\n @serialize()\r\n public _exposure = 1.0;\r\n /**\r\n * Gets the Exposure used in the effect.\r\n */\r\n public get exposure(): number {\r\n return this._exposure;\r\n }\r\n /**\r\n * Sets the Exposure used in the effect.\r\n */\r\n public set exposure(value: number) {\r\n if (this._exposure === value) {\r\n return;\r\n }\r\n\r\n this._exposure = value;\r\n this._updateParameters();\r\n }\r\n\r\n @serialize()\r\n private _toneMappingEnabled = false;\r\n /**\r\n * Gets whether the tone mapping effect is enabled.\r\n */\r\n public get toneMappingEnabled(): boolean {\r\n return this._toneMappingEnabled;\r\n }\r\n /**\r\n * Sets whether the tone mapping effect is enabled.\r\n */\r\n public set toneMappingEnabled(value: boolean) {\r\n if (this._toneMappingEnabled === value) {\r\n return;\r\n }\r\n\r\n this._toneMappingEnabled = value;\r\n this._updateParameters();\r\n }\r\n\r\n @serialize()\r\n private _toneMappingType = ImageProcessingConfiguration.TONEMAPPING_STANDARD;\r\n /**\r\n * Gets the type of tone mapping effect.\r\n */\r\n public get toneMappingType(): number {\r\n return this._toneMappingType;\r\n }\r\n /**\r\n * Sets the type of tone mapping effect used in BabylonJS.\r\n */\r\n public set toneMappingType(value: number) {\r\n if (this._toneMappingType === value) {\r\n return;\r\n }\r\n\r\n this._toneMappingType = value;\r\n this._updateParameters();\r\n }\r\n\r\n @serialize()\r\n protected _contrast = 1.0;\r\n /**\r\n * Gets the contrast used in the effect.\r\n */\r\n public get contrast(): number {\r\n return this._contrast;\r\n }\r\n /**\r\n * Sets the contrast used in the effect.\r\n */\r\n public set contrast(value: number) {\r\n if (this._contrast === value) {\r\n return;\r\n }\r\n\r\n this._contrast = value;\r\n this._updateParameters();\r\n }\r\n\r\n /**\r\n * Vignette stretch size.\r\n */\r\n @serialize()\r\n public vignetteStretch = 0;\r\n\r\n /**\r\n * Vignette center X Offset.\r\n */\r\n @serialize()\r\n public vignetteCenterX = 0;\r\n\r\n /**\r\n * Vignette center Y Offset.\r\n */\r\n @serialize()\r\n public vignetteCenterY = 0;\r\n\r\n /**\r\n * Back Compat: Vignette center Y Offset.\r\n * @deprecated use vignetteCenterY instead\r\n */\r\n public get vignetteCentreY(): number {\r\n return this.vignetteCenterY;\r\n }\r\n public set vignetteCentreY(value: number) {\r\n this.vignetteCenterY = value;\r\n }\r\n\r\n /**\r\n * Back Compat: Vignette center X Offset.\r\n * @deprecated use vignetteCenterX instead\r\n */\r\n public get vignetteCentreX(): number {\r\n return this.vignetteCenterX;\r\n }\r\n public set vignetteCentreX(value: number) {\r\n this.vignetteCenterX = value;\r\n }\r\n\r\n /**\r\n * Vignette weight or intensity of the vignette effect.\r\n */\r\n @serialize()\r\n public vignetteWeight = 1.5;\r\n\r\n /**\r\n * Color of the vignette applied on the screen through the chosen blend mode (vignetteBlendMode)\r\n * if vignetteEnabled is set to true.\r\n */\r\n @serializeAsColor4()\r\n public vignetteColor: Color4 = new Color4(0, 0, 0, 0);\r\n\r\n /**\r\n * Camera field of view used by the Vignette effect.\r\n */\r\n @serialize()\r\n public vignetteCameraFov = 0.5;\r\n\r\n @serialize()\r\n private _vignetteBlendMode = ImageProcessingConfiguration.VIGNETTEMODE_MULTIPLY;\r\n /**\r\n * Gets the vignette blend mode allowing different kind of effect.\r\n */\r\n public get vignetteBlendMode(): number {\r\n return this._vignetteBlendMode;\r\n }\r\n /**\r\n * Sets the vignette blend mode allowing different kind of effect.\r\n */\r\n public set vignetteBlendMode(value: number) {\r\n if (this._vignetteBlendMode === value) {\r\n return;\r\n }\r\n\r\n this._vignetteBlendMode = value;\r\n this._updateParameters();\r\n }\r\n\r\n @serialize()\r\n private _vignetteEnabled = false;\r\n /**\r\n * Gets whether the vignette effect is enabled.\r\n */\r\n public get vignetteEnabled(): boolean {\r\n return this._vignetteEnabled;\r\n }\r\n /**\r\n * Sets whether the vignette effect is enabled.\r\n */\r\n public set vignetteEnabled(value: boolean) {\r\n if (this._vignetteEnabled === value) {\r\n return;\r\n }\r\n\r\n this._vignetteEnabled = value;\r\n this._updateParameters();\r\n }\r\n\r\n @serialize()\r\n private _ditheringEnabled = false;\r\n /**\r\n * Gets whether the dithering effect is enabled.\r\n * The dithering effect can be used to reduce banding.\r\n */\r\n public get ditheringEnabled(): boolean {\r\n return this._ditheringEnabled;\r\n }\r\n /**\r\n * Sets whether the dithering effect is enabled.\r\n * The dithering effect can be used to reduce banding.\r\n */\r\n public set ditheringEnabled(value: boolean) {\r\n if (this._ditheringEnabled === value) {\r\n return;\r\n }\r\n\r\n this._ditheringEnabled = value;\r\n this._updateParameters();\r\n }\r\n\r\n @serialize()\r\n private _ditheringIntensity = 1.0 / 255.0;\r\n /**\r\n * Gets the dithering intensity. 0 is no dithering. Default is 1.0 / 255.0.\r\n */\r\n public get ditheringIntensity(): number {\r\n return this._ditheringIntensity;\r\n }\r\n /**\r\n * Sets the dithering intensity. 0 is no dithering. Default is 1.0 / 255.0.\r\n */\r\n public set ditheringIntensity(value: number) {\r\n if (this._ditheringIntensity === value) {\r\n return;\r\n }\r\n\r\n this._ditheringIntensity = value;\r\n this._updateParameters();\r\n }\r\n\r\n /** @internal */\r\n @serialize()\r\n public _skipFinalColorClamp = false;\r\n /**\r\n * If apply by post process is set to true, setting this to true will skip the the final color clamp step in the fragment shader\r\n * Applies to PBR materials.\r\n */\r\n public get skipFinalColorClamp(): boolean {\r\n return this._skipFinalColorClamp;\r\n }\r\n /**\r\n * If apply by post process is set to true, setting this to true will skip the the final color clamp step in the fragment shader\r\n * Applies to PBR materials.\r\n */\r\n public set skipFinalColorClamp(value: boolean) {\r\n if (this._skipFinalColorClamp === value) {\r\n return;\r\n }\r\n\r\n this._skipFinalColorClamp = value;\r\n this._updateParameters();\r\n }\r\n\r\n /** @internal */\r\n @serialize()\r\n public _applyByPostProcess = false;\r\n /**\r\n * Gets whether the image processing is applied through a post process or not.\r\n */\r\n public get applyByPostProcess(): boolean {\r\n return this._applyByPostProcess;\r\n }\r\n /**\r\n * Sets whether the image processing is applied through a post process or not.\r\n */\r\n public set applyByPostProcess(value: boolean) {\r\n if (this._applyByPostProcess === value) {\r\n return;\r\n }\r\n\r\n this._applyByPostProcess = value;\r\n this._updateParameters();\r\n }\r\n\r\n @serialize()\r\n private _isEnabled = true;\r\n /**\r\n * Gets whether the image processing is enabled or not.\r\n */\r\n public get isEnabled(): boolean {\r\n return this._isEnabled;\r\n }\r\n /**\r\n * Sets whether the image processing is enabled or not.\r\n */\r\n public set isEnabled(value: boolean) {\r\n if (this._isEnabled === value) {\r\n return;\r\n }\r\n\r\n this._isEnabled = value;\r\n this._updateParameters();\r\n }\r\n\r\n /**\r\n * An event triggered when the configuration changes and requires Shader to Update some parameters.\r\n */\r\n public onUpdateParameters = new Observable();\r\n\r\n /**\r\n * Method called each time the image processing information changes requires to recompile the effect.\r\n */\r\n protected _updateParameters(): void {\r\n this.onUpdateParameters.notifyObservers(this);\r\n }\r\n\r\n /**\r\n * Gets the current class name.\r\n * @returns \"ImageProcessingConfiguration\"\r\n */\r\n public getClassName(): string {\r\n return \"ImageProcessingConfiguration\";\r\n }\r\n\r\n /**\r\n * Prepare the list of uniforms associated with the Image Processing effects.\r\n * @param uniforms The list of uniforms used in the effect\r\n * @param defines the list of defines currently in use\r\n */\r\n public static PrepareUniforms(uniforms: string[], defines: IImageProcessingConfigurationDefines): void {\r\n if (defines.EXPOSURE) {\r\n uniforms.push(\"exposureLinear\");\r\n }\r\n if (defines.CONTRAST) {\r\n uniforms.push(\"contrast\");\r\n }\r\n if (defines.COLORGRADING) {\r\n uniforms.push(\"colorTransformSettings\");\r\n }\r\n if (defines.VIGNETTE || defines.DITHER) {\r\n uniforms.push(\"vInverseScreenSize\");\r\n }\r\n if (defines.VIGNETTE) {\r\n uniforms.push(\"vignetteSettings1\");\r\n uniforms.push(\"vignetteSettings2\");\r\n }\r\n if (defines.COLORCURVES) {\r\n ColorCurves.PrepareUniforms(uniforms);\r\n }\r\n if (defines.DITHER) {\r\n uniforms.push(\"ditherIntensity\");\r\n }\r\n }\r\n\r\n /**\r\n * Prepare the list of samplers associated with the Image Processing effects.\r\n * @param samplersList The list of uniforms used in the effect\r\n * @param defines the list of defines currently in use\r\n */\r\n public static PrepareSamplers(samplersList: string[], defines: IImageProcessingConfigurationDefines): void {\r\n if (defines.COLORGRADING) {\r\n samplersList.push(\"txColorTransform\");\r\n }\r\n }\r\n\r\n /**\r\n * Prepare the list of defines associated to the shader.\r\n * @param defines the list of defines to complete\r\n * @param forPostProcess Define if we are currently in post process mode or not\r\n */\r\n public prepareDefines(defines: IImageProcessingConfigurationDefines, forPostProcess = false): void {\r\n if (forPostProcess !== this.applyByPostProcess || !this._isEnabled) {\r\n defines.VIGNETTE = false;\r\n defines.TONEMAPPING = false;\r\n defines.TONEMAPPING_ACES = false;\r\n defines.CONTRAST = false;\r\n defines.EXPOSURE = false;\r\n defines.COLORCURVES = false;\r\n defines.COLORGRADING = false;\r\n defines.COLORGRADING3D = false;\r\n defines.DITHER = false;\r\n defines.IMAGEPROCESSING = false;\r\n defines.SKIPFINALCOLORCLAMP = this.skipFinalColorClamp;\r\n defines.IMAGEPROCESSINGPOSTPROCESS = this.applyByPostProcess && this._isEnabled;\r\n return;\r\n }\r\n\r\n defines.VIGNETTE = this.vignetteEnabled;\r\n defines.VIGNETTEBLENDMODEMULTIPLY = this.vignetteBlendMode === ImageProcessingConfiguration._VIGNETTEMODE_MULTIPLY;\r\n defines.VIGNETTEBLENDMODEOPAQUE = !defines.VIGNETTEBLENDMODEMULTIPLY;\r\n\r\n defines.TONEMAPPING = this.toneMappingEnabled;\r\n switch (this._toneMappingType) {\r\n case ImageProcessingConfiguration.TONEMAPPING_ACES:\r\n defines.TONEMAPPING_ACES = true;\r\n break;\r\n default:\r\n defines.TONEMAPPING_ACES = false;\r\n break;\r\n }\r\n\r\n defines.CONTRAST = this.contrast !== 1.0;\r\n defines.EXPOSURE = this.exposure !== 1.0;\r\n defines.COLORCURVES = this.colorCurvesEnabled && !!this.colorCurves;\r\n defines.COLORGRADING = this.colorGradingEnabled && !!this.colorGradingTexture;\r\n if (defines.COLORGRADING) {\r\n defines.COLORGRADING3D = this.colorGradingTexture!.is3D;\r\n } else {\r\n defines.COLORGRADING3D = false;\r\n }\r\n defines.SAMPLER3DGREENDEPTH = this.colorGradingWithGreenDepth;\r\n defines.SAMPLER3DBGRMAP = this.colorGradingBGR;\r\n defines.DITHER = this._ditheringEnabled;\r\n defines.IMAGEPROCESSINGPOSTPROCESS = this.applyByPostProcess;\r\n defines.SKIPFINALCOLORCLAMP = this.skipFinalColorClamp;\r\n defines.IMAGEPROCESSING = defines.VIGNETTE || defines.TONEMAPPING || defines.CONTRAST || defines.EXPOSURE || defines.COLORCURVES || defines.COLORGRADING || defines.DITHER;\r\n }\r\n\r\n /**\r\n * Returns true if all the image processing information are ready.\r\n * @returns True if ready, otherwise, false\r\n */\r\n public isReady() {\r\n // Color Grading texture can not be none blocking.\r\n return !this.colorGradingEnabled || !this.colorGradingTexture || this.colorGradingTexture.isReady();\r\n }\r\n\r\n /**\r\n * Binds the image processing to the shader.\r\n * @param effect The effect to bind to\r\n * @param overrideAspectRatio Override the aspect ratio of the effect\r\n */\r\n public bind(effect: Effect, overrideAspectRatio?: number): void {\r\n // Color Curves\r\n if (this._colorCurvesEnabled && this.colorCurves) {\r\n ColorCurves.Bind(this.colorCurves, effect);\r\n }\r\n\r\n // Vignette and dither handled together due to common uniform.\r\n if (this._vignetteEnabled || this._ditheringEnabled) {\r\n const inverseWidth = 1 / effect.getEngine().getRenderWidth();\r\n const inverseHeight = 1 / effect.getEngine().getRenderHeight();\r\n effect.setFloat2(\"vInverseScreenSize\", inverseWidth, inverseHeight);\r\n\r\n if (this._ditheringEnabled) {\r\n effect.setFloat(\"ditherIntensity\", 0.5 * this._ditheringIntensity);\r\n }\r\n\r\n if (this._vignetteEnabled) {\r\n const aspectRatio = overrideAspectRatio != null ? overrideAspectRatio : inverseHeight / inverseWidth;\r\n\r\n let vignetteScaleY = Math.tan(this.vignetteCameraFov * 0.5);\r\n let vignetteScaleX = vignetteScaleY * aspectRatio;\r\n\r\n const vignetteScaleGeometricMean = Math.sqrt(vignetteScaleX * vignetteScaleY);\r\n vignetteScaleX = Tools.Mix(vignetteScaleX, vignetteScaleGeometricMean, this.vignetteStretch);\r\n vignetteScaleY = Tools.Mix(vignetteScaleY, vignetteScaleGeometricMean, this.vignetteStretch);\r\n\r\n effect.setFloat4(\"vignetteSettings1\", vignetteScaleX, vignetteScaleY, -vignetteScaleX * this.vignetteCenterX, -vignetteScaleY * this.vignetteCenterY);\r\n\r\n const vignettePower = -2.0 * this.vignetteWeight;\r\n effect.setFloat4(\"vignetteSettings2\", this.vignetteColor.r, this.vignetteColor.g, this.vignetteColor.b, vignettePower);\r\n }\r\n }\r\n\r\n // Exposure\r\n effect.setFloat(\"exposureLinear\", this.exposure);\r\n\r\n // Contrast\r\n effect.setFloat(\"contrast\", this.contrast);\r\n\r\n // Color transform settings\r\n if (this.colorGradingTexture) {\r\n effect.setTexture(\"txColorTransform\", this.colorGradingTexture);\r\n const textureSize = this.colorGradingTexture.getSize().height;\r\n\r\n effect.setFloat4(\r\n \"colorTransformSettings\",\r\n (textureSize - 1) / textureSize, // textureScale\r\n 0.5 / textureSize, // textureOffset\r\n textureSize, // textureSize\r\n this.colorGradingTexture.level // weight\r\n );\r\n }\r\n }\r\n\r\n /**\r\n * Clones the current image processing instance.\r\n * @returns The cloned image processing\r\n */\r\n public clone(): ImageProcessingConfiguration {\r\n return SerializationHelper.Clone(() => new ImageProcessingConfiguration(), this);\r\n }\r\n\r\n /**\r\n * Serializes the current image processing instance to a json representation.\r\n * @returns a JSON representation\r\n */\r\n public serialize(): any {\r\n return SerializationHelper.Serialize(this);\r\n }\r\n\r\n /**\r\n * Parses the image processing from a json representation.\r\n * @param source the JSON source to parse\r\n * @returns The parsed image processing\r\n */\r\n public static Parse(source: any): ImageProcessingConfiguration {\r\n const parsed = SerializationHelper.Parse(() => new ImageProcessingConfiguration(), source, null, null);\r\n // Backward compatibility\r\n if (source.vignetteCentreX !== undefined) {\r\n parsed.vignetteCenterX = source.vignetteCentreX;\r\n }\r\n if (source.vignetteCentreY !== undefined) {\r\n parsed.vignetteCenterY = source.vignetteCentreY;\r\n }\r\n\r\n return parsed;\r\n }\r\n\r\n // Static constants associated to the image processing.\r\n private static _VIGNETTEMODE_MULTIPLY = 0;\r\n private static _VIGNETTEMODE_OPAQUE = 1;\r\n\r\n /**\r\n * Used to apply the vignette as a mix with the pixel color.\r\n */\r\n public static get VIGNETTEMODE_MULTIPLY(): number {\r\n return this._VIGNETTEMODE_MULTIPLY;\r\n }\r\n\r\n /**\r\n * Used to apply the vignette as a replacement of the pixel color.\r\n */\r\n public static get VIGNETTEMODE_OPAQUE(): number {\r\n return this._VIGNETTEMODE_OPAQUE;\r\n }\r\n}\r\n\r\n// References the dependencies.\r\nSerializationHelper._ImageProcessingConfigurationParser = ImageProcessingConfiguration.Parse;\r\n","/**\r\n * Enum for Device Types\r\n */\r\nexport enum DeviceType {\r\n /** Generic */\r\n Generic = 0,\r\n /** Keyboard */\r\n Keyboard = 1,\r\n /** Mouse */\r\n Mouse = 2,\r\n /** Touch Pointers */\r\n Touch = 3,\r\n /** PS4 Dual Shock */\r\n DualShock = 4,\r\n /** Xbox */\r\n Xbox = 5,\r\n /** Switch Controller */\r\n Switch = 6,\r\n /** PS5 DualSense */\r\n DualSense = 7,\r\n}\r\n\r\n// Device Enums\r\n/**\r\n * Enum for All Pointers (Touch/Mouse)\r\n */\r\nexport enum PointerInput {\r\n /** Horizontal Axis (Not used in events/observables; only in polling) */\r\n Horizontal = 0,\r\n /** Vertical Axis (Not used in events/observables; only in polling) */\r\n Vertical = 1,\r\n /** Left Click or Touch */\r\n LeftClick = 2,\r\n /** Middle Click */\r\n MiddleClick = 3,\r\n /** Right Click */\r\n RightClick = 4,\r\n /** Browser Back */\r\n BrowserBack = 5,\r\n /** Browser Forward */\r\n BrowserForward = 6,\r\n /** Mouse Wheel X */\r\n MouseWheelX = 7,\r\n /** Mouse Wheel Y */\r\n MouseWheelY = 8,\r\n /** Mouse Wheel Z */\r\n MouseWheelZ = 9,\r\n /** Used in events/observables to identify if x/y changes occurred */\r\n Move = 12,\r\n}\r\n\r\n/** @internal */\r\nexport enum NativePointerInput {\r\n /** Horizontal Axis */\r\n Horizontal = PointerInput.Horizontal,\r\n /** Vertical Axis */\r\n Vertical = 1,\r\n /** Left Click or Touch */\r\n LeftClick = 2,\r\n /** Middle Click */\r\n MiddleClick = 3,\r\n /** Right Click */\r\n RightClick = 4,\r\n /** Browser Back */\r\n BrowserBack = 5,\r\n /** Browser Forward */\r\n BrowserForward = 6,\r\n /** Mouse Wheel X */\r\n MouseWheelX = 7,\r\n /** Mouse Wheel Y */\r\n MouseWheelY = 8,\r\n /** Mouse Wheel Z */\r\n MouseWheelZ = 9,\r\n /** Delta X */\r\n DeltaHorizontal = 10,\r\n /** Delta Y */\r\n DeltaVertical = 11,\r\n}\r\n\r\n/**\r\n * Enum for Dual Shock Gamepad\r\n */\r\nexport enum DualShockInput {\r\n /** Cross */\r\n Cross = 0,\r\n /** Circle */\r\n Circle = 1,\r\n /** Square */\r\n Square = 2,\r\n /** Triangle */\r\n Triangle = 3,\r\n /** L1 */\r\n L1 = 4,\r\n /** R1 */\r\n R1 = 5,\r\n /** L2 */\r\n L2 = 6,\r\n /** R2 */\r\n R2 = 7,\r\n /** Share */\r\n Share = 8,\r\n /** Options */\r\n Options = 9,\r\n /** L3 */\r\n L3 = 10,\r\n /** R3 */\r\n R3 = 11,\r\n /** DPadUp */\r\n DPadUp = 12,\r\n /** DPadDown */\r\n DPadDown = 13,\r\n /** DPadLeft */\r\n DPadLeft = 14,\r\n /** DRight */\r\n DPadRight = 15,\r\n /** Home */\r\n Home = 16,\r\n /** TouchPad */\r\n TouchPad = 17,\r\n /** LStickXAxis */\r\n LStickXAxis = 18,\r\n /** LStickYAxis */\r\n LStickYAxis = 19,\r\n /** RStickXAxis */\r\n RStickXAxis = 20,\r\n /** RStickYAxis */\r\n RStickYAxis = 21,\r\n}\r\n\r\n/**\r\n * Enum for Dual Sense Gamepad\r\n */\r\nexport enum DualSenseInput {\r\n /** Cross */\r\n Cross = 0,\r\n /** Circle */\r\n Circle = 1,\r\n /** Square */\r\n Square = 2,\r\n /** Triangle */\r\n Triangle = 3,\r\n /** L1 */\r\n L1 = 4,\r\n /** R1 */\r\n R1 = 5,\r\n /** L2 */\r\n L2 = 6,\r\n /** R2 */\r\n R2 = 7,\r\n /** Create */\r\n Create = 8,\r\n /** Options */\r\n Options = 9,\r\n /** L3 */\r\n L3 = 10,\r\n /** R3 */\r\n R3 = 11,\r\n /** DPadUp */\r\n DPadUp = 12,\r\n /** DPadDown */\r\n DPadDown = 13,\r\n /** DPadLeft */\r\n DPadLeft = 14,\r\n /** DRight */\r\n DPadRight = 15,\r\n /** Home */\r\n Home = 16,\r\n /** TouchPad */\r\n TouchPad = 17,\r\n /** LStickXAxis */\r\n LStickXAxis = 18,\r\n /** LStickYAxis */\r\n LStickYAxis = 19,\r\n /** RStickXAxis */\r\n RStickXAxis = 20,\r\n /** RStickYAxis */\r\n RStickYAxis = 21,\r\n}\r\n\r\n/**\r\n * Enum for Xbox Gamepad\r\n */\r\nexport enum XboxInput {\r\n /** A */\r\n A = 0,\r\n /** B */\r\n B = 1,\r\n /** X */\r\n X = 2,\r\n /** Y */\r\n Y = 3,\r\n /** LB */\r\n LB = 4,\r\n /** RB */\r\n RB = 5,\r\n /** LT */\r\n LT = 6,\r\n /** RT */\r\n RT = 7,\r\n /** Back */\r\n Back = 8,\r\n /** Start */\r\n Start = 9,\r\n /** LS */\r\n LS = 10,\r\n /** RS */\r\n RS = 11,\r\n /** DPadUp */\r\n DPadUp = 12,\r\n /** DPadDown */\r\n DPadDown = 13,\r\n /** DPadLeft */\r\n DPadLeft = 14,\r\n /** DRight */\r\n DPadRight = 15,\r\n /** Home */\r\n Home = 16,\r\n /** LStickXAxis */\r\n LStickXAxis = 17,\r\n /** LStickYAxis */\r\n LStickYAxis = 18,\r\n /** RStickXAxis */\r\n RStickXAxis = 19,\r\n /** RStickYAxis */\r\n RStickYAxis = 20,\r\n}\r\n\r\n/**\r\n * Enum for Switch (Pro/JoyCon L+R) Gamepad\r\n */\r\nexport enum SwitchInput {\r\n /** B */\r\n B = 0,\r\n /** A */\r\n A = 1,\r\n /** Y */\r\n Y = 2,\r\n /** X */\r\n X = 3,\r\n /** L */\r\n L = 4,\r\n /** R */\r\n R = 5,\r\n /** ZL */\r\n ZL = 6,\r\n /** ZR */\r\n ZR = 7,\r\n /** Minus */\r\n Minus = 8,\r\n /** Plus */\r\n Plus = 9,\r\n /** LS */\r\n LS = 10,\r\n /** RS */\r\n RS = 11,\r\n /** DPadUp */\r\n DPadUp = 12,\r\n /** DPadDown */\r\n DPadDown = 13,\r\n /** DPadLeft */\r\n DPadLeft = 14,\r\n /** DRight */\r\n DPadRight = 15,\r\n /** Home */\r\n Home = 16,\r\n /** Capture */\r\n Capture = 17,\r\n /** LStickXAxis */\r\n LStickXAxis = 18,\r\n /** LStickYAxis */\r\n LStickYAxis = 19,\r\n /** RStickXAxis */\r\n RStickXAxis = 20,\r\n /** RStickYAxis */\r\n RStickYAxis = 21,\r\n}\r\n","import type { PointerInput } from \"../DeviceInput/InputDevices/deviceEnums\";\r\n\r\n/**\r\n * Event Types\r\n */\r\nexport enum DeviceInputEventType {\r\n // Pointers\r\n /** PointerMove */\r\n PointerMove,\r\n /** PointerDown */\r\n PointerDown,\r\n /** PointerUp */\r\n PointerUp,\r\n}\r\n\r\n/**\r\n * Native friendly interface for Event Object\r\n */\r\nexport interface IUIEvent {\r\n /**\r\n * Input array index\r\n */\r\n inputIndex: number;\r\n\r\n /**\r\n * Current target for an event\r\n */\r\n currentTarget?: any;\r\n\r\n /**\r\n * Alias for target\r\n * @deprecated Use target instead\r\n */\r\n srcElement?: any;\r\n\r\n /**\r\n * Type of event\r\n */\r\n type: string;\r\n\r\n /**\r\n * Reference to object where object was dispatched\r\n */\r\n target: any;\r\n\r\n /**\r\n * Tells user agent what to do when not explicitly handled\r\n */\r\n preventDefault: () => void;\r\n}\r\n\r\n/**\r\n * Native friendly interface for KeyboardEvent Object\r\n */\r\nexport interface IKeyboardEvent extends IUIEvent {\r\n /**\r\n * Status of Alt key being pressed\r\n */\r\n altKey: boolean;\r\n\r\n /**\r\n * Unicode value of character pressed\r\n * @deprecated Required for event, use keyCode instead.\r\n */\r\n charCode?: number;\r\n\r\n /**\r\n * Code for key based on layout\r\n */\r\n code: string;\r\n\r\n /**\r\n * Status of Ctrl key being pressed\r\n */\r\n ctrlKey: boolean;\r\n\r\n /**\r\n * String representation of key\r\n */\r\n key: string;\r\n /**\r\n * ASCII value of key\r\n * @deprecated Used with DeviceSourceManager\r\n */\r\n keyCode: number;\r\n\r\n /**\r\n * Status of Meta key (eg. Windows key) being pressed\r\n */\r\n metaKey: boolean;\r\n\r\n /**\r\n * Status of Shift key being pressed\r\n */\r\n shiftKey: boolean;\r\n}\r\n\r\n/**\r\n * Native friendly interface for MouseEvent Object\r\n */\r\nexport interface IMouseEvent extends IUIEvent {\r\n /**\r\n * Subset of possible PointerInput values for events, excluding ones that CANNOT be in events organically\r\n */\r\n inputIndex: Exclude;\r\n\r\n /**\r\n * Status of Alt key being pressed\r\n */\r\n altKey: boolean;\r\n\r\n /**\r\n * Value of single mouse button pressed\r\n */\r\n button: number;\r\n\r\n /**\r\n * Value of all mouse buttons pressed\r\n */\r\n buttons: number;\r\n\r\n /**\r\n * Current X coordinate\r\n */\r\n clientX: number;\r\n\r\n /**\r\n * Current Y coordinate\r\n */\r\n clientY: number;\r\n\r\n /**\r\n * Status of Ctrl key being pressed\r\n */\r\n ctrlKey: boolean;\r\n\r\n /**\r\n * Provides current click count\r\n */\r\n detail?: number;\r\n\r\n /**\r\n * Status of Meta key (eg. Windows key) being pressed\r\n */\r\n metaKey: boolean;\r\n\r\n /**\r\n * Delta of movement on X axis\r\n */\r\n movementX: number;\r\n\r\n /**\r\n * Delta of movement on Y axis\r\n */\r\n movementY: number;\r\n\r\n /**\r\n * Delta of movement on X axis\r\n * @deprecated Use 'movementX' instead\r\n */\r\n mozMovementX?: number;\r\n\r\n /**\r\n * Delta of movement on Y axis\r\n * @deprecated Use 'movementY' instead\r\n */\r\n mozMovementY?: number;\r\n\r\n /**\r\n * Delta of movement on X axis\r\n * @deprecated Use 'movementX' instead\r\n */\r\n msMovementX?: number;\r\n\r\n /**\r\n * Delta of movement on Y axis\r\n * @deprecated Use 'movementY' instead\r\n */\r\n msMovementY?: number;\r\n\r\n /**\r\n * Current coordinate of X within container\r\n */\r\n offsetX: number;\r\n\r\n /**\r\n * Current coordinate of Y within container\r\n */\r\n offsetY: number;\r\n\r\n /**\r\n * Horizontal coordinate of event\r\n */\r\n pageX: number;\r\n\r\n /**\r\n * Vertical coordinate of event\r\n */\r\n pageY: number;\r\n\r\n /**\r\n * Status of Shift key being pressed\r\n */\r\n shiftKey: boolean;\r\n\r\n /**\r\n * Delta of movement on X axis\r\n * @deprecated Use 'movementX' instead\r\n */\r\n webkitMovementX?: number;\r\n\r\n /**\r\n * Delta of movement on Y axis\r\n * @deprecated Use 'movementY' instead\r\n */\r\n webkitMovementY?: number;\r\n\r\n /**\r\n * Alias of clientX\r\n */\r\n x: number;\r\n\r\n /**\r\n * Alias of clientY\r\n */\r\n y: number;\r\n}\r\n\r\n/**\r\n * Native friendly interface for PointerEvent Object\r\n */\r\nexport interface IPointerEvent extends IMouseEvent {\r\n /**\r\n * Subset of possible PointerInput values for events, excluding ones that CANNOT be in events organically and mouse wheel values\r\n */\r\n inputIndex: Exclude;\r\n\r\n /**\r\n * Pointer Event ID\r\n */\r\n pointerId: number;\r\n\r\n /**\r\n * Type of pointer\r\n */\r\n pointerType: string;\r\n}\r\n\r\n/**\r\n * Native friendly interface for WheelEvent Object\r\n */\r\nexport interface IWheelEvent extends IMouseEvent {\r\n /**\r\n * Subset of possible PointerInput values for events that can only be used with mouse wheel\r\n */\r\n inputIndex: PointerInput.MouseWheelX | PointerInput.MouseWheelY | PointerInput.MouseWheelZ;\r\n\r\n /**\r\n * Units for delta value\r\n */\r\n deltaMode: number;\r\n\r\n /**\r\n * Horizontal scroll delta\r\n */\r\n deltaX: number;\r\n\r\n /**\r\n * Vertical scroll delta\r\n */\r\n deltaY: number;\r\n\r\n /**\r\n * Z-Axis scroll delta\r\n */\r\n deltaZ: number;\r\n\r\n /**\r\n * WheelDelta (From MouseWheel Event)\r\n * @deprecated\r\n */\r\n wheelDelta?: number;\r\n}\r\n\r\n/**\r\n * Constants used for Events\r\n */\r\nexport class EventConstants {\r\n /**\r\n * Pixel delta for Wheel Events (Default)\r\n */\r\n public static DOM_DELTA_PIXEL = 0x00;\r\n\r\n /**\r\n * Line delta for Wheel Events\r\n */\r\n public static DOM_DELTA_LINE = 0x01;\r\n\r\n /**\r\n * Page delta for Wheel Events\r\n */\r\n public static DOM_DELTA_PAGE = 0x02;\r\n}\r\n","import { ThinEngine } from \"../../Engines/thinEngine\";\r\nimport type { FloatArray, Nullable } from \"../../types\";\r\nimport type { DataBuffer } from \"../../Buffers/dataBuffer\";\r\nimport { WebGLDataBuffer } from \"../../Meshes/WebGL/webGLDataBuffer\";\r\nimport type { IPipelineContext } from \"../IPipelineContext\";\r\nimport type { WebGLPipelineContext } from \"../WebGL/webGLPipelineContext\";\r\n\r\ndeclare module \"../../Engines/thinEngine\" {\r\n export interface ThinEngine {\r\n /**\r\n * Create an uniform buffer\r\n * @see https://doc.babylonjs.com/setup/support/webGL2#uniform-buffer-objets\r\n * @param elements defines the content of the uniform buffer\r\n * @returns the webGL uniform buffer\r\n */\r\n createUniformBuffer(elements: FloatArray): DataBuffer;\r\n\r\n /**\r\n * Create a dynamic uniform buffer\r\n * @see https://doc.babylonjs.com/setup/support/webGL2#uniform-buffer-objets\r\n * @param elements defines the content of the uniform buffer\r\n * @returns the webGL uniform buffer\r\n */\r\n createDynamicUniformBuffer(elements: FloatArray): DataBuffer;\r\n\r\n /**\r\n * Update an existing uniform buffer\r\n * @see https://doc.babylonjs.com/setup/support/webGL2#uniform-buffer-objets\r\n * @param uniformBuffer defines the target uniform buffer\r\n * @param elements defines the content to update\r\n * @param offset defines the offset in the uniform buffer where update should start\r\n * @param count defines the size of the data to update\r\n */\r\n updateUniformBuffer(uniformBuffer: DataBuffer, elements: FloatArray, offset?: number, count?: number): void;\r\n\r\n /**\r\n * Bind an uniform buffer to the current webGL context\r\n * @param buffer defines the buffer to bind\r\n */\r\n bindUniformBuffer(buffer: Nullable): void;\r\n\r\n /**\r\n * Bind a buffer to the current webGL context at a given location\r\n * @param buffer defines the buffer to bind\r\n * @param location defines the index where to bind the buffer\r\n * @param name Name of the uniform variable to bind\r\n */\r\n bindUniformBufferBase(buffer: DataBuffer, location: number, name: string): void;\r\n\r\n /**\r\n * Bind a specific block at a given index in a specific shader program\r\n * @param pipelineContext defines the pipeline context to use\r\n * @param blockName defines the block name\r\n * @param index defines the index where to bind the block\r\n */\r\n bindUniformBlock(pipelineContext: IPipelineContext, blockName: string, index: number): void;\r\n }\r\n}\r\n\r\nThinEngine.prototype.createUniformBuffer = function (elements: FloatArray): DataBuffer {\r\n const ubo = this._gl.createBuffer();\r\n\r\n if (!ubo) {\r\n throw new Error(\"Unable to create uniform buffer\");\r\n }\r\n const result = new WebGLDataBuffer(ubo);\r\n\r\n this.bindUniformBuffer(result);\r\n\r\n if (elements instanceof Float32Array) {\r\n this._gl.bufferData(this._gl.UNIFORM_BUFFER, elements, this._gl.STATIC_DRAW);\r\n } else {\r\n this._gl.bufferData(this._gl.UNIFORM_BUFFER, new Float32Array(elements), this._gl.STATIC_DRAW);\r\n }\r\n\r\n this.bindUniformBuffer(null);\r\n\r\n result.references = 1;\r\n return result;\r\n};\r\n\r\nThinEngine.prototype.createDynamicUniformBuffer = function (elements: FloatArray): DataBuffer {\r\n const ubo = this._gl.createBuffer();\r\n\r\n if (!ubo) {\r\n throw new Error(\"Unable to create dynamic uniform buffer\");\r\n }\r\n\r\n const result = new WebGLDataBuffer(ubo);\r\n this.bindUniformBuffer(result);\r\n\r\n if (elements instanceof Float32Array) {\r\n this._gl.bufferData(this._gl.UNIFORM_BUFFER, elements, this._gl.DYNAMIC_DRAW);\r\n } else {\r\n this._gl.bufferData(this._gl.UNIFORM_BUFFER, new Float32Array(elements), this._gl.DYNAMIC_DRAW);\r\n }\r\n\r\n this.bindUniformBuffer(null);\r\n\r\n result.references = 1;\r\n return result;\r\n};\r\n\r\nThinEngine.prototype.updateUniformBuffer = function (uniformBuffer: DataBuffer, elements: FloatArray, offset?: number, count?: number): void {\r\n this.bindUniformBuffer(uniformBuffer);\r\n\r\n if (offset === undefined) {\r\n offset = 0;\r\n }\r\n\r\n if (count === undefined) {\r\n if (elements instanceof Float32Array) {\r\n this._gl.bufferSubData(this._gl.UNIFORM_BUFFER, offset, elements);\r\n } else {\r\n this._gl.bufferSubData(this._gl.UNIFORM_BUFFER, offset, new Float32Array(elements));\r\n }\r\n } else {\r\n if (elements instanceof Float32Array) {\r\n this._gl.bufferSubData(this._gl.UNIFORM_BUFFER, 0, elements.subarray(offset, offset + count));\r\n } else {\r\n this._gl.bufferSubData(this._gl.UNIFORM_BUFFER, 0, new Float32Array(elements).subarray(offset, offset + count));\r\n }\r\n }\r\n\r\n this.bindUniformBuffer(null);\r\n};\r\n\r\nThinEngine.prototype.bindUniformBuffer = function (buffer: Nullable): void {\r\n this._gl.bindBuffer(this._gl.UNIFORM_BUFFER, buffer ? buffer.underlyingResource : null);\r\n};\r\n\r\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\r\nThinEngine.prototype.bindUniformBufferBase = function (buffer: DataBuffer, location: number, name: string): void {\r\n this._gl.bindBufferBase(this._gl.UNIFORM_BUFFER, location, buffer ? buffer.underlyingResource : null);\r\n};\r\n\r\nThinEngine.prototype.bindUniformBlock = function (pipelineContext: IPipelineContext, blockName: string, index: number): void {\r\n const program = (pipelineContext as WebGLPipelineContext).program!;\r\n\r\n const uniformLocation = this._gl.getUniformBlockIndex(program, blockName);\r\n\r\n if (uniformLocation !== 0xffffffff) {\r\n this._gl.uniformBlockBinding(program, uniformLocation, index);\r\n }\r\n};\r\n","import { Logger } from \"../Misc/logger\";\r\nimport type { Nullable, FloatArray } from \"../types\";\r\nimport type { IMatrixLike, IVector3Like, IVector4Like, IColor3Like, IColor4Like } from \"../Maths/math.like\";\r\nimport type { Effect } from \"./effect\";\r\nimport type { ThinTexture } from \"../Materials/Textures/thinTexture\";\r\nimport type { DataBuffer } from \"../Buffers/dataBuffer\";\r\nimport type { ThinEngine } from \"../Engines/thinEngine\";\r\nimport { Tools } from \"../Misc/tools\";\r\n\r\nimport \"../Engines/Extensions/engine.uniformBuffer\";\r\n\r\n/**\r\n * Uniform buffer objects.\r\n *\r\n * Handles blocks of uniform on the GPU.\r\n *\r\n * If WebGL 2 is not available, this class falls back on traditional setUniformXXX calls.\r\n *\r\n * For more information, please refer to :\r\n * https://www.khronos.org/opengl/wiki/Uniform_Buffer_Object\r\n */\r\nexport class UniformBuffer {\r\n /** @internal */\r\n public static _UpdatedUbosInFrame: { [name: string]: number } = {};\r\n\r\n private _engine: ThinEngine;\r\n private _buffer: Nullable;\r\n private _buffers: Array<[DataBuffer, Float32Array | undefined]>;\r\n private _bufferIndex: number;\r\n private _createBufferOnWrite: boolean;\r\n private _data: number[];\r\n private _bufferData: Float32Array;\r\n private _dynamic?: boolean;\r\n private _uniformLocations: { [key: string]: number };\r\n private _uniformSizes: { [key: string]: number };\r\n private _uniformArraySizes: { [key: string]: { strideSize: number; arraySize: number } };\r\n private _uniformLocationPointer: number;\r\n private _needSync: boolean;\r\n private _noUBO: boolean;\r\n private _currentEffect: Effect;\r\n private _currentEffectName: string;\r\n private _name: string;\r\n private _currentFrameId: number;\r\n\r\n // Pool for avoiding memory leaks\r\n private static _MAX_UNIFORM_SIZE = 256;\r\n private static _TempBuffer = new Float32Array(UniformBuffer._MAX_UNIFORM_SIZE);\r\n private static _TempBufferInt32View = new Int32Array(UniformBuffer._TempBuffer.buffer);\r\n private static _TempBufferUInt32View = new Uint32Array(UniformBuffer._TempBuffer.buffer);\r\n\r\n /**\r\n * Lambda to Update a 3x3 Matrix in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateMatrix3x3: (name: string, matrix: Float32Array) => void;\r\n\r\n /**\r\n * Lambda to Update a 2x2 Matrix in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateMatrix2x2: (name: string, matrix: Float32Array) => void;\r\n\r\n /**\r\n * Lambda to Update a single float in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateFloat: (name: string, x: number) => void;\r\n\r\n /**\r\n * Lambda to Update a vec2 of float in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateFloat2: (name: string, x: number, y: number, suffix?: string) => void;\r\n\r\n /**\r\n * Lambda to Update a vec3 of float in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateFloat3: (name: string, x: number, y: number, z: number, suffix?: string) => void;\r\n\r\n /**\r\n * Lambda to Update a vec4 of float in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateFloat4: (name: string, x: number, y: number, z: number, w: number, suffix?: string) => void;\r\n\r\n /**\r\n * Lambda to Update an array of float in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateFloatArray: (name: string, array: Float32Array) => void;\r\n\r\n /**\r\n * Lambda to Update an array of number in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateArray: (name: string, array: number[]) => void;\r\n\r\n /**\r\n * Lambda to Update an array of number in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateIntArray: (name: string, array: Int32Array) => void;\r\n\r\n /**\r\n * Lambda to Update an array of number in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateUIntArray: (name: string, array: Uint32Array) => void;\r\n\r\n /**\r\n * Lambda to Update a 4x4 Matrix in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateMatrix: (name: string, mat: IMatrixLike) => void;\r\n\r\n /**\r\n * Lambda to Update an array of 4x4 Matrix in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateMatrices: (name: string, mat: Float32Array) => void;\r\n\r\n /**\r\n * Lambda to Update vec3 of float from a Vector in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateVector3: (name: string, vector: IVector3Like) => void;\r\n\r\n /**\r\n * Lambda to Update vec4 of float from a Vector in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateVector4: (name: string, vector: IVector4Like) => void;\r\n\r\n /**\r\n * Lambda to Update vec3 of float from a Color in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateColor3: (name: string, color: IColor3Like, suffix?: string) => void;\r\n\r\n /**\r\n * Lambda to Update vec4 of float from a Color in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateColor4: (name: string, color: IColor3Like, alpha: number, suffix?: string) => void;\r\n\r\n /**\r\n * Lambda to Update vec4 of float from a Color in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateDirectColor4: (name: string, color: IColor4Like, suffix?: string) => void;\r\n\r\n /**\r\n * Lambda to Update a int a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateInt: (name: string, x: number, suffix?: string) => void;\r\n\r\n /**\r\n * Lambda to Update a vec2 of int in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateInt2: (name: string, x: number, y: number, suffix?: string) => void;\r\n\r\n /**\r\n * Lambda to Update a vec3 of int in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateInt3: (name: string, x: number, y: number, z: number, suffix?: string) => void;\r\n\r\n /**\r\n * Lambda to Update a vec4 of int in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateInt4: (name: string, x: number, y: number, z: number, w: number, suffix?: string) => void;\r\n\r\n /**\r\n * Lambda to Update a unsigned int a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateUInt: (name: string, x: number, suffix?: string) => void;\r\n\r\n /**\r\n * Lambda to Update a vec2 of unsigned int in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateUInt2: (name: string, x: number, y: number, suffix?: string) => void;\r\n\r\n /**\r\n * Lambda to Update a vec3 of unsigned int in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateUInt3: (name: string, x: number, y: number, z: number, suffix?: string) => void;\r\n\r\n /**\r\n * Lambda to Update a vec4 of unsigned int in a uniform buffer.\r\n * This is dynamic to allow compat with webgl 1 and 2.\r\n * You will need to pass the name of the uniform as well as the value.\r\n */\r\n public updateUInt4: (name: string, x: number, y: number, z: number, w: number, suffix?: string) => void;\r\n\r\n /**\r\n * Instantiates a new Uniform buffer objects.\r\n *\r\n * Handles blocks of uniform on the GPU.\r\n *\r\n * If WebGL 2 is not available, this class falls back on traditional setUniformXXX calls.\r\n *\r\n * For more information, please refer to :\r\n * @see https://www.khronos.org/opengl/wiki/Uniform_Buffer_Object\r\n * @param engine Define the engine the buffer is associated with\r\n * @param data Define the data contained in the buffer\r\n * @param dynamic Define if the buffer is updatable\r\n * @param name to assign to the buffer (debugging purpose)\r\n * @param forceNoUniformBuffer define that this object must not rely on UBO objects\r\n */\r\n constructor(engine: ThinEngine, data?: number[], dynamic?: boolean, name?: string, forceNoUniformBuffer = false) {\r\n this._engine = engine;\r\n this._noUBO = !engine.supportsUniformBuffers || forceNoUniformBuffer;\r\n this._dynamic = dynamic;\r\n this._name = name ?? \"no-name\";\r\n\r\n this._data = data || [];\r\n\r\n this._uniformLocations = {};\r\n this._uniformSizes = {};\r\n this._uniformArraySizes = {};\r\n this._uniformLocationPointer = 0;\r\n this._needSync = false;\r\n\r\n if (this._engine._features.trackUbosInFrame) {\r\n this._buffers = [];\r\n this._bufferIndex = -1;\r\n this._createBufferOnWrite = false;\r\n this._currentFrameId = 0;\r\n }\r\n\r\n if (this._noUBO) {\r\n this.updateMatrix3x3 = this._updateMatrix3x3ForEffect;\r\n this.updateMatrix2x2 = this._updateMatrix2x2ForEffect;\r\n this.updateFloat = this._updateFloatForEffect;\r\n this.updateFloat2 = this._updateFloat2ForEffect;\r\n this.updateFloat3 = this._updateFloat3ForEffect;\r\n this.updateFloat4 = this._updateFloat4ForEffect;\r\n this.updateFloatArray = this._updateFloatArrayForEffect;\r\n this.updateArray = this._updateArrayForEffect;\r\n this.updateIntArray = this._updateIntArrayForEffect;\r\n this.updateUIntArray = this._updateUIntArrayForEffect;\r\n this.updateMatrix = this._updateMatrixForEffect;\r\n this.updateMatrices = this._updateMatricesForEffect;\r\n this.updateVector3 = this._updateVector3ForEffect;\r\n this.updateVector4 = this._updateVector4ForEffect;\r\n this.updateColor3 = this._updateColor3ForEffect;\r\n this.updateColor4 = this._updateColor4ForEffect;\r\n this.updateDirectColor4 = this._updateDirectColor4ForEffect;\r\n this.updateInt = this._updateIntForEffect;\r\n this.updateInt2 = this._updateInt2ForEffect;\r\n this.updateInt3 = this._updateInt3ForEffect;\r\n this.updateInt4 = this._updateInt4ForEffect;\r\n this.updateUInt = this._updateUIntForEffect;\r\n this.updateUInt2 = this._updateUInt2ForEffect;\r\n this.updateUInt3 = this._updateUInt3ForEffect;\r\n this.updateUInt4 = this._updateUInt4ForEffect;\r\n } else {\r\n this._engine._uniformBuffers.push(this);\r\n\r\n this.updateMatrix3x3 = this._updateMatrix3x3ForUniform;\r\n this.updateMatrix2x2 = this._updateMatrix2x2ForUniform;\r\n this.updateFloat = this._updateFloatForUniform;\r\n this.updateFloat2 = this._updateFloat2ForUniform;\r\n this.updateFloat3 = this._updateFloat3ForUniform;\r\n this.updateFloat4 = this._updateFloat4ForUniform;\r\n this.updateFloatArray = this._updateFloatArrayForUniform;\r\n this.updateArray = this._updateArrayForUniform;\r\n this.updateIntArray = this._updateIntArrayForUniform;\r\n this.updateUIntArray = this._updateUIntArrayForUniform;\r\n this.updateMatrix = this._updateMatrixForUniform;\r\n this.updateMatrices = this._updateMatricesForUniform;\r\n this.updateVector3 = this._updateVector3ForUniform;\r\n this.updateVector4 = this._updateVector4ForUniform;\r\n this.updateColor3 = this._updateColor3ForUniform;\r\n this.updateColor4 = this._updateColor4ForUniform;\r\n this.updateDirectColor4 = this._updateDirectColor4ForUniform;\r\n this.updateInt = this._updateIntForUniform;\r\n this.updateInt2 = this._updateInt2ForUniform;\r\n this.updateInt3 = this._updateInt3ForUniform;\r\n this.updateInt4 = this._updateInt4ForUniform;\r\n this.updateUInt = this._updateUIntForUniform;\r\n this.updateUInt2 = this._updateUInt2ForUniform;\r\n this.updateUInt3 = this._updateUInt3ForUniform;\r\n this.updateUInt4 = this._updateUInt4ForUniform;\r\n }\r\n }\r\n\r\n /**\r\n * Indicates if the buffer is using the WebGL2 UBO implementation,\r\n * or just falling back on setUniformXXX calls.\r\n */\r\n public get useUbo(): boolean {\r\n return !this._noUBO;\r\n }\r\n\r\n /**\r\n * Indicates if the WebGL underlying uniform buffer is in sync\r\n * with the javascript cache data.\r\n */\r\n public get isSync(): boolean {\r\n return !this._needSync;\r\n }\r\n\r\n /**\r\n * Indicates if the WebGL underlying uniform buffer is dynamic.\r\n * Also, a dynamic UniformBuffer will disable cache verification and always\r\n * update the underlying WebGL uniform buffer to the GPU.\r\n * @returns if Dynamic, otherwise false\r\n */\r\n public isDynamic(): boolean {\r\n return this._dynamic !== undefined;\r\n }\r\n\r\n /**\r\n * The data cache on JS side.\r\n * @returns the underlying data as a float array\r\n */\r\n public getData(): Float32Array {\r\n return this._bufferData;\r\n }\r\n\r\n /**\r\n * The underlying WebGL Uniform buffer.\r\n * @returns the webgl buffer\r\n */\r\n public getBuffer(): Nullable {\r\n return this._buffer;\r\n }\r\n\r\n /**\r\n * std140 layout specifies how to align data within an UBO structure.\r\n * See https://khronos.org/registry/OpenGL/specs/gl/glspec45.core.pdf#page=159\r\n * for specs.\r\n * @param size\r\n */\r\n private _fillAlignment(size: number) {\r\n // This code has been simplified because we only use floats, vectors of 1, 2, 3, 4 components\r\n // and 4x4 matrices\r\n // TODO : change if other types are used\r\n\r\n let alignment;\r\n if (size <= 2) {\r\n alignment = size;\r\n } else {\r\n alignment = 4;\r\n }\r\n\r\n if (this._uniformLocationPointer % alignment !== 0) {\r\n const oldPointer = this._uniformLocationPointer;\r\n this._uniformLocationPointer += alignment - (this._uniformLocationPointer % alignment);\r\n const diff = this._uniformLocationPointer - oldPointer;\r\n\r\n for (let i = 0; i < diff; i++) {\r\n this._data.push(0);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Adds an uniform in the buffer.\r\n * Warning : the subsequents calls of this function must be in the same order as declared in the shader\r\n * for the layout to be correct ! The addUniform function only handles types like float, vec2, vec3, vec4, mat4,\r\n * meaning size=1,2,3,4 or 16. It does not handle struct types.\r\n * @param name Name of the uniform, as used in the uniform block in the shader.\r\n * @param size Data size, or data directly.\r\n * @param arraySize The number of elements in the array, 0 if not an array.\r\n */\r\n public addUniform(name: string, size: number | number[], arraySize = 0) {\r\n if (this._noUBO) {\r\n return;\r\n }\r\n\r\n if (this._uniformLocations[name] !== undefined) {\r\n // Already existing uniform\r\n return;\r\n }\r\n // This function must be called in the order of the shader layout !\r\n // size can be the size of the uniform, or data directly\r\n let data;\r\n\r\n // std140 FTW...\r\n if (arraySize > 0) {\r\n if (size instanceof Array) {\r\n throw \"addUniform should not be use with Array in UBO: \" + name;\r\n }\r\n\r\n this._fillAlignment(4);\r\n\r\n this._uniformArraySizes[name] = { strideSize: size, arraySize };\r\n if (size == 16) {\r\n size = size * arraySize;\r\n } else {\r\n const perElementPadding = 4 - size;\r\n const totalPadding = perElementPadding * arraySize;\r\n size = size * arraySize + totalPadding;\r\n }\r\n\r\n data = [];\r\n // Fill with zeros\r\n for (let i = 0; i < size; i++) {\r\n data.push(0);\r\n }\r\n } else {\r\n if (size instanceof Array) {\r\n data = size;\r\n size = data.length;\r\n } else {\r\n size = size;\r\n data = [];\r\n\r\n // Fill with zeros\r\n for (let i = 0; i < size; i++) {\r\n data.push(0);\r\n }\r\n }\r\n this._fillAlignment(size);\r\n }\r\n\r\n this._uniformSizes[name] = size;\r\n this._uniformLocations[name] = this._uniformLocationPointer;\r\n this._uniformLocationPointer += size;\r\n\r\n for (let i = 0; i < size; i++) {\r\n this._data.push(data[i]);\r\n }\r\n\r\n this._needSync = true;\r\n }\r\n\r\n /**\r\n * Adds a Matrix 4x4 to the uniform buffer.\r\n * @param name Name of the uniform, as used in the uniform block in the shader.\r\n * @param mat A 4x4 matrix.\r\n */\r\n public addMatrix(name: string, mat: IMatrixLike) {\r\n this.addUniform(name, Array.prototype.slice.call(mat.toArray()));\r\n }\r\n\r\n /**\r\n * Adds a vec2 to the uniform buffer.\r\n * @param name Name of the uniform, as used in the uniform block in the shader.\r\n * @param x Define the x component value of the vec2\r\n * @param y Define the y component value of the vec2\r\n */\r\n public addFloat2(name: string, x: number, y: number) {\r\n const temp = [x, y];\r\n this.addUniform(name, temp);\r\n }\r\n\r\n /**\r\n * Adds a vec3 to the uniform buffer.\r\n * @param name Name of the uniform, as used in the uniform block in the shader.\r\n * @param x Define the x component value of the vec3\r\n * @param y Define the y component value of the vec3\r\n * @param z Define the z component value of the vec3\r\n */\r\n public addFloat3(name: string, x: number, y: number, z: number) {\r\n const temp = [x, y, z];\r\n this.addUniform(name, temp);\r\n }\r\n\r\n /**\r\n * Adds a vec3 to the uniform buffer.\r\n * @param name Name of the uniform, as used in the uniform block in the shader.\r\n * @param color Define the vec3 from a Color\r\n */\r\n public addColor3(name: string, color: IColor3Like) {\r\n const temp = [color.r, color.g, color.b];\r\n this.addUniform(name, temp);\r\n }\r\n\r\n /**\r\n * Adds a vec4 to the uniform buffer.\r\n * @param name Name of the uniform, as used in the uniform block in the shader.\r\n * @param color Define the rgb components from a Color\r\n * @param alpha Define the a component of the vec4\r\n */\r\n public addColor4(name: string, color: IColor3Like, alpha: number) {\r\n const temp = [color.r, color.g, color.b, alpha];\r\n this.addUniform(name, temp);\r\n }\r\n\r\n /**\r\n * Adds a vec3 to the uniform buffer.\r\n * @param name Name of the uniform, as used in the uniform block in the shader.\r\n * @param vector Define the vec3 components from a Vector\r\n */\r\n public addVector3(name: string, vector: IVector3Like) {\r\n const temp = [vector.x, vector.y, vector.z];\r\n this.addUniform(name, temp);\r\n }\r\n\r\n /**\r\n * Adds a Matrix 3x3 to the uniform buffer.\r\n * @param name Name of the uniform, as used in the uniform block in the shader.\r\n */\r\n public addMatrix3x3(name: string) {\r\n this.addUniform(name, 12);\r\n }\r\n\r\n /**\r\n * Adds a Matrix 2x2 to the uniform buffer.\r\n * @param name Name of the uniform, as used in the uniform block in the shader.\r\n */\r\n public addMatrix2x2(name: string) {\r\n this.addUniform(name, 8);\r\n }\r\n\r\n /**\r\n * Effectively creates the WebGL Uniform Buffer, once layout is completed with `addUniform`.\r\n */\r\n public create(): void {\r\n if (this._noUBO) {\r\n return;\r\n }\r\n if (this._buffer) {\r\n return; // nothing to do\r\n }\r\n\r\n // See spec, alignment must be filled as a vec4\r\n this._fillAlignment(4);\r\n this._bufferData = new Float32Array(this._data);\r\n\r\n this._rebuild();\r\n\r\n this._needSync = true;\r\n }\r\n\r\n /** @internal */\r\n public _rebuild(): void {\r\n if (this._noUBO || !this._bufferData) {\r\n return;\r\n }\r\n\r\n if (this._dynamic) {\r\n this._buffer = this._engine.createDynamicUniformBuffer(this._bufferData);\r\n } else {\r\n this._buffer = this._engine.createUniformBuffer(this._bufferData);\r\n }\r\n\r\n if (this._engine._features.trackUbosInFrame) {\r\n this._buffers.push([this._buffer, this._engine._features.checkUbosContentBeforeUpload ? this._bufferData.slice() : undefined]);\r\n this._bufferIndex = this._buffers.length - 1;\r\n this._createBufferOnWrite = false;\r\n }\r\n }\r\n\r\n /** @internal */\r\n public get _numBuffers(): number {\r\n return this._buffers.length;\r\n }\r\n\r\n /** @internal */\r\n public get _indexBuffer(): number {\r\n return this._bufferIndex;\r\n }\r\n\r\n /** Gets the name of this buffer */\r\n public get name(): string {\r\n return this._name;\r\n }\r\n\r\n /** Gets the current effect */\r\n public get currentEffect(): Nullable {\r\n return this._currentEffect;\r\n }\r\n\r\n private _buffersEqual(buf1: Float32Array, buf2: Float32Array): boolean {\r\n for (let i = 0; i < buf1.length; ++i) {\r\n if (buf1[i] !== buf2[i]) {\r\n return false;\r\n }\r\n }\r\n return true;\r\n }\r\n\r\n private _copyBuffer(src: Float32Array, dst: Float32Array): void {\r\n for (let i = 0; i < src.length; ++i) {\r\n dst[i] = src[i];\r\n }\r\n }\r\n\r\n /**\r\n * Updates the WebGL Uniform Buffer on the GPU.\r\n * If the `dynamic` flag is set to true, no cache comparison is done.\r\n * Otherwise, the buffer will be updated only if the cache differs.\r\n */\r\n public update(): void {\r\n if (this._noUBO) {\r\n return;\r\n }\r\n\r\n this.bindUniformBuffer();\r\n\r\n if (!this._buffer) {\r\n this.create();\r\n return;\r\n }\r\n\r\n if (!this._dynamic && !this._needSync) {\r\n this._createBufferOnWrite = this._engine._features.trackUbosInFrame;\r\n return;\r\n }\r\n\r\n if (this._buffers && this._buffers.length > 1 && this._buffers[this._bufferIndex][1]) {\r\n if (this._buffersEqual(this._bufferData, this._buffers[this._bufferIndex][1]!)) {\r\n this._needSync = false;\r\n this._createBufferOnWrite = this._engine._features.trackUbosInFrame;\r\n return;\r\n } else {\r\n this._copyBuffer(this._bufferData, this._buffers[this._bufferIndex][1]!);\r\n }\r\n }\r\n\r\n this._engine.updateUniformBuffer(this._buffer, this._bufferData);\r\n\r\n if (this._engine._features._collectUbosUpdatedInFrame) {\r\n if (!UniformBuffer._UpdatedUbosInFrame[this._name]) {\r\n UniformBuffer._UpdatedUbosInFrame[this._name] = 0;\r\n }\r\n UniformBuffer._UpdatedUbosInFrame[this._name]++;\r\n }\r\n\r\n this._needSync = false;\r\n this._createBufferOnWrite = this._engine._features.trackUbosInFrame;\r\n }\r\n\r\n private _createNewBuffer() {\r\n if (this._bufferIndex + 1 < this._buffers.length) {\r\n this._bufferIndex++;\r\n this._buffer = this._buffers[this._bufferIndex][0];\r\n this._createBufferOnWrite = false;\r\n this._needSync = true;\r\n } else {\r\n this._rebuild();\r\n }\r\n }\r\n\r\n private _checkNewFrame(): void {\r\n if (this._engine._features.trackUbosInFrame && this._currentFrameId !== this._engine.frameId) {\r\n this._currentFrameId = this._engine.frameId;\r\n this._createBufferOnWrite = false;\r\n if (this._buffers && this._buffers.length > 0) {\r\n this._needSync = this._bufferIndex !== 0;\r\n this._bufferIndex = 0;\r\n this._buffer = this._buffers[this._bufferIndex][0];\r\n } else {\r\n this._bufferIndex = -1;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Updates the value of an uniform. The `update` method must be called afterwards to make it effective in the GPU.\r\n * @param uniformName Define the name of the uniform, as used in the uniform block in the shader.\r\n * @param data Define the flattened data\r\n * @param size Define the size of the data.\r\n */\r\n public updateUniform(uniformName: string, data: FloatArray, size: number) {\r\n this._checkNewFrame();\r\n\r\n let location = this._uniformLocations[uniformName];\r\n if (location === undefined) {\r\n if (this._buffer) {\r\n // Cannot add an uniform if the buffer is already created\r\n Logger.Error(\"Cannot add an uniform after UBO has been created.\");\r\n return;\r\n }\r\n this.addUniform(uniformName, size);\r\n location = this._uniformLocations[uniformName];\r\n }\r\n\r\n if (!this._buffer) {\r\n this.create();\r\n }\r\n\r\n if (!this._dynamic) {\r\n // Cache for static uniform buffers\r\n let changed = false;\r\n\r\n for (let i = 0; i < size; i++) {\r\n // We are checking the matrix cache before calling updateUniform so we do not need to check it here\r\n // Hence the test for size === 16 to simply commit the matrix values\r\n if ((size === 16 && !this._engine._features.uniformBufferHardCheckMatrix) || this._bufferData[location + i] !== Tools.FloatRound(data[i])) {\r\n changed = true;\r\n if (this._createBufferOnWrite) {\r\n this._createNewBuffer();\r\n }\r\n this._bufferData[location + i] = data[i];\r\n }\r\n }\r\n\r\n this._needSync = this._needSync || changed;\r\n } else {\r\n // No cache for dynamic\r\n for (let i = 0; i < size; i++) {\r\n this._bufferData[location + i] = data[i];\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Updates the value of an uniform. The `update` method must be called afterwards to make it effective in the GPU.\r\n * @param uniformName Define the name of the uniform, as used in the uniform block in the shader.\r\n * @param data Define the flattened data\r\n * @param size Define the size of the data.\r\n */\r\n public updateUniformArray(uniformName: string, data: FloatArray, size: number) {\r\n this._checkNewFrame();\r\n\r\n const location = this._uniformLocations[uniformName];\r\n if (location === undefined) {\r\n Logger.Error(\"Cannot add an uniform Array dynamically. Please, add it using addUniform and make sure that uniform buffers are supported by the current engine.\");\r\n return;\r\n }\r\n\r\n if (!this._buffer) {\r\n this.create();\r\n }\r\n\r\n const arraySizes = this._uniformArraySizes[uniformName];\r\n\r\n if (!this._dynamic) {\r\n // Cache for static uniform buffers\r\n let changed = false;\r\n let countToFour = 0;\r\n let baseStride = 0;\r\n for (let i = 0; i < size; i++) {\r\n if (this._bufferData[location + baseStride * 4 + countToFour] !== Tools.FloatRound(data[i])) {\r\n changed = true;\r\n if (this._createBufferOnWrite) {\r\n this._createNewBuffer();\r\n }\r\n this._bufferData[location + baseStride * 4 + countToFour] = data[i];\r\n }\r\n countToFour++;\r\n if (countToFour === arraySizes.strideSize) {\r\n for (; countToFour < 4; countToFour++) {\r\n this._bufferData[location + baseStride * 4 + countToFour] = 0;\r\n }\r\n countToFour = 0;\r\n baseStride++;\r\n }\r\n }\r\n\r\n this._needSync = this._needSync || changed;\r\n } else {\r\n // No cache for dynamic\r\n for (let i = 0; i < size; i++) {\r\n this._bufferData[location + i] = data[i];\r\n }\r\n }\r\n }\r\n\r\n // Matrix cache\r\n private _valueCache: { [key: string]: number } = {};\r\n private _cacheMatrix(name: string, matrix: IMatrixLike): boolean {\r\n this._checkNewFrame();\r\n\r\n const cache = this._valueCache[name];\r\n const flag = matrix.updateFlag;\r\n if (cache !== undefined && cache === flag) {\r\n return false;\r\n }\r\n\r\n this._valueCache[name] = flag;\r\n return true;\r\n }\r\n\r\n // Update methods\r\n\r\n private _updateMatrix3x3ForUniform(name: string, matrix: Float32Array): void {\r\n // To match std140, matrix must be realigned\r\n for (let i = 0; i < 3; i++) {\r\n UniformBuffer._TempBuffer[i * 4] = matrix[i * 3];\r\n UniformBuffer._TempBuffer[i * 4 + 1] = matrix[i * 3 + 1];\r\n UniformBuffer._TempBuffer[i * 4 + 2] = matrix[i * 3 + 2];\r\n UniformBuffer._TempBuffer[i * 4 + 3] = 0.0;\r\n }\r\n\r\n this.updateUniform(name, UniformBuffer._TempBuffer, 12);\r\n }\r\n\r\n private _updateMatrix3x3ForEffect(name: string, matrix: Float32Array): void {\r\n this._currentEffect.setMatrix3x3(name, matrix);\r\n }\r\n\r\n private _updateMatrix2x2ForEffect(name: string, matrix: Float32Array): void {\r\n this._currentEffect.setMatrix2x2(name, matrix);\r\n }\r\n\r\n private _updateMatrix2x2ForUniform(name: string, matrix: Float32Array): void {\r\n // To match std140, matrix must be realigned\r\n for (let i = 0; i < 2; i++) {\r\n UniformBuffer._TempBuffer[i * 4] = matrix[i * 2];\r\n UniformBuffer._TempBuffer[i * 4 + 1] = matrix[i * 2 + 1];\r\n UniformBuffer._TempBuffer[i * 4 + 2] = 0.0;\r\n UniformBuffer._TempBuffer[i * 4 + 3] = 0.0;\r\n }\r\n\r\n this.updateUniform(name, UniformBuffer._TempBuffer, 8);\r\n }\r\n\r\n private _updateFloatForEffect(name: string, x: number) {\r\n this._currentEffect.setFloat(name, x);\r\n }\r\n\r\n private _updateFloatForUniform(name: string, x: number) {\r\n UniformBuffer._TempBuffer[0] = x;\r\n this.updateUniform(name, UniformBuffer._TempBuffer, 1);\r\n }\r\n\r\n private _updateFloat2ForEffect(name: string, x: number, y: number, suffix = \"\") {\r\n this._currentEffect.setFloat2(name + suffix, x, y);\r\n }\r\n\r\n private _updateFloat2ForUniform(name: string, x: number, y: number) {\r\n UniformBuffer._TempBuffer[0] = x;\r\n UniformBuffer._TempBuffer[1] = y;\r\n this.updateUniform(name, UniformBuffer._TempBuffer, 2);\r\n }\r\n\r\n private _updateFloat3ForEffect(name: string, x: number, y: number, z: number, suffix = \"\") {\r\n this._currentEffect.setFloat3(name + suffix, x, y, z);\r\n }\r\n\r\n private _updateFloat3ForUniform(name: string, x: number, y: number, z: number) {\r\n UniformBuffer._TempBuffer[0] = x;\r\n UniformBuffer._TempBuffer[1] = y;\r\n UniformBuffer._TempBuffer[2] = z;\r\n this.updateUniform(name, UniformBuffer._TempBuffer, 3);\r\n }\r\n\r\n private _updateFloat4ForEffect(name: string, x: number, y: number, z: number, w: number, suffix = \"\") {\r\n this._currentEffect.setFloat4(name + suffix, x, y, z, w);\r\n }\r\n\r\n private _updateFloat4ForUniform(name: string, x: number, y: number, z: number, w: number) {\r\n UniformBuffer._TempBuffer[0] = x;\r\n UniformBuffer._TempBuffer[1] = y;\r\n UniformBuffer._TempBuffer[2] = z;\r\n UniformBuffer._TempBuffer[3] = w;\r\n this.updateUniform(name, UniformBuffer._TempBuffer, 4);\r\n }\r\n\r\n private _updateFloatArrayForEffect(name: string, array: Float32Array) {\r\n this._currentEffect.setFloatArray(name, array);\r\n }\r\n\r\n private _updateFloatArrayForUniform(name: string, array: Float32Array) {\r\n this.updateUniformArray(name, array, array.length);\r\n }\r\n\r\n private _updateArrayForEffect(name: string, array: number[]) {\r\n this._currentEffect.setArray(name, array);\r\n }\r\n\r\n private _updateArrayForUniform(name: string, array: number[]) {\r\n this.updateUniformArray(name, array, array.length);\r\n }\r\n\r\n private _updateIntArrayForEffect(name: string, array: Int32Array) {\r\n this._currentEffect.setIntArray(name, array);\r\n }\r\n\r\n private _updateIntArrayForUniform(name: string, array: Int32Array) {\r\n UniformBuffer._TempBufferInt32View.set(array);\r\n this.updateUniformArray(name, UniformBuffer._TempBuffer, array.length);\r\n }\r\n\r\n private _updateUIntArrayForEffect(name: string, array: Uint32Array) {\r\n this._currentEffect.setUIntArray(name, array);\r\n }\r\n\r\n private _updateUIntArrayForUniform(name: string, array: Uint32Array) {\r\n UniformBuffer._TempBufferUInt32View.set(array);\r\n this.updateUniformArray(name, UniformBuffer._TempBuffer, array.length);\r\n }\r\n\r\n private _updateMatrixForEffect(name: string, mat: IMatrixLike) {\r\n this._currentEffect.setMatrix(name, mat);\r\n }\r\n\r\n private _updateMatrixForUniform(name: string, mat: IMatrixLike) {\r\n if (this._cacheMatrix(name, mat)) {\r\n this.updateUniform(name, mat.toArray(), 16);\r\n }\r\n }\r\n\r\n private _updateMatricesForEffect(name: string, mat: Float32Array) {\r\n this._currentEffect.setMatrices(name, mat);\r\n }\r\n\r\n private _updateMatricesForUniform(name: string, mat: Float32Array) {\r\n this.updateUniform(name, mat, mat.length);\r\n }\r\n\r\n private _updateVector3ForEffect(name: string, vector: IVector3Like) {\r\n this._currentEffect.setVector3(name, vector);\r\n }\r\n\r\n private _updateVector3ForUniform(name: string, vector: IVector3Like) {\r\n UniformBuffer._TempBuffer[0] = vector.x;\r\n UniformBuffer._TempBuffer[1] = vector.y;\r\n UniformBuffer._TempBuffer[2] = vector.z;\r\n this.updateUniform(name, UniformBuffer._TempBuffer, 3);\r\n }\r\n\r\n private _updateVector4ForEffect(name: string, vector: IVector4Like) {\r\n this._currentEffect.setVector4(name, vector);\r\n }\r\n\r\n private _updateVector4ForUniform(name: string, vector: IVector4Like) {\r\n UniformBuffer._TempBuffer[0] = vector.x;\r\n UniformBuffer._TempBuffer[1] = vector.y;\r\n UniformBuffer._TempBuffer[2] = vector.z;\r\n UniformBuffer._TempBuffer[3] = vector.w;\r\n this.updateUniform(name, UniformBuffer._TempBuffer, 4);\r\n }\r\n\r\n private _updateColor3ForEffect(name: string, color: IColor3Like, suffix = \"\") {\r\n this._currentEffect.setColor3(name + suffix, color);\r\n }\r\n\r\n private _updateColor3ForUniform(name: string, color: IColor3Like) {\r\n UniformBuffer._TempBuffer[0] = color.r;\r\n UniformBuffer._TempBuffer[1] = color.g;\r\n UniformBuffer._TempBuffer[2] = color.b;\r\n this.updateUniform(name, UniformBuffer._TempBuffer, 3);\r\n }\r\n\r\n private _updateColor4ForEffect(name: string, color: IColor3Like, alpha: number, suffix = \"\") {\r\n this._currentEffect.setColor4(name + suffix, color, alpha);\r\n }\r\n\r\n private _updateDirectColor4ForEffect(name: string, color: IColor4Like, suffix = \"\") {\r\n this._currentEffect.setDirectColor4(name + suffix, color);\r\n }\r\n\r\n private _updateColor4ForUniform(name: string, color: IColor3Like, alpha: number) {\r\n UniformBuffer._TempBuffer[0] = color.r;\r\n UniformBuffer._TempBuffer[1] = color.g;\r\n UniformBuffer._TempBuffer[2] = color.b;\r\n UniformBuffer._TempBuffer[3] = alpha;\r\n this.updateUniform(name, UniformBuffer._TempBuffer, 4);\r\n }\r\n\r\n private _updateDirectColor4ForUniform(name: string, color: IColor4Like) {\r\n UniformBuffer._TempBuffer[0] = color.r;\r\n UniformBuffer._TempBuffer[1] = color.g;\r\n UniformBuffer._TempBuffer[2] = color.b;\r\n UniformBuffer._TempBuffer[3] = color.a;\r\n this.updateUniform(name, UniformBuffer._TempBuffer, 4);\r\n }\r\n\r\n private _updateIntForEffect(name: string, x: number, suffix = \"\") {\r\n this._currentEffect.setInt(name + suffix, x);\r\n }\r\n\r\n private _updateIntForUniform(name: string, x: number) {\r\n UniformBuffer._TempBufferInt32View[0] = x;\r\n this.updateUniform(name, UniformBuffer._TempBuffer, 1);\r\n }\r\n\r\n private _updateInt2ForEffect(name: string, x: number, y: number, suffix = \"\") {\r\n this._currentEffect.setInt2(name + suffix, x, y);\r\n }\r\n\r\n private _updateInt2ForUniform(name: string, x: number, y: number) {\r\n UniformBuffer._TempBufferInt32View[0] = x;\r\n UniformBuffer._TempBufferInt32View[1] = y;\r\n this.updateUniform(name, UniformBuffer._TempBuffer, 2);\r\n }\r\n\r\n private _updateInt3ForEffect(name: string, x: number, y: number, z: number, suffix = \"\") {\r\n this._currentEffect.setInt3(name + suffix, x, y, z);\r\n }\r\n\r\n private _updateInt3ForUniform(name: string, x: number, y: number, z: number) {\r\n UniformBuffer._TempBufferInt32View[0] = x;\r\n UniformBuffer._TempBufferInt32View[1] = y;\r\n UniformBuffer._TempBufferInt32View[2] = z;\r\n this.updateUniform(name, UniformBuffer._TempBuffer, 3);\r\n }\r\n\r\n private _updateInt4ForEffect(name: string, x: number, y: number, z: number, w: number, suffix = \"\") {\r\n this._currentEffect.setInt4(name + suffix, x, y, z, w);\r\n }\r\n\r\n private _updateInt4ForUniform(name: string, x: number, y: number, z: number, w: number) {\r\n UniformBuffer._TempBufferInt32View[0] = x;\r\n UniformBuffer._TempBufferInt32View[1] = y;\r\n UniformBuffer._TempBufferInt32View[2] = z;\r\n UniformBuffer._TempBufferInt32View[3] = w;\r\n this.updateUniform(name, UniformBuffer._TempBuffer, 4);\r\n }\r\n\r\n private _updateUIntForEffect(name: string, x: number, suffix = \"\") {\r\n this._currentEffect.setUInt(name + suffix, x);\r\n }\r\n\r\n private _updateUIntForUniform(name: string, x: number) {\r\n UniformBuffer._TempBufferUInt32View[0] = x;\r\n this.updateUniform(name, UniformBuffer._TempBuffer, 1);\r\n }\r\n\r\n private _updateUInt2ForEffect(name: string, x: number, y: number, suffix = \"\") {\r\n this._currentEffect.setUInt2(name + suffix, x, y);\r\n }\r\n\r\n private _updateUInt2ForUniform(name: string, x: number, y: number) {\r\n UniformBuffer._TempBufferUInt32View[0] = x;\r\n UniformBuffer._TempBufferUInt32View[1] = y;\r\n this.updateUniform(name, UniformBuffer._TempBuffer, 2);\r\n }\r\n\r\n private _updateUInt3ForEffect(name: string, x: number, y: number, z: number, suffix = \"\") {\r\n this._currentEffect.setUInt3(name + suffix, x, y, z);\r\n }\r\n\r\n private _updateUInt3ForUniform(name: string, x: number, y: number, z: number) {\r\n UniformBuffer._TempBufferUInt32View[0] = x;\r\n UniformBuffer._TempBufferUInt32View[1] = y;\r\n UniformBuffer._TempBufferUInt32View[2] = z;\r\n this.updateUniform(name, UniformBuffer._TempBuffer, 3);\r\n }\r\n\r\n private _updateUInt4ForEffect(name: string, x: number, y: number, z: number, w: number, suffix = \"\") {\r\n this._currentEffect.setUInt4(name + suffix, x, y, z, w);\r\n }\r\n\r\n private _updateUInt4ForUniform(name: string, x: number, y: number, z: number, w: number) {\r\n UniformBuffer._TempBufferUInt32View[0] = x;\r\n UniformBuffer._TempBufferUInt32View[1] = y;\r\n UniformBuffer._TempBufferUInt32View[2] = z;\r\n UniformBuffer._TempBufferUInt32View[3] = w;\r\n this.updateUniform(name, UniformBuffer._TempBuffer, 4);\r\n }\r\n\r\n /**\r\n * Sets a sampler uniform on the effect.\r\n * @param name Define the name of the sampler.\r\n * @param texture Define the texture to set in the sampler\r\n */\r\n public setTexture(name: string, texture: Nullable) {\r\n this._currentEffect.setTexture(name, texture);\r\n }\r\n\r\n /**\r\n * Directly updates the value of the uniform in the cache AND on the GPU.\r\n * @param uniformName Define the name of the uniform, as used in the uniform block in the shader.\r\n * @param data Define the flattened data\r\n */\r\n public updateUniformDirectly(uniformName: string, data: FloatArray) {\r\n this.updateUniform(uniformName, data, data.length);\r\n\r\n this.update();\r\n }\r\n\r\n /**\r\n * Associates an effect to this uniform buffer\r\n * @param effect Define the effect to associate the buffer to\r\n * @param name Name of the uniform block in the shader.\r\n */\r\n public bindToEffect(effect: Effect, name: string): void {\r\n this._currentEffect = effect;\r\n this._currentEffectName = name;\r\n }\r\n\r\n /**\r\n * Binds the current (GPU) buffer to the effect\r\n */\r\n public bindUniformBuffer(): void {\r\n if (!this._noUBO && this._buffer && this._currentEffect) {\r\n this._currentEffect.bindUniformBuffer(this._buffer, this._currentEffectName);\r\n }\r\n }\r\n\r\n /**\r\n * Dissociates the current effect from this uniform buffer\r\n */\r\n public unbindEffect(): void {\r\n this._currentEffect = undefined as any;\r\n this._currentEffectName = undefined as any;\r\n }\r\n\r\n /**\r\n * Sets the current state of the class (_bufferIndex, _buffer) to point to the data buffer passed in parameter if this buffer is one of the buffers handled by the class (meaning if it can be found in the _buffers array)\r\n * This method is meant to be able to update a buffer at any time: just call setDataBuffer to set the class in the right state, call some updateXXX methods and then call udpate() => that will update the GPU buffer on the graphic card\r\n * @param dataBuffer buffer to look for\r\n * @returns true if the buffer has been found and the class internal state points to it, else false\r\n */\r\n public setDataBuffer(dataBuffer: DataBuffer): boolean {\r\n if (!this._buffers) {\r\n return this._buffer === dataBuffer;\r\n }\r\n\r\n for (let b = 0; b < this._buffers.length; ++b) {\r\n const buffer = this._buffers[b];\r\n if (buffer[0] === dataBuffer) {\r\n this._bufferIndex = b;\r\n this._buffer = dataBuffer;\r\n this._createBufferOnWrite = false;\r\n this._currentEffect = undefined as any;\r\n return true;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Disposes the uniform buffer.\r\n */\r\n public dispose(): void {\r\n if (this._noUBO) {\r\n return;\r\n }\r\n\r\n const uniformBuffers = this._engine._uniformBuffers;\r\n const index = uniformBuffers.indexOf(this);\r\n\r\n if (index !== -1) {\r\n uniformBuffers[index] = uniformBuffers[uniformBuffers.length - 1];\r\n uniformBuffers.pop();\r\n }\r\n\r\n if (this._engine._features.trackUbosInFrame && this._buffers) {\r\n for (let i = 0; i < this._buffers.length; ++i) {\r\n const buffer = this._buffers[i][0];\r\n this._engine._releaseBuffer(buffer!);\r\n }\r\n } else if (this._buffer && this._engine._releaseBuffer(this._buffer)) {\r\n this._buffer = null;\r\n }\r\n }\r\n}\r\n","import type { Nullable, DataArray, FloatArray } from \"../types\";\r\nimport type { ThinEngine } from \"../Engines/thinEngine\";\r\nimport { DataBuffer } from \"./dataBuffer\";\r\n\r\n/**\r\n * Class used to store data that will be store in GPU memory\r\n */\r\nexport class Buffer {\r\n private _engine: ThinEngine;\r\n private _buffer: Nullable;\r\n /** @internal */\r\n public _data: Nullable;\r\n private _updatable: boolean;\r\n private _instanced: boolean;\r\n private _divisor: number;\r\n private _isAlreadyOwned = false;\r\n\r\n /**\r\n * Gets the byte stride.\r\n */\r\n public readonly byteStride: number;\r\n\r\n /**\r\n * Constructor\r\n * @param engine the engine\r\n * @param data the data to use for this buffer\r\n * @param updatable whether the data is updatable\r\n * @param stride the stride (optional)\r\n * @param postponeInternalCreation whether to postpone creating the internal WebGL buffer (optional)\r\n * @param instanced whether the buffer is instanced (optional)\r\n * @param useBytes set to true if the stride in in bytes (optional)\r\n * @param divisor sets an optional divisor for instances (1 by default)\r\n */\r\n constructor(\r\n engine: any,\r\n data: DataArray | DataBuffer,\r\n updatable: boolean,\r\n stride = 0,\r\n postponeInternalCreation = false,\r\n instanced = false,\r\n useBytes = false,\r\n divisor?: number\r\n ) {\r\n if (engine.getScene) {\r\n // old versions of VertexBuffer accepted 'mesh' instead of 'engine'\r\n this._engine = engine.getScene().getEngine();\r\n } else {\r\n this._engine = engine;\r\n }\r\n\r\n this._updatable = updatable;\r\n this._instanced = instanced;\r\n this._divisor = divisor || 1;\r\n\r\n if (data instanceof DataBuffer) {\r\n this._data = null;\r\n this._buffer = data;\r\n } else {\r\n this._data = data;\r\n this._buffer = null;\r\n }\r\n\r\n this.byteStride = useBytes ? stride : stride * Float32Array.BYTES_PER_ELEMENT;\r\n\r\n if (!postponeInternalCreation) {\r\n // by default\r\n this.create();\r\n }\r\n }\r\n\r\n /**\r\n * Create a new VertexBuffer based on the current buffer\r\n * @param kind defines the vertex buffer kind (position, normal, etc.)\r\n * @param offset defines offset in the buffer (0 by default)\r\n * @param size defines the size in floats of attributes (position is 3 for instance)\r\n * @param stride defines the stride size in floats in the buffer (the offset to apply to reach next value when data is interleaved)\r\n * @param instanced defines if the vertex buffer contains indexed data\r\n * @param useBytes defines if the offset and stride are in bytes *\r\n * @param divisor sets an optional divisor for instances (1 by default)\r\n * @returns the new vertex buffer\r\n */\r\n public createVertexBuffer(kind: string, offset: number, size: number, stride?: number, instanced?: boolean, useBytes = false, divisor?: number): VertexBuffer {\r\n const byteOffset = useBytes ? offset : offset * Float32Array.BYTES_PER_ELEMENT;\r\n const byteStride = stride ? (useBytes ? stride : stride * Float32Array.BYTES_PER_ELEMENT) : this.byteStride;\r\n\r\n // a lot of these parameters are ignored as they are overridden by the buffer\r\n return new VertexBuffer(\r\n this._engine,\r\n this,\r\n kind,\r\n this._updatable,\r\n true,\r\n byteStride,\r\n instanced === undefined ? this._instanced : instanced,\r\n byteOffset,\r\n size,\r\n undefined,\r\n undefined,\r\n true,\r\n this._divisor || divisor\r\n );\r\n }\r\n\r\n // Properties\r\n\r\n /**\r\n * Gets a boolean indicating if the Buffer is updatable?\r\n * @returns true if the buffer is updatable\r\n */\r\n public isUpdatable(): boolean {\r\n return this._updatable;\r\n }\r\n\r\n /**\r\n * Gets current buffer's data\r\n * @returns a DataArray or null\r\n */\r\n public getData(): Nullable {\r\n return this._data;\r\n }\r\n\r\n /**\r\n * Gets underlying native buffer\r\n * @returns underlying native buffer\r\n */\r\n public getBuffer(): Nullable {\r\n return this._buffer;\r\n }\r\n\r\n /**\r\n * Gets the stride in float32 units (i.e. byte stride / 4).\r\n * May not be an integer if the byte stride is not divisible by 4.\r\n * @returns the stride in float32 units\r\n * @deprecated Please use byteStride instead.\r\n */\r\n public getStrideSize(): number {\r\n return this.byteStride / Float32Array.BYTES_PER_ELEMENT;\r\n }\r\n\r\n // Methods\r\n\r\n /**\r\n * Store data into the buffer. Creates the buffer if not used already.\r\n * If the buffer was already used, it will be updated only if it is updatable, otherwise it will do nothing.\r\n * @param data defines the data to store\r\n */\r\n public create(data: Nullable = null): void {\r\n if (!data && this._buffer) {\r\n return; // nothing to do\r\n }\r\n\r\n data = data || this._data;\r\n\r\n if (!data) {\r\n return;\r\n }\r\n\r\n if (!this._buffer) {\r\n // create buffer\r\n if (this._updatable) {\r\n this._buffer = this._engine.createDynamicVertexBuffer(data);\r\n this._data = data;\r\n } else {\r\n this._buffer = this._engine.createVertexBuffer(data);\r\n }\r\n } else if (this._updatable) {\r\n // update buffer\r\n this._engine.updateDynamicVertexBuffer(this._buffer, data);\r\n this._data = data;\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _rebuild(): void {\r\n this._buffer = null;\r\n this.create(this._data);\r\n }\r\n\r\n /**\r\n * Update current buffer data\r\n * @param data defines the data to store\r\n */\r\n public update(data: DataArray): void {\r\n this.create(data);\r\n }\r\n\r\n /**\r\n * Updates the data directly.\r\n * @param data the new data\r\n * @param offset the new offset\r\n * @param vertexCount the vertex count (optional)\r\n * @param useBytes set to true if the offset is in bytes\r\n */\r\n public updateDirectly(data: DataArray, offset: number, vertexCount?: number, useBytes: boolean = false): void {\r\n if (!this._buffer) {\r\n return;\r\n }\r\n\r\n if (this._updatable) {\r\n // update buffer\r\n this._engine.updateDynamicVertexBuffer(\r\n this._buffer,\r\n data,\r\n useBytes ? offset : offset * Float32Array.BYTES_PER_ELEMENT,\r\n vertexCount ? vertexCount * this.byteStride : undefined\r\n );\r\n if (offset === 0 && vertexCount === undefined) {\r\n // Keep the data if we easily can\r\n this._data = data;\r\n } else {\r\n this._data = null;\r\n }\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _increaseReferences() {\r\n if (!this._buffer) {\r\n return;\r\n }\r\n\r\n if (!this._isAlreadyOwned) {\r\n this._isAlreadyOwned = true;\r\n return;\r\n }\r\n\r\n this._buffer.references++;\r\n }\r\n\r\n /**\r\n * Release all resources\r\n */\r\n public dispose(): void {\r\n if (!this._buffer) {\r\n return;\r\n }\r\n if (this._engine._releaseBuffer(this._buffer)) {\r\n this._buffer = null;\r\n this._data = null;\r\n }\r\n }\r\n}\r\n\r\n/**\r\n * Specialized buffer used to store vertex data\r\n */\r\nexport class VertexBuffer {\r\n private static _Counter = 0;\r\n\r\n /** @internal */\r\n public _buffer: Buffer;\r\n /** @internal */\r\n public _validOffsetRange: boolean; // used internally by the engine\r\n private _kind: string;\r\n private _size: number;\r\n private _ownsBuffer: boolean;\r\n private _instanced: boolean;\r\n private _instanceDivisor: number;\r\n\r\n /**\r\n * The byte type.\r\n */\r\n public static readonly BYTE = 5120;\r\n\r\n /**\r\n * The unsigned byte type.\r\n */\r\n public static readonly UNSIGNED_BYTE = 5121;\r\n\r\n /**\r\n * The short type.\r\n */\r\n public static readonly SHORT = 5122;\r\n\r\n /**\r\n * The unsigned short type.\r\n */\r\n public static readonly UNSIGNED_SHORT = 5123;\r\n\r\n /**\r\n * The integer type.\r\n */\r\n public static readonly INT = 5124;\r\n\r\n /**\r\n * The unsigned integer type.\r\n */\r\n public static readonly UNSIGNED_INT = 5125;\r\n\r\n /**\r\n * The float type.\r\n */\r\n public static readonly FLOAT = 5126;\r\n\r\n /**\r\n * Gets or sets the instance divisor when in instanced mode\r\n */\r\n public get instanceDivisor(): number {\r\n return this._instanceDivisor;\r\n }\r\n\r\n public set instanceDivisor(value: number) {\r\n const isInstanced = value != 0;\r\n this._instanceDivisor = value;\r\n\r\n if (isInstanced !== this._instanced) {\r\n this._instanced = isInstanced;\r\n this._computeHashCode();\r\n }\r\n }\r\n\r\n /**\r\n * Gets the byte stride.\r\n */\r\n public readonly byteStride: number;\r\n\r\n /**\r\n * Gets the byte offset.\r\n */\r\n public readonly byteOffset: number;\r\n\r\n /**\r\n * Gets whether integer data values should be normalized into a certain range when being casted to a float.\r\n */\r\n public readonly normalized: boolean;\r\n\r\n /**\r\n * Gets the data type of each component in the array.\r\n */\r\n public readonly type: number;\r\n\r\n /**\r\n * Gets the unique id of this vertex buffer\r\n */\r\n public readonly uniqueId: number;\r\n\r\n /**\r\n * Gets a hash code representing the format (type, normalized, size, instanced, stride) of this buffer\r\n * All buffers with the same format will have the same hash code\r\n */\r\n public readonly hashCode: number;\r\n\r\n /**\r\n * Constructor\r\n * @param engine the engine\r\n * @param data the data to use for this vertex buffer\r\n * @param kind the vertex buffer kind\r\n * @param updatable whether the data is updatable\r\n * @param postponeInternalCreation whether to postpone creating the internal WebGL buffer (optional)\r\n * @param stride the stride (optional)\r\n * @param instanced whether the buffer is instanced (optional)\r\n * @param offset the offset of the data (optional)\r\n * @param size the number of components (optional)\r\n * @param type the type of the component (optional)\r\n * @param normalized whether the data contains normalized data (optional)\r\n * @param useBytes set to true if stride and offset are in bytes (optional)\r\n * @param divisor defines the instance divisor to use (1 by default)\r\n * @param takeBufferOwnership defines if the buffer should be released when the vertex buffer is disposed\r\n */\r\n constructor(\r\n engine: any,\r\n data: DataArray | Buffer | DataBuffer,\r\n kind: string,\r\n updatable: boolean,\r\n postponeInternalCreation?: boolean,\r\n stride?: number,\r\n instanced?: boolean,\r\n offset?: number,\r\n size?: number,\r\n type?: number,\r\n normalized = false,\r\n useBytes = false,\r\n divisor = 1,\r\n takeBufferOwnership = false\r\n ) {\r\n if (data instanceof Buffer) {\r\n this._buffer = data;\r\n this._ownsBuffer = takeBufferOwnership;\r\n } else {\r\n this._buffer = new Buffer(engine, data, updatable, stride, postponeInternalCreation, instanced, useBytes);\r\n this._ownsBuffer = true;\r\n }\r\n\r\n this.uniqueId = VertexBuffer._Counter++;\r\n this._kind = kind;\r\n\r\n if (type == undefined) {\r\n const vertexData = this.getData();\r\n this.type = VertexBuffer.FLOAT;\r\n if (vertexData instanceof Int8Array) {\r\n this.type = VertexBuffer.BYTE;\r\n } else if (vertexData instanceof Uint8Array) {\r\n this.type = VertexBuffer.UNSIGNED_BYTE;\r\n } else if (vertexData instanceof Int16Array) {\r\n this.type = VertexBuffer.SHORT;\r\n } else if (vertexData instanceof Uint16Array) {\r\n this.type = VertexBuffer.UNSIGNED_SHORT;\r\n } else if (vertexData instanceof Int32Array) {\r\n this.type = VertexBuffer.INT;\r\n } else if (vertexData instanceof Uint32Array) {\r\n this.type = VertexBuffer.UNSIGNED_INT;\r\n }\r\n } else {\r\n this.type = type;\r\n }\r\n\r\n const typeByteLength = VertexBuffer.GetTypeByteLength(this.type);\r\n\r\n if (useBytes) {\r\n this._size = size || (stride ? stride / typeByteLength : VertexBuffer.DeduceStride(kind));\r\n this.byteStride = stride || this._buffer.byteStride || this._size * typeByteLength;\r\n this.byteOffset = offset || 0;\r\n } else {\r\n this._size = size || stride || VertexBuffer.DeduceStride(kind);\r\n this.byteStride = stride ? stride * typeByteLength : this._buffer.byteStride || this._size * typeByteLength;\r\n this.byteOffset = (offset || 0) * typeByteLength;\r\n }\r\n\r\n this.normalized = normalized;\r\n\r\n this._instanced = instanced !== undefined ? instanced : false;\r\n this._instanceDivisor = instanced ? divisor : 0;\r\n\r\n this._computeHashCode();\r\n }\r\n\r\n private _computeHashCode(): void {\r\n // note: cast to any because the property is declared readonly\r\n (this.hashCode as any) =\r\n ((this.type - 5120) << 0) +\r\n ((this.normalized ? 1 : 0) << 3) +\r\n (this._size << 4) +\r\n ((this._instanced ? 1 : 0) << 6) +\r\n /* keep 5 bits free */\r\n (this.byteStride << 12);\r\n }\r\n\r\n /** @internal */\r\n public _rebuild(): void {\r\n if (!this._buffer) {\r\n return;\r\n }\r\n\r\n this._buffer._rebuild();\r\n }\r\n\r\n /**\r\n * Returns the kind of the VertexBuffer (string)\r\n * @returns a string\r\n */\r\n public getKind(): string {\r\n return this._kind;\r\n }\r\n\r\n // Properties\r\n\r\n /**\r\n * Gets a boolean indicating if the VertexBuffer is updatable?\r\n * @returns true if the buffer is updatable\r\n */\r\n public isUpdatable(): boolean {\r\n return this._buffer.isUpdatable();\r\n }\r\n\r\n /**\r\n * Gets current buffer's data\r\n * @returns a DataArray or null\r\n */\r\n public getData(): Nullable {\r\n return this._buffer.getData();\r\n }\r\n\r\n /**\r\n * Gets current buffer's data as a float array. Float data is constructed if the vertex buffer data cannot be returned directly.\r\n * @param totalVertices number of vertices in the buffer to take into account\r\n * @param forceCopy defines a boolean indicating that the returned array must be cloned upon returning it\r\n * @returns a float array containing vertex data\r\n */\r\n public getFloatData(totalVertices: number, forceCopy?: boolean): Nullable {\r\n const data = this.getData();\r\n if (!data) {\r\n return null;\r\n }\r\n\r\n const tightlyPackedByteStride = this.getSize() * VertexBuffer.GetTypeByteLength(this.type);\r\n const count = totalVertices * this.getSize();\r\n\r\n if (this.type !== VertexBuffer.FLOAT || this.byteStride !== tightlyPackedByteStride) {\r\n const copy = new Float32Array(count);\r\n this.forEach(count, (value, index) => (copy[index] = value));\r\n return copy;\r\n }\r\n\r\n if (!(data instanceof Array || data instanceof Float32Array) || this.byteOffset !== 0 || data.length !== count) {\r\n if (data instanceof Array) {\r\n const offset = this.byteOffset / 4;\r\n return data.slice(offset, offset + count);\r\n } else if (data instanceof ArrayBuffer) {\r\n return new Float32Array(data, this.byteOffset, count);\r\n } else {\r\n let offset = data.byteOffset + this.byteOffset;\r\n if (forceCopy) {\r\n const result = new Float32Array(count);\r\n const source = new Float32Array(data.buffer, offset, count);\r\n\r\n result.set(source);\r\n\r\n return result;\r\n }\r\n\r\n // Protect against bad data\r\n const remainder = offset % 4;\r\n\r\n if (remainder) {\r\n offset = Math.max(0, offset - remainder);\r\n }\r\n\r\n return new Float32Array(data.buffer, offset, count);\r\n }\r\n }\r\n\r\n if (forceCopy) {\r\n return data.slice();\r\n }\r\n\r\n return data;\r\n }\r\n\r\n /**\r\n * Gets underlying native buffer\r\n * @returns underlying native buffer\r\n */\r\n public getBuffer(): Nullable {\r\n return this._buffer.getBuffer();\r\n }\r\n\r\n /**\r\n * Gets the stride in float32 units (i.e. byte stride / 4).\r\n * May not be an integer if the byte stride is not divisible by 4.\r\n * @returns the stride in float32 units\r\n * @deprecated Please use byteStride instead.\r\n */\r\n public getStrideSize(): number {\r\n return this.byteStride / VertexBuffer.GetTypeByteLength(this.type);\r\n }\r\n\r\n /**\r\n * Returns the offset as a multiple of the type byte length.\r\n * @returns the offset in bytes\r\n * @deprecated Please use byteOffset instead.\r\n */\r\n public getOffset(): number {\r\n return this.byteOffset / VertexBuffer.GetTypeByteLength(this.type);\r\n }\r\n\r\n /**\r\n * Returns the number of components or the byte size per vertex attribute\r\n * @param sizeInBytes If true, returns the size in bytes or else the size in number of components of the vertex attribute (default: false)\r\n * @returns the number of components\r\n */\r\n public getSize(sizeInBytes = false): number {\r\n return sizeInBytes ? this._size * VertexBuffer.GetTypeByteLength(this.type) : this._size;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating is the internal buffer of the VertexBuffer is instanced\r\n * @returns true if this buffer is instanced\r\n */\r\n public getIsInstanced(): boolean {\r\n return this._instanced;\r\n }\r\n\r\n /**\r\n * Returns the instancing divisor, zero for non-instanced (integer).\r\n * @returns a number\r\n */\r\n public getInstanceDivisor(): number {\r\n return this._instanceDivisor;\r\n }\r\n\r\n // Methods\r\n\r\n /**\r\n * Store data into the buffer. If the buffer was already used it will be either recreated or updated depending on isUpdatable property\r\n * @param data defines the data to store\r\n */\r\n public create(data?: DataArray): void {\r\n this._buffer.create(data);\r\n }\r\n\r\n /**\r\n * Updates the underlying buffer according to the passed numeric array or Float32Array.\r\n * This function will create a new buffer if the current one is not updatable\r\n * @param data defines the data to store\r\n */\r\n public update(data: DataArray): void {\r\n this._buffer.update(data);\r\n }\r\n\r\n /**\r\n * Updates directly the underlying WebGLBuffer according to the passed numeric array or Float32Array.\r\n * Returns the directly updated WebGLBuffer.\r\n * @param data the new data\r\n * @param offset the new offset\r\n * @param useBytes set to true if the offset is in bytes\r\n */\r\n public updateDirectly(data: DataArray, offset: number, useBytes: boolean = false): void {\r\n this._buffer.updateDirectly(data, offset, undefined, useBytes);\r\n }\r\n\r\n /**\r\n * Disposes the VertexBuffer and the underlying WebGLBuffer.\r\n */\r\n public dispose(): void {\r\n if (this._ownsBuffer) {\r\n this._buffer.dispose();\r\n }\r\n }\r\n\r\n /**\r\n * Enumerates each value of this vertex buffer as numbers.\r\n * @param count the number of values to enumerate\r\n * @param callback the callback function called for each value\r\n */\r\n public forEach(count: number, callback: (value: number, index: number) => void): void {\r\n VertexBuffer.ForEach(this._buffer.getData()!, this.byteOffset, this.byteStride, this._size, this.type, count, this.normalized, callback);\r\n }\r\n\r\n // Enums\r\n /**\r\n * Positions\r\n */\r\n public static readonly PositionKind = \"position\";\r\n /**\r\n * Normals\r\n */\r\n public static readonly NormalKind = \"normal\";\r\n /**\r\n * Tangents\r\n */\r\n public static readonly TangentKind = \"tangent\";\r\n /**\r\n * Texture coordinates\r\n */\r\n public static readonly UVKind = \"uv\";\r\n /**\r\n * Texture coordinates 2\r\n */\r\n public static readonly UV2Kind = \"uv2\";\r\n /**\r\n * Texture coordinates 3\r\n */\r\n public static readonly UV3Kind = \"uv3\";\r\n /**\r\n * Texture coordinates 4\r\n */\r\n public static readonly UV4Kind = \"uv4\";\r\n /**\r\n * Texture coordinates 5\r\n */\r\n public static readonly UV5Kind = \"uv5\";\r\n /**\r\n * Texture coordinates 6\r\n */\r\n public static readonly UV6Kind = \"uv6\";\r\n /**\r\n * Colors\r\n */\r\n public static readonly ColorKind = \"color\";\r\n /**\r\n * Instance Colors\r\n */\r\n public static readonly ColorInstanceKind = \"instanceColor\";\r\n /**\r\n * Matrix indices (for bones)\r\n */\r\n public static readonly MatricesIndicesKind = \"matricesIndices\";\r\n /**\r\n * Matrix weights (for bones)\r\n */\r\n public static readonly MatricesWeightsKind = \"matricesWeights\";\r\n /**\r\n * Additional matrix indices (for bones)\r\n */\r\n public static readonly MatricesIndicesExtraKind = \"matricesIndicesExtra\";\r\n /**\r\n * Additional matrix weights (for bones)\r\n */\r\n public static readonly MatricesWeightsExtraKind = \"matricesWeightsExtra\";\r\n\r\n /**\r\n * Deduces the stride given a kind.\r\n * @param kind The kind string to deduce\r\n * @returns The deduced stride\r\n */\r\n public static DeduceStride(kind: string): number {\r\n switch (kind) {\r\n case VertexBuffer.UVKind:\r\n case VertexBuffer.UV2Kind:\r\n case VertexBuffer.UV3Kind:\r\n case VertexBuffer.UV4Kind:\r\n case VertexBuffer.UV5Kind:\r\n case VertexBuffer.UV6Kind:\r\n return 2;\r\n case VertexBuffer.NormalKind:\r\n case VertexBuffer.PositionKind:\r\n return 3;\r\n case VertexBuffer.ColorKind:\r\n case VertexBuffer.MatricesIndicesKind:\r\n case VertexBuffer.MatricesIndicesExtraKind:\r\n case VertexBuffer.MatricesWeightsKind:\r\n case VertexBuffer.MatricesWeightsExtraKind:\r\n case VertexBuffer.TangentKind:\r\n return 4;\r\n default:\r\n throw new Error(\"Invalid kind '\" + kind + \"'\");\r\n }\r\n }\r\n\r\n /**\r\n * Gets the byte length of the given type.\r\n * @param type the type\r\n * @returns the number of bytes\r\n */\r\n public static GetTypeByteLength(type: number): number {\r\n switch (type) {\r\n case VertexBuffer.BYTE:\r\n case VertexBuffer.UNSIGNED_BYTE:\r\n return 1;\r\n case VertexBuffer.SHORT:\r\n case VertexBuffer.UNSIGNED_SHORT:\r\n return 2;\r\n case VertexBuffer.INT:\r\n case VertexBuffer.UNSIGNED_INT:\r\n case VertexBuffer.FLOAT:\r\n return 4;\r\n default:\r\n throw new Error(`Invalid type '${type}'`);\r\n }\r\n }\r\n\r\n /**\r\n * Enumerates each value of the given parameters as numbers.\r\n * @param data the data to enumerate\r\n * @param byteOffset the byte offset of the data\r\n * @param byteStride the byte stride of the data\r\n * @param componentCount the number of components per element\r\n * @param componentType the type of the component\r\n * @param count the number of values to enumerate\r\n * @param normalized whether the data is normalized\r\n * @param callback the callback function called for each value\r\n */\r\n public static ForEach(\r\n data: DataArray,\r\n byteOffset: number,\r\n byteStride: number,\r\n componentCount: number,\r\n componentType: number,\r\n count: number,\r\n normalized: boolean,\r\n callback: (value: number, index: number) => void\r\n ): void {\r\n if (data instanceof Array) {\r\n let offset = byteOffset / 4;\r\n const stride = byteStride / 4;\r\n for (let index = 0; index < count; index += componentCount) {\r\n for (let componentIndex = 0; componentIndex < componentCount; componentIndex++) {\r\n callback(data[offset + componentIndex], index + componentIndex);\r\n }\r\n offset += stride;\r\n }\r\n } else {\r\n const dataView = data instanceof ArrayBuffer ? new DataView(data) : new DataView(data.buffer, data.byteOffset, data.byteLength);\r\n const componentByteLength = VertexBuffer.GetTypeByteLength(componentType);\r\n for (let index = 0; index < count; index += componentCount) {\r\n let componentByteOffset = byteOffset;\r\n for (let componentIndex = 0; componentIndex < componentCount; componentIndex++) {\r\n const value = VertexBuffer._GetFloatValue(dataView, componentType, componentByteOffset, normalized);\r\n callback(value, index + componentIndex);\r\n componentByteOffset += componentByteLength;\r\n }\r\n byteOffset += byteStride;\r\n }\r\n }\r\n }\r\n\r\n private static _GetFloatValue(dataView: DataView, type: number, byteOffset: number, normalized: boolean): number {\r\n switch (type) {\r\n case VertexBuffer.BYTE: {\r\n let value = dataView.getInt8(byteOffset);\r\n if (normalized) {\r\n value = Math.max(value / 127, -1);\r\n }\r\n return value;\r\n }\r\n case VertexBuffer.UNSIGNED_BYTE: {\r\n let value = dataView.getUint8(byteOffset);\r\n if (normalized) {\r\n value = value / 255;\r\n }\r\n return value;\r\n }\r\n case VertexBuffer.SHORT: {\r\n let value = dataView.getInt16(byteOffset, true);\r\n if (normalized) {\r\n value = Math.max(value / 32767, -1);\r\n }\r\n return value;\r\n }\r\n case VertexBuffer.UNSIGNED_SHORT: {\r\n let value = dataView.getUint16(byteOffset, true);\r\n if (normalized) {\r\n value = value / 65535;\r\n }\r\n return value;\r\n }\r\n case VertexBuffer.INT: {\r\n return dataView.getInt32(byteOffset, true);\r\n }\r\n case VertexBuffer.UNSIGNED_INT: {\r\n return dataView.getUint32(byteOffset, true);\r\n }\r\n case VertexBuffer.FLOAT: {\r\n return dataView.getFloat32(byteOffset, true);\r\n }\r\n default: {\r\n throw new Error(`Invalid component type ${type}`);\r\n }\r\n }\r\n }\r\n}\r\n","import type { Nullable, FloatArray } from \"../types\";\r\nimport { Vector3, Vector2, TmpVectors } from \"../Maths/math.vector\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { TransformNode } from \"../Meshes/transformNode\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport type { Sprite } from \"../Sprites/sprite\";\r\n\r\nimport type { Ray } from \"../Culling/ray\";\r\n\r\n/**\r\n * Information about the result of picking within a scene\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/interactions/picking_collisions\r\n */\r\nexport class PickingInfo {\r\n /**\r\n * If the pick collided with an object\r\n */\r\n public hit = false;\r\n /**\r\n * Distance away where the pick collided\r\n */\r\n public distance = 0;\r\n /**\r\n * The location of pick collision\r\n */\r\n public pickedPoint: Nullable = null;\r\n /**\r\n * The mesh corresponding the pick collision\r\n */\r\n public pickedMesh: Nullable = null;\r\n /** (See getTextureCoordinates) The barycentric U coordinate that is used when calculating the texture coordinates of the collision.*/\r\n public bu = 0;\r\n /** (See getTextureCoordinates) The barycentric V coordinate that is used when calculating the texture coordinates of the collision.*/\r\n public bv = 0;\r\n /** The index of the face on the mesh that was picked, or the index of the Line if the picked Mesh is a LinesMesh */\r\n public faceId = -1;\r\n /** The index of the face on the subMesh that was picked, or the index of the Line if the picked Mesh is a LinesMesh */\r\n public subMeshFaceId = -1;\r\n /** Id of the submesh that was picked */\r\n public subMeshId = 0;\r\n /** If a sprite was picked, this will be the sprite the pick collided with */\r\n public pickedSprite: Nullable = null;\r\n /** If we are picking a mesh with thin instance, this will give you the picked thin instance */\r\n public thinInstanceIndex = -1;\r\n /**\r\n * The ray that was used to perform the picking.\r\n */\r\n public ray: Nullable = null;\r\n /**\r\n * If a mesh was used to do the picking (eg. 6dof controller) as a \"near interaction\", this will be populated.\r\n */\r\n public originMesh: Nullable = null;\r\n /**\r\n * The aim-space transform of the input used for picking, if it is an XR input source.\r\n */\r\n public aimTransform: Nullable = null;\r\n /**\r\n * The grip-space transform of the input used for picking, if it is an XR input source.\r\n * Some XR sources, such as input coming from head mounted displays, do not have this.\r\n */\r\n public gripTransform: Nullable = null;\r\n\r\n /**\r\n * Gets the normal corresponding to the face the pick collided with\r\n * @param useWorldCoordinates If the resulting normal should be relative to the world (default: false)\r\n * @param useVerticesNormals If the vertices normals should be used to calculate the normal instead of the normal map (default: true)\r\n * @returns The normal corresponding to the face the pick collided with\r\n * @remarks Note that the returned normal will always point towards the picking ray.\r\n */\r\n public getNormal(useWorldCoordinates = false, useVerticesNormals = true): Nullable {\r\n if (!this.pickedMesh || (useVerticesNormals && !this.pickedMesh.isVerticesDataPresent(VertexBuffer.NormalKind))) {\r\n return null;\r\n }\r\n\r\n let indices = this.pickedMesh.getIndices();\r\n\r\n if (indices?.length === 0) {\r\n indices = null;\r\n }\r\n\r\n let result: Vector3;\r\n\r\n const tmp0 = TmpVectors.Vector3[0];\r\n const tmp1 = TmpVectors.Vector3[1];\r\n const tmp2 = TmpVectors.Vector3[2];\r\n\r\n if (useVerticesNormals) {\r\n const normals = this.pickedMesh.getVerticesData(VertexBuffer.NormalKind);\r\n\r\n let normal0 = indices\r\n ? Vector3.FromArrayToRef(normals, indices[this.faceId * 3] * 3, tmp0)\r\n : tmp0.copyFromFloats(normals[this.faceId * 3 * 3], normals[this.faceId * 3 * 3 + 1], normals[this.faceId * 3 * 3 + 2]);\r\n let normal1 = indices\r\n ? Vector3.FromArrayToRef(normals, indices[this.faceId * 3 + 1] * 3, tmp1)\r\n : tmp1.copyFromFloats(normals[(this.faceId * 3 + 1) * 3], normals[(this.faceId * 3 + 1) * 3 + 1], normals[(this.faceId * 3 + 1) * 3 + 2]);\r\n let normal2 = indices\r\n ? Vector3.FromArrayToRef(normals, indices[this.faceId * 3 + 2] * 3, tmp2)\r\n : tmp2.copyFromFloats(normals[(this.faceId * 3 + 2) * 3], normals[(this.faceId * 3 + 2) * 3 + 1], normals[(this.faceId * 3 + 2) * 3 + 2]);\r\n\r\n normal0 = normal0.scale(this.bu);\r\n normal1 = normal1.scale(this.bv);\r\n normal2 = normal2.scale(1.0 - this.bu - this.bv);\r\n\r\n result = new Vector3(normal0.x + normal1.x + normal2.x, normal0.y + normal1.y + normal2.y, normal0.z + normal1.z + normal2.z);\r\n } else {\r\n const positions = this.pickedMesh.getVerticesData(VertexBuffer.PositionKind);\r\n\r\n const vertex1 = indices\r\n ? Vector3.FromArrayToRef(positions, indices[this.faceId * 3] * 3, tmp0)\r\n : tmp0.copyFromFloats(positions[this.faceId * 3 * 3], positions[this.faceId * 3 * 3 + 1], positions[this.faceId * 3 * 3 + 2]);\r\n const vertex2 = indices\r\n ? Vector3.FromArrayToRef(positions, indices[this.faceId * 3 + 1] * 3, tmp1)\r\n : tmp1.copyFromFloats(positions[(this.faceId * 3 + 1) * 3], positions[(this.faceId * 3 + 1) * 3 + 1], positions[(this.faceId * 3 + 1) * 3 + 2]);\r\n const vertex3 = indices\r\n ? Vector3.FromArrayToRef(positions, indices[this.faceId * 3 + 2] * 3, tmp2)\r\n : tmp2.copyFromFloats(positions[(this.faceId * 3 + 2) * 3], positions[(this.faceId * 3 + 2) * 3 + 1], positions[(this.faceId * 3 + 2) * 3 + 2]);\r\n\r\n const p1p2 = vertex1.subtract(vertex2);\r\n const p3p2 = vertex3.subtract(vertex2);\r\n\r\n result = Vector3.Cross(p1p2, p3p2);\r\n }\r\n\r\n const transformNormalToWorld = (pickedMesh: AbstractMesh, n: Vector3) => {\r\n let wm = pickedMesh.getWorldMatrix();\r\n\r\n if (pickedMesh.nonUniformScaling) {\r\n TmpVectors.Matrix[0].copyFrom(wm);\r\n wm = TmpVectors.Matrix[0];\r\n wm.setTranslationFromFloats(0, 0, 0);\r\n wm.invert();\r\n wm.transposeToRef(TmpVectors.Matrix[1]);\r\n\r\n wm = TmpVectors.Matrix[1];\r\n }\r\n\r\n Vector3.TransformNormalToRef(n, wm, n);\r\n };\r\n\r\n if (useWorldCoordinates) {\r\n transformNormalToWorld(this.pickedMesh, result);\r\n }\r\n\r\n if (this.ray) {\r\n const normalForDirectionChecking = TmpVectors.Vector3[0].copyFrom(result);\r\n\r\n if (!useWorldCoordinates) {\r\n // the normal has not been transformed to world space as part as the normal processing, so we must do it now\r\n transformNormalToWorld(this.pickedMesh, normalForDirectionChecking);\r\n }\r\n\r\n // Flip the normal if the picking ray is in the same direction.\r\n if (Vector3.Dot(normalForDirectionChecking, this.ray.direction) > 0) {\r\n result.negateInPlace();\r\n }\r\n }\r\n\r\n result.normalize();\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Gets the texture coordinates of where the pick occurred\r\n * @param uvSet The UV set to use to calculate the texture coordinates (default: VertexBuffer.UVKind)\r\n * @returns The vector containing the coordinates of the texture\r\n */\r\n public getTextureCoordinates(uvSet = VertexBuffer.UVKind): Nullable {\r\n if (!this.pickedMesh || !this.pickedMesh.isVerticesDataPresent(uvSet)) {\r\n return null;\r\n }\r\n\r\n const indices = this.pickedMesh.getIndices();\r\n if (!indices) {\r\n return null;\r\n }\r\n\r\n const uvs = this.pickedMesh.getVerticesData(uvSet);\r\n if (!uvs) {\r\n return null;\r\n }\r\n\r\n let uv0 = Vector2.FromArray(uvs, indices[this.faceId * 3] * 2);\r\n let uv1 = Vector2.FromArray(uvs, indices[this.faceId * 3 + 1] * 2);\r\n let uv2 = Vector2.FromArray(uvs, indices[this.faceId * 3 + 2] * 2);\r\n\r\n uv0 = uv0.scale(this.bu);\r\n uv1 = uv1.scale(this.bv);\r\n uv2 = uv2.scale(1.0 - this.bu - this.bv);\r\n\r\n return new Vector2(uv0.x + uv1.x + uv2.x, uv0.y + uv1.y + uv2.y);\r\n }\r\n}\r\n","import type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { Sprite } from \"../Sprites/sprite\";\r\nimport type { Scene } from \"../scene\";\r\nimport type { Vector2 } from \"../Maths/math.vector\";\r\n\r\n/**\r\n * Interface used to define ActionEvent\r\n */\r\nexport interface IActionEvent {\r\n /** The mesh or sprite that triggered the action */\r\n source: any;\r\n /** The X mouse cursor position at the time of the event */\r\n pointerX: number;\r\n /** The Y mouse cursor position at the time of the event */\r\n pointerY: number;\r\n /** The mesh that is currently pointed at (can be null) */\r\n meshUnderPointer: Nullable;\r\n /** the original (browser) event that triggered the ActionEvent */\r\n sourceEvent?: any;\r\n /** additional data for the event */\r\n additionalData?: any;\r\n}\r\n\r\n/**\r\n * ActionEvent is the event being sent when an action is triggered.\r\n */\r\nexport class ActionEvent implements IActionEvent {\r\n /**\r\n * Creates a new ActionEvent\r\n * @param source The mesh or sprite that triggered the action\r\n * @param pointerX The X mouse cursor position at the time of the event\r\n * @param pointerY The Y mouse cursor position at the time of the event\r\n * @param meshUnderPointer The mesh that is currently pointed at (can be null)\r\n * @param sourceEvent the original (browser) event that triggered the ActionEvent\r\n * @param additionalData additional data for the event\r\n */\r\n constructor(\r\n /** The mesh or sprite that triggered the action */\r\n public source: any,\r\n /** The X mouse cursor position at the time of the event */\r\n public pointerX: number,\r\n /** The Y mouse cursor position at the time of the event */\r\n public pointerY: number,\r\n /** The mesh that is currently pointed at (can be null) */\r\n public meshUnderPointer: Nullable,\r\n /** the original (browser) event that triggered the ActionEvent */\r\n public sourceEvent?: any,\r\n /** additional data for the event */\r\n public additionalData?: any\r\n ) {}\r\n\r\n /**\r\n * Helper function to auto-create an ActionEvent from a source mesh.\r\n * @param source The source mesh that triggered the event\r\n * @param evt The original (browser) event\r\n * @param additionalData additional data for the event\r\n * @returns the new ActionEvent\r\n */\r\n public static CreateNew(source: AbstractMesh, evt?: any, additionalData?: any): ActionEvent {\r\n const scene = source.getScene();\r\n return new ActionEvent(source, scene.pointerX, scene.pointerY, scene.meshUnderPointer || source, evt, additionalData);\r\n }\r\n\r\n /**\r\n * Helper function to auto-create an ActionEvent from a source sprite\r\n * @param source The source sprite that triggered the event\r\n * @param scene Scene associated with the sprite\r\n * @param evt The original (browser) event\r\n * @param additionalData additional data for the event\r\n * @returns the new ActionEvent\r\n */\r\n public static CreateNewFromSprite(source: Sprite, scene: Scene, evt?: any, additionalData?: any): ActionEvent {\r\n return new ActionEvent(source, scene.pointerX, scene.pointerY, scene.meshUnderPointer, evt, additionalData);\r\n }\r\n\r\n /**\r\n * Helper function to auto-create an ActionEvent from a scene. If triggered by a mesh use ActionEvent.CreateNew\r\n * @param scene the scene where the event occurred\r\n * @param evt The original (browser) event\r\n * @returns the new ActionEvent\r\n */\r\n public static CreateNewFromScene(scene: Scene, evt: any): ActionEvent {\r\n return new ActionEvent(null, scene.pointerX, scene.pointerY, scene.meshUnderPointer, evt);\r\n }\r\n\r\n /**\r\n * Helper function to auto-create an ActionEvent from a primitive\r\n * @param prim defines the target primitive\r\n * @param pointerPos defines the pointer position\r\n * @param evt The original (browser) event\r\n * @param additionalData additional data for the event\r\n * @returns the new ActionEvent\r\n */\r\n public static CreateNewFromPrimitive(prim: any, pointerPos: Vector2, evt?: Event, additionalData?: any): ActionEvent {\r\n return new ActionEvent(prim, pointerPos.x, pointerPos.y, null, evt, additionalData);\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\nimport type { InternalTexture } from \"../Materials/Textures/internalTexture\";\r\nimport type { PostProcess } from \"./postProcess\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport type { DataBuffer } from \"../Buffers/dataBuffer\";\r\nimport type { RenderTargetWrapper } from \"../Engines/renderTargetWrapper\";\r\n\r\nimport type { Scene } from \"../scene\";\r\n\r\n/**\r\n * PostProcessManager is used to manage one or more post processes or post process pipelines\r\n * See https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/usePostProcesses\r\n */\r\nexport class PostProcessManager {\r\n private _scene: Scene;\r\n private _indexBuffer: Nullable;\r\n private _vertexBuffers: { [key: string]: Nullable } = {};\r\n\r\n /**\r\n * Creates a new instance PostProcess\r\n * @param scene The scene that the post process is associated with.\r\n */\r\n constructor(scene: Scene) {\r\n this._scene = scene;\r\n }\r\n\r\n private _prepareBuffers(): void {\r\n if (this._vertexBuffers[VertexBuffer.PositionKind]) {\r\n return;\r\n }\r\n\r\n // VBO\r\n const vertices = [];\r\n vertices.push(1, 1);\r\n vertices.push(-1, 1);\r\n vertices.push(-1, -1);\r\n vertices.push(1, -1);\r\n\r\n this._vertexBuffers[VertexBuffer.PositionKind] = new VertexBuffer(this._scene.getEngine(), vertices, VertexBuffer.PositionKind, false, false, 2);\r\n\r\n this._buildIndexBuffer();\r\n }\r\n\r\n private _buildIndexBuffer(): void {\r\n // Indices\r\n const indices = [];\r\n indices.push(0);\r\n indices.push(1);\r\n indices.push(2);\r\n\r\n indices.push(0);\r\n indices.push(2);\r\n indices.push(3);\r\n\r\n this._indexBuffer = this._scene.getEngine().createIndexBuffer(indices);\r\n }\r\n\r\n /**\r\n * Rebuilds the vertex buffers of the manager.\r\n * @internal\r\n */\r\n public _rebuild(): void {\r\n const vb = this._vertexBuffers[VertexBuffer.PositionKind];\r\n\r\n if (!vb) {\r\n return;\r\n }\r\n vb._rebuild();\r\n this._buildIndexBuffer();\r\n }\r\n\r\n // Methods\r\n /**\r\n * Prepares a frame to be run through a post process.\r\n * @param sourceTexture The input texture to the post processes. (default: null)\r\n * @param postProcesses An array of post processes to be run. (default: null)\r\n * @returns True if the post processes were able to be run.\r\n * @internal\r\n */\r\n public _prepareFrame(sourceTexture: Nullable = null, postProcesses: Nullable = null): boolean {\r\n const camera = this._scene.activeCamera;\r\n if (!camera) {\r\n return false;\r\n }\r\n\r\n postProcesses = postProcesses || >camera._postProcesses.filter((pp) => {\r\n return pp != null;\r\n });\r\n\r\n if (!postProcesses || postProcesses.length === 0 || !this._scene.postProcessesEnabled) {\r\n return false;\r\n }\r\n\r\n postProcesses[0].activate(camera, sourceTexture, postProcesses !== null && postProcesses !== undefined);\r\n return true;\r\n }\r\n\r\n /**\r\n * Manually render a set of post processes to a texture.\r\n * Please note, the frame buffer won't be unbound after the call in case you have more render to do.\r\n * @param postProcesses An array of post processes to be run.\r\n * @param targetTexture The render target wrapper to render to.\r\n * @param forceFullscreenViewport force gl.viewport to be full screen eg. 0,0,textureWidth,textureHeight\r\n * @param faceIndex defines the face to render to if a cubemap is defined as the target\r\n * @param lodLevel defines which lod of the texture to render to\r\n * @param doNotBindFrambuffer If set to true, assumes that the framebuffer has been bound previously\r\n */\r\n public directRender(\r\n postProcesses: PostProcess[],\r\n targetTexture: Nullable = null,\r\n forceFullscreenViewport = false,\r\n faceIndex = 0,\r\n lodLevel = 0,\r\n doNotBindFrambuffer = false\r\n ): void {\r\n const engine = this._scene.getEngine();\r\n\r\n for (let index = 0; index < postProcesses.length; index++) {\r\n if (index < postProcesses.length - 1) {\r\n postProcesses[index + 1].activate(this._scene.activeCamera, targetTexture?.texture);\r\n } else {\r\n if (targetTexture) {\r\n engine.bindFramebuffer(targetTexture, faceIndex, undefined, undefined, forceFullscreenViewport, lodLevel);\r\n } else if (!doNotBindFrambuffer) {\r\n engine.restoreDefaultFramebuffer();\r\n }\r\n engine._debugInsertMarker?.(`post process ${postProcesses[index].name} output`);\r\n }\r\n\r\n const pp = postProcesses[index];\r\n const effect = pp.apply();\r\n\r\n if (effect) {\r\n pp.onBeforeRenderObservable.notifyObservers(effect);\r\n\r\n // VBOs\r\n this._prepareBuffers();\r\n engine.bindBuffers(this._vertexBuffers, this._indexBuffer, effect);\r\n\r\n // Draw order\r\n engine.drawElementsType(Constants.MATERIAL_TriangleFillMode, 0, 6);\r\n\r\n pp.onAfterRenderObservable.notifyObservers(effect);\r\n }\r\n }\r\n\r\n // Restore depth buffer\r\n engine.setDepthBuffer(true);\r\n engine.setDepthWrite(true);\r\n }\r\n\r\n /**\r\n * Finalize the result of the output of the postprocesses.\r\n * @param doNotPresent If true the result will not be displayed to the screen.\r\n * @param targetTexture The render target wrapper to render to.\r\n * @param faceIndex The index of the face to bind the target texture to.\r\n * @param postProcesses The array of post processes to render.\r\n * @param forceFullscreenViewport force gl.viewport to be full screen eg. 0,0,textureWidth,textureHeight (default: false)\r\n * @internal\r\n */\r\n public _finalizeFrame(\r\n doNotPresent?: boolean,\r\n targetTexture?: RenderTargetWrapper,\r\n faceIndex?: number,\r\n postProcesses?: Array,\r\n forceFullscreenViewport = false\r\n ): void {\r\n const camera = this._scene.activeCamera;\r\n\r\n if (!camera) {\r\n return;\r\n }\r\n\r\n postProcesses = postProcesses || >camera._postProcesses.filter((pp) => {\r\n return pp != null;\r\n });\r\n if (postProcesses.length === 0 || !this._scene.postProcessesEnabled) {\r\n return;\r\n }\r\n const engine = this._scene.getEngine();\r\n\r\n for (let index = 0, len = postProcesses.length; index < len; index++) {\r\n const pp = postProcesses[index];\r\n\r\n if (index < len - 1) {\r\n pp._outputTexture = postProcesses[index + 1].activate(camera, targetTexture?.texture);\r\n } else {\r\n if (targetTexture) {\r\n engine.bindFramebuffer(targetTexture, faceIndex, undefined, undefined, forceFullscreenViewport);\r\n pp._outputTexture = targetTexture;\r\n } else {\r\n engine.restoreDefaultFramebuffer();\r\n pp._outputTexture = null;\r\n }\r\n engine._debugInsertMarker?.(`post process ${postProcesses[index].name} output`);\r\n }\r\n\r\n if (doNotPresent) {\r\n break;\r\n }\r\n\r\n const effect = pp.apply();\r\n\r\n if (effect) {\r\n pp.onBeforeRenderObservable.notifyObservers(effect);\r\n\r\n // VBOs\r\n this._prepareBuffers();\r\n engine.bindBuffers(this._vertexBuffers, this._indexBuffer, effect);\r\n\r\n // Draw order\r\n engine.drawElementsType(Constants.MATERIAL_TriangleFillMode, 0, 6);\r\n\r\n pp.onAfterRenderObservable.notifyObservers(effect);\r\n }\r\n }\r\n\r\n // Restore states\r\n engine.setDepthBuffer(true);\r\n engine.setDepthWrite(true);\r\n engine.setAlphaMode(Constants.ALPHA_DISABLE);\r\n }\r\n\r\n /**\r\n * Disposes of the post process manager.\r\n */\r\n public dispose(): void {\r\n const buffer = this._vertexBuffers[VertexBuffer.PositionKind];\r\n if (buffer) {\r\n buffer.dispose();\r\n this._vertexBuffers[VertexBuffer.PositionKind] = null;\r\n }\r\n\r\n if (this._indexBuffer) {\r\n this._scene.getEngine()._releaseBuffer(this._indexBuffer);\r\n this._indexBuffer = null;\r\n }\r\n }\r\n}\r\n","import { SmartArray, SmartArrayNoDuplicate } from \"../Misc/smartArray\";\r\nimport type { SubMesh } from \"../Meshes/subMesh\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { Nullable, DeepImmutable } from \"../types\";\r\nimport { Vector3 } from \"../Maths/math.vector\";\r\nimport type { IParticleSystem } from \"../Particles/IParticleSystem\";\r\nimport type { IEdgesRenderer } from \"./edgesRenderer\";\r\nimport type { ISpriteManager } from \"../Sprites/spriteManager\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport type { Material } from \"../Materials/material\";\r\nimport type { Scene } from \"../scene\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\n\r\n/**\r\n * This represents the object necessary to create a rendering group.\r\n * This is exclusively used and created by the rendering manager.\r\n * To modify the behavior, you use the available helpers in your scene or meshes.\r\n * @internal\r\n */\r\nexport class RenderingGroup {\r\n private static _ZeroVector: DeepImmutable = Vector3.Zero();\r\n private _scene: Scene;\r\n private _opaqueSubMeshes = new SmartArray(256);\r\n private _transparentSubMeshes = new SmartArray(256);\r\n private _alphaTestSubMeshes = new SmartArray(256);\r\n private _depthOnlySubMeshes = new SmartArray(256);\r\n private _particleSystems = new SmartArray(256);\r\n private _spriteManagers = new SmartArray(256);\r\n\r\n private _opaqueSortCompareFn: Nullable<(a: SubMesh, b: SubMesh) => number>;\r\n private _alphaTestSortCompareFn: Nullable<(a: SubMesh, b: SubMesh) => number>;\r\n private _transparentSortCompareFn: (a: SubMesh, b: SubMesh) => number;\r\n\r\n private _renderOpaque: (subMeshes: SmartArray) => void;\r\n private _renderAlphaTest: (subMeshes: SmartArray) => void;\r\n private _renderTransparent: (subMeshes: SmartArray) => void;\r\n\r\n /** @internal */\r\n public _empty = true;\r\n\r\n /** @internal */\r\n public _edgesRenderers = new SmartArrayNoDuplicate(16);\r\n\r\n public onBeforeTransparentRendering: () => void;\r\n\r\n /**\r\n * Set the opaque sort comparison function.\r\n * If null the sub meshes will be render in the order they were created\r\n */\r\n public set opaqueSortCompareFn(value: Nullable<(a: SubMesh, b: SubMesh) => number>) {\r\n if (value) {\r\n this._opaqueSortCompareFn = value;\r\n } else {\r\n this._opaqueSortCompareFn = RenderingGroup.PainterSortCompare;\r\n }\r\n this._renderOpaque = this._renderOpaqueSorted;\r\n }\r\n\r\n /**\r\n * Set the alpha test sort comparison function.\r\n * If null the sub meshes will be render in the order they were created\r\n */\r\n public set alphaTestSortCompareFn(value: Nullable<(a: SubMesh, b: SubMesh) => number>) {\r\n if (value) {\r\n this._alphaTestSortCompareFn = value;\r\n } else {\r\n this._alphaTestSortCompareFn = RenderingGroup.PainterSortCompare;\r\n }\r\n this._renderAlphaTest = this._renderAlphaTestSorted;\r\n }\r\n\r\n /**\r\n * Set the transparent sort comparison function.\r\n * If null the sub meshes will be render in the order they were created\r\n */\r\n public set transparentSortCompareFn(value: Nullable<(a: SubMesh, b: SubMesh) => number>) {\r\n if (value) {\r\n this._transparentSortCompareFn = value;\r\n } else {\r\n this._transparentSortCompareFn = RenderingGroup.defaultTransparentSortCompare;\r\n }\r\n this._renderTransparent = this._renderTransparentSorted;\r\n }\r\n\r\n /**\r\n * Creates a new rendering group.\r\n * @param index The rendering group index\r\n * @param scene\r\n * @param opaqueSortCompareFn The opaque sort comparison function. If null no order is applied\r\n * @param alphaTestSortCompareFn The alpha test sort comparison function. If null no order is applied\r\n * @param transparentSortCompareFn The transparent sort comparison function. If null back to front + alpha index sort is applied\r\n */\r\n constructor(\r\n public index: number,\r\n scene: Scene,\r\n opaqueSortCompareFn: Nullable<(a: SubMesh, b: SubMesh) => number> = null,\r\n alphaTestSortCompareFn: Nullable<(a: SubMesh, b: SubMesh) => number> = null,\r\n transparentSortCompareFn: Nullable<(a: SubMesh, b: SubMesh) => number> = null\r\n ) {\r\n this._scene = scene;\r\n\r\n this.opaqueSortCompareFn = opaqueSortCompareFn;\r\n this.alphaTestSortCompareFn = alphaTestSortCompareFn;\r\n this.transparentSortCompareFn = transparentSortCompareFn;\r\n }\r\n\r\n /**\r\n * Render all the sub meshes contained in the group.\r\n * @param customRenderFunction Used to override the default render behaviour of the group.\r\n * @param renderSprites\r\n * @param renderParticles\r\n * @param activeMeshes\r\n * @returns true if rendered some submeshes.\r\n */\r\n public render(\r\n customRenderFunction: Nullable<\r\n (\r\n opaqueSubMeshes: SmartArray,\r\n transparentSubMeshes: SmartArray,\r\n alphaTestSubMeshes: SmartArray,\r\n depthOnlySubMeshes: SmartArray\r\n ) => void\r\n >,\r\n renderSprites: boolean,\r\n renderParticles: boolean,\r\n activeMeshes: Nullable\r\n ): void {\r\n if (customRenderFunction) {\r\n customRenderFunction(this._opaqueSubMeshes, this._alphaTestSubMeshes, this._transparentSubMeshes, this._depthOnlySubMeshes);\r\n return;\r\n }\r\n\r\n const engine = this._scene.getEngine();\r\n\r\n // Depth only\r\n if (this._depthOnlySubMeshes.length !== 0) {\r\n engine.setColorWrite(false);\r\n this._renderAlphaTest(this._depthOnlySubMeshes);\r\n engine.setColorWrite(true);\r\n }\r\n\r\n // Opaque\r\n if (this._opaqueSubMeshes.length !== 0) {\r\n this._renderOpaque(this._opaqueSubMeshes);\r\n }\r\n\r\n // Alpha test\r\n if (this._alphaTestSubMeshes.length !== 0) {\r\n this._renderAlphaTest(this._alphaTestSubMeshes);\r\n }\r\n\r\n const stencilState = engine.getStencilBuffer();\r\n engine.setStencilBuffer(false);\r\n\r\n // Sprites\r\n if (renderSprites) {\r\n this._renderSprites();\r\n }\r\n\r\n // Particles\r\n if (renderParticles) {\r\n this._renderParticles(activeMeshes);\r\n }\r\n\r\n if (this.onBeforeTransparentRendering) {\r\n this.onBeforeTransparentRendering();\r\n }\r\n\r\n // Transparent\r\n if (this._transparentSubMeshes.length !== 0 || this._scene.useOrderIndependentTransparency) {\r\n engine.setStencilBuffer(stencilState);\r\n if (this._scene.useOrderIndependentTransparency) {\r\n const excludedMeshes = this._scene.depthPeelingRenderer!.render(this._transparentSubMeshes);\r\n if (excludedMeshes.length) {\r\n // Render leftover meshes that could not be processed by depth peeling\r\n this._renderTransparent(excludedMeshes);\r\n }\r\n } else {\r\n this._renderTransparent(this._transparentSubMeshes);\r\n }\r\n engine.setAlphaMode(Constants.ALPHA_DISABLE);\r\n }\r\n\r\n // Set back stencil to false in case it changes before the edge renderer.\r\n engine.setStencilBuffer(false);\r\n\r\n // Edges\r\n if (this._edgesRenderers.length) {\r\n for (let edgesRendererIndex = 0; edgesRendererIndex < this._edgesRenderers.length; edgesRendererIndex++) {\r\n this._edgesRenderers.data[edgesRendererIndex].render();\r\n }\r\n\r\n engine.setAlphaMode(Constants.ALPHA_DISABLE);\r\n }\r\n\r\n // Restore Stencil state.\r\n engine.setStencilBuffer(stencilState);\r\n }\r\n\r\n /**\r\n * Renders the opaque submeshes in the order from the opaqueSortCompareFn.\r\n * @param subMeshes The submeshes to render\r\n */\r\n private _renderOpaqueSorted(subMeshes: SmartArray): void {\r\n return RenderingGroup._RenderSorted(subMeshes, this._opaqueSortCompareFn, this._scene.activeCamera, false);\r\n }\r\n\r\n /**\r\n * Renders the opaque submeshes in the order from the alphatestSortCompareFn.\r\n * @param subMeshes The submeshes to render\r\n */\r\n private _renderAlphaTestSorted(subMeshes: SmartArray): void {\r\n return RenderingGroup._RenderSorted(subMeshes, this._alphaTestSortCompareFn, this._scene.activeCamera, false);\r\n }\r\n\r\n /**\r\n * Renders the opaque submeshes in the order from the transparentSortCompareFn.\r\n * @param subMeshes The submeshes to render\r\n */\r\n private _renderTransparentSorted(subMeshes: SmartArray): void {\r\n return RenderingGroup._RenderSorted(subMeshes, this._transparentSortCompareFn, this._scene.activeCamera, true);\r\n }\r\n\r\n /**\r\n * Renders the submeshes in a specified order.\r\n * @param subMeshes The submeshes to sort before render\r\n * @param sortCompareFn The comparison function use to sort\r\n * @param camera The camera position use to preprocess the submeshes to help sorting\r\n * @param transparent Specifies to activate blending if true\r\n */\r\n private static _RenderSorted(\r\n subMeshes: SmartArray,\r\n sortCompareFn: Nullable<(a: SubMesh, b: SubMesh) => number>,\r\n camera: Nullable,\r\n transparent: boolean\r\n ): void {\r\n let subIndex = 0;\r\n let subMesh: SubMesh;\r\n const cameraPosition = camera ? camera.globalPosition : RenderingGroup._ZeroVector;\r\n\r\n if (transparent) {\r\n for (; subIndex < subMeshes.length; subIndex++) {\r\n subMesh = subMeshes.data[subIndex];\r\n subMesh._alphaIndex = subMesh.getMesh().alphaIndex;\r\n subMesh._distanceToCamera = Vector3.Distance(subMesh.getBoundingInfo().boundingSphere.centerWorld, cameraPosition);\r\n }\r\n }\r\n\r\n const sortedArray = subMeshes.length === subMeshes.data.length ? subMeshes.data : subMeshes.data.slice(0, subMeshes.length);\r\n\r\n if (sortCompareFn) {\r\n sortedArray.sort(sortCompareFn);\r\n }\r\n\r\n const scene = sortedArray[0].getMesh().getScene();\r\n for (subIndex = 0; subIndex < sortedArray.length; subIndex++) {\r\n subMesh = sortedArray[subIndex];\r\n\r\n if (scene._activeMeshesFrozenButKeepClipping && !subMesh.isInFrustum(scene._frustumPlanes)) {\r\n continue;\r\n }\r\n\r\n if (transparent) {\r\n const material = subMesh.getMaterial();\r\n\r\n if (material && material.needDepthPrePass) {\r\n const engine = material.getScene().getEngine();\r\n engine.setColorWrite(false);\r\n engine.setAlphaMode(Constants.ALPHA_DISABLE);\r\n subMesh.render(false);\r\n engine.setColorWrite(true);\r\n }\r\n }\r\n\r\n subMesh.render(transparent);\r\n }\r\n }\r\n\r\n /**\r\n * Build in function which can be applied to ensure meshes of a special queue (opaque, alpha test, transparent)\r\n * are rendered back to front if in the same alpha index.\r\n *\r\n * @param a The first submesh\r\n * @param b The second submesh\r\n * @returns The result of the comparison\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public static defaultTransparentSortCompare(a: SubMesh, b: SubMesh): number {\r\n // Alpha index first\r\n if (a._alphaIndex > b._alphaIndex) {\r\n return 1;\r\n }\r\n if (a._alphaIndex < b._alphaIndex) {\r\n return -1;\r\n }\r\n\r\n // Then distance to camera\r\n return RenderingGroup.backToFrontSortCompare(a, b);\r\n }\r\n\r\n /**\r\n * Build in function which can be applied to ensure meshes of a special queue (opaque, alpha test, transparent)\r\n * are rendered back to front.\r\n *\r\n * @param a The first submesh\r\n * @param b The second submesh\r\n * @returns The result of the comparison\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public static backToFrontSortCompare(a: SubMesh, b: SubMesh): number {\r\n // Then distance to camera\r\n if (a._distanceToCamera < b._distanceToCamera) {\r\n return 1;\r\n }\r\n if (a._distanceToCamera > b._distanceToCamera) {\r\n return -1;\r\n }\r\n\r\n return 0;\r\n }\r\n\r\n /**\r\n * Build in function which can be applied to ensure meshes of a special queue (opaque, alpha test, transparent)\r\n * are rendered front to back (prevent overdraw).\r\n *\r\n * @param a The first submesh\r\n * @param b The second submesh\r\n * @returns The result of the comparison\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public static frontToBackSortCompare(a: SubMesh, b: SubMesh): number {\r\n // Then distance to camera\r\n if (a._distanceToCamera < b._distanceToCamera) {\r\n return -1;\r\n }\r\n if (a._distanceToCamera > b._distanceToCamera) {\r\n return 1;\r\n }\r\n\r\n return 0;\r\n }\r\n\r\n /**\r\n * Build in function which can be applied to ensure meshes of a special queue (opaque, alpha test, transparent)\r\n * are grouped by material then geometry.\r\n *\r\n * @param a The first submesh\r\n * @param b The second submesh\r\n * @returns The result of the comparison\r\n */\r\n public static PainterSortCompare(a: SubMesh, b: SubMesh): number {\r\n const meshA = a.getMesh();\r\n const meshB = b.getMesh();\r\n\r\n if (meshA.material && meshB.material) {\r\n return meshA.material.uniqueId - meshB.material.uniqueId;\r\n }\r\n\r\n return meshA.uniqueId - meshB.uniqueId;\r\n }\r\n\r\n /**\r\n * Resets the different lists of submeshes to prepare a new frame.\r\n */\r\n public prepare(): void {\r\n this._opaqueSubMeshes.reset();\r\n this._transparentSubMeshes.reset();\r\n this._alphaTestSubMeshes.reset();\r\n this._depthOnlySubMeshes.reset();\r\n this._particleSystems.reset();\r\n this.prepareSprites();\r\n this._edgesRenderers.reset();\r\n this._empty = true;\r\n }\r\n\r\n /**\r\n * Resets the different lists of sprites to prepare a new frame.\r\n */\r\n public prepareSprites(): void {\r\n this._spriteManagers.reset();\r\n }\r\n\r\n public dispose(): void {\r\n this._opaqueSubMeshes.dispose();\r\n this._transparentSubMeshes.dispose();\r\n this._alphaTestSubMeshes.dispose();\r\n this._depthOnlySubMeshes.dispose();\r\n this._particleSystems.dispose();\r\n this._spriteManagers.dispose();\r\n this._edgesRenderers.dispose();\r\n }\r\n\r\n /**\r\n * Inserts the submesh in its correct queue depending on its material.\r\n * @param subMesh The submesh to dispatch\r\n * @param [mesh] Optional reference to the submeshes's mesh. Provide if you have an exiting reference to improve performance.\r\n * @param [material] Optional reference to the submeshes's material. Provide if you have an exiting reference to improve performance.\r\n */\r\n public dispatch(subMesh: SubMesh, mesh?: AbstractMesh, material?: Nullable): void {\r\n // Get mesh and materials if not provided\r\n if (mesh === undefined) {\r\n mesh = subMesh.getMesh();\r\n }\r\n if (material === undefined) {\r\n material = subMesh.getMaterial();\r\n }\r\n\r\n if (material === null || material === undefined) {\r\n return;\r\n }\r\n\r\n if (material.needAlphaBlendingForMesh(mesh)) {\r\n // Transparent\r\n this._transparentSubMeshes.push(subMesh);\r\n } else if (material.needAlphaTesting()) {\r\n // Alpha test\r\n if (material.needDepthPrePass) {\r\n this._depthOnlySubMeshes.push(subMesh);\r\n }\r\n\r\n this._alphaTestSubMeshes.push(subMesh);\r\n } else {\r\n if (material.needDepthPrePass) {\r\n this._depthOnlySubMeshes.push(subMesh);\r\n }\r\n\r\n this._opaqueSubMeshes.push(subMesh); // Opaque\r\n }\r\n\r\n mesh._renderingGroup = this;\r\n\r\n if (mesh._edgesRenderer && mesh._edgesRenderer.isEnabled) {\r\n this._edgesRenderers.pushNoDuplicate(mesh._edgesRenderer);\r\n }\r\n\r\n this._empty = false;\r\n }\r\n\r\n public dispatchSprites(spriteManager: ISpriteManager) {\r\n this._spriteManagers.push(spriteManager);\r\n this._empty = false;\r\n }\r\n\r\n public dispatchParticles(particleSystem: IParticleSystem) {\r\n this._particleSystems.push(particleSystem);\r\n this._empty = false;\r\n }\r\n\r\n private _renderParticles(activeMeshes: Nullable): void {\r\n if (this._particleSystems.length === 0) {\r\n return;\r\n }\r\n\r\n // Particles\r\n const activeCamera = this._scene.activeCamera;\r\n this._scene.onBeforeParticlesRenderingObservable.notifyObservers(this._scene);\r\n for (let particleIndex = 0; particleIndex < this._particleSystems.length; particleIndex++) {\r\n const particleSystem = this._particleSystems.data[particleIndex];\r\n\r\n if ((activeCamera && activeCamera.layerMask & particleSystem.layerMask) === 0) {\r\n continue;\r\n }\r\n\r\n const emitter: any = particleSystem.emitter;\r\n if (!emitter.position || !activeMeshes || activeMeshes.indexOf(emitter) !== -1) {\r\n this._scene._activeParticles.addCount(particleSystem.render(), false);\r\n }\r\n }\r\n this._scene.onAfterParticlesRenderingObservable.notifyObservers(this._scene);\r\n }\r\n\r\n private _renderSprites(): void {\r\n if (!this._scene.spritesEnabled || this._spriteManagers.length === 0) {\r\n return;\r\n }\r\n\r\n // Sprites\r\n const activeCamera = this._scene.activeCamera;\r\n this._scene.onBeforeSpritesRenderingObservable.notifyObservers(this._scene);\r\n for (let id = 0; id < this._spriteManagers.length; id++) {\r\n const spriteManager = this._spriteManagers.data[id];\r\n\r\n if ((activeCamera && activeCamera.layerMask & spriteManager.layerMask) !== 0) {\r\n spriteManager.render();\r\n }\r\n }\r\n this._scene.onAfterSpritesRenderingObservable.notifyObservers(this._scene);\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\nimport type { SmartArray } from \"../Misc/smartArray\";\r\nimport type { ISpriteManager } from \"../Sprites/spriteManager\";\r\nimport type { IParticleSystem } from \"../Particles/IParticleSystem\";\r\nimport { RenderingGroup } from \"./renderingGroup\";\r\n\r\nimport type { Scene } from \"../scene\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\nimport type { Material } from \"../Materials/material\";\r\nimport type { SubMesh } from \"../Meshes/subMesh\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\n\r\n/**\r\n * Interface describing the different options available in the rendering manager\r\n * regarding Auto Clear between groups.\r\n */\r\nexport interface IRenderingManagerAutoClearSetup {\r\n /**\r\n * Defines whether or not autoclear is enable.\r\n */\r\n autoClear: boolean;\r\n /**\r\n * Defines whether or not to autoclear the depth buffer.\r\n */\r\n depth: boolean;\r\n /**\r\n * Defines whether or not to autoclear the stencil buffer.\r\n */\r\n stencil: boolean;\r\n}\r\n\r\n/**\r\n * This class is used by the onRenderingGroupObservable\r\n */\r\nexport class RenderingGroupInfo {\r\n /**\r\n * The Scene that being rendered\r\n */\r\n scene: Scene;\r\n\r\n /**\r\n * The camera currently used for the rendering pass\r\n */\r\n camera: Nullable;\r\n\r\n /**\r\n * The ID of the renderingGroup being processed\r\n */\r\n renderingGroupId: number;\r\n}\r\n\r\n/**\r\n * This is the manager responsible of all the rendering for meshes sprites and particles.\r\n * It is enable to manage the different groups as well as the different necessary sort functions.\r\n * This should not be used directly aside of the few static configurations\r\n */\r\nexport class RenderingManager {\r\n /**\r\n * The max id used for rendering groups (not included)\r\n */\r\n public static MAX_RENDERINGGROUPS = 4;\r\n\r\n /**\r\n * The min id used for rendering groups (included)\r\n */\r\n public static MIN_RENDERINGGROUPS = 0;\r\n\r\n /**\r\n * Used to globally prevent autoclearing scenes.\r\n */\r\n public static AUTOCLEAR = true;\r\n\r\n /**\r\n * @internal\r\n */\r\n public _useSceneAutoClearSetup = false;\r\n\r\n private _scene: Scene;\r\n private _renderingGroups = new Array();\r\n private _depthStencilBufferAlreadyCleaned: boolean;\r\n\r\n private _autoClearDepthStencil: { [id: number]: IRenderingManagerAutoClearSetup } = {};\r\n private _customOpaqueSortCompareFn: { [id: number]: Nullable<(a: SubMesh, b: SubMesh) => number> } = {};\r\n private _customAlphaTestSortCompareFn: { [id: number]: Nullable<(a: SubMesh, b: SubMesh) => number> } = {};\r\n private _customTransparentSortCompareFn: { [id: number]: Nullable<(a: SubMesh, b: SubMesh) => number> } = {};\r\n private _renderingGroupInfo: Nullable = new RenderingGroupInfo();\r\n\r\n private _maintainStateBetweenFrames = false;\r\n /**\r\n * Gets or sets a boolean indicating that the manager will not reset between frames.\r\n * This means that if a mesh becomes invisible or transparent it will not be visible until this boolean is set to false again.\r\n * By default, the rendering manager will dispatch all active meshes per frame (moving them to the transparent, opaque or alpha testing lists).\r\n * By turning this property on, you will accelerate the rendering by keeping all these lists unchanged between frames.\r\n */\r\n public get maintainStateBetweenFrames() {\r\n return this._maintainStateBetweenFrames;\r\n }\r\n\r\n public set maintainStateBetweenFrames(value: boolean) {\r\n if (value === this._maintainStateBetweenFrames) {\r\n return;\r\n }\r\n\r\n this._maintainStateBetweenFrames = value;\r\n\r\n // Restore wasDispatched flags when switching to maintainStateBetweenFrames to false\r\n if (!this._maintainStateBetweenFrames) {\r\n for (const mesh of this._scene.meshes) {\r\n if (mesh.subMeshes) {\r\n for (const subMesh of mesh.subMeshes) {\r\n subMesh._wasDispatched = false;\r\n }\r\n }\r\n }\r\n\r\n if (this._scene.spriteManagers) {\r\n for (const spriteManager of this._scene.spriteManagers) {\r\n spriteManager._wasDispatched = false;\r\n }\r\n }\r\n\r\n for (const particleSystem of this._scene.particleSystems) {\r\n particleSystem._wasDispatched = false;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Instantiates a new rendering group for a particular scene\r\n * @param scene Defines the scene the groups belongs to\r\n */\r\n constructor(scene: Scene) {\r\n this._scene = scene;\r\n\r\n for (let i = RenderingManager.MIN_RENDERINGGROUPS; i < RenderingManager.MAX_RENDERINGGROUPS; i++) {\r\n this._autoClearDepthStencil[i] = { autoClear: true, depth: true, stencil: true };\r\n }\r\n }\r\n\r\n /**\r\n * Gets the rendering group with the specified id.\r\n */\r\n public getRenderingGroup(id: number): RenderingGroup {\r\n const renderingGroupId = id || 0;\r\n\r\n this._prepareRenderingGroup(renderingGroupId);\r\n\r\n return this._renderingGroups[renderingGroupId];\r\n }\r\n\r\n private _clearDepthStencilBuffer(depth = true, stencil = true): void {\r\n if (this._depthStencilBufferAlreadyCleaned) {\r\n return;\r\n }\r\n\r\n this._scene.getEngine().clear(null, false, depth, stencil);\r\n this._depthStencilBufferAlreadyCleaned = true;\r\n }\r\n\r\n /**\r\n * Renders the entire managed groups. This is used by the scene or the different render targets.\r\n * @internal\r\n */\r\n public render(\r\n customRenderFunction: Nullable<\r\n (\r\n opaqueSubMeshes: SmartArray,\r\n transparentSubMeshes: SmartArray,\r\n alphaTestSubMeshes: SmartArray,\r\n depthOnlySubMeshes: SmartArray\r\n ) => void\r\n >,\r\n activeMeshes: Nullable,\r\n renderParticles: boolean,\r\n renderSprites: boolean\r\n ): void {\r\n // Update the observable context (not null as it only goes away on dispose)\r\n const info = this._renderingGroupInfo!;\r\n info.scene = this._scene;\r\n info.camera = this._scene.activeCamera;\r\n\r\n // Dispatch sprites\r\n if (this._scene.spriteManagers && renderSprites) {\r\n for (let index = 0; index < this._scene.spriteManagers.length; index++) {\r\n const manager = this._scene.spriteManagers[index];\r\n this.dispatchSprites(manager);\r\n }\r\n }\r\n\r\n // Render\r\n for (let index = RenderingManager.MIN_RENDERINGGROUPS; index < RenderingManager.MAX_RENDERINGGROUPS; index++) {\r\n this._depthStencilBufferAlreadyCleaned = index === RenderingManager.MIN_RENDERINGGROUPS;\r\n const renderingGroup = this._renderingGroups[index];\r\n if (!renderingGroup || renderingGroup._empty) {\r\n continue;\r\n }\r\n\r\n const renderingGroupMask = Math.pow(2, index);\r\n info.renderingGroupId = index;\r\n\r\n // Before Observable\r\n this._scene.onBeforeRenderingGroupObservable.notifyObservers(info, renderingGroupMask);\r\n\r\n // Clear depth/stencil if needed\r\n if (RenderingManager.AUTOCLEAR) {\r\n const autoClear = this._useSceneAutoClearSetup ? this._scene.getAutoClearDepthStencilSetup(index) : this._autoClearDepthStencil[index];\r\n\r\n if (autoClear && autoClear.autoClear) {\r\n this._clearDepthStencilBuffer(autoClear.depth, autoClear.stencil);\r\n }\r\n }\r\n\r\n // Render\r\n for (const step of this._scene._beforeRenderingGroupDrawStage) {\r\n step.action(index);\r\n }\r\n renderingGroup.render(customRenderFunction, renderSprites, renderParticles, activeMeshes);\r\n for (const step of this._scene._afterRenderingGroupDrawStage) {\r\n step.action(index);\r\n }\r\n\r\n // After Observable\r\n this._scene.onAfterRenderingGroupObservable.notifyObservers(info, renderingGroupMask);\r\n }\r\n }\r\n\r\n /**\r\n * Resets the different information of the group to prepare a new frame\r\n * @internal\r\n */\r\n public reset(): void {\r\n if (this.maintainStateBetweenFrames) {\r\n return;\r\n }\r\n\r\n for (let index = RenderingManager.MIN_RENDERINGGROUPS; index < RenderingManager.MAX_RENDERINGGROUPS; index++) {\r\n const renderingGroup = this._renderingGroups[index];\r\n if (renderingGroup) {\r\n renderingGroup.prepare();\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Resets the sprites information of the group to prepare a new frame\r\n * @internal\r\n */\r\n public resetSprites(): void {\r\n if (this.maintainStateBetweenFrames) {\r\n return;\r\n }\r\n\r\n for (let index = RenderingManager.MIN_RENDERINGGROUPS; index < RenderingManager.MAX_RENDERINGGROUPS; index++) {\r\n const renderingGroup = this._renderingGroups[index];\r\n if (renderingGroup) {\r\n renderingGroup.prepareSprites();\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Dispose and release the group and its associated resources.\r\n * @internal\r\n */\r\n public dispose(): void {\r\n this.freeRenderingGroups();\r\n this._renderingGroups.length = 0;\r\n this._renderingGroupInfo = null;\r\n }\r\n\r\n /**\r\n * Clear the info related to rendering groups preventing retention points during dispose.\r\n */\r\n public freeRenderingGroups(): void {\r\n for (let index = RenderingManager.MIN_RENDERINGGROUPS; index < RenderingManager.MAX_RENDERINGGROUPS; index++) {\r\n const renderingGroup = this._renderingGroups[index];\r\n if (renderingGroup) {\r\n renderingGroup.dispose();\r\n }\r\n }\r\n }\r\n\r\n private _prepareRenderingGroup(renderingGroupId: number): void {\r\n if (this._renderingGroups[renderingGroupId] === undefined) {\r\n this._renderingGroups[renderingGroupId] = new RenderingGroup(\r\n renderingGroupId,\r\n this._scene,\r\n this._customOpaqueSortCompareFn[renderingGroupId],\r\n this._customAlphaTestSortCompareFn[renderingGroupId],\r\n this._customTransparentSortCompareFn[renderingGroupId]\r\n );\r\n }\r\n }\r\n\r\n /**\r\n * Add a sprite manager to the rendering manager in order to render it this frame.\r\n * @param spriteManager Define the sprite manager to render\r\n */\r\n public dispatchSprites(spriteManager: ISpriteManager) {\r\n if (this.maintainStateBetweenFrames && spriteManager._wasDispatched) {\r\n return;\r\n }\r\n spriteManager._wasDispatched = true;\r\n this.getRenderingGroup(spriteManager.renderingGroupId).dispatchSprites(spriteManager);\r\n }\r\n\r\n /**\r\n * Add a particle system to the rendering manager in order to render it this frame.\r\n * @param particleSystem Define the particle system to render\r\n */\r\n public dispatchParticles(particleSystem: IParticleSystem) {\r\n if (this.maintainStateBetweenFrames && particleSystem._wasDispatched) {\r\n return;\r\n }\r\n particleSystem._wasDispatched = true;\r\n this.getRenderingGroup(particleSystem.renderingGroupId).dispatchParticles(particleSystem);\r\n }\r\n\r\n /**\r\n * Add a submesh to the manager in order to render it this frame\r\n * @param subMesh The submesh to dispatch\r\n * @param mesh Optional reference to the submeshes's mesh. Provide if you have an exiting reference to improve performance.\r\n * @param material Optional reference to the submeshes's material. Provide if you have an exiting reference to improve performance.\r\n */\r\n public dispatch(subMesh: SubMesh, mesh?: AbstractMesh, material?: Nullable): void {\r\n if (mesh === undefined) {\r\n mesh = subMesh.getMesh();\r\n }\r\n if (this.maintainStateBetweenFrames && subMesh._wasDispatched) {\r\n return;\r\n }\r\n subMesh._wasDispatched = true;\r\n this.getRenderingGroup(mesh.renderingGroupId).dispatch(subMesh, mesh, material);\r\n }\r\n\r\n /**\r\n * Overrides the default sort function applied in the rendering group to prepare the meshes.\r\n * This allowed control for front to back rendering or reversely depending of the special needs.\r\n *\r\n * @param renderingGroupId The rendering group id corresponding to its index\r\n * @param opaqueSortCompareFn The opaque queue comparison function use to sort.\r\n * @param alphaTestSortCompareFn The alpha test queue comparison function use to sort.\r\n * @param transparentSortCompareFn The transparent queue comparison function use to sort.\r\n */\r\n public setRenderingOrder(\r\n renderingGroupId: number,\r\n opaqueSortCompareFn: Nullable<(a: SubMesh, b: SubMesh) => number> = null,\r\n alphaTestSortCompareFn: Nullable<(a: SubMesh, b: SubMesh) => number> = null,\r\n transparentSortCompareFn: Nullable<(a: SubMesh, b: SubMesh) => number> = null\r\n ) {\r\n this._customOpaqueSortCompareFn[renderingGroupId] = opaqueSortCompareFn;\r\n this._customAlphaTestSortCompareFn[renderingGroupId] = alphaTestSortCompareFn;\r\n this._customTransparentSortCompareFn[renderingGroupId] = transparentSortCompareFn;\r\n\r\n if (this._renderingGroups[renderingGroupId]) {\r\n const group = this._renderingGroups[renderingGroupId];\r\n group.opaqueSortCompareFn = this._customOpaqueSortCompareFn[renderingGroupId];\r\n group.alphaTestSortCompareFn = this._customAlphaTestSortCompareFn[renderingGroupId];\r\n group.transparentSortCompareFn = this._customTransparentSortCompareFn[renderingGroupId];\r\n }\r\n }\r\n\r\n /**\r\n * Specifies whether or not the stencil and depth buffer are cleared between two rendering groups.\r\n *\r\n * @param renderingGroupId The rendering group id corresponding to its index\r\n * @param autoClearDepthStencil Automatically clears depth and stencil between groups if true.\r\n * @param depth Automatically clears depth between groups if true and autoClear is true.\r\n * @param stencil Automatically clears stencil between groups if true and autoClear is true.\r\n */\r\n public setRenderingAutoClearDepthStencil(renderingGroupId: number, autoClearDepthStencil: boolean, depth = true, stencil = true): void {\r\n this._autoClearDepthStencil[renderingGroupId] = {\r\n autoClear: autoClearDepthStencil,\r\n depth: depth,\r\n stencil: stencil,\r\n };\r\n }\r\n\r\n /**\r\n * Gets the current auto clear configuration for one rendering group of the rendering\r\n * manager.\r\n * @param index the rendering group index to get the information for\r\n * @returns The auto clear setup for the requested rendering group\r\n */\r\n public getAutoClearDepthStencilSetup(index: number): IRenderingManagerAutoClearSetup {\r\n return this._autoClearDepthStencil[index];\r\n }\r\n}\r\n","import type { Scene } from \"./scene\";\r\nimport type { SmartArrayNoDuplicate } from \"./Misc/smartArray\";\r\nimport type { Nullable } from \"./types\";\r\nimport type { PickingInfo } from \"./Collisions/pickingInfo\";\r\nimport type { AbstractScene } from \"./abstractScene\";\r\nimport type { IPointerEvent } from \"./Events/deviceInputEvents\";\r\n\r\nimport type { Mesh } from \"./Meshes/mesh\";\r\nimport type { Effect } from \"./Materials/effect\";\r\nimport type { Camera } from \"./Cameras/camera\";\r\nimport type { AbstractMesh } from \"./Meshes/abstractMesh\";\r\nimport type { SubMesh } from \"./Meshes/subMesh\";\r\nimport type { RenderTargetTexture } from \"./Materials/Textures/renderTargetTexture\";\r\n\r\n/**\r\n * Groups all the scene component constants in one place to ease maintenance.\r\n * @internal\r\n */\r\nexport class SceneComponentConstants {\r\n public static readonly NAME_EFFECTLAYER = \"EffectLayer\";\r\n public static readonly NAME_LAYER = \"Layer\";\r\n public static readonly NAME_LENSFLARESYSTEM = \"LensFlareSystem\";\r\n public static readonly NAME_BOUNDINGBOXRENDERER = \"BoundingBoxRenderer\";\r\n public static readonly NAME_PARTICLESYSTEM = \"ParticleSystem\";\r\n public static readonly NAME_GAMEPAD = \"Gamepad\";\r\n public static readonly NAME_SIMPLIFICATIONQUEUE = \"SimplificationQueue\";\r\n public static readonly NAME_GEOMETRYBUFFERRENDERER = \"GeometryBufferRenderer\";\r\n public static readonly NAME_PREPASSRENDERER = \"PrePassRenderer\";\r\n public static readonly NAME_DEPTHRENDERER = \"DepthRenderer\";\r\n public static readonly NAME_DEPTHPEELINGRENDERER = \"DepthPeelingRenderer\";\r\n public static readonly NAME_POSTPROCESSRENDERPIPELINEMANAGER = \"PostProcessRenderPipelineManager\";\r\n public static readonly NAME_SPRITE = \"Sprite\";\r\n public static readonly NAME_SUBSURFACE = \"SubSurface\";\r\n public static readonly NAME_OUTLINERENDERER = \"Outline\";\r\n public static readonly NAME_PROCEDURALTEXTURE = \"ProceduralTexture\";\r\n public static readonly NAME_SHADOWGENERATOR = \"ShadowGenerator\";\r\n public static readonly NAME_OCTREE = \"Octree\";\r\n public static readonly NAME_PHYSICSENGINE = \"PhysicsEngine\";\r\n public static readonly NAME_AUDIO = \"Audio\";\r\n public static readonly NAME_FLUIDRENDERER = \"FluidRenderer\";\r\n\r\n public static readonly STEP_ISREADYFORMESH_EFFECTLAYER = 0;\r\n\r\n public static readonly STEP_BEFOREEVALUATEACTIVEMESH_BOUNDINGBOXRENDERER = 0;\r\n\r\n public static readonly STEP_EVALUATESUBMESH_BOUNDINGBOXRENDERER = 0;\r\n\r\n public static readonly STEP_PREACTIVEMESH_BOUNDINGBOXRENDERER = 0;\r\n\r\n public static readonly STEP_CAMERADRAWRENDERTARGET_EFFECTLAYER = 1;\r\n\r\n public static readonly STEP_BEFORECAMERADRAW_PREPASS = 0;\r\n public static readonly STEP_BEFORECAMERADRAW_EFFECTLAYER = 1;\r\n public static readonly STEP_BEFORECAMERADRAW_LAYER = 2;\r\n\r\n public static readonly STEP_BEFORERENDERTARGETDRAW_PREPASS = 0;\r\n public static readonly STEP_BEFORERENDERTARGETDRAW_LAYER = 1;\r\n\r\n public static readonly STEP_BEFORERENDERINGMESH_PREPASS = 0;\r\n public static readonly STEP_BEFORERENDERINGMESH_OUTLINE = 1;\r\n\r\n public static readonly STEP_AFTERRENDERINGMESH_PREPASS = 0;\r\n public static readonly STEP_AFTERRENDERINGMESH_OUTLINE = 1;\r\n\r\n public static readonly STEP_AFTERRENDERINGGROUPDRAW_EFFECTLAYER_DRAW = 0;\r\n public static readonly STEP_AFTERRENDERINGGROUPDRAW_BOUNDINGBOXRENDERER = 1;\r\n\r\n public static readonly STEP_BEFORECAMERAUPDATE_SIMPLIFICATIONQUEUE = 0;\r\n public static readonly STEP_BEFORECAMERAUPDATE_GAMEPAD = 1;\r\n\r\n public static readonly STEP_BEFORECLEAR_PROCEDURALTEXTURE = 0;\r\n public static readonly STEP_BEFORECLEAR_PREPASS = 1;\r\n\r\n public static readonly STEP_BEFORERENDERTARGETCLEAR_PREPASS = 0;\r\n\r\n public static readonly STEP_AFTERRENDERTARGETDRAW_PREPASS = 0;\r\n public static readonly STEP_AFTERRENDERTARGETDRAW_LAYER = 1;\r\n\r\n public static readonly STEP_AFTERCAMERADRAW_PREPASS = 0;\r\n public static readonly STEP_AFTERCAMERADRAW_EFFECTLAYER = 1;\r\n public static readonly STEP_AFTERCAMERADRAW_LENSFLARESYSTEM = 2;\r\n public static readonly STEP_AFTERCAMERADRAW_EFFECTLAYER_DRAW = 3;\r\n public static readonly STEP_AFTERCAMERADRAW_LAYER = 4;\r\n public static readonly STEP_AFTERCAMERADRAW_FLUIDRENDERER = 5;\r\n\r\n public static readonly STEP_AFTERCAMERAPOSTPROCESS_LAYER = 0;\r\n\r\n public static readonly STEP_AFTERRENDERTARGETPOSTPROCESS_LAYER = 0;\r\n\r\n public static readonly STEP_AFTERRENDER_AUDIO = 0;\r\n\r\n public static readonly STEP_GATHERRENDERTARGETS_DEPTHRENDERER = 0;\r\n public static readonly STEP_GATHERRENDERTARGETS_GEOMETRYBUFFERRENDERER = 1;\r\n public static readonly STEP_GATHERRENDERTARGETS_SHADOWGENERATOR = 2;\r\n public static readonly STEP_GATHERRENDERTARGETS_POSTPROCESSRENDERPIPELINEMANAGER = 3;\r\n\r\n public static readonly STEP_GATHERACTIVECAMERARENDERTARGETS_DEPTHRENDERER = 0;\r\n public static readonly STEP_GATHERACTIVECAMERARENDERTARGETS_FLUIDRENDERER = 1;\r\n\r\n public static readonly STEP_POINTERMOVE_SPRITE = 0;\r\n public static readonly STEP_POINTERDOWN_SPRITE = 0;\r\n public static readonly STEP_POINTERUP_SPRITE = 0;\r\n}\r\n\r\n/**\r\n * This represents a scene component.\r\n *\r\n * This is used to decouple the dependency the scene is having on the different workloads like\r\n * layers, post processes...\r\n */\r\nexport interface ISceneComponent {\r\n /**\r\n * The name of the component. Each component must have a unique name.\r\n */\r\n name: string;\r\n\r\n /**\r\n * The scene the component belongs to.\r\n */\r\n scene: Scene;\r\n\r\n /**\r\n * Register the component to one instance of a scene.\r\n */\r\n register(): void;\r\n\r\n /**\r\n * Rebuilds the elements related to this component in case of\r\n * context lost for instance.\r\n */\r\n rebuild(): void;\r\n\r\n /**\r\n * Disposes the component and the associated ressources.\r\n */\r\n dispose(): void;\r\n}\r\n\r\n/**\r\n * This represents a SERIALIZABLE scene component.\r\n *\r\n * This extends Scene Component to add Serialization methods on top.\r\n */\r\nexport interface ISceneSerializableComponent extends ISceneComponent {\r\n /**\r\n * Adds all the elements from the container to the scene\r\n * @param container the container holding the elements\r\n */\r\n addFromContainer(container: AbstractScene): void;\r\n\r\n /**\r\n * Removes all the elements in the container from the scene\r\n * @param container contains the elements to remove\r\n * @param dispose if the removed element should be disposed (default: false)\r\n */\r\n removeFromContainer(container: AbstractScene, dispose?: boolean): void;\r\n\r\n /**\r\n * Serializes the component data to the specified json object\r\n * @param serializationObject The object to serialize to\r\n */\r\n serialize(serializationObject: any): void;\r\n}\r\n\r\n/**\r\n * Strong typing of a Mesh related stage step action\r\n */\r\nexport type MeshStageAction = (mesh: AbstractMesh, hardwareInstancedRendering: boolean) => boolean;\r\n\r\n/**\r\n * Strong typing of a Evaluate Sub Mesh related stage step action\r\n */\r\nexport type EvaluateSubMeshStageAction = (mesh: AbstractMesh, subMesh: SubMesh) => void;\r\n\r\n/**\r\n * Strong typing of a pre active Mesh related stage step action\r\n */\r\nexport type PreActiveMeshStageAction = (mesh: AbstractMesh) => void;\r\n\r\n/**\r\n * Strong typing of a Camera related stage step action\r\n */\r\nexport type CameraStageAction = (camera: Camera) => void;\r\n\r\n/**\r\n * Strong typing of a Camera Frame buffer related stage step action\r\n */\r\nexport type CameraStageFrameBufferAction = (camera: Camera) => boolean;\r\n\r\n/**\r\n * Strong typing of a Render Target related stage step action\r\n */\r\nexport type RenderTargetStageAction = (renderTarget: RenderTargetTexture, faceIndex?: number, layer?: number) => void;\r\n\r\n/**\r\n * Strong typing of a RenderingGroup related stage step action\r\n */\r\nexport type RenderingGroupStageAction = (renderingGroupId: number) => void;\r\n\r\n/**\r\n * Strong typing of a Mesh Render related stage step action\r\n */\r\nexport type RenderingMeshStageAction = (mesh: Mesh, subMesh: SubMesh, batch: any, effect: Nullable) => void;\r\n\r\n/**\r\n * Strong typing of a simple stage step action\r\n */\r\nexport type SimpleStageAction = () => void;\r\n\r\n/**\r\n * Strong typing of a render target action.\r\n */\r\nexport type RenderTargetsStageAction = (renderTargets: SmartArrayNoDuplicate) => void;\r\n\r\n/**\r\n * Strong typing of a pointer move action.\r\n */\r\nexport type PointerMoveStageAction = (\r\n unTranslatedPointerX: number,\r\n unTranslatedPointerY: number,\r\n pickResult: Nullable,\r\n isMeshPicked: boolean,\r\n element: Nullable\r\n) => Nullable;\r\n\r\n/**\r\n * Strong typing of a pointer up/down action.\r\n */\r\nexport type PointerUpDownStageAction = (\r\n unTranslatedPointerX: number,\r\n unTranslatedPointerY: number,\r\n pickResult: Nullable,\r\n evt: IPointerEvent,\r\n doubleClick: boolean\r\n) => Nullable;\r\n\r\n/**\r\n * Representation of a stage in the scene (Basically a list of ordered steps)\r\n * @internal\r\n */\r\nexport class Stage extends Array<{ index: number; component: ISceneComponent; action: T }> {\r\n /**\r\n * Hide ctor from the rest of the world.\r\n * @param items The items to add.\r\n */\r\n private constructor(items?: { index: number; component: ISceneComponent; action: T }[]) {\r\n super(...(items));\r\n }\r\n\r\n /**\r\n * Creates a new Stage.\r\n * @returns A new instance of a Stage\r\n */\r\n static Create(): Stage {\r\n return Object.create(Stage.prototype);\r\n }\r\n\r\n /**\r\n * Registers a step in an ordered way in the targeted stage.\r\n * @param index Defines the position to register the step in\r\n * @param component Defines the component attached to the step\r\n * @param action Defines the action to launch during the step\r\n */\r\n public registerStep(index: number, component: ISceneComponent, action: T): void {\r\n let i = 0;\r\n let maxIndex = Number.MAX_VALUE;\r\n for (; i < this.length; i++) {\r\n const step = this[i];\r\n maxIndex = step.index;\r\n if (index < maxIndex) {\r\n break;\r\n }\r\n }\r\n this.splice(i, 0, { index, component, action: action.bind(component) });\r\n }\r\n\r\n /**\r\n * Clears all the steps from the stage.\r\n */\r\n public clear(): void {\r\n this.length = 0;\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\nimport { Vector2 } from \"../Maths/math.vector\";\r\nimport type { PickingInfo } from \"../Collisions/pickingInfo\";\r\nimport type { IMouseEvent, IPointerEvent } from \"./deviceInputEvents\";\r\nimport type { InputManager } from \"../Inputs/scene.inputManager\";\r\n\r\nimport type { Ray } from \"../Culling/ray\";\r\n\r\n/**\r\n * Gather the list of pointer event types as constants.\r\n */\r\nexport class PointerEventTypes {\r\n /**\r\n * The pointerdown event is fired when a pointer becomes active. For mouse, it is fired when the device transitions from no buttons depressed to at least one button depressed. For touch, it is fired when physical contact is made with the digitizer. For pen, it is fired when the stylus makes physical contact with the digitizer.\r\n */\r\n public static readonly POINTERDOWN = 0x01;\r\n /**\r\n * The pointerup event is fired when a pointer is no longer active.\r\n */\r\n public static readonly POINTERUP = 0x02;\r\n /**\r\n * The pointermove event is fired when a pointer changes coordinates.\r\n */\r\n public static readonly POINTERMOVE = 0x04;\r\n /**\r\n * The pointerwheel event is fired when a mouse wheel has been rotated.\r\n */\r\n public static readonly POINTERWHEEL = 0x08;\r\n /**\r\n * The pointerpick event is fired when a mesh or sprite has been picked by the pointer.\r\n */\r\n public static readonly POINTERPICK = 0x10;\r\n /**\r\n * The pointertap event is fired when a the object has been touched and released without drag.\r\n */\r\n public static readonly POINTERTAP = 0x20;\r\n /**\r\n * The pointerdoubletap event is fired when a the object has been touched and released twice without drag.\r\n */\r\n public static readonly POINTERDOUBLETAP = 0x40;\r\n}\r\n\r\n/**\r\n * Base class of pointer info types.\r\n */\r\nexport class PointerInfoBase {\r\n /**\r\n * Instantiates the base class of pointers info.\r\n * @param type Defines the type of event (PointerEventTypes)\r\n * @param event Defines the related dom event\r\n */\r\n constructor(\r\n /**\r\n * Defines the type of event (PointerEventTypes)\r\n */\r\n public type: number,\r\n /**\r\n * Defines the related dom event\r\n */\r\n public event: IMouseEvent\r\n ) {}\r\n}\r\n\r\n/**\r\n * This class is used to store pointer related info for the onPrePointerObservable event.\r\n * Set the skipOnPointerObservable property to true if you want the engine to stop any process after this event is triggered, even not calling onPointerObservable\r\n */\r\nexport class PointerInfoPre extends PointerInfoBase {\r\n /**\r\n * Ray from a pointer if available (eg. 6dof controller)\r\n */\r\n public ray: Nullable = null;\r\n\r\n /**\r\n * Defines picking info coming from a near interaction (proximity instead of ray-based picking)\r\n */\r\n public nearInteractionPickingInfo: Nullable;\r\n\r\n /**\r\n * The original picking info that was used to trigger the pointer event\r\n */\r\n public originalPickingInfo: Nullable = null;\r\n\r\n /**\r\n * Defines the local position of the pointer on the canvas.\r\n */\r\n public localPosition: Vector2;\r\n\r\n /**\r\n * Defines whether the engine should skip the next OnPointerObservable associated to this pre.\r\n */\r\n public skipOnPointerObservable: boolean;\r\n\r\n /**\r\n * Instantiates a PointerInfoPre to store pointer related info to the onPrePointerObservable event.\r\n * @param type Defines the type of event (PointerEventTypes)\r\n * @param event Defines the related dom event\r\n * @param localX Defines the local x coordinates of the pointer when the event occured\r\n * @param localY Defines the local y coordinates of the pointer when the event occured\r\n */\r\n constructor(type: number, event: IMouseEvent, localX: number, localY: number) {\r\n super(type, event);\r\n this.skipOnPointerObservable = false;\r\n this.localPosition = new Vector2(localX, localY);\r\n }\r\n}\r\n\r\n/**\r\n * This type contains all the data related to a pointer event in Babylon.js.\r\n * The event member is an instance of PointerEvent for all types except PointerWheel and is of type MouseWheelEvent when type equals PointerWheel. The different event types can be found in the PointerEventTypes class.\r\n */\r\nexport class PointerInfo extends PointerInfoBase {\r\n private _pickInfo: Nullable;\r\n private _inputManager: Nullable;\r\n\r\n /**\r\n * Defines the picking info associated with this PointerInfo object (if applicable)\r\n */\r\n public get pickInfo(): Nullable {\r\n if (!this._pickInfo) {\r\n this._generatePickInfo();\r\n }\r\n\r\n return this._pickInfo;\r\n }\r\n /**\r\n * Instantiates a PointerInfo to store pointer related info to the onPointerObservable event.\r\n * @param type Defines the type of event (PointerEventTypes)\r\n * @param event Defines the related dom event\r\n * @param pickInfo Defines the picking info associated to the info (if any)\r\n * @param inputManager Defines the InputManager to use if there is no pickInfo\r\n */\r\n constructor(type: number, event: IMouseEvent, pickInfo: Nullable, inputManager: Nullable = null) {\r\n super(type, event);\r\n this._pickInfo = pickInfo;\r\n this._inputManager = inputManager;\r\n }\r\n\r\n /**\r\n * Generates the picking info if needed\r\n */\r\n /** @internal */\r\n public _generatePickInfo(): void {\r\n if (this._inputManager) {\r\n this._pickInfo = this._inputManager._pickMove(this.event as IPointerEvent);\r\n this._inputManager._setRayOnPointerInfo(this._pickInfo, this.event);\r\n this._inputManager = null;\r\n }\r\n }\r\n}\r\n\r\n/**\r\n * Data relating to a touch event on the screen.\r\n */\r\nexport interface PointerTouch {\r\n /**\r\n * X coordinate of touch.\r\n */\r\n x: number;\r\n /**\r\n * Y coordinate of touch.\r\n */\r\n y: number;\r\n /**\r\n * Id of touch. Unique for each finger.\r\n */\r\n pointerId: number;\r\n /**\r\n * Event type passed from DOM.\r\n */\r\n type: any;\r\n}\r\n","import type { IDisposable } from \"../scene\";\r\nimport type { IActionEvent } from \"./actionEvent\";\r\nimport type { IAction } from \"./action\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport type { Nullable } from \"../types\";\r\n\r\n/**\r\n * Abstract class used to decouple action Manager from scene and meshes.\r\n * Do not instantiate.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions\r\n */\r\nexport abstract class AbstractActionManager implements IDisposable {\r\n /** Gets the list of active triggers */\r\n public static Triggers: { [key: string]: number } = {};\r\n\r\n /** Gets the cursor to use when hovering items */\r\n public hoverCursor: string = \"\";\r\n\r\n /** Gets the list of actions */\r\n public actions = new Array();\r\n\r\n /**\r\n * Gets or sets a boolean indicating that the manager is recursive meaning that it can trigger action from children\r\n */\r\n public isRecursive = false;\r\n\r\n /**\r\n * Releases all associated resources\r\n */\r\n public abstract dispose(): void;\r\n\r\n /**\r\n * Does this action manager has pointer triggers\r\n */\r\n public abstract get hasPointerTriggers(): boolean;\r\n\r\n /**\r\n * Does this action manager has pick triggers\r\n */\r\n public abstract get hasPickTriggers(): boolean;\r\n\r\n /**\r\n * Process a specific trigger\r\n * @param trigger defines the trigger to process\r\n * @param evt defines the event details to be processed\r\n */\r\n public abstract processTrigger(trigger: number, evt?: IActionEvent): void;\r\n\r\n /**\r\n * Does this action manager handles actions of any of the given triggers\r\n * @param triggers defines the triggers to be tested\r\n * @returns a boolean indicating whether one (or more) of the triggers is handled\r\n */\r\n public abstract hasSpecificTriggers(triggers: number[]): boolean;\r\n\r\n /**\r\n * Does this action manager handles actions of any of the given triggers. This function takes two arguments for\r\n * speed.\r\n * @param triggerA defines the trigger to be tested\r\n * @param triggerB defines the trigger to be tested\r\n * @returns a boolean indicating whether one (or more) of the triggers is handled\r\n */\r\n public abstract hasSpecificTriggers2(triggerA: number, triggerB: number): boolean;\r\n\r\n /**\r\n * Does this action manager handles actions of a given trigger\r\n * @param trigger defines the trigger to be tested\r\n * @param parameterPredicate defines an optional predicate to filter triggers by parameter\r\n * @returns whether the trigger is handled\r\n */\r\n public abstract hasSpecificTrigger(trigger: number, parameterPredicate?: (parameter: any) => boolean): boolean;\r\n\r\n /**\r\n * Serialize this manager to a JSON object\r\n * @param name defines the property name to store this manager\r\n * @returns a JSON representation of this manager\r\n */\r\n public abstract serialize(name: string): any;\r\n\r\n /**\r\n * Registers an action to this action manager\r\n * @param action defines the action to be registered\r\n * @returns the action amended (prepared) after registration\r\n */\r\n public abstract registerAction(action: IAction): Nullable;\r\n\r\n /**\r\n * Unregisters an action to this action manager\r\n * @param action defines the action to be unregistered\r\n * @returns a boolean indicating whether the action has been unregistered\r\n */\r\n public abstract unregisterAction(action: IAction): Boolean;\r\n\r\n /**\r\n * Does exist one action manager with at least one trigger\r\n **/\r\n public static get HasTriggers(): boolean {\r\n for (const t in AbstractActionManager.Triggers) {\r\n if (Object.prototype.hasOwnProperty.call(AbstractActionManager.Triggers, t)) {\r\n return true;\r\n }\r\n }\r\n return false;\r\n }\r\n\r\n /**\r\n * Does exist one action manager with at least one pick trigger\r\n **/\r\n public static get HasPickTriggers(): boolean {\r\n for (const t in AbstractActionManager.Triggers) {\r\n if (Object.prototype.hasOwnProperty.call(AbstractActionManager.Triggers, t)) {\r\n const tAsInt = parseInt(t);\r\n if (tAsInt >= Constants.ACTION_OnPickTrigger && tAsInt <= Constants.ACTION_OnPickUpTrigger) {\r\n return true;\r\n }\r\n }\r\n }\r\n return false;\r\n }\r\n\r\n /**\r\n * Does exist one action manager that handles actions of a given trigger\r\n * @param trigger defines the trigger to be tested\r\n * @returns a boolean indicating whether the trigger is handled by at least one action manager\r\n **/\r\n public static HasSpecificTrigger(trigger: number): boolean {\r\n for (const t in AbstractActionManager.Triggers) {\r\n if (Object.prototype.hasOwnProperty.call(AbstractActionManager.Triggers, t)) {\r\n const tAsInt = parseInt(t);\r\n if (tAsInt === trigger) {\r\n return true;\r\n }\r\n }\r\n }\r\n return false;\r\n }\r\n}\r\n","import type { IKeyboardEvent } from \"./deviceInputEvents\";\r\n\r\n/**\r\n * Gather the list of keyboard event types as constants.\r\n */\r\nexport class KeyboardEventTypes {\r\n /**\r\n * The keydown event is fired when a key becomes active (pressed).\r\n */\r\n public static readonly KEYDOWN = 0x01;\r\n /**\r\n * The keyup event is fired when a key has been released.\r\n */\r\n public static readonly KEYUP = 0x02;\r\n}\r\n\r\n/**\r\n * This class is used to store keyboard related info for the onKeyboardObservable event.\r\n */\r\nexport class KeyboardInfo {\r\n /**\r\n * Instantiates a new keyboard info.\r\n * This class is used to store keyboard related info for the onKeyboardObservable event.\r\n * @param type Defines the type of event (KeyboardEventTypes)\r\n * @param event Defines the related dom event\r\n */\r\n constructor(\r\n /**\r\n * Defines the type of event (KeyboardEventTypes)\r\n */\r\n public type: number,\r\n /**\r\n * Defines the related dom event\r\n */\r\n public event: IKeyboardEvent\r\n ) {}\r\n}\r\n\r\n/**\r\n * This class is used to store keyboard related info for the onPreKeyboardObservable event.\r\n * Set the skipOnKeyboardObservable property to true if you want the engine to stop any process after this event is triggered, even not calling onKeyboardObservable\r\n */\r\nexport class KeyboardInfoPre extends KeyboardInfo {\r\n /**\r\n * Defines whether the engine should skip the next onKeyboardObservable associated to this pre.\r\n */\r\n public skipOnKeyboardObservable: boolean;\r\n\r\n /**\r\n * Defines whether the engine should skip the next onKeyboardObservable associated to this pre.\r\n * @deprecated use skipOnKeyboardObservable property instead\r\n */\r\n public get skipOnPointerObservable() {\r\n return this.skipOnKeyboardObservable;\r\n }\r\n public set skipOnPointerObservable(value) {\r\n this.skipOnKeyboardObservable = value;\r\n }\r\n\r\n /**\r\n * Instantiates a new keyboard pre info.\r\n * This class is used to store keyboard related info for the onPreKeyboardObservable event.\r\n * @param type Defines the type of event (KeyboardEventTypes)\r\n * @param event Defines the related dom event\r\n */\r\n constructor(\r\n /**\r\n * Defines the type of event (KeyboardEventTypes)\r\n */\r\n public type: number,\r\n /**\r\n * Defines the related dom event\r\n */\r\n public event: IKeyboardEvent\r\n ) {\r\n super(type, event);\r\n this.skipOnKeyboardObservable = false;\r\n }\r\n}\r\n","import { Constants } from \"../Engines/constants\";\r\nimport type { IUIEvent } from \"../Events/deviceInputEvents\";\r\nimport { EventConstants } from \"../Events/deviceInputEvents\";\r\nimport type { Nullable } from \"../types\";\r\nimport { DeviceType, NativePointerInput, PointerInput } from \"./InputDevices/deviceEnums\";\r\nimport type { IDeviceInputSystem } from \"./inputInterfaces\";\r\n\r\n/**\r\n * Class to wrap DeviceInputSystem data into an event object\r\n */\r\nexport class DeviceEventFactory {\r\n /**\r\n * Create device input events based on provided type and slot\r\n *\r\n * @param deviceType Type of device\r\n * @param deviceSlot \"Slot\" or index that device is referenced in\r\n * @param inputIndex Id of input to be checked\r\n * @param currentState Current value for given input\r\n * @param deviceInputSystem Reference to DeviceInputSystem\r\n * @param elementToAttachTo HTMLElement to reference as target for inputs\r\n * @returns IUIEvent object\r\n */\r\n public static CreateDeviceEvent(\r\n deviceType: DeviceType,\r\n deviceSlot: number,\r\n inputIndex: number,\r\n currentState: Nullable,\r\n deviceInputSystem: IDeviceInputSystem,\r\n elementToAttachTo?: any,\r\n pointerId?: number\r\n ): IUIEvent {\r\n switch (deviceType) {\r\n case DeviceType.Keyboard:\r\n return this._CreateKeyboardEvent(inputIndex, currentState, deviceInputSystem, elementToAttachTo);\r\n case DeviceType.Mouse:\r\n if (inputIndex === PointerInput.MouseWheelX || inputIndex === PointerInput.MouseWheelY || inputIndex === PointerInput.MouseWheelZ) {\r\n return this._CreateWheelEvent(deviceType, deviceSlot, inputIndex, currentState, deviceInputSystem, elementToAttachTo);\r\n }\r\n // eslint-disable-next-line no-fallthrough\r\n case DeviceType.Touch:\r\n return this._CreatePointerEvent(deviceType, deviceSlot, inputIndex, currentState, deviceInputSystem, elementToAttachTo, pointerId);\r\n default:\r\n throw `Unable to generate event for device ${DeviceType[deviceType]}`;\r\n }\r\n }\r\n\r\n /**\r\n * Creates pointer event\r\n *\r\n * @param deviceType Type of device\r\n * @param deviceSlot \"Slot\" or index that device is referenced in\r\n * @param inputIndex Id of input to be checked\r\n * @param currentState Current value for given input\r\n * @param deviceInputSystem Reference to DeviceInputSystem\r\n * @param elementToAttachTo HTMLElement to reference as target for inputs\r\n * @returns IUIEvent object (Pointer)\r\n */\r\n private static _CreatePointerEvent(\r\n deviceType: DeviceType,\r\n deviceSlot: number,\r\n inputIndex: number,\r\n currentState: Nullable,\r\n deviceInputSystem: IDeviceInputSystem,\r\n elementToAttachTo?: any,\r\n pointerId?: number\r\n ): any {\r\n const evt = this._CreateMouseEvent(deviceType, deviceSlot, inputIndex, currentState, deviceInputSystem, elementToAttachTo);\r\n\r\n if (deviceType === DeviceType.Mouse) {\r\n evt.deviceType = DeviceType.Mouse;\r\n evt.pointerId = 1;\r\n evt.pointerType = \"mouse\";\r\n } else {\r\n evt.deviceType = DeviceType.Touch;\r\n evt.pointerId = pointerId ?? deviceSlot;\r\n evt.pointerType = \"touch\";\r\n }\r\n\r\n let buttons = 0;\r\n\r\n // Populate buttons property with current state of all mouse buttons\r\n // Uses values found on: https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent/buttons\r\n buttons += deviceInputSystem.pollInput(deviceType, deviceSlot, PointerInput.LeftClick);\r\n buttons += deviceInputSystem.pollInput(deviceType, deviceSlot, PointerInput.RightClick) * 2;\r\n buttons += deviceInputSystem.pollInput(deviceType, deviceSlot, PointerInput.MiddleClick) * 4;\r\n evt.buttons = buttons;\r\n\r\n if (inputIndex === PointerInput.Move) {\r\n evt.type = \"pointermove\";\r\n } else if (inputIndex >= PointerInput.LeftClick && inputIndex <= PointerInput.RightClick) {\r\n evt.type = currentState === 1 ? \"pointerdown\" : \"pointerup\";\r\n evt.button = inputIndex - 2;\r\n }\r\n\r\n return evt;\r\n }\r\n\r\n /**\r\n * Create Mouse Wheel Event\r\n * @param deviceType Type of device\r\n * @param deviceSlot \"Slot\" or index that device is referenced in\r\n * @param inputIndex Id of input to be checked\r\n * @param currentState Current value for given input\r\n * @param deviceInputSystem Reference to DeviceInputSystem\r\n * @param elementToAttachTo HTMLElement to reference as target for inputs\r\n * @returns IUIEvent object (Wheel)\r\n */\r\n private static _CreateWheelEvent(\r\n deviceType: DeviceType,\r\n deviceSlot: number,\r\n inputIndex: number,\r\n currentState: Nullable,\r\n deviceInputSystem: IDeviceInputSystem,\r\n elementToAttachTo: any\r\n ): any {\r\n const evt = this._CreateMouseEvent(deviceType, deviceSlot, inputIndex, currentState, deviceInputSystem, elementToAttachTo);\r\n\r\n // While WheelEvents don't generally have a pointerId, we used to add one in the InputManager\r\n // This line has been added to make the InputManager more platform-agnostic\r\n // Similar code exists in the WebDeviceInputSystem to handle browser created events\r\n evt.pointerId = 1;\r\n evt.type = \"wheel\";\r\n evt.deltaMode = EventConstants.DOM_DELTA_PIXEL;\r\n evt.deltaX = 0;\r\n evt.deltaY = 0;\r\n evt.deltaZ = 0;\r\n\r\n switch (inputIndex) {\r\n case PointerInput.MouseWheelX:\r\n evt.deltaX = currentState;\r\n break;\r\n case PointerInput.MouseWheelY:\r\n evt.deltaY = currentState;\r\n break;\r\n case PointerInput.MouseWheelZ:\r\n evt.deltaZ = currentState;\r\n break;\r\n }\r\n\r\n return evt;\r\n }\r\n\r\n /**\r\n * Create Mouse Event\r\n * @param deviceType Type of device\r\n * @param deviceSlot \"Slot\" or index that device is referenced in\r\n * @param inputIndex Id of input to be checked\r\n * @param currentState Current value for given input\r\n * @param deviceInputSystem Reference to DeviceInputSystem\r\n * @param elementToAttachTo HTMLElement to reference as target for inputs\r\n * @returns IUIEvent object (Mouse)\r\n */\r\n private static _CreateMouseEvent(\r\n deviceType: DeviceType,\r\n deviceSlot: number,\r\n inputIndex: number,\r\n currentState: Nullable,\r\n deviceInputSystem: IDeviceInputSystem,\r\n elementToAttachTo?: any\r\n ): any {\r\n const evt = this._CreateEvent(elementToAttachTo);\r\n const pointerX = deviceInputSystem.pollInput(deviceType, deviceSlot, PointerInput.Horizontal);\r\n const pointerY = deviceInputSystem.pollInput(deviceType, deviceSlot, PointerInput.Vertical);\r\n\r\n // Handle offsets/deltas based on existence of HTMLElement\r\n if (elementToAttachTo) {\r\n evt.movementX = 0;\r\n evt.movementY = 0;\r\n evt.offsetX = evt.movementX - elementToAttachTo.getBoundingClientRect().x;\r\n evt.offsetY = evt.movementY - elementToAttachTo.getBoundingClientRect().y;\r\n } else {\r\n evt.movementX = deviceInputSystem.pollInput(deviceType, deviceSlot, NativePointerInput.DeltaHorizontal); // DeltaHorizontal\r\n evt.movementY = deviceInputSystem.pollInput(deviceType, deviceSlot, NativePointerInput.DeltaVertical); // DeltaVertical\r\n evt.offsetX = 0;\r\n evt.offsetY = 0;\r\n }\r\n this._CheckNonCharacterKeys(evt, deviceInputSystem);\r\n\r\n evt.clientX = pointerX;\r\n evt.clientY = pointerY;\r\n evt.x = pointerX;\r\n evt.y = pointerY;\r\n\r\n evt.deviceType = deviceType;\r\n evt.deviceSlot = deviceSlot;\r\n evt.inputIndex = inputIndex;\r\n\r\n return evt;\r\n }\r\n\r\n /**\r\n * Create Keyboard Event\r\n * @param inputIndex Id of input to be checked\r\n * @param currentState Current value for given input\r\n * @param deviceInputSystem Reference to DeviceInputSystem\r\n * @param elementToAttachTo HTMLElement to reference as target for inputs\r\n * @returns IEvent object (Keyboard)\r\n */\r\n private static _CreateKeyboardEvent(inputIndex: number, currentState: Nullable, deviceInputSystem: IDeviceInputSystem, elementToAttachTo?: any): any {\r\n const evt = this._CreateEvent(elementToAttachTo);\r\n this._CheckNonCharacterKeys(evt, deviceInputSystem);\r\n evt.deviceType = DeviceType.Keyboard;\r\n evt.deviceSlot = 0;\r\n evt.inputIndex = inputIndex;\r\n\r\n evt.type = currentState === 1 ? \"keydown\" : \"keyup\";\r\n evt.key = String.fromCharCode(inputIndex);\r\n evt.keyCode = inputIndex;\r\n\r\n return evt;\r\n }\r\n\r\n /**\r\n * Add parameters for non-character keys (Ctrl, Alt, Meta, Shift)\r\n * @param evt Event object to add parameters to\r\n * @param deviceInputSystem DeviceInputSystem to pull values from\r\n */\r\n private static _CheckNonCharacterKeys(evt: any, deviceInputSystem: IDeviceInputSystem): void {\r\n const isKeyboardActive = deviceInputSystem.isDeviceAvailable(DeviceType.Keyboard);\r\n const altKey = isKeyboardActive && deviceInputSystem.pollInput(DeviceType.Keyboard, 0, Constants.INPUT_ALT_KEY) === 1;\r\n const ctrlKey = isKeyboardActive && deviceInputSystem.pollInput(DeviceType.Keyboard, 0, Constants.INPUT_CTRL_KEY) === 1;\r\n const metaKey =\r\n isKeyboardActive &&\r\n (deviceInputSystem.pollInput(DeviceType.Keyboard, 0, Constants.INPUT_META_KEY1) === 1 ||\r\n deviceInputSystem.pollInput(DeviceType.Keyboard, 0, Constants.INPUT_META_KEY2) === 1 ||\r\n deviceInputSystem.pollInput(DeviceType.Keyboard, 0, Constants.INPUT_META_KEY3) === 1);\r\n const shiftKey = isKeyboardActive && deviceInputSystem.pollInput(DeviceType.Keyboard, 0, Constants.INPUT_SHIFT_KEY) === 1;\r\n\r\n evt.altKey = altKey;\r\n evt.ctrlKey = ctrlKey;\r\n evt.metaKey = metaKey;\r\n evt.shiftKey = shiftKey;\r\n }\r\n\r\n /**\r\n * Create base event object\r\n * @param elementToAttachTo Value to use as event target\r\n * @returns\r\n */\r\n private static _CreateEvent(elementToAttachTo: any): any {\r\n const evt: { [k: string]: any } = {};\r\n evt.preventDefault = () => {};\r\n evt.target = elementToAttachTo;\r\n\r\n return evt;\r\n }\r\n}\r\n","import type { INative } from \"../Engines/Native/nativeInterfaces\";\r\nimport type { IUIEvent } from \"../Events/deviceInputEvents\";\r\nimport { DeviceEventFactory } from \"./eventFactory\";\r\nimport { DeviceType } from \"./InputDevices/deviceEnums\";\r\nimport type { IDeviceInputSystem } from \"./inputInterfaces\";\r\n\r\ndeclare const _native: INative;\r\n\r\n/** @internal */\r\nexport class NativeDeviceInputSystem implements IDeviceInputSystem {\r\n private readonly _nativeInput: IDeviceInputSystem;\r\n\r\n public constructor(\r\n onDeviceConnected: (deviceType: DeviceType, deviceSlot: number) => void,\r\n onDeviceDisconnected: (deviceType: DeviceType, deviceSlot: number) => void,\r\n onInputChanged: (deviceType: DeviceType, deviceSlot: number, eventData: IUIEvent) => void\r\n ) {\r\n this._nativeInput = _native.DeviceInputSystem\r\n ? new _native.DeviceInputSystem(onDeviceConnected, onDeviceDisconnected, (deviceType, deviceSlot, inputIndex, currentState) => {\r\n const evt = DeviceEventFactory.CreateDeviceEvent(deviceType, deviceSlot, inputIndex, currentState, this);\r\n\r\n onInputChanged(deviceType, deviceSlot, evt);\r\n })\r\n : this._createDummyNativeInput();\r\n }\r\n\r\n // Public functions\r\n /**\r\n * Checks for current device input value, given an id and input index. Throws exception if requested device not initialized.\r\n * @param deviceType Enum specifying device type\r\n * @param deviceSlot \"Slot\" or index that device is referenced in\r\n * @param inputIndex Id of input to be checked\r\n * @returns Current value of input\r\n */\r\n public pollInput(deviceType: DeviceType, deviceSlot: number, inputIndex: number): number {\r\n return this._nativeInput.pollInput(deviceType, deviceSlot, inputIndex);\r\n }\r\n\r\n /**\r\n * Check for a specific device in the DeviceInputSystem\r\n * @param deviceType Type of device to check for\r\n * @returns bool with status of device's existence\r\n */\r\n public isDeviceAvailable(deviceType: DeviceType): boolean {\r\n //TODO: FIx native side first\r\n return deviceType === DeviceType.Mouse || deviceType === DeviceType.Touch;\r\n }\r\n\r\n /**\r\n * Dispose of all the observables\r\n */\r\n public dispose(): void {\r\n this._nativeInput.dispose();\r\n }\r\n\r\n /**\r\n * For versions of BabylonNative that don't have the NativeInput plugin initialized, create a dummy version\r\n * @returns Object with dummy functions\r\n */\r\n private _createDummyNativeInput() {\r\n const nativeInput = {\r\n pollInput: () => {\r\n return 0;\r\n },\r\n isDeviceAvailable: () => {\r\n return false;\r\n },\r\n dispose: () => {},\r\n };\r\n\r\n return nativeInput;\r\n }\r\n}\r\n","import type { Engine } from \"../Engines/engine\";\r\nimport type { IPointerEvent, IUIEvent } from \"../Events/deviceInputEvents\";\r\nimport { IsNavigatorAvailable } from \"../Misc/domManagement\";\r\nimport type { Observer } from \"../Misc/observable\";\r\nimport { Tools } from \"../Misc/tools\";\r\nimport type { Nullable } from \"../types\";\r\nimport { DeviceEventFactory } from \"./eventFactory\";\r\nimport { DeviceType, PointerInput } from \"./InputDevices/deviceEnums\";\r\nimport type { IDeviceInputSystem } from \"./inputInterfaces\";\r\n\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nconst MAX_KEYCODES = 255;\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nconst MAX_POINTER_INPUTS = Object.keys(PointerInput).length / 2;\r\n\r\n/** @internal */\r\nexport class WebDeviceInputSystem implements IDeviceInputSystem {\r\n // Private Members\r\n private _inputs: Array<{ [deviceSlot: number]: Array }> = [];\r\n private _gamepads: Array;\r\n private _keyboardActive: boolean = false;\r\n private _pointerActive: boolean = false;\r\n private _elementToAttachTo: HTMLElement;\r\n private _metaKeys: Array;\r\n private readonly _engine: Engine;\r\n private readonly _usingSafari: boolean = Tools.IsSafari();\r\n // Found solution for determining if MacOS is being used here:\r\n // https://stackoverflow.com/questions/10527983/best-way-to-detect-mac-os-x-or-windows-computers-with-javascript-or-jquery\r\n private readonly _usingMacOS: boolean = IsNavigatorAvailable() && /(Mac|iPhone|iPod|iPad)/i.test(navigator.platform);\r\n\r\n private _onDeviceConnected: (deviceType: DeviceType, deviceSlot: number) => void;\r\n private _onDeviceDisconnected: (deviceType: DeviceType, deviceSlot: number) => void;\r\n private _onInputChanged: (deviceType: DeviceType, deviceSlot: number, eventData: IUIEvent) => void;\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n private _keyboardDownEvent = (evt: any) => {};\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n private _keyboardUpEvent = (evt: any) => {};\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n private _keyboardBlurEvent = (evt: any) => {};\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n private _pointerMoveEvent = (evt: any) => {};\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n private _pointerDownEvent = (evt: any) => {};\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n private _pointerUpEvent = (evt: any) => {};\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n private _pointerCancelEvent = (evt: any) => {};\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n private _pointerWheelEvent = (evt: any) => {};\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n private _pointerBlurEvent = (evt: any) => {};\r\n private _wheelEventName: string;\r\n private _eventsAttached: boolean = false;\r\n\r\n private _mouseId = -1;\r\n private readonly _isUsingFirefox = IsNavigatorAvailable() && navigator.userAgent && navigator.userAgent.indexOf(\"Firefox\") !== -1;\r\n\r\n // Array to store active Pointer ID values; prevents issues with negative pointerIds\r\n private _activeTouchIds: Array;\r\n private _maxTouchPoints: number = 0;\r\n\r\n private _pointerInputClearObserver: Nullable> = null;\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n private _gamepadConnectedEvent = (evt: any) => {};\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n private _gamepadDisconnectedEvent = (evt: any) => {};\r\n\r\n private _eventPrefix: string;\r\n\r\n constructor(\r\n engine: Engine,\r\n onDeviceConnected: (deviceType: DeviceType, deviceSlot: number) => void,\r\n onDeviceDisconnected: (deviceType: DeviceType, deviceSlot: number) => void,\r\n onInputChanged: (deviceType: DeviceType, deviceSlot: number, eventData: IUIEvent) => void\r\n ) {\r\n this._eventPrefix = Tools.GetPointerPrefix(engine);\r\n this._engine = engine;\r\n\r\n this._onDeviceConnected = onDeviceConnected;\r\n this._onDeviceDisconnected = onDeviceDisconnected;\r\n this._onInputChanged = onInputChanged;\r\n\r\n // If we need a pointerId, set one for future use\r\n this._mouseId = this._isUsingFirefox ? 0 : 1;\r\n\r\n this._enableEvents();\r\n\r\n if (this._usingMacOS) {\r\n this._metaKeys = [];\r\n }\r\n\r\n // Set callback to enable event handler switching when inputElement changes\r\n if (!this._engine._onEngineViewChanged) {\r\n this._engine._onEngineViewChanged = () => {\r\n this._enableEvents();\r\n };\r\n }\r\n }\r\n\r\n // Public functions\r\n /**\r\n * Checks for current device input value, given an id and input index. Throws exception if requested device not initialized.\r\n * @param deviceType Enum specifying device type\r\n * @param deviceSlot \"Slot\" or index that device is referenced in\r\n * @param inputIndex Id of input to be checked\r\n * @returns Current value of input\r\n */\r\n public pollInput(deviceType: DeviceType, deviceSlot: number, inputIndex: number): number {\r\n const device = this._inputs[deviceType][deviceSlot];\r\n\r\n if (!device) {\r\n throw `Unable to find device ${DeviceType[deviceType]}`;\r\n }\r\n\r\n if (deviceType >= DeviceType.DualShock && deviceType <= DeviceType.DualSense) {\r\n this._updateDevice(deviceType, deviceSlot, inputIndex);\r\n }\r\n\r\n const currentValue = device[inputIndex];\r\n if (currentValue === undefined) {\r\n throw `Unable to find input ${inputIndex} for device ${DeviceType[deviceType]} in slot ${deviceSlot}`;\r\n }\r\n\r\n if (inputIndex === PointerInput.Move) {\r\n Tools.Warn(`Unable to provide information for PointerInput.Move. Try using PointerInput.Horizontal or PointerInput.Vertical for move data.`);\r\n }\r\n\r\n return currentValue;\r\n }\r\n\r\n /**\r\n * Check for a specific device in the DeviceInputSystem\r\n * @param deviceType Type of device to check for\r\n * @returns bool with status of device's existence\r\n */\r\n public isDeviceAvailable(deviceType: DeviceType): boolean {\r\n return this._inputs[deviceType] !== undefined;\r\n }\r\n\r\n /**\r\n * Dispose of all the eventlisteners\r\n */\r\n public dispose(): void {\r\n // Callbacks\r\n this._onDeviceConnected = () => {};\r\n this._onDeviceDisconnected = () => {};\r\n this._onInputChanged = () => {};\r\n delete this._engine._onEngineViewChanged;\r\n\r\n if (this._elementToAttachTo) {\r\n this._disableEvents();\r\n }\r\n }\r\n\r\n /**\r\n * Enable listening for user input events\r\n */\r\n private _enableEvents(): void {\r\n const inputElement = this?._engine.getInputElement();\r\n if (inputElement && (!this._eventsAttached || this._elementToAttachTo !== inputElement)) {\r\n // Remove events before adding to avoid double events or simultaneous events on multiple canvases\r\n this._disableEvents();\r\n\r\n // If the inputs array has already been created, zero it out to before setting up events\r\n if (this._inputs) {\r\n for (const inputs of this._inputs) {\r\n if (inputs) {\r\n for (const deviceSlotKey in inputs) {\r\n const deviceSlot = +deviceSlotKey;\r\n const device = inputs[deviceSlot];\r\n if (device) {\r\n for (let inputIndex = 0; inputIndex < device.length; inputIndex++) {\r\n device[inputIndex] = 0;\r\n }\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n this._elementToAttachTo = inputElement;\r\n // Set tab index for the inputElement to the engine's canvasTabIndex, if and only if the element's tab index is -1\r\n this._elementToAttachTo.tabIndex = this._elementToAttachTo.tabIndex !== -1 ? this._elementToAttachTo.tabIndex : this._engine.canvasTabIndex;\r\n this._handleKeyActions();\r\n this._handlePointerActions();\r\n this._handleGamepadActions();\r\n this._eventsAttached = true;\r\n\r\n // Check for devices that are already connected but aren't registered. Currently, only checks for gamepads and mouse\r\n this._checkForConnectedDevices();\r\n }\r\n }\r\n\r\n /**\r\n * Disable listening for user input events\r\n */\r\n private _disableEvents(): void {\r\n if (this._elementToAttachTo) {\r\n // Blur Events\r\n this._elementToAttachTo.removeEventListener(\"blur\", this._keyboardBlurEvent);\r\n this._elementToAttachTo.removeEventListener(\"blur\", this._pointerBlurEvent);\r\n\r\n // Keyboard Events\r\n this._elementToAttachTo.removeEventListener(\"keydown\", this._keyboardDownEvent);\r\n this._elementToAttachTo.removeEventListener(\"keyup\", this._keyboardUpEvent);\r\n\r\n // Pointer Events\r\n this._elementToAttachTo.removeEventListener(this._eventPrefix + \"move\", this._pointerMoveEvent);\r\n this._elementToAttachTo.removeEventListener(this._eventPrefix + \"down\", this._pointerDownEvent);\r\n this._elementToAttachTo.removeEventListener(this._eventPrefix + \"up\", this._pointerUpEvent);\r\n this._elementToAttachTo.removeEventListener(this._eventPrefix + \"cancel\", this._pointerCancelEvent);\r\n this._elementToAttachTo.removeEventListener(this._wheelEventName, this._pointerWheelEvent);\r\n\r\n // Gamepad Events\r\n window.removeEventListener(\"gamepadconnected\", this._gamepadConnectedEvent);\r\n window.removeEventListener(\"gamepaddisconnected\", this._gamepadDisconnectedEvent);\r\n }\r\n\r\n if (this._pointerInputClearObserver) {\r\n this._engine.onEndFrameObservable.remove(this._pointerInputClearObserver);\r\n }\r\n\r\n this._eventsAttached = false;\r\n }\r\n\r\n /**\r\n * Checks for existing connections to devices and register them, if necessary\r\n * Currently handles gamepads and mouse\r\n */\r\n private _checkForConnectedDevices(): void {\r\n if (navigator.getGamepads) {\r\n const gamepads = navigator.getGamepads();\r\n\r\n for (const gamepad of gamepads) {\r\n if (gamepad) {\r\n this._addGamePad(gamepad);\r\n }\r\n }\r\n }\r\n\r\n // If the device in use has mouse capabilities, pre-register mouse\r\n if (typeof matchMedia === \"function\" && matchMedia(\"(pointer:fine)\").matches) {\r\n // This will provide a dummy value for the cursor position and is expected to be overridden when the first mouse event happens.\r\n // There isn't any good way to get the current position outside of a pointer event so that's why this was done.\r\n this._addPointerDevice(DeviceType.Mouse, 0, 0, 0);\r\n }\r\n }\r\n\r\n // Private functions\r\n /**\r\n * Add a gamepad to the DeviceInputSystem\r\n * @param gamepad A single DOM Gamepad object\r\n */\r\n private _addGamePad(gamepad: any): void {\r\n const deviceType = this._getGamepadDeviceType(gamepad.id);\r\n const deviceSlot = gamepad.index;\r\n\r\n this._gamepads = this._gamepads || new Array(gamepad.index + 1);\r\n this._registerDevice(deviceType, deviceSlot, gamepad.buttons.length + gamepad.axes.length);\r\n\r\n this._gamepads[deviceSlot] = deviceType;\r\n }\r\n\r\n /**\r\n * Add pointer device to DeviceInputSystem\r\n * @param deviceType Type of Pointer to add\r\n * @param deviceSlot Pointer ID (0 for mouse, pointerId for Touch)\r\n * @param currentX Current X at point of adding\r\n * @param currentY Current Y at point of adding\r\n */\r\n private _addPointerDevice(deviceType: DeviceType, deviceSlot: number, currentX: number, currentY: number): void {\r\n if (!this._pointerActive) {\r\n this._pointerActive = true;\r\n }\r\n this._registerDevice(deviceType, deviceSlot, MAX_POINTER_INPUTS);\r\n const pointer = this._inputs[deviceType][deviceSlot]; /* initialize our pointer position immediately after registration */\r\n pointer[0] = currentX;\r\n pointer[1] = currentY;\r\n }\r\n\r\n /**\r\n * Add device and inputs to device array\r\n * @param deviceType Enum specifying device type\r\n * @param deviceSlot \"Slot\" or index that device is referenced in\r\n * @param numberOfInputs Number of input entries to create for given device\r\n */\r\n private _registerDevice(deviceType: DeviceType, deviceSlot: number, numberOfInputs: number): void {\r\n if (deviceSlot === undefined) {\r\n throw `Unable to register device ${DeviceType[deviceType]} to undefined slot.`;\r\n }\r\n\r\n if (!this._inputs[deviceType]) {\r\n this._inputs[deviceType] = {};\r\n }\r\n\r\n if (!this._inputs[deviceType][deviceSlot]) {\r\n const device = new Array(numberOfInputs);\r\n\r\n device.fill(0);\r\n\r\n this._inputs[deviceType][deviceSlot] = device;\r\n this._onDeviceConnected(deviceType, deviceSlot);\r\n }\r\n }\r\n\r\n /**\r\n * Given a specific device name, remove that device from the device map\r\n * @param deviceType Enum specifying device type\r\n * @param deviceSlot \"Slot\" or index that device is referenced in\r\n */\r\n private _unregisterDevice(deviceType: DeviceType, deviceSlot: number): void {\r\n if (this._inputs[deviceType][deviceSlot]) {\r\n delete this._inputs[deviceType][deviceSlot];\r\n this._onDeviceDisconnected(deviceType, deviceSlot);\r\n }\r\n }\r\n\r\n /**\r\n * Handle all actions that come from keyboard interaction\r\n */\r\n private _handleKeyActions(): void {\r\n this._keyboardDownEvent = (evt) => {\r\n if (!this._keyboardActive) {\r\n this._keyboardActive = true;\r\n this._registerDevice(DeviceType.Keyboard, 0, MAX_KEYCODES);\r\n }\r\n\r\n const kbKey = this._inputs[DeviceType.Keyboard][0];\r\n if (kbKey) {\r\n kbKey[evt.keyCode] = 1;\r\n\r\n const deviceEvent = evt as IUIEvent;\r\n deviceEvent.inputIndex = evt.keyCode;\r\n\r\n if (this._usingMacOS && evt.metaKey && evt.key !== \"Meta\") {\r\n if (!this._metaKeys.includes(evt.keyCode)) {\r\n this._metaKeys.push(evt.keyCode);\r\n }\r\n }\r\n\r\n this._onInputChanged(DeviceType.Keyboard, 0, deviceEvent);\r\n }\r\n };\r\n\r\n this._keyboardUpEvent = (evt) => {\r\n if (!this._keyboardActive) {\r\n this._keyboardActive = true;\r\n this._registerDevice(DeviceType.Keyboard, 0, MAX_KEYCODES);\r\n }\r\n\r\n const kbKey = this._inputs[DeviceType.Keyboard][0];\r\n if (kbKey) {\r\n kbKey[evt.keyCode] = 0;\r\n\r\n const deviceEvent = evt as IUIEvent;\r\n deviceEvent.inputIndex = evt.keyCode;\r\n\r\n if (this._usingMacOS && evt.key === \"Meta\" && this._metaKeys.length > 0) {\r\n for (const keyCode of this._metaKeys) {\r\n const deviceEvent: IUIEvent = DeviceEventFactory.CreateDeviceEvent(DeviceType.Keyboard, 0, keyCode, 0, this, this._elementToAttachTo);\r\n kbKey[keyCode] = 0;\r\n this._onInputChanged(DeviceType.Keyboard, 0, deviceEvent);\r\n }\r\n this._metaKeys.splice(0, this._metaKeys.length);\r\n }\r\n\r\n this._onInputChanged(DeviceType.Keyboard, 0, deviceEvent);\r\n }\r\n };\r\n\r\n this._keyboardBlurEvent = () => {\r\n if (this._keyboardActive) {\r\n const kbKey = this._inputs[DeviceType.Keyboard][0];\r\n\r\n for (let i = 0; i < kbKey.length; i++) {\r\n if (kbKey[i] !== 0) {\r\n kbKey[i] = 0;\r\n\r\n const deviceEvent: IUIEvent = DeviceEventFactory.CreateDeviceEvent(DeviceType.Keyboard, 0, i, 0, this, this._elementToAttachTo);\r\n\r\n this._onInputChanged(DeviceType.Keyboard, 0, deviceEvent);\r\n }\r\n }\r\n if (this._usingMacOS) {\r\n this._metaKeys.splice(0, this._metaKeys.length);\r\n }\r\n }\r\n };\r\n\r\n this._elementToAttachTo.addEventListener(\"keydown\", this._keyboardDownEvent);\r\n this._elementToAttachTo.addEventListener(\"keyup\", this._keyboardUpEvent);\r\n this._elementToAttachTo.addEventListener(\"blur\", this._keyboardBlurEvent);\r\n }\r\n\r\n /**\r\n * Handle all actions that come from pointer interaction\r\n */\r\n private _handlePointerActions(): void {\r\n // If maxTouchPoints is defined, use that value. Otherwise, allow for a minimum for supported gestures like pinch\r\n this._maxTouchPoints = (IsNavigatorAvailable() && navigator.maxTouchPoints) || 2;\r\n if (!this._activeTouchIds) {\r\n this._activeTouchIds = new Array(this._maxTouchPoints);\r\n }\r\n\r\n for (let i = 0; i < this._maxTouchPoints; i++) {\r\n this._activeTouchIds[i] = -1;\r\n }\r\n\r\n this._pointerMoveEvent = (evt) => {\r\n const deviceType = this._getPointerType(evt);\r\n const deviceSlot = deviceType === DeviceType.Mouse ? 0 : this._activeTouchIds.indexOf(evt.pointerId);\r\n\r\n if (!this._inputs[deviceType]) {\r\n this._inputs[deviceType] = {};\r\n }\r\n\r\n if (!this._inputs[deviceType][deviceSlot]) {\r\n this._addPointerDevice(deviceType, deviceSlot, evt.clientX, evt.clientY);\r\n }\r\n\r\n const pointer = this._inputs[deviceType][deviceSlot];\r\n if (pointer) {\r\n const deviceEvent = evt as IPointerEvent;\r\n deviceEvent.inputIndex = PointerInput.Move;\r\n\r\n pointer[PointerInput.Horizontal] = evt.clientX;\r\n pointer[PointerInput.Vertical] = evt.clientY;\r\n\r\n if (evt.pointerId === undefined) {\r\n evt.pointerId = this._mouseId;\r\n }\r\n\r\n this._onInputChanged(deviceType, deviceSlot, deviceEvent);\r\n\r\n // Lets Propagate the event for move with same position.\r\n if (!this._usingSafari && evt.button !== -1) {\r\n deviceEvent.inputIndex = evt.button + 2;\r\n pointer[evt.button + 2] = pointer[evt.button + 2] ? 0 : 1; // Reverse state of button if evt.button has value\r\n this._onInputChanged(deviceType, deviceSlot, deviceEvent);\r\n }\r\n }\r\n };\r\n\r\n this._pointerDownEvent = (evt) => {\r\n const deviceType = this._getPointerType(evt);\r\n let deviceSlot = deviceType === DeviceType.Mouse ? 0 : evt.pointerId;\r\n\r\n if (deviceType === DeviceType.Touch) {\r\n const idx = this._activeTouchIds.indexOf(-1);\r\n\r\n if (idx >= 0) {\r\n deviceSlot = idx;\r\n this._activeTouchIds[idx] = evt.pointerId;\r\n } else {\r\n // We can't find an open slot to store new pointer so just return (can only support max number of touches)\r\n Tools.Warn(`Max number of touches exceeded. Ignoring touches in excess of ${this._maxTouchPoints}`);\r\n return;\r\n }\r\n }\r\n\r\n if (!this._inputs[deviceType]) {\r\n this._inputs[deviceType] = {};\r\n }\r\n\r\n if (!this._inputs[deviceType][deviceSlot]) {\r\n this._addPointerDevice(deviceType, deviceSlot, evt.clientX, evt.clientY);\r\n } else if (deviceType === DeviceType.Touch) {\r\n this._onDeviceConnected(deviceType, deviceSlot);\r\n }\r\n\r\n const pointer = this._inputs[deviceType][deviceSlot];\r\n if (pointer) {\r\n const previousHorizontal = pointer[PointerInput.Horizontal];\r\n const previousVertical = pointer[PointerInput.Vertical];\r\n\r\n if (deviceType === DeviceType.Mouse) {\r\n // Mouse; Set pointerId if undefined\r\n if (evt.pointerId === undefined) {\r\n evt.pointerId = this._mouseId;\r\n }\r\n\r\n if (!document.pointerLockElement) {\r\n try {\r\n this._elementToAttachTo.setPointerCapture(this._mouseId);\r\n } catch (e) {\r\n // DO NOTHING\r\n }\r\n }\r\n } else {\r\n // Touch; Since touches are dynamically assigned, only set capture if we have an id\r\n if (evt.pointerId && !document.pointerLockElement) {\r\n try {\r\n this._elementToAttachTo.setPointerCapture(evt.pointerId);\r\n } catch (e) {\r\n // DO NOTHING\r\n }\r\n }\r\n }\r\n\r\n pointer[PointerInput.Horizontal] = evt.clientX;\r\n pointer[PointerInput.Vertical] = evt.clientY;\r\n pointer[evt.button + 2] = 1;\r\n\r\n const deviceEvent = evt as IUIEvent;\r\n\r\n // NOTE: The +2 used here to is because PointerInput has the same value progression for its mouse buttons as PointerEvent.button\r\n // However, we have our X and Y values front-loaded to group together the touch inputs but not break this progression\r\n // EG. ([X, Y, Left-click], Middle-click, etc...)\r\n deviceEvent.inputIndex = evt.button + 2;\r\n\r\n this._onInputChanged(deviceType, deviceSlot, deviceEvent);\r\n\r\n if (previousHorizontal !== evt.clientX || previousVertical !== evt.clientY) {\r\n deviceEvent.inputIndex = PointerInput.Move;\r\n this._onInputChanged(deviceType, deviceSlot, deviceEvent);\r\n }\r\n }\r\n };\r\n\r\n this._pointerUpEvent = (evt) => {\r\n const deviceType = this._getPointerType(evt);\r\n const deviceSlot = deviceType === DeviceType.Mouse ? 0 : this._activeTouchIds.indexOf(evt.pointerId);\r\n\r\n if (deviceType === DeviceType.Touch) {\r\n if (deviceSlot === -1) {\r\n return;\r\n } else {\r\n this._activeTouchIds[deviceSlot] = -1;\r\n }\r\n }\r\n\r\n const pointer = this._inputs[deviceType]?.[deviceSlot];\r\n if (pointer && pointer[evt.button + 2] !== 0) {\r\n const previousHorizontal = pointer[PointerInput.Horizontal];\r\n const previousVertical = pointer[PointerInput.Vertical];\r\n\r\n pointer[PointerInput.Horizontal] = evt.clientX;\r\n pointer[PointerInput.Vertical] = evt.clientY;\r\n pointer[evt.button + 2] = 0;\r\n\r\n const deviceEvent = evt as IUIEvent;\r\n\r\n if (evt.pointerId === undefined) {\r\n evt.pointerId = this._mouseId;\r\n }\r\n\r\n if (previousHorizontal !== evt.clientX || previousVertical !== evt.clientY) {\r\n deviceEvent.inputIndex = PointerInput.Move;\r\n this._onInputChanged(deviceType, deviceSlot, deviceEvent);\r\n }\r\n\r\n // NOTE: The +2 used here to is because PointerInput has the same value progression for its mouse buttons as PointerEvent.button\r\n // However, we have our X and Y values front-loaded to group together the touch inputs but not break this progression\r\n // EG. ([X, Y, Left-click], Middle-click, etc...)\r\n deviceEvent.inputIndex = evt.button + 2;\r\n\r\n if (deviceType === DeviceType.Mouse && this._mouseId >= 0 && this._elementToAttachTo.hasPointerCapture?.(this._mouseId)) {\r\n this._elementToAttachTo.releasePointerCapture(this._mouseId);\r\n } else if (evt.pointerId && this._elementToAttachTo.hasPointerCapture?.(evt.pointerId)) {\r\n this._elementToAttachTo.releasePointerCapture(evt.pointerId);\r\n }\r\n\r\n this._onInputChanged(deviceType, deviceSlot, deviceEvent);\r\n\r\n if (deviceType === DeviceType.Touch) {\r\n this._onDeviceDisconnected(deviceType, deviceSlot);\r\n }\r\n }\r\n };\r\n\r\n this._pointerCancelEvent = (evt) => {\r\n if (evt.pointerType === \"mouse\") {\r\n const pointer = this._inputs[DeviceType.Mouse][0];\r\n\r\n if (this._mouseId >= 0 && this._elementToAttachTo.hasPointerCapture?.(this._mouseId)) {\r\n this._elementToAttachTo.releasePointerCapture(this._mouseId);\r\n }\r\n\r\n for (let inputIndex = PointerInput.LeftClick; inputIndex <= PointerInput.BrowserForward; inputIndex++) {\r\n if (pointer[inputIndex] === 1) {\r\n pointer[inputIndex] = 0;\r\n\r\n const deviceEvent: IUIEvent = DeviceEventFactory.CreateDeviceEvent(DeviceType.Mouse, 0, inputIndex, 0, this, this._elementToAttachTo);\r\n\r\n this._onInputChanged(DeviceType.Mouse, 0, deviceEvent);\r\n }\r\n }\r\n } else {\r\n const deviceSlot = this._activeTouchIds.indexOf(evt.pointerId);\r\n\r\n if (this._elementToAttachTo.hasPointerCapture?.(evt.pointerId)) {\r\n this._elementToAttachTo.releasePointerCapture(evt.pointerId);\r\n }\r\n\r\n this._inputs[DeviceType.Touch][deviceSlot][PointerInput.LeftClick] = 0;\r\n\r\n const deviceEvent: IUIEvent = DeviceEventFactory.CreateDeviceEvent(\r\n DeviceType.Touch,\r\n deviceSlot,\r\n PointerInput.LeftClick,\r\n 0,\r\n this,\r\n this._elementToAttachTo,\r\n evt.pointerId\r\n );\r\n\r\n this._onInputChanged(DeviceType.Touch, deviceSlot, deviceEvent);\r\n\r\n this._activeTouchIds[deviceSlot] = -1;\r\n this._onDeviceDisconnected(DeviceType.Touch, deviceSlot);\r\n }\r\n };\r\n\r\n // Set Wheel Event Name, code originally from scene.inputManager\r\n this._wheelEventName =\r\n \"onwheel\" in document.createElement(\"div\")\r\n ? \"wheel\" // Modern browsers support \"wheel\"\r\n : (document).onmousewheel !== undefined\r\n ? \"mousewheel\" // Webkit and IE support at least \"mousewheel\"\r\n : \"DOMMouseScroll\"; // let's assume that remaining browsers are older Firefox\r\n\r\n // Code originally in scene.inputManager.ts\r\n // Chrome reports warning in console if wheel listener doesn't set an explicit passive option.\r\n // IE11 only supports captureEvent:boolean, not options:object, and it defaults to false.\r\n // Feature detection technique copied from: https://github.com/github/eventlistener-polyfill (MIT license)\r\n let passiveSupported = false;\r\n const noop = function () {};\r\n\r\n try {\r\n const options = Object.defineProperty({}, \"passive\", {\r\n get: function () {\r\n passiveSupported = true;\r\n },\r\n });\r\n\r\n this._elementToAttachTo.addEventListener(\"test\", noop, options);\r\n this._elementToAttachTo.removeEventListener(\"test\", noop, options);\r\n } catch (e) {\r\n /* */\r\n }\r\n\r\n this._pointerBlurEvent = () => {\r\n // Handle mouse buttons\r\n if (this.isDeviceAvailable(DeviceType.Mouse)) {\r\n const pointer = this._inputs[DeviceType.Mouse][0];\r\n\r\n if (this._mouseId >= 0 && this._elementToAttachTo.hasPointerCapture?.(this._mouseId)) {\r\n this._elementToAttachTo.releasePointerCapture(this._mouseId);\r\n }\r\n\r\n for (let inputIndex = PointerInput.LeftClick; inputIndex <= PointerInput.BrowserForward; inputIndex++) {\r\n if (pointer[inputIndex] === 1) {\r\n pointer[inputIndex] = 0;\r\n\r\n const deviceEvent: IUIEvent = DeviceEventFactory.CreateDeviceEvent(DeviceType.Mouse, 0, inputIndex, 0, this, this._elementToAttachTo);\r\n\r\n this._onInputChanged(DeviceType.Mouse, 0, deviceEvent);\r\n }\r\n }\r\n }\r\n\r\n // Handle Active Touches\r\n if (this.isDeviceAvailable(DeviceType.Touch)) {\r\n const pointer = this._inputs[DeviceType.Touch];\r\n\r\n for (let deviceSlot = 0; deviceSlot < this._activeTouchIds.length; deviceSlot++) {\r\n const pointerId = this._activeTouchIds[deviceSlot];\r\n\r\n if (this._elementToAttachTo.hasPointerCapture?.(pointerId)) {\r\n this._elementToAttachTo.releasePointerCapture(pointerId);\r\n }\r\n\r\n if (pointerId !== -1 && pointer[deviceSlot]?.[PointerInput.LeftClick] === 1) {\r\n pointer[deviceSlot][PointerInput.LeftClick] = 0;\r\n\r\n const deviceEvent: IUIEvent = DeviceEventFactory.CreateDeviceEvent(\r\n DeviceType.Touch,\r\n deviceSlot,\r\n PointerInput.LeftClick,\r\n 0,\r\n this,\r\n this._elementToAttachTo,\r\n pointerId\r\n );\r\n\r\n this._onInputChanged(DeviceType.Touch, deviceSlot, deviceEvent);\r\n\r\n this._activeTouchIds[deviceSlot] = -1;\r\n this._onDeviceDisconnected(DeviceType.Touch, deviceSlot);\r\n }\r\n }\r\n }\r\n };\r\n\r\n this._pointerWheelEvent = (evt) => {\r\n const deviceType = DeviceType.Mouse;\r\n const deviceSlot = 0;\r\n\r\n if (!this._inputs[deviceType]) {\r\n this._inputs[deviceType] = [];\r\n }\r\n\r\n if (!this._inputs[deviceType][deviceSlot]) {\r\n this._pointerActive = true;\r\n this._registerDevice(deviceType, deviceSlot, MAX_POINTER_INPUTS);\r\n }\r\n\r\n const pointer = this._inputs[deviceType][deviceSlot];\r\n if (pointer) {\r\n pointer[PointerInput.MouseWheelX] = evt.deltaX || 0;\r\n pointer[PointerInput.MouseWheelY] = evt.deltaY || evt.wheelDelta || 0;\r\n pointer[PointerInput.MouseWheelZ] = evt.deltaZ || 0;\r\n\r\n const deviceEvent = evt as IUIEvent;\r\n // By default, there is no pointerId for mouse wheel events so we'll add one here\r\n // This logic was originally in the InputManager but was added here to make the\r\n // InputManager more platform-agnostic\r\n if (evt.pointerId === undefined) {\r\n evt.pointerId = this._mouseId;\r\n }\r\n\r\n if (pointer[PointerInput.MouseWheelX] !== 0) {\r\n deviceEvent.inputIndex = PointerInput.MouseWheelX;\r\n this._onInputChanged(deviceType, deviceSlot, deviceEvent);\r\n }\r\n if (pointer[PointerInput.MouseWheelY] !== 0) {\r\n deviceEvent.inputIndex = PointerInput.MouseWheelY;\r\n this._onInputChanged(deviceType, deviceSlot, deviceEvent);\r\n }\r\n if (pointer[PointerInput.MouseWheelZ] !== 0) {\r\n deviceEvent.inputIndex = PointerInput.MouseWheelZ;\r\n this._onInputChanged(deviceType, deviceSlot, deviceEvent);\r\n }\r\n }\r\n };\r\n\r\n this._elementToAttachTo.addEventListener(this._eventPrefix + \"move\", this._pointerMoveEvent);\r\n this._elementToAttachTo.addEventListener(this._eventPrefix + \"down\", this._pointerDownEvent);\r\n this._elementToAttachTo.addEventListener(this._eventPrefix + \"up\", this._pointerUpEvent);\r\n this._elementToAttachTo.addEventListener(this._eventPrefix + \"cancel\", this._pointerCancelEvent);\r\n this._elementToAttachTo.addEventListener(\"blur\", this._pointerBlurEvent);\r\n this._elementToAttachTo.addEventListener(this._wheelEventName, this._pointerWheelEvent, passiveSupported ? { passive: false } : false);\r\n\r\n // Since there's no up or down event for mouse wheel or delta x/y, clear mouse values at end of frame\r\n this._pointerInputClearObserver = this._engine.onEndFrameObservable.add(() => {\r\n if (this.isDeviceAvailable(DeviceType.Mouse)) {\r\n const pointer = this._inputs[DeviceType.Mouse][0];\r\n pointer[PointerInput.MouseWheelX] = 0;\r\n pointer[PointerInput.MouseWheelY] = 0;\r\n pointer[PointerInput.MouseWheelZ] = 0;\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Handle all actions that come from gamepad interaction\r\n */\r\n private _handleGamepadActions(): void {\r\n this._gamepadConnectedEvent = (evt: any) => {\r\n this._addGamePad(evt.gamepad);\r\n };\r\n\r\n this._gamepadDisconnectedEvent = (evt: any) => {\r\n if (this._gamepads) {\r\n const deviceType = this._getGamepadDeviceType(evt.gamepad.id);\r\n const deviceSlot = evt.gamepad.index;\r\n\r\n this._unregisterDevice(deviceType, deviceSlot);\r\n delete this._gamepads[deviceSlot];\r\n }\r\n };\r\n\r\n window.addEventListener(\"gamepadconnected\", this._gamepadConnectedEvent);\r\n window.addEventListener(\"gamepaddisconnected\", this._gamepadDisconnectedEvent);\r\n }\r\n\r\n /**\r\n * Update all non-event based devices with each frame\r\n * @param deviceType Enum specifying device type\r\n * @param deviceSlot \"Slot\" or index that device is referenced in\r\n * @param inputIndex Id of input to be checked\r\n */\r\n private _updateDevice(deviceType: DeviceType, deviceSlot: number, inputIndex: number): void {\r\n // Gamepads\r\n const gp = navigator.getGamepads()[deviceSlot];\r\n\r\n if (gp && deviceType === this._gamepads[deviceSlot]) {\r\n const device = this._inputs[deviceType][deviceSlot];\r\n\r\n if (inputIndex >= gp.buttons.length) {\r\n device[inputIndex] = gp.axes[inputIndex - gp.buttons.length].valueOf();\r\n } else {\r\n device[inputIndex] = gp.buttons[inputIndex].value;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Gets DeviceType from the device name\r\n * @param deviceName Name of Device from DeviceInputSystem\r\n * @returns DeviceType enum value\r\n */\r\n private _getGamepadDeviceType(deviceName: string): DeviceType {\r\n if (deviceName.indexOf(\"054c\") !== -1) {\r\n // DualShock 4 Gamepad\r\n return deviceName.indexOf(\"0ce6\") !== -1 ? DeviceType.DualSense : DeviceType.DualShock;\r\n } else if (deviceName.indexOf(\"Xbox One\") !== -1 || deviceName.search(\"Xbox 360\") !== -1 || deviceName.search(\"xinput\") !== -1) {\r\n // Xbox Gamepad\r\n return DeviceType.Xbox;\r\n } else if (deviceName.indexOf(\"057e\") !== -1) {\r\n // Switch Gamepad\r\n return DeviceType.Switch;\r\n }\r\n\r\n return DeviceType.Generic;\r\n }\r\n\r\n /**\r\n * Get DeviceType from a given pointer/mouse/touch event.\r\n * @param evt PointerEvent to evaluate\r\n * @returns DeviceType interpreted from event\r\n */\r\n private _getPointerType(evt: any): DeviceType {\r\n let deviceType = DeviceType.Mouse;\r\n\r\n if (evt.pointerType === \"touch\" || evt.pointerType === \"pen\" || evt.touches) {\r\n deviceType = DeviceType.Touch;\r\n }\r\n\r\n return deviceType;\r\n }\r\n}\r\n","import type { DeviceType } from \"./deviceEnums\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport type { DeviceInput } from \"./deviceTypes\";\r\nimport type { IDeviceInputSystem } from \"../inputInterfaces\";\r\nimport type { IKeyboardEvent, IPointerEvent, IWheelEvent } from \"../../Events/deviceInputEvents\";\r\n\r\n/**\r\n * Subset of DeviceInput that only handles pointers and keyboard\r\n */\r\nexport type DeviceSourceEvent = T extends DeviceType.Keyboard\r\n ? IKeyboardEvent\r\n : T extends DeviceType.Mouse\r\n ? IWheelEvent | IPointerEvent\r\n : T extends DeviceType.Touch\r\n ? IPointerEvent\r\n : never;\r\n\r\n/**\r\n * Class that handles all input for a specific device\r\n */\r\nexport class DeviceSource {\r\n // Public Members\r\n /**\r\n * Observable to handle device input changes per device\r\n */\r\n public readonly onInputChangedObservable = new Observable>();\r\n\r\n // Private Members\r\n private readonly _deviceInputSystem: IDeviceInputSystem;\r\n\r\n /**\r\n * Default Constructor\r\n * @param deviceInputSystem - Reference to DeviceInputSystem\r\n * @param deviceType - Type of device\r\n * @param deviceSlot - \"Slot\" or index that device is referenced in\r\n */\r\n constructor(\r\n deviceInputSystem: IDeviceInputSystem,\r\n /** Type of device */\r\n public readonly deviceType: T,\r\n /** \"Slot\" or index that device is referenced in */\r\n public readonly deviceSlot: number = 0\r\n ) {\r\n this._deviceInputSystem = deviceInputSystem;\r\n }\r\n\r\n /**\r\n * Get input for specific input\r\n * @param inputIndex - index of specific input on device\r\n * @returns Input value from DeviceInputSystem\r\n */\r\n public getInput(inputIndex: DeviceInput): number {\r\n return this._deviceInputSystem.pollInput(this.deviceType, this.deviceSlot, inputIndex);\r\n }\r\n}\r\n","import type { IDisposable } from \"../scene\";\r\nimport { DeviceType } from \"./InputDevices/deviceEnums\";\r\nimport type { Observable } from \"../Misc/observable\";\r\nimport type { IDeviceInputSystem } from \"./inputInterfaces\";\r\nimport { NativeDeviceInputSystem } from \"./nativeDeviceInputSystem\";\r\nimport { WebDeviceInputSystem } from \"./webDeviceInputSystem\";\r\nimport { DeviceSource } from \"./InputDevices/deviceSource\";\r\nimport type { INative } from \"../Engines/Native/nativeInterfaces\";\r\nimport type { Engine } from \"../Engines/engine\";\r\nimport type { IUIEvent } from \"../Events/deviceInputEvents\";\r\n\r\ntype Distribute = T extends DeviceType ? DeviceSource : never;\r\n\r\nexport type DeviceSourceType = Distribute;\r\n\r\ndeclare const _native: INative;\r\n\r\ndeclare module \"../Engines/engine\" {\r\n interface Engine {\r\n /** @internal */\r\n _deviceSourceManager?: InternalDeviceSourceManager;\r\n }\r\n}\r\n\r\n/** @internal */\r\nexport interface IObservableManager {\r\n onDeviceConnectedObservable: Observable;\r\n onDeviceDisconnectedObservable: Observable;\r\n\r\n // Functions\r\n _onInputChanged(deviceType: DeviceType, deviceSlot: number, eventData: IUIEvent): void;\r\n _addDevice(deviceSource: DeviceSource): void;\r\n _removeDevice(deviceType: DeviceType, deviceSlot: number): void;\r\n}\r\n\r\n/** @internal */\r\nexport class InternalDeviceSourceManager implements IDisposable {\r\n // Public Members\r\n public readonly _deviceInputSystem: IDeviceInputSystem;\r\n\r\n // Private Members\r\n // This is a master list of all device type/slot combos\r\n private readonly _devices: Array>;\r\n\r\n private readonly _registeredManagers = new Array();\r\n\r\n public _refCount = 0;\r\n\r\n public constructor(engine: Engine) {\r\n const numberOfDeviceTypes = Object.keys(DeviceType).length / 2;\r\n this._devices = new Array>(numberOfDeviceTypes);\r\n\r\n const onDeviceConnected = (deviceType: DeviceType, deviceSlot: number) => {\r\n if (!this._devices[deviceType]) {\r\n this._devices[deviceType] = new Array();\r\n }\r\n\r\n if (!this._devices[deviceType][deviceSlot]) {\r\n this._devices[deviceType][deviceSlot] = deviceSlot;\r\n }\r\n for (const manager of this._registeredManagers) {\r\n const deviceSource = new DeviceSource(this._deviceInputSystem, deviceType, deviceSlot);\r\n manager._addDevice(deviceSource);\r\n }\r\n };\r\n\r\n const onDeviceDisconnected = (deviceType: DeviceType, deviceSlot: number) => {\r\n if (this._devices[deviceType]?.[deviceSlot]) {\r\n delete this._devices[deviceType][deviceSlot];\r\n }\r\n for (const manager of this._registeredManagers) {\r\n manager._removeDevice(deviceType, deviceSlot);\r\n }\r\n };\r\n\r\n const onInputChanged = (deviceType: DeviceType, deviceSlot: number, eventData: IUIEvent) => {\r\n if (eventData) {\r\n for (const manager of this._registeredManagers) {\r\n manager._onInputChanged(deviceType, deviceSlot, eventData);\r\n }\r\n }\r\n };\r\n\r\n if (typeof _native !== \"undefined\") {\r\n this._deviceInputSystem = new NativeDeviceInputSystem(onDeviceConnected, onDeviceDisconnected, onInputChanged);\r\n } else {\r\n this._deviceInputSystem = new WebDeviceInputSystem(engine, onDeviceConnected, onDeviceDisconnected, onInputChanged);\r\n }\r\n }\r\n\r\n // Public Functions\r\n public readonly registerManager = (manager: IObservableManager): void => {\r\n for (let deviceType = 0; deviceType < this._devices.length; deviceType++) {\r\n const device = this._devices[deviceType];\r\n for (const deviceSlotKey in device) {\r\n const deviceSlot = +deviceSlotKey;\r\n manager._addDevice(new DeviceSource(this._deviceInputSystem, deviceType, deviceSlot));\r\n }\r\n }\r\n this._registeredManagers.push(manager);\r\n };\r\n\r\n public readonly unregisterManager = (manager: IObservableManager): void => {\r\n const idx = this._registeredManagers.indexOf(manager);\r\n\r\n if (idx > -1) {\r\n this._registeredManagers.splice(idx, 1);\r\n }\r\n };\r\n\r\n public dispose(): void {\r\n this._deviceInputSystem.dispose();\r\n }\r\n}\r\n","import type { Engine } from \"../../Engines/engine\";\r\nimport { DeviceType } from \"./deviceEnums\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { Observer } from \"../../Misc/observable\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport type { DeviceSource } from \"./deviceSource\";\r\nimport type { IObservableManager, DeviceSourceType } from \"../internalDeviceSourceManager\";\r\nimport { InternalDeviceSourceManager } from \"../internalDeviceSourceManager\";\r\nimport type { IDisposable } from \"../../scene\";\r\nimport type { ThinEngine } from \"../../Engines/thinEngine\";\r\nimport type { IKeyboardEvent, IPointerEvent, IUIEvent, IWheelEvent } from \"../../Events/deviceInputEvents\";\r\n\r\n/**\r\n * Class to keep track of devices\r\n */\r\nexport class DeviceSourceManager implements IDisposable, IObservableManager {\r\n // Public Members\r\n /**\r\n * Observable to be triggered when after a device is connected, any new observers added will be triggered against already connected devices\r\n */\r\n public readonly onDeviceConnectedObservable: Observable;\r\n\r\n /**\r\n * Observable to be triggered when after a device is disconnected\r\n */\r\n public readonly onDeviceDisconnectedObservable: Observable;\r\n\r\n // Private Members\r\n private _engine: Engine;\r\n private _onDisposeObserver: Nullable>;\r\n private readonly _devices: Array>>;\r\n private readonly _firstDevice: Array;\r\n\r\n // Public Functions\r\n /**\r\n * Gets a DeviceSource, given a type and slot\r\n * @param deviceType - Type of Device\r\n * @param deviceSlot - Slot or ID of device\r\n * @returns DeviceSource\r\n */\r\n public getDeviceSource(deviceType: T, deviceSlot?: number): Nullable> {\r\n if (deviceSlot === undefined) {\r\n if (this._firstDevice[deviceType] === undefined) {\r\n return null;\r\n }\r\n\r\n deviceSlot = this._firstDevice[deviceType];\r\n }\r\n\r\n if (!this._devices[deviceType] || this._devices[deviceType][deviceSlot] === undefined) {\r\n return null;\r\n }\r\n\r\n return this._devices[deviceType][deviceSlot] as DeviceSource;\r\n }\r\n /**\r\n * Gets an array of DeviceSource objects for a given device type\r\n * @param deviceType - Type of Device\r\n * @returns All available DeviceSources of a given type\r\n */\r\n public getDeviceSources(deviceType: T): ReadonlyArray> {\r\n // If device type hasn't had any devices connected yet, return empty array.\r\n if (!this._devices[deviceType]) {\r\n return [];\r\n }\r\n return this._devices[deviceType].filter((source) => {\r\n return !!source;\r\n }) as Array>;\r\n }\r\n\r\n /**\r\n * Default constructor\r\n * @param engine - Used to get canvas (if applicable)\r\n */\r\n constructor(engine: Engine) {\r\n const numberOfDeviceTypes = Object.keys(DeviceType).length / 2;\r\n this._devices = new Array(numberOfDeviceTypes);\r\n this._firstDevice = new Array(numberOfDeviceTypes);\r\n this._engine = engine;\r\n\r\n if (!this._engine._deviceSourceManager) {\r\n this._engine._deviceSourceManager = new InternalDeviceSourceManager(engine);\r\n }\r\n this._engine._deviceSourceManager._refCount++;\r\n\r\n // Observables\r\n this.onDeviceConnectedObservable = new Observable((observer) => {\r\n for (const devices of this._devices) {\r\n if (devices) {\r\n for (const device of devices) {\r\n if (device) {\r\n this.onDeviceConnectedObservable.notifyObserver(observer, device as DeviceSourceType);\r\n }\r\n }\r\n }\r\n }\r\n });\r\n this.onDeviceDisconnectedObservable = new Observable();\r\n\r\n this._engine._deviceSourceManager.registerManager(this);\r\n\r\n this._onDisposeObserver = engine.onDisposeObservable.add(() => {\r\n this.dispose();\r\n });\r\n }\r\n\r\n /**\r\n * Dispose of DeviceSourceManager\r\n */\r\n public dispose(): void {\r\n // Null out observable refs\r\n this.onDeviceConnectedObservable.clear();\r\n this.onDeviceDisconnectedObservable.clear();\r\n\r\n if (this._engine._deviceSourceManager) {\r\n this._engine._deviceSourceManager.unregisterManager(this);\r\n if (--this._engine._deviceSourceManager._refCount < 1) {\r\n this._engine._deviceSourceManager.dispose();\r\n delete this._engine._deviceSourceManager;\r\n }\r\n }\r\n this._engine.onDisposeObservable.remove(this._onDisposeObserver);\r\n }\r\n\r\n // Hidden Functions\r\n /**\r\n * @param deviceSource - Source to add\r\n * @internal\r\n */\r\n public _addDevice(deviceSource: DeviceSourceType): void {\r\n if (!this._devices[deviceSource.deviceType]) {\r\n this._devices[deviceSource.deviceType] = new Array();\r\n }\r\n\r\n if (!this._devices[deviceSource.deviceType][deviceSource.deviceSlot]) {\r\n this._devices[deviceSource.deviceType][deviceSource.deviceSlot] = deviceSource;\r\n this._updateFirstDevices(deviceSource.deviceType);\r\n }\r\n\r\n this.onDeviceConnectedObservable.notifyObservers(deviceSource);\r\n }\r\n\r\n /**\r\n * @param deviceType - DeviceType\r\n * @param deviceSlot - DeviceSlot\r\n * @internal\r\n */\r\n public _removeDevice(deviceType: DeviceType, deviceSlot: number): void {\r\n const deviceSource = this._devices[deviceType]?.[deviceSlot]; // Grab local reference to use before removing from devices\r\n this.onDeviceDisconnectedObservable.notifyObservers(deviceSource as DeviceSourceType);\r\n if (this._devices[deviceType]?.[deviceSlot]) {\r\n delete this._devices[deviceType][deviceSlot];\r\n }\r\n // Even if we don't delete a device, we should still check for the first device as things may have gotten out of sync.\r\n this._updateFirstDevices(deviceType);\r\n }\r\n\r\n /**\r\n * @param deviceType - DeviceType\r\n * @param deviceSlot - DeviceSlot\r\n * @param eventData - Event\r\n * @internal\r\n */\r\n public _onInputChanged(deviceType: T, deviceSlot: number, eventData: IUIEvent): void {\r\n this._devices[deviceType]?.[deviceSlot]?.onInputChangedObservable.notifyObservers(eventData as IKeyboardEvent | IWheelEvent | IPointerEvent);\r\n }\r\n\r\n // Private Functions\r\n private _updateFirstDevices(type: DeviceType): void {\r\n switch (type) {\r\n case DeviceType.Keyboard:\r\n case DeviceType.Mouse:\r\n this._firstDevice[type] = 0;\r\n break;\r\n case DeviceType.Touch:\r\n case DeviceType.DualSense:\r\n case DeviceType.DualShock:\r\n case DeviceType.Xbox:\r\n case DeviceType.Switch:\r\n case DeviceType.Generic: {\r\n delete this._firstDevice[type];\r\n // eslint-disable-next-line no-case-declarations\r\n const devices = this._devices[type];\r\n if (devices) {\r\n for (let i = 0; i < devices.length; i++) {\r\n if (devices[i]) {\r\n this._firstDevice[type] = i;\r\n break;\r\n }\r\n }\r\n }\r\n break;\r\n }\r\n }\r\n }\r\n}\r\n","import type { EventState, Observable, Observer } from \"../Misc/observable\";\r\nimport { PointerInfoPre, PointerInfo, PointerEventTypes } from \"../Events/pointerEvents\";\r\nimport type { Nullable } from \"../types\";\r\nimport { AbstractActionManager } from \"../Actions/abstractActionManager\";\r\nimport { PickingInfo } from \"../Collisions/pickingInfo\";\r\nimport { Vector2, Matrix } from \"../Maths/math.vector\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport { ActionEvent } from \"../Actions/actionEvent\";\r\nimport { KeyboardEventTypes, KeyboardInfoPre, KeyboardInfo } from \"../Events/keyboardEvents\";\r\nimport { DeviceType, PointerInput } from \"../DeviceInput/InputDevices/deviceEnums\";\r\nimport type { IKeyboardEvent, IMouseEvent, IPointerEvent } from \"../Events/deviceInputEvents\";\r\nimport { DeviceSourceManager } from \"../DeviceInput/InputDevices/deviceSourceManager\";\r\nimport { EngineStore } from \"../Engines/engineStore\";\r\n\r\nimport type { Scene } from \"../scene\";\r\n\r\n/** @internal */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nclass _ClickInfo {\r\n private _singleClick = false;\r\n private _doubleClick = false;\r\n private _hasSwiped = false;\r\n private _ignore = false;\r\n\r\n public get singleClick(): boolean {\r\n return this._singleClick;\r\n }\r\n public get doubleClick(): boolean {\r\n return this._doubleClick;\r\n }\r\n public get hasSwiped(): boolean {\r\n return this._hasSwiped;\r\n }\r\n public get ignore(): boolean {\r\n return this._ignore;\r\n }\r\n\r\n public set singleClick(b: boolean) {\r\n this._singleClick = b;\r\n }\r\n public set doubleClick(b: boolean) {\r\n this._doubleClick = b;\r\n }\r\n public set hasSwiped(b: boolean) {\r\n this._hasSwiped = b;\r\n }\r\n public set ignore(b: boolean) {\r\n this._ignore = b;\r\n }\r\n}\r\n\r\n/** @internal */\r\ninterface _IClickEvent {\r\n clickInfo: _ClickInfo;\r\n evt: IPointerEvent;\r\n timeoutId: number;\r\n}\r\n\r\n/**\r\n * Class used to manage all inputs for the scene.\r\n */\r\nexport class InputManager {\r\n /** The distance in pixel that you have to move to prevent some events */\r\n public static DragMovementThreshold = 10; // in pixels\r\n /** Time in milliseconds to wait to raise long press events if button is still pressed */\r\n public static LongPressDelay = 500; // in milliseconds\r\n /** Time in milliseconds with two consecutive clicks will be considered as a double click */\r\n public static DoubleClickDelay = 300; // in milliseconds\r\n /**\r\n * This flag will modify the behavior so that, when true, a click will happen if and only if\r\n * another click DOES NOT happen within the DoubleClickDelay time frame. If another click does\r\n * happen within that time frame, the first click will not fire an event and and a double click will occur.\r\n */\r\n public static ExclusiveDoubleClickMode = false;\r\n\r\n /** This is a defensive check to not allow control attachment prior to an already active one. If already attached, previous control is unattached before attaching the new one. */\r\n private _alreadyAttached = false;\r\n private _alreadyAttachedTo: Nullable;\r\n\r\n // Pointers\r\n private _onPointerMove: (evt: IMouseEvent) => void;\r\n private _onPointerDown: (evt: IPointerEvent) => void;\r\n private _onPointerUp: (evt: IPointerEvent) => void;\r\n\r\n private _initClickEvent: (\r\n obs1: Observable,\r\n obs2: Observable,\r\n evt: IPointerEvent,\r\n cb: (clickInfo: _ClickInfo, pickResult: Nullable) => void\r\n ) => void;\r\n private _initActionManager: (act: Nullable, clickInfo: _ClickInfo) => Nullable;\r\n private _delayedSimpleClick: (btn: number, clickInfo: _ClickInfo, cb: (clickInfo: _ClickInfo, pickResult: Nullable) => void) => void;\r\n private _meshPickProceed = false;\r\n\r\n private _previousButtonPressed: number;\r\n private _currentPickResult: Nullable = null;\r\n private _previousPickResult: Nullable = null;\r\n private _totalPointersPressed = 0;\r\n private _doubleClickOccured = false;\r\n private _isSwiping: boolean = false;\r\n private _swipeButtonPressed: number = -1;\r\n private _skipPointerTap: boolean = false;\r\n private _isMultiTouchGesture: boolean = false;\r\n\r\n private _pointerOverMesh: Nullable;\r\n\r\n private _pickedDownMesh: Nullable;\r\n private _pickedUpMesh: Nullable;\r\n\r\n private _pointerX: number = 0;\r\n private _pointerY: number = 0;\r\n private _unTranslatedPointerX: number;\r\n private _unTranslatedPointerY: number;\r\n private _startingPointerPosition = new Vector2(0, 0);\r\n private _previousStartingPointerPosition = new Vector2(0, 0);\r\n private _startingPointerTime = 0;\r\n private _previousStartingPointerTime = 0;\r\n private _pointerCaptures: { [pointerId: number]: boolean } = {};\r\n private _meshUnderPointerId: { [pointerId: number]: Nullable } = {};\r\n private _movePointerInfo: Nullable = null;\r\n private _cameraObserverCount = 0;\r\n private _delayedClicks: Array> = [null, null, null, null, null];\r\n\r\n // Keyboard\r\n private _onKeyDown: (evt: IKeyboardEvent) => void;\r\n private _onKeyUp: (evt: IKeyboardEvent) => void;\r\n\r\n private _scene: Scene;\r\n private _deviceSourceManager: Nullable = null;\r\n\r\n /**\r\n * Creates a new InputManager\r\n * @param scene - defines the hosting scene\r\n */\r\n constructor(scene?: Scene) {\r\n this._scene = scene || EngineStore.LastCreatedScene;\r\n if (!this._scene) {\r\n return;\r\n }\r\n }\r\n\r\n /**\r\n * Gets the mesh that is currently under the pointer\r\n * @returns Mesh that the pointer is pointer is hovering over\r\n */\r\n public get meshUnderPointer(): Nullable {\r\n if (this._movePointerInfo) {\r\n // Because _pointerOverMesh is populated as part of _pickMove, we need to force a pick to update it.\r\n // Calling _pickMove calls _setCursorAndPointerOverMesh which calls setPointerOverMesh\r\n this._movePointerInfo._generatePickInfo();\r\n // Once we have what we need, we can clear _movePointerInfo because we don't need it anymore\r\n this._movePointerInfo = null;\r\n }\r\n return this._pointerOverMesh;\r\n }\r\n\r\n /**\r\n * When using more than one pointer (for example in XR) you can get the mesh under the specific pointer\r\n * @param pointerId - the pointer id to use\r\n * @returns The mesh under this pointer id or null if not found\r\n */\r\n public getMeshUnderPointerByPointerId(pointerId: number): Nullable {\r\n return this._meshUnderPointerId[pointerId] || null;\r\n }\r\n\r\n /**\r\n * Gets the pointer coordinates in 2D without any translation (ie. straight out of the pointer event)\r\n * @returns Vector with X/Y values directly from pointer event\r\n */\r\n public get unTranslatedPointer(): Vector2 {\r\n return new Vector2(this._unTranslatedPointerX, this._unTranslatedPointerY);\r\n }\r\n\r\n /**\r\n * Gets or sets the current on-screen X position of the pointer\r\n * @returns Translated X with respect to screen\r\n */\r\n public get pointerX(): number {\r\n return this._pointerX;\r\n }\r\n\r\n public set pointerX(value: number) {\r\n this._pointerX = value;\r\n }\r\n\r\n /**\r\n * Gets or sets the current on-screen Y position of the pointer\r\n * @returns Translated Y with respect to screen\r\n */\r\n public get pointerY(): number {\r\n return this._pointerY;\r\n }\r\n\r\n public set pointerY(value: number) {\r\n this._pointerY = value;\r\n }\r\n\r\n private _updatePointerPosition(evt: IPointerEvent): void {\r\n const canvasRect = this._scene.getEngine().getInputElementClientRect();\r\n\r\n if (!canvasRect) {\r\n return;\r\n }\r\n\r\n this._pointerX = evt.clientX - canvasRect.left;\r\n this._pointerY = evt.clientY - canvasRect.top;\r\n\r\n this._unTranslatedPointerX = this._pointerX;\r\n this._unTranslatedPointerY = this._pointerY;\r\n }\r\n\r\n private _processPointerMove(pickResult: Nullable, evt: IPointerEvent) {\r\n const scene = this._scene;\r\n const engine = scene.getEngine();\r\n const canvas = engine.getInputElement();\r\n\r\n if (canvas) {\r\n canvas.tabIndex = engine.canvasTabIndex;\r\n\r\n // Restore pointer\r\n if (!scene.doNotHandleCursors) {\r\n canvas.style.cursor = scene.defaultCursor;\r\n }\r\n }\r\n\r\n this._setCursorAndPointerOverMesh(pickResult, evt, scene);\r\n\r\n for (const step of scene._pointerMoveStage) {\r\n // If _pointerMoveState is defined, we have an active spriteManager and can't use Lazy Picking\r\n // Therefore, we need to force a pick to update the pickResult\r\n pickResult = pickResult || this._pickMove(evt);\r\n const isMeshPicked = pickResult?.pickedMesh ? true : false;\r\n pickResult = step.action(this._unTranslatedPointerX, this._unTranslatedPointerY, pickResult, isMeshPicked, canvas);\r\n }\r\n\r\n const type = evt.inputIndex >= PointerInput.MouseWheelX && evt.inputIndex <= PointerInput.MouseWheelZ ? PointerEventTypes.POINTERWHEEL : PointerEventTypes.POINTERMOVE;\r\n\r\n if (scene.onPointerMove) {\r\n // Because of lazy picking, we need to force a pick to update the pickResult\r\n pickResult = pickResult || this._pickMove(evt);\r\n scene.onPointerMove(evt, pickResult, type);\r\n }\r\n\r\n let pointerInfo: PointerInfo;\r\n if (pickResult) {\r\n pointerInfo = new PointerInfo(type, evt, pickResult);\r\n this._setRayOnPointerInfo(pickResult, evt);\r\n } else {\r\n pointerInfo = new PointerInfo(type, evt, null, this);\r\n this._movePointerInfo = pointerInfo;\r\n }\r\n\r\n if (scene.onPointerObservable.hasObservers()) {\r\n scene.onPointerObservable.notifyObservers(pointerInfo, type);\r\n }\r\n }\r\n\r\n // Pointers handling\r\n /** @internal */\r\n public _setRayOnPointerInfo(pickInfo: Nullable, event: IMouseEvent) {\r\n const scene = this._scene;\r\n if (pickInfo && scene._pickingAvailable) {\r\n if (!pickInfo.ray) {\r\n pickInfo.ray = scene.createPickingRay(event.offsetX, event.offsetY, Matrix.Identity(), scene.activeCamera);\r\n }\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _addCameraPointerObserver(observer: (p: PointerInfo, s: EventState) => void, mask?: number): Nullable> {\r\n this._cameraObserverCount++;\r\n return this._scene.onPointerObservable.add(observer, mask);\r\n }\r\n\r\n /** @internal */\r\n public _removeCameraPointerObserver(observer: Observer): boolean {\r\n this._cameraObserverCount--;\r\n return this._scene.onPointerObservable.remove(observer);\r\n }\r\n\r\n private _checkForPicking(): boolean {\r\n return !!(this._scene.onPointerObservable.observers.length > this._cameraObserverCount || this._scene.onPointerPick);\r\n }\r\n\r\n private _checkPrePointerObservable(pickResult: Nullable, evt: IPointerEvent, type: number) {\r\n const scene = this._scene;\r\n const pi = new PointerInfoPre(type, evt, this._unTranslatedPointerX, this._unTranslatedPointerY);\r\n if (pickResult) {\r\n pi.originalPickingInfo = pickResult;\r\n pi.ray = pickResult.ray;\r\n if (pickResult.originMesh) {\r\n pi.nearInteractionPickingInfo = pickResult;\r\n }\r\n }\r\n\r\n scene.onPrePointerObservable.notifyObservers(pi, type);\r\n if (pi.skipOnPointerObservable) {\r\n return true;\r\n } else {\r\n return false;\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _pickMove(evt: IPointerEvent): PickingInfo {\r\n const scene = this._scene;\r\n const pickResult = scene.pick(\r\n this._unTranslatedPointerX,\r\n this._unTranslatedPointerY,\r\n scene.pointerMovePredicate,\r\n scene.pointerMoveFastCheck,\r\n scene.cameraToUseForPointers,\r\n scene.pointerMoveTrianglePredicate\r\n );\r\n\r\n this._setCursorAndPointerOverMesh(pickResult, evt, scene);\r\n\r\n return pickResult;\r\n }\r\n\r\n private _setCursorAndPointerOverMesh(pickResult: Nullable, evt: IPointerEvent, scene: Scene) {\r\n const engine = scene.getEngine();\r\n const canvas = engine.getInputElement();\r\n\r\n if (pickResult?.pickedMesh) {\r\n this.setPointerOverMesh(pickResult.pickedMesh, evt.pointerId, pickResult, evt);\r\n\r\n if (!scene.doNotHandleCursors && canvas && this._pointerOverMesh) {\r\n const actionManager = this._pointerOverMesh._getActionManagerForTrigger();\r\n if (actionManager && actionManager.hasPointerTriggers) {\r\n canvas.style.cursor = actionManager.hoverCursor || scene.hoverCursor;\r\n }\r\n }\r\n } else {\r\n this.setPointerOverMesh(null, evt.pointerId, pickResult, evt);\r\n }\r\n }\r\n\r\n /**\r\n * Use this method to simulate a pointer move on a mesh\r\n * The pickResult parameter can be obtained from a scene.pick or scene.pickWithRay\r\n * @param pickResult - pickingInfo of the object wished to simulate pointer event on\r\n * @param pointerEventInit - pointer event state to be used when simulating the pointer event (eg. pointer id for multitouch)\r\n */\r\n public simulatePointerMove(pickResult: PickingInfo, pointerEventInit?: PointerEventInit): void {\r\n const evt = new PointerEvent(\"pointermove\", pointerEventInit);\r\n evt.inputIndex = PointerInput.Move;\r\n\r\n if (this._checkPrePointerObservable(pickResult, evt, PointerEventTypes.POINTERMOVE)) {\r\n return;\r\n }\r\n this._processPointerMove(pickResult, evt);\r\n }\r\n\r\n /**\r\n * Use this method to simulate a pointer down on a mesh\r\n * The pickResult parameter can be obtained from a scene.pick or scene.pickWithRay\r\n * @param pickResult - pickingInfo of the object wished to simulate pointer event on\r\n * @param pointerEventInit - pointer event state to be used when simulating the pointer event (eg. pointer id for multitouch)\r\n */\r\n public simulatePointerDown(pickResult: PickingInfo, pointerEventInit?: PointerEventInit): void {\r\n const evt = new PointerEvent(\"pointerdown\", pointerEventInit);\r\n evt.inputIndex = evt.button + 2;\r\n\r\n if (this._checkPrePointerObservable(pickResult, evt, PointerEventTypes.POINTERDOWN)) {\r\n return;\r\n }\r\n\r\n this._processPointerDown(pickResult, evt);\r\n }\r\n\r\n private _processPointerDown(pickResult: Nullable, evt: IPointerEvent): void {\r\n const scene = this._scene;\r\n if (pickResult?.pickedMesh) {\r\n this._pickedDownMesh = pickResult.pickedMesh;\r\n const actionManager = pickResult.pickedMesh._getActionManagerForTrigger();\r\n if (actionManager) {\r\n if (actionManager.hasPickTriggers) {\r\n actionManager.processTrigger(Constants.ACTION_OnPickDownTrigger, ActionEvent.CreateNew(pickResult.pickedMesh, evt, pickResult));\r\n switch (evt.button) {\r\n case 0:\r\n actionManager.processTrigger(Constants.ACTION_OnLeftPickTrigger, ActionEvent.CreateNew(pickResult.pickedMesh, evt, pickResult));\r\n break;\r\n case 1:\r\n actionManager.processTrigger(Constants.ACTION_OnCenterPickTrigger, ActionEvent.CreateNew(pickResult.pickedMesh, evt, pickResult));\r\n break;\r\n case 2:\r\n actionManager.processTrigger(Constants.ACTION_OnRightPickTrigger, ActionEvent.CreateNew(pickResult.pickedMesh, evt, pickResult));\r\n break;\r\n }\r\n }\r\n\r\n if (actionManager.hasSpecificTrigger(Constants.ACTION_OnLongPressTrigger)) {\r\n window.setTimeout(() => {\r\n const pickResult = scene.pick(\r\n this._unTranslatedPointerX,\r\n this._unTranslatedPointerY,\r\n (mesh: AbstractMesh): boolean =>\r\n (\r\n (mesh.isPickable &&\r\n mesh.isVisible &&\r\n mesh.isReady() &&\r\n mesh.actionManager &&\r\n mesh.actionManager.hasSpecificTrigger(Constants.ACTION_OnLongPressTrigger) &&\r\n mesh === this._pickedDownMesh)\r\n ),\r\n false,\r\n scene.cameraToUseForPointers\r\n );\r\n\r\n if (pickResult?.pickedMesh && actionManager) {\r\n if (this._totalPointersPressed !== 0 && Date.now() - this._startingPointerTime > InputManager.LongPressDelay && !this._isPointerSwiping()) {\r\n this._startingPointerTime = 0;\r\n actionManager.processTrigger(Constants.ACTION_OnLongPressTrigger, ActionEvent.CreateNew(pickResult.pickedMesh, evt));\r\n }\r\n }\r\n }, InputManager.LongPressDelay);\r\n }\r\n }\r\n } else {\r\n for (const step of scene._pointerDownStage) {\r\n pickResult = step.action(this._unTranslatedPointerX, this._unTranslatedPointerY, pickResult, evt, false);\r\n }\r\n }\r\n\r\n let pointerInfo: PointerInfo;\r\n const type = PointerEventTypes.POINTERDOWN;\r\n\r\n if (pickResult) {\r\n if (scene.onPointerDown) {\r\n scene.onPointerDown(evt, pickResult, type);\r\n }\r\n\r\n pointerInfo = new PointerInfo(type, evt, pickResult);\r\n this._setRayOnPointerInfo(pickResult, evt);\r\n } else {\r\n pointerInfo = new PointerInfo(type, evt, null, this);\r\n }\r\n\r\n if (scene.onPointerObservable.hasObservers()) {\r\n scene.onPointerObservable.notifyObservers(pointerInfo, type);\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n * @internals Boolean if delta for pointer exceeds drag movement threshold\r\n */\r\n public _isPointerSwiping(): boolean {\r\n return this._isSwiping;\r\n }\r\n\r\n /**\r\n * Use this method to simulate a pointer up on a mesh\r\n * The pickResult parameter can be obtained from a scene.pick or scene.pickWithRay\r\n * @param pickResult - pickingInfo of the object wished to simulate pointer event on\r\n * @param pointerEventInit - pointer event state to be used when simulating the pointer event (eg. pointer id for multitouch)\r\n * @param doubleTap - indicates that the pointer up event should be considered as part of a double click (false by default)\r\n */\r\n public simulatePointerUp(pickResult: PickingInfo, pointerEventInit?: PointerEventInit, doubleTap?: boolean): void {\r\n const evt = new PointerEvent(\"pointerup\", pointerEventInit);\r\n evt.inputIndex = PointerInput.Move;\r\n const clickInfo = new _ClickInfo();\r\n\r\n if (doubleTap) {\r\n clickInfo.doubleClick = true;\r\n } else {\r\n clickInfo.singleClick = true;\r\n }\r\n\r\n if (this._checkPrePointerObservable(pickResult, evt, PointerEventTypes.POINTERUP)) {\r\n return;\r\n }\r\n\r\n this._processPointerUp(pickResult, evt, clickInfo);\r\n }\r\n\r\n private _processPointerUp(pickResult: Nullable, evt: IPointerEvent, clickInfo: _ClickInfo): void {\r\n const scene = this._scene;\r\n if (pickResult?.pickedMesh) {\r\n this._pickedUpMesh = pickResult.pickedMesh;\r\n if (this._pickedDownMesh === this._pickedUpMesh) {\r\n if (scene.onPointerPick) {\r\n scene.onPointerPick(evt, pickResult);\r\n }\r\n if (clickInfo.singleClick && !clickInfo.ignore && scene.onPointerObservable.observers.length > this._cameraObserverCount) {\r\n const type = PointerEventTypes.POINTERPICK;\r\n const pi = new PointerInfo(type, evt, pickResult);\r\n this._setRayOnPointerInfo(pickResult, evt);\r\n scene.onPointerObservable.notifyObservers(pi, type);\r\n }\r\n }\r\n const actionManager = pickResult.pickedMesh._getActionManagerForTrigger();\r\n if (actionManager && !clickInfo.ignore) {\r\n actionManager.processTrigger(Constants.ACTION_OnPickUpTrigger, ActionEvent.CreateNew(pickResult.pickedMesh, evt, pickResult));\r\n\r\n if (!clickInfo.hasSwiped && clickInfo.singleClick) {\r\n actionManager.processTrigger(Constants.ACTION_OnPickTrigger, ActionEvent.CreateNew(pickResult.pickedMesh, evt, pickResult));\r\n }\r\n\r\n const doubleClickActionManager = pickResult.pickedMesh._getActionManagerForTrigger(Constants.ACTION_OnDoublePickTrigger);\r\n if (clickInfo.doubleClick && doubleClickActionManager) {\r\n doubleClickActionManager.processTrigger(Constants.ACTION_OnDoublePickTrigger, ActionEvent.CreateNew(pickResult.pickedMesh, evt, pickResult));\r\n }\r\n }\r\n } else {\r\n if (!clickInfo.ignore) {\r\n for (const step of scene._pointerUpStage) {\r\n pickResult = step.action(this._unTranslatedPointerX, this._unTranslatedPointerY, pickResult, evt, clickInfo.doubleClick);\r\n }\r\n }\r\n }\r\n\r\n if (this._pickedDownMesh && this._pickedDownMesh !== this._pickedUpMesh) {\r\n const pickedDownActionManager = this._pickedDownMesh._getActionManagerForTrigger(Constants.ACTION_OnPickOutTrigger);\r\n if (pickedDownActionManager) {\r\n pickedDownActionManager.processTrigger(Constants.ACTION_OnPickOutTrigger, ActionEvent.CreateNew(this._pickedDownMesh, evt));\r\n }\r\n }\r\n\r\n if (!clickInfo.ignore) {\r\n const pi = new PointerInfo(PointerEventTypes.POINTERUP, evt, pickResult);\r\n // Set ray on picking info. Note that this info will also be reused for the tap notification.\r\n this._setRayOnPointerInfo(pickResult, evt);\r\n scene.onPointerObservable.notifyObservers(pi, PointerEventTypes.POINTERUP);\r\n\r\n if (scene.onPointerUp) {\r\n scene.onPointerUp(evt, pickResult, PointerEventTypes.POINTERUP);\r\n }\r\n\r\n if (!clickInfo.hasSwiped && !this._skipPointerTap && !this._isMultiTouchGesture) {\r\n let type = 0;\r\n if (clickInfo.singleClick) {\r\n type = PointerEventTypes.POINTERTAP;\r\n } else if (clickInfo.doubleClick) {\r\n type = PointerEventTypes.POINTERDOUBLETAP;\r\n }\r\n\r\n if (type) {\r\n const pi = new PointerInfo(type, evt, pickResult);\r\n if (scene.onPointerObservable.hasObservers() && scene.onPointerObservable.hasSpecificMask(type)) {\r\n scene.onPointerObservable.notifyObservers(pi, type);\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if the current pointer event is captured (meaning that the scene has already handled the pointer down)\r\n * @param pointerId - defines the pointer id to use in a multi-touch scenario (0 by default)\r\n * @returns true if the pointer was captured\r\n */\r\n public isPointerCaptured(pointerId = 0): boolean {\r\n return this._pointerCaptures[pointerId];\r\n }\r\n\r\n /**\r\n * Attach events to the canvas (To handle actionManagers triggers and raise onPointerMove, onPointerDown and onPointerUp\r\n * @param attachUp - defines if you want to attach events to pointerup\r\n * @param attachDown - defines if you want to attach events to pointerdown\r\n * @param attachMove - defines if you want to attach events to pointermove\r\n * @param elementToAttachTo - defines the target DOM element to attach to (will use the canvas by default)\r\n */\r\n public attachControl(attachUp = true, attachDown = true, attachMove = true, elementToAttachTo: Nullable = null): void {\r\n const scene = this._scene;\r\n const engine = scene.getEngine();\r\n\r\n if (!elementToAttachTo) {\r\n elementToAttachTo = engine.getInputElement();\r\n }\r\n\r\n if (this._alreadyAttached) {\r\n this.detachControl();\r\n }\r\n\r\n if (elementToAttachTo) {\r\n this._alreadyAttachedTo = elementToAttachTo;\r\n }\r\n this._deviceSourceManager = new DeviceSourceManager(engine);\r\n\r\n // Because this is only called from _initClickEvent, which is called in _onPointerUp, we'll use the pointerUpPredicate for the pick call\r\n this._initActionManager = (act: Nullable): Nullable => {\r\n if (!this._meshPickProceed) {\r\n const pickResult =\r\n scene.skipPointerUpPicking || (scene._registeredActions === 0 && !this._checkForPicking() && !scene.onPointerUp)\r\n ? null\r\n : scene.pick(this._unTranslatedPointerX, this._unTranslatedPointerY, scene.pointerUpPredicate, scene.pointerUpFastCheck, scene.cameraToUseForPointers);\r\n this._currentPickResult = pickResult;\r\n if (pickResult) {\r\n act = pickResult.hit && pickResult.pickedMesh ? pickResult.pickedMesh._getActionManagerForTrigger() : null;\r\n }\r\n this._meshPickProceed = true;\r\n }\r\n return act;\r\n };\r\n\r\n this._delayedSimpleClick = (btn: number, clickInfo: _ClickInfo, cb: (clickInfo: _ClickInfo, pickResult: Nullable) => void) => {\r\n // double click delay is over and that no double click has been raised since, or the 2 consecutive keys pressed are different\r\n if ((Date.now() - this._previousStartingPointerTime > InputManager.DoubleClickDelay && !this._doubleClickOccured) || btn !== this._previousButtonPressed) {\r\n this._doubleClickOccured = false;\r\n clickInfo.singleClick = true;\r\n clickInfo.ignore = false;\r\n\r\n // If we have a delayed click, we need to resolve the TAP event\r\n if (this._delayedClicks[btn]) {\r\n const evt = this._delayedClicks[btn]!.evt;\r\n const type = PointerEventTypes.POINTERTAP;\r\n const pi = new PointerInfo(type, evt, this._currentPickResult);\r\n if (scene.onPointerObservable.hasObservers() && scene.onPointerObservable.hasSpecificMask(type)) {\r\n scene.onPointerObservable.notifyObservers(pi, type);\r\n }\r\n\r\n // Clear the delayed click\r\n this._delayedClicks[btn] = null;\r\n }\r\n }\r\n };\r\n\r\n this._initClickEvent = (\r\n obs1: Observable,\r\n obs2: Observable,\r\n evt: IPointerEvent,\r\n cb: (clickInfo: _ClickInfo, pickResult: Nullable) => void\r\n ): void => {\r\n const clickInfo = new _ClickInfo();\r\n this._currentPickResult = null;\r\n let act: Nullable = null;\r\n\r\n let checkPicking =\r\n obs1.hasSpecificMask(PointerEventTypes.POINTERPICK) ||\r\n obs2.hasSpecificMask(PointerEventTypes.POINTERPICK) ||\r\n obs1.hasSpecificMask(PointerEventTypes.POINTERTAP) ||\r\n obs2.hasSpecificMask(PointerEventTypes.POINTERTAP) ||\r\n obs1.hasSpecificMask(PointerEventTypes.POINTERDOUBLETAP) ||\r\n obs2.hasSpecificMask(PointerEventTypes.POINTERDOUBLETAP);\r\n if (!checkPicking && AbstractActionManager) {\r\n act = this._initActionManager(act, clickInfo);\r\n if (act) {\r\n checkPicking = act.hasPickTriggers;\r\n }\r\n }\r\n\r\n let needToIgnoreNext = false;\r\n\r\n if (checkPicking) {\r\n const btn = evt.button;\r\n clickInfo.hasSwiped = this._isPointerSwiping();\r\n\r\n if (!clickInfo.hasSwiped) {\r\n let checkSingleClickImmediately = !InputManager.ExclusiveDoubleClickMode;\r\n\r\n if (!checkSingleClickImmediately) {\r\n checkSingleClickImmediately = !obs1.hasSpecificMask(PointerEventTypes.POINTERDOUBLETAP) && !obs2.hasSpecificMask(PointerEventTypes.POINTERDOUBLETAP);\r\n\r\n if (checkSingleClickImmediately && !AbstractActionManager.HasSpecificTrigger(Constants.ACTION_OnDoublePickTrigger)) {\r\n act = this._initActionManager(act, clickInfo);\r\n if (act) {\r\n checkSingleClickImmediately = !act.hasSpecificTrigger(Constants.ACTION_OnDoublePickTrigger);\r\n }\r\n }\r\n }\r\n\r\n if (checkSingleClickImmediately) {\r\n // single click detected if double click delay is over or two different successive keys pressed without exclusive double click or no double click required\r\n if (Date.now() - this._previousStartingPointerTime > InputManager.DoubleClickDelay || btn !== this._previousButtonPressed) {\r\n clickInfo.singleClick = true;\r\n cb(clickInfo, this._currentPickResult);\r\n needToIgnoreNext = true;\r\n }\r\n }\r\n // at least one double click is required to be check and exclusive double click is enabled\r\n else {\r\n // Queue up a delayed click, just in case this isn't a double click\r\n // It should be noted that while this delayed event happens\r\n // because of user input, it shouldn't be considered as a direct,\r\n // timing-dependent result of that input. It's meant to just fire the TAP event\r\n const delayedClick = {\r\n evt: evt,\r\n clickInfo: clickInfo,\r\n timeoutId: window.setTimeout(this._delayedSimpleClick.bind(this, btn, clickInfo, cb), InputManager.DoubleClickDelay),\r\n };\r\n\r\n this._delayedClicks[btn] = delayedClick;\r\n }\r\n\r\n let checkDoubleClick = obs1.hasSpecificMask(PointerEventTypes.POINTERDOUBLETAP) || obs2.hasSpecificMask(PointerEventTypes.POINTERDOUBLETAP);\r\n if (!checkDoubleClick && AbstractActionManager.HasSpecificTrigger(Constants.ACTION_OnDoublePickTrigger)) {\r\n act = this._initActionManager(act, clickInfo);\r\n if (act) {\r\n checkDoubleClick = act.hasSpecificTrigger(Constants.ACTION_OnDoublePickTrigger);\r\n }\r\n }\r\n if (checkDoubleClick) {\r\n // two successive keys pressed are equal, double click delay is not over and double click has not just occurred\r\n if (btn === this._previousButtonPressed && Date.now() - this._previousStartingPointerTime < InputManager.DoubleClickDelay && !this._doubleClickOccured) {\r\n // pointer has not moved for 2 clicks, it's a double click\r\n if (!clickInfo.hasSwiped && !this._isPointerSwiping()) {\r\n this._previousStartingPointerTime = 0;\r\n this._doubleClickOccured = true;\r\n clickInfo.doubleClick = true;\r\n clickInfo.ignore = false;\r\n // If we have a pending click, we need to cancel it\r\n if (InputManager.ExclusiveDoubleClickMode && this._delayedClicks[btn]) {\r\n clearTimeout(this._delayedClicks[btn]?.timeoutId);\r\n this._delayedClicks[btn] = null;\r\n }\r\n\r\n cb(clickInfo, this._currentPickResult);\r\n }\r\n // if the two successive clicks are too far, it's just two simple clicks\r\n else {\r\n this._doubleClickOccured = false;\r\n this._previousStartingPointerTime = this._startingPointerTime;\r\n this._previousStartingPointerPosition.x = this._startingPointerPosition.x;\r\n this._previousStartingPointerPosition.y = this._startingPointerPosition.y;\r\n this._previousButtonPressed = btn;\r\n if (InputManager.ExclusiveDoubleClickMode) {\r\n // If we have a delayed click, we need to cancel it\r\n if (this._delayedClicks[btn]) {\r\n clearTimeout(this._delayedClicks[btn]?.timeoutId);\r\n this._delayedClicks[btn] = null;\r\n }\r\n cb(clickInfo, this._previousPickResult);\r\n } else {\r\n cb(clickInfo, this._currentPickResult);\r\n }\r\n }\r\n needToIgnoreNext = true;\r\n }\r\n // just the first click of the double has been raised\r\n else {\r\n this._doubleClickOccured = false;\r\n this._previousStartingPointerTime = this._startingPointerTime;\r\n this._previousStartingPointerPosition.x = this._startingPointerPosition.x;\r\n this._previousStartingPointerPosition.y = this._startingPointerPosition.y;\r\n this._previousButtonPressed = btn!;\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Even if ExclusiveDoubleClickMode is true, we need to always handle\r\n // up events at time of execution, unless we're explicitly ignoring them.\r\n if (!needToIgnoreNext) {\r\n cb(clickInfo, this._currentPickResult);\r\n }\r\n };\r\n\r\n this._onPointerMove = (evt: IMouseEvent) => {\r\n this._updatePointerPosition(evt as IPointerEvent);\r\n\r\n // Check if pointer leaves DragMovementThreshold range to determine if swipe is occurring\r\n if (!this._isSwiping && this._swipeButtonPressed !== -1) {\r\n this._isSwiping =\r\n Math.abs(this._startingPointerPosition.x - this._pointerX) > InputManager.DragMovementThreshold ||\r\n Math.abs(this._startingPointerPosition.y - this._pointerY) > InputManager.DragMovementThreshold;\r\n }\r\n\r\n // Because there's a race condition between pointermove and pointerlockchange events, we need to\r\n // verify that the pointer is still locked after each pointermove event.\r\n if (engine.isPointerLock) {\r\n engine._verifyPointerLock();\r\n }\r\n\r\n // PreObservable support\r\n if (\r\n this._checkPrePointerObservable(\r\n null,\r\n evt as IPointerEvent,\r\n evt.inputIndex >= PointerInput.MouseWheelX && evt.inputIndex <= PointerInput.MouseWheelZ ? PointerEventTypes.POINTERWHEEL : PointerEventTypes.POINTERMOVE\r\n )\r\n ) {\r\n return;\r\n }\r\n\r\n if (!scene.cameraToUseForPointers && !scene.activeCamera) {\r\n return;\r\n }\r\n\r\n if (scene.skipPointerMovePicking) {\r\n this._processPointerMove(new PickingInfo(), evt as IPointerEvent);\r\n return;\r\n }\r\n\r\n if (!scene.pointerMovePredicate) {\r\n scene.pointerMovePredicate = (mesh: AbstractMesh): boolean =>\r\n mesh.isPickable &&\r\n mesh.isVisible &&\r\n mesh.isReady() &&\r\n mesh.isEnabled() &&\r\n (mesh.enablePointerMoveEvents || scene.constantlyUpdateMeshUnderPointer || mesh._getActionManagerForTrigger() !== null) &&\r\n (!scene.cameraToUseForPointers || (scene.cameraToUseForPointers.layerMask & mesh.layerMask) !== 0);\r\n }\r\n\r\n const pickResult = scene._registeredActions > 0 || scene.constantlyUpdateMeshUnderPointer ? this._pickMove(evt as IPointerEvent) : null;\r\n this._processPointerMove(pickResult, evt as IPointerEvent);\r\n };\r\n\r\n this._onPointerDown = (evt: IPointerEvent) => {\r\n this._totalPointersPressed++;\r\n this._pickedDownMesh = null;\r\n this._meshPickProceed = false;\r\n\r\n // If ExclusiveDoubleClickMode is true, we need to resolve any pending delayed clicks\r\n if (InputManager.ExclusiveDoubleClickMode) {\r\n for (let i = 0; i < this._delayedClicks.length; i++) {\r\n if (this._delayedClicks[i]) {\r\n // If the button that was pressed is the same as the one that was released,\r\n // just clear the timer. This will be resolved in the up event.\r\n if (evt.button === i) {\r\n clearTimeout(this._delayedClicks[i]?.timeoutId);\r\n } else {\r\n // Otherwise, we need to resolve the click\r\n const clickInfo = this._delayedClicks[i]!.clickInfo;\r\n this._doubleClickOccured = false;\r\n clickInfo.singleClick = true;\r\n clickInfo.ignore = false;\r\n\r\n const prevEvt = this._delayedClicks[i]!.evt;\r\n const type = PointerEventTypes.POINTERTAP;\r\n const pi = new PointerInfo(type, prevEvt, this._currentPickResult);\r\n if (scene.onPointerObservable.hasObservers() && scene.onPointerObservable.hasSpecificMask(type)) {\r\n scene.onPointerObservable.notifyObservers(pi, type);\r\n }\r\n\r\n // Clear the delayed click\r\n this._delayedClicks[i] = null;\r\n }\r\n }\r\n }\r\n }\r\n\r\n this._updatePointerPosition(evt);\r\n\r\n if (this._swipeButtonPressed === -1) {\r\n this._swipeButtonPressed = evt.button;\r\n }\r\n\r\n if (scene.preventDefaultOnPointerDown && elementToAttachTo) {\r\n evt.preventDefault();\r\n elementToAttachTo.focus();\r\n }\r\n\r\n this._startingPointerPosition.x = this._pointerX;\r\n this._startingPointerPosition.y = this._pointerY;\r\n this._startingPointerTime = Date.now();\r\n\r\n // PreObservable support\r\n if (this._checkPrePointerObservable(null, evt, PointerEventTypes.POINTERDOWN)) {\r\n return;\r\n }\r\n\r\n if (!scene.cameraToUseForPointers && !scene.activeCamera) {\r\n return;\r\n }\r\n\r\n this._pointerCaptures[evt.pointerId] = true;\r\n\r\n if (!scene.pointerDownPredicate) {\r\n scene.pointerDownPredicate = (mesh: AbstractMesh): boolean => {\r\n return (\r\n mesh.isPickable &&\r\n mesh.isVisible &&\r\n mesh.isReady() &&\r\n mesh.isEnabled() &&\r\n (!scene.cameraToUseForPointers || (scene.cameraToUseForPointers.layerMask & mesh.layerMask) !== 0)\r\n );\r\n };\r\n }\r\n\r\n // Meshes\r\n this._pickedDownMesh = null;\r\n let pickResult;\r\n if (scene.skipPointerDownPicking || (scene._registeredActions === 0 && !this._checkForPicking() && !scene.onPointerDown)) {\r\n pickResult = new PickingInfo();\r\n } else {\r\n pickResult = scene.pick(\r\n this._unTranslatedPointerX,\r\n this._unTranslatedPointerY,\r\n scene.pointerDownPredicate,\r\n scene.pointerDownFastCheck,\r\n scene.cameraToUseForPointers\r\n );\r\n }\r\n\r\n this._processPointerDown(pickResult, evt);\r\n };\r\n\r\n this._onPointerUp = (evt: IPointerEvent) => {\r\n if (this._totalPointersPressed === 0) {\r\n // We are attaching the pointer up to windows because of a bug in FF\r\n return; // So we need to test it the pointer down was pressed before.\r\n }\r\n\r\n this._totalPointersPressed--;\r\n this._pickedUpMesh = null;\r\n this._meshPickProceed = false;\r\n\r\n this._updatePointerPosition(evt);\r\n\r\n if (scene.preventDefaultOnPointerUp && elementToAttachTo) {\r\n evt.preventDefault();\r\n elementToAttachTo.focus();\r\n }\r\n\r\n this._initClickEvent(scene.onPrePointerObservable, scene.onPointerObservable, evt, (clickInfo: _ClickInfo, pickResult: Nullable) => {\r\n // PreObservable support\r\n if (scene.onPrePointerObservable.hasObservers()) {\r\n this._skipPointerTap = false;\r\n if (!clickInfo.ignore) {\r\n if (this._checkPrePointerObservable(null, evt, PointerEventTypes.POINTERUP)) {\r\n // If we're skipping the next observable, we need to reset the swipe state before returning\r\n if (this._swipeButtonPressed === evt.button) {\r\n this._isSwiping = false;\r\n this._swipeButtonPressed = -1;\r\n }\r\n\r\n // If we're going to skip the POINTERUP, we need to reset the pointer capture\r\n if (evt.buttons === 0) {\r\n this._pointerCaptures[evt.pointerId] = false;\r\n }\r\n\r\n return;\r\n }\r\n if (!clickInfo.hasSwiped) {\r\n if (clickInfo.singleClick && scene.onPrePointerObservable.hasSpecificMask(PointerEventTypes.POINTERTAP)) {\r\n if (this._checkPrePointerObservable(null, evt, PointerEventTypes.POINTERTAP)) {\r\n this._skipPointerTap = true;\r\n }\r\n }\r\n if (clickInfo.doubleClick && scene.onPrePointerObservable.hasSpecificMask(PointerEventTypes.POINTERDOUBLETAP)) {\r\n if (this._checkPrePointerObservable(null, evt, PointerEventTypes.POINTERDOUBLETAP)) {\r\n this._skipPointerTap = true;\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n // There should be a pointer captured at this point so if there isn't we should reset and return\r\n if (!this._pointerCaptures[evt.pointerId]) {\r\n if (this._swipeButtonPressed === evt.button) {\r\n this._isSwiping = false;\r\n this._swipeButtonPressed = -1;\r\n }\r\n return;\r\n }\r\n\r\n // Only release capture if all buttons are released\r\n if (evt.buttons === 0) {\r\n this._pointerCaptures[evt.pointerId] = false;\r\n }\r\n if (!scene.cameraToUseForPointers && !scene.activeCamera) {\r\n return;\r\n }\r\n\r\n if (!scene.pointerUpPredicate) {\r\n scene.pointerUpPredicate = (mesh: AbstractMesh): boolean => {\r\n return (\r\n mesh.isPickable &&\r\n mesh.isVisible &&\r\n mesh.isReady() &&\r\n mesh.isEnabled() &&\r\n (!scene.cameraToUseForPointers || (scene.cameraToUseForPointers.layerMask & mesh.layerMask) !== 0)\r\n );\r\n };\r\n }\r\n\r\n // Meshes\r\n if (!this._meshPickProceed && ((AbstractActionManager && AbstractActionManager.HasTriggers) || this._checkForPicking() || scene.onPointerUp)) {\r\n this._initActionManager(null, clickInfo);\r\n }\r\n if (!pickResult) {\r\n pickResult = this._currentPickResult;\r\n }\r\n\r\n this._processPointerUp(pickResult, evt, clickInfo);\r\n\r\n this._previousPickResult = this._currentPickResult;\r\n\r\n if (this._swipeButtonPressed === evt.button) {\r\n this._isSwiping = false;\r\n this._swipeButtonPressed = -1;\r\n }\r\n });\r\n };\r\n\r\n this._onKeyDown = (evt: IKeyboardEvent) => {\r\n const type = KeyboardEventTypes.KEYDOWN;\r\n if (scene.onPreKeyboardObservable.hasObservers()) {\r\n const pi = new KeyboardInfoPre(type, evt);\r\n scene.onPreKeyboardObservable.notifyObservers(pi, type);\r\n if (pi.skipOnKeyboardObservable) {\r\n return;\r\n }\r\n }\r\n\r\n if (scene.onKeyboardObservable.hasObservers()) {\r\n const pi = new KeyboardInfo(type, evt);\r\n scene.onKeyboardObservable.notifyObservers(pi, type);\r\n }\r\n\r\n if (scene.actionManager) {\r\n scene.actionManager.processTrigger(Constants.ACTION_OnKeyDownTrigger, ActionEvent.CreateNewFromScene(scene, evt));\r\n }\r\n };\r\n\r\n this._onKeyUp = (evt: IKeyboardEvent) => {\r\n const type = KeyboardEventTypes.KEYUP;\r\n if (scene.onPreKeyboardObservable.hasObservers()) {\r\n const pi = new KeyboardInfoPre(type, evt);\r\n scene.onPreKeyboardObservable.notifyObservers(pi, type);\r\n if (pi.skipOnKeyboardObservable) {\r\n return;\r\n }\r\n }\r\n\r\n if (scene.onKeyboardObservable.hasObservers()) {\r\n const pi = new KeyboardInfo(type, evt);\r\n scene.onKeyboardObservable.notifyObservers(pi, type);\r\n }\r\n\r\n if (scene.actionManager) {\r\n scene.actionManager.processTrigger(Constants.ACTION_OnKeyUpTrigger, ActionEvent.CreateNewFromScene(scene, evt));\r\n }\r\n };\r\n\r\n // If a device connects that we can handle, wire up the observable\r\n this._deviceSourceManager.onDeviceConnectedObservable.add((deviceSource) => {\r\n if (deviceSource.deviceType === DeviceType.Mouse) {\r\n deviceSource.onInputChangedObservable.add((eventData) => {\r\n if (\r\n eventData.inputIndex === PointerInput.LeftClick ||\r\n eventData.inputIndex === PointerInput.MiddleClick ||\r\n eventData.inputIndex === PointerInput.RightClick ||\r\n eventData.inputIndex === PointerInput.BrowserBack ||\r\n eventData.inputIndex === PointerInput.BrowserForward\r\n ) {\r\n if (attachDown && deviceSource.getInput(eventData.inputIndex) === 1) {\r\n this._onPointerDown(eventData);\r\n } else if (attachUp && deviceSource.getInput(eventData.inputIndex) === 0) {\r\n this._onPointerUp(eventData);\r\n }\r\n } else if (attachMove) {\r\n if (eventData.inputIndex === PointerInput.Move) {\r\n this._onPointerMove(eventData);\r\n } else if (\r\n eventData.inputIndex === PointerInput.MouseWheelX ||\r\n eventData.inputIndex === PointerInput.MouseWheelY ||\r\n eventData.inputIndex === PointerInput.MouseWheelZ\r\n ) {\r\n this._onPointerMove(eventData);\r\n }\r\n }\r\n });\r\n } else if (deviceSource.deviceType === DeviceType.Touch) {\r\n deviceSource.onInputChangedObservable.add((eventData) => {\r\n if (eventData.inputIndex === PointerInput.LeftClick) {\r\n if (attachDown && deviceSource.getInput(eventData.inputIndex) === 1) {\r\n this._onPointerDown(eventData);\r\n if (this._totalPointersPressed > 1) {\r\n this._isMultiTouchGesture = true;\r\n }\r\n } else if (attachUp && deviceSource.getInput(eventData.inputIndex) === 0) {\r\n this._onPointerUp(eventData);\r\n if (this._totalPointersPressed === 0) {\r\n this._isMultiTouchGesture = false;\r\n }\r\n }\r\n }\r\n\r\n if (attachMove && eventData.inputIndex === PointerInput.Move) {\r\n this._onPointerMove(eventData);\r\n }\r\n });\r\n } else if (deviceSource.deviceType === DeviceType.Keyboard) {\r\n deviceSource.onInputChangedObservable.add((eventData) => {\r\n if (eventData.type === \"keydown\") {\r\n this._onKeyDown(eventData);\r\n } else if (eventData.type === \"keyup\") {\r\n this._onKeyUp(eventData);\r\n }\r\n });\r\n }\r\n });\r\n\r\n this._alreadyAttached = true;\r\n }\r\n\r\n /**\r\n * Detaches all event handlers\r\n */\r\n public detachControl() {\r\n if (this._alreadyAttached) {\r\n this._deviceSourceManager!.dispose();\r\n this._deviceSourceManager = null;\r\n\r\n // Cursor\r\n if (this._alreadyAttachedTo && !this._scene.doNotHandleCursors) {\r\n this._alreadyAttachedTo.style.cursor = this._scene.defaultCursor;\r\n }\r\n\r\n this._alreadyAttached = false;\r\n this._alreadyAttachedTo = null;\r\n }\r\n }\r\n\r\n /**\r\n * Force the value of meshUnderPointer\r\n * @param mesh - defines the mesh to use\r\n * @param pointerId - optional pointer id when using more than one pointer. Defaults to 0\r\n * @param pickResult - optional pickingInfo data used to find mesh\r\n * @param evt - optional pointer event\r\n */\r\n public setPointerOverMesh(mesh: Nullable, pointerId: number = 0, pickResult?: Nullable, evt?: IPointerEvent): void {\r\n if (this._meshUnderPointerId[pointerId] === mesh && (!mesh || !mesh._internalAbstractMeshDataInfo._pointerOverDisableMeshTesting)) {\r\n return;\r\n }\r\n\r\n const underPointerMesh = this._meshUnderPointerId[pointerId];\r\n\r\n let actionManager: Nullable;\r\n if (underPointerMesh) {\r\n actionManager = underPointerMesh._getActionManagerForTrigger(Constants.ACTION_OnPointerOutTrigger);\r\n if (actionManager) {\r\n actionManager.processTrigger(Constants.ACTION_OnPointerOutTrigger, ActionEvent.CreateNew(underPointerMesh, evt, { pointerId }));\r\n }\r\n }\r\n\r\n if (mesh) {\r\n this._meshUnderPointerId[pointerId] = mesh;\r\n this._pointerOverMesh = mesh;\r\n\r\n actionManager = mesh._getActionManagerForTrigger(Constants.ACTION_OnPointerOverTrigger);\r\n if (actionManager) {\r\n actionManager.processTrigger(Constants.ACTION_OnPointerOverTrigger, ActionEvent.CreateNew(mesh, evt, { pointerId, pickResult }));\r\n }\r\n } else {\r\n delete this._meshUnderPointerId[pointerId];\r\n this._pointerOverMesh = null;\r\n }\r\n }\r\n\r\n /**\r\n * Gets the mesh under the pointer\r\n * @returns a Mesh or null if no mesh is under the pointer\r\n */\r\n public getPointerOverMesh(): Nullable {\r\n return this.meshUnderPointer;\r\n }\r\n\r\n /**\r\n * @param mesh - Mesh to invalidate\r\n * @internal\r\n */\r\n public _invalidateMesh(mesh: AbstractMesh) {\r\n if (this._pointerOverMesh === mesh) {\r\n this._pointerOverMesh = null;\r\n }\r\n if (this._pickedDownMesh === mesh) {\r\n this._pickedDownMesh = null;\r\n }\r\n if (this._pickedUpMesh === mesh) {\r\n this._pickedUpMesh = null;\r\n }\r\n for (const pointerId in this._meshUnderPointerId) {\r\n if (this._meshUnderPointerId[pointerId] === mesh) {\r\n delete this._meshUnderPointerId[pointerId];\r\n }\r\n }\r\n }\r\n}\r\n","import { PrecisionDate } from \"./precisionDate\";\r\n\r\n/**\r\n * This class is used to track a performance counter which is number based.\r\n * The user has access to many properties which give statistics of different nature.\r\n *\r\n * The implementer can track two kinds of Performance Counter: time and count.\r\n * For time you can optionally call fetchNewFrame() to notify the start of a new frame to monitor, then call beginMonitoring() to start and endMonitoring() to record the lapsed time. endMonitoring takes a newFrame parameter for you to specify if the monitored time should be set for a new frame or accumulated to the current frame being monitored.\r\n * For count you first have to call fetchNewFrame() to notify the start of a new frame to monitor, then call addCount() how many time required to increment the count value you monitor.\r\n */\r\nexport class PerfCounter {\r\n /**\r\n * Gets or sets a global boolean to turn on and off all the counters\r\n */\r\n public static Enabled = true;\r\n\r\n /**\r\n * Returns the smallest value ever\r\n */\r\n public get min(): number {\r\n return this._min;\r\n }\r\n\r\n /**\r\n * Returns the biggest value ever\r\n */\r\n public get max(): number {\r\n return this._max;\r\n }\r\n\r\n /**\r\n * Returns the average value since the performance counter is running\r\n */\r\n public get average(): number {\r\n return this._average;\r\n }\r\n\r\n /**\r\n * Returns the average value of the last second the counter was monitored\r\n */\r\n public get lastSecAverage(): number {\r\n return this._lastSecAverage;\r\n }\r\n\r\n /**\r\n * Returns the current value\r\n */\r\n public get current(): number {\r\n return this._current;\r\n }\r\n\r\n /**\r\n * Gets the accumulated total\r\n */\r\n public get total(): number {\r\n return this._totalAccumulated;\r\n }\r\n\r\n /**\r\n * Gets the total value count\r\n */\r\n public get count(): number {\r\n return this._totalValueCount;\r\n }\r\n\r\n /**\r\n * Creates a new counter\r\n */\r\n constructor() {\r\n this._startMonitoringTime = 0;\r\n this._min = 0;\r\n this._max = 0;\r\n this._average = 0;\r\n this._lastSecAverage = 0;\r\n this._current = 0;\r\n this._totalValueCount = 0;\r\n this._totalAccumulated = 0;\r\n this._lastSecAccumulated = 0;\r\n this._lastSecTime = 0;\r\n this._lastSecValueCount = 0;\r\n }\r\n\r\n /**\r\n * Call this method to start monitoring a new frame.\r\n * This scenario is typically used when you accumulate monitoring time many times for a single frame, you call this method at the start of the frame, then beginMonitoring to start recording and endMonitoring(false) to accumulated the recorded time to the PerfCounter or addCount() to accumulate a monitored count.\r\n */\r\n public fetchNewFrame() {\r\n this._totalValueCount++;\r\n this._current = 0;\r\n this._lastSecValueCount++;\r\n }\r\n\r\n /**\r\n * Call this method to monitor a count of something (e.g. mesh drawn in viewport count)\r\n * @param newCount the count value to add to the monitored count\r\n * @param fetchResult true when it's the last time in the frame you add to the counter and you wish to update the statistics properties (min/max/average), false if you only want to update statistics.\r\n */\r\n public addCount(newCount: number, fetchResult: boolean) {\r\n if (!PerfCounter.Enabled) {\r\n return;\r\n }\r\n this._current += newCount;\r\n if (fetchResult) {\r\n this._fetchResult();\r\n }\r\n }\r\n\r\n /**\r\n * Start monitoring this performance counter\r\n */\r\n public beginMonitoring() {\r\n if (!PerfCounter.Enabled) {\r\n return;\r\n }\r\n this._startMonitoringTime = PrecisionDate.Now;\r\n }\r\n\r\n /**\r\n * Compute the time lapsed since the previous beginMonitoring() call.\r\n * @param newFrame true by default to fetch the result and monitor a new frame, if false the time monitored will be added to the current frame counter\r\n */\r\n public endMonitoring(newFrame: boolean = true) {\r\n if (!PerfCounter.Enabled) {\r\n return;\r\n }\r\n\r\n if (newFrame) {\r\n this.fetchNewFrame();\r\n }\r\n\r\n const currentTime = PrecisionDate.Now;\r\n this._current = currentTime - this._startMonitoringTime;\r\n\r\n if (newFrame) {\r\n this._fetchResult();\r\n }\r\n }\r\n\r\n /**\r\n * Call this method to end the monitoring of a frame.\r\n * This scenario is typically used when you accumulate monitoring time many times for a single frame, you call this method at the end of the frame, after beginMonitoring to start recording and endMonitoring(false) to accumulated the recorded time to the PerfCounter or addCount() to accumulate a monitored count.\r\n */\r\n public endFrame() {\r\n this._fetchResult();\r\n }\r\n\r\n private _fetchResult() {\r\n this._totalAccumulated += this._current;\r\n this._lastSecAccumulated += this._current;\r\n\r\n // Min/Max update\r\n this._min = Math.min(this._min, this._current);\r\n this._max = Math.max(this._max, this._current);\r\n this._average = this._totalAccumulated / this._totalValueCount;\r\n\r\n // Reset last sec?\r\n const now = PrecisionDate.Now;\r\n if (now - this._lastSecTime > 1000) {\r\n this._lastSecAverage = this._lastSecAccumulated / this._lastSecValueCount;\r\n this._lastSecTime = now;\r\n this._lastSecAccumulated = 0;\r\n this._lastSecValueCount = 0;\r\n }\r\n }\r\n\r\n private _startMonitoringTime: number;\r\n private _min: number;\r\n private _max: number;\r\n private _average: number;\r\n private _current: number;\r\n private _totalValueCount: number;\r\n private _totalAccumulated: number;\r\n private _lastSecAverage: number;\r\n private _lastSecAccumulated: number;\r\n private _lastSecTime: number;\r\n private _lastSecValueCount: number;\r\n}\r\n","/**\r\n * Helper class used to generate session unique ID\r\n */\r\nexport class UniqueIdGenerator {\r\n // Statics\r\n private static _UniqueIdCounter = 1;\r\n\r\n /**\r\n * Gets an unique (relatively to the current scene) Id\r\n */\r\n public static get UniqueId() {\r\n const result = this._UniqueIdCounter;\r\n this._UniqueIdCounter++;\r\n return result;\r\n }\r\n}\r\n","/** Defines the cross module constantsused by lights to avoid circular dependencies */\r\nexport class LightConstants {\r\n /**\r\n * Falloff Default: light is falling off following the material specification:\r\n * standard material is using standard falloff whereas pbr material can request special falloff per materials.\r\n */\r\n public static readonly FALLOFF_DEFAULT = 0;\r\n\r\n /**\r\n * Falloff Physical: light is falling off following the inverse squared distance law.\r\n */\r\n public static readonly FALLOFF_PHYSICAL = 1;\r\n\r\n /**\r\n * Falloff gltf: light is falling off as described in the gltf moving to PBR document\r\n * to enhance interoperability with other engines.\r\n */\r\n public static readonly FALLOFF_GLTF = 2;\r\n\r\n /**\r\n * Falloff Standard: light is falling off like in the standard material\r\n * to enhance interoperability with other materials.\r\n */\r\n public static readonly FALLOFF_STANDARD = 3;\r\n\r\n //lightmapMode Consts\r\n /**\r\n * If every light affecting the material is in this lightmapMode,\r\n * material.lightmapTexture adds or multiplies\r\n * (depends on material.useLightmapAsShadowmap)\r\n * after every other light calculations.\r\n */\r\n public static readonly LIGHTMAP_DEFAULT = 0;\r\n /**\r\n * material.lightmapTexture as only diffuse lighting from this light\r\n * adds only specular lighting from this light\r\n * adds dynamic shadows\r\n */\r\n public static readonly LIGHTMAP_SPECULAR = 1;\r\n /**\r\n * material.lightmapTexture as only lighting\r\n * no light calculation from this light\r\n * only adds dynamic shadows from this light\r\n */\r\n public static readonly LIGHTMAP_SHADOWSONLY = 2;\r\n\r\n // Intensity Mode Consts\r\n /**\r\n * Each light type uses the default quantity according to its type:\r\n * point/spot lights use luminous intensity\r\n * directional lights use illuminance\r\n */\r\n public static readonly INTENSITYMODE_AUTOMATIC = 0;\r\n /**\r\n * lumen (lm)\r\n */\r\n public static readonly INTENSITYMODE_LUMINOUSPOWER = 1;\r\n /**\r\n * candela (lm/sr)\r\n */\r\n public static readonly INTENSITYMODE_LUMINOUSINTENSITY = 2;\r\n /**\r\n * lux (lm/m^2)\r\n */\r\n public static readonly INTENSITYMODE_ILLUMINANCE = 3;\r\n /**\r\n * nit (cd/m^2)\r\n */\r\n public static readonly INTENSITYMODE_LUMINANCE = 4;\r\n\r\n // Light types ids const.\r\n /**\r\n * Light type const id of the point light.\r\n */\r\n public static readonly LIGHTTYPEID_POINTLIGHT = 0;\r\n /**\r\n * Light type const id of the directional light.\r\n */\r\n public static readonly LIGHTTYPEID_DIRECTIONALLIGHT = 1;\r\n /**\r\n * Light type const id of the spot light.\r\n */\r\n public static readonly LIGHTTYPEID_SPOTLIGHT = 2;\r\n /**\r\n * Light type const id of the hemispheric light.\r\n */\r\n public static readonly LIGHTTYPEID_HEMISPHERICLIGHT = 3;\r\n\r\n /**\r\n * Sort function to order lights for rendering.\r\n * @param a First Light object to compare to second.\r\n * @param b Second Light object to compare first.\r\n * @returns -1 to reduce's a's index relative to be, 0 for no change, 1 to increase a's index relative to b.\r\n */\r\n public static CompareLightsPriority(a: ISortableLight, b: ISortableLight): number {\r\n //shadow-casting lights have priority over non-shadow-casting lights\r\n //the renderPriority is a secondary sort criterion\r\n if (a.shadowEnabled !== b.shadowEnabled) {\r\n return (b.shadowEnabled ? 1 : 0) - (a.shadowEnabled ? 1 : 0);\r\n }\r\n return b.renderPriority - a.renderPriority;\r\n }\r\n}\r\n\r\n/**\r\n * Defines the common interface of sortable lights\r\n */\r\nexport interface ISortableLight {\r\n /**\r\n * Gets or sets whether or not the shadows are enabled for this light. This can help turning off/on shadow without detaching\r\n * the current shadow generator.\r\n */\r\n shadowEnabled: boolean;\r\n /**\r\n * Defines the rendering priority of the lights. It can help in case of fallback or number of lights\r\n * exceeding the number allowed of the materials.\r\n */\r\n renderPriority: number;\r\n}\r\n","import type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\n\r\n/**\r\n * Class used to store configuration data associated with pointer picking\r\n */\r\nexport class PointerPickingConfiguration {\r\n /**\r\n * Gets or sets a predicate used to select candidate meshes for a pointer down event\r\n */\r\n public pointerDownPredicate: (Mesh: AbstractMesh) => boolean;\r\n /**\r\n * Gets or sets a predicate used to select candidate meshes for a pointer up event\r\n */\r\n public pointerUpPredicate: (Mesh: AbstractMesh) => boolean;\r\n\r\n /**\r\n * Gets or sets a predicate used to select candidate meshes for a pointer move event\r\n */\r\n public pointerMovePredicate: (Mesh: AbstractMesh) => boolean;\r\n\r\n /**\r\n * Gets or sets a predicate used to select candidate meshes for a pointer down event\r\n */\r\n public pointerDownFastCheck = false;\r\n /**\r\n * Gets or sets a predicate used to select candidate meshes for a pointer up event\r\n */\r\n public pointerUpFastCheck = false;\r\n\r\n /**\r\n * Gets or sets a predicate used to select candidate meshes for a pointer move event\r\n */\r\n public pointerMoveFastCheck = false;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if the user want to entirely skip the picking phase when a pointer move event occurs.\r\n */\r\n public skipPointerMovePicking = false;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if the user want to entirely skip the picking phase when a pointer down event occurs.\r\n */\r\n public skipPointerDownPicking = false;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if the user want to entirely skip the picking phase when a pointer up event occurs. Off by default.\r\n */\r\n public skipPointerUpPicking = false;\r\n}\r\n","/* eslint-disable @typescript-eslint/no-unused-vars */\r\nimport type { Nullable } from \"./types\";\r\nimport { Tools } from \"./Misc/tools\";\r\nimport type { IAnimatable } from \"./Animations/animatable.interface\";\r\nimport { PrecisionDate } from \"./Misc/precisionDate\";\r\nimport type { Observer } from \"./Misc/observable\";\r\nimport { Observable } from \"./Misc/observable\";\r\nimport type { ISmartArrayLike } from \"./Misc/smartArray\";\r\nimport { SmartArrayNoDuplicate, SmartArray } from \"./Misc/smartArray\";\r\nimport { StringDictionary } from \"./Misc/stringDictionary\";\r\nimport { Tags } from \"./Misc/tags\";\r\nimport type { Vector2, Vector4 } from \"./Maths/math.vector\";\r\nimport { Vector3, Matrix, TmpVectors } from \"./Maths/math.vector\";\r\nimport type { IParticleSystem } from \"./Particles/IParticleSystem\";\r\nimport { AbstractScene } from \"./abstractScene\";\r\nimport { ImageProcessingConfiguration } from \"./Materials/imageProcessingConfiguration\";\r\nimport { UniformBuffer } from \"./Materials/uniformBuffer\";\r\nimport { PickingInfo } from \"./Collisions/pickingInfo\";\r\nimport type { ICollisionCoordinator } from \"./Collisions/collisionCoordinator\";\r\nimport type { PointerEventTypes, PointerInfoPre, PointerInfo } from \"./Events/pointerEvents\";\r\nimport type { KeyboardInfoPre, KeyboardInfo } from \"./Events/keyboardEvents\";\r\nimport { ActionEvent } from \"./Actions/actionEvent\";\r\nimport { PostProcessManager } from \"./PostProcesses/postProcessManager\";\r\nimport type { IOfflineProvider } from \"./Offline/IOfflineProvider\";\r\nimport type { RenderingGroupInfo, IRenderingManagerAutoClearSetup } from \"./Rendering/renderingManager\";\r\nimport { RenderingManager } from \"./Rendering/renderingManager\";\r\nimport type {\r\n ISceneComponent,\r\n ISceneSerializableComponent,\r\n SimpleStageAction,\r\n RenderTargetsStageAction,\r\n RenderTargetStageAction,\r\n MeshStageAction,\r\n EvaluateSubMeshStageAction,\r\n PreActiveMeshStageAction,\r\n CameraStageAction,\r\n RenderingGroupStageAction,\r\n RenderingMeshStageAction,\r\n PointerMoveStageAction,\r\n PointerUpDownStageAction,\r\n CameraStageFrameBufferAction,\r\n} from \"./sceneComponent\";\r\nimport { Stage } from \"./sceneComponent\";\r\nimport type { Engine } from \"./Engines/engine\";\r\nimport { Constants } from \"./Engines/constants\";\r\nimport { IsWindowObjectExist } from \"./Misc/domManagement\";\r\nimport { EngineStore } from \"./Engines/engineStore\";\r\nimport type { AbstractActionManager } from \"./Actions/abstractActionManager\";\r\nimport { _WarnImport } from \"./Misc/devTools\";\r\nimport type { WebRequest } from \"./Misc/webRequest\";\r\nimport { InputManager } from \"./Inputs/scene.inputManager\";\r\nimport { PerfCounter } from \"./Misc/perfCounter\";\r\nimport type { IFileRequest } from \"./Misc/fileRequest\";\r\nimport { Color4, Color3 } from \"./Maths/math.color\";\r\nimport type { Plane } from \"./Maths/math.plane\";\r\nimport { Frustum } from \"./Maths/math.frustum\";\r\nimport { UniqueIdGenerator } from \"./Misc/uniqueIdGenerator\";\r\nimport type { LoadFileError, RequestFileError, ReadFileError } from \"./Misc/fileTools\";\r\nimport { ReadFile, RequestFile, LoadFile } from \"./Misc/fileTools\";\r\nimport type { IClipPlanesHolder } from \"./Misc/interfaces/iClipPlanesHolder\";\r\nimport type { IPointerEvent } from \"./Events/deviceInputEvents\";\r\nimport { LightConstants } from \"./Lights/lightConstants\";\r\nimport { _ObserveArray } from \"./Misc/arrayTools\";\r\nimport type { IAction } from \"./Actions/action\";\r\nimport type { AnimationPropertiesOverride } from \"./Animations/animationPropertiesOverride\";\r\nimport type { AnimationGroup } from \"./Animations/animationGroup\";\r\nimport type { Skeleton } from \"./Bones/skeleton\";\r\nimport type { Bone } from \"./Bones/bone\";\r\nimport type { Camera } from \"./Cameras/camera\";\r\nimport type { WebVRFreeCamera } from \"./Cameras/VR/webVRCamera\";\r\nimport type { Collider } from \"./Collisions/collider\";\r\nimport type { Ray, TrianglePickingPredicate } from \"./Culling/ray\";\r\nimport type { Light } from \"./Lights/light\";\r\nimport type { PerformanceViewerCollector } from \"./Misc/PerformanceViewer/performanceViewerCollector\";\r\nimport type { MorphTarget } from \"./Morph/morphTarget\";\r\nimport type { MorphTargetManager } from \"./Morph/morphTargetManager\";\r\nimport type { PostProcess } from \"./PostProcesses/postProcess\";\r\nimport type { Material } from \"./Materials/material\";\r\nimport type { BaseTexture } from \"./Materials/Textures/baseTexture\";\r\nimport type { Geometry } from \"./Meshes/geometry\";\r\nimport type { TransformNode } from \"./Meshes/transformNode\";\r\nimport type { AbstractMesh } from \"./Meshes/abstractMesh\";\r\nimport type { MultiMaterial } from \"./Materials/multiMaterial\";\r\nimport type { Effect } from \"./Materials/effect\";\r\nimport type { RenderTargetTexture } from \"./Materials/Textures/renderTargetTexture\";\r\nimport type { Mesh } from \"./Meshes/mesh\";\r\nimport type { SubMesh } from \"./Meshes/subMesh\";\r\nimport type { Node } from \"./node\";\r\nimport type { Animation } from \"./Animations/animation\";\r\nimport type { Animatable } from \"./Animations/animatable\";\r\nimport type { Texture } from \"./Materials/Textures/texture\";\r\nimport { PointerPickingConfiguration } from \"./Inputs/pointerPickingConfiguration\";\r\n\r\n/**\r\n * Define an interface for all classes that will hold resources\r\n */\r\nexport interface IDisposable {\r\n /**\r\n * Releases all held resources\r\n */\r\n dispose(): void;\r\n}\r\n\r\n/** Interface defining initialization parameters for Scene class */\r\nexport interface SceneOptions {\r\n /**\r\n * Defines that scene should keep up-to-date a map of geometry to enable fast look-up by uniqueId\r\n * It will improve performance when the number of geometries becomes important.\r\n */\r\n useGeometryUniqueIdsMap?: boolean;\r\n\r\n /**\r\n * Defines that each material of the scene should keep up-to-date a map of referencing meshes for fast disposing\r\n * It will improve performance when the number of mesh becomes important, but might consume a bit more memory\r\n */\r\n useMaterialMeshMap?: boolean;\r\n\r\n /**\r\n * Defines that each mesh of the scene should keep up-to-date a map of referencing cloned meshes for fast disposing\r\n * It will improve performance when the number of mesh becomes important, but might consume a bit more memory\r\n */\r\n useClonedMeshMap?: boolean;\r\n\r\n /** Defines if the creation of the scene should impact the engine (Eg. UtilityLayer's scene) */\r\n virtual?: boolean;\r\n}\r\n\r\n/**\r\n * Define how the scene should favor performance over ease of use\r\n */\r\nexport enum ScenePerformancePriority {\r\n /** Default mode. No change. Performance will be treated as less important than backward compatibility */\r\n BackwardCompatible,\r\n /** Some performance options will be turned on trying to strike a balance between perf and ease of use */\r\n Intermediate,\r\n /** Performance will be top priority */\r\n Aggressive,\r\n}\r\n\r\n/**\r\n * Represents a scene to be rendered by the engine.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene\r\n */\r\nexport class Scene extends AbstractScene implements IAnimatable, IClipPlanesHolder {\r\n /** The fog is deactivated */\r\n public static readonly FOGMODE_NONE = 0;\r\n /** The fog density is following an exponential function */\r\n public static readonly FOGMODE_EXP = 1;\r\n /** The fog density is following an exponential function faster than FOGMODE_EXP */\r\n public static readonly FOGMODE_EXP2 = 2;\r\n /** The fog density is following a linear function. */\r\n public static readonly FOGMODE_LINEAR = 3;\r\n\r\n /**\r\n * Gets or sets the minimum deltatime when deterministic lock step is enabled\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep\r\n */\r\n public static MinDeltaTime = 1.0;\r\n /**\r\n * Gets or sets the maximum deltatime when deterministic lock step is enabled\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep\r\n */\r\n public static MaxDeltaTime = 1000.0;\r\n\r\n /**\r\n * Factory used to create the default material.\r\n * @param scene The scene to create the material for\r\n * @returns The default material\r\n */\r\n public static DefaultMaterialFactory(scene: Scene): Material {\r\n throw _WarnImport(\"StandardMaterial\");\r\n }\r\n\r\n /**\r\n * Factory used to create the a collision coordinator.\r\n * @returns The collision coordinator\r\n */\r\n public static CollisionCoordinatorFactory(): ICollisionCoordinator {\r\n throw _WarnImport(\"DefaultCollisionCoordinator\");\r\n }\r\n\r\n // Members\r\n\r\n /** @internal */\r\n public _inputManager = new InputManager(this);\r\n\r\n /** Define this parameter if you are using multiple cameras and you want to specify which one should be used for pointer position */\r\n public cameraToUseForPointers: Nullable = null;\r\n\r\n /** @internal */\r\n public readonly _isScene = true;\r\n\r\n /** @internal */\r\n public _blockEntityCollection = false;\r\n\r\n /**\r\n * Gets or sets a boolean that indicates if the scene must clear the render buffer before rendering a frame\r\n */\r\n public autoClear = true;\r\n /**\r\n * Gets or sets a boolean that indicates if the scene must clear the depth and stencil buffers before rendering a frame\r\n */\r\n public autoClearDepthAndStencil = true;\r\n /**\r\n * Defines the color used to clear the render buffer (Default is (0.2, 0.2, 0.3, 1.0))\r\n */\r\n public clearColor: Color4 = new Color4(0.2, 0.2, 0.3, 1.0);\r\n /**\r\n * Defines the color used to simulate the ambient color (Default is (0, 0, 0))\r\n */\r\n public ambientColor = new Color3(0, 0, 0);\r\n\r\n /**\r\n * This is use to store the default BRDF lookup for PBR materials in your scene.\r\n * It should only be one of the following (if not the default embedded one):\r\n * * For uncorrelated BRDF (pbr.brdf.useEnergyConservation = false and pbr.brdf.useSmithVisibilityHeightCorrelated = false) : https://assets.babylonjs.com/environments/uncorrelatedBRDF.dds\r\n * * For correlated BRDF (pbr.brdf.useEnergyConservation = false and pbr.brdf.useSmithVisibilityHeightCorrelated = true) : https://assets.babylonjs.com/environments/correlatedBRDF.dds\r\n * * For correlated multi scattering BRDF (pbr.brdf.useEnergyConservation = true and pbr.brdf.useSmithVisibilityHeightCorrelated = true) : https://assets.babylonjs.com/environments/correlatedMSBRDF.dds\r\n * The material properties need to be setup according to the type of texture in use.\r\n */\r\n public environmentBRDFTexture: BaseTexture;\r\n\r\n /**\r\n * Texture used in all pbr material as the reflection texture.\r\n * As in the majority of the scene they are the same (exception for multi room and so on),\r\n * this is easier to reference from here than from all the materials.\r\n */\r\n public get environmentTexture(): Nullable {\r\n return this._environmentTexture;\r\n }\r\n /**\r\n * Texture used in all pbr material as the reflection texture.\r\n * As in the majority of the scene they are the same (exception for multi room and so on),\r\n * this is easier to set here than in all the materials.\r\n */\r\n public set environmentTexture(value: Nullable) {\r\n if (this._environmentTexture === value) {\r\n return;\r\n }\r\n\r\n this._environmentTexture = value;\r\n this.markAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n /**\r\n * Intensity of the environment in all pbr material.\r\n * This dims or reinforces the IBL lighting overall (reflection and diffuse).\r\n * As in the majority of the scene they are the same (exception for multi room and so on),\r\n * this is easier to reference from here than from all the materials.\r\n */\r\n public environmentIntensity: number = 1;\r\n\r\n /** @internal */\r\n protected _imageProcessingConfiguration: ImageProcessingConfiguration;\r\n /**\r\n * Default image processing configuration used either in the rendering\r\n * Forward main pass or through the imageProcessingPostProcess if present.\r\n * As in the majority of the scene they are the same (exception for multi camera),\r\n * this is easier to reference from here than from all the materials and post process.\r\n *\r\n * No setter as we it is a shared configuration, you can set the values instead.\r\n */\r\n public get imageProcessingConfiguration(): ImageProcessingConfiguration {\r\n return this._imageProcessingConfiguration;\r\n }\r\n\r\n private _performancePriority = ScenePerformancePriority.BackwardCompatible;\r\n\r\n /**\r\n * Observable triggered when the performance priority is changed\r\n */\r\n public onScenePerformancePriorityChangedObservable = new Observable();\r\n /**\r\n * Gets or sets a value indicating how to treat performance relatively to ease of use and backward compatibility\r\n */\r\n public get performancePriority() {\r\n return this._performancePriority;\r\n }\r\n\r\n public set performancePriority(value) {\r\n if (value === this._performancePriority) {\r\n return;\r\n }\r\n\r\n this._performancePriority = value;\r\n\r\n switch (value) {\r\n case ScenePerformancePriority.BackwardCompatible:\r\n this.skipFrustumClipping = false;\r\n this._renderingManager.maintainStateBetweenFrames = false;\r\n this.skipPointerMovePicking = false;\r\n this.autoClear = true;\r\n break;\r\n case ScenePerformancePriority.Intermediate:\r\n this.skipFrustumClipping = false;\r\n this._renderingManager.maintainStateBetweenFrames = false;\r\n this.skipPointerMovePicking = true;\r\n this.autoClear = false;\r\n break;\r\n case ScenePerformancePriority.Aggressive:\r\n this.skipFrustumClipping = true;\r\n this._renderingManager.maintainStateBetweenFrames = true;\r\n this.skipPointerMovePicking = true;\r\n this.autoClear = false;\r\n break;\r\n }\r\n\r\n this.onScenePerformancePriorityChangedObservable.notifyObservers(value);\r\n }\r\n\r\n private _forceWireframe = false;\r\n /**\r\n * Gets or sets a boolean indicating if all rendering must be done in wireframe\r\n */\r\n public set forceWireframe(value: boolean) {\r\n if (this._forceWireframe === value) {\r\n return;\r\n }\r\n this._forceWireframe = value;\r\n this.markAllMaterialsAsDirty(Constants.MATERIAL_MiscDirtyFlag);\r\n }\r\n public get forceWireframe(): boolean {\r\n return this._forceWireframe;\r\n }\r\n\r\n private _skipFrustumClipping = false;\r\n /**\r\n * Gets or sets a boolean indicating if we should skip the frustum clipping part of the active meshes selection\r\n */\r\n public set skipFrustumClipping(value: boolean) {\r\n if (this._skipFrustumClipping === value) {\r\n return;\r\n }\r\n this._skipFrustumClipping = value;\r\n }\r\n public get skipFrustumClipping(): boolean {\r\n return this._skipFrustumClipping;\r\n }\r\n\r\n private _forcePointsCloud = false;\r\n /**\r\n * Gets or sets a boolean indicating if all rendering must be done in point cloud\r\n */\r\n public set forcePointsCloud(value: boolean) {\r\n if (this._forcePointsCloud === value) {\r\n return;\r\n }\r\n this._forcePointsCloud = value;\r\n this.markAllMaterialsAsDirty(Constants.MATERIAL_MiscDirtyFlag);\r\n }\r\n public get forcePointsCloud(): boolean {\r\n return this._forcePointsCloud;\r\n }\r\n\r\n /**\r\n * Gets or sets the active clipplane 1\r\n */\r\n public clipPlane: Nullable;\r\n\r\n /**\r\n * Gets or sets the active clipplane 2\r\n */\r\n public clipPlane2: Nullable;\r\n\r\n /**\r\n * Gets or sets the active clipplane 3\r\n */\r\n public clipPlane3: Nullable;\r\n\r\n /**\r\n * Gets or sets the active clipplane 4\r\n */\r\n public clipPlane4: Nullable;\r\n\r\n /**\r\n * Gets or sets the active clipplane 5\r\n */\r\n public clipPlane5: Nullable;\r\n\r\n /**\r\n * Gets or sets the active clipplane 6\r\n */\r\n public clipPlane6: Nullable;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if animations are enabled\r\n */\r\n public animationsEnabled = true;\r\n\r\n private _animationPropertiesOverride: Nullable = null;\r\n\r\n /**\r\n * Gets or sets the animation properties override\r\n */\r\n public get animationPropertiesOverride(): Nullable {\r\n return this._animationPropertiesOverride;\r\n }\r\n\r\n public set animationPropertiesOverride(value: Nullable) {\r\n this._animationPropertiesOverride = value;\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating if a constant deltatime has to be used\r\n * This is mostly useful for testing purposes when you do not want the animations to scale with the framerate\r\n */\r\n public useConstantAnimationDeltaTime = false;\r\n /**\r\n * Gets or sets a boolean indicating if the scene must keep the meshUnderPointer property updated\r\n * Please note that it requires to run a ray cast through the scene on every frame\r\n */\r\n public constantlyUpdateMeshUnderPointer = false;\r\n\r\n /**\r\n * Defines the HTML cursor to use when hovering over interactive elements\r\n */\r\n public hoverCursor = \"pointer\";\r\n /**\r\n * Defines the HTML default cursor to use (empty by default)\r\n */\r\n public defaultCursor: string = \"\";\r\n /**\r\n * Defines whether cursors are handled by the scene.\r\n */\r\n public doNotHandleCursors = false;\r\n /**\r\n * This is used to call preventDefault() on pointer down\r\n * in order to block unwanted artifacts like system double clicks\r\n */\r\n public preventDefaultOnPointerDown = true;\r\n\r\n /**\r\n * This is used to call preventDefault() on pointer up\r\n * in order to block unwanted artifacts like system double clicks\r\n */\r\n public preventDefaultOnPointerUp = true;\r\n\r\n // Metadata\r\n /**\r\n * Gets or sets user defined metadata\r\n */\r\n public metadata: any = null;\r\n\r\n /**\r\n * For internal use only. Please do not use.\r\n */\r\n public reservedDataStore: any = null;\r\n\r\n /**\r\n * Gets the name of the plugin used to load this scene (null by default)\r\n */\r\n public loadingPluginName: string;\r\n\r\n /**\r\n * Use this array to add regular expressions used to disable offline support for specific urls\r\n */\r\n public disableOfflineSupportExceptionRules = new Array();\r\n\r\n /**\r\n * An event triggered when the scene is disposed.\r\n */\r\n public onDisposeObservable = new Observable();\r\n\r\n private _onDisposeObserver: Nullable> = null;\r\n /** Sets a function to be executed when this scene is disposed. */\r\n public set onDispose(callback: () => void) {\r\n if (this._onDisposeObserver) {\r\n this.onDisposeObservable.remove(this._onDisposeObserver);\r\n }\r\n this._onDisposeObserver = this.onDisposeObservable.add(callback);\r\n }\r\n\r\n /**\r\n * An event triggered before rendering the scene (right after animations and physics)\r\n */\r\n public onBeforeRenderObservable = new Observable();\r\n\r\n private _onBeforeRenderObserver: Nullable> = null;\r\n /** Sets a function to be executed before rendering this scene */\r\n public set beforeRender(callback: Nullable<() => void>) {\r\n if (this._onBeforeRenderObserver) {\r\n this.onBeforeRenderObservable.remove(this._onBeforeRenderObserver);\r\n }\r\n if (callback) {\r\n this._onBeforeRenderObserver = this.onBeforeRenderObservable.add(callback);\r\n }\r\n }\r\n\r\n /**\r\n * An event triggered after rendering the scene\r\n */\r\n public onAfterRenderObservable = new Observable();\r\n\r\n /**\r\n * An event triggered after rendering the scene for an active camera (When scene.render is called this will be called after each camera)\r\n * This is triggered for each \"sub\" camera in a Camera Rig unlike onAfterCameraRenderObservable\r\n */\r\n public onAfterRenderCameraObservable = new Observable();\r\n\r\n private _onAfterRenderObserver: Nullable> = null;\r\n /** Sets a function to be executed after rendering this scene */\r\n public set afterRender(callback: Nullable<() => void>) {\r\n if (this._onAfterRenderObserver) {\r\n this.onAfterRenderObservable.remove(this._onAfterRenderObserver);\r\n }\r\n\r\n if (callback) {\r\n this._onAfterRenderObserver = this.onAfterRenderObservable.add(callback);\r\n }\r\n }\r\n\r\n /**\r\n * An event triggered before animating the scene\r\n */\r\n public onBeforeAnimationsObservable = new Observable();\r\n\r\n /**\r\n * An event triggered after animations processing\r\n */\r\n public onAfterAnimationsObservable = new Observable();\r\n\r\n /**\r\n * An event triggered before draw calls are ready to be sent\r\n */\r\n public onBeforeDrawPhaseObservable = new Observable();\r\n\r\n /**\r\n * An event triggered after draw calls have been sent\r\n */\r\n public onAfterDrawPhaseObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when the scene is ready\r\n */\r\n public onReadyObservable = new Observable();\r\n\r\n /**\r\n * An event triggered before rendering a camera\r\n */\r\n public onBeforeCameraRenderObservable = new Observable();\r\n\r\n private _onBeforeCameraRenderObserver: Nullable> = null;\r\n /** Sets a function to be executed before rendering a camera*/\r\n public set beforeCameraRender(callback: () => void) {\r\n if (this._onBeforeCameraRenderObserver) {\r\n this.onBeforeCameraRenderObservable.remove(this._onBeforeCameraRenderObserver);\r\n }\r\n\r\n this._onBeforeCameraRenderObserver = this.onBeforeCameraRenderObservable.add(callback);\r\n }\r\n\r\n /**\r\n * An event triggered after rendering a camera\r\n * This is triggered for the full rig Camera only unlike onAfterRenderCameraObservable\r\n */\r\n public onAfterCameraRenderObservable = new Observable();\r\n\r\n private _onAfterCameraRenderObserver: Nullable> = null;\r\n /** Sets a function to be executed after rendering a camera*/\r\n public set afterCameraRender(callback: () => void) {\r\n if (this._onAfterCameraRenderObserver) {\r\n this.onAfterCameraRenderObservable.remove(this._onAfterCameraRenderObserver);\r\n }\r\n this._onAfterCameraRenderObserver = this.onAfterCameraRenderObservable.add(callback);\r\n }\r\n\r\n /**\r\n * An event triggered when active meshes evaluation is about to start\r\n */\r\n public onBeforeActiveMeshesEvaluationObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when active meshes evaluation is done\r\n */\r\n public onAfterActiveMeshesEvaluationObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when particles rendering is about to start\r\n * Note: This event can be trigger more than once per frame (because particles can be rendered by render target textures as well)\r\n */\r\n public onBeforeParticlesRenderingObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when particles rendering is done\r\n * Note: This event can be trigger more than once per frame (because particles can be rendered by render target textures as well)\r\n */\r\n public onAfterParticlesRenderingObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when SceneLoader.Append or SceneLoader.Load or SceneLoader.ImportMesh were successfully executed\r\n */\r\n public onDataLoadedObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when a camera is created\r\n */\r\n public onNewCameraAddedObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when a camera is removed\r\n */\r\n public onCameraRemovedObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when a light is created\r\n */\r\n public onNewLightAddedObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when a light is removed\r\n */\r\n public onLightRemovedObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when a geometry is created\r\n */\r\n public onNewGeometryAddedObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when a geometry is removed\r\n */\r\n public onGeometryRemovedObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when a transform node is created\r\n */\r\n public onNewTransformNodeAddedObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when a transform node is removed\r\n */\r\n public onTransformNodeRemovedObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when a mesh is created\r\n */\r\n public onNewMeshAddedObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when a mesh is removed\r\n */\r\n public onMeshRemovedObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when a skeleton is created\r\n */\r\n public onNewSkeletonAddedObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when a skeleton is removed\r\n */\r\n public onSkeletonRemovedObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when a material is created\r\n */\r\n public onNewMaterialAddedObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when a multi material is created\r\n */\r\n public onNewMultiMaterialAddedObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when a material is removed\r\n */\r\n public onMaterialRemovedObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when a multi material is removed\r\n */\r\n public onMultiMaterialRemovedObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when a texture is created\r\n */\r\n public onNewTextureAddedObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when a texture is removed\r\n */\r\n public onTextureRemovedObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when render targets are about to be rendered\r\n * Can happen multiple times per frame.\r\n */\r\n public onBeforeRenderTargetsRenderObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when render targets were rendered.\r\n * Can happen multiple times per frame.\r\n */\r\n public onAfterRenderTargetsRenderObservable = new Observable();\r\n\r\n /**\r\n * An event triggered before calculating deterministic simulation step\r\n */\r\n public onBeforeStepObservable = new Observable();\r\n\r\n /**\r\n * An event triggered after calculating deterministic simulation step\r\n */\r\n public onAfterStepObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when the activeCamera property is updated\r\n */\r\n public onActiveCameraChanged = new Observable();\r\n\r\n /**\r\n * An event triggered when the activeCameras property is updated\r\n */\r\n public onActiveCamerasChanged = new Observable();\r\n\r\n /**\r\n * This Observable will be triggered before rendering each renderingGroup of each rendered camera.\r\n * The RenderingGroupInfo class contains all the information about the context in which the observable is called\r\n * If you wish to register an Observer only for a given set of renderingGroup, use the mask with a combination of the renderingGroup index elevated to the power of two (1 for renderingGroup 0, 2 for renderingrOup1, 4 for 2 and 8 for 3)\r\n */\r\n public onBeforeRenderingGroupObservable = new Observable();\r\n\r\n /**\r\n * This Observable will be triggered after rendering each renderingGroup of each rendered camera.\r\n * The RenderingGroupInfo class contains all the information about the context in which the observable is called\r\n * If you wish to register an Observer only for a given set of renderingGroup, use the mask with a combination of the renderingGroup index elevated to the power of two (1 for renderingGroup 0, 2 for renderingrOup1, 4 for 2 and 8 for 3)\r\n */\r\n public onAfterRenderingGroupObservable = new Observable();\r\n\r\n /**\r\n * This Observable will when a mesh has been imported into the scene.\r\n */\r\n public onMeshImportedObservable = new Observable();\r\n\r\n /**\r\n * This Observable will when an animation file has been imported into the scene.\r\n */\r\n public onAnimationFileImportedObservable = new Observable();\r\n\r\n /**\r\n * Gets or sets a user defined funtion to select LOD from a mesh and a camera.\r\n * By default this function is undefined and Babylon.js will select LOD based on distance to camera\r\n */\r\n public customLODSelector: (mesh: AbstractMesh, camera: Camera) => Nullable;\r\n\r\n // Animations\r\n\r\n /** @internal */\r\n public _registeredForLateAnimationBindings = new SmartArrayNoDuplicate(256);\r\n\r\n // Pointers\r\n private _pointerPickingConfiguration = new PointerPickingConfiguration();\r\n\r\n /**\r\n * Gets or sets a predicate used to select candidate meshes for a pointer down event\r\n */\r\n public get pointerDownPredicate() {\r\n return this._pointerPickingConfiguration.pointerDownPredicate;\r\n }\r\n\r\n public set pointerDownPredicate(value) {\r\n this._pointerPickingConfiguration.pointerDownPredicate = value;\r\n }\r\n\r\n /**\r\n * Gets or sets a predicate used to select candidate meshes for a pointer up event\r\n */\r\n public get pointerUpPredicate() {\r\n return this._pointerPickingConfiguration.pointerUpPredicate;\r\n }\r\n\r\n public set pointerUpPredicate(value) {\r\n this._pointerPickingConfiguration.pointerUpPredicate = value;\r\n }\r\n\r\n /**\r\n * Gets or sets a predicate used to select candidate meshes for a pointer move event\r\n */\r\n public get pointerMovePredicate() {\r\n return this._pointerPickingConfiguration.pointerMovePredicate;\r\n }\r\n\r\n public set pointerMovePredicate(value) {\r\n this._pointerPickingConfiguration.pointerMovePredicate = value;\r\n }\r\n\r\n /**\r\n * Gets or sets a predicate used to select candidate meshes for a pointer down event\r\n */\r\n public get pointerDownFastCheck() {\r\n return this._pointerPickingConfiguration.pointerDownFastCheck;\r\n }\r\n\r\n public set pointerDownFastCheck(value) {\r\n this._pointerPickingConfiguration.pointerDownFastCheck = value;\r\n }\r\n\r\n /**\r\n * Gets or sets a predicate used to select candidate meshes for a pointer up event\r\n */\r\n public get pointerUpFastCheck() {\r\n return this._pointerPickingConfiguration.pointerUpFastCheck;\r\n }\r\n\r\n public set pointerUpFastCheck(value) {\r\n this._pointerPickingConfiguration.pointerUpFastCheck = value;\r\n }\r\n\r\n /**\r\n * Gets or sets a predicate used to select candidate meshes for a pointer move event\r\n */\r\n public get pointerMoveFastCheck() {\r\n return this._pointerPickingConfiguration.pointerMoveFastCheck;\r\n }\r\n\r\n public set pointerMoveFastCheck(value) {\r\n this._pointerPickingConfiguration.pointerMoveFastCheck = value;\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating if the user want to entirely skip the picking phase when a pointer move event occurs.\r\n */\r\n public get skipPointerMovePicking() {\r\n return this._pointerPickingConfiguration.skipPointerMovePicking;\r\n }\r\n\r\n public set skipPointerMovePicking(value) {\r\n this._pointerPickingConfiguration.skipPointerMovePicking = value;\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating if the user want to entirely skip the picking phase when a pointer down event occurs.\r\n */\r\n public get skipPointerDownPicking() {\r\n return this._pointerPickingConfiguration.skipPointerDownPicking;\r\n }\r\n\r\n public set skipPointerDownPicking(value) {\r\n this._pointerPickingConfiguration.skipPointerDownPicking = value;\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating if the user want to entirely skip the picking phase when a pointer up event occurs. Off by default.\r\n */\r\n public get skipPointerUpPicking() {\r\n return this._pointerPickingConfiguration.skipPointerUpPicking;\r\n }\r\n\r\n public set skipPointerUpPicking(value) {\r\n this._pointerPickingConfiguration.skipPointerUpPicking = value;\r\n }\r\n\r\n /** Callback called when a pointer move is detected */\r\n public onPointerMove?: (evt: IPointerEvent, pickInfo: PickingInfo, type: PointerEventTypes) => void;\r\n /** Callback called when a pointer down is detected */\r\n public onPointerDown?: (evt: IPointerEvent, pickInfo: PickingInfo, type: PointerEventTypes) => void;\r\n /** Callback called when a pointer up is detected */\r\n public onPointerUp?: (evt: IPointerEvent, pickInfo: Nullable, type: PointerEventTypes) => void;\r\n /** Callback called when a pointer pick is detected */\r\n public onPointerPick?: (evt: IPointerEvent, pickInfo: PickingInfo) => void;\r\n\r\n /**\r\n * Gets or sets a predicate used to select candidate faces for a pointer move event\r\n */\r\n public pointerMoveTrianglePredicate: ((p0: Vector3, p1: Vector3, p2: Vector3, ray: Ray) => boolean) | undefined;\r\n\r\n /**\r\n * This observable event is triggered when any ponter event is triggered. It is registered during Scene.attachControl() and it is called BEFORE the 3D engine process anything (mesh/sprite picking for instance).\r\n * You have the possibility to skip the process and the call to onPointerObservable by setting PointerInfoPre.skipOnPointerObservable to true\r\n */\r\n public onPrePointerObservable = new Observable();\r\n\r\n /**\r\n * Observable event triggered each time an input event is received from the rendering canvas\r\n */\r\n public onPointerObservable = new Observable();\r\n\r\n /**\r\n * Gets the pointer coordinates without any translation (ie. straight out of the pointer event)\r\n */\r\n public get unTranslatedPointer(): Vector2 {\r\n return this._inputManager.unTranslatedPointer;\r\n }\r\n\r\n /**\r\n * Gets or sets the distance in pixel that you have to move to prevent some events. Default is 10 pixels\r\n */\r\n public static get DragMovementThreshold() {\r\n return InputManager.DragMovementThreshold;\r\n }\r\n\r\n public static set DragMovementThreshold(value: number) {\r\n InputManager.DragMovementThreshold = value;\r\n }\r\n\r\n /**\r\n * Time in milliseconds to wait to raise long press events if button is still pressed. Default is 500 ms\r\n */\r\n public static get LongPressDelay() {\r\n return InputManager.LongPressDelay;\r\n }\r\n\r\n public static set LongPressDelay(value: number) {\r\n InputManager.LongPressDelay = value;\r\n }\r\n\r\n /**\r\n * Time in milliseconds to wait to raise long press events if button is still pressed. Default is 300 ms\r\n */\r\n public static get DoubleClickDelay() {\r\n return InputManager.DoubleClickDelay;\r\n }\r\n\r\n public static set DoubleClickDelay(value: number) {\r\n InputManager.DoubleClickDelay = value;\r\n }\r\n\r\n /** If you need to check double click without raising a single click at first click, enable this flag */\r\n public static get ExclusiveDoubleClickMode() {\r\n return InputManager.ExclusiveDoubleClickMode;\r\n }\r\n\r\n public static set ExclusiveDoubleClickMode(value: boolean) {\r\n InputManager.ExclusiveDoubleClickMode = value;\r\n }\r\n\r\n /**\r\n * Bind the current view position to an effect.\r\n * @param effect The effect to be bound\r\n * @param variableName name of the shader variable that will hold the eye position\r\n * @param isVector3 true to indicates that variableName is a Vector3 and not a Vector4\r\n * @returns the computed eye position\r\n */\r\n public bindEyePosition(effect: Nullable, variableName = \"vEyePosition\", isVector3 = false): Vector4 {\r\n const eyePosition = this._forcedViewPosition\r\n ? this._forcedViewPosition\r\n : this._mirroredCameraPosition\r\n ? this._mirroredCameraPosition\r\n : this.activeCamera!.globalPosition ?? (this.activeCamera as WebVRFreeCamera).devicePosition;\r\n\r\n const invertNormal = this.useRightHandedSystem === (this._mirroredCameraPosition != null);\r\n\r\n TmpVectors.Vector4[0].set(eyePosition.x, eyePosition.y, eyePosition.z, invertNormal ? -1 : 1);\r\n\r\n if (effect) {\r\n if (isVector3) {\r\n effect.setFloat3(variableName, TmpVectors.Vector4[0].x, TmpVectors.Vector4[0].y, TmpVectors.Vector4[0].z);\r\n } else {\r\n effect.setVector4(variableName, TmpVectors.Vector4[0]);\r\n }\r\n }\r\n\r\n return TmpVectors.Vector4[0];\r\n }\r\n\r\n /**\r\n * Update the scene ubo before it can be used in rendering processing\r\n * @returns the scene UniformBuffer\r\n */\r\n public finalizeSceneUbo(): UniformBuffer {\r\n const ubo = this.getSceneUniformBuffer();\r\n const eyePosition = this.bindEyePosition(null);\r\n ubo.updateFloat4(\"vEyePosition\", eyePosition.x, eyePosition.y, eyePosition.z, eyePosition.w);\r\n\r\n ubo.update();\r\n\r\n return ubo;\r\n }\r\n\r\n // Mirror\r\n /** @internal */\r\n public _mirroredCameraPosition: Nullable;\r\n\r\n // Keyboard\r\n\r\n /**\r\n * This observable event is triggered when any keyboard event si raised and registered during Scene.attachControl()\r\n * You have the possibility to skip the process and the call to onKeyboardObservable by setting KeyboardInfoPre.skipOnPointerObservable to true\r\n */\r\n public onPreKeyboardObservable = new Observable();\r\n\r\n /**\r\n * Observable event triggered each time an keyboard event is received from the hosting window\r\n */\r\n public onKeyboardObservable = new Observable();\r\n\r\n // Coordinates system\r\n\r\n private _useRightHandedSystem = false;\r\n /**\r\n * Gets or sets a boolean indicating if the scene must use right-handed coordinates system\r\n */\r\n public set useRightHandedSystem(value: boolean) {\r\n if (this._useRightHandedSystem === value) {\r\n return;\r\n }\r\n this._useRightHandedSystem = value;\r\n this.markAllMaterialsAsDirty(Constants.MATERIAL_MiscDirtyFlag);\r\n }\r\n public get useRightHandedSystem(): boolean {\r\n return this._useRightHandedSystem;\r\n }\r\n\r\n // Deterministic lockstep\r\n private _timeAccumulator: number = 0;\r\n private _currentStepId: number = 0;\r\n private _currentInternalStep: number = 0;\r\n\r\n /**\r\n * Sets the step Id used by deterministic lock step\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep\r\n * @param newStepId defines the step Id\r\n */\r\n public setStepId(newStepId: number): void {\r\n this._currentStepId = newStepId;\r\n }\r\n\r\n /**\r\n * Gets the step Id used by deterministic lock step\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep\r\n * @returns the step Id\r\n */\r\n public getStepId(): number {\r\n return this._currentStepId;\r\n }\r\n\r\n /**\r\n * Gets the internal step used by deterministic lock step\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep\r\n * @returns the internal step\r\n */\r\n public getInternalStep(): number {\r\n return this._currentInternalStep;\r\n }\r\n\r\n // Fog\r\n\r\n private _fogEnabled = true;\r\n /**\r\n * Gets or sets a boolean indicating if fog is enabled on this scene\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/environment_introduction#fog\r\n * (Default is true)\r\n */\r\n public set fogEnabled(value: boolean) {\r\n if (this._fogEnabled === value) {\r\n return;\r\n }\r\n this._fogEnabled = value;\r\n this.markAllMaterialsAsDirty(Constants.MATERIAL_MiscDirtyFlag);\r\n }\r\n public get fogEnabled(): boolean {\r\n return this._fogEnabled;\r\n }\r\n\r\n private _fogMode = Scene.FOGMODE_NONE;\r\n /**\r\n * Gets or sets the fog mode to use\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/environment_introduction#fog\r\n * | mode | value |\r\n * | --- | --- |\r\n * | FOGMODE_NONE | 0 |\r\n * | FOGMODE_EXP | 1 |\r\n * | FOGMODE_EXP2 | 2 |\r\n * | FOGMODE_LINEAR | 3 |\r\n */\r\n public set fogMode(value: number) {\r\n if (this._fogMode === value) {\r\n return;\r\n }\r\n this._fogMode = value;\r\n this.markAllMaterialsAsDirty(Constants.MATERIAL_MiscDirtyFlag);\r\n }\r\n public get fogMode(): number {\r\n return this._fogMode;\r\n }\r\n\r\n /**\r\n * Gets or sets the fog color to use\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/environment_introduction#fog\r\n * (Default is Color3(0.2, 0.2, 0.3))\r\n */\r\n public fogColor = new Color3(0.2, 0.2, 0.3);\r\n /**\r\n * Gets or sets the fog density to use\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/environment_introduction#fog\r\n * (Default is 0.1)\r\n */\r\n public fogDensity = 0.1;\r\n /**\r\n * Gets or sets the fog start distance to use\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/environment_introduction#fog\r\n * (Default is 0)\r\n */\r\n public fogStart = 0;\r\n /**\r\n * Gets or sets the fog end distance to use\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/environment_introduction#fog\r\n * (Default is 1000)\r\n */\r\n public fogEnd = 1000.0;\r\n\r\n /**\r\n * Flag indicating that the frame buffer binding is handled by another component\r\n */\r\n public get prePass(): boolean {\r\n return !!this.prePassRenderer && this.prePassRenderer.defaultRT.enabled;\r\n }\r\n\r\n /**\r\n * Flag indicating if we need to store previous matrices when rendering\r\n */\r\n public needsPreviousWorldMatrices = false;\r\n\r\n // Lights\r\n private _shadowsEnabled = true;\r\n /**\r\n * Gets or sets a boolean indicating if shadows are enabled on this scene\r\n */\r\n public set shadowsEnabled(value: boolean) {\r\n if (this._shadowsEnabled === value) {\r\n return;\r\n }\r\n this._shadowsEnabled = value;\r\n this.markAllMaterialsAsDirty(Constants.MATERIAL_LightDirtyFlag);\r\n }\r\n public get shadowsEnabled(): boolean {\r\n return this._shadowsEnabled;\r\n }\r\n\r\n private _lightsEnabled = true;\r\n /**\r\n * Gets or sets a boolean indicating if lights are enabled on this scene\r\n */\r\n public set lightsEnabled(value: boolean) {\r\n if (this._lightsEnabled === value) {\r\n return;\r\n }\r\n this._lightsEnabled = value;\r\n this.markAllMaterialsAsDirty(Constants.MATERIAL_LightDirtyFlag);\r\n }\r\n\r\n public get lightsEnabled(): boolean {\r\n return this._lightsEnabled;\r\n }\r\n\r\n private _activeCameras: Nullable;\r\n private _unObserveActiveCameras: Nullable<() => void> = null;\r\n\r\n /** All of the active cameras added to this scene. */\r\n public get activeCameras(): Nullable {\r\n return this._activeCameras;\r\n }\r\n\r\n public set activeCameras(cameras: Nullable) {\r\n if (this._unObserveActiveCameras) {\r\n this._unObserveActiveCameras();\r\n this._unObserveActiveCameras = null;\r\n }\r\n\r\n if (cameras) {\r\n this._unObserveActiveCameras = _ObserveArray(cameras, () => {\r\n this.onActiveCamerasChanged.notifyObservers(this);\r\n });\r\n }\r\n\r\n this._activeCameras = cameras;\r\n }\r\n\r\n /** @internal */\r\n public _activeCamera: Nullable;\r\n /** Gets or sets the current active camera */\r\n public get activeCamera(): Nullable {\r\n return this._activeCamera;\r\n }\r\n\r\n public set activeCamera(value: Nullable) {\r\n if (value === this._activeCamera) {\r\n return;\r\n }\r\n\r\n this._activeCamera = value;\r\n this.onActiveCameraChanged.notifyObservers(this);\r\n }\r\n\r\n private _defaultMaterial: Material;\r\n\r\n /** The default material used on meshes when no material is affected */\r\n public get defaultMaterial(): Material {\r\n if (!this._defaultMaterial) {\r\n this._defaultMaterial = Scene.DefaultMaterialFactory(this);\r\n }\r\n\r\n return this._defaultMaterial;\r\n }\r\n\r\n /** The default material used on meshes when no material is affected */\r\n public set defaultMaterial(value: Material) {\r\n this._defaultMaterial = value;\r\n }\r\n\r\n // Textures\r\n private _texturesEnabled = true;\r\n /**\r\n * Gets or sets a boolean indicating if textures are enabled on this scene\r\n */\r\n public set texturesEnabled(value: boolean) {\r\n if (this._texturesEnabled === value) {\r\n return;\r\n }\r\n this._texturesEnabled = value;\r\n this.markAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n public get texturesEnabled(): boolean {\r\n return this._texturesEnabled;\r\n }\r\n\r\n // Physics\r\n /**\r\n * Gets or sets a boolean indicating if physic engines are enabled on this scene\r\n */\r\n public physicsEnabled = true;\r\n\r\n // Particles\r\n /**\r\n * Gets or sets a boolean indicating if particles are enabled on this scene\r\n */\r\n public particlesEnabled = true;\r\n\r\n // Sprites\r\n /**\r\n * Gets or sets a boolean indicating if sprites are enabled on this scene\r\n */\r\n public spritesEnabled = true;\r\n\r\n // Skeletons\r\n private _skeletonsEnabled = true;\r\n /**\r\n * Gets or sets a boolean indicating if skeletons are enabled on this scene\r\n */\r\n public set skeletonsEnabled(value: boolean) {\r\n if (this._skeletonsEnabled === value) {\r\n return;\r\n }\r\n this._skeletonsEnabled = value;\r\n this.markAllMaterialsAsDirty(Constants.MATERIAL_AttributesDirtyFlag);\r\n }\r\n\r\n public get skeletonsEnabled(): boolean {\r\n return this._skeletonsEnabled;\r\n }\r\n\r\n // Lens flares\r\n /**\r\n * Gets or sets a boolean indicating if lens flares are enabled on this scene\r\n */\r\n public lensFlaresEnabled = true;\r\n\r\n // Collisions\r\n /**\r\n * Gets or sets a boolean indicating if collisions are enabled on this scene\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions\r\n */\r\n public collisionsEnabled = true;\r\n\r\n private _collisionCoordinator: ICollisionCoordinator;\r\n\r\n /** @internal */\r\n public get collisionCoordinator(): ICollisionCoordinator {\r\n if (!this._collisionCoordinator) {\r\n this._collisionCoordinator = Scene.CollisionCoordinatorFactory();\r\n this._collisionCoordinator.init(this);\r\n }\r\n\r\n return this._collisionCoordinator;\r\n }\r\n\r\n /**\r\n * Defines the gravity applied to this scene (used only for collisions)\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions\r\n */\r\n public gravity = new Vector3(0, -9.807, 0);\r\n\r\n // Postprocesses\r\n /**\r\n * Gets or sets a boolean indicating if postprocesses are enabled on this scene\r\n */\r\n public postProcessesEnabled = true;\r\n /**\r\n * Gets the current postprocess manager\r\n */\r\n public postProcessManager: PostProcessManager;\r\n\r\n // Customs render targets\r\n /**\r\n * Gets or sets a boolean indicating if render targets are enabled on this scene\r\n */\r\n public renderTargetsEnabled = true;\r\n /**\r\n * Gets or sets a boolean indicating if next render targets must be dumped as image for debugging purposes\r\n * We recommend not using it and instead rely on Spector.js: http://spector.babylonjs.com\r\n */\r\n public dumpNextRenderTargets = false;\r\n /**\r\n * The list of user defined render targets added to the scene\r\n */\r\n public customRenderTargets = new Array();\r\n\r\n /**\r\n * Defines if texture loading must be delayed\r\n * If true, textures will only be loaded when they need to be rendered\r\n */\r\n public useDelayedTextureLoading: boolean;\r\n\r\n /**\r\n * Gets the list of meshes imported to the scene through SceneLoader\r\n */\r\n public importedMeshesFiles = new Array();\r\n\r\n // Probes\r\n /**\r\n * Gets or sets a boolean indicating if probes are enabled on this scene\r\n */\r\n public probesEnabled = true;\r\n\r\n // Offline support\r\n /**\r\n * Gets or sets the current offline provider to use to store scene data\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimizeCached\r\n */\r\n public offlineProvider: IOfflineProvider;\r\n\r\n /**\r\n * Gets or sets the action manager associated with the scene\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions\r\n */\r\n public actionManager: AbstractActionManager;\r\n\r\n private _meshesForIntersections = new SmartArrayNoDuplicate(256);\r\n\r\n // Procedural textures\r\n /**\r\n * Gets or sets a boolean indicating if procedural textures are enabled on this scene\r\n */\r\n public proceduralTexturesEnabled = true;\r\n\r\n // Private\r\n private _engine: Engine;\r\n\r\n // Performance counters\r\n private _totalVertices = new PerfCounter();\r\n /** @internal */\r\n public _activeIndices = new PerfCounter();\r\n /** @internal */\r\n public _activeParticles = new PerfCounter();\r\n /** @internal */\r\n public _activeBones = new PerfCounter();\r\n\r\n private _animationRatio: number;\r\n\r\n /** @internal */\r\n public _animationTimeLast: number;\r\n\r\n /** @internal */\r\n public _animationTime: number = 0;\r\n\r\n /**\r\n * Gets or sets a general scale for animation speed\r\n * @see https://www.babylonjs-playground.com/#IBU2W7#3\r\n */\r\n public animationTimeScale: number = 1;\r\n\r\n /** @internal */\r\n public _cachedMaterial: Nullable;\r\n /** @internal */\r\n public _cachedEffect: Nullable;\r\n /** @internal */\r\n public _cachedVisibility: Nullable;\r\n\r\n private _renderId = 0;\r\n private _frameId = 0;\r\n private _executeWhenReadyTimeoutId: Nullable> = null;\r\n private _intermediateRendering = false;\r\n private _defaultFrameBufferCleared = false;\r\n\r\n private _viewUpdateFlag = -1;\r\n private _projectionUpdateFlag = -1;\r\n\r\n /** @internal */\r\n public _toBeDisposed = new Array>(256);\r\n private _activeRequests = new Array();\r\n\r\n /** @internal */\r\n public _pendingData = new Array();\r\n private _isDisposed = false;\r\n\r\n /**\r\n * Gets or sets a boolean indicating that all submeshes of active meshes must be rendered\r\n * Use this boolean to avoid computing frustum clipping on submeshes (This could help when you are CPU bound)\r\n */\r\n public dispatchAllSubMeshesOfActiveMeshes: boolean = false;\r\n private _activeMeshes = new SmartArray(256);\r\n private _processedMaterials = new SmartArray(256);\r\n private _renderTargets = new SmartArrayNoDuplicate(256);\r\n private _materialsRenderTargets = new SmartArrayNoDuplicate(256);\r\n /** @internal */\r\n public _activeParticleSystems = new SmartArray(256);\r\n private _activeSkeletons = new SmartArrayNoDuplicate(32);\r\n private _softwareSkinnedMeshes = new SmartArrayNoDuplicate(32);\r\n\r\n private _renderingManager: RenderingManager;\r\n\r\n /**\r\n * Gets the scene's rendering manager\r\n */\r\n public get renderingManager(): RenderingManager {\r\n return this._renderingManager;\r\n }\r\n\r\n /** @internal */\r\n public _activeAnimatables = new Array();\r\n\r\n private _transformMatrix = Matrix.Zero();\r\n private _sceneUbo: UniformBuffer;\r\n\r\n /** @internal */\r\n public _viewMatrix: Matrix;\r\n /** @internal */\r\n public _projectionMatrix: Matrix;\r\n /** @internal */\r\n public _forcedViewPosition: Nullable;\r\n\r\n /** @internal */\r\n public _frustumPlanes: Plane[];\r\n /**\r\n * Gets the list of frustum planes (built from the active camera)\r\n */\r\n public get frustumPlanes(): Plane[] {\r\n return this._frustumPlanes;\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating if lights must be sorted by priority (off by default)\r\n * This is useful if there are more lights that the maximum simulteanous authorized\r\n */\r\n public requireLightSorting = false;\r\n\r\n /** @internal */\r\n public readonly useMaterialMeshMap: boolean;\r\n /** @internal */\r\n public readonly useClonedMeshMap: boolean;\r\n\r\n private _externalData: StringDictionary;\r\n private _uid: Nullable;\r\n\r\n /**\r\n * @internal\r\n * Backing store of defined scene components.\r\n */\r\n public _components: ISceneComponent[] = [];\r\n\r\n /**\r\n * @internal\r\n * Backing store of defined scene components.\r\n */\r\n public _serializableComponents: ISceneSerializableComponent[] = [];\r\n\r\n /**\r\n * List of components to register on the next registration step.\r\n */\r\n private _transientComponents: ISceneComponent[] = [];\r\n\r\n /**\r\n * Registers the transient components if needed.\r\n */\r\n private _registerTransientComponents(): void {\r\n // Register components that have been associated lately to the scene.\r\n if (this._transientComponents.length > 0) {\r\n for (const component of this._transientComponents) {\r\n component.register();\r\n }\r\n this._transientComponents.length = 0;\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n * Add a component to the scene.\r\n * Note that the ccomponent could be registered on th next frame if this is called after\r\n * the register component stage.\r\n * @param component Defines the component to add to the scene\r\n */\r\n public _addComponent(component: ISceneComponent) {\r\n this._components.push(component);\r\n this._transientComponents.push(component);\r\n\r\n const serializableComponent = component as any;\r\n if (serializableComponent.addFromContainer && serializableComponent.serialize) {\r\n this._serializableComponents.push(serializableComponent);\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n * Gets a component from the scene.\r\n * @param name defines the name of the component to retrieve\r\n * @returns the component or null if not present\r\n */\r\n public _getComponent(name: string): Nullable {\r\n for (const component of this._components) {\r\n if (component.name === name) {\r\n return component;\r\n }\r\n }\r\n return null;\r\n }\r\n\r\n /**\r\n * @internal\r\n * Defines the actions happening before camera updates.\r\n */\r\n public _beforeCameraUpdateStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening before clear the canvas.\r\n */\r\n public _beforeClearStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening before clear the canvas.\r\n */\r\n public _beforeRenderTargetClearStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions when collecting render targets for the frame.\r\n */\r\n public _gatherRenderTargetsStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening for one camera in the frame.\r\n */\r\n public _gatherActiveCameraRenderTargetsStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening during the per mesh ready checks.\r\n */\r\n public _isReadyForMeshStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening before evaluate active mesh checks.\r\n */\r\n public _beforeEvaluateActiveMeshStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening during the evaluate sub mesh checks.\r\n */\r\n public _evaluateSubMeshStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening during the active mesh stage.\r\n */\r\n public _preActiveMeshStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening during the per camera render target step.\r\n */\r\n public _cameraDrawRenderTargetStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening just before the active camera is drawing.\r\n */\r\n public _beforeCameraDrawStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening just before a render target is drawing.\r\n */\r\n public _beforeRenderTargetDrawStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening just before a rendering group is drawing.\r\n */\r\n public _beforeRenderingGroupDrawStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening just before a mesh is drawing.\r\n */\r\n public _beforeRenderingMeshStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening just after a mesh has been drawn.\r\n */\r\n public _afterRenderingMeshStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening just after a rendering group has been drawn.\r\n */\r\n public _afterRenderingGroupDrawStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening just after the active camera has been drawn.\r\n */\r\n public _afterCameraDrawStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening just after the post processing\r\n */\r\n public _afterCameraPostProcessStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening just after a render target has been drawn.\r\n */\r\n public _afterRenderTargetDrawStage = Stage.Create();\r\n /**\r\n * Defines the actions happening just after the post processing on a render target\r\n */\r\n public _afterRenderTargetPostProcessStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening just after rendering all cameras and computing intersections.\r\n */\r\n public _afterRenderStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening when a pointer move event happens.\r\n */\r\n public _pointerMoveStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening when a pointer down event happens.\r\n */\r\n public _pointerDownStage = Stage.Create();\r\n /**\r\n * @internal\r\n * Defines the actions happening when a pointer up event happens.\r\n */\r\n public _pointerUpStage = Stage.Create();\r\n\r\n /**\r\n * an optional map from Geometry Id to Geometry index in the 'geometries' array\r\n */\r\n private _geometriesByUniqueId: Nullable<{ [uniqueId: string]: number | undefined }> = null;\r\n\r\n /**\r\n * Creates a new Scene\r\n * @param engine defines the engine to use to render this scene\r\n * @param options defines the scene options\r\n */\r\n constructor(engine: Engine, options?: SceneOptions) {\r\n super();\r\n\r\n this.activeCameras = new Array();\r\n\r\n const fullOptions = {\r\n useGeometryUniqueIdsMap: true,\r\n useMaterialMeshMap: true,\r\n useClonedMeshMap: true,\r\n virtual: false,\r\n ...options,\r\n };\r\n\r\n this._engine = engine || EngineStore.LastCreatedEngine;\r\n if (!fullOptions.virtual) {\r\n EngineStore._LastCreatedScene = this;\r\n this._engine.scenes.push(this);\r\n } else {\r\n this._engine._virtualScenes.push(this);\r\n }\r\n\r\n this._uid = null;\r\n\r\n this._renderingManager = new RenderingManager(this);\r\n\r\n if (PostProcessManager) {\r\n this.postProcessManager = new PostProcessManager(this);\r\n }\r\n\r\n if (IsWindowObjectExist()) {\r\n this.attachControl();\r\n }\r\n\r\n // Uniform Buffer\r\n this._createUbo();\r\n\r\n // Default Image processing definition\r\n if (ImageProcessingConfiguration) {\r\n this._imageProcessingConfiguration = new ImageProcessingConfiguration();\r\n }\r\n\r\n this.setDefaultCandidateProviders();\r\n\r\n if (fullOptions.useGeometryUniqueIdsMap) {\r\n this._geometriesByUniqueId = {};\r\n }\r\n\r\n this.useMaterialMeshMap = fullOptions.useMaterialMeshMap;\r\n this.useClonedMeshMap = fullOptions.useClonedMeshMap;\r\n\r\n if (!options || !options.virtual) {\r\n this._engine.onNewSceneAddedObservable.notifyObservers(this);\r\n }\r\n }\r\n\r\n /**\r\n * Gets a string identifying the name of the class\r\n * @returns \"Scene\" string\r\n */\r\n public getClassName(): string {\r\n return \"Scene\";\r\n }\r\n\r\n private _defaultMeshCandidates: ISmartArrayLike = {\r\n data: [],\r\n length: 0,\r\n };\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getDefaultMeshCandidates(): ISmartArrayLike {\r\n this._defaultMeshCandidates.data = this.meshes;\r\n this._defaultMeshCandidates.length = this.meshes.length;\r\n return this._defaultMeshCandidates;\r\n }\r\n\r\n private _defaultSubMeshCandidates: ISmartArrayLike = {\r\n data: [],\r\n length: 0,\r\n };\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getDefaultSubMeshCandidates(mesh: AbstractMesh): ISmartArrayLike {\r\n this._defaultSubMeshCandidates.data = mesh.subMeshes;\r\n this._defaultSubMeshCandidates.length = mesh.subMeshes.length;\r\n return this._defaultSubMeshCandidates;\r\n }\r\n\r\n /**\r\n * Sets the default candidate providers for the scene.\r\n * This sets the getActiveMeshCandidates, getActiveSubMeshCandidates, getIntersectingSubMeshCandidates\r\n * and getCollidingSubMeshCandidates to their default function\r\n */\r\n public setDefaultCandidateProviders(): void {\r\n this.getActiveMeshCandidates = this._getDefaultMeshCandidates.bind(this);\r\n\r\n this.getActiveSubMeshCandidates = this._getDefaultSubMeshCandidates.bind(this);\r\n this.getIntersectingSubMeshCandidates = this._getDefaultSubMeshCandidates.bind(this);\r\n this.getCollidingSubMeshCandidates = this._getDefaultSubMeshCandidates.bind(this);\r\n }\r\n\r\n /**\r\n * Gets the mesh that is currently under the pointer\r\n */\r\n public get meshUnderPointer(): Nullable {\r\n return this._inputManager.meshUnderPointer;\r\n }\r\n\r\n /**\r\n * Gets or sets the current on-screen X position of the pointer\r\n */\r\n public get pointerX(): number {\r\n return this._inputManager.pointerX;\r\n }\r\n\r\n public set pointerX(value: number) {\r\n this._inputManager.pointerX = value;\r\n }\r\n\r\n /**\r\n * Gets or sets the current on-screen Y position of the pointer\r\n */\r\n public get pointerY(): number {\r\n return this._inputManager.pointerY;\r\n }\r\n\r\n public set pointerY(value: number) {\r\n this._inputManager.pointerY = value;\r\n }\r\n\r\n /**\r\n * Gets the cached material (ie. the latest rendered one)\r\n * @returns the cached material\r\n */\r\n public getCachedMaterial(): Nullable {\r\n return this._cachedMaterial;\r\n }\r\n\r\n /**\r\n * Gets the cached effect (ie. the latest rendered one)\r\n * @returns the cached effect\r\n */\r\n public getCachedEffect(): Nullable {\r\n return this._cachedEffect;\r\n }\r\n\r\n /**\r\n * Gets the cached visibility state (ie. the latest rendered one)\r\n * @returns the cached visibility state\r\n */\r\n public getCachedVisibility(): Nullable {\r\n return this._cachedVisibility;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if the current material / effect / visibility must be bind again\r\n * @param material defines the current material\r\n * @param effect defines the current effect\r\n * @param visibility defines the current visibility state\r\n * @returns true if one parameter is not cached\r\n */\r\n public isCachedMaterialInvalid(material: Material, effect: Effect, visibility: number = 1) {\r\n return this._cachedEffect !== effect || this._cachedMaterial !== material || this._cachedVisibility !== visibility;\r\n }\r\n\r\n /**\r\n * Gets the engine associated with the scene\r\n * @returns an Engine\r\n */\r\n public getEngine(): Engine {\r\n return this._engine;\r\n }\r\n\r\n /**\r\n * Gets the total number of vertices rendered per frame\r\n * @returns the total number of vertices rendered per frame\r\n */\r\n public getTotalVertices(): number {\r\n return this._totalVertices.current;\r\n }\r\n\r\n /**\r\n * Gets the performance counter for total vertices\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimize_your_scene#instrumentation\r\n */\r\n public get totalVerticesPerfCounter(): PerfCounter {\r\n return this._totalVertices;\r\n }\r\n\r\n /**\r\n * Gets the total number of active indices rendered per frame (You can deduce the number of rendered triangles by dividing this number by 3)\r\n * @returns the total number of active indices rendered per frame\r\n */\r\n public getActiveIndices(): number {\r\n return this._activeIndices.current;\r\n }\r\n\r\n /**\r\n * Gets the performance counter for active indices\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimize_your_scene#instrumentation\r\n */\r\n public get totalActiveIndicesPerfCounter(): PerfCounter {\r\n return this._activeIndices;\r\n }\r\n\r\n /**\r\n * Gets the total number of active particles rendered per frame\r\n * @returns the total number of active particles rendered per frame\r\n */\r\n public getActiveParticles(): number {\r\n return this._activeParticles.current;\r\n }\r\n\r\n /**\r\n * Gets the performance counter for active particles\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimize_your_scene#instrumentation\r\n */\r\n public get activeParticlesPerfCounter(): PerfCounter {\r\n return this._activeParticles;\r\n }\r\n\r\n /**\r\n * Gets the total number of active bones rendered per frame\r\n * @returns the total number of active bones rendered per frame\r\n */\r\n public getActiveBones(): number {\r\n return this._activeBones.current;\r\n }\r\n\r\n /**\r\n * Gets the performance counter for active bones\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimize_your_scene#instrumentation\r\n */\r\n public get activeBonesPerfCounter(): PerfCounter {\r\n return this._activeBones;\r\n }\r\n\r\n /**\r\n * Gets the array of active meshes\r\n * @returns an array of AbstractMesh\r\n */\r\n public getActiveMeshes(): SmartArray {\r\n return this._activeMeshes;\r\n }\r\n\r\n /**\r\n * Gets the animation ratio (which is 1.0 is the scene renders at 60fps and 2 if the scene renders at 30fps, etc.)\r\n * @returns a number\r\n */\r\n public getAnimationRatio(): number {\r\n return this._animationRatio !== undefined ? this._animationRatio : 1;\r\n }\r\n\r\n /**\r\n * Gets an unique Id for the current render phase\r\n * @returns a number\r\n */\r\n public getRenderId(): number {\r\n return this._renderId;\r\n }\r\n\r\n /**\r\n * Gets an unique Id for the current frame\r\n * @returns a number\r\n */\r\n public getFrameId(): number {\r\n return this._frameId;\r\n }\r\n\r\n /** Call this function if you want to manually increment the render Id*/\r\n public incrementRenderId(): void {\r\n this._renderId++;\r\n }\r\n\r\n private _createUbo(): void {\r\n this.setSceneUniformBuffer(this.createSceneUniformBuffer());\r\n }\r\n\r\n /**\r\n * Use this method to simulate a pointer move on a mesh\r\n * The pickResult parameter can be obtained from a scene.pick or scene.pickWithRay\r\n * @param pickResult pickingInfo of the object wished to simulate pointer event on\r\n * @param pointerEventInit pointer event state to be used when simulating the pointer event (eg. pointer id for multitouch)\r\n * @returns the current scene\r\n */\r\n public simulatePointerMove(pickResult: PickingInfo, pointerEventInit?: PointerEventInit): Scene {\r\n this._inputManager.simulatePointerMove(pickResult, pointerEventInit);\r\n return this;\r\n }\r\n\r\n /**\r\n * Use this method to simulate a pointer down on a mesh\r\n * The pickResult parameter can be obtained from a scene.pick or scene.pickWithRay\r\n * @param pickResult pickingInfo of the object wished to simulate pointer event on\r\n * @param pointerEventInit pointer event state to be used when simulating the pointer event (eg. pointer id for multitouch)\r\n * @returns the current scene\r\n */\r\n public simulatePointerDown(pickResult: PickingInfo, pointerEventInit?: PointerEventInit): Scene {\r\n this._inputManager.simulatePointerDown(pickResult, pointerEventInit);\r\n return this;\r\n }\r\n\r\n /**\r\n * Use this method to simulate a pointer up on a mesh\r\n * The pickResult parameter can be obtained from a scene.pick or scene.pickWithRay\r\n * @param pickResult pickingInfo of the object wished to simulate pointer event on\r\n * @param pointerEventInit pointer event state to be used when simulating the pointer event (eg. pointer id for multitouch)\r\n * @param doubleTap indicates that the pointer up event should be considered as part of a double click (false by default)\r\n * @returns the current scene\r\n */\r\n public simulatePointerUp(pickResult: PickingInfo, pointerEventInit?: PointerEventInit, doubleTap?: boolean): Scene {\r\n this._inputManager.simulatePointerUp(pickResult, pointerEventInit, doubleTap);\r\n return this;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if the current pointer event is captured (meaning that the scene has already handled the pointer down)\r\n * @param pointerId defines the pointer id to use in a multi-touch scenario (0 by default)\r\n * @returns true if the pointer was captured\r\n */\r\n public isPointerCaptured(pointerId = 0): boolean {\r\n return this._inputManager.isPointerCaptured(pointerId);\r\n }\r\n\r\n /**\r\n * Attach events to the canvas (To handle actionManagers triggers and raise onPointerMove, onPointerDown and onPointerUp\r\n * @param attachUp defines if you want to attach events to pointerup\r\n * @param attachDown defines if you want to attach events to pointerdown\r\n * @param attachMove defines if you want to attach events to pointermove\r\n */\r\n public attachControl(attachUp = true, attachDown = true, attachMove = true): void {\r\n this._inputManager.attachControl(attachUp, attachDown, attachMove);\r\n }\r\n\r\n /** Detaches all event handlers*/\r\n public detachControl() {\r\n this._inputManager.detachControl();\r\n }\r\n\r\n /**\r\n * This function will check if the scene can be rendered (textures are loaded, shaders are compiled)\r\n * Delay loaded resources are not taking in account\r\n * @param checkRenderTargets true to also check that the meshes rendered as part of a render target are ready (default: true)\r\n * @returns true if all required resources are ready\r\n */\r\n public isReady(checkRenderTargets = true): boolean {\r\n if (this._isDisposed) {\r\n return false;\r\n }\r\n\r\n let index: number;\r\n const engine = this.getEngine();\r\n\r\n const currentRenderPassId = engine.currentRenderPassId;\r\n\r\n engine.currentRenderPassId = this.activeCamera?.renderPassId ?? currentRenderPassId;\r\n\r\n let isReady = true;\r\n\r\n // Pending data\r\n if (this._pendingData.length > 0) {\r\n isReady = false;\r\n }\r\n\r\n // Ensures that the pre-pass renderer is enabled if it is to be enabled.\r\n this.prePassRenderer?.update();\r\n\r\n // Meshes\r\n if (checkRenderTargets) {\r\n this._processedMaterials.reset();\r\n this._materialsRenderTargets.reset();\r\n }\r\n\r\n for (index = 0; index < this.meshes.length; index++) {\r\n const mesh = this.meshes[index];\r\n\r\n if (!mesh.subMeshes || mesh.subMeshes.length === 0) {\r\n continue;\r\n }\r\n\r\n // Do not stop at the first encountered \"unready\" object as we want to ensure\r\n // all materials are starting off their compilation in parallel.\r\n if (!mesh.isReady(true)) {\r\n isReady = false;\r\n continue;\r\n }\r\n\r\n const hardwareInstancedRendering =\r\n mesh.hasThinInstances ||\r\n mesh.getClassName() === \"InstancedMesh\" ||\r\n mesh.getClassName() === \"InstancedLinesMesh\" ||\r\n (engine.getCaps().instancedArrays && (mesh).instances.length > 0);\r\n // Is Ready For Mesh\r\n for (const step of this._isReadyForMeshStage) {\r\n if (!step.action(mesh, hardwareInstancedRendering)) {\r\n isReady = false;\r\n }\r\n }\r\n\r\n if (!checkRenderTargets) {\r\n continue;\r\n }\r\n\r\n const mat = mesh.material || this.defaultMaterial;\r\n if (mat) {\r\n if (mat._storeEffectOnSubMeshes) {\r\n for (const subMesh of mesh.subMeshes) {\r\n const material = subMesh.getMaterial();\r\n if (material && material.hasRenderTargetTextures && material.getRenderTargetTextures != null) {\r\n if (this._processedMaterials.indexOf(material) === -1) {\r\n this._processedMaterials.push(material);\r\n\r\n this._materialsRenderTargets.concatWithNoDuplicate(material.getRenderTargetTextures!());\r\n }\r\n }\r\n }\r\n } else {\r\n if (mat.hasRenderTargetTextures && mat.getRenderTargetTextures != null) {\r\n if (this._processedMaterials.indexOf(mat) === -1) {\r\n this._processedMaterials.push(mat);\r\n\r\n this._materialsRenderTargets.concatWithNoDuplicate(mat.getRenderTargetTextures!());\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Render targets\r\n if (checkRenderTargets) {\r\n for (index = 0; index < this._materialsRenderTargets.length; ++index) {\r\n const rtt = this._materialsRenderTargets.data[index];\r\n if (!rtt.isReadyForRendering()) {\r\n isReady = false;\r\n }\r\n }\r\n }\r\n\r\n // Geometries\r\n for (index = 0; index < this.geometries.length; index++) {\r\n const geometry = this.geometries[index];\r\n\r\n if (geometry.delayLoadState === Constants.DELAYLOADSTATE_LOADING) {\r\n isReady = false;\r\n }\r\n }\r\n\r\n // Post-processes\r\n if (this.activeCameras && this.activeCameras.length > 0) {\r\n for (const camera of this.activeCameras) {\r\n if (!camera.isReady(true)) {\r\n isReady = false;\r\n }\r\n }\r\n } else if (this.activeCamera) {\r\n if (!this.activeCamera.isReady(true)) {\r\n isReady = false;\r\n }\r\n }\r\n\r\n // Particles\r\n for (const particleSystem of this.particleSystems) {\r\n if (!particleSystem.isReady()) {\r\n isReady = false;\r\n }\r\n }\r\n\r\n // Layers\r\n if (this.layers) {\r\n for (const layer of this.layers) {\r\n if (!layer.isReady()) {\r\n isReady = false;\r\n }\r\n }\r\n }\r\n\r\n // Effects\r\n if (!engine.areAllEffectsReady()) {\r\n isReady = false;\r\n }\r\n\r\n engine.currentRenderPassId = currentRenderPassId;\r\n\r\n return isReady;\r\n }\r\n\r\n /** Resets all cached information relative to material (including effect and visibility) */\r\n public resetCachedMaterial(): void {\r\n this._cachedMaterial = null;\r\n this._cachedEffect = null;\r\n this._cachedVisibility = null;\r\n }\r\n\r\n /**\r\n * Registers a function to be called before every frame render\r\n * @param func defines the function to register\r\n */\r\n public registerBeforeRender(func: () => void): void {\r\n this.onBeforeRenderObservable.add(func);\r\n }\r\n\r\n /**\r\n * Unregisters a function called before every frame render\r\n * @param func defines the function to unregister\r\n */\r\n public unregisterBeforeRender(func: () => void): void {\r\n this.onBeforeRenderObservable.removeCallback(func);\r\n }\r\n\r\n /**\r\n * Registers a function to be called after every frame render\r\n * @param func defines the function to register\r\n */\r\n public registerAfterRender(func: () => void): void {\r\n this.onAfterRenderObservable.add(func);\r\n }\r\n\r\n /**\r\n * Unregisters a function called after every frame render\r\n * @param func defines the function to unregister\r\n */\r\n public unregisterAfterRender(func: () => void): void {\r\n this.onAfterRenderObservable.removeCallback(func);\r\n }\r\n\r\n private _executeOnceBeforeRender(func: () => void): void {\r\n const execFunc = () => {\r\n func();\r\n setTimeout(() => {\r\n this.unregisterBeforeRender(execFunc);\r\n });\r\n };\r\n this.registerBeforeRender(execFunc);\r\n }\r\n\r\n /**\r\n * The provided function will run before render once and will be disposed afterwards.\r\n * A timeout delay can be provided so that the function will be executed in N ms.\r\n * The timeout is using the browser's native setTimeout so time percision cannot be guaranteed.\r\n * @param func The function to be executed.\r\n * @param timeout optional delay in ms\r\n */\r\n public executeOnceBeforeRender(func: () => void, timeout?: number): void {\r\n if (timeout !== undefined) {\r\n setTimeout(() => {\r\n this._executeOnceBeforeRender(func);\r\n }, timeout);\r\n } else {\r\n this._executeOnceBeforeRender(func);\r\n }\r\n }\r\n\r\n /**\r\n * This function can help adding any object to the list of data awaited to be ready in order to check for a complete scene loading.\r\n * @param data defines the object to wait for\r\n */\r\n public addPendingData(data: any): void {\r\n this._pendingData.push(data);\r\n }\r\n\r\n /**\r\n * Remove a pending data from the loading list which has previously been added with addPendingData.\r\n * @param data defines the object to remove from the pending list\r\n */\r\n public removePendingData(data: any): void {\r\n const wasLoading = this.isLoading;\r\n const index = this._pendingData.indexOf(data);\r\n\r\n if (index !== -1) {\r\n this._pendingData.splice(index, 1);\r\n }\r\n\r\n if (wasLoading && !this.isLoading) {\r\n this.onDataLoadedObservable.notifyObservers(this);\r\n }\r\n }\r\n\r\n /**\r\n * Returns the number of items waiting to be loaded\r\n * @returns the number of items waiting to be loaded\r\n */\r\n public getWaitingItemsCount(): number {\r\n return this._pendingData.length;\r\n }\r\n\r\n /**\r\n * Returns a boolean indicating if the scene is still loading data\r\n */\r\n public get isLoading(): boolean {\r\n return this._pendingData.length > 0;\r\n }\r\n\r\n /**\r\n * Registers a function to be executed when the scene is ready\r\n * @param func - the function to be executed\r\n * @param checkRenderTargets true to also check that the meshes rendered as part of a render target are ready (default: false)\r\n */\r\n public executeWhenReady(func: () => void, checkRenderTargets = false): void {\r\n this.onReadyObservable.addOnce(func);\r\n\r\n if (this._executeWhenReadyTimeoutId !== null) {\r\n return;\r\n }\r\n\r\n this._checkIsReady(checkRenderTargets);\r\n }\r\n\r\n /**\r\n * Returns a promise that resolves when the scene is ready\r\n * @param checkRenderTargets true to also check that the meshes rendered as part of a render target are ready (default: false)\r\n * @returns A promise that resolves when the scene is ready\r\n */\r\n public whenReadyAsync(checkRenderTargets = false): Promise {\r\n return new Promise((resolve) => {\r\n this.executeWhenReady(() => {\r\n resolve();\r\n }, checkRenderTargets);\r\n });\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _checkIsReady(checkRenderTargets = false) {\r\n this._registerTransientComponents();\r\n\r\n if (this.isReady(checkRenderTargets)) {\r\n this.onReadyObservable.notifyObservers(this);\r\n\r\n this.onReadyObservable.clear();\r\n this._executeWhenReadyTimeoutId = null;\r\n return;\r\n }\r\n\r\n if (this._isDisposed) {\r\n this.onReadyObservable.clear();\r\n this._executeWhenReadyTimeoutId = null;\r\n return;\r\n }\r\n\r\n this._executeWhenReadyTimeoutId = setTimeout(() => {\r\n // Ensure materials effects are checked outside render loops\r\n this.incrementRenderId();\r\n this._checkIsReady(checkRenderTargets);\r\n }, 100);\r\n }\r\n\r\n /**\r\n * Gets all animatable attached to the scene\r\n */\r\n public get animatables(): Animatable[] {\r\n return this._activeAnimatables;\r\n }\r\n\r\n /**\r\n * Resets the last animation time frame.\r\n * Useful to override when animations start running when loading a scene for the first time.\r\n */\r\n public resetLastAnimationTimeFrame(): void {\r\n this._animationTimeLast = PrecisionDate.Now;\r\n }\r\n\r\n // Matrix\r\n\r\n /**\r\n * Gets the current view matrix\r\n * @returns a Matrix\r\n */\r\n public getViewMatrix(): Matrix {\r\n return this._viewMatrix;\r\n }\r\n\r\n /**\r\n * Gets the current projection matrix\r\n * @returns a Matrix\r\n */\r\n public getProjectionMatrix(): Matrix {\r\n return this._projectionMatrix;\r\n }\r\n\r\n /**\r\n * Gets the current transform matrix\r\n * @returns a Matrix made of View * Projection\r\n */\r\n public getTransformMatrix(): Matrix {\r\n return this._transformMatrix;\r\n }\r\n\r\n /**\r\n * Sets the current transform matrix\r\n * @param viewL defines the View matrix to use\r\n * @param projectionL defines the Projection matrix to use\r\n * @param viewR defines the right View matrix to use (if provided)\r\n * @param projectionR defines the right Projection matrix to use (if provided)\r\n */\r\n public setTransformMatrix(viewL: Matrix, projectionL: Matrix, viewR?: Matrix, projectionR?: Matrix): void {\r\n // clear the multiviewSceneUbo if no viewR and projectionR are defined\r\n if (!viewR && !projectionR && this._multiviewSceneUbo) {\r\n this._multiviewSceneUbo.dispose();\r\n this._multiviewSceneUbo = null;\r\n }\r\n if (this._viewUpdateFlag === viewL.updateFlag && this._projectionUpdateFlag === projectionL.updateFlag) {\r\n return;\r\n }\r\n\r\n this._viewUpdateFlag = viewL.updateFlag;\r\n this._projectionUpdateFlag = projectionL.updateFlag;\r\n this._viewMatrix = viewL;\r\n this._projectionMatrix = projectionL;\r\n\r\n this._viewMatrix.multiplyToRef(this._projectionMatrix, this._transformMatrix);\r\n\r\n // Update frustum\r\n if (!this._frustumPlanes) {\r\n this._frustumPlanes = Frustum.GetPlanes(this._transformMatrix);\r\n } else {\r\n Frustum.GetPlanesToRef(this._transformMatrix, this._frustumPlanes);\r\n }\r\n\r\n if (this._multiviewSceneUbo && this._multiviewSceneUbo.useUbo) {\r\n this._updateMultiviewUbo(viewR, projectionR);\r\n } else if (this._sceneUbo.useUbo) {\r\n this._sceneUbo.updateMatrix(\"viewProjection\", this._transformMatrix);\r\n this._sceneUbo.updateMatrix(\"view\", this._viewMatrix);\r\n this._sceneUbo.updateMatrix(\"projection\", this._projectionMatrix);\r\n }\r\n }\r\n\r\n /**\r\n * Gets the uniform buffer used to store scene data\r\n * @returns a UniformBuffer\r\n */\r\n public getSceneUniformBuffer(): UniformBuffer {\r\n return this._multiviewSceneUbo ? this._multiviewSceneUbo : this._sceneUbo;\r\n }\r\n\r\n /**\r\n * Creates a scene UBO\r\n * @param name name of the uniform buffer (optional, for debugging purpose only)\r\n * @returns a new ubo\r\n */\r\n public createSceneUniformBuffer(name?: string): UniformBuffer {\r\n const sceneUbo = new UniformBuffer(this._engine, undefined, false, name ?? \"scene\");\r\n sceneUbo.addUniform(\"viewProjection\", 16);\r\n sceneUbo.addUniform(\"view\", 16);\r\n sceneUbo.addUniform(\"projection\", 16);\r\n sceneUbo.addUniform(\"vEyePosition\", 4);\r\n\r\n return sceneUbo;\r\n }\r\n\r\n /**\r\n * Sets the scene ubo\r\n * @param ubo the ubo to set for the scene\r\n */\r\n public setSceneUniformBuffer(ubo: UniformBuffer): void {\r\n this._sceneUbo = ubo;\r\n this._viewUpdateFlag = -1;\r\n this._projectionUpdateFlag = -1;\r\n }\r\n\r\n /**\r\n * Gets an unique (relatively to the current scene) Id\r\n * @returns an unique number for the scene\r\n */\r\n public getUniqueId() {\r\n return UniqueIdGenerator.UniqueId;\r\n }\r\n\r\n /**\r\n * Add a mesh to the list of scene's meshes\r\n * @param newMesh defines the mesh to add\r\n * @param recursive if all child meshes should also be added to the scene\r\n */\r\n public addMesh(newMesh: AbstractMesh, recursive = false) {\r\n if (this._blockEntityCollection) {\r\n return;\r\n }\r\n\r\n this.meshes.push(newMesh);\r\n\r\n newMesh._resyncLightSources();\r\n\r\n if (!newMesh.parent) {\r\n newMesh._addToSceneRootNodes();\r\n }\r\n\r\n this.onNewMeshAddedObservable.notifyObservers(newMesh);\r\n\r\n if (recursive) {\r\n newMesh.getChildMeshes().forEach((m) => {\r\n this.addMesh(m);\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Remove a mesh for the list of scene's meshes\r\n * @param toRemove defines the mesh to remove\r\n * @param recursive if all child meshes should also be removed from the scene\r\n * @returns the index where the mesh was in the mesh list\r\n */\r\n public removeMesh(toRemove: AbstractMesh, recursive = false): number {\r\n const index = this.meshes.indexOf(toRemove);\r\n if (index !== -1) {\r\n // Remove from the scene if mesh found\r\n this.meshes[index] = this.meshes[this.meshes.length - 1];\r\n this.meshes.pop();\r\n\r\n if (!toRemove.parent) {\r\n toRemove._removeFromSceneRootNodes();\r\n }\r\n }\r\n\r\n this._inputManager._invalidateMesh(toRemove);\r\n\r\n this.onMeshRemovedObservable.notifyObservers(toRemove);\r\n if (recursive) {\r\n toRemove.getChildMeshes().forEach((m) => {\r\n this.removeMesh(m);\r\n });\r\n }\r\n return index;\r\n }\r\n\r\n /**\r\n * Add a transform node to the list of scene's transform nodes\r\n * @param newTransformNode defines the transform node to add\r\n */\r\n public addTransformNode(newTransformNode: TransformNode) {\r\n if (this._blockEntityCollection) {\r\n return;\r\n }\r\n\r\n if (newTransformNode.getScene() === this && newTransformNode._indexInSceneTransformNodesArray !== -1) {\r\n // Already there?\r\n return;\r\n }\r\n\r\n newTransformNode._indexInSceneTransformNodesArray = this.transformNodes.length;\r\n this.transformNodes.push(newTransformNode);\r\n\r\n if (!newTransformNode.parent) {\r\n newTransformNode._addToSceneRootNodes();\r\n }\r\n\r\n this.onNewTransformNodeAddedObservable.notifyObservers(newTransformNode);\r\n }\r\n\r\n /**\r\n * Remove a transform node for the list of scene's transform nodes\r\n * @param toRemove defines the transform node to remove\r\n * @returns the index where the transform node was in the transform node list\r\n */\r\n public removeTransformNode(toRemove: TransformNode): number {\r\n const index = toRemove._indexInSceneTransformNodesArray;\r\n if (index !== -1) {\r\n if (index !== this.transformNodes.length - 1) {\r\n const lastNode = this.transformNodes[this.transformNodes.length - 1];\r\n this.transformNodes[index] = lastNode;\r\n lastNode._indexInSceneTransformNodesArray = index;\r\n }\r\n\r\n toRemove._indexInSceneTransformNodesArray = -1;\r\n this.transformNodes.pop();\r\n if (!toRemove.parent) {\r\n toRemove._removeFromSceneRootNodes();\r\n }\r\n }\r\n\r\n this.onTransformNodeRemovedObservable.notifyObservers(toRemove);\r\n\r\n return index;\r\n }\r\n\r\n /**\r\n * Remove a skeleton for the list of scene's skeletons\r\n * @param toRemove defines the skeleton to remove\r\n * @returns the index where the skeleton was in the skeleton list\r\n */\r\n public removeSkeleton(toRemove: Skeleton): number {\r\n const index = this.skeletons.indexOf(toRemove);\r\n if (index !== -1) {\r\n // Remove from the scene if found\r\n this.skeletons.splice(index, 1);\r\n this.onSkeletonRemovedObservable.notifyObservers(toRemove);\r\n\r\n // Clean active container\r\n this._executeActiveContainerCleanup(this._activeSkeletons);\r\n }\r\n\r\n return index;\r\n }\r\n\r\n /**\r\n * Remove a morph target for the list of scene's morph targets\r\n * @param toRemove defines the morph target to remove\r\n * @returns the index where the morph target was in the morph target list\r\n */\r\n public removeMorphTargetManager(toRemove: MorphTargetManager): number {\r\n const index = this.morphTargetManagers.indexOf(toRemove);\r\n if (index !== -1) {\r\n // Remove from the scene if found\r\n this.morphTargetManagers.splice(index, 1);\r\n }\r\n\r\n return index;\r\n }\r\n\r\n /**\r\n * Remove a light for the list of scene's lights\r\n * @param toRemove defines the light to remove\r\n * @returns the index where the light was in the light list\r\n */\r\n public removeLight(toRemove: Light): number {\r\n const index = this.lights.indexOf(toRemove);\r\n if (index !== -1) {\r\n // Remove from meshes\r\n for (const mesh of this.meshes) {\r\n mesh._removeLightSource(toRemove, false);\r\n }\r\n\r\n // Remove from the scene if mesh found\r\n this.lights.splice(index, 1);\r\n this.sortLightsByPriority();\r\n\r\n if (!toRemove.parent) {\r\n toRemove._removeFromSceneRootNodes();\r\n }\r\n }\r\n this.onLightRemovedObservable.notifyObservers(toRemove);\r\n return index;\r\n }\r\n\r\n /**\r\n * Remove a camera for the list of scene's cameras\r\n * @param toRemove defines the camera to remove\r\n * @returns the index where the camera was in the camera list\r\n */\r\n public removeCamera(toRemove: Camera): number {\r\n const index = this.cameras.indexOf(toRemove);\r\n if (index !== -1) {\r\n // Remove from the scene if mesh found\r\n this.cameras.splice(index, 1);\r\n if (!toRemove.parent) {\r\n toRemove._removeFromSceneRootNodes();\r\n }\r\n }\r\n // Remove from activeCameras\r\n if (this.activeCameras) {\r\n const index2 = this.activeCameras.indexOf(toRemove);\r\n if (index2 !== -1) {\r\n // Remove from the scene if mesh found\r\n this.activeCameras.splice(index2, 1);\r\n }\r\n }\r\n // Reset the activeCamera\r\n if (this.activeCamera === toRemove) {\r\n if (this.cameras.length > 0) {\r\n this.activeCamera = this.cameras[0];\r\n } else {\r\n this.activeCamera = null;\r\n }\r\n }\r\n this.onCameraRemovedObservable.notifyObservers(toRemove);\r\n return index;\r\n }\r\n\r\n /**\r\n * Remove a particle system for the list of scene's particle systems\r\n * @param toRemove defines the particle system to remove\r\n * @returns the index where the particle system was in the particle system list\r\n */\r\n public removeParticleSystem(toRemove: IParticleSystem): number {\r\n const index = this.particleSystems.indexOf(toRemove);\r\n if (index !== -1) {\r\n this.particleSystems.splice(index, 1);\r\n\r\n // Clean active container\r\n this._executeActiveContainerCleanup(this._activeParticleSystems);\r\n }\r\n return index;\r\n }\r\n\r\n /**\r\n * Remove a animation for the list of scene's animations\r\n * @param toRemove defines the animation to remove\r\n * @returns the index where the animation was in the animation list\r\n */\r\n public removeAnimation(toRemove: Animation): number {\r\n const index = this.animations.indexOf(toRemove);\r\n if (index !== -1) {\r\n this.animations.splice(index, 1);\r\n }\r\n return index;\r\n }\r\n\r\n /**\r\n * Will stop the animation of the given target\r\n * @param target - the target\r\n * @param animationName - the name of the animation to stop (all animations will be stopped if both this and targetMask are empty)\r\n * @param targetMask - a function that determines if the animation should be stopped based on its target (all animations will be stopped if both this and animationName are empty)\r\n */\r\n public stopAnimation(target: any, animationName?: string, targetMask?: (target: any) => boolean): void {\r\n // Do nothing as code will be provided by animation component\r\n }\r\n\r\n /**\r\n * Removes the given animation group from this scene.\r\n * @param toRemove The animation group to remove\r\n * @returns The index of the removed animation group\r\n */\r\n public removeAnimationGroup(toRemove: AnimationGroup): number {\r\n const index = this.animationGroups.indexOf(toRemove);\r\n if (index !== -1) {\r\n this.animationGroups.splice(index, 1);\r\n }\r\n return index;\r\n }\r\n\r\n /**\r\n * Removes the given multi-material from this scene.\r\n * @param toRemove The multi-material to remove\r\n * @returns The index of the removed multi-material\r\n */\r\n public removeMultiMaterial(toRemove: MultiMaterial): number {\r\n const index = this.multiMaterials.indexOf(toRemove);\r\n if (index !== -1) {\r\n this.multiMaterials.splice(index, 1);\r\n }\r\n\r\n this.onMultiMaterialRemovedObservable.notifyObservers(toRemove);\r\n\r\n return index;\r\n }\r\n\r\n /**\r\n * Removes the given material from this scene.\r\n * @param toRemove The material to remove\r\n * @returns The index of the removed material\r\n */\r\n public removeMaterial(toRemove: Material): number {\r\n const index = toRemove._indexInSceneMaterialArray;\r\n if (index !== -1 && index < this.materials.length) {\r\n if (index !== this.materials.length - 1) {\r\n const lastMaterial = this.materials[this.materials.length - 1];\r\n this.materials[index] = lastMaterial;\r\n lastMaterial._indexInSceneMaterialArray = index;\r\n }\r\n\r\n toRemove._indexInSceneMaterialArray = -1;\r\n this.materials.pop();\r\n }\r\n\r\n this.onMaterialRemovedObservable.notifyObservers(toRemove);\r\n\r\n return index;\r\n }\r\n\r\n /**\r\n * Removes the given action manager from this scene.\r\n * @deprecated\r\n * @param toRemove The action manager to remove\r\n * @returns The index of the removed action manager\r\n */\r\n public removeActionManager(toRemove: AbstractActionManager): number {\r\n const index = this.actionManagers.indexOf(toRemove);\r\n if (index !== -1) {\r\n this.actionManagers.splice(index, 1);\r\n }\r\n return index;\r\n }\r\n\r\n /**\r\n * Removes the given texture from this scene.\r\n * @param toRemove The texture to remove\r\n * @returns The index of the removed texture\r\n */\r\n public removeTexture(toRemove: BaseTexture): number {\r\n const index = this.textures.indexOf(toRemove);\r\n if (index !== -1) {\r\n this.textures.splice(index, 1);\r\n }\r\n this.onTextureRemovedObservable.notifyObservers(toRemove);\r\n\r\n return index;\r\n }\r\n\r\n /**\r\n * Adds the given light to this scene\r\n * @param newLight The light to add\r\n */\r\n public addLight(newLight: Light): void {\r\n if (this._blockEntityCollection) {\r\n return;\r\n }\r\n this.lights.push(newLight);\r\n this.sortLightsByPriority();\r\n\r\n if (!newLight.parent) {\r\n newLight._addToSceneRootNodes();\r\n }\r\n\r\n // Add light to all meshes (To support if the light is removed and then re-added)\r\n for (const mesh of this.meshes) {\r\n if (mesh.lightSources.indexOf(newLight) === -1) {\r\n mesh.lightSources.push(newLight);\r\n mesh._resyncLightSources();\r\n }\r\n }\r\n\r\n this.onNewLightAddedObservable.notifyObservers(newLight);\r\n }\r\n\r\n /**\r\n * Sorts the list list based on light priorities\r\n */\r\n public sortLightsByPriority(): void {\r\n if (this.requireLightSorting) {\r\n this.lights.sort(LightConstants.CompareLightsPriority);\r\n }\r\n }\r\n\r\n /**\r\n * Adds the given camera to this scene\r\n * @param newCamera The camera to add\r\n */\r\n public addCamera(newCamera: Camera): void {\r\n if (this._blockEntityCollection) {\r\n return;\r\n }\r\n\r\n this.cameras.push(newCamera);\r\n this.onNewCameraAddedObservable.notifyObservers(newCamera);\r\n\r\n if (!newCamera.parent) {\r\n newCamera._addToSceneRootNodes();\r\n }\r\n }\r\n\r\n /**\r\n * Adds the given skeleton to this scene\r\n * @param newSkeleton The skeleton to add\r\n */\r\n public addSkeleton(newSkeleton: Skeleton): void {\r\n if (this._blockEntityCollection) {\r\n return;\r\n }\r\n this.skeletons.push(newSkeleton);\r\n this.onNewSkeletonAddedObservable.notifyObservers(newSkeleton);\r\n }\r\n\r\n /**\r\n * Adds the given particle system to this scene\r\n * @param newParticleSystem The particle system to add\r\n */\r\n public addParticleSystem(newParticleSystem: IParticleSystem): void {\r\n if (this._blockEntityCollection) {\r\n return;\r\n }\r\n this.particleSystems.push(newParticleSystem);\r\n }\r\n\r\n /**\r\n * Adds the given animation to this scene\r\n * @param newAnimation The animation to add\r\n */\r\n public addAnimation(newAnimation: Animation): void {\r\n if (this._blockEntityCollection) {\r\n return;\r\n }\r\n this.animations.push(newAnimation);\r\n }\r\n\r\n /**\r\n * Adds the given animation group to this scene.\r\n * @param newAnimationGroup The animation group to add\r\n */\r\n public addAnimationGroup(newAnimationGroup: AnimationGroup): void {\r\n if (this._blockEntityCollection) {\r\n return;\r\n }\r\n this.animationGroups.push(newAnimationGroup);\r\n }\r\n\r\n /**\r\n * Adds the given multi-material to this scene\r\n * @param newMultiMaterial The multi-material to add\r\n */\r\n public addMultiMaterial(newMultiMaterial: MultiMaterial): void {\r\n if (this._blockEntityCollection) {\r\n return;\r\n }\r\n this.multiMaterials.push(newMultiMaterial);\r\n this.onNewMultiMaterialAddedObservable.notifyObservers(newMultiMaterial);\r\n }\r\n\r\n /**\r\n * Adds the given material to this scene\r\n * @param newMaterial The material to add\r\n */\r\n public addMaterial(newMaterial: Material): void {\r\n if (this._blockEntityCollection) {\r\n return;\r\n }\r\n\r\n if (newMaterial.getScene() === this && newMaterial._indexInSceneMaterialArray !== -1) {\r\n // Already there??\r\n return;\r\n }\r\n\r\n newMaterial._indexInSceneMaterialArray = this.materials.length;\r\n this.materials.push(newMaterial);\r\n this.onNewMaterialAddedObservable.notifyObservers(newMaterial);\r\n }\r\n\r\n /**\r\n * Adds the given morph target to this scene\r\n * @param newMorphTargetManager The morph target to add\r\n */\r\n public addMorphTargetManager(newMorphTargetManager: MorphTargetManager): void {\r\n if (this._blockEntityCollection) {\r\n return;\r\n }\r\n this.morphTargetManagers.push(newMorphTargetManager);\r\n }\r\n\r\n /**\r\n * Adds the given geometry to this scene\r\n * @param newGeometry The geometry to add\r\n */\r\n public addGeometry(newGeometry: Geometry): void {\r\n if (this._blockEntityCollection) {\r\n return;\r\n }\r\n\r\n if (this._geometriesByUniqueId) {\r\n this._geometriesByUniqueId[newGeometry.uniqueId] = this.geometries.length;\r\n }\r\n\r\n this.geometries.push(newGeometry);\r\n }\r\n\r\n /**\r\n * Adds the given action manager to this scene\r\n * @deprecated\r\n * @param newActionManager The action manager to add\r\n */\r\n public addActionManager(newActionManager: AbstractActionManager): void {\r\n this.actionManagers.push(newActionManager);\r\n }\r\n\r\n /**\r\n * Adds the given texture to this scene.\r\n * @param newTexture The texture to add\r\n */\r\n public addTexture(newTexture: BaseTexture): void {\r\n if (this._blockEntityCollection) {\r\n return;\r\n }\r\n this.textures.push(newTexture);\r\n this.onNewTextureAddedObservable.notifyObservers(newTexture);\r\n }\r\n\r\n /**\r\n * Switch active camera\r\n * @param newCamera defines the new active camera\r\n * @param attachControl defines if attachControl must be called for the new active camera (default: true)\r\n */\r\n public switchActiveCamera(newCamera: Camera, attachControl = true): void {\r\n const canvas = this._engine.getInputElement();\r\n\r\n if (!canvas) {\r\n return;\r\n }\r\n\r\n if (this.activeCamera) {\r\n this.activeCamera.detachControl();\r\n }\r\n this.activeCamera = newCamera;\r\n if (attachControl) {\r\n newCamera.attachControl();\r\n }\r\n }\r\n\r\n /**\r\n * sets the active camera of the scene using its Id\r\n * @param id defines the camera's Id\r\n * @returns the new active camera or null if none found.\r\n */\r\n public setActiveCameraById(id: string): Nullable {\r\n const camera = this.getCameraById(id);\r\n\r\n if (camera) {\r\n this.activeCamera = camera;\r\n return camera;\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * sets the active camera of the scene using its name\r\n * @param name defines the camera's name\r\n * @returns the new active camera or null if none found.\r\n */\r\n public setActiveCameraByName(name: string): Nullable {\r\n const camera = this.getCameraByName(name);\r\n\r\n if (camera) {\r\n this.activeCamera = camera;\r\n return camera;\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * get an animation group using its name\r\n * @param name defines the material's name\r\n * @returns the animation group or null if none found.\r\n */\r\n public getAnimationGroupByName(name: string): Nullable {\r\n for (let index = 0; index < this.animationGroups.length; index++) {\r\n if (this.animationGroups[index].name === name) {\r\n return this.animationGroups[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n private _getMaterial(allowMultiMaterials: boolean, predicate: (m: Material) => boolean): Nullable {\r\n for (let index = 0; index < this.materials.length; index++) {\r\n const material = this.materials[index];\r\n if (predicate(material)) {\r\n return material;\r\n }\r\n }\r\n if (allowMultiMaterials) {\r\n for (let index = 0; index < this.multiMaterials.length; index++) {\r\n const material = this.multiMaterials[index];\r\n if (predicate(material)) {\r\n return material;\r\n }\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Get a material using its unique id\r\n * @param uniqueId defines the material's unique id\r\n * @param allowMultiMaterials determines whether multimaterials should be considered\r\n * @returns the material or null if none found.\r\n */\r\n public getMaterialByUniqueID(uniqueId: number, allowMultiMaterials: boolean = false): Nullable {\r\n return this._getMaterial(allowMultiMaterials, (m) => m.uniqueId === uniqueId);\r\n }\r\n\r\n /**\r\n * get a material using its id\r\n * @param id defines the material's Id\r\n * @param allowMultiMaterials determines whether multimaterials should be considered\r\n * @returns the material or null if none found.\r\n */\r\n public getMaterialById(id: string, allowMultiMaterials: boolean = false): Nullable {\r\n return this._getMaterial(allowMultiMaterials, (m) => m.id === id);\r\n }\r\n\r\n /**\r\n * Gets a material using its name\r\n * @param name defines the material's name\r\n * @param allowMultiMaterials determines whether multimaterials should be considered\r\n * @returns the material or null if none found.\r\n */\r\n public getMaterialByName(name: string, allowMultiMaterials: boolean = false): Nullable {\r\n return this._getMaterial(allowMultiMaterials, (m) => m.name === name);\r\n }\r\n\r\n /**\r\n * Gets a last added material using a given id\r\n * @param id defines the material's id\r\n * @param allowMultiMaterials determines whether multimaterials should be considered\r\n * @returns the last material with the given id or null if none found.\r\n */\r\n public getLastMaterialById(id: string, allowMultiMaterials: boolean = false): Nullable {\r\n for (let index = this.materials.length - 1; index >= 0; index--) {\r\n if (this.materials[index].id === id) {\r\n return this.materials[index];\r\n }\r\n }\r\n if (allowMultiMaterials) {\r\n for (let index = this.multiMaterials.length - 1; index >= 0; index--) {\r\n if (this.multiMaterials[index].id === id) {\r\n return this.multiMaterials[index];\r\n }\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Get a texture using its unique id\r\n * @param uniqueId defines the texture's unique id\r\n * @returns the texture or null if none found.\r\n */\r\n public getTextureByUniqueId(uniqueId: number): Nullable {\r\n for (let index = 0; index < this.textures.length; index++) {\r\n if (this.textures[index].uniqueId === uniqueId) {\r\n return this.textures[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a texture using its name\r\n * @param name defines the texture's name\r\n * @returns the texture or null if none found.\r\n */\r\n public getTextureByName(name: string): Nullable {\r\n for (let index = 0; index < this.textures.length; index++) {\r\n if (this.textures[index].name === name) {\r\n return this.textures[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a camera using its Id\r\n * @param id defines the Id to look for\r\n * @returns the camera or null if not found\r\n */\r\n public getCameraById(id: string): Nullable {\r\n for (let index = 0; index < this.cameras.length; index++) {\r\n if (this.cameras[index].id === id) {\r\n return this.cameras[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a camera using its unique Id\r\n * @param uniqueId defines the unique Id to look for\r\n * @returns the camera or null if not found\r\n */\r\n public getCameraByUniqueId(uniqueId: number): Nullable {\r\n for (let index = 0; index < this.cameras.length; index++) {\r\n if (this.cameras[index].uniqueId === uniqueId) {\r\n return this.cameras[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a camera using its name\r\n * @param name defines the camera's name\r\n * @returns the camera or null if none found.\r\n */\r\n public getCameraByName(name: string): Nullable {\r\n for (let index = 0; index < this.cameras.length; index++) {\r\n if (this.cameras[index].name === name) {\r\n return this.cameras[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a bone using its Id\r\n * @param id defines the bone's Id\r\n * @returns the bone or null if not found\r\n */\r\n public getBoneById(id: string): Nullable {\r\n for (let skeletonIndex = 0; skeletonIndex < this.skeletons.length; skeletonIndex++) {\r\n const skeleton = this.skeletons[skeletonIndex];\r\n for (let boneIndex = 0; boneIndex < skeleton.bones.length; boneIndex++) {\r\n if (skeleton.bones[boneIndex].id === id) {\r\n return skeleton.bones[boneIndex];\r\n }\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a bone using its id\r\n * @param name defines the bone's name\r\n * @returns the bone or null if not found\r\n */\r\n public getBoneByName(name: string): Nullable {\r\n for (let skeletonIndex = 0; skeletonIndex < this.skeletons.length; skeletonIndex++) {\r\n const skeleton = this.skeletons[skeletonIndex];\r\n for (let boneIndex = 0; boneIndex < skeleton.bones.length; boneIndex++) {\r\n if (skeleton.bones[boneIndex].name === name) {\r\n return skeleton.bones[boneIndex];\r\n }\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a light node using its name\r\n * @param name defines the the light's name\r\n * @returns the light or null if none found.\r\n */\r\n public getLightByName(name: string): Nullable {\r\n for (let index = 0; index < this.lights.length; index++) {\r\n if (this.lights[index].name === name) {\r\n return this.lights[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a light node using its Id\r\n * @param id defines the light's Id\r\n * @returns the light or null if none found.\r\n */\r\n public getLightById(id: string): Nullable {\r\n for (let index = 0; index < this.lights.length; index++) {\r\n if (this.lights[index].id === id) {\r\n return this.lights[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a light node using its scene-generated unique Id\r\n * @param uniqueId defines the light's unique Id\r\n * @returns the light or null if none found.\r\n */\r\n public getLightByUniqueId(uniqueId: number): Nullable {\r\n for (let index = 0; index < this.lights.length; index++) {\r\n if (this.lights[index].uniqueId === uniqueId) {\r\n return this.lights[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a particle system by Id\r\n * @param id defines the particle system Id\r\n * @returns the corresponding system or null if none found\r\n */\r\n public getParticleSystemById(id: string): Nullable {\r\n for (let index = 0; index < this.particleSystems.length; index++) {\r\n if (this.particleSystems[index].id === id) {\r\n return this.particleSystems[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a geometry using its Id\r\n * @param id defines the geometry's Id\r\n * @returns the geometry or null if none found.\r\n */\r\n public getGeometryById(id: string): Nullable {\r\n for (let index = 0; index < this.geometries.length; index++) {\r\n if (this.geometries[index].id === id) {\r\n return this.geometries[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n private _getGeometryByUniqueId(uniqueId: number): Nullable {\r\n if (this._geometriesByUniqueId) {\r\n const index = this._geometriesByUniqueId[uniqueId];\r\n if (index !== undefined) {\r\n return this.geometries[index];\r\n }\r\n } else {\r\n for (let index = 0; index < this.geometries.length; index++) {\r\n if (this.geometries[index].uniqueId === uniqueId) {\r\n return this.geometries[index];\r\n }\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Add a new geometry to this scene\r\n * @param geometry defines the geometry to be added to the scene.\r\n * @param force defines if the geometry must be pushed even if a geometry with this id already exists\r\n * @returns a boolean defining if the geometry was added or not\r\n */\r\n public pushGeometry(geometry: Geometry, force?: boolean): boolean {\r\n if (!force && this._getGeometryByUniqueId(geometry.uniqueId)) {\r\n return false;\r\n }\r\n\r\n this.addGeometry(geometry);\r\n\r\n this.onNewGeometryAddedObservable.notifyObservers(geometry);\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Removes an existing geometry\r\n * @param geometry defines the geometry to be removed from the scene\r\n * @returns a boolean defining if the geometry was removed or not\r\n */\r\n public removeGeometry(geometry: Geometry): boolean {\r\n let index;\r\n if (this._geometriesByUniqueId) {\r\n index = this._geometriesByUniqueId[geometry.uniqueId];\r\n if (index === undefined) {\r\n return false;\r\n }\r\n } else {\r\n index = this.geometries.indexOf(geometry);\r\n if (index < 0) {\r\n return false;\r\n }\r\n }\r\n\r\n if (index !== this.geometries.length - 1) {\r\n const lastGeometry = this.geometries[this.geometries.length - 1];\r\n if (lastGeometry) {\r\n this.geometries[index] = lastGeometry;\r\n if (this._geometriesByUniqueId) {\r\n this._geometriesByUniqueId[lastGeometry.uniqueId] = index;\r\n }\r\n }\r\n }\r\n\r\n if (this._geometriesByUniqueId) {\r\n this._geometriesByUniqueId[geometry.uniqueId] = undefined;\r\n }\r\n\r\n this.geometries.pop();\r\n\r\n this.onGeometryRemovedObservable.notifyObservers(geometry);\r\n return true;\r\n }\r\n\r\n /**\r\n * Gets the list of geometries attached to the scene\r\n * @returns an array of Geometry\r\n */\r\n public getGeometries(): Geometry[] {\r\n return this.geometries;\r\n }\r\n\r\n /**\r\n * Gets the first added mesh found of a given Id\r\n * @param id defines the Id to search for\r\n * @returns the mesh found or null if not found at all\r\n */\r\n public getMeshById(id: string): Nullable {\r\n for (let index = 0; index < this.meshes.length; index++) {\r\n if (this.meshes[index].id === id) {\r\n return this.meshes[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a list of meshes using their Id\r\n * @param id defines the Id to search for\r\n * @returns a list of meshes\r\n */\r\n public getMeshesById(id: string): Array {\r\n return this.meshes.filter(function (m) {\r\n return m.id === id;\r\n });\r\n }\r\n\r\n /**\r\n * Gets the first added transform node found of a given Id\r\n * @param id defines the Id to search for\r\n * @returns the found transform node or null if not found at all.\r\n */\r\n public getTransformNodeById(id: string): Nullable {\r\n for (let index = 0; index < this.transformNodes.length; index++) {\r\n if (this.transformNodes[index].id === id) {\r\n return this.transformNodes[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a transform node with its auto-generated unique Id\r\n * @param uniqueId defines the unique Id to search for\r\n * @returns the found transform node or null if not found at all.\r\n */\r\n public getTransformNodeByUniqueId(uniqueId: number): Nullable {\r\n for (let index = 0; index < this.transformNodes.length; index++) {\r\n if (this.transformNodes[index].uniqueId === uniqueId) {\r\n return this.transformNodes[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a list of transform nodes using their Id\r\n * @param id defines the Id to search for\r\n * @returns a list of transform nodes\r\n */\r\n public getTransformNodesById(id: string): Array {\r\n return this.transformNodes.filter(function (m) {\r\n return m.id === id;\r\n });\r\n }\r\n\r\n /**\r\n * Gets a mesh with its auto-generated unique Id\r\n * @param uniqueId defines the unique Id to search for\r\n * @returns the found mesh or null if not found at all.\r\n */\r\n public getMeshByUniqueId(uniqueId: number): Nullable {\r\n for (let index = 0; index < this.meshes.length; index++) {\r\n if (this.meshes[index].uniqueId === uniqueId) {\r\n return this.meshes[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a the last added mesh using a given Id\r\n * @param id defines the Id to search for\r\n * @returns the found mesh or null if not found at all.\r\n */\r\n public getLastMeshById(id: string): Nullable {\r\n for (let index = this.meshes.length - 1; index >= 0; index--) {\r\n if (this.meshes[index].id === id) {\r\n return this.meshes[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a the last transform node using a given Id\r\n * @param id defines the Id to search for\r\n * @returns the found mesh or null if not found at all.\r\n */\r\n public getLastTransformNodeById(id: string): Nullable {\r\n for (let index = this.transformNodes.length - 1; index >= 0; index--) {\r\n if (this.transformNodes[index].id === id) {\r\n return this.transformNodes[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a the last added node (Mesh, Camera, Light) using a given Id\r\n * @param id defines the Id to search for\r\n * @returns the found node or null if not found at all\r\n */\r\n public getLastEntryById(id: string): Nullable {\r\n let index: number;\r\n for (index = this.meshes.length - 1; index >= 0; index--) {\r\n if (this.meshes[index].id === id) {\r\n return this.meshes[index];\r\n }\r\n }\r\n\r\n for (index = this.transformNodes.length - 1; index >= 0; index--) {\r\n if (this.transformNodes[index].id === id) {\r\n return this.transformNodes[index];\r\n }\r\n }\r\n\r\n for (index = this.cameras.length - 1; index >= 0; index--) {\r\n if (this.cameras[index].id === id) {\r\n return this.cameras[index];\r\n }\r\n }\r\n\r\n for (index = this.lights.length - 1; index >= 0; index--) {\r\n if (this.lights[index].id === id) {\r\n return this.lights[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a node (Mesh, Camera, Light) using a given Id\r\n * @param id defines the Id to search for\r\n * @returns the found node or null if not found at all\r\n */\r\n public getNodeById(id: string): Nullable {\r\n const mesh = this.getMeshById(id);\r\n if (mesh) {\r\n return mesh;\r\n }\r\n\r\n const transformNode = this.getTransformNodeById(id);\r\n if (transformNode) {\r\n return transformNode;\r\n }\r\n\r\n const light = this.getLightById(id);\r\n if (light) {\r\n return light;\r\n }\r\n\r\n const camera = this.getCameraById(id);\r\n if (camera) {\r\n return camera;\r\n }\r\n\r\n const bone = this.getBoneById(id);\r\n if (bone) {\r\n return bone;\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a node (Mesh, Camera, Light) using a given name\r\n * @param name defines the name to search for\r\n * @returns the found node or null if not found at all.\r\n */\r\n public getNodeByName(name: string): Nullable {\r\n const mesh = this.getMeshByName(name);\r\n if (mesh) {\r\n return mesh;\r\n }\r\n\r\n const transformNode = this.getTransformNodeByName(name);\r\n if (transformNode) {\r\n return transformNode;\r\n }\r\n\r\n const light = this.getLightByName(name);\r\n if (light) {\r\n return light;\r\n }\r\n\r\n const camera = this.getCameraByName(name);\r\n if (camera) {\r\n return camera;\r\n }\r\n\r\n const bone = this.getBoneByName(name);\r\n if (bone) {\r\n return bone;\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a mesh using a given name\r\n * @param name defines the name to search for\r\n * @returns the found mesh or null if not found at all.\r\n */\r\n public getMeshByName(name: string): Nullable {\r\n for (let index = 0; index < this.meshes.length; index++) {\r\n if (this.meshes[index].name === name) {\r\n return this.meshes[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a transform node using a given name\r\n * @param name defines the name to search for\r\n * @returns the found transform node or null if not found at all.\r\n */\r\n public getTransformNodeByName(name: string): Nullable {\r\n for (let index = 0; index < this.transformNodes.length; index++) {\r\n if (this.transformNodes[index].name === name) {\r\n return this.transformNodes[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a skeleton using a given Id (if many are found, this function will pick the last one)\r\n * @param id defines the Id to search for\r\n * @returns the found skeleton or null if not found at all.\r\n */\r\n public getLastSkeletonById(id: string): Nullable {\r\n for (let index = this.skeletons.length - 1; index >= 0; index--) {\r\n if (this.skeletons[index].id === id) {\r\n return this.skeletons[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a skeleton using a given auto generated unique id\r\n * @param uniqueId defines the unique id to search for\r\n * @returns the found skeleton or null if not found at all.\r\n */\r\n public getSkeletonByUniqueId(uniqueId: number): Nullable {\r\n for (let index = 0; index < this.skeletons.length; index++) {\r\n if (this.skeletons[index].uniqueId === uniqueId) {\r\n return this.skeletons[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a skeleton using a given id (if many are found, this function will pick the first one)\r\n * @param id defines the id to search for\r\n * @returns the found skeleton or null if not found at all.\r\n */\r\n public getSkeletonById(id: string): Nullable {\r\n for (let index = 0; index < this.skeletons.length; index++) {\r\n if (this.skeletons[index].id === id) {\r\n return this.skeletons[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a skeleton using a given name\r\n * @param name defines the name to search for\r\n * @returns the found skeleton or null if not found at all.\r\n */\r\n public getSkeletonByName(name: string): Nullable {\r\n for (let index = 0; index < this.skeletons.length; index++) {\r\n if (this.skeletons[index].name === name) {\r\n return this.skeletons[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a morph target manager using a given id (if many are found, this function will pick the last one)\r\n * @param id defines the id to search for\r\n * @returns the found morph target manager or null if not found at all.\r\n */\r\n public getMorphTargetManagerById(id: number): Nullable {\r\n for (let index = 0; index < this.morphTargetManagers.length; index++) {\r\n if (this.morphTargetManagers[index].uniqueId === id) {\r\n return this.morphTargetManagers[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a morph target using a given id (if many are found, this function will pick the first one)\r\n * @param id defines the id to search for\r\n * @returns the found morph target or null if not found at all.\r\n */\r\n public getMorphTargetById(id: string): Nullable {\r\n for (let managerIndex = 0; managerIndex < this.morphTargetManagers.length; ++managerIndex) {\r\n const morphTargetManager = this.morphTargetManagers[managerIndex];\r\n for (let index = 0; index < morphTargetManager.numTargets; ++index) {\r\n const target = morphTargetManager.getTarget(index);\r\n if (target.id === id) {\r\n return target;\r\n }\r\n }\r\n }\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a morph target using a given name (if many are found, this function will pick the first one)\r\n * @param name defines the name to search for\r\n * @returns the found morph target or null if not found at all.\r\n */\r\n public getMorphTargetByName(name: string): Nullable {\r\n for (let managerIndex = 0; managerIndex < this.morphTargetManagers.length; ++managerIndex) {\r\n const morphTargetManager = this.morphTargetManagers[managerIndex];\r\n for (let index = 0; index < morphTargetManager.numTargets; ++index) {\r\n const target = morphTargetManager.getTarget(index);\r\n if (target.name === name) {\r\n return target;\r\n }\r\n }\r\n }\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a post process using a given name (if many are found, this function will pick the first one)\r\n * @param name defines the name to search for\r\n * @returns the found post process or null if not found at all.\r\n */\r\n public getPostProcessByName(name: string): Nullable {\r\n for (let postProcessIndex = 0; postProcessIndex < this.postProcesses.length; ++postProcessIndex) {\r\n const postProcess = this.postProcesses[postProcessIndex];\r\n if (postProcess.name === name) {\r\n return postProcess;\r\n }\r\n }\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if the given mesh is active\r\n * @param mesh defines the mesh to look for\r\n * @returns true if the mesh is in the active list\r\n */\r\n public isActiveMesh(mesh: AbstractMesh): boolean {\r\n return this._activeMeshes.indexOf(mesh) !== -1;\r\n }\r\n\r\n /**\r\n * Return a unique id as a string which can serve as an identifier for the scene\r\n */\r\n public get uid(): string {\r\n if (!this._uid) {\r\n this._uid = Tools.RandomId();\r\n }\r\n return this._uid;\r\n }\r\n\r\n /**\r\n * Add an externally attached data from its key.\r\n * This method call will fail and return false, if such key already exists.\r\n * If you don't care and just want to get the data no matter what, use the more convenient getOrAddExternalDataWithFactory() method.\r\n * @param key the unique key that identifies the data\r\n * @param data the data object to associate to the key for this Engine instance\r\n * @returns true if no such key were already present and the data was added successfully, false otherwise\r\n */\r\n public addExternalData(key: string, data: T): boolean {\r\n if (!this._externalData) {\r\n this._externalData = new StringDictionary();\r\n }\r\n return this._externalData.add(key, data);\r\n }\r\n\r\n /**\r\n * Get an externally attached data from its key\r\n * @param key the unique key that identifies the data\r\n * @returns the associated data, if present (can be null), or undefined if not present\r\n */\r\n public getExternalData(key: string): Nullable {\r\n if (!this._externalData) {\r\n return null;\r\n }\r\n return this._externalData.get(key);\r\n }\r\n\r\n /**\r\n * Get an externally attached data from its key, create it using a factory if it's not already present\r\n * @param key the unique key that identifies the data\r\n * @param factory the factory that will be called to create the instance if and only if it doesn't exists\r\n * @returns the associated data, can be null if the factory returned null.\r\n */\r\n public getOrAddExternalDataWithFactory(key: string, factory: (k: string) => T): T {\r\n if (!this._externalData) {\r\n this._externalData = new StringDictionary();\r\n }\r\n return this._externalData.getOrAddWithFactory(key, factory);\r\n }\r\n\r\n /**\r\n * Remove an externally attached data from the Engine instance\r\n * @param key the unique key that identifies the data\r\n * @returns true if the data was successfully removed, false if it doesn't exist\r\n */\r\n public removeExternalData(key: string): boolean {\r\n return this._externalData.remove(key);\r\n }\r\n\r\n private _evaluateSubMesh(subMesh: SubMesh, mesh: AbstractMesh, initialMesh: AbstractMesh, forcePush: boolean): void {\r\n if (forcePush || subMesh.isInFrustum(this._frustumPlanes)) {\r\n for (const step of this._evaluateSubMeshStage) {\r\n step.action(mesh, subMesh);\r\n }\r\n\r\n const material = subMesh.getMaterial();\r\n if (material !== null && material !== undefined) {\r\n // Render targets\r\n if (material.hasRenderTargetTextures && material.getRenderTargetTextures != null) {\r\n if (this._processedMaterials.indexOf(material) === -1) {\r\n this._processedMaterials.push(material);\r\n\r\n this._materialsRenderTargets.concatWithNoDuplicate(material.getRenderTargetTextures!());\r\n }\r\n }\r\n\r\n // Dispatch\r\n this._renderingManager.dispatch(subMesh, mesh, material);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Clear the processed materials smart array preventing retention point in material dispose.\r\n */\r\n public freeProcessedMaterials(): void {\r\n this._processedMaterials.dispose();\r\n }\r\n\r\n private _preventFreeActiveMeshesAndRenderingGroups = false;\r\n\r\n /** Gets or sets a boolean blocking all the calls to freeActiveMeshes and freeRenderingGroups\r\n * It can be used in order to prevent going through methods freeRenderingGroups and freeActiveMeshes several times to improve performance\r\n * when disposing several meshes in a row or a hierarchy of meshes.\r\n * When used, it is the responsibility of the user to blockfreeActiveMeshesAndRenderingGroups back to false.\r\n */\r\n public get blockfreeActiveMeshesAndRenderingGroups(): boolean {\r\n return this._preventFreeActiveMeshesAndRenderingGroups;\r\n }\r\n\r\n public set blockfreeActiveMeshesAndRenderingGroups(value: boolean) {\r\n if (this._preventFreeActiveMeshesAndRenderingGroups === value) {\r\n return;\r\n }\r\n\r\n if (value) {\r\n this.freeActiveMeshes();\r\n this.freeRenderingGroups();\r\n }\r\n\r\n this._preventFreeActiveMeshesAndRenderingGroups = value;\r\n }\r\n\r\n /**\r\n * Clear the active meshes smart array preventing retention point in mesh dispose.\r\n */\r\n public freeActiveMeshes(): void {\r\n if (this.blockfreeActiveMeshesAndRenderingGroups) {\r\n return;\r\n }\r\n\r\n this._activeMeshes.dispose();\r\n if (this.activeCamera && this.activeCamera._activeMeshes) {\r\n this.activeCamera._activeMeshes.dispose();\r\n }\r\n if (this.activeCameras) {\r\n for (let i = 0; i < this.activeCameras.length; i++) {\r\n const activeCamera = this.activeCameras[i];\r\n if (activeCamera && activeCamera._activeMeshes) {\r\n activeCamera._activeMeshes.dispose();\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Clear the info related to rendering groups preventing retention points during dispose.\r\n */\r\n public freeRenderingGroups(): void {\r\n if (this.blockfreeActiveMeshesAndRenderingGroups) {\r\n return;\r\n }\r\n\r\n if (this._renderingManager) {\r\n this._renderingManager.freeRenderingGroups();\r\n }\r\n if (this.textures) {\r\n for (let i = 0; i < this.textures.length; i++) {\r\n const texture = this.textures[i];\r\n if (texture && (texture).renderList) {\r\n (texture).freeRenderingGroups();\r\n }\r\n }\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _isInIntermediateRendering(): boolean {\r\n return this._intermediateRendering;\r\n }\r\n\r\n /**\r\n * Lambda returning the list of potentially active meshes.\r\n */\r\n public getActiveMeshCandidates: () => ISmartArrayLike;\r\n\r\n /**\r\n * Lambda returning the list of potentially active sub meshes.\r\n */\r\n public getActiveSubMeshCandidates: (mesh: AbstractMesh) => ISmartArrayLike;\r\n\r\n /**\r\n * Lambda returning the list of potentially intersecting sub meshes.\r\n */\r\n public getIntersectingSubMeshCandidates: (mesh: AbstractMesh, localRay: Ray) => ISmartArrayLike;\r\n\r\n /**\r\n * Lambda returning the list of potentially colliding sub meshes.\r\n */\r\n public getCollidingSubMeshCandidates: (mesh: AbstractMesh, collider: Collider) => ISmartArrayLike;\r\n\r\n /** @internal */\r\n public _activeMeshesFrozen = false;\r\n /** @internal */\r\n public _activeMeshesFrozenButKeepClipping = false;\r\n private _skipEvaluateActiveMeshesCompletely = false;\r\n\r\n /**\r\n * Use this function to stop evaluating active meshes. The current list will be keep alive between frames\r\n * @param skipEvaluateActiveMeshes defines an optional boolean indicating that the evaluate active meshes step must be completely skipped\r\n * @param onSuccess optional success callback\r\n * @param onError optional error callback\r\n * @param freezeMeshes defines if meshes should be frozen (true by default)\r\n * @param keepFrustumCulling defines if you want to keep running the frustum clipping (false by default)\r\n * @returns the current scene\r\n */\r\n public freezeActiveMeshes(\r\n skipEvaluateActiveMeshes = false,\r\n onSuccess?: () => void,\r\n onError?: (message: string) => void,\r\n freezeMeshes = true,\r\n keepFrustumCulling = false\r\n ): Scene {\r\n this.executeWhenReady(() => {\r\n if (!this.activeCamera) {\r\n onError && onError(\"No active camera found\");\r\n return;\r\n }\r\n\r\n if (!this._frustumPlanes) {\r\n this.updateTransformMatrix();\r\n }\r\n\r\n this._evaluateActiveMeshes();\r\n this._activeMeshesFrozen = true;\r\n this._activeMeshesFrozenButKeepClipping = keepFrustumCulling;\r\n this._skipEvaluateActiveMeshesCompletely = skipEvaluateActiveMeshes;\r\n\r\n if (freezeMeshes) {\r\n for (let index = 0; index < this._activeMeshes.length; index++) {\r\n this._activeMeshes.data[index]._freeze();\r\n }\r\n }\r\n onSuccess && onSuccess();\r\n });\r\n return this;\r\n }\r\n\r\n /**\r\n * Use this function to restart evaluating active meshes on every frame\r\n * @returns the current scene\r\n */\r\n public unfreezeActiveMeshes(): Scene {\r\n for (let index = 0; index < this.meshes.length; index++) {\r\n const mesh = this.meshes[index];\r\n if (mesh._internalAbstractMeshDataInfo) {\r\n mesh._internalAbstractMeshDataInfo._isActive = false;\r\n }\r\n }\r\n\r\n for (let index = 0; index < this._activeMeshes.length; index++) {\r\n this._activeMeshes.data[index]._unFreeze();\r\n }\r\n\r\n this._activeMeshesFrozen = false;\r\n return this;\r\n }\r\n\r\n private _executeActiveContainerCleanup(container: SmartArray) {\r\n const isInFastMode = this._engine.snapshotRendering && this._engine.snapshotRenderingMode === Constants.SNAPSHOTRENDERING_FAST;\r\n\r\n if (!isInFastMode && this._activeMeshesFrozen && this._activeMeshes.length) {\r\n return; // Do not execute in frozen mode\r\n }\r\n\r\n // We need to ensure we are not in the rendering loop\r\n this.onBeforeRenderObservable.addOnce(() => container.dispose());\r\n }\r\n\r\n private _evaluateActiveMeshes(): void {\r\n if (this._engine.snapshotRendering && this._engine.snapshotRenderingMode === Constants.SNAPSHOTRENDERING_FAST) {\r\n if (this._activeMeshes.length > 0) {\r\n this.activeCamera?._activeMeshes.reset();\r\n this._activeMeshes.reset();\r\n this._renderingManager.reset();\r\n this._processedMaterials.reset();\r\n this._activeParticleSystems.reset();\r\n this._activeSkeletons.reset();\r\n this._softwareSkinnedMeshes.reset();\r\n }\r\n return;\r\n }\r\n\r\n if (this._activeMeshesFrozen && this._activeMeshes.length) {\r\n if (!this._skipEvaluateActiveMeshesCompletely) {\r\n const len = this._activeMeshes.length;\r\n for (let i = 0; i < len; i++) {\r\n const mesh = this._activeMeshes.data[i];\r\n mesh.computeWorldMatrix();\r\n }\r\n }\r\n\r\n if (this._activeParticleSystems) {\r\n const psLength = this._activeParticleSystems.length;\r\n for (let i = 0; i < psLength; i++) {\r\n this._activeParticleSystems.data[i].animate();\r\n }\r\n }\r\n\r\n this._renderingManager.resetSprites();\r\n\r\n return;\r\n }\r\n\r\n if (!this.activeCamera) {\r\n return;\r\n }\r\n\r\n this.onBeforeActiveMeshesEvaluationObservable.notifyObservers(this);\r\n\r\n this.activeCamera._activeMeshes.reset();\r\n this._activeMeshes.reset();\r\n this._renderingManager.reset();\r\n this._processedMaterials.reset();\r\n this._activeParticleSystems.reset();\r\n this._activeSkeletons.reset();\r\n this._softwareSkinnedMeshes.reset();\r\n this._materialsRenderTargets.reset();\r\n\r\n for (const step of this._beforeEvaluateActiveMeshStage) {\r\n step.action();\r\n }\r\n\r\n // Determine mesh candidates\r\n const meshes = this.getActiveMeshCandidates();\r\n\r\n // Check each mesh\r\n const len = meshes.length;\r\n for (let i = 0; i < len; i++) {\r\n const mesh = meshes.data[i];\r\n mesh._internalAbstractMeshDataInfo._currentLODIsUpToDate = false;\r\n if (mesh.isBlocked) {\r\n continue;\r\n }\r\n\r\n this._totalVertices.addCount(mesh.getTotalVertices(), false);\r\n\r\n if (!mesh.isReady() || !mesh.isEnabled() || mesh.scaling.hasAZeroComponent) {\r\n continue;\r\n }\r\n\r\n mesh.computeWorldMatrix();\r\n\r\n // Intersections\r\n if (mesh.actionManager && mesh.actionManager.hasSpecificTriggers2(Constants.ACTION_OnIntersectionEnterTrigger, Constants.ACTION_OnIntersectionExitTrigger)) {\r\n this._meshesForIntersections.pushNoDuplicate(mesh);\r\n }\r\n\r\n // Switch to current LOD\r\n let meshToRender = this.customLODSelector ? this.customLODSelector(mesh, this.activeCamera) : mesh.getLOD(this.activeCamera);\r\n mesh._internalAbstractMeshDataInfo._currentLOD = meshToRender;\r\n mesh._internalAbstractMeshDataInfo._currentLODIsUpToDate = true;\r\n if (meshToRender === undefined || meshToRender === null) {\r\n continue;\r\n }\r\n\r\n // Compute world matrix if LOD is billboard\r\n if (meshToRender !== mesh && meshToRender.billboardMode !== 0) {\r\n meshToRender.computeWorldMatrix();\r\n }\r\n\r\n mesh._preActivate();\r\n\r\n if (\r\n mesh.isVisible &&\r\n mesh.visibility > 0 &&\r\n (mesh.layerMask & this.activeCamera.layerMask) !== 0 &&\r\n (this._skipFrustumClipping || mesh.alwaysSelectAsActiveMesh || mesh.isInFrustum(this._frustumPlanes))\r\n ) {\r\n this._activeMeshes.push(mesh);\r\n this.activeCamera._activeMeshes.push(mesh);\r\n\r\n if (meshToRender !== mesh) {\r\n meshToRender._activate(this._renderId, false);\r\n }\r\n\r\n for (const step of this._preActiveMeshStage) {\r\n step.action(mesh);\r\n }\r\n\r\n if (mesh._activate(this._renderId, false)) {\r\n if (!mesh.isAnInstance) {\r\n meshToRender._internalAbstractMeshDataInfo._onlyForInstances = false;\r\n } else {\r\n if (mesh._internalAbstractMeshDataInfo._actAsRegularMesh) {\r\n meshToRender = mesh;\r\n }\r\n }\r\n meshToRender._internalAbstractMeshDataInfo._isActive = true;\r\n this._activeMesh(mesh, meshToRender);\r\n }\r\n\r\n mesh._postActivate();\r\n }\r\n }\r\n\r\n this.onAfterActiveMeshesEvaluationObservable.notifyObservers(this);\r\n\r\n // Particle systems\r\n if (this.particlesEnabled) {\r\n this.onBeforeParticlesRenderingObservable.notifyObservers(this);\r\n for (let particleIndex = 0; particleIndex < this.particleSystems.length; particleIndex++) {\r\n const particleSystem = this.particleSystems[particleIndex];\r\n\r\n if (!particleSystem.isStarted() || !particleSystem.emitter) {\r\n continue;\r\n }\r\n\r\n const emitter = particleSystem.emitter;\r\n if (!emitter.position || emitter.isEnabled()) {\r\n this._activeParticleSystems.push(particleSystem);\r\n particleSystem.animate();\r\n this._renderingManager.dispatchParticles(particleSystem);\r\n }\r\n }\r\n this.onAfterParticlesRenderingObservable.notifyObservers(this);\r\n }\r\n }\r\n\r\n private _activeMesh(sourceMesh: AbstractMesh, mesh: AbstractMesh): void {\r\n if (this._skeletonsEnabled && mesh.skeleton !== null && mesh.skeleton !== undefined) {\r\n if (this._activeSkeletons.pushNoDuplicate(mesh.skeleton)) {\r\n mesh.skeleton.prepare();\r\n this._activeBones.addCount(mesh.skeleton.bones.length, false);\r\n }\r\n\r\n if (!mesh.computeBonesUsingShaders) {\r\n this._softwareSkinnedMeshes.pushNoDuplicate(mesh);\r\n }\r\n }\r\n\r\n let forcePush = sourceMesh.hasInstances || sourceMesh.isAnInstance || this.dispatchAllSubMeshesOfActiveMeshes || this._skipFrustumClipping || mesh.alwaysSelectAsActiveMesh;\r\n\r\n if (mesh && mesh.subMeshes && mesh.subMeshes.length > 0) {\r\n const subMeshes = this.getActiveSubMeshCandidates(mesh);\r\n const len = subMeshes.length;\r\n forcePush = forcePush || len === 1;\r\n for (let i = 0; i < len; i++) {\r\n const subMesh = subMeshes.data[i];\r\n this._evaluateSubMesh(subMesh, mesh, sourceMesh, forcePush);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Update the transform matrix to update from the current active camera\r\n * @param force defines a boolean used to force the update even if cache is up to date\r\n */\r\n public updateTransformMatrix(force?: boolean): void {\r\n if (!this.activeCamera) {\r\n return;\r\n }\r\n\r\n if (this.activeCamera._renderingMultiview) {\r\n const leftCamera = this.activeCamera._rigCameras[0];\r\n const rightCamera = this.activeCamera._rigCameras[1];\r\n this.setTransformMatrix(leftCamera.getViewMatrix(), leftCamera.getProjectionMatrix(force), rightCamera.getViewMatrix(), rightCamera.getProjectionMatrix(force));\r\n } else {\r\n this.setTransformMatrix(this.activeCamera.getViewMatrix(), this.activeCamera.getProjectionMatrix(force));\r\n }\r\n }\r\n\r\n private _bindFrameBuffer(camera: Nullable, clear = true) {\r\n if (camera && camera._multiviewTexture) {\r\n camera._multiviewTexture._bindFrameBuffer();\r\n } else if (camera && camera.outputRenderTarget) {\r\n camera.outputRenderTarget._bindFrameBuffer();\r\n } else {\r\n if (!this._engine._currentFrameBufferIsDefaultFrameBuffer()) {\r\n this._engine.restoreDefaultFramebuffer();\r\n }\r\n }\r\n if (clear) {\r\n this._clearFrameBuffer(camera);\r\n }\r\n }\r\n\r\n private _clearFrameBuffer(camera: Nullable) {\r\n // we assume the framebuffer currently bound is the right one\r\n if (camera && camera._multiviewTexture) {\r\n // no clearing?\r\n } else if (camera && camera.outputRenderTarget && !camera._renderingMultiview) {\r\n const rtt = camera.outputRenderTarget;\r\n if (rtt.onClearObservable.hasObservers()) {\r\n rtt.onClearObservable.notifyObservers(this._engine);\r\n } else if (!rtt.skipInitialClear) {\r\n if (this.autoClear) {\r\n this._engine.clear(rtt.clearColor || this.clearColor, !rtt._cleared, true, true);\r\n }\r\n rtt._cleared = true;\r\n }\r\n } else {\r\n if (!this._defaultFrameBufferCleared) {\r\n this._defaultFrameBufferCleared = true;\r\n this._clear();\r\n } else {\r\n this._engine.clear(null, false, true, true);\r\n }\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _allowPostProcessClearColor = true;\r\n /**\r\n * @internal\r\n */\r\n public _renderForCamera(camera: Camera, rigParent?: Camera, bindFrameBuffer = true): void {\r\n if (camera && camera._skipRendering) {\r\n return;\r\n }\r\n\r\n const engine = this._engine;\r\n\r\n // Use _activeCamera instead of activeCamera to avoid onActiveCameraChanged\r\n this._activeCamera = camera;\r\n\r\n if (!this.activeCamera) {\r\n throw new Error(\"Active camera not set\");\r\n }\r\n\r\n // Viewport\r\n engine.setViewport(this.activeCamera.viewport);\r\n\r\n // Camera\r\n this.resetCachedMaterial();\r\n this._renderId++;\r\n\r\n if (!this.prePass && bindFrameBuffer) {\r\n let skipInitialClear = true;\r\n if (camera._renderingMultiview && camera.outputRenderTarget) {\r\n skipInitialClear = camera.outputRenderTarget.skipInitialClear;\r\n if (this.autoClear) {\r\n this._defaultFrameBufferCleared = false;\r\n camera.outputRenderTarget.skipInitialClear = false;\r\n }\r\n }\r\n this._bindFrameBuffer(this._activeCamera);\r\n if (camera._renderingMultiview && camera.outputRenderTarget) {\r\n camera.outputRenderTarget.skipInitialClear = skipInitialClear;\r\n }\r\n }\r\n\r\n this.updateTransformMatrix();\r\n\r\n this.onBeforeCameraRenderObservable.notifyObservers(this.activeCamera);\r\n\r\n // Meshes\r\n this._evaluateActiveMeshes();\r\n\r\n // Software skinning\r\n for (let softwareSkinnedMeshIndex = 0; softwareSkinnedMeshIndex < this._softwareSkinnedMeshes.length; softwareSkinnedMeshIndex++) {\r\n const mesh = this._softwareSkinnedMeshes.data[softwareSkinnedMeshIndex];\r\n\r\n mesh.applySkeleton(mesh.skeleton);\r\n }\r\n\r\n // Render targets\r\n this.onBeforeRenderTargetsRenderObservable.notifyObservers(this);\r\n\r\n this._renderTargets.concatWithNoDuplicate(this._materialsRenderTargets);\r\n\r\n if (camera.customRenderTargets && camera.customRenderTargets.length > 0) {\r\n this._renderTargets.concatWithNoDuplicate(camera.customRenderTargets);\r\n }\r\n\r\n if (rigParent && rigParent.customRenderTargets && rigParent.customRenderTargets.length > 0) {\r\n this._renderTargets.concatWithNoDuplicate(rigParent.customRenderTargets);\r\n }\r\n\r\n if (this.environmentTexture && this.environmentTexture.isRenderTarget) {\r\n this._renderTargets.pushNoDuplicate(this.environmentTexture as RenderTargetTexture);\r\n }\r\n\r\n // Collects render targets from external components.\r\n for (const step of this._gatherActiveCameraRenderTargetsStage) {\r\n step.action(this._renderTargets);\r\n }\r\n\r\n let needRebind = false;\r\n if (this.renderTargetsEnabled) {\r\n this._intermediateRendering = true;\r\n\r\n if (this._renderTargets.length > 0) {\r\n Tools.StartPerformanceCounter(\"Render targets\", this._renderTargets.length > 0);\r\n for (let renderIndex = 0; renderIndex < this._renderTargets.length; renderIndex++) {\r\n const renderTarget = this._renderTargets.data[renderIndex];\r\n if (renderTarget._shouldRender()) {\r\n this._renderId++;\r\n const hasSpecialRenderTargetCamera = renderTarget.activeCamera && renderTarget.activeCamera !== this.activeCamera;\r\n renderTarget.render(hasSpecialRenderTargetCamera, this.dumpNextRenderTargets);\r\n needRebind = true;\r\n }\r\n }\r\n Tools.EndPerformanceCounter(\"Render targets\", this._renderTargets.length > 0);\r\n\r\n this._renderId++;\r\n }\r\n\r\n for (const step of this._cameraDrawRenderTargetStage) {\r\n needRebind = step.action(this.activeCamera) || needRebind;\r\n }\r\n\r\n this._intermediateRendering = false;\r\n }\r\n\r\n this._engine.currentRenderPassId = camera.outputRenderTarget?.renderPassId ?? camera.renderPassId ?? Constants.RENDERPASS_MAIN;\r\n\r\n // Restore framebuffer after rendering to targets\r\n if (needRebind && !this.prePass) {\r\n this._bindFrameBuffer(this._activeCamera, false);\r\n }\r\n\r\n this.onAfterRenderTargetsRenderObservable.notifyObservers(this);\r\n\r\n // Prepare Frame\r\n if (this.postProcessManager && !camera._multiviewTexture && !this.prePass) {\r\n this.postProcessManager._prepareFrame();\r\n }\r\n\r\n // Before Camera Draw\r\n for (const step of this._beforeCameraDrawStage) {\r\n step.action(this.activeCamera);\r\n }\r\n\r\n // Render\r\n this.onBeforeDrawPhaseObservable.notifyObservers(this);\r\n\r\n if (engine.snapshotRendering && engine.snapshotRenderingMode === Constants.SNAPSHOTRENDERING_FAST) {\r\n this.finalizeSceneUbo();\r\n }\r\n this._renderingManager.render(null, null, true, true);\r\n this.onAfterDrawPhaseObservable.notifyObservers(this);\r\n\r\n // After Camera Draw\r\n for (const step of this._afterCameraDrawStage) {\r\n step.action(this.activeCamera);\r\n }\r\n\r\n // Finalize frame\r\n if (this.postProcessManager && !camera._multiviewTexture) {\r\n // if the camera has an output render target, render the post process to the render target\r\n const texture = camera.outputRenderTarget ? camera.outputRenderTarget.renderTarget! : undefined;\r\n this.postProcessManager._finalizeFrame(camera.isIntermediate, texture);\r\n }\r\n\r\n // After post process\r\n for (const step of this._afterCameraPostProcessStage) {\r\n step.action(this.activeCamera);\r\n }\r\n\r\n // Reset some special arrays\r\n this._renderTargets.reset();\r\n\r\n this.onAfterCameraRenderObservable.notifyObservers(this.activeCamera);\r\n }\r\n\r\n private _processSubCameras(camera: Camera, bindFrameBuffer = true): void {\r\n if (camera.cameraRigMode === Constants.RIG_MODE_NONE || camera._renderingMultiview) {\r\n if (camera._renderingMultiview && !this._multiviewSceneUbo) {\r\n this._createMultiviewUbo();\r\n }\r\n this._renderForCamera(camera, undefined, bindFrameBuffer);\r\n this.onAfterRenderCameraObservable.notifyObservers(camera);\r\n return;\r\n }\r\n\r\n if (camera._useMultiviewToSingleView) {\r\n this._renderMultiviewToSingleView(camera);\r\n } else {\r\n // rig cameras\r\n this.onBeforeCameraRenderObservable.notifyObservers(camera);\r\n for (let index = 0; index < camera._rigCameras.length; index++) {\r\n this._renderForCamera(camera._rigCameras[index], camera);\r\n }\r\n }\r\n\r\n // Use _activeCamera instead of activeCamera to avoid onActiveCameraChanged\r\n this._activeCamera = camera;\r\n this.updateTransformMatrix();\r\n this.onAfterRenderCameraObservable.notifyObservers(camera);\r\n }\r\n\r\n private _checkIntersections(): void {\r\n for (let index = 0; index < this._meshesForIntersections.length; index++) {\r\n const sourceMesh = this._meshesForIntersections.data[index];\r\n\r\n if (!sourceMesh.actionManager) {\r\n continue;\r\n }\r\n\r\n for (let actionIndex = 0; sourceMesh.actionManager && actionIndex < sourceMesh.actionManager.actions.length; actionIndex++) {\r\n const action: IAction = sourceMesh.actionManager.actions[actionIndex];\r\n\r\n if (action.trigger === Constants.ACTION_OnIntersectionEnterTrigger || action.trigger === Constants.ACTION_OnIntersectionExitTrigger) {\r\n const parameters = action.getTriggerParameter();\r\n const otherMesh = parameters.mesh ? parameters.mesh : parameters;\r\n\r\n const areIntersecting = otherMesh.intersectsMesh(sourceMesh, parameters.usePreciseIntersection);\r\n const currentIntersectionInProgress = sourceMesh._intersectionsInProgress.indexOf(otherMesh);\r\n\r\n if (areIntersecting && currentIntersectionInProgress === -1) {\r\n if (action.trigger === Constants.ACTION_OnIntersectionEnterTrigger) {\r\n action._executeCurrent(ActionEvent.CreateNew(sourceMesh, undefined, otherMesh));\r\n sourceMesh._intersectionsInProgress.push(otherMesh);\r\n } else if (action.trigger === Constants.ACTION_OnIntersectionExitTrigger) {\r\n sourceMesh._intersectionsInProgress.push(otherMesh);\r\n }\r\n } else if (!areIntersecting && currentIntersectionInProgress > -1) {\r\n //They intersected, and now they don't.\r\n\r\n //is this trigger an exit trigger? execute an event.\r\n if (action.trigger === Constants.ACTION_OnIntersectionExitTrigger) {\r\n action._executeCurrent(ActionEvent.CreateNew(sourceMesh, undefined, otherMesh));\r\n }\r\n\r\n //if this is an exit trigger, or no exit trigger exists, remove the id from the intersection in progress array.\r\n if (\r\n !sourceMesh.actionManager.hasSpecificTrigger(Constants.ACTION_OnIntersectionExitTrigger, (parameter) => {\r\n const parameterMesh = parameter.mesh ? parameter.mesh : parameter;\r\n return otherMesh === parameterMesh;\r\n }) ||\r\n action.trigger === Constants.ACTION_OnIntersectionExitTrigger\r\n ) {\r\n sourceMesh._intersectionsInProgress.splice(currentIntersectionInProgress, 1);\r\n }\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _advancePhysicsEngineStep(step: number) {\r\n // Do nothing. Code will be replaced if physics engine component is referenced\r\n }\r\n\r\n /**\r\n * User updatable function that will return a deterministic frame time when engine is in deterministic lock step mode\r\n */\r\n public getDeterministicFrameTime: () => number = () => {\r\n return this._engine.getTimeStep();\r\n };\r\n\r\n /** @internal */\r\n public _animate(): void {\r\n // Nothing to do as long as Animatable have not been imported.\r\n }\r\n\r\n /** Execute all animations (for a frame) */\r\n public animate() {\r\n if (this._engine.isDeterministicLockStep()) {\r\n let deltaTime = Math.max(Scene.MinDeltaTime, Math.min(this._engine.getDeltaTime(), Scene.MaxDeltaTime)) + this._timeAccumulator;\r\n\r\n const defaultFrameTime = this._engine.getTimeStep();\r\n const defaultFPS = 1000.0 / defaultFrameTime / 1000.0;\r\n\r\n let stepsTaken = 0;\r\n\r\n const maxSubSteps = this._engine.getLockstepMaxSteps();\r\n\r\n let internalSteps = Math.floor(deltaTime / defaultFrameTime);\r\n internalSteps = Math.min(internalSteps, maxSubSteps);\r\n\r\n while (deltaTime > 0 && stepsTaken < internalSteps) {\r\n this.onBeforeStepObservable.notifyObservers(this);\r\n\r\n // Animations\r\n this._animationRatio = defaultFrameTime * defaultFPS;\r\n this._animate();\r\n this.onAfterAnimationsObservable.notifyObservers(this);\r\n\r\n // Physics\r\n if (this.physicsEnabled) {\r\n this._advancePhysicsEngineStep(defaultFrameTime);\r\n }\r\n\r\n this.onAfterStepObservable.notifyObservers(this);\r\n this._currentStepId++;\r\n\r\n stepsTaken++;\r\n deltaTime -= defaultFrameTime;\r\n }\r\n\r\n this._timeAccumulator = deltaTime < 0 ? 0 : deltaTime;\r\n } else {\r\n // Animations\r\n const deltaTime = this.useConstantAnimationDeltaTime ? 16 : Math.max(Scene.MinDeltaTime, Math.min(this._engine.getDeltaTime(), Scene.MaxDeltaTime));\r\n this._animationRatio = deltaTime * (60.0 / 1000.0);\r\n this._animate();\r\n this.onAfterAnimationsObservable.notifyObservers(this);\r\n\r\n // Physics\r\n if (this.physicsEnabled) {\r\n this._advancePhysicsEngineStep(deltaTime);\r\n }\r\n }\r\n }\r\n\r\n private _clear(): void {\r\n if (this.autoClearDepthAndStencil || this.autoClear) {\r\n this._engine.clear(this.clearColor, this.autoClear || this.forceWireframe || this.forcePointsCloud, this.autoClearDepthAndStencil, this.autoClearDepthAndStencil);\r\n }\r\n }\r\n\r\n private _checkCameraRenderTarget(camera: Nullable) {\r\n if (camera?.outputRenderTarget && !camera?.isRigCamera) {\r\n camera.outputRenderTarget._cleared = false;\r\n }\r\n if (camera?.rigCameras?.length) {\r\n for (let i = 0; i < camera.rigCameras.length; ++i) {\r\n const rtt = camera.rigCameras[i].outputRenderTarget;\r\n if (rtt) {\r\n rtt._cleared = false;\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Resets the draw wrappers cache of all meshes\r\n * @param passId If provided, releases only the draw wrapper corresponding to this render pass id\r\n */\r\n public resetDrawCache(passId?: number): void {\r\n if (!this.meshes) {\r\n return;\r\n }\r\n\r\n for (const mesh of this.meshes) {\r\n mesh.resetDrawCache(passId);\r\n }\r\n }\r\n\r\n /**\r\n * Render the scene\r\n * @param updateCameras defines a boolean indicating if cameras must update according to their inputs (true by default)\r\n * @param ignoreAnimations defines a boolean indicating if animations should not be executed (false by default)\r\n */\r\n public render(updateCameras = true, ignoreAnimations = false): void {\r\n if (this.isDisposed) {\r\n return;\r\n }\r\n\r\n if (this.onReadyObservable.hasObservers() && this._executeWhenReadyTimeoutId === null) {\r\n this._checkIsReady();\r\n }\r\n\r\n this._frameId++;\r\n this._defaultFrameBufferCleared = false;\r\n this._checkCameraRenderTarget(this.activeCamera);\r\n if (this.activeCameras?.length) {\r\n this.activeCameras.forEach(this._checkCameraRenderTarget);\r\n }\r\n\r\n // Register components that have been associated lately to the scene.\r\n this._registerTransientComponents();\r\n\r\n this._activeParticles.fetchNewFrame();\r\n this._totalVertices.fetchNewFrame();\r\n this._activeIndices.fetchNewFrame();\r\n this._activeBones.fetchNewFrame();\r\n this._meshesForIntersections.reset();\r\n this.resetCachedMaterial();\r\n\r\n this.onBeforeAnimationsObservable.notifyObservers(this);\r\n\r\n // Actions\r\n if (this.actionManager) {\r\n this.actionManager.processTrigger(Constants.ACTION_OnEveryFrameTrigger);\r\n }\r\n\r\n // Animations\r\n if (!ignoreAnimations) {\r\n this.animate();\r\n }\r\n\r\n // Before camera update steps\r\n for (const step of this._beforeCameraUpdateStage) {\r\n step.action();\r\n }\r\n\r\n // Update Cameras\r\n if (updateCameras) {\r\n if (this.activeCameras && this.activeCameras.length > 0) {\r\n for (let cameraIndex = 0; cameraIndex < this.activeCameras.length; cameraIndex++) {\r\n const camera = this.activeCameras[cameraIndex];\r\n camera.update();\r\n if (camera.cameraRigMode !== Constants.RIG_MODE_NONE) {\r\n // rig cameras\r\n for (let index = 0; index < camera._rigCameras.length; index++) {\r\n camera._rigCameras[index].update();\r\n }\r\n }\r\n }\r\n } else if (this.activeCamera) {\r\n this.activeCamera.update();\r\n if (this.activeCamera.cameraRigMode !== Constants.RIG_MODE_NONE) {\r\n // rig cameras\r\n for (let index = 0; index < this.activeCamera._rigCameras.length; index++) {\r\n this.activeCamera._rigCameras[index].update();\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Before render\r\n this.onBeforeRenderObservable.notifyObservers(this);\r\n\r\n const engine = this.getEngine();\r\n\r\n // Customs render targets\r\n this.onBeforeRenderTargetsRenderObservable.notifyObservers(this);\r\n\r\n const currentActiveCamera = this.activeCameras?.length ? this.activeCameras[0] : this.activeCamera;\r\n if (this.renderTargetsEnabled) {\r\n Tools.StartPerformanceCounter(\"Custom render targets\", this.customRenderTargets.length > 0);\r\n this._intermediateRendering = true;\r\n for (let customIndex = 0; customIndex < this.customRenderTargets.length; customIndex++) {\r\n const renderTarget = this.customRenderTargets[customIndex];\r\n if (renderTarget._shouldRender()) {\r\n this._renderId++;\r\n\r\n this.activeCamera = renderTarget.activeCamera || this.activeCamera;\r\n\r\n if (!this.activeCamera) {\r\n throw new Error(\"Active camera not set\");\r\n }\r\n\r\n // Viewport\r\n engine.setViewport(this.activeCamera.viewport);\r\n\r\n // Camera\r\n this.updateTransformMatrix();\r\n\r\n renderTarget.render(currentActiveCamera !== this.activeCamera, this.dumpNextRenderTargets);\r\n }\r\n }\r\n Tools.EndPerformanceCounter(\"Custom render targets\", this.customRenderTargets.length > 0);\r\n this._intermediateRendering = false;\r\n this._renderId++;\r\n }\r\n\r\n this._engine.currentRenderPassId = currentActiveCamera?.renderPassId ?? Constants.RENDERPASS_MAIN;\r\n\r\n // Restore back buffer\r\n this.activeCamera = currentActiveCamera;\r\n if (this._activeCamera && this._activeCamera.cameraRigMode !== Constants.RIG_MODE_CUSTOM && !this.prePass) {\r\n this._bindFrameBuffer(this._activeCamera, false);\r\n }\r\n this.onAfterRenderTargetsRenderObservable.notifyObservers(this);\r\n\r\n for (const step of this._beforeClearStage) {\r\n step.action();\r\n }\r\n\r\n // Clear\r\n this._clearFrameBuffer(this.activeCamera);\r\n\r\n // Collects render targets from external components.\r\n for (const step of this._gatherRenderTargetsStage) {\r\n step.action(this._renderTargets);\r\n }\r\n\r\n // Multi-cameras?\r\n if (this.activeCameras && this.activeCameras.length > 0) {\r\n for (let cameraIndex = 0; cameraIndex < this.activeCameras.length; cameraIndex++) {\r\n this._processSubCameras(this.activeCameras[cameraIndex], cameraIndex > 0);\r\n }\r\n } else {\r\n if (!this.activeCamera) {\r\n throw new Error(\"No camera defined\");\r\n }\r\n\r\n this._processSubCameras(this.activeCamera, !!this.activeCamera.outputRenderTarget);\r\n }\r\n\r\n // Intersection checks\r\n this._checkIntersections();\r\n\r\n // Executes the after render stage actions.\r\n for (const step of this._afterRenderStage) {\r\n step.action();\r\n }\r\n\r\n // After render\r\n if (this.afterRender) {\r\n this.afterRender();\r\n }\r\n\r\n this.onAfterRenderObservable.notifyObservers(this);\r\n\r\n // Cleaning\r\n if (this._toBeDisposed.length) {\r\n for (let index = 0; index < this._toBeDisposed.length; index++) {\r\n const data = this._toBeDisposed[index];\r\n if (data) {\r\n data.dispose();\r\n }\r\n }\r\n\r\n this._toBeDisposed.length = 0;\r\n }\r\n\r\n if (this.dumpNextRenderTargets) {\r\n this.dumpNextRenderTargets = false;\r\n }\r\n\r\n this._activeBones.addCount(0, true);\r\n this._activeIndices.addCount(0, true);\r\n this._activeParticles.addCount(0, true);\r\n\r\n this._engine.restoreDefaultFramebuffer();\r\n }\r\n\r\n /**\r\n * Freeze all materials\r\n * A frozen material will not be updatable but should be faster to render\r\n * Note: multimaterials will not be frozen, but their submaterials will\r\n */\r\n public freezeMaterials(): void {\r\n for (let i = 0; i < this.materials.length; i++) {\r\n this.materials[i].freeze();\r\n }\r\n }\r\n\r\n /**\r\n * Unfreeze all materials\r\n * A frozen material will not be updatable but should be faster to render\r\n */\r\n public unfreezeMaterials(): void {\r\n for (let i = 0; i < this.materials.length; i++) {\r\n this.materials[i].unfreeze();\r\n }\r\n }\r\n\r\n /**\r\n * Releases all held resources\r\n */\r\n public dispose(): void {\r\n if (this.isDisposed) {\r\n return;\r\n }\r\n\r\n this.beforeRender = null;\r\n this.afterRender = null;\r\n this.metadata = null;\r\n\r\n this.skeletons.length = 0;\r\n this.morphTargetManagers.length = 0;\r\n this._transientComponents.length = 0;\r\n this._isReadyForMeshStage.clear();\r\n this._beforeEvaluateActiveMeshStage.clear();\r\n this._evaluateSubMeshStage.clear();\r\n this._preActiveMeshStage.clear();\r\n this._cameraDrawRenderTargetStage.clear();\r\n this._beforeCameraDrawStage.clear();\r\n this._beforeRenderTargetDrawStage.clear();\r\n this._beforeRenderingGroupDrawStage.clear();\r\n this._beforeRenderingMeshStage.clear();\r\n this._afterRenderingMeshStage.clear();\r\n this._afterRenderingGroupDrawStage.clear();\r\n this._afterCameraDrawStage.clear();\r\n this._afterRenderTargetDrawStage.clear();\r\n this._afterRenderStage.clear();\r\n this._beforeCameraUpdateStage.clear();\r\n this._beforeClearStage.clear();\r\n this._gatherRenderTargetsStage.clear();\r\n this._gatherActiveCameraRenderTargetsStage.clear();\r\n this._pointerMoveStage.clear();\r\n this._pointerDownStage.clear();\r\n this._pointerUpStage.clear();\r\n\r\n this.importedMeshesFiles = new Array();\r\n\r\n if (this.stopAllAnimations) {\r\n // Ensures that no animatable notifies a callback that could start a new animation group, constantly adding new animatables to the active list...\r\n this._activeAnimatables.forEach((animatable) => {\r\n animatable.onAnimationEndObservable.clear();\r\n animatable.onAnimationEnd = null;\r\n });\r\n this.stopAllAnimations();\r\n }\r\n\r\n this.resetCachedMaterial();\r\n\r\n // Smart arrays\r\n if (this.activeCamera) {\r\n this.activeCamera._activeMeshes.dispose();\r\n this.activeCamera = null;\r\n }\r\n this.activeCameras = null;\r\n\r\n this._activeMeshes.dispose();\r\n this._renderingManager.dispose();\r\n this._processedMaterials.dispose();\r\n this._activeParticleSystems.dispose();\r\n this._activeSkeletons.dispose();\r\n this._softwareSkinnedMeshes.dispose();\r\n this._renderTargets.dispose();\r\n this._materialsRenderTargets.dispose();\r\n this._registeredForLateAnimationBindings.dispose();\r\n this._meshesForIntersections.dispose();\r\n this._toBeDisposed.length = 0;\r\n\r\n // Abort active requests\r\n const activeRequests = this._activeRequests.slice();\r\n for (const request of activeRequests) {\r\n request.abort();\r\n }\r\n this._activeRequests.length = 0;\r\n\r\n // Events\r\n try {\r\n this.onDisposeObservable.notifyObservers(this);\r\n } catch (e) {\r\n console.error(\"An error occurred while calling onDisposeObservable!\", e);\r\n }\r\n\r\n this.detachControl();\r\n\r\n // Detach cameras\r\n const canvas = this._engine.getInputElement();\r\n\r\n if (canvas) {\r\n for (let index = 0; index < this.cameras.length; index++) {\r\n this.cameras[index].detachControl();\r\n }\r\n }\r\n\r\n // Release animation groups\r\n this._disposeList(this.animationGroups);\r\n\r\n // Release lights\r\n this._disposeList(this.lights);\r\n\r\n // Release meshes\r\n this._disposeList(this.meshes, (item) => item.dispose(true));\r\n this._disposeList(this.transformNodes, (item) => item.dispose(true));\r\n\r\n // Release cameras\r\n const cameras = this.cameras;\r\n this._disposeList(cameras);\r\n\r\n // Release materials\r\n if (this._defaultMaterial) {\r\n this._defaultMaterial.dispose();\r\n }\r\n this._disposeList(this.multiMaterials);\r\n this._disposeList(this.materials);\r\n\r\n // Release particles\r\n this._disposeList(this.particleSystems);\r\n\r\n // Release postProcesses\r\n this._disposeList(this.postProcesses);\r\n\r\n // Release textures\r\n this._disposeList(this.textures);\r\n\r\n // Release morph targets\r\n this._disposeList(this.morphTargetManagers);\r\n\r\n // Release UBO\r\n this._sceneUbo.dispose();\r\n\r\n if (this._multiviewSceneUbo) {\r\n this._multiviewSceneUbo.dispose();\r\n }\r\n\r\n // Post-processes\r\n this.postProcessManager.dispose();\r\n\r\n // Components\r\n this._disposeList(this._components);\r\n\r\n // Remove from engine\r\n let index = this._engine.scenes.indexOf(this);\r\n\r\n if (index > -1) {\r\n this._engine.scenes.splice(index, 1);\r\n }\r\n\r\n if (EngineStore._LastCreatedScene === this) {\r\n if (this._engine.scenes.length > 0) {\r\n EngineStore._LastCreatedScene = this._engine.scenes[this._engine.scenes.length - 1];\r\n } else {\r\n EngineStore._LastCreatedScene = null;\r\n }\r\n }\r\n\r\n index = this._engine._virtualScenes.indexOf(this);\r\n\r\n if (index > -1) {\r\n this._engine._virtualScenes.splice(index, 1);\r\n }\r\n\r\n this._engine.wipeCaches(true);\r\n this.onDisposeObservable.clear();\r\n this.onBeforeRenderObservable.clear();\r\n this.onAfterRenderObservable.clear();\r\n this.onBeforeRenderTargetsRenderObservable.clear();\r\n this.onAfterRenderTargetsRenderObservable.clear();\r\n this.onAfterStepObservable.clear();\r\n this.onBeforeStepObservable.clear();\r\n this.onBeforeActiveMeshesEvaluationObservable.clear();\r\n this.onAfterActiveMeshesEvaluationObservable.clear();\r\n this.onBeforeParticlesRenderingObservable.clear();\r\n this.onAfterParticlesRenderingObservable.clear();\r\n this.onBeforeDrawPhaseObservable.clear();\r\n this.onAfterDrawPhaseObservable.clear();\r\n this.onBeforeAnimationsObservable.clear();\r\n this.onAfterAnimationsObservable.clear();\r\n this.onDataLoadedObservable.clear();\r\n this.onBeforeRenderingGroupObservable.clear();\r\n this.onAfterRenderingGroupObservable.clear();\r\n this.onMeshImportedObservable.clear();\r\n this.onBeforeCameraRenderObservable.clear();\r\n this.onAfterCameraRenderObservable.clear();\r\n this.onAfterRenderCameraObservable.clear();\r\n this.onReadyObservable.clear();\r\n this.onNewCameraAddedObservable.clear();\r\n this.onCameraRemovedObservable.clear();\r\n this.onNewLightAddedObservable.clear();\r\n this.onLightRemovedObservable.clear();\r\n this.onNewGeometryAddedObservable.clear();\r\n this.onGeometryRemovedObservable.clear();\r\n this.onNewTransformNodeAddedObservable.clear();\r\n this.onTransformNodeRemovedObservable.clear();\r\n this.onNewMeshAddedObservable.clear();\r\n this.onMeshRemovedObservable.clear();\r\n this.onNewSkeletonAddedObservable.clear();\r\n this.onSkeletonRemovedObservable.clear();\r\n this.onNewMaterialAddedObservable.clear();\r\n this.onNewMultiMaterialAddedObservable.clear();\r\n this.onMaterialRemovedObservable.clear();\r\n this.onMultiMaterialRemovedObservable.clear();\r\n this.onNewTextureAddedObservable.clear();\r\n this.onTextureRemovedObservable.clear();\r\n this.onPrePointerObservable.clear();\r\n this.onPointerObservable.clear();\r\n this.onPreKeyboardObservable.clear();\r\n this.onKeyboardObservable.clear();\r\n this.onActiveCameraChanged.clear();\r\n this.onScenePerformancePriorityChangedObservable.clear();\r\n this._isDisposed = true;\r\n }\r\n\r\n private _disposeList(items: T[], callback?: (item: T) => void): void {\r\n const itemsCopy = items.slice(0);\r\n callback = callback ?? ((item) => item.dispose());\r\n for (const item of itemsCopy) {\r\n callback(item);\r\n }\r\n items.length = 0;\r\n }\r\n\r\n /**\r\n * Gets if the scene is already disposed\r\n */\r\n public get isDisposed(): boolean {\r\n return this._isDisposed;\r\n }\r\n\r\n /**\r\n * Call this function to reduce memory footprint of the scene.\r\n * Vertex buffers will not store CPU data anymore (this will prevent picking, collisions or physics to work correctly)\r\n */\r\n public clearCachedVertexData(): void {\r\n for (let meshIndex = 0; meshIndex < this.meshes.length; meshIndex++) {\r\n const mesh = this.meshes[meshIndex];\r\n const geometry = (mesh).geometry;\r\n\r\n if (geometry) {\r\n geometry.clearCachedData();\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * This function will remove the local cached buffer data from texture.\r\n * It will save memory but will prevent the texture from being rebuilt\r\n */\r\n public cleanCachedTextureBuffer(): void {\r\n for (const baseTexture of this.textures) {\r\n const buffer = (baseTexture)._buffer;\r\n\r\n if (buffer) {\r\n (baseTexture)._buffer = null;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Get the world extend vectors with an optional filter\r\n *\r\n * @param filterPredicate the predicate - which meshes should be included when calculating the world size\r\n * @returns {{ min: Vector3; max: Vector3 }} min and max vectors\r\n */\r\n public getWorldExtends(filterPredicate?: (mesh: AbstractMesh) => boolean): { min: Vector3; max: Vector3 } {\r\n const min = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\r\n const max = new Vector3(-Number.MAX_VALUE, -Number.MAX_VALUE, -Number.MAX_VALUE);\r\n filterPredicate = filterPredicate || (() => true);\r\n this.meshes.filter(filterPredicate).forEach((mesh) => {\r\n mesh.computeWorldMatrix(true);\r\n\r\n if (!mesh.subMeshes || mesh.subMeshes.length === 0 || mesh.infiniteDistance) {\r\n return;\r\n }\r\n\r\n const boundingInfo = mesh.getBoundingInfo();\r\n\r\n const minBox = boundingInfo.boundingBox.minimumWorld;\r\n const maxBox = boundingInfo.boundingBox.maximumWorld;\r\n\r\n Vector3.CheckExtends(minBox, min, max);\r\n Vector3.CheckExtends(maxBox, min, max);\r\n });\r\n\r\n return {\r\n min: min,\r\n max: max,\r\n };\r\n }\r\n\r\n // Picking\r\n\r\n /**\r\n * Creates a ray that can be used to pick in the scene\r\n * @param x defines the x coordinate of the origin (on-screen)\r\n * @param y defines the y coordinate of the origin (on-screen)\r\n * @param world defines the world matrix to use if you want to pick in object space (instead of world space)\r\n * @param camera defines the camera to use for the picking\r\n * @param cameraViewSpace defines if picking will be done in view space (false by default)\r\n * @returns a Ray\r\n */\r\n public createPickingRay(x: number, y: number, world: Nullable, camera: Nullable, cameraViewSpace = false): Ray {\r\n throw _WarnImport(\"Ray\");\r\n }\r\n\r\n /**\r\n * Creates a ray that can be used to pick in the scene\r\n * @param x defines the x coordinate of the origin (on-screen)\r\n * @param y defines the y coordinate of the origin (on-screen)\r\n * @param world defines the world matrix to use if you want to pick in object space (instead of world space)\r\n * @param result defines the ray where to store the picking ray\r\n * @param camera defines the camera to use for the picking\r\n * @param cameraViewSpace defines if picking will be done in view space (false by default)\r\n * @param enableDistantPicking defines if picking should handle large values for mesh position/scaling (false by default)\r\n * @returns the current scene\r\n */\r\n public createPickingRayToRef(\r\n x: number,\r\n y: number,\r\n world: Nullable,\r\n result: Ray,\r\n camera: Nullable,\r\n cameraViewSpace = false,\r\n enableDistantPicking = false\r\n ): Scene {\r\n throw _WarnImport(\"Ray\");\r\n }\r\n\r\n /**\r\n * Creates a ray that can be used to pick in the scene\r\n * @param x defines the x coordinate of the origin (on-screen)\r\n * @param y defines the y coordinate of the origin (on-screen)\r\n * @param camera defines the camera to use for the picking\r\n * @returns a Ray\r\n */\r\n public createPickingRayInCameraSpace(x: number, y: number, camera?: Camera): Ray {\r\n throw _WarnImport(\"Ray\");\r\n }\r\n\r\n /**\r\n * Creates a ray that can be used to pick in the scene\r\n * @param x defines the x coordinate of the origin (on-screen)\r\n * @param y defines the y coordinate of the origin (on-screen)\r\n * @param result defines the ray where to store the picking ray\r\n * @param camera defines the camera to use for the picking\r\n * @returns the current scene\r\n */\r\n public createPickingRayInCameraSpaceToRef(x: number, y: number, result: Ray, camera?: Camera): Scene {\r\n throw _WarnImport(\"Ray\");\r\n }\r\n\r\n /** @internal */\r\n public get _pickingAvailable(): boolean {\r\n return false;\r\n }\r\n\r\n /** @internal */\r\n public _registeredActions: number = 0;\r\n\r\n /** Launch a ray to try to pick a mesh in the scene\r\n * @param x position on screen\r\n * @param y position on screen\r\n * @param predicate Predicate function used to determine eligible meshes. Can be set to null. In this case, a mesh must be enabled, visible and with isPickable set to true\r\n * @param fastCheck defines if the first intersection will be used (and not the closest)\r\n * @param camera to use for computing the picking ray. Can be set to null. In this case, the scene.activeCamera will be used\r\n * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected\r\n * @returns a PickingInfo\r\n */\r\n public pick(\r\n x: number,\r\n y: number,\r\n predicate?: (mesh: AbstractMesh) => boolean,\r\n fastCheck?: boolean,\r\n camera?: Nullable,\r\n trianglePredicate?: TrianglePickingPredicate\r\n ): PickingInfo {\r\n // Dummy info if picking as not been imported\r\n return new PickingInfo();\r\n }\r\n\r\n /** Launch a ray to try to pick a mesh in the scene using only bounding information of the main mesh (not using submeshes)\r\n * @param x position on screen\r\n * @param y position on screen\r\n * @param predicate Predicate function used to determine eligible meshes. Can be set to null. In this case, a mesh must be enabled, visible and with isPickable set to true\r\n * @param fastCheck defines if the first intersection will be used (and not the closest)\r\n * @param camera to use for computing the picking ray. Can be set to null. In this case, the scene.activeCamera will be used\r\n * @returns a PickingInfo (Please note that some info will not be set like distance, bv, bu and everything that cannot be capture by only using bounding infos)\r\n */\r\n public pickWithBoundingInfo(x: number, y: number, predicate?: (mesh: AbstractMesh) => boolean, fastCheck?: boolean, camera?: Nullable): Nullable {\r\n // Dummy info if picking as not been imported\r\n return new PickingInfo();\r\n }\r\n\r\n /**\r\n * Use the given ray to pick a mesh in the scene. A mesh triangle can be picked both from its front and back sides,\r\n * irrespective of orientation.\r\n * @param ray The ray to use to pick meshes\r\n * @param predicate Predicate function used to determine eligible meshes. Can be set to null. In this case, a mesh must have isPickable set to true\r\n * @param fastCheck defines if the first intersection will be used (and not the closest)\r\n * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected\r\n * @returns a PickingInfo\r\n */\r\n public pickWithRay(ray: Ray, predicate?: (mesh: AbstractMesh) => boolean, fastCheck?: boolean, trianglePredicate?: TrianglePickingPredicate): Nullable {\r\n throw _WarnImport(\"Ray\");\r\n }\r\n\r\n /**\r\n * Launch a ray to try to pick a mesh in the scene. A mesh triangle can be picked both from its front and back sides,\r\n * irrespective of orientation.\r\n * @param x X position on screen\r\n * @param y Y position on screen\r\n * @param predicate Predicate function used to determine eligible meshes. Can be set to null. In this case, a mesh must be enabled, visible and with isPickable set to true\r\n * @param camera camera to use for computing the picking ray. Can be set to null. In this case, the scene.activeCamera will be used\r\n * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected\r\n * @returns an array of PickingInfo\r\n */\r\n public multiPick(x: number, y: number, predicate?: (mesh: AbstractMesh) => boolean, camera?: Camera, trianglePredicate?: TrianglePickingPredicate): Nullable {\r\n throw _WarnImport(\"Ray\");\r\n }\r\n\r\n /**\r\n * Launch a ray to try to pick a mesh in the scene\r\n * @param ray Ray to use\r\n * @param predicate Predicate function used to determine eligible meshes. Can be set to null. In this case, a mesh must be enabled, visible and with isPickable set to true\r\n * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected\r\n * @returns an array of PickingInfo\r\n */\r\n public multiPickWithRay(ray: Ray, predicate?: (mesh: AbstractMesh) => boolean, trianglePredicate?: TrianglePickingPredicate): Nullable {\r\n throw _WarnImport(\"Ray\");\r\n }\r\n\r\n /**\r\n * Force the value of meshUnderPointer\r\n * @param mesh defines the mesh to use\r\n * @param pointerId optional pointer id when using more than one pointer\r\n * @param pickResult optional pickingInfo data used to find mesh\r\n */\r\n public setPointerOverMesh(mesh: Nullable, pointerId?: number, pickResult?: Nullable): void {\r\n this._inputManager.setPointerOverMesh(mesh, pointerId, pickResult);\r\n }\r\n\r\n /**\r\n * Gets the mesh under the pointer\r\n * @returns a Mesh or null if no mesh is under the pointer\r\n */\r\n public getPointerOverMesh(): Nullable {\r\n return this._inputManager.getPointerOverMesh();\r\n }\r\n\r\n // Misc.\r\n /** @internal */\r\n public _rebuildGeometries(): void {\r\n for (const geometry of this.geometries) {\r\n geometry._rebuild();\r\n }\r\n\r\n for (const mesh of this.meshes) {\r\n mesh._rebuild();\r\n }\r\n\r\n if (this.postProcessManager) {\r\n this.postProcessManager._rebuild();\r\n }\r\n\r\n for (const component of this._components) {\r\n component.rebuild();\r\n }\r\n\r\n for (const system of this.particleSystems) {\r\n system.rebuild();\r\n }\r\n\r\n if (this.spriteManagers) {\r\n for (const spriteMgr of this.spriteManagers) {\r\n spriteMgr.rebuild();\r\n }\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _rebuildTextures(): void {\r\n for (const texture of this.textures) {\r\n texture._rebuild();\r\n }\r\n\r\n this.markAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n // Tags\r\n private _getByTags(list: any[], tagsQuery: string, forEach?: (item: any) => void): any[] {\r\n if (tagsQuery === undefined) {\r\n // returns the complete list (could be done with Tags.MatchesQuery but no need to have a for-loop here)\r\n return list;\r\n }\r\n\r\n const listByTags = [];\r\n\r\n forEach =\r\n forEach ||\r\n ((item: any) => {\r\n return;\r\n });\r\n\r\n for (const i in list) {\r\n const item = list[i];\r\n if (Tags && Tags.MatchesQuery(item, tagsQuery)) {\r\n listByTags.push(item);\r\n forEach(item);\r\n }\r\n }\r\n\r\n return listByTags;\r\n }\r\n\r\n /**\r\n * Get a list of meshes by tags\r\n * @param tagsQuery defines the tags query to use\r\n * @param forEach defines a predicate used to filter results\r\n * @returns an array of Mesh\r\n */\r\n public getMeshesByTags(tagsQuery: string, forEach?: (mesh: AbstractMesh) => void): Mesh[] {\r\n return this._getByTags(this.meshes, tagsQuery, forEach);\r\n }\r\n\r\n /**\r\n * Get a list of cameras by tags\r\n * @param tagsQuery defines the tags query to use\r\n * @param forEach defines a predicate used to filter results\r\n * @returns an array of Camera\r\n */\r\n public getCamerasByTags(tagsQuery: string, forEach?: (camera: Camera) => void): Camera[] {\r\n return this._getByTags(this.cameras, tagsQuery, forEach);\r\n }\r\n\r\n /**\r\n * Get a list of lights by tags\r\n * @param tagsQuery defines the tags query to use\r\n * @param forEach defines a predicate used to filter results\r\n * @returns an array of Light\r\n */\r\n public getLightsByTags(tagsQuery: string, forEach?: (light: Light) => void): Light[] {\r\n return this._getByTags(this.lights, tagsQuery, forEach);\r\n }\r\n\r\n /**\r\n * Get a list of materials by tags\r\n * @param tagsQuery defines the tags query to use\r\n * @param forEach defines a predicate used to filter results\r\n * @returns an array of Material\r\n */\r\n public getMaterialByTags(tagsQuery: string, forEach?: (material: Material) => void): Material[] {\r\n return this._getByTags(this.materials, tagsQuery, forEach).concat(this._getByTags(this.multiMaterials, tagsQuery, forEach));\r\n }\r\n\r\n /**\r\n * Get a list of transform nodes by tags\r\n * @param tagsQuery defines the tags query to use\r\n * @param forEach defines a predicate used to filter results\r\n * @returns an array of TransformNode\r\n */\r\n public getTransformNodesByTags(tagsQuery: string, forEach?: (transform: TransformNode) => void): TransformNode[] {\r\n return this._getByTags(this.transformNodes, tagsQuery, forEach);\r\n }\r\n\r\n /**\r\n * Overrides the default sort function applied in the rendering group to prepare the meshes.\r\n * This allowed control for front to back rendering or reversly depending of the special needs.\r\n *\r\n * @param renderingGroupId The rendering group id corresponding to its index\r\n * @param opaqueSortCompareFn The opaque queue comparison function use to sort.\r\n * @param alphaTestSortCompareFn The alpha test queue comparison function use to sort.\r\n * @param transparentSortCompareFn The transparent queue comparison function use to sort.\r\n */\r\n public setRenderingOrder(\r\n renderingGroupId: number,\r\n opaqueSortCompareFn: Nullable<(a: SubMesh, b: SubMesh) => number> = null,\r\n alphaTestSortCompareFn: Nullable<(a: SubMesh, b: SubMesh) => number> = null,\r\n transparentSortCompareFn: Nullable<(a: SubMesh, b: SubMesh) => number> = null\r\n ): void {\r\n this._renderingManager.setRenderingOrder(renderingGroupId, opaqueSortCompareFn, alphaTestSortCompareFn, transparentSortCompareFn);\r\n }\r\n\r\n /**\r\n * Specifies whether or not the stencil and depth buffer are cleared between two rendering groups.\r\n *\r\n * @param renderingGroupId The rendering group id corresponding to its index\r\n * @param autoClearDepthStencil Automatically clears depth and stencil between groups if true.\r\n * @param depth Automatically clears depth between groups if true and autoClear is true.\r\n * @param stencil Automatically clears stencil between groups if true and autoClear is true.\r\n */\r\n public setRenderingAutoClearDepthStencil(renderingGroupId: number, autoClearDepthStencil: boolean, depth = true, stencil = true): void {\r\n this._renderingManager.setRenderingAutoClearDepthStencil(renderingGroupId, autoClearDepthStencil, depth, stencil);\r\n }\r\n\r\n /**\r\n * Gets the current auto clear configuration for one rendering group of the rendering\r\n * manager.\r\n * @param index the rendering group index to get the information for\r\n * @returns The auto clear setup for the requested rendering group\r\n */\r\n public getAutoClearDepthStencilSetup(index: number): IRenderingManagerAutoClearSetup {\r\n return this._renderingManager.getAutoClearDepthStencilSetup(index);\r\n }\r\n\r\n private _blockMaterialDirtyMechanism = false;\r\n\r\n /** Gets or sets a boolean blocking all the calls to markAllMaterialsAsDirty (ie. the materials won't be updated if they are out of sync) */\r\n public get blockMaterialDirtyMechanism(): boolean {\r\n return this._blockMaterialDirtyMechanism;\r\n }\r\n\r\n public set blockMaterialDirtyMechanism(value: boolean) {\r\n if (this._blockMaterialDirtyMechanism === value) {\r\n return;\r\n }\r\n\r\n this._blockMaterialDirtyMechanism = value;\r\n\r\n if (!value) {\r\n // Do a complete update\r\n this.markAllMaterialsAsDirty(Constants.MATERIAL_AllDirtyFlag);\r\n }\r\n }\r\n\r\n /**\r\n * Will flag all materials as dirty to trigger new shader compilation\r\n * @param flag defines the flag used to specify which material part must be marked as dirty\r\n * @param predicate If not null, it will be used to specify if a material has to be marked as dirty\r\n */\r\n public markAllMaterialsAsDirty(flag: number, predicate?: (mat: Material) => boolean): void {\r\n if (this._blockMaterialDirtyMechanism) {\r\n return;\r\n }\r\n\r\n for (const material of this.materials) {\r\n if (predicate && !predicate(material)) {\r\n continue;\r\n }\r\n material.markAsDirty(flag);\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _loadFile(\r\n fileOrUrl: File | string,\r\n onSuccess: (data: string | ArrayBuffer, responseURL?: string) => void,\r\n onProgress?: (ev: ProgressEvent) => void,\r\n useOfflineSupport?: boolean,\r\n useArrayBuffer?: boolean,\r\n onError?: (request?: WebRequest, exception?: LoadFileError) => void,\r\n onOpened?: (request: WebRequest) => void\r\n ): IFileRequest {\r\n const request = LoadFile(fileOrUrl, onSuccess, onProgress, useOfflineSupport ? this.offlineProvider : undefined, useArrayBuffer, onError, onOpened);\r\n this._activeRequests.push(request);\r\n request.onCompleteObservable.add((request) => {\r\n this._activeRequests.splice(this._activeRequests.indexOf(request), 1);\r\n });\r\n return request;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _loadFileAsync(\r\n fileOrUrl: File | string,\r\n onProgress?: (data: any) => void,\r\n useOfflineSupport?: boolean,\r\n useArrayBuffer?: boolean,\r\n onOpened?: (request: WebRequest) => void\r\n ): Promise {\r\n return new Promise((resolve, reject) => {\r\n this._loadFile(\r\n fileOrUrl,\r\n (data) => {\r\n resolve(data);\r\n },\r\n onProgress,\r\n useOfflineSupport,\r\n useArrayBuffer,\r\n (request, exception) => {\r\n reject(exception);\r\n },\r\n onOpened\r\n );\r\n });\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _requestFile(\r\n url: string,\r\n onSuccess: (data: string | ArrayBuffer, request?: WebRequest) => void,\r\n onProgress?: (ev: ProgressEvent) => void,\r\n useOfflineSupport?: boolean,\r\n useArrayBuffer?: boolean,\r\n onError?: (error: RequestFileError) => void,\r\n onOpened?: (request: WebRequest) => void\r\n ): IFileRequest {\r\n const request = RequestFile(url, onSuccess, onProgress, useOfflineSupport ? this.offlineProvider : undefined, useArrayBuffer, onError, onOpened);\r\n this._activeRequests.push(request);\r\n request.onCompleteObservable.add((request) => {\r\n this._activeRequests.splice(this._activeRequests.indexOf(request), 1);\r\n });\r\n return request;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _requestFileAsync(\r\n url: string,\r\n onProgress?: (ev: ProgressEvent) => void,\r\n useOfflineSupport?: boolean,\r\n useArrayBuffer?: boolean,\r\n onOpened?: (request: WebRequest) => void\r\n ): Promise {\r\n return new Promise((resolve, reject) => {\r\n this._requestFile(\r\n url,\r\n (data) => {\r\n resolve(data);\r\n },\r\n onProgress,\r\n useOfflineSupport,\r\n useArrayBuffer,\r\n (error) => {\r\n reject(error);\r\n },\r\n onOpened\r\n );\r\n });\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _readFile(\r\n file: File,\r\n onSuccess: (data: string | ArrayBuffer) => void,\r\n onProgress?: (ev: ProgressEvent) => any,\r\n useArrayBuffer?: boolean,\r\n onError?: (error: ReadFileError) => void\r\n ): IFileRequest {\r\n const request = ReadFile(file, onSuccess, onProgress, useArrayBuffer, onError);\r\n this._activeRequests.push(request);\r\n request.onCompleteObservable.add((request) => {\r\n this._activeRequests.splice(this._activeRequests.indexOf(request), 1);\r\n });\r\n return request;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _readFileAsync(file: File, onProgress?: (ev: ProgressEvent) => any, useArrayBuffer?: boolean): Promise {\r\n return new Promise((resolve, reject) => {\r\n this._readFile(\r\n file,\r\n (data) => {\r\n resolve(data);\r\n },\r\n onProgress,\r\n useArrayBuffer,\r\n (error) => {\r\n reject(error);\r\n }\r\n );\r\n });\r\n }\r\n\r\n /**\r\n * Internal perfCollector instance used for sharing between inspector and playground.\r\n * Marked as protected to allow sharing between prototype extensions, but disallow access at toplevel.\r\n */\r\n protected _perfCollector: Nullable = null;\r\n\r\n /**\r\n * This method gets the performance collector belonging to the scene, which is generally shared with the inspector.\r\n * @returns the perf collector belonging to the scene.\r\n */\r\n public getPerfCollector(): PerformanceViewerCollector {\r\n throw _WarnImport(\"performanceViewerSceneExtension\");\r\n }\r\n\r\n // deprecated\r\n\r\n /**\r\n * Sets the active camera of the scene using its Id\r\n * @param id defines the camera's Id\r\n * @returns the new active camera or null if none found.\r\n * @deprecated Please use setActiveCameraById instead\r\n */\r\n setActiveCameraByID(id: string): Nullable {\r\n return this.setActiveCameraById(id);\r\n }\r\n /**\r\n * Get a material using its id\r\n * @param id defines the material's Id\r\n * @returns the material or null if none found.\r\n * @deprecated Please use getMaterialById instead\r\n */\r\n getMaterialByID(id: string): Nullable {\r\n return this.getMaterialById(id);\r\n }\r\n /**\r\n * Gets a the last added material using a given id\r\n * @param id defines the material's Id\r\n * @returns the last material with the given id or null if none found.\r\n * @deprecated Please use getLastMaterialById instead\r\n */\r\n getLastMaterialByID(id: string): Nullable {\r\n return this.getLastMaterialById(id);\r\n }\r\n\r\n /**\r\n * Get a texture using its unique id\r\n * @param uniqueId defines the texture's unique id\r\n * @returns the texture or null if none found.\r\n * @deprecated Please use getTextureByUniqueId instead\r\n */\r\n getTextureByUniqueID(uniqueId: number): Nullable {\r\n return this.getTextureByUniqueId(uniqueId);\r\n }\r\n /**\r\n * Gets a camera using its Id\r\n * @param id defines the Id to look for\r\n * @returns the camera or null if not found\r\n * @deprecated Please use getCameraById instead\r\n */\r\n getCameraByID(id: string): Nullable {\r\n return this.getCameraById(id);\r\n }\r\n /**\r\n * Gets a camera using its unique Id\r\n * @param uniqueId defines the unique Id to look for\r\n * @returns the camera or null if not found\r\n * @deprecated Please use getCameraByUniqueId instead\r\n */\r\n getCameraByUniqueID(uniqueId: number): Nullable {\r\n return this.getCameraByUniqueId(uniqueId);\r\n }\r\n /**\r\n * Gets a bone using its Id\r\n * @param id defines the bone's Id\r\n * @returns the bone or null if not found\r\n * @deprecated Please use getBoneById instead\r\n */\r\n getBoneByID(id: string): Nullable {\r\n return this.getBoneById(id);\r\n }\r\n /**\r\n * Gets a light node using its Id\r\n * @param id defines the light's Id\r\n * @returns the light or null if none found.\r\n * @deprecated Please use getLightById instead\r\n */\r\n getLightByID(id: string): Nullable {\r\n return this.getLightById(id);\r\n }\r\n /**\r\n * Gets a light node using its scene-generated unique Id\r\n * @param uniqueId defines the light's unique Id\r\n * @returns the light or null if none found.\r\n * @deprecated Please use getLightByUniqueId instead\r\n */\r\n getLightByUniqueID(uniqueId: number): Nullable {\r\n return this.getLightByUniqueId(uniqueId);\r\n }\r\n /**\r\n * Gets a particle system by Id\r\n * @param id defines the particle system Id\r\n * @returns the corresponding system or null if none found\r\n * @deprecated Please use getParticleSystemById instead\r\n */\r\n getParticleSystemByID(id: string): Nullable {\r\n return this.getParticleSystemById(id);\r\n }\r\n /**\r\n * Gets a geometry using its Id\r\n * @param id defines the geometry's Id\r\n * @returns the geometry or null if none found.\r\n * @deprecated Please use getGeometryById instead\r\n */\r\n getGeometryByID(id: string): Nullable {\r\n return this.getGeometryById(id);\r\n }\r\n /**\r\n * Gets the first added mesh found of a given Id\r\n * @param id defines the Id to search for\r\n * @returns the mesh found or null if not found at all\r\n * @deprecated Please use getMeshById instead\r\n */\r\n getMeshByID(id: string): Nullable {\r\n return this.getMeshById(id);\r\n }\r\n /**\r\n * Gets a mesh with its auto-generated unique Id\r\n * @param uniqueId defines the unique Id to search for\r\n * @returns the found mesh or null if not found at all.\r\n * @deprecated Please use getMeshByUniqueId instead\r\n */\r\n getMeshByUniqueID(uniqueId: number): Nullable {\r\n return this.getMeshByUniqueId(uniqueId);\r\n }\r\n /**\r\n * Gets a the last added mesh using a given Id\r\n * @param id defines the Id to search for\r\n * @returns the found mesh or null if not found at all.\r\n * @deprecated Please use getLastMeshById instead\r\n */\r\n getLastMeshByID(id: string): Nullable {\r\n return this.getLastMeshById(id);\r\n }\r\n /**\r\n * Gets a list of meshes using their Id\r\n * @param id defines the Id to search for\r\n * @returns a list of meshes\r\n * @deprecated Please use getMeshesById instead\r\n */\r\n getMeshesByID(id: string): Array {\r\n return this.getMeshesById(id);\r\n }\r\n /**\r\n * Gets the first added transform node found of a given Id\r\n * @param id defines the Id to search for\r\n * @returns the found transform node or null if not found at all.\r\n * @deprecated Please use getTransformNodeById instead\r\n */\r\n getTransformNodeByID(id: string): Nullable {\r\n return this.getTransformNodeById(id);\r\n }\r\n /**\r\n * Gets a transform node with its auto-generated unique Id\r\n * @param uniqueId defines the unique Id to search for\r\n * @returns the found transform node or null if not found at all.\r\n * @deprecated Please use getTransformNodeByUniqueId instead\r\n */\r\n getTransformNodeByUniqueID(uniqueId: number): Nullable {\r\n return this.getTransformNodeByUniqueId(uniqueId);\r\n }\r\n /**\r\n * Gets a list of transform nodes using their Id\r\n * @param id defines the Id to search for\r\n * @returns a list of transform nodes\r\n * @deprecated Please use getTransformNodesById instead\r\n */\r\n getTransformNodesByID(id: string): Array {\r\n return this.getTransformNodesById(id);\r\n }\r\n /**\r\n * Gets a node (Mesh, Camera, Light) using a given Id\r\n * @param id defines the Id to search for\r\n * @returns the found node or null if not found at all\r\n * @deprecated Please use getNodeById instead\r\n */\r\n getNodeByID(id: string): Nullable {\r\n return this.getNodeById(id);\r\n }\r\n /**\r\n * Gets a the last added node (Mesh, Camera, Light) using a given Id\r\n * @param id defines the Id to search for\r\n * @returns the found node or null if not found at all\r\n * @deprecated Please use getLastEntryById instead\r\n */\r\n getLastEntryByID(id: string): Nullable {\r\n return this.getLastEntryById(id);\r\n }\r\n /**\r\n * Gets a skeleton using a given Id (if many are found, this function will pick the last one)\r\n * @param id defines the Id to search for\r\n * @returns the found skeleton or null if not found at all.\r\n * @deprecated Please use getLastSkeletonById instead\r\n */\r\n getLastSkeletonByID(id: string): Nullable {\r\n return this.getLastSkeletonById(id);\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\n\r\n/**\r\n * @internal\r\n */\r\nexport class IntersectionInfo {\r\n public faceId = 0;\r\n public subMeshId = 0;\r\n\r\n constructor(public bu: Nullable, public bv: Nullable, public distance: number) {}\r\n}\r\n","import type { DeepImmutable, Nullable } from \"../types\";\r\nimport { ArrayTools } from \"../Misc/arrayTools\";\r\nimport { Matrix, Vector3 } from \"../Maths/math.vector\";\r\nimport type { BoundingSphere } from \"../Culling/boundingSphere\";\r\n\r\nimport type { ICullable } from \"./boundingInfo\";\r\nimport { Epsilon } from \"../Maths/math.constants\";\r\nimport type { Plane } from \"../Maths/math.plane\";\r\n\r\nimport type { DrawWrapper } from \"../Materials/drawWrapper\";\r\n\r\n/**\r\n * Class used to store bounding box information\r\n */\r\nexport class BoundingBox implements ICullable {\r\n /**\r\n * Gets the 8 vectors representing the bounding box in local space\r\n */\r\n public readonly vectors: Vector3[] = ArrayTools.BuildArray(8, Vector3.Zero);\r\n /**\r\n * Gets the center of the bounding box in local space\r\n */\r\n public readonly center: Vector3 = Vector3.Zero();\r\n /**\r\n * Gets the center of the bounding box in world space\r\n */\r\n public readonly centerWorld: Vector3 = Vector3.Zero();\r\n /**\r\n * Gets the extend size in local space\r\n */\r\n public readonly extendSize: Vector3 = Vector3.Zero();\r\n /**\r\n * Gets the extend size in world space\r\n */\r\n public readonly extendSizeWorld: Vector3 = Vector3.Zero();\r\n /**\r\n * Gets the OBB (object bounding box) directions\r\n */\r\n public readonly directions: Vector3[] = ArrayTools.BuildArray(3, Vector3.Zero);\r\n /**\r\n * Gets the 8 vectors representing the bounding box in world space\r\n */\r\n public readonly vectorsWorld: Vector3[] = ArrayTools.BuildArray(8, Vector3.Zero);\r\n /**\r\n * Gets the minimum vector in world space\r\n */\r\n public readonly minimumWorld: Vector3 = Vector3.Zero();\r\n /**\r\n * Gets the maximum vector in world space\r\n */\r\n public readonly maximumWorld: Vector3 = Vector3.Zero();\r\n /**\r\n * Gets the minimum vector in local space\r\n */\r\n public readonly minimum: Vector3 = Vector3.Zero();\r\n /**\r\n * Gets the maximum vector in local space\r\n */\r\n public readonly maximum: Vector3 = Vector3.Zero();\r\n\r\n private _worldMatrix: DeepImmutable;\r\n private static readonly _TmpVector3 = ArrayTools.BuildArray(3, Vector3.Zero);\r\n\r\n /**\r\n * @internal\r\n */\r\n public _tag: number;\r\n\r\n /** @internal */\r\n public _drawWrapperFront: Nullable = null;\r\n /** @internal */\r\n public _drawWrapperBack: Nullable = null;\r\n\r\n /**\r\n * Creates a new bounding box\r\n * @param min defines the minimum vector (in local space)\r\n * @param max defines the maximum vector (in local space)\r\n * @param worldMatrix defines the new world matrix\r\n */\r\n constructor(min: DeepImmutable, max: DeepImmutable, worldMatrix?: DeepImmutable) {\r\n this.reConstruct(min, max, worldMatrix);\r\n }\r\n\r\n // Methods\r\n\r\n /**\r\n * Recreates the entire bounding box from scratch as if we call the constructor in place\r\n * @param min defines the new minimum vector (in local space)\r\n * @param max defines the new maximum vector (in local space)\r\n * @param worldMatrix defines the new world matrix\r\n */\r\n public reConstruct(min: DeepImmutable, max: DeepImmutable, worldMatrix?: DeepImmutable) {\r\n const minX = min.x,\r\n minY = min.y,\r\n minZ = min.z,\r\n maxX = max.x,\r\n maxY = max.y,\r\n maxZ = max.z;\r\n const vectors = this.vectors;\r\n\r\n this.minimum.copyFromFloats(minX, minY, minZ);\r\n this.maximum.copyFromFloats(maxX, maxY, maxZ);\r\n vectors[0].copyFromFloats(minX, minY, minZ);\r\n vectors[1].copyFromFloats(maxX, maxY, maxZ);\r\n vectors[2].copyFromFloats(maxX, minY, minZ);\r\n vectors[3].copyFromFloats(minX, maxY, minZ);\r\n vectors[4].copyFromFloats(minX, minY, maxZ);\r\n vectors[5].copyFromFloats(maxX, maxY, minZ);\r\n vectors[6].copyFromFloats(minX, maxY, maxZ);\r\n vectors[7].copyFromFloats(maxX, minY, maxZ);\r\n\r\n // OBB\r\n max.addToRef(min, this.center).scaleInPlace(0.5);\r\n max.subtractToRef(min, this.extendSize).scaleInPlace(0.5);\r\n\r\n this._worldMatrix = worldMatrix || Matrix.IdentityReadOnly;\r\n\r\n this._update(this._worldMatrix);\r\n }\r\n\r\n /**\r\n * Scale the current bounding box by applying a scale factor\r\n * @param factor defines the scale factor to apply\r\n * @returns the current bounding box\r\n */\r\n public scale(factor: number): BoundingBox {\r\n const tmpVectors = BoundingBox._TmpVector3;\r\n const diff = this.maximum.subtractToRef(this.minimum, tmpVectors[0]);\r\n const len = diff.length();\r\n diff.normalizeFromLength(len);\r\n const distance = len * factor;\r\n const newRadius = diff.scaleInPlace(distance * 0.5);\r\n\r\n const min = this.center.subtractToRef(newRadius, tmpVectors[1]);\r\n const max = this.center.addToRef(newRadius, tmpVectors[2]);\r\n\r\n this.reConstruct(min, max, this._worldMatrix);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Gets the world matrix of the bounding box\r\n * @returns a matrix\r\n */\r\n public getWorldMatrix(): DeepImmutable {\r\n return this._worldMatrix;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _update(world: DeepImmutable): void {\r\n const minWorld = this.minimumWorld;\r\n const maxWorld = this.maximumWorld;\r\n const directions = this.directions;\r\n const vectorsWorld = this.vectorsWorld;\r\n const vectors = this.vectors;\r\n\r\n if (!world.isIdentity()) {\r\n minWorld.setAll(Number.MAX_VALUE);\r\n maxWorld.setAll(-Number.MAX_VALUE);\r\n\r\n for (let index = 0; index < 8; ++index) {\r\n const v = vectorsWorld[index];\r\n Vector3.TransformCoordinatesToRef(vectors[index], world, v);\r\n minWorld.minimizeInPlace(v);\r\n maxWorld.maximizeInPlace(v);\r\n }\r\n\r\n // Extend\r\n maxWorld.subtractToRef(minWorld, this.extendSizeWorld).scaleInPlace(0.5);\r\n maxWorld.addToRef(minWorld, this.centerWorld).scaleInPlace(0.5);\r\n } else {\r\n minWorld.copyFrom(this.minimum);\r\n maxWorld.copyFrom(this.maximum);\r\n for (let index = 0; index < 8; ++index) {\r\n vectorsWorld[index].copyFrom(vectors[index]);\r\n }\r\n\r\n // Extend\r\n this.extendSizeWorld.copyFrom(this.extendSize);\r\n this.centerWorld.copyFrom(this.center);\r\n }\r\n\r\n Vector3.FromArrayToRef(world.m, 0, directions[0]);\r\n Vector3.FromArrayToRef(world.m, 4, directions[1]);\r\n Vector3.FromArrayToRef(world.m, 8, directions[2]);\r\n\r\n this._worldMatrix = world;\r\n }\r\n\r\n /**\r\n * Tests if the bounding box is intersecting the frustum planes\r\n * @param frustumPlanes defines the frustum planes to test\r\n * @returns true if there is an intersection\r\n */\r\n public isInFrustum(frustumPlanes: Array>): boolean {\r\n return BoundingBox.IsInFrustum(this.vectorsWorld, frustumPlanes);\r\n }\r\n\r\n /**\r\n * Tests if the bounding box is entirely inside the frustum planes\r\n * @param frustumPlanes defines the frustum planes to test\r\n * @returns true if there is an inclusion\r\n */\r\n public isCompletelyInFrustum(frustumPlanes: Array>): boolean {\r\n return BoundingBox.IsCompletelyInFrustum(this.vectorsWorld, frustumPlanes);\r\n }\r\n\r\n /**\r\n * Tests if a point is inside the bounding box\r\n * @param point defines the point to test\r\n * @returns true if the point is inside the bounding box\r\n */\r\n public intersectsPoint(point: DeepImmutable): boolean {\r\n const min = this.minimumWorld;\r\n const max = this.maximumWorld;\r\n const minX = min.x,\r\n minY = min.y,\r\n minZ = min.z,\r\n maxX = max.x,\r\n maxY = max.y,\r\n maxZ = max.z;\r\n const pointX = point.x,\r\n pointY = point.y,\r\n pointZ = point.z;\r\n const delta = -Epsilon;\r\n\r\n if (maxX - pointX < delta || delta > pointX - minX) {\r\n return false;\r\n }\r\n\r\n if (maxY - pointY < delta || delta > pointY - minY) {\r\n return false;\r\n }\r\n\r\n if (maxZ - pointZ < delta || delta > pointZ - minZ) {\r\n return false;\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Tests if the bounding box intersects with a bounding sphere\r\n * @param sphere defines the sphere to test\r\n * @returns true if there is an intersection\r\n */\r\n public intersectsSphere(sphere: DeepImmutable): boolean {\r\n return BoundingBox.IntersectsSphere(this.minimumWorld, this.maximumWorld, sphere.centerWorld, sphere.radiusWorld);\r\n }\r\n\r\n /**\r\n * Tests if the bounding box intersects with a box defined by a min and max vectors\r\n * @param min defines the min vector to use\r\n * @param max defines the max vector to use\r\n * @returns true if there is an intersection\r\n */\r\n public intersectsMinMax(min: DeepImmutable, max: DeepImmutable): boolean {\r\n const myMin = this.minimumWorld;\r\n const myMax = this.maximumWorld;\r\n const myMinX = myMin.x,\r\n myMinY = myMin.y,\r\n myMinZ = myMin.z,\r\n myMaxX = myMax.x,\r\n myMaxY = myMax.y,\r\n myMaxZ = myMax.z;\r\n const minX = min.x,\r\n minY = min.y,\r\n minZ = min.z,\r\n maxX = max.x,\r\n maxY = max.y,\r\n maxZ = max.z;\r\n if (myMaxX < minX || myMinX > maxX) {\r\n return false;\r\n }\r\n\r\n if (myMaxY < minY || myMinY > maxY) {\r\n return false;\r\n }\r\n\r\n if (myMaxZ < minZ || myMinZ > maxZ) {\r\n return false;\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Disposes the resources of the class\r\n */\r\n public dispose(): void {\r\n this._drawWrapperFront?.dispose();\r\n this._drawWrapperBack?.dispose();\r\n }\r\n\r\n // Statics\r\n\r\n /**\r\n * Tests if two bounding boxes are intersections\r\n * @param box0 defines the first box to test\r\n * @param box1 defines the second box to test\r\n * @returns true if there is an intersection\r\n */\r\n public static Intersects(box0: DeepImmutable, box1: DeepImmutable): boolean {\r\n return box0.intersectsMinMax(box1.minimumWorld, box1.maximumWorld);\r\n }\r\n\r\n /**\r\n * Tests if a bounding box defines by a min/max vectors intersects a sphere\r\n * @param minPoint defines the minimum vector of the bounding box\r\n * @param maxPoint defines the maximum vector of the bounding box\r\n * @param sphereCenter defines the sphere center\r\n * @param sphereRadius defines the sphere radius\r\n * @returns true if there is an intersection\r\n */\r\n public static IntersectsSphere(minPoint: DeepImmutable, maxPoint: DeepImmutable, sphereCenter: DeepImmutable, sphereRadius: number): boolean {\r\n const vector = BoundingBox._TmpVector3[0];\r\n Vector3.ClampToRef(sphereCenter, minPoint, maxPoint, vector);\r\n const num = Vector3.DistanceSquared(sphereCenter, vector);\r\n return num <= sphereRadius * sphereRadius;\r\n }\r\n\r\n /**\r\n * Tests if a bounding box defined with 8 vectors is entirely inside frustum planes\r\n * @param boundingVectors defines an array of 8 vectors representing a bounding box\r\n * @param frustumPlanes defines the frustum planes to test\r\n * @returns true if there is an inclusion\r\n */\r\n public static IsCompletelyInFrustum(boundingVectors: Array>, frustumPlanes: Array>): boolean {\r\n for (let p = 0; p < 6; ++p) {\r\n const frustumPlane = frustumPlanes[p];\r\n for (let i = 0; i < 8; ++i) {\r\n if (frustumPlane.dotCoordinate(boundingVectors[i]) < 0) {\r\n return false;\r\n }\r\n }\r\n }\r\n return true;\r\n }\r\n\r\n /**\r\n * Tests if a bounding box defined with 8 vectors intersects frustum planes\r\n * @param boundingVectors defines an array of 8 vectors representing a bounding box\r\n * @param frustumPlanes defines the frustum planes to test\r\n * @returns true if there is an intersection\r\n */\r\n public static IsInFrustum(boundingVectors: Array>, frustumPlanes: Array>): boolean {\r\n for (let p = 0; p < 6; ++p) {\r\n let canReturnFalse = true;\r\n const frustumPlane = frustumPlanes[p];\r\n for (let i = 0; i < 8; ++i) {\r\n if (frustumPlane.dotCoordinate(boundingVectors[i]) >= 0) {\r\n canReturnFalse = false;\r\n break;\r\n }\r\n }\r\n if (canReturnFalse) {\r\n return false;\r\n }\r\n }\r\n return true;\r\n }\r\n}\r\n","import type { DeepImmutable } from \"../types\";\r\nimport { ArrayTools } from \"../Misc/arrayTools\";\r\nimport { Matrix, Vector3 } from \"../Maths/math.vector\";\r\nimport type { Plane } from \"../Maths/math.plane\";\r\n\r\n/**\r\n * Class used to store bounding sphere information\r\n */\r\nexport class BoundingSphere {\r\n /**\r\n * Gets the center of the bounding sphere in local space\r\n */\r\n public readonly center = Vector3.Zero();\r\n /**\r\n * Radius of the bounding sphere in local space\r\n */\r\n public radius: number;\r\n /**\r\n * Gets the center of the bounding sphere in world space\r\n */\r\n public readonly centerWorld = Vector3.Zero();\r\n /**\r\n * Radius of the bounding sphere in world space\r\n */\r\n public radiusWorld: number;\r\n /**\r\n * Gets the minimum vector in local space\r\n */\r\n public readonly minimum = Vector3.Zero();\r\n /**\r\n * Gets the maximum vector in local space\r\n */\r\n public readonly maximum = Vector3.Zero();\r\n\r\n private _worldMatrix: DeepImmutable;\r\n private static readonly _TmpVector3 = ArrayTools.BuildArray(3, Vector3.Zero);\r\n\r\n /**\r\n * Creates a new bounding sphere\r\n * @param min defines the minimum vector (in local space)\r\n * @param max defines the maximum vector (in local space)\r\n * @param worldMatrix defines the new world matrix\r\n */\r\n constructor(min: DeepImmutable, max: DeepImmutable, worldMatrix?: DeepImmutable) {\r\n this.reConstruct(min, max, worldMatrix);\r\n }\r\n\r\n /**\r\n * Recreates the entire bounding sphere from scratch as if we call the constructor in place\r\n * @param min defines the new minimum vector (in local space)\r\n * @param max defines the new maximum vector (in local space)\r\n * @param worldMatrix defines the new world matrix\r\n */\r\n public reConstruct(min: DeepImmutable, max: DeepImmutable, worldMatrix?: DeepImmutable) {\r\n this.minimum.copyFrom(min);\r\n this.maximum.copyFrom(max);\r\n\r\n const distance = Vector3.Distance(min, max);\r\n\r\n max.addToRef(min, this.center).scaleInPlace(0.5);\r\n this.radius = distance * 0.5;\r\n\r\n this._update(worldMatrix || Matrix.IdentityReadOnly);\r\n }\r\n\r\n /**\r\n * Scale the current bounding sphere by applying a scale factor\r\n * @param factor defines the scale factor to apply\r\n * @returns the current bounding box\r\n */\r\n public scale(factor: number): BoundingSphere {\r\n const newRadius = this.radius * factor;\r\n const tmpVectors = BoundingSphere._TmpVector3;\r\n const tempRadiusVector = tmpVectors[0].setAll(newRadius);\r\n const min = this.center.subtractToRef(tempRadiusVector, tmpVectors[1]);\r\n const max = this.center.addToRef(tempRadiusVector, tmpVectors[2]);\r\n\r\n this.reConstruct(min, max, this._worldMatrix);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Gets the world matrix of the bounding box\r\n * @returns a matrix\r\n */\r\n public getWorldMatrix(): DeepImmutable {\r\n return this._worldMatrix;\r\n }\r\n\r\n // Methods\r\n /**\r\n * @internal\r\n */\r\n public _update(worldMatrix: DeepImmutable): void {\r\n if (!worldMatrix.isIdentity()) {\r\n Vector3.TransformCoordinatesToRef(this.center, worldMatrix, this.centerWorld);\r\n const tempVector = BoundingSphere._TmpVector3[0];\r\n Vector3.TransformNormalFromFloatsToRef(1.0, 1.0, 1.0, worldMatrix, tempVector);\r\n this.radiusWorld = Math.max(Math.abs(tempVector.x), Math.abs(tempVector.y), Math.abs(tempVector.z)) * this.radius;\r\n } else {\r\n this.centerWorld.copyFrom(this.center);\r\n this.radiusWorld = this.radius;\r\n }\r\n }\r\n\r\n /**\r\n * Tests if the bounding sphere is intersecting the frustum planes\r\n * @param frustumPlanes defines the frustum planes to test\r\n * @returns true if there is an intersection\r\n */\r\n public isInFrustum(frustumPlanes: Array>): boolean {\r\n const center = this.centerWorld;\r\n const radius = this.radiusWorld;\r\n for (let i = 0; i < 6; i++) {\r\n if (frustumPlanes[i].dotCoordinate(center) <= -radius) {\r\n return false;\r\n }\r\n }\r\n return true;\r\n }\r\n\r\n /**\r\n * Tests if the bounding sphere center is in between the frustum planes.\r\n * Used for optimistic fast inclusion.\r\n * @param frustumPlanes defines the frustum planes to test\r\n * @returns true if the sphere center is in between the frustum planes\r\n */\r\n public isCenterInFrustum(frustumPlanes: Array>): boolean {\r\n const center = this.centerWorld;\r\n for (let i = 0; i < 6; i++) {\r\n if (frustumPlanes[i].dotCoordinate(center) < 0) {\r\n return false;\r\n }\r\n }\r\n return true;\r\n }\r\n\r\n /**\r\n * Tests if a point is inside the bounding sphere\r\n * @param point defines the point to test\r\n * @returns true if the point is inside the bounding sphere\r\n */\r\n public intersectsPoint(point: DeepImmutable): boolean {\r\n const squareDistance = Vector3.DistanceSquared(this.centerWorld, point);\r\n if (this.radiusWorld * this.radiusWorld < squareDistance) {\r\n return false;\r\n }\r\n\r\n return true;\r\n }\r\n\r\n // Statics\r\n /**\r\n * Checks if two sphere intersect\r\n * @param sphere0 sphere 0\r\n * @param sphere1 sphere 1\r\n * @returns true if the spheres intersect\r\n */\r\n public static Intersects(sphere0: DeepImmutable, sphere1: DeepImmutable): boolean {\r\n const squareDistance = Vector3.DistanceSquared(sphere0.centerWorld, sphere1.centerWorld);\r\n const radiusSum = sphere0.radiusWorld + sphere1.radiusWorld;\r\n\r\n if (radiusSum * radiusSum < squareDistance) {\r\n return false;\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Creates a sphere from a center and a radius\r\n * @param center The center\r\n * @param radius radius\r\n * @param matrix Optional worldMatrix\r\n * @returns The sphere\r\n */\r\n public static CreateFromCenterAndRadius(center: DeepImmutable, radius: number, matrix?: DeepImmutable): BoundingSphere {\r\n this._TmpVector3[0].copyFrom(center);\r\n this._TmpVector3[1].copyFromFloats(0, 0, radius);\r\n this._TmpVector3[2].copyFrom(center);\r\n this._TmpVector3[0].addInPlace(this._TmpVector3[1]);\r\n this._TmpVector3[2].subtractInPlace(this._TmpVector3[1]);\r\n\r\n const sphere = new BoundingSphere(this._TmpVector3[0], this._TmpVector3[2]);\r\n\r\n if (matrix) {\r\n sphere._worldMatrix = matrix;\r\n } else {\r\n sphere._worldMatrix = Matrix.Identity();\r\n }\r\n\r\n return sphere;\r\n }\r\n}\r\n","import type { DeepImmutable } from \"../types\";\r\nimport { ArrayTools } from \"../Misc/arrayTools\";\r\nimport type { Matrix } from \"../Maths/math.vector\";\r\nimport { TmpVectors } from \"../Maths/math.vector\";\r\nimport { Vector3 } from \"../Maths/math.vector\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport { BoundingBox } from \"./boundingBox\";\r\nimport { BoundingSphere } from \"./boundingSphere\";\r\nimport type { Plane } from \"../Maths/math.plane\";\r\n\r\nimport type { Collider } from \"../Collisions/collider\";\r\n\r\nconst _result0 = { min: 0, max: 0 };\r\nconst _result1 = { min: 0, max: 0 };\r\nconst computeBoxExtents = (axis: DeepImmutable, box: DeepImmutable, result: { min: number; max: number }) => {\r\n const p = Vector3.Dot(box.centerWorld, axis);\r\n\r\n const r0 = Math.abs(Vector3.Dot(box.directions[0], axis)) * box.extendSize.x;\r\n const r1 = Math.abs(Vector3.Dot(box.directions[1], axis)) * box.extendSize.y;\r\n const r2 = Math.abs(Vector3.Dot(box.directions[2], axis)) * box.extendSize.z;\r\n\r\n const r = r0 + r1 + r2;\r\n result.min = p - r;\r\n result.max = p + r;\r\n};\r\n\r\nconst axisOverlap = (axis: DeepImmutable, box0: DeepImmutable, box1: DeepImmutable): boolean => {\r\n computeBoxExtents(axis, box0, _result0);\r\n computeBoxExtents(axis, box1, _result1);\r\n return !(_result0.min > _result1.max || _result1.min > _result0.max);\r\n};\r\n\r\n/**\r\n * Interface for cullable objects\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/materials_introduction#back-face-culling\r\n */\r\nexport interface ICullable {\r\n /**\r\n * Checks if the object or part of the object is in the frustum\r\n * @param frustumPlanes Camera near/planes\r\n * @returns true if the object is in frustum otherwise false\r\n */\r\n isInFrustum(frustumPlanes: Plane[]): boolean;\r\n /**\r\n * Checks if a cullable object (mesh...) is in the camera frustum\r\n * Unlike isInFrustum this checks the full bounding box\r\n * @param frustumPlanes Camera near/planes\r\n * @returns true if the object is in frustum otherwise false\r\n */\r\n isCompletelyInFrustum(frustumPlanes: Plane[]): boolean;\r\n}\r\n\r\n/**\r\n * Info for a bounding data of a mesh\r\n */\r\nexport class BoundingInfo implements ICullable {\r\n /**\r\n * Bounding box for the mesh\r\n */\r\n public readonly boundingBox: BoundingBox;\r\n /**\r\n * Bounding sphere for the mesh\r\n */\r\n public readonly boundingSphere: BoundingSphere;\r\n\r\n private _isLocked = false;\r\n\r\n private static readonly _TmpVector3 = ArrayTools.BuildArray(2, Vector3.Zero);\r\n\r\n /**\r\n * Constructs bounding info\r\n * @param minimum min vector of the bounding box/sphere\r\n * @param maximum max vector of the bounding box/sphere\r\n * @param worldMatrix defines the new world matrix\r\n */\r\n constructor(minimum: DeepImmutable, maximum: DeepImmutable, worldMatrix?: DeepImmutable) {\r\n this.boundingBox = new BoundingBox(minimum, maximum, worldMatrix);\r\n this.boundingSphere = new BoundingSphere(minimum, maximum, worldMatrix);\r\n }\r\n\r\n /**\r\n * Recreates the entire bounding info from scratch as if we call the constructor in place\r\n * @param min defines the new minimum vector (in local space)\r\n * @param max defines the new maximum vector (in local space)\r\n * @param worldMatrix defines the new world matrix\r\n */\r\n public reConstruct(min: DeepImmutable, max: DeepImmutable, worldMatrix?: DeepImmutable) {\r\n this.boundingBox.reConstruct(min, max, worldMatrix);\r\n this.boundingSphere.reConstruct(min, max, worldMatrix);\r\n }\r\n\r\n /**\r\n * min vector of the bounding box/sphere\r\n */\r\n public get minimum(): Vector3 {\r\n return this.boundingBox.minimum;\r\n }\r\n\r\n /**\r\n * max vector of the bounding box/sphere\r\n */\r\n public get maximum(): Vector3 {\r\n return this.boundingBox.maximum;\r\n }\r\n\r\n /**\r\n * If the info is locked and won't be updated to avoid perf overhead\r\n */\r\n public get isLocked(): boolean {\r\n return this._isLocked;\r\n }\r\n\r\n public set isLocked(value: boolean) {\r\n this._isLocked = value;\r\n }\r\n\r\n // Methods\r\n /**\r\n * Updates the bounding sphere and box\r\n * @param world world matrix to be used to update\r\n */\r\n public update(world: DeepImmutable) {\r\n if (this._isLocked) {\r\n return;\r\n }\r\n this.boundingBox._update(world);\r\n this.boundingSphere._update(world);\r\n }\r\n\r\n /**\r\n * Recreate the bounding info to be centered around a specific point given a specific extend.\r\n * @param center New center of the bounding info\r\n * @param extend New extend of the bounding info\r\n * @returns the current bounding info\r\n */\r\n public centerOn(center: DeepImmutable, extend: DeepImmutable): BoundingInfo {\r\n const minimum = BoundingInfo._TmpVector3[0].copyFrom(center).subtractInPlace(extend);\r\n const maximum = BoundingInfo._TmpVector3[1].copyFrom(center).addInPlace(extend);\r\n\r\n this.boundingBox.reConstruct(minimum, maximum, this.boundingBox.getWorldMatrix());\r\n this.boundingSphere.reConstruct(minimum, maximum, this.boundingBox.getWorldMatrix());\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Grows the bounding info to include the given point.\r\n * @param point The point that will be included in the current bounding info (in local space)\r\n * @returns the current bounding info\r\n */\r\n public encapsulate(point: Vector3): BoundingInfo {\r\n const minimum = Vector3.Minimize(this.minimum, point);\r\n const maximum = Vector3.Maximize(this.maximum, point);\r\n this.reConstruct(minimum, maximum, this.boundingBox.getWorldMatrix());\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Grows the bounding info to encapsulate the given bounding info.\r\n * @param toEncapsulate The bounding info that will be encapsulated in the current bounding info\r\n * @returns the current bounding info\r\n */\r\n public encapsulateBoundingInfo(toEncapsulate: BoundingInfo): BoundingInfo {\r\n const invw = TmpVectors.Matrix[0];\r\n this.boundingBox.getWorldMatrix().invertToRef(invw);\r\n\r\n const v = TmpVectors.Vector3[0];\r\n\r\n Vector3.TransformCoordinatesToRef(toEncapsulate.boundingBox.minimumWorld, invw, v);\r\n this.encapsulate(v);\r\n\r\n Vector3.TransformCoordinatesToRef(toEncapsulate.boundingBox.maximumWorld, invw, v);\r\n this.encapsulate(v);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Scale the current bounding info by applying a scale factor\r\n * @param factor defines the scale factor to apply\r\n * @returns the current bounding info\r\n */\r\n public scale(factor: number): BoundingInfo {\r\n this.boundingBox.scale(factor);\r\n this.boundingSphere.scale(factor);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns `true` if the bounding info is within the frustum defined by the passed array of planes.\r\n * @param frustumPlanes defines the frustum to test\r\n * @param strategy defines the strategy to use for the culling (default is BABYLON.AbstractMesh.CULLINGSTRATEGY_STANDARD)\r\n * The different strategies available are:\r\n * * BABYLON.AbstractMesh.CULLINGSTRATEGY_STANDARD most accurate but slower @see https://doc.babylonjs.com/typedoc/classes/BABYLON.AbstractMesh#CULLINGSTRATEGY_STANDARD\r\n * * BABYLON.AbstractMesh.CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY faster but less accurate @see https://doc.babylonjs.com/typedoc/classes/BABYLON.AbstractMesh#CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY\r\n * * BABYLON.AbstractMesh.CULLINGSTRATEGY_OPTIMISTIC_INCLUSION can be faster if always visible @see https://doc.babylonjs.com/typedoc/classes/BABYLON.AbstractMesh#CULLINGSTRATEGY_OPTIMISTIC_INCLUSION\r\n * * BABYLON.AbstractMesh.CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY can be faster if always visible @see https://doc.babylonjs.com/typedoc/classes/BABYLON.AbstractMesh#CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY\r\n * @returns true if the bounding info is in the frustum planes\r\n */\r\n public isInFrustum(frustumPlanes: Array>, strategy: number = Constants.MESHES_CULLINGSTRATEGY_STANDARD): boolean {\r\n const inclusionTest =\r\n strategy === Constants.MESHES_CULLINGSTRATEGY_OPTIMISTIC_INCLUSION || strategy === Constants.MESHES_CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY;\r\n if (inclusionTest) {\r\n if (this.boundingSphere.isCenterInFrustum(frustumPlanes)) {\r\n return true;\r\n }\r\n }\r\n\r\n if (!this.boundingSphere.isInFrustum(frustumPlanes)) {\r\n return false;\r\n }\r\n\r\n const bSphereOnlyTest =\r\n strategy === Constants.MESHES_CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY || strategy === Constants.MESHES_CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY;\r\n if (bSphereOnlyTest) {\r\n return true;\r\n }\r\n\r\n return this.boundingBox.isInFrustum(frustumPlanes);\r\n }\r\n\r\n /**\r\n * Gets the world distance between the min and max points of the bounding box\r\n */\r\n public get diagonalLength(): number {\r\n const boundingBox = this.boundingBox;\r\n const diag = boundingBox.maximumWorld.subtractToRef(boundingBox.minimumWorld, BoundingInfo._TmpVector3[0]);\r\n return diag.length();\r\n }\r\n\r\n /**\r\n * Checks if a cullable object (mesh...) is in the camera frustum\r\n * Unlike isInFrustum this checks the full bounding box\r\n * @param frustumPlanes Camera near/planes\r\n * @returns true if the object is in frustum otherwise false\r\n */\r\n public isCompletelyInFrustum(frustumPlanes: Array>): boolean {\r\n return this.boundingBox.isCompletelyInFrustum(frustumPlanes);\r\n }\r\n /**\r\n * @internal\r\n */\r\n public _checkCollision(collider: Collider): boolean {\r\n return collider._canDoCollision(this.boundingSphere.centerWorld, this.boundingSphere.radiusWorld, this.boundingBox.minimumWorld, this.boundingBox.maximumWorld);\r\n }\r\n\r\n /**\r\n * Checks if a point is inside the bounding box and bounding sphere or the mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/interactions/mesh_intersect\r\n * @param point the point to check intersection with\r\n * @returns if the point intersects\r\n */\r\n public intersectsPoint(point: DeepImmutable): boolean {\r\n if (!this.boundingSphere.centerWorld) {\r\n return false;\r\n }\r\n\r\n if (!this.boundingSphere.intersectsPoint(point)) {\r\n return false;\r\n }\r\n\r\n if (!this.boundingBox.intersectsPoint(point)) {\r\n return false;\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Checks if another bounding info intersects the bounding box and bounding sphere or the mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/interactions/mesh_intersect\r\n * @param boundingInfo the bounding info to check intersection with\r\n * @param precise if the intersection should be done using OBB\r\n * @returns if the bounding info intersects\r\n */\r\n public intersects(boundingInfo: DeepImmutable, precise: boolean): boolean {\r\n if (!BoundingSphere.Intersects(this.boundingSphere, boundingInfo.boundingSphere)) {\r\n return false;\r\n }\r\n\r\n if (!BoundingBox.Intersects(this.boundingBox, boundingInfo.boundingBox)) {\r\n return false;\r\n }\r\n\r\n if (!precise) {\r\n return true;\r\n }\r\n\r\n const box0 = this.boundingBox;\r\n const box1 = boundingInfo.boundingBox;\r\n\r\n if (!axisOverlap(box0.directions[0], box0, box1)) {\r\n return false;\r\n }\r\n if (!axisOverlap(box0.directions[1], box0, box1)) {\r\n return false;\r\n }\r\n if (!axisOverlap(box0.directions[2], box0, box1)) {\r\n return false;\r\n }\r\n if (!axisOverlap(box1.directions[0], box0, box1)) {\r\n return false;\r\n }\r\n if (!axisOverlap(box1.directions[1], box0, box1)) {\r\n return false;\r\n }\r\n if (!axisOverlap(box1.directions[2], box0, box1)) {\r\n return false;\r\n }\r\n if (!axisOverlap(Vector3.Cross(box0.directions[0], box1.directions[0]), box0, box1)) {\r\n return false;\r\n }\r\n if (!axisOverlap(Vector3.Cross(box0.directions[0], box1.directions[1]), box0, box1)) {\r\n return false;\r\n }\r\n if (!axisOverlap(Vector3.Cross(box0.directions[0], box1.directions[2]), box0, box1)) {\r\n return false;\r\n }\r\n if (!axisOverlap(Vector3.Cross(box0.directions[1], box1.directions[0]), box0, box1)) {\r\n return false;\r\n }\r\n if (!axisOverlap(Vector3.Cross(box0.directions[1], box1.directions[1]), box0, box1)) {\r\n return false;\r\n }\r\n if (!axisOverlap(Vector3.Cross(box0.directions[1], box1.directions[2]), box0, box1)) {\r\n return false;\r\n }\r\n if (!axisOverlap(Vector3.Cross(box0.directions[2], box1.directions[0]), box0, box1)) {\r\n return false;\r\n }\r\n if (!axisOverlap(Vector3.Cross(box0.directions[2], box1.directions[1]), box0, box1)) {\r\n return false;\r\n }\r\n if (!axisOverlap(Vector3.Cross(box0.directions[2], box1.directions[2]), box0, box1)) {\r\n return false;\r\n }\r\n\r\n return true;\r\n }\r\n}\r\n","import type { FloatArray, Nullable, IndicesArray } from \"../types\";\r\nimport type { Vector2 } from \"./math.vector\";\r\nimport { Vector3 } from \"./math.vector\";\r\nimport { nativeOverride } from \"../Misc/decorators\";\r\n\r\n// This helper class is only here so we can apply the nativeOverride decorator to functions.\r\nclass MathHelpers {\r\n @nativeOverride.filter((...[positions, indices]: Parameters) => !Array.isArray(positions) && !Array.isArray(indices))\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public static extractMinAndMaxIndexed(positions: FloatArray, indices: IndicesArray, indexStart: number, indexCount: number, minimum: Vector3, maximum: Vector3): void {\r\n for (let index = indexStart; index < indexStart + indexCount; index++) {\r\n const offset = indices[index] * 3;\r\n const x = positions[offset];\r\n const y = positions[offset + 1];\r\n const z = positions[offset + 2];\r\n minimum.minimizeInPlaceFromFloats(x, y, z);\r\n maximum.maximizeInPlaceFromFloats(x, y, z);\r\n }\r\n }\r\n\r\n @nativeOverride.filter((...[positions]: Parameters) => !Array.isArray(positions))\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public static extractMinAndMax(positions: FloatArray, start: number, count: number, stride: number, minimum: Vector3, maximum: Vector3): void {\r\n for (let index = start, offset = start * stride; index < start + count; index++, offset += stride) {\r\n const x = positions[offset];\r\n const y = positions[offset + 1];\r\n const z = positions[offset + 2];\r\n minimum.minimizeInPlaceFromFloats(x, y, z);\r\n maximum.maximizeInPlaceFromFloats(x, y, z);\r\n }\r\n }\r\n}\r\n\r\n/**\r\n * Extracts minimum and maximum values from a list of indexed positions\r\n * @param positions defines the positions to use\r\n * @param indices defines the indices to the positions\r\n * @param indexStart defines the start index\r\n * @param indexCount defines the end index\r\n * @param bias defines bias value to add to the result\r\n * @returns minimum and maximum values\r\n */\r\nexport function extractMinAndMaxIndexed(\r\n positions: FloatArray,\r\n indices: IndicesArray,\r\n indexStart: number,\r\n indexCount: number,\r\n bias: Nullable = null\r\n): { minimum: Vector3; maximum: Vector3 } {\r\n const minimum = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\r\n const maximum = new Vector3(-Number.MAX_VALUE, -Number.MAX_VALUE, -Number.MAX_VALUE);\r\n\r\n MathHelpers.extractMinAndMaxIndexed(positions, indices, indexStart, indexCount, minimum, maximum);\r\n\r\n if (bias) {\r\n minimum.x -= minimum.x * bias.x + bias.y;\r\n minimum.y -= minimum.y * bias.x + bias.y;\r\n minimum.z -= minimum.z * bias.x + bias.y;\r\n maximum.x += maximum.x * bias.x + bias.y;\r\n maximum.y += maximum.y * bias.x + bias.y;\r\n maximum.z += maximum.z * bias.x + bias.y;\r\n }\r\n\r\n return {\r\n minimum: minimum,\r\n maximum: maximum,\r\n };\r\n}\r\n\r\n/**\r\n * Extracts minimum and maximum values from a list of positions\r\n * @param positions defines the positions to use\r\n * @param start defines the start index in the positions array\r\n * @param count defines the number of positions to handle\r\n * @param bias defines bias value to add to the result\r\n * @param stride defines the stride size to use (distance between two positions in the positions array)\r\n * @returns minimum and maximum values\r\n */\r\nexport function extractMinAndMax(positions: FloatArray, start: number, count: number, bias: Nullable = null, stride?: number): { minimum: Vector3; maximum: Vector3 } {\r\n const minimum = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\r\n const maximum = new Vector3(-Number.MAX_VALUE, -Number.MAX_VALUE, -Number.MAX_VALUE);\r\n\r\n if (!stride) {\r\n stride = 3;\r\n }\r\n\r\n MathHelpers.extractMinAndMax(positions, start, count, stride, minimum, maximum);\r\n\r\n if (bias) {\r\n minimum.x -= minimum.x * bias.x + bias.y;\r\n minimum.y -= minimum.y * bias.x + bias.y;\r\n minimum.z -= minimum.z * bias.x + bias.y;\r\n maximum.x += maximum.x * bias.x + bias.y;\r\n maximum.y += maximum.y * bias.x + bias.y;\r\n maximum.z += maximum.z * bias.x + bias.y;\r\n }\r\n\r\n return {\r\n minimum: minimum,\r\n maximum: maximum,\r\n };\r\n}\r\n","import type { Nullable, IndicesArray, DeepImmutable, FloatArray } from \"../types\";\r\nimport type { Matrix, Vector3 } from \"../Maths/math.vector\";\r\nimport type { Engine } from \"../Engines/engine\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport { IntersectionInfo } from \"../Collisions/intersectionInfo\";\r\nimport type { ICullable } from \"../Culling/boundingInfo\";\r\nimport { BoundingInfo } from \"../Culling/boundingInfo\";\r\nimport type { Effect } from \"../Materials/effect\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport type { DataBuffer } from \"../Buffers/dataBuffer\";\r\nimport { extractMinAndMaxIndexed } from \"../Maths/math.functions\";\r\nimport type { Plane } from \"../Maths/math.plane\";\r\nimport { DrawWrapper } from \"../Materials/drawWrapper\";\r\nimport type { IMaterialContext } from \"../Engines/IMaterialContext\";\r\n\r\nimport type { Collider } from \"../Collisions/collider\";\r\nimport type { Material } from \"../Materials/material\";\r\nimport type { MaterialDefines } from \"../Materials/materialDefines\";\r\nimport type { MultiMaterial } from \"../Materials/multiMaterial\";\r\nimport type { AbstractMesh } from \"./abstractMesh\";\r\nimport type { Mesh } from \"./mesh\";\r\nimport type { Ray } from \"../Culling/ray\";\r\nimport type { TrianglePickingPredicate } from \"../Culling/ray\";\r\n\r\n/**\r\n * Defines a subdivision inside a mesh\r\n */\r\nexport class SubMesh implements ICullable {\r\n private _engine: Engine;\r\n /** @internal */\r\n public _drawWrappers: Array; // index in this array = pass id\r\n private _mainDrawWrapperOverride: Nullable = null;\r\n\r\n /**\r\n * Gets material defines used by the effect associated to the sub mesh\r\n */\r\n public get materialDefines(): Nullable {\r\n return this._mainDrawWrapperOverride ? (this._mainDrawWrapperOverride.defines as MaterialDefines) : (this._getDrawWrapper()?.defines as Nullable);\r\n }\r\n\r\n /**\r\n * Sets material defines used by the effect associated to the sub mesh\r\n */\r\n public set materialDefines(defines: Nullable) {\r\n const drawWrapper = this._mainDrawWrapperOverride ?? this._getDrawWrapper(undefined, true)!;\r\n drawWrapper.defines = defines;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getDrawWrapper(passId?: number, createIfNotExisting = false): DrawWrapper | undefined {\r\n passId = passId ?? this._engine.currentRenderPassId;\r\n let drawWrapper = this._drawWrappers[passId];\r\n if (!drawWrapper && createIfNotExisting) {\r\n this._drawWrappers[passId] = drawWrapper = new DrawWrapper(this._mesh.getScene().getEngine());\r\n }\r\n return drawWrapper;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _removeDrawWrapper(passId: number, disposeWrapper = true) {\r\n if (disposeWrapper) {\r\n this._drawWrappers[passId]?.dispose();\r\n }\r\n this._drawWrappers[passId] = undefined as any;\r\n }\r\n\r\n /**\r\n * Gets associated (main) effect (possibly the effect override if defined)\r\n */\r\n public get effect(): Nullable {\r\n return this._mainDrawWrapperOverride ? this._mainDrawWrapperOverride.effect : this._getDrawWrapper()?.effect ?? null;\r\n }\r\n\r\n /** @internal */\r\n public get _drawWrapper(): DrawWrapper {\r\n return this._mainDrawWrapperOverride ?? this._getDrawWrapper(undefined, true)!;\r\n }\r\n\r\n /** @internal */\r\n public get _drawWrapperOverride(): Nullable {\r\n return this._mainDrawWrapperOverride;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _setMainDrawWrapperOverride(wrapper: Nullable): void {\r\n this._mainDrawWrapperOverride = wrapper;\r\n }\r\n\r\n /**\r\n * Sets associated effect (effect used to render this submesh)\r\n * @param effect defines the effect to associate with\r\n * @param defines defines the set of defines used to compile this effect\r\n * @param materialContext material context associated to the effect\r\n * @param resetContext true to reset the draw context\r\n */\r\n public setEffect(effect: Nullable, defines: Nullable = null, materialContext?: IMaterialContext, resetContext = true) {\r\n const drawWrapper = this._drawWrapper;\r\n drawWrapper.setEffect(effect, defines, resetContext);\r\n if (materialContext !== undefined) {\r\n drawWrapper.materialContext = materialContext;\r\n }\r\n if (!effect) {\r\n drawWrapper.defines = null;\r\n drawWrapper.materialContext = undefined;\r\n }\r\n }\r\n\r\n /**\r\n * Resets the draw wrappers cache\r\n * @param passId If provided, releases only the draw wrapper corresponding to this render pass id\r\n */\r\n public resetDrawCache(passId?: number): void {\r\n if (this._drawWrappers) {\r\n if (passId !== undefined) {\r\n this._removeDrawWrapper(passId);\r\n return;\r\n } else {\r\n for (const drawWrapper of this._drawWrappers) {\r\n drawWrapper?.dispose();\r\n }\r\n }\r\n }\r\n this._drawWrappers = [];\r\n }\r\n\r\n /** @internal */\r\n public _linesIndexCount: number = 0;\r\n private _mesh: AbstractMesh;\r\n private _renderingMesh: Mesh;\r\n private _boundingInfo: BoundingInfo;\r\n private _linesIndexBuffer: Nullable = null;\r\n /** @internal */\r\n public _lastColliderWorldVertices: Nullable = null;\r\n /** @internal */\r\n public _trianglePlanes: Plane[];\r\n /** @internal */\r\n public _lastColliderTransformMatrix: Nullable = null;\r\n /** @internal */\r\n public _wasDispatched = false;\r\n\r\n /** @internal */\r\n public _renderId = 0;\r\n /** @internal */\r\n public _alphaIndex: number = 0;\r\n /** @internal */\r\n public _distanceToCamera: number = 0;\r\n /** @internal */\r\n public _id: number;\r\n\r\n private _currentMaterial: Nullable = null;\r\n\r\n /**\r\n * Add a new submesh to a mesh\r\n * @param materialIndex defines the material index to use\r\n * @param verticesStart defines vertex index start\r\n * @param verticesCount defines vertices count\r\n * @param indexStart defines index start\r\n * @param indexCount defines indices count\r\n * @param mesh defines the parent mesh\r\n * @param renderingMesh defines an optional rendering mesh\r\n * @param createBoundingBox defines if bounding box should be created for this submesh\r\n * @returns the new submesh\r\n */\r\n public static AddToMesh(\r\n materialIndex: number,\r\n verticesStart: number,\r\n verticesCount: number,\r\n indexStart: number,\r\n indexCount: number,\r\n mesh: AbstractMesh,\r\n renderingMesh?: Mesh,\r\n createBoundingBox: boolean = true\r\n ): SubMesh {\r\n return new SubMesh(materialIndex, verticesStart, verticesCount, indexStart, indexCount, mesh, renderingMesh, createBoundingBox);\r\n }\r\n\r\n /**\r\n * Creates a new submesh\r\n * @param materialIndex defines the material index to use\r\n * @param verticesStart defines vertex index start\r\n * @param verticesCount defines vertices count\r\n * @param indexStart defines index start\r\n * @param indexCount defines indices count\r\n * @param mesh defines the parent mesh\r\n * @param renderingMesh defines an optional rendering mesh\r\n * @param createBoundingBox defines if bounding box should be created for this submesh\r\n * @param addToMesh defines a boolean indicating that the submesh must be added to the mesh.subMeshes array (true by default)\r\n */\r\n constructor(\r\n /** the material index to use */\r\n public materialIndex: number,\r\n /** vertex index start */\r\n public verticesStart: number,\r\n /** vertices count */\r\n public verticesCount: number,\r\n /** index start */\r\n public indexStart: number,\r\n /** indices count */\r\n public indexCount: number,\r\n mesh: AbstractMesh,\r\n renderingMesh?: Mesh,\r\n createBoundingBox: boolean = true,\r\n addToMesh = true\r\n ) {\r\n this._mesh = mesh;\r\n this._renderingMesh = renderingMesh || mesh;\r\n if (addToMesh) {\r\n mesh.subMeshes.push(this);\r\n }\r\n\r\n this._engine = this._mesh.getScene().getEngine();\r\n this.resetDrawCache();\r\n this._trianglePlanes = [];\r\n\r\n this._id = mesh.subMeshes.length - 1;\r\n\r\n if (createBoundingBox) {\r\n this.refreshBoundingInfo();\r\n mesh.computeWorldMatrix(true);\r\n }\r\n }\r\n\r\n /**\r\n * Returns true if this submesh covers the entire parent mesh\r\n * @ignorenaming\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public get IsGlobal(): boolean {\r\n return this.verticesStart === 0 && this.verticesCount === this._mesh.getTotalVertices() && this.indexStart === 0 && this.indexCount === this._mesh.getTotalIndices();\r\n }\r\n\r\n /**\r\n * Returns the submesh BoundingInfo object\r\n * @returns current bounding info (or mesh's one if the submesh is global)\r\n */\r\n public getBoundingInfo(): BoundingInfo {\r\n if (this.IsGlobal) {\r\n return this._mesh.getBoundingInfo();\r\n }\r\n\r\n return this._boundingInfo;\r\n }\r\n\r\n /**\r\n * Sets the submesh BoundingInfo\r\n * @param boundingInfo defines the new bounding info to use\r\n * @returns the SubMesh\r\n */\r\n public setBoundingInfo(boundingInfo: BoundingInfo): SubMesh {\r\n this._boundingInfo = boundingInfo;\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns the mesh of the current submesh\r\n * @returns the parent mesh\r\n */\r\n public getMesh(): AbstractMesh {\r\n return this._mesh;\r\n }\r\n\r\n /**\r\n * Returns the rendering mesh of the submesh\r\n * @returns the rendering mesh (could be different from parent mesh)\r\n */\r\n public getRenderingMesh(): Mesh {\r\n return this._renderingMesh;\r\n }\r\n\r\n /**\r\n * Returns the replacement mesh of the submesh\r\n * @returns the replacement mesh (could be different from parent mesh)\r\n */\r\n public getReplacementMesh(): Nullable {\r\n return this._mesh._internalAbstractMeshDataInfo._actAsRegularMesh ? this._mesh : null;\r\n }\r\n\r\n /**\r\n * Returns the effective mesh of the submesh\r\n * @returns the effective mesh (could be different from parent mesh)\r\n */\r\n public getEffectiveMesh(): AbstractMesh {\r\n const replacementMesh = this._mesh._internalAbstractMeshDataInfo._actAsRegularMesh ? this._mesh : null;\r\n\r\n return replacementMesh ? replacementMesh : this._renderingMesh;\r\n }\r\n\r\n /**\r\n * Returns the submesh material\r\n * @param getDefaultMaterial Defines whether or not to get the default material if nothing has been defined.\r\n * @returns null or the current material\r\n */\r\n public getMaterial(getDefaultMaterial = true): Nullable {\r\n const rootMaterial = this._renderingMesh.getMaterialForRenderPass(this._engine.currentRenderPassId) ?? this._renderingMesh.material;\r\n\r\n if (!rootMaterial) {\r\n return getDefaultMaterial ? this._mesh.getScene().defaultMaterial : null;\r\n } else if (this._isMultiMaterial(rootMaterial)) {\r\n const effectiveMaterial = rootMaterial.getSubMaterial(this.materialIndex);\r\n\r\n if (this._currentMaterial !== effectiveMaterial) {\r\n this._currentMaterial = effectiveMaterial;\r\n this.resetDrawCache();\r\n }\r\n\r\n return effectiveMaterial;\r\n }\r\n\r\n return rootMaterial;\r\n }\r\n\r\n private _isMultiMaterial(material: Material): material is MultiMaterial {\r\n return (material as MultiMaterial).getSubMaterial !== undefined;\r\n }\r\n\r\n // Methods\r\n\r\n /**\r\n * Sets a new updated BoundingInfo object to the submesh\r\n * @param data defines an optional position array to use to determine the bounding info\r\n * @returns the SubMesh\r\n */\r\n public refreshBoundingInfo(data: Nullable = null): SubMesh {\r\n this._lastColliderWorldVertices = null;\r\n\r\n if (this.IsGlobal || !this._renderingMesh || !this._renderingMesh.geometry) {\r\n return this;\r\n }\r\n\r\n if (!data) {\r\n data = this._renderingMesh.getVerticesData(VertexBuffer.PositionKind);\r\n }\r\n\r\n if (!data) {\r\n this._boundingInfo = this._mesh.getBoundingInfo();\r\n return this;\r\n }\r\n\r\n const indices = this._renderingMesh.getIndices();\r\n let extend: { minimum: Vector3; maximum: Vector3 };\r\n\r\n //is this the only submesh?\r\n if (this.indexStart === 0 && this.indexCount === indices.length) {\r\n const boundingInfo = this._renderingMesh.getBoundingInfo();\r\n\r\n //the rendering mesh's bounding info can be used, it is the standard submesh for all indices.\r\n extend = { minimum: boundingInfo.minimum.clone(), maximum: boundingInfo.maximum.clone() };\r\n } else {\r\n extend = extractMinAndMaxIndexed(data, indices, this.indexStart, this.indexCount, this._renderingMesh.geometry.boundingBias);\r\n }\r\n\r\n if (this._boundingInfo) {\r\n this._boundingInfo.reConstruct(extend.minimum, extend.maximum);\r\n } else {\r\n this._boundingInfo = new BoundingInfo(extend.minimum, extend.maximum);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _checkCollision(collider: Collider): boolean {\r\n const boundingInfo = this.getBoundingInfo();\r\n\r\n return boundingInfo._checkCollision(collider);\r\n }\r\n\r\n /**\r\n * Updates the submesh BoundingInfo\r\n * @param world defines the world matrix to use to update the bounding info\r\n * @returns the submesh\r\n */\r\n public updateBoundingInfo(world: DeepImmutable): SubMesh {\r\n let boundingInfo = this.getBoundingInfo();\r\n\r\n if (!boundingInfo) {\r\n this.refreshBoundingInfo();\r\n boundingInfo = this.getBoundingInfo();\r\n }\r\n if (boundingInfo) {\r\n (boundingInfo).update(world);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * True is the submesh bounding box intersects the frustum defined by the passed array of planes.\r\n * @param frustumPlanes defines the frustum planes\r\n * @returns true if the submesh is intersecting with the frustum\r\n */\r\n public isInFrustum(frustumPlanes: Plane[]): boolean {\r\n const boundingInfo = this.getBoundingInfo();\r\n\r\n if (!boundingInfo) {\r\n return false;\r\n }\r\n return boundingInfo.isInFrustum(frustumPlanes, this._mesh.cullingStrategy);\r\n }\r\n\r\n /**\r\n * True is the submesh bounding box is completely inside the frustum defined by the passed array of planes\r\n * @param frustumPlanes defines the frustum planes\r\n * @returns true if the submesh is inside the frustum\r\n */\r\n public isCompletelyInFrustum(frustumPlanes: Plane[]): boolean {\r\n const boundingInfo = this.getBoundingInfo();\r\n\r\n if (!boundingInfo) {\r\n return false;\r\n }\r\n return boundingInfo.isCompletelyInFrustum(frustumPlanes);\r\n }\r\n\r\n /**\r\n * Renders the submesh\r\n * @param enableAlphaMode defines if alpha needs to be used\r\n * @returns the submesh\r\n */\r\n public render(enableAlphaMode: boolean): SubMesh {\r\n this._renderingMesh.render(this, enableAlphaMode, this._mesh._internalAbstractMeshDataInfo._actAsRegularMesh ? this._mesh : undefined);\r\n return this;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getLinesIndexBuffer(indices: IndicesArray, engine: Engine): DataBuffer {\r\n if (!this._linesIndexBuffer) {\r\n const linesIndices = [];\r\n\r\n for (let index = this.indexStart; index < this.indexStart + this.indexCount; index += 3) {\r\n linesIndices.push(indices[index], indices[index + 1], indices[index + 1], indices[index + 2], indices[index + 2], indices[index]);\r\n }\r\n\r\n this._linesIndexBuffer = engine.createIndexBuffer(linesIndices);\r\n this._linesIndexCount = linesIndices.length;\r\n }\r\n return this._linesIndexBuffer;\r\n }\r\n\r\n /**\r\n * Checks if the submesh intersects with a ray\r\n * @param ray defines the ray to test\r\n * @returns true is the passed ray intersects the submesh bounding box\r\n */\r\n public canIntersects(ray: Ray): boolean {\r\n const boundingInfo = this.getBoundingInfo();\r\n\r\n if (!boundingInfo) {\r\n return false;\r\n }\r\n return ray.intersectsBox(boundingInfo.boundingBox);\r\n }\r\n\r\n /**\r\n * Intersects current submesh with a ray\r\n * @param ray defines the ray to test\r\n * @param positions defines mesh's positions array\r\n * @param indices defines mesh's indices array\r\n * @param fastCheck defines if the first intersection will be used (and not the closest)\r\n * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected\r\n * @returns intersection info or null if no intersection\r\n */\r\n public intersects(ray: Ray, positions: Vector3[], indices: IndicesArray, fastCheck?: boolean, trianglePredicate?: TrianglePickingPredicate): Nullable {\r\n const material = this.getMaterial();\r\n if (!material) {\r\n return null;\r\n }\r\n let step = 3;\r\n let checkStopper = false;\r\n\r\n switch (material.fillMode) {\r\n case Constants.MATERIAL_PointListDrawMode:\r\n case Constants.MATERIAL_LineLoopDrawMode:\r\n case Constants.MATERIAL_LineStripDrawMode:\r\n case Constants.MATERIAL_TriangleFanDrawMode:\r\n return null;\r\n case Constants.MATERIAL_TriangleStripDrawMode:\r\n step = 1;\r\n checkStopper = true;\r\n break;\r\n default:\r\n break;\r\n }\r\n\r\n // LineMesh first as it's also a Mesh...\r\n if (material.fillMode === Constants.MATERIAL_LineListDrawMode) {\r\n // Check if mesh is unindexed\r\n if (!indices.length) {\r\n return this._intersectUnIndexedLines(ray, positions, indices, (this._mesh as any).intersectionThreshold, fastCheck);\r\n }\r\n return this._intersectLines(ray, positions, indices, (this._mesh as any).intersectionThreshold, fastCheck);\r\n } else {\r\n // Check if mesh is unindexed\r\n if (!indices.length && this._mesh._unIndexed) {\r\n return this._intersectUnIndexedTriangles(ray, positions, indices, fastCheck, trianglePredicate);\r\n }\r\n\r\n return this._intersectTriangles(ray, positions, indices, step, checkStopper, fastCheck, trianglePredicate);\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n private _intersectLines(ray: Ray, positions: Vector3[], indices: IndicesArray, intersectionThreshold: number, fastCheck?: boolean): Nullable {\r\n let intersectInfo: Nullable = null;\r\n\r\n // Line test\r\n for (let index = this.indexStart; index < this.indexStart + this.indexCount; index += 2) {\r\n const p0 = positions[indices[index]];\r\n const p1 = positions[indices[index + 1]];\r\n\r\n const length = ray.intersectionSegment(p0, p1, intersectionThreshold);\r\n if (length < 0) {\r\n continue;\r\n }\r\n\r\n if (fastCheck || !intersectInfo || length < intersectInfo.distance) {\r\n intersectInfo = new IntersectionInfo(null, null, length);\r\n intersectInfo.faceId = index / 2;\r\n if (fastCheck) {\r\n break;\r\n }\r\n }\r\n }\r\n return intersectInfo;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n private _intersectUnIndexedLines(ray: Ray, positions: Vector3[], indices: IndicesArray, intersectionThreshold: number, fastCheck?: boolean): Nullable {\r\n let intersectInfo: Nullable = null;\r\n\r\n // Line test\r\n for (let index = this.verticesStart; index < this.verticesStart + this.verticesCount; index += 2) {\r\n const p0 = positions[index];\r\n const p1 = positions[index + 1];\r\n\r\n const length = ray.intersectionSegment(p0, p1, intersectionThreshold);\r\n if (length < 0) {\r\n continue;\r\n }\r\n\r\n if (fastCheck || !intersectInfo || length < intersectInfo.distance) {\r\n intersectInfo = new IntersectionInfo(null, null, length);\r\n intersectInfo.faceId = index / 2;\r\n if (fastCheck) {\r\n break;\r\n }\r\n }\r\n }\r\n\r\n return intersectInfo;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n private _intersectTriangles(\r\n ray: Ray,\r\n positions: Vector3[],\r\n indices: IndicesArray,\r\n step: number,\r\n checkStopper: boolean,\r\n fastCheck?: boolean,\r\n trianglePredicate?: TrianglePickingPredicate\r\n ): Nullable {\r\n let intersectInfo: Nullable = null;\r\n\r\n // Triangles test\r\n let faceId = -1;\r\n for (let index = this.indexStart; index < this.indexStart + this.indexCount - (3 - step); index += step) {\r\n faceId++;\r\n const indexA = indices[index];\r\n const indexB = indices[index + 1];\r\n const indexC = indices[index + 2];\r\n\r\n if (checkStopper && indexC === 0xffffffff) {\r\n index += 2;\r\n continue;\r\n }\r\n\r\n const p0 = positions[indexA];\r\n const p1 = positions[indexB];\r\n const p2 = positions[indexC];\r\n\r\n // stay defensive and don't check against undefined positions.\r\n if (!p0 || !p1 || !p2) {\r\n continue;\r\n }\r\n\r\n if (trianglePredicate && !trianglePredicate(p0, p1, p2, ray, indexA, indexB, indexC)) {\r\n continue;\r\n }\r\n\r\n const currentIntersectInfo = ray.intersectsTriangle(p0, p1, p2);\r\n\r\n if (currentIntersectInfo) {\r\n if (currentIntersectInfo.distance < 0) {\r\n continue;\r\n }\r\n\r\n if (fastCheck || !intersectInfo || currentIntersectInfo.distance < intersectInfo.distance) {\r\n intersectInfo = currentIntersectInfo;\r\n intersectInfo.faceId = faceId;\r\n\r\n if (fastCheck) {\r\n break;\r\n }\r\n }\r\n }\r\n }\r\n return intersectInfo;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n private _intersectUnIndexedTriangles(\r\n ray: Ray,\r\n positions: Vector3[],\r\n indices: IndicesArray,\r\n fastCheck?: boolean,\r\n trianglePredicate?: TrianglePickingPredicate\r\n ): Nullable {\r\n let intersectInfo: Nullable = null;\r\n // Triangles test\r\n for (let index = this.verticesStart; index < this.verticesStart + this.verticesCount; index += 3) {\r\n const p0 = positions[index];\r\n const p1 = positions[index + 1];\r\n const p2 = positions[index + 2];\r\n\r\n if (trianglePredicate && !trianglePredicate(p0, p1, p2, ray, -1, -1, -1)) {\r\n continue;\r\n }\r\n\r\n const currentIntersectInfo = ray.intersectsTriangle(p0, p1, p2);\r\n\r\n if (currentIntersectInfo) {\r\n if (currentIntersectInfo.distance < 0) {\r\n continue;\r\n }\r\n\r\n if (fastCheck || !intersectInfo || currentIntersectInfo.distance < intersectInfo.distance) {\r\n intersectInfo = currentIntersectInfo;\r\n intersectInfo.faceId = index / 3;\r\n\r\n if (fastCheck) {\r\n break;\r\n }\r\n }\r\n }\r\n }\r\n return intersectInfo;\r\n }\r\n\r\n /** @internal */\r\n public _rebuild(): void {\r\n if (this._linesIndexBuffer) {\r\n this._linesIndexBuffer = null;\r\n }\r\n }\r\n\r\n // Clone\r\n /**\r\n * Creates a new submesh from the passed mesh\r\n * @param newMesh defines the new hosting mesh\r\n * @param newRenderingMesh defines an optional rendering mesh\r\n * @returns the new submesh\r\n */\r\n public clone(newMesh: AbstractMesh, newRenderingMesh?: Mesh): SubMesh {\r\n const result = new SubMesh(this.materialIndex, this.verticesStart, this.verticesCount, this.indexStart, this.indexCount, newMesh, newRenderingMesh, false);\r\n\r\n if (!this.IsGlobal) {\r\n const boundingInfo = this.getBoundingInfo();\r\n\r\n if (!boundingInfo) {\r\n return result;\r\n }\r\n\r\n result._boundingInfo = new BoundingInfo(boundingInfo.minimum, boundingInfo.maximum);\r\n }\r\n\r\n return result;\r\n }\r\n\r\n // Dispose\r\n\r\n /**\r\n * Release associated resources\r\n */\r\n public dispose(): void {\r\n if (this._linesIndexBuffer) {\r\n this._mesh.getScene().getEngine()._releaseBuffer(this._linesIndexBuffer);\r\n this._linesIndexBuffer = null;\r\n }\r\n\r\n // Remove from mesh\r\n const index = this._mesh.subMeshes.indexOf(this);\r\n this._mesh.subMeshes.splice(index, 1);\r\n\r\n this.resetDrawCache();\r\n }\r\n\r\n /**\r\n * Gets the class name\r\n * @returns the string \"SubMesh\".\r\n */\r\n public getClassName(): string {\r\n return \"SubMesh\";\r\n }\r\n\r\n // Statics\r\n /**\r\n * Creates a new submesh from indices data\r\n * @param materialIndex the index of the main mesh material\r\n * @param startIndex the index where to start the copy in the mesh indices array\r\n * @param indexCount the number of indices to copy then from the startIndex\r\n * @param mesh the main mesh to create the submesh from\r\n * @param renderingMesh the optional rendering mesh\r\n * @param createBoundingBox defines if bounding box should be created for this submesh\r\n * @returns a new submesh\r\n */\r\n public static CreateFromIndices(\r\n materialIndex: number,\r\n startIndex: number,\r\n indexCount: number,\r\n mesh: AbstractMesh,\r\n renderingMesh?: Mesh,\r\n createBoundingBox: boolean = true\r\n ): SubMesh {\r\n let minVertexIndex = Number.MAX_VALUE;\r\n let maxVertexIndex = -Number.MAX_VALUE;\r\n\r\n const whatWillRender = renderingMesh || mesh;\r\n const indices = whatWillRender!.getIndices()!;\r\n\r\n for (let index = startIndex; index < startIndex + indexCount; index++) {\r\n const vertexIndex = indices[index];\r\n\r\n if (vertexIndex < minVertexIndex) {\r\n minVertexIndex = vertexIndex;\r\n }\r\n if (vertexIndex > maxVertexIndex) {\r\n maxVertexIndex = vertexIndex;\r\n }\r\n }\r\n\r\n return new SubMesh(materialIndex, minVertexIndex, maxVertexIndex - minVertexIndex + 1, startIndex, indexCount, mesh, renderingMesh, createBoundingBox);\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/no-unused-vars */\r\nimport type { Nullable, FloatArray, IndicesArray, DeepImmutable } from \"../types\";\r\nimport type { Matrix, Vector2 } from \"../Maths/math.vector\";\r\nimport { Vector3, Vector4, TmpVectors } from \"../Maths/math.vector\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport { _WarnImport } from \"../Misc/devTools\";\r\nimport type { Color3 } from \"../Maths/math.color\";\r\nimport { Color4 } from \"../Maths/math.color\";\r\nimport { Logger } from \"../Misc/logger\";\r\nimport { nativeOverride } from \"../Misc/decorators\";\r\nimport type { Coroutine } from \"../Misc/coroutine\";\r\nimport { makeSyncFunction, runCoroutineSync } from \"../Misc/coroutine\";\r\nimport type { ICreateCapsuleOptions } from \"./Builders/capsuleBuilder\";\r\nimport { RuntimeError, ErrorCodes } from \"../Misc/error\";\r\n\r\nimport type { Geometry } from \"../Meshes/geometry\";\r\nimport type { Mesh } from \"../Meshes/mesh\";\r\nimport { SubMesh } from \"./subMesh\";\r\n\r\n/**\r\n * Define an interface for all classes that will get and set the data on vertices\r\n */\r\nexport interface IGetSetVerticesData {\r\n /**\r\n * Gets a boolean indicating if specific vertex data is present\r\n * @param kind defines the vertex data kind to use\r\n * @returns true is data kind is present\r\n */\r\n isVerticesDataPresent(kind: string): boolean;\r\n /**\r\n * Gets a specific vertex data attached to this geometry. Float data is constructed if the vertex buffer data cannot be returned directly.\r\n * @param kind defines the data kind (Position, normal, etc...)\r\n * @param copyWhenShared defines if the returned array must be cloned upon returning it if the current geometry is shared between multiple meshes\r\n * @param forceCopy defines a boolean indicating that the returned array must be cloned upon returning it\r\n * @returns a float array containing vertex data\r\n */\r\n getVerticesData(kind: string, copyWhenShared?: boolean, forceCopy?: boolean): Nullable;\r\n /**\r\n * Returns an array of integers or a typed array (Int32Array, Uint32Array, Uint16Array) populated with the mesh indices.\r\n * @param copyWhenShared If true (default false) and and if the mesh geometry is shared among some other meshes, the returned array is a copy of the internal one.\r\n * @param forceCopy defines a boolean indicating that the returned array must be cloned upon returning it\r\n * @returns the indices array or an empty array if the mesh has no geometry\r\n */\r\n getIndices(copyWhenShared?: boolean, forceCopy?: boolean): Nullable;\r\n /**\r\n * Set specific vertex data\r\n * @param kind defines the data kind (Position, normal, etc...)\r\n * @param data defines the vertex data to use\r\n * @param updatable defines if the vertex must be flagged as updatable (false as default)\r\n * @param stride defines the stride to use (0 by default). This value is deduced from the kind value if not specified\r\n */\r\n setVerticesData(kind: string, data: FloatArray, updatable: boolean): void;\r\n /**\r\n * Update a specific associated vertex buffer\r\n * @param kind defines which buffer to write to (positions, indices, normals, etc). Possible `kind` values :\r\n * - VertexBuffer.PositionKind\r\n * - VertexBuffer.UVKind\r\n * - VertexBuffer.UV2Kind\r\n * - VertexBuffer.UV3Kind\r\n * - VertexBuffer.UV4Kind\r\n * - VertexBuffer.UV5Kind\r\n * - VertexBuffer.UV6Kind\r\n * - VertexBuffer.ColorKind\r\n * - VertexBuffer.MatricesIndicesKind\r\n * - VertexBuffer.MatricesIndicesExtraKind\r\n * - VertexBuffer.MatricesWeightsKind\r\n * - VertexBuffer.MatricesWeightsExtraKind\r\n * @param data defines the data source\r\n * @param updateExtends defines if extends info of the mesh must be updated (can be null). This is mostly useful for \"position\" kind\r\n * @param makeItUnique defines if the geometry associated with the mesh must be cloned to make the change only for this mesh (and not all meshes associated with the same geometry)\r\n */\r\n updateVerticesData(kind: string, data: FloatArray, updateExtends?: boolean, makeItUnique?: boolean): void;\r\n /**\r\n * Creates a new index buffer\r\n * @param indices defines the indices to store in the index buffer\r\n * @param totalVertices defines the total number of vertices (could be null)\r\n * @param updatable defines if the index buffer must be flagged as updatable (false by default)\r\n */\r\n setIndices(indices: IndicesArray, totalVertices: Nullable, updatable?: boolean): void;\r\n}\r\n\r\n/** Class used to attach material info to sub section of a vertex data class */\r\nexport class VertexDataMaterialInfo {\r\n /** Defines the material index to use */\r\n public materialIndex: number;\r\n /** Defines vertex index start*/\r\n public verticesStart: number;\r\n /** Defines vertices count */\r\n public verticesCount: number;\r\n /** Defines index start */\r\n public indexStart: number;\r\n /** Defines indices count */\r\n public indexCount: number;\r\n}\r\n\r\n/**\r\n * This class contains the various kinds of data on every vertex of a mesh used in determining its shape and appearance\r\n */\r\nexport class VertexData {\r\n /**\r\n * Mesh side orientation : usually the external or front surface\r\n */\r\n public static readonly FRONTSIDE = 0;\r\n /**\r\n * Mesh side orientation : usually the internal or back surface\r\n */\r\n public static readonly BACKSIDE = 1;\r\n /**\r\n * Mesh side orientation : both internal and external or front and back surfaces\r\n */\r\n public static readonly DOUBLESIDE = 2;\r\n /**\r\n * Mesh side orientation : by default, `FRONTSIDE`\r\n */\r\n public static readonly DEFAULTSIDE = 0;\r\n\r\n /**\r\n * An array of the x, y, z position of each vertex [...., x, y, z, .....]\r\n */\r\n public positions: Nullable;\r\n\r\n /**\r\n * An array of the x, y, z normal vector of each vertex [...., x, y, z, .....]\r\n */\r\n public normals: Nullable;\r\n\r\n /**\r\n * An array of the x, y, z tangent vector of each vertex [...., x, y, z, .....]\r\n */\r\n public tangents: Nullable;\r\n\r\n /**\r\n * An array of u,v which maps a texture image onto each vertex [...., u, v, .....]\r\n */\r\n public uvs: Nullable;\r\n\r\n /**\r\n * A second array of u,v which maps a texture image onto each vertex [...., u, v, .....]\r\n */\r\n public uvs2: Nullable;\r\n\r\n /**\r\n * A third array of u,v which maps a texture image onto each vertex [...., u, v, .....]\r\n */\r\n public uvs3: Nullable;\r\n\r\n /**\r\n * A fourth array of u,v which maps a texture image onto each vertex [...., u, v, .....]\r\n */\r\n public uvs4: Nullable;\r\n\r\n /**\r\n * A fifth array of u,v which maps a texture image onto each vertex [...., u, v, .....]\r\n */\r\n public uvs5: Nullable;\r\n\r\n /**\r\n * A sixth array of u,v which maps a texture image onto each vertex [...., u, v, .....]\r\n */\r\n public uvs6: Nullable;\r\n\r\n /**\r\n * An array of the r, g, b, a, color of each vertex [...., r, g, b, a, .....]\r\n */\r\n public colors: Nullable;\r\n\r\n /**\r\n * An array containing the list of indices to the array of matrices produced by bones, each vertex have up to 4 indices (8 if the matricesIndicesExtra is set).\r\n */\r\n public matricesIndices: Nullable;\r\n\r\n /**\r\n * An array containing the list of weights defining the weight of each indexed matrix in the final computation\r\n */\r\n public matricesWeights: Nullable;\r\n\r\n /**\r\n * An array extending the number of possible indices\r\n */\r\n public matricesIndicesExtra: Nullable;\r\n\r\n /**\r\n * An array extending the number of possible weights when the number of indices is extended\r\n */\r\n public matricesWeightsExtra: Nullable;\r\n\r\n /**\r\n * An array of i, j, k the three vertex indices required for each triangular facet [...., i, j, k .....]\r\n */\r\n public indices: Nullable;\r\n\r\n /**\r\n * An array defining material association for sub sections of the vertex data\r\n */\r\n public materialInfos: Nullable>;\r\n\r\n /**\r\n * Uses the passed data array to set the set the values for the specified kind of data\r\n * @param data a linear array of floating numbers\r\n * @param kind the type of data that is being set, eg positions, colors etc\r\n */\r\n public set(data: FloatArray, kind: string) {\r\n if (!data.length) {\r\n Logger.Warn(`Setting vertex data kind '${kind}' with an empty array`);\r\n }\r\n\r\n switch (kind) {\r\n case VertexBuffer.PositionKind:\r\n this.positions = data;\r\n break;\r\n case VertexBuffer.NormalKind:\r\n this.normals = data;\r\n break;\r\n case VertexBuffer.TangentKind:\r\n this.tangents = data;\r\n break;\r\n case VertexBuffer.UVKind:\r\n this.uvs = data;\r\n break;\r\n case VertexBuffer.UV2Kind:\r\n this.uvs2 = data;\r\n break;\r\n case VertexBuffer.UV3Kind:\r\n this.uvs3 = data;\r\n break;\r\n case VertexBuffer.UV4Kind:\r\n this.uvs4 = data;\r\n break;\r\n case VertexBuffer.UV5Kind:\r\n this.uvs5 = data;\r\n break;\r\n case VertexBuffer.UV6Kind:\r\n this.uvs6 = data;\r\n break;\r\n case VertexBuffer.ColorKind:\r\n this.colors = data;\r\n break;\r\n case VertexBuffer.MatricesIndicesKind:\r\n this.matricesIndices = data;\r\n break;\r\n case VertexBuffer.MatricesWeightsKind:\r\n this.matricesWeights = data;\r\n break;\r\n case VertexBuffer.MatricesIndicesExtraKind:\r\n this.matricesIndicesExtra = data;\r\n break;\r\n case VertexBuffer.MatricesWeightsExtraKind:\r\n this.matricesWeightsExtra = data;\r\n break;\r\n }\r\n }\r\n\r\n /**\r\n * Associates the vertexData to the passed Mesh.\r\n * Sets it as updatable or not (default `false`)\r\n * @param mesh the mesh the vertexData is applied to\r\n * @param updatable when used and having the value true allows new data to update the vertexData\r\n * @returns the VertexData\r\n */\r\n public applyToMesh(mesh: Mesh, updatable?: boolean): VertexData {\r\n this._applyTo(mesh, updatable, false);\r\n return this;\r\n }\r\n\r\n /**\r\n * Associates the vertexData to the passed Geometry.\r\n * Sets it as updatable or not (default `false`)\r\n * @param geometry the geometry the vertexData is applied to\r\n * @param updatable when used and having the value true allows new data to update the vertexData\r\n * @returns VertexData\r\n */\r\n public applyToGeometry(geometry: Geometry, updatable?: boolean): VertexData {\r\n this._applyTo(geometry, updatable, false);\r\n return this;\r\n }\r\n\r\n /**\r\n * Updates the associated mesh\r\n * @param mesh the mesh to be updated\r\n * @returns VertexData\r\n */\r\n public updateMesh(mesh: Mesh): VertexData {\r\n this._update(mesh);\r\n return this;\r\n }\r\n\r\n /**\r\n * Updates the associated geometry\r\n * @param geometry the geometry to be updated\r\n * @returns VertexData.\r\n */\r\n public updateGeometry(geometry: Geometry): VertexData {\r\n this._update(geometry);\r\n return this;\r\n }\r\n\r\n private readonly _applyTo = makeSyncFunction(this._applyToCoroutine.bind(this));\r\n\r\n /**\r\n * @internal\r\n */\r\n public *_applyToCoroutine(meshOrGeometry: IGetSetVerticesData, updatable: boolean = false, isAsync: boolean): Coroutine {\r\n if (this.positions) {\r\n meshOrGeometry.setVerticesData(VertexBuffer.PositionKind, this.positions, updatable);\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n\r\n if (this.normals) {\r\n meshOrGeometry.setVerticesData(VertexBuffer.NormalKind, this.normals, updatable);\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n\r\n if (this.tangents) {\r\n meshOrGeometry.setVerticesData(VertexBuffer.TangentKind, this.tangents, updatable);\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n\r\n if (this.uvs) {\r\n meshOrGeometry.setVerticesData(VertexBuffer.UVKind, this.uvs, updatable);\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n\r\n if (this.uvs2) {\r\n meshOrGeometry.setVerticesData(VertexBuffer.UV2Kind, this.uvs2, updatable);\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n\r\n if (this.uvs3) {\r\n meshOrGeometry.setVerticesData(VertexBuffer.UV3Kind, this.uvs3, updatable);\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n\r\n if (this.uvs4) {\r\n meshOrGeometry.setVerticesData(VertexBuffer.UV4Kind, this.uvs4, updatable);\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n\r\n if (this.uvs5) {\r\n meshOrGeometry.setVerticesData(VertexBuffer.UV5Kind, this.uvs5, updatable);\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n\r\n if (this.uvs6) {\r\n meshOrGeometry.setVerticesData(VertexBuffer.UV6Kind, this.uvs6, updatable);\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n\r\n if (this.colors) {\r\n meshOrGeometry.setVerticesData(VertexBuffer.ColorKind, this.colors, updatable);\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n\r\n if (this.matricesIndices) {\r\n meshOrGeometry.setVerticesData(VertexBuffer.MatricesIndicesKind, this.matricesIndices, updatable);\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n\r\n if (this.matricesWeights) {\r\n meshOrGeometry.setVerticesData(VertexBuffer.MatricesWeightsKind, this.matricesWeights, updatable);\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n\r\n if (this.matricesIndicesExtra) {\r\n meshOrGeometry.setVerticesData(VertexBuffer.MatricesIndicesExtraKind, this.matricesIndicesExtra, updatable);\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n\r\n if (this.matricesWeightsExtra) {\r\n meshOrGeometry.setVerticesData(VertexBuffer.MatricesWeightsExtraKind, this.matricesWeightsExtra, updatable);\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n\r\n if (this.indices) {\r\n meshOrGeometry.setIndices(this.indices, null, updatable);\r\n if (isAsync) {\r\n yield;\r\n }\r\n } else {\r\n meshOrGeometry.setIndices([], null);\r\n }\r\n\r\n if ((meshOrGeometry as Mesh).subMeshes && this.materialInfos && this.materialInfos.length > 1) {\r\n const mesh = meshOrGeometry as Mesh;\r\n mesh.subMeshes = [];\r\n for (const matInfo of this.materialInfos) {\r\n new SubMesh(matInfo.materialIndex, matInfo.verticesStart, matInfo.verticesCount, matInfo.indexStart, matInfo.indexCount, mesh);\r\n }\r\n }\r\n\r\n return this;\r\n }\r\n\r\n private _update(meshOrGeometry: IGetSetVerticesData, updateExtends?: boolean, makeItUnique?: boolean): VertexData {\r\n if (this.positions) {\r\n meshOrGeometry.updateVerticesData(VertexBuffer.PositionKind, this.positions, updateExtends, makeItUnique);\r\n }\r\n\r\n if (this.normals) {\r\n meshOrGeometry.updateVerticesData(VertexBuffer.NormalKind, this.normals, updateExtends, makeItUnique);\r\n }\r\n\r\n if (this.tangents) {\r\n meshOrGeometry.updateVerticesData(VertexBuffer.TangentKind, this.tangents, updateExtends, makeItUnique);\r\n }\r\n\r\n if (this.uvs) {\r\n meshOrGeometry.updateVerticesData(VertexBuffer.UVKind, this.uvs, updateExtends, makeItUnique);\r\n }\r\n\r\n if (this.uvs2) {\r\n meshOrGeometry.updateVerticesData(VertexBuffer.UV2Kind, this.uvs2, updateExtends, makeItUnique);\r\n }\r\n\r\n if (this.uvs3) {\r\n meshOrGeometry.updateVerticesData(VertexBuffer.UV3Kind, this.uvs3, updateExtends, makeItUnique);\r\n }\r\n\r\n if (this.uvs4) {\r\n meshOrGeometry.updateVerticesData(VertexBuffer.UV4Kind, this.uvs4, updateExtends, makeItUnique);\r\n }\r\n\r\n if (this.uvs5) {\r\n meshOrGeometry.updateVerticesData(VertexBuffer.UV5Kind, this.uvs5, updateExtends, makeItUnique);\r\n }\r\n\r\n if (this.uvs6) {\r\n meshOrGeometry.updateVerticesData(VertexBuffer.UV6Kind, this.uvs6, updateExtends, makeItUnique);\r\n }\r\n\r\n if (this.colors) {\r\n meshOrGeometry.updateVerticesData(VertexBuffer.ColorKind, this.colors, updateExtends, makeItUnique);\r\n }\r\n\r\n if (this.matricesIndices) {\r\n meshOrGeometry.updateVerticesData(VertexBuffer.MatricesIndicesKind, this.matricesIndices, updateExtends, makeItUnique);\r\n }\r\n\r\n if (this.matricesWeights) {\r\n meshOrGeometry.updateVerticesData(VertexBuffer.MatricesWeightsKind, this.matricesWeights, updateExtends, makeItUnique);\r\n }\r\n\r\n if (this.matricesIndicesExtra) {\r\n meshOrGeometry.updateVerticesData(VertexBuffer.MatricesIndicesExtraKind, this.matricesIndicesExtra, updateExtends, makeItUnique);\r\n }\r\n\r\n if (this.matricesWeightsExtra) {\r\n meshOrGeometry.updateVerticesData(VertexBuffer.MatricesWeightsExtraKind, this.matricesWeightsExtra, updateExtends, makeItUnique);\r\n }\r\n\r\n if (this.indices) {\r\n meshOrGeometry.setIndices(this.indices, null);\r\n }\r\n return this;\r\n }\r\n\r\n @nativeOverride.filter((...[coordinates]: Parameters) => !Array.isArray(coordinates))\r\n private static _TransformVector3Coordinates(coordinates: FloatArray, transformation: DeepImmutable, offset = 0, length = coordinates.length) {\r\n const coordinate = TmpVectors.Vector3[0];\r\n const transformedCoordinate = TmpVectors.Vector3[1];\r\n for (let index = offset; index < offset + length; index += 3) {\r\n Vector3.FromArrayToRef(coordinates, index, coordinate);\r\n Vector3.TransformCoordinatesToRef(coordinate, transformation, transformedCoordinate);\r\n coordinates[index] = transformedCoordinate.x;\r\n coordinates[index + 1] = transformedCoordinate.y;\r\n coordinates[index + 2] = transformedCoordinate.z;\r\n }\r\n }\r\n\r\n @nativeOverride.filter((...[normals]: Parameters) => !Array.isArray(normals))\r\n private static _TransformVector3Normals(normals: FloatArray, transformation: DeepImmutable, offset = 0, length = normals.length) {\r\n const normal = TmpVectors.Vector3[0];\r\n const transformedNormal = TmpVectors.Vector3[1];\r\n for (let index = offset; index < offset + length; index += 3) {\r\n Vector3.FromArrayToRef(normals, index, normal);\r\n Vector3.TransformNormalToRef(normal, transformation, transformedNormal);\r\n normals[index] = transformedNormal.x;\r\n normals[index + 1] = transformedNormal.y;\r\n normals[index + 2] = transformedNormal.z;\r\n }\r\n }\r\n\r\n @nativeOverride.filter((...[normals]: Parameters) => !Array.isArray(normals))\r\n private static _TransformVector4Normals(normals: FloatArray, transformation: DeepImmutable, offset = 0, length = normals.length) {\r\n const normal = TmpVectors.Vector4[0];\r\n const transformedNormal = TmpVectors.Vector4[1];\r\n for (let index = offset; index < offset + length; index += 4) {\r\n Vector4.FromArrayToRef(normals, index, normal);\r\n Vector4.TransformNormalToRef(normal, transformation, transformedNormal);\r\n normals[index] = transformedNormal.x;\r\n normals[index + 1] = transformedNormal.y;\r\n normals[index + 2] = transformedNormal.z;\r\n normals[index + 3] = transformedNormal.w;\r\n }\r\n }\r\n\r\n @nativeOverride.filter((...[indices]: Parameters) => !Array.isArray(indices))\r\n private static _FlipFaces(indices: IndicesArray, offset = 0, length = indices.length) {\r\n for (let index = offset; index < offset + length; index += 3) {\r\n const tmp = indices[index + 1];\r\n indices[index + 1] = indices[index + 2];\r\n indices[index + 2] = tmp;\r\n }\r\n }\r\n\r\n /**\r\n * Transforms each position and each normal of the vertexData according to the passed Matrix\r\n * @param matrix the transforming matrix\r\n * @returns the VertexData\r\n */\r\n public transform(matrix: Matrix): VertexData {\r\n const flip = matrix.determinant() < 0;\r\n if (this.positions) {\r\n VertexData._TransformVector3Coordinates(this.positions, matrix);\r\n }\r\n\r\n if (this.normals) {\r\n VertexData._TransformVector3Normals(this.normals, matrix);\r\n }\r\n\r\n if (this.tangents) {\r\n VertexData._TransformVector4Normals(this.tangents, matrix);\r\n }\r\n\r\n if (flip && this.indices) {\r\n VertexData._FlipFaces(this.indices);\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Generates an array of vertex data where each vertex data only has one material info\r\n * @returns An array of VertexData\r\n */\r\n public splitBasedOnMaterialID() {\r\n if (!this.materialInfos || this.materialInfos.length < 2) {\r\n return [this];\r\n }\r\n\r\n const result = new Array();\r\n for (const materialInfo of this.materialInfos) {\r\n const vertexData = new VertexData();\r\n\r\n if (this.positions) {\r\n vertexData.positions = this.positions.slice(materialInfo.verticesStart * 3, (materialInfo.verticesCount + materialInfo.verticesStart) * 3);\r\n }\r\n\r\n if (this.normals) {\r\n vertexData.normals = this.normals.slice(materialInfo.verticesStart * 3, (materialInfo.verticesCount + materialInfo.verticesStart) * 3);\r\n }\r\n\r\n if (this.tangents) {\r\n vertexData.tangents = this.tangents.slice(materialInfo.verticesStart * 4, (materialInfo.verticesCount + materialInfo.verticesStart) * 4);\r\n }\r\n\r\n if (this.colors) {\r\n vertexData.colors = this.colors.slice(materialInfo.verticesStart * 4, (materialInfo.verticesCount + materialInfo.verticesStart) * 4);\r\n }\r\n\r\n if (this.uvs) {\r\n vertexData.uvs = this.uvs.slice(materialInfo.verticesStart * 2, (materialInfo.verticesCount + materialInfo.verticesStart) * 2);\r\n }\r\n\r\n if (this.uvs2) {\r\n vertexData.uvs2 = this.uvs2.slice(materialInfo.verticesStart * 2, (materialInfo.verticesCount + materialInfo.verticesStart) * 2);\r\n }\r\n\r\n if (this.uvs3) {\r\n vertexData.uvs3 = this.uvs3.slice(materialInfo.verticesStart * 2, (materialInfo.verticesCount + materialInfo.verticesStart) * 2);\r\n }\r\n\r\n if (this.uvs4) {\r\n vertexData.uvs4 = this.uvs4.slice(materialInfo.verticesStart * 2, (materialInfo.verticesCount + materialInfo.verticesStart) * 2);\r\n }\r\n\r\n if (this.uvs5) {\r\n vertexData.uvs5 = this.uvs5.slice(materialInfo.verticesStart * 2, (materialInfo.verticesCount + materialInfo.verticesStart) * 2);\r\n }\r\n\r\n if (this.uvs6) {\r\n vertexData.uvs6 = this.uvs6.slice(materialInfo.verticesStart * 2, (materialInfo.verticesCount + materialInfo.verticesStart) * 2);\r\n }\r\n\r\n if (this.matricesIndices) {\r\n vertexData.matricesIndices = this.matricesIndices.slice(materialInfo.verticesStart * 4, (materialInfo.verticesCount + materialInfo.verticesStart) * 4);\r\n }\r\n\r\n if (this.matricesIndicesExtra) {\r\n vertexData.matricesIndicesExtra = this.matricesIndicesExtra.slice(materialInfo.verticesStart * 4, (materialInfo.verticesCount + materialInfo.verticesStart) * 4);\r\n }\r\n\r\n if (this.matricesWeights) {\r\n vertexData.matricesWeights = this.matricesWeights.slice(materialInfo.verticesStart * 4, (materialInfo.verticesCount + materialInfo.verticesStart) * 4);\r\n }\r\n\r\n if (this.matricesWeightsExtra) {\r\n vertexData.matricesWeightsExtra = this.matricesWeightsExtra.slice(materialInfo.verticesStart * 4, (materialInfo.verticesCount + materialInfo.verticesStart) * 4);\r\n }\r\n\r\n if (this.indices) {\r\n vertexData.indices = [];\r\n for (let index = materialInfo.indexStart; index < materialInfo.indexStart + materialInfo.indexCount; index++) {\r\n vertexData.indices.push(this.indices[index] - materialInfo.verticesStart);\r\n }\r\n }\r\n\r\n const newMaterialInfo = new VertexDataMaterialInfo();\r\n newMaterialInfo.indexStart = 0;\r\n newMaterialInfo.indexCount = vertexData.indices ? vertexData.indices.length : 0;\r\n newMaterialInfo.materialIndex = materialInfo.materialIndex;\r\n newMaterialInfo.verticesStart = 0;\r\n newMaterialInfo.verticesCount = (vertexData.positions ? vertexData.positions.length : 0) / 3;\r\n vertexData.materialInfos = [newMaterialInfo];\r\n\r\n result.push(vertexData);\r\n }\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Merges the passed VertexData into the current one\r\n * @param others the VertexData to be merged into the current one\r\n * @param use32BitsIndices defines a boolean indicating if indices must be store in a 32 bits array\r\n * @param forceCloneIndices defines a boolean indicating if indices are forced to be cloned\r\n * @param mergeMaterialIds defines a boolean indicating if we need to merge the material infos\r\n * @param enableCompletion defines a boolean indicating if the vertex data should be completed to be compatible\r\n * @returns the modified VertexData\r\n */\r\n public merge(others: VertexData | VertexData[], use32BitsIndices = false, forceCloneIndices = false, mergeMaterialIds = false, enableCompletion = false) {\r\n const vertexDatas: { vertexData: VertexData; transform?: Matrix }[] = Array.isArray(others)\r\n ? others.map((other) => {\r\n return { vertexData: other };\r\n })\r\n : [{ vertexData: others }];\r\n return runCoroutineSync(this._mergeCoroutine(undefined, vertexDatas, use32BitsIndices, false, forceCloneIndices, mergeMaterialIds, enableCompletion));\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public *_mergeCoroutine(\r\n transform: Matrix | undefined,\r\n vertexDatas: { vertexData: VertexData; transform?: Matrix }[],\r\n use32BitsIndices = false,\r\n isAsync: boolean,\r\n forceCloneIndices: boolean,\r\n mergeMaterialIds = false,\r\n enableCompletion = false\r\n ): Coroutine {\r\n this._validate();\r\n\r\n let others = vertexDatas.map((vertexData) => vertexData.vertexData);\r\n // eslint-disable-next-line @typescript-eslint/no-this-alias\r\n let root: VertexData = this;\r\n\r\n for (const other of others) {\r\n if (!other) {\r\n continue;\r\n }\r\n\r\n other._validate();\r\n\r\n if (!enableCompletion) {\r\n if (\r\n !this.normals !== !other.normals ||\r\n !this.tangents !== !other.tangents ||\r\n !this.uvs !== !other.uvs ||\r\n !this.uvs2 !== !other.uvs2 ||\r\n !this.uvs3 !== !other.uvs3 ||\r\n !this.uvs4 !== !other.uvs4 ||\r\n !this.uvs5 !== !other.uvs5 ||\r\n !this.uvs6 !== !other.uvs6 ||\r\n !this.colors !== !other.colors ||\r\n !this.matricesIndices !== !other.matricesIndices ||\r\n !this.matricesWeights !== !other.matricesWeights ||\r\n !this.matricesIndicesExtra !== !other.matricesIndicesExtra ||\r\n !this.matricesWeightsExtra !== !other.matricesWeightsExtra\r\n ) {\r\n throw new Error(\"Cannot merge vertex data that do not have the same set of attributes\");\r\n }\r\n } else {\r\n if (!this.normals !== !other.normals) {\r\n if (!this.normals) {\r\n this.normals = Array.from(other.normals!);\r\n } else {\r\n other.normals = Array.from(this.normals);\r\n }\r\n }\r\n\r\n if (!this.tangents !== !other.tangents) {\r\n if (!this.tangents) {\r\n this.tangents = Array.from(other.tangents!);\r\n } else {\r\n other.tangents = Array.from(this.tangents);\r\n }\r\n }\r\n\r\n if (!this.uvs !== !other.uvs) {\r\n if (!this.uvs) {\r\n this.uvs = Array.from(other.uvs!);\r\n } else {\r\n other.uvs = Array.from(this.uvs);\r\n }\r\n }\r\n\r\n if (!this.uvs2 !== !other.uvs2) {\r\n if (!this.uvs2) {\r\n this.uvs2 = Array.from(other.uvs2!);\r\n } else {\r\n other.uvs2 = Array.from(this.uvs2);\r\n }\r\n }\r\n\r\n if (!this.uvs3 !== !other.uvs3) {\r\n if (!this.uvs3) {\r\n this.uvs3 = Array.from(other.uvs3!);\r\n } else {\r\n other.uvs3 = Array.from(this.uvs3);\r\n }\r\n }\r\n\r\n if (!this.uvs4 !== !other.uvs4) {\r\n if (!this.uvs4) {\r\n this.uvs4 = Array.from(other.uvs4!);\r\n } else {\r\n other.uvs4 = Array.from(this.uvs4);\r\n }\r\n }\r\n\r\n if (!this.uvs5 !== !other.uvs5) {\r\n if (!this.uvs5) {\r\n this.uvs5 = Array.from(other.uvs5!);\r\n } else {\r\n other.uvs5 = Array.from(this.uvs5);\r\n }\r\n }\r\n\r\n if (!this.uvs6 !== !other.uvs6) {\r\n if (!this.uvs6) {\r\n this.uvs6 = Array.from(other.uvs6!);\r\n } else {\r\n other.uvs6 = Array.from(this.uvs6);\r\n }\r\n }\r\n\r\n if (!this.colors !== !other.colors) {\r\n if (!this.colors) {\r\n this.colors = Array.from(other.colors!);\r\n } else {\r\n other.colors = Array.from(this.colors);\r\n }\r\n }\r\n\r\n if (!this.matricesIndices !== !other.matricesIndices) {\r\n if (!this.matricesIndices) {\r\n this.matricesIndices = Array.from(other.matricesIndices!);\r\n } else {\r\n other.matricesIndices = Array.from(this.matricesIndices);\r\n }\r\n }\r\n\r\n if (!this.matricesWeights !== !other.matricesWeights) {\r\n if (!this.matricesWeights) {\r\n this.matricesWeights = Array.from(other.matricesWeights!);\r\n } else {\r\n other.matricesWeights = Array.from(this.matricesWeights);\r\n }\r\n }\r\n\r\n if (!this.matricesIndicesExtra !== !other.matricesIndicesExtra) {\r\n if (!this.matricesIndicesExtra) {\r\n this.matricesIndicesExtra = Array.from(other.matricesIndicesExtra!);\r\n } else {\r\n other.matricesIndicesExtra = Array.from(this.matricesIndicesExtra);\r\n }\r\n }\r\n\r\n if (!this.matricesWeightsExtra !== !other.matricesWeightsExtra) {\r\n if (!this.matricesWeightsExtra) {\r\n this.matricesWeightsExtra = Array.from(other.matricesWeightsExtra!);\r\n } else {\r\n other.matricesWeightsExtra = Array.from(this.matricesWeightsExtra);\r\n }\r\n }\r\n }\r\n }\r\n\r\n if (mergeMaterialIds) {\r\n // Merge material infos\r\n let materialIndex = 0;\r\n let indexOffset = 0;\r\n let vertexOffset = 0;\r\n const materialInfos: VertexDataMaterialInfo[] = [];\r\n let currentMaterialInfo: Nullable = null;\r\n const vertexDataList: { vertexData: VertexData; transform?: Matrix }[] = [];\r\n\r\n // We need to split vertexData with more than one materialInfo\r\n for (const split of this.splitBasedOnMaterialID()) {\r\n vertexDataList.push({ vertexData: split, transform: transform });\r\n }\r\n\r\n for (const vertexData of vertexDatas) {\r\n for (const split of vertexData.vertexData.splitBasedOnMaterialID()) {\r\n vertexDataList.push({ vertexData: split, transform: vertexData.transform });\r\n }\r\n }\r\n\r\n // Sort by material IDs\r\n vertexDataList.sort((a, b) => {\r\n const matInfoA = a.vertexData.materialInfos ? a.vertexData.materialInfos[0].materialIndex : 0;\r\n const matInfoB = b.vertexData.materialInfos ? b.vertexData.materialInfos[0].materialIndex : 0;\r\n\r\n if (matInfoA > matInfoB) {\r\n return 1;\r\n }\r\n\r\n if (matInfoA === matInfoB) {\r\n return 0;\r\n }\r\n\r\n return -1;\r\n });\r\n\r\n // Build the new material info\r\n for (const vertexDataSource of vertexDataList) {\r\n const vertexData = vertexDataSource.vertexData;\r\n if (vertexData.materialInfos) {\r\n materialIndex = vertexData.materialInfos[0].materialIndex;\r\n } else {\r\n materialIndex = 0;\r\n }\r\n if (currentMaterialInfo && currentMaterialInfo.materialIndex === materialIndex) {\r\n currentMaterialInfo.indexCount += vertexData.indices!.length;\r\n currentMaterialInfo.verticesCount += vertexData.positions!.length / 3;\r\n } else {\r\n const materialInfo = new VertexDataMaterialInfo();\r\n materialInfo.materialIndex = materialIndex;\r\n materialInfo.indexStart = indexOffset;\r\n materialInfo.indexCount = vertexData.indices!.length;\r\n materialInfo.verticesStart = vertexOffset;\r\n materialInfo.verticesCount = vertexData.positions!.length / 3;\r\n\r\n materialInfos.push(materialInfo);\r\n currentMaterialInfo = materialInfo;\r\n }\r\n indexOffset += vertexData.indices!.length;\r\n vertexOffset += vertexData.positions!.length / 3;\r\n }\r\n // Extract sorted values\r\n const first = vertexDataList.splice(0, 1)[0];\r\n root = first.vertexData;\r\n transform = first.transform;\r\n others = vertexDataList.map((v) => v.vertexData);\r\n vertexDatas = vertexDataList;\r\n\r\n this.materialInfos = materialInfos;\r\n }\r\n\r\n // Merge geometries\r\n const totalIndices = others.reduce((indexSum, vertexData) => indexSum + (vertexData.indices?.length ?? 0), root.indices?.length ?? 0);\r\n const sliceIndices = forceCloneIndices || others.some((vertexData) => vertexData.indices === root.indices);\r\n let indices = sliceIndices ? root.indices?.slice() : root.indices;\r\n if (totalIndices > 0) {\r\n let indicesOffset = indices?.length ?? 0;\r\n\r\n if (!indices) {\r\n indices = new Array(totalIndices);\r\n }\r\n\r\n if (indices.length !== totalIndices) {\r\n if (Array.isArray(indices)) {\r\n indices.length = totalIndices;\r\n } else {\r\n const temp = use32BitsIndices || indices instanceof Uint32Array ? new Uint32Array(totalIndices) : new Uint16Array(totalIndices);\r\n temp.set(indices);\r\n indices = temp;\r\n }\r\n\r\n if (transform && transform.determinant() < 0) {\r\n VertexData._FlipFaces(indices, 0, indicesOffset);\r\n }\r\n }\r\n\r\n let positionsOffset = root.positions ? root.positions.length / 3 : 0;\r\n for (const { vertexData: other, transform } of vertexDatas) {\r\n if (other.indices) {\r\n for (let index = 0; index < other.indices.length; index++) {\r\n indices[indicesOffset + index] = other.indices[index] + positionsOffset;\r\n }\r\n\r\n if (transform && transform.determinant() < 0) {\r\n VertexData._FlipFaces(indices, indicesOffset, other.indices.length);\r\n }\r\n\r\n // The call to _validate already checked for positions\r\n positionsOffset += other.positions!.length / 3;\r\n indicesOffset += other.indices.length;\r\n\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n }\r\n }\r\n\r\n this.indices = indices!;\r\n\r\n this.positions = VertexData._MergeElement(\r\n VertexBuffer.PositionKind,\r\n root.positions,\r\n transform,\r\n vertexDatas.map((other) => [other.vertexData.positions, other.transform])\r\n );\r\n if (isAsync) {\r\n yield;\r\n }\r\n if (root.normals) {\r\n this.normals = VertexData._MergeElement(\r\n VertexBuffer.NormalKind,\r\n root.normals,\r\n transform,\r\n vertexDatas.map((other) => [other.vertexData.normals, other.transform])\r\n );\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n if (root.tangents) {\r\n this.tangents = VertexData._MergeElement(\r\n VertexBuffer.TangentKind,\r\n root.tangents,\r\n transform,\r\n vertexDatas.map((other) => [other.vertexData.tangents, other.transform])\r\n );\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n if (root.uvs) {\r\n this.uvs = VertexData._MergeElement(\r\n VertexBuffer.UVKind,\r\n root.uvs,\r\n transform,\r\n vertexDatas.map((other) => [other.vertexData.uvs, other.transform])\r\n );\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n if (root.uvs2) {\r\n this.uvs2 = VertexData._MergeElement(\r\n VertexBuffer.UV2Kind,\r\n root.uvs2,\r\n transform,\r\n vertexDatas.map((other) => [other.vertexData.uvs2, other.transform])\r\n );\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n if (root.uvs3) {\r\n this.uvs3 = VertexData._MergeElement(\r\n VertexBuffer.UV3Kind,\r\n root.uvs3,\r\n transform,\r\n vertexDatas.map((other) => [other.vertexData.uvs3, other.transform])\r\n );\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n if (root.uvs4) {\r\n this.uvs4 = VertexData._MergeElement(\r\n VertexBuffer.UV4Kind,\r\n root.uvs4,\r\n transform,\r\n vertexDatas.map((other) => [other.vertexData.uvs4, other.transform])\r\n );\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n if (root.uvs5) {\r\n this.uvs5 = VertexData._MergeElement(\r\n VertexBuffer.UV5Kind,\r\n root.uvs5,\r\n transform,\r\n vertexDatas.map((other) => [other.vertexData.uvs5, other.transform])\r\n );\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n if (root.uvs6) {\r\n this.uvs6 = VertexData._MergeElement(\r\n VertexBuffer.UV6Kind,\r\n root.uvs6,\r\n transform,\r\n vertexDatas.map((other) => [other.vertexData.uvs6, other.transform])\r\n );\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n if (root.colors) {\r\n this.colors = VertexData._MergeElement(\r\n VertexBuffer.ColorKind,\r\n root.colors,\r\n transform,\r\n vertexDatas.map((other) => [other.vertexData.colors, other.transform])\r\n );\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n if (root.matricesIndices) {\r\n this.matricesIndices = VertexData._MergeElement(\r\n VertexBuffer.MatricesIndicesKind,\r\n root.matricesIndices,\r\n transform,\r\n vertexDatas.map((other) => [other.vertexData.matricesIndices, other.transform])\r\n );\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n if (root.matricesWeights) {\r\n this.matricesWeights = VertexData._MergeElement(\r\n VertexBuffer.MatricesWeightsKind,\r\n root.matricesWeights,\r\n transform,\r\n vertexDatas.map((other) => [other.vertexData.matricesWeights, other.transform])\r\n );\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n if (root.matricesIndicesExtra) {\r\n this.matricesIndicesExtra = VertexData._MergeElement(\r\n VertexBuffer.MatricesIndicesExtraKind,\r\n root.matricesIndicesExtra,\r\n transform,\r\n vertexDatas.map((other) => [other.vertexData.matricesIndicesExtra, other.transform])\r\n );\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n if (root.matricesWeightsExtra) {\r\n this.matricesWeightsExtra = VertexData._MergeElement(\r\n VertexBuffer.MatricesWeightsExtraKind,\r\n root.matricesWeightsExtra,\r\n transform,\r\n vertexDatas.map((other) => [other.vertexData.matricesWeightsExtra, other.transform])\r\n );\r\n }\r\n\r\n return this;\r\n }\r\n\r\n private static _MergeElement(\r\n kind: string,\r\n source: Nullable,\r\n transform: Matrix | undefined,\r\n others: readonly (readonly [element: Nullable, transform?: Matrix])[]\r\n ): Nullable {\r\n const nonNullOthers = others.filter((other): other is [element: FloatArray, transform?: Matrix] => other[0] !== null && other[0] !== undefined);\r\n\r\n // If there is no source to copy and no other non-null sources then skip this element.\r\n if (!source && nonNullOthers.length == 0) {\r\n return source;\r\n }\r\n\r\n if (!source) {\r\n return this._MergeElement(kind, nonNullOthers[0][0], nonNullOthers[0][1], nonNullOthers.slice(1));\r\n }\r\n\r\n const len = nonNullOthers.reduce((sumLen, elements) => sumLen + elements[0].length, source.length);\r\n\r\n const transformRange =\r\n kind === VertexBuffer.PositionKind\r\n ? VertexData._TransformVector3Coordinates\r\n : kind === VertexBuffer.NormalKind\r\n ? VertexData._TransformVector3Normals\r\n : kind === VertexBuffer.TangentKind\r\n ? VertexData._TransformVector4Normals\r\n : () => {};\r\n\r\n if (source instanceof Float32Array) {\r\n // use non-loop method when the source is Float32Array\r\n const ret32 = new Float32Array(len);\r\n ret32.set(source);\r\n transform && transformRange(ret32, transform, 0, source.length);\r\n\r\n let offset = source.length;\r\n for (const [vertexData, transform] of nonNullOthers) {\r\n ret32.set(vertexData, offset);\r\n transform && transformRange(ret32, transform, offset, vertexData.length);\r\n offset += vertexData.length;\r\n }\r\n return ret32;\r\n } else {\r\n // don't use concat as it is super slow, just loop for other cases\r\n const ret = new Array(len);\r\n for (let i = 0; i < source.length; i++) {\r\n ret[i] = source[i];\r\n }\r\n transform && transformRange(ret, transform, 0, source.length);\r\n\r\n let offset = source.length;\r\n for (const [vertexData, transform] of nonNullOthers) {\r\n for (let i = 0; i < vertexData.length; i++) {\r\n ret[offset + i] = vertexData[i];\r\n }\r\n transform && transformRange(ret, transform, offset, vertexData.length);\r\n offset += vertexData.length;\r\n }\r\n return ret;\r\n }\r\n }\r\n\r\n private _validate(): void {\r\n if (!this.positions) {\r\n throw new RuntimeError(\"Positions are required\", ErrorCodes.MeshInvalidPositionsError);\r\n }\r\n\r\n const getElementCount = (kind: string, values: FloatArray) => {\r\n const stride = VertexBuffer.DeduceStride(kind);\r\n if (values.length % stride !== 0) {\r\n throw new Error(\"The \" + kind + \"s array count must be a multiple of \" + stride);\r\n }\r\n\r\n return values.length / stride;\r\n };\r\n\r\n const positionsElementCount = getElementCount(VertexBuffer.PositionKind, this.positions);\r\n\r\n const validateElementCount = (kind: string, values: FloatArray) => {\r\n const elementCount = getElementCount(kind, values);\r\n if (elementCount !== positionsElementCount) {\r\n throw new Error(\"The \" + kind + \"s element count (\" + elementCount + \") does not match the positions count (\" + positionsElementCount + \")\");\r\n }\r\n };\r\n\r\n if (this.normals) {\r\n validateElementCount(VertexBuffer.NormalKind, this.normals);\r\n }\r\n if (this.tangents) {\r\n validateElementCount(VertexBuffer.TangentKind, this.tangents);\r\n }\r\n if (this.uvs) {\r\n validateElementCount(VertexBuffer.UVKind, this.uvs);\r\n }\r\n if (this.uvs2) {\r\n validateElementCount(VertexBuffer.UV2Kind, this.uvs2);\r\n }\r\n if (this.uvs3) {\r\n validateElementCount(VertexBuffer.UV3Kind, this.uvs3);\r\n }\r\n if (this.uvs4) {\r\n validateElementCount(VertexBuffer.UV4Kind, this.uvs4);\r\n }\r\n if (this.uvs5) {\r\n validateElementCount(VertexBuffer.UV5Kind, this.uvs5);\r\n }\r\n if (this.uvs6) {\r\n validateElementCount(VertexBuffer.UV6Kind, this.uvs6);\r\n }\r\n if (this.colors) {\r\n validateElementCount(VertexBuffer.ColorKind, this.colors);\r\n }\r\n if (this.matricesIndices) {\r\n validateElementCount(VertexBuffer.MatricesIndicesKind, this.matricesIndices);\r\n }\r\n if (this.matricesWeights) {\r\n validateElementCount(VertexBuffer.MatricesWeightsKind, this.matricesWeights);\r\n }\r\n if (this.matricesIndicesExtra) {\r\n validateElementCount(VertexBuffer.MatricesIndicesExtraKind, this.matricesIndicesExtra);\r\n }\r\n if (this.matricesWeightsExtra) {\r\n validateElementCount(VertexBuffer.MatricesWeightsExtraKind, this.matricesWeightsExtra);\r\n }\r\n }\r\n\r\n /**\r\n * Clone the current vertex data\r\n * @returns a copy of the current data\r\n */\r\n public clone() {\r\n const serializationObject = this.serialize();\r\n return VertexData.Parse(serializationObject);\r\n }\r\n\r\n /**\r\n * Serializes the VertexData\r\n * @returns a serialized object\r\n */\r\n public serialize(): any {\r\n const serializationObject: any = {};\r\n\r\n if (this.positions) {\r\n serializationObject.positions = Array.from(this.positions);\r\n }\r\n\r\n if (this.normals) {\r\n serializationObject.normals = Array.from(this.normals);\r\n }\r\n\r\n if (this.tangents) {\r\n serializationObject.tangents = Array.from(this.tangents);\r\n }\r\n\r\n if (this.uvs) {\r\n serializationObject.uvs = Array.from(this.uvs);\r\n }\r\n\r\n if (this.uvs2) {\r\n serializationObject.uvs2 = Array.from(this.uvs2);\r\n }\r\n\r\n if (this.uvs3) {\r\n serializationObject.uvs3 = Array.from(this.uvs3);\r\n }\r\n\r\n if (this.uvs4) {\r\n serializationObject.uvs4 = Array.from(this.uvs4);\r\n }\r\n\r\n if (this.uvs5) {\r\n serializationObject.uvs5 = Array.from(this.uvs5);\r\n }\r\n\r\n if (this.uvs6) {\r\n serializationObject.uvs6 = Array.from(this.uvs6);\r\n }\r\n\r\n if (this.colors) {\r\n serializationObject.colors = Array.from(this.colors);\r\n }\r\n\r\n if (this.matricesIndices) {\r\n serializationObject.matricesIndices = Array.from(this.matricesIndices);\r\n serializationObject.matricesIndices._isExpanded = true;\r\n }\r\n\r\n if (this.matricesWeights) {\r\n serializationObject.matricesWeights = Array.from(this.matricesWeights);\r\n }\r\n\r\n if (this.matricesIndicesExtra) {\r\n serializationObject.matricesIndicesExtra = Array.from(this.matricesIndicesExtra);\r\n serializationObject.matricesIndicesExtra._isExpanded = true;\r\n }\r\n\r\n if (this.matricesWeightsExtra) {\r\n serializationObject.matricesWeightsExtra = Array.from(this.matricesWeightsExtra);\r\n }\r\n\r\n serializationObject.indices = Array.from(this.indices as number[]);\r\n\r\n if (this.materialInfos) {\r\n serializationObject.materialInfos = [];\r\n for (const materialInfo of this.materialInfos) {\r\n const materialInfoSerializationObject = {\r\n indexStart: materialInfo.indexStart,\r\n indexCount: materialInfo.indexCount,\r\n materialIndex: materialInfo.materialIndex,\r\n verticesStart: materialInfo.verticesStart,\r\n verticesCount: materialInfo.verticesCount,\r\n };\r\n serializationObject.materialInfos.push(materialInfoSerializationObject);\r\n }\r\n }\r\n\r\n return serializationObject;\r\n }\r\n\r\n // Statics\r\n /**\r\n * Extracts the vertexData from a mesh\r\n * @param mesh the mesh from which to extract the VertexData\r\n * @param copyWhenShared defines if the VertexData must be cloned when shared between multiple meshes, optional, default false\r\n * @param forceCopy indicating that the VertexData must be cloned, optional, default false\r\n * @returns the object VertexData associated to the passed mesh\r\n */\r\n public static ExtractFromMesh(mesh: Mesh, copyWhenShared?: boolean, forceCopy?: boolean): VertexData {\r\n return VertexData._ExtractFrom(mesh, copyWhenShared, forceCopy);\r\n }\r\n\r\n /**\r\n * Extracts the vertexData from the geometry\r\n * @param geometry the geometry from which to extract the VertexData\r\n * @param copyWhenShared defines if the VertexData must be cloned when the geometry is shared between multiple meshes, optional, default false\r\n * @param forceCopy indicating that the VertexData must be cloned, optional, default false\r\n * @returns the object VertexData associated to the passed mesh\r\n */\r\n public static ExtractFromGeometry(geometry: Geometry, copyWhenShared?: boolean, forceCopy?: boolean): VertexData {\r\n return VertexData._ExtractFrom(geometry, copyWhenShared, forceCopy);\r\n }\r\n\r\n private static _ExtractFrom(meshOrGeometry: IGetSetVerticesData, copyWhenShared?: boolean, forceCopy?: boolean): VertexData {\r\n const result = new VertexData();\r\n\r\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.PositionKind)) {\r\n result.positions = meshOrGeometry.getVerticesData(VertexBuffer.PositionKind, copyWhenShared, forceCopy);\r\n }\r\n\r\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.NormalKind)) {\r\n result.normals = meshOrGeometry.getVerticesData(VertexBuffer.NormalKind, copyWhenShared, forceCopy);\r\n }\r\n\r\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.TangentKind)) {\r\n result.tangents = meshOrGeometry.getVerticesData(VertexBuffer.TangentKind, copyWhenShared, forceCopy);\r\n }\r\n\r\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.UVKind)) {\r\n result.uvs = meshOrGeometry.getVerticesData(VertexBuffer.UVKind, copyWhenShared, forceCopy);\r\n }\r\n\r\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.UV2Kind)) {\r\n result.uvs2 = meshOrGeometry.getVerticesData(VertexBuffer.UV2Kind, copyWhenShared, forceCopy);\r\n }\r\n\r\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.UV3Kind)) {\r\n result.uvs3 = meshOrGeometry.getVerticesData(VertexBuffer.UV3Kind, copyWhenShared, forceCopy);\r\n }\r\n\r\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.UV4Kind)) {\r\n result.uvs4 = meshOrGeometry.getVerticesData(VertexBuffer.UV4Kind, copyWhenShared, forceCopy);\r\n }\r\n\r\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.UV5Kind)) {\r\n result.uvs5 = meshOrGeometry.getVerticesData(VertexBuffer.UV5Kind, copyWhenShared, forceCopy);\r\n }\r\n\r\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.UV6Kind)) {\r\n result.uvs6 = meshOrGeometry.getVerticesData(VertexBuffer.UV6Kind, copyWhenShared, forceCopy);\r\n }\r\n\r\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.ColorKind)) {\r\n result.colors = meshOrGeometry.getVerticesData(VertexBuffer.ColorKind, copyWhenShared, forceCopy);\r\n }\r\n\r\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.MatricesIndicesKind)) {\r\n result.matricesIndices = meshOrGeometry.getVerticesData(VertexBuffer.MatricesIndicesKind, copyWhenShared, forceCopy);\r\n }\r\n\r\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.MatricesWeightsKind)) {\r\n result.matricesWeights = meshOrGeometry.getVerticesData(VertexBuffer.MatricesWeightsKind, copyWhenShared, forceCopy);\r\n }\r\n\r\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.MatricesIndicesExtraKind)) {\r\n result.matricesIndicesExtra = meshOrGeometry.getVerticesData(VertexBuffer.MatricesIndicesExtraKind, copyWhenShared, forceCopy);\r\n }\r\n\r\n if (meshOrGeometry.isVerticesDataPresent(VertexBuffer.MatricesWeightsExtraKind)) {\r\n result.matricesWeightsExtra = meshOrGeometry.getVerticesData(VertexBuffer.MatricesWeightsExtraKind, copyWhenShared, forceCopy);\r\n }\r\n\r\n result.indices = meshOrGeometry.getIndices(copyWhenShared, forceCopy);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates the VertexData for a Ribbon\r\n * @param options an object used to set the following optional parameters for the ribbon, required but can be empty\r\n * * pathArray array of paths, each of which an array of successive Vector3\r\n * * closeArray creates a seam between the first and the last paths of the pathArray, optional, default false\r\n * * closePath creates a seam between the first and the last points of each path of the path array, optional, default false\r\n * * offset a positive integer, only used when pathArray contains a single path (offset = 10 means the point 1 is joined to the point 11), default rounded half size of the pathArray length\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * * invertUV swaps in the U and V coordinates when applying a texture, optional, default false\r\n * * uvs a linear array, of length 2 * number of vertices, of custom UV values, optional\r\n * * colors a linear array, of length 4 * number of vertices, of custom color values, optional\r\n * @param options.pathArray\r\n * @param options.closeArray\r\n * @param options.closePath\r\n * @param options.offset\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param options.invertUV\r\n * @param options.uvs\r\n * @param options.colors\r\n * @returns the VertexData of the ribbon\r\n * @deprecated use CreateRibbonVertexData instead\r\n */\r\n public static CreateRibbon(options: {\r\n pathArray: Vector3[][];\r\n closeArray?: boolean;\r\n closePath?: boolean;\r\n offset?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n invertUV?: boolean;\r\n uvs?: Vector2[];\r\n colors?: Color4[];\r\n }): VertexData {\r\n throw _WarnImport(\"ribbonBuilder\");\r\n }\r\n\r\n /**\r\n * Creates the VertexData for a box\r\n * @param options an object used to set the following optional parameters for the box, required but can be empty\r\n * * size sets the width, height and depth of the box to the value of size, optional default 1\r\n * * width sets the width (x direction) of the box, overwrites the width set by size, optional, default size\r\n * * height sets the height (y direction) of the box, overwrites the height set by size, optional, default size\r\n * * depth sets the depth (z direction) of the box, overwrites the depth set by size, optional, default size\r\n * * faceUV an array of 6 Vector4 elements used to set different images to each box side\r\n * * faceColors an array of 6 Color3 elements used to set different colors to each box side\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.size\r\n * @param options.width\r\n * @param options.height\r\n * @param options.depth\r\n * @param options.faceUV\r\n * @param options.faceColors\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @returns the VertexData of the box\r\n * @deprecated Please use CreateBoxVertexData from the BoxBuilder file instead\r\n */\r\n public static CreateBox(options: {\r\n size?: number;\r\n width?: number;\r\n height?: number;\r\n depth?: number;\r\n faceUV?: Vector4[];\r\n faceColors?: Color4[];\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n }): VertexData {\r\n throw _WarnImport(\"boxBuilder\");\r\n }\r\n\r\n /**\r\n * Creates the VertexData for a tiled box\r\n * @param options an object used to set the following optional parameters for the box, required but can be empty\r\n * * faceTiles sets the pattern, tile size and number of tiles for a face\r\n * * faceUV an array of 6 Vector4 elements used to set different images to each box side\r\n * * faceColors an array of 6 Color3 elements used to set different colors to each box side\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * @param options.pattern\r\n * @param options.width\r\n * @param options.height\r\n * @param options.depth\r\n * @param options.tileSize\r\n * @param options.tileWidth\r\n * @param options.tileHeight\r\n * @param options.alignHorizontal\r\n * @param options.alignVertical\r\n * @param options.faceUV\r\n * @param options.faceColors\r\n * @param options.sideOrientation\r\n * @returns the VertexData of the box\r\n * @deprecated Please use CreateTiledBoxVertexData instead\r\n */\r\n public static CreateTiledBox(options: {\r\n pattern?: number;\r\n width?: number;\r\n height?: number;\r\n depth?: number;\r\n tileSize?: number;\r\n tileWidth?: number;\r\n tileHeight?: number;\r\n alignHorizontal?: number;\r\n alignVertical?: number;\r\n faceUV?: Vector4[];\r\n faceColors?: Color4[];\r\n sideOrientation?: number;\r\n }): VertexData {\r\n throw _WarnImport(\"tiledBoxBuilder\");\r\n }\r\n\r\n /**\r\n * Creates the VertexData for a tiled plane\r\n * @param options an object used to set the following optional parameters for the box, required but can be empty\r\n * * pattern a limited pattern arrangement depending on the number\r\n * * tileSize sets the width, height and depth of the tile to the value of size, optional default 1\r\n * * tileWidth sets the width (x direction) of the tile, overwrites the width set by size, optional, default size\r\n * * tileHeight sets the height (y direction) of the tile, overwrites the height set by size, optional, default size\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.pattern\r\n * @param options.tileSize\r\n * @param options.tileWidth\r\n * @param options.tileHeight\r\n * @param options.size\r\n * @param options.width\r\n * @param options.height\r\n * @param options.alignHorizontal\r\n * @param options.alignVertical\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @returns the VertexData of the tiled plane\r\n * @deprecated use CreateTiledPlaneVertexData instead\r\n */\r\n public static CreateTiledPlane(options: {\r\n pattern?: number;\r\n tileSize?: number;\r\n tileWidth?: number;\r\n tileHeight?: number;\r\n size?: number;\r\n width?: number;\r\n height?: number;\r\n alignHorizontal?: number;\r\n alignVertical?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n }): VertexData {\r\n throw _WarnImport(\"tiledPlaneBuilder\");\r\n }\r\n\r\n /**\r\n * Creates the VertexData for an ellipsoid, defaults to a sphere\r\n * @param options an object used to set the following optional parameters for the box, required but can be empty\r\n * * segments sets the number of horizontal strips optional, default 32\r\n * * diameter sets the axes dimensions, diameterX, diameterY and diameterZ to the value of diameter, optional default 1\r\n * * diameterX sets the diameterX (x direction) of the ellipsoid, overwrites the diameterX set by diameter, optional, default diameter\r\n * * diameterY sets the diameterY (y direction) of the ellipsoid, overwrites the diameterY set by diameter, optional, default diameter\r\n * * diameterZ sets the diameterZ (z direction) of the ellipsoid, overwrites the diameterZ set by diameter, optional, default diameter\r\n * * arc a number from 0 to 1, to create an unclosed ellipsoid based on the fraction of the circumference (latitude) given by the arc value, optional, default 1\r\n * * slice a number from 0 to 1, to create an unclosed ellipsoid based on the fraction of the height (latitude) given by the arc value, optional, default 1\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.segments\r\n * @param options.diameter\r\n * @param options.diameterX\r\n * @param options.diameterY\r\n * @param options.diameterZ\r\n * @param options.arc\r\n * @param options.slice\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @returns the VertexData of the ellipsoid\r\n * @deprecated use CreateSphereVertexData instead\r\n */\r\n public static CreateSphere(options: {\r\n segments?: number;\r\n diameter?: number;\r\n diameterX?: number;\r\n diameterY?: number;\r\n diameterZ?: number;\r\n arc?: number;\r\n slice?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n }): VertexData {\r\n throw _WarnImport(\"sphereBuilder\");\r\n }\r\n\r\n /**\r\n * Creates the VertexData for a cylinder, cone or prism\r\n * @param options an object used to set the following optional parameters for the box, required but can be empty\r\n * * height sets the height (y direction) of the cylinder, optional, default 2\r\n * * diameterTop sets the diameter of the top of the cone, overwrites diameter, optional, default diameter\r\n * * diameterBottom sets the diameter of the bottom of the cone, overwrites diameter, optional, default diameter\r\n * * diameter sets the diameter of the top and bottom of the cone, optional default 1\r\n * * tessellation the number of prism sides, 3 for a triangular prism, optional, default 24\r\n * * subdivisions` the number of rings along the cylinder height, optional, default 1\r\n * * arc a number from 0 to 1, to create an unclosed cylinder based on the fraction of the circumference given by the arc value, optional, default 1\r\n * * faceColors an array of Color3 elements used to set different colors to the top, rings and bottom respectively\r\n * * faceUV an array of Vector4 elements used to set different images to the top, rings and bottom respectively\r\n * * hasRings when true makes each subdivision independently treated as a face for faceUV and faceColors, optional, default false\r\n * * enclose when true closes an open cylinder by adding extra flat faces between the height axis and vertical edges, think cut cake\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.height\r\n * @param options.diameterTop\r\n * @param options.diameterBottom\r\n * @param options.diameter\r\n * @param options.tessellation\r\n * @param options.subdivisions\r\n * @param options.arc\r\n * @param options.faceColors\r\n * @param options.faceUV\r\n * @param options.hasRings\r\n * @param options.enclose\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @returns the VertexData of the cylinder, cone or prism\r\n * @deprecated please use CreateCylinderVertexData instead\r\n */\r\n public static CreateCylinder(options: {\r\n height?: number;\r\n diameterTop?: number;\r\n diameterBottom?: number;\r\n diameter?: number;\r\n tessellation?: number;\r\n subdivisions?: number;\r\n arc?: number;\r\n faceColors?: Color4[];\r\n faceUV?: Vector4[];\r\n hasRings?: boolean;\r\n enclose?: boolean;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n }): VertexData {\r\n throw _WarnImport(\"cylinderBuilder\");\r\n }\r\n\r\n /**\r\n * Creates the VertexData for a torus\r\n * @param options an object used to set the following optional parameters for the box, required but can be empty\r\n * * diameter the diameter of the torus, optional default 1\r\n * * thickness the diameter of the tube forming the torus, optional default 0.5\r\n * * tessellation the number of prism sides, 3 for a triangular prism, optional, default 24\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.diameter\r\n * @param options.thickness\r\n * @param options.tessellation\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @returns the VertexData of the torus\r\n * @deprecated use CreateTorusVertexData instead\r\n */\r\n public static CreateTorus(options: {\r\n diameter?: number;\r\n thickness?: number;\r\n tessellation?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n }): VertexData {\r\n throw _WarnImport(\"torusBuilder\");\r\n }\r\n\r\n /**\r\n * Creates the VertexData of the LineSystem\r\n * @param options an object used to set the following optional parameters for the LineSystem, required but can be empty\r\n * - lines an array of lines, each line being an array of successive Vector3\r\n * - colors an array of line colors, each of the line colors being an array of successive Color4, one per line point\r\n * @param options.lines\r\n * @param options.colors\r\n * @returns the VertexData of the LineSystem\r\n * @deprecated use CreateLineSystemVertexData instead\r\n */\r\n public static CreateLineSystem(options: { lines: Vector3[][]; colors?: Nullable }): VertexData {\r\n throw _WarnImport(\"linesBuilder\");\r\n }\r\n\r\n /**\r\n * Create the VertexData for a DashedLines\r\n * @param options an object used to set the following optional parameters for the DashedLines, required but can be empty\r\n * - points an array successive Vector3\r\n * - dashSize the size of the dashes relative to the dash number, optional, default 3\r\n * - gapSize the size of the gap between two successive dashes relative to the dash number, optional, default 1\r\n * - dashNb the intended total number of dashes, optional, default 200\r\n * @param options.points\r\n * @param options.dashSize\r\n * @param options.gapSize\r\n * @param options.dashNb\r\n * @returns the VertexData for the DashedLines\r\n * @deprecated use CreateDashedLinesVertexData instead\r\n */\r\n public static CreateDashedLines(options: { points: Vector3[]; dashSize?: number; gapSize?: number; dashNb?: number }): VertexData {\r\n throw _WarnImport(\"linesBuilder\");\r\n }\r\n\r\n /**\r\n * Creates the VertexData for a Ground\r\n * @param options an object used to set the following optional parameters for the Ground, required but can be empty\r\n * - width the width (x direction) of the ground, optional, default 1\r\n * - height the height (z direction) of the ground, optional, default 1\r\n * - subdivisions the number of subdivisions per side, optional, default 1\r\n * @param options.width\r\n * @param options.height\r\n * @param options.subdivisions\r\n * @param options.subdivisionsX\r\n * @param options.subdivisionsY\r\n * @returns the VertexData of the Ground\r\n * @deprecated Please use CreateGroundVertexData instead\r\n */\r\n public static CreateGround(options: { width?: number; height?: number; subdivisions?: number; subdivisionsX?: number; subdivisionsY?: number }): VertexData {\r\n throw _WarnImport(\"groundBuilder\");\r\n }\r\n\r\n /**\r\n * Creates the VertexData for a TiledGround by subdividing the ground into tiles\r\n * @param options an object used to set the following optional parameters for the Ground, required but can be empty\r\n * * xmin the ground minimum X coordinate, optional, default -1\r\n * * zmin the ground minimum Z coordinate, optional, default -1\r\n * * xmax the ground maximum X coordinate, optional, default 1\r\n * * zmax the ground maximum Z coordinate, optional, default 1\r\n * * subdivisions a javascript object {w: positive integer, h: positive integer}, `w` and `h` are the numbers of subdivisions on the ground width and height creating 'tiles', default {w: 6, h: 6}\r\n * * precision a javascript object {w: positive integer, h: positive integer}, `w` and `h` are the numbers of subdivisions on the tile width and height, default {w: 2, h: 2}\r\n * @param options.xmin\r\n * @param options.zmin\r\n * @param options.xmax\r\n * @param options.zmax\r\n * @param options.subdivisions\r\n * @param options.subdivisions.w\r\n * @param options.subdivisions.h\r\n * @param options.precision\r\n * @param options.precision.w\r\n * @param options.precision.h\r\n * @returns the VertexData of the TiledGround\r\n * @deprecated use CreateTiledGroundVertexData instead\r\n */\r\n public static CreateTiledGround(options: {\r\n xmin: number;\r\n zmin: number;\r\n xmax: number;\r\n zmax: number;\r\n subdivisions?: { w: number; h: number };\r\n precision?: { w: number; h: number };\r\n }): VertexData {\r\n throw _WarnImport(\"groundBuilder\");\r\n }\r\n\r\n /**\r\n * Creates the VertexData of the Ground designed from a heightmap\r\n * @param options an object used to set the following parameters for the Ground, required and provided by CreateGroundFromHeightMap\r\n * * width the width (x direction) of the ground\r\n * * height the height (z direction) of the ground\r\n * * subdivisions the number of subdivisions per side\r\n * * minHeight the minimum altitude on the ground, optional, default 0\r\n * * maxHeight the maximum altitude on the ground, optional default 1\r\n * * colorFilter the filter to apply to the image pixel colors to compute the height, optional Color3, default (0.3, 0.59, 0.11)\r\n * * buffer the array holding the image color data\r\n * * bufferWidth the width of image\r\n * * bufferHeight the height of image\r\n * * alphaFilter Remove any data where the alpha channel is below this value, defaults 0 (all data visible)\r\n * @param options.width\r\n * @param options.height\r\n * @param options.subdivisions\r\n * @param options.minHeight\r\n * @param options.maxHeight\r\n * @param options.colorFilter\r\n * @param options.buffer\r\n * @param options.bufferWidth\r\n * @param options.bufferHeight\r\n * @param options.alphaFilter\r\n * @returns the VertexData of the Ground designed from a heightmap\r\n * @deprecated use CreateGroundFromHeightMapVertexData instead\r\n */\r\n public static CreateGroundFromHeightMap(options: {\r\n width: number;\r\n height: number;\r\n subdivisions: number;\r\n minHeight: number;\r\n maxHeight: number;\r\n colorFilter: Color3;\r\n buffer: Uint8Array;\r\n bufferWidth: number;\r\n bufferHeight: number;\r\n alphaFilter: number;\r\n }): VertexData {\r\n throw _WarnImport(\"groundBuilder\");\r\n }\r\n\r\n /**\r\n * Creates the VertexData for a Plane\r\n * @param options an object used to set the following optional parameters for the plane, required but can be empty\r\n * * size sets the width and height of the plane to the value of size, optional default 1\r\n * * width sets the width (x direction) of the plane, overwrites the width set by size, optional, default size\r\n * * height sets the height (y direction) of the plane, overwrites the height set by size, optional, default size\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.size\r\n * @param options.width\r\n * @param options.height\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @returns the VertexData of the box\r\n * @deprecated use CreatePlaneVertexData instead\r\n */\r\n public static CreatePlane(options: { size?: number; width?: number; height?: number; sideOrientation?: number; frontUVs?: Vector4; backUVs?: Vector4 }): VertexData {\r\n throw _WarnImport(\"planeBuilder\");\r\n }\r\n\r\n /**\r\n * Creates the VertexData of the Disc or regular Polygon\r\n * @param options an object used to set the following optional parameters for the disc, required but can be empty\r\n * * radius the radius of the disc, optional default 0.5\r\n * * tessellation the number of polygon sides, optional, default 64\r\n * * arc a number from 0 to 1, to create an unclosed polygon based on the fraction of the circumference given by the arc value, optional, default 1\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.radius\r\n * @param options.tessellation\r\n * @param options.arc\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @returns the VertexData of the box\r\n * @deprecated use CreateDiscVertexData instead\r\n */\r\n public static CreateDisc(options: { radius?: number; tessellation?: number; arc?: number; sideOrientation?: number; frontUVs?: Vector4; backUVs?: Vector4 }): VertexData {\r\n throw _WarnImport(\"discBuilder\");\r\n }\r\n\r\n /**\r\n * Creates the VertexData for an irregular Polygon in the XoZ plane using a mesh built by polygonTriangulation.build()\r\n * All parameters are provided by CreatePolygon as needed\r\n * @param polygon a mesh built from polygonTriangulation.build()\r\n * @param sideOrientation takes the values Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * @param fUV an array of Vector4 elements used to set different images to the top, rings and bottom respectively\r\n * @param fColors an array of Color3 elements used to set different colors to the top, rings and bottom respectively\r\n * @param frontUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * @param backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param wrap a boolean, default false, when true and fUVs used texture is wrapped around all sides, when false texture is applied side\r\n * @returns the VertexData of the Polygon\r\n * @deprecated use CreatePolygonVertexData instead\r\n */\r\n public static CreatePolygon(polygon: Mesh, sideOrientation: number, fUV?: Vector4[], fColors?: Color4[], frontUVs?: Vector4, backUVs?: Vector4, wrap?: boolean): VertexData {\r\n throw _WarnImport(\"polygonBuilder\");\r\n }\r\n\r\n /**\r\n * Creates the VertexData of the IcoSphere\r\n * @param options an object used to set the following optional parameters for the IcoSphere, required but can be empty\r\n * * radius the radius of the IcoSphere, optional default 1\r\n * * radiusX allows stretching in the x direction, optional, default radius\r\n * * radiusY allows stretching in the y direction, optional, default radius\r\n * * radiusZ allows stretching in the z direction, optional, default radius\r\n * * flat when true creates a flat shaded mesh, optional, default true\r\n * * subdivisions increasing the subdivisions increases the number of faces, optional, default 4\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.radius\r\n * @param options.radiusX\r\n * @param options.radiusY\r\n * @param options.radiusZ\r\n * @param options.flat\r\n * @param options.subdivisions\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @returns the VertexData of the IcoSphere\r\n * @deprecated use CreateIcoSphereVertexData instead\r\n */\r\n public static CreateIcoSphere(options: {\r\n radius?: number;\r\n radiusX?: number;\r\n radiusY?: number;\r\n radiusZ?: number;\r\n flat?: boolean;\r\n subdivisions?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n }): VertexData {\r\n throw _WarnImport(\"icoSphereBuilder\");\r\n }\r\n\r\n // inspired from // http://stemkoski.github.io/Three.js/Polyhedra.html\r\n /**\r\n * Creates the VertexData for a Polyhedron\r\n * @param options an object used to set the following optional parameters for the polyhedron, required but can be empty\r\n * * type provided types are:\r\n * * 0 : Tetrahedron, 1 : Octahedron, 2 : Dodecahedron, 3 : Icosahedron, 4 : Rhombicuboctahedron, 5 : Triangular Prism, 6 : Pentagonal Prism, 7 : Hexagonal Prism, 8 : Square Pyramid (J1)\r\n * * 9 : Pentagonal Pyramid (J2), 10 : Triangular Dipyramid (J12), 11 : Pentagonal Dipyramid (J13), 12 : Elongated Square Dipyramid (J15), 13 : Elongated Pentagonal Dipyramid (J16), 14 : Elongated Pentagonal Cupola (J20)\r\n * * size the size of the IcoSphere, optional default 1\r\n * * sizeX allows stretching in the x direction, optional, default size\r\n * * sizeY allows stretching in the y direction, optional, default size\r\n * * sizeZ allows stretching in the z direction, optional, default size\r\n * * custom a number that overwrites the type to create from an extended set of polyhedron from https://www.babylonjs-playground.com/#21QRSK#15 with minimised editor\r\n * * faceUV an array of Vector4 elements used to set different images to the top, rings and bottom respectively\r\n * * faceColors an array of Color3 elements used to set different colors to the top, rings and bottom respectively\r\n * * flat when true creates a flat shaded mesh, optional, default true\r\n * * subdivisions increasing the subdivisions increases the number of faces, optional, default 4\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.type\r\n * @param options.size\r\n * @param options.sizeX\r\n * @param options.sizeY\r\n * @param options.sizeZ\r\n * @param options.custom\r\n * @param options.faceUV\r\n * @param options.faceColors\r\n * @param options.flat\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @returns the VertexData of the Polyhedron\r\n * @deprecated use CreatePolyhedronVertexData instead\r\n */\r\n public static CreatePolyhedron(options: {\r\n type?: number;\r\n size?: number;\r\n sizeX?: number;\r\n sizeY?: number;\r\n sizeZ?: number;\r\n custom?: any;\r\n faceUV?: Vector4[];\r\n faceColors?: Color4[];\r\n flat?: boolean;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n }): VertexData {\r\n throw _WarnImport(\"polyhedronBuilder\");\r\n }\r\n\r\n /**\r\n * Creates the VertexData for a Capsule, inspired from https://github.com/maximeq/three-js-capsule-geometry/blob/master/src/CapsuleBufferGeometry.js\r\n * @param options an object used to set the following optional parameters for the capsule, required but can be empty\r\n * @returns the VertexData of the Capsule\r\n * @deprecated Please use CreateCapsuleVertexData from the capsuleBuilder file instead\r\n */\r\n public static CreateCapsule(\r\n options: ICreateCapsuleOptions = {\r\n orientation: Vector3.Up(),\r\n subdivisions: 2,\r\n tessellation: 16,\r\n height: 1,\r\n radius: 0.25,\r\n capSubdivisions: 6,\r\n }\r\n ): VertexData {\r\n throw _WarnImport(\"capsuleBuilder\");\r\n }\r\n\r\n // based on http://code.google.com/p/away3d/source/browse/trunk/fp10/Away3D/src/away3d/primitives/TorusKnot.as?spec=svn2473&r=2473\r\n /**\r\n * Creates the VertexData for a TorusKnot\r\n * @param options an object used to set the following optional parameters for the TorusKnot, required but can be empty\r\n * * radius the radius of the torus knot, optional, default 2\r\n * * tube the thickness of the tube, optional, default 0.5\r\n * * radialSegments the number of sides on each tube segments, optional, default 32\r\n * * tubularSegments the number of tubes to decompose the knot into, optional, default 32\r\n * * p the number of windings around the z axis, optional, default 2\r\n * * q the number of windings around the x axis, optional, default 3\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.radius\r\n * @param options.tube\r\n * @param options.radialSegments\r\n * @param options.tubularSegments\r\n * @param options.p\r\n * @param options.q\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @returns the VertexData of the Torus Knot\r\n * @deprecated use CreateTorusKnotVertexData instead\r\n */\r\n public static CreateTorusKnot(options: {\r\n radius?: number;\r\n tube?: number;\r\n radialSegments?: number;\r\n tubularSegments?: number;\r\n p?: number;\r\n q?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n }): VertexData {\r\n throw _WarnImport(\"torusKnotBuilder\");\r\n }\r\n\r\n // Tools\r\n\r\n /**\r\n * Compute normals for given positions and indices\r\n * @param positions an array of vertex positions, [...., x, y, z, ......]\r\n * @param indices an array of indices in groups of three for each triangular facet, [...., i, j, k, ......]\r\n * @param normals an array of vertex normals, [...., x, y, z, ......]\r\n * @param options an object used to set the following optional parameters for the TorusKnot, optional\r\n * * facetNormals : optional array of facet normals (vector3)\r\n * * facetPositions : optional array of facet positions (vector3)\r\n * * facetPartitioning : optional partitioning array. facetPositions is required for facetPartitioning computation\r\n * * ratio : optional partitioning ratio / bounding box, required for facetPartitioning computation\r\n * * bInfo : optional bounding info, required for facetPartitioning computation\r\n * * bbSize : optional bounding box size data, required for facetPartitioning computation\r\n * * subDiv : optional partitioning data about subdivisions on each axis (int), required for facetPartitioning computation\r\n * * useRightHandedSystem: optional boolean to for right handed system computation\r\n * * depthSort : optional boolean to enable the facet depth sort computation\r\n * * distanceTo : optional Vector3 to compute the facet depth from this location\r\n * * depthSortedFacets : optional array of depthSortedFacets to store the facet distances from the reference location\r\n * @param options.facetNormals\r\n * @param options.facetPositions\r\n * @param options.facetPartitioning\r\n * @param options.ratio\r\n * @param options.bInfo\r\n * @param options.bbSize\r\n * @param options.subDiv\r\n * @param options.useRightHandedSystem\r\n * @param options.depthSort\r\n * @param options.distanceTo\r\n * @param options.depthSortedFacets\r\n */\r\n public static ComputeNormals(\r\n positions: any,\r\n indices: any,\r\n normals: any,\r\n options?: {\r\n facetNormals?: any;\r\n facetPositions?: any;\r\n facetPartitioning?: any;\r\n ratio?: number;\r\n bInfo?: any;\r\n bbSize?: Vector3;\r\n subDiv?: any;\r\n useRightHandedSystem?: boolean;\r\n depthSort?: boolean;\r\n distanceTo?: Vector3;\r\n depthSortedFacets?: any;\r\n }\r\n ): void {\r\n // temporary scalar variables\r\n let index = 0; // facet index\r\n let p1p2x = 0.0; // p1p2 vector x coordinate\r\n let p1p2y = 0.0; // p1p2 vector y coordinate\r\n let p1p2z = 0.0; // p1p2 vector z coordinate\r\n let p3p2x = 0.0; // p3p2 vector x coordinate\r\n let p3p2y = 0.0; // p3p2 vector y coordinate\r\n let p3p2z = 0.0; // p3p2 vector z coordinate\r\n let faceNormalx = 0.0; // facet normal x coordinate\r\n let faceNormaly = 0.0; // facet normal y coordinate\r\n let faceNormalz = 0.0; // facet normal z coordinate\r\n let length = 0.0; // facet normal length before normalization\r\n let v1x = 0; // vector1 x index in the positions array\r\n let v1y = 0; // vector1 y index in the positions array\r\n let v1z = 0; // vector1 z index in the positions array\r\n let v2x = 0; // vector2 x index in the positions array\r\n let v2y = 0; // vector2 y index in the positions array\r\n let v2z = 0; // vector2 z index in the positions array\r\n let v3x = 0; // vector3 x index in the positions array\r\n let v3y = 0; // vector3 y index in the positions array\r\n let v3z = 0; // vector3 z index in the positions array\r\n let computeFacetNormals = false;\r\n let computeFacetPositions = false;\r\n let computeFacetPartitioning = false;\r\n let computeDepthSort = false;\r\n let faceNormalSign = 1;\r\n let ratio = 0;\r\n let distanceTo: Nullable = null;\r\n if (options) {\r\n computeFacetNormals = options.facetNormals ? true : false;\r\n computeFacetPositions = options.facetPositions ? true : false;\r\n computeFacetPartitioning = options.facetPartitioning ? true : false;\r\n faceNormalSign = options.useRightHandedSystem === true ? -1 : 1;\r\n ratio = options.ratio || 0;\r\n computeDepthSort = options.depthSort ? true : false;\r\n distanceTo = options.distanceTo;\r\n if (computeDepthSort) {\r\n if (distanceTo === undefined) {\r\n distanceTo = Vector3.Zero();\r\n }\r\n }\r\n }\r\n\r\n // facetPartitioning reinit if needed\r\n let xSubRatio = 0;\r\n let ySubRatio = 0;\r\n let zSubRatio = 0;\r\n let subSq = 0;\r\n if (computeFacetPartitioning && options && options.bbSize) {\r\n //let bbSizeMax = options.bbSize.x > options.bbSize.y ? options.bbSize.x : options.bbSize.y;\r\n //bbSizeMax = bbSizeMax > options.bbSize.z ? bbSizeMax : options.bbSize.z;\r\n xSubRatio = (options.subDiv.X * ratio) / options.bbSize.x;\r\n ySubRatio = (options.subDiv.Y * ratio) / options.bbSize.y;\r\n zSubRatio = (options.subDiv.Z * ratio) / options.bbSize.z;\r\n subSq = options.subDiv.max * options.subDiv.max;\r\n options.facetPartitioning.length = 0;\r\n }\r\n\r\n // reset the normals\r\n for (index = 0; index < positions.length; index++) {\r\n normals[index] = 0.0;\r\n }\r\n\r\n // Loop : 1 indice triplet = 1 facet\r\n const nbFaces = (indices.length / 3) | 0;\r\n for (index = 0; index < nbFaces; index++) {\r\n // get the indexes of the coordinates of each vertex of the facet\r\n v1x = indices[index * 3] * 3;\r\n v1y = v1x + 1;\r\n v1z = v1x + 2;\r\n v2x = indices[index * 3 + 1] * 3;\r\n v2y = v2x + 1;\r\n v2z = v2x + 2;\r\n v3x = indices[index * 3 + 2] * 3;\r\n v3y = v3x + 1;\r\n v3z = v3x + 2;\r\n\r\n p1p2x = positions[v1x] - positions[v2x]; // compute two vectors per facet : p1p2 and p3p2\r\n p1p2y = positions[v1y] - positions[v2y];\r\n p1p2z = positions[v1z] - positions[v2z];\r\n\r\n p3p2x = positions[v3x] - positions[v2x];\r\n p3p2y = positions[v3y] - positions[v2y];\r\n p3p2z = positions[v3z] - positions[v2z];\r\n\r\n // compute the face normal with the cross product\r\n faceNormalx = faceNormalSign * (p1p2y * p3p2z - p1p2z * p3p2y);\r\n faceNormaly = faceNormalSign * (p1p2z * p3p2x - p1p2x * p3p2z);\r\n faceNormalz = faceNormalSign * (p1p2x * p3p2y - p1p2y * p3p2x);\r\n // normalize this normal and store it in the array facetData\r\n length = Math.sqrt(faceNormalx * faceNormalx + faceNormaly * faceNormaly + faceNormalz * faceNormalz);\r\n length = length === 0 ? 1.0 : length;\r\n faceNormalx /= length;\r\n faceNormaly /= length;\r\n faceNormalz /= length;\r\n\r\n if (computeFacetNormals && options) {\r\n options.facetNormals[index].x = faceNormalx;\r\n options.facetNormals[index].y = faceNormaly;\r\n options.facetNormals[index].z = faceNormalz;\r\n }\r\n\r\n if (computeFacetPositions && options) {\r\n // compute and the facet barycenter coordinates in the array facetPositions\r\n options.facetPositions[index].x = (positions[v1x] + positions[v2x] + positions[v3x]) / 3.0;\r\n options.facetPositions[index].y = (positions[v1y] + positions[v2y] + positions[v3y]) / 3.0;\r\n options.facetPositions[index].z = (positions[v1z] + positions[v2z] + positions[v3z]) / 3.0;\r\n }\r\n\r\n if (computeFacetPartitioning && options) {\r\n // store the facet indexes in arrays in the main facetPartitioning array :\r\n // compute each facet vertex (+ facet barycenter) index in the partiniong array\r\n const ox = Math.floor((options.facetPositions[index].x - options.bInfo.minimum.x * ratio) * xSubRatio);\r\n const oy = Math.floor((options.facetPositions[index].y - options.bInfo.minimum.y * ratio) * ySubRatio);\r\n const oz = Math.floor((options.facetPositions[index].z - options.bInfo.minimum.z * ratio) * zSubRatio);\r\n const b1x = Math.floor((positions[v1x] - options.bInfo.minimum.x * ratio) * xSubRatio);\r\n const b1y = Math.floor((positions[v1y] - options.bInfo.minimum.y * ratio) * ySubRatio);\r\n const b1z = Math.floor((positions[v1z] - options.bInfo.minimum.z * ratio) * zSubRatio);\r\n const b2x = Math.floor((positions[v2x] - options.bInfo.minimum.x * ratio) * xSubRatio);\r\n const b2y = Math.floor((positions[v2y] - options.bInfo.minimum.y * ratio) * ySubRatio);\r\n const b2z = Math.floor((positions[v2z] - options.bInfo.minimum.z * ratio) * zSubRatio);\r\n const b3x = Math.floor((positions[v3x] - options.bInfo.minimum.x * ratio) * xSubRatio);\r\n const b3y = Math.floor((positions[v3y] - options.bInfo.minimum.y * ratio) * ySubRatio);\r\n const b3z = Math.floor((positions[v3z] - options.bInfo.minimum.z * ratio) * zSubRatio);\r\n\r\n const block_idx_v1 = b1x + options.subDiv.max * b1y + subSq * b1z;\r\n const block_idx_v2 = b2x + options.subDiv.max * b2y + subSq * b2z;\r\n const block_idx_v3 = b3x + options.subDiv.max * b3y + subSq * b3z;\r\n const block_idx_o = ox + options.subDiv.max * oy + subSq * oz;\r\n\r\n options.facetPartitioning[block_idx_o] = options.facetPartitioning[block_idx_o] ? options.facetPartitioning[block_idx_o] : new Array();\r\n options.facetPartitioning[block_idx_v1] = options.facetPartitioning[block_idx_v1] ? options.facetPartitioning[block_idx_v1] : new Array();\r\n options.facetPartitioning[block_idx_v2] = options.facetPartitioning[block_idx_v2] ? options.facetPartitioning[block_idx_v2] : new Array();\r\n options.facetPartitioning[block_idx_v3] = options.facetPartitioning[block_idx_v3] ? options.facetPartitioning[block_idx_v3] : new Array();\r\n\r\n // push each facet index in each block containing the vertex\r\n options.facetPartitioning[block_idx_v1].push(index);\r\n if (block_idx_v2 != block_idx_v1) {\r\n options.facetPartitioning[block_idx_v2].push(index);\r\n }\r\n if (!(block_idx_v3 == block_idx_v2 || block_idx_v3 == block_idx_v1)) {\r\n options.facetPartitioning[block_idx_v3].push(index);\r\n }\r\n if (!(block_idx_o == block_idx_v1 || block_idx_o == block_idx_v2 || block_idx_o == block_idx_v3)) {\r\n options.facetPartitioning[block_idx_o].push(index);\r\n }\r\n }\r\n\r\n if (computeDepthSort && options && options.facetPositions) {\r\n const dsf = options.depthSortedFacets[index];\r\n dsf.ind = index * 3;\r\n dsf.sqDistance = Vector3.DistanceSquared(options.facetPositions[index], distanceTo!);\r\n }\r\n\r\n // compute the normals anyway\r\n normals[v1x] += faceNormalx; // accumulate all the normals per face\r\n normals[v1y] += faceNormaly;\r\n normals[v1z] += faceNormalz;\r\n normals[v2x] += faceNormalx;\r\n normals[v2y] += faceNormaly;\r\n normals[v2z] += faceNormalz;\r\n normals[v3x] += faceNormalx;\r\n normals[v3y] += faceNormaly;\r\n normals[v3z] += faceNormalz;\r\n }\r\n // last normalization of each normal\r\n for (index = 0; index < normals.length / 3; index++) {\r\n faceNormalx = normals[index * 3];\r\n faceNormaly = normals[index * 3 + 1];\r\n faceNormalz = normals[index * 3 + 2];\r\n\r\n length = Math.sqrt(faceNormalx * faceNormalx + faceNormaly * faceNormaly + faceNormalz * faceNormalz);\r\n length = length === 0 ? 1.0 : length;\r\n faceNormalx /= length;\r\n faceNormaly /= length;\r\n faceNormalz /= length;\r\n\r\n normals[index * 3] = faceNormalx;\r\n normals[index * 3 + 1] = faceNormaly;\r\n normals[index * 3 + 2] = faceNormalz;\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _ComputeSides(\r\n sideOrientation: number,\r\n positions: FloatArray,\r\n indices: FloatArray | IndicesArray,\r\n normals: FloatArray,\r\n uvs: FloatArray,\r\n frontUVs?: Vector4,\r\n backUVs?: Vector4\r\n ) {\r\n const li: number = indices.length;\r\n const ln: number = normals.length;\r\n let i: number;\r\n let n: number;\r\n sideOrientation = sideOrientation || VertexData.DEFAULTSIDE;\r\n\r\n switch (sideOrientation) {\r\n case VertexData.FRONTSIDE:\r\n // nothing changed\r\n break;\r\n\r\n case VertexData.BACKSIDE:\r\n // indices\r\n for (i = 0; i < li; i += 3) {\r\n const tmp = indices[i];\r\n indices[i] = indices[i + 2];\r\n indices[i + 2] = tmp;\r\n }\r\n // normals\r\n for (n = 0; n < ln; n++) {\r\n normals[n] = -normals[n];\r\n }\r\n break;\r\n\r\n case VertexData.DOUBLESIDE: {\r\n // positions\r\n const lp: number = positions.length;\r\n const l: number = lp / 3;\r\n for (let p = 0; p < lp; p++) {\r\n positions[lp + p] = positions[p];\r\n }\r\n // indices\r\n for (i = 0; i < li; i += 3) {\r\n indices[i + li] = indices[i + 2] + l;\r\n indices[i + 1 + li] = indices[i + 1] + l;\r\n indices[i + 2 + li] = indices[i] + l;\r\n }\r\n // normals\r\n for (n = 0; n < ln; n++) {\r\n normals[ln + n] = -normals[n];\r\n }\r\n\r\n // uvs\r\n const lu: number = uvs.length;\r\n let u: number = 0;\r\n for (u = 0; u < lu; u++) {\r\n uvs[u + lu] = uvs[u];\r\n }\r\n frontUVs = frontUVs ? frontUVs : new Vector4(0.0, 0.0, 1.0, 1.0);\r\n backUVs = backUVs ? backUVs : new Vector4(0.0, 0.0, 1.0, 1.0);\r\n u = 0;\r\n for (i = 0; i < lu / 2; i++) {\r\n uvs[u] = frontUVs.x + (frontUVs.z - frontUVs.x) * uvs[u];\r\n uvs[u + 1] = frontUVs.y + (frontUVs.w - frontUVs.y) * uvs[u + 1];\r\n uvs[u + lu] = backUVs.x + (backUVs.z - backUVs.x) * uvs[u + lu];\r\n uvs[u + lu + 1] = backUVs.y + (backUVs.w - backUVs.y) * uvs[u + lu + 1];\r\n u += 2;\r\n }\r\n break;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Creates a VertexData from serialized data\r\n * @param parsedVertexData the parsed data from an imported file\r\n * @returns a VertexData\r\n */\r\n public static Parse(parsedVertexData: any) {\r\n const vertexData = new VertexData();\r\n\r\n // positions\r\n const positions = parsedVertexData.positions;\r\n if (positions) {\r\n vertexData.set(positions, VertexBuffer.PositionKind);\r\n }\r\n\r\n // normals\r\n const normals = parsedVertexData.normals;\r\n if (normals) {\r\n vertexData.set(normals, VertexBuffer.NormalKind);\r\n }\r\n\r\n // tangents\r\n const tangents = parsedVertexData.tangents;\r\n if (tangents) {\r\n vertexData.set(tangents, VertexBuffer.TangentKind);\r\n }\r\n\r\n // uvs\r\n const uvs = parsedVertexData.uvs;\r\n if (uvs) {\r\n vertexData.set(uvs, VertexBuffer.UVKind);\r\n }\r\n\r\n // uv2s\r\n const uvs2 = parsedVertexData.uvs2;\r\n if (uvs2) {\r\n vertexData.set(uvs2, VertexBuffer.UV2Kind);\r\n }\r\n\r\n // uv3s\r\n const uvs3 = parsedVertexData.uvs3;\r\n if (uvs3) {\r\n vertexData.set(uvs3, VertexBuffer.UV3Kind);\r\n }\r\n\r\n // uv4s\r\n const uvs4 = parsedVertexData.uvs4;\r\n if (uvs4) {\r\n vertexData.set(uvs4, VertexBuffer.UV4Kind);\r\n }\r\n\r\n // uv5s\r\n const uvs5 = parsedVertexData.uvs5;\r\n if (uvs5) {\r\n vertexData.set(uvs5, VertexBuffer.UV5Kind);\r\n }\r\n\r\n // uv6s\r\n const uvs6 = parsedVertexData.uvs6;\r\n if (uvs6) {\r\n vertexData.set(uvs6, VertexBuffer.UV6Kind);\r\n }\r\n\r\n // colors\r\n const colors = parsedVertexData.colors;\r\n if (colors) {\r\n vertexData.set(Color4.CheckColors4(colors, positions.length / 3), VertexBuffer.ColorKind);\r\n }\r\n\r\n // matricesIndices\r\n const matricesIndices = parsedVertexData.matricesIndices;\r\n if (matricesIndices) {\r\n vertexData.set(matricesIndices, VertexBuffer.MatricesIndicesKind);\r\n }\r\n\r\n // matricesWeights\r\n const matricesWeights = parsedVertexData.matricesWeights;\r\n if (matricesWeights) {\r\n vertexData.set(matricesWeights, VertexBuffer.MatricesWeightsKind);\r\n }\r\n\r\n // indices\r\n const indices = parsedVertexData.indices;\r\n if (indices) {\r\n vertexData.indices = indices;\r\n }\r\n\r\n // MaterialInfos\r\n const materialInfos = parsedVertexData.materialInfos;\r\n if (materialInfos) {\r\n vertexData.materialInfos = [];\r\n for (const materialInfoFromJSON of materialInfos) {\r\n const materialInfo = new VertexDataMaterialInfo();\r\n materialInfo.indexCount = materialInfoFromJSON.indexCount;\r\n materialInfo.indexStart = materialInfoFromJSON.indexStart;\r\n materialInfo.verticesCount = materialInfoFromJSON.verticesCount;\r\n materialInfo.verticesStart = materialInfoFromJSON.verticesStart;\r\n materialInfo.materialIndex = materialInfoFromJSON.materialIndex;\r\n vertexData.materialInfos.push(materialInfo);\r\n }\r\n }\r\n\r\n return vertexData;\r\n }\r\n\r\n /**\r\n * Applies VertexData created from the imported parameters to the geometry\r\n * @param parsedVertexData the parsed data from an imported file\r\n * @param geometry the geometry to apply the VertexData to\r\n */\r\n public static ImportVertexData(parsedVertexData: any, geometry: Geometry) {\r\n const vertexData = VertexData.Parse(parsedVertexData);\r\n\r\n geometry.setAllVerticesData(vertexData, parsedVertexData.updatable);\r\n }\r\n}\r\n","import { Constants } from \"../Engines/constants\";\r\n\r\n/**\r\n * Class used to represent data loading progression\r\n */\r\nexport class SceneLoaderFlags {\r\n // Flags\r\n private static _ForceFullSceneLoadingForIncremental = false;\r\n private static _ShowLoadingScreen = true;\r\n private static _CleanBoneMatrixWeights = false;\r\n private static _LoggingLevel = Constants.SCENELOADER_NO_LOGGING;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if entire scene must be loaded even if scene contains incremental data\r\n */\r\n public static get ForceFullSceneLoadingForIncremental() {\r\n return SceneLoaderFlags._ForceFullSceneLoadingForIncremental;\r\n }\r\n\r\n public static set ForceFullSceneLoadingForIncremental(value: boolean) {\r\n SceneLoaderFlags._ForceFullSceneLoadingForIncremental = value;\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating if loading screen must be displayed while loading a scene\r\n */\r\n public static get ShowLoadingScreen(): boolean {\r\n return SceneLoaderFlags._ShowLoadingScreen;\r\n }\r\n\r\n public static set ShowLoadingScreen(value: boolean) {\r\n SceneLoaderFlags._ShowLoadingScreen = value;\r\n }\r\n\r\n /**\r\n * Defines the current logging level (while loading the scene)\r\n * @ignorenaming\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public static get loggingLevel(): number {\r\n return SceneLoaderFlags._LoggingLevel;\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public static set loggingLevel(value: number) {\r\n SceneLoaderFlags._LoggingLevel = value;\r\n }\r\n\r\n /**\r\n * Gets or set a boolean indicating if matrix weights must be cleaned upon loading\r\n */\r\n public static get CleanBoneMatrixWeights(): boolean {\r\n return SceneLoaderFlags._CleanBoneMatrixWeights;\r\n }\r\n\r\n public static set CleanBoneMatrixWeights(value: boolean) {\r\n SceneLoaderFlags._CleanBoneMatrixWeights = value;\r\n }\r\n}\r\n","/**\r\n * Options used to control default behaviors regarding compatibility support\r\n */\r\nexport class CompatibilityOptions {\r\n /**\r\n * Defines if the system should use OpenGL convention for UVs when creating geometry or loading .babylon files (false by default)\r\n */\r\n public static UseOpenGLOrientationForUV = false;\r\n}\r\n","import type { Nullable, FloatArray, DataArray, IndicesArray } from \"../types\";\r\nimport type { Scene } from \"../scene\";\r\nimport type { Vector2 } from \"../Maths/math.vector\";\r\nimport { Vector3 } from \"../Maths/math.vector\";\r\nimport { Color4 } from \"../Maths/math.color\";\r\nimport type { Engine } from \"../Engines/engine\";\r\nimport type { IGetSetVerticesData } from \"../Meshes/mesh.vertexData\";\r\nimport { VertexData } from \"../Meshes/mesh.vertexData\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport { SubMesh } from \"../Meshes/subMesh\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { Effect } from \"../Materials/effect\";\r\nimport { SceneLoaderFlags } from \"../Loading/sceneLoaderFlags\";\r\nimport { BoundingInfo } from \"../Culling/boundingInfo\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport { Tools } from \"../Misc/tools\";\r\nimport { Tags } from \"../Misc/tags\";\r\nimport type { DataBuffer } from \"../Buffers/dataBuffer\";\r\nimport { extractMinAndMax } from \"../Maths/math.functions\";\r\nimport type { AbstractScene } from \"../abstractScene\";\r\nimport { EngineStore } from \"../Engines/engineStore\";\r\nimport { CompatibilityOptions } from \"../Compat/compatibilityOptions\";\r\n\r\nimport type { Mesh } from \"../Meshes/mesh\";\r\n\r\n/**\r\n * Class used to store geometry data (vertex buffers + index buffer)\r\n */\r\nexport class Geometry implements IGetSetVerticesData {\r\n // Members\r\n /**\r\n * Gets or sets the ID of the geometry\r\n */\r\n public id: string;\r\n /**\r\n * Gets or sets the unique ID of the geometry\r\n */\r\n public uniqueId: number;\r\n /**\r\n * Gets the delay loading state of the geometry (none by default which means not delayed)\r\n */\r\n public delayLoadState = Constants.DELAYLOADSTATE_NONE;\r\n /**\r\n * Gets the file containing the data to load when running in delay load state\r\n */\r\n public delayLoadingFile: Nullable;\r\n /**\r\n * Callback called when the geometry is updated\r\n */\r\n public onGeometryUpdated: (geometry: Geometry, kind?: string) => void;\r\n\r\n // Private\r\n private _scene: Scene;\r\n private _engine: Engine;\r\n private _meshes: Mesh[];\r\n private _totalVertices = 0;\r\n /** @internal */\r\n public _loadedUniqueId: string;\r\n /** @internal */\r\n public _indices: IndicesArray;\r\n /** @internal */\r\n public _vertexBuffers: { [key: string]: VertexBuffer };\r\n private _isDisposed = false;\r\n private _extend: { minimum: Vector3; maximum: Vector3 };\r\n private _boundingBias: Vector2;\r\n /** @internal */\r\n public _delayInfo: Array;\r\n private _indexBuffer: Nullable;\r\n private _indexBufferIsUpdatable = false;\r\n /** @internal */\r\n public _boundingInfo: Nullable;\r\n /** @internal */\r\n public _delayLoadingFunction: Nullable<(any: any, geometry: Geometry) => void>;\r\n /** @internal */\r\n public _softwareSkinningFrameId: number;\r\n private _vertexArrayObjects: { [key: string]: WebGLVertexArrayObject };\r\n private _updatable: boolean;\r\n\r\n // Cache\r\n /** @internal */\r\n public _positions: Nullable;\r\n private _positionsCache: Vector3[] = [];\r\n\r\n /** @internal */\r\n public _parentContainer: Nullable = null;\r\n\r\n /**\r\n * Gets or sets the Bias Vector to apply on the bounding elements (box/sphere), the max extend is computed as v += v * bias.x + bias.y, the min is computed as v -= v * bias.x + bias.y\r\n */\r\n public get boundingBias(): Vector2 {\r\n return this._boundingBias;\r\n }\r\n\r\n /**\r\n * Gets or sets the Bias Vector to apply on the bounding elements (box/sphere), the max extend is computed as v += v * bias.x + bias.y, the min is computed as v -= v * bias.x + bias.y\r\n */\r\n public set boundingBias(value: Vector2) {\r\n if (this._boundingBias) {\r\n this._boundingBias.copyFrom(value);\r\n } else {\r\n this._boundingBias = value.clone();\r\n }\r\n\r\n this._updateBoundingInfo(true, null);\r\n }\r\n\r\n /**\r\n * Static function used to attach a new empty geometry to a mesh\r\n * @param mesh defines the mesh to attach the geometry to\r\n * @returns the new Geometry\r\n */\r\n public static CreateGeometryForMesh(mesh: Mesh): Geometry {\r\n const geometry = new Geometry(Geometry.RandomId(), mesh.getScene());\r\n\r\n geometry.applyToMesh(mesh);\r\n\r\n return geometry;\r\n }\r\n\r\n /** Get the list of meshes using this geometry */\r\n public get meshes(): Mesh[] {\r\n return this._meshes;\r\n }\r\n\r\n /**\r\n * If set to true (false by default), the bounding info applied to the meshes sharing this geometry will be the bounding info defined at the class level\r\n * and won't be computed based on the vertex positions (which is what we get when useBoundingInfoFromGeometry = false)\r\n */\r\n public useBoundingInfoFromGeometry = false;\r\n\r\n /**\r\n * Creates a new geometry\r\n * @param id defines the unique ID\r\n * @param scene defines the hosting scene\r\n * @param vertexData defines the VertexData used to get geometry data\r\n * @param updatable defines if geometry must be updatable (false by default)\r\n * @param mesh defines the mesh that will be associated with the geometry\r\n */\r\n constructor(id: string, scene?: Scene, vertexData?: VertexData, updatable: boolean = false, mesh: Nullable = null) {\r\n this._scene = scene || EngineStore.LastCreatedScene;\r\n if (!this._scene) {\r\n return;\r\n }\r\n this.id = id;\r\n this.uniqueId = this._scene.getUniqueId();\r\n this._engine = this._scene.getEngine();\r\n this._meshes = [];\r\n //Init vertex buffer cache\r\n this._vertexBuffers = {};\r\n this._indices = [];\r\n this._updatable = updatable;\r\n\r\n // vertexData\r\n if (vertexData) {\r\n this.setAllVerticesData(vertexData, updatable);\r\n } else {\r\n this._totalVertices = 0;\r\n }\r\n\r\n if (this._engine.getCaps().vertexArrayObject) {\r\n this._vertexArrayObjects = {};\r\n }\r\n\r\n // applyToMesh\r\n if (mesh) {\r\n this.applyToMesh(mesh);\r\n mesh.computeWorldMatrix(true);\r\n }\r\n }\r\n\r\n /**\r\n * Gets the current extend of the geometry\r\n */\r\n public get extend(): { minimum: Vector3; maximum: Vector3 } {\r\n return this._extend;\r\n }\r\n\r\n /**\r\n * Gets the hosting scene\r\n * @returns the hosting Scene\r\n */\r\n public getScene(): Scene {\r\n return this._scene;\r\n }\r\n\r\n /**\r\n * Gets the hosting engine\r\n * @returns the hosting Engine\r\n */\r\n public getEngine(): Engine {\r\n return this._engine;\r\n }\r\n\r\n /**\r\n * Defines if the geometry is ready to use\r\n * @returns true if the geometry is ready to be used\r\n */\r\n public isReady(): boolean {\r\n return this.delayLoadState === Constants.DELAYLOADSTATE_LOADED || this.delayLoadState === Constants.DELAYLOADSTATE_NONE;\r\n }\r\n\r\n /**\r\n * Gets a value indicating that the geometry should not be serialized\r\n */\r\n public get doNotSerialize(): boolean {\r\n for (let index = 0; index < this._meshes.length; index++) {\r\n if (!this._meshes[index].doNotSerialize) {\r\n return false;\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /** @internal */\r\n public _rebuild(): void {\r\n if (this._vertexArrayObjects) {\r\n this._vertexArrayObjects = {};\r\n }\r\n\r\n // Index buffer\r\n if (this._meshes.length !== 0 && this._indices) {\r\n this._indexBuffer = this._engine.createIndexBuffer(this._indices, this._updatable);\r\n }\r\n\r\n // Vertex buffers\r\n for (const key in this._vertexBuffers) {\r\n const vertexBuffer = this._vertexBuffers[key];\r\n vertexBuffer._rebuild();\r\n }\r\n }\r\n\r\n /**\r\n * Affects all geometry data in one call\r\n * @param vertexData defines the geometry data\r\n * @param updatable defines if the geometry must be flagged as updatable (false as default)\r\n */\r\n public setAllVerticesData(vertexData: VertexData, updatable?: boolean): void {\r\n vertexData.applyToGeometry(this, updatable);\r\n this._notifyUpdate();\r\n }\r\n\r\n /**\r\n * Set specific vertex data\r\n * @param kind defines the data kind (Position, normal, etc...)\r\n * @param data defines the vertex data to use\r\n * @param updatable defines if the vertex must be flagged as updatable (false as default)\r\n * @param stride defines the stride to use (0 by default). This value is deduced from the kind value if not specified\r\n */\r\n public setVerticesData(kind: string, data: FloatArray, updatable: boolean = false, stride?: number): void {\r\n if (updatable && Array.isArray(data)) {\r\n // to avoid converting to Float32Array at each draw call in engine.updateDynamicVertexBuffer, we make the conversion a single time here\r\n data = new Float32Array(data);\r\n }\r\n const buffer = new VertexBuffer(this._engine, data, kind, updatable, this._meshes.length === 0, stride);\r\n this.setVerticesBuffer(buffer);\r\n }\r\n\r\n /**\r\n * Removes a specific vertex data\r\n * @param kind defines the data kind (Position, normal, etc...)\r\n */\r\n public removeVerticesData(kind: string) {\r\n if (this._vertexBuffers[kind]) {\r\n this._vertexBuffers[kind].dispose();\r\n delete this._vertexBuffers[kind];\r\n }\r\n\r\n if (this._vertexArrayObjects) {\r\n this._disposeVertexArrayObjects();\r\n }\r\n }\r\n\r\n /**\r\n * Affect a vertex buffer to the geometry. the vertexBuffer.getKind() function is used to determine where to store the data\r\n * @param buffer defines the vertex buffer to use\r\n * @param totalVertices defines the total number of vertices for position kind (could be null)\r\n * @param disposeExistingBuffer disposes the existing buffer, if any (default: true)\r\n */\r\n public setVerticesBuffer(buffer: VertexBuffer, totalVertices: Nullable = null, disposeExistingBuffer = true): void {\r\n const kind = buffer.getKind();\r\n if (this._vertexBuffers[kind] && disposeExistingBuffer) {\r\n this._vertexBuffers[kind].dispose();\r\n }\r\n\r\n if (buffer._buffer) {\r\n buffer._buffer._increaseReferences();\r\n }\r\n\r\n this._vertexBuffers[kind] = buffer;\r\n const meshes = this._meshes;\r\n const numOfMeshes = meshes.length;\r\n\r\n if (kind === VertexBuffer.PositionKind) {\r\n const data = buffer.getData();\r\n if (totalVertices != null) {\r\n this._totalVertices = totalVertices;\r\n } else {\r\n if (data != null) {\r\n this._totalVertices = data.length / (buffer.type === VertexBuffer.BYTE ? buffer.byteStride : buffer.byteStride / 4);\r\n }\r\n }\r\n\r\n this._updateExtend(data);\r\n this._resetPointsArrayCache();\r\n\r\n for (let index = 0; index < numOfMeshes; index++) {\r\n const mesh = meshes[index];\r\n mesh.buildBoundingInfo(this._extend.minimum, this._extend.maximum);\r\n mesh._createGlobalSubMesh(mesh.isUnIndexed);\r\n mesh.computeWorldMatrix(true);\r\n mesh.synchronizeInstances();\r\n }\r\n }\r\n\r\n this._notifyUpdate(kind);\r\n }\r\n\r\n /**\r\n * Update a specific vertex buffer\r\n * This function will directly update the underlying DataBuffer according to the passed numeric array or Float32Array\r\n * It will do nothing if the buffer is not updatable\r\n * @param kind defines the data kind (Position, normal, etc...)\r\n * @param data defines the data to use\r\n * @param offset defines the offset in the target buffer where to store the data\r\n * @param useBytes set to true if the offset is in bytes\r\n */\r\n public updateVerticesDataDirectly(kind: string, data: DataArray, offset: number, useBytes: boolean = false): void {\r\n const vertexBuffer = this.getVertexBuffer(kind);\r\n\r\n if (!vertexBuffer) {\r\n return;\r\n }\r\n\r\n vertexBuffer.updateDirectly(data, offset, useBytes);\r\n this._notifyUpdate(kind);\r\n }\r\n\r\n /**\r\n * Update a specific vertex buffer\r\n * This function will create a new buffer if the current one is not updatable\r\n * @param kind defines the data kind (Position, normal, etc...)\r\n * @param data defines the data to use\r\n * @param updateExtends defines if the geometry extends must be recomputed (false by default)\r\n */\r\n public updateVerticesData(kind: string, data: FloatArray, updateExtends: boolean = false): void {\r\n const vertexBuffer = this.getVertexBuffer(kind);\r\n\r\n if (!vertexBuffer) {\r\n return;\r\n }\r\n\r\n vertexBuffer.update(data);\r\n\r\n if (kind === VertexBuffer.PositionKind) {\r\n this._updateBoundingInfo(updateExtends, data);\r\n }\r\n this._notifyUpdate(kind);\r\n }\r\n\r\n private _updateBoundingInfo(updateExtends: boolean, data: Nullable) {\r\n if (updateExtends) {\r\n this._updateExtend(data);\r\n }\r\n\r\n this._resetPointsArrayCache();\r\n\r\n if (updateExtends) {\r\n const meshes = this._meshes;\r\n for (const mesh of meshes) {\r\n if (mesh.hasBoundingInfo) {\r\n mesh.getBoundingInfo().reConstruct(this._extend.minimum, this._extend.maximum);\r\n } else {\r\n mesh.buildBoundingInfo(this._extend.minimum, this._extend.maximum);\r\n }\r\n\r\n const subMeshes = mesh.subMeshes;\r\n for (const subMesh of subMeshes) {\r\n subMesh.refreshBoundingInfo();\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _bind(\r\n effect: Nullable,\r\n indexToBind?: Nullable,\r\n overrideVertexBuffers?: { [kind: string]: Nullable },\r\n overrideVertexArrayObjects?: { [key: string]: WebGLVertexArrayObject }\r\n ): void {\r\n if (!effect) {\r\n return;\r\n }\r\n\r\n if (indexToBind === undefined) {\r\n indexToBind = this._indexBuffer;\r\n }\r\n const vbs = this.getVertexBuffers();\r\n\r\n if (!vbs) {\r\n return;\r\n }\r\n\r\n if (indexToBind != this._indexBuffer || (!this._vertexArrayObjects && !overrideVertexArrayObjects)) {\r\n this._engine.bindBuffers(vbs, indexToBind, effect, overrideVertexBuffers);\r\n return;\r\n }\r\n\r\n const vaos = overrideVertexArrayObjects ? overrideVertexArrayObjects : this._vertexArrayObjects;\r\n\r\n // Using VAO\r\n if (!vaos[effect.key]) {\r\n vaos[effect.key] = this._engine.recordVertexArrayObject(vbs, indexToBind, effect, overrideVertexBuffers);\r\n }\r\n\r\n this._engine.bindVertexArrayObject(vaos[effect.key], indexToBind);\r\n }\r\n\r\n /**\r\n * Gets total number of vertices\r\n * @returns the total number of vertices\r\n */\r\n public getTotalVertices(): number {\r\n if (!this.isReady()) {\r\n return 0;\r\n }\r\n\r\n return this._totalVertices;\r\n }\r\n\r\n /**\r\n * Gets a specific vertex data attached to this geometry. Float data is constructed if the vertex buffer data cannot be returned directly.\r\n * @param kind defines the data kind (Position, normal, etc...)\r\n * @param copyWhenShared defines if the returned array must be cloned upon returning it if the current geometry is shared between multiple meshes\r\n * @param forceCopy defines a boolean indicating that the returned array must be cloned upon returning it\r\n * @returns a float array containing vertex data\r\n */\r\n public getVerticesData(kind: string, copyWhenShared?: boolean, forceCopy?: boolean): Nullable {\r\n const vertexBuffer = this.getVertexBuffer(kind);\r\n if (!vertexBuffer) {\r\n return null;\r\n }\r\n\r\n return vertexBuffer.getFloatData(this._totalVertices, forceCopy || (copyWhenShared && this._meshes.length !== 1));\r\n }\r\n\r\n /**\r\n * Returns a boolean defining if the vertex data for the requested `kind` is updatable\r\n * @param kind defines the data kind (Position, normal, etc...)\r\n * @returns true if the vertex buffer with the specified kind is updatable\r\n */\r\n public isVertexBufferUpdatable(kind: string): boolean {\r\n const vb = this._vertexBuffers[kind];\r\n\r\n if (!vb) {\r\n return false;\r\n }\r\n\r\n return vb.isUpdatable();\r\n }\r\n\r\n /**\r\n * Gets a specific vertex buffer\r\n * @param kind defines the data kind (Position, normal, etc...)\r\n * @returns a VertexBuffer\r\n */\r\n public getVertexBuffer(kind: string): Nullable {\r\n if (!this.isReady()) {\r\n return null;\r\n }\r\n return this._vertexBuffers[kind];\r\n }\r\n\r\n /**\r\n * Returns all vertex buffers\r\n * @returns an object holding all vertex buffers indexed by kind\r\n */\r\n public getVertexBuffers(): Nullable<{ [key: string]: VertexBuffer }> {\r\n if (!this.isReady()) {\r\n return null;\r\n }\r\n return this._vertexBuffers;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if specific vertex buffer is present\r\n * @param kind defines the data kind (Position, normal, etc...)\r\n * @returns true if data is present\r\n */\r\n public isVerticesDataPresent(kind: string): boolean {\r\n if (!this._vertexBuffers) {\r\n if (this._delayInfo) {\r\n return this._delayInfo.indexOf(kind) !== -1;\r\n }\r\n return false;\r\n }\r\n return this._vertexBuffers[kind] !== undefined;\r\n }\r\n\r\n /**\r\n * Gets a list of all attached data kinds (Position, normal, etc...)\r\n * @returns a list of string containing all kinds\r\n */\r\n public getVerticesDataKinds(): string[] {\r\n const result = [];\r\n let kind;\r\n if (!this._vertexBuffers && this._delayInfo) {\r\n for (kind in this._delayInfo) {\r\n result.push(kind);\r\n }\r\n } else {\r\n for (kind in this._vertexBuffers) {\r\n result.push(kind);\r\n }\r\n }\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Update index buffer\r\n * @param indices defines the indices to store in the index buffer\r\n * @param offset defines the offset in the target buffer where to store the data\r\n * @param gpuMemoryOnly defines a boolean indicating that only the GPU memory must be updated leaving the CPU version of the indices unchanged (false by default)\r\n */\r\n public updateIndices(indices: IndicesArray, offset?: number, gpuMemoryOnly = false): void {\r\n if (!this._indexBuffer) {\r\n return;\r\n }\r\n\r\n if (!this._indexBufferIsUpdatable) {\r\n this.setIndices(indices, null, true);\r\n } else {\r\n const needToUpdateSubMeshes = indices.length !== this._indices.length;\r\n\r\n if (!gpuMemoryOnly) {\r\n this._indices = indices.slice();\r\n }\r\n this._engine.updateDynamicIndexBuffer(this._indexBuffer, indices, offset);\r\n if (needToUpdateSubMeshes) {\r\n for (const mesh of this._meshes) {\r\n mesh._createGlobalSubMesh(true);\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Creates a new index buffer\r\n * @param indices defines the indices to store in the index buffer\r\n * @param totalVertices defines the total number of vertices (could be null)\r\n * @param updatable defines if the index buffer must be flagged as updatable (false by default)\r\n */\r\n public setIndices(indices: IndicesArray, totalVertices: Nullable = null, updatable: boolean = false): void {\r\n if (this._indexBuffer) {\r\n this._engine._releaseBuffer(this._indexBuffer);\r\n }\r\n\r\n this._indices = indices;\r\n this._indexBufferIsUpdatable = updatable;\r\n if (this._meshes.length !== 0 && this._indices) {\r\n this._indexBuffer = this._engine.createIndexBuffer(this._indices, updatable);\r\n }\r\n\r\n if (totalVertices != undefined) {\r\n // including null and undefined\r\n this._totalVertices = totalVertices;\r\n }\r\n\r\n for (const mesh of this._meshes) {\r\n mesh._createGlobalSubMesh(true);\r\n mesh.synchronizeInstances();\r\n }\r\n\r\n this._notifyUpdate();\r\n }\r\n\r\n /**\r\n * Return the total number of indices\r\n * @returns the total number of indices\r\n */\r\n public getTotalIndices(): number {\r\n if (!this.isReady()) {\r\n return 0;\r\n }\r\n return this._indices.length;\r\n }\r\n\r\n /**\r\n * Gets the index buffer array\r\n * @param copyWhenShared defines if the returned array must be cloned upon returning it if the current geometry is shared between multiple meshes\r\n * @param forceCopy defines a boolean indicating that the returned array must be cloned upon returning it\r\n * @returns the index buffer array\r\n */\r\n public getIndices(copyWhenShared?: boolean, forceCopy?: boolean): Nullable {\r\n if (!this.isReady()) {\r\n return null;\r\n }\r\n const orig = this._indices;\r\n if (!forceCopy && (!copyWhenShared || this._meshes.length === 1)) {\r\n return orig;\r\n } else {\r\n return orig.slice();\r\n }\r\n }\r\n\r\n /**\r\n * Gets the index buffer\r\n * @returns the index buffer\r\n */\r\n public getIndexBuffer(): Nullable {\r\n if (!this.isReady()) {\r\n return null;\r\n }\r\n return this._indexBuffer;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _releaseVertexArrayObject(effect: Nullable = null) {\r\n if (!effect || !this._vertexArrayObjects) {\r\n return;\r\n }\r\n\r\n if (this._vertexArrayObjects[effect.key]) {\r\n this._engine.releaseVertexArrayObject(this._vertexArrayObjects[effect.key]);\r\n delete this._vertexArrayObjects[effect.key];\r\n }\r\n }\r\n\r\n /**\r\n * Release the associated resources for a specific mesh\r\n * @param mesh defines the source mesh\r\n * @param shouldDispose defines if the geometry must be disposed if there is no more mesh pointing to it\r\n */\r\n public releaseForMesh(mesh: Mesh, shouldDispose?: boolean): void {\r\n const meshes = this._meshes;\r\n const index = meshes.indexOf(mesh);\r\n\r\n if (index === -1) {\r\n return;\r\n }\r\n\r\n meshes.splice(index, 1);\r\n\r\n if (this._vertexArrayObjects) {\r\n mesh._invalidateInstanceVertexArrayObject();\r\n }\r\n\r\n mesh._geometry = null;\r\n\r\n if (meshes.length === 0 && shouldDispose) {\r\n this.dispose();\r\n }\r\n }\r\n\r\n /**\r\n * Apply current geometry to a given mesh\r\n * @param mesh defines the mesh to apply geometry to\r\n */\r\n public applyToMesh(mesh: Mesh): void {\r\n if (mesh._geometry === this) {\r\n return;\r\n }\r\n\r\n const previousGeometry = mesh._geometry;\r\n if (previousGeometry) {\r\n previousGeometry.releaseForMesh(mesh);\r\n }\r\n\r\n if (this._vertexArrayObjects) {\r\n mesh._invalidateInstanceVertexArrayObject();\r\n }\r\n\r\n const meshes = this._meshes;\r\n\r\n // must be done before setting vertexBuffers because of mesh._createGlobalSubMesh()\r\n mesh._geometry = this;\r\n mesh._internalAbstractMeshDataInfo._positions = null;\r\n\r\n this._scene.pushGeometry(this);\r\n\r\n meshes.push(mesh);\r\n\r\n if (this.isReady()) {\r\n this._applyToMesh(mesh);\r\n } else if (this._boundingInfo) {\r\n mesh.setBoundingInfo(this._boundingInfo);\r\n }\r\n }\r\n\r\n private _updateExtend(data: Nullable = null) {\r\n if (this.useBoundingInfoFromGeometry && this._boundingInfo) {\r\n this._extend = {\r\n minimum: this._boundingInfo.minimum.clone(),\r\n maximum: this._boundingInfo.maximum.clone(),\r\n };\r\n } else {\r\n if (!data) {\r\n data = this.getVerticesData(VertexBuffer.PositionKind)!;\r\n // This can happen if the buffer comes from a Hardware Buffer where\r\n // The data have not been uploaded by Babylon. (ex: Compute Shaders and Storage Buffers)\r\n if (!data) {\r\n return;\r\n }\r\n }\r\n\r\n this._extend = extractMinAndMax(data, 0, this._totalVertices, this.boundingBias, 3);\r\n }\r\n }\r\n\r\n private _applyToMesh(mesh: Mesh): void {\r\n const numOfMeshes = this._meshes.length;\r\n\r\n // vertexBuffers\r\n for (const kind in this._vertexBuffers) {\r\n if (numOfMeshes === 1) {\r\n this._vertexBuffers[kind].create();\r\n }\r\n\r\n if (kind === VertexBuffer.PositionKind) {\r\n if (!this._extend) {\r\n this._updateExtend();\r\n }\r\n mesh.buildBoundingInfo(this._extend.minimum, this._extend.maximum);\r\n\r\n mesh._createGlobalSubMesh(mesh.isUnIndexed);\r\n\r\n //bounding info was just created again, world matrix should be applied again.\r\n mesh._updateBoundingInfo();\r\n }\r\n }\r\n\r\n // indexBuffer\r\n if (numOfMeshes === 1 && this._indices && this._indices.length > 0) {\r\n this._indexBuffer = this._engine.createIndexBuffer(this._indices, this._updatable);\r\n }\r\n\r\n // morphTargets\r\n mesh._syncGeometryWithMorphTargetManager();\r\n\r\n // instances\r\n mesh.synchronizeInstances();\r\n }\r\n\r\n private _notifyUpdate(kind?: string) {\r\n if (this.onGeometryUpdated) {\r\n this.onGeometryUpdated(this, kind);\r\n }\r\n\r\n if (this._vertexArrayObjects) {\r\n this._disposeVertexArrayObjects();\r\n }\r\n\r\n for (const mesh of this._meshes) {\r\n mesh._markSubMeshesAsAttributesDirty();\r\n }\r\n }\r\n\r\n /**\r\n * Load the geometry if it was flagged as delay loaded\r\n * @param scene defines the hosting scene\r\n * @param onLoaded defines a callback called when the geometry is loaded\r\n */\r\n public load(scene: Scene, onLoaded?: () => void): void {\r\n if (this.delayLoadState === Constants.DELAYLOADSTATE_LOADING) {\r\n return;\r\n }\r\n\r\n if (this.isReady()) {\r\n if (onLoaded) {\r\n onLoaded();\r\n }\r\n return;\r\n }\r\n\r\n this.delayLoadState = Constants.DELAYLOADSTATE_LOADING;\r\n\r\n this._queueLoad(scene, onLoaded);\r\n }\r\n\r\n private _queueLoad(scene: Scene, onLoaded?: () => void): void {\r\n if (!this.delayLoadingFile) {\r\n return;\r\n }\r\n\r\n scene.addPendingData(this);\r\n scene._loadFile(\r\n this.delayLoadingFile,\r\n (data) => {\r\n if (!this._delayLoadingFunction) {\r\n return;\r\n }\r\n\r\n this._delayLoadingFunction(JSON.parse(data as string), this);\r\n\r\n this.delayLoadState = Constants.DELAYLOADSTATE_LOADED;\r\n this._delayInfo = [];\r\n\r\n scene.removePendingData(this);\r\n\r\n const meshes = this._meshes;\r\n const numOfMeshes = meshes.length;\r\n for (let index = 0; index < numOfMeshes; index++) {\r\n this._applyToMesh(meshes[index]);\r\n }\r\n\r\n if (onLoaded) {\r\n onLoaded();\r\n }\r\n },\r\n undefined,\r\n true\r\n );\r\n }\r\n\r\n /**\r\n * Invert the geometry to move from a right handed system to a left handed one.\r\n */\r\n public toLeftHanded(): void {\r\n // Flip faces\r\n const tIndices = this.getIndices(false);\r\n if (tIndices != null && tIndices.length > 0) {\r\n for (let i = 0; i < tIndices.length; i += 3) {\r\n const tTemp = tIndices[i + 0];\r\n tIndices[i + 0] = tIndices[i + 2];\r\n tIndices[i + 2] = tTemp;\r\n }\r\n this.setIndices(tIndices);\r\n }\r\n\r\n // Negate position.z\r\n const tPositions = this.getVerticesData(VertexBuffer.PositionKind, false);\r\n if (tPositions != null && tPositions.length > 0) {\r\n for (let i = 0; i < tPositions.length; i += 3) {\r\n tPositions[i + 2] = -tPositions[i + 2];\r\n }\r\n this.setVerticesData(VertexBuffer.PositionKind, tPositions, false);\r\n }\r\n\r\n // Negate normal.z\r\n const tNormals = this.getVerticesData(VertexBuffer.NormalKind, false);\r\n if (tNormals != null && tNormals.length > 0) {\r\n for (let i = 0; i < tNormals.length; i += 3) {\r\n tNormals[i + 2] = -tNormals[i + 2];\r\n }\r\n this.setVerticesData(VertexBuffer.NormalKind, tNormals, false);\r\n }\r\n }\r\n\r\n // Cache\r\n /** @internal */\r\n public _resetPointsArrayCache(): void {\r\n this._positions = null;\r\n }\r\n\r\n /** @internal */\r\n public _generatePointsArray(): boolean {\r\n if (this._positions) {\r\n return true;\r\n }\r\n\r\n const data = this.getVerticesData(VertexBuffer.PositionKind);\r\n\r\n if (!data || data.length === 0) {\r\n return false;\r\n }\r\n\r\n for (let index = this._positionsCache.length * 3, arrayIdx = this._positionsCache.length; index < data.length; index += 3, ++arrayIdx) {\r\n this._positionsCache[arrayIdx] = Vector3.FromArray(data, index);\r\n }\r\n\r\n for (let index = 0, arrayIdx = 0; index < data.length; index += 3, ++arrayIdx) {\r\n this._positionsCache[arrayIdx].set(data[0 + index], data[1 + index], data[2 + index]);\r\n }\r\n\r\n // just in case the number of positions was reduced, splice the array\r\n this._positionsCache.length = data.length / 3;\r\n\r\n this._positions = this._positionsCache;\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Gets a value indicating if the geometry is disposed\r\n * @returns true if the geometry was disposed\r\n */\r\n public isDisposed(): boolean {\r\n return this._isDisposed;\r\n }\r\n\r\n private _disposeVertexArrayObjects(): void {\r\n if (this._vertexArrayObjects) {\r\n for (const kind in this._vertexArrayObjects) {\r\n this._engine.releaseVertexArrayObject(this._vertexArrayObjects[kind]);\r\n }\r\n this._vertexArrayObjects = {}; // Will trigger a rebuild of the VAO if supported\r\n\r\n const meshes = this._meshes;\r\n const numOfMeshes = meshes.length;\r\n for (let index = 0; index < numOfMeshes; index++) {\r\n meshes[index]._invalidateInstanceVertexArrayObject();\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Free all associated resources\r\n */\r\n public dispose(): void {\r\n const meshes = this._meshes;\r\n const numOfMeshes = meshes.length;\r\n let index: number;\r\n for (index = 0; index < numOfMeshes; index++) {\r\n this.releaseForMesh(meshes[index]);\r\n }\r\n this._meshes.length = 0;\r\n\r\n this._disposeVertexArrayObjects();\r\n\r\n for (const kind in this._vertexBuffers) {\r\n this._vertexBuffers[kind].dispose();\r\n }\r\n this._vertexBuffers = {};\r\n this._totalVertices = 0;\r\n\r\n if (this._indexBuffer) {\r\n this._engine._releaseBuffer(this._indexBuffer);\r\n }\r\n this._indexBuffer = null;\r\n this._indices = [];\r\n\r\n this.delayLoadState = Constants.DELAYLOADSTATE_NONE;\r\n this.delayLoadingFile = null;\r\n this._delayLoadingFunction = null;\r\n this._delayInfo = [];\r\n\r\n this._boundingInfo = null;\r\n\r\n this._scene.removeGeometry(this);\r\n if (this._parentContainer) {\r\n const index = this._parentContainer.geometries.indexOf(this);\r\n if (index > -1) {\r\n this._parentContainer.geometries.splice(index, 1);\r\n }\r\n this._parentContainer = null;\r\n }\r\n\r\n this._isDisposed = true;\r\n }\r\n\r\n /**\r\n * Clone the current geometry into a new geometry\r\n * @param id defines the unique ID of the new geometry\r\n * @returns a new geometry object\r\n */\r\n public copy(id: string): Geometry {\r\n const vertexData = new VertexData();\r\n\r\n vertexData.indices = [];\r\n\r\n const indices = this.getIndices();\r\n if (indices) {\r\n for (let index = 0; index < indices.length; index++) {\r\n (vertexData.indices).push(indices[index]);\r\n }\r\n }\r\n\r\n let updatable = false;\r\n let stopChecking = false;\r\n let kind;\r\n for (kind in this._vertexBuffers) {\r\n // using slice() to make a copy of the array and not just reference it\r\n const data = this.getVerticesData(kind);\r\n\r\n if (data) {\r\n if (data instanceof Float32Array) {\r\n vertexData.set(new Float32Array(data), kind);\r\n } else {\r\n vertexData.set((data).slice(0), kind);\r\n }\r\n if (!stopChecking) {\r\n const vb = this.getVertexBuffer(kind);\r\n\r\n if (vb) {\r\n updatable = vb.isUpdatable();\r\n stopChecking = !updatable;\r\n }\r\n }\r\n }\r\n }\r\n\r\n const geometry = new Geometry(id, this._scene, vertexData, updatable);\r\n\r\n geometry.delayLoadState = this.delayLoadState;\r\n geometry.delayLoadingFile = this.delayLoadingFile;\r\n geometry._delayLoadingFunction = this._delayLoadingFunction;\r\n\r\n for (kind in this._delayInfo) {\r\n geometry._delayInfo = geometry._delayInfo || [];\r\n geometry._delayInfo.push(kind);\r\n }\r\n\r\n // Bounding info\r\n geometry._boundingInfo = new BoundingInfo(this._extend.minimum, this._extend.maximum);\r\n\r\n return geometry;\r\n }\r\n\r\n /**\r\n * Serialize the current geometry info (and not the vertices data) into a JSON object\r\n * @returns a JSON representation of the current geometry data (without the vertices data)\r\n */\r\n public serialize(): any {\r\n const serializationObject: any = {};\r\n\r\n serializationObject.id = this.id;\r\n serializationObject.uniqueId = this.uniqueId;\r\n serializationObject.updatable = this._updatable;\r\n\r\n if (Tags && Tags.HasTags(this)) {\r\n serializationObject.tags = Tags.GetTags(this);\r\n }\r\n\r\n return serializationObject;\r\n }\r\n\r\n private _toNumberArray(origin: Nullable): number[] {\r\n if (Array.isArray(origin)) {\r\n return origin;\r\n } else {\r\n return Array.prototype.slice.call(origin);\r\n }\r\n }\r\n\r\n /**\r\n * Release any memory retained by the cached data on the Geometry.\r\n *\r\n * Call this function to reduce memory footprint of the mesh.\r\n * Vertex buffers will not store CPU data anymore (this will prevent picking, collisions or physics to work correctly)\r\n */\r\n public clearCachedData(): void {\r\n this._indices = [];\r\n this._resetPointsArrayCache();\r\n\r\n for (const vbName in this._vertexBuffers) {\r\n if (!Object.prototype.hasOwnProperty.call(this._vertexBuffers, vbName)) {\r\n continue;\r\n }\r\n this._vertexBuffers[vbName]._buffer._data = null;\r\n }\r\n }\r\n\r\n /**\r\n * Serialize all vertices data into a JSON object\r\n * @returns a JSON representation of the current geometry data\r\n */\r\n public serializeVerticeData(): any {\r\n const serializationObject = this.serialize();\r\n\r\n if (this.isVerticesDataPresent(VertexBuffer.PositionKind)) {\r\n serializationObject.positions = this._toNumberArray(this.getVerticesData(VertexBuffer.PositionKind));\r\n if (this.isVertexBufferUpdatable(VertexBuffer.PositionKind)) {\r\n serializationObject.positions._updatable = true;\r\n }\r\n }\r\n\r\n if (this.isVerticesDataPresent(VertexBuffer.NormalKind)) {\r\n serializationObject.normals = this._toNumberArray(this.getVerticesData(VertexBuffer.NormalKind));\r\n if (this.isVertexBufferUpdatable(VertexBuffer.NormalKind)) {\r\n serializationObject.normals._updatable = true;\r\n }\r\n }\r\n\r\n if (this.isVerticesDataPresent(VertexBuffer.TangentKind)) {\r\n serializationObject.tangents = this._toNumberArray(this.getVerticesData(VertexBuffer.TangentKind));\r\n if (this.isVertexBufferUpdatable(VertexBuffer.TangentKind)) {\r\n serializationObject.tangents._updatable = true;\r\n }\r\n }\r\n\r\n if (this.isVerticesDataPresent(VertexBuffer.UVKind)) {\r\n serializationObject.uvs = this._toNumberArray(this.getVerticesData(VertexBuffer.UVKind));\r\n if (this.isVertexBufferUpdatable(VertexBuffer.UVKind)) {\r\n serializationObject.uvs._updatable = true;\r\n }\r\n }\r\n\r\n if (this.isVerticesDataPresent(VertexBuffer.UV2Kind)) {\r\n serializationObject.uv2s = this._toNumberArray(this.getVerticesData(VertexBuffer.UV2Kind));\r\n if (this.isVertexBufferUpdatable(VertexBuffer.UV2Kind)) {\r\n serializationObject.uv2s._updatable = true;\r\n }\r\n }\r\n\r\n if (this.isVerticesDataPresent(VertexBuffer.UV3Kind)) {\r\n serializationObject.uv3s = this._toNumberArray(this.getVerticesData(VertexBuffer.UV3Kind));\r\n if (this.isVertexBufferUpdatable(VertexBuffer.UV3Kind)) {\r\n serializationObject.uv3s._updatable = true;\r\n }\r\n }\r\n\r\n if (this.isVerticesDataPresent(VertexBuffer.UV4Kind)) {\r\n serializationObject.uv4s = this._toNumberArray(this.getVerticesData(VertexBuffer.UV4Kind));\r\n if (this.isVertexBufferUpdatable(VertexBuffer.UV4Kind)) {\r\n serializationObject.uv4s._updatable = true;\r\n }\r\n }\r\n\r\n if (this.isVerticesDataPresent(VertexBuffer.UV5Kind)) {\r\n serializationObject.uv5s = this._toNumberArray(this.getVerticesData(VertexBuffer.UV5Kind));\r\n if (this.isVertexBufferUpdatable(VertexBuffer.UV5Kind)) {\r\n serializationObject.uv5s._updatable = true;\r\n }\r\n }\r\n\r\n if (this.isVerticesDataPresent(VertexBuffer.UV6Kind)) {\r\n serializationObject.uv6s = this._toNumberArray(this.getVerticesData(VertexBuffer.UV6Kind));\r\n if (this.isVertexBufferUpdatable(VertexBuffer.UV6Kind)) {\r\n serializationObject.uv6s._updatable = true;\r\n }\r\n }\r\n\r\n if (this.isVerticesDataPresent(VertexBuffer.ColorKind)) {\r\n serializationObject.colors = this._toNumberArray(this.getVerticesData(VertexBuffer.ColorKind));\r\n if (this.isVertexBufferUpdatable(VertexBuffer.ColorKind)) {\r\n serializationObject.colors._updatable = true;\r\n }\r\n }\r\n\r\n if (this.isVerticesDataPresent(VertexBuffer.MatricesIndicesKind)) {\r\n serializationObject.matricesIndices = this._toNumberArray(this.getVerticesData(VertexBuffer.MatricesIndicesKind));\r\n serializationObject.matricesIndices._isExpanded = true;\r\n if (this.isVertexBufferUpdatable(VertexBuffer.MatricesIndicesKind)) {\r\n serializationObject.matricesIndices._updatable = true;\r\n }\r\n }\r\n\r\n if (this.isVerticesDataPresent(VertexBuffer.MatricesWeightsKind)) {\r\n serializationObject.matricesWeights = this._toNumberArray(this.getVerticesData(VertexBuffer.MatricesWeightsKind));\r\n if (this.isVertexBufferUpdatable(VertexBuffer.MatricesWeightsKind)) {\r\n serializationObject.matricesWeights._updatable = true;\r\n }\r\n }\r\n\r\n serializationObject.indices = this._toNumberArray(this.getIndices());\r\n\r\n return serializationObject;\r\n }\r\n\r\n // Statics\r\n\r\n /**\r\n * Extracts a clone of a mesh geometry\r\n * @param mesh defines the source mesh\r\n * @param id defines the unique ID of the new geometry object\r\n * @returns the new geometry object\r\n */\r\n public static ExtractFromMesh(mesh: Mesh, id: string): Nullable {\r\n const geometry = mesh._geometry;\r\n\r\n if (!geometry) {\r\n return null;\r\n }\r\n\r\n return geometry.copy(id);\r\n }\r\n\r\n /**\r\n * You should now use Tools.RandomId(), this method is still here for legacy reasons.\r\n * Implementation from http://stackoverflow.com/questions/105034/how-to-create-a-guid-uuid-in-javascript/2117523#answer-2117523\r\n * Be aware Math.random() could cause collisions, but:\r\n * \"All but 6 of the 128 bits of the ID are randomly generated, which means that for any two ids, there's a 1 in 2^^122 (or 5.3x10^^36) chance they'll collide\"\r\n * @returns a string containing a new GUID\r\n */\r\n public static RandomId(): string {\r\n return Tools.RandomId();\r\n }\r\n\r\n private static _GetGeometryByLoadedUniqueId(uniqueId: string, scene: Scene) {\r\n for (let index = 0; index < scene.geometries.length; index++) {\r\n if (scene.geometries[index]._loadedUniqueId === uniqueId) {\r\n return scene.geometries[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _ImportGeometry(parsedGeometry: any, mesh: Mesh): void {\r\n const scene = mesh.getScene();\r\n\r\n // Geometry\r\n const geometryUniqueId = parsedGeometry.geometryUniqueId;\r\n const geometryId = parsedGeometry.geometryId;\r\n if (geometryUniqueId || geometryId) {\r\n const geometry = geometryUniqueId ? this._GetGeometryByLoadedUniqueId(geometryUniqueId, scene) : scene.getGeometryById(geometryId);\r\n if (geometry) {\r\n geometry.applyToMesh(mesh);\r\n }\r\n } else if (parsedGeometry instanceof ArrayBuffer) {\r\n const binaryInfo = mesh._binaryInfo;\r\n\r\n if (binaryInfo.positionsAttrDesc && binaryInfo.positionsAttrDesc.count > 0) {\r\n const positionsData = new Float32Array(parsedGeometry, binaryInfo.positionsAttrDesc.offset, binaryInfo.positionsAttrDesc.count);\r\n mesh.setVerticesData(VertexBuffer.PositionKind, positionsData, false);\r\n }\r\n\r\n if (binaryInfo.normalsAttrDesc && binaryInfo.normalsAttrDesc.count > 0) {\r\n const normalsData = new Float32Array(parsedGeometry, binaryInfo.normalsAttrDesc.offset, binaryInfo.normalsAttrDesc.count);\r\n mesh.setVerticesData(VertexBuffer.NormalKind, normalsData, false);\r\n }\r\n\r\n if (binaryInfo.tangetsAttrDesc && binaryInfo.tangetsAttrDesc.count > 0) {\r\n const tangentsData = new Float32Array(parsedGeometry, binaryInfo.tangetsAttrDesc.offset, binaryInfo.tangetsAttrDesc.count);\r\n mesh.setVerticesData(VertexBuffer.TangentKind, tangentsData, false);\r\n }\r\n\r\n if (binaryInfo.uvsAttrDesc && binaryInfo.uvsAttrDesc.count > 0) {\r\n const uvsData = new Float32Array(parsedGeometry, binaryInfo.uvsAttrDesc.offset, binaryInfo.uvsAttrDesc.count);\r\n if (CompatibilityOptions.UseOpenGLOrientationForUV) {\r\n for (let index = 1; index < uvsData.length; index += 2) {\r\n uvsData[index] = 1 - uvsData[index];\r\n }\r\n }\r\n mesh.setVerticesData(VertexBuffer.UVKind, uvsData, false);\r\n }\r\n\r\n if (binaryInfo.uvs2AttrDesc && binaryInfo.uvs2AttrDesc.count > 0) {\r\n const uvs2Data = new Float32Array(parsedGeometry, binaryInfo.uvs2AttrDesc.offset, binaryInfo.uvs2AttrDesc.count);\r\n if (CompatibilityOptions.UseOpenGLOrientationForUV) {\r\n for (let index = 1; index < uvs2Data.length; index += 2) {\r\n uvs2Data[index] = 1 - uvs2Data[index];\r\n }\r\n }\r\n mesh.setVerticesData(VertexBuffer.UV2Kind, uvs2Data, false);\r\n }\r\n\r\n if (binaryInfo.uvs3AttrDesc && binaryInfo.uvs3AttrDesc.count > 0) {\r\n const uvs3Data = new Float32Array(parsedGeometry, binaryInfo.uvs3AttrDesc.offset, binaryInfo.uvs3AttrDesc.count);\r\n if (CompatibilityOptions.UseOpenGLOrientationForUV) {\r\n for (let index = 1; index < uvs3Data.length; index += 2) {\r\n uvs3Data[index] = 1 - uvs3Data[index];\r\n }\r\n }\r\n mesh.setVerticesData(VertexBuffer.UV3Kind, uvs3Data, false);\r\n }\r\n\r\n if (binaryInfo.uvs4AttrDesc && binaryInfo.uvs4AttrDesc.count > 0) {\r\n const uvs4Data = new Float32Array(parsedGeometry, binaryInfo.uvs4AttrDesc.offset, binaryInfo.uvs4AttrDesc.count);\r\n if (CompatibilityOptions.UseOpenGLOrientationForUV) {\r\n for (let index = 1; index < uvs4Data.length; index += 2) {\r\n uvs4Data[index] = 1 - uvs4Data[index];\r\n }\r\n }\r\n mesh.setVerticesData(VertexBuffer.UV4Kind, uvs4Data, false);\r\n }\r\n\r\n if (binaryInfo.uvs5AttrDesc && binaryInfo.uvs5AttrDesc.count > 0) {\r\n const uvs5Data = new Float32Array(parsedGeometry, binaryInfo.uvs5AttrDesc.offset, binaryInfo.uvs5AttrDesc.count);\r\n if (CompatibilityOptions.UseOpenGLOrientationForUV) {\r\n for (let index = 1; index < uvs5Data.length; index += 2) {\r\n uvs5Data[index] = 1 - uvs5Data[index];\r\n }\r\n }\r\n mesh.setVerticesData(VertexBuffer.UV5Kind, uvs5Data, false);\r\n }\r\n\r\n if (binaryInfo.uvs6AttrDesc && binaryInfo.uvs6AttrDesc.count > 0) {\r\n const uvs6Data = new Float32Array(parsedGeometry, binaryInfo.uvs6AttrDesc.offset, binaryInfo.uvs6AttrDesc.count);\r\n if (CompatibilityOptions.UseOpenGLOrientationForUV) {\r\n for (let index = 1; index < uvs6Data.length; index += 2) {\r\n uvs6Data[index] = 1 - uvs6Data[index];\r\n }\r\n }\r\n mesh.setVerticesData(VertexBuffer.UV6Kind, uvs6Data, false);\r\n }\r\n\r\n if (binaryInfo.colorsAttrDesc && binaryInfo.colorsAttrDesc.count > 0) {\r\n const colorsData = new Float32Array(parsedGeometry, binaryInfo.colorsAttrDesc.offset, binaryInfo.colorsAttrDesc.count);\r\n mesh.setVerticesData(VertexBuffer.ColorKind, colorsData, false, binaryInfo.colorsAttrDesc.stride);\r\n }\r\n\r\n if (binaryInfo.matricesIndicesAttrDesc && binaryInfo.matricesIndicesAttrDesc.count > 0) {\r\n const matricesIndicesData = new Int32Array(parsedGeometry, binaryInfo.matricesIndicesAttrDesc.offset, binaryInfo.matricesIndicesAttrDesc.count);\r\n const floatIndices = [];\r\n for (let i = 0; i < matricesIndicesData.length; i++) {\r\n const index = matricesIndicesData[i];\r\n floatIndices.push(index & 0x000000ff);\r\n floatIndices.push((index & 0x0000ff00) >> 8);\r\n floatIndices.push((index & 0x00ff0000) >> 16);\r\n floatIndices.push((index >> 24) & 0xff); // & 0xFF to convert to v + 256 if v < 0\r\n }\r\n mesh.setVerticesData(VertexBuffer.MatricesIndicesKind, floatIndices, false);\r\n }\r\n\r\n if (binaryInfo.matricesIndicesExtraAttrDesc && binaryInfo.matricesIndicesExtraAttrDesc.count > 0) {\r\n const matricesIndicesData = new Int32Array(parsedGeometry, binaryInfo.matricesIndicesExtraAttrDesc.offset, binaryInfo.matricesIndicesExtraAttrDesc.count);\r\n const floatIndices = [];\r\n for (let i = 0; i < matricesIndicesData.length; i++) {\r\n const index = matricesIndicesData[i];\r\n floatIndices.push(index & 0x000000ff);\r\n floatIndices.push((index & 0x0000ff00) >> 8);\r\n floatIndices.push((index & 0x00ff0000) >> 16);\r\n floatIndices.push((index >> 24) & 0xff); // & 0xFF to convert to v + 256 if v < 0\r\n }\r\n mesh.setVerticesData(VertexBuffer.MatricesIndicesExtraKind, floatIndices, false);\r\n }\r\n\r\n if (binaryInfo.matricesWeightsAttrDesc && binaryInfo.matricesWeightsAttrDesc.count > 0) {\r\n const matricesWeightsData = new Float32Array(parsedGeometry, binaryInfo.matricesWeightsAttrDesc.offset, binaryInfo.matricesWeightsAttrDesc.count);\r\n mesh.setVerticesData(VertexBuffer.MatricesWeightsKind, matricesWeightsData, false);\r\n }\r\n\r\n if (binaryInfo.indicesAttrDesc && binaryInfo.indicesAttrDesc.count > 0) {\r\n const indicesData = new Int32Array(parsedGeometry, binaryInfo.indicesAttrDesc.offset, binaryInfo.indicesAttrDesc.count);\r\n mesh.setIndices(indicesData, null);\r\n }\r\n\r\n if (binaryInfo.subMeshesAttrDesc && binaryInfo.subMeshesAttrDesc.count > 0) {\r\n const subMeshesData = new Int32Array(parsedGeometry, binaryInfo.subMeshesAttrDesc.offset, binaryInfo.subMeshesAttrDesc.count * 5);\r\n\r\n mesh.subMeshes = [];\r\n for (let i = 0; i < binaryInfo.subMeshesAttrDesc.count; i++) {\r\n const materialIndex = subMeshesData[i * 5 + 0];\r\n const verticesStart = subMeshesData[i * 5 + 1];\r\n const verticesCount = subMeshesData[i * 5 + 2];\r\n const indexStart = subMeshesData[i * 5 + 3];\r\n const indexCount = subMeshesData[i * 5 + 4];\r\n\r\n SubMesh.AddToMesh(materialIndex, verticesStart, verticesCount, indexStart, indexCount, mesh);\r\n }\r\n }\r\n } else if (parsedGeometry.positions && parsedGeometry.normals && parsedGeometry.indices) {\r\n mesh.setVerticesData(VertexBuffer.PositionKind, parsedGeometry.positions, parsedGeometry.positions._updatable);\r\n\r\n mesh.setVerticesData(VertexBuffer.NormalKind, parsedGeometry.normals, parsedGeometry.normals._updatable);\r\n\r\n if (parsedGeometry.tangents) {\r\n mesh.setVerticesData(VertexBuffer.TangentKind, parsedGeometry.tangents, parsedGeometry.tangents._updatable);\r\n }\r\n\r\n if (parsedGeometry.uvs) {\r\n mesh.setVerticesData(VertexBuffer.UVKind, parsedGeometry.uvs, parsedGeometry.uvs._updatable);\r\n }\r\n\r\n if (parsedGeometry.uvs2) {\r\n mesh.setVerticesData(VertexBuffer.UV2Kind, parsedGeometry.uvs2, parsedGeometry.uvs2._updatable);\r\n }\r\n\r\n if (parsedGeometry.uvs3) {\r\n mesh.setVerticesData(VertexBuffer.UV3Kind, parsedGeometry.uvs3, parsedGeometry.uvs3._updatable);\r\n }\r\n\r\n if (parsedGeometry.uvs4) {\r\n mesh.setVerticesData(VertexBuffer.UV4Kind, parsedGeometry.uvs4, parsedGeometry.uvs4._updatable);\r\n }\r\n\r\n if (parsedGeometry.uvs5) {\r\n mesh.setVerticesData(VertexBuffer.UV5Kind, parsedGeometry.uvs5, parsedGeometry.uvs5._updatable);\r\n }\r\n\r\n if (parsedGeometry.uvs6) {\r\n mesh.setVerticesData(VertexBuffer.UV6Kind, parsedGeometry.uvs6, parsedGeometry.uvs6._updatable);\r\n }\r\n\r\n if (parsedGeometry.colors) {\r\n mesh.setVerticesData(VertexBuffer.ColorKind, Color4.CheckColors4(parsedGeometry.colors, parsedGeometry.positions.length / 3), parsedGeometry.colors._updatable);\r\n }\r\n\r\n if (parsedGeometry.matricesIndices) {\r\n if (!parsedGeometry.matricesIndices._isExpanded) {\r\n const floatIndices = [];\r\n\r\n for (let i = 0; i < parsedGeometry.matricesIndices.length; i++) {\r\n const matricesIndex = parsedGeometry.matricesIndices[i];\r\n\r\n floatIndices.push(matricesIndex & 0x000000ff);\r\n floatIndices.push((matricesIndex & 0x0000ff00) >> 8);\r\n floatIndices.push((matricesIndex & 0x00ff0000) >> 16);\r\n floatIndices.push((matricesIndex >> 24) & 0xff); // & 0xFF to convert to v + 256 if v < 0\r\n }\r\n\r\n mesh.setVerticesData(VertexBuffer.MatricesIndicesKind, floatIndices, parsedGeometry.matricesIndices._updatable);\r\n } else {\r\n delete parsedGeometry.matricesIndices._isExpanded;\r\n mesh.setVerticesData(VertexBuffer.MatricesIndicesKind, parsedGeometry.matricesIndices, parsedGeometry.matricesIndices._updatable);\r\n }\r\n }\r\n\r\n if (parsedGeometry.matricesIndicesExtra) {\r\n if (!parsedGeometry.matricesIndicesExtra._isExpanded) {\r\n const floatIndices = [];\r\n\r\n for (let i = 0; i < parsedGeometry.matricesIndicesExtra.length; i++) {\r\n const matricesIndex = parsedGeometry.matricesIndicesExtra[i];\r\n\r\n floatIndices.push(matricesIndex & 0x000000ff);\r\n floatIndices.push((matricesIndex & 0x0000ff00) >> 8);\r\n floatIndices.push((matricesIndex & 0x00ff0000) >> 16);\r\n floatIndices.push((matricesIndex >> 24) & 0xff); // & 0xFF to convert to v + 256 if v < 0\r\n }\r\n\r\n mesh.setVerticesData(VertexBuffer.MatricesIndicesExtraKind, floatIndices, parsedGeometry.matricesIndicesExtra._updatable);\r\n } else {\r\n delete parsedGeometry.matricesIndices._isExpanded;\r\n mesh.setVerticesData(VertexBuffer.MatricesIndicesExtraKind, parsedGeometry.matricesIndicesExtra, parsedGeometry.matricesIndicesExtra._updatable);\r\n }\r\n }\r\n\r\n if (parsedGeometry.matricesWeights) {\r\n Geometry._CleanMatricesWeights(parsedGeometry, mesh);\r\n mesh.setVerticesData(VertexBuffer.MatricesWeightsKind, parsedGeometry.matricesWeights, parsedGeometry.matricesWeights._updatable);\r\n }\r\n\r\n if (parsedGeometry.matricesWeightsExtra) {\r\n mesh.setVerticesData(VertexBuffer.MatricesWeightsExtraKind, parsedGeometry.matricesWeightsExtra, parsedGeometry.matricesWeights._updatable);\r\n }\r\n\r\n mesh.setIndices(parsedGeometry.indices, null);\r\n }\r\n\r\n // SubMeshes\r\n if (parsedGeometry.subMeshes) {\r\n mesh.subMeshes = [];\r\n for (let subIndex = 0; subIndex < parsedGeometry.subMeshes.length; subIndex++) {\r\n const parsedSubMesh = parsedGeometry.subMeshes[subIndex];\r\n\r\n SubMesh.AddToMesh(\r\n parsedSubMesh.materialIndex,\r\n parsedSubMesh.verticesStart,\r\n parsedSubMesh.verticesCount,\r\n parsedSubMesh.indexStart,\r\n parsedSubMesh.indexCount,\r\n mesh\r\n );\r\n }\r\n }\r\n\r\n // Flat shading\r\n if (mesh._shouldGenerateFlatShading) {\r\n mesh.convertToFlatShadedMesh();\r\n mesh._shouldGenerateFlatShading = false;\r\n }\r\n\r\n // Update\r\n mesh.computeWorldMatrix(true);\r\n\r\n scene.onMeshImportedObservable.notifyObservers(mesh);\r\n }\r\n\r\n private static _CleanMatricesWeights(parsedGeometry: any, mesh: Mesh): void {\r\n const epsilon: number = 1e-3;\r\n if (!SceneLoaderFlags.CleanBoneMatrixWeights) {\r\n return;\r\n }\r\n let noInfluenceBoneIndex = 0.0;\r\n if (parsedGeometry.skeletonId > -1) {\r\n const skeleton = mesh.getScene().getLastSkeletonById(parsedGeometry.skeletonId);\r\n\r\n if (!skeleton) {\r\n return;\r\n }\r\n noInfluenceBoneIndex = skeleton.bones.length;\r\n } else {\r\n return;\r\n }\r\n const matricesIndices = mesh.getVerticesData(VertexBuffer.MatricesIndicesKind);\r\n const matricesIndicesExtra = mesh.getVerticesData(VertexBuffer.MatricesIndicesExtraKind);\r\n const matricesWeights = parsedGeometry.matricesWeights;\r\n const matricesWeightsExtra = parsedGeometry.matricesWeightsExtra;\r\n const influencers = parsedGeometry.numBoneInfluencer;\r\n const size = matricesWeights.length;\r\n\r\n for (let i = 0; i < size; i += 4) {\r\n let weight = 0.0;\r\n let firstZeroWeight = -1;\r\n for (let j = 0; j < 4; j++) {\r\n const w = matricesWeights[i + j];\r\n weight += w;\r\n if (w < epsilon && firstZeroWeight < 0) {\r\n firstZeroWeight = j;\r\n }\r\n }\r\n if (matricesWeightsExtra) {\r\n for (let j = 0; j < 4; j++) {\r\n const w = matricesWeightsExtra[i + j];\r\n weight += w;\r\n if (w < epsilon && firstZeroWeight < 0) {\r\n firstZeroWeight = j + 4;\r\n }\r\n }\r\n }\r\n if (firstZeroWeight < 0 || firstZeroWeight > influencers - 1) {\r\n firstZeroWeight = influencers - 1;\r\n }\r\n if (weight > epsilon) {\r\n const mweight = 1.0 / weight;\r\n for (let j = 0; j < 4; j++) {\r\n matricesWeights[i + j] *= mweight;\r\n }\r\n if (matricesWeightsExtra) {\r\n for (let j = 0; j < 4; j++) {\r\n matricesWeightsExtra[i + j] *= mweight;\r\n }\r\n }\r\n } else {\r\n if (firstZeroWeight >= 4) {\r\n matricesWeightsExtra[i + firstZeroWeight - 4] = 1.0 - weight;\r\n matricesIndicesExtra[i + firstZeroWeight - 4] = noInfluenceBoneIndex;\r\n } else {\r\n matricesWeights[i + firstZeroWeight] = 1.0 - weight;\r\n matricesIndices[i + firstZeroWeight] = noInfluenceBoneIndex;\r\n }\r\n }\r\n }\r\n\r\n mesh.setVerticesData(VertexBuffer.MatricesIndicesKind, matricesIndices);\r\n if (parsedGeometry.matricesWeightsExtra) {\r\n mesh.setVerticesData(VertexBuffer.MatricesIndicesExtraKind, matricesIndicesExtra);\r\n }\r\n }\r\n\r\n /**\r\n * Create a new geometry from persisted data (Using .babylon file format)\r\n * @param parsedVertexData defines the persisted data\r\n * @param scene defines the hosting scene\r\n * @param rootUrl defines the root url to use to load assets (like delayed data)\r\n * @returns the new geometry object\r\n */\r\n public static Parse(parsedVertexData: any, scene: Scene, rootUrl: string): Nullable {\r\n const geometry = new Geometry(parsedVertexData.id, scene, undefined, parsedVertexData.updatable);\r\n geometry._loadedUniqueId = parsedVertexData.uniqueId;\r\n\r\n if (Tags) {\r\n Tags.AddTagsTo(geometry, parsedVertexData.tags);\r\n }\r\n\r\n if (parsedVertexData.delayLoadingFile) {\r\n geometry.delayLoadState = Constants.DELAYLOADSTATE_NOTLOADED;\r\n geometry.delayLoadingFile = rootUrl + parsedVertexData.delayLoadingFile;\r\n geometry._boundingInfo = new BoundingInfo(Vector3.FromArray(parsedVertexData.boundingBoxMinimum), Vector3.FromArray(parsedVertexData.boundingBoxMaximum));\r\n\r\n geometry._delayInfo = [];\r\n if (parsedVertexData.hasUVs) {\r\n geometry._delayInfo.push(VertexBuffer.UVKind);\r\n }\r\n\r\n if (parsedVertexData.hasUVs2) {\r\n geometry._delayInfo.push(VertexBuffer.UV2Kind);\r\n }\r\n\r\n if (parsedVertexData.hasUVs3) {\r\n geometry._delayInfo.push(VertexBuffer.UV3Kind);\r\n }\r\n\r\n if (parsedVertexData.hasUVs4) {\r\n geometry._delayInfo.push(VertexBuffer.UV4Kind);\r\n }\r\n\r\n if (parsedVertexData.hasUVs5) {\r\n geometry._delayInfo.push(VertexBuffer.UV5Kind);\r\n }\r\n\r\n if (parsedVertexData.hasUVs6) {\r\n geometry._delayInfo.push(VertexBuffer.UV6Kind);\r\n }\r\n\r\n if (parsedVertexData.hasColors) {\r\n geometry._delayInfo.push(VertexBuffer.ColorKind);\r\n }\r\n\r\n if (parsedVertexData.hasMatricesIndices) {\r\n geometry._delayInfo.push(VertexBuffer.MatricesIndicesKind);\r\n }\r\n\r\n if (parsedVertexData.hasMatricesWeights) {\r\n geometry._delayInfo.push(VertexBuffer.MatricesWeightsKind);\r\n }\r\n\r\n geometry._delayLoadingFunction = VertexData.ImportVertexData;\r\n } else {\r\n VertexData.ImportVertexData(parsedVertexData, geometry);\r\n }\r\n\r\n scene.pushGeometry(geometry, true);\r\n\r\n return geometry;\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\nimport { PrecisionDate } from \"./precisionDate\";\r\n\r\n/**\r\n * Performance monitor tracks rolling average frame-time and frame-time variance over a user defined sliding-window\r\n */\r\nexport class PerformanceMonitor {\r\n private _enabled: boolean = true;\r\n private _rollingFrameTime: RollingAverage;\r\n private _lastFrameTimeMs: Nullable;\r\n\r\n /**\r\n * constructor\r\n * @param frameSampleSize The number of samples required to saturate the sliding window\r\n */\r\n constructor(frameSampleSize: number = 30) {\r\n this._rollingFrameTime = new RollingAverage(frameSampleSize);\r\n }\r\n\r\n /**\r\n * Samples current frame\r\n * @param timeMs A timestamp in milliseconds of the current frame to compare with other frames\r\n */\r\n public sampleFrame(timeMs: number = PrecisionDate.Now) {\r\n if (!this._enabled) {\r\n return;\r\n }\r\n\r\n if (this._lastFrameTimeMs != null) {\r\n const dt = timeMs - this._lastFrameTimeMs;\r\n this._rollingFrameTime.add(dt);\r\n }\r\n\r\n this._lastFrameTimeMs = timeMs;\r\n }\r\n\r\n /**\r\n * Returns the average frame time in milliseconds over the sliding window (or the subset of frames sampled so far)\r\n */\r\n public get averageFrameTime(): number {\r\n return this._rollingFrameTime.average;\r\n }\r\n\r\n /**\r\n * Returns the variance frame time in milliseconds over the sliding window (or the subset of frames sampled so far)\r\n */\r\n public get averageFrameTimeVariance(): number {\r\n return this._rollingFrameTime.variance;\r\n }\r\n\r\n /**\r\n * Returns the frame time of the most recent frame\r\n */\r\n public get instantaneousFrameTime(): number {\r\n return this._rollingFrameTime.history(0);\r\n }\r\n\r\n /**\r\n * Returns the average framerate in frames per second over the sliding window (or the subset of frames sampled so far)\r\n */\r\n public get averageFPS(): number {\r\n return 1000.0 / this._rollingFrameTime.average;\r\n }\r\n\r\n /**\r\n * Returns the average framerate in frames per second using the most recent frame time\r\n */\r\n public get instantaneousFPS(): number {\r\n const history = this._rollingFrameTime.history(0);\r\n\r\n if (history === 0) {\r\n return 0;\r\n }\r\n\r\n return 1000.0 / history;\r\n }\r\n\r\n /**\r\n * Returns true if enough samples have been taken to completely fill the sliding window\r\n */\r\n public get isSaturated(): boolean {\r\n return this._rollingFrameTime.isSaturated();\r\n }\r\n\r\n /**\r\n * Enables contributions to the sliding window sample set\r\n */\r\n public enable() {\r\n this._enabled = true;\r\n }\r\n\r\n /**\r\n * Disables contributions to the sliding window sample set\r\n * Samples will not be interpolated over the disabled period\r\n */\r\n public disable() {\r\n this._enabled = false;\r\n //clear last sample to avoid interpolating over the disabled period when next enabled\r\n this._lastFrameTimeMs = null;\r\n }\r\n\r\n /**\r\n * Returns true if sampling is enabled\r\n */\r\n public get isEnabled(): boolean {\r\n return this._enabled;\r\n }\r\n\r\n /**\r\n * Resets performance monitor\r\n */\r\n public reset() {\r\n //clear last sample to avoid interpolating over the disabled period when next enabled\r\n this._lastFrameTimeMs = null;\r\n //wipe record\r\n this._rollingFrameTime.reset();\r\n }\r\n}\r\n\r\n/**\r\n * RollingAverage\r\n *\r\n * Utility to efficiently compute the rolling average and variance over a sliding window of samples\r\n */\r\nexport class RollingAverage {\r\n /**\r\n * Current average\r\n */\r\n public average: number;\r\n /**\r\n * Current variance\r\n */\r\n public variance: number;\r\n\r\n protected _samples: Array;\r\n protected _sampleCount: number;\r\n protected _pos: number;\r\n protected _m2: number; //sum of squares of differences from the (current) mean\r\n\r\n /**\r\n * constructor\r\n * @param length The number of samples required to saturate the sliding window\r\n */\r\n constructor(length: number) {\r\n this._samples = new Array(length);\r\n this.reset();\r\n }\r\n\r\n /**\r\n * Adds a sample to the sample set\r\n * @param v The sample value\r\n */\r\n public add(v: number) {\r\n //http://en.wikipedia.org/wiki/Algorithms_for_calculating_variance\r\n let delta: number;\r\n\r\n //we need to check if we've already wrapped round\r\n if (this.isSaturated()) {\r\n //remove bottom of stack from mean\r\n const bottomValue = this._samples[this._pos];\r\n delta = bottomValue - this.average;\r\n this.average -= delta / (this._sampleCount - 1);\r\n this._m2 -= delta * (bottomValue - this.average);\r\n } else {\r\n this._sampleCount++;\r\n }\r\n\r\n //add new value to mean\r\n delta = v - this.average;\r\n this.average += delta / this._sampleCount;\r\n this._m2 += delta * (v - this.average);\r\n\r\n //set the new variance\r\n this.variance = this._m2 / (this._sampleCount - 1);\r\n\r\n this._samples[this._pos] = v;\r\n this._pos++;\r\n\r\n this._pos %= this._samples.length; //positive wrap around\r\n }\r\n\r\n /**\r\n * Returns previously added values or null if outside of history or outside the sliding window domain\r\n * @param i Index in history. For example, pass 0 for the most recent value and 1 for the value before that\r\n * @returns Value previously recorded with add() or null if outside of range\r\n */\r\n public history(i: number): number {\r\n if (i >= this._sampleCount || i >= this._samples.length) {\r\n return 0;\r\n }\r\n\r\n const i0 = this._wrapPosition(this._pos - 1.0);\r\n return this._samples[this._wrapPosition(i0 - i)];\r\n }\r\n\r\n /**\r\n * Returns true if enough samples have been taken to completely fill the sliding window\r\n * @returns true if sample-set saturated\r\n */\r\n public isSaturated(): boolean {\r\n return this._sampleCount >= this._samples.length;\r\n }\r\n\r\n /**\r\n * Resets the rolling average (equivalent to 0 samples taken so far)\r\n */\r\n public reset() {\r\n this.average = 0;\r\n this.variance = 0;\r\n this._sampleCount = 0;\r\n this._pos = 0;\r\n this._m2 = 0;\r\n }\r\n\r\n /**\r\n * Wraps a value around the sample range boundaries\r\n * @param i Position in sample range, for example if the sample length is 5, and i is -3, then 2 will be returned.\r\n * @returns Wrapped position in sample range\r\n */\r\n protected _wrapPosition(i: number): number {\r\n const max = this._samples.length;\r\n return ((i % max) + max) % max;\r\n }\r\n}\r\n","import { ThinEngine } from \"../../Engines/thinEngine\";\r\nimport { Constants } from \"../constants\";\r\n\r\ndeclare module \"../../Engines/thinEngine\" {\r\n export interface ThinEngine {\r\n /**\r\n * Sets alpha constants used by some alpha blending modes\r\n * @param r defines the red component\r\n * @param g defines the green component\r\n * @param b defines the blue component\r\n * @param a defines the alpha component\r\n */\r\n setAlphaConstants(r: number, g: number, b: number, a: number): void;\r\n\r\n /**\r\n * Sets the current alpha mode\r\n * @param mode defines the mode to use (one of the Engine.ALPHA_XXX)\r\n * @param noDepthWriteChange defines if depth writing state should remains unchanged (false by default)\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/advanced/transparent_rendering\r\n */\r\n setAlphaMode(mode: number, noDepthWriteChange?: boolean): void;\r\n\r\n /**\r\n * Gets the current alpha mode\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/advanced/transparent_rendering\r\n * @returns the current alpha mode\r\n */\r\n getAlphaMode(): number;\r\n\r\n /**\r\n * Sets the current alpha equation\r\n * @param equation defines the equation to use (one of the Engine.ALPHA_EQUATION_XXX)\r\n */\r\n setAlphaEquation(equation: number): void;\r\n\r\n /**\r\n * Gets the current alpha equation.\r\n * @returns the current alpha equation\r\n */\r\n getAlphaEquation(): number;\r\n }\r\n}\r\n\r\nThinEngine.prototype.setAlphaConstants = function (r: number, g: number, b: number, a: number) {\r\n this._alphaState.setAlphaBlendConstants(r, g, b, a);\r\n};\r\n\r\nThinEngine.prototype.setAlphaMode = function (mode: number, noDepthWriteChange: boolean = false): void {\r\n if (this._alphaMode === mode) {\r\n if (!noDepthWriteChange) {\r\n // Make sure we still have the correct depth mask according to the alpha mode (a transparent material could have forced writting to the depth buffer, for instance)\r\n const depthMask = mode === Constants.ALPHA_DISABLE;\r\n if (this.depthCullingState.depthMask !== depthMask) {\r\n this.depthCullingState.depthMask = depthMask;\r\n }\r\n }\r\n return;\r\n }\r\n\r\n switch (mode) {\r\n case Constants.ALPHA_DISABLE:\r\n this._alphaState.alphaBlend = false;\r\n break;\r\n case Constants.ALPHA_PREMULTIPLIED:\r\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA, this._gl.ONE, this._gl.ONE);\r\n this._alphaState.alphaBlend = true;\r\n break;\r\n case Constants.ALPHA_PREMULTIPLIED_PORTERDUFF:\r\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA, this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA);\r\n this._alphaState.alphaBlend = true;\r\n break;\r\n case Constants.ALPHA_COMBINE:\r\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.SRC_ALPHA, this._gl.ONE_MINUS_SRC_ALPHA, this._gl.ONE, this._gl.ONE);\r\n this._alphaState.alphaBlend = true;\r\n break;\r\n case Constants.ALPHA_ONEONE:\r\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE, this._gl.ZERO, this._gl.ONE);\r\n this._alphaState.alphaBlend = true;\r\n break;\r\n case Constants.ALPHA_ADD:\r\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.SRC_ALPHA, this._gl.ONE, this._gl.ZERO, this._gl.ONE);\r\n this._alphaState.alphaBlend = true;\r\n break;\r\n case Constants.ALPHA_SUBTRACT:\r\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.ZERO, this._gl.ONE_MINUS_SRC_COLOR, this._gl.ONE, this._gl.ONE);\r\n this._alphaState.alphaBlend = true;\r\n break;\r\n case Constants.ALPHA_MULTIPLY:\r\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.DST_COLOR, this._gl.ZERO, this._gl.ONE, this._gl.ONE);\r\n this._alphaState.alphaBlend = true;\r\n break;\r\n case Constants.ALPHA_MAXIMIZED:\r\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.SRC_ALPHA, this._gl.ONE_MINUS_SRC_COLOR, this._gl.ONE, this._gl.ONE);\r\n this._alphaState.alphaBlend = true;\r\n break;\r\n case Constants.ALPHA_INTERPOLATE:\r\n this._alphaState.setAlphaBlendFunctionParameters(\r\n this._gl.CONSTANT_COLOR,\r\n this._gl.ONE_MINUS_CONSTANT_COLOR,\r\n this._gl.CONSTANT_ALPHA,\r\n this._gl.ONE_MINUS_CONSTANT_ALPHA\r\n );\r\n this._alphaState.alphaBlend = true;\r\n break;\r\n case Constants.ALPHA_SCREENMODE:\r\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE_MINUS_SRC_COLOR, this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA);\r\n this._alphaState.alphaBlend = true;\r\n break;\r\n case Constants.ALPHA_ONEONE_ONEONE:\r\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE, this._gl.ONE, this._gl.ONE);\r\n this._alphaState.alphaBlend = true;\r\n break;\r\n case Constants.ALPHA_ALPHATOCOLOR:\r\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.DST_ALPHA, this._gl.ONE, this._gl.ZERO, this._gl.ZERO);\r\n this._alphaState.alphaBlend = true;\r\n break;\r\n case Constants.ALPHA_REVERSEONEMINUS:\r\n this._alphaState.setAlphaBlendFunctionParameters(\r\n this._gl.ONE_MINUS_DST_COLOR,\r\n this._gl.ONE_MINUS_SRC_COLOR,\r\n this._gl.ONE_MINUS_DST_ALPHA,\r\n this._gl.ONE_MINUS_SRC_ALPHA\r\n );\r\n this._alphaState.alphaBlend = true;\r\n break;\r\n case Constants.ALPHA_SRC_DSTONEMINUSSRCALPHA:\r\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA, this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA);\r\n this._alphaState.alphaBlend = true;\r\n break;\r\n case Constants.ALPHA_ONEONE_ONEZERO:\r\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE, this._gl.ONE, this._gl.ONE, this._gl.ZERO);\r\n this._alphaState.alphaBlend = true;\r\n break;\r\n case Constants.ALPHA_EXCLUSION:\r\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.ONE_MINUS_DST_COLOR, this._gl.ONE_MINUS_SRC_COLOR, this._gl.ZERO, this._gl.ONE);\r\n this._alphaState.alphaBlend = true;\r\n break;\r\n case Constants.ALPHA_LAYER_ACCUMULATE:\r\n // Same as ALPHA_COMBINE but accumulates (1 - alpha) values in the alpha channel for a later readout in order independant transparency\r\n this._alphaState.setAlphaBlendFunctionParameters(this._gl.SRC_ALPHA, this._gl.ONE_MINUS_SRC_ALPHA, this._gl.ONE, this._gl.ONE_MINUS_SRC_ALPHA);\r\n this._alphaState.alphaBlend = true;\r\n break;\r\n }\r\n if (!noDepthWriteChange) {\r\n this.depthCullingState.depthMask = mode === Constants.ALPHA_DISABLE;\r\n }\r\n this._alphaMode = mode;\r\n};\r\n\r\nThinEngine.prototype.getAlphaMode = function (): number {\r\n return this._alphaMode;\r\n};\r\n\r\nThinEngine.prototype.setAlphaEquation = function (equation: number): void {\r\n if (this._alphaEquation === equation) {\r\n return;\r\n }\r\n\r\n switch (equation) {\r\n case Constants.ALPHA_EQUATION_ADD:\r\n this._alphaState.setAlphaEquationParameters(Constants.GL_ALPHA_EQUATION_ADD, Constants.GL_ALPHA_EQUATION_ADD);\r\n break;\r\n case Constants.ALPHA_EQUATION_SUBSTRACT:\r\n this._alphaState.setAlphaEquationParameters(Constants.GL_ALPHA_EQUATION_SUBTRACT, Constants.GL_ALPHA_EQUATION_SUBTRACT);\r\n break;\r\n case Constants.ALPHA_EQUATION_REVERSE_SUBTRACT:\r\n this._alphaState.setAlphaEquationParameters(Constants.GL_ALPHA_EQUATION_REVERSE_SUBTRACT, Constants.GL_ALPHA_EQUATION_REVERSE_SUBTRACT);\r\n break;\r\n case Constants.ALPHA_EQUATION_MAX:\r\n this._alphaState.setAlphaEquationParameters(Constants.GL_ALPHA_EQUATION_MAX, Constants.GL_ALPHA_EQUATION_MAX);\r\n break;\r\n case Constants.ALPHA_EQUATION_MIN:\r\n this._alphaState.setAlphaEquationParameters(Constants.GL_ALPHA_EQUATION_MIN, Constants.GL_ALPHA_EQUATION_MIN);\r\n break;\r\n case Constants.ALPHA_EQUATION_DARKEN:\r\n this._alphaState.setAlphaEquationParameters(Constants.GL_ALPHA_EQUATION_MIN, Constants.GL_ALPHA_EQUATION_ADD);\r\n break;\r\n }\r\n this._alphaEquation = equation;\r\n};\r\n\r\nThinEngine.prototype.getAlphaEquation = function () {\r\n return this._alphaEquation;\r\n};\r\n","import { ThinEngine } from \"../../Engines/thinEngine\";\r\nimport type { InternalTexture } from \"../../Materials/Textures/internalTexture\";\r\nimport type { Nullable } from \"../../types\";\r\nimport { Constants } from \"../constants\";\r\n\r\ndeclare module \"../../Engines/thinEngine\" {\r\n export interface ThinEngine {\r\n /** @internal */\r\n _readTexturePixels(\r\n texture: InternalTexture,\r\n width: number,\r\n height: number,\r\n faceIndex?: number,\r\n level?: number,\r\n buffer?: Nullable,\r\n flushRenderer?: boolean,\r\n noDataConversion?: boolean,\r\n x?: number,\r\n y?: number\r\n ): Promise;\r\n\r\n /** @internal */\r\n _readTexturePixelsSync(\r\n texture: InternalTexture,\r\n width: number,\r\n height: number,\r\n faceIndex?: number,\r\n level?: number,\r\n buffer?: Nullable,\r\n flushRenderer?: boolean,\r\n noDataConversion?: boolean,\r\n x?: number,\r\n y?: number\r\n ): ArrayBufferView;\r\n }\r\n}\r\n\r\n/**\r\n * Allocate a typed array depending on a texture type. Optionally can copy existing data in the buffer.\r\n * @param type type of the texture\r\n * @param sizeOrDstBuffer size of the array OR an existing buffer that will be used as the destination of the copy (if copyBuffer is provided)\r\n * @param sizeInBytes true if the size of the array is given in bytes, false if it is the number of elements of the array\r\n * @param copyBuffer if provided, buffer to copy into the destination buffer (either a newly allocated buffer if sizeOrDstBuffer is a number or use sizeOrDstBuffer as the destination buffer otherwise)\r\n * @returns the allocated buffer or sizeOrDstBuffer if the latter is an ArrayBuffer\r\n */\r\nexport function allocateAndCopyTypedBuffer(type: number, sizeOrDstBuffer: number | ArrayBuffer, sizeInBytes = false, copyBuffer?: ArrayBuffer): ArrayBufferView {\r\n switch (type) {\r\n case Constants.TEXTURETYPE_BYTE: {\r\n const buffer = sizeOrDstBuffer instanceof ArrayBuffer ? new Int8Array(sizeOrDstBuffer) : new Int8Array(sizeOrDstBuffer);\r\n if (copyBuffer) {\r\n buffer.set(new Int8Array(copyBuffer));\r\n }\r\n return buffer;\r\n }\r\n case Constants.TEXTURETYPE_UNSIGNED_BYTE: {\r\n const buffer = sizeOrDstBuffer instanceof ArrayBuffer ? new Uint8Array(sizeOrDstBuffer) : new Uint8Array(sizeOrDstBuffer);\r\n if (copyBuffer) {\r\n buffer.set(new Uint8Array(copyBuffer));\r\n }\r\n return buffer;\r\n }\r\n case Constants.TEXTURETYPE_SHORT: {\r\n const buffer = sizeOrDstBuffer instanceof ArrayBuffer ? new Int16Array(sizeOrDstBuffer) : new Int16Array(sizeInBytes ? sizeOrDstBuffer / 2 : sizeOrDstBuffer);\r\n if (copyBuffer) {\r\n buffer.set(new Int16Array(copyBuffer));\r\n }\r\n return buffer;\r\n }\r\n case Constants.TEXTURETYPE_UNSIGNED_SHORT:\r\n case Constants.TEXTURETYPE_UNSIGNED_SHORT_4_4_4_4:\r\n case Constants.TEXTURETYPE_UNSIGNED_SHORT_5_5_5_1:\r\n case Constants.TEXTURETYPE_UNSIGNED_SHORT_5_6_5:\r\n case Constants.TEXTURETYPE_HALF_FLOAT: {\r\n const buffer = sizeOrDstBuffer instanceof ArrayBuffer ? new Uint16Array(sizeOrDstBuffer) : new Uint16Array(sizeInBytes ? sizeOrDstBuffer / 2 : sizeOrDstBuffer);\r\n if (copyBuffer) {\r\n buffer.set(new Uint16Array(copyBuffer));\r\n }\r\n return buffer;\r\n }\r\n case Constants.TEXTURETYPE_INT: {\r\n const buffer = sizeOrDstBuffer instanceof ArrayBuffer ? new Int32Array(sizeOrDstBuffer) : new Int32Array(sizeInBytes ? sizeOrDstBuffer / 4 : sizeOrDstBuffer);\r\n if (copyBuffer) {\r\n buffer.set(new Int32Array(copyBuffer));\r\n }\r\n return buffer;\r\n }\r\n case Constants.TEXTURETYPE_UNSIGNED_INTEGER:\r\n case Constants.TEXTURETYPE_UNSIGNED_INT_2_10_10_10_REV:\r\n case Constants.TEXTURETYPE_UNSIGNED_INT_24_8:\r\n case Constants.TEXTURETYPE_UNSIGNED_INT_10F_11F_11F_REV:\r\n case Constants.TEXTURETYPE_UNSIGNED_INT_5_9_9_9_REV:\r\n case Constants.TEXTURETYPE_FLOAT_32_UNSIGNED_INT_24_8_REV: {\r\n const buffer = sizeOrDstBuffer instanceof ArrayBuffer ? new Uint32Array(sizeOrDstBuffer) : new Uint32Array(sizeInBytes ? sizeOrDstBuffer / 4 : sizeOrDstBuffer);\r\n if (copyBuffer) {\r\n buffer.set(new Uint32Array(copyBuffer));\r\n }\r\n return buffer;\r\n }\r\n case Constants.TEXTURETYPE_FLOAT: {\r\n const buffer = sizeOrDstBuffer instanceof ArrayBuffer ? new Float32Array(sizeOrDstBuffer) : new Float32Array(sizeInBytes ? sizeOrDstBuffer / 4 : sizeOrDstBuffer);\r\n if (copyBuffer) {\r\n buffer.set(new Float32Array(copyBuffer));\r\n }\r\n return buffer;\r\n }\r\n }\r\n\r\n const buffer = sizeOrDstBuffer instanceof ArrayBuffer ? new Uint8Array(sizeOrDstBuffer) : new Uint8Array(sizeOrDstBuffer);\r\n if (copyBuffer) {\r\n buffer.set(new Uint8Array(copyBuffer));\r\n }\r\n return buffer;\r\n}\r\n\r\nThinEngine.prototype._readTexturePixelsSync = function (\r\n texture: InternalTexture,\r\n width: number,\r\n height: number,\r\n faceIndex = -1,\r\n level = 0,\r\n buffer: Nullable = null,\r\n flushRenderer = true,\r\n noDataConversion = false,\r\n x = 0,\r\n y = 0\r\n): ArrayBufferView {\r\n const gl = this._gl;\r\n if (!gl) {\r\n throw new Error(\"Engine does not have gl rendering context.\");\r\n }\r\n if (!this._dummyFramebuffer) {\r\n const dummy = gl.createFramebuffer();\r\n\r\n if (!dummy) {\r\n throw new Error(\"Unable to create dummy framebuffer\");\r\n }\r\n\r\n this._dummyFramebuffer = dummy;\r\n }\r\n gl.bindFramebuffer(gl.FRAMEBUFFER, this._dummyFramebuffer);\r\n\r\n if (faceIndex > -1) {\r\n gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex, texture._hardwareTexture?.underlyingResource, level);\r\n } else {\r\n gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, texture._hardwareTexture?.underlyingResource, level);\r\n }\r\n\r\n let readType = texture.type !== undefined ? this._getWebGLTextureType(texture.type) : gl.UNSIGNED_BYTE;\r\n\r\n if (!noDataConversion) {\r\n switch (readType) {\r\n case gl.UNSIGNED_BYTE:\r\n if (!buffer) {\r\n buffer = new Uint8Array(4 * width * height);\r\n }\r\n readType = gl.UNSIGNED_BYTE;\r\n break;\r\n default:\r\n if (!buffer) {\r\n buffer = new Float32Array(4 * width * height);\r\n }\r\n readType = gl.FLOAT;\r\n break;\r\n }\r\n } else if (!buffer) {\r\n buffer = allocateAndCopyTypedBuffer(texture.type, 4 * width * height);\r\n }\r\n\r\n if (flushRenderer) {\r\n this.flushFramebuffer();\r\n }\r\n\r\n gl.readPixels(x, y, width, height, gl.RGBA, readType, buffer);\r\n gl.bindFramebuffer(gl.FRAMEBUFFER, this._currentFramebuffer);\r\n\r\n return buffer;\r\n};\r\n\r\nThinEngine.prototype._readTexturePixels = function (\r\n texture: InternalTexture,\r\n width: number,\r\n height: number,\r\n faceIndex = -1,\r\n level = 0,\r\n buffer: Nullable = null,\r\n flushRenderer = true,\r\n noDataConversion = false,\r\n x = 0,\r\n y = 0\r\n): Promise {\r\n return Promise.resolve(this._readTexturePixelsSync(texture, width, height, faceIndex, level, buffer, flushRenderer, noDataConversion, x, y));\r\n};\r\n","import { ThinEngine } from \"../../Engines/thinEngine\";\r\nimport type { DataBuffer } from \"../../Buffers/dataBuffer\";\r\nimport type { IndicesArray, DataArray } from \"../../types\";\r\n\r\ndeclare module \"../../Engines/thinEngine\" {\r\n export interface ThinEngine {\r\n /**\r\n * Update a dynamic index buffer\r\n * @param indexBuffer defines the target index buffer\r\n * @param indices defines the data to update\r\n * @param offset defines the offset in the target index buffer where update should start\r\n */\r\n updateDynamicIndexBuffer(indexBuffer: DataBuffer, indices: IndicesArray, offset?: number): void;\r\n\r\n /**\r\n * Updates a dynamic vertex buffer.\r\n * @param vertexBuffer the vertex buffer to update\r\n * @param data the data used to update the vertex buffer\r\n * @param byteOffset the byte offset of the data\r\n * @param byteLength the byte length of the data\r\n */\r\n updateDynamicVertexBuffer(vertexBuffer: DataBuffer, data: DataArray, byteOffset?: number, byteLength?: number): void;\r\n }\r\n}\r\n\r\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\r\nThinEngine.prototype.updateDynamicIndexBuffer = function (this: ThinEngine, indexBuffer: DataBuffer, indices: IndicesArray, offset: number = 0): void {\r\n // Force cache update\r\n this._currentBoundBuffer[this._gl.ELEMENT_ARRAY_BUFFER] = null;\r\n this.bindIndexBuffer(indexBuffer);\r\n\r\n let view: ArrayBufferView;\r\n if (indexBuffer.is32Bits) {\r\n // anything else than Uint32Array needs to be converted to Uint32Array\r\n view = indices instanceof Uint32Array ? indices : new Uint32Array(indices);\r\n } else {\r\n // anything else than Uint16Array needs to be converted to Uint16Array\r\n view = indices instanceof Uint16Array ? indices : new Uint16Array(indices);\r\n }\r\n\r\n this._gl.bufferData(this._gl.ELEMENT_ARRAY_BUFFER, view, this._gl.DYNAMIC_DRAW);\r\n\r\n this._resetIndexBufferBinding();\r\n};\r\n\r\nThinEngine.prototype.updateDynamicVertexBuffer = function (this: ThinEngine, vertexBuffer: DataBuffer, data: DataArray, byteOffset?: number, byteLength?: number): void {\r\n this.bindArrayBuffer(vertexBuffer);\r\n\r\n if (byteOffset === undefined) {\r\n byteOffset = 0;\r\n }\r\n\r\n const dataLength = (data as ArrayBuffer).byteLength || (data as number[]).length;\r\n\r\n if (byteLength === undefined || (byteLength >= dataLength && byteOffset === 0)) {\r\n if (data instanceof Array) {\r\n this._gl.bufferSubData(this._gl.ARRAY_BUFFER, byteOffset, new Float32Array(data));\r\n } else {\r\n this._gl.bufferSubData(this._gl.ARRAY_BUFFER, byteOffset, data);\r\n }\r\n } else {\r\n if (data instanceof Array) {\r\n this._gl.bufferSubData(this._gl.ARRAY_BUFFER, 0, new Float32Array(data).subarray(byteOffset, byteOffset + byteLength));\r\n } else {\r\n if (data instanceof ArrayBuffer) {\r\n data = new Uint8Array(data, byteOffset, byteLength);\r\n } else {\r\n data = new Uint8Array(data.buffer, data.byteOffset + byteOffset, byteLength);\r\n }\r\n\r\n this._gl.bufferSubData(this._gl.ARRAY_BUFFER, 0, data);\r\n }\r\n }\r\n\r\n this._resetVertexBufferBinding();\r\n};\r\n","import { Observable } from \"../Misc/observable\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { Scene } from \"../scene\";\r\nimport { InternalTexture, InternalTextureSource } from \"../Materials/Textures/internalTexture\";\r\nimport type { IOfflineProvider } from \"../Offline/IOfflineProvider\";\r\nimport type { ILoadingScreen } from \"../Loading/loadingScreen\";\r\nimport { IsDocumentAvailable, IsWindowObjectExist } from \"../Misc/domManagement\";\r\nimport { EngineStore } from \"./engineStore\";\r\nimport { _WarnImport } from \"../Misc/devTools\";\r\nimport type { WebGLPipelineContext } from \"./WebGL/webGLPipelineContext\";\r\nimport type { IPipelineContext } from \"./IPipelineContext\";\r\nimport type { ICustomAnimationFrameRequester } from \"../Misc/customAnimationFrameRequester\";\r\nimport type { EngineOptions } from \"./thinEngine\";\r\nimport { ThinEngine } from \"./thinEngine\";\r\nimport { Constants } from \"./constants\";\r\nimport type { IViewportLike, IColor4Like } from \"../Maths/math.like\";\r\nimport type { RenderTargetTexture } from \"../Materials/Textures/renderTargetTexture\";\r\nimport { PerformanceMonitor } from \"../Misc/performanceMonitor\";\r\nimport type { DataBuffer } from \"../Buffers/dataBuffer\";\r\nimport { PerfCounter } from \"../Misc/perfCounter\";\r\nimport { WebGLDataBuffer } from \"../Meshes/WebGL/webGLDataBuffer\";\r\nimport { Logger } from \"../Misc/logger\";\r\nimport type { RenderTargetWrapper } from \"./renderTargetWrapper\";\r\nimport { WebGLHardwareTexture } from \"./WebGL/webGLHardwareTexture\";\r\n\r\nimport \"./Extensions/engine.alpha\";\r\nimport \"./Extensions/engine.readTexture\";\r\nimport \"./Extensions/engine.dynamicBuffer\";\r\nimport type { IAudioEngine } from \"../Audio/Interfaces/IAudioEngine\";\r\n\r\nimport type { Material } from \"../Materials/material\";\r\nimport type { PostProcess } from \"../PostProcesses/postProcess\";\r\n\r\n/**\r\n * Defines the interface used by display changed events\r\n */\r\nexport interface IDisplayChangedEventArgs {\r\n /** Gets the vrDisplay object (if any) */\r\n vrDisplay: Nullable;\r\n /** Gets a boolean indicating if webVR is supported */\r\n vrSupported: boolean;\r\n}\r\n\r\n/**\r\n * Defines the interface used by objects containing a viewport (like a camera)\r\n */\r\ninterface IViewportOwnerLike {\r\n /**\r\n * Gets or sets the viewport\r\n */\r\n viewport: IViewportLike;\r\n}\r\n\r\n/**\r\n * The engine class is responsible for interfacing with all lower-level APIs such as WebGL and Audio\r\n */\r\nexport class Engine extends ThinEngine {\r\n // Const statics\r\n\r\n /** Defines that alpha blending is disabled */\r\n public static readonly ALPHA_DISABLE = Constants.ALPHA_DISABLE;\r\n /** Defines that alpha blending to SRC ALPHA * SRC + DEST */\r\n public static readonly ALPHA_ADD = Constants.ALPHA_ADD;\r\n /** Defines that alpha blending to SRC ALPHA * SRC + (1 - SRC ALPHA) * DEST */\r\n public static readonly ALPHA_COMBINE = Constants.ALPHA_COMBINE;\r\n /** Defines that alpha blending to DEST - SRC * DEST */\r\n public static readonly ALPHA_SUBTRACT = Constants.ALPHA_SUBTRACT;\r\n /** Defines that alpha blending to SRC * DEST */\r\n public static readonly ALPHA_MULTIPLY = Constants.ALPHA_MULTIPLY;\r\n /** Defines that alpha blending to SRC ALPHA * SRC + (1 - SRC) * DEST */\r\n public static readonly ALPHA_MAXIMIZED = Constants.ALPHA_MAXIMIZED;\r\n /** Defines that alpha blending to SRC + DEST */\r\n public static readonly ALPHA_ONEONE = Constants.ALPHA_ONEONE;\r\n /** Defines that alpha blending to SRC + (1 - SRC ALPHA) * DEST */\r\n public static readonly ALPHA_PREMULTIPLIED = Constants.ALPHA_PREMULTIPLIED;\r\n /**\r\n * Defines that alpha blending to SRC + (1 - SRC ALPHA) * DEST\r\n * Alpha will be set to (1 - SRC ALPHA) * DEST ALPHA\r\n */\r\n public static readonly ALPHA_PREMULTIPLIED_PORTERDUFF = Constants.ALPHA_PREMULTIPLIED_PORTERDUFF;\r\n /** Defines that alpha blending to CST * SRC + (1 - CST) * DEST */\r\n public static readonly ALPHA_INTERPOLATE = Constants.ALPHA_INTERPOLATE;\r\n /**\r\n * Defines that alpha blending to SRC + (1 - SRC) * DEST\r\n * Alpha will be set to SRC ALPHA + (1 - SRC ALPHA) * DEST ALPHA\r\n */\r\n public static readonly ALPHA_SCREENMODE = Constants.ALPHA_SCREENMODE;\r\n\r\n /** Defines that the resource is not delayed*/\r\n public static readonly DELAYLOADSTATE_NONE = Constants.DELAYLOADSTATE_NONE;\r\n /** Defines that the resource was successfully delay loaded */\r\n public static readonly DELAYLOADSTATE_LOADED = Constants.DELAYLOADSTATE_LOADED;\r\n /** Defines that the resource is currently delay loading */\r\n public static readonly DELAYLOADSTATE_LOADING = Constants.DELAYLOADSTATE_LOADING;\r\n /** Defines that the resource is delayed and has not started loading */\r\n public static readonly DELAYLOADSTATE_NOTLOADED = Constants.DELAYLOADSTATE_NOTLOADED;\r\n\r\n // Depht or Stencil test Constants.\r\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will never pass. i.e. Nothing will be drawn */\r\n public static readonly NEVER = Constants.NEVER;\r\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will always pass. i.e. Pixels will be drawn in the order they are drawn */\r\n public static readonly ALWAYS = Constants.ALWAYS;\r\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is less than the stored value */\r\n public static readonly LESS = Constants.LESS;\r\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is equals to the stored value */\r\n public static readonly EQUAL = Constants.EQUAL;\r\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is less than or equal to the stored value */\r\n public static readonly LEQUAL = Constants.LEQUAL;\r\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is greater than the stored value */\r\n public static readonly GREATER = Constants.GREATER;\r\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is greater than or equal to the stored value */\r\n public static readonly GEQUAL = Constants.GEQUAL;\r\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is not equal to the stored value */\r\n public static readonly NOTEQUAL = Constants.NOTEQUAL;\r\n\r\n // Stencil Actions Constants.\r\n /** Passed to stencilOperation to specify that stencil value must be kept */\r\n public static readonly KEEP = Constants.KEEP;\r\n /** Passed to stencilOperation to specify that stencil value must be replaced */\r\n public static readonly REPLACE = Constants.REPLACE;\r\n /** Passed to stencilOperation to specify that stencil value must be incremented */\r\n public static readonly INCR = Constants.INCR;\r\n /** Passed to stencilOperation to specify that stencil value must be decremented */\r\n public static readonly DECR = Constants.DECR;\r\n /** Passed to stencilOperation to specify that stencil value must be inverted */\r\n public static readonly INVERT = Constants.INVERT;\r\n /** Passed to stencilOperation to specify that stencil value must be incremented with wrapping */\r\n public static readonly INCR_WRAP = Constants.INCR_WRAP;\r\n /** Passed to stencilOperation to specify that stencil value must be decremented with wrapping */\r\n public static readonly DECR_WRAP = Constants.DECR_WRAP;\r\n\r\n /** Texture is not repeating outside of 0..1 UVs */\r\n public static readonly TEXTURE_CLAMP_ADDRESSMODE = Constants.TEXTURE_CLAMP_ADDRESSMODE;\r\n /** Texture is repeating outside of 0..1 UVs */\r\n public static readonly TEXTURE_WRAP_ADDRESSMODE = Constants.TEXTURE_WRAP_ADDRESSMODE;\r\n /** Texture is repeating and mirrored */\r\n public static readonly TEXTURE_MIRROR_ADDRESSMODE = Constants.TEXTURE_MIRROR_ADDRESSMODE;\r\n\r\n /** ALPHA */\r\n public static readonly TEXTUREFORMAT_ALPHA = Constants.TEXTUREFORMAT_ALPHA;\r\n /** LUMINANCE */\r\n public static readonly TEXTUREFORMAT_LUMINANCE = Constants.TEXTUREFORMAT_LUMINANCE;\r\n /** LUMINANCE_ALPHA */\r\n public static readonly TEXTUREFORMAT_LUMINANCE_ALPHA = Constants.TEXTUREFORMAT_LUMINANCE_ALPHA;\r\n /** RGB */\r\n public static readonly TEXTUREFORMAT_RGB = Constants.TEXTUREFORMAT_RGB;\r\n /** RGBA */\r\n public static readonly TEXTUREFORMAT_RGBA = Constants.TEXTUREFORMAT_RGBA;\r\n /** RED */\r\n public static readonly TEXTUREFORMAT_RED = Constants.TEXTUREFORMAT_RED;\r\n /** RED (2nd reference) */\r\n public static readonly TEXTUREFORMAT_R = Constants.TEXTUREFORMAT_R;\r\n /** RG */\r\n public static readonly TEXTUREFORMAT_RG = Constants.TEXTUREFORMAT_RG;\r\n /** RED_INTEGER */\r\n public static readonly TEXTUREFORMAT_RED_INTEGER = Constants.TEXTUREFORMAT_RED_INTEGER;\r\n /** RED_INTEGER (2nd reference) */\r\n public static readonly TEXTUREFORMAT_R_INTEGER = Constants.TEXTUREFORMAT_R_INTEGER;\r\n /** RG_INTEGER */\r\n public static readonly TEXTUREFORMAT_RG_INTEGER = Constants.TEXTUREFORMAT_RG_INTEGER;\r\n /** RGB_INTEGER */\r\n public static readonly TEXTUREFORMAT_RGB_INTEGER = Constants.TEXTUREFORMAT_RGB_INTEGER;\r\n /** RGBA_INTEGER */\r\n public static readonly TEXTUREFORMAT_RGBA_INTEGER = Constants.TEXTUREFORMAT_RGBA_INTEGER;\r\n\r\n /** UNSIGNED_BYTE */\r\n public static readonly TEXTURETYPE_UNSIGNED_BYTE = Constants.TEXTURETYPE_UNSIGNED_BYTE;\r\n /** UNSIGNED_BYTE (2nd reference) */\r\n public static readonly TEXTURETYPE_UNSIGNED_INT = Constants.TEXTURETYPE_UNSIGNED_INT;\r\n /** FLOAT */\r\n public static readonly TEXTURETYPE_FLOAT = Constants.TEXTURETYPE_FLOAT;\r\n /** HALF_FLOAT */\r\n public static readonly TEXTURETYPE_HALF_FLOAT = Constants.TEXTURETYPE_HALF_FLOAT;\r\n /** BYTE */\r\n public static readonly TEXTURETYPE_BYTE = Constants.TEXTURETYPE_BYTE;\r\n /** SHORT */\r\n public static readonly TEXTURETYPE_SHORT = Constants.TEXTURETYPE_SHORT;\r\n /** UNSIGNED_SHORT */\r\n public static readonly TEXTURETYPE_UNSIGNED_SHORT = Constants.TEXTURETYPE_UNSIGNED_SHORT;\r\n /** INT */\r\n public static readonly TEXTURETYPE_INT = Constants.TEXTURETYPE_INT;\r\n /** UNSIGNED_INT */\r\n public static readonly TEXTURETYPE_UNSIGNED_INTEGER = Constants.TEXTURETYPE_UNSIGNED_INTEGER;\r\n /** UNSIGNED_SHORT_4_4_4_4 */\r\n public static readonly TEXTURETYPE_UNSIGNED_SHORT_4_4_4_4 = Constants.TEXTURETYPE_UNSIGNED_SHORT_4_4_4_4;\r\n /** UNSIGNED_SHORT_5_5_5_1 */\r\n public static readonly TEXTURETYPE_UNSIGNED_SHORT_5_5_5_1 = Constants.TEXTURETYPE_UNSIGNED_SHORT_5_5_5_1;\r\n /** UNSIGNED_SHORT_5_6_5 */\r\n public static readonly TEXTURETYPE_UNSIGNED_SHORT_5_6_5 = Constants.TEXTURETYPE_UNSIGNED_SHORT_5_6_5;\r\n /** UNSIGNED_INT_2_10_10_10_REV */\r\n public static readonly TEXTURETYPE_UNSIGNED_INT_2_10_10_10_REV = Constants.TEXTURETYPE_UNSIGNED_INT_2_10_10_10_REV;\r\n /** UNSIGNED_INT_24_8 */\r\n public static readonly TEXTURETYPE_UNSIGNED_INT_24_8 = Constants.TEXTURETYPE_UNSIGNED_INT_24_8;\r\n /** UNSIGNED_INT_10F_11F_11F_REV */\r\n public static readonly TEXTURETYPE_UNSIGNED_INT_10F_11F_11F_REV = Constants.TEXTURETYPE_UNSIGNED_INT_10F_11F_11F_REV;\r\n /** UNSIGNED_INT_5_9_9_9_REV */\r\n public static readonly TEXTURETYPE_UNSIGNED_INT_5_9_9_9_REV = Constants.TEXTURETYPE_UNSIGNED_INT_5_9_9_9_REV;\r\n /** FLOAT_32_UNSIGNED_INT_24_8_REV */\r\n public static readonly TEXTURETYPE_FLOAT_32_UNSIGNED_INT_24_8_REV = Constants.TEXTURETYPE_FLOAT_32_UNSIGNED_INT_24_8_REV;\r\n\r\n /** nearest is mag = nearest and min = nearest and mip = none */\r\n public static readonly TEXTURE_NEAREST_SAMPLINGMODE = Constants.TEXTURE_NEAREST_SAMPLINGMODE;\r\n /** Bilinear is mag = linear and min = linear and mip = nearest */\r\n public static readonly TEXTURE_BILINEAR_SAMPLINGMODE = Constants.TEXTURE_BILINEAR_SAMPLINGMODE;\r\n /** Trilinear is mag = linear and min = linear and mip = linear */\r\n public static readonly TEXTURE_TRILINEAR_SAMPLINGMODE = Constants.TEXTURE_TRILINEAR_SAMPLINGMODE;\r\n /** nearest is mag = nearest and min = nearest and mip = linear */\r\n public static readonly TEXTURE_NEAREST_NEAREST_MIPLINEAR = Constants.TEXTURE_NEAREST_NEAREST_MIPLINEAR;\r\n /** Bilinear is mag = linear and min = linear and mip = nearest */\r\n public static readonly TEXTURE_LINEAR_LINEAR_MIPNEAREST = Constants.TEXTURE_LINEAR_LINEAR_MIPNEAREST;\r\n /** Trilinear is mag = linear and min = linear and mip = linear */\r\n public static readonly TEXTURE_LINEAR_LINEAR_MIPLINEAR = Constants.TEXTURE_LINEAR_LINEAR_MIPLINEAR;\r\n /** mag = nearest and min = nearest and mip = nearest */\r\n public static readonly TEXTURE_NEAREST_NEAREST_MIPNEAREST = Constants.TEXTURE_NEAREST_NEAREST_MIPNEAREST;\r\n /** mag = nearest and min = linear and mip = nearest */\r\n public static readonly TEXTURE_NEAREST_LINEAR_MIPNEAREST = Constants.TEXTURE_NEAREST_LINEAR_MIPNEAREST;\r\n /** mag = nearest and min = linear and mip = linear */\r\n public static readonly TEXTURE_NEAREST_LINEAR_MIPLINEAR = Constants.TEXTURE_NEAREST_LINEAR_MIPLINEAR;\r\n /** mag = nearest and min = linear and mip = none */\r\n public static readonly TEXTURE_NEAREST_LINEAR = Constants.TEXTURE_NEAREST_LINEAR;\r\n /** mag = nearest and min = nearest and mip = none */\r\n public static readonly TEXTURE_NEAREST_NEAREST = Constants.TEXTURE_NEAREST_NEAREST;\r\n /** mag = linear and min = nearest and mip = nearest */\r\n public static readonly TEXTURE_LINEAR_NEAREST_MIPNEAREST = Constants.TEXTURE_LINEAR_NEAREST_MIPNEAREST;\r\n /** mag = linear and min = nearest and mip = linear */\r\n public static readonly TEXTURE_LINEAR_NEAREST_MIPLINEAR = Constants.TEXTURE_LINEAR_NEAREST_MIPLINEAR;\r\n /** mag = linear and min = linear and mip = none */\r\n public static readonly TEXTURE_LINEAR_LINEAR = Constants.TEXTURE_LINEAR_LINEAR;\r\n /** mag = linear and min = nearest and mip = none */\r\n public static readonly TEXTURE_LINEAR_NEAREST = Constants.TEXTURE_LINEAR_NEAREST;\r\n\r\n /** Explicit coordinates mode */\r\n public static readonly TEXTURE_EXPLICIT_MODE = Constants.TEXTURE_EXPLICIT_MODE;\r\n /** Spherical coordinates mode */\r\n public static readonly TEXTURE_SPHERICAL_MODE = Constants.TEXTURE_SPHERICAL_MODE;\r\n /** Planar coordinates mode */\r\n public static readonly TEXTURE_PLANAR_MODE = Constants.TEXTURE_PLANAR_MODE;\r\n /** Cubic coordinates mode */\r\n public static readonly TEXTURE_CUBIC_MODE = Constants.TEXTURE_CUBIC_MODE;\r\n /** Projection coordinates mode */\r\n public static readonly TEXTURE_PROJECTION_MODE = Constants.TEXTURE_PROJECTION_MODE;\r\n /** Skybox coordinates mode */\r\n public static readonly TEXTURE_SKYBOX_MODE = Constants.TEXTURE_SKYBOX_MODE;\r\n /** Inverse Cubic coordinates mode */\r\n public static readonly TEXTURE_INVCUBIC_MODE = Constants.TEXTURE_INVCUBIC_MODE;\r\n /** Equirectangular coordinates mode */\r\n public static readonly TEXTURE_EQUIRECTANGULAR_MODE = Constants.TEXTURE_EQUIRECTANGULAR_MODE;\r\n /** Equirectangular Fixed coordinates mode */\r\n public static readonly TEXTURE_FIXED_EQUIRECTANGULAR_MODE = Constants.TEXTURE_FIXED_EQUIRECTANGULAR_MODE;\r\n /** Equirectangular Fixed Mirrored coordinates mode */\r\n public static readonly TEXTURE_FIXED_EQUIRECTANGULAR_MIRRORED_MODE = Constants.TEXTURE_FIXED_EQUIRECTANGULAR_MIRRORED_MODE;\r\n\r\n // Texture rescaling mode\r\n /** Defines that texture rescaling will use a floor to find the closer power of 2 size */\r\n public static readonly SCALEMODE_FLOOR = Constants.SCALEMODE_FLOOR;\r\n /** Defines that texture rescaling will look for the nearest power of 2 size */\r\n public static readonly SCALEMODE_NEAREST = Constants.SCALEMODE_NEAREST;\r\n /** Defines that texture rescaling will use a ceil to find the closer power of 2 size */\r\n public static readonly SCALEMODE_CEILING = Constants.SCALEMODE_CEILING;\r\n\r\n /**\r\n * Returns the current npm package of the sdk\r\n */\r\n // Not mixed with Version for tooling purpose.\r\n public static get NpmPackage(): string {\r\n return ThinEngine.NpmPackage;\r\n }\r\n\r\n /**\r\n * Returns the current version of the framework\r\n */\r\n public static get Version(): string {\r\n return ThinEngine.Version;\r\n }\r\n\r\n /** Gets the list of created engines */\r\n public static get Instances(): Engine[] {\r\n return EngineStore.Instances;\r\n }\r\n\r\n /**\r\n * Gets the latest created engine\r\n */\r\n public static get LastCreatedEngine(): Nullable {\r\n return EngineStore.LastCreatedEngine;\r\n }\r\n\r\n /**\r\n * Gets the latest created scene\r\n */\r\n public static get LastCreatedScene(): Nullable {\r\n return EngineStore.LastCreatedScene;\r\n }\r\n\r\n /** @internal */\r\n /**\r\n * Engine abstraction for loading and creating an image bitmap from a given source string.\r\n * @param imageSource source to load the image from.\r\n * @param options An object that sets options for the image's extraction.\r\n * @returns ImageBitmap.\r\n */\r\n public _createImageBitmapFromSource(imageSource: string, options?: ImageBitmapOptions): Promise {\r\n const promise = new Promise((resolve, reject) => {\r\n const image = new Image();\r\n image.onload = () => {\r\n image.decode().then(() => {\r\n this.createImageBitmap(image, options).then((imageBitmap) => {\r\n resolve(imageBitmap);\r\n });\r\n });\r\n };\r\n image.onerror = () => {\r\n reject(`Error loading image ${image.src}`);\r\n };\r\n\r\n image.src = imageSource;\r\n });\r\n\r\n return promise;\r\n }\r\n\r\n /**\r\n * Engine abstraction for createImageBitmap\r\n * @param image source for image\r\n * @param options An object that sets options for the image's extraction.\r\n * @returns ImageBitmap\r\n */\r\n public createImageBitmap(image: ImageBitmapSource, options?: ImageBitmapOptions): Promise {\r\n return createImageBitmap(image, options);\r\n }\r\n\r\n /**\r\n * Resize an image and returns the image data as an uint8array\r\n * @param image image to resize\r\n * @param bufferWidth destination buffer width\r\n * @param bufferHeight destination buffer height\r\n * @returns an uint8array containing RGBA values of bufferWidth * bufferHeight size\r\n */\r\n public resizeImageBitmap(image: HTMLImageElement | ImageBitmap, bufferWidth: number, bufferHeight: number): Uint8Array {\r\n const canvas = this.createCanvas(bufferWidth, bufferHeight);\r\n const context = canvas.getContext(\"2d\");\r\n\r\n if (!context) {\r\n throw new Error(\"Unable to get 2d context for resizeImageBitmap\");\r\n }\r\n\r\n context.drawImage(image, 0, 0);\r\n\r\n // Create VertexData from map data\r\n // Cast is due to wrong definition in lib.d.ts from ts 1.3 - https://github.com/Microsoft/TypeScript/issues/949\r\n const buffer = (context.getImageData(0, 0, bufferWidth, bufferHeight).data);\r\n return buffer;\r\n }\r\n\r\n /**\r\n * Will flag all materials in all scenes in all engines as dirty to trigger new shader compilation\r\n * @param flag defines which part of the materials must be marked as dirty\r\n * @param predicate defines a predicate used to filter which materials should be affected\r\n */\r\n public static MarkAllMaterialsAsDirty(flag: number, predicate?: (mat: Material) => boolean): void {\r\n for (let engineIndex = 0; engineIndex < Engine.Instances.length; engineIndex++) {\r\n const engine = Engine.Instances[engineIndex];\r\n\r\n for (let sceneIndex = 0; sceneIndex < engine.scenes.length; sceneIndex++) {\r\n engine.scenes[sceneIndex].markAllMaterialsAsDirty(flag, predicate);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Method called to create the default loading screen.\r\n * This can be overridden in your own app.\r\n * @param canvas The rendering canvas element\r\n * @returns The loading screen\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public static DefaultLoadingScreenFactory(canvas: HTMLCanvasElement): ILoadingScreen {\r\n throw _WarnImport(\"LoadingScreen\");\r\n }\r\n\r\n /**\r\n * Method called to create the default rescale post process on each engine.\r\n */\r\n public static _RescalePostProcessFactory: Nullable<(engine: Engine) => PostProcess> = null;\r\n\r\n // Members\r\n\r\n /**\r\n * Gets or sets a boolean to enable/disable IndexedDB support and avoid XHR on .manifest\r\n **/\r\n public enableOfflineSupport = false;\r\n\r\n /**\r\n * Gets or sets a boolean to enable/disable checking manifest if IndexedDB support is enabled (js will always consider the database is up to date)\r\n **/\r\n public disableManifestCheck = false;\r\n\r\n /**\r\n * Gets or sets a boolean to enable/disable the context menu (right-click) from appearing on the main canvas\r\n */\r\n public disableContextMenu: boolean = true;\r\n\r\n /**\r\n * Gets the list of created scenes\r\n */\r\n public scenes = new Array();\r\n\r\n /** @internal */\r\n public _virtualScenes = new Array();\r\n\r\n /**\r\n * Event raised when a new scene is created\r\n */\r\n public onNewSceneAddedObservable = new Observable();\r\n\r\n /**\r\n * Gets the list of created postprocesses\r\n */\r\n public postProcesses = new Array();\r\n\r\n /**\r\n * Gets a boolean indicating if the pointer is currently locked\r\n */\r\n public isPointerLock = false;\r\n\r\n // Observables\r\n\r\n /**\r\n * Observable event triggered each time the rendering canvas is resized\r\n */\r\n public onResizeObservable = new Observable();\r\n\r\n /**\r\n * Observable event triggered each time the canvas loses focus\r\n */\r\n public onCanvasBlurObservable = new Observable();\r\n\r\n /**\r\n * Observable event triggered each time the canvas gains focus\r\n */\r\n public onCanvasFocusObservable = new Observable();\r\n\r\n /**\r\n * Observable event triggered each time the canvas receives pointerout event\r\n */\r\n public onCanvasPointerOutObservable = new Observable();\r\n\r\n /**\r\n * Observable raised when the engine begins a new frame\r\n */\r\n public onBeginFrameObservable = new Observable();\r\n\r\n /**\r\n * If set, will be used to request the next animation frame for the render loop\r\n */\r\n public customAnimationFrameRequester: Nullable = null;\r\n\r\n /**\r\n * Observable raised when the engine ends the current frame\r\n */\r\n public onEndFrameObservable = new Observable();\r\n\r\n /**\r\n * Observable raised when the engine is about to compile a shader\r\n */\r\n public onBeforeShaderCompilationObservable = new Observable();\r\n\r\n /**\r\n * Observable raised when the engine has just compiled a shader\r\n */\r\n public onAfterShaderCompilationObservable = new Observable();\r\n\r\n /**\r\n * Gets the audio engine\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/audio/playingSoundsMusic\r\n * @ignorenaming\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public static audioEngine: Nullable;\r\n\r\n /**\r\n * Default AudioEngine factory responsible of creating the Audio Engine.\r\n * By default, this will create a BabylonJS Audio Engine if the workload has been embedded.\r\n */\r\n public static AudioEngineFactory: (\r\n hostElement: Nullable,\r\n audioContext: Nullable,\r\n audioDestination: Nullable\r\n ) => IAudioEngine;\r\n\r\n /**\r\n * Default offline support factory responsible of creating a tool used to store data locally.\r\n * By default, this will create a Database object if the workload has been embedded.\r\n */\r\n public static OfflineProviderFactory: (urlToScene: string, callbackManifestChecked: (checked: boolean) => any, disableManifestCheck: boolean) => IOfflineProvider;\r\n\r\n private _loadingScreen: ILoadingScreen;\r\n private _pointerLockRequested: boolean;\r\n private _rescalePostProcess: Nullable;\r\n\r\n // Deterministic lockstepMaxSteps\r\n protected _deterministicLockstep: boolean = false;\r\n protected _lockstepMaxSteps: number = 4;\r\n protected _timeStep: number = 1 / 60;\r\n\r\n protected get _supportsHardwareTextureRescaling() {\r\n return !!Engine._RescalePostProcessFactory;\r\n }\r\n\r\n // FPS\r\n private _fps = 60;\r\n private _deltaTime = 0;\r\n\r\n /** @internal */\r\n public _drawCalls = new PerfCounter();\r\n\r\n /** Gets or sets the tab index to set to the rendering canvas. 1 is the minimum value to set to be able to capture keyboard events */\r\n public canvasTabIndex = 1;\r\n\r\n /**\r\n * Turn this value on if you want to pause FPS computation when in background\r\n */\r\n public disablePerformanceMonitorInBackground = false;\r\n\r\n private _performanceMonitor = new PerformanceMonitor();\r\n /**\r\n * Gets the performance monitor attached to this engine\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimize_your_scene#engineinstrumentation\r\n */\r\n public get performanceMonitor(): PerformanceMonitor {\r\n return this._performanceMonitor;\r\n }\r\n\r\n // Focus\r\n private _onFocus: () => void;\r\n private _onBlur: () => void;\r\n private _onCanvasPointerOut: (event: PointerEvent) => void;\r\n private _onCanvasBlur: () => void;\r\n private _onCanvasFocus: () => void;\r\n private _onCanvasContextMenu: (evt: Event) => void;\r\n\r\n private _onFullscreenChange: () => void;\r\n private _onPointerLockChange: () => void;\r\n\r\n protected _compatibilityMode = true;\r\n\r\n /**\r\n * (WebGPU only) True (default) to be in compatibility mode, meaning rendering all existing scenes without artifacts (same rendering than WebGL).\r\n * Setting the property to false will improve performances but may not work in some scenes if some precautions are not taken.\r\n * See https://doc.babylonjs.com/setup/support/webGPU/webGPUOptimization/webGPUNonCompatibilityMode for more details\r\n */\r\n public get compatibilityMode() {\r\n return this._compatibilityMode;\r\n }\r\n\r\n public set compatibilityMode(mode: boolean) {\r\n // not supported in WebGL\r\n this._compatibilityMode = true;\r\n }\r\n\r\n // Events\r\n\r\n /**\r\n * Gets the HTML element used to attach event listeners\r\n * @returns a HTML element\r\n */\r\n public getInputElement(): Nullable {\r\n return this._renderingCanvas;\r\n }\r\n\r\n /**\r\n * Creates a new engine\r\n * @param canvasOrContext defines the canvas or WebGL context to use for rendering. If you provide a WebGL context, Babylon.js will not hook events on the canvas (like pointers, keyboards, etc...) so no event observables will be available. This is mostly used when Babylon.js is used as a plugin on a system which already used the WebGL context\r\n * @param antialias defines enable antialiasing (default: false)\r\n * @param options defines further options to be sent to the getContext() function\r\n * @param adaptToDeviceRatio defines whether to adapt to the device's viewport characteristics (default: false)\r\n */\r\n constructor(\r\n canvasOrContext: Nullable,\r\n antialias?: boolean,\r\n options?: EngineOptions,\r\n adaptToDeviceRatio: boolean = false\r\n ) {\r\n super(canvasOrContext, antialias, options, adaptToDeviceRatio);\r\n\r\n Engine.Instances.push(this);\r\n\r\n if (!canvasOrContext) {\r\n return;\r\n }\r\n\r\n this._features.supportRenderPasses = true;\r\n\r\n options = this._creationOptions;\r\n\r\n if ((canvasOrContext).getContext) {\r\n const canvas = canvasOrContext;\r\n\r\n this._sharedInit(canvas);\r\n\r\n this._connectVREvents();\r\n }\r\n\r\n // Load WebVR Devices\r\n this._prepareVRComponent();\r\n if (options.autoEnableWebVR) {\r\n this.initWebVR();\r\n }\r\n }\r\n\r\n protected _initGLContext(): void {\r\n super._initGLContext();\r\n\r\n this._rescalePostProcess = null;\r\n }\r\n\r\n /**\r\n * Shared initialization across engines types.\r\n * @param canvas The canvas associated with this instance of the engine.\r\n */\r\n protected _sharedInit(canvas: HTMLCanvasElement) {\r\n super._sharedInit(canvas);\r\n\r\n this._onCanvasFocus = () => {\r\n this.onCanvasFocusObservable.notifyObservers(this);\r\n };\r\n\r\n this._onCanvasBlur = () => {\r\n this.onCanvasBlurObservable.notifyObservers(this);\r\n };\r\n\r\n this._onCanvasContextMenu = (evt: Event) => {\r\n if (this.disableContextMenu) {\r\n evt.preventDefault();\r\n }\r\n };\r\n\r\n canvas.addEventListener(\"focus\", this._onCanvasFocus);\r\n canvas.addEventListener(\"blur\", this._onCanvasBlur);\r\n canvas.addEventListener(\"contextmenu\", this._onCanvasContextMenu);\r\n\r\n this._onBlur = () => {\r\n if (this.disablePerformanceMonitorInBackground) {\r\n this._performanceMonitor.disable();\r\n }\r\n this._windowIsBackground = true;\r\n };\r\n\r\n this._onFocus = () => {\r\n if (this.disablePerformanceMonitorInBackground) {\r\n this._performanceMonitor.enable();\r\n }\r\n this._windowIsBackground = false;\r\n };\r\n\r\n this._onCanvasPointerOut = (ev) => {\r\n // Check that the element at the point of the pointer out isn't the canvas and if it isn't, notify observers\r\n // Note: This is a workaround for a bug with Safari\r\n if (document.elementFromPoint(ev.clientX, ev.clientY) !== canvas) {\r\n this.onCanvasPointerOutObservable.notifyObservers(ev);\r\n }\r\n };\r\n\r\n const hostWindow = this.getHostWindow(); // it calls IsWindowObjectExist()\r\n if (hostWindow && typeof hostWindow.addEventListener === \"function\") {\r\n hostWindow.addEventListener(\"blur\", this._onBlur);\r\n hostWindow.addEventListener(\"focus\", this._onFocus);\r\n }\r\n\r\n canvas.addEventListener(\"pointerout\", this._onCanvasPointerOut);\r\n\r\n if (!this._creationOptions.doNotHandleTouchAction) {\r\n this._disableTouchAction();\r\n }\r\n\r\n // Create Audio Engine if needed.\r\n if (!Engine.audioEngine && this._creationOptions.audioEngine && Engine.AudioEngineFactory) {\r\n Engine.audioEngine = Engine.AudioEngineFactory(this.getRenderingCanvas(), this.getAudioContext(), this.getAudioDestination());\r\n }\r\n if (IsDocumentAvailable()) {\r\n // Fullscreen\r\n this._onFullscreenChange = () => {\r\n this.isFullscreen = !!document.fullscreenElement;\r\n\r\n // Pointer lock\r\n if (this.isFullscreen && this._pointerLockRequested && canvas) {\r\n Engine._RequestPointerlock(canvas);\r\n }\r\n };\r\n\r\n document.addEventListener(\"fullscreenchange\", this._onFullscreenChange, false);\r\n document.addEventListener(\"webkitfullscreenchange\", this._onFullscreenChange, false);\r\n\r\n // Pointer lock\r\n this._onPointerLockChange = () => {\r\n this.isPointerLock = document.pointerLockElement === canvas;\r\n };\r\n\r\n document.addEventListener(\"pointerlockchange\", this._onPointerLockChange, false);\r\n document.addEventListener(\"webkitpointerlockchange\", this._onPointerLockChange, false);\r\n }\r\n\r\n this.enableOfflineSupport = Engine.OfflineProviderFactory !== undefined;\r\n\r\n this._deterministicLockstep = !!this._creationOptions.deterministicLockstep;\r\n this._lockstepMaxSteps = this._creationOptions.lockstepMaxSteps || 0;\r\n this._timeStep = this._creationOptions.timeStep || 1 / 60;\r\n }\r\n\r\n /** @internal */\r\n public _verifyPointerLock(): void {\r\n this._onPointerLockChange?.();\r\n }\r\n\r\n /**\r\n * Gets current aspect ratio\r\n * @param viewportOwner defines the camera to use to get the aspect ratio\r\n * @param useScreen defines if screen size must be used (or the current render target if any)\r\n * @returns a number defining the aspect ratio\r\n */\r\n public getAspectRatio(viewportOwner: IViewportOwnerLike, useScreen = false): number {\r\n const viewport = viewportOwner.viewport;\r\n return (this.getRenderWidth(useScreen) * viewport.width) / (this.getRenderHeight(useScreen) * viewport.height);\r\n }\r\n\r\n /**\r\n * Gets current screen aspect ratio\r\n * @returns a number defining the aspect ratio\r\n */\r\n public getScreenAspectRatio(): number {\r\n return this.getRenderWidth(true) / this.getRenderHeight(true);\r\n }\r\n\r\n /**\r\n * Gets the client rect of the HTML canvas attached with the current webGL context\r\n * @returns a client rectangle\r\n */\r\n public getRenderingCanvasClientRect(): Nullable {\r\n if (!this._renderingCanvas) {\r\n return null;\r\n }\r\n return this._renderingCanvas.getBoundingClientRect();\r\n }\r\n\r\n /**\r\n * Gets the client rect of the HTML element used for events\r\n * @returns a client rectangle\r\n */\r\n public getInputElementClientRect(): Nullable {\r\n if (!this._renderingCanvas) {\r\n return null;\r\n }\r\n return this.getInputElement()!.getBoundingClientRect();\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating that the engine is running in deterministic lock step mode\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep\r\n * @returns true if engine is in deterministic lock step mode\r\n */\r\n public isDeterministicLockStep(): boolean {\r\n return this._deterministicLockstep;\r\n }\r\n\r\n /**\r\n * Gets the max steps when engine is running in deterministic lock step\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#deterministic-lockstep\r\n * @returns the max steps\r\n */\r\n public getLockstepMaxSteps(): number {\r\n return this._lockstepMaxSteps;\r\n }\r\n\r\n /**\r\n * Returns the time in ms between steps when using deterministic lock step.\r\n * @returns time step in (ms)\r\n */\r\n public getTimeStep(): number {\r\n return this._timeStep * 1000;\r\n }\r\n\r\n /**\r\n * Force the mipmap generation for the given render target texture\r\n * @param texture defines the render target texture to use\r\n * @param unbind defines whether or not to unbind the texture after generation. Defaults to true.\r\n */\r\n public generateMipMapsForCubemap(texture: InternalTexture, unbind = true) {\r\n if (texture.generateMipMaps) {\r\n const gl = this._gl;\r\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, texture, true);\r\n gl.generateMipmap(gl.TEXTURE_CUBE_MAP);\r\n if (unbind) {\r\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, null);\r\n }\r\n }\r\n }\r\n\r\n /** States */\r\n\r\n /**\r\n * Gets a boolean indicating if depth writing is enabled\r\n * @returns the current depth writing state\r\n */\r\n public getDepthWrite(): boolean {\r\n return this._depthCullingState.depthMask;\r\n }\r\n\r\n /**\r\n * Enable or disable depth writing\r\n * @param enable defines the state to set\r\n */\r\n public setDepthWrite(enable: boolean): void {\r\n this._depthCullingState.depthMask = enable;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if stencil buffer is enabled\r\n * @returns the current stencil buffer state\r\n */\r\n public getStencilBuffer(): boolean {\r\n return this._stencilState.stencilTest;\r\n }\r\n\r\n /**\r\n * Enable or disable the stencil buffer\r\n * @param enable defines if the stencil buffer must be enabled or disabled\r\n */\r\n public setStencilBuffer(enable: boolean): void {\r\n this._stencilState.stencilTest = enable;\r\n }\r\n\r\n /**\r\n * Gets the current stencil mask\r\n * @returns a number defining the new stencil mask to use\r\n */\r\n public getStencilMask(): number {\r\n return this._stencilState.stencilMask;\r\n }\r\n\r\n /**\r\n * Sets the current stencil mask\r\n * @param mask defines the new stencil mask to use\r\n */\r\n public setStencilMask(mask: number): void {\r\n this._stencilState.stencilMask = mask;\r\n }\r\n\r\n /**\r\n * Gets the current stencil function\r\n * @returns a number defining the stencil function to use\r\n */\r\n public getStencilFunction(): number {\r\n return this._stencilState.stencilFunc;\r\n }\r\n\r\n /**\r\n * Gets the current stencil reference value\r\n * @returns a number defining the stencil reference value to use\r\n */\r\n public getStencilFunctionReference(): number {\r\n return this._stencilState.stencilFuncRef;\r\n }\r\n\r\n /**\r\n * Gets the current stencil mask\r\n * @returns a number defining the stencil mask to use\r\n */\r\n public getStencilFunctionMask(): number {\r\n return this._stencilState.stencilFuncMask;\r\n }\r\n\r\n /**\r\n * Sets the current stencil function\r\n * @param stencilFunc defines the new stencil function to use\r\n */\r\n public setStencilFunction(stencilFunc: number) {\r\n this._stencilState.stencilFunc = stencilFunc;\r\n }\r\n\r\n /**\r\n * Sets the current stencil reference\r\n * @param reference defines the new stencil reference to use\r\n */\r\n public setStencilFunctionReference(reference: number) {\r\n this._stencilState.stencilFuncRef = reference;\r\n }\r\n\r\n /**\r\n * Sets the current stencil mask\r\n * @param mask defines the new stencil mask to use\r\n */\r\n public setStencilFunctionMask(mask: number) {\r\n this._stencilState.stencilFuncMask = mask;\r\n }\r\n\r\n /**\r\n * Gets the current stencil operation when stencil fails\r\n * @returns a number defining stencil operation to use when stencil fails\r\n */\r\n public getStencilOperationFail(): number {\r\n return this._stencilState.stencilOpStencilFail;\r\n }\r\n\r\n /**\r\n * Gets the current stencil operation when depth fails\r\n * @returns a number defining stencil operation to use when depth fails\r\n */\r\n public getStencilOperationDepthFail(): number {\r\n return this._stencilState.stencilOpDepthFail;\r\n }\r\n\r\n /**\r\n * Gets the current stencil operation when stencil passes\r\n * @returns a number defining stencil operation to use when stencil passes\r\n */\r\n public getStencilOperationPass(): number {\r\n return this._stencilState.stencilOpStencilDepthPass;\r\n }\r\n\r\n /**\r\n * Sets the stencil operation to use when stencil fails\r\n * @param operation defines the stencil operation to use when stencil fails\r\n */\r\n public setStencilOperationFail(operation: number): void {\r\n this._stencilState.stencilOpStencilFail = operation;\r\n }\r\n\r\n /**\r\n * Sets the stencil operation to use when depth fails\r\n * @param operation defines the stencil operation to use when depth fails\r\n */\r\n public setStencilOperationDepthFail(operation: number): void {\r\n this._stencilState.stencilOpDepthFail = operation;\r\n }\r\n\r\n /**\r\n * Sets the stencil operation to use when stencil passes\r\n * @param operation defines the stencil operation to use when stencil passes\r\n */\r\n public setStencilOperationPass(operation: number): void {\r\n this._stencilState.stencilOpStencilDepthPass = operation;\r\n }\r\n\r\n /**\r\n * Sets a boolean indicating if the dithering state is enabled or disabled\r\n * @param value defines the dithering state\r\n */\r\n public setDitheringState(value: boolean): void {\r\n if (value) {\r\n this._gl.enable(this._gl.DITHER);\r\n } else {\r\n this._gl.disable(this._gl.DITHER);\r\n }\r\n }\r\n\r\n /**\r\n * Sets a boolean indicating if the rasterizer state is enabled or disabled\r\n * @param value defines the rasterizer state\r\n */\r\n public setRasterizerState(value: boolean): void {\r\n if (value) {\r\n this._gl.disable(this._gl.RASTERIZER_DISCARD);\r\n } else {\r\n this._gl.enable(this._gl.RASTERIZER_DISCARD);\r\n }\r\n }\r\n\r\n /**\r\n * Gets the current depth function\r\n * @returns a number defining the depth function\r\n */\r\n public getDepthFunction(): Nullable {\r\n return this._depthCullingState.depthFunc;\r\n }\r\n\r\n /**\r\n * Sets the current depth function\r\n * @param depthFunc defines the function to use\r\n */\r\n public setDepthFunction(depthFunc: number) {\r\n this._depthCullingState.depthFunc = depthFunc;\r\n }\r\n\r\n /**\r\n * Sets the current depth function to GREATER\r\n */\r\n public setDepthFunctionToGreater(): void {\r\n this.setDepthFunction(Constants.GREATER);\r\n }\r\n\r\n /**\r\n * Sets the current depth function to GEQUAL\r\n */\r\n public setDepthFunctionToGreaterOrEqual(): void {\r\n this.setDepthFunction(Constants.GEQUAL);\r\n }\r\n\r\n /**\r\n * Sets the current depth function to LESS\r\n */\r\n public setDepthFunctionToLess(): void {\r\n this.setDepthFunction(Constants.LESS);\r\n }\r\n\r\n /**\r\n * Sets the current depth function to LEQUAL\r\n */\r\n public setDepthFunctionToLessOrEqual(): void {\r\n this.setDepthFunction(Constants.LEQUAL);\r\n }\r\n\r\n private _cachedStencilBuffer: boolean;\r\n private _cachedStencilFunction: number;\r\n private _cachedStencilMask: number;\r\n private _cachedStencilOperationPass: number;\r\n private _cachedStencilOperationFail: number;\r\n private _cachedStencilOperationDepthFail: number;\r\n private _cachedStencilReference: number;\r\n\r\n /**\r\n * Caches the the state of the stencil buffer\r\n */\r\n public cacheStencilState() {\r\n this._cachedStencilBuffer = this.getStencilBuffer();\r\n this._cachedStencilFunction = this.getStencilFunction();\r\n this._cachedStencilMask = this.getStencilMask();\r\n this._cachedStencilOperationPass = this.getStencilOperationPass();\r\n this._cachedStencilOperationFail = this.getStencilOperationFail();\r\n this._cachedStencilOperationDepthFail = this.getStencilOperationDepthFail();\r\n this._cachedStencilReference = this.getStencilFunctionReference();\r\n }\r\n\r\n /**\r\n * Restores the state of the stencil buffer\r\n */\r\n public restoreStencilState() {\r\n this.setStencilFunction(this._cachedStencilFunction);\r\n this.setStencilMask(this._cachedStencilMask);\r\n this.setStencilBuffer(this._cachedStencilBuffer);\r\n this.setStencilOperationPass(this._cachedStencilOperationPass);\r\n this.setStencilOperationFail(this._cachedStencilOperationFail);\r\n this.setStencilOperationDepthFail(this._cachedStencilOperationDepthFail);\r\n this.setStencilFunctionReference(this._cachedStencilReference);\r\n }\r\n\r\n /**\r\n * Directly set the WebGL Viewport\r\n * @param x defines the x coordinate of the viewport (in screen space)\r\n * @param y defines the y coordinate of the viewport (in screen space)\r\n * @param width defines the width of the viewport (in screen space)\r\n * @param height defines the height of the viewport (in screen space)\r\n * @returns the current viewport Object (if any) that is being replaced by this call. You can restore this viewport later on to go back to the original state\r\n */\r\n public setDirectViewport(x: number, y: number, width: number, height: number): Nullable {\r\n const currentViewport = this._cachedViewport;\r\n this._cachedViewport = null;\r\n\r\n this._viewport(x, y, width, height);\r\n\r\n return currentViewport;\r\n }\r\n\r\n /**\r\n * Executes a scissor clear (ie. a clear on a specific portion of the screen)\r\n * @param x defines the x-coordinate of the bottom left corner of the clear rectangle\r\n * @param y defines the y-coordinate of the corner of the clear rectangle\r\n * @param width defines the width of the clear rectangle\r\n * @param height defines the height of the clear rectangle\r\n * @param clearColor defines the clear color\r\n */\r\n public scissorClear(x: number, y: number, width: number, height: number, clearColor: IColor4Like): void {\r\n this.enableScissor(x, y, width, height);\r\n this.clear(clearColor, true, true, true);\r\n this.disableScissor();\r\n }\r\n\r\n /**\r\n * Enable scissor test on a specific rectangle (ie. render will only be executed on a specific portion of the screen)\r\n * @param x defines the x-coordinate of the bottom left corner of the clear rectangle\r\n * @param y defines the y-coordinate of the corner of the clear rectangle\r\n * @param width defines the width of the clear rectangle\r\n * @param height defines the height of the clear rectangle\r\n */\r\n public enableScissor(x: number, y: number, width: number, height: number): void {\r\n const gl = this._gl;\r\n\r\n // Change state\r\n gl.enable(gl.SCISSOR_TEST);\r\n gl.scissor(x, y, width, height);\r\n }\r\n\r\n /**\r\n * Disable previously set scissor test rectangle\r\n */\r\n public disableScissor() {\r\n const gl = this._gl;\r\n\r\n gl.disable(gl.SCISSOR_TEST);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _reportDrawCall(numDrawCalls = 1) {\r\n this._drawCalls.addCount(numDrawCalls, false);\r\n }\r\n\r\n /**\r\n * Initializes a webVR display and starts listening to display change events\r\n * The onVRDisplayChangedObservable will be notified upon these changes\r\n * @returns The onVRDisplayChangedObservable\r\n */\r\n public initWebVR(): Observable {\r\n throw _WarnImport(\"WebVRCamera\");\r\n }\r\n\r\n /** @internal */\r\n public _prepareVRComponent() {\r\n // Do nothing as the engine side effect will overload it\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public _connectVREvents(canvas?: HTMLCanvasElement, document?: any) {\r\n // Do nothing as the engine side effect will overload it\r\n }\r\n\r\n /** @internal */\r\n public _submitVRFrame() {\r\n // Do nothing as the engine side effect will overload it\r\n }\r\n /**\r\n * Call this function to leave webVR mode\r\n * Will do nothing if webVR is not supported or if there is no webVR device\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/webVRCamera\r\n */\r\n public disableVR() {\r\n // Do nothing as the engine side effect will overload it\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating that the system is in VR mode and is presenting\r\n * @returns true if VR mode is engaged\r\n */\r\n public isVRPresenting() {\r\n return false;\r\n }\r\n\r\n /** @internal */\r\n public _requestVRFrame() {\r\n // Do nothing as the engine side effect will overload it\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _loadFileAsync(url: string, offlineProvider?: IOfflineProvider, useArrayBuffer?: boolean): Promise {\r\n return new Promise((resolve, reject) => {\r\n this._loadFile(\r\n url,\r\n (data) => {\r\n resolve(data);\r\n },\r\n undefined,\r\n offlineProvider,\r\n useArrayBuffer,\r\n (request, exception) => {\r\n reject(exception);\r\n }\r\n );\r\n });\r\n }\r\n\r\n /**\r\n * Gets the source code of the vertex shader associated with a specific webGL program\r\n * @param program defines the program to use\r\n * @returns a string containing the source code of the vertex shader associated with the program\r\n */\r\n public getVertexShaderSource(program: WebGLProgram): Nullable {\r\n const shaders = this._gl.getAttachedShaders(program);\r\n\r\n if (!shaders) {\r\n return null;\r\n }\r\n\r\n return this._gl.getShaderSource(shaders[0]);\r\n }\r\n\r\n /**\r\n * Gets the source code of the fragment shader associated with a specific webGL program\r\n * @param program defines the program to use\r\n * @returns a string containing the source code of the fragment shader associated with the program\r\n */\r\n public getFragmentShaderSource(program: WebGLProgram): Nullable {\r\n const shaders = this._gl.getAttachedShaders(program);\r\n\r\n if (!shaders) {\r\n return null;\r\n }\r\n\r\n return this._gl.getShaderSource(shaders[1]);\r\n }\r\n\r\n /**\r\n * Sets a depth stencil texture from a render target to the according uniform.\r\n * @param channel The texture channel\r\n * @param uniform The uniform to set\r\n * @param texture The render target texture containing the depth stencil texture to apply\r\n * @param name The texture name\r\n */\r\n public setDepthStencilTexture(channel: number, uniform: Nullable, texture: Nullable, name?: string): void {\r\n if (channel === undefined) {\r\n return;\r\n }\r\n\r\n if (uniform) {\r\n this._boundUniforms[channel] = uniform;\r\n }\r\n\r\n if (!texture || !texture.depthStencilTexture) {\r\n this._setTexture(channel, null, undefined, undefined, name);\r\n } else {\r\n this._setTexture(channel, texture, false, true, name);\r\n }\r\n }\r\n\r\n /**\r\n * Sets a texture to the webGL context from a postprocess\r\n * @param channel defines the channel to use\r\n * @param postProcess defines the source postprocess\r\n * @param name name of the channel\r\n */\r\n public setTextureFromPostProcess(channel: number, postProcess: Nullable, name: string): void {\r\n let postProcessInput = null;\r\n if (postProcess) {\r\n if (postProcess._forcedOutputTexture) {\r\n postProcessInput = postProcess._forcedOutputTexture;\r\n } else if (postProcess._textures.data[postProcess._currentRenderTextureInd]) {\r\n postProcessInput = postProcess._textures.data[postProcess._currentRenderTextureInd];\r\n }\r\n }\r\n\r\n this._bindTexture(channel, postProcessInput?.texture ?? null, name);\r\n }\r\n\r\n /**\r\n * Binds the output of the passed in post process to the texture channel specified\r\n * @param channel The channel the texture should be bound to\r\n * @param postProcess The post process which's output should be bound\r\n * @param name name of the channel\r\n */\r\n public setTextureFromPostProcessOutput(channel: number, postProcess: Nullable, name: string): void {\r\n this._bindTexture(channel, postProcess?._outputTexture?.texture ?? null, name);\r\n }\r\n\r\n protected _rebuildBuffers(): void {\r\n // Index / Vertex\r\n for (const scene of this.scenes) {\r\n scene.resetCachedMaterial();\r\n scene._rebuildGeometries();\r\n scene._rebuildTextures();\r\n }\r\n\r\n for (const scene of this._virtualScenes) {\r\n scene.resetCachedMaterial();\r\n scene._rebuildGeometries();\r\n scene._rebuildTextures();\r\n }\r\n\r\n super._rebuildBuffers();\r\n }\r\n\r\n /** @internal */\r\n public _renderFrame() {\r\n for (let index = 0; index < this._activeRenderLoops.length; index++) {\r\n const renderFunction = this._activeRenderLoops[index];\r\n\r\n renderFunction();\r\n }\r\n }\r\n\r\n protected _cancelFrame() {\r\n if (this._renderingQueueLaunched && this.customAnimationFrameRequester) {\r\n this._renderingQueueLaunched = false;\r\n const { cancelAnimationFrame } = this.customAnimationFrameRequester;\r\n if (cancelAnimationFrame) {\r\n cancelAnimationFrame(this.customAnimationFrameRequester.requestID);\r\n }\r\n } else {\r\n super._cancelFrame();\r\n }\r\n }\r\n\r\n public _renderLoop(): void {\r\n if (!this._contextWasLost) {\r\n let shouldRender = true;\r\n if (this.isDisposed || (!this.renderEvenInBackground && this._windowIsBackground)) {\r\n shouldRender = false;\r\n }\r\n\r\n if (shouldRender) {\r\n // Start new frame\r\n this.beginFrame();\r\n\r\n // Child canvases\r\n if (!this._renderViews()) {\r\n // Main frame\r\n this._renderFrame();\r\n }\r\n\r\n // Present\r\n this.endFrame();\r\n }\r\n }\r\n\r\n if (this._activeRenderLoops.length > 0) {\r\n // Register new frame\r\n if (this.customAnimationFrameRequester) {\r\n this.customAnimationFrameRequester.requestID = this._queueNewFrame(\r\n this.customAnimationFrameRequester.renderFunction || this._boundRenderFunction,\r\n this.customAnimationFrameRequester\r\n );\r\n this._frameHandler = this.customAnimationFrameRequester.requestID;\r\n } else if (this.isVRPresenting()) {\r\n this._requestVRFrame();\r\n } else {\r\n this._frameHandler = this._queueNewFrame(this._boundRenderFunction, this.getHostWindow());\r\n }\r\n } else {\r\n this._renderingQueueLaunched = false;\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _renderViews() {\r\n return false;\r\n }\r\n\r\n /**\r\n * Toggle full screen mode\r\n * @param requestPointerLock defines if a pointer lock should be requested from the user\r\n */\r\n public switchFullscreen(requestPointerLock: boolean): void {\r\n if (this.isFullscreen) {\r\n this.exitFullscreen();\r\n } else {\r\n this.enterFullscreen(requestPointerLock);\r\n }\r\n }\r\n\r\n /**\r\n * Enters full screen mode\r\n * @param requestPointerLock defines if a pointer lock should be requested from the user\r\n */\r\n public enterFullscreen(requestPointerLock: boolean): void {\r\n if (!this.isFullscreen) {\r\n this._pointerLockRequested = requestPointerLock;\r\n if (this._renderingCanvas) {\r\n Engine._RequestFullscreen(this._renderingCanvas);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Exits full screen mode\r\n */\r\n public exitFullscreen(): void {\r\n if (this.isFullscreen) {\r\n Engine._ExitFullscreen();\r\n }\r\n }\r\n\r\n /**\r\n * Enters Pointerlock mode\r\n */\r\n public enterPointerlock(): void {\r\n if (this._renderingCanvas) {\r\n Engine._RequestPointerlock(this._renderingCanvas);\r\n }\r\n }\r\n\r\n /**\r\n * Exits Pointerlock mode\r\n */\r\n public exitPointerlock(): void {\r\n Engine._ExitPointerlock();\r\n }\r\n\r\n /**\r\n * Begin a new frame\r\n */\r\n public beginFrame(): void {\r\n this._measureFps();\r\n\r\n this.onBeginFrameObservable.notifyObservers(this);\r\n super.beginFrame();\r\n }\r\n\r\n /**\r\n * End the current frame\r\n */\r\n public endFrame(): void {\r\n super.endFrame();\r\n this._submitVRFrame();\r\n\r\n this.onEndFrameObservable.notifyObservers(this);\r\n }\r\n\r\n /**\r\n * Resize the view according to the canvas' size\r\n * @param forceSetSize true to force setting the sizes of the underlying canvas\r\n */\r\n public resize(forceSetSize = false): void {\r\n // We're not resizing the size of the canvas while in VR mode & presenting\r\n if (this.isVRPresenting()) {\r\n return;\r\n }\r\n\r\n super.resize(forceSetSize);\r\n }\r\n\r\n /**\r\n * Force a specific size of the canvas\r\n * @param width defines the new canvas' width\r\n * @param height defines the new canvas' height\r\n * @param forceSetSize true to force setting the sizes of the underlying canvas\r\n * @returns true if the size was changed\r\n */\r\n public setSize(width: number, height: number, forceSetSize = false): boolean {\r\n if (!this._renderingCanvas) {\r\n return false;\r\n }\r\n\r\n if (!super.setSize(width, height, forceSetSize)) {\r\n return false;\r\n }\r\n\r\n if (this.scenes) {\r\n for (let index = 0; index < this.scenes.length; index++) {\r\n const scene = this.scenes[index];\r\n\r\n for (let camIndex = 0; camIndex < scene.cameras.length; camIndex++) {\r\n const cam = scene.cameras[camIndex];\r\n\r\n cam._currentRenderId = 0;\r\n }\r\n }\r\n\r\n if (this.onResizeObservable.hasObservers()) {\r\n this.onResizeObservable.notifyObservers(this);\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n public _deletePipelineContext(pipelineContext: IPipelineContext): void {\r\n const webGLPipelineContext = pipelineContext as WebGLPipelineContext;\r\n if (webGLPipelineContext && webGLPipelineContext.program) {\r\n if (webGLPipelineContext.transformFeedback) {\r\n this.deleteTransformFeedback(webGLPipelineContext.transformFeedback);\r\n webGLPipelineContext.transformFeedback = null;\r\n }\r\n }\r\n super._deletePipelineContext(pipelineContext);\r\n }\r\n\r\n public createShaderProgram(\r\n pipelineContext: IPipelineContext,\r\n vertexCode: string,\r\n fragmentCode: string,\r\n defines: Nullable,\r\n context?: WebGLRenderingContext,\r\n transformFeedbackVaryings: Nullable = null\r\n ): WebGLProgram {\r\n context = context || this._gl;\r\n\r\n this.onBeforeShaderCompilationObservable.notifyObservers(this);\r\n\r\n const program = super.createShaderProgram(pipelineContext, vertexCode, fragmentCode, defines, context, transformFeedbackVaryings);\r\n this.onAfterShaderCompilationObservable.notifyObservers(this);\r\n\r\n return program;\r\n }\r\n\r\n protected _createShaderProgram(\r\n pipelineContext: WebGLPipelineContext,\r\n vertexShader: WebGLShader,\r\n fragmentShader: WebGLShader,\r\n context: WebGLRenderingContext,\r\n transformFeedbackVaryings: Nullable = null\r\n ): WebGLProgram {\r\n const shaderProgram = context.createProgram();\r\n pipelineContext.program = shaderProgram;\r\n\r\n if (!shaderProgram) {\r\n throw new Error(\"Unable to create program\");\r\n }\r\n\r\n context.attachShader(shaderProgram, vertexShader);\r\n context.attachShader(shaderProgram, fragmentShader);\r\n\r\n if (this.webGLVersion > 1 && transformFeedbackVaryings) {\r\n const transformFeedback = this.createTransformFeedback();\r\n\r\n this.bindTransformFeedback(transformFeedback);\r\n this.setTranformFeedbackVaryings(shaderProgram, transformFeedbackVaryings);\r\n pipelineContext.transformFeedback = transformFeedback;\r\n }\r\n\r\n context.linkProgram(shaderProgram);\r\n\r\n if (this.webGLVersion > 1 && transformFeedbackVaryings) {\r\n this.bindTransformFeedback(null);\r\n }\r\n\r\n pipelineContext.context = context;\r\n pipelineContext.vertexShader = vertexShader;\r\n pipelineContext.fragmentShader = fragmentShader;\r\n\r\n if (!pipelineContext.isParallelCompiled) {\r\n this._finalizePipelineContext(pipelineContext);\r\n }\r\n\r\n return shaderProgram;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _releaseTexture(texture: InternalTexture): void {\r\n super._releaseTexture(texture);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _releaseRenderTargetWrapper(rtWrapper: RenderTargetWrapper): void {\r\n super._releaseRenderTargetWrapper(rtWrapper);\r\n\r\n // Set output texture of post process to null if the framebuffer has been released/disposed\r\n this.scenes.forEach((scene) => {\r\n scene.postProcesses.forEach((postProcess) => {\r\n if (postProcess._outputTexture === rtWrapper) {\r\n postProcess._outputTexture = null;\r\n }\r\n });\r\n scene.cameras.forEach((camera) => {\r\n camera._postProcesses.forEach((postProcess) => {\r\n if (postProcess) {\r\n if (postProcess._outputTexture === rtWrapper) {\r\n postProcess._outputTexture = null;\r\n }\r\n }\r\n });\r\n });\r\n });\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n protected static _RenderPassIdCounter = 0;\r\n /**\r\n * Gets or sets the current render pass id\r\n */\r\n public currentRenderPassId = Constants.RENDERPASS_MAIN;\r\n\r\n private _renderPassNames: string[] = [\"main\"];\r\n /**\r\n * Gets the names of the render passes that are currently created\r\n * @returns list of the render pass names\r\n */\r\n public getRenderPassNames(): string[] {\r\n return this._renderPassNames;\r\n }\r\n\r\n /**\r\n * Gets the name of the current render pass\r\n * @returns name of the current render pass\r\n */\r\n public getCurrentRenderPassName(): string {\r\n return this._renderPassNames[this.currentRenderPassId];\r\n }\r\n\r\n /**\r\n * Creates a render pass id\r\n * @param name Name of the render pass (for debug purpose only)\r\n * @returns the id of the new render pass\r\n */\r\n public createRenderPassId(name?: string) {\r\n // Note: render pass id == 0 is always for the main render pass\r\n const id = ++Engine._RenderPassIdCounter;\r\n this._renderPassNames[id] = name ?? \"NONAME\";\r\n return id;\r\n }\r\n\r\n /**\r\n * Releases a render pass id\r\n * @param id id of the render pass to release\r\n */\r\n public releaseRenderPassId(id: number): void {\r\n this._renderPassNames[id] = undefined as any;\r\n\r\n for (let s = 0; s < this.scenes.length; ++s) {\r\n const scene = this.scenes[s];\r\n for (let m = 0; m < scene.meshes.length; ++m) {\r\n const mesh = scene.meshes[m];\r\n if (mesh.subMeshes) {\r\n for (let b = 0; b < mesh.subMeshes.length; ++b) {\r\n const subMesh = mesh.subMeshes[b];\r\n subMesh._removeDrawWrapper(id);\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n * Rescales a texture\r\n * @param source input texture\r\n * @param destination destination texture\r\n * @param scene scene to use to render the resize\r\n * @param internalFormat format to use when resizing\r\n * @param onComplete callback to be called when resize has completed\r\n */\r\n public _rescaleTexture(source: InternalTexture, destination: InternalTexture, scene: Nullable, internalFormat: number, onComplete: () => void): void {\r\n this._gl.texParameteri(this._gl.TEXTURE_2D, this._gl.TEXTURE_MAG_FILTER, this._gl.LINEAR);\r\n this._gl.texParameteri(this._gl.TEXTURE_2D, this._gl.TEXTURE_MIN_FILTER, this._gl.LINEAR);\r\n this._gl.texParameteri(this._gl.TEXTURE_2D, this._gl.TEXTURE_WRAP_S, this._gl.CLAMP_TO_EDGE);\r\n this._gl.texParameteri(this._gl.TEXTURE_2D, this._gl.TEXTURE_WRAP_T, this._gl.CLAMP_TO_EDGE);\r\n\r\n const rtt = this.createRenderTargetTexture(\r\n {\r\n width: destination.width,\r\n height: destination.height,\r\n },\r\n {\r\n generateMipMaps: false,\r\n type: Constants.TEXTURETYPE_UNSIGNED_INT,\r\n samplingMode: Constants.TEXTURE_BILINEAR_SAMPLINGMODE,\r\n generateDepthBuffer: false,\r\n generateStencilBuffer: false,\r\n }\r\n );\r\n\r\n if (!this._rescalePostProcess && Engine._RescalePostProcessFactory) {\r\n this._rescalePostProcess = Engine._RescalePostProcessFactory(this);\r\n }\r\n\r\n if (this._rescalePostProcess) {\r\n this._rescalePostProcess.externalTextureSamplerBinding = true;\r\n this._rescalePostProcess.getEffect().executeWhenCompiled(() => {\r\n this._rescalePostProcess!.onApply = function (effect) {\r\n effect._bindTexture(\"textureSampler\", source);\r\n };\r\n\r\n let hostingScene: Scene = scene;\r\n\r\n if (!hostingScene) {\r\n hostingScene = this.scenes[this.scenes.length - 1];\r\n }\r\n hostingScene.postProcessManager.directRender([this._rescalePostProcess!], rtt, true);\r\n\r\n this._bindTextureDirectly(this._gl.TEXTURE_2D, destination, true);\r\n this._gl.copyTexImage2D(this._gl.TEXTURE_2D, 0, internalFormat, 0, 0, destination.width, destination.height, 0);\r\n\r\n this.unBindFramebuffer(rtt);\r\n rtt.dispose();\r\n\r\n if (onComplete) {\r\n onComplete();\r\n }\r\n });\r\n }\r\n }\r\n\r\n // FPS\r\n\r\n /**\r\n * Gets the current framerate\r\n * @returns a number representing the framerate\r\n */\r\n public getFps(): number {\r\n return this._fps;\r\n }\r\n\r\n /**\r\n * Gets the time spent between current and previous frame\r\n * @returns a number representing the delta time in ms\r\n */\r\n public getDeltaTime(): number {\r\n return this._deltaTime;\r\n }\r\n\r\n private _measureFps(): void {\r\n this._performanceMonitor.sampleFrame();\r\n this._fps = this._performanceMonitor.averageFPS;\r\n this._deltaTime = this._performanceMonitor.instantaneousFrameTime || 0;\r\n }\r\n\r\n /**\r\n * Wraps an external web gl texture in a Babylon texture.\r\n * @param texture defines the external texture\r\n * @param hasMipMaps defines whether the external texture has mip maps (default: false)\r\n * @param samplingMode defines the sampling mode for the external texture (default: Constants.TEXTURE_TRILINEAR_SAMPLINGMODE)\r\n * @returns the babylon internal texture\r\n */\r\n public wrapWebGLTexture(texture: WebGLTexture, hasMipMaps: boolean = false, samplingMode: number = Constants.TEXTURE_TRILINEAR_SAMPLINGMODE): InternalTexture {\r\n const hardwareTexture = new WebGLHardwareTexture(texture, this._gl);\r\n const internalTexture = new InternalTexture(this, InternalTextureSource.Unknown, true);\r\n internalTexture._hardwareTexture = hardwareTexture;\r\n internalTexture.isReady = true;\r\n internalTexture.useMipMaps = hasMipMaps;\r\n this.updateTextureSamplingMode(samplingMode, internalTexture);\r\n return internalTexture;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _uploadImageToTexture(texture: InternalTexture, image: HTMLImageElement | ImageBitmap, faceIndex: number = 0, lod: number = 0) {\r\n const gl = this._gl;\r\n\r\n const textureType = this._getWebGLTextureType(texture.type);\r\n const format = this._getInternalFormat(texture.format);\r\n const internalFormat = this._getRGBABufferInternalSizedFormat(texture.type, format);\r\n\r\n const bindTarget = texture.isCube ? gl.TEXTURE_CUBE_MAP : gl.TEXTURE_2D;\r\n\r\n this._bindTextureDirectly(bindTarget, texture, true);\r\n this._unpackFlipY(texture.invertY);\r\n\r\n let target: GLenum = gl.TEXTURE_2D;\r\n if (texture.isCube) {\r\n target = gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex;\r\n }\r\n\r\n gl.texImage2D(target, lod, internalFormat, format, textureType, image);\r\n this._bindTextureDirectly(bindTarget, null, true);\r\n }\r\n\r\n /**\r\n * Updates a depth texture Comparison Mode and Function.\r\n * If the comparison Function is equal to 0, the mode will be set to none.\r\n * Otherwise, this only works in webgl 2 and requires a shadow sampler in the shader.\r\n * @param texture The texture to set the comparison function for\r\n * @param comparisonFunction The comparison function to set, 0 if no comparison required\r\n */\r\n public updateTextureComparisonFunction(texture: InternalTexture, comparisonFunction: number): void {\r\n if (this.webGLVersion === 1) {\r\n Logger.Error(\"WebGL 1 does not support texture comparison.\");\r\n return;\r\n }\r\n\r\n const gl = this._gl;\r\n\r\n if (texture.isCube) {\r\n this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, texture, true);\r\n\r\n if (comparisonFunction === 0) {\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_COMPARE_FUNC, Constants.LEQUAL);\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_COMPARE_MODE, gl.NONE);\r\n } else {\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_COMPARE_FUNC, comparisonFunction);\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_COMPARE_MODE, gl.COMPARE_REF_TO_TEXTURE);\r\n }\r\n\r\n this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, null);\r\n } else {\r\n this._bindTextureDirectly(this._gl.TEXTURE_2D, texture, true);\r\n\r\n if (comparisonFunction === 0) {\r\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_COMPARE_FUNC, Constants.LEQUAL);\r\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_COMPARE_MODE, gl.NONE);\r\n } else {\r\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_COMPARE_FUNC, comparisonFunction);\r\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_COMPARE_MODE, gl.COMPARE_REF_TO_TEXTURE);\r\n }\r\n\r\n this._bindTextureDirectly(this._gl.TEXTURE_2D, null);\r\n }\r\n\r\n texture._comparisonFunction = comparisonFunction;\r\n }\r\n\r\n /**\r\n * Creates a webGL buffer to use with instantiation\r\n * @param capacity defines the size of the buffer\r\n * @returns the webGL buffer\r\n */\r\n public createInstancesBuffer(capacity: number): DataBuffer {\r\n const buffer = this._gl.createBuffer();\r\n\r\n if (!buffer) {\r\n throw new Error(\"Unable to create instance buffer\");\r\n }\r\n\r\n const result = new WebGLDataBuffer(buffer);\r\n result.capacity = capacity;\r\n\r\n this.bindArrayBuffer(result);\r\n this._gl.bufferData(this._gl.ARRAY_BUFFER, capacity, this._gl.DYNAMIC_DRAW);\r\n\r\n result.references = 1;\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Delete a webGL buffer used with instantiation\r\n * @param buffer defines the webGL buffer to delete\r\n */\r\n public deleteInstancesBuffer(buffer: WebGLBuffer): void {\r\n this._gl.deleteBuffer(buffer);\r\n }\r\n\r\n private _clientWaitAsync(sync: WebGLSync, flags = 0, intervalms = 10): Promise {\r\n const gl = (this._gl as any);\r\n return new Promise((resolve, reject) => {\r\n const check = () => {\r\n const res = gl.clientWaitSync(sync, flags, 0);\r\n if (res == gl.WAIT_FAILED) {\r\n reject();\r\n return;\r\n }\r\n if (res == gl.TIMEOUT_EXPIRED) {\r\n setTimeout(check, intervalms);\r\n return;\r\n }\r\n resolve();\r\n };\r\n\r\n check();\r\n });\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _readPixelsAsync(x: number, y: number, w: number, h: number, format: number, type: number, outputBuffer: ArrayBufferView) {\r\n if (this._webGLVersion < 2) {\r\n throw new Error(\"_readPixelsAsync only work on WebGL2+\");\r\n }\r\n\r\n const gl = (this._gl as any);\r\n const buf = gl.createBuffer();\r\n gl.bindBuffer(gl.PIXEL_PACK_BUFFER, buf);\r\n gl.bufferData(gl.PIXEL_PACK_BUFFER, outputBuffer.byteLength, gl.STREAM_READ);\r\n gl.readPixels(x, y, w, h, format, type, 0);\r\n gl.bindBuffer(gl.PIXEL_PACK_BUFFER, null);\r\n\r\n const sync = gl.fenceSync(gl.SYNC_GPU_COMMANDS_COMPLETE, 0);\r\n if (!sync) {\r\n return null;\r\n }\r\n\r\n gl.flush();\r\n\r\n return this._clientWaitAsync(sync, 0, 10).then(() => {\r\n gl.deleteSync(sync);\r\n\r\n gl.bindBuffer(gl.PIXEL_PACK_BUFFER, buf);\r\n gl.getBufferSubData(gl.PIXEL_PACK_BUFFER, 0, outputBuffer);\r\n gl.bindBuffer(gl.PIXEL_PACK_BUFFER, null);\r\n gl.deleteBuffer(buf);\r\n\r\n return outputBuffer;\r\n });\r\n }\r\n\r\n public dispose(): void {\r\n this.hideLoadingUI();\r\n\r\n this.onNewSceneAddedObservable.clear();\r\n\r\n // Release postProcesses\r\n while (this.postProcesses.length) {\r\n this.postProcesses[0].dispose();\r\n }\r\n\r\n // Rescale PP\r\n if (this._rescalePostProcess) {\r\n this._rescalePostProcess.dispose();\r\n }\r\n\r\n // Release scenes\r\n while (this.scenes.length) {\r\n this.scenes[0].dispose();\r\n }\r\n\r\n while (this._virtualScenes.length) {\r\n this._virtualScenes[0].dispose();\r\n }\r\n\r\n // Release audio engine\r\n if (EngineStore.Instances.length === 1 && Engine.audioEngine) {\r\n Engine.audioEngine.dispose();\r\n Engine.audioEngine = null;\r\n }\r\n\r\n //WebVR\r\n this.disableVR();\r\n\r\n // Events\r\n const hostWindow = this.getHostWindow(); // it calls IsWindowObjectExist()\r\n if (hostWindow && typeof hostWindow.removeEventListener === \"function\") {\r\n hostWindow.removeEventListener(\"blur\", this._onBlur);\r\n hostWindow.removeEventListener(\"focus\", this._onFocus);\r\n }\r\n\r\n if (this._renderingCanvas) {\r\n this._renderingCanvas.removeEventListener(\"focus\", this._onCanvasFocus);\r\n this._renderingCanvas.removeEventListener(\"blur\", this._onCanvasBlur);\r\n this._renderingCanvas.removeEventListener(\"pointerout\", this._onCanvasPointerOut);\r\n this._renderingCanvas.removeEventListener(\"contextmenu\", this._onCanvasContextMenu);\r\n }\r\n\r\n if (IsDocumentAvailable()) {\r\n document.removeEventListener(\"fullscreenchange\", this._onFullscreenChange);\r\n document.removeEventListener(\"mozfullscreenchange\", this._onFullscreenChange);\r\n document.removeEventListener(\"webkitfullscreenchange\", this._onFullscreenChange);\r\n document.removeEventListener(\"msfullscreenchange\", this._onFullscreenChange);\r\n document.removeEventListener(\"pointerlockchange\", this._onPointerLockChange);\r\n document.removeEventListener(\"mspointerlockchange\", this._onPointerLockChange);\r\n document.removeEventListener(\"mozpointerlockchange\", this._onPointerLockChange);\r\n document.removeEventListener(\"webkitpointerlockchange\", this._onPointerLockChange);\r\n }\r\n\r\n super.dispose();\r\n\r\n // Remove from Instances\r\n const index = EngineStore.Instances.indexOf(this);\r\n\r\n if (index >= 0) {\r\n EngineStore.Instances.splice(index, 1);\r\n }\r\n\r\n // no more engines left in the engine store? Notify!\r\n if (!Engine.Instances.length) {\r\n EngineStore.OnEnginesDisposedObservable.notifyObservers(this);\r\n }\r\n\r\n // Observables\r\n this.onResizeObservable.clear();\r\n this.onCanvasBlurObservable.clear();\r\n this.onCanvasFocusObservable.clear();\r\n this.onCanvasPointerOutObservable.clear();\r\n this.onBeginFrameObservable.clear();\r\n this.onEndFrameObservable.clear();\r\n }\r\n\r\n private _disableTouchAction(): void {\r\n if (!this._renderingCanvas || !this._renderingCanvas.setAttribute) {\r\n return;\r\n }\r\n\r\n this._renderingCanvas.setAttribute(\"touch-action\", \"none\");\r\n this._renderingCanvas.style.touchAction = \"none\";\r\n (this._renderingCanvas.style as any).webkitTapHighlightColor = \"transparent\";\r\n }\r\n\r\n // Loading screen\r\n\r\n /**\r\n * Display the loading screen\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/customLoadingScreen\r\n */\r\n public displayLoadingUI(): void {\r\n if (!IsWindowObjectExist()) {\r\n return;\r\n }\r\n const loadingScreen = this.loadingScreen;\r\n if (loadingScreen) {\r\n loadingScreen.displayLoadingUI();\r\n }\r\n }\r\n\r\n /**\r\n * Hide the loading screen\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/customLoadingScreen\r\n */\r\n public hideLoadingUI(): void {\r\n if (!IsWindowObjectExist()) {\r\n return;\r\n }\r\n const loadingScreen = this._loadingScreen;\r\n if (loadingScreen) {\r\n loadingScreen.hideLoadingUI();\r\n }\r\n }\r\n\r\n /**\r\n * Gets the current loading screen object\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/customLoadingScreen\r\n */\r\n public get loadingScreen(): ILoadingScreen {\r\n if (!this._loadingScreen && this._renderingCanvas) {\r\n this._loadingScreen = Engine.DefaultLoadingScreenFactory(this._renderingCanvas);\r\n }\r\n return this._loadingScreen;\r\n }\r\n\r\n /**\r\n * Sets the current loading screen object\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/customLoadingScreen\r\n */\r\n public set loadingScreen(loadingScreen: ILoadingScreen) {\r\n this._loadingScreen = loadingScreen;\r\n }\r\n\r\n /**\r\n * Sets the current loading screen text\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/customLoadingScreen\r\n */\r\n public set loadingUIText(text: string) {\r\n this.loadingScreen.loadingUIText = text;\r\n }\r\n\r\n /**\r\n * Sets the current loading screen background color\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/customLoadingScreen\r\n */\r\n public set loadingUIBackgroundColor(color: string) {\r\n this.loadingScreen.loadingUIBackgroundColor = color;\r\n }\r\n\r\n /**\r\n * creates and returns a new video element\r\n * @param constraints video constraints\r\n * @returns video element\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public createVideoElement(constraints: MediaTrackConstraints): any {\r\n return document.createElement(\"video\");\r\n }\r\n\r\n /** Pointerlock and fullscreen */\r\n\r\n /**\r\n * Ask the browser to promote the current element to pointerlock mode\r\n * @param element defines the DOM element to promote\r\n */\r\n static _RequestPointerlock(element: HTMLElement): void {\r\n if (element.requestPointerLock) {\r\n // In some browsers, requestPointerLock returns a promise.\r\n // Handle possible rejections to avoid an unhandled top-level exception.\r\n const promise: unknown = element.requestPointerLock();\r\n if (promise instanceof Promise)\r\n promise\r\n .then(() => {\r\n element.focus();\r\n })\r\n .catch(() => {});\r\n else element.focus();\r\n }\r\n }\r\n\r\n /**\r\n * Asks the browser to exit pointerlock mode\r\n */\r\n static _ExitPointerlock(): void {\r\n if (document.exitPointerLock) {\r\n document.exitPointerLock();\r\n }\r\n }\r\n\r\n /**\r\n * Ask the browser to promote the current element to fullscreen rendering mode\r\n * @param element defines the DOM element to promote\r\n */\r\n static _RequestFullscreen(element: HTMLElement): void {\r\n const requestFunction = element.requestFullscreen || (element).webkitRequestFullscreen;\r\n if (!requestFunction) {\r\n return;\r\n }\r\n requestFunction.call(element);\r\n }\r\n\r\n /**\r\n * Asks the browser to exit fullscreen mode\r\n */\r\n static _ExitFullscreen(): void {\r\n const anyDoc = document as any;\r\n\r\n if (document.exitFullscreen) {\r\n document.exitFullscreen();\r\n } else if (anyDoc.webkitCancelFullScreen) {\r\n anyDoc.webkitCancelFullScreen();\r\n }\r\n }\r\n\r\n /**\r\n * Get Font size information\r\n * @param font font name\r\n * @returns an object containing ascent, height and descent\r\n */\r\n public getFontOffset(font: string): { ascent: number; height: number; descent: number } {\r\n const text = document.createElement(\"span\");\r\n text.innerHTML = \"Hg\";\r\n text.setAttribute(\"style\", `font: ${font} !important`);\r\n\r\n const block = document.createElement(\"div\");\r\n block.style.display = \"inline-block\";\r\n block.style.width = \"1px\";\r\n block.style.height = \"0px\";\r\n block.style.verticalAlign = \"bottom\";\r\n\r\n const div = document.createElement(\"div\");\r\n div.style.whiteSpace = \"nowrap\";\r\n div.appendChild(text);\r\n div.appendChild(block);\r\n\r\n document.body.appendChild(div);\r\n\r\n let fontAscent = 0;\r\n let fontHeight = 0;\r\n try {\r\n fontHeight = block.getBoundingClientRect().top - text.getBoundingClientRect().top;\r\n block.style.verticalAlign = \"baseline\";\r\n fontAscent = block.getBoundingClientRect().top - text.getBoundingClientRect().top;\r\n } finally {\r\n document.body.removeChild(div);\r\n }\r\n return { ascent: fontAscent, height: fontHeight, descent: fontHeight - fontAscent };\r\n }\r\n}\r\n","import { Vector3 } from \"./math.vector\";\r\n\r\n/** Defines supported spaces */\r\nexport enum Space {\r\n /** Local (object) space */\r\n LOCAL = 0,\r\n /** World space */\r\n WORLD = 1,\r\n /** Bone space */\r\n BONE = 2,\r\n}\r\n\r\n/** Defines the 3 main axes */\r\nexport class Axis {\r\n /** X axis */\r\n public static X: Vector3 = new Vector3(1.0, 0.0, 0.0);\r\n /** Y axis */\r\n public static Y: Vector3 = new Vector3(0.0, 1.0, 0.0);\r\n /** Z axis */\r\n public static Z: Vector3 = new Vector3(0.0, 0.0, 1.0);\r\n}\r\n\r\n/**\r\n * Defines cartesian components.\r\n */\r\nexport enum Coordinate {\r\n /** X axis */\r\n X,\r\n /** Y axis */\r\n Y,\r\n /** Z axis */\r\n Z,\r\n}\r\n","import type { DeepImmutable, Nullable } from \"../types\";\r\nimport { serialize, serializeAsVector3, serializeAsQuaternion, SerializationHelper } from \"../Misc/decorators\";\r\nimport { Observable } from \"../Misc/observable\";\r\n\r\nimport type { Camera } from \"../Cameras/camera\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Quaternion, Matrix, Vector3, TmpVectors } from \"../Maths/math.vector\";\r\nimport { Node } from \"../node\";\r\nimport type { Bone } from \"../Bones/bone\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport { Space } from \"../Maths/math.axis\";\r\n\r\nconst convertRHSToLHS = Matrix.Compose(Vector3.One(), Quaternion.FromEulerAngles(0, Math.PI, 0), Vector3.Zero());\r\n\r\n/**\r\n * A TransformNode is an object that is not rendered but can be used as a center of transformation. This can decrease memory usage and increase rendering speed compared to using an empty mesh as a parent and is less complicated than using a pivot matrix.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/parent_pivot/transform_node\r\n */\r\nexport class TransformNode extends Node {\r\n // Statics\r\n /**\r\n * Object will not rotate to face the camera\r\n */\r\n public static BILLBOARDMODE_NONE = 0;\r\n /**\r\n * Object will rotate to face the camera but only on the x axis\r\n */\r\n public static BILLBOARDMODE_X = 1;\r\n /**\r\n * Object will rotate to face the camera but only on the y axis\r\n */\r\n public static BILLBOARDMODE_Y = 2;\r\n /**\r\n * Object will rotate to face the camera but only on the z axis\r\n */\r\n public static BILLBOARDMODE_Z = 4;\r\n /**\r\n * Object will rotate to face the camera\r\n */\r\n public static BILLBOARDMODE_ALL = 7;\r\n /**\r\n * Object will rotate to face the camera's position instead of orientation\r\n */\r\n public static BILLBOARDMODE_USE_POSITION = 128;\r\n /**\r\n * Child transform with Billboard flags should or should not apply parent rotation (default if off)\r\n */\r\n public static BillboardUseParentOrientation: boolean = false;\r\n\r\n private static _TmpRotation = Quaternion.Zero();\r\n private static _TmpScaling = Vector3.Zero();\r\n private static _TmpTranslation = Vector3.Zero();\r\n\r\n private _forward = new Vector3(0, 0, 1);\r\n private _up = new Vector3(0, 1, 0);\r\n private _right = new Vector3(1, 0, 0);\r\n\r\n // Properties\r\n @serializeAsVector3(\"position\")\r\n private _position = Vector3.Zero();\r\n\r\n @serializeAsVector3(\"rotation\")\r\n private _rotation = Vector3.Zero();\r\n\r\n @serializeAsQuaternion(\"rotationQuaternion\")\r\n private _rotationQuaternion: Nullable = null;\r\n\r\n @serializeAsVector3(\"scaling\")\r\n protected _scaling = Vector3.One();\r\n private _transformToBoneReferal: Nullable = null;\r\n private _currentParentWhenAttachingToBone: Nullable;\r\n private _isAbsoluteSynced = false;\r\n\r\n @serialize(\"billboardMode\")\r\n private _billboardMode = TransformNode.BILLBOARDMODE_NONE;\r\n\r\n /**\r\n * Gets or sets the billboard mode. Default is 0.\r\n *\r\n * | Value | Type | Description |\r\n * | --- | --- | --- |\r\n * | 0 | BILLBOARDMODE_NONE | |\r\n * | 1 | BILLBOARDMODE_X | |\r\n * | 2 | BILLBOARDMODE_Y | |\r\n * | 4 | BILLBOARDMODE_Z | |\r\n * | 7 | BILLBOARDMODE_ALL | |\r\n *\r\n */\r\n public get billboardMode() {\r\n return this._billboardMode;\r\n }\r\n\r\n public set billboardMode(value: number) {\r\n if (this._billboardMode === value) {\r\n return;\r\n }\r\n this._billboardMode = value;\r\n this._cache.useBillboardPosition = (this._billboardMode & TransformNode.BILLBOARDMODE_USE_POSITION) !== 0;\r\n this._computeUseBillboardPath();\r\n }\r\n\r\n private _preserveParentRotationForBillboard = false;\r\n /**\r\n * Gets or sets a boolean indicating that parent rotation should be preserved when using billboards.\r\n * This could be useful for glTF objects where parent rotation helps converting from right handed to left handed\r\n */\r\n public get preserveParentRotationForBillboard() {\r\n return this._preserveParentRotationForBillboard;\r\n }\r\n\r\n public set preserveParentRotationForBillboard(value: boolean) {\r\n if (value === this._preserveParentRotationForBillboard) {\r\n return;\r\n }\r\n this._preserveParentRotationForBillboard = value;\r\n this._computeUseBillboardPath();\r\n }\r\n\r\n private _computeUseBillboardPath(): void {\r\n this._cache.useBillboardPath = this._billboardMode !== TransformNode.BILLBOARDMODE_NONE && !this.preserveParentRotationForBillboard;\r\n }\r\n\r\n /**\r\n * Multiplication factor on scale x/y/z when computing the world matrix. Eg. for a 1x1x1 cube setting this to 2 will make it a 2x2x2 cube\r\n */\r\n @serialize()\r\n public scalingDeterminant = 1;\r\n\r\n @serialize(\"infiniteDistance\")\r\n private _infiniteDistance = false;\r\n\r\n /**\r\n * Gets or sets the distance of the object to max, often used by skybox\r\n */\r\n public get infiniteDistance() {\r\n return this._infiniteDistance;\r\n }\r\n\r\n public set infiniteDistance(value: boolean) {\r\n if (this._infiniteDistance === value) {\r\n return;\r\n }\r\n\r\n this._infiniteDistance = value;\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating that non uniform scaling (when at least one component is different from others) should be ignored.\r\n * By default the system will update normals to compensate\r\n */\r\n @serialize()\r\n public ignoreNonUniformScaling = false;\r\n\r\n /**\r\n * Gets or sets a boolean indicating that even if rotationQuaternion is defined, you can keep updating rotation property and Babylon.js will just mix both\r\n */\r\n @serialize()\r\n public reIntegrateRotationIntoRotationQuaternion = false;\r\n\r\n // Cache\r\n /** @internal */\r\n public _poseMatrix: Nullable = null;\r\n /** @internal */\r\n public _localMatrix = Matrix.Zero();\r\n\r\n private _usePivotMatrix = false;\r\n private _absolutePosition = Vector3.Zero();\r\n private _absoluteScaling = Vector3.Zero();\r\n private _absoluteRotationQuaternion = Quaternion.Identity();\r\n private _pivotMatrix = Matrix.Identity();\r\n private _pivotMatrixInverse: Matrix;\r\n /** @internal */\r\n public _postMultiplyPivotMatrix = false;\r\n\r\n protected _isWorldMatrixFrozen = false;\r\n\r\n /** @internal */\r\n public _indexInSceneTransformNodesArray = -1;\r\n\r\n /**\r\n * An event triggered after the world matrix is updated\r\n */\r\n public onAfterWorldMatrixUpdateObservable = new Observable();\r\n\r\n constructor(name: string, scene: Nullable = null, isPure = true) {\r\n super(name, scene);\r\n\r\n if (isPure) {\r\n this.getScene().addTransformNode(this);\r\n }\r\n }\r\n\r\n /**\r\n * Gets a string identifying the name of the class\r\n * @returns \"TransformNode\" string\r\n */\r\n public getClassName(): string {\r\n return \"TransformNode\";\r\n }\r\n\r\n /**\r\n * Gets or set the node position (default is (0.0, 0.0, 0.0))\r\n */\r\n public get position(): Vector3 {\r\n return this._position;\r\n }\r\n\r\n public set position(newPosition: Vector3) {\r\n this._position = newPosition;\r\n this._isDirty = true;\r\n }\r\n\r\n /**\r\n * return true if a pivot has been set\r\n * @returns true if a pivot matrix is used\r\n */\r\n public isUsingPivotMatrix(): boolean {\r\n return this._usePivotMatrix;\r\n }\r\n\r\n /**\r\n * Gets or sets the rotation property : a Vector3 defining the rotation value in radians around each local axis X, Y, Z (default is (0.0, 0.0, 0.0)).\r\n * If rotation quaternion is set, this Vector3 will be ignored and copy from the quaternion\r\n */\r\n public get rotation(): Vector3 {\r\n return this._rotation;\r\n }\r\n\r\n public set rotation(newRotation: Vector3) {\r\n this._rotation = newRotation;\r\n this._rotationQuaternion = null;\r\n this._isDirty = true;\r\n }\r\n\r\n /**\r\n * Gets or sets the scaling property : a Vector3 defining the node scaling along each local axis X, Y, Z (default is (1.0, 1.0, 1.0)).\r\n */\r\n public get scaling(): Vector3 {\r\n return this._scaling;\r\n }\r\n\r\n public set scaling(newScaling: Vector3) {\r\n this._scaling = newScaling;\r\n this._isDirty = true;\r\n }\r\n\r\n /**\r\n * Gets or sets the rotation Quaternion property : this a Quaternion object defining the node rotation by using a unit quaternion (undefined by default, but can be null).\r\n * If set, only the rotationQuaternion is then used to compute the node rotation (ie. node.rotation will be ignored)\r\n */\r\n public get rotationQuaternion(): Nullable {\r\n return this._rotationQuaternion;\r\n }\r\n\r\n public set rotationQuaternion(quaternion: Nullable) {\r\n this._rotationQuaternion = quaternion;\r\n //reset the rotation vector.\r\n if (quaternion) {\r\n this._rotation.setAll(0.0);\r\n }\r\n this._isDirty = true;\r\n }\r\n\r\n /**\r\n * The forward direction of that transform in world space.\r\n */\r\n public get forward(): Vector3 {\r\n Vector3.TransformNormalFromFloatsToRef(0, 0, this.getScene().useRightHandedSystem ? -1.0 : 1.0, this.getWorldMatrix(), this._forward);\r\n return this._forward.normalize();\r\n }\r\n\r\n /**\r\n * The up direction of that transform in world space.\r\n */\r\n public get up(): Vector3 {\r\n Vector3.TransformNormalFromFloatsToRef(0, 1, 0, this.getWorldMatrix(), this._up);\r\n return this._up.normalize();\r\n }\r\n\r\n /**\r\n * The right direction of that transform in world space.\r\n */\r\n public get right(): Vector3 {\r\n Vector3.TransformNormalFromFloatsToRef(this.getScene().useRightHandedSystem ? -1.0 : 1.0, 0, 0, this.getWorldMatrix(), this._right);\r\n return this._right.normalize();\r\n }\r\n\r\n /**\r\n * Copies the parameter passed Matrix into the mesh Pose matrix.\r\n * @param matrix the matrix to copy the pose from\r\n * @returns this TransformNode.\r\n */\r\n public updatePoseMatrix(matrix: Matrix): TransformNode {\r\n if (!this._poseMatrix) {\r\n this._poseMatrix = matrix.clone();\r\n return this;\r\n }\r\n this._poseMatrix.copyFrom(matrix);\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns the mesh Pose matrix.\r\n * @returns the pose matrix\r\n */\r\n public getPoseMatrix(): Matrix {\r\n if (!this._poseMatrix) {\r\n this._poseMatrix = Matrix.Identity();\r\n }\r\n return this._poseMatrix;\r\n }\r\n\r\n /** @internal */\r\n public _isSynchronized(): boolean {\r\n const cache = this._cache;\r\n\r\n if (this._billboardMode !== cache.billboardMode || this._billboardMode !== TransformNode.BILLBOARDMODE_NONE) {\r\n return false;\r\n }\r\n\r\n if (cache.pivotMatrixUpdated) {\r\n return false;\r\n }\r\n\r\n if (this._infiniteDistance) {\r\n return false;\r\n }\r\n\r\n if (this._position._isDirty) {\r\n return false;\r\n }\r\n\r\n if (this._scaling._isDirty) {\r\n return false;\r\n }\r\n\r\n if ((this._rotationQuaternion && this._rotationQuaternion._isDirty) || this._rotation._isDirty) {\r\n return false;\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /** @internal */\r\n public _initCache() {\r\n super._initCache();\r\n\r\n const cache = this._cache;\r\n cache.localMatrixUpdated = false;\r\n cache.billboardMode = -1;\r\n cache.infiniteDistance = false;\r\n cache.useBillboardPosition = false;\r\n cache.useBillboardPath = false;\r\n }\r\n\r\n /**\r\n * Returns the current mesh absolute position.\r\n * Returns a Vector3.\r\n */\r\n public get absolutePosition(): Vector3 {\r\n return this.getAbsolutePosition();\r\n }\r\n\r\n /**\r\n * Returns the current mesh absolute scaling.\r\n * Returns a Vector3.\r\n */\r\n public get absoluteScaling(): Vector3 {\r\n this._syncAbsoluteScalingAndRotation();\r\n return this._absoluteScaling;\r\n }\r\n\r\n /**\r\n * Returns the current mesh absolute rotation.\r\n * Returns a Quaternion.\r\n */\r\n public get absoluteRotationQuaternion(): Quaternion {\r\n this._syncAbsoluteScalingAndRotation();\r\n return this._absoluteRotationQuaternion;\r\n }\r\n\r\n /**\r\n * Sets a new matrix to apply before all other transformation\r\n * @param matrix defines the transform matrix\r\n * @returns the current TransformNode\r\n */\r\n public setPreTransformMatrix(matrix: Matrix): TransformNode {\r\n return this.setPivotMatrix(matrix, false);\r\n }\r\n\r\n /**\r\n * Sets a new pivot matrix to the current node\r\n * @param matrix defines the new pivot matrix to use\r\n * @param postMultiplyPivotMatrix defines if the pivot matrix must be cancelled in the world matrix. When this parameter is set to true (default), the inverse of the pivot matrix is also applied at the end to cancel the transformation effect\r\n * @returns the current TransformNode\r\n */\r\n public setPivotMatrix(matrix: DeepImmutable, postMultiplyPivotMatrix = true): TransformNode {\r\n this._pivotMatrix.copyFrom(matrix);\r\n this._usePivotMatrix = !this._pivotMatrix.isIdentity();\r\n\r\n this._cache.pivotMatrixUpdated = true;\r\n this._postMultiplyPivotMatrix = postMultiplyPivotMatrix;\r\n\r\n if (this._postMultiplyPivotMatrix) {\r\n if (!this._pivotMatrixInverse) {\r\n this._pivotMatrixInverse = Matrix.Invert(this._pivotMatrix);\r\n } else {\r\n this._pivotMatrix.invertToRef(this._pivotMatrixInverse);\r\n }\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns the mesh pivot matrix.\r\n * Default : Identity.\r\n * @returns the matrix\r\n */\r\n public getPivotMatrix(): Matrix {\r\n return this._pivotMatrix;\r\n }\r\n\r\n /**\r\n * Instantiate (when possible) or clone that node with its hierarchy\r\n * @param newParent defines the new parent to use for the instance (or clone)\r\n * @param options defines options to configure how copy is done\r\n * @param options.doNotInstantiate defines if the model must be instantiated or just cloned\r\n * @param onNewNodeCreated defines an option callback to call when a clone or an instance is created\r\n * @returns an instance (or a clone) of the current node with its hierarchy\r\n */\r\n public instantiateHierarchy(\r\n newParent: Nullable = null,\r\n options?: { doNotInstantiate: boolean | ((node: TransformNode) => boolean) },\r\n onNewNodeCreated?: (source: TransformNode, clone: TransformNode) => void\r\n ): Nullable {\r\n const clone = this.clone(\"Clone of \" + (this.name || this.id), newParent || this.parent, true);\r\n\r\n if (clone) {\r\n if (onNewNodeCreated) {\r\n onNewNodeCreated(this, clone);\r\n }\r\n }\r\n\r\n for (const child of this.getChildTransformNodes(true)) {\r\n child.instantiateHierarchy(clone, options, onNewNodeCreated);\r\n }\r\n\r\n return clone;\r\n }\r\n\r\n /**\r\n * Prevents the World matrix to be computed any longer\r\n * @param newWorldMatrix defines an optional matrix to use as world matrix\r\n * @param decompose defines whether to decompose the given newWorldMatrix or directly assign\r\n * @returns the TransformNode.\r\n */\r\n public freezeWorldMatrix(newWorldMatrix: Nullable = null, decompose = false): TransformNode {\r\n if (newWorldMatrix) {\r\n if (decompose) {\r\n this._rotation.setAll(0);\r\n this._rotationQuaternion = this._rotationQuaternion || Quaternion.Identity();\r\n newWorldMatrix.decompose(this._scaling, this._rotationQuaternion, this._position);\r\n this.computeWorldMatrix(true);\r\n } else {\r\n this._worldMatrix = newWorldMatrix;\r\n this._absolutePosition.copyFromFloats(this._worldMatrix.m[12], this._worldMatrix.m[13], this._worldMatrix.m[14]);\r\n this._afterComputeWorldMatrix();\r\n }\r\n } else {\r\n this._isWorldMatrixFrozen = false; // no guarantee world is not already frozen, switch off temporarily\r\n this.computeWorldMatrix(true);\r\n }\r\n this._isDirty = false;\r\n this._isWorldMatrixFrozen = true;\r\n return this;\r\n }\r\n\r\n /**\r\n * Allows back the World matrix computation.\r\n * @returns the TransformNode.\r\n */\r\n public unfreezeWorldMatrix() {\r\n this._isWorldMatrixFrozen = false;\r\n this.computeWorldMatrix(true);\r\n return this;\r\n }\r\n\r\n /**\r\n * True if the World matrix has been frozen.\r\n */\r\n public get isWorldMatrixFrozen(): boolean {\r\n return this._isWorldMatrixFrozen;\r\n }\r\n\r\n /**\r\n * Returns the mesh absolute position in the World.\r\n * @returns a Vector3.\r\n */\r\n public getAbsolutePosition(): Vector3 {\r\n this.computeWorldMatrix();\r\n return this._absolutePosition;\r\n }\r\n\r\n /**\r\n * Sets the mesh absolute position in the World from a Vector3 or an Array(3).\r\n * @param absolutePosition the absolute position to set\r\n * @returns the TransformNode.\r\n */\r\n public setAbsolutePosition(absolutePosition: Vector3): TransformNode {\r\n if (!absolutePosition) {\r\n return this;\r\n }\r\n let absolutePositionX;\r\n let absolutePositionY;\r\n let absolutePositionZ;\r\n if (absolutePosition.x === undefined) {\r\n if (arguments.length < 3) {\r\n return this;\r\n }\r\n absolutePositionX = arguments[0];\r\n absolutePositionY = arguments[1];\r\n absolutePositionZ = arguments[2];\r\n } else {\r\n absolutePositionX = absolutePosition.x;\r\n absolutePositionY = absolutePosition.y;\r\n absolutePositionZ = absolutePosition.z;\r\n }\r\n if (this.parent) {\r\n const invertParentWorldMatrix = TmpVectors.Matrix[0];\r\n this.parent.getWorldMatrix().invertToRef(invertParentWorldMatrix);\r\n Vector3.TransformCoordinatesFromFloatsToRef(absolutePositionX, absolutePositionY, absolutePositionZ, invertParentWorldMatrix, this.position);\r\n } else {\r\n this.position.x = absolutePositionX;\r\n this.position.y = absolutePositionY;\r\n this.position.z = absolutePositionZ;\r\n }\r\n\r\n this._absolutePosition.copyFrom(absolutePosition);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets the mesh position in its local space.\r\n * @param vector3 the position to set in localspace\r\n * @returns the TransformNode.\r\n */\r\n public setPositionWithLocalVector(vector3: Vector3): TransformNode {\r\n this.computeWorldMatrix();\r\n this.position = Vector3.TransformNormal(vector3, this._localMatrix);\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns the mesh position in the local space from the current World matrix values.\r\n * @returns a new Vector3.\r\n */\r\n public getPositionExpressedInLocalSpace(): Vector3 {\r\n this.computeWorldMatrix();\r\n const invLocalWorldMatrix = TmpVectors.Matrix[0];\r\n this._localMatrix.invertToRef(invLocalWorldMatrix);\r\n return Vector3.TransformNormal(this.position, invLocalWorldMatrix);\r\n }\r\n\r\n /**\r\n * Translates the mesh along the passed Vector3 in its local space.\r\n * @param vector3 the distance to translate in localspace\r\n * @returns the TransformNode.\r\n */\r\n public locallyTranslate(vector3: Vector3): TransformNode {\r\n this.computeWorldMatrix(true);\r\n this.position = Vector3.TransformCoordinates(vector3, this._localMatrix);\r\n return this;\r\n }\r\n\r\n private static _LookAtVectorCache = new Vector3(0, 0, 0);\r\n\r\n /**\r\n * Orients a mesh towards a target point. Mesh must be drawn facing user.\r\n * @param targetPoint the position (must be in same space as current mesh) to look at\r\n * @param yawCor optional yaw (y-axis) correction in radians\r\n * @param pitchCor optional pitch (x-axis) correction in radians\r\n * @param rollCor optional roll (z-axis) correction in radians\r\n * @param space the chosen space of the target\r\n * @returns the TransformNode.\r\n */\r\n public lookAt(targetPoint: Vector3, yawCor: number = 0, pitchCor: number = 0, rollCor: number = 0, space: Space = Space.LOCAL): TransformNode {\r\n const dv = TransformNode._LookAtVectorCache;\r\n const pos = space === Space.LOCAL ? this.position : this.getAbsolutePosition();\r\n targetPoint.subtractToRef(pos, dv);\r\n this.setDirection(dv, yawCor, pitchCor, rollCor);\r\n\r\n // Correct for parent's rotation offset\r\n if (space === Space.WORLD && this.parent) {\r\n if (this.rotationQuaternion) {\r\n // Get local rotation matrix of the looking object\r\n const rotationMatrix = TmpVectors.Matrix[0];\r\n this.rotationQuaternion.toRotationMatrix(rotationMatrix);\r\n\r\n // Offset rotation by parent's inverted rotation matrix to correct in world space\r\n const parentRotationMatrix = TmpVectors.Matrix[1];\r\n this.parent.getWorldMatrix().getRotationMatrixToRef(parentRotationMatrix);\r\n parentRotationMatrix.invert();\r\n rotationMatrix.multiplyToRef(parentRotationMatrix, rotationMatrix);\r\n this.rotationQuaternion.fromRotationMatrix(rotationMatrix);\r\n } else {\r\n // Get local rotation matrix of the looking object\r\n const quaternionRotation = TmpVectors.Quaternion[0];\r\n Quaternion.FromEulerVectorToRef(this.rotation, quaternionRotation);\r\n const rotationMatrix = TmpVectors.Matrix[0];\r\n quaternionRotation.toRotationMatrix(rotationMatrix);\r\n\r\n // Offset rotation by parent's inverted rotation matrix to correct in world space\r\n const parentRotationMatrix = TmpVectors.Matrix[1];\r\n this.parent.getWorldMatrix().getRotationMatrixToRef(parentRotationMatrix);\r\n parentRotationMatrix.invert();\r\n rotationMatrix.multiplyToRef(parentRotationMatrix, rotationMatrix);\r\n quaternionRotation.fromRotationMatrix(rotationMatrix);\r\n quaternionRotation.toEulerAnglesToRef(this.rotation);\r\n }\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 that is the localAxis, expressed in the mesh local space, rotated like the mesh.\r\n * This Vector3 is expressed in the World space.\r\n * @param localAxis axis to rotate\r\n * @returns a new Vector3 that is the localAxis, expressed in the mesh local space, rotated like the mesh.\r\n */\r\n public getDirection(localAxis: Vector3): Vector3 {\r\n const result = Vector3.Zero();\r\n\r\n this.getDirectionToRef(localAxis, result);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Sets the Vector3 \"result\" as the rotated Vector3 \"localAxis\" in the same rotation than the mesh.\r\n * localAxis is expressed in the mesh local space.\r\n * result is computed in the World space from the mesh World matrix.\r\n * @param localAxis axis to rotate\r\n * @param result the resulting transformnode\r\n * @returns this TransformNode.\r\n */\r\n public getDirectionToRef(localAxis: Vector3, result: Vector3): TransformNode {\r\n Vector3.TransformNormalToRef(localAxis, this.getWorldMatrix(), result);\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets this transform node rotation to the given local axis.\r\n * @param localAxis the axis in local space\r\n * @param yawCor optional yaw (y-axis) correction in radians\r\n * @param pitchCor optional pitch (x-axis) correction in radians\r\n * @param rollCor optional roll (z-axis) correction in radians\r\n * @returns this TransformNode\r\n */\r\n public setDirection(localAxis: Vector3, yawCor: number = 0, pitchCor: number = 0, rollCor: number = 0): TransformNode {\r\n const yaw = -Math.atan2(localAxis.z, localAxis.x) + Math.PI / 2;\r\n const len = Math.sqrt(localAxis.x * localAxis.x + localAxis.z * localAxis.z);\r\n const pitch = -Math.atan2(localAxis.y, len);\r\n if (this.rotationQuaternion) {\r\n Quaternion.RotationYawPitchRollToRef(yaw + yawCor, pitch + pitchCor, rollCor, this.rotationQuaternion);\r\n } else {\r\n this.rotation.x = pitch + pitchCor;\r\n this.rotation.y = yaw + yawCor;\r\n this.rotation.z = rollCor;\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets a new pivot point to the current node\r\n * @param point defines the new pivot point to use\r\n * @param space defines if the point is in world or local space (local by default)\r\n * @returns the current TransformNode\r\n */\r\n public setPivotPoint(point: Vector3, space: Space = Space.LOCAL): TransformNode {\r\n if (this.getScene().getRenderId() == 0) {\r\n this.computeWorldMatrix(true);\r\n }\r\n\r\n const wm = this.getWorldMatrix();\r\n\r\n if (space == Space.WORLD) {\r\n const tmat = TmpVectors.Matrix[0];\r\n wm.invertToRef(tmat);\r\n point = Vector3.TransformCoordinates(point, tmat);\r\n }\r\n\r\n return this.setPivotMatrix(Matrix.Translation(-point.x, -point.y, -point.z), true);\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 set with the mesh pivot point coordinates in the local space.\r\n * @returns the pivot point\r\n */\r\n public getPivotPoint(): Vector3 {\r\n const point = Vector3.Zero();\r\n this.getPivotPointToRef(point);\r\n return point;\r\n }\r\n\r\n /**\r\n * Sets the passed Vector3 \"result\" with the coordinates of the mesh pivot point in the local space.\r\n * @param result the vector3 to store the result\r\n * @returns this TransformNode.\r\n */\r\n public getPivotPointToRef(result: Vector3): TransformNode {\r\n result.x = -this._pivotMatrix.m[12];\r\n result.y = -this._pivotMatrix.m[13];\r\n result.z = -this._pivotMatrix.m[14];\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 set with the mesh pivot point World coordinates.\r\n * @returns a new Vector3 set with the mesh pivot point World coordinates.\r\n */\r\n public getAbsolutePivotPoint(): Vector3 {\r\n const point = Vector3.Zero();\r\n this.getAbsolutePivotPointToRef(point);\r\n return point;\r\n }\r\n\r\n /**\r\n * Sets the Vector3 \"result\" coordinates with the mesh pivot point World coordinates.\r\n * @param result vector3 to store the result\r\n * @returns this TransformNode.\r\n */\r\n public getAbsolutePivotPointToRef(result: Vector3): TransformNode {\r\n this.getPivotPointToRef(result);\r\n Vector3.TransformCoordinatesToRef(result, this.getWorldMatrix(), result);\r\n return this;\r\n }\r\n\r\n /**\r\n * Flag the transform node as dirty (Forcing it to update everything)\r\n * @param property if set to \"rotation\" the objects rotationQuaternion will be set to null\r\n * @returns this node\r\n */\r\n public markAsDirty(property?: string): Node {\r\n if (this._isDirty) {\r\n return this;\r\n }\r\n\r\n // We need to explicitly update the children\r\n // as the scene.evaluateActiveMeshes will not poll the transform nodes\r\n if (this._children) {\r\n for (const child of this._children) {\r\n child.markAsDirty(property);\r\n }\r\n }\r\n return super.markAsDirty(property);\r\n }\r\n\r\n /**\r\n * Defines the passed node as the parent of the current node.\r\n * The node will remain exactly where it is and its position / rotation will be updated accordingly.\r\n * Note that if the mesh has a pivot matrix / point defined it will be applied after the parent was updated.\r\n * In that case the node will not remain in the same space as it is, as the pivot will be applied.\r\n * To avoid this, you can set updatePivot to true and the pivot will be updated to identity\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/parent_pivot/parent\r\n * @param node the node ot set as the parent\r\n * @param preserveScalingSign if true, keep scaling sign of child. Otherwise, scaling sign might change.\r\n * @param updatePivot if true, update the pivot matrix to keep the node in the same space as before\r\n * @returns this TransformNode.\r\n */\r\n public setParent(node: Nullable, preserveScalingSign: boolean = false, updatePivot = false): TransformNode {\r\n if (!node && !this.parent) {\r\n return this;\r\n }\r\n\r\n const quatRotation = TmpVectors.Quaternion[0];\r\n const position = TmpVectors.Vector3[0];\r\n const scale = TmpVectors.Vector3[1];\r\n const invParentMatrix = TmpVectors.Matrix[1];\r\n Matrix.IdentityToRef(invParentMatrix);\r\n const composedMatrix = TmpVectors.Matrix[0];\r\n this.computeWorldMatrix(true);\r\n\r\n let currentRotation = this.rotationQuaternion;\r\n if (!currentRotation) {\r\n currentRotation = TransformNode._TmpRotation;\r\n Quaternion.RotationYawPitchRollToRef(this._rotation.y, this._rotation.x, this._rotation.z, currentRotation);\r\n }\r\n\r\n // current global transformation without pivot\r\n Matrix.ComposeToRef(this.scaling, currentRotation, this.position, composedMatrix);\r\n if (this.parent) {\r\n composedMatrix.multiplyToRef(this.parent.computeWorldMatrix(true), composedMatrix);\r\n }\r\n\r\n // is a node was set, calculate the difference between this and the node\r\n if (node) {\r\n node.computeWorldMatrix(true).invertToRef(invParentMatrix);\r\n composedMatrix.multiplyToRef(invParentMatrix, composedMatrix);\r\n }\r\n composedMatrix.decompose(scale, quatRotation, position, preserveScalingSign ? this : undefined);\r\n\r\n if (this.rotationQuaternion) {\r\n this.rotationQuaternion.copyFrom(quatRotation);\r\n } else {\r\n quatRotation.toEulerAnglesToRef(this.rotation);\r\n }\r\n\r\n this.scaling.copyFrom(scale);\r\n this.position.copyFrom(position);\r\n\r\n this.parent = node;\r\n\r\n if (updatePivot) {\r\n this.setPivotMatrix(Matrix.Identity());\r\n }\r\n\r\n return this;\r\n }\r\n\r\n private _nonUniformScaling = false;\r\n /**\r\n * True if the scaling property of this object is non uniform eg. (1,2,1)\r\n */\r\n public get nonUniformScaling(): boolean {\r\n return this._nonUniformScaling;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _updateNonUniformScalingState(value: boolean): boolean {\r\n if (this._nonUniformScaling === value) {\r\n return false;\r\n }\r\n\r\n this._nonUniformScaling = value;\r\n return true;\r\n }\r\n\r\n /**\r\n * Attach the current TransformNode to another TransformNode associated with a bone\r\n * @param bone Bone affecting the TransformNode\r\n * @param affectedTransformNode TransformNode associated with the bone\r\n * @returns this object\r\n */\r\n public attachToBone(bone: Bone, affectedTransformNode: TransformNode): TransformNode {\r\n this._currentParentWhenAttachingToBone = this.parent;\r\n this._transformToBoneReferal = affectedTransformNode;\r\n this.parent = bone;\r\n\r\n bone.getSkeleton().prepare(); // make sure bone.getFinalMatrix() is up to date\r\n\r\n if (bone.getFinalMatrix().determinant() < 0) {\r\n this.scalingDeterminant *= -1;\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Detach the transform node if its associated with a bone\r\n * @param resetToPreviousParent Indicates if the parent that was in effect when attachToBone was called should be set back or if we should set parent to null instead (defaults to the latter)\r\n * @returns this object\r\n */\r\n public detachFromBone(resetToPreviousParent = false): TransformNode {\r\n if (!this.parent) {\r\n if (resetToPreviousParent) {\r\n this.parent = this._currentParentWhenAttachingToBone;\r\n }\r\n return this;\r\n }\r\n\r\n if (this.parent.getWorldMatrix().determinant() < 0) {\r\n this.scalingDeterminant *= -1;\r\n }\r\n this._transformToBoneReferal = null;\r\n if (resetToPreviousParent) {\r\n this.parent = this._currentParentWhenAttachingToBone;\r\n } else {\r\n this.parent = null;\r\n }\r\n return this;\r\n }\r\n\r\n private static _RotationAxisCache = new Quaternion();\r\n /**\r\n * Rotates the mesh around the axis vector for the passed angle (amount) expressed in radians, in the given space.\r\n * space (default LOCAL) can be either Space.LOCAL, either Space.WORLD.\r\n * Note that the property `rotationQuaternion` is then automatically updated and the property `rotation` is set to (0,0,0) and no longer used.\r\n * The passed axis is also normalized.\r\n * @param axis the axis to rotate around\r\n * @param amount the amount to rotate in radians\r\n * @param space Space to rotate in (Default: local)\r\n * @returns the TransformNode.\r\n */\r\n public rotate(axis: Vector3, amount: number, space?: Space): TransformNode {\r\n axis.normalize();\r\n if (!this.rotationQuaternion) {\r\n this.rotationQuaternion = this.rotation.toQuaternion();\r\n this.rotation.setAll(0);\r\n }\r\n let rotationQuaternion: Quaternion;\r\n if (!space || (space as any) === Space.LOCAL) {\r\n rotationQuaternion = Quaternion.RotationAxisToRef(axis, amount, TransformNode._RotationAxisCache);\r\n this.rotationQuaternion.multiplyToRef(rotationQuaternion, this.rotationQuaternion);\r\n } else {\r\n if (this.parent) {\r\n const invertParentWorldMatrix = TmpVectors.Matrix[0];\r\n this.parent.getWorldMatrix().invertToRef(invertParentWorldMatrix);\r\n axis = Vector3.TransformNormal(axis, invertParentWorldMatrix);\r\n }\r\n rotationQuaternion = Quaternion.RotationAxisToRef(axis, amount, TransformNode._RotationAxisCache);\r\n rotationQuaternion.multiplyToRef(this.rotationQuaternion, this.rotationQuaternion);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Rotates the mesh around the axis vector for the passed angle (amount) expressed in radians, in world space.\r\n * Note that the property `rotationQuaternion` is then automatically updated and the property `rotation` is set to (0,0,0) and no longer used.\r\n * The passed axis is also normalized. .\r\n * Method is based on http://www.euclideanspace.com/maths/geometry/affine/aroundPoint/index.htm\r\n * @param point the point to rotate around\r\n * @param axis the axis to rotate around\r\n * @param amount the amount to rotate in radians\r\n * @returns the TransformNode\r\n */\r\n public rotateAround(point: Vector3, axis: Vector3, amount: number): TransformNode {\r\n axis.normalize();\r\n if (!this.rotationQuaternion) {\r\n this.rotationQuaternion = Quaternion.RotationYawPitchRoll(this.rotation.y, this.rotation.x, this.rotation.z);\r\n this.rotation.setAll(0);\r\n }\r\n\r\n const tmpVector = TmpVectors.Vector3[0];\r\n const finalScale = TmpVectors.Vector3[1];\r\n const finalTranslation = TmpVectors.Vector3[2];\r\n\r\n const finalRotation = TmpVectors.Quaternion[0];\r\n\r\n const translationMatrix = TmpVectors.Matrix[0]; // T\r\n const translationMatrixInv = TmpVectors.Matrix[1]; // T'\r\n const rotationMatrix = TmpVectors.Matrix[2]; // R\r\n const finalMatrix = TmpVectors.Matrix[3]; // T' x R x T\r\n\r\n point.subtractToRef(this.position, tmpVector);\r\n Matrix.TranslationToRef(tmpVector.x, tmpVector.y, tmpVector.z, translationMatrix); // T\r\n Matrix.TranslationToRef(-tmpVector.x, -tmpVector.y, -tmpVector.z, translationMatrixInv); // T'\r\n Matrix.RotationAxisToRef(axis, amount, rotationMatrix); // R\r\n\r\n translationMatrixInv.multiplyToRef(rotationMatrix, finalMatrix); // T' x R\r\n finalMatrix.multiplyToRef(translationMatrix, finalMatrix); // T' x R x T\r\n\r\n finalMatrix.decompose(finalScale, finalRotation, finalTranslation);\r\n\r\n this.position.addInPlace(finalTranslation);\r\n finalRotation.multiplyToRef(this.rotationQuaternion, this.rotationQuaternion);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Translates the mesh along the axis vector for the passed distance in the given space.\r\n * space (default LOCAL) can be either Space.LOCAL, either Space.WORLD.\r\n * @param axis the axis to translate in\r\n * @param distance the distance to translate\r\n * @param space Space to rotate in (Default: local)\r\n * @returns the TransformNode.\r\n */\r\n public translate(axis: Vector3, distance: number, space?: Space): TransformNode {\r\n const displacementVector = axis.scale(distance);\r\n if (!space || (space as any) === Space.LOCAL) {\r\n const tempV3 = this.getPositionExpressedInLocalSpace().add(displacementVector);\r\n this.setPositionWithLocalVector(tempV3);\r\n } else {\r\n this.setAbsolutePosition(this.getAbsolutePosition().add(displacementVector));\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds a rotation step to the mesh current rotation.\r\n * x, y, z are Euler angles expressed in radians.\r\n * This methods updates the current mesh rotation, either mesh.rotation, either mesh.rotationQuaternion if it's set.\r\n * This means this rotation is made in the mesh local space only.\r\n * It's useful to set a custom rotation order different from the BJS standard one YXZ.\r\n * Example : this rotates the mesh first around its local X axis, then around its local Z axis, finally around its local Y axis.\r\n * ```javascript\r\n * mesh.addRotation(x1, 0, 0).addRotation(0, 0, z2).addRotation(0, 0, y3);\r\n * ```\r\n * Note that `addRotation()` accumulates the passed rotation values to the current ones and computes the .rotation or .rotationQuaternion updated values.\r\n * Under the hood, only quaternions are used. So it's a little faster is you use .rotationQuaternion because it doesn't need to translate them back to Euler angles.\r\n * @param x Rotation to add\r\n * @param y Rotation to add\r\n * @param z Rotation to add\r\n * @returns the TransformNode.\r\n */\r\n public addRotation(x: number, y: number, z: number): TransformNode {\r\n let rotationQuaternion;\r\n if (this.rotationQuaternion) {\r\n rotationQuaternion = this.rotationQuaternion;\r\n } else {\r\n rotationQuaternion = TmpVectors.Quaternion[1];\r\n Quaternion.RotationYawPitchRollToRef(this.rotation.y, this.rotation.x, this.rotation.z, rotationQuaternion);\r\n }\r\n const accumulation = TmpVectors.Quaternion[0];\r\n Quaternion.RotationYawPitchRollToRef(y, x, z, accumulation);\r\n rotationQuaternion.multiplyInPlace(accumulation);\r\n if (!this.rotationQuaternion) {\r\n rotationQuaternion.toEulerAnglesToRef(this.rotation);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n protected _getEffectiveParent(): Nullable {\r\n return this.parent;\r\n }\r\n\r\n /**\r\n * Returns whether the transform node world matrix computation needs the camera information to be computed.\r\n * This is the case when the node is a billboard or has an infinite distance for instance.\r\n * @returns true if the world matrix computation needs the camera information to be computed\r\n */\r\n public isWorldMatrixCameraDependent(): boolean {\r\n return (this._infiniteDistance && !this.parent) || (this._billboardMode !== TransformNode.BILLBOARDMODE_NONE && !this.preserveParentRotationForBillboard);\r\n }\r\n\r\n /**\r\n * Computes the world matrix of the node\r\n * @param force defines if the cache version should be invalidated forcing the world matrix to be created from scratch\r\n * @param camera defines the camera used if different from the scene active camera (This is used with modes like Billboard or infinite distance)\r\n * @returns the world matrix\r\n */\r\n public computeWorldMatrix(force: boolean = false, camera: Nullable = null): Matrix {\r\n if (this._isWorldMatrixFrozen && !this._isDirty) {\r\n return this._worldMatrix;\r\n }\r\n\r\n const currentRenderId = this.getScene().getRenderId();\r\n if (!this._isDirty && !force && (this._currentRenderId === currentRenderId || this.isSynchronized())) {\r\n this._currentRenderId = currentRenderId;\r\n return this._worldMatrix;\r\n }\r\n\r\n camera = camera || this.getScene().activeCamera;\r\n\r\n this._updateCache();\r\n const cache = this._cache;\r\n cache.pivotMatrixUpdated = false;\r\n cache.billboardMode = this.billboardMode;\r\n cache.infiniteDistance = this.infiniteDistance;\r\n cache.parent = this._parentNode;\r\n\r\n this._currentRenderId = currentRenderId;\r\n this._childUpdateId += 1;\r\n this._isDirty = false;\r\n this._position._isDirty = false;\r\n this._rotation._isDirty = false;\r\n this._scaling._isDirty = false;\r\n const parent = this._getEffectiveParent();\r\n\r\n // Scaling\r\n const scaling: Vector3 = TransformNode._TmpScaling;\r\n let translation: Vector3 = this._position;\r\n\r\n // Translation\r\n if (this._infiniteDistance) {\r\n if (!this.parent && camera) {\r\n const cameraWorldMatrix = camera.getWorldMatrix();\r\n const cameraGlobalPosition = new Vector3(cameraWorldMatrix.m[12], cameraWorldMatrix.m[13], cameraWorldMatrix.m[14]);\r\n\r\n translation = TransformNode._TmpTranslation;\r\n translation.copyFromFloats(this._position.x + cameraGlobalPosition.x, this._position.y + cameraGlobalPosition.y, this._position.z + cameraGlobalPosition.z);\r\n }\r\n }\r\n\r\n // Scaling\r\n scaling.copyFromFloats(this._scaling.x * this.scalingDeterminant, this._scaling.y * this.scalingDeterminant, this._scaling.z * this.scalingDeterminant);\r\n\r\n // Rotation\r\n let rotation: Quaternion;\r\n if (this._rotationQuaternion) {\r\n this._rotationQuaternion._isDirty = false;\r\n rotation = this._rotationQuaternion;\r\n if (this.reIntegrateRotationIntoRotationQuaternion) {\r\n const len = this.rotation.lengthSquared();\r\n if (len) {\r\n this._rotationQuaternion.multiplyInPlace(Quaternion.RotationYawPitchRoll(this._rotation.y, this._rotation.x, this._rotation.z));\r\n this._rotation.copyFromFloats(0, 0, 0);\r\n }\r\n }\r\n } else {\r\n rotation = TransformNode._TmpRotation;\r\n Quaternion.RotationYawPitchRollToRef(this._rotation.y, this._rotation.x, this._rotation.z, rotation);\r\n }\r\n\r\n // Compose\r\n if (this._usePivotMatrix) {\r\n const scaleMatrix = TmpVectors.Matrix[1];\r\n Matrix.ScalingToRef(scaling.x, scaling.y, scaling.z, scaleMatrix);\r\n\r\n // Rotation\r\n const rotationMatrix = TmpVectors.Matrix[0];\r\n rotation.toRotationMatrix(rotationMatrix);\r\n\r\n // Composing transformations\r\n this._pivotMatrix.multiplyToRef(scaleMatrix, TmpVectors.Matrix[4]);\r\n TmpVectors.Matrix[4].multiplyToRef(rotationMatrix, this._localMatrix);\r\n\r\n // Post multiply inverse of pivotMatrix\r\n if (this._postMultiplyPivotMatrix) {\r\n this._localMatrix.multiplyToRef(this._pivotMatrixInverse, this._localMatrix);\r\n }\r\n\r\n this._localMatrix.addTranslationFromFloats(translation.x, translation.y, translation.z);\r\n } else {\r\n Matrix.ComposeToRef(scaling, rotation, translation, this._localMatrix);\r\n }\r\n\r\n // Parent\r\n if (parent && parent.getWorldMatrix) {\r\n if (force) {\r\n parent.computeWorldMatrix(force);\r\n }\r\n if (cache.useBillboardPath) {\r\n if (this._transformToBoneReferal) {\r\n parent.getWorldMatrix().multiplyToRef(this._transformToBoneReferal.getWorldMatrix(), TmpVectors.Matrix[7]);\r\n } else {\r\n TmpVectors.Matrix[7].copyFrom(parent.getWorldMatrix());\r\n }\r\n\r\n // Extract scaling and translation from parent\r\n const translation = TmpVectors.Vector3[5];\r\n const scale = TmpVectors.Vector3[6];\r\n const orientation = TmpVectors.Quaternion[0];\r\n TmpVectors.Matrix[7].decompose(scale, orientation, translation);\r\n Matrix.ScalingToRef(scale.x, scale.y, scale.z, TmpVectors.Matrix[7]);\r\n TmpVectors.Matrix[7].setTranslation(translation);\r\n\r\n if (TransformNode.BillboardUseParentOrientation) {\r\n // set localMatrix translation to be transformed against parent's orientation.\r\n this._position.applyRotationQuaternionToRef(orientation, translation);\r\n this._localMatrix.setTranslation(translation);\r\n }\r\n\r\n this._localMatrix.multiplyToRef(TmpVectors.Matrix[7], this._worldMatrix);\r\n } else {\r\n if (this._transformToBoneReferal) {\r\n this._localMatrix.multiplyToRef(parent.getWorldMatrix(), TmpVectors.Matrix[6]);\r\n TmpVectors.Matrix[6].multiplyToRef(this._transformToBoneReferal.getWorldMatrix(), this._worldMatrix);\r\n } else {\r\n this._localMatrix.multiplyToRef(parent.getWorldMatrix(), this._worldMatrix);\r\n }\r\n }\r\n this._markSyncedWithParent();\r\n } else {\r\n this._worldMatrix.copyFrom(this._localMatrix);\r\n }\r\n\r\n // Billboarding based on camera orientation (testing PG:http://www.babylonjs-playground.com/#UJEIL#13)\r\n if (cache.useBillboardPath && camera && this.billboardMode && !cache.useBillboardPosition) {\r\n const storedTranslation = TmpVectors.Vector3[0];\r\n this._worldMatrix.getTranslationToRef(storedTranslation); // Save translation\r\n\r\n // Cancel camera rotation\r\n TmpVectors.Matrix[1].copyFrom(camera.getViewMatrix());\r\n\r\n if (this._scene.useRightHandedSystem) {\r\n TmpVectors.Matrix[1].multiplyToRef(convertRHSToLHS, TmpVectors.Matrix[1]);\r\n }\r\n\r\n TmpVectors.Matrix[1].setTranslationFromFloats(0, 0, 0);\r\n TmpVectors.Matrix[1].invertToRef(TmpVectors.Matrix[0]);\r\n\r\n if ((this.billboardMode & TransformNode.BILLBOARDMODE_ALL) !== TransformNode.BILLBOARDMODE_ALL) {\r\n TmpVectors.Matrix[0].decompose(undefined, TmpVectors.Quaternion[0], undefined);\r\n const eulerAngles = TmpVectors.Vector3[1];\r\n TmpVectors.Quaternion[0].toEulerAnglesToRef(eulerAngles);\r\n\r\n if ((this.billboardMode & TransformNode.BILLBOARDMODE_X) !== TransformNode.BILLBOARDMODE_X) {\r\n eulerAngles.x = 0;\r\n }\r\n\r\n if ((this.billboardMode & TransformNode.BILLBOARDMODE_Y) !== TransformNode.BILLBOARDMODE_Y) {\r\n eulerAngles.y = 0;\r\n }\r\n\r\n if ((this.billboardMode & TransformNode.BILLBOARDMODE_Z) !== TransformNode.BILLBOARDMODE_Z) {\r\n eulerAngles.z = 0;\r\n }\r\n\r\n Matrix.RotationYawPitchRollToRef(eulerAngles.y, eulerAngles.x, eulerAngles.z, TmpVectors.Matrix[0]);\r\n }\r\n this._worldMatrix.setTranslationFromFloats(0, 0, 0);\r\n this._worldMatrix.multiplyToRef(TmpVectors.Matrix[0], this._worldMatrix);\r\n\r\n // Restore translation\r\n this._worldMatrix.setTranslation(TmpVectors.Vector3[0]);\r\n }\r\n // Billboarding based on camera position\r\n else if (cache.useBillboardPath && camera && cache.useBillboardPosition) {\r\n const storedTranslation = TmpVectors.Vector3[0];\r\n // Save translation\r\n this._worldMatrix.getTranslationToRef(storedTranslation);\r\n\r\n // Compute camera position in local space\r\n const cameraPosition = camera.globalPosition;\r\n this._worldMatrix.invertToRef(TmpVectors.Matrix[1]);\r\n const camInObjSpace = TmpVectors.Vector3[1];\r\n Vector3.TransformCoordinatesToRef(cameraPosition, TmpVectors.Matrix[1], camInObjSpace);\r\n camInObjSpace.normalize();\r\n\r\n // Find the lookAt info in local space\r\n const yaw = -Math.atan2(camInObjSpace.z, camInObjSpace.x) + Math.PI / 2;\r\n const len = Math.sqrt(camInObjSpace.x * camInObjSpace.x + camInObjSpace.z * camInObjSpace.z);\r\n const pitch = -Math.atan2(camInObjSpace.y, len);\r\n Quaternion.RotationYawPitchRollToRef(yaw, pitch, 0, TmpVectors.Quaternion[0]);\r\n\r\n if ((this.billboardMode & TransformNode.BILLBOARDMODE_ALL) !== TransformNode.BILLBOARDMODE_ALL) {\r\n const eulerAngles = TmpVectors.Vector3[1];\r\n TmpVectors.Quaternion[0].toEulerAnglesToRef(eulerAngles);\r\n\r\n if ((this.billboardMode & TransformNode.BILLBOARDMODE_X) !== TransformNode.BILLBOARDMODE_X) {\r\n eulerAngles.x = 0;\r\n }\r\n\r\n if ((this.billboardMode & TransformNode.BILLBOARDMODE_Y) !== TransformNode.BILLBOARDMODE_Y) {\r\n eulerAngles.y = 0;\r\n }\r\n\r\n if ((this.billboardMode & TransformNode.BILLBOARDMODE_Z) !== TransformNode.BILLBOARDMODE_Z) {\r\n eulerAngles.z = 0;\r\n }\r\n\r\n Matrix.RotationYawPitchRollToRef(eulerAngles.y, eulerAngles.x, eulerAngles.z, TmpVectors.Matrix[0]);\r\n } else {\r\n Matrix.FromQuaternionToRef(TmpVectors.Quaternion[0], TmpVectors.Matrix[0]);\r\n }\r\n\r\n // Cancel translation\r\n this._worldMatrix.setTranslationFromFloats(0, 0, 0);\r\n\r\n // Rotate according to lookat (diff from local to lookat)\r\n this._worldMatrix.multiplyToRef(TmpVectors.Matrix[0], this._worldMatrix);\r\n\r\n // Restore translation\r\n this._worldMatrix.setTranslation(TmpVectors.Vector3[0]);\r\n }\r\n\r\n // Normal matrix\r\n if (!this.ignoreNonUniformScaling) {\r\n if (this._scaling.isNonUniformWithinEpsilon(0.000001)) {\r\n this._updateNonUniformScalingState(true);\r\n } else if (parent && (parent)._nonUniformScaling) {\r\n this._updateNonUniformScalingState((parent)._nonUniformScaling);\r\n } else {\r\n this._updateNonUniformScalingState(false);\r\n }\r\n } else {\r\n this._updateNonUniformScalingState(false);\r\n }\r\n\r\n this._afterComputeWorldMatrix();\r\n\r\n // Absolute position\r\n this._absolutePosition.copyFromFloats(this._worldMatrix.m[12], this._worldMatrix.m[13], this._worldMatrix.m[14]);\r\n this._isAbsoluteSynced = false;\r\n\r\n // Callbacks\r\n this.onAfterWorldMatrixUpdateObservable.notifyObservers(this);\r\n\r\n if (!this._poseMatrix) {\r\n this._poseMatrix = Matrix.Invert(this._worldMatrix);\r\n }\r\n\r\n // Cache the determinant\r\n this._worldMatrixDeterminantIsDirty = true;\r\n\r\n return this._worldMatrix;\r\n }\r\n\r\n /**\r\n * Resets this nodeTransform's local matrix to Matrix.Identity().\r\n * @param independentOfChildren indicates if all child nodeTransform's world-space transform should be preserved.\r\n */\r\n public resetLocalMatrix(independentOfChildren: boolean = true): void {\r\n this.computeWorldMatrix();\r\n if (independentOfChildren) {\r\n const children = this.getChildren();\r\n for (let i = 0; i < children.length; ++i) {\r\n const child = children[i] as TransformNode;\r\n if (child) {\r\n child.computeWorldMatrix();\r\n const bakedMatrix = TmpVectors.Matrix[0];\r\n child._localMatrix.multiplyToRef(this._localMatrix, bakedMatrix);\r\n const tmpRotationQuaternion = TmpVectors.Quaternion[0];\r\n bakedMatrix.decompose(child.scaling, tmpRotationQuaternion, child.position);\r\n if (child.rotationQuaternion) {\r\n child.rotationQuaternion.copyFrom(tmpRotationQuaternion);\r\n } else {\r\n tmpRotationQuaternion.toEulerAnglesToRef(child.rotation);\r\n }\r\n }\r\n }\r\n }\r\n this.scaling.copyFromFloats(1, 1, 1);\r\n this.position.copyFromFloats(0, 0, 0);\r\n this.rotation.copyFromFloats(0, 0, 0);\r\n\r\n //only if quaternion is already set\r\n if (this.rotationQuaternion) {\r\n this.rotationQuaternion = Quaternion.Identity();\r\n }\r\n this._worldMatrix = Matrix.Identity();\r\n }\r\n\r\n protected _afterComputeWorldMatrix(): void {}\r\n\r\n /**\r\n * If you'd like to be called back after the mesh position, rotation or scaling has been updated.\r\n * @param func callback function to add\r\n *\r\n * @returns the TransformNode.\r\n */\r\n public registerAfterWorldMatrixUpdate(func: (mesh: TransformNode) => void): TransformNode {\r\n this.onAfterWorldMatrixUpdateObservable.add(func);\r\n return this;\r\n }\r\n\r\n /**\r\n * Removes a registered callback function.\r\n * @param func callback function to remove\r\n * @returns the TransformNode.\r\n */\r\n public unregisterAfterWorldMatrixUpdate(func: (mesh: TransformNode) => void): TransformNode {\r\n this.onAfterWorldMatrixUpdateObservable.removeCallback(func);\r\n return this;\r\n }\r\n\r\n /**\r\n * Gets the position of the current mesh in camera space\r\n * @param camera defines the camera to use\r\n * @returns a position\r\n */\r\n public getPositionInCameraSpace(camera: Nullable = null): Vector3 {\r\n if (!camera) {\r\n camera = this.getScene().activeCamera;\r\n }\r\n\r\n return Vector3.TransformCoordinates(this.getAbsolutePosition(), camera.getViewMatrix());\r\n }\r\n\r\n /**\r\n * Returns the distance from the mesh to the active camera\r\n * @param camera defines the camera to use\r\n * @returns the distance\r\n */\r\n public getDistanceToCamera(camera: Nullable = null): number {\r\n if (!camera) {\r\n camera = this.getScene().activeCamera;\r\n }\r\n return this.getAbsolutePosition().subtract(camera.globalPosition).length();\r\n }\r\n\r\n /**\r\n * Clone the current transform node\r\n * @param name Name of the new clone\r\n * @param newParent New parent for the clone\r\n * @param doNotCloneChildren Do not clone children hierarchy\r\n * @returns the new transform node\r\n */\r\n public clone(name: string, newParent: Nullable, doNotCloneChildren?: boolean): Nullable {\r\n const result = SerializationHelper.Clone(() => new TransformNode(name, this.getScene()), this);\r\n\r\n result.name = name;\r\n result.id = name;\r\n\r\n if (newParent) {\r\n result.parent = newParent;\r\n }\r\n\r\n if (!doNotCloneChildren) {\r\n // Children\r\n const directDescendants = this.getDescendants(true);\r\n for (let index = 0; index < directDescendants.length; index++) {\r\n const child = directDescendants[index];\r\n\r\n if ((child).clone) {\r\n (child).clone(name + \".\" + child.name, result);\r\n }\r\n }\r\n }\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Serializes the objects information.\r\n * @param currentSerializationObject defines the object to serialize in\r\n * @returns the serialized object\r\n */\r\n public serialize(currentSerializationObject?: any): any {\r\n const serializationObject = SerializationHelper.Serialize(this, currentSerializationObject);\r\n serializationObject.type = this.getClassName();\r\n serializationObject.uniqueId = this.uniqueId;\r\n\r\n // Parent\r\n if (this.parent) {\r\n this.parent._serializeAsParent(serializationObject);\r\n }\r\n\r\n serializationObject.localMatrix = this.getPivotMatrix().asArray();\r\n\r\n serializationObject.isEnabled = this.isEnabled();\r\n\r\n return serializationObject;\r\n }\r\n\r\n // Statics\r\n /**\r\n * Returns a new TransformNode object parsed from the source provided.\r\n * @param parsedTransformNode is the source.\r\n * @param scene the scene the object belongs to\r\n * @param rootUrl is a string, it's the root URL to prefix the `delayLoadingFile` property with\r\n * @returns a new TransformNode object parsed from the source provided.\r\n */\r\n public static Parse(parsedTransformNode: any, scene: Scene, rootUrl: string): TransformNode {\r\n const transformNode = SerializationHelper.Parse(() => new TransformNode(parsedTransformNode.name, scene), parsedTransformNode, scene, rootUrl);\r\n\r\n if (parsedTransformNode.localMatrix) {\r\n transformNode.setPreTransformMatrix(Matrix.FromArray(parsedTransformNode.localMatrix));\r\n } else if (parsedTransformNode.pivotMatrix) {\r\n transformNode.setPivotMatrix(Matrix.FromArray(parsedTransformNode.pivotMatrix));\r\n }\r\n\r\n transformNode.setEnabled(parsedTransformNode.isEnabled);\r\n\r\n transformNode._waitingParsedUniqueId = parsedTransformNode.uniqueId;\r\n\r\n // Parent\r\n if (parsedTransformNode.parentId !== undefined) {\r\n transformNode._waitingParentId = parsedTransformNode.parentId;\r\n }\r\n\r\n if (parsedTransformNode.parentInstanceIndex !== undefined) {\r\n transformNode._waitingParentInstanceIndex = parsedTransformNode.parentInstanceIndex;\r\n }\r\n\r\n return transformNode;\r\n }\r\n\r\n /**\r\n * Get all child-transformNodes of this node\r\n * @param directDescendantsOnly defines if true only direct descendants of 'this' will be considered, if false direct and also indirect (children of children, an so on in a recursive manner) descendants of 'this' will be considered\r\n * @param predicate defines an optional predicate that will be called on every evaluated child, the predicate must return true for a given child to be part of the result, otherwise it will be ignored\r\n * @returns an array of TransformNode\r\n */\r\n public getChildTransformNodes(directDescendantsOnly?: boolean, predicate?: (node: Node) => boolean): TransformNode[] {\r\n const results: Array = [];\r\n this._getDescendants(results, directDescendantsOnly, (node: Node) => {\r\n return (!predicate || predicate(node)) && node instanceof TransformNode;\r\n });\r\n return results;\r\n }\r\n\r\n /**\r\n * Releases resources associated with this transform node.\r\n * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default)\r\n * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default)\r\n */\r\n public dispose(doNotRecurse?: boolean, disposeMaterialAndTextures = false): void {\r\n // Animations\r\n this.getScene().stopAnimation(this);\r\n\r\n // Remove from scene\r\n this.getScene().removeTransformNode(this);\r\n\r\n if (this._parentContainer) {\r\n const index = this._parentContainer.transformNodes.indexOf(this);\r\n if (index > -1) {\r\n this._parentContainer.transformNodes.splice(index, 1);\r\n }\r\n this._parentContainer = null;\r\n }\r\n\r\n this.onAfterWorldMatrixUpdateObservable.clear();\r\n\r\n if (doNotRecurse) {\r\n const transformNodes = this.getChildTransformNodes(true);\r\n for (const transformNode of transformNodes) {\r\n transformNode.parent = null;\r\n transformNode.computeWorldMatrix(true);\r\n }\r\n }\r\n\r\n super.dispose(doNotRecurse, disposeMaterialAndTextures);\r\n }\r\n\r\n /**\r\n * Uniformly scales the mesh to fit inside of a unit cube (1 X 1 X 1 units)\r\n * @param includeDescendants Use the hierarchy's bounding box instead of the mesh's bounding box. Default is false\r\n * @param ignoreRotation ignore rotation when computing the scale (ie. object will be axis aligned). Default is false\r\n * @param predicate predicate that is passed in to getHierarchyBoundingVectors when selecting which object should be included when scaling\r\n * @returns the current mesh\r\n */\r\n public normalizeToUnitCube(includeDescendants = true, ignoreRotation = false, predicate?: Nullable<(node: AbstractMesh) => boolean>): TransformNode {\r\n let storedRotation: Nullable = null;\r\n let storedRotationQuaternion: Nullable = null;\r\n\r\n if (ignoreRotation) {\r\n if (this.rotationQuaternion) {\r\n storedRotationQuaternion = this.rotationQuaternion.clone();\r\n this.rotationQuaternion.copyFromFloats(0, 0, 0, 1);\r\n } else if (this.rotation) {\r\n storedRotation = this.rotation.clone();\r\n this.rotation.copyFromFloats(0, 0, 0);\r\n }\r\n }\r\n\r\n const boundingVectors = this.getHierarchyBoundingVectors(includeDescendants, predicate);\r\n const sizeVec = boundingVectors.max.subtract(boundingVectors.min);\r\n const maxDimension = Math.max(sizeVec.x, sizeVec.y, sizeVec.z);\r\n\r\n if (maxDimension === 0) {\r\n return this;\r\n }\r\n\r\n const scale = 1 / maxDimension;\r\n\r\n this.scaling.scaleInPlace(scale);\r\n\r\n if (ignoreRotation) {\r\n if (this.rotationQuaternion && storedRotationQuaternion) {\r\n this.rotationQuaternion.copyFrom(storedRotationQuaternion);\r\n } else if (this.rotation && storedRotation) {\r\n this.rotation.copyFrom(storedRotation);\r\n }\r\n }\r\n\r\n return this;\r\n }\r\n\r\n private _syncAbsoluteScalingAndRotation(): void {\r\n if (!this._isAbsoluteSynced) {\r\n this._worldMatrix.decompose(this._absoluteScaling, this._absoluteRotationQuaternion);\r\n this._isAbsoluteSynced = true;\r\n }\r\n }\r\n}\r\n","import type { Collider } from \"./collider\";\r\nimport { Vector3 } from \"../Maths/math.vector\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { Observer } from \"../Misc/observable\";\r\n\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\n\r\n/**\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class _MeshCollisionData {\r\n public _checkCollisions = false;\r\n public _collisionMask = -1;\r\n public _collisionGroup = -1;\r\n public _surroundingMeshes: Nullable = null;\r\n public _collider: Nullable = null;\r\n public _oldPositionForCollisions = new Vector3(0, 0, 0);\r\n public _diffPositionForCollisions = new Vector3(0, 0, 0);\r\n public _onCollideObserver: Nullable>;\r\n public _onCollisionPositionChangeObserver: Nullable>;\r\n public _collisionResponse = true;\r\n}\r\n","import { Observable } from \"../Misc/observable\";\r\nimport type { Nullable, FloatArray, IndicesArray, DeepImmutable } from \"../types\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\nimport type { Scene, IDisposable } from \"../scene\";\r\nimport { ScenePerformancePriority } from \"../scene\";\r\nimport type { Vector2 } from \"../Maths/math.vector\";\r\nimport { Quaternion, Matrix, Vector3, TmpVectors } from \"../Maths/math.vector\";\r\nimport { Engine } from \"../Engines/engine\";\r\nimport type { Node } from \"../node\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport type { IGetSetVerticesData } from \"../Meshes/mesh.vertexData\";\r\nimport { VertexData } from \"../Meshes/mesh.vertexData\";\r\nimport { TransformNode } from \"../Meshes/transformNode\";\r\nimport type { SubMesh } from \"../Meshes/subMesh\";\r\nimport { PickingInfo } from \"../Collisions/pickingInfo\";\r\nimport type { IntersectionInfo } from \"../Collisions/intersectionInfo\";\r\nimport type { ICullable } from \"../Culling/boundingInfo\";\r\nimport { BoundingInfo } from \"../Culling/boundingInfo\";\r\nimport type { Material } from \"../Materials/material\";\r\nimport type { MaterialDefines } from \"../Materials/materialDefines\";\r\nimport type { Light } from \"../Lights/light\";\r\nimport type { Skeleton } from \"../Bones/skeleton\";\r\nimport type { MorphTargetManager } from \"../Morph/morphTargetManager\";\r\nimport type { IBakedVertexAnimationManager } from \"../BakedVertexAnimation/bakedVertexAnimationManager\";\r\nimport type { IEdgesRenderer } from \"../Rendering/edgesRenderer\";\r\nimport type { SolidParticle } from \"../Particles/solidParticle\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport type { AbstractActionManager } from \"../Actions/abstractActionManager\";\r\nimport { UniformBuffer } from \"../Materials/uniformBuffer\";\r\nimport { _MeshCollisionData } from \"../Collisions/meshCollisionData\";\r\nimport { _WarnImport } from \"../Misc/devTools\";\r\nimport type { RawTexture } from \"../Materials/Textures/rawTexture\";\r\nimport { extractMinAndMax } from \"../Maths/math.functions\";\r\nimport { Color3, Color4 } from \"../Maths/math.color\";\r\nimport { Epsilon } from \"../Maths/math.constants\";\r\nimport type { Plane } from \"../Maths/math.plane\";\r\nimport { Axis } from \"../Maths/math.axis\";\r\nimport type { IParticleSystem } from \"../Particles/IParticleSystem\";\r\nimport { RegisterClass } from \"../Misc/typeStore\";\r\n\r\nimport type { Ray } from \"../Culling/ray\";\r\nimport type { Collider } from \"../Collisions/collider\";\r\nimport type { TrianglePickingPredicate } from \"../Culling/ray\";\r\nimport type { RenderingGroup } from \"../Rendering/renderingGroup\";\r\nimport type { IEdgesRendererOptions } from \"../Rendering/edgesRenderer\";\r\n\r\n/** @internal */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nclass _FacetDataStorage {\r\n // facetData private properties\r\n public facetPositions: Vector3[]; // facet local positions\r\n public facetNormals: Vector3[]; // facet local normals\r\n public facetPartitioning: number[][]; // partitioning array of facet index arrays\r\n public facetNb: number = 0; // facet number\r\n public partitioningSubdivisions: number = 10; // number of subdivisions per axis in the partitioning space\r\n public partitioningBBoxRatio: number = 1.01; // the partitioning array space is by default 1% bigger than the bounding box\r\n public facetDataEnabled: boolean = false; // is the facet data feature enabled on this mesh ?\r\n public facetParameters: any = {}; // keep a reference to the object parameters to avoid memory re-allocation\r\n public bbSize: Vector3 = Vector3.Zero(); // bbox size approximated for facet data\r\n public subDiv = {\r\n // actual number of subdivisions per axis for ComputeNormals()\r\n max: 1,\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n X: 1,\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n Y: 1,\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n Z: 1,\r\n };\r\n\r\n public facetDepthSort: boolean = false; // is the facet depth sort to be computed\r\n public facetDepthSortEnabled: boolean = false; // is the facet depth sort initialized\r\n public depthSortedIndices: IndicesArray; // copy of the indices array to store them once sorted\r\n public depthSortedFacets: { ind: number; sqDistance: number }[]; // array of depth sorted facets\r\n public facetDepthSortFunction: (f1: { ind: number; sqDistance: number }, f2: { ind: number; sqDistance: number }) => number; // facet depth sort function\r\n public facetDepthSortFrom: Vector3; // location where to depth sort from\r\n public facetDepthSortOrigin: Vector3; // same as facetDepthSortFrom but expressed in the mesh local space\r\n\r\n public invertedMatrix: Matrix; // Inverted world matrix.\r\n}\r\n\r\n/**\r\n * @internal\r\n **/\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nclass _InternalAbstractMeshDataInfo {\r\n public _hasVertexAlpha = false;\r\n public _useVertexColors = true;\r\n public _numBoneInfluencers = 4;\r\n public _applyFog = true;\r\n public _receiveShadows = false;\r\n public _facetData = new _FacetDataStorage();\r\n public _visibility = 1.0;\r\n public _skeleton: Nullable = null;\r\n public _layerMask: number = 0x0fffffff;\r\n public _computeBonesUsingShaders = true;\r\n public _isActive = false;\r\n public _onlyForInstances = false;\r\n public _isActiveIntermediate = false;\r\n public _onlyForInstancesIntermediate = false;\r\n public _actAsRegularMesh = false;\r\n public _currentLOD: Nullable = null;\r\n public _currentLODIsUpToDate: boolean = false;\r\n public _collisionRetryCount: number = 3;\r\n public _morphTargetManager: Nullable = null;\r\n public _renderingGroupId = 0;\r\n public _bakedVertexAnimationManager: Nullable = null;\r\n public _material: Nullable = null;\r\n public _materialForRenderPass: Array; // map a render pass id (index in the array) to a Material\r\n public _positions: Nullable = null;\r\n public _pointerOverDisableMeshTesting: boolean = false;\r\n // Collisions\r\n public _meshCollisionData = new _MeshCollisionData();\r\n public _enableDistantPicking = false;\r\n /** @internal\r\n * Bounding info that is unnafected by the addition of thin instances\r\n */\r\n public _rawBoundingInfo: Nullable = null;\r\n}\r\n\r\n/**\r\n * Class used to store all common mesh properties\r\n */\r\nexport class AbstractMesh extends TransformNode implements IDisposable, ICullable, IGetSetVerticesData {\r\n /** No occlusion */\r\n public static OCCLUSION_TYPE_NONE = 0;\r\n /** Occlusion set to optimistic */\r\n public static OCCLUSION_TYPE_OPTIMISTIC = 1;\r\n /** Occlusion set to strict */\r\n public static OCCLUSION_TYPE_STRICT = 2;\r\n /** Use an accurate occlusion algorithm */\r\n public static OCCLUSION_ALGORITHM_TYPE_ACCURATE = 0;\r\n /** Use a conservative occlusion algorithm */\r\n public static OCCLUSION_ALGORITHM_TYPE_CONSERVATIVE = 1;\r\n\r\n /** Default culling strategy : this is an exclusion test and it's the more accurate.\r\n * Test order :\r\n * Is the bounding sphere outside the frustum ?\r\n * If not, are the bounding box vertices outside the frustum ?\r\n * It not, then the cullable object is in the frustum.\r\n */\r\n public static readonly CULLINGSTRATEGY_STANDARD = Constants.MESHES_CULLINGSTRATEGY_STANDARD;\r\n /** Culling strategy : Bounding Sphere Only.\r\n * This is an exclusion test. It's faster than the standard strategy because the bounding box is not tested.\r\n * It's also less accurate than the standard because some not visible objects can still be selected.\r\n * Test : is the bounding sphere outside the frustum ?\r\n * If not, then the cullable object is in the frustum.\r\n */\r\n public static readonly CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY = Constants.MESHES_CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY;\r\n /** Culling strategy : Optimistic Inclusion.\r\n * This in an inclusion test first, then the standard exclusion test.\r\n * This can be faster when a cullable object is expected to be almost always in the camera frustum.\r\n * This could also be a little slower than the standard test when the tested object center is not the frustum but one of its bounding box vertex is still inside.\r\n * Anyway, it's as accurate as the standard strategy.\r\n * Test :\r\n * Is the cullable object bounding sphere center in the frustum ?\r\n * If not, apply the default culling strategy.\r\n */\r\n public static readonly CULLINGSTRATEGY_OPTIMISTIC_INCLUSION = Constants.MESHES_CULLINGSTRATEGY_OPTIMISTIC_INCLUSION;\r\n /** Culling strategy : Optimistic Inclusion then Bounding Sphere Only.\r\n * This in an inclusion test first, then the bounding sphere only exclusion test.\r\n * This can be the fastest test when a cullable object is expected to be almost always in the camera frustum.\r\n * This could also be a little slower than the BoundingSphereOnly strategy when the tested object center is not in the frustum but its bounding sphere still intersects it.\r\n * It's less accurate than the standard strategy and as accurate as the BoundingSphereOnly strategy.\r\n * Test :\r\n * Is the cullable object bounding sphere center in the frustum ?\r\n * If not, apply the Bounding Sphere Only strategy. No Bounding Box is tested here.\r\n */\r\n public static readonly CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY = Constants.MESHES_CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY;\r\n\r\n /**\r\n * No billboard\r\n */\r\n public static get BILLBOARDMODE_NONE(): number {\r\n return TransformNode.BILLBOARDMODE_NONE;\r\n }\r\n\r\n /** Billboard on X axis */\r\n public static get BILLBOARDMODE_X(): number {\r\n return TransformNode.BILLBOARDMODE_X;\r\n }\r\n\r\n /** Billboard on Y axis */\r\n public static get BILLBOARDMODE_Y(): number {\r\n return TransformNode.BILLBOARDMODE_Y;\r\n }\r\n\r\n /** Billboard on Z axis */\r\n public static get BILLBOARDMODE_Z(): number {\r\n return TransformNode.BILLBOARDMODE_Z;\r\n }\r\n\r\n /** Billboard on all axes */\r\n public static get BILLBOARDMODE_ALL(): number {\r\n return TransformNode.BILLBOARDMODE_ALL;\r\n }\r\n\r\n /** Billboard on using position instead of orientation */\r\n public static get BILLBOARDMODE_USE_POSITION(): number {\r\n return TransformNode.BILLBOARDMODE_USE_POSITION;\r\n }\r\n\r\n // Internal data\r\n /** @internal */\r\n public _internalAbstractMeshDataInfo = new _InternalAbstractMeshDataInfo();\r\n\r\n /** @internal */\r\n public _waitingMaterialId: Nullable = null;\r\n\r\n /**\r\n * The culling strategy to use to check whether the mesh must be rendered or not.\r\n * This value can be changed at any time and will be used on the next render mesh selection.\r\n * The possible values are :\r\n * - AbstractMesh.CULLINGSTRATEGY_STANDARD\r\n * - AbstractMesh.CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY\r\n * - AbstractMesh.CULLINGSTRATEGY_OPTIMISTIC_INCLUSION\r\n * - AbstractMesh.CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY\r\n * Please read each static variable documentation to get details about the culling process.\r\n * */\r\n public cullingStrategy = AbstractMesh.CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY;\r\n\r\n /**\r\n * Gets the number of facets in the mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData#what-is-a-mesh-facet\r\n */\r\n public get facetNb(): number {\r\n return this._internalAbstractMeshDataInfo._facetData.facetNb;\r\n }\r\n /**\r\n * Gets or set the number (integer) of subdivisions per axis in the partitioning space\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData#tweaking-the-partitioning\r\n */\r\n public get partitioningSubdivisions(): number {\r\n return this._internalAbstractMeshDataInfo._facetData.partitioningSubdivisions;\r\n }\r\n public set partitioningSubdivisions(nb: number) {\r\n this._internalAbstractMeshDataInfo._facetData.partitioningSubdivisions = nb;\r\n }\r\n /**\r\n * The ratio (float) to apply to the bounding box size to set to the partitioning space.\r\n * Ex : 1.01 (default) the partitioning space is 1% bigger than the bounding box\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData#tweaking-the-partitioning\r\n */\r\n public get partitioningBBoxRatio(): number {\r\n return this._internalAbstractMeshDataInfo._facetData.partitioningBBoxRatio;\r\n }\r\n public set partitioningBBoxRatio(ratio: number) {\r\n this._internalAbstractMeshDataInfo._facetData.partitioningBBoxRatio = ratio;\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating that the facets must be depth sorted on next call to `updateFacetData()`.\r\n * Works only for updatable meshes.\r\n * Doesn't work with multi-materials\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData#facet-depth-sort\r\n */\r\n public get mustDepthSortFacets(): boolean {\r\n return this._internalAbstractMeshDataInfo._facetData.facetDepthSort;\r\n }\r\n public set mustDepthSortFacets(sort: boolean) {\r\n this._internalAbstractMeshDataInfo._facetData.facetDepthSort = sort;\r\n }\r\n\r\n /**\r\n * The location (Vector3) where the facet depth sort must be computed from.\r\n * By default, the active camera position.\r\n * Used only when facet depth sort is enabled\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData#facet-depth-sort\r\n */\r\n public get facetDepthSortFrom(): Vector3 {\r\n return this._internalAbstractMeshDataInfo._facetData.facetDepthSortFrom;\r\n }\r\n public set facetDepthSortFrom(location: Vector3) {\r\n this._internalAbstractMeshDataInfo._facetData.facetDepthSortFrom = location;\r\n }\r\n\r\n /** number of collision detection tries. Change this value if not all collisions are detected and handled properly */\r\n public get collisionRetryCount(): number {\r\n return this._internalAbstractMeshDataInfo._collisionRetryCount;\r\n }\r\n public set collisionRetryCount(retryCount: number) {\r\n this._internalAbstractMeshDataInfo._collisionRetryCount = retryCount;\r\n }\r\n /**\r\n * gets a boolean indicating if facetData is enabled\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData#what-is-a-mesh-facet\r\n */\r\n public get isFacetDataEnabled(): boolean {\r\n return this._internalAbstractMeshDataInfo._facetData.facetDataEnabled;\r\n }\r\n\r\n /**\r\n * Gets or sets the morph target manager\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/morphTargets\r\n */\r\n public get morphTargetManager(): Nullable {\r\n return this._internalAbstractMeshDataInfo._morphTargetManager;\r\n }\r\n\r\n public set morphTargetManager(value: Nullable) {\r\n if (this._internalAbstractMeshDataInfo._morphTargetManager === value) {\r\n return;\r\n }\r\n this._internalAbstractMeshDataInfo._morphTargetManager = value;\r\n this._syncGeometryWithMorphTargetManager();\r\n }\r\n\r\n /**\r\n * Gets or sets the baked vertex animation manager\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/baked_texture_animations\r\n */\r\n public get bakedVertexAnimationManager(): Nullable {\r\n return this._internalAbstractMeshDataInfo._bakedVertexAnimationManager;\r\n }\r\n\r\n public set bakedVertexAnimationManager(value: Nullable) {\r\n if (this._internalAbstractMeshDataInfo._bakedVertexAnimationManager === value) {\r\n return;\r\n }\r\n this._internalAbstractMeshDataInfo._bakedVertexAnimationManager = value;\r\n this._markSubMeshesAsAttributesDirty();\r\n }\r\n\r\n /** @internal */\r\n public _syncGeometryWithMorphTargetManager(): void {}\r\n\r\n /**\r\n * @internal\r\n */\r\n public _updateNonUniformScalingState(value: boolean): boolean {\r\n if (!super._updateNonUniformScalingState(value)) {\r\n return false;\r\n }\r\n this._markSubMeshesAsMiscDirty();\r\n return true;\r\n }\r\n\r\n /** @internal */\r\n public get rawBoundingInfo(): Nullable {\r\n return this._internalAbstractMeshDataInfo._rawBoundingInfo;\r\n }\r\n public set rawBoundingInfo(boundingInfo: Nullable) {\r\n this._internalAbstractMeshDataInfo._rawBoundingInfo = boundingInfo;\r\n }\r\n\r\n // Events\r\n\r\n /**\r\n * An event triggered when this mesh collides with another one\r\n */\r\n public onCollideObservable = new Observable();\r\n\r\n /** Set a function to call when this mesh collides with another one */\r\n public set onCollide(callback: (collidedMesh?: AbstractMesh) => void) {\r\n if (this._internalAbstractMeshDataInfo._meshCollisionData._onCollideObserver) {\r\n this.onCollideObservable.remove(this._internalAbstractMeshDataInfo._meshCollisionData._onCollideObserver);\r\n }\r\n this._internalAbstractMeshDataInfo._meshCollisionData._onCollideObserver = this.onCollideObservable.add(callback);\r\n }\r\n\r\n /**\r\n * An event triggered when the collision's position changes\r\n */\r\n public onCollisionPositionChangeObservable = new Observable();\r\n\r\n /** Set a function to call when the collision's position changes */\r\n public set onCollisionPositionChange(callback: () => void) {\r\n if (this._internalAbstractMeshDataInfo._meshCollisionData._onCollisionPositionChangeObserver) {\r\n this.onCollisionPositionChangeObservable.remove(this._internalAbstractMeshDataInfo._meshCollisionData._onCollisionPositionChangeObserver);\r\n }\r\n this._internalAbstractMeshDataInfo._meshCollisionData._onCollisionPositionChangeObserver = this.onCollisionPositionChangeObservable.add(callback);\r\n }\r\n\r\n /**\r\n * An event triggered when material is changed\r\n */\r\n public onMaterialChangedObservable = new Observable();\r\n\r\n // Properties\r\n\r\n /**\r\n * Gets or sets the orientation for POV movement & rotation\r\n */\r\n public definedFacingForward = true;\r\n\r\n /** @internal */\r\n public _occlusionQuery: Nullable = null;\r\n\r\n /** @internal */\r\n public _renderingGroup: Nullable = null;\r\n\r\n /**\r\n * Gets or sets mesh visibility between 0 and 1 (default is 1)\r\n */\r\n public get visibility(): number {\r\n return this._internalAbstractMeshDataInfo._visibility;\r\n }\r\n\r\n /**\r\n * Gets or sets mesh visibility between 0 and 1 (default is 1)\r\n */\r\n public set visibility(value: number) {\r\n if (this._internalAbstractMeshDataInfo._visibility === value) {\r\n return;\r\n }\r\n\r\n const oldValue = this._internalAbstractMeshDataInfo._visibility;\r\n\r\n this._internalAbstractMeshDataInfo._visibility = value;\r\n\r\n if ((oldValue === 1 && value !== 1) || (oldValue !== 1 && value === 1)) {\r\n this._markSubMeshesAsDirty((defines) => {\r\n defines.markAsMiscDirty();\r\n defines.markAsPrePassDirty();\r\n });\r\n }\r\n }\r\n\r\n /** Gets or sets the alpha index used to sort transparent meshes\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/advanced/transparent_rendering#alpha-index\r\n */\r\n public alphaIndex = Number.MAX_VALUE;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if the mesh is visible (renderable). Default is true\r\n */\r\n public isVisible = true;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if the mesh can be picked (by scene.pick for instance or through actions). Default is true\r\n */\r\n public isPickable = true;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if the mesh can be near picked. Default is false\r\n */\r\n public isNearPickable = false;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if the mesh can be near grabbed. Default is false\r\n */\r\n public isNearGrabbable = false;\r\n\r\n /** Gets or sets a boolean indicating that bounding boxes of subMeshes must be rendered as well (false by default) */\r\n public showSubMeshesBoundingBox = false;\r\n\r\n /** Gets or sets a boolean indicating if the mesh must be considered as a ray blocker for lens flares (false by default)\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/lenseFlare\r\n */\r\n public isBlocker = false;\r\n\r\n /**\r\n * Gets or sets a boolean indicating that pointer move events must be supported on this mesh (false by default)\r\n */\r\n public enablePointerMoveEvents = false;\r\n\r\n /**\r\n * Gets or sets the property which disables the test that is checking that the mesh under the pointer is the same than the previous time we tested for it (default: false).\r\n * Set this property to true if you want thin instances picking to be reported accurately when moving over the mesh.\r\n * Note that setting this property to true will incur some performance penalties when dealing with pointer events for this mesh so use it sparingly.\r\n */\r\n public get pointerOverDisableMeshTesting() {\r\n return this._internalAbstractMeshDataInfo._pointerOverDisableMeshTesting;\r\n }\r\n\r\n public set pointerOverDisableMeshTesting(disable: boolean) {\r\n this._internalAbstractMeshDataInfo._pointerOverDisableMeshTesting = disable;\r\n }\r\n\r\n /**\r\n * Specifies the rendering group id for this mesh (0 by default)\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/advanced/transparent_rendering#rendering-groups\r\n */\r\n public get renderingGroupId() {\r\n return this._internalAbstractMeshDataInfo._renderingGroupId;\r\n }\r\n\r\n public set renderingGroupId(value: number) {\r\n this._internalAbstractMeshDataInfo._renderingGroupId = value;\r\n }\r\n\r\n /** Gets or sets current material */\r\n public get material(): Nullable {\r\n return this._internalAbstractMeshDataInfo._material;\r\n }\r\n public set material(value: Nullable) {\r\n if (this._internalAbstractMeshDataInfo._material === value) {\r\n return;\r\n }\r\n\r\n // remove from material mesh map id needed\r\n if (this._internalAbstractMeshDataInfo._material && this._internalAbstractMeshDataInfo._material.meshMap) {\r\n this._internalAbstractMeshDataInfo._material.meshMap[this.uniqueId] = undefined;\r\n }\r\n\r\n this._internalAbstractMeshDataInfo._material = value;\r\n\r\n if (value && value.meshMap) {\r\n value.meshMap[this.uniqueId] = this;\r\n }\r\n\r\n if (this.onMaterialChangedObservable.hasObservers()) {\r\n this.onMaterialChangedObservable.notifyObservers(this);\r\n }\r\n\r\n if (!this.subMeshes) {\r\n return;\r\n }\r\n\r\n this.resetDrawCache();\r\n this._unBindEffect();\r\n }\r\n\r\n /**\r\n * Gets the material used to render the mesh in a specific render pass\r\n * @param renderPassId render pass id\r\n * @returns material used for the render pass. If no specific material is used for this render pass, undefined is returned (meaning mesh.material is used for this pass)\r\n */\r\n public getMaterialForRenderPass(renderPassId: number): Material | undefined {\r\n return this._internalAbstractMeshDataInfo._materialForRenderPass?.[renderPassId];\r\n }\r\n\r\n /**\r\n * Sets the material to be used to render the mesh in a specific render pass\r\n * @param renderPassId render pass id\r\n * @param material material to use for this render pass. If undefined is passed, no specific material will be used for this render pass but the regular material will be used instead (mesh.material)\r\n */\r\n public setMaterialForRenderPass(renderPassId: number, material?: Material): void {\r\n this.resetDrawCache(renderPassId);\r\n if (!this._internalAbstractMeshDataInfo._materialForRenderPass) {\r\n this._internalAbstractMeshDataInfo._materialForRenderPass = [];\r\n }\r\n this._internalAbstractMeshDataInfo._materialForRenderPass[renderPassId] = material;\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating that this mesh can receive realtime shadows\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/lights/shadows\r\n */\r\n public get receiveShadows(): boolean {\r\n return this._internalAbstractMeshDataInfo._receiveShadows;\r\n }\r\n public set receiveShadows(value: boolean) {\r\n if (this._internalAbstractMeshDataInfo._receiveShadows === value) {\r\n return;\r\n }\r\n\r\n this._internalAbstractMeshDataInfo._receiveShadows = value;\r\n this._markSubMeshesAsLightDirty();\r\n }\r\n\r\n /** Defines color to use when rendering outline */\r\n public outlineColor = Color3.Red();\r\n /** Define width to use when rendering outline */\r\n public outlineWidth = 0.02;\r\n\r\n /** Defines color to use when rendering overlay */\r\n public overlayColor = Color3.Red();\r\n /** Defines alpha to use when rendering overlay */\r\n public overlayAlpha = 0.5;\r\n\r\n /** Gets or sets a boolean indicating that this mesh contains vertex color data with alpha values */\r\n public get hasVertexAlpha(): boolean {\r\n return this._internalAbstractMeshDataInfo._hasVertexAlpha;\r\n }\r\n public set hasVertexAlpha(value: boolean) {\r\n if (this._internalAbstractMeshDataInfo._hasVertexAlpha === value) {\r\n return;\r\n }\r\n\r\n this._internalAbstractMeshDataInfo._hasVertexAlpha = value;\r\n this._markSubMeshesAsAttributesDirty();\r\n this._markSubMeshesAsMiscDirty();\r\n }\r\n\r\n /** Gets or sets a boolean indicating that this mesh needs to use vertex color data to render (if this kind of vertex data is available in the geometry) */\r\n public get useVertexColors(): boolean {\r\n return this._internalAbstractMeshDataInfo._useVertexColors;\r\n }\r\n public set useVertexColors(value: boolean) {\r\n if (this._internalAbstractMeshDataInfo._useVertexColors === value) {\r\n return;\r\n }\r\n\r\n this._internalAbstractMeshDataInfo._useVertexColors = value;\r\n this._markSubMeshesAsAttributesDirty();\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating that bone animations must be computed by the GPU (true by default)\r\n */\r\n public get computeBonesUsingShaders(): boolean {\r\n return this._internalAbstractMeshDataInfo._computeBonesUsingShaders;\r\n }\r\n public set computeBonesUsingShaders(value: boolean) {\r\n if (this._internalAbstractMeshDataInfo._computeBonesUsingShaders === value) {\r\n return;\r\n }\r\n\r\n this._internalAbstractMeshDataInfo._computeBonesUsingShaders = value;\r\n this._markSubMeshesAsAttributesDirty();\r\n }\r\n\r\n /** Gets or sets the number of allowed bone influences per vertex (4 by default) */\r\n public get numBoneInfluencers(): number {\r\n return this._internalAbstractMeshDataInfo._numBoneInfluencers;\r\n }\r\n public set numBoneInfluencers(value: number) {\r\n if (this._internalAbstractMeshDataInfo._numBoneInfluencers === value) {\r\n return;\r\n }\r\n\r\n this._internalAbstractMeshDataInfo._numBoneInfluencers = value;\r\n this._markSubMeshesAsAttributesDirty();\r\n }\r\n\r\n /** Gets or sets a boolean indicating that this mesh will allow fog to be rendered on it (true by default) */\r\n public get applyFog(): boolean {\r\n return this._internalAbstractMeshDataInfo._applyFog;\r\n }\r\n public set applyFog(value: boolean) {\r\n if (this._internalAbstractMeshDataInfo._applyFog === value) {\r\n return;\r\n }\r\n\r\n this._internalAbstractMeshDataInfo._applyFog = value;\r\n this._markSubMeshesAsMiscDirty();\r\n }\r\n\r\n /** When enabled, decompose picking matrices for better precision with large values for mesh position and scling */\r\n public get enableDistantPicking(): boolean {\r\n return this._internalAbstractMeshDataInfo._enableDistantPicking;\r\n }\r\n public set enableDistantPicking(value: boolean) {\r\n this._internalAbstractMeshDataInfo._enableDistantPicking = value;\r\n }\r\n\r\n /** Gets or sets a boolean indicating that internal octree (if available) can be used to boost submeshes selection (true by default) */\r\n public useOctreeForRenderingSelection = true;\r\n /** Gets or sets a boolean indicating that internal octree (if available) can be used to boost submeshes picking (true by default) */\r\n public useOctreeForPicking = true;\r\n /** Gets or sets a boolean indicating that internal octree (if available) can be used to boost submeshes collision (true by default) */\r\n public useOctreeForCollisions = true;\r\n /**\r\n * Gets or sets the current layer mask (default is 0x0FFFFFFF)\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/layerMasksAndMultiCam\r\n */\r\n public get layerMask(): number {\r\n return this._internalAbstractMeshDataInfo._layerMask;\r\n }\r\n\r\n public set layerMask(value: number) {\r\n if (value === this._internalAbstractMeshDataInfo._layerMask) {\r\n return;\r\n }\r\n\r\n this._internalAbstractMeshDataInfo._layerMask = value;\r\n this._resyncLightSources();\r\n }\r\n\r\n /**\r\n * True if the mesh must be rendered in any case (this will shortcut the frustum clipping phase)\r\n */\r\n public alwaysSelectAsActiveMesh = false;\r\n\r\n /**\r\n * Gets or sets a boolean indicating that the bounding info does not need to be kept in sync (for performance reason)\r\n */\r\n public doNotSyncBoundingInfo = false;\r\n\r\n /**\r\n * Gets or sets the current action manager\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions\r\n */\r\n public actionManager: Nullable = null;\r\n\r\n /**\r\n * Gets or sets the ellipsoid used to impersonate this mesh when using collision engine (default is (0.5, 1, 0.5))\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions\r\n */\r\n public ellipsoid = new Vector3(0.5, 1, 0.5);\r\n /**\r\n * Gets or sets the ellipsoid offset used to impersonate this mesh when using collision engine (default is (0, 0, 0))\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions\r\n */\r\n public ellipsoidOffset = new Vector3(0, 0, 0);\r\n\r\n /**\r\n * Gets or sets a collision mask used to mask collisions (default is -1).\r\n * A collision between A and B will happen if A.collisionGroup & b.collisionMask !== 0\r\n */\r\n public get collisionMask(): number {\r\n return this._internalAbstractMeshDataInfo._meshCollisionData._collisionMask;\r\n }\r\n\r\n public set collisionMask(mask: number) {\r\n this._internalAbstractMeshDataInfo._meshCollisionData._collisionMask = !isNaN(mask) ? mask : -1;\r\n }\r\n\r\n /**\r\n * Gets or sets a collision response flag (default is true).\r\n * when collisionResponse is false, events are still triggered but colliding entity has no response\r\n * This helps creating trigger volume when user wants collision feedback events but not position/velocity\r\n * to respond to the collision.\r\n */\r\n public get collisionResponse(): boolean {\r\n return this._internalAbstractMeshDataInfo._meshCollisionData._collisionResponse;\r\n }\r\n\r\n public set collisionResponse(response: boolean) {\r\n this._internalAbstractMeshDataInfo._meshCollisionData._collisionResponse = response;\r\n }\r\n /**\r\n * Gets or sets the current collision group mask (-1 by default).\r\n * A collision between A and B will happen if A.collisionGroup & b.collisionMask !== 0\r\n */\r\n public get collisionGroup(): number {\r\n return this._internalAbstractMeshDataInfo._meshCollisionData._collisionGroup;\r\n }\r\n\r\n public set collisionGroup(mask: number) {\r\n this._internalAbstractMeshDataInfo._meshCollisionData._collisionGroup = !isNaN(mask) ? mask : -1;\r\n }\r\n\r\n /**\r\n * Gets or sets current surrounding meshes (null by default).\r\n *\r\n * By default collision detection is tested against every mesh in the scene.\r\n * It is possible to set surroundingMeshes to a defined list of meshes and then only these specified\r\n * meshes will be tested for the collision.\r\n *\r\n * Note: if set to an empty array no collision will happen when this mesh is moved.\r\n */\r\n public get surroundingMeshes(): Nullable {\r\n return this._internalAbstractMeshDataInfo._meshCollisionData._surroundingMeshes;\r\n }\r\n\r\n public set surroundingMeshes(meshes: Nullable) {\r\n this._internalAbstractMeshDataInfo._meshCollisionData._surroundingMeshes = meshes;\r\n }\r\n\r\n // Edges\r\n /**\r\n * Defines edge width used when edgesRenderer is enabled\r\n * @see https://www.babylonjs-playground.com/#10OJSG#13\r\n */\r\n public edgesWidth = 1;\r\n /**\r\n * Defines edge color used when edgesRenderer is enabled\r\n * @see https://www.babylonjs-playground.com/#10OJSG#13\r\n */\r\n public edgesColor = new Color4(1, 0, 0, 1);\r\n /** @internal */\r\n public _edgesRenderer: Nullable = null;\r\n\r\n /** @internal */\r\n public _masterMesh: Nullable = null;\r\n protected _boundingInfo: Nullable = null;\r\n protected _boundingInfoIsDirty = true;\r\n /** @internal */\r\n public _renderId = 0;\r\n\r\n /**\r\n * Gets or sets the list of subMeshes\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/multiMaterials\r\n */\r\n public subMeshes: SubMesh[];\r\n\r\n /** @internal */\r\n public _intersectionsInProgress = new Array();\r\n\r\n /** @internal */\r\n public _unIndexed = false;\r\n\r\n /** @internal */\r\n public _lightSources = new Array();\r\n\r\n /** Gets the list of lights affecting that mesh */\r\n public get lightSources(): Light[] {\r\n return this._lightSources;\r\n }\r\n\r\n /** @internal */\r\n public get _positions(): Nullable {\r\n return null;\r\n }\r\n\r\n // Loading properties\r\n /** @internal */\r\n public _waitingData: {\r\n lods: Nullable;\r\n actions: Nullable;\r\n freezeWorldMatrix: Nullable;\r\n } = {\r\n lods: null,\r\n actions: null,\r\n freezeWorldMatrix: null,\r\n };\r\n\r\n /** @internal */\r\n public _bonesTransformMatrices: Nullable = null;\r\n\r\n /** @internal */\r\n public _transformMatrixTexture: Nullable = null;\r\n\r\n /**\r\n * Gets or sets a skeleton to apply skinning transformations\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/bonesSkeletons\r\n */\r\n public set skeleton(value: Nullable) {\r\n const skeleton = this._internalAbstractMeshDataInfo._skeleton;\r\n if (skeleton && skeleton.needInitialSkinMatrix) {\r\n skeleton._unregisterMeshWithPoseMatrix(this);\r\n }\r\n\r\n if (value && value.needInitialSkinMatrix) {\r\n value._registerMeshWithPoseMatrix(this);\r\n }\r\n\r\n this._internalAbstractMeshDataInfo._skeleton = value;\r\n\r\n if (!this._internalAbstractMeshDataInfo._skeleton) {\r\n this._bonesTransformMatrices = null;\r\n }\r\n\r\n this._markSubMeshesAsAttributesDirty();\r\n }\r\n\r\n public get skeleton(): Nullable {\r\n return this._internalAbstractMeshDataInfo._skeleton;\r\n }\r\n\r\n /**\r\n * An event triggered when the mesh is rebuilt.\r\n */\r\n public onRebuildObservable = new Observable();\r\n\r\n /**\r\n * The current mesh uniform buffer.\r\n * @internal Internal use only.\r\n */\r\n public _uniformBuffer: UniformBuffer;\r\n\r\n // Constructor\r\n\r\n /**\r\n * Creates a new AbstractMesh\r\n * @param name defines the name of the mesh\r\n * @param scene defines the hosting scene\r\n */\r\n constructor(name: string, scene: Nullable = null) {\r\n super(name, scene, false);\r\n\r\n scene = this.getScene();\r\n\r\n scene.addMesh(this);\r\n\r\n this._resyncLightSources();\r\n\r\n // Mesh Uniform Buffer.\r\n this._uniformBuffer = new UniformBuffer(this.getScene().getEngine(), undefined, undefined, name, !this.getScene().getEngine().isWebGPU);\r\n this._buildUniformLayout();\r\n\r\n switch (scene.performancePriority) {\r\n case ScenePerformancePriority.Aggressive:\r\n this.doNotSyncBoundingInfo = true;\r\n // eslint-disable-next-line no-fallthrough\r\n case ScenePerformancePriority.Intermediate:\r\n this.alwaysSelectAsActiveMesh = true;\r\n this.isPickable = false;\r\n break;\r\n }\r\n }\r\n\r\n protected _buildUniformLayout(): void {\r\n this._uniformBuffer.addUniform(\"world\", 16);\r\n this._uniformBuffer.addUniform(\"visibility\", 1);\r\n this._uniformBuffer.create();\r\n }\r\n\r\n /**\r\n * Transfer the mesh values to its UBO.\r\n * @param world The world matrix associated with the mesh\r\n */\r\n public transferToEffect(world: Matrix): void {\r\n const ubo = this._uniformBuffer;\r\n\r\n ubo.updateMatrix(\"world\", world);\r\n ubo.updateFloat(\"visibility\", this._internalAbstractMeshDataInfo._visibility);\r\n\r\n ubo.update();\r\n }\r\n\r\n /**\r\n * Gets the mesh uniform buffer.\r\n * @returns the uniform buffer of the mesh.\r\n */\r\n public getMeshUniformBuffer(): UniformBuffer {\r\n return this._uniformBuffer;\r\n }\r\n\r\n /**\r\n * Returns the string \"AbstractMesh\"\r\n * @returns \"AbstractMesh\"\r\n */\r\n public getClassName(): string {\r\n return \"AbstractMesh\";\r\n }\r\n\r\n /**\r\n * Gets a string representation of the current mesh\r\n * @param fullDetails defines a boolean indicating if full details must be included\r\n * @returns a string representation of the current mesh\r\n */\r\n public toString(fullDetails?: boolean): string {\r\n let ret = \"Name: \" + this.name + \", isInstance: \" + (this.getClassName() !== \"InstancedMesh\" ? \"YES\" : \"NO\");\r\n ret += \", # of submeshes: \" + (this.subMeshes ? this.subMeshes.length : 0);\r\n\r\n const skeleton = this._internalAbstractMeshDataInfo._skeleton;\r\n if (skeleton) {\r\n ret += \", skeleton: \" + skeleton.name;\r\n }\r\n if (fullDetails) {\r\n ret += \", billboard mode: \" + [\"NONE\", \"X\", \"Y\", null, \"Z\", null, null, \"ALL\"][this.billboardMode];\r\n ret += \", freeze wrld mat: \" + (this._isWorldMatrixFrozen || this._waitingData.freezeWorldMatrix ? \"YES\" : \"NO\");\r\n }\r\n return ret;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n protected _getEffectiveParent(): Nullable {\r\n if (this._masterMesh && this.billboardMode !== TransformNode.BILLBOARDMODE_NONE) {\r\n return this._masterMesh;\r\n }\r\n\r\n return super._getEffectiveParent();\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getActionManagerForTrigger(trigger?: number, initialCall = true): Nullable {\r\n if (this.actionManager && (initialCall || this.actionManager.isRecursive)) {\r\n if (trigger) {\r\n if (this.actionManager.hasSpecificTrigger(trigger)) {\r\n return this.actionManager;\r\n }\r\n } else {\r\n return this.actionManager;\r\n }\r\n }\r\n\r\n if (!this.parent) {\r\n return null;\r\n }\r\n\r\n return this.parent._getActionManagerForTrigger(trigger, false);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public _rebuild(dispose = false): void {\r\n this.onRebuildObservable.notifyObservers(this);\r\n\r\n if (this._occlusionQuery !== null) {\r\n this._occlusionQuery = null;\r\n }\r\n\r\n if (!this.subMeshes) {\r\n return;\r\n }\r\n\r\n for (const subMesh of this.subMeshes) {\r\n subMesh._rebuild();\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _resyncLightSources(): void {\r\n this._lightSources.length = 0;\r\n\r\n for (const light of this.getScene().lights) {\r\n if (!light.isEnabled()) {\r\n continue;\r\n }\r\n\r\n if (light.canAffectMesh(this)) {\r\n this._lightSources.push(light);\r\n }\r\n }\r\n\r\n this._markSubMeshesAsLightDirty();\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _resyncLightSource(light: Light): void {\r\n const isIn = light.isEnabled() && light.canAffectMesh(this);\r\n\r\n const index = this._lightSources.indexOf(light);\r\n let removed = false;\r\n if (index === -1) {\r\n if (!isIn) {\r\n return;\r\n }\r\n this._lightSources.push(light);\r\n } else {\r\n if (isIn) {\r\n return;\r\n }\r\n removed = true;\r\n this._lightSources.splice(index, 1);\r\n }\r\n\r\n this._markSubMeshesAsLightDirty(removed);\r\n }\r\n\r\n /** @internal */\r\n public _unBindEffect() {\r\n for (const subMesh of this.subMeshes) {\r\n subMesh.setEffect(null);\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _removeLightSource(light: Light, dispose: boolean): void {\r\n const index = this._lightSources.indexOf(light);\r\n\r\n if (index === -1) {\r\n return;\r\n }\r\n this._lightSources.splice(index, 1);\r\n\r\n this._markSubMeshesAsLightDirty(dispose);\r\n }\r\n\r\n private _markSubMeshesAsDirty(func: (defines: MaterialDefines) => void) {\r\n if (!this.subMeshes) {\r\n return;\r\n }\r\n\r\n for (const subMesh of this.subMeshes) {\r\n for (let i = 0; i < subMesh._drawWrappers.length; ++i) {\r\n const drawWrapper = subMesh._drawWrappers[i];\r\n if (!drawWrapper || !drawWrapper.defines || !(drawWrapper.defines as MaterialDefines).markAllAsDirty) {\r\n continue;\r\n }\r\n func(drawWrapper.defines as MaterialDefines);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _markSubMeshesAsLightDirty(dispose: boolean = false) {\r\n this._markSubMeshesAsDirty((defines) => defines.markAsLightDirty(dispose));\r\n }\r\n\r\n /** @internal */\r\n public _markSubMeshesAsAttributesDirty() {\r\n this._markSubMeshesAsDirty((defines) => defines.markAsAttributesDirty());\r\n }\r\n\r\n /** @internal */\r\n public _markSubMeshesAsMiscDirty() {\r\n this._markSubMeshesAsDirty((defines) => defines.markAsMiscDirty());\r\n }\r\n\r\n /**\r\n * Flag the AbstractMesh as dirty (Forcing it to update everything)\r\n * @param property if set to \"rotation\" the objects rotationQuaternion will be set to null\r\n * @returns this AbstractMesh\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public markAsDirty(property?: string): AbstractMesh {\r\n this._currentRenderId = Number.MAX_VALUE;\r\n this._isDirty = true;\r\n return this;\r\n }\r\n\r\n /**\r\n * Resets the draw wrappers cache for all submeshes of this abstract mesh\r\n * @param passId If provided, releases only the draw wrapper corresponding to this render pass id\r\n */\r\n public resetDrawCache(passId?: number): void {\r\n if (!this.subMeshes) {\r\n return;\r\n }\r\n\r\n for (const subMesh of this.subMeshes) {\r\n subMesh.resetDrawCache(passId);\r\n }\r\n }\r\n\r\n // Methods\r\n /**\r\n * Returns true if the mesh is blocked. Implemented by child classes\r\n */\r\n public get isBlocked(): boolean {\r\n return false;\r\n }\r\n\r\n /**\r\n * Returns the mesh itself by default. Implemented by child classes\r\n * @param camera defines the camera to use to pick the right LOD level\r\n * @returns the currentAbstractMesh\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public getLOD(camera: Camera): Nullable {\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns 0 by default. Implemented by child classes\r\n * @returns an integer\r\n */\r\n public getTotalVertices(): number {\r\n return 0;\r\n }\r\n\r\n /**\r\n * Returns a positive integer : the total number of indices in this mesh geometry.\r\n * @returns the number of indices or zero if the mesh has no geometry.\r\n */\r\n public getTotalIndices(): number {\r\n return 0;\r\n }\r\n\r\n /**\r\n * Returns null by default. Implemented by child classes\r\n * @returns null\r\n */\r\n public getIndices(): Nullable {\r\n return null;\r\n }\r\n\r\n /**\r\n * Returns the array of the requested vertex data kind. Implemented by child classes\r\n * @param kind defines the vertex data kind to use\r\n * @returns null\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public getVerticesData(kind: string): Nullable {\r\n return null;\r\n }\r\n\r\n /**\r\n * Sets the vertex data of the mesh geometry for the requested `kind`.\r\n * If the mesh has no geometry, a new Geometry object is set to the mesh and then passed this vertex data.\r\n * Note that a new underlying VertexBuffer object is created each call.\r\n * If the `kind` is the `PositionKind`, the mesh BoundingInfo is renewed, so the bounding box and sphere, and the mesh World Matrix is recomputed.\r\n * @param kind defines vertex data kind:\r\n * * VertexBuffer.PositionKind\r\n * * VertexBuffer.UVKind\r\n * * VertexBuffer.UV2Kind\r\n * * VertexBuffer.UV3Kind\r\n * * VertexBuffer.UV4Kind\r\n * * VertexBuffer.UV5Kind\r\n * * VertexBuffer.UV6Kind\r\n * * VertexBuffer.ColorKind\r\n * * VertexBuffer.MatricesIndicesKind\r\n * * VertexBuffer.MatricesIndicesExtraKind\r\n * * VertexBuffer.MatricesWeightsKind\r\n * * VertexBuffer.MatricesWeightsExtraKind\r\n * @param data defines the data source\r\n * @param updatable defines if the data must be flagged as updatable (or static)\r\n * @param stride defines the vertex stride (size of an entire vertex). Can be null and in this case will be deduced from vertex data kind\r\n * @returns the current mesh\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public setVerticesData(kind: string, data: FloatArray, updatable?: boolean, stride?: number): AbstractMesh {\r\n return this;\r\n }\r\n\r\n /**\r\n * Updates the existing vertex data of the mesh geometry for the requested `kind`.\r\n * If the mesh has no geometry, it is simply returned as it is.\r\n * @param kind defines vertex data kind:\r\n * * VertexBuffer.PositionKind\r\n * * VertexBuffer.UVKind\r\n * * VertexBuffer.UV2Kind\r\n * * VertexBuffer.UV3Kind\r\n * * VertexBuffer.UV4Kind\r\n * * VertexBuffer.UV5Kind\r\n * * VertexBuffer.UV6Kind\r\n * * VertexBuffer.ColorKind\r\n * * VertexBuffer.MatricesIndicesKind\r\n * * VertexBuffer.MatricesIndicesExtraKind\r\n * * VertexBuffer.MatricesWeightsKind\r\n * * VertexBuffer.MatricesWeightsExtraKind\r\n * @param data defines the data source\r\n * @param updateExtends If `kind` is `PositionKind` and if `updateExtends` is true, the mesh BoundingInfo is renewed, so the bounding box and sphere, and the mesh World Matrix is recomputed\r\n * @param makeItUnique If true, a new global geometry is created from this data and is set to the mesh\r\n * @returns the current mesh\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public updateVerticesData(kind: string, data: FloatArray, updateExtends?: boolean, makeItUnique?: boolean): AbstractMesh {\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets the mesh indices,\r\n * If the mesh has no geometry, a new Geometry object is created and set to the mesh.\r\n * @param indices Expects an array populated with integers or a typed array (Int32Array, Uint32Array, Uint16Array)\r\n * @param totalVertices Defines the total number of vertices\r\n * @returns the current mesh\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public setIndices(indices: IndicesArray, totalVertices: Nullable): AbstractMesh {\r\n return this;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if specific vertex data is present\r\n * @param kind defines the vertex data kind to use\r\n * @returns true is data kind is present\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public isVerticesDataPresent(kind: string): boolean {\r\n return false;\r\n }\r\n\r\n /**\r\n * Returns the mesh BoundingInfo object or creates a new one and returns if it was undefined.\r\n * Note that it returns a shallow bounding of the mesh (i.e. it does not include children).\r\n * However, if the mesh contains thin instances, it will be expanded to include them. If you want the \"raw\" bounding data instead, then use `getRawBoundingInfo()`.\r\n * To get the full bounding of all children, call `getHierarchyBoundingVectors` instead.\r\n * @returns a BoundingInfo\r\n */\r\n public getBoundingInfo(): BoundingInfo {\r\n if (this._masterMesh) {\r\n return this._masterMesh.getBoundingInfo();\r\n }\r\n\r\n if (this._boundingInfoIsDirty) {\r\n this._boundingInfoIsDirty = false;\r\n // this._boundingInfo is being created if undefined\r\n this._updateBoundingInfo();\r\n }\r\n // cannot be null.\r\n return this._boundingInfo!;\r\n }\r\n\r\n /**\r\n * Returns the bounding info unnafected by instance data.\r\n * @returns the bounding info of the mesh unaffected by instance data.\r\n */\r\n public getRawBoundingInfo() {\r\n return this.rawBoundingInfo ?? this.getBoundingInfo();\r\n }\r\n\r\n /**\r\n * Overwrite the current bounding info\r\n * @param boundingInfo defines the new bounding info\r\n * @returns the current mesh\r\n */\r\n public setBoundingInfo(boundingInfo: BoundingInfo): AbstractMesh {\r\n this._boundingInfo = boundingInfo;\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns true if there is already a bounding info\r\n */\r\n public get hasBoundingInfo(): boolean {\r\n return this._boundingInfo !== null;\r\n }\r\n\r\n /**\r\n * Creates a new bounding info for the mesh\r\n * @param minimum min vector of the bounding box/sphere\r\n * @param maximum max vector of the bounding box/sphere\r\n * @param worldMatrix defines the new world matrix\r\n * @returns the new bounding info\r\n */\r\n public buildBoundingInfo(minimum: DeepImmutable, maximum: DeepImmutable, worldMatrix?: DeepImmutable) {\r\n this._boundingInfo = new BoundingInfo(minimum, maximum, worldMatrix);\r\n return this._boundingInfo;\r\n }\r\n\r\n /**\r\n * Uniformly scales the mesh to fit inside of a unit cube (1 X 1 X 1 units)\r\n * @param includeDescendants Use the hierarchy's bounding box instead of the mesh's bounding box. Default is false\r\n * @param ignoreRotation ignore rotation when computing the scale (ie. object will be axis aligned). Default is false\r\n * @param predicate predicate that is passed in to getHierarchyBoundingVectors when selecting which object should be included when scaling\r\n * @returns the current mesh\r\n */\r\n public normalizeToUnitCube(includeDescendants = true, ignoreRotation = false, predicate?: Nullable<(node: AbstractMesh) => boolean>): AbstractMesh {\r\n return super.normalizeToUnitCube(includeDescendants, ignoreRotation, predicate);\r\n }\r\n\r\n /** Gets a boolean indicating if this mesh has skinning data and an attached skeleton */\r\n public get useBones(): boolean {\r\n return (\r\n (this.skeleton &&\r\n this.getScene().skeletonsEnabled &&\r\n this.isVerticesDataPresent(VertexBuffer.MatricesIndicesKind) &&\r\n this.isVerticesDataPresent(VertexBuffer.MatricesWeightsKind))\r\n );\r\n }\r\n\r\n /** @internal */\r\n public _preActivate(): void {}\r\n\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public _preActivateForIntermediateRendering(renderId: number): void {}\r\n\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public _activate(renderId: number, intermediateRendering: boolean): boolean {\r\n this._renderId = renderId;\r\n return true;\r\n }\r\n\r\n /** @internal */\r\n public _postActivate(): void {\r\n // Do nothing\r\n }\r\n\r\n /** @internal */\r\n public _freeze() {\r\n // Do nothing\r\n }\r\n\r\n /** @internal */\r\n public _unFreeze() {\r\n // Do nothing\r\n }\r\n\r\n /**\r\n * Gets the current world matrix\r\n * @returns a Matrix\r\n */\r\n public getWorldMatrix(): Matrix {\r\n if (this._masterMesh && this.billboardMode === TransformNode.BILLBOARDMODE_NONE) {\r\n return this._masterMesh.getWorldMatrix();\r\n }\r\n\r\n return super.getWorldMatrix();\r\n }\r\n\r\n /** @internal */\r\n public _getWorldMatrixDeterminant(): number {\r\n if (this._masterMesh) {\r\n return this._masterMesh._getWorldMatrixDeterminant();\r\n }\r\n\r\n return super._getWorldMatrixDeterminant();\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if this mesh is an instance or a regular mesh\r\n */\r\n public get isAnInstance(): boolean {\r\n return false;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if this mesh has instances\r\n */\r\n public get hasInstances(): boolean {\r\n return false;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if this mesh has thin instances\r\n */\r\n public get hasThinInstances(): boolean {\r\n return false;\r\n }\r\n\r\n // ================================== Point of View Movement =================================\r\n\r\n /**\r\n * Perform relative position change from the point of view of behind the front of the mesh.\r\n * This is performed taking into account the meshes current rotation, so you do not have to care.\r\n * Supports definition of mesh facing forward or backward {@link definedFacingForwardSearch | See definedFacingForwardSearch }.\r\n * @param amountRight defines the distance on the right axis\r\n * @param amountUp defines the distance on the up axis\r\n * @param amountForward defines the distance on the forward axis\r\n * @returns the current mesh\r\n */\r\n public movePOV(amountRight: number, amountUp: number, amountForward: number): AbstractMesh {\r\n this.position.addInPlace(this.calcMovePOV(amountRight, amountUp, amountForward));\r\n return this;\r\n }\r\n\r\n /**\r\n * Calculate relative position change from the point of view of behind the front of the mesh.\r\n * This is performed taking into account the meshes current rotation, so you do not have to care.\r\n * Supports definition of mesh facing forward or backward {@link definedFacingForwardSearch | See definedFacingForwardSearch }.\r\n * @param amountRight defines the distance on the right axis\r\n * @param amountUp defines the distance on the up axis\r\n * @param amountForward defines the distance on the forward axis\r\n * @returns the new displacement vector\r\n */\r\n public calcMovePOV(amountRight: number, amountUp: number, amountForward: number): Vector3 {\r\n const rotMatrix = new Matrix();\r\n const rotQuaternion = this.rotationQuaternion ? this.rotationQuaternion : Quaternion.RotationYawPitchRoll(this.rotation.y, this.rotation.x, this.rotation.z);\r\n rotQuaternion.toRotationMatrix(rotMatrix);\r\n\r\n const translationDelta = Vector3.Zero();\r\n const defForwardMult = this.definedFacingForward ? -1 : 1;\r\n Vector3.TransformCoordinatesFromFloatsToRef(amountRight * defForwardMult, amountUp, amountForward * defForwardMult, rotMatrix, translationDelta);\r\n return translationDelta;\r\n }\r\n // ================================== Point of View Rotation =================================\r\n /**\r\n * Perform relative rotation change from the point of view of behind the front of the mesh.\r\n * Supports definition of mesh facing forward or backward {@link definedFacingForwardSearch | See definedFacingForwardSearch }.\r\n * @param flipBack defines the flip\r\n * @param twirlClockwise defines the twirl\r\n * @param tiltRight defines the tilt\r\n * @returns the current mesh\r\n */\r\n public rotatePOV(flipBack: number, twirlClockwise: number, tiltRight: number): AbstractMesh {\r\n this.rotation.addInPlace(this.calcRotatePOV(flipBack, twirlClockwise, tiltRight));\r\n return this;\r\n }\r\n\r\n /**\r\n * Calculate relative rotation change from the point of view of behind the front of the mesh.\r\n * Supports definition of mesh facing forward or backward {@link definedFacingForwardSearch | See definedFacingForwardSearch }.\r\n * @param flipBack defines the flip\r\n * @param twirlClockwise defines the twirl\r\n * @param tiltRight defines the tilt\r\n * @returns the new rotation vector\r\n */\r\n public calcRotatePOV(flipBack: number, twirlClockwise: number, tiltRight: number): Vector3 {\r\n const defForwardMult = this.definedFacingForward ? 1 : -1;\r\n return new Vector3(flipBack * defForwardMult, twirlClockwise, tiltRight * defForwardMult);\r\n }\r\n\r\n /**\r\n * This method recomputes and sets a new BoundingInfo to the mesh unless it is locked.\r\n * This means the mesh underlying bounding box and sphere are recomputed.\r\n * @param applySkeleton defines whether to apply the skeleton before computing the bounding info\r\n * @param applyMorph defines whether to apply the morph target before computing the bounding info\r\n * @returns the current mesh\r\n */\r\n public refreshBoundingInfo(applySkeleton: boolean = false, applyMorph: boolean = false): AbstractMesh {\r\n if (this._boundingInfo && this._boundingInfo.isLocked) {\r\n return this;\r\n }\r\n\r\n this._refreshBoundingInfo(this._getPositionData(applySkeleton, applyMorph), null);\r\n return this;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _refreshBoundingInfo(data: Nullable, bias: Nullable): void {\r\n if (data) {\r\n const extend = extractMinAndMax(data, 0, this.getTotalVertices(), bias);\r\n if (this._boundingInfo) {\r\n this._boundingInfo.reConstruct(extend.minimum, extend.maximum);\r\n } else {\r\n this._boundingInfo = new BoundingInfo(extend.minimum, extend.maximum);\r\n }\r\n }\r\n\r\n if (this.subMeshes) {\r\n for (let index = 0; index < this.subMeshes.length; index++) {\r\n this.subMeshes[index].refreshBoundingInfo(data);\r\n }\r\n }\r\n\r\n this._updateBoundingInfo();\r\n }\r\n\r\n /**\r\n * Internal function to get buffer data and possibly apply morphs and normals\r\n * @param applySkeleton\r\n * @param applyMorph\r\n * @param data\r\n * @param kind the kind of data you want. Can be Normal or Position\r\n */\r\n private _getData(applySkeleton: boolean = false, applyMorph: boolean = false, data?: Nullable, kind: string = VertexBuffer.PositionKind): Nullable {\r\n data = data ?? this.getVerticesData(kind)!.slice();\r\n\r\n if (data && applyMorph && this.morphTargetManager) {\r\n let faceIndexCount = 0;\r\n let positionIndex = 0;\r\n for (let vertexCount = 0; vertexCount < data.length; vertexCount++) {\r\n for (let targetCount = 0; targetCount < this.morphTargetManager.numTargets; targetCount++) {\r\n const targetMorph = this.morphTargetManager.getTarget(targetCount);\r\n const influence = targetMorph.influence;\r\n if (influence > 0.0) {\r\n const morphTargetPositions = targetMorph.getPositions();\r\n if (morphTargetPositions) {\r\n data[vertexCount] += (morphTargetPositions[vertexCount] - data[vertexCount]) * influence;\r\n }\r\n }\r\n }\r\n\r\n faceIndexCount++;\r\n if (kind === VertexBuffer.PositionKind) {\r\n if (this._positions && faceIndexCount === 3) {\r\n // We want to merge into positions every 3 indices starting (but not 0)\r\n faceIndexCount = 0;\r\n const index = positionIndex * 3;\r\n this._positions[positionIndex++].copyFromFloats(data[index], data[index + 1], data[index + 2]);\r\n }\r\n }\r\n }\r\n }\r\n\r\n if (data && applySkeleton && this.skeleton) {\r\n const matricesIndicesData = this.getVerticesData(VertexBuffer.MatricesIndicesKind);\r\n const matricesWeightsData = this.getVerticesData(VertexBuffer.MatricesWeightsKind);\r\n if (matricesWeightsData && matricesIndicesData) {\r\n const needExtras = this.numBoneInfluencers > 4;\r\n const matricesIndicesExtraData = needExtras ? this.getVerticesData(VertexBuffer.MatricesIndicesExtraKind) : null;\r\n const matricesWeightsExtraData = needExtras ? this.getVerticesData(VertexBuffer.MatricesWeightsExtraKind) : null;\r\n\r\n const skeletonMatrices = this.skeleton.getTransformMatrices(this);\r\n\r\n const tempVector = TmpVectors.Vector3[0];\r\n const finalMatrix = TmpVectors.Matrix[0];\r\n const tempMatrix = TmpVectors.Matrix[1];\r\n\r\n let matWeightIdx = 0;\r\n for (let index = 0; index < data.length; index += 3, matWeightIdx += 4) {\r\n finalMatrix.reset();\r\n\r\n let inf: number;\r\n let weight: number;\r\n for (inf = 0; inf < 4; inf++) {\r\n weight = matricesWeightsData[matWeightIdx + inf];\r\n if (weight > 0) {\r\n Matrix.FromFloat32ArrayToRefScaled(skeletonMatrices, Math.floor(matricesIndicesData[matWeightIdx + inf] * 16), weight, tempMatrix);\r\n finalMatrix.addToSelf(tempMatrix);\r\n }\r\n }\r\n if (needExtras) {\r\n for (inf = 0; inf < 4; inf++) {\r\n weight = matricesWeightsExtraData![matWeightIdx + inf];\r\n if (weight > 0) {\r\n Matrix.FromFloat32ArrayToRefScaled(skeletonMatrices, Math.floor(matricesIndicesExtraData![matWeightIdx + inf] * 16), weight, tempMatrix);\r\n finalMatrix.addToSelf(tempMatrix);\r\n }\r\n }\r\n }\r\n\r\n if (kind === VertexBuffer.NormalKind) {\r\n Vector3.TransformNormalFromFloatsToRef(data[index], data[index + 1], data[index + 2], finalMatrix, tempVector);\r\n } else {\r\n Vector3.TransformCoordinatesFromFloatsToRef(data[index], data[index + 1], data[index + 2], finalMatrix, tempVector);\r\n }\r\n tempVector.toArray(data, index);\r\n\r\n if (kind === VertexBuffer.PositionKind && this._positions) {\r\n this._positions[index / 3].copyFrom(tempVector);\r\n }\r\n }\r\n }\r\n }\r\n\r\n return data;\r\n }\r\n\r\n /**\r\n * Get the normals vertex data and optionally apply skeleton and morphing.\r\n * @param applySkeleton defines whether to apply the skeleton\r\n * @param applyMorph defines whether to apply the morph target\r\n * @returns the normals data\r\n */\r\n public getNormalsData(applySkeleton = false, applyMorph = false): Nullable {\r\n return this._getData(applySkeleton, applyMorph, null, VertexBuffer.NormalKind);\r\n }\r\n\r\n /**\r\n * Get the position vertex data and optionally apply skeleton and morphing.\r\n * @param applySkeleton defines whether to apply the skeleton\r\n * @param applyMorph defines whether to apply the morph target\r\n * @param data defines the position data to apply the skeleton and morph to\r\n * @returns the position data\r\n */\r\n public getPositionData(applySkeleton: boolean = false, applyMorph: boolean = false, data?: Nullable): Nullable {\r\n return this._getData(applySkeleton, applyMorph, data, VertexBuffer.PositionKind);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getPositionData(applySkeleton: boolean, applyMorph: boolean): Nullable {\r\n let data = this.getVerticesData(VertexBuffer.PositionKind);\r\n\r\n if (this._internalAbstractMeshDataInfo._positions) {\r\n this._internalAbstractMeshDataInfo._positions = null;\r\n }\r\n\r\n if (data && ((applySkeleton && this.skeleton) || (applyMorph && this.morphTargetManager))) {\r\n data = data.slice();\r\n this._generatePointsArray();\r\n if (this._positions) {\r\n const pos = this._positions;\r\n this._internalAbstractMeshDataInfo._positions = new Array(pos.length);\r\n for (let i = 0; i < pos.length; i++) {\r\n this._internalAbstractMeshDataInfo._positions[i] = pos[i]?.clone() || new Vector3();\r\n }\r\n }\r\n return this.getPositionData(applySkeleton, applyMorph, data);\r\n }\r\n\r\n return data;\r\n }\r\n\r\n /** @internal */\r\n public _updateBoundingInfo(): AbstractMesh {\r\n if (this._boundingInfo) {\r\n this._boundingInfo.update(this.worldMatrixFromCache);\r\n } else {\r\n this._boundingInfo = new BoundingInfo(Vector3.Zero(), Vector3.Zero(), this.worldMatrixFromCache);\r\n }\r\n this._updateSubMeshesBoundingInfo(this.worldMatrixFromCache);\r\n return this;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _updateSubMeshesBoundingInfo(matrix: DeepImmutable): AbstractMesh {\r\n if (!this.subMeshes) {\r\n return this;\r\n }\r\n const count = this.subMeshes.length;\r\n for (let subIndex = 0; subIndex < count; subIndex++) {\r\n const subMesh = this.subMeshes[subIndex];\r\n if (count > 1 || !subMesh.IsGlobal) {\r\n subMesh.updateBoundingInfo(matrix);\r\n }\r\n }\r\n return this;\r\n }\r\n\r\n /** @internal */\r\n protected _afterComputeWorldMatrix(): void {\r\n if (this.doNotSyncBoundingInfo) {\r\n return;\r\n }\r\n // Bounding info\r\n this._boundingInfoIsDirty = true;\r\n }\r\n\r\n /**\r\n * Returns `true` if the mesh is within the frustum defined by the passed array of planes.\r\n * A mesh is in the frustum if its bounding box intersects the frustum\r\n * @param frustumPlanes defines the frustum to test\r\n * @returns true if the mesh is in the frustum planes\r\n */\r\n public isInFrustum(frustumPlanes: Plane[]): boolean {\r\n return this.getBoundingInfo().isInFrustum(frustumPlanes, this.cullingStrategy);\r\n }\r\n\r\n /**\r\n * Returns `true` if the mesh is completely in the frustum defined be the passed array of planes.\r\n * A mesh is completely in the frustum if its bounding box it completely inside the frustum.\r\n * @param frustumPlanes defines the frustum to test\r\n * @returns true if the mesh is completely in the frustum planes\r\n */\r\n public isCompletelyInFrustum(frustumPlanes: Plane[]): boolean {\r\n return this.getBoundingInfo().isCompletelyInFrustum(frustumPlanes);\r\n }\r\n\r\n /**\r\n * True if the mesh intersects another mesh or a SolidParticle object\r\n * @param mesh defines a target mesh or SolidParticle to test\r\n * @param precise Unless the parameter `precise` is set to `true` the intersection is computed according to Axis Aligned Bounding Boxes (AABB), else according to OBB (Oriented BBoxes)\r\n * @param includeDescendants Can be set to true to test if the mesh defined in parameters intersects with the current mesh or any child meshes\r\n * @returns true if there is an intersection\r\n */\r\n public intersectsMesh(mesh: AbstractMesh | SolidParticle, precise: boolean = false, includeDescendants?: boolean): boolean {\r\n const boundingInfo = this.getBoundingInfo();\r\n const otherBoundingInfo = mesh.getBoundingInfo();\r\n\r\n if (boundingInfo.intersects(otherBoundingInfo, precise)) {\r\n return true;\r\n }\r\n\r\n if (includeDescendants) {\r\n for (const child of this.getChildMeshes()) {\r\n if (child.intersectsMesh(mesh, precise, true)) {\r\n return true;\r\n }\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Returns true if the passed point (Vector3) is inside the mesh bounding box\r\n * @param point defines the point to test\r\n * @returns true if there is an intersection\r\n */\r\n public intersectsPoint(point: Vector3): boolean {\r\n return this.getBoundingInfo().intersectsPoint(point);\r\n }\r\n\r\n // Collisions\r\n\r\n /**\r\n * Gets or sets a boolean indicating that this mesh can be used in the collision engine\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions\r\n */\r\n public get checkCollisions(): boolean {\r\n return this._internalAbstractMeshDataInfo._meshCollisionData._checkCollisions;\r\n }\r\n\r\n public set checkCollisions(collisionEnabled: boolean) {\r\n this._internalAbstractMeshDataInfo._meshCollisionData._checkCollisions = collisionEnabled;\r\n }\r\n\r\n /**\r\n * Gets Collider object used to compute collisions (not physics)\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions\r\n */\r\n public get collider(): Nullable {\r\n return this._internalAbstractMeshDataInfo._meshCollisionData._collider;\r\n }\r\n\r\n /**\r\n * Move the mesh using collision engine\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions\r\n * @param displacement defines the requested displacement vector\r\n * @returns the current mesh\r\n */\r\n public moveWithCollisions(displacement: Vector3): AbstractMesh {\r\n const globalPosition = this.getAbsolutePosition();\r\n\r\n globalPosition.addToRef(this.ellipsoidOffset, this._internalAbstractMeshDataInfo._meshCollisionData._oldPositionForCollisions);\r\n const coordinator = this.getScene().collisionCoordinator;\r\n\r\n if (!this._internalAbstractMeshDataInfo._meshCollisionData._collider) {\r\n this._internalAbstractMeshDataInfo._meshCollisionData._collider = coordinator.createCollider();\r\n }\r\n\r\n this._internalAbstractMeshDataInfo._meshCollisionData._collider._radius = this.ellipsoid;\r\n\r\n coordinator.getNewPosition(\r\n this._internalAbstractMeshDataInfo._meshCollisionData._oldPositionForCollisions,\r\n displacement,\r\n this._internalAbstractMeshDataInfo._meshCollisionData._collider,\r\n this.collisionRetryCount,\r\n this,\r\n this._onCollisionPositionChange,\r\n this.uniqueId\r\n );\r\n return this;\r\n }\r\n\r\n private _onCollisionPositionChange = (collisionId: number, newPosition: Vector3, collidedMesh: Nullable = null) => {\r\n newPosition.subtractToRef(\r\n this._internalAbstractMeshDataInfo._meshCollisionData._oldPositionForCollisions,\r\n this._internalAbstractMeshDataInfo._meshCollisionData._diffPositionForCollisions\r\n );\r\n\r\n if (this._internalAbstractMeshDataInfo._meshCollisionData._diffPositionForCollisions.length() > Engine.CollisionsEpsilon) {\r\n this.position.addInPlace(this._internalAbstractMeshDataInfo._meshCollisionData._diffPositionForCollisions);\r\n }\r\n\r\n if (collidedMesh) {\r\n this.onCollideObservable.notifyObservers(collidedMesh);\r\n }\r\n\r\n this.onCollisionPositionChangeObservable.notifyObservers(this.position);\r\n };\r\n\r\n // Collisions\r\n /**\r\n * @internal\r\n */\r\n public _collideForSubMesh(subMesh: SubMesh, transformMatrix: Matrix, collider: Collider): AbstractMesh {\r\n this._generatePointsArray();\r\n\r\n if (!this._positions) {\r\n return this;\r\n }\r\n\r\n // Transformation\r\n if (!subMesh._lastColliderWorldVertices || !subMesh._lastColliderTransformMatrix!.equals(transformMatrix)) {\r\n subMesh._lastColliderTransformMatrix = transformMatrix.clone();\r\n subMesh._lastColliderWorldVertices = [];\r\n subMesh._trianglePlanes = [];\r\n const start = subMesh.verticesStart;\r\n const end = subMesh.verticesStart + subMesh.verticesCount;\r\n for (let i = start; i < end; i++) {\r\n subMesh._lastColliderWorldVertices.push(Vector3.TransformCoordinates(this._positions[i], transformMatrix));\r\n }\r\n }\r\n\r\n // Collide\r\n collider._collide(\r\n subMesh._trianglePlanes,\r\n subMesh._lastColliderWorldVertices,\r\n this.getIndices(),\r\n subMesh.indexStart,\r\n subMesh.indexStart + subMesh.indexCount,\r\n subMesh.verticesStart,\r\n !!subMesh.getMaterial(),\r\n this,\r\n this._shouldConvertRHS(),\r\n subMesh.getMaterial()?.fillMode === Constants.MATERIAL_TriangleStripDrawMode\r\n );\r\n return this;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _processCollisionsForSubMeshes(collider: Collider, transformMatrix: Matrix): AbstractMesh {\r\n const subMeshes = this._scene.getCollidingSubMeshCandidates(this, collider);\r\n const len = subMeshes.length;\r\n\r\n for (let index = 0; index < len; index++) {\r\n const subMesh = subMeshes.data[index];\r\n\r\n // Bounding test\r\n if (len > 1 && !subMesh._checkCollision(collider)) {\r\n continue;\r\n }\r\n\r\n this._collideForSubMesh(subMesh, transformMatrix, collider);\r\n }\r\n return this;\r\n }\r\n\r\n /** @internal */\r\n public _shouldConvertRHS() {\r\n return false;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _checkCollision(collider: Collider): AbstractMesh {\r\n // Bounding box test\r\n if (!this.getBoundingInfo()._checkCollision(collider)) {\r\n return this;\r\n }\r\n\r\n // Transformation matrix\r\n const collisionsScalingMatrix = TmpVectors.Matrix[0];\r\n const collisionsTransformMatrix = TmpVectors.Matrix[1];\r\n Matrix.ScalingToRef(1.0 / collider._radius.x, 1.0 / collider._radius.y, 1.0 / collider._radius.z, collisionsScalingMatrix);\r\n this.worldMatrixFromCache.multiplyToRef(collisionsScalingMatrix, collisionsTransformMatrix);\r\n this._processCollisionsForSubMeshes(collider, collisionsTransformMatrix);\r\n return this;\r\n }\r\n\r\n // Picking\r\n /** @internal */\r\n public _generatePointsArray(): boolean {\r\n return false;\r\n }\r\n\r\n /**\r\n * Checks if the passed Ray intersects with the mesh. A mesh triangle can be picked both from its front and back sides,\r\n * irrespective of orientation.\r\n * @param ray defines the ray to use. It should be in the mesh's LOCAL coordinate space.\r\n * @param fastCheck defines if fast mode (but less precise) must be used (false by default)\r\n * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected\r\n * @param onlyBoundingInfo defines a boolean indicating if picking should only happen using bounding info (false by default)\r\n * @param worldToUse defines the world matrix to use to get the world coordinate of the intersection point\r\n * @param skipBoundingInfo a boolean indicating if we should skip the bounding info check\r\n * @returns the picking info\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/interactions/mesh_intersect\r\n */\r\n public intersects(\r\n ray: Ray,\r\n fastCheck?: boolean,\r\n trianglePredicate?: TrianglePickingPredicate,\r\n onlyBoundingInfo = false,\r\n worldToUse?: Matrix,\r\n skipBoundingInfo = false\r\n ): PickingInfo {\r\n const pickingInfo = new PickingInfo();\r\n const className = this.getClassName();\r\n const intersectionThreshold = className === \"InstancedLinesMesh\" || className === \"LinesMesh\" || className === \"GreasedLineMesh\" ? (this as any).intersectionThreshold : 0;\r\n const boundingInfo = this.getBoundingInfo();\r\n if (!this.subMeshes) {\r\n return pickingInfo;\r\n }\r\n if (\r\n !skipBoundingInfo &&\r\n (!ray.intersectsSphere(boundingInfo.boundingSphere, intersectionThreshold) || !ray.intersectsBox(boundingInfo.boundingBox, intersectionThreshold))\r\n ) {\r\n return pickingInfo;\r\n }\r\n\r\n if (onlyBoundingInfo) {\r\n pickingInfo.hit = skipBoundingInfo ? false : true;\r\n pickingInfo.pickedMesh = skipBoundingInfo ? null : this;\r\n pickingInfo.distance = skipBoundingInfo ? 0 : Vector3.Distance(ray.origin, boundingInfo.boundingSphere.center);\r\n pickingInfo.subMeshId = 0;\r\n return pickingInfo;\r\n }\r\n\r\n if (!this._generatePointsArray()) {\r\n return pickingInfo;\r\n }\r\n\r\n let intersectInfo: Nullable = null;\r\n\r\n const subMeshes = this._scene.getIntersectingSubMeshCandidates(this, ray);\r\n const len: number = subMeshes.length;\r\n\r\n // Check if all submeshes are using a material that don't allow picking (point/lines rendering)\r\n // if no submesh can be picked that way, then fallback to BBox picking\r\n let anySubmeshSupportIntersect = false;\r\n for (let index = 0; index < len; index++) {\r\n const subMesh = subMeshes.data[index];\r\n const material = subMesh.getMaterial();\r\n if (!material) {\r\n continue;\r\n }\r\n if (\r\n material.fillMode == Constants.MATERIAL_TriangleStripDrawMode ||\r\n material.fillMode == Constants.MATERIAL_TriangleFillMode ||\r\n material.fillMode == Constants.MATERIAL_WireFrameFillMode ||\r\n material.fillMode == Constants.MATERIAL_PointFillMode ||\r\n material.fillMode == Constants.MATERIAL_LineListDrawMode\r\n ) {\r\n anySubmeshSupportIntersect = true;\r\n break;\r\n }\r\n }\r\n\r\n // no sub mesh support intersection, fallback to BBox that has already be done\r\n if (!anySubmeshSupportIntersect) {\r\n pickingInfo.hit = true;\r\n pickingInfo.pickedMesh = this;\r\n pickingInfo.distance = Vector3.Distance(ray.origin, boundingInfo.boundingSphere.center);\r\n pickingInfo.subMeshId = -1;\r\n return pickingInfo;\r\n }\r\n\r\n // at least 1 submesh supports intersection, keep going\r\n for (let index = 0; index < len; index++) {\r\n const subMesh = subMeshes.data[index];\r\n\r\n // Bounding test\r\n if (len > 1 && !subMesh.canIntersects(ray)) {\r\n continue;\r\n }\r\n\r\n const currentIntersectInfo = subMesh.intersects(ray, this._positions, this.getIndices(), fastCheck, trianglePredicate);\r\n\r\n if (currentIntersectInfo) {\r\n if (fastCheck || !intersectInfo || currentIntersectInfo.distance < intersectInfo.distance) {\r\n intersectInfo = currentIntersectInfo;\r\n intersectInfo.subMeshId = index;\r\n\r\n if (fastCheck) {\r\n break;\r\n }\r\n }\r\n }\r\n }\r\n\r\n if (intersectInfo) {\r\n // Get picked point\r\n const world = worldToUse ?? this.getWorldMatrix();\r\n const worldOrigin = TmpVectors.Vector3[0];\r\n const direction = TmpVectors.Vector3[1];\r\n Vector3.TransformCoordinatesToRef(ray.origin, world, worldOrigin);\r\n ray.direction.scaleToRef(intersectInfo.distance, direction);\r\n const worldDirection = Vector3.TransformNormal(direction, world);\r\n const pickedPoint = worldDirection.addInPlace(worldOrigin);\r\n\r\n // Return result\r\n pickingInfo.hit = true;\r\n pickingInfo.distance = Vector3.Distance(worldOrigin, pickedPoint);\r\n pickingInfo.pickedPoint = pickedPoint;\r\n pickingInfo.pickedMesh = this;\r\n pickingInfo.bu = intersectInfo.bu || 0;\r\n pickingInfo.bv = intersectInfo.bv || 0;\r\n pickingInfo.subMeshFaceId = intersectInfo.faceId;\r\n pickingInfo.faceId = intersectInfo.faceId + subMeshes.data[intersectInfo.subMeshId].indexStart / (this.getClassName().indexOf(\"LinesMesh\") !== -1 ? 2 : 3);\r\n pickingInfo.subMeshId = intersectInfo.subMeshId;\r\n return pickingInfo;\r\n }\r\n\r\n return pickingInfo;\r\n }\r\n\r\n /**\r\n * Clones the current mesh\r\n * @param name defines the mesh name\r\n * @param newParent defines the new mesh parent\r\n * @param doNotCloneChildren defines a boolean indicating that children must not be cloned (false by default)\r\n * @returns the new mesh\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public clone(name: string, newParent: Nullable, doNotCloneChildren?: boolean): Nullable {\r\n return null;\r\n }\r\n\r\n /**\r\n * Disposes all the submeshes of the current meshnp\r\n * @returns the current mesh\r\n */\r\n public releaseSubMeshes(): AbstractMesh {\r\n if (this.subMeshes) {\r\n while (this.subMeshes.length) {\r\n this.subMeshes[0].dispose();\r\n }\r\n } else {\r\n this.subMeshes = new Array();\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Releases resources associated with this abstract mesh.\r\n * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default)\r\n * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default)\r\n */\r\n public dispose(doNotRecurse?: boolean, disposeMaterialAndTextures = false): void {\r\n let index: number;\r\n\r\n // mesh map release.\r\n if (this._scene.useMaterialMeshMap) {\r\n // remove from material mesh map id needed\r\n if (this._internalAbstractMeshDataInfo._material && this._internalAbstractMeshDataInfo._material.meshMap) {\r\n this._internalAbstractMeshDataInfo._material.meshMap[this.uniqueId] = undefined;\r\n }\r\n }\r\n\r\n // Smart Array Retainers.\r\n this.getScene().freeActiveMeshes();\r\n this.getScene().freeRenderingGroups();\r\n\r\n // Action manager\r\n if (this.actionManager !== undefined && this.actionManager !== null) {\r\n this.actionManager.dispose();\r\n this.actionManager = null;\r\n }\r\n\r\n // Skeleton\r\n this._internalAbstractMeshDataInfo._skeleton = null;\r\n\r\n if (this._transformMatrixTexture) {\r\n this._transformMatrixTexture.dispose();\r\n this._transformMatrixTexture = null;\r\n }\r\n\r\n // Intersections in progress\r\n for (index = 0; index < this._intersectionsInProgress.length; index++) {\r\n const other = this._intersectionsInProgress[index];\r\n\r\n const pos = other._intersectionsInProgress.indexOf(this);\r\n other._intersectionsInProgress.splice(pos, 1);\r\n }\r\n\r\n this._intersectionsInProgress.length = 0;\r\n\r\n // Lights\r\n const lights = this.getScene().lights;\r\n\r\n lights.forEach((light: Light) => {\r\n let meshIndex = light.includedOnlyMeshes.indexOf(this);\r\n\r\n if (meshIndex !== -1) {\r\n light.includedOnlyMeshes.splice(meshIndex, 1);\r\n }\r\n\r\n meshIndex = light.excludedMeshes.indexOf(this);\r\n\r\n if (meshIndex !== -1) {\r\n light.excludedMeshes.splice(meshIndex, 1);\r\n }\r\n\r\n // Shadow generators\r\n const generators = light.getShadowGenerators();\r\n if (generators) {\r\n const iterator = generators.values();\r\n for (let key = iterator.next(); key.done !== true; key = iterator.next()) {\r\n const generator = key.value;\r\n const shadowMap = generator.getShadowMap();\r\n\r\n if (shadowMap && shadowMap.renderList) {\r\n meshIndex = shadowMap.renderList.indexOf(this);\r\n\r\n if (meshIndex !== -1) {\r\n shadowMap.renderList.splice(meshIndex, 1);\r\n }\r\n }\r\n }\r\n }\r\n });\r\n\r\n // SubMeshes\r\n if (this.getClassName() !== \"InstancedMesh\" || this.getClassName() !== \"InstancedLinesMesh\") {\r\n this.releaseSubMeshes();\r\n }\r\n\r\n // Query\r\n const engine = this.getScene().getEngine();\r\n if (this._occlusionQuery !== null) {\r\n this.isOcclusionQueryInProgress = false;\r\n engine.deleteQuery(this._occlusionQuery);\r\n this._occlusionQuery = null;\r\n }\r\n\r\n // Engine\r\n engine.wipeCaches();\r\n\r\n // Remove from scene\r\n this.getScene().removeMesh(this);\r\n\r\n if (this._parentContainer) {\r\n const index = this._parentContainer.meshes.indexOf(this);\r\n if (index > -1) {\r\n this._parentContainer.meshes.splice(index, 1);\r\n }\r\n this._parentContainer = null;\r\n }\r\n\r\n if (disposeMaterialAndTextures) {\r\n if (this.material) {\r\n if (this.material.getClassName() === \"MultiMaterial\") {\r\n this.material.dispose(false, true, true);\r\n } else {\r\n this.material.dispose(false, true);\r\n }\r\n }\r\n }\r\n\r\n if (!doNotRecurse) {\r\n // Particles\r\n for (index = 0; index < this.getScene().particleSystems.length; index++) {\r\n if (this.getScene().particleSystems[index].emitter === this) {\r\n this.getScene().particleSystems[index].dispose();\r\n index--;\r\n }\r\n }\r\n }\r\n\r\n // facet data\r\n if (this._internalAbstractMeshDataInfo._facetData.facetDataEnabled) {\r\n this.disableFacetData();\r\n }\r\n\r\n this._uniformBuffer.dispose();\r\n\r\n this.onAfterWorldMatrixUpdateObservable.clear();\r\n this.onCollideObservable.clear();\r\n this.onCollisionPositionChangeObservable.clear();\r\n this.onRebuildObservable.clear();\r\n\r\n super.dispose(doNotRecurse, disposeMaterialAndTextures);\r\n }\r\n\r\n /**\r\n * Adds the passed mesh as a child to the current mesh\r\n * @param mesh defines the child mesh\r\n * @param preserveScalingSign if true, keep scaling sign of child. Otherwise, scaling sign might change.\r\n * @returns the current mesh\r\n */\r\n public addChild(mesh: AbstractMesh, preserveScalingSign: boolean = false): AbstractMesh {\r\n mesh.setParent(this, preserveScalingSign);\r\n return this;\r\n }\r\n\r\n /**\r\n * Removes the passed mesh from the current mesh children list\r\n * @param mesh defines the child mesh\r\n * @param preserveScalingSign if true, keep scaling sign of child. Otherwise, scaling sign might change.\r\n * @returns the current mesh\r\n */\r\n public removeChild(mesh: AbstractMesh, preserveScalingSign: boolean = false): AbstractMesh {\r\n mesh.setParent(null, preserveScalingSign);\r\n return this;\r\n }\r\n\r\n // Facet data\r\n /** @internal */\r\n private _initFacetData(): AbstractMesh {\r\n const data = this._internalAbstractMeshDataInfo._facetData;\r\n if (!data.facetNormals) {\r\n data.facetNormals = new Array();\r\n }\r\n if (!data.facetPositions) {\r\n data.facetPositions = new Array();\r\n }\r\n if (!data.facetPartitioning) {\r\n data.facetPartitioning = new Array();\r\n }\r\n data.facetNb = ((this.getIndices()).length / 3) | 0;\r\n data.partitioningSubdivisions = data.partitioningSubdivisions ? data.partitioningSubdivisions : 10; // default nb of partitioning subdivisions = 10\r\n data.partitioningBBoxRatio = data.partitioningBBoxRatio ? data.partitioningBBoxRatio : 1.01; // default ratio 1.01 = the partitioning is 1% bigger than the bounding box\r\n for (let f = 0; f < data.facetNb; f++) {\r\n data.facetNormals[f] = Vector3.Zero();\r\n data.facetPositions[f] = Vector3.Zero();\r\n }\r\n data.facetDataEnabled = true;\r\n return this;\r\n }\r\n\r\n /**\r\n * Updates the mesh facetData arrays and the internal partitioning when the mesh is morphed or updated.\r\n * This method can be called within the render loop.\r\n * You don't need to call this method by yourself in the render loop when you update/morph a mesh with the methods CreateXXX() as they automatically manage this computation\r\n * @returns the current mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\r\n */\r\n public updateFacetData(): AbstractMesh {\r\n const data = this._internalAbstractMeshDataInfo._facetData;\r\n if (!data.facetDataEnabled) {\r\n this._initFacetData();\r\n }\r\n const positions = this.getVerticesData(VertexBuffer.PositionKind);\r\n const indices = this.getIndices();\r\n const normals = this.getVerticesData(VertexBuffer.NormalKind);\r\n const bInfo = this.getBoundingInfo();\r\n\r\n if (data.facetDepthSort && !data.facetDepthSortEnabled) {\r\n // init arrays, matrix and sort function on first call\r\n data.facetDepthSortEnabled = true;\r\n if (indices instanceof Uint16Array) {\r\n data.depthSortedIndices = new Uint16Array(indices!);\r\n } else if (indices instanceof Uint32Array) {\r\n data.depthSortedIndices = new Uint32Array(indices!);\r\n } else {\r\n let needs32bits = false;\r\n for (let i = 0; i < indices!.length; i++) {\r\n if (indices![i] > 65535) {\r\n needs32bits = true;\r\n break;\r\n }\r\n }\r\n if (needs32bits) {\r\n data.depthSortedIndices = new Uint32Array(indices!);\r\n } else {\r\n data.depthSortedIndices = new Uint16Array(indices!);\r\n }\r\n }\r\n data.facetDepthSortFunction = function (f1, f2) {\r\n return f2.sqDistance - f1.sqDistance;\r\n };\r\n if (!data.facetDepthSortFrom) {\r\n const camera = this.getScene().activeCamera;\r\n data.facetDepthSortFrom = camera ? camera.position : Vector3.Zero();\r\n }\r\n data.depthSortedFacets = [];\r\n for (let f = 0; f < data.facetNb; f++) {\r\n const depthSortedFacet = { ind: f * 3, sqDistance: 0.0 };\r\n data.depthSortedFacets.push(depthSortedFacet);\r\n }\r\n data.invertedMatrix = Matrix.Identity();\r\n data.facetDepthSortOrigin = Vector3.Zero();\r\n }\r\n\r\n data.bbSize.x = bInfo.maximum.x - bInfo.minimum.x > Epsilon ? bInfo.maximum.x - bInfo.minimum.x : Epsilon;\r\n data.bbSize.y = bInfo.maximum.y - bInfo.minimum.y > Epsilon ? bInfo.maximum.y - bInfo.minimum.y : Epsilon;\r\n data.bbSize.z = bInfo.maximum.z - bInfo.minimum.z > Epsilon ? bInfo.maximum.z - bInfo.minimum.z : Epsilon;\r\n let bbSizeMax = data.bbSize.x > data.bbSize.y ? data.bbSize.x : data.bbSize.y;\r\n bbSizeMax = bbSizeMax > data.bbSize.z ? bbSizeMax : data.bbSize.z;\r\n data.subDiv.max = data.partitioningSubdivisions;\r\n data.subDiv.X = Math.floor((data.subDiv.max * data.bbSize.x) / bbSizeMax); // adjust the number of subdivisions per axis\r\n data.subDiv.Y = Math.floor((data.subDiv.max * data.bbSize.y) / bbSizeMax); // according to each bbox size per axis\r\n data.subDiv.Z = Math.floor((data.subDiv.max * data.bbSize.z) / bbSizeMax);\r\n data.subDiv.X = data.subDiv.X < 1 ? 1 : data.subDiv.X; // at least one subdivision\r\n data.subDiv.Y = data.subDiv.Y < 1 ? 1 : data.subDiv.Y;\r\n data.subDiv.Z = data.subDiv.Z < 1 ? 1 : data.subDiv.Z;\r\n // set the parameters for ComputeNormals()\r\n data.facetParameters.facetNormals = this.getFacetLocalNormals();\r\n data.facetParameters.facetPositions = this.getFacetLocalPositions();\r\n data.facetParameters.facetPartitioning = this.getFacetLocalPartitioning();\r\n data.facetParameters.bInfo = bInfo;\r\n data.facetParameters.bbSize = data.bbSize;\r\n data.facetParameters.subDiv = data.subDiv;\r\n data.facetParameters.ratio = this.partitioningBBoxRatio;\r\n data.facetParameters.depthSort = data.facetDepthSort;\r\n if (data.facetDepthSort && data.facetDepthSortEnabled) {\r\n this.computeWorldMatrix(true);\r\n this._worldMatrix.invertToRef(data.invertedMatrix);\r\n Vector3.TransformCoordinatesToRef(data.facetDepthSortFrom, data.invertedMatrix, data.facetDepthSortOrigin);\r\n data.facetParameters.distanceTo = data.facetDepthSortOrigin;\r\n }\r\n data.facetParameters.depthSortedFacets = data.depthSortedFacets;\r\n if (normals) {\r\n VertexData.ComputeNormals(positions, indices, normals, data.facetParameters);\r\n }\r\n\r\n if (data.facetDepthSort && data.facetDepthSortEnabled) {\r\n data.depthSortedFacets.sort(data.facetDepthSortFunction);\r\n const l = (data.depthSortedIndices.length / 3) | 0;\r\n for (let f = 0; f < l; f++) {\r\n const sind = data.depthSortedFacets[f].ind;\r\n data.depthSortedIndices[f * 3] = indices![sind];\r\n data.depthSortedIndices[f * 3 + 1] = indices![sind + 1];\r\n data.depthSortedIndices[f * 3 + 2] = indices![sind + 2];\r\n }\r\n this.updateIndices(data.depthSortedIndices, undefined, true);\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns the facetLocalNormals array.\r\n * The normals are expressed in the mesh local spac\r\n * @returns an array of Vector3\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\r\n */\r\n public getFacetLocalNormals(): Vector3[] {\r\n const facetData = this._internalAbstractMeshDataInfo._facetData;\r\n if (!facetData.facetNormals) {\r\n this.updateFacetData();\r\n }\r\n return facetData.facetNormals;\r\n }\r\n\r\n /**\r\n * Returns the facetLocalPositions array.\r\n * The facet positions are expressed in the mesh local space\r\n * @returns an array of Vector3\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\r\n */\r\n public getFacetLocalPositions(): Vector3[] {\r\n const facetData = this._internalAbstractMeshDataInfo._facetData;\r\n if (!facetData.facetPositions) {\r\n this.updateFacetData();\r\n }\r\n return facetData.facetPositions;\r\n }\r\n\r\n /**\r\n * Returns the facetLocalPartitioning array\r\n * @returns an array of array of numbers\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\r\n */\r\n public getFacetLocalPartitioning(): number[][] {\r\n const facetData = this._internalAbstractMeshDataInfo._facetData;\r\n\r\n if (!facetData.facetPartitioning) {\r\n this.updateFacetData();\r\n }\r\n return facetData.facetPartitioning;\r\n }\r\n\r\n /**\r\n * Returns the i-th facet position in the world system.\r\n * This method allocates a new Vector3 per call\r\n * @param i defines the facet index\r\n * @returns a new Vector3\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\r\n */\r\n public getFacetPosition(i: number): Vector3 {\r\n const pos = Vector3.Zero();\r\n this.getFacetPositionToRef(i, pos);\r\n return pos;\r\n }\r\n\r\n /**\r\n * Sets the reference Vector3 with the i-th facet position in the world system\r\n * @param i defines the facet index\r\n * @param ref defines the target vector\r\n * @returns the current mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\r\n */\r\n public getFacetPositionToRef(i: number, ref: Vector3): AbstractMesh {\r\n const localPos = this.getFacetLocalPositions()[i];\r\n const world = this.getWorldMatrix();\r\n Vector3.TransformCoordinatesToRef(localPos, world, ref);\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns the i-th facet normal in the world system.\r\n * This method allocates a new Vector3 per call\r\n * @param i defines the facet index\r\n * @returns a new Vector3\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\r\n */\r\n public getFacetNormal(i: number): Vector3 {\r\n const norm = Vector3.Zero();\r\n this.getFacetNormalToRef(i, norm);\r\n return norm;\r\n }\r\n\r\n /**\r\n * Sets the reference Vector3 with the i-th facet normal in the world system\r\n * @param i defines the facet index\r\n * @param ref defines the target vector\r\n * @returns the current mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\r\n */\r\n public getFacetNormalToRef(i: number, ref: Vector3) {\r\n const localNorm = this.getFacetLocalNormals()[i];\r\n Vector3.TransformNormalToRef(localNorm, this.getWorldMatrix(), ref);\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns the facets (in an array) in the same partitioning block than the one the passed coordinates are located (expressed in the mesh local system)\r\n * @param x defines x coordinate\r\n * @param y defines y coordinate\r\n * @param z defines z coordinate\r\n * @returns the array of facet indexes\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\r\n */\r\n public getFacetsAtLocalCoordinates(x: number, y: number, z: number): Nullable {\r\n const bInfo = this.getBoundingInfo();\r\n const data = this._internalAbstractMeshDataInfo._facetData;\r\n\r\n const ox = Math.floor(((x - bInfo.minimum.x * data.partitioningBBoxRatio) * data.subDiv.X * data.partitioningBBoxRatio) / data.bbSize.x);\r\n const oy = Math.floor(((y - bInfo.minimum.y * data.partitioningBBoxRatio) * data.subDiv.Y * data.partitioningBBoxRatio) / data.bbSize.y);\r\n const oz = Math.floor(((z - bInfo.minimum.z * data.partitioningBBoxRatio) * data.subDiv.Z * data.partitioningBBoxRatio) / data.bbSize.z);\r\n if (ox < 0 || ox > data.subDiv.max || oy < 0 || oy > data.subDiv.max || oz < 0 || oz > data.subDiv.max) {\r\n return null;\r\n }\r\n return data.facetPartitioning[ox + data.subDiv.max * oy + data.subDiv.max * data.subDiv.max * oz];\r\n }\r\n\r\n /**\r\n * Returns the closest mesh facet index at (x,y,z) World coordinates, null if not found\r\n * @param x defines x coordinate\r\n * @param y defines y coordinate\r\n * @param z defines z coordinate\r\n * @param projected sets as the (x,y,z) world projection on the facet\r\n * @param checkFace if true (default false), only the facet \"facing\" to (x,y,z) or only the ones \"turning their backs\", according to the parameter \"facing\" are returned\r\n * @param facing if facing and checkFace are true, only the facet \"facing\" to (x, y, z) are returned : positive dot (x, y, z) * facet position. If facing si false and checkFace is true, only the facet \"turning their backs\" to (x, y, z) are returned : negative dot (x, y, z) * facet position\r\n * @returns the face index if found (or null instead)\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\r\n */\r\n public getClosestFacetAtCoordinates(x: number, y: number, z: number, projected?: Vector3, checkFace: boolean = false, facing: boolean = true): Nullable {\r\n const world = this.getWorldMatrix();\r\n const invMat = TmpVectors.Matrix[5];\r\n world.invertToRef(invMat);\r\n const invVect = TmpVectors.Vector3[8];\r\n Vector3.TransformCoordinatesFromFloatsToRef(x, y, z, invMat, invVect); // transform (x,y,z) to coordinates in the mesh local space\r\n const closest = this.getClosestFacetAtLocalCoordinates(invVect.x, invVect.y, invVect.z, projected, checkFace, facing);\r\n if (projected) {\r\n // transform the local computed projected vector to world coordinates\r\n Vector3.TransformCoordinatesFromFloatsToRef(projected.x, projected.y, projected.z, world, projected);\r\n }\r\n return closest;\r\n }\r\n\r\n /**\r\n * Returns the closest mesh facet index at (x,y,z) local coordinates, null if not found\r\n * @param x defines x coordinate\r\n * @param y defines y coordinate\r\n * @param z defines z coordinate\r\n * @param projected sets as the (x,y,z) local projection on the facet\r\n * @param checkFace if true (default false), only the facet \"facing\" to (x,y,z) or only the ones \"turning their backs\", according to the parameter \"facing\" are returned\r\n * @param facing if facing and checkFace are true, only the facet \"facing\" to (x, y, z) are returned : positive dot (x, y, z) * facet position. If facing si false and checkFace is true, only the facet \"turning their backs\" to (x, y, z) are returned : negative dot (x, y, z) * facet position\r\n * @returns the face index if found (or null instead)\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\r\n */\r\n public getClosestFacetAtLocalCoordinates(x: number, y: number, z: number, projected?: Vector3, checkFace: boolean = false, facing: boolean = true): Nullable {\r\n let closest = null;\r\n let tmpx = 0.0;\r\n let tmpy = 0.0;\r\n let tmpz = 0.0;\r\n let d = 0.0; // tmp dot facet normal * facet position\r\n let t0 = 0.0;\r\n let projx = 0.0;\r\n let projy = 0.0;\r\n let projz = 0.0;\r\n // Get all the facets in the same partitioning block than (x, y, z)\r\n const facetPositions = this.getFacetLocalPositions();\r\n const facetNormals = this.getFacetLocalNormals();\r\n const facetsInBlock = this.getFacetsAtLocalCoordinates(x, y, z);\r\n if (!facetsInBlock) {\r\n return null;\r\n }\r\n // Get the closest facet to (x, y, z)\r\n let shortest = Number.MAX_VALUE; // init distance vars\r\n let tmpDistance = shortest;\r\n let fib; // current facet in the block\r\n let norm; // current facet normal\r\n let p0; // current facet barycenter position\r\n // loop on all the facets in the current partitioning block\r\n for (let idx = 0; idx < facetsInBlock.length; idx++) {\r\n fib = facetsInBlock[idx];\r\n norm = facetNormals[fib];\r\n p0 = facetPositions[fib];\r\n\r\n d = (x - p0.x) * norm.x + (y - p0.y) * norm.y + (z - p0.z) * norm.z;\r\n if (!checkFace || (checkFace && facing && d >= 0.0) || (checkFace && !facing && d <= 0.0)) {\r\n // compute (x,y,z) projection on the facet = (projx, projy, projz)\r\n d = norm.x * p0.x + norm.y * p0.y + norm.z * p0.z;\r\n t0 = -(norm.x * x + norm.y * y + norm.z * z - d) / (norm.x * norm.x + norm.y * norm.y + norm.z * norm.z);\r\n projx = x + norm.x * t0;\r\n projy = y + norm.y * t0;\r\n projz = z + norm.z * t0;\r\n\r\n tmpx = projx - x;\r\n tmpy = projy - y;\r\n tmpz = projz - z;\r\n tmpDistance = tmpx * tmpx + tmpy * tmpy + tmpz * tmpz; // compute length between (x, y, z) and its projection on the facet\r\n if (tmpDistance < shortest) {\r\n // just keep the closest facet to (x, y, z)\r\n shortest = tmpDistance;\r\n closest = fib;\r\n if (projected) {\r\n projected.x = projx;\r\n projected.y = projy;\r\n projected.z = projz;\r\n }\r\n }\r\n }\r\n }\r\n return closest;\r\n }\r\n\r\n /**\r\n * Returns the object \"parameter\" set with all the expected parameters for facetData computation by ComputeNormals()\r\n * @returns the parameters\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\r\n */\r\n public getFacetDataParameters(): any {\r\n return this._internalAbstractMeshDataInfo._facetData.facetParameters;\r\n }\r\n\r\n /**\r\n * Disables the feature FacetData and frees the related memory\r\n * @returns the current mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/facetData\r\n */\r\n public disableFacetData(): AbstractMesh {\r\n const facetData = this._internalAbstractMeshDataInfo._facetData;\r\n if (facetData.facetDataEnabled) {\r\n facetData.facetDataEnabled = false;\r\n facetData.facetPositions = new Array();\r\n facetData.facetNormals = new Array();\r\n facetData.facetPartitioning = new Array();\r\n facetData.facetParameters = null;\r\n facetData.depthSortedIndices = new Uint32Array(0);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Updates the AbstractMesh indices array\r\n * @param indices defines the data source\r\n * @param offset defines the offset in the index buffer where to store the new data (can be null)\r\n * @param gpuMemoryOnly defines a boolean indicating that only the GPU memory must be updated leaving the CPU version of the indices unchanged (false by default)\r\n * @returns the current mesh\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public updateIndices(indices: IndicesArray, offset?: number, gpuMemoryOnly = false): AbstractMesh {\r\n return this;\r\n }\r\n\r\n /**\r\n * Creates new normals data for the mesh\r\n * @param updatable defines if the normal vertex buffer must be flagged as updatable\r\n * @returns the current mesh\r\n */\r\n public createNormals(updatable: boolean): AbstractMesh {\r\n const positions = this.getVerticesData(VertexBuffer.PositionKind);\r\n const indices = this.getIndices();\r\n let normals: FloatArray;\r\n\r\n if (this.isVerticesDataPresent(VertexBuffer.NormalKind)) {\r\n normals = this.getVerticesData(VertexBuffer.NormalKind);\r\n } else {\r\n normals = [];\r\n }\r\n\r\n VertexData.ComputeNormals(positions, indices, normals, { useRightHandedSystem: this.getScene().useRightHandedSystem });\r\n this.setVerticesData(VertexBuffer.NormalKind, normals, updatable);\r\n return this;\r\n }\r\n\r\n /**\r\n * Align the mesh with a normal\r\n * @param normal defines the normal to use\r\n * @param upDirection can be used to redefined the up vector to use (will use the (0, 1, 0) by default)\r\n * @returns the current mesh\r\n */\r\n public alignWithNormal(normal: Vector3, upDirection?: Vector3): AbstractMesh {\r\n if (!upDirection) {\r\n upDirection = Axis.Y;\r\n }\r\n\r\n const axisX = TmpVectors.Vector3[0];\r\n const axisZ = TmpVectors.Vector3[1];\r\n Vector3.CrossToRef(upDirection, normal, axisZ);\r\n Vector3.CrossToRef(normal, axisZ, axisX);\r\n\r\n if (this.rotationQuaternion) {\r\n Quaternion.RotationQuaternionFromAxisToRef(axisX, normal, axisZ, this.rotationQuaternion);\r\n } else {\r\n Vector3.RotationFromAxisToRef(axisX, normal, axisZ, this.rotation);\r\n }\r\n return this;\r\n }\r\n\r\n /** @internal */\r\n public _checkOcclusionQuery(): boolean {\r\n // Will be replaced by correct code if Occlusion queries are referenced\r\n return false;\r\n }\r\n\r\n /**\r\n * Disables the mesh edge rendering mode\r\n * @returns the currentAbstractMesh\r\n */\r\n disableEdgesRendering(): AbstractMesh {\r\n throw _WarnImport(\"EdgesRenderer\");\r\n }\r\n\r\n /**\r\n * Enables the edge rendering mode on the mesh.\r\n * This mode makes the mesh edges visible\r\n * @param epsilon defines the maximal distance between two angles to detect a face\r\n * @param checkVerticesInsteadOfIndices indicates that we should check vertex list directly instead of faces\r\n * @param options options to the edge renderer\r\n * @returns the currentAbstractMesh\r\n * @see https://www.babylonjs-playground.com/#19O9TU#0\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n enableEdgesRendering(epsilon?: number, checkVerticesInsteadOfIndices?: boolean, options?: IEdgesRendererOptions): AbstractMesh {\r\n throw _WarnImport(\"EdgesRenderer\");\r\n }\r\n\r\n /**\r\n * This function returns all of the particle systems in the scene that use the mesh as an emitter.\r\n * @returns an array of particle systems in the scene that use the mesh as an emitter\r\n */\r\n public getConnectedParticleSystems(): IParticleSystem[] {\r\n return this._scene.particleSystems.filter((particleSystem) => particleSystem.emitter === this);\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.AbstractMesh\", AbstractMesh);\r\n","import type { Effect } from \"./effect\";\r\nimport type { IClipPlanesHolder } from \"../Misc/interfaces/iClipPlanesHolder\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { Plane } from \"../Maths/math.plane\";\r\n\r\n/** @internal */\r\nexport function addClipPlaneUniforms(uniforms: string[]): void {\r\n if (uniforms.indexOf(\"vClipPlane\") === -1) {\r\n uniforms.push(\"vClipPlane\");\r\n }\r\n if (uniforms.indexOf(\"vClipPlane2\") === -1) {\r\n uniforms.push(\"vClipPlane2\");\r\n }\r\n if (uniforms.indexOf(\"vClipPlane3\") === -1) {\r\n uniforms.push(\"vClipPlane3\");\r\n }\r\n if (uniforms.indexOf(\"vClipPlane4\") === -1) {\r\n uniforms.push(\"vClipPlane4\");\r\n }\r\n if (uniforms.indexOf(\"vClipPlane5\") === -1) {\r\n uniforms.push(\"vClipPlane5\");\r\n }\r\n if (uniforms.indexOf(\"vClipPlane6\") === -1) {\r\n uniforms.push(\"vClipPlane6\");\r\n }\r\n}\r\n\r\n/** @internal */\r\nexport function prepareStringDefinesForClipPlanes(primaryHolder: IClipPlanesHolder, secondaryHolder: IClipPlanesHolder, defines: string[]): void {\r\n const clipPlane = !!(primaryHolder.clipPlane ?? secondaryHolder.clipPlane);\r\n const clipPlane2 = !!(primaryHolder.clipPlane2 ?? secondaryHolder.clipPlane2);\r\n const clipPlane3 = !!(primaryHolder.clipPlane3 ?? secondaryHolder.clipPlane3);\r\n const clipPlane4 = !!(primaryHolder.clipPlane4 ?? secondaryHolder.clipPlane4);\r\n const clipPlane5 = !!(primaryHolder.clipPlane5 ?? secondaryHolder.clipPlane5);\r\n const clipPlane6 = !!(primaryHolder.clipPlane6 ?? secondaryHolder.clipPlane6);\r\n\r\n if (clipPlane) defines.push(\"#define CLIPPLANE\");\r\n if (clipPlane2) defines.push(\"#define CLIPPLANE2\");\r\n if (clipPlane3) defines.push(\"#define CLIPPLANE3\");\r\n if (clipPlane4) defines.push(\"#define CLIPPLANE4\");\r\n if (clipPlane5) defines.push(\"#define CLIPPLANE5\");\r\n if (clipPlane6) defines.push(\"#define CLIPPLANE6\");\r\n}\r\n\r\n/** @internal */\r\nexport function prepareDefinesForClipPlanes(primaryHolder: IClipPlanesHolder, secondaryHolder: IClipPlanesHolder, defines: Record): boolean {\r\n let changed = false;\r\n\r\n const clipPlane = !!(primaryHolder.clipPlane ?? secondaryHolder.clipPlane);\r\n const clipPlane2 = !!(primaryHolder.clipPlane2 ?? secondaryHolder.clipPlane2);\r\n const clipPlane3 = !!(primaryHolder.clipPlane3 ?? secondaryHolder.clipPlane3);\r\n const clipPlane4 = !!(primaryHolder.clipPlane4 ?? secondaryHolder.clipPlane4);\r\n const clipPlane5 = !!(primaryHolder.clipPlane5 ?? secondaryHolder.clipPlane5);\r\n const clipPlane6 = !!(primaryHolder.clipPlane6 ?? secondaryHolder.clipPlane6);\r\n\r\n // Do not factorize this code, it breaks browsers optimizations.\r\n if (defines[\"CLIPPLANE\"] !== clipPlane) {\r\n defines[\"CLIPPLANE\"] = clipPlane;\r\n changed = true;\r\n }\r\n if (defines[\"CLIPPLANE2\"] !== clipPlane2) {\r\n defines[\"CLIPPLANE2\"] = clipPlane2;\r\n changed = true;\r\n }\r\n if (defines[\"CLIPPLANE3\"] !== clipPlane3) {\r\n defines[\"CLIPPLANE3\"] = clipPlane3;\r\n changed = true;\r\n }\r\n if (defines[\"CLIPPLANE4\"] !== clipPlane4) {\r\n defines[\"CLIPPLANE4\"] = clipPlane4;\r\n changed = true;\r\n }\r\n if (defines[\"CLIPPLANE5\"] !== clipPlane5) {\r\n defines[\"CLIPPLANE5\"] = clipPlane5;\r\n changed = true;\r\n }\r\n if (defines[\"CLIPPLANE6\"] !== clipPlane6) {\r\n defines[\"CLIPPLANE6\"] = clipPlane6;\r\n changed = true;\r\n }\r\n\r\n return changed;\r\n}\r\n\r\n/** @internal */\r\nexport function bindClipPlane(effect: Effect, primaryHolder: IClipPlanesHolder, secondaryHolder: IClipPlanesHolder): void {\r\n let clipPlane = primaryHolder.clipPlane ?? secondaryHolder.clipPlane;\r\n setClipPlane(effect, \"vClipPlane\", clipPlane);\r\n clipPlane = primaryHolder.clipPlane2 ?? secondaryHolder.clipPlane2;\r\n setClipPlane(effect, \"vClipPlane2\", clipPlane);\r\n clipPlane = primaryHolder.clipPlane3 ?? secondaryHolder.clipPlane3;\r\n setClipPlane(effect, \"vClipPlane3\", clipPlane);\r\n clipPlane = primaryHolder.clipPlane4 ?? secondaryHolder.clipPlane4;\r\n setClipPlane(effect, \"vClipPlane4\", clipPlane);\r\n clipPlane = primaryHolder.clipPlane5 ?? secondaryHolder.clipPlane5;\r\n setClipPlane(effect, \"vClipPlane5\", clipPlane);\r\n clipPlane = primaryHolder.clipPlane6 ?? secondaryHolder.clipPlane6;\r\n setClipPlane(effect, \"vClipPlane6\", clipPlane);\r\n}\r\n\r\nfunction setClipPlane(effect: Effect, uniformName: string, clipPlane: Nullable): void {\r\n if (clipPlane) {\r\n effect.setFloat4(uniformName, clipPlane.normal.x, clipPlane.normal.y, clipPlane.normal.z, clipPlane.d);\r\n }\r\n}\r\n","import { Logger } from \"../Misc/logger\";\r\nimport type { Nullable } from \"../types\";\r\nimport { Camera } from \"../Cameras/camera\";\r\nimport { Scene } from \"../scene\";\r\nimport type { Engine } from \"../Engines/engine\";\r\nimport { EngineStore } from \"../Engines/engineStore\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { Mesh } from \"../Meshes/mesh\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport type { Light } from \"../Lights/light\";\r\nimport { LightConstants } from \"../Lights/lightConstants\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport type { PrePassConfiguration } from \"../Materials/prePassConfiguration\";\r\n\r\nimport type { UniformBuffer } from \"./uniformBuffer\";\r\nimport type { Effect, IEffectCreationOptions } from \"./effect\";\r\nimport type { BaseTexture } from \"../Materials/Textures/baseTexture\";\r\nimport type { MaterialDefines } from \"./materialDefines\";\r\nimport { Color3 } from \"../Maths/math.color\";\r\nimport type { EffectFallbacks } from \"./effectFallbacks\";\r\nimport { prepareDefinesForClipPlanes } from \"./clipPlaneMaterialHelper\";\r\nimport type { Material } from \"./material\";\r\n\r\n/**\r\n * \"Static Class\" containing the most commonly used helper while dealing with material for rendering purpose.\r\n *\r\n * It contains the basic tools to help defining defines, binding uniform for the common part of the materials.\r\n *\r\n * This works by convention in BabylonJS but is meant to be use only with shader following the in place naming rules and conventions.\r\n */\r\nexport class MaterialHelper {\r\n /**\r\n * Binds the scene's uniform buffer to the effect.\r\n * @param effect defines the effect to bind to the scene uniform buffer\r\n * @param sceneUbo defines the uniform buffer storing scene data\r\n */\r\n public static BindSceneUniformBuffer(effect: Effect, sceneUbo: UniformBuffer): void {\r\n sceneUbo.bindToEffect(effect, \"Scene\");\r\n }\r\n\r\n /**\r\n * Helps preparing the defines values about the UVs in used in the effect.\r\n * UVs are shared as much as we can across channels in the shaders.\r\n * @param texture The texture we are preparing the UVs for\r\n * @param defines The defines to update\r\n * @param key The channel key \"diffuse\", \"specular\"... used in the shader\r\n */\r\n public static PrepareDefinesForMergedUV(texture: BaseTexture, defines: any, key: string): void {\r\n defines._needUVs = true;\r\n defines[key] = true;\r\n if (texture.optimizeUVAllocation && texture.getTextureMatrix().isIdentityAs3x2()) {\r\n defines[key + \"DIRECTUV\"] = texture.coordinatesIndex + 1;\r\n defines[\"MAINUV\" + (texture.coordinatesIndex + 1)] = true;\r\n } else {\r\n defines[key + \"DIRECTUV\"] = 0;\r\n }\r\n }\r\n\r\n /**\r\n * Binds a texture matrix value to its corresponding uniform\r\n * @param texture The texture to bind the matrix for\r\n * @param uniformBuffer The uniform buffer receiving the data\r\n * @param key The channel key \"diffuse\", \"specular\"... used in the shader\r\n */\r\n public static BindTextureMatrix(texture: BaseTexture, uniformBuffer: UniformBuffer, key: string): void {\r\n const matrix = texture.getTextureMatrix();\r\n\r\n uniformBuffer.updateMatrix(key + \"Matrix\", matrix);\r\n }\r\n\r\n /**\r\n * Gets the current status of the fog (should it be enabled?)\r\n * @param mesh defines the mesh to evaluate for fog support\r\n * @param scene defines the hosting scene\r\n * @returns true if fog must be enabled\r\n */\r\n public static GetFogState(mesh: AbstractMesh, scene: Scene) {\r\n return scene.fogEnabled && mesh.applyFog && scene.fogMode !== Scene.FOGMODE_NONE;\r\n }\r\n\r\n /**\r\n * Helper used to prepare the list of defines associated with misc. values for shader compilation\r\n * @param mesh defines the current mesh\r\n * @param scene defines the current scene\r\n * @param useLogarithmicDepth defines if logarithmic depth has to be turned on\r\n * @param pointsCloud defines if point cloud rendering has to be turned on\r\n * @param fogEnabled defines if fog has to be turned on\r\n * @param alphaTest defines if alpha testing has to be turned on\r\n * @param defines defines the current list of defines\r\n * @param applyDecalAfterDetail Defines if the decal is applied after or before the detail\r\n */\r\n public static PrepareDefinesForMisc(\r\n mesh: AbstractMesh,\r\n scene: Scene,\r\n useLogarithmicDepth: boolean,\r\n pointsCloud: boolean,\r\n fogEnabled: boolean,\r\n alphaTest: boolean,\r\n defines: any,\r\n applyDecalAfterDetail: boolean = false\r\n ): void {\r\n if (defines._areMiscDirty) {\r\n defines[\"LOGARITHMICDEPTH\"] = useLogarithmicDepth;\r\n defines[\"POINTSIZE\"] = pointsCloud;\r\n defines[\"FOG\"] = fogEnabled && this.GetFogState(mesh, scene);\r\n defines[\"NONUNIFORMSCALING\"] = mesh.nonUniformScaling;\r\n defines[\"ALPHATEST\"] = alphaTest;\r\n defines[\"DECAL_AFTER_DETAIL\"] = applyDecalAfterDetail;\r\n }\r\n }\r\n\r\n /**\r\n * Helper used to prepare the defines relative to the active camera\r\n * @param scene defines the current scene\r\n * @param defines specifies the list of active defines\r\n * @returns true if the defines have been updated, else false\r\n */\r\n public static PrepareDefinesForCamera(scene: Scene, defines: any): boolean {\r\n let changed = false;\r\n\r\n if (scene.activeCamera) {\r\n const wasOrtho = defines[\"CAMERA_ORTHOGRAPHIC\"] ? 1 : 0;\r\n const wasPersp = defines[\"CAMERA_PERSPECTIVE\"] ? 1 : 0;\r\n const isOrtho = scene.activeCamera.mode === Camera.ORTHOGRAPHIC_CAMERA ? 1 : 0;\r\n const isPersp = scene.activeCamera.mode === Camera.PERSPECTIVE_CAMERA ? 1 : 0;\r\n\r\n if (wasOrtho ^ isOrtho || wasPersp ^ isPersp) {\r\n defines[\"CAMERA_ORTHOGRAPHIC\"] = isOrtho === 1;\r\n defines[\"CAMERA_PERSPECTIVE\"] = isPersp === 1;\r\n changed = true;\r\n }\r\n }\r\n\r\n return changed;\r\n }\r\n\r\n /**\r\n * Helper used to prepare the list of defines associated with frame values for shader compilation\r\n * @param scene defines the current scene\r\n * @param engine defines the current engine\r\n * @param material defines the material we are compiling the shader for\r\n * @param defines specifies the list of active defines\r\n * @param useInstances defines if instances have to be turned on\r\n * @param useClipPlane defines if clip plane have to be turned on\r\n * @param useThinInstances defines if thin instances have to be turned on\r\n */\r\n public static PrepareDefinesForFrameBoundValues(\r\n scene: Scene,\r\n engine: Engine,\r\n material: Material,\r\n defines: any,\r\n useInstances: boolean,\r\n useClipPlane: Nullable = null,\r\n useThinInstances: boolean = false\r\n ): void {\r\n let changed = MaterialHelper.PrepareDefinesForCamera(scene, defines);\r\n\r\n if (useClipPlane !== false) {\r\n changed = prepareDefinesForClipPlanes(material, scene, defines);\r\n }\r\n\r\n if (defines[\"DEPTHPREPASS\"] !== !engine.getColorWrite()) {\r\n defines[\"DEPTHPREPASS\"] = !defines[\"DEPTHPREPASS\"];\r\n changed = true;\r\n }\r\n\r\n if (defines[\"INSTANCES\"] !== useInstances) {\r\n defines[\"INSTANCES\"] = useInstances;\r\n changed = true;\r\n }\r\n\r\n if (defines[\"THIN_INSTANCES\"] !== useThinInstances) {\r\n defines[\"THIN_INSTANCES\"] = useThinInstances;\r\n changed = true;\r\n }\r\n\r\n if (changed) {\r\n defines.markAsUnprocessed();\r\n }\r\n }\r\n\r\n /**\r\n * Prepares the defines for bones\r\n * @param mesh The mesh containing the geometry data we will draw\r\n * @param defines The defines to update\r\n */\r\n public static PrepareDefinesForBones(mesh: AbstractMesh, defines: any) {\r\n if (mesh.useBones && mesh.computeBonesUsingShaders && mesh.skeleton) {\r\n defines[\"NUM_BONE_INFLUENCERS\"] = mesh.numBoneInfluencers;\r\n\r\n const materialSupportsBoneTexture = defines[\"BONETEXTURE\"] !== undefined;\r\n\r\n if (mesh.skeleton.isUsingTextureForMatrices && materialSupportsBoneTexture) {\r\n defines[\"BONETEXTURE\"] = true;\r\n } else {\r\n defines[\"BonesPerMesh\"] = mesh.skeleton.bones.length + 1;\r\n defines[\"BONETEXTURE\"] = materialSupportsBoneTexture ? false : undefined;\r\n\r\n const prePassRenderer = mesh.getScene().prePassRenderer;\r\n if (prePassRenderer && prePassRenderer.enabled) {\r\n const nonExcluded = prePassRenderer.excludedSkinnedMesh.indexOf(mesh) === -1;\r\n defines[\"BONES_VELOCITY_ENABLED\"] = nonExcluded;\r\n }\r\n }\r\n } else {\r\n defines[\"NUM_BONE_INFLUENCERS\"] = 0;\r\n defines[\"BonesPerMesh\"] = 0;\r\n if (defines[\"BONETEXTURE\"] !== undefined) {\r\n defines[\"BONETEXTURE\"] = false;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Prepares the defines for morph targets\r\n * @param mesh The mesh containing the geometry data we will draw\r\n * @param defines The defines to update\r\n */\r\n public static PrepareDefinesForMorphTargets(mesh: AbstractMesh, defines: any) {\r\n const manager = (mesh).morphTargetManager;\r\n if (manager) {\r\n defines[\"MORPHTARGETS_UV\"] = manager.supportsUVs && defines[\"UV1\"];\r\n defines[\"MORPHTARGETS_TANGENT\"] = manager.supportsTangents && defines[\"TANGENT\"];\r\n defines[\"MORPHTARGETS_NORMAL\"] = manager.supportsNormals && defines[\"NORMAL\"];\r\n defines[\"MORPHTARGETS\"] = manager.numInfluencers > 0;\r\n defines[\"NUM_MORPH_INFLUENCERS\"] = manager.numInfluencers;\r\n\r\n defines[\"MORPHTARGETS_TEXTURE\"] = manager.isUsingTextureForTargets;\r\n } else {\r\n defines[\"MORPHTARGETS_UV\"] = false;\r\n defines[\"MORPHTARGETS_TANGENT\"] = false;\r\n defines[\"MORPHTARGETS_NORMAL\"] = false;\r\n defines[\"MORPHTARGETS\"] = false;\r\n defines[\"NUM_MORPH_INFLUENCERS\"] = 0;\r\n }\r\n }\r\n\r\n /**\r\n * Prepares the defines for baked vertex animation\r\n * @param mesh The mesh containing the geometry data we will draw\r\n * @param defines The defines to update\r\n */\r\n public static PrepareDefinesForBakedVertexAnimation(mesh: AbstractMesh, defines: any) {\r\n const manager = (mesh).bakedVertexAnimationManager;\r\n defines[\"BAKED_VERTEX_ANIMATION_TEXTURE\"] = manager && manager.isEnabled ? true : false;\r\n }\r\n\r\n /**\r\n * Prepares the defines used in the shader depending on the attributes data available in the mesh\r\n * @param mesh The mesh containing the geometry data we will draw\r\n * @param defines The defines to update\r\n * @param useVertexColor Precise whether vertex colors should be used or not (override mesh info)\r\n * @param useBones Precise whether bones should be used or not (override mesh info)\r\n * @param useMorphTargets Precise whether morph targets should be used or not (override mesh info)\r\n * @param useVertexAlpha Precise whether vertex alpha should be used or not (override mesh info)\r\n * @param useBakedVertexAnimation Precise whether baked vertex animation should be used or not (override mesh info)\r\n * @returns false if defines are considered not dirty and have not been checked\r\n */\r\n public static PrepareDefinesForAttributes(\r\n mesh: AbstractMesh,\r\n defines: any,\r\n useVertexColor: boolean,\r\n useBones: boolean,\r\n useMorphTargets = false,\r\n useVertexAlpha = true,\r\n useBakedVertexAnimation = true\r\n ): boolean {\r\n if (!defines._areAttributesDirty && defines._needNormals === defines._normals && defines._needUVs === defines._uvs) {\r\n return false;\r\n }\r\n\r\n defines._normals = defines._needNormals;\r\n defines._uvs = defines._needUVs;\r\n\r\n defines[\"NORMAL\"] = defines._needNormals && mesh.isVerticesDataPresent(VertexBuffer.NormalKind);\r\n\r\n if (defines._needNormals && mesh.isVerticesDataPresent(VertexBuffer.TangentKind)) {\r\n defines[\"TANGENT\"] = true;\r\n }\r\n\r\n for (let i = 1; i <= Constants.MAX_SUPPORTED_UV_SETS; ++i) {\r\n defines[\"UV\" + i] = defines._needUVs ? mesh.isVerticesDataPresent(`uv${i === 1 ? \"\" : i}`) : false;\r\n }\r\n\r\n if (useVertexColor) {\r\n const hasVertexColors = mesh.useVertexColors && mesh.isVerticesDataPresent(VertexBuffer.ColorKind);\r\n defines[\"VERTEXCOLOR\"] = hasVertexColors;\r\n defines[\"VERTEXALPHA\"] = mesh.hasVertexAlpha && hasVertexColors && useVertexAlpha;\r\n }\r\n\r\n if (mesh.isVerticesDataPresent(VertexBuffer.ColorInstanceKind) && (mesh.hasInstances || mesh.hasThinInstances)) {\r\n defines[\"INSTANCESCOLOR\"] = true;\r\n }\r\n\r\n if (useBones) {\r\n this.PrepareDefinesForBones(mesh, defines);\r\n }\r\n\r\n if (useMorphTargets) {\r\n this.PrepareDefinesForMorphTargets(mesh, defines);\r\n }\r\n\r\n if (useBakedVertexAnimation) {\r\n this.PrepareDefinesForBakedVertexAnimation(mesh, defines);\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Prepares the defines related to multiview\r\n * @param scene The scene we are intending to draw\r\n * @param defines The defines to update\r\n */\r\n public static PrepareDefinesForMultiview(scene: Scene, defines: any) {\r\n if (scene.activeCamera) {\r\n const previousMultiview = defines.MULTIVIEW;\r\n defines.MULTIVIEW = scene.activeCamera.outputRenderTarget !== null && scene.activeCamera.outputRenderTarget.getViewCount() > 1;\r\n if (defines.MULTIVIEW != previousMultiview) {\r\n defines.markAsUnprocessed();\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Prepares the defines related to order independant transparency\r\n * @param scene The scene we are intending to draw\r\n * @param defines The defines to update\r\n * @param needAlphaBlending Determines if the material needs alpha blending\r\n */\r\n public static PrepareDefinesForOIT(scene: Scene, defines: any, needAlphaBlending: boolean) {\r\n const previousDefine = defines.ORDER_INDEPENDENT_TRANSPARENCY;\r\n const previousDefine16Bits = defines.ORDER_INDEPENDENT_TRANSPARENCY_16BITS;\r\n\r\n defines.ORDER_INDEPENDENT_TRANSPARENCY = scene.useOrderIndependentTransparency && needAlphaBlending;\r\n defines.ORDER_INDEPENDENT_TRANSPARENCY_16BITS = !scene.getEngine().getCaps().textureFloatLinearFiltering;\r\n\r\n if (previousDefine !== defines.ORDER_INDEPENDENT_TRANSPARENCY || previousDefine16Bits !== defines.ORDER_INDEPENDENT_TRANSPARENCY_16BITS) {\r\n defines.markAsUnprocessed();\r\n }\r\n }\r\n\r\n /**\r\n * Prepares the defines related to the prepass\r\n * @param scene The scene we are intending to draw\r\n * @param defines The defines to update\r\n * @param canRenderToMRT Indicates if this material renders to several textures in the prepass\r\n */\r\n public static PrepareDefinesForPrePass(scene: Scene, defines: any, canRenderToMRT: boolean) {\r\n const previousPrePass = defines.PREPASS;\r\n\r\n if (!defines._arePrePassDirty) {\r\n return;\r\n }\r\n\r\n const texturesList = [\r\n {\r\n type: Constants.PREPASS_POSITION_TEXTURE_TYPE,\r\n define: \"PREPASS_POSITION\",\r\n index: \"PREPASS_POSITION_INDEX\",\r\n },\r\n {\r\n type: Constants.PREPASS_VELOCITY_TEXTURE_TYPE,\r\n define: \"PREPASS_VELOCITY\",\r\n index: \"PREPASS_VELOCITY_INDEX\",\r\n },\r\n {\r\n type: Constants.PREPASS_REFLECTIVITY_TEXTURE_TYPE,\r\n define: \"PREPASS_REFLECTIVITY\",\r\n index: \"PREPASS_REFLECTIVITY_INDEX\",\r\n },\r\n {\r\n type: Constants.PREPASS_IRRADIANCE_TEXTURE_TYPE,\r\n define: \"PREPASS_IRRADIANCE\",\r\n index: \"PREPASS_IRRADIANCE_INDEX\",\r\n },\r\n {\r\n type: Constants.PREPASS_ALBEDO_SQRT_TEXTURE_TYPE,\r\n define: \"PREPASS_ALBEDO_SQRT\",\r\n index: \"PREPASS_ALBEDO_SQRT_INDEX\",\r\n },\r\n {\r\n type: Constants.PREPASS_DEPTH_TEXTURE_TYPE,\r\n define: \"PREPASS_DEPTH\",\r\n index: \"PREPASS_DEPTH_INDEX\",\r\n },\r\n {\r\n type: Constants.PREPASS_NORMAL_TEXTURE_TYPE,\r\n define: \"PREPASS_NORMAL\",\r\n index: \"PREPASS_NORMAL_INDEX\",\r\n },\r\n ];\r\n\r\n if (scene.prePassRenderer && scene.prePassRenderer.enabled && canRenderToMRT) {\r\n defines.PREPASS = true;\r\n defines.SCENE_MRT_COUNT = scene.prePassRenderer.mrtCount;\r\n\r\n for (let i = 0; i < texturesList.length; i++) {\r\n const index = scene.prePassRenderer.getIndex(texturesList[i].type);\r\n if (index !== -1) {\r\n defines[texturesList[i].define] = true;\r\n defines[texturesList[i].index] = index;\r\n } else {\r\n defines[texturesList[i].define] = false;\r\n }\r\n }\r\n } else {\r\n defines.PREPASS = false;\r\n for (let i = 0; i < texturesList.length; i++) {\r\n defines[texturesList[i].define] = false;\r\n }\r\n }\r\n\r\n if (defines.PREPASS != previousPrePass) {\r\n defines.markAsUnprocessed();\r\n defines.markAsImageProcessingDirty();\r\n }\r\n }\r\n\r\n /**\r\n * Prepares the defines related to the light information passed in parameter\r\n * @param scene The scene we are intending to draw\r\n * @param mesh The mesh the effect is compiling for\r\n * @param light The light the effect is compiling for\r\n * @param lightIndex The index of the light\r\n * @param defines The defines to update\r\n * @param specularSupported Specifies whether specular is supported or not (override lights data)\r\n * @param state Defines the current state regarding what is needed (normals, etc...)\r\n * @param state.needNormals\r\n * @param state.needRebuild\r\n * @param state.shadowEnabled\r\n * @param state.specularEnabled\r\n * @param state.lightmapMode\r\n */\r\n public static PrepareDefinesForLight(\r\n scene: Scene,\r\n mesh: AbstractMesh,\r\n light: Light,\r\n lightIndex: number,\r\n defines: any,\r\n specularSupported: boolean,\r\n state: {\r\n needNormals: boolean;\r\n needRebuild: boolean;\r\n shadowEnabled: boolean;\r\n specularEnabled: boolean;\r\n lightmapMode: boolean;\r\n }\r\n ) {\r\n state.needNormals = true;\r\n\r\n if (defines[\"LIGHT\" + lightIndex] === undefined) {\r\n state.needRebuild = true;\r\n }\r\n\r\n defines[\"LIGHT\" + lightIndex] = true;\r\n\r\n defines[\"SPOTLIGHT\" + lightIndex] = false;\r\n defines[\"HEMILIGHT\" + lightIndex] = false;\r\n defines[\"POINTLIGHT\" + lightIndex] = false;\r\n defines[\"DIRLIGHT\" + lightIndex] = false;\r\n\r\n light.prepareLightSpecificDefines(defines, lightIndex);\r\n\r\n // FallOff.\r\n defines[\"LIGHT_FALLOFF_PHYSICAL\" + lightIndex] = false;\r\n defines[\"LIGHT_FALLOFF_GLTF\" + lightIndex] = false;\r\n defines[\"LIGHT_FALLOFF_STANDARD\" + lightIndex] = false;\r\n\r\n switch (light.falloffType) {\r\n case LightConstants.FALLOFF_GLTF:\r\n defines[\"LIGHT_FALLOFF_GLTF\" + lightIndex] = true;\r\n break;\r\n case LightConstants.FALLOFF_PHYSICAL:\r\n defines[\"LIGHT_FALLOFF_PHYSICAL\" + lightIndex] = true;\r\n break;\r\n case LightConstants.FALLOFF_STANDARD:\r\n defines[\"LIGHT_FALLOFF_STANDARD\" + lightIndex] = true;\r\n break;\r\n }\r\n\r\n // Specular\r\n if (specularSupported && !light.specular.equalsFloats(0, 0, 0)) {\r\n state.specularEnabled = true;\r\n }\r\n\r\n // Shadows\r\n defines[\"SHADOW\" + lightIndex] = false;\r\n defines[\"SHADOWCSM\" + lightIndex] = false;\r\n defines[\"SHADOWCSMDEBUG\" + lightIndex] = false;\r\n defines[\"SHADOWCSMNUM_CASCADES\" + lightIndex] = false;\r\n defines[\"SHADOWCSMUSESHADOWMAXZ\" + lightIndex] = false;\r\n defines[\"SHADOWCSMNOBLEND\" + lightIndex] = false;\r\n defines[\"SHADOWCSM_RIGHTHANDED\" + lightIndex] = false;\r\n defines[\"SHADOWPCF\" + lightIndex] = false;\r\n defines[\"SHADOWPCSS\" + lightIndex] = false;\r\n defines[\"SHADOWPOISSON\" + lightIndex] = false;\r\n defines[\"SHADOWESM\" + lightIndex] = false;\r\n defines[\"SHADOWCLOSEESM\" + lightIndex] = false;\r\n defines[\"SHADOWCUBE\" + lightIndex] = false;\r\n defines[\"SHADOWLOWQUALITY\" + lightIndex] = false;\r\n defines[\"SHADOWMEDIUMQUALITY\" + lightIndex] = false;\r\n\r\n if (mesh && mesh.receiveShadows && scene.shadowsEnabled && light.shadowEnabled) {\r\n const shadowGenerator = light.getShadowGenerator(scene.activeCamera) ?? light.getShadowGenerator();\r\n if (shadowGenerator) {\r\n const shadowMap = shadowGenerator.getShadowMap();\r\n if (shadowMap) {\r\n if (shadowMap.renderList && shadowMap.renderList.length > 0) {\r\n state.shadowEnabled = true;\r\n shadowGenerator.prepareDefines(defines, lightIndex);\r\n }\r\n }\r\n }\r\n }\r\n\r\n if (light.lightmapMode != LightConstants.LIGHTMAP_DEFAULT) {\r\n state.lightmapMode = true;\r\n defines[\"LIGHTMAPEXCLUDED\" + lightIndex] = true;\r\n defines[\"LIGHTMAPNOSPECULAR\" + lightIndex] = light.lightmapMode == LightConstants.LIGHTMAP_SHADOWSONLY;\r\n } else {\r\n defines[\"LIGHTMAPEXCLUDED\" + lightIndex] = false;\r\n defines[\"LIGHTMAPNOSPECULAR\" + lightIndex] = false;\r\n }\r\n }\r\n\r\n /**\r\n * Prepares the defines related to the light information passed in parameter\r\n * @param scene The scene we are intending to draw\r\n * @param mesh The mesh the effect is compiling for\r\n * @param defines The defines to update\r\n * @param specularSupported Specifies whether specular is supported or not (override lights data)\r\n * @param maxSimultaneousLights Specifies how manuy lights can be added to the effect at max\r\n * @param disableLighting Specifies whether the lighting is disabled (override scene and light)\r\n * @returns true if normals will be required for the rest of the effect\r\n */\r\n public static PrepareDefinesForLights(scene: Scene, mesh: AbstractMesh, defines: any, specularSupported: boolean, maxSimultaneousLights = 4, disableLighting = false): boolean {\r\n if (!defines._areLightsDirty) {\r\n return defines._needNormals;\r\n }\r\n\r\n let lightIndex = 0;\r\n const state = {\r\n needNormals: defines._needNormals, // prevents overriding previous reflection or other needs for normals\r\n needRebuild: false,\r\n lightmapMode: false,\r\n shadowEnabled: false,\r\n specularEnabled: false,\r\n };\r\n\r\n if (scene.lightsEnabled && !disableLighting) {\r\n for (const light of mesh.lightSources) {\r\n this.PrepareDefinesForLight(scene, mesh, light, lightIndex, defines, specularSupported, state);\r\n\r\n lightIndex++;\r\n if (lightIndex === maxSimultaneousLights) {\r\n break;\r\n }\r\n }\r\n }\r\n\r\n defines[\"SPECULARTERM\"] = state.specularEnabled;\r\n defines[\"SHADOWS\"] = state.shadowEnabled;\r\n\r\n // Resetting all other lights if any\r\n for (let index = lightIndex; index < maxSimultaneousLights; index++) {\r\n if (defines[\"LIGHT\" + index] !== undefined) {\r\n defines[\"LIGHT\" + index] = false;\r\n defines[\"HEMILIGHT\" + index] = false;\r\n defines[\"POINTLIGHT\" + index] = false;\r\n defines[\"DIRLIGHT\" + index] = false;\r\n defines[\"SPOTLIGHT\" + index] = false;\r\n defines[\"SHADOW\" + index] = false;\r\n defines[\"SHADOWCSM\" + index] = false;\r\n defines[\"SHADOWCSMDEBUG\" + index] = false;\r\n defines[\"SHADOWCSMNUM_CASCADES\" + index] = false;\r\n defines[\"SHADOWCSMUSESHADOWMAXZ\" + index] = false;\r\n defines[\"SHADOWCSMNOBLEND\" + index] = false;\r\n defines[\"SHADOWCSM_RIGHTHANDED\" + index] = false;\r\n defines[\"SHADOWPCF\" + index] = false;\r\n defines[\"SHADOWPCSS\" + index] = false;\r\n defines[\"SHADOWPOISSON\" + index] = false;\r\n defines[\"SHADOWESM\" + index] = false;\r\n defines[\"SHADOWCLOSEESM\" + index] = false;\r\n defines[\"SHADOWCUBE\" + index] = false;\r\n defines[\"SHADOWLOWQUALITY\" + index] = false;\r\n defines[\"SHADOWMEDIUMQUALITY\" + index] = false;\r\n }\r\n }\r\n\r\n const caps = scene.getEngine().getCaps();\r\n\r\n if (defines[\"SHADOWFLOAT\"] === undefined) {\r\n state.needRebuild = true;\r\n }\r\n\r\n defines[\"SHADOWFLOAT\"] =\r\n state.shadowEnabled && ((caps.textureFloatRender && caps.textureFloatLinearFiltering) || (caps.textureHalfFloatRender && caps.textureHalfFloatLinearFiltering));\r\n defines[\"LIGHTMAPEXCLUDED\"] = state.lightmapMode;\r\n\r\n if (state.needRebuild) {\r\n defines.rebuild();\r\n }\r\n\r\n return state.needNormals;\r\n }\r\n\r\n /**\r\n * Prepares the uniforms and samplers list to be used in the effect (for a specific light)\r\n * @param lightIndex defines the light index\r\n * @param uniformsList The uniform list\r\n * @param samplersList The sampler list\r\n * @param projectedLightTexture defines if projected texture must be used\r\n * @param uniformBuffersList defines an optional list of uniform buffers\r\n * @param updateOnlyBuffersList True to only update the uniformBuffersList array\r\n */\r\n public static PrepareUniformsAndSamplersForLight(\r\n lightIndex: number,\r\n uniformsList: string[],\r\n samplersList: string[],\r\n projectedLightTexture?: any,\r\n uniformBuffersList: Nullable = null,\r\n updateOnlyBuffersList = false\r\n ) {\r\n if (uniformBuffersList) {\r\n uniformBuffersList.push(\"Light\" + lightIndex);\r\n }\r\n\r\n if (updateOnlyBuffersList) {\r\n return;\r\n }\r\n\r\n uniformsList.push(\r\n \"vLightData\" + lightIndex,\r\n \"vLightDiffuse\" + lightIndex,\r\n \"vLightSpecular\" + lightIndex,\r\n \"vLightDirection\" + lightIndex,\r\n \"vLightFalloff\" + lightIndex,\r\n \"vLightGround\" + lightIndex,\r\n \"lightMatrix\" + lightIndex,\r\n \"shadowsInfo\" + lightIndex,\r\n \"depthValues\" + lightIndex\r\n );\r\n\r\n samplersList.push(\"shadowSampler\" + lightIndex);\r\n samplersList.push(\"depthSampler\" + lightIndex);\r\n\r\n uniformsList.push(\r\n \"viewFrustumZ\" + lightIndex,\r\n \"cascadeBlendFactor\" + lightIndex,\r\n \"lightSizeUVCorrection\" + lightIndex,\r\n \"depthCorrection\" + lightIndex,\r\n \"penumbraDarkness\" + lightIndex,\r\n \"frustumLengths\" + lightIndex\r\n );\r\n\r\n if (projectedLightTexture) {\r\n samplersList.push(\"projectionLightSampler\" + lightIndex);\r\n uniformsList.push(\"textureProjectionMatrix\" + lightIndex);\r\n }\r\n }\r\n\r\n /**\r\n * Prepares the uniforms and samplers list to be used in the effect\r\n * @param uniformsListOrOptions The uniform names to prepare or an EffectCreationOptions containing the list and extra information\r\n * @param samplersList The sampler list\r\n * @param defines The defines helping in the list generation\r\n * @param maxSimultaneousLights The maximum number of simultaneous light allowed in the effect\r\n */\r\n public static PrepareUniformsAndSamplersList(\r\n uniformsListOrOptions: string[] | IEffectCreationOptions,\r\n samplersList?: string[],\r\n defines?: any,\r\n maxSimultaneousLights = 4\r\n ): void {\r\n let uniformsList: string[];\r\n let uniformBuffersList: Nullable = null;\r\n\r\n if ((uniformsListOrOptions).uniformsNames) {\r\n const options = uniformsListOrOptions;\r\n uniformsList = options.uniformsNames;\r\n uniformBuffersList = options.uniformBuffersNames;\r\n samplersList = options.samplers;\r\n defines = options.defines;\r\n maxSimultaneousLights = options.maxSimultaneousLights || 0;\r\n } else {\r\n uniformsList = uniformsListOrOptions;\r\n if (!samplersList) {\r\n samplersList = [];\r\n }\r\n }\r\n\r\n for (let lightIndex = 0; lightIndex < maxSimultaneousLights; lightIndex++) {\r\n if (!defines[\"LIGHT\" + lightIndex]) {\r\n break;\r\n }\r\n this.PrepareUniformsAndSamplersForLight(lightIndex, uniformsList, samplersList, defines[\"PROJECTEDLIGHTTEXTURE\" + lightIndex], uniformBuffersList);\r\n }\r\n\r\n if (defines[\"NUM_MORPH_INFLUENCERS\"]) {\r\n uniformsList.push(\"morphTargetInfluences\");\r\n }\r\n\r\n if (defines[\"BAKED_VERTEX_ANIMATION_TEXTURE\"]) {\r\n uniformsList.push(\"bakedVertexAnimationSettings\");\r\n uniformsList.push(\"bakedVertexAnimationTextureSizeInverted\");\r\n uniformsList.push(\"bakedVertexAnimationTime\");\r\n samplersList.push(\"bakedVertexAnimationTexture\");\r\n }\r\n }\r\n\r\n /**\r\n * This helps decreasing rank by rank the shadow quality (0 being the highest rank and quality)\r\n * @param defines The defines to update while falling back\r\n * @param fallbacks The authorized effect fallbacks\r\n * @param maxSimultaneousLights The maximum number of lights allowed\r\n * @param rank the current rank of the Effect\r\n * @returns The newly affected rank\r\n */\r\n public static HandleFallbacksForShadows(defines: any, fallbacks: EffectFallbacks, maxSimultaneousLights = 4, rank = 0): number {\r\n let lightFallbackRank = 0;\r\n for (let lightIndex = 0; lightIndex < maxSimultaneousLights; lightIndex++) {\r\n if (!defines[\"LIGHT\" + lightIndex]) {\r\n break;\r\n }\r\n\r\n if (lightIndex > 0) {\r\n lightFallbackRank = rank + lightIndex;\r\n fallbacks.addFallback(lightFallbackRank, \"LIGHT\" + lightIndex);\r\n }\r\n\r\n if (!defines[\"SHADOWS\"]) {\r\n if (defines[\"SHADOW\" + lightIndex]) {\r\n fallbacks.addFallback(rank, \"SHADOW\" + lightIndex);\r\n }\r\n\r\n if (defines[\"SHADOWPCF\" + lightIndex]) {\r\n fallbacks.addFallback(rank, \"SHADOWPCF\" + lightIndex);\r\n }\r\n\r\n if (defines[\"SHADOWPCSS\" + lightIndex]) {\r\n fallbacks.addFallback(rank, \"SHADOWPCSS\" + lightIndex);\r\n }\r\n\r\n if (defines[\"SHADOWPOISSON\" + lightIndex]) {\r\n fallbacks.addFallback(rank, \"SHADOWPOISSON\" + lightIndex);\r\n }\r\n\r\n if (defines[\"SHADOWESM\" + lightIndex]) {\r\n fallbacks.addFallback(rank, \"SHADOWESM\" + lightIndex);\r\n }\r\n\r\n if (defines[\"SHADOWCLOSEESM\" + lightIndex]) {\r\n fallbacks.addFallback(rank, \"SHADOWCLOSEESM\" + lightIndex);\r\n }\r\n }\r\n }\r\n return lightFallbackRank++;\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n private static _TmpMorphInfluencers = { NUM_MORPH_INFLUENCERS: 0 };\r\n /**\r\n * Prepares the list of attributes required for morph targets according to the effect defines.\r\n * @param attribs The current list of supported attribs\r\n * @param mesh The mesh to prepare the morph targets attributes for\r\n * @param influencers The number of influencers\r\n */\r\n public static PrepareAttributesForMorphTargetsInfluencers(attribs: string[], mesh: AbstractMesh, influencers: number): void {\r\n this._TmpMorphInfluencers.NUM_MORPH_INFLUENCERS = influencers;\r\n this.PrepareAttributesForMorphTargets(attribs, mesh, this._TmpMorphInfluencers);\r\n }\r\n\r\n /**\r\n * Prepares the list of attributes required for morph targets according to the effect defines.\r\n * @param attribs The current list of supported attribs\r\n * @param mesh The mesh to prepare the morph targets attributes for\r\n * @param defines The current Defines of the effect\r\n */\r\n public static PrepareAttributesForMorphTargets(attribs: string[], mesh: AbstractMesh, defines: any): void {\r\n const influencers = defines[\"NUM_MORPH_INFLUENCERS\"];\r\n\r\n if (influencers > 0 && EngineStore.LastCreatedEngine) {\r\n const maxAttributesCount = EngineStore.LastCreatedEngine.getCaps().maxVertexAttribs;\r\n const manager = (mesh).morphTargetManager;\r\n if (manager?.isUsingTextureForTargets) {\r\n return;\r\n }\r\n const normal = manager && manager.supportsNormals && defines[\"NORMAL\"];\r\n const tangent = manager && manager.supportsTangents && defines[\"TANGENT\"];\r\n const uv = manager && manager.supportsUVs && defines[\"UV1\"];\r\n for (let index = 0; index < influencers; index++) {\r\n attribs.push(VertexBuffer.PositionKind + index);\r\n\r\n if (normal) {\r\n attribs.push(VertexBuffer.NormalKind + index);\r\n }\r\n\r\n if (tangent) {\r\n attribs.push(VertexBuffer.TangentKind + index);\r\n }\r\n\r\n if (uv) {\r\n attribs.push(VertexBuffer.UVKind + \"_\" + index);\r\n }\r\n\r\n if (attribs.length > maxAttributesCount) {\r\n Logger.Error(\"Cannot add more vertex attributes for mesh \" + mesh.name);\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Prepares the list of attributes required for baked vertex animations according to the effect defines.\r\n * @param attribs The current list of supported attribs\r\n * @param mesh The mesh to prepare the morph targets attributes for\r\n * @param defines The current Defines of the effect\r\n */\r\n public static PrepareAttributesForBakedVertexAnimation(attribs: string[], mesh: AbstractMesh, defines: any): void {\r\n const enabled = defines[\"BAKED_VERTEX_ANIMATION_TEXTURE\"] && defines[\"INSTANCES\"];\r\n\r\n if (enabled) {\r\n attribs.push(\"bakedVertexAnimationSettingsInstanced\");\r\n }\r\n }\r\n\r\n /**\r\n * Prepares the list of attributes required for bones according to the effect defines.\r\n * @param attribs The current list of supported attribs\r\n * @param mesh The mesh to prepare the bones attributes for\r\n * @param defines The current Defines of the effect\r\n * @param fallbacks The current effect fallback strategy\r\n */\r\n public static PrepareAttributesForBones(attribs: string[], mesh: AbstractMesh, defines: any, fallbacks: EffectFallbacks): void {\r\n if (defines[\"NUM_BONE_INFLUENCERS\"] > 0) {\r\n fallbacks.addCPUSkinningFallback(0, mesh);\r\n\r\n attribs.push(VertexBuffer.MatricesIndicesKind);\r\n attribs.push(VertexBuffer.MatricesWeightsKind);\r\n if (defines[\"NUM_BONE_INFLUENCERS\"] > 4) {\r\n attribs.push(VertexBuffer.MatricesIndicesExtraKind);\r\n attribs.push(VertexBuffer.MatricesWeightsExtraKind);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Check and prepare the list of attributes required for instances according to the effect defines.\r\n * @param attribs The current list of supported attribs\r\n * @param defines The current MaterialDefines of the effect\r\n */\r\n public static PrepareAttributesForInstances(attribs: string[], defines: MaterialDefines): void {\r\n if (defines[\"INSTANCES\"] || defines[\"THIN_INSTANCES\"]) {\r\n this.PushAttributesForInstances(attribs, !!defines[\"PREPASS_VELOCITY\"]);\r\n }\r\n\r\n if (defines.INSTANCESCOLOR) {\r\n attribs.push(VertexBuffer.ColorInstanceKind);\r\n }\r\n }\r\n\r\n /**\r\n * Add the list of attributes required for instances to the attribs array.\r\n * @param attribs The current list of supported attribs\r\n * @param needsPreviousMatrices If the shader needs previous matrices\r\n */\r\n public static PushAttributesForInstances(attribs: string[], needsPreviousMatrices: boolean = false): void {\r\n attribs.push(\"world0\");\r\n attribs.push(\"world1\");\r\n attribs.push(\"world2\");\r\n attribs.push(\"world3\");\r\n if (needsPreviousMatrices) {\r\n attribs.push(\"previousWorld0\");\r\n attribs.push(\"previousWorld1\");\r\n attribs.push(\"previousWorld2\");\r\n attribs.push(\"previousWorld3\");\r\n }\r\n }\r\n\r\n /**\r\n * Binds the light information to the effect.\r\n * @param light The light containing the generator\r\n * @param effect The effect we are binding the data to\r\n * @param lightIndex The light index in the effect used to render\r\n */\r\n public static BindLightProperties(light: Light, effect: Effect, lightIndex: number): void {\r\n light.transferToEffect(effect, lightIndex + \"\");\r\n }\r\n\r\n /**\r\n * Binds the lights information from the scene to the effect for the given mesh.\r\n * @param light Light to bind\r\n * @param lightIndex Light index\r\n * @param scene The scene where the light belongs to\r\n * @param effect The effect we are binding the data to\r\n * @param useSpecular Defines if specular is supported\r\n * @param receiveShadows Defines if the effect (mesh) we bind the light for receives shadows\r\n */\r\n public static BindLight(light: Light, lightIndex: number, scene: Scene, effect: Effect, useSpecular: boolean, receiveShadows = true): void {\r\n light._bindLight(lightIndex, scene, effect, useSpecular, receiveShadows);\r\n }\r\n\r\n /**\r\n * Binds the lights information from the scene to the effect for the given mesh.\r\n * @param scene The scene the lights belongs to\r\n * @param mesh The mesh we are binding the information to render\r\n * @param effect The effect we are binding the data to\r\n * @param defines The generated defines for the effect\r\n * @param maxSimultaneousLights The maximum number of light that can be bound to the effect\r\n */\r\n public static BindLights(scene: Scene, mesh: AbstractMesh, effect: Effect, defines: any, maxSimultaneousLights = 4): void {\r\n const len = Math.min(mesh.lightSources.length, maxSimultaneousLights);\r\n\r\n for (let i = 0; i < len; i++) {\r\n const light = mesh.lightSources[i];\r\n this.BindLight(light, i, scene, effect, typeof defines === \"boolean\" ? defines : defines[\"SPECULARTERM\"], mesh.receiveShadows);\r\n }\r\n }\r\n\r\n private static _TempFogColor = Color3.Black();\r\n /**\r\n * Binds the fog information from the scene to the effect for the given mesh.\r\n * @param scene The scene the lights belongs to\r\n * @param mesh The mesh we are binding the information to render\r\n * @param effect The effect we are binding the data to\r\n * @param linearSpace Defines if the fog effect is applied in linear space\r\n */\r\n public static BindFogParameters(scene: Scene, mesh: AbstractMesh, effect: Effect, linearSpace = false): void {\r\n if (scene.fogEnabled && mesh.applyFog && scene.fogMode !== Scene.FOGMODE_NONE) {\r\n effect.setFloat4(\"vFogInfos\", scene.fogMode, scene.fogStart, scene.fogEnd, scene.fogDensity);\r\n // Convert fog color to linear space if used in a linear space computed shader.\r\n if (linearSpace) {\r\n scene.fogColor.toLinearSpaceToRef(this._TempFogColor, scene.getEngine().useExactSrgbConversions);\r\n effect.setColor3(\"vFogColor\", this._TempFogColor);\r\n } else {\r\n effect.setColor3(\"vFogColor\", scene.fogColor);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Binds the bones information from the mesh to the effect.\r\n * @param mesh The mesh we are binding the information to render\r\n * @param effect The effect we are binding the data to\r\n * @param prePassConfiguration Configuration for the prepass, in case prepass is activated\r\n */\r\n public static BindBonesParameters(mesh?: AbstractMesh, effect?: Effect, prePassConfiguration?: PrePassConfiguration): void {\r\n if (!effect || !mesh) {\r\n return;\r\n }\r\n if (mesh.computeBonesUsingShaders && effect._bonesComputationForcedToCPU) {\r\n mesh.computeBonesUsingShaders = false;\r\n }\r\n\r\n if (mesh.useBones && mesh.computeBonesUsingShaders && mesh.skeleton) {\r\n const skeleton = mesh.skeleton;\r\n\r\n if (skeleton.isUsingTextureForMatrices && effect.getUniformIndex(\"boneTextureWidth\") > -1) {\r\n const boneTexture = skeleton.getTransformMatrixTexture(mesh);\r\n effect.setTexture(\"boneSampler\", boneTexture);\r\n effect.setFloat(\"boneTextureWidth\", 4.0 * (skeleton.bones.length + 1));\r\n } else {\r\n const matrices = skeleton.getTransformMatrices(mesh);\r\n\r\n if (matrices) {\r\n effect.setMatrices(\"mBones\", matrices);\r\n if (prePassConfiguration && mesh.getScene().prePassRenderer && mesh.getScene().prePassRenderer!.getIndex(Constants.PREPASS_VELOCITY_TEXTURE_TYPE)) {\r\n if (!prePassConfiguration.previousBones[mesh.uniqueId]) {\r\n prePassConfiguration.previousBones[mesh.uniqueId] = matrices.slice();\r\n }\r\n effect.setMatrices(\"mPreviousBones\", prePassConfiguration.previousBones[mesh.uniqueId]);\r\n MaterialHelper._CopyBonesTransformationMatrices(matrices, prePassConfiguration.previousBones[mesh.uniqueId]);\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Copies the bones transformation matrices into the target array and returns the target's reference\r\n private static _CopyBonesTransformationMatrices(source: Float32Array, target: Float32Array): Float32Array {\r\n target.set(source);\r\n\r\n return target;\r\n }\r\n\r\n /**\r\n * Binds the morph targets information from the mesh to the effect.\r\n * @param abstractMesh The mesh we are binding the information to render\r\n * @param effect The effect we are binding the data to\r\n */\r\n public static BindMorphTargetParameters(abstractMesh: AbstractMesh, effect: Effect): void {\r\n const manager = (abstractMesh).morphTargetManager;\r\n if (!abstractMesh || !manager) {\r\n return;\r\n }\r\n\r\n effect.setFloatArray(\"morphTargetInfluences\", manager.influences);\r\n }\r\n\r\n /**\r\n * Binds the logarithmic depth information from the scene to the effect for the given defines.\r\n * @param defines The generated defines used in the effect\r\n * @param effect The effect we are binding the data to\r\n * @param scene The scene we are willing to render with logarithmic scale for\r\n */\r\n public static BindLogDepth(defines: any, effect: Effect, scene: Scene): void {\r\n if (!defines || defines[\"LOGARITHMICDEPTH\"] || (defines.indexOf && defines.indexOf(\"LOGARITHMICDEPTH\") >= 0)) {\r\n const camera = scene.activeCamera;\r\n if (camera.mode === Camera.ORTHOGRAPHIC_CAMERA) {\r\n Logger.Error(\"Logarithmic depth is not compatible with orthographic cameras!\", 20);\r\n }\r\n effect.setFloat(\"logarithmicDepthConstant\", 2.0 / (Math.log(camera.maxZ + 1.0) / Math.LN2));\r\n }\r\n }\r\n}\r\n","import { Constants } from \"../Engines/constants\";\r\nimport { SerializationHelper, serialize } from \"../Misc/decorators\";\r\nimport type { IStencilState } from \"../States/IStencilState\";\r\n\r\nimport type { Scene } from \"../scene\";\r\n\r\n/**\r\n * Class that holds the different stencil states of a material\r\n * Usage example: https://playground.babylonjs.com/#CW5PRI#10\r\n */\r\nexport class MaterialStencilState implements IStencilState {\r\n /**\r\n * Creates a material stencil state instance\r\n */\r\n public constructor() {\r\n this.reset();\r\n }\r\n\r\n /**\r\n * Resets all the stencil states to default values\r\n */\r\n public reset() {\r\n this.enabled = false;\r\n this.mask = 0xff;\r\n\r\n this.func = Constants.ALWAYS;\r\n this.funcRef = 1;\r\n this.funcMask = 0xff;\r\n\r\n this.opStencilFail = Constants.KEEP;\r\n this.opDepthFail = Constants.KEEP;\r\n this.opStencilDepthPass = Constants.REPLACE;\r\n }\r\n\r\n private _func: number;\r\n /**\r\n * Gets or sets the stencil function\r\n */\r\n @serialize()\r\n public get func(): number {\r\n return this._func;\r\n }\r\n\r\n public set func(value: number) {\r\n this._func = value;\r\n }\r\n\r\n private _funcRef: number;\r\n /**\r\n * Gets or sets the stencil function reference\r\n */\r\n @serialize()\r\n public get funcRef(): number {\r\n return this._funcRef;\r\n }\r\n\r\n public set funcRef(value: number) {\r\n this._funcRef = value;\r\n }\r\n\r\n private _funcMask: number;\r\n /**\r\n * Gets or sets the stencil function mask\r\n */\r\n @serialize()\r\n public get funcMask(): number {\r\n return this._funcMask;\r\n }\r\n\r\n public set funcMask(value: number) {\r\n this._funcMask = value;\r\n }\r\n\r\n private _opStencilFail: number;\r\n /**\r\n * Gets or sets the operation when the stencil test fails\r\n */\r\n @serialize()\r\n public get opStencilFail(): number {\r\n return this._opStencilFail;\r\n }\r\n\r\n public set opStencilFail(value: number) {\r\n this._opStencilFail = value;\r\n }\r\n\r\n private _opDepthFail: number;\r\n /**\r\n * Gets or sets the operation when the depth test fails\r\n */\r\n @serialize()\r\n public get opDepthFail(): number {\r\n return this._opDepthFail;\r\n }\r\n\r\n public set opDepthFail(value: number) {\r\n this._opDepthFail = value;\r\n }\r\n\r\n private _opStencilDepthPass: number;\r\n /**\r\n * Gets or sets the operation when the stencil+depth test succeeds\r\n */\r\n @serialize()\r\n public get opStencilDepthPass(): number {\r\n return this._opStencilDepthPass;\r\n }\r\n\r\n public set opStencilDepthPass(value: number) {\r\n this._opStencilDepthPass = value;\r\n }\r\n\r\n private _mask: number;\r\n /**\r\n * Gets or sets the stencil mask\r\n */\r\n @serialize()\r\n public get mask(): number {\r\n return this._mask;\r\n }\r\n\r\n public set mask(value: number) {\r\n this._mask = value;\r\n }\r\n\r\n private _enabled: boolean;\r\n /**\r\n * Enables or disables the stencil test\r\n */\r\n @serialize()\r\n public get enabled(): boolean {\r\n return this._enabled;\r\n }\r\n\r\n public set enabled(value: boolean) {\r\n this._enabled = value;\r\n }\r\n\r\n /**\r\n * Get the current class name, useful for serialization or dynamic coding.\r\n * @returns \"MaterialStencilState\"\r\n */\r\n public getClassName(): string {\r\n return \"MaterialStencilState\";\r\n }\r\n\r\n /**\r\n * Makes a duplicate of the current configuration into another one.\r\n * @param stencilState defines stencil state where to copy the info\r\n */\r\n public copyTo(stencilState: MaterialStencilState): void {\r\n SerializationHelper.Clone(() => stencilState, this);\r\n }\r\n\r\n /**\r\n * Serializes this stencil configuration.\r\n * @returns - An object with the serialized config.\r\n */\r\n public serialize(): any {\r\n return SerializationHelper.Serialize(this);\r\n }\r\n\r\n /**\r\n * Parses a stencil state configuration from a serialized object.\r\n * @param source - Serialized object.\r\n * @param scene Defines the scene we are parsing for\r\n * @param rootUrl Defines the rootUrl to load from\r\n */\r\n public parse(source: any, scene: Scene, rootUrl: string): void {\r\n SerializationHelper.Parse(() => this, source, scene, rootUrl);\r\n }\r\n}\r\n","import type { ShaderCustomProcessingFunction } from \"../Engines/Processors/shaderProcessingOptions\";\r\nimport type { SmartArray } from \"../Misc/smartArray\";\r\n\r\nimport type { BaseTexture } from \"./Textures/baseTexture\";\r\nimport type { EffectFallbacks } from \"./effectFallbacks\";\r\nimport type { MaterialDefines } from \"./materialDefines\";\r\nimport type { UniformBuffer } from \"./uniformBuffer\";\r\nimport type { SubMesh } from \"../Meshes/subMesh\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { IAnimatable } from \"../Animations/animatable.interface\";\r\nimport type { RenderTargetTexture } from \"./Textures/renderTargetTexture\";\r\n\r\n/** @internal */\r\nexport type MaterialPluginCreated = {};\r\n\r\n/** @internal */\r\nexport type MaterialPluginDisposed = {\r\n forceDisposeTextures?: boolean;\r\n};\r\n\r\n/** @internal */\r\nexport type MaterialPluginHasTexture = {\r\n hasTexture: boolean;\r\n texture: BaseTexture;\r\n};\r\n\r\n/** @internal */\r\nexport type MaterialPluginIsReadyForSubMesh = {\r\n isReadyForSubMesh: boolean;\r\n defines: MaterialDefines;\r\n subMesh: SubMesh;\r\n};\r\n\r\n/** @internal */\r\nexport type MaterialPluginGetDefineNames = {\r\n defineNames?: { [name: string]: { type: string; default: any } };\r\n};\r\n\r\n/** @internal */\r\nexport type MaterialPluginPrepareEffect = {\r\n defines: MaterialDefines;\r\n fallbacks: EffectFallbacks;\r\n fallbackRank: number;\r\n customCode?: ShaderCustomProcessingFunction;\r\n attributes: string[];\r\n uniforms: string[];\r\n samplers: string[];\r\n uniformBuffersNames: string[];\r\n mesh: AbstractMesh;\r\n indexParameters: any;\r\n};\r\n\r\n/** @internal */\r\nexport type MaterialPluginPrepareDefines = {\r\n defines: MaterialDefines;\r\n mesh: AbstractMesh;\r\n};\r\n\r\n/** @internal */\r\nexport type MaterialPluginPrepareUniformBuffer = {\r\n ubo: UniformBuffer;\r\n};\r\n\r\n/** @internal */\r\nexport type MaterialPluginBindForSubMesh = {\r\n subMesh: SubMesh;\r\n};\r\n\r\n/** @internal */\r\nexport type MaterialPluginGetAnimatables = {\r\n animatables: IAnimatable[];\r\n};\r\n\r\n/** @internal */\r\nexport type MaterialPluginGetActiveTextures = {\r\n activeTextures: BaseTexture[];\r\n};\r\n\r\n/** @internal */\r\nexport type MaterialPluginFillRenderTargetTextures = {\r\n renderTargets: SmartArray;\r\n};\r\n\r\n/** @internal */\r\nexport type MaterialPluginHasRenderTargetTextures = {\r\n hasRenderTargetTextures: boolean;\r\n};\r\n\r\n/** @internal */\r\nexport type MaterialPluginHardBindForSubMesh = {\r\n subMesh: SubMesh;\r\n};\r\n\r\n/**\r\n * @internal\r\n */\r\nexport enum MaterialPluginEvent {\r\n Created = 0x0001,\r\n Disposed = 0x0002,\r\n GetDefineNames = 0x0004,\r\n PrepareUniformBuffer = 0x0008,\r\n IsReadyForSubMesh = 0x0010,\r\n PrepareDefines = 0x0020,\r\n BindForSubMesh = 0x0040,\r\n PrepareEffect = 0x0080,\r\n GetAnimatables = 0x0100,\r\n GetActiveTextures = 0x0200,\r\n HasTexture = 0x0400,\r\n FillRenderTargetTextures = 0x0800,\r\n HasRenderTargetTextures = 0x1000,\r\n HardBindForSubMesh = 0x2000,\r\n}\r\n","import type { DeepImmutable, Nullable } from \"../types\";\r\nimport { Scalar } from \"./math.scalar\";\r\nimport { Vector2, Vector3, Quaternion, Matrix } from \"./math.vector\";\r\nimport { Epsilon } from \"./math.constants\";\r\n\r\n/**\r\n * Defines potential orientation for back face culling\r\n */\r\nexport enum Orientation {\r\n /**\r\n * Clockwise\r\n */\r\n CW = 0,\r\n /** Counter clockwise */\r\n CCW = 1,\r\n}\r\n\r\n/** Class used to represent a Bezier curve */\r\nexport class BezierCurve {\r\n /**\r\n * Returns the cubic Bezier interpolated value (float) at \"t\" (float) from the given x1, y1, x2, y2 floats\r\n * @param t defines the time\r\n * @param x1 defines the left coordinate on X axis\r\n * @param y1 defines the left coordinate on Y axis\r\n * @param x2 defines the right coordinate on X axis\r\n * @param y2 defines the right coordinate on Y axis\r\n * @returns the interpolated value\r\n */\r\n public static Interpolate(t: number, x1: number, y1: number, x2: number, y2: number): number {\r\n // Extract X (which is equal to time here)\r\n const f0 = 1 - 3 * x2 + 3 * x1;\r\n const f1 = 3 * x2 - 6 * x1;\r\n const f2 = 3 * x1;\r\n\r\n let refinedT = t;\r\n for (let i = 0; i < 5; i++) {\r\n const refinedT2 = refinedT * refinedT;\r\n const refinedT3 = refinedT2 * refinedT;\r\n\r\n const x = f0 * refinedT3 + f1 * refinedT2 + f2 * refinedT;\r\n const slope = 1.0 / (3.0 * f0 * refinedT2 + 2.0 * f1 * refinedT + f2);\r\n refinedT -= (x - t) * slope;\r\n refinedT = Math.min(1, Math.max(0, refinedT));\r\n }\r\n\r\n // Resolve cubic bezier for the given x\r\n return 3 * Math.pow(1 - refinedT, 2) * refinedT * y1 + 3 * (1 - refinedT) * Math.pow(refinedT, 2) * y2 + Math.pow(refinedT, 3);\r\n }\r\n}\r\n\r\n/**\r\n * Defines angle representation\r\n */\r\nexport class Angle {\r\n private _radians: number;\r\n\r\n /**\r\n * Creates an Angle object of \"radians\" radians (float).\r\n * @param radians the angle in radians\r\n */\r\n constructor(radians: number) {\r\n this._radians = radians;\r\n if (this._radians < 0.0) {\r\n this._radians += 2.0 * Math.PI;\r\n }\r\n }\r\n\r\n /**\r\n * Get value in degrees\r\n * @returns the Angle value in degrees (float)\r\n */\r\n public degrees() {\r\n return (this._radians * 180.0) / Math.PI;\r\n }\r\n\r\n /**\r\n * Get value in radians\r\n * @returns the Angle value in radians (float)\r\n */\r\n public radians() {\r\n return this._radians;\r\n }\r\n\r\n /**\r\n * Gets a new Angle object valued with the gradient angle, in radians, of the line joining two points\r\n * @param a defines first point as the origin\r\n * @param b defines point\r\n * @returns a new Angle\r\n */\r\n public static BetweenTwoPoints(a: DeepImmutable, b: DeepImmutable): Angle {\r\n const delta = b.subtract(a);\r\n const theta = Math.atan2(delta.y, delta.x);\r\n return new Angle(theta);\r\n }\r\n\r\n /**\r\n * Gets a new Angle object from the given float in radians\r\n * @param radians defines the angle value in radians\r\n * @returns a new Angle\r\n */\r\n public static FromRadians(radians: number): Angle {\r\n return new Angle(radians);\r\n }\r\n /**\r\n * Gets a new Angle object from the given float in degrees\r\n * @param degrees defines the angle value in degrees\r\n * @returns a new Angle\r\n */\r\n public static FromDegrees(degrees: number): Angle {\r\n return new Angle((degrees * Math.PI) / 180.0);\r\n }\r\n}\r\n\r\n/**\r\n * This represents an arc in a 2d space.\r\n */\r\nexport class Arc2 {\r\n /**\r\n * Defines the center point of the arc.\r\n */\r\n public centerPoint: Vector2;\r\n /**\r\n * Defines the radius of the arc.\r\n */\r\n public radius: number;\r\n /**\r\n * Defines the angle of the arc (from mid point to end point).\r\n */\r\n public angle: Angle;\r\n /**\r\n * Defines the start angle of the arc (from start point to middle point).\r\n */\r\n public startAngle: Angle;\r\n /**\r\n * Defines the orientation of the arc (clock wise/counter clock wise).\r\n */\r\n public orientation: Orientation;\r\n\r\n /**\r\n * Creates an Arc object from the three given points : start, middle and end.\r\n * @param startPoint Defines the start point of the arc\r\n * @param midPoint Defines the middle point of the arc\r\n * @param endPoint Defines the end point of the arc\r\n */\r\n constructor(\r\n /** Defines the start point of the arc */\r\n public startPoint: Vector2,\r\n /** Defines the mid point of the arc */\r\n public midPoint: Vector2,\r\n /** Defines the end point of the arc */\r\n public endPoint: Vector2\r\n ) {\r\n const temp = Math.pow(midPoint.x, 2) + Math.pow(midPoint.y, 2);\r\n const startToMid = (Math.pow(startPoint.x, 2) + Math.pow(startPoint.y, 2) - temp) / 2;\r\n const midToEnd = (temp - Math.pow(endPoint.x, 2) - Math.pow(endPoint.y, 2)) / 2;\r\n const det = (startPoint.x - midPoint.x) * (midPoint.y - endPoint.y) - (midPoint.x - endPoint.x) * (startPoint.y - midPoint.y);\r\n\r\n this.centerPoint = new Vector2(\r\n (startToMid * (midPoint.y - endPoint.y) - midToEnd * (startPoint.y - midPoint.y)) / det,\r\n ((startPoint.x - midPoint.x) * midToEnd - (midPoint.x - endPoint.x) * startToMid) / det\r\n );\r\n\r\n this.radius = this.centerPoint.subtract(this.startPoint).length();\r\n\r\n this.startAngle = Angle.BetweenTwoPoints(this.centerPoint, this.startPoint);\r\n\r\n const a1 = this.startAngle.degrees();\r\n let a2 = Angle.BetweenTwoPoints(this.centerPoint, this.midPoint).degrees();\r\n let a3 = Angle.BetweenTwoPoints(this.centerPoint, this.endPoint).degrees();\r\n\r\n // angles correction\r\n if (a2 - a1 > +180.0) {\r\n a2 -= 360.0;\r\n }\r\n if (a2 - a1 < -180.0) {\r\n a2 += 360.0;\r\n }\r\n if (a3 - a2 > +180.0) {\r\n a3 -= 360.0;\r\n }\r\n if (a3 - a2 < -180.0) {\r\n a3 += 360.0;\r\n }\r\n\r\n this.orientation = a2 - a1 < 0 ? Orientation.CW : Orientation.CCW;\r\n this.angle = Angle.FromDegrees(this.orientation === Orientation.CW ? a1 - a3 : a3 - a1);\r\n }\r\n}\r\n\r\n/**\r\n * Represents a 2D path made up of multiple 2D points\r\n */\r\nexport class Path2 {\r\n private _points = new Array();\r\n private _length = 0.0;\r\n\r\n /**\r\n * If the path start and end point are the same\r\n */\r\n public closed = false;\r\n\r\n /**\r\n * Creates a Path2 object from the starting 2D coordinates x and y.\r\n * @param x the starting points x value\r\n * @param y the starting points y value\r\n */\r\n constructor(x: number, y: number) {\r\n this._points.push(new Vector2(x, y));\r\n }\r\n\r\n /**\r\n * Adds a new segment until the given coordinates (x, y) to the current Path2.\r\n * @param x the added points x value\r\n * @param y the added points y value\r\n * @returns the updated Path2.\r\n */\r\n public addLineTo(x: number, y: number): Path2 {\r\n if (this.closed) {\r\n return this;\r\n }\r\n const newPoint = new Vector2(x, y);\r\n const previousPoint = this._points[this._points.length - 1];\r\n this._points.push(newPoint);\r\n this._length += newPoint.subtract(previousPoint).length();\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds _numberOfSegments_ segments according to the arc definition (middle point coordinates, end point coordinates, the arc start point being the current Path2 last point) to the current Path2.\r\n * @param midX middle point x value\r\n * @param midY middle point y value\r\n * @param endX end point x value\r\n * @param endY end point y value\r\n * @param numberOfSegments (default: 36)\r\n * @returns the updated Path2.\r\n */\r\n public addArcTo(midX: number, midY: number, endX: number, endY: number, numberOfSegments = 36): Path2 {\r\n if (this.closed) {\r\n return this;\r\n }\r\n const startPoint = this._points[this._points.length - 1];\r\n const midPoint = new Vector2(midX, midY);\r\n const endPoint = new Vector2(endX, endY);\r\n\r\n const arc = new Arc2(startPoint, midPoint, endPoint);\r\n\r\n let increment = arc.angle.radians() / numberOfSegments;\r\n if (arc.orientation === Orientation.CW) {\r\n increment *= -1;\r\n }\r\n let currentAngle = arc.startAngle.radians() + increment;\r\n\r\n for (let i = 0; i < numberOfSegments; i++) {\r\n const x = Math.cos(currentAngle) * arc.radius + arc.centerPoint.x;\r\n const y = Math.sin(currentAngle) * arc.radius + arc.centerPoint.y;\r\n this.addLineTo(x, y);\r\n currentAngle += increment;\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds _numberOfSegments_ segments according to the quadratic curve definition to the current Path2.\r\n * @param controlX control point x value\r\n * @param controlY control point y value\r\n * @param endX end point x value\r\n * @param endY end point y value\r\n * @param numberOfSegments (default: 36)\r\n * @returns the updated Path2.\r\n */\r\n public addQuadraticCurveTo(controlX: number, controlY: number, endX: number, endY: number, numberOfSegments = 36): Path2 {\r\n if (this.closed) {\r\n return this;\r\n }\r\n\r\n const equation = (t: number, val0: number, val1: number, val2: number) => {\r\n const res = (1.0 - t) * (1.0 - t) * val0 + 2.0 * t * (1.0 - t) * val1 + t * t * val2;\r\n return res;\r\n };\r\n const startPoint = this._points[this._points.length - 1];\r\n for (let i = 0; i <= numberOfSegments; i++) {\r\n const step = i / numberOfSegments;\r\n const x = equation(step, startPoint.x, controlX, endX);\r\n const y = equation(step, startPoint.y, controlY, endY);\r\n this.addLineTo(x, y);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds _numberOfSegments_ segments according to the bezier curve definition to the current Path2.\r\n * @param originTangentX tangent vector at the origin point x value\r\n * @param originTangentY tangent vector at the origin point y value\r\n * @param destinationTangentX tangent vector at the destination point x value\r\n * @param destinationTangentY tangent vector at the destination point y value\r\n * @param endX end point x value\r\n * @param endY end point y value\r\n * @param numberOfSegments (default: 36)\r\n * @returns the updated Path2.\r\n */\r\n public addBezierCurveTo(\r\n originTangentX: number,\r\n originTangentY: number,\r\n destinationTangentX: number,\r\n destinationTangentY: number,\r\n endX: number,\r\n endY: number,\r\n numberOfSegments = 36\r\n ): Path2 {\r\n if (this.closed) {\r\n return this;\r\n }\r\n\r\n const equation = (t: number, val0: number, val1: number, val2: number, val3: number) => {\r\n const res = (1.0 - t) * (1.0 - t) * (1.0 - t) * val0 + 3.0 * t * (1.0 - t) * (1.0 - t) * val1 + 3.0 * t * t * (1.0 - t) * val2 + t * t * t * val3;\r\n return res;\r\n };\r\n const startPoint = this._points[this._points.length - 1];\r\n for (let i = 0; i <= numberOfSegments; i++) {\r\n const step = i / numberOfSegments;\r\n const x = equation(step, startPoint.x, originTangentX, destinationTangentX, endX);\r\n const y = equation(step, startPoint.y, originTangentY, destinationTangentY, endY);\r\n this.addLineTo(x, y);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Defines if a given point is inside the polygon defines by the path\r\n * @param point defines the point to test\r\n * @returns true if the point is inside\r\n */\r\n public isPointInside(point: Vector2) {\r\n let isInside = false;\r\n const count = this._points.length;\r\n for (let p = count - 1, q = 0; q < count; p = q++) {\r\n let edgeLow = this._points[p];\r\n let edgeHigh = this._points[q];\r\n\r\n let edgeDx = edgeHigh.x - edgeLow.x;\r\n let edgeDy = edgeHigh.y - edgeLow.y;\r\n\r\n if (Math.abs(edgeDy) > Number.EPSILON) {\r\n // Not parallel\r\n if (edgeDy < 0) {\r\n edgeLow = this._points[q];\r\n edgeDx = -edgeDx;\r\n edgeHigh = this._points[p];\r\n edgeDy = -edgeDy;\r\n }\r\n\r\n if (point.y < edgeLow.y || point.y > edgeHigh.y) {\r\n continue;\r\n }\r\n\r\n if (point.y === edgeLow.y && point.x === edgeLow.x) {\r\n return true;\r\n } else {\r\n const perpEdge = edgeDy * (point.x - edgeLow.x) - edgeDx * (point.y - edgeLow.y);\r\n if (perpEdge === 0) {\r\n return true;\r\n }\r\n if (perpEdge < 0) {\r\n continue;\r\n }\r\n isInside = !isInside;\r\n }\r\n } else {\r\n // parallel or collinear\r\n if (point.y !== edgeLow.y) {\r\n continue;\r\n }\r\n\r\n if ((edgeHigh.x <= point.x && point.x <= edgeLow.x) || (edgeLow.x <= point.x && point.x <= edgeHigh.x)) {\r\n return true;\r\n }\r\n }\r\n }\r\n\r\n return isInside;\r\n }\r\n\r\n /**\r\n * Closes the Path2.\r\n * @returns the Path2.\r\n */\r\n public close(): Path2 {\r\n this.closed = true;\r\n return this;\r\n }\r\n /**\r\n * Gets the sum of the distance between each sequential point in the path\r\n * @returns the Path2 total length (float).\r\n */\r\n public length(): number {\r\n let result = this._length;\r\n\r\n if (this.closed) {\r\n const lastPoint = this._points[this._points.length - 1];\r\n const firstPoint = this._points[0];\r\n result += firstPoint.subtract(lastPoint).length();\r\n }\r\n return result;\r\n }\r\n\r\n /**\r\n * Gets the area of the polygon defined by the path\r\n * @returns area value\r\n */\r\n public area(): number {\r\n const n = this._points.length;\r\n let value = 0.0;\r\n\r\n for (let p = n - 1, q = 0; q < n; p = q++) {\r\n value += this._points[p].x * this._points[q].y - this._points[q].x * this._points[p].y;\r\n }\r\n\r\n return value * 0.5;\r\n }\r\n\r\n /**\r\n * Gets the points which construct the path\r\n * @returns the Path2 internal array of points.\r\n */\r\n public getPoints(): Vector2[] {\r\n return this._points;\r\n }\r\n\r\n /**\r\n * Retreives the point at the distance aways from the starting point\r\n * @param normalizedLengthPosition the length along the path to retrieve the point from\r\n * @returns a new Vector2 located at a percentage of the Path2 total length on this path.\r\n */\r\n public getPointAtLengthPosition(normalizedLengthPosition: number): Vector2 {\r\n if (normalizedLengthPosition < 0 || normalizedLengthPosition > 1) {\r\n return Vector2.Zero();\r\n }\r\n\r\n const lengthPosition = normalizedLengthPosition * this.length();\r\n\r\n let previousOffset = 0;\r\n for (let i = 0; i < this._points.length; i++) {\r\n const j = (i + 1) % this._points.length;\r\n\r\n const a = this._points[i];\r\n const b = this._points[j];\r\n const bToA = b.subtract(a);\r\n\r\n const nextOffset = bToA.length() + previousOffset;\r\n if (lengthPosition >= previousOffset && lengthPosition <= nextOffset) {\r\n const dir = bToA.normalize();\r\n const localOffset = lengthPosition - previousOffset;\r\n\r\n return new Vector2(a.x + dir.x * localOffset, a.y + dir.y * localOffset);\r\n }\r\n previousOffset = nextOffset;\r\n }\r\n\r\n return Vector2.Zero();\r\n }\r\n\r\n /**\r\n * Creates a new path starting from an x and y position\r\n * @param x starting x value\r\n * @param y starting y value\r\n * @returns a new Path2 starting at the coordinates (x, y).\r\n */\r\n public static StartingAt(x: number, y: number): Path2 {\r\n return new Path2(x, y);\r\n }\r\n}\r\n\r\n/**\r\n * Represents a 3D path made up of multiple 3D points\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/path3D\r\n */\r\nexport class Path3D {\r\n private _curve = new Array();\r\n private _distances = new Array();\r\n private _tangents = new Array();\r\n private _normals = new Array();\r\n private _binormals = new Array();\r\n private _raw: boolean;\r\n private _alignTangentsWithPath: boolean;\r\n\r\n // holds interpolated point data\r\n private readonly _pointAtData = {\r\n id: 0,\r\n point: Vector3.Zero(),\r\n previousPointArrayIndex: 0,\r\n\r\n position: 0,\r\n subPosition: 0,\r\n\r\n interpolateReady: false,\r\n interpolationMatrix: Matrix.Identity(),\r\n };\r\n\r\n /**\r\n * new Path3D(path, normal, raw)\r\n * Creates a Path3D. A Path3D is a logical math object, so not a mesh.\r\n * please read the description in the tutorial : https://doc.babylonjs.com/features/featuresDeepDive/mesh/path3D\r\n * @param path an array of Vector3, the curve axis of the Path3D\r\n * @param firstNormal (options) Vector3, the first wanted normal to the curve. Ex (0, 1, 0) for a vertical normal.\r\n * @param raw (optional, default false) : boolean, if true the returned Path3D isn't normalized. Useful to depict path acceleration or speed.\r\n * @param alignTangentsWithPath (optional, default false) : boolean, if true the tangents will be aligned with the path.\r\n */\r\n constructor(\r\n /**\r\n * an array of Vector3, the curve axis of the Path3D\r\n */\r\n public path: Vector3[],\r\n firstNormal: Nullable = null,\r\n raw?: boolean,\r\n alignTangentsWithPath = false\r\n ) {\r\n for (let p = 0; p < path.length; p++) {\r\n this._curve[p] = path[p].clone(); // hard copy\r\n }\r\n this._raw = raw || false;\r\n this._alignTangentsWithPath = alignTangentsWithPath;\r\n this._compute(firstNormal, alignTangentsWithPath);\r\n }\r\n\r\n /**\r\n * Returns the Path3D array of successive Vector3 designing its curve.\r\n * @returns the Path3D array of successive Vector3 designing its curve.\r\n */\r\n public getCurve(): Vector3[] {\r\n return this._curve;\r\n }\r\n\r\n /**\r\n * Returns the Path3D array of successive Vector3 designing its curve.\r\n * @returns the Path3D array of successive Vector3 designing its curve.\r\n */\r\n public getPoints(): Vector3[] {\r\n return this._curve;\r\n }\r\n\r\n /**\r\n * @returns the computed length (float) of the path.\r\n */\r\n public length() {\r\n return this._distances[this._distances.length - 1];\r\n }\r\n\r\n /**\r\n * Returns an array populated with tangent vectors on each Path3D curve point.\r\n * @returns an array populated with tangent vectors on each Path3D curve point.\r\n */\r\n public getTangents(): Vector3[] {\r\n return this._tangents;\r\n }\r\n\r\n /**\r\n * Returns an array populated with normal vectors on each Path3D curve point.\r\n * @returns an array populated with normal vectors on each Path3D curve point.\r\n */\r\n public getNormals(): Vector3[] {\r\n return this._normals;\r\n }\r\n\r\n /**\r\n * Returns an array populated with binormal vectors on each Path3D curve point.\r\n * @returns an array populated with binormal vectors on each Path3D curve point.\r\n */\r\n public getBinormals(): Vector3[] {\r\n return this._binormals;\r\n }\r\n\r\n /**\r\n * Returns an array populated with distances (float) of the i-th point from the first curve point.\r\n * @returns an array populated with distances (float) of the i-th point from the first curve point.\r\n */\r\n public getDistances(): number[] {\r\n return this._distances;\r\n }\r\n\r\n /**\r\n * Returns an interpolated point along this path\r\n * @param position the position of the point along this path, from 0.0 to 1.0\r\n * @returns a new Vector3 as the point\r\n */\r\n public getPointAt(position: number): Vector3 {\r\n return this._updatePointAtData(position).point;\r\n }\r\n\r\n /**\r\n * Returns the tangent vector of an interpolated Path3D curve point at the specified position along this path.\r\n * @param position the position of the point along this path, from 0.0 to 1.0\r\n * @param interpolated (optional, default false) : boolean, if true returns an interpolated tangent instead of the tangent of the previous path point.\r\n * @returns a tangent vector corresponding to the interpolated Path3D curve point, if not interpolated, the tangent is taken from the precomputed tangents array.\r\n */\r\n public getTangentAt(position: number, interpolated = false): Vector3 {\r\n this._updatePointAtData(position, interpolated);\r\n return interpolated ? Vector3.TransformCoordinates(Vector3.Forward(), this._pointAtData.interpolationMatrix) : this._tangents[this._pointAtData.previousPointArrayIndex];\r\n }\r\n\r\n /**\r\n * Returns the tangent vector of an interpolated Path3D curve point at the specified position along this path.\r\n * @param position the position of the point along this path, from 0.0 to 1.0\r\n * @param interpolated (optional, default false) : boolean, if true returns an interpolated normal instead of the normal of the previous path point.\r\n * @returns a normal vector corresponding to the interpolated Path3D curve point, if not interpolated, the normal is taken from the precomputed normals array.\r\n */\r\n public getNormalAt(position: number, interpolated = false): Vector3 {\r\n this._updatePointAtData(position, interpolated);\r\n return interpolated ? Vector3.TransformCoordinates(Vector3.Right(), this._pointAtData.interpolationMatrix) : this._normals[this._pointAtData.previousPointArrayIndex];\r\n }\r\n\r\n /**\r\n * Returns the binormal vector of an interpolated Path3D curve point at the specified position along this path.\r\n * @param position the position of the point along this path, from 0.0 to 1.0\r\n * @param interpolated (optional, default false) : boolean, if true returns an interpolated binormal instead of the binormal of the previous path point.\r\n * @returns a binormal vector corresponding to the interpolated Path3D curve point, if not interpolated, the binormal is taken from the precomputed binormals array.\r\n */\r\n public getBinormalAt(position: number, interpolated = false): Vector3 {\r\n this._updatePointAtData(position, interpolated);\r\n return interpolated ? Vector3.TransformCoordinates(Vector3.UpReadOnly, this._pointAtData.interpolationMatrix) : this._binormals[this._pointAtData.previousPointArrayIndex];\r\n }\r\n\r\n /**\r\n * Returns the distance (float) of an interpolated Path3D curve point at the specified position along this path.\r\n * @param position the position of the point along this path, from 0.0 to 1.0\r\n * @returns the distance of the interpolated Path3D curve point at the specified position along this path.\r\n */\r\n public getDistanceAt(position: number): number {\r\n return this.length() * position;\r\n }\r\n\r\n /**\r\n * Returns the array index of the previous point of an interpolated point along this path\r\n * @param position the position of the point to interpolate along this path, from 0.0 to 1.0\r\n * @returns the array index\r\n */\r\n public getPreviousPointIndexAt(position: number) {\r\n this._updatePointAtData(position);\r\n return this._pointAtData.previousPointArrayIndex;\r\n }\r\n\r\n /**\r\n * Returns the position of an interpolated point relative to the two path points it lies between, from 0.0 (point A) to 1.0 (point B)\r\n * @param position the position of the point to interpolate along this path, from 0.0 to 1.0\r\n * @returns the sub position\r\n */\r\n public getSubPositionAt(position: number) {\r\n this._updatePointAtData(position);\r\n return this._pointAtData.subPosition;\r\n }\r\n\r\n /**\r\n * Returns the position of the closest virtual point on this path to an arbitrary Vector3, from 0.0 to 1.0\r\n * @param target the vector of which to get the closest position to\r\n * @returns the position of the closest virtual point on this path to the target vector\r\n */\r\n public getClosestPositionTo(target: Vector3) {\r\n let smallestDistance = Number.MAX_VALUE;\r\n let closestPosition = 0.0;\r\n for (let i = 0; i < this._curve.length - 1; i++) {\r\n const point = this._curve[i + 0];\r\n const tangent = this._curve[i + 1].subtract(point).normalize();\r\n const subLength = this._distances[i + 1] - this._distances[i + 0];\r\n const subPosition = Math.min((Math.max(Vector3.Dot(tangent, target.subtract(point).normalize()), 0.0) * Vector3.Distance(point, target)) / subLength, 1.0);\r\n const distance = Vector3.Distance(point.add(tangent.scale(subPosition * subLength)), target);\r\n\r\n if (distance < smallestDistance) {\r\n smallestDistance = distance;\r\n closestPosition = (this._distances[i + 0] + subLength * subPosition) / this.length();\r\n }\r\n }\r\n return closestPosition;\r\n }\r\n\r\n /**\r\n * Returns a sub path (slice) of this path\r\n * @param start the position of the fist path point, from 0.0 to 1.0, or a negative value, which will get wrapped around from the end of the path to 0.0 to 1.0 values\r\n * @param end the position of the last path point, from 0.0 to 1.0, or a negative value, which will get wrapped around from the end of the path to 0.0 to 1.0 values\r\n * @returns a sub path (slice) of this path\r\n */\r\n public slice(start: number = 0.0, end: number = 1.0) {\r\n if (start < 0.0) {\r\n start = 1 - ((start * -1.0) % 1.0);\r\n }\r\n if (end < 0.0) {\r\n end = 1 - ((end * -1.0) % 1.0);\r\n }\r\n if (start > end) {\r\n const _start = start;\r\n start = end;\r\n end = _start;\r\n }\r\n const curvePoints = this.getCurve();\r\n\r\n const startPoint = this.getPointAt(start);\r\n let startIndex = this.getPreviousPointIndexAt(start);\r\n\r\n const endPoint = this.getPointAt(end);\r\n const endIndex = this.getPreviousPointIndexAt(end) + 1;\r\n\r\n const slicePoints: Vector3[] = [];\r\n if (start !== 0.0) {\r\n startIndex++;\r\n slicePoints.push(startPoint);\r\n }\r\n\r\n slicePoints.push(...curvePoints.slice(startIndex, endIndex));\r\n if (end !== 1.0 || start === 1.0) {\r\n slicePoints.push(endPoint);\r\n }\r\n return new Path3D(slicePoints, this.getNormalAt(start), this._raw, this._alignTangentsWithPath);\r\n }\r\n\r\n /**\r\n * Forces the Path3D tangent, normal, binormal and distance recomputation.\r\n * @param path path which all values are copied into the curves points\r\n * @param firstNormal which should be projected onto the curve\r\n * @param alignTangentsWithPath (optional, default false) : boolean, if true the tangents will be aligned with the path\r\n * @returns the same object updated.\r\n */\r\n public update(path: Vector3[], firstNormal: Nullable = null, alignTangentsWithPath = false): Path3D {\r\n for (let p = 0; p < path.length; p++) {\r\n this._curve[p].x = path[p].x;\r\n this._curve[p].y = path[p].y;\r\n this._curve[p].z = path[p].z;\r\n }\r\n this._compute(firstNormal, alignTangentsWithPath);\r\n return this;\r\n }\r\n\r\n // private function compute() : computes tangents, normals and binormals\r\n private _compute(firstNormal: Nullable, alignTangentsWithPath = false): void {\r\n const l = this._curve.length;\r\n\r\n if (l < 2) {\r\n return;\r\n }\r\n\r\n // first and last tangents\r\n this._tangents[0] = this._getFirstNonNullVector(0);\r\n if (!this._raw) {\r\n this._tangents[0].normalize();\r\n }\r\n this._tangents[l - 1] = this._curve[l - 1].subtract(this._curve[l - 2]);\r\n if (!this._raw) {\r\n this._tangents[l - 1].normalize();\r\n }\r\n\r\n // normals and binormals at first point : arbitrary vector with _normalVector()\r\n const tg0 = this._tangents[0];\r\n const pp0 = this._normalVector(tg0, firstNormal);\r\n this._normals[0] = pp0;\r\n if (!this._raw) {\r\n this._normals[0].normalize();\r\n }\r\n this._binormals[0] = Vector3.Cross(tg0, this._normals[0]);\r\n if (!this._raw) {\r\n this._binormals[0].normalize();\r\n }\r\n this._distances[0] = 0.0;\r\n\r\n // normals and binormals : next points\r\n let prev: Vector3; // previous vector (segment)\r\n let cur: Vector3; // current vector (segment)\r\n let curTang: Vector3; // current tangent\r\n // previous normal\r\n let prevNor: Vector3; // previous normal\r\n let prevBinor: Vector3; // previous binormal\r\n\r\n for (let i = 1; i < l; i++) {\r\n // tangents\r\n prev = this._getLastNonNullVector(i);\r\n if (i < l - 1) {\r\n cur = this._getFirstNonNullVector(i);\r\n this._tangents[i] = alignTangentsWithPath ? cur : prev.add(cur);\r\n this._tangents[i].normalize();\r\n }\r\n this._distances[i] = this._distances[i - 1] + this._curve[i].subtract(this._curve[i - 1]).length();\r\n\r\n // normals and binormals\r\n // http://www.cs.cmu.edu/afs/andrew/scs/cs/15-462/web/old/asst2camera.html\r\n curTang = this._tangents[i];\r\n prevBinor = this._binormals[i - 1];\r\n this._normals[i] = Vector3.Cross(prevBinor, curTang);\r\n if (!this._raw) {\r\n if (this._normals[i].length() === 0) {\r\n prevNor = this._normals[i - 1];\r\n this._normals[i] = prevNor.clone();\r\n } else {\r\n this._normals[i].normalize();\r\n }\r\n }\r\n this._binormals[i] = Vector3.Cross(curTang, this._normals[i]);\r\n if (!this._raw) {\r\n this._binormals[i].normalize();\r\n }\r\n }\r\n this._pointAtData.id = NaN;\r\n }\r\n\r\n // private function getFirstNonNullVector(index)\r\n // returns the first non null vector from index : curve[index + N].subtract(curve[index])\r\n private _getFirstNonNullVector(index: number): Vector3 {\r\n let i = 1;\r\n let nNVector: Vector3 = this._curve[index + i].subtract(this._curve[index]);\r\n while (nNVector.length() === 0 && index + i + 1 < this._curve.length) {\r\n i++;\r\n nNVector = this._curve[index + i].subtract(this._curve[index]);\r\n }\r\n return nNVector;\r\n }\r\n\r\n // private function getLastNonNullVector(index)\r\n // returns the last non null vector from index : curve[index].subtract(curve[index - N])\r\n private _getLastNonNullVector(index: number): Vector3 {\r\n let i = 1;\r\n let nLVector: Vector3 = this._curve[index].subtract(this._curve[index - i]);\r\n while (nLVector.length() === 0 && index > i + 1) {\r\n i++;\r\n nLVector = this._curve[index].subtract(this._curve[index - i]);\r\n }\r\n return nLVector;\r\n }\r\n\r\n // private function normalVector(v0, vt, va) :\r\n // returns an arbitrary point in the plane defined by the point v0 and the vector vt orthogonal to this plane\r\n // if va is passed, it returns the va projection on the plane orthogonal to vt at the point v0\r\n private _normalVector(vt: Vector3, va: Nullable): Vector3 {\r\n let normal0: Vector3;\r\n let tgl = vt.length();\r\n if (tgl === 0.0) {\r\n tgl = 1.0;\r\n }\r\n\r\n if (va === undefined || va === null) {\r\n let point: Vector3;\r\n if (!Scalar.WithinEpsilon(Math.abs(vt.y) / tgl, 1.0, Epsilon)) {\r\n // search for a point in the plane\r\n point = new Vector3(0.0, -1.0, 0.0);\r\n } else if (!Scalar.WithinEpsilon(Math.abs(vt.x) / tgl, 1.0, Epsilon)) {\r\n point = new Vector3(1.0, 0.0, 0.0);\r\n } else if (!Scalar.WithinEpsilon(Math.abs(vt.z) / tgl, 1.0, Epsilon)) {\r\n point = new Vector3(0.0, 0.0, 1.0);\r\n } else {\r\n point = Vector3.Zero();\r\n }\r\n normal0 = Vector3.Cross(vt, point);\r\n } else {\r\n normal0 = Vector3.Cross(vt, va);\r\n Vector3.CrossToRef(normal0, vt, normal0);\r\n }\r\n normal0.normalize();\r\n return normal0;\r\n }\r\n\r\n /**\r\n * Updates the point at data for an interpolated point along this curve\r\n * @param position the position of the point along this curve, from 0.0 to 1.0\r\n * @param interpolateTNB\r\n * @interpolateTNB whether to compute the interpolated tangent, normal and binormal\r\n * @returns the (updated) point at data\r\n */\r\n private _updatePointAtData(position: number, interpolateTNB: boolean = false) {\r\n // set an id for caching the result\r\n if (this._pointAtData.id === position) {\r\n if (!this._pointAtData.interpolateReady) {\r\n this._updateInterpolationMatrix();\r\n }\r\n return this._pointAtData;\r\n } else {\r\n this._pointAtData.id = position;\r\n }\r\n const curvePoints = this.getPoints();\r\n\r\n // clamp position between 0.0 and 1.0\r\n if (position <= 0.0) {\r\n return this._setPointAtData(0.0, 0.0, curvePoints[0], 0, interpolateTNB);\r\n } else if (position >= 1.0) {\r\n return this._setPointAtData(1.0, 1.0, curvePoints[curvePoints.length - 1], curvePoints.length - 1, interpolateTNB);\r\n }\r\n\r\n let previousPoint: Vector3 = curvePoints[0];\r\n let currentPoint: Vector3;\r\n let currentLength = 0.0;\r\n const targetLength = position * this.length();\r\n\r\n for (let i = 1; i < curvePoints.length; i++) {\r\n currentPoint = curvePoints[i];\r\n const distance = Vector3.Distance(previousPoint, currentPoint);\r\n currentLength += distance;\r\n if (currentLength === targetLength) {\r\n return this._setPointAtData(position, 1.0, currentPoint, i, interpolateTNB);\r\n } else if (currentLength > targetLength) {\r\n const toLength = currentLength - targetLength;\r\n const diff = toLength / distance;\r\n const dir = previousPoint.subtract(currentPoint);\r\n const point = currentPoint.add(dir.scaleInPlace(diff));\r\n return this._setPointAtData(position, 1 - diff, point, i - 1, interpolateTNB);\r\n }\r\n previousPoint = currentPoint;\r\n }\r\n return this._pointAtData;\r\n }\r\n\r\n /**\r\n * Updates the point at data from the specified parameters\r\n * @param position where along the path the interpolated point is, from 0.0 to 1.0\r\n * @param subPosition\r\n * @param point the interpolated point\r\n * @param parentIndex the index of an existing curve point that is on, or else positionally the first behind, the interpolated point\r\n * @param interpolateTNB\r\n */\r\n private _setPointAtData(position: number, subPosition: number, point: Vector3, parentIndex: number, interpolateTNB: boolean) {\r\n this._pointAtData.point = point;\r\n this._pointAtData.position = position;\r\n this._pointAtData.subPosition = subPosition;\r\n this._pointAtData.previousPointArrayIndex = parentIndex;\r\n this._pointAtData.interpolateReady = interpolateTNB;\r\n\r\n if (interpolateTNB) {\r\n this._updateInterpolationMatrix();\r\n }\r\n return this._pointAtData;\r\n }\r\n\r\n /**\r\n * Updates the point at interpolation matrix for the tangents, normals and binormals\r\n */\r\n private _updateInterpolationMatrix() {\r\n this._pointAtData.interpolationMatrix = Matrix.Identity();\r\n const parentIndex = this._pointAtData.previousPointArrayIndex;\r\n\r\n if (parentIndex !== this._tangents.length - 1) {\r\n const index = parentIndex + 1;\r\n\r\n const tangentFrom = this._tangents[parentIndex].clone();\r\n const normalFrom = this._normals[parentIndex].clone();\r\n const binormalFrom = this._binormals[parentIndex].clone();\r\n\r\n const tangentTo = this._tangents[index].clone();\r\n const normalTo = this._normals[index].clone();\r\n const binormalTo = this._binormals[index].clone();\r\n\r\n const quatFrom = Quaternion.RotationQuaternionFromAxis(normalFrom, binormalFrom, tangentFrom);\r\n const quatTo = Quaternion.RotationQuaternionFromAxis(normalTo, binormalTo, tangentTo);\r\n const quatAt = Quaternion.Slerp(quatFrom, quatTo, this._pointAtData.subPosition);\r\n\r\n quatAt.toRotationMatrix(this._pointAtData.interpolationMatrix);\r\n }\r\n }\r\n}\r\n\r\n/**\r\n * A Curve3 object is a logical object, so not a mesh, to handle curves in the 3D geometric space.\r\n * A Curve3 is designed from a series of successive Vector3.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/drawCurves\r\n */\r\nexport class Curve3 {\r\n private _points: Vector3[];\r\n private _length: number = 0.0;\r\n\r\n /**\r\n * Returns a Curve3 object along a Quadratic Bezier curve : https://doc.babylonjs.com/features/featuresDeepDive/mesh/drawCurves#quadratic-bezier-curve\r\n * @param v0 (Vector3) the origin point of the Quadratic Bezier\r\n * @param v1 (Vector3) the control point\r\n * @param v2 (Vector3) the end point of the Quadratic Bezier\r\n * @param nbPoints (integer) the wanted number of points in the curve\r\n * @returns the created Curve3\r\n */\r\n public static CreateQuadraticBezier(v0: DeepImmutable, v1: DeepImmutable, v2: DeepImmutable, nbPoints: number): Curve3 {\r\n nbPoints = nbPoints > 2 ? nbPoints : 3;\r\n const bez = new Array();\r\n const equation = (t: number, val0: number, val1: number, val2: number) => {\r\n const res = (1.0 - t) * (1.0 - t) * val0 + 2.0 * t * (1.0 - t) * val1 + t * t * val2;\r\n return res;\r\n };\r\n for (let i = 0; i <= nbPoints; i++) {\r\n bez.push(new Vector3(equation(i / nbPoints, v0.x, v1.x, v2.x), equation(i / nbPoints, v0.y, v1.y, v2.y), equation(i / nbPoints, v0.z, v1.z, v2.z)));\r\n }\r\n return new Curve3(bez);\r\n }\r\n\r\n /**\r\n * Returns a Curve3 object along a Cubic Bezier curve : https://doc.babylonjs.com/features/featuresDeepDive/mesh/drawCurves#cubic-bezier-curve\r\n * @param v0 (Vector3) the origin point of the Cubic Bezier\r\n * @param v1 (Vector3) the first control point\r\n * @param v2 (Vector3) the second control point\r\n * @param v3 (Vector3) the end point of the Cubic Bezier\r\n * @param nbPoints (integer) the wanted number of points in the curve\r\n * @returns the created Curve3\r\n */\r\n public static CreateCubicBezier(v0: DeepImmutable, v1: DeepImmutable, v2: DeepImmutable, v3: DeepImmutable, nbPoints: number): Curve3 {\r\n nbPoints = nbPoints > 3 ? nbPoints : 4;\r\n const bez = new Array();\r\n const equation = (t: number, val0: number, val1: number, val2: number, val3: number) => {\r\n const res = (1.0 - t) * (1.0 - t) * (1.0 - t) * val0 + 3.0 * t * (1.0 - t) * (1.0 - t) * val1 + 3.0 * t * t * (1.0 - t) * val2 + t * t * t * val3;\r\n return res;\r\n };\r\n for (let i = 0; i <= nbPoints; i++) {\r\n bez.push(new Vector3(equation(i / nbPoints, v0.x, v1.x, v2.x, v3.x), equation(i / nbPoints, v0.y, v1.y, v2.y, v3.y), equation(i / nbPoints, v0.z, v1.z, v2.z, v3.z)));\r\n }\r\n return new Curve3(bez);\r\n }\r\n\r\n /**\r\n * Returns a Curve3 object along a Hermite Spline curve : https://doc.babylonjs.com/features/featuresDeepDive/mesh/drawCurves#hermite-spline\r\n * @param p1 (Vector3) the origin point of the Hermite Spline\r\n * @param t1 (Vector3) the tangent vector at the origin point\r\n * @param p2 (Vector3) the end point of the Hermite Spline\r\n * @param t2 (Vector3) the tangent vector at the end point\r\n * @param nSeg (integer) the number of curve segments or nSeg + 1 points in the array\r\n * @returns the created Curve3\r\n */\r\n public static CreateHermiteSpline(p1: DeepImmutable, t1: DeepImmutable, p2: DeepImmutable, t2: DeepImmutable, nSeg: number): Curve3 {\r\n const hermite = new Array();\r\n const step = 1.0 / nSeg;\r\n for (let i = 0; i <= nSeg; i++) {\r\n hermite.push(Vector3.Hermite(p1, t1, p2, t2, i * step));\r\n }\r\n return new Curve3(hermite);\r\n }\r\n\r\n /**\r\n * Returns a Curve3 object along a CatmullRom Spline curve :\r\n * @param points (array of Vector3) the points the spline must pass through. At least, four points required\r\n * @param nbPoints (integer) the wanted number of points between each curve control points\r\n * @param closed (boolean) optional with default false, when true forms a closed loop from the points\r\n * @returns the created Curve3\r\n */\r\n public static CreateCatmullRomSpline(points: DeepImmutable, nbPoints: number, closed?: boolean): Curve3 {\r\n const catmullRom = new Array();\r\n const step = 1.0 / nbPoints;\r\n let amount = 0.0;\r\n if (closed) {\r\n const pointsCount = points.length;\r\n for (let i = 0; i < pointsCount; i++) {\r\n amount = 0;\r\n for (let c = 0; c < nbPoints; c++) {\r\n catmullRom.push(\r\n Vector3.CatmullRom(points[i % pointsCount], points[(i + 1) % pointsCount], points[(i + 2) % pointsCount], points[(i + 3) % pointsCount], amount)\r\n );\r\n amount += step;\r\n }\r\n }\r\n catmullRom.push(catmullRom[0]);\r\n } else {\r\n const totalPoints = new Array();\r\n totalPoints.push(points[0].clone());\r\n Array.prototype.push.apply(totalPoints, points);\r\n totalPoints.push(points[points.length - 1].clone());\r\n let i = 0;\r\n for (; i < totalPoints.length - 3; i++) {\r\n amount = 0;\r\n for (let c = 0; c < nbPoints; c++) {\r\n catmullRom.push(Vector3.CatmullRom(totalPoints[i], totalPoints[i + 1], totalPoints[i + 2], totalPoints[i + 3], amount));\r\n amount += step;\r\n }\r\n }\r\n i--;\r\n catmullRom.push(Vector3.CatmullRom(totalPoints[i], totalPoints[i + 1], totalPoints[i + 2], totalPoints[i + 3], amount));\r\n }\r\n return new Curve3(catmullRom);\r\n }\r\n\r\n /**\r\n * Returns a Curve3 object along an arc through three vector3 points:\r\n * The three points should not be colinear. When they are the Curve3 is empty.\r\n * @param first (Vector3) the first point the arc must pass through.\r\n * @param second (Vector3) the second point the arc must pass through.\r\n * @param third (Vector3) the third point the arc must pass through.\r\n * @param steps (number) the larger the number of steps the more detailed the arc.\r\n * @param closed (boolean) optional with default false, when true forms the chord from the first and third point\r\n * @param fullCircle Circle (boolean) optional with default false, when true forms the complete circle through the three points\r\n * @returns the created Curve3\r\n */\r\n public static ArcThru3Points(first: Vector3, second: Vector3, third: Vector3, steps: number = 32, closed: boolean = false, fullCircle: boolean = false): Curve3 {\r\n const arc = new Array();\r\n const vec1 = second.subtract(first);\r\n const vec2 = third.subtract(second);\r\n const vec3 = first.subtract(third);\r\n const zAxis = Vector3.Cross(vec1, vec2);\r\n const len4 = zAxis.length();\r\n if (len4 < Math.pow(10, -8)) {\r\n return new Curve3(arc); // colinear points arc is empty\r\n }\r\n const len1_sq = vec1.lengthSquared();\r\n const len2_sq = vec2.lengthSquared();\r\n const len3_sq = vec3.lengthSquared();\r\n const len4_sq = zAxis.lengthSquared();\r\n const len1 = vec1.length();\r\n const len2 = vec2.length();\r\n const len3 = vec3.length();\r\n const radius = (0.5 * len1 * len2 * len3) / len4;\r\n const dot1 = Vector3.Dot(vec1, vec3);\r\n const dot2 = Vector3.Dot(vec1, vec2);\r\n const dot3 = Vector3.Dot(vec2, vec3);\r\n const a = (-0.5 * len2_sq * dot1) / len4_sq;\r\n const b = (-0.5 * len3_sq * dot2) / len4_sq;\r\n const c = (-0.5 * len1_sq * dot3) / len4_sq;\r\n const center = first.scale(a).add(second.scale(b)).add(third.scale(c));\r\n const radiusVec = first.subtract(center);\r\n const xAxis = radiusVec.normalize();\r\n const yAxis = Vector3.Cross(zAxis, xAxis).normalize();\r\n if (fullCircle) {\r\n const dStep = (2 * Math.PI) / steps;\r\n for (let theta = 0; theta <= 2 * Math.PI; theta += dStep) {\r\n arc.push(center.add(xAxis.scale(radius * Math.cos(theta)).add(yAxis.scale(radius * Math.sin(theta)))));\r\n }\r\n arc.push(first);\r\n } else {\r\n const dStep = 1 / steps;\r\n let theta = 0;\r\n let point = Vector3.Zero();\r\n do {\r\n point = center.add(xAxis.scale(radius * Math.cos(theta)).add(yAxis.scale(radius * Math.sin(theta))));\r\n arc.push(point);\r\n theta += dStep;\r\n } while (!point.equalsWithEpsilon(third, radius * dStep * 1.1));\r\n arc.push(third);\r\n if (closed) {\r\n arc.push(first);\r\n }\r\n }\r\n return new Curve3(arc);\r\n }\r\n\r\n /**\r\n * A Curve3 object is a logical object, so not a mesh, to handle curves in the 3D geometric space.\r\n * A Curve3 is designed from a series of successive Vector3.\r\n * Tuto : https://doc.babylonjs.com/features/featuresDeepDive/mesh/drawCurves#curve3-object\r\n * @param points points which make up the curve\r\n */\r\n constructor(points: Vector3[]) {\r\n this._points = points;\r\n this._length = this._computeLength(points);\r\n }\r\n\r\n /**\r\n * @returns the Curve3 stored array of successive Vector3\r\n */\r\n public getPoints() {\r\n return this._points;\r\n }\r\n\r\n /**\r\n * @returns the computed length (float) of the curve.\r\n */\r\n public length() {\r\n return this._length;\r\n }\r\n\r\n /**\r\n * Returns a new instance of Curve3 object : var curve = curveA.continue(curveB);\r\n * This new Curve3 is built by translating and sticking the curveB at the end of the curveA.\r\n * curveA and curveB keep unchanged.\r\n * @param curve the curve to continue from this curve\r\n * @returns the newly constructed curve\r\n */\r\n public continue(curve: DeepImmutable): Curve3 {\r\n const lastPoint = this._points[this._points.length - 1];\r\n const continuedPoints = this._points.slice();\r\n const curvePoints = curve.getPoints();\r\n for (let i = 1; i < curvePoints.length; i++) {\r\n continuedPoints.push(curvePoints[i].subtract(curvePoints[0]).add(lastPoint));\r\n }\r\n const continuedCurve = new Curve3(continuedPoints);\r\n return continuedCurve;\r\n }\r\n\r\n private _computeLength(path: DeepImmutable): number {\r\n let l = 0;\r\n for (let i = 1; i < path.length; i++) {\r\n l += path[i].subtract(path[i - 1]).length();\r\n }\r\n return l;\r\n }\r\n}\r\n","/**\r\n * Defines an interface which represents an animation key frame\r\n */\r\nexport interface IAnimationKey {\r\n /**\r\n * Frame of the key frame\r\n */\r\n frame: number;\r\n /**\r\n * Value at the specifies key frame\r\n */\r\n value: any;\r\n /**\r\n * The input tangent for the cubic hermite spline\r\n */\r\n inTangent?: any;\r\n /**\r\n * The output tangent for the cubic hermite spline\r\n */\r\n outTangent?: any;\r\n /**\r\n * The animation interpolation type\r\n */\r\n interpolation?: AnimationKeyInterpolation;\r\n /**\r\n * Property defined by UI tools to link (or not ) the tangents\r\n */\r\n lockedTangent?: boolean;\r\n}\r\n\r\n/**\r\n * Enum for the animation key frame interpolation type\r\n */\r\nexport enum AnimationKeyInterpolation {\r\n /**\r\n * Use tangents to interpolate between start and end values.\r\n */\r\n NONE = 0,\r\n /**\r\n * Do not interpolate between keys and use the start key value only. Tangents are ignored\r\n */\r\n STEP = 1,\r\n}\r\n","import { serialize, SerializationHelper } from \"../Misc/decorators\";\r\nimport { Tools } from \"../Misc/tools\";\r\nimport type { IAnimatable } from \"../Animations/animatable.interface\";\r\nimport type { SmartArray } from \"../Misc/smartArray\";\r\nimport type { Observer } from \"../Misc/observable\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { Matrix } from \"../Maths/math.vector\";\r\nimport { EngineStore } from \"../Engines/engineStore\";\r\nimport { SubMesh } from \"../Meshes/subMesh\";\r\nimport type { Geometry } from \"../Meshes/geometry\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport { UniformBuffer } from \"./uniformBuffer\";\r\nimport type { Effect } from \"./effect\";\r\nimport type { BaseTexture } from \"../Materials/Textures/baseTexture\";\r\nimport type { RenderTargetTexture } from \"../Materials/Textures/renderTargetTexture\";\r\nimport type { MaterialDefines } from \"./materialDefines\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport { Logger } from \"../Misc/logger\";\r\nimport type { IInspectable } from \"../Misc/iInspectable\";\r\nimport { Plane } from \"../Maths/math.plane\";\r\nimport type { ShadowDepthWrapper } from \"./shadowDepthWrapper\";\r\nimport { MaterialHelper } from \"./materialHelper\";\r\nimport type { IMaterialContext } from \"../Engines/IMaterialContext\";\r\nimport { DrawWrapper } from \"./drawWrapper\";\r\nimport { MaterialStencilState } from \"./materialStencilState\";\r\nimport { ScenePerformancePriority } from \"../scene\";\r\nimport type { Scene } from \"../scene\";\r\nimport type { AbstractScene } from \"../abstractScene\";\r\nimport type {\r\n MaterialPluginDisposed,\r\n MaterialPluginIsReadyForSubMesh,\r\n MaterialPluginGetDefineNames,\r\n MaterialPluginBindForSubMesh,\r\n MaterialPluginGetActiveTextures,\r\n MaterialPluginHasTexture,\r\n MaterialPluginGetAnimatables,\r\n MaterialPluginPrepareDefines,\r\n MaterialPluginPrepareEffect,\r\n MaterialPluginPrepareUniformBuffer,\r\n MaterialPluginCreated,\r\n MaterialPluginFillRenderTargetTextures,\r\n MaterialPluginHasRenderTargetTextures,\r\n MaterialPluginHardBindForSubMesh,\r\n} from \"./materialPluginEvent\";\r\nimport { MaterialPluginEvent } from \"./materialPluginEvent\";\r\nimport type { ShaderCustomProcessingFunction } from \"../Engines/Processors/shaderProcessingOptions\";\r\nimport type { IClipPlanesHolder } from \"../Misc/interfaces/iClipPlanesHolder\";\r\n\r\nimport type { PrePassRenderer } from \"../Rendering/prePassRenderer\";\r\nimport type { Mesh } from \"../Meshes/mesh\";\r\nimport type { Animation } from \"../Animations/animation\";\r\nimport type { InstancedMesh } from \"../Meshes/instancedMesh\";\r\n\r\ndeclare let BABYLON: any;\r\n\r\n/**\r\n * Options for compiling materials.\r\n */\r\nexport interface IMaterialCompilationOptions {\r\n /**\r\n * Defines whether clip planes are enabled.\r\n */\r\n clipPlane: boolean;\r\n\r\n /**\r\n * Defines whether instances are enabled.\r\n */\r\n useInstances: boolean;\r\n}\r\n\r\n/**\r\n * Options passed when calling customShaderNameResolve\r\n */\r\nexport interface ICustomShaderNameResolveOptions {\r\n /**\r\n * If provided, will be called two times with the vertex and fragment code so that this code can be updated before it is compiled by the GPU\r\n */\r\n processFinalCode?: Nullable;\r\n}\r\n\r\n/**\r\n * Base class for the main features of a material in Babylon.js\r\n */\r\nexport class Material implements IAnimatable, IClipPlanesHolder {\r\n /**\r\n * Returns the triangle fill mode\r\n */\r\n public static readonly TriangleFillMode = Constants.MATERIAL_TriangleFillMode;\r\n /**\r\n * Returns the wireframe mode\r\n */\r\n public static readonly WireFrameFillMode = Constants.MATERIAL_WireFrameFillMode;\r\n /**\r\n * Returns the point fill mode\r\n */\r\n public static readonly PointFillMode = Constants.MATERIAL_PointFillMode;\r\n /**\r\n * Returns the point list draw mode\r\n */\r\n public static readonly PointListDrawMode = Constants.MATERIAL_PointListDrawMode;\r\n /**\r\n * Returns the line list draw mode\r\n */\r\n public static readonly LineListDrawMode = Constants.MATERIAL_LineListDrawMode;\r\n /**\r\n * Returns the line loop draw mode\r\n */\r\n public static readonly LineLoopDrawMode = Constants.MATERIAL_LineLoopDrawMode;\r\n /**\r\n * Returns the line strip draw mode\r\n */\r\n public static readonly LineStripDrawMode = Constants.MATERIAL_LineStripDrawMode;\r\n /**\r\n * Returns the triangle strip draw mode\r\n */\r\n public static readonly TriangleStripDrawMode = Constants.MATERIAL_TriangleStripDrawMode;\r\n /**\r\n * Returns the triangle fan draw mode\r\n */\r\n public static readonly TriangleFanDrawMode = Constants.MATERIAL_TriangleFanDrawMode;\r\n\r\n /**\r\n * Stores the clock-wise side orientation\r\n */\r\n public static readonly ClockWiseSideOrientation = Constants.MATERIAL_ClockWiseSideOrientation;\r\n\r\n /**\r\n * Stores the counter clock-wise side orientation\r\n */\r\n public static readonly CounterClockWiseSideOrientation = Constants.MATERIAL_CounterClockWiseSideOrientation;\r\n\r\n /**\r\n * The dirty texture flag value\r\n */\r\n public static readonly TextureDirtyFlag = Constants.MATERIAL_TextureDirtyFlag;\r\n\r\n /**\r\n * The dirty light flag value\r\n */\r\n public static readonly LightDirtyFlag = Constants.MATERIAL_LightDirtyFlag;\r\n\r\n /**\r\n * The dirty fresnel flag value\r\n */\r\n public static readonly FresnelDirtyFlag = Constants.MATERIAL_FresnelDirtyFlag;\r\n\r\n /**\r\n * The dirty attribute flag value\r\n */\r\n public static readonly AttributesDirtyFlag = Constants.MATERIAL_AttributesDirtyFlag;\r\n\r\n /**\r\n * The dirty misc flag value\r\n */\r\n public static readonly MiscDirtyFlag = Constants.MATERIAL_MiscDirtyFlag;\r\n\r\n /**\r\n * The dirty prepass flag value\r\n */\r\n public static readonly PrePassDirtyFlag = Constants.MATERIAL_PrePassDirtyFlag;\r\n\r\n /**\r\n * The all dirty flag value\r\n */\r\n public static readonly AllDirtyFlag = Constants.MATERIAL_AllDirtyFlag;\r\n\r\n /**\r\n * MaterialTransparencyMode: No transparency mode, Alpha channel is not use.\r\n */\r\n public static readonly MATERIAL_OPAQUE = 0;\r\n\r\n /**\r\n * MaterialTransparencyMode: Alpha Test mode, pixel are discarded below a certain threshold defined by the alpha cutoff value.\r\n */\r\n public static readonly MATERIAL_ALPHATEST = 1;\r\n\r\n /**\r\n * MaterialTransparencyMode: Pixels are blended (according to the alpha mode) with the already drawn pixels in the current frame buffer.\r\n */\r\n public static readonly MATERIAL_ALPHABLEND = 2;\r\n\r\n /**\r\n * MaterialTransparencyMode: Pixels are blended (according to the alpha mode) with the already drawn pixels in the current frame buffer.\r\n * They are also discarded below the alpha cutoff threshold to improve performances.\r\n */\r\n public static readonly MATERIAL_ALPHATESTANDBLEND = 3;\r\n\r\n /**\r\n * The Whiteout method is used to blend normals.\r\n * Details of the algorithm can be found here: https://blog.selfshadow.com/publications/blending-in-detail/\r\n */\r\n public static readonly MATERIAL_NORMALBLENDMETHOD_WHITEOUT = 0;\r\n\r\n /**\r\n * The Reoriented Normal Mapping method is used to blend normals.\r\n * Details of the algorithm can be found here: https://blog.selfshadow.com/publications/blending-in-detail/\r\n */\r\n public static readonly MATERIAL_NORMALBLENDMETHOD_RNM = 1;\r\n\r\n /**\r\n * Event observable which raises global events common to all materials (like MaterialPluginEvent.Created)\r\n */\r\n public static OnEventObservable = new Observable();\r\n\r\n /**\r\n * Custom callback helping to override the default shader used in the material.\r\n */\r\n public customShaderNameResolve: (\r\n shaderName: string,\r\n uniforms: string[],\r\n uniformBuffers: string[],\r\n samplers: string[],\r\n defines: MaterialDefines | string[],\r\n attributes?: string[],\r\n options?: ICustomShaderNameResolveOptions\r\n ) => string;\r\n\r\n /**\r\n * Custom shadow depth material to use for shadow rendering instead of the in-built one\r\n */\r\n public shadowDepthWrapper: Nullable = null;\r\n\r\n /**\r\n * Gets or sets a boolean indicating that the material is allowed (if supported) to do shader hot swapping.\r\n * This means that the material can keep using a previous shader while a new one is being compiled.\r\n * This is mostly used when shader parallel compilation is supported (true by default)\r\n */\r\n public allowShaderHotSwapping = true;\r\n\r\n /**\r\n * The ID of the material\r\n */\r\n @serialize()\r\n public id: string;\r\n\r\n /**\r\n * Gets or sets the unique id of the material\r\n */\r\n @serialize()\r\n public uniqueId: number;\r\n\r\n /** @internal */\r\n public _loadedUniqueId: string;\r\n\r\n /**\r\n * The name of the material\r\n */\r\n @serialize()\r\n public name: string;\r\n\r\n /**\r\n * Gets or sets user defined metadata\r\n */\r\n @serialize()\r\n public metadata: any = null;\r\n\r\n /** @internal */\r\n public _internalMetadata: any;\r\n\r\n /**\r\n * For internal use only. Please do not use.\r\n */\r\n public reservedDataStore: any = null;\r\n\r\n /**\r\n * Specifies if the ready state should be checked on each call\r\n */\r\n @serialize()\r\n public checkReadyOnEveryCall = false;\r\n\r\n /**\r\n * Specifies if the ready state should be checked once\r\n */\r\n @serialize()\r\n public checkReadyOnlyOnce = false;\r\n\r\n /**\r\n * The state of the material\r\n */\r\n @serialize()\r\n public state = \"\";\r\n\r\n /**\r\n * If the material can be rendered to several textures with MRT extension\r\n */\r\n public get canRenderToMRT(): boolean {\r\n // By default, shaders are not compatible with MRTs\r\n // Base classes should override that if their shader supports MRT\r\n return false;\r\n }\r\n\r\n /**\r\n * The alpha value of the material\r\n */\r\n @serialize(\"alpha\")\r\n protected _alpha = 1.0;\r\n\r\n /**\r\n * List of inspectable custom properties (used by the Inspector)\r\n * @see https://doc.babylonjs.com/toolsAndResources/inspector#extensibility\r\n */\r\n public inspectableCustomProperties: IInspectable[];\r\n\r\n /**\r\n * Sets the alpha value of the material\r\n */\r\n public set alpha(value: number) {\r\n if (this._alpha === value) {\r\n return;\r\n }\r\n\r\n const oldValue = this._alpha;\r\n this._alpha = value;\r\n\r\n // Only call dirty when there is a state change (no alpha / alpha)\r\n if (oldValue === 1 || value === 1) {\r\n this.markAsDirty(Material.MiscDirtyFlag + Material.PrePassDirtyFlag);\r\n }\r\n }\r\n\r\n /**\r\n * Gets the alpha value of the material\r\n */\r\n public get alpha(): number {\r\n return this._alpha;\r\n }\r\n\r\n /**\r\n * Specifies if back face culling is enabled\r\n */\r\n @serialize(\"backFaceCulling\")\r\n protected _backFaceCulling = true;\r\n\r\n /**\r\n * Sets the culling state (true to enable culling, false to disable)\r\n */\r\n public set backFaceCulling(value: boolean) {\r\n if (this._backFaceCulling === value) {\r\n return;\r\n }\r\n this._backFaceCulling = value;\r\n this.markAsDirty(Material.TextureDirtyFlag);\r\n }\r\n\r\n /**\r\n * Gets the culling state\r\n */\r\n public get backFaceCulling(): boolean {\r\n return this._backFaceCulling;\r\n }\r\n\r\n /**\r\n * Specifies if back or front faces should be culled (when culling is enabled)\r\n */\r\n @serialize(\"cullBackFaces\")\r\n protected _cullBackFaces = true;\r\n\r\n /**\r\n * Sets the type of faces that should be culled (true for back faces, false for front faces)\r\n */\r\n public set cullBackFaces(value: boolean) {\r\n if (this._cullBackFaces === value) {\r\n return;\r\n }\r\n this._cullBackFaces = value;\r\n this.markAsDirty(Material.TextureDirtyFlag);\r\n }\r\n\r\n /**\r\n * Gets the type of faces that should be culled\r\n */\r\n public get cullBackFaces(): boolean {\r\n return this._cullBackFaces;\r\n }\r\n\r\n private _blockDirtyMechanism = false;\r\n\r\n /**\r\n * Block the dirty-mechanism for this specific material\r\n * When set to false after being true the material will be marked as dirty.\r\n */\r\n public get blockDirtyMechanism(): boolean {\r\n return this._blockDirtyMechanism;\r\n }\r\n\r\n public set blockDirtyMechanism(value: boolean) {\r\n if (this._blockDirtyMechanism === value) {\r\n return;\r\n }\r\n\r\n this._blockDirtyMechanism = value;\r\n\r\n if (!value) {\r\n this.markDirty();\r\n }\r\n }\r\n\r\n /**\r\n * This allows you to modify the material without marking it as dirty after every change.\r\n * This function should be used if you need to make more than one dirty-enabling change to the material - adding a texture, setting a new fill mode and so on.\r\n * The callback will pass the material as an argument, so you can make your changes to it.\r\n * @param callback the callback to be executed that will update the material\r\n */\r\n public atomicMaterialsUpdate(callback: (material: this) => void): void {\r\n this.blockDirtyMechanism = true;\r\n try {\r\n callback(this);\r\n } finally {\r\n this.blockDirtyMechanism = false;\r\n }\r\n }\r\n\r\n /**\r\n * Stores the value for side orientation\r\n */\r\n @serialize()\r\n public sideOrientation: number;\r\n\r\n /**\r\n * Callback triggered when the material is compiled\r\n */\r\n public onCompiled: Nullable<(effect: Effect) => void> = null;\r\n\r\n /**\r\n * Callback triggered when an error occurs\r\n */\r\n public onError: Nullable<(effect: Effect, errors: string) => void> = null;\r\n\r\n /**\r\n * Callback triggered to get the render target textures\r\n */\r\n public getRenderTargetTextures: Nullable<() => SmartArray> = null;\r\n\r\n /**\r\n * Gets a boolean indicating that current material needs to register RTT\r\n */\r\n public get hasRenderTargetTextures(): boolean {\r\n this._eventInfo.hasRenderTargetTextures = false;\r\n this._callbackPluginEventHasRenderTargetTextures(this._eventInfo);\r\n return this._eventInfo.hasRenderTargetTextures;\r\n }\r\n\r\n /**\r\n * Specifies if the material should be serialized\r\n */\r\n public doNotSerialize = false;\r\n\r\n /**\r\n * @internal\r\n */\r\n public _storeEffectOnSubMeshes = false;\r\n\r\n /**\r\n * Stores the animations for the material\r\n */\r\n public animations: Nullable> = null;\r\n\r\n /**\r\n * An event triggered when the material is disposed\r\n */\r\n public onDisposeObservable = new Observable();\r\n\r\n /**\r\n * An observer which watches for dispose events\r\n */\r\n private _onDisposeObserver: Nullable> = null;\r\n private _onUnBindObservable: Nullable> = null;\r\n\r\n /**\r\n * Called during a dispose event\r\n */\r\n public set onDispose(callback: () => void) {\r\n if (this._onDisposeObserver) {\r\n this.onDisposeObservable.remove(this._onDisposeObserver);\r\n }\r\n this._onDisposeObserver = this.onDisposeObservable.add(callback);\r\n }\r\n\r\n private _onBindObservable: Nullable>;\r\n\r\n /**\r\n * An event triggered when the material is bound\r\n */\r\n public get onBindObservable(): Observable {\r\n if (!this._onBindObservable) {\r\n this._onBindObservable = new Observable();\r\n }\r\n\r\n return this._onBindObservable;\r\n }\r\n\r\n /**\r\n * An observer which watches for bind events\r\n */\r\n private _onBindObserver: Nullable> = null;\r\n\r\n /**\r\n * Called during a bind event\r\n */\r\n public set onBind(callback: (Mesh: AbstractMesh) => void) {\r\n if (this._onBindObserver) {\r\n this.onBindObservable.remove(this._onBindObserver);\r\n }\r\n this._onBindObserver = this.onBindObservable.add(callback);\r\n }\r\n\r\n /**\r\n * An event triggered when the material is unbound\r\n */\r\n public get onUnBindObservable(): Observable {\r\n if (!this._onUnBindObservable) {\r\n this._onUnBindObservable = new Observable();\r\n }\r\n\r\n return this._onUnBindObservable;\r\n }\r\n\r\n protected _onEffectCreatedObservable: Nullable }>>;\r\n\r\n /**\r\n * An event triggered when the effect is (re)created\r\n */\r\n public get onEffectCreatedObservable(): Observable<{ effect: Effect; subMesh: Nullable }> {\r\n if (!this._onEffectCreatedObservable) {\r\n this._onEffectCreatedObservable = new Observable<{ effect: Effect; subMesh: Nullable }>();\r\n }\r\n\r\n return this._onEffectCreatedObservable;\r\n }\r\n\r\n /**\r\n * Stores the value of the alpha mode\r\n */\r\n @serialize(\"alphaMode\")\r\n private _alphaMode: number = Constants.ALPHA_COMBINE;\r\n\r\n /**\r\n * Sets the value of the alpha mode.\r\n *\r\n * | Value | Type | Description |\r\n * | --- | --- | --- |\r\n * | 0 | ALPHA_DISABLE | |\r\n * | 1 | ALPHA_ADD | |\r\n * | 2 | ALPHA_COMBINE | |\r\n * | 3 | ALPHA_SUBTRACT | |\r\n * | 4 | ALPHA_MULTIPLY | |\r\n * | 5 | ALPHA_MAXIMIZED | |\r\n * | 6 | ALPHA_ONEONE | |\r\n * | 7 | ALPHA_PREMULTIPLIED | |\r\n * | 8 | ALPHA_PREMULTIPLIED_PORTERDUFF | |\r\n * | 9 | ALPHA_INTERPOLATE | |\r\n * | 10 | ALPHA_SCREENMODE | |\r\n *\r\n */\r\n public set alphaMode(value: number) {\r\n if (this._alphaMode === value) {\r\n return;\r\n }\r\n this._alphaMode = value;\r\n this.markAsDirty(Material.TextureDirtyFlag);\r\n }\r\n\r\n /**\r\n * Gets the value of the alpha mode\r\n */\r\n public get alphaMode(): number {\r\n return this._alphaMode;\r\n }\r\n\r\n /**\r\n * Stores the state of the need depth pre-pass value\r\n */\r\n @serialize()\r\n private _needDepthPrePass = false;\r\n\r\n /**\r\n * Sets the need depth pre-pass value\r\n */\r\n public set needDepthPrePass(value: boolean) {\r\n if (this._needDepthPrePass === value) {\r\n return;\r\n }\r\n this._needDepthPrePass = value;\r\n if (this._needDepthPrePass) {\r\n this.checkReadyOnEveryCall = true;\r\n }\r\n }\r\n\r\n /**\r\n * Gets the depth pre-pass value\r\n */\r\n public get needDepthPrePass(): boolean {\r\n return this._needDepthPrePass;\r\n }\r\n\r\n /**\r\n * Can this material render to prepass\r\n */\r\n public get isPrePassCapable(): boolean {\r\n return false;\r\n }\r\n\r\n /**\r\n * Specifies if depth writing should be disabled\r\n */\r\n @serialize()\r\n public disableDepthWrite = false;\r\n\r\n /**\r\n * Specifies if color writing should be disabled\r\n */\r\n @serialize()\r\n public disableColorWrite = false;\r\n\r\n /**\r\n * Specifies if depth writing should be forced\r\n */\r\n @serialize()\r\n public forceDepthWrite = false;\r\n\r\n /**\r\n * Specifies the depth function that should be used. 0 means the default engine function\r\n */\r\n @serialize()\r\n public depthFunction = 0;\r\n\r\n /**\r\n * Specifies if there should be a separate pass for culling\r\n */\r\n @serialize()\r\n public separateCullingPass = false;\r\n\r\n /**\r\n * Stores the state specifying if fog should be enabled\r\n */\r\n @serialize(\"fogEnabled\")\r\n private _fogEnabled = true;\r\n\r\n /**\r\n * Sets the state for enabling fog\r\n */\r\n public set fogEnabled(value: boolean) {\r\n if (this._fogEnabled === value) {\r\n return;\r\n }\r\n this._fogEnabled = value;\r\n this.markAsDirty(Material.MiscDirtyFlag);\r\n }\r\n\r\n /**\r\n * Gets the value of the fog enabled state\r\n */\r\n public get fogEnabled(): boolean {\r\n return this._fogEnabled;\r\n }\r\n\r\n /**\r\n * Stores the size of points\r\n */\r\n @serialize()\r\n public pointSize = 1.0;\r\n\r\n /**\r\n * Stores the z offset Factor value\r\n */\r\n @serialize()\r\n public zOffset = 0;\r\n\r\n /**\r\n * Stores the z offset Units value\r\n */\r\n @serialize()\r\n public zOffsetUnits = 0;\r\n\r\n public get wireframe(): boolean {\r\n switch (this._fillMode) {\r\n case Material.WireFrameFillMode:\r\n case Material.LineListDrawMode:\r\n case Material.LineLoopDrawMode:\r\n case Material.LineStripDrawMode:\r\n return true;\r\n }\r\n\r\n return this._scene.forceWireframe;\r\n }\r\n\r\n /**\r\n * Sets the state of wireframe mode\r\n */\r\n public set wireframe(value: boolean) {\r\n this.fillMode = value ? Material.WireFrameFillMode : Material.TriangleFillMode;\r\n }\r\n\r\n /**\r\n * Gets the value specifying if point clouds are enabled\r\n */\r\n @serialize()\r\n public get pointsCloud(): boolean {\r\n switch (this._fillMode) {\r\n case Material.PointFillMode:\r\n case Material.PointListDrawMode:\r\n return true;\r\n }\r\n\r\n return this._scene.forcePointsCloud;\r\n }\r\n\r\n /**\r\n * Sets the state of point cloud mode\r\n */\r\n public set pointsCloud(value: boolean) {\r\n this.fillMode = value ? Material.PointFillMode : Material.TriangleFillMode;\r\n }\r\n\r\n /**\r\n * Gets the material fill mode\r\n */\r\n @serialize()\r\n public get fillMode(): number {\r\n return this._fillMode;\r\n }\r\n\r\n /**\r\n * Sets the material fill mode\r\n */\r\n public set fillMode(value: number) {\r\n if (this._fillMode === value) {\r\n return;\r\n }\r\n\r\n this._fillMode = value;\r\n this.markAsDirty(Material.MiscDirtyFlag);\r\n }\r\n\r\n /**\r\n * Gets or sets the active clipplane 1\r\n */\r\n public clipPlane: Nullable;\r\n\r\n /**\r\n * Gets or sets the active clipplane 2\r\n */\r\n public clipPlane2: Nullable;\r\n\r\n /**\r\n * Gets or sets the active clipplane 3\r\n */\r\n public clipPlane3: Nullable;\r\n\r\n /**\r\n * Gets or sets the active clipplane 4\r\n */\r\n public clipPlane4: Nullable;\r\n\r\n /**\r\n * Gets or sets the active clipplane 5\r\n */\r\n public clipPlane5: Nullable;\r\n\r\n /**\r\n * Gets or sets the active clipplane 6\r\n */\r\n public clipPlane6: Nullable;\r\n\r\n /**\r\n * Gives access to the stencil properties of the material\r\n */\r\n public readonly stencil = new MaterialStencilState();\r\n\r\n /**\r\n * @internal\r\n * Stores the effects for the material\r\n */\r\n protected _materialContext: IMaterialContext | undefined;\r\n\r\n protected _drawWrapper: DrawWrapper;\r\n /** @internal */\r\n public _getDrawWrapper(): DrawWrapper {\r\n return this._drawWrapper;\r\n }\r\n /**\r\n * @internal\r\n */\r\n public _setDrawWrapper(drawWrapper: DrawWrapper) {\r\n this._drawWrapper = drawWrapper;\r\n }\r\n\r\n /**\r\n * Specifies if uniform buffers should be used\r\n */\r\n private _useUBO: boolean = false;\r\n\r\n /**\r\n * Stores a reference to the scene\r\n */\r\n private _scene: Scene;\r\n protected _needToBindSceneUbo: boolean;\r\n\r\n /**\r\n * Stores the fill mode state\r\n */\r\n private _fillMode = Material.TriangleFillMode;\r\n\r\n /**\r\n * Specifies if the depth write state should be cached\r\n */\r\n private _cachedDepthWriteState: boolean = false;\r\n\r\n /**\r\n * Specifies if the color write state should be cached\r\n */\r\n private _cachedColorWriteState: boolean = false;\r\n\r\n /**\r\n * Specifies if the depth function state should be cached\r\n */\r\n private _cachedDepthFunctionState: number = 0;\r\n\r\n /**\r\n * Stores the uniform buffer\r\n * @internal\r\n */\r\n public _uniformBuffer: UniformBuffer;\r\n\r\n /** @internal */\r\n public _indexInSceneMaterialArray = -1;\r\n\r\n /** @internal */\r\n public meshMap: Nullable<{ [id: string]: AbstractMesh | undefined }> = null;\r\n\r\n /** @internal */\r\n public _parentContainer: Nullable = null;\r\n\r\n /** @internal */\r\n public _dirtyCallbacks: { [code: number]: () => void };\r\n\r\n /** @internal */\r\n public _uniformBufferLayoutBuilt = false;\r\n\r\n protected _eventInfo: MaterialPluginCreated &\r\n MaterialPluginDisposed &\r\n MaterialPluginHasTexture &\r\n MaterialPluginIsReadyForSubMesh &\r\n MaterialPluginGetDefineNames &\r\n MaterialPluginPrepareEffect &\r\n MaterialPluginPrepareDefines &\r\n MaterialPluginPrepareUniformBuffer &\r\n MaterialPluginBindForSubMesh &\r\n MaterialPluginGetAnimatables &\r\n MaterialPluginGetActiveTextures &\r\n MaterialPluginFillRenderTargetTextures &\r\n MaterialPluginHasRenderTargetTextures &\r\n MaterialPluginHardBindForSubMesh = {} as any; // will be initialized before each event notification\r\n\r\n /** @internal */\r\n public _callbackPluginEventGeneric: (\r\n id: number,\r\n info:\r\n | MaterialPluginGetActiveTextures\r\n | MaterialPluginGetAnimatables\r\n | MaterialPluginHasTexture\r\n | MaterialPluginDisposed\r\n | MaterialPluginGetDefineNames\r\n | MaterialPluginPrepareEffect\r\n | MaterialPluginPrepareUniformBuffer\r\n ) => void = () => void 0;\r\n /** @internal */\r\n public _callbackPluginEventIsReadyForSubMesh: (eventData: MaterialPluginIsReadyForSubMesh) => void = () => void 0;\r\n /** @internal */\r\n public _callbackPluginEventPrepareDefines: (eventData: MaterialPluginPrepareDefines) => void = () => void 0;\r\n /** @internal */\r\n public _callbackPluginEventPrepareDefinesBeforeAttributes: (eventData: MaterialPluginPrepareDefines) => void = () => void 0;\r\n /** @internal */\r\n public _callbackPluginEventHardBindForSubMesh: (eventData: MaterialPluginHardBindForSubMesh) => void = () => void 0;\r\n /** @internal */\r\n public _callbackPluginEventBindForSubMesh: (eventData: MaterialPluginBindForSubMesh) => void = () => void 0;\r\n /** @internal */\r\n public _callbackPluginEventHasRenderTargetTextures: (eventData: MaterialPluginHasRenderTargetTextures) => void = () => void 0;\r\n /** @internal */\r\n public _callbackPluginEventFillRenderTargetTextures: (eventData: MaterialPluginFillRenderTargetTextures) => void = () => void 0;\r\n\r\n /**\r\n * Creates a material instance\r\n * @param name defines the name of the material\r\n * @param scene defines the scene to reference\r\n * @param doNotAdd specifies if the material should be added to the scene\r\n */\r\n constructor(name: string, scene?: Nullable, doNotAdd?: boolean) {\r\n this.name = name;\r\n const setScene = scene || EngineStore.LastCreatedScene;\r\n if (!setScene) {\r\n return;\r\n }\r\n this._scene = setScene;\r\n this._dirtyCallbacks = {};\r\n\r\n this._dirtyCallbacks[Constants.MATERIAL_TextureDirtyFlag] = this._markAllSubMeshesAsTexturesDirty.bind(this);\r\n this._dirtyCallbacks[Constants.MATERIAL_LightDirtyFlag] = this._markAllSubMeshesAsLightsDirty.bind(this);\r\n this._dirtyCallbacks[Constants.MATERIAL_FresnelDirtyFlag] = this._markAllSubMeshesAsFresnelDirty.bind(this);\r\n this._dirtyCallbacks[Constants.MATERIAL_AttributesDirtyFlag] = this._markAllSubMeshesAsAttributesDirty.bind(this);\r\n this._dirtyCallbacks[Constants.MATERIAL_MiscDirtyFlag] = this._markAllSubMeshesAsMiscDirty.bind(this);\r\n this._dirtyCallbacks[Constants.MATERIAL_PrePassDirtyFlag] = this._markAllSubMeshesAsPrePassDirty.bind(this);\r\n this._dirtyCallbacks[Constants.MATERIAL_AllDirtyFlag] = this._markAllSubMeshesAsAllDirty.bind(this);\r\n\r\n this.id = name || Tools.RandomId();\r\n this.uniqueId = this._scene.getUniqueId();\r\n this._materialContext = this._scene.getEngine().createMaterialContext();\r\n this._drawWrapper = new DrawWrapper(this._scene.getEngine(), false);\r\n this._drawWrapper.materialContext = this._materialContext;\r\n\r\n if (this._scene.useRightHandedSystem) {\r\n this.sideOrientation = Material.ClockWiseSideOrientation;\r\n } else {\r\n this.sideOrientation = Material.CounterClockWiseSideOrientation;\r\n }\r\n\r\n this._uniformBuffer = new UniformBuffer(this._scene.getEngine(), undefined, undefined, name);\r\n this._useUBO = this.getScene().getEngine().supportsUniformBuffers;\r\n\r\n if (!doNotAdd) {\r\n this._scene.addMaterial(this);\r\n }\r\n\r\n if (this._scene.useMaterialMeshMap) {\r\n this.meshMap = {};\r\n }\r\n\r\n Material.OnEventObservable.notifyObservers(this, MaterialPluginEvent.Created);\r\n }\r\n\r\n /**\r\n * Returns a string representation of the current material\r\n * @param fullDetails defines a boolean indicating which levels of logging is desired\r\n * @returns a string with material information\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public toString(fullDetails?: boolean): string {\r\n const ret = \"Name: \" + this.name;\r\n return ret;\r\n }\r\n\r\n /**\r\n * Gets the class name of the material\r\n * @returns a string with the class name of the material\r\n */\r\n public getClassName(): string {\r\n return \"Material\";\r\n }\r\n\r\n /** @internal */\r\n public get _isMaterial() {\r\n return true;\r\n }\r\n\r\n /**\r\n * Specifies if updates for the material been locked\r\n */\r\n public get isFrozen(): boolean {\r\n return this.checkReadyOnlyOnce;\r\n }\r\n\r\n /**\r\n * Locks updates for the material\r\n */\r\n public freeze(): void {\r\n this.markDirty();\r\n this.checkReadyOnlyOnce = true;\r\n }\r\n\r\n /**\r\n * Unlocks updates for the material\r\n */\r\n public unfreeze(): void {\r\n this.markDirty();\r\n this.checkReadyOnlyOnce = false;\r\n }\r\n\r\n /**\r\n * Specifies if the material is ready to be used\r\n * @param mesh defines the mesh to check\r\n * @param useInstances specifies if instances should be used\r\n * @returns a boolean indicating if the material is ready to be used\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public isReady(mesh?: AbstractMesh, useInstances?: boolean): boolean {\r\n return true;\r\n }\r\n\r\n /**\r\n * Specifies that the submesh is ready to be used\r\n * @param mesh defines the mesh to check\r\n * @param subMesh defines which submesh to check\r\n * @param useInstances specifies that instances should be used\r\n * @returns a boolean indicating that the submesh is ready or not\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public isReadyForSubMesh(mesh: AbstractMesh, subMesh: SubMesh, useInstances?: boolean): boolean {\r\n const defines = subMesh.materialDefines;\r\n if (!defines) {\r\n return false;\r\n }\r\n\r\n this._eventInfo.isReadyForSubMesh = true;\r\n this._eventInfo.defines = defines;\r\n this._callbackPluginEventIsReadyForSubMesh(this._eventInfo);\r\n\r\n return this._eventInfo.isReadyForSubMesh;\r\n }\r\n\r\n /**\r\n * Returns the material effect\r\n * @returns the effect associated with the material\r\n */\r\n public getEffect(): Nullable {\r\n return this._drawWrapper.effect;\r\n }\r\n\r\n /**\r\n * Returns the current scene\r\n * @returns a Scene\r\n */\r\n public getScene(): Scene {\r\n return this._scene;\r\n }\r\n\r\n /**\r\n * Enforces alpha test in opaque or blend mode in order to improve the performances of some situations.\r\n */\r\n protected _forceAlphaTest = false;\r\n\r\n /**\r\n * The transparency mode of the material.\r\n */\r\n protected _transparencyMode: Nullable = null;\r\n\r\n /**\r\n * Gets the current transparency mode.\r\n */\r\n @serialize()\r\n public get transparencyMode(): Nullable {\r\n return this._transparencyMode;\r\n }\r\n\r\n /**\r\n * Sets the transparency mode of the material.\r\n *\r\n * | Value | Type | Description |\r\n * | ----- | ----------------------------------- | ----------- |\r\n * | 0 | OPAQUE | |\r\n * | 1 | ALPHATEST | |\r\n * | 2 | ALPHABLEND | |\r\n * | 3 | ALPHATESTANDBLEND | |\r\n *\r\n */\r\n public set transparencyMode(value: Nullable) {\r\n if (this._transparencyMode === value) {\r\n return;\r\n }\r\n\r\n this._transparencyMode = value;\r\n\r\n this._forceAlphaTest = value === Material.MATERIAL_ALPHATESTANDBLEND;\r\n\r\n this._markAllSubMeshesAsTexturesAndMiscDirty();\r\n }\r\n\r\n /**\r\n * Returns true if alpha blending should be disabled.\r\n */\r\n protected get _disableAlphaBlending(): boolean {\r\n return this._transparencyMode === Material.MATERIAL_OPAQUE || this._transparencyMode === Material.MATERIAL_ALPHATEST;\r\n }\r\n\r\n /**\r\n * Specifies whether or not this material should be rendered in alpha blend mode.\r\n * @returns a boolean specifying if alpha blending is needed\r\n */\r\n public needAlphaBlending(): boolean {\r\n if (this._disableAlphaBlending) {\r\n return false;\r\n }\r\n\r\n return this.alpha < 1.0;\r\n }\r\n\r\n /**\r\n * Specifies if the mesh will require alpha blending\r\n * @param mesh defines the mesh to check\r\n * @returns a boolean specifying if alpha blending is needed for the mesh\r\n */\r\n public needAlphaBlendingForMesh(mesh: AbstractMesh): boolean {\r\n if (mesh.visibility < 1.0) {\r\n return true;\r\n }\r\n\r\n if (this._disableAlphaBlending) {\r\n return false;\r\n }\r\n\r\n return mesh.hasVertexAlpha || this.needAlphaBlending();\r\n }\r\n\r\n /**\r\n * Specifies whether or not this material should be rendered in alpha test mode.\r\n * @returns a boolean specifying if an alpha test is needed.\r\n */\r\n public needAlphaTesting(): boolean {\r\n if (this._forceAlphaTest) {\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Specifies if material alpha testing should be turned on for the mesh\r\n * @param mesh defines the mesh to check\r\n */\r\n protected _shouldTurnAlphaTestOn(mesh: AbstractMesh): boolean {\r\n return !this.needAlphaBlendingForMesh(mesh) && this.needAlphaTesting();\r\n }\r\n\r\n /**\r\n * Gets the texture used for the alpha test\r\n * @returns the texture to use for alpha testing\r\n */\r\n public getAlphaTestTexture(): Nullable {\r\n return null;\r\n }\r\n\r\n /**\r\n * Marks the material to indicate that it needs to be re-calculated\r\n * @param forceMaterialDirty - Forces the material to be marked as dirty for all components (same as this.markAsDirty(Material.AllDirtyFlag)). You should use this flag if the material is frozen and you want to force a recompilation.\r\n */\r\n public markDirty(forceMaterialDirty = false): void {\r\n const meshes = this.getScene().meshes;\r\n for (const mesh of meshes) {\r\n if (!mesh.subMeshes) {\r\n continue;\r\n }\r\n for (const subMesh of mesh.subMeshes) {\r\n if (subMesh.getMaterial() !== this) {\r\n continue;\r\n }\r\n\r\n if (!subMesh.effect) {\r\n continue;\r\n }\r\n\r\n subMesh.effect._wasPreviouslyReady = false;\r\n subMesh.effect._wasPreviouslyUsingInstances = null;\r\n subMesh.effect._forceRebindOnNextCall = forceMaterialDirty;\r\n }\r\n }\r\n\r\n if (forceMaterialDirty) {\r\n this.markAsDirty(Material.AllDirtyFlag);\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _preBind(effect?: Effect | DrawWrapper, overrideOrientation: Nullable = null): boolean {\r\n const engine = this._scene.getEngine();\r\n\r\n const orientation = overrideOrientation == null ? this.sideOrientation : overrideOrientation;\r\n const reverse = orientation === Material.ClockWiseSideOrientation;\r\n\r\n engine.enableEffect(effect ? effect : this._getDrawWrapper());\r\n engine.setState(\r\n this.backFaceCulling,\r\n this.zOffset,\r\n false,\r\n reverse,\r\n this._scene._mirroredCameraPosition ? !this.cullBackFaces : this.cullBackFaces,\r\n this.stencil,\r\n this.zOffsetUnits\r\n );\r\n\r\n return reverse;\r\n }\r\n\r\n /**\r\n * Binds the material to the mesh\r\n * @param world defines the world transformation matrix\r\n * @param mesh defines the mesh to bind the material to\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public bind(world: Matrix, mesh?: Mesh): void {}\r\n\r\n /**\r\n * Initializes the uniform buffer layout for the shader.\r\n */\r\n public buildUniformLayout(): void {\r\n const ubo = this._uniformBuffer;\r\n\r\n this._eventInfo.ubo = ubo;\r\n this._callbackPluginEventGeneric(MaterialPluginEvent.PrepareUniformBuffer, this._eventInfo);\r\n\r\n ubo.create();\r\n\r\n this._uniformBufferLayoutBuilt = true;\r\n }\r\n\r\n /**\r\n * Binds the submesh to the material\r\n * @param world defines the world transformation matrix\r\n * @param mesh defines the mesh containing the submesh\r\n * @param subMesh defines the submesh to bind the material to\r\n */\r\n public bindForSubMesh(world: Matrix, mesh: Mesh, subMesh: SubMesh): void {\r\n const effect = subMesh.effect;\r\n if (!effect) {\r\n return;\r\n }\r\n\r\n this._eventInfo.subMesh = subMesh;\r\n this._callbackPluginEventBindForSubMesh(this._eventInfo);\r\n effect._forceRebindOnNextCall = false;\r\n }\r\n\r\n /**\r\n * Binds the world matrix to the material\r\n * @param world defines the world transformation matrix\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public bindOnlyWorldMatrix(world: Matrix): void {}\r\n\r\n /**\r\n * Binds the view matrix to the effect\r\n * @param effect defines the effect to bind the view matrix to\r\n */\r\n public bindView(effect: Effect): void {\r\n if (!this._useUBO) {\r\n effect.setMatrix(\"view\", this.getScene().getViewMatrix());\r\n } else {\r\n this._needToBindSceneUbo = true;\r\n }\r\n }\r\n\r\n /**\r\n * Binds the view projection and projection matrices to the effect\r\n * @param effect defines the effect to bind the view projection and projection matrices to\r\n */\r\n public bindViewProjection(effect: Effect): void {\r\n if (!this._useUBO) {\r\n effect.setMatrix(\"viewProjection\", this.getScene().getTransformMatrix());\r\n effect.setMatrix(\"projection\", this.getScene().getProjectionMatrix());\r\n } else {\r\n this._needToBindSceneUbo = true;\r\n }\r\n }\r\n\r\n /**\r\n * Binds the view matrix to the effect\r\n * @param effect defines the effect to bind the view matrix to\r\n * @param variableName name of the shader variable that will hold the eye position\r\n */\r\n public bindEyePosition(effect: Effect, variableName?: string): void {\r\n if (!this._useUBO) {\r\n this._scene.bindEyePosition(effect, variableName);\r\n } else {\r\n this._needToBindSceneUbo = true;\r\n }\r\n }\r\n\r\n /**\r\n * Processes to execute after binding the material to a mesh\r\n * @param mesh defines the rendered mesh\r\n * @param effect\r\n */\r\n protected _afterBind(mesh?: Mesh, effect: Nullable = null): void {\r\n this._scene._cachedMaterial = this;\r\n if (this._needToBindSceneUbo) {\r\n if (effect) {\r\n this._needToBindSceneUbo = false;\r\n MaterialHelper.BindSceneUniformBuffer(effect, this.getScene().getSceneUniformBuffer());\r\n this._scene.finalizeSceneUbo();\r\n }\r\n }\r\n if (mesh) {\r\n this._scene._cachedVisibility = mesh.visibility;\r\n } else {\r\n this._scene._cachedVisibility = 1;\r\n }\r\n\r\n if (this._onBindObservable && mesh) {\r\n this._onBindObservable.notifyObservers(mesh);\r\n }\r\n\r\n if (this.disableDepthWrite) {\r\n const engine = this._scene.getEngine();\r\n this._cachedDepthWriteState = engine.getDepthWrite();\r\n engine.setDepthWrite(false);\r\n }\r\n\r\n if (this.disableColorWrite) {\r\n const engine = this._scene.getEngine();\r\n this._cachedColorWriteState = engine.getColorWrite();\r\n engine.setColorWrite(false);\r\n }\r\n\r\n if (this.depthFunction !== 0) {\r\n const engine = this._scene.getEngine();\r\n this._cachedDepthFunctionState = engine.getDepthFunction() || 0;\r\n engine.setDepthFunction(this.depthFunction);\r\n }\r\n }\r\n\r\n /**\r\n * Unbinds the material from the mesh\r\n */\r\n public unbind(): void {\r\n if (this._onUnBindObservable) {\r\n this._onUnBindObservable.notifyObservers(this);\r\n }\r\n\r\n if (this.depthFunction !== 0) {\r\n const engine = this._scene.getEngine();\r\n engine.setDepthFunction(this._cachedDepthFunctionState);\r\n }\r\n\r\n if (this.disableDepthWrite) {\r\n const engine = this._scene.getEngine();\r\n engine.setDepthWrite(this._cachedDepthWriteState);\r\n }\r\n\r\n if (this.disableColorWrite) {\r\n const engine = this._scene.getEngine();\r\n engine.setColorWrite(this._cachedColorWriteState);\r\n }\r\n }\r\n\r\n /**\r\n * Returns the animatable textures.\r\n * @returns - Array of animatable textures.\r\n */\r\n public getAnimatables(): IAnimatable[] {\r\n this._eventInfo.animatables = [];\r\n this._callbackPluginEventGeneric(MaterialPluginEvent.GetAnimatables, this._eventInfo);\r\n return this._eventInfo.animatables;\r\n }\r\n\r\n /**\r\n * Gets the active textures from the material\r\n * @returns an array of textures\r\n */\r\n public getActiveTextures(): BaseTexture[] {\r\n this._eventInfo.activeTextures = [];\r\n this._callbackPluginEventGeneric(MaterialPluginEvent.GetActiveTextures, this._eventInfo);\r\n return this._eventInfo.activeTextures;\r\n }\r\n\r\n /**\r\n * Specifies if the material uses a texture\r\n * @param texture defines the texture to check against the material\r\n * @returns a boolean specifying if the material uses the texture\r\n */\r\n public hasTexture(texture: BaseTexture): boolean {\r\n this._eventInfo.hasTexture = false;\r\n this._eventInfo.texture = texture;\r\n this._callbackPluginEventGeneric(MaterialPluginEvent.HasTexture, this._eventInfo);\r\n return this._eventInfo.hasTexture;\r\n }\r\n\r\n /**\r\n * Makes a duplicate of the material, and gives it a new name\r\n * @param name defines the new name for the duplicated material\r\n * @returns the cloned material\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public clone(name: string): Nullable {\r\n return null;\r\n }\r\n\r\n protected _clonePlugins(targetMaterial: Material, rootUrl: string) {\r\n const serializationObject: any = {};\r\n\r\n // Create plugins in targetMaterial in case they don't exist\r\n this._serializePlugins(serializationObject);\r\n\r\n Material._parsePlugins(serializationObject, targetMaterial, this._scene, rootUrl);\r\n\r\n // Copy the properties of the current plugins to the cloned material's plugins\r\n if (this.pluginManager) {\r\n for (const plugin of this.pluginManager._plugins) {\r\n const targetPlugin = targetMaterial.pluginManager!.getPlugin(plugin.name)!;\r\n plugin.copyTo(targetPlugin);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Gets the meshes bound to the material\r\n * @returns an array of meshes bound to the material\r\n */\r\n public getBindedMeshes(): AbstractMesh[] {\r\n if (this.meshMap) {\r\n const result = new Array();\r\n for (const meshId in this.meshMap) {\r\n const mesh = this.meshMap[meshId];\r\n if (mesh) {\r\n result.push(mesh);\r\n }\r\n }\r\n return result;\r\n } else {\r\n const meshes = this._scene.meshes;\r\n return meshes.filter((mesh) => mesh.material === this);\r\n }\r\n }\r\n\r\n /**\r\n * Force shader compilation\r\n * @param mesh defines the mesh associated with this material\r\n * @param onCompiled defines a function to execute once the material is compiled\r\n * @param options defines the options to configure the compilation\r\n * @param onError defines a function to execute if the material fails compiling\r\n */\r\n public forceCompilation(\r\n mesh: AbstractMesh,\r\n onCompiled?: (material: Material) => void,\r\n options?: Partial,\r\n onError?: (reason: string) => void\r\n ): void {\r\n const localOptions = {\r\n clipPlane: false,\r\n useInstances: false,\r\n ...options,\r\n };\r\n\r\n const scene = this.getScene();\r\n const currentHotSwapingState = this.allowShaderHotSwapping;\r\n this.allowShaderHotSwapping = false; // Turned off to let us evaluate the real compilation state\r\n\r\n const checkReady = () => {\r\n if (!this._scene || !this._scene.getEngine()) {\r\n return;\r\n }\r\n\r\n const clipPlaneState = scene.clipPlane;\r\n\r\n if (localOptions.clipPlane) {\r\n scene.clipPlane = new Plane(0, 0, 0, 1);\r\n }\r\n\r\n if (this._storeEffectOnSubMeshes) {\r\n let allDone = true,\r\n lastError = null;\r\n if (mesh.subMeshes) {\r\n const tempSubMesh = new SubMesh(0, 0, 0, 0, 0, mesh, undefined, false, false);\r\n if (tempSubMesh.materialDefines) {\r\n tempSubMesh.materialDefines._renderId = -1;\r\n }\r\n if (!this.isReadyForSubMesh(mesh, tempSubMesh, localOptions.useInstances)) {\r\n if (tempSubMesh.effect && tempSubMesh.effect.getCompilationError() && tempSubMesh.effect.allFallbacksProcessed()) {\r\n lastError = tempSubMesh.effect.getCompilationError();\r\n } else {\r\n allDone = false;\r\n setTimeout(checkReady, 16);\r\n }\r\n }\r\n }\r\n if (allDone) {\r\n this.allowShaderHotSwapping = currentHotSwapingState;\r\n if (lastError) {\r\n if (onError) {\r\n onError(lastError);\r\n }\r\n }\r\n if (onCompiled) {\r\n onCompiled(this);\r\n }\r\n }\r\n } else {\r\n if (this.isReady()) {\r\n this.allowShaderHotSwapping = currentHotSwapingState;\r\n if (onCompiled) {\r\n onCompiled(this);\r\n }\r\n } else {\r\n setTimeout(checkReady, 16);\r\n }\r\n }\r\n\r\n if (localOptions.clipPlane) {\r\n scene.clipPlane = clipPlaneState;\r\n }\r\n };\r\n\r\n checkReady();\r\n }\r\n\r\n /**\r\n * Force shader compilation\r\n * @param mesh defines the mesh that will use this material\r\n * @param options defines additional options for compiling the shaders\r\n * @returns a promise that resolves when the compilation completes\r\n */\r\n public forceCompilationAsync(mesh: AbstractMesh, options?: Partial): Promise {\r\n return new Promise((resolve, reject) => {\r\n this.forceCompilation(\r\n mesh,\r\n () => {\r\n resolve();\r\n },\r\n options,\r\n (reason) => {\r\n reject(reason);\r\n }\r\n );\r\n });\r\n }\r\n\r\n private static readonly _AllDirtyCallBack = (defines: MaterialDefines) => defines.markAllAsDirty();\r\n private static readonly _ImageProcessingDirtyCallBack = (defines: MaterialDefines) => defines.markAsImageProcessingDirty();\r\n private static readonly _TextureDirtyCallBack = (defines: MaterialDefines) => defines.markAsTexturesDirty();\r\n private static readonly _FresnelDirtyCallBack = (defines: MaterialDefines) => defines.markAsFresnelDirty();\r\n private static readonly _MiscDirtyCallBack = (defines: MaterialDefines) => defines.markAsMiscDirty();\r\n private static readonly _PrePassDirtyCallBack = (defines: MaterialDefines) => defines.markAsPrePassDirty();\r\n private static readonly _LightsDirtyCallBack = (defines: MaterialDefines) => defines.markAsLightDirty();\r\n private static readonly _AttributeDirtyCallBack = (defines: MaterialDefines) => defines.markAsAttributesDirty();\r\n\r\n private static _FresnelAndMiscDirtyCallBack = (defines: MaterialDefines) => {\r\n Material._FresnelDirtyCallBack(defines);\r\n Material._MiscDirtyCallBack(defines);\r\n };\r\n\r\n private static _TextureAndMiscDirtyCallBack = (defines: MaterialDefines) => {\r\n Material._TextureDirtyCallBack(defines);\r\n Material._MiscDirtyCallBack(defines);\r\n };\r\n\r\n private static readonly _DirtyCallbackArray: Array<(defines: MaterialDefines) => void> = [];\r\n private static readonly _RunDirtyCallBacks = (defines: MaterialDefines) => {\r\n for (const cb of Material._DirtyCallbackArray) {\r\n cb(defines);\r\n }\r\n };\r\n\r\n /**\r\n * Marks a define in the material to indicate that it needs to be re-computed\r\n * @param flag defines a flag used to determine which parts of the material have to be marked as dirty\r\n */\r\n public markAsDirty(flag: number): void {\r\n if (this.getScene().blockMaterialDirtyMechanism || this._blockDirtyMechanism) {\r\n return;\r\n }\r\n\r\n Material._DirtyCallbackArray.length = 0;\r\n\r\n if (flag & Material.TextureDirtyFlag) {\r\n Material._DirtyCallbackArray.push(Material._TextureDirtyCallBack);\r\n }\r\n\r\n if (flag & Material.LightDirtyFlag) {\r\n Material._DirtyCallbackArray.push(Material._LightsDirtyCallBack);\r\n }\r\n\r\n if (flag & Material.FresnelDirtyFlag) {\r\n Material._DirtyCallbackArray.push(Material._FresnelDirtyCallBack);\r\n }\r\n\r\n if (flag & Material.AttributesDirtyFlag) {\r\n Material._DirtyCallbackArray.push(Material._AttributeDirtyCallBack);\r\n }\r\n\r\n if (flag & Material.MiscDirtyFlag) {\r\n Material._DirtyCallbackArray.push(Material._MiscDirtyCallBack);\r\n }\r\n\r\n if (flag & Material.PrePassDirtyFlag) {\r\n Material._DirtyCallbackArray.push(Material._PrePassDirtyCallBack);\r\n }\r\n\r\n if (Material._DirtyCallbackArray.length) {\r\n this._markAllSubMeshesAsDirty(Material._RunDirtyCallBacks);\r\n }\r\n\r\n this.getScene().resetCachedMaterial();\r\n }\r\n\r\n /**\r\n * Resets the draw wrappers cache for all submeshes that are using this material\r\n */\r\n public resetDrawCache(): void {\r\n const meshes = this.getScene().meshes;\r\n for (const mesh of meshes) {\r\n if (!mesh.subMeshes) {\r\n continue;\r\n }\r\n for (const subMesh of mesh.subMeshes) {\r\n if (subMesh.getMaterial() !== this) {\r\n continue;\r\n }\r\n\r\n subMesh.resetDrawCache();\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Marks all submeshes of a material to indicate that their material defines need to be re-calculated\r\n * @param func defines a function which checks material defines against the submeshes\r\n */\r\n protected _markAllSubMeshesAsDirty(func: (defines: MaterialDefines) => void) {\r\n if (this.getScene().blockMaterialDirtyMechanism || this._blockDirtyMechanism) {\r\n return;\r\n }\r\n\r\n const meshes = this.getScene().meshes;\r\n for (const mesh of meshes) {\r\n if (!mesh.subMeshes) {\r\n continue;\r\n }\r\n for (const subMesh of mesh.subMeshes) {\r\n // We want to skip the submeshes which are not using this material or which have not yet rendered at least once\r\n if (subMesh.getMaterial(false) !== this) {\r\n continue;\r\n }\r\n\r\n for (const drawWrapper of subMesh._drawWrappers) {\r\n if (!drawWrapper || !drawWrapper.defines || !(drawWrapper.defines as MaterialDefines).markAllAsDirty) {\r\n continue;\r\n }\r\n if (this._materialContext === drawWrapper.materialContext) {\r\n func(drawWrapper.defines as MaterialDefines);\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Indicates that the scene should check if the rendering now needs a prepass\r\n */\r\n protected _markScenePrePassDirty() {\r\n if (this.getScene().blockMaterialDirtyMechanism || this._blockDirtyMechanism) {\r\n return;\r\n }\r\n\r\n const prePassRenderer = this.getScene().enablePrePassRenderer();\r\n if (prePassRenderer) {\r\n prePassRenderer.markAsDirty();\r\n }\r\n }\r\n\r\n /**\r\n * Indicates that we need to re-calculated for all submeshes\r\n */\r\n protected _markAllSubMeshesAsAllDirty() {\r\n this._markAllSubMeshesAsDirty(Material._AllDirtyCallBack);\r\n }\r\n\r\n /**\r\n * Indicates that image processing needs to be re-calculated for all submeshes\r\n */\r\n protected _markAllSubMeshesAsImageProcessingDirty() {\r\n this._markAllSubMeshesAsDirty(Material._ImageProcessingDirtyCallBack);\r\n }\r\n\r\n /**\r\n * Indicates that textures need to be re-calculated for all submeshes\r\n */\r\n protected _markAllSubMeshesAsTexturesDirty() {\r\n this._markAllSubMeshesAsDirty(Material._TextureDirtyCallBack);\r\n }\r\n\r\n /**\r\n * Indicates that fresnel needs to be re-calculated for all submeshes\r\n */\r\n protected _markAllSubMeshesAsFresnelDirty() {\r\n this._markAllSubMeshesAsDirty(Material._FresnelDirtyCallBack);\r\n }\r\n\r\n /**\r\n * Indicates that fresnel and misc need to be re-calculated for all submeshes\r\n */\r\n protected _markAllSubMeshesAsFresnelAndMiscDirty() {\r\n this._markAllSubMeshesAsDirty(Material._FresnelAndMiscDirtyCallBack);\r\n }\r\n\r\n /**\r\n * Indicates that lights need to be re-calculated for all submeshes\r\n */\r\n protected _markAllSubMeshesAsLightsDirty() {\r\n this._markAllSubMeshesAsDirty(Material._LightsDirtyCallBack);\r\n }\r\n\r\n /**\r\n * Indicates that attributes need to be re-calculated for all submeshes\r\n */\r\n protected _markAllSubMeshesAsAttributesDirty() {\r\n this._markAllSubMeshesAsDirty(Material._AttributeDirtyCallBack);\r\n }\r\n\r\n /**\r\n * Indicates that misc needs to be re-calculated for all submeshes\r\n */\r\n protected _markAllSubMeshesAsMiscDirty() {\r\n this._markAllSubMeshesAsDirty(Material._MiscDirtyCallBack);\r\n }\r\n\r\n /**\r\n * Indicates that prepass needs to be re-calculated for all submeshes\r\n */\r\n protected _markAllSubMeshesAsPrePassDirty() {\r\n this._markAllSubMeshesAsDirty(Material._MiscDirtyCallBack);\r\n }\r\n\r\n /**\r\n * Indicates that textures and misc need to be re-calculated for all submeshes\r\n */\r\n protected _markAllSubMeshesAsTexturesAndMiscDirty() {\r\n this._markAllSubMeshesAsDirty(Material._TextureAndMiscDirtyCallBack);\r\n }\r\n\r\n protected _checkScenePerformancePriority() {\r\n if (this._scene.performancePriority !== ScenePerformancePriority.BackwardCompatible) {\r\n this.checkReadyOnlyOnce = true;\r\n // re-set the flag when the perf priority changes\r\n const observer = this._scene.onScenePerformancePriorityChangedObservable.addOnce(() => {\r\n this.checkReadyOnlyOnce = false;\r\n });\r\n // if this material is disposed before the scene is disposed, cleanup the observer\r\n this.onDisposeObservable.add(() => {\r\n this._scene.onScenePerformancePriorityChangedObservable.remove(observer);\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Sets the required values to the prepass renderer.\r\n * @param prePassRenderer defines the prepass renderer to setup.\r\n * @returns true if the pre pass is needed.\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public setPrePassRenderer(prePassRenderer: PrePassRenderer): boolean {\r\n // Do Nothing by default\r\n return false;\r\n }\r\n\r\n /**\r\n * Disposes the material\r\n * @param forceDisposeEffect specifies if effects should be forcefully disposed\r\n * @param forceDisposeTextures specifies if textures should be forcefully disposed\r\n * @param notBoundToMesh specifies if the material that is being disposed is known to be not bound to any mesh\r\n */\r\n public dispose(forceDisposeEffect?: boolean, forceDisposeTextures?: boolean, notBoundToMesh?: boolean): void {\r\n const scene = this.getScene();\r\n // Animations\r\n scene.stopAnimation(this);\r\n scene.freeProcessedMaterials();\r\n\r\n // Remove from scene\r\n scene.removeMaterial(this);\r\n\r\n this._eventInfo.forceDisposeTextures = forceDisposeTextures;\r\n this._callbackPluginEventGeneric(MaterialPluginEvent.Disposed, this._eventInfo);\r\n\r\n if (this._parentContainer) {\r\n const index = this._parentContainer.materials.indexOf(this);\r\n if (index > -1) {\r\n this._parentContainer.materials.splice(index, 1);\r\n }\r\n this._parentContainer = null;\r\n }\r\n\r\n if (notBoundToMesh !== true) {\r\n // Remove from meshes\r\n if (this.meshMap) {\r\n for (const meshId in this.meshMap) {\r\n const mesh = this.meshMap[meshId];\r\n if (mesh) {\r\n mesh.material = null; // will set the entry in the map to undefined\r\n this.releaseVertexArrayObject(mesh, forceDisposeEffect);\r\n }\r\n }\r\n } else {\r\n const meshes = scene.meshes;\r\n for (const mesh of meshes) {\r\n if (mesh.material === this && !(mesh as InstancedMesh).sourceMesh) {\r\n mesh.material = null;\r\n this.releaseVertexArrayObject(mesh, forceDisposeEffect);\r\n }\r\n }\r\n }\r\n }\r\n\r\n this._uniformBuffer.dispose();\r\n\r\n // Shader are kept in cache for further use but we can get rid of this by using forceDisposeEffect\r\n if (forceDisposeEffect && this._drawWrapper.effect) {\r\n if (!this._storeEffectOnSubMeshes) {\r\n this._drawWrapper.effect.dispose();\r\n }\r\n\r\n this._drawWrapper.effect = null;\r\n }\r\n\r\n this.metadata = null;\r\n\r\n // Callback\r\n this.onDisposeObservable.notifyObservers(this);\r\n\r\n this.onDisposeObservable.clear();\r\n if (this._onBindObservable) {\r\n this._onBindObservable.clear();\r\n }\r\n\r\n if (this._onUnBindObservable) {\r\n this._onUnBindObservable.clear();\r\n }\r\n\r\n if (this._onEffectCreatedObservable) {\r\n this._onEffectCreatedObservable.clear();\r\n }\r\n\r\n if (this._eventInfo) {\r\n this._eventInfo = {} as any;\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n private releaseVertexArrayObject(mesh: AbstractMesh, forceDisposeEffect?: boolean) {\r\n if ((mesh).geometry) {\r\n const geometry = (mesh).geometry;\r\n if (this._storeEffectOnSubMeshes) {\r\n for (const subMesh of mesh.subMeshes) {\r\n geometry._releaseVertexArrayObject(subMesh.effect);\r\n if (forceDisposeEffect && subMesh.effect) {\r\n subMesh.effect.dispose();\r\n }\r\n }\r\n } else {\r\n geometry._releaseVertexArrayObject(this._drawWrapper.effect);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Serializes this material\r\n * @returns the serialized material object\r\n */\r\n public serialize(): any {\r\n const serializationObject = SerializationHelper.Serialize(this);\r\n\r\n serializationObject.stencil = this.stencil.serialize();\r\n serializationObject.uniqueId = this.uniqueId;\r\n\r\n this._serializePlugins(serializationObject);\r\n\r\n return serializationObject;\r\n }\r\n\r\n protected _serializePlugins(serializationObject: any) {\r\n serializationObject.plugins = {};\r\n\r\n if (this.pluginManager) {\r\n for (const plugin of this.pluginManager._plugins) {\r\n serializationObject.plugins[plugin.getClassName()] = plugin.serialize();\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Creates a material from parsed material data\r\n * @param parsedMaterial defines parsed material data\r\n * @param scene defines the hosting scene\r\n * @param rootUrl defines the root URL to use to load textures\r\n * @returns a new material\r\n */\r\n public static Parse(parsedMaterial: any, scene: Scene, rootUrl: string): Nullable {\r\n if (!parsedMaterial.customType) {\r\n parsedMaterial.customType = \"BABYLON.StandardMaterial\";\r\n } else if (parsedMaterial.customType === \"BABYLON.PBRMaterial\" && parsedMaterial.overloadedAlbedo) {\r\n parsedMaterial.customType = \"BABYLON.LegacyPBRMaterial\";\r\n if (!BABYLON.LegacyPBRMaterial) {\r\n Logger.Error(\"Your scene is trying to load a legacy version of the PBRMaterial, please, include it from the materials library.\");\r\n return null;\r\n }\r\n }\r\n\r\n const materialType = Tools.Instantiate(parsedMaterial.customType);\r\n const material = materialType.Parse(parsedMaterial, scene, rootUrl);\r\n material._loadedUniqueId = parsedMaterial.uniqueId;\r\n\r\n return material;\r\n }\r\n\r\n protected static _parsePlugins(serializationObject: any, material: Material, scene: Scene, rootUrl: string) {\r\n if (!serializationObject.plugins) {\r\n return;\r\n }\r\n\r\n for (const pluginClassName in serializationObject.plugins) {\r\n const pluginData = serializationObject.plugins[pluginClassName];\r\n\r\n let plugin = material.pluginManager?.getPlugin(pluginData.name);\r\n\r\n if (!plugin) {\r\n const pluginClassType = Tools.Instantiate(\"BABYLON.\" + pluginClassName);\r\n if (pluginClassType) {\r\n plugin = new pluginClassType(material);\r\n }\r\n }\r\n\r\n plugin?.parse(pluginData, scene, rootUrl);\r\n }\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\nimport type { Scene } from \"../scene\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { SubMesh } from \"../Meshes/subMesh\";\r\nimport type { BaseTexture } from \"../Materials/Textures/baseTexture\";\r\nimport { Material } from \"../Materials/material\";\r\nimport { Tags } from \"../Misc/tags\";\r\nimport { RegisterClass } from \"../Misc/typeStore\";\r\n\r\n/**\r\n * A multi-material is used to apply different materials to different parts of the same object without the need of\r\n * separate meshes. This can be use to improve performances.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/multiMaterials\r\n */\r\nexport class MultiMaterial extends Material {\r\n private _subMaterials: Nullable[];\r\n /** @internal */\r\n public _waitingSubMaterialsUniqueIds: string[] = [];\r\n\r\n /**\r\n * Gets or Sets the list of Materials used within the multi material.\r\n * They need to be ordered according to the submeshes order in the associated mesh\r\n */\r\n public get subMaterials(): Nullable[] {\r\n return this._subMaterials;\r\n }\r\n\r\n public set subMaterials(value: Nullable[]) {\r\n this._subMaterials = value;\r\n this._hookArray(value);\r\n }\r\n\r\n /**\r\n * Function used to align with Node.getChildren()\r\n * @returns the list of Materials used within the multi material\r\n */\r\n public getChildren(): Nullable[] {\r\n return this.subMaterials;\r\n }\r\n\r\n /**\r\n * Instantiates a new Multi Material\r\n * A multi-material is used to apply different materials to different parts of the same object without the need of\r\n * separate meshes. This can be use to improve performances.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/multiMaterials\r\n * @param name Define the name in the scene\r\n * @param scene Define the scene the material belongs to\r\n */\r\n constructor(name: string, scene?: Scene) {\r\n super(name, scene, true);\r\n\r\n this.getScene().addMultiMaterial(this);\r\n\r\n this.subMaterials = new Array();\r\n\r\n this._storeEffectOnSubMeshes = true; // multimaterial is considered like a push material\r\n }\r\n\r\n private _hookArray(array: Nullable[]): void {\r\n const oldPush = array.push;\r\n array.push = (...items: Nullable[]) => {\r\n const result = oldPush.apply(array, items);\r\n\r\n this._markAllSubMeshesAsTexturesDirty();\r\n\r\n return result;\r\n };\r\n\r\n const oldSplice = array.splice;\r\n array.splice = (index: number, deleteCount?: number) => {\r\n const deleted = oldSplice.apply(array, [index, deleteCount]);\r\n\r\n this._markAllSubMeshesAsTexturesDirty();\r\n\r\n return deleted;\r\n };\r\n }\r\n\r\n /**\r\n * Get one of the submaterial by its index in the submaterials array\r\n * @param index The index to look the sub material at\r\n * @returns The Material if the index has been defined\r\n */\r\n public getSubMaterial(index: number): Nullable {\r\n if (index < 0 || index >= this.subMaterials.length) {\r\n return this.getScene().defaultMaterial;\r\n }\r\n\r\n return this.subMaterials[index];\r\n }\r\n\r\n /**\r\n * Get the list of active textures for the whole sub materials list.\r\n * @returns All the textures that will be used during the rendering\r\n */\r\n public getActiveTextures(): BaseTexture[] {\r\n return super.getActiveTextures().concat(\r\n ...this.subMaterials.map((subMaterial) => {\r\n if (subMaterial) {\r\n return subMaterial.getActiveTextures();\r\n } else {\r\n return [];\r\n }\r\n })\r\n );\r\n }\r\n\r\n /**\r\n * Specifies if any sub-materials of this multi-material use a given texture.\r\n * @param texture Defines the texture to check against this multi-material's sub-materials.\r\n * @returns A boolean specifying if any sub-material of this multi-material uses the texture.\r\n */\r\n public hasTexture(texture: BaseTexture): boolean {\r\n if (super.hasTexture(texture)) {\r\n return true;\r\n }\r\n\r\n for (let i = 0; i < this.subMaterials.length; i++) {\r\n if (this.subMaterials[i]?.hasTexture(texture)) {\r\n return true;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Gets the current class name of the material e.g. \"MultiMaterial\"\r\n * Mainly use in serialization.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"MultiMaterial\";\r\n }\r\n\r\n /**\r\n * Checks if the material is ready to render the requested sub mesh\r\n * @param mesh Define the mesh the submesh belongs to\r\n * @param subMesh Define the sub mesh to look readiness for\r\n * @param useInstances Define whether or not the material is used with instances\r\n * @returns true if ready, otherwise false\r\n */\r\n public isReadyForSubMesh(mesh: AbstractMesh, subMesh: SubMesh, useInstances?: boolean): boolean {\r\n for (let index = 0; index < this.subMaterials.length; index++) {\r\n const subMaterial = this.subMaterials[index];\r\n if (subMaterial) {\r\n if (subMaterial._storeEffectOnSubMeshes) {\r\n if (!subMaterial.isReadyForSubMesh(mesh, subMesh, useInstances)) {\r\n return false;\r\n }\r\n continue;\r\n }\r\n\r\n if (!subMaterial.isReady(mesh)) {\r\n return false;\r\n }\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Clones the current material and its related sub materials\r\n * @param name Define the name of the newly cloned material\r\n * @param cloneChildren Define if submaterial will be cloned or shared with the parent instance\r\n * @returns the cloned material\r\n */\r\n public clone(name: string, cloneChildren?: boolean): MultiMaterial {\r\n const newMultiMaterial = new MultiMaterial(name, this.getScene());\r\n\r\n for (let index = 0; index < this.subMaterials.length; index++) {\r\n let subMaterial: Nullable = null;\r\n const current = this.subMaterials[index];\r\n if (cloneChildren && current) {\r\n subMaterial = current.clone(name + \"-\" + current.name);\r\n } else {\r\n subMaterial = this.subMaterials[index];\r\n }\r\n newMultiMaterial.subMaterials.push(subMaterial);\r\n }\r\n\r\n return newMultiMaterial;\r\n }\r\n\r\n /**\r\n * Serializes the materials into a JSON representation.\r\n * @returns the JSON representation\r\n */\r\n public serialize(): any {\r\n const serializationObject: any = {};\r\n\r\n serializationObject.name = this.name;\r\n serializationObject.id = this.id;\r\n serializationObject.uniqueId = this.uniqueId;\r\n if (Tags) {\r\n serializationObject.tags = Tags.GetTags(this);\r\n }\r\n serializationObject.materialsUniqueIds = [];\r\n serializationObject.materials = [];\r\n\r\n for (let matIndex = 0; matIndex < this.subMaterials.length; matIndex++) {\r\n const subMat = this.subMaterials[matIndex];\r\n\r\n if (subMat) {\r\n serializationObject.materialsUniqueIds.push(subMat.uniqueId);\r\n serializationObject.materials.push(subMat.id);\r\n } else {\r\n serializationObject.materialsUniqueIds.push(null);\r\n serializationObject.materials.push(null);\r\n }\r\n }\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Dispose the material and release its associated resources\r\n * @param forceDisposeEffect Define if we want to force disposing the associated effect (if false the shader is not released and could be reuse later on)\r\n * @param forceDisposeTextures Define if we want to force disposing the associated textures (if false, they will not be disposed and can still be use elsewhere in the app)\r\n * @param forceDisposeChildren Define if we want to force disposing the associated submaterials (if false, they will not be disposed and can still be use elsewhere in the app)\r\n */\r\n public dispose(forceDisposeEffect?: boolean, forceDisposeTextures?: boolean, forceDisposeChildren?: boolean): void {\r\n const scene = this.getScene();\r\n if (!scene) {\r\n return;\r\n }\r\n\r\n if (forceDisposeChildren) {\r\n for (let index = 0; index < this.subMaterials.length; index++) {\r\n const subMaterial = this.subMaterials[index];\r\n if (subMaterial) {\r\n subMaterial.dispose(forceDisposeEffect, forceDisposeTextures);\r\n }\r\n }\r\n }\r\n\r\n const index = scene.multiMaterials.indexOf(this);\r\n if (index >= 0) {\r\n scene.multiMaterials.splice(index, 1);\r\n }\r\n\r\n super.dispose(forceDisposeEffect, forceDisposeTextures);\r\n }\r\n\r\n /**\r\n * Creates a MultiMaterial from parsed MultiMaterial data.\r\n * @param parsedMultiMaterial defines parsed MultiMaterial data.\r\n * @param scene defines the hosting scene\r\n * @returns a new MultiMaterial\r\n */\r\n public static ParseMultiMaterial(parsedMultiMaterial: any, scene: Scene): MultiMaterial {\r\n const multiMaterial = new MultiMaterial(parsedMultiMaterial.name, scene);\r\n\r\n multiMaterial.id = parsedMultiMaterial.id;\r\n multiMaterial._loadedUniqueId = parsedMultiMaterial.uniqueId;\r\n\r\n if (Tags) {\r\n Tags.AddTagsTo(multiMaterial, parsedMultiMaterial.tags);\r\n }\r\n\r\n if (parsedMultiMaterial.materialsUniqueIds) {\r\n multiMaterial._waitingSubMaterialsUniqueIds = parsedMultiMaterial.materialsUniqueIds;\r\n } else {\r\n parsedMultiMaterial.materials.forEach((subMatId: string) => multiMaterial.subMaterials.push(scene.getLastMaterialById(subMatId)));\r\n }\r\n\r\n return multiMaterial;\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.MultiMaterial\", MultiMaterial);\r\n","import type { Mesh } from \"./mesh\";\r\nimport type { Nullable } from \"../types\";\r\n\r\n/**\r\n * Class used to represent a specific level of detail of a mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/LOD\r\n */\r\nexport class MeshLODLevel {\r\n /**\r\n * Creates a new LOD level\r\n * @param distanceOrScreenCoverage defines either the distance or the screen coverage where this level should start being displayed\r\n * @param mesh defines the mesh to use to render this level\r\n */\r\n constructor(\r\n /** Either distance from the center of the object to show this level or the screen coverage if `useLODScreenCoverage` is set to `true` on the mesh*/\r\n public distanceOrScreenCoverage: number,\r\n /** Defines the mesh to use to render this level */\r\n public mesh: Nullable\r\n ) {}\r\n}\r\n","import type { Observer } from \"../Misc/observable\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport { Tools, AsyncLoop } from \"../Misc/tools\";\r\nimport type { IAnimatable } from \"../Animations/animatable.interface\";\r\nimport { DeepCopier } from \"../Misc/deepCopier\";\r\nimport { Tags } from \"../Misc/tags\";\r\nimport type { Coroutine } from \"../Misc/coroutine\";\r\nimport { runCoroutineSync, runCoroutineAsync, createYieldingScheduler } from \"../Misc/coroutine\";\r\nimport type { Nullable, FloatArray, IndicesArray } from \"../types\";\r\nimport { Camera } from \"../Cameras/camera\";\r\nimport type { Scene } from \"../scene\";\r\nimport { ScenePerformancePriority } from \"../scene\";\r\nimport type { Vector4 } from \"../Maths/math.vector\";\r\nimport { Quaternion, Matrix, Vector3, Vector2 } from \"../Maths/math.vector\";\r\nimport type { Color4 } from \"../Maths/math.color\";\r\nimport { Color3 } from \"../Maths/math.color\";\r\nimport type { Engine } from \"../Engines/engine\";\r\nimport { Node } from \"../node\";\r\nimport { VertexBuffer, Buffer } from \"../Buffers/buffer\";\r\nimport type { IGetSetVerticesData } from \"./mesh.vertexData\";\r\nimport { VertexData } from \"./mesh.vertexData\";\r\n\r\nimport { Geometry } from \"./geometry\";\r\nimport { AbstractMesh } from \"./abstractMesh\";\r\nimport { SubMesh } from \"./subMesh\";\r\nimport type { BoundingSphere } from \"../Culling/boundingSphere\";\r\nimport type { Effect } from \"../Materials/effect\";\r\nimport { Material } from \"../Materials/material\";\r\nimport { MultiMaterial } from \"../Materials/multiMaterial\";\r\nimport { SceneLoaderFlags } from \"../Loading/sceneLoaderFlags\";\r\nimport type { Skeleton } from \"../Bones/skeleton\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport { SerializationHelper } from \"../Misc/decorators\";\r\nimport { Logger } from \"../Misc/logger\";\r\nimport { GetClass, RegisterClass } from \"../Misc/typeStore\";\r\nimport { _WarnImport } from \"../Misc/devTools\";\r\nimport { SceneComponentConstants } from \"../sceneComponent\";\r\nimport { MeshLODLevel } from \"./meshLODLevel\";\r\nimport type { Path3D } from \"../Maths/math.path\";\r\nimport type { Plane } from \"../Maths/math.plane\";\r\nimport type { TransformNode } from \"./transformNode\";\r\nimport type { DrawWrapper } from \"../Materials/drawWrapper\";\r\nimport type { PhysicsEngine as PhysicsEngineV1 } from \"../Physics/v1/physicsEngine\";\r\n\r\nimport type { GoldbergMesh } from \"./goldbergMesh\";\r\nimport type { InstancedMesh } from \"./instancedMesh\";\r\nimport type { IPhysicsEnabledObject, PhysicsImpostor } from \"../Physics/v1/physicsImpostor\";\r\nimport type { ICreateCapsuleOptions } from \"./Builders/capsuleBuilder\";\r\nimport type { LinesMesh } from \"./linesMesh\";\r\nimport type { GroundMesh } from \"./groundMesh\";\r\n\r\n/**\r\n * @internal\r\n **/\r\nexport class _CreationDataStorage {\r\n public closePath?: boolean;\r\n public closeArray?: boolean;\r\n public idx: number[];\r\n public dashSize: number;\r\n public gapSize: number;\r\n public path3D: Path3D;\r\n public pathArray: Vector3[][];\r\n public arc: number;\r\n public radius: number;\r\n public cap: number;\r\n public tessellation: number;\r\n}\r\n\r\n/**\r\n * @internal\r\n **/\r\nclass _InstanceDataStorage {\r\n public visibleInstances: any = {};\r\n public batchCache = new _InstancesBatch();\r\n public batchCacheReplacementModeInFrozenMode = new _InstancesBatch();\r\n public instancesBufferSize = 32 * 16 * 4; // let's start with a maximum of 32 instances\r\n public instancesBuffer: Nullable;\r\n public instancesPreviousBuffer: Nullable;\r\n public instancesData: Float32Array;\r\n public instancesPreviousData: Float32Array;\r\n public overridenInstanceCount: number;\r\n public isFrozen: boolean;\r\n public forceMatrixUpdates: boolean;\r\n public previousBatch: Nullable<_InstancesBatch>;\r\n public hardwareInstancedRendering: boolean;\r\n public sideOrientation: number;\r\n public manualUpdate: boolean;\r\n public previousManualUpdate: boolean;\r\n public previousRenderId: number;\r\n public masterMeshPreviousWorldMatrix: Nullable;\r\n}\r\n\r\n/**\r\n * @internal\r\n **/\r\nexport class _InstancesBatch {\r\n public mustReturn = false;\r\n public visibleInstances = new Array>>();\r\n public renderSelf = new Array();\r\n public hardwareInstancedRendering = new Array();\r\n}\r\n\r\n/**\r\n * @internal\r\n **/\r\nclass _ThinInstanceDataStorage {\r\n public instancesCount: number = 0;\r\n public matrixBuffer: Nullable = null;\r\n public previousMatrixBuffer: Nullable = null;\r\n public matrixBufferSize = 32 * 16; // let's start with a maximum of 32 thin instances\r\n public matrixData: Nullable = null;\r\n public previousMatrixData: Nullable;\r\n public boundingVectors: Array = [];\r\n public worldMatrices: Nullable = null;\r\n public masterMeshPreviousWorldMatrix: Nullable;\r\n}\r\n\r\n/**\r\n * @internal\r\n **/\r\nclass _InternalMeshDataInfo {\r\n // Events\r\n public _onBeforeRenderObservable: Nullable>;\r\n public _onBeforeBindObservable: Nullable>;\r\n public _onAfterRenderObservable: Nullable>;\r\n public _onBeforeDrawObservable: Nullable>;\r\n public _onBetweenPassObservable: Nullable>;\r\n\r\n public _areNormalsFrozen: boolean = false; // Will be used by ribbons mainly\r\n public _sourcePositions: Nullable; // Will be used to save original positions when using software skinning\r\n public _sourceNormals: Nullable; // Will be used to save original normals when using software skinning\r\n\r\n // Will be used to save a source mesh reference, If any\r\n public _source: Nullable = null;\r\n // Will be used to for fast cloned mesh lookup\r\n public meshMap: Nullable<{ [id: string]: Mesh | undefined }> = null;\r\n\r\n public _preActivateId: number = -1;\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public _LODLevels = new Array();\r\n /** Alternative definition of LOD level, using screen coverage instead of distance */\r\n public _useLODScreenCoverage: boolean = false;\r\n public _checkReadinessObserver: Nullable>;\r\n\r\n public _onMeshReadyObserverAdded: (observer: Observer) => void;\r\n\r\n public _effectiveMaterial: Nullable = null;\r\n\r\n public _forcedInstanceCount: number = 0;\r\n\r\n public _overrideRenderingFillMode: Nullable = null;\r\n}\r\n\r\n/**\r\n * Class used to represent renderable models\r\n */\r\nexport class Mesh extends AbstractMesh implements IGetSetVerticesData {\r\n // Consts\r\n\r\n /**\r\n * Mesh side orientation : usually the external or front surface\r\n */\r\n public static readonly FRONTSIDE = VertexData.FRONTSIDE;\r\n\r\n /**\r\n * Mesh side orientation : usually the internal or back surface\r\n */\r\n public static readonly BACKSIDE = VertexData.BACKSIDE;\r\n /**\r\n * Mesh side orientation : both internal and external or front and back surfaces\r\n */\r\n public static readonly DOUBLESIDE = VertexData.DOUBLESIDE;\r\n /**\r\n * Mesh side orientation : by default, `FRONTSIDE`\r\n */\r\n public static readonly DEFAULTSIDE = VertexData.DEFAULTSIDE;\r\n /**\r\n * Mesh cap setting : no cap\r\n */\r\n public static readonly NO_CAP = 0;\r\n /**\r\n * Mesh cap setting : one cap at the beginning of the mesh\r\n */\r\n public static readonly CAP_START = 1;\r\n /**\r\n * Mesh cap setting : one cap at the end of the mesh\r\n */\r\n public static readonly CAP_END = 2;\r\n /**\r\n * Mesh cap setting : two caps, one at the beginning and one at the end of the mesh\r\n */\r\n public static readonly CAP_ALL = 3;\r\n /**\r\n * Mesh pattern setting : no flip or rotate\r\n */\r\n public static readonly NO_FLIP = 0;\r\n /**\r\n * Mesh pattern setting : flip (reflect in y axis) alternate tiles on each row or column\r\n */\r\n public static readonly FLIP_TILE = 1;\r\n /**\r\n * Mesh pattern setting : rotate (180degs) alternate tiles on each row or column\r\n */\r\n public static readonly ROTATE_TILE = 2;\r\n /**\r\n * Mesh pattern setting : flip (reflect in y axis) all tiles on alternate rows\r\n */\r\n public static readonly FLIP_ROW = 3;\r\n /**\r\n * Mesh pattern setting : rotate (180degs) all tiles on alternate rows\r\n */\r\n public static readonly ROTATE_ROW = 4;\r\n /**\r\n * Mesh pattern setting : flip and rotate alternate tiles on each row or column\r\n */\r\n public static readonly FLIP_N_ROTATE_TILE = 5;\r\n /**\r\n * Mesh pattern setting : rotate pattern and rotate\r\n */\r\n public static readonly FLIP_N_ROTATE_ROW = 6;\r\n /**\r\n * Mesh tile positioning : part tiles same on left/right or top/bottom\r\n */\r\n public static readonly CENTER = 0;\r\n /**\r\n * Mesh tile positioning : part tiles on left\r\n */\r\n public static readonly LEFT = 1;\r\n /**\r\n * Mesh tile positioning : part tiles on right\r\n */\r\n public static readonly RIGHT = 2;\r\n /**\r\n * Mesh tile positioning : part tiles on top\r\n */\r\n public static readonly TOP = 3;\r\n /**\r\n * Mesh tile positioning : part tiles on bottom\r\n */\r\n public static readonly BOTTOM = 4;\r\n\r\n /**\r\n * Indicates that the instanced meshes should be sorted from back to front before rendering if their material is transparent\r\n */\r\n public static INSTANCEDMESH_SORT_TRANSPARENT = false;\r\n\r\n /**\r\n * Gets the default side orientation.\r\n * @param orientation the orientation to value to attempt to get\r\n * @returns the default orientation\r\n * @internal\r\n */\r\n public static _GetDefaultSideOrientation(orientation?: number): number {\r\n return orientation || Mesh.FRONTSIDE; // works as Mesh.FRONTSIDE is 0\r\n }\r\n\r\n // Internal data\r\n private _internalMeshDataInfo = new _InternalMeshDataInfo();\r\n\r\n /**\r\n * Determines if the LOD levels are intended to be calculated using screen coverage (surface area ratio) instead of distance.\r\n */\r\n public get useLODScreenCoverage() {\r\n return this._internalMeshDataInfo._useLODScreenCoverage;\r\n }\r\n\r\n public set useLODScreenCoverage(value: boolean) {\r\n this._internalMeshDataInfo._useLODScreenCoverage = value;\r\n this._sortLODLevels();\r\n }\r\n\r\n /**\r\n * Will notify when the mesh is completely ready, including materials.\r\n * Observers added to this observable will be removed once triggered\r\n */\r\n public onMeshReadyObservable: Observable;\r\n\r\n public get computeBonesUsingShaders(): boolean {\r\n return this._internalAbstractMeshDataInfo._computeBonesUsingShaders;\r\n }\r\n public set computeBonesUsingShaders(value: boolean) {\r\n if (this._internalAbstractMeshDataInfo._computeBonesUsingShaders === value) {\r\n return;\r\n }\r\n\r\n if (value && this._internalMeshDataInfo._sourcePositions) {\r\n // switch from software to GPU computation: we need to reset the vertex and normal buffers that have been updated by the software process\r\n this.setVerticesData(VertexBuffer.PositionKind, this._internalMeshDataInfo._sourcePositions, true);\r\n if (this._internalMeshDataInfo._sourceNormals) {\r\n this.setVerticesData(VertexBuffer.NormalKind, this._internalMeshDataInfo._sourceNormals, true);\r\n }\r\n\r\n this._internalMeshDataInfo._sourcePositions = null;\r\n this._internalMeshDataInfo._sourceNormals = null;\r\n }\r\n\r\n this._internalAbstractMeshDataInfo._computeBonesUsingShaders = value;\r\n this._markSubMeshesAsAttributesDirty();\r\n }\r\n\r\n /**\r\n * An event triggered before rendering the mesh\r\n */\r\n public get onBeforeRenderObservable(): Observable {\r\n if (!this._internalMeshDataInfo._onBeforeRenderObservable) {\r\n this._internalMeshDataInfo._onBeforeRenderObservable = new Observable();\r\n }\r\n\r\n return this._internalMeshDataInfo._onBeforeRenderObservable;\r\n }\r\n\r\n /**\r\n * An event triggered before binding the mesh\r\n */\r\n public get onBeforeBindObservable(): Observable {\r\n if (!this._internalMeshDataInfo._onBeforeBindObservable) {\r\n this._internalMeshDataInfo._onBeforeBindObservable = new Observable();\r\n }\r\n\r\n return this._internalMeshDataInfo._onBeforeBindObservable;\r\n }\r\n\r\n /**\r\n * An event triggered after rendering the mesh\r\n */\r\n public get onAfterRenderObservable(): Observable {\r\n if (!this._internalMeshDataInfo._onAfterRenderObservable) {\r\n this._internalMeshDataInfo._onAfterRenderObservable = new Observable();\r\n }\r\n\r\n return this._internalMeshDataInfo._onAfterRenderObservable;\r\n }\r\n\r\n /**\r\n * An event triggeredbetween rendering pass when using separateCullingPass = true\r\n */\r\n public get onBetweenPassObservable(): Observable {\r\n if (!this._internalMeshDataInfo._onBetweenPassObservable) {\r\n this._internalMeshDataInfo._onBetweenPassObservable = new Observable();\r\n }\r\n\r\n return this._internalMeshDataInfo._onBetweenPassObservable;\r\n }\r\n\r\n /**\r\n * An event triggered before drawing the mesh\r\n */\r\n public get onBeforeDrawObservable(): Observable {\r\n if (!this._internalMeshDataInfo._onBeforeDrawObservable) {\r\n this._internalMeshDataInfo._onBeforeDrawObservable = new Observable();\r\n }\r\n\r\n return this._internalMeshDataInfo._onBeforeDrawObservable;\r\n }\r\n\r\n private _onBeforeDrawObserver: Nullable>;\r\n\r\n /**\r\n * Sets a callback to call before drawing the mesh. It is recommended to use onBeforeDrawObservable instead\r\n */\r\n public set onBeforeDraw(callback: () => void) {\r\n if (this._onBeforeDrawObserver) {\r\n this.onBeforeDrawObservable.remove(this._onBeforeDrawObserver);\r\n }\r\n this._onBeforeDrawObserver = this.onBeforeDrawObservable.add(callback);\r\n }\r\n\r\n public get hasInstances(): boolean {\r\n return this.instances.length > 0;\r\n }\r\n\r\n public get hasThinInstances(): boolean {\r\n return (this._thinInstanceDataStorage.instancesCount ?? 0) > 0;\r\n }\r\n\r\n // Members\r\n\r\n /**\r\n * Gets the delay loading state of the mesh (when delay loading is turned on)\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/importers/incrementalLoading\r\n */\r\n public delayLoadState = Constants.DELAYLOADSTATE_NONE;\r\n\r\n /**\r\n * Gets the list of instances created from this mesh\r\n * it is not supposed to be modified manually.\r\n * Note also that the order of the InstancedMesh wihin the array is not significant and might change.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/copies/instances\r\n */\r\n public instances = new Array();\r\n\r\n /**\r\n * Gets the file containing delay loading data for this mesh\r\n */\r\n public delayLoadingFile: string;\r\n\r\n /** @internal */\r\n public _binaryInfo: any;\r\n\r\n /**\r\n * User defined function used to change how LOD level selection is done\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/LOD\r\n */\r\n public onLODLevelSelection: (distance: number, mesh: Mesh, selectedLevel: Nullable) => void;\r\n\r\n // Private\r\n /** @internal */\r\n public _creationDataStorage: Nullable<_CreationDataStorage> = null;\r\n\r\n /** @internal */\r\n public _geometry: Nullable = null;\r\n /** @internal */\r\n public _delayInfo: Array;\r\n /** @internal */\r\n public _delayLoadingFunction: (any: any, mesh: Mesh) => void;\r\n\r\n /**\r\n * Gets or sets the forced number of instances to display.\r\n * If 0 (default value), the number of instances is not forced and depends on the draw type\r\n * (regular / instance / thin instances mesh)\r\n */\r\n public get forcedInstanceCount(): number {\r\n return this._internalMeshDataInfo._forcedInstanceCount;\r\n }\r\n\r\n public set forcedInstanceCount(count: number) {\r\n this._internalMeshDataInfo._forcedInstanceCount = count;\r\n }\r\n\r\n /** @internal */\r\n public _instanceDataStorage = new _InstanceDataStorage();\r\n\r\n /** @internal */\r\n public _thinInstanceDataStorage = new _ThinInstanceDataStorage();\r\n\r\n /** @internal */\r\n public _shouldGenerateFlatShading: boolean = false;\r\n\r\n // Use by builder only to know what orientation were the mesh build in.\r\n /** @internal */\r\n public _originalBuilderSideOrientation: number = Mesh.DEFAULTSIDE;\r\n\r\n /**\r\n * Use this property to change the original side orientation defined at construction time\r\n */\r\n public overrideMaterialSideOrientation: Nullable = null;\r\n\r\n /**\r\n * Use this property to override the Material's fillMode value\r\n */\r\n public get overrideRenderingFillMode(): Nullable {\r\n return this._internalMeshDataInfo._overrideRenderingFillMode;\r\n }\r\n\r\n public set overrideRenderingFillMode(fillMode: Nullable) {\r\n this._internalMeshDataInfo._overrideRenderingFillMode = fillMode;\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating whether to render ignoring the active camera's max z setting. (false by default)\r\n * You should not mix meshes that have this property set to true with meshes that have it set to false if they all write\r\n * to the depth buffer, because the z-values are not comparable in the two cases and you will get rendering artifacts if you do.\r\n * You can set the property to true for meshes that do not write to the depth buffer, or set the same value (either false or true) otherwise.\r\n * Note this will reduce performance when set to true.\r\n */\r\n public ignoreCameraMaxZ = false;\r\n\r\n /**\r\n * Gets the source mesh (the one used to clone this one from)\r\n */\r\n public get source(): Nullable {\r\n return this._internalMeshDataInfo._source;\r\n }\r\n\r\n /**\r\n * Gets the list of clones of this mesh\r\n * The scene must have been constructed with useClonedMeshMap=true for this to work!\r\n * Note that useClonedMeshMap=true is the default setting\r\n */\r\n public get cloneMeshMap(): Nullable<{ [id: string]: Mesh | undefined }> {\r\n return this._internalMeshDataInfo.meshMap;\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating that this mesh does not use index buffer\r\n */\r\n public get isUnIndexed(): boolean {\r\n return this._unIndexed;\r\n }\r\n\r\n public set isUnIndexed(value: boolean) {\r\n if (this._unIndexed !== value) {\r\n this._unIndexed = value;\r\n this._markSubMeshesAsAttributesDirty();\r\n }\r\n }\r\n\r\n /** Gets the array buffer used to store the instanced buffer used for instances' world matrices */\r\n public get worldMatrixInstancedBuffer() {\r\n return this._instanceDataStorage.instancesData;\r\n }\r\n\r\n /** Gets the array buffer used to store the instanced buffer used for instances' previous world matrices */\r\n public get previousWorldMatrixInstancedBuffer() {\r\n return this._instanceDataStorage.instancesPreviousData;\r\n }\r\n\r\n /** Gets or sets a boolean indicating that the update of the instance buffer of the world matrices is manual */\r\n public get manualUpdateOfWorldMatrixInstancedBuffer() {\r\n return this._instanceDataStorage.manualUpdate;\r\n }\r\n\r\n public set manualUpdateOfWorldMatrixInstancedBuffer(value: boolean) {\r\n this._instanceDataStorage.manualUpdate = value;\r\n }\r\n\r\n /** Gets or sets a boolean indicating that the update of the instance buffer of the world matrices is manual */\r\n public get manualUpdateOfPreviousWorldMatrixInstancedBuffer() {\r\n return this._instanceDataStorage.previousManualUpdate;\r\n }\r\n\r\n public set manualUpdateOfPreviousWorldMatrixInstancedBuffer(value: boolean) {\r\n this._instanceDataStorage.previousManualUpdate = value;\r\n }\r\n\r\n /** Gets or sets a boolean indicating that the update of the instance buffer of the world matrices must be performed in all cases (and notably even in frozen mode) */\r\n public get forceWorldMatrixInstancedBufferUpdate() {\r\n return this._instanceDataStorage.forceMatrixUpdates;\r\n }\r\n\r\n public set forceWorldMatrixInstancedBufferUpdate(value: boolean) {\r\n this._instanceDataStorage.forceMatrixUpdates = value;\r\n }\r\n\r\n /**\r\n * @constructor\r\n * @param name The value used by scene.getMeshByName() to do a lookup.\r\n * @param scene The scene to add this mesh to.\r\n * @param parent The parent of this mesh, if it has one\r\n * @param source An optional Mesh from which geometry is shared, cloned.\r\n * @param doNotCloneChildren When cloning, skip cloning child meshes of source, default False.\r\n * When false, achieved by calling a clone(), also passing False.\r\n * This will make creation of children, recursive.\r\n * @param clonePhysicsImpostor When cloning, include cloning mesh physics impostor, default True.\r\n */\r\n constructor(\r\n name: string,\r\n scene: Nullable = null,\r\n parent: Nullable = null,\r\n source: Nullable = null,\r\n doNotCloneChildren?: boolean,\r\n clonePhysicsImpostor: boolean = true\r\n ) {\r\n super(name, scene);\r\n\r\n scene = this.getScene();\r\n\r\n this._onBeforeDraw = (isInstance: boolean, world: Matrix, effectiveMaterial?: Material) => {\r\n if (isInstance && effectiveMaterial) {\r\n if (this._uniformBuffer) {\r\n this.transferToEffect(world);\r\n } else {\r\n effectiveMaterial.bindOnlyWorldMatrix(world);\r\n }\r\n }\r\n };\r\n\r\n if (source) {\r\n // Geometry\r\n if (source._geometry) {\r\n source._geometry.applyToMesh(this);\r\n }\r\n\r\n // Deep copy\r\n DeepCopier.DeepCopy(\r\n source,\r\n this,\r\n [\r\n \"name\",\r\n \"material\",\r\n \"skeleton\",\r\n \"instances\",\r\n \"parent\",\r\n \"uniqueId\",\r\n \"source\",\r\n \"metadata\",\r\n \"morphTargetManager\",\r\n \"hasInstances\",\r\n \"worldMatrixInstancedBuffer\",\r\n \"previousWorldMatrixInstancedBuffer\",\r\n \"hasLODLevels\",\r\n \"geometry\",\r\n \"isBlocked\",\r\n \"areNormalsFrozen\",\r\n \"facetNb\",\r\n \"isFacetDataEnabled\",\r\n \"lightSources\",\r\n \"useBones\",\r\n \"isAnInstance\",\r\n \"collider\",\r\n \"edgesRenderer\",\r\n \"forward\",\r\n \"up\",\r\n \"right\",\r\n \"absolutePosition\",\r\n \"absoluteScaling\",\r\n \"absoluteRotationQuaternion\",\r\n \"isWorldMatrixFrozen\",\r\n \"nonUniformScaling\",\r\n \"behaviors\",\r\n \"worldMatrixFromCache\",\r\n \"hasThinInstances\",\r\n \"cloneMeshMap\",\r\n \"hasBoundingInfo\",\r\n \"physicsBody\",\r\n \"physicsImpostor\",\r\n ],\r\n [\"_poseMatrix\"]\r\n );\r\n\r\n // Source mesh\r\n this._internalMeshDataInfo._source = source;\r\n if (scene.useClonedMeshMap) {\r\n if (!source._internalMeshDataInfo.meshMap) {\r\n source._internalMeshDataInfo.meshMap = {};\r\n }\r\n source._internalMeshDataInfo.meshMap[this.uniqueId] = this;\r\n }\r\n\r\n // Construction Params\r\n // Clone parameters allowing mesh to be updated in case of parametric shapes.\r\n this._originalBuilderSideOrientation = source._originalBuilderSideOrientation;\r\n this._creationDataStorage = source._creationDataStorage;\r\n\r\n // Animation ranges\r\n if (source._ranges) {\r\n const ranges = source._ranges;\r\n for (const name in ranges) {\r\n if (!Object.prototype.hasOwnProperty.call(ranges, name)) {\r\n continue;\r\n }\r\n\r\n if (!ranges[name]) {\r\n continue;\r\n }\r\n\r\n this.createAnimationRange(name, ranges[name]!.from, ranges[name]!.to);\r\n }\r\n }\r\n\r\n // Metadata\r\n if (source.metadata && source.metadata.clone) {\r\n this.metadata = source.metadata.clone();\r\n } else {\r\n this.metadata = source.metadata;\r\n }\r\n this._internalMetadata = source._internalMetadata;\r\n\r\n // Tags\r\n if (Tags && Tags.HasTags(source)) {\r\n Tags.AddTagsTo(this, Tags.GetTags(source, true));\r\n }\r\n\r\n // Enabled. We shouldn't need to check the source's ancestors, as this mesh\r\n // will have the same ones.\r\n this.setEnabled(source.isEnabled(false));\r\n\r\n // Parent\r\n this.parent = source.parent;\r\n\r\n // Pivot\r\n this.setPivotMatrix(source.getPivotMatrix());\r\n\r\n this.id = name + \".\" + source.id;\r\n\r\n // Material\r\n this.material = source.material;\r\n\r\n if (!doNotCloneChildren) {\r\n // Children\r\n const directDescendants = source.getDescendants(true);\r\n for (let index = 0; index < directDescendants.length; index++) {\r\n const child = directDescendants[index];\r\n\r\n if ((child).clone) {\r\n (child).clone(name + \".\" + child.name, this);\r\n }\r\n }\r\n }\r\n\r\n // Morphs\r\n if (source.morphTargetManager) {\r\n this.morphTargetManager = source.morphTargetManager;\r\n }\r\n\r\n // Physics clone\r\n if (scene.getPhysicsEngine) {\r\n const physicsEngine = scene.getPhysicsEngine();\r\n if (clonePhysicsImpostor && physicsEngine) {\r\n if (physicsEngine.getPluginVersion() === 1) {\r\n const impostor = (physicsEngine as PhysicsEngineV1).getImpostorForPhysicsObject(source);\r\n if (impostor) {\r\n this.physicsImpostor = impostor.clone(this);\r\n }\r\n } else if (physicsEngine.getPluginVersion() === 2) {\r\n if (source.physicsBody) {\r\n source.physicsBody.clone(this);\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Particles\r\n for (let index = 0; index < scene.particleSystems.length; index++) {\r\n const system = scene.particleSystems[index];\r\n\r\n if (system.emitter === source) {\r\n system.clone(system.name, this);\r\n }\r\n }\r\n\r\n // Skeleton\r\n this.skeleton = source.skeleton;\r\n\r\n this.refreshBoundingInfo(true, true);\r\n this.computeWorldMatrix(true);\r\n }\r\n\r\n // Parent\r\n if (parent !== null) {\r\n this.parent = parent;\r\n }\r\n\r\n this._instanceDataStorage.hardwareInstancedRendering = this.getEngine().getCaps().instancedArrays;\r\n\r\n this._internalMeshDataInfo._onMeshReadyObserverAdded = (observer: Observer) => {\r\n // only notify once! then unregister the observer\r\n observer.unregisterOnNextCall = true;\r\n if (this.isReady(true)) {\r\n this.onMeshReadyObservable.notifyObservers(this);\r\n } else {\r\n if (!this._internalMeshDataInfo._checkReadinessObserver) {\r\n this._internalMeshDataInfo._checkReadinessObserver = this._scene.onBeforeRenderObservable.add(() => {\r\n // check for complete readiness\r\n if (this.isReady(true)) {\r\n this._scene.onBeforeRenderObservable.remove(this._internalMeshDataInfo._checkReadinessObserver);\r\n this._internalMeshDataInfo._checkReadinessObserver = null;\r\n this.onMeshReadyObservable.notifyObservers(this);\r\n }\r\n });\r\n }\r\n }\r\n };\r\n\r\n this.onMeshReadyObservable = new Observable(this._internalMeshDataInfo._onMeshReadyObserverAdded);\r\n\r\n if (source) {\r\n source.onClonedObservable.notifyObservers(this);\r\n }\r\n }\r\n\r\n public instantiateHierarchy(\r\n newParent: Nullable = null,\r\n options?: { doNotInstantiate: boolean | ((node: TransformNode) => boolean) },\r\n onNewNodeCreated?: (source: TransformNode, clone: TransformNode) => void\r\n ): Nullable {\r\n const instance =\r\n this.getTotalVertices() === 0 || (options && options.doNotInstantiate && (options.doNotInstantiate === true || options.doNotInstantiate(this)))\r\n ? this.clone(\"Clone of \" + (this.name || this.id), newParent || this.parent, true)\r\n : this.createInstance(\"instance of \" + (this.name || this.id));\r\n\r\n instance.parent = newParent || this.parent;\r\n instance.position = this.position.clone();\r\n instance.scaling = this.scaling.clone();\r\n if (this.rotationQuaternion) {\r\n instance.rotationQuaternion = this.rotationQuaternion.clone();\r\n } else {\r\n instance.rotation = this.rotation.clone();\r\n }\r\n\r\n if (onNewNodeCreated) {\r\n onNewNodeCreated(this, instance);\r\n }\r\n\r\n for (const child of this.getChildTransformNodes(true)) {\r\n // instancedMesh should have a different sourced mesh\r\n if (child.getClassName() === \"InstancedMesh\" && instance.getClassName() === \"Mesh\" && (child as InstancedMesh).sourceMesh === this) {\r\n (child as InstancedMesh).instantiateHierarchy(\r\n instance,\r\n {\r\n doNotInstantiate: (options && options.doNotInstantiate) || false,\r\n newSourcedMesh: instance as Mesh,\r\n },\r\n onNewNodeCreated\r\n );\r\n } else {\r\n child.instantiateHierarchy(instance, options, onNewNodeCreated);\r\n }\r\n }\r\n\r\n return instance;\r\n }\r\n\r\n /**\r\n * Gets the class name\r\n * @returns the string \"Mesh\".\r\n */\r\n public getClassName(): string {\r\n return \"Mesh\";\r\n }\r\n\r\n /** @internal */\r\n public get _isMesh() {\r\n return true;\r\n }\r\n\r\n /**\r\n * Returns a description of this mesh\r\n * @param fullDetails define if full details about this mesh must be used\r\n * @returns a descriptive string representing this mesh\r\n */\r\n public toString(fullDetails?: boolean): string {\r\n let ret = super.toString(fullDetails);\r\n ret += \", n vertices: \" + this.getTotalVertices();\r\n ret += \", parent: \" + (this._waitingParentId ? this._waitingParentId : this.parent ? this.parent.name : \"NONE\");\r\n\r\n if (this.animations) {\r\n for (let i = 0; i < this.animations.length; i++) {\r\n ret += \", animation[0]: \" + this.animations[i].toString(fullDetails);\r\n }\r\n }\r\n\r\n if (fullDetails) {\r\n if (this._geometry) {\r\n const ib = this.getIndices();\r\n const vb = this.getVerticesData(VertexBuffer.PositionKind);\r\n\r\n if (vb && ib) {\r\n ret += \", flat shading: \" + (vb.length / 3 === ib.length ? \"YES\" : \"NO\");\r\n }\r\n } else {\r\n ret += \", flat shading: UNKNOWN\";\r\n }\r\n }\r\n return ret;\r\n }\r\n\r\n /** @internal */\r\n public _unBindEffect() {\r\n super._unBindEffect();\r\n\r\n for (const instance of this.instances) {\r\n instance._unBindEffect();\r\n }\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if this mesh has LOD\r\n */\r\n public get hasLODLevels(): boolean {\r\n return this._internalMeshDataInfo._LODLevels.length > 0;\r\n }\r\n\r\n /**\r\n * Gets the list of MeshLODLevel associated with the current mesh\r\n * @returns an array of MeshLODLevel\r\n */\r\n public getLODLevels(): MeshLODLevel[] {\r\n return this._internalMeshDataInfo._LODLevels;\r\n }\r\n\r\n private _sortLODLevels(): void {\r\n const sortingOrderFactor = this._internalMeshDataInfo._useLODScreenCoverage ? -1 : 1;\r\n this._internalMeshDataInfo._LODLevels.sort((a, b) => {\r\n if (a.distanceOrScreenCoverage < b.distanceOrScreenCoverage) {\r\n return sortingOrderFactor;\r\n }\r\n if (a.distanceOrScreenCoverage > b.distanceOrScreenCoverage) {\r\n return -sortingOrderFactor;\r\n }\r\n\r\n return 0;\r\n });\r\n }\r\n\r\n /**\r\n * Add a mesh as LOD level triggered at the given distance.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/LOD\r\n * @param distanceOrScreenCoverage Either distance from the center of the object to show this level or the screen coverage if `useScreenCoverage` is set to `true`.\r\n * If screen coverage, value is a fraction of the screen's total surface, between 0 and 1.\r\n * Example Playground for distance https://playground.babylonjs.com/#QE7KM#197\r\n * Example Playground for screen coverage https://playground.babylonjs.com/#QE7KM#196\r\n * @param mesh The mesh to be added as LOD level (can be null)\r\n * @returns This mesh (for chaining)\r\n */\r\n public addLODLevel(distanceOrScreenCoverage: number, mesh: Nullable): Mesh {\r\n if (mesh && mesh._masterMesh) {\r\n Logger.Warn(\"You cannot use a mesh as LOD level twice\");\r\n return this;\r\n }\r\n\r\n const level = new MeshLODLevel(distanceOrScreenCoverage, mesh);\r\n this._internalMeshDataInfo._LODLevels.push(level);\r\n\r\n if (mesh) {\r\n mesh._masterMesh = this;\r\n }\r\n\r\n this._sortLODLevels();\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns the LOD level mesh at the passed distance or null if not found.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/LOD\r\n * @param distance The distance from the center of the object to show this level\r\n * @returns a Mesh or `null`\r\n */\r\n public getLODLevelAtDistance(distance: number): Nullable {\r\n const internalDataInfo = this._internalMeshDataInfo;\r\n for (let index = 0; index < internalDataInfo._LODLevels.length; index++) {\r\n const level = internalDataInfo._LODLevels[index];\r\n\r\n if (level.distanceOrScreenCoverage === distance) {\r\n return level.mesh;\r\n }\r\n }\r\n return null;\r\n }\r\n\r\n /**\r\n * Remove a mesh from the LOD array\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/LOD\r\n * @param mesh defines the mesh to be removed\r\n * @returns This mesh (for chaining)\r\n */\r\n public removeLODLevel(mesh: Nullable): Mesh {\r\n const internalDataInfo = this._internalMeshDataInfo;\r\n for (let index = 0; index < internalDataInfo._LODLevels.length; index++) {\r\n if (internalDataInfo._LODLevels[index].mesh === mesh) {\r\n internalDataInfo._LODLevels.splice(index, 1);\r\n if (mesh) {\r\n mesh._masterMesh = null;\r\n }\r\n }\r\n }\r\n\r\n this._sortLODLevels();\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns the registered LOD mesh distant from the parameter `camera` position if any, else returns the current mesh.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/LOD\r\n * @param camera defines the camera to use to compute distance\r\n * @param boundingSphere defines a custom bounding sphere to use instead of the one from this mesh\r\n * @returns This mesh (for chaining)\r\n */\r\n public getLOD(camera: Camera, boundingSphere?: BoundingSphere): Nullable {\r\n const internalDataInfo = this._internalMeshDataInfo;\r\n if (!internalDataInfo._LODLevels || internalDataInfo._LODLevels.length === 0) {\r\n return this;\r\n }\r\n\r\n const bSphere = boundingSphere || this.getBoundingInfo().boundingSphere;\r\n\r\n const distanceToCamera = camera.mode === Camera.ORTHOGRAPHIC_CAMERA ? camera.minZ : bSphere.centerWorld.subtract(camera.globalPosition).length();\r\n let compareValue = distanceToCamera;\r\n let compareSign = 1;\r\n\r\n if (internalDataInfo._useLODScreenCoverage) {\r\n const screenArea = camera.screenArea;\r\n let meshArea = (bSphere.radiusWorld * camera.minZ) / distanceToCamera;\r\n meshArea = meshArea * meshArea * Math.PI;\r\n compareValue = meshArea / screenArea;\r\n compareSign = -1;\r\n }\r\n\r\n if (compareSign * internalDataInfo._LODLevels[internalDataInfo._LODLevels.length - 1].distanceOrScreenCoverage > compareSign * compareValue) {\r\n if (this.onLODLevelSelection) {\r\n this.onLODLevelSelection(compareValue, this, this);\r\n }\r\n return this;\r\n }\r\n\r\n for (let index = 0; index < internalDataInfo._LODLevels.length; index++) {\r\n const level = internalDataInfo._LODLevels[index];\r\n\r\n if (compareSign * level.distanceOrScreenCoverage < compareSign * compareValue) {\r\n if (level.mesh) {\r\n if (level.mesh.delayLoadState === Constants.DELAYLOADSTATE_NOTLOADED) {\r\n level.mesh._checkDelayState();\r\n return this;\r\n }\r\n\r\n if (level.mesh.delayLoadState === Constants.DELAYLOADSTATE_LOADING) {\r\n return this;\r\n }\r\n\r\n level.mesh._preActivate();\r\n level.mesh._updateSubMeshesBoundingInfo(this.worldMatrixFromCache);\r\n }\r\n\r\n if (this.onLODLevelSelection) {\r\n this.onLODLevelSelection(compareValue, this, level.mesh);\r\n }\r\n\r\n return level.mesh;\r\n }\r\n }\r\n\r\n if (this.onLODLevelSelection) {\r\n this.onLODLevelSelection(compareValue, this, this);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Gets the mesh internal Geometry object\r\n */\r\n public get geometry(): Nullable {\r\n return this._geometry;\r\n }\r\n\r\n /**\r\n * Returns the total number of vertices within the mesh geometry or zero if the mesh has no geometry.\r\n * @returns the total number of vertices\r\n */\r\n public getTotalVertices(): number {\r\n if (this._geometry === null || this._geometry === undefined) {\r\n return 0;\r\n }\r\n return this._geometry.getTotalVertices();\r\n }\r\n\r\n /**\r\n * Returns the content of an associated vertex buffer\r\n * @param kind defines which buffer to read from (positions, indices, normals, etc). Possible `kind` values :\r\n * - VertexBuffer.PositionKind\r\n * - VertexBuffer.UVKind\r\n * - VertexBuffer.UV2Kind\r\n * - VertexBuffer.UV3Kind\r\n * - VertexBuffer.UV4Kind\r\n * - VertexBuffer.UV5Kind\r\n * - VertexBuffer.UV6Kind\r\n * - VertexBuffer.ColorKind\r\n * - VertexBuffer.MatricesIndicesKind\r\n * - VertexBuffer.MatricesIndicesExtraKind\r\n * - VertexBuffer.MatricesWeightsKind\r\n * - VertexBuffer.MatricesWeightsExtraKind\r\n * @param copyWhenShared defines a boolean indicating that if the mesh geometry is shared among some other meshes, the returned array is a copy of the internal one\r\n * @param forceCopy defines a boolean forcing the copy of the buffer no matter what the value of copyWhenShared is\r\n * @param bypassInstanceData defines a boolean indicating that the function should not take into account the instance data (applies only if the mesh has instances). Default: false\r\n * @returns a FloatArray or null if the mesh has no geometry or no vertex buffer for this kind.\r\n */\r\n public getVerticesData(kind: string, copyWhenShared?: boolean, forceCopy?: boolean, bypassInstanceData?: boolean): Nullable {\r\n if (!this._geometry) {\r\n return null;\r\n }\r\n let data = bypassInstanceData\r\n ? undefined\r\n : this._userInstancedBuffersStorage?.vertexBuffers[kind]?.getFloatData(\r\n this.instances.length + 1, // +1 because the master mesh is not included in the instances array\r\n forceCopy || (copyWhenShared && this._geometry.meshes.length !== 1)\r\n );\r\n if (!data) {\r\n data = this._geometry.getVerticesData(kind, copyWhenShared, forceCopy);\r\n }\r\n return data;\r\n }\r\n\r\n /**\r\n * Returns the mesh VertexBuffer object from the requested `kind`\r\n * @param kind defines which buffer to read from (positions, indices, normals, etc). Possible `kind` values :\r\n * - VertexBuffer.PositionKind\r\n * - VertexBuffer.NormalKind\r\n * - VertexBuffer.UVKind\r\n * - VertexBuffer.UV2Kind\r\n * - VertexBuffer.UV3Kind\r\n * - VertexBuffer.UV4Kind\r\n * - VertexBuffer.UV5Kind\r\n * - VertexBuffer.UV6Kind\r\n * - VertexBuffer.ColorKind\r\n * - VertexBuffer.MatricesIndicesKind\r\n * - VertexBuffer.MatricesIndicesExtraKind\r\n * - VertexBuffer.MatricesWeightsKind\r\n * - VertexBuffer.MatricesWeightsExtraKind\r\n * @param bypassInstanceData defines a boolean indicating that the function should not take into account the instance data (applies only if the mesh has instances). Default: false\r\n * @returns a FloatArray or null if the mesh has no vertex buffer for this kind.\r\n */\r\n public getVertexBuffer(kind: string, bypassInstanceData?: boolean): Nullable {\r\n if (!this._geometry) {\r\n return null;\r\n }\r\n\r\n return (bypassInstanceData ? undefined : this._userInstancedBuffersStorage?.vertexBuffers[kind]) ?? this._geometry.getVertexBuffer(kind);\r\n }\r\n\r\n /**\r\n * Tests if a specific vertex buffer is associated with this mesh\r\n * @param kind defines which buffer to check (positions, indices, normals, etc). Possible `kind` values :\r\n * - VertexBuffer.PositionKind\r\n * - VertexBuffer.NormalKind\r\n * - VertexBuffer.UVKind\r\n * - VertexBuffer.UV2Kind\r\n * - VertexBuffer.UV3Kind\r\n * - VertexBuffer.UV4Kind\r\n * - VertexBuffer.UV5Kind\r\n * - VertexBuffer.UV6Kind\r\n * - VertexBuffer.ColorKind\r\n * - VertexBuffer.MatricesIndicesKind\r\n * - VertexBuffer.MatricesIndicesExtraKind\r\n * - VertexBuffer.MatricesWeightsKind\r\n * - VertexBuffer.MatricesWeightsExtraKind\r\n * @param bypassInstanceData defines a boolean indicating that the function should not take into account the instance data (applies only if the mesh has instances). Default: false\r\n * @returns a boolean\r\n */\r\n public isVerticesDataPresent(kind: string, bypassInstanceData?: boolean): boolean {\r\n if (!this._geometry) {\r\n if (this._delayInfo) {\r\n return this._delayInfo.indexOf(kind) !== -1;\r\n }\r\n return false;\r\n }\r\n return (!bypassInstanceData && this._userInstancedBuffersStorage?.vertexBuffers[kind] !== undefined) || this._geometry.isVerticesDataPresent(kind);\r\n }\r\n\r\n /**\r\n * Returns a boolean defining if the vertex data for the requested `kind` is updatable.\r\n * @param kind defines which buffer to check (positions, indices, normals, etc). Possible `kind` values :\r\n * - VertexBuffer.PositionKind\r\n * - VertexBuffer.UVKind\r\n * - VertexBuffer.UV2Kind\r\n * - VertexBuffer.UV3Kind\r\n * - VertexBuffer.UV4Kind\r\n * - VertexBuffer.UV5Kind\r\n * - VertexBuffer.UV6Kind\r\n * - VertexBuffer.ColorKind\r\n * - VertexBuffer.MatricesIndicesKind\r\n * - VertexBuffer.MatricesIndicesExtraKind\r\n * - VertexBuffer.MatricesWeightsKind\r\n * - VertexBuffer.MatricesWeightsExtraKind\r\n * @param bypassInstanceData defines a boolean indicating that the function should not take into account the instance data (applies only if the mesh has instances). Default: false\r\n * @returns a boolean\r\n */\r\n public isVertexBufferUpdatable(kind: string, bypassInstanceData?: boolean): boolean {\r\n if (!this._geometry) {\r\n if (this._delayInfo) {\r\n return this._delayInfo.indexOf(kind) !== -1;\r\n }\r\n return false;\r\n }\r\n if (!bypassInstanceData) {\r\n const buffer = this._userInstancedBuffersStorage?.vertexBuffers[kind];\r\n if (buffer) {\r\n return buffer.isUpdatable();\r\n }\r\n }\r\n return this._geometry.isVertexBufferUpdatable(kind);\r\n }\r\n\r\n /**\r\n * Returns a string which contains the list of existing `kinds` of Vertex Data associated with this mesh.\r\n * @param bypassInstanceData defines a boolean indicating that the function should not take into account the instance data (applies only if the mesh has instances). Default: false\r\n * @returns an array of strings\r\n */\r\n public getVerticesDataKinds(bypassInstanceData?: boolean): string[] {\r\n if (!this._geometry) {\r\n const result = new Array();\r\n if (this._delayInfo) {\r\n this._delayInfo.forEach(function (kind) {\r\n result.push(kind);\r\n });\r\n }\r\n return result;\r\n }\r\n const kinds = this._geometry.getVerticesDataKinds();\r\n if (!bypassInstanceData && this._userInstancedBuffersStorage) {\r\n for (const kind in this._userInstancedBuffersStorage.vertexBuffers) {\r\n if (kinds.indexOf(kind) === -1) {\r\n kinds.push(kind);\r\n }\r\n }\r\n }\r\n return kinds;\r\n }\r\n\r\n /**\r\n * Returns a positive integer : the total number of indices in this mesh geometry.\r\n * @returns the numner of indices or zero if the mesh has no geometry.\r\n */\r\n public getTotalIndices(): number {\r\n if (!this._geometry) {\r\n return 0;\r\n }\r\n return this._geometry.getTotalIndices();\r\n }\r\n\r\n /**\r\n * Returns an array of integers or a typed array (Int32Array, Uint32Array, Uint16Array) populated with the mesh indices.\r\n * @param copyWhenShared If true (default false) and and if the mesh geometry is shared among some other meshes, the returned array is a copy of the internal one.\r\n * @param forceCopy defines a boolean indicating that the returned array must be cloned upon returning it\r\n * @returns the indices array or an empty array if the mesh has no geometry\r\n */\r\n public getIndices(copyWhenShared?: boolean, forceCopy?: boolean): Nullable {\r\n if (!this._geometry) {\r\n return [];\r\n }\r\n return this._geometry.getIndices(copyWhenShared, forceCopy);\r\n }\r\n\r\n public get isBlocked(): boolean {\r\n return this._masterMesh !== null && this._masterMesh !== undefined;\r\n }\r\n\r\n /**\r\n * Determine if the current mesh is ready to be rendered\r\n * @param completeCheck defines if a complete check (including materials and lights) has to be done (false by default)\r\n * @param forceInstanceSupport will check if the mesh will be ready when used with instances (false by default)\r\n * @returns true if all associated assets are ready (material, textures, shaders)\r\n */\r\n public isReady(completeCheck = false, forceInstanceSupport = false): boolean {\r\n if (this.delayLoadState === Constants.DELAYLOADSTATE_LOADING) {\r\n return false;\r\n }\r\n\r\n if (!super.isReady(completeCheck)) {\r\n return false;\r\n }\r\n\r\n if (!this.subMeshes || this.subMeshes.length === 0) {\r\n return true;\r\n }\r\n\r\n if (!completeCheck) {\r\n return true;\r\n }\r\n\r\n const engine = this.getEngine();\r\n const scene = this.getScene();\r\n const hardwareInstancedRendering = forceInstanceSupport || (engine.getCaps().instancedArrays && (this.instances.length > 0 || this.hasThinInstances));\r\n\r\n this.computeWorldMatrix();\r\n\r\n const mat = this.material || scene.defaultMaterial;\r\n if (mat) {\r\n if (mat._storeEffectOnSubMeshes) {\r\n for (const subMesh of this.subMeshes) {\r\n const effectiveMaterial = subMesh.getMaterial();\r\n if (effectiveMaterial) {\r\n if (effectiveMaterial._storeEffectOnSubMeshes) {\r\n if (!effectiveMaterial.isReadyForSubMesh(this, subMesh, hardwareInstancedRendering)) {\r\n return false;\r\n }\r\n } else {\r\n if (!effectiveMaterial.isReady(this, hardwareInstancedRendering)) {\r\n return false;\r\n }\r\n }\r\n }\r\n }\r\n } else {\r\n if (!mat.isReady(this, hardwareInstancedRendering)) {\r\n return false;\r\n }\r\n }\r\n }\r\n\r\n // Shadows\r\n const currentRenderPassId = engine.currentRenderPassId;\r\n for (const light of this.lightSources) {\r\n const generators = light.getShadowGenerators();\r\n\r\n if (!generators) {\r\n continue;\r\n }\r\n\r\n const iterator = generators.values();\r\n for (let key = iterator.next(); key.done !== true; key = iterator.next()) {\r\n const generator = key.value;\r\n if (generator && (!generator.getShadowMap()?.renderList || (generator.getShadowMap()?.renderList && generator.getShadowMap()?.renderList?.indexOf(this) !== -1))) {\r\n const shadowMap = generator.getShadowMap()!;\r\n const renderPassIds = shadowMap.renderPassIds ?? [engine.currentRenderPassId];\r\n for (let p = 0; p < renderPassIds.length; ++p) {\r\n engine.currentRenderPassId = renderPassIds[p];\r\n for (const subMesh of this.subMeshes) {\r\n if (!generator.isReady(subMesh, hardwareInstancedRendering, subMesh.getMaterial()?.needAlphaBlendingForMesh(this) ?? false)) {\r\n engine.currentRenderPassId = currentRenderPassId;\r\n return false;\r\n }\r\n }\r\n }\r\n engine.currentRenderPassId = currentRenderPassId;\r\n }\r\n }\r\n }\r\n\r\n // LOD\r\n for (const lod of this._internalMeshDataInfo._LODLevels) {\r\n if (lod.mesh && !lod.mesh.isReady(hardwareInstancedRendering)) {\r\n return false;\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if the normals aren't to be recomputed on next mesh `positions` array update. This property is pertinent only for updatable parametric shapes.\r\n */\r\n public get areNormalsFrozen(): boolean {\r\n return this._internalMeshDataInfo._areNormalsFrozen;\r\n }\r\n\r\n /**\r\n * This function affects parametric shapes on vertex position update only : ribbons, tubes, etc. It has no effect at all on other shapes. It prevents the mesh normals from being recomputed on next `positions` array update.\r\n * @returns the current mesh\r\n */\r\n public freezeNormals(): Mesh {\r\n this._internalMeshDataInfo._areNormalsFrozen = true;\r\n return this;\r\n }\r\n\r\n /**\r\n * This function affects parametric shapes on vertex position update only : ribbons, tubes, etc. It has no effect at all on other shapes. It reactivates the mesh normals computation if it was previously frozen\r\n * @returns the current mesh\r\n */\r\n public unfreezeNormals(): Mesh {\r\n this._internalMeshDataInfo._areNormalsFrozen = false;\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets a value overriding the instance count. Only applicable when custom instanced InterleavedVertexBuffer are used rather than InstancedMeshs\r\n */\r\n public set overridenInstanceCount(count: number) {\r\n this._instanceDataStorage.overridenInstanceCount = count;\r\n }\r\n\r\n // Methods\r\n /** @internal */\r\n public _preActivate(): Mesh {\r\n const internalDataInfo = this._internalMeshDataInfo;\r\n const sceneRenderId = this.getScene().getRenderId();\r\n if (internalDataInfo._preActivateId === sceneRenderId) {\r\n return this;\r\n }\r\n\r\n internalDataInfo._preActivateId = sceneRenderId;\r\n this._instanceDataStorage.visibleInstances = null;\r\n return this;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _preActivateForIntermediateRendering(renderId: number): Mesh {\r\n if (this._instanceDataStorage.visibleInstances) {\r\n this._instanceDataStorage.visibleInstances.intermediateDefaultRenderId = renderId;\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _registerInstanceForRenderId(instance: InstancedMesh, renderId: number): Mesh {\r\n if (!this._instanceDataStorage.visibleInstances) {\r\n this._instanceDataStorage.visibleInstances = {\r\n defaultRenderId: renderId,\r\n selfDefaultRenderId: this._renderId,\r\n };\r\n }\r\n\r\n if (!this._instanceDataStorage.visibleInstances[renderId]) {\r\n if (this._instanceDataStorage.previousRenderId !== undefined && this._instanceDataStorage.isFrozen) {\r\n this._instanceDataStorage.visibleInstances[this._instanceDataStorage.previousRenderId] = null;\r\n }\r\n this._instanceDataStorage.previousRenderId = renderId;\r\n this._instanceDataStorage.visibleInstances[renderId] = new Array();\r\n }\r\n\r\n this._instanceDataStorage.visibleInstances[renderId].push(instance);\r\n return this;\r\n }\r\n\r\n protected _afterComputeWorldMatrix(): void {\r\n super._afterComputeWorldMatrix();\r\n\r\n if (!this.hasThinInstances) {\r\n return;\r\n }\r\n\r\n if (!this.doNotSyncBoundingInfo) {\r\n this.thinInstanceRefreshBoundingInfo(false);\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _postActivate(): void {\r\n if (this.edgesShareWithInstances && this.edgesRenderer && this.edgesRenderer.isEnabled && this._renderingGroup) {\r\n this._renderingGroup._edgesRenderers.pushNoDuplicate(this.edgesRenderer);\r\n this.edgesRenderer.customInstances.push(this.getWorldMatrix());\r\n }\r\n }\r\n\r\n /**\r\n * This method recomputes and sets a new BoundingInfo to the mesh unless it is locked.\r\n * This means the mesh underlying bounding box and sphere are recomputed.\r\n * @param applySkeleton defines whether to apply the skeleton before computing the bounding info\r\n * @param applyMorph defines whether to apply the morph target before computing the bounding info\r\n * @returns the current mesh\r\n */\r\n public refreshBoundingInfo(applySkeleton: boolean = false, applyMorph: boolean = false): Mesh {\r\n if (this.hasBoundingInfo && this.getBoundingInfo().isLocked) {\r\n return this;\r\n }\r\n\r\n const bias = this.geometry ? this.geometry.boundingBias : null;\r\n this._refreshBoundingInfo(this._getPositionData(applySkeleton, applyMorph), bias);\r\n return this;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _createGlobalSubMesh(force: boolean): Nullable {\r\n const totalVertices = this.getTotalVertices();\r\n if (!totalVertices || !this.getIndices()) {\r\n return null;\r\n }\r\n\r\n // Check if we need to recreate the submeshes\r\n if (this.subMeshes && this.subMeshes.length > 0) {\r\n const ib = this.getIndices();\r\n\r\n if (!ib) {\r\n return null;\r\n }\r\n\r\n const totalIndices = ib.length;\r\n let needToRecreate = false;\r\n\r\n if (force) {\r\n needToRecreate = true;\r\n } else {\r\n for (const submesh of this.subMeshes) {\r\n if (submesh.indexStart + submesh.indexCount > totalIndices) {\r\n needToRecreate = true;\r\n break;\r\n }\r\n\r\n if (submesh.verticesStart + submesh.verticesCount > totalVertices) {\r\n needToRecreate = true;\r\n break;\r\n }\r\n }\r\n }\r\n\r\n if (!needToRecreate) {\r\n return this.subMeshes[0];\r\n }\r\n }\r\n\r\n this.releaseSubMeshes();\r\n return new SubMesh(0, 0, totalVertices, 0, this.getTotalIndices(), this);\r\n }\r\n\r\n /**\r\n * This function will subdivide the mesh into multiple submeshes\r\n * @param count defines the expected number of submeshes\r\n */\r\n public subdivide(count: number): void {\r\n if (count < 1) {\r\n return;\r\n }\r\n\r\n const totalIndices = this.getTotalIndices();\r\n let subdivisionSize = (totalIndices / count) | 0;\r\n let offset = 0;\r\n\r\n // Ensure that subdivisionSize is a multiple of 3\r\n while (subdivisionSize % 3 !== 0) {\r\n subdivisionSize++;\r\n }\r\n\r\n this.releaseSubMeshes();\r\n for (let index = 0; index < count; index++) {\r\n if (offset >= totalIndices) {\r\n break;\r\n }\r\n\r\n SubMesh.CreateFromIndices(0, offset, index === count - 1 ? totalIndices - offset : subdivisionSize, this, undefined, false);\r\n\r\n offset += subdivisionSize;\r\n }\r\n\r\n this.refreshBoundingInfo();\r\n this.synchronizeInstances();\r\n }\r\n\r\n /**\r\n * Copy a FloatArray into a specific associated vertex buffer\r\n * @param kind defines which buffer to write to (positions, indices, normals, etc). Possible `kind` values :\r\n * - VertexBuffer.PositionKind\r\n * - VertexBuffer.UVKind\r\n * - VertexBuffer.UV2Kind\r\n * - VertexBuffer.UV3Kind\r\n * - VertexBuffer.UV4Kind\r\n * - VertexBuffer.UV5Kind\r\n * - VertexBuffer.UV6Kind\r\n * - VertexBuffer.ColorKind\r\n * - VertexBuffer.MatricesIndicesKind\r\n * - VertexBuffer.MatricesIndicesExtraKind\r\n * - VertexBuffer.MatricesWeightsKind\r\n * - VertexBuffer.MatricesWeightsExtraKind\r\n * @param data defines the data source\r\n * @param updatable defines if the updated vertex buffer must be flagged as updatable\r\n * @param stride defines the data stride size (can be null)\r\n * @returns the current mesh\r\n */\r\n public setVerticesData(kind: string, data: FloatArray, updatable: boolean = false, stride?: number): AbstractMesh {\r\n if (!this._geometry) {\r\n const vertexData = new VertexData();\r\n vertexData.set(data, kind);\r\n\r\n const scene = this.getScene();\r\n\r\n new Geometry(Geometry.RandomId(), scene, vertexData, updatable, this);\r\n } else {\r\n this._geometry.setVerticesData(kind, data, updatable, stride);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Delete a vertex buffer associated with this mesh\r\n * @param kind defines which buffer to delete (positions, indices, normals, etc). Possible `kind` values :\r\n * - VertexBuffer.PositionKind\r\n * - VertexBuffer.UVKind\r\n * - VertexBuffer.UV2Kind\r\n * - VertexBuffer.UV3Kind\r\n * - VertexBuffer.UV4Kind\r\n * - VertexBuffer.UV5Kind\r\n * - VertexBuffer.UV6Kind\r\n * - VertexBuffer.ColorKind\r\n * - VertexBuffer.MatricesIndicesKind\r\n * - VertexBuffer.MatricesIndicesExtraKind\r\n * - VertexBuffer.MatricesWeightsKind\r\n * - VertexBuffer.MatricesWeightsExtraKind\r\n */\r\n public removeVerticesData(kind: string) {\r\n if (!this._geometry) {\r\n return;\r\n }\r\n\r\n this._geometry.removeVerticesData(kind);\r\n }\r\n\r\n /**\r\n * Flags an associated vertex buffer as updatable\r\n * @param kind defines which buffer to use (positions, indices, normals, etc). Possible `kind` values :\r\n * - VertexBuffer.PositionKind\r\n * - VertexBuffer.UVKind\r\n * - VertexBuffer.UV2Kind\r\n * - VertexBuffer.UV3Kind\r\n * - VertexBuffer.UV4Kind\r\n * - VertexBuffer.UV5Kind\r\n * - VertexBuffer.UV6Kind\r\n * - VertexBuffer.ColorKind\r\n * - VertexBuffer.MatricesIndicesKind\r\n * - VertexBuffer.MatricesIndicesExtraKind\r\n * - VertexBuffer.MatricesWeightsKind\r\n * - VertexBuffer.MatricesWeightsExtraKind\r\n * @param updatable defines if the updated vertex buffer must be flagged as updatable\r\n */\r\n public markVerticesDataAsUpdatable(kind: string, updatable = true) {\r\n const vb = this.getVertexBuffer(kind);\r\n\r\n if (!vb || vb.isUpdatable() === updatable) {\r\n return;\r\n }\r\n\r\n this.setVerticesData(kind, this.getVerticesData(kind), updatable);\r\n }\r\n\r\n /**\r\n * Sets the mesh global Vertex Buffer\r\n * @param buffer defines the buffer to use\r\n * @param disposeExistingBuffer disposes the existing buffer, if any (default: true)\r\n * @returns the current mesh\r\n */\r\n public setVerticesBuffer(buffer: VertexBuffer, disposeExistingBuffer = true): Mesh {\r\n if (!this._geometry) {\r\n this._geometry = Geometry.CreateGeometryForMesh(this);\r\n }\r\n\r\n this._geometry.setVerticesBuffer(buffer, null, disposeExistingBuffer);\r\n return this;\r\n }\r\n\r\n /**\r\n * Update a specific associated vertex buffer\r\n * @param kind defines which buffer to write to (positions, indices, normals, etc). Possible `kind` values :\r\n * - VertexBuffer.PositionKind\r\n * - VertexBuffer.UVKind\r\n * - VertexBuffer.UV2Kind\r\n * - VertexBuffer.UV3Kind\r\n * - VertexBuffer.UV4Kind\r\n * - VertexBuffer.UV5Kind\r\n * - VertexBuffer.UV6Kind\r\n * - VertexBuffer.ColorKind\r\n * - VertexBuffer.MatricesIndicesKind\r\n * - VertexBuffer.MatricesIndicesExtraKind\r\n * - VertexBuffer.MatricesWeightsKind\r\n * - VertexBuffer.MatricesWeightsExtraKind\r\n * @param data defines the data source\r\n * @param updateExtends defines if extends info of the mesh must be updated (can be null). This is mostly useful for \"position\" kind\r\n * @param makeItUnique defines if the geometry associated with the mesh must be cloned to make the change only for this mesh (and not all meshes associated with the same geometry)\r\n * @returns the current mesh\r\n */\r\n public updateVerticesData(kind: string, data: FloatArray, updateExtends?: boolean, makeItUnique?: boolean): AbstractMesh {\r\n if (!this._geometry) {\r\n return this;\r\n }\r\n if (!makeItUnique) {\r\n this._geometry.updateVerticesData(kind, data, updateExtends);\r\n } else {\r\n this.makeGeometryUnique();\r\n this.updateVerticesData(kind, data, updateExtends, false);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * This method updates the vertex positions of an updatable mesh according to the `positionFunction` returned values.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#other-shapes-updatemeshpositions\r\n * @param positionFunction is a simple JS function what is passed the mesh `positions` array. It doesn't need to return anything\r\n * @param computeNormals is a boolean (default true) to enable/disable the mesh normal recomputation after the vertex position update\r\n * @returns the current mesh\r\n */\r\n public updateMeshPositions(positionFunction: (data: FloatArray) => void, computeNormals: boolean = true): Mesh {\r\n const positions = this.getVerticesData(VertexBuffer.PositionKind);\r\n if (!positions) {\r\n return this;\r\n }\r\n\r\n positionFunction(positions);\r\n this.updateVerticesData(VertexBuffer.PositionKind, positions, false, false);\r\n\r\n if (computeNormals) {\r\n const indices = this.getIndices();\r\n const normals = this.getVerticesData(VertexBuffer.NormalKind);\r\n\r\n if (!normals) {\r\n return this;\r\n }\r\n\r\n VertexData.ComputeNormals(positions, indices, normals);\r\n this.updateVerticesData(VertexBuffer.NormalKind, normals, false, false);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Creates a un-shared specific occurence of the geometry for the mesh.\r\n * @returns the current mesh\r\n */\r\n public makeGeometryUnique(): Mesh {\r\n if (!this._geometry) {\r\n return this;\r\n }\r\n\r\n if (this._geometry.meshes.length === 1) {\r\n return this;\r\n }\r\n\r\n const oldGeometry = this._geometry;\r\n const geometry = this._geometry.copy(Geometry.RandomId());\r\n oldGeometry.releaseForMesh(this, true);\r\n geometry.applyToMesh(this);\r\n return this;\r\n }\r\n\r\n /**\r\n * Set the index buffer of this mesh\r\n * @param indices defines the source data\r\n * @param totalVertices defines the total number of vertices referenced by this index data (can be null)\r\n * @param updatable defines if the updated index buffer must be flagged as updatable (default is false)\r\n * @returns the current mesh\r\n */\r\n public setIndices(indices: IndicesArray, totalVertices: Nullable = null, updatable: boolean = false): AbstractMesh {\r\n if (!this._geometry) {\r\n const vertexData = new VertexData();\r\n vertexData.indices = indices;\r\n\r\n const scene = this.getScene();\r\n\r\n new Geometry(Geometry.RandomId(), scene, vertexData, updatable, this);\r\n } else {\r\n this._geometry.setIndices(indices, totalVertices, updatable);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Update the current index buffer\r\n * @param indices defines the source data\r\n * @param offset defines the offset in the index buffer where to store the new data (can be null)\r\n * @param gpuMemoryOnly defines a boolean indicating that only the GPU memory must be updated leaving the CPU version of the indices unchanged (false by default)\r\n * @returns the current mesh\r\n */\r\n public updateIndices(indices: IndicesArray, offset?: number, gpuMemoryOnly = false): AbstractMesh {\r\n if (!this._geometry) {\r\n return this;\r\n }\r\n\r\n this._geometry.updateIndices(indices, offset, gpuMemoryOnly);\r\n return this;\r\n }\r\n\r\n /**\r\n * Invert the geometry to move from a right handed system to a left handed one.\r\n * @returns the current mesh\r\n */\r\n public toLeftHanded(): Mesh {\r\n if (!this._geometry) {\r\n return this;\r\n }\r\n this._geometry.toLeftHanded();\r\n return this;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _bind(subMesh: SubMesh, effect: Effect, fillMode: number, allowInstancedRendering = true): Mesh {\r\n if (!this._geometry) {\r\n return this;\r\n }\r\n\r\n const engine = this.getScene().getEngine();\r\n\r\n // Morph targets\r\n if (this.morphTargetManager && this.morphTargetManager.isUsingTextureForTargets) {\r\n this.morphTargetManager._bind(effect);\r\n }\r\n\r\n // Wireframe\r\n let indexToBind;\r\n if (this._unIndexed) {\r\n indexToBind = null;\r\n } else {\r\n switch (this._getRenderingFillMode(fillMode)) {\r\n case Material.PointFillMode:\r\n indexToBind = null;\r\n break;\r\n case Material.WireFrameFillMode:\r\n indexToBind = subMesh._getLinesIndexBuffer(this.getIndices(), engine);\r\n break;\r\n default:\r\n case Material.TriangleFillMode:\r\n indexToBind = this._geometry.getIndexBuffer();\r\n break;\r\n }\r\n }\r\n\r\n // VBOs\r\n if (!allowInstancedRendering || !this._userInstancedBuffersStorage || this.hasThinInstances) {\r\n this._geometry._bind(effect, indexToBind);\r\n } else {\r\n this._geometry._bind(effect, indexToBind, this._userInstancedBuffersStorage.vertexBuffers, this._userInstancedBuffersStorage.vertexArrayObjects);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _draw(subMesh: SubMesh, fillMode: number, instancesCount?: number): Mesh {\r\n if (!this._geometry || !this._geometry.getVertexBuffers() || (!this._unIndexed && !this._geometry.getIndexBuffer())) {\r\n return this;\r\n }\r\n\r\n if (this._internalMeshDataInfo._onBeforeDrawObservable) {\r\n this._internalMeshDataInfo._onBeforeDrawObservable.notifyObservers(this);\r\n }\r\n\r\n const scene = this.getScene();\r\n const engine = scene.getEngine();\r\n\r\n if (this._unIndexed || fillMode == Material.PointFillMode) {\r\n // or triangles as points\r\n engine.drawArraysType(fillMode, subMesh.verticesStart, subMesh.verticesCount, this.forcedInstanceCount || instancesCount);\r\n } else if (fillMode == Material.WireFrameFillMode) {\r\n // Triangles as wireframe\r\n engine.drawElementsType(fillMode, 0, subMesh._linesIndexCount, this.forcedInstanceCount || instancesCount);\r\n } else {\r\n engine.drawElementsType(fillMode, subMesh.indexStart, subMesh.indexCount, this.forcedInstanceCount || instancesCount);\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Registers for this mesh a javascript function called just before the rendering process\r\n * @param func defines the function to call before rendering this mesh\r\n * @returns the current mesh\r\n */\r\n public registerBeforeRender(func: (mesh: AbstractMesh) => void): Mesh {\r\n this.onBeforeRenderObservable.add(func);\r\n return this;\r\n }\r\n\r\n /**\r\n * Disposes a previously registered javascript function called before the rendering\r\n * @param func defines the function to remove\r\n * @returns the current mesh\r\n */\r\n public unregisterBeforeRender(func: (mesh: AbstractMesh) => void): Mesh {\r\n this.onBeforeRenderObservable.removeCallback(func);\r\n return this;\r\n }\r\n\r\n /**\r\n * Registers for this mesh a javascript function called just after the rendering is complete\r\n * @param func defines the function to call after rendering this mesh\r\n * @returns the current mesh\r\n */\r\n public registerAfterRender(func: (mesh: AbstractMesh) => void): Mesh {\r\n this.onAfterRenderObservable.add(func);\r\n return this;\r\n }\r\n\r\n /**\r\n * Disposes a previously registered javascript function called after the rendering.\r\n * @param func defines the function to remove\r\n * @returns the current mesh\r\n */\r\n public unregisterAfterRender(func: (mesh: AbstractMesh) => void): Mesh {\r\n this.onAfterRenderObservable.removeCallback(func);\r\n return this;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getInstancesRenderList(subMeshId: number, isReplacementMode: boolean = false): _InstancesBatch {\r\n if (this._instanceDataStorage.isFrozen) {\r\n if (isReplacementMode) {\r\n this._instanceDataStorage.batchCacheReplacementModeInFrozenMode.hardwareInstancedRendering[subMeshId] = false;\r\n this._instanceDataStorage.batchCacheReplacementModeInFrozenMode.renderSelf[subMeshId] = true;\r\n return this._instanceDataStorage.batchCacheReplacementModeInFrozenMode;\r\n }\r\n if (this._instanceDataStorage.previousBatch) {\r\n return this._instanceDataStorage.previousBatch;\r\n }\r\n }\r\n const scene = this.getScene();\r\n const isInIntermediateRendering = scene._isInIntermediateRendering();\r\n const onlyForInstances = isInIntermediateRendering\r\n ? this._internalAbstractMeshDataInfo._onlyForInstancesIntermediate\r\n : this._internalAbstractMeshDataInfo._onlyForInstances;\r\n const batchCache = this._instanceDataStorage.batchCache;\r\n batchCache.mustReturn = false;\r\n batchCache.renderSelf[subMeshId] = isReplacementMode || (!onlyForInstances && this.isEnabled() && this.isVisible);\r\n batchCache.visibleInstances[subMeshId] = null;\r\n\r\n if (this._instanceDataStorage.visibleInstances && !isReplacementMode) {\r\n const visibleInstances = this._instanceDataStorage.visibleInstances;\r\n const currentRenderId = scene.getRenderId();\r\n const defaultRenderId = isInIntermediateRendering ? visibleInstances.intermediateDefaultRenderId : visibleInstances.defaultRenderId;\r\n batchCache.visibleInstances[subMeshId] = visibleInstances[currentRenderId];\r\n\r\n if (!batchCache.visibleInstances[subMeshId] && defaultRenderId) {\r\n batchCache.visibleInstances[subMeshId] = visibleInstances[defaultRenderId];\r\n }\r\n }\r\n batchCache.hardwareInstancedRendering[subMeshId] =\r\n !isReplacementMode &&\r\n this._instanceDataStorage.hardwareInstancedRendering &&\r\n batchCache.visibleInstances[subMeshId] !== null &&\r\n batchCache.visibleInstances[subMeshId] !== undefined;\r\n this._instanceDataStorage.previousBatch = batchCache;\r\n\r\n return batchCache;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _renderWithInstances(subMesh: SubMesh, fillMode: number, batch: _InstancesBatch, effect: Effect, engine: Engine): Mesh {\r\n const visibleInstances = batch.visibleInstances[subMesh._id];\r\n const visibleInstanceCount = visibleInstances ? visibleInstances.length : 0;\r\n\r\n const instanceStorage = this._instanceDataStorage;\r\n const currentInstancesBufferSize = instanceStorage.instancesBufferSize;\r\n let instancesBuffer = instanceStorage.instancesBuffer;\r\n let instancesPreviousBuffer = instanceStorage.instancesPreviousBuffer;\r\n const matricesCount = visibleInstanceCount + 1;\r\n const bufferSize = matricesCount * 16 * 4;\r\n\r\n while (instanceStorage.instancesBufferSize < bufferSize) {\r\n instanceStorage.instancesBufferSize *= 2;\r\n }\r\n\r\n if (!instanceStorage.instancesData || currentInstancesBufferSize != instanceStorage.instancesBufferSize) {\r\n instanceStorage.instancesData = new Float32Array(instanceStorage.instancesBufferSize / 4);\r\n }\r\n if ((this._scene.needsPreviousWorldMatrices && !instanceStorage.instancesPreviousData) || currentInstancesBufferSize != instanceStorage.instancesBufferSize) {\r\n instanceStorage.instancesPreviousData = new Float32Array(instanceStorage.instancesBufferSize / 4);\r\n }\r\n\r\n let offset = 0;\r\n let instancesCount = 0;\r\n\r\n const renderSelf = batch.renderSelf[subMesh._id];\r\n\r\n const needUpdateBuffer =\r\n !instancesBuffer ||\r\n currentInstancesBufferSize !== instanceStorage.instancesBufferSize ||\r\n (this._scene.needsPreviousWorldMatrices && !instanceStorage.instancesPreviousBuffer);\r\n\r\n if (!this._instanceDataStorage.manualUpdate && (!instanceStorage.isFrozen || needUpdateBuffer)) {\r\n const world = this.getWorldMatrix();\r\n if (renderSelf) {\r\n if (this._scene.needsPreviousWorldMatrices) {\r\n if (!instanceStorage.masterMeshPreviousWorldMatrix) {\r\n instanceStorage.masterMeshPreviousWorldMatrix = world.clone();\r\n instanceStorage.masterMeshPreviousWorldMatrix.copyToArray(instanceStorage.instancesPreviousData, offset);\r\n } else {\r\n instanceStorage.masterMeshPreviousWorldMatrix.copyToArray(instanceStorage.instancesPreviousData, offset);\r\n instanceStorage.masterMeshPreviousWorldMatrix.copyFrom(world);\r\n }\r\n }\r\n world.copyToArray(instanceStorage.instancesData, offset);\r\n offset += 16;\r\n instancesCount++;\r\n }\r\n\r\n if (visibleInstances) {\r\n if (Mesh.INSTANCEDMESH_SORT_TRANSPARENT && this._scene.activeCamera && subMesh.getMaterial()?.needAlphaBlendingForMesh(subMesh.getRenderingMesh())) {\r\n const cameraPosition = this._scene.activeCamera.globalPosition;\r\n for (let instanceIndex = 0; instanceIndex < visibleInstances.length; instanceIndex++) {\r\n const instanceMesh = visibleInstances[instanceIndex];\r\n instanceMesh._distanceToCamera = Vector3.Distance(instanceMesh.getBoundingInfo().boundingSphere.centerWorld, cameraPosition);\r\n }\r\n visibleInstances.sort((m1, m2) => {\r\n return m1._distanceToCamera > m2._distanceToCamera ? -1 : m1._distanceToCamera < m2._distanceToCamera ? 1 : 0;\r\n });\r\n }\r\n for (let instanceIndex = 0; instanceIndex < visibleInstances.length; instanceIndex++) {\r\n const instance = visibleInstances[instanceIndex];\r\n const matrix = instance.getWorldMatrix();\r\n matrix.copyToArray(instanceStorage.instancesData, offset);\r\n\r\n if (this._scene.needsPreviousWorldMatrices) {\r\n if (!instance._previousWorldMatrix) {\r\n instance._previousWorldMatrix = matrix.clone();\r\n instance._previousWorldMatrix.copyToArray(instanceStorage.instancesPreviousData, offset);\r\n } else {\r\n instance._previousWorldMatrix.copyToArray(instanceStorage.instancesPreviousData, offset);\r\n instance._previousWorldMatrix.copyFrom(matrix);\r\n }\r\n }\r\n\r\n offset += 16;\r\n instancesCount++;\r\n }\r\n }\r\n } else {\r\n instancesCount = (renderSelf ? 1 : 0) + visibleInstanceCount;\r\n }\r\n\r\n if (needUpdateBuffer) {\r\n if (instancesBuffer) {\r\n instancesBuffer.dispose();\r\n }\r\n\r\n if (instancesPreviousBuffer) {\r\n instancesPreviousBuffer.dispose();\r\n }\r\n\r\n instancesBuffer = new Buffer(engine, instanceStorage.instancesData, true, 16, false, true);\r\n instanceStorage.instancesBuffer = instancesBuffer;\r\n if (!this._userInstancedBuffersStorage) {\r\n this._userInstancedBuffersStorage = {\r\n data: {},\r\n vertexBuffers: {},\r\n strides: {},\r\n sizes: {},\r\n vertexArrayObjects: this.getEngine().getCaps().vertexArrayObject ? {} : undefined,\r\n };\r\n }\r\n\r\n this._userInstancedBuffersStorage.vertexBuffers[\"world0\"] = instancesBuffer.createVertexBuffer(\"world0\", 0, 4);\r\n this._userInstancedBuffersStorage.vertexBuffers[\"world1\"] = instancesBuffer.createVertexBuffer(\"world1\", 4, 4);\r\n this._userInstancedBuffersStorage.vertexBuffers[\"world2\"] = instancesBuffer.createVertexBuffer(\"world2\", 8, 4);\r\n this._userInstancedBuffersStorage.vertexBuffers[\"world3\"] = instancesBuffer.createVertexBuffer(\"world3\", 12, 4);\r\n\r\n if (this._scene.needsPreviousWorldMatrices) {\r\n instancesPreviousBuffer = new Buffer(engine, instanceStorage.instancesPreviousData, true, 16, false, true);\r\n instanceStorage.instancesPreviousBuffer = instancesPreviousBuffer;\r\n\r\n this._userInstancedBuffersStorage.vertexBuffers[\"previousWorld0\"] = instancesPreviousBuffer.createVertexBuffer(\"previousWorld0\", 0, 4);\r\n this._userInstancedBuffersStorage.vertexBuffers[\"previousWorld1\"] = instancesPreviousBuffer.createVertexBuffer(\"previousWorld1\", 4, 4);\r\n this._userInstancedBuffersStorage.vertexBuffers[\"previousWorld2\"] = instancesPreviousBuffer.createVertexBuffer(\"previousWorld2\", 8, 4);\r\n this._userInstancedBuffersStorage.vertexBuffers[\"previousWorld3\"] = instancesPreviousBuffer.createVertexBuffer(\"previousWorld3\", 12, 4);\r\n }\r\n this._invalidateInstanceVertexArrayObject();\r\n } else {\r\n if (!this._instanceDataStorage.isFrozen || this._instanceDataStorage.forceMatrixUpdates) {\r\n instancesBuffer!.updateDirectly(instanceStorage.instancesData, 0, instancesCount);\r\n if (this._scene.needsPreviousWorldMatrices && (!this._instanceDataStorage.manualUpdate || this._instanceDataStorage.previousManualUpdate)) {\r\n instancesPreviousBuffer!.updateDirectly(instanceStorage.instancesPreviousData, 0, instancesCount);\r\n }\r\n }\r\n }\r\n\r\n this._processInstancedBuffers(visibleInstances, renderSelf);\r\n\r\n // Stats\r\n this.getScene()._activeIndices.addCount(subMesh.indexCount * instancesCount, false);\r\n\r\n // Draw\r\n if (engine._currentDrawContext) {\r\n engine._currentDrawContext.useInstancing = true;\r\n }\r\n this._bind(subMesh, effect, fillMode);\r\n this._draw(subMesh, fillMode, instancesCount);\r\n\r\n // Write current matrices as previous matrices in case of manual update\r\n // Default behaviour when previous matrices are not specified explicitly\r\n // Will break if instances number/order changes\r\n if (\r\n this._scene.needsPreviousWorldMatrices &&\r\n !needUpdateBuffer &&\r\n this._instanceDataStorage.manualUpdate &&\r\n (!this._instanceDataStorage.isFrozen || this._instanceDataStorage.forceMatrixUpdates) &&\r\n !this._instanceDataStorage.previousManualUpdate\r\n ) {\r\n instancesPreviousBuffer!.updateDirectly(instanceStorage.instancesData, 0, instancesCount);\r\n }\r\n\r\n engine.unbindInstanceAttributes();\r\n return this;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _renderWithThinInstances(subMesh: SubMesh, fillMode: number, effect: Effect, engine: Engine) {\r\n // Stats\r\n const instancesCount = this._thinInstanceDataStorage?.instancesCount ?? 0;\r\n\r\n this.getScene()._activeIndices.addCount(subMesh.indexCount * instancesCount, false);\r\n\r\n // Draw\r\n if (engine._currentDrawContext) {\r\n engine._currentDrawContext.useInstancing = true;\r\n }\r\n this._bind(subMesh, effect, fillMode);\r\n this._draw(subMesh, fillMode, instancesCount);\r\n\r\n // Write current matrices as previous matrices\r\n // Default behaviour when previous matrices are not specified explicitly\r\n // Will break if instances number/order changes\r\n if (this._scene.needsPreviousWorldMatrices && !this._thinInstanceDataStorage.previousMatrixData && this._thinInstanceDataStorage.matrixData) {\r\n if (!this._thinInstanceDataStorage.previousMatrixBuffer) {\r\n this._thinInstanceDataStorage.previousMatrixBuffer = this._thinInstanceCreateMatrixBuffer(\"previousWorld\", this._thinInstanceDataStorage.matrixData, false);\r\n } else {\r\n this._thinInstanceDataStorage.previousMatrixBuffer!.updateDirectly(this._thinInstanceDataStorage.matrixData, 0, instancesCount);\r\n }\r\n }\r\n\r\n engine.unbindInstanceAttributes();\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public _processInstancedBuffers(visibleInstances: Nullable, renderSelf: boolean) {\r\n // Do nothing\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _processRendering(\r\n renderingMesh: AbstractMesh,\r\n subMesh: SubMesh,\r\n effect: Effect,\r\n fillMode: number,\r\n batch: _InstancesBatch,\r\n hardwareInstancedRendering: boolean,\r\n onBeforeDraw: (isInstance: boolean, world: Matrix, effectiveMaterial?: Material) => void,\r\n effectiveMaterial?: Material\r\n ): Mesh {\r\n const scene = this.getScene();\r\n const engine = scene.getEngine();\r\n fillMode = this._getRenderingFillMode(fillMode);\r\n\r\n if (hardwareInstancedRendering && subMesh.getRenderingMesh().hasThinInstances) {\r\n this._renderWithThinInstances(subMesh, fillMode, effect, engine);\r\n return this;\r\n }\r\n\r\n if (hardwareInstancedRendering) {\r\n this._renderWithInstances(subMesh, fillMode, batch, effect, engine);\r\n } else {\r\n if (engine._currentDrawContext) {\r\n engine._currentDrawContext.useInstancing = false;\r\n }\r\n\r\n let instanceCount = 0;\r\n if (batch.renderSelf[subMesh._id]) {\r\n // Draw\r\n if (onBeforeDraw) {\r\n onBeforeDraw(false, renderingMesh.getWorldMatrix(), effectiveMaterial);\r\n }\r\n instanceCount++;\r\n\r\n this._draw(subMesh, fillMode, this._instanceDataStorage.overridenInstanceCount);\r\n }\r\n\r\n const visibleInstancesForSubMesh = batch.visibleInstances[subMesh._id];\r\n\r\n if (visibleInstancesForSubMesh) {\r\n const visibleInstanceCount = visibleInstancesForSubMesh.length;\r\n instanceCount += visibleInstanceCount;\r\n\r\n // Stats\r\n for (let instanceIndex = 0; instanceIndex < visibleInstanceCount; instanceIndex++) {\r\n const instance = visibleInstancesForSubMesh[instanceIndex];\r\n\r\n // World\r\n const world = instance.getWorldMatrix();\r\n if (onBeforeDraw) {\r\n onBeforeDraw(true, world, effectiveMaterial);\r\n }\r\n // Draw\r\n this._draw(subMesh, fillMode);\r\n }\r\n }\r\n\r\n // Stats\r\n scene._activeIndices.addCount(subMesh.indexCount * instanceCount, false);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _rebuild(dispose = false): void {\r\n if (this._instanceDataStorage.instancesBuffer) {\r\n // Dispose instance buffer to be recreated in _renderWithInstances when rendered\r\n if (dispose) {\r\n this._instanceDataStorage.instancesBuffer.dispose();\r\n }\r\n this._instanceDataStorage.instancesBuffer = null;\r\n }\r\n if (this._userInstancedBuffersStorage) {\r\n for (const kind in this._userInstancedBuffersStorage.vertexBuffers) {\r\n const buffer = this._userInstancedBuffersStorage.vertexBuffers[kind];\r\n if (buffer) {\r\n // Dispose instance buffer to be recreated in _renderWithInstances when rendered\r\n if (dispose) {\r\n buffer.dispose();\r\n }\r\n this._userInstancedBuffersStorage.vertexBuffers[kind] = null;\r\n }\r\n }\r\n if (this._userInstancedBuffersStorage.vertexArrayObjects) {\r\n this._userInstancedBuffersStorage.vertexArrayObjects = {};\r\n }\r\n }\r\n this._internalMeshDataInfo._effectiveMaterial = null;\r\n super._rebuild(dispose);\r\n }\r\n\r\n /** @internal */\r\n public _freeze() {\r\n if (!this.subMeshes) {\r\n return;\r\n }\r\n\r\n // Prepare batches\r\n for (let index = 0; index < this.subMeshes.length; index++) {\r\n this._getInstancesRenderList(index);\r\n }\r\n\r\n this._internalMeshDataInfo._effectiveMaterial = null;\r\n this._instanceDataStorage.isFrozen = true;\r\n }\r\n\r\n /** @internal */\r\n public _unFreeze() {\r\n this._instanceDataStorage.isFrozen = false;\r\n this._instanceDataStorage.previousBatch = null;\r\n }\r\n\r\n /**\r\n * Triggers the draw call for the mesh. Usually, you don't need to call this method by your own because the mesh rendering is handled by the scene rendering manager\r\n * @param subMesh defines the subMesh to render\r\n * @param enableAlphaMode defines if alpha mode can be changed\r\n * @param effectiveMeshReplacement defines an optional mesh used to provide info for the rendering\r\n * @returns the current mesh\r\n */\r\n public render(subMesh: SubMesh, enableAlphaMode: boolean, effectiveMeshReplacement?: AbstractMesh): Mesh {\r\n const scene = this.getScene();\r\n\r\n if (this._internalAbstractMeshDataInfo._isActiveIntermediate) {\r\n this._internalAbstractMeshDataInfo._isActiveIntermediate = false;\r\n } else {\r\n this._internalAbstractMeshDataInfo._isActive = false;\r\n }\r\n\r\n if (this._checkOcclusionQuery() && !this._occlusionDataStorage.forceRenderingWhenOccluded) {\r\n return this;\r\n }\r\n\r\n // Managing instances\r\n const batch = this._getInstancesRenderList(subMesh._id, !!effectiveMeshReplacement);\r\n\r\n if (batch.mustReturn) {\r\n return this;\r\n }\r\n\r\n // Checking geometry state\r\n if (!this._geometry || !this._geometry.getVertexBuffers() || (!this._unIndexed && !this._geometry.getIndexBuffer())) {\r\n return this;\r\n }\r\n\r\n const engine = scene.getEngine();\r\n let oldCameraMaxZ = 0;\r\n let oldCamera: Nullable = null;\r\n if (this.ignoreCameraMaxZ && scene.activeCamera && !scene._isInIntermediateRendering()) {\r\n oldCameraMaxZ = scene.activeCamera.maxZ;\r\n oldCamera = scene.activeCamera;\r\n scene.activeCamera.maxZ = 0;\r\n scene.updateTransformMatrix(true);\r\n }\r\n\r\n if (this._internalMeshDataInfo._onBeforeRenderObservable) {\r\n this._internalMeshDataInfo._onBeforeRenderObservable.notifyObservers(this);\r\n }\r\n\r\n const renderingMesh = subMesh.getRenderingMesh();\r\n const hardwareInstancedRendering =\r\n batch.hardwareInstancedRendering[subMesh._id] ||\r\n renderingMesh.hasThinInstances ||\r\n (!!this._userInstancedBuffersStorage && !subMesh.getMesh()._internalAbstractMeshDataInfo._actAsRegularMesh);\r\n const instanceDataStorage = this._instanceDataStorage;\r\n\r\n const material = subMesh.getMaterial();\r\n if (!material) {\r\n if (oldCamera) {\r\n oldCamera.maxZ = oldCameraMaxZ;\r\n scene.updateTransformMatrix(true);\r\n }\r\n return this;\r\n }\r\n\r\n // Material\r\n if (!instanceDataStorage.isFrozen || !this._internalMeshDataInfo._effectiveMaterial || this._internalMeshDataInfo._effectiveMaterial !== material) {\r\n if (material._storeEffectOnSubMeshes) {\r\n if (!material.isReadyForSubMesh(this, subMesh, hardwareInstancedRendering)) {\r\n if (oldCamera) {\r\n oldCamera.maxZ = oldCameraMaxZ;\r\n scene.updateTransformMatrix(true);\r\n }\r\n return this;\r\n }\r\n } else if (!material.isReady(this, hardwareInstancedRendering)) {\r\n if (oldCamera) {\r\n oldCamera.maxZ = oldCameraMaxZ;\r\n scene.updateTransformMatrix(true);\r\n }\r\n return this;\r\n }\r\n\r\n this._internalMeshDataInfo._effectiveMaterial = material;\r\n } else if (\r\n (material._storeEffectOnSubMeshes && !subMesh.effect?._wasPreviouslyReady) ||\r\n (!material._storeEffectOnSubMeshes && !material.getEffect()?._wasPreviouslyReady)\r\n ) {\r\n if (oldCamera) {\r\n oldCamera.maxZ = oldCameraMaxZ;\r\n scene.updateTransformMatrix(true);\r\n }\r\n return this;\r\n }\r\n\r\n // Alpha mode\r\n if (enableAlphaMode) {\r\n engine.setAlphaMode(this._internalMeshDataInfo._effectiveMaterial.alphaMode);\r\n }\r\n\r\n let drawWrapper: Nullable;\r\n if (this._internalMeshDataInfo._effectiveMaterial._storeEffectOnSubMeshes) {\r\n drawWrapper = subMesh._drawWrapper;\r\n } else {\r\n drawWrapper = this._internalMeshDataInfo._effectiveMaterial._getDrawWrapper();\r\n }\r\n\r\n const effect = drawWrapper?.effect ?? null;\r\n\r\n for (const step of scene._beforeRenderingMeshStage) {\r\n step.action(this, subMesh, batch, effect);\r\n }\r\n\r\n if (!drawWrapper || !effect) {\r\n if (oldCamera) {\r\n oldCamera.maxZ = oldCameraMaxZ;\r\n scene.updateTransformMatrix(true);\r\n }\r\n return this;\r\n }\r\n\r\n const effectiveMesh = effectiveMeshReplacement || this;\r\n\r\n let sideOrientation: Nullable;\r\n\r\n if (!instanceDataStorage.isFrozen && (this._internalMeshDataInfo._effectiveMaterial.backFaceCulling || this.overrideMaterialSideOrientation !== null)) {\r\n const mainDeterminant = effectiveMesh._getWorldMatrixDeterminant();\r\n sideOrientation = this.overrideMaterialSideOrientation;\r\n if (sideOrientation == null) {\r\n sideOrientation = this._internalMeshDataInfo._effectiveMaterial.sideOrientation;\r\n }\r\n if (mainDeterminant < 0) {\r\n sideOrientation = sideOrientation === Material.ClockWiseSideOrientation ? Material.CounterClockWiseSideOrientation : Material.ClockWiseSideOrientation;\r\n }\r\n instanceDataStorage.sideOrientation = sideOrientation!;\r\n } else {\r\n sideOrientation = instanceDataStorage.sideOrientation;\r\n }\r\n\r\n const reverse = this._internalMeshDataInfo._effectiveMaterial._preBind(drawWrapper, sideOrientation);\r\n\r\n if (this._internalMeshDataInfo._effectiveMaterial.forceDepthWrite) {\r\n engine.setDepthWrite(true);\r\n }\r\n\r\n // Bind\r\n const effectiveMaterial = this._internalMeshDataInfo._effectiveMaterial;\r\n const fillMode = effectiveMaterial.fillMode;\r\n\r\n if (this._internalMeshDataInfo._onBeforeBindObservable) {\r\n this._internalMeshDataInfo._onBeforeBindObservable.notifyObservers(this);\r\n }\r\n\r\n if (!hardwareInstancedRendering) {\r\n // Binding will be done later because we need to add more info to the VB\r\n this._bind(subMesh, effect, fillMode, false);\r\n }\r\n\r\n const world = effectiveMesh.getWorldMatrix();\r\n if (effectiveMaterial._storeEffectOnSubMeshes) {\r\n effectiveMaterial.bindForSubMesh(world, this, subMesh);\r\n } else {\r\n effectiveMaterial.bind(world, this);\r\n }\r\n\r\n if (!effectiveMaterial.backFaceCulling && effectiveMaterial.separateCullingPass) {\r\n engine.setState(true, effectiveMaterial.zOffset, false, !reverse, effectiveMaterial.cullBackFaces, effectiveMaterial.stencil, effectiveMaterial.zOffsetUnits);\r\n this._processRendering(this, subMesh, effect, fillMode, batch, hardwareInstancedRendering, this._onBeforeDraw, this._internalMeshDataInfo._effectiveMaterial);\r\n engine.setState(true, effectiveMaterial.zOffset, false, reverse, effectiveMaterial.cullBackFaces, effectiveMaterial.stencil, effectiveMaterial.zOffsetUnits);\r\n\r\n if (this._internalMeshDataInfo._onBetweenPassObservable) {\r\n this._internalMeshDataInfo._onBetweenPassObservable.notifyObservers(subMesh);\r\n }\r\n }\r\n\r\n // Draw\r\n this._processRendering(this, subMesh, effect, fillMode, batch, hardwareInstancedRendering, this._onBeforeDraw, this._internalMeshDataInfo._effectiveMaterial);\r\n\r\n // Unbind\r\n this._internalMeshDataInfo._effectiveMaterial.unbind();\r\n\r\n for (const step of scene._afterRenderingMeshStage) {\r\n step.action(this, subMesh, batch, effect);\r\n }\r\n\r\n if (this._internalMeshDataInfo._onAfterRenderObservable) {\r\n this._internalMeshDataInfo._onAfterRenderObservable.notifyObservers(this);\r\n }\r\n\r\n if (oldCamera) {\r\n oldCamera.maxZ = oldCameraMaxZ;\r\n scene.updateTransformMatrix(true);\r\n }\r\n\r\n if (scene.performancePriority === ScenePerformancePriority.Aggressive && !instanceDataStorage.isFrozen) {\r\n this._freeze();\r\n }\r\n\r\n return this;\r\n }\r\n\r\n private _onBeforeDraw: (isInstance: boolean, world: Matrix, effectiveMaterial?: Material) => void;\r\n\r\n /**\r\n * Renormalize the mesh and patch it up if there are no weights\r\n * Similar to normalization by adding the weights compute the reciprocal and multiply all elements, this wil ensure that everything adds to 1.\r\n * However in the case of zero weights then we set just a single influence to 1.\r\n * We check in the function for extra's present and if so we use the normalizeSkinWeightsWithExtras rather than the FourWeights version.\r\n */\r\n public cleanMatrixWeights(): void {\r\n if (this.isVerticesDataPresent(VertexBuffer.MatricesWeightsKind)) {\r\n if (this.isVerticesDataPresent(VertexBuffer.MatricesWeightsExtraKind)) {\r\n this._normalizeSkinWeightsAndExtra();\r\n } else {\r\n this._normalizeSkinFourWeights();\r\n }\r\n }\r\n }\r\n\r\n // faster 4 weight version.\r\n private _normalizeSkinFourWeights(): void {\r\n const matricesWeights = this.getVerticesData(VertexBuffer.MatricesWeightsKind);\r\n const numWeights = matricesWeights.length;\r\n\r\n for (let a = 0; a < numWeights; a += 4) {\r\n // accumulate weights\r\n const t = matricesWeights[a] + matricesWeights[a + 1] + matricesWeights[a + 2] + matricesWeights[a + 3];\r\n // check for invalid weight and just set it to 1.\r\n if (t === 0) {\r\n matricesWeights[a] = 1;\r\n } else {\r\n // renormalize so everything adds to 1 use reciprocal\r\n const recip = 1 / t;\r\n matricesWeights[a] *= recip;\r\n matricesWeights[a + 1] *= recip;\r\n matricesWeights[a + 2] *= recip;\r\n matricesWeights[a + 3] *= recip;\r\n }\r\n }\r\n this.setVerticesData(VertexBuffer.MatricesWeightsKind, matricesWeights);\r\n }\r\n // handle special case of extra verts. (in theory gltf can handle 12 influences)\r\n private _normalizeSkinWeightsAndExtra(): void {\r\n const matricesWeightsExtra = this.getVerticesData(VertexBuffer.MatricesWeightsExtraKind);\r\n const matricesWeights = this.getVerticesData(VertexBuffer.MatricesWeightsKind);\r\n const numWeights = matricesWeights.length;\r\n\r\n for (let a = 0; a < numWeights; a += 4) {\r\n // accumulate weights\r\n let t = matricesWeights[a] + matricesWeights[a + 1] + matricesWeights[a + 2] + matricesWeights[a + 3];\r\n t += matricesWeightsExtra[a] + matricesWeightsExtra[a + 1] + matricesWeightsExtra[a + 2] + matricesWeightsExtra[a + 3];\r\n // check for invalid weight and just set it to 1.\r\n if (t === 0) {\r\n matricesWeights[a] = 1;\r\n } else {\r\n // renormalize so everything adds to 1 use reciprocal\r\n const recip = 1 / t;\r\n matricesWeights[a] *= recip;\r\n matricesWeights[a + 1] *= recip;\r\n matricesWeights[a + 2] *= recip;\r\n matricesWeights[a + 3] *= recip;\r\n // same goes for extras\r\n matricesWeightsExtra[a] *= recip;\r\n matricesWeightsExtra[a + 1] *= recip;\r\n matricesWeightsExtra[a + 2] *= recip;\r\n matricesWeightsExtra[a + 3] *= recip;\r\n }\r\n }\r\n this.setVerticesData(VertexBuffer.MatricesWeightsKind, matricesWeights);\r\n this.setVerticesData(VertexBuffer.MatricesWeightsKind, matricesWeightsExtra);\r\n }\r\n\r\n /**\r\n * ValidateSkinning is used to determine that a mesh has valid skinning data along with skin metrics, if missing weights,\r\n * or not normalized it is returned as invalid mesh the string can be used for console logs, or on screen messages to let\r\n * the user know there was an issue with importing the mesh\r\n * @returns a validation object with skinned, valid and report string\r\n */\r\n public validateSkinning(): { skinned: boolean; valid: boolean; report: string } {\r\n const matricesWeightsExtra = this.getVerticesData(VertexBuffer.MatricesWeightsExtraKind);\r\n const matricesWeights = this.getVerticesData(VertexBuffer.MatricesWeightsKind);\r\n if (matricesWeights === null || this.skeleton == null) {\r\n return { skinned: false, valid: true, report: \"not skinned\" };\r\n }\r\n\r\n const numWeights = matricesWeights.length;\r\n let numberNotSorted: number = 0;\r\n let missingWeights: number = 0;\r\n let maxUsedWeights: number = 0;\r\n let numberNotNormalized: number = 0;\r\n const numInfluences: number = matricesWeightsExtra === null ? 4 : 8;\r\n const usedWeightCounts = new Array();\r\n for (let a = 0; a <= numInfluences; a++) {\r\n usedWeightCounts[a] = 0;\r\n }\r\n const toleranceEpsilon: number = 0.001;\r\n\r\n for (let a = 0; a < numWeights; a += 4) {\r\n let lastWeight: number = matricesWeights[a];\r\n let t = lastWeight;\r\n let usedWeights: number = t === 0 ? 0 : 1;\r\n\r\n for (let b = 1; b < numInfluences; b++) {\r\n const d = b < 4 ? matricesWeights[a + b] : matricesWeightsExtra[a + b - 4];\r\n if (d > lastWeight) {\r\n numberNotSorted++;\r\n }\r\n if (d !== 0) {\r\n usedWeights++;\r\n }\r\n t += d;\r\n lastWeight = d;\r\n }\r\n // count the buffer weights usage\r\n usedWeightCounts[usedWeights]++;\r\n\r\n // max influences\r\n if (usedWeights > maxUsedWeights) {\r\n maxUsedWeights = usedWeights;\r\n }\r\n\r\n // check for invalid weight and just set it to 1.\r\n if (t === 0) {\r\n missingWeights++;\r\n } else {\r\n // renormalize so everything adds to 1 use reciprocal\r\n const recip = 1 / t;\r\n let tolerance = 0;\r\n for (let b = 0; b < numInfluences; b++) {\r\n if (b < 4) {\r\n tolerance += Math.abs(matricesWeights[a + b] - matricesWeights[a + b] * recip);\r\n } else {\r\n tolerance += Math.abs(matricesWeightsExtra[a + b - 4] - matricesWeightsExtra[a + b - 4] * recip);\r\n }\r\n }\r\n // arbitrary epsilon value for dictating not normalized\r\n if (tolerance > toleranceEpsilon) {\r\n numberNotNormalized++;\r\n }\r\n }\r\n }\r\n\r\n // validate bone indices are in range of the skeleton\r\n const numBones: number = this.skeleton.bones.length;\r\n const matricesIndices = this.getVerticesData(VertexBuffer.MatricesIndicesKind);\r\n const matricesIndicesExtra = this.getVerticesData(VertexBuffer.MatricesIndicesExtraKind);\r\n let numBadBoneIndices: number = 0;\r\n for (let a = 0; a < numWeights; a += 4) {\r\n for (let b = 0; b < numInfluences; b++) {\r\n const index = b < 4 ? matricesIndices[a + b] : matricesIndicesExtra[a + b - 4];\r\n if (index >= numBones || index < 0) {\r\n numBadBoneIndices++;\r\n }\r\n }\r\n }\r\n\r\n // log mesh stats\r\n const output =\r\n \"Number of Weights = \" +\r\n numWeights / 4 +\r\n \"\\nMaximum influences = \" +\r\n maxUsedWeights +\r\n \"\\nMissing Weights = \" +\r\n missingWeights +\r\n \"\\nNot Sorted = \" +\r\n numberNotSorted +\r\n \"\\nNot Normalized = \" +\r\n numberNotNormalized +\r\n \"\\nWeightCounts = [\" +\r\n usedWeightCounts +\r\n \"]\" +\r\n \"\\nNumber of bones = \" +\r\n numBones +\r\n \"\\nBad Bone Indices = \" +\r\n numBadBoneIndices;\r\n\r\n return { skinned: true, valid: missingWeights === 0 && numberNotNormalized === 0 && numBadBoneIndices === 0, report: output };\r\n }\r\n\r\n /** @internal */\r\n public _checkDelayState(): Mesh {\r\n const scene = this.getScene();\r\n if (this._geometry) {\r\n this._geometry.load(scene);\r\n } else if (this.delayLoadState === Constants.DELAYLOADSTATE_NOTLOADED) {\r\n this.delayLoadState = Constants.DELAYLOADSTATE_LOADING;\r\n\r\n this._queueLoad(scene);\r\n }\r\n return this;\r\n }\r\n\r\n private _queueLoad(scene: Scene): Mesh {\r\n scene.addPendingData(this);\r\n\r\n const getBinaryData = this.delayLoadingFile.indexOf(\".babylonbinarymeshdata\") !== -1;\r\n\r\n Tools.LoadFile(\r\n this.delayLoadingFile,\r\n (data) => {\r\n if (data instanceof ArrayBuffer) {\r\n this._delayLoadingFunction(data, this);\r\n } else {\r\n this._delayLoadingFunction(JSON.parse(data), this);\r\n }\r\n\r\n this.instances.forEach((instance) => {\r\n instance.refreshBoundingInfo();\r\n instance._syncSubMeshes();\r\n });\r\n\r\n this.delayLoadState = Constants.DELAYLOADSTATE_LOADED;\r\n scene.removePendingData(this);\r\n },\r\n () => {},\r\n scene.offlineProvider,\r\n getBinaryData\r\n );\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns `true` if the mesh is within the frustum defined by the passed array of planes.\r\n * A mesh is in the frustum if its bounding box intersects the frustum\r\n * @param frustumPlanes defines the frustum to test\r\n * @returns true if the mesh is in the frustum planes\r\n */\r\n public isInFrustum(frustumPlanes: Plane[]): boolean {\r\n if (this.delayLoadState === Constants.DELAYLOADSTATE_LOADING) {\r\n return false;\r\n }\r\n\r\n if (!super.isInFrustum(frustumPlanes)) {\r\n return false;\r\n }\r\n\r\n this._checkDelayState();\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Sets the mesh material by the material or multiMaterial `id` property\r\n * @param id is a string identifying the material or the multiMaterial\r\n * @returns the current mesh\r\n */\r\n public setMaterialById(id: string): Mesh {\r\n const materials = this.getScene().materials;\r\n let index: number;\r\n for (index = materials.length - 1; index > -1; index--) {\r\n if (materials[index].id === id) {\r\n this.material = materials[index];\r\n return this;\r\n }\r\n }\r\n\r\n // Multi\r\n const multiMaterials = this.getScene().multiMaterials;\r\n for (index = multiMaterials.length - 1; index > -1; index--) {\r\n if (multiMaterials[index].id === id) {\r\n this.material = multiMaterials[index];\r\n return this;\r\n }\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns as a new array populated with the mesh material and/or skeleton, if any.\r\n * @returns an array of IAnimatable\r\n */\r\n public getAnimatables(): IAnimatable[] {\r\n const results = new Array();\r\n\r\n if (this.material) {\r\n results.push(this.material);\r\n }\r\n\r\n if (this.skeleton) {\r\n results.push(this.skeleton);\r\n }\r\n\r\n return results;\r\n }\r\n\r\n /**\r\n * Modifies the mesh geometry according to the passed transformation matrix.\r\n * This method returns nothing, but it really modifies the mesh even if it's originally not set as updatable.\r\n * The mesh normals are modified using the same transformation.\r\n * Note that, under the hood, this method sets a new VertexBuffer each call.\r\n * @param transform defines the transform matrix to use\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/center_origin/bakingTransforms\r\n * @returns the current mesh\r\n */\r\n public bakeTransformIntoVertices(transform: Matrix): Mesh {\r\n // Position\r\n if (!this.isVerticesDataPresent(VertexBuffer.PositionKind)) {\r\n return this;\r\n }\r\n\r\n const submeshes = this.subMeshes.splice(0);\r\n\r\n this._resetPointsArrayCache();\r\n\r\n let data = this.getVerticesData(VertexBuffer.PositionKind);\r\n\r\n const temp = Vector3.Zero();\r\n let index: number;\r\n for (index = 0; index < data.length; index += 3) {\r\n Vector3.TransformCoordinatesFromFloatsToRef(data[index], data[index + 1], data[index + 2], transform, temp).toArray(data, index);\r\n }\r\n\r\n this.setVerticesData(VertexBuffer.PositionKind, data, (this.getVertexBuffer(VertexBuffer.PositionKind)).isUpdatable());\r\n\r\n // Normals\r\n if (this.isVerticesDataPresent(VertexBuffer.NormalKind)) {\r\n data = this.getVerticesData(VertexBuffer.NormalKind);\r\n for (index = 0; index < data.length; index += 3) {\r\n Vector3.TransformNormalFromFloatsToRef(data[index], data[index + 1], data[index + 2], transform, temp)\r\n .normalize()\r\n .toArray(data, index);\r\n }\r\n this.setVerticesData(VertexBuffer.NormalKind, data, (this.getVertexBuffer(VertexBuffer.NormalKind)).isUpdatable());\r\n }\r\n\r\n // flip faces?\r\n if (transform.determinant() < 0) {\r\n this.flipFaces();\r\n }\r\n\r\n // Restore submeshes\r\n this.releaseSubMeshes();\r\n this.subMeshes = submeshes;\r\n return this;\r\n }\r\n\r\n /**\r\n * Modifies the mesh geometry according to its own current World Matrix.\r\n * The mesh World Matrix is then reset.\r\n * This method returns nothing but really modifies the mesh even if it's originally not set as updatable.\r\n * Note that, under the hood, this method sets a new VertexBuffer each call.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/center_origin/bakingTransforms\r\n * @param bakeIndependentlyOfChildren indicates whether to preserve all child nodes' World Matrix during baking\r\n * @returns the current mesh\r\n */\r\n public bakeCurrentTransformIntoVertices(bakeIndependentlyOfChildren: boolean = true): Mesh {\r\n this.bakeTransformIntoVertices(this.computeWorldMatrix(true));\r\n this.resetLocalMatrix(bakeIndependentlyOfChildren);\r\n return this;\r\n }\r\n\r\n // Cache\r\n\r\n /** @internal */\r\n public get _positions(): Nullable {\r\n if (this._internalAbstractMeshDataInfo._positions) {\r\n return this._internalAbstractMeshDataInfo._positions;\r\n }\r\n\r\n if (this._geometry) {\r\n return this._geometry._positions;\r\n }\r\n return null;\r\n }\r\n\r\n /** @internal */\r\n public _resetPointsArrayCache(): Mesh {\r\n if (this._geometry) {\r\n this._geometry._resetPointsArrayCache();\r\n }\r\n return this;\r\n }\r\n\r\n /** @internal */\r\n public _generatePointsArray(): boolean {\r\n if (this._geometry) {\r\n return this._geometry._generatePointsArray();\r\n }\r\n return false;\r\n }\r\n\r\n /**\r\n * Returns a new Mesh object generated from the current mesh properties.\r\n * This method must not get confused with createInstance()\r\n * @param name is a string, the name given to the new mesh\r\n * @param newParent can be any Node object (default `null`)\r\n * @param doNotCloneChildren allows/denies the recursive cloning of the original mesh children if any (default `false`)\r\n * @param clonePhysicsImpostor allows/denies the cloning in the same time of the original mesh `body` used by the physics engine, if any (default `true`)\r\n * @returns a new mesh\r\n */\r\n public clone(name: string = \"\", newParent: Nullable = null, doNotCloneChildren?: boolean, clonePhysicsImpostor: boolean = true): Mesh {\r\n return new Mesh(name, this.getScene(), newParent, this, doNotCloneChildren, clonePhysicsImpostor);\r\n }\r\n\r\n /**\r\n * Releases resources associated with this mesh.\r\n * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default)\r\n * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default)\r\n */\r\n public dispose(doNotRecurse?: boolean, disposeMaterialAndTextures = false): void {\r\n this.morphTargetManager = null;\r\n\r\n if (this._geometry) {\r\n this._geometry.releaseForMesh(this, true);\r\n }\r\n\r\n const internalDataInfo = this._internalMeshDataInfo;\r\n\r\n if (internalDataInfo._onBeforeDrawObservable) {\r\n internalDataInfo._onBeforeDrawObservable.clear();\r\n }\r\n\r\n if (internalDataInfo._onBeforeBindObservable) {\r\n internalDataInfo._onBeforeBindObservable.clear();\r\n }\r\n\r\n if (internalDataInfo._onBeforeRenderObservable) {\r\n internalDataInfo._onBeforeRenderObservable.clear();\r\n }\r\n\r\n if (internalDataInfo._onAfterRenderObservable) {\r\n internalDataInfo._onAfterRenderObservable.clear();\r\n }\r\n\r\n if (internalDataInfo._onBetweenPassObservable) {\r\n internalDataInfo._onBetweenPassObservable.clear();\r\n }\r\n\r\n // Sources\r\n if (this._scene.useClonedMeshMap) {\r\n if (internalDataInfo.meshMap) {\r\n for (const uniqueId in internalDataInfo.meshMap) {\r\n const mesh = internalDataInfo.meshMap[uniqueId];\r\n if (mesh) {\r\n mesh._internalMeshDataInfo._source = null;\r\n internalDataInfo.meshMap[uniqueId] = undefined;\r\n }\r\n }\r\n }\r\n\r\n if (internalDataInfo._source && internalDataInfo._source._internalMeshDataInfo.meshMap) {\r\n internalDataInfo._source._internalMeshDataInfo.meshMap[this.uniqueId] = undefined;\r\n }\r\n } else {\r\n const meshes = this.getScene().meshes;\r\n for (const abstractMesh of meshes) {\r\n const mesh = abstractMesh as Mesh;\r\n if (mesh._internalMeshDataInfo && mesh._internalMeshDataInfo._source && mesh._internalMeshDataInfo._source === this) {\r\n mesh._internalMeshDataInfo._source = null;\r\n }\r\n }\r\n }\r\n\r\n internalDataInfo._source = null;\r\n this._instanceDataStorage.visibleInstances = {};\r\n\r\n // Instances\r\n this._disposeInstanceSpecificData();\r\n\r\n // Thin instances\r\n this._disposeThinInstanceSpecificData();\r\n\r\n if (this._internalMeshDataInfo._checkReadinessObserver) {\r\n this._scene.onBeforeRenderObservable.remove(this._internalMeshDataInfo._checkReadinessObserver);\r\n }\r\n\r\n super.dispose(doNotRecurse, disposeMaterialAndTextures);\r\n }\r\n\r\n /** @internal */\r\n public _disposeInstanceSpecificData() {\r\n // Do nothing\r\n }\r\n\r\n /** @internal */\r\n public _disposeThinInstanceSpecificData() {\r\n // Do nothing\r\n }\r\n\r\n /** @internal */\r\n public _invalidateInstanceVertexArrayObject() {\r\n // Do nothing\r\n }\r\n\r\n /**\r\n * Modifies the mesh geometry according to a displacement map.\r\n * A displacement map is a colored image. Each pixel color value (actually a gradient computed from red, green, blue values) will give the displacement to apply to each mesh vertex.\r\n * The mesh must be set as updatable. Its internal geometry is directly modified, no new buffer are allocated.\r\n * @param url is a string, the URL from the image file is to be downloaded.\r\n * @param minHeight is the lower limit of the displacement.\r\n * @param maxHeight is the upper limit of the displacement.\r\n * @param onSuccess is an optional Javascript function to be called just after the mesh is modified. It is passed the modified mesh and must return nothing.\r\n * @param uvOffset is an optional vector2 used to offset UV.\r\n * @param uvScale is an optional vector2 used to scale UV.\r\n * @param forceUpdate defines whether or not to force an update of the generated buffers. This is useful to apply on a deserialized model for instance.\r\n * @returns the Mesh.\r\n */\r\n public applyDisplacementMap(\r\n url: string,\r\n minHeight: number,\r\n maxHeight: number,\r\n onSuccess?: (mesh: Mesh) => void,\r\n uvOffset?: Vector2,\r\n uvScale?: Vector2,\r\n forceUpdate = false\r\n ): Mesh {\r\n const scene = this.getScene();\r\n\r\n const onload = (img: HTMLImageElement | ImageBitmap) => {\r\n // Getting height map data\r\n const heightMapWidth = img.width;\r\n const heightMapHeight = img.height;\r\n const canvas = this.getEngine().createCanvas(heightMapWidth, heightMapHeight);\r\n const context = canvas.getContext(\"2d\");\r\n\r\n context.drawImage(img, 0, 0);\r\n\r\n // Create VertexData from map data\r\n //Cast is due to wrong definition in lib.d.ts from ts 1.3 - https://github.com/Microsoft/TypeScript/issues/949\r\n const buffer = (context.getImageData(0, 0, heightMapWidth, heightMapHeight).data);\r\n\r\n this.applyDisplacementMapFromBuffer(buffer, heightMapWidth, heightMapHeight, minHeight, maxHeight, uvOffset, uvScale, forceUpdate);\r\n //execute success callback, if set\r\n if (onSuccess) {\r\n onSuccess(this);\r\n }\r\n };\r\n\r\n Tools.LoadImage(url, onload, () => {}, scene.offlineProvider);\r\n return this;\r\n }\r\n\r\n /**\r\n * Modifies the mesh geometry according to a displacementMap buffer.\r\n * A displacement map is a colored image. Each pixel color value (actually a gradient computed from red, green, blue values) will give the displacement to apply to each mesh vertex.\r\n * The mesh must be set as updatable. Its internal geometry is directly modified, no new buffer are allocated.\r\n * @param buffer is a `Uint8Array` buffer containing series of `Uint8` lower than 255, the red, green, blue and alpha values of each successive pixel.\r\n * @param heightMapWidth is the width of the buffer image.\r\n * @param heightMapHeight is the height of the buffer image.\r\n * @param minHeight is the lower limit of the displacement.\r\n * @param maxHeight is the upper limit of the displacement.\r\n * @param uvOffset is an optional vector2 used to offset UV.\r\n * @param uvScale is an optional vector2 used to scale UV.\r\n * @param forceUpdate defines whether or not to force an update of the generated buffers. This is useful to apply on a deserialized model for instance.\r\n * @returns the Mesh.\r\n */\r\n public applyDisplacementMapFromBuffer(\r\n buffer: Uint8Array,\r\n heightMapWidth: number,\r\n heightMapHeight: number,\r\n minHeight: number,\r\n maxHeight: number,\r\n uvOffset?: Vector2,\r\n uvScale?: Vector2,\r\n forceUpdate = false\r\n ): Mesh {\r\n if (!this.isVerticesDataPresent(VertexBuffer.PositionKind) || !this.isVerticesDataPresent(VertexBuffer.NormalKind) || !this.isVerticesDataPresent(VertexBuffer.UVKind)) {\r\n Logger.Warn(\"Cannot call applyDisplacementMap: Given mesh is not complete. Position, Normal or UV are missing\");\r\n return this;\r\n }\r\n\r\n const positions = this.getVerticesData(VertexBuffer.PositionKind, true, true);\r\n const normals = this.getVerticesData(VertexBuffer.NormalKind);\r\n const uvs = this.getVerticesData(VertexBuffer.UVKind);\r\n let position = Vector3.Zero();\r\n const normal = Vector3.Zero();\r\n const uv = Vector2.Zero();\r\n\r\n uvOffset = uvOffset || Vector2.Zero();\r\n uvScale = uvScale || new Vector2(1, 1);\r\n\r\n for (let index = 0; index < positions.length; index += 3) {\r\n Vector3.FromArrayToRef(positions, index, position);\r\n Vector3.FromArrayToRef(normals, index, normal);\r\n Vector2.FromArrayToRef(uvs, (index / 3) * 2, uv);\r\n\r\n // Compute height\r\n const u = (Math.abs(uv.x * uvScale.x + (uvOffset.x % 1)) * (heightMapWidth - 1)) % heightMapWidth | 0;\r\n const v = (Math.abs(uv.y * uvScale.y + (uvOffset.y % 1)) * (heightMapHeight - 1)) % heightMapHeight | 0;\r\n\r\n const pos = (u + v * heightMapWidth) * 4;\r\n const r = buffer[pos] / 255.0;\r\n const g = buffer[pos + 1] / 255.0;\r\n const b = buffer[pos + 2] / 255.0;\r\n\r\n const gradient = r * 0.3 + g * 0.59 + b * 0.11;\r\n\r\n normal.normalize();\r\n normal.scaleInPlace(minHeight + (maxHeight - minHeight) * gradient);\r\n position = position.add(normal);\r\n\r\n position.toArray(positions, index);\r\n }\r\n\r\n VertexData.ComputeNormals(positions, this.getIndices(), normals);\r\n\r\n if (forceUpdate) {\r\n this.setVerticesData(VertexBuffer.PositionKind, positions);\r\n this.setVerticesData(VertexBuffer.NormalKind, normals);\r\n this.setVerticesData(VertexBuffer.UVKind, uvs);\r\n } else {\r\n this.updateVerticesData(VertexBuffer.PositionKind, positions);\r\n this.updateVerticesData(VertexBuffer.NormalKind, normals);\r\n }\r\n return this;\r\n }\r\n\r\n private _getFlattenedNormals(indices: IndicesArray, positions: FloatArray): Float32Array {\r\n const normals = new Float32Array(indices.length * 3);\r\n let normalsCount = 0;\r\n\r\n // Decide if normals should be flipped\r\n const flipNormalGeneration =\r\n this.overrideMaterialSideOrientation ===\r\n (this._scene.useRightHandedSystem ? Constants.MATERIAL_CounterClockWiseSideOrientation : Constants.MATERIAL_ClockWiseSideOrientation);\r\n\r\n // Generate new normals\r\n for (let index = 0; index < indices.length; index += 3) {\r\n const p1 = Vector3.FromArray(positions, indices[index] * 3);\r\n const p2 = Vector3.FromArray(positions, indices[index + 1] * 3);\r\n const p3 = Vector3.FromArray(positions, indices[index + 2] * 3);\r\n\r\n const p1p2 = p1.subtract(p2);\r\n const p3p2 = p3.subtract(p2);\r\n\r\n const normal = Vector3.Normalize(Vector3.Cross(p1p2, p3p2));\r\n if (flipNormalGeneration) {\r\n normal.scaleInPlace(-1);\r\n }\r\n\r\n // Store same normals for every vertex\r\n for (let localIndex = 0; localIndex < 3; localIndex++) {\r\n normals[normalsCount++] = normal.x;\r\n normals[normalsCount++] = normal.y;\r\n normals[normalsCount++] = normal.z;\r\n }\r\n }\r\n\r\n return normals;\r\n }\r\n\r\n private _convertToUnIndexedMesh(flattenNormals: boolean = false): Mesh {\r\n const kinds = this.getVerticesDataKinds();\r\n const indices = this.getIndices()!;\r\n const data: { [kind: string]: FloatArray } = {};\r\n\r\n const separateVertices = (data: FloatArray, stride: number): Float32Array => {\r\n const newData = new Float32Array(indices.length * stride);\r\n let count = 0;\r\n for (let index = 0; index < indices.length; index++) {\r\n for (let offset = 0; offset < stride; offset++) {\r\n newData[count++] = data[indices[index] * stride + offset];\r\n }\r\n }\r\n return newData;\r\n };\r\n\r\n // Save previous submeshes\r\n const previousSubmeshes = this.geometry ? this.subMeshes.slice(0) : [];\r\n\r\n // Cache vertex data\r\n for (const kind of kinds) {\r\n data[kind] = this.getVerticesData(kind)!;\r\n }\r\n\r\n // Update vertex data\r\n for (const kind of kinds) {\r\n const vertexBuffer = this.getVertexBuffer(kind)!;\r\n const stride = vertexBuffer.getStrideSize();\r\n\r\n if (flattenNormals && kind === VertexBuffer.NormalKind) {\r\n const normals = this._getFlattenedNormals(indices, data[VertexBuffer.PositionKind]);\r\n this.setVerticesData(VertexBuffer.NormalKind, normals, vertexBuffer.isUpdatable(), stride);\r\n } else {\r\n this.setVerticesData(kind, separateVertices(data[kind], stride), vertexBuffer.isUpdatable(), stride);\r\n }\r\n }\r\n\r\n // Update morph targets\r\n if (this.morphTargetManager) {\r\n for (let targetIndex = 0; targetIndex < this.morphTargetManager.numTargets; targetIndex++) {\r\n const target = this.morphTargetManager.getTarget(targetIndex);\r\n\r\n const positions = target.getPositions()!;\r\n target.setPositions(separateVertices(positions, 3));\r\n\r\n const normals = target.getNormals();\r\n if (normals) {\r\n target.setNormals(flattenNormals ? this._getFlattenedNormals(indices, positions) : separateVertices(normals, 3));\r\n }\r\n\r\n const tangents = target.getTangents();\r\n if (tangents) {\r\n target.setTangents(separateVertices(tangents, 3));\r\n }\r\n\r\n const uvs = target.getUVs();\r\n if (uvs) {\r\n target.setUVs(separateVertices(uvs, 2));\r\n }\r\n }\r\n this.morphTargetManager.synchronize();\r\n }\r\n\r\n // Update indices\r\n for (let index = 0; index < indices.length; index++) {\r\n indices[index] = index;\r\n }\r\n this.setIndices(indices);\r\n\r\n this._unIndexed = true;\r\n\r\n // Update submeshes\r\n this.releaseSubMeshes();\r\n for (const previousOne of previousSubmeshes) {\r\n SubMesh.AddToMesh(previousOne.materialIndex, previousOne.indexStart, previousOne.indexCount, previousOne.indexStart, previousOne.indexCount, this);\r\n }\r\n\r\n this.synchronizeInstances();\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Modify the mesh to get a flat shading rendering.\r\n * This means each mesh facet will then have its own normals. Usually new vertices are added in the mesh geometry to get this result.\r\n * Warning : the mesh is really modified even if not set originally as updatable and, under the hood, a new VertexBuffer is allocated.\r\n * @returns current mesh\r\n */\r\n public convertToFlatShadedMesh(): Mesh {\r\n return this._convertToUnIndexedMesh(true);\r\n }\r\n\r\n /**\r\n * This method removes all the mesh indices and add new vertices (duplication) in order to unfold facets into buffers.\r\n * In other words, more vertices, no more indices and a single bigger VBO.\r\n * The mesh is really modified even if not set originally as updatable. Under the hood, a new VertexBuffer is allocated.\r\n * @returns current mesh\r\n */\r\n public convertToUnIndexedMesh(): Mesh {\r\n return this._convertToUnIndexedMesh();\r\n }\r\n\r\n /**\r\n * Inverses facet orientations.\r\n * Warning : the mesh is really modified even if not set originally as updatable. A new VertexBuffer is created under the hood each call.\r\n * @param flipNormals will also inverts the normals\r\n * @returns current mesh\r\n */\r\n public flipFaces(flipNormals: boolean = false): Mesh {\r\n const vertex_data = VertexData.ExtractFromMesh(this);\r\n let i: number;\r\n if (flipNormals && this.isVerticesDataPresent(VertexBuffer.NormalKind) && vertex_data.normals) {\r\n for (i = 0; i < vertex_data.normals.length; i++) {\r\n vertex_data.normals[i] *= -1;\r\n }\r\n }\r\n\r\n if (vertex_data.indices) {\r\n let temp;\r\n for (i = 0; i < vertex_data.indices.length; i += 3) {\r\n // reassign indices\r\n temp = vertex_data.indices[i + 1];\r\n vertex_data.indices[i + 1] = vertex_data.indices[i + 2];\r\n vertex_data.indices[i + 2] = temp;\r\n }\r\n }\r\n\r\n vertex_data.applyToMesh(this, this.isVertexBufferUpdatable(VertexBuffer.PositionKind));\r\n return this;\r\n }\r\n\r\n /**\r\n * Increase the number of facets and hence vertices in a mesh\r\n * Vertex normals are interpolated from existing vertex normals\r\n * Warning : the mesh is really modified even if not set originally as updatable. A new VertexBuffer is created under the hood each call.\r\n * @param numberPerEdge the number of new vertices to add to each edge of a facet, optional default 1\r\n */\r\n public increaseVertices(numberPerEdge: number = 1): void {\r\n const vertex_data = VertexData.ExtractFromMesh(this);\r\n const currentIndices = vertex_data.indices && !Array.isArray(vertex_data.indices) && Array.from ? Array.from(vertex_data.indices) : vertex_data.indices;\r\n const positions = vertex_data.positions && !Array.isArray(vertex_data.positions) && Array.from ? Array.from(vertex_data.positions) : vertex_data.positions;\r\n const uvs = vertex_data.uvs && !Array.isArray(vertex_data.uvs) && Array.from ? Array.from(vertex_data.uvs) : vertex_data.uvs;\r\n const normals = vertex_data.normals && !Array.isArray(vertex_data.normals) && Array.from ? Array.from(vertex_data.normals) : vertex_data.normals;\r\n\r\n if (!currentIndices || !positions) {\r\n Logger.Warn(\"Couldn't increase number of vertices : VertexData must contain at least indices and positions\");\r\n } else {\r\n vertex_data.indices = currentIndices;\r\n vertex_data.positions = positions;\r\n if (uvs) {\r\n vertex_data.uvs = uvs;\r\n }\r\n if (normals) {\r\n vertex_data.normals = normals;\r\n }\r\n\r\n const segments: number = numberPerEdge + 1; //segments per current facet edge, become sides of new facets\r\n const tempIndices: Array> = new Array();\r\n for (let i = 0; i < segments + 1; i++) {\r\n tempIndices[i] = new Array();\r\n }\r\n let a: number; //vertex index of one end of a side\r\n let b: number; //vertex index of other end of the side\r\n const deltaPosition: Vector3 = new Vector3(0, 0, 0);\r\n const deltaNormal: Vector3 = new Vector3(0, 0, 0);\r\n const deltaUV: Vector2 = new Vector2(0, 0);\r\n const indices: number[] = new Array();\r\n const vertexIndex: number[] = new Array();\r\n const side: Array>> = new Array();\r\n let len: number;\r\n let positionPtr: number = positions.length;\r\n let uvPtr: number;\r\n if (uvs) {\r\n uvPtr = uvs.length;\r\n }\r\n let normalsPtr: number;\r\n if (normals) {\r\n normalsPtr = normals.length;\r\n }\r\n\r\n for (let i = 0; i < currentIndices.length; i += 3) {\r\n vertexIndex[0] = currentIndices[i];\r\n vertexIndex[1] = currentIndices[i + 1];\r\n vertexIndex[2] = currentIndices[i + 2];\r\n for (let j = 0; j < 3; j++) {\r\n a = vertexIndex[j];\r\n b = vertexIndex[(j + 1) % 3];\r\n if (side[a] === undefined && side[b] === undefined) {\r\n side[a] = new Array();\r\n side[b] = new Array();\r\n } else {\r\n if (side[a] === undefined) {\r\n side[a] = new Array();\r\n }\r\n if (side[b] === undefined) {\r\n side[b] = new Array();\r\n }\r\n }\r\n if (side[a][b] === undefined && side[b][a] === undefined) {\r\n side[a][b] = [];\r\n deltaPosition.x = (positions[3 * b] - positions[3 * a]) / segments;\r\n deltaPosition.y = (positions[3 * b + 1] - positions[3 * a + 1]) / segments;\r\n deltaPosition.z = (positions[3 * b + 2] - positions[3 * a + 2]) / segments;\r\n if (normals) {\r\n deltaNormal.x = (normals[3 * b] - normals[3 * a]) / segments;\r\n deltaNormal.y = (normals[3 * b + 1] - normals[3 * a + 1]) / segments;\r\n deltaNormal.z = (normals[3 * b + 2] - normals[3 * a + 2]) / segments;\r\n }\r\n if (uvs) {\r\n deltaUV.x = (uvs[2 * b] - uvs[2 * a]) / segments;\r\n deltaUV.y = (uvs[2 * b + 1] - uvs[2 * a + 1]) / segments;\r\n }\r\n side[a][b].push(a);\r\n for (let k = 1; k < segments; k++) {\r\n side[a][b].push(positions.length / 3);\r\n positions[positionPtr++] = positions[3 * a] + k * deltaPosition.x;\r\n positions[positionPtr++] = positions[3 * a + 1] + k * deltaPosition.y;\r\n positions[positionPtr++] = positions[3 * a + 2] + k * deltaPosition.z;\r\n if (normals) {\r\n normals[normalsPtr!++] = normals[3 * a] + k * deltaNormal.x;\r\n normals[normalsPtr!++] = normals[3 * a + 1] + k * deltaNormal.y;\r\n normals[normalsPtr!++] = normals[3 * a + 2] + k * deltaNormal.z;\r\n }\r\n if (uvs) {\r\n uvs[uvPtr!++] = uvs[2 * a] + k * deltaUV.x;\r\n uvs[uvPtr!++] = uvs[2 * a + 1] + k * deltaUV.y;\r\n }\r\n }\r\n side[a][b].push(b);\r\n side[b][a] = new Array();\r\n len = side[a][b].length;\r\n for (let idx = 0; idx < len; idx++) {\r\n side[b][a][idx] = side[a][b][len - 1 - idx];\r\n }\r\n }\r\n }\r\n //Calculate positions, normals and uvs of new internal vertices\r\n tempIndices[0][0] = currentIndices[i];\r\n tempIndices[1][0] = side[currentIndices[i]][currentIndices[i + 1]][1];\r\n tempIndices[1][1] = side[currentIndices[i]][currentIndices[i + 2]][1];\r\n for (let k = 2; k < segments; k++) {\r\n tempIndices[k][0] = side[currentIndices[i]][currentIndices[i + 1]][k];\r\n tempIndices[k][k] = side[currentIndices[i]][currentIndices[i + 2]][k];\r\n deltaPosition.x = (positions[3 * tempIndices[k][k]] - positions[3 * tempIndices[k][0]]) / k;\r\n deltaPosition.y = (positions[3 * tempIndices[k][k] + 1] - positions[3 * tempIndices[k][0] + 1]) / k;\r\n deltaPosition.z = (positions[3 * tempIndices[k][k] + 2] - positions[3 * tempIndices[k][0] + 2]) / k;\r\n if (normals) {\r\n deltaNormal.x = (normals[3 * tempIndices[k][k]] - normals[3 * tempIndices[k][0]]) / k;\r\n deltaNormal.y = (normals[3 * tempIndices[k][k] + 1] - normals[3 * tempIndices[k][0] + 1]) / k;\r\n deltaNormal.z = (normals[3 * tempIndices[k][k] + 2] - normals[3 * tempIndices[k][0] + 2]) / k;\r\n }\r\n if (uvs) {\r\n deltaUV.x = (uvs[2 * tempIndices[k][k]] - uvs[2 * tempIndices[k][0]]) / k;\r\n deltaUV.y = (uvs[2 * tempIndices[k][k] + 1] - uvs[2 * tempIndices[k][0] + 1]) / k;\r\n }\r\n for (let j = 1; j < k; j++) {\r\n tempIndices[k][j] = positions.length / 3;\r\n positions[positionPtr++] = positions[3 * tempIndices[k][0]] + j * deltaPosition.x;\r\n positions[positionPtr++] = positions[3 * tempIndices[k][0] + 1] + j * deltaPosition.y;\r\n positions[positionPtr++] = positions[3 * tempIndices[k][0] + 2] + j * deltaPosition.z;\r\n if (normals) {\r\n normals[normalsPtr!++] = normals[3 * tempIndices[k][0]] + j * deltaNormal.x;\r\n normals[normalsPtr!++] = normals[3 * tempIndices[k][0] + 1] + j * deltaNormal.y;\r\n normals[normalsPtr!++] = normals[3 * tempIndices[k][0] + 2] + j * deltaNormal.z;\r\n }\r\n if (uvs) {\r\n uvs[uvPtr!++] = uvs[2 * tempIndices[k][0]] + j * deltaUV.x;\r\n uvs[uvPtr!++] = uvs[2 * tempIndices[k][0] + 1] + j * deltaUV.y;\r\n }\r\n }\r\n }\r\n tempIndices[segments] = side[currentIndices[i + 1]][currentIndices[i + 2]];\r\n\r\n // reform indices\r\n indices.push(tempIndices[0][0], tempIndices[1][0], tempIndices[1][1]);\r\n for (let k = 1; k < segments; k++) {\r\n let j: number;\r\n for (j = 0; j < k; j++) {\r\n indices.push(tempIndices[k][j], tempIndices[k + 1][j], tempIndices[k + 1][j + 1]);\r\n indices.push(tempIndices[k][j], tempIndices[k + 1][j + 1], tempIndices[k][j + 1]);\r\n }\r\n indices.push(tempIndices[k][j], tempIndices[k + 1][j], tempIndices[k + 1][j + 1]);\r\n }\r\n }\r\n\r\n vertex_data.indices = indices;\r\n vertex_data.applyToMesh(this, this.isVertexBufferUpdatable(VertexBuffer.PositionKind));\r\n }\r\n }\r\n\r\n /**\r\n * Force adjacent facets to share vertices and remove any facets that have all vertices in a line\r\n * This will undo any application of covertToFlatShadedMesh\r\n * Warning : the mesh is really modified even if not set originally as updatable. A new VertexBuffer is created under the hood each call.\r\n */\r\n public forceSharedVertices(): void {\r\n const vertex_data = VertexData.ExtractFromMesh(this);\r\n const currentUVs = vertex_data.uvs;\r\n const currentIndices = vertex_data.indices;\r\n const currentPositions = vertex_data.positions;\r\n const currentColors = vertex_data.colors;\r\n const currentMatrixIndices = vertex_data.matricesIndices;\r\n const currentMatrixWeights = vertex_data.matricesWeights;\r\n const currentMatrixIndicesExtra = vertex_data.matricesIndicesExtra;\r\n const currentMatrixWeightsExtra = vertex_data.matricesWeightsExtra;\r\n\r\n if (currentIndices === void 0 || currentPositions === void 0 || currentIndices === null || currentPositions === null) {\r\n Logger.Warn(\"VertexData contains empty entries\");\r\n } else {\r\n const positions: Array = new Array();\r\n const indices: Array = new Array();\r\n const uvs: Array = new Array();\r\n const colors: Array = new Array();\r\n const matrixIndices: Array = new Array();\r\n const matrixWeights: Array = new Array();\r\n const matrixIndicesExtra: Array = new Array();\r\n const matrixWeightsExtra: Array = new Array();\r\n let pstring: Array = new Array(); //lists facet vertex positions (a,b,c) as string \"a|b|c\"\r\n\r\n let indexPtr: number = 0; // pointer to next available index value\r\n const uniquePositions: { [key: string]: number } = {}; // unique vertex positions\r\n let ptr: number; // pointer to element in uniquePositions\r\n let facet: Array;\r\n\r\n for (let i = 0; i < currentIndices.length; i += 3) {\r\n facet = [currentIndices[i], currentIndices[i + 1], currentIndices[i + 2]]; //facet vertex indices\r\n pstring = new Array();\r\n for (let j = 0; j < 3; j++) {\r\n pstring[j] = \"\";\r\n for (let k = 0; k < 3; k++) {\r\n //small values make 0\r\n if (Math.abs(currentPositions[3 * facet[j] + k]) < 0.00000001) {\r\n currentPositions[3 * facet[j] + k] = 0;\r\n }\r\n pstring[j] += currentPositions[3 * facet[j] + k] + \"|\";\r\n }\r\n }\r\n //check facet vertices to see that none are repeated\r\n // do not process any facet that has a repeated vertex, ie is a line\r\n if (!(pstring[0] == pstring[1] || pstring[0] == pstring[2] || pstring[1] == pstring[2])) {\r\n //for each facet position check if already listed in uniquePositions\r\n // if not listed add to uniquePositions and set index pointer\r\n // if listed use its index in uniquePositions and new index pointer\r\n for (let j = 0; j < 3; j++) {\r\n ptr = uniquePositions[pstring[j]];\r\n if (ptr === undefined) {\r\n uniquePositions[pstring[j]] = indexPtr;\r\n ptr = indexPtr++;\r\n //not listed so add individual x, y, z coordinates to positions\r\n for (let k = 0; k < 3; k++) {\r\n positions.push(currentPositions[3 * facet[j] + k]);\r\n }\r\n if (currentColors !== null && currentColors !== void 0) {\r\n for (let k = 0; k < 4; k++) {\r\n colors.push(currentColors[4 * facet[j] + k]);\r\n }\r\n }\r\n if (currentUVs !== null && currentUVs !== void 0) {\r\n for (let k = 0; k < 2; k++) {\r\n uvs.push(currentUVs[2 * facet[j] + k]);\r\n }\r\n }\r\n if (currentMatrixIndices !== null && currentMatrixIndices !== void 0) {\r\n for (let k = 0; k < 4; k++) {\r\n matrixIndices.push(currentMatrixIndices[4 * facet[j] + k]);\r\n }\r\n }\r\n if (currentMatrixWeights !== null && currentMatrixWeights !== void 0) {\r\n for (let k = 0; k < 4; k++) {\r\n matrixWeights.push(currentMatrixWeights[4 * facet[j] + k]);\r\n }\r\n }\r\n if (currentMatrixIndicesExtra !== null && currentMatrixIndicesExtra !== void 0) {\r\n for (let k = 0; k < 4; k++) {\r\n matrixIndicesExtra.push(currentMatrixIndicesExtra[4 * facet[j] + k]);\r\n }\r\n }\r\n if (currentMatrixWeightsExtra !== null && currentMatrixWeightsExtra !== void 0) {\r\n for (let k = 0; k < 4; k++) {\r\n matrixWeightsExtra.push(currentMatrixWeightsExtra[4 * facet[j] + k]);\r\n }\r\n }\r\n }\r\n // add new index pointer to indices array\r\n indices.push(ptr);\r\n }\r\n }\r\n }\r\n\r\n const normals: Array = new Array();\r\n VertexData.ComputeNormals(positions, indices, normals);\r\n\r\n //create new vertex data object and update\r\n vertex_data.positions = positions;\r\n vertex_data.indices = indices;\r\n vertex_data.normals = normals;\r\n if (currentUVs !== null && currentUVs !== void 0) {\r\n vertex_data.uvs = uvs;\r\n }\r\n if (currentColors !== null && currentColors !== void 0) {\r\n vertex_data.colors = colors;\r\n }\r\n if (currentMatrixIndices !== null && currentMatrixIndices !== void 0) {\r\n vertex_data.matricesIndices = matrixIndices;\r\n }\r\n if (currentMatrixWeights !== null && currentMatrixWeights !== void 0) {\r\n vertex_data.matricesWeights = matrixWeights;\r\n }\r\n if (currentMatrixIndicesExtra !== null && currentMatrixIndicesExtra !== void 0) {\r\n vertex_data.matricesIndicesExtra = matrixIndicesExtra;\r\n }\r\n if (currentMatrixWeights !== null && currentMatrixWeights !== void 0) {\r\n vertex_data.matricesWeightsExtra = matrixWeightsExtra;\r\n }\r\n\r\n vertex_data.applyToMesh(this, this.isVertexBufferUpdatable(VertexBuffer.PositionKind));\r\n }\r\n }\r\n\r\n // Instances\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/naming-convention\r\n public static _instancedMeshFactory(name: string, mesh: Mesh): InstancedMesh {\r\n throw _WarnImport(\"InstancedMesh\");\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public static _PhysicsImpostorParser(scene: Scene, physicObject: IPhysicsEnabledObject, jsonObject: any): PhysicsImpostor {\r\n throw _WarnImport(\"PhysicsImpostor\");\r\n }\r\n\r\n /**\r\n * Creates a new InstancedMesh object from the mesh model.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/copies/instances\r\n * @param name defines the name of the new instance\r\n * @returns a new InstancedMesh\r\n */\r\n public createInstance(name: string): InstancedMesh {\r\n return Mesh._instancedMeshFactory(name, this);\r\n }\r\n\r\n /**\r\n * Synchronises all the mesh instance submeshes to the current mesh submeshes, if any.\r\n * After this call, all the mesh instances have the same submeshes than the current mesh.\r\n * @returns the current mesh\r\n */\r\n public synchronizeInstances(): Mesh {\r\n for (let instanceIndex = 0; instanceIndex < this.instances.length; instanceIndex++) {\r\n const instance = this.instances[instanceIndex];\r\n instance._syncSubMeshes();\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Optimization of the mesh's indices, in case a mesh has duplicated vertices.\r\n * The function will only reorder the indices and will not remove unused vertices to avoid problems with submeshes.\r\n * This should be used together with the simplification to avoid disappearing triangles.\r\n * @param successCallback an optional success callback to be called after the optimization finished.\r\n * @returns the current mesh\r\n */\r\n public optimizeIndices(successCallback?: (mesh?: Mesh) => void): Mesh {\r\n const indices = this.getIndices();\r\n const positions = this.getVerticesData(VertexBuffer.PositionKind);\r\n\r\n if (!positions || !indices) {\r\n return this;\r\n }\r\n\r\n const vectorPositions = new Array();\r\n for (let pos = 0; pos < positions.length; pos = pos + 3) {\r\n vectorPositions.push(Vector3.FromArray(positions, pos));\r\n }\r\n const dupes = new Array();\r\n\r\n AsyncLoop.SyncAsyncForLoop(\r\n vectorPositions.length,\r\n 40,\r\n (iteration) => {\r\n const realPos = vectorPositions.length - 1 - iteration;\r\n const testedPosition = vectorPositions[realPos];\r\n for (let j = 0; j < realPos; ++j) {\r\n const againstPosition = vectorPositions[j];\r\n if (testedPosition.equals(againstPosition)) {\r\n dupes[realPos] = j;\r\n break;\r\n }\r\n }\r\n },\r\n () => {\r\n for (let i = 0; i < indices.length; ++i) {\r\n indices[i] = dupes[indices[i]] || indices[i];\r\n }\r\n\r\n //indices are now reordered\r\n const originalSubMeshes = this.subMeshes.slice(0);\r\n this.setIndices(indices);\r\n this.subMeshes = originalSubMeshes;\r\n if (successCallback) {\r\n successCallback(this);\r\n }\r\n }\r\n );\r\n return this;\r\n }\r\n\r\n /**\r\n * Serialize current mesh\r\n * @param serializationObject defines the object which will receive the serialization data\r\n */\r\n public serialize(serializationObject: any = {}): any {\r\n serializationObject.name = this.name;\r\n serializationObject.id = this.id;\r\n serializationObject.uniqueId = this.uniqueId;\r\n serializationObject.type = this.getClassName();\r\n\r\n if (Tags && Tags.HasTags(this)) {\r\n serializationObject.tags = Tags.GetTags(this);\r\n }\r\n\r\n serializationObject.position = this.position.asArray();\r\n\r\n if (this.rotationQuaternion) {\r\n serializationObject.rotationQuaternion = this.rotationQuaternion.asArray();\r\n } else if (this.rotation) {\r\n serializationObject.rotation = this.rotation.asArray();\r\n }\r\n\r\n serializationObject.scaling = this.scaling.asArray();\r\n if (this._postMultiplyPivotMatrix) {\r\n serializationObject.pivotMatrix = this.getPivotMatrix().asArray();\r\n } else {\r\n serializationObject.localMatrix = this.getPivotMatrix().asArray();\r\n }\r\n\r\n serializationObject.isEnabled = this.isEnabled(false);\r\n serializationObject.isVisible = this.isVisible;\r\n serializationObject.infiniteDistance = this.infiniteDistance;\r\n serializationObject.pickable = this.isPickable;\r\n\r\n serializationObject.receiveShadows = this.receiveShadows;\r\n\r\n serializationObject.billboardMode = this.billboardMode;\r\n serializationObject.visibility = this.visibility;\r\n\r\n serializationObject.checkCollisions = this.checkCollisions;\r\n serializationObject.isBlocker = this.isBlocker;\r\n serializationObject.overrideMaterialSideOrientation = this.overrideMaterialSideOrientation;\r\n\r\n // Parent\r\n if (this.parent) {\r\n this.parent._serializeAsParent(serializationObject);\r\n }\r\n\r\n // Geometry\r\n serializationObject.isUnIndexed = this.isUnIndexed;\r\n const geometry = this._geometry;\r\n if (geometry && this.subMeshes) {\r\n serializationObject.geometryUniqueId = geometry.uniqueId;\r\n serializationObject.geometryId = geometry.id;\r\n\r\n // SubMeshes\r\n serializationObject.subMeshes = [];\r\n for (let subIndex = 0; subIndex < this.subMeshes.length; subIndex++) {\r\n const subMesh = this.subMeshes[subIndex];\r\n\r\n serializationObject.subMeshes.push({\r\n materialIndex: subMesh.materialIndex,\r\n verticesStart: subMesh.verticesStart,\r\n verticesCount: subMesh.verticesCount,\r\n indexStart: subMesh.indexStart,\r\n indexCount: subMesh.indexCount,\r\n });\r\n }\r\n }\r\n\r\n // Material\r\n if (this.material) {\r\n if (!this.material.doNotSerialize) {\r\n serializationObject.materialUniqueId = this.material.uniqueId;\r\n serializationObject.materialId = this.material.id; // back compat\r\n }\r\n } else {\r\n this.material = null;\r\n serializationObject.materialUniqueId = this._scene.defaultMaterial.uniqueId;\r\n serializationObject.materialId = this._scene.defaultMaterial.id; // back compat\r\n }\r\n\r\n // Morph targets\r\n if (this.morphTargetManager) {\r\n serializationObject.morphTargetManagerId = this.morphTargetManager.uniqueId;\r\n }\r\n\r\n // Skeleton\r\n if (this.skeleton) {\r\n serializationObject.skeletonId = this.skeleton.id;\r\n serializationObject.numBoneInfluencers = this.numBoneInfluencers;\r\n }\r\n\r\n // Physics\r\n //TODO implement correct serialization for physics impostors.\r\n if (this.getScene()._getComponent(SceneComponentConstants.NAME_PHYSICSENGINE)) {\r\n const impostor = this.getPhysicsImpostor();\r\n if (impostor) {\r\n serializationObject.physicsMass = impostor.getParam(\"mass\");\r\n serializationObject.physicsFriction = impostor.getParam(\"friction\");\r\n serializationObject.physicsRestitution = impostor.getParam(\"mass\");\r\n serializationObject.physicsImpostor = impostor.type;\r\n }\r\n }\r\n\r\n // Metadata\r\n if (this.metadata) {\r\n serializationObject.metadata = this.metadata;\r\n }\r\n\r\n // Instances\r\n serializationObject.instances = [];\r\n for (let index = 0; index < this.instances.length; index++) {\r\n const instance = this.instances[index];\r\n if (instance.doNotSerialize) {\r\n continue;\r\n }\r\n\r\n const serializationInstance: any = {\r\n name: instance.name,\r\n id: instance.id,\r\n isEnabled: instance.isEnabled(false),\r\n isVisible: instance.isVisible,\r\n isPickable: instance.isPickable,\r\n checkCollisions: instance.checkCollisions,\r\n position: instance.position.asArray(),\r\n scaling: instance.scaling.asArray(),\r\n };\r\n\r\n if (instance.parent) {\r\n instance.parent._serializeAsParent(serializationInstance);\r\n }\r\n\r\n if (instance.rotationQuaternion) {\r\n serializationInstance.rotationQuaternion = instance.rotationQuaternion.asArray();\r\n } else if (instance.rotation) {\r\n serializationInstance.rotation = instance.rotation.asArray();\r\n }\r\n\r\n // Physics\r\n //TODO implement correct serialization for physics impostors.\r\n if (this.getScene()._getComponent(SceneComponentConstants.NAME_PHYSICSENGINE)) {\r\n const impostor = instance.getPhysicsImpostor();\r\n if (impostor) {\r\n serializationInstance.physicsMass = impostor.getParam(\"mass\");\r\n serializationInstance.physicsFriction = impostor.getParam(\"friction\");\r\n serializationInstance.physicsRestitution = impostor.getParam(\"mass\");\r\n serializationInstance.physicsImpostor = impostor.type;\r\n }\r\n }\r\n\r\n // Metadata\r\n if (instance.metadata) {\r\n serializationInstance.metadata = instance.metadata;\r\n }\r\n\r\n // Action Manager\r\n if (instance.actionManager) {\r\n serializationInstance.actions = instance.actionManager.serialize(instance.name);\r\n }\r\n\r\n serializationObject.instances.push(serializationInstance);\r\n\r\n // Animations\r\n SerializationHelper.AppendSerializedAnimations(instance, serializationInstance);\r\n serializationInstance.ranges = instance.serializeAnimationRanges();\r\n }\r\n\r\n // Thin instances\r\n if (this._thinInstanceDataStorage.instancesCount && this._thinInstanceDataStorage.matrixData) {\r\n serializationObject.thinInstances = {\r\n instancesCount: this._thinInstanceDataStorage.instancesCount,\r\n matrixData: Array.from(this._thinInstanceDataStorage.matrixData),\r\n matrixBufferSize: this._thinInstanceDataStorage.matrixBufferSize,\r\n enablePicking: this.thinInstanceEnablePicking,\r\n };\r\n\r\n if (this._userThinInstanceBuffersStorage) {\r\n const userThinInstance: any = {\r\n data: {},\r\n sizes: {},\r\n strides: {},\r\n };\r\n\r\n for (const kind in this._userThinInstanceBuffersStorage.data) {\r\n userThinInstance.data[kind] = Array.from(this._userThinInstanceBuffersStorage.data[kind]);\r\n userThinInstance.sizes[kind] = this._userThinInstanceBuffersStorage.sizes[kind];\r\n userThinInstance.strides[kind] = this._userThinInstanceBuffersStorage.strides[kind];\r\n }\r\n\r\n serializationObject.thinInstances.userThinInstance = userThinInstance;\r\n }\r\n }\r\n\r\n // Animations\r\n SerializationHelper.AppendSerializedAnimations(this, serializationObject);\r\n serializationObject.ranges = this.serializeAnimationRanges();\r\n\r\n // Layer mask\r\n serializationObject.layerMask = this.layerMask;\r\n\r\n // Alpha\r\n serializationObject.alphaIndex = this.alphaIndex;\r\n serializationObject.hasVertexAlpha = this.hasVertexAlpha;\r\n\r\n // Overlay\r\n serializationObject.overlayAlpha = this.overlayAlpha;\r\n serializationObject.overlayColor = this.overlayColor.asArray();\r\n serializationObject.renderOverlay = this.renderOverlay;\r\n\r\n // Fog\r\n serializationObject.applyFog = this.applyFog;\r\n\r\n // Action Manager\r\n if (this.actionManager) {\r\n serializationObject.actions = this.actionManager.serialize(this.name);\r\n }\r\n\r\n return serializationObject;\r\n }\r\n\r\n /** @internal */\r\n public _syncGeometryWithMorphTargetManager() {\r\n if (!this.geometry) {\r\n return;\r\n }\r\n\r\n this._markSubMeshesAsAttributesDirty();\r\n\r\n const morphTargetManager = this._internalAbstractMeshDataInfo._morphTargetManager;\r\n if (morphTargetManager && morphTargetManager.vertexCount) {\r\n if (morphTargetManager.vertexCount !== this.getTotalVertices()) {\r\n Logger.Error(\"Mesh is incompatible with morph targets. Targets and mesh must all have the same vertices count.\");\r\n this.morphTargetManager = null;\r\n return;\r\n }\r\n\r\n if (morphTargetManager.isUsingTextureForTargets) {\r\n return;\r\n }\r\n\r\n for (let index = 0; index < morphTargetManager.numInfluencers; index++) {\r\n const morphTarget = morphTargetManager.getActiveTarget(index);\r\n\r\n const positions = morphTarget.getPositions();\r\n if (!positions) {\r\n Logger.Error(\"Invalid morph target. Target must have positions.\");\r\n return;\r\n }\r\n\r\n this.geometry.setVerticesData(VertexBuffer.PositionKind + index, positions, false, 3);\r\n\r\n const normals = morphTarget.getNormals();\r\n if (normals) {\r\n this.geometry.setVerticesData(VertexBuffer.NormalKind + index, normals, false, 3);\r\n }\r\n\r\n const tangents = morphTarget.getTangents();\r\n if (tangents) {\r\n this.geometry.setVerticesData(VertexBuffer.TangentKind + index, tangents, false, 3);\r\n }\r\n\r\n const uvs = morphTarget.getUVs();\r\n if (uvs) {\r\n this.geometry.setVerticesData(VertexBuffer.UVKind + \"_\" + index, uvs, false, 2);\r\n }\r\n }\r\n } else {\r\n let index = 0;\r\n\r\n // Positions\r\n while (this.geometry.isVerticesDataPresent(VertexBuffer.PositionKind + index)) {\r\n this.geometry.removeVerticesData(VertexBuffer.PositionKind + index);\r\n\r\n if (this.geometry.isVerticesDataPresent(VertexBuffer.NormalKind + index)) {\r\n this.geometry.removeVerticesData(VertexBuffer.NormalKind + index);\r\n }\r\n if (this.geometry.isVerticesDataPresent(VertexBuffer.TangentKind + index)) {\r\n this.geometry.removeVerticesData(VertexBuffer.TangentKind + index);\r\n }\r\n if (this.geometry.isVerticesDataPresent(VertexBuffer.UVKind + index)) {\r\n this.geometry.removeVerticesData(VertexBuffer.UVKind + \"_\" + index);\r\n }\r\n index++;\r\n }\r\n }\r\n }\r\n\r\n // Statics\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public static _GroundMeshParser = (parsedMesh: any, scene: Scene): Mesh => {\r\n throw _WarnImport(\"GroundMesh\");\r\n };\r\n\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public static _GoldbergMeshParser = (parsedMesh: any, scene: Scene): GoldbergMesh => {\r\n throw _WarnImport(\"GoldbergMesh\");\r\n };\r\n\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public static _LinesMeshParser = (parsedMesh: any, scene: Scene): Mesh => {\r\n throw _WarnImport(\"LinesMesh\");\r\n };\r\n\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public static _GreasedLineMeshParser = (parsedMesh: any, scene: Scene): Mesh => {\r\n throw _WarnImport(\"GreasedLineMesh\");\r\n };\r\n\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public static _TrailMeshParser = (parsedMesh: any, scene: Scene): Mesh => {\r\n throw _WarnImport(\"TrailMesh\");\r\n };\r\n\r\n /**\r\n * Returns a new Mesh object parsed from the source provided.\r\n * @param parsedMesh is the source\r\n * @param scene defines the hosting scene\r\n * @param rootUrl is the root URL to prefix the `delayLoadingFile` property with\r\n * @returns a new Mesh\r\n */\r\n public static Parse(parsedMesh: any, scene: Scene, rootUrl: string): Mesh {\r\n let mesh: Mesh;\r\n\r\n if (parsedMesh.type && parsedMesh.type === \"LinesMesh\") {\r\n mesh = Mesh._LinesMeshParser(parsedMesh, scene);\r\n } else if (parsedMesh.type && parsedMesh.type === \"GroundMesh\") {\r\n mesh = Mesh._GroundMeshParser(parsedMesh, scene);\r\n } else if (parsedMesh.type && parsedMesh.type === \"GoldbergMesh\") {\r\n mesh = Mesh._GoldbergMeshParser(parsedMesh, scene);\r\n } else if (parsedMesh.type && parsedMesh.type === \"GreasedLineMesh\") {\r\n mesh = Mesh._GreasedLineMeshParser(parsedMesh, scene);\r\n } else if (parsedMesh.type && parsedMesh.type === \"TrailMesh\") {\r\n mesh = Mesh._TrailMeshParser(parsedMesh, scene);\r\n } else {\r\n mesh = new Mesh(parsedMesh.name, scene);\r\n }\r\n mesh.id = parsedMesh.id;\r\n mesh._waitingParsedUniqueId = parsedMesh.uniqueId;\r\n\r\n if (Tags) {\r\n Tags.AddTagsTo(mesh, parsedMesh.tags);\r\n }\r\n\r\n mesh.position = Vector3.FromArray(parsedMesh.position);\r\n\r\n if (parsedMesh.metadata !== undefined) {\r\n mesh.metadata = parsedMesh.metadata;\r\n }\r\n\r\n if (parsedMesh.rotationQuaternion) {\r\n mesh.rotationQuaternion = Quaternion.FromArray(parsedMesh.rotationQuaternion);\r\n } else if (parsedMesh.rotation) {\r\n mesh.rotation = Vector3.FromArray(parsedMesh.rotation);\r\n }\r\n\r\n mesh.scaling = Vector3.FromArray(parsedMesh.scaling);\r\n\r\n if (parsedMesh.localMatrix) {\r\n mesh.setPreTransformMatrix(Matrix.FromArray(parsedMesh.localMatrix));\r\n } else if (parsedMesh.pivotMatrix) {\r\n mesh.setPivotMatrix(Matrix.FromArray(parsedMesh.pivotMatrix));\r\n }\r\n\r\n mesh.setEnabled(parsedMesh.isEnabled);\r\n mesh.isVisible = parsedMesh.isVisible;\r\n mesh.infiniteDistance = parsedMesh.infiniteDistance;\r\n\r\n mesh.showBoundingBox = parsedMesh.showBoundingBox;\r\n mesh.showSubMeshesBoundingBox = parsedMesh.showSubMeshesBoundingBox;\r\n\r\n if (parsedMesh.applyFog !== undefined) {\r\n mesh.applyFog = parsedMesh.applyFog;\r\n }\r\n\r\n if (parsedMesh.pickable !== undefined) {\r\n mesh.isPickable = parsedMesh.pickable;\r\n }\r\n\r\n if (parsedMesh.alphaIndex !== undefined) {\r\n mesh.alphaIndex = parsedMesh.alphaIndex;\r\n }\r\n\r\n mesh.receiveShadows = parsedMesh.receiveShadows;\r\n\r\n if (parsedMesh.billboardMode !== undefined) {\r\n mesh.billboardMode = parsedMesh.billboardMode;\r\n }\r\n\r\n if (parsedMesh.visibility !== undefined) {\r\n mesh.visibility = parsedMesh.visibility;\r\n }\r\n\r\n mesh.checkCollisions = parsedMesh.checkCollisions;\r\n mesh.overrideMaterialSideOrientation = parsedMesh.overrideMaterialSideOrientation;\r\n\r\n if (parsedMesh.isBlocker !== undefined) {\r\n mesh.isBlocker = parsedMesh.isBlocker;\r\n }\r\n\r\n mesh._shouldGenerateFlatShading = parsedMesh.useFlatShading;\r\n\r\n // freezeWorldMatrix\r\n if (parsedMesh.freezeWorldMatrix) {\r\n mesh._waitingData.freezeWorldMatrix = parsedMesh.freezeWorldMatrix;\r\n }\r\n\r\n // Parent\r\n if (parsedMesh.parentId !== undefined) {\r\n mesh._waitingParentId = parsedMesh.parentId;\r\n }\r\n\r\n if (parsedMesh.parentInstanceIndex !== undefined) {\r\n mesh._waitingParentInstanceIndex = parsedMesh.parentInstanceIndex;\r\n }\r\n\r\n // Actions\r\n if (parsedMesh.actions !== undefined) {\r\n mesh._waitingData.actions = parsedMesh.actions;\r\n }\r\n\r\n // Overlay\r\n if (parsedMesh.overlayAlpha !== undefined) {\r\n mesh.overlayAlpha = parsedMesh.overlayAlpha;\r\n }\r\n\r\n if (parsedMesh.overlayColor !== undefined) {\r\n mesh.overlayColor = Color3.FromArray(parsedMesh.overlayColor);\r\n }\r\n\r\n if (parsedMesh.renderOverlay !== undefined) {\r\n mesh.renderOverlay = parsedMesh.renderOverlay;\r\n }\r\n\r\n // Geometry\r\n mesh.isUnIndexed = !!parsedMesh.isUnIndexed;\r\n mesh.hasVertexAlpha = parsedMesh.hasVertexAlpha;\r\n\r\n if (parsedMesh.delayLoadingFile) {\r\n mesh.delayLoadState = Constants.DELAYLOADSTATE_NOTLOADED;\r\n mesh.delayLoadingFile = rootUrl + parsedMesh.delayLoadingFile;\r\n mesh.buildBoundingInfo(Vector3.FromArray(parsedMesh.boundingBoxMinimum), Vector3.FromArray(parsedMesh.boundingBoxMaximum));\r\n\r\n if (parsedMesh._binaryInfo) {\r\n mesh._binaryInfo = parsedMesh._binaryInfo;\r\n }\r\n\r\n mesh._delayInfo = [];\r\n if (parsedMesh.hasUVs) {\r\n mesh._delayInfo.push(VertexBuffer.UVKind);\r\n }\r\n\r\n if (parsedMesh.hasUVs2) {\r\n mesh._delayInfo.push(VertexBuffer.UV2Kind);\r\n }\r\n\r\n if (parsedMesh.hasUVs3) {\r\n mesh._delayInfo.push(VertexBuffer.UV3Kind);\r\n }\r\n\r\n if (parsedMesh.hasUVs4) {\r\n mesh._delayInfo.push(VertexBuffer.UV4Kind);\r\n }\r\n\r\n if (parsedMesh.hasUVs5) {\r\n mesh._delayInfo.push(VertexBuffer.UV5Kind);\r\n }\r\n\r\n if (parsedMesh.hasUVs6) {\r\n mesh._delayInfo.push(VertexBuffer.UV6Kind);\r\n }\r\n\r\n if (parsedMesh.hasColors) {\r\n mesh._delayInfo.push(VertexBuffer.ColorKind);\r\n }\r\n\r\n if (parsedMesh.hasMatricesIndices) {\r\n mesh._delayInfo.push(VertexBuffer.MatricesIndicesKind);\r\n }\r\n\r\n if (parsedMesh.hasMatricesWeights) {\r\n mesh._delayInfo.push(VertexBuffer.MatricesWeightsKind);\r\n }\r\n\r\n mesh._delayLoadingFunction = Geometry._ImportGeometry;\r\n\r\n if (SceneLoaderFlags.ForceFullSceneLoadingForIncremental) {\r\n mesh._checkDelayState();\r\n }\r\n } else {\r\n Geometry._ImportGeometry(parsedMesh, mesh);\r\n }\r\n\r\n // Material\r\n if (parsedMesh.materialUniqueId) {\r\n mesh._waitingMaterialId = parsedMesh.materialUniqueId;\r\n } else if (parsedMesh.materialId) {\r\n mesh._waitingMaterialId = parsedMesh.materialId;\r\n }\r\n\r\n // Morph targets\r\n if (parsedMesh.morphTargetManagerId > -1) {\r\n mesh.morphTargetManager = scene.getMorphTargetManagerById(parsedMesh.morphTargetManagerId);\r\n }\r\n\r\n // Skeleton\r\n if (parsedMesh.skeletonId !== undefined && parsedMesh.skeletonId !== null) {\r\n mesh.skeleton = scene.getLastSkeletonById(parsedMesh.skeletonId);\r\n if (parsedMesh.numBoneInfluencers) {\r\n mesh.numBoneInfluencers = parsedMesh.numBoneInfluencers;\r\n }\r\n }\r\n\r\n // Animations\r\n if (parsedMesh.animations) {\r\n for (let animationIndex = 0; animationIndex < parsedMesh.animations.length; animationIndex++) {\r\n const parsedAnimation = parsedMesh.animations[animationIndex];\r\n const internalClass = GetClass(\"BABYLON.Animation\");\r\n if (internalClass) {\r\n mesh.animations.push(internalClass.Parse(parsedAnimation));\r\n }\r\n }\r\n Node.ParseAnimationRanges(mesh, parsedMesh, scene);\r\n }\r\n\r\n if (parsedMesh.autoAnimate) {\r\n scene.beginAnimation(mesh, parsedMesh.autoAnimateFrom, parsedMesh.autoAnimateTo, parsedMesh.autoAnimateLoop, parsedMesh.autoAnimateSpeed || 1.0);\r\n }\r\n\r\n // Layer Mask\r\n if (parsedMesh.layerMask && !isNaN(parsedMesh.layerMask)) {\r\n mesh.layerMask = Math.abs(parseInt(parsedMesh.layerMask));\r\n } else {\r\n mesh.layerMask = 0x0fffffff;\r\n }\r\n\r\n // Physics\r\n if (parsedMesh.physicsImpostor) {\r\n Mesh._PhysicsImpostorParser(scene, mesh, parsedMesh);\r\n }\r\n\r\n // Levels\r\n if (parsedMesh.lodMeshIds) {\r\n mesh._waitingData.lods = {\r\n ids: parsedMesh.lodMeshIds,\r\n distances: parsedMesh.lodDistances ? parsedMesh.lodDistances : null,\r\n coverages: parsedMesh.lodCoverages ? parsedMesh.lodCoverages : null,\r\n };\r\n }\r\n\r\n // Instances\r\n if (parsedMesh.instances) {\r\n for (let index = 0; index < parsedMesh.instances.length; index++) {\r\n const parsedInstance = parsedMesh.instances[index];\r\n const instance = mesh.createInstance(parsedInstance.name);\r\n\r\n if (parsedInstance.id) {\r\n instance.id = parsedInstance.id;\r\n }\r\n\r\n if (Tags) {\r\n if (parsedInstance.tags) {\r\n Tags.AddTagsTo(instance, parsedInstance.tags);\r\n } else {\r\n Tags.AddTagsTo(instance, parsedMesh.tags);\r\n }\r\n }\r\n\r\n instance.position = Vector3.FromArray(parsedInstance.position);\r\n\r\n if (parsedInstance.metadata !== undefined) {\r\n instance.metadata = parsedInstance.metadata;\r\n }\r\n\r\n if (parsedInstance.parentId !== undefined) {\r\n instance._waitingParentId = parsedInstance.parentId;\r\n }\r\n\r\n if (parsedInstance.parentInstanceIndex !== undefined) {\r\n instance._waitingParentInstanceIndex = parsedInstance.parentInstanceIndex;\r\n }\r\n\r\n if (parsedInstance.isEnabled !== undefined && parsedInstance.isEnabled !== null) {\r\n instance.setEnabled(parsedInstance.isEnabled);\r\n }\r\n\r\n if (parsedInstance.isVisible !== undefined && parsedInstance.isVisible !== null) {\r\n instance.isVisible = parsedInstance.isVisible;\r\n }\r\n\r\n if (parsedInstance.isPickable !== undefined && parsedInstance.isPickable !== null) {\r\n instance.isPickable = parsedInstance.isPickable;\r\n }\r\n\r\n if (parsedInstance.rotationQuaternion) {\r\n instance.rotationQuaternion = Quaternion.FromArray(parsedInstance.rotationQuaternion);\r\n } else if (parsedInstance.rotation) {\r\n instance.rotation = Vector3.FromArray(parsedInstance.rotation);\r\n }\r\n\r\n instance.scaling = Vector3.FromArray(parsedInstance.scaling);\r\n\r\n if (parsedInstance.checkCollisions != undefined && parsedInstance.checkCollisions != null) {\r\n instance.checkCollisions = parsedInstance.checkCollisions;\r\n }\r\n if (parsedInstance.pickable != undefined && parsedInstance.pickable != null) {\r\n instance.isPickable = parsedInstance.pickable;\r\n }\r\n if (parsedInstance.showBoundingBox != undefined && parsedInstance.showBoundingBox != null) {\r\n instance.showBoundingBox = parsedInstance.showBoundingBox;\r\n }\r\n if (parsedInstance.showSubMeshesBoundingBox != undefined && parsedInstance.showSubMeshesBoundingBox != null) {\r\n instance.showSubMeshesBoundingBox = parsedInstance.showSubMeshesBoundingBox;\r\n }\r\n if (parsedInstance.alphaIndex != undefined && parsedInstance.showSubMeshesBoundingBox != null) {\r\n instance.alphaIndex = parsedInstance.alphaIndex;\r\n }\r\n\r\n // Physics\r\n if (parsedInstance.physicsImpostor) {\r\n Mesh._PhysicsImpostorParser(scene, instance, parsedInstance);\r\n }\r\n\r\n // Actions\r\n if (parsedInstance.actions !== undefined) {\r\n instance._waitingData.actions = parsedInstance.actions;\r\n }\r\n\r\n // Animation\r\n if (parsedInstance.animations) {\r\n for (let animationIndex = 0; animationIndex < parsedInstance.animations.length; animationIndex++) {\r\n const parsedAnimation = parsedInstance.animations[animationIndex];\r\n const internalClass = GetClass(\"BABYLON.Animation\");\r\n if (internalClass) {\r\n instance.animations.push(internalClass.Parse(parsedAnimation));\r\n }\r\n }\r\n Node.ParseAnimationRanges(instance, parsedInstance, scene);\r\n\r\n if (parsedInstance.autoAnimate) {\r\n scene.beginAnimation(\r\n instance,\r\n parsedInstance.autoAnimateFrom,\r\n parsedInstance.autoAnimateTo,\r\n parsedInstance.autoAnimateLoop,\r\n parsedInstance.autoAnimateSpeed || 1.0\r\n );\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Thin instances\r\n if (parsedMesh.thinInstances) {\r\n const thinInstances = parsedMesh.thinInstances;\r\n\r\n mesh.thinInstanceEnablePicking = !!thinInstances.enablePicking;\r\n\r\n if (thinInstances.matrixData) {\r\n mesh.thinInstanceSetBuffer(\"matrix\", new Float32Array(thinInstances.matrixData), 16, false);\r\n\r\n mesh._thinInstanceDataStorage.matrixBufferSize = thinInstances.matrixBufferSize;\r\n mesh._thinInstanceDataStorage.instancesCount = thinInstances.instancesCount;\r\n } else {\r\n mesh._thinInstanceDataStorage.matrixBufferSize = thinInstances.matrixBufferSize;\r\n }\r\n\r\n if (parsedMesh.thinInstances.userThinInstance) {\r\n const userThinInstance = parsedMesh.thinInstances.userThinInstance;\r\n\r\n for (const kind in userThinInstance.data) {\r\n mesh.thinInstanceSetBuffer(kind, new Float32Array(userThinInstance.data[kind]), userThinInstance.strides[kind], false);\r\n mesh._userThinInstanceBuffersStorage.sizes[kind] = userThinInstance.sizes[kind];\r\n }\r\n }\r\n }\r\n\r\n return mesh;\r\n }\r\n\r\n // Skeletons\r\n\r\n /**\r\n * Prepare internal position array for software CPU skinning\r\n * @returns original positions used for CPU skinning. Useful for integrating Morphing with skeletons in same mesh\r\n */\r\n public setPositionsForCPUSkinning(): Nullable {\r\n const internalDataInfo = this._internalMeshDataInfo;\r\n if (!internalDataInfo._sourcePositions) {\r\n const source = this.getVerticesData(VertexBuffer.PositionKind);\r\n if (!source) {\r\n return internalDataInfo._sourcePositions;\r\n }\r\n\r\n internalDataInfo._sourcePositions = new Float32Array(source);\r\n\r\n if (!this.isVertexBufferUpdatable(VertexBuffer.PositionKind)) {\r\n this.setVerticesData(VertexBuffer.PositionKind, source, true);\r\n }\r\n }\r\n return internalDataInfo._sourcePositions;\r\n }\r\n\r\n /**\r\n * Prepare internal normal array for software CPU skinning\r\n * @returns original normals used for CPU skinning. Useful for integrating Morphing with skeletons in same mesh.\r\n */\r\n public setNormalsForCPUSkinning(): Nullable {\r\n const internalDataInfo = this._internalMeshDataInfo;\r\n\r\n if (!internalDataInfo._sourceNormals) {\r\n const source = this.getVerticesData(VertexBuffer.NormalKind);\r\n\r\n if (!source) {\r\n return internalDataInfo._sourceNormals;\r\n }\r\n\r\n internalDataInfo._sourceNormals = new Float32Array(source);\r\n\r\n if (!this.isVertexBufferUpdatable(VertexBuffer.NormalKind)) {\r\n this.setVerticesData(VertexBuffer.NormalKind, source, true);\r\n }\r\n }\r\n return internalDataInfo._sourceNormals;\r\n }\r\n\r\n /**\r\n * Updates the vertex buffer by applying transformation from the bones\r\n * @param skeleton defines the skeleton to apply to current mesh\r\n * @returns the current mesh\r\n */\r\n public applySkeleton(skeleton: Skeleton): Mesh {\r\n if (!this.geometry) {\r\n return this;\r\n }\r\n\r\n if (this.geometry._softwareSkinningFrameId == this.getScene().getFrameId()) {\r\n return this;\r\n }\r\n\r\n this.geometry._softwareSkinningFrameId = this.getScene().getFrameId();\r\n\r\n if (!this.isVerticesDataPresent(VertexBuffer.PositionKind)) {\r\n return this;\r\n }\r\n if (!this.isVerticesDataPresent(VertexBuffer.MatricesIndicesKind)) {\r\n return this;\r\n }\r\n if (!this.isVerticesDataPresent(VertexBuffer.MatricesWeightsKind)) {\r\n return this;\r\n }\r\n\r\n const hasNormals = this.isVerticesDataPresent(VertexBuffer.NormalKind);\r\n\r\n const internalDataInfo = this._internalMeshDataInfo;\r\n\r\n if (!internalDataInfo._sourcePositions) {\r\n const submeshes = this.subMeshes.slice();\r\n this.setPositionsForCPUSkinning();\r\n this.subMeshes = submeshes;\r\n }\r\n\r\n if (hasNormals && !internalDataInfo._sourceNormals) {\r\n this.setNormalsForCPUSkinning();\r\n }\r\n\r\n // positionsData checks for not being Float32Array will only pass at most once\r\n let positionsData = this.getVerticesData(VertexBuffer.PositionKind);\r\n\r\n if (!positionsData) {\r\n return this;\r\n }\r\n\r\n if (!(positionsData instanceof Float32Array)) {\r\n positionsData = new Float32Array(positionsData);\r\n }\r\n\r\n // normalsData checks for not being Float32Array will only pass at most once\r\n let normalsData = this.getVerticesData(VertexBuffer.NormalKind);\r\n\r\n if (hasNormals) {\r\n if (!normalsData) {\r\n return this;\r\n }\r\n\r\n if (!(normalsData instanceof Float32Array)) {\r\n normalsData = new Float32Array(normalsData);\r\n }\r\n }\r\n\r\n const matricesIndicesData = this.getVerticesData(VertexBuffer.MatricesIndicesKind);\r\n const matricesWeightsData = this.getVerticesData(VertexBuffer.MatricesWeightsKind);\r\n\r\n if (!matricesWeightsData || !matricesIndicesData) {\r\n return this;\r\n }\r\n\r\n const needExtras = this.numBoneInfluencers > 4;\r\n const matricesIndicesExtraData = needExtras ? this.getVerticesData(VertexBuffer.MatricesIndicesExtraKind) : null;\r\n const matricesWeightsExtraData = needExtras ? this.getVerticesData(VertexBuffer.MatricesWeightsExtraKind) : null;\r\n\r\n const skeletonMatrices = skeleton.getTransformMatrices(this);\r\n\r\n const tempVector3 = Vector3.Zero();\r\n const finalMatrix = new Matrix();\r\n const tempMatrix = new Matrix();\r\n\r\n let matWeightIdx = 0;\r\n let inf: number;\r\n for (let index = 0; index < positionsData.length; index += 3, matWeightIdx += 4) {\r\n let weight: number;\r\n for (inf = 0; inf < 4; inf++) {\r\n weight = matricesWeightsData[matWeightIdx + inf];\r\n if (weight > 0) {\r\n Matrix.FromFloat32ArrayToRefScaled(skeletonMatrices, Math.floor(matricesIndicesData[matWeightIdx + inf] * 16), weight, tempMatrix);\r\n finalMatrix.addToSelf(tempMatrix);\r\n }\r\n }\r\n if (needExtras) {\r\n for (inf = 0; inf < 4; inf++) {\r\n weight = matricesWeightsExtraData![matWeightIdx + inf];\r\n if (weight > 0) {\r\n Matrix.FromFloat32ArrayToRefScaled(skeletonMatrices, Math.floor(matricesIndicesExtraData![matWeightIdx + inf] * 16), weight, tempMatrix);\r\n finalMatrix.addToSelf(tempMatrix);\r\n }\r\n }\r\n }\r\n\r\n Vector3.TransformCoordinatesFromFloatsToRef(\r\n internalDataInfo._sourcePositions![index],\r\n internalDataInfo._sourcePositions![index + 1],\r\n internalDataInfo._sourcePositions![index + 2],\r\n finalMatrix,\r\n tempVector3\r\n );\r\n tempVector3.toArray(positionsData, index);\r\n\r\n if (hasNormals) {\r\n Vector3.TransformNormalFromFloatsToRef(\r\n internalDataInfo._sourceNormals![index],\r\n internalDataInfo._sourceNormals![index + 1],\r\n internalDataInfo._sourceNormals![index + 2],\r\n finalMatrix,\r\n tempVector3\r\n );\r\n tempVector3.toArray(normalsData!, index);\r\n }\r\n\r\n finalMatrix.reset();\r\n }\r\n\r\n this.updateVerticesData(VertexBuffer.PositionKind, positionsData);\r\n if (hasNormals) {\r\n this.updateVerticesData(VertexBuffer.NormalKind, normalsData!);\r\n }\r\n\r\n return this;\r\n }\r\n\r\n // Tools\r\n\r\n /**\r\n * Returns an object containing a min and max Vector3 which are the minimum and maximum vectors of each mesh bounding box from the passed array, in the world coordinates\r\n * @param meshes defines the list of meshes to scan\r\n * @returns an object `{min:` Vector3`, max:` Vector3`}`\r\n */\r\n public static MinMax(meshes: AbstractMesh[]): { min: Vector3; max: Vector3 } {\r\n let minVector: Nullable = null;\r\n let maxVector: Nullable = null;\r\n\r\n meshes.forEach(function (mesh) {\r\n const boundingInfo = mesh.getBoundingInfo();\r\n\r\n const boundingBox = boundingInfo.boundingBox;\r\n if (!minVector || !maxVector) {\r\n minVector = boundingBox.minimumWorld;\r\n maxVector = boundingBox.maximumWorld;\r\n } else {\r\n minVector.minimizeInPlace(boundingBox.minimumWorld);\r\n maxVector.maximizeInPlace(boundingBox.maximumWorld);\r\n }\r\n });\r\n\r\n if (!minVector || !maxVector) {\r\n return {\r\n min: Vector3.Zero(),\r\n max: Vector3.Zero(),\r\n };\r\n }\r\n\r\n return {\r\n min: minVector,\r\n max: maxVector,\r\n };\r\n }\r\n\r\n /**\r\n * Returns the center of the `{min:` Vector3`, max:` Vector3`}` or the center of MinMax vector3 computed from a mesh array\r\n * @param meshesOrMinMaxVector could be an array of meshes or a `{min:` Vector3`, max:` Vector3`}` object\r\n * @returns a vector3\r\n */\r\n public static Center(meshesOrMinMaxVector: { min: Vector3; max: Vector3 } | AbstractMesh[]): Vector3 {\r\n const minMaxVector = meshesOrMinMaxVector instanceof Array ? Mesh.MinMax(meshesOrMinMaxVector) : meshesOrMinMaxVector;\r\n return Vector3.Center(minMaxVector.min, minMaxVector.max);\r\n }\r\n\r\n /**\r\n * Merge the array of meshes into a single mesh for performance reasons.\r\n * @param meshes array of meshes with the vertices to merge. Entries cannot be empty meshes.\r\n * @param disposeSource when true (default), dispose of the vertices from the source meshes.\r\n * @param allow32BitsIndices when the sum of the vertices > 64k, this must be set to true.\r\n * @param meshSubclass (optional) can be set to a Mesh where the merged vertices will be inserted.\r\n * @param subdivideWithSubMeshes when true (false default), subdivide mesh into subMeshes.\r\n * @param multiMultiMaterials when true (false default), subdivide mesh into subMeshes with multiple materials, ignores subdivideWithSubMeshes.\r\n * @returns a new mesh\r\n */\r\n public static MergeMeshes(\r\n meshes: Array,\r\n disposeSource = true,\r\n allow32BitsIndices?: boolean,\r\n meshSubclass?: Mesh,\r\n subdivideWithSubMeshes?: boolean,\r\n multiMultiMaterials?: boolean\r\n ) {\r\n return runCoroutineSync(Mesh._MergeMeshesCoroutine(meshes, disposeSource, allow32BitsIndices, meshSubclass, subdivideWithSubMeshes, multiMultiMaterials, false));\r\n }\r\n\r\n /**\r\n * Merge the array of meshes into a single mesh for performance reasons.\r\n * @param meshes array of meshes with the vertices to merge. Entries cannot be empty meshes.\r\n * @param disposeSource when true (default), dispose of the vertices from the source meshes.\r\n * @param allow32BitsIndices when the sum of the vertices > 64k, this must be set to true.\r\n * @param meshSubclass (optional) can be set to a Mesh where the merged vertices will be inserted.\r\n * @param subdivideWithSubMeshes when true (false default), subdivide mesh into subMeshes.\r\n * @param multiMultiMaterials when true (false default), subdivide mesh into subMeshes with multiple materials, ignores subdivideWithSubMeshes.\r\n * @returns a new mesh\r\n */\r\n public static MergeMeshesAsync(\r\n meshes: Array,\r\n disposeSource = true,\r\n allow32BitsIndices?: boolean,\r\n meshSubclass?: Mesh,\r\n subdivideWithSubMeshes?: boolean,\r\n multiMultiMaterials?: boolean\r\n ) {\r\n return runCoroutineAsync(\r\n Mesh._MergeMeshesCoroutine(meshes, disposeSource, allow32BitsIndices, meshSubclass, subdivideWithSubMeshes, multiMultiMaterials, true),\r\n createYieldingScheduler()\r\n );\r\n }\r\n\r\n private static *_MergeMeshesCoroutine(\r\n meshes: Array,\r\n disposeSource = true,\r\n allow32BitsIndices: boolean | undefined,\r\n meshSubclass: Mesh | undefined,\r\n subdivideWithSubMeshes: boolean | undefined,\r\n multiMultiMaterials: boolean | undefined,\r\n isAsync: boolean\r\n ): Coroutine> {\r\n // Remove any null/undefined entries from the mesh array\r\n meshes = meshes.filter(Boolean);\r\n\r\n if (meshes.length === 0) {\r\n return null;\r\n }\r\n\r\n let index: number;\r\n if (!allow32BitsIndices) {\r\n let totalVertices = 0;\r\n\r\n // Counting vertices\r\n for (index = 0; index < meshes.length; index++) {\r\n totalVertices += meshes[index].getTotalVertices();\r\n\r\n if (totalVertices >= 65536) {\r\n Logger.Warn(\"Cannot merge meshes because resulting mesh will have more than 65536 vertices. Please use allow32BitsIndices = true to use 32 bits indices\");\r\n return null;\r\n }\r\n }\r\n }\r\n if (multiMultiMaterials) {\r\n subdivideWithSubMeshes = false;\r\n }\r\n const materialArray: Array = new Array();\r\n const materialIndexArray: Array = new Array();\r\n // Merge\r\n const indiceArray: Array = new Array();\r\n const currentOverrideMaterialSideOrientation = meshes[0].overrideMaterialSideOrientation;\r\n\r\n for (index = 0; index < meshes.length; index++) {\r\n const mesh = meshes[index];\r\n if (mesh.isAnInstance) {\r\n Logger.Warn(\"Cannot merge instance meshes.\");\r\n return null;\r\n }\r\n\r\n if (currentOverrideMaterialSideOrientation !== mesh.overrideMaterialSideOrientation) {\r\n Logger.Warn(\"Cannot merge meshes with different overrideMaterialSideOrientation values.\");\r\n return null;\r\n }\r\n\r\n if (subdivideWithSubMeshes) {\r\n indiceArray.push(mesh.getTotalIndices());\r\n }\r\n\r\n if (multiMultiMaterials) {\r\n if (mesh.material) {\r\n const material = mesh.material;\r\n if (material instanceof MultiMaterial) {\r\n for (let matIndex = 0; matIndex < material.subMaterials.length; matIndex++) {\r\n if (materialArray.indexOf(material.subMaterials[matIndex]) < 0) {\r\n materialArray.push(material.subMaterials[matIndex]);\r\n }\r\n }\r\n for (let subIndex = 0; subIndex < mesh.subMeshes.length; subIndex++) {\r\n materialIndexArray.push(materialArray.indexOf(material.subMaterials[mesh.subMeshes[subIndex].materialIndex]));\r\n indiceArray.push(mesh.subMeshes[subIndex].indexCount);\r\n }\r\n } else {\r\n if (materialArray.indexOf(material) < 0) {\r\n materialArray.push(material);\r\n }\r\n for (let subIndex = 0; subIndex < mesh.subMeshes.length; subIndex++) {\r\n materialIndexArray.push(materialArray.indexOf(material));\r\n indiceArray.push(mesh.subMeshes[subIndex].indexCount);\r\n }\r\n }\r\n } else {\r\n for (let subIndex = 0; subIndex < mesh.subMeshes.length; subIndex++) {\r\n materialIndexArray.push(0);\r\n indiceArray.push(mesh.subMeshes[subIndex].indexCount);\r\n }\r\n }\r\n }\r\n }\r\n\r\n const source = meshes[0];\r\n\r\n const getVertexDataFromMesh = (mesh: Mesh) => {\r\n const wm = mesh.computeWorldMatrix(true);\r\n const vertexData = VertexData.ExtractFromMesh(mesh, false, false);\r\n return { vertexData, transform: wm };\r\n };\r\n\r\n const { vertexData: sourceVertexData, transform: sourceTransform } = getVertexDataFromMesh(source);\r\n if (isAsync) {\r\n yield;\r\n }\r\n\r\n const meshVertexDatas = new Array<{ vertexData: VertexData; transform?: Matrix }>(meshes.length - 1);\r\n for (let i = 1; i < meshes.length; i++) {\r\n meshVertexDatas[i - 1] = getVertexDataFromMesh(meshes[i]);\r\n if (isAsync) {\r\n yield;\r\n }\r\n }\r\n\r\n const mergeCoroutine = sourceVertexData._mergeCoroutine(sourceTransform, meshVertexDatas, allow32BitsIndices, isAsync, !disposeSource);\r\n let mergeCoroutineStep = mergeCoroutine.next();\r\n while (!mergeCoroutineStep.done) {\r\n if (isAsync) {\r\n yield;\r\n }\r\n mergeCoroutineStep = mergeCoroutine.next();\r\n }\r\n const vertexData = mergeCoroutineStep.value;\r\n\r\n if (!meshSubclass) {\r\n meshSubclass = new Mesh(source.name + \"_merged\", source.getScene());\r\n }\r\n\r\n const applyToCoroutine = vertexData._applyToCoroutine(meshSubclass, undefined, isAsync);\r\n let applyToCoroutineStep = applyToCoroutine.next();\r\n while (!applyToCoroutineStep.done) {\r\n if (isAsync) {\r\n yield;\r\n }\r\n applyToCoroutineStep = applyToCoroutine.next();\r\n }\r\n\r\n // Setting properties\r\n meshSubclass.checkCollisions = source.checkCollisions;\r\n meshSubclass.overrideMaterialSideOrientation = source.overrideMaterialSideOrientation;\r\n\r\n // Cleaning\r\n if (disposeSource) {\r\n for (index = 0; index < meshes.length; index++) {\r\n meshes[index].dispose();\r\n }\r\n }\r\n\r\n // Subdivide\r\n if (subdivideWithSubMeshes || multiMultiMaterials) {\r\n //-- removal of global submesh\r\n meshSubclass.releaseSubMeshes();\r\n index = 0;\r\n let offset = 0;\r\n\r\n //-- apply subdivision according to index table\r\n while (index < indiceArray.length) {\r\n SubMesh.CreateFromIndices(0, offset, indiceArray[index], meshSubclass, undefined, false);\r\n offset += indiceArray[index];\r\n index++;\r\n }\r\n\r\n for (const subMesh of meshSubclass.subMeshes) {\r\n subMesh.refreshBoundingInfo();\r\n }\r\n\r\n meshSubclass.computeWorldMatrix(true);\r\n }\r\n\r\n if (multiMultiMaterials) {\r\n const newMultiMaterial = new MultiMaterial(source.name + \"_merged\", source.getScene());\r\n newMultiMaterial.subMaterials = materialArray;\r\n for (let subIndex = 0; subIndex < meshSubclass.subMeshes.length; subIndex++) {\r\n meshSubclass.subMeshes[subIndex].materialIndex = materialIndexArray[subIndex];\r\n }\r\n meshSubclass.material = newMultiMaterial;\r\n } else {\r\n meshSubclass.material = source.material;\r\n }\r\n\r\n return meshSubclass;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public addInstance(instance: InstancedMesh) {\r\n instance._indexInSourceMeshInstanceArray = this.instances.length;\r\n this.instances.push(instance);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public removeInstance(instance: InstancedMesh) {\r\n // Remove from mesh\r\n const index = instance._indexInSourceMeshInstanceArray;\r\n if (index != -1) {\r\n if (index !== this.instances.length - 1) {\r\n const last = this.instances[this.instances.length - 1];\r\n this.instances[index] = last;\r\n last._indexInSourceMeshInstanceArray = index;\r\n }\r\n\r\n instance._indexInSourceMeshInstanceArray = -1;\r\n this.instances.pop();\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _shouldConvertRHS() {\r\n return this.overrideMaterialSideOrientation === Material.CounterClockWiseSideOrientation;\r\n }\r\n\r\n /** @internal */\r\n public _getRenderingFillMode(fillMode: number): number {\r\n const scene = this.getScene();\r\n\r\n if (scene.forcePointsCloud) return Material.PointFillMode;\r\n\r\n if (scene.forceWireframe) return Material.WireFrameFillMode;\r\n\r\n return this.overrideRenderingFillMode ?? fillMode;\r\n }\r\n\r\n // deprecated methods\r\n /**\r\n * Sets the mesh material by the material or multiMaterial `id` property\r\n * @param id is a string identifying the material or the multiMaterial\r\n * @returns the current mesh\r\n * @deprecated Please use MeshBuilder instead Please use setMaterialById instead\r\n */\r\n public setMaterialByID(id: string): Mesh {\r\n return this.setMaterialById(id);\r\n }\r\n\r\n /**\r\n * Creates a ribbon mesh.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param\r\n * @param name defines the name of the mesh to create\r\n * @param pathArray is a required array of paths, what are each an array of successive Vector3. The pathArray parameter depicts the ribbon geometry.\r\n * @param closeArray creates a seam between the first and the last paths of the path array (default is false)\r\n * @param closePath creates a seam between the first and the last points of each path of the path array\r\n * @param offset is taken in account only if the `pathArray` is containing a single path\r\n * @param scene defines the hosting scene\r\n * @param updatable defines if the mesh must be flagged as updatable\r\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\r\n * @param instance defines an instance of an existing Ribbon object to be updated with the passed `pathArray` parameter (https://doc.babylonjs.com/how_to/How_to_dynamically_morph_a_mesh#ribbon)\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreateRibbon(\r\n name: string,\r\n pathArray: Vector3[][],\r\n closeArray: boolean,\r\n closePath: boolean,\r\n offset: number,\r\n scene?: Scene,\r\n updatable?: boolean,\r\n sideOrientation?: number,\r\n instance?: Mesh\r\n ): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates a plane polygonal mesh. By default, this is a disc.\r\n * @param name defines the name of the mesh to create\r\n * @param radius sets the radius size (float) of the polygon (default 0.5)\r\n * @param tessellation sets the number of polygon sides (positive integer, default 64). So a tessellation valued to 3 will build a triangle, to 4 a square, etc\r\n * @param scene defines the hosting scene\r\n * @param updatable defines if the mesh must be flagged as updatable\r\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreateDisc(name: string, radius: number, tessellation: number, scene: Nullable, updatable?: boolean, sideOrientation?: number): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates a box mesh.\r\n * @param name defines the name of the mesh to create\r\n * @param size sets the size (float) of each box side (default 1)\r\n * @param scene defines the hosting scene\r\n * @param updatable defines if the mesh must be flagged as updatable\r\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreateBox(name: string, size: number, scene: Nullable, updatable?: boolean, sideOrientation?: number): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates a sphere mesh.\r\n * @param name defines the name of the mesh to create\r\n * @param segments sets the sphere number of horizontal stripes (positive integer, default 32)\r\n * @param diameter sets the diameter size (float) of the sphere (default 1)\r\n * @param scene defines the hosting scene\r\n * @param updatable defines if the mesh must be flagged as updatable\r\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreateSphere(name: string, segments: number, diameter: number, scene?: Scene, updatable?: boolean, sideOrientation?: number): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates a hemisphere mesh.\r\n * @param name defines the name of the mesh to create\r\n * @param segments sets the sphere number of horizontal stripes (positive integer, default 32)\r\n * @param diameter sets the diameter size (float) of the sphere (default 1)\r\n * @param scene defines the hosting scene\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreateHemisphere(name: string, segments: number, diameter: number, scene?: Scene): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates a cylinder or a cone mesh.\r\n * @param name defines the name of the mesh to create\r\n * @param height sets the height size (float) of the cylinder/cone (float, default 2)\r\n * @param diameterTop set the top cap diameter (floats, default 1)\r\n * @param diameterBottom set the bottom cap diameter (floats, default 1). This value can't be zero\r\n * @param tessellation sets the number of cylinder sides (positive integer, default 24). Set it to 3 to get a prism for instance\r\n * @param subdivisions sets the number of rings along the cylinder height (positive integer, default 1)\r\n * @param scene defines the hosting scene\r\n * @param updatable defines if the mesh must be flagged as updatable\r\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreateCylinder(\r\n name: string,\r\n height: number,\r\n diameterTop: number,\r\n diameterBottom: number,\r\n tessellation: number,\r\n subdivisions: any,\r\n scene?: Scene,\r\n updatable?: any,\r\n sideOrientation?: number\r\n ): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n // Torus (Code from SharpDX.org)\r\n /**\r\n * Creates a torus mesh.\r\n * @param name defines the name of the mesh to create\r\n * @param diameter sets the diameter size (float) of the torus (default 1)\r\n * @param thickness sets the diameter size of the tube of the torus (float, default 0.5)\r\n * @param tessellation sets the number of torus sides (positive integer, default 16)\r\n * @param scene defines the hosting scene\r\n * @param updatable defines if the mesh must be flagged as updatable\r\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreateTorus(name: string, diameter: number, thickness: number, tessellation: number, scene?: Scene, updatable?: boolean, sideOrientation?: number): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates a torus knot mesh.\r\n * @param name defines the name of the mesh to create\r\n * @param radius sets the global radius size (float) of the torus knot (default 2)\r\n * @param tube sets the diameter size of the tube of the torus (float, default 0.5)\r\n * @param radialSegments sets the number of sides on each tube segments (positive integer, default 32)\r\n * @param tubularSegments sets the number of tubes to decompose the knot into (positive integer, default 32)\r\n * @param p the number of windings on X axis (positive integers, default 2)\r\n * @param q the number of windings on Y axis (positive integers, default 3)\r\n * @param scene defines the hosting scene\r\n * @param updatable defines if the mesh must be flagged as updatable\r\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreateTorusKnot(\r\n name: string,\r\n radius: number,\r\n tube: number,\r\n radialSegments: number,\r\n tubularSegments: number,\r\n p: number,\r\n q: number,\r\n scene?: Scene,\r\n updatable?: boolean,\r\n sideOrientation?: number\r\n ): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates a line mesh..\r\n * @param name defines the name of the mesh to create\r\n * @param points is an array successive Vector3\r\n * @param scene defines the hosting scene\r\n * @param updatable defines if the mesh must be flagged as updatable\r\n * @param instance is an instance of an existing LineMesh object to be updated with the passed `points` parameter (https://doc.babylonjs.com/how_to/How_to_dynamically_morph_a_mesh#lines-and-dashedlines).\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreateLines(name: string, points: Vector3[], scene: Nullable, updatable: boolean, instance?: Nullable): LinesMesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates a dashed line mesh.\r\n * @param name defines the name of the mesh to create\r\n * @param points is an array successive Vector3\r\n * @param dashSize is the size of the dashes relatively the dash number (positive float, default 3)\r\n * @param gapSize is the size of the gap between two successive dashes relatively the dash number (positive float, default 1)\r\n * @param dashNb is the intended total number of dashes (positive integer, default 200)\r\n * @param scene defines the hosting scene\r\n * @param updatable defines if the mesh must be flagged as updatable\r\n * @param instance is an instance of an existing LineMesh object to be updated with the passed `points` parameter (https://doc.babylonjs.com/how_to/How_to_dynamically_morph_a_mesh#lines-and-dashedlines)\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreateDashedLines(\r\n name: string,\r\n points: Vector3[],\r\n dashSize: number,\r\n gapSize: number,\r\n dashNb: number,\r\n scene: Nullable,\r\n updatable?: boolean,\r\n instance?: LinesMesh\r\n ): LinesMesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates a polygon mesh.Please consider using the same method from the MeshBuilder class instead\r\n * The polygon's shape will depend on the input parameters and is constructed parallel to a ground mesh.\r\n * The parameter `shape` is a required array of successive Vector3 representing the corners of the polygon in th XoZ plane, that is y = 0 for all vectors.\r\n * You can set the mesh side orientation with the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\r\n * Remember you can only change the shape positions, not their number when updating a polygon.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#non-regular-polygon\r\n * @param name defines the name of the mesh to create\r\n * @param shape is a required array of successive Vector3 representing the corners of the polygon in th XoZ plane, that is y = 0 for all vectors\r\n * @param scene defines the hosting scene\r\n * @param holes is a required array of arrays of successive Vector3 used to defines holes in the polygon\r\n * @param updatable defines if the mesh must be flagged as updatable\r\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\r\n * @param earcutInjection can be used to inject your own earcut reference\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreatePolygon(name: string, shape: Vector3[], scene: Scene, holes?: Vector3[][], updatable?: boolean, sideOrientation?: number, earcutInjection?: any): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates an extruded polygon mesh, with depth in the Y direction..\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#extruded-non-regular-polygon\r\n * @param name defines the name of the mesh to create\r\n * @param shape is a required array of successive Vector3 representing the corners of the polygon in th XoZ plane, that is y = 0 for all vectors\r\n * @param depth defines the height of extrusion\r\n * @param scene defines the hosting scene\r\n * @param holes is a required array of arrays of successive Vector3 used to defines holes in the polygon\r\n * @param updatable defines if the mesh must be flagged as updatable\r\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\r\n * @param earcutInjection can be used to inject your own earcut reference\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static ExtrudePolygon(\r\n name: string,\r\n shape: Vector3[],\r\n depth: number,\r\n scene: Scene,\r\n holes?: Vector3[][],\r\n updatable?: boolean,\r\n sideOrientation?: number,\r\n earcutInjection?: any\r\n ): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates an extruded shape mesh.\r\n * The extrusion is a parametric shape. It has no predefined shape. Its final shape will depend on the input parameters.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#extruded-shapes\r\n * @param name defines the name of the mesh to create\r\n * @param shape is a required array of successive Vector3. This array depicts the shape to be extruded in its local space : the shape must be designed in the xOy plane and will be extruded along the Z axis\r\n * @param path is a required array of successive Vector3. This is the axis curve the shape is extruded along\r\n * @param scale is the value to scale the shape\r\n * @param rotation is the angle value to rotate the shape each step (each path point), from the former step (so rotation added each step) along the curve\r\n * @param cap sets the way the extruded shape is capped. Possible values : Mesh.NO_CAP (default), Mesh.CAP_START, Mesh.CAP_END, Mesh.CAP_ALL\r\n * @param scene defines the hosting scene\r\n * @param updatable defines if the mesh must be flagged as updatable\r\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\r\n * @param instance is an instance of an existing ExtrudedShape object to be updated with the passed `shape`, `path`, `scale` or `rotation` parameters (https://doc.babylonjs.com/how_to/How_to_dynamically_morph_a_mesh#extruded-shape)\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static ExtrudeShape(\r\n name: string,\r\n shape: Vector3[],\r\n path: Vector3[],\r\n scale: number,\r\n rotation: number,\r\n cap: number,\r\n scene: Nullable,\r\n updatable?: boolean,\r\n sideOrientation?: number,\r\n instance?: Mesh\r\n ): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates an custom extruded shape mesh.\r\n * The custom extrusion is a parametric shape.\r\n * It has no predefined shape. Its final shape will depend on the input parameters.\r\n *\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#extruded-shapes\r\n * @param name defines the name of the mesh to create\r\n * @param shape is a required array of successive Vector3. This array depicts the shape to be extruded in its local space : the shape must be designed in the xOy plane and will be extruded along the Z axis\r\n * @param path is a required array of successive Vector3. This is the axis curve the shape is extruded along\r\n * @param scaleFunction is a custom Javascript function called on each path point\r\n * @param rotationFunction is a custom Javascript function called on each path point\r\n * @param ribbonCloseArray forces the extrusion underlying ribbon to close all the paths in its `pathArray`\r\n * @param ribbonClosePath forces the extrusion underlying ribbon to close its `pathArray`\r\n * @param cap sets the way the extruded shape is capped. Possible values : Mesh.NO_CAP (default), Mesh.CAP_START, Mesh.CAP_END, Mesh.CAP_ALL\r\n * @param scene defines the hosting scene\r\n * @param updatable defines if the mesh must be flagged as updatable\r\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\r\n * @param instance is an instance of an existing ExtrudedShape object to be updated with the passed `shape`, `path`, `scale` or `rotation` parameters (https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#extruded-shape)\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static ExtrudeShapeCustom(\r\n name: string,\r\n shape: Vector3[],\r\n path: Vector3[],\r\n scaleFunction: Nullable<{ (i: number, distance: number): number }>,\r\n rotationFunction: Nullable<{ (i: number, distance: number): number }>,\r\n ribbonCloseArray: boolean,\r\n ribbonClosePath: boolean,\r\n cap: number,\r\n scene: Scene,\r\n updatable?: boolean,\r\n sideOrientation?: number,\r\n instance?: Mesh\r\n ): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates lathe mesh.\r\n * The lathe is a shape with a symmetry axis : a 2D model shape is rotated around this axis to design the lathe.\r\n * @param name defines the name of the mesh to create\r\n * @param shape is a required array of successive Vector3. This array depicts the shape to be rotated in its local space : the shape must be designed in the xOy plane and will be rotated around the Y axis. It's usually a 2D shape, so the Vector3 z coordinates are often set to zero\r\n * @param radius is the radius value of the lathe\r\n * @param tessellation is the side number of the lathe.\r\n * @param scene defines the hosting scene\r\n * @param updatable defines if the mesh must be flagged as updatable\r\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreateLathe(name: string, shape: Vector3[], radius: number, tessellation: number, scene: Scene, updatable?: boolean, sideOrientation?: number): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates a plane mesh.\r\n * @param name defines the name of the mesh to create\r\n * @param size sets the size (float) of both sides of the plane at once (default 1)\r\n * @param scene defines the hosting scene\r\n * @param updatable defines if the mesh must be flagged as updatable\r\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreatePlane(name: string, size: number, scene: Scene, updatable?: boolean, sideOrientation?: number): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates a ground mesh.\r\n * @param name defines the name of the mesh to create\r\n * @param width set the width of the ground\r\n * @param height set the height of the ground\r\n * @param subdivisions sets the number of subdivisions per side\r\n * @param scene defines the hosting scene\r\n * @param updatable defines if the mesh must be flagged as updatable\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreateGround(name: string, width: number, height: number, subdivisions: number, scene?: Scene, updatable?: boolean): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates a tiled ground mesh.\r\n * @param name defines the name of the mesh to create\r\n * @param xmin set the ground minimum X coordinate\r\n * @param zmin set the ground minimum Y coordinate\r\n * @param xmax set the ground maximum X coordinate\r\n * @param zmax set the ground maximum Z coordinate\r\n * @param subdivisions is an object `{w: positive integer, h: positive integer}` (default `{w: 6, h: 6}`). `w` and `h` are the numbers of subdivisions on the ground width and height. Each subdivision is called a tile\r\n * @param precision is an object `{w: positive integer, h: positive integer}` (default `{w: 2, h: 2}`). `w` and `h` are the numbers of subdivisions on the ground width and height of each tile\r\n * @param scene defines the hosting scene\r\n * @param updatable defines if the mesh must be flagged as updatable\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreateTiledGround(\r\n name: string,\r\n xmin: number,\r\n zmin: number,\r\n xmax: number,\r\n zmax: number,\r\n subdivisions: { w: number; h: number },\r\n precision: { w: number; h: number },\r\n scene: Scene,\r\n updatable?: boolean\r\n ): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates a ground mesh from a height map.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set/height_map\r\n * @param name defines the name of the mesh to create\r\n * @param url sets the URL of the height map image resource\r\n * @param width set the ground width size\r\n * @param height set the ground height size\r\n * @param subdivisions sets the number of subdivision per side\r\n * @param minHeight is the minimum altitude on the ground\r\n * @param maxHeight is the maximum altitude on the ground\r\n * @param scene defines the hosting scene\r\n * @param updatable defines if the mesh must be flagged as updatable\r\n * @param onReady is a callback function that will be called once the mesh is built (the height map download can last some time)\r\n * @param alphaFilter will filter any data where the alpha channel is below this value, defaults 0 (all data visible)\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreateGroundFromHeightMap(\r\n name: string,\r\n url: string,\r\n width: number,\r\n height: number,\r\n subdivisions: number,\r\n minHeight: number,\r\n maxHeight: number,\r\n scene: Scene,\r\n updatable?: boolean,\r\n onReady?: (mesh: GroundMesh) => void,\r\n alphaFilter?: number\r\n ): GroundMesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates a tube mesh.\r\n * The tube is a parametric shape.\r\n * It has no predefined shape. Its final shape will depend on the input parameters.\r\n *\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param\r\n * @param name defines the name of the mesh to create\r\n * @param path is a required array of successive Vector3. It is the curve used as the axis of the tube\r\n * @param radius sets the tube radius size\r\n * @param tessellation is the number of sides on the tubular surface\r\n * @param radiusFunction is a custom function. If it is not null, it overrides the parameter `radius`. This function is called on each point of the tube path and is passed the index `i` of the i-th point and the distance of this point from the first point of the path\r\n * @param cap sets the way the extruded shape is capped. Possible values : Mesh.NO_CAP (default), Mesh.CAP_START, Mesh.CAP_END, Mesh.CAP_ALL\r\n * @param scene defines the hosting scene\r\n * @param updatable defines if the mesh must be flagged as updatable\r\n * @param sideOrientation defines the mesh side orientation (https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation)\r\n * @param instance is an instance of an existing Tube object to be updated with the passed `pathArray` parameter (https://doc.babylonjs.com/how_to/How_to_dynamically_morph_a_mesh#tube)\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreateTube(\r\n name: string,\r\n path: Vector3[],\r\n radius: number,\r\n tessellation: number,\r\n radiusFunction: { (i: number, distance: number): number },\r\n cap: number,\r\n scene: Scene,\r\n updatable?: boolean,\r\n sideOrientation?: number,\r\n instance?: Mesh\r\n ): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates a polyhedron mesh.\r\n *.\r\n * * The parameter `type` (positive integer, max 14, default 0) sets the polyhedron type to build among the 15 embedded types. Please refer to the type sheet in the tutorial to choose the wanted type\r\n * * The parameter `size` (positive float, default 1) sets the polygon size\r\n * * You can overwrite the `size` on each dimension bu using the parameters `sizeX`, `sizeY` or `sizeZ` (positive floats, default to `size` value)\r\n * * You can build other polyhedron types than the 15 embbeded ones by setting the parameter `custom` (`polyhedronObject`, default null). If you set the parameter `custom`, this overwrittes the parameter `type`\r\n * * A `polyhedronObject` is a formatted javascript object. You'll find a full file with pre-set polyhedra here : https://github.com/BabylonJS/Extensions/tree/master/Polyhedron\r\n * * You can set the color and the UV of each side of the polyhedron with the parameters `faceColors` (Color4, default `(1, 1, 1, 1)`) and faceUV (Vector4, default `(0, 0, 1, 1)`)\r\n * * To understand how to set `faceUV` or `faceColors`, please read this by considering the right number of faces of your polyhedron, instead of only 6 for the box : https://doc.babylonjs.com/features/featuresDeepDive/materials/using/texturePerBoxFace\r\n * * The parameter `flat` (boolean, default true). If set to false, it gives the polyhedron a single global face, so less vertices and shared normals. In this case, `faceColors` and `faceUV` are ignored\r\n * * You can also set the mesh side orientation with the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\r\n * @param name defines the name of the mesh to create\r\n * @param options defines the options used to create the mesh\r\n * @param scene defines the hosting scene\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreatePolyhedron(\r\n name: string,\r\n options: {\r\n type?: number;\r\n size?: number;\r\n sizeX?: number;\r\n sizeY?: number;\r\n sizeZ?: number;\r\n custom?: any;\r\n faceUV?: Vector4[];\r\n faceColors?: Color4[];\r\n updatable?: boolean;\r\n sideOrientation?: number;\r\n },\r\n scene: Scene\r\n ): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates a sphere based upon an icosahedron with 20 triangular faces which can be subdivided\r\n * * The parameter `radius` sets the radius size (float) of the icosphere (default 1)\r\n * * You can set some different icosphere dimensions, for instance to build an ellipsoid, by using the parameters `radiusX`, `radiusY` and `radiusZ` (all by default have the same value than `radius`)\r\n * * The parameter `subdivisions` sets the number of subdivisions (positive integer, default 4). The more subdivisions, the more faces on the icosphere whatever its size\r\n * * The parameter `flat` (boolean, default true) gives each side its own normals. Set it to false to get a smooth continuous light reflection on the surface\r\n * * You can also set the mesh side orientation with the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/polyhedra#icosphere\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param scene defines the hosting scene\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreateIcoSphere(\r\n name: string,\r\n options: { radius?: number; flat?: boolean; subdivisions?: number; sideOrientation?: number; updatable?: boolean },\r\n scene: Scene\r\n ): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Creates a decal mesh.\r\n *.\r\n * A decal is a mesh usually applied as a model onto the surface of another mesh\r\n * @param name defines the name of the mesh\r\n * @param sourceMesh defines the mesh receiving the decal\r\n * @param position sets the position of the decal in world coordinates\r\n * @param normal sets the normal of the mesh where the decal is applied onto in world coordinates\r\n * @param size sets the decal scaling\r\n * @param angle sets the angle to rotate the decal\r\n * @returns a new Mesh\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreateDecal(name: string, sourceMesh: AbstractMesh, position: Vector3, normal: Vector3, size: Vector3, angle: number): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /** Creates a Capsule Mesh\r\n * @param name defines the name of the mesh.\r\n * @param options the constructors options used to shape the mesh.\r\n * @param scene defines the scene the mesh is scoped to.\r\n * @returns the capsule mesh\r\n * @see https://doc.babylonjs.com/how_to/capsule_shape\r\n * @deprecated Please use MeshBuilder instead\r\n */\r\n public static CreateCapsule(name: string, options: ICreateCapsuleOptions, scene: Scene): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n\r\n /**\r\n * Extends a mesh to a Goldberg mesh\r\n * Warning the mesh to convert MUST be an import of a perviously exported Goldberg mesh\r\n * @param mesh the mesh to convert\r\n * @returns the extended mesh\r\n * @deprecated Please use ExtendMeshToGoldberg instead\r\n */\r\n public static ExtendToGoldberg(mesh: Mesh): Mesh {\r\n throw new Error(\"Import MeshBuilder to populate this function\");\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.Mesh\", Mesh);\r\n","import type { Behavior } from \"../../Behaviors/behavior\";\r\nimport type { Camera } from \"../../Cameras/camera\";\r\nimport type { ArcRotateCamera } from \"../../Cameras/arcRotateCamera\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { Observer } from \"../../Misc/observable\";\r\nimport type { PointerInfoPre } from \"../../Events/pointerEvents\";\r\nimport { PointerEventTypes } from \"../../Events/pointerEvents\";\r\nimport { PrecisionDate } from \"../../Misc/precisionDate\";\r\nimport { Epsilon } from \"../../Maths/math.constants\";\r\n\r\n/**\r\n * The autoRotation behavior (AutoRotationBehavior) is designed to create a smooth rotation of an ArcRotateCamera when there is no user interaction.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#autorotation-behavior\r\n */\r\nexport class AutoRotationBehavior implements Behavior {\r\n /**\r\n * Gets the name of the behavior.\r\n */\r\n public get name(): string {\r\n return \"AutoRotation\";\r\n }\r\n\r\n private _zoomStopsAnimation = false;\r\n private _idleRotationSpeed = 0.05;\r\n private _idleRotationWaitTime = 2000;\r\n private _idleRotationSpinupTime = 2000;\r\n\r\n public targetAlpha: Nullable = null;\r\n\r\n /**\r\n * Sets the flag that indicates if user zooming should stop animation.\r\n */\r\n public set zoomStopsAnimation(flag: boolean) {\r\n this._zoomStopsAnimation = flag;\r\n }\r\n\r\n /**\r\n * Gets the flag that indicates if user zooming should stop animation.\r\n */\r\n public get zoomStopsAnimation(): boolean {\r\n return this._zoomStopsAnimation;\r\n }\r\n\r\n /**\r\n * Sets the default speed at which the camera rotates around the model.\r\n */\r\n public set idleRotationSpeed(speed: number) {\r\n this._idleRotationSpeed = speed;\r\n }\r\n\r\n /**\r\n * Gets the default speed at which the camera rotates around the model.\r\n */\r\n public get idleRotationSpeed() {\r\n return this._idleRotationSpeed;\r\n }\r\n\r\n /**\r\n * Sets the time (in milliseconds) to wait after user interaction before the camera starts rotating.\r\n */\r\n public set idleRotationWaitTime(time: number) {\r\n this._idleRotationWaitTime = time;\r\n }\r\n\r\n /**\r\n * Gets the time (milliseconds) to wait after user interaction before the camera starts rotating.\r\n */\r\n public get idleRotationWaitTime() {\r\n return this._idleRotationWaitTime;\r\n }\r\n\r\n /**\r\n * Sets the time (milliseconds) to take to spin up to the full idle rotation speed.\r\n */\r\n public set idleRotationSpinupTime(time: number) {\r\n this._idleRotationSpinupTime = time;\r\n }\r\n\r\n /**\r\n * Gets the time (milliseconds) to take to spin up to the full idle rotation speed.\r\n */\r\n public get idleRotationSpinupTime() {\r\n return this._idleRotationSpinupTime;\r\n }\r\n\r\n /**\r\n * Gets a value indicating if the camera is currently rotating because of this behavior\r\n */\r\n public get rotationInProgress(): boolean {\r\n return Math.abs(this._cameraRotationSpeed) > 0;\r\n }\r\n\r\n // Default behavior functions\r\n private _onPrePointerObservableObserver: Nullable>;\r\n private _onAfterCheckInputsObserver: Nullable>;\r\n private _attachedCamera: Nullable;\r\n private _isPointerDown = false;\r\n private _lastFrameTime: Nullable = null;\r\n private _lastInteractionTime = -Infinity;\r\n private _cameraRotationSpeed: number = 0;\r\n\r\n /**\r\n * Initializes the behavior.\r\n */\r\n public init(): void {\r\n // Do nothing\r\n }\r\n\r\n /**\r\n * Attaches the behavior to its arc rotate camera.\r\n * @param camera Defines the camera to attach the behavior to\r\n */\r\n public attach(camera: ArcRotateCamera): void {\r\n this._attachedCamera = camera;\r\n const scene = this._attachedCamera.getScene();\r\n\r\n this._onPrePointerObservableObserver = scene.onPrePointerObservable.add((pointerInfoPre) => {\r\n if (pointerInfoPre.type === PointerEventTypes.POINTERDOWN) {\r\n this._isPointerDown = true;\r\n return;\r\n }\r\n\r\n if (pointerInfoPre.type === PointerEventTypes.POINTERUP) {\r\n this._isPointerDown = false;\r\n }\r\n });\r\n\r\n this._onAfterCheckInputsObserver = camera.onAfterCheckInputsObservable.add(() => {\r\n if (this._reachTargetAlpha()) {\r\n return;\r\n }\r\n const now = PrecisionDate.Now;\r\n let dt = 0;\r\n if (this._lastFrameTime != null) {\r\n dt = now - this._lastFrameTime;\r\n }\r\n this._lastFrameTime = now;\r\n\r\n // Stop the animation if there is user interaction and the animation should stop for this interaction\r\n this._applyUserInteraction();\r\n\r\n const timeToRotation = now - this._lastInteractionTime - this._idleRotationWaitTime;\r\n const scale = Math.max(Math.min(timeToRotation / this._idleRotationSpinupTime, 1), 0);\r\n this._cameraRotationSpeed = this._idleRotationSpeed * scale;\r\n\r\n // Step camera rotation by rotation speed\r\n if (this._attachedCamera) {\r\n this._attachedCamera.alpha -= this._cameraRotationSpeed * (dt / 1000);\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Detaches the behavior from its current arc rotate camera.\r\n */\r\n public detach(): void {\r\n if (!this._attachedCamera) {\r\n return;\r\n }\r\n const scene = this._attachedCamera.getScene();\r\n\r\n if (this._onPrePointerObservableObserver) {\r\n scene.onPrePointerObservable.remove(this._onPrePointerObservableObserver);\r\n }\r\n\r\n this._attachedCamera.onAfterCheckInputsObservable.remove(this._onAfterCheckInputsObserver);\r\n this._attachedCamera = null;\r\n }\r\n\r\n /**\r\n * Force-reset the last interaction time\r\n * @param customTime an optional time that will be used instead of the current last interaction time. For example `Date.now()`\r\n */\r\n public resetLastInteractionTime(customTime?: number): void {\r\n this._lastInteractionTime = customTime ?? PrecisionDate.Now;\r\n }\r\n\r\n /**\r\n * Returns true if camera alpha reaches the target alpha\r\n * @returns true if camera alpha reaches the target alpha\r\n */\r\n private _reachTargetAlpha(): boolean {\r\n if (this._attachedCamera && this.targetAlpha) {\r\n return Math.abs(this._attachedCamera.alpha - this.targetAlpha) < Epsilon;\r\n }\r\n return false;\r\n }\r\n\r\n /**\r\n * Returns true if user is scrolling.\r\n * @returns true if user is scrolling.\r\n */\r\n private _userIsZooming(): boolean {\r\n if (!this._attachedCamera) {\r\n return false;\r\n }\r\n return this._attachedCamera.inertialRadiusOffset !== 0;\r\n }\r\n\r\n private _lastFrameRadius = 0;\r\n private _shouldAnimationStopForInteraction(): boolean {\r\n if (!this._attachedCamera) {\r\n return false;\r\n }\r\n\r\n let zoomHasHitLimit = false;\r\n if (this._lastFrameRadius === this._attachedCamera.radius && this._attachedCamera.inertialRadiusOffset !== 0) {\r\n zoomHasHitLimit = true;\r\n }\r\n\r\n // Update the record of previous radius - works as an approx. indicator of hitting radius limits\r\n this._lastFrameRadius = this._attachedCamera.radius;\r\n return this._zoomStopsAnimation ? zoomHasHitLimit : this._userIsZooming();\r\n }\r\n\r\n /**\r\n * Applies any current user interaction to the camera. Takes into account maximum alpha rotation.\r\n */\r\n private _applyUserInteraction(): void {\r\n if (this._userIsMoving() && !this._shouldAnimationStopForInteraction()) {\r\n this._lastInteractionTime = PrecisionDate.Now;\r\n }\r\n }\r\n\r\n // Tools\r\n private _userIsMoving(): boolean {\r\n if (!this._attachedCamera) {\r\n return false;\r\n }\r\n\r\n return (\r\n this._attachedCamera.inertialAlphaOffset !== 0 ||\r\n this._attachedCamera.inertialBetaOffset !== 0 ||\r\n this._attachedCamera.inertialRadiusOffset !== 0 ||\r\n this._attachedCamera.inertialPanningX !== 0 ||\r\n this._attachedCamera.inertialPanningY !== 0 ||\r\n this._isPointerDown\r\n );\r\n }\r\n}\r\n","import { BezierCurve } from \"../Maths/math.path\";\r\n\r\n/**\r\n * This represents the main contract an easing function should follow.\r\n * Easing functions are used throughout the animation system.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#easing-functions\r\n */\r\nexport interface IEasingFunction {\r\n /**\r\n * Given an input gradient between 0 and 1, this returns the corresponding value\r\n * of the easing function.\r\n * The link below provides some of the most common examples of easing functions.\r\n * @see https://easings.net/\r\n * @param gradient Defines the value between 0 and 1 we want the easing value for\r\n * @returns the corresponding value on the curve defined by the easing function\r\n */\r\n ease(gradient: number): number;\r\n}\r\n\r\n/**\r\n * Base class used for every default easing function.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#easing-functions\r\n */\r\nexport class EasingFunction implements IEasingFunction {\r\n /**\r\n * Interpolation follows the mathematical formula associated with the easing function.\r\n */\r\n public static readonly EASINGMODE_EASEIN = 0;\r\n\r\n /**\r\n * Interpolation follows 100% interpolation minus the output of the formula associated with the easing function.\r\n */\r\n public static readonly EASINGMODE_EASEOUT = 1;\r\n\r\n /**\r\n * Interpolation uses EaseIn for the first half of the animation and EaseOut for the second half.\r\n */\r\n public static readonly EASINGMODE_EASEINOUT = 2;\r\n\r\n private _easingMode = EasingFunction.EASINGMODE_EASEIN;\r\n\r\n /**\r\n * Sets the easing mode of the current function.\r\n * @param easingMode Defines the willing mode (EASINGMODE_EASEIN, EASINGMODE_EASEOUT or EASINGMODE_EASEINOUT)\r\n */\r\n public setEasingMode(easingMode: number) {\r\n const n = Math.min(Math.max(easingMode, 0), 2);\r\n this._easingMode = n;\r\n }\r\n /**\r\n * Gets the current easing mode.\r\n * @returns the easing mode\r\n */\r\n public getEasingMode(): number {\r\n return this._easingMode;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public easeInCore(gradient: number): number {\r\n throw new Error(\"You must implement this method\");\r\n }\r\n\r\n /**\r\n * Given an input gradient between 0 and 1, this returns the corresponding value\r\n * of the easing function.\r\n * @param gradient Defines the value between 0 and 1 we want the easing value for\r\n * @returns the corresponding value on the curve defined by the easing function\r\n */\r\n public ease(gradient: number): number {\r\n switch (this._easingMode) {\r\n case EasingFunction.EASINGMODE_EASEIN:\r\n return this.easeInCore(gradient);\r\n case EasingFunction.EASINGMODE_EASEOUT:\r\n return 1 - this.easeInCore(1 - gradient);\r\n }\r\n\r\n if (gradient >= 0.5) {\r\n return (1 - this.easeInCore((1 - gradient) * 2)) * 0.5 + 0.5;\r\n }\r\n\r\n return this.easeInCore(gradient * 2) * 0.5;\r\n }\r\n}\r\n\r\n/**\r\n * Easing function with a circle shape (see link below).\r\n * @see https://easings.net/#easeInCirc\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#easing-functions\r\n */\r\nexport class CircleEase extends EasingFunction implements IEasingFunction {\r\n /**\r\n * @internal\r\n */\r\n public easeInCore(gradient: number): number {\r\n gradient = Math.max(0, Math.min(1, gradient));\r\n return 1.0 - Math.sqrt(1.0 - gradient * gradient);\r\n }\r\n}\r\n\r\n/**\r\n * Easing function with a ease back shape (see link below).\r\n * @see https://easings.net/#easeInBack\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#easing-functions\r\n */\r\nexport class BackEase extends EasingFunction implements IEasingFunction {\r\n /**\r\n * Instantiates a back ease easing\r\n * @see https://easings.net/#easeInBack\r\n * @param amplitude Defines the amplitude of the function\r\n */\r\n constructor(\r\n /** Defines the amplitude of the function */\r\n public amplitude: number = 1\r\n ) {\r\n super();\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public easeInCore(gradient: number): number {\r\n const num = Math.max(0, this.amplitude);\r\n return Math.pow(gradient, 3.0) - gradient * num * Math.sin(3.1415926535897931 * gradient);\r\n }\r\n}\r\n\r\n/**\r\n * Easing function with a bouncing shape (see link below).\r\n * @see https://easings.net/#easeInBounce\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#easing-functions\r\n */\r\nexport class BounceEase extends EasingFunction implements IEasingFunction {\r\n /**\r\n * Instantiates a bounce easing\r\n * @see https://easings.net/#easeInBounce\r\n * @param bounces Defines the number of bounces\r\n * @param bounciness Defines the amplitude of the bounce\r\n */\r\n constructor(\r\n /** Defines the number of bounces */\r\n public bounces: number = 3,\r\n /** Defines the amplitude of the bounce */\r\n public bounciness: number = 2\r\n ) {\r\n super();\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public easeInCore(gradient: number): number {\r\n const y = Math.max(0.0, this.bounces);\r\n let bounciness = this.bounciness;\r\n if (bounciness <= 1.0) {\r\n bounciness = 1.001;\r\n }\r\n const num9 = Math.pow(bounciness, y);\r\n const num5 = 1.0 - bounciness;\r\n const num4 = (1.0 - num9) / num5 + num9 * 0.5;\r\n const num15 = gradient * num4;\r\n const num65 = Math.log(-num15 * (1.0 - bounciness) + 1.0) / Math.log(bounciness);\r\n const num3 = Math.floor(num65);\r\n const num13 = num3 + 1.0;\r\n const num8 = (1.0 - Math.pow(bounciness, num3)) / (num5 * num4);\r\n const num12 = (1.0 - Math.pow(bounciness, num13)) / (num5 * num4);\r\n const num7 = (num8 + num12) * 0.5;\r\n const num6 = gradient - num7;\r\n const num2 = num7 - num8;\r\n return (-Math.pow(1.0 / bounciness, y - num3) / (num2 * num2)) * (num6 - num2) * (num6 + num2);\r\n }\r\n}\r\n\r\n/**\r\n * Easing function with a power of 3 shape (see link below).\r\n * @see https://easings.net/#easeInCubic\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#easing-functions\r\n */\r\nexport class CubicEase extends EasingFunction implements IEasingFunction {\r\n /**\r\n * @internal\r\n */\r\n public easeInCore(gradient: number): number {\r\n return gradient * gradient * gradient;\r\n }\r\n}\r\n\r\n/**\r\n * Easing function with an elastic shape (see link below).\r\n * @see https://easings.net/#easeInElastic\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#easing-functions\r\n */\r\nexport class ElasticEase extends EasingFunction implements IEasingFunction {\r\n /**\r\n * Instantiates an elastic easing function\r\n * @see https://easings.net/#easeInElastic\r\n * @param oscillations Defines the number of oscillations\r\n * @param springiness Defines the amplitude of the oscillations\r\n */\r\n constructor(\r\n /** Defines the number of oscillations*/\r\n public oscillations: number = 3,\r\n /** Defines the amplitude of the oscillations*/\r\n public springiness: number = 3\r\n ) {\r\n super();\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public easeInCore(gradient: number): number {\r\n let num2;\r\n const num3 = Math.max(0.0, this.oscillations);\r\n const num = Math.max(0.0, this.springiness);\r\n\r\n if (num == 0) {\r\n num2 = gradient;\r\n } else {\r\n num2 = (Math.exp(num * gradient) - 1.0) / (Math.exp(num) - 1.0);\r\n }\r\n return num2 * Math.sin((6.2831853071795862 * num3 + 1.5707963267948966) * gradient);\r\n }\r\n}\r\n\r\n/**\r\n * Easing function with an exponential shape (see link below).\r\n * @see https://easings.net/#easeInExpo\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#easing-functions\r\n */\r\nexport class ExponentialEase extends EasingFunction implements IEasingFunction {\r\n /**\r\n * Instantiates an exponential easing function\r\n * @see https://easings.net/#easeInExpo\r\n * @param exponent Defines the exponent of the function\r\n */\r\n constructor(\r\n /** Defines the exponent of the function */\r\n public exponent: number = 2\r\n ) {\r\n super();\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public easeInCore(gradient: number): number {\r\n if (this.exponent <= 0) {\r\n return gradient;\r\n }\r\n\r\n return (Math.exp(this.exponent * gradient) - 1.0) / (Math.exp(this.exponent) - 1.0);\r\n }\r\n}\r\n\r\n/**\r\n * Easing function with a power shape (see link below).\r\n * @see https://easings.net/#easeInQuad\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#easing-functions\r\n */\r\nexport class PowerEase extends EasingFunction implements IEasingFunction {\r\n /**\r\n * Instantiates an power base easing function\r\n * @see https://easings.net/#easeInQuad\r\n * @param power Defines the power of the function\r\n */\r\n constructor(\r\n /** Defines the power of the function */\r\n public power: number = 2\r\n ) {\r\n super();\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public easeInCore(gradient: number): number {\r\n const y = Math.max(0.0, this.power);\r\n return Math.pow(gradient, y);\r\n }\r\n}\r\n\r\n/**\r\n * Easing function with a power of 2 shape (see link below).\r\n * @see https://easings.net/#easeInQuad\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#easing-functions\r\n */\r\nexport class QuadraticEase extends EasingFunction implements IEasingFunction {\r\n /**\r\n * @internal\r\n */\r\n public easeInCore(gradient: number): number {\r\n return gradient * gradient;\r\n }\r\n}\r\n\r\n/**\r\n * Easing function with a power of 4 shape (see link below).\r\n * @see https://easings.net/#easeInQuart\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#easing-functions\r\n */\r\nexport class QuarticEase extends EasingFunction implements IEasingFunction {\r\n /**\r\n * @internal\r\n */\r\n public easeInCore(gradient: number): number {\r\n return gradient * gradient * gradient * gradient;\r\n }\r\n}\r\n\r\n/**\r\n * Easing function with a power of 5 shape (see link below).\r\n * @see https://easings.net/#easeInQuint\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#easing-functions\r\n */\r\nexport class QuinticEase extends EasingFunction implements IEasingFunction {\r\n /**\r\n * @internal\r\n */\r\n public easeInCore(gradient: number): number {\r\n return gradient * gradient * gradient * gradient * gradient;\r\n }\r\n}\r\n\r\n/**\r\n * Easing function with a sin shape (see link below).\r\n * @see https://easings.net/#easeInSine\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#easing-functions\r\n */\r\nexport class SineEase extends EasingFunction implements IEasingFunction {\r\n /**\r\n * @internal\r\n */\r\n public easeInCore(gradient: number): number {\r\n return 1.0 - Math.sin(1.5707963267948966 * (1.0 - gradient));\r\n }\r\n}\r\n\r\n/**\r\n * Easing function with a bezier shape (see link below).\r\n * @see http://cubic-bezier.com/#.17,.67,.83,.67\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#easing-functions\r\n */\r\nexport class BezierCurveEase extends EasingFunction implements IEasingFunction {\r\n /**\r\n * Instantiates a bezier function\r\n * @see http://cubic-bezier.com/#.17,.67,.83,.67\r\n * @param x1 Defines the x component of the start tangent in the bezier curve\r\n * @param y1 Defines the y component of the start tangent in the bezier curve\r\n * @param x2 Defines the x component of the end tangent in the bezier curve\r\n * @param y2 Defines the y component of the end tangent in the bezier curve\r\n */\r\n constructor(\r\n /** Defines the x component of the start tangent in the bezier curve */\r\n public x1: number = 0,\r\n /** Defines the y component of the start tangent in the bezier curve */\r\n public y1: number = 0,\r\n /** Defines the x component of the end tangent in the bezier curve */\r\n public x2: number = 1,\r\n /** Defines the y component of the end tangent in the bezier curve */\r\n public y2: number = 1\r\n ) {\r\n super();\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public easeInCore(gradient: number): number {\r\n return BezierCurve.Interpolate(gradient, this.x1, this.y1, this.x2, this.y2);\r\n }\r\n}\r\n","/**\r\n * Represents the range of an animation\r\n */\r\nexport class AnimationRange {\r\n /**\r\n * Initializes the range of an animation\r\n * @param name The name of the animation range\r\n * @param from The starting frame of the animation\r\n * @param to The ending frame of the animation\r\n */\r\n constructor(\r\n /**The name of the animation range**/\r\n public name: string,\r\n /**The starting frame of the animation */\r\n public from: number,\r\n /**The ending frame of the animation*/\r\n public to: number\r\n ) {}\r\n\r\n /**\r\n * Makes a copy of the animation range\r\n * @returns A copy of the animation range\r\n */\r\n public clone(): AnimationRange {\r\n return new AnimationRange(this.name, this.from, this.to);\r\n }\r\n}\r\n","/**\r\n * Interface for the size containing width and height\r\n */\r\nexport interface ISize {\r\n /**\r\n * Width\r\n */\r\n width: number;\r\n /**\r\n * Height\r\n */\r\n height: number;\r\n}\r\n\r\n/**\r\n * Size containing width and height\r\n */\r\nexport class Size implements ISize {\r\n /**\r\n * Width\r\n */\r\n public width: number;\r\n /**\r\n * Height\r\n */\r\n public height: number;\r\n\r\n /**\r\n * Creates a Size object from the given width and height (floats).\r\n * @param width width of the new size\r\n * @param height height of the new size\r\n */\r\n public constructor(width: number, height: number) {\r\n this.width = width;\r\n this.height = height;\r\n }\r\n\r\n /**\r\n * Returns a string with the Size width and height\r\n * @returns a string with the Size width and height\r\n */\r\n public toString(): string {\r\n return `{W: ${this.width}, H: ${this.height}}`;\r\n }\r\n /**\r\n * \"Size\"\r\n * @returns the string \"Size\"\r\n */\r\n public getClassName(): string {\r\n return \"Size\";\r\n }\r\n /**\r\n * Returns the Size hash code.\r\n * @returns a hash code for a unique width and height\r\n */\r\n public getHashCode(): number {\r\n let hash = this.width | 0;\r\n hash = (hash * 397) ^ (this.height | 0);\r\n return hash;\r\n }\r\n /**\r\n * Updates the current size from the given one.\r\n * @param src the given size\r\n */\r\n public copyFrom(src: Size) {\r\n this.width = src.width;\r\n this.height = src.height;\r\n }\r\n /**\r\n * Updates in place the current Size from the given floats.\r\n * @param width width of the new size\r\n * @param height height of the new size\r\n * @returns the updated Size.\r\n */\r\n public copyFromFloats(width: number, height: number): Size {\r\n this.width = width;\r\n this.height = height;\r\n return this;\r\n }\r\n /**\r\n * Updates in place the current Size from the given floats.\r\n * @param width width to set\r\n * @param height height to set\r\n * @returns the updated Size.\r\n */\r\n public set(width: number, height: number): Size {\r\n return this.copyFromFloats(width, height);\r\n }\r\n /**\r\n * Multiplies the width and height by numbers\r\n * @param w factor to multiple the width by\r\n * @param h factor to multiple the height by\r\n * @returns a new Size set with the multiplication result of the current Size and the given floats.\r\n */\r\n public multiplyByFloats(w: number, h: number): Size {\r\n return new Size(this.width * w, this.height * h);\r\n }\r\n /**\r\n * Clones the size\r\n * @returns a new Size copied from the given one.\r\n */\r\n public clone(): Size {\r\n return new Size(this.width, this.height);\r\n }\r\n /**\r\n * True if the current Size and the given one width and height are strictly equal.\r\n * @param other the other size to compare against\r\n * @returns True if the current Size and the given one width and height are strictly equal.\r\n */\r\n public equals(other: Size): boolean {\r\n if (!other) {\r\n return false;\r\n }\r\n return this.width === other.width && this.height === other.height;\r\n }\r\n /**\r\n * The surface of the Size : width * height (float).\r\n */\r\n public get surface(): number {\r\n return this.width * this.height;\r\n }\r\n /**\r\n * Create a new size of zero\r\n * @returns a new Size set to (0.0, 0.0)\r\n */\r\n public static Zero(): Size {\r\n return new Size(0.0, 0.0);\r\n }\r\n /**\r\n * Sums the width and height of two sizes\r\n * @param otherSize size to add to this size\r\n * @returns a new Size set as the addition result of the current Size and the given one.\r\n */\r\n public add(otherSize: Size): Size {\r\n const r = new Size(this.width + otherSize.width, this.height + otherSize.height);\r\n return r;\r\n }\r\n /**\r\n * Subtracts the width and height of two\r\n * @param otherSize size to subtract to this size\r\n * @returns a new Size set as the subtraction result of the given one from the current Size.\r\n */\r\n public subtract(otherSize: Size): Size {\r\n const r = new Size(this.width - otherSize.width, this.height - otherSize.height);\r\n return r;\r\n }\r\n /**\r\n * Creates a new Size set at the linear interpolation \"amount\" between \"start\" and \"end\"\r\n * @param start starting size to lerp between\r\n * @param end end size to lerp between\r\n * @param amount amount to lerp between the start and end values\r\n * @returns a new Size set at the linear interpolation \"amount\" between \"start\" and \"end\"\r\n */\r\n public static Lerp(start: Size, end: Size, amount: number): Size {\r\n const w = start.width + (end.width - start.width) * amount;\r\n const h = start.height + (end.height - start.height) * amount;\r\n\r\n return new Size(w, h);\r\n }\r\n}\r\n","import type { IEasingFunction, EasingFunction } from \"./easing\";\r\nimport { Vector3, Quaternion, Vector2, Matrix, TmpVectors } from \"../Maths/math.vector\";\r\nimport { Color3, Color4 } from \"../Maths/math.color\";\r\nimport { Scalar } from \"../Maths/math.scalar\";\r\n\r\nimport type { Nullable } from \"../types\";\r\nimport type { Scene } from \"../scene\";\r\nimport { SerializationHelper } from \"../Misc/decorators\";\r\nimport { RegisterClass } from \"../Misc/typeStore\";\r\nimport type { IAnimationKey } from \"./animationKey\";\r\nimport { AnimationKeyInterpolation } from \"./animationKey\";\r\nimport { AnimationRange } from \"./animationRange\";\r\nimport type { AnimationEvent } from \"./animationEvent\";\r\nimport { Node } from \"../node\";\r\nimport type { IAnimatable } from \"./animatable.interface\";\r\nimport { Size } from \"../Maths/math.size\";\r\nimport { WebRequest } from \"../Misc/webRequest\";\r\nimport { Constants } from \"../Engines/constants\";\r\n\r\nimport type { Animatable } from \"./animatable\";\r\nimport type { RuntimeAnimation } from \"./runtimeAnimation\";\r\n\r\n/**\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class _IAnimationState {\r\n key: number;\r\n repeatCount: number;\r\n workValue?: any;\r\n loopMode?: number;\r\n offsetValue?: any;\r\n highLimitValue?: any;\r\n}\r\n\r\n/**\r\n * Class used to store any kind of animation\r\n */\r\nexport class Animation {\r\n private static _UniqueIdGenerator = 0;\r\n\r\n /**\r\n * Use matrix interpolation instead of using direct key value when animating matrices\r\n */\r\n public static AllowMatricesInterpolation = false;\r\n\r\n /**\r\n * When matrix interpolation is enabled, this boolean forces the system to use Matrix.DecomposeLerp instead of Matrix.Lerp. Interpolation is more precise but slower\r\n */\r\n public static AllowMatrixDecomposeForInterpolation = true;\r\n\r\n /**\r\n * Gets or sets the unique id of the animation (the uniqueness is solely among other animations)\r\n */\r\n public uniqueId: number;\r\n\r\n /** Define the Url to load snippets */\r\n public static SnippetUrl = Constants.SnippetUrl;\r\n\r\n /** Snippet ID if the animation was created from the snippet server */\r\n public snippetId: string;\r\n\r\n /**\r\n * Stores the key frames of the animation\r\n */\r\n private _keys: Array;\r\n\r\n /**\r\n * Stores the easing function of the animation\r\n */\r\n private _easingFunction: Nullable = null;\r\n\r\n /**\r\n * @internal Internal use only\r\n */\r\n public _runtimeAnimations = new Array();\r\n\r\n /**\r\n * The set of event that will be linked to this animation\r\n */\r\n private _events = new Array();\r\n\r\n /**\r\n * Stores an array of target property paths\r\n */\r\n public targetPropertyPath: string[];\r\n\r\n /**\r\n * Stores the blending speed of the animation\r\n */\r\n public blendingSpeed = 0.01;\r\n\r\n /**\r\n * Stores the animation ranges for the animation\r\n */\r\n private _ranges: { [name: string]: Nullable } = {};\r\n\r\n /**\r\n * @internal Internal use\r\n */\r\n public static _PrepareAnimation(\r\n name: string,\r\n targetProperty: string,\r\n framePerSecond: number,\r\n totalFrame: number,\r\n from: any,\r\n to: any,\r\n loopMode?: number,\r\n easingFunction?: EasingFunction\r\n ): Nullable {\r\n let dataType = undefined;\r\n\r\n if (!isNaN(parseFloat(from)) && isFinite(from)) {\r\n dataType = Animation.ANIMATIONTYPE_FLOAT;\r\n } else if (from instanceof Quaternion) {\r\n dataType = Animation.ANIMATIONTYPE_QUATERNION;\r\n } else if (from instanceof Vector3) {\r\n dataType = Animation.ANIMATIONTYPE_VECTOR3;\r\n } else if (from instanceof Vector2) {\r\n dataType = Animation.ANIMATIONTYPE_VECTOR2;\r\n } else if (from instanceof Color3) {\r\n dataType = Animation.ANIMATIONTYPE_COLOR3;\r\n } else if (from instanceof Color4) {\r\n dataType = Animation.ANIMATIONTYPE_COLOR4;\r\n } else if (from instanceof Size) {\r\n dataType = Animation.ANIMATIONTYPE_SIZE;\r\n }\r\n\r\n if (dataType == undefined) {\r\n return null;\r\n }\r\n\r\n const animation = new Animation(name, targetProperty, framePerSecond, dataType, loopMode);\r\n\r\n const keys: Array = [\r\n { frame: 0, value: from },\r\n { frame: totalFrame, value: to },\r\n ];\r\n animation.setKeys(keys);\r\n\r\n if (easingFunction !== undefined) {\r\n animation.setEasingFunction(easingFunction);\r\n }\r\n\r\n return animation;\r\n }\r\n\r\n /**\r\n * Sets up an animation\r\n * @param property The property to animate\r\n * @param animationType The animation type to apply\r\n * @param framePerSecond The frames per second of the animation\r\n * @param easingFunction The easing function used in the animation\r\n * @returns The created animation\r\n */\r\n public static CreateAnimation(property: string, animationType: number, framePerSecond: number, easingFunction: EasingFunction): Animation {\r\n const animation: Animation = new Animation(property + \"Animation\", property, framePerSecond, animationType, Animation.ANIMATIONLOOPMODE_CONSTANT);\r\n\r\n animation.setEasingFunction(easingFunction);\r\n\r\n return animation;\r\n }\r\n\r\n /**\r\n * Create and start an animation on a node\r\n * @param name defines the name of the global animation that will be run on all nodes\r\n * @param target defines the target where the animation will take place\r\n * @param targetProperty defines property to animate\r\n * @param framePerSecond defines the number of frame per second yo use\r\n * @param totalFrame defines the number of frames in total\r\n * @param from defines the initial value\r\n * @param to defines the final value\r\n * @param loopMode defines which loop mode you want to use (off by default)\r\n * @param easingFunction defines the easing function to use (linear by default)\r\n * @param onAnimationEnd defines the callback to call when animation end\r\n * @param scene defines the hosting scene\r\n * @returns the animatable created for this animation\r\n */\r\n public static CreateAndStartAnimation(\r\n name: string,\r\n target: any,\r\n targetProperty: string,\r\n framePerSecond: number,\r\n totalFrame: number,\r\n from: any,\r\n to: any,\r\n loopMode?: number,\r\n easingFunction?: EasingFunction,\r\n onAnimationEnd?: () => void,\r\n scene?: Scene\r\n ): Nullable {\r\n const animation = Animation._PrepareAnimation(name, targetProperty, framePerSecond, totalFrame, from, to, loopMode, easingFunction);\r\n\r\n if (!animation) {\r\n return null;\r\n }\r\n\r\n if (target.getScene) {\r\n scene = target.getScene();\r\n }\r\n\r\n if (!scene) {\r\n return null;\r\n }\r\n\r\n return scene.beginDirectAnimation(target, [animation], 0, totalFrame, animation.loopMode === 1, 1.0, onAnimationEnd);\r\n }\r\n\r\n /**\r\n * Create and start an animation on a node and its descendants\r\n * @param name defines the name of the global animation that will be run on all nodes\r\n * @param node defines the root node where the animation will take place\r\n * @param directDescendantsOnly if true only direct descendants will be used, if false direct and also indirect (children of children, an so on in a recursive manner) descendants will be used\r\n * @param targetProperty defines property to animate\r\n * @param framePerSecond defines the number of frame per second to use\r\n * @param totalFrame defines the number of frames in total\r\n * @param from defines the initial value\r\n * @param to defines the final value\r\n * @param loopMode defines which loop mode you want to use (off by default)\r\n * @param easingFunction defines the easing function to use (linear by default)\r\n * @param onAnimationEnd defines the callback to call when an animation ends (will be called once per node)\r\n * @returns the list of animatables created for all nodes\r\n * @example https://www.babylonjs-playground.com/#MH0VLI\r\n */\r\n public static CreateAndStartHierarchyAnimation(\r\n name: string,\r\n node: Node,\r\n directDescendantsOnly: boolean,\r\n targetProperty: string,\r\n framePerSecond: number,\r\n totalFrame: number,\r\n from: any,\r\n to: any,\r\n loopMode?: number,\r\n easingFunction?: EasingFunction,\r\n onAnimationEnd?: () => void\r\n ): Nullable {\r\n const animation = Animation._PrepareAnimation(name, targetProperty, framePerSecond, totalFrame, from, to, loopMode, easingFunction);\r\n\r\n if (!animation) {\r\n return null;\r\n }\r\n\r\n const scene = node.getScene();\r\n return scene.beginDirectHierarchyAnimation(node, directDescendantsOnly, [animation], 0, totalFrame, animation.loopMode === 1, 1.0, onAnimationEnd);\r\n }\r\n\r\n /**\r\n * Creates a new animation, merges it with the existing animations and starts it\r\n * @param name Name of the animation\r\n * @param node Node which contains the scene that begins the animations\r\n * @param targetProperty Specifies which property to animate\r\n * @param framePerSecond The frames per second of the animation\r\n * @param totalFrame The total number of frames\r\n * @param from The frame at the beginning of the animation\r\n * @param to The frame at the end of the animation\r\n * @param loopMode Specifies the loop mode of the animation\r\n * @param easingFunction (Optional) The easing function of the animation, which allow custom mathematical formulas for animations\r\n * @param onAnimationEnd Callback to run once the animation is complete\r\n * @returns Nullable animation\r\n */\r\n public static CreateMergeAndStartAnimation(\r\n name: string,\r\n node: Node,\r\n targetProperty: string,\r\n framePerSecond: number,\r\n totalFrame: number,\r\n from: any,\r\n to: any,\r\n loopMode?: number,\r\n easingFunction?: EasingFunction,\r\n onAnimationEnd?: () => void\r\n ): Nullable {\r\n const animation = Animation._PrepareAnimation(name, targetProperty, framePerSecond, totalFrame, from, to, loopMode, easingFunction);\r\n\r\n if (!animation) {\r\n return null;\r\n }\r\n\r\n node.animations.push(animation);\r\n\r\n return node.getScene().beginAnimation(node, 0, totalFrame, animation.loopMode === 1, 1.0, onAnimationEnd);\r\n }\r\n\r\n /**\r\n * Convert the keyframes for all animations belonging to the group to be relative to a given reference frame.\r\n * @param sourceAnimation defines the Animation containing keyframes to convert\r\n * @param referenceFrame defines the frame that keyframes in the range will be relative to\r\n * @param range defines the name of the AnimationRange belonging to the Animation to convert\r\n * @param cloneOriginal defines whether or not to clone the animation and convert the clone or convert the original animation (default is false)\r\n * @param clonedName defines the name of the resulting cloned Animation if cloneOriginal is true\r\n * @returns a new Animation if cloneOriginal is true or the original Animation if cloneOriginal is false\r\n */\r\n public static MakeAnimationAdditive(sourceAnimation: Animation, referenceFrame = 0, range?: string, cloneOriginal = false, clonedName?: string): Animation {\r\n let animation = sourceAnimation;\r\n\r\n if (cloneOriginal) {\r\n animation = sourceAnimation.clone();\r\n animation.name = clonedName || animation.name;\r\n }\r\n\r\n if (!animation._keys.length) {\r\n return animation;\r\n }\r\n\r\n referenceFrame = referenceFrame >= 0 ? referenceFrame : 0;\r\n let startIndex = 0;\r\n const firstKey = animation._keys[0];\r\n let endIndex = animation._keys.length - 1;\r\n const lastKey = animation._keys[endIndex];\r\n const valueStore = {\r\n referenceValue: firstKey.value,\r\n referencePosition: TmpVectors.Vector3[0],\r\n referenceQuaternion: TmpVectors.Quaternion[0],\r\n referenceScaling: TmpVectors.Vector3[1],\r\n keyPosition: TmpVectors.Vector3[2],\r\n keyQuaternion: TmpVectors.Quaternion[1],\r\n keyScaling: TmpVectors.Vector3[3],\r\n };\r\n let referenceFound = false;\r\n let from = firstKey.frame;\r\n let to = lastKey.frame;\r\n if (range) {\r\n const rangeValue = animation.getRange(range);\r\n\r\n if (rangeValue) {\r\n from = rangeValue.from;\r\n to = rangeValue.to;\r\n }\r\n }\r\n let fromKeyFound = firstKey.frame === from;\r\n let toKeyFound = lastKey.frame === to;\r\n\r\n // There's only one key, so use it\r\n if (animation._keys.length === 1) {\r\n const value = animation._getKeyValue(animation._keys[0]);\r\n valueStore.referenceValue = value.clone ? value.clone() : value;\r\n referenceFound = true;\r\n }\r\n\r\n // Reference frame is before the first frame, so just use the first frame\r\n else if (referenceFrame <= firstKey.frame) {\r\n const value = animation._getKeyValue(firstKey.value);\r\n valueStore.referenceValue = value.clone ? value.clone() : value;\r\n referenceFound = true;\r\n }\r\n\r\n // Reference frame is after the last frame, so just use the last frame\r\n else if (referenceFrame >= lastKey.frame) {\r\n const value = animation._getKeyValue(lastKey.value);\r\n valueStore.referenceValue = value.clone ? value.clone() : value;\r\n referenceFound = true;\r\n }\r\n\r\n // Find key bookends, create them if they don't exist\r\n let index = 0;\r\n while (!referenceFound || !fromKeyFound || (!toKeyFound && index < animation._keys.length - 1)) {\r\n const currentKey = animation._keys[index];\r\n const nextKey = animation._keys[index + 1];\r\n\r\n // If reference frame wasn't found yet, check if we can interpolate to it\r\n if (!referenceFound && referenceFrame >= currentKey.frame && referenceFrame <= nextKey.frame) {\r\n let value;\r\n\r\n if (referenceFrame === currentKey.frame) {\r\n value = animation._getKeyValue(currentKey.value);\r\n } else if (referenceFrame === nextKey.frame) {\r\n value = animation._getKeyValue(nextKey.value);\r\n } else {\r\n const animationState = {\r\n key: index,\r\n repeatCount: 0,\r\n loopMode: this.ANIMATIONLOOPMODE_CONSTANT,\r\n };\r\n value = animation._interpolate(referenceFrame, animationState);\r\n }\r\n\r\n valueStore.referenceValue = value.clone ? value.clone() : value;\r\n referenceFound = true;\r\n }\r\n\r\n // If from key wasn't found yet, check if we can interpolate to it\r\n if (!fromKeyFound && from >= currentKey.frame && from <= nextKey.frame) {\r\n if (from === currentKey.frame) {\r\n startIndex = index;\r\n } else if (from === nextKey.frame) {\r\n startIndex = index + 1;\r\n } else {\r\n const animationState = {\r\n key: index,\r\n repeatCount: 0,\r\n loopMode: this.ANIMATIONLOOPMODE_CONSTANT,\r\n };\r\n const value = animation._interpolate(from, animationState);\r\n const key: IAnimationKey = {\r\n frame: from,\r\n value: value.clone ? value.clone() : value,\r\n };\r\n animation._keys.splice(index + 1, 0, key);\r\n startIndex = index + 1;\r\n }\r\n\r\n fromKeyFound = true;\r\n }\r\n\r\n // If to key wasn't found yet, check if we can interpolate to it\r\n if (!toKeyFound && to >= currentKey.frame && to <= nextKey.frame) {\r\n if (to === currentKey.frame) {\r\n endIndex = index;\r\n } else if (to === nextKey.frame) {\r\n endIndex = index + 1;\r\n } else {\r\n const animationState = {\r\n key: index,\r\n repeatCount: 0,\r\n loopMode: this.ANIMATIONLOOPMODE_CONSTANT,\r\n };\r\n const value = animation._interpolate(to, animationState);\r\n const key: IAnimationKey = {\r\n frame: to,\r\n value: value.clone ? value.clone() : value,\r\n };\r\n animation._keys.splice(index + 1, 0, key);\r\n endIndex = index + 1;\r\n }\r\n\r\n toKeyFound = true;\r\n }\r\n\r\n index++;\r\n }\r\n\r\n // Conjugate the quaternion\r\n if (animation.dataType === Animation.ANIMATIONTYPE_QUATERNION) {\r\n valueStore.referenceValue.normalize().conjugateInPlace();\r\n }\r\n\r\n // Decompose matrix and conjugate the quaternion\r\n else if (animation.dataType === Animation.ANIMATIONTYPE_MATRIX) {\r\n valueStore.referenceValue.decompose(valueStore.referenceScaling, valueStore.referenceQuaternion, valueStore.referencePosition);\r\n valueStore.referenceQuaternion.normalize().conjugateInPlace();\r\n }\r\n\r\n // Subtract the reference value from all of the key values\r\n for (index = startIndex; index <= endIndex; index++) {\r\n const key = animation._keys[index];\r\n\r\n // If this key was duplicated to create a frame 0 key, skip it because its value has already been updated\r\n if (index && animation.dataType !== Animation.ANIMATIONTYPE_FLOAT && key.value === firstKey.value) {\r\n continue;\r\n }\r\n\r\n switch (animation.dataType) {\r\n case Animation.ANIMATIONTYPE_MATRIX:\r\n key.value.decompose(valueStore.keyScaling, valueStore.keyQuaternion, valueStore.keyPosition);\r\n valueStore.keyPosition.subtractInPlace(valueStore.referencePosition);\r\n valueStore.keyScaling.divideInPlace(valueStore.referenceScaling);\r\n valueStore.referenceQuaternion.multiplyToRef(valueStore.keyQuaternion, valueStore.keyQuaternion);\r\n Matrix.ComposeToRef(valueStore.keyScaling, valueStore.keyQuaternion, valueStore.keyPosition, key.value);\r\n break;\r\n\r\n case Animation.ANIMATIONTYPE_QUATERNION:\r\n valueStore.referenceValue.multiplyToRef(key.value, key.value);\r\n break;\r\n\r\n case Animation.ANIMATIONTYPE_VECTOR2:\r\n case Animation.ANIMATIONTYPE_VECTOR3:\r\n case Animation.ANIMATIONTYPE_COLOR3:\r\n case Animation.ANIMATIONTYPE_COLOR4:\r\n key.value.subtractToRef(valueStore.referenceValue, key.value);\r\n break;\r\n\r\n case Animation.ANIMATIONTYPE_SIZE:\r\n key.value.width -= valueStore.referenceValue.width;\r\n key.value.height -= valueStore.referenceValue.height;\r\n break;\r\n\r\n default:\r\n key.value -= valueStore.referenceValue;\r\n }\r\n }\r\n\r\n return animation;\r\n }\r\n\r\n /**\r\n * Transition property of an host to the target Value\r\n * @param property The property to transition\r\n * @param targetValue The target Value of the property\r\n * @param host The object where the property to animate belongs\r\n * @param scene Scene used to run the animation\r\n * @param frameRate Framerate (in frame/s) to use\r\n * @param transition The transition type we want to use\r\n * @param duration The duration of the animation, in milliseconds\r\n * @param onAnimationEnd Callback trigger at the end of the animation\r\n * @returns Nullable animation\r\n */\r\n public static TransitionTo(\r\n property: string,\r\n targetValue: any,\r\n host: any,\r\n scene: Scene,\r\n frameRate: number,\r\n transition: Animation,\r\n duration: number,\r\n onAnimationEnd: Nullable<() => void> = null\r\n ): Nullable {\r\n if (duration <= 0) {\r\n host[property] = targetValue;\r\n if (onAnimationEnd) {\r\n onAnimationEnd();\r\n }\r\n return null;\r\n }\r\n\r\n const endFrame: number = frameRate * (duration / 1000);\r\n\r\n transition.setKeys([\r\n {\r\n frame: 0,\r\n value: host[property].clone ? host[property].clone() : host[property],\r\n },\r\n {\r\n frame: endFrame,\r\n value: targetValue,\r\n },\r\n ]);\r\n\r\n if (!host.animations) {\r\n host.animations = [];\r\n }\r\n\r\n host.animations.push(transition);\r\n\r\n const animation: Animatable = scene.beginAnimation(host, 0, endFrame, false);\r\n animation.onAnimationEnd = onAnimationEnd;\r\n return animation;\r\n }\r\n\r\n /**\r\n * Return the array of runtime animations currently using this animation\r\n */\r\n public get runtimeAnimations(): RuntimeAnimation[] {\r\n return this._runtimeAnimations;\r\n }\r\n\r\n /**\r\n * Specifies if any of the runtime animations are currently running\r\n */\r\n public get hasRunningRuntimeAnimations(): boolean {\r\n for (const runtimeAnimation of this._runtimeAnimations) {\r\n if (!runtimeAnimation.isStopped()) {\r\n return true;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Initializes the animation\r\n * @param name Name of the animation\r\n * @param targetProperty Property to animate\r\n * @param framePerSecond The frames per second of the animation\r\n * @param dataType The data type of the animation\r\n * @param loopMode The loop mode of the animation\r\n * @param enableBlending Specifies if blending should be enabled\r\n */\r\n constructor(\r\n /**Name of the animation */\r\n public name: string,\r\n /**Property to animate */\r\n public targetProperty: string,\r\n /**The frames per second of the animation */\r\n public framePerSecond: number,\r\n /**The data type of the animation */\r\n public dataType: number,\r\n /**The loop mode of the animation */\r\n public loopMode?: number,\r\n /**Specifies if blending should be enabled */\r\n public enableBlending?: boolean\r\n ) {\r\n this.targetPropertyPath = targetProperty.split(\".\");\r\n this.dataType = dataType;\r\n this.loopMode = loopMode === undefined ? Animation.ANIMATIONLOOPMODE_CYCLE : loopMode;\r\n this.uniqueId = Animation._UniqueIdGenerator++;\r\n }\r\n\r\n // Methods\r\n /**\r\n * Converts the animation to a string\r\n * @param fullDetails support for multiple levels of logging within scene loading\r\n * @returns String form of the animation\r\n */\r\n public toString(fullDetails?: boolean): string {\r\n let ret = \"Name: \" + this.name + \", property: \" + this.targetProperty;\r\n ret += \", datatype: \" + [\"Float\", \"Vector3\", \"Quaternion\", \"Matrix\", \"Color3\", \"Vector2\"][this.dataType];\r\n ret += \", nKeys: \" + (this._keys ? this._keys.length : \"none\");\r\n ret += \", nRanges: \" + (this._ranges ? Object.keys(this._ranges).length : \"none\");\r\n if (fullDetails) {\r\n ret += \", Ranges: {\";\r\n let first = true;\r\n for (const name in this._ranges) {\r\n if (first) {\r\n ret += \", \";\r\n first = false;\r\n }\r\n ret += name;\r\n }\r\n ret += \"}\";\r\n }\r\n return ret;\r\n }\r\n\r\n /**\r\n * Add an event to this animation\r\n * @param event Event to add\r\n */\r\n public addEvent(event: AnimationEvent): void {\r\n this._events.push(event);\r\n this._events.sort((a, b) => a.frame - b.frame);\r\n }\r\n\r\n /**\r\n * Remove all events found at the given frame\r\n * @param frame The frame to remove events from\r\n */\r\n public removeEvents(frame: number): void {\r\n for (let index = 0; index < this._events.length; index++) {\r\n if (this._events[index].frame === frame) {\r\n this._events.splice(index, 1);\r\n index--;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Retrieves all the events from the animation\r\n * @returns Events from the animation\r\n */\r\n public getEvents(): AnimationEvent[] {\r\n return this._events;\r\n }\r\n\r\n /**\r\n * Creates an animation range\r\n * @param name Name of the animation range\r\n * @param from Starting frame of the animation range\r\n * @param to Ending frame of the animation\r\n */\r\n public createRange(name: string, from: number, to: number): void {\r\n // check name not already in use; could happen for bones after serialized\r\n if (!this._ranges[name]) {\r\n this._ranges[name] = new AnimationRange(name, from, to);\r\n }\r\n }\r\n\r\n /**\r\n * Deletes an animation range by name\r\n * @param name Name of the animation range to delete\r\n * @param deleteFrames Specifies if the key frames for the range should also be deleted (true) or not (false)\r\n */\r\n public deleteRange(name: string, deleteFrames = true): void {\r\n const range = this._ranges[name];\r\n if (!range) {\r\n return;\r\n }\r\n if (deleteFrames) {\r\n const from = range.from;\r\n const to = range.to;\r\n\r\n // this loop MUST go high to low for multiple splices to work\r\n for (let key = this._keys.length - 1; key >= 0; key--) {\r\n if (this._keys[key].frame >= from && this._keys[key].frame <= to) {\r\n this._keys.splice(key, 1);\r\n }\r\n }\r\n }\r\n this._ranges[name] = null; // said much faster than 'delete this._range[name]'\r\n }\r\n\r\n /**\r\n * Gets the animation range by name, or null if not defined\r\n * @param name Name of the animation range\r\n * @returns Nullable animation range\r\n */\r\n public getRange(name: string): Nullable {\r\n return this._ranges[name];\r\n }\r\n\r\n /**\r\n * Gets the key frames from the animation\r\n * @returns The key frames of the animation\r\n */\r\n public getKeys(): Array {\r\n return this._keys;\r\n }\r\n\r\n /**\r\n * Gets the highest frame rate of the animation\r\n * @returns Highest frame rate of the animation\r\n */\r\n public getHighestFrame(): number {\r\n let ret = 0;\r\n\r\n for (let key = 0, nKeys = this._keys.length; key < nKeys; key++) {\r\n if (ret < this._keys[key].frame) {\r\n ret = this._keys[key].frame;\r\n }\r\n }\r\n return ret;\r\n }\r\n\r\n /**\r\n * Gets the easing function of the animation\r\n * @returns Easing function of the animation\r\n */\r\n public getEasingFunction(): Nullable {\r\n return this._easingFunction;\r\n }\r\n\r\n /**\r\n * Sets the easing function of the animation\r\n * @param easingFunction A custom mathematical formula for animation\r\n */\r\n public setEasingFunction(easingFunction: Nullable): void {\r\n this._easingFunction = easingFunction;\r\n }\r\n\r\n /**\r\n * Interpolates a scalar linearly\r\n * @param startValue Start value of the animation curve\r\n * @param endValue End value of the animation curve\r\n * @param gradient Scalar amount to interpolate\r\n * @returns Interpolated scalar value\r\n */\r\n public floatInterpolateFunction(startValue: number, endValue: number, gradient: number): number {\r\n return Scalar.Lerp(startValue, endValue, gradient);\r\n }\r\n\r\n /**\r\n * Interpolates a scalar cubically\r\n * @param startValue Start value of the animation curve\r\n * @param outTangent End tangent of the animation\r\n * @param endValue End value of the animation curve\r\n * @param inTangent Start tangent of the animation curve\r\n * @param gradient Scalar amount to interpolate\r\n * @returns Interpolated scalar value\r\n */\r\n public floatInterpolateFunctionWithTangents(startValue: number, outTangent: number, endValue: number, inTangent: number, gradient: number): number {\r\n return Scalar.Hermite(startValue, outTangent, endValue, inTangent, gradient);\r\n }\r\n\r\n /**\r\n * Interpolates a quaternion using a spherical linear interpolation\r\n * @param startValue Start value of the animation curve\r\n * @param endValue End value of the animation curve\r\n * @param gradient Scalar amount to interpolate\r\n * @returns Interpolated quaternion value\r\n */\r\n public quaternionInterpolateFunction(startValue: Quaternion, endValue: Quaternion, gradient: number): Quaternion {\r\n return Quaternion.Slerp(startValue, endValue, gradient);\r\n }\r\n\r\n /**\r\n * Interpolates a quaternion cubically\r\n * @param startValue Start value of the animation curve\r\n * @param outTangent End tangent of the animation curve\r\n * @param endValue End value of the animation curve\r\n * @param inTangent Start tangent of the animation curve\r\n * @param gradient Scalar amount to interpolate\r\n * @returns Interpolated quaternion value\r\n */\r\n public quaternionInterpolateFunctionWithTangents(startValue: Quaternion, outTangent: Quaternion, endValue: Quaternion, inTangent: Quaternion, gradient: number): Quaternion {\r\n return Quaternion.Hermite(startValue, outTangent, endValue, inTangent, gradient).normalize();\r\n }\r\n\r\n /**\r\n * Interpolates a Vector3 linearly\r\n * @param startValue Start value of the animation curve\r\n * @param endValue End value of the animation curve\r\n * @param gradient Scalar amount to interpolate (value between 0 and 1)\r\n * @returns Interpolated scalar value\r\n */\r\n public vector3InterpolateFunction(startValue: Vector3, endValue: Vector3, gradient: number): Vector3 {\r\n return Vector3.Lerp(startValue, endValue, gradient);\r\n }\r\n\r\n /**\r\n * Interpolates a Vector3 cubically\r\n * @param startValue Start value of the animation curve\r\n * @param outTangent End tangent of the animation\r\n * @param endValue End value of the animation curve\r\n * @param inTangent Start tangent of the animation curve\r\n * @param gradient Scalar amount to interpolate (value between 0 and 1)\r\n * @returns InterpolatedVector3 value\r\n */\r\n public vector3InterpolateFunctionWithTangents(startValue: Vector3, outTangent: Vector3, endValue: Vector3, inTangent: Vector3, gradient: number): Vector3 {\r\n return Vector3.Hermite(startValue, outTangent, endValue, inTangent, gradient);\r\n }\r\n\r\n /**\r\n * Interpolates a Vector2 linearly\r\n * @param startValue Start value of the animation curve\r\n * @param endValue End value of the animation curve\r\n * @param gradient Scalar amount to interpolate (value between 0 and 1)\r\n * @returns Interpolated Vector2 value\r\n */\r\n public vector2InterpolateFunction(startValue: Vector2, endValue: Vector2, gradient: number): Vector2 {\r\n return Vector2.Lerp(startValue, endValue, gradient);\r\n }\r\n\r\n /**\r\n * Interpolates a Vector2 cubically\r\n * @param startValue Start value of the animation curve\r\n * @param outTangent End tangent of the animation\r\n * @param endValue End value of the animation curve\r\n * @param inTangent Start tangent of the animation curve\r\n * @param gradient Scalar amount to interpolate (value between 0 and 1)\r\n * @returns Interpolated Vector2 value\r\n */\r\n public vector2InterpolateFunctionWithTangents(startValue: Vector2, outTangent: Vector2, endValue: Vector2, inTangent: Vector2, gradient: number): Vector2 {\r\n return Vector2.Hermite(startValue, outTangent, endValue, inTangent, gradient);\r\n }\r\n\r\n /**\r\n * Interpolates a size linearly\r\n * @param startValue Start value of the animation curve\r\n * @param endValue End value of the animation curve\r\n * @param gradient Scalar amount to interpolate\r\n * @returns Interpolated Size value\r\n */\r\n public sizeInterpolateFunction(startValue: Size, endValue: Size, gradient: number): Size {\r\n return Size.Lerp(startValue, endValue, gradient);\r\n }\r\n\r\n /**\r\n * Interpolates a Color3 linearly\r\n * @param startValue Start value of the animation curve\r\n * @param endValue End value of the animation curve\r\n * @param gradient Scalar amount to interpolate\r\n * @returns Interpolated Color3 value\r\n */\r\n public color3InterpolateFunction(startValue: Color3, endValue: Color3, gradient: number): Color3 {\r\n return Color3.Lerp(startValue, endValue, gradient);\r\n }\r\n\r\n /**\r\n * Interpolates a Color3 cubically\r\n * @param startValue Start value of the animation curve\r\n * @param outTangent End tangent of the animation\r\n * @param endValue End value of the animation curve\r\n * @param inTangent Start tangent of the animation curve\r\n * @param gradient Scalar amount to interpolate\r\n * @returns interpolated value\r\n */\r\n public color3InterpolateFunctionWithTangents(startValue: Color3, outTangent: Color3, endValue: Color3, inTangent: Color3, gradient: number): Color3 {\r\n return Color3.Hermite(startValue, outTangent, endValue, inTangent, gradient);\r\n }\r\n\r\n /**\r\n * Interpolates a Color4 linearly\r\n * @param startValue Start value of the animation curve\r\n * @param endValue End value of the animation curve\r\n * @param gradient Scalar amount to interpolate\r\n * @returns Interpolated Color3 value\r\n */\r\n public color4InterpolateFunction(startValue: Color4, endValue: Color4, gradient: number): Color4 {\r\n return Color4.Lerp(startValue, endValue, gradient);\r\n }\r\n\r\n /**\r\n * Interpolates a Color4 cubically\r\n * @param startValue Start value of the animation curve\r\n * @param outTangent End tangent of the animation\r\n * @param endValue End value of the animation curve\r\n * @param inTangent Start tangent of the animation curve\r\n * @param gradient Scalar amount to interpolate\r\n * @returns interpolated value\r\n */\r\n public color4InterpolateFunctionWithTangents(startValue: Color4, outTangent: Color4, endValue: Color4, inTangent: Color4, gradient: number): Color4 {\r\n return Color4.Hermite(startValue, outTangent, endValue, inTangent, gradient);\r\n }\r\n\r\n /**\r\n * @internal Internal use only\r\n */\r\n public _getKeyValue(value: any): any {\r\n if (typeof value === \"function\") {\r\n return value();\r\n }\r\n\r\n return value;\r\n }\r\n\r\n /**\r\n * Evaluate the animation value at a given frame\r\n * @param currentFrame defines the frame where we want to evaluate the animation\r\n * @returns the animation value\r\n */\r\n public evaluate(currentFrame: number) {\r\n return this._interpolate(currentFrame, {\r\n key: 0,\r\n repeatCount: 0,\r\n loopMode: Animation.ANIMATIONLOOPMODE_CONSTANT,\r\n });\r\n }\r\n\r\n /**\r\n * @internal Internal use only\r\n */\r\n public _interpolate(currentFrame: number, state: _IAnimationState): any {\r\n if (state.loopMode === Animation.ANIMATIONLOOPMODE_CONSTANT && state.repeatCount > 0) {\r\n return state.highLimitValue.clone ? state.highLimitValue.clone() : state.highLimitValue;\r\n }\r\n\r\n const keys = this._keys;\r\n const keysLength = keys.length;\r\n\r\n let key = state.key;\r\n\r\n while (key >= 0 && currentFrame < keys[key].frame) {\r\n --key;\r\n }\r\n\r\n while (key + 1 <= keysLength - 1 && currentFrame >= keys[key + 1].frame) {\r\n ++key;\r\n }\r\n\r\n state.key = key;\r\n\r\n if (key < 0) {\r\n return this._getKeyValue(keys[0].value);\r\n } else if (key + 1 > keysLength - 1) {\r\n return this._getKeyValue(keys[keysLength - 1].value);\r\n }\r\n\r\n const startKey = keys[key];\r\n const endKey = keys[key + 1];\r\n const startValue = this._getKeyValue(startKey.value);\r\n const endValue = this._getKeyValue(endKey.value);\r\n if (startKey.interpolation === AnimationKeyInterpolation.STEP) {\r\n if (endKey.frame > currentFrame) {\r\n return startValue;\r\n } else {\r\n return endValue;\r\n }\r\n }\r\n\r\n const useTangent = startKey.outTangent !== undefined && endKey.inTangent !== undefined;\r\n const frameDelta = endKey.frame - startKey.frame;\r\n\r\n // gradient : percent of currentFrame between the frame inf and the frame sup\r\n let gradient = (currentFrame - startKey.frame) / frameDelta;\r\n\r\n // check for easingFunction and correction of gradient\r\n const easingFunction = this.getEasingFunction();\r\n if (easingFunction !== null) {\r\n gradient = easingFunction.ease(gradient);\r\n }\r\n\r\n switch (this.dataType) {\r\n // Float\r\n case Animation.ANIMATIONTYPE_FLOAT: {\r\n const floatValue = useTangent\r\n ? this.floatInterpolateFunctionWithTangents(startValue, startKey.outTangent * frameDelta, endValue, endKey.inTangent * frameDelta, gradient)\r\n : this.floatInterpolateFunction(startValue, endValue, gradient);\r\n switch (state.loopMode) {\r\n case Animation.ANIMATIONLOOPMODE_CYCLE:\r\n case Animation.ANIMATIONLOOPMODE_CONSTANT:\r\n case Animation.ANIMATIONLOOPMODE_YOYO:\r\n return floatValue;\r\n case Animation.ANIMATIONLOOPMODE_RELATIVE:\r\n return state.offsetValue * state.repeatCount + floatValue;\r\n }\r\n break;\r\n }\r\n // Quaternion\r\n case Animation.ANIMATIONTYPE_QUATERNION: {\r\n const quatValue = useTangent\r\n ? this.quaternionInterpolateFunctionWithTangents(startValue, startKey.outTangent.scale(frameDelta), endValue, endKey.inTangent.scale(frameDelta), gradient)\r\n : this.quaternionInterpolateFunction(startValue, endValue, gradient);\r\n switch (state.loopMode) {\r\n case Animation.ANIMATIONLOOPMODE_CYCLE:\r\n case Animation.ANIMATIONLOOPMODE_CONSTANT:\r\n case Animation.ANIMATIONLOOPMODE_YOYO:\r\n return quatValue;\r\n case Animation.ANIMATIONLOOPMODE_RELATIVE:\r\n return quatValue.addInPlace(state.offsetValue.scale(state.repeatCount));\r\n }\r\n\r\n return quatValue;\r\n }\r\n // Vector3\r\n case Animation.ANIMATIONTYPE_VECTOR3: {\r\n const vec3Value = useTangent\r\n ? this.vector3InterpolateFunctionWithTangents(startValue, startKey.outTangent.scale(frameDelta), endValue, endKey.inTangent.scale(frameDelta), gradient)\r\n : this.vector3InterpolateFunction(startValue, endValue, gradient);\r\n switch (state.loopMode) {\r\n case Animation.ANIMATIONLOOPMODE_CYCLE:\r\n case Animation.ANIMATIONLOOPMODE_CONSTANT:\r\n case Animation.ANIMATIONLOOPMODE_YOYO:\r\n return vec3Value;\r\n case Animation.ANIMATIONLOOPMODE_RELATIVE:\r\n return vec3Value.add(state.offsetValue.scale(state.repeatCount));\r\n }\r\n break;\r\n }\r\n // Vector2\r\n case Animation.ANIMATIONTYPE_VECTOR2: {\r\n const vec2Value = useTangent\r\n ? this.vector2InterpolateFunctionWithTangents(startValue, startKey.outTangent.scale(frameDelta), endValue, endKey.inTangent.scale(frameDelta), gradient)\r\n : this.vector2InterpolateFunction(startValue, endValue, gradient);\r\n switch (state.loopMode) {\r\n case Animation.ANIMATIONLOOPMODE_CYCLE:\r\n case Animation.ANIMATIONLOOPMODE_CONSTANT:\r\n case Animation.ANIMATIONLOOPMODE_YOYO:\r\n return vec2Value;\r\n case Animation.ANIMATIONLOOPMODE_RELATIVE:\r\n return vec2Value.add(state.offsetValue.scale(state.repeatCount));\r\n }\r\n break;\r\n }\r\n // Size\r\n case Animation.ANIMATIONTYPE_SIZE: {\r\n switch (state.loopMode) {\r\n case Animation.ANIMATIONLOOPMODE_CYCLE:\r\n case Animation.ANIMATIONLOOPMODE_CONSTANT:\r\n case Animation.ANIMATIONLOOPMODE_YOYO:\r\n return this.sizeInterpolateFunction(startValue, endValue, gradient);\r\n case Animation.ANIMATIONLOOPMODE_RELATIVE:\r\n return this.sizeInterpolateFunction(startValue, endValue, gradient).add(state.offsetValue.scale(state.repeatCount));\r\n }\r\n break;\r\n }\r\n // Color3\r\n case Animation.ANIMATIONTYPE_COLOR3: {\r\n const color3Value = useTangent\r\n ? this.color3InterpolateFunctionWithTangents(startValue, startKey.outTangent.scale(frameDelta), endValue, endKey.inTangent.scale(frameDelta), gradient)\r\n : this.color3InterpolateFunction(startValue, endValue, gradient);\r\n switch (state.loopMode) {\r\n case Animation.ANIMATIONLOOPMODE_CYCLE:\r\n case Animation.ANIMATIONLOOPMODE_CONSTANT:\r\n case Animation.ANIMATIONLOOPMODE_YOYO:\r\n return color3Value;\r\n case Animation.ANIMATIONLOOPMODE_RELATIVE:\r\n return color3Value.add(state.offsetValue.scale(state.repeatCount));\r\n }\r\n break;\r\n }\r\n // Color4\r\n case Animation.ANIMATIONTYPE_COLOR4: {\r\n const color4Value = useTangent\r\n ? this.color4InterpolateFunctionWithTangents(startValue, startKey.outTangent.scale(frameDelta), endValue, endKey.inTangent.scale(frameDelta), gradient)\r\n : this.color4InterpolateFunction(startValue, endValue, gradient);\r\n switch (state.loopMode) {\r\n case Animation.ANIMATIONLOOPMODE_CYCLE:\r\n case Animation.ANIMATIONLOOPMODE_CONSTANT:\r\n case Animation.ANIMATIONLOOPMODE_YOYO:\r\n return color4Value;\r\n case Animation.ANIMATIONLOOPMODE_RELATIVE:\r\n return color4Value.add(state.offsetValue.scale(state.repeatCount));\r\n }\r\n break;\r\n }\r\n // Matrix\r\n case Animation.ANIMATIONTYPE_MATRIX: {\r\n switch (state.loopMode) {\r\n case Animation.ANIMATIONLOOPMODE_CYCLE:\r\n case Animation.ANIMATIONLOOPMODE_CONSTANT:\r\n case Animation.ANIMATIONLOOPMODE_YOYO: {\r\n if (Animation.AllowMatricesInterpolation) {\r\n return this.matrixInterpolateFunction(startValue, endValue, gradient, state.workValue);\r\n }\r\n return startValue;\r\n }\r\n case Animation.ANIMATIONLOOPMODE_RELATIVE: {\r\n return startValue;\r\n }\r\n }\r\n break;\r\n }\r\n }\r\n\r\n return 0;\r\n }\r\n\r\n /**\r\n * Defines the function to use to interpolate matrices\r\n * @param startValue defines the start matrix\r\n * @param endValue defines the end matrix\r\n * @param gradient defines the gradient between both matrices\r\n * @param result defines an optional target matrix where to store the interpolation\r\n * @returns the interpolated matrix\r\n */\r\n public matrixInterpolateFunction(startValue: Matrix, endValue: Matrix, gradient: number, result?: Matrix): Matrix {\r\n if (Animation.AllowMatrixDecomposeForInterpolation) {\r\n if (result) {\r\n Matrix.DecomposeLerpToRef(startValue, endValue, gradient, result);\r\n return result;\r\n }\r\n return Matrix.DecomposeLerp(startValue, endValue, gradient);\r\n }\r\n\r\n if (result) {\r\n Matrix.LerpToRef(startValue, endValue, gradient, result);\r\n return result;\r\n }\r\n return Matrix.Lerp(startValue, endValue, gradient);\r\n }\r\n\r\n /**\r\n * Makes a copy of the animation\r\n * @returns Cloned animation\r\n */\r\n public clone(): Animation {\r\n const clone = new Animation(this.name, this.targetPropertyPath.join(\".\"), this.framePerSecond, this.dataType, this.loopMode);\r\n\r\n clone.enableBlending = this.enableBlending;\r\n clone.blendingSpeed = this.blendingSpeed;\r\n\r\n if (this._keys) {\r\n clone.setKeys(this._keys);\r\n }\r\n\r\n if (this._ranges) {\r\n clone._ranges = {};\r\n for (const name in this._ranges) {\r\n const range = this._ranges[name];\r\n if (!range) {\r\n continue;\r\n }\r\n clone._ranges[name] = range.clone();\r\n }\r\n }\r\n\r\n return clone;\r\n }\r\n\r\n /**\r\n * Sets the key frames of the animation\r\n * @param values The animation key frames to set\r\n */\r\n public setKeys(values: Array): void {\r\n this._keys = values.slice(0);\r\n }\r\n\r\n /**\r\n * Serializes the animation to an object\r\n * @returns Serialized object\r\n */\r\n public serialize(): any {\r\n const serializationObject: any = {};\r\n\r\n serializationObject.name = this.name;\r\n serializationObject.property = this.targetProperty;\r\n serializationObject.framePerSecond = this.framePerSecond;\r\n serializationObject.dataType = this.dataType;\r\n serializationObject.loopBehavior = this.loopMode;\r\n serializationObject.enableBlending = this.enableBlending;\r\n serializationObject.blendingSpeed = this.blendingSpeed;\r\n\r\n const dataType = this.dataType;\r\n serializationObject.keys = [];\r\n const keys = this.getKeys();\r\n for (let index = 0; index < keys.length; index++) {\r\n const animationKey = keys[index];\r\n\r\n const key: any = {};\r\n key.frame = animationKey.frame;\r\n\r\n switch (dataType) {\r\n case Animation.ANIMATIONTYPE_FLOAT:\r\n key.values = [animationKey.value];\r\n if (animationKey.inTangent !== undefined) {\r\n key.values.push(animationKey.inTangent);\r\n }\r\n if (animationKey.outTangent !== undefined) {\r\n if (animationKey.inTangent === undefined) {\r\n key.values.push(undefined);\r\n }\r\n key.values.push(animationKey.outTangent);\r\n }\r\n if (animationKey.interpolation !== undefined) {\r\n if (animationKey.inTangent === undefined) {\r\n key.values.push(undefined);\r\n }\r\n if (animationKey.outTangent === undefined) {\r\n key.values.push(undefined);\r\n }\r\n key.values.push(animationKey.interpolation);\r\n }\r\n break;\r\n case Animation.ANIMATIONTYPE_QUATERNION:\r\n case Animation.ANIMATIONTYPE_MATRIX:\r\n case Animation.ANIMATIONTYPE_VECTOR3:\r\n case Animation.ANIMATIONTYPE_COLOR3:\r\n case Animation.ANIMATIONTYPE_COLOR4:\r\n key.values = animationKey.value.asArray();\r\n if (animationKey.inTangent != undefined) {\r\n key.values.push(animationKey.inTangent.asArray());\r\n }\r\n if (animationKey.outTangent != undefined) {\r\n if (animationKey.inTangent === undefined) {\r\n key.values.push(undefined);\r\n }\r\n key.values.push(animationKey.outTangent.asArray());\r\n }\r\n if (animationKey.interpolation !== undefined) {\r\n if (animationKey.inTangent === undefined) {\r\n key.values.push(undefined);\r\n }\r\n if (animationKey.outTangent === undefined) {\r\n key.values.push(undefined);\r\n }\r\n key.values.push(animationKey.interpolation);\r\n }\r\n break;\r\n }\r\n\r\n serializationObject.keys.push(key);\r\n }\r\n\r\n serializationObject.ranges = [];\r\n for (const name in this._ranges) {\r\n const source = this._ranges[name];\r\n\r\n if (!source) {\r\n continue;\r\n }\r\n const range: any = {};\r\n range.name = name;\r\n range.from = source.from;\r\n range.to = source.to;\r\n serializationObject.ranges.push(range);\r\n }\r\n\r\n return serializationObject;\r\n }\r\n\r\n // Statics\r\n /**\r\n * Float animation type\r\n */\r\n public static readonly ANIMATIONTYPE_FLOAT = 0;\r\n /**\r\n * Vector3 animation type\r\n */\r\n public static readonly ANIMATIONTYPE_VECTOR3 = 1;\r\n /**\r\n * Quaternion animation type\r\n */\r\n public static readonly ANIMATIONTYPE_QUATERNION = 2;\r\n /**\r\n * Matrix animation type\r\n */\r\n public static readonly ANIMATIONTYPE_MATRIX = 3;\r\n /**\r\n * Color3 animation type\r\n */\r\n public static readonly ANIMATIONTYPE_COLOR3 = 4;\r\n /**\r\n * Color3 animation type\r\n */\r\n public static readonly ANIMATIONTYPE_COLOR4 = 7;\r\n /**\r\n * Vector2 animation type\r\n */\r\n public static readonly ANIMATIONTYPE_VECTOR2 = 5;\r\n /**\r\n * Size animation type\r\n */\r\n public static readonly ANIMATIONTYPE_SIZE = 6;\r\n /**\r\n * Relative Loop Mode\r\n */\r\n public static readonly ANIMATIONLOOPMODE_RELATIVE = 0;\r\n /**\r\n * Cycle Loop Mode\r\n */\r\n public static readonly ANIMATIONLOOPMODE_CYCLE = 1;\r\n /**\r\n * Constant Loop Mode\r\n */\r\n public static readonly ANIMATIONLOOPMODE_CONSTANT = 2;\r\n /**\r\n * Yoyo Loop Mode\r\n */\r\n public static readonly ANIMATIONLOOPMODE_YOYO = 4;\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _UniversalLerp(left: any, right: any, amount: number): any {\r\n const constructor = left.constructor;\r\n if (constructor.Lerp) {\r\n // Lerp supported\r\n return constructor.Lerp(left, right, amount);\r\n } else if (constructor.Slerp) {\r\n // Slerp supported\r\n return constructor.Slerp(left, right, amount);\r\n } else if (left.toFixed) {\r\n // Number\r\n return left * (1.0 - amount) + amount * right;\r\n } else {\r\n // Blending not supported\r\n return right;\r\n }\r\n }\r\n\r\n /**\r\n * Parses an animation object and creates an animation\r\n * @param parsedAnimation Parsed animation object\r\n * @returns Animation object\r\n */\r\n public static Parse(parsedAnimation: any): Animation {\r\n const animation = new Animation(parsedAnimation.name, parsedAnimation.property, parsedAnimation.framePerSecond, parsedAnimation.dataType, parsedAnimation.loopBehavior);\r\n\r\n const dataType = parsedAnimation.dataType;\r\n const keys: Array = [];\r\n let data;\r\n let index: number;\r\n\r\n if (parsedAnimation.enableBlending) {\r\n animation.enableBlending = parsedAnimation.enableBlending;\r\n }\r\n\r\n if (parsedAnimation.blendingSpeed) {\r\n animation.blendingSpeed = parsedAnimation.blendingSpeed;\r\n }\r\n\r\n for (index = 0; index < parsedAnimation.keys.length; index++) {\r\n const key = parsedAnimation.keys[index];\r\n let inTangent: any = undefined;\r\n let outTangent: any = undefined;\r\n let interpolation: any = undefined;\r\n\r\n switch (dataType) {\r\n case Animation.ANIMATIONTYPE_FLOAT:\r\n data = key.values[0];\r\n if (key.values.length >= 2) {\r\n inTangent = key.values[1];\r\n }\r\n if (key.values.length >= 3) {\r\n outTangent = key.values[2];\r\n }\r\n if (key.values.length >= 4) {\r\n interpolation = key.values[3];\r\n }\r\n break;\r\n case Animation.ANIMATIONTYPE_QUATERNION:\r\n data = Quaternion.FromArray(key.values);\r\n if (key.values.length >= 8) {\r\n const _inTangent = Quaternion.FromArray(key.values.slice(4, 8));\r\n if (!_inTangent.equals(Quaternion.Zero())) {\r\n inTangent = _inTangent;\r\n }\r\n }\r\n if (key.values.length >= 12) {\r\n const _outTangent = Quaternion.FromArray(key.values.slice(8, 12));\r\n if (!_outTangent.equals(Quaternion.Zero())) {\r\n outTangent = _outTangent;\r\n }\r\n }\r\n if (key.values.length >= 13) {\r\n interpolation = key.values[12];\r\n }\r\n break;\r\n case Animation.ANIMATIONTYPE_MATRIX:\r\n data = Matrix.FromArray(key.values);\r\n if (key.values.length >= 17) {\r\n interpolation = key.values[16];\r\n }\r\n break;\r\n case Animation.ANIMATIONTYPE_COLOR3:\r\n data = Color3.FromArray(key.values);\r\n if (key.values[3]) {\r\n inTangent = Color3.FromArray(key.values[3]);\r\n }\r\n if (key.values[4]) {\r\n outTangent = Color3.FromArray(key.values[4]);\r\n }\r\n if (key.values[5]) {\r\n interpolation = key.values[5];\r\n }\r\n break;\r\n case Animation.ANIMATIONTYPE_COLOR4:\r\n data = Color4.FromArray(key.values);\r\n if (key.values[4]) {\r\n inTangent = Color4.FromArray(key.values[4]);\r\n }\r\n if (key.values[5]) {\r\n outTangent = Color4.FromArray(key.values[5]);\r\n }\r\n if (key.values[6]) {\r\n interpolation = Color4.FromArray(key.values[6]);\r\n }\r\n break;\r\n case Animation.ANIMATIONTYPE_VECTOR3:\r\n default:\r\n data = Vector3.FromArray(key.values);\r\n if (key.values[3]) {\r\n inTangent = Vector3.FromArray(key.values[3]);\r\n }\r\n if (key.values[4]) {\r\n outTangent = Vector3.FromArray(key.values[4]);\r\n }\r\n if (key.values[5]) {\r\n interpolation = key.values[5];\r\n }\r\n break;\r\n }\r\n\r\n const keyData: any = {};\r\n keyData.frame = key.frame;\r\n keyData.value = data;\r\n\r\n if (inTangent != undefined) {\r\n keyData.inTangent = inTangent;\r\n }\r\n if (outTangent != undefined) {\r\n keyData.outTangent = outTangent;\r\n }\r\n if (interpolation != undefined) {\r\n keyData.interpolation = interpolation;\r\n }\r\n keys.push(keyData);\r\n }\r\n\r\n animation.setKeys(keys);\r\n\r\n if (parsedAnimation.ranges) {\r\n for (index = 0; index < parsedAnimation.ranges.length; index++) {\r\n data = parsedAnimation.ranges[index];\r\n animation.createRange(data.name, data.from, data.to);\r\n }\r\n }\r\n\r\n return animation;\r\n }\r\n\r\n /**\r\n * Appends the serialized animations from the source animations\r\n * @param source Source containing the animations\r\n * @param destination Target to store the animations\r\n */\r\n public static AppendSerializedAnimations(source: IAnimatable, destination: any): void {\r\n SerializationHelper.AppendSerializedAnimations(source, destination);\r\n }\r\n\r\n /**\r\n * Creates a new animation or an array of animations from a snippet saved in a remote file\r\n * @param name defines the name of the animation to create (can be null or empty to use the one from the json data)\r\n * @param url defines the url to load from\r\n * @returns a promise that will resolve to the new animation or an array of animations\r\n */\r\n public static ParseFromFileAsync(name: Nullable, url: string): Promise> {\r\n return new Promise((resolve, reject) => {\r\n const request = new WebRequest();\r\n request.addEventListener(\"readystatechange\", () => {\r\n if (request.readyState == 4) {\r\n if (request.status == 200) {\r\n let serializationObject = JSON.parse(request.responseText);\r\n if (serializationObject.animations) {\r\n serializationObject = serializationObject.animations;\r\n }\r\n\r\n if (serializationObject.length) {\r\n const output = new Array();\r\n for (const serializedAnimation of serializationObject) {\r\n output.push(this.Parse(serializedAnimation));\r\n }\r\n\r\n resolve(output);\r\n } else {\r\n const output = this.Parse(serializationObject);\r\n\r\n if (name) {\r\n output.name = name;\r\n }\r\n\r\n resolve(output);\r\n }\r\n } else {\r\n reject(\"Unable to load the animation\");\r\n }\r\n }\r\n });\r\n\r\n request.open(\"GET\", url);\r\n request.send();\r\n });\r\n }\r\n\r\n /**\r\n * Creates an animation or an array of animations from a snippet saved by the Inspector\r\n * @param snippetId defines the snippet to load\r\n * @returns a promise that will resolve to the new animation or a new array of animations\r\n */\r\n public static ParseFromSnippetAsync(snippetId: string): Promise> {\r\n return new Promise((resolve, reject) => {\r\n const request = new WebRequest();\r\n request.addEventListener(\"readystatechange\", () => {\r\n if (request.readyState == 4) {\r\n if (request.status == 200) {\r\n const snippet = JSON.parse(JSON.parse(request.responseText).jsonPayload);\r\n\r\n if (snippet.animations) {\r\n const serializationObject = JSON.parse(snippet.animations);\r\n const outputs = new Array();\r\n for (const serializedAnimation of serializationObject.animations) {\r\n const output = this.Parse(serializedAnimation);\r\n output.snippetId = snippetId;\r\n outputs.push(output);\r\n }\r\n\r\n resolve(outputs);\r\n } else {\r\n const serializationObject = JSON.parse(snippet.animation);\r\n const output = this.Parse(serializationObject);\r\n\r\n output.snippetId = snippetId;\r\n\r\n resolve(output);\r\n }\r\n } else {\r\n reject(\"Unable to load the snippet \" + snippetId);\r\n }\r\n }\r\n });\r\n\r\n request.open(\"GET\", this.SnippetUrl + \"/\" + snippetId.replace(/#/g, \"/\"));\r\n request.send();\r\n });\r\n }\r\n\r\n /**\r\n * Creates an animation or an array of animations from a snippet saved by the Inspector\r\n * @deprecated Please use ParseFromSnippetAsync instead\r\n * @param snippetId defines the snippet to load\r\n * @returns a promise that will resolve to the new animation or a new array of animations\r\n */\r\n public static CreateFromSnippetAsync = Animation.ParseFromSnippetAsync;\r\n}\r\n\r\nRegisterClass(\"BABYLON.Animation\", Animation);\r\nNode._AnimationRangeFactory = (name: string, from: number, to: number) => new AnimationRange(name, from, to);\r\n","import type { Behavior } from \"../../Behaviors/behavior\";\r\nimport type { Camera } from \"../../Cameras/camera\";\r\nimport type { ArcRotateCamera } from \"../../Cameras/arcRotateCamera\";\r\nimport { BackEase, EasingFunction } from \"../../Animations/easing\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { Observer } from \"../../Misc/observable\";\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport type { Animatable } from \"../../Animations/animatable\";\r\nimport { Animation } from \"../../Animations/animation\";\r\n\r\n/**\r\n * Add a bouncing effect to an ArcRotateCamera when reaching a specified minimum and maximum radius\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#bouncing-behavior\r\n */\r\nexport class BouncingBehavior implements Behavior {\r\n /**\r\n * Gets the name of the behavior.\r\n */\r\n public get name(): string {\r\n return \"Bouncing\";\r\n }\r\n\r\n /**\r\n * The easing function used by animations\r\n */\r\n public static EasingFunction = new BackEase(0.3);\r\n\r\n /**\r\n * The easing mode used by animations\r\n */\r\n public static EasingMode = EasingFunction.EASINGMODE_EASEOUT;\r\n\r\n /**\r\n * The duration of the animation, in milliseconds\r\n */\r\n public transitionDuration = 450;\r\n\r\n /**\r\n * Length of the distance animated by the transition when lower radius is reached\r\n */\r\n public lowerRadiusTransitionRange = 2;\r\n\r\n /**\r\n * Length of the distance animated by the transition when upper radius is reached\r\n */\r\n public upperRadiusTransitionRange = -2;\r\n\r\n private _autoTransitionRange = false;\r\n\r\n /**\r\n * Gets a value indicating if the lowerRadiusTransitionRange and upperRadiusTransitionRange are defined automatically\r\n */\r\n public get autoTransitionRange(): boolean {\r\n return this._autoTransitionRange;\r\n }\r\n\r\n /**\r\n * Sets a value indicating if the lowerRadiusTransitionRange and upperRadiusTransitionRange are defined automatically\r\n * Transition ranges will be set to 5% of the bounding box diagonal in world space\r\n */\r\n public set autoTransitionRange(value: boolean) {\r\n if (this._autoTransitionRange === value) {\r\n return;\r\n }\r\n\r\n this._autoTransitionRange = value;\r\n\r\n const camera = this._attachedCamera;\r\n if (!camera) {\r\n return;\r\n }\r\n\r\n if (value) {\r\n this._onMeshTargetChangedObserver = camera.onMeshTargetChangedObservable.add((mesh) => {\r\n if (!mesh) {\r\n return;\r\n }\r\n\r\n mesh.computeWorldMatrix(true);\r\n const diagonal = mesh.getBoundingInfo().diagonalLength;\r\n\r\n this.lowerRadiusTransitionRange = diagonal * 0.05;\r\n this.upperRadiusTransitionRange = diagonal * 0.05;\r\n });\r\n } else if (this._onMeshTargetChangedObserver) {\r\n camera.onMeshTargetChangedObservable.remove(this._onMeshTargetChangedObserver);\r\n }\r\n }\r\n\r\n // Connection\r\n private _attachedCamera: Nullable;\r\n private _onAfterCheckInputsObserver: Nullable>;\r\n private _onMeshTargetChangedObserver: Nullable>>;\r\n\r\n /**\r\n * Initializes the behavior.\r\n */\r\n public init(): void {\r\n // Do nothing\r\n }\r\n\r\n /**\r\n * Attaches the behavior to its arc rotate camera.\r\n * @param camera Defines the camera to attach the behavior to\r\n */\r\n public attach(camera: ArcRotateCamera): void {\r\n this._attachedCamera = camera;\r\n this._onAfterCheckInputsObserver = camera.onAfterCheckInputsObservable.add(() => {\r\n if (!this._attachedCamera) {\r\n return;\r\n }\r\n\r\n // Add the bounce animation to the lower radius limit\r\n if (this._isRadiusAtLimit(this._attachedCamera.lowerRadiusLimit)) {\r\n this._applyBoundRadiusAnimation(this.lowerRadiusTransitionRange);\r\n }\r\n\r\n // Add the bounce animation to the upper radius limit\r\n if (this._isRadiusAtLimit(this._attachedCamera.upperRadiusLimit)) {\r\n this._applyBoundRadiusAnimation(this.upperRadiusTransitionRange);\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Detaches the behavior from its current arc rotate camera.\r\n */\r\n public detach(): void {\r\n if (!this._attachedCamera) {\r\n return;\r\n }\r\n if (this._onAfterCheckInputsObserver) {\r\n this._attachedCamera.onAfterCheckInputsObservable.remove(this._onAfterCheckInputsObserver);\r\n }\r\n if (this._onMeshTargetChangedObserver) {\r\n this._attachedCamera.onMeshTargetChangedObservable.remove(this._onMeshTargetChangedObserver);\r\n }\r\n this._attachedCamera = null;\r\n }\r\n\r\n // Animations\r\n private _radiusIsAnimating: boolean = false;\r\n private _radiusBounceTransition: Nullable = null;\r\n private _animatables = new Array();\r\n private _cachedWheelPrecision: number;\r\n\r\n /**\r\n * Checks if the camera radius is at the specified limit. Takes into account animation locks.\r\n * @param radiusLimit The limit to check against.\r\n * @returns Bool to indicate if at limit.\r\n */\r\n private _isRadiusAtLimit(radiusLimit: Nullable): boolean {\r\n if (!this._attachedCamera) {\r\n return false;\r\n }\r\n\r\n if (this._attachedCamera.radius === radiusLimit && !this._radiusIsAnimating) {\r\n return true;\r\n }\r\n return false;\r\n }\r\n\r\n /**\r\n * Applies an animation to the radius of the camera, extending by the radiusDelta.\r\n * @param radiusDelta The delta by which to animate to. Can be negative.\r\n */\r\n private _applyBoundRadiusAnimation(radiusDelta: number): void {\r\n if (!this._attachedCamera) {\r\n return;\r\n }\r\n\r\n if (!this._radiusBounceTransition) {\r\n BouncingBehavior.EasingFunction.setEasingMode(BouncingBehavior.EasingMode);\r\n this._radiusBounceTransition = Animation.CreateAnimation(\"radius\", Animation.ANIMATIONTYPE_FLOAT, 60, BouncingBehavior.EasingFunction);\r\n }\r\n // Prevent zoom until bounce has completed\r\n this._cachedWheelPrecision = this._attachedCamera.wheelPrecision;\r\n this._attachedCamera.wheelPrecision = Infinity;\r\n this._attachedCamera.inertialRadiusOffset = 0;\r\n\r\n // Animate to the radius limit\r\n this.stopAllAnimations();\r\n this._radiusIsAnimating = true;\r\n const animatable = Animation.TransitionTo(\r\n \"radius\",\r\n this._attachedCamera.radius + radiusDelta,\r\n this._attachedCamera,\r\n this._attachedCamera.getScene(),\r\n 60,\r\n this._radiusBounceTransition,\r\n this.transitionDuration,\r\n () => this._clearAnimationLocks()\r\n );\r\n\r\n if (animatable) {\r\n this._animatables.push(animatable);\r\n }\r\n }\r\n\r\n /**\r\n * Removes all animation locks. Allows new animations to be added to any of the camera properties.\r\n */\r\n protected _clearAnimationLocks(): void {\r\n this._radiusIsAnimating = false;\r\n\r\n if (this._attachedCamera) {\r\n this._attachedCamera.wheelPrecision = this._cachedWheelPrecision;\r\n }\r\n }\r\n\r\n /**\r\n * Stops and removes all animations that have been applied to the camera\r\n */\r\n public stopAllAnimations(): void {\r\n if (this._attachedCamera) {\r\n this._attachedCamera.animations = [];\r\n }\r\n while (this._animatables.length) {\r\n this._animatables[0].onAnimationEnd = null;\r\n this._animatables[0].stop();\r\n this._animatables.shift();\r\n }\r\n }\r\n}\r\n","import type { Behavior } from \"../../Behaviors/behavior\";\r\nimport type { Camera } from \"../../Cameras/camera\";\r\nimport type { ArcRotateCamera } from \"../../Cameras/arcRotateCamera\";\r\nimport { ExponentialEase, EasingFunction } from \"../../Animations/easing\";\r\nimport type { Observer } from \"../../Misc/observable\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { PointerInfoPre } from \"../../Events/pointerEvents\";\r\nimport { PointerEventTypes } from \"../../Events/pointerEvents\";\r\nimport { PrecisionDate } from \"../../Misc/precisionDate\";\r\n\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport { Vector3, Vector2 } from \"../../Maths/math.vector\";\r\nimport type { Animatable } from \"../../Animations/animatable\";\r\nimport { Animation } from \"../../Animations/animation\";\r\n\r\n/**\r\n * The framing behavior (FramingBehavior) is designed to automatically position an ArcRotateCamera when its target is set to a mesh. It is also useful if you want to prevent the camera to go under a virtual horizontal plane.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#framing-behavior\r\n */\r\nexport class FramingBehavior implements Behavior {\r\n /**\r\n * Gets the name of the behavior.\r\n */\r\n public get name(): string {\r\n return \"Framing\";\r\n }\r\n\r\n /**\r\n * An event triggered when the animation to zoom on target mesh has ended\r\n */\r\n public onTargetFramingAnimationEndObservable = new Observable();\r\n\r\n private _mode = FramingBehavior.FitFrustumSidesMode;\r\n private _radiusScale = 1.0;\r\n private _positionScale = 0.5;\r\n private _defaultElevation = 0.3;\r\n private _elevationReturnTime = 1500;\r\n private _elevationReturnWaitTime = 1000;\r\n private _zoomStopsAnimation = false;\r\n private _framingTime = 1500;\r\n\r\n /**\r\n * The easing function used by animations\r\n */\r\n public static EasingFunction = new ExponentialEase();\r\n\r\n /**\r\n * The easing mode used by animations\r\n */\r\n public static EasingMode = EasingFunction.EASINGMODE_EASEINOUT;\r\n\r\n /**\r\n * Sets the current mode used by the behavior\r\n */\r\n public set mode(mode: number) {\r\n this._mode = mode;\r\n }\r\n\r\n /**\r\n * Gets current mode used by the behavior.\r\n */\r\n public get mode(): number {\r\n return this._mode;\r\n }\r\n\r\n /**\r\n * Sets the scale applied to the radius (1 by default)\r\n */\r\n public set radiusScale(radius: number) {\r\n this._radiusScale = radius;\r\n }\r\n\r\n /**\r\n * Gets the scale applied to the radius\r\n */\r\n public get radiusScale(): number {\r\n return this._radiusScale;\r\n }\r\n\r\n /**\r\n * Sets the scale to apply on Y axis to position camera focus. 0.5 by default which means the center of the bounding box.\r\n */\r\n public set positionScale(scale: number) {\r\n this._positionScale = scale;\r\n }\r\n\r\n /**\r\n * Gets the scale to apply on Y axis to position camera focus. 0.5 by default which means the center of the bounding box.\r\n */\r\n public get positionScale(): number {\r\n return this._positionScale;\r\n }\r\n\r\n /**\r\n * Sets the angle above/below the horizontal plane to return to when the return to default elevation idle\r\n * behaviour is triggered, in radians.\r\n */\r\n public set defaultElevation(elevation: number) {\r\n this._defaultElevation = elevation;\r\n }\r\n\r\n /**\r\n * Gets the angle above/below the horizontal plane to return to when the return to default elevation idle\r\n * behaviour is triggered, in radians.\r\n */\r\n public get defaultElevation() {\r\n return this._defaultElevation;\r\n }\r\n\r\n /**\r\n * Sets the time (in milliseconds) taken to return to the default beta position.\r\n * Negative value indicates camera should not return to default.\r\n */\r\n public set elevationReturnTime(speed: number) {\r\n this._elevationReturnTime = speed;\r\n }\r\n\r\n /**\r\n * Gets the time (in milliseconds) taken to return to the default beta position.\r\n * Negative value indicates camera should not return to default.\r\n */\r\n public get elevationReturnTime(): number {\r\n return this._elevationReturnTime;\r\n }\r\n\r\n /**\r\n * Sets the delay (in milliseconds) taken before the camera returns to the default beta position.\r\n */\r\n public set elevationReturnWaitTime(time: number) {\r\n this._elevationReturnWaitTime = time;\r\n }\r\n\r\n /**\r\n * Gets the delay (in milliseconds) taken before the camera returns to the default beta position.\r\n */\r\n public get elevationReturnWaitTime(): number {\r\n return this._elevationReturnWaitTime;\r\n }\r\n\r\n /**\r\n * Sets the flag that indicates if user zooming should stop animation.\r\n */\r\n public set zoomStopsAnimation(flag: boolean) {\r\n this._zoomStopsAnimation = flag;\r\n }\r\n\r\n /**\r\n * Gets the flag that indicates if user zooming should stop animation.\r\n */\r\n public get zoomStopsAnimation(): boolean {\r\n return this._zoomStopsAnimation;\r\n }\r\n\r\n /**\r\n * Sets the transition time when framing the mesh, in milliseconds\r\n */\r\n public set framingTime(time: number) {\r\n this._framingTime = time;\r\n }\r\n\r\n /**\r\n * Gets the transition time when framing the mesh, in milliseconds\r\n */\r\n public get framingTime() {\r\n return this._framingTime;\r\n }\r\n\r\n /**\r\n * Define if the behavior should automatically change the configured\r\n * camera limits and sensibilities.\r\n */\r\n public autoCorrectCameraLimitsAndSensibility = true;\r\n\r\n // Default behavior functions\r\n private _onPrePointerObservableObserver: Nullable>;\r\n private _onAfterCheckInputsObserver: Nullable>;\r\n private _onMeshTargetChangedObserver: Nullable>>;\r\n private _attachedCamera: Nullable;\r\n private _isPointerDown = false;\r\n private _lastInteractionTime = -Infinity;\r\n\r\n /**\r\n * Initializes the behavior.\r\n */\r\n public init(): void {\r\n // Do nothing\r\n }\r\n\r\n /**\r\n * Attaches the behavior to its arc rotate camera.\r\n * @param camera Defines the camera to attach the behavior to\r\n */\r\n public attach(camera: ArcRotateCamera): void {\r\n this._attachedCamera = camera;\r\n const scene = this._attachedCamera.getScene();\r\n\r\n FramingBehavior.EasingFunction.setEasingMode(FramingBehavior.EasingMode);\r\n\r\n this._onPrePointerObservableObserver = scene.onPrePointerObservable.add((pointerInfoPre) => {\r\n if (pointerInfoPre.type === PointerEventTypes.POINTERDOWN) {\r\n this._isPointerDown = true;\r\n return;\r\n }\r\n\r\n if (pointerInfoPre.type === PointerEventTypes.POINTERUP) {\r\n this._isPointerDown = false;\r\n }\r\n });\r\n\r\n this._onMeshTargetChangedObserver = camera.onMeshTargetChangedObservable.add((mesh) => {\r\n if (mesh) {\r\n this.zoomOnMesh(mesh, undefined, () => {\r\n this.onTargetFramingAnimationEndObservable.notifyObservers();\r\n });\r\n }\r\n });\r\n\r\n this._onAfterCheckInputsObserver = camera.onAfterCheckInputsObservable.add(() => {\r\n // Stop the animation if there is user interaction and the animation should stop for this interaction\r\n this._applyUserInteraction();\r\n\r\n // Maintain the camera above the ground. If the user pulls the camera beneath the ground plane, lift it\r\n // back to the default position after a given timeout\r\n this._maintainCameraAboveGround();\r\n });\r\n }\r\n\r\n /**\r\n * Detaches the behavior from its current arc rotate camera.\r\n */\r\n public detach(): void {\r\n if (!this._attachedCamera) {\r\n return;\r\n }\r\n\r\n const scene = this._attachedCamera.getScene();\r\n\r\n if (this._onPrePointerObservableObserver) {\r\n scene.onPrePointerObservable.remove(this._onPrePointerObservableObserver);\r\n }\r\n\r\n if (this._onAfterCheckInputsObserver) {\r\n this._attachedCamera.onAfterCheckInputsObservable.remove(this._onAfterCheckInputsObserver);\r\n }\r\n\r\n if (this._onMeshTargetChangedObserver) {\r\n this._attachedCamera.onMeshTargetChangedObservable.remove(this._onMeshTargetChangedObserver);\r\n }\r\n\r\n this._attachedCamera = null;\r\n }\r\n\r\n // Framing control\r\n private _animatables = new Array();\r\n private _betaIsAnimating = false;\r\n private _betaTransition: Animation;\r\n private _radiusTransition: Animation;\r\n private _vectorTransition: Animation;\r\n\r\n /**\r\n * Targets the given mesh and updates zoom level accordingly.\r\n * @param mesh The mesh to target.\r\n * @param focusOnOriginXZ Determines if the camera should focus on 0 in the X and Z axis instead of the mesh\r\n * @param onAnimationEnd Callback triggered at the end of the framing animation\r\n */\r\n public zoomOnMesh(mesh: AbstractMesh, focusOnOriginXZ: boolean = false, onAnimationEnd: Nullable<() => void> = null): void {\r\n mesh.computeWorldMatrix(true);\r\n\r\n const boundingBox = mesh.getBoundingInfo().boundingBox;\r\n this.zoomOnBoundingInfo(boundingBox.minimumWorld, boundingBox.maximumWorld, focusOnOriginXZ, onAnimationEnd);\r\n }\r\n\r\n /**\r\n * Targets the given mesh with its children and updates zoom level accordingly.\r\n * @param mesh The mesh to target.\r\n * @param focusOnOriginXZ Determines if the camera should focus on 0 in the X and Z axis instead of the mesh\r\n * @param onAnimationEnd Callback triggered at the end of the framing animation\r\n */\r\n public zoomOnMeshHierarchy(mesh: AbstractMesh, focusOnOriginXZ: boolean = false, onAnimationEnd: Nullable<() => void> = null): void {\r\n mesh.computeWorldMatrix(true);\r\n\r\n const boundingBox = mesh.getHierarchyBoundingVectors(true);\r\n this.zoomOnBoundingInfo(boundingBox.min, boundingBox.max, focusOnOriginXZ, onAnimationEnd);\r\n }\r\n\r\n /**\r\n * Targets the given meshes with their children and updates zoom level accordingly.\r\n * @param meshes The mesh to target.\r\n * @param focusOnOriginXZ Determines if the camera should focus on 0 in the X and Z axis instead of the mesh\r\n * @param onAnimationEnd Callback triggered at the end of the framing animation\r\n */\r\n public zoomOnMeshesHierarchy(meshes: AbstractMesh[], focusOnOriginXZ: boolean = false, onAnimationEnd: Nullable<() => void> = null): void {\r\n const min = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\r\n const max = new Vector3(-Number.MAX_VALUE, -Number.MAX_VALUE, -Number.MAX_VALUE);\r\n\r\n for (let i = 0; i < meshes.length; i++) {\r\n const boundingInfo = meshes[i].getHierarchyBoundingVectors(true);\r\n Vector3.CheckExtends(boundingInfo.min, min, max);\r\n Vector3.CheckExtends(boundingInfo.max, min, max);\r\n }\r\n\r\n this.zoomOnBoundingInfo(min, max, focusOnOriginXZ, onAnimationEnd);\r\n }\r\n\r\n /**\r\n * Targets the bounding box info defined by its extends and updates zoom level accordingly.\r\n * @param minimumWorld Determines the smaller position of the bounding box extend\r\n * @param maximumWorld Determines the bigger position of the bounding box extend\r\n * @param focusOnOriginXZ Determines if the camera should focus on 0 in the X and Z axis instead of the mesh\r\n * @param onAnimationEnd Callback triggered at the end of the framing animation\r\n * @returns true if the zoom was done\r\n */\r\n public zoomOnBoundingInfo(minimumWorld: Vector3, maximumWorld: Vector3, focusOnOriginXZ: boolean = false, onAnimationEnd: Nullable<() => void> = null): boolean {\r\n let zoomTarget: Vector3;\r\n\r\n if (!this._attachedCamera) {\r\n return false;\r\n }\r\n\r\n // Find target by interpolating from bottom of bounding box in world-space to top via framingPositionY\r\n const bottom = minimumWorld.y;\r\n const top = maximumWorld.y;\r\n const zoomTargetY = bottom + (top - bottom) * this._positionScale;\r\n const radiusWorld = maximumWorld.subtract(minimumWorld).scale(0.5);\r\n\r\n if (focusOnOriginXZ) {\r\n zoomTarget = new Vector3(0, zoomTargetY, 0);\r\n } else {\r\n const centerWorld = minimumWorld.add(radiusWorld);\r\n zoomTarget = new Vector3(centerWorld.x, zoomTargetY, centerWorld.z);\r\n }\r\n\r\n if (!this._vectorTransition) {\r\n this._vectorTransition = Animation.CreateAnimation(\"target\", Animation.ANIMATIONTYPE_VECTOR3, 60, FramingBehavior.EasingFunction);\r\n }\r\n\r\n this._betaIsAnimating = true;\r\n let animatable = Animation.TransitionTo(\"target\", zoomTarget, this._attachedCamera, this._attachedCamera.getScene(), 60, this._vectorTransition, this._framingTime);\r\n if (animatable) {\r\n this._animatables.push(animatable);\r\n }\r\n\r\n // sets the radius and lower radius bounds\r\n // Small delta ensures camera is not always at lower zoom limit.\r\n let radius = 0;\r\n if (this._mode === FramingBehavior.FitFrustumSidesMode) {\r\n const position = this._calculateLowerRadiusFromModelBoundingSphere(minimumWorld, maximumWorld);\r\n if (this.autoCorrectCameraLimitsAndSensibility) {\r\n this._attachedCamera.lowerRadiusLimit = radiusWorld.length() + this._attachedCamera.minZ;\r\n }\r\n radius = position;\r\n } else if (this._mode === FramingBehavior.IgnoreBoundsSizeMode) {\r\n radius = this._calculateLowerRadiusFromModelBoundingSphere(minimumWorld, maximumWorld);\r\n if (this.autoCorrectCameraLimitsAndSensibility && this._attachedCamera.lowerRadiusLimit === null) {\r\n this._attachedCamera.lowerRadiusLimit = this._attachedCamera.minZ;\r\n }\r\n }\r\n\r\n // Set sensibilities\r\n if (this.autoCorrectCameraLimitsAndSensibility) {\r\n const extend = maximumWorld.subtract(minimumWorld).length();\r\n this._attachedCamera.panningSensibility = 5000 / extend;\r\n this._attachedCamera.wheelPrecision = 100 / radius;\r\n }\r\n\r\n // transition to new radius\r\n if (!this._radiusTransition) {\r\n this._radiusTransition = Animation.CreateAnimation(\"radius\", Animation.ANIMATIONTYPE_FLOAT, 60, FramingBehavior.EasingFunction);\r\n }\r\n\r\n animatable = Animation.TransitionTo(\"radius\", radius, this._attachedCamera, this._attachedCamera.getScene(), 60, this._radiusTransition, this._framingTime, () => {\r\n this.stopAllAnimations();\r\n if (onAnimationEnd) {\r\n onAnimationEnd();\r\n }\r\n\r\n if (this._attachedCamera && this._attachedCamera.useInputToRestoreState) {\r\n this._attachedCamera.storeState();\r\n }\r\n });\r\n\r\n if (animatable) {\r\n this._animatables.push(animatable);\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Calculates the lowest radius for the camera based on the bounding box of the mesh.\r\n * @param minimumWorld\r\n * @param maximumWorld\r\n * @returns The minimum distance from the primary mesh's center point at which the camera must be kept in order\r\n *\t\t to fully enclose the mesh in the viewing frustum.\r\n */\r\n protected _calculateLowerRadiusFromModelBoundingSphere(minimumWorld: Vector3, maximumWorld: Vector3): number {\r\n const size = maximumWorld.subtract(minimumWorld);\r\n const boxVectorGlobalDiagonal = size.length();\r\n const frustumSlope: Vector2 = this._getFrustumSlope();\r\n\r\n // Formula for setting distance\r\n // (Good explanation: http://stackoverflow.com/questions/2866350/move-camera-to-fit-3d-scene)\r\n const radiusWithoutFraming = boxVectorGlobalDiagonal * 0.5;\r\n\r\n // Horizon distance\r\n const radius = radiusWithoutFraming * this._radiusScale;\r\n const distanceForHorizontalFrustum = radius * Math.sqrt(1.0 + 1.0 / (frustumSlope.x * frustumSlope.x));\r\n const distanceForVerticalFrustum = radius * Math.sqrt(1.0 + 1.0 / (frustumSlope.y * frustumSlope.y));\r\n let distance = Math.max(distanceForHorizontalFrustum, distanceForVerticalFrustum);\r\n const camera = this._attachedCamera;\r\n\r\n if (!camera) {\r\n return 0;\r\n }\r\n\r\n if (camera.lowerRadiusLimit && this._mode === FramingBehavior.IgnoreBoundsSizeMode) {\r\n // Don't exceed the requested limit\r\n distance = distance < camera.lowerRadiusLimit ? camera.lowerRadiusLimit : distance;\r\n }\r\n\r\n // Don't exceed the upper radius limit\r\n if (camera.upperRadiusLimit) {\r\n distance = distance > camera.upperRadiusLimit ? camera.upperRadiusLimit : distance;\r\n }\r\n\r\n return distance;\r\n }\r\n\r\n /**\r\n * Keeps the camera above the ground plane. If the user pulls the camera below the ground plane, the camera\r\n * is automatically returned to its default position (expected to be above ground plane).\r\n */\r\n private _maintainCameraAboveGround(): void {\r\n if (this._elevationReturnTime < 0) {\r\n return;\r\n }\r\n\r\n const timeSinceInteraction = PrecisionDate.Now - this._lastInteractionTime;\r\n const defaultBeta = Math.PI * 0.5 - this._defaultElevation;\r\n const limitBeta = Math.PI * 0.5;\r\n\r\n // Bring the camera back up if below the ground plane\r\n if (this._attachedCamera && !this._betaIsAnimating && this._attachedCamera.beta > limitBeta && timeSinceInteraction >= this._elevationReturnWaitTime) {\r\n this._betaIsAnimating = true;\r\n\r\n //Transition to new position\r\n this.stopAllAnimations();\r\n\r\n if (!this._betaTransition) {\r\n this._betaTransition = Animation.CreateAnimation(\"beta\", Animation.ANIMATIONTYPE_FLOAT, 60, FramingBehavior.EasingFunction);\r\n }\r\n\r\n const animatabe = Animation.TransitionTo(\r\n \"beta\",\r\n defaultBeta,\r\n this._attachedCamera,\r\n this._attachedCamera.getScene(),\r\n 60,\r\n this._betaTransition,\r\n this._elevationReturnTime,\r\n () => {\r\n this._clearAnimationLocks();\r\n this.stopAllAnimations();\r\n }\r\n );\r\n\r\n if (animatabe) {\r\n this._animatables.push(animatabe);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Returns the frustum slope based on the canvas ratio and camera FOV\r\n * @returns The frustum slope represented as a Vector2 with X and Y slopes\r\n */\r\n private _getFrustumSlope(): Vector2 {\r\n // Calculate the viewport ratio\r\n // Aspect Ratio is Height/Width.\r\n const camera = this._attachedCamera;\r\n\r\n if (!camera) {\r\n return Vector2.Zero();\r\n }\r\n\r\n const engine = camera.getScene().getEngine();\r\n const aspectRatio = engine.getAspectRatio(camera);\r\n\r\n // Camera FOV is the vertical field of view (top-bottom) in radians.\r\n // Slope of the frustum top/bottom planes in view space, relative to the forward vector.\r\n const frustumSlopeY = Math.tan(camera.fov / 2);\r\n\r\n // Slope of the frustum left/right planes in view space, relative to the forward vector.\r\n // Provides the amount that one side (e.g. left) of the frustum gets wider for every unit\r\n // along the forward vector.\r\n const frustumSlopeX = frustumSlopeY * aspectRatio;\r\n\r\n return new Vector2(frustumSlopeX, frustumSlopeY);\r\n }\r\n\r\n /**\r\n * Removes all animation locks. Allows new animations to be added to any of the arcCamera properties.\r\n */\r\n private _clearAnimationLocks(): void {\r\n this._betaIsAnimating = false;\r\n }\r\n\r\n /**\r\n * Applies any current user interaction to the camera. Takes into account maximum alpha rotation.\r\n */\r\n private _applyUserInteraction(): void {\r\n if (this.isUserIsMoving) {\r\n this._lastInteractionTime = PrecisionDate.Now;\r\n this.stopAllAnimations();\r\n this._clearAnimationLocks();\r\n }\r\n }\r\n\r\n /**\r\n * Stops and removes all animations that have been applied to the camera\r\n */\r\n public stopAllAnimations(): void {\r\n if (this._attachedCamera) {\r\n this._attachedCamera.animations = [];\r\n }\r\n\r\n while (this._animatables.length) {\r\n if (this._animatables[0]) {\r\n this._animatables[0].onAnimationEnd = null;\r\n this._animatables[0].stop();\r\n }\r\n this._animatables.shift();\r\n }\r\n }\r\n\r\n /**\r\n * Gets a value indicating if the user is moving the camera\r\n */\r\n public get isUserIsMoving(): boolean {\r\n if (!this._attachedCamera) {\r\n return false;\r\n }\r\n\r\n return (\r\n this._attachedCamera.inertialAlphaOffset !== 0 ||\r\n this._attachedCamera.inertialBetaOffset !== 0 ||\r\n this._attachedCamera.inertialRadiusOffset !== 0 ||\r\n this._attachedCamera.inertialPanningX !== 0 ||\r\n this._attachedCamera.inertialPanningY !== 0 ||\r\n this._isPointerDown\r\n );\r\n }\r\n\r\n // Statics\r\n\r\n /**\r\n * The camera can move all the way towards the mesh.\r\n */\r\n public static IgnoreBoundsSizeMode = 0;\r\n\r\n /**\r\n * The camera is not allowed to zoom closer to the mesh than the point at which the adjusted bounding sphere touches the frustum sides\r\n */\r\n public static FitFrustumSidesMode = 1;\r\n}\r\n","import { serialize, serializeAsVector3, serializeAsMeshReference } from \"../Misc/decorators\";\r\nimport type { Nullable } from \"../types\";\r\nimport { Camera } from \"./camera\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Quaternion, Matrix, Vector3, Vector2, TmpVectors } from \"../Maths/math.vector\";\r\nimport { Epsilon } from \"../Maths/math.constants\";\r\nimport { Axis } from \"../Maths/math.axis\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\n/**\r\n * A target camera takes a mesh or position as a target and continues to look at it while it moves.\r\n * This is the base of the follow, arc rotate cameras and Free camera\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras\r\n */\r\nexport class TargetCamera extends Camera {\r\n private static _RigCamTransformMatrix = new Matrix();\r\n private static _TargetTransformMatrix = new Matrix();\r\n private static _TargetFocalPoint = new Vector3();\r\n\r\n private _tmpUpVector = Vector3.Zero();\r\n private _tmpTargetVector = Vector3.Zero();\r\n\r\n /**\r\n * Define the current direction the camera is moving to\r\n */\r\n public cameraDirection = new Vector3(0, 0, 0);\r\n /**\r\n * Define the current rotation the camera is rotating to\r\n */\r\n public cameraRotation = new Vector2(0, 0);\r\n\r\n /** Gets or sets a boolean indicating that the scaling of the parent hierarchy will not be taken in account by the camera */\r\n public ignoreParentScaling = false;\r\n\r\n /**\r\n * When set, the up vector of the camera will be updated by the rotation of the camera\r\n */\r\n public updateUpVectorFromRotation = false;\r\n private _tmpQuaternion = new Quaternion();\r\n\r\n /**\r\n * Define the current rotation of the camera\r\n */\r\n @serializeAsVector3()\r\n public rotation = new Vector3(0, 0, 0);\r\n\r\n /**\r\n * Define the current rotation of the camera as a quaternion to prevent Gimbal lock\r\n */\r\n public rotationQuaternion: Quaternion;\r\n\r\n /**\r\n * Define the current speed of the camera\r\n */\r\n @serialize()\r\n public speed = 2.0;\r\n\r\n /**\r\n * Add constraint to the camera to prevent it to move freely in all directions and\r\n * around all axis.\r\n */\r\n public noRotationConstraint = false;\r\n\r\n /**\r\n * Reverses mouselook direction to 'natural' panning as opposed to traditional direct\r\n * panning\r\n */\r\n public invertRotation = false;\r\n\r\n /**\r\n * Speed multiplier for inverse camera panning\r\n */\r\n public inverseRotationSpeed = 0.2;\r\n\r\n /**\r\n * Define the current target of the camera as an object or a position.\r\n * Please note that locking a target will disable panning.\r\n */\r\n @serializeAsMeshReference(\"lockedTargetId\")\r\n public lockedTarget: any = null;\r\n\r\n /** @internal */\r\n public _currentTarget = Vector3.Zero();\r\n /** @internal */\r\n public _initialFocalDistance = 1;\r\n /** @internal */\r\n public _viewMatrix = Matrix.Zero();\r\n /** @internal */\r\n public _camMatrix = Matrix.Zero();\r\n /** @internal */\r\n public _cameraTransformMatrix = Matrix.Zero();\r\n /** @internal */\r\n public _cameraRotationMatrix = Matrix.Zero();\r\n\r\n /** @internal */\r\n public _referencePoint = new Vector3(0, 0, 1);\r\n /** @internal */\r\n public _transformedReferencePoint = Vector3.Zero();\r\n\r\n protected _deferredPositionUpdate = new Vector3();\r\n protected _deferredRotationQuaternionUpdate = new Quaternion();\r\n protected _deferredRotationUpdate = new Vector3();\r\n protected _deferredUpdated = false;\r\n protected _deferOnly: boolean = false;\r\n\r\n /** @internal */\r\n public _reset: () => void;\r\n\r\n private _defaultUp = Vector3.Up();\r\n\r\n /**\r\n * Instantiates a target camera that takes a mesh or position as a target and continues to look at it while it moves.\r\n * This is the base of the follow, arc rotate cameras and Free camera\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras\r\n * @param name Defines the name of the camera in the scene\r\n * @param position Defines the start position of the camera in the scene\r\n * @param scene Defines the scene the camera belongs to\r\n * @param setActiveOnSceneIfNoneActive Defines whether the camera should be marked as active if not other active cameras have been defined\r\n */\r\n constructor(name: string, position: Vector3, scene?: Scene, setActiveOnSceneIfNoneActive = true) {\r\n super(name, position, scene, setActiveOnSceneIfNoneActive);\r\n }\r\n\r\n /**\r\n * Gets the position in front of the camera at a given distance.\r\n * @param distance The distance from the camera we want the position to be\r\n * @returns the position\r\n */\r\n public getFrontPosition(distance: number): Vector3 {\r\n this.getWorldMatrix();\r\n const direction = this.getTarget().subtract(this.position);\r\n direction.normalize();\r\n direction.scaleInPlace(distance);\r\n return this.globalPosition.add(direction);\r\n }\r\n\r\n /** @internal */\r\n public _getLockedTargetPosition(): Nullable {\r\n if (!this.lockedTarget) {\r\n return null;\r\n }\r\n\r\n if (this.lockedTarget.absolutePosition) {\r\n const lockedTarget = this.lockedTarget as AbstractMesh;\r\n const m = lockedTarget.computeWorldMatrix();\r\n // in some cases the absolute position resets externally, but doesn't update since the matrix is cached.\r\n m.getTranslationToRef(lockedTarget.absolutePosition);\r\n }\r\n\r\n return this.lockedTarget.absolutePosition || this.lockedTarget;\r\n }\r\n\r\n private _storedPosition: Vector3;\r\n private _storedRotation: Vector3;\r\n private _storedRotationQuaternion: Quaternion;\r\n\r\n /**\r\n * Store current camera state of the camera (fov, position, rotation, etc..)\r\n * @returns the camera\r\n */\r\n public storeState(): Camera {\r\n this._storedPosition = this.position.clone();\r\n this._storedRotation = this.rotation.clone();\r\n if (this.rotationQuaternion) {\r\n this._storedRotationQuaternion = this.rotationQuaternion.clone();\r\n }\r\n\r\n return super.storeState();\r\n }\r\n\r\n /**\r\n * Restored camera state. You must call storeState() first\r\n * @returns whether it was successful or not\r\n * @internal\r\n */\r\n public _restoreStateValues(): boolean {\r\n if (!super._restoreStateValues()) {\r\n return false;\r\n }\r\n\r\n this.position = this._storedPosition.clone();\r\n this.rotation = this._storedRotation.clone();\r\n\r\n if (this.rotationQuaternion) {\r\n this.rotationQuaternion = this._storedRotationQuaternion.clone();\r\n }\r\n\r\n this.cameraDirection.copyFromFloats(0, 0, 0);\r\n this.cameraRotation.copyFromFloats(0, 0);\r\n\r\n return true;\r\n }\r\n\r\n /** @internal */\r\n public _initCache() {\r\n super._initCache();\r\n this._cache.lockedTarget = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\r\n this._cache.rotation = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\r\n this._cache.rotationQuaternion = new Quaternion(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _updateCache(ignoreParentClass?: boolean): void {\r\n if (!ignoreParentClass) {\r\n super._updateCache();\r\n }\r\n\r\n const lockedTargetPosition = this._getLockedTargetPosition();\r\n if (!lockedTargetPosition) {\r\n this._cache.lockedTarget = null;\r\n } else {\r\n if (!this._cache.lockedTarget) {\r\n this._cache.lockedTarget = lockedTargetPosition.clone();\r\n } else {\r\n this._cache.lockedTarget.copyFrom(lockedTargetPosition);\r\n }\r\n }\r\n\r\n this._cache.rotation.copyFrom(this.rotation);\r\n if (this.rotationQuaternion) {\r\n this._cache.rotationQuaternion.copyFrom(this.rotationQuaternion);\r\n }\r\n }\r\n\r\n // Synchronized\r\n /** @internal */\r\n public _isSynchronizedViewMatrix(): boolean {\r\n if (!super._isSynchronizedViewMatrix()) {\r\n return false;\r\n }\r\n\r\n const lockedTargetPosition = this._getLockedTargetPosition();\r\n\r\n return (\r\n (this._cache.lockedTarget ? this._cache.lockedTarget.equals(lockedTargetPosition) : !lockedTargetPosition) &&\r\n (this.rotationQuaternion ? this.rotationQuaternion.equals(this._cache.rotationQuaternion) : this._cache.rotation.equals(this.rotation))\r\n );\r\n }\r\n\r\n // Methods\r\n /** @internal */\r\n public _computeLocalCameraSpeed(): number {\r\n const engine = this.getEngine();\r\n return this.speed * Math.sqrt(engine.getDeltaTime() / (engine.getFps() * 100.0));\r\n }\r\n\r\n // Target\r\n\r\n /**\r\n * Defines the target the camera should look at.\r\n * @param target Defines the new target as a Vector\r\n */\r\n public setTarget(target: Vector3): void {\r\n this.upVector.normalize();\r\n\r\n this._initialFocalDistance = target.subtract(this.position).length();\r\n\r\n if (this.position.z === target.z) {\r\n this.position.z += Epsilon;\r\n }\r\n\r\n this._referencePoint.normalize().scaleInPlace(this._initialFocalDistance);\r\n\r\n Matrix.LookAtLHToRef(this.position, target, this._defaultUp, this._camMatrix);\r\n this._camMatrix.invert();\r\n\r\n this.rotation.x = Math.atan(this._camMatrix.m[6] / this._camMatrix.m[10]);\r\n\r\n const vDir = target.subtract(this.position);\r\n\r\n if (vDir.x >= 0.0) {\r\n this.rotation.y = -Math.atan(vDir.z / vDir.x) + Math.PI / 2.0;\r\n } else {\r\n this.rotation.y = -Math.atan(vDir.z / vDir.x) - Math.PI / 2.0;\r\n }\r\n\r\n this.rotation.z = 0;\r\n\r\n if (isNaN(this.rotation.x)) {\r\n this.rotation.x = 0;\r\n }\r\n\r\n if (isNaN(this.rotation.y)) {\r\n this.rotation.y = 0;\r\n }\r\n\r\n if (isNaN(this.rotation.z)) {\r\n this.rotation.z = 0;\r\n }\r\n\r\n if (this.rotationQuaternion) {\r\n Quaternion.RotationYawPitchRollToRef(this.rotation.y, this.rotation.x, this.rotation.z, this.rotationQuaternion);\r\n }\r\n }\r\n\r\n /**\r\n * Defines the target point of the camera.\r\n * The camera looks towards it form the radius distance.\r\n */\r\n public get target(): Vector3 {\r\n return this.getTarget();\r\n }\r\n public set target(value: Vector3) {\r\n this.setTarget(value);\r\n }\r\n\r\n /**\r\n * Return the current target position of the camera. This value is expressed in local space.\r\n * @returns the target position\r\n */\r\n public getTarget(): Vector3 {\r\n return this._currentTarget;\r\n }\r\n\r\n /** @internal */\r\n public _decideIfNeedsToMove(): boolean {\r\n return Math.abs(this.cameraDirection.x) > 0 || Math.abs(this.cameraDirection.y) > 0 || Math.abs(this.cameraDirection.z) > 0;\r\n }\r\n\r\n /** @internal */\r\n public _updatePosition(): void {\r\n if (this.parent) {\r\n this.parent.getWorldMatrix().invertToRef(TmpVectors.Matrix[0]);\r\n Vector3.TransformNormalToRef(this.cameraDirection, TmpVectors.Matrix[0], TmpVectors.Vector3[0]);\r\n this._deferredPositionUpdate.addInPlace(TmpVectors.Vector3[0]);\r\n if (!this._deferOnly) {\r\n this.position.copyFrom(this._deferredPositionUpdate);\r\n } else {\r\n this._deferredUpdated = true;\r\n }\r\n return;\r\n }\r\n this._deferredPositionUpdate.addInPlace(this.cameraDirection);\r\n if (!this._deferOnly) {\r\n this.position.copyFrom(this._deferredPositionUpdate);\r\n } else {\r\n this._deferredUpdated = true;\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _checkInputs(): void {\r\n const directionMultiplier = this.invertRotation ? -this.inverseRotationSpeed : 1.0;\r\n const needToMove = this._decideIfNeedsToMove();\r\n const needToRotate = this.cameraRotation.x || this.cameraRotation.y;\r\n\r\n this._deferredUpdated = false;\r\n this._deferredRotationUpdate.copyFrom(this.rotation);\r\n this._deferredPositionUpdate.copyFrom(this.position);\r\n if (this.rotationQuaternion) {\r\n this._deferredRotationQuaternionUpdate.copyFrom(this.rotationQuaternion);\r\n }\r\n\r\n // Move\r\n if (needToMove) {\r\n this._updatePosition();\r\n }\r\n\r\n // Rotate\r\n if (needToRotate) {\r\n //rotate, if quaternion is set and rotation was used\r\n if (this.rotationQuaternion) {\r\n this.rotationQuaternion.toEulerAnglesToRef(this._deferredRotationUpdate);\r\n }\r\n\r\n this._deferredRotationUpdate.x += this.cameraRotation.x * directionMultiplier;\r\n this._deferredRotationUpdate.y += this.cameraRotation.y * directionMultiplier;\r\n\r\n // Apply constraints\r\n if (!this.noRotationConstraint) {\r\n const limit = 1.570796;\r\n\r\n if (this._deferredRotationUpdate.x > limit) {\r\n this._deferredRotationUpdate.x = limit;\r\n }\r\n if (this._deferredRotationUpdate.x < -limit) {\r\n this._deferredRotationUpdate.x = -limit;\r\n }\r\n }\r\n\r\n if (!this._deferOnly) {\r\n this.rotation.copyFrom(this._deferredRotationUpdate);\r\n } else {\r\n this._deferredUpdated = true;\r\n }\r\n\r\n //rotate, if quaternion is set and rotation was used\r\n if (this.rotationQuaternion) {\r\n const len = this._deferredRotationUpdate.lengthSquared();\r\n if (len) {\r\n Quaternion.RotationYawPitchRollToRef(\r\n this._deferredRotationUpdate.y,\r\n this._deferredRotationUpdate.x,\r\n this._deferredRotationUpdate.z,\r\n this._deferredRotationQuaternionUpdate\r\n );\r\n if (!this._deferOnly) {\r\n this.rotationQuaternion.copyFrom(this._deferredRotationQuaternionUpdate);\r\n } else {\r\n this._deferredUpdated = true;\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Inertia\r\n if (needToMove) {\r\n if (Math.abs(this.cameraDirection.x) < this.speed * Epsilon) {\r\n this.cameraDirection.x = 0;\r\n }\r\n\r\n if (Math.abs(this.cameraDirection.y) < this.speed * Epsilon) {\r\n this.cameraDirection.y = 0;\r\n }\r\n\r\n if (Math.abs(this.cameraDirection.z) < this.speed * Epsilon) {\r\n this.cameraDirection.z = 0;\r\n }\r\n\r\n this.cameraDirection.scaleInPlace(this.inertia);\r\n }\r\n if (needToRotate) {\r\n if (Math.abs(this.cameraRotation.x) < this.speed * Epsilon) {\r\n this.cameraRotation.x = 0;\r\n }\r\n\r\n if (Math.abs(this.cameraRotation.y) < this.speed * Epsilon) {\r\n this.cameraRotation.y = 0;\r\n }\r\n this.cameraRotation.scaleInPlace(this.inertia);\r\n }\r\n\r\n super._checkInputs();\r\n }\r\n\r\n protected _updateCameraRotationMatrix() {\r\n if (this.rotationQuaternion) {\r\n this.rotationQuaternion.toRotationMatrix(this._cameraRotationMatrix);\r\n } else {\r\n Matrix.RotationYawPitchRollToRef(this.rotation.y, this.rotation.x, this.rotation.z, this._cameraRotationMatrix);\r\n }\r\n }\r\n\r\n /**\r\n * Update the up vector to apply the rotation of the camera (So if you changed the camera rotation.z this will let you update the up vector as well)\r\n * @returns the current camera\r\n */\r\n private _rotateUpVectorWithCameraRotationMatrix(): TargetCamera {\r\n Vector3.TransformNormalToRef(this._defaultUp, this._cameraRotationMatrix, this.upVector);\r\n return this;\r\n }\r\n\r\n private _cachedRotationZ = 0;\r\n private _cachedQuaternionRotationZ = 0;\r\n /** @internal */\r\n public _getViewMatrix(): Matrix {\r\n if (this.lockedTarget) {\r\n this.setTarget(this._getLockedTargetPosition()!);\r\n }\r\n\r\n // Compute\r\n this._updateCameraRotationMatrix();\r\n\r\n // Apply the changed rotation to the upVector\r\n if (this.rotationQuaternion && this._cachedQuaternionRotationZ != this.rotationQuaternion.z) {\r\n this._rotateUpVectorWithCameraRotationMatrix();\r\n this._cachedQuaternionRotationZ = this.rotationQuaternion.z;\r\n } else if (this._cachedRotationZ !== this.rotation.z) {\r\n this._rotateUpVectorWithCameraRotationMatrix();\r\n this._cachedRotationZ = this.rotation.z;\r\n }\r\n\r\n Vector3.TransformCoordinatesToRef(this._referencePoint, this._cameraRotationMatrix, this._transformedReferencePoint);\r\n\r\n // Computing target and final matrix\r\n this.position.addToRef(this._transformedReferencePoint, this._currentTarget);\r\n if (this.updateUpVectorFromRotation) {\r\n if (this.rotationQuaternion) {\r\n Axis.Y.rotateByQuaternionToRef(this.rotationQuaternion, this.upVector);\r\n } else {\r\n Quaternion.FromEulerVectorToRef(this.rotation, this._tmpQuaternion);\r\n Axis.Y.rotateByQuaternionToRef(this._tmpQuaternion, this.upVector);\r\n }\r\n }\r\n this._computeViewMatrix(this.position, this._currentTarget, this.upVector);\r\n return this._viewMatrix;\r\n }\r\n\r\n protected _computeViewMatrix(position: Vector3, target: Vector3, up: Vector3): void {\r\n if (this.ignoreParentScaling) {\r\n if (this.parent) {\r\n const parentWorldMatrix = this.parent.getWorldMatrix();\r\n Vector3.TransformCoordinatesToRef(position, parentWorldMatrix, this._globalPosition);\r\n Vector3.TransformCoordinatesToRef(target, parentWorldMatrix, this._tmpTargetVector);\r\n Vector3.TransformNormalToRef(up, parentWorldMatrix, this._tmpUpVector);\r\n this._markSyncedWithParent();\r\n } else {\r\n this._globalPosition.copyFrom(position);\r\n this._tmpTargetVector.copyFrom(target);\r\n this._tmpUpVector.copyFrom(up);\r\n }\r\n\r\n if (this.getScene().useRightHandedSystem) {\r\n Matrix.LookAtRHToRef(this._globalPosition, this._tmpTargetVector, this._tmpUpVector, this._viewMatrix);\r\n } else {\r\n Matrix.LookAtLHToRef(this._globalPosition, this._tmpTargetVector, this._tmpUpVector, this._viewMatrix);\r\n }\r\n return;\r\n }\r\n\r\n if (this.getScene().useRightHandedSystem) {\r\n Matrix.LookAtRHToRef(position, target, up, this._viewMatrix);\r\n } else {\r\n Matrix.LookAtLHToRef(position, target, up, this._viewMatrix);\r\n }\r\n\r\n if (this.parent) {\r\n const parentWorldMatrix = this.parent.getWorldMatrix();\r\n this._viewMatrix.invert();\r\n this._viewMatrix.multiplyToRef(parentWorldMatrix, this._viewMatrix);\r\n this._viewMatrix.getTranslationToRef(this._globalPosition);\r\n this._viewMatrix.invert();\r\n this._markSyncedWithParent();\r\n } else {\r\n this._globalPosition.copyFrom(position);\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public createRigCamera(name: string, cameraIndex: number): Nullable {\r\n if (this.cameraRigMode !== Camera.RIG_MODE_NONE) {\r\n const rigCamera = new TargetCamera(name, this.position.clone(), this.getScene());\r\n rigCamera.isRigCamera = true;\r\n rigCamera.rigParent = this;\r\n if (this.cameraRigMode === Camera.RIG_MODE_VR || this.cameraRigMode === Camera.RIG_MODE_WEBVR) {\r\n if (!this.rotationQuaternion) {\r\n this.rotationQuaternion = new Quaternion();\r\n }\r\n rigCamera._cameraRigParams = {};\r\n rigCamera.rotationQuaternion = new Quaternion();\r\n }\r\n\r\n rigCamera.mode = this.mode;\r\n rigCamera.orthoLeft = this.orthoLeft;\r\n rigCamera.orthoRight = this.orthoRight;\r\n rigCamera.orthoTop = this.orthoTop;\r\n rigCamera.orthoBottom = this.orthoBottom;\r\n\r\n return rigCamera;\r\n }\r\n return null;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _updateRigCameras() {\r\n const camLeft = this._rigCameras[0];\r\n const camRight = this._rigCameras[1];\r\n\r\n this.computeWorldMatrix();\r\n\r\n switch (this.cameraRigMode) {\r\n case Camera.RIG_MODE_STEREOSCOPIC_ANAGLYPH:\r\n case Camera.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL:\r\n case Camera.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED:\r\n case Camera.RIG_MODE_STEREOSCOPIC_OVERUNDER:\r\n case Camera.RIG_MODE_STEREOSCOPIC_INTERLACED: {\r\n //provisionnaly using _cameraRigParams.stereoHalfAngle instead of calculations based on _cameraRigParams.interaxialDistance:\r\n const leftSign = this.cameraRigMode === Camera.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED ? 1 : -1;\r\n const rightSign = this.cameraRigMode === Camera.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED ? -1 : 1;\r\n this._getRigCamPositionAndTarget(this._cameraRigParams.stereoHalfAngle * leftSign, camLeft);\r\n this._getRigCamPositionAndTarget(this._cameraRigParams.stereoHalfAngle * rightSign, camRight);\r\n break;\r\n }\r\n case Camera.RIG_MODE_VR:\r\n if (camLeft.rotationQuaternion) {\r\n camLeft.rotationQuaternion.copyFrom(this.rotationQuaternion);\r\n camRight.rotationQuaternion.copyFrom(this.rotationQuaternion);\r\n } else {\r\n camLeft.rotation.copyFrom(this.rotation);\r\n camRight.rotation.copyFrom(this.rotation);\r\n }\r\n camLeft.position.copyFrom(this.position);\r\n camRight.position.copyFrom(this.position);\r\n\r\n break;\r\n }\r\n super._updateRigCameras();\r\n }\r\n\r\n private _getRigCamPositionAndTarget(halfSpace: number, rigCamera: TargetCamera) {\r\n const target = this.getTarget();\r\n target.subtractToRef(this.position, TargetCamera._TargetFocalPoint);\r\n\r\n TargetCamera._TargetFocalPoint.normalize().scaleInPlace(this._initialFocalDistance);\r\n const newFocalTarget = TargetCamera._TargetFocalPoint.addInPlace(this.position);\r\n\r\n Matrix.TranslationToRef(-newFocalTarget.x, -newFocalTarget.y, -newFocalTarget.z, TargetCamera._TargetTransformMatrix);\r\n TargetCamera._TargetTransformMatrix.multiplyToRef(Matrix.RotationAxis(rigCamera.upVector, halfSpace), TargetCamera._RigCamTransformMatrix);\r\n Matrix.TranslationToRef(newFocalTarget.x, newFocalTarget.y, newFocalTarget.z, TargetCamera._TargetTransformMatrix);\r\n\r\n TargetCamera._RigCamTransformMatrix.multiplyToRef(TargetCamera._TargetTransformMatrix, TargetCamera._RigCamTransformMatrix);\r\n\r\n Vector3.TransformCoordinatesToRef(this.position, TargetCamera._RigCamTransformMatrix, rigCamera.position);\r\n rigCamera.setTarget(newFocalTarget);\r\n }\r\n\r\n /**\r\n * Gets the current object class name.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"TargetCamera\";\r\n }\r\n}\r\n","import { Logger } from \"../Misc/logger\";\r\nimport { SerializationHelper } from \"../Misc/decorators\";\r\nimport type { Nullable } from \"../types\";\r\nimport { Camera } from \"./camera\";\r\n/**\r\n * @ignore\r\n * This is a list of all the different input types that are available in the application.\r\n * Fo instance: ArcRotateCameraGamepadInput...\r\n */\r\n// eslint-disable-next-line no-var, @typescript-eslint/naming-convention\r\nexport var CameraInputTypes = {};\r\n\r\n/**\r\n * This is the contract to implement in order to create a new input class.\r\n * Inputs are dealing with listening to user actions and moving the camera accordingly.\r\n */\r\nexport interface ICameraInput {\r\n /**\r\n * Defines the camera the input is attached to.\r\n */\r\n camera: Nullable;\r\n /**\r\n * Gets the class name of the current input.\r\n * @returns the class name\r\n */\r\n getClassName(): string;\r\n /**\r\n * Get the friendly name associated with the input class.\r\n * @returns the input friendly name\r\n */\r\n getSimpleName(): string;\r\n /**\r\n * Attach the input controls to a specific dom element to get the input from.\r\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n */\r\n attachControl(noPreventDefault?: boolean): void;\r\n /**\r\n * Detach the current controls from the specified dom element.\r\n */\r\n detachControl(): void;\r\n /**\r\n * Update the current camera state depending on the inputs that have been used this frame.\r\n * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop.\r\n */\r\n checkInputs?: () => void;\r\n}\r\n\r\n/**\r\n * Represents a map of input types to input instance or input index to input instance.\r\n */\r\nexport interface CameraInputsMap {\r\n /**\r\n * Accessor to the input by input type.\r\n */\r\n [name: string]: ICameraInput;\r\n /**\r\n * Accessor to the input by input index.\r\n */\r\n [idx: number]: ICameraInput;\r\n}\r\n\r\n/**\r\n * This represents the input manager used within a camera.\r\n * It helps dealing with all the different kind of input attached to a camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\r\n */\r\nexport class CameraInputsManager {\r\n /**\r\n * Defines the list of inputs attached to the camera.\r\n */\r\n public attached: CameraInputsMap;\r\n\r\n /**\r\n * Defines the dom element the camera is collecting inputs from.\r\n * This is null if the controls have not been attached.\r\n */\r\n public attachedToElement: boolean = false;\r\n\r\n /**\r\n * Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n */\r\n public noPreventDefault: boolean;\r\n\r\n /**\r\n * Defined the camera the input manager belongs to.\r\n */\r\n public camera: TCamera;\r\n\r\n /**\r\n * Update the current camera state depending on the inputs that have been used this frame.\r\n * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop.\r\n */\r\n public checkInputs: () => void;\r\n\r\n /**\r\n * Instantiate a new Camera Input Manager.\r\n * @param camera Defines the camera the input manager belongs to\r\n */\r\n constructor(camera: TCamera) {\r\n this.attached = {};\r\n this.camera = camera;\r\n this.checkInputs = () => {};\r\n }\r\n\r\n /**\r\n * Add an input method to a camera\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\r\n * @param input Camera input method\r\n */\r\n public add(input: ICameraInput): void {\r\n const type = input.getSimpleName();\r\n if (this.attached[type]) {\r\n Logger.Warn(\"camera input of type \" + type + \" already exists on camera\");\r\n return;\r\n }\r\n\r\n this.attached[type] = input;\r\n\r\n input.camera = this.camera;\r\n\r\n // for checkInputs, we are dynamically creating a function\r\n // the goal is to avoid the performance penalty of looping for inputs in the render loop\r\n if (input.checkInputs) {\r\n this.checkInputs = this._addCheckInputs(input.checkInputs.bind(input));\r\n }\r\n\r\n if (this.attachedToElement) {\r\n input.attachControl(this.noPreventDefault);\r\n }\r\n }\r\n\r\n /**\r\n * Remove a specific input method from a camera\r\n * example: camera.inputs.remove(camera.inputs.attached.mouse);\r\n * @param inputToRemove camera input method\r\n */\r\n public remove(inputToRemove: ICameraInput): void {\r\n for (const cam in this.attached) {\r\n const input = this.attached[cam];\r\n if (input === inputToRemove) {\r\n input.detachControl();\r\n input.camera = null;\r\n delete this.attached[cam];\r\n this.rebuildInputCheck();\r\n\r\n return;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Remove a specific input type from a camera\r\n * example: camera.inputs.remove(\"ArcRotateCameraGamepadInput\");\r\n * @param inputType the type of the input to remove\r\n */\r\n public removeByType(inputType: string): void {\r\n for (const cam in this.attached) {\r\n const input = this.attached[cam];\r\n if (input.getClassName() === inputType) {\r\n input.detachControl();\r\n input.camera = null;\r\n delete this.attached[cam];\r\n this.rebuildInputCheck();\r\n }\r\n }\r\n }\r\n\r\n private _addCheckInputs(fn: () => void) {\r\n const current = this.checkInputs;\r\n return () => {\r\n current();\r\n fn();\r\n };\r\n }\r\n\r\n /**\r\n * Attach the input controls to the currently attached dom element to listen the events from.\r\n * @param input Defines the input to attach\r\n */\r\n public attachInput(input: ICameraInput): void {\r\n if (this.attachedToElement) {\r\n input.attachControl(this.noPreventDefault);\r\n }\r\n }\r\n\r\n /**\r\n * Attach the current manager inputs controls to a specific dom element to listen the events from.\r\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n */\r\n public attachElement(noPreventDefault: boolean = false): void {\r\n if (this.attachedToElement) {\r\n return;\r\n }\r\n\r\n noPreventDefault = Camera.ForceAttachControlToAlwaysPreventDefault ? false : noPreventDefault;\r\n this.attachedToElement = true;\r\n this.noPreventDefault = noPreventDefault;\r\n\r\n for (const cam in this.attached) {\r\n this.attached[cam].attachControl(noPreventDefault);\r\n }\r\n }\r\n\r\n /**\r\n * Detach the current manager inputs controls from a specific dom element.\r\n * @param disconnect Defines whether the input should be removed from the current list of attached inputs\r\n */\r\n public detachElement(disconnect = false): void {\r\n for (const cam in this.attached) {\r\n this.attached[cam].detachControl();\r\n\r\n if (disconnect) {\r\n this.attached[cam].camera = null;\r\n }\r\n }\r\n this.attachedToElement = false;\r\n }\r\n\r\n /**\r\n * Rebuild the dynamic inputCheck function from the current list of\r\n * defined inputs in the manager.\r\n */\r\n public rebuildInputCheck(): void {\r\n this.checkInputs = () => {};\r\n\r\n for (const cam in this.attached) {\r\n const input = this.attached[cam];\r\n if (input.checkInputs) {\r\n this.checkInputs = this._addCheckInputs(input.checkInputs.bind(input));\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Remove all attached input methods from a camera\r\n */\r\n public clear(): void {\r\n if (this.attachedToElement) {\r\n this.detachElement(true);\r\n }\r\n this.attached = {};\r\n this.attachedToElement = false;\r\n this.checkInputs = () => {};\r\n }\r\n\r\n /**\r\n * Serialize the current input manager attached to a camera.\r\n * This ensures than once parsed,\r\n * the input associated to the camera will be identical to the current ones\r\n * @param serializedCamera Defines the camera serialization JSON the input serialization should write to\r\n */\r\n public serialize(serializedCamera: any): void {\r\n const inputs: { [key: string]: any } = {};\r\n for (const cam in this.attached) {\r\n const input = this.attached[cam];\r\n const res = SerializationHelper.Serialize(input);\r\n inputs[input.getClassName()] = res;\r\n }\r\n\r\n serializedCamera.inputsmgr = inputs;\r\n }\r\n\r\n /**\r\n * Parses an input manager serialized JSON to restore the previous list of inputs\r\n * and states associated to a camera.\r\n * @param parsedCamera Defines the JSON to parse\r\n */\r\n public parse(parsedCamera: any): void {\r\n const parsedInputs = parsedCamera.inputsmgr;\r\n if (parsedInputs) {\r\n this.clear();\r\n\r\n for (const n in parsedInputs) {\r\n const construct = (CameraInputTypes)[n];\r\n if (construct) {\r\n const parsedinput = parsedInputs[n];\r\n const input = SerializationHelper.Parse(\r\n () => {\r\n return new construct();\r\n },\r\n parsedinput,\r\n null\r\n );\r\n this.add(input as any);\r\n }\r\n }\r\n } else {\r\n //2016-03-08 this part is for managing backward compatibility\r\n for (const n in this.attached) {\r\n const construct = (CameraInputTypes)[this.attached[n].getClassName()];\r\n if (construct) {\r\n const input = SerializationHelper.Parse(\r\n () => {\r\n return new construct();\r\n },\r\n parsedCamera,\r\n null\r\n );\r\n this.remove(this.attached[n]);\r\n this.add(input as any);\r\n }\r\n }\r\n }\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/no-unused-vars */\r\nimport type { Nullable } from \"../../types\";\r\nimport { serialize } from \"../../Misc/decorators\";\r\nimport type { EventState, Observer } from \"../../Misc/observable\";\r\nimport { Tools } from \"../../Misc/tools\";\r\nimport type { Camera } from \"../../Cameras/camera\";\r\nimport type { ICameraInput } from \"../../Cameras/cameraInputsManager\";\r\nimport type { PointerInfo, PointerTouch } from \"../../Events/pointerEvents\";\r\nimport { PointerEventTypes } from \"../../Events/pointerEvents\";\r\nimport type { IPointerEvent } from \"../../Events/deviceInputEvents\";\r\n\r\n/**\r\n * Base class for Camera Pointer Inputs.\r\n * See FollowCameraPointersInput in src/Cameras/Inputs/followCameraPointersInput.ts\r\n * for example usage.\r\n */\r\nexport abstract class BaseCameraPointersInput implements ICameraInput {\r\n /**\r\n * Defines the camera the input is attached to.\r\n */\r\n public abstract camera: Camera;\r\n\r\n /**\r\n * Whether keyboard modifier keys are pressed at time of last mouse event.\r\n */\r\n protected _altKey: boolean;\r\n protected _ctrlKey: boolean;\r\n protected _metaKey: boolean;\r\n protected _shiftKey: boolean;\r\n\r\n /**\r\n * Which mouse buttons were pressed at time of last mouse event.\r\n * https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent/buttons\r\n */\r\n protected _buttonsPressed: number;\r\n\r\n private _currentActiveButton: number = -1;\r\n private _contextMenuBind: EventListener;\r\n\r\n /**\r\n * Defines the buttons associated with the input to handle camera move.\r\n */\r\n @serialize()\r\n public buttons = [0, 1, 2];\r\n\r\n /**\r\n * Attach the input controls to a specific dom element to get the input from.\r\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n */\r\n public attachControl(noPreventDefault?: boolean): void {\r\n // eslint-disable-next-line prefer-rest-params\r\n noPreventDefault = Tools.BackCompatCameraNoPreventDefault(arguments);\r\n const engine = this.camera.getEngine();\r\n const element = engine.getInputElement();\r\n let previousPinchSquaredDistance = 0;\r\n let previousMultiTouchPanPosition: Nullable = null;\r\n\r\n this._pointA = null;\r\n this._pointB = null;\r\n\r\n this._altKey = false;\r\n this._ctrlKey = false;\r\n this._metaKey = false;\r\n this._shiftKey = false;\r\n this._buttonsPressed = 0;\r\n\r\n this._pointerInput = (p) => {\r\n const evt = p.event;\r\n const isTouch = evt.pointerType === \"touch\";\r\n\r\n if (engine.isInVRExclusivePointerMode) {\r\n return;\r\n }\r\n\r\n if (p.type !== PointerEventTypes.POINTERMOVE && this.buttons.indexOf(evt.button) === -1) {\r\n return;\r\n }\r\n\r\n const srcElement = evt.target;\r\n\r\n this._altKey = evt.altKey;\r\n this._ctrlKey = evt.ctrlKey;\r\n this._metaKey = evt.metaKey;\r\n this._shiftKey = evt.shiftKey;\r\n this._buttonsPressed = evt.buttons;\r\n\r\n if (engine.isPointerLock) {\r\n const offsetX = evt.movementX;\r\n const offsetY = evt.movementY;\r\n\r\n this.onTouch(null, offsetX, offsetY);\r\n this._pointA = null;\r\n this._pointB = null;\r\n } else if (p.type !== PointerEventTypes.POINTERDOWN && isTouch && this._pointA?.pointerId !== evt.pointerId && this._pointB?.pointerId !== evt.pointerId) {\r\n return; // If we get a non-down event for a touch that we're not tracking, ignore it\r\n } else if (p.type === PointerEventTypes.POINTERDOWN && (this._currentActiveButton === -1 || isTouch)) {\r\n try {\r\n srcElement?.setPointerCapture(evt.pointerId);\r\n } catch (e) {\r\n //Nothing to do with the error. Execution will continue.\r\n }\r\n\r\n if (this._pointA === null) {\r\n this._pointA = {\r\n x: evt.clientX,\r\n y: evt.clientY,\r\n pointerId: evt.pointerId,\r\n type: evt.pointerType,\r\n };\r\n } else if (this._pointB === null) {\r\n this._pointB = {\r\n x: evt.clientX,\r\n y: evt.clientY,\r\n pointerId: evt.pointerId,\r\n type: evt.pointerType,\r\n };\r\n } else {\r\n return; // We are already tracking two pointers so ignore this one\r\n }\r\n\r\n if (this._currentActiveButton === -1 && !isTouch) {\r\n this._currentActiveButton = evt.button;\r\n }\r\n this.onButtonDown(evt);\r\n\r\n if (!noPreventDefault) {\r\n evt.preventDefault();\r\n element && element.focus();\r\n }\r\n } else if (p.type === PointerEventTypes.POINTERDOUBLETAP) {\r\n this.onDoubleTap(evt.pointerType);\r\n } else if (p.type === PointerEventTypes.POINTERUP && (this._currentActiveButton === evt.button || isTouch)) {\r\n try {\r\n srcElement?.releasePointerCapture(evt.pointerId);\r\n } catch (e) {\r\n //Nothing to do with the error.\r\n }\r\n\r\n if (!isTouch) {\r\n this._pointB = null; // Mouse and pen are mono pointer\r\n }\r\n\r\n //would be better to use pointers.remove(evt.pointerId) for multitouch gestures,\r\n //but emptying completely pointers collection is required to fix a bug on iPhone :\r\n //when changing orientation while pinching camera,\r\n //one pointer stay pressed forever if we don't release all pointers\r\n //will be ok to put back pointers.remove(evt.pointerId); when iPhone bug corrected\r\n if (engine._badOS) {\r\n this._pointA = this._pointB = null;\r\n } else {\r\n //only remove the impacted pointer in case of multitouch allowing on most\r\n //platforms switching from rotate to zoom and pan seamlessly.\r\n if (this._pointB && this._pointA && this._pointA.pointerId == evt.pointerId) {\r\n this._pointA = this._pointB;\r\n this._pointB = null;\r\n } else if (this._pointA && this._pointB && this._pointB.pointerId == evt.pointerId) {\r\n this._pointB = null;\r\n } else {\r\n this._pointA = this._pointB = null;\r\n }\r\n }\r\n\r\n if (previousPinchSquaredDistance !== 0 || previousMultiTouchPanPosition) {\r\n // Previous pinch data is populated but a button has been lifted\r\n // so pinch has ended.\r\n this.onMultiTouch(\r\n this._pointA,\r\n this._pointB,\r\n previousPinchSquaredDistance,\r\n 0, // pinchSquaredDistance\r\n previousMultiTouchPanPosition,\r\n null // multiTouchPanPosition\r\n );\r\n previousPinchSquaredDistance = 0;\r\n previousMultiTouchPanPosition = null;\r\n }\r\n\r\n this._currentActiveButton = -1;\r\n this.onButtonUp(evt);\r\n\r\n if (!noPreventDefault) {\r\n evt.preventDefault();\r\n }\r\n } else if (p.type === PointerEventTypes.POINTERMOVE) {\r\n if (!noPreventDefault) {\r\n evt.preventDefault();\r\n }\r\n\r\n // One button down\r\n if (this._pointA && this._pointB === null) {\r\n const offsetX = evt.clientX - this._pointA.x;\r\n const offsetY = evt.clientY - this._pointA.y;\r\n this.onTouch(this._pointA, offsetX, offsetY);\r\n\r\n this._pointA.x = evt.clientX;\r\n this._pointA.y = evt.clientY;\r\n }\r\n // Two buttons down: pinch\r\n else if (this._pointA && this._pointB) {\r\n const ed = this._pointA.pointerId === evt.pointerId ? this._pointA : this._pointB;\r\n ed.x = evt.clientX;\r\n ed.y = evt.clientY;\r\n const distX = this._pointA.x - this._pointB.x;\r\n const distY = this._pointA.y - this._pointB.y;\r\n const pinchSquaredDistance = distX * distX + distY * distY;\r\n const multiTouchPanPosition = {\r\n x: (this._pointA.x + this._pointB.x) / 2,\r\n y: (this._pointA.y + this._pointB.y) / 2,\r\n pointerId: evt.pointerId,\r\n type: p.type,\r\n };\r\n\r\n this.onMultiTouch(this._pointA, this._pointB, previousPinchSquaredDistance, pinchSquaredDistance, previousMultiTouchPanPosition, multiTouchPanPosition);\r\n\r\n previousMultiTouchPanPosition = multiTouchPanPosition;\r\n previousPinchSquaredDistance = pinchSquaredDistance;\r\n }\r\n }\r\n };\r\n\r\n this._observer = this.camera\r\n .getScene()\r\n ._inputManager._addCameraPointerObserver(\r\n this._pointerInput,\r\n PointerEventTypes.POINTERDOWN | PointerEventTypes.POINTERUP | PointerEventTypes.POINTERMOVE | PointerEventTypes.POINTERDOUBLETAP\r\n );\r\n\r\n this._onLostFocus = () => {\r\n this._pointA = this._pointB = null;\r\n previousPinchSquaredDistance = 0;\r\n previousMultiTouchPanPosition = null;\r\n this.onLostFocus();\r\n };\r\n\r\n this._contextMenuBind = this.onContextMenu.bind(this);\r\n\r\n element && element.addEventListener(\"contextmenu\", this._contextMenuBind, false);\r\n\r\n const hostWindow = this.camera.getScene().getEngine().getHostWindow();\r\n\r\n if (hostWindow) {\r\n Tools.RegisterTopRootEvents(hostWindow, [{ name: \"blur\", handler: this._onLostFocus }]);\r\n }\r\n }\r\n\r\n /**\r\n * Detach the current controls from the specified dom element.\r\n */\r\n public detachControl(): void {\r\n if (this._onLostFocus) {\r\n const hostWindow = this.camera.getScene().getEngine().getHostWindow();\r\n if (hostWindow) {\r\n Tools.UnregisterTopRootEvents(hostWindow, [{ name: \"blur\", handler: this._onLostFocus }]);\r\n }\r\n }\r\n\r\n if (this._observer) {\r\n this.camera.getScene()._inputManager._removeCameraPointerObserver(this._observer);\r\n this._observer = null;\r\n\r\n if (this._contextMenuBind) {\r\n const inputElement = this.camera.getScene().getEngine().getInputElement();\r\n inputElement && inputElement.removeEventListener(\"contextmenu\", this._contextMenuBind);\r\n }\r\n\r\n this._onLostFocus = null;\r\n }\r\n\r\n this._altKey = false;\r\n this._ctrlKey = false;\r\n this._metaKey = false;\r\n this._shiftKey = false;\r\n this._buttonsPressed = 0;\r\n this._currentActiveButton = -1;\r\n }\r\n\r\n /**\r\n * Gets the class name of the current input.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"BaseCameraPointersInput\";\r\n }\r\n\r\n /**\r\n * Get the friendly name associated with the input class.\r\n * @returns the input friendly name\r\n */\r\n public getSimpleName(): string {\r\n return \"pointers\";\r\n }\r\n\r\n /**\r\n * Called on pointer POINTERDOUBLETAP event.\r\n * Override this method to provide functionality on POINTERDOUBLETAP event.\r\n * @param type\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public onDoubleTap(type: string) {}\r\n\r\n /**\r\n * Called on pointer POINTERMOVE event if only a single touch is active.\r\n * Override this method to provide functionality.\r\n * @param point\r\n * @param offsetX\r\n * @param offsetY\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public onTouch(point: Nullable, offsetX: number, offsetY: number): void {}\r\n\r\n /**\r\n * Called on pointer POINTERMOVE event if multiple touches are active.\r\n * Override this method to provide functionality.\r\n * @param _pointA\r\n * @param _pointB\r\n * @param previousPinchSquaredDistance\r\n * @param pinchSquaredDistance\r\n * @param previousMultiTouchPanPosition\r\n * @param multiTouchPanPosition\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public onMultiTouch(\r\n _pointA: Nullable,\r\n _pointB: Nullable,\r\n previousPinchSquaredDistance: number,\r\n pinchSquaredDistance: number,\r\n previousMultiTouchPanPosition: Nullable,\r\n multiTouchPanPosition: Nullable\r\n ): void {}\r\n\r\n /**\r\n * Called on JS contextmenu event.\r\n * Override this method to provide functionality.\r\n * @param evt\r\n */\r\n public onContextMenu(evt: PointerEvent): void {\r\n evt.preventDefault();\r\n }\r\n\r\n /**\r\n * Called each time a new POINTERDOWN event occurs. Ie, for each button\r\n * press.\r\n * Override this method to provide functionality.\r\n * @param evt\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public onButtonDown(evt: IPointerEvent): void {}\r\n\r\n /**\r\n * Called each time a new POINTERUP event occurs. Ie, for each button\r\n * release.\r\n * Override this method to provide functionality.\r\n * @param evt\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public onButtonUp(evt: IPointerEvent): void {}\r\n\r\n /**\r\n * Called when window becomes inactive.\r\n * Override this method to provide functionality.\r\n */\r\n public onLostFocus(): void {}\r\n\r\n private _pointerInput: (p: PointerInfo, s: EventState) => void;\r\n private _observer: Nullable>;\r\n private _onLostFocus: Nullable<(e: FocusEvent) => any>;\r\n private _pointA: Nullable;\r\n private _pointB: Nullable;\r\n}\r\n","import type { Nullable } from \"../../types\";\r\nimport { serialize } from \"../../Misc/decorators\";\r\nimport type { ArcRotateCamera } from \"../../Cameras/arcRotateCamera\";\r\nimport { CameraInputTypes } from \"../../Cameras/cameraInputsManager\";\r\nimport { BaseCameraPointersInput } from \"../../Cameras/Inputs/BaseCameraPointersInput\";\r\nimport type { PointerTouch } from \"../../Events/pointerEvents\";\r\nimport type { IPointerEvent } from \"../../Events/deviceInputEvents\";\r\n\r\n/**\r\n * Manage the pointers inputs to control an arc rotate camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\r\n */\r\nexport class ArcRotateCameraPointersInput extends BaseCameraPointersInput {\r\n /**\r\n * Defines the camera the input is attached to.\r\n */\r\n public camera: ArcRotateCamera;\r\n\r\n /**\r\n * The minimum radius used for pinch, to avoid radius lock at 0\r\n */\r\n public static MinimumRadiusForPinch: number = 0.001;\r\n\r\n /**\r\n * Gets the class name of the current input.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"ArcRotateCameraPointersInput\";\r\n }\r\n\r\n /**\r\n * Defines the buttons associated with the input to handle camera move.\r\n */\r\n @serialize()\r\n public buttons = [0, 1, 2];\r\n\r\n /**\r\n * Defines the pointer angular sensibility along the X axis or how fast is\r\n * the camera rotating.\r\n */\r\n @serialize()\r\n public angularSensibilityX = 1000.0;\r\n\r\n /**\r\n * Defines the pointer angular sensibility along the Y axis or how fast is\r\n * the camera rotating.\r\n */\r\n @serialize()\r\n public angularSensibilityY = 1000.0;\r\n\r\n /**\r\n * Defines the pointer pinch precision or how fast is the camera zooming.\r\n */\r\n @serialize()\r\n public pinchPrecision = 12.0;\r\n\r\n /**\r\n * pinchDeltaPercentage will be used instead of pinchPrecision if different\r\n * from 0.\r\n * It defines the percentage of current camera.radius to use as delta when\r\n * pinch zoom is used.\r\n */\r\n @serialize()\r\n public pinchDeltaPercentage = 0;\r\n\r\n /**\r\n * When useNaturalPinchZoom is true, multi touch zoom will zoom in such\r\n * that any object in the plane at the camera's target point will scale\r\n * perfectly with finger motion.\r\n * Overrides pinchDeltaPercentage and pinchPrecision.\r\n */\r\n @serialize()\r\n public useNaturalPinchZoom: boolean = false;\r\n\r\n /**\r\n * Defines whether zoom (2 fingers pinch) is enabled through multitouch\r\n */\r\n @serialize()\r\n public pinchZoom: boolean = true;\r\n\r\n /**\r\n * Defines the pointer panning sensibility or how fast is the camera moving.\r\n */\r\n @serialize()\r\n public panningSensibility: number = 1000.0;\r\n\r\n /**\r\n * Defines whether panning (2 fingers swipe) is enabled through multitouch.\r\n */\r\n @serialize()\r\n public multiTouchPanning: boolean = true;\r\n\r\n /**\r\n * Defines whether panning is enabled for both pan (2 fingers swipe) and\r\n * zoom (pinch) through multitouch.\r\n */\r\n @serialize()\r\n public multiTouchPanAndZoom: boolean = true;\r\n\r\n /**\r\n * Revers pinch action direction.\r\n */\r\n public pinchInwards = true;\r\n\r\n private _isPanClick: boolean = false;\r\n private _twoFingerActivityCount: number = 0;\r\n private _isPinching: boolean = false;\r\n\r\n /**\r\n * Move camera from multi touch panning positions.\r\n * @param previousMultiTouchPanPosition\r\n * @param multiTouchPanPosition\r\n */\r\n private _computeMultiTouchPanning(previousMultiTouchPanPosition: Nullable, multiTouchPanPosition: Nullable): void {\r\n if (this.panningSensibility !== 0 && previousMultiTouchPanPosition && multiTouchPanPosition) {\r\n const moveDeltaX = multiTouchPanPosition.x - previousMultiTouchPanPosition.x;\r\n const moveDeltaY = multiTouchPanPosition.y - previousMultiTouchPanPosition.y;\r\n this.camera.inertialPanningX += -moveDeltaX / this.panningSensibility;\r\n this.camera.inertialPanningY += moveDeltaY / this.panningSensibility;\r\n }\r\n }\r\n\r\n /**\r\n * Move camera from pinch zoom distances.\r\n * @param previousPinchSquaredDistance\r\n * @param pinchSquaredDistance\r\n */\r\n private _computePinchZoom(previousPinchSquaredDistance: number, pinchSquaredDistance: number): void {\r\n const radius = this.camera.radius || ArcRotateCameraPointersInput.MinimumRadiusForPinch;\r\n if (this.useNaturalPinchZoom) {\r\n this.camera.radius = (radius * Math.sqrt(previousPinchSquaredDistance)) / Math.sqrt(pinchSquaredDistance);\r\n } else if (this.pinchDeltaPercentage) {\r\n this.camera.inertialRadiusOffset += (pinchSquaredDistance - previousPinchSquaredDistance) * 0.001 * radius * this.pinchDeltaPercentage;\r\n } else {\r\n this.camera.inertialRadiusOffset +=\r\n (pinchSquaredDistance - previousPinchSquaredDistance) /\r\n ((this.pinchPrecision * (this.pinchInwards ? 1 : -1) * (this.angularSensibilityX + this.angularSensibilityY)) / 2);\r\n }\r\n }\r\n\r\n /**\r\n * Called on pointer POINTERMOVE event if only a single touch is active.\r\n * @param point\r\n * @param offsetX\r\n * @param offsetY\r\n */\r\n public onTouch(point: Nullable, offsetX: number, offsetY: number): void {\r\n if (this.panningSensibility !== 0 && ((this._ctrlKey && this.camera._useCtrlForPanning) || this._isPanClick)) {\r\n this.camera.inertialPanningX += -offsetX / this.panningSensibility;\r\n this.camera.inertialPanningY += offsetY / this.panningSensibility;\r\n } else {\r\n this.camera.inertialAlphaOffset -= offsetX / this.angularSensibilityX;\r\n this.camera.inertialBetaOffset -= offsetY / this.angularSensibilityY;\r\n }\r\n }\r\n\r\n /**\r\n * Called on pointer POINTERDOUBLETAP event.\r\n */\r\n public onDoubleTap() {\r\n if (this.camera.useInputToRestoreState) {\r\n this.camera.restoreState();\r\n }\r\n }\r\n\r\n /**\r\n * Called on pointer POINTERMOVE event if multiple touches are active.\r\n * @param pointA\r\n * @param pointB\r\n * @param previousPinchSquaredDistance\r\n * @param pinchSquaredDistance\r\n * @param previousMultiTouchPanPosition\r\n * @param multiTouchPanPosition\r\n */\r\n public onMultiTouch(\r\n pointA: Nullable,\r\n pointB: Nullable,\r\n previousPinchSquaredDistance: number,\r\n pinchSquaredDistance: number,\r\n previousMultiTouchPanPosition: Nullable,\r\n multiTouchPanPosition: Nullable\r\n ): void {\r\n if (previousPinchSquaredDistance === 0 && previousMultiTouchPanPosition === null) {\r\n // First time this method is called for new pinch.\r\n // Next time this is called there will be a\r\n // previousPinchSquaredDistance and pinchSquaredDistance to compare.\r\n return;\r\n }\r\n if (pinchSquaredDistance === 0 && multiTouchPanPosition === null) {\r\n // Last time this method is called at the end of a pinch.\r\n return;\r\n }\r\n\r\n // Zoom and panning enabled together\r\n if (this.multiTouchPanAndZoom) {\r\n this._computePinchZoom(previousPinchSquaredDistance, pinchSquaredDistance);\r\n this._computeMultiTouchPanning(previousMultiTouchPanPosition, multiTouchPanPosition);\r\n\r\n // Zoom and panning enabled but only one at a time\r\n } else if (this.multiTouchPanning && this.pinchZoom) {\r\n this._twoFingerActivityCount++;\r\n\r\n if (\r\n this._isPinching ||\r\n (this._twoFingerActivityCount < 20 && Math.abs(Math.sqrt(pinchSquaredDistance) - Math.sqrt(previousPinchSquaredDistance)) > this.camera.pinchToPanMaxDistance)\r\n ) {\r\n // Since pinch has not been active long, assume we intend to zoom.\r\n this._computePinchZoom(previousPinchSquaredDistance, pinchSquaredDistance);\r\n\r\n // Since we are pinching, remain pinching on next iteration.\r\n this._isPinching = true;\r\n } else {\r\n // Pause between pinch starting and moving implies not a zoom event. Pan instead.\r\n this._computeMultiTouchPanning(previousMultiTouchPanPosition, multiTouchPanPosition);\r\n }\r\n\r\n // Panning enabled, zoom disabled\r\n } else if (this.multiTouchPanning) {\r\n this._computeMultiTouchPanning(previousMultiTouchPanPosition, multiTouchPanPosition);\r\n\r\n // Zoom enabled, panning disabled\r\n } else if (this.pinchZoom) {\r\n this._computePinchZoom(previousPinchSquaredDistance, pinchSquaredDistance);\r\n }\r\n }\r\n\r\n /**\r\n * Called each time a new POINTERDOWN event occurs. Ie, for each button\r\n * press.\r\n * @param evt\r\n */\r\n public onButtonDown(evt: IPointerEvent): void {\r\n this._isPanClick = evt.button === this.camera._panningMouseButton;\r\n }\r\n\r\n /**\r\n * Called each time a new POINTERUP event occurs. Ie, for each button\r\n * release.\r\n */\r\n public onButtonUp(): void {\r\n this._twoFingerActivityCount = 0;\r\n this._isPinching = false;\r\n }\r\n\r\n /**\r\n * Called when window becomes inactive.\r\n */\r\n public onLostFocus(): void {\r\n this._isPanClick = false;\r\n this._twoFingerActivityCount = 0;\r\n this._isPinching = false;\r\n }\r\n}\r\n(CameraInputTypes)[\"ArcRotateCameraPointersInput\"] = ArcRotateCameraPointersInput;\r\n","import type { Nullable } from \"../../types\";\r\nimport { serialize } from \"../../Misc/decorators\";\r\nimport type { Observer } from \"../../Misc/observable\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { ArcRotateCamera } from \"../../Cameras/arcRotateCamera\";\r\nimport type { ICameraInput } from \"../../Cameras/cameraInputsManager\";\r\nimport { CameraInputTypes } from \"../../Cameras/cameraInputsManager\";\r\nimport type { Engine } from \"../../Engines/engine\";\r\nimport type { KeyboardInfo } from \"../../Events/keyboardEvents\";\r\nimport { KeyboardEventTypes } from \"../../Events/keyboardEvents\";\r\nimport { Tools } from \"../../Misc/tools\";\r\n\r\n/**\r\n * Manage the keyboard inputs to control the movement of an arc rotate camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\r\n */\r\nexport class ArcRotateCameraKeyboardMoveInput implements ICameraInput {\r\n /**\r\n * Defines the camera the input is attached to.\r\n */\r\n public camera: ArcRotateCamera;\r\n\r\n /**\r\n * Defines the list of key codes associated with the up action (increase alpha)\r\n */\r\n @serialize()\r\n public keysUp = [38];\r\n\r\n /**\r\n * Defines the list of key codes associated with the down action (decrease alpha)\r\n */\r\n @serialize()\r\n public keysDown = [40];\r\n\r\n /**\r\n * Defines the list of key codes associated with the left action (increase beta)\r\n */\r\n @serialize()\r\n public keysLeft = [37];\r\n\r\n /**\r\n * Defines the list of key codes associated with the right action (decrease beta)\r\n */\r\n @serialize()\r\n public keysRight = [39];\r\n\r\n /**\r\n * Defines the list of key codes associated with the reset action.\r\n * Those keys reset the camera to its last stored state (with the method camera.storeState())\r\n */\r\n @serialize()\r\n public keysReset = [220];\r\n\r\n /**\r\n * Defines the panning sensibility of the inputs.\r\n * (How fast is the camera panning)\r\n */\r\n @serialize()\r\n public panningSensibility: number = 50.0;\r\n\r\n /**\r\n * Defines the zooming sensibility of the inputs.\r\n * (How fast is the camera zooming)\r\n */\r\n @serialize()\r\n public zoomingSensibility: number = 25.0;\r\n\r\n /**\r\n * Defines whether maintaining the alt key down switch the movement mode from\r\n * orientation to zoom.\r\n */\r\n @serialize()\r\n public useAltToZoom: boolean = true;\r\n\r\n /**\r\n * Rotation speed of the camera\r\n */\r\n @serialize()\r\n public angularSpeed = 0.01;\r\n\r\n private _keys = new Array();\r\n private _ctrlPressed: boolean;\r\n private _altPressed: boolean;\r\n private _onCanvasBlurObserver: Nullable>;\r\n private _onKeyboardObserver: Nullable>;\r\n private _engine: Engine;\r\n private _scene: Scene;\r\n\r\n /**\r\n * Attach the input controls to a specific dom element to get the input from.\r\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n */\r\n public attachControl(noPreventDefault?: boolean): void {\r\n // was there a second variable defined?\r\n // eslint-disable-next-line prefer-rest-params\r\n noPreventDefault = Tools.BackCompatCameraNoPreventDefault(arguments);\r\n\r\n if (this._onCanvasBlurObserver) {\r\n return;\r\n }\r\n\r\n this._scene = this.camera.getScene();\r\n this._engine = this._scene.getEngine();\r\n\r\n this._onCanvasBlurObserver = this._engine.onCanvasBlurObservable.add(() => {\r\n this._keys.length = 0;\r\n });\r\n\r\n this._onKeyboardObserver = this._scene.onKeyboardObservable.add((info) => {\r\n const evt = info.event;\r\n if (!evt.metaKey) {\r\n if (info.type === KeyboardEventTypes.KEYDOWN) {\r\n this._ctrlPressed = evt.ctrlKey;\r\n this._altPressed = evt.altKey;\r\n\r\n if (\r\n this.keysUp.indexOf(evt.keyCode) !== -1 ||\r\n this.keysDown.indexOf(evt.keyCode) !== -1 ||\r\n this.keysLeft.indexOf(evt.keyCode) !== -1 ||\r\n this.keysRight.indexOf(evt.keyCode) !== -1 ||\r\n this.keysReset.indexOf(evt.keyCode) !== -1\r\n ) {\r\n const index = this._keys.indexOf(evt.keyCode);\r\n\r\n if (index === -1) {\r\n this._keys.push(evt.keyCode);\r\n }\r\n\r\n if (evt.preventDefault) {\r\n if (!noPreventDefault) {\r\n evt.preventDefault();\r\n }\r\n }\r\n }\r\n } else {\r\n if (\r\n this.keysUp.indexOf(evt.keyCode) !== -1 ||\r\n this.keysDown.indexOf(evt.keyCode) !== -1 ||\r\n this.keysLeft.indexOf(evt.keyCode) !== -1 ||\r\n this.keysRight.indexOf(evt.keyCode) !== -1 ||\r\n this.keysReset.indexOf(evt.keyCode) !== -1\r\n ) {\r\n const index = this._keys.indexOf(evt.keyCode);\r\n\r\n if (index >= 0) {\r\n this._keys.splice(index, 1);\r\n }\r\n\r\n if (evt.preventDefault) {\r\n if (!noPreventDefault) {\r\n evt.preventDefault();\r\n }\r\n }\r\n }\r\n }\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Detach the current controls from the specified dom element.\r\n */\r\n public detachControl(): void {\r\n if (this._scene) {\r\n if (this._onKeyboardObserver) {\r\n this._scene.onKeyboardObservable.remove(this._onKeyboardObserver);\r\n }\r\n if (this._onCanvasBlurObserver) {\r\n this._engine.onCanvasBlurObservable.remove(this._onCanvasBlurObserver);\r\n }\r\n this._onKeyboardObserver = null;\r\n this._onCanvasBlurObserver = null;\r\n }\r\n\r\n this._keys.length = 0;\r\n }\r\n\r\n /**\r\n * Update the current camera state depending on the inputs that have been used this frame.\r\n * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop.\r\n */\r\n public checkInputs(): void {\r\n if (this._onKeyboardObserver) {\r\n const camera = this.camera;\r\n\r\n for (let index = 0; index < this._keys.length; index++) {\r\n const keyCode = this._keys[index];\r\n if (this.keysLeft.indexOf(keyCode) !== -1) {\r\n if (this._ctrlPressed && this.camera._useCtrlForPanning) {\r\n camera.inertialPanningX -= 1 / this.panningSensibility;\r\n } else {\r\n camera.inertialAlphaOffset -= this.angularSpeed;\r\n }\r\n } else if (this.keysUp.indexOf(keyCode) !== -1) {\r\n if (this._ctrlPressed && this.camera._useCtrlForPanning) {\r\n camera.inertialPanningY += 1 / this.panningSensibility;\r\n } else if (this._altPressed && this.useAltToZoom) {\r\n camera.inertialRadiusOffset += 1 / this.zoomingSensibility;\r\n } else {\r\n camera.inertialBetaOffset -= this.angularSpeed;\r\n }\r\n } else if (this.keysRight.indexOf(keyCode) !== -1) {\r\n if (this._ctrlPressed && this.camera._useCtrlForPanning) {\r\n camera.inertialPanningX += 1 / this.panningSensibility;\r\n } else {\r\n camera.inertialAlphaOffset += this.angularSpeed;\r\n }\r\n } else if (this.keysDown.indexOf(keyCode) !== -1) {\r\n if (this._ctrlPressed && this.camera._useCtrlForPanning) {\r\n camera.inertialPanningY -= 1 / this.panningSensibility;\r\n } else if (this._altPressed && this.useAltToZoom) {\r\n camera.inertialRadiusOffset -= 1 / this.zoomingSensibility;\r\n } else {\r\n camera.inertialBetaOffset += this.angularSpeed;\r\n }\r\n } else if (this.keysReset.indexOf(keyCode) !== -1) {\r\n if (camera.useInputToRestoreState) {\r\n camera.restoreState();\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Gets the class name of the current input.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"ArcRotateCameraKeyboardMoveInput\";\r\n }\r\n\r\n /**\r\n * Get the friendly name associated with the input class.\r\n * @returns the input friendly name\r\n */\r\n public getSimpleName(): string {\r\n return \"keyboard\";\r\n }\r\n}\r\n\r\n(CameraInputTypes)[\"ArcRotateCameraKeyboardMoveInput\"] = ArcRotateCameraKeyboardMoveInput;\r\n","import type { Nullable } from \"../../types\";\r\nimport { serialize } from \"../../Misc/decorators\";\r\nimport type { EventState, Observer } from \"../../Misc/observable\";\r\nimport type { ArcRotateCamera } from \"../../Cameras/arcRotateCamera\";\r\nimport type { ICameraInput } from \"../../Cameras/cameraInputsManager\";\r\nimport { CameraInputTypes } from \"../../Cameras/cameraInputsManager\";\r\nimport type { PointerInfo } from \"../../Events/pointerEvents\";\r\nimport { PointerEventTypes } from \"../../Events/pointerEvents\";\r\nimport { Plane } from \"../../Maths/math.plane\";\r\nimport { Vector3, Matrix, TmpVectors } from \"../../Maths/math.vector\";\r\nimport { Epsilon } from \"../../Maths/math.constants\";\r\nimport type { IWheelEvent } from \"../../Events/deviceInputEvents\";\r\nimport { EventConstants } from \"../../Events/deviceInputEvents\";\r\nimport { Scalar } from \"../../Maths/math.scalar\";\r\nimport { Tools } from \"../../Misc/tools\";\r\n\r\n/**\r\n * Firefox uses a different scheme to report scroll distances to other\r\n * browsers. Rather than use complicated methods to calculate the exact\r\n * multiple we need to apply, let's just cheat and use a constant.\r\n * https://developer.mozilla.org/en-US/docs/Web/API/WheelEvent/deltaMode\r\n * https://stackoverflow.com/questions/20110224/what-is-the-height-of-a-line-in-a-wheel-event-deltamode-dom-delta-line\r\n */\r\nconst ffMultiplier = 40;\r\n\r\n/**\r\n * Manage the mouse wheel inputs to control an arc rotate camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\r\n */\r\nexport class ArcRotateCameraMouseWheelInput implements ICameraInput {\r\n /**\r\n * Defines the camera the input is attached to.\r\n */\r\n public camera: ArcRotateCamera;\r\n\r\n /**\r\n * Gets or Set the mouse wheel precision or how fast is the camera zooming.\r\n */\r\n @serialize()\r\n public wheelPrecision = 3.0;\r\n\r\n /**\r\n * Gets or Set the boolean value that controls whether or not the mouse wheel\r\n * zooms to the location of the mouse pointer or not. The default is false.\r\n */\r\n @serialize()\r\n public zoomToMouseLocation = false;\r\n\r\n /**\r\n * wheelDeltaPercentage will be used instead of wheelPrecision if different from 0.\r\n * It defines the percentage of current camera.radius to use as delta when wheel is used.\r\n */\r\n @serialize()\r\n public wheelDeltaPercentage = 0;\r\n\r\n /**\r\n * If set, this function will be used to set the radius delta that will be added to the current camera radius\r\n */\r\n public customComputeDeltaFromMouseWheel: Nullable<(wheelDelta: number, input: ArcRotateCameraMouseWheelInput, event: IWheelEvent) => number> = null;\r\n\r\n private _wheel: Nullable<(p: PointerInfo, s: EventState) => void>;\r\n private _observer: Nullable>;\r\n private _hitPlane: Nullable;\r\n\r\n protected _computeDeltaFromMouseWheelLegacyEvent(mouseWheelDelta: number, radius: number) {\r\n let delta = 0;\r\n const wheelDelta = mouseWheelDelta * 0.01 * this.wheelDeltaPercentage * radius;\r\n if (mouseWheelDelta > 0) {\r\n delta = wheelDelta / (1.0 + this.wheelDeltaPercentage);\r\n } else {\r\n delta = wheelDelta * (1.0 + this.wheelDeltaPercentage);\r\n }\r\n return delta;\r\n }\r\n\r\n /**\r\n * Attach the input controls to a specific dom element to get the input from.\r\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n */\r\n public attachControl(noPreventDefault?: boolean): void {\r\n noPreventDefault = Tools.BackCompatCameraNoPreventDefault(arguments);\r\n this._wheel = (p) => {\r\n //sanity check - this should be a PointerWheel event.\r\n if (p.type !== PointerEventTypes.POINTERWHEEL) {\r\n return;\r\n }\r\n const event = p.event;\r\n let delta = 0;\r\n const platformScale = event.deltaMode === EventConstants.DOM_DELTA_LINE ? ffMultiplier : 1; // If this happens to be set to DOM_DELTA_LINE, adjust accordingly\r\n\r\n const wheelDelta = -(event.deltaY * platformScale);\r\n\r\n if (this.customComputeDeltaFromMouseWheel) {\r\n delta = this.customComputeDeltaFromMouseWheel(wheelDelta, this, event);\r\n } else {\r\n if (this.wheelDeltaPercentage) {\r\n delta = this._computeDeltaFromMouseWheelLegacyEvent(wheelDelta, this.camera.radius);\r\n\r\n // If zooming in, estimate the target radius and use that to compute the delta for inertia\r\n // this will stop multiple scroll events zooming in from adding too much inertia\r\n if (delta > 0) {\r\n let estimatedTargetRadius = this.camera.radius;\r\n let targetInertia = this.camera.inertialRadiusOffset + delta;\r\n for (let i = 0; i < 20 && Math.abs(targetInertia) > 0.001; i++) {\r\n estimatedTargetRadius -= targetInertia;\r\n targetInertia *= this.camera.inertia;\r\n }\r\n estimatedTargetRadius = Scalar.Clamp(estimatedTargetRadius, 0, Number.MAX_VALUE);\r\n delta = this._computeDeltaFromMouseWheelLegacyEvent(wheelDelta, estimatedTargetRadius);\r\n }\r\n } else {\r\n delta = wheelDelta / (this.wheelPrecision * 40);\r\n }\r\n }\r\n\r\n if (delta) {\r\n if (this.zoomToMouseLocation) {\r\n // If we are zooming to the mouse location, then we need to get the hit plane at the start of the zoom gesture if it doesn't exist\r\n // The hit plane is normally calculated after the first motion and each time there's motion so if we don't do this first,\r\n // the first zoom will be to the center of the screen\r\n if (!this._hitPlane) {\r\n this._updateHitPlane();\r\n }\r\n\r\n this._zoomToMouse(delta);\r\n } else {\r\n this.camera.inertialRadiusOffset += delta;\r\n }\r\n }\r\n\r\n if (event.preventDefault) {\r\n if (!noPreventDefault) {\r\n event.preventDefault();\r\n }\r\n }\r\n };\r\n\r\n this._observer = this.camera.getScene()._inputManager._addCameraPointerObserver(this._wheel, PointerEventTypes.POINTERWHEEL);\r\n\r\n if (this.zoomToMouseLocation) {\r\n this._inertialPanning.setAll(0);\r\n }\r\n }\r\n\r\n /**\r\n * Detach the current controls from the specified dom element.\r\n */\r\n public detachControl(): void {\r\n if (this._observer) {\r\n this.camera.getScene()._inputManager._removeCameraPointerObserver(this._observer);\r\n this._observer = null;\r\n this._wheel = null;\r\n }\r\n }\r\n\r\n /**\r\n * Update the current camera state depending on the inputs that have been used this frame.\r\n * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop.\r\n */\r\n public checkInputs(): void {\r\n if (!this.zoomToMouseLocation) {\r\n return;\r\n }\r\n\r\n const camera = this.camera;\r\n const motion = 0.0 + camera.inertialAlphaOffset + camera.inertialBetaOffset + camera.inertialRadiusOffset;\r\n if (motion) {\r\n // if zooming is still happening as a result of inertia, then we also need to update\r\n // the hit plane.\r\n this._updateHitPlane();\r\n\r\n // Note we cannot use arcRotateCamera.inertialPlanning here because arcRotateCamera panning\r\n // uses a different panningInertia which could cause this panning to get out of sync with\r\n // the zooming, and for this to work they must be exactly in sync.\r\n camera.target.addInPlace(this._inertialPanning);\r\n this._inertialPanning.scaleInPlace(camera.inertia);\r\n this._zeroIfClose(this._inertialPanning);\r\n }\r\n }\r\n\r\n /**\r\n * Gets the class name of the current input.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"ArcRotateCameraMouseWheelInput\";\r\n }\r\n\r\n /**\r\n * Get the friendly name associated with the input class.\r\n * @returns the input friendly name\r\n */\r\n public getSimpleName(): string {\r\n return \"mousewheel\";\r\n }\r\n\r\n private _updateHitPlane() {\r\n const camera = this.camera;\r\n const direction = camera.target.subtract(camera.position);\r\n this._hitPlane = Plane.FromPositionAndNormal(camera.target, direction);\r\n }\r\n\r\n // Get position on the hit plane\r\n private _getPosition(): Vector3 {\r\n const camera = this.camera;\r\n const scene = camera.getScene();\r\n\r\n // since the _hitPlane is always updated to be orthogonal to the camera position vector\r\n // we don't have to worry about this ray shooting off to infinity. This ray creates\r\n // a vector defining where we want to zoom to.\r\n const ray = scene.createPickingRay(scene.pointerX, scene.pointerY, Matrix.Identity(), camera, false);\r\n // Since the camera is the origin of the picking ray, we need to offset it by the camera's offset manually\r\n ray.origin.x -= camera.targetScreenOffset.x;\r\n ray.origin.y -= camera.targetScreenOffset.y;\r\n let distance = 0;\r\n if (this._hitPlane) {\r\n distance = ray.intersectsPlane(this._hitPlane) ?? 0;\r\n }\r\n\r\n // not using this ray again, so modifying its vectors here is fine\r\n return ray.origin.addInPlace(ray.direction.scaleInPlace(distance));\r\n }\r\n\r\n private _inertialPanning: Vector3 = Vector3.Zero();\r\n\r\n private _zoomToMouse(delta: number) {\r\n const camera = this.camera;\r\n const inertiaComp = 1 - camera.inertia;\r\n if (camera.lowerRadiusLimit) {\r\n const lowerLimit = camera.lowerRadiusLimit ?? 0;\r\n if (camera.radius - (camera.inertialRadiusOffset + delta) / inertiaComp < lowerLimit) {\r\n delta = (camera.radius - lowerLimit) * inertiaComp - camera.inertialRadiusOffset;\r\n }\r\n }\r\n if (camera.upperRadiusLimit) {\r\n const upperLimit = camera.upperRadiusLimit ?? 0;\r\n if (camera.radius - (camera.inertialRadiusOffset + delta) / inertiaComp > upperLimit) {\r\n delta = (camera.radius - upperLimit) * inertiaComp - camera.inertialRadiusOffset;\r\n }\r\n }\r\n\r\n const zoomDistance = delta / inertiaComp;\r\n const ratio = zoomDistance / camera.radius;\r\n const vec = this._getPosition();\r\n\r\n // Now this vector tells us how much we also need to pan the camera\r\n // so the targeted mouse location becomes the center of zooming.\r\n\r\n const directionToZoomLocation = TmpVectors.Vector3[6];\r\n vec.subtractToRef(camera.target, directionToZoomLocation);\r\n directionToZoomLocation.scaleInPlace(ratio);\r\n directionToZoomLocation.scaleInPlace(inertiaComp);\r\n this._inertialPanning.addInPlace(directionToZoomLocation);\r\n\r\n camera.inertialRadiusOffset += delta;\r\n }\r\n\r\n // Sets x y or z of passed in vector to zero if less than Epsilon.\r\n private _zeroIfClose(vec: Vector3) {\r\n if (Math.abs(vec.x) < Epsilon) {\r\n vec.x = 0;\r\n }\r\n if (Math.abs(vec.y) < Epsilon) {\r\n vec.y = 0;\r\n }\r\n if (Math.abs(vec.z) < Epsilon) {\r\n vec.z = 0;\r\n }\r\n }\r\n}\r\n\r\n(CameraInputTypes)[\"ArcRotateCameraMouseWheelInput\"] = ArcRotateCameraMouseWheelInput;\r\n","import type { ArcRotateCamera } from \"./arcRotateCamera\";\r\nimport { ArcRotateCameraPointersInput } from \"../Cameras/Inputs/arcRotateCameraPointersInput\";\r\nimport { ArcRotateCameraKeyboardMoveInput } from \"../Cameras/Inputs/arcRotateCameraKeyboardMoveInput\";\r\nimport { ArcRotateCameraMouseWheelInput } from \"../Cameras/Inputs/arcRotateCameraMouseWheelInput\";\r\nimport { CameraInputsManager } from \"../Cameras/cameraInputsManager\";\r\n\r\n/**\r\n * Default Inputs manager for the ArcRotateCamera.\r\n * It groups all the default supported inputs for ease of use.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\r\n */\r\nexport class ArcRotateCameraInputsManager extends CameraInputsManager {\r\n /**\r\n * Instantiates a new ArcRotateCameraInputsManager.\r\n * @param camera Defines the camera the inputs belong to\r\n */\r\n constructor(camera: ArcRotateCamera) {\r\n super(camera);\r\n }\r\n\r\n /**\r\n * Add mouse wheel input support to the input manager.\r\n * @returns the current input manager\r\n */\r\n public addMouseWheel(): ArcRotateCameraInputsManager {\r\n this.add(new ArcRotateCameraMouseWheelInput());\r\n return this;\r\n }\r\n\r\n /**\r\n * Add pointers input support to the input manager.\r\n * @returns the current input manager\r\n */\r\n public addPointers(): ArcRotateCameraInputsManager {\r\n this.add(new ArcRotateCameraPointersInput());\r\n return this;\r\n }\r\n\r\n /**\r\n * Add keyboard input support to the input manager.\r\n * @returns the current input manager\r\n */\r\n public addKeyboard(): ArcRotateCameraInputsManager {\r\n this.add(new ArcRotateCameraKeyboardMoveInput());\r\n return this;\r\n }\r\n}\r\n","import { serialize, serializeAsVector3, serializeAsMeshReference, serializeAsVector2 } from \"../Misc/decorators\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Matrix, Vector3, Vector2 } from \"../Maths/math.vector\";\r\nimport { Node } from \"../node\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport { Mesh } from \"../Meshes/mesh\";\r\nimport { AutoRotationBehavior } from \"../Behaviors/Cameras/autoRotationBehavior\";\r\nimport { BouncingBehavior } from \"../Behaviors/Cameras/bouncingBehavior\";\r\nimport { FramingBehavior } from \"../Behaviors/Cameras/framingBehavior\";\r\nimport { Camera } from \"./camera\";\r\nimport { TargetCamera } from \"./targetCamera\";\r\nimport type { ArcRotateCameraPointersInput } from \"../Cameras/Inputs/arcRotateCameraPointersInput\";\r\nimport type { ArcRotateCameraKeyboardMoveInput } from \"../Cameras/Inputs/arcRotateCameraKeyboardMoveInput\";\r\nimport type { ArcRotateCameraMouseWheelInput } from \"../Cameras/Inputs/arcRotateCameraMouseWheelInput\";\r\nimport { ArcRotateCameraInputsManager } from \"../Cameras/arcRotateCameraInputsManager\";\r\nimport { Epsilon } from \"../Maths/math.constants\";\r\nimport { Tools } from \"../Misc/tools\";\r\n\r\nimport type { Collider } from \"../Collisions/collider\";\r\n\r\nNode.AddNodeConstructor(\"ArcRotateCamera\", (name, scene) => {\r\n return () => new ArcRotateCamera(name, 0, 0, 1.0, Vector3.Zero(), scene);\r\n});\r\n\r\n/**\r\n * This represents an orbital type of camera.\r\n *\r\n * This camera always points towards a given target position and can be rotated around that target with the target as the centre of rotation. It can be controlled with cursors and mouse, or with touch events.\r\n * Think of this camera as one orbiting its target position, or more imaginatively as a spy satellite orbiting the earth. Its position relative to the target (earth) can be set by three parameters, alpha (radians) the longitudinal rotation, beta (radians) the latitudinal rotation and radius the distance from the target position.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#arc-rotate-camera\r\n */\r\nexport class ArcRotateCamera extends TargetCamera {\r\n /**\r\n * Defines the rotation angle of the camera along the longitudinal axis.\r\n */\r\n @serialize()\r\n public alpha: number;\r\n\r\n /**\r\n * Defines the rotation angle of the camera along the latitudinal axis.\r\n */\r\n @serialize()\r\n public beta: number;\r\n\r\n /**\r\n * Defines the radius of the camera from it s target point.\r\n */\r\n @serialize()\r\n public radius: number;\r\n\r\n /**\r\n * Defines an override value to use as the parameter to setTarget.\r\n * This allows the parameter to be specified when animating the target (e.g. using FramingBehavior).\r\n */\r\n @serialize()\r\n public overrideCloneAlphaBetaRadius: Nullable;\r\n\r\n @serializeAsVector3(\"target\")\r\n protected _target: Vector3;\r\n @serializeAsMeshReference(\"targetHost\")\r\n protected _targetHost: Nullable;\r\n\r\n /**\r\n * Defines the target point of the camera.\r\n * The camera looks towards it from the radius distance.\r\n */\r\n public get target(): Vector3 {\r\n return this._target;\r\n }\r\n public set target(value: Vector3) {\r\n this.setTarget(value);\r\n }\r\n\r\n /**\r\n * Defines the target mesh of the camera.\r\n * The camera looks towards it from the radius distance.\r\n * Please note that setting a target host will disable panning.\r\n */\r\n public get targetHost(): Nullable {\r\n return this._targetHost;\r\n }\r\n public set targetHost(value: Nullable) {\r\n if (value) {\r\n this.setTarget(value);\r\n }\r\n }\r\n\r\n /**\r\n * Return the current target position of the camera. This value is expressed in local space.\r\n * @returns the target position\r\n */\r\n public getTarget(): Vector3 {\r\n return this.target;\r\n }\r\n\r\n /**\r\n * Define the current local position of the camera in the scene\r\n */\r\n public get position(): Vector3 {\r\n return this._position;\r\n }\r\n\r\n public set position(newPosition: Vector3) {\r\n this.setPosition(newPosition);\r\n }\r\n\r\n protected _upToYMatrix: Matrix;\r\n protected _yToUpMatrix: Matrix;\r\n\r\n /**\r\n * The vector the camera should consider as up. (default is Vector3(0, 1, 0) as returned by Vector3.Up())\r\n * Setting this will copy the given vector to the camera's upVector, and set rotation matrices to and from Y up.\r\n * DO NOT set the up vector using copyFrom or copyFromFloats, as this bypasses setting the above matrices.\r\n */\r\n set upVector(vec: Vector3) {\r\n if (!this._upToYMatrix) {\r\n this._yToUpMatrix = new Matrix();\r\n this._upToYMatrix = new Matrix();\r\n\r\n this._upVector = Vector3.Zero();\r\n }\r\n\r\n vec.normalize();\r\n this._upVector.copyFrom(vec);\r\n this.setMatUp();\r\n }\r\n\r\n get upVector() {\r\n return this._upVector;\r\n }\r\n\r\n /**\r\n * Sets the Y-up to camera up-vector rotation matrix, and the up-vector to Y-up rotation matrix.\r\n */\r\n public setMatUp() {\r\n // from y-up to custom-up (used in _getViewMatrix)\r\n Matrix.RotationAlignToRef(Vector3.UpReadOnly, this._upVector, this._yToUpMatrix);\r\n\r\n // from custom-up to y-up (used in rebuildAnglesAndRadius)\r\n Matrix.RotationAlignToRef(this._upVector, Vector3.UpReadOnly, this._upToYMatrix);\r\n }\r\n\r\n /**\r\n * Current inertia value on the longitudinal axis.\r\n * The bigger this number the longer it will take for the camera to stop.\r\n */\r\n @serialize()\r\n public inertialAlphaOffset = 0;\r\n\r\n /**\r\n * Current inertia value on the latitudinal axis.\r\n * The bigger this number the longer it will take for the camera to stop.\r\n */\r\n @serialize()\r\n public inertialBetaOffset = 0;\r\n\r\n /**\r\n * Current inertia value on the radius axis.\r\n * The bigger this number the longer it will take for the camera to stop.\r\n */\r\n @serialize()\r\n public inertialRadiusOffset = 0;\r\n\r\n /**\r\n * Minimum allowed angle on the longitudinal axis.\r\n * This can help limiting how the Camera is able to move in the scene.\r\n */\r\n @serialize()\r\n public lowerAlphaLimit: Nullable = null;\r\n\r\n /**\r\n * Maximum allowed angle on the longitudinal axis.\r\n * This can help limiting how the Camera is able to move in the scene.\r\n */\r\n @serialize()\r\n public upperAlphaLimit: Nullable = null;\r\n\r\n /**\r\n * Minimum allowed angle on the latitudinal axis.\r\n * This can help limiting how the Camera is able to move in the scene.\r\n */\r\n @serialize()\r\n public lowerBetaLimit: Nullable = 0.01;\r\n\r\n /**\r\n * Maximum allowed angle on the latitudinal axis.\r\n * This can help limiting how the Camera is able to move in the scene.\r\n */\r\n @serialize()\r\n public upperBetaLimit: Nullable = Math.PI - 0.01;\r\n\r\n /**\r\n * Minimum allowed distance of the camera to the target (The camera can not get closer).\r\n * This can help limiting how the Camera is able to move in the scene.\r\n */\r\n @serialize()\r\n public lowerRadiusLimit: Nullable = null;\r\n\r\n /**\r\n * Maximum allowed distance of the camera to the target (The camera can not get further).\r\n * This can help limiting how the Camera is able to move in the scene.\r\n */\r\n @serialize()\r\n public upperRadiusLimit: Nullable = null;\r\n\r\n /**\r\n * Defines the current inertia value used during panning of the camera along the X axis.\r\n */\r\n @serialize()\r\n public inertialPanningX: number = 0;\r\n\r\n /**\r\n * Defines the current inertia value used during panning of the camera along the Y axis.\r\n */\r\n @serialize()\r\n public inertialPanningY: number = 0;\r\n\r\n /**\r\n * Defines the distance used to consider the camera in pan mode vs pinch/zoom.\r\n * Basically if your fingers moves away from more than this distance you will be considered\r\n * in pinch mode.\r\n */\r\n @serialize()\r\n public pinchToPanMaxDistance: number = 20;\r\n\r\n /**\r\n * Defines the maximum distance the camera can pan.\r\n * This could help keeping the camera always in your scene.\r\n */\r\n @serialize()\r\n public panningDistanceLimit: Nullable = null;\r\n\r\n /**\r\n * Defines the target of the camera before panning.\r\n */\r\n @serializeAsVector3()\r\n public panningOriginTarget: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * Defines the value of the inertia used during panning.\r\n * 0 would mean stop inertia and one would mean no deceleration at all.\r\n */\r\n @serialize()\r\n public panningInertia = 0.9;\r\n\r\n //-- begin properties for backward compatibility for inputs\r\n\r\n /**\r\n * Gets or Set the pointer angular sensibility along the X axis or how fast is the camera rotating.\r\n */\r\n public get angularSensibilityX(): number {\r\n const pointers = this.inputs.attached[\"pointers\"];\r\n if (pointers) {\r\n return pointers.angularSensibilityX;\r\n }\r\n\r\n return 0;\r\n }\r\n\r\n public set angularSensibilityX(value: number) {\r\n const pointers = this.inputs.attached[\"pointers\"];\r\n if (pointers) {\r\n pointers.angularSensibilityX = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the pointer angular sensibility along the Y axis or how fast is the camera rotating.\r\n */\r\n public get angularSensibilityY(): number {\r\n const pointers = this.inputs.attached[\"pointers\"];\r\n if (pointers) {\r\n return pointers.angularSensibilityY;\r\n }\r\n\r\n return 0;\r\n }\r\n\r\n public set angularSensibilityY(value: number) {\r\n const pointers = this.inputs.attached[\"pointers\"];\r\n if (pointers) {\r\n pointers.angularSensibilityY = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the pointer pinch precision or how fast is the camera zooming.\r\n */\r\n public get pinchPrecision(): number {\r\n const pointers = this.inputs.attached[\"pointers\"];\r\n if (pointers) {\r\n return pointers.pinchPrecision;\r\n }\r\n\r\n return 0;\r\n }\r\n\r\n public set pinchPrecision(value: number) {\r\n const pointers = this.inputs.attached[\"pointers\"];\r\n if (pointers) {\r\n pointers.pinchPrecision = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the pointer pinch delta percentage or how fast is the camera zooming.\r\n * It will be used instead of pinchDeltaPrecision if different from 0.\r\n * It defines the percentage of current camera.radius to use as delta when pinch zoom is used.\r\n */\r\n public get pinchDeltaPercentage(): number {\r\n const pointers = this.inputs.attached[\"pointers\"];\r\n if (pointers) {\r\n return pointers.pinchDeltaPercentage;\r\n }\r\n\r\n return 0;\r\n }\r\n\r\n public set pinchDeltaPercentage(value: number) {\r\n const pointers = this.inputs.attached[\"pointers\"];\r\n if (pointers) {\r\n pointers.pinchDeltaPercentage = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the pointer use natural pinch zoom to override the pinch precision\r\n * and pinch delta percentage.\r\n * When useNaturalPinchZoom is true, multi touch zoom will zoom in such\r\n * that any object in the plane at the camera's target point will scale\r\n * perfectly with finger motion.\r\n */\r\n public get useNaturalPinchZoom(): boolean {\r\n const pointers = this.inputs.attached[\"pointers\"];\r\n if (pointers) {\r\n return pointers.useNaturalPinchZoom;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n public set useNaturalPinchZoom(value: boolean) {\r\n const pointers = this.inputs.attached[\"pointers\"];\r\n if (pointers) {\r\n pointers.useNaturalPinchZoom = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the pointer panning sensibility or how fast is the camera moving.\r\n */\r\n public get panningSensibility(): number {\r\n const pointers = this.inputs.attached[\"pointers\"];\r\n if (pointers) {\r\n return pointers.panningSensibility;\r\n }\r\n\r\n return 0;\r\n }\r\n\r\n public set panningSensibility(value: number) {\r\n const pointers = this.inputs.attached[\"pointers\"];\r\n if (pointers) {\r\n pointers.panningSensibility = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control beta angle in a positive direction.\r\n */\r\n public get keysUp(): number[] {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n return keyboard.keysUp;\r\n }\r\n\r\n return [];\r\n }\r\n\r\n public set keysUp(value: number[]) {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n keyboard.keysUp = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control beta angle in a negative direction.\r\n */\r\n public get keysDown(): number[] {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n return keyboard.keysDown;\r\n }\r\n\r\n return [];\r\n }\r\n\r\n public set keysDown(value: number[]) {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n keyboard.keysDown = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control alpha angle in a negative direction.\r\n */\r\n public get keysLeft(): number[] {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n return keyboard.keysLeft;\r\n }\r\n\r\n return [];\r\n }\r\n\r\n public set keysLeft(value: number[]) {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n keyboard.keysLeft = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control alpha angle in a positive direction.\r\n */\r\n public get keysRight(): number[] {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n return keyboard.keysRight;\r\n }\r\n\r\n return [];\r\n }\r\n\r\n public set keysRight(value: number[]) {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n keyboard.keysRight = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the mouse wheel precision or how fast is the camera zooming.\r\n */\r\n public get wheelPrecision(): number {\r\n const mousewheel = this.inputs.attached[\"mousewheel\"];\r\n if (mousewheel) {\r\n return mousewheel.wheelPrecision;\r\n }\r\n\r\n return 0;\r\n }\r\n\r\n public set wheelPrecision(value: number) {\r\n const mousewheel = this.inputs.attached[\"mousewheel\"];\r\n if (mousewheel) {\r\n mousewheel.wheelPrecision = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the boolean value that controls whether or not the mouse wheel\r\n * zooms to the location of the mouse pointer or not. The default is false.\r\n */\r\n @serialize()\r\n public get zoomToMouseLocation(): boolean {\r\n const mousewheel = this.inputs.attached[\"mousewheel\"];\r\n if (mousewheel) {\r\n return mousewheel.zoomToMouseLocation;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n public set zoomToMouseLocation(value: boolean) {\r\n const mousewheel = this.inputs.attached[\"mousewheel\"];\r\n if (mousewheel) {\r\n mousewheel.zoomToMouseLocation = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the mouse wheel delta percentage or how fast is the camera zooming.\r\n * It will be used instead of pinchDeltaPrecision if different from 0.\r\n * It defines the percentage of current camera.radius to use as delta when pinch zoom is used.\r\n */\r\n public get wheelDeltaPercentage(): number {\r\n const mousewheel = this.inputs.attached[\"mousewheel\"];\r\n if (mousewheel) {\r\n return mousewheel.wheelDeltaPercentage;\r\n }\r\n\r\n return 0;\r\n }\r\n\r\n public set wheelDeltaPercentage(value: number) {\r\n const mousewheel = this.inputs.attached[\"mousewheel\"];\r\n if (mousewheel) {\r\n mousewheel.wheelDeltaPercentage = value;\r\n }\r\n }\r\n\r\n //-- end properties for backward compatibility for inputs\r\n\r\n /**\r\n * Defines how much the radius should be scaled while zooming on a particular mesh (through the zoomOn function)\r\n */\r\n @serialize()\r\n public zoomOnFactor = 1;\r\n\r\n /**\r\n * Defines a screen offset for the camera position.\r\n */\r\n @serializeAsVector2()\r\n public targetScreenOffset = Vector2.Zero();\r\n\r\n /**\r\n * Allows the camera to be completely reversed.\r\n * If false the camera can not arrive upside down.\r\n */\r\n @serialize()\r\n public allowUpsideDown = true;\r\n\r\n /**\r\n * Define if double tap/click is used to restore the previously saved state of the camera.\r\n */\r\n @serialize()\r\n public useInputToRestoreState = true;\r\n\r\n /** @internal */\r\n public _viewMatrix = new Matrix();\r\n /** @internal */\r\n public _useCtrlForPanning: boolean;\r\n /** @internal */\r\n public _panningMouseButton: number;\r\n\r\n /**\r\n * Defines the input associated to the camera.\r\n */\r\n public inputs: ArcRotateCameraInputsManager;\r\n\r\n /** @internal */\r\n public _reset: () => void;\r\n\r\n /**\r\n * Defines the allowed panning axis.\r\n */\r\n public panningAxis: Vector3 = new Vector3(1, 1, 0);\r\n protected _transformedDirection: Vector3 = new Vector3();\r\n\r\n /**\r\n * Defines if camera will eliminate transform on y axis.\r\n */\r\n public mapPanning: boolean = false;\r\n\r\n // Behaviors\r\n private _bouncingBehavior: Nullable;\r\n\r\n /**\r\n * Gets the bouncing behavior of the camera if it has been enabled.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#bouncing-behavior\r\n */\r\n public get bouncingBehavior(): Nullable {\r\n return this._bouncingBehavior;\r\n }\r\n\r\n /**\r\n * Defines if the bouncing behavior of the camera is enabled on the camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#bouncing-behavior\r\n */\r\n public get useBouncingBehavior(): boolean {\r\n return this._bouncingBehavior != null;\r\n }\r\n\r\n public set useBouncingBehavior(value: boolean) {\r\n if (value === this.useBouncingBehavior) {\r\n return;\r\n }\r\n\r\n if (value) {\r\n this._bouncingBehavior = new BouncingBehavior();\r\n this.addBehavior(this._bouncingBehavior);\r\n } else if (this._bouncingBehavior) {\r\n this.removeBehavior(this._bouncingBehavior);\r\n this._bouncingBehavior = null;\r\n }\r\n }\r\n\r\n private _framingBehavior: Nullable;\r\n\r\n /**\r\n * Gets the framing behavior of the camera if it has been enabled.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#framing-behavior\r\n */\r\n public get framingBehavior(): Nullable {\r\n return this._framingBehavior;\r\n }\r\n\r\n /**\r\n * Defines if the framing behavior of the camera is enabled on the camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#framing-behavior\r\n */\r\n public get useFramingBehavior(): boolean {\r\n return this._framingBehavior != null;\r\n }\r\n\r\n public set useFramingBehavior(value: boolean) {\r\n if (value === this.useFramingBehavior) {\r\n return;\r\n }\r\n\r\n if (value) {\r\n this._framingBehavior = new FramingBehavior();\r\n this.addBehavior(this._framingBehavior);\r\n } else if (this._framingBehavior) {\r\n this.removeBehavior(this._framingBehavior);\r\n this._framingBehavior = null;\r\n }\r\n }\r\n\r\n private _autoRotationBehavior: Nullable;\r\n\r\n /**\r\n * Gets the auto rotation behavior of the camera if it has been enabled.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#autorotation-behavior\r\n */\r\n public get autoRotationBehavior(): Nullable {\r\n return this._autoRotationBehavior;\r\n }\r\n\r\n /**\r\n * Defines if the auto rotation behavior of the camera is enabled on the camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/behaviors/cameraBehaviors#autorotation-behavior\r\n */\r\n public get useAutoRotationBehavior(): boolean {\r\n return this._autoRotationBehavior != null;\r\n }\r\n\r\n public set useAutoRotationBehavior(value: boolean) {\r\n if (value === this.useAutoRotationBehavior) {\r\n return;\r\n }\r\n\r\n if (value) {\r\n this._autoRotationBehavior = new AutoRotationBehavior();\r\n this.addBehavior(this._autoRotationBehavior);\r\n } else if (this._autoRotationBehavior) {\r\n this.removeBehavior(this._autoRotationBehavior);\r\n this._autoRotationBehavior = null;\r\n }\r\n }\r\n\r\n /**\r\n * Observable triggered when the mesh target has been changed on the camera.\r\n */\r\n public onMeshTargetChangedObservable = new Observable>();\r\n\r\n /**\r\n * Event raised when the camera is colliding with a mesh.\r\n */\r\n public onCollide: (collidedMesh: AbstractMesh) => void;\r\n\r\n /**\r\n * Defines whether the camera should check collision with the objects oh the scene.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions#how-can-i-do-this-\r\n */\r\n public checkCollisions = false;\r\n\r\n /**\r\n * Defines the collision radius of the camera.\r\n * This simulates a sphere around the camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions#arcrotatecamera\r\n */\r\n public collisionRadius = new Vector3(0.5, 0.5, 0.5);\r\n\r\n protected _collider: Collider;\r\n protected _previousPosition = Vector3.Zero();\r\n protected _collisionVelocity = Vector3.Zero();\r\n protected _newPosition = Vector3.Zero();\r\n protected _previousAlpha: number;\r\n protected _previousBeta: number;\r\n protected _previousRadius: number;\r\n //due to async collision inspection\r\n protected _collisionTriggered: boolean;\r\n\r\n protected _targetBoundingCenter: Nullable;\r\n\r\n private _computationVector: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * Instantiates a new ArcRotateCamera in a given scene\r\n * @param name Defines the name of the camera\r\n * @param alpha Defines the camera rotation along the longitudinal axis\r\n * @param beta Defines the camera rotation along the latitudinal axis\r\n * @param radius Defines the camera distance from its target\r\n * @param target Defines the camera target\r\n * @param scene Defines the scene the camera belongs to\r\n * @param setActiveOnSceneIfNoneActive Defines whether the camera should be marked as active if not other active cameras have been defined\r\n */\r\n constructor(name: string, alpha: number, beta: number, radius: number, target: Vector3, scene?: Scene, setActiveOnSceneIfNoneActive = true) {\r\n super(name, Vector3.Zero(), scene, setActiveOnSceneIfNoneActive);\r\n\r\n this._target = Vector3.Zero();\r\n if (target) {\r\n this.setTarget(target);\r\n }\r\n\r\n this.alpha = alpha;\r\n this.beta = beta;\r\n this.radius = radius;\r\n\r\n this.getViewMatrix();\r\n this.inputs = new ArcRotateCameraInputsManager(this);\r\n this.inputs.addKeyboard().addMouseWheel().addPointers();\r\n }\r\n\r\n // Cache\r\n /** @internal */\r\n public _initCache(): void {\r\n super._initCache();\r\n this._cache._target = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\r\n this._cache.alpha = undefined;\r\n this._cache.beta = undefined;\r\n this._cache.radius = undefined;\r\n this._cache.targetScreenOffset = Vector2.Zero();\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _updateCache(ignoreParentClass?: boolean): void {\r\n if (!ignoreParentClass) {\r\n super._updateCache();\r\n }\r\n\r\n this._cache._target.copyFrom(this._getTargetPosition());\r\n this._cache.alpha = this.alpha;\r\n this._cache.beta = this.beta;\r\n this._cache.radius = this.radius;\r\n this._cache.targetScreenOffset.copyFrom(this.targetScreenOffset);\r\n }\r\n\r\n protected _getTargetPosition(): Vector3 {\r\n if (this._targetHost && this._targetHost.getAbsolutePosition) {\r\n const pos: Vector3 = this._targetHost.getAbsolutePosition();\r\n if (this._targetBoundingCenter) {\r\n pos.addToRef(this._targetBoundingCenter, this._target);\r\n } else {\r\n this._target.copyFrom(pos);\r\n }\r\n }\r\n\r\n const lockedTargetPosition = this._getLockedTargetPosition();\r\n\r\n if (lockedTargetPosition) {\r\n return lockedTargetPosition;\r\n }\r\n\r\n return this._target;\r\n }\r\n\r\n private _storedAlpha: number;\r\n private _storedBeta: number;\r\n private _storedRadius: number;\r\n private _storedTarget: Vector3;\r\n private _storedTargetScreenOffset: Vector2;\r\n\r\n /**\r\n * Stores the current state of the camera (alpha, beta, radius and target)\r\n * @returns the camera itself\r\n */\r\n public storeState(): Camera {\r\n this._storedAlpha = this.alpha;\r\n this._storedBeta = this.beta;\r\n this._storedRadius = this.radius;\r\n this._storedTarget = this._getTargetPosition().clone();\r\n this._storedTargetScreenOffset = this.targetScreenOffset.clone();\r\n\r\n return super.storeState();\r\n }\r\n\r\n /**\r\n * @internal\r\n * Restored camera state. You must call storeState() first\r\n */\r\n public _restoreStateValues(): boolean {\r\n if (!super._restoreStateValues()) {\r\n return false;\r\n }\r\n\r\n this.setTarget(this._storedTarget.clone());\r\n this.alpha = this._storedAlpha;\r\n this.beta = this._storedBeta;\r\n this.radius = this._storedRadius;\r\n this.targetScreenOffset = this._storedTargetScreenOffset.clone();\r\n\r\n this.inertialAlphaOffset = 0;\r\n this.inertialBetaOffset = 0;\r\n this.inertialRadiusOffset = 0;\r\n this.inertialPanningX = 0;\r\n this.inertialPanningY = 0;\r\n\r\n return true;\r\n }\r\n\r\n // Synchronized\r\n /** @internal */\r\n public _isSynchronizedViewMatrix(): boolean {\r\n if (!super._isSynchronizedViewMatrix()) {\r\n return false;\r\n }\r\n\r\n return (\r\n this._cache._target.equals(this._getTargetPosition()) &&\r\n this._cache.alpha === this.alpha &&\r\n this._cache.beta === this.beta &&\r\n this._cache.radius === this.radius &&\r\n this._cache.targetScreenOffset.equals(this.targetScreenOffset)\r\n );\r\n }\r\n\r\n /**\r\n * Attach the input controls to a specific dom element to get the input from.\r\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n */\r\n public attachControl(noPreventDefault?: boolean): void;\r\n /**\r\n * Attach the input controls to a specific dom element to get the input from.\r\n * @param ignored defines an ignored parameter kept for backward compatibility.\r\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n */\r\n public attachControl(ignored: any, noPreventDefault?: boolean): void;\r\n /**\r\n * Attached controls to the current camera.\r\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n * @param useCtrlForPanning Defines whether ctrl is used for panning within the controls\r\n */\r\n public attachControl(noPreventDefault: boolean, useCtrlForPanning: boolean): void;\r\n /**\r\n * Attached controls to the current camera.\r\n * @param ignored defines an ignored parameter kept for backward compatibility.\r\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n * @param useCtrlForPanning Defines whether ctrl is used for panning within the controls\r\n */\r\n public attachControl(ignored: any, noPreventDefault: boolean, useCtrlForPanning: boolean): void;\r\n /**\r\n * Attached controls to the current camera.\r\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n * @param useCtrlForPanning Defines whether ctrl is used for panning within the controls\r\n * @param panningMouseButton Defines whether panning is allowed through mouse click button\r\n */\r\n public attachControl(noPreventDefault: boolean, useCtrlForPanning: boolean, panningMouseButton: number): void;\r\n /**\r\n * Attached controls to the current camera.\r\n * @param ignored defines an ignored parameter kept for backward compatibility.\r\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n * @param useCtrlForPanning Defines whether ctrl is used for panning within the controls\r\n * @param panningMouseButton Defines whether panning is allowed through mouse click button\r\n */\r\n public attachControl(ignored: any, noPreventDefault?: boolean, useCtrlForPanning: boolean | number = true, panningMouseButton: number = 2): void {\r\n // eslint-disable-next-line prefer-rest-params\r\n const args = arguments;\r\n\r\n noPreventDefault = Tools.BackCompatCameraNoPreventDefault(args);\r\n this._useCtrlForPanning = useCtrlForPanning as boolean;\r\n this._panningMouseButton = panningMouseButton;\r\n // backwards compatibility\r\n if (typeof args[0] === \"boolean\") {\r\n if (args.length > 1) {\r\n this._useCtrlForPanning = args[1];\r\n }\r\n if (args.length > 2) {\r\n this._panningMouseButton = args[2];\r\n }\r\n }\r\n\r\n this.inputs.attachElement(noPreventDefault);\r\n\r\n this._reset = () => {\r\n this.inertialAlphaOffset = 0;\r\n this.inertialBetaOffset = 0;\r\n this.inertialRadiusOffset = 0;\r\n this.inertialPanningX = 0;\r\n this.inertialPanningY = 0;\r\n };\r\n }\r\n\r\n /**\r\n * Detach the current controls from the specified dom element.\r\n */\r\n public detachControl(): void {\r\n this.inputs.detachElement();\r\n\r\n if (this._reset) {\r\n this._reset();\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _checkInputs(): void {\r\n //if (async) collision inspection was triggered, don't update the camera's position - until the collision callback was called.\r\n if (this._collisionTriggered) {\r\n return;\r\n }\r\n\r\n this.inputs.checkInputs();\r\n // Inertia\r\n if (this.inertialAlphaOffset !== 0 || this.inertialBetaOffset !== 0 || this.inertialRadiusOffset !== 0) {\r\n const directionModifier = this.invertRotation ? -1 : 1;\r\n let inertialAlphaOffset = this.inertialAlphaOffset;\r\n if (this.beta <= 0) {\r\n inertialAlphaOffset *= -1;\r\n }\r\n if (this.getScene().useRightHandedSystem) {\r\n inertialAlphaOffset *= -1;\r\n }\r\n if (this.parent && this.parent._getWorldMatrixDeterminant() < 0) {\r\n inertialAlphaOffset *= -1;\r\n }\r\n this.alpha += inertialAlphaOffset * directionModifier;\r\n\r\n this.beta += this.inertialBetaOffset * directionModifier;\r\n\r\n this.radius -= this.inertialRadiusOffset;\r\n this.inertialAlphaOffset *= this.inertia;\r\n this.inertialBetaOffset *= this.inertia;\r\n this.inertialRadiusOffset *= this.inertia;\r\n if (Math.abs(this.inertialAlphaOffset) < Epsilon) {\r\n this.inertialAlphaOffset = 0;\r\n }\r\n if (Math.abs(this.inertialBetaOffset) < Epsilon) {\r\n this.inertialBetaOffset = 0;\r\n }\r\n if (Math.abs(this.inertialRadiusOffset) < this.speed * Epsilon) {\r\n this.inertialRadiusOffset = 0;\r\n }\r\n }\r\n\r\n // Panning inertia\r\n if (this.inertialPanningX !== 0 || this.inertialPanningY !== 0) {\r\n const localDirection = new Vector3(this.inertialPanningX, this.inertialPanningY, this.inertialPanningY);\r\n\r\n this._viewMatrix.invertToRef(this._cameraTransformMatrix);\r\n localDirection.multiplyInPlace(this.panningAxis);\r\n Vector3.TransformNormalToRef(localDirection, this._cameraTransformMatrix, this._transformedDirection);\r\n // Eliminate y if mapPanning is enabled\r\n if (this.mapPanning || !this.panningAxis.y) {\r\n this._transformedDirection.y = 0;\r\n }\r\n\r\n if (!this._targetHost) {\r\n if (this.panningDistanceLimit) {\r\n this._transformedDirection.addInPlace(this._target);\r\n const distanceSquared = Vector3.DistanceSquared(this._transformedDirection, this.panningOriginTarget);\r\n if (distanceSquared <= this.panningDistanceLimit * this.panningDistanceLimit) {\r\n this._target.copyFrom(this._transformedDirection);\r\n }\r\n } else {\r\n this._target.addInPlace(this._transformedDirection);\r\n }\r\n }\r\n\r\n this.inertialPanningX *= this.panningInertia;\r\n this.inertialPanningY *= this.panningInertia;\r\n\r\n if (Math.abs(this.inertialPanningX) < this.speed * Epsilon) {\r\n this.inertialPanningX = 0;\r\n }\r\n if (Math.abs(this.inertialPanningY) < this.speed * Epsilon) {\r\n this.inertialPanningY = 0;\r\n }\r\n }\r\n\r\n // Limits\r\n this._checkLimits();\r\n\r\n super._checkInputs();\r\n }\r\n\r\n protected _checkLimits() {\r\n if (this.lowerBetaLimit === null || this.lowerBetaLimit === undefined) {\r\n if (this.allowUpsideDown && this.beta > Math.PI) {\r\n this.beta = this.beta - 2 * Math.PI;\r\n }\r\n } else {\r\n if (this.beta < this.lowerBetaLimit) {\r\n this.beta = this.lowerBetaLimit;\r\n }\r\n }\r\n\r\n if (this.upperBetaLimit === null || this.upperBetaLimit === undefined) {\r\n if (this.allowUpsideDown && this.beta < -Math.PI) {\r\n this.beta = this.beta + 2 * Math.PI;\r\n }\r\n } else {\r\n if (this.beta > this.upperBetaLimit) {\r\n this.beta = this.upperBetaLimit;\r\n }\r\n }\r\n\r\n if (this.lowerAlphaLimit !== null && this.alpha < this.lowerAlphaLimit) {\r\n this.alpha = this.lowerAlphaLimit;\r\n }\r\n if (this.upperAlphaLimit !== null && this.alpha > this.upperAlphaLimit) {\r\n this.alpha = this.upperAlphaLimit;\r\n }\r\n\r\n if (this.lowerRadiusLimit !== null && this.radius < this.lowerRadiusLimit) {\r\n this.radius = this.lowerRadiusLimit;\r\n this.inertialRadiusOffset = 0;\r\n }\r\n if (this.upperRadiusLimit !== null && this.radius > this.upperRadiusLimit) {\r\n this.radius = this.upperRadiusLimit;\r\n this.inertialRadiusOffset = 0;\r\n }\r\n }\r\n\r\n /**\r\n * Rebuilds angles (alpha, beta) and radius from the give position and target\r\n */\r\n public rebuildAnglesAndRadius(): void {\r\n this._position.subtractToRef(this._getTargetPosition(), this._computationVector);\r\n\r\n // need to rotate to Y up equivalent if up vector not Axis.Y\r\n if (this._upVector.x !== 0 || this._upVector.y !== 1.0 || this._upVector.z !== 0) {\r\n Vector3.TransformCoordinatesToRef(this._computationVector, this._upToYMatrix, this._computationVector);\r\n }\r\n\r\n this.radius = this._computationVector.length();\r\n\r\n if (this.radius === 0) {\r\n this.radius = 0.0001; // Just to avoid division by zero\r\n }\r\n\r\n // Alpha\r\n const previousAlpha = this.alpha;\r\n if (this._computationVector.x === 0 && this._computationVector.z === 0) {\r\n this.alpha = Math.PI / 2; // avoid division by zero when looking along up axis, and set to acos(0)\r\n } else {\r\n this.alpha = Math.acos(this._computationVector.x / Math.sqrt(Math.pow(this._computationVector.x, 2) + Math.pow(this._computationVector.z, 2)));\r\n }\r\n\r\n if (this._computationVector.z < 0) {\r\n this.alpha = 2 * Math.PI - this.alpha;\r\n }\r\n\r\n // Calculate the number of revolutions between the new and old alpha values.\r\n const alphaCorrectionTurns = Math.round((previousAlpha - this.alpha) / (2.0 * Math.PI));\r\n // Adjust alpha so that its numerical representation is the closest one to the old value.\r\n this.alpha += alphaCorrectionTurns * 2.0 * Math.PI;\r\n\r\n // Beta\r\n this.beta = Math.acos(this._computationVector.y / this.radius);\r\n\r\n this._checkLimits();\r\n }\r\n\r\n /**\r\n * Use a position to define the current camera related information like alpha, beta and radius\r\n * @param position Defines the position to set the camera at\r\n */\r\n public setPosition(position: Vector3): void {\r\n if (this._position.equals(position)) {\r\n return;\r\n }\r\n this._position.copyFrom(position);\r\n\r\n this.rebuildAnglesAndRadius();\r\n }\r\n\r\n /**\r\n * Defines the target the camera should look at.\r\n * This will automatically adapt alpha beta and radius to fit within the new target.\r\n * Please note that setting a target as a mesh will disable panning.\r\n * @param target Defines the new target as a Vector or a mesh\r\n * @param toBoundingCenter In case of a mesh target, defines whether to target the mesh position or its bounding information center\r\n * @param allowSamePosition If false, prevents reapplying the new computed position if it is identical to the current one (optim)\r\n * @param cloneAlphaBetaRadius If true, replicate the current setup (alpha, beta, radius) on the new target\r\n */\r\n public setTarget(target: AbstractMesh | Vector3, toBoundingCenter = false, allowSamePosition = false, cloneAlphaBetaRadius = false): void {\r\n cloneAlphaBetaRadius = this.overrideCloneAlphaBetaRadius ?? cloneAlphaBetaRadius;\r\n\r\n if ((target).getBoundingInfo) {\r\n if (toBoundingCenter) {\r\n this._targetBoundingCenter = (target).getBoundingInfo().boundingBox.centerWorld.clone();\r\n } else {\r\n this._targetBoundingCenter = null;\r\n }\r\n (target).computeWorldMatrix();\r\n this._targetHost = target;\r\n this._target = this._getTargetPosition();\r\n\r\n this.onMeshTargetChangedObservable.notifyObservers(this._targetHost);\r\n } else {\r\n const newTarget = target;\r\n const currentTarget = this._getTargetPosition();\r\n if (currentTarget && !allowSamePosition && currentTarget.equals(newTarget)) {\r\n return;\r\n }\r\n this._targetHost = null;\r\n this._target = newTarget;\r\n this._targetBoundingCenter = null;\r\n this.onMeshTargetChangedObservable.notifyObservers(null);\r\n }\r\n\r\n if (!cloneAlphaBetaRadius) {\r\n this.rebuildAnglesAndRadius();\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _getViewMatrix(): Matrix {\r\n // Compute\r\n const cosa = Math.cos(this.alpha);\r\n const sina = Math.sin(this.alpha);\r\n const cosb = Math.cos(this.beta);\r\n let sinb = Math.sin(this.beta);\r\n\r\n if (sinb === 0) {\r\n sinb = 0.0001;\r\n }\r\n\r\n if (this.radius === 0) {\r\n this.radius = 0.0001; // Just to avoid division by zero\r\n }\r\n\r\n const target = this._getTargetPosition();\r\n this._computationVector.copyFromFloats(this.radius * cosa * sinb, this.radius * cosb, this.radius * sina * sinb);\r\n\r\n // Rotate according to up vector\r\n if (this._upVector.x !== 0 || this._upVector.y !== 1.0 || this._upVector.z !== 0) {\r\n Vector3.TransformCoordinatesToRef(this._computationVector, this._yToUpMatrix, this._computationVector);\r\n }\r\n\r\n target.addToRef(this._computationVector, this._newPosition);\r\n if (this.getScene().collisionsEnabled && this.checkCollisions) {\r\n const coordinator = this.getScene().collisionCoordinator;\r\n if (!this._collider) {\r\n this._collider = coordinator.createCollider();\r\n }\r\n this._collider._radius = this.collisionRadius;\r\n this._newPosition.subtractToRef(this._position, this._collisionVelocity);\r\n this._collisionTriggered = true;\r\n coordinator.getNewPosition(this._position, this._collisionVelocity, this._collider, 3, null, this._onCollisionPositionChange, this.uniqueId);\r\n } else {\r\n this._position.copyFrom(this._newPosition);\r\n\r\n let up = this.upVector;\r\n if (this.allowUpsideDown && sinb < 0) {\r\n up = up.negate();\r\n }\r\n\r\n this._computeViewMatrix(this._position, target, up);\r\n\r\n this._viewMatrix.addAtIndex(12, this.targetScreenOffset.x);\r\n this._viewMatrix.addAtIndex(13, this.targetScreenOffset.y);\r\n }\r\n this._currentTarget = target;\r\n return this._viewMatrix;\r\n }\r\n\r\n protected _onCollisionPositionChange = (collisionId: number, newPosition: Vector3, collidedMesh: Nullable = null) => {\r\n if (!collidedMesh) {\r\n this._previousPosition.copyFrom(this._position);\r\n } else {\r\n this.setPosition(newPosition);\r\n\r\n if (this.onCollide) {\r\n this.onCollide(collidedMesh);\r\n }\r\n }\r\n\r\n // Recompute because of constraints\r\n const cosa = Math.cos(this.alpha);\r\n const sina = Math.sin(this.alpha);\r\n const cosb = Math.cos(this.beta);\r\n let sinb = Math.sin(this.beta);\r\n\r\n if (sinb === 0) {\r\n sinb = 0.0001;\r\n }\r\n\r\n const target = this._getTargetPosition();\r\n this._computationVector.copyFromFloats(this.radius * cosa * sinb, this.radius * cosb, this.radius * sina * sinb);\r\n target.addToRef(this._computationVector, this._newPosition);\r\n this._position.copyFrom(this._newPosition);\r\n\r\n let up = this.upVector;\r\n if (this.allowUpsideDown && this.beta < 0) {\r\n up = up.clone();\r\n up = up.negate();\r\n }\r\n\r\n this._computeViewMatrix(this._position, target, up);\r\n this._viewMatrix.addAtIndex(12, this.targetScreenOffset.x);\r\n this._viewMatrix.addAtIndex(13, this.targetScreenOffset.y);\r\n\r\n this._collisionTriggered = false;\r\n };\r\n\r\n /**\r\n * Zooms on a mesh to be at the min distance where we could see it fully in the current viewport.\r\n * @param meshes Defines the mesh to zoom on\r\n * @param doNotUpdateMaxZ Defines whether or not maxZ should be updated whilst zooming on the mesh (this can happen if the mesh is big and the maxradius pretty small for instance)\r\n */\r\n public zoomOn(meshes?: AbstractMesh[], doNotUpdateMaxZ = false): void {\r\n meshes = meshes || this.getScene().meshes;\r\n\r\n const minMaxVector = Mesh.MinMax(meshes);\r\n const distance = Vector3.Distance(minMaxVector.min, minMaxVector.max);\r\n\r\n this.radius = distance * this.zoomOnFactor;\r\n\r\n this.focusOn({ min: minMaxVector.min, max: minMaxVector.max, distance: distance }, doNotUpdateMaxZ);\r\n }\r\n\r\n /**\r\n * Focus on a mesh or a bounding box. This adapts the target and maxRadius if necessary but does not update the current radius.\r\n * The target will be changed but the radius\r\n * @param meshesOrMinMaxVectorAndDistance Defines the mesh or bounding info to focus on\r\n * @param doNotUpdateMaxZ Defines whether or not maxZ should be updated whilst zooming on the mesh (this can happen if the mesh is big and the maxradius pretty small for instance)\r\n */\r\n public focusOn(meshesOrMinMaxVectorAndDistance: AbstractMesh[] | { min: Vector3; max: Vector3; distance: number }, doNotUpdateMaxZ = false): void {\r\n let meshesOrMinMaxVector: { min: Vector3; max: Vector3 };\r\n let distance: number;\r\n\r\n if ((meshesOrMinMaxVectorAndDistance).min === undefined) {\r\n // meshes\r\n const meshes = meshesOrMinMaxVectorAndDistance || this.getScene().meshes;\r\n meshesOrMinMaxVector = Mesh.MinMax(meshes);\r\n distance = Vector3.Distance(meshesOrMinMaxVector.min, meshesOrMinMaxVector.max);\r\n } else {\r\n //minMaxVector and distance\r\n const minMaxVectorAndDistance = meshesOrMinMaxVectorAndDistance;\r\n meshesOrMinMaxVector = minMaxVectorAndDistance;\r\n distance = minMaxVectorAndDistance.distance;\r\n }\r\n\r\n this._target = Mesh.Center(meshesOrMinMaxVector);\r\n\r\n if (!doNotUpdateMaxZ) {\r\n this.maxZ = distance * 2;\r\n }\r\n }\r\n\r\n /**\r\n * @override\r\n * Override Camera.createRigCamera\r\n */\r\n public createRigCamera(name: string, cameraIndex: number): Camera {\r\n let alphaShift: number = 0;\r\n switch (this.cameraRigMode) {\r\n case Camera.RIG_MODE_STEREOSCOPIC_ANAGLYPH:\r\n case Camera.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL:\r\n case Camera.RIG_MODE_STEREOSCOPIC_OVERUNDER:\r\n case Camera.RIG_MODE_STEREOSCOPIC_INTERLACED:\r\n case Camera.RIG_MODE_VR:\r\n alphaShift = this._cameraRigParams.stereoHalfAngle * (cameraIndex === 0 ? 1 : -1);\r\n break;\r\n case Camera.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED:\r\n alphaShift = this._cameraRigParams.stereoHalfAngle * (cameraIndex === 0 ? -1 : 1);\r\n break;\r\n }\r\n const rigCam = new ArcRotateCamera(name, this.alpha + alphaShift, this.beta, this.radius, this._target, this.getScene());\r\n rigCam._cameraRigParams = {};\r\n rigCam.isRigCamera = true;\r\n rigCam.rigParent = this;\r\n rigCam.upVector = this.upVector;\r\n\r\n rigCam.mode = this.mode;\r\n rigCam.orthoLeft = this.orthoLeft;\r\n rigCam.orthoRight = this.orthoRight;\r\n rigCam.orthoBottom = this.orthoBottom;\r\n rigCam.orthoTop = this.orthoTop;\r\n\r\n return rigCam;\r\n }\r\n\r\n /**\r\n * @internal\r\n * @override\r\n * Override Camera._updateRigCameras\r\n */\r\n public _updateRigCameras() {\r\n const camLeft = this._rigCameras[0];\r\n const camRight = this._rigCameras[1];\r\n\r\n camLeft.beta = camRight.beta = this.beta;\r\n\r\n switch (this.cameraRigMode) {\r\n case Camera.RIG_MODE_STEREOSCOPIC_ANAGLYPH:\r\n case Camera.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL:\r\n case Camera.RIG_MODE_STEREOSCOPIC_OVERUNDER:\r\n case Camera.RIG_MODE_STEREOSCOPIC_INTERLACED:\r\n case Camera.RIG_MODE_VR:\r\n camLeft.alpha = this.alpha - this._cameraRigParams.stereoHalfAngle;\r\n camRight.alpha = this.alpha + this._cameraRigParams.stereoHalfAngle;\r\n break;\r\n case Camera.RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED:\r\n camLeft.alpha = this.alpha + this._cameraRigParams.stereoHalfAngle;\r\n camRight.alpha = this.alpha - this._cameraRigParams.stereoHalfAngle;\r\n break;\r\n }\r\n super._updateRigCameras();\r\n }\r\n\r\n /**\r\n * Destroy the camera and release the current resources hold by it.\r\n */\r\n public dispose(): void {\r\n this.inputs.clear();\r\n super.dispose();\r\n }\r\n\r\n /**\r\n * Gets the current object class name.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"ArcRotateCamera\";\r\n }\r\n}\r\n","import type { DeepImmutable, Nullable, float } from \"../types\";\r\nimport { ArrayTools } from \"../Misc/arrayTools\";\r\nimport { Matrix, Vector3, TmpVectors } from \"../Maths/math.vector\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport { PickingInfo } from \"../Collisions/pickingInfo\";\r\nimport { IntersectionInfo } from \"../Collisions/intersectionInfo\";\r\nimport type { BoundingBox } from \"./boundingBox\";\r\nimport type { BoundingSphere } from \"./boundingSphere\";\r\nimport { Scene } from \"../scene\";\r\nimport { Camera } from \"../Cameras/camera\";\r\nimport type { Plane } from \"../Maths/math.plane\";\r\nimport { EngineStore } from \"../Engines/engineStore\";\r\n\r\nimport type { Mesh } from \"../Meshes/mesh\";\r\n\r\n/**\r\n * Class representing a ray with position and direction\r\n */\r\nexport class Ray {\r\n private static readonly _TmpVector3 = ArrayTools.BuildArray(6, Vector3.Zero);\r\n private static _RayDistant = Ray.Zero();\r\n private _tmpRay: Ray;\r\n\r\n /**\r\n * Creates a new ray\r\n * @param origin origin point\r\n * @param direction direction\r\n * @param length length of the ray\r\n */\r\n constructor(\r\n /** origin point */\r\n public origin: Vector3,\r\n /** direction */\r\n public direction: Vector3,\r\n /** length of the ray */\r\n public length: number = Number.MAX_VALUE\r\n ) {}\r\n\r\n // Methods\r\n\r\n /**\r\n * Clone the current ray\r\n * @returns a new ray\r\n */\r\n public clone(): Ray {\r\n return new Ray(this.origin.clone(), this.direction.clone(), this.length);\r\n }\r\n\r\n /**\r\n * Checks if the ray intersects a box\r\n * This does not account for the ray length by design to improve perfs.\r\n * @param minimum bound of the box\r\n * @param maximum bound of the box\r\n * @param intersectionTreshold extra extend to be added to the box in all direction\r\n * @returns if the box was hit\r\n */\r\n public intersectsBoxMinMax(minimum: DeepImmutable, maximum: DeepImmutable, intersectionTreshold: number = 0): boolean {\r\n const newMinimum = Ray._TmpVector3[0].copyFromFloats(minimum.x - intersectionTreshold, minimum.y - intersectionTreshold, minimum.z - intersectionTreshold);\r\n const newMaximum = Ray._TmpVector3[1].copyFromFloats(maximum.x + intersectionTreshold, maximum.y + intersectionTreshold, maximum.z + intersectionTreshold);\r\n let d = 0.0;\r\n let maxValue = Number.MAX_VALUE;\r\n let inv: number;\r\n let min: number;\r\n let max: number;\r\n let temp: number;\r\n if (Math.abs(this.direction.x) < 0.0000001) {\r\n if (this.origin.x < newMinimum.x || this.origin.x > newMaximum.x) {\r\n return false;\r\n }\r\n } else {\r\n inv = 1.0 / this.direction.x;\r\n min = (newMinimum.x - this.origin.x) * inv;\r\n max = (newMaximum.x - this.origin.x) * inv;\r\n if (max === -Infinity) {\r\n max = Infinity;\r\n }\r\n\r\n if (min > max) {\r\n temp = min;\r\n min = max;\r\n max = temp;\r\n }\r\n\r\n d = Math.max(min, d);\r\n maxValue = Math.min(max, maxValue);\r\n\r\n if (d > maxValue) {\r\n return false;\r\n }\r\n }\r\n\r\n if (Math.abs(this.direction.y) < 0.0000001) {\r\n if (this.origin.y < newMinimum.y || this.origin.y > newMaximum.y) {\r\n return false;\r\n }\r\n } else {\r\n inv = 1.0 / this.direction.y;\r\n min = (newMinimum.y - this.origin.y) * inv;\r\n max = (newMaximum.y - this.origin.y) * inv;\r\n\r\n if (max === -Infinity) {\r\n max = Infinity;\r\n }\r\n\r\n if (min > max) {\r\n temp = min;\r\n min = max;\r\n max = temp;\r\n }\r\n\r\n d = Math.max(min, d);\r\n maxValue = Math.min(max, maxValue);\r\n\r\n if (d > maxValue) {\r\n return false;\r\n }\r\n }\r\n\r\n if (Math.abs(this.direction.z) < 0.0000001) {\r\n if (this.origin.z < newMinimum.z || this.origin.z > newMaximum.z) {\r\n return false;\r\n }\r\n } else {\r\n inv = 1.0 / this.direction.z;\r\n min = (newMinimum.z - this.origin.z) * inv;\r\n max = (newMaximum.z - this.origin.z) * inv;\r\n\r\n if (max === -Infinity) {\r\n max = Infinity;\r\n }\r\n\r\n if (min > max) {\r\n temp = min;\r\n min = max;\r\n max = temp;\r\n }\r\n\r\n d = Math.max(min, d);\r\n maxValue = Math.min(max, maxValue);\r\n\r\n if (d > maxValue) {\r\n return false;\r\n }\r\n }\r\n return true;\r\n }\r\n\r\n /**\r\n * Checks if the ray intersects a box\r\n * This does not account for the ray lenght by design to improve perfs.\r\n * @param box the bounding box to check\r\n * @param intersectionTreshold extra extend to be added to the BoundingBox in all direction\r\n * @returns if the box was hit\r\n */\r\n public intersectsBox(box: DeepImmutable, intersectionTreshold: number = 0): boolean {\r\n return this.intersectsBoxMinMax(box.minimum, box.maximum, intersectionTreshold);\r\n }\r\n\r\n /**\r\n * If the ray hits a sphere\r\n * @param sphere the bounding sphere to check\r\n * @param intersectionTreshold extra extend to be added to the BoundingSphere in all direction\r\n * @returns true if it hits the sphere\r\n */\r\n public intersectsSphere(sphere: DeepImmutable, intersectionTreshold: number = 0): boolean {\r\n const x = sphere.center.x - this.origin.x;\r\n const y = sphere.center.y - this.origin.y;\r\n const z = sphere.center.z - this.origin.z;\r\n const pyth = x * x + y * y + z * z;\r\n const radius = sphere.radius + intersectionTreshold;\r\n const rr = radius * radius;\r\n\r\n if (pyth <= rr) {\r\n return true;\r\n }\r\n\r\n const dot = x * this.direction.x + y * this.direction.y + z * this.direction.z;\r\n if (dot < 0.0) {\r\n return false;\r\n }\r\n\r\n const temp = pyth - dot * dot;\r\n\r\n return temp <= rr;\r\n }\r\n\r\n /**\r\n * If the ray hits a triange\r\n * @param vertex0 triangle vertex\r\n * @param vertex1 triangle vertex\r\n * @param vertex2 triangle vertex\r\n * @returns intersection information if hit\r\n */\r\n public intersectsTriangle(vertex0: DeepImmutable, vertex1: DeepImmutable, vertex2: DeepImmutable): Nullable {\r\n const edge1 = Ray._TmpVector3[0];\r\n const edge2 = Ray._TmpVector3[1];\r\n const pvec = Ray._TmpVector3[2];\r\n const tvec = Ray._TmpVector3[3];\r\n const qvec = Ray._TmpVector3[4];\r\n\r\n vertex1.subtractToRef(vertex0, edge1);\r\n vertex2.subtractToRef(vertex0, edge2);\r\n Vector3.CrossToRef(this.direction, edge2, pvec);\r\n const det = Vector3.Dot(edge1, pvec);\r\n\r\n if (det === 0) {\r\n return null;\r\n }\r\n\r\n const invdet = 1 / det;\r\n\r\n this.origin.subtractToRef(vertex0, tvec);\r\n\r\n const bv = Vector3.Dot(tvec, pvec) * invdet;\r\n\r\n if (bv < 0 || bv > 1.0) {\r\n return null;\r\n }\r\n\r\n Vector3.CrossToRef(tvec, edge1, qvec);\r\n\r\n const bw = Vector3.Dot(this.direction, qvec) * invdet;\r\n\r\n if (bw < 0 || bv + bw > 1.0) {\r\n return null;\r\n }\r\n\r\n //check if the distance is longer than the predefined length.\r\n const distance = Vector3.Dot(edge2, qvec) * invdet;\r\n if (distance > this.length) {\r\n return null;\r\n }\r\n\r\n return new IntersectionInfo(1 - bv - bw, bv, distance);\r\n }\r\n\r\n /**\r\n * Checks if ray intersects a plane\r\n * @param plane the plane to check\r\n * @returns the distance away it was hit\r\n */\r\n public intersectsPlane(plane: DeepImmutable): Nullable {\r\n let distance: number;\r\n const result1 = Vector3.Dot(plane.normal, this.direction);\r\n if (Math.abs(result1) < 9.99999997475243e-7) {\r\n return null;\r\n } else {\r\n const result2 = Vector3.Dot(plane.normal, this.origin);\r\n distance = (-plane.d - result2) / result1;\r\n if (distance < 0.0) {\r\n if (distance < -9.99999997475243e-7) {\r\n return null;\r\n } else {\r\n return 0;\r\n }\r\n }\r\n\r\n return distance;\r\n }\r\n }\r\n /**\r\n * Calculate the intercept of a ray on a given axis\r\n * @param axis to check 'x' | 'y' | 'z'\r\n * @param offset from axis interception (i.e. an offset of 1y is intercepted above ground)\r\n * @returns a vector containing the coordinates where 'axis' is equal to zero (else offset), or null if there is no intercept.\r\n */\r\n public intersectsAxis(axis: string, offset: number = 0): Nullable {\r\n switch (axis) {\r\n case \"y\": {\r\n const t = (this.origin.y - offset) / this.direction.y;\r\n if (t > 0) {\r\n return null;\r\n }\r\n return new Vector3(this.origin.x + this.direction.x * -t, offset, this.origin.z + this.direction.z * -t);\r\n }\r\n case \"x\": {\r\n const t = (this.origin.x - offset) / this.direction.x;\r\n if (t > 0) {\r\n return null;\r\n }\r\n return new Vector3(offset, this.origin.y + this.direction.y * -t, this.origin.z + this.direction.z * -t);\r\n }\r\n case \"z\": {\r\n const t = (this.origin.z - offset) / this.direction.z;\r\n if (t > 0) {\r\n return null;\r\n }\r\n return new Vector3(this.origin.x + this.direction.x * -t, this.origin.y + this.direction.y * -t, offset);\r\n }\r\n default:\r\n return null;\r\n }\r\n }\r\n\r\n /**\r\n * Checks if ray intersects a mesh. The ray is defined in WORLD space. A mesh triangle can be picked both from its front and back sides,\r\n * irrespective of orientation.\r\n * @param mesh the mesh to check\r\n * @param fastCheck defines if the first intersection will be used (and not the closest)\r\n * @param trianglePredicate defines an optional predicate used to select faces when a mesh intersection is detected\r\n * @param onlyBoundingInfo defines a boolean indicating if picking should only happen using bounding info (false by default)\r\n * @param worldToUse defines the world matrix to use to get the world coordinate of the intersection point\r\n * @param skipBoundingInfo a boolean indicating if we should skip the bounding info check\r\n * @returns picking info of the intersection\r\n */\r\n public intersectsMesh(\r\n mesh: DeepImmutable,\r\n fastCheck?: boolean,\r\n trianglePredicate?: TrianglePickingPredicate,\r\n onlyBoundingInfo = false,\r\n worldToUse?: Matrix,\r\n skipBoundingInfo = false\r\n ): PickingInfo {\r\n const tm = TmpVectors.Matrix[0];\r\n\r\n mesh.getWorldMatrix().invertToRef(tm);\r\n\r\n if (this._tmpRay) {\r\n Ray.TransformToRef(this, tm, this._tmpRay);\r\n } else {\r\n this._tmpRay = Ray.Transform(this, tm);\r\n }\r\n\r\n return mesh.intersects(this._tmpRay, fastCheck, trianglePredicate, onlyBoundingInfo, worldToUse, skipBoundingInfo);\r\n }\r\n\r\n /**\r\n * Checks if ray intersects a mesh\r\n * @param meshes the meshes to check\r\n * @param fastCheck defines if the first intersection will be used (and not the closest)\r\n * @param results array to store result in\r\n * @returns Array of picking infos\r\n */\r\n public intersectsMeshes(meshes: Array>, fastCheck?: boolean, results?: Array): Array {\r\n if (results) {\r\n results.length = 0;\r\n } else {\r\n results = [];\r\n }\r\n\r\n for (let i = 0; i < meshes.length; i++) {\r\n const pickInfo = this.intersectsMesh(meshes[i], fastCheck);\r\n\r\n if (pickInfo.hit) {\r\n results.push(pickInfo);\r\n }\r\n }\r\n\r\n results.sort(this._comparePickingInfo);\r\n\r\n return results;\r\n }\r\n\r\n private _comparePickingInfo(pickingInfoA: DeepImmutable, pickingInfoB: DeepImmutable): number {\r\n if (pickingInfoA.distance < pickingInfoB.distance) {\r\n return -1;\r\n } else if (pickingInfoA.distance > pickingInfoB.distance) {\r\n return 1;\r\n } else {\r\n return 0;\r\n }\r\n }\r\n\r\n private static _Smallnum = 0.00000001;\r\n private static _Rayl = 10e8;\r\n\r\n /**\r\n * Intersection test between the ray and a given segment within a given tolerance (threshold)\r\n * @param sega the first point of the segment to test the intersection against\r\n * @param segb the second point of the segment to test the intersection against\r\n * @param threshold the tolerance margin, if the ray doesn't intersect the segment but is close to the given threshold, the intersection is successful\r\n * @returns the distance from the ray origin to the intersection point if there's intersection, or -1 if there's no intersection\r\n */\r\n intersectionSegment(sega: DeepImmutable, segb: DeepImmutable, threshold: number): number {\r\n const o = this.origin;\r\n const u = TmpVectors.Vector3[0];\r\n const rsegb = TmpVectors.Vector3[1];\r\n const v = TmpVectors.Vector3[2];\r\n const w = TmpVectors.Vector3[3];\r\n\r\n segb.subtractToRef(sega, u);\r\n\r\n this.direction.scaleToRef(Ray._Rayl, v);\r\n o.addToRef(v, rsegb);\r\n\r\n sega.subtractToRef(o, w);\r\n\r\n const a = Vector3.Dot(u, u); // always >= 0\r\n const b = Vector3.Dot(u, v);\r\n const c = Vector3.Dot(v, v); // always >= 0\r\n const d = Vector3.Dot(u, w);\r\n const e = Vector3.Dot(v, w);\r\n const D = a * c - b * b; // always >= 0\r\n let sN: number,\r\n sD = D; // sc = sN / sD, default sD = D >= 0\r\n let tN: number,\r\n tD = D; // tc = tN / tD, default tD = D >= 0\r\n\r\n // compute the line parameters of the two closest points\r\n if (D < Ray._Smallnum) {\r\n // the lines are almost parallel\r\n sN = 0.0; // force using point P0 on segment S1\r\n sD = 1.0; // to prevent possible division by 0.0 later\r\n tN = e;\r\n tD = c;\r\n } else {\r\n // get the closest points on the infinite lines\r\n sN = b * e - c * d;\r\n tN = a * e - b * d;\r\n if (sN < 0.0) {\r\n // sc < 0 => the s=0 edge is visible\r\n sN = 0.0;\r\n tN = e;\r\n tD = c;\r\n } else if (sN > sD) {\r\n // sc > 1 => the s=1 edge is visible\r\n sN = sD;\r\n tN = e + b;\r\n tD = c;\r\n }\r\n }\r\n\r\n if (tN < 0.0) {\r\n // tc < 0 => the t=0 edge is visible\r\n tN = 0.0;\r\n // recompute sc for this edge\r\n if (-d < 0.0) {\r\n sN = 0.0;\r\n } else if (-d > a) {\r\n sN = sD;\r\n } else {\r\n sN = -d;\r\n sD = a;\r\n }\r\n } else if (tN > tD) {\r\n // tc > 1 => the t=1 edge is visible\r\n tN = tD;\r\n // recompute sc for this edge\r\n if (-d + b < 0.0) {\r\n sN = 0;\r\n } else if (-d + b > a) {\r\n sN = sD;\r\n } else {\r\n sN = -d + b;\r\n sD = a;\r\n }\r\n }\r\n // finally do the division to get sc and tc\r\n const sc = Math.abs(sN) < Ray._Smallnum ? 0.0 : sN / sD;\r\n const tc = Math.abs(tN) < Ray._Smallnum ? 0.0 : tN / tD;\r\n\r\n // get the difference of the two closest points\r\n const qtc = TmpVectors.Vector3[4];\r\n v.scaleToRef(tc, qtc);\r\n const qsc = TmpVectors.Vector3[5];\r\n u.scaleToRef(sc, qsc);\r\n qsc.addInPlace(w);\r\n const dP = TmpVectors.Vector3[6];\r\n qsc.subtractToRef(qtc, dP); // = S1(sc) - S2(tc)\r\n\r\n const isIntersected = tc > 0 && tc <= this.length && dP.lengthSquared() < threshold * threshold; // return intersection result\r\n\r\n if (isIntersected) {\r\n return qsc.length();\r\n }\r\n return -1;\r\n }\r\n\r\n /**\r\n * Update the ray from viewport position\r\n * @param x position\r\n * @param y y position\r\n * @param viewportWidth viewport width\r\n * @param viewportHeight viewport height\r\n * @param world world matrix\r\n * @param view view matrix\r\n * @param projection projection matrix\r\n * @param enableDistantPicking defines if picking should handle large values for mesh position/scaling (false by default)\r\n * @returns this ray updated\r\n */\r\n public update(\r\n x: number,\r\n y: number,\r\n viewportWidth: number,\r\n viewportHeight: number,\r\n world: DeepImmutable,\r\n view: DeepImmutable,\r\n projection: DeepImmutable,\r\n enableDistantPicking: boolean = false\r\n ): Ray {\r\n if (enableDistantPicking) {\r\n // With world matrices having great values (like 8000000000 on 1 or more scaling or position axis),\r\n // multiplying view/projection/world and doing invert will result in loss of float precision in the matrix.\r\n // One way to fix it is to compute the ray with world at identity then transform the ray in object space.\r\n // This is slower (2 matrix inverts instead of 1) but precision is preserved.\r\n // This is hidden behind `EnableDistantPicking` flag (default is false)\r\n if (!Ray._RayDistant) {\r\n Ray._RayDistant = Ray.Zero();\r\n }\r\n\r\n Ray._RayDistant.unprojectRayToRef(x, y, viewportWidth, viewportHeight, Matrix.IdentityReadOnly, view, projection);\r\n\r\n const tm = TmpVectors.Matrix[0];\r\n world.invertToRef(tm);\r\n Ray.TransformToRef(Ray._RayDistant, tm, this);\r\n } else {\r\n this.unprojectRayToRef(x, y, viewportWidth, viewportHeight, world, view, projection);\r\n }\r\n\r\n return this;\r\n }\r\n\r\n // Statics\r\n /**\r\n * Creates a ray with origin and direction of 0,0,0\r\n * @returns the new ray\r\n */\r\n public static Zero(): Ray {\r\n return new Ray(Vector3.Zero(), Vector3.Zero());\r\n }\r\n\r\n /**\r\n * Creates a new ray from screen space and viewport\r\n * @param x position\r\n * @param y y position\r\n * @param viewportWidth viewport width\r\n * @param viewportHeight viewport height\r\n * @param world world matrix\r\n * @param view view matrix\r\n * @param projection projection matrix\r\n * @returns new ray\r\n */\r\n public static CreateNew(\r\n x: number,\r\n y: number,\r\n viewportWidth: number,\r\n viewportHeight: number,\r\n world: DeepImmutable,\r\n view: DeepImmutable,\r\n projection: DeepImmutable\r\n ): Ray {\r\n const result = Ray.Zero();\r\n\r\n return result.update(x, y, viewportWidth, viewportHeight, world, view, projection);\r\n }\r\n\r\n /**\r\n * Function will create a new transformed ray starting from origin and ending at the end point. Ray's length will be set, and ray will be\r\n * transformed to the given world matrix.\r\n * @param origin The origin point\r\n * @param end The end point\r\n * @param world a matrix to transform the ray to. Default is the identity matrix.\r\n * @returns the new ray\r\n */\r\n public static CreateNewFromTo(origin: Vector3, end: Vector3, world: DeepImmutable = Matrix.IdentityReadOnly): Ray {\r\n const direction = end.subtract(origin);\r\n const length = Math.sqrt(direction.x * direction.x + direction.y * direction.y + direction.z * direction.z);\r\n direction.normalize();\r\n\r\n return Ray.Transform(new Ray(origin, direction, length), world);\r\n }\r\n\r\n /**\r\n * Transforms a ray by a matrix\r\n * @param ray ray to transform\r\n * @param matrix matrix to apply\r\n * @returns the resulting new ray\r\n */\r\n public static Transform(ray: DeepImmutable, matrix: DeepImmutable): Ray {\r\n const result = new Ray(new Vector3(0, 0, 0), new Vector3(0, 0, 0));\r\n Ray.TransformToRef(ray, matrix, result);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Transforms a ray by a matrix\r\n * @param ray ray to transform\r\n * @param matrix matrix to apply\r\n * @param result ray to store result in\r\n */\r\n public static TransformToRef(ray: DeepImmutable, matrix: DeepImmutable, result: Ray): void {\r\n Vector3.TransformCoordinatesToRef(ray.origin, matrix, result.origin);\r\n Vector3.TransformNormalToRef(ray.direction, matrix, result.direction);\r\n result.length = ray.length;\r\n\r\n const dir = result.direction;\r\n const len = dir.length();\r\n\r\n if (!(len === 0 || len === 1)) {\r\n const num = 1.0 / len;\r\n dir.x *= num;\r\n dir.y *= num;\r\n dir.z *= num;\r\n result.length *= len;\r\n }\r\n }\r\n\r\n /**\r\n * Unproject a ray from screen space to object space\r\n * @param sourceX defines the screen space x coordinate to use\r\n * @param sourceY defines the screen space y coordinate to use\r\n * @param viewportWidth defines the current width of the viewport\r\n * @param viewportHeight defines the current height of the viewport\r\n * @param world defines the world matrix to use (can be set to Identity to go to world space)\r\n * @param view defines the view matrix to use\r\n * @param projection defines the projection matrix to use\r\n */\r\n public unprojectRayToRef(\r\n sourceX: float,\r\n sourceY: float,\r\n viewportWidth: number,\r\n viewportHeight: number,\r\n world: DeepImmutable,\r\n view: DeepImmutable,\r\n projection: DeepImmutable\r\n ): void {\r\n const matrix = TmpVectors.Matrix[0];\r\n world.multiplyToRef(view, matrix);\r\n matrix.multiplyToRef(projection, matrix);\r\n matrix.invert();\r\n\r\n const engine = EngineStore.LastCreatedEngine;\r\n const nearScreenSource = TmpVectors.Vector3[0];\r\n nearScreenSource.x = (sourceX / viewportWidth) * 2 - 1;\r\n nearScreenSource.y = -((sourceY / viewportHeight) * 2 - 1);\r\n nearScreenSource.z = engine?.useReverseDepthBuffer ? 1 : engine?.isNDCHalfZRange ? 0 : -1;\r\n\r\n // far Z need to be close but < to 1 or camera projection matrix with maxZ = 0 will NaN\r\n const farScreenSource = TmpVectors.Vector3[1].copyFromFloats(nearScreenSource.x, nearScreenSource.y, 1.0 - 1e-8);\r\n const nearVec3 = TmpVectors.Vector3[2];\r\n const farVec3 = TmpVectors.Vector3[3];\r\n Vector3._UnprojectFromInvertedMatrixToRef(nearScreenSource, matrix, nearVec3);\r\n Vector3._UnprojectFromInvertedMatrixToRef(farScreenSource, matrix, farVec3);\r\n\r\n this.origin.copyFrom(nearVec3);\r\n farVec3.subtractToRef(nearVec3, this.direction);\r\n this.direction.normalize();\r\n }\r\n}\r\n\r\n// Picking\r\n/**\r\n * Type used to define predicate used to select faces when a mesh intersection is detected\r\n */\r\nexport type TrianglePickingPredicate = (p0: Vector3, p1: Vector3, p2: Vector3, ray: Ray, i0: number, i1: number, i2: number) => boolean;\r\n\r\ndeclare module \"../scene\" {\r\n export interface Scene {\r\n /** @internal */\r\n _tempPickingRay: Nullable;\r\n\r\n /** @internal */\r\n _cachedRayForTransform: Ray;\r\n\r\n /** @internal */\r\n _pickWithRayInverseMatrix: Matrix;\r\n\r\n /** @internal */\r\n _internalPick(\r\n rayFunction: (world: Matrix, enableDistantPicking: boolean) => Ray,\r\n predicate?: (mesh: AbstractMesh) => boolean,\r\n fastCheck?: boolean,\r\n onlyBoundingInfo?: boolean,\r\n trianglePredicate?: TrianglePickingPredicate\r\n ): PickingInfo;\r\n\r\n /** @internal */\r\n _internalMultiPick(\r\n rayFunction: (world: Matrix, enableDistantPicking: boolean) => Ray,\r\n predicate?: (mesh: AbstractMesh) => boolean,\r\n trianglePredicate?: TrianglePickingPredicate\r\n ): Nullable;\r\n\r\n /** @internal */\r\n _internalPickForMesh(\r\n pickingInfo: Nullable,\r\n rayFunction: (world: Matrix, enableDistantPicking: boolean) => Ray,\r\n mesh: AbstractMesh,\r\n world: Matrix,\r\n fastCheck?: boolean,\r\n onlyBoundingInfo?: boolean,\r\n trianglePredicate?: TrianglePickingPredicate,\r\n skipBoundingInfo?: boolean\r\n ): Nullable;\r\n }\r\n}\r\n\r\nScene.prototype.createPickingRay = function (x: number, y: number, world: Nullable, camera: Nullable, cameraViewSpace = false): Ray {\r\n const result = Ray.Zero();\r\n\r\n this.createPickingRayToRef(x, y, world, result, camera, cameraViewSpace);\r\n\r\n return result;\r\n};\r\n\r\nScene.prototype.createPickingRayToRef = function (\r\n x: number,\r\n y: number,\r\n world: Nullable,\r\n result: Ray,\r\n camera: Nullable,\r\n cameraViewSpace = false,\r\n enableDistantPicking = false\r\n): Scene {\r\n const engine = this.getEngine();\r\n\r\n if (!camera) {\r\n if (!this.activeCamera) {\r\n return this;\r\n }\r\n\r\n camera = this.activeCamera;\r\n }\r\n\r\n const cameraViewport = camera.viewport;\r\n const viewport = cameraViewport.toGlobal(engine.getRenderWidth(), engine.getRenderHeight());\r\n\r\n // Moving coordinates to local viewport world\r\n x = x / engine.getHardwareScalingLevel() - viewport.x;\r\n y = y / engine.getHardwareScalingLevel() - (engine.getRenderHeight() - viewport.y - viewport.height);\r\n\r\n result.update(\r\n x,\r\n y,\r\n viewport.width,\r\n viewport.height,\r\n world ? world : Matrix.IdentityReadOnly,\r\n cameraViewSpace ? Matrix.IdentityReadOnly : camera.getViewMatrix(),\r\n camera.getProjectionMatrix(),\r\n enableDistantPicking\r\n );\r\n return this;\r\n};\r\n\r\nScene.prototype.createPickingRayInCameraSpace = function (x: number, y: number, camera?: Camera): Ray {\r\n const result = Ray.Zero();\r\n\r\n this.createPickingRayInCameraSpaceToRef(x, y, result, camera);\r\n\r\n return result;\r\n};\r\n\r\nScene.prototype.createPickingRayInCameraSpaceToRef = function (x: number, y: number, result: Ray, camera?: Camera): Scene {\r\n if (!PickingInfo) {\r\n return this;\r\n }\r\n\r\n const engine = this.getEngine();\r\n\r\n if (!camera) {\r\n if (!this.activeCamera) {\r\n throw new Error(\"Active camera not set\");\r\n }\r\n\r\n camera = this.activeCamera;\r\n }\r\n\r\n const cameraViewport = camera.viewport;\r\n const viewport = cameraViewport.toGlobal(engine.getRenderWidth(), engine.getRenderHeight());\r\n const identity = Matrix.Identity();\r\n\r\n // Moving coordinates to local viewport world\r\n x = x / engine.getHardwareScalingLevel() - viewport.x;\r\n y = y / engine.getHardwareScalingLevel() - (engine.getRenderHeight() - viewport.y - viewport.height);\r\n result.update(x, y, viewport.width, viewport.height, identity, identity, camera.getProjectionMatrix());\r\n return this;\r\n};\r\n\r\nScene.prototype._internalPickForMesh = function (\r\n pickingInfo: Nullable,\r\n rayFunction: (world: Matrix, enableDistantPicking: boolean) => Ray,\r\n mesh: AbstractMesh,\r\n world: Matrix,\r\n fastCheck?: boolean,\r\n onlyBoundingInfo?: boolean,\r\n trianglePredicate?: TrianglePickingPredicate,\r\n skipBoundingInfo?: boolean\r\n) {\r\n const ray = rayFunction(world, mesh.enableDistantPicking);\r\n\r\n const result = mesh.intersects(ray, fastCheck, trianglePredicate, onlyBoundingInfo, world, skipBoundingInfo);\r\n if (!result || !result.hit) {\r\n return null;\r\n }\r\n\r\n if (!fastCheck && pickingInfo != null && result.distance >= pickingInfo.distance) {\r\n return null;\r\n }\r\n\r\n return result;\r\n};\r\n\r\nScene.prototype._internalPick = function (\r\n rayFunction: (world: Matrix, enableDistantPicking: boolean) => Ray,\r\n predicate?: (mesh: AbstractMesh) => boolean,\r\n fastCheck?: boolean,\r\n onlyBoundingInfo?: boolean,\r\n trianglePredicate?: TrianglePickingPredicate\r\n): PickingInfo {\r\n let pickingInfo = null;\r\n\r\n const computeWorldMatrixForCamera = !!(this.activeCameras && this.activeCameras.length > 1 && this.cameraToUseForPointers !== this.activeCamera);\r\n const currentCamera = this.cameraToUseForPointers || this.activeCamera;\r\n\r\n for (let meshIndex = 0; meshIndex < this.meshes.length; meshIndex++) {\r\n const mesh = this.meshes[meshIndex];\r\n\r\n if (predicate) {\r\n if (!predicate(mesh)) {\r\n continue;\r\n }\r\n } else if (!mesh.isEnabled() || !mesh.isVisible || !mesh.isPickable) {\r\n continue;\r\n }\r\n\r\n const forceCompute = computeWorldMatrixForCamera && mesh.isWorldMatrixCameraDependent();\r\n const world = mesh.computeWorldMatrix(forceCompute, currentCamera);\r\n\r\n if (mesh.hasThinInstances && (mesh as Mesh).thinInstanceEnablePicking) {\r\n // first check if the ray intersects the whole bounding box/sphere of the mesh\r\n const result = this._internalPickForMesh(pickingInfo, rayFunction, mesh, world, true, true, trianglePredicate);\r\n if (result) {\r\n if (onlyBoundingInfo) {\r\n // the user only asked for a bounding info check so we can return\r\n return result;\r\n }\r\n const tmpMatrix = TmpVectors.Matrix[1];\r\n const thinMatrices = (mesh as Mesh).thinInstanceGetWorldMatrices();\r\n for (let index = 0; index < thinMatrices.length; index++) {\r\n const thinMatrix = thinMatrices[index];\r\n thinMatrix.multiplyToRef(world, tmpMatrix);\r\n const result = this._internalPickForMesh(pickingInfo, rayFunction, mesh, tmpMatrix, fastCheck, onlyBoundingInfo, trianglePredicate, true);\r\n\r\n if (result) {\r\n pickingInfo = result;\r\n pickingInfo.thinInstanceIndex = index;\r\n\r\n if (fastCheck) {\r\n return pickingInfo;\r\n }\r\n }\r\n }\r\n }\r\n } else {\r\n const result = this._internalPickForMesh(pickingInfo, rayFunction, mesh, world, fastCheck, onlyBoundingInfo, trianglePredicate);\r\n\r\n if (result) {\r\n pickingInfo = result;\r\n\r\n if (fastCheck) {\r\n return pickingInfo;\r\n }\r\n }\r\n }\r\n }\r\n\r\n return pickingInfo || new PickingInfo();\r\n};\r\n\r\nScene.prototype._internalMultiPick = function (\r\n rayFunction: (world: Matrix, enableDistantPicking: boolean) => Ray,\r\n predicate?: (mesh: AbstractMesh) => boolean,\r\n trianglePredicate?: TrianglePickingPredicate\r\n): Nullable {\r\n if (!PickingInfo) {\r\n return null;\r\n }\r\n const pickingInfos = new Array();\r\n const computeWorldMatrixForCamera = !!(this.activeCameras && this.activeCameras.length > 1 && this.cameraToUseForPointers !== this.activeCamera);\r\n const currentCamera = this.cameraToUseForPointers || this.activeCamera;\r\n\r\n for (let meshIndex = 0; meshIndex < this.meshes.length; meshIndex++) {\r\n const mesh = this.meshes[meshIndex];\r\n\r\n if (predicate) {\r\n if (!predicate(mesh)) {\r\n continue;\r\n }\r\n } else if (!mesh.isEnabled() || !mesh.isVisible || !mesh.isPickable) {\r\n continue;\r\n }\r\n\r\n const forceCompute = computeWorldMatrixForCamera && mesh.isWorldMatrixCameraDependent();\r\n const world = mesh.computeWorldMatrix(forceCompute, currentCamera);\r\n\r\n if (mesh.hasThinInstances && (mesh as Mesh).thinInstanceEnablePicking) {\r\n const result = this._internalPickForMesh(null, rayFunction, mesh, world, true, true, trianglePredicate);\r\n if (result) {\r\n const tmpMatrix = TmpVectors.Matrix[1];\r\n const thinMatrices = (mesh as Mesh).thinInstanceGetWorldMatrices();\r\n for (let index = 0; index < thinMatrices.length; index++) {\r\n const thinMatrix = thinMatrices[index];\r\n thinMatrix.multiplyToRef(world, tmpMatrix);\r\n const result = this._internalPickForMesh(null, rayFunction, mesh, tmpMatrix, false, false, trianglePredicate, true);\r\n\r\n if (result) {\r\n result.thinInstanceIndex = index;\r\n pickingInfos.push(result);\r\n }\r\n }\r\n }\r\n } else {\r\n const result = this._internalPickForMesh(null, rayFunction, mesh, world, false, false, trianglePredicate);\r\n\r\n if (result) {\r\n pickingInfos.push(result);\r\n }\r\n }\r\n }\r\n\r\n return pickingInfos;\r\n};\r\n\r\nScene.prototype.pickWithBoundingInfo = function (\r\n x: number,\r\n y: number,\r\n predicate?: (mesh: AbstractMesh) => boolean,\r\n fastCheck?: boolean,\r\n camera?: Nullable\r\n): Nullable {\r\n if (!PickingInfo) {\r\n return null;\r\n }\r\n const result = this._internalPick(\r\n (world) => {\r\n if (!this._tempPickingRay) {\r\n this._tempPickingRay = Ray.Zero();\r\n }\r\n\r\n this.createPickingRayToRef(x, y, world, this._tempPickingRay, camera || null);\r\n return this._tempPickingRay;\r\n },\r\n predicate,\r\n fastCheck,\r\n true\r\n );\r\n if (result) {\r\n result.ray = this.createPickingRay(x, y, Matrix.Identity(), camera || null);\r\n }\r\n return result;\r\n};\r\n\r\nObject.defineProperty(Scene.prototype, \"_pickingAvailable\", {\r\n get: () => true,\r\n enumerable: false,\r\n configurable: false,\r\n});\r\n\r\nScene.prototype.pick = function (\r\n x: number,\r\n y: number,\r\n predicate?: (mesh: AbstractMesh) => boolean,\r\n fastCheck?: boolean,\r\n camera?: Nullable,\r\n trianglePredicate?: TrianglePickingPredicate,\r\n _enableDistantPicking = false\r\n): PickingInfo {\r\n const result = this._internalPick(\r\n (world, enableDistantPicking) => {\r\n if (!this._tempPickingRay) {\r\n this._tempPickingRay = Ray.Zero();\r\n }\r\n\r\n this.createPickingRayToRef(x, y, world, this._tempPickingRay, camera || null, false, enableDistantPicking);\r\n return this._tempPickingRay;\r\n },\r\n predicate,\r\n fastCheck,\r\n false,\r\n trianglePredicate\r\n );\r\n if (result) {\r\n result.ray = this.createPickingRay(x, y, Matrix.Identity(), camera || null);\r\n }\r\n return result;\r\n};\r\n\r\nScene.prototype.pickWithRay = function (\r\n ray: Ray,\r\n predicate?: (mesh: AbstractMesh) => boolean,\r\n fastCheck?: boolean,\r\n trianglePredicate?: TrianglePickingPredicate\r\n): Nullable {\r\n const result = this._internalPick(\r\n (world) => {\r\n if (!this._pickWithRayInverseMatrix) {\r\n this._pickWithRayInverseMatrix = Matrix.Identity();\r\n }\r\n world.invertToRef(this._pickWithRayInverseMatrix);\r\n\r\n if (!this._cachedRayForTransform) {\r\n this._cachedRayForTransform = Ray.Zero();\r\n }\r\n\r\n Ray.TransformToRef(ray, this._pickWithRayInverseMatrix, this._cachedRayForTransform);\r\n return this._cachedRayForTransform;\r\n },\r\n predicate,\r\n fastCheck,\r\n false,\r\n trianglePredicate\r\n );\r\n if (result) {\r\n result.ray = ray;\r\n }\r\n return result;\r\n};\r\n\r\nScene.prototype.multiPick = function (\r\n x: number,\r\n y: number,\r\n predicate?: (mesh: AbstractMesh) => boolean,\r\n camera?: Camera,\r\n trianglePredicate?: TrianglePickingPredicate\r\n): Nullable {\r\n return this._internalMultiPick((world) => this.createPickingRay(x, y, world, camera || null), predicate, trianglePredicate);\r\n};\r\n\r\nScene.prototype.multiPickWithRay = function (ray: Ray, predicate?: (mesh: AbstractMesh) => boolean, trianglePredicate?: TrianglePickingPredicate): Nullable {\r\n return this._internalMultiPick(\r\n (world) => {\r\n if (!this._pickWithRayInverseMatrix) {\r\n this._pickWithRayInverseMatrix = Matrix.Identity();\r\n }\r\n world.invertToRef(this._pickWithRayInverseMatrix);\r\n\r\n if (!this._cachedRayForTransform) {\r\n this._cachedRayForTransform = Ray.Zero();\r\n }\r\n\r\n Ray.TransformToRef(ray, this._pickWithRayInverseMatrix, this._cachedRayForTransform);\r\n return this._cachedRayForTransform;\r\n },\r\n predicate,\r\n trianglePredicate\r\n );\r\n};\r\n\r\nCamera.prototype.getForwardRay = function (length = 100, transform?: Matrix, origin?: Vector3): Ray {\r\n return this.getForwardRayToRef(new Ray(Vector3.Zero(), Vector3.Zero(), length), length, transform, origin);\r\n};\r\n\r\nCamera.prototype.getForwardRayToRef = function (refRay: Ray, length = 100, transform?: Matrix, origin?: Vector3): Ray {\r\n if (!transform) {\r\n transform = this.getWorldMatrix();\r\n }\r\n refRay.length = length;\r\n\r\n if (!origin) {\r\n refRay.origin.copyFrom(this.position);\r\n } else {\r\n refRay.origin.copyFrom(origin);\r\n }\r\n TmpVectors.Vector3[2].set(0, 0, this._scene.useRightHandedSystem ? -1 : 1);\r\n Vector3.TransformNormalToRef(TmpVectors.Vector3[2], transform, TmpVectors.Vector3[3]);\r\n\r\n Vector3.NormalizeToRef(TmpVectors.Vector3[3], refRay.direction);\r\n\r\n return refRay;\r\n};\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport { Vector3 } from \"../Maths/math.vector\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { Color3 } from \"../Maths/math.color\";\r\nimport { TmpVectors } from \"./math\";\r\n\r\n// https://dickyjim.wordpress.com/2013/09/04/spherical-harmonics-for-beginners/\r\n// http://silviojemma.com/public/papers/lighting/spherical-harmonic-lighting.pdf\r\n// https://www.ppsloan.org/publications/StupidSH36.pdf\r\n// http://cseweb.ucsd.edu/~ravir/papers/envmap/envmap.pdf\r\n// https://www.ppsloan.org/publications/SHJCGT.pdf\r\n// https://www.ppsloan.org/publications/shdering.pdf\r\n// https://google.github.io/filament/Filament.md.html#annex/sphericalharmonics\r\n// https://patapom.com/blog/SHPortal/\r\n// https://imdoingitwrong.wordpress.com/2011/04/14/spherical-harmonics-wtf/\r\n\r\n// Using real SH basis:\r\n// m>0 m m\r\n// y = sqrt(2) * K * P * cos(m*phi) * cos(theta)\r\n// l l l\r\n//\r\n// m<0 m |m|\r\n// y = sqrt(2) * K * P * sin(m*phi) * cos(theta)\r\n// l l l\r\n//\r\n// m=0 0 0\r\n// y = K * P * trigono terms\r\n// l l l\r\n//\r\n// m (2l + 1)(l - |m|)!\r\n// K = sqrt(------------------)\r\n// l 4pi(l + |m|)!\r\n//\r\n// and P by recursion:\r\n//\r\n// P00(x) = 1\r\n// P01(x) = x\r\n// Pll(x) = (-1^l)(2l - 1)!!(1-x*x)^(1/2)\r\n// ((2l - 1)x[Pl-1/m]-(l + m - 1)[Pl-2/m])\r\n// Plm(x) = ---------------------------------------\r\n// l - m\r\n// Leaving the trigonometric terms aside we can precompute the constants to :\r\nconst SH3ylmBasisConstants = [\r\n Math.sqrt(1 / (4 * Math.PI)), // l00\r\n\r\n -Math.sqrt(3 / (4 * Math.PI)), // l1_1\r\n Math.sqrt(3 / (4 * Math.PI)), // l10\r\n -Math.sqrt(3 / (4 * Math.PI)), // l11\r\n\r\n Math.sqrt(15 / (4 * Math.PI)), // l2_2\r\n -Math.sqrt(15 / (4 * Math.PI)), // l2_1\r\n Math.sqrt(5 / (16 * Math.PI)), // l20\r\n -Math.sqrt(15 / (4 * Math.PI)), // l21\r\n Math.sqrt(15 / (16 * Math.PI)), // l22\r\n];\r\n\r\n// cm = cos(m * phi)\r\n// sm = sin(m * phi)\r\n// {x,y,z} = {cos(phi)sin(theta), sin(phi)sin(theta), cos(theta)}\r\n// By recursion on using trigo identities:\r\nconst SH3ylmBasisTrigonometricTerms = [\r\n () => 1, // l00\r\n\r\n (direction: Vector3) => direction.y, // l1_1\r\n (direction: Vector3) => direction.z, // l10\r\n (direction: Vector3) => direction.x, // l11\r\n\r\n (direction: Vector3) => direction.x * direction.y, // l2_2\r\n (direction: Vector3) => direction.y * direction.z, // l2_1\r\n (direction: Vector3) => 3 * direction.z * direction.z - 1, // l20\r\n (direction: Vector3) => direction.x * direction.z, // l21\r\n (direction: Vector3) => direction.x * direction.x - direction.y * direction.y, // l22\r\n];\r\n\r\n// Wrap the full compute\r\nconst applySH3 = (lm: number, direction: Vector3) => {\r\n return SH3ylmBasisConstants[lm] * SH3ylmBasisTrigonometricTerms[lm](direction);\r\n};\r\n\r\n// Derived from the integration of the a kernel convolution to SH.\r\n// Great explanation here: https://patapom.com/blog/SHPortal/#about-distant-radiance-and-irradiance-environments\r\nconst SHCosKernelConvolution = [Math.PI, (2 * Math.PI) / 3, (2 * Math.PI) / 3, (2 * Math.PI) / 3, Math.PI / 4, Math.PI / 4, Math.PI / 4, Math.PI / 4, Math.PI / 4];\r\n\r\n/**\r\n * Class representing spherical harmonics coefficients to the 3rd degree\r\n */\r\nexport class SphericalHarmonics {\r\n /**\r\n * Defines whether or not the harmonics have been prescaled for rendering.\r\n */\r\n public preScaled = false;\r\n\r\n /**\r\n * The l0,0 coefficients of the spherical harmonics\r\n */\r\n public l00: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * The l1,-1 coefficients of the spherical harmonics\r\n */\r\n public l1_1: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * The l1,0 coefficients of the spherical harmonics\r\n */\r\n public l10: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * The l1,1 coefficients of the spherical harmonics\r\n */\r\n public l11: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * The l2,-2 coefficients of the spherical harmonics\r\n */\r\n public l2_2: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * The l2,-1 coefficients of the spherical harmonics\r\n */\r\n public l2_1: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * The l2,0 coefficients of the spherical harmonics\r\n */\r\n public l20: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * The l2,1 coefficients of the spherical harmonics\r\n */\r\n public l21: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * The l2,2 coefficients of the spherical harmonics\r\n */\r\n public l22: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * Adds a light to the spherical harmonics\r\n * @param direction the direction of the light\r\n * @param color the color of the light\r\n * @param deltaSolidAngle the delta solid angle of the light\r\n */\r\n public addLight(direction: Vector3, color: Color3, deltaSolidAngle: number): void {\r\n TmpVectors.Vector3[0].set(color.r, color.g, color.b);\r\n const colorVector = TmpVectors.Vector3[0];\r\n const c = TmpVectors.Vector3[1];\r\n colorVector.scaleToRef(deltaSolidAngle, c);\r\n\r\n c.scaleToRef(applySH3(0, direction), TmpVectors.Vector3[2]);\r\n this.l00.addInPlace(TmpVectors.Vector3[2]);\r\n\r\n c.scaleToRef(applySH3(1, direction), TmpVectors.Vector3[2]);\r\n this.l1_1.addInPlace(TmpVectors.Vector3[2]);\r\n c.scaleToRef(applySH3(2, direction), TmpVectors.Vector3[2]);\r\n this.l10.addInPlace(TmpVectors.Vector3[2]);\r\n c.scaleToRef(applySH3(3, direction), TmpVectors.Vector3[2]);\r\n this.l11.addInPlace(TmpVectors.Vector3[2]);\r\n\r\n c.scaleToRef(applySH3(4, direction), TmpVectors.Vector3[2]);\r\n this.l2_2.addInPlace(TmpVectors.Vector3[2]);\r\n c.scaleToRef(applySH3(5, direction), TmpVectors.Vector3[2]);\r\n this.l2_1.addInPlace(TmpVectors.Vector3[2]);\r\n c.scaleToRef(applySH3(6, direction), TmpVectors.Vector3[2]);\r\n this.l20.addInPlace(TmpVectors.Vector3[2]);\r\n c.scaleToRef(applySH3(7, direction), TmpVectors.Vector3[2]);\r\n this.l21.addInPlace(TmpVectors.Vector3[2]);\r\n c.scaleToRef(applySH3(8, direction), TmpVectors.Vector3[2]);\r\n this.l22.addInPlace(TmpVectors.Vector3[2]);\r\n }\r\n\r\n /**\r\n * Scales the spherical harmonics by the given amount\r\n * @param scale the amount to scale\r\n */\r\n public scaleInPlace(scale: number): void {\r\n this.l00.scaleInPlace(scale);\r\n this.l1_1.scaleInPlace(scale);\r\n this.l10.scaleInPlace(scale);\r\n this.l11.scaleInPlace(scale);\r\n this.l2_2.scaleInPlace(scale);\r\n this.l2_1.scaleInPlace(scale);\r\n this.l20.scaleInPlace(scale);\r\n this.l21.scaleInPlace(scale);\r\n this.l22.scaleInPlace(scale);\r\n }\r\n\r\n /**\r\n * Convert from incident radiance (Li) to irradiance (E) by applying convolution with the cosine-weighted hemisphere.\r\n *\r\n * ```\r\n * E_lm = A_l * L_lm\r\n * ```\r\n *\r\n * In spherical harmonics this convolution amounts to scaling factors for each frequency band.\r\n * This corresponds to equation 5 in \"An Efficient Representation for Irradiance Environment Maps\", where\r\n * the scaling factors are given in equation 9.\r\n */\r\n public convertIncidentRadianceToIrradiance(): void {\r\n // Constant (Band 0)\r\n this.l00.scaleInPlace(SHCosKernelConvolution[0]);\r\n\r\n // Linear (Band 1)\r\n this.l1_1.scaleInPlace(SHCosKernelConvolution[1]);\r\n this.l10.scaleInPlace(SHCosKernelConvolution[2]);\r\n this.l11.scaleInPlace(SHCosKernelConvolution[3]);\r\n\r\n // Quadratic (Band 2)\r\n this.l2_2.scaleInPlace(SHCosKernelConvolution[4]);\r\n this.l2_1.scaleInPlace(SHCosKernelConvolution[5]);\r\n this.l20.scaleInPlace(SHCosKernelConvolution[6]);\r\n this.l21.scaleInPlace(SHCosKernelConvolution[7]);\r\n this.l22.scaleInPlace(SHCosKernelConvolution[8]);\r\n }\r\n\r\n /**\r\n * Convert from irradiance to outgoing radiance for Lambertian BDRF, suitable for efficient shader evaluation.\r\n *\r\n * ```\r\n * L = (1/pi) * E * rho\r\n * ```\r\n *\r\n * This is done by an additional scale by 1/pi, so is a fairly trivial operation but important conceptually.\r\n */\r\n public convertIrradianceToLambertianRadiance(): void {\r\n this.scaleInPlace(1.0 / Math.PI);\r\n\r\n // The resultant SH now represents outgoing radiance, so includes the Lambert 1/pi normalisation factor but without albedo (rho) applied\r\n // (The pixel shader must apply albedo after texture fetches, etc).\r\n }\r\n\r\n /**\r\n * Integrates the reconstruction coefficients directly in to the SH preventing further\r\n * required operations at run time.\r\n *\r\n * This is simply done by scaling back the SH with Ylm constants parameter.\r\n * The trigonometric part being applied by the shader at run time.\r\n */\r\n public preScaleForRendering(): void {\r\n this.preScaled = true;\r\n\r\n this.l00.scaleInPlace(SH3ylmBasisConstants[0]);\r\n\r\n this.l1_1.scaleInPlace(SH3ylmBasisConstants[1]);\r\n this.l10.scaleInPlace(SH3ylmBasisConstants[2]);\r\n this.l11.scaleInPlace(SH3ylmBasisConstants[3]);\r\n\r\n this.l2_2.scaleInPlace(SH3ylmBasisConstants[4]);\r\n this.l2_1.scaleInPlace(SH3ylmBasisConstants[5]);\r\n this.l20.scaleInPlace(SH3ylmBasisConstants[6]);\r\n this.l21.scaleInPlace(SH3ylmBasisConstants[7]);\r\n this.l22.scaleInPlace(SH3ylmBasisConstants[8]);\r\n }\r\n\r\n /**\r\n * update the spherical harmonics coefficients from the given array\r\n * @param data defines the 9x3 coefficients (l00, l1-1, l10, l11, l2-2, l2-1, l20, l21, l22)\r\n * @returns the spherical harmonics (this)\r\n */\r\n public updateFromArray(data: ArrayLike>): SphericalHarmonics {\r\n Vector3.FromArrayToRef(data[0], 0, this.l00);\r\n Vector3.FromArrayToRef(data[1], 0, this.l1_1);\r\n Vector3.FromArrayToRef(data[2], 0, this.l10);\r\n Vector3.FromArrayToRef(data[3], 0, this.l11);\r\n Vector3.FromArrayToRef(data[4], 0, this.l2_2);\r\n Vector3.FromArrayToRef(data[5], 0, this.l2_1);\r\n Vector3.FromArrayToRef(data[6], 0, this.l20);\r\n Vector3.FromArrayToRef(data[7], 0, this.l21);\r\n Vector3.FromArrayToRef(data[8], 0, this.l22);\r\n return this;\r\n }\r\n\r\n /**\r\n * update the spherical harmonics coefficients from the given floats array\r\n * @param data defines the 9x3 coefficients (l00, l1-1, l10, l11, l2-2, l2-1, l20, l21, l22)\r\n * @returns the spherical harmonics (this)\r\n */\r\n public updateFromFloatsArray(data: ArrayLike): SphericalHarmonics {\r\n Vector3.FromFloatsToRef(data[0], data[1], data[2], this.l00);\r\n Vector3.FromFloatsToRef(data[3], data[4], data[5], this.l1_1);\r\n Vector3.FromFloatsToRef(data[6], data[7], data[8], this.l10);\r\n Vector3.FromFloatsToRef(data[9], data[10], data[11], this.l11);\r\n Vector3.FromFloatsToRef(data[12], data[13], data[14], this.l2_2);\r\n Vector3.FromFloatsToRef(data[15], data[16], data[17], this.l2_1);\r\n Vector3.FromFloatsToRef(data[18], data[19], data[20], this.l20);\r\n Vector3.FromFloatsToRef(data[21], data[22], data[23], this.l21);\r\n Vector3.FromFloatsToRef(data[24], data[25], data[26], this.l22);\r\n return this;\r\n }\r\n\r\n /**\r\n * Constructs a spherical harmonics from an array.\r\n * @param data defines the 9x3 coefficients (l00, l1-1, l10, l11, l2-2, l2-1, l20, l21, l22)\r\n * @returns the spherical harmonics\r\n */\r\n public static FromArray(data: ArrayLike>): SphericalHarmonics {\r\n const sh = new SphericalHarmonics();\r\n return sh.updateFromArray(data);\r\n }\r\n\r\n // Keep for references.\r\n /**\r\n * Gets the spherical harmonics from polynomial\r\n * @param polynomial the spherical polynomial\r\n * @returns the spherical harmonics\r\n */\r\n public static FromPolynomial(polynomial: SphericalPolynomial): SphericalHarmonics {\r\n const result = new SphericalHarmonics();\r\n\r\n result.l00 = polynomial.xx.scale(0.376127).add(polynomial.yy.scale(0.376127)).add(polynomial.zz.scale(0.376126));\r\n result.l1_1 = polynomial.y.scale(0.977204);\r\n result.l10 = polynomial.z.scale(0.977204);\r\n result.l11 = polynomial.x.scale(0.977204);\r\n result.l2_2 = polynomial.xy.scale(1.16538);\r\n result.l2_1 = polynomial.yz.scale(1.16538);\r\n result.l20 = polynomial.zz.scale(1.34567).subtract(polynomial.xx.scale(0.672834)).subtract(polynomial.yy.scale(0.672834));\r\n result.l21 = polynomial.zx.scale(1.16538);\r\n result.l22 = polynomial.xx.scale(1.16538).subtract(polynomial.yy.scale(1.16538));\r\n\r\n result.l1_1.scaleInPlace(-1);\r\n result.l11.scaleInPlace(-1);\r\n result.l2_1.scaleInPlace(-1);\r\n result.l21.scaleInPlace(-1);\r\n\r\n result.scaleInPlace(Math.PI);\r\n\r\n return result;\r\n }\r\n}\r\n\r\n/**\r\n * Class representing spherical polynomial coefficients to the 3rd degree\r\n */\r\nexport class SphericalPolynomial {\r\n private _harmonics: Nullable;\r\n\r\n /**\r\n * The spherical harmonics used to create the polynomials.\r\n */\r\n public get preScaledHarmonics(): SphericalHarmonics {\r\n if (!this._harmonics) {\r\n this._harmonics = SphericalHarmonics.FromPolynomial(this);\r\n }\r\n if (!this._harmonics.preScaled) {\r\n this._harmonics.preScaleForRendering();\r\n }\r\n return this._harmonics;\r\n }\r\n\r\n /**\r\n * The x coefficients of the spherical polynomial\r\n */\r\n public x: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * The y coefficients of the spherical polynomial\r\n */\r\n public y: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * The z coefficients of the spherical polynomial\r\n */\r\n public z: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * The xx coefficients of the spherical polynomial\r\n */\r\n public xx: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * The yy coefficients of the spherical polynomial\r\n */\r\n public yy: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * The zz coefficients of the spherical polynomial\r\n */\r\n public zz: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * The xy coefficients of the spherical polynomial\r\n */\r\n public xy: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * The yz coefficients of the spherical polynomial\r\n */\r\n public yz: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * The zx coefficients of the spherical polynomial\r\n */\r\n public zx: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * Adds an ambient color to the spherical polynomial\r\n * @param color the color to add\r\n */\r\n public addAmbient(color: Color3): void {\r\n TmpVectors.Vector3[0].copyFromFloats(color.r, color.g, color.b);\r\n const colorVector = TmpVectors.Vector3[0];\r\n this.xx.addInPlace(colorVector);\r\n this.yy.addInPlace(colorVector);\r\n this.zz.addInPlace(colorVector);\r\n }\r\n\r\n /**\r\n * Scales the spherical polynomial by the given amount\r\n * @param scale the amount to scale\r\n */\r\n public scaleInPlace(scale: number) {\r\n this.x.scaleInPlace(scale);\r\n this.y.scaleInPlace(scale);\r\n this.z.scaleInPlace(scale);\r\n this.xx.scaleInPlace(scale);\r\n this.yy.scaleInPlace(scale);\r\n this.zz.scaleInPlace(scale);\r\n this.yz.scaleInPlace(scale);\r\n this.zx.scaleInPlace(scale);\r\n this.xy.scaleInPlace(scale);\r\n }\r\n\r\n /**\r\n * Updates the spherical polynomial from harmonics\r\n * @param harmonics the spherical harmonics\r\n * @returns the spherical polynomial\r\n */\r\n public updateFromHarmonics(harmonics: SphericalHarmonics): SphericalPolynomial {\r\n this._harmonics = harmonics;\r\n\r\n this.x.copyFrom(harmonics.l11);\r\n this.x.scaleInPlace(1.02333).scaleInPlace(-1);\r\n this.y.copyFrom(harmonics.l1_1);\r\n this.y.scaleInPlace(1.02333).scaleInPlace(-1);\r\n this.z.copyFrom(harmonics.l10);\r\n this.z.scaleInPlace(1.02333);\r\n\r\n this.xx.copyFrom(harmonics.l00);\r\n TmpVectors.Vector3[0].copyFrom(harmonics.l20).scaleInPlace(0.247708);\r\n TmpVectors.Vector3[1].copyFrom(harmonics.l22).scaleInPlace(0.429043);\r\n this.xx.scaleInPlace(0.886277).subtractInPlace(TmpVectors.Vector3[0]).addInPlace(TmpVectors.Vector3[1]);\r\n this.yy.copyFrom(harmonics.l00);\r\n this.yy.scaleInPlace(0.886277).subtractInPlace(TmpVectors.Vector3[0]).subtractInPlace(TmpVectors.Vector3[1]);\r\n this.zz.copyFrom(harmonics.l00);\r\n TmpVectors.Vector3[0].copyFrom(harmonics.l20).scaleInPlace(0.495417);\r\n this.zz.scaleInPlace(0.886277).addInPlace(TmpVectors.Vector3[0]);\r\n\r\n this.yz.copyFrom(harmonics.l2_1);\r\n this.yz.scaleInPlace(0.858086).scaleInPlace(-1);\r\n this.zx.copyFrom(harmonics.l21);\r\n this.zx.scaleInPlace(0.858086).scaleInPlace(-1);\r\n this.xy.copyFrom(harmonics.l2_2);\r\n this.xy.scaleInPlace(0.858086);\r\n\r\n this.scaleInPlace(1.0 / Math.PI);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Gets the spherical polynomial from harmonics\r\n * @param harmonics the spherical harmonics\r\n * @returns the spherical polynomial\r\n */\r\n public static FromHarmonics(harmonics: SphericalHarmonics): SphericalPolynomial {\r\n const result = new SphericalPolynomial();\r\n return result.updateFromHarmonics(harmonics);\r\n }\r\n\r\n /**\r\n * Constructs a spherical polynomial from an array.\r\n * @param data defines the 9x3 coefficients (x, y, z, xx, yy, zz, yz, zx, xy)\r\n * @returns the spherical polynomial\r\n */\r\n public static FromArray(data: ArrayLike>): SphericalPolynomial {\r\n const sp = new SphericalPolynomial();\r\n Vector3.FromArrayToRef(data[0], 0, sp.x);\r\n Vector3.FromArrayToRef(data[1], 0, sp.y);\r\n Vector3.FromArrayToRef(data[2], 0, sp.z);\r\n Vector3.FromArrayToRef(data[3], 0, sp.xx);\r\n Vector3.FromArrayToRef(data[4], 0, sp.yy);\r\n Vector3.FromArrayToRef(data[5], 0, sp.zz);\r\n Vector3.FromArrayToRef(data[6], 0, sp.yz);\r\n Vector3.FromArrayToRef(data[7], 0, sp.zx);\r\n Vector3.FromArrayToRef(data[8], 0, sp.xy);\r\n return sp;\r\n }\r\n}\r\n","import { ArcRotateCamera } from '@babylonjs/core/Cameras/arcRotateCamera';\r\nimport { Camera } from '@babylonjs/core/Cameras/camera';\r\nimport { Ray } from '@babylonjs/core/Culling/ray';\r\nimport { Matrix, Quaternion, Vector3 } from '@babylonjs/core/Maths';\r\nimport { MeshNode } from '@models/classes/meshes/mesh-node';\r\nimport { KbQuaternion } from '@models/classes/primitives/kb-quaternion';\r\nimport type { IKbVectorLike } from '@models/classes/primitives/kb-vector';\r\n\r\nconst ray1 = Ray.Zero();\r\nconst vector1 = Vector3.Zero();\r\nconst vector2 = Vector3.Zero();\r\nconst vector3 = Vector3.Zero();\r\n\r\nexport type RoundDirection = 'ceil' | 'floor' | 'round';\r\n\r\nexport function getCameraPositionWithOffset(camera: Camera) {\r\n if (camera instanceof ArcRotateCamera) {\r\n camera.getForwardRayToRef(ray1);\r\n Vector3.CrossToRef(ray1.direction, Vector3.Up(), vector1);\r\n Vector3.CrossToRef(ray1.direction, vector1, vector2);\r\n vector1.normalize().scaleInPlace(-camera.targetScreenOffset.x);\r\n vector2.normalize().scaleInPlace(camera.targetScreenOffset.y);\r\n vector3.copyFrom(camera.globalPosition);\r\n vector3.addInPlace(vector1).addInPlace(vector2);\r\n\r\n return vector3;\r\n } else {\r\n return camera.globalPosition;\r\n }\r\n}\r\n\r\nexport function toRadians(degrees: number) {\r\n const scalar = Math.PI / 180;\r\n return degrees * scalar;\r\n}\r\n\r\nexport function roundVectorToPlaces(vector: Vector3, significance = 1, direction?: RoundDirection) {\r\n vector.x = roundToPlaces(vector.x, significance, direction);\r\n vector.y = roundToPlaces(vector.y, significance, direction);\r\n vector.z = roundToPlaces(vector.z, significance, direction);\r\n}\r\n\r\nexport function roundToPlaces(value: number, significance = 1, direction: RoundDirection = 'round') {\r\n const isRound = Math.round(value / significance) * significance === value;\r\n if (isRound) {\r\n return value;\r\n }\r\n const n = (value / significance) * (1 + Number.EPSILON);\r\n let tmp: number;\r\n if (direction === 'ceil') {\r\n tmp = Math.ceil(n);\r\n } else if (direction === 'floor') {\r\n tmp = Math.floor(n);\r\n } else {\r\n tmp = Math.round(n);\r\n }\r\n return tmp * significance;\r\n}\r\n\r\nexport function rotationToVector(toVector: Vector3, fromVector?: Vector3) {\r\n const result = new Quaternion();\r\n if (!fromVector) {\r\n fromVector = new Vector3(0, 1, 0);\r\n }\r\n rotationToVectorToRef(toVector, fromVector, result);\r\n\r\n return result;\r\n}\r\n\r\nexport function transformVectorWithQuat(vector: Vector3, quat: Quaternion) {\r\n const transform = new Matrix();\r\n quat.toRotationMatrix(transform);\r\n const transformVector = Vector3.TransformCoordinates(vector, transform);\r\n return transformVector;\r\n}\r\n\r\nexport function applyTransformToVector(vector: IKbVectorLike, transform: Matrix) {\r\n const v = new Vector3(vector.x, vector.y, vector.z);\r\n const matrix = Matrix.Compose(new Vector3(1, 1, 1), Quaternion.Identity(), v);\r\n return applyTransformToSpace(matrix, transform);\r\n}\r\n\r\nexport function applyTransformToSpace(worldTransform: Matrix | number[], inSpace?: Matrix) {\r\n if (!inSpace) {\r\n inSpace = Matrix.Identity();\r\n }\r\n const inSpaceInv = inSpace.clone().invert();\r\n\r\n if (Array.isArray(worldTransform)) {\r\n worldTransform = Matrix.FromArray(worldTransform);\r\n }\r\n\r\n const spaceTransform = inSpace.multiply(worldTransform);\r\n spaceTransform.multiplyToRef(inSpaceInv, spaceTransform);\r\n\r\n const t = new Vector3();\r\n const q = new Quaternion();\r\n const s = new Vector3();\r\n spaceTransform.decompose(s, q, t);\r\n\r\n return {\r\n position: { x: t.x, y: t.y, z: t.z } as IKbVectorLike,\r\n rotation: KbQuaternion.FromQuat(q),\r\n scaling: { x: s.x, y: s.y, z: s.z } as IKbVectorLike,\r\n };\r\n\r\n // const fromSpaceRotation = fromSpace.rotationQuaternion ? fromSpace.rotationQuaternion : Quaternion.Identity();\r\n // const toSpaceRotation = toSpace.rotationQuaternion ? toSpace.rotationQuaternion : Quaternion.Identity();\r\n\r\n // const fromSpaceRotationInverse = fromSpaceRotation.clone().invert();\r\n // const toSpaceRotationInverse = toSpaceRotation.clone().invert();\r\n\r\n // const q1 = q.clone().multiply(fromSpaceRotationInverse);\r\n // const q2 = toSpaceRotationInverse.multiply(q1);\r\n // const q3 = q2.multiply(toSpaceRotation);\r\n\r\n // return q3;\r\n}\r\n\r\nexport function computeLocalTransform(node: MeshNode) {\r\n let rotation: Quaternion;\r\n if (node.useEuler) {\r\n rotation = Quaternion.FromEulerAngles(\r\n toRadians(node.eulerRotation.x),\r\n toRadians(node.eulerRotation.y),\r\n toRadians(node.eulerRotation.z)\r\n );\r\n } else {\r\n rotation = Quaternion.RotationAxis(node.rotationAxis.toVec3(), node.rotationAngle * (Math.PI / 180));\r\n }\r\n\r\n const pivotTransform = node.pivot ? Matrix.Translation(-node.pivot.x, -node.pivot.y, -node.pivot.z) : undefined;\r\n\r\n const scalingTransform = Matrix.Scaling(node.scaling, node.scaling, node.scaling);\r\n const rotationTransform = new Matrix();\r\n Matrix.FromQuaternionToRef(rotation, rotationTransform);\r\n const translationTransform = Matrix.Translation(node.position.x, node.position.y, node.position.z);\r\n\r\n const transform = Matrix.Identity();\r\n if (pivotTransform) {\r\n transform.multiplyToRef(pivotTransform, transform);\r\n }\r\n transform.multiplyToRef(scalingTransform, transform);\r\n transform.multiplyToRef(rotationTransform, transform);\r\n transform.multiplyToRef(translationTransform, transform);\r\n if (pivotTransform) {\r\n transform.multiplyToRef(pivotTransform.invert(), transform);\r\n }\r\n\r\n return transform;\r\n}\r\n\r\nexport function rotationToVectorToRef(_toVector: Vector3, _fromVector: Vector3, result: Quaternion) {\r\n const toVector = _toVector.clone().normalize();\r\n const fromVector = _fromVector.clone().normalize();\r\n\r\n if (Vector3.Dot(fromVector, toVector) > 0.9999999) {\r\n result.set(0, 0, 0, 1); // Identity Quaternion\r\n }\r\n if (Vector3.Dot(fromVector, toVector) < -0.9999999) {\r\n // Rotate 180 degrees about arbitrary axis\r\n return Quaternion.RotationAxisToRef(\r\n Vector3.Right().equals(toVector) ? Vector3.Up().cross(toVector) : Vector3.Right().cross(toVector),\r\n Math.PI,\r\n result\r\n );\r\n }\r\n const rotAxis = toVector.cross(fromVector);\r\n const w = -Math.acos(Vector3.Dot(fromVector, toVector));\r\n\r\n return Quaternion.RotationAxisToRef(rotAxis, w, result);\r\n}\r\n\r\nexport function degreeToRadian(vectorDegree: Vector3, optionalResult?: Vector3): Vector3 {\r\n optionalResult = optionalResult || new Vector3();\r\n if (vectorDegree !== optionalResult) {\r\n optionalResult.copyFrom(vectorDegree);\r\n }\r\n return multiplyScalar(optionalResult, Math.PI / 180);\r\n}\r\n\r\nexport function multiplyScalar(vec: Vector3, scalar: number): Vector3 {\r\n vec.x *= scalar;\r\n vec.y *= scalar;\r\n vec.z *= scalar;\r\n return vec;\r\n}\r\n\r\nexport function computeNormal(normals: Vector3[]): Vector3 {\r\n const vertex1 = normals[0];\r\n const vertex2 = normals[1];\r\n const vertex3 = normals[2];\r\n const p1p2 = vertex1.subtract(vertex2);\r\n const p3p2 = vertex3.subtract(vertex2);\r\n\r\n const normal = Vector3.Normalize(Vector3.Cross(p1p2, p3p2));\r\n return normal;\r\n}\r\n\r\nexport function applyMatrix4(thisVect: Vector3, m: Matrix) {\r\n const x = thisVect.x,\r\n y = thisVect.y,\r\n z = thisVect.z;\r\n const e = m.asArray();\r\n thisVect.x = e[0] * x + e[4] * y + e[8] * z + e[12];\r\n thisVect.y = e[1] * x + e[5] * y + e[9] * z + e[13];\r\n thisVect.z = e[2] * x + e[6] * y + e[10] * z + e[14];\r\n return thisVect;\r\n}\r\n\r\nexport function mathClamp(x: number, a: number, b: number): number {\r\n return x < a ? a : x > b ? b : x;\r\n}\r\n\r\nexport function makeRotationAxis(axis: Vector3, angle: number): Matrix {\r\n const c = Math.cos(angle);\r\n const s = Math.sin(angle);\r\n const t = 1 - c;\r\n const x = axis.x,\r\n y = axis.y,\r\n z = axis.z;\r\n const tx = t * x,\r\n ty = t * y;\r\n const matrix = Matrix.FromValues(\r\n tx * x + c,\r\n tx * y - s * z,\r\n tx * z + s * y,\r\n 0,\r\n tx * y + s * z,\r\n ty * y + c,\r\n ty * z - s * x,\r\n 0,\r\n tx * z - s * y,\r\n ty * z + s * x,\r\n t * z * z + c,\r\n 0,\r\n 0,\r\n 0,\r\n 0,\r\n 1\r\n );\r\n return matrix;\r\n}\r\n\r\nexport function getRotationBtwVectors(u: Vector3, v: Vector3) {\r\n //http://lolengine.net/blog/2014/02/24/quaternion-from-two-vectors-final\r\n v.clone().normalize();\r\n u.clone().normalize();\r\n const dot = Vector3.Dot(u, v);\r\n const k = 1; //we are using unit vectors, so just use \"1\" instead of: Math.sqrt(u.length()^2 * v.length()^2);\r\n let w = k + dot;\r\n let r: Vector3;\r\n\r\n if (w < 0.0000001) {\r\n /* If u and v are exactly opposite, rotate 180 degrees\r\n * around an arbitrary orthogonal axis. Axis normalisation\r\n * can happen later, when we normalise the quaternion. */\r\n w = 0;\r\n r = Math.abs(u.x) > Math.abs(u.z) ? new Vector3(-u.y, u.x, 0) : new Vector3(0, -u.z, u.y);\r\n } else {\r\n r = Vector3.Cross(u, v);\r\n }\r\n\r\n return new Quaternion(r.x, r.y, r.z, w).normalize();\r\n}\r\n\r\nexport function addArrays(arrayA: number[], arrayB: number[]) {\r\n if (arrayA.length == 0) {\r\n return arrayB;\r\n }\r\n if (arrayB.length == 0) {\r\n return arrayA;\r\n }\r\n const result = [];\r\n const n = Math.min(arrayA.length, arrayB.length);\r\n for (let i = 0; i < n; i++) {\r\n result[i] = arrayA[i] + arrayB[i];\r\n }\r\n return result;\r\n}\r\n\r\nexport enum AxisType {\r\n X = 1,\r\n Y = 2,\r\n Z = 3,\r\n Custom = 4,\r\n}\r\n","import { Ignore } from '../decorators/inject-decorator';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { IKbQuaternion } from '../kb3d-models';\r\nimport { Token } from '../token';\r\nimport { KbVector } from './kb-vector';\r\n\r\nexport interface IKbQuaternionLike {\r\n x: number;\r\n y: number;\r\n z: number;\r\n w: number;\r\n}\r\n\r\n@Kb3dModel({\r\n token: Token.KbQuaternion,\r\n trackChanges: false,\r\n create: args => new KbQuaternion(args.x || 0, args.y || 0, args.z || 0, args.w || 1),\r\n})\r\nexport class KbQuaternion implements IKbQuaternion {\r\n constructor(\r\n @Ignore() readonly x: number,\r\n @Ignore() readonly y: number,\r\n @Ignore() readonly z: number,\r\n @Ignore() readonly w: number\r\n ) {\r\n if (this.w > 1) {\r\n this.w = 1;\r\n }\r\n if (this.w < -1) {\r\n this.w = -1;\r\n }\r\n }\r\n\r\n clone() {\r\n return new KbQuaternion(this.x, this.y, this.z, this.w);\r\n }\r\n\r\n assign(v: { x?: number | null; y?: number | null; z?: number | null; w?: number | null }) {\r\n return new KbQuaternion(\r\n typeof v.x === 'number' ? v.x : this.x,\r\n typeof v.y === 'number' ? v.y : this.y,\r\n typeof v.z === 'number' ? v.z : this.z,\r\n typeof v.w === 'number' ? v.w : this.w\r\n );\r\n }\r\n\r\n toEuler(): KbVector {\r\n let x: number, y: number, z: number;\r\n\r\n const qz = this.z;\r\n const qx = this.x;\r\n const qy = this.y;\r\n const qw = this.w;\r\n const sqw = qw * qw;\r\n const sqz = qz * qz;\r\n const sqx = qx * qx;\r\n const sqy = qy * qy;\r\n const zAxisY = qy * qz - qx * qw;\r\n const limit = 0.4999999;\r\n if (zAxisY < -limit) {\r\n y = 2 * Math.atan2(qy, qw);\r\n x = Math.PI / 2;\r\n z = 0;\r\n } else if (zAxisY > limit) {\r\n y = 2 * Math.atan2(qy, qw);\r\n x = -Math.PI / 2;\r\n z = 0;\r\n } else {\r\n z = Math.atan2(2.0 * (qx * qy + qz * qw), -sqz - sqx + sqy + sqw);\r\n x = Math.asin(-2.0 * (qz * qy - qx * qw));\r\n y = Math.atan2(2.0 * (qz * qx + qy * qw), sqz - sqx - sqy + sqw);\r\n }\r\n\r\n const scalar = 180 / Math.PI;\r\n\r\n return new KbVector(x * scalar, y * scalar, z * scalar);\r\n }\r\n\r\n multiply(q: { x: number; y: number; z: number; w: number }) {\r\n const x = this.x * q.w + this.y * q.z - this.z * q.y + this.w * q.x;\r\n const y = -this.x * q.z + this.y * q.w + this.z * q.x + this.w * q.y;\r\n const z = this.x * q.y - this.y * q.x + this.z * q.w + this.w * q.z;\r\n const w = -this.x * q.x - this.y * q.y - this.z * q.z + this.w * q.w;\r\n\r\n return new KbQuaternion(x, y, z, w);\r\n }\r\n\r\n equal(q: IKbQuaternionLike | undefined): boolean {\r\n return (\r\n !!q &&\r\n basicallyEqual(q.x, this.x) &&\r\n basicallyEqual(q.y, this.y) &&\r\n basicallyEqual(q.z, this.z) &&\r\n basicallyEqual(q.w, this.w)\r\n );\r\n }\r\n\r\n toObject() {\r\n return { x: this.x, y: this.y, z: this.z, w: this.w } as IKbQuaternionLike;\r\n }\r\n\r\n getAngle() {\r\n return this._getRadAngle() * (180 / Math.PI);\r\n }\r\n\r\n private _getRadAngle() {\r\n return Math.acos(this.w) * 2;\r\n }\r\n\r\n getAxis() {\r\n const angle = this._getRadAngle();\r\n const sin = Math.sin(angle / 2);\r\n\r\n if (sin === 0) {\r\n return KbVector.Zero();\r\n } else {\r\n return new KbVector(this.x / sin, this.y / sin, this.z / sin).normalize();\r\n }\r\n }\r\n\r\n static FromRotationAxis(axis: KbVector, angle: number) {\r\n // var sin = Math.sin(angle / 2);\r\n // axis.normalize();\r\n // result.w = Math.cos(angle / 2);\r\n // result.x = axis._x * sin;\r\n // result.y = axis._y * sin;\r\n // result.z = axis._z * sin;\r\n\r\n const rad = (angle * Math.PI) / 180;\r\n const sin = Math.sin(rad / 2);\r\n const nAxis = axis.normalize();\r\n const result = new KbQuaternion(nAxis.x * sin, nAxis.y * sin, nAxis.z * sin, Math.cos(rad / 2));\r\n return result;\r\n }\r\n\r\n static FromYawPitchRoll(dV: KbVector) {\r\n const v = dV.scale(Math.PI / 180);\r\n const halfRoll = v.z * 0.5;\r\n const halfPitch = v.x * 0.5;\r\n const halfYaw = v.y * 0.5;\r\n const sinRoll = Math.sin(halfRoll);\r\n const cosRoll = Math.cos(halfRoll);\r\n const sinPitch = Math.sin(halfPitch);\r\n const cosPitch = Math.cos(halfPitch);\r\n const sinYaw = Math.sin(halfYaw);\r\n const cosYaw = Math.cos(halfYaw);\r\n const result = new KbQuaternion(\r\n cosYaw * sinPitch * cosRoll + sinYaw * cosPitch * sinRoll,\r\n sinYaw * cosPitch * cosRoll - cosYaw * sinPitch * sinRoll,\r\n cosYaw * cosPitch * sinRoll - sinYaw * sinPitch * cosRoll,\r\n cosYaw * cosPitch * cosRoll + sinYaw * sinPitch * sinRoll\r\n );\r\n\r\n return result;\r\n }\r\n\r\n static Identity() {\r\n return new KbQuaternion(0, 0, 0, 1);\r\n }\r\n\r\n static From(q: IKbQuaternionLike) {\r\n return new KbQuaternion(q.x, q.y, q.z, q.w);\r\n }\r\n}\r\n\r\nfunction basicallyEqual(a: number, b: number) {\r\n return Math.abs(a - b) < 0.0000000001; // To deal with floating point errors\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { instanceOf } from '../decorators/model-utils';\r\nimport { IBoxNode } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { Token } from '../token';\r\nimport { PrimitiveMeshNode } from './primitive-mesh-node';\r\n\r\n@Kb3dModel({\r\n token: Token.BoxNode,\r\n})\r\nexport class BoxNode extends PrimitiveMeshNode implements IBoxNode {\r\n public constructor() {\r\n super();\r\n }\r\n public height: number;\r\n public width: number;\r\n public depth: number;\r\n public tessellation: number;\r\n}\r\n\r\nexport function isBoxMesh(node: Kb3dObject): node is BoxNode {\r\n return instanceOf(node, Token.BoxNode);\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { instanceOf } from '../decorators/model-utils';\r\nimport { ICylinderNode } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { Token } from '../token';\r\nimport { PrimitiveMeshNode } from './primitive-mesh-node';\r\n\r\n@Kb3dModel({\r\n token: Token.ConeNode,\r\n})\r\nexport class ConeNode extends PrimitiveMeshNode implements ICylinderNode {\r\n public constructor() {\r\n super();\r\n }\r\n public height: number;\r\n public radiusTop: number;\r\n public radiusBottom: number;\r\n public tessellation: number;\r\n}\r\n\r\nexport function isConeMesh(node: Kb3dObject): node is ConeNode {\r\n return instanceOf(node, Token.ConeNode);\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { instanceOf } from '../decorators/model-utils';\r\nimport { eCylinderCap } from '../enums';\r\nimport { ICylinderNode } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { Token } from '../token';\r\nimport { PrimitiveMeshNode } from './primitive-mesh-node';\r\n\r\n@Kb3dModel({\r\n token: Token.CylinderNode,\r\n})\r\nexport class CylinderNode extends PrimitiveMeshNode implements ICylinderNode {\r\n public constructor() {\r\n super();\r\n }\r\n public height: number;\r\n public radius: number;\r\n public arc: number;\r\n\r\n public tessellation: number;\r\n public subdivisions: number;\r\n public capMode: eCylinderCap;\r\n}\r\n\r\nexport function isCylinderMesh(node: Kb3dObject): node is CylinderNode {\r\n return instanceOf(node, Token.CylinderNode);\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { instanceOf } from '../decorators/model-utils';\r\nimport { IDiscNode } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { Token } from '../token';\r\nimport { PrimitiveMeshNode } from './primitive-mesh-node';\r\n\r\n@Kb3dModel({\r\n token: Token.DiscNode,\r\n})\r\nexport class DiscNode extends PrimitiveMeshNode implements IDiscNode {\r\n public constructor() {\r\n super();\r\n }\r\n public radius: number;\r\n public tessellation: number;\r\n}\r\n\r\nexport function isDiscMesh(node: Kb3dObject): node is DiscNode {\r\n return instanceOf(node, Token.DiscNode);\r\n}\r\n","import { isTrackedInstanceOfAny, ITrackedClass } from '@models/interfaces/tracked-class';\r\nimport { AnnotationNode } from '../annotations';\r\nimport { instanceOf, instanceOfAny } from '../decorators/model-utils';\r\nimport { SketchControlPoint } from '../features/sketch-control-point';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { Token } from '../token';\r\nimport { CustomMeshNode } from './custom-mesh-node';\r\nimport { MeshGroupNode } from './mesh-group-node';\r\nimport { MeshNode } from './mesh-node';\r\nimport { PrimitiveMeshNode } from './primitive-mesh-node';\r\nimport { SketchNode } from './sketch-node';\r\nimport { SpaceNode } from './space-node';\r\nimport { TextNode } from './text-node';\r\n\r\nexport function isMeshGroupNode(node: Kb3dObject): node is MeshGroupNode {\r\n return instanceOf(node, Token.MeshGroupNode);\r\n}\r\n\r\nexport function isCustomMeshNode(node: Kb3dObject): node is CustomMeshNode {\r\n return instanceOf(node, Token.CustomMeshNode);\r\n}\r\n\r\nexport function isSketchNode(node: Kb3dObject): node is SketchNode {\r\n return instanceOf(node, Token.SketchNode);\r\n}\r\n\r\nexport function isTextNode(node: Kb3dObject): node is TextNode {\r\n return instanceOf(node, Token.TextNode);\r\n}\r\n\r\nexport function isSketchControlPoint(node: Kb3dObject): node is SketchControlPoint {\r\n return instanceOf(node, Token.SketchControlPoint);\r\n}\r\n\r\nexport function isMeshNode(node: Kb3dObject): node is MeshNode {\r\n return instanceOfAny(node, ...Token.MeshTokens);\r\n}\r\n\r\nexport function isPrimitiveMeshNode(node: Kb3dObject): node is PrimitiveMeshNode {\r\n return instanceOf(node, Token.PrimitiveMeshNode);\r\n}\r\n\r\nexport function isSpaceNode(node: Kb3dObject): node is SpaceNode {\r\n return instanceOf(node, Token.SpaceNode);\r\n}\r\n\r\nexport function isPickable(node: Kb3dObject): node is SpaceNode | AnnotationNode {\r\n if (node instanceof SpaceNode || node instanceof AnnotationNode) {\r\n return true;\r\n } else {\r\n return false;\r\n }\r\n}\r\n\r\nexport function isTrackedMeshNode(node: Kb3dObject): node is ITrackedClass {\r\n return isMeshNode(node) && isTrackedInstanceOfAny(node, ...Token.MeshAndGroupTokens);\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { instanceOf } from '../decorators/model-utils';\r\nimport { IPlaneNode } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { Token } from '../token';\r\nimport { PrimitiveMeshNode } from './primitive-mesh-node';\r\n\r\n@Kb3dModel({\r\n token: Token.PlaneNode,\r\n})\r\nexport class PlaneNode extends PrimitiveMeshNode implements IPlaneNode {\r\n public constructor() {\r\n super();\r\n }\r\n public depth: number;\r\n public width: number;\r\n public segments: number;\r\n}\r\n\r\nexport function isPlaneMesh(node: Kb3dObject): node is PlaneNode {\r\n return instanceOf(node, Token.PlaneNode);\r\n}\r\n","import { RoundDirection, roundToPlaces } from '@view/helpers/math-helper';\r\nimport { Ignore } from '../decorators/inject-decorator';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { Token } from '../token';\r\n\r\nexport interface IKbVectorLike {\r\n x: number;\r\n y: number;\r\n z: number;\r\n}\r\n\r\n@Kb3dModel({\r\n token: Token.KbVector,\r\n trackChanges: false,\r\n create: args => new KbVector(args.x || 0, args.y || 0, args.z || 0),\r\n})\r\nexport class KbVector implements IKbVectorLike {\r\n constructor(@Ignore() readonly x: number, @Ignore() readonly y: number, @Ignore() readonly z: number) {\r\n if (!isFinite(x) || isNaN(x)) {\r\n this.x = 0;\r\n console.error(\r\n 'Error: Non numeric or infinite value detected in x-value of a vector, please check snapcode for mistakes.'\r\n );\r\n }\r\n if (!isFinite(y) || isNaN(y)) {\r\n this.y = 0;\r\n console.error(\r\n 'Error: Non numeric or infinite value detected in y-value of a vector, please check snapcode for mistakes.'\r\n );\r\n }\r\n if (!isFinite(z) || isNaN(z)) {\r\n this.z = 0;\r\n console.error(\r\n 'Error: Non numeric or infinite value detected in z-value of a vector, please check snapcode for mistakes.'\r\n );\r\n }\r\n }\r\n\r\n add(v: IKbVectorLike): KbVector {\r\n return new KbVector(this.x + v.x, this.y + v.y, this.z + v.z);\r\n }\r\n subtract(v: IKbVectorLike): KbVector {\r\n return new KbVector(this.x - v.x, this.y - v.y, this.z - v.z);\r\n }\r\n multiply(v: IKbVectorLike): KbVector {\r\n return new KbVector(this.x * v.x, this.y * v.y, this.z * v.z);\r\n }\r\n divide(v: { x: number; y: number; z: number }): KbVector {\r\n const newX = v.x !== 0 ? this.x / v.x : 0;\r\n const newY = v.y !== 0 ? this.y / v.y : 0;\r\n const newZ = v.z !== 0 ? this.z / v.z : 0;\r\n return new KbVector(newX, newY, newZ);\r\n }\r\n crossProduct(v: IKbVectorLike) {\r\n const x = this.y * v.z - this.z * v.y;\r\n const y = this.z * v.x - this.x * v.z;\r\n const z = this.x * v.y - this.y * v.x;\r\n return new KbVector(x, y, z);\r\n }\r\n dotProduct(v: IKbVectorLike) {\r\n return this.x * v.x + this.y * v.y + this.z * v.z;\r\n }\r\n length() {\r\n return Math.sqrt(this.x * this.x + this.y * this.y + this.z * this.z);\r\n }\r\n normalize() {\r\n const l = this.length();\r\n if (l === 0 || l === 1.0) {\r\n return this.clone();\r\n }\r\n return this.scale(1.0 / l);\r\n }\r\n scale(s: number): KbVector {\r\n return new KbVector(this.x * s, this.y * s, this.z * s);\r\n }\r\n equal(v: IKbVectorLike): boolean {\r\n return v.x == this.x && v.y == this.y && v.z == this.z;\r\n }\r\n clone(): KbVector {\r\n return new KbVector(this.x, this.y, this.z);\r\n }\r\n assign(v: { x?: number | null; y?: number | null; z?: number | null }) {\r\n return new KbVector(\r\n typeof v.x === 'number' ? v.x : this.x,\r\n typeof v.y === 'number' ? v.y : this.y,\r\n typeof v.z === 'number' ? v.z : this.z\r\n );\r\n }\r\n shiftClone(component: 'x' | 'y' | 'z', amount: number) {\r\n if (component === 'x') {\r\n return new KbVector(amount, this.y, this.z);\r\n } else if (component === 'y') {\r\n return new KbVector(this.x, amount, this.z);\r\n } else {\r\n return new KbVector(this.x, this.y, amount);\r\n }\r\n }\r\n\r\n toString() {\r\n return `{ x: ${this.x}, y: ${this.y}, z: ${this.z} }`;\r\n }\r\n toObject() {\r\n return { x: this.x, y: this.y, z: this.z } as IKbVectorLike;\r\n }\r\n round(significance = 1, direction?: RoundDirection) {\r\n return new KbVector(\r\n roundToPlaces(this.x, significance, direction),\r\n roundToPlaces(this.y, significance, direction),\r\n roundToPlaces(this.z, significance, direction)\r\n );\r\n }\r\n static Zero() {\r\n return new KbVector(0, 0, 0);\r\n }\r\n static Identity() {\r\n return new KbVector(1, 1, 1);\r\n }\r\n static From(v: IKbVectorLike) {\r\n return new KbVector(v.x, v.y, v.z);\r\n }\r\n}\r\n","import { BaseKb3dModel } from '../decorators';\r\nimport { Enumerable } from '../decorators/enumerable';\r\nimport { Connector } from '../features/connector';\r\nimport { Feature } from '../features/feature';\r\nimport { ISpaceNode } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { KbQuaternion } from '../primitives/kb-quaternion';\r\nimport { KbVector } from '../primitives/kb-vector';\r\nimport { Token } from '../token';\r\n\r\n@BaseKb3dModel({\r\n token: Token.SpaceNode,\r\n})\r\nexport abstract class SpaceNode extends Kb3dObject implements ISpaceNode {\r\n constructor() {\r\n super();\r\n // this._rotationQuaternion = nodeConfig.rotationQuaternion ? KbQuaternion.FromVec3(nodeConfig.rotationQuaternion) : KbQuaternion.Identity();\r\n // this._rotation = this._rotationQuaternion.toEuler();\r\n }\r\n visible: boolean;\r\n position: KbVector;\r\n useEuler: boolean;\r\n eulerRotation: KbVector;\r\n rotationAngle: number;\r\n rotationAxis: KbVector;\r\n scaling: number;\r\n pivot: KbVector;\r\n fixed: boolean;\r\n _primeMover: boolean;\r\n /** override base parent for more specific typing*/\r\n _parent: SpaceNode | undefined;\r\n _showPivot = false;\r\n\r\n preTransform?: number[];\r\n\r\n getRotationQuaternion() {\r\n if (this.useEuler && this.eulerRotation) {\r\n return KbQuaternion.FromYawPitchRoll(this.eulerRotation);\r\n } else if (this.rotationAxis && this.rotationAngle !== undefined) {\r\n return KbQuaternion.FromRotationAxis(this.rotationAxis, this.rotationAngle);\r\n }\r\n return KbQuaternion.Identity();\r\n }\r\n setRotationQuaternion(q: KbQuaternion) {\r\n if (this.useEuler) {\r\n this.eulerRotation = q.toEuler();\r\n } else {\r\n this.rotationAngle = q.getAngle();\r\n this.rotationAxis = q.getAxis();\r\n }\r\n }\r\n\r\n private _features: Feature[];\r\n @Enumerable()\r\n public get features(): Feature[] {\r\n if (!this._features) this._features = [];\r\n return this._features;\r\n }\r\n public set features(val) {\r\n this._features = val;\r\n }\r\n\r\n private _connectors: Connector[];\r\n @Enumerable()\r\n public get connectors(): Connector[] {\r\n if (!this._connectors) this._connectors = [];\r\n return this._connectors;\r\n }\r\n public set connectors(val) {\r\n this._connectors = val;\r\n }\r\n\r\n public getChildren(): Kb3dObject[] {\r\n return [...this.features, ...this.connectors];\r\n }\r\n\r\n public getChildArrayForType(token: Token): Kb3dObject[] | undefined {\r\n if (token.isFeature()) {\r\n return this.features;\r\n } else if (token == Token.Connector) {\r\n return this.connectors;\r\n } else {\r\n throw Error('unexpected token');\r\n }\r\n }\r\n\r\n /** overriden by derived classes to get only children that derive from SpaceNode (like groups, meshes, nested scenes) */\r\n public abstract getSpaceChildren(): SpaceNode[];\r\n\r\n public getAncestors(): SpaceNode[] {\r\n return super.getAncestors() as SpaceNode[];\r\n }\r\n\r\n public copyPropertiesTo(target: SpaceNode) {\r\n target.name = this.name;\r\n target.visible = this.visible;\r\n target.position = this.position;\r\n target.useEuler = this.useEuler;\r\n target.eulerRotation = this.eulerRotation;\r\n target.rotationAngle = this.rotationAngle;\r\n target.rotationAxis = this.rotationAxis;\r\n target.scaling = this.scaling;\r\n target.pivot = this.pivot;\r\n target.fixed = this.fixed;\r\n target._primeMover = this._primeMover;\r\n target._showPivot = this._showPivot;\r\n target.preTransform = this.preTransform;\r\n }\r\n\r\n public getFeatureByName(name: string) {\r\n return [...this.features, ...this.connectors].find(f => f.name.isEqual(name));\r\n }\r\n\r\n public rotate(axis: KbVector, angle: number) {\r\n axis.normalize();\r\n\r\n const q = KbQuaternion.FromRotationAxis(axis, angle);\r\n this.setRotationQuaternion(this.getRotationQuaternion().multiply(q));\r\n }\r\n}\r\n","import { BaseKb3dModel, NodeReference } from '../decorators';\r\nimport { IMeshNode } from '../kb3d-models';\r\nimport { MaterialNode } from '../materials/material-node';\r\nimport { Token } from '../token';\r\nimport { SpaceNode } from './space-node';\r\n\r\n@BaseKb3dModel({\r\n token: Token.MeshNode,\r\n})\r\nexport abstract class MeshNode extends SpaceNode implements IMeshNode {\r\n constructor() {\r\n super();\r\n }\r\n\r\n /**\r\n * Dynamically created id that is used to track assets in the 3d engine. Ensures there is\r\n * no duplicate id's when there are nested scenes.\r\n */\r\n public _activeDynamicId?: string;\r\n public _polygonCount?: number = 0;\r\n public _verticesCount?: number = 0;\r\n\r\n public flipSurfaceOrientation?: boolean;\r\n public receiveShadows?: boolean;\r\n public castShadows?: boolean;\r\n public checkCollisions?: boolean;\r\n public isPickable?: boolean;\r\n\r\n @NodeReference(MaterialNode, '' as any)\r\n public materialId?: string = null as any;\r\n\r\n public opacity?: number;\r\n\r\n public getSpaceChildren() {\r\n return new Array();\r\n }\r\n\r\n copyPropertiesTo(target: SpaceNode) {\r\n super.copyPropertiesTo(target);\r\n if (target instanceof MeshNode) {\r\n target.flipSurfaceOrientation = this.flipSurfaceOrientation;\r\n target.receiveShadows = this.receiveShadows;\r\n target.castShadows = this.castShadows;\r\n target.checkCollisions = this.checkCollisions;\r\n target.materialId = this.materialId;\r\n target.opacity = this.opacity;\r\n }\r\n }\r\n}\r\n","import { BaseKb3dModel } from '../decorators/kb3d-decorator';\r\nimport { eSideOrientation } from '../enums';\r\nimport { IPrimitiveMeshNode, IUvData } from '../kb3d-models';\r\nimport { Token } from '../token';\r\nimport { MeshNode } from './mesh-node';\r\nimport { SpaceNode } from './space-node';\r\n\r\n@BaseKb3dModel({\r\n token: Token.PrimitiveMeshNode,\r\n})\r\nexport abstract class PrimitiveMeshNode extends MeshNode implements IPrimitiveMeshNode {\r\n constructor() {\r\n super();\r\n }\r\n\r\n /** for baking in uv map data into the primitive mesh */\r\n public uvOverrides?: IUvData;\r\n public sideOrientation: eSideOrientation;\r\n\r\n copyPropertiesTo(target: SpaceNode) {\r\n super.copyPropertiesTo(target);\r\n if (target instanceof PrimitiveMeshNode) {\r\n target.uvOverrides = this.uvOverrides;\r\n target.sideOrientation = this.sideOrientation;\r\n }\r\n }\r\n}\r\n","import { SpaceNode } from '.';\r\nimport { IgnoreChanges } from '../decorators/change-tracker';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { IMeshNode } from '../kb3d-models';\r\nimport { KbGeometry } from '../primitives/kb-geometry';\r\nimport { Token } from '../token';\r\nimport { MeshNode } from './mesh-node';\r\n\r\n/**\r\n * A Mesh Node. Must be a descendant of the MeshRoot node.\r\n * @param {Object} nodeConfig The configuration object of the node.\r\n * @param {string} nodeConfig.name The name of the node. Required.\r\n * @param {Object} parent The parent of this node. Optional, but mesh nodes should have parents if they are to be displayed.\r\n * @property\r\n */\r\n\r\n@Kb3dModel({\r\n token: Token.CustomMeshNode,\r\n})\r\nexport class CustomMeshNode extends MeshNode implements IMeshNode {\r\n constructor() {\r\n super();\r\n }\r\n\r\n /** Loading state of mesh. Used to keep track of when a request is in flight */\r\n @IgnoreChanges\r\n public _loading: string | null = null;\r\n\r\n /** The mesh geometry ID. Blank string is there is no geometry, otherwise is the same as the geometry.id */\r\n public geometryId: string;\r\n\r\n /** The mesh geometry. Undefined if not loaded or doesn't exist. Should be lazy loaded using the geometryId property. */\r\n public _geometry?: KbGeometry;\r\n\r\n get hasGeometry() {\r\n return this.geometryId != null && this.geometryId != '';\r\n }\r\n get geometryLoaded() {\r\n return this._geometry != null;\r\n }\r\n\r\n copyPropertiesTo(target: SpaceNode) {\r\n super.copyPropertiesTo(target);\r\n if (target instanceof CustomMeshNode) {\r\n target.geometryId = this.geometryId;\r\n }\r\n }\r\n}\r\n","import { Enumerable } from '../decorators/enumerable';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { IMeshGroupNode } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { Token } from '../token';\r\nimport { SpaceNode } from './space-node';\r\n\r\n@Kb3dModel({\r\n token: Token.MeshGroupNode,\r\n})\r\nexport class MeshGroupNode extends SpaceNode implements IMeshGroupNode {\r\n constructor() {\r\n super();\r\n }\r\n\r\n private _meshes: SpaceNode[];\r\n @Enumerable()\r\n public get meshes(): SpaceNode[] {\r\n if (!this._meshes) this._meshes = [];\r\n return this._meshes;\r\n }\r\n public set meshes(val) {\r\n this._meshes = val;\r\n }\r\n\r\n public getChildren(): Kb3dObject[] {\r\n return super.getChildren().concat(this.meshes);\r\n }\r\n\r\n public getChildArrayForType(token: Token): Kb3dObject[] | undefined {\r\n if (token.isMeshOrGroup() || token.isLight()) {\r\n return this.meshes;\r\n } else if (token.isConnector()) {\r\n return this.connectors;\r\n } else {\r\n throw 'unexpected token';\r\n }\r\n }\r\n\r\n public getSpaceChildren() {\r\n return this.meshes;\r\n }\r\n}\r\n","import { Ignore } from '../decorators/inject-decorator';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { IKbColor } from '../kb3d-models';\r\nimport { Token } from '../token';\r\n\r\nexport interface IKbColorLike {\r\n r: number;\r\n g: number;\r\n b: number;\r\n a?: number;\r\n}\r\n\r\n@Kb3dModel({\r\n token: Token.KbColor,\r\n trackChanges: false,\r\n create: args => new KbColor(args.r || 0, args.g || 0, args.b || 0, args.a),\r\n})\r\nexport class KbColor implements IKbColor {\r\n constructor(\r\n @Ignore() readonly r: number,\r\n @Ignore() readonly g: number,\r\n @Ignore() readonly b: number,\r\n @Ignore() readonly a: number = 1\r\n ) {}\r\n\r\n clone() {\r\n return new KbColor(this.r, this.g, this.b, this.a);\r\n }\r\n assign(c: { r?: number | null; g?: number | null; b?: number | null; a?: number | null }) {\r\n return new KbColor(\r\n typeof c.r === 'number' ? c.r : this.r,\r\n typeof c.g === 'number' ? c.g : this.g,\r\n typeof c.b === 'number' ? c.b : this.b,\r\n typeof c.a === 'number' ? c.a : this.a\r\n );\r\n }\r\n equal(c?: KbColor): boolean {\r\n return !!c && c.r == this.r && c.g == this.g && c.b == this.b && c.a == this.a;\r\n }\r\n cssString() {\r\n if (this.a < 1) {\r\n return `rgba(${this.r * 255}, ${this.g * 255}, ${this.b * 255}, ${this.a})`;\r\n } else {\r\n return `rgb(${this.r * 255}, ${this.g * 255}, ${this.b * 255})`;\r\n }\r\n }\r\n toObject() {\r\n return { r: this.r, g: this.g, b: this.b, a: this.a } as IKbColorLike;\r\n }\r\n toIntObject() {\r\n return { r: this.r * 255, g: this.g * 255, b: this.b * 255, a: this.a };\r\n }\r\n static From(c: IKbColorLike) {\r\n return new KbColor(c.r, c.g, c.b, c.a);\r\n }\r\n static FromIntObject(c: IKbColorLike) {\r\n return new KbColor(c.r / 255, c.g / 255, c.b / 255, c.a != null ? c.a : 1);\r\n }\r\n static White() {\r\n return new KbColor(1, 1, 1, 1);\r\n }\r\n static Black() {\r\n return new KbColor(0, 0, 0, 1);\r\n }\r\n static Transparent() {\r\n return new KbColor(0, 0, 0, 0);\r\n }\r\n static Green() {\r\n return new KbColor(0, 1, 0);\r\n }\r\n static Red() {\r\n return new KbColor(1, 0, 0);\r\n }\r\n static KbmaxOrange() {\r\n return new KbColor(227 / 255, 131 / 255, 28 / 255);\r\n }\r\n static KbmaxBlue() {\r\n return new KbColor(8 / 255, 72 / 255, 96 / 255);\r\n }\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { IUvData } from '../kb3d-models';\r\nimport { MeshNode } from '../meshes/mesh-node';\r\nimport { Token } from '../token';\r\n\r\nexport const NEW_GEOMETRY = 'NEW_GEOMETRY';\r\n\r\n/** IUvData has number[] props needed for server, but we want\r\n * them to be float32array here\r\n */\r\nexport type UvBuffer = {\r\n [P in keyof IUvData]: Float32Array;\r\n};\r\n\r\nexport interface KBVertexData extends UvBuffer {\r\n indices?: Uint32Array;\r\n positions?: Float32Array;\r\n normals?: Float32Array;\r\n tangents?: Float32Array;\r\n}\r\n\r\nconst MAX_INT16 = 32767;\r\nconst MAX_UINT16 = 65535;\r\n\r\n@Kb3dModel({\r\n token: Token.KbGeometry,\r\n trackChanges: false,\r\n})\r\nexport class KbGeometry implements KBVertexData {\r\n indices?: Uint32Array;\r\n positions?: Float32Array;\r\n normals?: Float32Array;\r\n tangents?: Float32Array;\r\n uvs?: Float32Array;\r\n /**\r\n * This property is used by the mesh renderer when a node has a pretransform (baked transform) defined.\r\n * The first time the geometry is transformed, it's mutated. The 4x4 transform matrix is stored here.\r\n * When this geometry is used in the future, this transform has to be taken into account for any future transforms (or lack of a transform)\r\n * This is so that we can reuse a transformed geometry if many nodes have the same transform or if a single node needs to rebuild\r\n * it's own geometry often, which is the most common use case.\r\n */\r\n _hasTransform?: number[];\r\n\r\n shared = false;\r\n\r\n /**\r\n * A geometry object. Geometries are immutable, once they are created they should not be modified.\r\n */\r\n constructor() {}\r\n\r\n clone() {\r\n const newGeometry = new KbGeometry();\r\n newGeometry.indices = this.indices;\r\n newGeometry.positions = this.positions;\r\n newGeometry.normals = this.normals;\r\n newGeometry.tangents = this.tangents;\r\n newGeometry.uvs = this.uvs;\r\n\r\n return newGeometry;\r\n }\r\n\r\n /**\r\n * Freeze a geometry to enforce the immutability of geometries\r\n */\r\n // freeze() {\r\n // Object.freeze(this);\r\n // Object.freeze(this.vertexData);\r\n // }\r\n\r\n // update(vertexData: Partial) {\r\n // const newVertexData = { ...this.vertexData, ...vertexData };\r\n // const newGeometry = new KbGeometry(newVertexData, this.id);\r\n // return newGeometry;\r\n // }\r\n\r\n extractFromBinary(fileData: ArrayBuffer, meshNode: MeshNode): void {\r\n const dataView = new DataView(fileData);\r\n let curPos: number;\r\n\r\n // indices\r\n const indicesOffset = dataView.getInt32(0, true);\r\n curPos = 4;\r\n const indicesDataSize = dataView.getInt32(curPos, true);\r\n curPos += 4;\r\n\r\n let indices: Uint32Array;\r\n if (indicesDataSize === 16) {\r\n const indices16 = new Uint16Array(fileData, curPos, indicesOffset / 2);\r\n indices = Uint32Array.from(indices16);\r\n curPos += indicesOffset;\r\n } else if (indicesDataSize === 32) {\r\n indices = new Uint32Array(fileData, curPos, indicesOffset / 4);\r\n curPos += indicesOffset;\r\n } else {\r\n throw Error('Unknown indices data size ' + meshNode.name);\r\n }\r\n\r\n // positions\r\n const posOffset = dataView.getInt32(curPos, true);\r\n curPos += 4;\r\n const posDataSize = dataView.getInt32(curPos, true);\r\n curPos += 4;\r\n const posRatio = dataView.getInt32(curPos, true);\r\n curPos += 4;\r\n\r\n let positions: Float32Array;\r\n if (posDataSize === 32) {\r\n positions = new Float32Array(fileData.slice(curPos, curPos + posOffset));\r\n for (let i = 0; i < positions.length; i++) {\r\n positions[i] /= posRatio;\r\n }\r\n curPos += posOffset;\r\n } else {\r\n throw Error('Invalid positions in binary ' + meshNode.name);\r\n }\r\n\r\n // normals\r\n const norOffset = dataView.getInt32(curPos, true);\r\n curPos += 4;\r\n const norDataSize = dataView.getInt32(curPos, true);\r\n curPos += 4;\r\n const norRatio = dataView.getInt32(curPos, true);\r\n curPos += 4;\r\n\r\n let normals: Float32Array;\r\n if (norDataSize === 16) {\r\n const normalsInt = new Int16Array(fileData.slice(curPos, curPos + norOffset));\r\n normals = new Float32Array(normalsInt.length);\r\n for (let i = 0; i < normalsInt.length; i++) {\r\n normals[i] = normalsInt[i] / norRatio;\r\n }\r\n curPos += norOffset;\r\n } else {\r\n throw Error('Invalid normals in binary ' + meshNode.name);\r\n }\r\n\r\n const uvsOffset = dataView.getInt32(curPos, true);\r\n curPos += 4;\r\n const uvsDataSize = dataView.getInt32(curPos, true);\r\n curPos += 4;\r\n const uvsRatio = dataView.getInt32(curPos, true);\r\n curPos += 4;\r\n\r\n let uvs: Float32Array;\r\n if (uvsDataSize === 16) {\r\n const uvsInt = new Int16Array(fileData.slice(curPos, curPos + uvsOffset));\r\n uvs = new Float32Array(uvsInt.length);\r\n for (let i = 0; i < uvsInt.length; i++) {\r\n uvs[i] = uvsInt[i] / uvsRatio;\r\n }\r\n curPos += uvsOffset;\r\n } else {\r\n throw Error('Invalid uvs in binary ' + meshNode.name);\r\n }\r\n\r\n // Error reporting\r\n if (indices.length % 3 > 0) {\r\n console.warn('Incomplete indices triangles in mesh ' + meshNode.name);\r\n }\r\n if (positions.length % 3 > 0) {\r\n console.warn('Incomplete position vectors in mesh ' + meshNode.name);\r\n }\r\n if (normals.length % 3 > 0) {\r\n console.warn('Incomplete normal vectors in mesh ' + meshNode.name);\r\n }\r\n if (uvs.length % 2 > 0) {\r\n console.warn('Incomplete uv vectors in mesh ' + meshNode.name);\r\n }\r\n if (uvs.length > 0 && positions.length !== normals.length) {\r\n console.warn('Normals and Positions array size mismatch in ' + meshNode.name);\r\n }\r\n if (uvs.length > 0 && positions.length / 3 !== uvs.length / 2) {\r\n console.warn('UV array size mismatch in ' + meshNode.name);\r\n }\r\n // End error reporting\r\n\r\n this.positions = positions;\r\n this.indices = indices;\r\n if (normals.length > 0) {\r\n this.normals = normals;\r\n }\r\n if (uvs.length > 0) {\r\n this.uvs = uvs;\r\n }\r\n }\r\n\r\n toBinary(): ArrayBuffer | undefined {\r\n if (!this.positions || !this.indices) {\r\n return;\r\n }\r\n const positions = this.positions;\r\n const indices = this.indices;\r\n const normals = this.normals || new Float32Array([]);\r\n const uvs = this.uvs || new Float32Array([]);\r\n\r\n let fileLen = 0;\r\n let indicesOffset = 8; // indices data markers\r\n let indicesBits = 0;\r\n let indicesLength = 0;\r\n let posOffset = 0;\r\n let posBits = 0;\r\n let posLength = 0;\r\n let normalsOffset = 0;\r\n let normalsBits = 0;\r\n let normalsLength = 0;\r\n let uvsOffset = 0;\r\n let uvsBits = 0;\r\n let uvsLength = 0;\r\n // calculate the total size of the array\r\n\r\n fileLen += indicesOffset;\r\n if (canFitInUInt16Bit(indices)) {\r\n // write a 16bit array\r\n indicesBits = 16;\r\n indicesLength = indices.length * 2;\r\n fileLen += indicesLength;\r\n } else {\r\n // write a 32bit array\r\n indicesBits = 32;\r\n indicesLength = indices.length * 4;\r\n fileLen += indicesLength;\r\n }\r\n\r\n fileLen += 12; // position data markers\r\n posOffset = fileLen;\r\n posBits = 32;\r\n fileLen += positions.length * 4;\r\n posLength = positions.length * 4;\r\n\r\n fileLen += 12; // normals data markers\r\n normalsOffset = fileLen;\r\n // write the normals 16 bit array\r\n normalsBits = 16;\r\n fileLen += normals.length * 2;\r\n normalsLength = normals.length * 2;\r\n\r\n fileLen += 12; // uv data markers\r\n uvsOffset = fileLen;\r\n uvsBits = 16;\r\n fileLen += uvs.length * 2;\r\n uvsLength = uvs.length * 2;\r\n\r\n // create the buffer that represent the binary file\r\n const buffer = new ArrayBuffer(fileLen);\r\n const bufferView = new DataView(buffer);\r\n\r\n // let's write the data into the bufferView\r\n // write the indices\r\n bufferView.setInt32(indicesOffset - 8, indicesLength, true); // NOTICE that by default JS is big endian, but since our servers are x86 we are little endian\r\n bufferView.setInt32(indicesOffset - 4, indicesBits, true);\r\n\r\n if (indicesBits == 16) {\r\n for (let i = 0; i < indices.length; i++) {\r\n bufferView.setUint16(indicesOffset, indices[i], true);\r\n indicesOffset += 2;\r\n }\r\n } else if (indicesBits == 32) {\r\n for (let i = 0; i < indices.length; i++) {\r\n bufferView.setUint32(indicesOffset, indices[i], true);\r\n indicesOffset += 4;\r\n }\r\n }\r\n\r\n // write the positions\r\n bufferView.setInt32(posOffset - 12, posLength, true);\r\n bufferView.setInt32(posOffset - 8, posBits, true);\r\n bufferView.setInt32(posOffset - 4, 1, true); // ratio\r\n\r\n if (posBits == 32) {\r\n for (let i = 0; i < positions.length; i++) {\r\n bufferView.setFloat32(posOffset, positions[i], true);\r\n posOffset += 4;\r\n }\r\n } else {\r\n console.error('Fix me - non-32bit positions not supported');\r\n }\r\n\r\n // write the normals\r\n bufferView.setInt32(normalsOffset - 12, normalsLength, true);\r\n bufferView.setInt32(normalsOffset - 8, normalsBits, true);\r\n bufferView.setInt32(normalsOffset - 4, MAX_INT16, true);\r\n\r\n if (normalsBits == 16) {\r\n for (let i = 0; i < normals.length; i++) {\r\n const n = floatToIntStore(normals[i], MAX_INT16);\r\n bufferView.setInt16(normalsOffset, n, true);\r\n normalsOffset += 2;\r\n }\r\n } else {\r\n console.error('Fix me - non-16bit normals not supported');\r\n }\r\n\r\n // write the UVs\r\n bufferView.setInt32(uvsOffset - 12, uvsLength, true);\r\n bufferView.setInt32(uvsOffset - 8, uvsBits, true);\r\n bufferView.setInt32(uvsOffset - 4, MAX_INT16, true);\r\n\r\n if (uvsBits == 16) {\r\n for (let i = 0; i < uvs.length; i++) {\r\n const n = floatToIntStore(uvs[i], MAX_INT16);\r\n bufferView.setInt16(uvsOffset, n, true);\r\n uvsOffset += 2;\r\n }\r\n } else {\r\n console.error('Fix me - non-16bit uvs not supported');\r\n }\r\n\r\n return buffer;\r\n }\r\n}\r\n\r\n// function canFitInInt16Bit(inArray: Float32Array): boolean {\r\n\r\n// for (let i = 0; i < inArray.length; i++) {\r\n// const val = inArray[i];\r\n// if (val > 327.68 || val < -327.68) {\r\n// return false;\r\n// }\r\n// }\r\n// return true;\r\n// }\r\n\r\n// function canFitInInt8Bit(inArray: Float32Array): boolean {\r\n// for (let i = 0; i < inArray.length; i++) {\r\n// const val = inArray[i];\r\n// if (val > 1.28 || val < -1.28) {\r\n// return false;\r\n// }\r\n// }\r\n// return true;\r\n// }\r\n\r\nfunction canFitInUInt16Bit(inArray: Int32Array | Uint32Array) {\r\n for (let i = 0; i < inArray.length; i++) {\r\n const val = inArray[i];\r\n if (val > MAX_UINT16) {\r\n return false;\r\n }\r\n }\r\n return true;\r\n}\r\n\r\nfunction floatToIntStore(float: number, maxInt: number, maxVal = 1): number {\r\n // this function convert a value between -327.68 and 327.68 to an int 16. we use the convention to have only 2 decimal numbers\r\n float = (Math.min(Math.max(-maxVal, float), maxVal) * maxInt) / maxVal;\r\n\r\n const outVal = Math.trunc(float);\r\n return outVal;\r\n}\r\n","export enum eContainerItemLayout {\r\n horizontal = \"horizontal\",\r\n vertical = \"vertical\",\r\n absolute = \"absolute\",\r\n}\r\nexport enum eContainerChildAlignment {\r\n inherit = \"inherit\",\r\n stretch = \"stretch\",\r\n start = \"start\",\r\n center = \"center\",\r\n end = \"end\",\r\n}\r\nexport enum eContainerItemAlignment {\r\n stretch = \"stretch\",\r\n start = \"start\",\r\n center = \"center\",\r\n end = \"end\",\r\n}\r\nexport enum eContainerItemJustify {\r\n start = \"start\",\r\n center = \"center\",\r\n end = \"end\",\r\n}\r\nexport enum eScroll {\r\n none = \"none\",\r\n auto = \"auto\",\r\n x = \"x\",\r\n y = \"y\",\r\n}\r\nexport enum eRelativeSize {\r\n auto = \"auto\",\r\n x1 = \"x1\",\r\n x2 = \"x2\",\r\n x3 = \"x3\",\r\n x4 = \"x4\",\r\n x5 = \"x5\",\r\n x6 = \"x6\",\r\n x7 = \"x7\",\r\n x8 = \"x8\",\r\n x9 = \"x9\",\r\n x10 = \"x10\",\r\n x11 = \"x11\",\r\n x12 = \"x12\",\r\n custom = \"custom\",\r\n}\r\nexport enum ePadding {\r\n none = \"none\",\r\n small = \"small\",\r\n medium = \"medium\",\r\n large = \"large\",\r\n}\r\nexport enum eColorShade {\r\n default = \"default\",\r\n lighter1 = \"lighter1\",\r\n lighter2 = \"lighter2\",\r\n lighter3 = \"lighter3\",\r\n lighter4 = \"lighter4\",\r\n darker1 = \"darker1\",\r\n darker2 = \"darker2\",\r\n darker3 = \"darker3\",\r\n darker4 = \"darker4\",\r\n}\r\nexport enum eBackgroundColor {\r\n background = \"background\",\r\n primary = \"primary\",\r\n accent = \"accent\",\r\n warning = \"warning\",\r\n success = \"success\",\r\n input = \"input\",\r\n header = \"header\",\r\n}\r\nexport enum eElevation {\r\n none = \"none\",\r\n z1 = \"z1\",\r\n z2 = \"z2\",\r\n z3 = \"z3\",\r\n z4 = \"z4\",\r\n z5 = \"z5\",\r\n z6 = \"z6\",\r\n z7 = \"z7\",\r\n z8 = \"z8\",\r\n}\r\nexport enum eRuleType {\r\n value = \"value\",\r\n visibility = \"visibility\",\r\n validation = \"validation\",\r\n pricing = \"pricing\",\r\n message = \"message\",\r\n loaded = \"loaded\",\r\n submit = \"submit\",\r\n pageChanged = \"pageChanged\",\r\n tabChanged = \"tabChanged\",\r\n expanderChanged = \"expanderChanged\",\r\n output = \"output\",\r\n action = \"action\",\r\n function = \"function\",\r\n global = \"global\",\r\n globalEvent = \"globalEvent\",\r\n safeFunction = \"safeFunction\",\r\n naming = \"naming\",\r\n field = \"field\",\r\n button = \"button\",\r\n svgViewer = \"svgViewer\",\r\n dragStart = \"dragStart\",\r\n dragMove = \"dragMove\",\r\n dragEnd = \"dragEnd\",\r\n optionFilter = \"optionFilter\",\r\n nested = \"nested\",\r\n resize = \"resize\",\r\n nestedSetSelected = \"nestedSetSelected\",\r\n keyboard = \"keyboard\",\r\n manufacture = \"manufacture\",\r\n kineticDocumentRules = \"kineticDocumentRules\",\r\n layout = \"layout\",\r\n configuratorAdded = \"configuratorAdded\",\r\n configuratorRemoved = \"configuratorRemoved\",\r\n configuratorCopied = \"configuratorCopied\",\r\n product = \"product\",\r\n workflow = \"workflow\",\r\n beforeBuild = \"beforeBuild\",\r\n globalDefinition = \"globalDefinition\",\r\n quote = \"quote\",\r\n scene = \"scene\",\r\n sceneLoaded = \"sceneLoaded\",\r\n xrEnter = \"xrEnter\",\r\n xrExit = \"xrExit\",\r\n}\r\nexport enum eToolbarPosition {\r\n hidden = \"hidden\",\r\n topLeft = \"topLeft\",\r\n topRight = \"topRight\",\r\n bottomLeft = \"bottomLeft\",\r\n bottomRight = \"bottomRight\",\r\n}\r\nexport enum eValidationTiming {\r\n always = \"always\",\r\n onTouched = \"onTouched\",\r\n}\r\nexport enum eViewerMode {\r\n none = \"none\",\r\n media = \"media\",\r\n scene = \"scene\",\r\n}\r\nexport enum ePosition {\r\n left = \"left\",\r\n right = \"right\",\r\n}\r\nexport enum eIconStyle {\r\n none = \"none\",\r\n icon = \"icon\",\r\n media = \"media\",\r\n}\r\nexport enum eExportType {\r\n _custom = \"_custom\",\r\n pdf = \"pdf\",\r\n bmp = \"bmp\",\r\n doc = \"doc\",\r\n docm = \"docm\",\r\n dot = \"dot\",\r\n dotm = \"dotm\",\r\n dotx = \"dotx\",\r\n emf = \"emf\",\r\n epub = \"epub\",\r\n html = \"html\",\r\n jpg = \"jpg\",\r\n gif = \"gif\",\r\n tiff = \"tiff\",\r\n dwf = \"dwf\",\r\n edrw = \"edrw\",\r\n png = \"png\",\r\n odt = \"odt\",\r\n rtf = \"rtf\",\r\n tif = \"tif\",\r\n xaml = \"xaml\",\r\n xps = \"xps\",\r\n ods = \"ods\",\r\n svg = \"svg\",\r\n txt = \"txt\",\r\n easm = \"easm\",\r\n prt = \"prt\",\r\n g = \"g\",\r\n dwg = \"dwg\",\r\n dxf = \"dxf\",\r\n sldprt = \"sldprt\",\r\n sldasm = \"sldasm\",\r\n slddrw = \"slddrw\",\r\n sldlfp = \"sldlfp\",\r\n x_t = \"x_t\",\r\n x_b = \"x_b\",\r\n igs = \"igs\",\r\n stl = \"stl\",\r\n wrl = \"wrl\",\r\n u3d = \"u3d\",\r\n psd = \"psd\",\r\n ai = \"ai\",\r\n cgr = \"cgr\",\r\n asm = \"asm\",\r\n hcg = \"hcg\",\r\n hsf = \"hsf\",\r\n sat = \"sat\",\r\n stp = \"stp\",\r\n step = \"step\",\r\n vda = \"vda\",\r\n neu = \"neu\",\r\n iv = \"iv\",\r\n slp = \"slp\",\r\n eps = \"eps\",\r\n catproduct = \"catproduct\",\r\n jt = \"jt\",\r\n zgl = \"zgl\",\r\n xgl = \"xgl\",\r\n csv = \"csv\",\r\n xml = \"xml\",\r\n}\r\nexport enum eHotspotSource {\r\n none = \"none\",\r\n text = \"text\",\r\n media = \"media\",\r\n element = \"element\",\r\n}\r\nexport enum eHotspotAttach {\r\n topLeft = \"topLeft\",\r\n top = \"top\",\r\n topRight = \"topRight\",\r\n left = \"left\",\r\n center = \"center\",\r\n right = \"right\",\r\n bottomLeft = \"bottomLeft\",\r\n bottom = \"bottom\",\r\n bottomRight = \"bottomRight\",\r\n}\r\nexport enum eHotspotPosition {\r\n manual = \"manual\",\r\n target = \"target\",\r\n}\r\nexport enum eHotspotShape {\r\n circle = \"circle\",\r\n circleWithText = \"circleWithText\",\r\n image = \"image\",\r\n mesh = \"mesh\",\r\n text = \"text\",\r\n}\r\nexport enum eAnimationEasing {\r\n linear = \"linear\",\r\n easeInQuad = \"easeInQuad\",\r\n easeOutQuad = \"easeOutQuad\",\r\n easeInOutQuad = \"easeInOutQuad\",\r\n easeInCubic = \"easeInCubic\",\r\n easeOutCubic = \"easeOutCubic\",\r\n easeInOutCubic = \"easeInOutCubic\",\r\n easeInQuart = \"easeInQuart\",\r\n easeOutQuart = \"easeOutQuart\",\r\n easeInOutQuart = \"easeInOutQuart\",\r\n easeInQuint = \"easeInQuint\",\r\n easeOutQuint = \"easeOutQuint\",\r\n easeInOutQuint = \"easeInOutQuint\",\r\n easeInSine = \"easeInSine\",\r\n easeOutSine = \"easeOutSine\",\r\n easeInOutSine = \"easeInOutSine\",\r\n easeInExpo = \"easeInExpo\",\r\n easeOutExpo = \"easeOutExpo\",\r\n easeInOutExpo = \"easeInOutExpo\",\r\n easeInCirc = \"easeInCirc\",\r\n easeOutCirc = \"easeOutCirc\",\r\n easeInOutCirc = \"easeInOutCirc\",\r\n easeInElastic = \"easeInElastic\",\r\n easeOutElastic = \"easeOutElastic\",\r\n easeInOutElastic = \"easeInOutElastic\",\r\n easeInBack = \"easeInBack\",\r\n easeOutBack = \"easeOutBack\",\r\n easeInOutBack = \"easeInOutBack\",\r\n easeInBounce = \"easeInBounce\",\r\n easeOutBounce = \"easeOutBounce\",\r\n easeInOutBounce = \"easeInOutBounce\",\r\n}\r\nexport enum eQueryMode {\r\n none = \"none\",\r\n table = \"table\",\r\n database = \"database\",\r\n array = \"array\",\r\n optionFilter = \"optionFilter\",\r\n parentOptionFilter = \"parentOptionFilter\",\r\n}\r\nexport enum eOptionFilterSource {\r\n table = \"table\",\r\n database = \"database\",\r\n query = \"query\",\r\n safeFunction = \"safeFunction\",\r\n optionFilter = \"optionFilter\",\r\n}\r\nexport enum eOutputType {\r\n none = \"none\",\r\n word = \"word\",\r\n excel = \"excel\",\r\n image = \"image\",\r\n solidworks = \"solidworks\",\r\n inventor = \"inventor\",\r\n creo = \"creo\",\r\n creo3 = \"creo3\",\r\n cadDrawing = \"cadDrawing\",\r\n bom = \"bom\",\r\n sceneRender = \"sceneRender\",\r\n pdf = \"pdf\",\r\n salesforce = \"salesforce\",\r\n autoCad = \"autoCad\",\r\n zip = \"zip\",\r\n svg = \"svg\",\r\n text = \"text\",\r\n}\r\nexport enum eOutputFieldType {\r\n wordText = \"wordText\",\r\n wordContainer = \"wordContainer\",\r\n wordImage = \"wordImage\",\r\n wordList = \"wordList\",\r\n excelSheet = \"excelSheet\",\r\n excelCell = \"excelCell\",\r\n excelImage = \"excelImage\",\r\n excelTable = \"excelTable\",\r\n excelChart = \"excelChart\",\r\n cadReplaceComponent = \"cadReplaceComponent\",\r\n cadAngleMate = \"cadAngleMate\",\r\n cadCoincidentMate = \"cadCoincidentMate\",\r\n cadCoordinateSystemMate = \"cadCoordinateSystemMate\",\r\n cadDistanceMate = \"cadDistanceMate\",\r\n cadPositionConstraint = \"cadPositionConstraint\",\r\n cadDimension = \"cadDimension\",\r\n cadFeature = \"cadFeature\",\r\n cadLibFeature = \"cadLibFeature\",\r\n cadCustomProperty = \"cadCustomProperty\",\r\n cadComponent = \"cadComponent\",\r\n cadDrawingView = \"cadDrawingView\",\r\n cadDrawingSheet = \"cadDrawingSheet\",\r\n cadDrawingAnnotation = \"cadDrawingAnnotation\",\r\n cadDynamicDrawingDimension = \"cadDynamicDrawingDimension\",\r\n cadDynamicDrawingAnnotation = \"cadDynamicDrawingAnnotation\",\r\n cadDynamicComponent = \"cadDynamicComponent\",\r\n cadNestedComponent = \"cadNestedComponent\",\r\n cadScaleFeature = \"cadScaleFeature\",\r\n cadAnnotation = \"cadAnnotation\",\r\n cadText = \"cadText\",\r\n cadParameter = \"cadParameter\",\r\n cadRepresentation = \"cadRepresentation\",\r\n cadPoint = \"cadPoint\",\r\n cadLayer = \"cadLayer\",\r\n bomMapping = \"bomMapping\",\r\n salesForceObject = \"salesForceObject\",\r\n cadDrawingDimension = \"cadDrawingDimension\",\r\n pdfSection = \"pdfSection\",\r\n cadLibReference = \"cadLibReference\",\r\n cadLibDimension = \"cadLibDimension\",\r\n cadBlock = \"cadBlock\",\r\n cadBlockArray = \"cadBlockArray\",\r\n zipSection = \"zipSection\",\r\n}\r\nexport enum eSingleOptionBehavior {\r\n nothing = \"nothing\",\r\n disable = \"disable\",\r\n hide = \"hide\",\r\n}\r\nexport enum eConfiguratorImageSource {\r\n none = \"none\",\r\n media = \"media\",\r\n scene = \"scene\",\r\n}\r\nexport enum eTextColor {\r\n normal = \"normal\",\r\n header = \"header\",\r\n primary = \"primary\",\r\n accent = \"accent\",\r\n error = \"error\",\r\n success = \"success\",\r\n}\r\nexport enum eConfiguratorEntityType {\r\n product = \"product\",\r\n quoteHeader = \"quoteHeader\",\r\n}\r\nexport enum eBackground {\r\n color = \"color\",\r\n transparent = \"transparent\",\r\n environment = \"environment\",\r\n}\r\nexport enum eSideOrientation {\r\n frontSide = \"frontSide\",\r\n backSide = \"backSide\",\r\n doubleSide = \"doubleSide\",\r\n}\r\nexport enum eUvMode {\r\n planar = \"planar\",\r\n box = \"box\",\r\n cylindrical = \"cylindrical\",\r\n spherical = \"spherical\",\r\n useExisting = \"useExisting\",\r\n}\r\nexport enum eMaterialViewMode {\r\n shaded = \"shaded\",\r\n edges = \"edges\",\r\n wireframe = \"wireframe\",\r\n}\r\nexport enum eMaterialLocation {\r\n local = \"local\",\r\n global = \"global\",\r\n library = \"library\",\r\n}\r\nexport enum eAlphaMode {\r\n disable = \"disable\",\r\n add = \"add\",\r\n combine = \"combine\",\r\n subtract = \"subtract\",\r\n multiply = \"multiply\",\r\n}\r\nexport enum eViewpointType {\r\n orbit = \"orbit\",\r\n firstPerson = \"firstPerson\",\r\n overhead = \"overhead\",\r\n}\r\nexport enum eViewpointProjection {\r\n perspective = \"perspective\",\r\n orthographic = \"orthographic\",\r\n}\r\nexport enum eViewpointTargetMode {\r\n rotation = \"rotation\",\r\n point = \"point\",\r\n mesh = \"mesh\",\r\n}\r\nexport enum eConnectorPositionMode {\r\n boundingBox = \"boundingBox\",\r\n absolute = \"absolute\",\r\n moveAlongSketch = \"moveAlongSketch\",\r\n}\r\nexport enum eDragMode {\r\n plane = \"plane\",\r\n free = \"free\",\r\n surface = \"surface\",\r\n}\r\nexport enum eConnectionMode {\r\n none = \"none\",\r\n world = \"world\",\r\n screen = \"screen\",\r\n}\r\nexport enum eEasing {\r\n linear = \"linear\",\r\n quadraticIn = \"quadraticIn\",\r\n quadraticOut = \"quadraticOut\",\r\n quadraticInOut = \"quadraticInOut\",\r\n cubicIn = \"cubicIn\",\r\n cubicOut = \"cubicOut\",\r\n cubicInOut = \"cubicInOut\",\r\n quarticIn = \"quarticIn\",\r\n quarticOut = \"quarticOut\",\r\n quarticInOut = \"quarticInOut\",\r\n quinticIn = \"quinticIn\",\r\n quinticOut = \"quinticOut\",\r\n quinticInOut = \"quinticInOut\",\r\n sinusoidalIn = \"sinusoidalIn\",\r\n sinusoidalOut = \"sinusoidalOut\",\r\n sinusoidalInOut = \"sinusoidalInOut\",\r\n exponentialIn = \"exponentialIn\",\r\n exponentialOut = \"exponentialOut\",\r\n exponentialInOut = \"exponentialInOut\",\r\n circularIn = \"circularIn\",\r\n circularOut = \"circularOut\",\r\n circularInOut = \"circularInOut\",\r\n elasticIn = \"elasticIn\",\r\n elasticOut = \"elasticOut\",\r\n elasticInOut = \"elasticInOut\",\r\n backIn = \"backIn\",\r\n backOut = \"backOut\",\r\n backInOut = \"backInOut\",\r\n bounceIn = \"bounceIn\",\r\n bounceOut = \"bounceOut\",\r\n bounceInOut = \"bounceInOut\",\r\n}\r\nexport enum eCylinderCap {\r\n none = \"none\",\r\n start = \"start\",\r\n end = \"end\",\r\n both = \"both\",\r\n}\r\nexport enum eJoinType {\r\n union = \"union\",\r\n intersect = \"intersect\",\r\n cut = \"cut\",\r\n}\r\nexport enum eShadowQuality {\r\n low = \"low\",\r\n medium = \"medium\",\r\n high = \"high\",\r\n}\r\nexport enum eShadowResolution {\r\n low = \"low\",\r\n medium = \"medium\",\r\n high = \"high\",\r\n}\r\nexport enum eShadowFilter {\r\n none = \"none\",\r\n simple = \"simple\",\r\n advanced = \"advanced\",\r\n}\r\nexport enum eMaterialWorkflow {\r\n metallicRoughness = \"metallicRoughness\",\r\n specularGlossiness = \"specularGlossiness\",\r\n}\r\nexport enum eFeatureCenter {\r\n barycenter = \"barycenter\",\r\n meshPivotPoint = \"meshPivotPoint\",\r\n}\r\nexport enum eJobStatus {\r\n none = \"none\",\r\n waiting = \"waiting\",\r\n queued = \"queued\",\r\n working = \"working\",\r\n complete = \"complete\",\r\n failed = \"failed\",\r\n}\r\nexport enum eJobType {\r\n build = \"build\",\r\n render = \"render\",\r\n notification = \"notification\",\r\n upsertProductsToSfdc = \"upsertProductsToSfdc\",\r\n pullSfdcProducts = \"pullSfdcProducts\",\r\n submitFromSfdcCpq = \"submitFromSfdcCpq\",\r\n completeBuildState = \"completeBuildState\",\r\n cleanDb = \"cleanDb\",\r\n deployment = \"deployment\",\r\n sceneImport = \"sceneImport\",\r\n sceneExport = \"sceneExport\",\r\n sceneConvert = \"sceneConvert\",\r\n completeQuoteProductBuild = \"completeQuoteProductBuild\",\r\n configuratorExport = \"configuratorExport\",\r\n configuratorImport = \"configuratorImport\",\r\n}\r\nexport enum eEnvironment {\r\n prod = \"prod\",\r\n test = \"test\",\r\n dev = \"dev\",\r\n}\r\nexport enum eImportType {\r\n external = \"external\",\r\n kb3d = \"kb3d\",\r\n}\r\nexport enum eQueueMessageType {\r\n builder = \"builder\",\r\n cleanDb = \"cleanDb\",\r\n completeBuildState = \"completeBuildState\",\r\n notification = \"notification\",\r\n render = \"render\",\r\n pullSfdcProducts = \"pullSfdcProducts\",\r\n upsertProductsToSfdc = \"upsertProductsToSfdc\",\r\n submitFromSfdcCpq = \"submitFromSfdcCpq\",\r\n deployment = \"deployment\",\r\n completeQuoteProductBuild = \"completeQuoteProductBuild\",\r\n sceneImport = \"sceneImport\",\r\n sceneExport = \"sceneExport\",\r\n sceneConvert = \"sceneConvert\",\r\n}\r\nexport enum eMateType {\r\n fastened = \"fastened\",\r\n revolute = \"revolute\",\r\n slider = \"slider\",\r\n planar = \"planar\",\r\n cylindrical = \"cylindrical\",\r\n ball = \"ball\",\r\n parallel = \"parallel\",\r\n}\r\nexport enum eValidationType {\r\n error = \"error\",\r\n warning = \"warning\",\r\n info = \"info\",\r\n}\r\nexport enum eValidationSource {\r\n snap = \"snap\",\r\n configurator = \"configurator\",\r\n scene = \"scene\",\r\n}\r\nexport enum eTextureChannel {\r\n emissive = \"emissive\",\r\n albedo = \"albedo\",\r\n ambient = \"ambient\",\r\n opacity = \"opacity\",\r\n reflection = \"reflection\",\r\n specular = \"specular\",\r\n metallic = \"metallic\",\r\n roughness = \"roughness\",\r\n glossiness = \"glossiness\",\r\n normal = \"normal\",\r\n ambientOcclusion = \"ambientOcclusion\",\r\n anisotropyTangent = \"anisotropyTangent\",\r\n subsurfaceThickness = \"subsurfaceThickness\",\r\n clearCoatNormal = \"clearCoatNormal\",\r\n}\r\nexport enum eTextureHorizontalAlign {\r\n left = \"left\",\r\n center = \"center\",\r\n right = \"right\",\r\n}\r\nexport enum eTextureVerticalAlign {\r\n top = \"top\",\r\n middle = \"middle\",\r\n bottom = \"bottom\",\r\n}\r\nexport enum eTextureFit {\r\n none = \"none\",\r\n stretch = \"stretch\",\r\n contain = \"contain\",\r\n}\r\nexport enum eTextureBlendMode {\r\n normal = \"normal\",\r\n multiply = \"multiply\",\r\n screen = \"screen\",\r\n overlay = \"overlay\",\r\n darken = \"darken\",\r\n lighten = \"lighten\",\r\n hue = \"hue\",\r\n saturation = \"saturation\",\r\n color = \"color\",\r\n luminosity = \"luminosity\",\r\n difference = \"difference\",\r\n mask = \"mask\",\r\n vector = \"vector\",\r\n}\r\nexport enum eTextureWrapAddressMode {\r\n repeat = \"repeat\",\r\n clamp = \"clamp\",\r\n}\r\nexport enum eAnnotationLineStyle {\r\n solid = \"solid\",\r\n dashed = \"dashed\",\r\n dotted = \"dotted\",\r\n none = \"none\",\r\n}\r\nexport enum eDimensionLineCapStyle {\r\n none = \"none\",\r\n arrow = \"arrow\",\r\n line = \"line\",\r\n}\r\nexport enum eDimensionAxis {\r\n free = \"free\",\r\n x = \"x\",\r\n y = \"y\",\r\n z = \"z\",\r\n}\r\nexport enum eLoadingExperience {\r\n none = \"none\",\r\n fade = \"fade\",\r\n spinner = \"spinner\",\r\n image = \"image\",\r\n}\r\nexport enum eReflectionType {\r\n plane = \"plane\",\r\n probe = \"probe\",\r\n}\r\nexport enum eRenderMode {\r\n shaded = \"shaded\",\r\n wireframe = \"wireframe\",\r\n edge = \"edge\",\r\n hiddenEdges = \"hiddenEdges\",\r\n}\r\nexport enum eTextureSamplingMode {\r\n nearest = \"nearest\",\r\n bilinear = \"bilinear\",\r\n trilinear = \"trilinear\",\r\n}\r\nexport enum eAxis {\r\n x = \"x\",\r\n y = \"y\",\r\n z = \"z\",\r\n}\r\nexport enum eSketchPathType {\r\n points = \"points\",\r\n arcThrough3Points = \"arcThrough3Points\",\r\n centerRadiusCircle = \"centerRadiusCircle\",\r\n catmullRom = \"catmullRom\",\r\n cubicBezier = \"cubicBezier\",\r\n quadraticBezier = \"quadraticBezier\",\r\n hermite = \"hermite\",\r\n}\r\nexport enum eFilterControl {\r\n none = \"none\",\r\n text = \"text\",\r\n select = \"select\",\r\n multiSelect = \"multiSelect\",\r\n dateRange = \"dateRange\",\r\n numberRange = \"numberRange\",\r\n suggest = \"suggest\",\r\n checkbox = \"checkbox\",\r\n}\r\nexport enum eFilterSource {\r\n none = \"none\",\r\n enum = \"enum\",\r\n properties = \"properties\",\r\n users = \"users\",\r\n states = \"states\",\r\n configurators = \"configurators\",\r\n fields = \"fields\",\r\n customers = \"customers\",\r\n roles = \"roles\",\r\n productSorts = \"productSorts\",\r\n manufacturers = \"manufacturers\",\r\n attributes = \"attributes\",\r\n suggestContacts = \"suggestContacts\",\r\n suggestCustomers = \"suggestCustomers\",\r\n suggestQuotes = \"suggestQuotes\",\r\n suggestProducts = \"suggestProducts\",\r\n headerFields = \"headerFields\",\r\n}\r\nexport enum eFilterType {\r\n property = \"property\",\r\n fieldValue = \"fieldValue\",\r\n sort = \"sort\",\r\n attribute = \"attribute\",\r\n headerValue = \"headerValue\",\r\n}\r\nexport enum eFieldType {\r\n text = \"text\",\r\n number = \"number\",\r\n boolean = \"boolean\",\r\n color = \"color\",\r\n date = \"date\",\r\n upload = \"upload\",\r\n textArray = \"textArray\",\r\n numberArray = \"numberArray\",\r\n colorArray = \"colorArray\",\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { instanceOf } from '../decorators/model-utils';\r\nimport { ISphereNode } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { Token } from '../token';\r\nimport { PrimitiveMeshNode } from './primitive-mesh-node';\r\n\r\n@Kb3dModel({\r\n token: Token.SphereNode,\r\n})\r\nexport class SphereNode extends PrimitiveMeshNode implements ISphereNode {\r\n public constructor() {\r\n super();\r\n }\r\n\r\n public tessellation: number;\r\n public radius: number;\r\n public arc: number;\r\n public latitudeStart: number;\r\n public latitudeSlice: number;\r\n}\r\n\r\nexport function isSphereMesh(node: Kb3dObject): node is SphereNode {\r\n return instanceOf(node, Token.SphereNode);\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { instanceOf } from '../decorators/model-utils';\r\nimport { ITorusNode } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { Token } from '../token';\r\nimport { PrimitiveMeshNode } from './primitive-mesh-node';\r\n\r\n@Kb3dModel({\r\n token: Token.TorusNode,\r\n})\r\nexport class TorusNode extends PrimitiveMeshNode implements ITorusNode {\r\n public constructor() {\r\n super();\r\n }\r\n public radius: number;\r\n public thickness: number;\r\n public arc: number;\r\n public latitudeStart: number;\r\n public latitudeSlice: number;\r\n public tessellation: number;\r\n}\r\n\r\nexport function isTorusMesh(node: Kb3dObject): node is TorusNode {\r\n return instanceOf(node, Token.TorusNode);\r\n}\r\n","import { BoundingBox } from '@babylonjs/core/Culling/boundingBox';\r\nimport { Subject } from 'rxjs';\r\nimport { NodeReference } from '../decorators/change-tracker';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { eAxis } from '../enums';\r\nimport { ISketchNode } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { IKbVectorLike, KbColor, KbVector } from '../primitives';\r\nimport { IBoundingBox } from '../primitives/primitive-utilities';\r\nimport { Token } from '../token';\r\nimport { MeshNode } from './mesh-node';\r\nimport { SketchPath } from './sketch-node-path';\r\n\r\n@Kb3dModel({\r\n token: Token.SketchNode,\r\n})\r\nexport class SketchNode extends MeshNode implements ISketchNode {\r\n public constructor() {\r\n super();\r\n }\r\n\r\n _previewVisible: boolean;\r\n _boundingBoxMin = KbVector.Zero();\r\n _boundingBoxMax = KbVector.Zero();\r\n _boundingBoxChanges = new Subject();\r\n _controlPoints: IKbVectorLike[] = [];\r\n _lineSegmentStartingPoints: number[] = [];\r\n\r\n // todo remove\r\n _sketchPoints = new Float32Array();\r\n\r\n calculatedBoundingBox: BoundingBox;\r\n boundingXOverride?: number;\r\n boundingYOverride?: number;\r\n boundingZOverride?: number;\r\n paths: SketchPath[] = [];\r\n normal: eAxis;\r\n /** Whether to generate a face for this path */\r\n addFace = false;\r\n closePath = false;\r\n flipPath = false;\r\n\r\n tracePath = false;\r\n lineThickness: number;\r\n pathColor: KbColor;\r\n\r\n /** (Not Implemented) Path to clone. Will cause the paths property to be ignored */\r\n @NodeReference(SketchNode) clonePath?: string;\r\n\r\n public getChildren(): Kb3dObject[] {\r\n return super.getChildren().concat(this.paths);\r\n }\r\n\r\n public getChildArrayForType(token: Token): Kb3dObject[] | undefined {\r\n if (token.isSketchPath()) {\r\n return this.paths;\r\n } else {\r\n return super.getChildArrayForType(token);\r\n }\r\n }\r\n\r\n public getBoundingBox() {\r\n const min = new KbVector(\r\n this.boundingXOverride == null ? this._boundingBoxMin.x : -this.boundingXOverride / 2,\r\n this.boundingYOverride == null ? this._boundingBoxMin.y : -this.boundingYOverride / 2,\r\n this.boundingZOverride == null ? this._boundingBoxMin.z : -this.boundingZOverride / 2\r\n );\r\n const max = new KbVector(\r\n this.boundingXOverride == null ? this._boundingBoxMax.x : this.boundingXOverride / 2,\r\n this.boundingYOverride == null ? this._boundingBoxMax.y : this.boundingYOverride / 2,\r\n this.boundingZOverride == null ? this._boundingBoxMax.z : this.boundingZOverride / 2\r\n );\r\n const centerBB = this._boundingBoxMin.add(this._boundingBoxMax).scale(0.5);\r\n const center = new KbVector(\r\n this.boundingXOverride == null ? centerBB.x : 0,\r\n this.boundingYOverride == null ? centerBB.y : 0,\r\n this.boundingZOverride == null ? centerBB.z : 0\r\n );\r\n return {\r\n min,\r\n max,\r\n center,\r\n };\r\n }\r\n\r\n public getSketchLength() {\r\n let sketchLength = 0;\r\n this.paths.forEach(path => {\r\n let lastPosition: IKbVectorLike | undefined = undefined;\r\n path._points.forEach(point => {\r\n if (!lastPosition) {\r\n lastPosition = point;\r\n } else {\r\n const currentPos = point;\r\n const distance = Math.sqrt(\r\n Math.pow(lastPosition.x - currentPos.x, 2) +\r\n Math.pow(lastPosition.y - currentPos.y, 2) +\r\n Math.pow(lastPosition.z - currentPos.z, 2)\r\n );\r\n sketchLength += distance;\r\n lastPosition = currentPos;\r\n }\r\n });\r\n });\r\n return sketchLength;\r\n }\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { eSketchPathType } from '../enums';\r\nimport { SketchControlPoint } from '../features/sketch-control-point';\r\nimport { ISketchPath } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { IKbVectorLike } from '../primitives';\r\nimport { Token } from '../token';\r\nimport { SketchNode } from './sketch-node';\r\n\r\n@Kb3dModel({\r\n token: Token.SketchPath,\r\n})\r\nexport class SketchPath extends Kb3dObject implements ISketchPath {\r\n /** Next target path connected to this one. Full paths are built linked-list style */\r\n toIndex: number;\r\n /** Path control points */\r\n controlPoints: SketchControlPoint[] = [];\r\n /** How smooth to make this path when rasterizing to geometry */\r\n tessellation = 16;\r\n /** Type of arc to use for this path */\r\n type: eSketchPathType;\r\n\r\n /** For grouping paths as one contiguous shape. Only required for hole-path matching */\r\n pathGroupKey = 'default';\r\n\r\n /** Whether to treat this path as part of a hole for extrusions and fills */\r\n holeForPathGroup: string;\r\n\r\n _expanded = false;\r\n _points: IKbVectorLike[] = [];\r\n\r\n /** override base parent for more specific typing*/\r\n public _parent: SketchNode | undefined;\r\n\r\n public getChildren(): Kb3dObject[] {\r\n return super.getChildren().concat(this.controlPoints);\r\n }\r\n\r\n public getChildArrayForType(token: Token): Kb3dObject[] | undefined {\r\n if (token.isConnector()) {\r\n return this.controlPoints;\r\n } else {\r\n return super.getChildArrayForType(token);\r\n }\r\n }\r\n\r\n public getAncestors() {\r\n return super.getAncestors() as SketchNode[];\r\n }\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { ITextNode } from '../kb3d-models';\r\nimport { Token } from '../token';\r\nimport { MeshNode } from './mesh-node';\r\n\r\n@Kb3dModel({\r\n token: Token.TextNode,\r\n})\r\nexport class TextNode extends MeshNode implements ITextNode {\r\n curveRadius = 0;\r\n charSpacing = 1;\r\n\r\n text = '';\r\n\r\n fontFile: string;\r\n fontSize: number;\r\n}\r\n","import { NodeReference } from '../decorators';\r\nimport { instanceOfAny } from '../decorators/model-utils';\r\nimport { IPatternFeature } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { SketchNode } from '../meshes';\r\nimport { Token } from '../token';\r\nimport { ALL_GEOMETRY, Feature } from './feature';\r\n\r\nexport abstract class PatternFeature extends Feature implements IPatternFeature {\r\n static _affects = new Set(ALL_GEOMETRY);\r\n static _affectedBy = new Set(ALL_GEOMETRY);\r\n\r\n @NodeReference(SketchNode) meshId: string;\r\n count: number;\r\n distance: number;\r\n addEndMesh: boolean;\r\n}\r\n\r\nexport function isPattern(o: Kb3dObject): o is PatternFeature {\r\n return instanceOfAny(o, Token.LinearPatternFeature, Token.CircularPatternFeature);\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { instanceOf } from '../decorators/model-utils';\r\nimport { IConnector } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { SpaceNode } from '../meshes/space-node';\r\nimport { Token } from '../token';\r\nimport { BaseConnector } from './base-connector';\r\n\r\n@Kb3dModel({\r\n token: Token.Connector,\r\n})\r\nexport class Connector extends BaseConnector implements IConnector {\r\n /** override base parent for more specific typing*/\r\n public _parent: SpaceNode | undefined;\r\n\r\n public getAncestors() {\r\n return super.getAncestors() as SpaceNode[];\r\n }\r\n\r\n public getSpaceParent() {\r\n return this._parent;\r\n }\r\n}\r\n\r\nexport function isConnector(o: Kb3dObject): o is Connector {\r\n return instanceOf(o, Token.Connector);\r\n}\r\n","import { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { instanceOf } from '../decorators/model-utils';\r\nimport { ICircularPatternFeature } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { Token } from '../token';\r\nimport { ALL_GEOMETRY } from './feature';\r\nimport { PatternFeature } from './pattern-feature';\r\n\r\n@Kb3dModel({\r\n token: Token.CircularPatternFeature,\r\n})\r\nexport class CircularPatternFeature extends PatternFeature implements ICircularPatternFeature {\r\n static _affects = new Set(ALL_GEOMETRY);\r\n static _affectedBy = new Set(ALL_GEOMETRY);\r\n\r\n arc: number;\r\n pivot: KbVector;\r\n axis: KbVector;\r\n}\r\n\r\nexport function isCircularPattern(o: Kb3dObject): o is CircularPatternFeature {\r\n return instanceOf(o, Token.CircularPatternFeature);\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { IDeleteFacesFeature, IVertexGroup } from '../kb3d-models';\r\nimport { Token } from '../token';\r\nimport { Feature, eMeshChangeTypes } from './feature';\r\n\r\n@Kb3dModel({\r\n token: Token.DeleteFacesFeature,\r\n})\r\nexport class DeleteFacesFeature extends Feature implements IDeleteFacesFeature {\r\n static _dependsOn = new Set([eMeshChangeTypes.VertexIndices]);\r\n static _affects = new Set([eMeshChangeTypes.VertexIndices, eMeshChangeTypes.Submeshes]);\r\n static _affectedBy = new Set([eMeshChangeTypes.VertexIndices, eMeshChangeTypes.Submeshes]);\r\n\r\n vertexGroups: Array> = [];\r\n}\r\n","import { MaterialService } from '@models/services';\r\nimport { Enumerable, Inject } from '../decorators';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { IDisplacementMapFeature } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { TextureLayer } from '../materials';\r\nimport { Token } from '../token';\r\nimport { ALL_GEOMETRY, eMeshChangeTypes, Feature, FeatureWithTextures } from './feature';\r\n\r\n@Kb3dModel({\r\n token: Token.DisplacementMapFeature,\r\n})\r\nexport class DisplacementMapFeature extends Feature implements IDisplacementMapFeature, FeatureWithTextures {\r\n static _affects = new Set([eMeshChangeTypes.VertexNormals, eMeshChangeTypes.VertexPositions]);\r\n static _affectedBy = new Set(ALL_GEOMETRY);\r\n private _textures: TextureLayer[];\r\n constructor(@Inject(Token.MaterialService) public _materialService: MaterialService) {\r\n super();\r\n\r\n this._initialized = false;\r\n }\r\n\r\n @Enumerable()\r\n public get textures(): TextureLayer[] {\r\n if (!this._textures) this._textures = [];\r\n return this._textures;\r\n }\r\n public set textures(val) {\r\n this._textures = val;\r\n }\r\n\r\n minDisplacement: number;\r\n maxDisplacement: number;\r\n _initialized: boolean;\r\n\r\n public getChildren(): Kb3dObject[] {\r\n return [...this.textures];\r\n }\r\n\r\n public getChildArrayForType(token: Token): Kb3dObject[] | undefined {\r\n if (token.isTexture()) {\r\n return this.textures;\r\n } else {\r\n throw 'unexpected token';\r\n }\r\n }\r\n}\r\n","import { IExtrudeFeature } from 'viewer';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { IVertexGroup } from '../kb3d-models';\r\nimport { Token } from '../token';\r\nimport { ALL_GEOMETRY, Feature } from './feature';\r\n\r\n@Kb3dModel({\r\n token: Token.ExtrudeFeature,\r\n})\r\nexport class ExtrudeFeature extends Feature implements IExtrudeFeature {\r\n static _affects = new Set(ALL_GEOMETRY);\r\n static _affectedBy = new Set(ALL_GEOMETRY);\r\n\r\n size: number;\r\n capEnd: boolean;\r\n capStart: boolean;\r\n vertexGroups: Array> = [];\r\n}\r\n","import { NodeReference } from '../decorators/change-tracker';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { eJoinType } from '../enums';\r\nimport { IJoinGeometryFeature } from '../kb3d-models';\r\nimport { MeshNode } from '../meshes/mesh-node';\r\nimport { Token } from '../token';\r\nimport { ALL_CHANGES, ALL_GEOMETRY, Feature } from './feature';\r\n\r\n@Kb3dModel({\r\n token: Token.JoinGeometryFeature,\r\n})\r\nexport class JoinGeometryFeature extends Feature implements IJoinGeometryFeature {\r\n static _affects = new Set(ALL_GEOMETRY);\r\n static _affectedBy = new Set(ALL_CHANGES);\r\n\r\n _previewVisible: boolean;\r\n joinType: eJoinType;\r\n @NodeReference(MeshNode) meshId: string;\r\n}\r\n","import { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { instanceOf } from '../decorators/model-utils';\r\nimport { ILinearPatternFeature } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { Token } from '../token';\r\nimport { ALL_GEOMETRY } from './feature';\r\nimport { PatternFeature } from './pattern-feature';\r\n\r\n@Kb3dModel({\r\n token: Token.LinearPatternFeature,\r\n})\r\nexport class LinearPatternFeature extends PatternFeature implements ILinearPatternFeature {\r\n static _affects = new Set(ALL_GEOMETRY);\r\n static _affectedBy = new Set(ALL_GEOMETRY);\r\n\r\n translation: KbVector;\r\n}\r\n\r\nexport function isLinearPattern(o: Kb3dObject): o is LinearPatternFeature {\r\n return instanceOf(o, Token.LinearPatternFeature);\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { ISliceFeature } from '../kb3d-models';\r\nimport { KbVector } from '../primitives/kb-vector';\r\nimport { Token } from '../token';\r\nimport { ALL_GEOMETRY, Feature } from './feature';\r\n\r\n@Kb3dModel({\r\n token: Token.SliceFeature,\r\n})\r\nexport class SliceFeature extends Feature implements ISliceFeature {\r\n static _affectedBy = new Set(ALL_GEOMETRY);\r\n static _affects = new Set(ALL_GEOMETRY);\r\n\r\n _previewVisible: boolean;\r\n normal: KbVector;\r\n intersect: KbVector;\r\n removeGeometry: boolean;\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { IMirrorFeature } from '../kb3d-models';\r\nimport { Token } from '../token';\r\nimport { ALL_GEOMETRY } from './feature';\r\nimport { SliceFeature } from './slice-feature';\r\n\r\n@Kb3dModel({\r\n token: Token.MirrorFeature,\r\n})\r\nexport class MirrorFeature extends SliceFeature implements IMirrorFeature {\r\n static _affectedBy = new Set(ALL_GEOMETRY);\r\n static _affects = new Set(ALL_GEOMETRY);\r\n\r\n cloneGeometry: boolean;\r\n}\r\n","import { IgnoreChanges } from '../decorators/change-tracker';\r\nimport { Enumerable } from '../decorators/enumerable';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { eFeatureCenter } from '../enums';\r\nimport { IMoveVerticesFeature } from '../kb3d-models';\r\nimport { KbVector } from '../primitives/kb-vector';\r\nimport { Token } from '../token';\r\nimport { eMeshChangeTypes, Feature } from './feature';\r\n\r\n@Kb3dModel({\r\n token: Token.MoveVerticesFeature,\r\n})\r\nexport class MoveVerticesFeature extends Feature implements IMoveVerticesFeature {\r\n static _dependsOn = new Set([eMeshChangeTypes.VertexIndices]);\r\n static _affectedBy = new Set([eMeshChangeTypes.VertexPositions]);\r\n static _affects = new Set([eMeshChangeTypes.VertexPositions]);\r\n\r\n translation: KbVector;\r\n rotationAngle: number;\r\n rotationAxis: KbVector;\r\n scale: KbVector;\r\n center: eFeatureCenter;\r\n _previewParticleSize = 0.005;\r\n\r\n @IgnoreChanges\r\n private _vertexIndices: number[];\r\n @Enumerable()\r\n public get vertexIndices(): number[] {\r\n if (!this._vertexIndices) this._vertexIndices = [];\r\n return this._vertexIndices;\r\n }\r\n public set vertexIndices(val) {\r\n this._vertexIndices = val;\r\n }\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { INormalSmoothingFeature, IVertexGroup } from '../kb3d-models';\r\nimport { Token } from '../token';\r\nimport { ALL_GEOMETRY, Feature } from './feature';\r\n\r\n@Kb3dModel({\r\n token: Token.NormalSmoothingFeature,\r\n})\r\nexport class NormalSmoothingFeature extends Feature implements INormalSmoothingFeature {\r\n //static _dependsOn = new Set([eMeshChangeTypes.VertexIndices]);\r\n static _affects = new Set(ALL_GEOMETRY);\r\n static _affectedBy = new Set(ALL_GEOMETRY);\r\n\r\n vertexGroups: Array> = [];\r\n smooth = true;\r\n}\r\n","import { PickingInfo } from '@babylonjs/core/Collisions/pickingInfo';\r\nimport { KeyboardInfo } from '@babylonjs/core/Events/keyboardEvents';\r\nimport { PointerInfo, PointerInfoBase } from '@babylonjs/core/Events/pointerEvents';\r\nimport { BoundingBoxInfo } from '@view/modules';\r\nimport { AnnotationNode } from './annotations';\r\nimport { eConnectionMode, eDragMode, eMateType } from './enums';\r\nimport { Connector } from './features/connector';\r\nimport { Kb3dManager } from './kb3d-manager';\r\nimport { Kb3dObject } from './kb3d-object';\r\nimport { MateNode } from './mates/mate-node';\r\nimport { isPickable } from './meshes/mesh-utilities';\r\nimport { SpaceNode } from './meshes/space-node';\r\nimport { KbVector } from './primitives';\r\n\r\nexport enum eHandler {\r\n click = 'click',\r\n doubleClick = 'doubleClick',\r\n draggable = 'draggable',\r\n hotspot = 'hotspot',\r\n boundingBox = 'boundingBox',\r\n}\r\n\r\nexport type PointerInfoMinimal = PointerInfoBase & Pick;\r\n\r\nexport interface IKb3dPickInfo {\r\n pickInfo: PickingInfo;\r\n\r\n /** the specific node that the pointer is over */\r\n targetNode?: SpaceNode;\r\n\r\n /** the 3d point in the world where the mesh was clicked */\r\n pickedPoint?: KbVector;\r\n /** the 3d point of the target mesh that was clicked */\r\n meshPoint?: KbVector;\r\n\r\n /** UV intersection coordinates */\r\n u?: number;\r\n v?: number;\r\n\r\n getNormal: () => KbVector | undefined;\r\n}\r\n\r\nexport interface IKb3dPointerArgs extends PointerInfoBase, IKb3dPickInfo {\r\n /** the node that the handler was set on (could be an ancestor of the targetNode) */\r\n sourceNode?: SpaceNode;\r\n /** whether the event should continue to bubble up the ancestor nodes */\r\n propagate?: boolean;\r\n /** the type of pointer device. Eg: 'mouse', 'touch', etc. */\r\n pointerType?: string;\r\n /** the mouse button that was pressed (if there is one). 0 = left, 1 = center, 2 = right */\r\n button?: number;\r\n /** whether the ctrl key was pressed during the click */\r\n ctrlKey?: boolean;\r\n /** whether the shift key was pressed during the click */\r\n shiftKey?: boolean;\r\n /** whether the alt key was pressed during the click */\r\n altKey?: boolean;\r\n /** the screen coordinates of the click in relation to the canvas viewer */\r\n canvasX?: number;\r\n canvasY?: number;\r\n}\r\n\r\nexport interface IKb3dFacetSelectArgs extends IKb3dPointerArgs {\r\n selectedFaces: number[];\r\n}\r\n\r\nexport interface IKeyboardArgs extends KeyboardInfo {\r\n key: string;\r\n keyCode: number;\r\n altKey: boolean;\r\n ctrlKey: boolean;\r\n shiftKey: boolean;\r\n repeat: boolean;\r\n}\r\n\r\nexport type Kb3dPointerHandler = (args: IKb3dPointerArgs) => void;\r\n\r\nexport type Kb3dBoundingBoxChangeHandler = {\r\n mode: 'once' | 'forever';\r\n handler: (args: BoundingBoxInfo) => void;\r\n};\r\n\r\nexport interface IDragArgs {\r\n dragNode: SpaceNode;\r\n newPos: KbVector;\r\n startPos: KbVector;\r\n newRotationEuler: KbVector;\r\n newRotationAxis: KbVector;\r\n newRotationAngle: number;\r\n allow: boolean;\r\n u: number;\r\n v: number;\r\n //relativeSurfacePos: IVector2;\r\n\r\n /*following only used for connections*/\r\n connection?: IConnection;\r\n\r\n /** if withMates is true for the draggable, then this will contain the added mate in the dropped rule */\r\n mate?: MateNode;\r\n}\r\n\r\nexport interface IConnection {\r\n dragConnector: Connector;\r\n otherConnector: Connector;\r\n dragNode?: SpaceNode;\r\n otherNode?: SpaceNode;\r\n}\r\n\r\nexport interface ICanConnectArgs extends IConnection {\r\n allow: boolean;\r\n}\r\n\r\n// export enum eDragMode {\r\n// free,\r\n// plane,\r\n// //axis,\r\n// surface\r\n// }\r\n\r\n// export enum eConnectorMode {\r\n// none,\r\n// world,\r\n// screen\r\n// }\r\n\r\nexport interface IDraggable {\r\n /** The id, name, or wildcard of the node to define as draggable */\r\n id?: string | SpaceNode;\r\n /** The mode of the drag */\r\n mode?: eDragMode; // free, plane, surface\r\n /** In plane or linear mode, is a vector normal to the drag plane. This normal + the picked point on the mesh will determine the intersection plane */\r\n planeNormal?: KbVector;\r\n // /** In plane mode, it is the origin of the plane. If not provided, the origin of the drag node is used (usually the desired effect) */\r\n // planeOrigin?: KbVector;\r\n /** if disabled, the node origin will snap to the mouse position when you start dragging */\r\n disableOffset?: boolean;\r\n\r\n //axis?: KbVector;\r\n\r\n /** In surface mode, holds the id (if a singular surface) or ids (if multiple surfaces) of the surfaces */\r\n surfaceNodes?: string | string[];\r\n\r\n /** whether connectors are enabled during this drag operation and if so how the gauge proximity */\r\n connectionMode?: eConnectionMode;\r\n /** how to align the connectors when a connection is made */\r\n connectionType?: eMateType;\r\n /** whether to show the connector spheres or not */\r\n showConnectors?: boolean;\r\n /** the proximity distance between connectors that counts as a possible connection */\r\n connectRadius?: number;\r\n /** when a connector drag, controls the opacity of the dragged items */\r\n dragOpacity?: number;\r\n /**\r\n * Will add mates once a connection is made. Also uses mates to understand what connectors\r\n * are open for connection (meaning not part of mate). Will dynamically swap mate connectors\r\n * based on the drag node. Only valid if connectionMode is not 'none'\r\n * */\r\n withMates?: boolean;\r\n\r\n /** Called when the drag is started */\r\n dragStart?: (args: IDragArgs) => void;\r\n // /** Called at every frame to see if we want to allow the move */\r\n // allow?: (args: IDragArgs) => boolean;\r\n /** Called at every frame while dragging. Can be used to custom update the position by modifying newPos */\r\n dragging?: (args: IDragArgs) => void;\r\n /** Called when the node is dropped */\r\n dropped?: (args: IDragArgs) => void;\r\n\r\n canConnect?: (args: ICanConnectArgs) => void;\r\n}\r\n\r\n/** A 2 layer map that stores values by a key 1st, then a 2nd level of a string tag, then an array of values */\r\nexport class TagArrayMap {\r\n private _db = new Map>();\r\n\r\n public get(key: TKey, tag: string) {\r\n const v = this._db.get(key)?.get(tag);\r\n return v || [];\r\n }\r\n\r\n public deleteByKeyValue(key: TKey, value: TValue) {\r\n const e = this._db.get(key);\r\n if (e) {\r\n for (const values of e.values()) {\r\n values.removeWhere(v => v == value);\r\n }\r\n this.deleteEntryIfEmpty(key);\r\n }\r\n }\r\n\r\n public getByKey(key: TKey): TValue[] {\r\n if (this._db.has(key)) {\r\n return (this._db.get(key) as ArrayMap).all();\r\n }\r\n return [];\r\n }\r\n\r\n public set(key: TKey, tag: string, value: TValue) {\r\n if (!this._db.has(key)) this._db.set(key, new ArrayMap());\r\n this._db.get(key)?.add(tag, value);\r\n }\r\n\r\n public deleteByTag(...tags: string[]) {\r\n const result: Array<{ node: TKey; handlers: Record }> = [];\r\n for (const [key, value] of this._db.entries()) {\r\n const nodeResult: Record = {};\r\n result.push({ node: key, handlers: nodeResult });\r\n\r\n for (const tag of tags) {\r\n const handler = value.get(tag);\r\n if (handler) {\r\n nodeResult[tag] = handler;\r\n }\r\n\r\n value.delete(tag);\r\n }\r\n this.deleteEntryIfEmpty(key);\r\n }\r\n return result;\r\n }\r\n\r\n public delete(key: TKey, tag: string) {\r\n const e = this._db.get(key);\r\n if (e) {\r\n const result = e.get(tag);\r\n e.delete(tag);\r\n this.deleteEntryIfEmpty(key);\r\n return result;\r\n }\r\n return [];\r\n }\r\n\r\n public deleteByKey(key: TKey) {\r\n const handlers = this._db.get(key);\r\n const result: Record = {};\r\n if (handlers) {\r\n handlers.forEach((value, key) => {\r\n result[key] = value;\r\n });\r\n }\r\n\r\n this._db.delete(key);\r\n\r\n return result;\r\n }\r\n\r\n public clear() {\r\n this._db.clear();\r\n }\r\n\r\n public has(key: TKey, tag: string) {\r\n return this.get(key, tag).length > 0;\r\n }\r\n\r\n protected deleteEntryIfEmpty(key: TKey) {\r\n const e = this._db.get(key);\r\n if (e && !e.size) this._db.delete(key);\r\n }\r\n}\r\n\r\nexport class ArrayMap extends Map {\r\n public get(key: TKey) {\r\n return super.get(key) || [];\r\n }\r\n\r\n public add(key: TKey, ...values: TValue[]) {\r\n if (!this.has(key)) {\r\n super.set(key, [...values]);\r\n } else {\r\n this.get(key).push(...values);\r\n }\r\n }\r\n\r\n public removeValue(key: TKey, ...values: TValue[]) {\r\n const arr = super.get(key);\r\n if (arr) {\r\n arr.removeWhere(i => values.some(v => v == i));\r\n }\r\n }\r\n\r\n public all() {\r\n return [...this.values()].flat();\r\n }\r\n}\r\n\r\nexport class SetMap extends Map> {\r\n public get(key: TKey) {\r\n const set = super.get(key);\r\n if (!set) super.set(key, new Set());\r\n return super.get(key)!;\r\n }\r\n\r\n public add(key: TKey, ...values: TValue[]) {\r\n const s = this.get(key);\r\n for (const v of values) s.add(v);\r\n }\r\n\r\n public removeValue(key: TKey, ...values: TValue[]) {\r\n const s = super.get(key);\r\n if (s) {\r\n for (const v of values) s.delete(v);\r\n }\r\n }\r\n\r\n public all() {\r\n return [...this.values()].flat();\r\n }\r\n}\r\n\r\nexport class HandlerMap extends TagArrayMap {\r\n constructor(private manager: Kb3dManager) {\r\n super();\r\n }\r\n\r\n public add(idNameOrNode: string | SpaceNode | AnnotationNode, tag: string, handler: THandler) {\r\n let nodes = new Array();\r\n if (typeof idNameOrNode == 'string') {\r\n nodes = this.manager.getNodesByIdOrName(idNameOrNode, true);\r\n } else {\r\n nodes.push(idNameOrNode);\r\n }\r\n for (const n of nodes) {\r\n if (isPickable(n)) {\r\n this.set(n, tag, handler);\r\n }\r\n }\r\n }\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { eConnectorPositionMode } from '../enums';\r\nimport { ISketchControlPoint } from '../kb3d-models';\r\nimport { SketchPath } from '../meshes/sketch-node-path';\r\nimport { KbVector } from '../primitives';\r\nimport { Token } from '../token';\r\nimport { BaseConnector } from './base-connector';\r\n\r\n@Kb3dModel({\r\n token: Token.SketchControlPoint,\r\n})\r\nexport class SketchControlPoint extends BaseConnector implements ISketchControlPoint {\r\n enabled = true; // non-configurable\r\n positionMode = eConnectorPositionMode.absolute;\r\n direction2?: KbVector;\r\n\r\n /** override base parent for more specific typing*/\r\n public _parent: SketchPath | undefined;\r\n\r\n public getSketch() {\r\n return this._parent?._parent;\r\n }\r\n\r\n public getSpaceParent() {\r\n return this.getSketch();\r\n }\r\n\r\n public getAncestors() {\r\n return super.getAncestors() as SketchPath[];\r\n }\r\n}\r\n\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { eUvMode } from '../enums';\r\nimport { IUvMapFeature } from '../kb3d-models';\r\nimport { KbVector } from '../primitives/kb-vector';\r\nimport { Token } from '../token';\r\nimport { ALL_GEOMETRY, Feature } from './feature';\r\n\r\n@Kb3dModel({\r\n token: Token.UvMapFeature,\r\n})\r\nexport class UvMapFeature extends Feature implements IUvMapFeature {\r\n static _affects = new Set(ALL_GEOMETRY);\r\n static _affectedBy = new Set(ALL_GEOMETRY);\r\n\r\n center: KbVector;\r\n projectionAxis: KbVector;\r\n worldPosition = this._parent?.position;\r\n angle: number;\r\n scale: KbVector;\r\n uTile: number;\r\n vTile: number;\r\n uOffset: number;\r\n vOffset: number;\r\n mode: eUvMode;\r\n _previewMaterialVisible: boolean;\r\n _showMaterial: boolean;\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { instanceOf } from '../decorators/model-utils';\r\nimport { ISubmeshFeature, IVertexGroup } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { Token } from '../token';\r\nimport { ALL_GEOMETRY, eMeshChangeTypes } from './feature';\r\nimport { UvMapFeature } from './uv-map-feature';\r\n\r\nexport interface SubmeshData {\r\n materialIndex: number;\r\n verticesStart: number;\r\n verticesCount: number;\r\n indexStart: number;\r\n indexCount: number;\r\n}\r\n\r\n@Kb3dModel({\r\n token: Token.SubmeshFeature,\r\n})\r\nexport class SubmeshFeature extends UvMapFeature implements ISubmeshFeature {\r\n static _dependsOn = new Set([eMeshChangeTypes.VertexIndices]);\r\n static _affects = new Set(ALL_GEOMETRY);\r\n static _affectedBy = new Set(ALL_GEOMETRY);\r\n\r\n vertexGroups: Array> = [];\r\n materialId: string;\r\n uvMapping: boolean;\r\n}\r\n\r\nexport function isSubmesh(o: Kb3dObject): o is SubmeshFeature {\r\n return instanceOf(o, Token.SubmeshFeature);\r\n}\r\n","import { NodeReference } from '../decorators/change-tracker';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { ISweepFeature } from '../kb3d-models';\r\nimport { SketchNode } from '../meshes/sketch-node';\r\nimport { Token } from '../token';\r\nimport { BaseConnector } from './base-connector';\r\nimport { ExtrudeFeature } from './extrude-feature';\r\nimport { ALL_GEOMETRY, eMeshChangeTypes } from './feature';\r\n\r\n@Kb3dModel({\r\n token: Token.SweepFeature,\r\n})\r\nexport class SweepFeature extends ExtrudeFeature implements ISweepFeature {\r\n static _affects = new Set(ALL_GEOMETRY);\r\n static _affectedBy = new Set([...ALL_GEOMETRY, eMeshChangeTypes.Transform]);\r\n\r\n @NodeReference(BaseConnector) start?: string;\r\n @NodeReference(BaseConnector) end?: string;\r\n @NodeReference(SketchNode) path?: string;\r\n\r\n size = 0;\r\n mitredExtrusion: boolean;\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { ITransformFeature } from '../kb3d-models';\r\nimport { KbVector } from '../primitives/kb-vector';\r\nimport { Token } from '../token';\r\nimport { eMeshChangeTypes, Feature } from './feature';\r\n\r\n@Kb3dModel({\r\n token: Token.TransformFeature,\r\n})\r\nexport class TransformFeature extends Feature implements ITransformFeature {\r\n static _affectedBy = new Set([eMeshChangeTypes.Transform, eMeshChangeTypes.VertexPositions]);\r\n static _affects = new Set([eMeshChangeTypes.VertexPositions]);\r\n\r\n scale: KbVector;\r\n}\r\n","import { IgnoreChanges } from '../decorators/change-tracker';\r\nimport { Enumerable } from '../decorators/enumerable';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { eFeatureCenter } from '../enums';\r\nimport { IWeldVerticesFeature } from '../kb3d-models';\r\nimport { KbVector } from '../primitives/kb-vector';\r\nimport { Token } from '../token';\r\nimport { eMeshChangeTypes, Feature } from './feature';\r\n\r\n@Kb3dModel({\r\n token: Token.WeldVerticesFeature,\r\n})\r\nexport class WeldVerticesFeature extends Feature implements IWeldVerticesFeature {\r\n static _dependsOn = new Set([eMeshChangeTypes.VertexIndices]);\r\n static _affectedBy = new Set([eMeshChangeTypes.VertexPositions, eMeshChangeTypes.VertexIndices]);\r\n static _affects = new Set([\r\n eMeshChangeTypes.VertexPositions,\r\n eMeshChangeTypes.VertexIndices,\r\n eMeshChangeTypes.VertexUvs,\r\n eMeshChangeTypes.VertexNormals,\r\n eMeshChangeTypes.Submeshes,\r\n ]);\r\n\r\n translation: KbVector;\r\n rotationAngle: number;\r\n _previewParticleSize: number;\r\n rotationAxis: KbVector;\r\n scale: KbVector;\r\n center: eFeatureCenter;\r\n\r\n @IgnoreChanges\r\n private _vertexIndices: number[];\r\n @Enumerable()\r\n public get vertexIndices(): number[] {\r\n if (!this._vertexIndices) this._vertexIndices = [];\r\n return this._vertexIndices;\r\n }\r\n public set vertexIndices(val) {\r\n this._vertexIndices = val;\r\n }\r\n}\r\n","import { KbColor } from '@models/classes/primitives/kb-color';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { NodeReference } from '../decorators/change-tracker';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { instanceOfAny } from '../decorators/model-utils';\r\nimport { eAnnotationLineStyle, eTextureHorizontalAlign, eTextureVerticalAlign } from '../enums';\r\nimport { Connector } from '../features';\r\nimport { IAnnotationNode } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { Token } from '../token';\r\n\r\n@Kb3dModel({\r\n token: Token.AnnotationNode,\r\n})\r\nexport class AnnotationNode extends Kb3dObject implements IAnnotationNode {\r\n public constructor() {\r\n super();\r\n }\r\n\r\n isPickable: boolean;\r\n enabled: boolean;\r\n billboard: boolean;\r\n color: KbColor;\r\n fontFamily: string;\r\n fontSize: number;\r\n keepOnTop: boolean;\r\n offset: KbVector;\r\n lineStyle: eAnnotationLineStyle;\r\n lineSpacing: number;\r\n horizontalTextAlign: eTextureHorizontalAlign;\r\n horizontalBoxAlign: eTextureHorizontalAlign;\r\n verticalBoxAlign: eTextureVerticalAlign;\r\n label = '';\r\n preloadFontStylesheet = '';\r\n\r\n @NodeReference(Connector, 'position', 'direction', 'angle')\r\n targetId?: string;\r\n}\r\n\r\nexport function isAnnotation(node: Kb3dObject): node is AnnotationNode {\r\n return instanceOfAny(node, Token.AnnotationNode, Token.DimensionNode);\r\n}\r\n","import { NodeReference } from '../decorators/change-tracker';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { eDimensionAxis, eDimensionLineCapStyle } from '../enums';\r\nimport { Connector } from '../features/connector';\r\nimport { IDimensionNode } from '../kb3d-models';\r\nimport { KbVector } from '../primitives';\r\nimport { Token } from '../token';\r\nimport { AnnotationNode } from './annotation-node';\r\n\r\n@Kb3dModel({\r\n token: Token.DimensionNode,\r\n})\r\nexport class DimensionNode extends AnnotationNode implements IDimensionNode {\r\n public constructor() {\r\n super();\r\n }\r\n\r\n lineCapStyle: eDimensionLineCapStyle;\r\n axis: eDimensionAxis;\r\n lineThickness: number;\r\n dimensionGap: number;\r\n labelOffset: KbVector;\r\n dimensionUnitPrecision: number;\r\n scaleWithCamera: boolean;\r\n\r\n @NodeReference(Connector, 'position')\r\n target2Id?: string;\r\n}\r\n","import { Kb3dModel } from './decorators/kb3d-decorator';\r\nimport { IImportInfo } from './kb3d-models';\r\nimport { Token } from './token';\r\n\r\n@Kb3dModel({\r\n token: Token.ImportInfo,\r\n trackChanges: false,\r\n})\r\nexport class ImportInfo implements IImportInfo {}\r\n","import { isAbsoluteUrl } from '../classes/utilities';\r\nimport { HttpService } from './http-service';\r\n\r\nconst fontFilesLoading = new Map>();\r\nconst stylesheetsLoading = new Map>();\r\nconst fontFamilySrcs = new Map();\r\n\r\nexport class FontService {\r\n constructor(private httpService: HttpService, private assetRoot = '') {}\r\n\r\n /** Can be both the font face name as well as the font file URL */\r\n public readyFonts = new Map();\r\n\r\n loadFont(stylesheetUrl: string, fontFamilies: string[]): Promise {\r\n const fontFaceSet: FontFaceSet = (document as any).fonts;\r\n if (typeof fontFaceSet === 'undefined') {\r\n return Promise.reject('FontFaceSet api not available');\r\n }\r\n\r\n return this.loadStylesheet(stylesheetUrl).then(() => this.loadFontFamily(fontFaceSet, fontFamilies));\r\n }\r\n\r\n loadFontByFile(path: string) {\r\n const absPath = isAbsoluteUrl(path) ? path : this.getAbsoluteFontPath(path);\r\n let loadPromise: Promise;\r\n if (fontFilesLoading.has(absPath)) {\r\n loadPromise = fontFilesLoading.get(absPath)!;\r\n } else {\r\n const promise = this.httpService.getPublicStatic(absPath, 'arraybuffer');\r\n fontFilesLoading.set(absPath, promise);\r\n loadPromise = promise;\r\n }\r\n\r\n return loadPromise.then(buffer => {\r\n if (buffer.byteLength === 0) {\r\n return Promise.reject('Could not fetch font');\r\n } else {\r\n this.readyFonts.set(path, buffer);\r\n }\r\n return buffer;\r\n });\r\n }\r\n\r\n loadFontData(stylesheetUrl: string, fontFamilies: string[]) {\r\n return this.loadStylesheet(stylesheetUrl).then(() => this.getFontData(fontFamilies));\r\n }\r\n\r\n getAbsoluteFontPath(relativePath: string) {\r\n return `${this.assetRoot}/media/${relativePath}`; //it's a relative path to the media folder\r\n }\r\n\r\n private loadStylesheet(stylesheetUrl: string) {\r\n let stylesheetLoad = stylesheetsLoading.get(stylesheetUrl);\r\n if (!stylesheetLoad) {\r\n stylesheetLoad = this.httpService.getPublicStatic(stylesheetUrl).then(res => {\r\n // Parse the css and store the srcs to the font files\r\n const fontFaces = res.matchAll(/@font-face\\s*{([\\s\\S]*?)}/gm);\r\n for (const face of fontFaces) {\r\n const fontFamilyMatch = face[1].match(/font-family\\s*:\\s*['\"]?(.*?)['\"]?[;,\\n]/m);\r\n\r\n if (fontFamilyMatch) {\r\n const fontFamilyName = fontFamilyMatch[1];\r\n const srcMatch = face[1].match(/src\\s*:.*?url\\s*\\([\\s\"']*(.*?)[\\s\"']*\\)/m);\r\n if (srcMatch) {\r\n const fontSrc = srcMatch[1];\r\n fontFamilySrcs.set(fontFamilyName.trim().toLowerCase(), fontSrc.trim());\r\n }\r\n }\r\n }\r\n });\r\n stylesheetsLoading.set(stylesheetUrl, stylesheetLoad);\r\n }\r\n\r\n return stylesheetLoad;\r\n }\r\n\r\n private getFontData(fontFamilies: string[]) {\r\n const loadFaces: string[] = [];\r\n const loadPromises: Array> = [];\r\n\r\n fontFamilies.forEach(fontFamily => {\r\n const fontSrc = fontFamilySrcs.get(fontFamily.trim().toLowerCase());\r\n if (!fontSrc) {\r\n console.warn(`Font family ${fontFamily} not found. Is it being loaded by the preloaded stylesheet?`);\r\n } else if (fontFilesLoading.has(fontFamily)) {\r\n loadPromises.push(fontFilesLoading.get(fontFamily)!);\r\n loadFaces.push(fontFamily);\r\n } else {\r\n const promise = this.httpService.getPublicStatic(fontSrc, 'arraybuffer');\r\n fontFilesLoading.set(fontFamily, promise);\r\n loadPromises.push(promise);\r\n loadFaces.push(fontFamily);\r\n }\r\n });\r\n\r\n const resMap: Record = {};\r\n return Promise.all(loadPromises).then(buffers => {\r\n buffers.forEach((buffer, index) => {\r\n resMap[loadFaces[index]] = buffer;\r\n this.readyFonts.set(loadFaces[index], buffer);\r\n });\r\n return resMap;\r\n });\r\n }\r\n\r\n private loadFontFamily(fontFaceSet: FontFaceSet, fontFamilies: string[]): Promise {\r\n return this.getFontData(fontFamilies).then(resultData => {\r\n const resultFonts: FontFace[] = [];\r\n fontFamilies.forEach(fontFamily => {\r\n if (resultData[fontFamily]) {\r\n const newFontFace = new FontFace(fontFamily, resultData[fontFamily]);\r\n // @ts-expect-error TS 5.1.6 definition does not include add on FontFaceSet\r\n fontFaceSet.add(newFontFace);\r\n resultFonts.push(newFontFace);\r\n } else {\r\n console.warn('Could not load font ' + fontFamily);\r\n }\r\n });\r\n return resultFonts;\r\n });\r\n }\r\n}\r\n","import { isAbsoluteUrl } from '@models/classes/utilities';\r\nimport { Observable } from 'rxjs';\r\n\r\nexport enum eContentType {\r\n json = 'application/json',\r\n formData = 'multipart/form-data',\r\n}\r\n\r\nexport interface IHttpEnvironment {\r\n lastDeploy?: number;\r\n cacheBuster?: string;\r\n}\r\n\r\ninterface IDeferredAssetCall {\r\n func: () => Promise;\r\n resolve: (value: T | PromiseLike) => void;\r\n reject: (reason?: any) => void;\r\n priority: number;\r\n}\r\n\r\nexport interface IBinaryResponse {\r\n data: T;\r\n width?: number;\r\n height?: number;\r\n}\r\n\r\nexport const ABORTED = 'ABORTED';\r\n\r\nexport class HttpService {\r\n public trackedAssetCalls = new Array>();\r\n private deferredAssetCalls = new Array>();\r\n private httpCallsToAbort = new Map();\r\n private deferredExecuted = false;\r\n\r\n constructor(private environment?: IHttpEnvironment) {}\r\n\r\n public addCacheParams(url: string): string {\r\n return this.getUrlWithParams(url, this.getEnvironmentCacheParams());\r\n }\r\n\r\n public get(url: string, params?: Record, trackAssetCall = true) {\r\n if (!isAbsoluteUrl(url) && this.environment) {\r\n // api calls in the test and prod environments should send the last deploy,\r\n // which allows the browser cache to store results for some calls\r\n if (this.environment) {\r\n const addParams = this.getEnvironmentCacheParams();\r\n params = params ? { ...params, ...addParams } : addParams;\r\n }\r\n }\r\n\r\n const u = this.getUrlWithParams(url, params);\r\n const p = this.fetchWrapper(u, 'GET');\r\n if (trackAssetCall) {\r\n this.trackAssetCall(p);\r\n }\r\n return p;\r\n }\r\n\r\n public getPublicStatic(\r\n url: string,\r\n type: 'json',\r\n params?: Record,\r\n trackAssetCall?: boolean\r\n ): Promise;\r\n public getPublicStatic(\r\n url: string,\r\n type: 'blob',\r\n params?: Record,\r\n trackAssetCall?: boolean\r\n ): Promise;\r\n public getPublicStatic(\r\n url: string,\r\n type: 'arraybuffer',\r\n params?: Record,\r\n trackAssetCall?: boolean\r\n ): Promise;\r\n public getPublicStatic(\r\n url: string,\r\n type?: 'text',\r\n params?: Record,\r\n trackAssetCall?: boolean\r\n ): Promise;\r\n public getPublicStatic(\r\n url: string,\r\n type: 'text' | 'blob' | 'json' | 'arraybuffer' = 'text',\r\n params?: Record,\r\n trackAssetCall = true\r\n ) {\r\n const u = this.getUrlWithParams(url, params);\r\n\r\n const requestInit = {\r\n method: 'GET',\r\n credentials: 'omit',\r\n } as RequestInit;\r\n const p = fetch(u, requestInit).then(res => {\r\n if (type === 'text') {\r\n return res.text();\r\n } else if (type === 'blob') {\r\n return res.blob();\r\n } else if (type === 'arraybuffer') {\r\n return res.arrayBuffer();\r\n } else {\r\n return res.json();\r\n }\r\n });\r\n\r\n if (trackAssetCall) {\r\n this.trackAssetCall(p);\r\n }\r\n return p;\r\n }\r\n\r\n public post(url: string, body?: any, contentType = eContentType.json) {\r\n return this.fetchWrapper(url, 'POST', body, contentType);\r\n }\r\n public put(url: string, body?: any, contentType = eContentType.json) {\r\n return this.fetchWrapper(url, 'PUT', body, contentType);\r\n }\r\n public delete(url: string, params?: Record) {\r\n const u = this.getUrlWithParams(url, params);\r\n return this.fetchWrapper(u, 'DELETE');\r\n }\r\n\r\n getBinary(\r\n url: string,\r\n responseType: 'arraybuffer',\r\n trackAssetCall: boolean,\r\n appendCacheParameters?: boolean,\r\n abortSubject?: Observable\r\n ): Promise>;\r\n getBinary(\r\n url: string,\r\n responseType: 'blob',\r\n trackAssetCall: boolean,\r\n appendCacheParameters?: boolean,\r\n abortSubject?: Observable\r\n ): Promise>;\r\n public getBinary(\r\n url: string,\r\n responseType: 'arraybuffer' | 'blob',\r\n trackAssetCall: boolean,\r\n appendCacheParameters = false,\r\n abortSubject?: Observable\r\n ): Promise> {\r\n const xhr = new XMLHttpRequest();\r\n if (appendCacheParameters && !isAbsoluteUrl(url) && this.environment) {\r\n if (this.environment) {\r\n const addParams = this.getEnvironmentCacheParams();\r\n url = this.getUrlWithParams(url, addParams);\r\n }\r\n }\r\n const abortSub = abortSubject?.subscribe(() => {\r\n xhr.abort();\r\n });\r\n\r\n const p = new Promise>((resolve, reject) => {\r\n xhr.open('GET', url, true);\r\n xhr.responseType = responseType;\r\n\r\n xhr.onload = function (oEvent) {\r\n if (xhr.status >= 200 && xhr.status < 300) {\r\n const allHeaders = xhr.getAllResponseHeaders();\r\n let height = null,\r\n width = null;\r\n if (allHeaders.match(/Original-Height/i)) {\r\n height = xhr.getResponseHeader('X-Original-Height');\r\n width = xhr.getResponseHeader('X-Original-Width');\r\n }\r\n const response: IBinaryResponse = {\r\n data: xhr.response,\r\n };\r\n if (height && width) {\r\n response.height = parseInt(height);\r\n response.width = parseInt(width);\r\n }\r\n resolve(response);\r\n } else {\r\n reject(xhr.statusText);\r\n }\r\n abortSub?.unsubscribe();\r\n };\r\n xhr.onerror = e => {\r\n reject(e);\r\n abortSub?.unsubscribe();\r\n };\r\n xhr.onabort = () => {\r\n reject(ABORTED);\r\n // do nothing\r\n };\r\n\r\n xhr.send(null);\r\n });\r\n\r\n if (trackAssetCall) {\r\n this.trackAssetCall(p);\r\n }\r\n\r\n return p;\r\n }\r\n\r\n /**\r\n * delays execution of parameter function until all blocking assets have been loaded for the scene.\r\n * executes immediately if scene is already loaded.\r\n */\r\n defer(func: () => Promise, priority = 100): Promise {\r\n if (this.deferredExecuted) {\r\n return func();\r\n } else {\r\n return new Promise((resolve, reject) => {\r\n this.deferredAssetCalls.push({ func, resolve, reject, priority });\r\n });\r\n }\r\n }\r\n\r\n executeDeferredCalls() {\r\n this.deferredAssetCalls.sort((a, b) => a.priority - b.priority);\r\n for (const call of this.deferredAssetCalls) {\r\n call.func().then(call.resolve).catch(call.reject);\r\n }\r\n this.deferredAssetCalls = [];\r\n this.deferredExecuted = true;\r\n }\r\n\r\n protected trackAssetCall(p: Promise) {\r\n const trackedPromise = p.finally(() => {\r\n this.trackedAssetCalls.remove(trackedPromise);\r\n });\r\n this.trackedAssetCalls.push(trackedPromise);\r\n }\r\n\r\n private getEnvironmentCacheParams(): Record {\r\n if (this.environment) {\r\n const params: Record = {};\r\n if (this.environment.lastDeploy) {\r\n params.d = this.environment.lastDeploy.toString();\r\n }\r\n if (this.environment.cacheBuster) {\r\n params.v = this.environment.cacheBuster;\r\n }\r\n return params;\r\n }\r\n return {};\r\n }\r\n\r\n private getUrlWithParams(url: string, params?: Record) {\r\n if (!params) return url;\r\n const urlParams = new URLSearchParams(Object.entries(params));\r\n\r\n let prefix = '?';\r\n if (url.match(/\\?/)) {\r\n prefix = '&';\r\n }\r\n\r\n return `${url}${prefix}${urlParams.toString()}`;\r\n }\r\n\r\n private fetchWrapper(\r\n url: string,\r\n method: string,\r\n body?: T,\r\n contentType: eContentType = eContentType.json\r\n ): Promise {\r\n const headers: Headers = new Headers();\r\n\r\n const requestInit = {\r\n method,\r\n headers,\r\n credentials: 'include',\r\n } as RequestInit;\r\n\r\n if (method.equalsAny('POST', 'PUT')) {\r\n if (contentType == eContentType.json) {\r\n headers.append('Content-Type', contentType);\r\n requestInit.body = JSON.stringify(body);\r\n } else if (contentType == eContentType.formData) {\r\n requestInit.body = body as any as FormData;\r\n }\r\n }\r\n\r\n const p = fetch(url, requestInit).then(r => {\r\n if (r.ok) {\r\n return r.json().catch(() => body);\r\n } else {\r\n throw new Error(JSON.stringify(r));\r\n }\r\n });\r\n return p as Promise;\r\n }\r\n}\r\n","export function getRotatedDimensions(width: number, height: number, rAngle: number) {\r\n const rad = (rAngle * Math.PI) / 180;\r\n return {\r\n width: width * Math.cos(rad) + height * Math.sin(rad),\r\n height: width * Math.sin(rad) + height * Math.cos(rad),\r\n };\r\n}\r\n\r\nexport function scaleImagePreserveAspect(\r\n targetWidth: number,\r\n targetHeight: number,\r\n naturalWidth?: number,\r\n naturalHeight?: number\r\n) {\r\n let drawWidth: number, drawHeight: number;\r\n const imageWidth = naturalWidth || 256;\r\n const imageHeight = naturalHeight || 256;\r\n if (imageWidth > imageHeight) {\r\n drawWidth = targetWidth;\r\n drawHeight = (imageHeight / imageWidth) * targetWidth;\r\n } else {\r\n drawWidth = (imageWidth / imageHeight) * targetHeight;\r\n drawHeight = targetHeight;\r\n }\r\n\r\n return {\r\n drawWidth: Math.ceil(drawWidth),\r\n drawHeight: Math.ceil(drawHeight),\r\n };\r\n}\r\n\r\nexport function getBinClampedSize(width: number, height: number, rounding: 'floor' | 'ceil' = 'floor') {\r\n if (rounding === 'floor') {\r\n const log2 = Math.min(Math.log2(width), Math.log2(height));\r\n return Math.pow(2, Math.floor(log2));\r\n } else {\r\n const log2 = Math.max(Math.log2(width), Math.log2(height));\r\n return Math.pow(2, Math.ceil(log2));\r\n }\r\n}\r\n","export const KBMAXPREFIX = '____kbm____';\r\nexport const LOCAL_ENVIRONMENT_DOMAIN = 'library://';\r\n","const SLOW_NETWORK_THRESHOLD = 100; // kB per second\r\n\r\nlet slowWarningThrown = false;\r\n\r\nexport function attachLoadingTimer(p: Promise) {\r\n const startTime = performance.now();\r\n\r\n p.then(result => {\r\n if (slowWarningThrown) {\r\n return;\r\n }\r\n\r\n let fileSize = 0;\r\n const endTime = performance.now();\r\n if ('byteLength' in result.data) {\r\n fileSize = result.data.byteLength;\r\n } else if ('size' in result.data) {\r\n fileSize = result.data.size;\r\n }\r\n\r\n if (startTime && endTime && fileSize > 50000) {\r\n const resultTime = endTime - startTime; // in milliseconds\r\n\r\n // if download speed is less than 100 kB per second, throw slow loading warning\r\n if (fileSize / resultTime < SLOW_NETWORK_THRESHOLD) {\r\n console.warn('Slow network detected. Scene experience may be degraded.');\r\n slowWarningThrown = true;\r\n }\r\n }\r\n }).catch(() => {\r\n // Do nothing\r\n });\r\n\r\n return p;\r\n}\r\n","import { isMobile } from '@common-util/device-helper';\r\nimport { scaleImagePreserveAspect } from '@common-util/texture-util';\r\nimport { FileTextureLayer } from '@models/classes';\r\nimport { Kb3dService } from '@models/classes/decorators/service-decorator';\r\nimport { eMaterialLocation, eTextureChannel } from '@models/classes/enums';\r\nimport { IAdvancedMaterialNode, IFilter, IMaterial, ISearchArgs, ITextureLayer } from '@models/classes/kb3d-models';\r\nimport { AdvancedMaterialNode } from '@models/classes/materials/advanced-material-node';\r\nimport { Token } from '@models/classes/token';\r\nimport { isAbsoluteUrl } from '@models/classes/utilities';\r\nimport { LOCAL_ENVIRONMENT_DOMAIN } from '@models/constants';\r\nimport { BehaviorSubject, Observable, Subject, combineLatest, from, of } from 'rxjs';\r\nimport { map, shareReplay, switchMap, take } from 'rxjs/operators';\r\nimport { ABORTED, HttpService, IBinaryResponse } from './http-service';\r\nimport { attachLoadingTimer } from './loading-timer-helper';\r\n\r\nconst LIBRARY_PATH = '%LIBRARY%';\r\n\r\n// If this is a raster image, we store the bitmap. If this is an SVG, then we store the blob so it can be resized efficiently\r\nconst textureCache = new Map>();\r\nconst materialCache = new Map>();\r\n\r\nexport interface IMaterialService {\r\n getMaterialById: (id: string) => Observable;\r\n getMaterialLibraryCategories: () => IMaterialCategory[];\r\n getMaterialLibraryOptions: (category: string) => IMaterial[];\r\n getTexturePath: (node: FileTextureLayer, detailLevel?: number) => string;\r\n getTextureImage: (node: FileTextureLayer) => Observable<{ bitmap: ImageBitmap; url: string }>;\r\n getLibraryTexturePath: (path: string, quality?: number) => string;\r\n}\r\n\r\nexport interface IMaterialCategory {\r\n id: string;\r\n label: string;\r\n}\r\n\r\nconst TEXTURE_SIZES = [128, 256, 512, 1024, 2048, null];\r\nconst LIBRARY_TEXTURE_SIZES = [256, 256, 512, 1024, 2048, 2048];\r\nconst LIBRARY_ENV_SIZES = [128, 256, 512, 1024];\r\nconst LIBRARY_MOBILE_ENV_SIZES = [128, 128, 256, 512];\r\nconst MAX_DETAIL_LEVEL = 5;\r\n\r\n@Kb3dService({\r\n token: Token.MaterialService,\r\n})\r\nexport class MaterialService implements IMaterialService {\r\n public trackedAssetCalls = new Array>();\r\n\r\n materialUpdateChannel = new Map>();\r\n private abortSubjects = new Map>();\r\n textureDetailLevel = MAX_DETAIL_LEVEL;\r\n\r\n constructor(\r\n protected httpService: HttpService,\r\n protected assetRoot?: string,\r\n protected forceFullSizeTextures = false\r\n ) {\r\n if (forceFullSizeTextures) {\r\n this.textureDetailLevel = -1;\r\n } else if (window.screen.width <= 1024 || window.screen.height <= 1024) {\r\n this.textureDetailLevel = 2;\r\n }\r\n }\r\n\r\n public getMaterialLibraryCategories(): IMaterialCategory[] {\r\n const categoryIds = Object.keys(materialDefinitionSet);\r\n\r\n return categoryIds.map(id => ({\r\n id,\r\n label: materialDefinitionSet[id].label,\r\n }));\r\n }\r\n\r\n public getMaterialLibraryOptions(category: string): IMaterial[] {\r\n return LibraryMaterialsByCategory.get(category) || [];\r\n }\r\n\r\n public getMaterialById(guid: string): Observable {\r\n if (materialCache.has(guid)) {\r\n return materialCache.get(guid) as Observable;\r\n } else {\r\n let obs: Observable;\r\n const material = LibraryMaterials.get(guid);\r\n\r\n if (material) {\r\n //it's a library material\r\n obs = of(\r\n material\r\n //asapScheduler\r\n );\r\n } else {\r\n //maybe a global material\r\n obs = from(this.httpService.get(`/api/materials/${guid}`));\r\n }\r\n const updateChannel = new BehaviorSubject(null);\r\n this.materialUpdateChannel.set(guid, updateChannel);\r\n\r\n obs = combineLatest([obs, updateChannel])\r\n .pipe(\r\n map(([material, updatedMaterial]) => {\r\n if (!material) {\r\n return undefined;\r\n }\r\n return { ...material, ...(updatedMaterial || {}) };\r\n })\r\n )\r\n .pipe(shareReplay(1));\r\n\r\n materialCache.set(guid, obs);\r\n return obs;\r\n }\r\n }\r\n\r\n public getTexturePath(node: FileTextureLayer, detailLevel?: number) {\r\n let width: number | null = null;\r\n let height: number | null = null;\r\n if (!node.path) {\r\n return '';\r\n }\r\n\r\n if (detailLevel === undefined) {\r\n detailLevel = this.textureDetailLevel;\r\n }\r\n detailLevel = Math.min(MAX_DETAIL_LEVEL, detailLevel);\r\n\r\n if (node.path.startsWith(LIBRARY_PATH)) {\r\n return this.getLibraryTexturePath(node.path, detailLevel);\r\n } else if (!isAbsoluteUrl(node.path)) {\r\n const url = this.getAbsoluteTexturePath(node.path);\r\n if (detailLevel >= 0) {\r\n width = TEXTURE_SIZES[detailLevel];\r\n }\r\n if (detailLevel >= 0) {\r\n height = TEXTURE_SIZES[detailLevel];\r\n }\r\n\r\n const params: string[] = [];\r\n if (width != null) params.push(`width=${width}`);\r\n if (height != null) params.push(`height=${height}`);\r\n if (params.length > 0) {\r\n return url + '?' + params.join('&');\r\n } else {\r\n return url;\r\n }\r\n } else {\r\n return node.path;\r\n }\r\n }\r\n\r\n public getTextureImage(node: FileTextureLayer, trackAssetCall = true) {\r\n const initialUrl = this.getTexturePath(node, 0);\r\n const url = this.getTexturePath(node);\r\n\r\n let bitmapObs: Observable<{ bitmap: ImageBitmap | Blob; url: string }>;\r\n\r\n if (textureCache.has(url)) {\r\n bitmapObs = textureCache.get(url)!;\r\n } else {\r\n bitmapObs = new Observable<{ bitmap: ImageBitmap | Blob; url: string }>(subject => {\r\n const activeLinkAbort = this.abortSubjects.get(node._dynamicId);\r\n // if there already is an httpService call on this node. Abort the previous one\r\n if (activeLinkAbort) {\r\n activeLinkAbort.next();\r\n }\r\n\r\n const abortTextureLoad = new Subject();\r\n this.abortSubjects.set(node._dynamicId, abortTextureLoad);\r\n\r\n // Textures are absolute so we have to add the cache params before passing to http service\r\n // With this code library textures are always progressively loaded regardless of the scene setting because they do not have a root scene\r\n // revisit this if we want to change this behavior but it is non-trivial to do so\r\n if (this.forceFullSizeTextures || (node._rootScene && !node._rootScene.progressiveTextureLoad)) {\r\n const cacheParamUrl =\r\n this.assetRoot && url.startsWith(this.assetRoot) ? this.httpService.addCacheParams(url) : url;\r\n\r\n attachLoadingTimer(\r\n this.httpService.getBinary(cacheParamUrl, 'blob', false, true, abortTextureLoad)\r\n ).then(\r\n blobResult => {\r\n // clear the abort once the full image is loaded\r\n const activeLinkAbort = this.abortSubjects.get(node._dynamicId);\r\n if (activeLinkAbort) {\r\n activeLinkAbort.complete();\r\n this.abortSubjects.delete(node._dynamicId);\r\n }\r\n\r\n processImageBlob(blobResult).then(result => {\r\n if (result) {\r\n subject.next({ bitmap: result, url: cacheParamUrl });\r\n subject.complete();\r\n }\r\n });\r\n },\r\n err => {\r\n if (err !== ABORTED) {\r\n console.warn(`Could not load texture ${cacheParamUrl}: ` + err.toString());\r\n }\r\n subject.complete();\r\n }\r\n );\r\n } else {\r\n const initialCacheParamUrl =\r\n this.assetRoot && url.startsWith(this.assetRoot)\r\n ? this.httpService.addCacheParams(initialUrl)\r\n : initialUrl;\r\n\r\n const loadStartTime = performance.now();\r\n attachLoadingTimer(\r\n this.httpService.getBinary(initialCacheParamUrl, 'blob', false, true, abortTextureLoad)\r\n ).then(\r\n initialBlobResult => {\r\n processImageBlob(initialBlobResult).then(result => {\r\n const loadTime = performance.now() - loadStartTime;\r\n if (loadTime > 2000) {\r\n console.warn(`Texture ${initialCacheParamUrl} took ${loadTime}ms to load`);\r\n }\r\n if (result) {\r\n subject.next({ bitmap: result, url: initialCacheParamUrl });\r\n }\r\n const cacheParamUrl =\r\n this.assetRoot && url.startsWith(this.assetRoot)\r\n ? this.httpService.addCacheParams(url)\r\n : url;\r\n this.httpService\r\n .defer(() =>\r\n this.httpService.getBinary(cacheParamUrl, 'blob', false, true, abortTextureLoad)\r\n )\r\n .then(\r\n blobResult => {\r\n // clear the abort once the full image is loaded\r\n const activeLinkAbort = this.abortSubjects.get(node._dynamicId);\r\n if (activeLinkAbort) {\r\n activeLinkAbort.complete();\r\n this.abortSubjects.delete(node._dynamicId);\r\n }\r\n\r\n processImageBlob(blobResult).then(result => {\r\n if (result) {\r\n subject.next({ bitmap: result, url: cacheParamUrl });\r\n subject.complete();\r\n }\r\n });\r\n },\r\n err => {\r\n if (err !== ABORTED) {\r\n console.warn(\r\n `Could not load texture ${cacheParamUrl}: ` + err.toString()\r\n );\r\n }\r\n subject.complete();\r\n }\r\n );\r\n });\r\n },\r\n err => {\r\n if (err !== ABORTED) {\r\n console.warn(`Could not load texture ${initialCacheParamUrl}: ` + err.toString());\r\n }\r\n subject.complete();\r\n }\r\n );\r\n }\r\n }).pipe(shareReplay(1));\r\n }\r\n\r\n function processImageBlob(blob: IBinaryResponse): Promise {\r\n if (blob && blob.data) {\r\n if (blob.data.type === 'image/svg+xml') {\r\n return Promise.resolve(blob.data);\r\n } else {\r\n if (blob.data.type.match(/image/)) {\r\n const opts =\r\n blob.width && blob.height\r\n ? {\r\n resizeWidth: blob.width,\r\n resizeHeight: blob.height,\r\n resizeQuality: 'medium' as ResizeQuality,\r\n }\r\n : undefined;\r\n return createImageBitmap(blob.data, opts).then(bitmap => {\r\n return Promise.resolve(bitmap);\r\n });\r\n } else {\r\n console.warn('source file is not an image: ', url);\r\n }\r\n }\r\n }\r\n return Promise.resolve(null);\r\n }\r\n\r\n textureCache.set(url, bitmapObs);\r\n\r\n const obs = bitmapObs.pipe(\r\n switchMap(res => {\r\n /** bitmap is returned as blob for SVGs and imageBitmap for raster images */\r\n if (res.bitmap instanceof Blob) {\r\n const blobToConvert = res.bitmap;\r\n return new Promise<{ bitmap: ImageBitmap; url: string }>((resolve, reject) => {\r\n const imageElement = document.createElement('img');\r\n\r\n const blobUrl = URL.createObjectURL(blobToConvert);\r\n\r\n imageElement.onload = () => {\r\n // Figure out what resolution to actually draw the SVG at to fit within the envelope\r\n const scaled = scaleImagePreserveAspect(\r\n node.width,\r\n node.height,\r\n imageElement.naturalWidth,\r\n imageElement.naturalHeight\r\n );\r\n\r\n createImageBitmap(imageElement, {\r\n resizeWidth: scaled.drawWidth,\r\n resizeHeight: scaled.drawHeight,\r\n }).then(bitmap => {\r\n resolve({ bitmap: bitmap, url: res.url });\r\n });\r\n };\r\n imageElement.onerror = e => {\r\n reject(e);\r\n };\r\n imageElement.src = blobUrl;\r\n });\r\n } else {\r\n return of({ bitmap: res.bitmap, url: res.url });\r\n }\r\n })\r\n );\r\n\r\n if (trackAssetCall) {\r\n this.trackAssetCall(obs.pipe(take(1)).toPromise());\r\n }\r\n\r\n return obs;\r\n }\r\n\r\n public getLibraryTexturePath(relativePath: string, quality = 0) {\r\n const path = relativePath.substring(LIBRARY_PATH.length);\r\n let res: number;\r\n if (quality < 0) {\r\n res = LIBRARY_TEXTURE_SIZES[LIBRARY_TEXTURE_SIZES.length - 1];\r\n } else {\r\n res = LIBRARY_TEXTURE_SIZES[quality];\r\n }\r\n return `${this.assetRoot}/assets/textures/${path}_${res}.jpg`;\r\n }\r\n\r\n public getAbsoluteTexturePath(relativePath: string) {\r\n return `${this.assetRoot}/media/${relativePath}`; //it's a relative path to the media folder\r\n }\r\n\r\n public getAbsoluteEnvironmentPath(path: string, quality: 0 | 1 | 2 | 3) {\r\n if (path) {\r\n let url: string;\r\n if (path.startsWith(LOCAL_ENVIRONMENT_DOMAIN)) {\r\n const res = isMobile() ? LIBRARY_MOBILE_ENV_SIZES[quality] : LIBRARY_ENV_SIZES[quality];\r\n\r\n url = `${this.assetRoot}/assets/environments/${path.replace(LOCAL_ENVIRONMENT_DOMAIN, '')}_${res}.env`;\r\n } else {\r\n url = this.getAbsoluteTexturePath(path);\r\n }\r\n return this.httpService.addCacheParams(url);\r\n }\r\n return '';\r\n }\r\n\r\n public searchMaterialByName(name: string, loadArchivedMaterials: boolean) {\r\n const archivedFilter = { type: 'property', property: 'archived', values: [false] } as IFilter;\r\n const searchArgs = { query: name + '*', sortField: 'Name', filters: [], descending: false } as ISearchArgs;\r\n if (!loadArchivedMaterials) {\r\n searchArgs.filters?.push(archivedFilter);\r\n }\r\n let result = null;\r\n if (name) {\r\n result = this.httpService.post('/api/materials/search', searchArgs);\r\n }\r\n return result;\r\n }\r\n\r\n public getMaterialNameList() {\r\n const archivedFilter = { type: 'property', property: 'archived', values: [false] } as IFilter;\r\n const searchArgs = { fields: ['Guid', 'Name'], take: 10000, filters: [archivedFilter] } as ISearchArgs;\r\n return this.httpService.post('/api/materials/search', searchArgs);\r\n }\r\n\r\n protected setMaterialCache(guid: string, mat: IMaterial) {\r\n materialCache.set(guid, of(mat));\r\n }\r\n\r\n protected trackAssetCall(p: Promise) {\r\n const trackedPromise = p.finally(() => {\r\n this.trackedAssetCalls.remove(trackedPromise);\r\n });\r\n this.trackedAssetCalls.push(trackedPromise);\r\n }\r\n}\r\n\r\nexport interface IEnvironmentInfo {\r\n name: string;\r\n label: string;\r\n path: string;\r\n previewUrl: string;\r\n}\r\n\r\nexport const LibraryMaterials = new Map();\r\nexport const LibraryMaterialsByCategory = new Map();\r\nexport const LibraryEnvironments = new Array();\r\nexport const LibraryEnvironmentsMap = new Map();\r\n\r\nfunction makeLibraryMaterial(tag: string, index: number) {\r\n //pad the index string with zeros\r\n let indexStr = index + '';\r\n while (indexStr.length < 2) indexStr = '0' + indexStr;\r\n const iname = tag + indexStr;\r\n const baseUrl = `${tag}/${iname}`;\r\n\r\n return {\r\n id: -1,\r\n name: iname,\r\n guid: iname,\r\n tags: [{ name: tag }],\r\n data: {\r\n $type: Token.AdvancedMaterialNode.key,\r\n name: iname,\r\n id: iname,\r\n location: eMaterialLocation.library,\r\n textures: [\r\n {\r\n $type: 'FileTextureLayer',\r\n channel: eTextureChannel.ambientOcclusion,\r\n path: `${LIBRARY_PATH}${baseUrl}_ao`,\r\n } as ITextureLayer,\r\n {\r\n $type: 'FileTextureLayer',\r\n channel: eTextureChannel.albedo,\r\n path: `${LIBRARY_PATH}${baseUrl}_col`,\r\n } as ITextureLayer,\r\n {\r\n $type: 'FileTextureLayer',\r\n channel: eTextureChannel.metallic,\r\n path: `${LIBRARY_PATH}${baseUrl}_met`,\r\n } as ITextureLayer,\r\n {\r\n $type: 'FileTextureLayer',\r\n channel: eTextureChannel.normal,\r\n path: `${LIBRARY_PATH}${baseUrl}_nrm`,\r\n } as ITextureLayer,\r\n {\r\n $type: 'FileTextureLayer',\r\n channel: eTextureChannel.roughness,\r\n path: `${LIBRARY_PATH}${baseUrl}_rgh`,\r\n } as ITextureLayer,\r\n //{ $type: \"FileTextureLayer\", channel: eTextureChannel.displacement, path: `${LIBRARY_PATH}${baseUrl}_disp` },\r\n ],\r\n },\r\n };\r\n}\r\n\r\ninterface IMaterialDefinitionSet {\r\n [tagName: string]: {\r\n label: string;\r\n count: number;\r\n omit?: Partial<{\r\n //the textures to omit for\r\n [T in eTextureChannel]: boolean | number[];\r\n }>;\r\n props?: Partial & { [key: string]: any };\r\n };\r\n}\r\n\r\nfunction fillLibraryMaterialsMap() {\r\n for (const tag in materialDefinitionSet) {\r\n const entry = materialDefinitionSet[tag];\r\n for (let index = 1; index <= entry.count; index++) {\r\n const mat = makeLibraryMaterial(tag, index);\r\n const matData = mat.data as AdvancedMaterialNode;\r\n matData.roughness = 1;\r\n matData.metallic = 0;\r\n\r\n if (entry.omit) {\r\n const omit = entry.omit;\r\n matData.textures = matData.textures.filter(t => {\r\n if (omit[t.channel]) {\r\n if (omit[t.channel] === true) {\r\n return false;\r\n }\r\n if (Array.isArray(omit[t.channel]) && (omit[t.channel] as number[]).some(i => i === index)) {\r\n return false;\r\n }\r\n }\r\n return true;\r\n });\r\n }\r\n // Metallic is multiplied against the texture, so if the material has a metallic texture then set metallic to 1.\r\n if (matData.textures.findIndex(t => t.channel === eTextureChannel.metallic) > -1) {\r\n matData.metallic = 1;\r\n }\r\n\r\n if (entry.props) {\r\n for (const p in entry.props) {\r\n (mat.data as any)[p] = entry.props[p];\r\n }\r\n }\r\n LibraryMaterials.set(mat.guid as string, mat);\r\n\r\n let categoryList = LibraryMaterialsByCategory.get(tag);\r\n if (!categoryList) {\r\n categoryList = [];\r\n }\r\n categoryList.push(mat);\r\n LibraryMaterialsByCategory.set(tag, categoryList);\r\n }\r\n }\r\n}\r\n\r\nconst materialDefinitionSet: IMaterialDefinitionSet = {\r\n asphalt: {\r\n label: 'Asphalt',\r\n count: 15,\r\n omit: {\r\n [eTextureChannel.metallic]: true,\r\n [eTextureChannel.ambientOcclusion]: [1, 2, 9, 10, 11, 12, 13, 14, 15],\r\n },\r\n },\r\n bark: {\r\n label: 'Bark',\r\n count: 6,\r\n omit: {\r\n [eTextureChannel.metallic]: true,\r\n [eTextureChannel.ambientOcclusion]: [1],\r\n },\r\n },\r\n bricks: {\r\n label: 'Bricks',\r\n count: 17,\r\n omit: {\r\n [eTextureChannel.metallic]: true,\r\n [eTextureChannel.ambientOcclusion]: [1, 2, 3, 4],\r\n },\r\n },\r\n chipboard: {\r\n label: 'Chipboard',\r\n count: 3,\r\n omit: {\r\n [eTextureChannel.metallic]: true,\r\n },\r\n },\r\n concrete: {\r\n label: 'Concrete',\r\n count: 17,\r\n omit: {\r\n [eTextureChannel.metallic]: true,\r\n [eTextureChannel.ambientOcclusion]: [1, 2, 12, 14, 15, 16, 17],\r\n [eTextureChannel.roughness]: [1],\r\n },\r\n },\r\n corrugatedsteel: {\r\n label: 'Corrugated Steel',\r\n count: 5,\r\n },\r\n diamondplate: {\r\n label: 'Diamond Plate',\r\n count: 4,\r\n omit: {\r\n [eTextureChannel.ambientOcclusion]: true,\r\n [eTextureChannel.albedo]: [4],\r\n },\r\n },\r\n fabric: {\r\n label: 'Fabric',\r\n count: 29,\r\n omit: {\r\n [eTextureChannel.ambientOcclusion]: true,\r\n [eTextureChannel.metallic]: true,\r\n },\r\n },\r\n gravel: {\r\n label: 'Gravel',\r\n count: 8,\r\n omit: {\r\n [eTextureChannel.metallic]: true,\r\n [eTextureChannel.ambientOcclusion]: [1, 2],\r\n },\r\n },\r\n ground: {\r\n label: 'Ground',\r\n count: 28,\r\n omit: {\r\n [eTextureChannel.metallic]: true,\r\n [eTextureChannel.ambientOcclusion]: [1, 2, 3, 17],\r\n },\r\n },\r\n ice: {\r\n label: 'Ice',\r\n count: 4,\r\n omit: {\r\n [eTextureChannel.metallic]: true,\r\n [eTextureChannel.ambientOcclusion]: true,\r\n },\r\n },\r\n leather: {\r\n label: 'Leather',\r\n count: 14,\r\n omit: {\r\n [eTextureChannel.metallic]: true,\r\n [eTextureChannel.ambientOcclusion]: [1, 2, 3, 4, 5, 6, 7, 9, 10, 11, 12, 13, 14],\r\n },\r\n },\r\n marble: {\r\n label: 'Marble',\r\n count: 5,\r\n omit: {\r\n [eTextureChannel.metallic]: true,\r\n [eTextureChannel.ambientOcclusion]: true,\r\n },\r\n },\r\n metal: {\r\n label: 'Metal',\r\n count: 27,\r\n omit: {\r\n [eTextureChannel.ambientOcclusion]: true,\r\n },\r\n },\r\n metalplates: {\r\n label: 'Metal Plate',\r\n count: 3,\r\n omit: {\r\n [eTextureChannel.ambientOcclusion]: true,\r\n },\r\n },\r\n paint: {\r\n label: 'Paint',\r\n count: 6,\r\n omit: {\r\n [eTextureChannel.metallic]: true,\r\n },\r\n },\r\n paintedmetal: {\r\n label: 'Painted Metal',\r\n count: 5,\r\n omit: {\r\n [eTextureChannel.ambientOcclusion]: true,\r\n },\r\n },\r\n paintedplaster: {\r\n label: 'Painted Plaster',\r\n count: 5,\r\n omit: {\r\n [eTextureChannel.ambientOcclusion]: true,\r\n [eTextureChannel.metallic]: true,\r\n },\r\n },\r\n pavingstones: {\r\n label: 'Paving Stones',\r\n count: 41,\r\n omit: {\r\n [eTextureChannel.metallic]: true,\r\n [eTextureChannel.ambientOcclusion]: [1, 3, 4],\r\n },\r\n },\r\n pipe: {\r\n label: 'Pipe',\r\n count: 4,\r\n omit: {\r\n [eTextureChannel.ambientOcclusion]: true,\r\n },\r\n },\r\n planks: {\r\n label: 'Planks',\r\n count: 9,\r\n omit: {\r\n [eTextureChannel.ambientOcclusion]: [1, 2],\r\n },\r\n },\r\n plastic: {\r\n label: 'Plastic',\r\n count: 5,\r\n omit: {\r\n [eTextureChannel.metallic]: true,\r\n [eTextureChannel.ambientOcclusion]: true,\r\n },\r\n },\r\n rock: {\r\n label: 'Rock',\r\n count: 14,\r\n omit: {\r\n [eTextureChannel.metallic]: true,\r\n [eTextureChannel.ambientOcclusion]: [1, 6, 7, 8, 9],\r\n [eTextureChannel.albedo]: [4],\r\n },\r\n },\r\n rocks: {\r\n label: 'Rocks',\r\n count: 6,\r\n omit: {\r\n [eTextureChannel.metallic]: true,\r\n },\r\n },\r\n roofingtiles: {\r\n label: 'Roofing Tiles',\r\n count: 5,\r\n omit: {\r\n [eTextureChannel.metallic]: true,\r\n [eTextureChannel.ambientOcclusion]: true,\r\n },\r\n },\r\n rust: {\r\n label: 'Rust',\r\n count: 5,\r\n omit: {\r\n [eTextureChannel.ambientOcclusion]: true,\r\n },\r\n },\r\n scratches: {\r\n label: 'Scratches',\r\n count: 5,\r\n omit: {\r\n [eTextureChannel.ambientOcclusion]: true,\r\n [eTextureChannel.metallic]: true,\r\n [eTextureChannel.albedo]: true,\r\n },\r\n },\r\n terrazzo: {\r\n label: 'Terrazzo',\r\n count: 8,\r\n omit: {\r\n [eTextureChannel.ambientOcclusion]: true,\r\n [eTextureChannel.metallic]: true,\r\n },\r\n },\r\n tiles: {\r\n label: 'Tiles',\r\n count: 37,\r\n omit: {\r\n [eTextureChannel.metallic]: true,\r\n [eTextureChannel.ambientOcclusion]: [\r\n 1, 2, 3, 4, 5, 12, 13, 14, 15, 20, 21, 22, 23, 24, 25, 29, 30, 31, 32, 33, 34, 35, 36, 37,\r\n ],\r\n },\r\n },\r\n wood: {\r\n label: 'Wood',\r\n count: 20,\r\n omit: {\r\n [eTextureChannel.metallic]: true,\r\n [eTextureChannel.ambientOcclusion]: [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 12, 13, 14, 15, 16, 17, 18, 19, 20],\r\n },\r\n },\r\n woodfloor: {\r\n label: 'Wood Floor',\r\n count: 28,\r\n omit: {\r\n [eTextureChannel.metallic]: true,\r\n [eTextureChannel.ambientOcclusion]: [1, 2, 4, 5, 6, 8],\r\n },\r\n },\r\n woodsiding: {\r\n label: 'Wood Siding',\r\n count: 8,\r\n omit: {\r\n [eTextureChannel.metallic]: [5, 6, 7],\r\n },\r\n },\r\n};\r\n\r\nfillLibraryMaterialsMap();\r\n\r\n/** create library environments */\r\nfor (const env of [\r\n // \"country\",\r\n // \"environment\",\r\n // \"forest\",\r\n // \"night\",\r\n // \"parking\",\r\n // \"room\",\r\n\r\n 'adams_place_bridge',\r\n 'aerodynamics_workshop',\r\n 'autoshop',\r\n 'ballroom',\r\n 'boiler_room',\r\n 'cape_hill',\r\n 'colorful_studio',\r\n 'fireplace',\r\n 'greenwich_park',\r\n 'hospital_room',\r\n 'industrial_pipe_and_valve',\r\n 'leadenhall_market',\r\n 'machine_shop',\r\n 'modern_buildings_night',\r\n 'museum_of_ethnography',\r\n 'mutianyu',\r\n 'old_bus_depot',\r\n 'photo_studio',\r\n 'preller_drive',\r\n 'red_wall',\r\n 'rooitou_park',\r\n 'royal_esplanade',\r\n 'schadowplatz',\r\n 'small_cathedral',\r\n 'spruit_sunrise',\r\n 'studio_small',\r\n 'surgery',\r\n 'venice_sunset',\r\n 'winter_evening',\r\n 'wooden_lounge',\r\n]) {\r\n const label = env\r\n .split('_')\r\n .map(s => s.capitalize())\r\n .join(' ');\r\n const o = {\r\n name: env,\r\n label,\r\n path: `${LOCAL_ENVIRONMENT_DOMAIN}${env}`,\r\n previewUrl: `environments/${env}.jpg`,\r\n } as IEnvironmentInfo;\r\n LibraryEnvironments.push(o);\r\n LibraryEnvironmentsMap.set(o.path, o);\r\n}\r\n","export const MOBILE_WIDTH = 847;\r\n\r\nexport function isMobile() {\r\n return window != null && window.innerWidth < MOBILE_WIDTH;\r\n}\r\n","import { KbGeometry } from '@models/classes';\r\nimport { Token } from '@models/classes/token';\r\nimport { MeshNode } from '../classes/meshes/mesh-node';\r\nimport { isCustomMeshNode } from '../classes/meshes/mesh-utilities';\r\nimport { HttpService } from './http-service';\r\nimport { attachLoadingTimer } from './loading-timer-helper';\r\n\r\nconst geometryCache = new Map>();\r\n\r\nexport class MeshService {\r\n constructor(private httpService: HttpService, private assetRoot?: string) {}\r\n /**\r\n * Loads the geometry for a mesh node and adds it to the node, mutating the node object.\r\n * @param meshNode Mesh node to load geometry for\r\n */\r\n public hydrateGeometry(meshNode: T): Promise {\r\n if (!isCustomMeshNode(meshNode) || !meshNode.geometryId) {\r\n return Promise.resolve(meshNode);\r\n } else {\r\n let request = geometryCache.get(meshNode.geometryId);\r\n let shared = false;\r\n\r\n if (!request) {\r\n request = attachLoadingTimer(\r\n this.httpService.getBinary(`${this.assetRoot}/meshes/${meshNode.geometryId}`, 'arraybuffer', true)\r\n ).then(\r\n response => {\r\n if (response && response.data) {\r\n const geometry = meshNode._manager!.create(Token.KbGeometry);\r\n geometry.extractFromBinary(response.data, meshNode);\r\n return geometry;\r\n }\r\n return Promise.reject();\r\n },\r\n err => {\r\n console.error(`Geometry for \"${meshNode.name}\" could not be loaded`, err);\r\n return null;\r\n }\r\n );\r\n geometryCache.set(meshNode.geometryId, request);\r\n } else {\r\n shared = true;\r\n }\r\n\r\n return request.then(geometry => {\r\n if (geometry) {\r\n if (shared) {\r\n geometry.shared = true;\r\n }\r\n\r\n meshNode._geometry = geometry;\r\n }\r\n return meshNode;\r\n });\r\n }\r\n }\r\n}\r\n","import { BaseKb3dModel } from '../decorators/kb3d-decorator';\r\nimport { instanceOfAny } from '../decorators/model-utils';\r\nimport { ILightNode } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { SpaceNode } from '../meshes';\r\nimport { KbColor } from '../primitives/kb-color';\r\nimport { Token } from '../token';\r\n\r\n@BaseKb3dModel({\r\n token: Token.LightNode,\r\n})\r\nexport abstract class LightNode extends SpaceNode implements ILightNode {\r\n constructor() {\r\n super();\r\n }\r\n public visible: boolean;\r\n public intensity: number;\r\n public color: KbColor;\r\n\r\n getSpaceChildren() {\r\n return [];\r\n }\r\n}\r\n\r\nexport function isLightNode(node: Kb3dObject): node is LightNode {\r\n return instanceOfAny(node, ...Token.LightTokens);\r\n}\r\n","import { SketchControlPoint } from '.';\r\nimport { AnnotationNode } from './annotations';\r\nimport { IgnoreChanges } from './decorators/change-tracker';\r\nimport { Enumerable } from './decorators/enumerable';\r\nimport { Kb3dModel } from './decorators/kb3d-decorator';\r\nimport { instanceOf } from './decorators/model-utils';\r\nimport { eAxis, eBackground, eLoadingExperience, eRenderMode, eTextureSamplingMode } from './enums';\r\nimport { Connector } from './features/connector';\r\nimport { IImportInfo, ISceneNode } from './kb3d-models';\r\nimport { Kb3dObject } from './kb3d-object';\r\nimport { LightNode } from './lights';\r\nimport { MaterialNode } from './materials/material-node';\r\nimport { MateNode } from './mates/mate-node';\r\nimport { MeshGroupNode } from './meshes/mesh-group-node';\r\nimport { SpaceNode } from './meshes/space-node';\r\nimport { KbColor } from './primitives';\r\nimport { Token } from './token';\r\nimport { ViewpointNode } from './viewpoints/viewpoint-node';\r\n\r\n@Kb3dModel({\r\n token: Token.SceneNode,\r\n})\r\nexport class SceneNode extends MeshGroupNode implements ISceneNode {\r\n constructor() {\r\n super();\r\n }\r\n\r\n public nestedId: string;\r\n public background: eBackground;\r\n public environment: string;\r\n public environmentBlur: number;\r\n public environmentIntensity: number;\r\n public backgroundColor: KbColor;\r\n public ambientColor: KbColor;\r\n public renderMode: eRenderMode;\r\n public edgeColor: KbColor;\r\n public edgeWidth: number;\r\n public edgeDisplayTolerance: number;\r\n public viewpointId?: string;\r\n public frameViewpoint: boolean;\r\n public progressiveTextureLoad: boolean;\r\n public enableHoverEffects: boolean;\r\n public enableSceneOptimization: boolean;\r\n public enableDeferredSceneUpdates: boolean;\r\n public enableGrid?: boolean;\r\n public enableGridAxes?: boolean;\r\n public gridMajorInterval?: number;\r\n public gridNormal: eAxis;\r\n public gridSize?: number;\r\n public gridStepSize?: number;\r\n public connectorSize?: number;\r\n public arEnabled?: boolean;\r\n public fullscreenEnabled?: boolean;\r\n public xrAllowScaling: boolean;\r\n public unitsPerMeter: number;\r\n public xrAddShadow: boolean;\r\n public xrLightingEstimation: boolean;\r\n public enableFxaaPostProcess?: boolean;\r\n public enableLensEffectsPostProcess?: boolean;\r\n public enableSSAOPostProcess?: boolean;\r\n public environmentRotation: number;\r\n public dimensionUnitScalar?: number;\r\n public dimensionUnitSymbol?: string;\r\n public loadingExperience: eLoadingExperience;\r\n public loadingImagePath: string;\r\n public textureSamplingMode: eTextureSamplingMode;\r\n\r\n public _forceGrid = false;\r\n\r\n /** override base parent for more specific typing*/\r\n public _parent: SceneNode | undefined;\r\n\r\n @IgnoreChanges\r\n public conversionMap?: Record;\r\n\r\n @IgnoreChanges\r\n public _configurator: any; //filled in by kbmax for easy retrieval of the associated configurator\r\n\r\n private _materials: MaterialNode[];\r\n @Enumerable()\r\n public get materials(): MaterialNode[] {\r\n if (!this._materials) this._materials = [];\r\n return this._materials;\r\n }\r\n public set materials(val) {\r\n this._materials = val;\r\n }\r\n\r\n private _lights: LightNode[];\r\n @Enumerable()\r\n public get lights(): LightNode[] {\r\n if (!this._lights) this._lights = [];\r\n return this._lights;\r\n }\r\n public set lights(val) {\r\n this._lights = val;\r\n }\r\n\r\n private _viewpoints: ViewpointNode[];\r\n @Enumerable()\r\n public get viewpoints(): ViewpointNode[] {\r\n if (!this._viewpoints) this._viewpoints = [];\r\n return this._viewpoints;\r\n }\r\n public set viewpoints(val) {\r\n this._viewpoints = val;\r\n }\r\n\r\n private _scenes: SceneNode[];\r\n @Enumerable()\r\n public get scenes(): SceneNode[] {\r\n if (!this._scenes) this._scenes = [];\r\n return this._scenes;\r\n }\r\n public set scenes(val) {\r\n this._scenes = val;\r\n }\r\n\r\n private _mates: MateNode[];\r\n @Enumerable()\r\n public get mates(): MateNode[] {\r\n if (!this._mates) this._mates = [];\r\n return this._mates;\r\n }\r\n public set mates(val) {\r\n this._mates = val;\r\n }\r\n\r\n private _annotations: AnnotationNode[];\r\n @Enumerable()\r\n public get annotations(): AnnotationNode[] {\r\n if (!this._annotations) this._annotations = [];\r\n return this._annotations;\r\n }\r\n public set annotations(val) {\r\n this._annotations = val;\r\n }\r\n\r\n private _importHistory: IImportInfo[];\r\n @Enumerable()\r\n public get importHistory(): IImportInfo[] {\r\n if (!this._importHistory) this._importHistory = [];\r\n return this._importHistory;\r\n }\r\n public set importHistory(val) {\r\n this._importHistory = val;\r\n }\r\n\r\n private _exportHistory: IImportInfo[];\r\n @Enumerable()\r\n public get exportHistory(): IImportInfo[] {\r\n if (!this._exportHistory) this._exportHistory = [];\r\n return this._importHistory;\r\n }\r\n public set exportHistory(val) {\r\n this._exportHistory = val;\r\n }\r\n\r\n public getChildren() {\r\n return [\r\n ...this.meshes,\r\n ...this.materials,\r\n ...this.lights,\r\n ...this.viewpoints,\r\n ...this.mates,\r\n ...this.connectors,\r\n ...this.annotations,\r\n ];\r\n }\r\n\r\n public getChildArrayForType(token: Token): Kb3dObject[] {\r\n if (token.isMeshOrGroup() || token.isLight()) {\r\n return this.meshes;\r\n } else if (token.isMaterial()) {\r\n return this.materials;\r\n } else if (token.isViewpoint()) {\r\n return this.viewpoints;\r\n } else if (token.isMate()) {\r\n return this.mates;\r\n } else if (token.isConnector()) {\r\n return this.connectors;\r\n } else if (token.isAnnotation()) {\r\n return this.annotations;\r\n } else {\r\n throw 'unexpected token';\r\n }\r\n }\r\n\r\n public getChildArray(node: Kb3dObject) {\r\n if (Token.withKey(node.$type).isLight()) {\r\n // Lights can be defined as meshes but can be in the light array as well, so we have to check there\r\n if (this.lights.contains(node as LightNode)) {\r\n return this.lights;\r\n }\r\n }\r\n return super.getChildArray(node);\r\n }\r\n\r\n public isNested() {\r\n return this._parent != null;\r\n }\r\n\r\n public runOnAllScenes(action: (sceneNode: SceneNode) => void) {\r\n action(this);\r\n for (const nested of this.scenes) {\r\n nested.runOnAllScenes(action);\r\n }\r\n }\r\n\r\n public getSpaceChildren(): SpaceNode[] {\r\n return this.meshes.concat(this.scenes);\r\n }\r\n\r\n public getNestedPathTo(node: Kb3dObject) {\r\n let path = node.id;\r\n let parent = node._parentScene;\r\n let isPath = false;\r\n while (parent) {\r\n if (parent === this) break;\r\n path = parent.nestedId + '/' + path;\r\n isPath = true;\r\n parent = parent._parent;\r\n }\r\n return `${isPath ? '@' : ''}${path}`;\r\n }\r\n\r\n public getAllConnectors() {\r\n const connectors = new Array();\r\n this.runOnAllScenes(s => {\r\n connectors.push(...s._manager!.getByToken(Token.Connector).filter(c => c.enabled));\r\n });\r\n return connectors;\r\n }\r\n\r\n public getAllSketchControlPoints() {\r\n const points = new Array();\r\n this.runOnAllScenes(s => {\r\n points.push(...s._manager!.getByToken(Token.SketchControlPoint).filter(c => c.enabled));\r\n });\r\n return points;\r\n }\r\n}\r\n\r\nexport function isSceneNode(node: Kb3dObject): node is SceneNode {\r\n return instanceOf(node, Token.SceneNode);\r\n}\r\n","import { isTrackedClass } from '@models/interfaces/tracked-class';\r\nimport { FontService } from '@models/services/font-service';\r\nimport { HttpService, IHttpEnvironment } from '@models/services/http-service';\r\nimport { MaterialService } from '@models/services/material-service';\r\nimport { MeshService } from '@models/services/mesh-service';\r\nimport { getConstructorParamsForModel } from './decorators/inject-decorator';\r\nimport { IDecoratedModel } from './decorators/kb3d-decorator';\r\nimport { Constructor, IKb3dModel, Kb3dModels, instanceOf, instanceOfAny } from './decorators/model-utils';\r\nimport { defaults } from './default-values';\r\nimport {\r\n HandlerMap,\r\n IDraggable,\r\n Kb3dBoundingBoxChangeHandler,\r\n Kb3dPointerHandler,\r\n SetMap,\r\n TagArrayMap,\r\n} from './handlers';\r\nimport { IKb3dObject, ISceneInfo } from './kb3d-models';\r\nimport { Kb3dObject } from './kb3d-object';\r\nimport { LightNode, isLightNode } from './lights/light-node';\r\nimport { MaterialNode, isMaterialNode } from './materials/material-node';\r\nimport { MeshGroupNode } from './meshes/mesh-group-node';\r\nimport { isMeshNode } from './meshes/mesh-utilities';\r\nimport { SpaceNode } from './meshes/space-node';\r\nimport { SceneNode } from './scene-node';\r\nimport { Token } from './token';\r\nimport { isGuid, isObject } from './utilities';\r\n\r\nconst SET_TYPE = '__set';\r\n\r\n/** contains all kb3dobjects keyed by their dynamic Id, to fetch nodes from the 3d engine ID */\r\nconst dynamicDb: Map = new Map();\r\n\r\nexport function getByDynamicId(dynamicId: string, throwOnNotFound: true): T;\r\nexport function getByDynamicId(dynamicId: string, throwOnNotFound?: boolean): T | undefined;\r\nexport function getByDynamicId(dynamicId: string, throwOnNotFound = false): T | undefined {\r\n if (throwOnNotFound && !dynamicDb.has(dynamicId)) {\r\n throw `kb3d object with id ${dynamicId} not found`;\r\n }\r\n return dynamicDb.get(dynamicId) as T | undefined;\r\n}\r\n\r\nexport interface ISerializeOptions {\r\n /** whether to include children in the serialization. Defaults to true */\r\n includeChildNodes?: boolean;\r\n /** whether to omit properties that are the same as the default value of that property. Defaults to true */\r\n omitDefaultValues?: boolean;\r\n /** internal use */\r\n isRoot?: boolean;\r\n}\r\n\r\nexport interface IJsonSerializeOptions extends ISerializeOptions {\r\n prettyPrint?: boolean;\r\n}\r\n\r\nexport interface IDeserializeOptions {\r\n /** whether to create children while deserializing. Defaults to true */\r\n includeChildNodes?: boolean;\r\n /** when deserializing, set to true to issue new id's to all the kb3dobjects. Defaults to false */\r\n issueNewIds?: boolean;\r\n /** when copying properties from one node to another, we don't want to copy certain properties */\r\n copyingToDifferentNode?: boolean;\r\n}\r\n\r\nexport enum ImportOperation {\r\n ignore,\r\n add,\r\n replace,\r\n}\r\n\r\nexport interface IImportPreview {\r\n type: Token;\r\n node: Partial;\r\n namePath: string;\r\n matchingNode: Kb3dObject;\r\n operation: ImportOperation;\r\n children: IImportPreview[];\r\n parent?: IImportPreview;\r\n}\r\n\r\nexport interface IKb3dManagerOptions {\r\n services?: Map;\r\n /** the root domain that holds assets like meshes and textures. Leave empty to call for them on the host server */\r\n assetRoot?: string;\r\n forceFullSizeTextures?: boolean;\r\n environment?: IHttpEnvironment;\r\n}\r\n\r\nexport class Kb3dManager {\r\n services: Map;\r\n isDevEnvironment = true;\r\n\r\n constructor(options?: IKb3dManagerOptions) {\r\n //if no services are provided, then use the default services\r\n if (options && options.services) {\r\n this.services = options.services;\r\n } else {\r\n const httpService = new HttpService(options?.environment);\r\n if (options?.environment && options.environment.lastDeploy) {\r\n this.isDevEnvironment = false;\r\n }\r\n this.services = new Map([\r\n [Token.HttpService, httpService],\r\n [\r\n Token.MaterialService,\r\n new MaterialService(httpService, options?.assetRoot, options?.forceFullSizeTextures),\r\n ],\r\n [Token.MeshService, new MeshService(httpService, options?.assetRoot)],\r\n [Token.FontService, new FontService(httpService, options?.assetRoot)],\r\n ]);\r\n }\r\n }\r\n\r\n /** contains all kb3dobjects */\r\n private _db = new Map();\r\n /** contains all objects keyed by their token */\r\n private _tokenDb = new SetMap();\r\n /** contains all nodes (but not other kinds of objects) */\r\n private _nodeDb = new Map();\r\n /** holds a cache of name searches that have been performed with their results */\r\n private _nameSearchCache = new Map();\r\n /** contains nested scenes keyed by their nestedId */\r\n private _nestedSceneDb = new Map();\r\n\r\n /** holds a cache of click handlers that have been added to nodes in the context of this manager */\r\n public readonly clickHandlers = new HandlerMap(this);\r\n public readonly doubleClickHandlers = new HandlerMap(this);\r\n public readonly draggables = new HandlerMap(this);\r\n public readonly hotspotClickHandlers = new TagArrayMap void>();\r\n public readonly boundingboxChangeHandlers = new HandlerMap(this);\r\n\r\n public getService(serviceToken: Token): TService {\r\n return this.services.get(serviceToken);\r\n }\r\n\r\n public dispose() {\r\n /**only clear items out of the dynamic db that were part of this manager (as it is static and holds dynamic ids for all managers) */\r\n for (const [, value] of this._db) {\r\n dynamicDb.delete(value._dynamicId);\r\n }\r\n this._db.clear();\r\n this._nodeDb.clear();\r\n this.clickHandlers.clear();\r\n this.doubleClickHandlers.clear();\r\n this.draggables.clear();\r\n this.hotspotClickHandlers.clear();\r\n this.boundingboxChangeHandlers.clear();\r\n this._nameSearchCache.clear();\r\n this._tokenDb.clear();\r\n this._nestedSceneDb.clear();\r\n }\r\n\r\n /**\r\n * Accepts either a node id or a name to search for nodes. If using\r\n * node name wildcards * are allowed. The action provided will be run\r\n * on each of the matching nodes, with the args passed in to that action.\r\n * @param idOrName\r\n * @param action\r\n * @param args\r\n * @param useCache\r\n */\r\n public runOnNodes(\r\n idOrName: string,\r\n action: (node: T, args?: TArgs) => void,\r\n args?: TArgs,\r\n useCache = true\r\n ) {\r\n if (isGuid(idOrName)) {\r\n const node = this.getById(idOrName);\r\n if (node) action(node, args);\r\n } else {\r\n const nodes = this.getNodesByName(idOrName, useCache);\r\n for (const node of nodes) {\r\n action(node as T, args);\r\n }\r\n }\r\n }\r\n // @child1/xxxxx-xxxxxx-xxxx\r\n public getById(id: string): T | undefined {\r\n if (id && id.startsWith('@')) {\r\n //it's a nested path id\r\n const segments = id.split('/');\r\n const nestedId = segments.shift()!.substring(1).toLowerCase();\r\n const nestedScene = this._nestedSceneDb.get(nestedId);\r\n if (nestedScene) {\r\n let subId = segments.join('/');\r\n if (segments.length > 1) subId = '@' + subId;\r\n return nestedScene._manager!.getNodeByIdOrName(subId);\r\n }\r\n } else {\r\n return this._db.get(id) as T | undefined;\r\n }\r\n }\r\n\r\n public getAll() {\r\n return [...this._db.values()];\r\n }\r\n\r\n public getAllNodes() {\r\n return [...this._nodeDb.values()];\r\n }\r\n\r\n /**\r\n * Accepts either a node id or a name to search for nodes. Only returns the first\r\n * matching node. If using node name wildcards * are allowed.\r\n * @param idOrName\r\n * @param useCache\r\n */\r\n public getNodeByIdOrName(idOrName: string | T, useCache = true): T | undefined {\r\n if (typeof idOrName == 'string') {\r\n if (isGuid(idOrName) || idOrName.startsWith('@')) {\r\n return this.getById(idOrName);\r\n } else {\r\n return this.getFirstNodeByName(idOrName, useCache) as any as T;\r\n }\r\n } else if (idOrName instanceof Kb3dObject) {\r\n return idOrName;\r\n }\r\n }\r\n\r\n /**\r\n * Accepts either a node id or a name to search for nodes. If using\r\n * node name wildcards * are allowed.\r\n * @param idOrName\r\n * @param useCache\r\n */\r\n public getNodesByIdOrName(idOrName: string | T, useCache = true): T[] {\r\n const res = [];\r\n\r\n if (typeof idOrName == 'string') {\r\n if (isGuid(idOrName)) {\r\n const n = this.getById(idOrName);\r\n if (n) res.push(n);\r\n } else {\r\n res.push(...this.getNodesByName(idOrName, useCache));\r\n }\r\n } else if (idOrName instanceof Kb3dObject) {\r\n res.push(idOrName);\r\n }\r\n return res as T[];\r\n }\r\n\r\n /**\r\n * Get's the first node that matches the name given. Wildcards * are allowed.\r\n * @param name\r\n * @param useCache\r\n */\r\n public getFirstNodeByName(name: string, useCache = true) {\r\n const nodes = this.getNodesByName(name, useCache);\r\n if (nodes && nodes.length) {\r\n return nodes[0];\r\n }\r\n }\r\n\r\n /**\r\n * Returns a list of nodes that match the name given.\r\n * Wildcards * are allowed.\r\n * Results will be fed from a search cache if useCache is true\r\n * @param name\r\n * @param useCache\r\n */\r\n public getNodesByName(name: string, useCache = true): T[] {\r\n if (useCache && this._nameSearchCache.has(name)) {\r\n return this._nameSearchCache.get(name) as T[];\r\n }\r\n\r\n const results = new Array();\r\n const re = this.getWildcardRegex(name);\r\n\r\n for (const [, node] of this._db) {\r\n if (re.test(node.name)) {\r\n results.push(node as T);\r\n }\r\n }\r\n\r\n this._nameSearchCache.set(name, results);\r\n\r\n return results;\r\n }\r\n\r\n public getNodesContainingName(name: string, useCache = true): T[] {\r\n if (useCache && this._nameSearchCache.has(name)) {\r\n return this._nameSearchCache.get(name) as T[];\r\n }\r\n\r\n const results = new Array();\r\n const re = this.getWildcardRegex(name);\r\n\r\n for (const [, node] of this._db) {\r\n if (re.test(node.name)) {\r\n results.push(node as T);\r\n }\r\n }\r\n\r\n this._nameSearchCache.set(name, results);\r\n\r\n return results;\r\n }\r\n\r\n public getByToken(token: Token) {\r\n return [...this._tokenDb.get(token).values()] as T[];\r\n }\r\n\r\n public getByTokens(...tokens: Token[]) {\r\n return tokens.flatMap(t => [...this._tokenDb.get(t).values()]) as T[];\r\n }\r\n\r\n /** escapes the given string so that any special characters will be processed literally in a regex search */\r\n protected escapeRegex(str: string) {\r\n return str.replace(/([.*+?^=!:${}()|[\\]/\\\\])/g, '\\\\$1');\r\n }\r\n\r\n /**\r\n * Given a string with wildcards *, will provide a regex that can be used to\r\n * match strings.\r\n * @param rule\r\n */\r\n protected getWildcardRegex(rule: string): RegExp {\r\n // \".\" => Find a single character, except newline or line terminator\r\n // \".*\" => Matches any string that contains zero or more characters\r\n rule = rule.split('*').map(this.escapeRegex).join('.*');\r\n\r\n // \"^\" => Matches any string with the following at the beginning of it\r\n // \"$\" => Matches any string with that in front at the end of it\r\n rule = '^' + rule + '$';\r\n\r\n //Create a regular expression object for matching string\r\n return new RegExp(rule, 'i');\r\n }\r\n\r\n /**\r\n * absorbs a kb3d object that already exists into the manager.\r\n * @param o\r\n */\r\n public register(o: T) {\r\n if (o instanceof Kb3dObject) {\r\n if (!this._db.has(o.id)) {\r\n if (isTrackedClass(o)) {\r\n o.isModified = true; //force a full round of change detection if this is a newly registered object\r\n }\r\n }\r\n\r\n this._db.set(o.id, o);\r\n dynamicDb.set(o._dynamicId, o);\r\n this._tokenDb.add(Token.withKey(o.$type), o);\r\n\r\n if (o.isNode()) {\r\n this._nodeDb.set(o.id, o);\r\n }\r\n }\r\n }\r\n\r\n public unregister(o: T) {\r\n if (o instanceof Kb3dObject) {\r\n this.clickHandlers.deleteByKey(o as any as SpaceNode);\r\n this.doubleClickHandlers.deleteByKey(o as any as SpaceNode);\r\n this.draggables.deleteByKey(o as any as SpaceNode);\r\n this.boundingboxChangeHandlers.deleteByKey(o as any as SpaceNode);\r\n this._db.delete(o.id);\r\n dynamicDb.delete(o._dynamicId);\r\n this._tokenDb.removeValue(Token.withKey(o.$type), o);\r\n\r\n if (o.isNode()) {\r\n this._nodeDb.delete(o.id);\r\n }\r\n\r\n if (o instanceof SceneNode && o.isNested()) {\r\n this._nestedSceneDb.delete(o.nestedId);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Creates an instance of a kb3d model, given a token\r\n * and an anonymous object in the shape of the model that\r\n * can be used to set properties on it.\r\n *\r\n * As part of the creation process, we do things in this order:\r\n *\r\n * 1. Create an instance of the class. If a create function was provided\r\n * in the decorator for the class, that is used to create the instance.\r\n * If not, then it's created by calling the constructor of the class\r\n * with no arguments.\r\n * 2. We then apply default values to the properties of the class\r\n * 3. We then apply the args that were passed in to set properties on the class\r\n * 4. We then register the object into the map using it's id so it can be quickly retrieved\r\n */\r\n public create(token: Token, args: Partial = {}, options?: IDeserializeOptions): T {\r\n options = { includeChildNodes: true, ...options };\r\n\r\n const modelDef = Kb3dModels.get(token);\r\n if (!modelDef) {\r\n throw `could not find model with token: ${token.key}`;\r\n }\r\n const constructorParameters = getConstructorParamsForModel(modelDef);\r\n const hydratedParameters: any[] = [];\r\n for (const i of constructorParameters) {\r\n if (!i.ignore) {\r\n const service = this.services?.get(i.token);\r\n if (!service) {\r\n throw `Injected service of type '${i.token.key} could not be found to inject into the constructor of type '${token.key}`;\r\n }\r\n hydratedParameters.push(service);\r\n }\r\n }\r\n\r\n let model: T;\r\n if (modelDef.create) {\r\n model = modelDef.create(args);\r\n } else {\r\n model = new (modelDef.classConstructor as Constructor)(...hydratedParameters);\r\n }\r\n\r\n //now initialize the model for when no create function was given\r\n if (!modelDef.create) {\r\n this.initializeModel(isTrackedClass(model) ? model.rawNode : model, args, modelDef, options);\r\n }\r\n\r\n if (model instanceof SceneNode) {\r\n model._manager = this;\r\n model.calculateParents(); //because calculateParents also registers the item, we must call this after initialization, otherwise it won't have the right id\r\n }\r\n\r\n if (modelDef.afterCreate) {\r\n modelDef.afterCreate(model);\r\n }\r\n\r\n //need to clear the name cache since we just created a new node\r\n //no need to clear if the node is a primitive as they are never fetched by name\r\n if(!token.isPrimitive()) {\r\n this._nameSearchCache.clear();\r\n }\r\n\r\n return model;\r\n }\r\n\r\n public serializeToJson(instance: T, options?: IJsonSerializeOptions): string {\r\n const o = this.serialize(instance, options);\r\n const json = JSON.stringify(o, null, options?.prettyPrint ? 2 : undefined);\r\n return json;\r\n }\r\n\r\n public serialize(instance: T, options?: ISerializeOptions): Partial | undefined {\r\n options = {\r\n includeChildNodes: true,\r\n isRoot: true,\r\n omitDefaultValues: true,\r\n ...options,\r\n };\r\n\r\n let result: any = { $type: instance.$type };\r\n const defaultValues = (defaults as any)[instance.$type];\r\n\r\n for (const key in instance) {\r\n if (\r\n // Never persist properties beginning with an underscore\r\n key.substr(0, 1) !== '_' &&\r\n // If a property is the same as a default value, remove it\r\n (!options.omitDefaultValues ||\r\n !defaultValues ||\r\n !this.shallowEquals(defaultValues[key], (instance as any)[key]))\r\n ) {\r\n if (!isObject(instance[key])) {\r\n // For primitives, just copy the values over\r\n result[key] = instance[key];\r\n } else if (Array.isArray(instance[key])) {\r\n const sourceArray = instance[key] as T[];\r\n if (sourceArray.length && typeof sourceArray[0] !== 'object') {\r\n //it's an array of primitives\r\n result[key] = sourceArray.slice(0); //clone the array of primitives\r\n } else {\r\n result[key] = []; //make sure there is an array on target prop. Recreate it if one is there already.\r\n for (const sourceItem of sourceArray) {\r\n if (\r\n !sourceItem.$type ||\r\n !Token.withKey(sourceItem.$type).isNode() ||\r\n options.includeChildNodes\r\n ) {\r\n // exclude it if it's a kb3d model and includeChildren is false\r\n const targetItem = this.serialize(sourceItem, { ...options, isRoot: false });\r\n if (targetItem) {\r\n result[key].push(targetItem);\r\n }\r\n }\r\n }\r\n }\r\n } else if ((instance[key] as any) instanceof Map) {\r\n console.error('implement serializing maps');\r\n } else if ((instance[key] as any) instanceof Set) {\r\n result[key] = {\r\n $type: SET_TYPE,\r\n values: Array.from((instance[key] as Set).values()),\r\n };\r\n } else {\r\n // For anything else, try deserializing. The deserializer will just copy the property if it is not a serialized class.\r\n result[key] = this.serialize(instance[key], { ...options, isRoot: false });\r\n }\r\n }\r\n }\r\n\r\n if (instance.$type) {\r\n const modelDef = Kb3dModels.get(Token.withKey(instance.$type));\r\n if (modelDef && modelDef.onSerializing) {\r\n const include = modelDef.onSerializing(instance, result);\r\n // Always serialize the object if it is the root object. Otherwise, let the class decide.\r\n if (include === false && !options.isRoot) {\r\n result = undefined;\r\n }\r\n }\r\n }\r\n return result;\r\n }\r\n\r\n /**\r\n * See notes for the create() function as this is virtually the same.\r\n * The only difference is that the type here is discovered through the $type\r\n * property of the args object instead of an explicit token.\r\n * @param object\r\n */\r\n public deserialize(object: any, options?: IDeserializeOptions): T {\r\n if (!object || !object.$type) {\r\n return object;\r\n } else if (object.$type === SET_TYPE) {\r\n return new Set(object.values) as any;\r\n }\r\n return this.create(Token.withKey(object.$type), object, options);\r\n }\r\n\r\n public deserializeJson(json: string): T {\r\n const object = JSON.parse(json);\r\n return this.deserialize(object);\r\n }\r\n\r\n public clone(objectToClone: T): T {\r\n //make sure it's a kb3d model\r\n if (!objectToClone) throw 'no object provided to clone';\r\n if (!('$type' in objectToClone)) {\r\n throw 'object passed to clone function must be a kb3d model and have a $type defined';\r\n }\r\n const json = this.serializeToJson(objectToClone);\r\n const object = JSON.parse(json) as Kb3dObject;\r\n\r\n return this.deserialize(object, { issueNewIds: true });\r\n }\r\n\r\n public nestScene(nestedId: string, parent: SceneNode, serializedNested: Partial = {}) {\r\n const nestedMan = new Kb3dManager({ services: this.services });\r\n if (serializedNested.lights) {\r\n serializedNested.lights.clear();\r\n }\r\n const nested = nestedMan.create(Token.SceneNode, serializedNested);\r\n nested.nestedId = nestedId.replace('/', '');\r\n parent.scenes.push(nested);\r\n this._nestedSceneDb.set(nested.nestedId.toLowerCase(), nested);\r\n\r\n return nested;\r\n }\r\n\r\n protected getNodeNamePathMap() {\r\n const meshMap = new Map();\r\n const matMap = new Map();\r\n const lightMap = new Map();\r\n const allNodes = this._nodeDb.values();\r\n for (const n of allNodes) {\r\n if (isLightNode(n)) {\r\n lightMap.set(n.getNamePath(), n);\r\n } else if (isMaterialNode(n)) {\r\n matMap.set(n.getNamePath(), n);\r\n } else {\r\n meshMap.set(n.getNamePath(), n);\r\n }\r\n }\r\n return {\r\n meshes: meshMap,\r\n lights: lightMap,\r\n materials: matMap,\r\n };\r\n }\r\n\r\n public previewImport(parent: SceneNode | MeshGroupNode, imported: ISceneInfo) {\r\n const sceneNode = parent._parentScene;\r\n if (!sceneNode || !imported.scene) return;\r\n\r\n const nMap = this.getNodeNamePathMap();\r\n\r\n const processNode = (node: Partial, parentNamePath: string) => {\r\n const token = Token.withKey((node as any)['$type']);\r\n const namePath =\r\n token !== Token.SceneNode && node.name\r\n ? [parentNamePath, node.name].filter(Boolean).join('/')\r\n : parentNamePath;\r\n\r\n let matchingNode: Kb3dObject | undefined;\r\n if (token.isLight()) {\r\n matchingNode = nMap.lights.get(namePath);\r\n } else if (token.isMaterial()) {\r\n matchingNode = nMap.materials.get(namePath);\r\n } else {\r\n matchingNode = nMap.meshes.get(namePath);\r\n }\r\n\r\n const p = {\r\n node,\r\n type: Token.withKey((node as any)['$type']),\r\n namePath,\r\n matchingNode,\r\n operation: matchingNode ? ImportOperation.replace : ImportOperation.add,\r\n } as IImportPreview;\r\n p.children = ['meshes', 'materials', 'lights', 'viewpoints'].flatMap(childProp => {\r\n const childArr = (node as any)[childProp] as Kb3dObject[];\r\n return childArr\r\n ? childArr.map(c => {\r\n const cNode = processNode(c, childProp === 'meshes' ? namePath : '');\r\n cNode.parent = p;\r\n return cNode;\r\n })\r\n : [];\r\n });\r\n return p;\r\n };\r\n\r\n const pathToImportParent = parent.getNamePath();\r\n (imported.scene as any)['$type'] = 'SceneNode';\r\n return processNode(imported.scene, pathToImportParent).children;\r\n }\r\n\r\n public import(previews: IImportPreview[], parentNode: SpaceNode | MaterialNode, root = true) {\r\n const nMap = this.getNodeNamePathMap();\r\n\r\n // Materials have to come first so we know how they reconnect to the meshes to them\r\n previews.sort((a, b) => {\r\n if (a.type.isMaterial() && !b.type.isMaterial()) {\r\n return -1;\r\n } else {\r\n return 0;\r\n }\r\n });\r\n\r\n for (const p of previews) {\r\n if (p.operation !== ImportOperation.ignore) {\r\n let thisNode: Kb3dObject;\r\n\r\n if (p.operation === ImportOperation.add || !p.matchingNode) {\r\n const parentPath = p.namePath.substring(0, p.namePath.lastIndexOf('/'));\r\n\r\n let parent: Kb3dObject | undefined;\r\n if (!parentPath) {\r\n parent = parentNode;\r\n } else if (p.type.isLight()) {\r\n parent = nMap.lights.get(parentPath);\r\n } else if (p.type.isMaterial()) {\r\n parent = nMap.materials.get(parentPath);\r\n } else {\r\n parent = nMap.meshes.get(parentPath);\r\n }\r\n\r\n const newNode = this.deserialize(p.node, { includeChildNodes: false });\r\n if (\r\n instanceOfAny(newNode, ...Token.GlobalLightTokens) &&\r\n instanceOf(parentNode, Token.SceneNode)\r\n ) {\r\n parentNode.lights.push(newNode);\r\n } else if (parent?.id === parentNode.id) {\r\n parentNode?.addChildren(newNode);\r\n }\r\n thisNode = newNode;\r\n } else {\r\n // replace\r\n const originalId = p.matchingNode.id;\r\n const matchingMeshNode = isMeshNode(p.matchingNode) ? p.matchingNode : undefined;\r\n const originalMaterialId = matchingMeshNode ? matchingMeshNode.materialId : undefined;\r\n this.assign(p.matchingNode, p.node, { includeChildNodes: false });\r\n\r\n p.matchingNode.id = originalId;\r\n // If the new materialId from the import exists, that means the material was imported and added and can be used.\r\n // If the new materialId from the import does not exist, use the old materialId because the material was updated but\r\n // kept the original ID.\r\n if (matchingMeshNode && originalMaterialId && matchingMeshNode.materialId) {\r\n if (!this.getById(matchingMeshNode.materialId)) {\r\n matchingMeshNode.materialId = originalMaterialId;\r\n }\r\n }\r\n // Trigger change detection on this existing node\r\n if (isTrackedClass(p.matchingNode)) {\r\n p.matchingNode.isModified = true;\r\n }\r\n thisNode = p.matchingNode;\r\n }\r\n\r\n if (thisNode instanceof SpaceNode || thisNode instanceof MaterialNode) {\r\n this.import(p.children, thisNode, false);\r\n } else if (p.children && p.children.length > 0) {\r\n throw Error('Children on non-space node or non-material-node, how did this happen');\r\n }\r\n }\r\n }\r\n }\r\n\r\n protected initializeModel(model: T, args: Partial, modelDef: IKb3dModel, options?: IDeserializeOptions) {\r\n this.setModelDefaultValues(model);\r\n this.assign(model, args, options);\r\n }\r\n\r\n protected setModelDefaultValues(o: any) {\r\n const values = (defaults as any)[o.$type];\r\n if (values) {\r\n for (const prop in values) {\r\n if (values[prop] && values[prop].$type) {\r\n o[prop] = this.deserialize(values[prop]);\r\n } else if (isObject(values[prop])) {\r\n o[prop] = { ...values[prop] }; //need to clone the object from default values otherwise the class and defaults will both be pointing to same instance!\r\n } else {\r\n o[prop] = values[prop];\r\n }\r\n }\r\n }\r\n }\r\n\r\n public assign(model: any, args: any, options?: IDeserializeOptions) {\r\n options = {\r\n includeChildNodes: true,\r\n issueNewIds: false,\r\n copyingToDifferentNode: false,\r\n ...options,\r\n };\r\n\r\n if (args) {\r\n if (options.copyingToDifferentNode) {\r\n // If we are using assign to copy properties from one node to another, serialize the data\r\n if (args instanceof Kb3dObject) {\r\n const defaultValues = { $type: args.$type };\r\n this.setModelDefaultValues(defaultValues);\r\n args = { ...defaultValues, ...this.serialize(args) };\r\n }\r\n }\r\n\r\n for (const prop in args) {\r\n // Never persist properties beginning with an underscore\r\n if (prop.substr(0, 1) == '_') continue;\r\n if (options.copyingToDifferentNode) {\r\n if ((prop.isEqual('id') && model.id) || prop.isEqual('$type')) {\r\n continue; // skip id's if copying to an existing node so the node retains its ID\r\n }\r\n }\r\n if (options.issueNewIds && prop.isEqual('id') && args.$type) continue; //skip id's if options specify so it will use the new one it generates during construction\r\n\r\n const sourceProp = args[prop];\r\n\r\n if (typeof model === 'undefined') {\r\n throw Error(); // this should never happen, this means a reference is being lost somewhere\r\n }\r\n\r\n if (prop === 'parent' || sourceProp === model[prop]) {\r\n // Don't break parent relationships\r\n continue; // Don't break parent relationships\r\n }\r\n\r\n if (!isObject(args[prop])) {\r\n // For primitives, just copy the values over\r\n model[prop] = args[prop];\r\n } else if (Array.isArray(sourceProp)) {\r\n // make sure there is an array on target prop. Recreate it if one is there already.\r\n if (model[prop]) {\r\n if (\r\n options.includeChildNodes ||\r\n !(\r\n model[prop].length > 0 &&\r\n model[prop][0].$type &&\r\n Token.withKey(model[prop][0].$type).isNode()\r\n )\r\n ) {\r\n model[prop].clear();\r\n }\r\n } else {\r\n model[prop] = [];\r\n }\r\n\r\n for (const sourceItem of sourceProp) {\r\n let targetItem: any;\r\n if (sourceItem.$type) {\r\n //it's one of our types so deserialize it\r\n if (options.includeChildNodes || !Token.withKey(sourceItem.$type).isNode()) {\r\n targetItem = this.deserialize(sourceItem, options);\r\n model[prop].push(targetItem);\r\n }\r\n } else if (isObject(sourceItem)) {\r\n targetItem = {};\r\n model[prop].push(targetItem);\r\n this.assign(targetItem, sourceItem, options);\r\n } else {\r\n //primitive type\r\n model[prop].push(sourceItem);\r\n }\r\n }\r\n } else {\r\n if (sourceProp.$type) {\r\n //it's one of our types so deserialize it\r\n if (options.includeChildNodes || !Token.withKey(sourceProp.$type).isNode()) {\r\n model[prop] = this.deserialize(sourceProp, options);\r\n }\r\n } else if (isObject(sourceProp)) {\r\n const targetItem: any = {};\r\n model[prop] = targetItem;\r\n this.assign(targetItem, sourceProp, options);\r\n } else {\r\n //primitive type\r\n model[prop] = sourceProp;\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n protected shallowEquals(a: A, b: B) {\r\n if (!isObject(a)) {\r\n return (a as any) === (b as any);\r\n }\r\n\r\n for (const aProp in a) {\r\n if (aProp.substring(0, 1) !== '_') {\r\n if (!(aProp in b)) {\r\n return false;\r\n }\r\n if (a[aProp] !== (b as any)[aProp]) {\r\n return false;\r\n }\r\n }\r\n }\r\n for (const bProp in b) {\r\n if (bProp.substring(0, 1) !== '_') {\r\n if (!(bProp in a)) {\r\n return false;\r\n }\r\n if (b[bProp] !== (a as any)[bProp]) {\r\n return false;\r\n }\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n}\r\n","import { eShadowFilter, eShadowQuality, eShadowResolution } from '../enums';\r\nimport { IShadowLightNode } from '../kb3d-models';\r\nimport { LightNode } from './light-node';\r\n\r\nexport abstract class ShadowLightNode extends LightNode implements IShadowLightNode {\r\n constructor() {\r\n super();\r\n }\r\n\r\n public generatesShadows: boolean;\r\n public shadowQuality: eShadowQuality;\r\n public shadowResolution: eShadowResolution;\r\n public shadowFilter: eShadowFilter;\r\n public processTransparency: boolean;\r\n}\r\n","import { KbColor } from '@models/classes/primitives/kb-color';\r\nimport { isTrackedClass } from '@models/interfaces/tracked-class';\r\nimport { IgnoreHistory } from '../decorators/change-tracker';\r\nimport { BaseKb3dModel } from '../decorators/kb3d-decorator';\r\nimport { instanceOfAny } from '../decorators/model-utils';\r\nimport {\r\n eTextureBlendMode,\r\n eTextureChannel,\r\n eTextureFit,\r\n eTextureHorizontalAlign,\r\n eTextureVerticalAlign,\r\n eTextureWrapAddressMode,\r\n} from '../enums';\r\nimport { FeatureWithTextures } from '../features/feature';\r\nimport { ITextureLayer } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { Token } from '../token';\r\nimport { MaterialMapping } from './material-mapping';\r\nimport { MaterialNode } from './material-node';\r\n\r\n@BaseKb3dModel({\r\n token: Token.TextureLayer,\r\n})\r\nexport abstract class TextureLayer extends MaterialMapping implements ITextureLayer {\r\n channel: eTextureChannel;\r\n blendMode: eTextureBlendMode;\r\n horizontalAlign: eTextureHorizontalAlign;\r\n verticalAlign: eTextureVerticalAlign;\r\n horizontalFit: eTextureFit;\r\n verticalFit: eTextureFit;\r\n strength: number;\r\n colorize?: KbColor;\r\n tile: boolean;\r\n uWrap: eTextureWrapAddressMode;\r\n vWrap: eTextureWrapAddressMode;\r\n maintainAspect: boolean;\r\n _visualize: boolean;\r\n useClippingMask: boolean;\r\n\r\n @IgnoreHistory\r\n _init = false;\r\n\r\n @IgnoreHistory\r\n _external = false;\r\n\r\n enabled: boolean;\r\n _suppressed: boolean;\r\n\r\n /** override base parent for more specific typing*/\r\n public _parent: MaterialNode | FeatureWithTextures | undefined;\r\n\r\n init(parent: MaterialNode | FeatureWithTextures) {\r\n if (!this._init) {\r\n this._parent = parent;\r\n // Normally this function is called when the node is added to an array on the parent, but textures live in defined properties on a material.\r\n // So we have to explicitly call it here, on texture init. The texture won't be doing anything before init anyways.\r\n this.calculateParents();\r\n if (isTrackedClass(this)) {\r\n // Always track changes on initialized textures because they may be attached to global materials\r\n this.trackChanges(true);\r\n this._init = true;\r\n }\r\n }\r\n }\r\n}\r\n\r\nexport function isTextureLayer(node: Kb3dObject): node is TextureLayer {\r\n return instanceOfAny(node, ...Token.TextureLayerTokens);\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { IDirectionalLightNode } from '../kb3d-models';\r\nimport { KbVector } from '../primitives/kb-vector';\r\nimport { Token } from '../token';\r\nimport { ShadowLightNode } from './shadow-light-node';\r\n\r\n@Kb3dModel({\r\n token: Token.DirectionalLightNode,\r\n})\r\nexport class DirectionalLightNode extends ShadowLightNode implements IDirectionalLightNode {\r\n constructor() {\r\n super();\r\n }\r\n\r\n public direction: KbVector;\r\n}\r\n","import { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { IHemisphericLightNode } from '../kb3d-models';\r\nimport { KbColor } from '../primitives/kb-color';\r\nimport { Token } from '../token';\r\nimport { LightNode } from './light-node';\r\n\r\n@Kb3dModel({\r\n token: Token.HemisphericLightNode,\r\n})\r\nexport class HemisphericLightNode extends LightNode implements IHemisphericLightNode {\r\n constructor() {\r\n super();\r\n }\r\n public direction: KbVector;\r\n public groundColor: KbColor;\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { IPointLightNode } from '../kb3d-models';\r\nimport { KbVector } from '../primitives/kb-vector';\r\nimport { Token } from '../token';\r\nimport { ShadowLightNode } from './shadow-light-node';\r\n\r\n@Kb3dModel({\r\n token: Token.PointLightNode,\r\n})\r\nexport class PointLightNode extends ShadowLightNode implements IPointLightNode {\r\n constructor() {\r\n super();\r\n }\r\n\r\n public position: KbVector;\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { ISpotLightNode } from '../kb3d-models';\r\nimport { KbVector } from '../primitives/kb-vector';\r\nimport { Token } from '../token';\r\nimport { DirectionalLightNode } from './directional-light-node';\r\n\r\n@Kb3dModel({\r\n token: Token.SpotLightNode,\r\n})\r\nexport class SpotLightNode extends DirectionalLightNode implements ISpotLightNode {\r\n constructor() {\r\n super();\r\n }\r\n\r\n public position: KbVector;\r\n public coneAngle: number;\r\n}\r\n","import { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { MaterialService } from '@models/services/material-service';\r\nimport { Enumerable, IgnoreChanges } from '../decorators';\r\nimport { Inject } from '../decorators/inject-decorator';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { eAlphaMode, eMaterialLocation, eMaterialWorkflow, eReflectionType } from '../enums';\r\nimport { IAdvancedMaterialNode } from '../kb3d-models';\r\nimport { KbColor } from '../primitives/kb-color';\r\nimport { Token } from '../token';\r\nimport { MaterialNode } from './material-node';\r\nimport { TextureLayer } from './texture-layer';\r\n\r\n@Kb3dModel({\r\n token: Token.AdvancedMaterialNode,\r\n onSerializing: (node, o) => {\r\n if (node.parentMaterialId == null) {\r\n return true;\r\n }\r\n\r\n // remove any textures that are from the parent material\r\n o.textures = o.textures?.filter(texture => !texture._external);\r\n\r\n // remove any properties from the serialization that are inherited from a global or library material\r\n const keys = Object.keys(o);\r\n for (const p of keys) {\r\n if (node._overriddenPropertiesSet && !node._overriddenPropertiesSet.has(p) && p !== 'textures') {\r\n delete (o as any)[p];\r\n }\r\n }\r\n if (node._overriddenPropertiesSet) {\r\n o.overriddenProperties = Array.from(node._overriddenPropertiesSet).map(val => val.toString());\r\n }\r\n\r\n return node.location == eMaterialLocation.local;\r\n },\r\n addedToChildArray: (node, child) => {\r\n if (child instanceof TextureLayer) {\r\n if (node._initialized) {\r\n child.init(node);\r\n }\r\n }\r\n },\r\n afterCreate: node => {\r\n node.overriddenProperties.forEach(val => node._overriddenPropertiesSet.add(val));\r\n\r\n node.calculateParents(); //parents need to be calculated again because the texture children don't exist yet the first time it is run\r\n node.refreshParentMaterial();\r\n },\r\n})\r\nexport class AdvancedMaterialNode extends MaterialNode implements IAdvancedMaterialNode {\r\n constructor(@Inject(Token.MaterialService) _materialService: MaterialService) {\r\n super(_materialService);\r\n\r\n this._initialized = false;\r\n }\r\n\r\n protected _materials: AdvancedMaterialNode[];\r\n @Enumerable()\r\n get materials(): AdvancedMaterialNode[] {\r\n if (!this._materials) this._materials = [];\r\n return this._materials;\r\n }\r\n set materials(val) {\r\n this._materials = val;\r\n }\r\n\r\n @IgnoreChanges\r\n _initialized = false;\r\n\r\n guid: string;\r\n\r\n alpha: number;\r\n alphaMode?: eAlphaMode;\r\n\r\n workflowMode: eMaterialWorkflow;\r\n\r\n /**\r\n * Intensity of the direct lights e.g. the four lights available in your scene.\r\n * This impacts both the direct diffuse and specular highlights.\r\n */\r\n directIntensity: number;\r\n\r\n /**\r\n * Intensity of the emissive part of the material.\r\n * This helps controlling the emissive effect without modifying the emissive color.\r\n */\r\n emissiveIntensity: number;\r\n\r\n /**\r\n * Intensity of the environment e.g. how much the environment will light the object\r\n * either through harmonics for rough material or through the refelction for shiny ones.\r\n */\r\n environmentIntensity: number;\r\n\r\n /**\r\n * This is a special control allowing the reduction of the specular highlights coming from the\r\n * four lights of the scene. Those highlights may not be needed in full environment lighting.\r\n */\r\n specularIntensity: number;\r\n\r\n /**\r\n * AKA Occlusion Texture Intensity in other nomenclature.\r\n */\r\n ambientTextureStrength?: number;\r\n\r\n /**\r\n * Specifies the metallic scalar of the metallic/roughness workflow.\r\n * Can also be used to scale the metalness values of the metallic texture.\r\n */\r\n metallic: number;\r\n\r\n /**\r\n * Specifies the roughness scalar of the metallic/roughness workflow.\r\n * Can also be used to scale the roughness values of the metallic texture.\r\n */\r\n roughness: number;\r\n\r\n /**\r\n * The color of a material in ambient lighting.\r\n */\r\n ambientColor: KbColor;\r\n\r\n /**\r\n * AKA Diffuse Color in other nomenclature.\r\n */\r\n albedoColor: KbColor;\r\n\r\n /**\r\n * AKA Specular Color in other nomenclature.\r\n */\r\n specularColor: KbColor;\r\n\r\n /**\r\n * The color reflected from the material.\r\n */\r\n reflectionColor: KbColor;\r\n\r\n /**\r\n * The type of reflection or refraction being used used (Plane or probe)\r\n */\r\n reflectionType: eReflectionType;\r\n\r\n /**\r\n * AKA Glossiness in other nomenclature.\r\n */\r\n microSurface: number;\r\n\r\n /**\r\n * Specifies that the material will keep the specular highlights over a transparent surface (only the most limunous ones).\r\n * A car glass is a good exemple of that. When sun reflects on it you can not see what is behind.\r\n */\r\n useSpecularOverAlpha: boolean;\r\n\r\n /**\r\n * Specifies that the material will keeps the reflection highlights over a transparent surface (only the most limunous ones).\r\n * A car glass is a good exemple of that. When the street lights reflects on it you can not see what is behind.\r\n */\r\n useRadianceOverAlpha: boolean;\r\n\r\n /** Anisotropy settings */\r\n anisotropyEnabled: boolean;\r\n anisotropicDirection: KbVector;\r\n anisotropicIntensity: number;\r\n\r\n /** Clearcoat settings */\r\n clearcoatEnabled: boolean;\r\n clearcoatIntensity: number;\r\n clearcoatTintColor: KbColor;\r\n clearcoatTintEnabled: boolean;\r\n clearcoatTintIntensity: number;\r\n\r\n /** Refraction settings */\r\n subsurfaceRefractionEnabled: boolean;\r\n subsurfaceReflectionEnabled: boolean;\r\n subsurfaceRefractionIndex: number;\r\n subsurfaceRefractionIntensity: number;\r\n subsurfaceRefractionLinkTransparency: boolean;\r\n\r\n /** Translucency settings */\r\n subsurfaceTranslucencyEnabled: boolean;\r\n subsurfaceTranslucencyIntensity: number;\r\n\r\n /** Common subsurface settings shared between refraction and translucency */\r\n subsurfaceThickness: number;\r\n subsurfaceTintColor: KbColor;\r\n subsurfaceUseAlbedoForTintColor: boolean;\r\n}\r\n","import { KbColor } from '@models/classes/primitives/kb-color';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { IColorFillTextureLayer } from '../kb3d-models';\r\nimport { Token } from '../token';\r\nimport { TextureLayer } from './texture-layer';\r\n\r\n@Kb3dModel({\r\n token: Token.ColorFillTextureLayer,\r\n})\r\nexport class ColorFillTextureLayer extends TextureLayer implements IColorFillTextureLayer {\r\n color: KbColor;\r\n\r\n width: number;\r\n height: number;\r\n}\r\n","import { IgnoreChanges, IgnoreHistory } from '../decorators/change-tracker';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { IFileTextureLayer } from '../kb3d-models';\r\nimport { Token } from '../token';\r\nimport { TextureLayer } from './texture-layer';\r\n\r\n@Kb3dModel({\r\n token: Token.FileTextureLayer,\r\n onSerializing: (node, o) => {\r\n return !node._external; // TODO Copy this to other layer types\r\n },\r\n})\r\nexport class FileTextureLayer extends TextureLayer implements IFileTextureLayer {\r\n path = '';\r\n\r\n @IgnoreChanges _needsImageUpdate = false;\r\n @IgnoreHistory\r\n _buffer?: ImageBitmap; // the image data is saved here\r\n _bufferSource: string; // to trigger changes \r\n\r\n width: number;\r\n height: number;\r\n\r\n flipNormalsRg = false;\r\n}\r\n","import { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { ISvgTextureLayer } from '../kb3d-models';\r\nimport { Token } from '../token';\r\nimport { TextureLayer } from './texture-layer';\r\n\r\n@Kb3dModel({\r\n token: Token.SvgTextureLayer,\r\n})\r\nexport class SvgTextureLayer extends TextureLayer implements ISvgTextureLayer {\r\n svgXml = '';\r\n height = 256;\r\n width = 256;\r\n preloadFontStylesheet = '';\r\n\r\n _raster?: ImageBitmap;\r\n}\r\n","import { KbColor } from '@models/classes/primitives/kb-color';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { ITextTextureLayer } from '../kb3d-models';\r\nimport { Token } from '../token';\r\nimport { TextureLayer } from './texture-layer';\r\n\r\n@Kb3dModel({\r\n token: Token.TextTextureLayer,\r\n})\r\nexport class TextTextureLayer extends TextureLayer implements ITextTextureLayer {\r\n color: KbColor;\r\n font: string;\r\n text: string;\r\n}\r\n","import { IgnoreChanges, NodeReference } from '../decorators/change-tracker';\r\nimport { Kb3dModel } from '../decorators/kb3d-decorator';\r\nimport { instanceOf } from '../decorators/model-utils';\r\nimport { eMateType } from '../enums';\r\nimport { BaseConnector } from '../features';\r\nimport { IMateNode } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { Token } from '../token';\r\n\r\n@Kb3dModel({\r\n token: Token.MateNode,\r\n validate: (n, v) => {\r\n if (n.changes.hasAny('connector1', 'connector2')) {\r\n v.assert(\r\n n,\r\n 'connector1',\r\n n.connector1 != null && n._manager?.getById(n.connector1) != null,\r\n 'Connector is required'\r\n );\r\n v.assert(\r\n n,\r\n 'connector2',\r\n n.connector2 != null && n._manager?.getById(n.connector2) != null,\r\n 'Connector is required'\r\n );\r\n v.assert(\r\n n,\r\n 'connector2',\r\n n._manager?.getById(n.connector1)?._parent !== n._manager?.getById(n.connector2)?._parent ||\r\n n._manager?.getById(n.connector1) == null,\r\n 'Connectors cannot have the same parent node'\r\n );\r\n }\r\n },\r\n})\r\nexport class MateNode extends Kb3dObject implements IMateNode {\r\n constructor() {\r\n super();\r\n }\r\n\r\n enabled: boolean;\r\n mateType: eMateType;\r\n @NodeReference(BaseConnector, 'position', 'direction', 'angle') connector1: string;\r\n @NodeReference(BaseConnector, 'position', 'direction', 'angle') connector2: string;\r\n flip: boolean;\r\n offsetX: number;\r\n offsetY: number;\r\n offsetZ: number;\r\n offsetAngle: number;\r\n\r\n /** used internally when processing mates to control whether c1/c2 need to be swapped */\r\n @IgnoreChanges\r\n _swap: boolean;\r\n\r\n getC1() {\r\n return this._manager?.getById(this.connector1);\r\n }\r\n getC2() {\r\n return this._manager?.getById(this.connector2);\r\n }\r\n isActive() {\r\n return this.enabled && this.getC1() != null && this.getC2() != null;\r\n }\r\n}\r\n\r\nexport function isMate(node: Kb3dObject): node is MateNode {\r\n return instanceOf(node, Token.MateNode);\r\n}\r\n","import { SetMap } from '@models/classes/handlers';\r\n\r\nexport interface TopologicalSortResult {\r\n /** the topologically sorted result set */\r\n sorted: Set;\r\n /** if any cycles were detected (which is not good), they are returned here */\r\n cycles: T[][];\r\n}\r\n\r\n/**\r\n * Does a topological sort of directed graph with cycle detection and warning.\r\n * @param nodes a map where the key is the node and value is an array containing the nodes the key node depends on\r\n * @param nameSelector function that returns a descriptive name for the node\r\n */\r\nexport function topologicalSort(nodes: SetMap, nameSelector: (node: T) => string) {\r\n const visited = new Set();\r\n const path = new Set();\r\n const sorted = new Set();\r\n const cycles = new Array();\r\n\r\n for (const [node] of nodes) {\r\n recurse(nodes, nameSelector, node, visited, path, sorted, cycles);\r\n }\r\n\r\n // console.log(`Original graph: ${getDescriptor(nameSelector, ...nodes.keys())}`);\r\n // console.log(`Sorted graph: ${getDescriptor(nameSelector, ...sorted)}`);\r\n\r\n return {\r\n sorted,\r\n cycles,\r\n } as TopologicalSortResult;\r\n}\r\n\r\nfunction recurse(\r\n nodes: SetMap,\r\n nameSelector: (node: T) => string,\r\n node: T,\r\n visited: Set,\r\n path: Set,\r\n sorted: Set,\r\n cycles: T[][]\r\n) {\r\n if (path.has(node)) {\r\n const items = [...path].reverse();\r\n cycles.push(items);\r\n //console.warn(`Cycle detected: ${getDescriptor(nameSelector, ...path, node)}`);\r\n }\r\n\r\n if (visited.has(node)) return;\r\n visited.add(node);\r\n path.add(node);\r\n\r\n const edges = nodes.get(node);\r\n if (edges) {\r\n for (const edge of edges) {\r\n recurse(nodes, nameSelector, edge, visited, path, sorted, cycles);\r\n }\r\n }\r\n\r\n path.delete(node);\r\n sorted.add(node);\r\n}\r\n\r\n// function getDescriptor(nameSelector: (n: any) => string, ...nodes: any[]){\r\n// return nodes.map(n => nameSelector(n)).join(\" -> \");\r\n// }\r\n","import { Enumerable, instanceOf, Kb3dModel } from '../decorators';\r\nimport { defaults } from '../default-values';\r\nimport { eViewpointProjection, eViewpointTargetMode, eViewpointType } from '../enums';\r\nimport { IViewpointNode } from '../kb3d-models';\r\nimport { Kb3dObject } from '../kb3d-object';\r\nimport { KbVector } from '../primitives';\r\nimport { Token } from '../token';\r\n\r\n@Kb3dModel({\r\n token: Token.ViewpointNode,\r\n onSerializing: (node, o) => {\r\n // remove any target properties not relevant to the target mode\r\n if (node.targetMode === eViewpointTargetMode.rotation) {\r\n delete o.targetPoint;\r\n delete node.targetMeshId;\r\n } else if (node.targetMode === eViewpointTargetMode.point) {\r\n delete o.rotation;\r\n delete node.targetMeshId;\r\n } else if (node.targetMode === eViewpointTargetMode.mesh) {\r\n delete o.targetPoint;\r\n delete o.rotation;\r\n }\r\n return true;\r\n },\r\n validate: (n, v) => {\r\n if (n.changes.hasAny('targetMode', 'targetMeshId')) {\r\n const exp =\r\n n.targetMode != eViewpointTargetMode.mesh ||\r\n (n.targetMeshId != null && n._manager?.getById(n.targetMeshId) != null);\r\n v.assert(n, 'targetMeshId', exp, 'Target mesh required');\r\n }\r\n },\r\n})\r\nexport class ViewpointNode extends Kb3dObject implements IViewpointNode {\r\n constructor() {\r\n super();\r\n\r\n this.type = eViewpointType.orbit;\r\n this.targetMode = eViewpointTargetMode.point;\r\n this.position = KbVector.Zero();\r\n this.rotation = KbVector.Zero();\r\n this.targetPoint = KbVector.Zero();\r\n this.fov = defaults.ViewpointNode.fov;\r\n this.projection = eViewpointProjection.perspective;\r\n }\r\n\r\n private _type?: eViewpointType;\r\n @Enumerable()\r\n public get type() {\r\n return this._type;\r\n }\r\n public set type(newType) {\r\n this._type = newType;\r\n switch (newType) {\r\n case eViewpointType.firstPerson:\r\n this.targetMode = eViewpointTargetMode.rotation;\r\n this.projection = eViewpointProjection.perspective;\r\n break;\r\n }\r\n }\r\n targetMode: eViewpointTargetMode;\r\n position: KbVector;\r\n rotation: KbVector;\r\n targetPoint: KbVector;\r\n targetMeshId: string | undefined;\r\n orbitAxis: KbVector;\r\n fov: number;\r\n projection: eViewpointProjection;\r\n minTargetDistance: number;\r\n maxTargetDistance: number;\r\n\r\n offsetX = 0; // For panning\r\n offsetY = 0;\r\n\r\n allowOrbit = true;\r\n allowPan = true;\r\n allowZoom = true;\r\n limitMinimumTargetDistance = false;\r\n limitLatOrbit = false;\r\n limitMinLatitude: number;\r\n limitMaxLatitude: number;\r\n limitLngOrbit = false;\r\n limitMinLongitude: number;\r\n limitMaxLongitude: number;\r\n}\r\n\r\nexport function isViewpointNode(node: Kb3dObject): node is ViewpointNode {\r\n return instanceOf(node, Token.ViewpointNode);\r\n}\r\n","import type { Nullable } from \"../../types\";\r\nimport type { InternalTexture } from \"../../Materials/Textures/internalTexture\";\r\nimport { Constants } from \"../../Engines/constants\";\r\n\r\nimport type { ISize } from \"../../Maths/math.size\";\r\nimport { Size } from \"../../Maths/math.size\";\r\n\r\nimport type { ThinEngine } from \"../../Engines/thinEngine\";\r\nimport type { RenderTargetWrapper } from \"core/Engines/renderTargetWrapper\";\r\n\r\n/**\r\n * Base class of all the textures in babylon.\r\n * It groups all the common properties required to work with Thin Engine.\r\n */\r\nexport class ThinTexture {\r\n protected _wrapU = Constants.TEXTURE_WRAP_ADDRESSMODE;\r\n /**\r\n * | Value | Type | Description |\r\n * | ----- | ------------------ | ----------- |\r\n * | 0 | CLAMP_ADDRESSMODE | |\r\n * | 1 | WRAP_ADDRESSMODE | |\r\n * | 2 | MIRROR_ADDRESSMODE | |\r\n */\r\n public get wrapU() {\r\n return this._wrapU;\r\n }\r\n\r\n public set wrapU(value: number) {\r\n this._wrapU = value;\r\n }\r\n\r\n protected _wrapV = Constants.TEXTURE_WRAP_ADDRESSMODE;\r\n /**\r\n * | Value | Type | Description |\r\n * | ----- | ------------------ | ----------- |\r\n * | 0 | CLAMP_ADDRESSMODE | |\r\n * | 1 | WRAP_ADDRESSMODE | |\r\n * | 2 | MIRROR_ADDRESSMODE | |\r\n */\r\n public get wrapV() {\r\n return this._wrapV;\r\n }\r\n\r\n public set wrapV(value: number) {\r\n this._wrapV = value;\r\n }\r\n\r\n /**\r\n * | Value | Type | Description |\r\n * | ----- | ------------------ | ----------- |\r\n * | 0 | CLAMP_ADDRESSMODE | |\r\n * | 1 | WRAP_ADDRESSMODE | |\r\n * | 2 | MIRROR_ADDRESSMODE | |\r\n */\r\n public wrapR = Constants.TEXTURE_WRAP_ADDRESSMODE;\r\n\r\n /**\r\n * With compliant hardware and browser (supporting anisotropic filtering)\r\n * this defines the level of anisotropic filtering in the texture.\r\n * The higher the better but the slower. This defaults to 4 as it seems to be the best tradeoff.\r\n */\r\n public anisotropicFilteringLevel = 4;\r\n\r\n /**\r\n * Define the current state of the loading sequence when in delayed load mode.\r\n */\r\n public delayLoadState = Constants.DELAYLOADSTATE_NONE;\r\n\r\n /**\r\n * How a texture is mapped.\r\n * Unused in thin texture mode.\r\n */\r\n public get coordinatesMode(): number {\r\n return 0;\r\n }\r\n\r\n /**\r\n * Define if the texture is a cube texture or if false a 2d texture.\r\n */\r\n public get isCube(): boolean {\r\n if (!this._texture) {\r\n return false;\r\n }\r\n\r\n return this._texture.isCube;\r\n }\r\n\r\n protected set isCube(value: boolean) {\r\n if (!this._texture) {\r\n return;\r\n }\r\n\r\n this._texture.isCube = value;\r\n }\r\n\r\n /**\r\n * Define if the texture is a 3d texture (webgl 2) or if false a 2d texture.\r\n */\r\n public get is3D(): boolean {\r\n if (!this._texture) {\r\n return false;\r\n }\r\n\r\n return this._texture.is3D;\r\n }\r\n\r\n protected set is3D(value: boolean) {\r\n if (!this._texture) {\r\n return;\r\n }\r\n\r\n this._texture.is3D = value;\r\n }\r\n\r\n /**\r\n * Define if the texture is a 2d array texture (webgl 2) or if false a 2d texture.\r\n */\r\n public get is2DArray(): boolean {\r\n if (!this._texture) {\r\n return false;\r\n }\r\n\r\n return this._texture.is2DArray;\r\n }\r\n\r\n protected set is2DArray(value: boolean) {\r\n if (!this._texture) {\r\n return;\r\n }\r\n\r\n this._texture.is2DArray = value;\r\n }\r\n\r\n /**\r\n * Get the class name of the texture.\r\n * @returns \"ThinTexture\"\r\n */\r\n public getClassName(): string {\r\n return \"ThinTexture\";\r\n }\r\n\r\n /** @internal */\r\n public _texture: Nullable = null;\r\n\r\n protected _engine: Nullable = null;\r\n\r\n private _cachedSize: ISize = Size.Zero();\r\n private _cachedBaseSize: ISize = Size.Zero();\r\n\r\n private static _IsRenderTargetWrapper(texture: Nullable | Nullable): texture is RenderTargetWrapper {\r\n return (texture as RenderTargetWrapper)?._shareDepth !== undefined;\r\n }\r\n\r\n /**\r\n * Instantiates a new ThinTexture.\r\n * Base class of all the textures in babylon.\r\n * This can be used as an internal texture wrapper in ThinEngine to benefit from the cache\r\n * @param internalTexture Define the internalTexture to wrap. You can also pass a RenderTargetWrapper, in which case the texture will be the render target's texture\r\n */\r\n constructor(internalTexture: Nullable) {\r\n this._texture = ThinTexture._IsRenderTargetWrapper(internalTexture) ? internalTexture.texture : internalTexture;\r\n if (this._texture) {\r\n this._engine = this._texture.getEngine();\r\n }\r\n }\r\n\r\n /**\r\n * Get if the texture is ready to be used (downloaded, converted, mip mapped...).\r\n * @returns true if fully ready\r\n */\r\n public isReady(): boolean {\r\n if (this.delayLoadState === Constants.DELAYLOADSTATE_NOTLOADED) {\r\n this.delayLoad();\r\n return false;\r\n }\r\n\r\n if (this._texture) {\r\n return this._texture.isReady;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Triggers the load sequence in delayed load mode.\r\n */\r\n public delayLoad(): void {}\r\n\r\n /**\r\n * Get the underlying lower level texture from Babylon.\r\n * @returns the internal texture\r\n */\r\n public getInternalTexture(): Nullable {\r\n return this._texture;\r\n }\r\n\r\n /**\r\n * Get the size of the texture.\r\n * @returns the texture size.\r\n */\r\n public getSize(): ISize {\r\n if (this._texture) {\r\n if (this._texture.width) {\r\n this._cachedSize.width = this._texture.width;\r\n this._cachedSize.height = this._texture.height;\r\n return this._cachedSize;\r\n }\r\n\r\n if (this._texture._size) {\r\n this._cachedSize.width = this._texture._size;\r\n this._cachedSize.height = this._texture._size;\r\n return this._cachedSize;\r\n }\r\n }\r\n\r\n return this._cachedSize;\r\n }\r\n\r\n /**\r\n * Get the base size of the texture.\r\n * It can be different from the size if the texture has been resized for POT for instance\r\n * @returns the base size\r\n */\r\n public getBaseSize(): ISize {\r\n if (!this.isReady() || !this._texture) {\r\n this._cachedBaseSize.width = 0;\r\n this._cachedBaseSize.height = 0;\r\n return this._cachedBaseSize;\r\n }\r\n\r\n if (this._texture._size) {\r\n this._cachedBaseSize.width = this._texture._size;\r\n this._cachedBaseSize.height = this._texture._size;\r\n return this._cachedBaseSize;\r\n }\r\n\r\n this._cachedBaseSize.width = this._texture.baseWidth;\r\n this._cachedBaseSize.height = this._texture.baseHeight;\r\n return this._cachedBaseSize;\r\n }\r\n\r\n /** @internal */\r\n protected _initialSamplingMode = Constants.TEXTURE_BILINEAR_SAMPLINGMODE;\r\n\r\n /**\r\n * Get the current sampling mode associated with the texture.\r\n */\r\n public get samplingMode(): number {\r\n if (!this._texture) {\r\n return this._initialSamplingMode;\r\n }\r\n\r\n return this._texture.samplingMode;\r\n }\r\n\r\n /**\r\n * Update the sampling mode of the texture.\r\n * Default is Trilinear mode.\r\n *\r\n * | Value | Type | Description |\r\n * | ----- | ------------------ | ----------- |\r\n * | 1 | NEAREST_SAMPLINGMODE or NEAREST_NEAREST_MIPLINEAR | Nearest is: mag = nearest, min = nearest, mip = linear |\r\n * | 2 | BILINEAR_SAMPLINGMODE or LINEAR_LINEAR_MIPNEAREST | Bilinear is: mag = linear, min = linear, mip = nearest |\r\n * | 3 | TRILINEAR_SAMPLINGMODE or LINEAR_LINEAR_MIPLINEAR | Trilinear is: mag = linear, min = linear, mip = linear |\r\n * | 4 | NEAREST_NEAREST_MIPNEAREST | |\r\n * | 5 | NEAREST_LINEAR_MIPNEAREST | |\r\n * | 6 | NEAREST_LINEAR_MIPLINEAR | |\r\n * | 7 | NEAREST_LINEAR | |\r\n * | 8 | NEAREST_NEAREST | |\r\n * | 9 | LINEAR_NEAREST_MIPNEAREST | |\r\n * | 10 | LINEAR_NEAREST_MIPLINEAR | |\r\n * | 11 | LINEAR_LINEAR | |\r\n * | 12 | LINEAR_NEAREST | |\r\n *\r\n * > _mag_: magnification filter (close to the viewer)\r\n * > _min_: minification filter (far from the viewer)\r\n * > _mip_: filter used between mip map levels\r\n *@param samplingMode Define the new sampling mode of the texture\r\n */\r\n public updateSamplingMode(samplingMode: number): void {\r\n if (this._texture && this._engine) {\r\n this._engine.updateTextureSamplingMode(samplingMode, this._texture);\r\n }\r\n }\r\n\r\n /**\r\n * Release and destroy the underlying lower level texture aka internalTexture.\r\n */\r\n public releaseInternalTexture(): void {\r\n if (this._texture) {\r\n this._texture.dispose();\r\n this._texture = null;\r\n }\r\n }\r\n\r\n /**\r\n * Dispose the texture and release its associated resources.\r\n */\r\n public dispose(): void {\r\n if (this._texture) {\r\n this.releaseInternalTexture();\r\n this._engine = null;\r\n }\r\n }\r\n}\r\n","import { serialize, SerializationHelper, serializeAsTexture } from \"../../Misc/decorators\";\r\nimport type { Observer } from \"../../Misc/observable\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { Matrix } from \"../../Maths/math.vector\";\r\nimport { EngineStore } from \"../../Engines/engineStore\";\r\nimport type { InternalTexture } from \"../../Materials/Textures/internalTexture\";\r\nimport { Constants } from \"../../Engines/constants\";\r\nimport type { IAnimatable } from \"../../Animations/animatable.interface\";\r\nimport { RandomGUID } from \"../../Misc/guid\";\r\n\r\nimport \"../../Misc/fileTools\";\r\nimport type { ThinEngine } from \"../../Engines/thinEngine\";\r\nimport { ThinTexture } from \"./thinTexture\";\r\nimport type { AbstractScene } from \"../../abstractScene\";\r\n\r\nimport type { Animation } from \"../../Animations/animation\";\r\n\r\n/**\r\n * Base class of all the textures in babylon.\r\n * It groups all the common properties the materials, post process, lights... might need\r\n * in order to make a correct use of the texture.\r\n */\r\nexport class BaseTexture extends ThinTexture implements IAnimatable {\r\n /**\r\n * Default anisotropic filtering level for the application.\r\n * It is set to 4 as a good tradeoff between perf and quality.\r\n */\r\n public static DEFAULT_ANISOTROPIC_FILTERING_LEVEL = 4;\r\n\r\n /**\r\n * Gets or sets the unique id of the texture\r\n */\r\n @serialize()\r\n public uniqueId: number;\r\n\r\n /**\r\n * Define the name of the texture.\r\n */\r\n @serialize()\r\n public name: string;\r\n\r\n /**\r\n * Gets or sets an object used to store user defined information.\r\n */\r\n @serialize()\r\n public metadata: any = null;\r\n\r\n /** @internal */\r\n public _internalMetadata: any;\r\n\r\n /**\r\n * For internal use only. Please do not use.\r\n */\r\n public reservedDataStore: any = null;\r\n\r\n @serialize(\"hasAlpha\")\r\n private _hasAlpha = false;\r\n /**\r\n * Define if the texture is having a usable alpha value (can be use for transparency or glossiness for instance).\r\n */\r\n public set hasAlpha(value: boolean) {\r\n if (this._hasAlpha === value) {\r\n return;\r\n }\r\n this._hasAlpha = value;\r\n if (this._scene) {\r\n this._scene.markAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag, (mat) => {\r\n return mat.hasTexture(this);\r\n });\r\n }\r\n }\r\n public get hasAlpha(): boolean {\r\n return this._hasAlpha;\r\n }\r\n\r\n @serialize(\"getAlphaFromRGB\")\r\n private _getAlphaFromRGB = false;\r\n /**\r\n * Defines if the alpha value should be determined via the rgb values.\r\n * If true the luminance of the pixel might be used to find the corresponding alpha value.\r\n */\r\n public set getAlphaFromRGB(value: boolean) {\r\n if (this._getAlphaFromRGB === value) {\r\n return;\r\n }\r\n this._getAlphaFromRGB = value;\r\n if (this._scene) {\r\n this._scene.markAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag, (mat) => {\r\n return mat.hasTexture(this);\r\n });\r\n }\r\n }\r\n public get getAlphaFromRGB(): boolean {\r\n return this._getAlphaFromRGB;\r\n }\r\n\r\n /**\r\n * Intensity or strength of the texture.\r\n * It is commonly used by materials to fine tune the intensity of the texture\r\n */\r\n @serialize()\r\n public level = 1;\r\n\r\n @serialize(\"coordinatesIndex\")\r\n protected _coordinatesIndex = 0;\r\n\r\n /**\r\n * Gets or sets a boolean indicating that the texture should try to reduce shader code if there is no UV manipulation.\r\n * (ie. when texture.getTextureMatrix().isIdentityAs3x2() returns true)\r\n */\r\n @serialize()\r\n public optimizeUVAllocation = true;\r\n\r\n /**\r\n * Define the UV channel to use starting from 0 and defaulting to 0.\r\n * This is part of the texture as textures usually maps to one uv set.\r\n */\r\n public set coordinatesIndex(value: number) {\r\n if (this._coordinatesIndex === value) {\r\n return;\r\n }\r\n this._coordinatesIndex = value;\r\n if (this._scene) {\r\n this._scene.markAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag, (mat) => {\r\n return mat.hasTexture(this);\r\n });\r\n }\r\n }\r\n public get coordinatesIndex(): number {\r\n return this._coordinatesIndex;\r\n }\r\n\r\n @serialize(\"coordinatesMode\")\r\n protected _coordinatesMode = Constants.TEXTURE_EXPLICIT_MODE;\r\n\r\n /**\r\n * How a texture is mapped.\r\n *\r\n * | Value | Type | Description |\r\n * | ----- | ----------------------------------- | ----------- |\r\n * | 0 | EXPLICIT_MODE | |\r\n * | 1 | SPHERICAL_MODE | |\r\n * | 2 | PLANAR_MODE | |\r\n * | 3 | CUBIC_MODE | |\r\n * | 4 | PROJECTION_MODE | |\r\n * | 5 | SKYBOX_MODE | |\r\n * | 6 | INVCUBIC_MODE | |\r\n * | 7 | EQUIRECTANGULAR_MODE | |\r\n * | 8 | FIXED_EQUIRECTANGULAR_MODE | |\r\n * | 9 | FIXED_EQUIRECTANGULAR_MIRRORED_MODE | |\r\n */\r\n public set coordinatesMode(value: number) {\r\n if (this._coordinatesMode === value) {\r\n return;\r\n }\r\n this._coordinatesMode = value;\r\n if (this._scene) {\r\n this._scene.markAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag, (mat) => {\r\n return mat.hasTexture(this);\r\n });\r\n }\r\n }\r\n public get coordinatesMode(): number {\r\n return this._coordinatesMode;\r\n }\r\n\r\n /**\r\n * | Value | Type | Description |\r\n * | ----- | ------------------ | ----------- |\r\n * | 0 | CLAMP_ADDRESSMODE | |\r\n * | 1 | WRAP_ADDRESSMODE | |\r\n * | 2 | MIRROR_ADDRESSMODE | |\r\n */\r\n @serialize()\r\n public get wrapU() {\r\n return this._wrapU;\r\n }\r\n public set wrapU(value: number) {\r\n this._wrapU = value;\r\n }\r\n\r\n /**\r\n * | Value | Type | Description |\r\n * | ----- | ------------------ | ----------- |\r\n * | 0 | CLAMP_ADDRESSMODE | |\r\n * | 1 | WRAP_ADDRESSMODE | |\r\n * | 2 | MIRROR_ADDRESSMODE | |\r\n */\r\n @serialize()\r\n public get wrapV() {\r\n return this._wrapV;\r\n }\r\n public set wrapV(value: number) {\r\n this._wrapV = value;\r\n }\r\n\r\n /**\r\n * | Value | Type | Description |\r\n * | ----- | ------------------ | ----------- |\r\n * | 0 | CLAMP_ADDRESSMODE | |\r\n * | 1 | WRAP_ADDRESSMODE | |\r\n * | 2 | MIRROR_ADDRESSMODE | |\r\n */\r\n @serialize()\r\n public wrapR = Constants.TEXTURE_WRAP_ADDRESSMODE;\r\n\r\n /**\r\n * With compliant hardware and browser (supporting anisotropic filtering)\r\n * this defines the level of anisotropic filtering in the texture.\r\n * The higher the better but the slower. This defaults to 4 as it seems to be the best tradeoff.\r\n */\r\n @serialize()\r\n public anisotropicFilteringLevel = BaseTexture.DEFAULT_ANISOTROPIC_FILTERING_LEVEL;\r\n\r\n /** @internal */\r\n public _isCube = false;\r\n /**\r\n * Define if the texture is a cube texture or if false a 2d texture.\r\n */\r\n @serialize()\r\n public get isCube(): boolean {\r\n if (!this._texture) {\r\n return this._isCube;\r\n }\r\n\r\n return this._texture.isCube;\r\n }\r\n\r\n protected set isCube(value: boolean) {\r\n if (!this._texture) {\r\n this._isCube = value;\r\n } else {\r\n this._texture.isCube = value;\r\n }\r\n }\r\n\r\n /**\r\n * Define if the texture is a 3d texture (webgl 2) or if false a 2d texture.\r\n */\r\n @serialize()\r\n public get is3D(): boolean {\r\n if (!this._texture) {\r\n return false;\r\n }\r\n\r\n return this._texture.is3D;\r\n }\r\n\r\n protected set is3D(value: boolean) {\r\n if (!this._texture) {\r\n return;\r\n }\r\n\r\n this._texture.is3D = value;\r\n }\r\n\r\n /**\r\n * Define if the texture is a 2d array texture (webgl 2) or if false a 2d texture.\r\n */\r\n @serialize()\r\n public get is2DArray(): boolean {\r\n if (!this._texture) {\r\n return false;\r\n }\r\n\r\n return this._texture.is2DArray;\r\n }\r\n\r\n protected set is2DArray(value: boolean) {\r\n if (!this._texture) {\r\n return;\r\n }\r\n\r\n this._texture.is2DArray = value;\r\n }\r\n\r\n private _gammaSpace = true;\r\n /**\r\n * Define if the texture contains data in gamma space (most of the png/jpg aside bump).\r\n * HDR texture are usually stored in linear space.\r\n * This only impacts the PBR and Background materials\r\n */\r\n @serialize()\r\n public get gammaSpace(): boolean {\r\n if (!this._texture) {\r\n return this._gammaSpace;\r\n } else {\r\n if (this._texture._gammaSpace === null) {\r\n this._texture._gammaSpace = this._gammaSpace;\r\n }\r\n }\r\n\r\n return this._texture._gammaSpace && !this._texture._useSRGBBuffer;\r\n }\r\n\r\n public set gammaSpace(gamma: boolean) {\r\n if (!this._texture) {\r\n if (this._gammaSpace === gamma) {\r\n return;\r\n }\r\n\r\n this._gammaSpace = gamma;\r\n } else {\r\n if (this._texture._gammaSpace === gamma) {\r\n return;\r\n }\r\n this._texture._gammaSpace = gamma;\r\n }\r\n\r\n this._markAllSubMeshesAsTexturesDirty();\r\n }\r\n\r\n /**\r\n * Gets or sets whether or not the texture contains RGBD data.\r\n */\r\n public get isRGBD(): boolean {\r\n return this._texture != null && this._texture._isRGBD;\r\n }\r\n public set isRGBD(value: boolean) {\r\n if (this._texture) {\r\n this._texture._isRGBD = value;\r\n }\r\n }\r\n\r\n /**\r\n * Is Z inverted in the texture (useful in a cube texture).\r\n */\r\n @serialize()\r\n public invertZ = false;\r\n\r\n /**\r\n * Are mip maps generated for this texture or not.\r\n */\r\n public get noMipmap(): boolean {\r\n return false;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n @serialize()\r\n public lodLevelInAlpha = false;\r\n\r\n /**\r\n * With prefiltered texture, defined the offset used during the prefiltering steps.\r\n */\r\n @serialize()\r\n public get lodGenerationOffset(): number {\r\n if (this._texture) {\r\n return this._texture._lodGenerationOffset;\r\n }\r\n\r\n return 0.0;\r\n }\r\n public set lodGenerationOffset(value: number) {\r\n if (this._texture) {\r\n this._texture._lodGenerationOffset = value;\r\n }\r\n }\r\n\r\n /**\r\n * With prefiltered texture, defined the scale used during the prefiltering steps.\r\n */\r\n @serialize()\r\n public get lodGenerationScale(): number {\r\n if (this._texture) {\r\n return this._texture._lodGenerationScale;\r\n }\r\n\r\n return 0.0;\r\n }\r\n public set lodGenerationScale(value: number) {\r\n if (this._texture) {\r\n this._texture._lodGenerationScale = value;\r\n }\r\n }\r\n\r\n /**\r\n * With prefiltered texture, defined if the specular generation is based on a linear ramp.\r\n * By default we are using a log2 of the linear roughness helping to keep a better resolution for\r\n * average roughness values.\r\n */\r\n @serialize()\r\n public get linearSpecularLOD(): boolean {\r\n if (this._texture) {\r\n return this._texture._linearSpecularLOD;\r\n }\r\n\r\n return false;\r\n }\r\n public set linearSpecularLOD(value: boolean) {\r\n if (this._texture) {\r\n this._texture._linearSpecularLOD = value;\r\n }\r\n }\r\n\r\n /**\r\n * In case a better definition than spherical harmonics is required for the diffuse part of the environment.\r\n * You can set the irradiance texture to rely on a texture instead of the spherical approach.\r\n * This texture need to have the same characteristics than its parent (Cube vs 2d, coordinates mode, Gamma/Linear, RGBD).\r\n */\r\n @serializeAsTexture()\r\n public get irradianceTexture(): Nullable {\r\n if (this._texture) {\r\n return this._texture._irradianceTexture;\r\n }\r\n\r\n return null;\r\n }\r\n public set irradianceTexture(value: Nullable) {\r\n if (this._texture) {\r\n this._texture._irradianceTexture = value;\r\n }\r\n }\r\n\r\n /**\r\n * Define if the texture is a render target.\r\n */\r\n @serialize()\r\n public isRenderTarget = false;\r\n\r\n /**\r\n * Define the unique id of the texture in the scene.\r\n */\r\n public get uid(): string {\r\n if (!this._uid) {\r\n this._uid = RandomGUID();\r\n }\r\n return this._uid;\r\n }\r\n\r\n /** @internal */\r\n public _prefiltered: boolean = false;\r\n /** @internal */\r\n public _forceSerialize: boolean = false;\r\n\r\n /**\r\n * Return a string representation of the texture.\r\n * @returns the texture as a string\r\n */\r\n public toString(): string {\r\n return this.name;\r\n }\r\n\r\n /**\r\n * Get the class name of the texture.\r\n * @returns \"BaseTexture\"\r\n */\r\n public getClassName(): string {\r\n return \"BaseTexture\";\r\n }\r\n\r\n /**\r\n * Define the list of animation attached to the texture.\r\n */\r\n public animations = new Array();\r\n\r\n /**\r\n * An event triggered when the texture is disposed.\r\n */\r\n public onDisposeObservable = new Observable();\r\n\r\n private _onDisposeObserver: Nullable> = null;\r\n /**\r\n * Callback triggered when the texture has been disposed.\r\n * Kept for back compatibility, you can use the onDisposeObservable instead.\r\n */\r\n public set onDispose(callback: () => void) {\r\n if (this._onDisposeObserver) {\r\n this.onDisposeObservable.remove(this._onDisposeObserver);\r\n }\r\n this._onDisposeObserver = this.onDisposeObservable.add(callback);\r\n }\r\n\r\n protected _scene: Nullable = null;\r\n\r\n /** @internal */\r\n private _uid: Nullable = null;\r\n\r\n /**\r\n * Define if the texture is preventing a material to render or not.\r\n * If not and the texture is not ready, the engine will use a default black texture instead.\r\n */\r\n public get isBlocking(): boolean {\r\n return true;\r\n }\r\n\r\n /** @internal */\r\n public _parentContainer: Nullable = null;\r\n\r\n protected _loadingError: boolean = false;\r\n protected _errorObject?: {\r\n message?: string;\r\n exception?: any;\r\n };\r\n\r\n /**\r\n * Was there any loading error?\r\n */\r\n public get loadingError(): boolean {\r\n return this._loadingError;\r\n }\r\n\r\n /**\r\n * If a loading error occurred this object will be populated with information about the error.\r\n */\r\n public get errorObject():\r\n | {\r\n message?: string;\r\n exception?: any;\r\n }\r\n | undefined {\r\n return this._errorObject;\r\n }\r\n\r\n /**\r\n * Instantiates a new BaseTexture.\r\n * Base class of all the textures in babylon.\r\n * It groups all the common properties the materials, post process, lights... might need\r\n * in order to make a correct use of the texture.\r\n * @param sceneOrEngine Define the scene or engine the texture belongs to\r\n * @param internalTexture Define the internal texture associated with the texture\r\n */\r\n constructor(sceneOrEngine?: Nullable, internalTexture: Nullable = null) {\r\n super(null);\r\n\r\n if (sceneOrEngine) {\r\n if (BaseTexture._IsScene(sceneOrEngine)) {\r\n this._scene = sceneOrEngine;\r\n } else {\r\n this._engine = sceneOrEngine;\r\n }\r\n } else {\r\n this._scene = EngineStore.LastCreatedScene;\r\n }\r\n\r\n if (this._scene) {\r\n this.uniqueId = this._scene.getUniqueId();\r\n this._scene.addTexture(this);\r\n this._engine = this._scene.getEngine();\r\n }\r\n\r\n this._texture = internalTexture;\r\n\r\n this._uid = null;\r\n }\r\n\r\n /**\r\n * Get the scene the texture belongs to.\r\n * @returns the scene or null if undefined\r\n */\r\n public getScene(): Nullable {\r\n return this._scene;\r\n }\r\n\r\n /** @internal */\r\n protected _getEngine(): Nullable {\r\n return this._engine;\r\n }\r\n\r\n /**\r\n * Checks if the texture has the same transform matrix than another texture\r\n * @param texture texture to check against\r\n * @returns true if the transforms are the same, else false\r\n */\r\n public checkTransformsAreIdentical(texture: Nullable): boolean {\r\n return texture !== null;\r\n }\r\n\r\n /**\r\n * Get the texture transform matrix used to offset tile the texture for instance.\r\n * @returns the transformation matrix\r\n */\r\n public getTextureMatrix(): Matrix {\r\n return Matrix.IdentityReadOnly;\r\n }\r\n\r\n /**\r\n * Get the texture reflection matrix used to rotate/transform the reflection.\r\n * @returns the reflection matrix\r\n */\r\n public getReflectionTextureMatrix(): Matrix {\r\n return Matrix.IdentityReadOnly;\r\n }\r\n\r\n /**\r\n * Gets a suitable rotate/transform matrix when the texture is used for refraction.\r\n * There's a separate function from getReflectionTextureMatrix because refraction requires a special configuration of the matrix in right-handed mode.\r\n * @returns The refraction matrix\r\n */\r\n public getRefractionTextureMatrix(): Matrix {\r\n return this.getReflectionTextureMatrix();\r\n }\r\n\r\n /**\r\n * Get if the texture is ready to be consumed (either it is ready or it is not blocking)\r\n * @returns true if ready, not blocking or if there was an error loading the texture\r\n */\r\n public isReadyOrNotBlocking(): boolean {\r\n return !this.isBlocking || this.isReady() || this.loadingError;\r\n }\r\n\r\n /**\r\n * Scales the texture if is `canRescale()`\r\n * @param ratio the resize factor we want to use to rescale\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public scale(ratio: number): void {}\r\n\r\n /**\r\n * Get if the texture can rescale.\r\n */\r\n public get canRescale(): boolean {\r\n return false;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getFromCache(url: Nullable, noMipmap: boolean, sampling?: number, invertY?: boolean, useSRGBBuffer?: boolean, isCube?: boolean): Nullable {\r\n const engine = this._getEngine();\r\n if (!engine) {\r\n return null;\r\n }\r\n\r\n const correctedUseSRGBBuffer = engine._getUseSRGBBuffer(!!useSRGBBuffer, noMipmap);\r\n\r\n const texturesCache = engine.getLoadedTexturesCache();\r\n for (let index = 0; index < texturesCache.length; index++) {\r\n const texturesCacheEntry = texturesCache[index];\r\n\r\n if (useSRGBBuffer === undefined || correctedUseSRGBBuffer === texturesCacheEntry._useSRGBBuffer) {\r\n if (invertY === undefined || invertY === texturesCacheEntry.invertY) {\r\n if (texturesCacheEntry.url === url && texturesCacheEntry.generateMipMaps === !noMipmap) {\r\n if (!sampling || sampling === texturesCacheEntry.samplingMode) {\r\n if (isCube === undefined || isCube === texturesCacheEntry.isCube) {\r\n texturesCacheEntry.incrementReferences();\r\n return texturesCacheEntry;\r\n }\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /** @internal */\r\n public _rebuild(): void {}\r\n\r\n /**\r\n * Clones the texture.\r\n * @returns the cloned texture\r\n */\r\n public clone(): Nullable {\r\n return null;\r\n }\r\n\r\n /**\r\n * Get the texture underlying type (INT, FLOAT...)\r\n */\r\n public get textureType(): number {\r\n if (!this._texture) {\r\n return Constants.TEXTURETYPE_UNSIGNED_INT;\r\n }\r\n\r\n return this._texture.type !== undefined ? this._texture.type : Constants.TEXTURETYPE_UNSIGNED_INT;\r\n }\r\n\r\n /**\r\n * Get the texture underlying format (RGB, RGBA...)\r\n */\r\n public get textureFormat(): number {\r\n if (!this._texture) {\r\n return Constants.TEXTUREFORMAT_RGBA;\r\n }\r\n\r\n return this._texture.format !== undefined ? this._texture.format : Constants.TEXTUREFORMAT_RGBA;\r\n }\r\n\r\n /**\r\n * Indicates that textures need to be re-calculated for all materials\r\n */\r\n protected _markAllSubMeshesAsTexturesDirty() {\r\n const scene = this.getScene();\r\n\r\n if (!scene) {\r\n return;\r\n }\r\n\r\n scene.markAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n /**\r\n * Reads the pixels stored in the webgl texture and returns them as an ArrayBuffer.\r\n * This will returns an RGBA array buffer containing either in values (0-255) or\r\n * float values (0-1) depending of the underlying buffer type.\r\n * @param faceIndex defines the face of the texture to read (in case of cube texture)\r\n * @param level defines the LOD level of the texture to read (in case of Mip Maps)\r\n * @param buffer defines a user defined buffer to fill with data (can be null)\r\n * @param flushRenderer true to flush the renderer from the pending commands before reading the pixels\r\n * @param noDataConversion false to convert the data to Uint8Array (if texture type is UNSIGNED_BYTE) or to Float32Array (if texture type is anything but UNSIGNED_BYTE). If true, the type of the generated buffer (if buffer==null) will depend on the type of the texture\r\n * @param x defines the region x coordinates to start reading from (default to 0)\r\n * @param y defines the region y coordinates to start reading from (default to 0)\r\n * @param width defines the region width to read from (default to the texture size at level)\r\n * @param height defines the region width to read from (default to the texture size at level)\r\n * @returns The Array buffer promise containing the pixels data.\r\n */\r\n public readPixels(\r\n faceIndex = 0,\r\n level = 0,\r\n buffer: Nullable = null,\r\n flushRenderer = true,\r\n noDataConversion = false,\r\n x = 0,\r\n y = 0,\r\n width = Number.MAX_VALUE,\r\n height = Number.MAX_VALUE\r\n ): Nullable> {\r\n if (!this._texture) {\r\n return null;\r\n }\r\n\r\n const engine = this._getEngine();\r\n if (!engine) {\r\n return null;\r\n }\r\n\r\n const size = this.getSize();\r\n let maxWidth = size.width;\r\n let maxHeight = size.height;\r\n if (level !== 0) {\r\n maxWidth = maxWidth / Math.pow(2, level);\r\n maxHeight = maxHeight / Math.pow(2, level);\r\n maxWidth = Math.round(maxWidth);\r\n maxHeight = Math.round(maxHeight);\r\n }\r\n\r\n width = Math.min(maxWidth, width);\r\n height = Math.min(maxHeight, height);\r\n\r\n try {\r\n if (this._texture.isCube) {\r\n return engine._readTexturePixels(this._texture, width, height, faceIndex, level, buffer, flushRenderer, noDataConversion, x, y);\r\n }\r\n\r\n return engine._readTexturePixels(this._texture, width, height, -1, level, buffer, flushRenderer, noDataConversion, x, y);\r\n } catch (e) {\r\n return null;\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _readPixelsSync(faceIndex = 0, level = 0, buffer: Nullable = null, flushRenderer = true, noDataConversion = false): Nullable {\r\n if (!this._texture) {\r\n return null;\r\n }\r\n\r\n const size = this.getSize();\r\n let width = size.width;\r\n let height = size.height;\r\n\r\n const engine = this._getEngine();\r\n if (!engine) {\r\n return null;\r\n }\r\n\r\n if (level != 0) {\r\n width = width / Math.pow(2, level);\r\n height = height / Math.pow(2, level);\r\n\r\n width = Math.round(width);\r\n height = Math.round(height);\r\n }\r\n\r\n try {\r\n if (this._texture.isCube) {\r\n return engine._readTexturePixelsSync(this._texture, width, height, faceIndex, level, buffer, flushRenderer, noDataConversion);\r\n }\r\n\r\n return engine._readTexturePixelsSync(this._texture, width, height, -1, level, buffer, flushRenderer, noDataConversion);\r\n } catch (e) {\r\n return null;\r\n }\r\n }\r\n\r\n /** @internal */\r\n public get _lodTextureHigh(): Nullable {\r\n if (this._texture) {\r\n return this._texture._lodTextureHigh;\r\n }\r\n return null;\r\n }\r\n\r\n /** @internal */\r\n public get _lodTextureMid(): Nullable {\r\n if (this._texture) {\r\n return this._texture._lodTextureMid;\r\n }\r\n return null;\r\n }\r\n\r\n /** @internal */\r\n public get _lodTextureLow(): Nullable {\r\n if (this._texture) {\r\n return this._texture._lodTextureLow;\r\n }\r\n return null;\r\n }\r\n\r\n /**\r\n * Dispose the texture and release its associated resources.\r\n */\r\n public dispose(): void {\r\n if (this._scene) {\r\n // Animations\r\n if (this._scene.stopAnimation) {\r\n this._scene.stopAnimation(this);\r\n }\r\n\r\n // Remove from scene\r\n this._scene.removePendingData(this);\r\n const index = this._scene.textures.indexOf(this);\r\n\r\n if (index >= 0) {\r\n this._scene.textures.splice(index, 1);\r\n }\r\n this._scene.onTextureRemovedObservable.notifyObservers(this);\r\n this._scene = null;\r\n\r\n if (this._parentContainer) {\r\n const index = this._parentContainer.textures.indexOf(this);\r\n if (index > -1) {\r\n this._parentContainer.textures.splice(index, 1);\r\n }\r\n this._parentContainer = null;\r\n }\r\n }\r\n\r\n // Callback\r\n this.onDisposeObservable.notifyObservers(this);\r\n this.onDisposeObservable.clear();\r\n\r\n this.metadata = null;\r\n\r\n super.dispose();\r\n }\r\n\r\n /**\r\n * Serialize the texture into a JSON representation that can be parsed later on.\r\n * @param allowEmptyName True to force serialization even if name is empty. Default: false\r\n * @returns the JSON representation of the texture\r\n */\r\n public serialize(allowEmptyName = false): any {\r\n if (!this.name && !allowEmptyName) {\r\n return null;\r\n }\r\n\r\n const serializationObject = SerializationHelper.Serialize(this);\r\n\r\n // Animations\r\n SerializationHelper.AppendSerializedAnimations(this, serializationObject);\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Helper function to be called back once a list of texture contains only ready textures.\r\n * @param textures Define the list of textures to wait for\r\n * @param callback Define the callback triggered once the entire list will be ready\r\n */\r\n public static WhenAllReady(textures: BaseTexture[], callback: () => void): void {\r\n let numRemaining = textures.length;\r\n if (numRemaining === 0) {\r\n callback();\r\n return;\r\n }\r\n\r\n for (let i = 0; i < textures.length; i++) {\r\n const texture = textures[i];\r\n\r\n if (texture.isReady()) {\r\n if (--numRemaining === 0) {\r\n callback();\r\n }\r\n } else {\r\n const onLoadObservable = (texture as any).onLoadObservable as Observable;\r\n\r\n if (onLoadObservable) {\r\n onLoadObservable.addOnce(() => {\r\n if (--numRemaining === 0) {\r\n callback();\r\n }\r\n });\r\n } else {\r\n if (--numRemaining === 0) {\r\n callback();\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n private static _IsScene(sceneOrEngine: Scene | ThinEngine): sceneOrEngine is Scene {\r\n return sceneOrEngine.getClassName() === \"Scene\";\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport type { ISize } from \"../Maths/math.size\";\r\nimport type { Nullable } from \"../types\";\r\n\r\nimport type { BaseTexture } from \"../Materials/Textures/baseTexture\";\r\n\r\n/**\r\n * Transform some pixel data to a base64 string\r\n * @param pixels defines the pixel data to transform to base64\r\n * @param size defines the width and height of the (texture) data\r\n * @param invertY true if the data must be inverted for the Y coordinate during the conversion\r\n * @returns The base64 encoded string or null\r\n */\r\nexport function GenerateBase64StringFromPixelData(pixels: ArrayBufferView, size: ISize, invertY = false): Nullable {\r\n const width = size.width;\r\n const height = size.height;\r\n\r\n if (pixels instanceof Float32Array) {\r\n let len = pixels.byteLength / pixels.BYTES_PER_ELEMENT;\r\n const npixels = new Uint8Array(len);\r\n\r\n while (--len >= 0) {\r\n let val = pixels[len];\r\n if (val < 0) {\r\n val = 0;\r\n } else if (val > 1) {\r\n val = 1;\r\n }\r\n npixels[len] = val * 255;\r\n }\r\n\r\n pixels = npixels;\r\n }\r\n\r\n const canvas = document.createElement(\"canvas\");\r\n canvas.width = width;\r\n canvas.height = height;\r\n\r\n const ctx = canvas.getContext(\"2d\");\r\n if (!ctx) {\r\n return null;\r\n }\r\n\r\n const imageData = ctx.createImageData(width, height);\r\n const castData = imageData.data;\r\n castData.set(pixels);\r\n ctx.putImageData(imageData, 0, 0);\r\n\r\n if (invertY) {\r\n const canvas2 = document.createElement(\"canvas\");\r\n canvas2.width = width;\r\n canvas2.height = height;\r\n\r\n const ctx2 = canvas2.getContext(\"2d\");\r\n if (!ctx2) {\r\n return null;\r\n }\r\n\r\n ctx2.translate(0, height);\r\n ctx2.scale(1, -1);\r\n ctx2.drawImage(canvas, 0, 0);\r\n\r\n return canvas2.toDataURL(\"image/png\");\r\n }\r\n\r\n return canvas.toDataURL(\"image/png\");\r\n}\r\n\r\n/**\r\n * Reads the pixels stored in the webgl texture and returns them as a base64 string\r\n * @param texture defines the texture to read pixels from\r\n * @param faceIndex defines the face of the texture to read (in case of cube texture)\r\n * @param level defines the LOD level of the texture to read (in case of Mip Maps)\r\n * @returns The base64 encoded string or null\r\n */\r\nexport function GenerateBase64StringFromTexture(texture: BaseTexture, faceIndex = 0, level = 0): Nullable {\r\n const internalTexture = texture.getInternalTexture();\r\n if (!internalTexture) {\r\n return null;\r\n }\r\n\r\n const pixels = texture._readPixelsSync(faceIndex, level);\r\n if (!pixels) {\r\n return null;\r\n }\r\n\r\n return GenerateBase64StringFromPixelData(pixels, texture.getSize(), internalTexture.invertY);\r\n}\r\n\r\n/**\r\n * Reads the pixels stored in the webgl texture and returns them as a base64 string\r\n * @param texture defines the texture to read pixels from\r\n * @param faceIndex defines the face of the texture to read (in case of cube texture)\r\n * @param level defines the LOD level of the texture to read (in case of Mip Maps)\r\n * @returns The base64 encoded string or null wrapped in a promise\r\n */\r\nexport async function GenerateBase64StringFromTextureAsync(texture: BaseTexture, faceIndex = 0, level = 0): Promise> {\r\n const internalTexture = texture.getInternalTexture();\r\n if (!internalTexture) {\r\n return null;\r\n }\r\n\r\n const pixels = await texture.readPixels(faceIndex, level);\r\n if (!pixels) {\r\n return null;\r\n }\r\n\r\n return GenerateBase64StringFromPixelData(pixels, texture.getSize(), internalTexture.invertY);\r\n}\r\n\r\n/**\r\n * Class used to host copy specific utilities\r\n * (Back-compat)\r\n */\r\nexport const CopyTools = {\r\n /**\r\n * Transform some pixel data to a base64 string\r\n * @param pixels defines the pixel data to transform to base64\r\n * @param size defines the width and height of the (texture) data\r\n * @param invertY true if the data must be inverted for the Y coordinate during the conversion\r\n * @returns The base64 encoded string or null\r\n */\r\n GenerateBase64StringFromPixelData,\r\n\r\n /**\r\n * Reads the pixels stored in the webgl texture and returns them as a base64 string\r\n * @param texture defines the texture to read pixels from\r\n * @param faceIndex defines the face of the texture to read (in case of cube texture)\r\n * @param level defines the LOD level of the texture to read (in case of Mip Maps)\r\n * @returns The base64 encoded string or null\r\n */\r\n GenerateBase64StringFromTexture,\r\n\r\n /**\r\n * Reads the pixels stored in the webgl texture and returns them as a base64 string\r\n * @param texture defines the texture to read pixels from\r\n * @param faceIndex defines the face of the texture to read (in case of cube texture)\r\n * @param level defines the LOD level of the texture to read (in case of Mip Maps)\r\n * @returns The base64 encoded string or null wrapped in a promise\r\n */\r\n GenerateBase64StringFromTextureAsync,\r\n};\r\n","import { serialize, SerializationHelper } from \"../../Misc/decorators\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport type { Nullable } from \"../../types\";\r\nimport { Matrix, TmpVectors, Vector3 } from \"../../Maths/math.vector\";\r\nimport { BaseTexture } from \"../../Materials/Textures/baseTexture\";\r\nimport { Constants } from \"../../Engines/constants\";\r\nimport { GetClass, RegisterClass } from \"../../Misc/typeStore\";\r\nimport { _WarnImport } from \"../../Misc/devTools\";\r\nimport type { IInspectable } from \"../../Misc/iInspectable\";\r\nimport type { ThinEngine } from \"../../Engines/thinEngine\";\r\nimport { TimingTools } from \"../../Misc/timingTools\";\r\nimport { InstantiationTools } from \"../../Misc/instantiationTools\";\r\nimport { Plane } from \"../../Maths/math.plane\";\r\nimport { EncodeArrayBufferToBase64 } from \"../../Misc/stringTools\";\r\nimport { GenerateBase64StringFromTexture, GenerateBase64StringFromTextureAsync } from \"../../Misc/copyTools\";\r\nimport { CompatibilityOptions } from \"../../Compat/compatibilityOptions\";\r\nimport type { InternalTexture } from \"./internalTexture\";\r\n\r\nimport type { CubeTexture } from \"../../Materials/Textures/cubeTexture\";\r\nimport type { MirrorTexture } from \"../../Materials/Textures/mirrorTexture\";\r\nimport type { RenderTargetTexture } from \"../../Materials/Textures/renderTargetTexture\";\r\nimport type { Scene } from \"../../scene\";\r\n\r\n/**\r\n * Defines the available options when creating a texture\r\n */\r\nexport interface ITextureCreationOptions {\r\n /** Defines if the texture will require mip maps or not (default: false) */\r\n noMipmap?: boolean;\r\n\r\n /** Defines if the texture needs to be inverted on the y axis during loading (default: true) */\r\n invertY?: boolean;\r\n\r\n /** Defines the sampling mode we want for the texture while fetching from it (Texture.NEAREST_SAMPLINGMODE...) (default: Texture.TRILINEAR_SAMPLINGMODE) */\r\n samplingMode?: number;\r\n\r\n /** Defines a callback triggered when the texture has been loaded (default: null) */\r\n onLoad?: Nullable<() => void>;\r\n\r\n /** Defines a callback triggered when an error occurred during the loading session (default: null) */\r\n onError?: Nullable<(message?: string, exception?: any) => void>;\r\n\r\n /** Defines the buffer to load the texture from in case the texture is loaded from a buffer representation (default: null) */\r\n buffer?: Nullable;\r\n\r\n /** Defines if the buffer we are loading the texture from should be deleted after load (default: false) */\r\n deleteBuffer?: boolean;\r\n\r\n /** Defines the format of the texture we are trying to load (Engine.TEXTUREFORMAT_RGBA...) (default: ) */\r\n format?: number;\r\n\r\n /** Defines an optional mime type information (default: undefined) */\r\n mimeType?: string;\r\n\r\n /** Options to be passed to the loader (default: undefined) */\r\n loaderOptions?: any;\r\n\r\n /** Specific flags to use when creating the texture (Constants.TEXTURE_CREATIONFLAG_STORAGE for storage textures, for eg) (default: undefined) */\r\n creationFlags?: number;\r\n\r\n /** Defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU) (default: false) */\r\n useSRGBBuffer?: boolean;\r\n\r\n /** Defines the underlying texture from an already existing one */\r\n internalTexture?: InternalTexture;\r\n}\r\n\r\n/**\r\n * This represents a texture in babylon. It can be easily loaded from a network, base64 or html input.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/materials_introduction#texture\r\n */\r\nexport class Texture extends BaseTexture {\r\n /**\r\n * Gets or sets a general boolean used to indicate that textures containing direct data (buffers) must be saved as part of the serialization process\r\n */\r\n public static SerializeBuffers = true;\r\n\r\n /**\r\n * Gets or sets a general boolean used to indicate that texture buffers must be saved as part of the serialization process.\r\n * If no buffer exists, one will be created as base64 string from the internal webgl data.\r\n */\r\n public static ForceSerializeBuffers = false;\r\n\r\n /**\r\n * This observable will notify when any texture had a loading error\r\n */\r\n public static OnTextureLoadErrorObservable = new Observable();\r\n\r\n /** @internal */\r\n public static _SerializeInternalTextureUniqueId = false;\r\n\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public static _CubeTextureParser = (jsonTexture: any, scene: Scene, rootUrl: string): CubeTexture => {\r\n throw _WarnImport(\"CubeTexture\");\r\n };\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public static _CreateMirror = (name: string, renderTargetSize: number, scene: Scene, generateMipMaps: boolean): MirrorTexture => {\r\n throw _WarnImport(\"MirrorTexture\");\r\n };\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public static _CreateRenderTargetTexture = (name: string, renderTargetSize: number, scene: Scene, generateMipMaps: boolean, creationFlags?: number): RenderTargetTexture => {\r\n throw _WarnImport(\"RenderTargetTexture\");\r\n };\r\n\r\n /** nearest is mag = nearest and min = nearest and no mip */\r\n public static readonly NEAREST_SAMPLINGMODE = Constants.TEXTURE_NEAREST_SAMPLINGMODE;\r\n /** nearest is mag = nearest and min = nearest and mip = linear */\r\n public static readonly NEAREST_NEAREST_MIPLINEAR = Constants.TEXTURE_NEAREST_NEAREST_MIPLINEAR; // nearest is mag = nearest and min = nearest and mip = linear\r\n\r\n /** Bilinear is mag = linear and min = linear and no mip */\r\n public static readonly BILINEAR_SAMPLINGMODE = Constants.TEXTURE_BILINEAR_SAMPLINGMODE;\r\n /** Bilinear is mag = linear and min = linear and mip = nearest */\r\n public static readonly LINEAR_LINEAR_MIPNEAREST = Constants.TEXTURE_LINEAR_LINEAR_MIPNEAREST; // Bilinear is mag = linear and min = linear and mip = nearest\r\n\r\n /** Trilinear is mag = linear and min = linear and mip = linear */\r\n public static readonly TRILINEAR_SAMPLINGMODE = Constants.TEXTURE_TRILINEAR_SAMPLINGMODE;\r\n /** Trilinear is mag = linear and min = linear and mip = linear */\r\n public static readonly LINEAR_LINEAR_MIPLINEAR = Constants.TEXTURE_LINEAR_LINEAR_MIPLINEAR; // Trilinear is mag = linear and min = linear and mip = linear\r\n\r\n /** mag = nearest and min = nearest and mip = nearest */\r\n public static readonly NEAREST_NEAREST_MIPNEAREST = Constants.TEXTURE_NEAREST_NEAREST_MIPNEAREST;\r\n /** mag = nearest and min = linear and mip = nearest */\r\n public static readonly NEAREST_LINEAR_MIPNEAREST = Constants.TEXTURE_NEAREST_LINEAR_MIPNEAREST;\r\n /** mag = nearest and min = linear and mip = linear */\r\n public static readonly NEAREST_LINEAR_MIPLINEAR = Constants.TEXTURE_NEAREST_LINEAR_MIPLINEAR;\r\n /** mag = nearest and min = linear and mip = none */\r\n public static readonly NEAREST_LINEAR = Constants.TEXTURE_NEAREST_LINEAR;\r\n /** mag = nearest and min = nearest and mip = none */\r\n public static readonly NEAREST_NEAREST = Constants.TEXTURE_NEAREST_NEAREST;\r\n /** mag = linear and min = nearest and mip = nearest */\r\n public static readonly LINEAR_NEAREST_MIPNEAREST = Constants.TEXTURE_LINEAR_NEAREST_MIPNEAREST;\r\n /** mag = linear and min = nearest and mip = linear */\r\n public static readonly LINEAR_NEAREST_MIPLINEAR = Constants.TEXTURE_LINEAR_NEAREST_MIPLINEAR;\r\n /** mag = linear and min = linear and mip = none */\r\n public static readonly LINEAR_LINEAR = Constants.TEXTURE_LINEAR_LINEAR;\r\n /** mag = linear and min = nearest and mip = none */\r\n public static readonly LINEAR_NEAREST = Constants.TEXTURE_LINEAR_NEAREST;\r\n\r\n /** Explicit coordinates mode */\r\n public static readonly EXPLICIT_MODE = Constants.TEXTURE_EXPLICIT_MODE;\r\n /** Spherical coordinates mode */\r\n public static readonly SPHERICAL_MODE = Constants.TEXTURE_SPHERICAL_MODE;\r\n /** Planar coordinates mode */\r\n public static readonly PLANAR_MODE = Constants.TEXTURE_PLANAR_MODE;\r\n /** Cubic coordinates mode */\r\n public static readonly CUBIC_MODE = Constants.TEXTURE_CUBIC_MODE;\r\n /** Projection coordinates mode */\r\n public static readonly PROJECTION_MODE = Constants.TEXTURE_PROJECTION_MODE;\r\n /** Inverse Cubic coordinates mode */\r\n public static readonly SKYBOX_MODE = Constants.TEXTURE_SKYBOX_MODE;\r\n /** Inverse Cubic coordinates mode */\r\n public static readonly INVCUBIC_MODE = Constants.TEXTURE_INVCUBIC_MODE;\r\n /** Equirectangular coordinates mode */\r\n public static readonly EQUIRECTANGULAR_MODE = Constants.TEXTURE_EQUIRECTANGULAR_MODE;\r\n /** Equirectangular Fixed coordinates mode */\r\n public static readonly FIXED_EQUIRECTANGULAR_MODE = Constants.TEXTURE_FIXED_EQUIRECTANGULAR_MODE;\r\n /** Equirectangular Fixed Mirrored coordinates mode */\r\n public static readonly FIXED_EQUIRECTANGULAR_MIRRORED_MODE = Constants.TEXTURE_FIXED_EQUIRECTANGULAR_MIRRORED_MODE;\r\n\r\n /** Texture is not repeating outside of 0..1 UVs */\r\n public static readonly CLAMP_ADDRESSMODE = Constants.TEXTURE_CLAMP_ADDRESSMODE;\r\n /** Texture is repeating outside of 0..1 UVs */\r\n public static readonly WRAP_ADDRESSMODE = Constants.TEXTURE_WRAP_ADDRESSMODE;\r\n /** Texture is repeating and mirrored */\r\n public static readonly MIRROR_ADDRESSMODE = Constants.TEXTURE_MIRROR_ADDRESSMODE;\r\n\r\n /**\r\n * Gets or sets a boolean which defines if the texture url must be build from the serialized URL instead of just using the name and loading them side by side with the scene file\r\n */\r\n public static UseSerializedUrlIfAny = false;\r\n\r\n /**\r\n * Define the url of the texture.\r\n */\r\n @serialize()\r\n public url: Nullable = null;\r\n\r\n /**\r\n * Define an offset on the texture to offset the u coordinates of the UVs\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/moreMaterials#offsetting\r\n */\r\n @serialize()\r\n public uOffset = 0;\r\n\r\n /**\r\n * Define an offset on the texture to offset the v coordinates of the UVs\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/moreMaterials#offsetting\r\n */\r\n @serialize()\r\n public vOffset = 0;\r\n\r\n /**\r\n * Define an offset on the texture to scale the u coordinates of the UVs\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/moreMaterials#tiling\r\n */\r\n @serialize()\r\n public uScale = 1.0;\r\n\r\n /**\r\n * Define an offset on the texture to scale the v coordinates of the UVs\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/moreMaterials#tiling\r\n */\r\n @serialize()\r\n public vScale = 1.0;\r\n\r\n /**\r\n * Define an offset on the texture to rotate around the u coordinates of the UVs\r\n * The angle is defined in radians.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/moreMaterials\r\n */\r\n @serialize()\r\n public uAng = 0;\r\n\r\n /**\r\n * Define an offset on the texture to rotate around the v coordinates of the UVs\r\n * The angle is defined in radians.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/moreMaterials\r\n */\r\n @serialize()\r\n public vAng = 0;\r\n\r\n /**\r\n * Define an offset on the texture to rotate around the w coordinates of the UVs (in case of 3d texture)\r\n * The angle is defined in radians.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/moreMaterials\r\n */\r\n @serialize()\r\n public wAng = 0;\r\n\r\n /**\r\n * Defines the center of rotation (U)\r\n */\r\n @serialize()\r\n public uRotationCenter = 0.5;\r\n\r\n /**\r\n * Defines the center of rotation (V)\r\n */\r\n @serialize()\r\n public vRotationCenter = 0.5;\r\n\r\n /**\r\n * Defines the center of rotation (W)\r\n */\r\n @serialize()\r\n public wRotationCenter = 0.5;\r\n\r\n /**\r\n * Sets this property to true to avoid deformations when rotating the texture with non-uniform scaling\r\n */\r\n @serialize()\r\n public homogeneousRotationInUVTransform = false;\r\n\r\n /**\r\n * Are mip maps generated for this texture or not.\r\n */\r\n get noMipmap(): boolean {\r\n return this._noMipmap;\r\n }\r\n\r\n /**\r\n * List of inspectable custom properties (used by the Inspector)\r\n * @see https://doc.babylonjs.com/toolsAndResources/inspector#extensibility\r\n */\r\n public inspectableCustomProperties: Nullable = null;\r\n\r\n /** @internal */\r\n public _noMipmap: boolean = false;\r\n /** @internal */\r\n public _invertY: boolean = false;\r\n private _rowGenerationMatrix: Nullable = null;\r\n private _cachedTextureMatrix: Nullable = null;\r\n private _projectionModeMatrix: Nullable = null;\r\n private _t0: Nullable = null;\r\n private _t1: Nullable = null;\r\n private _t2: Nullable = null;\r\n\r\n private _cachedUOffset: number = -1;\r\n private _cachedVOffset: number = -1;\r\n private _cachedUScale: number = 0;\r\n private _cachedVScale: number = 0;\r\n private _cachedUAng: number = -1;\r\n private _cachedVAng: number = -1;\r\n private _cachedWAng: number = -1;\r\n private _cachedReflectionProjectionMatrixId: number = -1;\r\n private _cachedURotationCenter: number = -1;\r\n private _cachedVRotationCenter: number = -1;\r\n private _cachedWRotationCenter: number = -1;\r\n private _cachedHomogeneousRotationInUVTransform: boolean = false;\r\n\r\n private _cachedReflectionTextureMatrix: Nullable = null;\r\n private _cachedReflectionUOffset = -1;\r\n private _cachedReflectionVOffset = -1;\r\n private _cachedReflectionUScale = 0;\r\n private _cachedReflectionVScale = 0;\r\n private _cachedReflectionCoordinatesMode = -1;\r\n\r\n /** @internal */\r\n public _buffer: Nullable = null;\r\n private _deleteBuffer: boolean = false;\r\n protected _format: Nullable = null;\r\n private _delayedOnLoad: Nullable<() => void> = null;\r\n private _delayedOnError: Nullable<() => void> = null;\r\n private _mimeType?: string;\r\n private _loaderOptions?: any;\r\n private _creationFlags?: number;\r\n /** @internal */\r\n public _useSRGBBuffer?: boolean;\r\n private _forcedExtension?: string;\r\n\r\n /** Returns the texture mime type if it was defined by a loader (undefined else) */\r\n public get mimeType() {\r\n return this._mimeType;\r\n }\r\n\r\n /**\r\n * Observable triggered once the texture has been loaded.\r\n */\r\n public onLoadObservable: Observable = new Observable();\r\n\r\n protected _isBlocking: boolean = true;\r\n /**\r\n * Is the texture preventing material to render while loading.\r\n * If false, a default texture will be used instead of the loading one during the preparation step.\r\n */\r\n public set isBlocking(value: boolean) {\r\n this._isBlocking = value;\r\n }\r\n @serialize()\r\n public get isBlocking(): boolean {\r\n return this._isBlocking;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if the texture needs to be inverted on the y axis during loading\r\n */\r\n public get invertY(): boolean {\r\n return this._invertY;\r\n }\r\n\r\n /**\r\n * Instantiates a new texture.\r\n * This represents a texture in babylon. It can be easily loaded from a network, base64 or html input.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/materials_introduction#texture\r\n * @param url defines the url of the picture to load as a texture\r\n * @param sceneOrEngine defines the scene or engine the texture will belong to\r\n * @param noMipmapOrOptions defines if the texture will require mip maps or not or set of all options to create the texture\r\n * @param invertY defines if the texture needs to be inverted on the y axis during loading\r\n * @param samplingMode defines the sampling mode we want for the texture while fetching from it (Texture.NEAREST_SAMPLINGMODE...)\r\n * @param onLoad defines a callback triggered when the texture has been loaded\r\n * @param onError defines a callback triggered when an error occurred during the loading session\r\n * @param buffer defines the buffer to load the texture from in case the texture is loaded from a buffer representation\r\n * @param deleteBuffer defines if the buffer we are loading the texture from should be deleted after load\r\n * @param format defines the format of the texture we are trying to load (Engine.TEXTUREFORMAT_RGBA...)\r\n * @param mimeType defines an optional mime type information\r\n * @param loaderOptions options to be passed to the loader\r\n * @param creationFlags specific flags to use when creating the texture (Constants.TEXTURE_CREATIONFLAG_STORAGE for storage textures, for eg)\r\n * @param forcedExtension defines the extension to use to pick the right loader\r\n */\r\n constructor(\r\n url: Nullable,\r\n sceneOrEngine?: Nullable,\r\n noMipmapOrOptions?: boolean | ITextureCreationOptions,\r\n invertY?: boolean,\r\n samplingMode: number = Texture.TRILINEAR_SAMPLINGMODE,\r\n onLoad: Nullable<() => void> = null,\r\n onError: Nullable<(message?: string, exception?: any) => void> = null,\r\n buffer: Nullable = null,\r\n deleteBuffer: boolean = false,\r\n format?: number,\r\n mimeType?: string,\r\n loaderOptions?: any,\r\n creationFlags?: number,\r\n forcedExtension?: string\r\n ) {\r\n super(sceneOrEngine);\r\n\r\n this.name = url || \"\";\r\n this.url = url;\r\n\r\n let noMipmap: boolean;\r\n let useSRGBBuffer: boolean = false;\r\n let internalTexture: Nullable = null;\r\n\r\n if (typeof noMipmapOrOptions === \"object\" && noMipmapOrOptions !== null) {\r\n noMipmap = noMipmapOrOptions.noMipmap ?? false;\r\n invertY = noMipmapOrOptions.invertY ?? (CompatibilityOptions.UseOpenGLOrientationForUV ? false : true);\r\n samplingMode = noMipmapOrOptions.samplingMode ?? Texture.TRILINEAR_SAMPLINGMODE;\r\n onLoad = noMipmapOrOptions.onLoad ?? null;\r\n onError = noMipmapOrOptions.onError ?? null;\r\n buffer = noMipmapOrOptions.buffer ?? null;\r\n deleteBuffer = noMipmapOrOptions.deleteBuffer ?? false;\r\n format = noMipmapOrOptions.format;\r\n mimeType = noMipmapOrOptions.mimeType;\r\n loaderOptions = noMipmapOrOptions.loaderOptions;\r\n creationFlags = noMipmapOrOptions.creationFlags;\r\n useSRGBBuffer = noMipmapOrOptions.useSRGBBuffer ?? false;\r\n internalTexture = noMipmapOrOptions.internalTexture ?? null;\r\n } else {\r\n noMipmap = !!noMipmapOrOptions;\r\n }\r\n\r\n this._noMipmap = noMipmap;\r\n this._invertY = invertY === undefined ? (CompatibilityOptions.UseOpenGLOrientationForUV ? false : true) : invertY;\r\n this._initialSamplingMode = samplingMode;\r\n this._buffer = buffer;\r\n this._deleteBuffer = deleteBuffer;\r\n this._mimeType = mimeType;\r\n this._loaderOptions = loaderOptions;\r\n this._creationFlags = creationFlags;\r\n this._useSRGBBuffer = useSRGBBuffer;\r\n this._forcedExtension = forcedExtension;\r\n if (format) {\r\n this._format = format;\r\n }\r\n\r\n const scene = this.getScene();\r\n const engine = this._getEngine();\r\n if (!engine) {\r\n return;\r\n }\r\n\r\n engine.onBeforeTextureInitObservable.notifyObservers(this);\r\n\r\n const load = () => {\r\n if (this._texture) {\r\n if (this._texture._invertVScale) {\r\n this.vScale *= -1;\r\n this.vOffset += 1;\r\n }\r\n\r\n // Update texture to match internal texture's wrapping\r\n if (this._texture._cachedWrapU !== null) {\r\n this.wrapU = this._texture._cachedWrapU;\r\n this._texture._cachedWrapU = null;\r\n }\r\n if (this._texture._cachedWrapV !== null) {\r\n this.wrapV = this._texture._cachedWrapV;\r\n this._texture._cachedWrapV = null;\r\n }\r\n if (this._texture._cachedWrapR !== null) {\r\n this.wrapR = this._texture._cachedWrapR;\r\n this._texture._cachedWrapR = null;\r\n }\r\n }\r\n\r\n if (this.onLoadObservable.hasObservers()) {\r\n this.onLoadObservable.notifyObservers(this);\r\n }\r\n if (onLoad) {\r\n onLoad();\r\n }\r\n\r\n if (!this.isBlocking && scene) {\r\n scene.resetCachedMaterial();\r\n }\r\n };\r\n\r\n const errorHandler = (message?: string, exception?: any) => {\r\n this._loadingError = true;\r\n this._errorObject = { message, exception };\r\n if (onError) {\r\n onError(message, exception);\r\n }\r\n Texture.OnTextureLoadErrorObservable.notifyObservers(this);\r\n };\r\n\r\n if (!this.url && !internalTexture) {\r\n this._delayedOnLoad = load;\r\n this._delayedOnError = errorHandler;\r\n return;\r\n }\r\n\r\n this._texture = internalTexture ?? this._getFromCache(this.url, noMipmap, samplingMode, this._invertY, useSRGBBuffer);\r\n\r\n if (!this._texture) {\r\n if (!scene || !scene.useDelayedTextureLoading) {\r\n try {\r\n this._texture = engine.createTexture(\r\n this.url,\r\n noMipmap,\r\n this._invertY,\r\n scene,\r\n samplingMode,\r\n load,\r\n errorHandler,\r\n this._buffer,\r\n undefined,\r\n this._format,\r\n this._forcedExtension,\r\n mimeType,\r\n loaderOptions,\r\n creationFlags,\r\n useSRGBBuffer\r\n );\r\n } catch (e) {\r\n errorHandler(\"error loading\", e);\r\n throw e;\r\n }\r\n if (deleteBuffer) {\r\n this._buffer = null;\r\n }\r\n } else {\r\n this.delayLoadState = Constants.DELAYLOADSTATE_NOTLOADED;\r\n\r\n this._delayedOnLoad = load;\r\n this._delayedOnError = errorHandler;\r\n }\r\n } else {\r\n if (this._texture.isReady) {\r\n TimingTools.SetImmediate(() => load());\r\n } else {\r\n const loadObserver = this._texture.onLoadedObservable.add(load);\r\n this._texture.onErrorObservable.add((e) => {\r\n errorHandler(e.message, e.exception);\r\n this._texture?.onLoadedObservable.remove(loadObserver);\r\n });\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Update the url (and optional buffer) of this texture if url was null during construction.\r\n * @param url the url of the texture\r\n * @param buffer the buffer of the texture (defaults to null)\r\n * @param onLoad callback called when the texture is loaded (defaults to null)\r\n * @param forcedExtension defines the extension to use to pick the right loader\r\n */\r\n public updateURL(\r\n url: string,\r\n buffer: Nullable = null,\r\n onLoad?: () => void,\r\n forcedExtension?: string\r\n ): void {\r\n if (this.url) {\r\n this.releaseInternalTexture();\r\n this.getScene()!.markAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n if (!this.name || this.name.startsWith(\"data:\")) {\r\n this.name = url;\r\n }\r\n this.url = url;\r\n this._buffer = buffer;\r\n this._forcedExtension = forcedExtension;\r\n this.delayLoadState = Constants.DELAYLOADSTATE_NOTLOADED;\r\n\r\n if (onLoad) {\r\n this._delayedOnLoad = onLoad;\r\n }\r\n this.delayLoad();\r\n }\r\n\r\n /**\r\n * Finish the loading sequence of a texture flagged as delayed load.\r\n * @internal\r\n */\r\n public delayLoad(): void {\r\n if (this.delayLoadState !== Constants.DELAYLOADSTATE_NOTLOADED) {\r\n return;\r\n }\r\n\r\n const scene = this.getScene();\r\n if (!scene) {\r\n return;\r\n }\r\n\r\n this.delayLoadState = Constants.DELAYLOADSTATE_LOADED;\r\n this._texture = this._getFromCache(this.url, this._noMipmap, this.samplingMode, this._invertY, this._useSRGBBuffer);\r\n\r\n if (!this._texture) {\r\n this._texture = scene\r\n .getEngine()\r\n .createTexture(\r\n this.url,\r\n this._noMipmap,\r\n this._invertY,\r\n scene,\r\n this.samplingMode,\r\n this._delayedOnLoad,\r\n this._delayedOnError,\r\n this._buffer,\r\n null,\r\n this._format,\r\n this._forcedExtension,\r\n this._mimeType,\r\n this._loaderOptions,\r\n this._creationFlags,\r\n this._useSRGBBuffer\r\n );\r\n if (this._deleteBuffer) {\r\n this._buffer = null;\r\n }\r\n } else {\r\n if (this._delayedOnLoad) {\r\n if (this._texture.isReady) {\r\n TimingTools.SetImmediate(this._delayedOnLoad);\r\n } else {\r\n this._texture.onLoadedObservable.add(this._delayedOnLoad);\r\n }\r\n }\r\n }\r\n\r\n this._delayedOnLoad = null;\r\n this._delayedOnError = null;\r\n }\r\n\r\n private _prepareRowForTextureGeneration(x: number, y: number, z: number, t: Vector3): void {\r\n x *= this._cachedUScale;\r\n y *= this._cachedVScale;\r\n\r\n x -= this.uRotationCenter * this._cachedUScale;\r\n y -= this.vRotationCenter * this._cachedVScale;\r\n z -= this.wRotationCenter;\r\n\r\n Vector3.TransformCoordinatesFromFloatsToRef(x, y, z, this._rowGenerationMatrix!, t);\r\n\r\n t.x += this.uRotationCenter * this._cachedUScale + this._cachedUOffset;\r\n t.y += this.vRotationCenter * this._cachedVScale + this._cachedVOffset;\r\n t.z += this.wRotationCenter;\r\n }\r\n\r\n /**\r\n * Checks if the texture has the same transform matrix than another texture\r\n * @param texture texture to check against\r\n * @returns true if the transforms are the same, else false\r\n */\r\n public checkTransformsAreIdentical(texture: Nullable): boolean {\r\n return (\r\n texture !== null &&\r\n this.uOffset === texture.uOffset &&\r\n this.vOffset === texture.vOffset &&\r\n this.uScale === texture.uScale &&\r\n this.vScale === texture.vScale &&\r\n this.uAng === texture.uAng &&\r\n this.vAng === texture.vAng &&\r\n this.wAng === texture.wAng\r\n );\r\n }\r\n\r\n /**\r\n * Get the current texture matrix which includes the requested offsetting, tiling and rotation components.\r\n * @param uBase\r\n * @returns the transform matrix of the texture.\r\n */\r\n public getTextureMatrix(uBase = 1): Matrix {\r\n if (\r\n this.uOffset === this._cachedUOffset &&\r\n this.vOffset === this._cachedVOffset &&\r\n this.uScale * uBase === this._cachedUScale &&\r\n this.vScale === this._cachedVScale &&\r\n this.uAng === this._cachedUAng &&\r\n this.vAng === this._cachedVAng &&\r\n this.wAng === this._cachedWAng &&\r\n this.uRotationCenter === this._cachedURotationCenter &&\r\n this.vRotationCenter === this._cachedVRotationCenter &&\r\n this.wRotationCenter === this._cachedWRotationCenter &&\r\n this.homogeneousRotationInUVTransform === this._cachedHomogeneousRotationInUVTransform\r\n ) {\r\n return this._cachedTextureMatrix!;\r\n }\r\n\r\n this._cachedUOffset = this.uOffset;\r\n this._cachedVOffset = this.vOffset;\r\n this._cachedUScale = this.uScale * uBase;\r\n this._cachedVScale = this.vScale;\r\n this._cachedUAng = this.uAng;\r\n this._cachedVAng = this.vAng;\r\n this._cachedWAng = this.wAng;\r\n this._cachedURotationCenter = this.uRotationCenter;\r\n this._cachedVRotationCenter = this.vRotationCenter;\r\n this._cachedWRotationCenter = this.wRotationCenter;\r\n this._cachedHomogeneousRotationInUVTransform = this.homogeneousRotationInUVTransform;\r\n\r\n if (!this._cachedTextureMatrix || !this._rowGenerationMatrix) {\r\n this._cachedTextureMatrix = Matrix.Zero();\r\n this._rowGenerationMatrix = new Matrix();\r\n this._t0 = Vector3.Zero();\r\n this._t1 = Vector3.Zero();\r\n this._t2 = Vector3.Zero();\r\n }\r\n\r\n Matrix.RotationYawPitchRollToRef(this.vAng, this.uAng, this.wAng, this._rowGenerationMatrix!);\r\n\r\n if (this.homogeneousRotationInUVTransform) {\r\n Matrix.TranslationToRef(-this._cachedURotationCenter, -this._cachedVRotationCenter, -this._cachedWRotationCenter, TmpVectors.Matrix[0]);\r\n Matrix.TranslationToRef(this._cachedURotationCenter, this._cachedVRotationCenter, this._cachedWRotationCenter, TmpVectors.Matrix[1]);\r\n Matrix.ScalingToRef(this._cachedUScale, this._cachedVScale, 0, TmpVectors.Matrix[2]);\r\n Matrix.TranslationToRef(this._cachedUOffset, this._cachedVOffset, 0, TmpVectors.Matrix[3]);\r\n\r\n TmpVectors.Matrix[0].multiplyToRef(this._rowGenerationMatrix!, this._cachedTextureMatrix);\r\n this._cachedTextureMatrix.multiplyToRef(TmpVectors.Matrix[1], this._cachedTextureMatrix);\r\n this._cachedTextureMatrix.multiplyToRef(TmpVectors.Matrix[2], this._cachedTextureMatrix);\r\n this._cachedTextureMatrix.multiplyToRef(TmpVectors.Matrix[3], this._cachedTextureMatrix);\r\n\r\n // copy the translation row to the 3rd row of the matrix so that we don't need to update the shaders (which expects the translation to be on the 3rd row)\r\n this._cachedTextureMatrix.setRowFromFloats(2, this._cachedTextureMatrix.m[12], this._cachedTextureMatrix.m[13], this._cachedTextureMatrix.m[14], 1);\r\n } else {\r\n this._prepareRowForTextureGeneration(0, 0, 0, this._t0!);\r\n this._prepareRowForTextureGeneration(1.0, 0, 0, this._t1!);\r\n this._prepareRowForTextureGeneration(0, 1.0, 0, this._t2!);\r\n\r\n this._t1!.subtractInPlace(this._t0!);\r\n this._t2!.subtractInPlace(this._t0!);\r\n\r\n Matrix.FromValuesToRef(\r\n this._t1!.x,\r\n this._t1!.y,\r\n this._t1!.z,\r\n 0.0,\r\n this._t2!.x,\r\n this._t2!.y,\r\n this._t2!.z,\r\n 0.0,\r\n this._t0!.x,\r\n this._t0!.y,\r\n this._t0!.z,\r\n 0.0,\r\n 0.0,\r\n 0.0,\r\n 0.0,\r\n 1.0,\r\n this._cachedTextureMatrix\r\n );\r\n }\r\n\r\n const scene = this.getScene();\r\n\r\n if (!scene) {\r\n return this._cachedTextureMatrix;\r\n }\r\n\r\n if (this.optimizeUVAllocation) {\r\n // We flag the materials that are using this texture as \"texture dirty\" because depending on the fact that the matrix is the identity or not, some defines\r\n // will get different values (see MaterialHelper.PrepareDefinesForMergedUV), meaning we should regenerate the effect accordingly\r\n scene.markAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag, (mat) => {\r\n return mat.hasTexture(this);\r\n });\r\n }\r\n\r\n return this._cachedTextureMatrix;\r\n }\r\n\r\n /**\r\n * Get the current matrix used to apply reflection. This is useful to rotate an environment texture for instance.\r\n * @returns The reflection texture transform\r\n */\r\n public getReflectionTextureMatrix(): Matrix {\r\n const scene = this.getScene();\r\n\r\n if (!scene) {\r\n return this._cachedReflectionTextureMatrix!;\r\n }\r\n\r\n if (\r\n this.uOffset === this._cachedReflectionUOffset &&\r\n this.vOffset === this._cachedReflectionVOffset &&\r\n this.uScale === this._cachedReflectionUScale &&\r\n this.vScale === this._cachedReflectionVScale &&\r\n this.coordinatesMode === this._cachedReflectionCoordinatesMode\r\n ) {\r\n if (this.coordinatesMode === Texture.PROJECTION_MODE) {\r\n if (this._cachedReflectionProjectionMatrixId === scene.getProjectionMatrix().updateFlag) {\r\n return this._cachedReflectionTextureMatrix!;\r\n }\r\n } else {\r\n return this._cachedReflectionTextureMatrix!;\r\n }\r\n }\r\n\r\n if (!this._cachedReflectionTextureMatrix) {\r\n this._cachedReflectionTextureMatrix = Matrix.Zero();\r\n }\r\n\r\n if (!this._projectionModeMatrix) {\r\n this._projectionModeMatrix = Matrix.Zero();\r\n }\r\n\r\n const flagMaterialsAsTextureDirty = this._cachedReflectionCoordinatesMode !== this.coordinatesMode;\r\n\r\n this._cachedReflectionUOffset = this.uOffset;\r\n this._cachedReflectionVOffset = this.vOffset;\r\n this._cachedReflectionUScale = this.uScale;\r\n this._cachedReflectionVScale = this.vScale;\r\n this._cachedReflectionCoordinatesMode = this.coordinatesMode;\r\n\r\n switch (this.coordinatesMode) {\r\n case Texture.PLANAR_MODE: {\r\n Matrix.IdentityToRef(this._cachedReflectionTextureMatrix);\r\n (this._cachedReflectionTextureMatrix)[0] = this.uScale;\r\n (this._cachedReflectionTextureMatrix)[5] = this.vScale;\r\n (this._cachedReflectionTextureMatrix)[12] = this.uOffset;\r\n (this._cachedReflectionTextureMatrix)[13] = this.vOffset;\r\n break;\r\n }\r\n case Texture.PROJECTION_MODE: {\r\n Matrix.FromValuesToRef(0.5, 0.0, 0.0, 0.0, 0.0, -0.5, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.5, 0.5, 1.0, 1.0, this._projectionModeMatrix);\r\n\r\n const projectionMatrix = scene.getProjectionMatrix();\r\n this._cachedReflectionProjectionMatrixId = projectionMatrix.updateFlag;\r\n projectionMatrix.multiplyToRef(this._projectionModeMatrix, this._cachedReflectionTextureMatrix);\r\n break;\r\n }\r\n default:\r\n Matrix.IdentityToRef(this._cachedReflectionTextureMatrix);\r\n break;\r\n }\r\n\r\n if (flagMaterialsAsTextureDirty) {\r\n // We flag the materials that are using this texture as \"texture dirty\" if the coordinatesMode has changed.\r\n // Indeed, this property is used to set the value of some defines used to generate the effect (in material.isReadyForSubMesh), so we must make sure this code will be re-executed and the effect recreated if necessary\r\n scene.markAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag, (mat) => {\r\n return mat.getActiveTextures().indexOf(this) !== -1;\r\n });\r\n }\r\n\r\n return this._cachedReflectionTextureMatrix;\r\n }\r\n\r\n /**\r\n * Clones the texture.\r\n * @returns the cloned texture\r\n */\r\n public clone(): Texture {\r\n const options: ITextureCreationOptions = {\r\n noMipmap: this._noMipmap,\r\n invertY: this._invertY,\r\n samplingMode: this.samplingMode,\r\n onLoad: undefined,\r\n onError: undefined,\r\n buffer: this._texture ? this._texture._buffer : undefined,\r\n deleteBuffer: this._deleteBuffer,\r\n format: this.textureFormat,\r\n mimeType: this.mimeType,\r\n loaderOptions: this._loaderOptions,\r\n creationFlags: this._creationFlags,\r\n useSRGBBuffer: this._useSRGBBuffer,\r\n };\r\n\r\n return SerializationHelper.Clone(() => {\r\n return new Texture(this._texture ? this._texture.url : null, this.getScene(), options);\r\n }, this);\r\n }\r\n\r\n /**\r\n * Serialize the texture to a JSON representation we can easily use in the respective Parse function.\r\n * @returns The JSON representation of the texture\r\n */\r\n public serialize(): any {\r\n const savedName = this.name;\r\n\r\n if (!Texture.SerializeBuffers) {\r\n if (this.name.startsWith(\"data:\")) {\r\n this.name = \"\";\r\n }\r\n }\r\n\r\n if (this.name.startsWith(\"data:\") && this.url === this.name) {\r\n this.url = \"\";\r\n }\r\n\r\n const serializationObject = super.serialize(Texture._SerializeInternalTextureUniqueId);\r\n\r\n if (!serializationObject) {\r\n return null;\r\n }\r\n\r\n if (Texture.SerializeBuffers || Texture.ForceSerializeBuffers) {\r\n if (typeof this._buffer === \"string\" && (this._buffer as string).substr(0, 5) === \"data:\") {\r\n serializationObject.base64String = this._buffer;\r\n serializationObject.name = serializationObject.name.replace(\"data:\", \"\");\r\n } else if (this.url && this.url.startsWith(\"data:\") && this._buffer instanceof Uint8Array) {\r\n serializationObject.base64String = \"data:image/png;base64,\" + EncodeArrayBufferToBase64(this._buffer);\r\n } else if (Texture.ForceSerializeBuffers || (this.url && this.url.startsWith(\"blob:\")) || this._forceSerialize) {\r\n serializationObject.base64String =\r\n !this._engine || this._engine._features.supportSyncTextureRead ? GenerateBase64StringFromTexture(this) : GenerateBase64StringFromTextureAsync(this);\r\n }\r\n }\r\n\r\n serializationObject.invertY = this._invertY;\r\n serializationObject.samplingMode = this.samplingMode;\r\n serializationObject._creationFlags = this._creationFlags;\r\n serializationObject._useSRGBBuffer = this._useSRGBBuffer;\r\n if (Texture._SerializeInternalTextureUniqueId) {\r\n serializationObject.internalTextureUniqueId = this._texture?.uniqueId ?? undefined;\r\n }\r\n\r\n this.name = savedName;\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Get the current class name of the texture useful for serialization or dynamic coding.\r\n * @returns \"Texture\"\r\n */\r\n public getClassName(): string {\r\n return \"Texture\";\r\n }\r\n\r\n /**\r\n * Dispose the texture and release its associated resources.\r\n */\r\n public dispose(): void {\r\n super.dispose();\r\n\r\n this.onLoadObservable.clear();\r\n\r\n this._delayedOnLoad = null;\r\n this._delayedOnError = null;\r\n this._buffer = null;\r\n }\r\n\r\n /**\r\n * Parse the JSON representation of a texture in order to recreate the texture in the given scene.\r\n * @param parsedTexture Define the JSON representation of the texture\r\n * @param scene Define the scene the parsed texture should be instantiated in\r\n * @param rootUrl Define the root url of the parsing sequence in the case of relative dependencies\r\n * @returns The parsed texture if successful\r\n */\r\n public static Parse(parsedTexture: any, scene: Scene, rootUrl: string): Nullable {\r\n if (parsedTexture.customType) {\r\n const customTexture = InstantiationTools.Instantiate(parsedTexture.customType);\r\n // Update Sampling Mode\r\n const parsedCustomTexture: any = customTexture.Parse(parsedTexture, scene, rootUrl);\r\n if (parsedTexture.samplingMode && parsedCustomTexture.updateSamplingMode && parsedCustomTexture._samplingMode) {\r\n if (parsedCustomTexture._samplingMode !== parsedTexture.samplingMode) {\r\n parsedCustomTexture.updateSamplingMode(parsedTexture.samplingMode);\r\n }\r\n }\r\n return parsedCustomTexture;\r\n }\r\n\r\n if (parsedTexture.isCube && !parsedTexture.isRenderTarget) {\r\n return Texture._CubeTextureParser(parsedTexture, scene, rootUrl);\r\n }\r\n\r\n const hasInternalTextureUniqueId = parsedTexture.internalTextureUniqueId !== undefined;\r\n\r\n if (!parsedTexture.name && !parsedTexture.isRenderTarget && !hasInternalTextureUniqueId) {\r\n return null;\r\n }\r\n\r\n let internalTexture: InternalTexture | undefined;\r\n\r\n if (hasInternalTextureUniqueId) {\r\n const cache = scene.getEngine().getLoadedTexturesCache();\r\n for (const texture of cache) {\r\n if (texture.uniqueId === parsedTexture.internalTextureUniqueId) {\r\n internalTexture = texture;\r\n break;\r\n }\r\n }\r\n }\r\n\r\n const onLoaded = (texture: Texture | null) => {\r\n // Clear cache\r\n if (texture && texture._texture) {\r\n texture._texture._cachedWrapU = null;\r\n texture._texture._cachedWrapV = null;\r\n texture._texture._cachedWrapR = null;\r\n }\r\n\r\n // Update Sampling Mode\r\n if (parsedTexture.samplingMode) {\r\n const sampling: number = parsedTexture.samplingMode;\r\n if (texture && texture.samplingMode !== sampling) {\r\n texture.updateSamplingMode(sampling);\r\n }\r\n }\r\n // Animations\r\n if (texture && parsedTexture.animations) {\r\n for (let animationIndex = 0; animationIndex < parsedTexture.animations.length; animationIndex++) {\r\n const parsedAnimation = parsedTexture.animations[animationIndex];\r\n const internalClass = GetClass(\"BABYLON.Animation\");\r\n if (internalClass) {\r\n texture.animations.push(internalClass.Parse(parsedAnimation));\r\n }\r\n }\r\n }\r\n\r\n if (hasInternalTextureUniqueId && !internalTexture) {\r\n texture?._texture?._setUniqueId(parsedTexture.internalTextureUniqueId);\r\n }\r\n };\r\n\r\n const texture = SerializationHelper.Parse(\r\n () => {\r\n let generateMipMaps: boolean = true;\r\n if (parsedTexture.noMipmap) {\r\n generateMipMaps = false;\r\n }\r\n if (parsedTexture.mirrorPlane) {\r\n const mirrorTexture = Texture._CreateMirror(parsedTexture.name, parsedTexture.renderTargetSize, scene, generateMipMaps);\r\n mirrorTexture._waitingRenderList = parsedTexture.renderList;\r\n mirrorTexture.mirrorPlane = Plane.FromArray(parsedTexture.mirrorPlane);\r\n onLoaded(mirrorTexture);\r\n return mirrorTexture;\r\n } else if (parsedTexture.isRenderTarget) {\r\n let renderTargetTexture: Nullable = null;\r\n if (parsedTexture.isCube) {\r\n // Search for an existing reflection probe (which contains a cube render target texture)\r\n if (scene.reflectionProbes) {\r\n for (let index = 0; index < scene.reflectionProbes.length; index++) {\r\n const probe = scene.reflectionProbes[index];\r\n if (probe.name === parsedTexture.name) {\r\n return probe.cubeTexture;\r\n }\r\n }\r\n }\r\n } else {\r\n renderTargetTexture = Texture._CreateRenderTargetTexture(\r\n parsedTexture.name,\r\n parsedTexture.renderTargetSize,\r\n scene,\r\n generateMipMaps,\r\n parsedTexture._creationFlags ?? 0\r\n );\r\n renderTargetTexture._waitingRenderList = parsedTexture.renderList;\r\n }\r\n onLoaded(renderTargetTexture);\r\n return renderTargetTexture;\r\n } else {\r\n let texture: Texture;\r\n\r\n if (parsedTexture.base64String && !internalTexture) {\r\n // name and url are the same to ensure caching happens from the actual base64 string\r\n texture = Texture.CreateFromBase64String(\r\n parsedTexture.base64String,\r\n parsedTexture.base64String,\r\n scene,\r\n !generateMipMaps,\r\n parsedTexture.invertY,\r\n parsedTexture.samplingMode,\r\n () => {\r\n onLoaded(texture);\r\n },\r\n parsedTexture._creationFlags ?? 0,\r\n parsedTexture._useSRGBBuffer ?? false\r\n );\r\n\r\n // prettier name to fit with the loaded data\r\n texture.name = parsedTexture.name;\r\n } else {\r\n let url: string;\r\n if (parsedTexture.name && (parsedTexture.name.indexOf(\"://\") > 0 || parsedTexture.name.startsWith(\"data:\"))) {\r\n url = parsedTexture.name;\r\n } else {\r\n url = rootUrl + parsedTexture.name;\r\n }\r\n\r\n if (parsedTexture.url && (parsedTexture.url.startsWith(\"data:\") || Texture.UseSerializedUrlIfAny)) {\r\n url = parsedTexture.url;\r\n }\r\n\r\n const options: ITextureCreationOptions = {\r\n noMipmap: !generateMipMaps,\r\n invertY: parsedTexture.invertY,\r\n samplingMode: parsedTexture.samplingMode,\r\n onLoad: () => {\r\n onLoaded(texture);\r\n },\r\n internalTexture,\r\n };\r\n\r\n texture = new Texture(url, scene, options);\r\n }\r\n\r\n return texture;\r\n }\r\n },\r\n parsedTexture,\r\n scene\r\n );\r\n\r\n return texture;\r\n }\r\n\r\n /**\r\n * Creates a texture from its base 64 representation.\r\n * @param data Define the base64 payload without the data: prefix\r\n * @param name Define the name of the texture in the scene useful fo caching purpose for instance\r\n * @param scene Define the scene the texture should belong to\r\n * @param noMipmapOrOptions defines if the texture will require mip maps or not or set of all options to create the texture\r\n * @param invertY define if the texture needs to be inverted on the y axis during loading\r\n * @param samplingMode define the sampling mode we want for the texture while fetching from it (Texture.NEAREST_SAMPLINGMODE...)\r\n * @param onLoad define a callback triggered when the texture has been loaded\r\n * @param onError define a callback triggered when an error occurred during the loading session\r\n * @param format define the format of the texture we are trying to load (Engine.TEXTUREFORMAT_RGBA...)\r\n * @param creationFlags specific flags to use when creating the texture (Constants.TEXTURE_CREATIONFLAG_STORAGE for storage textures, for eg)\r\n * @returns the created texture\r\n */\r\n public static CreateFromBase64String(\r\n data: string,\r\n name: string,\r\n scene: Scene,\r\n noMipmapOrOptions?: boolean | ITextureCreationOptions,\r\n invertY?: boolean,\r\n samplingMode: number = Texture.TRILINEAR_SAMPLINGMODE,\r\n onLoad: Nullable<() => void> = null,\r\n onError: Nullable<() => void> = null,\r\n format: number = Constants.TEXTUREFORMAT_RGBA,\r\n creationFlags?: number\r\n ): Texture {\r\n return new Texture(\"data:\" + name, scene, noMipmapOrOptions, invertY, samplingMode, onLoad, onError, data, false, format, undefined, undefined, creationFlags);\r\n }\r\n\r\n /**\r\n * Creates a texture from its data: representation. (data: will be added in case only the payload has been passed in)\r\n * @param name Define the name of the texture in the scene useful fo caching purpose for instance\r\n * @param buffer define the buffer to load the texture from in case the texture is loaded from a buffer representation\r\n * @param scene Define the scene the texture should belong to\r\n * @param deleteBuffer define if the buffer we are loading the texture from should be deleted after load\r\n * @param noMipmapOrOptions defines if the texture will require mip maps or not or set of all options to create the texture\r\n * @param invertY define if the texture needs to be inverted on the y axis during loading\r\n * @param samplingMode define the sampling mode we want for the texture while fetching from it (Texture.NEAREST_SAMPLINGMODE...)\r\n * @param onLoad define a callback triggered when the texture has been loaded\r\n * @param onError define a callback triggered when an error occurred during the loading session\r\n * @param format define the format of the texture we are trying to load (Engine.TEXTUREFORMAT_RGBA...)\r\n * @param creationFlags specific flags to use when creating the texture (Constants.TEXTURE_CREATIONFLAG_STORAGE for storage textures, for eg)\r\n * @returns the created texture\r\n */\r\n public static LoadFromDataString(\r\n name: string,\r\n buffer: any,\r\n scene: Scene,\r\n deleteBuffer: boolean = false,\r\n noMipmapOrOptions?: boolean | ITextureCreationOptions,\r\n invertY: boolean = true,\r\n samplingMode: number = Texture.TRILINEAR_SAMPLINGMODE,\r\n onLoad: Nullable<() => void> = null,\r\n onError: Nullable<(message?: string, exception?: any) => void> = null,\r\n format: number = Constants.TEXTUREFORMAT_RGBA,\r\n creationFlags?: number\r\n ): Texture {\r\n if (name.substr(0, 5) !== \"data:\") {\r\n name = \"data:\" + name;\r\n }\r\n\r\n return new Texture(name, scene, noMipmapOrOptions, invertY, samplingMode, onLoad, onError, buffer, deleteBuffer, format, undefined, undefined, creationFlags);\r\n }\r\n}\r\n\r\n// References the dependencies.\r\nRegisterClass(\"BABYLON.Texture\", Texture);\r\nSerializationHelper._TextureParser = Texture.Parse;\r\n","import type { Matrix } from \"../Maths/math.vector\";\r\nimport type { Mesh } from \"../Meshes/mesh\";\r\nimport type { Scene } from \"../scene\";\r\nimport type { Effect } from \"../Materials/effect\";\r\nimport { Constants } from \"../Engines/constants\";\r\n\r\n/**\r\n * Configuration needed for prepass-capable materials\r\n */\r\nexport class PrePassConfiguration {\r\n /**\r\n * Previous world matrices of meshes carrying this material\r\n * Used for computing velocity\r\n */\r\n public previousWorldMatrices: { [index: number]: Matrix } = {};\r\n /**\r\n * Previous view project matrix\r\n * Used for computing velocity\r\n */\r\n public previousViewProjection: Matrix;\r\n /**\r\n * Current view projection matrix\r\n * Used for computing velocity\r\n */\r\n public currentViewProjection: Matrix;\r\n /**\r\n * Previous bones of meshes carrying this material\r\n * Used for computing velocity\r\n */\r\n public previousBones: { [index: number]: Float32Array } = {};\r\n\r\n private _lastUpdateFrameId: number;\r\n\r\n /**\r\n * Add the required uniforms to the current list.\r\n * @param uniforms defines the current uniform list.\r\n */\r\n public static AddUniforms(uniforms: string[]): void {\r\n uniforms.push(\"previousWorld\", \"previousViewProjection\", \"mPreviousBones\");\r\n }\r\n\r\n /**\r\n * Add the required samplers to the current list.\r\n * @param samplers defines the current sampler list.\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public static AddSamplers(samplers: string[]): void {\r\n // pass\r\n }\r\n\r\n /**\r\n * Binds the material data.\r\n * @param effect defines the effect to update\r\n * @param scene defines the scene the material belongs to.\r\n * @param mesh The mesh\r\n * @param world World matrix of this mesh\r\n * @param isFrozen Is the material frozen\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public bindForSubMesh(effect: Effect, scene: Scene, mesh: Mesh, world: Matrix, isFrozen: boolean): void {\r\n if (scene.prePassRenderer && scene.prePassRenderer.enabled && scene.prePassRenderer.currentRTisSceneRT) {\r\n if (scene.prePassRenderer.getIndex(Constants.PREPASS_VELOCITY_TEXTURE_TYPE) !== -1) {\r\n if (!this.previousWorldMatrices[mesh.uniqueId]) {\r\n this.previousWorldMatrices[mesh.uniqueId] = world.clone();\r\n }\r\n\r\n if (!this.previousViewProjection) {\r\n this.previousViewProjection = scene.getTransformMatrix().clone();\r\n this.currentViewProjection = scene.getTransformMatrix().clone();\r\n }\r\n\r\n const engine = scene.getEngine();\r\n\r\n if (this.currentViewProjection.updateFlag !== scene.getTransformMatrix().updateFlag) {\r\n // First update of the prepass configuration for this rendering pass\r\n this._lastUpdateFrameId = engine.frameId;\r\n this.previousViewProjection.copyFrom(this.currentViewProjection);\r\n this.currentViewProjection.copyFrom(scene.getTransformMatrix());\r\n } else if (this._lastUpdateFrameId !== engine.frameId) {\r\n // The scene transformation did not change from the previous frame (so no camera motion), we must update previousViewProjection accordingly\r\n this._lastUpdateFrameId = engine.frameId;\r\n this.previousViewProjection.copyFrom(this.currentViewProjection);\r\n }\r\n\r\n effect.setMatrix(\"previousWorld\", this.previousWorldMatrices[mesh.uniqueId]);\r\n effect.setMatrix(\"previousViewProjection\", this.previousViewProjection);\r\n\r\n this.previousWorldMatrices[mesh.uniqueId] = world.clone();\r\n }\r\n }\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Matrix } from \"../Maths/math.vector\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { Mesh } from \"../Meshes/mesh\";\r\nimport { Material } from \"../Materials/material\";\r\nimport type { Effect } from \"../Materials/effect\";\r\nimport type { SubMesh } from \"../Meshes/subMesh\";\r\n/**\r\n * Base class of materials working in push mode in babylon JS\r\n * @internal\r\n */\r\nexport class PushMaterial extends Material {\r\n protected _activeEffect?: Effect;\r\n\r\n protected _normalMatrix: Matrix = new Matrix();\r\n\r\n constructor(name: string, scene?: Scene, storeEffectOnSubMeshes = true) {\r\n super(name, scene);\r\n this._storeEffectOnSubMeshes = storeEffectOnSubMeshes;\r\n }\r\n\r\n public getEffect(): Effect {\r\n return this._storeEffectOnSubMeshes ? this._activeEffect! : super.getEffect()!;\r\n }\r\n\r\n public isReady(mesh?: AbstractMesh, useInstances?: boolean): boolean {\r\n if (!mesh) {\r\n return false;\r\n }\r\n\r\n if (!this._storeEffectOnSubMeshes) {\r\n return true;\r\n }\r\n\r\n if (!mesh.subMeshes || mesh.subMeshes.length === 0) {\r\n return true;\r\n }\r\n\r\n return this.isReadyForSubMesh(mesh, mesh.subMeshes[0], useInstances);\r\n }\r\n\r\n protected _isReadyForSubMesh(subMesh: SubMesh) {\r\n const defines = subMesh.materialDefines;\r\n if (!this.checkReadyOnEveryCall && subMesh.effect && defines) {\r\n if (defines._renderId === this.getScene().getRenderId()) {\r\n return true;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Binds the given world matrix to the active effect\r\n *\r\n * @param world the matrix to bind\r\n */\r\n public bindOnlyWorldMatrix(world: Matrix): void {\r\n this._activeEffect!.setMatrix(\"world\", world);\r\n }\r\n\r\n /**\r\n * Binds the given normal matrix to the active effect\r\n *\r\n * @param normalMatrix the matrix to bind\r\n */\r\n public bindOnlyNormalMatrix(normalMatrix: Matrix): void {\r\n this._activeEffect!.setMatrix(\"normalMatrix\", normalMatrix);\r\n }\r\n\r\n public bind(world: Matrix, mesh?: Mesh): void {\r\n if (!mesh) {\r\n return;\r\n }\r\n\r\n this.bindForSubMesh(world, mesh, mesh.subMeshes[0]);\r\n }\r\n\r\n protected _afterBind(mesh?: Mesh, effect: Nullable = null): void {\r\n super._afterBind(mesh, effect);\r\n this.getScene()._cachedEffect = effect;\r\n if (effect) {\r\n effect._forceRebindOnNextCall = false;\r\n }\r\n }\r\n\r\n protected _mustRebind(scene: Scene, effect: Effect, visibility: number = 1) {\r\n return scene.isCachedMaterialInvalid(this, effect, visibility);\r\n }\r\n\r\n public dispose(forceDisposeEffect?: boolean, forceDisposeTextures?: boolean, notBoundToMesh?: boolean) {\r\n this._activeEffect = undefined;\r\n super.dispose(forceDisposeEffect, forceDisposeTextures, notBoundToMesh);\r\n }\r\n}\r\n","import { Engine } from \"../Engines/engine\";\r\nimport { Constants } from \"../Engines/constants\";\r\n\r\n/**\r\n * This groups all the flags used to control the materials channel.\r\n */\r\nexport class MaterialFlags {\r\n // Flags used to enable or disable a type of texture for all Standard Materials\r\n private static _DiffuseTextureEnabled = true;\r\n /**\r\n * Are diffuse textures enabled in the application.\r\n */\r\n public static get DiffuseTextureEnabled(): boolean {\r\n return this._DiffuseTextureEnabled;\r\n }\r\n public static set DiffuseTextureEnabled(value: boolean) {\r\n if (this._DiffuseTextureEnabled === value) {\r\n return;\r\n }\r\n\r\n this._DiffuseTextureEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private static _DetailTextureEnabled = true;\r\n /**\r\n * Are detail textures enabled in the application.\r\n */\r\n public static get DetailTextureEnabled(): boolean {\r\n return this._DetailTextureEnabled;\r\n }\r\n public static set DetailTextureEnabled(value: boolean) {\r\n if (this._DetailTextureEnabled === value) {\r\n return;\r\n }\r\n\r\n this._DetailTextureEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private static _DecalMapEnabled = true;\r\n /**\r\n * Are decal maps enabled in the application.\r\n */\r\n public static get DecalMapEnabled(): boolean {\r\n return this._DecalMapEnabled;\r\n }\r\n public static set DecalMapEnabled(value: boolean) {\r\n if (this._DecalMapEnabled === value) {\r\n return;\r\n }\r\n\r\n this._DecalMapEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private static _AmbientTextureEnabled = true;\r\n /**\r\n * Are ambient textures enabled in the application.\r\n */\r\n public static get AmbientTextureEnabled(): boolean {\r\n return this._AmbientTextureEnabled;\r\n }\r\n public static set AmbientTextureEnabled(value: boolean) {\r\n if (this._AmbientTextureEnabled === value) {\r\n return;\r\n }\r\n\r\n this._AmbientTextureEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private static _OpacityTextureEnabled = true;\r\n /**\r\n * Are opacity textures enabled in the application.\r\n */\r\n public static get OpacityTextureEnabled(): boolean {\r\n return this._OpacityTextureEnabled;\r\n }\r\n public static set OpacityTextureEnabled(value: boolean) {\r\n if (this._OpacityTextureEnabled === value) {\r\n return;\r\n }\r\n\r\n this._OpacityTextureEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private static _ReflectionTextureEnabled = true;\r\n /**\r\n * Are reflection textures enabled in the application.\r\n */\r\n public static get ReflectionTextureEnabled(): boolean {\r\n return this._ReflectionTextureEnabled;\r\n }\r\n public static set ReflectionTextureEnabled(value: boolean) {\r\n if (this._ReflectionTextureEnabled === value) {\r\n return;\r\n }\r\n\r\n this._ReflectionTextureEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private static _EmissiveTextureEnabled = true;\r\n /**\r\n * Are emissive textures enabled in the application.\r\n */\r\n public static get EmissiveTextureEnabled(): boolean {\r\n return this._EmissiveTextureEnabled;\r\n }\r\n public static set EmissiveTextureEnabled(value: boolean) {\r\n if (this._EmissiveTextureEnabled === value) {\r\n return;\r\n }\r\n\r\n this._EmissiveTextureEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private static _SpecularTextureEnabled = true;\r\n /**\r\n * Are specular textures enabled in the application.\r\n */\r\n public static get SpecularTextureEnabled(): boolean {\r\n return this._SpecularTextureEnabled;\r\n }\r\n public static set SpecularTextureEnabled(value: boolean) {\r\n if (this._SpecularTextureEnabled === value) {\r\n return;\r\n }\r\n\r\n this._SpecularTextureEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private static _BumpTextureEnabled = true;\r\n /**\r\n * Are bump textures enabled in the application.\r\n */\r\n public static get BumpTextureEnabled(): boolean {\r\n return this._BumpTextureEnabled;\r\n }\r\n public static set BumpTextureEnabled(value: boolean) {\r\n if (this._BumpTextureEnabled === value) {\r\n return;\r\n }\r\n\r\n this._BumpTextureEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private static _LightmapTextureEnabled = true;\r\n /**\r\n * Are lightmap textures enabled in the application.\r\n */\r\n public static get LightmapTextureEnabled(): boolean {\r\n return this._LightmapTextureEnabled;\r\n }\r\n public static set LightmapTextureEnabled(value: boolean) {\r\n if (this._LightmapTextureEnabled === value) {\r\n return;\r\n }\r\n\r\n this._LightmapTextureEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private static _RefractionTextureEnabled = true;\r\n /**\r\n * Are refraction textures enabled in the application.\r\n */\r\n public static get RefractionTextureEnabled(): boolean {\r\n return this._RefractionTextureEnabled;\r\n }\r\n public static set RefractionTextureEnabled(value: boolean) {\r\n if (this._RefractionTextureEnabled === value) {\r\n return;\r\n }\r\n\r\n this._RefractionTextureEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private static _ColorGradingTextureEnabled = true;\r\n /**\r\n * Are color grading textures enabled in the application.\r\n */\r\n public static get ColorGradingTextureEnabled(): boolean {\r\n return this._ColorGradingTextureEnabled;\r\n }\r\n public static set ColorGradingTextureEnabled(value: boolean) {\r\n if (this._ColorGradingTextureEnabled === value) {\r\n return;\r\n }\r\n\r\n this._ColorGradingTextureEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private static _FresnelEnabled = true;\r\n /**\r\n * Are fresnels enabled in the application.\r\n */\r\n public static get FresnelEnabled(): boolean {\r\n return this._FresnelEnabled;\r\n }\r\n public static set FresnelEnabled(value: boolean) {\r\n if (this._FresnelEnabled === value) {\r\n return;\r\n }\r\n\r\n this._FresnelEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_FresnelDirtyFlag);\r\n }\r\n\r\n private static _ClearCoatTextureEnabled = true;\r\n /**\r\n * Are clear coat textures enabled in the application.\r\n */\r\n public static get ClearCoatTextureEnabled(): boolean {\r\n return this._ClearCoatTextureEnabled;\r\n }\r\n public static set ClearCoatTextureEnabled(value: boolean) {\r\n if (this._ClearCoatTextureEnabled === value) {\r\n return;\r\n }\r\n\r\n this._ClearCoatTextureEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private static _ClearCoatBumpTextureEnabled = true;\r\n /**\r\n * Are clear coat bump textures enabled in the application.\r\n */\r\n public static get ClearCoatBumpTextureEnabled(): boolean {\r\n return this._ClearCoatBumpTextureEnabled;\r\n }\r\n public static set ClearCoatBumpTextureEnabled(value: boolean) {\r\n if (this._ClearCoatBumpTextureEnabled === value) {\r\n return;\r\n }\r\n\r\n this._ClearCoatBumpTextureEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private static _ClearCoatTintTextureEnabled = true;\r\n /**\r\n * Are clear coat tint textures enabled in the application.\r\n */\r\n public static get ClearCoatTintTextureEnabled(): boolean {\r\n return this._ClearCoatTintTextureEnabled;\r\n }\r\n public static set ClearCoatTintTextureEnabled(value: boolean) {\r\n if (this._ClearCoatTintTextureEnabled === value) {\r\n return;\r\n }\r\n\r\n this._ClearCoatTintTextureEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private static _SheenTextureEnabled = true;\r\n /**\r\n * Are sheen textures enabled in the application.\r\n */\r\n public static get SheenTextureEnabled(): boolean {\r\n return this._SheenTextureEnabled;\r\n }\r\n public static set SheenTextureEnabled(value: boolean) {\r\n if (this._SheenTextureEnabled === value) {\r\n return;\r\n }\r\n\r\n this._SheenTextureEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private static _AnisotropicTextureEnabled = true;\r\n /**\r\n * Are anisotropic textures enabled in the application.\r\n */\r\n public static get AnisotropicTextureEnabled(): boolean {\r\n return this._AnisotropicTextureEnabled;\r\n }\r\n public static set AnisotropicTextureEnabled(value: boolean) {\r\n if (this._AnisotropicTextureEnabled === value) {\r\n return;\r\n }\r\n\r\n this._AnisotropicTextureEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private static _ThicknessTextureEnabled = true;\r\n /**\r\n * Are thickness textures enabled in the application.\r\n */\r\n public static get ThicknessTextureEnabled(): boolean {\r\n return this._ThicknessTextureEnabled;\r\n }\r\n public static set ThicknessTextureEnabled(value: boolean) {\r\n if (this._ThicknessTextureEnabled === value) {\r\n return;\r\n }\r\n\r\n this._ThicknessTextureEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private static _RefractionIntensityTextureEnabled = true;\r\n /**\r\n * Are refraction intensity textures enabled in the application.\r\n */\r\n public static get RefractionIntensityTextureEnabled(): boolean {\r\n return this._ThicknessTextureEnabled;\r\n }\r\n public static set RefractionIntensityTextureEnabled(value: boolean) {\r\n if (this._RefractionIntensityTextureEnabled === value) {\r\n return;\r\n }\r\n\r\n this._RefractionIntensityTextureEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private static _TranslucencyIntensityTextureEnabled = true;\r\n /**\r\n * Are translucency intensity textures enabled in the application.\r\n */\r\n public static get TranslucencyIntensityTextureEnabled(): boolean {\r\n return this._ThicknessTextureEnabled;\r\n }\r\n public static set TranslucencyIntensityTextureEnabled(value: boolean) {\r\n if (this._TranslucencyIntensityTextureEnabled === value) {\r\n return;\r\n }\r\n\r\n this._TranslucencyIntensityTextureEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private static _IridescenceTextureEnabled = true;\r\n /**\r\n * Are translucency intensity textures enabled in the application.\r\n */\r\n public static get IridescenceTextureEnabled(): boolean {\r\n return this._IridescenceTextureEnabled;\r\n }\r\n public static set IridescenceTextureEnabled(value: boolean) {\r\n if (this._IridescenceTextureEnabled === value) {\r\n return;\r\n }\r\n\r\n this._IridescenceTextureEnabled = value;\r\n Engine.MarkAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n}\r\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"decalFragmentDeclaration\";\nconst shader = `#ifdef DECAL\nuniform vec4 vDecalInfos;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const decalFragmentDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\nimport \"./decalFragmentDeclaration\";\n\nconst name = \"defaultFragmentDeclaration\";\nconst shader = `uniform vec4 vEyePosition;uniform vec4 vDiffuseColor;\n#ifdef SPECULARTERM\nuniform vec4 vSpecularColor;\n#endif\nuniform vec3 vEmissiveColor;uniform vec3 vAmbientColor;uniform float visibility;\n#ifdef DIFFUSE\nuniform vec2 vDiffuseInfos;\n#endif\n#ifdef AMBIENT\nuniform vec2 vAmbientInfos;\n#endif\n#ifdef OPACITY \nuniform vec2 vOpacityInfos;\n#endif\n#ifdef EMISSIVE\nuniform vec2 vEmissiveInfos;\n#endif\n#ifdef LIGHTMAP\nuniform vec2 vLightmapInfos;\n#endif\n#ifdef BUMP\nuniform vec3 vBumpInfos;uniform vec2 vTangentSpaceParams;\n#endif\n#ifdef ALPHATEST\nuniform float alphaCutOff;\n#endif\n#if defined(REFLECTIONMAP_SPHERICAL) || defined(REFLECTIONMAP_PROJECTION) || defined(REFRACTION) || defined(PREPASS)\nuniform mat4 view;\n#endif\n#ifdef REFRACTION\nuniform vec4 vRefractionInfos;\n#ifndef REFRACTIONMAP_3D\nuniform mat4 refractionMatrix;\n#endif\n#ifdef REFRACTIONFRESNEL\nuniform vec4 refractionLeftColor;uniform vec4 refractionRightColor;\n#endif\n#if defined(USE_LOCAL_REFRACTIONMAP_CUBIC) && defined(REFRACTIONMAP_3D)\nuniform vec3 vRefractionPosition;uniform vec3 vRefractionSize; \n#endif\n#endif\n#if defined(SPECULAR) && defined(SPECULARTERM)\nuniform vec2 vSpecularInfos;\n#endif\n#ifdef DIFFUSEFRESNEL\nuniform vec4 diffuseLeftColor;uniform vec4 diffuseRightColor;\n#endif\n#ifdef OPACITYFRESNEL\nuniform vec4 opacityParts;\n#endif\n#ifdef EMISSIVEFRESNEL\nuniform vec4 emissiveLeftColor;uniform vec4 emissiveRightColor;\n#endif\n#ifdef REFLECTION\nuniform vec2 vReflectionInfos;\n#if defined(REFLECTIONMAP_PLANAR) || defined(REFLECTIONMAP_CUBIC) || defined(REFLECTIONMAP_PROJECTION) || defined(REFLECTIONMAP_EQUIRECTANGULAR) || defined(REFLECTIONMAP_SPHERICAL) || defined(REFLECTIONMAP_SKYBOX)\nuniform mat4 reflectionMatrix;\n#endif\n#ifndef REFLECTIONMAP_SKYBOX\n#if defined(USE_LOCAL_REFLECTIONMAP_CUBIC) && defined(REFLECTIONMAP_CUBIC)\nuniform vec3 vReflectionPosition;uniform vec3 vReflectionSize; \n#endif\n#endif\n#ifdef REFLECTIONFRESNEL\nuniform vec4 reflectionLeftColor;uniform vec4 reflectionRightColor;\n#endif\n#endif\n#ifdef DETAIL\nuniform vec4 vDetailInfos;\n#endif\n#include\n#define ADDITIONAL_FRAGMENT_DECLARATION\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const defaultFragmentDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"sceneUboDeclaration\";\nconst shader = `layout(std140,column_major) uniform;uniform Scene {mat4 viewProjection;\n#ifdef MULTIVIEW\nmat4 viewProjectionR;\n#endif \nmat4 view;mat4 projection;vec4 vEyePosition;};\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const sceneUboDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"meshUboDeclaration\";\nconst shader = `#ifdef WEBGL2\nuniform mat4 world;uniform float visibility;\n#else\nlayout(std140,column_major) uniform;uniform Mesh\n{mat4 world;float visibility;};\n#endif\n#define WORLD_UBO\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const meshUboDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\nimport \"./sceneUboDeclaration\";\nimport \"./meshUboDeclaration\";\n\nconst name = \"defaultUboDeclaration\";\nconst shader = `layout(std140,column_major) uniform;uniform Material\n{vec4 diffuseLeftColor;vec4 diffuseRightColor;vec4 opacityParts;vec4 reflectionLeftColor;vec4 reflectionRightColor;vec4 refractionLeftColor;vec4 refractionRightColor;vec4 emissiveLeftColor;vec4 emissiveRightColor;vec2 vDiffuseInfos;vec2 vAmbientInfos;vec2 vOpacityInfos;vec2 vReflectionInfos;vec3 vReflectionPosition;vec3 vReflectionSize;vec2 vEmissiveInfos;vec2 vLightmapInfos;vec2 vSpecularInfos;vec3 vBumpInfos;mat4 diffuseMatrix;mat4 ambientMatrix;mat4 opacityMatrix;mat4 reflectionMatrix;mat4 emissiveMatrix;mat4 lightmapMatrix;mat4 specularMatrix;mat4 bumpMatrix;vec2 vTangentSpaceParams;float pointSize;float alphaCutOff;mat4 refractionMatrix;vec4 vRefractionInfos;vec3 vRefractionPosition;vec3 vRefractionSize;vec4 vSpecularColor;vec3 vEmissiveColor;vec4 vDiffuseColor;vec3 vAmbientColor;\n#define ADDITIONAL_UBO_DECLARATION\n};\n#include\n#include\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const defaultUboDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"prePassDeclaration\";\nconst shader = `#ifdef PREPASS\n#extension GL_EXT_draw_buffers : require\nlayout(location=0) out highp vec4 glFragData[{X}];highp vec4 gl_FragColor;\n#ifdef PREPASS_DEPTH\nvarying highp vec3 vViewPos;\n#endif\n#ifdef PREPASS_VELOCITY\nvarying highp vec4 vCurrentPosition;varying highp vec4 vPreviousPosition;\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const prePassDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"oitDeclaration\";\nconst shader = `#ifdef ORDER_INDEPENDENT_TRANSPARENCY\n#extension GL_EXT_draw_buffers : require\nlayout(location=0) out vec2 depth; \nlayout(location=1) out vec4 frontColor;layout(location=2) out vec4 backColor;\n#define MAX_DEPTH 99999.0\nhighp vec4 gl_FragColor;uniform sampler2D oitDepthSampler;uniform sampler2D oitFrontColorSampler;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const oitDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"mainUVVaryingDeclaration\";\nconst shader = `#ifdef MAINUV{X}\nvarying vec2 vMainUV{X};\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const mainUVVaryingDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"helperFunctions\";\nconst shader = `const float PI=3.1415926535897932384626433832795;const float HALF_MIN=5.96046448e-08; \nconst float LinearEncodePowerApprox=2.2;const float GammaEncodePowerApprox=1.0/LinearEncodePowerApprox;const vec3 LuminanceEncodeApprox=vec3(0.2126,0.7152,0.0722);const float Epsilon=0.0000001;\n#define saturate(x) clamp(x,0.0,1.0)\n#define absEps(x) abs(x)+Epsilon\n#define maxEps(x) max(x,Epsilon)\n#define saturateEps(x) clamp(x,Epsilon,1.0)\nmat3 transposeMat3(mat3 inMatrix) {vec3 i0=inMatrix[0];vec3 i1=inMatrix[1];vec3 i2=inMatrix[2];mat3 outMatrix=mat3(\nvec3(i0.x,i1.x,i2.x),\nvec3(i0.y,i1.y,i2.y),\nvec3(i0.z,i1.z,i2.z)\n);return outMatrix;}\nmat3 inverseMat3(mat3 inMatrix) {float a00=inMatrix[0][0],a01=inMatrix[0][1],a02=inMatrix[0][2];float a10=inMatrix[1][0],a11=inMatrix[1][1],a12=inMatrix[1][2];float a20=inMatrix[2][0],a21=inMatrix[2][1],a22=inMatrix[2][2];float b01=a22*a11-a12*a21;float b11=-a22*a10+a12*a20;float b21=a21*a10-a11*a20;float det=a00*b01+a01*b11+a02*b21;return mat3(b01,(-a22*a01+a02*a21),(a12*a01-a02*a11),\nb11,(a22*a00-a02*a20),(-a12*a00+a02*a10),\nb21,(-a21*a00+a01*a20),(a11*a00-a01*a10))/det;}\n#if USE_EXACT_SRGB_CONVERSIONS\nvec3 toLinearSpaceExact(vec3 color)\n{vec3 nearZeroSection=0.0773993808*color;vec3 remainingSection=pow(0.947867299*(color+vec3(0.055)),vec3(2.4));\n#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE)\nreturn mix(remainingSection,nearZeroSection,lessThanEqual(color,vec3(0.04045)));\n#else\nreturn\nvec3(\ncolor.r<=0.04045 ? nearZeroSection.r : remainingSection.r,\ncolor.g<=0.04045 ? nearZeroSection.g : remainingSection.g,\ncolor.b<=0.04045 ? nearZeroSection.b : remainingSection.b);\n#endif\n}\nvec3 toGammaSpaceExact(vec3 color)\n{vec3 nearZeroSection=12.92*color;vec3 remainingSection=1.055*pow(color,vec3(0.41666))-vec3(0.055);\n#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE)\nreturn mix(remainingSection,nearZeroSection,lessThanEqual(color,vec3(0.0031308)));\n#else\nreturn\nvec3(\ncolor.r<=0.0031308 ? nearZeroSection.r : remainingSection.r,\ncolor.g<=0.0031308 ? nearZeroSection.g : remainingSection.g,\ncolor.b<=0.0031308 ? nearZeroSection.b : remainingSection.b);\n#endif\n}\n#endif\nfloat toLinearSpace(float color)\n{\n#if USE_EXACT_SRGB_CONVERSIONS\nfloat nearZeroSection=0.0773993808*color;float remainingSection=pow(0.947867299*(color+0.055),2.4);return color<=0.04045 ? nearZeroSection : remainingSection;\n#else\nreturn pow(color,LinearEncodePowerApprox);\n#endif\n}\nvec3 toLinearSpace(vec3 color)\n{\n#if USE_EXACT_SRGB_CONVERSIONS\nreturn toLinearSpaceExact(color);\n#else\nreturn pow(color,vec3(LinearEncodePowerApprox));\n#endif\n}\nvec4 toLinearSpace(vec4 color)\n{\n#if USE_EXACT_SRGB_CONVERSIONS\nreturn vec4(toLinearSpaceExact(color.rgb),color.a);\n#else\nreturn vec4(pow(color.rgb,vec3(LinearEncodePowerApprox)),color.a);\n#endif\n}\nfloat toGammaSpace(float color)\n{\n#if USE_EXACT_SRGB_CONVERSIONS\nfloat nearZeroSection=12.92*color;float remainingSection=1.055*pow(color,0.41666)-0.055;return color<=0.0031308 ? nearZeroSection : remainingSection;\n#else\nreturn pow(color,GammaEncodePowerApprox);\n#endif\n}\nvec3 toGammaSpace(vec3 color)\n{\n#if USE_EXACT_SRGB_CONVERSIONS\nreturn toGammaSpaceExact(color);\n#else\nreturn pow(color,vec3(GammaEncodePowerApprox));\n#endif\n}\nvec4 toGammaSpace(vec4 color)\n{\n#if USE_EXACT_SRGB_CONVERSIONS\nreturn vec4(toGammaSpaceExact(color.rgb),color.a);\n#else\nreturn vec4(pow(color.rgb,vec3(GammaEncodePowerApprox)),color.a);\n#endif\n}\nfloat square(float value)\n{return value*value;}\nvec3 square(vec3 value)\n{return value*value;}\nfloat pow5(float value) {float sq=value*value;return sq*sq*value;}\nfloat getLuminance(vec3 color)\n{return clamp(dot(color,LuminanceEncodeApprox),0.,1.);}\nfloat getRand(vec2 seed) {return fract(sin(dot(seed.xy ,vec2(12.9898,78.233)))*43758.5453);}\nfloat dither(vec2 seed,float varianceAmount) {float rand=getRand(seed);float normVariance=varianceAmount/255.0;float dither=mix(-normVariance,normVariance,rand);return dither;}\nconst float rgbdMaxRange=255.0;vec4 toRGBD(vec3 color) {float maxRGB=maxEps(max(color.r,max(color.g,color.b)));float D =max(rgbdMaxRange/maxRGB,1.);D =clamp(floor(D)/255.0,0.,1.);vec3 rgb=color.rgb*D;rgb=toGammaSpace(rgb);return vec4(clamp(rgb,0.,1.),D); }\nvec3 fromRGBD(vec4 rgbd) {rgbd.rgb=toLinearSpace(rgbd.rgb);return rgbd.rgb/rgbd.a;}\nvec3 parallaxCorrectNormal( vec3 vertexPos,vec3 origVec,vec3 cubeSize,vec3 cubePos ) {vec3 invOrigVec=vec3(1.0,1.0,1.0)/origVec;vec3 halfSize=cubeSize*0.5;vec3 intersecAtMaxPlane=(cubePos+halfSize-vertexPos)*invOrigVec;vec3 intersecAtMinPlane=(cubePos-halfSize-vertexPos)*invOrigVec;vec3 largestIntersec=max(intersecAtMaxPlane,intersecAtMinPlane);float distance=min(min(largestIntersec.x,largestIntersec.y),largestIntersec.z);vec3 intersectPositionWS=vertexPos+origVec*distance;return intersectPositionWS-cubePos;}\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const helperFunctions = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"lightFragmentDeclaration\";\nconst shader = `#ifdef LIGHT{X}\nuniform vec4 vLightData{X};uniform vec4 vLightDiffuse{X};\n#ifdef SPECULARTERM\nuniform vec4 vLightSpecular{X};\n#else\nvec4 vLightSpecular{X}=vec4(0.);\n#endif\n#ifdef SHADOW{X}\n#ifdef SHADOWCSM{X}\nuniform mat4 lightMatrix{X}[SHADOWCSMNUM_CASCADES{X}];uniform float viewFrustumZ{X}[SHADOWCSMNUM_CASCADES{X}];uniform float frustumLengths{X}[SHADOWCSMNUM_CASCADES{X}];uniform float cascadeBlendFactor{X};varying vec4 vPositionFromLight{X}[SHADOWCSMNUM_CASCADES{X}];varying float vDepthMetric{X}[SHADOWCSMNUM_CASCADES{X}];varying vec4 vPositionFromCamera{X};\n#if defined(SHADOWPCSS{X})\nuniform highp sampler2DArrayShadow shadowSampler{X};uniform highp sampler2DArray depthSampler{X};uniform vec2 lightSizeUVCorrection{X}[SHADOWCSMNUM_CASCADES{X}];uniform float depthCorrection{X}[SHADOWCSMNUM_CASCADES{X}];uniform float penumbraDarkness{X};\n#elif defined(SHADOWPCF{X})\nuniform highp sampler2DArrayShadow shadowSampler{X};\n#else\nuniform highp sampler2DArray shadowSampler{X};\n#endif\n#ifdef SHADOWCSMDEBUG{X}\nconst vec3 vCascadeColorsMultiplier{X}[8]=vec3[8]\n(\nvec3 ( 1.5,0.0,0.0 ),\nvec3 ( 0.0,1.5,0.0 ),\nvec3 ( 0.0,0.0,5.5 ),\nvec3 ( 1.5,0.0,5.5 ),\nvec3 ( 1.5,1.5,0.0 ),\nvec3 ( 1.0,1.0,1.0 ),\nvec3 ( 0.0,1.0,5.5 ),\nvec3 ( 0.5,3.5,0.75 )\n);vec3 shadowDebug{X};\n#endif\n#ifdef SHADOWCSMUSESHADOWMAXZ{X}\nint index{X}=-1;\n#else\nint index{X}=SHADOWCSMNUM_CASCADES{X}-1;\n#endif\nfloat diff{X}=0.;\n#elif defined(SHADOWCUBE{X})\nuniform samplerCube shadowSampler{X};\n#else\nvarying vec4 vPositionFromLight{X};varying float vDepthMetric{X};\n#if defined(SHADOWPCSS{X})\nuniform highp sampler2DShadow shadowSampler{X};uniform highp sampler2D depthSampler{X};\n#elif defined(SHADOWPCF{X})\nuniform highp sampler2DShadow shadowSampler{X};\n#else\nuniform sampler2D shadowSampler{X};\n#endif\nuniform mat4 lightMatrix{X};\n#endif\nuniform vec4 shadowsInfo{X};uniform vec2 depthValues{X};\n#endif\n#ifdef SPOTLIGHT{X}\nuniform vec4 vLightDirection{X};uniform vec4 vLightFalloff{X};\n#elif defined(POINTLIGHT{X})\nuniform vec4 vLightFalloff{X};\n#elif defined(HEMILIGHT{X})\nuniform vec3 vLightGround{X};\n#endif\n#ifdef PROJECTEDLIGHTTEXTURE{X}\nuniform mat4 textureProjectionMatrix{X};uniform sampler2D projectionLightSampler{X};\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const lightFragmentDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"lightUboDeclaration\";\nconst shader = `#ifdef LIGHT{X}\nuniform Light{X}\n{vec4 vLightData;vec4 vLightDiffuse;vec4 vLightSpecular;\n#ifdef SPOTLIGHT{X}\nvec4 vLightDirection;vec4 vLightFalloff;\n#elif defined(POINTLIGHT{X})\nvec4 vLightFalloff;\n#elif defined(HEMILIGHT{X})\nvec3 vLightGround;\n#endif\nvec4 shadowsInfo;vec2 depthValues;} light{X};\n#ifdef PROJECTEDLIGHTTEXTURE{X}\nuniform mat4 textureProjectionMatrix{X};uniform sampler2D projectionLightSampler{X};\n#endif\n#ifdef SHADOW{X}\n#ifdef SHADOWCSM{X}\nuniform mat4 lightMatrix{X}[SHADOWCSMNUM_CASCADES{X}];uniform float viewFrustumZ{X}[SHADOWCSMNUM_CASCADES{X}];uniform float frustumLengths{X}[SHADOWCSMNUM_CASCADES{X}];uniform float cascadeBlendFactor{X};varying vec4 vPositionFromLight{X}[SHADOWCSMNUM_CASCADES{X}];varying float vDepthMetric{X}[SHADOWCSMNUM_CASCADES{X}];varying vec4 vPositionFromCamera{X};\n#if defined(SHADOWPCSS{X})\nuniform highp sampler2DArrayShadow shadowSampler{X};uniform highp sampler2DArray depthSampler{X};uniform vec2 lightSizeUVCorrection{X}[SHADOWCSMNUM_CASCADES{X}];uniform float depthCorrection{X}[SHADOWCSMNUM_CASCADES{X}];uniform float penumbraDarkness{X};\n#elif defined(SHADOWPCF{X})\nuniform highp sampler2DArrayShadow shadowSampler{X};\n#else\nuniform highp sampler2DArray shadowSampler{X};\n#endif\n#ifdef SHADOWCSMDEBUG{X}\nconst vec3 vCascadeColorsMultiplier{X}[8]=vec3[8]\n(\nvec3 ( 1.5,0.0,0.0 ),\nvec3 ( 0.0,1.5,0.0 ),\nvec3 ( 0.0,0.0,5.5 ),\nvec3 ( 1.5,0.0,5.5 ),\nvec3 ( 1.5,1.5,0.0 ),\nvec3 ( 1.0,1.0,1.0 ),\nvec3 ( 0.0,1.0,5.5 ),\nvec3 ( 0.5,3.5,0.75 )\n);vec3 shadowDebug{X};\n#endif\n#ifdef SHADOWCSMUSESHADOWMAXZ{X}\nint index{X}=-1;\n#else\nint index{X}=SHADOWCSMNUM_CASCADES{X}-1;\n#endif\nfloat diff{X}=0.;\n#elif defined(SHADOWCUBE{X})\nuniform samplerCube shadowSampler{X}; \n#else\nvarying vec4 vPositionFromLight{X};varying float vDepthMetric{X};\n#if defined(SHADOWPCSS{X})\nuniform highp sampler2DShadow shadowSampler{X};uniform highp sampler2D depthSampler{X};\n#elif defined(SHADOWPCF{X})\nuniform highp sampler2DShadow shadowSampler{X};\n#else\nuniform sampler2D shadowSampler{X};\n#endif\nuniform mat4 lightMatrix{X};\n#endif\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const lightUboDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"lightsFragmentFunctions\";\nconst shader = `struct lightingInfo\n{vec3 diffuse;\n#ifdef SPECULARTERM\nvec3 specular;\n#endif\n#ifdef NDOTL\nfloat ndl;\n#endif\n};lightingInfo computeLighting(vec3 viewDirectionW,vec3 vNormal,vec4 lightData,vec3 diffuseColor,vec3 specularColor,float range,float glossiness) {lightingInfo result;vec3 lightVectorW;float attenuation=1.0;if (lightData.w==0.)\n{vec3 direction=lightData.xyz-vPositionW;attenuation=max(0.,1.0-length(direction)/range);lightVectorW=normalize(direction);}\nelse\n{lightVectorW=normalize(-lightData.xyz);}\nfloat ndl=max(0.,dot(vNormal,lightVectorW));\n#ifdef NDOTL\nresult.ndl=ndl;\n#endif\nresult.diffuse=ndl*diffuseColor*attenuation;\n#ifdef SPECULARTERM\nvec3 angleW=normalize(viewDirectionW+lightVectorW);float specComp=max(0.,dot(vNormal,angleW));specComp=pow(specComp,max(1.,glossiness));result.specular=specComp*specularColor*attenuation;\n#endif\nreturn result;}\nlightingInfo computeSpotLighting(vec3 viewDirectionW,vec3 vNormal,vec4 lightData,vec4 lightDirection,vec3 diffuseColor,vec3 specularColor,float range,float glossiness) {lightingInfo result;vec3 direction=lightData.xyz-vPositionW;vec3 lightVectorW=normalize(direction);float attenuation=max(0.,1.0-length(direction)/range);float cosAngle=max(0.,dot(lightDirection.xyz,-lightVectorW));if (cosAngle>=lightDirection.w)\n{cosAngle=max(0.,pow(cosAngle,lightData.w));attenuation*=cosAngle;float ndl=max(0.,dot(vNormal,lightVectorW));\n#ifdef NDOTL\nresult.ndl=ndl;\n#endif\nresult.diffuse=ndl*diffuseColor*attenuation;\n#ifdef SPECULARTERM\nvec3 angleW=normalize(viewDirectionW+lightVectorW);float specComp=max(0.,dot(vNormal,angleW));specComp=pow(specComp,max(1.,glossiness));result.specular=specComp*specularColor*attenuation;\n#endif\nreturn result;}\nresult.diffuse=vec3(0.);\n#ifdef SPECULARTERM\nresult.specular=vec3(0.);\n#endif\n#ifdef NDOTL\nresult.ndl=0.;\n#endif\nreturn result;}\nlightingInfo computeHemisphericLighting(vec3 viewDirectionW,vec3 vNormal,vec4 lightData,vec3 diffuseColor,vec3 specularColor,vec3 groundColor,float glossiness) {lightingInfo result;float ndl=dot(vNormal,lightData.xyz)*0.5+0.5;\n#ifdef NDOTL\nresult.ndl=ndl;\n#endif\nresult.diffuse=mix(groundColor,diffuseColor,ndl);\n#ifdef SPECULARTERM\nvec3 angleW=normalize(viewDirectionW+lightData.xyz);float specComp=max(0.,dot(vNormal,angleW));specComp=pow(specComp,max(1.,glossiness));result.specular=specComp*specularColor;\n#endif\nreturn result;}\n#define inline\nvec3 computeProjectionTextureDiffuseLighting(sampler2D projectionLightSampler,mat4 textureProjectionMatrix){vec4 strq=textureProjectionMatrix*vec4(vPositionW,1.0);strq/=strq.w;vec3 textureColor=texture2D(projectionLightSampler,strq.xy).rgb;return textureColor;}`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const lightsFragmentFunctions = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"shadowsFragmentFunctions\";\nconst shader = `#ifdef SHADOWS\n#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE)\n#define TEXTUREFUNC(s,c,l) texture2DLodEXT(s,c,l)\n#else\n#define TEXTUREFUNC(s,c,b) texture2D(s,c,b)\n#endif\n#ifndef SHADOWFLOAT\nfloat unpack(vec4 color)\n{const vec4 bit_shift=vec4(1.0/(255.0*255.0*255.0),1.0/(255.0*255.0),1.0/255.0,1.0);return dot(color,bit_shift);}\n#endif\nfloat computeFallOff(float value,vec2 clipSpace,float frustumEdgeFalloff)\n{float mask=smoothstep(1.0-frustumEdgeFalloff,1.00000012,clamp(dot(clipSpace,clipSpace),0.,1.));return mix(value,1.0,mask);}\n#define inline\nfloat computeShadowCube(vec3 lightPosition,samplerCube shadowSampler,float darkness,vec2 depthValues)\n{vec3 directionToLight=vPositionW-lightPosition;float depth=length(directionToLight);depth=(depth+depthValues.x)/(depthValues.y);depth=clamp(depth,0.,1.0);directionToLight=normalize(directionToLight);directionToLight.y=-directionToLight.y;\n#ifndef SHADOWFLOAT\nfloat shadow=unpack(textureCube(shadowSampler,directionToLight));\n#else\nfloat shadow=textureCube(shadowSampler,directionToLight).x;\n#endif\nreturn depth>shadow ? darkness : 1.0;}\n#define inline\nfloat computeShadowWithPoissonSamplingCube(vec3 lightPosition,samplerCube shadowSampler,float mapSize,float darkness,vec2 depthValues)\n{vec3 directionToLight=vPositionW-lightPosition;float depth=length(directionToLight);depth=(depth+depthValues.x)/(depthValues.y);depth=clamp(depth,0.,1.0);directionToLight=normalize(directionToLight);directionToLight.y=-directionToLight.y;float visibility=1.;vec3 poissonDisk[4];poissonDisk[0]=vec3(-1.0,1.0,-1.0);poissonDisk[1]=vec3(1.0,-1.0,-1.0);poissonDisk[2]=vec3(-1.0,-1.0,-1.0);poissonDisk[3]=vec3(1.0,-1.0,1.0);\n#ifndef SHADOWFLOAT\nif (unpack(textureCube(shadowSampler,directionToLight+poissonDisk[0]*mapSize))shadow ? computeFallOff(darkness,clipSpace.xy,frustumEdgeFalloff) : 1.;}\n#endif\n#define inline\nfloat computeShadow(vec4 vPositionFromLight,float depthMetric,sampler2D shadowSampler,float darkness,float frustumEdgeFalloff)\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec2 uv=0.5*clipSpace.xy+vec2(0.5);if (uv.x<0. || uv.x>1.0 || uv.y<0. || uv.y>1.0)\n{return 1.0;}\nelse\n{float shadowPixelDepth=clamp(depthMetric,0.,1.0);\n#ifndef SHADOWFLOAT\nfloat shadow=unpack(TEXTUREFUNC(shadowSampler,uv,0.));\n#else\nfloat shadow=TEXTUREFUNC(shadowSampler,uv,0.).x;\n#endif\nreturn shadowPixelDepth>shadow ? computeFallOff(darkness,clipSpace.xy,frustumEdgeFalloff) : 1.;}}\n#define inline\nfloat computeShadowWithPoissonSampling(vec4 vPositionFromLight,float depthMetric,sampler2D shadowSampler,float mapSize,float darkness,float frustumEdgeFalloff)\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec2 uv=0.5*clipSpace.xy+vec2(0.5);if (uv.x<0. || uv.x>1.0 || uv.y<0. || uv.y>1.0)\n{return 1.0;}\nelse\n{float shadowPixelDepth=clamp(depthMetric,0.,1.0);float visibility=1.;vec2 poissonDisk[4];poissonDisk[0]=vec2(-0.94201624,-0.39906216);poissonDisk[1]=vec2(0.94558609,-0.76890725);poissonDisk[2]=vec2(-0.094184101,-0.92938870);poissonDisk[3]=vec2(0.34495938,0.29387760);\n#ifndef SHADOWFLOAT\nif (unpack(TEXTUREFUNC(shadowSampler,uv+poissonDisk[0]*mapSize,0.))1.0 || uv.y<0. || uv.y>1.0)\n{return 1.0;}\nelse\n{float shadowPixelDepth=clamp(depthMetric,0.,1.0);\n#ifndef SHADOWFLOAT\nfloat shadowMapSample=unpack(TEXTUREFUNC(shadowSampler,uv,0.));\n#else\nfloat shadowMapSample=TEXTUREFUNC(shadowSampler,uv,0.).x;\n#endif\nfloat esm=1.0-clamp(exp(min(87.,depthScale*shadowPixelDepth))*shadowMapSample,0.,1.-darkness);return computeFallOff(esm,clipSpace.xy,frustumEdgeFalloff);}}\n#define inline\nfloat computeShadowWithCloseESM(vec4 vPositionFromLight,float depthMetric,sampler2D shadowSampler,float darkness,float depthScale,float frustumEdgeFalloff)\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec2 uv=0.5*clipSpace.xy+vec2(0.5);if (uv.x<0. || uv.x>1.0 || uv.y<0. || uv.y>1.0)\n{return 1.0;}\nelse\n{float shadowPixelDepth=clamp(depthMetric,0.,1.0); \n#ifndef SHADOWFLOAT\nfloat shadowMapSample=unpack(TEXTUREFUNC(shadowSampler,uv,0.));\n#else\nfloat shadowMapSample=TEXTUREFUNC(shadowSampler,uv,0.).x;\n#endif\nfloat esm=clamp(exp(min(87.,-depthScale*(shadowPixelDepth-shadowMapSample))),darkness,1.);return computeFallOff(esm,clipSpace.xy,frustumEdgeFalloff);}}\n#ifdef IS_NDC_HALF_ZRANGE\n#define ZINCLIP clipSpace.z\n#else\n#define ZINCLIP uvDepth.z\n#endif\n#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE)\n#define GREATEST_LESS_THAN_ONE 0.99999994\n#define inline\nfloat computeShadowWithCSMPCF1(float layer,vec4 vPositionFromLight,float depthMetric,highp sampler2DArrayShadow shadowSampler,float darkness,float frustumEdgeFalloff)\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=clamp(ZINCLIP,0.,GREATEST_LESS_THAN_ONE);vec4 uvDepthLayer=vec4(uvDepth.x,uvDepth.y,layer,uvDepth.z);float shadow=texture2D(shadowSampler,uvDepthLayer);shadow=mix(darkness,1.,shadow);return computeFallOff(shadow,clipSpace.xy,frustumEdgeFalloff);}\n#define inline\nfloat computeShadowWithCSMPCF3(float layer,vec4 vPositionFromLight,float depthMetric,highp sampler2DArrayShadow shadowSampler,vec2 shadowMapSizeAndInverse,float darkness,float frustumEdgeFalloff)\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=clamp(ZINCLIP,0.,GREATEST_LESS_THAN_ONE);vec2 uv=uvDepth.xy*shadowMapSizeAndInverse.x; \nuv+=0.5; \nvec2 st=fract(uv); \nvec2 base_uv=floor(uv)-0.5; \nbase_uv*=shadowMapSizeAndInverse.y; \nvec2 uvw0=3.-2.*st;vec2 uvw1=1.+2.*st;vec2 u=vec2((2.-st.x)/uvw0.x-1.,st.x/uvw1.x+1.)*shadowMapSizeAndInverse.y;vec2 v=vec2((2.-st.y)/uvw0.y-1.,st.y/uvw1.y+1.)*shadowMapSizeAndInverse.y;float shadow=0.;shadow+=uvw0.x*uvw0.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[0],v[0]),layer,uvDepth.z));shadow+=uvw1.x*uvw0.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[1],v[0]),layer,uvDepth.z));shadow+=uvw0.x*uvw1.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[0],v[1]),layer,uvDepth.z));shadow+=uvw1.x*uvw1.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[1],v[1]),layer,uvDepth.z));shadow=shadow/16.;shadow=mix(darkness,1.,shadow);return computeFallOff(shadow,clipSpace.xy,frustumEdgeFalloff);}\n#define inline\nfloat computeShadowWithCSMPCF5(float layer,vec4 vPositionFromLight,float depthMetric,highp sampler2DArrayShadow shadowSampler,vec2 shadowMapSizeAndInverse,float darkness,float frustumEdgeFalloff)\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=clamp(ZINCLIP,0.,GREATEST_LESS_THAN_ONE);vec2 uv=uvDepth.xy*shadowMapSizeAndInverse.x; \nuv+=0.5; \nvec2 st=fract(uv); \nvec2 base_uv=floor(uv)-0.5; \nbase_uv*=shadowMapSizeAndInverse.y; \nvec2 uvw0=4.-3.*st;vec2 uvw1=vec2(7.);vec2 uvw2=1.+3.*st;vec3 u=vec3((3.-2.*st.x)/uvw0.x-2.,(3.+st.x)/uvw1.x,st.x/uvw2.x+2.)*shadowMapSizeAndInverse.y;vec3 v=vec3((3.-2.*st.y)/uvw0.y-2.,(3.+st.y)/uvw1.y,st.y/uvw2.y+2.)*shadowMapSizeAndInverse.y;float shadow=0.;shadow+=uvw0.x*uvw0.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[0],v[0]),layer,uvDepth.z));shadow+=uvw1.x*uvw0.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[1],v[0]),layer,uvDepth.z));shadow+=uvw2.x*uvw0.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[2],v[0]),layer,uvDepth.z));shadow+=uvw0.x*uvw1.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[0],v[1]),layer,uvDepth.z));shadow+=uvw1.x*uvw1.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[1],v[1]),layer,uvDepth.z));shadow+=uvw2.x*uvw1.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[2],v[1]),layer,uvDepth.z));shadow+=uvw0.x*uvw2.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[0],v[2]),layer,uvDepth.z));shadow+=uvw1.x*uvw2.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[1],v[2]),layer,uvDepth.z));shadow+=uvw2.x*uvw2.y*texture2D(shadowSampler,vec4(base_uv.xy+vec2(u[2],v[2]),layer,uvDepth.z));shadow=shadow/144.;shadow=mix(darkness,1.,shadow);return computeFallOff(shadow,clipSpace.xy,frustumEdgeFalloff);}\n#define inline\nfloat computeShadowWithPCF1(vec4 vPositionFromLight,float depthMetric,highp sampler2DShadow shadowSampler,float darkness,float frustumEdgeFalloff)\n{if (depthMetric>1.0 || depthMetric<0.0) {return 1.0;}\nelse\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=ZINCLIP;float shadow=TEXTUREFUNC(shadowSampler,uvDepth,0.);shadow=mix(darkness,1.,shadow);return computeFallOff(shadow,clipSpace.xy,frustumEdgeFalloff);}}\n#define inline\nfloat computeShadowWithPCF3(vec4 vPositionFromLight,float depthMetric,highp sampler2DShadow shadowSampler,vec2 shadowMapSizeAndInverse,float darkness,float frustumEdgeFalloff)\n{if (depthMetric>1.0 || depthMetric<0.0) {return 1.0;}\nelse\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=ZINCLIP;vec2 uv=uvDepth.xy*shadowMapSizeAndInverse.x; \nuv+=0.5; \nvec2 st=fract(uv); \nvec2 base_uv=floor(uv)-0.5; \nbase_uv*=shadowMapSizeAndInverse.y; \nvec2 uvw0=3.-2.*st;vec2 uvw1=1.+2.*st;vec2 u=vec2((2.-st.x)/uvw0.x-1.,st.x/uvw1.x+1.)*shadowMapSizeAndInverse.y;vec2 v=vec2((2.-st.y)/uvw0.y-1.,st.y/uvw1.y+1.)*shadowMapSizeAndInverse.y;float shadow=0.;shadow+=uvw0.x*uvw0.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[0],v[0]),uvDepth.z),0.);shadow+=uvw1.x*uvw0.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[1],v[0]),uvDepth.z),0.);shadow+=uvw0.x*uvw1.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[0],v[1]),uvDepth.z),0.);shadow+=uvw1.x*uvw1.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[1],v[1]),uvDepth.z),0.);shadow=shadow/16.;shadow=mix(darkness,1.,shadow);return computeFallOff(shadow,clipSpace.xy,frustumEdgeFalloff);}}\n#define inline\nfloat computeShadowWithPCF5(vec4 vPositionFromLight,float depthMetric,highp sampler2DShadow shadowSampler,vec2 shadowMapSizeAndInverse,float darkness,float frustumEdgeFalloff)\n{if (depthMetric>1.0 || depthMetric<0.0) {return 1.0;}\nelse\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=ZINCLIP;vec2 uv=uvDepth.xy*shadowMapSizeAndInverse.x; \nuv+=0.5; \nvec2 st=fract(uv); \nvec2 base_uv=floor(uv)-0.5; \nbase_uv*=shadowMapSizeAndInverse.y; \nvec2 uvw0=4.-3.*st;vec2 uvw1=vec2(7.);vec2 uvw2=1.+3.*st;vec3 u=vec3((3.-2.*st.x)/uvw0.x-2.,(3.+st.x)/uvw1.x,st.x/uvw2.x+2.)*shadowMapSizeAndInverse.y;vec3 v=vec3((3.-2.*st.y)/uvw0.y-2.,(3.+st.y)/uvw1.y,st.y/uvw2.y+2.)*shadowMapSizeAndInverse.y;float shadow=0.;shadow+=uvw0.x*uvw0.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[0],v[0]),uvDepth.z),0.);shadow+=uvw1.x*uvw0.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[1],v[0]),uvDepth.z),0.);shadow+=uvw2.x*uvw0.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[2],v[0]),uvDepth.z),0.);shadow+=uvw0.x*uvw1.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[0],v[1]),uvDepth.z),0.);shadow+=uvw1.x*uvw1.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[1],v[1]),uvDepth.z),0.);shadow+=uvw2.x*uvw1.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[2],v[1]),uvDepth.z),0.);shadow+=uvw0.x*uvw2.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[0],v[2]),uvDepth.z),0.);shadow+=uvw1.x*uvw2.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[1],v[2]),uvDepth.z),0.);shadow+=uvw2.x*uvw2.y*TEXTUREFUNC(shadowSampler,vec3(base_uv.xy+vec2(u[2],v[2]),uvDepth.z),0.);shadow=shadow/144.;shadow=mix(darkness,1.,shadow);return computeFallOff(shadow,clipSpace.xy,frustumEdgeFalloff);}}\nconst vec3 PoissonSamplers32[64]=vec3[64](\nvec3(0.06407013,0.05409927,0.),\nvec3(0.7366577,0.5789394,0.),\nvec3(-0.6270542,-0.5320278,0.),\nvec3(-0.4096107,0.8411095,0.),\nvec3(0.6849564,-0.4990818,0.),\nvec3(-0.874181,-0.04579735,0.),\nvec3(0.9989998,0.0009880066,0.),\nvec3(-0.004920578,-0.9151649,0.),\nvec3(0.1805763,0.9747483,0.),\nvec3(-0.2138451,0.2635818,0.),\nvec3(0.109845,0.3884785,0.),\nvec3(0.06876755,-0.3581074,0.),\nvec3(0.374073,-0.7661266,0.),\nvec3(0.3079132,-0.1216763,0.),\nvec3(-0.3794335,-0.8271583,0.),\nvec3(-0.203878,-0.07715034,0.),\nvec3(0.5912697,0.1469799,0.),\nvec3(-0.88069,0.3031784,0.),\nvec3(0.5040108,0.8283722,0.),\nvec3(-0.5844124,0.5494877,0.),\nvec3(0.6017799,-0.1726654,0.),\nvec3(-0.5554981,0.1559997,0.),\nvec3(-0.3016369,-0.3900928,0.),\nvec3(-0.5550632,-0.1723762,0.),\nvec3(0.925029,0.2995041,0.),\nvec3(-0.2473137,0.5538505,0.),\nvec3(0.9183037,-0.2862392,0.),\nvec3(0.2469421,0.6718712,0.),\nvec3(0.3916397,-0.4328209,0.),\nvec3(-0.03576927,-0.6220032,0.),\nvec3(-0.04661255,0.7995201,0.),\nvec3(0.4402924,0.3640312,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.),\nvec3(0.,0.,0.)\n);const vec3 PoissonSamplers64[64]=vec3[64](\nvec3(-0.613392,0.617481,0.),\nvec3(0.170019,-0.040254,0.),\nvec3(-0.299417,0.791925,0.),\nvec3(0.645680,0.493210,0.),\nvec3(-0.651784,0.717887,0.),\nvec3(0.421003,0.027070,0.),\nvec3(-0.817194,-0.271096,0.),\nvec3(-0.705374,-0.668203,0.),\nvec3(0.977050,-0.108615,0.),\nvec3(0.063326,0.142369,0.),\nvec3(0.203528,0.214331,0.),\nvec3(-0.667531,0.326090,0.),\nvec3(-0.098422,-0.295755,0.),\nvec3(-0.885922,0.215369,0.),\nvec3(0.566637,0.605213,0.),\nvec3(0.039766,-0.396100,0.),\nvec3(0.751946,0.453352,0.),\nvec3(0.078707,-0.715323,0.),\nvec3(-0.075838,-0.529344,0.),\nvec3(0.724479,-0.580798,0.),\nvec3(0.222999,-0.215125,0.),\nvec3(-0.467574,-0.405438,0.),\nvec3(-0.248268,-0.814753,0.),\nvec3(0.354411,-0.887570,0.),\nvec3(0.175817,0.382366,0.),\nvec3(0.487472,-0.063082,0.),\nvec3(-0.084078,0.898312,0.),\nvec3(0.488876,-0.783441,0.),\nvec3(0.470016,0.217933,0.),\nvec3(-0.696890,-0.549791,0.),\nvec3(-0.149693,0.605762,0.),\nvec3(0.034211,0.979980,0.),\nvec3(0.503098,-0.308878,0.),\nvec3(-0.016205,-0.872921,0.),\nvec3(0.385784,-0.393902,0.),\nvec3(-0.146886,-0.859249,0.),\nvec3(0.643361,0.164098,0.),\nvec3(0.634388,-0.049471,0.),\nvec3(-0.688894,0.007843,0.),\nvec3(0.464034,-0.188818,0.),\nvec3(-0.440840,0.137486,0.),\nvec3(0.364483,0.511704,0.),\nvec3(0.034028,0.325968,0.),\nvec3(0.099094,-0.308023,0.),\nvec3(0.693960,-0.366253,0.),\nvec3(0.678884,-0.204688,0.),\nvec3(0.001801,0.780328,0.),\nvec3(0.145177,-0.898984,0.),\nvec3(0.062655,-0.611866,0.),\nvec3(0.315226,-0.604297,0.),\nvec3(-0.780145,0.486251,0.),\nvec3(-0.371868,0.882138,0.),\nvec3(0.200476,0.494430,0.),\nvec3(-0.494552,-0.711051,0.),\nvec3(0.612476,0.705252,0.),\nvec3(-0.578845,-0.768792,0.),\nvec3(-0.772454,-0.090976,0.),\nvec3(0.504440,0.372295,0.),\nvec3(0.155736,0.065157,0.),\nvec3(0.391522,0.849605,0.),\nvec3(-0.620106,-0.328104,0.),\nvec3(0.789239,-0.419965,0.),\nvec3(-0.545396,0.538133,0.),\nvec3(-0.178564,-0.596057,0.)\n);\n#define inline\nfloat computeShadowWithCSMPCSS(float layer,vec4 vPositionFromLight,float depthMetric,highp sampler2DArray depthSampler,highp sampler2DArrayShadow shadowSampler,float shadowMapSizeInverse,float lightSizeUV,float darkness,float frustumEdgeFalloff,int searchTapCount,int pcfTapCount,vec3[64] poissonSamplers,vec2 lightSizeUVCorrection,float depthCorrection,float penumbraDarkness)\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=clamp(ZINCLIP,0.,GREATEST_LESS_THAN_ONE);vec4 uvDepthLayer=vec4(uvDepth.x,uvDepth.y,layer,uvDepth.z);float blockerDepth=0.0;float sumBlockerDepth=0.0;float numBlocker=0.0;for (int i=0; i1.0 || depthMetric<0.0) {return 1.0;}\nelse\n{vec3 clipSpace=vPositionFromLight.xyz/vPositionFromLight.w;vec3 uvDepth=vec3(0.5*clipSpace.xyz+vec3(0.5));uvDepth.z=ZINCLIP;float blockerDepth=0.0;float sumBlockerDepth=0.0;float numBlocker=0.0;for (int i=0; i(_DEFINENAME_,BUMP,_VARYINGNAME_,Bump,_SAMPLERNAME_,bump)\n#endif\n#if defined(DETAIL)\n#include(_DEFINENAME_,DETAIL,_VARYINGNAME_,Detail,_SAMPLERNAME_,detail)\n#endif\n#if defined(BUMP) && defined(PARALLAX)\nconst float minSamples=4.;const float maxSamples=15.;const int iMaxSamples=15;vec2 parallaxOcclusion(vec3 vViewDirCoT,vec3 vNormalCoT,vec2 texCoord,float parallaxScale) {float parallaxLimit=length(vViewDirCoT.xy)/vViewDirCoT.z;parallaxLimit*=parallaxScale;vec2 vOffsetDir=normalize(vViewDirCoT.xy);vec2 vMaxOffset=vOffsetDir*parallaxLimit;float numSamples=maxSamples+(dot(vViewDirCoT,vNormalCoT)*(minSamples-maxSamples));float stepSize=1.0/numSamples;float currRayHeight=1.0;vec2 vCurrOffset=vec2(0,0);vec2 vLastOffset=vec2(0,0);float lastSampledHeight=1.0;float currSampledHeight=1.0;bool keepWorking=true;for (int i=0; icurrRayHeight)\n{float delta1=currSampledHeight-currRayHeight;float delta2=(currRayHeight+stepSize)-lastSampledHeight;float ratio=delta1/(delta1+delta2);vCurrOffset=(ratio)* vLastOffset+(1.0-ratio)*vCurrOffset;keepWorking=false;}\nelse\n{currRayHeight-=stepSize;vLastOffset=vCurrOffset;vCurrOffset+=stepSize*vMaxOffset;lastSampledHeight=currSampledHeight;}}\nreturn vCurrOffset;}\nvec2 parallaxOffset(vec3 viewDir,float heightScale)\n{float height=texture2D(bumpSampler,vBumpUV).w;vec2 texCoordOffset=heightScale*viewDir.xy*height;return -texCoordOffset;}\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const bumpFragmentFunctions = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"clipPlaneFragmentDeclaration\";\nconst shader = `#ifdef CLIPPLANE\nvarying float fClipDistance;\n#endif\n#ifdef CLIPPLANE2\nvarying float fClipDistance2;\n#endif\n#ifdef CLIPPLANE3\nvarying float fClipDistance3;\n#endif\n#ifdef CLIPPLANE4\nvarying float fClipDistance4;\n#endif\n#ifdef CLIPPLANE5\nvarying float fClipDistance5;\n#endif\n#ifdef CLIPPLANE6\nvarying float fClipDistance6;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const clipPlaneFragmentDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"logDepthDeclaration\";\nconst shader = `#ifdef LOGARITHMICDEPTH\nuniform float logarithmicDepthConstant;varying float vFragmentDepth;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const logDepthDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"fogFragmentDeclaration\";\nconst shader = `#ifdef FOG\n#define FOGMODE_NONE 0.\n#define FOGMODE_EXP 1.\n#define FOGMODE_EXP2 2.\n#define FOGMODE_LINEAR 3.\n#define E 2.71828\nuniform vec4 vFogInfos;uniform vec3 vFogColor;varying vec3 vFogDistance;float CalcFogFactor()\n{float fogCoeff=1.0;float fogStart=vFogInfos.y;float fogEnd=vFogInfos.z;float fogDensity=vFogInfos.w;float fogDistance=length(vFogDistance);if (FOGMODE_LINEAR==vFogInfos.x)\n{fogCoeff=(fogEnd-fogDistance)/(fogEnd-fogStart);}\nelse if (FOGMODE_EXP==vFogInfos.x)\n{fogCoeff=1.0/pow(E,fogDistance*fogDensity);}\nelse if (FOGMODE_EXP2==vFogInfos.x)\n{fogCoeff=1.0/pow(E,fogDistance*fogDistance*fogDensity*fogDensity);}\nreturn clamp(fogCoeff,0.0,1.0);}\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const fogFragmentDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"clipPlaneFragment\";\nconst shader = `#if defined(CLIPPLANE) || defined(CLIPPLANE2) || defined(CLIPPLANE3) || defined(CLIPPLANE4) || defined(CLIPPLANE5) || defined(CLIPPLANE6)\nif (false) {}\n#endif\n#ifdef CLIPPLANE\nelse if (fClipDistance>0.0)\n{discard;}\n#endif\n#ifdef CLIPPLANE2\nelse if (fClipDistance2>0.0)\n{discard;}\n#endif\n#ifdef CLIPPLANE3\nelse if (fClipDistance3>0.0)\n{discard;}\n#endif\n#ifdef CLIPPLANE4\nelse if (fClipDistance4>0.0)\n{discard;}\n#endif\n#ifdef CLIPPLANE5\nelse if (fClipDistance5>0.0)\n{discard;}\n#endif\n#ifdef CLIPPLANE6\nelse if (fClipDistance6>0.0)\n{discard;}\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const clipPlaneFragment = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"bumpFragment\";\nconst shader = `vec2 uvOffset=vec2(0.0,0.0);\n#if defined(BUMP) || defined(PARALLAX) || defined(DETAIL)\n#ifdef NORMALXYSCALE\nfloat normalScale=1.0;\n#elif defined(BUMP)\nfloat normalScale=vBumpInfos.y;\n#else\nfloat normalScale=1.0;\n#endif\n#if defined(TANGENT) && defined(NORMAL)\nmat3 TBN=vTBN;\n#elif defined(BUMP)\nvec2 TBNUV=gl_FrontFacing ? vBumpUV : -vBumpUV;mat3 TBN=cotangent_frame(normalW*normalScale,vPositionW,TBNUV,vTangentSpaceParams);\n#else\nvec2 TBNUV=gl_FrontFacing ? vDetailUV : -vDetailUV;mat3 TBN=cotangent_frame(normalW*normalScale,vPositionW,TBNUV,vec2(1.,1.));\n#endif\n#elif defined(ANISOTROPIC)\n#if defined(TANGENT) && defined(NORMAL)\nmat3 TBN=vTBN;\n#else\nvec2 TBNUV=gl_FrontFacing ? vMainUV1 : -vMainUV1;mat3 TBN=cotangent_frame(normalW,vPositionW,TBNUV,vec2(1.,1.));\n#endif\n#endif\n#ifdef PARALLAX\nmat3 invTBN=transposeMat3(TBN);\n#ifdef PARALLAXOCCLUSION\nuvOffset=parallaxOcclusion(invTBN*-viewDirectionW,invTBN*normalW,vBumpUV,vBumpInfos.z);\n#else\nuvOffset=parallaxOffset(invTBN*viewDirectionW,vBumpInfos.z);\n#endif\n#endif\n#ifdef DETAIL\nvec4 detailColor=texture2D(detailSampler,vDetailUV+uvOffset);vec2 detailNormalRG=detailColor.wy*2.0-1.0;float detailNormalB=sqrt(1.-saturate(dot(detailNormalRG,detailNormalRG)));vec3 detailNormal=vec3(detailNormalRG,detailNormalB);\n#endif\n#ifdef BUMP\n#ifdef OBJECTSPACE_NORMALMAP\n#define CUSTOM_FRAGMENT_BUMP_FRAGMENT\nnormalW=normalize(texture2D(bumpSampler,vBumpUV).xyz *2.0-1.0);normalW=normalize(mat3(normalMatrix)*normalW);\n#elif !defined(DETAIL)\nnormalW=perturbNormal(TBN,texture2D(bumpSampler,vBumpUV+uvOffset).xyz,vBumpInfos.y);\n#else\nvec3 bumpNormal=texture2D(bumpSampler,vBumpUV+uvOffset).xyz*2.0-1.0;\n#if DETAIL_NORMALBLENDMETHOD==0 \ndetailNormal.xy*=vDetailInfos.z;vec3 blendedNormal=normalize(vec3(bumpNormal.xy+detailNormal.xy,bumpNormal.z*detailNormal.z));\n#elif DETAIL_NORMALBLENDMETHOD==1 \ndetailNormal.xy*=vDetailInfos.z;bumpNormal+=vec3(0.0,0.0,1.0);detailNormal*=vec3(-1.0,-1.0,1.0);vec3 blendedNormal=bumpNormal*dot(bumpNormal,detailNormal)/bumpNormal.z-detailNormal;\n#endif\nnormalW=perturbNormalBase(TBN,blendedNormal,vBumpInfos.y);\n#endif\n#elif defined(DETAIL)\ndetailNormal.xy*=vDetailInfos.z;normalW=perturbNormalBase(TBN,detailNormal,vDetailInfos.z);\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const bumpFragment = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"decalFragment\";\nconst shader = `#ifdef DECAL\n#ifdef GAMMADECAL\ndecalColor.rgb=toLinearSpace(decalColor.rgb);\n#endif\n#ifdef DECAL_SMOOTHALPHA\ndecalColor.a*=decalColor.a;\n#endif\nsurfaceAlbedo.rgb=mix(surfaceAlbedo.rgb,decalColor.rgb,decalColor.a);\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const decalFragment = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"depthPrePass\";\nconst shader = `#ifdef DEPTHPREPASS\ngl_FragColor=vec4(0.,0.,0.,1.0);return;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const depthPrePass = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"lightFragment\";\nconst shader = `#ifdef LIGHT{X}\n#if defined(SHADOWONLY) || defined(LIGHTMAP) && defined(LIGHTMAPEXCLUDED{X}) && defined(LIGHTMAPNOSPECULAR{X})\n#else\n#ifdef PBR\n#ifdef SPOTLIGHT{X}\npreInfo=computePointAndSpotPreLightingInfo(light{X}.vLightData,viewDirectionW,normalW);\n#elif defined(POINTLIGHT{X})\npreInfo=computePointAndSpotPreLightingInfo(light{X}.vLightData,viewDirectionW,normalW);\n#elif defined(HEMILIGHT{X})\npreInfo=computeHemisphericPreLightingInfo(light{X}.vLightData,viewDirectionW,normalW);\n#elif defined(DIRLIGHT{X})\npreInfo=computeDirectionalPreLightingInfo(light{X}.vLightData,viewDirectionW,normalW);\n#endif\npreInfo.NdotV=NdotV;\n#ifdef SPOTLIGHT{X}\n#ifdef LIGHT_FALLOFF_GLTF{X}\npreInfo.attenuation=computeDistanceLightFalloff_GLTF(preInfo.lightDistanceSquared,light{X}.vLightFalloff.y);preInfo.attenuation*=computeDirectionalLightFalloff_GLTF(light{X}.vLightDirection.xyz,preInfo.L,light{X}.vLightFalloff.z,light{X}.vLightFalloff.w);\n#elif defined(LIGHT_FALLOFF_PHYSICAL{X})\npreInfo.attenuation=computeDistanceLightFalloff_Physical(preInfo.lightDistanceSquared);preInfo.attenuation*=computeDirectionalLightFalloff_Physical(light{X}.vLightDirection.xyz,preInfo.L,light{X}.vLightDirection.w);\n#elif defined(LIGHT_FALLOFF_STANDARD{X})\npreInfo.attenuation=computeDistanceLightFalloff_Standard(preInfo.lightOffset,light{X}.vLightFalloff.x);preInfo.attenuation*=computeDirectionalLightFalloff_Standard(light{X}.vLightDirection.xyz,preInfo.L,light{X}.vLightDirection.w,light{X}.vLightData.w);\n#else\npreInfo.attenuation=computeDistanceLightFalloff(preInfo.lightOffset,preInfo.lightDistanceSquared,light{X}.vLightFalloff.x,light{X}.vLightFalloff.y);preInfo.attenuation*=computeDirectionalLightFalloff(light{X}.vLightDirection.xyz,preInfo.L,light{X}.vLightDirection.w,light{X}.vLightData.w,light{X}.vLightFalloff.z,light{X}.vLightFalloff.w);\n#endif\n#elif defined(POINTLIGHT{X})\n#ifdef LIGHT_FALLOFF_GLTF{X}\npreInfo.attenuation=computeDistanceLightFalloff_GLTF(preInfo.lightDistanceSquared,light{X}.vLightFalloff.y);\n#elif defined(LIGHT_FALLOFF_PHYSICAL{X})\npreInfo.attenuation=computeDistanceLightFalloff_Physical(preInfo.lightDistanceSquared);\n#elif defined(LIGHT_FALLOFF_STANDARD{X})\npreInfo.attenuation=computeDistanceLightFalloff_Standard(preInfo.lightOffset,light{X}.vLightFalloff.x);\n#else\npreInfo.attenuation=computeDistanceLightFalloff(preInfo.lightOffset,preInfo.lightDistanceSquared,light{X}.vLightFalloff.x,light{X}.vLightFalloff.y);\n#endif\n#else\npreInfo.attenuation=1.0;\n#endif\n#ifdef HEMILIGHT{X}\npreInfo.roughness=roughness;\n#else\npreInfo.roughness=adjustRoughnessFromLightProperties(roughness,light{X}.vLightSpecular.a,preInfo.lightDistance);\n#endif\n#ifdef IRIDESCENCE\npreInfo.iridescenceIntensity=iridescenceIntensity;\n#endif\n#ifdef HEMILIGHT{X}\ninfo.diffuse=computeHemisphericDiffuseLighting(preInfo,light{X}.vLightDiffuse.rgb,light{X}.vLightGround);\n#elif defined(SS_TRANSLUCENCY)\ninfo.diffuse=computeDiffuseAndTransmittedLighting(preInfo,light{X}.vLightDiffuse.rgb,subSurfaceOut.transmittance);\n#else\ninfo.diffuse=computeDiffuseLighting(preInfo,light{X}.vLightDiffuse.rgb);\n#endif\n#ifdef SPECULARTERM\n#ifdef ANISOTROPIC\ninfo.specular=computeAnisotropicSpecularLighting(preInfo,viewDirectionW,normalW,anisotropicOut.anisotropicTangent,anisotropicOut.anisotropicBitangent,anisotropicOut.anisotropy,clearcoatOut.specularEnvironmentR0,specularEnvironmentR90,AARoughnessFactors.x,light{X}.vLightDiffuse.rgb);\n#else\ninfo.specular=computeSpecularLighting(preInfo,normalW,clearcoatOut.specularEnvironmentR0,specularEnvironmentR90,AARoughnessFactors.x,light{X}.vLightDiffuse.rgb);\n#endif\n#endif\n#ifdef SHEEN\n#ifdef SHEEN_LINKWITHALBEDO\npreInfo.roughness=sheenOut.sheenIntensity;\n#else\n#ifdef HEMILIGHT{X}\npreInfo.roughness=sheenOut.sheenRoughness;\n#else\npreInfo.roughness=adjustRoughnessFromLightProperties(sheenOut.sheenRoughness,light{X}.vLightSpecular.a,preInfo.lightDistance);\n#endif\n#endif\ninfo.sheen=computeSheenLighting(preInfo,normalW,sheenOut.sheenColor,specularEnvironmentR90,AARoughnessFactors.x,light{X}.vLightDiffuse.rgb);\n#endif\n#ifdef CLEARCOAT\n#ifdef HEMILIGHT{X}\npreInfo.roughness=clearcoatOut.clearCoatRoughness;\n#else\npreInfo.roughness=adjustRoughnessFromLightProperties(clearcoatOut.clearCoatRoughness,light{X}.vLightSpecular.a,preInfo.lightDistance);\n#endif\ninfo.clearCoat=computeClearCoatLighting(preInfo,clearcoatOut.clearCoatNormalW,clearcoatOut.clearCoatAARoughnessFactors.x,clearcoatOut.clearCoatIntensity,light{X}.vLightDiffuse.rgb);\n#ifdef CLEARCOAT_TINT\nabsorption=computeClearCoatLightingAbsorption(clearcoatOut.clearCoatNdotVRefract,preInfo.L,clearcoatOut.clearCoatNormalW,clearcoatOut.clearCoatColor,clearcoatOut.clearCoatThickness,clearcoatOut.clearCoatIntensity);info.diffuse*=absorption;\n#ifdef SPECULARTERM\ninfo.specular*=absorption;\n#endif\n#endif\ninfo.diffuse*=info.clearCoat.w;\n#ifdef SPECULARTERM\ninfo.specular*=info.clearCoat.w;\n#endif\n#ifdef SHEEN\ninfo.sheen*=info.clearCoat.w;\n#endif\n#endif\n#else\n#ifdef SPOTLIGHT{X}\ninfo=computeSpotLighting(viewDirectionW,normalW,light{X}.vLightData,light{X}.vLightDirection,light{X}.vLightDiffuse.rgb,light{X}.vLightSpecular.rgb,light{X}.vLightDiffuse.a,glossiness);\n#elif defined(HEMILIGHT{X})\ninfo=computeHemisphericLighting(viewDirectionW,normalW,light{X}.vLightData,light{X}.vLightDiffuse.rgb,light{X}.vLightSpecular.rgb,light{X}.vLightGround,glossiness);\n#elif defined(POINTLIGHT{X}) || defined(DIRLIGHT{X})\ninfo=computeLighting(viewDirectionW,normalW,light{X}.vLightData,light{X}.vLightDiffuse.rgb,light{X}.vLightSpecular.rgb,light{X}.vLightDiffuse.a,glossiness);\n#endif\n#endif\n#ifdef PROJECTEDLIGHTTEXTURE{X}\ninfo.diffuse*=computeProjectionTextureDiffuseLighting(projectionLightSampler{X},textureProjectionMatrix{X});\n#endif\n#endif\n#ifdef SHADOW{X}\n#ifdef SHADOWCSM{X}\nfor (int i=0; i=0.) {index{X}=i;break;}}\n#ifdef SHADOWCSMUSESHADOWMAXZ{X}\nif (index{X}>=0)\n#endif\n{\n#if defined(SHADOWPCF{X})\n#if defined(SHADOWLOWQUALITY{X})\nshadow=computeShadowWithCSMPCF1(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#elif defined(SHADOWMEDIUMQUALITY{X})\nshadow=computeShadowWithCSMPCF3(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.yz,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#else\nshadow=computeShadowWithCSMPCF5(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.yz,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#endif\n#elif defined(SHADOWPCSS{X})\n#if defined(SHADOWLOWQUALITY{X})\nshadow=computeShadowWithCSMPCSS16(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w,lightSizeUVCorrection{X}[index{X}],depthCorrection{X}[index{X}],penumbraDarkness{X});\n#elif defined(SHADOWMEDIUMQUALITY{X})\nshadow=computeShadowWithCSMPCSS32(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w,lightSizeUVCorrection{X}[index{X}],depthCorrection{X}[index{X}],penumbraDarkness{X});\n#else\nshadow=computeShadowWithCSMPCSS64(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w,lightSizeUVCorrection{X}[index{X}],depthCorrection{X}[index{X}],penumbraDarkness{X});\n#endif\n#else\nshadow=computeShadowCSM(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#endif\n#ifdef SHADOWCSMDEBUG{X}\nshadowDebug{X}=vec3(shadow)*vCascadeColorsMultiplier{X}[index{X}];\n#endif\n#ifndef SHADOWCSMNOBLEND{X}\nfloat frustumLength=frustumLengths{X}[index{X}];float diffRatio=clamp(diff{X}/frustumLength,0.,1.)*cascadeBlendFactor{X};if (index{X}<(SHADOWCSMNUM_CASCADES{X}-1) && diffRatio<1.)\n{index{X}+=1;float nextShadow=0.;\n#if defined(SHADOWPCF{X})\n#if defined(SHADOWLOWQUALITY{X})\nnextShadow=computeShadowWithCSMPCF1(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#elif defined(SHADOWMEDIUMQUALITY{X})\nnextShadow=computeShadowWithCSMPCF3(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.yz,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#else\nnextShadow=computeShadowWithCSMPCF5(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.yz,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#endif\n#elif defined(SHADOWPCSS{X})\n#if defined(SHADOWLOWQUALITY{X})\nnextShadow=computeShadowWithCSMPCSS16(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w,lightSizeUVCorrection{X}[index{X}],depthCorrection{X}[index{X}],penumbraDarkness{X});\n#elif defined(SHADOWMEDIUMQUALITY{X})\nnextShadow=computeShadowWithCSMPCSS32(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w,lightSizeUVCorrection{X}[index{X}],depthCorrection{X}[index{X}],penumbraDarkness{X});\n#else\nnextShadow=computeShadowWithCSMPCSS64(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w,lightSizeUVCorrection{X}[index{X}],depthCorrection{X}[index{X}],penumbraDarkness{X});\n#endif\n#else\nnextShadow=computeShadowCSM(float(index{X}),vPositionFromLight{X}[index{X}],vDepthMetric{X}[index{X}],shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#endif\nshadow=mix(nextShadow,shadow,diffRatio);\n#ifdef SHADOWCSMDEBUG{X}\nshadowDebug{X}=mix(vec3(nextShadow)*vCascadeColorsMultiplier{X}[index{X}],shadowDebug{X},diffRatio);\n#endif\n}\n#endif\n}\n#elif defined(SHADOWCLOSEESM{X})\n#if defined(SHADOWCUBE{X})\nshadow=computeShadowWithCloseESMCube(light{X}.vLightData.xyz,shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.z,light{X}.depthValues);\n#else\nshadow=computeShadowWithCloseESM(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.z,light{X}.shadowsInfo.w);\n#endif\n#elif defined(SHADOWESM{X})\n#if defined(SHADOWCUBE{X})\nshadow=computeShadowWithESMCube(light{X}.vLightData.xyz,shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.z,light{X}.depthValues);\n#else\nshadow=computeShadowWithESM(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.z,light{X}.shadowsInfo.w);\n#endif\n#elif defined(SHADOWPOISSON{X})\n#if defined(SHADOWCUBE{X})\nshadow=computeShadowWithPoissonSamplingCube(light{X}.vLightData.xyz,shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.x,light{X}.depthValues);\n#else\nshadow=computeShadowWithPoissonSampling(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#endif\n#elif defined(SHADOWPCF{X})\n#if defined(SHADOWLOWQUALITY{X})\nshadow=computeShadowWithPCF1(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#elif defined(SHADOWMEDIUMQUALITY{X})\nshadow=computeShadowWithPCF3(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.yz,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#else\nshadow=computeShadowWithPCF5(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.yz,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#endif\n#elif defined(SHADOWPCSS{X})\n#if defined(SHADOWLOWQUALITY{X})\nshadow=computeShadowWithPCSS16(vPositionFromLight{X},vDepthMetric{X},depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#elif defined(SHADOWMEDIUMQUALITY{X})\nshadow=computeShadowWithPCSS32(vPositionFromLight{X},vDepthMetric{X},depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#else\nshadow=computeShadowWithPCSS64(vPositionFromLight{X},vDepthMetric{X},depthSampler{X},shadowSampler{X},light{X}.shadowsInfo.y,light{X}.shadowsInfo.z,light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#endif\n#else\n#if defined(SHADOWCUBE{X})\nshadow=computeShadowCube(light{X}.vLightData.xyz,shadowSampler{X},light{X}.shadowsInfo.x,light{X}.depthValues);\n#else\nshadow=computeShadow(vPositionFromLight{X},vDepthMetric{X},shadowSampler{X},light{X}.shadowsInfo.x,light{X}.shadowsInfo.w);\n#endif\n#endif\n#ifdef SHADOWONLY\n#ifndef SHADOWINUSE\n#define SHADOWINUSE\n#endif\nglobalShadow+=shadow;shadowLightCount+=1.0;\n#endif\n#else\nshadow=1.;\n#endif\n#ifndef SHADOWONLY\n#ifdef CUSTOMUSERLIGHTING\ndiffuseBase+=computeCustomDiffuseLighting(info,diffuseBase,shadow);\n#ifdef SPECULARTERM\nspecularBase+=computeCustomSpecularLighting(info,specularBase,shadow);\n#endif\n#elif defined(LIGHTMAP) && defined(LIGHTMAPEXCLUDED{X})\ndiffuseBase+=lightmapColor.rgb*shadow;\n#ifdef SPECULARTERM\n#ifndef LIGHTMAPNOSPECULAR{X}\nspecularBase+=info.specular*shadow*lightmapColor.rgb;\n#endif\n#endif\n#ifdef CLEARCOAT\n#ifndef LIGHTMAPNOSPECULAR{X}\nclearCoatBase+=info.clearCoat.rgb*shadow*lightmapColor.rgb;\n#endif\n#endif\n#ifdef SHEEN\n#ifndef LIGHTMAPNOSPECULAR{X}\nsheenBase+=info.sheen.rgb*shadow;\n#endif\n#endif\n#else\n#ifdef SHADOWCSMDEBUG{X}\ndiffuseBase+=info.diffuse*shadowDebug{X};\n#else \ndiffuseBase+=info.diffuse*shadow;\n#endif\n#ifdef SPECULARTERM\nspecularBase+=info.specular*shadow;\n#endif\n#ifdef CLEARCOAT\nclearCoatBase+=info.clearCoat.rgb*shadow;\n#endif\n#ifdef SHEEN\nsheenBase+=info.sheen.rgb*shadow;\n#endif\n#endif\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const lightFragment = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"logDepthFragment\";\nconst shader = `#ifdef LOGARITHMICDEPTH\ngl_FragDepthEXT=log2(vFragmentDepth)*logarithmicDepthConstant*0.5;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const logDepthFragment = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"fogFragment\";\nconst shader = `#ifdef FOG\nfloat fog=CalcFogFactor();\n#ifdef PBR\nfog=toLinearSpace(fog);\n#endif\ncolor.rgb=mix(vFogColor,color.rgb,fog);\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const fogFragment = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"oitFragment\";\nconst shader = `#ifdef ORDER_INDEPENDENT_TRANSPARENCY\nfloat fragDepth=gl_FragCoord.z; \n#ifdef ORDER_INDEPENDENT_TRANSPARENCY_16BITS\nuint halfFloat=packHalf2x16(vec2(fragDepth));vec2 full=unpackHalf2x16(halfFloat);fragDepth=full.x;\n#endif\nivec2 fragCoord=ivec2(gl_FragCoord.xy);vec2 lastDepth=texelFetch(oitDepthSampler,fragCoord,0).rg;vec4 lastFrontColor=texelFetch(oitFrontColorSampler,fragCoord,0);depth.rg=vec2(-MAX_DEPTH);frontColor=lastFrontColor;backColor=vec4(0.0);\n#ifdef USE_REVERSE_DEPTHBUFFER\nfloat furthestDepth=-lastDepth.x;float nearestDepth=lastDepth.y;\n#else\nfloat nearestDepth=-lastDepth.x;float furthestDepth=lastDepth.y;\n#endif\nfloat alphaMultiplier=1.0-lastFrontColor.a;\n#ifdef USE_REVERSE_DEPTHBUFFER\nif (fragDepth>nearestDepth || fragDepthfurthestDepth) {\n#endif\nreturn;}\n#ifdef USE_REVERSE_DEPTHBUFFER\nif (fragDepthfurthestDepth) {\n#else\nif (fragDepth>nearestDepth && fragDepth\n#if defined(BUMP) || !defined(NORMAL)\n#extension GL_OES_standard_derivatives : enable\n#endif\n#include[SCENE_MRT_COUNT]\n#include\n#define CUSTOM_FRAGMENT_BEGIN\n#ifdef LOGARITHMICDEPTH\n#extension GL_EXT_frag_depth : enable\n#endif\n#define RECIPROCAL_PI2 0.15915494\nvarying vec3 vPositionW;\n#ifdef NORMAL\nvarying vec3 vNormalW;\n#endif\n#if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES)\nvarying vec4 vColor;\n#endif\n#include[1..7]\n#include\n#include<__decl__lightFragment>[0..maxSimultaneousLights]\n#include\n#include\n#include(_DEFINENAME_,DIFFUSE,_VARYINGNAME_,Diffuse,_SAMPLERNAME_,diffuse)\n#include(_DEFINENAME_,AMBIENT,_VARYINGNAME_,Ambient,_SAMPLERNAME_,ambient)\n#include(_DEFINENAME_,OPACITY,_VARYINGNAME_,Opacity,_SAMPLERNAME_,opacity)\n#include(_DEFINENAME_,EMISSIVE,_VARYINGNAME_,Emissive,_SAMPLERNAME_,emissive)\n#include(_DEFINENAME_,LIGHTMAP,_VARYINGNAME_,Lightmap,_SAMPLERNAME_,lightmap)\n#include(_DEFINENAME_,DECAL,_VARYINGNAME_,Decal,_SAMPLERNAME_,decal)\n#ifdef REFRACTION\n#ifdef REFRACTIONMAP_3D\nuniform samplerCube refractionCubeSampler;\n#else\nuniform sampler2D refraction2DSampler;\n#endif\n#endif\n#if defined(SPECULARTERM)\n#include(_DEFINENAME_,SPECULAR,_VARYINGNAME_,Specular,_SAMPLERNAME_,specular)\n#endif\n#include\n#ifdef REFLECTION\n#ifdef REFLECTIONMAP_3D\nuniform samplerCube reflectionCubeSampler;\n#else\nuniform sampler2D reflection2DSampler;\n#endif\n#ifdef REFLECTIONMAP_SKYBOX\nvarying vec3 vPositionUVW;\n#else\n#if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED)\nvarying vec3 vDirectionW;\n#endif\n#endif\n#include\n#endif\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_FRAGMENT_MAIN_BEGIN\n#include\nvec3 viewDirectionW=normalize(vEyePosition.xyz-vPositionW);vec4 baseColor=vec4(1.,1.,1.,1.);vec3 diffuseColor=vDiffuseColor.rgb;float alpha=vDiffuseColor.a;\n#ifdef NORMAL\nvec3 normalW=normalize(vNormalW);\n#else\nvec3 normalW=normalize(-cross(dFdx(vPositionW),dFdy(vPositionW)));\n#endif\n#include\n#ifdef TWOSIDEDLIGHTING\nnormalW=gl_FrontFacing ? normalW : -normalW;\n#endif\n#ifdef DIFFUSE\nbaseColor=texture2D(diffuseSampler,vDiffuseUV+uvOffset);\n#if defined(ALPHATEST) && !defined(ALPHATEST_AFTERALLALPHACOMPUTATIONS)\nif (baseColor.a(surfaceAlbedo,baseColor,GAMMADECAL,_GAMMADECAL_NOTUSED_)\n#endif\n#include\n#if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES)\nbaseColor.rgb*=vColor.rgb;\n#endif\n#ifdef DETAIL\nbaseColor.rgb=baseColor.rgb*2.0*mix(0.5,detailColor.r,vDetailInfos.y);\n#endif\n#if defined(DECAL) && defined(DECAL_AFTER_DETAIL)\nvec4 decalColor=texture2D(decalSampler,vDecalUV+uvOffset);\n#include(surfaceAlbedo,baseColor,GAMMADECAL,_GAMMADECAL_NOTUSED_)\n#endif\n#define CUSTOM_FRAGMENT_UPDATE_DIFFUSE\nvec3 baseAmbientColor=vec3(1.,1.,1.);\n#ifdef AMBIENT\nbaseAmbientColor=texture2D(ambientSampler,vAmbientUV+uvOffset).rgb*vAmbientInfos.y;\n#endif\n#define CUSTOM_FRAGMENT_BEFORE_LIGHTS\n#ifdef SPECULARTERM\nfloat glossiness=vSpecularColor.a;vec3 specularColor=vSpecularColor.rgb;\n#ifdef SPECULAR\nvec4 specularMapColor=texture2D(specularSampler,vSpecularUV+uvOffset);specularColor=specularMapColor.rgb;\n#ifdef GLOSSINESS\nglossiness=glossiness*specularMapColor.a;\n#endif\n#endif\n#else\nfloat glossiness=0.;\n#endif\nvec3 diffuseBase=vec3(0.,0.,0.);lightingInfo info;\n#ifdef SPECULARTERM\nvec3 specularBase=vec3(0.,0.,0.);\n#endif\nfloat shadow=1.;\n#ifdef LIGHTMAP\nvec4 lightmapColor=texture2D(lightmapSampler,vLightmapUV+uvOffset);\n#ifdef RGBDLIGHTMAP\nlightmapColor.rgb=fromRGBD(lightmapColor);\n#endif\nlightmapColor.rgb*=vLightmapInfos.y;\n#endif\n#include[0..maxSimultaneousLights]\nvec4 refractionColor=vec4(0.,0.,0.,1.);\n#ifdef REFRACTION\nvec3 refractionVector=normalize(refract(-viewDirectionW,normalW,vRefractionInfos.y));\n#ifdef REFRACTIONMAP_3D\n#ifdef USE_LOCAL_REFRACTIONMAP_CUBIC\nrefractionVector=parallaxCorrectNormal(vPositionW,refractionVector,vRefractionSize,vRefractionPosition);\n#endif\nrefractionVector.y=refractionVector.y*vRefractionInfos.w;vec4 refractionLookup=textureCube(refractionCubeSampler,refractionVector);if (dot(refractionVector,viewDirectionW)<1.0) {refractionColor=refractionLookup;}\n#else\nvec3 vRefractionUVW=vec3(refractionMatrix*(view*vec4(vPositionW+refractionVector*vRefractionInfos.z,1.0)));vec2 refractionCoords=vRefractionUVW.xy/vRefractionUVW.z;refractionCoords.y=1.0-refractionCoords.y;refractionColor=texture2D(refraction2DSampler,refractionCoords);\n#endif\n#ifdef RGBDREFRACTION\nrefractionColor.rgb=fromRGBD(refractionColor);\n#endif\n#ifdef IS_REFRACTION_LINEAR\nrefractionColor.rgb=toGammaSpace(refractionColor.rgb);\n#endif\nrefractionColor.rgb*=vRefractionInfos.x;\n#endif\nvec4 reflectionColor=vec4(0.,0.,0.,1.);\n#ifdef REFLECTION\nvec3 vReflectionUVW=computeReflectionCoords(vec4(vPositionW,1.0),normalW);\n#ifdef REFLECTIONMAP_OPPOSITEZ\nvReflectionUVW.z*=-1.0;\n#endif\n#ifdef REFLECTIONMAP_3D\n#ifdef ROUGHNESS\nfloat bias=vReflectionInfos.y;\n#ifdef SPECULARTERM\n#ifdef SPECULAR\n#ifdef GLOSSINESS\nbias*=(1.0-specularMapColor.a);\n#endif\n#endif\n#endif\nreflectionColor=textureCube(reflectionCubeSampler,vReflectionUVW,bias);\n#else\nreflectionColor=textureCube(reflectionCubeSampler,vReflectionUVW);\n#endif\n#else\nvec2 coords=vReflectionUVW.xy;\n#ifdef REFLECTIONMAP_PROJECTION\ncoords/=vReflectionUVW.z;\n#endif\ncoords.y=1.0-coords.y;reflectionColor=texture2D(reflection2DSampler,coords);\n#endif\n#ifdef RGBDREFLECTION\nreflectionColor.rgb=fromRGBD(reflectionColor);\n#endif\n#ifdef IS_REFLECTION_LINEAR\nreflectionColor.rgb=toGammaSpace(reflectionColor.rgb);\n#endif\nreflectionColor.rgb*=vReflectionInfos.x;\n#ifdef REFLECTIONFRESNEL\nfloat reflectionFresnelTerm=computeFresnelTerm(viewDirectionW,normalW,reflectionRightColor.a,reflectionLeftColor.a);\n#ifdef REFLECTIONFRESNELFROMSPECULAR\n#ifdef SPECULARTERM\nreflectionColor.rgb*=specularColor.rgb*(1.0-reflectionFresnelTerm)+reflectionFresnelTerm*reflectionRightColor.rgb;\n#else\nreflectionColor.rgb*=reflectionLeftColor.rgb*(1.0-reflectionFresnelTerm)+reflectionFresnelTerm*reflectionRightColor.rgb;\n#endif\n#else\nreflectionColor.rgb*=reflectionLeftColor.rgb*(1.0-reflectionFresnelTerm)+reflectionFresnelTerm*reflectionRightColor.rgb;\n#endif\n#endif\n#endif\n#ifdef REFRACTIONFRESNEL\nfloat refractionFresnelTerm=computeFresnelTerm(viewDirectionW,normalW,refractionRightColor.a,refractionLeftColor.a);refractionColor.rgb*=refractionLeftColor.rgb*(1.0-refractionFresnelTerm)+refractionFresnelTerm*refractionRightColor.rgb;\n#endif\n#ifdef OPACITY\nvec4 opacityMap=texture2D(opacitySampler,vOpacityUV+uvOffset);\n#ifdef OPACITYRGB\nopacityMap.rgb=opacityMap.rgb*vec3(0.3,0.59,0.11);alpha*=(opacityMap.x+opacityMap.y+opacityMap.z)* vOpacityInfos.y;\n#else\nalpha*=opacityMap.a*vOpacityInfos.y;\n#endif\n#endif\n#if defined(VERTEXALPHA) || defined(INSTANCESCOLOR) && defined(INSTANCES)\nalpha*=vColor.a;\n#endif\n#ifdef OPACITYFRESNEL\nfloat opacityFresnelTerm=computeFresnelTerm(viewDirectionW,normalW,opacityParts.z,opacityParts.w);alpha+=opacityParts.x*(1.0-opacityFresnelTerm)+opacityFresnelTerm*opacityParts.y;\n#endif\n#ifdef ALPHATEST\n#ifdef ALPHATEST_AFTERALLALPHACOMPUTATIONS\nif (alpha\n#include\n#ifdef IMAGEPROCESSINGPOSTPROCESS\ncolor.rgb=toLinearSpace(color.rgb);\n#else\n#ifdef IMAGEPROCESSING\ncolor.rgb=toLinearSpace(color.rgb);color=applyImageProcessing(color);\n#endif\n#endif\ncolor.a*=visibility;\n#ifdef PREMULTIPLYALPHA\ncolor.rgb*=color.a;\n#endif\n#define CUSTOM_FRAGMENT_BEFORE_FRAGCOLOR\n#ifdef PREPASS\nfloat writeGeometryInfo=color.a>0.4 ? 1.0 : 0.0;gl_FragData[0]=color; \n#ifdef PREPASS_POSITION\ngl_FragData[PREPASS_POSITION_INDEX]=vec4(vPositionW,writeGeometryInfo);\n#endif\n#ifdef PREPASS_VELOCITY\nvec2 a=(vCurrentPosition.xy/vCurrentPosition.w)*0.5+0.5;vec2 b=(vPreviousPosition.xy/vPreviousPosition.w)*0.5+0.5;vec2 velocity=abs(a-b);velocity=vec2(pow(velocity.x,1.0/3.0),pow(velocity.y,1.0/3.0))*sign(a-b)*0.5+0.5;gl_FragData[PREPASS_VELOCITY_INDEX]=vec4(velocity,0.0,writeGeometryInfo);\n#endif\n#ifdef PREPASS_IRRADIANCE\ngl_FragData[PREPASS_IRRADIANCE_INDEX]=vec4(0.0,0.0,0.0,writeGeometryInfo); \n#endif\n#ifdef PREPASS_DEPTH\ngl_FragData[PREPASS_DEPTH_INDEX]=vec4(vViewPos.z,0.0,0.0,writeGeometryInfo); \n#endif\n#ifdef PREPASS_NORMAL\ngl_FragData[PREPASS_NORMAL_INDEX]=vec4(normalize((view*vec4(normalW,0.0)).rgb),writeGeometryInfo); \n#endif\n#ifdef PREPASS_ALBEDO_SQRT\ngl_FragData[PREPASS_ALBEDO_SQRT_INDEX]=vec4(0.0,0.0,0.0,writeGeometryInfo); \n#endif\n#ifdef PREPASS_REFLECTIVITY\n#if defined(SPECULARTERM)\n#if defined(SPECULAR)\ngl_FragData[PREPASS_REFLECTIVITY_INDEX]=vec4(toLinearSpace(specularMapColor))*writeGeometryInfo; \n#else\ngl_FragData[PREPASS_REFLECTIVITY_INDEX]=vec4(toLinearSpace(specularColor),1.0)*writeGeometryInfo;\n#endif\n#else\ngl_FragData[PREPASS_REFLECTIVITY_INDEX]=vec4(0.0,0.0,0.0,1.0)*writeGeometryInfo;\n#endif\n#endif\n#endif\n#if !defined(PREPASS) || defined(WEBGL2)\ngl_FragColor=color;\n#endif\n#include\n#if ORDER_INDEPENDENT_TRANSPARENCY\nif (fragDepth==nearestDepth) {frontColor.rgb+=color.rgb*color.a*alphaMultiplier;frontColor.a=1.0-alphaMultiplier*(1.0-color.a);} else {backColor+=color;}\n#endif\n#define CUSTOM_FRAGMENT_MAIN_END\n}\n`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const defaultPixelShader = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"decalVertexDeclaration\";\nconst shader = `#ifdef DECAL\nuniform vec4 vDecalInfos;uniform mat4 decalMatrix;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const decalVertexDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\nimport \"./decalVertexDeclaration\";\n\nconst name = \"defaultVertexDeclaration\";\nconst shader = `uniform mat4 viewProjection;uniform mat4 view;\n#ifdef DIFFUSE\nuniform mat4 diffuseMatrix;uniform vec2 vDiffuseInfos;\n#endif\n#ifdef AMBIENT\nuniform mat4 ambientMatrix;uniform vec2 vAmbientInfos;\n#endif\n#ifdef OPACITY\nuniform mat4 opacityMatrix;uniform vec2 vOpacityInfos;\n#endif\n#ifdef EMISSIVE\nuniform vec2 vEmissiveInfos;uniform mat4 emissiveMatrix;\n#endif\n#ifdef LIGHTMAP\nuniform vec2 vLightmapInfos;uniform mat4 lightmapMatrix;\n#endif\n#if defined(SPECULAR) && defined(SPECULARTERM)\nuniform vec2 vSpecularInfos;uniform mat4 specularMatrix;\n#endif\n#ifdef BUMP\nuniform vec3 vBumpInfos;uniform mat4 bumpMatrix;\n#endif\n#ifdef REFLECTION\nuniform mat4 reflectionMatrix;\n#endif\n#ifdef POINTSIZE\nuniform float pointSize;\n#endif\n#ifdef DETAIL\nuniform vec4 vDetailInfos;uniform mat4 detailMatrix;\n#endif\n#include\n#define ADDITIONAL_VERTEX_DECLARATION\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const defaultVertexDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"uvAttributeDeclaration\";\nconst shader = `#ifdef UV{X}\nattribute vec2 uv{X};\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const uvAttributeDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"bonesDeclaration\";\nconst shader = `#if NUM_BONE_INFLUENCERS>0\nattribute vec4 matricesIndices;attribute vec4 matricesWeights;\n#if NUM_BONE_INFLUENCERS>4\nattribute vec4 matricesIndicesExtra;attribute vec4 matricesWeightsExtra;\n#endif\n#ifndef BAKED_VERTEX_ANIMATION_TEXTURE\n#ifdef BONETEXTURE\nuniform sampler2D boneSampler;uniform float boneTextureWidth;\n#else\nuniform mat4 mBones[BonesPerMesh];\n#ifdef BONES_VELOCITY_ENABLED\nuniform mat4 mPreviousBones[BonesPerMesh];\n#endif\n#endif\n#ifdef BONETEXTURE\n#define inline\nmat4 readMatrixFromRawSampler(sampler2D smp,float index)\n{float offset=index *4.0;float dx=1.0/boneTextureWidth;vec4 m0=texture2D(smp,vec2(dx*(offset+0.5),0.));vec4 m1=texture2D(smp,vec2(dx*(offset+1.5),0.));vec4 m2=texture2D(smp,vec2(dx*(offset+2.5),0.));vec4 m3=texture2D(smp,vec2(dx*(offset+3.5),0.));return mat4(m0,m1,m2,m3);}\n#endif\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const bonesDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"bakedVertexAnimationDeclaration\";\nconst shader = `#ifdef BAKED_VERTEX_ANIMATION_TEXTURE\nuniform float bakedVertexAnimationTime;uniform vec2 bakedVertexAnimationTextureSizeInverted;uniform vec4 bakedVertexAnimationSettings;uniform sampler2D bakedVertexAnimationTexture;\n#ifdef INSTANCES\nattribute vec4 bakedVertexAnimationSettingsInstanced;\n#endif\n#define inline\nmat4 readMatrixFromRawSamplerVAT(sampler2D smp,float index,float frame)\n{float offset=index*4.0;float frameUV=(frame+0.5)*bakedVertexAnimationTextureSizeInverted.y;float dx=bakedVertexAnimationTextureSizeInverted.x;vec4 m0=texture2D(smp,vec2(dx*(offset+0.5),frameUV));vec4 m1=texture2D(smp,vec2(dx*(offset+1.5),frameUV));vec4 m2=texture2D(smp,vec2(dx*(offset+2.5),frameUV));vec4 m3=texture2D(smp,vec2(dx*(offset+3.5),frameUV));return mat4(m0,m1,m2,m3);}\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const bakedVertexAnimationDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"instancesDeclaration\";\nconst shader = `#ifdef INSTANCES\nattribute vec4 world0;attribute vec4 world1;attribute vec4 world2;attribute vec4 world3;\n#ifdef INSTANCESCOLOR\nattribute vec4 instanceColor;\n#endif\n#if defined(THIN_INSTANCES) && !defined(WORLD_UBO)\nuniform mat4 world;\n#endif\n#if defined(VELOCITY) || defined(PREPASS_VELOCITY)\nattribute vec4 previousWorld0;attribute vec4 previousWorld1;attribute vec4 previousWorld2;attribute vec4 previousWorld3;\n#ifdef THIN_INSTANCES\nuniform mat4 previousWorld;\n#endif\n#endif\n#else\n#if !defined(WORLD_UBO)\nuniform mat4 world;\n#endif\n#if defined(VELOCITY) || defined(PREPASS_VELOCITY)\nuniform mat4 previousWorld;\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const instancesDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"prePassVertexDeclaration\";\nconst shader = `#ifdef PREPASS\n#ifdef PREPASS_DEPTH\nvarying vec3 vViewPos;\n#endif\n#ifdef PREPASS_VELOCITY\nuniform mat4 previousViewProjection;varying vec4 vCurrentPosition;varying vec4 vPreviousPosition;\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const prePassVertexDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"samplerVertexDeclaration\";\nconst shader = `#if defined(_DEFINENAME_) && _DEFINENAME_DIRECTUV==0\nvarying vec2 v_VARYINGNAME_UV;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const samplerVertexDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"bumpVertexDeclaration\";\nconst shader = `#if defined(BUMP) || defined(PARALLAX) || defined(CLEARCOAT_BUMP) || defined(ANISOTROPIC)\n#if defined(TANGENT) && defined(NORMAL) \nvarying mat3 vTBN;\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const bumpVertexDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"clipPlaneVertexDeclaration\";\nconst shader = `#ifdef CLIPPLANE\nuniform vec4 vClipPlane;varying float fClipDistance;\n#endif\n#ifdef CLIPPLANE2\nuniform vec4 vClipPlane2;varying float fClipDistance2;\n#endif\n#ifdef CLIPPLANE3\nuniform vec4 vClipPlane3;varying float fClipDistance3;\n#endif\n#ifdef CLIPPLANE4\nuniform vec4 vClipPlane4;varying float fClipDistance4;\n#endif\n#ifdef CLIPPLANE5\nuniform vec4 vClipPlane5;varying float fClipDistance5;\n#endif\n#ifdef CLIPPLANE6\nuniform vec4 vClipPlane6;varying float fClipDistance6;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const clipPlaneVertexDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"fogVertexDeclaration\";\nconst shader = `#ifdef FOG\nvarying vec3 vFogDistance;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const fogVertexDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"lightVxFragmentDeclaration\";\nconst shader = `#ifdef LIGHT{X}\nuniform vec4 vLightData{X};uniform vec4 vLightDiffuse{X};\n#ifdef SPECULARTERM\nuniform vec4 vLightSpecular{X};\n#else\nvec4 vLightSpecular{X}=vec4(0.);\n#endif\n#ifdef SHADOW{X}\n#ifdef SHADOWCSM{X}\nuniform mat4 lightMatrix{X}[SHADOWCSMNUM_CASCADES{X}];varying vec4 vPositionFromLight{X}[SHADOWCSMNUM_CASCADES{X}];varying float vDepthMetric{X}[SHADOWCSMNUM_CASCADES{X}];varying vec4 vPositionFromCamera{X};\n#elif defined(SHADOWCUBE{X})\n#else\nvarying vec4 vPositionFromLight{X};varying float vDepthMetric{X};uniform mat4 lightMatrix{X};\n#endif\nuniform vec4 shadowsInfo{X};uniform vec2 depthValues{X};\n#endif\n#ifdef SPOTLIGHT{X}\nuniform vec4 vLightDirection{X};uniform vec4 vLightFalloff{X};\n#elif defined(POINTLIGHT{X})\nuniform vec4 vLightFalloff{X};\n#elif defined(HEMILIGHT{X})\nuniform vec3 vLightGround{X};\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const lightVxFragmentDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"lightVxUboDeclaration\";\nconst shader = `#ifdef LIGHT{X}\nuniform Light{X}\n{vec4 vLightData;vec4 vLightDiffuse;vec4 vLightSpecular;\n#ifdef SPOTLIGHT{X}\nvec4 vLightDirection;vec4 vLightFalloff;\n#elif defined(POINTLIGHT{X})\nvec4 vLightFalloff;\n#elif defined(HEMILIGHT{X})\nvec3 vLightGround;\n#endif\nvec4 shadowsInfo;vec2 depthValues;} light{X};\n#ifdef SHADOW{X}\n#ifdef SHADOWCSM{X}\nuniform mat4 lightMatrix{X}[SHADOWCSMNUM_CASCADES{X}];varying vec4 vPositionFromLight{X}[SHADOWCSMNUM_CASCADES{X}];varying float vDepthMetric{X}[SHADOWCSMNUM_CASCADES{X}];varying vec4 vPositionFromCamera{X};\n#elif defined(SHADOWCUBE{X})\n#else\nvarying vec4 vPositionFromLight{X};varying float vDepthMetric{X};uniform mat4 lightMatrix{X};\n#endif\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const lightVxUboDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"morphTargetsVertexGlobalDeclaration\";\nconst shader = `#ifdef MORPHTARGETS\nuniform float morphTargetInfluences[NUM_MORPH_INFLUENCERS];\n#ifdef MORPHTARGETS_TEXTURE \nprecision mediump sampler2DArray; \nuniform float morphTargetTextureIndices[NUM_MORPH_INFLUENCERS];uniform vec3 morphTargetTextureInfo;uniform sampler2DArray morphTargets;vec3 readVector3FromRawSampler(int targetIndex,float vertexIndex)\n{ \nfloat y=floor(vertexIndex/morphTargetTextureInfo.y);float x=vertexIndex-y*morphTargetTextureInfo.y;vec3 textureUV=vec3((x+0.5)/morphTargetTextureInfo.y,(y+0.5)/morphTargetTextureInfo.z,morphTargetTextureIndices[targetIndex]);return texture(morphTargets,textureUV).xyz;}\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const morphTargetsVertexGlobalDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"morphTargetsVertexDeclaration\";\nconst shader = `#ifdef MORPHTARGETS\n#ifndef MORPHTARGETS_TEXTURE\nattribute vec3 position{X};\n#ifdef MORPHTARGETS_NORMAL\nattribute vec3 normal{X};\n#endif\n#ifdef MORPHTARGETS_TANGENT\nattribute vec3 tangent{X};\n#endif\n#ifdef MORPHTARGETS_UV\nattribute vec2 uv_{X};\n#endif\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const morphTargetsVertexDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"morphTargetsVertexGlobal\";\nconst shader = `#ifdef MORPHTARGETS\n#ifdef MORPHTARGETS_TEXTURE\nfloat vertexID;\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const morphTargetsVertexGlobal = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"morphTargetsVertex\";\nconst shader = `#ifdef MORPHTARGETS\n#ifdef MORPHTARGETS_TEXTURE \nvertexID=float(gl_VertexID)*morphTargetTextureInfo.x;positionUpdated+=(readVector3FromRawSampler({X},vertexID)-position)*morphTargetInfluences[{X}];vertexID+=1.0;\n#ifdef MORPHTARGETS_NORMAL\nnormalUpdated+=(readVector3FromRawSampler({X},vertexID) -normal)*morphTargetInfluences[{X}];vertexID+=1.0;\n#endif\n#ifdef MORPHTARGETS_UV\nuvUpdated+=(readVector3FromRawSampler({X},vertexID).xy-uv)*morphTargetInfluences[{X}];vertexID+=1.0;\n#endif\n#ifdef MORPHTARGETS_TANGENT\ntangentUpdated.xyz+=(readVector3FromRawSampler({X},vertexID) -tangent.xyz)*morphTargetInfluences[{X}];\n#endif\n#else\npositionUpdated+=(position{X}-position)*morphTargetInfluences[{X}];\n#ifdef MORPHTARGETS_NORMAL\nnormalUpdated+=(normal{X}-normal)*morphTargetInfluences[{X}];\n#endif\n#ifdef MORPHTARGETS_TANGENT\ntangentUpdated.xyz+=(tangent{X}-tangent.xyz)*morphTargetInfluences[{X}];\n#endif\n#ifdef MORPHTARGETS_UV\nuvUpdated+=(uv_{X}-uv)*morphTargetInfluences[{X}];\n#endif\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const morphTargetsVertex = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"instancesVertex\";\nconst shader = `#ifdef INSTANCES\nmat4 finalWorld=mat4(world0,world1,world2,world3);\n#if defined(PREPASS_VELOCITY) || defined(VELOCITY)\nmat4 finalPreviousWorld=mat4(previousWorld0,previousWorld1,previousWorld2,previousWorld3);\n#endif\n#ifdef THIN_INSTANCES\nfinalWorld=world*finalWorld;\n#if defined(PREPASS_VELOCITY) || defined(VELOCITY)\nfinalPreviousWorld=previousWorld*finalPreviousWorld;\n#endif\n#endif\n#else\nmat4 finalWorld=world;\n#if defined(PREPASS_VELOCITY) || defined(VELOCITY)\nmat4 finalPreviousWorld=previousWorld;\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const instancesVertex = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"bonesVertex\";\nconst shader = `#ifndef BAKED_VERTEX_ANIMATION_TEXTURE\n#if NUM_BONE_INFLUENCERS>0\nmat4 influence;\n#ifdef BONETEXTURE\ninfluence=readMatrixFromRawSampler(boneSampler,matricesIndices[0])*matricesWeights[0];\n#if NUM_BONE_INFLUENCERS>1\ninfluence+=readMatrixFromRawSampler(boneSampler,matricesIndices[1])*matricesWeights[1];\n#endif\n#if NUM_BONE_INFLUENCERS>2\ninfluence+=readMatrixFromRawSampler(boneSampler,matricesIndices[2])*matricesWeights[2];\n#endif\n#if NUM_BONE_INFLUENCERS>3\ninfluence+=readMatrixFromRawSampler(boneSampler,matricesIndices[3])*matricesWeights[3];\n#endif\n#if NUM_BONE_INFLUENCERS>4\ninfluence+=readMatrixFromRawSampler(boneSampler,matricesIndicesExtra[0])*matricesWeightsExtra[0];\n#endif\n#if NUM_BONE_INFLUENCERS>5\ninfluence+=readMatrixFromRawSampler(boneSampler,matricesIndicesExtra[1])*matricesWeightsExtra[1];\n#endif\n#if NUM_BONE_INFLUENCERS>6\ninfluence+=readMatrixFromRawSampler(boneSampler,matricesIndicesExtra[2])*matricesWeightsExtra[2];\n#endif\n#if NUM_BONE_INFLUENCERS>7\ninfluence+=readMatrixFromRawSampler(boneSampler,matricesIndicesExtra[3])*matricesWeightsExtra[3];\n#endif\n#else\ninfluence=mBones[int(matricesIndices[0])]*matricesWeights[0];\n#if NUM_BONE_INFLUENCERS>1\ninfluence+=mBones[int(matricesIndices[1])]*matricesWeights[1];\n#endif\n#if NUM_BONE_INFLUENCERS>2\ninfluence+=mBones[int(matricesIndices[2])]*matricesWeights[2];\n#endif\n#if NUM_BONE_INFLUENCERS>3\ninfluence+=mBones[int(matricesIndices[3])]*matricesWeights[3];\n#endif\n#if NUM_BONE_INFLUENCERS>4\ninfluence+=mBones[int(matricesIndicesExtra[0])]*matricesWeightsExtra[0];\n#endif\n#if NUM_BONE_INFLUENCERS>5\ninfluence+=mBones[int(matricesIndicesExtra[1])]*matricesWeightsExtra[1];\n#endif\n#if NUM_BONE_INFLUENCERS>6\ninfluence+=mBones[int(matricesIndicesExtra[2])]*matricesWeightsExtra[2];\n#endif\n#if NUM_BONE_INFLUENCERS>7\ninfluence+=mBones[int(matricesIndicesExtra[3])]*matricesWeightsExtra[3];\n#endif\n#endif\nfinalWorld=finalWorld*influence;\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const bonesVertex = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"bakedVertexAnimation\";\nconst shader = `#ifdef BAKED_VERTEX_ANIMATION_TEXTURE\n{\n#ifdef INSTANCES\n#define BVASNAME bakedVertexAnimationSettingsInstanced\n#else\n#define BVASNAME bakedVertexAnimationSettings\n#endif\nfloat VATStartFrame=BVASNAME.x;float VATEndFrame=BVASNAME.y;float VATOffsetFrame=BVASNAME.z;float VATSpeed=BVASNAME.w;float totalFrames=VATEndFrame-VATStartFrame+1.0;float time=bakedVertexAnimationTime*VATSpeed/totalFrames;float frameCorrection=time<1.0 ? 0.0 : 1.0;float numOfFrames=totalFrames-frameCorrection;float VATFrameNum=fract(time)*numOfFrames;VATFrameNum=mod(VATFrameNum+VATOffsetFrame,numOfFrames);VATFrameNum=floor(VATFrameNum);VATFrameNum+=VATStartFrame+frameCorrection;mat4 VATInfluence;VATInfluence=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndices[0],VATFrameNum)*matricesWeights[0];\n#if NUM_BONE_INFLUENCERS>1\nVATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndices[1],VATFrameNum)*matricesWeights[1];\n#endif\n#if NUM_BONE_INFLUENCERS>2\nVATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndices[2],VATFrameNum)*matricesWeights[2];\n#endif\n#if NUM_BONE_INFLUENCERS>3\nVATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndices[3],VATFrameNum)*matricesWeights[3];\n#endif\n#if NUM_BONE_INFLUENCERS>4\nVATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndicesExtra[0],VATFrameNum)*matricesWeightsExtra[0];\n#endif\n#if NUM_BONE_INFLUENCERS>5\nVATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndicesExtra[1],VATFrameNum)*matricesWeightsExtra[1];\n#endif\n#if NUM_BONE_INFLUENCERS>6\nVATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndicesExtra[2],VATFrameNum)*matricesWeightsExtra[2];\n#endif\n#if NUM_BONE_INFLUENCERS>7\nVATInfluence+=readMatrixFromRawSamplerVAT(bakedVertexAnimationTexture,matricesIndicesExtra[3],VATFrameNum)*matricesWeightsExtra[3];\n#endif\nfinalWorld=finalWorld*VATInfluence;}\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const bakedVertexAnimation = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"prePassVertex\";\nconst shader = `#ifdef PREPASS_DEPTH\nvViewPos=(view*worldPos).rgb;\n#endif\n#if defined(PREPASS_VELOCITY) && defined(BONES_VELOCITY_ENABLED)\nvCurrentPosition=viewProjection*worldPos;\n#if NUM_BONE_INFLUENCERS>0\nmat4 previousInfluence;previousInfluence=mPreviousBones[int(matricesIndices[0])]*matricesWeights[0];\n#if NUM_BONE_INFLUENCERS>1\npreviousInfluence+=mPreviousBones[int(matricesIndices[1])]*matricesWeights[1];\n#endif \n#if NUM_BONE_INFLUENCERS>2\npreviousInfluence+=mPreviousBones[int(matricesIndices[2])]*matricesWeights[2];\n#endif \n#if NUM_BONE_INFLUENCERS>3\npreviousInfluence+=mPreviousBones[int(matricesIndices[3])]*matricesWeights[3];\n#endif\n#if NUM_BONE_INFLUENCERS>4\npreviousInfluence+=mPreviousBones[int(matricesIndicesExtra[0])]*matricesWeightsExtra[0];\n#endif \n#if NUM_BONE_INFLUENCERS>5\npreviousInfluence+=mPreviousBones[int(matricesIndicesExtra[1])]*matricesWeightsExtra[1];\n#endif \n#if NUM_BONE_INFLUENCERS>6\npreviousInfluence+=mPreviousBones[int(matricesIndicesExtra[2])]*matricesWeightsExtra[2];\n#endif \n#if NUM_BONE_INFLUENCERS>7\npreviousInfluence+=mPreviousBones[int(matricesIndicesExtra[3])]*matricesWeightsExtra[3];\n#endif\nvPreviousPosition=previousViewProjection*finalPreviousWorld*previousInfluence*vec4(positionUpdated,1.0);\n#else\nvPreviousPosition=previousViewProjection*finalPreviousWorld*vec4(positionUpdated,1.0);\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const prePassVertex = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"uvVariableDeclaration\";\nconst shader = `#if !defined(UV{X}) && defined(MAINUV{X})\nvec2 uv{X}=vec2(0.,0.);\n#endif\n#ifdef MAINUV{X}\nvMainUV{X}=uv{X};\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const uvVariableDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"samplerVertexImplementation\";\nconst shader = `#if defined(_DEFINENAME_) && _DEFINENAME_DIRECTUV==0\nif (v_INFONAME_==0.)\n{v_VARYINGNAME_UV=vec2(_MATRIXNAME_Matrix*vec4(uvUpdated,1.0,0.0));}\n#ifdef UV2\nelse if (v_INFONAME_==1.)\n{v_VARYINGNAME_UV=vec2(_MATRIXNAME_Matrix*vec4(uv2,1.0,0.0));}\n#endif\n#ifdef UV3\nelse if (v_INFONAME_==2.)\n{v_VARYINGNAME_UV=vec2(_MATRIXNAME_Matrix*vec4(uv3,1.0,0.0));}\n#endif\n#ifdef UV4\nelse if (v_INFONAME_==3.)\n{v_VARYINGNAME_UV=vec2(_MATRIXNAME_Matrix*vec4(uv4,1.0,0.0));}\n#endif\n#ifdef UV5\nelse if (v_INFONAME_==4.)\n{v_VARYINGNAME_UV=vec2(_MATRIXNAME_Matrix*vec4(uv5,1.0,0.0));}\n#endif\n#ifdef UV6\nelse if (v_INFONAME_==5.)\n{v_VARYINGNAME_UV=vec2(_MATRIXNAME_Matrix*vec4(uv6,1.0,0.0));}\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const samplerVertexImplementation = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"bumpVertex\";\nconst shader = `#if defined(BUMP) || defined(PARALLAX) || defined(CLEARCOAT_BUMP) || defined(ANISOTROPIC)\n#if defined(TANGENT) && defined(NORMAL)\nvec3 tbnNormal=normalize(normalUpdated);vec3 tbnTangent=normalize(tangentUpdated.xyz);vec3 tbnBitangent=cross(tbnNormal,tbnTangent)*tangentUpdated.w;vTBN=mat3(finalWorld)*mat3(tbnTangent,tbnBitangent,tbnNormal);\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const bumpVertex = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"clipPlaneVertex\";\nconst shader = `#ifdef CLIPPLANE\nfClipDistance=dot(worldPos,vClipPlane);\n#endif\n#ifdef CLIPPLANE2\nfClipDistance2=dot(worldPos,vClipPlane2);\n#endif\n#ifdef CLIPPLANE3\nfClipDistance3=dot(worldPos,vClipPlane3);\n#endif\n#ifdef CLIPPLANE4\nfClipDistance4=dot(worldPos,vClipPlane4);\n#endif\n#ifdef CLIPPLANE5\nfClipDistance5=dot(worldPos,vClipPlane5);\n#endif\n#ifdef CLIPPLANE6\nfClipDistance6=dot(worldPos,vClipPlane6);\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const clipPlaneVertex = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"fogVertex\";\nconst shader = `#ifdef FOG\nvFogDistance=(view*worldPos).xyz;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const fogVertex = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"shadowsVertex\";\nconst shader = `#ifdef SHADOWS\n#if defined(SHADOWCSM{X})\nvPositionFromCamera{X}=view*worldPos;for (int i=0; i\n#define CUSTOM_VERTEX_BEGIN\nattribute vec3 position;\n#ifdef NORMAL\nattribute vec3 normal;\n#endif\n#ifdef TANGENT\nattribute vec4 tangent;\n#endif\n#ifdef UV1\nattribute vec2 uv;\n#endif\n#include[2..7]\n#ifdef VERTEXCOLOR\nattribute vec4 color;\n#endif\n#include\n#include\n#include\n#include\n#include\n#include[1..7]\n#include(_DEFINENAME_,DIFFUSE,_VARYINGNAME_,Diffuse)\n#include(_DEFINENAME_,DETAIL,_VARYINGNAME_,Detail)\n#include(_DEFINENAME_,AMBIENT,_VARYINGNAME_,Ambient)\n#include(_DEFINENAME_,OPACITY,_VARYINGNAME_,Opacity)\n#include(_DEFINENAME_,EMISSIVE,_VARYINGNAME_,Emissive)\n#include(_DEFINENAME_,LIGHTMAP,_VARYINGNAME_,Lightmap)\n#if defined(SPECULARTERM)\n#include(_DEFINENAME_,SPECULAR,_VARYINGNAME_,Specular)\n#endif\n#include(_DEFINENAME_,BUMP,_VARYINGNAME_,Bump)\n#include(_DEFINENAME_,DECAL,_VARYINGNAME_,Decal)\nvarying vec3 vPositionW;\n#ifdef NORMAL\nvarying vec3 vNormalW;\n#endif\n#if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES)\nvarying vec4 vColor;\n#endif\n#include\n#include\n#include\n#include<__decl__lightVxFragment>[0..maxSimultaneousLights]\n#include\n#include[0..maxSimultaneousMorphTargets]\n#ifdef REFLECTIONMAP_SKYBOX\nvarying vec3 vPositionUVW;\n#endif\n#if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED)\nvarying vec3 vDirectionW;\n#endif\n#include\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\nvec3 positionUpdated=position;\n#ifdef NORMAL\nvec3 normalUpdated=normal;\n#endif\n#ifdef TANGENT\nvec4 tangentUpdated=tangent;\n#endif\n#ifdef UV1\nvec2 uvUpdated=uv;\n#endif\n#include\n#include[0..maxSimultaneousMorphTargets]\n#ifdef REFLECTIONMAP_SKYBOX\nvPositionUVW=positionUpdated;\n#endif\n#define CUSTOM_VERTEX_UPDATE_POSITION\n#define CUSTOM_VERTEX_UPDATE_NORMAL\n#include\n#if defined(PREPASS) && defined(PREPASS_VELOCITY) && !defined(BONES_VELOCITY_ENABLED)\nvCurrentPosition=viewProjection*finalWorld*vec4(positionUpdated,1.0);vPreviousPosition=previousViewProjection*finalPreviousWorld*vec4(positionUpdated,1.0);\n#endif\n#include\n#include\nvec4 worldPos=finalWorld*vec4(positionUpdated,1.0);\n#ifdef NORMAL\nmat3 normalWorld=mat3(finalWorld);\n#if defined(INSTANCES) && defined(THIN_INSTANCES)\nvNormalW=normalUpdated/vec3(dot(normalWorld[0],normalWorld[0]),dot(normalWorld[1],normalWorld[1]),dot(normalWorld[2],normalWorld[2]));vNormalW=normalize(normalWorld*vNormalW);\n#else\n#ifdef NONUNIFORMSCALING\nnormalWorld=transposeMat3(inverseMat3(normalWorld));\n#endif\nvNormalW=normalize(normalWorld*normalUpdated);\n#endif\n#endif\n#define CUSTOM_VERTEX_UPDATE_WORLDPOS\n#ifdef MULTIVIEW\nif (gl_ViewID_OVR==0u) {gl_Position=viewProjection*worldPos;} else {gl_Position=viewProjectionR*worldPos;}\n#else\ngl_Position=viewProjection*worldPos;\n#endif\nvPositionW=vec3(worldPos);\n#include\n#if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED)\nvDirectionW=normalize(vec3(finalWorld*vec4(positionUpdated,0.0)));\n#endif\n#ifndef UV1\nvec2 uvUpdated=vec2(0.,0.);\n#endif\n#ifdef MAINUV1\nvMainUV1=uvUpdated;\n#endif\n#include[2..7]\n#include(_DEFINENAME_,DIFFUSE,_VARYINGNAME_,Diffuse,_MATRIXNAME_,diffuse,_INFONAME_,DiffuseInfos.x)\n#include(_DEFINENAME_,DETAIL,_VARYINGNAME_,Detail,_MATRIXNAME_,detail,_INFONAME_,DetailInfos.x)\n#include(_DEFINENAME_,AMBIENT,_VARYINGNAME_,Ambient,_MATRIXNAME_,ambient,_INFONAME_,AmbientInfos.x)\n#include(_DEFINENAME_,OPACITY,_VARYINGNAME_,Opacity,_MATRIXNAME_,opacity,_INFONAME_,OpacityInfos.x)\n#include(_DEFINENAME_,EMISSIVE,_VARYINGNAME_,Emissive,_MATRIXNAME_,emissive,_INFONAME_,EmissiveInfos.x)\n#include(_DEFINENAME_,LIGHTMAP,_VARYINGNAME_,Lightmap,_MATRIXNAME_,lightmap,_INFONAME_,LightmapInfos.x)\n#if defined(SPECULARTERM)\n#include(_DEFINENAME_,SPECULAR,_VARYINGNAME_,Specular,_MATRIXNAME_,specular,_INFONAME_,SpecularInfos.x)\n#endif\n#include(_DEFINENAME_,BUMP,_VARYINGNAME_,Bump,_MATRIXNAME_,bump,_INFONAME_,BumpInfos.x)\n#include(_DEFINENAME_,DECAL,_VARYINGNAME_,Decal,_MATRIXNAME_,decal,_INFONAME_,DecalInfos.x)\n#include\n#include\n#include\n#include[0..maxSimultaneousLights]\n#include\n#include\n#include\n#define CUSTOM_VERTEX_MAIN_END\n}\n`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const defaultVertexShader = { name, shader };\n","import type { Nullable } from \"../types\";\r\nimport type { IEffectFallbacks } from \"./iEffectFallbacks\";\r\n\r\nimport type { Effect } from \"./effect\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\n\r\n/**\r\n * EffectFallbacks can be used to add fallbacks (properties to disable) to certain properties when desired to improve performance.\r\n * (Eg. Start at high quality with reflection and fog, if fps is low, remove reflection, if still low remove fog)\r\n */\r\nexport class EffectFallbacks implements IEffectFallbacks {\r\n private _defines: { [key: string]: Array } = {};\r\n\r\n private _currentRank = 32;\r\n private _maxRank = -1;\r\n\r\n private _mesh: Nullable = null;\r\n\r\n /**\r\n * Removes the fallback from the bound mesh.\r\n */\r\n public unBindMesh() {\r\n this._mesh = null;\r\n }\r\n\r\n /**\r\n * Adds a fallback on the specified property.\r\n * @param rank The rank of the fallback (Lower ranks will be fallbacked to first)\r\n * @param define The name of the define in the shader\r\n */\r\n public addFallback(rank: number, define: string): void {\r\n if (!this._defines[rank]) {\r\n if (rank < this._currentRank) {\r\n this._currentRank = rank;\r\n }\r\n\r\n if (rank > this._maxRank) {\r\n this._maxRank = rank;\r\n }\r\n\r\n this._defines[rank] = new Array();\r\n }\r\n\r\n this._defines[rank].push(define);\r\n }\r\n\r\n /**\r\n * Sets the mesh to use CPU skinning when needing to fallback.\r\n * @param rank The rank of the fallback (Lower ranks will be fallbacked to first)\r\n * @param mesh The mesh to use the fallbacks.\r\n */\r\n public addCPUSkinningFallback(rank: number, mesh: AbstractMesh) {\r\n this._mesh = mesh;\r\n\r\n if (rank < this._currentRank) {\r\n this._currentRank = rank;\r\n }\r\n if (rank > this._maxRank) {\r\n this._maxRank = rank;\r\n }\r\n }\r\n\r\n /**\r\n * Checks to see if more fallbacks are still available.\r\n */\r\n public get hasMoreFallbacks(): boolean {\r\n return this._currentRank <= this._maxRank;\r\n }\r\n\r\n /**\r\n * Removes the defines that should be removed when falling back.\r\n * @param currentDefines defines the current define statements for the shader.\r\n * @param effect defines the current effect we try to compile\r\n * @returns The resulting defines with defines of the current rank removed.\r\n */\r\n public reduce(currentDefines: string, effect: Effect): string {\r\n // First we try to switch to CPU skinning\r\n if (this._mesh && this._mesh.computeBonesUsingShaders && this._mesh.numBoneInfluencers > 0) {\r\n this._mesh.computeBonesUsingShaders = false;\r\n currentDefines = currentDefines.replace(\"#define NUM_BONE_INFLUENCERS \" + this._mesh.numBoneInfluencers, \"#define NUM_BONE_INFLUENCERS 0\");\r\n effect._bonesComputationForcedToCPU = true;\r\n\r\n const scene = this._mesh.getScene();\r\n for (let index = 0; index < scene.meshes.length; index++) {\r\n const otherMesh = scene.meshes[index];\r\n\r\n if (!otherMesh.material) {\r\n if (!this._mesh.material && otherMesh.computeBonesUsingShaders && otherMesh.numBoneInfluencers > 0) {\r\n otherMesh.computeBonesUsingShaders = false;\r\n }\r\n continue;\r\n }\r\n\r\n if (!otherMesh.computeBonesUsingShaders || otherMesh.numBoneInfluencers === 0) {\r\n continue;\r\n }\r\n\r\n if (otherMesh.material.getEffect() === effect) {\r\n otherMesh.computeBonesUsingShaders = false;\r\n } else if (otherMesh.subMeshes) {\r\n for (const subMesh of otherMesh.subMeshes) {\r\n const subMeshEffect = subMesh.effect;\r\n\r\n if (subMeshEffect === effect) {\r\n otherMesh.computeBonesUsingShaders = false;\r\n break;\r\n }\r\n }\r\n }\r\n }\r\n } else {\r\n const currentFallbacks = this._defines[this._currentRank];\r\n if (currentFallbacks) {\r\n for (let index = 0; index < currentFallbacks.length; index++) {\r\n currentDefines = currentDefines.replace(\"#define \" + currentFallbacks[index], \"\");\r\n }\r\n }\r\n\r\n this._currentRank++;\r\n }\r\n\r\n return currentDefines;\r\n }\r\n}\r\n","import type { ProcessingOptions, ShaderCustomProcessingFunction } from \"../Engines/Processors/shaderProcessingOptions\";\r\nimport type { Nullable } from \"../types\";\r\nimport { Material } from \"./material\";\r\nimport type {\r\n MaterialPluginPrepareEffect,\r\n MaterialPluginBindForSubMesh,\r\n MaterialPluginDisposed,\r\n MaterialPluginGetActiveTextures,\r\n MaterialPluginGetAnimatables,\r\n MaterialPluginGetDefineNames,\r\n MaterialPluginHasTexture,\r\n MaterialPluginIsReadyForSubMesh,\r\n MaterialPluginPrepareDefines,\r\n MaterialPluginPrepareUniformBuffer,\r\n MaterialPluginHardBindForSubMesh,\r\n MaterialPluginHasRenderTargetTextures,\r\n MaterialPluginFillRenderTargetTextures,\r\n} from \"./materialPluginEvent\";\r\nimport { MaterialPluginEvent } from \"./materialPluginEvent\";\r\nimport type { Observer } from \"core/Misc/observable\";\r\nimport { EngineStore } from \"../Engines/engineStore\";\r\n\r\nimport type { Scene } from \"../scene\";\r\nimport type { Engine } from \"../Engines/engine\";\r\nimport type { MaterialPluginBase } from \"./materialPluginBase\";\r\nimport { ShaderProcessor } from \"../Engines/Processors/shaderProcessor\";\r\nimport { ShaderLanguage } from \"./shaderLanguage\";\r\nimport { ShaderStore } from \"../Engines/shaderStore\";\r\n\r\ndeclare module \"./material\" {\r\n export interface Material {\r\n /**\r\n * Plugin manager for this material\r\n */\r\n pluginManager?: MaterialPluginManager;\r\n }\r\n}\r\n\r\nconst rxOption = new RegExp(\"^([gimus]+)!\");\r\n\r\n/**\r\n * Class that manages the plugins of a material\r\n * @since 5.0\r\n */\r\nexport class MaterialPluginManager {\r\n /** Map a plugin class name to a #define name (used in the vertex/fragment shaders as a marker of the plugin usage) */\r\n private static _MaterialPluginClassToMainDefine: { [name: string]: string } = {};\r\n private static _MaterialPluginCounter: number = 0;\r\n\r\n protected _material: Material;\r\n protected _scene: Scene;\r\n protected _engine: Engine;\r\n /** @internal */\r\n public _plugins: MaterialPluginBase[] = [];\r\n protected _activePlugins: MaterialPluginBase[] = [];\r\n protected _activePluginsForExtraEvents: MaterialPluginBase[] = [];\r\n protected _codeInjectionPoints: { [shaderType: string]: { [codeName: string]: boolean } };\r\n protected _defineNamesFromPlugins?: { [name: string]: { type: string; default: any } };\r\n protected _uboDeclaration: string;\r\n protected _vertexDeclaration: string;\r\n protected _fragmentDeclaration: string;\r\n protected _uniformList: string[];\r\n protected _samplerList: string[];\r\n protected _uboList: string[];\r\n\r\n static {\r\n EngineStore.OnEnginesDisposedObservable.add(() => {\r\n UnregisterAllMaterialPlugins();\r\n });\r\n }\r\n\r\n /**\r\n * Creates a new instance of the plugin manager\r\n * @param material material that this manager will manage the plugins for\r\n */\r\n constructor(material: Material) {\r\n this._material = material;\r\n this._scene = material.getScene();\r\n this._engine = this._scene.getEngine();\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _addPlugin(plugin: MaterialPluginBase): boolean {\r\n for (let i = 0; i < this._plugins.length; ++i) {\r\n if (this._plugins[i].name === plugin.name) {\r\n return false;\r\n }\r\n }\r\n\r\n if (this._material._uniformBufferLayoutBuilt) {\r\n throw `The plugin \"${plugin.name}\" can't be added to the material \"${this._material.name}\" because this material has already been used for rendering! Please add plugins to materials before any rendering with this material occurs.`;\r\n }\r\n\r\n const pluginClassName = plugin.getClassName();\r\n if (!MaterialPluginManager._MaterialPluginClassToMainDefine[pluginClassName]) {\r\n MaterialPluginManager._MaterialPluginClassToMainDefine[pluginClassName] = \"MATERIALPLUGIN_\" + ++MaterialPluginManager._MaterialPluginCounter;\r\n }\r\n\r\n this._material._callbackPluginEventGeneric = this._handlePluginEvent.bind(this);\r\n\r\n this._plugins.push(plugin);\r\n this._plugins.sort((a, b) => a.priority - b.priority);\r\n\r\n this._codeInjectionPoints = {};\r\n\r\n const defineNamesFromPlugins: { [name: string]: { type: string; default: any } } = {};\r\n defineNamesFromPlugins[MaterialPluginManager._MaterialPluginClassToMainDefine[pluginClassName]] = {\r\n type: \"boolean\",\r\n default: true,\r\n };\r\n\r\n for (const plugin of this._plugins) {\r\n plugin.collectDefines(defineNamesFromPlugins);\r\n this._collectPointNames(\"vertex\", plugin.getCustomCode(\"vertex\"));\r\n this._collectPointNames(\"fragment\", plugin.getCustomCode(\"fragment\"));\r\n }\r\n\r\n this._defineNamesFromPlugins = defineNamesFromPlugins;\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _activatePlugin(plugin: MaterialPluginBase): void {\r\n if (this._activePlugins.indexOf(plugin) === -1) {\r\n this._activePlugins.push(plugin);\r\n this._activePlugins.sort((a, b) => a.priority - b.priority);\r\n\r\n this._material._callbackPluginEventIsReadyForSubMesh = this._handlePluginEventIsReadyForSubMesh.bind(this);\r\n this._material._callbackPluginEventPrepareDefinesBeforeAttributes = this._handlePluginEventPrepareDefinesBeforeAttributes.bind(this);\r\n this._material._callbackPluginEventPrepareDefines = this._handlePluginEventPrepareDefines.bind(this);\r\n this._material._callbackPluginEventBindForSubMesh = this._handlePluginEventBindForSubMesh.bind(this);\r\n\r\n if (plugin.registerForExtraEvents) {\r\n this._activePluginsForExtraEvents.push(plugin);\r\n this._activePluginsForExtraEvents.sort((a, b) => a.priority - b.priority);\r\n this._material._callbackPluginEventHasRenderTargetTextures = this._handlePluginEventHasRenderTargetTextures.bind(this);\r\n this._material._callbackPluginEventFillRenderTargetTextures = this._handlePluginEventFillRenderTargetTextures.bind(this);\r\n this._material._callbackPluginEventHardBindForSubMesh = this._handlePluginEventHardBindForSubMesh.bind(this);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Gets a plugin from the list of plugins managed by this manager\r\n * @param name name of the plugin\r\n * @returns the plugin if found, else null\r\n */\r\n public getPlugin(name: string): Nullable {\r\n for (let i = 0; i < this._plugins.length; ++i) {\r\n if (this._plugins[i].name === name) {\r\n return this._plugins[i];\r\n }\r\n }\r\n return null;\r\n }\r\n\r\n protected _handlePluginEventIsReadyForSubMesh(eventData: MaterialPluginIsReadyForSubMesh): void {\r\n let isReady = true;\r\n for (const plugin of this._activePlugins) {\r\n isReady = isReady && plugin.isReadyForSubMesh(eventData.defines, this._scene, this._engine, eventData.subMesh);\r\n }\r\n eventData.isReadyForSubMesh = isReady;\r\n }\r\n\r\n protected _handlePluginEventPrepareDefinesBeforeAttributes(eventData: MaterialPluginPrepareDefines): void {\r\n for (const plugin of this._activePlugins) {\r\n plugin.prepareDefinesBeforeAttributes(eventData.defines, this._scene, eventData.mesh);\r\n }\r\n }\r\n\r\n protected _handlePluginEventPrepareDefines(eventData: MaterialPluginPrepareDefines): void {\r\n for (const plugin of this._activePlugins) {\r\n plugin.prepareDefines(eventData.defines, this._scene, eventData.mesh);\r\n }\r\n }\r\n\r\n protected _handlePluginEventHardBindForSubMesh(eventData: MaterialPluginHardBindForSubMesh): void {\r\n for (const plugin of this._activePluginsForExtraEvents) {\r\n plugin.hardBindForSubMesh(this._material._uniformBuffer, this._scene, this._engine, eventData.subMesh);\r\n }\r\n }\r\n\r\n protected _handlePluginEventBindForSubMesh(eventData: MaterialPluginBindForSubMesh): void {\r\n for (const plugin of this._activePlugins) {\r\n plugin.bindForSubMesh(this._material._uniformBuffer, this._scene, this._engine, eventData.subMesh);\r\n }\r\n }\r\n\r\n protected _handlePluginEventHasRenderTargetTextures(eventData: MaterialPluginHasRenderTargetTextures): void {\r\n let hasRenderTargetTextures = false;\r\n for (const plugin of this._activePluginsForExtraEvents) {\r\n hasRenderTargetTextures = plugin.hasRenderTargetTextures();\r\n if (hasRenderTargetTextures) {\r\n break;\r\n }\r\n }\r\n eventData.hasRenderTargetTextures = hasRenderTargetTextures;\r\n }\r\n\r\n protected _handlePluginEventFillRenderTargetTextures(eventData: MaterialPluginFillRenderTargetTextures): void {\r\n for (const plugin of this._activePluginsForExtraEvents) {\r\n plugin.fillRenderTargetTextures(eventData.renderTargets);\r\n }\r\n }\r\n\r\n protected _handlePluginEvent(\r\n id: number,\r\n info:\r\n | MaterialPluginGetActiveTextures\r\n | MaterialPluginGetAnimatables\r\n | MaterialPluginHasTexture\r\n | MaterialPluginDisposed\r\n | MaterialPluginGetDefineNames\r\n | MaterialPluginPrepareEffect\r\n | MaterialPluginPrepareUniformBuffer\r\n ): void {\r\n switch (id) {\r\n case MaterialPluginEvent.GetActiveTextures: {\r\n const eventData = info as MaterialPluginGetActiveTextures;\r\n for (const plugin of this._activePlugins) {\r\n plugin.getActiveTextures(eventData.activeTextures);\r\n }\r\n break;\r\n }\r\n\r\n case MaterialPluginEvent.GetAnimatables: {\r\n const eventData = info as MaterialPluginGetAnimatables;\r\n for (const plugin of this._activePlugins) {\r\n plugin.getAnimatables(eventData.animatables);\r\n }\r\n break;\r\n }\r\n\r\n case MaterialPluginEvent.HasTexture: {\r\n const eventData = info as MaterialPluginHasTexture;\r\n let hasTexture = false;\r\n for (const plugin of this._activePlugins) {\r\n hasTexture = plugin.hasTexture(eventData.texture);\r\n if (hasTexture) {\r\n break;\r\n }\r\n }\r\n eventData.hasTexture = hasTexture;\r\n break;\r\n }\r\n\r\n case MaterialPluginEvent.Disposed: {\r\n const eventData = info as MaterialPluginDisposed;\r\n for (const plugin of this._plugins) {\r\n plugin.dispose(eventData.forceDisposeTextures);\r\n }\r\n break;\r\n }\r\n\r\n case MaterialPluginEvent.GetDefineNames: {\r\n const eventData = info as MaterialPluginGetDefineNames;\r\n eventData.defineNames = this._defineNamesFromPlugins;\r\n break;\r\n }\r\n\r\n case MaterialPluginEvent.PrepareEffect: {\r\n const eventData = info as MaterialPluginPrepareEffect;\r\n for (const plugin of this._activePlugins) {\r\n eventData.fallbackRank = plugin.addFallbacks(eventData.defines, eventData.fallbacks, eventData.fallbackRank);\r\n plugin.getAttributes(eventData.attributes, this._scene, eventData.mesh);\r\n }\r\n if (this._uniformList.length > 0) {\r\n eventData.uniforms.push(...this._uniformList);\r\n }\r\n if (this._samplerList.length > 0) {\r\n eventData.samplers.push(...this._samplerList);\r\n }\r\n if (this._uboList.length > 0) {\r\n eventData.uniformBuffersNames.push(...this._uboList);\r\n }\r\n eventData.customCode = this._injectCustomCode(eventData, eventData.customCode);\r\n break;\r\n }\r\n\r\n case MaterialPluginEvent.PrepareUniformBuffer: {\r\n const eventData = info as MaterialPluginPrepareUniformBuffer;\r\n this._uboDeclaration = \"\";\r\n this._vertexDeclaration = \"\";\r\n this._fragmentDeclaration = \"\";\r\n this._uniformList = [];\r\n this._samplerList = [];\r\n this._uboList = [];\r\n for (const plugin of this._plugins) {\r\n const uniforms = plugin.getUniforms();\r\n if (uniforms) {\r\n if (uniforms.ubo) {\r\n for (const uniform of uniforms.ubo) {\r\n if (uniform.size && uniform.type) {\r\n const arraySize = uniform.arraySize ?? 0;\r\n eventData.ubo.addUniform(uniform.name, uniform.size, arraySize);\r\n this._uboDeclaration += `${uniform.type} ${uniform.name}${arraySize > 0 ? `[${arraySize}]` : \"\"};\\n`;\r\n }\r\n this._uniformList.push(uniform.name);\r\n }\r\n }\r\n if (uniforms.vertex) {\r\n this._vertexDeclaration += uniforms.vertex + \"\\n\";\r\n }\r\n if (uniforms.fragment) {\r\n this._fragmentDeclaration += uniforms.fragment + \"\\n\";\r\n }\r\n }\r\n plugin.getSamplers(this._samplerList);\r\n plugin.getUniformBuffersNames(this._uboList);\r\n }\r\n break;\r\n }\r\n }\r\n }\r\n\r\n protected _collectPointNames(shaderType: string, customCode: Nullable<{ [pointName: string]: string }> | undefined): void {\r\n if (!customCode) {\r\n return;\r\n }\r\n for (const pointName in customCode) {\r\n if (!this._codeInjectionPoints[shaderType]) {\r\n this._codeInjectionPoints[shaderType] = {};\r\n }\r\n this._codeInjectionPoints[shaderType][pointName] = true;\r\n }\r\n }\r\n\r\n protected _injectCustomCode(eventData: MaterialPluginPrepareEffect, existingCallback?: (shaderType: string, code: string) => string): ShaderCustomProcessingFunction {\r\n return (shaderType: string, code: string) => {\r\n if (existingCallback) {\r\n code = existingCallback(shaderType, code);\r\n }\r\n if (this._uboDeclaration) {\r\n code = code.replace(\"#define ADDITIONAL_UBO_DECLARATION\", this._uboDeclaration);\r\n }\r\n if (this._vertexDeclaration) {\r\n code = code.replace(\"#define ADDITIONAL_VERTEX_DECLARATION\", this._vertexDeclaration);\r\n }\r\n if (this._fragmentDeclaration) {\r\n code = code.replace(\"#define ADDITIONAL_FRAGMENT_DECLARATION\", this._fragmentDeclaration);\r\n }\r\n const points = this._codeInjectionPoints?.[shaderType];\r\n if (!points) {\r\n return code;\r\n }\r\n let processorOptions: Nullable = null;\r\n for (let pointName in points) {\r\n let injectedCode = \"\";\r\n for (const plugin of this._activePlugins) {\r\n let customCode = plugin.getCustomCode(shaderType)?.[pointName];\r\n if (!customCode) {\r\n continue;\r\n }\r\n if (plugin.resolveIncludes) {\r\n if (processorOptions === null) {\r\n const shaderLanguage = ShaderLanguage.GLSL;\r\n processorOptions = {\r\n defines: [], // not used by _ProcessIncludes\r\n indexParameters: eventData.indexParameters,\r\n isFragment: false,\r\n shouldUseHighPrecisionShader: this._engine._shouldUseHighPrecisionShader,\r\n processor: undefined as any, // not used by _ProcessIncludes\r\n supportsUniformBuffers: this._engine.supportsUniformBuffers,\r\n shadersRepository: ShaderStore.GetShadersRepository(shaderLanguage),\r\n includesShadersStore: ShaderStore.GetIncludesShadersStore(shaderLanguage),\r\n version: undefined as any, // not used by _ProcessIncludes\r\n platformName: this._engine.shaderPlatformName,\r\n processingContext: undefined as any, // not used by _ProcessIncludes\r\n isNDCHalfZRange: this._engine.isNDCHalfZRange,\r\n useReverseDepthBuffer: this._engine.useReverseDepthBuffer,\r\n processCodeAfterIncludes: undefined as any, // not used by _ProcessIncludes\r\n };\r\n }\r\n processorOptions.isFragment = shaderType === \"fragment\";\r\n ShaderProcessor._ProcessIncludes(customCode, processorOptions, (code) => (customCode = code));\r\n }\r\n injectedCode += customCode + \"\\n\";\r\n }\r\n if (injectedCode.length > 0) {\r\n if (pointName.charAt(0) === \"!\") {\r\n // pointName is a regular expression\r\n pointName = pointName.substring(1);\r\n\r\n let regexFlags = \"g\";\r\n if (pointName.charAt(0) === \"!\") {\r\n // no flags\r\n regexFlags = \"\";\r\n pointName = pointName.substring(1);\r\n } else {\r\n // get the flag(s)\r\n const matchOption = rxOption.exec(pointName);\r\n if (matchOption && matchOption.length >= 2) {\r\n regexFlags = matchOption[1];\r\n pointName = pointName.substring(regexFlags.length + 1);\r\n }\r\n }\r\n\r\n if (regexFlags.indexOf(\"g\") < 0) {\r\n // we force the \"g\" flag so that the regexp object is stateful!\r\n regexFlags += \"g\";\r\n }\r\n\r\n const sourceCode = code;\r\n const rx = new RegExp(pointName, regexFlags);\r\n let match = rx.exec(sourceCode);\r\n while (match !== null) {\r\n let newCode = injectedCode;\r\n for (let i = 0; i < match.length; ++i) {\r\n newCode = newCode.replace(\"$\" + i, match[i]);\r\n }\r\n code = code.replace(match[0], newCode);\r\n match = rx.exec(sourceCode);\r\n }\r\n } else {\r\n const fullPointName = \"#define \" + pointName;\r\n code = code.replace(fullPointName, \"\\n\" + injectedCode + \"\\n\" + fullPointName);\r\n }\r\n }\r\n }\r\n return code;\r\n };\r\n }\r\n}\r\n\r\n/**\r\n * Type for plugin material factories.\r\n */\r\nexport type PluginMaterialFactory = (material: Material) => Nullable;\r\n\r\nconst plugins: Array<[string, PluginMaterialFactory]> = [];\r\nlet inited = false;\r\nlet observer: Nullable> = null;\r\n\r\n/**\r\n * Registers a new material plugin through a factory, or updates it. This makes the plugin available to all materials instantiated after its registration.\r\n * @param pluginName The plugin name\r\n * @param factory The factory function which allows to create the plugin\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport function RegisterMaterialPlugin(pluginName: string, factory: PluginMaterialFactory): void {\r\n if (!inited) {\r\n observer = Material.OnEventObservable.add((material: Material) => {\r\n for (const [, factory] of plugins) {\r\n factory(material);\r\n }\r\n }, MaterialPluginEvent.Created);\r\n inited = true;\r\n }\r\n const existing = plugins.filter(([name, _factory]) => name === pluginName);\r\n if (existing.length > 0) {\r\n existing[0][1] = factory;\r\n } else {\r\n plugins.push([pluginName, factory]);\r\n }\r\n}\r\n\r\n/**\r\n * Removes a material plugin from the list of global plugins.\r\n * @param pluginName The plugin name\r\n * @returns true if the plugin has been removed, else false\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport function UnregisterMaterialPlugin(pluginName: string): boolean {\r\n for (let i = 0; i < plugins.length; ++i) {\r\n if (plugins[i][0] === pluginName) {\r\n plugins.splice(i, 1);\r\n if (plugins.length === 0) {\r\n UnregisterAllMaterialPlugins();\r\n }\r\n return true;\r\n }\r\n }\r\n return false;\r\n}\r\n\r\n/**\r\n * Clear the list of global material plugins\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport function UnregisterAllMaterialPlugins(): void {\r\n plugins.length = 0;\r\n inited = false;\r\n Material.OnEventObservable.remove(observer);\r\n observer = null;\r\n}\r\n","import { SerializationHelper, serialize } from \"../Misc/decorators\";\r\nimport type { Nullable } from \"../types\";\r\nimport { MaterialPluginManager } from \"./materialPluginManager\";\r\nimport type { SmartArray } from \"../Misc/smartArray\";\r\nimport { Constants } from \"../Engines/constants\";\r\n\r\nimport type { Engine } from \"../Engines/engine\";\r\nimport type { Scene } from \"../scene\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { SubMesh } from \"../Meshes/subMesh\";\r\nimport type { IAnimatable } from \"../Animations/animatable.interface\";\r\nimport type { UniformBuffer } from \"./uniformBuffer\";\r\nimport type { EffectFallbacks } from \"./effectFallbacks\";\r\nimport type { MaterialDefines } from \"./materialDefines\";\r\nimport type { Material } from \"./material\";\r\nimport type { BaseTexture } from \"./Textures/baseTexture\";\r\nimport type { RenderTargetTexture } from \"./Textures/renderTargetTexture\";\r\n\r\n/**\r\n * Base class for material plugins.\r\n * @since 5.0\r\n */\r\nexport class MaterialPluginBase {\r\n /**\r\n * Defines the name of the plugin\r\n */\r\n @serialize()\r\n public name: string;\r\n\r\n /**\r\n * Defines the priority of the plugin. Lower numbers run first.\r\n */\r\n @serialize()\r\n public priority: number = 500;\r\n\r\n /**\r\n * Indicates that any #include directive in the plugin code must be replaced by the corresponding code.\r\n */\r\n @serialize()\r\n public resolveIncludes: boolean = false;\r\n\r\n /**\r\n * Indicates that this plugin should be notified for the extra events (HasRenderTargetTextures / FillRenderTargetTextures / HardBindForSubMesh)\r\n */\r\n @serialize()\r\n public registerForExtraEvents: boolean = false;\r\n\r\n protected _material: Material;\r\n protected _pluginManager: MaterialPluginManager;\r\n protected _pluginDefineNames?: { [name: string]: any };\r\n\r\n protected _enable(enable: boolean) {\r\n if (enable) {\r\n this._pluginManager._activatePlugin(this);\r\n }\r\n }\r\n\r\n /**\r\n * Helper function to mark defines as being dirty.\r\n */\r\n public readonly markAllDefinesAsDirty: () => void;\r\n\r\n /**\r\n * Creates a new material plugin\r\n * @param material parent material of the plugin\r\n * @param name name of the plugin\r\n * @param priority priority of the plugin\r\n * @param defines list of defines used by the plugin. The value of the property is the default value for this property\r\n * @param addToPluginList true to add the plugin to the list of plugins managed by the material plugin manager of the material (default: true)\r\n * @param enable true to enable the plugin (it is handy if the plugin does not handle properties to switch its current activation)\r\n * @param resolveIncludes Indicates that any #include directive in the plugin code must be replaced by the corresponding code (default: false)\r\n */\r\n constructor(material: Material, name: string, priority: number, defines?: { [key: string]: any }, addToPluginList = true, enable = false, resolveIncludes = false) {\r\n this._material = material;\r\n this.name = name;\r\n this.priority = priority;\r\n this.resolveIncludes = resolveIncludes;\r\n\r\n if (!material.pluginManager) {\r\n material.pluginManager = new MaterialPluginManager(material);\r\n material.onDisposeObservable.add(() => {\r\n material.pluginManager = undefined;\r\n });\r\n }\r\n\r\n this._pluginDefineNames = defines;\r\n this._pluginManager = material.pluginManager;\r\n\r\n if (addToPluginList) {\r\n this._pluginManager._addPlugin(this);\r\n }\r\n\r\n if (enable) {\r\n this._enable(true);\r\n }\r\n\r\n this.markAllDefinesAsDirty = material._dirtyCallbacks[Constants.MATERIAL_AllDirtyFlag];\r\n }\r\n\r\n /**\r\n * Gets the current class name useful for serialization or dynamic coding.\r\n * @returns The class name.\r\n */\r\n public getClassName(): string {\r\n return \"MaterialPluginBase\";\r\n }\r\n\r\n /**\r\n * Specifies that the submesh is ready to be used.\r\n * @param defines the list of \"defines\" to update.\r\n * @param scene defines the scene the material belongs to.\r\n * @param engine the engine this scene belongs to.\r\n * @param subMesh the submesh to check for readiness\r\n * @returns - boolean indicating that the submesh is ready or not.\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public isReadyForSubMesh(defines: MaterialDefines, scene: Scene, engine: Engine, subMesh: SubMesh): boolean {\r\n return true;\r\n }\r\n\r\n /**\r\n * Binds the material data (this function is called even if mustRebind() returns false)\r\n * @param uniformBuffer defines the Uniform buffer to fill in.\r\n * @param scene defines the scene the material belongs to.\r\n * @param engine defines the engine the material belongs to.\r\n * @param subMesh the submesh to bind data for\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public hardBindForSubMesh(uniformBuffer: UniformBuffer, scene: Scene, engine: Engine, subMesh: SubMesh): void {}\r\n\r\n /**\r\n * Binds the material data.\r\n * @param uniformBuffer defines the Uniform buffer to fill in.\r\n * @param scene defines the scene the material belongs to.\r\n * @param engine the engine this scene belongs to.\r\n * @param subMesh the submesh to bind data for\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public bindForSubMesh(uniformBuffer: UniformBuffer, scene: Scene, engine: Engine, subMesh: SubMesh): void {}\r\n\r\n /**\r\n * Disposes the resources of the material.\r\n * @param forceDisposeTextures - Forces the disposal of all textures.\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public dispose(forceDisposeTextures?: boolean): void {}\r\n\r\n /**\r\n * Returns a list of custom shader code fragments to customize the shader.\r\n * @param shaderType \"vertex\" or \"fragment\"\r\n * @returns null if no code to be added, or a list of pointName =\\> code.\r\n * Note that `pointName` can also be a regular expression if it starts with a `!`.\r\n * In that case, the string found by the regular expression (if any) will be\r\n * replaced by the code provided.\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public getCustomCode(shaderType: string): Nullable<{ [pointName: string]: string }> {\r\n return null;\r\n }\r\n\r\n /**\r\n * Collects all defines.\r\n * @param defines The object to append to.\r\n */\r\n public collectDefines(defines: { [name: string]: { type: string; default: any } }): void {\r\n if (!this._pluginDefineNames) {\r\n return;\r\n }\r\n for (const key of Object.keys(this._pluginDefineNames)) {\r\n if (key[0] === \"_\") {\r\n continue;\r\n }\r\n\r\n const type = typeof this._pluginDefineNames[key];\r\n defines[key] = {\r\n type: type === \"number\" ? \"number\" : type === \"string\" ? \"string\" : type === \"boolean\" ? \"boolean\" : \"object\",\r\n default: this._pluginDefineNames[key],\r\n };\r\n }\r\n }\r\n\r\n /**\r\n * Sets the defines for the next rendering. Called before MaterialHelper.PrepareDefinesForAttributes is called.\r\n * @param defines the list of \"defines\" to update.\r\n * @param scene defines the scene to the material belongs to.\r\n * @param mesh the mesh being rendered\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public prepareDefinesBeforeAttributes(defines: MaterialDefines, scene: Scene, mesh: AbstractMesh): void {}\r\n\r\n /**\r\n * Sets the defines for the next rendering\r\n * @param defines the list of \"defines\" to update.\r\n * @param scene defines the scene to the material belongs to.\r\n * @param mesh the mesh being rendered\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public prepareDefines(defines: MaterialDefines, scene: Scene, mesh: AbstractMesh): void {}\r\n\r\n /**\r\n * Checks to see if a texture is used in the material.\r\n * @param texture - Base texture to use.\r\n * @returns - Boolean specifying if a texture is used in the material.\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public hasTexture(texture: BaseTexture): boolean {\r\n return false;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating that current material needs to register RTT\r\n * @returns true if this uses a render target otherwise false.\r\n */\r\n public hasRenderTargetTextures(): boolean {\r\n return false;\r\n }\r\n\r\n /**\r\n * Fills the list of render target textures.\r\n * @param renderTargets the list of render targets to update\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public fillRenderTargetTextures(renderTargets: SmartArray): void {}\r\n\r\n /**\r\n * Returns an array of the actively used textures.\r\n * @param activeTextures Array of BaseTextures\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public getActiveTextures(activeTextures: BaseTexture[]): void {}\r\n\r\n /**\r\n * Returns the animatable textures.\r\n * @param animatables Array of animatable textures.\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public getAnimatables(animatables: IAnimatable[]): void {}\r\n\r\n /**\r\n * Add fallbacks to the effect fallbacks list.\r\n * @param defines defines the Base texture to use.\r\n * @param fallbacks defines the current fallback list.\r\n * @param currentRank defines the current fallback rank.\r\n * @returns the new fallback rank.\r\n */\r\n public addFallbacks(defines: MaterialDefines, fallbacks: EffectFallbacks, currentRank: number): number {\r\n return currentRank;\r\n }\r\n\r\n /**\r\n * Gets the samplers used by the plugin.\r\n * @param samplers list that the sampler names should be added to.\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public getSamplers(samplers: string[]): void {}\r\n\r\n /**\r\n * Gets the attributes used by the plugin.\r\n * @param attributes list that the attribute names should be added to.\r\n * @param scene the scene that the material belongs to.\r\n * @param mesh the mesh being rendered.\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public getAttributes(attributes: string[], scene: Scene, mesh: AbstractMesh): void {}\r\n\r\n /**\r\n * Gets the uniform buffers names added by the plugin.\r\n * @param ubos list that the ubo names should be added to.\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public getUniformBuffersNames(ubos: string[]): void {}\r\n\r\n /**\r\n * Gets the description of the uniforms to add to the ubo (if engine supports ubos) or to inject directly in the vertex/fragment shaders (if engine does not support ubos)\r\n * @returns the description of the uniforms\r\n */\r\n public getUniforms(): { ubo?: Array<{ name: string; size?: number; type?: string; arraySize?: number }>; vertex?: string; fragment?: string } {\r\n return {};\r\n }\r\n\r\n /**\r\n * Makes a duplicate of the current configuration into another one.\r\n * @param plugin define the config where to copy the info\r\n */\r\n public copyTo(plugin: MaterialPluginBase): void {\r\n SerializationHelper.Clone(() => plugin, this);\r\n }\r\n\r\n /**\r\n * Serializes this plugin configuration.\r\n * @returns - An object with the serialized config.\r\n */\r\n public serialize(): any {\r\n return SerializationHelper.Serialize(this);\r\n }\r\n\r\n /**\r\n * Parses a plugin configuration from a serialized object.\r\n * @param source - Serialized object.\r\n * @param scene Defines the scene we are parsing for\r\n * @param rootUrl Defines the rootUrl to load from\r\n */\r\n public parse(source: any, scene: Scene, rootUrl: string): void {\r\n SerializationHelper.Parse(() => this, source, scene, rootUrl);\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport type { Nullable } from \"../types\";\r\nimport { Material } from \"./material\";\r\nimport { serialize, expandToProperty, serializeAsTexture } from \"../Misc/decorators\";\r\nimport { MaterialFlags } from \"./materialFlags\";\r\nimport { MaterialHelper } from \"./materialHelper\";\r\nimport type { BaseTexture } from \"./Textures/baseTexture\";\r\nimport type { UniformBuffer } from \"./uniformBuffer\";\r\nimport type { IAnimatable } from \"../Animations/animatable.interface\";\r\nimport { MaterialDefines } from \"./materialDefines\";\r\nimport { MaterialPluginBase } from \"./materialPluginBase\";\r\nimport { Constants } from \"../Engines/constants\";\r\n\r\nimport type { Engine } from \"../Engines/engine\";\r\nimport type { Scene } from \"../scene\";\r\nimport type { StandardMaterial } from \"./standardMaterial\";\r\nimport type { PBRBaseMaterial } from \"./PBR/pbrBaseMaterial\";\r\n\r\n/**\r\n * @internal\r\n */\r\nexport class MaterialDetailMapDefines extends MaterialDefines {\r\n DETAIL = false;\r\n DETAILDIRECTUV = 0;\r\n DETAIL_NORMALBLENDMETHOD = 0;\r\n}\r\n\r\n/**\r\n * Plugin that implements the detail map component of a material\r\n *\r\n * Inspired from:\r\n * Unity: https://docs.unity3d.com/Packages/com.unity.render-pipelines.high-definition@9.0/manual/Mask-Map-and-Detail-Map.html and https://docs.unity3d.com/Manual/StandardShaderMaterialParameterDetail.html\r\n * Unreal: https://docs.unrealengine.com/en-US/Engine/Rendering/Materials/HowTo/DetailTexturing/index.html\r\n * Cryengine: https://docs.cryengine.com/display/SDKDOC2/Detail+Maps\r\n */\r\nexport class DetailMapConfiguration extends MaterialPluginBase {\r\n private _texture: Nullable = null;\r\n /**\r\n * The detail texture of the material.\r\n */\r\n @serializeAsTexture(\"detailTexture\")\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public texture: Nullable;\r\n\r\n /**\r\n * Defines how strongly the detail diffuse/albedo channel is blended with the regular diffuse/albedo texture\r\n * Bigger values mean stronger blending\r\n */\r\n @serialize()\r\n public diffuseBlendLevel = 1;\r\n\r\n /**\r\n * Defines how strongly the detail roughness channel is blended with the regular roughness value\r\n * Bigger values mean stronger blending. Only used with PBR materials\r\n */\r\n @serialize()\r\n public roughnessBlendLevel = 1;\r\n\r\n /**\r\n * Defines how strong the bump effect from the detail map is\r\n * Bigger values mean stronger effect\r\n */\r\n @serialize()\r\n public bumpLevel = 1;\r\n\r\n private _normalBlendMethod = Material.MATERIAL_NORMALBLENDMETHOD_WHITEOUT;\r\n /**\r\n * The method used to blend the bump and detail normals together\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public normalBlendMethod: number;\r\n\r\n private _isEnabled = false;\r\n /**\r\n * Enable or disable the detail map on this material\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public isEnabled = false;\r\n\r\n /** @internal */\r\n private _internalMarkAllSubMeshesAsTexturesDirty: () => void;\r\n\r\n /** @internal */\r\n public _markAllSubMeshesAsTexturesDirty(): void {\r\n this._enable(this._isEnabled);\r\n this._internalMarkAllSubMeshesAsTexturesDirty();\r\n }\r\n\r\n constructor(material: PBRBaseMaterial | StandardMaterial, addToPluginList = true) {\r\n super(material, \"DetailMap\", 140, new MaterialDetailMapDefines(), addToPluginList);\r\n\r\n this._internalMarkAllSubMeshesAsTexturesDirty = material._dirtyCallbacks[Constants.MATERIAL_TextureDirtyFlag];\r\n }\r\n\r\n public isReadyForSubMesh(defines: MaterialDetailMapDefines, scene: Scene, engine: Engine): boolean {\r\n if (!this._isEnabled) {\r\n return true;\r\n }\r\n\r\n if (defines._areTexturesDirty && scene.texturesEnabled) {\r\n if (engine.getCaps().standardDerivatives && this._texture && MaterialFlags.DetailTextureEnabled) {\r\n // Detail texture cannot be not blocking.\r\n if (!this._texture.isReady()) {\r\n return false;\r\n }\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n public prepareDefines(defines: MaterialDetailMapDefines, scene: Scene): void {\r\n if (this._isEnabled) {\r\n defines.DETAIL_NORMALBLENDMETHOD = this._normalBlendMethod;\r\n\r\n const engine = scene.getEngine();\r\n\r\n if (defines._areTexturesDirty) {\r\n if (engine.getCaps().standardDerivatives && this._texture && MaterialFlags.DetailTextureEnabled && this._isEnabled) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._texture, defines, \"DETAIL\");\r\n defines.DETAIL_NORMALBLENDMETHOD = this._normalBlendMethod;\r\n } else {\r\n defines.DETAIL = false;\r\n }\r\n }\r\n } else {\r\n defines.DETAIL = false;\r\n }\r\n }\r\n\r\n public bindForSubMesh(uniformBuffer: UniformBuffer, scene: Scene): void {\r\n if (!this._isEnabled) {\r\n return;\r\n }\r\n\r\n const isFrozen = this._material.isFrozen;\r\n\r\n if (!uniformBuffer.useUbo || !isFrozen || !uniformBuffer.isSync) {\r\n if (this._texture && MaterialFlags.DetailTextureEnabled) {\r\n uniformBuffer.updateFloat4(\"vDetailInfos\", this._texture.coordinatesIndex, this.diffuseBlendLevel, this.bumpLevel, this.roughnessBlendLevel);\r\n MaterialHelper.BindTextureMatrix(this._texture, uniformBuffer, \"detail\");\r\n }\r\n }\r\n\r\n // Textures\r\n if (scene.texturesEnabled) {\r\n if (this._texture && MaterialFlags.DetailTextureEnabled) {\r\n uniformBuffer.setTexture(\"detailSampler\", this._texture);\r\n }\r\n }\r\n }\r\n\r\n public hasTexture(texture: BaseTexture): boolean {\r\n if (this._texture === texture) {\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n public getActiveTextures(activeTextures: BaseTexture[]): void {\r\n if (this._texture) {\r\n activeTextures.push(this._texture);\r\n }\r\n }\r\n\r\n public getAnimatables(animatables: IAnimatable[]): void {\r\n if (this._texture && this._texture.animations && this._texture.animations.length > 0) {\r\n animatables.push(this._texture);\r\n }\r\n }\r\n\r\n public dispose(forceDisposeTextures?: boolean): void {\r\n if (forceDisposeTextures) {\r\n this._texture?.dispose();\r\n }\r\n }\r\n\r\n public getClassName(): string {\r\n return \"DetailMapConfiguration\";\r\n }\r\n\r\n public getSamplers(samplers: string[]): void {\r\n samplers.push(\"detailSampler\");\r\n }\r\n\r\n public getUniforms(): { ubo?: Array<{ name: string; size: number; type: string }>; vertex?: string; fragment?: string } {\r\n return {\r\n ubo: [\r\n { name: \"vDetailInfos\", size: 4, type: \"vec4\" },\r\n { name: \"detailMatrix\", size: 16, type: \"mat4\" },\r\n ],\r\n };\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport { serialize, SerializationHelper, serializeAsColor3, expandToProperty, serializeAsFresnelParameters, serializeAsTexture } from \"../Misc/decorators\";\r\nimport type { Observer } from \"../Misc/observable\";\r\nimport { SmartArray } from \"../Misc/smartArray\";\r\nimport type { IAnimatable } from \"../Animations/animatable.interface\";\r\n\r\nimport type { Nullable } from \"../types\";\r\nimport { Scene } from \"../scene\";\r\nimport { Matrix } from \"../Maths/math.vector\";\r\nimport { Color3 } from \"../Maths/math.color\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport type { SubMesh } from \"../Meshes/subMesh\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { Mesh } from \"../Meshes/mesh\";\r\nimport { PrePassConfiguration } from \"./prePassConfiguration\";\r\n\r\nimport type { IImageProcessingConfigurationDefines } from \"./imageProcessingConfiguration\";\r\nimport { ImageProcessingConfiguration } from \"./imageProcessingConfiguration\";\r\nimport type { ColorCurves } from \"./colorCurves\";\r\nimport type { FresnelParameters } from \"./fresnelParameters\";\r\nimport type { ICustomShaderNameResolveOptions } from \"../Materials/material\";\r\nimport { Material } from \"../Materials/material\";\r\nimport { MaterialPluginEvent } from \"./materialPluginEvent\";\r\nimport { MaterialDefines } from \"../Materials/materialDefines\";\r\nimport { PushMaterial } from \"./pushMaterial\";\r\nimport { MaterialHelper } from \"./materialHelper\";\r\n\r\nimport type { BaseTexture } from \"../Materials/Textures/baseTexture\";\r\nimport { Texture } from \"../Materials/Textures/texture\";\r\nimport type { CubeTexture } from \"../Materials/Textures/cubeTexture\";\r\nimport type { RenderTargetTexture } from \"../Materials/Textures/renderTargetTexture\";\r\nimport { RegisterClass } from \"../Misc/typeStore\";\r\nimport { MaterialFlags } from \"./materialFlags\";\r\n\r\nimport \"../Shaders/default.fragment\";\r\nimport \"../Shaders/default.vertex\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport { EffectFallbacks } from \"./effectFallbacks\";\r\nimport type { Effect, IEffectCreationOptions } from \"./effect\";\r\nimport { DetailMapConfiguration } from \"./material.detailMapConfiguration\";\r\nimport { addClipPlaneUniforms, bindClipPlane } from \"./clipPlaneMaterialHelper\";\r\n\r\nconst onCreatedEffectParameters = { effect: null as unknown as Effect, subMesh: null as unknown as Nullable };\r\n\r\n/** @internal */\r\nexport class StandardMaterialDefines extends MaterialDefines implements IImageProcessingConfigurationDefines {\r\n public MAINUV1 = false;\r\n public MAINUV2 = false;\r\n public MAINUV3 = false;\r\n public MAINUV4 = false;\r\n public MAINUV5 = false;\r\n public MAINUV6 = false;\r\n public DIFFUSE = false;\r\n public DIFFUSEDIRECTUV = 0;\r\n public BAKED_VERTEX_ANIMATION_TEXTURE = false;\r\n public AMBIENT = false;\r\n public AMBIENTDIRECTUV = 0;\r\n public OPACITY = false;\r\n public OPACITYDIRECTUV = 0;\r\n public OPACITYRGB = false;\r\n public REFLECTION = false;\r\n public EMISSIVE = false;\r\n public EMISSIVEDIRECTUV = 0;\r\n public SPECULAR = false;\r\n public SPECULARDIRECTUV = 0;\r\n public BUMP = false;\r\n public BUMPDIRECTUV = 0;\r\n public PARALLAX = false;\r\n public PARALLAXOCCLUSION = false;\r\n public SPECULAROVERALPHA = false;\r\n public CLIPPLANE = false;\r\n public CLIPPLANE2 = false;\r\n public CLIPPLANE3 = false;\r\n public CLIPPLANE4 = false;\r\n public CLIPPLANE5 = false;\r\n public CLIPPLANE6 = false;\r\n public ALPHATEST = false;\r\n public DEPTHPREPASS = false;\r\n public ALPHAFROMDIFFUSE = false;\r\n public POINTSIZE = false;\r\n public FOG = false;\r\n public SPECULARTERM = false;\r\n public DIFFUSEFRESNEL = false;\r\n public OPACITYFRESNEL = false;\r\n public REFLECTIONFRESNEL = false;\r\n public REFRACTIONFRESNEL = false;\r\n public EMISSIVEFRESNEL = false;\r\n public FRESNEL = false;\r\n public NORMAL = false;\r\n public TANGENT = false;\r\n public UV1 = false;\r\n public UV2 = false;\r\n public UV3 = false;\r\n public UV4 = false;\r\n public UV5 = false;\r\n public UV6 = false;\r\n public VERTEXCOLOR = false;\r\n public VERTEXALPHA = false;\r\n public NUM_BONE_INFLUENCERS = 0;\r\n public BonesPerMesh = 0;\r\n public BONETEXTURE = false;\r\n public BONES_VELOCITY_ENABLED = false;\r\n public INSTANCES = false;\r\n public THIN_INSTANCES = false;\r\n public INSTANCESCOLOR = false;\r\n public GLOSSINESS = false;\r\n public ROUGHNESS = false;\r\n public EMISSIVEASILLUMINATION = false;\r\n public LINKEMISSIVEWITHDIFFUSE = false;\r\n public REFLECTIONFRESNELFROMSPECULAR = false;\r\n public LIGHTMAP = false;\r\n public LIGHTMAPDIRECTUV = 0;\r\n public OBJECTSPACE_NORMALMAP = false;\r\n public USELIGHTMAPASSHADOWMAP = false;\r\n public REFLECTIONMAP_3D = false;\r\n public REFLECTIONMAP_SPHERICAL = false;\r\n public REFLECTIONMAP_PLANAR = false;\r\n public REFLECTIONMAP_CUBIC = false;\r\n public USE_LOCAL_REFLECTIONMAP_CUBIC = false;\r\n public USE_LOCAL_REFRACTIONMAP_CUBIC = false;\r\n public REFLECTIONMAP_PROJECTION = false;\r\n public REFLECTIONMAP_SKYBOX = false;\r\n public REFLECTIONMAP_EXPLICIT = false;\r\n public REFLECTIONMAP_EQUIRECTANGULAR = false;\r\n public REFLECTIONMAP_EQUIRECTANGULAR_FIXED = false;\r\n public REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = false;\r\n public REFLECTIONMAP_OPPOSITEZ = false;\r\n public INVERTCUBICMAP = false;\r\n public LOGARITHMICDEPTH = false;\r\n public REFRACTION = false;\r\n public REFRACTIONMAP_3D = false;\r\n public REFLECTIONOVERALPHA = false;\r\n public TWOSIDEDLIGHTING = false;\r\n public SHADOWFLOAT = false;\r\n public MORPHTARGETS = false;\r\n public MORPHTARGETS_NORMAL = false;\r\n public MORPHTARGETS_TANGENT = false;\r\n public MORPHTARGETS_UV = false;\r\n public NUM_MORPH_INFLUENCERS = 0;\r\n public MORPHTARGETS_TEXTURE = false;\r\n public NONUNIFORMSCALING = false; // https://playground.babylonjs.com#V6DWIH\r\n public PREMULTIPLYALPHA = false; // https://playground.babylonjs.com#LNVJJ7\r\n public ALPHATEST_AFTERALLALPHACOMPUTATIONS = false;\r\n public ALPHABLEND = true;\r\n\r\n public PREPASS = false;\r\n public PREPASS_IRRADIANCE = false;\r\n public PREPASS_IRRADIANCE_INDEX = -1;\r\n public PREPASS_ALBEDO_SQRT = false;\r\n public PREPASS_ALBEDO_SQRT_INDEX = -1;\r\n public PREPASS_DEPTH = false;\r\n public PREPASS_DEPTH_INDEX = -1;\r\n public PREPASS_NORMAL = false;\r\n public PREPASS_NORMAL_INDEX = -1;\r\n public PREPASS_POSITION = false;\r\n public PREPASS_POSITION_INDEX = -1;\r\n public PREPASS_VELOCITY = false;\r\n public PREPASS_VELOCITY_INDEX = -1;\r\n public PREPASS_REFLECTIVITY = false;\r\n public PREPASS_REFLECTIVITY_INDEX = -1;\r\n public SCENE_MRT_COUNT = 0;\r\n\r\n public RGBDLIGHTMAP = false;\r\n public RGBDREFLECTION = false;\r\n public RGBDREFRACTION = false;\r\n\r\n public IMAGEPROCESSING = false;\r\n public VIGNETTE = false;\r\n public VIGNETTEBLENDMODEMULTIPLY = false;\r\n public VIGNETTEBLENDMODEOPAQUE = false;\r\n public TONEMAPPING = false;\r\n public TONEMAPPING_ACES = false;\r\n public CONTRAST = false;\r\n public COLORCURVES = false;\r\n public COLORGRADING = false;\r\n public COLORGRADING3D = false;\r\n public SAMPLER3DGREENDEPTH = false;\r\n public SAMPLER3DBGRMAP = false;\r\n public DITHER = false;\r\n public IMAGEPROCESSINGPOSTPROCESS = false;\r\n public SKIPFINALCOLORCLAMP = false;\r\n public MULTIVIEW = false;\r\n public ORDER_INDEPENDENT_TRANSPARENCY = false;\r\n public ORDER_INDEPENDENT_TRANSPARENCY_16BITS = false;\r\n public CAMERA_ORTHOGRAPHIC = false;\r\n public CAMERA_PERSPECTIVE = false;\r\n\r\n /**\r\n * If the reflection texture on this material is in linear color space\r\n * @internal\r\n */\r\n public IS_REFLECTION_LINEAR = false;\r\n /**\r\n * If the refraction texture on this material is in linear color space\r\n * @internal\r\n */\r\n public IS_REFRACTION_LINEAR = false;\r\n public EXPOSURE = false;\r\n\r\n public DECAL_AFTER_DETAIL = false;\r\n\r\n /**\r\n * Initializes the Standard Material defines.\r\n * @param externalProperties The external properties\r\n */\r\n constructor(externalProperties?: { [name: string]: { type: string; default: any } }) {\r\n super(externalProperties);\r\n this.rebuild();\r\n }\r\n\r\n public setReflectionMode(modeToEnable: string) {\r\n const modes = [\r\n \"REFLECTIONMAP_CUBIC\",\r\n \"REFLECTIONMAP_EXPLICIT\",\r\n \"REFLECTIONMAP_PLANAR\",\r\n \"REFLECTIONMAP_PROJECTION\",\r\n \"REFLECTIONMAP_PROJECTION\",\r\n \"REFLECTIONMAP_SKYBOX\",\r\n \"REFLECTIONMAP_SPHERICAL\",\r\n \"REFLECTIONMAP_EQUIRECTANGULAR\",\r\n \"REFLECTIONMAP_EQUIRECTANGULAR_FIXED\",\r\n \"REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED\",\r\n ];\r\n\r\n for (const mode of modes) {\r\n (this)[mode] = mode === modeToEnable;\r\n }\r\n }\r\n}\r\n\r\n/**\r\n * This is the default material used in Babylon. It is the best trade off between quality\r\n * and performances.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/materials_introduction\r\n */\r\nexport class StandardMaterial extends PushMaterial {\r\n @serializeAsTexture(\"diffuseTexture\")\r\n private _diffuseTexture: Nullable = null;\r\n /**\r\n * The basic texture of the material as viewed under a light.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesAndMiscDirty\")\r\n public diffuseTexture: Nullable;\r\n\r\n @serializeAsTexture(\"ambientTexture\")\r\n private _ambientTexture: Nullable = null;\r\n /**\r\n * AKA Occlusion Texture in other nomenclature, it helps adding baked shadows into your material.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public ambientTexture: Nullable;\r\n\r\n @serializeAsTexture(\"opacityTexture\")\r\n private _opacityTexture: Nullable = null;\r\n /**\r\n * Define the transparency of the material from a texture.\r\n * The final alpha value can be read either from the red channel (if texture.getAlphaFromRGB is false)\r\n * or from the luminance or the current texel (if texture.getAlphaFromRGB is true)\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesAndMiscDirty\")\r\n public opacityTexture: Nullable;\r\n\r\n @serializeAsTexture(\"reflectionTexture\")\r\n private _reflectionTexture: Nullable = null;\r\n /**\r\n * Define the texture used to display the reflection.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#how-to-obtain-reflections-and-refractions\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public reflectionTexture: Nullable;\r\n\r\n @serializeAsTexture(\"emissiveTexture\")\r\n private _emissiveTexture: Nullable = null;\r\n /**\r\n * Define texture of the material as if self lit.\r\n * This will be mixed in the final result even in the absence of light.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public emissiveTexture: Nullable;\r\n\r\n @serializeAsTexture(\"specularTexture\")\r\n private _specularTexture: Nullable = null;\r\n /**\r\n * Define how the color and intensity of the highlight given by the light in the material.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public specularTexture: Nullable;\r\n\r\n @serializeAsTexture(\"bumpTexture\")\r\n private _bumpTexture: Nullable = null;\r\n /**\r\n * Bump mapping is a technique to simulate bump and dents on a rendered surface.\r\n * These are made by creating a normal map from an image. The means to do this can be found on the web, a search for 'normal map generator' will bring up free and paid for methods of doing this.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/moreMaterials#bump-map\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public bumpTexture: Nullable;\r\n\r\n @serializeAsTexture(\"lightmapTexture\")\r\n private _lightmapTexture: Nullable = null;\r\n /**\r\n * Complex lighting can be computationally expensive to compute at runtime.\r\n * To save on computation, lightmaps may be used to store calculated lighting in a texture which will be applied to a given mesh.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction#lightmaps\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public lightmapTexture: Nullable;\r\n\r\n @serializeAsTexture(\"refractionTexture\")\r\n private _refractionTexture: Nullable = null;\r\n /**\r\n * Define the texture used to display the refraction.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#how-to-obtain-reflections-and-refractions\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public refractionTexture: Nullable;\r\n\r\n /**\r\n * The color of the material lit by the environmental background lighting.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/materials_introduction#ambient-color-example\r\n */\r\n @serializeAsColor3(\"ambient\")\r\n public ambientColor = new Color3(0, 0, 0);\r\n\r\n /**\r\n * The basic color of the material as viewed under a light.\r\n */\r\n @serializeAsColor3(\"diffuse\")\r\n public diffuseColor = new Color3(1, 1, 1);\r\n\r\n /**\r\n * Define how the color and intensity of the highlight given by the light in the material.\r\n */\r\n @serializeAsColor3(\"specular\")\r\n public specularColor = new Color3(1, 1, 1);\r\n\r\n /**\r\n * Define the color of the material as if self lit.\r\n * This will be mixed in the final result even in the absence of light.\r\n */\r\n @serializeAsColor3(\"emissive\")\r\n public emissiveColor = new Color3(0, 0, 0);\r\n\r\n /**\r\n * Defines how sharp are the highlights in the material.\r\n * The bigger the value the sharper giving a more glossy feeling to the result.\r\n * Reversely, the smaller the value the blurrier giving a more rough feeling to the result.\r\n */\r\n @serialize()\r\n public specularPower = 64;\r\n\r\n @serialize(\"useAlphaFromDiffuseTexture\")\r\n private _useAlphaFromDiffuseTexture = false;\r\n /**\r\n * Does the transparency come from the diffuse texture alpha channel.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesAndMiscDirty\")\r\n public useAlphaFromDiffuseTexture: boolean;\r\n\r\n @serialize(\"useEmissiveAsIllumination\")\r\n private _useEmissiveAsIllumination = false;\r\n /**\r\n * If true, the emissive value is added into the end result, otherwise it is multiplied in.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useEmissiveAsIllumination: boolean;\r\n\r\n @serialize(\"linkEmissiveWithDiffuse\")\r\n private _linkEmissiveWithDiffuse = false;\r\n /**\r\n * If true, some kind of energy conservation will prevent the end result to be more than 1 by reducing\r\n * the emissive level when the final color is close to one.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public linkEmissiveWithDiffuse: boolean;\r\n\r\n @serialize(\"useSpecularOverAlpha\")\r\n private _useSpecularOverAlpha = false;\r\n /**\r\n * Specifies that the material will keep the specular highlights over a transparent surface (only the most luminous ones).\r\n * A car glass is a good exemple of that. When sun reflects on it you can not see what is behind.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useSpecularOverAlpha: boolean;\r\n\r\n @serialize(\"useReflectionOverAlpha\")\r\n private _useReflectionOverAlpha = false;\r\n /**\r\n * Specifies that the material will keeps the reflection highlights over a transparent surface (only the most luminous ones).\r\n * A car glass is a good exemple of that. When the street lights reflects on it you can not see what is behind.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useReflectionOverAlpha: boolean;\r\n\r\n @serialize(\"disableLighting\")\r\n private _disableLighting = false;\r\n /**\r\n * Does lights from the scene impacts this material.\r\n * It can be a nice trick for performance to disable lighting on a fully emissive material.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsLightsDirty\")\r\n public disableLighting: boolean;\r\n\r\n @serialize(\"useObjectSpaceNormalMap\")\r\n private _useObjectSpaceNormalMap = false;\r\n /**\r\n * Allows using an object space normal map (instead of tangent space).\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useObjectSpaceNormalMap: boolean;\r\n\r\n @serialize(\"useParallax\")\r\n private _useParallax = false;\r\n /**\r\n * Is parallax enabled or not.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/parallaxMapping\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useParallax: boolean;\r\n\r\n @serialize(\"useParallaxOcclusion\")\r\n private _useParallaxOcclusion = false;\r\n /**\r\n * Is parallax occlusion enabled or not.\r\n * If true, the outcome is way more realistic than traditional Parallax but you can expect a performance hit that worthes consideration.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/parallaxMapping\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useParallaxOcclusion: boolean;\r\n\r\n /**\r\n * Apply a scaling factor that determine which \"depth\" the height map should reprensent. A value between 0.05 and 0.1 is reasonnable in Parallax, you can reach 0.2 using Parallax Occlusion.\r\n */\r\n @serialize()\r\n public parallaxScaleBias = 0.05;\r\n\r\n @serialize(\"roughness\")\r\n private _roughness = 0;\r\n /**\r\n * Helps to define how blurry the reflections should appears in the material.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public roughness: number;\r\n\r\n /**\r\n * In case of refraction, define the value of the index of refraction.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#how-to-obtain-reflections-and-refractions\r\n */\r\n @serialize()\r\n public indexOfRefraction = 0.98;\r\n\r\n /**\r\n * Invert the refraction texture alongside the y axis.\r\n * It can be useful with procedural textures or probe for instance.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#how-to-obtain-reflections-and-refractions\r\n */\r\n @serialize()\r\n public invertRefractionY = true;\r\n\r\n /**\r\n * Defines the alpha limits in alpha test mode.\r\n */\r\n @serialize()\r\n public alphaCutOff = 0.4;\r\n\r\n @serialize(\"useLightmapAsShadowmap\")\r\n private _useLightmapAsShadowmap = false;\r\n /**\r\n * In case of light mapping, define whether the map contains light or shadow informations.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useLightmapAsShadowmap: boolean;\r\n\r\n // Fresnel\r\n @serializeAsFresnelParameters(\"diffuseFresnelParameters\")\r\n private _diffuseFresnelParameters: FresnelParameters;\r\n /**\r\n * Define the diffuse fresnel parameters of the material.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/fresnelParameters\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsFresnelDirty\")\r\n public diffuseFresnelParameters: FresnelParameters;\r\n\r\n @serializeAsFresnelParameters(\"opacityFresnelParameters\")\r\n private _opacityFresnelParameters: FresnelParameters;\r\n /**\r\n * Define the opacity fresnel parameters of the material.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/fresnelParameters\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsFresnelAndMiscDirty\")\r\n public opacityFresnelParameters: FresnelParameters;\r\n\r\n @serializeAsFresnelParameters(\"reflectionFresnelParameters\")\r\n private _reflectionFresnelParameters: FresnelParameters;\r\n /**\r\n * Define the reflection fresnel parameters of the material.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/fresnelParameters\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsFresnelDirty\")\r\n public reflectionFresnelParameters: FresnelParameters;\r\n\r\n @serializeAsFresnelParameters(\"refractionFresnelParameters\")\r\n private _refractionFresnelParameters: FresnelParameters;\r\n /**\r\n * Define the refraction fresnel parameters of the material.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/fresnelParameters\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsFresnelDirty\")\r\n public refractionFresnelParameters: FresnelParameters;\r\n\r\n @serializeAsFresnelParameters(\"emissiveFresnelParameters\")\r\n private _emissiveFresnelParameters: FresnelParameters;\r\n /**\r\n * Define the emissive fresnel parameters of the material.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/fresnelParameters\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsFresnelDirty\")\r\n public emissiveFresnelParameters: FresnelParameters;\r\n\r\n @serialize(\"useReflectionFresnelFromSpecular\")\r\n private _useReflectionFresnelFromSpecular = false;\r\n /**\r\n * If true automatically deducts the fresnels values from the material specularity.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/fresnelParameters\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsFresnelDirty\")\r\n public useReflectionFresnelFromSpecular: boolean;\r\n\r\n @serialize(\"useGlossinessFromSpecularMapAlpha\")\r\n private _useGlossinessFromSpecularMapAlpha = false;\r\n /**\r\n * Defines if the glossiness/roughness of the material should be read from the specular map alpha channel\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useGlossinessFromSpecularMapAlpha: boolean;\r\n\r\n @serialize(\"maxSimultaneousLights\")\r\n private _maxSimultaneousLights = 4;\r\n /**\r\n * Defines the maximum number of lights that can be used in the material\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsLightsDirty\")\r\n public maxSimultaneousLights: number;\r\n\r\n @serialize(\"invertNormalMapX\")\r\n private _invertNormalMapX = false;\r\n /**\r\n * If sets to true, x component of normal map value will invert (x = 1.0 - x).\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public invertNormalMapX: boolean;\r\n\r\n @serialize(\"invertNormalMapY\")\r\n private _invertNormalMapY = false;\r\n /**\r\n * If sets to true, y component of normal map value will invert (y = 1.0 - y).\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public invertNormalMapY: boolean;\r\n\r\n @serialize(\"twoSidedLighting\")\r\n private _twoSidedLighting = false;\r\n /**\r\n * If sets to true and backfaceCulling is false, normals will be flipped on the backside.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public twoSidedLighting: boolean;\r\n\r\n @serialize(\"applyDecalMapAfterDetailMap\")\r\n private _applyDecalMapAfterDetailMap = false;\r\n /**\r\n * If sets to true, the decal map will be applied after the detail map. Else, it is applied before (default: false)\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsMiscDirty\")\r\n public applyDecalMapAfterDetailMap: boolean;\r\n\r\n /**\r\n * Default configuration related to image processing available in the standard Material.\r\n */\r\n protected _imageProcessingConfiguration: ImageProcessingConfiguration;\r\n\r\n /**\r\n * Gets the image processing configuration used either in this material.\r\n */\r\n public get imageProcessingConfiguration(): ImageProcessingConfiguration {\r\n return this._imageProcessingConfiguration;\r\n }\r\n\r\n /**\r\n * Sets the Default image processing configuration used either in the this material.\r\n *\r\n * If sets to null, the scene one is in use.\r\n */\r\n public set imageProcessingConfiguration(value: ImageProcessingConfiguration) {\r\n this._attachImageProcessingConfiguration(value);\r\n\r\n // Ensure the effect will be rebuilt.\r\n this._markAllSubMeshesAsTexturesDirty();\r\n }\r\n\r\n /**\r\n * Keep track of the image processing observer to allow dispose and replace.\r\n */\r\n private _imageProcessingObserver: Nullable>;\r\n\r\n /**\r\n * Attaches a new image processing configuration to the Standard Material.\r\n * @param configuration\r\n */\r\n protected _attachImageProcessingConfiguration(configuration: Nullable): void {\r\n if (configuration === this._imageProcessingConfiguration) {\r\n return;\r\n }\r\n\r\n // Detaches observer\r\n if (this._imageProcessingConfiguration && this._imageProcessingObserver) {\r\n this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver);\r\n }\r\n\r\n // Pick the scene configuration if needed\r\n if (!configuration) {\r\n this._imageProcessingConfiguration = this.getScene().imageProcessingConfiguration;\r\n } else {\r\n this._imageProcessingConfiguration = configuration;\r\n }\r\n\r\n // Attaches observer\r\n if (this._imageProcessingConfiguration) {\r\n this._imageProcessingObserver = this._imageProcessingConfiguration.onUpdateParameters.add(() => {\r\n this._markAllSubMeshesAsImageProcessingDirty();\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Defines additional PrePass parameters for the material.\r\n */\r\n public readonly prePassConfiguration: PrePassConfiguration;\r\n\r\n /**\r\n * Can this material render to prepass\r\n */\r\n public get isPrePassCapable(): boolean {\r\n return !this.disableDepthWrite;\r\n }\r\n\r\n /**\r\n * Gets whether the color curves effect is enabled.\r\n */\r\n public get cameraColorCurvesEnabled(): boolean {\r\n return this.imageProcessingConfiguration.colorCurvesEnabled;\r\n }\r\n /**\r\n * Sets whether the color curves effect is enabled.\r\n */\r\n public set cameraColorCurvesEnabled(value: boolean) {\r\n this.imageProcessingConfiguration.colorCurvesEnabled = value;\r\n }\r\n\r\n /**\r\n * Gets whether the color grading effect is enabled.\r\n */\r\n public get cameraColorGradingEnabled(): boolean {\r\n return this.imageProcessingConfiguration.colorGradingEnabled;\r\n }\r\n /**\r\n * Gets whether the color grading effect is enabled.\r\n */\r\n public set cameraColorGradingEnabled(value: boolean) {\r\n this.imageProcessingConfiguration.colorGradingEnabled = value;\r\n }\r\n\r\n /**\r\n * Gets whether tonemapping is enabled or not.\r\n */\r\n public get cameraToneMappingEnabled(): boolean {\r\n return this._imageProcessingConfiguration.toneMappingEnabled;\r\n }\r\n /**\r\n * Sets whether tonemapping is enabled or not\r\n */\r\n public set cameraToneMappingEnabled(value: boolean) {\r\n this._imageProcessingConfiguration.toneMappingEnabled = value;\r\n }\r\n\r\n /**\r\n * The camera exposure used on this material.\r\n * This property is here and not in the camera to allow controlling exposure without full screen post process.\r\n * This corresponds to a photographic exposure.\r\n */\r\n public get cameraExposure(): number {\r\n return this._imageProcessingConfiguration.exposure;\r\n }\r\n /**\r\n * The camera exposure used on this material.\r\n * This property is here and not in the camera to allow controlling exposure without full screen post process.\r\n * This corresponds to a photographic exposure.\r\n */\r\n public set cameraExposure(value: number) {\r\n this._imageProcessingConfiguration.exposure = value;\r\n }\r\n\r\n /**\r\n * Gets The camera contrast used on this material.\r\n */\r\n public get cameraContrast(): number {\r\n return this._imageProcessingConfiguration.contrast;\r\n }\r\n\r\n /**\r\n * Sets The camera contrast used on this material.\r\n */\r\n public set cameraContrast(value: number) {\r\n this._imageProcessingConfiguration.contrast = value;\r\n }\r\n\r\n /**\r\n * Gets the Color Grading 2D Lookup Texture.\r\n */\r\n public get cameraColorGradingTexture(): Nullable {\r\n return this._imageProcessingConfiguration.colorGradingTexture;\r\n }\r\n /**\r\n * Sets the Color Grading 2D Lookup Texture.\r\n */\r\n public set cameraColorGradingTexture(value: Nullable) {\r\n this._imageProcessingConfiguration.colorGradingTexture = value;\r\n }\r\n\r\n /**\r\n * The color grading curves provide additional color adjustmnent that is applied after any color grading transform (3D LUT).\r\n * They allow basic adjustment of saturation and small exposure adjustments, along with color filter tinting to provide white balance adjustment or more stylistic effects.\r\n * These are similar to controls found in many professional imaging or colorist software. The global controls are applied to the entire image. For advanced tuning, extra controls are provided to adjust the shadow, midtone and highlight areas of the image;\r\n * corresponding to low luminance, medium luminance, and high luminance areas respectively.\r\n */\r\n public get cameraColorCurves(): Nullable {\r\n return this._imageProcessingConfiguration.colorCurves;\r\n }\r\n /**\r\n * The color grading curves provide additional color adjustment that is applied after any color grading transform (3D LUT).\r\n * They allow basic adjustment of saturation and small exposure adjustments, along with color filter tinting to provide white balance adjustment or more stylistic effects.\r\n * These are similar to controls found in many professional imaging or colorist software. The global controls are applied to the entire image. For advanced tuning, extra controls are provided to adjust the shadow, midtone and highlight areas of the image;\r\n * corresponding to low luminance, medium luminance, and high luminance areas respectively.\r\n */\r\n public set cameraColorCurves(value: Nullable) {\r\n this._imageProcessingConfiguration.colorCurves = value;\r\n }\r\n\r\n /**\r\n * Can this material render to several textures at once\r\n */\r\n public get canRenderToMRT() {\r\n return true;\r\n }\r\n\r\n /**\r\n * Defines the detail map parameters for the material.\r\n */\r\n public readonly detailMap: DetailMapConfiguration;\r\n\r\n protected _renderTargets = new SmartArray(16);\r\n protected _worldViewProjectionMatrix = Matrix.Zero();\r\n protected _globalAmbientColor = new Color3(0, 0, 0);\r\n protected _useLogarithmicDepth: boolean;\r\n protected _cacheHasRenderTargetTextures = false;\r\n\r\n /**\r\n * Instantiates a new standard material.\r\n * This is the default material used in Babylon. It is the best trade off between quality\r\n * and performances.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/materials_introduction\r\n * @param name Define the name of the material in the scene\r\n * @param scene Define the scene the material belong to\r\n */\r\n constructor(name: string, scene?: Scene) {\r\n super(name, scene);\r\n\r\n this.detailMap = new DetailMapConfiguration(this);\r\n\r\n // Setup the default processing configuration to the scene.\r\n this._attachImageProcessingConfiguration(null);\r\n this.prePassConfiguration = new PrePassConfiguration();\r\n\r\n this.getRenderTargetTextures = (): SmartArray => {\r\n this._renderTargets.reset();\r\n\r\n if (StandardMaterial.ReflectionTextureEnabled && this._reflectionTexture && this._reflectionTexture.isRenderTarget) {\r\n this._renderTargets.push(this._reflectionTexture);\r\n }\r\n\r\n if (StandardMaterial.RefractionTextureEnabled && this._refractionTexture && this._refractionTexture.isRenderTarget) {\r\n this._renderTargets.push(this._refractionTexture);\r\n }\r\n\r\n this._eventInfo.renderTargets = this._renderTargets;\r\n this._callbackPluginEventFillRenderTargetTextures(this._eventInfo);\r\n\r\n return this._renderTargets;\r\n };\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating that current material needs to register RTT\r\n */\r\n public get hasRenderTargetTextures(): boolean {\r\n if (StandardMaterial.ReflectionTextureEnabled && this._reflectionTexture && this._reflectionTexture.isRenderTarget) {\r\n return true;\r\n }\r\n\r\n if (StandardMaterial.RefractionTextureEnabled && this._refractionTexture && this._refractionTexture.isRenderTarget) {\r\n return true;\r\n }\r\n\r\n return this._cacheHasRenderTargetTextures;\r\n }\r\n\r\n /**\r\n * Gets the current class name of the material e.g. \"StandardMaterial\"\r\n * Mainly use in serialization.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"StandardMaterial\";\r\n }\r\n\r\n /**\r\n * In case the depth buffer does not allow enough depth precision for your scene (might be the case in large scenes)\r\n * You can try switching to logarithmic depth.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/advanced/logarithmicDepthBuffer\r\n */\r\n @serialize()\r\n public get useLogarithmicDepth(): boolean {\r\n return this._useLogarithmicDepth;\r\n }\r\n\r\n public set useLogarithmicDepth(value: boolean) {\r\n this._useLogarithmicDepth = value && this.getScene().getEngine().getCaps().fragmentDepthSupported;\r\n\r\n this._markAllSubMeshesAsMiscDirty();\r\n }\r\n\r\n /**\r\n * Specifies if the material will require alpha blending\r\n * @returns a boolean specifying if alpha blending is needed\r\n */\r\n public needAlphaBlending(): boolean {\r\n if (this._disableAlphaBlending) {\r\n return false;\r\n }\r\n\r\n return (\r\n this.alpha < 1.0 ||\r\n this._opacityTexture != null ||\r\n this._shouldUseAlphaFromDiffuseTexture() ||\r\n (this._opacityFresnelParameters && this._opacityFresnelParameters.isEnabled)\r\n );\r\n }\r\n\r\n /**\r\n * Specifies if this material should be rendered in alpha test mode\r\n * @returns a boolean specifying if an alpha test is needed.\r\n */\r\n public needAlphaTesting(): boolean {\r\n if (this._forceAlphaTest) {\r\n return true;\r\n }\r\n\r\n return this._hasAlphaChannel() && (this._transparencyMode == null || this._transparencyMode === Material.MATERIAL_ALPHATEST);\r\n }\r\n\r\n /**\r\n * Specifies whether or not the alpha value of the diffuse texture should be used for alpha blending.\r\n */\r\n protected _shouldUseAlphaFromDiffuseTexture(): boolean {\r\n return this._diffuseTexture != null && this._diffuseTexture.hasAlpha && this._useAlphaFromDiffuseTexture && this._transparencyMode !== Material.MATERIAL_OPAQUE;\r\n }\r\n\r\n /**\r\n * Specifies whether or not there is a usable alpha channel for transparency.\r\n */\r\n protected _hasAlphaChannel(): boolean {\r\n return (this._diffuseTexture != null && this._diffuseTexture.hasAlpha) || this._opacityTexture != null;\r\n }\r\n\r\n /**\r\n * Get the texture used for alpha test purpose.\r\n * @returns the diffuse texture in case of the standard material.\r\n */\r\n public getAlphaTestTexture(): Nullable {\r\n return this._diffuseTexture;\r\n }\r\n\r\n /**\r\n * Get if the submesh is ready to be used and all its information available.\r\n * Child classes can use it to update shaders\r\n * @param mesh defines the mesh to check\r\n * @param subMesh defines which submesh to check\r\n * @param useInstances specifies that instances should be used\r\n * @returns a boolean indicating that the submesh is ready or not\r\n */\r\n public isReadyForSubMesh(mesh: AbstractMesh, subMesh: SubMesh, useInstances: boolean = false): boolean {\r\n if (!this._uniformBufferLayoutBuilt) {\r\n this.buildUniformLayout();\r\n }\r\n\r\n if (subMesh.effect && this.isFrozen) {\r\n if (subMesh.effect._wasPreviouslyReady && subMesh.effect._wasPreviouslyUsingInstances === useInstances) {\r\n return true;\r\n }\r\n }\r\n\r\n if (!subMesh.materialDefines) {\r\n this._callbackPluginEventGeneric(MaterialPluginEvent.GetDefineNames, this._eventInfo);\r\n subMesh.materialDefines = new StandardMaterialDefines(this._eventInfo.defineNames);\r\n }\r\n\r\n const scene = this.getScene();\r\n const defines = subMesh.materialDefines;\r\n if (this._isReadyForSubMesh(subMesh)) {\r\n return true;\r\n }\r\n\r\n const engine = scene.getEngine();\r\n\r\n // Lights\r\n defines._needNormals = MaterialHelper.PrepareDefinesForLights(scene, mesh, defines, true, this._maxSimultaneousLights, this._disableLighting);\r\n\r\n // Multiview\r\n MaterialHelper.PrepareDefinesForMultiview(scene, defines);\r\n\r\n // PrePass\r\n const oit = this.needAlphaBlendingForMesh(mesh) && this.getScene().useOrderIndependentTransparency;\r\n MaterialHelper.PrepareDefinesForPrePass(scene, defines, this.canRenderToMRT && !oit);\r\n\r\n // Order independant transparency\r\n MaterialHelper.PrepareDefinesForOIT(scene, defines, oit);\r\n\r\n // Textures\r\n if (defines._areTexturesDirty) {\r\n this._eventInfo.hasRenderTargetTextures = false;\r\n this._callbackPluginEventHasRenderTargetTextures(this._eventInfo);\r\n this._cacheHasRenderTargetTextures = this._eventInfo.hasRenderTargetTextures;\r\n defines._needUVs = false;\r\n for (let i = 1; i <= Constants.MAX_SUPPORTED_UV_SETS; ++i) {\r\n defines[\"MAINUV\" + i] = false;\r\n }\r\n if (scene.texturesEnabled) {\r\n defines.DIFFUSEDIRECTUV = 0;\r\n defines.BUMPDIRECTUV = 0;\r\n defines.AMBIENTDIRECTUV = 0;\r\n defines.OPACITYDIRECTUV = 0;\r\n defines.EMISSIVEDIRECTUV = 0;\r\n defines.SPECULARDIRECTUV = 0;\r\n defines.LIGHTMAPDIRECTUV = 0;\r\n\r\n if (this._diffuseTexture && StandardMaterial.DiffuseTextureEnabled) {\r\n if (!this._diffuseTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n } else {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._diffuseTexture, defines, \"DIFFUSE\");\r\n }\r\n } else {\r\n defines.DIFFUSE = false;\r\n }\r\n\r\n if (this._ambientTexture && StandardMaterial.AmbientTextureEnabled) {\r\n if (!this._ambientTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n } else {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._ambientTexture, defines, \"AMBIENT\");\r\n }\r\n } else {\r\n defines.AMBIENT = false;\r\n }\r\n\r\n if (this._opacityTexture && StandardMaterial.OpacityTextureEnabled) {\r\n if (!this._opacityTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n } else {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._opacityTexture, defines, \"OPACITY\");\r\n defines.OPACITYRGB = this._opacityTexture.getAlphaFromRGB;\r\n }\r\n } else {\r\n defines.OPACITY = false;\r\n }\r\n\r\n if (this._reflectionTexture && StandardMaterial.ReflectionTextureEnabled) {\r\n if (!this._reflectionTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n } else {\r\n defines._needNormals = true;\r\n defines.REFLECTION = true;\r\n\r\n defines.ROUGHNESS = this._roughness > 0;\r\n defines.REFLECTIONOVERALPHA = this._useReflectionOverAlpha;\r\n defines.INVERTCUBICMAP = this._reflectionTexture.coordinatesMode === Texture.INVCUBIC_MODE;\r\n defines.REFLECTIONMAP_3D = this._reflectionTexture.isCube;\r\n defines.REFLECTIONMAP_OPPOSITEZ =\r\n defines.REFLECTIONMAP_3D && this.getScene().useRightHandedSystem ? !this._reflectionTexture.invertZ : this._reflectionTexture.invertZ;\r\n defines.RGBDREFLECTION = this._reflectionTexture.isRGBD;\r\n\r\n switch (this._reflectionTexture.coordinatesMode) {\r\n case Texture.EXPLICIT_MODE:\r\n defines.setReflectionMode(\"REFLECTIONMAP_EXPLICIT\");\r\n break;\r\n case Texture.PLANAR_MODE:\r\n defines.setReflectionMode(\"REFLECTIONMAP_PLANAR\");\r\n break;\r\n case Texture.PROJECTION_MODE:\r\n defines.setReflectionMode(\"REFLECTIONMAP_PROJECTION\");\r\n break;\r\n case Texture.SKYBOX_MODE:\r\n defines.setReflectionMode(\"REFLECTIONMAP_SKYBOX\");\r\n break;\r\n case Texture.SPHERICAL_MODE:\r\n defines.setReflectionMode(\"REFLECTIONMAP_SPHERICAL\");\r\n break;\r\n case Texture.EQUIRECTANGULAR_MODE:\r\n defines.setReflectionMode(\"REFLECTIONMAP_EQUIRECTANGULAR\");\r\n break;\r\n case Texture.FIXED_EQUIRECTANGULAR_MODE:\r\n defines.setReflectionMode(\"REFLECTIONMAP_EQUIRECTANGULAR_FIXED\");\r\n break;\r\n case Texture.FIXED_EQUIRECTANGULAR_MIRRORED_MODE:\r\n defines.setReflectionMode(\"REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED\");\r\n break;\r\n case Texture.CUBIC_MODE:\r\n case Texture.INVCUBIC_MODE:\r\n default:\r\n defines.setReflectionMode(\"REFLECTIONMAP_CUBIC\");\r\n break;\r\n }\r\n\r\n defines.USE_LOCAL_REFLECTIONMAP_CUBIC = (this._reflectionTexture).boundingBoxSize ? true : false;\r\n }\r\n } else {\r\n defines.REFLECTION = false;\r\n defines.REFLECTIONMAP_OPPOSITEZ = false;\r\n }\r\n\r\n if (this._emissiveTexture && StandardMaterial.EmissiveTextureEnabled) {\r\n if (!this._emissiveTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n } else {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._emissiveTexture, defines, \"EMISSIVE\");\r\n }\r\n } else {\r\n defines.EMISSIVE = false;\r\n }\r\n\r\n if (this._lightmapTexture && StandardMaterial.LightmapTextureEnabled) {\r\n if (!this._lightmapTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n } else {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._lightmapTexture, defines, \"LIGHTMAP\");\r\n defines.USELIGHTMAPASSHADOWMAP = this._useLightmapAsShadowmap;\r\n defines.RGBDLIGHTMAP = this._lightmapTexture.isRGBD;\r\n }\r\n } else {\r\n defines.LIGHTMAP = false;\r\n }\r\n\r\n if (this._specularTexture && StandardMaterial.SpecularTextureEnabled) {\r\n if (!this._specularTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n } else {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._specularTexture, defines, \"SPECULAR\");\r\n defines.GLOSSINESS = this._useGlossinessFromSpecularMapAlpha;\r\n }\r\n } else {\r\n defines.SPECULAR = false;\r\n }\r\n\r\n if (scene.getEngine().getCaps().standardDerivatives && this._bumpTexture && StandardMaterial.BumpTextureEnabled) {\r\n // Bump texture can not be not blocking.\r\n if (!this._bumpTexture.isReady()) {\r\n return false;\r\n } else {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._bumpTexture, defines, \"BUMP\");\r\n\r\n defines.PARALLAX = this._useParallax;\r\n defines.PARALLAXOCCLUSION = this._useParallaxOcclusion;\r\n }\r\n\r\n defines.OBJECTSPACE_NORMALMAP = this._useObjectSpaceNormalMap;\r\n } else {\r\n defines.BUMP = false;\r\n defines.PARALLAX = false;\r\n defines.PARALLAXOCCLUSION = false;\r\n }\r\n\r\n if (this._refractionTexture && StandardMaterial.RefractionTextureEnabled) {\r\n if (!this._refractionTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n } else {\r\n defines._needUVs = true;\r\n defines.REFRACTION = true;\r\n\r\n defines.REFRACTIONMAP_3D = this._refractionTexture.isCube;\r\n defines.RGBDREFRACTION = this._refractionTexture.isRGBD;\r\n defines.USE_LOCAL_REFRACTIONMAP_CUBIC = (this._refractionTexture).boundingBoxSize ? true : false;\r\n }\r\n } else {\r\n defines.REFRACTION = false;\r\n }\r\n\r\n defines.TWOSIDEDLIGHTING = !this._backFaceCulling && this._twoSidedLighting;\r\n } else {\r\n defines.DIFFUSE = false;\r\n defines.AMBIENT = false;\r\n defines.OPACITY = false;\r\n defines.REFLECTION = false;\r\n defines.EMISSIVE = false;\r\n defines.LIGHTMAP = false;\r\n defines.BUMP = false;\r\n defines.REFRACTION = false;\r\n }\r\n\r\n defines.ALPHAFROMDIFFUSE = this._shouldUseAlphaFromDiffuseTexture();\r\n\r\n defines.EMISSIVEASILLUMINATION = this._useEmissiveAsIllumination;\r\n\r\n defines.LINKEMISSIVEWITHDIFFUSE = this._linkEmissiveWithDiffuse;\r\n\r\n defines.SPECULAROVERALPHA = this._useSpecularOverAlpha;\r\n\r\n defines.PREMULTIPLYALPHA = this.alphaMode === Constants.ALPHA_PREMULTIPLIED || this.alphaMode === Constants.ALPHA_PREMULTIPLIED_PORTERDUFF;\r\n\r\n defines.ALPHATEST_AFTERALLALPHACOMPUTATIONS = this.transparencyMode !== null;\r\n\r\n defines.ALPHABLEND = this.transparencyMode === null || this.needAlphaBlendingForMesh(mesh); // check on null for backward compatibility\r\n }\r\n\r\n this._eventInfo.isReadyForSubMesh = true;\r\n this._eventInfo.defines = defines;\r\n this._eventInfo.subMesh = subMesh;\r\n this._callbackPluginEventIsReadyForSubMesh(this._eventInfo);\r\n\r\n if (!this._eventInfo.isReadyForSubMesh) {\r\n return false;\r\n }\r\n\r\n if (defines._areImageProcessingDirty && this._imageProcessingConfiguration) {\r\n if (!this._imageProcessingConfiguration.isReady()) {\r\n return false;\r\n }\r\n\r\n this._imageProcessingConfiguration.prepareDefines(defines);\r\n\r\n defines.IS_REFLECTION_LINEAR = this.reflectionTexture != null && !this.reflectionTexture.gammaSpace;\r\n defines.IS_REFRACTION_LINEAR = this.refractionTexture != null && !this.refractionTexture.gammaSpace;\r\n }\r\n\r\n if (defines._areFresnelDirty) {\r\n if (StandardMaterial.FresnelEnabled) {\r\n // Fresnel\r\n if (\r\n this._diffuseFresnelParameters ||\r\n this._opacityFresnelParameters ||\r\n this._emissiveFresnelParameters ||\r\n this._refractionFresnelParameters ||\r\n this._reflectionFresnelParameters\r\n ) {\r\n defines.DIFFUSEFRESNEL = this._diffuseFresnelParameters && this._diffuseFresnelParameters.isEnabled;\r\n\r\n defines.OPACITYFRESNEL = this._opacityFresnelParameters && this._opacityFresnelParameters.isEnabled;\r\n\r\n defines.REFLECTIONFRESNEL = this._reflectionFresnelParameters && this._reflectionFresnelParameters.isEnabled;\r\n\r\n defines.REFLECTIONFRESNELFROMSPECULAR = this._useReflectionFresnelFromSpecular;\r\n\r\n defines.REFRACTIONFRESNEL = this._refractionFresnelParameters && this._refractionFresnelParameters.isEnabled;\r\n\r\n defines.EMISSIVEFRESNEL = this._emissiveFresnelParameters && this._emissiveFresnelParameters.isEnabled;\r\n\r\n defines._needNormals = true;\r\n defines.FRESNEL = true;\r\n }\r\n } else {\r\n defines.FRESNEL = false;\r\n }\r\n }\r\n\r\n // Misc.\r\n MaterialHelper.PrepareDefinesForMisc(\r\n mesh,\r\n scene,\r\n this._useLogarithmicDepth,\r\n this.pointsCloud,\r\n this.fogEnabled,\r\n this._shouldTurnAlphaTestOn(mesh) || this._forceAlphaTest,\r\n defines,\r\n this._applyDecalMapAfterDetailMap\r\n );\r\n\r\n // Values that need to be evaluated on every frame\r\n MaterialHelper.PrepareDefinesForFrameBoundValues(scene, engine, this, defines, useInstances, null, subMesh.getRenderingMesh().hasThinInstances);\r\n\r\n // External config\r\n this._eventInfo.defines = defines;\r\n this._eventInfo.mesh = mesh;\r\n this._callbackPluginEventPrepareDefinesBeforeAttributes(this._eventInfo);\r\n\r\n // Attribs\r\n MaterialHelper.PrepareDefinesForAttributes(mesh, defines, true, true, true);\r\n\r\n // External config\r\n this._callbackPluginEventPrepareDefines(this._eventInfo);\r\n\r\n // Get correct effect\r\n let forceWasNotReadyPreviously = false;\r\n\r\n if (defines.isDirty) {\r\n const lightDisposed = defines._areLightsDisposed;\r\n defines.markAsProcessed();\r\n\r\n // Fallbacks\r\n const fallbacks = new EffectFallbacks();\r\n if (defines.REFLECTION) {\r\n fallbacks.addFallback(0, \"REFLECTION\");\r\n }\r\n\r\n if (defines.SPECULAR) {\r\n fallbacks.addFallback(0, \"SPECULAR\");\r\n }\r\n\r\n if (defines.BUMP) {\r\n fallbacks.addFallback(0, \"BUMP\");\r\n }\r\n\r\n if (defines.PARALLAX) {\r\n fallbacks.addFallback(1, \"PARALLAX\");\r\n }\r\n\r\n if (defines.PARALLAXOCCLUSION) {\r\n fallbacks.addFallback(0, \"PARALLAXOCCLUSION\");\r\n }\r\n\r\n if (defines.SPECULAROVERALPHA) {\r\n fallbacks.addFallback(0, \"SPECULAROVERALPHA\");\r\n }\r\n\r\n if (defines.FOG) {\r\n fallbacks.addFallback(1, \"FOG\");\r\n }\r\n\r\n if (defines.POINTSIZE) {\r\n fallbacks.addFallback(0, \"POINTSIZE\");\r\n }\r\n\r\n if (defines.LOGARITHMICDEPTH) {\r\n fallbacks.addFallback(0, \"LOGARITHMICDEPTH\");\r\n }\r\n\r\n MaterialHelper.HandleFallbacksForShadows(defines, fallbacks, this._maxSimultaneousLights);\r\n\r\n if (defines.SPECULARTERM) {\r\n fallbacks.addFallback(0, \"SPECULARTERM\");\r\n }\r\n\r\n if (defines.DIFFUSEFRESNEL) {\r\n fallbacks.addFallback(1, \"DIFFUSEFRESNEL\");\r\n }\r\n\r\n if (defines.OPACITYFRESNEL) {\r\n fallbacks.addFallback(2, \"OPACITYFRESNEL\");\r\n }\r\n\r\n if (defines.REFLECTIONFRESNEL) {\r\n fallbacks.addFallback(3, \"REFLECTIONFRESNEL\");\r\n }\r\n\r\n if (defines.EMISSIVEFRESNEL) {\r\n fallbacks.addFallback(4, \"EMISSIVEFRESNEL\");\r\n }\r\n\r\n if (defines.FRESNEL) {\r\n fallbacks.addFallback(4, \"FRESNEL\");\r\n }\r\n\r\n if (defines.MULTIVIEW) {\r\n fallbacks.addFallback(0, \"MULTIVIEW\");\r\n }\r\n\r\n //Attributes\r\n const attribs = [VertexBuffer.PositionKind];\r\n\r\n if (defines.NORMAL) {\r\n attribs.push(VertexBuffer.NormalKind);\r\n }\r\n\r\n if (defines.TANGENT) {\r\n attribs.push(VertexBuffer.TangentKind);\r\n }\r\n\r\n for (let i = 1; i <= Constants.MAX_SUPPORTED_UV_SETS; ++i) {\r\n if (defines[\"UV\" + i]) {\r\n attribs.push(`uv${i === 1 ? \"\" : i}`);\r\n }\r\n }\r\n\r\n if (defines.VERTEXCOLOR) {\r\n attribs.push(VertexBuffer.ColorKind);\r\n }\r\n\r\n MaterialHelper.PrepareAttributesForBones(attribs, mesh, defines, fallbacks);\r\n MaterialHelper.PrepareAttributesForInstances(attribs, defines);\r\n MaterialHelper.PrepareAttributesForMorphTargets(attribs, mesh, defines);\r\n MaterialHelper.PrepareAttributesForBakedVertexAnimation(attribs, mesh, defines);\r\n\r\n let shaderName = \"default\";\r\n\r\n const uniforms = [\r\n \"world\",\r\n \"view\",\r\n \"viewProjection\",\r\n \"vEyePosition\",\r\n \"vLightsType\",\r\n \"vAmbientColor\",\r\n \"vDiffuseColor\",\r\n \"vSpecularColor\",\r\n \"vEmissiveColor\",\r\n \"visibility\",\r\n \"vFogInfos\",\r\n \"vFogColor\",\r\n \"pointSize\",\r\n \"vDiffuseInfos\",\r\n \"vAmbientInfos\",\r\n \"vOpacityInfos\",\r\n \"vReflectionInfos\",\r\n \"vEmissiveInfos\",\r\n \"vSpecularInfos\",\r\n \"vBumpInfos\",\r\n \"vLightmapInfos\",\r\n \"vRefractionInfos\",\r\n \"mBones\",\r\n \"diffuseMatrix\",\r\n \"ambientMatrix\",\r\n \"opacityMatrix\",\r\n \"reflectionMatrix\",\r\n \"emissiveMatrix\",\r\n \"specularMatrix\",\r\n \"bumpMatrix\",\r\n \"normalMatrix\",\r\n \"lightmapMatrix\",\r\n \"refractionMatrix\",\r\n \"diffuseLeftColor\",\r\n \"diffuseRightColor\",\r\n \"opacityParts\",\r\n \"reflectionLeftColor\",\r\n \"reflectionRightColor\",\r\n \"emissiveLeftColor\",\r\n \"emissiveRightColor\",\r\n \"refractionLeftColor\",\r\n \"refractionRightColor\",\r\n \"vReflectionPosition\",\r\n \"vReflectionSize\",\r\n \"vRefractionPosition\",\r\n \"vRefractionSize\",\r\n \"logarithmicDepthConstant\",\r\n \"vTangentSpaceParams\",\r\n \"alphaCutOff\",\r\n \"boneTextureWidth\",\r\n \"morphTargetTextureInfo\",\r\n \"morphTargetTextureIndices\",\r\n ];\r\n\r\n const samplers = [\r\n \"diffuseSampler\",\r\n \"ambientSampler\",\r\n \"opacitySampler\",\r\n \"reflectionCubeSampler\",\r\n \"reflection2DSampler\",\r\n \"emissiveSampler\",\r\n \"specularSampler\",\r\n \"bumpSampler\",\r\n \"lightmapSampler\",\r\n \"refractionCubeSampler\",\r\n \"refraction2DSampler\",\r\n \"boneSampler\",\r\n \"morphTargets\",\r\n \"oitDepthSampler\",\r\n \"oitFrontColorSampler\",\r\n ];\r\n\r\n const uniformBuffers = [\"Material\", \"Scene\", \"Mesh\"];\r\n\r\n const indexParameters = { maxSimultaneousLights: this._maxSimultaneousLights, maxSimultaneousMorphTargets: defines.NUM_MORPH_INFLUENCERS };\r\n\r\n this._eventInfo.fallbacks = fallbacks;\r\n this._eventInfo.fallbackRank = 0;\r\n this._eventInfo.defines = defines;\r\n this._eventInfo.uniforms = uniforms;\r\n this._eventInfo.attributes = attribs;\r\n this._eventInfo.samplers = samplers;\r\n this._eventInfo.uniformBuffersNames = uniformBuffers;\r\n this._eventInfo.customCode = undefined;\r\n this._eventInfo.mesh = mesh;\r\n this._eventInfo.indexParameters = indexParameters;\r\n this._callbackPluginEventGeneric(MaterialPluginEvent.PrepareEffect, this._eventInfo);\r\n\r\n PrePassConfiguration.AddUniforms(uniforms);\r\n PrePassConfiguration.AddSamplers(samplers);\r\n\r\n if (ImageProcessingConfiguration) {\r\n ImageProcessingConfiguration.PrepareUniforms(uniforms, defines);\r\n ImageProcessingConfiguration.PrepareSamplers(samplers, defines);\r\n }\r\n\r\n MaterialHelper.PrepareUniformsAndSamplersList({\r\n uniformsNames: uniforms,\r\n uniformBuffersNames: uniformBuffers,\r\n samplers: samplers,\r\n defines: defines,\r\n maxSimultaneousLights: this._maxSimultaneousLights,\r\n });\r\n\r\n addClipPlaneUniforms(uniforms);\r\n\r\n const csnrOptions: ICustomShaderNameResolveOptions = {};\r\n\r\n if (this.customShaderNameResolve) {\r\n shaderName = this.customShaderNameResolve(shaderName, uniforms, uniformBuffers, samplers, defines, attribs, csnrOptions);\r\n }\r\n\r\n const join = defines.toString();\r\n\r\n const previousEffect = subMesh.effect;\r\n let effect = scene.getEngine().createEffect(\r\n shaderName,\r\n {\r\n attributes: attribs,\r\n uniformsNames: uniforms,\r\n uniformBuffersNames: uniformBuffers,\r\n samplers: samplers,\r\n defines: join,\r\n fallbacks: fallbacks,\r\n onCompiled: this.onCompiled,\r\n onError: this.onError,\r\n indexParameters,\r\n processFinalCode: csnrOptions.processFinalCode,\r\n processCodeAfterIncludes: this._eventInfo.customCode,\r\n multiTarget: defines.PREPASS,\r\n },\r\n engine\r\n );\r\n\r\n this._eventInfo.customCode = undefined;\r\n\r\n if (effect) {\r\n if (this._onEffectCreatedObservable) {\r\n onCreatedEffectParameters.effect = effect;\r\n onCreatedEffectParameters.subMesh = subMesh;\r\n this._onEffectCreatedObservable.notifyObservers(onCreatedEffectParameters);\r\n }\r\n\r\n // Use previous effect while new one is compiling\r\n if (this.allowShaderHotSwapping && previousEffect && !effect.isReady()) {\r\n effect = previousEffect;\r\n defines.markAsUnprocessed();\r\n\r\n forceWasNotReadyPreviously = this.isFrozen;\r\n\r\n if (lightDisposed) {\r\n // re register in case it takes more than one frame.\r\n defines._areLightsDisposed = true;\r\n return false;\r\n }\r\n } else {\r\n scene.resetCachedMaterial();\r\n subMesh.setEffect(effect, defines, this._materialContext);\r\n }\r\n }\r\n }\r\n\r\n if (!subMesh.effect || !subMesh.effect.isReady()) {\r\n return false;\r\n }\r\n\r\n defines._renderId = scene.getRenderId();\r\n subMesh.effect._wasPreviouslyReady = forceWasNotReadyPreviously ? false : true;\r\n subMesh.effect._wasPreviouslyUsingInstances = useInstances;\r\n\r\n this._checkScenePerformancePriority();\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Builds the material UBO layouts.\r\n * Used internally during the effect preparation.\r\n */\r\n public buildUniformLayout(): void {\r\n // Order is important !\r\n const ubo = this._uniformBuffer;\r\n ubo.addUniform(\"diffuseLeftColor\", 4);\r\n ubo.addUniform(\"diffuseRightColor\", 4);\r\n ubo.addUniform(\"opacityParts\", 4);\r\n ubo.addUniform(\"reflectionLeftColor\", 4);\r\n ubo.addUniform(\"reflectionRightColor\", 4);\r\n ubo.addUniform(\"refractionLeftColor\", 4);\r\n ubo.addUniform(\"refractionRightColor\", 4);\r\n ubo.addUniform(\"emissiveLeftColor\", 4);\r\n ubo.addUniform(\"emissiveRightColor\", 4);\r\n\r\n ubo.addUniform(\"vDiffuseInfos\", 2);\r\n ubo.addUniform(\"vAmbientInfos\", 2);\r\n ubo.addUniform(\"vOpacityInfos\", 2);\r\n ubo.addUniform(\"vReflectionInfos\", 2);\r\n ubo.addUniform(\"vReflectionPosition\", 3);\r\n ubo.addUniform(\"vReflectionSize\", 3);\r\n ubo.addUniform(\"vEmissiveInfos\", 2);\r\n ubo.addUniform(\"vLightmapInfos\", 2);\r\n ubo.addUniform(\"vSpecularInfos\", 2);\r\n ubo.addUniform(\"vBumpInfos\", 3);\r\n\r\n ubo.addUniform(\"diffuseMatrix\", 16);\r\n ubo.addUniform(\"ambientMatrix\", 16);\r\n ubo.addUniform(\"opacityMatrix\", 16);\r\n ubo.addUniform(\"reflectionMatrix\", 16);\r\n ubo.addUniform(\"emissiveMatrix\", 16);\r\n ubo.addUniform(\"lightmapMatrix\", 16);\r\n ubo.addUniform(\"specularMatrix\", 16);\r\n ubo.addUniform(\"bumpMatrix\", 16);\r\n ubo.addUniform(\"vTangentSpaceParams\", 2);\r\n ubo.addUniform(\"pointSize\", 1);\r\n ubo.addUniform(\"alphaCutOff\", 1);\r\n ubo.addUniform(\"refractionMatrix\", 16);\r\n ubo.addUniform(\"vRefractionInfos\", 4);\r\n ubo.addUniform(\"vRefractionPosition\", 3);\r\n ubo.addUniform(\"vRefractionSize\", 3);\r\n ubo.addUniform(\"vSpecularColor\", 4);\r\n ubo.addUniform(\"vEmissiveColor\", 3);\r\n ubo.addUniform(\"vDiffuseColor\", 4);\r\n ubo.addUniform(\"vAmbientColor\", 3);\r\n\r\n super.buildUniformLayout();\r\n }\r\n\r\n /**\r\n * Binds the submesh to this material by preparing the effect and shader to draw\r\n * @param world defines the world transformation matrix\r\n * @param mesh defines the mesh containing the submesh\r\n * @param subMesh defines the submesh to bind the material to\r\n */\r\n public bindForSubMesh(world: Matrix, mesh: Mesh, subMesh: SubMesh): void {\r\n const scene = this.getScene();\r\n\r\n const defines = subMesh.materialDefines;\r\n if (!defines) {\r\n return;\r\n }\r\n\r\n const effect = subMesh.effect;\r\n if (!effect) {\r\n return;\r\n }\r\n this._activeEffect = effect;\r\n\r\n // Matrices Mesh.\r\n mesh.getMeshUniformBuffer().bindToEffect(effect, \"Mesh\");\r\n mesh.transferToEffect(world);\r\n\r\n // Binding unconditionally\r\n this._uniformBuffer.bindToEffect(effect, \"Material\");\r\n\r\n this.prePassConfiguration.bindForSubMesh(this._activeEffect, scene, mesh, world, this.isFrozen);\r\n\r\n this._eventInfo.subMesh = subMesh;\r\n this._callbackPluginEventHardBindForSubMesh(this._eventInfo);\r\n\r\n // Normal Matrix\r\n if (defines.OBJECTSPACE_NORMALMAP) {\r\n world.toNormalMatrix(this._normalMatrix);\r\n this.bindOnlyNormalMatrix(this._normalMatrix);\r\n }\r\n\r\n const mustRebind = effect._forceRebindOnNextCall || this._mustRebind(scene, effect, mesh.visibility);\r\n\r\n // Bones\r\n MaterialHelper.BindBonesParameters(mesh, effect);\r\n const ubo = this._uniformBuffer;\r\n if (mustRebind) {\r\n this.bindViewProjection(effect);\r\n if (!ubo.useUbo || !this.isFrozen || !ubo.isSync || effect._forceRebindOnNextCall) {\r\n if (StandardMaterial.FresnelEnabled && defines.FRESNEL) {\r\n // Fresnel\r\n if (this.diffuseFresnelParameters && this.diffuseFresnelParameters.isEnabled) {\r\n ubo.updateColor4(\"diffuseLeftColor\", this.diffuseFresnelParameters.leftColor, this.diffuseFresnelParameters.power);\r\n ubo.updateColor4(\"diffuseRightColor\", this.diffuseFresnelParameters.rightColor, this.diffuseFresnelParameters.bias);\r\n }\r\n\r\n if (this.opacityFresnelParameters && this.opacityFresnelParameters.isEnabled) {\r\n ubo.updateColor4(\r\n \"opacityParts\",\r\n new Color3(\r\n this.opacityFresnelParameters.leftColor.toLuminance(),\r\n this.opacityFresnelParameters.rightColor.toLuminance(),\r\n this.opacityFresnelParameters.bias\r\n ),\r\n this.opacityFresnelParameters.power\r\n );\r\n }\r\n\r\n if (this.reflectionFresnelParameters && this.reflectionFresnelParameters.isEnabled) {\r\n ubo.updateColor4(\"reflectionLeftColor\", this.reflectionFresnelParameters.leftColor, this.reflectionFresnelParameters.power);\r\n ubo.updateColor4(\"reflectionRightColor\", this.reflectionFresnelParameters.rightColor, this.reflectionFresnelParameters.bias);\r\n }\r\n\r\n if (this.refractionFresnelParameters && this.refractionFresnelParameters.isEnabled) {\r\n ubo.updateColor4(\"refractionLeftColor\", this.refractionFresnelParameters.leftColor, this.refractionFresnelParameters.power);\r\n ubo.updateColor4(\"refractionRightColor\", this.refractionFresnelParameters.rightColor, this.refractionFresnelParameters.bias);\r\n }\r\n\r\n if (this.emissiveFresnelParameters && this.emissiveFresnelParameters.isEnabled) {\r\n ubo.updateColor4(\"emissiveLeftColor\", this.emissiveFresnelParameters.leftColor, this.emissiveFresnelParameters.power);\r\n ubo.updateColor4(\"emissiveRightColor\", this.emissiveFresnelParameters.rightColor, this.emissiveFresnelParameters.bias);\r\n }\r\n }\r\n\r\n // Textures\r\n if (scene.texturesEnabled) {\r\n if (this._diffuseTexture && StandardMaterial.DiffuseTextureEnabled) {\r\n ubo.updateFloat2(\"vDiffuseInfos\", this._diffuseTexture.coordinatesIndex, this._diffuseTexture.level);\r\n MaterialHelper.BindTextureMatrix(this._diffuseTexture, ubo, \"diffuse\");\r\n }\r\n\r\n if (this._ambientTexture && StandardMaterial.AmbientTextureEnabled) {\r\n ubo.updateFloat2(\"vAmbientInfos\", this._ambientTexture.coordinatesIndex, this._ambientTexture.level);\r\n MaterialHelper.BindTextureMatrix(this._ambientTexture, ubo, \"ambient\");\r\n }\r\n\r\n if (this._opacityTexture && StandardMaterial.OpacityTextureEnabled) {\r\n ubo.updateFloat2(\"vOpacityInfos\", this._opacityTexture.coordinatesIndex, this._opacityTexture.level);\r\n MaterialHelper.BindTextureMatrix(this._opacityTexture, ubo, \"opacity\");\r\n }\r\n\r\n if (this._hasAlphaChannel()) {\r\n ubo.updateFloat(\"alphaCutOff\", this.alphaCutOff);\r\n }\r\n\r\n if (this._reflectionTexture && StandardMaterial.ReflectionTextureEnabled) {\r\n ubo.updateFloat2(\"vReflectionInfos\", this._reflectionTexture.level, this.roughness);\r\n ubo.updateMatrix(\"reflectionMatrix\", this._reflectionTexture.getReflectionTextureMatrix());\r\n\r\n if ((this._reflectionTexture).boundingBoxSize) {\r\n const cubeTexture = this._reflectionTexture;\r\n\r\n ubo.updateVector3(\"vReflectionPosition\", cubeTexture.boundingBoxPosition);\r\n ubo.updateVector3(\"vReflectionSize\", cubeTexture.boundingBoxSize);\r\n }\r\n }\r\n\r\n if (this._emissiveTexture && StandardMaterial.EmissiveTextureEnabled) {\r\n ubo.updateFloat2(\"vEmissiveInfos\", this._emissiveTexture.coordinatesIndex, this._emissiveTexture.level);\r\n MaterialHelper.BindTextureMatrix(this._emissiveTexture, ubo, \"emissive\");\r\n }\r\n\r\n if (this._lightmapTexture && StandardMaterial.LightmapTextureEnabled) {\r\n ubo.updateFloat2(\"vLightmapInfos\", this._lightmapTexture.coordinatesIndex, this._lightmapTexture.level);\r\n MaterialHelper.BindTextureMatrix(this._lightmapTexture, ubo, \"lightmap\");\r\n }\r\n\r\n if (this._specularTexture && StandardMaterial.SpecularTextureEnabled) {\r\n ubo.updateFloat2(\"vSpecularInfos\", this._specularTexture.coordinatesIndex, this._specularTexture.level);\r\n MaterialHelper.BindTextureMatrix(this._specularTexture, ubo, \"specular\");\r\n }\r\n\r\n if (this._bumpTexture && scene.getEngine().getCaps().standardDerivatives && StandardMaterial.BumpTextureEnabled) {\r\n ubo.updateFloat3(\"vBumpInfos\", this._bumpTexture.coordinatesIndex, 1.0 / this._bumpTexture.level, this.parallaxScaleBias);\r\n MaterialHelper.BindTextureMatrix(this._bumpTexture, ubo, \"bump\");\r\n\r\n if (scene._mirroredCameraPosition) {\r\n ubo.updateFloat2(\"vTangentSpaceParams\", this._invertNormalMapX ? 1.0 : -1.0, this._invertNormalMapY ? 1.0 : -1.0);\r\n } else {\r\n ubo.updateFloat2(\"vTangentSpaceParams\", this._invertNormalMapX ? -1.0 : 1.0, this._invertNormalMapY ? -1.0 : 1.0);\r\n }\r\n }\r\n\r\n if (this._refractionTexture && StandardMaterial.RefractionTextureEnabled) {\r\n let depth = 1.0;\r\n if (!this._refractionTexture.isCube) {\r\n ubo.updateMatrix(\"refractionMatrix\", this._refractionTexture.getReflectionTextureMatrix());\r\n\r\n if ((this._refractionTexture).depth) {\r\n depth = (this._refractionTexture).depth;\r\n }\r\n }\r\n ubo.updateFloat4(\"vRefractionInfos\", this._refractionTexture.level, this.indexOfRefraction, depth, this.invertRefractionY ? -1 : 1);\r\n\r\n if ((this._refractionTexture).boundingBoxSize) {\r\n const cubeTexture = this._refractionTexture;\r\n\r\n ubo.updateVector3(\"vRefractionPosition\", cubeTexture.boundingBoxPosition);\r\n ubo.updateVector3(\"vRefractionSize\", cubeTexture.boundingBoxSize);\r\n }\r\n }\r\n }\r\n\r\n // Point size\r\n if (this.pointsCloud) {\r\n ubo.updateFloat(\"pointSize\", this.pointSize);\r\n }\r\n\r\n if (defines.SPECULARTERM) {\r\n ubo.updateColor4(\"vSpecularColor\", this.specularColor, this.specularPower);\r\n }\r\n\r\n ubo.updateColor3(\"vEmissiveColor\", StandardMaterial.EmissiveTextureEnabled ? this.emissiveColor : Color3.BlackReadOnly);\r\n ubo.updateColor4(\"vDiffuseColor\", this.diffuseColor, this.alpha);\r\n\r\n scene.ambientColor.multiplyToRef(this.ambientColor, this._globalAmbientColor);\r\n ubo.updateColor3(\"vAmbientColor\", this._globalAmbientColor);\r\n }\r\n\r\n // Textures\r\n if (scene.texturesEnabled) {\r\n if (this._diffuseTexture && StandardMaterial.DiffuseTextureEnabled) {\r\n effect.setTexture(\"diffuseSampler\", this._diffuseTexture);\r\n }\r\n\r\n if (this._ambientTexture && StandardMaterial.AmbientTextureEnabled) {\r\n effect.setTexture(\"ambientSampler\", this._ambientTexture);\r\n }\r\n\r\n if (this._opacityTexture && StandardMaterial.OpacityTextureEnabled) {\r\n effect.setTexture(\"opacitySampler\", this._opacityTexture);\r\n }\r\n\r\n if (this._reflectionTexture && StandardMaterial.ReflectionTextureEnabled) {\r\n if (this._reflectionTexture.isCube) {\r\n effect.setTexture(\"reflectionCubeSampler\", this._reflectionTexture);\r\n } else {\r\n effect.setTexture(\"reflection2DSampler\", this._reflectionTexture);\r\n }\r\n }\r\n\r\n if (this._emissiveTexture && StandardMaterial.EmissiveTextureEnabled) {\r\n effect.setTexture(\"emissiveSampler\", this._emissiveTexture);\r\n }\r\n\r\n if (this._lightmapTexture && StandardMaterial.LightmapTextureEnabled) {\r\n effect.setTexture(\"lightmapSampler\", this._lightmapTexture);\r\n }\r\n\r\n if (this._specularTexture && StandardMaterial.SpecularTextureEnabled) {\r\n effect.setTexture(\"specularSampler\", this._specularTexture);\r\n }\r\n\r\n if (this._bumpTexture && scene.getEngine().getCaps().standardDerivatives && StandardMaterial.BumpTextureEnabled) {\r\n effect.setTexture(\"bumpSampler\", this._bumpTexture);\r\n }\r\n\r\n if (this._refractionTexture && StandardMaterial.RefractionTextureEnabled) {\r\n if (this._refractionTexture.isCube) {\r\n effect.setTexture(\"refractionCubeSampler\", this._refractionTexture);\r\n } else {\r\n effect.setTexture(\"refraction2DSampler\", this._refractionTexture);\r\n }\r\n }\r\n }\r\n\r\n // OIT with depth peeling\r\n if (this.getScene().useOrderIndependentTransparency && this.needAlphaBlendingForMesh(mesh)) {\r\n this.getScene().depthPeelingRenderer!.bind(effect);\r\n }\r\n\r\n this._eventInfo.subMesh = subMesh;\r\n this._callbackPluginEventBindForSubMesh(this._eventInfo);\r\n\r\n // Clip plane\r\n bindClipPlane(effect, this, scene);\r\n\r\n // Colors\r\n this.bindEyePosition(effect);\r\n } else if (scene.getEngine()._features.needToAlwaysBindUniformBuffers) {\r\n this._needToBindSceneUbo = true;\r\n }\r\n\r\n if (mustRebind || !this.isFrozen) {\r\n // Lights\r\n if (scene.lightsEnabled && !this._disableLighting) {\r\n MaterialHelper.BindLights(scene, mesh, effect, defines, this._maxSimultaneousLights);\r\n }\r\n\r\n // View\r\n if (\r\n (scene.fogEnabled && mesh.applyFog && scene.fogMode !== Scene.FOGMODE_NONE) ||\r\n this._reflectionTexture ||\r\n this._refractionTexture ||\r\n mesh.receiveShadows ||\r\n defines.PREPASS\r\n ) {\r\n this.bindView(effect);\r\n }\r\n\r\n // Fog\r\n MaterialHelper.BindFogParameters(scene, mesh, effect);\r\n\r\n // Morph targets\r\n if (defines.NUM_MORPH_INFLUENCERS) {\r\n MaterialHelper.BindMorphTargetParameters(mesh, effect);\r\n }\r\n\r\n if (defines.BAKED_VERTEX_ANIMATION_TEXTURE) {\r\n mesh.bakedVertexAnimationManager?.bind(effect, defines.INSTANCES);\r\n }\r\n\r\n // Log. depth\r\n if (this.useLogarithmicDepth) {\r\n MaterialHelper.BindLogDepth(defines, effect, scene);\r\n }\r\n\r\n // image processing\r\n if (this._imageProcessingConfiguration && !this._imageProcessingConfiguration.applyByPostProcess) {\r\n this._imageProcessingConfiguration.bind(this._activeEffect);\r\n }\r\n }\r\n\r\n this._afterBind(mesh, this._activeEffect);\r\n ubo.update();\r\n }\r\n\r\n /**\r\n * Get the list of animatables in the material.\r\n * @returns the list of animatables object used in the material\r\n */\r\n public getAnimatables(): IAnimatable[] {\r\n const results = super.getAnimatables();\r\n\r\n if (this._diffuseTexture && this._diffuseTexture.animations && this._diffuseTexture.animations.length > 0) {\r\n results.push(this._diffuseTexture);\r\n }\r\n\r\n if (this._ambientTexture && this._ambientTexture.animations && this._ambientTexture.animations.length > 0) {\r\n results.push(this._ambientTexture);\r\n }\r\n\r\n if (this._opacityTexture && this._opacityTexture.animations && this._opacityTexture.animations.length > 0) {\r\n results.push(this._opacityTexture);\r\n }\r\n\r\n if (this._reflectionTexture && this._reflectionTexture.animations && this._reflectionTexture.animations.length > 0) {\r\n results.push(this._reflectionTexture);\r\n }\r\n\r\n if (this._emissiveTexture && this._emissiveTexture.animations && this._emissiveTexture.animations.length > 0) {\r\n results.push(this._emissiveTexture);\r\n }\r\n\r\n if (this._specularTexture && this._specularTexture.animations && this._specularTexture.animations.length > 0) {\r\n results.push(this._specularTexture);\r\n }\r\n\r\n if (this._bumpTexture && this._bumpTexture.animations && this._bumpTexture.animations.length > 0) {\r\n results.push(this._bumpTexture);\r\n }\r\n\r\n if (this._lightmapTexture && this._lightmapTexture.animations && this._lightmapTexture.animations.length > 0) {\r\n results.push(this._lightmapTexture);\r\n }\r\n\r\n if (this._refractionTexture && this._refractionTexture.animations && this._refractionTexture.animations.length > 0) {\r\n results.push(this._refractionTexture);\r\n }\r\n\r\n return results;\r\n }\r\n\r\n /**\r\n * Gets the active textures from the material\r\n * @returns an array of textures\r\n */\r\n public getActiveTextures(): BaseTexture[] {\r\n const activeTextures = super.getActiveTextures();\r\n\r\n if (this._diffuseTexture) {\r\n activeTextures.push(this._diffuseTexture);\r\n }\r\n\r\n if (this._ambientTexture) {\r\n activeTextures.push(this._ambientTexture);\r\n }\r\n\r\n if (this._opacityTexture) {\r\n activeTextures.push(this._opacityTexture);\r\n }\r\n\r\n if (this._reflectionTexture) {\r\n activeTextures.push(this._reflectionTexture);\r\n }\r\n\r\n if (this._emissiveTexture) {\r\n activeTextures.push(this._emissiveTexture);\r\n }\r\n\r\n if (this._specularTexture) {\r\n activeTextures.push(this._specularTexture);\r\n }\r\n\r\n if (this._bumpTexture) {\r\n activeTextures.push(this._bumpTexture);\r\n }\r\n\r\n if (this._lightmapTexture) {\r\n activeTextures.push(this._lightmapTexture);\r\n }\r\n\r\n if (this._refractionTexture) {\r\n activeTextures.push(this._refractionTexture);\r\n }\r\n\r\n return activeTextures;\r\n }\r\n\r\n /**\r\n * Specifies if the material uses a texture\r\n * @param texture defines the texture to check against the material\r\n * @returns a boolean specifying if the material uses the texture\r\n */\r\n public hasTexture(texture: BaseTexture): boolean {\r\n if (super.hasTexture(texture)) {\r\n return true;\r\n }\r\n\r\n if (this._diffuseTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._ambientTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._opacityTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._reflectionTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._emissiveTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._specularTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._bumpTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._lightmapTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._refractionTexture === texture) {\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Disposes the material\r\n * @param forceDisposeEffect specifies if effects should be forcefully disposed\r\n * @param forceDisposeTextures specifies if textures should be forcefully disposed\r\n */\r\n public dispose(forceDisposeEffect?: boolean, forceDisposeTextures?: boolean): void {\r\n if (forceDisposeTextures) {\r\n this._diffuseTexture?.dispose();\r\n this._ambientTexture?.dispose();\r\n this._opacityTexture?.dispose();\r\n this._reflectionTexture?.dispose();\r\n this._emissiveTexture?.dispose();\r\n this._specularTexture?.dispose();\r\n this._bumpTexture?.dispose();\r\n this._lightmapTexture?.dispose();\r\n this._refractionTexture?.dispose();\r\n }\r\n\r\n if (this._imageProcessingConfiguration && this._imageProcessingObserver) {\r\n this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver);\r\n }\r\n\r\n super.dispose(forceDisposeEffect, forceDisposeTextures);\r\n }\r\n\r\n /**\r\n * Makes a duplicate of the material, and gives it a new name\r\n * @param name defines the new name for the duplicated material\r\n * @param cloneTexturesOnlyOnce - if a texture is used in more than one channel (e.g diffuse and opacity), only clone it once and reuse it on the other channels. Default false.\r\n * @param rootUrl defines the root URL to use to load textures\r\n * @returns the cloned material\r\n */\r\n public clone(name: string, cloneTexturesOnlyOnce: boolean = true, rootUrl = \"\"): StandardMaterial {\r\n const result = SerializationHelper.Clone(() => new StandardMaterial(name, this.getScene()), this, { cloneTexturesOnlyOnce });\r\n\r\n result.name = name;\r\n result.id = name;\r\n\r\n this.stencil.copyTo(result.stencil);\r\n\r\n this._clonePlugins(result, rootUrl);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a standard material from parsed material data\r\n * @param source defines the JSON representation of the material\r\n * @param scene defines the hosting scene\r\n * @param rootUrl defines the root URL to use to load textures and relative dependencies\r\n * @returns a new standard material\r\n */\r\n public static Parse(source: any, scene: Scene, rootUrl: string): StandardMaterial {\r\n const material = SerializationHelper.Parse(() => new StandardMaterial(source.name, scene), source, scene, rootUrl);\r\n\r\n if (source.stencil) {\r\n material.stencil.parse(source.stencil, scene, rootUrl);\r\n }\r\n\r\n Material._parsePlugins(source, material, scene, rootUrl);\r\n\r\n return material;\r\n }\r\n\r\n // Flags used to enable or disable a type of texture for all Standard Materials\r\n /**\r\n * Are diffuse textures enabled in the application.\r\n */\r\n public static get DiffuseTextureEnabled(): boolean {\r\n return MaterialFlags.DiffuseTextureEnabled;\r\n }\r\n public static set DiffuseTextureEnabled(value: boolean) {\r\n MaterialFlags.DiffuseTextureEnabled = value;\r\n }\r\n\r\n /**\r\n * Are detail textures enabled in the application.\r\n */\r\n public static get DetailTextureEnabled(): boolean {\r\n return MaterialFlags.DetailTextureEnabled;\r\n }\r\n public static set DetailTextureEnabled(value: boolean) {\r\n MaterialFlags.DetailTextureEnabled = value;\r\n }\r\n\r\n /**\r\n * Are ambient textures enabled in the application.\r\n */\r\n public static get AmbientTextureEnabled(): boolean {\r\n return MaterialFlags.AmbientTextureEnabled;\r\n }\r\n public static set AmbientTextureEnabled(value: boolean) {\r\n MaterialFlags.AmbientTextureEnabled = value;\r\n }\r\n\r\n /**\r\n * Are opacity textures enabled in the application.\r\n */\r\n public static get OpacityTextureEnabled(): boolean {\r\n return MaterialFlags.OpacityTextureEnabled;\r\n }\r\n public static set OpacityTextureEnabled(value: boolean) {\r\n MaterialFlags.OpacityTextureEnabled = value;\r\n }\r\n\r\n /**\r\n * Are reflection textures enabled in the application.\r\n */\r\n public static get ReflectionTextureEnabled(): boolean {\r\n return MaterialFlags.ReflectionTextureEnabled;\r\n }\r\n public static set ReflectionTextureEnabled(value: boolean) {\r\n MaterialFlags.ReflectionTextureEnabled = value;\r\n }\r\n\r\n /**\r\n * Are emissive textures enabled in the application.\r\n */\r\n public static get EmissiveTextureEnabled(): boolean {\r\n return MaterialFlags.EmissiveTextureEnabled;\r\n }\r\n public static set EmissiveTextureEnabled(value: boolean) {\r\n MaterialFlags.EmissiveTextureEnabled = value;\r\n }\r\n\r\n /**\r\n * Are specular textures enabled in the application.\r\n */\r\n public static get SpecularTextureEnabled(): boolean {\r\n return MaterialFlags.SpecularTextureEnabled;\r\n }\r\n public static set SpecularTextureEnabled(value: boolean) {\r\n MaterialFlags.SpecularTextureEnabled = value;\r\n }\r\n\r\n /**\r\n * Are bump textures enabled in the application.\r\n */\r\n public static get BumpTextureEnabled(): boolean {\r\n return MaterialFlags.BumpTextureEnabled;\r\n }\r\n public static set BumpTextureEnabled(value: boolean) {\r\n MaterialFlags.BumpTextureEnabled = value;\r\n }\r\n\r\n /**\r\n * Are lightmap textures enabled in the application.\r\n */\r\n public static get LightmapTextureEnabled(): boolean {\r\n return MaterialFlags.LightmapTextureEnabled;\r\n }\r\n public static set LightmapTextureEnabled(value: boolean) {\r\n MaterialFlags.LightmapTextureEnabled = value;\r\n }\r\n\r\n /**\r\n * Are refraction textures enabled in the application.\r\n */\r\n public static get RefractionTextureEnabled(): boolean {\r\n return MaterialFlags.RefractionTextureEnabled;\r\n }\r\n public static set RefractionTextureEnabled(value: boolean) {\r\n MaterialFlags.RefractionTextureEnabled = value;\r\n }\r\n\r\n /**\r\n * Are color grading textures enabled in the application.\r\n */\r\n public static get ColorGradingTextureEnabled(): boolean {\r\n return MaterialFlags.ColorGradingTextureEnabled;\r\n }\r\n public static set ColorGradingTextureEnabled(value: boolean) {\r\n MaterialFlags.ColorGradingTextureEnabled = value;\r\n }\r\n\r\n /**\r\n * Are fresnels enabled in the application.\r\n */\r\n public static get FresnelEnabled(): boolean {\r\n return MaterialFlags.FresnelEnabled;\r\n }\r\n public static set FresnelEnabled(value: boolean) {\r\n MaterialFlags.FresnelEnabled = value;\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.StandardMaterial\", StandardMaterial);\r\n\r\nScene.DefaultMaterialFactory = (scene: Scene) => {\r\n return new StandardMaterial(\"default material\", scene);\r\n};\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\n\nconst name = \"postprocessVertexShader\";\nconst shader = `attribute vec2 position;uniform vec2 scale;varying vec2 vUV;const vec2 madd=vec2(0.5,0.5);\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\nvUV=(position*madd+madd)*scale;gl_Position=vec4(position,0.0,1.0);\n#define CUSTOM_VERTEX_MAIN_END\n}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const postprocessVertexShader = { name, shader };\n","import type { InternalTexture } from \"../../Materials/Textures/internalTexture\";\r\nimport type { TextureSize } from \"../../Materials/Textures/textureCreationOptions\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { Engine } from \"../engine\";\r\nimport { RenderTargetWrapper } from \"../renderTargetWrapper\";\r\nimport type { ThinEngine } from \"../thinEngine\";\r\n\r\n/** @internal */\r\nexport class WebGLRenderTargetWrapper extends RenderTargetWrapper {\r\n private _context: WebGLRenderingContext;\r\n\r\n /**\r\n * @internal\r\n */\r\n public _framebuffer: Nullable = null;\r\n /**\r\n * @internal\r\n */\r\n public _depthStencilBuffer: Nullable = null;\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n /**\r\n * @internal\r\n */\r\n public _MSAAFramebuffer: Nullable = null;\r\n\r\n // Multiview\r\n /**\r\n * @internal\r\n */\r\n public _colorTextureArray: Nullable = null;\r\n /**\r\n * @internal\r\n */\r\n public _depthStencilTextureArray: Nullable = null;\r\n /**\r\n * @internal\r\n */\r\n public _disposeOnlyFramebuffers = false;\r\n\r\n constructor(isMulti: boolean, isCube: boolean, size: TextureSize, engine: ThinEngine, context: WebGLRenderingContext) {\r\n super(isMulti, isCube, size, engine);\r\n\r\n this._context = context;\r\n }\r\n\r\n protected _cloneRenderTargetWrapper(): Nullable {\r\n let rtw: Nullable = null;\r\n\r\n if (this._colorTextureArray && this._depthStencilTextureArray) {\r\n rtw = (this._engine as Engine).createMultiviewRenderTargetTexture(this.width, this.height);\r\n rtw.texture!.isReady = true;\r\n } else {\r\n rtw = super._cloneRenderTargetWrapper();\r\n }\r\n\r\n return rtw;\r\n }\r\n\r\n protected _swapRenderTargetWrapper(target: WebGLRenderTargetWrapper): void {\r\n super._swapRenderTargetWrapper(target);\r\n\r\n target._framebuffer = this._framebuffer;\r\n target._depthStencilBuffer = this._depthStencilBuffer;\r\n target._MSAAFramebuffer = this._MSAAFramebuffer;\r\n target._colorTextureArray = this._colorTextureArray;\r\n target._depthStencilTextureArray = this._depthStencilTextureArray;\r\n\r\n this._framebuffer = this._depthStencilBuffer = this._MSAAFramebuffer = this._colorTextureArray = this._depthStencilTextureArray = null;\r\n }\r\n\r\n /**\r\n * Shares the depth buffer of this render target with another render target.\r\n * @internal\r\n * @param renderTarget Destination renderTarget\r\n */\r\n public _shareDepth(renderTarget: WebGLRenderTargetWrapper): void {\r\n super._shareDepth(renderTarget);\r\n\r\n const gl = this._context;\r\n const depthbuffer = this._depthStencilBuffer;\r\n const framebuffer = renderTarget._MSAAFramebuffer || renderTarget._framebuffer;\r\n\r\n if (renderTarget._depthStencilBuffer) {\r\n gl.deleteRenderbuffer(renderTarget._depthStencilBuffer);\r\n }\r\n renderTarget._depthStencilBuffer = this._depthStencilBuffer;\r\n\r\n this._engine._bindUnboundFramebuffer(framebuffer);\r\n gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_ATTACHMENT, gl.RENDERBUFFER, depthbuffer);\r\n this._engine._bindUnboundFramebuffer(null);\r\n }\r\n\r\n /**\r\n * Binds a texture to this render target on a specific attachment\r\n * @param texture The texture to bind to the framebuffer\r\n * @param attachmentIndex Index of the attachment\r\n * @param faceIndexOrLayer The face or layer of the texture to render to in case of cube texture or array texture\r\n * @param lodLevel defines the lod level to bind to the frame buffer\r\n */\r\n private _bindTextureRenderTarget(texture: InternalTexture, attachmentIndex: number = 0, faceIndexOrLayer?: number, lodLevel: number = 0) {\r\n if (!texture._hardwareTexture) {\r\n return;\r\n }\r\n\r\n const framebuffer = this._framebuffer;\r\n\r\n const currentFB = this._engine._currentFramebuffer;\r\n this._engine._bindUnboundFramebuffer(framebuffer);\r\n\r\n if (this._engine.webGLVersion > 1) {\r\n const gl = this._context as WebGL2RenderingContext;\r\n\r\n const attachment = (gl)[\"COLOR_ATTACHMENT\" + attachmentIndex];\r\n if (texture.is2DArray || texture.is3D) {\r\n faceIndexOrLayer = faceIndexOrLayer ?? this.layerIndices?.[attachmentIndex] ?? 0;\r\n gl.framebufferTextureLayer(gl.FRAMEBUFFER, attachment, texture._hardwareTexture.underlyingResource, lodLevel, faceIndexOrLayer);\r\n } else if (texture.isCube) {\r\n // if face index is not specified, try to query it from faceIndices\r\n // default is face 0\r\n faceIndexOrLayer = faceIndexOrLayer ?? this.faceIndices?.[attachmentIndex] ?? 0;\r\n gl.framebufferTexture2D(gl.FRAMEBUFFER, attachment, gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndexOrLayer, texture._hardwareTexture.underlyingResource, lodLevel);\r\n } else {\r\n gl.framebufferTexture2D(gl.FRAMEBUFFER, attachment, gl.TEXTURE_2D, texture._hardwareTexture.underlyingResource, lodLevel);\r\n }\r\n } else {\r\n // Default behavior (WebGL)\r\n const gl = this._context;\r\n\r\n const attachment = (gl)[\"COLOR_ATTACHMENT\" + attachmentIndex + \"_WEBGL\"];\r\n const target = faceIndexOrLayer !== undefined ? gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndexOrLayer : gl.TEXTURE_2D;\r\n\r\n gl.framebufferTexture2D(gl.FRAMEBUFFER, attachment, target, texture._hardwareTexture.underlyingResource, lodLevel);\r\n }\r\n\r\n this._engine._bindUnboundFramebuffer(currentFB);\r\n }\r\n\r\n /**\r\n * Set a texture in the textures array\r\n * @param texture the texture to set\r\n * @param index the index in the textures array to set\r\n * @param disposePrevious If this function should dispose the previous texture\r\n */\r\n public setTexture(texture: InternalTexture, index: number = 0, disposePrevious: boolean = true) {\r\n super.setTexture(texture, index, disposePrevious);\r\n this._bindTextureRenderTarget(texture, index);\r\n }\r\n\r\n /**\r\n * Sets the layer and face indices of every render target texture\r\n * @param layers The layer of the texture to be set (make negative to not modify)\r\n * @param faces The face of the texture to be set (make negative to not modify)\r\n */\r\n public setLayerAndFaceIndices(layers: number[], faces: number[]) {\r\n super.setLayerAndFaceIndices(layers, faces);\r\n\r\n if (!this.textures || !this.layerIndices || !this.faceIndices) {\r\n return;\r\n }\r\n\r\n // the length of this._attachments is the right one as it does not count the depth texture, in case we generated it\r\n const textureCount = this._attachments?.length ?? this.textures.length;\r\n for (let index = 0; index < textureCount; index++) {\r\n const texture = this.textures[index];\r\n if (!texture) {\r\n // The target type was probably -1 at creation time and setTexture has not been called yet for this index\r\n continue;\r\n }\r\n if (texture.is2DArray || texture.is3D) {\r\n this._bindTextureRenderTarget(texture, index, this.layerIndices[index]);\r\n } else if (texture.isCube) {\r\n this._bindTextureRenderTarget(texture, index, this.faceIndices[index]);\r\n } else {\r\n this._bindTextureRenderTarget(texture, index);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Set the face and layer indices of a texture in the textures array\r\n * @param index The index of the texture in the textures array to modify\r\n * @param layer The layer of the texture to be set\r\n * @param face The face of the texture to be set\r\n */\r\n public setLayerAndFaceIndex(index: number = 0, layer?: number, face?: number): void {\r\n super.setLayerAndFaceIndex(index, layer, face);\r\n\r\n if (!this.textures || !this.layerIndices || !this.faceIndices) {\r\n return;\r\n }\r\n\r\n const texture = this.textures[index];\r\n if (texture.is2DArray || texture.is3D) {\r\n this._bindTextureRenderTarget(this.textures[index], index, this.layerIndices[index]);\r\n } else if (texture.isCube) {\r\n this._bindTextureRenderTarget(this.textures[index], index, this.faceIndices[index]);\r\n }\r\n }\r\n\r\n public dispose(disposeOnlyFramebuffers = this._disposeOnlyFramebuffers): void {\r\n const gl = this._context;\r\n\r\n if (!disposeOnlyFramebuffers) {\r\n if (this._colorTextureArray) {\r\n this._context.deleteTexture(this._colorTextureArray);\r\n this._colorTextureArray = null;\r\n }\r\n if (this._depthStencilTextureArray) {\r\n this._context.deleteTexture(this._depthStencilTextureArray);\r\n this._depthStencilTextureArray = null;\r\n }\r\n }\r\n\r\n if (this._framebuffer) {\r\n gl.deleteFramebuffer(this._framebuffer);\r\n this._framebuffer = null;\r\n }\r\n\r\n if (this._depthStencilBuffer) {\r\n gl.deleteRenderbuffer(this._depthStencilBuffer);\r\n this._depthStencilBuffer = null;\r\n }\r\n\r\n if (this._MSAAFramebuffer) {\r\n gl.deleteFramebuffer(this._MSAAFramebuffer);\r\n this._MSAAFramebuffer = null;\r\n }\r\n\r\n super.dispose(disposeOnlyFramebuffers);\r\n }\r\n}\r\n","import type { InternalTexture } from \"../Materials/Textures/internalTexture\";\r\nimport { InternalTextureSource } from \"../Materials/Textures/internalTexture\";\r\nimport type { RenderTargetCreationOptions, TextureSize } from \"../Materials/Textures/textureCreationOptions\";\r\nimport type { Nullable } from \"../types\";\r\nimport { Constants } from \"./constants\";\r\nimport type { ThinEngine } from \"./thinEngine\";\r\nimport type { IMultiRenderTargetOptions } from \"../Materials/Textures/multiRenderTarget\";\r\n\r\n/**\r\n * An interface enforcing the renderTarget accessor to used by render target textures.\r\n */\r\nexport interface IRenderTargetTexture {\r\n /**\r\n * Entry point to access the wrapper on a texture.\r\n */\r\n renderTarget: Nullable;\r\n}\r\n\r\n/**\r\n * Wrapper around a render target (either single or multi textures)\r\n */\r\nexport class RenderTargetWrapper {\r\n protected _engine: ThinEngine;\r\n private _size: TextureSize;\r\n private _isCube: boolean;\r\n private _isMulti: boolean;\r\n private _textures: Nullable = null;\r\n private _faceIndices: Nullable = null;\r\n private _layerIndices: Nullable = null;\r\n /** @internal */\r\n public _samples = 1;\r\n\r\n /** @internal */\r\n public _attachments: Nullable = null;\r\n /** @internal */\r\n public _generateStencilBuffer: boolean = false;\r\n /** @internal */\r\n public _generateDepthBuffer: boolean = false;\r\n\r\n /** @internal */\r\n public _depthStencilTexture: Nullable;\r\n /** @internal */\r\n public _depthStencilTextureWithStencil: boolean = false;\r\n\r\n /**\r\n * Gets the depth/stencil texture (if created by a createDepthStencilTexture() call)\r\n */\r\n public get depthStencilTexture() {\r\n return this._depthStencilTexture;\r\n }\r\n\r\n /**\r\n * Indicates if the depth/stencil texture has a stencil aspect\r\n */\r\n public get depthStencilTextureWithStencil() {\r\n return this._depthStencilTextureWithStencil;\r\n }\r\n\r\n /**\r\n * Defines if the render target wrapper is for a cube texture or if false a 2d texture\r\n */\r\n public get isCube(): boolean {\r\n return this._isCube;\r\n }\r\n\r\n /**\r\n * Defines if the render target wrapper is for a single or multi target render wrapper\r\n */\r\n public get isMulti(): boolean {\r\n return this._isMulti;\r\n }\r\n\r\n /**\r\n * Defines if the render target wrapper is for a single or an array of textures\r\n */\r\n public get is2DArray(): boolean {\r\n return this.layers > 0;\r\n }\r\n\r\n /**\r\n * Gets the size of the render target wrapper (used for cubes, as width=height in this case)\r\n */\r\n public get size(): number {\r\n return this.width;\r\n }\r\n\r\n /**\r\n * Gets the width of the render target wrapper\r\n */\r\n public get width(): number {\r\n return (<{ width: number; height: number }>this._size).width || this._size;\r\n }\r\n\r\n /**\r\n * Gets the height of the render target wrapper\r\n */\r\n public get height(): number {\r\n return (<{ width: number; height: number }>this._size).height || this._size;\r\n }\r\n\r\n /**\r\n * Gets the number of layers of the render target wrapper (only used if is2DArray is true and wrapper is not a multi render target)\r\n */\r\n public get layers(): number {\r\n return (<{ width: number; height: number; layers?: number }>this._size).layers || 0;\r\n }\r\n\r\n /**\r\n * Gets the render texture. If this is a multi render target, gets the first texture\r\n */\r\n public get texture(): Nullable {\r\n return this._textures?.[0] ?? null;\r\n }\r\n\r\n /**\r\n * Gets the list of render textures. If we are not in a multi render target, the list will be null (use the texture getter instead)\r\n */\r\n public get textures(): Nullable {\r\n return this._textures;\r\n }\r\n\r\n /**\r\n * Gets the face indices that correspond to the list of render textures. If we are not in a multi render target, the list will be null\r\n */\r\n public get faceIndices(): Nullable {\r\n return this._faceIndices;\r\n }\r\n\r\n /**\r\n * Gets the layer indices that correspond to the list of render textures. If we are not in a multi render target, the list will be null\r\n */\r\n public get layerIndices(): Nullable {\r\n return this._layerIndices;\r\n }\r\n\r\n /**\r\n * Gets the sample count of the render target\r\n */\r\n public get samples(): number {\r\n return this._samples;\r\n }\r\n\r\n /**\r\n * Sets the sample count of the render target\r\n * @param value sample count\r\n * @param initializeBuffers If set to true, the engine will make an initializing call to drawBuffers (only used when isMulti=true).\r\n * @param force true to force calling the update sample count engine function even if the current sample count is equal to value\r\n * @returns the sample count that has been set\r\n */\r\n public setSamples(value: number, initializeBuffers = true, force = false): number {\r\n if (this.samples === value && !force) {\r\n return value;\r\n }\r\n\r\n const result = this._isMulti\r\n ? this._engine.updateMultipleRenderTargetTextureSampleCount(this, value, initializeBuffers)\r\n : this._engine.updateRenderTargetTextureSampleCount(this, value);\r\n this._samples = value;\r\n return result;\r\n }\r\n\r\n /**\r\n * Initializes the render target wrapper\r\n * @param isMulti true if the wrapper is a multi render target\r\n * @param isCube true if the wrapper should render to a cube texture\r\n * @param size size of the render target (width/height/layers)\r\n * @param engine engine used to create the render target\r\n */\r\n constructor(isMulti: boolean, isCube: boolean, size: TextureSize, engine: ThinEngine) {\r\n this._isMulti = isMulti;\r\n this._isCube = isCube;\r\n this._size = size;\r\n this._engine = engine;\r\n this._depthStencilTexture = null;\r\n }\r\n\r\n /**\r\n * Sets the render target texture(s)\r\n * @param textures texture(s) to set\r\n */\r\n public setTextures(textures: Nullable | Nullable): void {\r\n if (Array.isArray(textures)) {\r\n this._textures = textures;\r\n } else if (textures) {\r\n this._textures = [textures];\r\n } else {\r\n this._textures = null;\r\n }\r\n }\r\n\r\n /**\r\n * Set a texture in the textures array\r\n * @param texture The texture to set\r\n * @param index The index in the textures array to set\r\n * @param disposePrevious If this function should dispose the previous texture\r\n */\r\n public setTexture(texture: InternalTexture, index: number = 0, disposePrevious: boolean = true): void {\r\n if (!this._textures) {\r\n this._textures = [];\r\n }\r\n if (this._textures[index] && disposePrevious) {\r\n this._textures[index].dispose();\r\n }\r\n\r\n this._textures[index] = texture;\r\n }\r\n\r\n /**\r\n * Sets the layer and face indices of every render target texture bound to each color attachment\r\n * @param layers The layers of each texture to be set\r\n * @param faces The faces of each texture to be set\r\n */\r\n public setLayerAndFaceIndices(layers: number[], faces: number[]) {\r\n this._layerIndices = layers;\r\n this._faceIndices = faces;\r\n }\r\n\r\n /**\r\n * Sets the layer and face indices of a texture in the textures array that should be bound to each color attachment\r\n * @param index The index of the texture in the textures array to modify\r\n * @param layer The layer of the texture to be set\r\n * @param face The face of the texture to be set\r\n */\r\n public setLayerAndFaceIndex(index: number = 0, layer?: number, face?: number): void {\r\n if (!this._layerIndices) {\r\n this._layerIndices = [];\r\n }\r\n if (!this._faceIndices) {\r\n this._faceIndices = [];\r\n }\r\n\r\n if (layer !== undefined && layer >= 0) {\r\n this._layerIndices[index] = layer;\r\n }\r\n if (face !== undefined && face >= 0) {\r\n this._faceIndices[index] = face;\r\n }\r\n }\r\n\r\n /**\r\n * Creates the depth/stencil texture\r\n * @param comparisonFunction Comparison function to use for the texture\r\n * @param bilinearFiltering true if bilinear filtering should be used when sampling the texture\r\n * @param generateStencil true if the stencil aspect should also be created\r\n * @param samples sample count to use when creating the texture\r\n * @param format format of the depth texture\r\n * @param label defines the label to use for the texture (for debugging purpose only)\r\n * @returns the depth/stencil created texture\r\n */\r\n public createDepthStencilTexture(\r\n comparisonFunction: number = 0,\r\n bilinearFiltering: boolean = true,\r\n generateStencil: boolean = false,\r\n samples: number = 1,\r\n format: number = Constants.TEXTUREFORMAT_DEPTH32_FLOAT,\r\n label?: string\r\n ): InternalTexture {\r\n this._depthStencilTexture?.dispose();\r\n\r\n this._depthStencilTextureWithStencil = generateStencil;\r\n this._depthStencilTexture = this._engine.createDepthStencilTexture(\r\n this._size,\r\n {\r\n bilinearFiltering,\r\n comparisonFunction,\r\n generateStencil,\r\n isCube: this._isCube,\r\n samples,\r\n depthTextureFormat: format,\r\n label,\r\n },\r\n this\r\n );\r\n\r\n return this._depthStencilTexture;\r\n }\r\n\r\n /**\r\n * Shares the depth buffer of this render target with another render target.\r\n * @internal\r\n * @param renderTarget Destination renderTarget\r\n */\r\n public _shareDepth(renderTarget: RenderTargetWrapper): void {\r\n if (this._depthStencilTexture) {\r\n if (renderTarget._depthStencilTexture) {\r\n renderTarget._depthStencilTexture.dispose();\r\n }\r\n\r\n renderTarget._depthStencilTexture = this._depthStencilTexture;\r\n this._depthStencilTexture.incrementReferences();\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _swapAndDie(target: InternalTexture): void {\r\n if (this.texture) {\r\n this.texture._swapAndDie(target);\r\n }\r\n this._textures = null;\r\n this.dispose(true);\r\n }\r\n\r\n protected _cloneRenderTargetWrapper(): Nullable {\r\n let rtw: Nullable = null;\r\n\r\n if (this._isMulti) {\r\n const textureArray = this.textures;\r\n if (textureArray && textureArray.length > 0) {\r\n let generateDepthTexture = false;\r\n let textureCount = textureArray.length;\r\n\r\n const lastTextureSource = textureArray[textureArray.length - 1]._source;\r\n if (lastTextureSource === InternalTextureSource.Depth || lastTextureSource === InternalTextureSource.DepthStencil) {\r\n generateDepthTexture = true;\r\n textureCount--;\r\n }\r\n\r\n const samplingModes: number[] = [];\r\n const types: number[] = [];\r\n const formats: number[] = [];\r\n const targetTypes: number[] = [];\r\n const faceIndex: number[] = [];\r\n const layerIndex: number[] = [];\r\n const layerCounts: number[] = [];\r\n const internalTexture2Index: { [id: number]: number } = {};\r\n\r\n for (let i = 0; i < textureCount; ++i) {\r\n const texture = textureArray[i];\r\n\r\n samplingModes.push(texture.samplingMode);\r\n types.push(texture.type);\r\n formats.push(texture.format);\r\n\r\n const index = internalTexture2Index[texture.uniqueId];\r\n if (index !== undefined) {\r\n targetTypes.push(-1);\r\n layerCounts.push(0);\r\n } else {\r\n internalTexture2Index[texture.uniqueId] = i;\r\n if (texture.is2DArray) {\r\n targetTypes.push(Constants.TEXTURE_2D_ARRAY);\r\n layerCounts.push(texture.depth);\r\n } else if (texture.isCube) {\r\n targetTypes.push(Constants.TEXTURE_CUBE_MAP);\r\n layerCounts.push(0);\r\n } /*else if (texture.isCubeArray) {\r\n targetTypes.push(Constants.TEXTURE_CUBE_MAP_ARRAY);\r\n layerCounts.push(texture.depth);\r\n }*/ else if (texture.is3D) {\r\n targetTypes.push(Constants.TEXTURE_3D);\r\n layerCounts.push(texture.depth);\r\n } else {\r\n targetTypes.push(Constants.TEXTURE_2D);\r\n layerCounts.push(0);\r\n }\r\n }\r\n\r\n if (this._faceIndices) {\r\n faceIndex.push(this._faceIndices[i] ?? 0);\r\n }\r\n if (this._layerIndices) {\r\n layerIndex.push(this._layerIndices[i] ?? 0);\r\n }\r\n }\r\n\r\n const optionsMRT: IMultiRenderTargetOptions = {\r\n samplingModes,\r\n generateMipMaps: textureArray[0].generateMipMaps,\r\n generateDepthBuffer: this._generateDepthBuffer,\r\n generateStencilBuffer: this._generateStencilBuffer,\r\n generateDepthTexture,\r\n types,\r\n formats,\r\n textureCount,\r\n targetTypes,\r\n faceIndex,\r\n layerIndex,\r\n layerCounts,\r\n };\r\n const size = {\r\n width: this.width,\r\n height: this.height,\r\n };\r\n\r\n rtw = this._engine.createMultipleRenderTarget(size, optionsMRT);\r\n\r\n for (let i = 0; i < textureCount; ++i) {\r\n if (targetTypes[i] !== -1) {\r\n continue;\r\n }\r\n const index = internalTexture2Index[textureArray[i].uniqueId];\r\n rtw.setTexture(rtw.textures![index], i);\r\n }\r\n }\r\n } else {\r\n const options: RenderTargetCreationOptions = {};\r\n\r\n options.generateDepthBuffer = this._generateDepthBuffer;\r\n options.generateMipMaps = this.texture?.generateMipMaps ?? false;\r\n options.generateStencilBuffer = this._generateStencilBuffer;\r\n options.samplingMode = this.texture?.samplingMode;\r\n options.type = this.texture?.type;\r\n options.format = this.texture?.format;\r\n\r\n if (this.isCube) {\r\n rtw = this._engine.createRenderTargetCubeTexture(this.width, options);\r\n } else {\r\n const size = {\r\n width: this.width,\r\n height: this.height,\r\n layers: this.is2DArray ? this.texture?.depth : undefined,\r\n };\r\n\r\n rtw = this._engine.createRenderTargetTexture(size, options);\r\n }\r\n rtw.texture!.isReady = true;\r\n }\r\n\r\n return rtw;\r\n }\r\n\r\n protected _swapRenderTargetWrapper(target: RenderTargetWrapper): void {\r\n if (this._textures && target._textures) {\r\n for (let i = 0; i < this._textures.length; ++i) {\r\n this._textures[i]._swapAndDie(target._textures[i], false);\r\n target._textures[i].isReady = true;\r\n }\r\n }\r\n if (this._depthStencilTexture && target._depthStencilTexture) {\r\n this._depthStencilTexture._swapAndDie(target._depthStencilTexture);\r\n target._depthStencilTexture.isReady = true;\r\n }\r\n\r\n this._textures = null;\r\n this._depthStencilTexture = null;\r\n }\r\n\r\n /** @internal */\r\n public _rebuild(): void {\r\n const rtw = this._cloneRenderTargetWrapper();\r\n if (!rtw) {\r\n return;\r\n }\r\n\r\n if (this._depthStencilTexture) {\r\n const samplingMode = this._depthStencilTexture.samplingMode;\r\n const bilinear =\r\n samplingMode === Constants.TEXTURE_BILINEAR_SAMPLINGMODE ||\r\n samplingMode === Constants.TEXTURE_TRILINEAR_SAMPLINGMODE ||\r\n samplingMode === Constants.TEXTURE_LINEAR_LINEAR_MIPNEAREST;\r\n rtw.createDepthStencilTexture(this._depthStencilTexture._comparisonFunction, bilinear, this._depthStencilTextureWithStencil, this._depthStencilTexture.samples);\r\n }\r\n\r\n if (this.samples > 1) {\r\n rtw.setSamples(this.samples);\r\n }\r\n\r\n rtw._swapRenderTargetWrapper(this);\r\n rtw.dispose();\r\n }\r\n\r\n /**\r\n * Releases the internal render textures\r\n */\r\n public releaseTextures(): void {\r\n if (this._textures) {\r\n for (let i = 0; i < this._textures?.length ?? 0; ++i) {\r\n this._textures[i].dispose();\r\n }\r\n }\r\n this._textures = null;\r\n }\r\n\r\n /**\r\n * Disposes the whole render target wrapper\r\n * @param disposeOnlyFramebuffers true if only the frame buffers should be released (used for the WebGL engine). If false, all the textures will also be released\r\n */\r\n public dispose(disposeOnlyFramebuffers = false): void {\r\n if (!disposeOnlyFramebuffers) {\r\n this._depthStencilTexture?.dispose();\r\n this._depthStencilTexture = null;\r\n this.releaseTextures();\r\n }\r\n\r\n this._engine._releaseRenderTargetWrapper(this);\r\n }\r\n}\r\n","import { InternalTexture, InternalTextureSource } from \"../../Materials/Textures/internalTexture\";\r\nimport { Logger } from \"../../Misc/logger\";\r\nimport type { RenderTargetCreationOptions, DepthTextureCreationOptions, TextureSize } from \"../../Materials/Textures/textureCreationOptions\";\r\nimport { ThinEngine } from \"../thinEngine\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { RenderTargetWrapper } from \"../renderTargetWrapper\";\r\nimport { WebGLRenderTargetWrapper } from \"../WebGL/webGLRenderTargetWrapper\";\r\nimport type { WebGLHardwareTexture } from \"../WebGL/webGLHardwareTexture\";\r\n\r\nimport { Constants } from \"../constants\";\r\n\r\n/**\r\n * Type used to define a texture size (either with a number or with a rect width and height)\r\n * @deprecated please use TextureSize instead\r\n */\r\nexport type RenderTargetTextureSize = TextureSize;\r\n\r\ndeclare module \"../../Engines/thinEngine\" {\r\n export interface ThinEngine {\r\n /**\r\n * Creates a new render target texture\r\n * @param size defines the size of the texture\r\n * @param options defines the options used to create the texture\r\n * @returns a new render target wrapper ready to render texture\r\n */\r\n createRenderTargetTexture(size: TextureSize, options: boolean | RenderTargetCreationOptions): RenderTargetWrapper;\r\n\r\n /**\r\n * Creates a depth stencil texture.\r\n * This is only available in WebGL 2 or with the depth texture extension available.\r\n * @param size The size of face edge in the texture.\r\n * @param options The options defining the texture.\r\n * @param rtWrapper The render target wrapper for which the depth/stencil texture must be created\r\n * @returns The texture\r\n */\r\n createDepthStencilTexture(size: TextureSize, options: DepthTextureCreationOptions, rtWrapper: RenderTargetWrapper): InternalTexture;\r\n\r\n /**\r\n * Updates the sample count of a render target texture\r\n * @see https://doc.babylonjs.com/setup/support/webGL2#multisample-render-targets\r\n * @param rtWrapper defines the render target wrapper to update\r\n * @param samples defines the sample count to set\r\n * @returns the effective sample count (could be 0 if multisample render targets are not supported)\r\n */\r\n updateRenderTargetTextureSampleCount(rtWrapper: Nullable, samples: number): number;\r\n\r\n /** @internal */\r\n _createDepthStencilTexture(size: TextureSize, options: DepthTextureCreationOptions, rtWrapper: RenderTargetWrapper): InternalTexture;\r\n\r\n /** @internal */\r\n _createHardwareRenderTargetWrapper(isMulti: boolean, isCube: boolean, size: TextureSize): RenderTargetWrapper;\r\n }\r\n}\r\n\r\nThinEngine.prototype._createHardwareRenderTargetWrapper = function (isMulti: boolean, isCube: boolean, size: TextureSize): RenderTargetWrapper {\r\n const rtWrapper = new WebGLRenderTargetWrapper(isMulti, isCube, size, this, this._gl);\r\n this._renderTargetWrapperCache.push(rtWrapper);\r\n return rtWrapper;\r\n};\r\n\r\nThinEngine.prototype.createRenderTargetTexture = function (this: ThinEngine, size: TextureSize, options: boolean | RenderTargetCreationOptions): RenderTargetWrapper {\r\n const rtWrapper = this._createHardwareRenderTargetWrapper(false, false, size) as WebGLRenderTargetWrapper;\r\n\r\n let generateDepthBuffer = true;\r\n let generateStencilBuffer = false;\r\n let noColorAttachment = false;\r\n let colorAttachment: InternalTexture | undefined = undefined;\r\n let samples = 1;\r\n if (options !== undefined && typeof options === \"object\") {\r\n generateDepthBuffer = options.generateDepthBuffer ?? true;\r\n generateStencilBuffer = !!options.generateStencilBuffer;\r\n noColorAttachment = !!options.noColorAttachment;\r\n colorAttachment = options.colorAttachment;\r\n samples = options.samples ?? 1;\r\n }\r\n\r\n const texture = colorAttachment || (noColorAttachment ? null : this._createInternalTexture(size, options, true, InternalTextureSource.RenderTarget));\r\n const width = (<{ width: number; height: number; layers?: number }>size).width || size;\r\n const height = (<{ width: number; height: number; layers?: number }>size).height || size;\r\n\r\n const currentFrameBuffer = this._currentFramebuffer;\r\n const gl = this._gl;\r\n\r\n // Create the framebuffer\r\n const framebuffer = gl.createFramebuffer();\r\n this._bindUnboundFramebuffer(framebuffer);\r\n rtWrapper._depthStencilBuffer = this._setupFramebufferDepthAttachments(generateStencilBuffer, generateDepthBuffer, width, height);\r\n\r\n // No need to rebind on every frame\r\n if (texture && !texture.is2DArray) {\r\n gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, texture._hardwareTexture!.underlyingResource, 0);\r\n }\r\n\r\n this._bindUnboundFramebuffer(currentFrameBuffer);\r\n\r\n rtWrapper._framebuffer = framebuffer;\r\n rtWrapper._generateDepthBuffer = generateDepthBuffer;\r\n rtWrapper._generateStencilBuffer = generateStencilBuffer;\r\n\r\n rtWrapper.setTextures(texture);\r\n\r\n this.updateRenderTargetTextureSampleCount(rtWrapper, samples);\r\n\r\n return rtWrapper;\r\n};\r\n\r\nThinEngine.prototype.createDepthStencilTexture = function (size: TextureSize, options: DepthTextureCreationOptions, rtWrapper: RenderTargetWrapper): InternalTexture {\r\n if (options.isCube) {\r\n const width = (<{ width: number; height: number }>size).width || size;\r\n return this._createDepthStencilCubeTexture(width, options, rtWrapper);\r\n } else {\r\n return this._createDepthStencilTexture(size, options, rtWrapper);\r\n }\r\n};\r\n\r\nThinEngine.prototype._createDepthStencilTexture = function (size: TextureSize, options: DepthTextureCreationOptions, rtWrapper: RenderTargetWrapper): InternalTexture {\r\n const gl = this._gl;\r\n const layers = (<{ width: number; height: number; layers?: number }>size).layers || 0;\r\n const target = layers !== 0 ? gl.TEXTURE_2D_ARRAY : gl.TEXTURE_2D;\r\n const internalTexture = new InternalTexture(this, InternalTextureSource.DepthStencil);\r\n if (!this._caps.depthTextureExtension) {\r\n Logger.Error(\"Depth texture is not supported by your browser or hardware.\");\r\n return internalTexture;\r\n }\r\n\r\n const internalOptions = {\r\n bilinearFiltering: false,\r\n comparisonFunction: 0,\r\n generateStencil: false,\r\n ...options,\r\n };\r\n\r\n this._bindTextureDirectly(target, internalTexture, true);\r\n\r\n this._setupDepthStencilTexture(\r\n internalTexture,\r\n size,\r\n internalOptions.generateStencil,\r\n internalOptions.comparisonFunction === 0 ? false : internalOptions.bilinearFiltering,\r\n internalOptions.comparisonFunction,\r\n internalOptions.samples\r\n );\r\n\r\n if (internalOptions.depthTextureFormat !== undefined) {\r\n if (\r\n internalOptions.depthTextureFormat !== Constants.TEXTUREFORMAT_DEPTH16 &&\r\n internalOptions.depthTextureFormat !== Constants.TEXTUREFORMAT_DEPTH24 &&\r\n internalOptions.depthTextureFormat !== Constants.TEXTUREFORMAT_DEPTH24UNORM_STENCIL8 &&\r\n internalOptions.depthTextureFormat !== Constants.TEXTUREFORMAT_DEPTH24_STENCIL8 &&\r\n internalOptions.depthTextureFormat !== Constants.TEXTUREFORMAT_DEPTH32_FLOAT &&\r\n internalOptions.depthTextureFormat !== Constants.TEXTUREFORMAT_DEPTH32FLOAT_STENCIL8\r\n ) {\r\n Logger.Error(\"Depth texture format is not supported.\");\r\n return internalTexture;\r\n }\r\n internalTexture.format = internalOptions.depthTextureFormat;\r\n } else {\r\n internalTexture.format = internalOptions.generateStencil ? Constants.TEXTUREFORMAT_DEPTH24_STENCIL8 : Constants.TEXTUREFORMAT_DEPTH24;\r\n }\r\n\r\n const hasStencil =\r\n internalTexture.format === Constants.TEXTUREFORMAT_DEPTH24UNORM_STENCIL8 ||\r\n internalTexture.format === Constants.TEXTUREFORMAT_DEPTH24_STENCIL8 ||\r\n internalTexture.format === Constants.TEXTUREFORMAT_DEPTH32FLOAT_STENCIL8;\r\n\r\n rtWrapper._depthStencilTexture = internalTexture;\r\n rtWrapper._depthStencilTextureWithStencil = hasStencil;\r\n\r\n let type: GLenum = gl.UNSIGNED_INT;\r\n if (internalTexture.format === Constants.TEXTUREFORMAT_DEPTH16) {\r\n type = gl.UNSIGNED_SHORT;\r\n } else if (internalTexture.format === Constants.TEXTUREFORMAT_DEPTH24UNORM_STENCIL8 || internalTexture.format === Constants.TEXTUREFORMAT_DEPTH24_STENCIL8) {\r\n type = gl.UNSIGNED_INT_24_8;\r\n } else if (internalTexture.format === Constants.TEXTUREFORMAT_DEPTH32_FLOAT) {\r\n type = gl.FLOAT;\r\n } else if (internalTexture.format === Constants.TEXTUREFORMAT_DEPTH32FLOAT_STENCIL8) {\r\n type = gl.FLOAT_32_UNSIGNED_INT_24_8_REV;\r\n }\r\n\r\n const format: GLenum = hasStencil ? gl.DEPTH_STENCIL : gl.DEPTH_COMPONENT;\r\n let internalFormat = format;\r\n if (this.webGLVersion > 1) {\r\n if (internalTexture.format === Constants.TEXTUREFORMAT_DEPTH16) {\r\n internalFormat = gl.DEPTH_COMPONENT16;\r\n } else if (internalTexture.format === Constants.TEXTUREFORMAT_DEPTH24) {\r\n internalFormat = gl.DEPTH_COMPONENT24;\r\n } else if (internalTexture.format === Constants.TEXTUREFORMAT_DEPTH24UNORM_STENCIL8 || internalTexture.format === Constants.TEXTUREFORMAT_DEPTH24_STENCIL8) {\r\n internalFormat = gl.DEPTH24_STENCIL8;\r\n } else if (internalTexture.format === Constants.TEXTUREFORMAT_DEPTH32_FLOAT) {\r\n internalFormat = gl.DEPTH_COMPONENT32F;\r\n } else if (internalTexture.format === Constants.TEXTUREFORMAT_DEPTH32FLOAT_STENCIL8) {\r\n internalFormat = gl.DEPTH32F_STENCIL8;\r\n }\r\n }\r\n\r\n if (internalTexture.is2DArray) {\r\n gl.texImage3D(target, 0, internalFormat, internalTexture.width, internalTexture.height, layers, 0, format, type, null);\r\n } else {\r\n gl.texImage2D(target, 0, internalFormat, internalTexture.width, internalTexture.height, 0, format, type, null);\r\n }\r\n\r\n this._bindTextureDirectly(target, null);\r\n\r\n this._internalTexturesCache.push(internalTexture);\r\n\r\n // Dispose previous depth/stencil render buffers and clear the corresponding attachment.\r\n // Next time this framebuffer is bound, the new depth/stencil texture will be attached.\r\n const glRtWrapper = (rtWrapper as any);\r\n if (glRtWrapper._depthStencilBuffer) {\r\n const currentFrameBuffer = this._currentFramebuffer;\r\n this._bindUnboundFramebuffer(glRtWrapper._framebuffer);\r\n gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_STENCIL_ATTACHMENT, gl.RENDERBUFFER, null);\r\n gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_ATTACHMENT, gl.RENDERBUFFER, null);\r\n gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.STENCIL_ATTACHMENT, gl.RENDERBUFFER, null);\r\n this._bindUnboundFramebuffer(currentFrameBuffer);\r\n\r\n gl.deleteRenderbuffer(glRtWrapper._depthStencilBuffer);\r\n glRtWrapper._depthStencilBuffer = null;\r\n }\r\n\r\n return internalTexture;\r\n};\r\n\r\nThinEngine.prototype.updateRenderTargetTextureSampleCount = function (rtWrapper: Nullable, samples: number): number {\r\n if (this.webGLVersion < 2 || !rtWrapper || !rtWrapper.texture) {\r\n return 1;\r\n }\r\n\r\n if (rtWrapper.samples === samples) {\r\n return samples;\r\n }\r\n\r\n const gl = this._gl;\r\n\r\n samples = Math.min(samples, this.getCaps().maxMSAASamples);\r\n\r\n // Dispose previous render buffers\r\n if (rtWrapper._depthStencilBuffer) {\r\n gl.deleteRenderbuffer(rtWrapper._depthStencilBuffer);\r\n rtWrapper._depthStencilBuffer = null;\r\n }\r\n\r\n if (rtWrapper._MSAAFramebuffer) {\r\n gl.deleteFramebuffer(rtWrapper._MSAAFramebuffer);\r\n rtWrapper._MSAAFramebuffer = null;\r\n }\r\n\r\n const hardwareTexture = rtWrapper.texture._hardwareTexture as WebGLHardwareTexture;\r\n hardwareTexture.releaseMSAARenderBuffers();\r\n\r\n if (samples > 1 && typeof gl.renderbufferStorageMultisample === \"function\") {\r\n const framebuffer = gl.createFramebuffer();\r\n\r\n if (!framebuffer) {\r\n throw new Error(\"Unable to create multi sampled framebuffer\");\r\n }\r\n\r\n rtWrapper._MSAAFramebuffer = framebuffer;\r\n this._bindUnboundFramebuffer(rtWrapper._MSAAFramebuffer);\r\n\r\n const colorRenderbuffer = this._createRenderBuffer(\r\n rtWrapper.texture.width,\r\n rtWrapper.texture.height,\r\n samples,\r\n -1 /* not used */,\r\n this._getRGBAMultiSampleBufferFormat(rtWrapper.texture.type),\r\n gl.COLOR_ATTACHMENT0,\r\n false\r\n );\r\n\r\n if (!colorRenderbuffer) {\r\n throw new Error(\"Unable to create multi sampled framebuffer\");\r\n }\r\n\r\n hardwareTexture.addMSAARenderBuffer(colorRenderbuffer);\r\n } else {\r\n this._bindUnboundFramebuffer(rtWrapper._framebuffer);\r\n }\r\n\r\n rtWrapper.texture.samples = samples;\r\n rtWrapper._samples = samples;\r\n rtWrapper._depthStencilBuffer = this._setupFramebufferDepthAttachments(\r\n rtWrapper._generateStencilBuffer,\r\n rtWrapper._generateDepthBuffer,\r\n rtWrapper.texture.width,\r\n rtWrapper.texture.height,\r\n samples\r\n );\r\n\r\n this._bindUnboundFramebuffer(null);\r\n\r\n return samples;\r\n};\r\n","import type { Nullable } from \"../types\";\r\nimport { SmartArray } from \"../Misc/smartArray\";\r\nimport type { Observer } from \"../Misc/observable\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport { Vector2 } from \"../Maths/math.vector\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\nimport type { Effect } from \"../Materials/effect\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport type { RenderTargetCreationOptions } from \"../Materials/Textures/textureCreationOptions\";\r\nimport \"../Shaders/postprocess.vertex\";\r\nimport type { IInspectable } from \"../Misc/iInspectable\";\r\nimport { Engine } from \"../Engines/engine\";\r\nimport type { Color4 } from \"../Maths/math.color\";\r\n\r\nimport \"../Engines/Extensions/engine.renderTarget\";\r\nimport type { NodeMaterial } from \"../Materials/Node/nodeMaterial\";\r\nimport { serialize, serializeAsColor4, SerializationHelper } from \"../Misc/decorators\";\r\nimport { GetClass, RegisterClass } from \"../Misc/typeStore\";\r\nimport { DrawWrapper } from \"../Materials/drawWrapper\";\r\nimport type { AbstractScene } from \"../abstractScene\";\r\nimport type { RenderTargetWrapper } from \"../Engines/renderTargetWrapper\";\r\nimport { ShaderLanguage } from \"../Materials/shaderLanguage\";\r\n\r\nimport type { Scene } from \"../scene\";\r\nimport type { InternalTexture } from \"../Materials/Textures/internalTexture\";\r\nimport type { WebVRFreeCamera } from \"../Cameras/VR/webVRCamera\";\r\nimport type { Animation } from \"../Animations/animation\";\r\nimport type { PrePassRenderer } from \"../Rendering/prePassRenderer\";\r\nimport type { PrePassEffectConfiguration } from \"../Rendering/prePassEffectConfiguration\";\r\n\r\n/**\r\n * Allows for custom processing of the shader code used by a post process\r\n */\r\nexport type PostProcessCustomShaderCodeProcessing = {\r\n /**\r\n * If provided, will be called two times with the vertex and fragment code so that this code can be updated after the #include have been processed\r\n */\r\n processCodeAfterIncludes?: (postProcessName: string, shaderType: string, code: string) => string;\r\n /**\r\n * If provided, will be called two times with the vertex and fragment code so that this code can be updated before it is compiled by the GPU\r\n */\r\n processFinalCode?: (postProcessName: string, shaderType: string, code: string) => string;\r\n /**\r\n * If provided, will be called before creating the effect to collect additional custom bindings (defines, uniforms, samplers)\r\n */\r\n defineCustomBindings?: (postProcessName: string, defines: Nullable, uniforms: string[], samplers: string[]) => Nullable;\r\n /**\r\n * If provided, will be called when binding inputs to the shader code to allow the user to add custom bindings\r\n */\r\n bindCustomBindings?: (postProcessName: string, effect: Effect) => void;\r\n};\r\n\r\n/**\r\n * Size options for a post process\r\n */\r\nexport type PostProcessOptions = { width: number; height: number };\r\n\r\ntype TextureCache = { texture: RenderTargetWrapper; postProcessChannel: number; lastUsedRenderId: number };\r\n\r\n/**\r\n * PostProcess can be used to apply a shader to a texture after it has been rendered\r\n * See https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/usePostProcesses\r\n */\r\nexport class PostProcess {\r\n /** @internal */\r\n public _parentContainer: Nullable = null;\r\n\r\n private static _CustomShaderCodeProcessing: { [postProcessName: string]: PostProcessCustomShaderCodeProcessing } = {};\r\n\r\n /**\r\n * Registers a shader code processing with a post process name.\r\n * @param postProcessName name of the post process. Use null for the fallback shader code processing. This is the shader code processing that will be used in case no specific shader code processing has been associated to a post process name\r\n * @param customShaderCodeProcessing shader code processing to associate to the post process name\r\n * @returns\r\n */\r\n public static RegisterShaderCodeProcessing(postProcessName: Nullable, customShaderCodeProcessing?: PostProcessCustomShaderCodeProcessing) {\r\n if (!customShaderCodeProcessing) {\r\n delete PostProcess._CustomShaderCodeProcessing[postProcessName ?? \"\"];\r\n return;\r\n }\r\n\r\n PostProcess._CustomShaderCodeProcessing[postProcessName ?? \"\"] = customShaderCodeProcessing;\r\n }\r\n\r\n private static _GetShaderCodeProcessing(postProcessName: string) {\r\n return PostProcess._CustomShaderCodeProcessing[postProcessName] ?? PostProcess._CustomShaderCodeProcessing[\"\"];\r\n }\r\n\r\n /**\r\n * Gets or sets the unique id of the post process\r\n */\r\n @serialize()\r\n public uniqueId: number;\r\n\r\n /** Name of the PostProcess. */\r\n @serialize()\r\n public name: string;\r\n\r\n /**\r\n * Width of the texture to apply the post process on\r\n */\r\n @serialize()\r\n public width = -1;\r\n\r\n /**\r\n * Height of the texture to apply the post process on\r\n */\r\n @serialize()\r\n public height = -1;\r\n\r\n /**\r\n * Gets the node material used to create this postprocess (null if the postprocess was manually created)\r\n */\r\n public nodeMaterialSource: Nullable = null;\r\n\r\n /**\r\n * Internal, reference to the location where this postprocess was output to. (Typically the texture on the next postprocess in the chain)\r\n * @internal\r\n */\r\n public _outputTexture: Nullable = null;\r\n /**\r\n * Sampling mode used by the shader\r\n * See https://doc.babylonjs.com/classes/3.1/texture\r\n */\r\n @serialize()\r\n public renderTargetSamplingMode: number;\r\n /**\r\n * Clear color to use when screen clearing\r\n */\r\n @serializeAsColor4()\r\n public clearColor: Color4;\r\n /**\r\n * If the buffer needs to be cleared before applying the post process. (default: true)\r\n * Should be set to false if shader will overwrite all previous pixels.\r\n */\r\n @serialize()\r\n public autoClear = true;\r\n /**\r\n * If clearing the buffer should be forced in autoClear mode, even when alpha mode is enabled (default: false).\r\n * By default, the buffer will only be cleared if alpha mode is disabled (and autoClear is true).\r\n */\r\n @serialize()\r\n public forceAutoClearInAlphaMode = false;\r\n /**\r\n * Type of alpha mode to use when performing the post process (default: Engine.ALPHA_DISABLE)\r\n */\r\n @serialize()\r\n public alphaMode = Constants.ALPHA_DISABLE;\r\n /**\r\n * Sets the setAlphaBlendConstants of the babylon engine\r\n */\r\n @serialize()\r\n public alphaConstants: Color4;\r\n /**\r\n * Animations to be used for the post processing\r\n */\r\n public animations = new Array();\r\n\r\n /**\r\n * Enable Pixel Perfect mode where texture is not scaled to be power of 2.\r\n * Can only be used on a single postprocess or on the last one of a chain. (default: false)\r\n */\r\n @serialize()\r\n public enablePixelPerfectMode = false;\r\n\r\n /**\r\n * Force the postprocess to be applied without taking in account viewport\r\n */\r\n @serialize()\r\n public forceFullscreenViewport = true;\r\n\r\n /**\r\n * List of inspectable custom properties (used by the Inspector)\r\n * @see https://doc.babylonjs.com/toolsAndResources/inspector#extensibility\r\n */\r\n public inspectableCustomProperties: IInspectable[];\r\n\r\n /**\r\n * Scale mode for the post process (default: Engine.SCALEMODE_FLOOR)\r\n *\r\n * | Value | Type | Description |\r\n * | ----- | ----------------------------------- | ----------- |\r\n * | 1 | SCALEMODE_FLOOR | [engine.scalemode_floor](https://doc.babylonjs.com/api/classes/babylon.engine#scalemode_floor) |\r\n * | 2 | SCALEMODE_NEAREST | [engine.scalemode_nearest](https://doc.babylonjs.com/api/classes/babylon.engine#scalemode_nearest) |\r\n * | 3 | SCALEMODE_CEILING | [engine.scalemode_ceiling](https://doc.babylonjs.com/api/classes/babylon.engine#scalemode_ceiling) |\r\n *\r\n */\r\n @serialize()\r\n public scaleMode = Constants.SCALEMODE_FLOOR;\r\n /**\r\n * Force textures to be a power of two (default: false)\r\n */\r\n @serialize()\r\n public alwaysForcePOT = false;\r\n\r\n @serialize(\"samples\")\r\n private _samples = 1;\r\n\r\n /**\r\n * Number of sample textures (default: 1)\r\n */\r\n public get samples() {\r\n return this._samples;\r\n }\r\n\r\n public set samples(n: number) {\r\n this._samples = Math.min(n, this._engine.getCaps().maxMSAASamples);\r\n\r\n this._textures.forEach((texture) => {\r\n texture.setSamples(this._samples);\r\n });\r\n }\r\n\r\n /**\r\n * Modify the scale of the post process to be the same as the viewport (default: false)\r\n */\r\n @serialize()\r\n public adaptScaleToCurrentViewport = false;\r\n\r\n private _camera: Camera;\r\n protected _scene: Scene;\r\n private _engine: Engine;\r\n\r\n private _options: number | PostProcessOptions;\r\n private _reusable = false;\r\n private _renderId = 0;\r\n private _textureType: number;\r\n private _textureFormat: number;\r\n private _shaderLanguage: ShaderLanguage;\r\n\r\n /**\r\n * if externalTextureSamplerBinding is true, the \"apply\" method won't bind the textureSampler texture, it is expected to be done by the \"outside\" (by the onApplyObservable observer most probably).\r\n * counter-productive in some cases because if the texture bound by \"apply\" is different from the currently texture bound, (the one set by the onApplyObservable observer, for eg) some\r\n * internal structures (materialContext) will be dirtified, which may impact performances\r\n */\r\n public externalTextureSamplerBinding = false;\r\n\r\n /**\r\n * Smart array of input and output textures for the post process.\r\n * @internal\r\n */\r\n public _textures = new SmartArray(2);\r\n /**\r\n * Smart array of input and output textures for the post process.\r\n * @internal\r\n */\r\n private _textureCache: TextureCache[] = [];\r\n /**\r\n * The index in _textures that corresponds to the output texture.\r\n * @internal\r\n */\r\n public _currentRenderTextureInd = 0;\r\n private _drawWrapper: DrawWrapper;\r\n private _samplers: string[];\r\n private _fragmentUrl: string;\r\n private _vertexUrl: string;\r\n private _parameters: string[];\r\n protected _postProcessDefines: Nullable;\r\n private _scaleRatio = new Vector2(1, 1);\r\n protected _indexParameters: any;\r\n private _shareOutputWithPostProcess: Nullable;\r\n private _texelSize = Vector2.Zero();\r\n\r\n /** @internal */\r\n public _forcedOutputTexture: Nullable;\r\n\r\n /**\r\n * Prepass configuration in case this post process needs a texture from prepass\r\n * @internal\r\n */\r\n public _prePassEffectConfiguration: PrePassEffectConfiguration;\r\n\r\n /**\r\n * Returns the fragment url or shader name used in the post process.\r\n * @returns the fragment url or name in the shader store.\r\n */\r\n public getEffectName(): string {\r\n return this._fragmentUrl;\r\n }\r\n\r\n // Events\r\n\r\n /**\r\n * An event triggered when the postprocess is activated.\r\n */\r\n public onActivateObservable = new Observable();\r\n\r\n private _onActivateObserver: Nullable>;\r\n /**\r\n * A function that is added to the onActivateObservable\r\n */\r\n public set onActivate(callback: Nullable<(camera: Camera) => void>) {\r\n if (this._onActivateObserver) {\r\n this.onActivateObservable.remove(this._onActivateObserver);\r\n }\r\n if (callback) {\r\n this._onActivateObserver = this.onActivateObservable.add(callback);\r\n }\r\n }\r\n\r\n /**\r\n * An event triggered when the postprocess changes its size.\r\n */\r\n public onSizeChangedObservable = new Observable();\r\n\r\n private _onSizeChangedObserver: Nullable>;\r\n /**\r\n * A function that is added to the onSizeChangedObservable\r\n */\r\n public set onSizeChanged(callback: (postProcess: PostProcess) => void) {\r\n if (this._onSizeChangedObserver) {\r\n this.onSizeChangedObservable.remove(this._onSizeChangedObserver);\r\n }\r\n this._onSizeChangedObserver = this.onSizeChangedObservable.add(callback);\r\n }\r\n\r\n /**\r\n * An event triggered when the postprocess applies its effect.\r\n */\r\n public onApplyObservable = new Observable();\r\n\r\n private _onApplyObserver: Nullable>;\r\n /**\r\n * A function that is added to the onApplyObservable\r\n */\r\n public set onApply(callback: (effect: Effect) => void) {\r\n if (this._onApplyObserver) {\r\n this.onApplyObservable.remove(this._onApplyObserver);\r\n }\r\n this._onApplyObserver = this.onApplyObservable.add(callback);\r\n }\r\n\r\n /**\r\n * An event triggered before rendering the postprocess\r\n */\r\n public onBeforeRenderObservable = new Observable();\r\n\r\n private _onBeforeRenderObserver: Nullable>;\r\n /**\r\n * A function that is added to the onBeforeRenderObservable\r\n */\r\n public set onBeforeRender(callback: (effect: Effect) => void) {\r\n if (this._onBeforeRenderObserver) {\r\n this.onBeforeRenderObservable.remove(this._onBeforeRenderObserver);\r\n }\r\n this._onBeforeRenderObserver = this.onBeforeRenderObservable.add(callback);\r\n }\r\n\r\n /**\r\n * An event triggered after rendering the postprocess\r\n */\r\n public onAfterRenderObservable = new Observable();\r\n\r\n private _onAfterRenderObserver: Nullable>;\r\n /**\r\n * A function that is added to the onAfterRenderObservable\r\n */\r\n public set onAfterRender(callback: (efect: Effect) => void) {\r\n if (this._onAfterRenderObserver) {\r\n this.onAfterRenderObservable.remove(this._onAfterRenderObserver);\r\n }\r\n this._onAfterRenderObserver = this.onAfterRenderObservable.add(callback);\r\n }\r\n\r\n /**\r\n * The input texture for this post process and the output texture of the previous post process. When added to a pipeline the previous post process will\r\n * render it's output into this texture and this texture will be used as textureSampler in the fragment shader of this post process.\r\n */\r\n public get inputTexture(): RenderTargetWrapper {\r\n return this._textures.data[this._currentRenderTextureInd];\r\n }\r\n\r\n public set inputTexture(value: RenderTargetWrapper) {\r\n this._forcedOutputTexture = value;\r\n }\r\n\r\n /**\r\n * Since inputTexture should always be defined, if we previously manually set `inputTexture`,\r\n * the only way to unset it is to use this function to restore its internal state\r\n */\r\n public restoreDefaultInputTexture() {\r\n if (this._forcedOutputTexture) {\r\n this._forcedOutputTexture = null;\r\n this.markTextureDirty();\r\n }\r\n }\r\n\r\n /**\r\n * Gets the camera which post process is applied to.\r\n * @returns The camera the post process is applied to.\r\n */\r\n public getCamera(): Camera {\r\n return this._camera;\r\n }\r\n\r\n /**\r\n * Gets the texel size of the postprocess.\r\n * See https://en.wikipedia.org/wiki/Texel_(graphics)\r\n */\r\n public get texelSize(): Vector2 {\r\n if (this._shareOutputWithPostProcess) {\r\n return this._shareOutputWithPostProcess.texelSize;\r\n }\r\n\r\n if (this._forcedOutputTexture) {\r\n this._texelSize.copyFromFloats(1.0 / this._forcedOutputTexture.width, 1.0 / this._forcedOutputTexture.height);\r\n }\r\n\r\n return this._texelSize;\r\n }\r\n\r\n /**\r\n * Creates a new instance PostProcess\r\n * @param name The name of the PostProcess.\r\n * @param fragmentUrl The url of the fragment shader to be used.\r\n * @param parameters Array of the names of uniform non-sampler2D variables that will be passed to the shader.\r\n * @param samplers Array of the names of uniform sampler2D variables that will be passed to the shader.\r\n * @param options The required width/height ratio to downsize to before computing the render pass. (Use 1.0 for full size)\r\n * @param camera The camera to apply the render pass to.\r\n * @param samplingMode The sampling mode to be used when computing the pass. (default: 0)\r\n * @param engine The engine which the post process will be applied. (default: current engine)\r\n * @param reusable If the post process can be reused on the same frame. (default: false)\r\n * @param defines String of defines that will be set when running the fragment shader. (default: null)\r\n * @param textureType Type of textures used when performing the post process. (default: 0)\r\n * @param vertexUrl The url of the vertex shader to be used. (default: \"postprocess\")\r\n * @param indexParameters The index parameters to be used for babylons include syntax \"#include[0..varyingCount]\". (default: undefined) See usage in babylon.blurPostProcess.ts and kernelBlur.vertex.fx\r\n * @param blockCompilation If the shader should not be compiled immediatly. (default: false)\r\n * @param textureFormat Format of textures used when performing the post process. (default: TEXTUREFORMAT_RGBA)\r\n */\r\n constructor(\r\n name: string,\r\n fragmentUrl: string,\r\n parameters: Nullable,\r\n samplers: Nullable,\r\n options: number | PostProcessOptions,\r\n camera: Nullable,\r\n samplingMode: number = Constants.TEXTURE_NEAREST_SAMPLINGMODE,\r\n engine?: Engine,\r\n reusable?: boolean,\r\n defines: Nullable = null,\r\n textureType: number = Constants.TEXTURETYPE_UNSIGNED_INT,\r\n vertexUrl: string = \"postprocess\",\r\n indexParameters?: any,\r\n blockCompilation = false,\r\n textureFormat = Constants.TEXTUREFORMAT_RGBA,\r\n shaderLanguage = ShaderLanguage.GLSL\r\n ) {\r\n this.name = name;\r\n if (camera != null) {\r\n this._camera = camera;\r\n this._scene = camera.getScene();\r\n camera.attachPostProcess(this);\r\n this._engine = this._scene.getEngine();\r\n\r\n this._scene.postProcesses.push(this);\r\n this.uniqueId = this._scene.getUniqueId();\r\n } else if (engine) {\r\n this._engine = engine;\r\n this._engine.postProcesses.push(this);\r\n }\r\n this._options = options;\r\n this.renderTargetSamplingMode = samplingMode ? samplingMode : Constants.TEXTURE_NEAREST_SAMPLINGMODE;\r\n this._reusable = reusable || false;\r\n this._textureType = textureType;\r\n this._textureFormat = textureFormat;\r\n this._shaderLanguage = shaderLanguage;\r\n\r\n this._samplers = samplers || [];\r\n this._samplers.push(\"textureSampler\");\r\n\r\n this._fragmentUrl = fragmentUrl;\r\n this._vertexUrl = vertexUrl;\r\n this._parameters = parameters || [];\r\n\r\n this._parameters.push(\"scale\");\r\n\r\n this._indexParameters = indexParameters;\r\n this._drawWrapper = new DrawWrapper(this._engine);\r\n\r\n if (!blockCompilation) {\r\n this.updateEffect(defines);\r\n }\r\n }\r\n\r\n /**\r\n * Gets a string identifying the name of the class\r\n * @returns \"PostProcess\" string\r\n */\r\n public getClassName(): string {\r\n return \"PostProcess\";\r\n }\r\n\r\n /**\r\n * Gets the engine which this post process belongs to.\r\n * @returns The engine the post process was enabled with.\r\n */\r\n public getEngine(): Engine {\r\n return this._engine;\r\n }\r\n\r\n /**\r\n * The effect that is created when initializing the post process.\r\n * @returns The created effect corresponding the the postprocess.\r\n */\r\n public getEffect(): Effect {\r\n return this._drawWrapper.effect!;\r\n }\r\n\r\n /**\r\n * To avoid multiple redundant textures for multiple post process, the output the output texture for this post process can be shared with another.\r\n * @param postProcess The post process to share the output with.\r\n * @returns This post process.\r\n */\r\n public shareOutputWith(postProcess: PostProcess): PostProcess {\r\n this._disposeTextures();\r\n\r\n this._shareOutputWithPostProcess = postProcess;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Reverses the effect of calling shareOutputWith and returns the post process back to its original state.\r\n * This should be called if the post process that shares output with this post process is disabled/disposed.\r\n */\r\n public useOwnOutput() {\r\n if (this._textures.length == 0) {\r\n this._textures = new SmartArray(2);\r\n }\r\n\r\n this._shareOutputWithPostProcess = null;\r\n }\r\n\r\n /**\r\n * Updates the effect with the current post process compile time values and recompiles the shader.\r\n * @param defines Define statements that should be added at the beginning of the shader. (default: null)\r\n * @param uniforms Set of uniform variables that will be passed to the shader. (default: null)\r\n * @param samplers Set of Texture2D variables that will be passed to the shader. (default: null)\r\n * @param indexParameters The index parameters to be used for babylons include syntax \"#include[0..varyingCount]\". (default: undefined) See usage in babylon.blurPostProcess.ts and kernelBlur.vertex.fx\r\n * @param onCompiled Called when the shader has been compiled.\r\n * @param onError Called if there is an error when compiling a shader.\r\n * @param vertexUrl The url of the vertex shader to be used (default: the one given at construction time)\r\n * @param fragmentUrl The url of the fragment shader to be used (default: the one given at construction time)\r\n */\r\n public updateEffect(\r\n defines: Nullable = null,\r\n uniforms: Nullable = null,\r\n samplers: Nullable = null,\r\n indexParameters?: any,\r\n onCompiled?: (effect: Effect) => void,\r\n onError?: (effect: Effect, errors: string) => void,\r\n vertexUrl?: string,\r\n fragmentUrl?: string\r\n ) {\r\n const customShaderCodeProcessing = PostProcess._GetShaderCodeProcessing(this.name);\r\n if (customShaderCodeProcessing?.defineCustomBindings) {\r\n const newUniforms = uniforms?.slice() ?? [];\r\n newUniforms.push(...this._parameters);\r\n\r\n const newSamplers = samplers?.slice() ?? [];\r\n newSamplers.push(...this._samplers);\r\n\r\n defines = customShaderCodeProcessing.defineCustomBindings(this.name, defines, newUniforms, newSamplers);\r\n uniforms = newUniforms;\r\n samplers = newSamplers;\r\n }\r\n this._postProcessDefines = defines;\r\n this._drawWrapper.effect = this._engine.createEffect(\r\n { vertex: vertexUrl ?? this._vertexUrl, fragment: fragmentUrl ?? this._fragmentUrl },\r\n {\r\n attributes: [\"position\"],\r\n uniformsNames: uniforms || this._parameters,\r\n uniformBuffersNames: [],\r\n samplers: samplers || this._samplers,\r\n defines: defines !== null ? defines : \"\",\r\n fallbacks: null,\r\n onCompiled: onCompiled ?? null,\r\n onError: onError ?? null,\r\n indexParameters: indexParameters || this._indexParameters,\r\n processCodeAfterIncludes: customShaderCodeProcessing?.processCodeAfterIncludes\r\n ? (shaderType: string, code: string) => customShaderCodeProcessing!.processCodeAfterIncludes!(this.name, shaderType, code)\r\n : null,\r\n processFinalCode: customShaderCodeProcessing?.processFinalCode\r\n ? (shaderType: string, code: string) => customShaderCodeProcessing!.processFinalCode!(this.name, shaderType, code)\r\n : null,\r\n shaderLanguage: this._shaderLanguage,\r\n },\r\n this._engine\r\n );\r\n }\r\n\r\n /**\r\n * The post process is reusable if it can be used multiple times within one frame.\r\n * @returns If the post process is reusable\r\n */\r\n public isReusable(): boolean {\r\n return this._reusable;\r\n }\r\n\r\n /** invalidate frameBuffer to hint the postprocess to create a depth buffer */\r\n public markTextureDirty(): void {\r\n this.width = -1;\r\n }\r\n\r\n private _createRenderTargetTexture(textureSize: { width: number; height: number }, textureOptions: RenderTargetCreationOptions, channel = 0) {\r\n for (let i = 0; i < this._textureCache.length; i++) {\r\n if (\r\n this._textureCache[i].texture.width === textureSize.width &&\r\n this._textureCache[i].texture.height === textureSize.height &&\r\n this._textureCache[i].postProcessChannel === channel &&\r\n this._textureCache[i].texture._generateDepthBuffer === textureOptions.generateDepthBuffer &&\r\n this._textureCache[i].texture.samples === textureOptions.samples\r\n ) {\r\n return this._textureCache[i].texture;\r\n }\r\n }\r\n\r\n const tex = this._engine.createRenderTargetTexture(textureSize, textureOptions);\r\n this._textureCache.push({ texture: tex, postProcessChannel: channel, lastUsedRenderId: -1 });\r\n\r\n return tex;\r\n }\r\n\r\n private _flushTextureCache() {\r\n const currentRenderId = this._renderId;\r\n\r\n for (let i = this._textureCache.length - 1; i >= 0; i--) {\r\n if (currentRenderId - this._textureCache[i].lastUsedRenderId > 100) {\r\n let currentlyUsed = false;\r\n for (let j = 0; j < this._textures.length; j++) {\r\n if (this._textures.data[j] === this._textureCache[i].texture) {\r\n currentlyUsed = true;\r\n break;\r\n }\r\n }\r\n\r\n if (!currentlyUsed) {\r\n this._textureCache[i].texture.dispose();\r\n this._textureCache.splice(i, 1);\r\n }\r\n }\r\n }\r\n }\r\n\r\n private _resize(width: number, height: number, camera: Camera, needMipMaps: boolean, forceDepthStencil?: boolean) {\r\n if (this._textures.length > 0) {\r\n this._textures.reset();\r\n }\r\n\r\n this.width = width;\r\n this.height = height;\r\n\r\n let firstPP = null;\r\n for (let i = 0; i < camera._postProcesses.length; i++) {\r\n if (camera._postProcesses[i] !== null) {\r\n firstPP = camera._postProcesses[i];\r\n break;\r\n }\r\n }\r\n\r\n const textureSize = { width: this.width, height: this.height };\r\n const textureOptions = {\r\n generateMipMaps: needMipMaps,\r\n generateDepthBuffer: forceDepthStencil || firstPP === this,\r\n generateStencilBuffer: (forceDepthStencil || firstPP === this) && this._engine.isStencilEnable,\r\n samplingMode: this.renderTargetSamplingMode,\r\n type: this._textureType,\r\n format: this._textureFormat,\r\n samples: this._samples,\r\n label: \"PostProcessRTT-\" + this.name,\r\n };\r\n\r\n this._textures.push(this._createRenderTargetTexture(textureSize, textureOptions, 0));\r\n\r\n if (this._reusable) {\r\n this._textures.push(this._createRenderTargetTexture(textureSize, textureOptions, 1));\r\n }\r\n\r\n this._texelSize.copyFromFloats(1.0 / this.width, 1.0 / this.height);\r\n\r\n this.onSizeChangedObservable.notifyObservers(this);\r\n }\r\n\r\n /**\r\n * Activates the post process by intializing the textures to be used when executed. Notifies onActivateObservable.\r\n * When this post process is used in a pipeline, this is call will bind the input texture of this post process to the output of the previous.\r\n * @param camera The camera that will be used in the post process. This camera will be used when calling onActivateObservable.\r\n * @param sourceTexture The source texture to be inspected to get the width and height if not specified in the post process constructor. (default: null)\r\n * @param forceDepthStencil If true, a depth and stencil buffer will be generated. (default: false)\r\n * @returns The render target wrapper that was bound to be written to.\r\n */\r\n public activate(camera: Nullable, sourceTexture: Nullable = null, forceDepthStencil?: boolean): RenderTargetWrapper {\r\n camera = camera || this._camera;\r\n\r\n const scene = camera.getScene();\r\n const engine = scene.getEngine();\r\n const maxSize = engine.getCaps().maxTextureSize;\r\n\r\n let requiredWidth = ((sourceTexture ? sourceTexture.width : this._engine.getRenderWidth(true)) * this._options) | 0;\r\n const requiredHeight = ((sourceTexture ? sourceTexture.height : this._engine.getRenderHeight(true)) * this._options) | 0;\r\n\r\n // If rendering to a webvr camera's left or right eye only half the width should be used to avoid resize when rendered to screen\r\n const webVRCamera = camera.parent;\r\n if (webVRCamera && (webVRCamera.leftCamera == camera || webVRCamera.rightCamera == camera)) {\r\n requiredWidth /= 2;\r\n }\r\n\r\n let desiredWidth = (this._options).width || requiredWidth;\r\n let desiredHeight = (this._options).height || requiredHeight;\r\n\r\n const needMipMaps =\r\n this.renderTargetSamplingMode !== Constants.TEXTURE_NEAREST_LINEAR &&\r\n this.renderTargetSamplingMode !== Constants.TEXTURE_NEAREST_NEAREST &&\r\n this.renderTargetSamplingMode !== Constants.TEXTURE_LINEAR_LINEAR;\r\n\r\n if (!this._shareOutputWithPostProcess && !this._forcedOutputTexture) {\r\n if (this.adaptScaleToCurrentViewport) {\r\n const currentViewport = engine.currentViewport;\r\n\r\n if (currentViewport) {\r\n desiredWidth *= currentViewport.width;\r\n desiredHeight *= currentViewport.height;\r\n }\r\n }\r\n\r\n if (needMipMaps || this.alwaysForcePOT) {\r\n if (!(this._options).width) {\r\n desiredWidth = engine.needPOTTextures ? Engine.GetExponentOfTwo(desiredWidth, maxSize, this.scaleMode) : desiredWidth;\r\n }\r\n\r\n if (!(this._options).height) {\r\n desiredHeight = engine.needPOTTextures ? Engine.GetExponentOfTwo(desiredHeight, maxSize, this.scaleMode) : desiredHeight;\r\n }\r\n }\r\n\r\n if (this.width !== desiredWidth || this.height !== desiredHeight) {\r\n this._resize(desiredWidth, desiredHeight, camera, needMipMaps, forceDepthStencil);\r\n }\r\n\r\n this._textures.forEach((texture) => {\r\n if (texture.samples !== this.samples) {\r\n this._engine.updateRenderTargetTextureSampleCount(texture, this.samples);\r\n }\r\n });\r\n\r\n this._flushTextureCache();\r\n this._renderId++;\r\n }\r\n\r\n let target: RenderTargetWrapper;\r\n\r\n if (this._shareOutputWithPostProcess) {\r\n target = this._shareOutputWithPostProcess.inputTexture;\r\n } else if (this._forcedOutputTexture) {\r\n target = this._forcedOutputTexture;\r\n\r\n this.width = this._forcedOutputTexture.width;\r\n this.height = this._forcedOutputTexture.height;\r\n } else {\r\n target = this.inputTexture;\r\n\r\n let cache;\r\n for (let i = 0; i < this._textureCache.length; i++) {\r\n if (this._textureCache[i].texture === target) {\r\n cache = this._textureCache[i];\r\n break;\r\n }\r\n }\r\n\r\n if (cache) {\r\n cache.lastUsedRenderId = this._renderId;\r\n }\r\n }\r\n\r\n // Bind the input of this post process to be used as the output of the previous post process.\r\n if (this.enablePixelPerfectMode) {\r\n this._scaleRatio.copyFromFloats(requiredWidth / desiredWidth, requiredHeight / desiredHeight);\r\n this._engine.bindFramebuffer(target, 0, requiredWidth, requiredHeight, this.forceFullscreenViewport);\r\n } else {\r\n this._scaleRatio.copyFromFloats(1, 1);\r\n this._engine.bindFramebuffer(target, 0, undefined, undefined, this.forceFullscreenViewport);\r\n }\r\n\r\n this._engine._debugInsertMarker?.(`post process ${this.name} input`);\r\n\r\n this.onActivateObservable.notifyObservers(camera);\r\n\r\n // Clear\r\n if (this.autoClear && (this.alphaMode === Constants.ALPHA_DISABLE || this.forceAutoClearInAlphaMode)) {\r\n this._engine.clear(this.clearColor ? this.clearColor : scene.clearColor, scene._allowPostProcessClearColor, true, true);\r\n }\r\n\r\n if (this._reusable) {\r\n this._currentRenderTextureInd = (this._currentRenderTextureInd + 1) % 2;\r\n }\r\n return target;\r\n }\r\n\r\n /**\r\n * If the post process is supported.\r\n */\r\n public get isSupported(): boolean {\r\n return this._drawWrapper.effect!.isSupported;\r\n }\r\n\r\n /**\r\n * The aspect ratio of the output texture.\r\n */\r\n public get aspectRatio(): number {\r\n if (this._shareOutputWithPostProcess) {\r\n return this._shareOutputWithPostProcess.aspectRatio;\r\n }\r\n\r\n if (this._forcedOutputTexture) {\r\n return this._forcedOutputTexture.width / this._forcedOutputTexture.height;\r\n }\r\n return this.width / this.height;\r\n }\r\n\r\n /**\r\n * Get a value indicating if the post-process is ready to be used\r\n * @returns true if the post-process is ready (shader is compiled)\r\n */\r\n public isReady(): boolean {\r\n return this._drawWrapper.effect?.isReady() ?? false;\r\n }\r\n\r\n /**\r\n * Binds all textures and uniforms to the shader, this will be run on every pass.\r\n * @returns the effect corresponding to this post process. Null if not compiled or not ready.\r\n */\r\n public apply(): Nullable {\r\n // Check\r\n if (!this._drawWrapper.effect?.isReady()) {\r\n return null;\r\n }\r\n\r\n // States\r\n this._engine.enableEffect(this._drawWrapper);\r\n this._engine.setState(false);\r\n this._engine.setDepthBuffer(false);\r\n this._engine.setDepthWrite(false);\r\n\r\n // Alpha\r\n this._engine.setAlphaMode(this.alphaMode);\r\n if (this.alphaConstants) {\r\n this.getEngine().setAlphaConstants(this.alphaConstants.r, this.alphaConstants.g, this.alphaConstants.b, this.alphaConstants.a);\r\n }\r\n\r\n // Bind the output texture of the preivous post process as the input to this post process.\r\n let source: RenderTargetWrapper;\r\n if (this._shareOutputWithPostProcess) {\r\n source = this._shareOutputWithPostProcess.inputTexture;\r\n } else if (this._forcedOutputTexture) {\r\n source = this._forcedOutputTexture;\r\n } else {\r\n source = this.inputTexture;\r\n }\r\n\r\n if (!this.externalTextureSamplerBinding) {\r\n this._drawWrapper.effect._bindTexture(\"textureSampler\", source?.texture);\r\n }\r\n\r\n // Parameters\r\n this._drawWrapper.effect.setVector2(\"scale\", this._scaleRatio);\r\n this.onApplyObservable.notifyObservers(this._drawWrapper.effect);\r\n\r\n PostProcess._GetShaderCodeProcessing(this.name)?.bindCustomBindings?.(this.name, this._drawWrapper.effect);\r\n\r\n return this._drawWrapper.effect;\r\n }\r\n\r\n private _disposeTextures() {\r\n if (this._shareOutputWithPostProcess || this._forcedOutputTexture) {\r\n this._disposeTextureCache();\r\n return;\r\n }\r\n\r\n this._disposeTextureCache();\r\n this._textures.dispose();\r\n }\r\n\r\n private _disposeTextureCache() {\r\n for (let i = this._textureCache.length - 1; i >= 0; i--) {\r\n this._textureCache[i].texture.dispose();\r\n }\r\n\r\n this._textureCache.length = 0;\r\n }\r\n\r\n /**\r\n * Sets the required values to the prepass renderer.\r\n * @param prePassRenderer defines the prepass renderer to setup.\r\n * @returns true if the pre pass is needed.\r\n */\r\n public setPrePassRenderer(prePassRenderer: PrePassRenderer): boolean {\r\n if (this._prePassEffectConfiguration) {\r\n this._prePassEffectConfiguration = prePassRenderer.addEffectConfiguration(this._prePassEffectConfiguration);\r\n this._prePassEffectConfiguration.enabled = true;\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Disposes the post process.\r\n * @param camera The camera to dispose the post process on.\r\n */\r\n public dispose(camera?: Camera): void {\r\n camera = camera || this._camera;\r\n\r\n this._disposeTextures();\r\n\r\n let index;\r\n if (this._scene) {\r\n index = this._scene.postProcesses.indexOf(this);\r\n if (index !== -1) {\r\n this._scene.postProcesses.splice(index, 1);\r\n }\r\n }\r\n\r\n if (this._parentContainer) {\r\n const index = this._parentContainer.postProcesses.indexOf(this);\r\n if (index > -1) {\r\n this._parentContainer.postProcesses.splice(index, 1);\r\n }\r\n this._parentContainer = null;\r\n }\r\n\r\n index = this._engine.postProcesses.indexOf(this);\r\n if (index !== -1) {\r\n this._engine.postProcesses.splice(index, 1);\r\n }\r\n\r\n if (!camera) {\r\n return;\r\n }\r\n camera.detachPostProcess(this);\r\n\r\n index = camera._postProcesses.indexOf(this);\r\n if (index === 0 && camera._postProcesses.length > 0) {\r\n const firstPostProcess = this._camera._getFirstPostProcess();\r\n if (firstPostProcess) {\r\n firstPostProcess.markTextureDirty();\r\n }\r\n }\r\n\r\n this.onActivateObservable.clear();\r\n this.onAfterRenderObservable.clear();\r\n this.onApplyObservable.clear();\r\n this.onBeforeRenderObservable.clear();\r\n this.onSizeChangedObservable.clear();\r\n }\r\n\r\n /**\r\n * Serializes the post process to a JSON object\r\n * @returns the JSON object\r\n */\r\n public serialize(): any {\r\n const serializationObject = SerializationHelper.Serialize(this);\r\n const camera = this.getCamera() || (this._scene && this._scene.activeCamera);\r\n serializationObject.customType = \"BABYLON.\" + this.getClassName();\r\n serializationObject.cameraId = camera ? camera.id : null;\r\n serializationObject.reusable = this._reusable;\r\n serializationObject.textureType = this._textureType;\r\n serializationObject.fragmentUrl = this._fragmentUrl;\r\n serializationObject.parameters = this._parameters;\r\n serializationObject.samplers = this._samplers;\r\n serializationObject.options = this._options;\r\n serializationObject.defines = this._postProcessDefines;\r\n serializationObject.textureFormat = this._textureFormat;\r\n serializationObject.vertexUrl = this._vertexUrl;\r\n serializationObject.indexParameters = this._indexParameters;\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Clones this post process\r\n * @returns a new post process similar to this one\r\n */\r\n public clone(): Nullable {\r\n const serializationObject = this.serialize();\r\n serializationObject._engine = this._engine;\r\n serializationObject.cameraId = null;\r\n\r\n const result = PostProcess.Parse(serializationObject, this._scene, \"\");\r\n\r\n if (!result) {\r\n return null;\r\n }\r\n\r\n result.onActivateObservable = this.onActivateObservable.clone();\r\n result.onSizeChangedObservable = this.onSizeChangedObservable.clone();\r\n result.onApplyObservable = this.onApplyObservable.clone();\r\n result.onBeforeRenderObservable = this.onBeforeRenderObservable.clone();\r\n result.onAfterRenderObservable = this.onAfterRenderObservable.clone();\r\n\r\n result._prePassEffectConfiguration = this._prePassEffectConfiguration;\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a material from parsed material data\r\n * @param parsedPostProcess defines parsed post process data\r\n * @param scene defines the hosting scene\r\n * @param rootUrl defines the root URL to use to load textures\r\n * @returns a new post process\r\n */\r\n public static Parse(parsedPostProcess: any, scene: Scene, rootUrl: string): Nullable {\r\n const postProcessType = GetClass(parsedPostProcess.customType);\r\n\r\n if (!postProcessType || !postProcessType._Parse) {\r\n return null;\r\n }\r\n\r\n const camera = scene ? scene.getCameraById(parsedPostProcess.cameraId) : null;\r\n return postProcessType._Parse(parsedPostProcess, camera, scene, rootUrl);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _Parse(parsedPostProcess: any, targetCamera: Camera, scene: Scene, rootUrl: string): Nullable {\r\n return SerializationHelper.Parse(\r\n () => {\r\n return new PostProcess(\r\n parsedPostProcess.name,\r\n parsedPostProcess.fragmentUrl,\r\n parsedPostProcess.parameters,\r\n parsedPostProcess.samplers,\r\n parsedPostProcess.options,\r\n targetCamera,\r\n parsedPostProcess.renderTargetSamplingMode,\r\n parsedPostProcess._engine,\r\n parsedPostProcess.reusable,\r\n parsedPostProcess.defines,\r\n parsedPostProcess.textureType,\r\n parsedPostProcess.vertexUrl,\r\n parsedPostProcess.indexParameters,\r\n false,\r\n parsedPostProcess.textureFormat\r\n );\r\n },\r\n parsedPostProcess,\r\n scene,\r\n rootUrl\r\n );\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.PostProcess\", PostProcess);\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/helperFunctions\";\n\nconst name = \"rgbdDecodePixelShader\";\nconst shader = `varying vec2 vUV;uniform sampler2D textureSampler;\n#include\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) \n{gl_FragColor=vec4(fromRGBD(texture2D(textureSampler,vUV)),1.0);}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const rgbdDecodePixelShader = { name, shader };\n","import { InternalTexture, InternalTextureSource } from \"../../Materials/Textures/internalTexture\";\r\nimport { Logger } from \"../../Misc/logger\";\r\nimport { Constants } from \"../constants\";\r\nimport { ThinEngine } from \"../thinEngine\";\r\nimport type { RenderTargetWrapper } from \"../renderTargetWrapper\";\r\nimport type { WebGLRenderTargetWrapper } from \"../WebGL/webGLRenderTargetWrapper\";\r\nimport type { RenderTargetCreationOptions } from \"../../Materials/Textures/textureCreationOptions\";\r\n\r\ndeclare module \"../../Engines/thinEngine\" {\r\n export interface ThinEngine {\r\n /**\r\n * Creates a new render target cube wrapper\r\n * @param size defines the size of the texture\r\n * @param options defines the options used to create the texture\r\n * @returns a new render target cube wrapper\r\n */\r\n createRenderTargetCubeTexture(size: number, options?: RenderTargetCreationOptions): RenderTargetWrapper;\r\n }\r\n}\r\n\r\nThinEngine.prototype.createRenderTargetCubeTexture = function (size: number, options?: RenderTargetCreationOptions): RenderTargetWrapper {\r\n const rtWrapper = this._createHardwareRenderTargetWrapper(false, true, size) as WebGLRenderTargetWrapper;\r\n\r\n const fullOptions = {\r\n generateMipMaps: true,\r\n generateDepthBuffer: true,\r\n generateStencilBuffer: false,\r\n type: Constants.TEXTURETYPE_UNSIGNED_INT,\r\n samplingMode: Constants.TEXTURE_TRILINEAR_SAMPLINGMODE,\r\n format: Constants.TEXTUREFORMAT_RGBA,\r\n ...options,\r\n };\r\n fullOptions.generateStencilBuffer = fullOptions.generateDepthBuffer && fullOptions.generateStencilBuffer;\r\n\r\n if (fullOptions.type === Constants.TEXTURETYPE_FLOAT && !this._caps.textureFloatLinearFiltering) {\r\n // if floating point linear (gl.FLOAT) then force to NEAREST_SAMPLINGMODE\r\n fullOptions.samplingMode = Constants.TEXTURE_NEAREST_SAMPLINGMODE;\r\n } else if (fullOptions.type === Constants.TEXTURETYPE_HALF_FLOAT && !this._caps.textureHalfFloatLinearFiltering) {\r\n // if floating point linear (HALF_FLOAT) then force to NEAREST_SAMPLINGMODE\r\n fullOptions.samplingMode = Constants.TEXTURE_NEAREST_SAMPLINGMODE;\r\n }\r\n const gl = this._gl;\r\n\r\n const texture = new InternalTexture(this, InternalTextureSource.RenderTarget);\r\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, texture, true);\r\n\r\n const filters = this._getSamplingParameters(fullOptions.samplingMode, fullOptions.generateMipMaps);\r\n\r\n if (fullOptions.type === Constants.TEXTURETYPE_FLOAT && !this._caps.textureFloat) {\r\n fullOptions.type = Constants.TEXTURETYPE_UNSIGNED_INT;\r\n Logger.Warn(\"Float textures are not supported. Cube render target forced to TEXTURETYPE_UNESIGNED_BYTE type\");\r\n }\r\n\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_MAG_FILTER, filters.mag);\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_MIN_FILTER, filters.min);\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\r\n\r\n for (let face = 0; face < 6; face++) {\r\n gl.texImage2D(\r\n gl.TEXTURE_CUBE_MAP_POSITIVE_X + face,\r\n 0,\r\n this._getRGBABufferInternalSizedFormat(fullOptions.type, fullOptions.format),\r\n size,\r\n size,\r\n 0,\r\n this._getInternalFormat(fullOptions.format),\r\n this._getWebGLTextureType(fullOptions.type),\r\n null\r\n );\r\n }\r\n\r\n // Create the framebuffer\r\n const framebuffer = gl.createFramebuffer();\r\n this._bindUnboundFramebuffer(framebuffer);\r\n\r\n rtWrapper._depthStencilBuffer = this._setupFramebufferDepthAttachments(fullOptions.generateStencilBuffer, fullOptions.generateDepthBuffer, size, size);\r\n\r\n // MipMaps\r\n if (fullOptions.generateMipMaps) {\r\n gl.generateMipmap(gl.TEXTURE_CUBE_MAP);\r\n }\r\n\r\n // Unbind\r\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, null);\r\n this._bindUnboundFramebuffer(null);\r\n\r\n rtWrapper._framebuffer = framebuffer;\r\n rtWrapper._generateDepthBuffer = fullOptions.generateDepthBuffer;\r\n rtWrapper._generateStencilBuffer = fullOptions.generateStencilBuffer;\r\n\r\n texture.width = size;\r\n texture.height = size;\r\n texture.isReady = true;\r\n texture.isCube = true;\r\n texture.samples = 1;\r\n texture.generateMipMaps = fullOptions.generateMipMaps;\r\n texture.samplingMode = fullOptions.samplingMode;\r\n texture.type = fullOptions.type;\r\n texture.format = fullOptions.format;\r\n\r\n this._internalTexturesCache.push(texture);\r\n rtWrapper.setTextures(texture);\r\n\r\n return rtWrapper;\r\n};\r\n","import type { Nullable } from \"../types\";\r\nimport type { ThinEngine } from \"../Engines/thinEngine\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport { Viewport } from \"../Maths/math.viewport\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport type { Observer } from \"../Misc/observable\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport { Effect } from \"./effect\";\r\nimport type { DataBuffer } from \"../Buffers/dataBuffer\";\r\nimport { DrawWrapper } from \"./drawWrapper\";\r\nimport type { IRenderTargetTexture, RenderTargetWrapper } from \"../Engines/renderTargetWrapper\";\r\nimport type { ShaderLanguage } from \"./shaderLanguage\";\r\n\r\n// Prevents ES6 Crash if not imported.\r\nimport \"../Shaders/postprocess.vertex\";\r\n\r\n/**\r\n * Effect Render Options\r\n */\r\nexport interface IEffectRendererOptions {\r\n /**\r\n * Defines the vertices positions.\r\n */\r\n positions?: number[];\r\n /**\r\n * Defines the indices.\r\n */\r\n indices?: number[];\r\n}\r\n\r\n// Fullscreen quad buffers by default.\r\nconst defaultOptions = {\r\n positions: [1, 1, -1, 1, -1, -1, 1, -1],\r\n indices: [0, 1, 2, 0, 2, 3],\r\n};\r\n\r\n/**\r\n * Helper class to render one or more effects.\r\n * You can access the previous rendering in your shader by declaring a sampler named textureSampler\r\n */\r\nexport class EffectRenderer {\r\n /**\r\n * The engine the effect renderer has been created for.\r\n */\r\n public readonly engine: ThinEngine;\r\n\r\n private _vertexBuffers: { [key: string]: VertexBuffer };\r\n private _indexBuffer: DataBuffer;\r\n\r\n private _fullscreenViewport = new Viewport(0, 0, 1, 1);\r\n private _onContextRestoredObserver: Nullable>;\r\n\r\n private _savedStateDepthTest: boolean;\r\n private _savedStateStencilTest: boolean;\r\n\r\n /**\r\n * Creates an effect renderer\r\n * @param engine the engine to use for rendering\r\n * @param options defines the options of the effect renderer\r\n */\r\n constructor(engine: ThinEngine, options: IEffectRendererOptions = defaultOptions) {\r\n const positions = options.positions ?? defaultOptions.positions;\r\n const indices = options.indices ?? defaultOptions.indices;\r\n\r\n this.engine = engine;\r\n this._vertexBuffers = {\r\n [VertexBuffer.PositionKind]: new VertexBuffer(engine, positions, VertexBuffer.PositionKind, false, false, 2),\r\n };\r\n this._indexBuffer = engine.createIndexBuffer(indices);\r\n\r\n this._onContextRestoredObserver = engine.onContextRestoredObservable.add(() => {\r\n this._indexBuffer = engine.createIndexBuffer(indices);\r\n\r\n for (const key in this._vertexBuffers) {\r\n const vertexBuffer = this._vertexBuffers[key];\r\n vertexBuffer._rebuild();\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Sets the current viewport in normalized coordinates 0-1\r\n * @param viewport Defines the viewport to set (defaults to 0 0 1 1)\r\n */\r\n public setViewport(viewport = this._fullscreenViewport): void {\r\n this.engine.setViewport(viewport);\r\n }\r\n\r\n /**\r\n * Binds the embedded attributes buffer to the effect.\r\n * @param effect Defines the effect to bind the attributes for\r\n */\r\n public bindBuffers(effect: Effect): void {\r\n this.engine.bindBuffers(this._vertexBuffers, this._indexBuffer, effect);\r\n }\r\n\r\n /**\r\n * Sets the current effect wrapper to use during draw.\r\n * The effect needs to be ready before calling this api.\r\n * This also sets the default full screen position attribute.\r\n * @param effectWrapper Defines the effect to draw with\r\n */\r\n public applyEffectWrapper(effectWrapper: EffectWrapper): void {\r\n this.engine.setState(true);\r\n this.engine.depthCullingState.depthTest = false;\r\n this.engine.stencilState.stencilTest = false;\r\n this.engine.enableEffect(effectWrapper._drawWrapper);\r\n this.bindBuffers(effectWrapper.effect);\r\n effectWrapper.onApplyObservable.notifyObservers({});\r\n }\r\n\r\n /**\r\n * Saves engine states\r\n */\r\n public saveStates(): void {\r\n this._savedStateDepthTest = this.engine.depthCullingState.depthTest;\r\n this._savedStateStencilTest = this.engine.stencilState.stencilTest;\r\n }\r\n\r\n /**\r\n * Restores engine states\r\n */\r\n public restoreStates(): void {\r\n this.engine.depthCullingState.depthTest = this._savedStateDepthTest;\r\n this.engine.stencilState.stencilTest = this._savedStateStencilTest;\r\n }\r\n\r\n /**\r\n * Draws a full screen quad.\r\n */\r\n public draw(): void {\r\n this.engine.drawElementsType(Constants.MATERIAL_TriangleFillMode, 0, 6);\r\n }\r\n\r\n private _isRenderTargetTexture(texture: RenderTargetWrapper | IRenderTargetTexture): texture is IRenderTargetTexture {\r\n return (texture as IRenderTargetTexture).renderTarget !== undefined;\r\n }\r\n\r\n /**\r\n * renders one or more effects to a specified texture\r\n * @param effectWrapper the effect to renderer\r\n * @param outputTexture texture to draw to, if null it will render to the screen.\r\n */\r\n public render(effectWrapper: EffectWrapper, outputTexture: Nullable = null) {\r\n // Ensure effect is ready\r\n if (!effectWrapper.effect.isReady()) {\r\n return;\r\n }\r\n\r\n this.saveStates();\r\n\r\n // Reset state\r\n this.setViewport();\r\n\r\n const out = outputTexture === null ? null : this._isRenderTargetTexture(outputTexture) ? outputTexture.renderTarget! : outputTexture;\r\n\r\n if (out) {\r\n this.engine.bindFramebuffer(out);\r\n }\r\n\r\n this.applyEffectWrapper(effectWrapper);\r\n\r\n this.draw();\r\n\r\n if (out) {\r\n this.engine.unBindFramebuffer(out);\r\n }\r\n\r\n this.restoreStates();\r\n }\r\n\r\n /**\r\n * Disposes of the effect renderer\r\n */\r\n dispose() {\r\n const vertexBuffer = this._vertexBuffers[VertexBuffer.PositionKind];\r\n if (vertexBuffer) {\r\n vertexBuffer.dispose();\r\n delete this._vertexBuffers[VertexBuffer.PositionKind];\r\n }\r\n\r\n if (this._indexBuffer) {\r\n this.engine._releaseBuffer(this._indexBuffer);\r\n }\r\n\r\n if (this._onContextRestoredObserver) {\r\n this.engine.onContextRestoredObservable.remove(this._onContextRestoredObserver);\r\n this._onContextRestoredObserver = null;\r\n }\r\n }\r\n}\r\n\r\n/**\r\n * Options to create an EffectWrapper\r\n */\r\ninterface EffectWrapperCreationOptions {\r\n /**\r\n * Engine to use to create the effect\r\n */\r\n engine: ThinEngine;\r\n /**\r\n * Fragment shader for the effect\r\n */\r\n fragmentShader: string;\r\n /**\r\n * Use the shader store instead of direct source code\r\n */\r\n useShaderStore?: boolean;\r\n /**\r\n * Vertex shader for the effect\r\n */\r\n vertexShader?: string;\r\n /**\r\n * Attributes to use in the shader\r\n */\r\n attributeNames?: Array;\r\n /**\r\n * Uniforms to use in the shader\r\n */\r\n uniformNames?: Array;\r\n /**\r\n * Texture sampler names to use in the shader\r\n */\r\n samplerNames?: Array;\r\n /**\r\n * Defines to use in the shader\r\n */\r\n defines?: Array;\r\n /**\r\n * Callback when effect is compiled\r\n */\r\n onCompiled?: Nullable<(effect: Effect) => void>;\r\n /**\r\n * The friendly name of the effect displayed in Spector.\r\n */\r\n name?: string;\r\n /**\r\n * The language the shader is written in (default: GLSL)\r\n */\r\n shaderLanguage?: ShaderLanguage;\r\n}\r\n\r\n/**\r\n * Wraps an effect to be used for rendering\r\n */\r\nexport class EffectWrapper {\r\n /**\r\n * Event that is fired right before the effect is drawn (should be used to update uniforms)\r\n */\r\n public onApplyObservable = new Observable<{}>();\r\n /**\r\n * The underlying effect\r\n */\r\n public get effect(): Effect {\r\n return this._drawWrapper.effect!;\r\n }\r\n\r\n public set effect(effect: Effect) {\r\n this._drawWrapper.effect = effect;\r\n }\r\n\r\n /** @internal */\r\n public _drawWrapper: DrawWrapper;\r\n\r\n private _onContextRestoredObserver: Nullable>;\r\n\r\n /**\r\n * Creates an effect to be renderer\r\n * @param creationOptions options to create the effect\r\n */\r\n constructor(creationOptions: EffectWrapperCreationOptions) {\r\n let effectCreationOptions: any;\r\n const uniformNames = creationOptions.uniformNames || [];\r\n\r\n if (creationOptions.vertexShader) {\r\n effectCreationOptions = {\r\n fragmentSource: creationOptions.fragmentShader,\r\n vertexSource: creationOptions.vertexShader,\r\n spectorName: creationOptions.name || \"effectWrapper\",\r\n };\r\n } else {\r\n // Default scale to use in post process vertex shader.\r\n uniformNames.push(\"scale\");\r\n\r\n effectCreationOptions = {\r\n fragmentSource: creationOptions.fragmentShader,\r\n vertex: \"postprocess\",\r\n spectorName: creationOptions.name || \"effectWrapper\",\r\n };\r\n\r\n // Sets the default scale to identity for the post process vertex shader.\r\n this.onApplyObservable.add(() => {\r\n this.effect.setFloat2(\"scale\", 1, 1);\r\n });\r\n }\r\n\r\n const defines = creationOptions.defines ? creationOptions.defines.join(\"\\n\") : \"\";\r\n this._drawWrapper = new DrawWrapper(creationOptions.engine);\r\n\r\n if (creationOptions.useShaderStore) {\r\n effectCreationOptions.fragment = effectCreationOptions.fragmentSource;\r\n if (!effectCreationOptions.vertex) {\r\n effectCreationOptions.vertex = effectCreationOptions.vertexSource;\r\n }\r\n\r\n delete effectCreationOptions.fragmentSource;\r\n delete effectCreationOptions.vertexSource;\r\n\r\n this.effect = creationOptions.engine.createEffect(\r\n effectCreationOptions,\r\n creationOptions.attributeNames || [\"position\"],\r\n uniformNames,\r\n creationOptions.samplerNames,\r\n defines,\r\n undefined,\r\n creationOptions.onCompiled,\r\n undefined,\r\n undefined,\r\n creationOptions.shaderLanguage\r\n );\r\n } else {\r\n this.effect = new Effect(\r\n effectCreationOptions,\r\n creationOptions.attributeNames || [\"position\"],\r\n uniformNames,\r\n creationOptions.samplerNames,\r\n creationOptions.engine,\r\n defines,\r\n undefined,\r\n creationOptions.onCompiled,\r\n undefined,\r\n undefined,\r\n undefined,\r\n creationOptions.shaderLanguage\r\n );\r\n\r\n this._onContextRestoredObserver = creationOptions.engine.onContextRestoredObservable.add(() => {\r\n this.effect._pipelineContext = null; // because _prepareEffect will try to dispose this pipeline before recreating it and that would lead to webgl errors\r\n this.effect._wasPreviouslyReady = false;\r\n this.effect._prepareEffect();\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Disposes of the effect wrapper\r\n */\r\n public dispose() {\r\n if (this._onContextRestoredObserver) {\r\n this.effect.getEngine().onContextRestoredObservable.remove(this._onContextRestoredObserver);\r\n this._onContextRestoredObserver = null;\r\n }\r\n this.effect.dispose();\r\n }\r\n}\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\n\nconst name = \"passPixelShader\";\nconst shader = `varying vec2 vUV;uniform sampler2D textureSampler;\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) \n{gl_FragColor=texture2D(textureSampler,vUV);}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const passPixelShader = { name, shader };\n","/* eslint-disable @typescript-eslint/no-unused-vars */\r\nimport { _WarnImport } from \"./devTools\";\r\nimport type { Engine } from \"../Engines/engine\";\r\n\r\nimport { ThinEngine } from \"../Engines/thinEngine\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport { EffectRenderer, EffectWrapper } from \"../Materials/effectRenderer\";\r\nimport { Tools } from \"./tools\";\r\nimport type { Nullable } from \"../types\";\r\n\r\nimport { passPixelShader } from \"../Shaders/pass.fragment\";\r\nimport { Scalar } from \"../Maths/math.scalar\";\r\n\r\ntype DumpToolsEngine = {\r\n canvas: HTMLCanvasElement | OffscreenCanvas;\r\n engine: ThinEngine;\r\n renderer: EffectRenderer;\r\n wrapper: EffectWrapper;\r\n};\r\n\r\n/**\r\n * Class containing a set of static utilities functions to dump data from a canvas\r\n */\r\nexport class DumpTools {\r\n private static _DumpToolsEngine: Nullable;\r\n\r\n private static _CreateDumpRenderer(): DumpToolsEngine {\r\n if (!DumpTools._DumpToolsEngine) {\r\n let canvas: HTMLCanvasElement | OffscreenCanvas = new OffscreenCanvas(100, 100); // will be resized later\r\n let engine: Nullable = null;\r\n const options = {\r\n preserveDrawingBuffer: true,\r\n depth: false,\r\n stencil: false,\r\n alpha: true,\r\n premultipliedAlpha: false,\r\n antialias: false,\r\n failIfMajorPerformanceCaveat: false,\r\n };\r\n try {\r\n engine = new ThinEngine(canvas, false, options);\r\n } catch (e) {\r\n // The browser does not support WebGL context in OffscreenCanvas, fallback on a regular canvas\r\n canvas = document.createElement(\"canvas\");\r\n engine = new ThinEngine(canvas, false, options);\r\n }\r\n engine.getCaps().parallelShaderCompile = undefined;\r\n const renderer = new EffectRenderer(engine);\r\n const wrapper = new EffectWrapper({\r\n engine,\r\n name: passPixelShader.name,\r\n fragmentShader: passPixelShader.shader,\r\n samplerNames: [\"textureSampler\"],\r\n });\r\n DumpTools._DumpToolsEngine = {\r\n canvas,\r\n engine,\r\n renderer,\r\n wrapper,\r\n };\r\n }\r\n return DumpTools._DumpToolsEngine!;\r\n }\r\n\r\n /**\r\n * Dumps the current bound framebuffer\r\n * @param width defines the rendering width\r\n * @param height defines the rendering height\r\n * @param engine defines the hosting engine\r\n * @param successCallback defines the callback triggered once the data are available\r\n * @param mimeType defines the mime type of the result\r\n * @param fileName defines the filename to download. If present, the result will automatically be downloaded\r\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\r\n * @returns a void promise\r\n */\r\n public static async DumpFramebuffer(\r\n width: number,\r\n height: number,\r\n engine: Engine,\r\n successCallback?: (data: string) => void,\r\n mimeType = \"image/png\",\r\n fileName?: string,\r\n quality?: number\r\n ) {\r\n // Read the contents of the framebuffer\r\n const bufferView = await engine.readPixels(0, 0, width, height);\r\n\r\n const data = new Uint8Array(bufferView.buffer);\r\n\r\n DumpTools.DumpData(width, height, data, successCallback as (data: string | ArrayBuffer) => void, mimeType, fileName, true, undefined, quality);\r\n }\r\n\r\n /**\r\n * Dumps an array buffer\r\n * @param width defines the rendering width\r\n * @param height defines the rendering height\r\n * @param data the data array\r\n * @param mimeType defines the mime type of the result\r\n * @param fileName defines the filename to download. If present, the result will automatically be downloaded\r\n * @param invertY true to invert the picture in the Y dimension\r\n * @param toArrayBuffer true to convert the data to an ArrayBuffer (encoded as `mimeType`) instead of a base64 string\r\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\r\n * @returns a promise that resolve to the final data\r\n */\r\n public static DumpDataAsync(\r\n width: number,\r\n height: number,\r\n data: ArrayBufferView,\r\n mimeType = \"image/png\",\r\n fileName?: string,\r\n invertY = false,\r\n toArrayBuffer = false,\r\n quality?: number\r\n ): Promise {\r\n return new Promise((resolve) => {\r\n DumpTools.DumpData(width, height, data, (result) => resolve(result), mimeType, fileName, invertY, toArrayBuffer, quality);\r\n });\r\n }\r\n\r\n /**\r\n * Dumps an array buffer\r\n * @param width defines the rendering width\r\n * @param height defines the rendering height\r\n * @param data the data array\r\n * @param successCallback defines the callback triggered once the data are available\r\n * @param mimeType defines the mime type of the result\r\n * @param fileName defines the filename to download. If present, the result will automatically be downloaded\r\n * @param invertY true to invert the picture in the Y dimension\r\n * @param toArrayBuffer true to convert the data to an ArrayBuffer (encoded as `mimeType`) instead of a base64 string\r\n * @param quality The quality of the image if lossy mimeType is used (e.g. image/jpeg, image/webp). See {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob | HTMLCanvasElement.toBlob()}'s `quality` parameter.\r\n */\r\n public static DumpData(\r\n width: number,\r\n height: number,\r\n data: ArrayBufferView,\r\n successCallback?: (data: string | ArrayBuffer) => void,\r\n mimeType = \"image/png\",\r\n fileName?: string,\r\n invertY = false,\r\n toArrayBuffer = false,\r\n quality?: number\r\n ) {\r\n const renderer = DumpTools._CreateDumpRenderer();\r\n renderer.engine.setSize(width, height, true);\r\n\r\n // Convert if data are float32\r\n if (data instanceof Float32Array) {\r\n const data2 = new Uint8Array(data.length);\r\n let n = data.length;\r\n while (n--) {\r\n const v = data[n];\r\n data2[n] = Math.round(Scalar.Clamp(v) * 255);\r\n }\r\n data = data2;\r\n }\r\n\r\n // Create the image\r\n const texture = renderer.engine.createRawTexture(data, width, height, Constants.TEXTUREFORMAT_RGBA, false, !invertY, Constants.TEXTURE_NEAREST_NEAREST);\r\n\r\n renderer.renderer.setViewport();\r\n renderer.renderer.applyEffectWrapper(renderer.wrapper);\r\n renderer.wrapper.effect._bindTexture(\"textureSampler\", texture);\r\n renderer.renderer.draw();\r\n\r\n if (toArrayBuffer) {\r\n Tools.ToBlob(\r\n renderer.canvas,\r\n (blob) => {\r\n const fileReader = new FileReader();\r\n fileReader.onload = (event: any) => {\r\n const arrayBuffer = event.target!.result as ArrayBuffer;\r\n if (successCallback) {\r\n successCallback(arrayBuffer);\r\n }\r\n };\r\n fileReader.readAsArrayBuffer(blob!);\r\n },\r\n mimeType,\r\n quality\r\n );\r\n } else {\r\n Tools.EncodeScreenshotCanvasData(renderer.canvas, successCallback, mimeType, fileName, quality);\r\n }\r\n\r\n texture.dispose();\r\n }\r\n\r\n /**\r\n * Dispose the dump tools associated resources\r\n */\r\n public static Dispose() {\r\n if (DumpTools._DumpToolsEngine) {\r\n DumpTools._DumpToolsEngine.wrapper.dispose();\r\n DumpTools._DumpToolsEngine.renderer.dispose();\r\n DumpTools._DumpToolsEngine.engine.dispose();\r\n }\r\n DumpTools._DumpToolsEngine = null;\r\n }\r\n}\r\n\r\n/**\r\n * This will be executed automatically for UMD and es5.\r\n * If esm dev wants the side effects to execute they will have to run it manually\r\n * Once we build native modules those need to be exported.\r\n * @internal\r\n */\r\nconst initSideEffects = () => {\r\n // References the dependencies.\r\n Tools.DumpData = DumpTools.DumpData;\r\n Tools.DumpDataAsync = DumpTools.DumpDataAsync;\r\n Tools.DumpFramebuffer = DumpTools.DumpFramebuffer;\r\n};\r\n\r\ninitSideEffects();\r\n","import type { Observer } from \"../../Misc/observable\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport type { SmartArray } from \"../../Misc/smartArray\";\r\nimport type { Nullable, Immutable } from \"../../types\";\r\nimport type { Camera } from \"../../Cameras/camera\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { Matrix, Vector3 } from \"../../Maths/math.vector\";\r\nimport type { Color4 } from \"../../Maths/math.color\";\r\nimport type { RenderTargetCreationOptions, TextureSize } from \"../../Materials/Textures/textureCreationOptions\";\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport type { SubMesh } from \"../../Meshes/subMesh\";\r\nimport type { InternalTexture } from \"../../Materials/Textures/internalTexture\";\r\nimport { Texture } from \"../../Materials/Textures/texture\";\r\nimport { PostProcessManager } from \"../../PostProcesses/postProcessManager\";\r\nimport type { PostProcess } from \"../../PostProcesses/postProcess\";\r\nimport { RenderingManager } from \"../../Rendering/renderingManager\";\r\nimport { Constants } from \"../../Engines/constants\";\r\nimport type { IRenderTargetTexture, RenderTargetWrapper } from \"../../Engines/renderTargetWrapper\";\r\n\r\nimport \"../../Engines/Extensions/engine.renderTarget\";\r\nimport \"../../Engines/Extensions/engine.renderTargetCube\";\r\nimport { Engine } from \"../../Engines/engine\";\r\nimport { _ObserveArray } from \"../../Misc/arrayTools\";\r\nimport { DumpTools } from \"../../Misc/dumpTools\";\r\n\r\nimport type { Material } from \"../material\";\r\n\r\n/**\r\n * Options for the RenderTargetTexture constructor\r\n */\r\nexport interface RenderTargetTextureOptions {\r\n /** True (default: false) if mipmaps need to be generated after render */\r\n generateMipMaps?: boolean;\r\n\r\n /** True (default) to not change the aspect ratio of the scene in the RTT */\r\n doNotChangeAspectRatio?: boolean;\r\n\r\n /** The type of the buffer in the RTT (byte (default), half float, float...) */\r\n type?: number;\r\n\r\n /** True (default: false) if a cube texture needs to be created */\r\n isCube?: boolean;\r\n\r\n /** The sampling mode to be used with the render target (Trilinear (default), Linear, Nearest...) */\r\n samplingMode?: number;\r\n\r\n /** True (default) to generate a depth buffer */\r\n generateDepthBuffer?: boolean;\r\n\r\n /** True (default: false) to generate a stencil buffer */\r\n generateStencilBuffer?: boolean;\r\n\r\n /** True (default: false) if multiple textures need to be created (Draw Buffers) */\r\n isMulti?: boolean;\r\n\r\n /** The internal format of the buffer in the RTT (RED, RG, RGB, RGBA (default), ALPHA...) */\r\n format?: number;\r\n\r\n /** True (default: false) if the texture allocation should be delayed */\r\n delayAllocation?: boolean;\r\n\r\n /** Sample count to use when creating the RTT */\r\n samples?: number;\r\n\r\n /** specific flags to use when creating the texture (e.g., Constants.TEXTURE_CREATIONFLAG_STORAGE for storage textures) */\r\n creationFlags?: number;\r\n\r\n /** True (default: false) to indicate that no color target should be created. (e.g., if you only want to write to the depth buffer) */\r\n noColorAttachment?: boolean;\r\n\r\n /** Specifies the internal texture to use directly instead of creating one (ignores `noColorAttachment` flag when set) **/\r\n colorAttachment?: InternalTexture;\r\n\r\n /** True (default: false) to create a SRGB texture */\r\n useSRGBBuffer?: boolean;\r\n}\r\n\r\n/**\r\n * This Helps creating a texture that will be created from a camera in your scene.\r\n * It is basically a dynamic texture that could be used to create special effects for instance.\r\n * Actually, It is the base of lot of effects in the framework like post process, shadows, effect layers and rendering pipelines...\r\n */\r\nexport class RenderTargetTexture extends Texture implements IRenderTargetTexture {\r\n /**\r\n * The texture will only be rendered once which can be useful to improve performance if everything in your render is static for instance.\r\n */\r\n public static readonly REFRESHRATE_RENDER_ONCE: number = 0;\r\n /**\r\n * The texture will only be rendered rendered every frame and is recommended for dynamic contents.\r\n */\r\n public static readonly REFRESHRATE_RENDER_ONEVERYFRAME: number = 1;\r\n /**\r\n * The texture will be rendered every 2 frames which could be enough if your dynamic objects are not\r\n * the central point of your effect and can save a lot of performances.\r\n */\r\n public static readonly REFRESHRATE_RENDER_ONEVERYTWOFRAMES: number = 2;\r\n\r\n /**\r\n * Use this predicate to dynamically define the list of mesh you want to render.\r\n * If set, the renderList property will be overwritten.\r\n */\r\n public renderListPredicate: (AbstractMesh: AbstractMesh) => boolean;\r\n\r\n private _renderList: Nullable>;\r\n private _unObserveRenderList: Nullable<() => void> = null;\r\n\r\n /**\r\n * Use this list to define the list of mesh you want to render.\r\n */\r\n public get renderList(): Nullable> {\r\n return this._renderList;\r\n }\r\n\r\n public set renderList(value: Nullable>) {\r\n if (this._unObserveRenderList) {\r\n this._unObserveRenderList();\r\n this._unObserveRenderList = null;\r\n }\r\n\r\n if (value) {\r\n this._unObserveRenderList = _ObserveArray(value, this._renderListHasChanged);\r\n }\r\n\r\n this._renderList = value;\r\n }\r\n\r\n private _renderListHasChanged = (_functionName: String, previousLength: number) => {\r\n const newLength = this._renderList ? this._renderList.length : 0;\r\n if ((previousLength === 0 && newLength > 0) || newLength === 0) {\r\n this.getScene()?.meshes.forEach((mesh) => {\r\n mesh._markSubMeshesAsLightDirty();\r\n });\r\n }\r\n };\r\n\r\n /**\r\n * Use this function to overload the renderList array at rendering time.\r\n * Return null to render with the current renderList, else return the list of meshes to use for rendering.\r\n * For 2DArray RTT, layerOrFace is the index of the layer that is going to be rendered, else it is the faceIndex of\r\n * the cube (if the RTT is a cube, else layerOrFace=0).\r\n * The renderList passed to the function is the current render list (the one that will be used if the function returns null).\r\n * The length of this list is passed through renderListLength: don't use renderList.length directly because the array can\r\n * hold dummy elements!\r\n */\r\n public getCustomRenderList: (layerOrFace: number, renderList: Nullable>>, renderListLength: number) => Nullable>;\r\n\r\n /**\r\n * Define if particles should be rendered in your texture.\r\n */\r\n public renderParticles = true;\r\n /**\r\n * Define if sprites should be rendered in your texture.\r\n */\r\n public renderSprites = false;\r\n\r\n /**\r\n * Force checking the layerMask property even if a custom list of meshes is provided (ie. if renderList is not undefined)\r\n */\r\n public forceLayerMaskCheck = false;\r\n\r\n /**\r\n * Define the camera used to render the texture.\r\n */\r\n public activeCamera: Nullable;\r\n /**\r\n * Override the mesh isReady function with your own one.\r\n */\r\n public customIsReadyFunction: (mesh: AbstractMesh, refreshRate: number, preWarm?: boolean) => boolean;\r\n /**\r\n * Override the render function of the texture with your own one.\r\n */\r\n public customRenderFunction: (\r\n opaqueSubMeshes: SmartArray,\r\n alphaTestSubMeshes: SmartArray,\r\n transparentSubMeshes: SmartArray,\r\n depthOnlySubMeshes: SmartArray,\r\n beforeTransparents?: () => void\r\n ) => void;\r\n /**\r\n * Define if camera post processes should be use while rendering the texture.\r\n */\r\n public useCameraPostProcesses: boolean;\r\n /**\r\n * Define if the camera viewport should be respected while rendering the texture or if the render should be done to the entire texture.\r\n */\r\n public ignoreCameraViewport: boolean = false;\r\n\r\n private _postProcessManager: Nullable;\r\n\r\n /**\r\n * Post-processes for this render target\r\n */\r\n public get postProcesses() {\r\n return this._postProcesses;\r\n }\r\n private _postProcesses: PostProcess[];\r\n private _resizeObserver: Nullable>;\r\n\r\n private get _prePassEnabled() {\r\n return !!this._prePassRenderTarget && this._prePassRenderTarget.enabled;\r\n }\r\n\r\n /**\r\n * An event triggered when the texture is unbind.\r\n */\r\n public onBeforeBindObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when the texture is unbind.\r\n */\r\n public onAfterUnbindObservable = new Observable();\r\n\r\n private _onAfterUnbindObserver: Nullable>;\r\n /**\r\n * Set a after unbind callback in the texture.\r\n * This has been kept for backward compatibility and use of onAfterUnbindObservable is recommended.\r\n */\r\n public set onAfterUnbind(callback: () => void) {\r\n if (this._onAfterUnbindObserver) {\r\n this.onAfterUnbindObservable.remove(this._onAfterUnbindObserver);\r\n }\r\n this._onAfterUnbindObserver = this.onAfterUnbindObservable.add(callback);\r\n }\r\n\r\n /**\r\n * An event triggered before rendering the texture\r\n */\r\n public onBeforeRenderObservable = new Observable();\r\n\r\n private _onBeforeRenderObserver: Nullable>;\r\n /**\r\n * Set a before render callback in the texture.\r\n * This has been kept for backward compatibility and use of onBeforeRenderObservable is recommended.\r\n */\r\n public set onBeforeRender(callback: (faceIndex: number) => void) {\r\n if (this._onBeforeRenderObserver) {\r\n this.onBeforeRenderObservable.remove(this._onBeforeRenderObserver);\r\n }\r\n this._onBeforeRenderObserver = this.onBeforeRenderObservable.add(callback);\r\n }\r\n\r\n /**\r\n * An event triggered after rendering the texture\r\n */\r\n public onAfterRenderObservable = new Observable();\r\n\r\n private _onAfterRenderObserver: Nullable>;\r\n /**\r\n * Set a after render callback in the texture.\r\n * This has been kept for backward compatibility and use of onAfterRenderObservable is recommended.\r\n */\r\n public set onAfterRender(callback: (faceIndex: number) => void) {\r\n if (this._onAfterRenderObserver) {\r\n this.onAfterRenderObservable.remove(this._onAfterRenderObserver);\r\n }\r\n this._onAfterRenderObserver = this.onAfterRenderObservable.add(callback);\r\n }\r\n\r\n /**\r\n * An event triggered after the texture clear\r\n */\r\n public onClearObservable = new Observable();\r\n\r\n private _onClearObserver: Nullable>;\r\n /**\r\n * Set a clear callback in the texture.\r\n * This has been kept for backward compatibility and use of onClearObservable is recommended.\r\n */\r\n public set onClear(callback: (Engine: Engine) => void) {\r\n if (this._onClearObserver) {\r\n this.onClearObservable.remove(this._onClearObserver);\r\n }\r\n this._onClearObserver = this.onClearObservable.add(callback);\r\n }\r\n\r\n /**\r\n * An event triggered when the texture is resized.\r\n */\r\n public onResizeObservable = new Observable();\r\n\r\n /**\r\n * Define the clear color of the Render Target if it should be different from the scene.\r\n */\r\n public clearColor: Color4;\r\n protected _size: TextureSize;\r\n protected _initialSizeParameter: number | { width: number; height: number } | { ratio: number };\r\n protected _sizeRatio: Nullable;\r\n /** @internal */\r\n public _generateMipMaps: boolean;\r\n /** @internal */\r\n public _cleared = false;\r\n /**\r\n * Skip the initial clear of the rtt at the beginning of the frame render loop\r\n */\r\n public skipInitialClear = false;\r\n protected _renderingManager: RenderingManager;\r\n /** @internal */\r\n public _waitingRenderList?: string[];\r\n protected _doNotChangeAspectRatio: boolean;\r\n protected _currentRefreshId = -1;\r\n protected _refreshRate = 1;\r\n protected _textureMatrix: Matrix;\r\n protected _samples = 1;\r\n protected _renderTargetOptions: RenderTargetCreationOptions;\r\n private _canRescale = true;\r\n protected _renderTarget: Nullable = null;\r\n /**\r\n * Current render pass id of the render target texture. Note it can change over the rendering as there's a separate id for each face of a cube / each layer of an array layer!\r\n */\r\n public renderPassId: number;\r\n private _renderPassIds: number[];\r\n /**\r\n * Gets the render pass ids used by the render target texture. For a single render target the array length will be 1, for a cube texture it will be 6 and for\r\n * a 2D texture array it will return an array of ids the size of the 2D texture array\r\n */\r\n public get renderPassIds(): readonly number[] {\r\n return this._renderPassIds;\r\n }\r\n\r\n /**\r\n * Gets the current value of the refreshId counter\r\n */\r\n public get currentRefreshId() {\r\n return this._currentRefreshId;\r\n }\r\n\r\n /**\r\n * Sets a specific material to be used to render a mesh/a list of meshes in this render target texture\r\n * @param mesh mesh or array of meshes\r\n * @param material material or array of materials to use for this render pass. If undefined is passed, no specific material will be used but the regular material instead (mesh.material). It's possible to provide an array of materials to use a different material for each rendering in the case of a cube texture (6 rendering) and a 2D texture array (as many rendering as the length of the array)\r\n */\r\n public setMaterialForRendering(mesh: AbstractMesh | AbstractMesh[], material?: Material | Material[]): void {\r\n let meshes;\r\n if (!Array.isArray(mesh)) {\r\n meshes = [mesh];\r\n } else {\r\n meshes = mesh;\r\n }\r\n for (let j = 0; j < meshes.length; ++j) {\r\n for (let i = 0; i < this._renderPassIds.length; ++i) {\r\n meshes[j].setMaterialForRenderPass(this._renderPassIds[i], material !== undefined ? (Array.isArray(material) ? material[i] : material) : undefined);\r\n }\r\n }\r\n }\r\n\r\n private _isCubeData: boolean;\r\n\r\n /**\r\n * Define if the texture has multiple draw buffers or if false a single draw buffer.\r\n */\r\n public get isMulti(): boolean {\r\n return this._renderTarget?.isMulti ?? false;\r\n }\r\n\r\n /**\r\n * Gets render target creation options that were used.\r\n */\r\n public get renderTargetOptions(): RenderTargetCreationOptions {\r\n return this._renderTargetOptions;\r\n }\r\n\r\n /**\r\n * Gets the render target wrapper associated with this render target\r\n */\r\n public get renderTarget(): Nullable {\r\n return this._renderTarget;\r\n }\r\n\r\n protected _onRatioRescale(): void {\r\n if (this._sizeRatio) {\r\n this.resize(this._initialSizeParameter);\r\n }\r\n }\r\n\r\n /**\r\n * Gets or sets the center of the bounding box associated with the texture (when in cube mode)\r\n * It must define where the camera used to render the texture is set\r\n */\r\n public boundingBoxPosition = Vector3.Zero();\r\n\r\n private _boundingBoxSize: Vector3;\r\n\r\n /**\r\n * Gets or sets the size of the bounding box associated with the texture (when in cube mode)\r\n * When defined, the cubemap will switch to local mode\r\n * @see https://community.arm.com/graphics/b/blog/posts/reflections-based-on-local-cubemaps-in-unity\r\n * @example https://www.babylonjs-playground.com/#RNASML\r\n */\r\n public set boundingBoxSize(value: Vector3) {\r\n if (this._boundingBoxSize && this._boundingBoxSize.equals(value)) {\r\n return;\r\n }\r\n this._boundingBoxSize = value;\r\n const scene = this.getScene();\r\n if (scene) {\r\n scene.markAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n }\r\n public get boundingBoxSize(): Vector3 {\r\n return this._boundingBoxSize;\r\n }\r\n\r\n /**\r\n * In case the RTT has been created with a depth texture, get the associated\r\n * depth texture.\r\n * Otherwise, return null.\r\n */\r\n public get depthStencilTexture(): Nullable {\r\n return this._renderTarget?._depthStencilTexture ?? null;\r\n }\r\n\r\n /**\r\n * Instantiate a render target texture. This is mainly used to render of screen the scene to for instance apply post process\r\n * or used a shadow, depth texture...\r\n * @param name The friendly name of the texture\r\n * @param size The size of the RTT (number if square, or {width: number, height:number} or {ratio:} to define a ratio from the main scene)\r\n * @param scene The scene the RTT belongs to. Default is the last created scene.\r\n * @param options The options for creating the render target texture.\r\n */\r\n constructor(name: string, size: number | { width: number; height: number; layers?: number } | { ratio: number }, scene?: Nullable, options?: RenderTargetTextureOptions);\r\n\r\n /**\r\n * Instantiate a render target texture. This is mainly used to render of screen the scene to for instance apply post process\r\n * or used a shadow, depth texture...\r\n * @param name The friendly name of the texture\r\n * @param size The size of the RTT (number if square, or {width: number, height:number} or {ratio:} to define a ratio from the main scene)\r\n * @param scene The scene the RTT belongs to. Default is the last created scene\r\n * @param generateMipMaps True (default: false) if mipmaps need to be generated after render\r\n * @param doNotChangeAspectRatio True (default) to not change the aspect ratio of the scene in the RTT\r\n * @param type The type of the buffer in the RTT (byte (default), half float, float...)\r\n * @param isCube True (default: false) if a cube texture needs to be created\r\n * @param samplingMode The sampling mode to be used with the render target (Trilinear (default), Linear, Nearest...)\r\n * @param generateDepthBuffer True (default) to generate a depth buffer\r\n * @param generateStencilBuffer True (default: false) to generate a stencil buffer\r\n * @param isMulti True (default: false) if multiple textures need to be created (Draw Buffers)\r\n * @param format The internal format of the buffer in the RTT (RED, RG, RGB, RGBA (default), ALPHA...)\r\n * @param delayAllocation True (default: false) if the texture allocation should be delayed\r\n * @param samples Sample count to use when creating the RTT\r\n * @param creationFlags specific flags to use when creating the texture (e.g., Constants.TEXTURE_CREATIONFLAG_STORAGE for storage textures)\r\n * @param noColorAttachment True (default: false) to indicate that no color target should be created. (e.g., if you only want to write to the depth buffer)\r\n * @param useSRGBBuffer True (default: false) to create a SRGB texture\r\n */\r\n constructor(\r\n name: string,\r\n size: number | { width: number; height: number; layers?: number } | { ratio: number },\r\n scene?: Nullable,\r\n generateMipMaps?: boolean,\r\n doNotChangeAspectRatio?: boolean,\r\n type?: number,\r\n isCube?: boolean,\r\n samplingMode?: number,\r\n generateDepthBuffer?: boolean,\r\n generateStencilBuffer?: boolean,\r\n isMulti?: boolean,\r\n format?: number,\r\n delayAllocation?: boolean,\r\n samples?: number,\r\n creationFlags?: number,\r\n noColorAttachment?: boolean,\r\n useSRGBBuffer?: boolean\r\n );\r\n\r\n /** @internal */\r\n constructor(\r\n name: string,\r\n size: number | { width: number; height: number; layers?: number } | { ratio: number },\r\n scene?: Nullable,\r\n generateMipMaps: boolean | RenderTargetTextureOptions = false,\r\n doNotChangeAspectRatio: boolean = true,\r\n type: number = Constants.TEXTURETYPE_UNSIGNED_INT,\r\n isCube = false,\r\n samplingMode = Texture.TRILINEAR_SAMPLINGMODE,\r\n generateDepthBuffer = true,\r\n generateStencilBuffer = false,\r\n isMulti = false,\r\n format = Constants.TEXTUREFORMAT_RGBA,\r\n delayAllocation = false,\r\n samples?: number,\r\n creationFlags?: number,\r\n noColorAttachment = false,\r\n useSRGBBuffer = false\r\n ) {\r\n let colorAttachment: InternalTexture | undefined = undefined;\r\n if (typeof generateMipMaps === \"object\") {\r\n const options = generateMipMaps;\r\n generateMipMaps = !!options.generateMipMaps;\r\n doNotChangeAspectRatio = options.doNotChangeAspectRatio ?? true;\r\n type = options.type ?? Constants.TEXTURETYPE_UNSIGNED_BYTE;\r\n isCube = !!options.isCube;\r\n samplingMode = options.samplingMode ?? Texture.TRILINEAR_SAMPLINGMODE;\r\n generateDepthBuffer = options.generateDepthBuffer ?? true;\r\n generateStencilBuffer = !!options.generateStencilBuffer;\r\n isMulti = !!options.isMulti;\r\n format = options.format ?? Constants.TEXTUREFORMAT_RGBA;\r\n delayAllocation = !!options.delayAllocation;\r\n samples = options.samples;\r\n creationFlags = options.creationFlags;\r\n noColorAttachment = !!options.noColorAttachment;\r\n useSRGBBuffer = !!options.useSRGBBuffer;\r\n colorAttachment = options.colorAttachment;\r\n }\r\n\r\n super(null, scene, !generateMipMaps, undefined, samplingMode, undefined, undefined, undefined, undefined, format);\r\n\r\n scene = this.getScene();\r\n if (!scene) {\r\n return;\r\n }\r\n\r\n const engine = this.getScene()!.getEngine();\r\n\r\n this._coordinatesMode = Texture.PROJECTION_MODE;\r\n this.renderList = new Array();\r\n this.name = name;\r\n this.isRenderTarget = true;\r\n this._initialSizeParameter = size;\r\n this._renderPassIds = [];\r\n this._isCubeData = isCube;\r\n\r\n this._processSizeParameter(size);\r\n\r\n this.renderPassId = this._renderPassIds[0];\r\n\r\n this._resizeObserver = engine.onResizeObservable.add(() => {});\r\n\r\n this._generateMipMaps = generateMipMaps ? true : false;\r\n this._doNotChangeAspectRatio = doNotChangeAspectRatio;\r\n\r\n // Rendering groups\r\n this._renderingManager = new RenderingManager(scene);\r\n this._renderingManager._useSceneAutoClearSetup = true;\r\n\r\n if (isMulti) {\r\n return;\r\n }\r\n\r\n this._renderTargetOptions = {\r\n generateMipMaps: generateMipMaps,\r\n type: type,\r\n format: this._format ?? undefined,\r\n samplingMode: this.samplingMode,\r\n generateDepthBuffer: generateDepthBuffer,\r\n generateStencilBuffer: generateStencilBuffer,\r\n samples,\r\n creationFlags,\r\n noColorAttachment: noColorAttachment,\r\n useSRGBBuffer,\r\n colorAttachment: colorAttachment,\r\n label: this.name,\r\n };\r\n\r\n if (this.samplingMode === Texture.NEAREST_SAMPLINGMODE) {\r\n this.wrapU = Texture.CLAMP_ADDRESSMODE;\r\n this.wrapV = Texture.CLAMP_ADDRESSMODE;\r\n }\r\n\r\n if (!delayAllocation) {\r\n if (isCube) {\r\n this._renderTarget = scene.getEngine().createRenderTargetCubeTexture(this.getRenderSize(), this._renderTargetOptions);\r\n this.coordinatesMode = Texture.INVCUBIC_MODE;\r\n this._textureMatrix = Matrix.Identity();\r\n } else {\r\n this._renderTarget = scene.getEngine().createRenderTargetTexture(this._size, this._renderTargetOptions);\r\n }\r\n this._texture = this._renderTarget.texture;\r\n if (samples !== undefined) {\r\n this.samples = samples;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Creates a depth stencil texture.\r\n * This is only available in WebGL 2 or with the depth texture extension available.\r\n * @param comparisonFunction Specifies the comparison function to set on the texture. If 0 or undefined, the texture is not in comparison mode (default: 0)\r\n * @param bilinearFiltering Specifies whether or not bilinear filtering is enable on the texture (default: true)\r\n * @param generateStencil Specifies whether or not a stencil should be allocated in the texture (default: false)\r\n * @param samples sample count of the depth/stencil texture (default: 1)\r\n * @param format format of the depth texture (default: Constants.TEXTUREFORMAT_DEPTH32_FLOAT)\r\n */\r\n public createDepthStencilTexture(\r\n comparisonFunction: number = 0,\r\n bilinearFiltering: boolean = true,\r\n generateStencil: boolean = false,\r\n samples: number = 1,\r\n format: number = Constants.TEXTUREFORMAT_DEPTH32_FLOAT\r\n ): void {\r\n this._renderTarget?.createDepthStencilTexture(comparisonFunction, bilinearFiltering, generateStencil, samples, format);\r\n }\r\n\r\n private _releaseRenderPassId(): void {\r\n if (this._scene) {\r\n const engine = this._scene.getEngine();\r\n for (let i = 0; i < this._renderPassIds.length; ++i) {\r\n engine.releaseRenderPassId(this._renderPassIds[i]);\r\n }\r\n }\r\n this._renderPassIds = [];\r\n }\r\n\r\n private _createRenderPassId(): void {\r\n this._releaseRenderPassId();\r\n\r\n const engine = this._scene!.getEngine(); // scene can't be null in a RenderTargetTexture, see constructor\r\n const numPasses = this._isCubeData ? 6 : this.getRenderLayers() || 1;\r\n\r\n for (let i = 0; i < numPasses; ++i) {\r\n this._renderPassIds[i] = engine.createRenderPassId(`RenderTargetTexture - ${this.name}#${i}`);\r\n }\r\n }\r\n\r\n protected _processSizeParameter(size: number | { width: number; height: number } | { ratio: number }, createRenderPassIds = true): void {\r\n if ((<{ ratio: number }>size).ratio) {\r\n this._sizeRatio = (<{ ratio: number }>size).ratio;\r\n const engine = this._getEngine()!;\r\n this._size = {\r\n width: this._bestReflectionRenderTargetDimension(engine.getRenderWidth(), this._sizeRatio),\r\n height: this._bestReflectionRenderTargetDimension(engine.getRenderHeight(), this._sizeRatio),\r\n };\r\n } else {\r\n this._size = size;\r\n }\r\n\r\n if (createRenderPassIds) {\r\n this._createRenderPassId();\r\n }\r\n }\r\n\r\n /**\r\n * Define the number of samples to use in case of MSAA.\r\n * It defaults to one meaning no MSAA has been enabled.\r\n */\r\n public get samples(): number {\r\n return this._renderTarget?.samples ?? this._samples;\r\n }\r\n\r\n public set samples(value: number) {\r\n if (this._renderTarget) {\r\n this._samples = this._renderTarget.setSamples(value);\r\n }\r\n }\r\n\r\n /**\r\n * Resets the refresh counter of the texture and start bak from scratch.\r\n * Could be useful to regenerate the texture if it is setup to render only once.\r\n */\r\n public resetRefreshCounter(): void {\r\n this._currentRefreshId = -1;\r\n }\r\n\r\n /**\r\n * Define the refresh rate of the texture or the rendering frequency.\r\n * Use 0 to render just once, 1 to render on every frame, 2 to render every two frames and so on...\r\n */\r\n public get refreshRate(): number {\r\n return this._refreshRate;\r\n }\r\n public set refreshRate(value: number) {\r\n this._refreshRate = value;\r\n this.resetRefreshCounter();\r\n }\r\n\r\n /**\r\n * Adds a post process to the render target rendering passes.\r\n * @param postProcess define the post process to add\r\n */\r\n public addPostProcess(postProcess: PostProcess): void {\r\n if (!this._postProcessManager) {\r\n const scene = this.getScene();\r\n\r\n if (!scene) {\r\n return;\r\n }\r\n this._postProcessManager = new PostProcessManager(scene);\r\n this._postProcesses = new Array();\r\n }\r\n\r\n this._postProcesses.push(postProcess);\r\n this._postProcesses[0].autoClear = false;\r\n }\r\n\r\n /**\r\n * Clear all the post processes attached to the render target\r\n * @param dispose define if the cleared post processes should also be disposed (false by default)\r\n */\r\n public clearPostProcesses(dispose: boolean = false): void {\r\n if (!this._postProcesses) {\r\n return;\r\n }\r\n\r\n if (dispose) {\r\n for (const postProcess of this._postProcesses) {\r\n postProcess.dispose();\r\n }\r\n }\r\n\r\n this._postProcesses = [];\r\n }\r\n\r\n /**\r\n * Remove one of the post process from the list of attached post processes to the texture\r\n * @param postProcess define the post process to remove from the list\r\n */\r\n public removePostProcess(postProcess: PostProcess): void {\r\n if (!this._postProcesses) {\r\n return;\r\n }\r\n\r\n const index = this._postProcesses.indexOf(postProcess);\r\n\r\n if (index === -1) {\r\n return;\r\n }\r\n\r\n this._postProcesses.splice(index, 1);\r\n\r\n if (this._postProcesses.length > 0) {\r\n this._postProcesses[0].autoClear = false;\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _shouldRender(): boolean {\r\n if (this._currentRefreshId === -1) {\r\n // At least render once\r\n this._currentRefreshId = 1;\r\n return true;\r\n }\r\n\r\n if (this.refreshRate === this._currentRefreshId) {\r\n this._currentRefreshId = 1;\r\n return true;\r\n }\r\n\r\n this._currentRefreshId++;\r\n return false;\r\n }\r\n\r\n /**\r\n * Gets the actual render size of the texture.\r\n * @returns the width of the render size\r\n */\r\n public getRenderSize(): number {\r\n return this.getRenderWidth();\r\n }\r\n\r\n /**\r\n * Gets the actual render width of the texture.\r\n * @returns the width of the render size\r\n */\r\n public getRenderWidth(): number {\r\n if ((<{ width: number; height: number }>this._size).width) {\r\n return (<{ width: number; height: number }>this._size).width;\r\n }\r\n\r\n return this._size;\r\n }\r\n\r\n /**\r\n * Gets the actual render height of the texture.\r\n * @returns the height of the render size\r\n */\r\n public getRenderHeight(): number {\r\n if ((<{ width: number; height: number }>this._size).width) {\r\n return (<{ width: number; height: number }>this._size).height;\r\n }\r\n\r\n return this._size;\r\n }\r\n\r\n /**\r\n * Gets the actual number of layers of the texture.\r\n * @returns the number of layers\r\n */\r\n public getRenderLayers(): number {\r\n const layers = (<{ width: number; height: number; layers?: number }>this._size).layers;\r\n if (layers) {\r\n return layers;\r\n }\r\n\r\n return 0;\r\n }\r\n\r\n /**\r\n * Don't allow this render target texture to rescale. Mainly used to prevent rescaling by the scene optimizer.\r\n */\r\n public disableRescaling() {\r\n this._canRescale = false;\r\n }\r\n\r\n /**\r\n * Get if the texture can be rescaled or not.\r\n */\r\n public get canRescale(): boolean {\r\n return this._canRescale;\r\n }\r\n\r\n /**\r\n * Resize the texture using a ratio.\r\n * @param ratio the ratio to apply to the texture size in order to compute the new target size\r\n */\r\n public scale(ratio: number): void {\r\n const newSize = Math.max(1, this.getRenderSize() * ratio);\r\n\r\n this.resize(newSize);\r\n }\r\n\r\n /**\r\n * Get the texture reflection matrix used to rotate/transform the reflection.\r\n * @returns the reflection matrix\r\n */\r\n public getReflectionTextureMatrix(): Matrix {\r\n if (this.isCube) {\r\n return this._textureMatrix;\r\n }\r\n\r\n return super.getReflectionTextureMatrix();\r\n }\r\n\r\n /**\r\n * Resize the texture to a new desired size.\r\n * Be careful as it will recreate all the data in the new texture.\r\n * @param size Define the new size. It can be:\r\n * - a number for squared texture,\r\n * - an object containing { width: number, height: number }\r\n * - or an object containing a ratio { ratio: number }\r\n */\r\n public resize(size: number | { width: number; height: number } | { ratio: number }): void {\r\n const wasCube = this.isCube;\r\n\r\n this._renderTarget?.dispose();\r\n this._renderTarget = null;\r\n\r\n const scene = this.getScene();\r\n\r\n if (!scene) {\r\n return;\r\n }\r\n\r\n this._processSizeParameter(size, false);\r\n\r\n if (wasCube) {\r\n this._renderTarget = scene.getEngine().createRenderTargetCubeTexture(this.getRenderSize(), this._renderTargetOptions);\r\n } else {\r\n this._renderTarget = scene.getEngine().createRenderTargetTexture(this._size, this._renderTargetOptions);\r\n }\r\n this._texture = this._renderTarget.texture;\r\n\r\n if (this._renderTargetOptions.samples !== undefined) {\r\n this.samples = this._renderTargetOptions.samples;\r\n }\r\n\r\n if (this.onResizeObservable.hasObservers()) {\r\n this.onResizeObservable.notifyObservers(this);\r\n }\r\n }\r\n\r\n private _defaultRenderListPrepared: boolean;\r\n\r\n /**\r\n * Renders all the objects from the render list into the texture.\r\n * @param useCameraPostProcess Define if camera post processes should be used during the rendering\r\n * @param dumpForDebug Define if the rendering result should be dumped (copied) for debugging purpose\r\n */\r\n public render(useCameraPostProcess: boolean = false, dumpForDebug: boolean = false): void {\r\n this._render(useCameraPostProcess, dumpForDebug);\r\n }\r\n\r\n /**\r\n * This function will check if the render target texture can be rendered (textures are loaded, shaders are compiled)\r\n * @returns true if all required resources are ready\r\n */\r\n public isReadyForRendering(): boolean {\r\n return this._render(false, false, true);\r\n }\r\n\r\n private _render(useCameraPostProcess: boolean = false, dumpForDebug: boolean = false, checkReadiness: boolean = false): boolean {\r\n const scene = this.getScene();\r\n\r\n if (!scene) {\r\n return checkReadiness;\r\n }\r\n\r\n const engine = scene.getEngine();\r\n\r\n if (this.useCameraPostProcesses !== undefined) {\r\n useCameraPostProcess = this.useCameraPostProcesses;\r\n }\r\n\r\n if (this._waitingRenderList) {\r\n this.renderList = [];\r\n for (let index = 0; index < this._waitingRenderList.length; index++) {\r\n const id = this._waitingRenderList[index];\r\n const mesh = scene.getMeshById(id);\r\n if (mesh) {\r\n this.renderList.push(mesh);\r\n }\r\n }\r\n\r\n this._waitingRenderList = undefined;\r\n }\r\n\r\n // Is predicate defined?\r\n if (this.renderListPredicate) {\r\n if (this.renderList) {\r\n this.renderList.length = 0; // Clear previous renderList\r\n } else {\r\n this.renderList = [];\r\n }\r\n\r\n const scene = this.getScene();\r\n\r\n if (!scene) {\r\n return checkReadiness;\r\n }\r\n\r\n const sceneMeshes = scene.meshes;\r\n\r\n for (let index = 0; index < sceneMeshes.length; index++) {\r\n const mesh = sceneMeshes[index];\r\n if (this.renderListPredicate(mesh)) {\r\n this.renderList.push(mesh);\r\n }\r\n }\r\n }\r\n\r\n const currentRenderPassId = engine.currentRenderPassId;\r\n\r\n this.onBeforeBindObservable.notifyObservers(this);\r\n\r\n // Set custom projection.\r\n // Needs to be before binding to prevent changing the aspect ratio.\r\n const camera: Nullable = this.activeCamera ?? scene.activeCamera;\r\n const sceneCamera = scene.activeCamera;\r\n\r\n if (camera) {\r\n if (camera !== scene.activeCamera) {\r\n scene.setTransformMatrix(camera.getViewMatrix(), camera.getProjectionMatrix(true));\r\n scene.activeCamera = camera;\r\n }\r\n engine.setViewport(camera.rigParent ? camera.rigParent.viewport : camera.viewport, this.getRenderWidth(), this.getRenderHeight());\r\n }\r\n\r\n this._defaultRenderListPrepared = false;\r\n\r\n let returnValue = checkReadiness;\r\n\r\n if (!checkReadiness) {\r\n if (this.is2DArray && !this.isMulti) {\r\n for (let layer = 0; layer < this.getRenderLayers(); layer++) {\r\n this._renderToTarget(0, useCameraPostProcess, dumpForDebug, layer, camera);\r\n scene.incrementRenderId();\r\n scene.resetCachedMaterial();\r\n }\r\n } else if (this.isCube && !this.isMulti) {\r\n for (let face = 0; face < 6; face++) {\r\n this._renderToTarget(face, useCameraPostProcess, dumpForDebug, undefined, camera);\r\n scene.incrementRenderId();\r\n scene.resetCachedMaterial();\r\n }\r\n } else {\r\n this._renderToTarget(0, useCameraPostProcess, dumpForDebug, undefined, camera);\r\n }\r\n } else {\r\n if (!scene.getViewMatrix()) {\r\n // We probably didn't execute scene.render() yet, so make sure we have a view/projection matrix setup for the scene\r\n scene.updateTransformMatrix();\r\n }\r\n const numLayers = this.is2DArray ? this.getRenderLayers() : this.isCube ? 6 : 1;\r\n for (let layer = 0; layer < numLayers && returnValue; layer++) {\r\n let currentRenderList: Nullable> = null;\r\n const defaultRenderList = this.renderList ? this.renderList : scene.getActiveMeshes().data;\r\n const defaultRenderListLength = this.renderList ? this.renderList.length : scene.getActiveMeshes().length;\r\n\r\n engine.currentRenderPassId = this._renderPassIds[layer];\r\n\r\n this.onBeforeRenderObservable.notifyObservers(layer);\r\n\r\n if (this.getCustomRenderList) {\r\n currentRenderList = this.getCustomRenderList(layer, defaultRenderList, defaultRenderListLength);\r\n }\r\n\r\n if (!currentRenderList) {\r\n currentRenderList = defaultRenderList;\r\n }\r\n\r\n if (!this._doNotChangeAspectRatio) {\r\n scene.updateTransformMatrix(true);\r\n }\r\n\r\n for (let i = 0; i < currentRenderList.length && returnValue; ++i) {\r\n const mesh = currentRenderList[i];\r\n\r\n if (!mesh.isEnabled() || mesh.isBlocked || !mesh.isVisible || !mesh.subMeshes) {\r\n continue;\r\n }\r\n\r\n if (this.customIsReadyFunction) {\r\n if (!this.customIsReadyFunction(mesh, this.refreshRate, checkReadiness)) {\r\n returnValue = false;\r\n continue;\r\n }\r\n } else if (!mesh.isReady(true)) {\r\n returnValue = false;\r\n continue;\r\n }\r\n }\r\n\r\n this.onAfterRenderObservable.notifyObservers(layer);\r\n\r\n if (this.is2DArray || this.isCube) {\r\n scene.incrementRenderId();\r\n scene.resetCachedMaterial();\r\n }\r\n }\r\n }\r\n\r\n this.onAfterUnbindObservable.notifyObservers(this);\r\n\r\n engine.currentRenderPassId = currentRenderPassId;\r\n\r\n if (sceneCamera) {\r\n scene.activeCamera = sceneCamera;\r\n // Do not avoid setting uniforms when multiple scenes are active as another camera may have overwrite these\r\n if (scene.getEngine().scenes.length > 1 || (this.activeCamera && this.activeCamera !== scene.activeCamera)) {\r\n scene.setTransformMatrix(scene.activeCamera.getViewMatrix(), scene.activeCamera.getProjectionMatrix(true));\r\n }\r\n engine.setViewport(scene.activeCamera.viewport);\r\n }\r\n\r\n scene.resetCachedMaterial();\r\n\r\n return returnValue;\r\n }\r\n\r\n private _bestReflectionRenderTargetDimension(renderDimension: number, scale: number): number {\r\n const minimum = 128;\r\n const x = renderDimension * scale;\r\n const curved = Engine.NearestPOT(x + (minimum * minimum) / (minimum + x));\r\n\r\n // Ensure we don't exceed the render dimension (while staying POT)\r\n return Math.min(Engine.FloorPOT(renderDimension), curved);\r\n }\r\n\r\n private _prepareRenderingManager(currentRenderList: Array, currentRenderListLength: number, camera: Nullable, checkLayerMask: boolean): void {\r\n const scene = this.getScene();\r\n\r\n if (!scene) {\r\n return;\r\n }\r\n\r\n this._renderingManager.reset();\r\n\r\n const sceneRenderId = scene.getRenderId();\r\n for (let meshIndex = 0; meshIndex < currentRenderListLength; meshIndex++) {\r\n const mesh = currentRenderList[meshIndex];\r\n\r\n if (mesh && !mesh.isBlocked) {\r\n if (this.customIsReadyFunction) {\r\n if (!this.customIsReadyFunction(mesh, this.refreshRate, false)) {\r\n this.resetRefreshCounter();\r\n continue;\r\n }\r\n } else if (!mesh.isReady(this.refreshRate === 0)) {\r\n this.resetRefreshCounter();\r\n continue;\r\n }\r\n\r\n if (!mesh._internalAbstractMeshDataInfo._currentLODIsUpToDate && scene.activeCamera) {\r\n mesh._internalAbstractMeshDataInfo._currentLOD = scene.customLODSelector\r\n ? scene.customLODSelector(mesh, this.activeCamera || scene.activeCamera)\r\n : mesh.getLOD(this.activeCamera || scene.activeCamera);\r\n mesh._internalAbstractMeshDataInfo._currentLODIsUpToDate = true;\r\n }\r\n if (!mesh._internalAbstractMeshDataInfo._currentLOD) {\r\n continue;\r\n }\r\n\r\n let meshToRender = mesh._internalAbstractMeshDataInfo._currentLOD;\r\n\r\n meshToRender._preActivateForIntermediateRendering(sceneRenderId);\r\n\r\n let isMasked;\r\n if (checkLayerMask && camera) {\r\n isMasked = (mesh.layerMask & camera.layerMask) === 0;\r\n } else {\r\n isMasked = false;\r\n }\r\n\r\n if (mesh.isEnabled() && mesh.isVisible && mesh.subMeshes && !isMasked) {\r\n if (meshToRender !== mesh) {\r\n meshToRender._activate(sceneRenderId, true);\r\n }\r\n if (mesh._activate(sceneRenderId, true) && mesh.subMeshes.length) {\r\n if (!mesh.isAnInstance) {\r\n meshToRender._internalAbstractMeshDataInfo._onlyForInstancesIntermediate = false;\r\n } else {\r\n if (mesh._internalAbstractMeshDataInfo._actAsRegularMesh) {\r\n meshToRender = mesh;\r\n }\r\n }\r\n meshToRender._internalAbstractMeshDataInfo._isActiveIntermediate = true;\r\n\r\n for (let subIndex = 0; subIndex < meshToRender.subMeshes.length; subIndex++) {\r\n const subMesh = meshToRender.subMeshes[subIndex];\r\n this._renderingManager.dispatch(subMesh, meshToRender);\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n for (let particleIndex = 0; particleIndex < scene.particleSystems.length; particleIndex++) {\r\n const particleSystem = scene.particleSystems[particleIndex];\r\n\r\n const emitter: any = particleSystem.emitter;\r\n\r\n if (!particleSystem.isStarted() || !emitter || (emitter.position && !emitter.isEnabled())) {\r\n continue;\r\n }\r\n\r\n this._renderingManager.dispatchParticles(particleSystem);\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n * @param faceIndex face index to bind to if this is a cubetexture\r\n * @param layer defines the index of the texture to bind in the array\r\n */\r\n public _bindFrameBuffer(faceIndex: number = 0, layer = 0) {\r\n const scene = this.getScene();\r\n if (!scene) {\r\n return;\r\n }\r\n\r\n const engine = scene.getEngine();\r\n if (this._renderTarget) {\r\n engine.bindFramebuffer(this._renderTarget, this.isCube ? faceIndex : undefined, undefined, undefined, this.ignoreCameraViewport, 0, layer);\r\n }\r\n }\r\n\r\n protected _unbindFrameBuffer(engine: Engine, faceIndex: number): void {\r\n if (!this._renderTarget) {\r\n return;\r\n }\r\n engine.unBindFramebuffer(this._renderTarget, this.isCube, () => {\r\n this.onAfterRenderObservable.notifyObservers(faceIndex);\r\n });\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _prepareFrame(scene: Scene, faceIndex?: number, layer?: number, useCameraPostProcess?: boolean) {\r\n if (this._postProcessManager) {\r\n if (!this._prePassEnabled) {\r\n this._postProcessManager._prepareFrame(this._texture, this._postProcesses);\r\n }\r\n } else if (!useCameraPostProcess || !scene.postProcessManager._prepareFrame(this._texture)) {\r\n this._bindFrameBuffer(faceIndex, layer);\r\n }\r\n }\r\n\r\n private _renderToTarget(faceIndex: number, useCameraPostProcess: boolean, dumpForDebug: boolean, layer = 0, camera: Nullable = null): void {\r\n const scene = this.getScene();\r\n\r\n if (!scene) {\r\n return;\r\n }\r\n\r\n const engine = scene.getEngine();\r\n\r\n engine._debugPushGroup?.(`render to face #${faceIndex} layer #${layer}`, 1);\r\n\r\n // Bind\r\n this._prepareFrame(scene, faceIndex, layer, useCameraPostProcess);\r\n\r\n if (this.is2DArray) {\r\n engine.currentRenderPassId = this._renderPassIds[layer];\r\n this.onBeforeRenderObservable.notifyObservers(layer);\r\n } else {\r\n engine.currentRenderPassId = this._renderPassIds[faceIndex];\r\n this.onBeforeRenderObservable.notifyObservers(faceIndex);\r\n }\r\n\r\n const fastPath = engine.snapshotRendering && engine.snapshotRenderingMode === Constants.SNAPSHOTRENDERING_FAST;\r\n\r\n if (!fastPath) {\r\n // Get the list of meshes to render\r\n let currentRenderList: Nullable> = null;\r\n const defaultRenderList = this.renderList ? this.renderList : scene.getActiveMeshes().data;\r\n const defaultRenderListLength = this.renderList ? this.renderList.length : scene.getActiveMeshes().length;\r\n\r\n if (this.getCustomRenderList) {\r\n currentRenderList = this.getCustomRenderList(this.is2DArray ? layer : faceIndex, defaultRenderList, defaultRenderListLength);\r\n }\r\n\r\n if (!currentRenderList) {\r\n // No custom render list provided, we prepare the rendering for the default list, but check\r\n // first if we did not already performed the preparation before so as to avoid re-doing it several times\r\n if (!this._defaultRenderListPrepared) {\r\n this._prepareRenderingManager(defaultRenderList, defaultRenderListLength, camera, !this.renderList || this.forceLayerMaskCheck);\r\n this._defaultRenderListPrepared = true;\r\n }\r\n currentRenderList = defaultRenderList;\r\n } else {\r\n // Prepare the rendering for the custom render list provided\r\n this._prepareRenderingManager(currentRenderList, currentRenderList.length, camera, this.forceLayerMaskCheck);\r\n }\r\n\r\n // Before clear\r\n for (const step of scene._beforeRenderTargetClearStage) {\r\n step.action(this, faceIndex, layer);\r\n }\r\n\r\n // Clear\r\n if (this.onClearObservable.hasObservers()) {\r\n this.onClearObservable.notifyObservers(engine);\r\n } else {\r\n if (!this.skipInitialClear) {\r\n engine.clear(this.clearColor || scene.clearColor, true, true, true);\r\n }\r\n }\r\n\r\n if (!this._doNotChangeAspectRatio) {\r\n scene.updateTransformMatrix(true);\r\n }\r\n\r\n // Before Camera Draw\r\n for (const step of scene._beforeRenderTargetDrawStage) {\r\n step.action(this, faceIndex, layer);\r\n }\r\n\r\n // Render\r\n this._renderingManager.render(this.customRenderFunction, currentRenderList, this.renderParticles, this.renderSprites);\r\n\r\n // After Camera Draw\r\n for (const step of scene._afterRenderTargetDrawStage) {\r\n step.action(this, faceIndex, layer);\r\n }\r\n\r\n const saveGenerateMipMaps = this._texture?.generateMipMaps ?? false;\r\n\r\n if (this._texture) {\r\n this._texture.generateMipMaps = false; // if left true, the mipmaps will be generated (if this._texture.generateMipMaps = true) when the first post process binds its own RTT: by doing so it will unbind the current RTT,\r\n // which will trigger a mipmap generation. We don't want this because it's a wasted work, we will do an unbind of the current RTT at the end of the process (see unbindFrameBuffer) which will\r\n // trigger the generation of the final mipmaps\r\n }\r\n\r\n if (this._postProcessManager) {\r\n this._postProcessManager._finalizeFrame(false, this._renderTarget ?? undefined, faceIndex, this._postProcesses, this.ignoreCameraViewport);\r\n } else if (useCameraPostProcess) {\r\n scene.postProcessManager._finalizeFrame(false, this._renderTarget ?? undefined, faceIndex);\r\n }\r\n\r\n for (const step of scene._afterRenderTargetPostProcessStage) {\r\n step.action(this, faceIndex, layer);\r\n }\r\n\r\n if (this._texture) {\r\n this._texture.generateMipMaps = saveGenerateMipMaps;\r\n }\r\n\r\n if (!this._doNotChangeAspectRatio) {\r\n scene.updateTransformMatrix(true);\r\n }\r\n\r\n // Dump ?\r\n if (dumpForDebug) {\r\n DumpTools.DumpFramebuffer(this.getRenderWidth(), this.getRenderHeight(), engine);\r\n }\r\n } else {\r\n // Clear\r\n if (this.onClearObservable.hasObservers()) {\r\n this.onClearObservable.notifyObservers(engine);\r\n } else {\r\n if (!this.skipInitialClear) {\r\n engine.clear(this.clearColor || scene.clearColor, true, true, true);\r\n }\r\n }\r\n }\r\n\r\n // Unbind\r\n this._unbindFrameBuffer(engine, faceIndex);\r\n\r\n if (this._texture && this.isCube && faceIndex === 5) {\r\n engine.generateMipMapsForCubemap(this._texture);\r\n }\r\n\r\n engine._debugPopGroup?.(1);\r\n }\r\n\r\n /**\r\n * Overrides the default sort function applied in the rendering group to prepare the meshes.\r\n * This allowed control for front to back rendering or reversely depending of the special needs.\r\n *\r\n * @param renderingGroupId The rendering group id corresponding to its index\r\n * @param opaqueSortCompareFn The opaque queue comparison function use to sort.\r\n * @param alphaTestSortCompareFn The alpha test queue comparison function use to sort.\r\n * @param transparentSortCompareFn The transparent queue comparison function use to sort.\r\n */\r\n public setRenderingOrder(\r\n renderingGroupId: number,\r\n opaqueSortCompareFn: Nullable<(a: SubMesh, b: SubMesh) => number> = null,\r\n alphaTestSortCompareFn: Nullable<(a: SubMesh, b: SubMesh) => number> = null,\r\n transparentSortCompareFn: Nullable<(a: SubMesh, b: SubMesh) => number> = null\r\n ): void {\r\n this._renderingManager.setRenderingOrder(renderingGroupId, opaqueSortCompareFn, alphaTestSortCompareFn, transparentSortCompareFn);\r\n }\r\n\r\n /**\r\n * Specifies whether or not the stencil and depth buffer are cleared between two rendering groups.\r\n *\r\n * @param renderingGroupId The rendering group id corresponding to its index\r\n * @param autoClearDepthStencil Automatically clears depth and stencil between groups if true.\r\n */\r\n public setRenderingAutoClearDepthStencil(renderingGroupId: number, autoClearDepthStencil: boolean): void {\r\n this._renderingManager.setRenderingAutoClearDepthStencil(renderingGroupId, autoClearDepthStencil);\r\n this._renderingManager._useSceneAutoClearSetup = false;\r\n }\r\n\r\n /**\r\n * Clones the texture.\r\n * @returns the cloned texture\r\n */\r\n public clone(): RenderTargetTexture {\r\n const textureSize = this.getSize();\r\n const newTexture = new RenderTargetTexture(\r\n this.name,\r\n textureSize,\r\n this.getScene(),\r\n this._renderTargetOptions.generateMipMaps,\r\n this._doNotChangeAspectRatio,\r\n this._renderTargetOptions.type,\r\n this.isCube,\r\n this._renderTargetOptions.samplingMode,\r\n this._renderTargetOptions.generateDepthBuffer,\r\n this._renderTargetOptions.generateStencilBuffer,\r\n undefined,\r\n this._renderTargetOptions.format,\r\n undefined,\r\n this._renderTargetOptions.samples\r\n );\r\n\r\n // Base texture\r\n newTexture.hasAlpha = this.hasAlpha;\r\n newTexture.level = this.level;\r\n\r\n // RenderTarget Texture\r\n newTexture.coordinatesMode = this.coordinatesMode;\r\n if (this.renderList) {\r\n newTexture.renderList = this.renderList.slice(0);\r\n }\r\n\r\n return newTexture;\r\n }\r\n\r\n /**\r\n * Serialize the texture to a JSON representation we can easily use in the respective Parse function.\r\n * @returns The JSON representation of the texture\r\n */\r\n public serialize(): any {\r\n if (!this.name) {\r\n return null;\r\n }\r\n\r\n const serializationObject = super.serialize();\r\n\r\n serializationObject.renderTargetSize = this.getRenderSize();\r\n serializationObject.renderList = [];\r\n\r\n if (this.renderList) {\r\n for (let index = 0; index < this.renderList.length; index++) {\r\n serializationObject.renderList.push(this.renderList[index].id);\r\n }\r\n }\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * This will remove the attached framebuffer objects. The texture will not be able to be used as render target anymore\r\n */\r\n public disposeFramebufferObjects(): void {\r\n this._renderTarget?.dispose(true);\r\n }\r\n\r\n /**\r\n * Release and destroy the underlying lower level texture aka internalTexture.\r\n */\r\n public releaseInternalTexture(): void {\r\n this._renderTarget?.releaseTextures();\r\n this._texture = null;\r\n }\r\n\r\n /**\r\n * Dispose the texture and release its associated resources.\r\n */\r\n public dispose(): void {\r\n this.onResizeObservable.clear();\r\n this.onClearObservable.clear();\r\n this.onAfterRenderObservable.clear();\r\n this.onAfterUnbindObservable.clear();\r\n this.onBeforeBindObservable.clear();\r\n this.onBeforeRenderObservable.clear();\r\n\r\n if (this._postProcessManager) {\r\n this._postProcessManager.dispose();\r\n this._postProcessManager = null;\r\n }\r\n\r\n if (this._prePassRenderTarget) {\r\n this._prePassRenderTarget.dispose();\r\n }\r\n\r\n this._releaseRenderPassId();\r\n this.clearPostProcesses(true);\r\n\r\n if (this._resizeObserver) {\r\n this.getScene()!.getEngine().onResizeObservable.remove(this._resizeObserver);\r\n this._resizeObserver = null;\r\n }\r\n\r\n this.renderList = null;\r\n\r\n // Remove from custom render targets\r\n const scene = this.getScene();\r\n\r\n if (!scene) {\r\n return;\r\n }\r\n\r\n let index = scene.customRenderTargets.indexOf(this);\r\n\r\n if (index >= 0) {\r\n scene.customRenderTargets.splice(index, 1);\r\n }\r\n\r\n for (const camera of scene.cameras) {\r\n index = camera.customRenderTargets.indexOf(this);\r\n\r\n if (index >= 0) {\r\n camera.customRenderTargets.splice(index, 1);\r\n }\r\n }\r\n\r\n this._renderTarget?.dispose();\r\n this._renderTarget = null;\r\n this._texture = null;\r\n\r\n super.dispose();\r\n }\r\n\r\n /** @internal */\r\n public _rebuild(): void {\r\n if (this.refreshRate === RenderTargetTexture.REFRESHRATE_RENDER_ONCE) {\r\n this.refreshRate = RenderTargetTexture.REFRESHRATE_RENDER_ONCE;\r\n }\r\n\r\n if (this._postProcessManager) {\r\n this._postProcessManager._rebuild();\r\n }\r\n }\r\n\r\n /**\r\n * Clear the info related to rendering groups preventing retention point in material dispose.\r\n */\r\n public freeRenderingGroups(): void {\r\n if (this._renderingManager) {\r\n this._renderingManager.freeRenderingGroups();\r\n }\r\n }\r\n\r\n /**\r\n * Gets the number of views the corresponding to the texture (eg. a MultiviewRenderTarget will have > 1)\r\n * @returns the view count\r\n */\r\n public getViewCount() {\r\n return 1;\r\n }\r\n}\r\n\r\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\r\nTexture._CreateRenderTargetTexture = (name: string, renderTargetSize: number, scene: Scene, generateMipMaps: boolean, creationFlags?: number) => {\r\n return new RenderTargetTexture(name, renderTargetSize, scene, generateMipMaps);\r\n};\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\n\nconst name = \"passCubePixelShader\";\nconst shader = `varying vec2 vUV;uniform samplerCube textureSampler;\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) \n{vec2 uv=vUV*2.0-1.0;\n#ifdef POSITIVEX\ngl_FragColor=textureCube(textureSampler,vec3(1.001,uv.y,uv.x));\n#endif\n#ifdef NEGATIVEX\ngl_FragColor=textureCube(textureSampler,vec3(-1.001,uv.y,uv.x));\n#endif\n#ifdef POSITIVEY\ngl_FragColor=textureCube(textureSampler,vec3(uv.y,1.001,uv.x));\n#endif\n#ifdef NEGATIVEY\ngl_FragColor=textureCube(textureSampler,vec3(uv.y,-1.001,uv.x));\n#endif\n#ifdef POSITIVEZ\ngl_FragColor=textureCube(textureSampler,vec3(uv,1.001));\n#endif\n#ifdef NEGATIVEZ\ngl_FragColor=textureCube(textureSampler,vec3(uv,-1.001));\n#endif\n}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const passCubePixelShader = { name, shader };\n","import type { Nullable } from \"../types\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\nimport type { PostProcessOptions } from \"./postProcess\";\r\nimport { PostProcess } from \"./postProcess\";\r\nimport { Engine } from \"../Engines/engine\";\r\n\r\nimport \"../Shaders/pass.fragment\";\r\nimport \"../Shaders/passCube.fragment\";\r\nimport { RegisterClass } from \"../Misc/typeStore\";\r\nimport { SerializationHelper } from \"../Misc/decorators\";\r\n\r\nimport type { Scene } from \"../scene\";\r\n\r\n/**\r\n * PassPostProcess which produces an output the same as it's input\r\n */\r\nexport class PassPostProcess extends PostProcess {\r\n /**\r\n * Gets a string identifying the name of the class\r\n * @returns \"PassPostProcess\" string\r\n */\r\n public getClassName(): string {\r\n return \"PassPostProcess\";\r\n }\r\n\r\n /**\r\n * Creates the PassPostProcess\r\n * @param name The name of the effect.\r\n * @param options The required width/height ratio to downsize to before computing the render pass.\r\n * @param camera The camera to apply the render pass to.\r\n * @param samplingMode The sampling mode to be used when computing the pass. (default: 0)\r\n * @param engine The engine which the post process will be applied. (default: current engine)\r\n * @param reusable If the post process can be reused on the same frame. (default: false)\r\n * @param textureType The type of texture to be used when performing the post processing.\r\n * @param blockCompilation If compilation of the shader should not be done in the constructor. The updateEffect method can be used to compile the shader at a later time. (default: false)\r\n */\r\n constructor(\r\n name: string,\r\n options: number | PostProcessOptions,\r\n camera: Nullable = null,\r\n samplingMode?: number,\r\n engine?: Engine,\r\n reusable?: boolean,\r\n textureType: number = Constants.TEXTURETYPE_UNSIGNED_INT,\r\n blockCompilation = false\r\n ) {\r\n super(name, \"pass\", null, null, options, camera, samplingMode, engine, reusable, undefined, textureType, undefined, null, blockCompilation);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _Parse(parsedPostProcess: any, targetCamera: Camera, scene: Scene, rootUrl: string) {\r\n return SerializationHelper.Parse(\r\n () => {\r\n return new PassPostProcess(\r\n parsedPostProcess.name,\r\n parsedPostProcess.options,\r\n targetCamera,\r\n parsedPostProcess.renderTargetSamplingMode,\r\n parsedPostProcess._engine,\r\n parsedPostProcess.reusable\r\n );\r\n },\r\n parsedPostProcess,\r\n scene,\r\n rootUrl\r\n );\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.PassPostProcess\", PassPostProcess);\r\n\r\n/**\r\n * PassCubePostProcess which produces an output the same as it's input (which must be a cube texture)\r\n */\r\nexport class PassCubePostProcess extends PostProcess {\r\n private _face = 0;\r\n\r\n /**\r\n * Gets or sets the cube face to display.\r\n * * 0 is +X\r\n * * 1 is -X\r\n * * 2 is +Y\r\n * * 3 is -Y\r\n * * 4 is +Z\r\n * * 5 is -Z\r\n */\r\n public get face(): number {\r\n return this._face;\r\n }\r\n\r\n public set face(value: number) {\r\n if (value < 0 || value > 5) {\r\n return;\r\n }\r\n\r\n this._face = value;\r\n switch (this._face) {\r\n case 0:\r\n this.updateEffect(\"#define POSITIVEX\");\r\n break;\r\n case 1:\r\n this.updateEffect(\"#define NEGATIVEX\");\r\n break;\r\n case 2:\r\n this.updateEffect(\"#define POSITIVEY\");\r\n break;\r\n case 3:\r\n this.updateEffect(\"#define NEGATIVEY\");\r\n break;\r\n case 4:\r\n this.updateEffect(\"#define POSITIVEZ\");\r\n break;\r\n case 5:\r\n this.updateEffect(\"#define NEGATIVEZ\");\r\n break;\r\n }\r\n }\r\n\r\n /**\r\n * Gets a string identifying the name of the class\r\n * @returns \"PassCubePostProcess\" string\r\n */\r\n public getClassName(): string {\r\n return \"PassCubePostProcess\";\r\n }\r\n\r\n /**\r\n * Creates the PassCubePostProcess\r\n * @param name The name of the effect.\r\n * @param options The required width/height ratio to downsize to before computing the render pass.\r\n * @param camera The camera to apply the render pass to.\r\n * @param samplingMode The sampling mode to be used when computing the pass. (default: 0)\r\n * @param engine The engine which the post process will be applied. (default: current engine)\r\n * @param reusable If the post process can be reused on the same frame. (default: false)\r\n * @param textureType The type of texture to be used when performing the post processing.\r\n * @param blockCompilation If compilation of the shader should not be done in the constructor. The updateEffect method can be used to compile the shader at a later time. (default: false)\r\n */\r\n constructor(\r\n name: string,\r\n options: number | PostProcessOptions,\r\n camera: Nullable = null,\r\n samplingMode?: number,\r\n engine?: Engine,\r\n reusable?: boolean,\r\n textureType: number = Constants.TEXTURETYPE_UNSIGNED_INT,\r\n blockCompilation = false\r\n ) {\r\n super(name, \"passCube\", null, null, options, camera, samplingMode, engine, reusable, \"#define POSITIVEX\", textureType, undefined, null, blockCompilation);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _Parse(parsedPostProcess: any, targetCamera: Camera, scene: Scene, rootUrl: string) {\r\n return SerializationHelper.Parse(\r\n () => {\r\n return new PassCubePostProcess(\r\n parsedPostProcess.name,\r\n parsedPostProcess.options,\r\n targetCamera,\r\n parsedPostProcess.renderTargetSamplingMode,\r\n parsedPostProcess._engine,\r\n parsedPostProcess.reusable\r\n );\r\n },\r\n parsedPostProcess,\r\n scene,\r\n rootUrl\r\n );\r\n }\r\n}\r\n\r\nEngine._RescalePostProcessFactory = (engine: Engine) => {\r\n return new PassPostProcess(\"rescale\", 1, null, Constants.TEXTURE_BILINEAR_SAMPLINGMODE, engine, false, Constants.TEXTURETYPE_UNSIGNED_INT);\r\n};\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport type { InternalTexture } from \"../Materials/Textures/internalTexture\";\r\nimport { Texture } from \"../Materials/Textures/texture\";\r\nimport { RenderTargetTexture } from \"../Materials/Textures/renderTargetTexture\";\r\nimport { PassPostProcess } from \"../PostProcesses/passPostProcess\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport type { Scene } from \"../scene\";\r\nimport { PostProcess } from \"../PostProcesses/postProcess\";\r\nimport type { Engine } from \"../Engines/engine\";\r\n\r\n/**\r\n * Uses the GPU to create a copy texture rescaled at a given size\r\n * @param texture Texture to copy from\r\n * @param width defines the desired width\r\n * @param height defines the desired height\r\n * @param useBilinearMode defines if bilinear mode has to be used\r\n * @returns the generated texture\r\n */\r\nexport function CreateResizedCopy(texture: Texture, width: number, height: number, useBilinearMode: boolean = true): Texture {\r\n const scene = texture.getScene();\r\n const engine = scene.getEngine();\r\n\r\n const rtt = new RenderTargetTexture(\r\n \"resized\" + texture.name,\r\n { width: width, height: height },\r\n scene,\r\n !texture.noMipmap,\r\n true,\r\n (texture._texture).type,\r\n false,\r\n texture.samplingMode,\r\n false\r\n );\r\n\r\n rtt.wrapU = texture.wrapU;\r\n rtt.wrapV = texture.wrapV;\r\n rtt.uOffset = texture.uOffset;\r\n rtt.vOffset = texture.vOffset;\r\n rtt.uScale = texture.uScale;\r\n rtt.vScale = texture.vScale;\r\n rtt.uAng = texture.uAng;\r\n rtt.vAng = texture.vAng;\r\n rtt.wAng = texture.wAng;\r\n rtt.coordinatesIndex = texture.coordinatesIndex;\r\n rtt.level = texture.level;\r\n rtt.anisotropicFilteringLevel = texture.anisotropicFilteringLevel;\r\n (rtt._texture).isReady = false;\r\n\r\n texture.wrapU = Texture.CLAMP_ADDRESSMODE;\r\n texture.wrapV = Texture.CLAMP_ADDRESSMODE;\r\n\r\n const passPostProcess = new PassPostProcess(\r\n \"pass\",\r\n 1,\r\n null,\r\n useBilinearMode ? Texture.BILINEAR_SAMPLINGMODE : Texture.NEAREST_SAMPLINGMODE,\r\n engine,\r\n false,\r\n Constants.TEXTURETYPE_UNSIGNED_INT\r\n );\r\n passPostProcess.externalTextureSamplerBinding = true;\r\n passPostProcess.getEffect().executeWhenCompiled(() => {\r\n passPostProcess.onApply = function (effect) {\r\n effect.setTexture(\"textureSampler\", texture);\r\n };\r\n\r\n const internalTexture = rtt.renderTarget;\r\n\r\n if (internalTexture) {\r\n scene.postProcessManager.directRender([passPostProcess], internalTexture);\r\n\r\n engine.unBindFramebuffer(internalTexture);\r\n rtt.disposeFramebufferObjects();\r\n passPostProcess.dispose();\r\n\r\n rtt.getInternalTexture()!.isReady = true;\r\n }\r\n });\r\n\r\n return rtt;\r\n}\r\n\r\n/**\r\n * Apply a post process to a texture\r\n * @param postProcessName name of the fragment post process\r\n * @param internalTexture the texture to encode\r\n * @param scene the scene hosting the texture\r\n * @param type type of the output texture. If not provided, use the one from internalTexture\r\n * @param samplingMode sampling mode to use to sample the source texture. If not provided, use the one from internalTexture\r\n * @param format format of the output texture. If not provided, use the one from internalTexture\r\n * @returns a promise with the internalTexture having its texture replaced by the result of the processing\r\n */\r\nexport function ApplyPostProcess(\r\n postProcessName: string,\r\n internalTexture: InternalTexture,\r\n scene: Scene,\r\n type?: number,\r\n samplingMode?: number,\r\n format?: number,\r\n width?: number,\r\n height?: number\r\n): Promise {\r\n // Gets everything ready.\r\n const engine = internalTexture.getEngine() as Engine;\r\n\r\n internalTexture.isReady = false;\r\n\r\n samplingMode = samplingMode ?? internalTexture.samplingMode;\r\n type = type ?? internalTexture.type;\r\n format = format ?? internalTexture.format;\r\n width = width ?? internalTexture.width;\r\n height = height ?? internalTexture.height;\r\n\r\n if (type === -1) {\r\n type = Constants.TEXTURETYPE_UNSIGNED_BYTE;\r\n }\r\n\r\n return new Promise((resolve) => {\r\n // Create the post process\r\n const postProcess = new PostProcess(\"postprocess\", postProcessName, null, null, 1, null, samplingMode, engine, false, undefined, type, undefined, null, false, format);\r\n postProcess.externalTextureSamplerBinding = true;\r\n\r\n // Hold the output of the decoding.\r\n const encodedTexture = engine.createRenderTargetTexture(\r\n { width: width as number, height: height as number },\r\n {\r\n generateDepthBuffer: false,\r\n generateMipMaps: false,\r\n generateStencilBuffer: false,\r\n samplingMode,\r\n type,\r\n format,\r\n }\r\n );\r\n\r\n postProcess.getEffect().executeWhenCompiled(() => {\r\n // PP Render Pass\r\n postProcess.onApply = (effect) => {\r\n effect._bindTexture(\"textureSampler\", internalTexture);\r\n effect.setFloat2(\"scale\", 1, 1);\r\n };\r\n scene.postProcessManager.directRender([postProcess!], encodedTexture, true);\r\n\r\n // Cleanup\r\n engine.restoreDefaultFramebuffer();\r\n engine._releaseTexture(internalTexture);\r\n if (postProcess) {\r\n postProcess.dispose();\r\n }\r\n\r\n // Internal Swap\r\n encodedTexture._swapAndDie(internalTexture);\r\n\r\n // Ready to get rolling again.\r\n internalTexture.type = type!;\r\n internalTexture.format = Constants.TEXTUREFORMAT_RGBA;\r\n internalTexture.isReady = true;\r\n\r\n resolve(internalTexture);\r\n });\r\n });\r\n}\r\n\r\n// ref: http://stackoverflow.com/questions/32633585/how-do-you-convert-to-half-floats-in-javascript\r\nlet floatView: Float32Array;\r\nlet int32View: Int32Array;\r\n/**\r\n * Converts a number to half float\r\n * @param value number to convert\r\n * @returns converted number\r\n */\r\nexport function ToHalfFloat(value: number): number {\r\n if (!floatView) {\r\n floatView = new Float32Array(1);\r\n int32View = new Int32Array(floatView.buffer);\r\n }\r\n\r\n floatView[0] = value;\r\n const x = int32View[0];\r\n\r\n let bits = (x >> 16) & 0x8000; /* Get the sign */\r\n let m = (x >> 12) & 0x07ff; /* Keep one extra bit for rounding */\r\n const e = (x >> 23) & 0xff; /* Using int is faster here */\r\n\r\n /* If zero, or denormal, or exponent underflows too much for a denormal\r\n * half, return signed zero. */\r\n if (e < 103) {\r\n return bits;\r\n }\r\n\r\n /* If NaN, return NaN. If Inf or exponent overflow, return Inf. */\r\n if (e > 142) {\r\n bits |= 0x7c00;\r\n /* If exponent was 0xff and one mantissa bit was set, it means NaN,\r\n * not Inf, so make sure we set one mantissa bit too. */\r\n bits |= (e == 255 ? 0 : 1) && x & 0x007fffff;\r\n return bits;\r\n }\r\n\r\n /* If exponent underflows but not too much, return a denormal */\r\n if (e < 113) {\r\n m |= 0x0800;\r\n /* Extra rounding may overflow and set mantissa to 0 and exponent\r\n * to 1, which is OK. */\r\n bits |= (m >> (114 - e)) + ((m >> (113 - e)) & 1);\r\n return bits;\r\n }\r\n\r\n bits |= ((e - 112) << 10) | (m >> 1);\r\n bits += m & 1;\r\n return bits;\r\n}\r\n\r\n/**\r\n * Converts a half float to a number\r\n * @param value half float to convert\r\n * @returns converted half float\r\n */\r\nexport function FromHalfFloat(value: number): number {\r\n const s = (value & 0x8000) >> 15;\r\n const e = (value & 0x7c00) >> 10;\r\n const f = value & 0x03ff;\r\n\r\n if (e === 0) {\r\n return (s ? -1 : 1) * Math.pow(2, -14) * (f / Math.pow(2, 10));\r\n } else if (e == 0x1f) {\r\n return f ? NaN : (s ? -1 : 1) * Infinity;\r\n }\r\n\r\n return (s ? -1 : 1) * Math.pow(2, e - 15) * (1 + f / Math.pow(2, 10));\r\n}\r\n\r\n/**\r\n * Class used to host texture specific utilities\r\n */\r\nexport const TextureTools = {\r\n /**\r\n * Uses the GPU to create a copy texture rescaled at a given size\r\n * @param texture Texture to copy from\r\n * @param width defines the desired width\r\n * @param height defines the desired height\r\n * @param useBilinearMode defines if bilinear mode has to be used\r\n * @returns the generated texture\r\n */\r\n CreateResizedCopy,\r\n\r\n /**\r\n * Apply a post process to a texture\r\n * @param postProcessName name of the fragment post process\r\n * @param internalTexture the texture to encode\r\n * @param scene the scene hosting the texture\r\n * @param type type of the output texture. If not provided, use the one from internalTexture\r\n * @param samplingMode sampling mode to use to sample the source texture. If not provided, use the one from internalTexture\r\n * @param format format of the output texture. If not provided, use the one from internalTexture\r\n * @returns a promise with the internalTexture having its texture replaced by the result of the processing\r\n */\r\n ApplyPostProcess,\r\n /**\r\n * Converts a number to half float\r\n * @param value number to convert\r\n * @returns converted number\r\n */\r\n ToHalfFloat,\r\n\r\n /**\r\n * Converts a half float to a number\r\n * @param value half float to convert\r\n * @returns converted half float\r\n */\r\n FromHalfFloat,\r\n};\r\n","import { Constants } from \"../Engines/constants\";\r\nimport { PostProcess } from \"../PostProcesses/postProcess\";\r\nimport \"../Shaders/rgbdDecode.fragment\";\r\nimport type { Engine } from \"../Engines/engine\";\r\n\r\nimport \"../Engines/Extensions/engine.renderTarget\";\r\nimport { ApplyPostProcess } from \"./textureTools\";\r\n\r\nimport type { Texture } from \"../Materials/Textures/texture\";\r\nimport type { InternalTexture } from \"../Materials/Textures/internalTexture\";\r\nimport type { Scene } from \"../scene\";\r\n\r\n/**\r\n * Class used to host RGBD texture specific utilities\r\n */\r\nexport class RGBDTextureTools {\r\n /**\r\n * Expand the RGBD Texture from RGBD to Half Float if possible.\r\n * @param texture the texture to expand.\r\n */\r\n public static ExpandRGBDTexture(texture: Texture) {\r\n const internalTexture = texture._texture;\r\n if (!internalTexture || !texture.isRGBD) {\r\n return;\r\n }\r\n\r\n // Gets everything ready.\r\n const engine = internalTexture.getEngine() as Engine;\r\n const caps = engine.getCaps();\r\n const isReady = internalTexture.isReady;\r\n let expandTexture = false;\r\n\r\n // If half float available we can uncompress the texture\r\n if (caps.textureHalfFloatRender && caps.textureHalfFloatLinearFiltering) {\r\n expandTexture = true;\r\n internalTexture.type = Constants.TEXTURETYPE_HALF_FLOAT;\r\n }\r\n // If full float available we can uncompress the texture\r\n else if (caps.textureFloatRender && caps.textureFloatLinearFiltering) {\r\n expandTexture = true;\r\n internalTexture.type = Constants.TEXTURETYPE_FLOAT;\r\n }\r\n\r\n if (expandTexture) {\r\n // Do not use during decode.\r\n internalTexture.isReady = false;\r\n internalTexture._isRGBD = false;\r\n internalTexture.invertY = false;\r\n }\r\n\r\n const expandRGBDTexture = () => {\r\n // Expand the texture if possible\r\n if (expandTexture) {\r\n // Simply run through the decode PP.\r\n const rgbdPostProcess = new PostProcess(\r\n \"rgbdDecode\",\r\n \"rgbdDecode\",\r\n null,\r\n null,\r\n 1,\r\n null,\r\n Constants.TEXTURE_TRILINEAR_SAMPLINGMODE,\r\n engine,\r\n false,\r\n undefined,\r\n internalTexture.type,\r\n undefined,\r\n null,\r\n false\r\n );\r\n rgbdPostProcess.externalTextureSamplerBinding = true;\r\n\r\n // Hold the output of the decoding.\r\n const expandedTexture = engine.createRenderTargetTexture(internalTexture.width, {\r\n generateDepthBuffer: false,\r\n generateMipMaps: false,\r\n generateStencilBuffer: false,\r\n samplingMode: internalTexture.samplingMode,\r\n type: internalTexture.type,\r\n format: Constants.TEXTUREFORMAT_RGBA,\r\n });\r\n\r\n rgbdPostProcess.getEffect().executeWhenCompiled(() => {\r\n // PP Render Pass\r\n rgbdPostProcess.onApply = (effect) => {\r\n effect._bindTexture(\"textureSampler\", internalTexture);\r\n effect.setFloat2(\"scale\", 1, 1);\r\n };\r\n texture.getScene()!.postProcessManager.directRender([rgbdPostProcess!], expandedTexture, true);\r\n\r\n // Cleanup\r\n engine.restoreDefaultFramebuffer();\r\n engine._releaseTexture(internalTexture);\r\n if (rgbdPostProcess) {\r\n rgbdPostProcess.dispose();\r\n }\r\n\r\n // Internal Swap\r\n expandedTexture._swapAndDie(internalTexture);\r\n\r\n // Ready to get rolling again.\r\n internalTexture.isReady = true;\r\n });\r\n }\r\n };\r\n\r\n if (isReady) {\r\n expandRGBDTexture();\r\n } else {\r\n texture.onLoadObservable.addOnce(expandRGBDTexture);\r\n }\r\n }\r\n\r\n /**\r\n * Encode the texture to RGBD if possible.\r\n * @param internalTexture the texture to encode\r\n * @param scene the scene hosting the texture\r\n * @param outputTextureType type of the texture in which the encoding is performed\r\n * @returns a promise with the internalTexture having its texture replaced by the result of the processing\r\n */\r\n public static EncodeTextureToRGBD(internalTexture: InternalTexture, scene: Scene, outputTextureType = Constants.TEXTURETYPE_UNSIGNED_BYTE): Promise {\r\n return ApplyPostProcess(\"rgbdEncode\", internalTexture, scene, outputTextureType, Constants.TEXTURE_NEAREST_SAMPLINGMODE, Constants.TEXTUREFORMAT_RGBA);\r\n }\r\n}\r\n","import type { BaseTexture } from \"../Materials/Textures/baseTexture\";\r\nimport { Texture } from \"../Materials/Textures/texture\";\r\nimport type { Scene } from \"../scene\";\r\nimport { RGBDTextureTools } from \"./rgbdTextureTools\";\r\nimport { Tools } from \"./tools\";\r\n\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nconst _environmentBRDFBase64Texture =\r\n \"\";\r\n\r\nlet _instanceNumber = 0;\r\n\r\n/**\r\n * Gets a default environment BRDF for MS-BRDF Height Correlated BRDF\r\n * @param scene defines the hosting scene\r\n * @returns the environment BRDF texture\r\n */\r\nexport const GetEnvironmentBRDFTexture = (scene: Scene): BaseTexture => {\r\n if (!scene.environmentBRDFTexture) {\r\n // Forces Delayed Texture Loading to prevent undefined error whilst setting RGBD values.\r\n const useDelayedTextureLoading = scene.useDelayedTextureLoading;\r\n scene.useDelayedTextureLoading = false;\r\n\r\n const previousState = scene._blockEntityCollection;\r\n scene._blockEntityCollection = false;\r\n const texture = Texture.CreateFromBase64String(\r\n _environmentBRDFBase64Texture,\r\n \"EnvironmentBRDFTexture\" + _instanceNumber++,\r\n scene,\r\n true,\r\n false,\r\n Texture.BILINEAR_SAMPLINGMODE\r\n );\r\n scene._blockEntityCollection = previousState;\r\n // BRDF Texture should not be cached here due to pre processing and redundant scene caches.\r\n const texturesCache = scene.getEngine().getLoadedTexturesCache();\r\n const index = texturesCache.indexOf(texture.getInternalTexture()!);\r\n if (index !== -1) {\r\n texturesCache.splice(index, 1);\r\n }\r\n\r\n texture.isRGBD = true;\r\n texture.wrapU = Texture.CLAMP_ADDRESSMODE;\r\n texture.wrapV = Texture.CLAMP_ADDRESSMODE;\r\n scene.environmentBRDFTexture = texture;\r\n\r\n scene.useDelayedTextureLoading = useDelayedTextureLoading;\r\n\r\n RGBDTextureTools.ExpandRGBDTexture(texture);\r\n\r\n const observer = scene.getEngine().onContextRestoredObservable.add(() => {\r\n texture.isRGBD = true;\r\n const checkReady = () => {\r\n if (texture.isReady()) {\r\n RGBDTextureTools.ExpandRGBDTexture(texture);\r\n } else {\r\n Tools.SetImmediate(checkReady);\r\n }\r\n };\r\n checkReady();\r\n });\r\n\r\n scene.onDisposeObservable.add(() => {\r\n scene.getEngine().onContextRestoredObservable.remove(observer);\r\n });\r\n }\r\n\r\n return scene.environmentBRDFTexture;\r\n};\r\n\r\n/**\r\n * Class used to host texture specific utilities\r\n */\r\nexport const BRDFTextureTools = {\r\n /**\r\n * Gets a default environment BRDF for MS-BRDF Height Correlated BRDF\r\n * @param scene defines the hosting scene\r\n * @returns the environment BRDF texture\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n GetEnvironmentBRDFTexture,\r\n};\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport { Constants } from \"../../Engines/constants\";\r\nimport { serialize, expandToProperty } from \"../../Misc/decorators\";\r\nimport { MaterialDefines } from \"../materialDefines\";\r\nimport { MaterialPluginBase } from \"../materialPluginBase\";\r\nimport type { PBRBaseMaterial } from \"./pbrBaseMaterial\";\r\n\r\n/**\r\n * @internal\r\n */\r\nexport class MaterialBRDFDefines extends MaterialDefines {\r\n BRDF_V_HEIGHT_CORRELATED = false;\r\n MS_BRDF_ENERGY_CONSERVATION = false;\r\n SPHERICAL_HARMONICS = false;\r\n SPECULAR_GLOSSINESS_ENERGY_CONSERVATION = false;\r\n}\r\n\r\n/**\r\n * Plugin that implements the BRDF component of the PBR material\r\n */\r\nexport class PBRBRDFConfiguration extends MaterialPluginBase {\r\n /**\r\n * Default value used for the energy conservation.\r\n * This should only be changed to adapt to the type of texture in scene.environmentBRDFTexture.\r\n */\r\n public static DEFAULT_USE_ENERGY_CONSERVATION = true;\r\n\r\n /**\r\n * Default value used for the Smith Visibility Height Correlated mode.\r\n * This should only be changed to adapt to the type of texture in scene.environmentBRDFTexture.\r\n */\r\n public static DEFAULT_USE_SMITH_VISIBILITY_HEIGHT_CORRELATED = true;\r\n\r\n /**\r\n * Default value used for the IBL diffuse part.\r\n * This can help switching back to the polynomials mode globally which is a tiny bit\r\n * less GPU intensive at the drawback of a lower quality.\r\n */\r\n public static DEFAULT_USE_SPHERICAL_HARMONICS = true;\r\n\r\n /**\r\n * Default value used for activating energy conservation for the specular workflow.\r\n * If activated, the albedo color is multiplied with (1. - maxChannel(specular color)).\r\n * If deactivated, a material is only physically plausible, when (albedo color + specular color) < 1.\r\n */\r\n public static DEFAULT_USE_SPECULAR_GLOSSINESS_INPUT_ENERGY_CONSERVATION = true;\r\n\r\n private _useEnergyConservation = PBRBRDFConfiguration.DEFAULT_USE_ENERGY_CONSERVATION;\r\n /**\r\n * Defines if the material uses energy conservation.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsMiscDirty\")\r\n public useEnergyConservation = PBRBRDFConfiguration.DEFAULT_USE_ENERGY_CONSERVATION;\r\n\r\n private _useSmithVisibilityHeightCorrelated = PBRBRDFConfiguration.DEFAULT_USE_SMITH_VISIBILITY_HEIGHT_CORRELATED;\r\n /**\r\n * LEGACY Mode set to false\r\n * Defines if the material uses height smith correlated visibility term.\r\n * If you intent to not use our default BRDF, you need to load a separate BRDF Texture for the PBR\r\n * You can either load https://assets.babylonjs.com/environments/uncorrelatedBRDF.png\r\n * or https://assets.babylonjs.com/environments/uncorrelatedBRDF.dds to have more precision\r\n * Not relying on height correlated will also disable energy conservation.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsMiscDirty\")\r\n public useSmithVisibilityHeightCorrelated = PBRBRDFConfiguration.DEFAULT_USE_SMITH_VISIBILITY_HEIGHT_CORRELATED;\r\n\r\n private _useSphericalHarmonics = PBRBRDFConfiguration.DEFAULT_USE_SPHERICAL_HARMONICS;\r\n /**\r\n * LEGACY Mode set to false\r\n * Defines if the material uses spherical harmonics vs spherical polynomials for the\r\n * diffuse part of the IBL.\r\n * The harmonics despite a tiny bigger cost has been proven to provide closer results\r\n * to the ground truth.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsMiscDirty\")\r\n public useSphericalHarmonics = PBRBRDFConfiguration.DEFAULT_USE_SPHERICAL_HARMONICS;\r\n\r\n private _useSpecularGlossinessInputEnergyConservation = PBRBRDFConfiguration.DEFAULT_USE_SPECULAR_GLOSSINESS_INPUT_ENERGY_CONSERVATION;\r\n /**\r\n * Defines if the material uses energy conservation, when the specular workflow is active.\r\n * If activated, the albedo color is multiplied with (1. - maxChannel(specular color)).\r\n * If deactivated, a material is only physically plausible, when (albedo color + specular color) < 1.\r\n * In the deactivated case, the material author has to ensure energy conservation, for a physically plausible rendering.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsMiscDirty\")\r\n public useSpecularGlossinessInputEnergyConservation = PBRBRDFConfiguration.DEFAULT_USE_SPECULAR_GLOSSINESS_INPUT_ENERGY_CONSERVATION;\r\n\r\n /** @internal */\r\n private _internalMarkAllSubMeshesAsMiscDirty: () => void;\r\n\r\n /** @internal */\r\n public _markAllSubMeshesAsMiscDirty(): void {\r\n this._internalMarkAllSubMeshesAsMiscDirty();\r\n }\r\n\r\n constructor(material: PBRBaseMaterial, addToPluginList = true) {\r\n super(material, \"PBRBRDF\", 90, new MaterialBRDFDefines(), addToPluginList);\r\n\r\n this._internalMarkAllSubMeshesAsMiscDirty = material._dirtyCallbacks[Constants.MATERIAL_MiscDirtyFlag];\r\n this._enable(true);\r\n }\r\n\r\n public prepareDefines(defines: MaterialBRDFDefines): void {\r\n defines.BRDF_V_HEIGHT_CORRELATED = this._useSmithVisibilityHeightCorrelated;\r\n defines.MS_BRDF_ENERGY_CONSERVATION = this._useEnergyConservation && this._useSmithVisibilityHeightCorrelated;\r\n defines.SPHERICAL_HARMONICS = this._useSphericalHarmonics;\r\n defines.SPECULAR_GLOSSINESS_ENERGY_CONSERVATION = this._useSpecularGlossinessInputEnergyConservation;\r\n }\r\n\r\n public getClassName(): string {\r\n return \"PBRBRDFConfiguration\";\r\n }\r\n}\r\n","import { Vector3 } from \"../../Maths/math.vector\";\r\nimport { Scalar } from \"../../Maths/math.scalar\";\r\nimport { SphericalPolynomial, SphericalHarmonics } from \"../../Maths/sphericalPolynomial\";\r\nimport type { BaseTexture } from \"../../Materials/Textures/baseTexture\";\r\nimport type { Nullable } from \"../../types\";\r\nimport { Constants } from \"../../Engines/constants\";\r\nimport type { CubeMapInfo } from \"./panoramaToCubemap\";\r\nimport { ToLinearSpace } from \"../../Maths/math.constants\";\r\nimport { Color3 } from \"../../Maths/math.color\";\r\n\r\nclass FileFaceOrientation {\r\n public name: string;\r\n public worldAxisForNormal: Vector3; // the world axis corresponding to the normal to the face\r\n public worldAxisForFileX: Vector3; // the world axis corresponding to texture right x-axis in file\r\n public worldAxisForFileY: Vector3; // the world axis corresponding to texture down y-axis in file\r\n\r\n public constructor(name: string, worldAxisForNormal: Vector3, worldAxisForFileX: Vector3, worldAxisForFileY: Vector3) {\r\n this.name = name;\r\n this.worldAxisForNormal = worldAxisForNormal;\r\n this.worldAxisForFileX = worldAxisForFileX;\r\n this.worldAxisForFileY = worldAxisForFileY;\r\n }\r\n}\r\n\r\n/**\r\n * Helper class dealing with the extraction of spherical polynomial dataArray\r\n * from a cube map.\r\n */\r\nexport class CubeMapToSphericalPolynomialTools {\r\n private static _FileFaces: FileFaceOrientation[] = [\r\n new FileFaceOrientation(\"right\", new Vector3(1, 0, 0), new Vector3(0, 0, -1), new Vector3(0, -1, 0)), // +X east\r\n new FileFaceOrientation(\"left\", new Vector3(-1, 0, 0), new Vector3(0, 0, 1), new Vector3(0, -1, 0)), // -X west\r\n new FileFaceOrientation(\"up\", new Vector3(0, 1, 0), new Vector3(1, 0, 0), new Vector3(0, 0, 1)), // +Y north\r\n new FileFaceOrientation(\"down\", new Vector3(0, -1, 0), new Vector3(1, 0, 0), new Vector3(0, 0, -1)), // -Y south\r\n new FileFaceOrientation(\"front\", new Vector3(0, 0, 1), new Vector3(1, 0, 0), new Vector3(0, -1, 0)), // +Z top\r\n new FileFaceOrientation(\"back\", new Vector3(0, 0, -1), new Vector3(-1, 0, 0), new Vector3(0, -1, 0)), // -Z bottom\r\n ];\r\n\r\n /** @internal */\r\n public static MAX_HDRI_VALUE = 4096;\r\n /** @internal */\r\n public static PRESERVE_CLAMPED_COLORS = false;\r\n\r\n /**\r\n * Converts a texture to the according Spherical Polynomial data.\r\n * This extracts the first 3 orders only as they are the only one used in the lighting.\r\n *\r\n * @param texture The texture to extract the information from.\r\n * @returns The Spherical Polynomial data.\r\n */\r\n public static ConvertCubeMapTextureToSphericalPolynomial(texture: BaseTexture): Nullable> {\r\n if (!texture.isCube) {\r\n // Only supports cube Textures currently.\r\n return null;\r\n }\r\n\r\n texture.getScene()?.getEngine().flushFramebuffer();\r\n\r\n const size = texture.getSize().width;\r\n const rightPromise = texture.readPixels(0, undefined, undefined, false);\r\n const leftPromise = texture.readPixels(1, undefined, undefined, false);\r\n\r\n let upPromise: Nullable>;\r\n let downPromise: Nullable>;\r\n if (texture.isRenderTarget) {\r\n upPromise = texture.readPixels(3, undefined, undefined, false);\r\n downPromise = texture.readPixels(2, undefined, undefined, false);\r\n } else {\r\n upPromise = texture.readPixels(2, undefined, undefined, false);\r\n downPromise = texture.readPixels(3, undefined, undefined, false);\r\n }\r\n\r\n const frontPromise = texture.readPixels(4, undefined, undefined, false);\r\n const backPromise = texture.readPixels(5, undefined, undefined, false);\r\n\r\n const gammaSpace = texture.gammaSpace;\r\n // Always read as RGBA.\r\n const format = Constants.TEXTUREFORMAT_RGBA;\r\n let type = Constants.TEXTURETYPE_UNSIGNED_INT;\r\n if (texture.textureType == Constants.TEXTURETYPE_FLOAT || texture.textureType == Constants.TEXTURETYPE_HALF_FLOAT) {\r\n type = Constants.TEXTURETYPE_FLOAT;\r\n }\r\n\r\n return new Promise((resolve) => {\r\n Promise.all([leftPromise, rightPromise, upPromise, downPromise, frontPromise, backPromise]).then(([left, right, up, down, front, back]) => {\r\n const cubeInfo: CubeMapInfo = {\r\n size,\r\n right,\r\n left,\r\n up,\r\n down,\r\n front,\r\n back,\r\n format,\r\n type,\r\n gammaSpace,\r\n };\r\n\r\n resolve(this.ConvertCubeMapToSphericalPolynomial(cubeInfo));\r\n });\r\n });\r\n }\r\n\r\n /**\r\n * Compute the area on the unit sphere of the rectangle defined by (x,y) and the origin\r\n * See https://www.rorydriscoll.com/2012/01/15/cubemap-texel-solid-angle/\r\n * @param x\r\n * @param y\r\n */\r\n private static _AreaElement(x: number, y: number): number {\r\n return Math.atan2(x * y, Math.sqrt(x * x + y * y + 1));\r\n }\r\n\r\n /**\r\n * Converts a cubemap to the according Spherical Polynomial data.\r\n * This extracts the first 3 orders only as they are the only one used in the lighting.\r\n *\r\n * @param cubeInfo The Cube map to extract the information from.\r\n * @returns The Spherical Polynomial data.\r\n */\r\n public static ConvertCubeMapToSphericalPolynomial(cubeInfo: CubeMapInfo): SphericalPolynomial {\r\n const sphericalHarmonics = new SphericalHarmonics();\r\n let totalSolidAngle = 0.0;\r\n\r\n // The (u,v) range is [-1,+1], so the distance between each texel is 2/Size.\r\n const du = 2.0 / cubeInfo.size;\r\n const dv = du;\r\n\r\n const halfTexel = 0.5 * du;\r\n\r\n // The (u,v) of the first texel is half a texel from the corner (-1,-1).\r\n const minUV = halfTexel - 1.0;\r\n\r\n for (let faceIndex = 0; faceIndex < 6; faceIndex++) {\r\n const fileFace = this._FileFaces[faceIndex];\r\n const dataArray = (cubeInfo)[fileFace.name];\r\n let v = minUV;\r\n\r\n // TODO: we could perform the summation directly into a SphericalPolynomial (SP), which is more efficient than SphericalHarmonic (SH).\r\n // This is possible because during the summation we do not need the SH-specific properties, e.g. orthogonality.\r\n // Because SP is still linear, so summation is fine in that basis.\r\n const stride = cubeInfo.format === Constants.TEXTUREFORMAT_RGBA ? 4 : 3;\r\n for (let y = 0; y < cubeInfo.size; y++) {\r\n let u = minUV;\r\n\r\n for (let x = 0; x < cubeInfo.size; x++) {\r\n // World direction (not normalised)\r\n const worldDirection = fileFace.worldAxisForFileX.scale(u).add(fileFace.worldAxisForFileY.scale(v)).add(fileFace.worldAxisForNormal);\r\n worldDirection.normalize();\r\n\r\n const deltaSolidAngle =\r\n this._AreaElement(u - halfTexel, v - halfTexel) -\r\n this._AreaElement(u - halfTexel, v + halfTexel) -\r\n this._AreaElement(u + halfTexel, v - halfTexel) +\r\n this._AreaElement(u + halfTexel, v + halfTexel);\r\n\r\n let r = dataArray[y * cubeInfo.size * stride + x * stride + 0];\r\n let g = dataArray[y * cubeInfo.size * stride + x * stride + 1];\r\n let b = dataArray[y * cubeInfo.size * stride + x * stride + 2];\r\n\r\n // Prevent NaN harmonics with extreme HDRI data.\r\n if (isNaN(r)) {\r\n r = 0;\r\n }\r\n if (isNaN(g)) {\r\n g = 0;\r\n }\r\n if (isNaN(b)) {\r\n b = 0;\r\n }\r\n\r\n // Handle Integer types.\r\n if (cubeInfo.type === Constants.TEXTURETYPE_UNSIGNED_INT) {\r\n r /= 255;\r\n g /= 255;\r\n b /= 255;\r\n }\r\n\r\n // Handle Gamma space textures.\r\n if (cubeInfo.gammaSpace) {\r\n r = Math.pow(Scalar.Clamp(r), ToLinearSpace);\r\n g = Math.pow(Scalar.Clamp(g), ToLinearSpace);\r\n b = Math.pow(Scalar.Clamp(b), ToLinearSpace);\r\n }\r\n\r\n // Prevent to explode in case of really high dynamic ranges.\r\n // sh 3 would not be enough to accurately represent it.\r\n const max = this.MAX_HDRI_VALUE;\r\n if (this.PRESERVE_CLAMPED_COLORS) {\r\n const currentMax = Math.max(r, g, b);\r\n if (currentMax > max) {\r\n const factor = max / currentMax;\r\n r *= factor;\r\n g *= factor;\r\n b *= factor;\r\n }\r\n } else {\r\n r = Scalar.Clamp(r, 0, max);\r\n g = Scalar.Clamp(g, 0, max);\r\n b = Scalar.Clamp(b, 0, max);\r\n }\r\n\r\n const color = new Color3(r, g, b);\r\n\r\n sphericalHarmonics.addLight(worldDirection, color, deltaSolidAngle);\r\n\r\n totalSolidAngle += deltaSolidAngle;\r\n\r\n u += du;\r\n }\r\n\r\n v += dv;\r\n }\r\n }\r\n\r\n // Solid angle for entire sphere is 4*pi\r\n const sphereSolidAngle = 4.0 * Math.PI;\r\n\r\n // Adjust the solid angle to allow for how many faces we processed.\r\n const facesProcessed = 6.0;\r\n const expectedSolidAngle = (sphereSolidAngle * facesProcessed) / 6.0;\r\n\r\n // Adjust the harmonics so that the accumulated solid angle matches the expected solid angle.\r\n // This is needed because the numerical integration over the cube uses a\r\n // small angle approximation of solid angle for each texel (see deltaSolidAngle),\r\n // and also to compensate for accumulative error due to float precision in the summation.\r\n const correctionFactor = expectedSolidAngle / totalSolidAngle;\r\n sphericalHarmonics.scaleInPlace(correctionFactor);\r\n\r\n sphericalHarmonics.convertIncidentRadianceToIrradiance();\r\n sphericalHarmonics.convertIrradianceToLambertianRadiance();\r\n\r\n return SphericalPolynomial.FromHarmonics(sphericalHarmonics);\r\n }\r\n}\r\n","import type { Nullable } from \"../../types\";\r\nimport { CubeMapToSphericalPolynomialTools } from \"../../Misc/HighDynamicRange/cubemapToSphericalPolynomial\";\r\nimport type { SphericalPolynomial } from \"../../Maths/sphericalPolynomial\";\r\nimport { BaseTexture } from \"./baseTexture\";\r\n\r\ndeclare module \"./baseTexture\" {\r\n export interface BaseTexture {\r\n /**\r\n * Get the polynomial representation of the texture data.\r\n * This is mainly use as a fast way to recover IBL Diffuse irradiance data.\r\n * @see https://learnopengl.com/PBR/IBL/Diffuse-irradiance\r\n */\r\n sphericalPolynomial: Nullable;\r\n\r\n /**\r\n * Force recomputation of spherical polynomials.\r\n * Can be useful if you generate a cubemap multiple times (from a probe for eg) and you need the proper polynomials each time\r\n */\r\n forceSphericalPolynomialsRecompute(): void;\r\n }\r\n}\r\n\r\nBaseTexture.prototype.forceSphericalPolynomialsRecompute = function (): void {\r\n if (this._texture) {\r\n this._texture._sphericalPolynomial = null;\r\n this._texture._sphericalPolynomialPromise = null;\r\n this._texture._sphericalPolynomialComputed = false;\r\n }\r\n};\r\n\r\nObject.defineProperty(BaseTexture.prototype, \"sphericalPolynomial\", {\r\n get: function (this: BaseTexture) {\r\n if (this._texture) {\r\n if (this._texture._sphericalPolynomial || this._texture._sphericalPolynomialComputed) {\r\n return this._texture._sphericalPolynomial;\r\n }\r\n\r\n if (this._texture.isReady) {\r\n if (!this._texture._sphericalPolynomialPromise) {\r\n this._texture._sphericalPolynomialPromise = CubeMapToSphericalPolynomialTools.ConvertCubeMapTextureToSphericalPolynomial(this);\r\n if (this._texture._sphericalPolynomialPromise === null) {\r\n this._texture._sphericalPolynomialComputed = true;\r\n } else {\r\n this._texture._sphericalPolynomialPromise.then((sphericalPolynomial) => {\r\n this._texture!._sphericalPolynomial = sphericalPolynomial;\r\n this._texture!._sphericalPolynomialComputed = true;\r\n });\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n }\r\n\r\n return null;\r\n },\r\n set: function (this: BaseTexture, value: Nullable) {\r\n if (this._texture) {\r\n this._texture._sphericalPolynomial = value;\r\n }\r\n },\r\n enumerable: true,\r\n configurable: true,\r\n});\r\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\nimport \"./decalFragmentDeclaration\";\n\nconst name = \"pbrFragmentDeclaration\";\nconst shader = `uniform vec4 vEyePosition;uniform vec3 vReflectionColor;uniform vec4 vAlbedoColor;uniform vec4 vLightingIntensity;uniform vec4 vReflectivityColor;uniform vec4 vMetallicReflectanceFactors;uniform vec3 vEmissiveColor;uniform float visibility;uniform vec3 vAmbientColor;\n#ifdef ALBEDO\nuniform vec2 vAlbedoInfos;\n#endif\n#ifdef AMBIENT\nuniform vec4 vAmbientInfos;\n#endif\n#ifdef BUMP\nuniform vec3 vBumpInfos;uniform vec2 vTangentSpaceParams;\n#endif\n#ifdef OPACITY\nuniform vec2 vOpacityInfos;\n#endif\n#ifdef EMISSIVE\nuniform vec2 vEmissiveInfos;\n#endif\n#ifdef LIGHTMAP\nuniform vec2 vLightmapInfos;\n#endif\n#ifdef REFLECTIVITY\nuniform vec3 vReflectivityInfos;\n#endif\n#ifdef MICROSURFACEMAP\nuniform vec2 vMicroSurfaceSamplerInfos;\n#endif\n#if defined(REFLECTIONMAP_SPHERICAL) || defined(REFLECTIONMAP_PROJECTION) || defined(SS_REFRACTION) || defined(PREPASS)\nuniform mat4 view;\n#endif\n#ifdef REFLECTION\nuniform vec2 vReflectionInfos;\n#ifdef REALTIME_FILTERING\nuniform vec2 vReflectionFilteringInfo;\n#endif\nuniform mat4 reflectionMatrix;uniform vec3 vReflectionMicrosurfaceInfos;\n#if defined(USE_LOCAL_REFLECTIONMAP_CUBIC) && defined(REFLECTIONMAP_CUBIC)\nuniform vec3 vReflectionPosition;uniform vec3 vReflectionSize; \n#endif\n#endif\n#if defined(SS_REFRACTION) && defined(SS_USE_LOCAL_REFRACTIONMAP_CUBIC)\nuniform vec3 vRefractionPosition;uniform vec3 vRefractionSize; \n#endif\n#ifdef CLEARCOAT\nuniform vec2 vClearCoatParams;uniform vec4 vClearCoatRefractionParams;\n#if defined(CLEARCOAT_TEXTURE) || defined(CLEARCOAT_TEXTURE_ROUGHNESS)\nuniform vec4 vClearCoatInfos;\n#endif\n#ifdef CLEARCOAT_TEXTURE\nuniform mat4 clearCoatMatrix;\n#endif\n#ifdef CLEARCOAT_TEXTURE_ROUGHNESS\nuniform mat4 clearCoatRoughnessMatrix;\n#endif\n#ifdef CLEARCOAT_BUMP\nuniform vec2 vClearCoatBumpInfos;uniform vec2 vClearCoatTangentSpaceParams;uniform mat4 clearCoatBumpMatrix;\n#endif\n#ifdef CLEARCOAT_TINT\nuniform vec4 vClearCoatTintParams;uniform float clearCoatColorAtDistance;\n#ifdef CLEARCOAT_TINT_TEXTURE\nuniform vec2 vClearCoatTintInfos;uniform mat4 clearCoatTintMatrix;\n#endif\n#endif\n#endif\n#ifdef IRIDESCENCE\nuniform vec4 vIridescenceParams;\n#if defined(IRIDESCENCE_TEXTURE) || defined(IRIDESCENCE_THICKNESS_TEXTURE)\nuniform vec4 vIridescenceInfos;\n#endif\n#ifdef IRIDESCENCE_TEXTURE\nuniform mat4 iridescenceMatrix;\n#endif\n#ifdef IRIDESCENCE_THICKNESS_TEXTURE\nuniform mat4 iridescenceThicknessMatrix;\n#endif\n#endif\n#ifdef ANISOTROPIC\nuniform vec3 vAnisotropy;\n#ifdef ANISOTROPIC_TEXTURE\nuniform vec2 vAnisotropyInfos;uniform mat4 anisotropyMatrix;\n#endif\n#endif\n#ifdef SHEEN\nuniform vec4 vSheenColor;\n#ifdef SHEEN_ROUGHNESS\nuniform float vSheenRoughness;\n#endif\n#if defined(SHEEN_TEXTURE) || defined(SHEEN_TEXTURE_ROUGHNESS)\nuniform vec4 vSheenInfos;\n#endif\n#ifdef SHEEN_TEXTURE\nuniform mat4 sheenMatrix;\n#endif\n#ifdef SHEEN_TEXTURE_ROUGHNESS\nuniform mat4 sheenRoughnessMatrix;\n#endif\n#endif\n#ifdef SUBSURFACE\n#ifdef SS_REFRACTION\nuniform vec4 vRefractionMicrosurfaceInfos;uniform vec4 vRefractionInfos;uniform mat4 refractionMatrix;\n#ifdef REALTIME_FILTERING\nuniform vec2 vRefractionFilteringInfo;\n#endif\n#endif\n#ifdef SS_THICKNESSANDMASK_TEXTURE\nuniform vec2 vThicknessInfos;uniform mat4 thicknessMatrix;\n#endif\n#ifdef SS_REFRACTIONINTENSITY_TEXTURE\nuniform vec2 vRefractionIntensityInfos;uniform mat4 refractionIntensityMatrix;\n#endif\n#ifdef SS_TRANSLUCENCYINTENSITY_TEXTURE\nuniform vec2 vTranslucencyIntensityInfos;uniform mat4 translucencyIntensityMatrix;\n#endif\nuniform vec2 vThicknessParam;uniform vec3 vDiffusionDistance;uniform vec4 vTintColor;uniform vec3 vSubSurfaceIntensity;\n#endif\n#ifdef PREPASS\n#ifdef SS_SCATTERING\nuniform float scatteringDiffusionProfile;\n#endif\n#endif\n#if DEBUGMODE>0\nuniform vec2 vDebugMode;\n#endif\n#ifdef DETAIL\nuniform vec4 vDetailInfos;\n#endif\n#include\n#ifdef USESPHERICALFROMREFLECTIONMAP\n#ifdef SPHERICAL_HARMONICS\nuniform vec3 vSphericalL00;uniform vec3 vSphericalL1_1;uniform vec3 vSphericalL10;uniform vec3 vSphericalL11;uniform vec3 vSphericalL2_2;uniform vec3 vSphericalL2_1;uniform vec3 vSphericalL20;uniform vec3 vSphericalL21;uniform vec3 vSphericalL22;\n#else\nuniform vec3 vSphericalX;uniform vec3 vSphericalY;uniform vec3 vSphericalZ;uniform vec3 vSphericalXX_ZZ;uniform vec3 vSphericalYY_ZZ;uniform vec3 vSphericalZZ;uniform vec3 vSphericalXY;uniform vec3 vSphericalYZ;uniform vec3 vSphericalZX;\n#endif\n#endif\n#define ADDITIONAL_FRAGMENT_DECLARATION\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrFragmentDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\nimport \"./sceneUboDeclaration\";\nimport \"./meshUboDeclaration\";\n\nconst name = \"pbrUboDeclaration\";\nconst shader = `layout(std140,column_major) uniform;uniform Material {vec2 vAlbedoInfos;vec4 vAmbientInfos;vec2 vOpacityInfos;vec2 vEmissiveInfos;vec2 vLightmapInfos;vec3 vReflectivityInfos;vec2 vMicroSurfaceSamplerInfos;vec2 vReflectionInfos;vec2 vReflectionFilteringInfo;vec3 vReflectionPosition;vec3 vReflectionSize;vec3 vBumpInfos;mat4 albedoMatrix;mat4 ambientMatrix;mat4 opacityMatrix;mat4 emissiveMatrix;mat4 lightmapMatrix;mat4 reflectivityMatrix;mat4 microSurfaceSamplerMatrix;mat4 bumpMatrix;vec2 vTangentSpaceParams;mat4 reflectionMatrix;vec3 vReflectionColor;vec4 vAlbedoColor;vec4 vLightingIntensity;vec3 vReflectionMicrosurfaceInfos;float pointSize;vec4 vReflectivityColor;vec3 vEmissiveColor;vec3 vAmbientColor;vec2 vDebugMode;vec4 vMetallicReflectanceFactors;vec2 vMetallicReflectanceInfos;mat4 metallicReflectanceMatrix;vec2 vReflectanceInfos;mat4 reflectanceMatrix;vec3 vSphericalL00;vec3 vSphericalL1_1;vec3 vSphericalL10;vec3 vSphericalL11;vec3 vSphericalL2_2;vec3 vSphericalL2_1;vec3 vSphericalL20;vec3 vSphericalL21;vec3 vSphericalL22;vec3 vSphericalX;vec3 vSphericalY;vec3 vSphericalZ;vec3 vSphericalXX_ZZ;vec3 vSphericalYY_ZZ;vec3 vSphericalZZ;vec3 vSphericalXY;vec3 vSphericalYZ;vec3 vSphericalZX;\n#define ADDITIONAL_UBO_DECLARATION\n};\n#include\n#include\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrUboDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\nimport \"./mainUVVaryingDeclaration\";\n\nconst name = \"pbrFragmentExtraDeclaration\";\nconst shader = `varying vec3 vPositionW;\n#if DEBUGMODE>0\nvarying vec4 vClipSpacePosition;\n#endif\n#include[1..7]\n#ifdef NORMAL\nvarying vec3 vNormalW;\n#if defined(USESPHERICALFROMREFLECTIONMAP) && defined(USESPHERICALINVERTEX)\nvarying vec3 vEnvironmentIrradiance;\n#endif\n#endif\n#if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES)\nvarying vec4 vColor;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrFragmentExtraDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"samplerFragmentAlternateDeclaration\";\nconst shader = `#ifdef _DEFINENAME_\n#if _DEFINENAME_DIRECTUV==1\n#define v_VARYINGNAME_UV vMainUV1\n#elif _DEFINENAME_DIRECTUV==2\n#define v_VARYINGNAME_UV vMainUV2\n#elif _DEFINENAME_DIRECTUV==3\n#define v_VARYINGNAME_UV vMainUV3\n#elif _DEFINENAME_DIRECTUV==4\n#define v_VARYINGNAME_UV vMainUV4\n#elif _DEFINENAME_DIRECTUV==5\n#define v_VARYINGNAME_UV vMainUV5\n#elif _DEFINENAME_DIRECTUV==6\n#define v_VARYINGNAME_UV vMainUV6\n#else\nvarying vec2 v_VARYINGNAME_UV;\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const samplerFragmentAlternateDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\nimport \"./samplerFragmentDeclaration\";\nimport \"./samplerFragmentAlternateDeclaration\";\n\nconst name = \"pbrFragmentSamplersDeclaration\";\nconst shader = `#include(_DEFINENAME_,ALBEDO,_VARYINGNAME_,Albedo,_SAMPLERNAME_,albedo)\n#include(_DEFINENAME_,AMBIENT,_VARYINGNAME_,Ambient,_SAMPLERNAME_,ambient)\n#include(_DEFINENAME_,OPACITY,_VARYINGNAME_,Opacity,_SAMPLERNAME_,opacity)\n#include(_DEFINENAME_,EMISSIVE,_VARYINGNAME_,Emissive,_SAMPLERNAME_,emissive)\n#include(_DEFINENAME_,LIGHTMAP,_VARYINGNAME_,Lightmap,_SAMPLERNAME_,lightmap)\n#include(_DEFINENAME_,REFLECTIVITY,_VARYINGNAME_,Reflectivity,_SAMPLERNAME_,reflectivity)\n#include(_DEFINENAME_,MICROSURFACEMAP,_VARYINGNAME_,MicroSurfaceSampler,_SAMPLERNAME_,microSurface)\n#include(_DEFINENAME_,METALLIC_REFLECTANCE,_VARYINGNAME_,MetallicReflectance,_SAMPLERNAME_,metallicReflectance)\n#include(_DEFINENAME_,REFLECTANCE,_VARYINGNAME_,Reflectance,_SAMPLERNAME_,reflectance)\n#include(_DEFINENAME_,DECAL,_VARYINGNAME_,Decal,_SAMPLERNAME_,decal)\n#ifdef CLEARCOAT\n#include(_DEFINENAME_,CLEARCOAT_TEXTURE,_VARYINGNAME_,ClearCoat,_SAMPLERNAME_,clearCoat)\n#include(_DEFINENAME_,CLEARCOAT_TEXTURE_ROUGHNESS,_VARYINGNAME_,ClearCoatRoughness)\n#if defined(CLEARCOAT_TEXTURE_ROUGHNESS) && !defined(CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL)\nuniform sampler2D clearCoatRoughnessSampler;\n#endif\n#include(_DEFINENAME_,CLEARCOAT_BUMP,_VARYINGNAME_,ClearCoatBump,_SAMPLERNAME_,clearCoatBump)\n#include(_DEFINENAME_,CLEARCOAT_TINT_TEXTURE,_VARYINGNAME_,ClearCoatTint,_SAMPLERNAME_,clearCoatTint)\n#endif\n#ifdef IRIDESCENCE\n#include(_DEFINENAME_,IRIDESCENCE_TEXTURE,_VARYINGNAME_,Iridescence,_SAMPLERNAME_,iridescence)\n#include(_DEFINENAME_,IRIDESCENCE_THICKNESS_TEXTURE,_VARYINGNAME_,IridescenceThickness,_SAMPLERNAME_,iridescenceThickness)\n#endif\n#ifdef SHEEN\n#include(_DEFINENAME_,SHEEN_TEXTURE,_VARYINGNAME_,Sheen,_SAMPLERNAME_,sheen)\n#include(_DEFINENAME_,SHEEN_TEXTURE_ROUGHNESS,_VARYINGNAME_,SheenRoughness)\n#if defined(SHEEN_ROUGHNESS) && defined(SHEEN_TEXTURE_ROUGHNESS) && !defined(SHEEN_TEXTURE_ROUGHNESS_IDENTICAL)\nuniform sampler2D sheenRoughnessSampler;\n#endif\n#endif\n#ifdef ANISOTROPIC\n#include(_DEFINENAME_,ANISOTROPIC_TEXTURE,_VARYINGNAME_,Anisotropy,_SAMPLERNAME_,anisotropy)\n#endif\n#ifdef REFLECTION\n#ifdef REFLECTIONMAP_3D\n#define sampleReflection(s,c) textureCube(s,c)\nuniform samplerCube reflectionSampler;\n#ifdef LODBASEDMICROSFURACE\n#define sampleReflectionLod(s,c,l) textureCubeLodEXT(s,c,l)\n#else\nuniform samplerCube reflectionSamplerLow;uniform samplerCube reflectionSamplerHigh;\n#endif\n#ifdef USEIRRADIANCEMAP\nuniform samplerCube irradianceSampler;\n#endif\n#else\n#define sampleReflection(s,c) texture2D(s,c)\nuniform sampler2D reflectionSampler;\n#ifdef LODBASEDMICROSFURACE\n#define sampleReflectionLod(s,c,l) texture2DLodEXT(s,c,l)\n#else\nuniform sampler2D reflectionSamplerLow;uniform sampler2D reflectionSamplerHigh;\n#endif\n#ifdef USEIRRADIANCEMAP\nuniform sampler2D irradianceSampler;\n#endif\n#endif\n#ifdef REFLECTIONMAP_SKYBOX\nvarying vec3 vPositionUVW;\n#else\n#if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED)\nvarying vec3 vDirectionW;\n#endif\n#endif\n#endif\n#ifdef ENVIRONMENTBRDF\nuniform sampler2D environmentBrdfSampler;\n#endif\n#ifdef SUBSURFACE\n#ifdef SS_REFRACTION\n#ifdef SS_REFRACTIONMAP_3D\n#define sampleRefraction(s,c) textureCube(s,c)\nuniform samplerCube refractionSampler;\n#ifdef LODBASEDMICROSFURACE\n#define sampleRefractionLod(s,c,l) textureCubeLodEXT(s,c,l)\n#else\nuniform samplerCube refractionSamplerLow;uniform samplerCube refractionSamplerHigh;\n#endif\n#else\n#define sampleRefraction(s,c) texture2D(s,c)\nuniform sampler2D refractionSampler;\n#ifdef LODBASEDMICROSFURACE\n#define sampleRefractionLod(s,c,l) texture2DLodEXT(s,c,l)\n#else\nuniform sampler2D refractionSamplerLow;uniform sampler2D refractionSamplerHigh;\n#endif\n#endif\n#endif\n#include(_DEFINENAME_,SS_THICKNESSANDMASK_TEXTURE,_VARYINGNAME_,Thickness,_SAMPLERNAME_,thickness)\n#include(_DEFINENAME_,SS_REFRACTIONINTENSITY_TEXTURE,_VARYINGNAME_,RefractionIntensity,_SAMPLERNAME_,refractionIntensity)\n#include(_DEFINENAME_,SS_TRANSLUCENCYINTENSITY_TEXTURE,_VARYINGNAME_,TranslucencyIntensity,_SAMPLERNAME_,translucencyIntensity)\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrFragmentSamplersDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"subSurfaceScatteringFunctions\";\nconst shader = `bool testLightingForSSS(float diffusionProfile)\n{return diffusionProfile<1.;}`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const subSurfaceScatteringFunctions = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"importanceSampling\";\nconst shader = `vec3 hemisphereCosSample(vec2 u) {float phi=2.*PI*u.x;float cosTheta2=1.-u.y;float cosTheta=sqrt(cosTheta2);float sinTheta=sqrt(1.-cosTheta2);return vec3(sinTheta*cos(phi),sinTheta*sin(phi),cosTheta);}\nvec3 hemisphereImportanceSampleDggx(vec2 u,float a) {float phi=2.*PI*u.x;float cosTheta2=(1.-u.y)/(1.+(a+1.)*((a-1.)*u.y));float cosTheta=sqrt(cosTheta2);float sinTheta=sqrt(1.-cosTheta2);return vec3(sinTheta*cos(phi),sinTheta*sin(phi),cosTheta);}\nvec3 hemisphereImportanceSampleDCharlie(vec2 u,float a) { \nfloat phi=2.*PI*u.x;float sinTheta=pow(u.y,a/(2.*a+1.));float cosTheta=sqrt(1.-sinTheta*sinTheta);return vec3(sinTheta*cos(phi),sinTheta*sin(phi),cosTheta);}`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const importanceSampling = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrHelperFunctions\";\nconst shader = `#define RECIPROCAL_PI2 0.15915494\n#define RECIPROCAL_PI 0.31830988618\n#define MINIMUMVARIANCE 0.0005\nfloat convertRoughnessToAverageSlope(float roughness)\n{return square(roughness)+MINIMUMVARIANCE;}\nfloat fresnelGrazingReflectance(float reflectance0) {float reflectance90=saturate(reflectance0*25.0);return reflectance90;}\nvec2 getAARoughnessFactors(vec3 normalVector) {\n#ifdef SPECULARAA\nvec3 nDfdx=dFdx(normalVector.xyz);vec3 nDfdy=dFdy(normalVector.xyz);float slopeSquare=max(dot(nDfdx,nDfdx),dot(nDfdy,nDfdy));float geometricRoughnessFactor=pow(saturate(slopeSquare),0.333);float geometricAlphaGFactor=sqrt(slopeSquare);geometricAlphaGFactor*=0.75;return vec2(geometricRoughnessFactor,geometricAlphaGFactor);\n#else\nreturn vec2(0.);\n#endif\n}\n#ifdef ANISOTROPIC\n#ifdef ANISOTROPIC_LEGACY\nvec2 getAnisotropicRoughness(float alphaG,float anisotropy) {float alphaT=max(alphaG*(1.0+anisotropy),MINIMUMVARIANCE);float alphaB=max(alphaG*(1.0-anisotropy),MINIMUMVARIANCE);return vec2(alphaT,alphaB);}\nvec3 getAnisotropicBentNormals(const vec3 T,const vec3 B,const vec3 N,const vec3 V,float anisotropy,float roughness) {vec3 anisotropicFrameDirection=anisotropy>=0.0 ? B : T;vec3 anisotropicFrameTangent=cross(normalize(anisotropicFrameDirection),V);vec3 anisotropicFrameNormal=cross(anisotropicFrameTangent,anisotropicFrameDirection);vec3 anisotropicNormal=normalize(mix(N,anisotropicFrameNormal,abs(anisotropy)));return anisotropicNormal;}\n#else\nvec2 getAnisotropicRoughness(float alphaG,float anisotropy) {float alphaT=max(mix(alphaG,1.0,anisotropy*anisotropy),MINIMUMVARIANCE);float alphaB=max(alphaG,MINIMUMVARIANCE);return vec2(alphaT,alphaB);}\nvec3 getAnisotropicBentNormals(const vec3 T,const vec3 B,const vec3 N,const vec3 V,float anisotropy,float roughness) {vec3 bentNormal=cross(B,V);bentNormal=normalize(cross(bentNormal,B));float a=square(square(1.0-anisotropy*(1.0-roughness)));bentNormal=normalize(mix(bentNormal,N,a));return bentNormal;}\n#endif\n#endif\n#if defined(CLEARCOAT) || defined(SS_REFRACTION)\nvec3 cocaLambert(vec3 alpha,float distance) {return exp(-alpha*distance);}\nvec3 cocaLambert(float NdotVRefract,float NdotLRefract,vec3 alpha,float thickness) {return cocaLambert(alpha,(thickness*((NdotLRefract+NdotVRefract)/(NdotLRefract*NdotVRefract))));}\nvec3 computeColorAtDistanceInMedia(vec3 color,float distance) {return -log(color)/distance;}\nvec3 computeClearCoatAbsorption(float NdotVRefract,float NdotLRefract,vec3 clearCoatColor,float clearCoatThickness,float clearCoatIntensity) {vec3 clearCoatAbsorption=mix(vec3(1.0),\ncocaLambert(NdotVRefract,NdotLRefract,clearCoatColor,clearCoatThickness),\nclearCoatIntensity);return clearCoatAbsorption;}\n#endif\n#ifdef MICROSURFACEAUTOMATIC\nfloat computeDefaultMicroSurface(float microSurface,vec3 reflectivityColor)\n{const float kReflectivityNoAlphaWorkflow_SmoothnessMax=0.95;float reflectivityLuminance=getLuminance(reflectivityColor);float reflectivityLuma=sqrt(reflectivityLuminance);microSurface=reflectivityLuma*kReflectivityNoAlphaWorkflow_SmoothnessMax;return microSurface;}\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrHelperFunctions = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"harmonicsFunctions\";\nconst shader = `#ifdef USESPHERICALFROMREFLECTIONMAP\n#ifdef SPHERICAL_HARMONICS\nvec3 computeEnvironmentIrradiance(vec3 normal) {return vSphericalL00\n+ vSphericalL1_1*(normal.y)\n+ vSphericalL10*(normal.z)\n+ vSphericalL11*(normal.x)\n+ vSphericalL2_2*(normal.y*normal.x)\n+ vSphericalL2_1*(normal.y*normal.z)\n+ vSphericalL20*((3.0*normal.z*normal.z)-1.0)\n+ vSphericalL21*(normal.z*normal.x)\n+ vSphericalL22*(normal.x*normal.x-(normal.y*normal.y));}\n#else\nvec3 computeEnvironmentIrradiance(vec3 normal) {float Nx=normal.x;float Ny=normal.y;float Nz=normal.z;vec3 C1=vSphericalZZ.rgb;vec3 Cx=vSphericalX.rgb;vec3 Cy=vSphericalY.rgb;vec3 Cz=vSphericalZ.rgb;vec3 Cxx_zz=vSphericalXX_ZZ.rgb;vec3 Cyy_zz=vSphericalYY_ZZ.rgb;vec3 Cxy=vSphericalXY.rgb;vec3 Cyz=vSphericalYZ.rgb;vec3 Czx=vSphericalZX.rgb;vec3 a1=Cyy_zz*Ny+Cy;vec3 a2=Cyz*Nz+a1;vec3 b1=Czx*Nz+Cx;vec3 b2=Cxy*Ny+b1;vec3 b3=Cxx_zz*Nx+b2;vec3 t1=Cz *Nz+C1;vec3 t2=a2 *Ny+t1;vec3 t3=b3 *Nx+t2;return t3;}\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const harmonicsFunctions = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrDirectLightingSetupFunctions\";\nconst shader = `struct preLightingInfo\n{vec3 lightOffset;float lightDistanceSquared;float lightDistance;float attenuation;vec3 L;vec3 H;float NdotV;float NdotLUnclamped;float NdotL;float VdotH;float roughness;\n#ifdef IRIDESCENCE\nfloat iridescenceIntensity;\n#endif\n};preLightingInfo computePointAndSpotPreLightingInfo(vec4 lightData,vec3 V,vec3 N) {preLightingInfo result;result.lightOffset=lightData.xyz-vPositionW;result.lightDistanceSquared=dot(result.lightOffset,result.lightOffset);result.lightDistance=sqrt(result.lightDistanceSquared);result.L=normalize(result.lightOffset);result.H=normalize(V+result.L);result.VdotH=saturate(dot(V,result.H));result.NdotLUnclamped=dot(N,result.L);result.NdotL=saturateEps(result.NdotLUnclamped);return result;}\npreLightingInfo computeDirectionalPreLightingInfo(vec4 lightData,vec3 V,vec3 N) {preLightingInfo result;result.lightDistance=length(-lightData.xyz);result.L=normalize(-lightData.xyz);result.H=normalize(V+result.L);result.VdotH=saturate(dot(V,result.H));result.NdotLUnclamped=dot(N,result.L);result.NdotL=saturateEps(result.NdotLUnclamped);return result;}\npreLightingInfo computeHemisphericPreLightingInfo(vec4 lightData,vec3 V,vec3 N) {preLightingInfo result;result.NdotL=dot(N,lightData.xyz)*0.5+0.5;result.NdotL=saturateEps(result.NdotL);result.NdotLUnclamped=result.NdotL;\n#ifdef SPECULARTERM\nresult.L=normalize(lightData.xyz);result.H=normalize(V+result.L);result.VdotH=saturate(dot(V,result.H));\n#endif\nreturn result;}`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrDirectLightingSetupFunctions = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrDirectLightingFalloffFunctions\";\nconst shader = `float computeDistanceLightFalloff_Standard(vec3 lightOffset,float range)\n{return max(0.,1.0-length(lightOffset)/range);}\nfloat computeDistanceLightFalloff_Physical(float lightDistanceSquared)\n{return 1.0/maxEps(lightDistanceSquared);}\nfloat computeDistanceLightFalloff_GLTF(float lightDistanceSquared,float inverseSquaredRange)\n{float lightDistanceFalloff=1.0/maxEps(lightDistanceSquared);float factor=lightDistanceSquared*inverseSquaredRange;float attenuation=saturate(1.0-factor*factor);attenuation*=attenuation;lightDistanceFalloff*=attenuation;return lightDistanceFalloff;}\nfloat computeDistanceLightFalloff(vec3 lightOffset,float lightDistanceSquared,float range,float inverseSquaredRange)\n{\n#ifdef USEPHYSICALLIGHTFALLOFF\nreturn computeDistanceLightFalloff_Physical(lightDistanceSquared);\n#elif defined(USEGLTFLIGHTFALLOFF)\nreturn computeDistanceLightFalloff_GLTF(lightDistanceSquared,inverseSquaredRange);\n#else\nreturn computeDistanceLightFalloff_Standard(lightOffset,range);\n#endif\n}\nfloat computeDirectionalLightFalloff_Standard(vec3 lightDirection,vec3 directionToLightCenterW,float cosHalfAngle,float exponent)\n{float falloff=0.0;float cosAngle=maxEps(dot(-lightDirection,directionToLightCenterW));if (cosAngle>=cosHalfAngle)\n{falloff=max(0.,pow(cosAngle,exponent));}\nreturn falloff;}\nfloat computeDirectionalLightFalloff_Physical(vec3 lightDirection,vec3 directionToLightCenterW,float cosHalfAngle)\n{const float kMinusLog2ConeAngleIntensityRatio=6.64385618977; \nfloat concentrationKappa=kMinusLog2ConeAngleIntensityRatio/(1.0-cosHalfAngle);vec4 lightDirectionSpreadSG=vec4(-lightDirection*concentrationKappa,-concentrationKappa);float falloff=exp2(dot(vec4(directionToLightCenterW,1.0),lightDirectionSpreadSG));return falloff;}\nfloat computeDirectionalLightFalloff_GLTF(vec3 lightDirection,vec3 directionToLightCenterW,float lightAngleScale,float lightAngleOffset)\n{float cd=dot(-lightDirection,directionToLightCenterW);float falloff=saturate(cd*lightAngleScale+lightAngleOffset);falloff*=falloff;return falloff;}\nfloat computeDirectionalLightFalloff(vec3 lightDirection,vec3 directionToLightCenterW,float cosHalfAngle,float exponent,float lightAngleScale,float lightAngleOffset)\n{\n#ifdef USEPHYSICALLIGHTFALLOFF\nreturn computeDirectionalLightFalloff_Physical(lightDirection,directionToLightCenterW,cosHalfAngle);\n#elif defined(USEGLTFLIGHTFALLOFF)\nreturn computeDirectionalLightFalloff_GLTF(lightDirection,directionToLightCenterW,lightAngleScale,lightAngleOffset);\n#else\nreturn computeDirectionalLightFalloff_Standard(lightDirection,directionToLightCenterW,cosHalfAngle,exponent);\n#endif\n}`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrDirectLightingFalloffFunctions = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrBRDFFunctions\";\nconst shader = `#define FRESNEL_MAXIMUM_ON_ROUGH 0.25\n#ifdef MS_BRDF_ENERGY_CONSERVATION\nvec3 getEnergyConservationFactor(const vec3 specularEnvironmentR0,const vec3 environmentBrdf) {return 1.0+specularEnvironmentR0*(1.0/environmentBrdf.y-1.0);}\n#endif\n#ifdef ENVIRONMENTBRDF\nvec3 getBRDFLookup(float NdotV,float perceptualRoughness) {vec2 UV=vec2(NdotV,perceptualRoughness);vec4 brdfLookup=texture2D(environmentBrdfSampler,UV);\n#ifdef ENVIRONMENTBRDF_RGBD\nbrdfLookup.rgb=fromRGBD(brdfLookup.rgba);\n#endif\nreturn brdfLookup.rgb;}\nvec3 getReflectanceFromBRDFLookup(const vec3 specularEnvironmentR0,const vec3 specularEnvironmentR90,const vec3 environmentBrdf) {\n#ifdef BRDF_V_HEIGHT_CORRELATED\nvec3 reflectance=(specularEnvironmentR90-specularEnvironmentR0)*environmentBrdf.x+specularEnvironmentR0*environmentBrdf.y;\n#else\nvec3 reflectance=specularEnvironmentR0*environmentBrdf.x+specularEnvironmentR90*environmentBrdf.y;\n#endif\nreturn reflectance;}\nvec3 getReflectanceFromBRDFLookup(const vec3 specularEnvironmentR0,const vec3 environmentBrdf) {\n#ifdef BRDF_V_HEIGHT_CORRELATED\nvec3 reflectance=mix(environmentBrdf.xxx,environmentBrdf.yyy,specularEnvironmentR0);\n#else\nvec3 reflectance=specularEnvironmentR0*environmentBrdf.x+environmentBrdf.y;\n#endif\nreturn reflectance;}\n#endif\n/* NOT USED\n#if defined(SHEEN) && defined(SHEEN_SOFTER)\nfloat getBRDFLookupCharlieSheen(float NdotV,float perceptualRoughness)\n{float c=1.0-NdotV;float c3=c*c*c;return 0.65584461*c3+1.0/(4.16526551+exp(-7.97291361*perceptualRoughness+6.33516894));}\n#endif\n*/\n#if !defined(ENVIRONMENTBRDF) || defined(REFLECTIONMAP_SKYBOX) || defined(ALPHAFRESNEL)\nvec3 getReflectanceFromAnalyticalBRDFLookup_Jones(float VdotN,vec3 reflectance0,vec3 reflectance90,float smoothness)\n{float weight=mix(FRESNEL_MAXIMUM_ON_ROUGH,1.0,smoothness);return reflectance0+weight*(reflectance90-reflectance0)*pow5(saturate(1.0-VdotN));}\n#endif\n#if defined(SHEEN) && defined(ENVIRONMENTBRDF)\n/**\n* The sheen BRDF not containing F can be easily stored in the blue channel of the BRDF texture.\n* The blue channel contains DCharlie*VAshikhmin*NdotL as a lokkup table\n*/\nvec3 getSheenReflectanceFromBRDFLookup(const vec3 reflectance0,const vec3 environmentBrdf) {vec3 sheenEnvironmentReflectance=reflectance0*environmentBrdf.b;return sheenEnvironmentReflectance;}\n#endif\nvec3 fresnelSchlickGGX(float VdotH,vec3 reflectance0,vec3 reflectance90)\n{return reflectance0+(reflectance90-reflectance0)*pow5(1.0-VdotH);}\nfloat fresnelSchlickGGX(float VdotH,float reflectance0,float reflectance90)\n{return reflectance0+(reflectance90-reflectance0)*pow5(1.0-VdotH);}\n#ifdef CLEARCOAT\nvec3 getR0RemappedForClearCoat(vec3 f0) {\n#ifdef CLEARCOAT_DEFAULTIOR\n#ifdef MOBILE\nreturn saturate(f0*(f0*0.526868+0.529324)-0.0482256);\n#else\nreturn saturate(f0*(f0*(0.941892-0.263008*f0)+0.346479)-0.0285998);\n#endif\n#else\nvec3 s=sqrt(f0);vec3 t=(vClearCoatRefractionParams.z+vClearCoatRefractionParams.w*s)/(vClearCoatRefractionParams.w+vClearCoatRefractionParams.z*s);return square(t);\n#endif\n}\n#endif\n#ifdef IRIDESCENCE\nconst mat3 XYZ_TO_REC709=mat3(\n3.2404542,-0.9692660, 0.0556434,\n-1.5371385, 1.8760108,-0.2040259,\n-0.4985314, 0.0415560, 1.0572252\n);vec3 getIORTfromAirToSurfaceR0(vec3 f0) {vec3 sqrtF0=sqrt(f0);return (1.+sqrtF0)/(1.-sqrtF0);}\nvec3 getR0fromIORs(vec3 iorT,float iorI) {return square((iorT-vec3(iorI))/(iorT+vec3(iorI)));}\nfloat getR0fromIORs(float iorT,float iorI) {return square((iorT-iorI)/(iorT+iorI));}\nvec3 evalSensitivity(float opd,vec3 shift) {float phase=2.0*PI*opd*1.0e-9;const vec3 val=vec3(5.4856e-13,4.4201e-13,5.2481e-13);const vec3 pos=vec3(1.6810e+06,1.7953e+06,2.2084e+06);const vec3 var=vec3(4.3278e+09,9.3046e+09,6.6121e+09);vec3 xyz=val*sqrt(2.0*PI*var)*cos(pos*phase+shift)*exp(-square(phase)*var);xyz.x+=9.7470e-14*sqrt(2.0*PI*4.5282e+09)*cos(2.2399e+06*phase+shift[0])*exp(-4.5282e+09*square(phase));xyz/=1.0685e-7;vec3 srgb=XYZ_TO_REC709*xyz;return srgb;}\nvec3 evalIridescence(float outsideIOR,float eta2,float cosTheta1,float thinFilmThickness,vec3 baseF0) {vec3 I=vec3(1.0);float iridescenceIOR=mix(outsideIOR,eta2,smoothstep(0.0,0.03,thinFilmThickness));float sinTheta2Sq=square(outsideIOR/iridescenceIOR)*(1.0-square(cosTheta1));float cosTheta2Sq=1.0-sinTheta2Sq;if (cosTheta2Sq<0.0) {return I;}\nfloat cosTheta2=sqrt(cosTheta2Sq);float R0=getR0fromIORs(iridescenceIOR,outsideIOR);float R12=fresnelSchlickGGX(cosTheta1,R0,1.);float R21=R12;float T121=1.0-R12;float phi12=0.0;if (iridescenceIOR0\n#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE)\nfloat radicalInverse_VdC(uint bits) \n{bits=(bits<<16u) | (bits>>16u);bits=((bits & 0x55555555u)<<1u) | ((bits & 0xAAAAAAAAu)>>1u);bits=((bits & 0x33333333u)<<2u) | ((bits & 0xCCCCCCCCu)>>2u);bits=((bits & 0x0F0F0F0Fu)<<4u) | ((bits & 0xF0F0F0F0u)>>4u);bits=((bits & 0x00FF00FFu)<<8u) | ((bits & 0xFF00FF00u)>>8u);return float(bits)*2.3283064365386963e-10; }\nvec2 hammersley(uint i,uint N)\n{return vec2(float(i)/float(N),radicalInverse_VdC(i));}\n#else\nfloat vanDerCorpus(int n,int base)\n{float invBase=1.0/float(base);float denom =1.0;float result =0.0;for(int i=0; i<32; ++i)\n{if(n>0)\n{denom =mod(float(n),2.0);result+=denom*invBase;invBase=invBase/2.0;n =int(float(n)/2.0);}}\nreturn result;}\nvec2 hammersley(int i,int N)\n{return vec2(float(i)/float(N),vanDerCorpus(i,2));}\n#endif\nfloat log4(float x) {return log2(x)/2.;}\nconst float NUM_SAMPLES_FLOAT=float(NUM_SAMPLES);const float NUM_SAMPLES_FLOAT_INVERSED=1./NUM_SAMPLES_FLOAT;const float K=4.;\n#define inline\nvec3 irradiance(samplerCube inputTexture,vec3 inputN,vec2 filteringInfo)\n{vec3 n=normalize(inputN);vec3 result=vec3(0.0);vec3 tangent=abs(n.z)<0.999 ? vec3(0.,0.,1.) : vec3(1.,0.,0.);tangent=normalize(cross(tangent,n));vec3 bitangent=cross(n,tangent);mat3 tbn=mat3(tangent,bitangent,n);float maxLevel=filteringInfo.y;float dim0=filteringInfo.x;float omegaP=(4.*PI)/(6.*dim0*dim0);\n#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE)\nfor(uint i=0u; i0.) {float pdf_inversed=PI/NoL;float omegaS=NUM_SAMPLES_FLOAT_INVERSED*pdf_inversed;float l=log4(omegaS)-log4(omegaP)+log4(K);float mipLevel=clamp(l,0.0,maxLevel);vec3 c=textureCubeLodEXT(inputTexture,tbn*Ls,mipLevel).rgb;\n#ifdef GAMMA_INPUT\nc=toLinearSpace(c);\n#endif\nresult+=c;}}\nresult=result*NUM_SAMPLES_FLOAT_INVERSED;return result;}\n#define inline\nvec3 radiance(float alphaG,samplerCube inputTexture,vec3 inputN,vec2 filteringInfo)\n{vec3 n=normalize(inputN);if (alphaG==0.) {vec3 c=textureCube(inputTexture,n).rgb;\n#ifdef GAMMA_INPUT\nc=toLinearSpace(c);\n#endif\nreturn c;} else {vec3 result=vec3(0.);vec3 tangent=abs(n.z)<0.999 ? vec3(0.,0.,1.) : vec3(1.,0.,0.);tangent=normalize(cross(tangent,n));vec3 bitangent=cross(n,tangent);mat3 tbn=mat3(tangent,bitangent,n);float maxLevel=filteringInfo.y;float dim0=filteringInfo.x;float omegaP=(4.*PI)/(6.*dim0*dim0);float weight=0.;\n#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE)\nfor(uint i=0u; i0.) {float pdf_inversed=4./normalDistributionFunction_TrowbridgeReitzGGX(NoH,alphaG);float omegaS=NUM_SAMPLES_FLOAT_INVERSED*pdf_inversed;float l=log4(omegaS)-log4(omegaP)+log4(K);float mipLevel=clamp(float(l),0.0,maxLevel);weight+=NoL;vec3 c=textureCubeLodEXT(inputTexture,tbn*L,mipLevel).rgb;\n#ifdef GAMMA_INPUT\nc=toLinearSpace(c);\n#endif\nresult+=c*NoL;}}\nresult=result/weight;return result;}}\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const hdrFilteringFunctions = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrDirectLightingFunctions\";\nconst shader = `#define CLEARCOATREFLECTANCE90 1.0\nstruct lightingInfo\n{vec3 diffuse;\n#ifdef SPECULARTERM\nvec3 specular;\n#endif\n#ifdef CLEARCOAT\nvec4 clearCoat;\n#endif\n#ifdef SHEEN\nvec3 sheen;\n#endif\n};float adjustRoughnessFromLightProperties(float roughness,float lightRadius,float lightDistance) {\n#if defined(USEPHYSICALLIGHTFALLOFF) || defined(USEGLTFLIGHTFALLOFF)\nfloat lightRoughness=lightRadius/lightDistance;float totalRoughness=saturate(lightRoughness+roughness);return totalRoughness;\n#else\nreturn roughness;\n#endif\n}\nvec3 computeHemisphericDiffuseLighting(preLightingInfo info,vec3 lightColor,vec3 groundColor) {return mix(groundColor,lightColor,info.NdotL);}\nvec3 computeDiffuseLighting(preLightingInfo info,vec3 lightColor) {float diffuseTerm=diffuseBRDF_Burley(info.NdotL,info.NdotV,info.VdotH,info.roughness);return diffuseTerm*info.attenuation*info.NdotL*lightColor;}\n#define inline\nvec3 computeProjectionTextureDiffuseLighting(sampler2D projectionLightSampler,mat4 textureProjectionMatrix){vec4 strq=textureProjectionMatrix*vec4(vPositionW,1.0);strq/=strq.w;vec3 textureColor=texture2D(projectionLightSampler,strq.xy).rgb;return toLinearSpace(textureColor);}\n#ifdef SS_TRANSLUCENCY\nvec3 computeDiffuseAndTransmittedLighting(preLightingInfo info,vec3 lightColor,vec3 transmittance) {float NdotL=absEps(info.NdotLUnclamped);float wrapNdotL=computeWrappedDiffuseNdotL(NdotL,0.02);float trAdapt=step(0.,info.NdotLUnclamped);vec3 transmittanceNdotL=mix(transmittance*wrapNdotL,vec3(wrapNdotL),trAdapt);float diffuseTerm=diffuseBRDF_Burley(NdotL,info.NdotV,info.VdotH,info.roughness);return diffuseTerm*transmittanceNdotL*info.attenuation*lightColor;}\n#endif\n#ifdef SPECULARTERM\nvec3 computeSpecularLighting(preLightingInfo info,vec3 N,vec3 reflectance0,vec3 reflectance90,float geometricRoughnessFactor,vec3 lightColor) {float NdotH=saturateEps(dot(N,info.H));float roughness=max(info.roughness,geometricRoughnessFactor);float alphaG=convertRoughnessToAverageSlope(roughness);vec3 fresnel=fresnelSchlickGGX(info.VdotH,reflectance0,reflectance90);\n#ifdef IRIDESCENCE\nfresnel=mix(fresnel,reflectance0,info.iridescenceIntensity);\n#endif\nfloat distribution=normalDistributionFunction_TrowbridgeReitzGGX(NdotH,alphaG);\n#ifdef BRDF_V_HEIGHT_CORRELATED\nfloat smithVisibility=smithVisibility_GGXCorrelated(info.NdotL,info.NdotV,alphaG);\n#else\nfloat smithVisibility=smithVisibility_TrowbridgeReitzGGXFast(info.NdotL,info.NdotV,alphaG);\n#endif\nvec3 specTerm=fresnel*distribution*smithVisibility;return specTerm*info.attenuation*info.NdotL*lightColor;}\n#endif\n#ifdef ANISOTROPIC\nvec3 computeAnisotropicSpecularLighting(preLightingInfo info,vec3 V,vec3 N,vec3 T,vec3 B,float anisotropy,vec3 reflectance0,vec3 reflectance90,float geometricRoughnessFactor,vec3 lightColor) {float NdotH=saturateEps(dot(N,info.H));float TdotH=dot(T,info.H);float BdotH=dot(B,info.H);float TdotV=dot(T,V);float BdotV=dot(B,V);float TdotL=dot(T,info.L);float BdotL=dot(B,info.L);float alphaG=convertRoughnessToAverageSlope(info.roughness);vec2 alphaTB=getAnisotropicRoughness(alphaG,anisotropy);alphaTB=max(alphaTB,square(geometricRoughnessFactor));vec3 fresnel=fresnelSchlickGGX(info.VdotH,reflectance0,reflectance90);\n#ifdef IRIDESCENCE\nfresnel=mix(fresnel,reflectance0,info.iridescenceIntensity);\n#endif\nfloat distribution=normalDistributionFunction_BurleyGGX_Anisotropic(NdotH,TdotH,BdotH,alphaTB);float smithVisibility=smithVisibility_GGXCorrelated_Anisotropic(info.NdotL,info.NdotV,TdotV,BdotV,TdotL,BdotL,alphaTB);vec3 specTerm=fresnel*distribution*smithVisibility;return specTerm*info.attenuation*info.NdotL*lightColor;}\n#endif\n#ifdef CLEARCOAT\nvec4 computeClearCoatLighting(preLightingInfo info,vec3 Ncc,float geometricRoughnessFactor,float clearCoatIntensity,vec3 lightColor) {float NccdotL=saturateEps(dot(Ncc,info.L));float NccdotH=saturateEps(dot(Ncc,info.H));float clearCoatRoughness=max(info.roughness,geometricRoughnessFactor);float alphaG=convertRoughnessToAverageSlope(clearCoatRoughness);float fresnel=fresnelSchlickGGX(info.VdotH,vClearCoatRefractionParams.x,CLEARCOATREFLECTANCE90);fresnel*=clearCoatIntensity;float distribution=normalDistributionFunction_TrowbridgeReitzGGX(NccdotH,alphaG);float kelemenVisibility=visibility_Kelemen(info.VdotH);float clearCoatTerm=fresnel*distribution*kelemenVisibility;return vec4(\nclearCoatTerm*info.attenuation*NccdotL*lightColor,\n1.0-fresnel\n);}\nvec3 computeClearCoatLightingAbsorption(float NdotVRefract,vec3 L,vec3 Ncc,vec3 clearCoatColor,float clearCoatThickness,float clearCoatIntensity) {vec3 LRefract=-refract(L,Ncc,vClearCoatRefractionParams.y);float NdotLRefract=saturateEps(dot(Ncc,LRefract));vec3 absorption=computeClearCoatAbsorption(NdotVRefract,NdotLRefract,clearCoatColor,clearCoatThickness,clearCoatIntensity);return absorption;}\n#endif\n#ifdef SHEEN\nvec3 computeSheenLighting(preLightingInfo info,vec3 N,vec3 reflectance0,vec3 reflectance90,float geometricRoughnessFactor,vec3 lightColor) {float NdotH=saturateEps(dot(N,info.H));float roughness=max(info.roughness,geometricRoughnessFactor);float alphaG=convertRoughnessToAverageSlope(roughness);float fresnel=1.;float distribution=normalDistributionFunction_CharlieSheen(NdotH,alphaG);/*#ifdef SHEEN_SOFTER\nfloat visibility=visibility_CharlieSheen(info.NdotL,info.NdotV,alphaG);\n#else */\nfloat visibility=visibility_Ashikhmin(info.NdotL,info.NdotV);/* #endif */\nfloat sheenTerm=fresnel*distribution*visibility;return sheenTerm*info.attenuation*info.NdotL*lightColor;}\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrDirectLightingFunctions = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrIBLFunctions\";\nconst shader = `#if defined(REFLECTION) || defined(SS_REFRACTION)\nfloat getLodFromAlphaG(float cubeMapDimensionPixels,float microsurfaceAverageSlope) {float microsurfaceAverageSlopeTexels=cubeMapDimensionPixels*microsurfaceAverageSlope;float lod=log2(microsurfaceAverageSlopeTexels);return lod;}\nfloat getLinearLodFromRoughness(float cubeMapDimensionPixels,float roughness) {float lod=log2(cubeMapDimensionPixels)*roughness;return lod;}\n#endif\n#if defined(ENVIRONMENTBRDF) && defined(RADIANCEOCCLUSION)\nfloat environmentRadianceOcclusion(float ambientOcclusion,float NdotVUnclamped) {float temp=NdotVUnclamped+ambientOcclusion;return saturate(square(temp)-1.0+ambientOcclusion);}\n#endif\n#if defined(ENVIRONMENTBRDF) && defined(HORIZONOCCLUSION)\nfloat environmentHorizonOcclusion(vec3 view,vec3 normal,vec3 geometricNormal) {vec3 reflection=reflect(view,normal);float temp=saturate(1.0+1.1*dot(reflection,geometricNormal));return square(temp);}\n#endif\n#if defined(LODINREFLECTIONALPHA) || defined(SS_LODINREFRACTIONALPHA)\n#define UNPACK_LOD(x) (1.0-x)*255.0\nfloat getLodFromAlphaG(float cubeMapDimensionPixels,float alphaG,float NdotV) {float microsurfaceAverageSlope=alphaG;microsurfaceAverageSlope*=sqrt(abs(NdotV));return getLodFromAlphaG(cubeMapDimensionPixels,microsurfaceAverageSlope);}\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrIBLFunctions = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\nimport \"./decalFragment\";\n\nconst name = \"pbrBlockAlbedoOpacity\";\nconst shader = `struct albedoOpacityOutParams\n{vec3 surfaceAlbedo;float alpha;};\n#define pbr_inline\nvoid albedoOpacityBlock(\nin vec4 vAlbedoColor,\n#ifdef ALBEDO\nin vec4 albedoTexture,\nin vec2 albedoInfos,\n#endif\n#ifdef OPACITY\nin vec4 opacityMap,\nin vec2 vOpacityInfos,\n#endif\n#ifdef DETAIL\nin vec4 detailColor,\nin vec4 vDetailInfos,\n#endif\n#ifdef DECAL\nin vec4 decalColor,\nin vec4 vDecalInfos,\n#endif\nout albedoOpacityOutParams outParams\n)\n{vec3 surfaceAlbedo=vAlbedoColor.rgb;float alpha=vAlbedoColor.a;\n#ifdef ALBEDO\n#if defined(ALPHAFROMALBEDO) || defined(ALPHATEST)\nalpha*=albedoTexture.a;\n#endif\n#ifdef GAMMAALBEDO\nsurfaceAlbedo*=toLinearSpace(albedoTexture.rgb);\n#else\nsurfaceAlbedo*=albedoTexture.rgb;\n#endif\nsurfaceAlbedo*=albedoInfos.y;\n#endif\n#ifndef DECAL_AFTER_DETAIL\n#include\n#endif\n#if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES)\nsurfaceAlbedo*=vColor.rgb;\n#endif\n#ifdef DETAIL\nfloat detailAlbedo=2.0*mix(0.5,detailColor.r,vDetailInfos.y);surfaceAlbedo.rgb=surfaceAlbedo.rgb*detailAlbedo*detailAlbedo; \n#endif\n#ifdef DECAL_AFTER_DETAIL\n#include\n#endif\n#define CUSTOM_FRAGMENT_UPDATE_ALBEDO\n#ifdef OPACITY\n#ifdef OPACITYRGB\nalpha=getLuminance(opacityMap.rgb);\n#else\nalpha*=opacityMap.a;\n#endif\nalpha*=vOpacityInfos.y;\n#endif\n#if defined(VERTEXALPHA) || defined(INSTANCESCOLOR) && defined(INSTANCES)\nalpha*=vColor.a;\n#endif\n#if !defined(SS_LINKREFRACTIONTOTRANSPARENCY) && !defined(ALPHAFRESNEL)\n#ifdef ALPHATEST \n#if DEBUGMODE != 88\nif (alpha0\n#ifdef METALLICWORKFLOW\nvec2 metallicRoughness;\n#ifdef REFLECTIVITY\nvec4 surfaceMetallicColorMap;\n#endif\n#ifndef FROSTBITE_REFLECTANCE\nvec3 metallicF0;\n#endif\n#else\n#ifdef REFLECTIVITY\nvec4 surfaceReflectivityColorMap;\n#endif\n#endif\n#endif\n};\n#define pbr_inline\nvoid reflectivityBlock(\nin vec4 vReflectivityColor,\n#ifdef METALLICWORKFLOW\nin vec3 surfaceAlbedo,\nin vec4 metallicReflectanceFactors,\n#endif\n#ifdef REFLECTIVITY\nin vec3 reflectivityInfos,\nin vec4 surfaceMetallicOrReflectivityColorMap,\n#endif\n#if defined(METALLICWORKFLOW) && defined(REFLECTIVITY) && defined(AOSTOREINMETALMAPRED)\nin vec3 ambientOcclusionColorIn,\n#endif\n#ifdef MICROSURFACEMAP\nin vec4 microSurfaceTexel,\n#endif\n#ifdef DETAIL\nin vec4 detailColor,\nin vec4 vDetailInfos,\n#endif\nout reflectivityOutParams outParams\n)\n{float microSurface=vReflectivityColor.a;vec3 surfaceReflectivityColor=vReflectivityColor.rgb;\n#ifdef METALLICWORKFLOW\nvec2 metallicRoughness=surfaceReflectivityColor.rg;\n#ifdef REFLECTIVITY\n#if DEBUGMODE>0\noutParams.surfaceMetallicColorMap=surfaceMetallicOrReflectivityColorMap;\n#endif\n#ifdef AOSTOREINMETALMAPRED\nvec3 aoStoreInMetalMap=vec3(surfaceMetallicOrReflectivityColorMap.r,surfaceMetallicOrReflectivityColorMap.r,surfaceMetallicOrReflectivityColorMap.r);outParams.ambientOcclusionColor=mix(ambientOcclusionColorIn,aoStoreInMetalMap,reflectivityInfos.z);\n#endif\n#ifdef METALLNESSSTOREINMETALMAPBLUE\nmetallicRoughness.r*=surfaceMetallicOrReflectivityColorMap.b;\n#else\nmetallicRoughness.r*=surfaceMetallicOrReflectivityColorMap.r;\n#endif\n#ifdef ROUGHNESSSTOREINMETALMAPALPHA\nmetallicRoughness.g*=surfaceMetallicOrReflectivityColorMap.a;\n#else\n#ifdef ROUGHNESSSTOREINMETALMAPGREEN\nmetallicRoughness.g*=surfaceMetallicOrReflectivityColorMap.g;\n#endif\n#endif\n#endif\n#ifdef DETAIL\nfloat detailRoughness=mix(0.5,detailColor.b,vDetailInfos.w);float loLerp=mix(0.,metallicRoughness.g,detailRoughness*2.);float hiLerp=mix(metallicRoughness.g,1.,(detailRoughness-0.5)*2.);metallicRoughness.g=mix(loLerp,hiLerp,step(detailRoughness,0.5));\n#endif\n#ifdef MICROSURFACEMAP\nmetallicRoughness.g*=microSurfaceTexel.r;\n#endif\n#if DEBUGMODE>0\noutParams.metallicRoughness=metallicRoughness;\n#endif\n#define CUSTOM_FRAGMENT_UPDATE_METALLICROUGHNESS\nmicroSurface=1.0-metallicRoughness.g;vec3 baseColor=surfaceAlbedo;\n#ifdef FROSTBITE_REFLECTANCE\noutParams.surfaceAlbedo=baseColor.rgb*(1.0-metallicRoughness.r);surfaceReflectivityColor=mix(0.16*reflectance*reflectance,baseColor,metallicRoughness.r);\n#else\nvec3 metallicF0=metallicReflectanceFactors.rgb;\n#if DEBUGMODE>0\noutParams.metallicF0=metallicF0;\n#endif\noutParams.surfaceAlbedo=mix(baseColor.rgb*(1.0-metallicF0),vec3(0.,0.,0.),metallicRoughness.r);surfaceReflectivityColor=mix(metallicF0,baseColor,metallicRoughness.r);\n#endif\n#else\n#ifdef REFLECTIVITY\nsurfaceReflectivityColor*=surfaceMetallicOrReflectivityColorMap.rgb;\n#if DEBUGMODE>0\noutParams.surfaceReflectivityColorMap=surfaceMetallicOrReflectivityColorMap;\n#endif\n#ifdef MICROSURFACEFROMREFLECTIVITYMAP\nmicroSurface*=surfaceMetallicOrReflectivityColorMap.a;microSurface*=reflectivityInfos.z;\n#else\n#ifdef MICROSURFACEAUTOMATIC\nmicroSurface*=computeDefaultMicroSurface(microSurface,surfaceReflectivityColor);\n#endif\n#ifdef MICROSURFACEMAP\nmicroSurface*=microSurfaceTexel.r;\n#endif\n#define CUSTOM_FRAGMENT_UPDATE_MICROSURFACE\n#endif\n#endif\n#endif\nmicroSurface=saturate(microSurface);float roughness=1.-microSurface;outParams.microSurface=microSurface;outParams.roughness=roughness;outParams.surfaceReflectivityColor=surfaceReflectivityColor;}\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrBlockReflectivity = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrBlockAmbientOcclusion\";\nconst shader = `struct ambientOcclusionOutParams\n{vec3 ambientOcclusionColor;\n#if DEBUGMODE>0 && defined(AMBIENT)\nvec3 ambientOcclusionColorMap;\n#endif\n};\n#define pbr_inline\nvoid ambientOcclusionBlock(\n#ifdef AMBIENT\nin vec3 ambientOcclusionColorMap_,\nin vec4 vAmbientInfos,\n#endif\nout ambientOcclusionOutParams outParams\n)\n{vec3 ambientOcclusionColor=vec3(1.,1.,1.);\n#ifdef AMBIENT\nvec3 ambientOcclusionColorMap=ambientOcclusionColorMap_*vAmbientInfos.y;\n#ifdef AMBIENTINGRAYSCALE\nambientOcclusionColorMap=vec3(ambientOcclusionColorMap.r,ambientOcclusionColorMap.r,ambientOcclusionColorMap.r);\n#endif\nambientOcclusionColor=mix(ambientOcclusionColor,ambientOcclusionColorMap,vAmbientInfos.z);\n#if DEBUGMODE>0\noutParams.ambientOcclusionColorMap=ambientOcclusionColorMap;\n#endif\n#endif\noutParams.ambientOcclusionColor=ambientOcclusionColor;}\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrBlockAmbientOcclusion = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrBlockAlphaFresnel\";\nconst shader = `#ifdef ALPHAFRESNEL\n#if defined(ALPHATEST) || defined(ALPHABLEND)\nstruct alphaFresnelOutParams\n{float alpha;};\n#define pbr_inline\nvoid alphaFresnelBlock(\nin vec3 normalW,\nin vec3 viewDirectionW,\nin float alpha,\nin float microSurface,\nout alphaFresnelOutParams outParams\n)\n{float opacityPerceptual=alpha;\n#ifdef LINEARALPHAFRESNEL\nfloat opacity0=opacityPerceptual;\n#else\nfloat opacity0=opacityPerceptual*opacityPerceptual;\n#endif\nfloat opacity90=fresnelGrazingReflectance(opacity0);vec3 normalForward=faceforward(normalW,-viewDirectionW,normalW);outParams.alpha=getReflectanceFromAnalyticalBRDFLookup_Jones(saturate(dot(viewDirectionW,normalForward)),vec3(opacity0),vec3(opacity90),sqrt(microSurface)).x;\n#ifdef ALPHATEST\nif (outParams.alpha0 && defined(ANISOTROPIC_TEXTURE)\nvec3 anisotropyMapData;\n#endif\n};\n#define pbr_inline\nvoid anisotropicBlock(\nin vec3 vAnisotropy,\nin float roughness,\n#ifdef ANISOTROPIC_TEXTURE\nin vec3 anisotropyMapData,\n#endif\nin mat3 TBN,\nin vec3 normalW,\nin vec3 viewDirectionW,\nout anisotropicOutParams outParams\n)\n{float anisotropy=vAnisotropy.b;vec3 anisotropyDirection=vec3(vAnisotropy.xy,0.);\n#ifdef ANISOTROPIC_TEXTURE\nanisotropy*=anisotropyMapData.b;\n#if DEBUGMODE>0\noutParams.anisotropyMapData=anisotropyMapData;\n#endif\nanisotropyMapData.rg=anisotropyMapData.rg*2.0-1.0;\n#ifdef ANISOTROPIC_LEGACY\nanisotropyDirection.rg*=anisotropyMapData.rg;\n#else\nanisotropyDirection.xy=mat2(anisotropyDirection.x,anisotropyDirection.y,-anisotropyDirection.y,anisotropyDirection.x)*normalize(anisotropyMapData.rg);\n#endif\n#endif\nmat3 anisoTBN=mat3(normalize(TBN[0]),normalize(TBN[1]),normalize(TBN[2]));vec3 anisotropicTangent=normalize(anisoTBN*anisotropyDirection);vec3 anisotropicBitangent=normalize(cross(anisoTBN[2],anisotropicTangent));outParams.anisotropy=anisotropy;outParams.anisotropicTangent=anisotropicTangent;outParams.anisotropicBitangent=anisotropicBitangent;outParams.anisotropicNormal=getAnisotropicBentNormals(anisotropicTangent,anisotropicBitangent,normalW,viewDirectionW,anisotropy,roughness);}\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrBlockAnisotropic = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrBlockReflection\";\nconst shader = `#ifdef REFLECTION\nstruct reflectionOutParams\n{vec4 environmentRadiance;vec3 environmentIrradiance;\n#ifdef REFLECTIONMAP_3D\nvec3 reflectionCoords;\n#else\nvec2 reflectionCoords;\n#endif\n#ifdef SS_TRANSLUCENCY\n#ifdef USESPHERICALFROMREFLECTIONMAP\n#if !defined(NORMAL) || !defined(USESPHERICALINVERTEX)\nvec3 irradianceVector;\n#endif\n#endif\n#endif\n};\n#define pbr_inline\nvoid createReflectionCoords(\nin vec3 vPositionW,\nin vec3 normalW,\n#ifdef ANISOTROPIC\nin anisotropicOutParams anisotropicOut,\n#endif\n#ifdef REFLECTIONMAP_3D\nout vec3 reflectionCoords\n#else\nout vec2 reflectionCoords\n#endif\n)\n{\n#ifdef ANISOTROPIC\nvec3 reflectionVector=computeReflectionCoords(vec4(vPositionW,1.0),anisotropicOut.anisotropicNormal);\n#else\nvec3 reflectionVector=computeReflectionCoords(vec4(vPositionW,1.0),normalW);\n#endif\n#ifdef REFLECTIONMAP_OPPOSITEZ\nreflectionVector.z*=-1.0;\n#endif\n#ifdef REFLECTIONMAP_3D\nreflectionCoords=reflectionVector;\n#else\nreflectionCoords=reflectionVector.xy;\n#ifdef REFLECTIONMAP_PROJECTION\nreflectionCoords/=reflectionVector.z;\n#endif\nreflectionCoords.y=1.0-reflectionCoords.y;\n#endif\n}\n#define pbr_inline\n#define inline\nvoid sampleReflectionTexture(\nin float alphaG,\nin vec3 vReflectionMicrosurfaceInfos,\nin vec2 vReflectionInfos,\nin vec3 vReflectionColor,\n#if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX)\nin float NdotVUnclamped,\n#endif\n#ifdef LINEARSPECULARREFLECTION\nin float roughness,\n#endif\n#ifdef REFLECTIONMAP_3D\nin samplerCube reflectionSampler,\nconst vec3 reflectionCoords,\n#else\nin sampler2D reflectionSampler,\nconst vec2 reflectionCoords,\n#endif\n#ifndef LODBASEDMICROSFURACE\n#ifdef REFLECTIONMAP_3D\nin samplerCube reflectionSamplerLow,\nin samplerCube reflectionSamplerHigh,\n#else\nin sampler2D reflectionSamplerLow,\nin sampler2D reflectionSamplerHigh,\n#endif\n#endif\n#ifdef REALTIME_FILTERING\nin vec2 vReflectionFilteringInfo,\n#endif\nout vec4 environmentRadiance\n)\n{\n#if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX)\nfloat reflectionLOD=getLodFromAlphaG(vReflectionMicrosurfaceInfos.x,alphaG,NdotVUnclamped);\n#elif defined(LINEARSPECULARREFLECTION)\nfloat reflectionLOD=getLinearLodFromRoughness(vReflectionMicrosurfaceInfos.x,roughness);\n#else\nfloat reflectionLOD=getLodFromAlphaG(vReflectionMicrosurfaceInfos.x,alphaG);\n#endif\n#ifdef LODBASEDMICROSFURACE\nreflectionLOD=reflectionLOD*vReflectionMicrosurfaceInfos.y+vReflectionMicrosurfaceInfos.z;\n#ifdef LODINREFLECTIONALPHA\nfloat automaticReflectionLOD=UNPACK_LOD(sampleReflection(reflectionSampler,reflectionCoords).a);float requestedReflectionLOD=max(automaticReflectionLOD,reflectionLOD);\n#else\nfloat requestedReflectionLOD=reflectionLOD;\n#endif\n#ifdef REALTIME_FILTERING\nenvironmentRadiance=vec4(radiance(alphaG,reflectionSampler,reflectionCoords,vReflectionFilteringInfo),1.0);\n#else\nenvironmentRadiance=sampleReflectionLod(reflectionSampler,reflectionCoords,reflectionLOD);\n#endif\n#else\nfloat lodReflectionNormalized=saturate(reflectionLOD/log2(vReflectionMicrosurfaceInfos.x));float lodReflectionNormalizedDoubled=lodReflectionNormalized*2.0;vec4 environmentMid=sampleReflection(reflectionSampler,reflectionCoords);if (lodReflectionNormalizedDoubled<1.0){environmentRadiance=mix(\nsampleReflection(reflectionSamplerHigh,reflectionCoords),\nenvironmentMid,\nlodReflectionNormalizedDoubled\n);} else {environmentRadiance=mix(\nenvironmentMid,\nsampleReflection(reflectionSamplerLow,reflectionCoords),\nlodReflectionNormalizedDoubled-1.0\n);}\n#endif\n#ifdef RGBDREFLECTION\nenvironmentRadiance.rgb=fromRGBD(environmentRadiance);\n#endif\n#ifdef GAMMAREFLECTION\nenvironmentRadiance.rgb=toLinearSpace(environmentRadiance.rgb);\n#endif\nenvironmentRadiance.rgb*=vReflectionInfos.x;environmentRadiance.rgb*=vReflectionColor.rgb;}\n#define pbr_inline\n#define inline\nvoid reflectionBlock(\nin vec3 vPositionW,\nin vec3 normalW,\nin float alphaG,\nin vec3 vReflectionMicrosurfaceInfos,\nin vec2 vReflectionInfos,\nin vec3 vReflectionColor,\n#ifdef ANISOTROPIC\nin anisotropicOutParams anisotropicOut,\n#endif\n#if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX)\nin float NdotVUnclamped,\n#endif\n#ifdef LINEARSPECULARREFLECTION\nin float roughness,\n#endif\n#ifdef REFLECTIONMAP_3D\nin samplerCube reflectionSampler,\n#else\nin sampler2D reflectionSampler,\n#endif\n#if defined(NORMAL) && defined(USESPHERICALINVERTEX)\nin vec3 vEnvironmentIrradiance,\n#endif\n#ifdef USESPHERICALFROMREFLECTIONMAP\n#if !defined(NORMAL) || !defined(USESPHERICALINVERTEX)\nin mat4 reflectionMatrix,\n#endif\n#endif\n#ifdef USEIRRADIANCEMAP\n#ifdef REFLECTIONMAP_3D\nin samplerCube irradianceSampler,\n#else\nin sampler2D irradianceSampler,\n#endif\n#endif\n#ifndef LODBASEDMICROSFURACE\n#ifdef REFLECTIONMAP_3D\nin samplerCube reflectionSamplerLow,\nin samplerCube reflectionSamplerHigh,\n#else\nin sampler2D reflectionSamplerLow,\nin sampler2D reflectionSamplerHigh,\n#endif\n#endif\n#ifdef REALTIME_FILTERING\nin vec2 vReflectionFilteringInfo,\n#endif\nout reflectionOutParams outParams\n)\n{vec4 environmentRadiance=vec4(0.,0.,0.,0.);\n#ifdef REFLECTIONMAP_3D\nvec3 reflectionCoords=vec3(0.);\n#else\nvec2 reflectionCoords=vec2(0.);\n#endif\ncreateReflectionCoords(\nvPositionW,\nnormalW,\n#ifdef ANISOTROPIC\nanisotropicOut,\n#endif\nreflectionCoords\n);sampleReflectionTexture(\nalphaG,\nvReflectionMicrosurfaceInfos,\nvReflectionInfos,\nvReflectionColor,\n#if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX)\nNdotVUnclamped,\n#endif\n#ifdef LINEARSPECULARREFLECTION\nroughness,\n#endif\n#ifdef REFLECTIONMAP_3D\nreflectionSampler,\nreflectionCoords,\n#else\nreflectionSampler,\nreflectionCoords,\n#endif\n#ifndef LODBASEDMICROSFURACE\nreflectionSamplerLow,\nreflectionSamplerHigh,\n#endif\n#ifdef REALTIME_FILTERING\nvReflectionFilteringInfo,\n#endif\nenvironmentRadiance\n);vec3 environmentIrradiance=vec3(0.,0.,0.);\n#ifdef USESPHERICALFROMREFLECTIONMAP\n#if defined(NORMAL) && defined(USESPHERICALINVERTEX)\nenvironmentIrradiance=vEnvironmentIrradiance;\n#else\n#ifdef ANISOTROPIC\nvec3 irradianceVector=vec3(reflectionMatrix*vec4(anisotropicOut.anisotropicNormal,0)).xyz;\n#else\nvec3 irradianceVector=vec3(reflectionMatrix*vec4(normalW,0)).xyz;\n#endif\n#ifdef REFLECTIONMAP_OPPOSITEZ\nirradianceVector.z*=-1.0;\n#endif\n#ifdef INVERTCUBICMAP\nirradianceVector.y*=-1.0;\n#endif\n#if defined(REALTIME_FILTERING)\nenvironmentIrradiance=irradiance(reflectionSampler,irradianceVector,vReflectionFilteringInfo);\n#else\nenvironmentIrradiance=computeEnvironmentIrradiance(irradianceVector);\n#endif\n#ifdef SS_TRANSLUCENCY\noutParams.irradianceVector=irradianceVector;\n#endif\n#endif\n#elif defined(USEIRRADIANCEMAP)\nvec4 environmentIrradiance4=sampleReflection(irradianceSampler,reflectionCoords);environmentIrradiance=environmentIrradiance4.rgb;\n#ifdef RGBDREFLECTION\nenvironmentIrradiance.rgb=fromRGBD(environmentIrradiance4);\n#endif\n#ifdef GAMMAREFLECTION\nenvironmentIrradiance.rgb=toLinearSpace(environmentIrradiance.rgb);\n#endif\n#endif\nenvironmentIrradiance*=vReflectionColor.rgb;outParams.environmentRadiance=environmentRadiance;outParams.environmentIrradiance=environmentIrradiance;outParams.reflectionCoords=reflectionCoords;}\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrBlockReflection = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrBlockSheen\";\nconst shader = `#ifdef SHEEN\nstruct sheenOutParams\n{float sheenIntensity;vec3 sheenColor;float sheenRoughness;\n#ifdef SHEEN_LINKWITHALBEDO\nvec3 surfaceAlbedo;\n#endif\n#if defined(ENVIRONMENTBRDF) && defined(SHEEN_ALBEDOSCALING)\nfloat sheenAlbedoScaling;\n#endif\n#if defined(REFLECTION) && defined(ENVIRONMENTBRDF)\nvec3 finalSheenRadianceScaled;\n#endif\n#if DEBUGMODE>0\n#ifdef SHEEN_TEXTURE\nvec4 sheenMapData;\n#endif\n#if defined(REFLECTION) && defined(ENVIRONMENTBRDF)\nvec3 sheenEnvironmentReflectance;\n#endif\n#endif\n};\n#define pbr_inline\n#define inline\nvoid sheenBlock(\nin vec4 vSheenColor,\n#ifdef SHEEN_ROUGHNESS\nin float vSheenRoughness,\n#if defined(SHEEN_TEXTURE_ROUGHNESS) && !defined(SHEEN_TEXTURE_ROUGHNESS_IDENTICAL) && !defined(SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE)\nin vec4 sheenMapRoughnessData,\n#endif\n#endif\nin float roughness,\n#ifdef SHEEN_TEXTURE\nin vec4 sheenMapData,\nin float sheenMapLevel,\n#endif\nin float reflectance,\n#ifdef SHEEN_LINKWITHALBEDO\nin vec3 baseColor,\nin vec3 surfaceAlbedo,\n#endif\n#ifdef ENVIRONMENTBRDF\nin float NdotV,\nin vec3 environmentBrdf,\n#endif\n#if defined(REFLECTION) && defined(ENVIRONMENTBRDF)\nin vec2 AARoughnessFactors,\nin vec3 vReflectionMicrosurfaceInfos,\nin vec2 vReflectionInfos,\nin vec3 vReflectionColor,\nin vec4 vLightingIntensity,\n#ifdef REFLECTIONMAP_3D\nin samplerCube reflectionSampler,\nin vec3 reflectionCoords,\n#else\nin sampler2D reflectionSampler,\nin vec2 reflectionCoords,\n#endif\nin float NdotVUnclamped,\n#ifndef LODBASEDMICROSFURACE\n#ifdef REFLECTIONMAP_3D\nin samplerCube reflectionSamplerLow,\nin samplerCube reflectionSamplerHigh,\n#else\nin sampler2D reflectionSamplerLow,\nin sampler2D reflectionSamplerHigh,\n#endif\n#endif\n#ifdef REALTIME_FILTERING\nin vec2 vReflectionFilteringInfo,\n#endif\n#if !defined(REFLECTIONMAP_SKYBOX) && defined(RADIANCEOCCLUSION)\nin float seo,\n#endif\n#if !defined(REFLECTIONMAP_SKYBOX) && defined(HORIZONOCCLUSION) && defined(BUMP) && defined(REFLECTIONMAP_3D)\nin float eho,\n#endif\n#endif\nout sheenOutParams outParams\n)\n{float sheenIntensity=vSheenColor.a;\n#ifdef SHEEN_TEXTURE\n#if DEBUGMODE>0\noutParams.sheenMapData=sheenMapData;\n#endif\n#endif\n#ifdef SHEEN_LINKWITHALBEDO\nfloat sheenFactor=pow5(1.0-sheenIntensity);vec3 sheenColor=baseColor.rgb*(1.0-sheenFactor);float sheenRoughness=sheenIntensity;outParams.surfaceAlbedo=surfaceAlbedo*sheenFactor;\n#ifdef SHEEN_TEXTURE\nsheenIntensity*=sheenMapData.a;\n#endif\n#else\nvec3 sheenColor=vSheenColor.rgb;\n#ifdef SHEEN_TEXTURE\n#ifdef SHEEN_GAMMATEXTURE\nsheenColor.rgb*=toLinearSpace(sheenMapData.rgb);\n#else\nsheenColor.rgb*=sheenMapData.rgb;\n#endif\nsheenColor.rgb*=sheenMapLevel;\n#endif\n#ifdef SHEEN_ROUGHNESS\nfloat sheenRoughness=vSheenRoughness;\n#ifdef SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE\n#if defined(SHEEN_TEXTURE)\nsheenRoughness*=sheenMapData.a;\n#endif\n#elif defined(SHEEN_TEXTURE_ROUGHNESS)\n#ifdef SHEEN_TEXTURE_ROUGHNESS_IDENTICAL\nsheenRoughness*=sheenMapData.a;\n#else\nsheenRoughness*=sheenMapRoughnessData.a;\n#endif\n#endif\n#else\nfloat sheenRoughness=roughness;\n#ifdef SHEEN_TEXTURE\nsheenIntensity*=sheenMapData.a;\n#endif\n#endif\n#if !defined(SHEEN_ALBEDOSCALING)\nsheenIntensity*=(1.-reflectance);\n#endif\nsheenColor*=sheenIntensity;\n#endif\n#ifdef ENVIRONMENTBRDF\n/*#ifdef SHEEN_SOFTER\nvec3 environmentSheenBrdf=vec3(0.,0.,getBRDFLookupCharlieSheen(NdotV,sheenRoughness));\n#else*/\n#ifdef SHEEN_ROUGHNESS\nvec3 environmentSheenBrdf=getBRDFLookup(NdotV,sheenRoughness);\n#else\nvec3 environmentSheenBrdf=environmentBrdf;\n#endif\n/*#endif*/\n#endif\n#if defined(REFLECTION) && defined(ENVIRONMENTBRDF)\nfloat sheenAlphaG=convertRoughnessToAverageSlope(sheenRoughness);\n#ifdef SPECULARAA\nsheenAlphaG+=AARoughnessFactors.y;\n#endif\nvec4 environmentSheenRadiance=vec4(0.,0.,0.,0.);sampleReflectionTexture(\nsheenAlphaG,\nvReflectionMicrosurfaceInfos,\nvReflectionInfos,\nvReflectionColor,\n#if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX)\nNdotVUnclamped,\n#endif\n#ifdef LINEARSPECULARREFLECTION\nsheenRoughness,\n#endif\nreflectionSampler,\nreflectionCoords,\n#ifndef LODBASEDMICROSFURACE\nreflectionSamplerLow,\nreflectionSamplerHigh,\n#endif\n#ifdef REALTIME_FILTERING\nvReflectionFilteringInfo,\n#endif\nenvironmentSheenRadiance\n);vec3 sheenEnvironmentReflectance=getSheenReflectanceFromBRDFLookup(sheenColor,environmentSheenBrdf);\n#if !defined(REFLECTIONMAP_SKYBOX) && defined(RADIANCEOCCLUSION)\nsheenEnvironmentReflectance*=seo;\n#endif\n#if !defined(REFLECTIONMAP_SKYBOX) && defined(HORIZONOCCLUSION) && defined(BUMP) && defined(REFLECTIONMAP_3D)\nsheenEnvironmentReflectance*=eho;\n#endif\n#if DEBUGMODE>0\noutParams.sheenEnvironmentReflectance=sheenEnvironmentReflectance;\n#endif\noutParams.finalSheenRadianceScaled=\nenvironmentSheenRadiance.rgb *\nsheenEnvironmentReflectance *\nvLightingIntensity.z;\n#endif\n#if defined(ENVIRONMENTBRDF) && defined(SHEEN_ALBEDOSCALING)\noutParams.sheenAlbedoScaling=1.0-sheenIntensity*max(max(sheenColor.r,sheenColor.g),sheenColor.b)*environmentSheenBrdf.b;\n#endif\noutParams.sheenIntensity=sheenIntensity;outParams.sheenColor=sheenColor;outParams.sheenRoughness=sheenRoughness;}\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrBlockSheen = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrBlockClearcoat\";\nconst shader = `struct clearcoatOutParams\n{vec3 specularEnvironmentR0;float conservationFactor;vec3 clearCoatNormalW;vec2 clearCoatAARoughnessFactors;float clearCoatIntensity;float clearCoatRoughness;\n#ifdef REFLECTION\nvec3 finalClearCoatRadianceScaled;\n#endif\n#ifdef CLEARCOAT_TINT\nvec3 absorption;float clearCoatNdotVRefract;vec3 clearCoatColor;float clearCoatThickness;\n#endif\n#if defined(ENVIRONMENTBRDF) && defined(MS_BRDF_ENERGY_CONSERVATION)\nvec3 energyConservationFactorClearCoat;\n#endif\n#if DEBUGMODE>0\n#ifdef CLEARCOAT_BUMP\nmat3 TBNClearCoat;\n#endif\n#ifdef CLEARCOAT_TEXTURE\nvec2 clearCoatMapData;\n#endif\n#if defined(CLEARCOAT_TINT) && defined(CLEARCOAT_TINT_TEXTURE)\nvec4 clearCoatTintMapData;\n#endif\n#ifdef REFLECTION\nvec4 environmentClearCoatRadiance;vec3 clearCoatEnvironmentReflectance;\n#endif\nfloat clearCoatNdotV;\n#endif\n};\n#ifdef CLEARCOAT\n#define pbr_inline\n#define inline\nvoid clearcoatBlock(\nin vec3 vPositionW,\nin vec3 geometricNormalW,\nin vec3 viewDirectionW,\nin vec2 vClearCoatParams,\n#if defined(CLEARCOAT_TEXTURE_ROUGHNESS) && !defined(CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL) && !defined(CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE)\nin vec4 clearCoatMapRoughnessData,\n#endif\nin vec3 specularEnvironmentR0,\n#ifdef CLEARCOAT_TEXTURE\nin vec2 clearCoatMapData,\n#endif\n#ifdef CLEARCOAT_TINT\nin vec4 vClearCoatTintParams,\nin float clearCoatColorAtDistance,\nin vec4 vClearCoatRefractionParams,\n#ifdef CLEARCOAT_TINT_TEXTURE\nin vec4 clearCoatTintMapData,\n#endif\n#endif\n#ifdef CLEARCOAT_BUMP\nin vec2 vClearCoatBumpInfos,\nin vec4 clearCoatBumpMapData,\nin vec2 vClearCoatBumpUV,\n#if defined(TANGENT) && defined(NORMAL)\nin mat3 vTBN,\n#else\nin vec2 vClearCoatTangentSpaceParams,\n#endif\n#ifdef OBJECTSPACE_NORMALMAP\nin mat4 normalMatrix,\n#endif\n#endif\n#if defined(FORCENORMALFORWARD) && defined(NORMAL)\nin vec3 faceNormal,\n#endif\n#ifdef REFLECTION\nin vec3 vReflectionMicrosurfaceInfos,\nin vec2 vReflectionInfos,\nin vec3 vReflectionColor,\nin vec4 vLightingIntensity,\n#ifdef REFLECTIONMAP_3D\nin samplerCube reflectionSampler,\n#else\nin sampler2D reflectionSampler,\n#endif\n#ifndef LODBASEDMICROSFURACE\n#ifdef REFLECTIONMAP_3D\nin samplerCube reflectionSamplerLow,\nin samplerCube reflectionSamplerHigh,\n#else\nin sampler2D reflectionSamplerLow,\nin sampler2D reflectionSamplerHigh,\n#endif\n#endif\n#ifdef REALTIME_FILTERING\nin vec2 vReflectionFilteringInfo,\n#endif\n#endif\n#if defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX)\n#ifdef RADIANCEOCCLUSION\nin float ambientMonochrome,\n#endif\n#endif\n#if defined(CLEARCOAT_BUMP) || defined(TWOSIDEDLIGHTING)\nin float frontFacingMultiplier,\n#endif\nout clearcoatOutParams outParams\n)\n{float clearCoatIntensity=vClearCoatParams.x;float clearCoatRoughness=vClearCoatParams.y;\n#ifdef CLEARCOAT_TEXTURE\nclearCoatIntensity*=clearCoatMapData.x;\n#ifdef CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE\nclearCoatRoughness*=clearCoatMapData.y;\n#endif\n#if DEBUGMODE>0\noutParams.clearCoatMapData=clearCoatMapData;\n#endif\n#endif\n#if defined(CLEARCOAT_TEXTURE_ROUGHNESS) && !defined(CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE)\n#ifdef CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL\nclearCoatRoughness*=clearCoatMapData.y;\n#else\nclearCoatRoughness*=clearCoatMapRoughnessData.y;\n#endif\n#endif\noutParams.clearCoatIntensity=clearCoatIntensity;outParams.clearCoatRoughness=clearCoatRoughness;\n#ifdef CLEARCOAT_TINT\nvec3 clearCoatColor=vClearCoatTintParams.rgb;float clearCoatThickness=vClearCoatTintParams.a;\n#ifdef CLEARCOAT_TINT_TEXTURE\n#ifdef CLEARCOAT_TINT_GAMMATEXTURE\nclearCoatColor*=toLinearSpace(clearCoatTintMapData.rgb);\n#else\nclearCoatColor*=clearCoatTintMapData.rgb;\n#endif\nclearCoatThickness*=clearCoatTintMapData.a;\n#if DEBUGMODE>0\noutParams.clearCoatTintMapData=clearCoatTintMapData;\n#endif\n#endif\noutParams.clearCoatColor=computeColorAtDistanceInMedia(clearCoatColor,clearCoatColorAtDistance);outParams.clearCoatThickness=clearCoatThickness;\n#endif\n#ifdef CLEARCOAT_REMAP_F0\nvec3 specularEnvironmentR0Updated=getR0RemappedForClearCoat(specularEnvironmentR0);\n#else\nvec3 specularEnvironmentR0Updated=specularEnvironmentR0;\n#endif\noutParams.specularEnvironmentR0=mix(specularEnvironmentR0,specularEnvironmentR0Updated,clearCoatIntensity);vec3 clearCoatNormalW=geometricNormalW;\n#ifdef CLEARCOAT_BUMP\n#ifdef NORMALXYSCALE\nfloat clearCoatNormalScale=1.0;\n#else\nfloat clearCoatNormalScale=vClearCoatBumpInfos.y;\n#endif\n#if defined(TANGENT) && defined(NORMAL)\nmat3 TBNClearCoat=vTBN;\n#else\nvec2 TBNClearCoatUV=vClearCoatBumpUV*frontFacingMultiplier;mat3 TBNClearCoat=cotangent_frame(clearCoatNormalW*clearCoatNormalScale,vPositionW,TBNClearCoatUV,vClearCoatTangentSpaceParams);\n#endif\n#if DEBUGMODE>0\noutParams.TBNClearCoat=TBNClearCoat;\n#endif\n#ifdef OBJECTSPACE_NORMALMAP\nclearCoatNormalW=normalize(clearCoatBumpMapData.xyz *2.0-1.0);clearCoatNormalW=normalize(mat3(normalMatrix)*clearCoatNormalW);\n#else\nclearCoatNormalW=perturbNormal(TBNClearCoat,clearCoatBumpMapData.xyz,vClearCoatBumpInfos.y);\n#endif\n#endif\n#if defined(FORCENORMALFORWARD) && defined(NORMAL)\nclearCoatNormalW*=sign(dot(clearCoatNormalW,faceNormal));\n#endif\n#if defined(TWOSIDEDLIGHTING) && defined(NORMAL)\nclearCoatNormalW=clearCoatNormalW*frontFacingMultiplier;\n#endif\noutParams.clearCoatNormalW=clearCoatNormalW;outParams.clearCoatAARoughnessFactors=getAARoughnessFactors(clearCoatNormalW.xyz);float clearCoatNdotVUnclamped=dot(clearCoatNormalW,viewDirectionW);float clearCoatNdotV=absEps(clearCoatNdotVUnclamped);\n#if DEBUGMODE>0\noutParams.clearCoatNdotV=clearCoatNdotV;\n#endif\n#ifdef CLEARCOAT_TINT\nvec3 clearCoatVRefract=refract(-viewDirectionW,clearCoatNormalW,vClearCoatRefractionParams.y);outParams.clearCoatNdotVRefract=absEps(dot(clearCoatNormalW,clearCoatVRefract));\n#endif\n#if defined(ENVIRONMENTBRDF) && (!defined(REFLECTIONMAP_SKYBOX) || defined(MS_BRDF_ENERGY_CONSERVATION))\nvec3 environmentClearCoatBrdf=getBRDFLookup(clearCoatNdotV,clearCoatRoughness);\n#endif\n#if defined(REFLECTION)\nfloat clearCoatAlphaG=convertRoughnessToAverageSlope(clearCoatRoughness);\n#ifdef SPECULARAA\nclearCoatAlphaG+=outParams.clearCoatAARoughnessFactors.y;\n#endif\nvec4 environmentClearCoatRadiance=vec4(0.,0.,0.,0.);vec3 clearCoatReflectionVector=computeReflectionCoords(vec4(vPositionW,1.0),clearCoatNormalW);\n#ifdef REFLECTIONMAP_OPPOSITEZ\nclearCoatReflectionVector.z*=-1.0;\n#endif\n#ifdef REFLECTIONMAP_3D\nvec3 clearCoatReflectionCoords=clearCoatReflectionVector;\n#else\nvec2 clearCoatReflectionCoords=clearCoatReflectionVector.xy;\n#ifdef REFLECTIONMAP_PROJECTION\nclearCoatReflectionCoords/=clearCoatReflectionVector.z;\n#endif\nclearCoatReflectionCoords.y=1.0-clearCoatReflectionCoords.y;\n#endif\nsampleReflectionTexture(\nclearCoatAlphaG,\nvReflectionMicrosurfaceInfos,\nvReflectionInfos,\nvReflectionColor,\n#if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX)\nclearCoatNdotVUnclamped,\n#endif\n#ifdef LINEARSPECULARREFLECTION\nclearCoatRoughness,\n#endif\nreflectionSampler,\nclearCoatReflectionCoords,\n#ifndef LODBASEDMICROSFURACE\nreflectionSamplerLow,\nreflectionSamplerHigh,\n#endif\n#ifdef REALTIME_FILTERING\nvReflectionFilteringInfo,\n#endif\nenvironmentClearCoatRadiance\n);\n#if DEBUGMODE>0\noutParams.environmentClearCoatRadiance=environmentClearCoatRadiance;\n#endif\n#if defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX)\nvec3 clearCoatEnvironmentReflectance=getReflectanceFromBRDFLookup(vec3(vClearCoatRefractionParams.x),environmentClearCoatBrdf);\n#ifdef HORIZONOCCLUSION\n#ifdef BUMP\n#ifdef REFLECTIONMAP_3D\nfloat clearCoatEho=environmentHorizonOcclusion(-viewDirectionW,clearCoatNormalW,geometricNormalW);clearCoatEnvironmentReflectance*=clearCoatEho;\n#endif\n#endif\n#endif\n#else\nvec3 clearCoatEnvironmentReflectance=getReflectanceFromAnalyticalBRDFLookup_Jones(clearCoatNdotV,vec3(1.),vec3(1.),sqrt(1.-clearCoatRoughness));\n#endif\nclearCoatEnvironmentReflectance*=clearCoatIntensity;\n#if DEBUGMODE>0\noutParams.clearCoatEnvironmentReflectance=clearCoatEnvironmentReflectance;\n#endif\noutParams.finalClearCoatRadianceScaled=\nenvironmentClearCoatRadiance.rgb *\nclearCoatEnvironmentReflectance *\nvLightingIntensity.z;\n#endif\n#if defined(CLEARCOAT_TINT)\noutParams.absorption=computeClearCoatAbsorption(outParams.clearCoatNdotVRefract,outParams.clearCoatNdotVRefract,outParams.clearCoatColor,clearCoatThickness,clearCoatIntensity);\n#endif\nfloat fresnelIBLClearCoat=fresnelSchlickGGX(clearCoatNdotV,vClearCoatRefractionParams.x,CLEARCOATREFLECTANCE90);fresnelIBLClearCoat*=clearCoatIntensity;outParams.conservationFactor=(1.-fresnelIBLClearCoat);\n#if defined(ENVIRONMENTBRDF) && defined(MS_BRDF_ENERGY_CONSERVATION)\noutParams.energyConservationFactorClearCoat=getEnergyConservationFactor(outParams.specularEnvironmentR0,environmentClearCoatBrdf);\n#endif\n}\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrBlockClearcoat = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrBlockIridescence\";\nconst shader = `struct iridescenceOutParams\n{float iridescenceIntensity;float iridescenceIOR;float iridescenceThickness;vec3 specularEnvironmentR0;};\n#ifdef IRIDESCENCE\n#define pbr_inline\n#define inline\nvoid iridescenceBlock(\nin vec4 vIridescenceParams,\nin float viewAngle,\nin vec3 specularEnvironmentR0,\n#ifdef IRIDESCENCE_TEXTURE\nin vec2 iridescenceMapData,\n#endif\n#ifdef IRIDESCENCE_THICKNESS_TEXTURE\nin vec2 iridescenceThicknessMapData,\n#endif\n#ifdef CLEARCOAT\nin float NdotVUnclamped,\n#ifdef CLEARCOAT_TEXTURE\nin vec2 clearCoatMapData,\n#endif\n#endif\nout iridescenceOutParams outParams\n)\n{float iridescenceIntensity=vIridescenceParams.x;float iridescenceIOR=vIridescenceParams.y;float iridescenceThicknessMin=vIridescenceParams.z;float iridescenceThicknessMax=vIridescenceParams.w;float iridescenceThicknessWeight=1.;\n#ifdef IRIDESCENCE_TEXTURE\niridescenceIntensity*=iridescenceMapData.x;\n#ifdef IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE\niridescenceThicknessWeight=iridescenceMapData.g;\n#endif\n#endif\n#if defined(IRIDESCENCE_THICKNESS_TEXTURE)\niridescenceThicknessWeight=iridescenceThicknessMapData.g;\n#endif\nfloat iridescenceThickness=mix(iridescenceThicknessMin,iridescenceThicknessMax,iridescenceThicknessWeight);float topIor=1.; \n#ifdef CLEARCOAT\nfloat clearCoatIntensity=vClearCoatParams.x;\n#ifdef CLEARCOAT_TEXTURE\nclearCoatIntensity*=clearCoatMapData.x;\n#endif\ntopIor=mix(1.0,vClearCoatRefractionParams.w-1.,clearCoatIntensity);viewAngle=sqrt(1.0+square(1.0/topIor)*(square(NdotVUnclamped)-1.0));\n#endif\nvec3 iridescenceFresnel=evalIridescence(topIor,iridescenceIOR,viewAngle,iridescenceThickness,specularEnvironmentR0);outParams.specularEnvironmentR0=mix(specularEnvironmentR0,iridescenceFresnel,iridescenceIntensity);outParams.iridescenceIntensity=iridescenceIntensity;outParams.iridescenceThickness=iridescenceThickness;outParams.iridescenceIOR=iridescenceIOR;}\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrBlockIridescence = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrBlockSubSurface\";\nconst shader = `struct subSurfaceOutParams\n{vec3 specularEnvironmentReflectance;\n#ifdef SS_REFRACTION\nvec3 finalRefraction;vec3 surfaceAlbedo;\n#ifdef SS_LINKREFRACTIONTOTRANSPARENCY\nfloat alpha;\n#endif\n#ifdef REFLECTION\nfloat refractionFactorForIrradiance;\n#endif\n#endif\n#ifdef SS_TRANSLUCENCY\nvec3 transmittance;float translucencyIntensity;\n#ifdef REFLECTION\nvec3 refractionIrradiance;\n#endif\n#endif\n#if DEBUGMODE>0\n#ifdef SS_THICKNESSANDMASK_TEXTURE\nvec4 thicknessMap;\n#endif\n#ifdef SS_REFRACTION\nvec4 environmentRefraction;vec3 refractionTransmittance;\n#endif\n#endif\n};\n#ifdef SUBSURFACE\n#define pbr_inline\n#define inline\nvoid subSurfaceBlock(\nin vec3 vSubSurfaceIntensity,\nin vec2 vThicknessParam,\nin vec4 vTintColor,\nin vec3 normalW,\nin vec3 specularEnvironmentReflectance,\n#ifdef SS_THICKNESSANDMASK_TEXTURE\nin vec4 thicknessMap,\n#endif\n#ifdef SS_REFRACTIONINTENSITY_TEXTURE\nin vec4 refractionIntensityMap,\n#endif\n#ifdef SS_TRANSLUCENCYINTENSITY_TEXTURE\nin vec4 translucencyIntensityMap,\n#endif\n#ifdef REFLECTION\n#ifdef SS_TRANSLUCENCY\nin mat4 reflectionMatrix,\n#ifdef USESPHERICALFROMREFLECTIONMAP\n#if !defined(NORMAL) || !defined(USESPHERICALINVERTEX)\nin vec3 irradianceVector_,\n#endif\n#if defined(REALTIME_FILTERING)\nin samplerCube reflectionSampler,\nin vec2 vReflectionFilteringInfo,\n#endif\n#endif\n#ifdef USEIRRADIANCEMAP\n#ifdef REFLECTIONMAP_3D\nin samplerCube irradianceSampler,\n#else\nin sampler2D irradianceSampler,\n#endif\n#endif\n#endif\n#endif\n#if defined(SS_REFRACTION) || defined(SS_TRANSLUCENCY)\nin vec3 surfaceAlbedo,\n#endif\n#ifdef SS_REFRACTION\nin vec3 vPositionW,\nin vec3 viewDirectionW,\nin mat4 view,\nin vec4 vRefractionInfos,\nin mat4 refractionMatrix,\nin vec4 vRefractionMicrosurfaceInfos,\nin vec4 vLightingIntensity,\n#ifdef SS_LINKREFRACTIONTOTRANSPARENCY\nin float alpha,\n#endif\n#ifdef SS_LODINREFRACTIONALPHA\nin float NdotVUnclamped,\n#endif\n#ifdef SS_LINEARSPECULARREFRACTION\nin float roughness,\n#endif\nin float alphaG,\n#ifdef SS_REFRACTIONMAP_3D\nin samplerCube refractionSampler,\n#ifndef LODBASEDMICROSFURACE\nin samplerCube refractionSamplerLow,\nin samplerCube refractionSamplerHigh,\n#endif\n#else\nin sampler2D refractionSampler,\n#ifndef LODBASEDMICROSFURACE\nin sampler2D refractionSamplerLow,\nin sampler2D refractionSamplerHigh,\n#endif\n#endif\n#ifdef ANISOTROPIC\nin anisotropicOutParams anisotropicOut,\n#endif\n#ifdef REALTIME_FILTERING\nin vec2 vRefractionFilteringInfo,\n#endif\n#ifdef SS_USE_LOCAL_REFRACTIONMAP_CUBIC\nin vec3 refractionPosition,\nin vec3 refractionSize,\n#endif\n#endif\n#ifdef SS_TRANSLUCENCY\nin vec3 vDiffusionDistance,\n#endif\nout subSurfaceOutParams outParams\n)\n{outParams.specularEnvironmentReflectance=specularEnvironmentReflectance;\n#ifdef SS_REFRACTION\nfloat refractionIntensity=vSubSurfaceIntensity.x;\n#ifdef SS_LINKREFRACTIONTOTRANSPARENCY\nrefractionIntensity*=(1.0-alpha);outParams.alpha=1.0;\n#endif\n#endif\n#ifdef SS_TRANSLUCENCY\nfloat translucencyIntensity=vSubSurfaceIntensity.y;\n#endif\n#ifdef SS_THICKNESSANDMASK_TEXTURE\n#if defined(SS_USE_GLTF_TEXTURES)\nfloat thickness=thicknessMap.g*vThicknessParam.y+vThicknessParam.x;\n#else\nfloat thickness=thicknessMap.r*vThicknessParam.y+vThicknessParam.x;\n#endif\n#if DEBUGMODE>0\noutParams.thicknessMap=thicknessMap;\n#endif\n#ifdef SS_MASK_FROM_THICKNESS_TEXTURE\n#if defined(SS_REFRACTION) && defined(SS_REFRACTION_USE_INTENSITY_FROM_TEXTURE)\n#if defined(SS_USE_GLTF_TEXTURES)\nrefractionIntensity*=thicknessMap.r;\n#else\nrefractionIntensity*=thicknessMap.g;\n#endif\n#endif\n#if defined(SS_TRANSLUCENCY) && defined(SS_TRANSLUCENCY_USE_INTENSITY_FROM_TEXTURE)\ntranslucencyIntensity*=thicknessMap.b;\n#endif\n#endif\n#else\nfloat thickness=vThicknessParam.y;\n#endif\n#ifdef SS_REFRACTIONINTENSITY_TEXTURE\n#ifdef SS_USE_GLTF_TEXTURES\nrefractionIntensity*=refractionIntensityMap.r;\n#else\nrefractionIntensity*=refractionIntensityMap.g;\n#endif\n#endif\n#ifdef SS_TRANSLUCENCYINTENSITY_TEXTURE\ntranslucencyIntensity*=translucencyIntensityMap.b;\n#endif\n#ifdef SS_TRANSLUCENCY\nthickness=maxEps(thickness);vec3 transmittance=transmittanceBRDF_Burley(vTintColor.rgb,vDiffusionDistance,thickness);transmittance*=translucencyIntensity;outParams.transmittance=transmittance;outParams.translucencyIntensity=translucencyIntensity;\n#endif\n#ifdef SS_REFRACTION\nvec4 environmentRefraction=vec4(0.,0.,0.,0.);\n#ifdef ANISOTROPIC\nvec3 refractionVector=refract(-viewDirectionW,anisotropicOut.anisotropicNormal,vRefractionInfos.y);\n#else\nvec3 refractionVector=refract(-viewDirectionW,normalW,vRefractionInfos.y);\n#endif\n#ifdef SS_REFRACTIONMAP_OPPOSITEZ\nrefractionVector.z*=-1.0;\n#endif\n#ifdef SS_REFRACTIONMAP_3D\n#ifdef SS_USE_LOCAL_REFRACTIONMAP_CUBIC\nrefractionVector=parallaxCorrectNormal(vPositionW,refractionVector,refractionSize,refractionPosition);\n#endif\nrefractionVector.y=refractionVector.y*vRefractionInfos.w;vec3 refractionCoords=refractionVector;refractionCoords=vec3(refractionMatrix*vec4(refractionCoords,0));\n#else\n#ifdef SS_USE_THICKNESS_AS_DEPTH\nvec3 vRefractionUVW=vec3(refractionMatrix*(view*vec4(vPositionW+refractionVector*thickness,1.0)));\n#else\nvec3 vRefractionUVW=vec3(refractionMatrix*(view*vec4(vPositionW+refractionVector*vRefractionInfos.z,1.0)));\n#endif\nvec2 refractionCoords=vRefractionUVW.xy/vRefractionUVW.z;refractionCoords.y=1.0-refractionCoords.y;\n#endif\n#ifdef SS_HAS_THICKNESS\nfloat ior=vRefractionInfos.y;\n#else\nfloat ior=vRefractionMicrosurfaceInfos.w;\n#endif\n#ifdef SS_LODINREFRACTIONALPHA\nfloat refractionAlphaG=alphaG;refractionAlphaG=mix(alphaG,0.0,clamp(ior*3.0-2.0,0.0,1.0));float refractionLOD=getLodFromAlphaG(vRefractionMicrosurfaceInfos.x,refractionAlphaG,NdotVUnclamped);\n#elif defined(SS_LINEARSPECULARREFRACTION)\nfloat refractionRoughness=alphaG;refractionRoughness=mix(alphaG,0.0,clamp(ior*3.0-2.0,0.0,1.0));float refractionLOD=getLinearLodFromRoughness(vRefractionMicrosurfaceInfos.x,refractionRoughness);\n#else\nfloat refractionAlphaG=alphaG;refractionAlphaG=mix(alphaG,0.0,clamp(ior*3.0-2.0,0.0,1.0));float refractionLOD=getLodFromAlphaG(vRefractionMicrosurfaceInfos.x,refractionAlphaG);\n#endif\n#ifdef LODBASEDMICROSFURACE\nrefractionLOD=refractionLOD*vRefractionMicrosurfaceInfos.y+vRefractionMicrosurfaceInfos.z;\n#ifdef SS_LODINREFRACTIONALPHA\nfloat automaticRefractionLOD=UNPACK_LOD(sampleRefraction(refractionSampler,refractionCoords).a);float requestedRefractionLOD=max(automaticRefractionLOD,refractionLOD);\n#else\nfloat requestedRefractionLOD=refractionLOD;\n#endif\n#if defined(REALTIME_FILTERING) && defined(SS_REFRACTIONMAP_3D)\nenvironmentRefraction=vec4(radiance(alphaG,refractionSampler,refractionCoords,vRefractionFilteringInfo),1.0);\n#else\nenvironmentRefraction=sampleRefractionLod(refractionSampler,refractionCoords,requestedRefractionLOD);\n#endif\n#else\nfloat lodRefractionNormalized=saturate(refractionLOD/log2(vRefractionMicrosurfaceInfos.x));float lodRefractionNormalizedDoubled=lodRefractionNormalized*2.0;vec4 environmentRefractionMid=sampleRefraction(refractionSampler,refractionCoords);if (lodRefractionNormalizedDoubled<1.0){environmentRefraction=mix(\nsampleRefraction(refractionSamplerHigh,refractionCoords),\nenvironmentRefractionMid,\nlodRefractionNormalizedDoubled\n);} else {environmentRefraction=mix(\nenvironmentRefractionMid,\nsampleRefraction(refractionSamplerLow,refractionCoords),\nlodRefractionNormalizedDoubled-1.0\n);}\n#endif\n#ifdef SS_RGBDREFRACTION\nenvironmentRefraction.rgb=fromRGBD(environmentRefraction);\n#endif\n#ifdef SS_GAMMAREFRACTION\nenvironmentRefraction.rgb=toLinearSpace(environmentRefraction.rgb);\n#endif\nenvironmentRefraction.rgb*=vRefractionInfos.x;\n#endif\n#ifdef SS_REFRACTION\nvec3 refractionTransmittance=vec3(refractionIntensity);\n#ifdef SS_THICKNESSANDMASK_TEXTURE\nvec3 volumeAlbedo=computeColorAtDistanceInMedia(vTintColor.rgb,vTintColor.w);refractionTransmittance*=cocaLambert(volumeAlbedo,thickness);\n#elif defined(SS_LINKREFRACTIONTOTRANSPARENCY)\nfloat maxChannel=max(max(surfaceAlbedo.r,surfaceAlbedo.g),surfaceAlbedo.b);vec3 volumeAlbedo=saturate(maxChannel*surfaceAlbedo);environmentRefraction.rgb*=volumeAlbedo;\n#else\nvec3 volumeAlbedo=computeColorAtDistanceInMedia(vTintColor.rgb,vTintColor.w);refractionTransmittance*=cocaLambert(volumeAlbedo,vThicknessParam.y);\n#endif\n#ifdef SS_ALBEDOFORREFRACTIONTINT\nenvironmentRefraction.rgb*=surfaceAlbedo.rgb;\n#endif\noutParams.surfaceAlbedo=surfaceAlbedo*(1.-refractionIntensity);\n#ifdef REFLECTION\noutParams.refractionFactorForIrradiance=(1.-refractionIntensity);\n#endif\n#ifdef UNUSED_MULTIPLEBOUNCES\nvec3 bounceSpecularEnvironmentReflectance=(2.0*specularEnvironmentReflectance)/(1.0+specularEnvironmentReflectance);outParams.specularEnvironmentReflectance=mix(bounceSpecularEnvironmentReflectance,specularEnvironmentReflectance,refractionIntensity);\n#endif\nrefractionTransmittance*=1.0-outParams.specularEnvironmentReflectance;\n#if DEBUGMODE>0\noutParams.refractionTransmittance=refractionTransmittance;\n#endif\noutParams.finalRefraction=environmentRefraction.rgb*refractionTransmittance*vLightingIntensity.z;\n#if DEBUGMODE>0\noutParams.environmentRefraction=environmentRefraction;\n#endif\n#endif\n#if defined(REFLECTION) && defined(SS_TRANSLUCENCY)\n#if defined(NORMAL) && defined(USESPHERICALINVERTEX) || !defined(USESPHERICALFROMREFLECTIONMAP)\nvec3 irradianceVector=vec3(reflectionMatrix*vec4(normalW,0)).xyz;\n#ifdef REFLECTIONMAP_OPPOSITEZ\nirradianceVector.z*=-1.0;\n#endif\n#ifdef INVERTCUBICMAP\nirradianceVector.y*=-1.0;\n#endif\n#else\nvec3 irradianceVector=irradianceVector_;\n#endif\n#if defined(USESPHERICALFROMREFLECTIONMAP)\n#if defined(REALTIME_FILTERING)\nvec3 refractionIrradiance=irradiance(reflectionSampler,-irradianceVector,vReflectionFilteringInfo);\n#else\nvec3 refractionIrradiance=computeEnvironmentIrradiance(-irradianceVector);\n#endif\n#elif defined(USEIRRADIANCEMAP)\n#ifdef REFLECTIONMAP_3D\nvec3 irradianceCoords=irradianceVector;\n#else\nvec2 irradianceCoords=irradianceVector.xy;\n#ifdef REFLECTIONMAP_PROJECTION\nirradianceCoords/=irradianceVector.z;\n#endif\nirradianceCoords.y=1.0-irradianceCoords.y;\n#endif\nvec4 refractionIrradiance=sampleReflection(irradianceSampler,-irradianceCoords);\n#ifdef RGBDREFLECTION\nrefractionIrradiance.rgb=fromRGBD(refractionIrradiance);\n#endif\n#ifdef GAMMAREFLECTION\nrefractionIrradiance.rgb=toLinearSpace(refractionIrradiance.rgb);\n#endif\n#else\nvec4 refractionIrradiance=vec4(0.);\n#endif\nrefractionIrradiance.rgb*=transmittance;\n#ifdef SS_ALBEDOFORTRANSLUCENCYTINT\nrefractionIrradiance.rgb*=surfaceAlbedo.rgb;\n#endif\noutParams.refractionIrradiance=refractionIrradiance.rgb;\n#endif\n}\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrBlockSubSurface = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrBlockNormalGeometric\";\nconst shader = `vec3 viewDirectionW=normalize(vEyePosition.xyz-vPositionW);\n#ifdef NORMAL\nvec3 normalW=normalize(vNormalW);\n#else\nvec3 normalW=normalize(cross(dFdx(vPositionW),dFdy(vPositionW)))*vEyePosition.w;\n#endif\nvec3 geometricNormalW=normalW;\n#if defined(TWOSIDEDLIGHTING) && defined(NORMAL)\ngeometricNormalW=gl_FrontFacing ? geometricNormalW : -geometricNormalW;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrBlockNormalGeometric = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrBlockNormalFinal\";\nconst shader = `#if defined(FORCENORMALFORWARD) && defined(NORMAL)\nvec3 faceNormal=normalize(cross(dFdx(vPositionW),dFdy(vPositionW)))*vEyePosition.w;\n#if defined(TWOSIDEDLIGHTING)\nfaceNormal=gl_FrontFacing ? faceNormal : -faceNormal;\n#endif\nnormalW*=sign(dot(normalW,faceNormal));\n#endif\n#if defined(TWOSIDEDLIGHTING) && defined(NORMAL)\nnormalW=gl_FrontFacing ? normalW : -normalW;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrBlockNormalFinal = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrBlockLightmapInit\";\nconst shader = `#ifdef LIGHTMAP\nvec4 lightmapColor=texture2D(lightmapSampler,vLightmapUV+uvOffset);\n#ifdef RGBDLIGHTMAP\nlightmapColor.rgb=fromRGBD(lightmapColor);\n#endif\n#ifdef GAMMALIGHTMAP\nlightmapColor.rgb=toLinearSpace(lightmapColor.rgb);\n#endif\nlightmapColor.rgb*=vLightmapInfos.y;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrBlockLightmapInit = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrBlockGeometryInfo\";\nconst shader = `float NdotVUnclamped=dot(normalW,viewDirectionW);float NdotV=absEps(NdotVUnclamped);float alphaG=convertRoughnessToAverageSlope(roughness);vec2 AARoughnessFactors=getAARoughnessFactors(normalW.xyz);\n#ifdef SPECULARAA\nalphaG+=AARoughnessFactors.y;\n#endif\n#if defined(ENVIRONMENTBRDF)\nvec3 environmentBrdf=getBRDFLookup(NdotV,roughness);\n#endif\n#if defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX)\n#ifdef RADIANCEOCCLUSION\n#ifdef AMBIENTINGRAYSCALE\nfloat ambientMonochrome=aoOut.ambientOcclusionColor.r;\n#else\nfloat ambientMonochrome=getLuminance(aoOut.ambientOcclusionColor);\n#endif\nfloat seo=environmentRadianceOcclusion(ambientMonochrome,NdotVUnclamped);\n#endif\n#ifdef HORIZONOCCLUSION\n#ifdef BUMP\n#ifdef REFLECTIONMAP_3D\nfloat eho=environmentHorizonOcclusion(-viewDirectionW,normalW,geometricNormalW);\n#endif\n#endif\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrBlockGeometryInfo = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrBlockReflectance0\";\nconst shader = `float reflectance=max(max(reflectivityOut.surfaceReflectivityColor.r,reflectivityOut.surfaceReflectivityColor.g),reflectivityOut.surfaceReflectivityColor.b);vec3 specularEnvironmentR0=reflectivityOut.surfaceReflectivityColor.rgb;\n#ifdef METALLICWORKFLOW\nvec3 specularEnvironmentR90=vec3(metallicReflectanceFactors.a);\n#else \nvec3 specularEnvironmentR90=vec3(1.0,1.0,1.0);\n#endif\n#ifdef ALPHAFRESNEL\nfloat reflectance90=fresnelGrazingReflectance(reflectance);specularEnvironmentR90=specularEnvironmentR90*reflectance90;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrBlockReflectance0 = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrBlockReflectance\";\nconst shader = `#if defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX)\nvec3 specularEnvironmentReflectance=getReflectanceFromBRDFLookup(clearcoatOut.specularEnvironmentR0,specularEnvironmentR90,environmentBrdf);\n#ifdef RADIANCEOCCLUSION\nspecularEnvironmentReflectance*=seo;\n#endif\n#ifdef HORIZONOCCLUSION\n#ifdef BUMP\n#ifdef REFLECTIONMAP_3D\nspecularEnvironmentReflectance*=eho;\n#endif\n#endif\n#endif\n#else\nvec3 specularEnvironmentReflectance=getReflectanceFromAnalyticalBRDFLookup_Jones(NdotV,clearcoatOut.specularEnvironmentR0,specularEnvironmentR90,sqrt(microSurface));\n#endif\n#ifdef CLEARCOAT\nspecularEnvironmentReflectance*=clearcoatOut.conservationFactor;\n#if defined(CLEARCOAT_TINT)\nspecularEnvironmentReflectance*=clearcoatOut.absorption;\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrBlockReflectance = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrBlockDirectLighting\";\nconst shader = `vec3 diffuseBase=vec3(0.,0.,0.);\n#ifdef SPECULARTERM\nvec3 specularBase=vec3(0.,0.,0.);\n#endif\n#ifdef CLEARCOAT\nvec3 clearCoatBase=vec3(0.,0.,0.);\n#endif\n#ifdef SHEEN\nvec3 sheenBase=vec3(0.,0.,0.);\n#endif\npreLightingInfo preInfo;lightingInfo info;float shadow=1.; \n#if defined(CLEARCOAT) && defined(CLEARCOAT_TINT)\nvec3 absorption=vec3(0.);\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrBlockDirectLighting = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrBlockFinalLitComponents\";\nconst shader = `#if defined(ENVIRONMENTBRDF)\n#ifdef MS_BRDF_ENERGY_CONSERVATION\nvec3 energyConservationFactor=getEnergyConservationFactor(clearcoatOut.specularEnvironmentR0,environmentBrdf);\n#endif\n#endif\n#ifndef METALLICWORKFLOW\n#ifdef SPECULAR_GLOSSINESS_ENERGY_CONSERVATION\nsurfaceAlbedo.rgb=(1.-reflectance)*surfaceAlbedo.rgb;\n#endif\n#endif\n#if defined(SHEEN) && defined(SHEEN_ALBEDOSCALING) && defined(ENVIRONMENTBRDF)\nsurfaceAlbedo.rgb=sheenOut.sheenAlbedoScaling*surfaceAlbedo.rgb;\n#endif\n#ifdef REFLECTION\nvec3 finalIrradiance=reflectionOut.environmentIrradiance;\n#if defined(CLEARCOAT)\nfinalIrradiance*=clearcoatOut.conservationFactor;\n#if defined(CLEARCOAT_TINT)\nfinalIrradiance*=clearcoatOut.absorption;\n#endif\n#endif\n#if defined(SS_REFRACTION)\nfinalIrradiance*=subSurfaceOut.refractionFactorForIrradiance;\n#endif\n#if defined(SS_TRANSLUCENCY)\nfinalIrradiance*=(1.0-subSurfaceOut.translucencyIntensity);finalIrradiance+=subSurfaceOut.refractionIrradiance;\n#endif\nfinalIrradiance*=surfaceAlbedo.rgb;finalIrradiance*=vLightingIntensity.z;finalIrradiance*=aoOut.ambientOcclusionColor;\n#endif\n#ifdef SPECULARTERM\nvec3 finalSpecular=specularBase;finalSpecular=max(finalSpecular,0.0);vec3 finalSpecularScaled=finalSpecular*vLightingIntensity.x*vLightingIntensity.w;\n#if defined(ENVIRONMENTBRDF) && defined(MS_BRDF_ENERGY_CONSERVATION)\nfinalSpecularScaled*=energyConservationFactor;\n#endif\n#if defined(SHEEN) && defined(ENVIRONMENTBRDF) && defined(SHEEN_ALBEDOSCALING)\nfinalSpecularScaled*=sheenOut.sheenAlbedoScaling;\n#endif\n#endif\n#ifdef REFLECTION\nvec3 finalRadiance=reflectionOut.environmentRadiance.rgb;finalRadiance*=subSurfaceOut.specularEnvironmentReflectance;vec3 finalRadianceScaled=finalRadiance*vLightingIntensity.z;\n#if defined(ENVIRONMENTBRDF) && defined(MS_BRDF_ENERGY_CONSERVATION)\nfinalRadianceScaled*=energyConservationFactor;\n#endif\n#if defined(SHEEN) && defined(ENVIRONMENTBRDF) && defined(SHEEN_ALBEDOSCALING)\nfinalRadianceScaled*=sheenOut.sheenAlbedoScaling;\n#endif\n#endif\n#ifdef SHEEN\nvec3 finalSheen=sheenBase*sheenOut.sheenColor;finalSheen=max(finalSheen,0.0);vec3 finalSheenScaled=finalSheen*vLightingIntensity.x*vLightingIntensity.w;\n#if defined(CLEARCOAT) && defined(REFLECTION) && defined(ENVIRONMENTBRDF)\nsheenOut.finalSheenRadianceScaled*=clearcoatOut.conservationFactor;\n#if defined(CLEARCOAT_TINT)\nsheenOut.finalSheenRadianceScaled*=clearcoatOut.absorption;\n#endif\n#endif\n#endif\n#ifdef CLEARCOAT\nvec3 finalClearCoat=clearCoatBase;finalClearCoat=max(finalClearCoat,0.0);vec3 finalClearCoatScaled=finalClearCoat*vLightingIntensity.x*vLightingIntensity.w;\n#if defined(ENVIRONMENTBRDF) && defined(MS_BRDF_ENERGY_CONSERVATION)\nfinalClearCoatScaled*=clearcoatOut.energyConservationFactorClearCoat;\n#endif\n#ifdef SS_REFRACTION\nsubSurfaceOut.finalRefraction*=clearcoatOut.conservationFactor;\n#ifdef CLEARCOAT_TINT\nsubSurfaceOut.finalRefraction*=clearcoatOut.absorption;\n#endif\n#endif\n#endif\n#ifdef ALPHABLEND\nfloat luminanceOverAlpha=0.0;\n#if defined(REFLECTION) && defined(RADIANCEOVERALPHA)\nluminanceOverAlpha+=getLuminance(finalRadianceScaled);\n#if defined(CLEARCOAT)\nluminanceOverAlpha+=getLuminance(clearcoatOut.finalClearCoatRadianceScaled);\n#endif\n#endif\n#if defined(SPECULARTERM) && defined(SPECULAROVERALPHA)\nluminanceOverAlpha+=getLuminance(finalSpecularScaled);\n#endif\n#if defined(CLEARCOAT) && defined(CLEARCOATOVERALPHA)\nluminanceOverAlpha+=getLuminance(finalClearCoatScaled);\n#endif\n#if defined(RADIANCEOVERALPHA) || defined(SPECULAROVERALPHA) || defined(CLEARCOATOVERALPHA)\nalpha=saturate(alpha+luminanceOverAlpha*luminanceOverAlpha);\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrBlockFinalLitComponents = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrBlockFinalUnlitComponents\";\nconst shader = `vec3 finalDiffuse=diffuseBase;finalDiffuse*=surfaceAlbedo.rgb;finalDiffuse=max(finalDiffuse,0.0);finalDiffuse*=vLightingIntensity.x;vec3 finalAmbient=vAmbientColor;finalAmbient*=surfaceAlbedo.rgb;vec3 finalEmissive=vEmissiveColor;\n#ifdef EMISSIVE\nvec3 emissiveColorTex=texture2D(emissiveSampler,vEmissiveUV+uvOffset).rgb;\n#ifdef GAMMAEMISSIVE\nfinalEmissive*=toLinearSpace(emissiveColorTex.rgb);\n#else\nfinalEmissive*=emissiveColorTex.rgb;\n#endif\nfinalEmissive*= vEmissiveInfos.y;\n#endif\nfinalEmissive*=vLightingIntensity.y;\n#ifdef AMBIENT\nvec3 ambientOcclusionForDirectDiffuse=mix(vec3(1.),aoOut.ambientOcclusionColor,vAmbientInfos.w);\n#else\nvec3 ambientOcclusionForDirectDiffuse=aoOut.ambientOcclusionColor;\n#endif\nfinalAmbient*=aoOut.ambientOcclusionColor;finalDiffuse*=ambientOcclusionForDirectDiffuse;\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrBlockFinalUnlitComponents = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrBlockFinalColorComposition\";\nconst shader = `vec4 finalColor=vec4(\n#ifndef UNLIT\n#ifdef REFLECTION\nfinalIrradiance +\n#endif\n#ifdef SPECULARTERM\nfinalSpecularScaled +\n#endif\n#ifdef SHEEN\nfinalSheenScaled +\n#endif\n#ifdef CLEARCOAT\nfinalClearCoatScaled +\n#endif\n#ifdef REFLECTION\nfinalRadianceScaled +\n#if defined(SHEEN) && defined(ENVIRONMENTBRDF)\nsheenOut.finalSheenRadianceScaled +\n#endif\n#ifdef CLEARCOAT\nclearcoatOut.finalClearCoatRadianceScaled +\n#endif\n#endif\n#ifdef SS_REFRACTION\nsubSurfaceOut.finalRefraction +\n#endif\n#endif\nfinalAmbient +\nfinalDiffuse,\nalpha);\n#ifdef LIGHTMAP\n#ifndef LIGHTMAPEXCLUDED\n#ifdef USELIGHTMAPASSHADOWMAP\nfinalColor.rgb*=lightmapColor.rgb;\n#else\nfinalColor.rgb+=lightmapColor.rgb;\n#endif\n#endif\n#endif\nfinalColor.rgb+=finalEmissive;\n#define CUSTOM_FRAGMENT_BEFORE_FOG\nfinalColor=max(finalColor,0.0);\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrBlockFinalColorComposition = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrBlockImageProcessing\";\nconst shader = `#if defined(IMAGEPROCESSINGPOSTPROCESS) || defined(SS_SCATTERING)\n#if !defined(SKIPFINALCOLORCLAMP)\nfinalColor.rgb=clamp(finalColor.rgb,0.,30.0);\n#endif\n#else\nfinalColor=applyImageProcessing(finalColor);\n#endif\nfinalColor.a*=visibility;\n#ifdef PREMULTIPLYALPHA\nfinalColor.rgb*=finalColor.a;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrBlockImageProcessing = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"pbrDebug\";\nconst shader = `#if DEBUGMODE>0\nif (vClipSpacePosition.x/vClipSpacePosition.w>=vDebugMode.x) {\n#if DEBUGMODE==1\ngl_FragColor.rgb=vPositionW.rgb;\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==2 && defined(NORMAL)\ngl_FragColor.rgb=vNormalW.rgb;\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==3 && defined(BUMP) || DEBUGMODE==3 && defined(PARALLAX) || DEBUGMODE==3 && defined(ANISOTROPIC)\ngl_FragColor.rgb=TBN[0];\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==4 && defined(BUMP) || DEBUGMODE==4 && defined(PARALLAX) || DEBUGMODE==4 && defined(ANISOTROPIC)\ngl_FragColor.rgb=TBN[1];\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==5\ngl_FragColor.rgb=normalW;\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==6 && defined(MAINUV1)\ngl_FragColor.rgb=vec3(vMainUV1,0.0);\n#elif DEBUGMODE==7 && defined(MAINUV2)\ngl_FragColor.rgb=vec3(vMainUV2,0.0);\n#elif DEBUGMODE==8 && defined(CLEARCOAT) && defined(CLEARCOAT_BUMP)\ngl_FragColor.rgb=clearcoatOut.TBNClearCoat[0];\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==9 && defined(CLEARCOAT) && defined(CLEARCOAT_BUMP)\ngl_FragColor.rgb=clearcoatOut.TBNClearCoat[1];\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==10 && defined(CLEARCOAT)\ngl_FragColor.rgb=clearcoatOut.clearCoatNormalW;\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==11 && defined(ANISOTROPIC)\ngl_FragColor.rgb=anisotropicOut.anisotropicNormal;\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==12 && defined(ANISOTROPIC)\ngl_FragColor.rgb=anisotropicOut.anisotropicTangent;\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==13 && defined(ANISOTROPIC)\ngl_FragColor.rgb=anisotropicOut.anisotropicBitangent;\n#define DEBUGMODE_NORMALIZE\n#elif DEBUGMODE==20 && defined(ALBEDO)\ngl_FragColor.rgb=albedoTexture.rgb;\n#ifndef GAMMAALBEDO\n#define DEBUGMODE_GAMMA\n#endif\n#elif DEBUGMODE==21 && defined(AMBIENT)\ngl_FragColor.rgb=aoOut.ambientOcclusionColorMap.rgb;\n#elif DEBUGMODE==22 && defined(OPACITY)\ngl_FragColor.rgb=opacityMap.rgb;\n#elif DEBUGMODE==23 && defined(EMISSIVE)\ngl_FragColor.rgb=emissiveColorTex.rgb;\n#ifndef GAMMAEMISSIVE\n#define DEBUGMODE_GAMMA\n#endif\n#elif DEBUGMODE==24 && defined(LIGHTMAP)\ngl_FragColor.rgb=lightmapColor.rgb;\n#ifndef GAMMALIGHTMAP\n#define DEBUGMODE_GAMMA\n#endif\n#elif DEBUGMODE==25 && defined(REFLECTIVITY) && defined(METALLICWORKFLOW)\ngl_FragColor.rgb=reflectivityOut.surfaceMetallicColorMap.rgb;\n#elif DEBUGMODE==26 && defined(REFLECTIVITY) && !defined(METALLICWORKFLOW)\ngl_FragColor.rgb=reflectivityOut.surfaceReflectivityColorMap.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==27 && defined(CLEARCOAT) && defined(CLEARCOAT_TEXTURE)\ngl_FragColor.rgb=vec3(clearcoatOut.clearCoatMapData.rg,0.0);\n#elif DEBUGMODE==28 && defined(CLEARCOAT) && defined(CLEARCOAT_TINT) && defined(CLEARCOAT_TINT_TEXTURE)\ngl_FragColor.rgb=clearcoatOut.clearCoatTintMapData.rgb;\n#elif DEBUGMODE==29 && defined(SHEEN) && defined(SHEEN_TEXTURE)\ngl_FragColor.rgb=sheenOut.sheenMapData.rgb;\n#elif DEBUGMODE==30 && defined(ANISOTROPIC) && defined(ANISOTROPIC_TEXTURE)\ngl_FragColor.rgb=anisotropicOut.anisotropyMapData.rgb;\n#elif DEBUGMODE==31 && defined(SUBSURFACE) && defined(SS_THICKNESSANDMASK_TEXTURE)\ngl_FragColor.rgb=subSurfaceOut.thicknessMap.rgb;\n#elif DEBUGMODE==32 && defined(BUMP)\ngl_FragColor.rgb=texture2D(bumpSampler,vBumpUV).rgb;\n#elif DEBUGMODE==40 && defined(SS_REFRACTION)\ngl_FragColor.rgb=subSurfaceOut.environmentRefraction.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==41 && defined(REFLECTION)\ngl_FragColor.rgb=reflectionOut.environmentRadiance.rgb;\n#ifndef GAMMAREFLECTION\n#define DEBUGMODE_GAMMA\n#endif\n#elif DEBUGMODE==42 && defined(CLEARCOAT) && defined(REFLECTION)\ngl_FragColor.rgb=clearcoatOut.environmentClearCoatRadiance.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==50\ngl_FragColor.rgb=diffuseBase.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==51 && defined(SPECULARTERM)\ngl_FragColor.rgb=specularBase.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==52 && defined(CLEARCOAT)\ngl_FragColor.rgb=clearCoatBase.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==53 && defined(SHEEN)\ngl_FragColor.rgb=sheenBase.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==54 && defined(REFLECTION)\ngl_FragColor.rgb=reflectionOut.environmentIrradiance.rgb;\n#ifndef GAMMAREFLECTION\n#define DEBUGMODE_GAMMA\n#endif\n#elif DEBUGMODE==60\ngl_FragColor.rgb=surfaceAlbedo.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==61\ngl_FragColor.rgb=clearcoatOut.specularEnvironmentR0;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==62 && defined(METALLICWORKFLOW)\ngl_FragColor.rgb=vec3(reflectivityOut.metallicRoughness.r);\n#elif DEBUGMODE==71 && defined(METALLICWORKFLOW)\ngl_FragColor.rgb=reflectivityOut.metallicF0;\n#elif DEBUGMODE==63\ngl_FragColor.rgb=vec3(roughness);\n#elif DEBUGMODE==64\ngl_FragColor.rgb=vec3(alphaG);\n#elif DEBUGMODE==65\ngl_FragColor.rgb=vec3(NdotV);\n#elif DEBUGMODE==66 && defined(CLEARCOAT) && defined(CLEARCOAT_TINT)\ngl_FragColor.rgb=clearcoatOut.clearCoatColor.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==67 && defined(CLEARCOAT)\ngl_FragColor.rgb=vec3(clearcoatOut.clearCoatRoughness);\n#elif DEBUGMODE==68 && defined(CLEARCOAT)\ngl_FragColor.rgb=vec3(clearcoatOut.clearCoatNdotV);\n#elif DEBUGMODE==69 && defined(SUBSURFACE) && defined(SS_TRANSLUCENCY)\ngl_FragColor.rgb=subSurfaceOut.transmittance;\n#elif DEBUGMODE==70 && defined(SUBSURFACE) && defined(SS_REFRACTION)\ngl_FragColor.rgb=subSurfaceOut.refractionTransmittance;\n#elif DEBUGMODE==72\ngl_FragColor.rgb=vec3(microSurface);\n#elif DEBUGMODE==73\ngl_FragColor.rgb=vAlbedoColor.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==74 && !defined(METALLICWORKFLOW)\ngl_FragColor.rgb=vReflectivityColor.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==75\ngl_FragColor.rgb=vEmissiveColor.rgb;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==80 && defined(RADIANCEOCCLUSION)\ngl_FragColor.rgb=vec3(seo);\n#elif DEBUGMODE==81 && defined(HORIZONOCCLUSION)\ngl_FragColor.rgb=vec3(eho);\n#elif DEBUGMODE==82 && defined(MS_BRDF_ENERGY_CONSERVATION)\ngl_FragColor.rgb=vec3(energyConservationFactor);\n#elif DEBUGMODE==83 && defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX)\ngl_FragColor.rgb=specularEnvironmentReflectance;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==84 && defined(CLEARCOAT) && defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX)\ngl_FragColor.rgb=clearcoatOut.clearCoatEnvironmentReflectance;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==85 && defined(SHEEN) && defined(REFLECTION)\ngl_FragColor.rgb=sheenOut.sheenEnvironmentReflectance;\n#define DEBUGMODE_GAMMA\n#elif DEBUGMODE==86 && defined(ALPHABLEND)\ngl_FragColor.rgb=vec3(luminanceOverAlpha);\n#elif DEBUGMODE==87\ngl_FragColor.rgb=vec3(alpha);\n#elif DEBUGMODE==88 && defined(ALBEDO)\ngl_FragColor.rgb=vec3(albedoTexture.a);\n#else\nfloat stripeWidth=30.;float stripePos=floor((gl_FragCoord.x+gl_FragCoord.y)/stripeWidth);float whichColor=mod(stripePos,2.);vec3 color1=vec3(.6,.2,.2);vec3 color2=vec3(.3,.1,.1);gl_FragColor.rgb=mix(color1,color2,whichColor);\n#endif\ngl_FragColor.rgb*=vDebugMode.y;\n#ifdef DEBUGMODE_NORMALIZE\ngl_FragColor.rgb=normalize(gl_FragColor.rgb)*0.5+0.5;\n#endif\n#ifdef DEBUGMODE_GAMMA\ngl_FragColor.rgb=toGammaSpace(gl_FragColor.rgb);\n#endif\ngl_FragColor.a=1.0;\n#ifdef PREPASS\ngl_FragData[0]=toLinearSpace(gl_FragColor); \ngl_FragData[1]=vec4(0.,0.,0.,0.); \n#endif\n}\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrDebug = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/prePassDeclaration\";\nimport \"./ShadersInclude/oitDeclaration\";\nimport \"./ShadersInclude/pbrFragmentDeclaration\";\nimport \"./ShadersInclude/pbrUboDeclaration\";\nimport \"./ShadersInclude/pbrFragmentExtraDeclaration\";\nimport \"./ShadersInclude/lightFragmentDeclaration\";\nimport \"./ShadersInclude/lightUboDeclaration\";\nimport \"./ShadersInclude/pbrFragmentSamplersDeclaration\";\nimport \"./ShadersInclude/imageProcessingDeclaration\";\nimport \"./ShadersInclude/clipPlaneFragmentDeclaration\";\nimport \"./ShadersInclude/logDepthDeclaration\";\nimport \"./ShadersInclude/fogFragmentDeclaration\";\nimport \"./ShadersInclude/helperFunctions\";\nimport \"./ShadersInclude/subSurfaceScatteringFunctions\";\nimport \"./ShadersInclude/importanceSampling\";\nimport \"./ShadersInclude/pbrHelperFunctions\";\nimport \"./ShadersInclude/imageProcessingFunctions\";\nimport \"./ShadersInclude/shadowsFragmentFunctions\";\nimport \"./ShadersInclude/harmonicsFunctions\";\nimport \"./ShadersInclude/pbrDirectLightingSetupFunctions\";\nimport \"./ShadersInclude/pbrDirectLightingFalloffFunctions\";\nimport \"./ShadersInclude/pbrBRDFFunctions\";\nimport \"./ShadersInclude/hdrFilteringFunctions\";\nimport \"./ShadersInclude/pbrDirectLightingFunctions\";\nimport \"./ShadersInclude/pbrIBLFunctions\";\nimport \"./ShadersInclude/bumpFragmentMainFunctions\";\nimport \"./ShadersInclude/bumpFragmentFunctions\";\nimport \"./ShadersInclude/reflectionFunction\";\nimport \"./ShadersInclude/pbrBlockAlbedoOpacity\";\nimport \"./ShadersInclude/pbrBlockReflectivity\";\nimport \"./ShadersInclude/pbrBlockAmbientOcclusion\";\nimport \"./ShadersInclude/pbrBlockAlphaFresnel\";\nimport \"./ShadersInclude/pbrBlockAnisotropic\";\nimport \"./ShadersInclude/pbrBlockReflection\";\nimport \"./ShadersInclude/pbrBlockSheen\";\nimport \"./ShadersInclude/pbrBlockClearcoat\";\nimport \"./ShadersInclude/pbrBlockIridescence\";\nimport \"./ShadersInclude/pbrBlockSubSurface\";\nimport \"./ShadersInclude/clipPlaneFragment\";\nimport \"./ShadersInclude/pbrBlockNormalGeometric\";\nimport \"./ShadersInclude/bumpFragment\";\nimport \"./ShadersInclude/pbrBlockNormalFinal\";\nimport \"./ShadersInclude/depthPrePass\";\nimport \"./ShadersInclude/pbrBlockLightmapInit\";\nimport \"./ShadersInclude/pbrBlockGeometryInfo\";\nimport \"./ShadersInclude/pbrBlockReflectance0\";\nimport \"./ShadersInclude/pbrBlockReflectance\";\nimport \"./ShadersInclude/pbrBlockDirectLighting\";\nimport \"./ShadersInclude/lightFragment\";\nimport \"./ShadersInclude/pbrBlockFinalLitComponents\";\nimport \"./ShadersInclude/pbrBlockFinalUnlitComponents\";\nimport \"./ShadersInclude/pbrBlockFinalColorComposition\";\nimport \"./ShadersInclude/logDepthFragment\";\nimport \"./ShadersInclude/fogFragment\";\nimport \"./ShadersInclude/pbrBlockImageProcessing\";\nimport \"./ShadersInclude/oitFragment\";\nimport \"./ShadersInclude/pbrDebug\";\n\nconst name = \"pbrPixelShader\";\nconst shader = `#if defined(BUMP) || !defined(NORMAL) || defined(FORCENORMALFORWARD) || defined(SPECULARAA) || defined(CLEARCOAT_BUMP) || defined(ANISOTROPIC)\n#extension GL_OES_standard_derivatives : enable\n#endif\n#ifdef LODBASEDMICROSFURACE\n#extension GL_EXT_shader_texture_lod : enable\n#endif\n#define CUSTOM_FRAGMENT_BEGIN\n#ifdef LOGARITHMICDEPTH\n#extension GL_EXT_frag_depth : enable\n#endif\n#include[SCENE_MRT_COUNT]\nprecision highp float;\n#include\n#ifndef FROMLINEARSPACE\n#define FROMLINEARSPACE\n#endif\n#include<__decl__pbrFragment>\n#include\n#include<__decl__lightFragment>[0..maxSimultaneousLights]\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#ifdef REFLECTION\n#include\n#endif\n#define CUSTOM_FRAGMENT_DEFINITIONS\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\n#include\nvoid main(void) {\n#define CUSTOM_FRAGMENT_MAIN_BEGIN\n#include\n#include\n#include\n#include\nalbedoOpacityOutParams albedoOpacityOut;\n#ifdef ALBEDO\nvec4 albedoTexture=texture2D(albedoSampler,vAlbedoUV+uvOffset);\n#endif\n#ifdef OPACITY\nvec4 opacityMap=texture2D(opacitySampler,vOpacityUV+uvOffset);\n#endif\n#ifdef DECAL\nvec4 decalColor=texture2D(decalSampler,vDecalUV+uvOffset);\n#endif\nalbedoOpacityBlock(\nvAlbedoColor,\n#ifdef ALBEDO\nalbedoTexture,\nvAlbedoInfos,\n#endif\n#ifdef OPACITY\nopacityMap,\nvOpacityInfos,\n#endif\n#ifdef DETAIL\ndetailColor,\nvDetailInfos,\n#endif\n#ifdef DECAL\ndecalColor,\nvDecalInfos,\n#endif\nalbedoOpacityOut\n);vec3 surfaceAlbedo=albedoOpacityOut.surfaceAlbedo;float alpha=albedoOpacityOut.alpha;\n#define CUSTOM_FRAGMENT_UPDATE_ALPHA\n#include\n#define CUSTOM_FRAGMENT_BEFORE_LIGHTS\nambientOcclusionOutParams aoOut;\n#ifdef AMBIENT\nvec3 ambientOcclusionColorMap=texture2D(ambientSampler,vAmbientUV+uvOffset).rgb;\n#endif\nambientOcclusionBlock(\n#ifdef AMBIENT\nambientOcclusionColorMap,\nvAmbientInfos,\n#endif\naoOut\n);\n#include\n#ifdef UNLIT\nvec3 diffuseBase=vec3(1.,1.,1.);\n#else\nvec3 baseColor=surfaceAlbedo;reflectivityOutParams reflectivityOut;\n#if defined(REFLECTIVITY)\nvec4 surfaceMetallicOrReflectivityColorMap=texture2D(reflectivitySampler,vReflectivityUV+uvOffset);vec4 baseReflectivity=surfaceMetallicOrReflectivityColorMap;\n#ifndef METALLICWORKFLOW\n#ifdef REFLECTIVITY_GAMMA\nsurfaceMetallicOrReflectivityColorMap=toLinearSpace(surfaceMetallicOrReflectivityColorMap);\n#endif\nsurfaceMetallicOrReflectivityColorMap.rgb*=vReflectivityInfos.y;\n#endif\n#endif\n#if defined(MICROSURFACEMAP)\nvec4 microSurfaceTexel=texture2D(microSurfaceSampler,vMicroSurfaceSamplerUV+uvOffset)*vMicroSurfaceSamplerInfos.y;\n#endif\n#ifdef METALLICWORKFLOW\nvec4 metallicReflectanceFactors=vMetallicReflectanceFactors;\n#ifdef REFLECTANCE\nvec4 reflectanceFactorsMap=texture2D(reflectanceSampler,vReflectanceUV+uvOffset);\n#ifdef REFLECTANCE_GAMMA\nreflectanceFactorsMap=toLinearSpace(reflectanceFactorsMap);\n#endif\nmetallicReflectanceFactors.rgb*=reflectanceFactorsMap.rgb;\n#endif\n#ifdef METALLIC_REFLECTANCE\nvec4 metallicReflectanceFactorsMap=texture2D(metallicReflectanceSampler,vMetallicReflectanceUV+uvOffset);\n#ifdef METALLIC_REFLECTANCE_GAMMA\nmetallicReflectanceFactorsMap=toLinearSpace(metallicReflectanceFactorsMap);\n#endif\n#ifndef METALLIC_REFLECTANCE_USE_ALPHA_ONLY\nmetallicReflectanceFactors.rgb*=metallicReflectanceFactorsMap.rgb;\n#endif\nmetallicReflectanceFactors*=metallicReflectanceFactorsMap.a;\n#endif\n#endif\nreflectivityBlock(\nvReflectivityColor,\n#ifdef METALLICWORKFLOW\nsurfaceAlbedo,\nmetallicReflectanceFactors,\n#endif\n#ifdef REFLECTIVITY\nvReflectivityInfos,\nsurfaceMetallicOrReflectivityColorMap,\n#endif\n#if defined(METALLICWORKFLOW) && defined(REFLECTIVITY) && defined(AOSTOREINMETALMAPRED)\naoOut.ambientOcclusionColor,\n#endif\n#ifdef MICROSURFACEMAP\nmicroSurfaceTexel,\n#endif\n#ifdef DETAIL\ndetailColor,\nvDetailInfos,\n#endif\nreflectivityOut\n);float microSurface=reflectivityOut.microSurface;float roughness=reflectivityOut.roughness;\n#ifdef METALLICWORKFLOW\nsurfaceAlbedo=reflectivityOut.surfaceAlbedo;\n#endif\n#if defined(METALLICWORKFLOW) && defined(REFLECTIVITY) && defined(AOSTOREINMETALMAPRED)\naoOut.ambientOcclusionColor=reflectivityOut.ambientOcclusionColor;\n#endif\n#ifdef ALPHAFRESNEL\n#if defined(ALPHATEST) || defined(ALPHABLEND)\nalphaFresnelOutParams alphaFresnelOut;alphaFresnelBlock(\nnormalW,\nviewDirectionW,\nalpha,\nmicroSurface,\nalphaFresnelOut\n);alpha=alphaFresnelOut.alpha;\n#endif\n#endif\n#include\n#ifdef ANISOTROPIC\nanisotropicOutParams anisotropicOut;\n#ifdef ANISOTROPIC_TEXTURE\nvec3 anisotropyMapData=texture2D(anisotropySampler,vAnisotropyUV+uvOffset).rgb*vAnisotropyInfos.y;\n#endif\nanisotropicBlock(\nvAnisotropy,\nroughness,\n#ifdef ANISOTROPIC_TEXTURE\nanisotropyMapData,\n#endif\nTBN,\nnormalW,\nviewDirectionW,\nanisotropicOut\n);\n#endif\n#ifdef REFLECTION\nreflectionOutParams reflectionOut;\n#ifndef USE_CUSTOM_REFLECTION\nreflectionBlock(\nvPositionW,\nnormalW,\nalphaG,\nvReflectionMicrosurfaceInfos,\nvReflectionInfos,\nvReflectionColor,\n#ifdef ANISOTROPIC\nanisotropicOut,\n#endif\n#if defined(LODINREFLECTIONALPHA) && !defined(REFLECTIONMAP_SKYBOX)\nNdotVUnclamped,\n#endif\n#ifdef LINEARSPECULARREFLECTION\nroughness,\n#endif\nreflectionSampler,\n#if defined(NORMAL) && defined(USESPHERICALINVERTEX)\nvEnvironmentIrradiance,\n#endif\n#ifdef USESPHERICALFROMREFLECTIONMAP\n#if !defined(NORMAL) || !defined(USESPHERICALINVERTEX)\nreflectionMatrix,\n#endif\n#endif\n#ifdef USEIRRADIANCEMAP\nirradianceSampler,\n#endif\n#ifndef LODBASEDMICROSFURACE\nreflectionSamplerLow,\nreflectionSamplerHigh,\n#endif\n#ifdef REALTIME_FILTERING\nvReflectionFilteringInfo,\n#endif\nreflectionOut\n);\n#else\n#define CUSTOM_REFLECTION\n#endif\n#endif\n#include\n#ifdef SHEEN\nsheenOutParams sheenOut;\n#ifdef SHEEN_TEXTURE\nvec4 sheenMapData=texture2D(sheenSampler,vSheenUV+uvOffset);\n#endif\n#if defined(SHEEN_ROUGHNESS) && defined(SHEEN_TEXTURE_ROUGHNESS) && !defined(SHEEN_TEXTURE_ROUGHNESS_IDENTICAL) && !defined(SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE)\nvec4 sheenMapRoughnessData=texture2D(sheenRoughnessSampler,vSheenRoughnessUV+uvOffset)*vSheenInfos.w;\n#endif\nsheenBlock(\nvSheenColor,\n#ifdef SHEEN_ROUGHNESS\nvSheenRoughness,\n#if defined(SHEEN_TEXTURE_ROUGHNESS) && !defined(SHEEN_TEXTURE_ROUGHNESS_IDENTICAL) && !defined(SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE)\nsheenMapRoughnessData,\n#endif\n#endif\nroughness,\n#ifdef SHEEN_TEXTURE\nsheenMapData,\nvSheenInfos.y,\n#endif\nreflectance,\n#ifdef SHEEN_LINKWITHALBEDO\nbaseColor,\nsurfaceAlbedo,\n#endif\n#ifdef ENVIRONMENTBRDF\nNdotV,\nenvironmentBrdf,\n#endif\n#if defined(REFLECTION) && defined(ENVIRONMENTBRDF)\nAARoughnessFactors,\nvReflectionMicrosurfaceInfos,\nvReflectionInfos,\nvReflectionColor,\nvLightingIntensity,\nreflectionSampler,\nreflectionOut.reflectionCoords,\nNdotVUnclamped,\n#ifndef LODBASEDMICROSFURACE\nreflectionSamplerLow,\nreflectionSamplerHigh,\n#endif\n#ifdef REALTIME_FILTERING\nvReflectionFilteringInfo,\n#endif\n#if !defined(REFLECTIONMAP_SKYBOX) && defined(RADIANCEOCCLUSION)\nseo,\n#endif\n#if !defined(REFLECTIONMAP_SKYBOX) && defined(HORIZONOCCLUSION) && defined(BUMP) && defined(REFLECTIONMAP_3D)\neho,\n#endif\n#endif\nsheenOut\n);\n#ifdef SHEEN_LINKWITHALBEDO\nsurfaceAlbedo=sheenOut.surfaceAlbedo;\n#endif\n#endif\n#ifdef CLEARCOAT\n#ifdef CLEARCOAT_TEXTURE\nvec2 clearCoatMapData=texture2D(clearCoatSampler,vClearCoatUV+uvOffset).rg*vClearCoatInfos.y;\n#endif\n#endif\n#ifdef IRIDESCENCE\niridescenceOutParams iridescenceOut;\n#ifdef IRIDESCENCE_TEXTURE\nvec2 iridescenceMapData=texture2D(iridescenceSampler,vIridescenceUV+uvOffset).rg*vIridescenceInfos.y;\n#endif\n#ifdef IRIDESCENCE_THICKNESS_TEXTURE\nvec2 iridescenceThicknessMapData=texture2D(iridescenceThicknessSampler,vIridescenceThicknessUV+uvOffset).rg*vIridescenceInfos.w;\n#endif\niridescenceBlock(\nvIridescenceParams,\nNdotV,\nspecularEnvironmentR0,\n#ifdef IRIDESCENCE_TEXTURE\niridescenceMapData,\n#endif\n#ifdef IRIDESCENCE_THICKNESS_TEXTURE\niridescenceThicknessMapData,\n#endif\n#ifdef CLEARCOAT\nNdotVUnclamped,\n#ifdef CLEARCOAT_TEXTURE\nclearCoatMapData,\n#endif\n#endif\niridescenceOut\n);float iridescenceIntensity=iridescenceOut.iridescenceIntensity;specularEnvironmentR0=iridescenceOut.specularEnvironmentR0;\n#endif\nclearcoatOutParams clearcoatOut;\n#ifdef CLEARCOAT\n#if defined(CLEARCOAT_TEXTURE_ROUGHNESS) && !defined(CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL) && !defined(CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE)\nvec4 clearCoatMapRoughnessData=texture2D(clearCoatRoughnessSampler,vClearCoatRoughnessUV+uvOffset)*vClearCoatInfos.w;\n#endif\n#if defined(CLEARCOAT_TINT) && defined(CLEARCOAT_TINT_TEXTURE)\nvec4 clearCoatTintMapData=texture2D(clearCoatTintSampler,vClearCoatTintUV+uvOffset);\n#endif\n#ifdef CLEARCOAT_BUMP\nvec4 clearCoatBumpMapData=texture2D(clearCoatBumpSampler,vClearCoatBumpUV+uvOffset);\n#endif\nclearcoatBlock(\nvPositionW,\ngeometricNormalW,\nviewDirectionW,\nvClearCoatParams,\n#if defined(CLEARCOAT_TEXTURE_ROUGHNESS) && !defined(CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL) && !defined(CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE)\nclearCoatMapRoughnessData,\n#endif\nspecularEnvironmentR0,\n#ifdef CLEARCOAT_TEXTURE\nclearCoatMapData,\n#endif\n#ifdef CLEARCOAT_TINT\nvClearCoatTintParams,\nclearCoatColorAtDistance,\nvClearCoatRefractionParams,\n#ifdef CLEARCOAT_TINT_TEXTURE\nclearCoatTintMapData,\n#endif\n#endif\n#ifdef CLEARCOAT_BUMP\nvClearCoatBumpInfos,\nclearCoatBumpMapData,\nvClearCoatBumpUV,\n#if defined(TANGENT) && defined(NORMAL)\nvTBN,\n#else\nvClearCoatTangentSpaceParams,\n#endif\n#ifdef OBJECTSPACE_NORMALMAP\nnormalMatrix,\n#endif\n#endif\n#if defined(FORCENORMALFORWARD) && defined(NORMAL)\nfaceNormal,\n#endif\n#ifdef REFLECTION\nvReflectionMicrosurfaceInfos,\nvReflectionInfos,\nvReflectionColor,\nvLightingIntensity,\nreflectionSampler,\n#ifndef LODBASEDMICROSFURACE\nreflectionSamplerLow,\nreflectionSamplerHigh,\n#endif\n#ifdef REALTIME_FILTERING\nvReflectionFilteringInfo,\n#endif\n#endif\n#if defined(ENVIRONMENTBRDF) && !defined(REFLECTIONMAP_SKYBOX)\n#ifdef RADIANCEOCCLUSION\nambientMonochrome,\n#endif\n#endif\n#if defined(CLEARCOAT_BUMP) || defined(TWOSIDEDLIGHTING)\n(gl_FrontFacing ? 1. : -1.),\n#endif\nclearcoatOut\n);\n#else\nclearcoatOut.specularEnvironmentR0=specularEnvironmentR0;\n#endif\n#include\nsubSurfaceOutParams subSurfaceOut;\n#ifdef SUBSURFACE\n#ifdef SS_THICKNESSANDMASK_TEXTURE\nvec4 thicknessMap=texture2D(thicknessSampler,vThicknessUV+uvOffset);\n#endif\n#ifdef SS_REFRACTIONINTENSITY_TEXTURE\nvec4 refractionIntensityMap=texture2D(refractionIntensitySampler,vRefractionIntensityUV+uvOffset);\n#endif\n#ifdef SS_TRANSLUCENCYINTENSITY_TEXTURE\nvec4 translucencyIntensityMap=texture2D(translucencyIntensitySampler,vTranslucencyIntensityUV+uvOffset);\n#endif\nsubSurfaceBlock(\nvSubSurfaceIntensity,\nvThicknessParam,\nvTintColor,\nnormalW,\nspecularEnvironmentReflectance,\n#ifdef SS_THICKNESSANDMASK_TEXTURE\nthicknessMap,\n#endif\n#ifdef SS_REFRACTIONINTENSITY_TEXTURE\nrefractionIntensityMap,\n#endif\n#ifdef SS_TRANSLUCENCYINTENSITY_TEXTURE\ntranslucencyIntensityMap,\n#endif\n#ifdef REFLECTION\n#ifdef SS_TRANSLUCENCY\nreflectionMatrix,\n#ifdef USESPHERICALFROMREFLECTIONMAP\n#if !defined(NORMAL) || !defined(USESPHERICALINVERTEX)\nreflectionOut.irradianceVector,\n#endif\n#if defined(REALTIME_FILTERING)\nreflectionSampler,\nvReflectionFilteringInfo,\n#endif\n#endif\n#ifdef USEIRRADIANCEMAP\nirradianceSampler,\n#endif\n#endif\n#endif\n#if defined(SS_REFRACTION) || defined(SS_TRANSLUCENCY)\nsurfaceAlbedo,\n#endif\n#ifdef SS_REFRACTION\nvPositionW,\nviewDirectionW,\nview,\nvRefractionInfos,\nrefractionMatrix,\nvRefractionMicrosurfaceInfos,\nvLightingIntensity,\n#ifdef SS_LINKREFRACTIONTOTRANSPARENCY\nalpha,\n#endif\n#ifdef SS_LODINREFRACTIONALPHA\nNdotVUnclamped,\n#endif\n#ifdef SS_LINEARSPECULARREFRACTION\nroughness,\n#endif\nalphaG,\nrefractionSampler,\n#ifndef LODBASEDMICROSFURACE\nrefractionSamplerLow,\nrefractionSamplerHigh,\n#endif\n#ifdef ANISOTROPIC\nanisotropicOut,\n#endif\n#ifdef REALTIME_FILTERING\nvRefractionFilteringInfo,\n#endif\n#ifdef SS_USE_LOCAL_REFRACTIONMAP_CUBIC\nvRefractionPosition,\nvRefractionSize,\n#endif\n#endif\n#ifdef SS_TRANSLUCENCY\nvDiffusionDistance,\n#endif\nsubSurfaceOut\n);\n#ifdef SS_REFRACTION\nsurfaceAlbedo=subSurfaceOut.surfaceAlbedo;\n#ifdef SS_LINKREFRACTIONTOTRANSPARENCY\nalpha=subSurfaceOut.alpha;\n#endif\n#endif\n#else\nsubSurfaceOut.specularEnvironmentReflectance=specularEnvironmentReflectance;\n#endif\n#include\n#include[0..maxSimultaneousLights]\n#include\n#endif \n#include\n#define CUSTOM_FRAGMENT_BEFORE_FINALCOLORCOMPOSITION\n#include\n#include\n#include(color,finalColor)\n#include\n#define CUSTOM_FRAGMENT_BEFORE_FRAGCOLOR\n#ifdef PREPASS\nfloat writeGeometryInfo=finalColor.a>0.4 ? 1.0 : 0.0;\n#ifdef PREPASS_POSITION\ngl_FragData[PREPASS_POSITION_INDEX]=vec4(vPositionW,writeGeometryInfo);\n#endif\n#ifdef PREPASS_VELOCITY\nvec2 a=(vCurrentPosition.xy/vCurrentPosition.w)*0.5+0.5;vec2 b=(vPreviousPosition.xy/vPreviousPosition.w)*0.5+0.5;vec2 velocity=abs(a-b);velocity=vec2(pow(velocity.x,1.0/3.0),pow(velocity.y,1.0/3.0))*sign(a-b)*0.5+0.5;gl_FragData[PREPASS_VELOCITY_INDEX]=vec4(velocity,0.0,writeGeometryInfo);\n#endif\n#ifdef PREPASS_ALBEDO_SQRT\nvec3 sqAlbedo=sqrt(surfaceAlbedo); \n#endif\n#ifdef PREPASS_IRRADIANCE\nvec3 irradiance=finalDiffuse;\n#ifndef UNLIT\n#ifdef REFLECTION\nirradiance+=finalIrradiance;\n#endif\n#endif\n#ifdef SS_SCATTERING\ngl_FragData[0]=vec4(finalColor.rgb-irradiance,finalColor.a); \nirradiance/=sqAlbedo;\n#else\ngl_FragData[0]=finalColor; \nfloat scatteringDiffusionProfile=255.;\n#endif\ngl_FragData[PREPASS_IRRADIANCE_INDEX]=vec4(clamp(irradiance,vec3(0.),vec3(1.)),writeGeometryInfo*scatteringDiffusionProfile/255.); \n#else\ngl_FragData[0]=vec4(finalColor.rgb,finalColor.a);\n#endif\n#ifdef PREPASS_DEPTH\ngl_FragData[PREPASS_DEPTH_INDEX]=vec4(vViewPos.z,0.0,0.0,writeGeometryInfo); \n#endif\n#ifdef PREPASS_NORMAL\ngl_FragData[PREPASS_NORMAL_INDEX]=vec4(normalize((view*vec4(normalW,0.0)).rgb),writeGeometryInfo); \n#endif\n#ifdef PREPASS_ALBEDO_SQRT\ngl_FragData[PREPASS_ALBEDO_SQRT_INDEX]=vec4(sqAlbedo,writeGeometryInfo); \n#endif\n#ifdef PREPASS_REFLECTIVITY\n#ifndef UNLIT\ngl_FragData[PREPASS_REFLECTIVITY_INDEX]=vec4(specularEnvironmentR0,microSurface)*writeGeometryInfo;\n#else\ngl_FragData[PREPASS_REFLECTIVITY_INDEX]=vec4( 0.0,0.0,0.0,1.0 )*writeGeometryInfo;\n#endif\n#endif\n#endif\n#if !defined(PREPASS) || defined(WEBGL2)\ngl_FragColor=finalColor;\n#endif\n#include\n#if ORDER_INDEPENDENT_TRANSPARENCY\nif (fragDepth==nearestDepth) {frontColor.rgb+=finalColor.rgb*finalColor.a*alphaMultiplier;frontColor.a=1.0-alphaMultiplier*(1.0-finalColor.a);} else {backColor+=finalColor;}\n#endif\n#include\n#define CUSTOM_FRAGMENT_MAIN_END\n}\n`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const pbrPixelShader = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\nimport \"./decalVertexDeclaration\";\n\nconst name = \"pbrVertexDeclaration\";\nconst shader = `uniform mat4 view;uniform mat4 viewProjection;\n#ifdef ALBEDO\nuniform mat4 albedoMatrix;uniform vec2 vAlbedoInfos;\n#endif\n#ifdef AMBIENT\nuniform mat4 ambientMatrix;uniform vec4 vAmbientInfos;\n#endif\n#ifdef OPACITY\nuniform mat4 opacityMatrix;uniform vec2 vOpacityInfos;\n#endif\n#ifdef EMISSIVE\nuniform vec2 vEmissiveInfos;uniform mat4 emissiveMatrix;\n#endif\n#ifdef LIGHTMAP\nuniform vec2 vLightmapInfos;uniform mat4 lightmapMatrix;\n#endif\n#ifdef REFLECTIVITY \nuniform vec3 vReflectivityInfos;uniform mat4 reflectivityMatrix;\n#endif\n#ifdef METALLIC_REFLECTANCE\nuniform vec2 vMetallicReflectanceInfos;uniform mat4 metallicReflectanceMatrix;\n#endif\n#ifdef REFLECTANCE\nuniform vec2 vReflectanceInfos;uniform mat4 reflectanceMatrix;\n#endif\n#ifdef MICROSURFACEMAP\nuniform vec2 vMicroSurfaceSamplerInfos;uniform mat4 microSurfaceSamplerMatrix;\n#endif\n#ifdef BUMP\nuniform vec3 vBumpInfos;uniform mat4 bumpMatrix;\n#endif\n#ifdef POINTSIZE\nuniform float pointSize;\n#endif\n#ifdef REFLECTION\nuniform vec2 vReflectionInfos;uniform mat4 reflectionMatrix;\n#endif\n#ifdef CLEARCOAT\n#if defined(CLEARCOAT_TEXTURE) || defined(CLEARCOAT_TEXTURE_ROUGHNESS)\nuniform vec4 vClearCoatInfos;\n#endif\n#ifdef CLEARCOAT_TEXTURE\nuniform mat4 clearCoatMatrix;\n#endif\n#ifdef CLEARCOAT_TEXTURE_ROUGHNESS\nuniform mat4 clearCoatRoughnessMatrix;\n#endif\n#ifdef CLEARCOAT_BUMP\nuniform vec2 vClearCoatBumpInfos;uniform mat4 clearCoatBumpMatrix;\n#endif\n#ifdef CLEARCOAT_TINT_TEXTURE\nuniform vec2 vClearCoatTintInfos;uniform mat4 clearCoatTintMatrix;\n#endif\n#endif\n#ifdef IRIDESCENCE\n#if defined(IRIDESCENCE_TEXTURE) || defined(IRIDESCENCE_THICKNESS_TEXTURE)\nuniform vec4 vIridescenceInfos;\n#endif\n#ifdef IRIDESCENCE_TEXTURE\nuniform mat4 iridescenceMatrix;\n#endif\n#ifdef IRIDESCENCE_THICKNESS_TEXTURE\nuniform mat4 iridescenceThicknessMatrix;\n#endif\n#endif\n#ifdef ANISOTROPIC\n#ifdef ANISOTROPIC_TEXTURE\nuniform vec2 vAnisotropyInfos;uniform mat4 anisotropyMatrix;\n#endif\n#endif\n#ifdef SHEEN\n#if defined(SHEEN_TEXTURE) || defined(SHEEN_TEXTURE_ROUGHNESS)\nuniform vec4 vSheenInfos;\n#endif\n#ifdef SHEEN_TEXTURE\nuniform mat4 sheenMatrix;\n#endif\n#ifdef SHEEN_TEXTURE_ROUGHNESS\nuniform mat4 sheenRoughnessMatrix;\n#endif\n#endif\n#ifdef SUBSURFACE\n#ifdef SS_REFRACTION\nuniform vec4 vRefractionInfos;uniform mat4 refractionMatrix;\n#endif\n#ifdef SS_THICKNESSANDMASK_TEXTURE\nuniform vec2 vThicknessInfos;uniform mat4 thicknessMatrix;\n#endif\n#ifdef SS_REFRACTIONINTENSITY_TEXTURE\nuniform vec2 vRefractionIntensityInfos;uniform mat4 refractionIntensityMatrix;\n#endif\n#ifdef SS_TRANSLUCENCYINTENSITY_TEXTURE\nuniform vec2 vTranslucencyIntensityInfos;uniform mat4 translucencyIntensityMatrix;\n#endif\n#endif\n#ifdef NORMAL\n#if defined(USESPHERICALFROMREFLECTIONMAP) && defined(USESPHERICALINVERTEX)\n#ifdef USESPHERICALFROMREFLECTIONMAP\n#ifdef SPHERICAL_HARMONICS\nuniform vec3 vSphericalL00;uniform vec3 vSphericalL1_1;uniform vec3 vSphericalL10;uniform vec3 vSphericalL11;uniform vec3 vSphericalL2_2;uniform vec3 vSphericalL2_1;uniform vec3 vSphericalL20;uniform vec3 vSphericalL21;uniform vec3 vSphericalL22;\n#else\nuniform vec3 vSphericalX;uniform vec3 vSphericalY;uniform vec3 vSphericalZ;uniform vec3 vSphericalXX_ZZ;uniform vec3 vSphericalYY_ZZ;uniform vec3 vSphericalZZ;uniform vec3 vSphericalXY;uniform vec3 vSphericalYZ;uniform vec3 vSphericalZX;\n#endif\n#endif\n#endif\n#endif\n#ifdef DETAIL\nuniform vec4 vDetailInfos;uniform mat4 detailMatrix;\n#endif\n#include\n#define ADDITIONAL_VERTEX_DECLARATION\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const pbrVertexDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/pbrVertexDeclaration\";\nimport \"./ShadersInclude/pbrUboDeclaration\";\nimport \"./ShadersInclude/uvAttributeDeclaration\";\nimport \"./ShadersInclude/mainUVVaryingDeclaration\";\nimport \"./ShadersInclude/helperFunctions\";\nimport \"./ShadersInclude/bonesDeclaration\";\nimport \"./ShadersInclude/bakedVertexAnimationDeclaration\";\nimport \"./ShadersInclude/instancesDeclaration\";\nimport \"./ShadersInclude/prePassVertexDeclaration\";\nimport \"./ShadersInclude/samplerVertexDeclaration\";\nimport \"./ShadersInclude/harmonicsFunctions\";\nimport \"./ShadersInclude/bumpVertexDeclaration\";\nimport \"./ShadersInclude/clipPlaneVertexDeclaration\";\nimport \"./ShadersInclude/fogVertexDeclaration\";\nimport \"./ShadersInclude/lightVxFragmentDeclaration\";\nimport \"./ShadersInclude/lightVxUboDeclaration\";\nimport \"./ShadersInclude/morphTargetsVertexGlobalDeclaration\";\nimport \"./ShadersInclude/morphTargetsVertexDeclaration\";\nimport \"./ShadersInclude/logDepthDeclaration\";\nimport \"./ShadersInclude/morphTargetsVertexGlobal\";\nimport \"./ShadersInclude/morphTargetsVertex\";\nimport \"./ShadersInclude/instancesVertex\";\nimport \"./ShadersInclude/bonesVertex\";\nimport \"./ShadersInclude/bakedVertexAnimation\";\nimport \"./ShadersInclude/prePassVertex\";\nimport \"./ShadersInclude/uvVariableDeclaration\";\nimport \"./ShadersInclude/samplerVertexImplementation\";\nimport \"./ShadersInclude/bumpVertex\";\nimport \"./ShadersInclude/clipPlaneVertex\";\nimport \"./ShadersInclude/fogVertex\";\nimport \"./ShadersInclude/shadowsVertex\";\nimport \"./ShadersInclude/vertexColorMixing\";\nimport \"./ShadersInclude/logDepthVertex\";\n\nconst name = \"pbrVertexShader\";\nconst shader = `precision highp float;\n#include<__decl__pbrVertex>\n#define CUSTOM_VERTEX_BEGIN\nattribute vec3 position;\n#ifdef NORMAL\nattribute vec3 normal;\n#endif\n#ifdef TANGENT\nattribute vec4 tangent;\n#endif\n#ifdef UV1\nattribute vec2 uv;\n#endif\n#include[2..7]\n#include[1..7]\n#ifdef VERTEXCOLOR\nattribute vec4 color;\n#endif\n#include\n#include\n#include\n#include\n#include\n#include(_DEFINENAME_,ALBEDO,_VARYINGNAME_,Albedo)\n#include(_DEFINENAME_,DETAIL,_VARYINGNAME_,Detail)\n#include(_DEFINENAME_,AMBIENT,_VARYINGNAME_,Ambient)\n#include(_DEFINENAME_,OPACITY,_VARYINGNAME_,Opacity)\n#include(_DEFINENAME_,EMISSIVE,_VARYINGNAME_,Emissive)\n#include(_DEFINENAME_,LIGHTMAP,_VARYINGNAME_,Lightmap)\n#include(_DEFINENAME_,REFLECTIVITY,_VARYINGNAME_,Reflectivity)\n#include(_DEFINENAME_,MICROSURFACEMAP,_VARYINGNAME_,MicroSurfaceSampler)\n#include(_DEFINENAME_,METALLIC_REFLECTANCE,_VARYINGNAME_,MetallicReflectance)\n#include(_DEFINENAME_,REFLECTANCE,_VARYINGNAME_,Reflectance)\n#include(_DEFINENAME_,BUMP,_VARYINGNAME_,Bump)\n#include(_DEFINENAME_,DECAL,_VARYINGNAME_,Decal)\n#ifdef CLEARCOAT\n#include(_DEFINENAME_,CLEARCOAT_TEXTURE,_VARYINGNAME_,ClearCoat)\n#include(_DEFINENAME_,CLEARCOAT_TEXTURE_ROUGHNESS,_VARYINGNAME_,ClearCoatRoughness)\n#include(_DEFINENAME_,CLEARCOAT_BUMP,_VARYINGNAME_,ClearCoatBump)\n#include(_DEFINENAME_,CLEARCOAT_TINT_TEXTURE,_VARYINGNAME_,ClearCoatTint)\n#endif\n#ifdef IRIDESCENCE\n#include(_DEFINENAME_,IRIDESCENCE_TEXTURE,_VARYINGNAME_,Iridescence)\n#include(_DEFINENAME_,IRIDESCENCE_THICKNESS_TEXTURE,_VARYINGNAME_,IridescenceThickness)\n#endif\n#ifdef SHEEN\n#include(_DEFINENAME_,SHEEN_TEXTURE,_VARYINGNAME_,Sheen)\n#include(_DEFINENAME_,SHEEN_TEXTURE_ROUGHNESS,_VARYINGNAME_,SheenRoughness)\n#endif\n#ifdef ANISOTROPIC\n#include(_DEFINENAME_,ANISOTROPIC_TEXTURE,_VARYINGNAME_,Anisotropy)\n#endif\n#ifdef SUBSURFACE\n#include(_DEFINENAME_,SS_THICKNESSANDMASK_TEXTURE,_VARYINGNAME_,Thickness)\n#include(_DEFINENAME_,SS_REFRACTIONINTENSITY_TEXTURE,_VARYINGNAME_,RefractionIntensity)\n#include(_DEFINENAME_,SS_TRANSLUCENCYINTENSITY_TEXTURE,_VARYINGNAME_,TranslucencyIntensity)\n#endif\nvarying vec3 vPositionW;\n#if DEBUGMODE>0\nvarying vec4 vClipSpacePosition;\n#endif\n#ifdef NORMAL\nvarying vec3 vNormalW;\n#if defined(USESPHERICALFROMREFLECTIONMAP) && defined(USESPHERICALINVERTEX)\nvarying vec3 vEnvironmentIrradiance;\n#include\n#endif\n#endif\n#if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES)\nvarying vec4 vColor;\n#endif\n#include\n#include\n#include\n#include<__decl__lightVxFragment>[0..maxSimultaneousLights]\n#include\n#include[0..maxSimultaneousMorphTargets]\n#ifdef REFLECTIONMAP_SKYBOX\nvarying vec3 vPositionUVW;\n#endif\n#if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED)\nvarying vec3 vDirectionW;\n#endif\n#include\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\nvec3 positionUpdated=position;\n#ifdef NORMAL\nvec3 normalUpdated=normal;\n#endif\n#ifdef TANGENT\nvec4 tangentUpdated=tangent;\n#endif\n#ifdef UV1\nvec2 uvUpdated=uv;\n#endif\n#include\n#include[0..maxSimultaneousMorphTargets]\n#ifdef REFLECTIONMAP_SKYBOX\nvPositionUVW=positionUpdated;\n#endif\n#define CUSTOM_VERTEX_UPDATE_POSITION\n#define CUSTOM_VERTEX_UPDATE_NORMAL\n#include\n#if defined(PREPASS) && defined(PREPASS_VELOCITY) && !defined(BONES_VELOCITY_ENABLED)\nvCurrentPosition=viewProjection*finalWorld*vec4(positionUpdated,1.0);vPreviousPosition=previousViewProjection*finalPreviousWorld*vec4(positionUpdated,1.0);\n#endif\n#include\n#include\nvec4 worldPos=finalWorld*vec4(positionUpdated,1.0);vPositionW=vec3(worldPos);\n#include\n#ifdef NORMAL\nmat3 normalWorld=mat3(finalWorld);\n#if defined(INSTANCES) && defined(THIN_INSTANCES)\nvNormalW=normalUpdated/vec3(dot(normalWorld[0],normalWorld[0]),dot(normalWorld[1],normalWorld[1]),dot(normalWorld[2],normalWorld[2]));vNormalW=normalize(normalWorld*vNormalW);\n#else\n#ifdef NONUNIFORMSCALING\nnormalWorld=transposeMat3(inverseMat3(normalWorld));\n#endif\nvNormalW=normalize(normalWorld*normalUpdated);\n#endif\n#if defined(USESPHERICALFROMREFLECTIONMAP) && defined(USESPHERICALINVERTEX)\nvec3 reflectionVector=vec3(reflectionMatrix*vec4(vNormalW,0)).xyz;\n#ifdef REFLECTIONMAP_OPPOSITEZ\nreflectionVector.z*=-1.0;\n#endif\nvEnvironmentIrradiance=computeEnvironmentIrradiance(reflectionVector);\n#endif\n#endif\n#define CUSTOM_VERTEX_UPDATE_WORLDPOS\n#ifdef MULTIVIEW\nif (gl_ViewID_OVR==0u) {gl_Position=viewProjection*worldPos;} else {gl_Position=viewProjectionR*worldPos;}\n#else\ngl_Position=viewProjection*worldPos;\n#endif\n#if DEBUGMODE>0\nvClipSpacePosition=gl_Position;\n#endif\n#if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED)\nvDirectionW=normalize(vec3(finalWorld*vec4(positionUpdated,0.0)));\n#endif\n#ifndef UV1\nvec2 uvUpdated=vec2(0.,0.);\n#endif\n#ifdef MAINUV1\nvMainUV1=uvUpdated;\n#endif\n#include[2..7]\n#include(_DEFINENAME_,ALBEDO,_VARYINGNAME_,Albedo,_MATRIXNAME_,albedo,_INFONAME_,AlbedoInfos.x)\n#include(_DEFINENAME_,DETAIL,_VARYINGNAME_,Detail,_MATRIXNAME_,detail,_INFONAME_,DetailInfos.x)\n#include(_DEFINENAME_,AMBIENT,_VARYINGNAME_,Ambient,_MATRIXNAME_,ambient,_INFONAME_,AmbientInfos.x)\n#include(_DEFINENAME_,OPACITY,_VARYINGNAME_,Opacity,_MATRIXNAME_,opacity,_INFONAME_,OpacityInfos.x)\n#include(_DEFINENAME_,EMISSIVE,_VARYINGNAME_,Emissive,_MATRIXNAME_,emissive,_INFONAME_,EmissiveInfos.x)\n#include(_DEFINENAME_,LIGHTMAP,_VARYINGNAME_,Lightmap,_MATRIXNAME_,lightmap,_INFONAME_,LightmapInfos.x)\n#include(_DEFINENAME_,REFLECTIVITY,_VARYINGNAME_,Reflectivity,_MATRIXNAME_,reflectivity,_INFONAME_,ReflectivityInfos.x)\n#include(_DEFINENAME_,MICROSURFACEMAP,_VARYINGNAME_,MicroSurfaceSampler,_MATRIXNAME_,microSurfaceSampler,_INFONAME_,MicroSurfaceSamplerInfos.x)\n#include(_DEFINENAME_,METALLIC_REFLECTANCE,_VARYINGNAME_,MetallicReflectance,_MATRIXNAME_,metallicReflectance,_INFONAME_,MetallicReflectanceInfos.x)\n#include(_DEFINENAME_,REFLECTANCE,_VARYINGNAME_,Reflectance,_MATRIXNAME_,reflectance,_INFONAME_,ReflectanceInfos.x)\n#include(_DEFINENAME_,BUMP,_VARYINGNAME_,Bump,_MATRIXNAME_,bump,_INFONAME_,BumpInfos.x)\n#include(_DEFINENAME_,DECAL,_VARYINGNAME_,Decal,_MATRIXNAME_,decal,_INFONAME_,DecalInfos.x)\n#ifdef CLEARCOAT\n#include(_DEFINENAME_,CLEARCOAT_TEXTURE,_VARYINGNAME_,ClearCoat,_MATRIXNAME_,clearCoat,_INFONAME_,ClearCoatInfos.x)\n#include(_DEFINENAME_,CLEARCOAT_TEXTURE_ROUGHNESS,_VARYINGNAME_,ClearCoatRoughness,_MATRIXNAME_,clearCoatRoughness,_INFONAME_,ClearCoatInfos.z)\n#include(_DEFINENAME_,CLEARCOAT_BUMP,_VARYINGNAME_,ClearCoatBump,_MATRIXNAME_,clearCoatBump,_INFONAME_,ClearCoatBumpInfos.x)\n#include(_DEFINENAME_,CLEARCOAT_TINT_TEXTURE,_VARYINGNAME_,ClearCoatTint,_MATRIXNAME_,clearCoatTint,_INFONAME_,ClearCoatTintInfos.x)\n#endif\n#ifdef IRIDESCENCE\n#include(_DEFINENAME_,IRIDESCENCE_TEXTURE,_VARYINGNAME_,Iridescence,_MATRIXNAME_,iridescence,_INFONAME_,IridescenceInfos.x)\n#include(_DEFINENAME_,IRIDESCENCE_THICKNESS_TEXTURE,_VARYINGNAME_,IridescenceThickness,_MATRIXNAME_,iridescenceThickness,_INFONAME_,IridescenceInfos.z)\n#endif\n#ifdef SHEEN\n#include(_DEFINENAME_,SHEEN_TEXTURE,_VARYINGNAME_,Sheen,_MATRIXNAME_,sheen,_INFONAME_,SheenInfos.x)\n#include(_DEFINENAME_,SHEEN_TEXTURE_ROUGHNESS,_VARYINGNAME_,SheenRoughness,_MATRIXNAME_,sheen,_INFONAME_,SheenInfos.z)\n#endif\n#ifdef ANISOTROPIC\n#include(_DEFINENAME_,ANISOTROPIC_TEXTURE,_VARYINGNAME_,Anisotropy,_MATRIXNAME_,anisotropy,_INFONAME_,AnisotropyInfos.x)\n#endif\n#ifdef SUBSURFACE\n#include(_DEFINENAME_,SS_THICKNESSANDMASK_TEXTURE,_VARYINGNAME_,Thickness,_MATRIXNAME_,thickness,_INFONAME_,ThicknessInfos.x)\n#include(_DEFINENAME_,SS_REFRACTIONINTENSITY_TEXTURE,_VARYINGNAME_,RefractionIntensity,_MATRIXNAME_,refractionIntensity,_INFONAME_,RefractionIntensityInfos.x)\n#include(_DEFINENAME_,SS_TRANSLUCENCYINTENSITY_TEXTURE,_VARYINGNAME_,TranslucencyIntensity,_MATRIXNAME_,translucencyIntensity,_INFONAME_,TranslucencyIntensityInfos.x)\n#endif\n#include\n#include\n#include\n#include[0..maxSimultaneousLights]\n#include\n#if defined(POINTSIZE) && !defined(WEBGPU)\ngl_PointSize=pointSize;\n#endif\n#include\n#define CUSTOM_VERTEX_MAIN_END\n}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const pbrVertexShader = { name, shader };\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport type { Nullable } from \"../../types\";\r\nimport { serialize, serializeAsTexture, expandToProperty, serializeAsColor3 } from \"../../Misc/decorators\";\r\nimport { Color3 } from \"../../Maths/math.color\";\r\nimport type { BaseTexture } from \"../../Materials/Textures/baseTexture\";\r\nimport { MaterialFlags } from \"../materialFlags\";\r\nimport type { UniformBuffer } from \"../../Materials/uniformBuffer\";\r\nimport { MaterialHelper } from \"../../Materials/materialHelper\";\r\nimport type { IAnimatable } from \"../../Animations/animatable.interface\";\r\nimport type { EffectFallbacks } from \"../effectFallbacks\";\r\nimport type { SubMesh } from \"../../Meshes/subMesh\";\r\nimport { Constants } from \"../../Engines/constants\";\r\nimport { MaterialPluginBase } from \"../materialPluginBase\";\r\nimport { MaterialDefines } from \"../materialDefines\";\r\n\r\nimport type { Engine } from \"../../Engines/engine\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { PBRBaseMaterial } from \"./pbrBaseMaterial\";\r\n\r\n/**\r\n * @internal\r\n */\r\nexport class MaterialClearCoatDefines extends MaterialDefines {\r\n public CLEARCOAT = false;\r\n public CLEARCOAT_DEFAULTIOR = false;\r\n public CLEARCOAT_TEXTURE = false;\r\n public CLEARCOAT_TEXTURE_ROUGHNESS = false;\r\n public CLEARCOAT_TEXTUREDIRECTUV = 0;\r\n public CLEARCOAT_TEXTURE_ROUGHNESSDIRECTUV = 0;\r\n public CLEARCOAT_BUMP = false;\r\n public CLEARCOAT_BUMPDIRECTUV = 0;\r\n public CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE = false;\r\n public CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL = false;\r\n public CLEARCOAT_REMAP_F0 = false;\r\n\r\n public CLEARCOAT_TINT = false;\r\n public CLEARCOAT_TINT_TEXTURE = false;\r\n public CLEARCOAT_TINT_TEXTUREDIRECTUV = 0;\r\n public CLEARCOAT_TINT_GAMMATEXTURE = false;\r\n}\r\n\r\n/**\r\n * Plugin that implements the clear coat component of the PBR material\r\n */\r\nexport class PBRClearCoatConfiguration extends MaterialPluginBase {\r\n protected _material: PBRBaseMaterial;\r\n\r\n /**\r\n * This defaults to 1.5 corresponding to a 0.04 f0 or a 4% reflectance at normal incidence\r\n * The default fits with a polyurethane material.\r\n * @internal\r\n */\r\n public static readonly _DefaultIndexOfRefraction = 1.5;\r\n\r\n private _isEnabled = false;\r\n /**\r\n * Defines if the clear coat is enabled in the material.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public isEnabled = false;\r\n\r\n /**\r\n * Defines the clear coat layer strength (between 0 and 1) it defaults to 1.\r\n */\r\n @serialize()\r\n public intensity: number = 1;\r\n\r\n /**\r\n * Defines the clear coat layer roughness.\r\n */\r\n @serialize()\r\n public roughness: number = 0;\r\n\r\n private _indexOfRefraction = PBRClearCoatConfiguration._DefaultIndexOfRefraction;\r\n /**\r\n * Defines the index of refraction of the clear coat.\r\n * This defaults to 1.5 corresponding to a 0.04 f0 or a 4% reflectance at normal incidence\r\n * The default fits with a polyurethane material.\r\n * Changing the default value is more performance intensive.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public indexOfRefraction = PBRClearCoatConfiguration._DefaultIndexOfRefraction;\r\n\r\n private _texture: Nullable = null;\r\n /**\r\n * Stores the clear coat values in a texture (red channel is intensity and green channel is roughness)\r\n * If useRoughnessFromMainTexture is false, the green channel of texture is not used and the green channel of textureRoughness is used instead\r\n * if textureRoughness is not empty, else no texture roughness is used\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public texture: Nullable = null;\r\n\r\n private _useRoughnessFromMainTexture = true;\r\n /**\r\n * Indicates that the green channel of the texture property will be used for roughness (default: true)\r\n * If false, the green channel from textureRoughness is used for roughness\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useRoughnessFromMainTexture = true;\r\n\r\n private _textureRoughness: Nullable = null;\r\n /**\r\n * Stores the clear coat roughness in a texture (green channel)\r\n * Not used if useRoughnessFromMainTexture is true\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public textureRoughness: Nullable = null;\r\n\r\n private _remapF0OnInterfaceChange = true;\r\n /**\r\n * Defines if the F0 value should be remapped to account for the interface change in the material.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public remapF0OnInterfaceChange = true;\r\n\r\n private _bumpTexture: Nullable = null;\r\n /**\r\n * Define the clear coat specific bump texture.\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public bumpTexture: Nullable = null;\r\n\r\n private _isTintEnabled = false;\r\n /**\r\n * Defines if the clear coat tint is enabled in the material.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public isTintEnabled = false;\r\n\r\n /**\r\n * Defines the clear coat tint of the material.\r\n * This is only use if tint is enabled\r\n */\r\n @serializeAsColor3()\r\n public tintColor = Color3.White();\r\n\r\n /**\r\n * Defines the distance at which the tint color should be found in the\r\n * clear coat media.\r\n * This is only use if tint is enabled\r\n */\r\n @serialize()\r\n public tintColorAtDistance = 1;\r\n\r\n /**\r\n * Defines the clear coat layer thickness.\r\n * This is only use if tint is enabled\r\n */\r\n @serialize()\r\n public tintThickness: number = 1;\r\n\r\n private _tintTexture: Nullable = null;\r\n /**\r\n * Stores the clear tint values in a texture.\r\n * rgb is tint\r\n * a is a thickness factor\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public tintTexture: Nullable = null;\r\n\r\n /** @internal */\r\n private _internalMarkAllSubMeshesAsTexturesDirty: () => void;\r\n\r\n /** @internal */\r\n public _markAllSubMeshesAsTexturesDirty(): void {\r\n this._enable(this._isEnabled);\r\n this._internalMarkAllSubMeshesAsTexturesDirty();\r\n }\r\n\r\n constructor(material: PBRBaseMaterial, addToPluginList = true) {\r\n super(material, \"PBRClearCoat\", 100, new MaterialClearCoatDefines(), addToPluginList);\r\n\r\n this._internalMarkAllSubMeshesAsTexturesDirty = material._dirtyCallbacks[Constants.MATERIAL_TextureDirtyFlag];\r\n }\r\n\r\n public isReadyForSubMesh(defines: MaterialClearCoatDefines, scene: Scene, engine: Engine): boolean {\r\n if (!this._isEnabled) {\r\n return true;\r\n }\r\n\r\n const disableBumpMap = this._material._disableBumpMap;\r\n if (defines._areTexturesDirty) {\r\n if (scene.texturesEnabled) {\r\n if (this._texture && MaterialFlags.ClearCoatTextureEnabled) {\r\n if (!this._texture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n }\r\n\r\n if (this._textureRoughness && MaterialFlags.ClearCoatTextureEnabled) {\r\n if (!this._textureRoughness.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n }\r\n\r\n if (engine.getCaps().standardDerivatives && this._bumpTexture && MaterialFlags.ClearCoatBumpTextureEnabled && !disableBumpMap) {\r\n // Bump texture cannot be not blocking.\r\n if (!this._bumpTexture.isReady()) {\r\n return false;\r\n }\r\n }\r\n\r\n if (this._isTintEnabled && this._tintTexture && MaterialFlags.ClearCoatTintTextureEnabled) {\r\n if (!this._tintTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n }\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n public prepareDefinesBeforeAttributes(defines: MaterialClearCoatDefines, scene: Scene): void {\r\n if (this._isEnabled) {\r\n defines.CLEARCOAT = true;\r\n defines.CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE = this._useRoughnessFromMainTexture;\r\n defines.CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL =\r\n this._texture !== null && this._texture._texture === this._textureRoughness?._texture && this._texture.checkTransformsAreIdentical(this._textureRoughness);\r\n defines.CLEARCOAT_REMAP_F0 = this._remapF0OnInterfaceChange;\r\n\r\n if (defines._areTexturesDirty) {\r\n if (scene.texturesEnabled) {\r\n if (this._texture && MaterialFlags.ClearCoatTextureEnabled) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._texture, defines, \"CLEARCOAT_TEXTURE\");\r\n } else {\r\n defines.CLEARCOAT_TEXTURE = false;\r\n }\r\n\r\n if (this._textureRoughness && MaterialFlags.ClearCoatTextureEnabled) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._textureRoughness, defines, \"CLEARCOAT_TEXTURE_ROUGHNESS\");\r\n } else {\r\n defines.CLEARCOAT_TEXTURE_ROUGHNESS = false;\r\n }\r\n\r\n if (this._bumpTexture && MaterialFlags.ClearCoatBumpTextureEnabled) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._bumpTexture, defines, \"CLEARCOAT_BUMP\");\r\n } else {\r\n defines.CLEARCOAT_BUMP = false;\r\n }\r\n\r\n defines.CLEARCOAT_DEFAULTIOR = this._indexOfRefraction === PBRClearCoatConfiguration._DefaultIndexOfRefraction;\r\n\r\n if (this._isTintEnabled) {\r\n defines.CLEARCOAT_TINT = true;\r\n if (this._tintTexture && MaterialFlags.ClearCoatTintTextureEnabled) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._tintTexture, defines, \"CLEARCOAT_TINT_TEXTURE\");\r\n defines.CLEARCOAT_TINT_GAMMATEXTURE = this._tintTexture.gammaSpace;\r\n } else {\r\n defines.CLEARCOAT_TINT_TEXTURE = false;\r\n }\r\n } else {\r\n defines.CLEARCOAT_TINT = false;\r\n defines.CLEARCOAT_TINT_TEXTURE = false;\r\n }\r\n }\r\n }\r\n } else {\r\n defines.CLEARCOAT = false;\r\n defines.CLEARCOAT_TEXTURE = false;\r\n defines.CLEARCOAT_TEXTURE_ROUGHNESS = false;\r\n defines.CLEARCOAT_BUMP = false;\r\n defines.CLEARCOAT_TINT = false;\r\n defines.CLEARCOAT_TINT_TEXTURE = false;\r\n defines.CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE = false;\r\n defines.CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL = false;\r\n defines.CLEARCOAT_DEFAULTIOR = false;\r\n defines.CLEARCOAT_TEXTUREDIRECTUV = 0;\r\n defines.CLEARCOAT_TEXTURE_ROUGHNESSDIRECTUV = 0;\r\n defines.CLEARCOAT_BUMPDIRECTUV = 0;\r\n defines.CLEARCOAT_REMAP_F0 = false;\r\n defines.CLEARCOAT_TINT_TEXTUREDIRECTUV = 0;\r\n defines.CLEARCOAT_TINT_GAMMATEXTURE = false;\r\n }\r\n }\r\n\r\n public bindForSubMesh(uniformBuffer: UniformBuffer, scene: Scene, engine: Engine, subMesh: SubMesh): void {\r\n if (!this._isEnabled) {\r\n return;\r\n }\r\n\r\n const defines = subMesh!.materialDefines as unknown as MaterialClearCoatDefines;\r\n\r\n const isFrozen = this._material.isFrozen;\r\n\r\n const disableBumpMap = this._material._disableBumpMap;\r\n const invertNormalMapX = this._material._invertNormalMapX;\r\n const invertNormalMapY = this._material._invertNormalMapY;\r\n\r\n const identicalTextures = defines.CLEARCOAT_TEXTURE_ROUGHNESS_IDENTICAL;\r\n\r\n if (!uniformBuffer.useUbo || !isFrozen || !uniformBuffer.isSync) {\r\n if (identicalTextures && MaterialFlags.ClearCoatTextureEnabled) {\r\n uniformBuffer.updateFloat4(\"vClearCoatInfos\", this._texture!.coordinatesIndex, this._texture!.level, -1, -1);\r\n MaterialHelper.BindTextureMatrix(this._texture!, uniformBuffer, \"clearCoat\");\r\n } else if ((this._texture || this._textureRoughness) && MaterialFlags.ClearCoatTextureEnabled) {\r\n uniformBuffer.updateFloat4(\r\n \"vClearCoatInfos\",\r\n this._texture?.coordinatesIndex ?? 0,\r\n this._texture?.level ?? 0,\r\n this._textureRoughness?.coordinatesIndex ?? 0,\r\n this._textureRoughness?.level ?? 0\r\n );\r\n if (this._texture) {\r\n MaterialHelper.BindTextureMatrix(this._texture, uniformBuffer, \"clearCoat\");\r\n }\r\n if (this._textureRoughness && !identicalTextures && !defines.CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE) {\r\n MaterialHelper.BindTextureMatrix(this._textureRoughness, uniformBuffer, \"clearCoatRoughness\");\r\n }\r\n }\r\n\r\n if (this._bumpTexture && engine.getCaps().standardDerivatives && MaterialFlags.ClearCoatTextureEnabled && !disableBumpMap) {\r\n uniformBuffer.updateFloat2(\"vClearCoatBumpInfos\", this._bumpTexture.coordinatesIndex, this._bumpTexture.level);\r\n MaterialHelper.BindTextureMatrix(this._bumpTexture, uniformBuffer, \"clearCoatBump\");\r\n\r\n if (scene._mirroredCameraPosition) {\r\n uniformBuffer.updateFloat2(\"vClearCoatTangentSpaceParams\", invertNormalMapX ? 1.0 : -1.0, invertNormalMapY ? 1.0 : -1.0);\r\n } else {\r\n uniformBuffer.updateFloat2(\"vClearCoatTangentSpaceParams\", invertNormalMapX ? -1.0 : 1.0, invertNormalMapY ? -1.0 : 1.0);\r\n }\r\n }\r\n\r\n if (this._tintTexture && MaterialFlags.ClearCoatTintTextureEnabled) {\r\n uniformBuffer.updateFloat2(\"vClearCoatTintInfos\", this._tintTexture.coordinatesIndex, this._tintTexture.level);\r\n MaterialHelper.BindTextureMatrix(this._tintTexture, uniformBuffer, \"clearCoatTint\");\r\n }\r\n\r\n // Clear Coat General params\r\n uniformBuffer.updateFloat2(\"vClearCoatParams\", this.intensity, this.roughness);\r\n\r\n // Clear Coat Refraction params\r\n const a = 1 - this._indexOfRefraction;\r\n const b = 1 + this._indexOfRefraction;\r\n const f0 = Math.pow(-a / b, 2); // Schlicks approx: (ior1 - ior2) / (ior1 + ior2) where ior2 for air is close to vacuum = 1.\r\n const eta = 1 / this._indexOfRefraction;\r\n uniformBuffer.updateFloat4(\"vClearCoatRefractionParams\", f0, eta, a, b);\r\n\r\n if (this._isTintEnabled) {\r\n uniformBuffer.updateFloat4(\"vClearCoatTintParams\", this.tintColor.r, this.tintColor.g, this.tintColor.b, Math.max(0.00001, this.tintThickness));\r\n uniformBuffer.updateFloat(\"clearCoatColorAtDistance\", Math.max(0.00001, this.tintColorAtDistance));\r\n }\r\n }\r\n\r\n // Textures\r\n if (scene.texturesEnabled) {\r\n if (this._texture && MaterialFlags.ClearCoatTextureEnabled) {\r\n uniformBuffer.setTexture(\"clearCoatSampler\", this._texture);\r\n }\r\n\r\n if (this._textureRoughness && !identicalTextures && !defines.CLEARCOAT_USE_ROUGHNESS_FROM_MAINTEXTURE && MaterialFlags.ClearCoatTextureEnabled) {\r\n uniformBuffer.setTexture(\"clearCoatRoughnessSampler\", this._textureRoughness);\r\n }\r\n\r\n if (this._bumpTexture && engine.getCaps().standardDerivatives && MaterialFlags.ClearCoatBumpTextureEnabled && !disableBumpMap) {\r\n uniformBuffer.setTexture(\"clearCoatBumpSampler\", this._bumpTexture);\r\n }\r\n\r\n if (this._isTintEnabled && this._tintTexture && MaterialFlags.ClearCoatTintTextureEnabled) {\r\n uniformBuffer.setTexture(\"clearCoatTintSampler\", this._tintTexture);\r\n }\r\n }\r\n }\r\n\r\n public hasTexture(texture: BaseTexture): boolean {\r\n if (this._texture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._textureRoughness === texture) {\r\n return true;\r\n }\r\n\r\n if (this._bumpTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._tintTexture === texture) {\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n public getActiveTextures(activeTextures: BaseTexture[]): void {\r\n if (this._texture) {\r\n activeTextures.push(this._texture);\r\n }\r\n\r\n if (this._textureRoughness) {\r\n activeTextures.push(this._textureRoughness);\r\n }\r\n\r\n if (this._bumpTexture) {\r\n activeTextures.push(this._bumpTexture);\r\n }\r\n\r\n if (this._tintTexture) {\r\n activeTextures.push(this._tintTexture);\r\n }\r\n }\r\n\r\n public getAnimatables(animatables: IAnimatable[]): void {\r\n if (this._texture && this._texture.animations && this._texture.animations.length > 0) {\r\n animatables.push(this._texture);\r\n }\r\n\r\n if (this._textureRoughness && this._textureRoughness.animations && this._textureRoughness.animations.length > 0) {\r\n animatables.push(this._textureRoughness);\r\n }\r\n\r\n if (this._bumpTexture && this._bumpTexture.animations && this._bumpTexture.animations.length > 0) {\r\n animatables.push(this._bumpTexture);\r\n }\r\n\r\n if (this._tintTexture && this._tintTexture.animations && this._tintTexture.animations.length > 0) {\r\n animatables.push(this._tintTexture);\r\n }\r\n }\r\n\r\n public dispose(forceDisposeTextures?: boolean): void {\r\n if (forceDisposeTextures) {\r\n this._texture?.dispose();\r\n this._textureRoughness?.dispose();\r\n this._bumpTexture?.dispose();\r\n this._tintTexture?.dispose();\r\n }\r\n }\r\n\r\n public getClassName(): string {\r\n return \"PBRClearCoatConfiguration\";\r\n }\r\n\r\n public addFallbacks(defines: MaterialClearCoatDefines, fallbacks: EffectFallbacks, currentRank: number): number {\r\n if (defines.CLEARCOAT_BUMP) {\r\n fallbacks.addFallback(currentRank++, \"CLEARCOAT_BUMP\");\r\n }\r\n if (defines.CLEARCOAT_TINT) {\r\n fallbacks.addFallback(currentRank++, \"CLEARCOAT_TINT\");\r\n }\r\n if (defines.CLEARCOAT) {\r\n fallbacks.addFallback(currentRank++, \"CLEARCOAT\");\r\n }\r\n return currentRank;\r\n }\r\n\r\n public getSamplers(samplers: string[]): void {\r\n samplers.push(\"clearCoatSampler\", \"clearCoatRoughnessSampler\", \"clearCoatBumpSampler\", \"clearCoatTintSampler\");\r\n }\r\n\r\n public getUniforms(): { ubo?: Array<{ name: string; size: number; type: string }>; vertex?: string; fragment?: string } {\r\n return {\r\n ubo: [\r\n { name: \"vClearCoatParams\", size: 2, type: \"vec2\" },\r\n { name: \"vClearCoatRefractionParams\", size: 4, type: \"vec4\" },\r\n { name: \"vClearCoatInfos\", size: 4, type: \"vec4\" },\r\n { name: \"clearCoatMatrix\", size: 16, type: \"mat4\" },\r\n { name: \"clearCoatRoughnessMatrix\", size: 16, type: \"mat4\" },\r\n { name: \"vClearCoatBumpInfos\", size: 2, type: \"vec2\" },\r\n { name: \"vClearCoatTangentSpaceParams\", size: 2, type: \"vec2\" },\r\n { name: \"clearCoatBumpMatrix\", size: 16, type: \"mat4\" },\r\n { name: \"vClearCoatTintParams\", size: 4, type: \"vec4\" },\r\n { name: \"clearCoatColorAtDistance\", size: 1, type: \"float\" },\r\n { name: \"vClearCoatTintInfos\", size: 2, type: \"vec2\" },\r\n { name: \"clearCoatTintMatrix\", size: 16, type: \"mat4\" },\r\n ],\r\n };\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport type { Nullable } from \"../../types\";\r\nimport { serialize, serializeAsTexture, expandToProperty } from \"../../Misc/decorators\";\r\nimport type { BaseTexture } from \"../../Materials/Textures/baseTexture\";\r\nimport { MaterialFlags } from \"../materialFlags\";\r\nimport type { UniformBuffer } from \"../../Materials/uniformBuffer\";\r\nimport { MaterialHelper } from \"../../Materials/materialHelper\";\r\nimport type { IAnimatable } from \"../../Animations/animatable.interface\";\r\nimport type { EffectFallbacks } from \"../effectFallbacks\";\r\nimport type { SubMesh } from \"../../Meshes/subMesh\";\r\nimport { Constants } from \"../../Engines/constants\";\r\nimport { MaterialPluginBase } from \"../materialPluginBase\";\r\nimport { MaterialDefines } from \"../materialDefines\";\r\n\r\nimport type { Engine } from \"../../Engines/engine\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { PBRBaseMaterial } from \"./pbrBaseMaterial\";\r\n\r\n/**\r\n * @internal\r\n */\r\nexport class MaterialIridescenceDefines extends MaterialDefines {\r\n public IRIDESCENCE = false;\r\n public IRIDESCENCE_TEXTURE = false;\r\n public IRIDESCENCE_TEXTUREDIRECTUV = 0;\r\n public IRIDESCENCE_THICKNESS_TEXTURE = false;\r\n public IRIDESCENCE_THICKNESS_TEXTUREDIRECTUV = 0;\r\n public IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE = false;\r\n}\r\n\r\n/**\r\n * Plugin that implements the iridescence (thin film) component of the PBR material\r\n */\r\nexport class PBRIridescenceConfiguration extends MaterialPluginBase {\r\n protected _material: PBRBaseMaterial;\r\n\r\n /**\r\n * The default minimum thickness of the thin-film layer given in nanometers (nm).\r\n * Defaults to 100 nm.\r\n * @internal\r\n */\r\n public static readonly _DefaultMinimumThickness = 100;\r\n\r\n /**\r\n * The default maximum thickness of the thin-film layer given in nanometers (nm).\r\n * Defaults to 400 nm.\r\n * @internal\r\n */\r\n public static readonly _DefaultMaximumThickness = 400;\r\n\r\n /**\r\n * The default index of refraction of the thin-film layer.\r\n * Defaults to 1.3\r\n * @internal\r\n */\r\n public static readonly _DefaultIndexOfRefraction = 1.3;\r\n\r\n private _isEnabled = false;\r\n /**\r\n * Defines if the iridescence is enabled in the material.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public isEnabled = false;\r\n\r\n /**\r\n * Defines the iridescence layer strength (between 0 and 1) it defaults to 1.\r\n */\r\n @serialize()\r\n public intensity: number = 1;\r\n\r\n /**\r\n * Defines the minimum thickness of the thin-film layer given in nanometers (nm).\r\n */\r\n @serialize()\r\n public minimumThickness: number = PBRIridescenceConfiguration._DefaultMinimumThickness;\r\n\r\n /**\r\n * Defines the maximum thickness of the thin-film layer given in nanometers (nm). This will be the thickness used if not thickness texture has been set.\r\n */\r\n @serialize()\r\n public maximumThickness: number = PBRIridescenceConfiguration._DefaultMaximumThickness;\r\n\r\n /**\r\n * Defines the maximum thickness of the thin-film layer given in nanometers (nm).\r\n */\r\n @serialize()\r\n public indexOfRefraction: number = PBRIridescenceConfiguration._DefaultIndexOfRefraction;\r\n\r\n private _texture: Nullable = null;\r\n /**\r\n * Stores the iridescence intensity in a texture (red channel)\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public texture: Nullable = null;\r\n\r\n private _thicknessTexture: Nullable = null;\r\n /**\r\n * Stores the iridescence thickness in a texture (green channel)\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public thicknessTexture: Nullable = null;\r\n\r\n /** @internal */\r\n private _internalMarkAllSubMeshesAsTexturesDirty: () => void;\r\n\r\n /** @internal */\r\n public _markAllSubMeshesAsTexturesDirty(): void {\r\n this._enable(this._isEnabled);\r\n this._internalMarkAllSubMeshesAsTexturesDirty();\r\n }\r\n\r\n constructor(material: PBRBaseMaterial, addToPluginList = true) {\r\n super(material, \"PBRIridescence\", 110, new MaterialIridescenceDefines(), addToPluginList);\r\n\r\n this._internalMarkAllSubMeshesAsTexturesDirty = material._dirtyCallbacks[Constants.MATERIAL_TextureDirtyFlag];\r\n }\r\n\r\n public isReadyForSubMesh(defines: MaterialIridescenceDefines, scene: Scene): boolean {\r\n if (!this._isEnabled) {\r\n return true;\r\n }\r\n\r\n if (defines._areTexturesDirty) {\r\n if (scene.texturesEnabled) {\r\n if (this._texture && MaterialFlags.IridescenceTextureEnabled) {\r\n if (!this._texture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n }\r\n\r\n if (this._thicknessTexture && MaterialFlags.IridescenceTextureEnabled) {\r\n if (!this._thicknessTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n }\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n public prepareDefinesBeforeAttributes(defines: MaterialIridescenceDefines, scene: Scene): void {\r\n if (this._isEnabled) {\r\n defines.IRIDESCENCE = true;\r\n defines.IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE =\r\n this._texture !== null && this._texture._texture === this._thicknessTexture?._texture && this._texture.checkTransformsAreIdentical(this._thicknessTexture);\r\n\r\n if (defines._areTexturesDirty) {\r\n if (scene.texturesEnabled) {\r\n if (this._texture && MaterialFlags.IridescenceTextureEnabled) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._texture, defines, \"IRIDESCENCE_TEXTURE\");\r\n } else {\r\n defines.IRIDESCENCE_TEXTURE = false;\r\n }\r\n\r\n if (!defines.IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE && this._thicknessTexture && MaterialFlags.IridescenceTextureEnabled) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._thicknessTexture, defines, \"IRIDESCENCE_THICKNESS_TEXTURE\");\r\n } else {\r\n defines.IRIDESCENCE_THICKNESS_TEXTURE = false;\r\n }\r\n }\r\n }\r\n } else {\r\n defines.IRIDESCENCE = false;\r\n defines.IRIDESCENCE_TEXTURE = false;\r\n defines.IRIDESCENCE_THICKNESS_TEXTURE = false;\r\n defines.IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE = false;\r\n defines.IRIDESCENCE_TEXTUREDIRECTUV = 0;\r\n defines.IRIDESCENCE_THICKNESS_TEXTUREDIRECTUV = 0;\r\n }\r\n }\r\n\r\n public bindForSubMesh(uniformBuffer: UniformBuffer, scene: Scene, engine: Engine, subMesh: SubMesh): void {\r\n if (!this._isEnabled) {\r\n return;\r\n }\r\n\r\n const defines = subMesh!.materialDefines as unknown as MaterialIridescenceDefines;\r\n\r\n const isFrozen = this._material.isFrozen;\r\n\r\n const identicalTextures = defines.IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE;\r\n\r\n if (!uniformBuffer.useUbo || !isFrozen || !uniformBuffer.isSync) {\r\n if (identicalTextures && MaterialFlags.IridescenceTextureEnabled) {\r\n uniformBuffer.updateFloat4(\"vIridescenceInfos\", this._texture!.coordinatesIndex, this._texture!.level, -1, -1);\r\n MaterialHelper.BindTextureMatrix(this._texture!, uniformBuffer, \"iridescence\");\r\n } else if ((this._texture || this._thicknessTexture) && MaterialFlags.IridescenceTextureEnabled) {\r\n uniformBuffer.updateFloat4(\r\n \"vIridescenceInfos\",\r\n this._texture?.coordinatesIndex ?? 0,\r\n this._texture?.level ?? 0,\r\n this._thicknessTexture?.coordinatesIndex ?? 0,\r\n this._thicknessTexture?.level ?? 0\r\n );\r\n if (this._texture) {\r\n MaterialHelper.BindTextureMatrix(this._texture, uniformBuffer, \"iridescence\");\r\n }\r\n if (this._thicknessTexture && !identicalTextures && !defines.IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE) {\r\n MaterialHelper.BindTextureMatrix(this._thicknessTexture, uniformBuffer, \"iridescenceThickness\");\r\n }\r\n }\r\n\r\n // Clear Coat General params\r\n uniformBuffer.updateFloat4(\"vIridescenceParams\", this.intensity, this.indexOfRefraction, this.minimumThickness, this.maximumThickness);\r\n }\r\n\r\n // Textures\r\n if (scene.texturesEnabled) {\r\n if (this._texture && MaterialFlags.IridescenceTextureEnabled) {\r\n uniformBuffer.setTexture(\"iridescenceSampler\", this._texture);\r\n }\r\n\r\n if (this._thicknessTexture && !identicalTextures && !defines.IRIDESCENCE_USE_THICKNESS_FROM_MAINTEXTURE && MaterialFlags.IridescenceTextureEnabled) {\r\n uniformBuffer.setTexture(\"iridescenceThicknessSampler\", this._thicknessTexture);\r\n }\r\n }\r\n }\r\n\r\n public hasTexture(texture: BaseTexture): boolean {\r\n if (this._texture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._thicknessTexture === texture) {\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n public getActiveTextures(activeTextures: BaseTexture[]): void {\r\n if (this._texture) {\r\n activeTextures.push(this._texture);\r\n }\r\n\r\n if (this._thicknessTexture) {\r\n activeTextures.push(this._thicknessTexture);\r\n }\r\n }\r\n\r\n public getAnimatables(animatables: IAnimatable[]): void {\r\n if (this._texture && this._texture.animations && this._texture.animations.length > 0) {\r\n animatables.push(this._texture);\r\n }\r\n\r\n if (this._thicknessTexture && this._thicknessTexture.animations && this._thicknessTexture.animations.length > 0) {\r\n animatables.push(this._thicknessTexture);\r\n }\r\n }\r\n\r\n public dispose(forceDisposeTextures?: boolean): void {\r\n if (forceDisposeTextures) {\r\n this._texture?.dispose();\r\n this._thicknessTexture?.dispose();\r\n }\r\n }\r\n\r\n public getClassName(): string {\r\n return \"PBRIridescenceConfiguration\";\r\n }\r\n\r\n public addFallbacks(defines: MaterialIridescenceDefines, fallbacks: EffectFallbacks, currentRank: number): number {\r\n if (defines.IRIDESCENCE) {\r\n fallbacks.addFallback(currentRank++, \"IRIDESCENCE\");\r\n }\r\n return currentRank;\r\n }\r\n\r\n public getSamplers(samplers: string[]): void {\r\n samplers.push(\"iridescenceSampler\", \"iridescenceThicknessSampler\");\r\n }\r\n\r\n public getUniforms(): { ubo?: Array<{ name: string; size: number; type: string }>; vertex?: string; fragment?: string } {\r\n return {\r\n ubo: [\r\n { name: \"vIridescenceParams\", size: 4, type: \"vec4\" },\r\n { name: \"vIridescenceInfos\", size: 4, type: \"vec4\" },\r\n { name: \"iridescenceMatrix\", size: 16, type: \"mat4\" },\r\n { name: \"iridescenceThicknessMatrix\", size: 16, type: \"mat4\" },\r\n ],\r\n };\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport { serialize, expandToProperty, serializeAsVector2, serializeAsTexture } from \"../../Misc/decorators\";\r\nimport type { UniformBuffer } from \"../../Materials/uniformBuffer\";\r\nimport { VertexBuffer } from \"../../Buffers/buffer\";\r\nimport { Vector2 } from \"../../Maths/math.vector\";\r\nimport { MaterialFlags } from \"../../Materials/materialFlags\";\r\nimport { MaterialHelper } from \"../../Materials/materialHelper\";\r\nimport type { BaseTexture } from \"../../Materials/Textures/baseTexture\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { IAnimatable } from \"../../Animations/animatable.interface\";\r\nimport type { EffectFallbacks } from \"../effectFallbacks\";\r\nimport { MaterialPluginBase } from \"../materialPluginBase\";\r\nimport { Constants } from \"../../Engines/constants\";\r\nimport { MaterialDefines } from \"../materialDefines\";\r\n\r\nimport type { Scene } from \"../../scene\";\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport type { PBRBaseMaterial } from \"./pbrBaseMaterial\";\r\n\r\n/**\r\n * @internal\r\n */\r\nexport class MaterialAnisotropicDefines extends MaterialDefines {\r\n public ANISOTROPIC = false;\r\n public ANISOTROPIC_TEXTURE = false;\r\n public ANISOTROPIC_TEXTUREDIRECTUV = 0;\r\n public ANISOTROPIC_LEGACY = false;\r\n public MAINUV1 = false;\r\n}\r\n\r\n/**\r\n * Plugin that implements the anisotropic component of the PBR material\r\n */\r\nexport class PBRAnisotropicConfiguration extends MaterialPluginBase {\r\n private _isEnabled = false;\r\n /**\r\n * Defines if the anisotropy is enabled in the material.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public isEnabled = false;\r\n\r\n /**\r\n * Defines the anisotropy strength (between 0 and 1) it defaults to 1.\r\n */\r\n @serialize()\r\n public intensity: number = 1;\r\n\r\n /**\r\n * Defines if the effect is along the tangents, bitangents or in between.\r\n * By default, the effect is \"stretching\" the highlights along the tangents.\r\n */\r\n @serializeAsVector2()\r\n public direction = new Vector2(1, 0);\r\n\r\n /**\r\n * Sets the anisotropy direction as an angle.\r\n */\r\n public set angle(value: number) {\r\n this.direction.x = Math.cos(value);\r\n this.direction.y = Math.sin(value);\r\n }\r\n\r\n /**\r\n * Gets the anisotropy angle value in radians.\r\n * @returns the anisotropy angle value in radians.\r\n */\r\n public get angle(): number {\r\n return Math.atan2(this.direction.y, this.direction.x);\r\n }\r\n\r\n private _texture: Nullable = null;\r\n /**\r\n * Stores the anisotropy values in a texture.\r\n * rg is direction (like normal from -1 to 1)\r\n * b is a intensity\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public texture: Nullable = null;\r\n\r\n private _legacy = false;\r\n /**\r\n * Defines if the anisotropy is in legacy mode for backwards compatibility before 6.4.0.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsMiscDirty\")\r\n public legacy: boolean = false;\r\n\r\n /** @internal */\r\n private _internalMarkAllSubMeshesAsTexturesDirty: () => void;\r\n\r\n /** @internal */\r\n public _markAllSubMeshesAsTexturesDirty(): void {\r\n this._enable(this._isEnabled);\r\n this._internalMarkAllSubMeshesAsTexturesDirty();\r\n }\r\n\r\n /** @internal */\r\n private _internalMarkAllSubMeshesAsMiscDirty: () => void;\r\n\r\n /** @internal */\r\n public _markAllSubMeshesAsMiscDirty(): void {\r\n this._enable(this._isEnabled);\r\n this._internalMarkAllSubMeshesAsMiscDirty();\r\n }\r\n\r\n constructor(material: PBRBaseMaterial, addToPluginList = true) {\r\n super(material, \"PBRAnisotropic\", 110, new MaterialAnisotropicDefines(), addToPluginList);\r\n\r\n this._internalMarkAllSubMeshesAsTexturesDirty = material._dirtyCallbacks[Constants.MATERIAL_TextureDirtyFlag];\r\n this._internalMarkAllSubMeshesAsMiscDirty = material._dirtyCallbacks[Constants.MATERIAL_MiscDirtyFlag];\r\n }\r\n\r\n public isReadyForSubMesh(defines: MaterialAnisotropicDefines, scene: Scene): boolean {\r\n if (!this._isEnabled) {\r\n return true;\r\n }\r\n\r\n if (defines._areTexturesDirty) {\r\n if (scene.texturesEnabled) {\r\n if (this._texture && MaterialFlags.AnisotropicTextureEnabled) {\r\n if (!this._texture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n }\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n public prepareDefinesBeforeAttributes(defines: MaterialAnisotropicDefines, scene: Scene, mesh: AbstractMesh): void {\r\n if (this._isEnabled) {\r\n defines.ANISOTROPIC = this._isEnabled;\r\n if (this._isEnabled && !mesh.isVerticesDataPresent(VertexBuffer.TangentKind)) {\r\n defines._needUVs = true;\r\n defines.MAINUV1 = true;\r\n }\r\n\r\n if (defines._areTexturesDirty) {\r\n if (scene.texturesEnabled) {\r\n if (this._texture && MaterialFlags.AnisotropicTextureEnabled) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._texture, defines, \"ANISOTROPIC_TEXTURE\");\r\n } else {\r\n defines.ANISOTROPIC_TEXTURE = false;\r\n }\r\n }\r\n }\r\n\r\n if (defines._areMiscDirty) {\r\n defines.ANISOTROPIC_LEGACY = this._legacy;\r\n }\r\n } else {\r\n defines.ANISOTROPIC = false;\r\n defines.ANISOTROPIC_TEXTURE = false;\r\n defines.ANISOTROPIC_TEXTUREDIRECTUV = 0;\r\n defines.ANISOTROPIC_LEGACY = false;\r\n }\r\n }\r\n\r\n public bindForSubMesh(uniformBuffer: UniformBuffer, scene: Scene): void {\r\n if (!this._isEnabled) {\r\n return;\r\n }\r\n\r\n const isFrozen = this._material.isFrozen;\r\n\r\n if (!uniformBuffer.useUbo || !isFrozen || !uniformBuffer.isSync) {\r\n if (this._texture && MaterialFlags.AnisotropicTextureEnabled) {\r\n uniformBuffer.updateFloat2(\"vAnisotropyInfos\", this._texture.coordinatesIndex, this._texture.level);\r\n MaterialHelper.BindTextureMatrix(this._texture, uniformBuffer, \"anisotropy\");\r\n }\r\n\r\n // Anisotropy\r\n uniformBuffer.updateFloat3(\"vAnisotropy\", this.direction.x, this.direction.y, this.intensity);\r\n }\r\n\r\n // Textures\r\n if (scene.texturesEnabled) {\r\n if (this._texture && MaterialFlags.AnisotropicTextureEnabled) {\r\n uniformBuffer.setTexture(\"anisotropySampler\", this._texture);\r\n }\r\n }\r\n }\r\n\r\n public hasTexture(texture: BaseTexture): boolean {\r\n if (this._texture === texture) {\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n public getActiveTextures(activeTextures: BaseTexture[]): void {\r\n if (this._texture) {\r\n activeTextures.push(this._texture);\r\n }\r\n }\r\n\r\n public getAnimatables(animatables: IAnimatable[]): void {\r\n if (this._texture && this._texture.animations && this._texture.animations.length > 0) {\r\n animatables.push(this._texture);\r\n }\r\n }\r\n\r\n public dispose(forceDisposeTextures?: boolean): void {\r\n if (forceDisposeTextures) {\r\n if (this._texture) {\r\n this._texture.dispose();\r\n }\r\n }\r\n }\r\n\r\n public getClassName(): string {\r\n return \"PBRAnisotropicConfiguration\";\r\n }\r\n\r\n public addFallbacks(defines: MaterialAnisotropicDefines, fallbacks: EffectFallbacks, currentRank: number): number {\r\n if (defines.ANISOTROPIC) {\r\n fallbacks.addFallback(currentRank++, \"ANISOTROPIC\");\r\n }\r\n return currentRank;\r\n }\r\n\r\n public getSamplers(samplers: string[]): void {\r\n samplers.push(\"anisotropySampler\");\r\n }\r\n\r\n public getUniforms(): { ubo?: Array<{ name: string; size: number; type: string }>; vertex?: string; fragment?: string } {\r\n return {\r\n ubo: [\r\n { name: \"vAnisotropy\", size: 3, type: \"vec3\" },\r\n { name: \"vAnisotropyInfos\", size: 2, type: \"vec2\" },\r\n { name: \"anisotropyMatrix\", size: 16, type: \"mat4\" },\r\n ],\r\n };\r\n }\r\n\r\n /**\r\n * Parses a anisotropy Configuration from a serialized object.\r\n * @param source - Serialized object.\r\n * @param scene Defines the scene we are parsing for\r\n * @param rootUrl Defines the rootUrl to load from\r\n */\r\n public parse(source: any, scene: Scene, rootUrl: string): void {\r\n super.parse(source, scene, rootUrl);\r\n\r\n // Backward compatibility\r\n if (source.legacy === undefined) {\r\n this.legacy = true;\r\n }\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport { serialize, expandToProperty, serializeAsColor3, serializeAsTexture } from \"../../Misc/decorators\";\r\nimport type { UniformBuffer } from \"../../Materials/uniformBuffer\";\r\nimport { Color3 } from \"../../Maths/math.color\";\r\nimport { MaterialFlags } from \"../../Materials/materialFlags\";\r\nimport { MaterialHelper } from \"../../Materials/materialHelper\";\r\nimport type { BaseTexture } from \"../../Materials/Textures/baseTexture\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { IAnimatable } from \"../../Animations/animatable.interface\";\r\nimport type { EffectFallbacks } from \"../effectFallbacks\";\r\nimport type { SubMesh } from \"../../Meshes/subMesh\";\r\nimport { Constants } from \"../../Engines/constants\";\r\nimport { MaterialPluginBase } from \"../materialPluginBase\";\r\nimport { MaterialDefines } from \"../materialDefines\";\r\n\r\nimport type { Engine } from \"../../Engines/engine\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { PBRBaseMaterial } from \"./pbrBaseMaterial\";\r\n\r\n/**\r\n * @internal\r\n */\r\nexport class MaterialSheenDefines extends MaterialDefines {\r\n public SHEEN = false;\r\n public SHEEN_TEXTURE = false;\r\n public SHEEN_GAMMATEXTURE = false;\r\n public SHEEN_TEXTURE_ROUGHNESS = false;\r\n public SHEEN_TEXTUREDIRECTUV = 0;\r\n public SHEEN_TEXTURE_ROUGHNESSDIRECTUV = 0;\r\n public SHEEN_LINKWITHALBEDO = false;\r\n public SHEEN_ROUGHNESS = false;\r\n public SHEEN_ALBEDOSCALING = false;\r\n public SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE = false;\r\n public SHEEN_TEXTURE_ROUGHNESS_IDENTICAL = false;\r\n}\r\n\r\n/**\r\n * Plugin that implements the sheen component of the PBR material.\r\n */\r\nexport class PBRSheenConfiguration extends MaterialPluginBase {\r\n private _isEnabled = false;\r\n /**\r\n * Defines if the material uses sheen.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public isEnabled = false;\r\n\r\n private _linkSheenWithAlbedo = false;\r\n /**\r\n * Defines if the sheen is linked to the sheen color.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public linkSheenWithAlbedo = false;\r\n\r\n /**\r\n * Defines the sheen intensity.\r\n */\r\n @serialize()\r\n public intensity = 1;\r\n\r\n /**\r\n * Defines the sheen color.\r\n */\r\n @serializeAsColor3()\r\n public color = Color3.White();\r\n\r\n private _texture: Nullable = null;\r\n /**\r\n * Stores the sheen tint values in a texture.\r\n * rgb is tint\r\n * a is a intensity or roughness if the roughness property has been defined and useRoughnessFromTexture is true (in that case, textureRoughness won't be used)\r\n * If the roughness property has been defined and useRoughnessFromTexture is false then the alpha channel is not used to modulate roughness\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public texture: Nullable = null;\r\n\r\n private _useRoughnessFromMainTexture = true;\r\n /**\r\n * Indicates that the alpha channel of the texture property will be used for roughness.\r\n * Has no effect if the roughness (and texture!) property is not defined\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useRoughnessFromMainTexture = true;\r\n\r\n private _roughness: Nullable = null;\r\n /**\r\n * Defines the sheen roughness.\r\n * It is not taken into account if linkSheenWithAlbedo is true.\r\n * To stay backward compatible, material roughness is used instead if sheen roughness = null\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public roughness: Nullable = null;\r\n\r\n private _textureRoughness: Nullable = null;\r\n /**\r\n * Stores the sheen roughness in a texture.\r\n * alpha channel is the roughness. This texture won't be used if the texture property is not empty and useRoughnessFromTexture is true\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public textureRoughness: Nullable = null;\r\n\r\n private _albedoScaling = false;\r\n /**\r\n * If true, the sheen effect is layered above the base BRDF with the albedo-scaling technique.\r\n * It allows the strength of the sheen effect to not depend on the base color of the material,\r\n * making it easier to setup and tweak the effect\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public albedoScaling = false;\r\n\r\n /** @internal */\r\n private _internalMarkAllSubMeshesAsTexturesDirty: () => void;\r\n\r\n /** @internal */\r\n public _markAllSubMeshesAsTexturesDirty(): void {\r\n this._enable(this._isEnabled);\r\n this._internalMarkAllSubMeshesAsTexturesDirty();\r\n }\r\n\r\n constructor(material: PBRBaseMaterial, addToPluginList = true) {\r\n super(material, \"Sheen\", 120, new MaterialSheenDefines(), addToPluginList);\r\n\r\n this._internalMarkAllSubMeshesAsTexturesDirty = material._dirtyCallbacks[Constants.MATERIAL_TextureDirtyFlag];\r\n }\r\n\r\n public isReadyForSubMesh(defines: MaterialSheenDefines, scene: Scene): boolean {\r\n if (!this._isEnabled) {\r\n return true;\r\n }\r\n\r\n if (defines._areTexturesDirty) {\r\n if (scene.texturesEnabled) {\r\n if (this._texture && MaterialFlags.SheenTextureEnabled) {\r\n if (!this._texture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n }\r\n\r\n if (this._textureRoughness && MaterialFlags.SheenTextureEnabled) {\r\n if (!this._textureRoughness.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n }\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n public prepareDefinesBeforeAttributes(defines: MaterialSheenDefines, scene: Scene): void {\r\n if (this._isEnabled) {\r\n defines.SHEEN = true;\r\n defines.SHEEN_LINKWITHALBEDO = this._linkSheenWithAlbedo;\r\n defines.SHEEN_ROUGHNESS = this._roughness !== null;\r\n defines.SHEEN_ALBEDOSCALING = this._albedoScaling;\r\n defines.SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE = this._useRoughnessFromMainTexture;\r\n defines.SHEEN_TEXTURE_ROUGHNESS_IDENTICAL =\r\n this._texture !== null && this._texture._texture === this._textureRoughness?._texture && this._texture.checkTransformsAreIdentical(this._textureRoughness);\r\n\r\n if (defines._areTexturesDirty) {\r\n if (scene.texturesEnabled) {\r\n if (this._texture && MaterialFlags.SheenTextureEnabled) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._texture, defines, \"SHEEN_TEXTURE\");\r\n defines.SHEEN_GAMMATEXTURE = this._texture.gammaSpace;\r\n } else {\r\n defines.SHEEN_TEXTURE = false;\r\n }\r\n\r\n if (this._textureRoughness && MaterialFlags.SheenTextureEnabled) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._textureRoughness, defines, \"SHEEN_TEXTURE_ROUGHNESS\");\r\n } else {\r\n defines.SHEEN_TEXTURE_ROUGHNESS = false;\r\n }\r\n }\r\n }\r\n } else {\r\n defines.SHEEN = false;\r\n defines.SHEEN_TEXTURE = false;\r\n defines.SHEEN_TEXTURE_ROUGHNESS = false;\r\n defines.SHEEN_LINKWITHALBEDO = false;\r\n defines.SHEEN_ROUGHNESS = false;\r\n defines.SHEEN_ALBEDOSCALING = false;\r\n defines.SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE = false;\r\n defines.SHEEN_TEXTURE_ROUGHNESS_IDENTICAL = false;\r\n defines.SHEEN_GAMMATEXTURE = false;\r\n defines.SHEEN_TEXTUREDIRECTUV = 0;\r\n defines.SHEEN_TEXTURE_ROUGHNESSDIRECTUV = 0;\r\n }\r\n }\r\n\r\n public bindForSubMesh(uniformBuffer: UniformBuffer, scene: Scene, engine: Engine, subMesh: SubMesh): void {\r\n if (!this._isEnabled) {\r\n return;\r\n }\r\n\r\n const defines = subMesh!.materialDefines as unknown as MaterialSheenDefines;\r\n\r\n const isFrozen = this._material.isFrozen;\r\n\r\n const identicalTextures = defines.SHEEN_TEXTURE_ROUGHNESS_IDENTICAL;\r\n\r\n if (!uniformBuffer.useUbo || !isFrozen || !uniformBuffer.isSync) {\r\n if (identicalTextures && MaterialFlags.SheenTextureEnabled) {\r\n uniformBuffer.updateFloat4(\"vSheenInfos\", this._texture!.coordinatesIndex, this._texture!.level, -1, -1);\r\n MaterialHelper.BindTextureMatrix(this._texture!, uniformBuffer, \"sheen\");\r\n } else if ((this._texture || this._textureRoughness) && MaterialFlags.SheenTextureEnabled) {\r\n uniformBuffer.updateFloat4(\r\n \"vSheenInfos\",\r\n this._texture?.coordinatesIndex ?? 0,\r\n this._texture?.level ?? 0,\r\n this._textureRoughness?.coordinatesIndex ?? 0,\r\n this._textureRoughness?.level ?? 0\r\n );\r\n if (this._texture) {\r\n MaterialHelper.BindTextureMatrix(this._texture, uniformBuffer, \"sheen\");\r\n }\r\n if (this._textureRoughness && !identicalTextures && !defines.SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE) {\r\n MaterialHelper.BindTextureMatrix(this._textureRoughness, uniformBuffer, \"sheenRoughness\");\r\n }\r\n }\r\n\r\n // Sheen\r\n uniformBuffer.updateFloat4(\"vSheenColor\", this.color.r, this.color.g, this.color.b, this.intensity);\r\n\r\n if (this._roughness !== null) {\r\n uniformBuffer.updateFloat(\"vSheenRoughness\", this._roughness);\r\n }\r\n }\r\n\r\n // Textures\r\n if (scene.texturesEnabled) {\r\n if (this._texture && MaterialFlags.SheenTextureEnabled) {\r\n uniformBuffer.setTexture(\"sheenSampler\", this._texture);\r\n }\r\n\r\n if (this._textureRoughness && !identicalTextures && !defines.SHEEN_USE_ROUGHNESS_FROM_MAINTEXTURE && MaterialFlags.SheenTextureEnabled) {\r\n uniformBuffer.setTexture(\"sheenRoughnessSampler\", this._textureRoughness);\r\n }\r\n }\r\n }\r\n\r\n public hasTexture(texture: BaseTexture): boolean {\r\n if (this._texture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._textureRoughness === texture) {\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n public getActiveTextures(activeTextures: BaseTexture[]): void {\r\n if (this._texture) {\r\n activeTextures.push(this._texture);\r\n }\r\n\r\n if (this._textureRoughness) {\r\n activeTextures.push(this._textureRoughness);\r\n }\r\n }\r\n\r\n public getAnimatables(animatables: IAnimatable[]): void {\r\n if (this._texture && this._texture.animations && this._texture.animations.length > 0) {\r\n animatables.push(this._texture);\r\n }\r\n\r\n if (this._textureRoughness && this._textureRoughness.animations && this._textureRoughness.animations.length > 0) {\r\n animatables.push(this._textureRoughness);\r\n }\r\n }\r\n\r\n public dispose(forceDisposeTextures?: boolean): void {\r\n if (forceDisposeTextures) {\r\n this._texture?.dispose();\r\n this._textureRoughness?.dispose();\r\n }\r\n }\r\n\r\n public getClassName(): string {\r\n return \"PBRSheenConfiguration\";\r\n }\r\n\r\n public addFallbacks(defines: MaterialSheenDefines, fallbacks: EffectFallbacks, currentRank: number): number {\r\n if (defines.SHEEN) {\r\n fallbacks.addFallback(currentRank++, \"SHEEN\");\r\n }\r\n return currentRank;\r\n }\r\n\r\n public getSamplers(samplers: string[]): void {\r\n samplers.push(\"sheenSampler\", \"sheenRoughnessSampler\");\r\n }\r\n\r\n public getUniforms(): { ubo?: Array<{ name: string; size: number; type: string }>; vertex?: string; fragment?: string } {\r\n return {\r\n ubo: [\r\n { name: \"vSheenColor\", size: 4, type: \"vec4\" },\r\n { name: \"vSheenRoughness\", size: 1, type: \"float\" },\r\n { name: \"vSheenInfos\", size: 4, type: \"vec4\" },\r\n { name: \"sheenMatrix\", size: 16, type: \"mat4\" },\r\n { name: \"sheenRoughnessMatrix\", size: 16, type: \"mat4\" },\r\n ],\r\n };\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport type { Nullable } from \"../../types\";\r\nimport type { IAnimatable } from \"../../Animations/animatable.interface\";\r\nimport { serialize, serializeAsTexture, expandToProperty, serializeAsColor3 } from \"../../Misc/decorators\";\r\nimport { Color3 } from \"../../Maths/math.color\";\r\nimport type { SmartArray } from \"../../Misc/smartArray\";\r\nimport type { BaseTexture } from \"../../Materials/Textures/baseTexture\";\r\nimport type { RenderTargetTexture } from \"../../Materials/Textures/renderTargetTexture\";\r\nimport { MaterialFlags } from \"../materialFlags\";\r\nimport type { UniformBuffer } from \"../../Materials/uniformBuffer\";\r\nimport { MaterialHelper } from \"../../Materials/materialHelper\";\r\nimport type { EffectFallbacks } from \"../effectFallbacks\";\r\nimport { Scalar } from \"../../Maths/math.scalar\";\r\nimport type { CubeTexture } from \"../Textures/cubeTexture\";\r\nimport { TmpVectors } from \"../../Maths/math.vector\";\r\nimport type { SubMesh } from \"../../Meshes/subMesh\";\r\nimport { MaterialPluginBase } from \"../materialPluginBase\";\r\nimport { Constants } from \"../../Engines/constants\";\r\nimport { MaterialDefines } from \"../materialDefines\";\r\n\r\nimport type { Engine } from \"../../Engines/engine\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { PBRBaseMaterial } from \"./pbrBaseMaterial\";\r\n\r\n/**\r\n * @internal\r\n */\r\nexport class MaterialSubSurfaceDefines extends MaterialDefines {\r\n public SUBSURFACE = false;\r\n\r\n public SS_REFRACTION = false;\r\n public SS_REFRACTION_USE_INTENSITY_FROM_TEXTURE = false;\r\n public SS_TRANSLUCENCY = false;\r\n public SS_TRANSLUCENCY_USE_INTENSITY_FROM_TEXTURE = false;\r\n public SS_SCATTERING = false;\r\n\r\n public SS_THICKNESSANDMASK_TEXTURE = false;\r\n public SS_THICKNESSANDMASK_TEXTUREDIRECTUV = 0;\r\n public SS_HAS_THICKNESS = false;\r\n public SS_REFRACTIONINTENSITY_TEXTURE = false;\r\n public SS_REFRACTIONINTENSITY_TEXTUREDIRECTUV = 0;\r\n public SS_TRANSLUCENCYINTENSITY_TEXTURE = false;\r\n public SS_TRANSLUCENCYINTENSITY_TEXTUREDIRECTUV = 0;\r\n\r\n public SS_REFRACTIONMAP_3D = false;\r\n public SS_REFRACTIONMAP_OPPOSITEZ = false;\r\n public SS_LODINREFRACTIONALPHA = false;\r\n public SS_GAMMAREFRACTION = false;\r\n public SS_RGBDREFRACTION = false;\r\n public SS_LINEARSPECULARREFRACTION = false;\r\n public SS_LINKREFRACTIONTOTRANSPARENCY = false;\r\n public SS_ALBEDOFORREFRACTIONTINT = false;\r\n public SS_ALBEDOFORTRANSLUCENCYTINT = false;\r\n public SS_USE_LOCAL_REFRACTIONMAP_CUBIC = false;\r\n public SS_USE_THICKNESS_AS_DEPTH = false;\r\n\r\n public SS_MASK_FROM_THICKNESS_TEXTURE = false;\r\n public SS_USE_GLTF_TEXTURES = false;\r\n}\r\n\r\n/**\r\n * Plugin that implements the sub surface component of the PBR material\r\n */\r\nexport class PBRSubSurfaceConfiguration extends MaterialPluginBase {\r\n protected _material: PBRBaseMaterial;\r\n\r\n private _isRefractionEnabled = false;\r\n /**\r\n * Defines if the refraction is enabled in the material.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public isRefractionEnabled = false;\r\n\r\n private _isTranslucencyEnabled = false;\r\n /**\r\n * Defines if the translucency is enabled in the material.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public isTranslucencyEnabled = false;\r\n\r\n private _isScatteringEnabled = false;\r\n /**\r\n * Defines if the sub surface scattering is enabled in the material.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markScenePrePassDirty\")\r\n public isScatteringEnabled = false;\r\n\r\n @serialize()\r\n private _scatteringDiffusionProfileIndex = 0;\r\n\r\n /**\r\n * Diffusion profile for subsurface scattering.\r\n * Useful for better scattering in the skins or foliages.\r\n */\r\n public get scatteringDiffusionProfile(): Nullable {\r\n if (!this._scene.subSurfaceConfiguration) {\r\n return null;\r\n }\r\n\r\n return this._scene.subSurfaceConfiguration.ssDiffusionProfileColors[this._scatteringDiffusionProfileIndex];\r\n }\r\n\r\n public set scatteringDiffusionProfile(c: Nullable) {\r\n if (!this._scene.enableSubSurfaceForPrePass()) {\r\n // Not supported\r\n return;\r\n }\r\n\r\n // addDiffusionProfile automatically checks for doubles\r\n if (c) {\r\n this._scatteringDiffusionProfileIndex = this._scene.subSurfaceConfiguration!.addDiffusionProfile(c);\r\n }\r\n }\r\n\r\n /**\r\n * Defines the refraction intensity of the material.\r\n * The refraction when enabled replaces the Diffuse part of the material.\r\n * The intensity helps transitioning between diffuse and refraction.\r\n */\r\n @serialize()\r\n public refractionIntensity: number = 1;\r\n\r\n /**\r\n * Defines the translucency intensity of the material.\r\n * When translucency has been enabled, this defines how much of the \"translucency\"\r\n * is added to the diffuse part of the material.\r\n */\r\n @serialize()\r\n public translucencyIntensity: number = 1;\r\n\r\n /**\r\n * When enabled, transparent surfaces will be tinted with the albedo colour (independent of thickness)\r\n */\r\n @serialize()\r\n public useAlbedoToTintRefraction: boolean = false;\r\n\r\n /**\r\n * When enabled, translucent surfaces will be tinted with the albedo colour (independent of thickness)\r\n */\r\n @serialize()\r\n public useAlbedoToTintTranslucency: boolean = false;\r\n\r\n private _thicknessTexture: Nullable = null;\r\n /**\r\n * Stores the average thickness of a mesh in a texture (The texture is holding the values linearly).\r\n * The red (or green if useGltfStyleTextures=true) channel of the texture should contain the thickness remapped between 0 and 1.\r\n * 0 would mean minimumThickness\r\n * 1 would mean maximumThickness\r\n * The other channels might be use as a mask to vary the different effects intensity.\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public thicknessTexture: Nullable = null;\r\n\r\n private _refractionTexture: Nullable = null;\r\n /**\r\n * Defines the texture to use for refraction.\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public refractionTexture: Nullable = null;\r\n\r\n /** @internal */\r\n public _indexOfRefraction = 1.5;\r\n /**\r\n * Index of refraction of the material base layer.\r\n * https://en.wikipedia.org/wiki/List_of_refractive_indices\r\n *\r\n * This does not only impact refraction but also the Base F0 of Dielectric Materials.\r\n *\r\n * From dielectric fresnel rules: F0 = square((iorT - iorI) / (iorT + iorI))\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public indexOfRefraction = 1.5;\r\n\r\n @serialize()\r\n private _volumeIndexOfRefraction = -1.0;\r\n\r\n /**\r\n * Index of refraction of the material's volume.\r\n * https://en.wikipedia.org/wiki/List_of_refractive_indices\r\n *\r\n * This ONLY impacts refraction. If not provided or given a non-valid value,\r\n * the volume will use the same IOR as the surface.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public get volumeIndexOfRefraction(): number {\r\n if (this._volumeIndexOfRefraction >= 1.0) {\r\n return this._volumeIndexOfRefraction;\r\n }\r\n return this._indexOfRefraction;\r\n }\r\n public set volumeIndexOfRefraction(value: number) {\r\n if (value >= 1.0) {\r\n this._volumeIndexOfRefraction = value;\r\n } else {\r\n this._volumeIndexOfRefraction = -1.0;\r\n }\r\n }\r\n\r\n private _invertRefractionY = false;\r\n /**\r\n * Controls if refraction needs to be inverted on Y. This could be useful for procedural texture.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public invertRefractionY = false;\r\n\r\n /** @internal */\r\n public _linkRefractionWithTransparency = false;\r\n /**\r\n * This parameters will make the material used its opacity to control how much it is refracting against not.\r\n * Materials half opaque for instance using refraction could benefit from this control.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public linkRefractionWithTransparency = false;\r\n\r\n /**\r\n * Defines the minimum thickness stored in the thickness map.\r\n * If no thickness map is defined, this value will be used to simulate thickness.\r\n */\r\n @serialize()\r\n public minimumThickness: number = 0;\r\n\r\n /**\r\n * Defines the maximum thickness stored in the thickness map.\r\n */\r\n @serialize()\r\n public maximumThickness: number = 1;\r\n\r\n /**\r\n * Defines that the thickness should be used as a measure of the depth volume.\r\n */\r\n @serialize()\r\n public useThicknessAsDepth = false;\r\n\r\n /**\r\n * Defines the volume tint of the material.\r\n * This is used for both translucency and scattering.\r\n */\r\n @serializeAsColor3()\r\n public tintColor = Color3.White();\r\n\r\n /**\r\n * Defines the distance at which the tint color should be found in the media.\r\n * This is used for refraction only.\r\n */\r\n @serialize()\r\n public tintColorAtDistance = 1;\r\n\r\n /**\r\n * Defines how far each channel transmit through the media.\r\n * It is defined as a color to simplify it selection.\r\n */\r\n @serializeAsColor3()\r\n public diffusionDistance = Color3.White();\r\n\r\n private _useMaskFromThicknessTexture = false;\r\n /**\r\n * Stores the intensity of the different subsurface effects in the thickness texture.\r\n * Note that if refractionIntensityTexture and/or translucencyIntensityTexture is provided it takes precedence over thicknessTexture + useMaskFromThicknessTexture\r\n * * the green (red if useGltfStyleTextures = true) channel is the refraction intensity.\r\n * * the blue channel is the translucency intensity.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useMaskFromThicknessTexture: boolean = false;\r\n\r\n private _refractionIntensityTexture: Nullable = null;\r\n /**\r\n * Stores the intensity of the refraction. If provided, it takes precedence over thicknessTexture + useMaskFromThicknessTexture\r\n * * the green (red if useGltfStyleTextures = true) channel is the refraction intensity.\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public refractionIntensityTexture: Nullable = null;\r\n\r\n private _translucencyIntensityTexture: Nullable = null;\r\n /**\r\n * Stores the intensity of the translucency. If provided, it takes precedence over thicknessTexture + useMaskFromThicknessTexture\r\n * * the blue channel is the translucency intensity.\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public translucencyIntensityTexture: Nullable = null;\r\n\r\n private _scene: Scene;\r\n private _useGltfStyleTextures = false;\r\n /**\r\n * Use channels layout used by glTF:\r\n * * thicknessTexture: the green (instead of red) channel is the thickness\r\n * * thicknessTexture/refractionIntensityTexture: the red (instead of green) channel is the refraction intensity\r\n * * thicknessTexture/translucencyIntensityTexture: no change, use the blue channel for the translucency intensity\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useGltfStyleTextures: boolean = false;\r\n\r\n /** @internal */\r\n private _internalMarkAllSubMeshesAsTexturesDirty: () => void;\r\n private _internalMarkScenePrePassDirty: () => void;\r\n\r\n /** @internal */\r\n public _markAllSubMeshesAsTexturesDirty(): void {\r\n this._enable(this._isRefractionEnabled || this._isTranslucencyEnabled || this._isScatteringEnabled);\r\n this._internalMarkAllSubMeshesAsTexturesDirty();\r\n }\r\n /** @internal */\r\n public _markScenePrePassDirty(): void {\r\n this._internalMarkAllSubMeshesAsTexturesDirty();\r\n this._internalMarkScenePrePassDirty();\r\n }\r\n\r\n constructor(material: PBRBaseMaterial, addToPluginList = true) {\r\n super(material, \"PBRSubSurface\", 130, new MaterialSubSurfaceDefines(), addToPluginList);\r\n\r\n this._scene = material.getScene();\r\n this.registerForExtraEvents = true;\r\n\r\n this._internalMarkAllSubMeshesAsTexturesDirty = material._dirtyCallbacks[Constants.MATERIAL_TextureDirtyFlag];\r\n this._internalMarkScenePrePassDirty = material._dirtyCallbacks[Constants.MATERIAL_PrePassDirtyFlag];\r\n }\r\n\r\n public isReadyForSubMesh(defines: MaterialSubSurfaceDefines, scene: Scene): boolean {\r\n if (!this._isRefractionEnabled && !this._isTranslucencyEnabled && !this._isScatteringEnabled) {\r\n return true;\r\n }\r\n\r\n if (defines._areTexturesDirty) {\r\n if (scene.texturesEnabled) {\r\n if (this._thicknessTexture && MaterialFlags.ThicknessTextureEnabled) {\r\n if (!this._thicknessTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n }\r\n\r\n const refractionTexture = this._getRefractionTexture(scene);\r\n if (refractionTexture && MaterialFlags.RefractionTextureEnabled) {\r\n if (!refractionTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n }\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n public prepareDefinesBeforeAttributes(defines: MaterialSubSurfaceDefines, scene: Scene): void {\r\n if (!this._isRefractionEnabled && !this._isTranslucencyEnabled && !this._isScatteringEnabled) {\r\n defines.SUBSURFACE = false;\r\n defines.SS_TRANSLUCENCY = false;\r\n defines.SS_SCATTERING = false;\r\n defines.SS_REFRACTION = false;\r\n defines.SS_REFRACTION_USE_INTENSITY_FROM_TEXTURE = false;\r\n defines.SS_TRANSLUCENCY_USE_INTENSITY_FROM_TEXTURE = false;\r\n defines.SS_THICKNESSANDMASK_TEXTURE = false;\r\n defines.SS_THICKNESSANDMASK_TEXTUREDIRECTUV = 0;\r\n defines.SS_HAS_THICKNESS = false;\r\n defines.SS_REFRACTIONINTENSITY_TEXTURE = false;\r\n defines.SS_REFRACTIONINTENSITY_TEXTUREDIRECTUV = 0;\r\n defines.SS_TRANSLUCENCYINTENSITY_TEXTURE = false;\r\n defines.SS_TRANSLUCENCYINTENSITY_TEXTUREDIRECTUV = 0;\r\n defines.SS_REFRACTIONMAP_3D = false;\r\n defines.SS_REFRACTIONMAP_OPPOSITEZ = false;\r\n defines.SS_LODINREFRACTIONALPHA = false;\r\n defines.SS_GAMMAREFRACTION = false;\r\n defines.SS_RGBDREFRACTION = false;\r\n defines.SS_LINEARSPECULARREFRACTION = false;\r\n defines.SS_LINKREFRACTIONTOTRANSPARENCY = false;\r\n defines.SS_ALBEDOFORREFRACTIONTINT = false;\r\n defines.SS_ALBEDOFORTRANSLUCENCYTINT = false;\r\n defines.SS_USE_LOCAL_REFRACTIONMAP_CUBIC = false;\r\n defines.SS_USE_THICKNESS_AS_DEPTH = false;\r\n defines.SS_MASK_FROM_THICKNESS_TEXTURE = false;\r\n defines.SS_USE_GLTF_TEXTURES = false;\r\n return;\r\n }\r\n\r\n if (defines._areTexturesDirty) {\r\n defines.SUBSURFACE = true;\r\n\r\n defines.SS_TRANSLUCENCY = this._isTranslucencyEnabled;\r\n defines.SS_TRANSLUCENCY_USE_INTENSITY_FROM_TEXTURE = false;\r\n defines.SS_SCATTERING = this._isScatteringEnabled;\r\n defines.SS_THICKNESSANDMASK_TEXTURE = false;\r\n defines.SS_REFRACTIONINTENSITY_TEXTURE = false;\r\n defines.SS_TRANSLUCENCYINTENSITY_TEXTURE = false;\r\n defines.SS_HAS_THICKNESS = false;\r\n defines.SS_MASK_FROM_THICKNESS_TEXTURE = false;\r\n defines.SS_USE_GLTF_TEXTURES = false;\r\n defines.SS_REFRACTION = false;\r\n defines.SS_REFRACTION_USE_INTENSITY_FROM_TEXTURE = false;\r\n defines.SS_REFRACTIONMAP_3D = false;\r\n defines.SS_GAMMAREFRACTION = false;\r\n defines.SS_RGBDREFRACTION = false;\r\n defines.SS_LINEARSPECULARREFRACTION = false;\r\n defines.SS_REFRACTIONMAP_OPPOSITEZ = false;\r\n defines.SS_LODINREFRACTIONALPHA = false;\r\n defines.SS_LINKREFRACTIONTOTRANSPARENCY = false;\r\n defines.SS_ALBEDOFORREFRACTIONTINT = false;\r\n defines.SS_ALBEDOFORTRANSLUCENCYTINT = false;\r\n defines.SS_USE_LOCAL_REFRACTIONMAP_CUBIC = false;\r\n defines.SS_USE_THICKNESS_AS_DEPTH = false;\r\n\r\n const refractionIntensityTextureIsThicknessTexture =\r\n !!this._thicknessTexture &&\r\n !!this._refractionIntensityTexture &&\r\n this._refractionIntensityTexture.checkTransformsAreIdentical(this._thicknessTexture) &&\r\n this._refractionIntensityTexture._texture === this._thicknessTexture._texture;\r\n\r\n const translucencyIntensityTextureIsThicknessTexture =\r\n !!this._thicknessTexture &&\r\n !!this._translucencyIntensityTexture &&\r\n this._translucencyIntensityTexture.checkTransformsAreIdentical(this._thicknessTexture) &&\r\n this._translucencyIntensityTexture._texture === this._thicknessTexture._texture;\r\n\r\n // if true, it means the refraction/translucency textures are the same than the thickness texture so there's no need to pass them to the shader, only thicknessTexture\r\n const useOnlyThicknessTexture =\r\n (refractionIntensityTextureIsThicknessTexture || !this._refractionIntensityTexture) &&\r\n (translucencyIntensityTextureIsThicknessTexture || !this._translucencyIntensityTexture);\r\n\r\n if (defines._areTexturesDirty) {\r\n if (scene.texturesEnabled) {\r\n if (this._thicknessTexture && MaterialFlags.ThicknessTextureEnabled) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._thicknessTexture, defines, \"SS_THICKNESSANDMASK_TEXTURE\");\r\n }\r\n\r\n if (this._refractionIntensityTexture && MaterialFlags.RefractionIntensityTextureEnabled && !useOnlyThicknessTexture) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._refractionIntensityTexture, defines, \"SS_REFRACTIONINTENSITY_TEXTURE\");\r\n }\r\n\r\n if (this._translucencyIntensityTexture && MaterialFlags.TranslucencyIntensityTextureEnabled && !useOnlyThicknessTexture) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._translucencyIntensityTexture, defines, \"SS_TRANSLUCENCYINTENSITY_TEXTURE\");\r\n }\r\n }\r\n }\r\n\r\n defines.SS_HAS_THICKNESS = this.maximumThickness - this.minimumThickness !== 0.0;\r\n defines.SS_MASK_FROM_THICKNESS_TEXTURE =\r\n (this._useMaskFromThicknessTexture || !!this._refractionIntensityTexture || !!this._translucencyIntensityTexture) && useOnlyThicknessTexture;\r\n defines.SS_USE_GLTF_TEXTURES = this._useGltfStyleTextures;\r\n defines.SS_REFRACTION_USE_INTENSITY_FROM_TEXTURE = (this._useMaskFromThicknessTexture || !!this._refractionIntensityTexture) && useOnlyThicknessTexture;\r\n defines.SS_TRANSLUCENCY_USE_INTENSITY_FROM_TEXTURE = (this._useMaskFromThicknessTexture || !!this._translucencyIntensityTexture) && useOnlyThicknessTexture;\r\n\r\n if (this._isRefractionEnabled) {\r\n if (scene.texturesEnabled) {\r\n const refractionTexture = this._getRefractionTexture(scene);\r\n if (refractionTexture && MaterialFlags.RefractionTextureEnabled) {\r\n defines.SS_REFRACTION = true;\r\n defines.SS_REFRACTIONMAP_3D = refractionTexture.isCube;\r\n defines.SS_GAMMAREFRACTION = refractionTexture.gammaSpace;\r\n defines.SS_RGBDREFRACTION = refractionTexture.isRGBD;\r\n defines.SS_LINEARSPECULARREFRACTION = refractionTexture.linearSpecularLOD;\r\n defines.SS_REFRACTIONMAP_OPPOSITEZ = this._scene.useRightHandedSystem && refractionTexture.isCube ? !refractionTexture.invertZ : refractionTexture.invertZ;\r\n defines.SS_LODINREFRACTIONALPHA = refractionTexture.lodLevelInAlpha;\r\n defines.SS_LINKREFRACTIONTOTRANSPARENCY = this._linkRefractionWithTransparency;\r\n defines.SS_ALBEDOFORREFRACTIONTINT = this.useAlbedoToTintRefraction;\r\n defines.SS_USE_LOCAL_REFRACTIONMAP_CUBIC = refractionTexture.isCube && (refractionTexture).boundingBoxSize;\r\n defines.SS_USE_THICKNESS_AS_DEPTH = this.useThicknessAsDepth;\r\n }\r\n }\r\n }\r\n\r\n if (this._isTranslucencyEnabled) {\r\n defines.SS_ALBEDOFORTRANSLUCENCYTINT = this.useAlbedoToTintTranslucency;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Binds the material data (this function is called even if mustRebind() returns false)\r\n * @param uniformBuffer defines the Uniform buffer to fill in.\r\n * @param scene defines the scene the material belongs to.\r\n * @param engine defines the engine the material belongs to.\r\n * @param subMesh the submesh to bind data for\r\n */\r\n public hardBindForSubMesh(uniformBuffer: UniformBuffer, scene: Scene, engine: Engine, subMesh: SubMesh): void {\r\n if (!this._isRefractionEnabled && !this._isTranslucencyEnabled && !this._isScatteringEnabled) {\r\n return;\r\n }\r\n\r\n subMesh.getRenderingMesh().getWorldMatrix().decompose(TmpVectors.Vector3[0]);\r\n\r\n const thicknessScale = Math.max(Math.abs(TmpVectors.Vector3[0].x), Math.abs(TmpVectors.Vector3[0].y), Math.abs(TmpVectors.Vector3[0].z));\r\n\r\n uniformBuffer.updateFloat2(\"vThicknessParam\", this.minimumThickness * thicknessScale, (this.maximumThickness - this.minimumThickness) * thicknessScale);\r\n }\r\n\r\n public bindForSubMesh(uniformBuffer: UniformBuffer, scene: Scene, engine: Engine, subMesh: SubMesh): void {\r\n if (!this._isRefractionEnabled && !this._isTranslucencyEnabled && !this._isScatteringEnabled) {\r\n return;\r\n }\r\n\r\n const defines = subMesh!.materialDefines as unknown as MaterialSubSurfaceDefines;\r\n\r\n const isFrozen = this._material.isFrozen;\r\n const realTimeFiltering = this._material.realTimeFiltering;\r\n const lodBasedMicrosurface = defines.LODBASEDMICROSFURACE;\r\n\r\n const refractionTexture = this._getRefractionTexture(scene);\r\n\r\n if (!uniformBuffer.useUbo || !isFrozen || !uniformBuffer.isSync) {\r\n if (this._thicknessTexture && MaterialFlags.ThicknessTextureEnabled) {\r\n uniformBuffer.updateFloat2(\"vThicknessInfos\", this._thicknessTexture.coordinatesIndex, this._thicknessTexture.level);\r\n MaterialHelper.BindTextureMatrix(this._thicknessTexture, uniformBuffer, \"thickness\");\r\n }\r\n\r\n if (this._refractionIntensityTexture && MaterialFlags.RefractionIntensityTextureEnabled && defines.SS_REFRACTIONINTENSITY_TEXTURE) {\r\n uniformBuffer.updateFloat2(\"vRefractionIntensityInfos\", this._refractionIntensityTexture.coordinatesIndex, this._refractionIntensityTexture.level);\r\n MaterialHelper.BindTextureMatrix(this._refractionIntensityTexture, uniformBuffer, \"refractionIntensity\");\r\n }\r\n\r\n if (this._translucencyIntensityTexture && MaterialFlags.TranslucencyIntensityTextureEnabled && defines.SS_TRANSLUCENCYINTENSITY_TEXTURE) {\r\n uniformBuffer.updateFloat2(\"vTranslucencyIntensityInfos\", this._translucencyIntensityTexture.coordinatesIndex, this._translucencyIntensityTexture.level);\r\n MaterialHelper.BindTextureMatrix(this._translucencyIntensityTexture, uniformBuffer, \"translucencyIntensity\");\r\n }\r\n\r\n if (refractionTexture && MaterialFlags.RefractionTextureEnabled) {\r\n uniformBuffer.updateMatrix(\"refractionMatrix\", refractionTexture.getRefractionTextureMatrix());\r\n\r\n let depth = 1.0;\r\n if (!refractionTexture.isCube) {\r\n if ((refractionTexture).depth) {\r\n depth = (refractionTexture).depth;\r\n }\r\n }\r\n\r\n const width = refractionTexture.getSize().width;\r\n const refractionIor = this.volumeIndexOfRefraction;\r\n uniformBuffer.updateFloat4(\"vRefractionInfos\", refractionTexture.level, 1 / refractionIor, depth, this._invertRefractionY ? -1 : 1);\r\n uniformBuffer.updateFloat4(\r\n \"vRefractionMicrosurfaceInfos\",\r\n width,\r\n refractionTexture.lodGenerationScale,\r\n refractionTexture.lodGenerationOffset,\r\n 1.0 / this.indexOfRefraction\r\n );\r\n\r\n if (realTimeFiltering) {\r\n uniformBuffer.updateFloat2(\"vRefractionFilteringInfo\", width, Scalar.Log2(width));\r\n }\r\n\r\n if ((refractionTexture).boundingBoxSize) {\r\n const cubeTexture = refractionTexture;\r\n\r\n uniformBuffer.updateVector3(\"vRefractionPosition\", cubeTexture.boundingBoxPosition);\r\n uniformBuffer.updateVector3(\"vRefractionSize\", cubeTexture.boundingBoxSize);\r\n }\r\n }\r\n\r\n if (this._isScatteringEnabled) {\r\n uniformBuffer.updateFloat(\"scatteringDiffusionProfile\", this._scatteringDiffusionProfileIndex);\r\n }\r\n uniformBuffer.updateColor3(\"vDiffusionDistance\", this.diffusionDistance);\r\n\r\n uniformBuffer.updateFloat4(\"vTintColor\", this.tintColor.r, this.tintColor.g, this.tintColor.b, Math.max(0.00001, this.tintColorAtDistance));\r\n\r\n uniformBuffer.updateFloat3(\"vSubSurfaceIntensity\", this.refractionIntensity, this.translucencyIntensity, 0);\r\n }\r\n\r\n // Textures\r\n if (scene.texturesEnabled) {\r\n if (this._thicknessTexture && MaterialFlags.ThicknessTextureEnabled) {\r\n uniformBuffer.setTexture(\"thicknessSampler\", this._thicknessTexture);\r\n }\r\n\r\n if (this._refractionIntensityTexture && MaterialFlags.RefractionIntensityTextureEnabled && defines.SS_REFRACTIONINTENSITY_TEXTURE) {\r\n uniformBuffer.setTexture(\"refractionIntensitySampler\", this._refractionIntensityTexture);\r\n }\r\n\r\n if (this._translucencyIntensityTexture && MaterialFlags.TranslucencyIntensityTextureEnabled && defines.SS_TRANSLUCENCYINTENSITY_TEXTURE) {\r\n uniformBuffer.setTexture(\"translucencyIntensitySampler\", this._translucencyIntensityTexture);\r\n }\r\n\r\n if (refractionTexture && MaterialFlags.RefractionTextureEnabled) {\r\n if (lodBasedMicrosurface) {\r\n uniformBuffer.setTexture(\"refractionSampler\", refractionTexture);\r\n } else {\r\n uniformBuffer.setTexture(\"refractionSampler\", refractionTexture._lodTextureMid || refractionTexture);\r\n uniformBuffer.setTexture(\"refractionSamplerLow\", refractionTexture._lodTextureLow || refractionTexture);\r\n uniformBuffer.setTexture(\"refractionSamplerHigh\", refractionTexture._lodTextureHigh || refractionTexture);\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Returns the texture used for refraction or null if none is used.\r\n * @param scene defines the scene the material belongs to.\r\n * @returns - Refraction texture if present. If no refraction texture and refraction\r\n * is linked with transparency, returns environment texture. Otherwise, returns null.\r\n */\r\n private _getRefractionTexture(scene: Scene): Nullable {\r\n if (this._refractionTexture) {\r\n return this._refractionTexture;\r\n }\r\n\r\n if (this._isRefractionEnabled) {\r\n return scene.environmentTexture;\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Returns true if alpha blending should be disabled.\r\n */\r\n public get disableAlphaBlending(): boolean {\r\n return this._isRefractionEnabled && this._linkRefractionWithTransparency;\r\n }\r\n\r\n /**\r\n * Fills the list of render target textures.\r\n * @param renderTargets the list of render targets to update\r\n */\r\n public fillRenderTargetTextures(renderTargets: SmartArray): void {\r\n if (MaterialFlags.RefractionTextureEnabled && this._refractionTexture && this._refractionTexture.isRenderTarget) {\r\n renderTargets.push(this._refractionTexture);\r\n }\r\n }\r\n\r\n public hasTexture(texture: BaseTexture): boolean {\r\n if (this._thicknessTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._refractionTexture === texture) {\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n public hasRenderTargetTextures(): boolean {\r\n if (MaterialFlags.RefractionTextureEnabled && this._refractionTexture && this._refractionTexture.isRenderTarget) {\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n public getActiveTextures(activeTextures: BaseTexture[]): void {\r\n if (this._thicknessTexture) {\r\n activeTextures.push(this._thicknessTexture);\r\n }\r\n\r\n if (this._refractionTexture) {\r\n activeTextures.push(this._refractionTexture);\r\n }\r\n }\r\n\r\n public getAnimatables(animatables: IAnimatable[]): void {\r\n if (this._thicknessTexture && this._thicknessTexture.animations && this._thicknessTexture.animations.length > 0) {\r\n animatables.push(this._thicknessTexture);\r\n }\r\n\r\n if (this._refractionTexture && this._refractionTexture.animations && this._refractionTexture.animations.length > 0) {\r\n animatables.push(this._refractionTexture);\r\n }\r\n }\r\n\r\n public dispose(forceDisposeTextures?: boolean): void {\r\n if (forceDisposeTextures) {\r\n if (this._thicknessTexture) {\r\n this._thicknessTexture.dispose();\r\n }\r\n\r\n if (this._refractionTexture) {\r\n this._refractionTexture.dispose();\r\n }\r\n }\r\n }\r\n\r\n public getClassName(): string {\r\n return \"PBRSubSurfaceConfiguration\";\r\n }\r\n\r\n public addFallbacks(defines: MaterialSubSurfaceDefines, fallbacks: EffectFallbacks, currentRank: number): number {\r\n if (defines.SS_SCATTERING) {\r\n fallbacks.addFallback(currentRank++, \"SS_SCATTERING\");\r\n }\r\n if (defines.SS_TRANSLUCENCY) {\r\n fallbacks.addFallback(currentRank++, \"SS_TRANSLUCENCY\");\r\n }\r\n return currentRank;\r\n }\r\n\r\n public getSamplers(samplers: string[]): void {\r\n samplers.push(\"thicknessSampler\", \"refractionIntensitySampler\", \"translucencyIntensitySampler\", \"refractionSampler\", \"refractionSamplerLow\", \"refractionSamplerHigh\");\r\n }\r\n\r\n public getUniforms(): { ubo?: Array<{ name: string; size: number; type: string }>; vertex?: string; fragment?: string } {\r\n return {\r\n ubo: [\r\n { name: \"vRefractionMicrosurfaceInfos\", size: 4, type: \"vec4\" },\r\n { name: \"vRefractionFilteringInfo\", size: 2, type: \"vec2\" },\r\n { name: \"vTranslucencyIntensityInfos\", size: 2, type: \"vec2\" },\r\n { name: \"vRefractionInfos\", size: 4, type: \"vec4\" },\r\n { name: \"refractionMatrix\", size: 16, type: \"mat4\" },\r\n { name: \"vThicknessInfos\", size: 2, type: \"vec2\" },\r\n { name: \"vRefractionIntensityInfos\", size: 2, type: \"vec2\" },\r\n { name: \"thicknessMatrix\", size: 16, type: \"mat4\" },\r\n { name: \"refractionIntensityMatrix\", size: 16, type: \"mat4\" },\r\n { name: \"translucencyIntensityMatrix\", size: 16, type: \"mat4\" },\r\n { name: \"vThicknessParam\", size: 2, type: \"vec2\" },\r\n { name: \"vDiffusionDistance\", size: 3, type: \"vec3\" },\r\n { name: \"vTintColor\", size: 4, type: \"vec4\" },\r\n { name: \"vSubSurfaceIntensity\", size: 3, type: \"vec3\" },\r\n { name: \"vRefractionPosition\", size: 3, type: \"vec3\" },\r\n { name: \"vRefractionSize\", size: 3, type: \"vec3\" },\r\n { name: \"scatteringDiffusionProfile\", size: 1, type: \"float\" },\r\n ],\r\n };\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport { serialize, serializeAsImageProcessingConfiguration, expandToProperty } from \"../../Misc/decorators\";\r\nimport type { Observer } from \"../../Misc/observable\";\r\nimport { Logger } from \"../../Misc/logger\";\r\nimport { SmartArray } from \"../../Misc/smartArray\";\r\nimport { GetEnvironmentBRDFTexture } from \"../../Misc/brdfTextureTools\";\r\nimport type { Nullable } from \"../../types\";\r\nimport { Scene } from \"../../scene\";\r\nimport type { Matrix } from \"../../Maths/math.vector\";\r\nimport { Vector4 } from \"../../Maths/math.vector\";\r\nimport { VertexBuffer } from \"../../Buffers/buffer\";\r\nimport type { SubMesh } from \"../../Meshes/subMesh\";\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport type { Mesh } from \"../../Meshes/mesh\";\r\nimport { PBRBRDFConfiguration } from \"./pbrBRDFConfiguration\";\r\nimport { PrePassConfiguration } from \"../prePassConfiguration\";\r\nimport { Color3, TmpColors } from \"../../Maths/math.color\";\r\nimport { Scalar } from \"../../Maths/math.scalar\";\r\n\r\nimport type { IImageProcessingConfigurationDefines } from \"../../Materials/imageProcessingConfiguration\";\r\nimport { ImageProcessingConfiguration } from \"../../Materials/imageProcessingConfiguration\";\r\nimport type { Effect, IEffectCreationOptions } from \"../../Materials/effect\";\r\nimport type { IMaterialCompilationOptions, ICustomShaderNameResolveOptions } from \"../../Materials/material\";\r\nimport { Material } from \"../../Materials/material\";\r\nimport { MaterialPluginEvent } from \"../materialPluginEvent\";\r\nimport { MaterialDefines } from \"../../Materials/materialDefines\";\r\nimport { PushMaterial } from \"../../Materials/pushMaterial\";\r\nimport { MaterialHelper } from \"../../Materials/materialHelper\";\r\n\r\nimport type { BaseTexture } from \"../../Materials/Textures/baseTexture\";\r\nimport { Texture } from \"../../Materials/Textures/texture\";\r\nimport type { RenderTargetTexture } from \"../../Materials/Textures/renderTargetTexture\";\r\nimport type { CubeTexture } from \"../../Materials/Textures/cubeTexture\";\r\n\r\nimport { MaterialFlags } from \"../materialFlags\";\r\nimport { Constants } from \"../../Engines/constants\";\r\nimport type { IAnimatable } from \"../../Animations/animatable.interface\";\r\n\r\nimport \"../../Materials/Textures/baseTexture.polynomial\";\r\nimport \"../../Shaders/pbr.fragment\";\r\nimport \"../../Shaders/pbr.vertex\";\r\n\r\nimport { EffectFallbacks } from \"../effectFallbacks\";\r\nimport { PBRClearCoatConfiguration } from \"./pbrClearCoatConfiguration\";\r\nimport { PBRIridescenceConfiguration } from \"./pbrIridescenceConfiguration\";\r\nimport { PBRAnisotropicConfiguration } from \"./pbrAnisotropicConfiguration\";\r\nimport { PBRSheenConfiguration } from \"./pbrSheenConfiguration\";\r\nimport { PBRSubSurfaceConfiguration } from \"./pbrSubSurfaceConfiguration\";\r\nimport { DetailMapConfiguration } from \"../material.detailMapConfiguration\";\r\nimport { addClipPlaneUniforms, bindClipPlane } from \"../clipPlaneMaterialHelper\";\r\n\r\nconst onCreatedEffectParameters = { effect: null as unknown as Effect, subMesh: null as unknown as Nullable };\r\n\r\n/**\r\n * Manages the defines for the PBR Material.\r\n * @internal\r\n */\r\nexport class PBRMaterialDefines extends MaterialDefines implements IImageProcessingConfigurationDefines {\r\n public PBR = true;\r\n\r\n public NUM_SAMPLES = \"0\";\r\n public REALTIME_FILTERING = false;\r\n\r\n public MAINUV1 = false;\r\n public MAINUV2 = false;\r\n public MAINUV3 = false;\r\n public MAINUV4 = false;\r\n public MAINUV5 = false;\r\n public MAINUV6 = false;\r\n public UV1 = false;\r\n public UV2 = false;\r\n public UV3 = false;\r\n public UV4 = false;\r\n public UV5 = false;\r\n public UV6 = false;\r\n\r\n public ALBEDO = false;\r\n public GAMMAALBEDO = false;\r\n public ALBEDODIRECTUV = 0;\r\n public VERTEXCOLOR = false;\r\n\r\n public BAKED_VERTEX_ANIMATION_TEXTURE = false;\r\n\r\n public AMBIENT = false;\r\n public AMBIENTDIRECTUV = 0;\r\n public AMBIENTINGRAYSCALE = false;\r\n\r\n public OPACITY = false;\r\n public VERTEXALPHA = false;\r\n public OPACITYDIRECTUV = 0;\r\n public OPACITYRGB = false;\r\n public ALPHATEST = false;\r\n public DEPTHPREPASS = false;\r\n public ALPHABLEND = false;\r\n public ALPHAFROMALBEDO = false;\r\n public ALPHATESTVALUE = \"0.5\";\r\n public SPECULAROVERALPHA = false;\r\n public RADIANCEOVERALPHA = false;\r\n public ALPHAFRESNEL = false;\r\n public LINEARALPHAFRESNEL = false;\r\n public PREMULTIPLYALPHA = false;\r\n\r\n public EMISSIVE = false;\r\n public EMISSIVEDIRECTUV = 0;\r\n public GAMMAEMISSIVE = false;\r\n\r\n public REFLECTIVITY = false;\r\n public REFLECTIVITY_GAMMA = false;\r\n public REFLECTIVITYDIRECTUV = 0;\r\n public SPECULARTERM = false;\r\n\r\n public MICROSURFACEFROMREFLECTIVITYMAP = false;\r\n public MICROSURFACEAUTOMATIC = false;\r\n public LODBASEDMICROSFURACE = false;\r\n public MICROSURFACEMAP = false;\r\n public MICROSURFACEMAPDIRECTUV = 0;\r\n\r\n public METALLICWORKFLOW = false;\r\n public ROUGHNESSSTOREINMETALMAPALPHA = false;\r\n public ROUGHNESSSTOREINMETALMAPGREEN = false;\r\n public METALLNESSSTOREINMETALMAPBLUE = false;\r\n public AOSTOREINMETALMAPRED = false;\r\n public METALLIC_REFLECTANCE = false;\r\n public METALLIC_REFLECTANCE_GAMMA = false;\r\n public METALLIC_REFLECTANCEDIRECTUV = 0;\r\n public METALLIC_REFLECTANCE_USE_ALPHA_ONLY = false;\r\n public REFLECTANCE = false;\r\n public REFLECTANCE_GAMMA = false;\r\n public REFLECTANCEDIRECTUV = 0;\r\n\r\n public ENVIRONMENTBRDF = false;\r\n public ENVIRONMENTBRDF_RGBD = false;\r\n\r\n public NORMAL = false;\r\n public TANGENT = false;\r\n public BUMP = false;\r\n public BUMPDIRECTUV = 0;\r\n public OBJECTSPACE_NORMALMAP = false;\r\n public PARALLAX = false;\r\n public PARALLAXOCCLUSION = false;\r\n public NORMALXYSCALE = true;\r\n\r\n public LIGHTMAP = false;\r\n public LIGHTMAPDIRECTUV = 0;\r\n public USELIGHTMAPASSHADOWMAP = false;\r\n public GAMMALIGHTMAP = false;\r\n public RGBDLIGHTMAP = false;\r\n\r\n public REFLECTION = false;\r\n public REFLECTIONMAP_3D = false;\r\n public REFLECTIONMAP_SPHERICAL = false;\r\n public REFLECTIONMAP_PLANAR = false;\r\n public REFLECTIONMAP_CUBIC = false;\r\n public USE_LOCAL_REFLECTIONMAP_CUBIC = false;\r\n public REFLECTIONMAP_PROJECTION = false;\r\n public REFLECTIONMAP_SKYBOX = false;\r\n public REFLECTIONMAP_EXPLICIT = false;\r\n public REFLECTIONMAP_EQUIRECTANGULAR = false;\r\n public REFLECTIONMAP_EQUIRECTANGULAR_FIXED = false;\r\n public REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = false;\r\n public INVERTCUBICMAP = false;\r\n public USESPHERICALFROMREFLECTIONMAP = false;\r\n public USEIRRADIANCEMAP = false;\r\n public USESPHERICALINVERTEX = false;\r\n public REFLECTIONMAP_OPPOSITEZ = false;\r\n public LODINREFLECTIONALPHA = false;\r\n public GAMMAREFLECTION = false;\r\n public RGBDREFLECTION = false;\r\n public LINEARSPECULARREFLECTION = false;\r\n public RADIANCEOCCLUSION = false;\r\n public HORIZONOCCLUSION = false;\r\n\r\n public INSTANCES = false;\r\n public THIN_INSTANCES = false;\r\n public INSTANCESCOLOR = false;\r\n\r\n public PREPASS = false;\r\n public PREPASS_IRRADIANCE = false;\r\n public PREPASS_IRRADIANCE_INDEX = -1;\r\n public PREPASS_ALBEDO_SQRT = false;\r\n public PREPASS_ALBEDO_SQRT_INDEX = -1;\r\n public PREPASS_DEPTH = false;\r\n public PREPASS_DEPTH_INDEX = -1;\r\n public PREPASS_NORMAL = false;\r\n public PREPASS_NORMAL_INDEX = -1;\r\n public PREPASS_POSITION = false;\r\n public PREPASS_POSITION_INDEX = -1;\r\n public PREPASS_VELOCITY = false;\r\n public PREPASS_VELOCITY_INDEX = -1;\r\n public PREPASS_REFLECTIVITY = false;\r\n public PREPASS_REFLECTIVITY_INDEX = -1;\r\n public SCENE_MRT_COUNT = 0;\r\n\r\n public NUM_BONE_INFLUENCERS = 0;\r\n public BonesPerMesh = 0;\r\n public BONETEXTURE = false;\r\n public BONES_VELOCITY_ENABLED = false;\r\n\r\n public NONUNIFORMSCALING = false;\r\n\r\n public MORPHTARGETS = false;\r\n public MORPHTARGETS_NORMAL = false;\r\n public MORPHTARGETS_TANGENT = false;\r\n public MORPHTARGETS_UV = false;\r\n public NUM_MORPH_INFLUENCERS = 0;\r\n public MORPHTARGETS_TEXTURE = false;\r\n\r\n public IMAGEPROCESSING = false;\r\n public VIGNETTE = false;\r\n public VIGNETTEBLENDMODEMULTIPLY = false;\r\n public VIGNETTEBLENDMODEOPAQUE = false;\r\n public TONEMAPPING = false;\r\n public TONEMAPPING_ACES = false;\r\n public CONTRAST = false;\r\n public COLORCURVES = false;\r\n public COLORGRADING = false;\r\n public COLORGRADING3D = false;\r\n public SAMPLER3DGREENDEPTH = false;\r\n public SAMPLER3DBGRMAP = false;\r\n public DITHER = false;\r\n public IMAGEPROCESSINGPOSTPROCESS = false;\r\n public SKIPFINALCOLORCLAMP = false;\r\n public EXPOSURE = false;\r\n public MULTIVIEW = false;\r\n public ORDER_INDEPENDENT_TRANSPARENCY = false;\r\n public ORDER_INDEPENDENT_TRANSPARENCY_16BITS = false;\r\n\r\n public USEPHYSICALLIGHTFALLOFF = false;\r\n public USEGLTFLIGHTFALLOFF = false;\r\n public TWOSIDEDLIGHTING = false;\r\n public SHADOWFLOAT = false;\r\n public CLIPPLANE = false;\r\n public CLIPPLANE2 = false;\r\n public CLIPPLANE3 = false;\r\n public CLIPPLANE4 = false;\r\n public CLIPPLANE5 = false;\r\n public CLIPPLANE6 = false;\r\n public POINTSIZE = false;\r\n public FOG = false;\r\n public LOGARITHMICDEPTH = false;\r\n public CAMERA_ORTHOGRAPHIC = false;\r\n public CAMERA_PERSPECTIVE = false;\r\n\r\n public FORCENORMALFORWARD = false;\r\n\r\n public SPECULARAA = false;\r\n\r\n public UNLIT = false;\r\n\r\n public DECAL_AFTER_DETAIL = false;\r\n\r\n public DEBUGMODE = 0;\r\n\r\n /**\r\n * Initializes the PBR Material defines.\r\n * @param externalProperties The external properties\r\n */\r\n constructor(externalProperties?: { [name: string]: { type: string; default: any } }) {\r\n super(externalProperties);\r\n this.rebuild();\r\n }\r\n\r\n /**\r\n * Resets the PBR Material defines.\r\n */\r\n public reset(): void {\r\n super.reset();\r\n this.ALPHATESTVALUE = \"0.5\";\r\n this.PBR = true;\r\n this.NORMALXYSCALE = true;\r\n }\r\n}\r\n\r\n/**\r\n * The Physically based material base class of BJS.\r\n *\r\n * This offers the main features of a standard PBR material.\r\n * For more information, please refer to the documentation :\r\n * https://doc.babylonjs.com/features/featuresDeepDive/materials/using/introToPBR\r\n */\r\nexport abstract class PBRBaseMaterial extends PushMaterial {\r\n /**\r\n * PBRMaterialTransparencyMode: No transparency mode, Alpha channel is not use.\r\n */\r\n public static readonly PBRMATERIAL_OPAQUE = Material.MATERIAL_OPAQUE;\r\n\r\n /**\r\n * PBRMaterialTransparencyMode: Alpha Test mode, pixel are discarded below a certain threshold defined by the alpha cutoff value.\r\n */\r\n public static readonly PBRMATERIAL_ALPHATEST = Material.MATERIAL_ALPHATEST;\r\n\r\n /**\r\n * PBRMaterialTransparencyMode: Pixels are blended (according to the alpha mode) with the already drawn pixels in the current frame buffer.\r\n */\r\n public static readonly PBRMATERIAL_ALPHABLEND = Material.MATERIAL_ALPHABLEND;\r\n\r\n /**\r\n * PBRMaterialTransparencyMode: Pixels are blended (according to the alpha mode) with the already drawn pixels in the current frame buffer.\r\n * They are also discarded below the alpha cutoff threshold to improve performances.\r\n */\r\n public static readonly PBRMATERIAL_ALPHATESTANDBLEND = Material.MATERIAL_ALPHATESTANDBLEND;\r\n\r\n /**\r\n * Defines the default value of how much AO map is occluding the analytical lights\r\n * (point spot...).\r\n */\r\n public static DEFAULT_AO_ON_ANALYTICAL_LIGHTS = 0;\r\n\r\n /**\r\n * PBRMaterialLightFalloff Physical: light is falling off following the inverse squared distance law.\r\n */\r\n public static readonly LIGHTFALLOFF_PHYSICAL = 0;\r\n\r\n /**\r\n * PBRMaterialLightFalloff gltf: light is falling off as described in the gltf moving to PBR document\r\n * to enhance interoperability with other engines.\r\n */\r\n public static readonly LIGHTFALLOFF_GLTF = 1;\r\n\r\n /**\r\n * PBRMaterialLightFalloff Standard: light is falling off like in the standard material\r\n * to enhance interoperability with other materials.\r\n */\r\n public static readonly LIGHTFALLOFF_STANDARD = 2;\r\n\r\n /**\r\n * Intensity of the direct lights e.g. the four lights available in your scene.\r\n * This impacts both the direct diffuse and specular highlights.\r\n * @internal\r\n */\r\n public _directIntensity: number = 1.0;\r\n\r\n /**\r\n * Intensity of the emissive part of the material.\r\n * This helps controlling the emissive effect without modifying the emissive color.\r\n * @internal\r\n */\r\n public _emissiveIntensity: number = 1.0;\r\n\r\n /**\r\n * Intensity of the environment e.g. how much the environment will light the object\r\n * either through harmonics for rough material or through the reflection for shiny ones.\r\n * @internal\r\n */\r\n public _environmentIntensity: number = 1.0;\r\n\r\n /**\r\n * This is a special control allowing the reduction of the specular highlights coming from the\r\n * four lights of the scene. Those highlights may not be needed in full environment lighting.\r\n * @internal\r\n */\r\n public _specularIntensity: number = 1.0;\r\n\r\n /**\r\n * This stores the direct, emissive, environment, and specular light intensities into a Vector4.\r\n */\r\n private _lightingInfos: Vector4 = new Vector4(this._directIntensity, this._emissiveIntensity, this._environmentIntensity, this._specularIntensity);\r\n\r\n /**\r\n * Debug Control allowing disabling the bump map on this material.\r\n * @internal\r\n */\r\n public _disableBumpMap: boolean = false;\r\n\r\n /**\r\n * AKA Diffuse Texture in standard nomenclature.\r\n * @internal\r\n */\r\n public _albedoTexture: Nullable = null;\r\n\r\n /**\r\n * AKA Occlusion Texture in other nomenclature.\r\n * @internal\r\n */\r\n public _ambientTexture: Nullable = null;\r\n\r\n /**\r\n * AKA Occlusion Texture Intensity in other nomenclature.\r\n * @internal\r\n */\r\n public _ambientTextureStrength: number = 1.0;\r\n\r\n /**\r\n * Defines how much the AO map is occluding the analytical lights (point spot...).\r\n * 1 means it completely occludes it\r\n * 0 mean it has no impact\r\n * @internal\r\n */\r\n public _ambientTextureImpactOnAnalyticalLights: number = PBRBaseMaterial.DEFAULT_AO_ON_ANALYTICAL_LIGHTS;\r\n\r\n /**\r\n * Stores the alpha values in a texture.\r\n * @internal\r\n */\r\n public _opacityTexture: Nullable = null;\r\n\r\n /**\r\n * Stores the reflection values in a texture.\r\n * @internal\r\n */\r\n public _reflectionTexture: Nullable = null;\r\n\r\n /**\r\n * Stores the emissive values in a texture.\r\n * @internal\r\n */\r\n public _emissiveTexture: Nullable = null;\r\n\r\n /**\r\n * AKA Specular texture in other nomenclature.\r\n * @internal\r\n */\r\n public _reflectivityTexture: Nullable = null;\r\n\r\n /**\r\n * Used to switch from specular/glossiness to metallic/roughness workflow.\r\n * @internal\r\n */\r\n public _metallicTexture: Nullable = null;\r\n\r\n /**\r\n * Specifies the metallic scalar of the metallic/roughness workflow.\r\n * Can also be used to scale the metalness values of the metallic texture.\r\n * @internal\r\n */\r\n public _metallic: Nullable = null;\r\n\r\n /**\r\n * Specifies the roughness scalar of the metallic/roughness workflow.\r\n * Can also be used to scale the roughness values of the metallic texture.\r\n * @internal\r\n */\r\n public _roughness: Nullable = null;\r\n\r\n /**\r\n * In metallic workflow, specifies an F0 factor to help configuring the material F0.\r\n * By default the indexOfrefraction is used to compute F0;\r\n *\r\n * This is used as a factor against the default reflectance at normal incidence to tweak it.\r\n *\r\n * F0 = defaultF0 * metallicF0Factor * metallicReflectanceColor;\r\n * F90 = metallicReflectanceColor;\r\n * @internal\r\n */\r\n public _metallicF0Factor = 1;\r\n\r\n /**\r\n * In metallic workflow, specifies an F0 color.\r\n * By default the F90 is always 1;\r\n *\r\n * Please note that this factor is also used as a factor against the default reflectance at normal incidence.\r\n *\r\n * F0 = defaultF0_from_IOR * metallicF0Factor * metallicReflectanceColor\r\n * F90 = metallicF0Factor;\r\n * @internal\r\n */\r\n public _metallicReflectanceColor = Color3.White();\r\n\r\n /**\r\n * Specifies that only the A channel from _metallicReflectanceTexture should be used.\r\n * If false, both RGB and A channels will be used\r\n * @internal\r\n */\r\n public _useOnlyMetallicFromMetallicReflectanceTexture = false;\r\n\r\n /**\r\n * Defines to store metallicReflectanceColor in RGB and metallicF0Factor in A\r\n * This is multiply against the scalar values defined in the material.\r\n * @internal\r\n */\r\n public _metallicReflectanceTexture: Nullable = null;\r\n\r\n /**\r\n * Defines to store reflectanceColor in RGB\r\n * This is multiplied against the scalar values defined in the material.\r\n * If both _reflectanceTexture and _metallicReflectanceTexture textures are provided and _useOnlyMetallicFromMetallicReflectanceTexture\r\n * is false, _metallicReflectanceTexture takes precedence and _reflectanceTexture is not used\r\n * @internal\r\n */\r\n public _reflectanceTexture: Nullable = null;\r\n\r\n /**\r\n * Used to enable roughness/glossiness fetch from a separate channel depending on the current mode.\r\n * Gray Scale represents roughness in metallic mode and glossiness in specular mode.\r\n * @internal\r\n */\r\n public _microSurfaceTexture: Nullable = null;\r\n\r\n /**\r\n * Stores surface normal data used to displace a mesh in a texture.\r\n * @internal\r\n */\r\n public _bumpTexture: Nullable = null;\r\n\r\n /**\r\n * Stores the pre-calculated light information of a mesh in a texture.\r\n * @internal\r\n */\r\n public _lightmapTexture: Nullable = null;\r\n\r\n /**\r\n * The color of a material in ambient lighting.\r\n * @internal\r\n */\r\n public _ambientColor = new Color3(0, 0, 0);\r\n\r\n /**\r\n * AKA Diffuse Color in other nomenclature.\r\n * @internal\r\n */\r\n public _albedoColor = new Color3(1, 1, 1);\r\n\r\n /**\r\n * AKA Specular Color in other nomenclature.\r\n * @internal\r\n */\r\n public _reflectivityColor = new Color3(1, 1, 1);\r\n\r\n /**\r\n * The color applied when light is reflected from a material.\r\n * @internal\r\n */\r\n public _reflectionColor = new Color3(1, 1, 1);\r\n\r\n /**\r\n * The color applied when light is emitted from a material.\r\n * @internal\r\n */\r\n public _emissiveColor = new Color3(0, 0, 0);\r\n\r\n /**\r\n * AKA Glossiness in other nomenclature.\r\n * @internal\r\n */\r\n public _microSurface = 0.9;\r\n\r\n /**\r\n * Specifies that the material will use the light map as a show map.\r\n * @internal\r\n */\r\n public _useLightmapAsShadowmap = false;\r\n\r\n /**\r\n * This parameters will enable/disable Horizon occlusion to prevent normal maps to look shiny when the normal\r\n * makes the reflect vector face the model (under horizon).\r\n * @internal\r\n */\r\n public _useHorizonOcclusion = true;\r\n\r\n /**\r\n * This parameters will enable/disable radiance occlusion by preventing the radiance to lit\r\n * too much the area relying on ambient texture to define their ambient occlusion.\r\n * @internal\r\n */\r\n public _useRadianceOcclusion = true;\r\n\r\n /**\r\n * Specifies that the alpha is coming form the albedo channel alpha channel for alpha blending.\r\n * @internal\r\n */\r\n public _useAlphaFromAlbedoTexture = false;\r\n\r\n /**\r\n * Specifies that the material will keeps the specular highlights over a transparent surface (only the most luminous ones).\r\n * A car glass is a good example of that. When sun reflects on it you can not see what is behind.\r\n * @internal\r\n */\r\n public _useSpecularOverAlpha = true;\r\n\r\n /**\r\n * Specifies if the reflectivity texture contains the glossiness information in its alpha channel.\r\n * @internal\r\n */\r\n public _useMicroSurfaceFromReflectivityMapAlpha = false;\r\n\r\n /**\r\n * Specifies if the metallic texture contains the roughness information in its alpha channel.\r\n * @internal\r\n */\r\n public _useRoughnessFromMetallicTextureAlpha = true;\r\n\r\n /**\r\n * Specifies if the metallic texture contains the roughness information in its green channel.\r\n * @internal\r\n */\r\n public _useRoughnessFromMetallicTextureGreen = false;\r\n\r\n /**\r\n * Specifies if the metallic texture contains the metallness information in its blue channel.\r\n * @internal\r\n */\r\n public _useMetallnessFromMetallicTextureBlue = false;\r\n\r\n /**\r\n * Specifies if the metallic texture contains the ambient occlusion information in its red channel.\r\n * @internal\r\n */\r\n public _useAmbientOcclusionFromMetallicTextureRed = false;\r\n\r\n /**\r\n * Specifies if the ambient texture contains the ambient occlusion information in its red channel only.\r\n * @internal\r\n */\r\n public _useAmbientInGrayScale = false;\r\n\r\n /**\r\n * In case the reflectivity map does not contain the microsurface information in its alpha channel,\r\n * The material will try to infer what glossiness each pixel should be.\r\n * @internal\r\n */\r\n public _useAutoMicroSurfaceFromReflectivityMap = false;\r\n\r\n /**\r\n * Defines the falloff type used in this material.\r\n * It by default is Physical.\r\n * @internal\r\n */\r\n public _lightFalloff = PBRBaseMaterial.LIGHTFALLOFF_PHYSICAL;\r\n\r\n /**\r\n * Specifies that the material will keeps the reflection highlights over a transparent surface (only the most luminous ones).\r\n * A car glass is a good example of that. When the street lights reflects on it you can not see what is behind.\r\n * @internal\r\n */\r\n public _useRadianceOverAlpha = true;\r\n\r\n /**\r\n * Allows using an object space normal map (instead of tangent space).\r\n * @internal\r\n */\r\n public _useObjectSpaceNormalMap = false;\r\n\r\n /**\r\n * Allows using the bump map in parallax mode.\r\n * @internal\r\n */\r\n public _useParallax = false;\r\n\r\n /**\r\n * Allows using the bump map in parallax occlusion mode.\r\n * @internal\r\n */\r\n public _useParallaxOcclusion = false;\r\n\r\n /**\r\n * Controls the scale bias of the parallax mode.\r\n * @internal\r\n */\r\n public _parallaxScaleBias = 0.05;\r\n\r\n /**\r\n * If sets to true, disables all the lights affecting the material.\r\n * @internal\r\n */\r\n public _disableLighting = false;\r\n\r\n /**\r\n * Number of Simultaneous lights allowed on the material.\r\n * @internal\r\n */\r\n public _maxSimultaneousLights = 4;\r\n\r\n /**\r\n * If sets to true, x component of normal map value will be inverted (x = 1.0 - x).\r\n * @internal\r\n */\r\n public _invertNormalMapX = false;\r\n\r\n /**\r\n * If sets to true, y component of normal map value will be inverted (y = 1.0 - y).\r\n * @internal\r\n */\r\n public _invertNormalMapY = false;\r\n\r\n /**\r\n * If sets to true and backfaceCulling is false, normals will be flipped on the backside.\r\n * @internal\r\n */\r\n public _twoSidedLighting = false;\r\n\r\n /**\r\n * Defines the alpha limits in alpha test mode.\r\n * @internal\r\n */\r\n public _alphaCutOff = 0.4;\r\n\r\n /**\r\n * Enforces alpha test in opaque or blend mode in order to improve the performances of some situations.\r\n * @internal\r\n */\r\n public _forceAlphaTest = false;\r\n\r\n /**\r\n * A fresnel is applied to the alpha of the model to ensure grazing angles edges are not alpha tested.\r\n * And/Or occlude the blended part. (alpha is converted to gamma to compute the fresnel)\r\n * @internal\r\n */\r\n public _useAlphaFresnel = false;\r\n\r\n /**\r\n * A fresnel is applied to the alpha of the model to ensure grazing angles edges are not alpha tested.\r\n * And/Or occlude the blended part. (alpha stays linear to compute the fresnel)\r\n * @internal\r\n */\r\n public _useLinearAlphaFresnel = false;\r\n\r\n /**\r\n * Specifies the environment BRDF texture used to compute the scale and offset roughness values\r\n * from cos theta and roughness:\r\n * http://blog.selfshadow.com/publications/s2013-shading-course/karis/s2013_pbs_epic_notes_v2.pdf\r\n * @internal\r\n */\r\n public _environmentBRDFTexture: Nullable = null;\r\n\r\n /**\r\n * Force the shader to compute irradiance in the fragment shader in order to take bump in account.\r\n * @internal\r\n */\r\n public _forceIrradianceInFragment = false;\r\n\r\n private _realTimeFiltering: boolean = false;\r\n /**\r\n * Enables realtime filtering on the texture.\r\n */\r\n public get realTimeFiltering() {\r\n return this._realTimeFiltering;\r\n }\r\n public set realTimeFiltering(b: boolean) {\r\n this._realTimeFiltering = b;\r\n this.markAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n private _realTimeFilteringQuality: number = Constants.TEXTURE_FILTERING_QUALITY_LOW;\r\n /**\r\n * Quality switch for realtime filtering\r\n */\r\n public get realTimeFilteringQuality(): number {\r\n return this._realTimeFilteringQuality;\r\n }\r\n public set realTimeFilteringQuality(n: number) {\r\n this._realTimeFilteringQuality = n;\r\n this.markAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n\r\n /**\r\n * Can this material render to several textures at once\r\n */\r\n public get canRenderToMRT() {\r\n return true;\r\n }\r\n\r\n /**\r\n * Force normal to face away from face.\r\n * @internal\r\n */\r\n public _forceNormalForward = false;\r\n\r\n /**\r\n * Enables specular anti aliasing in the PBR shader.\r\n * It will both interacts on the Geometry for analytical and IBL lighting.\r\n * It also prefilter the roughness map based on the bump values.\r\n * @internal\r\n */\r\n public _enableSpecularAntiAliasing = false;\r\n\r\n /**\r\n * Default configuration related to image processing available in the PBR Material.\r\n */\r\n @serializeAsImageProcessingConfiguration()\r\n protected _imageProcessingConfiguration: ImageProcessingConfiguration;\r\n\r\n /**\r\n * Keep track of the image processing observer to allow dispose and replace.\r\n */\r\n private _imageProcessingObserver: Nullable> = null;\r\n\r\n /**\r\n * Attaches a new image processing configuration to the PBR Material.\r\n * @param configuration\r\n */\r\n protected _attachImageProcessingConfiguration(configuration: Nullable): void {\r\n if (configuration === this._imageProcessingConfiguration) {\r\n return;\r\n }\r\n\r\n // Detaches observer.\r\n if (this._imageProcessingConfiguration && this._imageProcessingObserver) {\r\n this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver);\r\n }\r\n\r\n // Pick the scene configuration if needed.\r\n if (!configuration) {\r\n this._imageProcessingConfiguration = this.getScene().imageProcessingConfiguration;\r\n } else {\r\n this._imageProcessingConfiguration = configuration;\r\n }\r\n\r\n // Attaches observer.\r\n if (this._imageProcessingConfiguration) {\r\n this._imageProcessingObserver = this._imageProcessingConfiguration.onUpdateParameters.add(() => {\r\n this._markAllSubMeshesAsImageProcessingDirty();\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Stores the available render targets.\r\n */\r\n private _renderTargets = new SmartArray(16);\r\n\r\n /**\r\n * Sets the global ambient color for the material used in lighting calculations.\r\n */\r\n private _globalAmbientColor = new Color3(0, 0, 0);\r\n\r\n /**\r\n * Enables the use of logarithmic depth buffers, which is good for wide depth buffers.\r\n */\r\n private _useLogarithmicDepth: boolean = false;\r\n\r\n /**\r\n * If set to true, no lighting calculations will be applied.\r\n */\r\n private _unlit = false;\r\n\r\n /**\r\n * If sets to true, the decal map will be applied after the detail map. Else, it is applied before (default: false)\r\n */\r\n private _applyDecalMapAfterDetailMap = false;\r\n\r\n private _debugMode = 0;\r\n /**\r\n * @internal\r\n * This is reserved for the inspector.\r\n * Defines the material debug mode.\r\n * It helps seeing only some components of the material while troubleshooting.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsMiscDirty\")\r\n public debugMode = 0;\r\n\r\n /**\r\n * @internal\r\n * This is reserved for the inspector.\r\n * Specify from where on screen the debug mode should start.\r\n * The value goes from -1 (full screen) to 1 (not visible)\r\n * It helps with side by side comparison against the final render\r\n * This defaults to -1\r\n */\r\n public debugLimit = -1;\r\n\r\n /**\r\n * @internal\r\n * This is reserved for the inspector.\r\n * As the default viewing range might not be enough (if the ambient is really small for instance)\r\n * You can use the factor to better multiply the final value.\r\n */\r\n public debugFactor = 1;\r\n\r\n /**\r\n * Defines the clear coat layer parameters for the material.\r\n */\r\n public readonly clearCoat: PBRClearCoatConfiguration;\r\n\r\n /**\r\n * Defines the iridescence layer parameters for the material.\r\n */\r\n public readonly iridescence: PBRIridescenceConfiguration;\r\n\r\n /**\r\n * Defines the anisotropic parameters for the material.\r\n */\r\n public readonly anisotropy: PBRAnisotropicConfiguration;\r\n\r\n /**\r\n * Defines the BRDF parameters for the material.\r\n */\r\n public readonly brdf: PBRBRDFConfiguration;\r\n\r\n /**\r\n * Defines the Sheen parameters for the material.\r\n */\r\n public readonly sheen: PBRSheenConfiguration;\r\n\r\n /**\r\n * Defines the SubSurface parameters for the material.\r\n */\r\n public readonly subSurface: PBRSubSurfaceConfiguration;\r\n\r\n /**\r\n * Defines additional PrePass parameters for the material.\r\n */\r\n public readonly prePassConfiguration: PrePassConfiguration;\r\n\r\n /**\r\n * Defines the detail map parameters for the material.\r\n */\r\n public readonly detailMap: DetailMapConfiguration;\r\n\r\n protected _cacheHasRenderTargetTextures = false;\r\n\r\n /**\r\n * Instantiates a new PBRMaterial instance.\r\n *\r\n * @param name The material name\r\n * @param scene The scene the material will be use in.\r\n */\r\n constructor(name: string, scene?: Scene) {\r\n super(name, scene);\r\n\r\n this.brdf = new PBRBRDFConfiguration(this);\r\n this.clearCoat = new PBRClearCoatConfiguration(this);\r\n this.iridescence = new PBRIridescenceConfiguration(this);\r\n this.anisotropy = new PBRAnisotropicConfiguration(this);\r\n this.sheen = new PBRSheenConfiguration(this);\r\n this.subSurface = new PBRSubSurfaceConfiguration(this);\r\n this.detailMap = new DetailMapConfiguration(this);\r\n\r\n // Setup the default processing configuration to the scene.\r\n this._attachImageProcessingConfiguration(null);\r\n\r\n this.getRenderTargetTextures = (): SmartArray => {\r\n this._renderTargets.reset();\r\n\r\n if (MaterialFlags.ReflectionTextureEnabled && this._reflectionTexture && this._reflectionTexture.isRenderTarget) {\r\n this._renderTargets.push(this._reflectionTexture);\r\n }\r\n\r\n this._eventInfo.renderTargets = this._renderTargets;\r\n this._callbackPluginEventFillRenderTargetTextures(this._eventInfo);\r\n\r\n return this._renderTargets;\r\n };\r\n\r\n this._environmentBRDFTexture = GetEnvironmentBRDFTexture(this.getScene());\r\n this.prePassConfiguration = new PrePassConfiguration();\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating that current material needs to register RTT\r\n */\r\n public get hasRenderTargetTextures(): boolean {\r\n if (MaterialFlags.ReflectionTextureEnabled && this._reflectionTexture && this._reflectionTexture.isRenderTarget) {\r\n return true;\r\n }\r\n\r\n return this._cacheHasRenderTargetTextures;\r\n }\r\n\r\n /**\r\n * Can this material render to prepass\r\n */\r\n public get isPrePassCapable(): boolean {\r\n return !this.disableDepthWrite;\r\n }\r\n\r\n /**\r\n * Gets the name of the material class.\r\n */\r\n public getClassName(): string {\r\n return \"PBRBaseMaterial\";\r\n }\r\n\r\n /**\r\n * Enabled the use of logarithmic depth buffers, which is good for wide depth buffers.\r\n */\r\n @serialize()\r\n public get useLogarithmicDepth(): boolean {\r\n return this._useLogarithmicDepth;\r\n }\r\n\r\n /**\r\n * Enabled the use of logarithmic depth buffers, which is good for wide depth buffers.\r\n */\r\n public set useLogarithmicDepth(value: boolean) {\r\n this._useLogarithmicDepth = value && this.getScene().getEngine().getCaps().fragmentDepthSupported;\r\n }\r\n\r\n /**\r\n * Returns true if alpha blending should be disabled.\r\n */\r\n protected get _disableAlphaBlending(): boolean {\r\n return (\r\n this._transparencyMode === PBRBaseMaterial.PBRMATERIAL_OPAQUE ||\r\n this._transparencyMode === PBRBaseMaterial.PBRMATERIAL_ALPHATEST ||\r\n this.subSurface?.disableAlphaBlending\r\n );\r\n }\r\n\r\n /**\r\n * Specifies whether or not this material should be rendered in alpha blend mode.\r\n */\r\n public needAlphaBlending(): boolean {\r\n if (this._disableAlphaBlending) {\r\n return false;\r\n }\r\n\r\n return this.alpha < 1.0 || this._opacityTexture != null || this._shouldUseAlphaFromAlbedoTexture();\r\n }\r\n\r\n /**\r\n * Specifies whether or not this material should be rendered in alpha test mode.\r\n */\r\n public needAlphaTesting(): boolean {\r\n if (this._forceAlphaTest) {\r\n return true;\r\n }\r\n\r\n if (this.subSurface?.disableAlphaBlending) {\r\n return false;\r\n }\r\n\r\n return this._hasAlphaChannel() && (this._transparencyMode == null || this._transparencyMode === PBRBaseMaterial.PBRMATERIAL_ALPHATEST);\r\n }\r\n\r\n /**\r\n * Specifies whether or not the alpha value of the albedo texture should be used for alpha blending.\r\n */\r\n protected _shouldUseAlphaFromAlbedoTexture(): boolean {\r\n return this._albedoTexture != null && this._albedoTexture.hasAlpha && this._useAlphaFromAlbedoTexture && this._transparencyMode !== PBRBaseMaterial.PBRMATERIAL_OPAQUE;\r\n }\r\n\r\n /**\r\n * Specifies whether or not there is a usable alpha channel for transparency.\r\n */\r\n protected _hasAlphaChannel(): boolean {\r\n return (this._albedoTexture != null && this._albedoTexture.hasAlpha) || this._opacityTexture != null;\r\n }\r\n\r\n /**\r\n * Gets the texture used for the alpha test.\r\n */\r\n public getAlphaTestTexture(): Nullable {\r\n return this._albedoTexture;\r\n }\r\n\r\n /**\r\n * Specifies that the submesh is ready to be used.\r\n * @param mesh - BJS mesh.\r\n * @param subMesh - A submesh of the BJS mesh. Used to check if it is ready.\r\n * @param useInstances - Specifies that instances should be used.\r\n * @returns - boolean indicating that the submesh is ready or not.\r\n */\r\n public isReadyForSubMesh(mesh: AbstractMesh, subMesh: SubMesh, useInstances?: boolean): boolean {\r\n if (!this._uniformBufferLayoutBuilt) {\r\n this.buildUniformLayout();\r\n }\r\n\r\n if (subMesh.effect && this.isFrozen) {\r\n if (subMesh.effect._wasPreviouslyReady && subMesh.effect._wasPreviouslyUsingInstances === useInstances) {\r\n return true;\r\n }\r\n }\r\n\r\n if (!subMesh.materialDefines) {\r\n this._callbackPluginEventGeneric(MaterialPluginEvent.GetDefineNames, this._eventInfo);\r\n subMesh.materialDefines = new PBRMaterialDefines(this._eventInfo.defineNames);\r\n }\r\n\r\n const defines = subMesh.materialDefines;\r\n if (this._isReadyForSubMesh(subMesh)) {\r\n return true;\r\n }\r\n\r\n const scene = this.getScene();\r\n const engine = scene.getEngine();\r\n\r\n if (defines._areTexturesDirty) {\r\n this._eventInfo.hasRenderTargetTextures = false;\r\n this._callbackPluginEventHasRenderTargetTextures(this._eventInfo);\r\n this._cacheHasRenderTargetTextures = this._eventInfo.hasRenderTargetTextures;\r\n if (scene.texturesEnabled) {\r\n if (this._albedoTexture && MaterialFlags.DiffuseTextureEnabled) {\r\n if (!this._albedoTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n }\r\n\r\n if (this._ambientTexture && MaterialFlags.AmbientTextureEnabled) {\r\n if (!this._ambientTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n }\r\n\r\n if (this._opacityTexture && MaterialFlags.OpacityTextureEnabled) {\r\n if (!this._opacityTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n }\r\n\r\n const reflectionTexture = this._getReflectionTexture();\r\n if (reflectionTexture && MaterialFlags.ReflectionTextureEnabled) {\r\n if (!reflectionTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n if (reflectionTexture.irradianceTexture) {\r\n if (!reflectionTexture.irradianceTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n } else {\r\n // Not ready until spherical are ready too.\r\n if (!reflectionTexture.sphericalPolynomial && reflectionTexture.getInternalTexture()?._sphericalPolynomialPromise) {\r\n return false;\r\n }\r\n }\r\n }\r\n\r\n if (this._lightmapTexture && MaterialFlags.LightmapTextureEnabled) {\r\n if (!this._lightmapTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n }\r\n\r\n if (this._emissiveTexture && MaterialFlags.EmissiveTextureEnabled) {\r\n if (!this._emissiveTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n }\r\n\r\n if (MaterialFlags.SpecularTextureEnabled) {\r\n if (this._metallicTexture) {\r\n if (!this._metallicTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n } else if (this._reflectivityTexture) {\r\n if (!this._reflectivityTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n }\r\n\r\n if (this._metallicReflectanceTexture) {\r\n if (!this._metallicReflectanceTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n }\r\n\r\n if (this._reflectanceTexture) {\r\n if (!this._reflectanceTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n }\r\n\r\n if (this._microSurfaceTexture) {\r\n if (!this._microSurfaceTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n }\r\n }\r\n\r\n if (engine.getCaps().standardDerivatives && this._bumpTexture && MaterialFlags.BumpTextureEnabled && !this._disableBumpMap) {\r\n // Bump texture cannot be not blocking.\r\n if (!this._bumpTexture.isReady()) {\r\n return false;\r\n }\r\n }\r\n\r\n if (this._environmentBRDFTexture && MaterialFlags.ReflectionTextureEnabled) {\r\n // This is blocking.\r\n if (!this._environmentBRDFTexture.isReady()) {\r\n return false;\r\n }\r\n }\r\n }\r\n }\r\n\r\n this._eventInfo.isReadyForSubMesh = true;\r\n this._eventInfo.defines = defines;\r\n this._eventInfo.subMesh = subMesh;\r\n this._callbackPluginEventIsReadyForSubMesh(this._eventInfo);\r\n\r\n if (!this._eventInfo.isReadyForSubMesh) {\r\n return false;\r\n }\r\n\r\n if (defines._areImageProcessingDirty && this._imageProcessingConfiguration) {\r\n if (!this._imageProcessingConfiguration.isReady()) {\r\n return false;\r\n }\r\n }\r\n\r\n if (!engine.getCaps().standardDerivatives && !mesh.isVerticesDataPresent(VertexBuffer.NormalKind)) {\r\n mesh.createNormals(true);\r\n Logger.Warn(\"PBRMaterial: Normals have been created for the mesh: \" + mesh.name);\r\n }\r\n\r\n const previousEffect = subMesh.effect;\r\n const lightDisposed = defines._areLightsDisposed;\r\n let effect = this._prepareEffect(mesh, defines, this.onCompiled, this.onError, useInstances, null, subMesh.getRenderingMesh().hasThinInstances);\r\n\r\n let forceWasNotReadyPreviously = false;\r\n\r\n if (effect) {\r\n if (this._onEffectCreatedObservable) {\r\n onCreatedEffectParameters.effect = effect;\r\n onCreatedEffectParameters.subMesh = subMesh;\r\n this._onEffectCreatedObservable.notifyObservers(onCreatedEffectParameters);\r\n }\r\n\r\n // Use previous effect while new one is compiling\r\n if (this.allowShaderHotSwapping && previousEffect && !effect.isReady()) {\r\n effect = previousEffect;\r\n defines.markAsUnprocessed();\r\n\r\n forceWasNotReadyPreviously = this.isFrozen;\r\n\r\n if (lightDisposed) {\r\n // re register in case it takes more than one frame.\r\n defines._areLightsDisposed = true;\r\n return false;\r\n }\r\n } else {\r\n scene.resetCachedMaterial();\r\n subMesh.setEffect(effect, defines, this._materialContext);\r\n }\r\n }\r\n\r\n if (!subMesh.effect || !subMesh.effect.isReady()) {\r\n return false;\r\n }\r\n\r\n defines._renderId = scene.getRenderId();\r\n subMesh.effect._wasPreviouslyReady = forceWasNotReadyPreviously ? false : true;\r\n subMesh.effect._wasPreviouslyUsingInstances = !!useInstances;\r\n\r\n this._checkScenePerformancePriority();\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Specifies if the material uses metallic roughness workflow.\r\n * @returns boolean specifying if the material uses metallic roughness workflow.\r\n */\r\n public isMetallicWorkflow(): boolean {\r\n if (this._metallic != null || this._roughness != null || this._metallicTexture) {\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n private _prepareEffect(\r\n mesh: AbstractMesh,\r\n defines: PBRMaterialDefines,\r\n onCompiled: Nullable<(effect: Effect) => void> = null,\r\n onError: Nullable<(effect: Effect, errors: string) => void> = null,\r\n useInstances: Nullable = null,\r\n useClipPlane: Nullable = null,\r\n useThinInstances: boolean\r\n ): Nullable {\r\n this._prepareDefines(mesh, defines, useInstances, useClipPlane, useThinInstances);\r\n\r\n if (!defines.isDirty) {\r\n return null;\r\n }\r\n\r\n defines.markAsProcessed();\r\n\r\n const scene = this.getScene();\r\n const engine = scene.getEngine();\r\n\r\n // Fallbacks\r\n const fallbacks = new EffectFallbacks();\r\n let fallbackRank = 0;\r\n if (defines.USESPHERICALINVERTEX) {\r\n fallbacks.addFallback(fallbackRank++, \"USESPHERICALINVERTEX\");\r\n }\r\n\r\n if (defines.FOG) {\r\n fallbacks.addFallback(fallbackRank, \"FOG\");\r\n }\r\n if (defines.SPECULARAA) {\r\n fallbacks.addFallback(fallbackRank, \"SPECULARAA\");\r\n }\r\n if (defines.POINTSIZE) {\r\n fallbacks.addFallback(fallbackRank, \"POINTSIZE\");\r\n }\r\n if (defines.LOGARITHMICDEPTH) {\r\n fallbacks.addFallback(fallbackRank, \"LOGARITHMICDEPTH\");\r\n }\r\n if (defines.PARALLAX) {\r\n fallbacks.addFallback(fallbackRank, \"PARALLAX\");\r\n }\r\n if (defines.PARALLAXOCCLUSION) {\r\n fallbacks.addFallback(fallbackRank++, \"PARALLAXOCCLUSION\");\r\n }\r\n\r\n if (defines.ENVIRONMENTBRDF) {\r\n fallbacks.addFallback(fallbackRank++, \"ENVIRONMENTBRDF\");\r\n }\r\n\r\n if (defines.TANGENT) {\r\n fallbacks.addFallback(fallbackRank++, \"TANGENT\");\r\n }\r\n\r\n if (defines.BUMP) {\r\n fallbacks.addFallback(fallbackRank++, \"BUMP\");\r\n }\r\n\r\n fallbackRank = MaterialHelper.HandleFallbacksForShadows(defines, fallbacks, this._maxSimultaneousLights, fallbackRank++);\r\n\r\n if (defines.SPECULARTERM) {\r\n fallbacks.addFallback(fallbackRank++, \"SPECULARTERM\");\r\n }\r\n\r\n if (defines.USESPHERICALFROMREFLECTIONMAP) {\r\n fallbacks.addFallback(fallbackRank++, \"USESPHERICALFROMREFLECTIONMAP\");\r\n }\r\n\r\n if (defines.USEIRRADIANCEMAP) {\r\n fallbacks.addFallback(fallbackRank++, \"USEIRRADIANCEMAP\");\r\n }\r\n\r\n if (defines.LIGHTMAP) {\r\n fallbacks.addFallback(fallbackRank++, \"LIGHTMAP\");\r\n }\r\n\r\n if (defines.NORMAL) {\r\n fallbacks.addFallback(fallbackRank++, \"NORMAL\");\r\n }\r\n\r\n if (defines.AMBIENT) {\r\n fallbacks.addFallback(fallbackRank++, \"AMBIENT\");\r\n }\r\n\r\n if (defines.EMISSIVE) {\r\n fallbacks.addFallback(fallbackRank++, \"EMISSIVE\");\r\n }\r\n\r\n if (defines.VERTEXCOLOR) {\r\n fallbacks.addFallback(fallbackRank++, \"VERTEXCOLOR\");\r\n }\r\n\r\n if (defines.MORPHTARGETS) {\r\n fallbacks.addFallback(fallbackRank++, \"MORPHTARGETS\");\r\n }\r\n\r\n if (defines.MULTIVIEW) {\r\n fallbacks.addFallback(0, \"MULTIVIEW\");\r\n }\r\n\r\n //Attributes\r\n const attribs = [VertexBuffer.PositionKind];\r\n\r\n if (defines.NORMAL) {\r\n attribs.push(VertexBuffer.NormalKind);\r\n }\r\n\r\n if (defines.TANGENT) {\r\n attribs.push(VertexBuffer.TangentKind);\r\n }\r\n\r\n for (let i = 1; i <= Constants.MAX_SUPPORTED_UV_SETS; ++i) {\r\n if (defines[\"UV\" + i]) {\r\n attribs.push(`uv${i === 1 ? \"\" : i}`);\r\n }\r\n }\r\n\r\n if (defines.VERTEXCOLOR) {\r\n attribs.push(VertexBuffer.ColorKind);\r\n }\r\n\r\n if (defines.INSTANCESCOLOR) {\r\n attribs.push(VertexBuffer.ColorInstanceKind);\r\n }\r\n\r\n MaterialHelper.PrepareAttributesForBones(attribs, mesh, defines, fallbacks);\r\n MaterialHelper.PrepareAttributesForInstances(attribs, defines);\r\n MaterialHelper.PrepareAttributesForMorphTargets(attribs, mesh, defines);\r\n MaterialHelper.PrepareAttributesForBakedVertexAnimation(attribs, mesh, defines);\r\n\r\n let shaderName = \"pbr\";\r\n\r\n const uniforms = [\r\n \"world\",\r\n \"view\",\r\n \"viewProjection\",\r\n \"vEyePosition\",\r\n \"vLightsType\",\r\n \"vAmbientColor\",\r\n \"vAlbedoColor\",\r\n \"vReflectivityColor\",\r\n \"vMetallicReflectanceFactors\",\r\n \"vEmissiveColor\",\r\n \"visibility\",\r\n \"vReflectionColor\",\r\n \"vFogInfos\",\r\n \"vFogColor\",\r\n \"pointSize\",\r\n \"vAlbedoInfos\",\r\n \"vAmbientInfos\",\r\n \"vOpacityInfos\",\r\n \"vReflectionInfos\",\r\n \"vReflectionPosition\",\r\n \"vReflectionSize\",\r\n \"vEmissiveInfos\",\r\n \"vReflectivityInfos\",\r\n \"vReflectionFilteringInfo\",\r\n \"vMetallicReflectanceInfos\",\r\n \"vReflectanceInfos\",\r\n \"vMicroSurfaceSamplerInfos\",\r\n \"vBumpInfos\",\r\n \"vLightmapInfos\",\r\n \"mBones\",\r\n \"albedoMatrix\",\r\n \"ambientMatrix\",\r\n \"opacityMatrix\",\r\n \"reflectionMatrix\",\r\n \"emissiveMatrix\",\r\n \"reflectivityMatrix\",\r\n \"normalMatrix\",\r\n \"microSurfaceSamplerMatrix\",\r\n \"bumpMatrix\",\r\n \"lightmapMatrix\",\r\n \"metallicReflectanceMatrix\",\r\n \"reflectanceMatrix\",\r\n \"vLightingIntensity\",\r\n \"logarithmicDepthConstant\",\r\n \"vSphericalX\",\r\n \"vSphericalY\",\r\n \"vSphericalZ\",\r\n \"vSphericalXX_ZZ\",\r\n \"vSphericalYY_ZZ\",\r\n \"vSphericalZZ\",\r\n \"vSphericalXY\",\r\n \"vSphericalYZ\",\r\n \"vSphericalZX\",\r\n \"vSphericalL00\",\r\n \"vSphericalL1_1\",\r\n \"vSphericalL10\",\r\n \"vSphericalL11\",\r\n \"vSphericalL2_2\",\r\n \"vSphericalL2_1\",\r\n \"vSphericalL20\",\r\n \"vSphericalL21\",\r\n \"vSphericalL22\",\r\n \"vReflectionMicrosurfaceInfos\",\r\n \"vTangentSpaceParams\",\r\n \"boneTextureWidth\",\r\n \"vDebugMode\",\r\n \"morphTargetTextureInfo\",\r\n \"morphTargetTextureIndices\",\r\n ];\r\n\r\n const samplers = [\r\n \"albedoSampler\",\r\n \"reflectivitySampler\",\r\n \"ambientSampler\",\r\n \"emissiveSampler\",\r\n \"bumpSampler\",\r\n \"lightmapSampler\",\r\n \"opacitySampler\",\r\n \"reflectionSampler\",\r\n \"reflectionSamplerLow\",\r\n \"reflectionSamplerHigh\",\r\n \"irradianceSampler\",\r\n \"microSurfaceSampler\",\r\n \"environmentBrdfSampler\",\r\n \"boneSampler\",\r\n \"metallicReflectanceSampler\",\r\n \"reflectanceSampler\",\r\n \"morphTargets\",\r\n \"oitDepthSampler\",\r\n \"oitFrontColorSampler\",\r\n ];\r\n\r\n const uniformBuffers = [\"Material\", \"Scene\", \"Mesh\"];\r\n\r\n const indexParameters = { maxSimultaneousLights: this._maxSimultaneousLights, maxSimultaneousMorphTargets: defines.NUM_MORPH_INFLUENCERS };\r\n\r\n this._eventInfo.fallbacks = fallbacks;\r\n this._eventInfo.fallbackRank = fallbackRank;\r\n this._eventInfo.defines = defines;\r\n this._eventInfo.uniforms = uniforms;\r\n this._eventInfo.attributes = attribs;\r\n this._eventInfo.samplers = samplers;\r\n this._eventInfo.uniformBuffersNames = uniformBuffers;\r\n this._eventInfo.customCode = undefined;\r\n this._eventInfo.mesh = mesh;\r\n this._eventInfo.indexParameters = indexParameters;\r\n this._callbackPluginEventGeneric(MaterialPluginEvent.PrepareEffect, this._eventInfo);\r\n\r\n PrePassConfiguration.AddUniforms(uniforms);\r\n PrePassConfiguration.AddSamplers(samplers);\r\n addClipPlaneUniforms(uniforms);\r\n\r\n if (ImageProcessingConfiguration) {\r\n ImageProcessingConfiguration.PrepareUniforms(uniforms, defines);\r\n ImageProcessingConfiguration.PrepareSamplers(samplers, defines);\r\n }\r\n\r\n MaterialHelper.PrepareUniformsAndSamplersList({\r\n uniformsNames: uniforms,\r\n uniformBuffersNames: uniformBuffers,\r\n samplers: samplers,\r\n defines: defines,\r\n maxSimultaneousLights: this._maxSimultaneousLights,\r\n });\r\n\r\n const csnrOptions: ICustomShaderNameResolveOptions = {};\r\n\r\n if (this.customShaderNameResolve) {\r\n shaderName = this.customShaderNameResolve(shaderName, uniforms, uniformBuffers, samplers, defines, attribs, csnrOptions);\r\n }\r\n\r\n const join = defines.toString();\r\n const effect = engine.createEffect(\r\n shaderName,\r\n {\r\n attributes: attribs,\r\n uniformsNames: uniforms,\r\n uniformBuffersNames: uniformBuffers,\r\n samplers: samplers,\r\n defines: join,\r\n fallbacks: fallbacks,\r\n onCompiled: onCompiled,\r\n onError: onError,\r\n indexParameters,\r\n processFinalCode: csnrOptions.processFinalCode,\r\n processCodeAfterIncludes: this._eventInfo.customCode,\r\n multiTarget: defines.PREPASS,\r\n },\r\n engine\r\n );\r\n\r\n this._eventInfo.customCode = undefined;\r\n\r\n return effect;\r\n }\r\n\r\n private _prepareDefines(\r\n mesh: AbstractMesh,\r\n defines: PBRMaterialDefines,\r\n useInstances: Nullable = null,\r\n useClipPlane: Nullable = null,\r\n useThinInstances: boolean = false\r\n ): void {\r\n const scene = this.getScene();\r\n const engine = scene.getEngine();\r\n\r\n // Lights\r\n MaterialHelper.PrepareDefinesForLights(scene, mesh, defines, true, this._maxSimultaneousLights, this._disableLighting);\r\n defines._needNormals = true;\r\n\r\n // Multiview\r\n MaterialHelper.PrepareDefinesForMultiview(scene, defines);\r\n\r\n // PrePass\r\n const oit = this.needAlphaBlendingForMesh(mesh) && this.getScene().useOrderIndependentTransparency;\r\n MaterialHelper.PrepareDefinesForPrePass(scene, defines, this.canRenderToMRT && !oit);\r\n\r\n // Order independant transparency\r\n MaterialHelper.PrepareDefinesForOIT(scene, defines, oit);\r\n\r\n // Textures\r\n defines.METALLICWORKFLOW = this.isMetallicWorkflow();\r\n if (defines._areTexturesDirty) {\r\n defines._needUVs = false;\r\n for (let i = 1; i <= Constants.MAX_SUPPORTED_UV_SETS; ++i) {\r\n defines[\"MAINUV\" + i] = false;\r\n }\r\n if (scene.texturesEnabled) {\r\n defines.ALBEDODIRECTUV = 0;\r\n defines.AMBIENTDIRECTUV = 0;\r\n defines.OPACITYDIRECTUV = 0;\r\n defines.EMISSIVEDIRECTUV = 0;\r\n defines.REFLECTIVITYDIRECTUV = 0;\r\n defines.MICROSURFACEMAPDIRECTUV = 0;\r\n defines.METALLIC_REFLECTANCEDIRECTUV = 0;\r\n defines.REFLECTANCEDIRECTUV = 0;\r\n defines.BUMPDIRECTUV = 0;\r\n defines.LIGHTMAPDIRECTUV = 0;\r\n\r\n if (engine.getCaps().textureLOD) {\r\n defines.LODBASEDMICROSFURACE = true;\r\n }\r\n\r\n if (this._albedoTexture && MaterialFlags.DiffuseTextureEnabled) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._albedoTexture, defines, \"ALBEDO\");\r\n defines.GAMMAALBEDO = this._albedoTexture.gammaSpace;\r\n } else {\r\n defines.ALBEDO = false;\r\n }\r\n\r\n if (this._ambientTexture && MaterialFlags.AmbientTextureEnabled) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._ambientTexture, defines, \"AMBIENT\");\r\n defines.AMBIENTINGRAYSCALE = this._useAmbientInGrayScale;\r\n } else {\r\n defines.AMBIENT = false;\r\n }\r\n\r\n if (this._opacityTexture && MaterialFlags.OpacityTextureEnabled) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._opacityTexture, defines, \"OPACITY\");\r\n defines.OPACITYRGB = this._opacityTexture.getAlphaFromRGB;\r\n } else {\r\n defines.OPACITY = false;\r\n }\r\n\r\n const reflectionTexture = this._getReflectionTexture();\r\n if (reflectionTexture && MaterialFlags.ReflectionTextureEnabled) {\r\n defines.REFLECTION = true;\r\n defines.GAMMAREFLECTION = reflectionTexture.gammaSpace;\r\n defines.RGBDREFLECTION = reflectionTexture.isRGBD;\r\n defines.LODINREFLECTIONALPHA = reflectionTexture.lodLevelInAlpha;\r\n defines.LINEARSPECULARREFLECTION = reflectionTexture.linearSpecularLOD;\r\n\r\n if (this.realTimeFiltering && this.realTimeFilteringQuality > 0) {\r\n defines.NUM_SAMPLES = \"\" + this.realTimeFilteringQuality;\r\n if (engine._features.needTypeSuffixInShaderConstants) {\r\n defines.NUM_SAMPLES = defines.NUM_SAMPLES + \"u\";\r\n }\r\n\r\n defines.REALTIME_FILTERING = true;\r\n } else {\r\n defines.REALTIME_FILTERING = false;\r\n }\r\n\r\n defines.INVERTCUBICMAP = reflectionTexture.coordinatesMode === Texture.INVCUBIC_MODE;\r\n defines.REFLECTIONMAP_3D = reflectionTexture.isCube;\r\n defines.REFLECTIONMAP_OPPOSITEZ = defines.REFLECTIONMAP_3D && this.getScene().useRightHandedSystem ? !reflectionTexture.invertZ : reflectionTexture.invertZ;\r\n\r\n defines.REFLECTIONMAP_CUBIC = false;\r\n defines.REFLECTIONMAP_EXPLICIT = false;\r\n defines.REFLECTIONMAP_PLANAR = false;\r\n defines.REFLECTIONMAP_PROJECTION = false;\r\n defines.REFLECTIONMAP_SKYBOX = false;\r\n defines.REFLECTIONMAP_SPHERICAL = false;\r\n defines.REFLECTIONMAP_EQUIRECTANGULAR = false;\r\n defines.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = false;\r\n defines.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = false;\r\n\r\n switch (reflectionTexture.coordinatesMode) {\r\n case Texture.EXPLICIT_MODE:\r\n defines.REFLECTIONMAP_EXPLICIT = true;\r\n break;\r\n case Texture.PLANAR_MODE:\r\n defines.REFLECTIONMAP_PLANAR = true;\r\n break;\r\n case Texture.PROJECTION_MODE:\r\n defines.REFLECTIONMAP_PROJECTION = true;\r\n break;\r\n case Texture.SKYBOX_MODE:\r\n defines.REFLECTIONMAP_SKYBOX = true;\r\n break;\r\n case Texture.SPHERICAL_MODE:\r\n defines.REFLECTIONMAP_SPHERICAL = true;\r\n break;\r\n case Texture.EQUIRECTANGULAR_MODE:\r\n defines.REFLECTIONMAP_EQUIRECTANGULAR = true;\r\n break;\r\n case Texture.FIXED_EQUIRECTANGULAR_MODE:\r\n defines.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = true;\r\n break;\r\n case Texture.FIXED_EQUIRECTANGULAR_MIRRORED_MODE:\r\n defines.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = true;\r\n break;\r\n case Texture.CUBIC_MODE:\r\n case Texture.INVCUBIC_MODE:\r\n default:\r\n defines.REFLECTIONMAP_CUBIC = true;\r\n defines.USE_LOCAL_REFLECTIONMAP_CUBIC = (reflectionTexture).boundingBoxSize ? true : false;\r\n break;\r\n }\r\n\r\n if (reflectionTexture.coordinatesMode !== Texture.SKYBOX_MODE) {\r\n if (reflectionTexture.irradianceTexture) {\r\n defines.USEIRRADIANCEMAP = true;\r\n defines.USESPHERICALFROMREFLECTIONMAP = false;\r\n }\r\n // Assume using spherical polynomial if the reflection texture is a cube map\r\n else if (reflectionTexture.isCube) {\r\n defines.USESPHERICALFROMREFLECTIONMAP = true;\r\n defines.USEIRRADIANCEMAP = false;\r\n if (this._forceIrradianceInFragment || this.realTimeFiltering || engine.getCaps().maxVaryingVectors <= 8) {\r\n defines.USESPHERICALINVERTEX = false;\r\n } else {\r\n defines.USESPHERICALINVERTEX = true;\r\n }\r\n }\r\n }\r\n } else {\r\n defines.REFLECTION = false;\r\n defines.REFLECTIONMAP_3D = false;\r\n defines.REFLECTIONMAP_SPHERICAL = false;\r\n defines.REFLECTIONMAP_PLANAR = false;\r\n defines.REFLECTIONMAP_CUBIC = false;\r\n defines.USE_LOCAL_REFLECTIONMAP_CUBIC = false;\r\n defines.REFLECTIONMAP_PROJECTION = false;\r\n defines.REFLECTIONMAP_SKYBOX = false;\r\n defines.REFLECTIONMAP_EXPLICIT = false;\r\n defines.REFLECTIONMAP_EQUIRECTANGULAR = false;\r\n defines.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = false;\r\n defines.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = false;\r\n defines.INVERTCUBICMAP = false;\r\n defines.USESPHERICALFROMREFLECTIONMAP = false;\r\n defines.USEIRRADIANCEMAP = false;\r\n defines.USESPHERICALINVERTEX = false;\r\n defines.REFLECTIONMAP_OPPOSITEZ = false;\r\n defines.LODINREFLECTIONALPHA = false;\r\n defines.GAMMAREFLECTION = false;\r\n defines.RGBDREFLECTION = false;\r\n defines.LINEARSPECULARREFLECTION = false;\r\n }\r\n\r\n if (this._lightmapTexture && MaterialFlags.LightmapTextureEnabled) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._lightmapTexture, defines, \"LIGHTMAP\");\r\n defines.USELIGHTMAPASSHADOWMAP = this._useLightmapAsShadowmap;\r\n defines.GAMMALIGHTMAP = this._lightmapTexture.gammaSpace;\r\n defines.RGBDLIGHTMAP = this._lightmapTexture.isRGBD;\r\n } else {\r\n defines.LIGHTMAP = false;\r\n }\r\n\r\n if (this._emissiveTexture && MaterialFlags.EmissiveTextureEnabled) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._emissiveTexture, defines, \"EMISSIVE\");\r\n defines.GAMMAEMISSIVE = this._emissiveTexture.gammaSpace;\r\n } else {\r\n defines.EMISSIVE = false;\r\n }\r\n\r\n if (MaterialFlags.SpecularTextureEnabled) {\r\n if (this._metallicTexture) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._metallicTexture, defines, \"REFLECTIVITY\");\r\n defines.ROUGHNESSSTOREINMETALMAPALPHA = this._useRoughnessFromMetallicTextureAlpha;\r\n defines.ROUGHNESSSTOREINMETALMAPGREEN = !this._useRoughnessFromMetallicTextureAlpha && this._useRoughnessFromMetallicTextureGreen;\r\n defines.METALLNESSSTOREINMETALMAPBLUE = this._useMetallnessFromMetallicTextureBlue;\r\n defines.AOSTOREINMETALMAPRED = this._useAmbientOcclusionFromMetallicTextureRed;\r\n defines.REFLECTIVITY_GAMMA = false;\r\n } else if (this._reflectivityTexture) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._reflectivityTexture, defines, \"REFLECTIVITY\");\r\n defines.MICROSURFACEFROMREFLECTIVITYMAP = this._useMicroSurfaceFromReflectivityMapAlpha;\r\n defines.MICROSURFACEAUTOMATIC = this._useAutoMicroSurfaceFromReflectivityMap;\r\n defines.REFLECTIVITY_GAMMA = this._reflectivityTexture.gammaSpace;\r\n } else {\r\n defines.REFLECTIVITY = false;\r\n }\r\n\r\n if (this._metallicReflectanceTexture || this._reflectanceTexture) {\r\n const identicalTextures =\r\n this._metallicReflectanceTexture !== null &&\r\n this._metallicReflectanceTexture._texture === this._reflectanceTexture?._texture &&\r\n this._metallicReflectanceTexture.checkTransformsAreIdentical(this._reflectanceTexture);\r\n\r\n defines.METALLIC_REFLECTANCE_USE_ALPHA_ONLY = this._useOnlyMetallicFromMetallicReflectanceTexture && !identicalTextures;\r\n if (this._metallicReflectanceTexture) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._metallicReflectanceTexture, defines, \"METALLIC_REFLECTANCE\");\r\n defines.METALLIC_REFLECTANCE_GAMMA = this._metallicReflectanceTexture.gammaSpace;\r\n } else {\r\n defines.METALLIC_REFLECTANCE = false;\r\n }\r\n if (\r\n this._reflectanceTexture &&\r\n !identicalTextures &&\r\n (!this._metallicReflectanceTexture || (this._metallicReflectanceTexture && this._useOnlyMetallicFromMetallicReflectanceTexture))\r\n ) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._reflectanceTexture, defines, \"REFLECTANCE\");\r\n defines.REFLECTANCE_GAMMA = this._reflectanceTexture.gammaSpace;\r\n } else {\r\n defines.REFLECTANCE = false;\r\n }\r\n } else {\r\n defines.METALLIC_REFLECTANCE = false;\r\n defines.REFLECTANCE = false;\r\n }\r\n\r\n if (this._microSurfaceTexture) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._microSurfaceTexture, defines, \"MICROSURFACEMAP\");\r\n } else {\r\n defines.MICROSURFACEMAP = false;\r\n }\r\n } else {\r\n defines.REFLECTIVITY = false;\r\n defines.MICROSURFACEMAP = false;\r\n }\r\n\r\n if (engine.getCaps().standardDerivatives && this._bumpTexture && MaterialFlags.BumpTextureEnabled && !this._disableBumpMap) {\r\n MaterialHelper.PrepareDefinesForMergedUV(this._bumpTexture, defines, \"BUMP\");\r\n\r\n if (this._useParallax && this._albedoTexture && MaterialFlags.DiffuseTextureEnabled) {\r\n defines.PARALLAX = true;\r\n defines.PARALLAXOCCLUSION = !!this._useParallaxOcclusion;\r\n } else {\r\n defines.PARALLAX = false;\r\n }\r\n\r\n defines.OBJECTSPACE_NORMALMAP = this._useObjectSpaceNormalMap;\r\n } else {\r\n defines.BUMP = false;\r\n defines.PARALLAX = false;\r\n defines.PARALLAXOCCLUSION = false;\r\n defines.OBJECTSPACE_NORMALMAP = false;\r\n }\r\n\r\n if (this._environmentBRDFTexture && MaterialFlags.ReflectionTextureEnabled) {\r\n defines.ENVIRONMENTBRDF = true;\r\n defines.ENVIRONMENTBRDF_RGBD = this._environmentBRDFTexture.isRGBD;\r\n } else {\r\n defines.ENVIRONMENTBRDF = false;\r\n defines.ENVIRONMENTBRDF_RGBD = false;\r\n }\r\n\r\n if (this._shouldUseAlphaFromAlbedoTexture()) {\r\n defines.ALPHAFROMALBEDO = true;\r\n } else {\r\n defines.ALPHAFROMALBEDO = false;\r\n }\r\n }\r\n\r\n defines.SPECULAROVERALPHA = this._useSpecularOverAlpha;\r\n\r\n if (this._lightFalloff === PBRBaseMaterial.LIGHTFALLOFF_STANDARD) {\r\n defines.USEPHYSICALLIGHTFALLOFF = false;\r\n defines.USEGLTFLIGHTFALLOFF = false;\r\n } else if (this._lightFalloff === PBRBaseMaterial.LIGHTFALLOFF_GLTF) {\r\n defines.USEPHYSICALLIGHTFALLOFF = false;\r\n defines.USEGLTFLIGHTFALLOFF = true;\r\n } else {\r\n defines.USEPHYSICALLIGHTFALLOFF = true;\r\n defines.USEGLTFLIGHTFALLOFF = false;\r\n }\r\n\r\n defines.RADIANCEOVERALPHA = this._useRadianceOverAlpha;\r\n\r\n if (!this.backFaceCulling && this._twoSidedLighting) {\r\n defines.TWOSIDEDLIGHTING = true;\r\n } else {\r\n defines.TWOSIDEDLIGHTING = false;\r\n }\r\n\r\n defines.SPECULARAA = engine.getCaps().standardDerivatives && this._enableSpecularAntiAliasing;\r\n }\r\n\r\n if (defines._areTexturesDirty || defines._areMiscDirty) {\r\n defines.ALPHATESTVALUE = `${this._alphaCutOff}${this._alphaCutOff % 1 === 0 ? \".\" : \"\"}`;\r\n defines.PREMULTIPLYALPHA = this.alphaMode === Constants.ALPHA_PREMULTIPLIED || this.alphaMode === Constants.ALPHA_PREMULTIPLIED_PORTERDUFF;\r\n defines.ALPHABLEND = this.needAlphaBlendingForMesh(mesh);\r\n defines.ALPHAFRESNEL = this._useAlphaFresnel || this._useLinearAlphaFresnel;\r\n defines.LINEARALPHAFRESNEL = this._useLinearAlphaFresnel;\r\n }\r\n\r\n if (defines._areImageProcessingDirty && this._imageProcessingConfiguration) {\r\n this._imageProcessingConfiguration.prepareDefines(defines);\r\n }\r\n\r\n defines.FORCENORMALFORWARD = this._forceNormalForward;\r\n\r\n defines.RADIANCEOCCLUSION = this._useRadianceOcclusion;\r\n\r\n defines.HORIZONOCCLUSION = this._useHorizonOcclusion;\r\n\r\n // Misc.\r\n if (defines._areMiscDirty) {\r\n MaterialHelper.PrepareDefinesForMisc(\r\n mesh,\r\n scene,\r\n this._useLogarithmicDepth,\r\n this.pointsCloud,\r\n this.fogEnabled,\r\n this._shouldTurnAlphaTestOn(mesh) || this._forceAlphaTest,\r\n defines,\r\n this._applyDecalMapAfterDetailMap\r\n );\r\n defines.UNLIT = this._unlit || ((this.pointsCloud || this.wireframe) && !mesh.isVerticesDataPresent(VertexBuffer.NormalKind));\r\n defines.DEBUGMODE = this._debugMode;\r\n }\r\n\r\n // Values that need to be evaluated on every frame\r\n MaterialHelper.PrepareDefinesForFrameBoundValues(scene, engine, this, defines, useInstances ? true : false, useClipPlane, useThinInstances);\r\n\r\n // External config\r\n this._eventInfo.defines = defines;\r\n this._eventInfo.mesh = mesh;\r\n this._callbackPluginEventPrepareDefinesBeforeAttributes(this._eventInfo);\r\n\r\n // Attribs\r\n MaterialHelper.PrepareDefinesForAttributes(mesh, defines, true, true, true, this._transparencyMode !== PBRBaseMaterial.PBRMATERIAL_OPAQUE);\r\n\r\n // External config\r\n this._callbackPluginEventPrepareDefines(this._eventInfo);\r\n }\r\n\r\n /**\r\n * Force shader compilation\r\n * @param mesh\r\n * @param onCompiled\r\n * @param options\r\n */\r\n public forceCompilation(mesh: AbstractMesh, onCompiled?: (material: Material) => void, options?: Partial): void {\r\n const localOptions = {\r\n clipPlane: false,\r\n useInstances: false,\r\n ...options,\r\n };\r\n\r\n if (!this._uniformBufferLayoutBuilt) {\r\n this.buildUniformLayout();\r\n }\r\n\r\n this._callbackPluginEventGeneric(MaterialPluginEvent.GetDefineNames, this._eventInfo);\r\n const defines = new PBRMaterialDefines(this._eventInfo.defineNames);\r\n const effect = this._prepareEffect(mesh, defines, undefined, undefined, localOptions.useInstances, localOptions.clipPlane, mesh.hasThinInstances)!;\r\n if (this._onEffectCreatedObservable) {\r\n onCreatedEffectParameters.effect = effect;\r\n onCreatedEffectParameters.subMesh = null;\r\n this._onEffectCreatedObservable.notifyObservers(onCreatedEffectParameters);\r\n }\r\n if (effect.isReady()) {\r\n if (onCompiled) {\r\n onCompiled(this);\r\n }\r\n } else {\r\n effect.onCompileObservable.add(() => {\r\n if (onCompiled) {\r\n onCompiled(this);\r\n }\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Initializes the uniform buffer layout for the shader.\r\n */\r\n public buildUniformLayout(): void {\r\n // Order is important !\r\n const ubo = this._uniformBuffer;\r\n ubo.addUniform(\"vAlbedoInfos\", 2);\r\n ubo.addUniform(\"vAmbientInfos\", 4);\r\n ubo.addUniform(\"vOpacityInfos\", 2);\r\n ubo.addUniform(\"vEmissiveInfos\", 2);\r\n ubo.addUniform(\"vLightmapInfos\", 2);\r\n ubo.addUniform(\"vReflectivityInfos\", 3);\r\n ubo.addUniform(\"vMicroSurfaceSamplerInfos\", 2);\r\n ubo.addUniform(\"vReflectionInfos\", 2);\r\n ubo.addUniform(\"vReflectionFilteringInfo\", 2);\r\n ubo.addUniform(\"vReflectionPosition\", 3);\r\n ubo.addUniform(\"vReflectionSize\", 3);\r\n ubo.addUniform(\"vBumpInfos\", 3);\r\n ubo.addUniform(\"albedoMatrix\", 16);\r\n ubo.addUniform(\"ambientMatrix\", 16);\r\n ubo.addUniform(\"opacityMatrix\", 16);\r\n ubo.addUniform(\"emissiveMatrix\", 16);\r\n ubo.addUniform(\"lightmapMatrix\", 16);\r\n ubo.addUniform(\"reflectivityMatrix\", 16);\r\n ubo.addUniform(\"microSurfaceSamplerMatrix\", 16);\r\n ubo.addUniform(\"bumpMatrix\", 16);\r\n ubo.addUniform(\"vTangentSpaceParams\", 2);\r\n ubo.addUniform(\"reflectionMatrix\", 16);\r\n\r\n ubo.addUniform(\"vReflectionColor\", 3);\r\n ubo.addUniform(\"vAlbedoColor\", 4);\r\n ubo.addUniform(\"vLightingIntensity\", 4);\r\n\r\n ubo.addUniform(\"vReflectionMicrosurfaceInfos\", 3);\r\n ubo.addUniform(\"pointSize\", 1);\r\n ubo.addUniform(\"vReflectivityColor\", 4);\r\n ubo.addUniform(\"vEmissiveColor\", 3);\r\n ubo.addUniform(\"vAmbientColor\", 3);\r\n\r\n ubo.addUniform(\"vDebugMode\", 2);\r\n\r\n ubo.addUniform(\"vMetallicReflectanceFactors\", 4);\r\n ubo.addUniform(\"vMetallicReflectanceInfos\", 2);\r\n ubo.addUniform(\"metallicReflectanceMatrix\", 16);\r\n ubo.addUniform(\"vReflectanceInfos\", 2);\r\n ubo.addUniform(\"reflectanceMatrix\", 16);\r\n\r\n ubo.addUniform(\"vSphericalL00\", 3);\r\n ubo.addUniform(\"vSphericalL1_1\", 3);\r\n ubo.addUniform(\"vSphericalL10\", 3);\r\n ubo.addUniform(\"vSphericalL11\", 3);\r\n ubo.addUniform(\"vSphericalL2_2\", 3);\r\n ubo.addUniform(\"vSphericalL2_1\", 3);\r\n ubo.addUniform(\"vSphericalL20\", 3);\r\n ubo.addUniform(\"vSphericalL21\", 3);\r\n ubo.addUniform(\"vSphericalL22\", 3);\r\n\r\n ubo.addUniform(\"vSphericalX\", 3);\r\n ubo.addUniform(\"vSphericalY\", 3);\r\n ubo.addUniform(\"vSphericalZ\", 3);\r\n ubo.addUniform(\"vSphericalXX_ZZ\", 3);\r\n ubo.addUniform(\"vSphericalYY_ZZ\", 3);\r\n ubo.addUniform(\"vSphericalZZ\", 3);\r\n ubo.addUniform(\"vSphericalXY\", 3);\r\n ubo.addUniform(\"vSphericalYZ\", 3);\r\n ubo.addUniform(\"vSphericalZX\", 3);\r\n\r\n super.buildUniformLayout();\r\n }\r\n\r\n /**\r\n * Binds the submesh data.\r\n * @param world - The world matrix.\r\n * @param mesh - The BJS mesh.\r\n * @param subMesh - A submesh of the BJS mesh.\r\n */\r\n public bindForSubMesh(world: Matrix, mesh: Mesh, subMesh: SubMesh): void {\r\n const scene = this.getScene();\r\n\r\n const defines = subMesh.materialDefines;\r\n if (!defines) {\r\n return;\r\n }\r\n\r\n const effect = subMesh.effect;\r\n\r\n if (!effect) {\r\n return;\r\n }\r\n\r\n this._activeEffect = effect;\r\n\r\n // Matrices Mesh.\r\n mesh.getMeshUniformBuffer().bindToEffect(effect, \"Mesh\");\r\n mesh.transferToEffect(world);\r\n\r\n const engine = scene.getEngine();\r\n\r\n // Binding unconditionally\r\n this._uniformBuffer.bindToEffect(effect, \"Material\");\r\n\r\n this.prePassConfiguration.bindForSubMesh(this._activeEffect, scene, mesh, world, this.isFrozen);\r\n\r\n this._eventInfo.subMesh = subMesh;\r\n this._callbackPluginEventHardBindForSubMesh(this._eventInfo);\r\n\r\n // Normal Matrix\r\n if (defines.OBJECTSPACE_NORMALMAP) {\r\n world.toNormalMatrix(this._normalMatrix);\r\n this.bindOnlyNormalMatrix(this._normalMatrix);\r\n }\r\n\r\n const mustRebind = effect._forceRebindOnNextCall || this._mustRebind(scene, effect, mesh.visibility);\r\n\r\n // Bones\r\n MaterialHelper.BindBonesParameters(mesh, this._activeEffect, this.prePassConfiguration);\r\n\r\n let reflectionTexture: Nullable = null;\r\n const ubo = this._uniformBuffer;\r\n if (mustRebind) {\r\n this.bindViewProjection(effect);\r\n reflectionTexture = this._getReflectionTexture();\r\n\r\n if (!ubo.useUbo || !this.isFrozen || !ubo.isSync || effect._forceRebindOnNextCall) {\r\n // Texture uniforms\r\n if (scene.texturesEnabled) {\r\n if (this._albedoTexture && MaterialFlags.DiffuseTextureEnabled) {\r\n ubo.updateFloat2(\"vAlbedoInfos\", this._albedoTexture.coordinatesIndex, this._albedoTexture.level);\r\n MaterialHelper.BindTextureMatrix(this._albedoTexture, ubo, \"albedo\");\r\n }\r\n\r\n if (this._ambientTexture && MaterialFlags.AmbientTextureEnabled) {\r\n ubo.updateFloat4(\r\n \"vAmbientInfos\",\r\n this._ambientTexture.coordinatesIndex,\r\n this._ambientTexture.level,\r\n this._ambientTextureStrength,\r\n this._ambientTextureImpactOnAnalyticalLights\r\n );\r\n MaterialHelper.BindTextureMatrix(this._ambientTexture, ubo, \"ambient\");\r\n }\r\n\r\n if (this._opacityTexture && MaterialFlags.OpacityTextureEnabled) {\r\n ubo.updateFloat2(\"vOpacityInfos\", this._opacityTexture.coordinatesIndex, this._opacityTexture.level);\r\n MaterialHelper.BindTextureMatrix(this._opacityTexture, ubo, \"opacity\");\r\n }\r\n\r\n if (reflectionTexture && MaterialFlags.ReflectionTextureEnabled) {\r\n ubo.updateMatrix(\"reflectionMatrix\", reflectionTexture.getReflectionTextureMatrix());\r\n ubo.updateFloat2(\"vReflectionInfos\", reflectionTexture.level, 0);\r\n\r\n if ((reflectionTexture).boundingBoxSize) {\r\n const cubeTexture = reflectionTexture;\r\n\r\n ubo.updateVector3(\"vReflectionPosition\", cubeTexture.boundingBoxPosition);\r\n ubo.updateVector3(\"vReflectionSize\", cubeTexture.boundingBoxSize);\r\n }\r\n\r\n if (this.realTimeFiltering) {\r\n const width = reflectionTexture.getSize().width;\r\n ubo.updateFloat2(\"vReflectionFilteringInfo\", width, Scalar.Log2(width));\r\n }\r\n\r\n if (!defines.USEIRRADIANCEMAP) {\r\n const polynomials = reflectionTexture.sphericalPolynomial;\r\n if (defines.USESPHERICALFROMREFLECTIONMAP && polynomials) {\r\n if (defines.SPHERICAL_HARMONICS) {\r\n const preScaledHarmonics = polynomials.preScaledHarmonics;\r\n ubo.updateVector3(\"vSphericalL00\", preScaledHarmonics.l00);\r\n ubo.updateVector3(\"vSphericalL1_1\", preScaledHarmonics.l1_1);\r\n ubo.updateVector3(\"vSphericalL10\", preScaledHarmonics.l10);\r\n ubo.updateVector3(\"vSphericalL11\", preScaledHarmonics.l11);\r\n ubo.updateVector3(\"vSphericalL2_2\", preScaledHarmonics.l2_2);\r\n ubo.updateVector3(\"vSphericalL2_1\", preScaledHarmonics.l2_1);\r\n ubo.updateVector3(\"vSphericalL20\", preScaledHarmonics.l20);\r\n ubo.updateVector3(\"vSphericalL21\", preScaledHarmonics.l21);\r\n ubo.updateVector3(\"vSphericalL22\", preScaledHarmonics.l22);\r\n } else {\r\n ubo.updateFloat3(\"vSphericalX\", polynomials.x.x, polynomials.x.y, polynomials.x.z);\r\n ubo.updateFloat3(\"vSphericalY\", polynomials.y.x, polynomials.y.y, polynomials.y.z);\r\n ubo.updateFloat3(\"vSphericalZ\", polynomials.z.x, polynomials.z.y, polynomials.z.z);\r\n ubo.updateFloat3(\r\n \"vSphericalXX_ZZ\",\r\n polynomials.xx.x - polynomials.zz.x,\r\n polynomials.xx.y - polynomials.zz.y,\r\n polynomials.xx.z - polynomials.zz.z\r\n );\r\n ubo.updateFloat3(\r\n \"vSphericalYY_ZZ\",\r\n polynomials.yy.x - polynomials.zz.x,\r\n polynomials.yy.y - polynomials.zz.y,\r\n polynomials.yy.z - polynomials.zz.z\r\n );\r\n ubo.updateFloat3(\"vSphericalZZ\", polynomials.zz.x, polynomials.zz.y, polynomials.zz.z);\r\n ubo.updateFloat3(\"vSphericalXY\", polynomials.xy.x, polynomials.xy.y, polynomials.xy.z);\r\n ubo.updateFloat3(\"vSphericalYZ\", polynomials.yz.x, polynomials.yz.y, polynomials.yz.z);\r\n ubo.updateFloat3(\"vSphericalZX\", polynomials.zx.x, polynomials.zx.y, polynomials.zx.z);\r\n }\r\n }\r\n }\r\n\r\n ubo.updateFloat3(\r\n \"vReflectionMicrosurfaceInfos\",\r\n reflectionTexture.getSize().width,\r\n reflectionTexture.lodGenerationScale,\r\n reflectionTexture.lodGenerationOffset\r\n );\r\n }\r\n\r\n if (this._emissiveTexture && MaterialFlags.EmissiveTextureEnabled) {\r\n ubo.updateFloat2(\"vEmissiveInfos\", this._emissiveTexture.coordinatesIndex, this._emissiveTexture.level);\r\n MaterialHelper.BindTextureMatrix(this._emissiveTexture, ubo, \"emissive\");\r\n }\r\n\r\n if (this._lightmapTexture && MaterialFlags.LightmapTextureEnabled) {\r\n ubo.updateFloat2(\"vLightmapInfos\", this._lightmapTexture.coordinatesIndex, this._lightmapTexture.level);\r\n MaterialHelper.BindTextureMatrix(this._lightmapTexture, ubo, \"lightmap\");\r\n }\r\n\r\n if (MaterialFlags.SpecularTextureEnabled) {\r\n if (this._metallicTexture) {\r\n ubo.updateFloat3(\"vReflectivityInfos\", this._metallicTexture.coordinatesIndex, this._metallicTexture.level, this._ambientTextureStrength);\r\n MaterialHelper.BindTextureMatrix(this._metallicTexture, ubo, \"reflectivity\");\r\n } else if (this._reflectivityTexture) {\r\n ubo.updateFloat3(\"vReflectivityInfos\", this._reflectivityTexture.coordinatesIndex, this._reflectivityTexture.level, 1.0);\r\n MaterialHelper.BindTextureMatrix(this._reflectivityTexture, ubo, \"reflectivity\");\r\n }\r\n\r\n if (this._metallicReflectanceTexture) {\r\n ubo.updateFloat2(\"vMetallicReflectanceInfos\", this._metallicReflectanceTexture.coordinatesIndex, this._metallicReflectanceTexture.level);\r\n MaterialHelper.BindTextureMatrix(this._metallicReflectanceTexture, ubo, \"metallicReflectance\");\r\n }\r\n\r\n if (this._reflectanceTexture && defines.REFLECTANCE) {\r\n ubo.updateFloat2(\"vReflectanceInfos\", this._reflectanceTexture.coordinatesIndex, this._reflectanceTexture.level);\r\n MaterialHelper.BindTextureMatrix(this._reflectanceTexture, ubo, \"reflectance\");\r\n }\r\n\r\n if (this._microSurfaceTexture) {\r\n ubo.updateFloat2(\"vMicroSurfaceSamplerInfos\", this._microSurfaceTexture.coordinatesIndex, this._microSurfaceTexture.level);\r\n MaterialHelper.BindTextureMatrix(this._microSurfaceTexture, ubo, \"microSurfaceSampler\");\r\n }\r\n }\r\n\r\n if (this._bumpTexture && engine.getCaps().standardDerivatives && MaterialFlags.BumpTextureEnabled && !this._disableBumpMap) {\r\n ubo.updateFloat3(\"vBumpInfos\", this._bumpTexture.coordinatesIndex, this._bumpTexture.level, this._parallaxScaleBias);\r\n MaterialHelper.BindTextureMatrix(this._bumpTexture, ubo, \"bump\");\r\n\r\n if (scene._mirroredCameraPosition) {\r\n ubo.updateFloat2(\"vTangentSpaceParams\", this._invertNormalMapX ? 1.0 : -1.0, this._invertNormalMapY ? 1.0 : -1.0);\r\n } else {\r\n ubo.updateFloat2(\"vTangentSpaceParams\", this._invertNormalMapX ? -1.0 : 1.0, this._invertNormalMapY ? -1.0 : 1.0);\r\n }\r\n }\r\n }\r\n\r\n // Point size\r\n if (this.pointsCloud) {\r\n ubo.updateFloat(\"pointSize\", this.pointSize);\r\n }\r\n\r\n // Colors\r\n if (defines.METALLICWORKFLOW) {\r\n TmpColors.Color3[0].r = this._metallic === undefined || this._metallic === null ? 1 : this._metallic;\r\n TmpColors.Color3[0].g = this._roughness === undefined || this._roughness === null ? 1 : this._roughness;\r\n ubo.updateColor4(\"vReflectivityColor\", TmpColors.Color3[0], 1);\r\n\r\n const ior = this.subSurface?._indexOfRefraction ?? 1.5;\r\n const outsideIOR = 1; // consider air as clear coat and other layers would remap in the shader.\r\n\r\n // We are here deriving our default reflectance from a common value for none metallic surface.\r\n // Based of the schlick fresnel approximation model\r\n // for dielectrics.\r\n const f0 = Math.pow((ior - outsideIOR) / (ior + outsideIOR), 2);\r\n\r\n // Tweak the default F0 and F90 based on our given setup\r\n this._metallicReflectanceColor.scaleToRef(f0 * this._metallicF0Factor, TmpColors.Color3[0]);\r\n const metallicF90 = this._metallicF0Factor;\r\n\r\n ubo.updateColor4(\"vMetallicReflectanceFactors\", TmpColors.Color3[0], metallicF90);\r\n } else {\r\n ubo.updateColor4(\"vReflectivityColor\", this._reflectivityColor, this._microSurface);\r\n }\r\n\r\n ubo.updateColor3(\"vEmissiveColor\", MaterialFlags.EmissiveTextureEnabled ? this._emissiveColor : Color3.BlackReadOnly);\r\n ubo.updateColor3(\"vReflectionColor\", this._reflectionColor);\r\n if (!defines.SS_REFRACTION && this.subSurface?._linkRefractionWithTransparency) {\r\n ubo.updateColor4(\"vAlbedoColor\", this._albedoColor, 1);\r\n } else {\r\n ubo.updateColor4(\"vAlbedoColor\", this._albedoColor, this.alpha);\r\n }\r\n\r\n // Misc\r\n this._lightingInfos.x = this._directIntensity;\r\n this._lightingInfos.y = this._emissiveIntensity;\r\n this._lightingInfos.z = this._environmentIntensity * scene.environmentIntensity;\r\n this._lightingInfos.w = this._specularIntensity;\r\n\r\n ubo.updateVector4(\"vLightingIntensity\", this._lightingInfos);\r\n\r\n // Colors\r\n scene.ambientColor.multiplyToRef(this._ambientColor, this._globalAmbientColor);\r\n\r\n ubo.updateColor3(\"vAmbientColor\", this._globalAmbientColor);\r\n\r\n ubo.updateFloat2(\"vDebugMode\", this.debugLimit, this.debugFactor);\r\n }\r\n\r\n // Textures\r\n if (scene.texturesEnabled) {\r\n if (this._albedoTexture && MaterialFlags.DiffuseTextureEnabled) {\r\n ubo.setTexture(\"albedoSampler\", this._albedoTexture);\r\n }\r\n\r\n if (this._ambientTexture && MaterialFlags.AmbientTextureEnabled) {\r\n ubo.setTexture(\"ambientSampler\", this._ambientTexture);\r\n }\r\n\r\n if (this._opacityTexture && MaterialFlags.OpacityTextureEnabled) {\r\n ubo.setTexture(\"opacitySampler\", this._opacityTexture);\r\n }\r\n\r\n if (reflectionTexture && MaterialFlags.ReflectionTextureEnabled) {\r\n if (defines.LODBASEDMICROSFURACE) {\r\n ubo.setTexture(\"reflectionSampler\", reflectionTexture);\r\n } else {\r\n ubo.setTexture(\"reflectionSampler\", reflectionTexture._lodTextureMid || reflectionTexture);\r\n ubo.setTexture(\"reflectionSamplerLow\", reflectionTexture._lodTextureLow || reflectionTexture);\r\n ubo.setTexture(\"reflectionSamplerHigh\", reflectionTexture._lodTextureHigh || reflectionTexture);\r\n }\r\n\r\n if (defines.USEIRRADIANCEMAP) {\r\n ubo.setTexture(\"irradianceSampler\", reflectionTexture.irradianceTexture);\r\n }\r\n }\r\n\r\n if (defines.ENVIRONMENTBRDF) {\r\n ubo.setTexture(\"environmentBrdfSampler\", this._environmentBRDFTexture);\r\n }\r\n\r\n if (this._emissiveTexture && MaterialFlags.EmissiveTextureEnabled) {\r\n ubo.setTexture(\"emissiveSampler\", this._emissiveTexture);\r\n }\r\n\r\n if (this._lightmapTexture && MaterialFlags.LightmapTextureEnabled) {\r\n ubo.setTexture(\"lightmapSampler\", this._lightmapTexture);\r\n }\r\n\r\n if (MaterialFlags.SpecularTextureEnabled) {\r\n if (this._metallicTexture) {\r\n ubo.setTexture(\"reflectivitySampler\", this._metallicTexture);\r\n } else if (this._reflectivityTexture) {\r\n ubo.setTexture(\"reflectivitySampler\", this._reflectivityTexture);\r\n }\r\n\r\n if (this._metallicReflectanceTexture) {\r\n ubo.setTexture(\"metallicReflectanceSampler\", this._metallicReflectanceTexture);\r\n }\r\n\r\n if (this._reflectanceTexture && defines.REFLECTANCE) {\r\n ubo.setTexture(\"reflectanceSampler\", this._reflectanceTexture);\r\n }\r\n\r\n if (this._microSurfaceTexture) {\r\n ubo.setTexture(\"microSurfaceSampler\", this._microSurfaceTexture);\r\n }\r\n }\r\n\r\n if (this._bumpTexture && engine.getCaps().standardDerivatives && MaterialFlags.BumpTextureEnabled && !this._disableBumpMap) {\r\n ubo.setTexture(\"bumpSampler\", this._bumpTexture);\r\n }\r\n }\r\n\r\n // OIT with depth peeling\r\n if (this.getScene().useOrderIndependentTransparency && this.needAlphaBlendingForMesh(mesh)) {\r\n this.getScene().depthPeelingRenderer!.bind(effect);\r\n }\r\n\r\n this._eventInfo.subMesh = subMesh;\r\n this._callbackPluginEventBindForSubMesh(this._eventInfo);\r\n\r\n // Clip plane\r\n bindClipPlane(this._activeEffect, this, scene);\r\n\r\n this.bindEyePosition(effect);\r\n } else if (scene.getEngine()._features.needToAlwaysBindUniformBuffers) {\r\n this._needToBindSceneUbo = true;\r\n }\r\n\r\n if (mustRebind || !this.isFrozen) {\r\n // Lights\r\n if (scene.lightsEnabled && !this._disableLighting) {\r\n MaterialHelper.BindLights(scene, mesh, this._activeEffect, defines, this._maxSimultaneousLights);\r\n }\r\n\r\n // View\r\n if ((scene.fogEnabled && mesh.applyFog && scene.fogMode !== Scene.FOGMODE_NONE) || reflectionTexture || mesh.receiveShadows || defines.PREPASS) {\r\n this.bindView(effect);\r\n }\r\n\r\n // Fog\r\n MaterialHelper.BindFogParameters(scene, mesh, this._activeEffect, true);\r\n\r\n // Morph targets\r\n if (defines.NUM_MORPH_INFLUENCERS) {\r\n MaterialHelper.BindMorphTargetParameters(mesh, this._activeEffect);\r\n }\r\n\r\n if (defines.BAKED_VERTEX_ANIMATION_TEXTURE) {\r\n mesh.bakedVertexAnimationManager?.bind(effect, defines.INSTANCES);\r\n }\r\n\r\n // image processing\r\n this._imageProcessingConfiguration!.bind(this._activeEffect);\r\n\r\n // Log. depth\r\n MaterialHelper.BindLogDepth(defines, this._activeEffect, scene);\r\n }\r\n\r\n this._afterBind(mesh, this._activeEffect);\r\n\r\n ubo.update();\r\n }\r\n\r\n /**\r\n * Returns the animatable textures.\r\n * If material have animatable metallic texture, then reflectivity texture will not be returned, even if it has animations.\r\n * @returns - Array of animatable textures.\r\n */\r\n public getAnimatables(): IAnimatable[] {\r\n const results = super.getAnimatables();\r\n\r\n if (this._albedoTexture && this._albedoTexture.animations && this._albedoTexture.animations.length > 0) {\r\n results.push(this._albedoTexture);\r\n }\r\n\r\n if (this._ambientTexture && this._ambientTexture.animations && this._ambientTexture.animations.length > 0) {\r\n results.push(this._ambientTexture);\r\n }\r\n\r\n if (this._opacityTexture && this._opacityTexture.animations && this._opacityTexture.animations.length > 0) {\r\n results.push(this._opacityTexture);\r\n }\r\n\r\n if (this._reflectionTexture && this._reflectionTexture.animations && this._reflectionTexture.animations.length > 0) {\r\n results.push(this._reflectionTexture);\r\n }\r\n\r\n if (this._emissiveTexture && this._emissiveTexture.animations && this._emissiveTexture.animations.length > 0) {\r\n results.push(this._emissiveTexture);\r\n }\r\n\r\n if (this._metallicTexture && this._metallicTexture.animations && this._metallicTexture.animations.length > 0) {\r\n results.push(this._metallicTexture);\r\n } else if (this._reflectivityTexture && this._reflectivityTexture.animations && this._reflectivityTexture.animations.length > 0) {\r\n results.push(this._reflectivityTexture);\r\n }\r\n\r\n if (this._bumpTexture && this._bumpTexture.animations && this._bumpTexture.animations.length > 0) {\r\n results.push(this._bumpTexture);\r\n }\r\n\r\n if (this._lightmapTexture && this._lightmapTexture.animations && this._lightmapTexture.animations.length > 0) {\r\n results.push(this._lightmapTexture);\r\n }\r\n\r\n if (this._metallicReflectanceTexture && this._metallicReflectanceTexture.animations && this._metallicReflectanceTexture.animations.length > 0) {\r\n results.push(this._metallicReflectanceTexture);\r\n }\r\n\r\n if (this._reflectanceTexture && this._reflectanceTexture.animations && this._reflectanceTexture.animations.length > 0) {\r\n results.push(this._reflectanceTexture);\r\n }\r\n\r\n if (this._microSurfaceTexture && this._microSurfaceTexture.animations && this._microSurfaceTexture.animations.length > 0) {\r\n results.push(this._microSurfaceTexture);\r\n }\r\n\r\n return results;\r\n }\r\n\r\n /**\r\n * Returns the texture used for reflections.\r\n * @returns - Reflection texture if present. Otherwise, returns the environment texture.\r\n */\r\n private _getReflectionTexture(): Nullable {\r\n if (this._reflectionTexture) {\r\n return this._reflectionTexture;\r\n }\r\n\r\n return this.getScene().environmentTexture;\r\n }\r\n\r\n /**\r\n * Returns an array of the actively used textures.\r\n * @returns - Array of BaseTextures\r\n */\r\n public getActiveTextures(): BaseTexture[] {\r\n const activeTextures = super.getActiveTextures();\r\n\r\n if (this._albedoTexture) {\r\n activeTextures.push(this._albedoTexture);\r\n }\r\n\r\n if (this._ambientTexture) {\r\n activeTextures.push(this._ambientTexture);\r\n }\r\n\r\n if (this._opacityTexture) {\r\n activeTextures.push(this._opacityTexture);\r\n }\r\n\r\n if (this._reflectionTexture) {\r\n activeTextures.push(this._reflectionTexture);\r\n }\r\n\r\n if (this._emissiveTexture) {\r\n activeTextures.push(this._emissiveTexture);\r\n }\r\n\r\n if (this._reflectivityTexture) {\r\n activeTextures.push(this._reflectivityTexture);\r\n }\r\n\r\n if (this._metallicTexture) {\r\n activeTextures.push(this._metallicTexture);\r\n }\r\n\r\n if (this._metallicReflectanceTexture) {\r\n activeTextures.push(this._metallicReflectanceTexture);\r\n }\r\n\r\n if (this._reflectanceTexture) {\r\n activeTextures.push(this._reflectanceTexture);\r\n }\r\n\r\n if (this._microSurfaceTexture) {\r\n activeTextures.push(this._microSurfaceTexture);\r\n }\r\n\r\n if (this._bumpTexture) {\r\n activeTextures.push(this._bumpTexture);\r\n }\r\n\r\n if (this._lightmapTexture) {\r\n activeTextures.push(this._lightmapTexture);\r\n }\r\n\r\n return activeTextures;\r\n }\r\n\r\n /**\r\n * Checks to see if a texture is used in the material.\r\n * @param texture - Base texture to use.\r\n * @returns - Boolean specifying if a texture is used in the material.\r\n */\r\n public hasTexture(texture: BaseTexture): boolean {\r\n if (super.hasTexture(texture)) {\r\n return true;\r\n }\r\n\r\n if (this._albedoTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._ambientTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._opacityTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._reflectionTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._emissiveTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._reflectivityTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._metallicTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._metallicReflectanceTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._reflectanceTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._microSurfaceTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._bumpTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._lightmapTexture === texture) {\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Sets the required values to the prepass renderer.\r\n * It can't be sets when subsurface scattering of this material is disabled.\r\n * When scene have ability to enable subsurface prepass effect, it will enable.\r\n */\r\n public setPrePassRenderer(): boolean {\r\n if (!this.subSurface?.isScatteringEnabled) {\r\n return false;\r\n }\r\n\r\n const subSurfaceConfiguration = this.getScene().enableSubSurfaceForPrePass();\r\n if (subSurfaceConfiguration) {\r\n subSurfaceConfiguration.enabled = true;\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Disposes the resources of the material.\r\n * @param forceDisposeEffect - Forces the disposal of effects.\r\n * @param forceDisposeTextures - Forces the disposal of all textures.\r\n */\r\n public dispose(forceDisposeEffect?: boolean, forceDisposeTextures?: boolean): void {\r\n if (forceDisposeTextures) {\r\n if (this._environmentBRDFTexture && this.getScene().environmentBRDFTexture !== this._environmentBRDFTexture) {\r\n this._environmentBRDFTexture.dispose();\r\n }\r\n\r\n this._albedoTexture?.dispose();\r\n this._ambientTexture?.dispose();\r\n this._opacityTexture?.dispose();\r\n this._reflectionTexture?.dispose();\r\n this._emissiveTexture?.dispose();\r\n this._metallicTexture?.dispose();\r\n this._reflectivityTexture?.dispose();\r\n this._bumpTexture?.dispose();\r\n this._lightmapTexture?.dispose();\r\n this._metallicReflectanceTexture?.dispose();\r\n this._reflectanceTexture?.dispose();\r\n this._microSurfaceTexture?.dispose();\r\n }\r\n\r\n this._renderTargets.dispose();\r\n\r\n if (this._imageProcessingConfiguration && this._imageProcessingObserver) {\r\n this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver);\r\n }\r\n\r\n super.dispose(forceDisposeEffect, forceDisposeTextures);\r\n }\r\n}\r\n","import { serialize, SerializationHelper, serializeAsColor3, expandToProperty, serializeAsTexture } from \"../../Misc/decorators\";\r\nimport { GetEnvironmentBRDFTexture } from \"../../Misc/brdfTextureTools\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { Color3 } from \"../../Maths/math.color\";\r\nimport type { ImageProcessingConfiguration } from \"../../Materials/imageProcessingConfiguration\";\r\nimport type { ColorCurves } from \"../../Materials/colorCurves\";\r\nimport type { BaseTexture } from \"../../Materials/Textures/baseTexture\";\r\nimport { PBRBaseMaterial } from \"./pbrBaseMaterial\";\r\nimport { RegisterClass } from \"../../Misc/typeStore\";\r\nimport { Material } from \"../material\";\r\n\r\n/**\r\n * The Physically based material of BJS.\r\n *\r\n * This offers the main features of a standard PBR material.\r\n * For more information, please refer to the documentation :\r\n * https://doc.babylonjs.com/features/featuresDeepDive/materials/using/introToPBR\r\n */\r\nexport class PBRMaterial extends PBRBaseMaterial {\r\n /**\r\n * PBRMaterialTransparencyMode: No transparency mode, Alpha channel is not use.\r\n */\r\n public static readonly PBRMATERIAL_OPAQUE = PBRBaseMaterial.PBRMATERIAL_OPAQUE;\r\n\r\n /**\r\n * PBRMaterialTransparencyMode: Alpha Test mode, pixel are discarded below a certain threshold defined by the alpha cutoff value.\r\n */\r\n public static readonly PBRMATERIAL_ALPHATEST = PBRBaseMaterial.PBRMATERIAL_ALPHATEST;\r\n\r\n /**\r\n * PBRMaterialTransparencyMode: Pixels are blended (according to the alpha mode) with the already drawn pixels in the current frame buffer.\r\n */\r\n public static readonly PBRMATERIAL_ALPHABLEND = PBRBaseMaterial.PBRMATERIAL_ALPHABLEND;\r\n\r\n /**\r\n * PBRMaterialTransparencyMode: Pixels are blended (according to the alpha mode) with the already drawn pixels in the current frame buffer.\r\n * They are also discarded below the alpha cutoff threshold to improve performances.\r\n */\r\n public static readonly PBRMATERIAL_ALPHATESTANDBLEND = PBRBaseMaterial.PBRMATERIAL_ALPHATESTANDBLEND;\r\n\r\n /**\r\n * Defines the default value of how much AO map is occluding the analytical lights\r\n * (point spot...).\r\n */\r\n public static DEFAULT_AO_ON_ANALYTICAL_LIGHTS = PBRBaseMaterial.DEFAULT_AO_ON_ANALYTICAL_LIGHTS;\r\n\r\n /**\r\n * Intensity of the direct lights e.g. the four lights available in your scene.\r\n * This impacts both the direct diffuse and specular highlights.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public directIntensity: number = 1.0;\r\n\r\n /**\r\n * Intensity of the emissive part of the material.\r\n * This helps controlling the emissive effect without modifying the emissive color.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public emissiveIntensity: number = 1.0;\r\n\r\n /**\r\n * Intensity of the environment e.g. how much the environment will light the object\r\n * either through harmonics for rough material or through the reflection for shiny ones.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public environmentIntensity: number = 1.0;\r\n\r\n /**\r\n * This is a special control allowing the reduction of the specular highlights coming from the\r\n * four lights of the scene. Those highlights may not be needed in full environment lighting.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public specularIntensity: number = 1.0;\r\n\r\n /**\r\n * Debug Control allowing disabling the bump map on this material.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public disableBumpMap: boolean = false;\r\n\r\n /**\r\n * AKA Diffuse Texture in standard nomenclature.\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public albedoTexture: Nullable;\r\n\r\n /**\r\n * AKA Occlusion Texture in other nomenclature.\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public ambientTexture: Nullable;\r\n\r\n /**\r\n * AKA Occlusion Texture Intensity in other nomenclature.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public ambientTextureStrength: number = 1.0;\r\n\r\n /**\r\n * Defines how much the AO map is occluding the analytical lights (point spot...).\r\n * 1 means it completely occludes it\r\n * 0 mean it has no impact\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public ambientTextureImpactOnAnalyticalLights: number = PBRMaterial.DEFAULT_AO_ON_ANALYTICAL_LIGHTS;\r\n\r\n /**\r\n * Stores the alpha values in a texture. Use luminance if texture.getAlphaFromRGB is true.\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesAndMiscDirty\")\r\n public opacityTexture: Nullable;\r\n\r\n /**\r\n * Stores the reflection values in a texture.\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public reflectionTexture: Nullable;\r\n\r\n /**\r\n * Stores the emissive values in a texture.\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public emissiveTexture: Nullable;\r\n\r\n /**\r\n * AKA Specular texture in other nomenclature.\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public reflectivityTexture: Nullable;\r\n\r\n /**\r\n * Used to switch from specular/glossiness to metallic/roughness workflow.\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public metallicTexture: Nullable;\r\n\r\n /**\r\n * Specifies the metallic scalar of the metallic/roughness workflow.\r\n * Can also be used to scale the metalness values of the metallic texture.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public metallic: Nullable;\r\n\r\n /**\r\n * Specifies the roughness scalar of the metallic/roughness workflow.\r\n * Can also be used to scale the roughness values of the metallic texture.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public roughness: Nullable;\r\n\r\n /**\r\n * In metallic workflow, specifies an F0 factor to help configuring the material F0.\r\n * By default the indexOfrefraction is used to compute F0;\r\n *\r\n * This is used as a factor against the default reflectance at normal incidence to tweak it.\r\n *\r\n * F0 = defaultF0 * metallicF0Factor * metallicReflectanceColor;\r\n * F90 = metallicReflectanceColor;\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public metallicF0Factor = 1;\r\n\r\n /**\r\n * In metallic workflow, specifies an F0 color.\r\n * By default the F90 is always 1;\r\n *\r\n * Please note that this factor is also used as a factor against the default reflectance at normal incidence.\r\n *\r\n * F0 = defaultF0_from_IOR * metallicF0Factor * metallicReflectanceColor\r\n * F90 = metallicF0Factor;\r\n */\r\n @serializeAsColor3()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public metallicReflectanceColor = Color3.White();\r\n\r\n /**\r\n * Specifies that only the A channel from metallicReflectanceTexture should be used.\r\n * If false, both RGB and A channels will be used\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useOnlyMetallicFromMetallicReflectanceTexture = false;\r\n\r\n /**\r\n * Defines to store metallicReflectanceColor in RGB and metallicF0Factor in A\r\n * This is multiplied against the scalar values defined in the material.\r\n * If useOnlyMetallicFromMetallicReflectanceTexture is true, don't use the RGB channels, only A\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public metallicReflectanceTexture: Nullable;\r\n\r\n /**\r\n * Defines to store reflectanceColor in RGB\r\n * This is multiplied against the scalar values defined in the material.\r\n * If both reflectanceTexture and metallicReflectanceTexture textures are provided and useOnlyMetallicFromMetallicReflectanceTexture\r\n * is false, metallicReflectanceTexture takes priority and reflectanceTexture is not used\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public reflectanceTexture: Nullable;\r\n\r\n /**\r\n * Used to enable roughness/glossiness fetch from a separate channel depending on the current mode.\r\n * Gray Scale represents roughness in metallic mode and glossiness in specular mode.\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public microSurfaceTexture: Nullable;\r\n\r\n /**\r\n * Stores surface normal data used to displace a mesh in a texture.\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public bumpTexture: Nullable;\r\n\r\n /**\r\n * Stores the pre-calculated light information of a mesh in a texture.\r\n */\r\n @serializeAsTexture()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\", null)\r\n public lightmapTexture: Nullable;\r\n\r\n /**\r\n * Stores the refracted light information in a texture.\r\n */\r\n public get refractionTexture(): Nullable {\r\n return this.subSurface.refractionTexture;\r\n }\r\n public set refractionTexture(value: Nullable) {\r\n this.subSurface.refractionTexture = value;\r\n if (value) {\r\n this.subSurface.isRefractionEnabled = true;\r\n } else if (!this.subSurface.linkRefractionWithTransparency) {\r\n this.subSurface.isRefractionEnabled = false;\r\n }\r\n }\r\n\r\n /**\r\n * The color of a material in ambient lighting.\r\n */\r\n @serializeAsColor3(\"ambient\")\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public ambientColor = new Color3(0, 0, 0);\r\n\r\n /**\r\n * AKA Diffuse Color in other nomenclature.\r\n */\r\n @serializeAsColor3(\"albedo\")\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public albedoColor = new Color3(1, 1, 1);\r\n\r\n /**\r\n * AKA Specular Color in other nomenclature.\r\n */\r\n @serializeAsColor3(\"reflectivity\")\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public reflectivityColor = new Color3(1, 1, 1);\r\n\r\n /**\r\n * The color reflected from the material.\r\n */\r\n @serializeAsColor3(\"reflection\")\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public reflectionColor = new Color3(1.0, 1.0, 1.0);\r\n\r\n /**\r\n * The color emitted from the material.\r\n */\r\n @serializeAsColor3(\"emissive\")\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public emissiveColor = new Color3(0, 0, 0);\r\n\r\n /**\r\n * AKA Glossiness in other nomenclature.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public microSurface = 1.0;\r\n\r\n /**\r\n * Index of refraction of the material base layer.\r\n * https://en.wikipedia.org/wiki/List_of_refractive_indices\r\n *\r\n * This does not only impact refraction but also the Base F0 of Dielectric Materials.\r\n *\r\n * From dielectric fresnel rules: F0 = square((iorT - iorI) / (iorT + iorI))\r\n */\r\n public get indexOfRefraction(): number {\r\n return this.subSurface.indexOfRefraction;\r\n }\r\n public set indexOfRefraction(value: number) {\r\n this.subSurface.indexOfRefraction = value;\r\n }\r\n\r\n /**\r\n * Controls if refraction needs to be inverted on Y. This could be useful for procedural texture.\r\n */\r\n public get invertRefractionY(): boolean {\r\n return this.subSurface.invertRefractionY;\r\n }\r\n public set invertRefractionY(value: boolean) {\r\n this.subSurface.invertRefractionY = value;\r\n }\r\n\r\n /**\r\n * This parameters will make the material used its opacity to control how much it is refracting against not.\r\n * Materials half opaque for instance using refraction could benefit from this control.\r\n */\r\n public get linkRefractionWithTransparency(): boolean {\r\n return this.subSurface.linkRefractionWithTransparency;\r\n }\r\n public set linkRefractionWithTransparency(value: boolean) {\r\n this.subSurface.linkRefractionWithTransparency = value;\r\n if (value) {\r\n this.subSurface.isRefractionEnabled = true;\r\n }\r\n }\r\n\r\n /**\r\n * If true, the light map contains occlusion information instead of lighting info.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useLightmapAsShadowmap = false;\r\n\r\n /**\r\n * Specifies that the alpha is coming form the albedo channel alpha channel for alpha blending.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesAndMiscDirty\")\r\n public useAlphaFromAlbedoTexture = false;\r\n\r\n /**\r\n * Enforces alpha test in opaque or blend mode in order to improve the performances of some situations.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesAndMiscDirty\")\r\n public forceAlphaTest = false;\r\n\r\n /**\r\n * Defines the alpha limits in alpha test mode.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesAndMiscDirty\")\r\n public alphaCutOff = 0.4;\r\n\r\n /**\r\n * Specifies that the material will keep the specular highlights over a transparent surface (only the most luminous ones).\r\n * A car glass is a good example of that. When sun reflects on it you can not see what is behind.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useSpecularOverAlpha = true;\r\n\r\n /**\r\n * Specifies if the reflectivity texture contains the glossiness information in its alpha channel.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useMicroSurfaceFromReflectivityMapAlpha = false;\r\n\r\n /**\r\n * Specifies if the metallic texture contains the roughness information in its alpha channel.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useRoughnessFromMetallicTextureAlpha = true;\r\n\r\n /**\r\n * Specifies if the metallic texture contains the roughness information in its green channel.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useRoughnessFromMetallicTextureGreen = false;\r\n\r\n /**\r\n * Specifies if the metallic texture contains the metallness information in its blue channel.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useMetallnessFromMetallicTextureBlue = false;\r\n\r\n /**\r\n * Specifies if the metallic texture contains the ambient occlusion information in its red channel.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useAmbientOcclusionFromMetallicTextureRed = false;\r\n\r\n /**\r\n * Specifies if the ambient texture contains the ambient occlusion information in its red channel only.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useAmbientInGrayScale = false;\r\n\r\n /**\r\n * In case the reflectivity map does not contain the microsurface information in its alpha channel,\r\n * The material will try to infer what glossiness each pixel should be.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useAutoMicroSurfaceFromReflectivityMap = false;\r\n\r\n /**\r\n * BJS is using an hardcoded light falloff based on a manually sets up range.\r\n * In PBR, one way to represents the falloff is to use the inverse squared root algorithm.\r\n * This parameter can help you switch back to the BJS mode in order to create scenes using both materials.\r\n */\r\n @serialize()\r\n public get usePhysicalLightFalloff(): boolean {\r\n return this._lightFalloff === PBRBaseMaterial.LIGHTFALLOFF_PHYSICAL;\r\n }\r\n\r\n /**\r\n * BJS is using an hardcoded light falloff based on a manually sets up range.\r\n * In PBR, one way to represents the falloff is to use the inverse squared root algorithm.\r\n * This parameter can help you switch back to the BJS mode in order to create scenes using both materials.\r\n */\r\n public set usePhysicalLightFalloff(value: boolean) {\r\n if (value !== this.usePhysicalLightFalloff) {\r\n // Ensure the effect will be rebuilt.\r\n this._markAllSubMeshesAsTexturesDirty();\r\n\r\n if (value) {\r\n this._lightFalloff = PBRBaseMaterial.LIGHTFALLOFF_PHYSICAL;\r\n } else {\r\n this._lightFalloff = PBRBaseMaterial.LIGHTFALLOFF_STANDARD;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * In order to support the falloff compatibility with gltf, a special mode has been added\r\n * to reproduce the gltf light falloff.\r\n */\r\n @serialize()\r\n public get useGLTFLightFalloff(): boolean {\r\n return this._lightFalloff === PBRBaseMaterial.LIGHTFALLOFF_GLTF;\r\n }\r\n\r\n /**\r\n * In order to support the falloff compatibility with gltf, a special mode has been added\r\n * to reproduce the gltf light falloff.\r\n */\r\n public set useGLTFLightFalloff(value: boolean) {\r\n if (value !== this.useGLTFLightFalloff) {\r\n // Ensure the effect will be rebuilt.\r\n this._markAllSubMeshesAsTexturesDirty();\r\n\r\n if (value) {\r\n this._lightFalloff = PBRBaseMaterial.LIGHTFALLOFF_GLTF;\r\n } else {\r\n this._lightFalloff = PBRBaseMaterial.LIGHTFALLOFF_STANDARD;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Specifies that the material will keeps the reflection highlights over a transparent surface (only the most luminous ones).\r\n * A car glass is a good example of that. When the street lights reflects on it you can not see what is behind.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useRadianceOverAlpha = true;\r\n\r\n /**\r\n * Allows using an object space normal map (instead of tangent space).\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useObjectSpaceNormalMap = false;\r\n\r\n /**\r\n * Allows using the bump map in parallax mode.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useParallax = false;\r\n\r\n /**\r\n * Allows using the bump map in parallax occlusion mode.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useParallaxOcclusion = false;\r\n\r\n /**\r\n * Controls the scale bias of the parallax mode.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public parallaxScaleBias = 0.05;\r\n\r\n /**\r\n * If sets to true, disables all the lights affecting the material.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsLightsDirty\")\r\n public disableLighting = false;\r\n\r\n /**\r\n * Force the shader to compute irradiance in the fragment shader in order to take bump in account.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public forceIrradianceInFragment = false;\r\n\r\n /**\r\n * Number of Simultaneous lights allowed on the material.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsLightsDirty\")\r\n public maxSimultaneousLights = 4;\r\n\r\n /**\r\n * If sets to true, x component of normal map value will invert (x = 1.0 - x).\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public invertNormalMapX = false;\r\n\r\n /**\r\n * If sets to true, y component of normal map value will invert (y = 1.0 - y).\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public invertNormalMapY = false;\r\n\r\n /**\r\n * If sets to true and backfaceCulling is false, normals will be flipped on the backside.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public twoSidedLighting = false;\r\n\r\n /**\r\n * A fresnel is applied to the alpha of the model to ensure grazing angles edges are not alpha tested.\r\n * And/Or occlude the blended part. (alpha is converted to gamma to compute the fresnel)\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useAlphaFresnel = false;\r\n\r\n /**\r\n * A fresnel is applied to the alpha of the model to ensure grazing angles edges are not alpha tested.\r\n * And/Or occlude the blended part. (alpha stays linear to compute the fresnel)\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useLinearAlphaFresnel = false;\r\n\r\n /**\r\n * Let user defines the brdf lookup texture used for IBL.\r\n * A default 8bit version is embedded but you could point at :\r\n * * Default texture: https://assets.babylonjs.com/environments/correlatedMSBRDF_RGBD.png\r\n * * Default 16bit pixel depth texture: https://assets.babylonjs.com/environments/correlatedMSBRDF.dds\r\n * * LEGACY Default None correlated https://assets.babylonjs.com/environments/uncorrelatedBRDF_RGBD.png\r\n * * LEGACY Default None correlated 16bit pixel depth https://assets.babylonjs.com/environments/uncorrelatedBRDF.dds\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public environmentBRDFTexture: Nullable = null;\r\n\r\n /**\r\n * Force normal to face away from face.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public forceNormalForward = false;\r\n\r\n /**\r\n * Enables specular anti aliasing in the PBR shader.\r\n * It will both interacts on the Geometry for analytical and IBL lighting.\r\n * It also prefilter the roughness map based on the bump values.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public enableSpecularAntiAliasing = false;\r\n\r\n /**\r\n * This parameters will enable/disable Horizon occlusion to prevent normal maps to look shiny when the normal\r\n * makes the reflect vector face the model (under horizon).\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useHorizonOcclusion = true;\r\n\r\n /**\r\n * This parameters will enable/disable radiance occlusion by preventing the radiance to lit\r\n * too much the area relying on ambient texture to define their ambient occlusion.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useRadianceOcclusion = true;\r\n\r\n /**\r\n * If set to true, no lighting calculations will be applied.\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsMiscDirty\")\r\n public unlit = false;\r\n\r\n /**\r\n * If sets to true, the decal map will be applied after the detail map. Else, it is applied before (default: false)\r\n */\r\n @serialize()\r\n @expandToProperty(\"_markAllSubMeshesAsMiscDirty\")\r\n public applyDecalMapAfterDetailMap = false;\r\n\r\n /**\r\n * Gets the image processing configuration used either in this material.\r\n */\r\n public get imageProcessingConfiguration(): ImageProcessingConfiguration {\r\n return this._imageProcessingConfiguration;\r\n }\r\n\r\n /**\r\n * Sets the Default image processing configuration used either in the this material.\r\n *\r\n * If sets to null, the scene one is in use.\r\n */\r\n public set imageProcessingConfiguration(value: ImageProcessingConfiguration) {\r\n this._attachImageProcessingConfiguration(value);\r\n\r\n // Ensure the effect will be rebuilt.\r\n this._markAllSubMeshesAsTexturesDirty();\r\n }\r\n\r\n /**\r\n * Gets whether the color curves effect is enabled.\r\n */\r\n public get cameraColorCurvesEnabled(): boolean {\r\n return this.imageProcessingConfiguration.colorCurvesEnabled;\r\n }\r\n /**\r\n * Sets whether the color curves effect is enabled.\r\n */\r\n public set cameraColorCurvesEnabled(value: boolean) {\r\n this.imageProcessingConfiguration.colorCurvesEnabled = value;\r\n }\r\n\r\n /**\r\n * Gets whether the color grading effect is enabled.\r\n */\r\n public get cameraColorGradingEnabled(): boolean {\r\n return this.imageProcessingConfiguration.colorGradingEnabled;\r\n }\r\n /**\r\n * Gets whether the color grading effect is enabled.\r\n */\r\n public set cameraColorGradingEnabled(value: boolean) {\r\n this.imageProcessingConfiguration.colorGradingEnabled = value;\r\n }\r\n\r\n /**\r\n * Gets whether tonemapping is enabled or not.\r\n */\r\n public get cameraToneMappingEnabled(): boolean {\r\n return this._imageProcessingConfiguration.toneMappingEnabled;\r\n }\r\n /**\r\n * Sets whether tonemapping is enabled or not\r\n */\r\n public set cameraToneMappingEnabled(value: boolean) {\r\n this._imageProcessingConfiguration.toneMappingEnabled = value;\r\n }\r\n\r\n /**\r\n * The camera exposure used on this material.\r\n * This property is here and not in the camera to allow controlling exposure without full screen post process.\r\n * This corresponds to a photographic exposure.\r\n */\r\n public get cameraExposure(): number {\r\n return this._imageProcessingConfiguration.exposure;\r\n }\r\n /**\r\n * The camera exposure used on this material.\r\n * This property is here and not in the camera to allow controlling exposure without full screen post process.\r\n * This corresponds to a photographic exposure.\r\n */\r\n public set cameraExposure(value: number) {\r\n this._imageProcessingConfiguration.exposure = value;\r\n }\r\n\r\n /**\r\n * Gets The camera contrast used on this material.\r\n */\r\n public get cameraContrast(): number {\r\n return this._imageProcessingConfiguration.contrast;\r\n }\r\n\r\n /**\r\n * Sets The camera contrast used on this material.\r\n */\r\n public set cameraContrast(value: number) {\r\n this._imageProcessingConfiguration.contrast = value;\r\n }\r\n\r\n /**\r\n * Gets the Color Grading 2D Lookup Texture.\r\n */\r\n public get cameraColorGradingTexture(): Nullable {\r\n return this._imageProcessingConfiguration.colorGradingTexture;\r\n }\r\n /**\r\n * Sets the Color Grading 2D Lookup Texture.\r\n */\r\n public set cameraColorGradingTexture(value: Nullable) {\r\n this._imageProcessingConfiguration.colorGradingTexture = value;\r\n }\r\n\r\n /**\r\n * The color grading curves provide additional color adjustment that is applied after any color grading transform (3D LUT).\r\n * They allow basic adjustment of saturation and small exposure adjustments, along with color filter tinting to provide white balance adjustment or more stylistic effects.\r\n * These are similar to controls found in many professional imaging or colorist software. The global controls are applied to the entire image. For advanced tuning, extra controls are provided to adjust the shadow, midtone and highlight areas of the image;\r\n * corresponding to low luminance, medium luminance, and high luminance areas respectively.\r\n */\r\n public get cameraColorCurves(): Nullable {\r\n return this._imageProcessingConfiguration.colorCurves;\r\n }\r\n /**\r\n * The color grading curves provide additional color adjustment that is applied after any color grading transform (3D LUT).\r\n * They allow basic adjustment of saturation and small exposure adjustments, along with color filter tinting to provide white balance adjustment or more stylistic effects.\r\n * These are similar to controls found in many professional imaging or colorist software. The global controls are applied to the entire image. For advanced tuning, extra controls are provided to adjust the shadow, midtone and highlight areas of the image;\r\n * corresponding to low luminance, medium luminance, and high luminance areas respectively.\r\n */\r\n public set cameraColorCurves(value: Nullable) {\r\n this._imageProcessingConfiguration.colorCurves = value;\r\n }\r\n\r\n /**\r\n * Instantiates a new PBRMaterial instance.\r\n *\r\n * @param name The material name\r\n * @param scene The scene the material will be use in.\r\n */\r\n constructor(name: string, scene?: Scene) {\r\n super(name, scene);\r\n\r\n this._environmentBRDFTexture = GetEnvironmentBRDFTexture(this.getScene());\r\n }\r\n\r\n /**\r\n * Returns the name of this material class.\r\n */\r\n public getClassName(): string {\r\n return \"PBRMaterial\";\r\n }\r\n\r\n /**\r\n * Makes a duplicate of the current material.\r\n * @param name - name to use for the new material.\r\n * @param cloneTexturesOnlyOnce - if a texture is used in more than one channel (e.g diffuse and opacity), only clone it once and reuse it on the other channels. Default false.\r\n * @param rootUrl defines the root URL to use to load textures\r\n */\r\n public clone(name: string, cloneTexturesOnlyOnce: boolean = true, rootUrl = \"\"): PBRMaterial {\r\n const clone = SerializationHelper.Clone(() => new PBRMaterial(name, this.getScene()), this, { cloneTexturesOnlyOnce });\r\n\r\n clone.id = name;\r\n clone.name = name;\r\n\r\n this.stencil.copyTo(clone.stencil);\r\n\r\n this._clonePlugins(clone, rootUrl);\r\n\r\n return clone;\r\n }\r\n\r\n /**\r\n * Serializes this PBR Material.\r\n * @returns - An object with the serialized material.\r\n */\r\n public serialize(): any {\r\n const serializationObject = super.serialize();\r\n serializationObject.customType = \"BABYLON.PBRMaterial\";\r\n\r\n return serializationObject;\r\n }\r\n\r\n // Statics\r\n /**\r\n * Parses a PBR Material from a serialized object.\r\n * @param source - Serialized object.\r\n * @param scene - BJS scene instance.\r\n * @param rootUrl - url for the scene object\r\n * @returns - PBRMaterial\r\n */\r\n public static Parse(source: any, scene: Scene, rootUrl: string): PBRMaterial {\r\n const material = SerializationHelper.Parse(() => new PBRMaterial(source.name, scene), source, scene, rootUrl);\r\n\r\n if (source.stencil) {\r\n material.stencil.parse(source.stencil, scene, rootUrl);\r\n }\r\n\r\n Material._parsePlugins(source, material, scene, rootUrl);\r\n\r\n // The code block below ensures backward compatibility with serialized materials before plugins are automatically serialized.\r\n if (source.clearCoat) {\r\n material.clearCoat.parse(source.clearCoat, scene, rootUrl);\r\n }\r\n if (source.anisotropy) {\r\n material.anisotropy.parse(source.anisotropy, scene, rootUrl);\r\n }\r\n if (source.brdf) {\r\n material.brdf.parse(source.brdf, scene, rootUrl);\r\n }\r\n if (source.sheen) {\r\n material.sheen.parse(source.sheen, scene, rootUrl);\r\n }\r\n if (source.subSurface) {\r\n material.subSurface.parse(source.subSurface, scene, rootUrl);\r\n }\r\n if (source.iridescence) {\r\n material.iridescence.parse(source.iridescence, scene, rootUrl);\r\n }\r\n\r\n return material;\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.PBRMaterial\", PBRMaterial);\r\n","import { serialize, SerializationHelper, serializeAsColor3, expandToProperty } from \"../Misc/decorators\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Vector3 } from \"../Maths/math.vector\";\r\nimport { Color3, TmpColors } from \"../Maths/math.color\";\r\nimport { Node } from \"../node\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { Effect } from \"../Materials/effect\";\r\nimport { UniformBuffer } from \"../Materials/uniformBuffer\";\r\nimport type { IShadowGenerator } from \"./Shadows/shadowGenerator\";\r\nimport { GetClass } from \"../Misc/typeStore\";\r\nimport type { ISortableLight } from \"./lightConstants\";\r\nimport { LightConstants } from \"./lightConstants\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\n\r\n/**\r\n * Base class of all the lights in Babylon. It groups all the generic information about lights.\r\n * Lights are used, as you would expect, to affect how meshes are seen, in terms of both illumination and colour.\r\n * All meshes allow light to pass through them unless shadow generation is activated. The default number of lights allowed is four but this can be increased.\r\n */\r\nexport abstract class Light extends Node implements ISortableLight {\r\n /**\r\n * Falloff Default: light is falling off following the material specification:\r\n * standard material is using standard falloff whereas pbr material can request special falloff per materials.\r\n */\r\n public static readonly FALLOFF_DEFAULT = LightConstants.FALLOFF_DEFAULT;\r\n\r\n /**\r\n * Falloff Physical: light is falling off following the inverse squared distance law.\r\n */\r\n public static readonly FALLOFF_PHYSICAL = LightConstants.FALLOFF_PHYSICAL;\r\n\r\n /**\r\n * Falloff gltf: light is falling off as described in the gltf moving to PBR document\r\n * to enhance interoperability with other engines.\r\n */\r\n public static readonly FALLOFF_GLTF = LightConstants.FALLOFF_GLTF;\r\n\r\n /**\r\n * Falloff Standard: light is falling off like in the standard material\r\n * to enhance interoperability with other materials.\r\n */\r\n public static readonly FALLOFF_STANDARD = LightConstants.FALLOFF_STANDARD;\r\n\r\n //lightmapMode Consts\r\n /**\r\n * If every light affecting the material is in this lightmapMode,\r\n * material.lightmapTexture adds or multiplies\r\n * (depends on material.useLightmapAsShadowmap)\r\n * after every other light calculations.\r\n */\r\n public static readonly LIGHTMAP_DEFAULT = LightConstants.LIGHTMAP_DEFAULT;\r\n /**\r\n * material.lightmapTexture as only diffuse lighting from this light\r\n * adds only specular lighting from this light\r\n * adds dynamic shadows\r\n */\r\n public static readonly LIGHTMAP_SPECULAR = LightConstants.LIGHTMAP_SPECULAR;\r\n /**\r\n * material.lightmapTexture as only lighting\r\n * no light calculation from this light\r\n * only adds dynamic shadows from this light\r\n */\r\n public static readonly LIGHTMAP_SHADOWSONLY = LightConstants.LIGHTMAP_SHADOWSONLY;\r\n\r\n // Intensity Mode Consts\r\n /**\r\n * Each light type uses the default quantity according to its type:\r\n * point/spot lights use luminous intensity\r\n * directional lights use illuminance\r\n */\r\n public static readonly INTENSITYMODE_AUTOMATIC = LightConstants.INTENSITYMODE_AUTOMATIC;\r\n /**\r\n * lumen (lm)\r\n */\r\n public static readonly INTENSITYMODE_LUMINOUSPOWER = LightConstants.INTENSITYMODE_LUMINOUSPOWER;\r\n /**\r\n * candela (lm/sr)\r\n */\r\n public static readonly INTENSITYMODE_LUMINOUSINTENSITY = LightConstants.INTENSITYMODE_LUMINOUSINTENSITY;\r\n /**\r\n * lux (lm/m^2)\r\n */\r\n public static readonly INTENSITYMODE_ILLUMINANCE = LightConstants.INTENSITYMODE_ILLUMINANCE;\r\n /**\r\n * nit (cd/m^2)\r\n */\r\n public static readonly INTENSITYMODE_LUMINANCE = LightConstants.INTENSITYMODE_LUMINANCE;\r\n\r\n // Light types ids const.\r\n /**\r\n * Light type const id of the point light.\r\n */\r\n public static readonly LIGHTTYPEID_POINTLIGHT = LightConstants.LIGHTTYPEID_POINTLIGHT;\r\n /**\r\n * Light type const id of the directional light.\r\n */\r\n public static readonly LIGHTTYPEID_DIRECTIONALLIGHT = LightConstants.LIGHTTYPEID_DIRECTIONALLIGHT;\r\n /**\r\n * Light type const id of the spot light.\r\n */\r\n public static readonly LIGHTTYPEID_SPOTLIGHT = LightConstants.LIGHTTYPEID_SPOTLIGHT;\r\n /**\r\n * Light type const id of the hemispheric light.\r\n */\r\n public static readonly LIGHTTYPEID_HEMISPHERICLIGHT = LightConstants.LIGHTTYPEID_HEMISPHERICLIGHT;\r\n\r\n /**\r\n * Diffuse gives the basic color to an object.\r\n */\r\n @serializeAsColor3()\r\n public diffuse = new Color3(1.0, 1.0, 1.0);\r\n\r\n /**\r\n * Specular produces a highlight color on an object.\r\n * Note: This is not affecting PBR materials.\r\n */\r\n @serializeAsColor3()\r\n public specular = new Color3(1.0, 1.0, 1.0);\r\n\r\n /**\r\n * Defines the falloff type for this light. This lets overriding how punctual light are\r\n * falling off base on range or angle.\r\n * This can be set to any values in Light.FALLOFF_x.\r\n *\r\n * Note: This is only useful for PBR Materials at the moment. This could be extended if required to\r\n * other types of materials.\r\n */\r\n @serialize()\r\n public falloffType = Light.FALLOFF_DEFAULT;\r\n\r\n /**\r\n * Strength of the light.\r\n * Note: By default it is define in the framework own unit.\r\n * Note: In PBR materials the intensityMode can be use to chose what unit the intensity is defined in.\r\n */\r\n @serialize()\r\n public intensity = 1.0;\r\n\r\n private _range = Number.MAX_VALUE;\r\n protected _inverseSquaredRange = 0;\r\n\r\n /**\r\n * Defines how far from the source the light is impacting in scene units.\r\n * Note: Unused in PBR material as the distance light falloff is defined following the inverse squared falloff.\r\n */\r\n @serialize()\r\n public get range(): number {\r\n return this._range;\r\n }\r\n /**\r\n * Defines how far from the source the light is impacting in scene units.\r\n * Note: Unused in PBR material as the distance light falloff is defined following the inverse squared falloff.\r\n */\r\n public set range(value: number) {\r\n this._range = value;\r\n this._inverseSquaredRange = 1.0 / (this.range * this.range);\r\n }\r\n\r\n /**\r\n * Cached photometric scale default to 1.0 as the automatic intensity mode defaults to 1.0 for every type\r\n * of light.\r\n */\r\n private _photometricScale = 1.0;\r\n\r\n private _intensityMode: number = Light.INTENSITYMODE_AUTOMATIC;\r\n /**\r\n * Gets the photometric scale used to interpret the intensity.\r\n * This is only relevant with PBR Materials where the light intensity can be defined in a physical way.\r\n */\r\n @serialize()\r\n public get intensityMode(): number {\r\n return this._intensityMode;\r\n }\r\n /**\r\n * Sets the photometric scale used to interpret the intensity.\r\n * This is only relevant with PBR Materials where the light intensity can be defined in a physical way.\r\n */\r\n public set intensityMode(value: number) {\r\n this._intensityMode = value;\r\n this._computePhotometricScale();\r\n }\r\n\r\n private _radius = 0.00001;\r\n /**\r\n * Gets the light radius used by PBR Materials to simulate soft area lights.\r\n */\r\n @serialize()\r\n public get radius(): number {\r\n return this._radius;\r\n }\r\n /**\r\n * sets the light radius used by PBR Materials to simulate soft area lights.\r\n */\r\n public set radius(value: number) {\r\n this._radius = value;\r\n this._computePhotometricScale();\r\n }\r\n\r\n @serialize()\r\n private _renderPriority: number;\r\n /**\r\n * Defines the rendering priority of the lights. It can help in case of fallback or number of lights\r\n * exceeding the number allowed of the materials.\r\n */\r\n @expandToProperty(\"_reorderLightsInScene\")\r\n public renderPriority: number = 0;\r\n\r\n @serialize(\"shadowEnabled\")\r\n private _shadowEnabled: boolean = true;\r\n /**\r\n * Gets whether or not the shadows are enabled for this light. This can help turning off/on shadow without detaching\r\n * the current shadow generator.\r\n */\r\n public get shadowEnabled(): boolean {\r\n return this._shadowEnabled;\r\n }\r\n /**\r\n * Sets whether or not the shadows are enabled for this light. This can help turning off/on shadow without detaching\r\n * the current shadow generator.\r\n */\r\n public set shadowEnabled(value: boolean) {\r\n if (this._shadowEnabled === value) {\r\n return;\r\n }\r\n\r\n this._shadowEnabled = value;\r\n this._markMeshesAsLightDirty();\r\n }\r\n\r\n private _includedOnlyMeshes: AbstractMesh[];\r\n /**\r\n * Gets the only meshes impacted by this light.\r\n */\r\n public get includedOnlyMeshes(): AbstractMesh[] {\r\n return this._includedOnlyMeshes;\r\n }\r\n /**\r\n * Sets the only meshes impacted by this light.\r\n */\r\n public set includedOnlyMeshes(value: AbstractMesh[]) {\r\n this._includedOnlyMeshes = value;\r\n this._hookArrayForIncludedOnly(value);\r\n }\r\n\r\n private _excludedMeshes: AbstractMesh[];\r\n /**\r\n * Gets the meshes not impacted by this light.\r\n */\r\n public get excludedMeshes(): AbstractMesh[] {\r\n return this._excludedMeshes;\r\n }\r\n /**\r\n * Sets the meshes not impacted by this light.\r\n */\r\n public set excludedMeshes(value: AbstractMesh[]) {\r\n this._excludedMeshes = value;\r\n this._hookArrayForExcluded(value);\r\n }\r\n\r\n @serialize(\"excludeWithLayerMask\")\r\n private _excludeWithLayerMask = 0;\r\n /**\r\n * Gets the layer id use to find what meshes are not impacted by the light.\r\n * Inactive if 0\r\n */\r\n public get excludeWithLayerMask(): number {\r\n return this._excludeWithLayerMask;\r\n }\r\n /**\r\n * Sets the layer id use to find what meshes are not impacted by the light.\r\n * Inactive if 0\r\n */\r\n public set excludeWithLayerMask(value: number) {\r\n this._excludeWithLayerMask = value;\r\n this._resyncMeshes();\r\n }\r\n\r\n @serialize(\"includeOnlyWithLayerMask\")\r\n private _includeOnlyWithLayerMask = 0;\r\n /**\r\n * Gets the layer id use to find what meshes are impacted by the light.\r\n * Inactive if 0\r\n */\r\n public get includeOnlyWithLayerMask(): number {\r\n return this._includeOnlyWithLayerMask;\r\n }\r\n /**\r\n * Sets the layer id use to find what meshes are impacted by the light.\r\n * Inactive if 0\r\n */\r\n public set includeOnlyWithLayerMask(value: number) {\r\n this._includeOnlyWithLayerMask = value;\r\n this._resyncMeshes();\r\n }\r\n\r\n @serialize(\"lightmapMode\")\r\n private _lightmapMode = 0;\r\n /**\r\n * Gets the lightmap mode of this light (should be one of the constants defined by Light.LIGHTMAP_x)\r\n */\r\n public get lightmapMode(): number {\r\n return this._lightmapMode;\r\n }\r\n /**\r\n * Sets the lightmap mode of this light (should be one of the constants defined by Light.LIGHTMAP_x)\r\n */\r\n public set lightmapMode(value: number) {\r\n if (this._lightmapMode === value) {\r\n return;\r\n }\r\n\r\n this._lightmapMode = value;\r\n this._markMeshesAsLightDirty();\r\n }\r\n\r\n /**\r\n * Shadow generators associated to the light.\r\n * @internal Internal use only.\r\n */\r\n public _shadowGenerators: Nullable, IShadowGenerator>> = null;\r\n\r\n /**\r\n * @internal Internal use only.\r\n */\r\n public _excludedMeshesIds = new Array();\r\n\r\n /**\r\n * @internal Internal use only.\r\n */\r\n public _includedOnlyMeshesIds = new Array();\r\n\r\n /**\r\n * The current light uniform buffer.\r\n * @internal Internal use only.\r\n */\r\n public _uniformBuffer: UniformBuffer;\r\n\r\n /** @internal */\r\n public _renderId: number;\r\n\r\n private _lastUseSpecular: boolean;\r\n\r\n /**\r\n * Creates a Light object in the scene.\r\n * Documentation : https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction\r\n * @param name The friendly name of the light\r\n * @param scene The scene the light belongs too\r\n */\r\n constructor(name: string, scene?: Scene) {\r\n super(name, scene);\r\n this.getScene().addLight(this);\r\n this._uniformBuffer = new UniformBuffer(this.getScene().getEngine(), undefined, undefined, name);\r\n this._buildUniformLayout();\r\n\r\n this.includedOnlyMeshes = new Array();\r\n this.excludedMeshes = new Array();\r\n\r\n this._resyncMeshes();\r\n }\r\n\r\n protected abstract _buildUniformLayout(): void;\r\n\r\n /**\r\n * Sets the passed Effect \"effect\" with the Light information.\r\n * @param effect The effect to update\r\n * @param lightIndex The index of the light in the effect to update\r\n * @returns The light\r\n */\r\n public abstract transferToEffect(effect: Effect, lightIndex: string): Light;\r\n\r\n /**\r\n * Sets the passed Effect \"effect\" with the Light textures.\r\n * @param effect The effect to update\r\n * @param lightIndex The index of the light in the effect to update\r\n * @returns The light\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public transferTexturesToEffect(effect: Effect, lightIndex: string): Light {\r\n // Do nothing by default.\r\n return this;\r\n }\r\n\r\n /**\r\n * Binds the lights information from the scene to the effect for the given mesh.\r\n * @param lightIndex Light index\r\n * @param scene The scene where the light belongs to\r\n * @param effect The effect we are binding the data to\r\n * @param useSpecular Defines if specular is supported\r\n * @param receiveShadows Defines if the effect (mesh) we bind the light for receives shadows\r\n */\r\n public _bindLight(lightIndex: number, scene: Scene, effect: Effect, useSpecular: boolean, receiveShadows = true): void {\r\n const iAsString = lightIndex.toString();\r\n let needUpdate = false;\r\n\r\n this._uniformBuffer.bindToEffect(effect, \"Light\" + iAsString);\r\n\r\n if (this._renderId !== scene.getRenderId() || this._lastUseSpecular !== useSpecular || !this._uniformBuffer.useUbo) {\r\n this._renderId = scene.getRenderId();\r\n this._lastUseSpecular = useSpecular;\r\n\r\n const scaledIntensity = this.getScaledIntensity();\r\n\r\n this.transferToEffect(effect, iAsString);\r\n\r\n this.diffuse.scaleToRef(scaledIntensity, TmpColors.Color3[0]);\r\n this._uniformBuffer.updateColor4(\"vLightDiffuse\", TmpColors.Color3[0], this.range, iAsString);\r\n if (useSpecular) {\r\n this.specular.scaleToRef(scaledIntensity, TmpColors.Color3[1]);\r\n this._uniformBuffer.updateColor4(\"vLightSpecular\", TmpColors.Color3[1], this.radius, iAsString);\r\n }\r\n needUpdate = true;\r\n }\r\n\r\n // Textures might still need to be rebound.\r\n this.transferTexturesToEffect(effect, iAsString);\r\n\r\n // Shadows\r\n if (scene.shadowsEnabled && this.shadowEnabled && receiveShadows) {\r\n const shadowGenerator = this.getShadowGenerator(scene.activeCamera) ?? this.getShadowGenerator();\r\n if (shadowGenerator) {\r\n shadowGenerator.bindShadowLight(iAsString, effect);\r\n needUpdate = true;\r\n }\r\n }\r\n\r\n if (needUpdate) {\r\n this._uniformBuffer.update();\r\n } else {\r\n this._uniformBuffer.bindUniformBuffer();\r\n }\r\n }\r\n\r\n /**\r\n * Sets the passed Effect \"effect\" with the Light information.\r\n * @param effect The effect to update\r\n * @param lightDataUniformName The uniform used to store light data (position or direction)\r\n * @returns The light\r\n */\r\n public abstract transferToNodeMaterialEffect(effect: Effect, lightDataUniformName: string): Light;\r\n\r\n /**\r\n * Returns the string \"Light\".\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"Light\";\r\n }\r\n\r\n /** @internal */\r\n public readonly _isLight = true;\r\n\r\n /**\r\n * Converts the light information to a readable string for debug purpose.\r\n * @param fullDetails Supports for multiple levels of logging within scene loading\r\n * @returns the human readable light info\r\n */\r\n public toString(fullDetails?: boolean): string {\r\n let ret = \"Name: \" + this.name;\r\n ret += \", type: \" + [\"Point\", \"Directional\", \"Spot\", \"Hemispheric\"][this.getTypeID()];\r\n if (this.animations) {\r\n for (let i = 0; i < this.animations.length; i++) {\r\n ret += \", animation[0]: \" + this.animations[i].toString(fullDetails);\r\n }\r\n }\r\n return ret;\r\n }\r\n\r\n /** @internal */\r\n protected _syncParentEnabledState() {\r\n super._syncParentEnabledState();\r\n if (!this.isDisposed()) {\r\n this._resyncMeshes();\r\n }\r\n }\r\n\r\n /**\r\n * Set the enabled state of this node.\r\n * @param value - the new enabled state\r\n */\r\n public setEnabled(value: boolean): void {\r\n super.setEnabled(value);\r\n\r\n this._resyncMeshes();\r\n }\r\n\r\n /**\r\n * Returns the Light associated shadow generator if any.\r\n * @param camera Camera for which the shadow generator should be retrieved (default: null). If null, retrieves the default shadow generator\r\n * @returns the associated shadow generator.\r\n */\r\n public getShadowGenerator(camera: Nullable = null): Nullable {\r\n if (this._shadowGenerators === null) {\r\n return null;\r\n }\r\n\r\n return this._shadowGenerators.get(camera) ?? null;\r\n }\r\n\r\n /**\r\n * Returns all the shadow generators associated to this light\r\n * @returns\r\n */\r\n public getShadowGenerators(): Nullable, IShadowGenerator>> {\r\n return this._shadowGenerators;\r\n }\r\n\r\n /**\r\n * Returns a Vector3, the absolute light position in the World.\r\n * @returns the world space position of the light\r\n */\r\n public getAbsolutePosition(): Vector3 {\r\n return Vector3.Zero();\r\n }\r\n\r\n /**\r\n * Specifies if the light will affect the passed mesh.\r\n * @param mesh The mesh to test against the light\r\n * @returns true the mesh is affected otherwise, false.\r\n */\r\n public canAffectMesh(mesh: AbstractMesh): boolean {\r\n if (!mesh) {\r\n return true;\r\n }\r\n\r\n if (this.includedOnlyMeshes && this.includedOnlyMeshes.length > 0 && this.includedOnlyMeshes.indexOf(mesh) === -1) {\r\n return false;\r\n }\r\n\r\n if (this.excludedMeshes && this.excludedMeshes.length > 0 && this.excludedMeshes.indexOf(mesh) !== -1) {\r\n return false;\r\n }\r\n\r\n if (this.includeOnlyWithLayerMask !== 0 && (this.includeOnlyWithLayerMask & mesh.layerMask) === 0) {\r\n return false;\r\n }\r\n\r\n if (this.excludeWithLayerMask !== 0 && this.excludeWithLayerMask & mesh.layerMask) {\r\n return false;\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Releases resources associated with this node.\r\n * @param doNotRecurse Set to true to not recurse into each children (recurse into each children by default)\r\n * @param disposeMaterialAndTextures Set to true to also dispose referenced materials and textures (false by default)\r\n */\r\n public dispose(doNotRecurse?: boolean, disposeMaterialAndTextures = false): void {\r\n if (this._shadowGenerators) {\r\n const iterator = this._shadowGenerators.values();\r\n for (let key = iterator.next(); key.done !== true; key = iterator.next()) {\r\n const shadowGenerator = key.value;\r\n shadowGenerator.dispose();\r\n }\r\n this._shadowGenerators = null;\r\n }\r\n\r\n // Animations\r\n this.getScene().stopAnimation(this);\r\n\r\n if (this._parentContainer) {\r\n const index = this._parentContainer.lights.indexOf(this);\r\n if (index > -1) {\r\n this._parentContainer.lights.splice(index, 1);\r\n }\r\n this._parentContainer = null;\r\n }\r\n\r\n // Remove from meshes\r\n for (const mesh of this.getScene().meshes) {\r\n mesh._removeLightSource(this, true);\r\n }\r\n\r\n this._uniformBuffer.dispose();\r\n\r\n // Remove from scene\r\n this.getScene().removeLight(this);\r\n super.dispose(doNotRecurse, disposeMaterialAndTextures);\r\n }\r\n\r\n /**\r\n * Returns the light type ID (integer).\r\n * @returns The light Type id as a constant defines in Light.LIGHTTYPEID_x\r\n */\r\n public getTypeID(): number {\r\n return 0;\r\n }\r\n\r\n /**\r\n * Returns the intensity scaled by the Photometric Scale according to the light type and intensity mode.\r\n * @returns the scaled intensity in intensity mode unit\r\n */\r\n public getScaledIntensity() {\r\n return this._photometricScale * this.intensity;\r\n }\r\n\r\n /**\r\n * Returns a new Light object, named \"name\", from the current one.\r\n * @param name The name of the cloned light\r\n * @param newParent The parent of this light, if it has one\r\n * @returns the new created light\r\n */\r\n public clone(name: string, newParent: Nullable = null): Nullable {\r\n const constructor = Light.GetConstructorFromName(this.getTypeID(), name, this.getScene());\r\n\r\n if (!constructor) {\r\n return null;\r\n }\r\n const clonedLight = SerializationHelper.Clone(constructor, this);\r\n if (name) {\r\n clonedLight.name = name;\r\n }\r\n if (newParent) {\r\n clonedLight.parent = newParent;\r\n }\r\n clonedLight.setEnabled(this.isEnabled());\r\n\r\n this.onClonedObservable.notifyObservers(clonedLight);\r\n\r\n return clonedLight;\r\n }\r\n\r\n /**\r\n * Serializes the current light into a Serialization object.\r\n * @returns the serialized object.\r\n */\r\n public serialize(): any {\r\n const serializationObject = SerializationHelper.Serialize(this);\r\n serializationObject.uniqueId = this.uniqueId;\r\n\r\n // Type\r\n serializationObject.type = this.getTypeID();\r\n\r\n // Parent\r\n if (this.parent) {\r\n this.parent._serializeAsParent(serializationObject);\r\n }\r\n\r\n // Inclusion / exclusions\r\n if (this.excludedMeshes.length > 0) {\r\n serializationObject.excludedMeshesIds = [];\r\n this.excludedMeshes.forEach((mesh: AbstractMesh) => {\r\n serializationObject.excludedMeshesIds.push(mesh.id);\r\n });\r\n }\r\n\r\n if (this.includedOnlyMeshes.length > 0) {\r\n serializationObject.includedOnlyMeshesIds = [];\r\n this.includedOnlyMeshes.forEach((mesh: AbstractMesh) => {\r\n serializationObject.includedOnlyMeshesIds.push(mesh.id);\r\n });\r\n }\r\n\r\n // Animations\r\n SerializationHelper.AppendSerializedAnimations(this, serializationObject);\r\n serializationObject.ranges = this.serializeAnimationRanges();\r\n\r\n serializationObject.isEnabled = this.isEnabled();\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Creates a new typed light from the passed type (integer) : point light = 0, directional light = 1, spot light = 2, hemispheric light = 3.\r\n * This new light is named \"name\" and added to the passed scene.\r\n * @param type Type according to the types available in Light.LIGHTTYPEID_x\r\n * @param name The friendly name of the light\r\n * @param scene The scene the new light will belong to\r\n * @returns the constructor function\r\n */\r\n static GetConstructorFromName(type: number, name: string, scene: Scene): Nullable<() => Light> {\r\n const constructorFunc = Node.Construct(\"Light_Type_\" + type, name, scene);\r\n\r\n if (constructorFunc) {\r\n return <() => Light>constructorFunc;\r\n }\r\n\r\n // Default to no light for none present once.\r\n return null;\r\n }\r\n\r\n /**\r\n * Parses the passed \"parsedLight\" and returns a new instanced Light from this parsing.\r\n * @param parsedLight The JSON representation of the light\r\n * @param scene The scene to create the parsed light in\r\n * @returns the created light after parsing\r\n */\r\n public static Parse(parsedLight: any, scene: Scene): Nullable {\r\n const constructor = Light.GetConstructorFromName(parsedLight.type, parsedLight.name, scene);\r\n\r\n if (!constructor) {\r\n return null;\r\n }\r\n\r\n const light = SerializationHelper.Parse(constructor, parsedLight, scene);\r\n\r\n // Inclusion / exclusions\r\n if (parsedLight.excludedMeshesIds) {\r\n light._excludedMeshesIds = parsedLight.excludedMeshesIds;\r\n }\r\n\r\n if (parsedLight.includedOnlyMeshesIds) {\r\n light._includedOnlyMeshesIds = parsedLight.includedOnlyMeshesIds;\r\n }\r\n\r\n // Parent\r\n if (parsedLight.parentId !== undefined) {\r\n light._waitingParentId = parsedLight.parentId;\r\n }\r\n\r\n if (parsedLight.parentInstanceIndex !== undefined) {\r\n light._waitingParentInstanceIndex = parsedLight.parentInstanceIndex;\r\n }\r\n\r\n // Falloff\r\n if (parsedLight.falloffType !== undefined) {\r\n light.falloffType = parsedLight.falloffType;\r\n }\r\n\r\n // Lightmaps\r\n if (parsedLight.lightmapMode !== undefined) {\r\n light.lightmapMode = parsedLight.lightmapMode;\r\n }\r\n\r\n // Animations\r\n if (parsedLight.animations) {\r\n for (let animationIndex = 0; animationIndex < parsedLight.animations.length; animationIndex++) {\r\n const parsedAnimation = parsedLight.animations[animationIndex];\r\n const internalClass = GetClass(\"BABYLON.Animation\");\r\n if (internalClass) {\r\n light.animations.push(internalClass.Parse(parsedAnimation));\r\n }\r\n }\r\n Node.ParseAnimationRanges(light, parsedLight, scene);\r\n }\r\n\r\n if (parsedLight.autoAnimate) {\r\n scene.beginAnimation(light, parsedLight.autoAnimateFrom, parsedLight.autoAnimateTo, parsedLight.autoAnimateLoop, parsedLight.autoAnimateSpeed || 1.0);\r\n }\r\n\r\n // Check if isEnabled is defined to be back compatible with prior serialized versions.\r\n if (parsedLight.isEnabled !== undefined) {\r\n light.setEnabled(parsedLight.isEnabled);\r\n }\r\n\r\n return light;\r\n }\r\n\r\n private _hookArrayForExcluded(array: AbstractMesh[]): void {\r\n const oldPush = array.push;\r\n array.push = (...items: AbstractMesh[]) => {\r\n const result = oldPush.apply(array, items);\r\n\r\n for (const item of items) {\r\n item._resyncLightSource(this);\r\n }\r\n\r\n return result;\r\n };\r\n\r\n const oldSplice = array.splice;\r\n array.splice = (index: number, deleteCount?: number) => {\r\n const deleted = oldSplice.apply(array, [index, deleteCount]);\r\n\r\n for (const item of deleted) {\r\n item._resyncLightSource(this);\r\n }\r\n\r\n return deleted;\r\n };\r\n\r\n for (const item of array) {\r\n item._resyncLightSource(this);\r\n }\r\n }\r\n\r\n private _hookArrayForIncludedOnly(array: AbstractMesh[]): void {\r\n const oldPush = array.push;\r\n array.push = (...items: AbstractMesh[]) => {\r\n const result = oldPush.apply(array, items);\r\n\r\n this._resyncMeshes();\r\n\r\n return result;\r\n };\r\n\r\n const oldSplice = array.splice;\r\n array.splice = (index: number, deleteCount?: number) => {\r\n const deleted = oldSplice.apply(array, [index, deleteCount]);\r\n\r\n this._resyncMeshes();\r\n\r\n return deleted;\r\n };\r\n\r\n this._resyncMeshes();\r\n }\r\n\r\n private _resyncMeshes() {\r\n for (const mesh of this.getScene().meshes) {\r\n mesh._resyncLightSource(this);\r\n }\r\n }\r\n\r\n /**\r\n * Forces the meshes to update their light related information in their rendering used effects\r\n * @internal Internal Use Only\r\n */\r\n public _markMeshesAsLightDirty() {\r\n for (const mesh of this.getScene().meshes) {\r\n if (mesh.lightSources.indexOf(this) !== -1) {\r\n mesh._markSubMeshesAsLightDirty();\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Recomputes the cached photometric scale if needed.\r\n */\r\n private _computePhotometricScale(): void {\r\n this._photometricScale = this._getPhotometricScale();\r\n this.getScene().resetCachedMaterial();\r\n }\r\n\r\n /**\r\n * Returns the Photometric Scale according to the light type and intensity mode.\r\n */\r\n private _getPhotometricScale() {\r\n let photometricScale = 0.0;\r\n const lightTypeID = this.getTypeID();\r\n\r\n //get photometric mode\r\n let photometricMode = this.intensityMode;\r\n if (photometricMode === Light.INTENSITYMODE_AUTOMATIC) {\r\n if (lightTypeID === Light.LIGHTTYPEID_DIRECTIONALLIGHT) {\r\n photometricMode = Light.INTENSITYMODE_ILLUMINANCE;\r\n } else {\r\n photometricMode = Light.INTENSITYMODE_LUMINOUSINTENSITY;\r\n }\r\n }\r\n\r\n //compute photometric scale\r\n switch (lightTypeID) {\r\n case Light.LIGHTTYPEID_POINTLIGHT:\r\n case Light.LIGHTTYPEID_SPOTLIGHT:\r\n switch (photometricMode) {\r\n case Light.INTENSITYMODE_LUMINOUSPOWER:\r\n photometricScale = 1.0 / (4.0 * Math.PI);\r\n break;\r\n case Light.INTENSITYMODE_LUMINOUSINTENSITY:\r\n photometricScale = 1.0;\r\n break;\r\n case Light.INTENSITYMODE_LUMINANCE:\r\n photometricScale = this.radius * this.radius;\r\n break;\r\n }\r\n break;\r\n\r\n case Light.LIGHTTYPEID_DIRECTIONALLIGHT:\r\n switch (photometricMode) {\r\n case Light.INTENSITYMODE_ILLUMINANCE:\r\n photometricScale = 1.0;\r\n break;\r\n case Light.INTENSITYMODE_LUMINANCE: {\r\n // When radius (and therefore solid angle) is non-zero a directional lights brightness can be specified via central (peak) luminance.\r\n // For a directional light the 'radius' defines the angular radius (in radians) rather than world-space radius (e.g. in metres).\r\n let apexAngleRadians = this.radius;\r\n // Impose a minimum light angular size to avoid the light becoming an infinitely small angular light source (i.e. a dirac delta function).\r\n apexAngleRadians = Math.max(apexAngleRadians, 0.001);\r\n const solidAngle = 2.0 * Math.PI * (1.0 - Math.cos(apexAngleRadians));\r\n photometricScale = solidAngle;\r\n break;\r\n }\r\n }\r\n break;\r\n\r\n case Light.LIGHTTYPEID_HEMISPHERICLIGHT:\r\n // No fall off in hemispheric light.\r\n photometricScale = 1.0;\r\n break;\r\n }\r\n return photometricScale;\r\n }\r\n\r\n /**\r\n * Reorder the light in the scene according to their defined priority.\r\n * @internal Internal Use Only\r\n */\r\n public _reorderLightsInScene(): void {\r\n const scene = this.getScene();\r\n if (this._renderPriority != 0) {\r\n scene.requireLightSorting = true;\r\n }\r\n this.getScene().sortLightsByPriority();\r\n }\r\n\r\n /**\r\n * Prepares the list of defines specific to the light type.\r\n * @param defines the list of defines\r\n * @param lightIndex defines the index of the light for the effect\r\n */\r\n public abstract prepareLightSpecificDefines(defines: any, lightIndex: number): void;\r\n}\r\n","import { serializeAsColor3, serializeAsVector3 } from \"../Misc/decorators\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Matrix, Vector3 } from \"../Maths/math.vector\";\r\nimport { Color3 } from \"../Maths/math.color\";\r\nimport { Node } from \"../node\";\r\nimport type { Effect } from \"../Materials/effect\";\r\nimport { Light } from \"./light\";\r\nimport type { IShadowGenerator } from \"./Shadows/shadowGenerator\";\r\n\r\nNode.AddNodeConstructor(\"Light_Type_3\", (name, scene) => {\r\n return () => new HemisphericLight(name, Vector3.Zero(), scene);\r\n});\r\n\r\n/**\r\n * The HemisphericLight simulates the ambient environment light,\r\n * so the passed direction is the light reflection direction, not the incoming direction.\r\n */\r\nexport class HemisphericLight extends Light {\r\n /**\r\n * The groundColor is the light in the opposite direction to the one specified during creation.\r\n * You can think of the diffuse and specular light as coming from the centre of the object in the given direction and the groundColor light in the opposite direction.\r\n */\r\n @serializeAsColor3()\r\n public groundColor = new Color3(0.0, 0.0, 0.0);\r\n\r\n /**\r\n * The light reflection direction, not the incoming direction.\r\n */\r\n @serializeAsVector3()\r\n public direction: Vector3;\r\n\r\n /**\r\n * Creates a HemisphericLight object in the scene according to the passed direction (Vector3).\r\n * The HemisphericLight simulates the ambient environment light, so the passed direction is the light reflection direction, not the incoming direction.\r\n * The HemisphericLight can't cast shadows.\r\n * Documentation : https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction\r\n * @param name The friendly name of the light\r\n * @param direction The direction of the light reflection\r\n * @param scene The scene the light belongs to\r\n */\r\n constructor(name: string, direction: Vector3, scene?: Scene) {\r\n super(name, scene);\r\n this.direction = direction || Vector3.Up();\r\n }\r\n\r\n protected _buildUniformLayout(): void {\r\n this._uniformBuffer.addUniform(\"vLightData\", 4);\r\n this._uniformBuffer.addUniform(\"vLightDiffuse\", 4);\r\n this._uniformBuffer.addUniform(\"vLightSpecular\", 4);\r\n this._uniformBuffer.addUniform(\"vLightGround\", 3);\r\n this._uniformBuffer.addUniform(\"shadowsInfo\", 3);\r\n this._uniformBuffer.addUniform(\"depthValues\", 2);\r\n this._uniformBuffer.create();\r\n }\r\n\r\n /**\r\n * Returns the string \"HemisphericLight\".\r\n * @returns The class name\r\n */\r\n public getClassName(): string {\r\n return \"HemisphericLight\";\r\n }\r\n\r\n /**\r\n * Sets the HemisphericLight direction towards the passed target (Vector3).\r\n * Returns the updated direction.\r\n * @param target The target the direction should point to\r\n * @returns The computed direction\r\n */\r\n public setDirectionToTarget(target: Vector3): Vector3 {\r\n this.direction = Vector3.Normalize(target.subtract(Vector3.Zero()));\r\n return this.direction;\r\n }\r\n\r\n /**\r\n * Returns the shadow generator associated to the light.\r\n * @returns Always null for hemispheric lights because it does not support shadows.\r\n */\r\n public getShadowGenerator(): Nullable {\r\n return null;\r\n }\r\n\r\n /**\r\n * Sets the passed Effect object with the HemisphericLight normalized direction and color and the passed name (string).\r\n * @param _effect The effect to update\r\n * @param lightIndex The index of the light in the effect to update\r\n * @returns The hemispheric light\r\n */\r\n public transferToEffect(_effect: Effect, lightIndex: string): HemisphericLight {\r\n const normalizeDirection = Vector3.Normalize(this.direction);\r\n this._uniformBuffer.updateFloat4(\"vLightData\", normalizeDirection.x, normalizeDirection.y, normalizeDirection.z, 0.0, lightIndex);\r\n this._uniformBuffer.updateColor3(\"vLightGround\", this.groundColor.scale(this.intensity), lightIndex);\r\n return this;\r\n }\r\n\r\n public transferToNodeMaterialEffect(effect: Effect, lightDataUniformName: string) {\r\n const normalizeDirection = Vector3.Normalize(this.direction);\r\n effect.setFloat3(lightDataUniformName, normalizeDirection.x, normalizeDirection.y, normalizeDirection.z);\r\n return this;\r\n }\r\n\r\n /**\r\n * Computes the world matrix of the node\r\n * @returns the world matrix\r\n */\r\n public computeWorldMatrix(): Matrix {\r\n if (!this._worldMatrix) {\r\n this._worldMatrix = Matrix.Identity();\r\n }\r\n return this._worldMatrix;\r\n }\r\n\r\n /**\r\n * Returns the integer 3.\r\n * @returns The light Type id as a constant defines in Light.LIGHTTYPEID_x\r\n */\r\n public getTypeID(): number {\r\n return Light.LIGHTTYPEID_HEMISPHERICLIGHT;\r\n }\r\n\r\n /**\r\n * Prepares the list of defines specific to the light type.\r\n * @param defines the list of defines\r\n * @param lightIndex defines the index of the light for the effect\r\n */\r\n public prepareLightSpecificDefines(defines: any, lightIndex: number): void {\r\n defines[\"HEMILIGHT\" + lightIndex] = true;\r\n }\r\n}\r\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"kernelBlurVaryingDeclaration\";\nconst shader = `varying vec2 sampleCoord{X};`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const kernelBlurVaryingDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"packingFunctions\";\nconst shader = `vec4 pack(float depth)\n{const vec4 bit_shift=vec4(255.0*255.0*255.0,255.0*255.0,255.0,1.0);const vec4 bit_mask=vec4(0.0,1.0/255.0,1.0/255.0,1.0/255.0);vec4 res=fract(depth*bit_shift);res-=res.xxyz*bit_mask;return res;}\nfloat unpack(vec4 color)\n{const vec4 bit_shift=vec4(1.0/(255.0*255.0*255.0),1.0/(255.0*255.0),1.0/255.0,1.0);return dot(color,bit_shift);}`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const packingFunctions = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"kernelBlurFragment\";\nconst shader = `#ifdef DOF\nfactor=sampleCoC(sampleCoord{X}); \ncomputedWeight=KERNEL_WEIGHT{X}*factor;sumOfWeights+=computedWeight;\n#else\ncomputedWeight=KERNEL_WEIGHT{X};\n#endif\n#ifdef PACKEDFLOAT\nblend+=unpack(texture2D(textureSampler,sampleCoord{X}))*computedWeight;\n#else\nblend+=texture2D(textureSampler,sampleCoord{X})*computedWeight;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const kernelBlurFragment = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"kernelBlurFragment2\";\nconst shader = `#ifdef DOF\nfactor=sampleCoC(sampleCenter+delta*KERNEL_DEP_OFFSET{X});computedWeight=KERNEL_DEP_WEIGHT{X}*factor;sumOfWeights+=computedWeight;\n#else\ncomputedWeight=KERNEL_DEP_WEIGHT{X};\n#endif\n#ifdef PACKEDFLOAT\nblend+=unpack(texture2D(textureSampler,sampleCenter+delta*KERNEL_DEP_OFFSET{X}))*computedWeight;\n#else\nblend+=texture2D(textureSampler,sampleCenter+delta*KERNEL_DEP_OFFSET{X})*computedWeight;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const kernelBlurFragment2 = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/kernelBlurVaryingDeclaration\";\nimport \"./ShadersInclude/packingFunctions\";\nimport \"./ShadersInclude/kernelBlurFragment\";\nimport \"./ShadersInclude/kernelBlurFragment2\";\n\nconst name = \"kernelBlurPixelShader\";\nconst shader = `uniform sampler2D textureSampler;uniform vec2 delta;varying vec2 sampleCenter;\n#ifdef DOF\nuniform sampler2D circleOfConfusionSampler;float sampleCoC(in vec2 offset) {float coc=texture2D(circleOfConfusionSampler,offset).r;return coc; }\n#endif\n#include[0..varyingCount]\n#ifdef PACKEDFLOAT\n#include\n#endif\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{float computedWeight=0.0;\n#ifdef PACKEDFLOAT\nfloat blend=0.;\n#else\nvec4 blend=vec4(0.);\n#endif\n#ifdef DOF\nfloat sumOfWeights=CENTER_WEIGHT; \nfloat factor=0.0;\n#ifdef PACKEDFLOAT\nblend+=unpack(texture2D(textureSampler,sampleCenter))*CENTER_WEIGHT;\n#else\nblend+=texture2D(textureSampler,sampleCenter)*CENTER_WEIGHT;\n#endif\n#endif\n#include[0..varyingCount]\n#include[0..depCount]\n#ifdef PACKEDFLOAT\ngl_FragColor=pack(blend);\n#else\ngl_FragColor=blend;\n#endif\n#ifdef DOF\ngl_FragColor/=sumOfWeights;\n#endif\n}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const kernelBlurPixelShader = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"kernelBlurVertex\";\nconst shader = `sampleCoord{X}=sampleCenter+delta*KERNEL_OFFSET{X};`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const kernelBlurVertex = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/kernelBlurVaryingDeclaration\";\nimport \"./ShadersInclude/kernelBlurVertex\";\n\nconst name = \"kernelBlurVertexShader\";\nconst shader = `attribute vec2 position;uniform vec2 delta;varying vec2 sampleCenter;\n#include[0..varyingCount]\nconst vec2 madd=vec2(0.5,0.5);\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\nsampleCenter=(position*madd+madd);\n#include[0..varyingCount]\ngl_Position=vec4(position,0.0,1.0);\n#define CUSTOM_VERTEX_MAIN_END\n}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const kernelBlurVertexShader = { name, shader };\n","/* eslint-disable @typescript-eslint/no-unused-vars */\r\nimport type { Vector2 } from \"../Maths/math.vector\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { PostProcessOptions } from \"./postProcess\";\r\nimport { PostProcess } from \"./postProcess\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\nimport type { Effect } from \"../Materials/effect\";\r\nimport { Texture } from \"../Materials/Textures/texture\";\r\nimport type { Engine } from \"../Engines/engine\";\r\nimport { Constants } from \"../Engines/constants\";\r\n\r\nimport \"../Shaders/kernelBlur.fragment\";\r\nimport \"../Shaders/kernelBlur.vertex\";\r\nimport { RegisterClass } from \"../Misc/typeStore\";\r\nimport { serialize, serializeAsVector2, SerializationHelper } from \"../Misc/decorators\";\r\n\r\nimport type { Scene } from \"../scene\";\r\n\r\n/**\r\n * The Blur Post Process which blurs an image based on a kernel and direction.\r\n * Can be used twice in x and y directions to perform a gaussian blur in two passes.\r\n */\r\nexport class BlurPostProcess extends PostProcess {\r\n @serialize(\"kernel\")\r\n protected _kernel: number;\r\n protected _idealKernel: number;\r\n @serialize(\"packedFloat\")\r\n protected _packedFloat: boolean = false;\r\n private _staticDefines: string = \"\";\r\n\r\n /** The direction in which to blur the image. */\r\n @serializeAsVector2()\r\n public direction: Vector2;\r\n\r\n /**\r\n * Sets the length in pixels of the blur sample region\r\n */\r\n public set kernel(v: number) {\r\n if (this._idealKernel === v) {\r\n return;\r\n }\r\n\r\n v = Math.max(v, 1);\r\n this._idealKernel = v;\r\n this._kernel = this._nearestBestKernel(v);\r\n if (!this._blockCompilation) {\r\n this._updateParameters();\r\n }\r\n }\r\n\r\n /**\r\n * Gets the length in pixels of the blur sample region\r\n */\r\n public get kernel(): number {\r\n return this._idealKernel;\r\n }\r\n\r\n /**\r\n * Sets whether or not the blur needs to unpack/repack floats\r\n */\r\n public set packedFloat(v: boolean) {\r\n if (this._packedFloat === v) {\r\n return;\r\n }\r\n this._packedFloat = v;\r\n if (!this._blockCompilation) {\r\n this._updateParameters();\r\n }\r\n }\r\n\r\n /**\r\n * Gets whether or not the blur is unpacking/repacking floats\r\n */\r\n public get packedFloat(): boolean {\r\n return this._packedFloat;\r\n }\r\n\r\n /**\r\n * Gets a string identifying the name of the class\r\n * @returns \"BlurPostProcess\" string\r\n */\r\n public getClassName(): string {\r\n return \"BlurPostProcess\";\r\n }\r\n\r\n /**\r\n * Creates a new instance BlurPostProcess\r\n * @param name The name of the effect.\r\n * @param direction The direction in which to blur the image.\r\n * @param kernel The size of the kernel to be used when computing the blur. eg. Size of 3 will blur the center pixel by 2 pixels surrounding it.\r\n * @param options The required width/height ratio to downsize to before computing the render pass. (Use 1.0 for full size)\r\n * @param camera The camera to apply the render pass to.\r\n * @param samplingMode The sampling mode to be used when computing the pass. (default: 0)\r\n * @param engine The engine which the post process will be applied. (default: current engine)\r\n * @param reusable If the post process can be reused on the same frame. (default: false)\r\n * @param textureType Type of textures used when performing the post process. (default: 0)\r\n * @param defines\r\n * @param _blockCompilation If compilation of the shader should not be done in the constructor. The updateEffect method can be used to compile the shader at a later time. (default: false)\r\n * @param textureFormat Format of textures used when performing the post process. (default: TEXTUREFORMAT_RGBA)\r\n */\r\n constructor(\r\n name: string,\r\n direction: Vector2,\r\n kernel: number,\r\n options: number | PostProcessOptions,\r\n camera: Nullable,\r\n samplingMode: number = Texture.BILINEAR_SAMPLINGMODE,\r\n engine?: Engine,\r\n reusable?: boolean,\r\n textureType = Constants.TEXTURETYPE_UNSIGNED_INT,\r\n defines = \"\",\r\n private _blockCompilation = false,\r\n textureFormat = Constants.TEXTUREFORMAT_RGBA\r\n ) {\r\n super(\r\n name,\r\n \"kernelBlur\",\r\n [\"delta\", \"direction\"],\r\n [\"circleOfConfusionSampler\"],\r\n options,\r\n camera,\r\n samplingMode,\r\n engine,\r\n reusable,\r\n null,\r\n textureType,\r\n \"kernelBlur\",\r\n { varyingCount: 0, depCount: 0 },\r\n true,\r\n textureFormat\r\n );\r\n this._staticDefines = defines;\r\n this.direction = direction;\r\n this.onApplyObservable.add((effect: Effect) => {\r\n if (this._outputTexture) {\r\n effect.setFloat2(\"delta\", (1 / this._outputTexture.width) * this.direction.x, (1 / this._outputTexture.height) * this.direction.y);\r\n } else {\r\n effect.setFloat2(\"delta\", (1 / this.width) * this.direction.x, (1 / this.height) * this.direction.y);\r\n }\r\n });\r\n\r\n this.kernel = kernel;\r\n }\r\n\r\n /**\r\n * Updates the effect with the current post process compile time values and recompiles the shader.\r\n * @param defines Define statements that should be added at the beginning of the shader. (default: null)\r\n * @param uniforms Set of uniform variables that will be passed to the shader. (default: null)\r\n * @param samplers Set of Texture2D variables that will be passed to the shader. (default: null)\r\n * @param indexParameters The index parameters to be used for babylons include syntax \"#include[0..varyingCount]\". (default: undefined) See usage in babylon.blurPostProcess.ts and kernelBlur.vertex.fx\r\n * @param onCompiled Called when the shader has been compiled.\r\n * @param onError Called if there is an error when compiling a shader.\r\n */\r\n public updateEffect(\r\n defines: Nullable = null,\r\n uniforms: Nullable = null,\r\n samplers: Nullable = null,\r\n indexParameters?: any,\r\n onCompiled?: (effect: Effect) => void,\r\n onError?: (effect: Effect, errors: string) => void\r\n ) {\r\n this._updateParameters(onCompiled, onError);\r\n }\r\n\r\n protected _updateParameters(onCompiled?: (effect: Effect) => void, onError?: (effect: Effect, errors: string) => void): void {\r\n // Generate sampling offsets and weights\r\n const N = this._kernel;\r\n const centerIndex = (N - 1) / 2;\r\n\r\n // Generate Gaussian sampling weights over kernel\r\n let offsets = [];\r\n let weights = [];\r\n let totalWeight = 0;\r\n for (let i = 0; i < N; i++) {\r\n const u = i / (N - 1);\r\n const w = this._gaussianWeight(u * 2.0 - 1);\r\n offsets[i] = i - centerIndex;\r\n weights[i] = w;\r\n totalWeight += w;\r\n }\r\n\r\n // Normalize weights\r\n for (let i = 0; i < weights.length; i++) {\r\n weights[i] /= totalWeight;\r\n }\r\n\r\n // Optimize: combine samples to take advantage of hardware linear sampling\r\n // Walk from left to center, combining pairs (symmetrically)\r\n const linearSamplingWeights = [];\r\n const linearSamplingOffsets = [];\r\n\r\n const linearSamplingMap = [];\r\n\r\n for (let i = 0; i <= centerIndex; i += 2) {\r\n const j = Math.min(i + 1, Math.floor(centerIndex));\r\n\r\n const singleCenterSample = i === j;\r\n\r\n if (singleCenterSample) {\r\n linearSamplingMap.push({ o: offsets[i], w: weights[i] });\r\n } else {\r\n const sharedCell = j === centerIndex;\r\n\r\n const weightLinear = weights[i] + weights[j] * (sharedCell ? 0.5 : 1);\r\n const offsetLinear = offsets[i] + 1 / (1 + weights[i] / weights[j]);\r\n\r\n if (offsetLinear === 0) {\r\n linearSamplingMap.push({ o: offsets[i], w: weights[i] });\r\n linearSamplingMap.push({ o: offsets[i + 1], w: weights[i + 1] });\r\n } else {\r\n linearSamplingMap.push({ o: offsetLinear, w: weightLinear });\r\n linearSamplingMap.push({ o: -offsetLinear, w: weightLinear });\r\n }\r\n }\r\n }\r\n\r\n for (let i = 0; i < linearSamplingMap.length; i++) {\r\n linearSamplingOffsets[i] = linearSamplingMap[i].o;\r\n linearSamplingWeights[i] = linearSamplingMap[i].w;\r\n }\r\n\r\n // Replace with optimized\r\n offsets = linearSamplingOffsets;\r\n weights = linearSamplingWeights;\r\n\r\n // Generate shaders\r\n const maxVaryingRows = this.getEngine().getCaps().maxVaryingVectors;\r\n const freeVaryingVec2 = Math.max(maxVaryingRows, 0) - 1; // Because of sampleCenter\r\n\r\n let varyingCount = Math.min(offsets.length, freeVaryingVec2);\r\n\r\n let defines = \"\";\r\n defines += this._staticDefines;\r\n\r\n // The DOF fragment should ignore the center pixel when looping as it is handled manually in the fragment shader.\r\n if (this._staticDefines.indexOf(\"DOF\") != -1) {\r\n defines += `#define CENTER_WEIGHT ${this._glslFloat(weights[varyingCount - 1])}\\n`;\r\n varyingCount--;\r\n }\r\n\r\n for (let i = 0; i < varyingCount; i++) {\r\n defines += `#define KERNEL_OFFSET${i} ${this._glslFloat(offsets[i])}\\n`;\r\n defines += `#define KERNEL_WEIGHT${i} ${this._glslFloat(weights[i])}\\n`;\r\n }\r\n\r\n let depCount = 0;\r\n for (let i = freeVaryingVec2; i < offsets.length; i++) {\r\n defines += `#define KERNEL_DEP_OFFSET${depCount} ${this._glslFloat(offsets[i])}\\n`;\r\n defines += `#define KERNEL_DEP_WEIGHT${depCount} ${this._glslFloat(weights[i])}\\n`;\r\n depCount++;\r\n }\r\n\r\n if (this.packedFloat) {\r\n defines += `#define PACKEDFLOAT 1`;\r\n }\r\n\r\n this._blockCompilation = false;\r\n super.updateEffect(\r\n defines,\r\n null,\r\n null,\r\n {\r\n varyingCount: varyingCount,\r\n depCount: depCount,\r\n },\r\n onCompiled,\r\n onError\r\n );\r\n }\r\n\r\n /**\r\n * Best kernels are odd numbers that when divided by 2, their integer part is even, so 5, 9 or 13.\r\n * Other odd kernels optimize correctly but require proportionally more samples, even kernels are\r\n * possible but will produce minor visual artifacts. Since each new kernel requires a new shader we\r\n * want to minimize kernel changes, having gaps between physical kernels is helpful in that regard.\r\n * The gaps between physical kernels are compensated for in the weighting of the samples\r\n * @param idealKernel Ideal blur kernel.\r\n * @returns Nearest best kernel.\r\n */\r\n protected _nearestBestKernel(idealKernel: number): number {\r\n const v = Math.round(idealKernel);\r\n for (const k of [v, v - 1, v + 1, v - 2, v + 2]) {\r\n if (k % 2 !== 0 && Math.floor(k / 2) % 2 === 0 && k > 0) {\r\n return Math.max(k, 3);\r\n }\r\n }\r\n return Math.max(v, 3);\r\n }\r\n\r\n /**\r\n * Calculates the value of a Gaussian distribution with sigma 3 at a given point.\r\n * @param x The point on the Gaussian distribution to sample.\r\n * @returns the value of the Gaussian function at x.\r\n */\r\n protected _gaussianWeight(x: number): number {\r\n //reference: Engines/ImageProcessingBlur.cpp #dcc760\r\n // We are evaluating the Gaussian (normal) distribution over a kernel parameter space of [-1,1],\r\n // so we truncate at three standard deviations by setting stddev (sigma) to 1/3.\r\n // The choice of 3-sigma truncation is common but arbitrary, and means that the signal is\r\n // truncated at around 1.3% of peak strength.\r\n\r\n //the distribution is scaled to account for the difference between the actual kernel size and the requested kernel size\r\n const sigma = 1 / 3;\r\n const denominator = Math.sqrt(2.0 * Math.PI) * sigma;\r\n const exponent = -((x * x) / (2.0 * sigma * sigma));\r\n const weight = (1.0 / denominator) * Math.exp(exponent);\r\n return weight;\r\n }\r\n\r\n /**\r\n * Generates a string that can be used as a floating point number in GLSL.\r\n * @param x Value to print.\r\n * @param decimalFigures Number of decimal places to print the number to (excluding trailing 0s).\r\n * @returns GLSL float string.\r\n */\r\n protected _glslFloat(x: number, decimalFigures = 8) {\r\n return x.toFixed(decimalFigures).replace(/0+$/, \"\");\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _Parse(parsedPostProcess: any, targetCamera: Camera, scene: Scene, rootUrl: string): Nullable {\r\n return SerializationHelper.Parse(\r\n () => {\r\n return new BlurPostProcess(\r\n parsedPostProcess.name,\r\n parsedPostProcess.direction,\r\n parsedPostProcess.kernel,\r\n parsedPostProcess.options,\r\n targetCamera,\r\n parsedPostProcess.renderTargetSamplingMode,\r\n scene.getEngine(),\r\n parsedPostProcess.reusable,\r\n parsedPostProcess.textureType,\r\n undefined,\r\n false\r\n );\r\n },\r\n parsedPostProcess,\r\n scene,\r\n rootUrl\r\n );\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.BlurPostProcess\", BlurPostProcess);\r\n","import type { Observer } from \"../../Misc/observable\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { Camera } from \"../../Cameras/camera\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { Matrix, Vector3, Vector2 } from \"../../Maths/math.vector\";\r\nimport { Texture } from \"../../Materials/Textures/texture\";\r\nimport { RenderTargetTexture } from \"../../Materials/Textures/renderTargetTexture\";\r\nimport type { ImageProcessingConfiguration } from \"../../Materials/imageProcessingConfiguration\";\r\nimport { BlurPostProcess } from \"../../PostProcesses/blurPostProcess\";\r\nimport { Constants } from \"../../Engines/constants\";\r\nimport { Plane } from \"../../Maths/math.plane\";\r\nimport type { UniformBuffer } from \"../uniformBuffer\";\r\n/**\r\n * Mirror texture can be used to simulate the view from a mirror in a scene.\r\n * It will dynamically be rendered every frame to adapt to the camera point of view.\r\n * You can then easily use it as a reflectionTexture on a flat surface.\r\n * In case the surface is not a plane, please consider relying on reflection probes.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#mirrortexture\r\n */\r\nexport class MirrorTexture extends RenderTargetTexture {\r\n /**\r\n * Define the reflection plane we want to use. The mirrorPlane is usually set to the constructed reflector.\r\n * It is possible to directly set the mirrorPlane by directly using a Plane(a, b, c, d) where a, b and c give the plane normal vector (a, b, c) and d is a scalar displacement from the mirrorPlane to the origin. However in all but the very simplest of situations it is more straight forward to set it to the reflector as stated in the doc.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#mirrors\r\n */\r\n public mirrorPlane = new Plane(0, 1, 0, 1);\r\n\r\n /**\r\n * Define the blur ratio used to blur the reflection if needed.\r\n */\r\n public set blurRatio(value: number) {\r\n if (this._blurRatio === value) {\r\n return;\r\n }\r\n\r\n this._blurRatio = value;\r\n this._preparePostProcesses();\r\n }\r\n\r\n public get blurRatio(): number {\r\n return this._blurRatio;\r\n }\r\n\r\n /**\r\n * Define the adaptive blur kernel used to blur the reflection if needed.\r\n * This will autocompute the closest best match for the `blurKernel`\r\n */\r\n public set adaptiveBlurKernel(value: number) {\r\n this._adaptiveBlurKernel = value;\r\n this._autoComputeBlurKernel();\r\n }\r\n\r\n /**\r\n * Define the blur kernel used to blur the reflection if needed.\r\n * Please consider using `adaptiveBlurKernel` as it could find the closest best value for you.\r\n */\r\n public set blurKernel(value: number) {\r\n this.blurKernelX = value;\r\n this.blurKernelY = value;\r\n }\r\n\r\n /**\r\n * Define the blur kernel on the X Axis used to blur the reflection if needed.\r\n * Please consider using `adaptiveBlurKernel` as it could find the closest best value for you.\r\n */\r\n public set blurKernelX(value: number) {\r\n if (this._blurKernelX === value) {\r\n return;\r\n }\r\n\r\n this._blurKernelX = value;\r\n this._preparePostProcesses();\r\n }\r\n\r\n public get blurKernelX(): number {\r\n return this._blurKernelX;\r\n }\r\n\r\n /**\r\n * Define the blur kernel on the Y Axis used to blur the reflection if needed.\r\n * Please consider using `adaptiveBlurKernel` as it could find the closest best value for you.\r\n */\r\n public set blurKernelY(value: number) {\r\n if (this._blurKernelY === value) {\r\n return;\r\n }\r\n\r\n this._blurKernelY = value;\r\n this._preparePostProcesses();\r\n }\r\n\r\n public get blurKernelY(): number {\r\n return this._blurKernelY;\r\n }\r\n\r\n private _autoComputeBlurKernel(): void {\r\n const engine = this.getScene()!.getEngine();\r\n\r\n const dw = this.getRenderWidth() / engine.getRenderWidth();\r\n const dh = this.getRenderHeight() / engine.getRenderHeight();\r\n this.blurKernelX = this._adaptiveBlurKernel * dw;\r\n this.blurKernelY = this._adaptiveBlurKernel * dh;\r\n }\r\n\r\n protected _onRatioRescale(): void {\r\n if (this._sizeRatio) {\r\n this.resize(this._initialSizeParameter);\r\n if (!this._adaptiveBlurKernel) {\r\n this._preparePostProcesses();\r\n }\r\n }\r\n\r\n if (this._adaptiveBlurKernel) {\r\n this._autoComputeBlurKernel();\r\n }\r\n }\r\n\r\n private _updateGammaSpace() {\r\n const scene = this.getScene();\r\n\r\n if (!scene) {\r\n return;\r\n }\r\n\r\n this.gammaSpace = !scene.imageProcessingConfiguration.isEnabled || !scene.imageProcessingConfiguration.applyByPostProcess;\r\n }\r\n\r\n private _imageProcessingConfigChangeObserver: Nullable>;\r\n\r\n private _transformMatrix = Matrix.Zero();\r\n private _mirrorMatrix = Matrix.Zero();\r\n\r\n private _blurX: Nullable;\r\n private _blurY: Nullable;\r\n private _adaptiveBlurKernel = 0;\r\n private _blurKernelX = 0;\r\n private _blurKernelY = 0;\r\n private _blurRatio = 1.0;\r\n private _sceneUBO: UniformBuffer;\r\n private _currentSceneUBO: UniformBuffer;\r\n\r\n /**\r\n * Instantiates a Mirror Texture.\r\n * Mirror texture can be used to simulate the view from a mirror in a scene.\r\n * It will dynamically be rendered every frame to adapt to the camera point of view.\r\n * You can then easily use it as a reflectionTexture on a flat surface.\r\n * In case the surface is not a plane, please consider relying on reflection probes.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#mirrors\r\n * @param name\r\n * @param size\r\n * @param scene\r\n * @param generateMipMaps\r\n * @param type\r\n * @param samplingMode\r\n * @param generateDepthBuffer\r\n */\r\n constructor(\r\n name: string,\r\n size: number | { width: number; height: number } | { ratio: number },\r\n scene?: Scene,\r\n generateMipMaps?: boolean,\r\n type: number = Constants.TEXTURETYPE_UNSIGNED_INT,\r\n samplingMode = Texture.BILINEAR_SAMPLINGMODE,\r\n generateDepthBuffer = true\r\n ) {\r\n super(name, size, scene, generateMipMaps, true, type, false, samplingMode, generateDepthBuffer);\r\n\r\n scene = this.getScene();\r\n\r\n if (!scene) {\r\n return this;\r\n }\r\n this.ignoreCameraViewport = true;\r\n\r\n this._updateGammaSpace();\r\n this._imageProcessingConfigChangeObserver = scene.imageProcessingConfiguration.onUpdateParameters.add(() => {\r\n this._updateGammaSpace();\r\n });\r\n\r\n const engine = scene.getEngine();\r\n\r\n if (engine.supportsUniformBuffers) {\r\n this._sceneUBO = scene.createSceneUniformBuffer(`Scene for Mirror Texture (name \"${name}\")`);\r\n }\r\n\r\n this.onBeforeBindObservable.add(() => {\r\n engine._debugPushGroup?.(`mirror generation for ${name}`, 1);\r\n });\r\n\r\n this.onAfterUnbindObservable.add(() => {\r\n engine._debugPopGroup?.(1);\r\n });\r\n\r\n let saveClipPlane: Nullable;\r\n\r\n this.onBeforeRenderObservable.add(() => {\r\n if (this._sceneUBO) {\r\n this._currentSceneUBO = scene!.getSceneUniformBuffer();\r\n scene!.setSceneUniformBuffer(this._sceneUBO);\r\n scene!.getSceneUniformBuffer().unbindEffect();\r\n }\r\n\r\n Matrix.ReflectionToRef(this.mirrorPlane, this._mirrorMatrix);\r\n this._mirrorMatrix.multiplyToRef(scene!.getViewMatrix(), this._transformMatrix);\r\n\r\n scene!.setTransformMatrix(this._transformMatrix, scene!.getProjectionMatrix());\r\n\r\n saveClipPlane = scene!.clipPlane;\r\n scene!.clipPlane = this.mirrorPlane;\r\n\r\n scene!._mirroredCameraPosition = Vector3.TransformCoordinates((scene!.activeCamera).globalPosition, this._mirrorMatrix);\r\n });\r\n\r\n this.onAfterRenderObservable.add(() => {\r\n if (this._sceneUBO) {\r\n scene!.setSceneUniformBuffer(this._currentSceneUBO);\r\n }\r\n scene!.updateTransformMatrix();\r\n scene!._mirroredCameraPosition = null;\r\n\r\n scene!.clipPlane = saveClipPlane;\r\n });\r\n }\r\n\r\n private _preparePostProcesses(): void {\r\n this.clearPostProcesses(true);\r\n\r\n if (this._blurKernelX && this._blurKernelY) {\r\n const engine = (this.getScene()).getEngine();\r\n\r\n const textureType =\r\n engine.getCaps().textureFloatRender && engine.getCaps().textureFloatLinearFiltering ? Constants.TEXTURETYPE_FLOAT : Constants.TEXTURETYPE_HALF_FLOAT;\r\n\r\n this._blurX = new BlurPostProcess(\r\n \"horizontal blur\",\r\n new Vector2(1.0, 0),\r\n this._blurKernelX,\r\n this._blurRatio,\r\n null,\r\n Texture.BILINEAR_SAMPLINGMODE,\r\n engine,\r\n false,\r\n textureType\r\n );\r\n this._blurX.autoClear = false;\r\n\r\n if (this._blurRatio === 1 && this.samples < 2 && this._texture) {\r\n this._blurX.inputTexture = this._renderTarget!;\r\n } else {\r\n this._blurX.alwaysForcePOT = true;\r\n }\r\n\r\n this._blurY = new BlurPostProcess(\r\n \"vertical blur\",\r\n new Vector2(0, 1.0),\r\n this._blurKernelY,\r\n this._blurRatio,\r\n null,\r\n Texture.BILINEAR_SAMPLINGMODE,\r\n engine,\r\n false,\r\n textureType\r\n );\r\n this._blurY.autoClear = false;\r\n this._blurY.alwaysForcePOT = this._blurRatio !== 1;\r\n\r\n this.addPostProcess(this._blurX);\r\n this.addPostProcess(this._blurY);\r\n } else {\r\n if (this._blurY) {\r\n this.removePostProcess(this._blurY);\r\n this._blurY.dispose();\r\n this._blurY = null;\r\n }\r\n if (this._blurX) {\r\n this.removePostProcess(this._blurX);\r\n this._blurX.dispose();\r\n this._blurX = null;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Clone the mirror texture.\r\n * @returns the cloned texture\r\n */\r\n public clone(): MirrorTexture {\r\n const scene = this.getScene();\r\n\r\n if (!scene) {\r\n return this;\r\n }\r\n\r\n const textureSize = this.getSize();\r\n const newTexture = new MirrorTexture(\r\n this.name,\r\n textureSize.width,\r\n scene,\r\n this._renderTargetOptions.generateMipMaps,\r\n this._renderTargetOptions.type,\r\n this._renderTargetOptions.samplingMode,\r\n this._renderTargetOptions.generateDepthBuffer\r\n );\r\n\r\n // Base texture\r\n newTexture.hasAlpha = this.hasAlpha;\r\n newTexture.level = this.level;\r\n\r\n // Mirror Texture\r\n newTexture.mirrorPlane = this.mirrorPlane.clone();\r\n if (this.renderList) {\r\n newTexture.renderList = this.renderList.slice(0);\r\n }\r\n\r\n return newTexture;\r\n }\r\n\r\n /**\r\n * Serialize the texture to a JSON representation you could use in Parse later on\r\n * @returns the serialized JSON representation\r\n */\r\n public serialize(): any {\r\n if (!this.name) {\r\n return null;\r\n }\r\n\r\n const serializationObject = super.serialize();\r\n\r\n serializationObject.mirrorPlane = this.mirrorPlane.asArray();\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Dispose the texture and release its associated resources.\r\n */\r\n public dispose() {\r\n super.dispose();\r\n const scene = this.getScene();\r\n\r\n if (scene) {\r\n scene.imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingConfigChangeObserver);\r\n }\r\n this._sceneUBO?.dispose();\r\n }\r\n}\r\n\r\nTexture._CreateMirror = (name: string, renderTargetSize: number, scene: Scene, generateMipMaps: boolean): MirrorTexture => {\r\n return new MirrorTexture(name, renderTargetSize, scene, generateMipMaps);\r\n};\r\n","import { ThinEngine } from \"../../Engines/thinEngine\";\r\nimport { InternalTexture, InternalTextureSource } from \"../../Materials/Textures/internalTexture\";\r\nimport { Logger } from \"../../Misc/logger\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { IInternalTextureLoader } from \"../../Materials/Textures/internalTextureLoader\";\r\nimport { LoadImage } from \"../../Misc/fileTools\";\r\nimport { RandomGUID } from \"../../Misc/guid\";\r\nimport type { IWebRequest } from \"../../Misc/interfaces/iWebRequest\";\r\nimport { Constants } from \"../constants\";\r\nimport type { DepthTextureCreationOptions } from \"../../Materials/Textures/textureCreationOptions\";\r\nimport type { RenderTargetWrapper } from \"../renderTargetWrapper\";\r\n\r\ndeclare module \"../../Engines/thinEngine\" {\r\n export interface ThinEngine {\r\n /**\r\n * Creates a depth stencil cube texture.\r\n * This is only available in WebGL 2.\r\n * @param size The size of face edge in the cube texture.\r\n * @param options The options defining the cube texture.\r\n * @param rtWrapper The render target wrapper for which the depth/stencil texture must be created\r\n * @returns The cube texture\r\n */\r\n _createDepthStencilCubeTexture(size: number, options: DepthTextureCreationOptions, rtWrapper: RenderTargetWrapper): InternalTexture;\r\n\r\n /**\r\n * Creates a cube texture\r\n * @param rootUrl defines the url where the files to load is located\r\n * @param scene defines the current scene\r\n * @param files defines the list of files to load (1 per face)\r\n * @param noMipmap defines a boolean indicating that no mipmaps shall be generated (false by default)\r\n * @param onLoad defines an optional callback raised when the texture is loaded\r\n * @param onError defines an optional callback raised if there is an issue to load the texture\r\n * @param format defines the format of the data\r\n * @param forcedExtension defines the extension to use to pick the right loader\r\n * @param createPolynomials if a polynomial sphere should be created for the cube texture\r\n * @param lodScale defines the scale applied to environment texture. This manages the range of LOD level used for IBL according to the roughness\r\n * @param lodOffset defines the offset applied to environment texture. This manages first LOD level used for IBL according to the roughness\r\n * @param fallback defines texture to use while falling back when (compressed) texture file not found.\r\n * @param loaderOptions options to be passed to the loader\r\n * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU).\r\n * @returns the cube texture as an InternalTexture\r\n */\r\n createCubeTexture(\r\n rootUrl: string,\r\n scene: Nullable,\r\n files: Nullable,\r\n noMipmap: boolean | undefined,\r\n onLoad: Nullable<(data?: any) => void>,\r\n onError: Nullable<(message?: string, exception?: any) => void>,\r\n format: number | undefined,\r\n forcedExtension: any,\r\n createPolynomials: boolean,\r\n lodScale: number,\r\n lodOffset: number,\r\n fallback: Nullable,\r\n loaderOptions: any,\r\n useSRGBBuffer: boolean\r\n ): InternalTexture;\r\n\r\n /**\r\n * Creates a cube texture\r\n * @param rootUrl defines the url where the files to load is located\r\n * @param scene defines the current scene\r\n * @param files defines the list of files to load (1 per face)\r\n * @param noMipmap defines a boolean indicating that no mipmaps shall be generated (false by default)\r\n * @param onLoad defines an optional callback raised when the texture is loaded\r\n * @param onError defines an optional callback raised if there is an issue to load the texture\r\n * @param format defines the format of the data\r\n * @param forcedExtension defines the extension to use to pick the right loader\r\n * @returns the cube texture as an InternalTexture\r\n */\r\n createCubeTexture(\r\n rootUrl: string,\r\n scene: Nullable,\r\n files: Nullable,\r\n noMipmap: boolean,\r\n onLoad: Nullable<(data?: any) => void>,\r\n onError: Nullable<(message?: string, exception?: any) => void>,\r\n format: number | undefined,\r\n forcedExtension: any\r\n ): InternalTexture;\r\n\r\n /**\r\n * Creates a cube texture\r\n * @param rootUrl defines the url where the files to load is located\r\n * @param scene defines the current scene\r\n * @param files defines the list of files to load (1 per face)\r\n * @param noMipmap defines a boolean indicating that no mipmaps shall be generated (false by default)\r\n * @param onLoad defines an optional callback raised when the texture is loaded\r\n * @param onError defines an optional callback raised if there is an issue to load the texture\r\n * @param format defines the format of the data\r\n * @param forcedExtension defines the extension to use to pick the right loader\r\n * @param createPolynomials if a polynomial sphere should be created for the cube texture\r\n * @param lodScale defines the scale applied to environment texture. This manages the range of LOD level used for IBL according to the roughness\r\n * @param lodOffset defines the offset applied to environment texture. This manages first LOD level used for IBL according to the roughness\r\n * @returns the cube texture as an InternalTexture\r\n */\r\n createCubeTexture(\r\n rootUrl: string,\r\n scene: Nullable,\r\n files: Nullable,\r\n noMipmap: boolean,\r\n onLoad: Nullable<(data?: any) => void>,\r\n onError: Nullable<(message?: string, exception?: any) => void>,\r\n format: number | undefined,\r\n forcedExtension: any,\r\n createPolynomials: boolean,\r\n lodScale: number,\r\n lodOffset: number\r\n ): InternalTexture;\r\n\r\n /** @internal */\r\n createCubeTextureBase(\r\n rootUrl: string,\r\n scene: Nullable,\r\n files: Nullable,\r\n noMipmap: boolean,\r\n onLoad: Nullable<(data?: any) => void>,\r\n onError: Nullable<(message?: string, exception?: any) => void>,\r\n format: number | undefined,\r\n forcedExtension: any,\r\n createPolynomials: boolean,\r\n lodScale: number,\r\n lodOffset: number,\r\n fallback: Nullable,\r\n beforeLoadCubeDataCallback: Nullable<(texture: InternalTexture, data: ArrayBufferView | ArrayBufferView[]) => void>,\r\n imageHandler: Nullable<(texture: InternalTexture, imgs: HTMLImageElement[] | ImageBitmap[]) => void>,\r\n useSRGBBuffer: boolean\r\n ): InternalTexture;\r\n\r\n /** @internal */\r\n _partialLoadFile(\r\n url: string,\r\n index: number,\r\n loadedFiles: ArrayBuffer[],\r\n onfinish: (files: ArrayBuffer[]) => void,\r\n onErrorCallBack: Nullable<(message?: string, exception?: any) => void>\r\n ): void;\r\n\r\n /** @internal */\r\n _cascadeLoadFiles(scene: Nullable, onfinish: (images: ArrayBuffer[]) => void, files: string[], onError: Nullable<(message?: string, exception?: any) => void>): void;\r\n\r\n /** @internal */\r\n _cascadeLoadImgs(\r\n scene: Nullable,\r\n texture: InternalTexture,\r\n onfinish: Nullable<(texture: InternalTexture, images: HTMLImageElement[] | ImageBitmap[]) => void>,\r\n files: string[],\r\n onError: Nullable<(message?: string, exception?: any) => void>,\r\n mimeType?: string\r\n ): void;\r\n\r\n /** @internal */\r\n _partialLoadImg(\r\n url: string,\r\n index: number,\r\n loadedImages: HTMLImageElement[] | ImageBitmap[],\r\n scene: Nullable,\r\n texture: InternalTexture,\r\n onfinish: Nullable<(texture: InternalTexture, images: HTMLImageElement[] | ImageBitmap[]) => void>,\r\n onErrorCallBack: Nullable<(message?: string, exception?: any) => void>,\r\n mimeType?: string\r\n ): void;\r\n\r\n /**\r\n * @internal\r\n */\r\n _setCubeMapTextureParams(texture: InternalTexture, loadMipmap: boolean, maxLevel?: number): void;\r\n }\r\n}\r\n\r\nThinEngine.prototype._createDepthStencilCubeTexture = function (size: number, options: DepthTextureCreationOptions, rtWrapper: RenderTargetWrapper): InternalTexture {\r\n const internalTexture = new InternalTexture(this, InternalTextureSource.DepthStencil);\r\n internalTexture.isCube = true;\r\n\r\n if (this.webGLVersion === 1) {\r\n Logger.Error(\"Depth cube texture is not supported by WebGL 1.\");\r\n return internalTexture;\r\n }\r\n\r\n const internalOptions = {\r\n bilinearFiltering: false,\r\n comparisonFunction: 0,\r\n generateStencil: false,\r\n ...options,\r\n };\r\n\r\n const gl = this._gl;\r\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, internalTexture, true);\r\n\r\n this._setupDepthStencilTexture(internalTexture, size, internalOptions.generateStencil, internalOptions.bilinearFiltering, internalOptions.comparisonFunction);\r\n\r\n rtWrapper._depthStencilTexture = internalTexture;\r\n rtWrapper._depthStencilTextureWithStencil = internalOptions.generateStencil;\r\n\r\n // Create the depth/stencil buffer\r\n for (let face = 0; face < 6; face++) {\r\n if (internalOptions.generateStencil) {\r\n gl.texImage2D(gl.TEXTURE_CUBE_MAP_POSITIVE_X + face, 0, gl.DEPTH24_STENCIL8, size, size, 0, gl.DEPTH_STENCIL, gl.UNSIGNED_INT_24_8, null);\r\n } else {\r\n gl.texImage2D(gl.TEXTURE_CUBE_MAP_POSITIVE_X + face, 0, gl.DEPTH_COMPONENT24, size, size, 0, gl.DEPTH_COMPONENT, gl.UNSIGNED_INT, null);\r\n }\r\n }\r\n\r\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, null);\r\n\r\n this._internalTexturesCache.push(internalTexture);\r\n\r\n return internalTexture;\r\n};\r\n\r\nThinEngine.prototype._partialLoadFile = function (\r\n url: string,\r\n index: number,\r\n loadedFiles: ArrayBuffer[],\r\n onfinish: (files: ArrayBuffer[]) => void,\r\n onErrorCallBack: Nullable<(message?: string, exception?: any) => void> = null\r\n): void {\r\n const onload = (data: ArrayBuffer) => {\r\n loadedFiles[index] = data;\r\n (loadedFiles)._internalCount++;\r\n\r\n if ((loadedFiles)._internalCount === 6) {\r\n onfinish(loadedFiles);\r\n }\r\n };\r\n\r\n const onerror = (request?: IWebRequest, exception?: any) => {\r\n if (onErrorCallBack && request) {\r\n onErrorCallBack(request.status + \" \" + request.statusText, exception);\r\n }\r\n };\r\n\r\n this._loadFile(url, onload as (data: string | ArrayBuffer) => void, undefined, undefined, true, onerror);\r\n};\r\n\r\nThinEngine.prototype._cascadeLoadFiles = function (\r\n scene: Nullable,\r\n onfinish: (images: ArrayBuffer[]) => void,\r\n files: string[],\r\n onError: Nullable<(message?: string, exception?: any) => void> = null\r\n): void {\r\n const loadedFiles: ArrayBuffer[] = [];\r\n (loadedFiles)._internalCount = 0;\r\n\r\n for (let index = 0; index < 6; index++) {\r\n this._partialLoadFile(files[index], index, loadedFiles, onfinish, onError);\r\n }\r\n};\r\n\r\nThinEngine.prototype._cascadeLoadImgs = function (\r\n scene: Nullable,\r\n texture: InternalTexture,\r\n onfinish: Nullable<(texture: InternalTexture, images: HTMLImageElement[] | ImageBitmap[]) => void>,\r\n files: string[],\r\n onError: Nullable<(message?: string, exception?: any) => void> = null,\r\n mimeType?: string\r\n) {\r\n const loadedImages: HTMLImageElement[] | ImageBitmap[] = [];\r\n (loadedImages)._internalCount = 0;\r\n\r\n for (let index = 0; index < 6; index++) {\r\n this._partialLoadImg(files[index], index, loadedImages, scene, texture, onfinish, onError, mimeType);\r\n }\r\n};\r\n\r\nThinEngine.prototype._partialLoadImg = function (\r\n url: string,\r\n index: number,\r\n loadedImages: HTMLImageElement[] | ImageBitmap[],\r\n scene: Nullable,\r\n texture: InternalTexture,\r\n onfinish: Nullable<(texture: InternalTexture, images: HTMLImageElement[] | ImageBitmap[]) => void>,\r\n onErrorCallBack: Nullable<(message?: string, exception?: any) => void> = null,\r\n mimeType?: string\r\n) {\r\n const tokenPendingData = RandomGUID();\r\n\r\n const onload = (img: HTMLImageElement | ImageBitmap) => {\r\n loadedImages[index] = img;\r\n (loadedImages)._internalCount++;\r\n\r\n if (scene) {\r\n scene.removePendingData(tokenPendingData);\r\n }\r\n\r\n if ((loadedImages)._internalCount === 6 && onfinish) {\r\n onfinish(texture, loadedImages);\r\n }\r\n };\r\n\r\n const onerror = (message?: string, exception?: any) => {\r\n if (scene) {\r\n scene.removePendingData(tokenPendingData);\r\n }\r\n\r\n if (onErrorCallBack) {\r\n onErrorCallBack(message, exception);\r\n }\r\n };\r\n\r\n LoadImage(url, onload, onerror, scene ? scene.offlineProvider : null, mimeType);\r\n if (scene) {\r\n scene.addPendingData(tokenPendingData);\r\n }\r\n};\r\n\r\nThinEngine.prototype._setCubeMapTextureParams = function (texture: InternalTexture, loadMipmap: boolean, maxLevel?: number): void {\r\n const gl = this._gl;\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_MAG_FILTER, gl.LINEAR);\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_MIN_FILTER, loadMipmap ? gl.LINEAR_MIPMAP_LINEAR : gl.LINEAR);\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\r\n texture.samplingMode = loadMipmap ? Constants.TEXTURE_TRILINEAR_SAMPLINGMODE : Constants.TEXTURE_LINEAR_LINEAR;\r\n\r\n if (loadMipmap && this.getCaps().textureMaxLevel && maxLevel !== undefined && maxLevel > 0) {\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_MAX_LEVEL, maxLevel);\r\n texture._maxLodLevel = maxLevel;\r\n }\r\n\r\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, null);\r\n};\r\n\r\nThinEngine.prototype.createCubeTextureBase = function (\r\n rootUrl: string,\r\n scene: Nullable,\r\n files: Nullable,\r\n noMipmap?: boolean,\r\n onLoad: Nullable<(data?: any) => void> = null,\r\n onError: Nullable<(message?: string, exception?: any) => void> = null,\r\n format?: number,\r\n forcedExtension: any = null,\r\n createPolynomials: boolean = false,\r\n lodScale: number = 0,\r\n lodOffset: number = 0,\r\n fallback: Nullable = null,\r\n beforeLoadCubeDataCallback: Nullable<(texture: InternalTexture, data: ArrayBufferView | ArrayBufferView[]) => void> = null,\r\n imageHandler: Nullable<(texture: InternalTexture, imgs: HTMLImageElement[] | ImageBitmap[]) => void> = null,\r\n useSRGBBuffer = false\r\n): InternalTexture {\r\n const texture = fallback ? fallback : new InternalTexture(this, InternalTextureSource.Cube);\r\n texture.isCube = true;\r\n texture.url = rootUrl;\r\n texture.generateMipMaps = !noMipmap;\r\n texture._lodGenerationScale = lodScale;\r\n texture._lodGenerationOffset = lodOffset;\r\n texture._useSRGBBuffer = !!useSRGBBuffer && this._caps.supportSRGBBuffers && (this.webGLVersion > 1 || this.isWebGPU || !!noMipmap);\r\n if (texture !== fallback) {\r\n texture.label = rootUrl.substring(0, 60); // default label, can be overriden by the caller\r\n }\r\n\r\n if (!this._doNotHandleContextLost) {\r\n texture._extension = forcedExtension;\r\n texture._files = files;\r\n }\r\n\r\n const originalRootUrl = rootUrl;\r\n if (this._transformTextureUrl && !fallback) {\r\n rootUrl = this._transformTextureUrl(rootUrl);\r\n }\r\n\r\n const rootUrlWithoutUriParams = rootUrl.split(\"?\")[0];\r\n const lastDot = rootUrlWithoutUriParams.lastIndexOf(\".\");\r\n const extension = forcedExtension ? forcedExtension : lastDot > -1 ? rootUrlWithoutUriParams.substring(lastDot).toLowerCase() : \"\";\r\n\r\n let loader: Nullable = null;\r\n for (const availableLoader of ThinEngine._TextureLoaders) {\r\n if (availableLoader.canLoad(extension)) {\r\n loader = availableLoader;\r\n break;\r\n }\r\n }\r\n\r\n const onInternalError = (request?: IWebRequest, exception?: any) => {\r\n if (rootUrl === originalRootUrl) {\r\n if (onError && request) {\r\n onError(request.status + \" \" + request.statusText, exception);\r\n }\r\n } else {\r\n // fall back to the original url if the transformed url fails to load\r\n Logger.Warn(`Failed to load ${rootUrl}, falling back to the ${originalRootUrl}`);\r\n this.createCubeTextureBase(\r\n originalRootUrl,\r\n scene,\r\n files,\r\n !!noMipmap,\r\n onLoad,\r\n onError,\r\n format,\r\n forcedExtension,\r\n createPolynomials,\r\n lodScale,\r\n lodOffset,\r\n texture,\r\n beforeLoadCubeDataCallback,\r\n imageHandler,\r\n useSRGBBuffer\r\n );\r\n }\r\n };\r\n\r\n if (loader) {\r\n const onloaddata = (data: ArrayBufferView | ArrayBufferView[]) => {\r\n if (beforeLoadCubeDataCallback) {\r\n beforeLoadCubeDataCallback(texture, data);\r\n }\r\n loader!.loadCubeData(data, texture, createPolynomials, onLoad, onError);\r\n };\r\n if (files && files.length === 6) {\r\n if (loader.supportCascades) {\r\n this._cascadeLoadFiles(scene, (images) => onloaddata(images.map((image) => new Uint8Array(image))), files, onError);\r\n } else {\r\n if (onError) {\r\n onError(\"Textures type does not support cascades.\");\r\n } else {\r\n Logger.Warn(\"Texture loader does not support cascades.\");\r\n }\r\n }\r\n } else {\r\n this._loadFile(rootUrl, (data) => onloaddata(new Uint8Array(data as ArrayBuffer)), undefined, undefined, true, onInternalError);\r\n }\r\n } else {\r\n if (!files) {\r\n throw new Error(\"Cannot load cubemap because files were not defined\");\r\n }\r\n\r\n this._cascadeLoadImgs(\r\n scene,\r\n texture,\r\n (texture: InternalTexture, imgs: HTMLImageElement[] | ImageBitmap[]) => {\r\n if (imageHandler) {\r\n imageHandler(texture, imgs);\r\n }\r\n },\r\n files,\r\n onError\r\n );\r\n }\r\n\r\n this._internalTexturesCache.push(texture);\r\n\r\n return texture;\r\n};\r\n\r\nThinEngine.prototype.createCubeTexture = function (\r\n rootUrl: string,\r\n scene: Nullable,\r\n files: Nullable,\r\n noMipmap?: boolean,\r\n onLoad: Nullable<(data?: any) => void> = null,\r\n onError: Nullable<(message?: string, exception?: any) => void> = null,\r\n format?: number,\r\n forcedExtension: any = null,\r\n createPolynomials: boolean = false,\r\n lodScale: number = 0,\r\n lodOffset: number = 0,\r\n fallback: Nullable = null,\r\n loaderOptions?: any,\r\n useSRGBBuffer = false\r\n): InternalTexture {\r\n const gl = this._gl;\r\n\r\n return this.createCubeTextureBase(\r\n rootUrl,\r\n scene,\r\n files,\r\n !!noMipmap,\r\n onLoad,\r\n onError,\r\n format,\r\n forcedExtension,\r\n createPolynomials,\r\n lodScale,\r\n lodOffset,\r\n fallback,\r\n (texture: InternalTexture) => this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, texture, true),\r\n (texture: InternalTexture, imgs: HTMLImageElement[] | ImageBitmap[]) => {\r\n const width = this.needPOTTextures ? ThinEngine.GetExponentOfTwo(imgs[0].width, this._caps.maxCubemapTextureSize) : imgs[0].width;\r\n const height = width;\r\n\r\n const faces = [\r\n gl.TEXTURE_CUBE_MAP_POSITIVE_X,\r\n gl.TEXTURE_CUBE_MAP_POSITIVE_Y,\r\n gl.TEXTURE_CUBE_MAP_POSITIVE_Z,\r\n gl.TEXTURE_CUBE_MAP_NEGATIVE_X,\r\n gl.TEXTURE_CUBE_MAP_NEGATIVE_Y,\r\n gl.TEXTURE_CUBE_MAP_NEGATIVE_Z,\r\n ];\r\n\r\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, texture, true);\r\n this._unpackFlipY(false);\r\n\r\n const internalFormat = format ? this._getInternalFormat(format, texture._useSRGBBuffer) : texture._useSRGBBuffer ? this._glSRGBExtensionValues.SRGB8_ALPHA8 : gl.RGBA;\r\n let texelFormat = format ? this._getInternalFormat(format) : gl.RGBA;\r\n\r\n if (texture._useSRGBBuffer && this.webGLVersion === 1) {\r\n texelFormat = internalFormat;\r\n }\r\n\r\n for (let index = 0; index < faces.length; index++) {\r\n if (imgs[index].width !== width || imgs[index].height !== height) {\r\n this._prepareWorkingCanvas();\r\n\r\n if (!this._workingCanvas || !this._workingContext) {\r\n Logger.Warn(\"Cannot create canvas to resize texture.\");\r\n return;\r\n }\r\n this._workingCanvas.width = width;\r\n this._workingCanvas.height = height;\r\n\r\n this._workingContext.drawImage(imgs[index], 0, 0, imgs[index].width, imgs[index].height, 0, 0, width, height);\r\n gl.texImage2D(faces[index], 0, internalFormat, texelFormat, gl.UNSIGNED_BYTE, this._workingCanvas as TexImageSource);\r\n } else {\r\n gl.texImage2D(faces[index], 0, internalFormat, texelFormat, gl.UNSIGNED_BYTE, imgs[index]);\r\n }\r\n }\r\n\r\n if (!noMipmap) {\r\n gl.generateMipmap(gl.TEXTURE_CUBE_MAP);\r\n }\r\n\r\n this._setCubeMapTextureParams(texture, !noMipmap);\r\n\r\n texture.width = width;\r\n texture.height = height;\r\n texture.isReady = true;\r\n if (format) {\r\n texture.format = format;\r\n }\r\n\r\n texture.onLoadedObservable.notifyObservers(texture);\r\n texture.onLoadedObservable.clear();\r\n\r\n if (onLoad) {\r\n onLoad();\r\n }\r\n },\r\n !!useSRGBBuffer\r\n );\r\n};\r\n","import { serialize, serializeAsMatrix, SerializationHelper, serializeAsVector3 } from \"../../Misc/decorators\";\r\nimport { Tools } from \"../../Misc/tools\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { Matrix, TmpVectors, Vector3 } from \"../../Maths/math.vector\";\r\nimport { BaseTexture } from \"../../Materials/Textures/baseTexture\";\r\nimport { Texture } from \"../../Materials/Textures/texture\";\r\nimport { Constants } from \"../../Engines/constants\";\r\nimport { GetClass, RegisterClass } from \"../../Misc/typeStore\";\r\nimport type { ThinEngine } from \"../../Engines/thinEngine\";\r\n\r\nimport \"../../Engines/Extensions/engine.cubeTexture\";\r\nimport { Observable } from \"../../Misc/observable\";\r\n\r\n/**\r\n * Class for creating a cube texture\r\n */\r\nexport class CubeTexture extends BaseTexture {\r\n private _delayedOnLoad: Nullable<() => void>;\r\n private _delayedOnError: Nullable<(message?: string, exception?: any) => void>;\r\n private _lodScale: number = 0.8;\r\n private _lodOffset: number = 0;\r\n\r\n /**\r\n * Observable triggered once the texture has been loaded.\r\n */\r\n public onLoadObservable: Observable = new Observable();\r\n\r\n /**\r\n * The url of the texture\r\n */\r\n @serialize()\r\n public url: string;\r\n\r\n /**\r\n * Gets or sets the center of the bounding box associated with the cube texture.\r\n * It must define where the camera used to render the texture was set\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#using-local-cubemap-mode\r\n */\r\n @serializeAsVector3()\r\n public boundingBoxPosition = Vector3.Zero();\r\n\r\n private _boundingBoxSize: Vector3;\r\n\r\n /**\r\n * Gets or sets the size of the bounding box associated with the cube texture\r\n * When defined, the cubemap will switch to local mode\r\n * @see https://community.arm.com/graphics/b/blog/posts/reflections-based-on-local-cubemaps-in-unity\r\n * @example https://www.babylonjs-playground.com/#RNASML\r\n */\r\n public set boundingBoxSize(value: Vector3) {\r\n if (this._boundingBoxSize && this._boundingBoxSize.equals(value)) {\r\n return;\r\n }\r\n this._boundingBoxSize = value;\r\n const scene = this.getScene();\r\n if (scene) {\r\n scene.markAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n }\r\n /**\r\n * Returns the bounding box size\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#using-local-cubemap-mode\r\n */\r\n @serializeAsVector3()\r\n public get boundingBoxSize(): Vector3 {\r\n return this._boundingBoxSize;\r\n }\r\n\r\n protected _rotationY: number = 0;\r\n\r\n /**\r\n * Sets texture matrix rotation angle around Y axis in radians.\r\n */\r\n @serialize(\"rotationY\")\r\n public set rotationY(value: number) {\r\n this._rotationY = value;\r\n this.setReflectionTextureMatrix(Matrix.RotationY(this._rotationY));\r\n }\r\n /**\r\n * Gets texture matrix rotation angle around Y axis radians.\r\n */\r\n public get rotationY(): number {\r\n return this._rotationY;\r\n }\r\n\r\n /**\r\n * Are mip maps generated for this texture or not.\r\n */\r\n public get noMipmap(): boolean {\r\n return this._noMipmap;\r\n }\r\n\r\n private _noMipmap: boolean;\r\n\r\n /** @internal */\r\n @serialize(\"files\")\r\n public _files: Nullable = null;\r\n\r\n @serialize(\"forcedExtension\")\r\n protected _forcedExtension: Nullable = null;\r\n\r\n /**\r\n * Gets the forced extension (if any)\r\n */\r\n public get forcedExtension(): Nullable {\r\n return this._forcedExtension;\r\n }\r\n\r\n @serialize(\"extensions\")\r\n private _extensions: Nullable = null;\r\n\r\n @serializeAsMatrix(\"textureMatrix\")\r\n private _textureMatrix: Matrix;\r\n\r\n @serializeAsMatrix(\"textureMatrixRefraction\")\r\n private _textureMatrixRefraction: Matrix = new Matrix();\r\n\r\n private _format: number;\r\n private _createPolynomials: boolean;\r\n private _loaderOptions: any;\r\n private _useSRGBBuffer?: boolean;\r\n\r\n /**\r\n * Creates a cube texture from an array of image urls\r\n * @param files defines an array of image urls\r\n * @param scene defines the hosting scene\r\n * @param noMipmap specifies if mip maps are not used\r\n * @returns a cube texture\r\n */\r\n public static CreateFromImages(files: string[], scene: Scene, noMipmap?: boolean): CubeTexture {\r\n let rootUrlKey = \"\";\r\n\r\n files.forEach((url) => (rootUrlKey += url));\r\n\r\n return new CubeTexture(rootUrlKey, scene, null, noMipmap, files);\r\n }\r\n\r\n /**\r\n * Creates and return a texture created from prefilterd data by tools like IBL Baker or Lys.\r\n * @param url defines the url of the prefiltered texture\r\n * @param scene defines the scene the texture is attached to\r\n * @param forcedExtension defines the extension of the file if different from the url\r\n * @param createPolynomials defines whether or not to create polynomial harmonics from the texture data if necessary\r\n * @returns the prefiltered texture\r\n */\r\n public static CreateFromPrefilteredData(url: string, scene: Scene, forcedExtension: any = null, createPolynomials: boolean = true) {\r\n const oldValue = scene.useDelayedTextureLoading;\r\n scene.useDelayedTextureLoading = false;\r\n\r\n const result = new CubeTexture(url, scene, null, false, null, null, null, undefined, true, forcedExtension, createPolynomials);\r\n\r\n scene.useDelayedTextureLoading = oldValue;\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a cube texture to use with reflection for instance. It can be based upon dds or six images as well\r\n * as prefiltered data.\r\n * @param rootUrl defines the url of the texture or the root name of the six images\r\n * @param sceneOrEngine defines the scene or engine the texture is attached to\r\n * @param extensions defines the suffixes add to the picture name in case six images are in use like _px.jpg...\r\n * @param noMipmap defines if mipmaps should be created or not\r\n * @param files defines the six files to load for the different faces in that order: px, py, pz, nx, ny, nz\r\n * @param onLoad defines a callback triggered at the end of the file load if no errors occurred\r\n * @param onError defines a callback triggered in case of error during load\r\n * @param format defines the internal format to use for the texture once loaded\r\n * @param prefiltered defines whether or not the texture is created from prefiltered data\r\n * @param forcedExtension defines the extensions to use (force a special type of file to load) in case it is different from the file name\r\n * @param createPolynomials defines whether or not to create polynomial harmonics from the texture data if necessary\r\n * @param lodScale defines the scale applied to environment texture. This manages the range of LOD level used for IBL according to the roughness\r\n * @param lodOffset defines the offset applied to environment texture. This manages first LOD level used for IBL according to the roughness\r\n * @param loaderOptions options to be passed to the loader\r\n * @param useSRGBBuffer Defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU) (default: false)\r\n * @returns the cube texture\r\n */\r\n constructor(\r\n rootUrl: string,\r\n sceneOrEngine: Scene | ThinEngine,\r\n extensions: Nullable = null,\r\n noMipmap: boolean = false,\r\n files: Nullable = null,\r\n onLoad: Nullable<() => void> = null,\r\n onError: Nullable<(message?: string, exception?: any) => void> = null,\r\n format: number = Constants.TEXTUREFORMAT_RGBA,\r\n prefiltered = false,\r\n forcedExtension: any = null,\r\n createPolynomials: boolean = false,\r\n lodScale: number = 0.8,\r\n lodOffset: number = 0,\r\n loaderOptions?: any,\r\n useSRGBBuffer?: boolean\r\n ) {\r\n super(sceneOrEngine);\r\n\r\n this.name = rootUrl;\r\n this.url = rootUrl;\r\n this._noMipmap = noMipmap;\r\n this.hasAlpha = false;\r\n this._format = format;\r\n this.isCube = true;\r\n this._textureMatrix = Matrix.Identity();\r\n this._createPolynomials = createPolynomials;\r\n this.coordinatesMode = Texture.CUBIC_MODE;\r\n this._extensions = extensions;\r\n this._files = files;\r\n this._forcedExtension = forcedExtension;\r\n this._loaderOptions = loaderOptions;\r\n this._useSRGBBuffer = useSRGBBuffer;\r\n this._lodScale = lodScale;\r\n this._lodOffset = lodOffset;\r\n\r\n if (!rootUrl && !files) {\r\n return;\r\n }\r\n\r\n this.updateURL(rootUrl, forcedExtension, onLoad, prefiltered, onError, extensions, this.getScene()?.useDelayedTextureLoading, files);\r\n }\r\n\r\n /**\r\n * Get the current class name of the texture useful for serialization or dynamic coding.\r\n * @returns \"CubeTexture\"\r\n */\r\n public getClassName(): string {\r\n return \"CubeTexture\";\r\n }\r\n\r\n /**\r\n * Update the url (and optional buffer) of this texture if url was null during construction.\r\n * @param url the url of the texture\r\n * @param forcedExtension defines the extension to use\r\n * @param onLoad callback called when the texture is loaded (defaults to null)\r\n * @param prefiltered Defines whether the updated texture is prefiltered or not\r\n * @param onError callback called if there was an error during the loading process (defaults to null)\r\n * @param extensions defines the suffixes add to the picture name in case six images are in use like _px.jpg...\r\n * @param delayLoad defines if the texture should be loaded now (false by default)\r\n * @param files defines the six files to load for the different faces in that order: px, py, pz, nx, ny, nz\r\n */\r\n public updateURL(\r\n url: string,\r\n forcedExtension?: string,\r\n onLoad: Nullable<() => void> = null,\r\n prefiltered: boolean = false,\r\n onError: Nullable<(message?: string, exception?: any) => void> = null,\r\n extensions: Nullable = null,\r\n delayLoad = false,\r\n files: Nullable = null\r\n ): void {\r\n if (!this.name || this.name.startsWith(\"data:\")) {\r\n this.name = url;\r\n }\r\n this.url = url;\r\n\r\n if (forcedExtension) {\r\n this._forcedExtension = forcedExtension;\r\n }\r\n\r\n const lastDot = url.lastIndexOf(\".\");\r\n const extension = forcedExtension ? forcedExtension : lastDot > -1 ? url.substring(lastDot).toLowerCase() : \"\";\r\n const isDDS = extension.indexOf(\".dds\") === 0;\r\n const isEnv = extension.indexOf(\".env\") === 0;\r\n const isBasis = extension.indexOf(\".basis\") === 0;\r\n\r\n if (isEnv) {\r\n this.gammaSpace = false;\r\n this._prefiltered = false;\r\n this.anisotropicFilteringLevel = 1;\r\n } else {\r\n this._prefiltered = prefiltered;\r\n\r\n if (prefiltered) {\r\n this.gammaSpace = false;\r\n this.anisotropicFilteringLevel = 1;\r\n }\r\n }\r\n\r\n if (files) {\r\n this._files = files;\r\n } else {\r\n if (!isBasis && !isEnv && !isDDS && !extensions) {\r\n extensions = [\"_px.jpg\", \"_py.jpg\", \"_pz.jpg\", \"_nx.jpg\", \"_ny.jpg\", \"_nz.jpg\"];\r\n }\r\n\r\n this._files = this._files || [];\r\n this._files.length = 0;\r\n\r\n if (extensions) {\r\n for (let index = 0; index < extensions.length; index++) {\r\n this._files.push(url + extensions[index]);\r\n }\r\n this._extensions = extensions;\r\n }\r\n }\r\n\r\n if (delayLoad) {\r\n this.delayLoadState = Constants.DELAYLOADSTATE_NOTLOADED;\r\n this._delayedOnLoad = onLoad;\r\n this._delayedOnError = onError;\r\n } else {\r\n this._loadTexture(onLoad, onError);\r\n }\r\n }\r\n\r\n /**\r\n * Delays loading of the cube texture\r\n * @param forcedExtension defines the extension to use\r\n */\r\n public delayLoad(forcedExtension?: string): void {\r\n if (this.delayLoadState !== Constants.DELAYLOADSTATE_NOTLOADED) {\r\n return;\r\n }\r\n if (forcedExtension) {\r\n this._forcedExtension = forcedExtension;\r\n }\r\n\r\n this.delayLoadState = Constants.DELAYLOADSTATE_LOADED;\r\n this._loadTexture(this._delayedOnLoad, this._delayedOnError);\r\n }\r\n\r\n /**\r\n * Returns the reflection texture matrix\r\n * @returns the reflection texture matrix\r\n */\r\n public getReflectionTextureMatrix(): Matrix {\r\n return this._textureMatrix;\r\n }\r\n\r\n /**\r\n * Sets the reflection texture matrix\r\n * @param value Reflection texture matrix\r\n */\r\n public setReflectionTextureMatrix(value: Matrix): void {\r\n if (value.updateFlag === this._textureMatrix.updateFlag) {\r\n return;\r\n }\r\n\r\n if (value.isIdentity() !== this._textureMatrix.isIdentity()) {\r\n this.getScene()?.markAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag, (mat) => mat.getActiveTextures().indexOf(this) !== -1);\r\n }\r\n\r\n this._textureMatrix = value;\r\n\r\n if (!this.getScene()?.useRightHandedSystem) {\r\n return;\r\n }\r\n\r\n const scale = TmpVectors.Vector3[0];\r\n const quat = TmpVectors.Quaternion[0];\r\n const trans = TmpVectors.Vector3[1];\r\n\r\n this._textureMatrix.decompose(scale, quat, trans);\r\n\r\n quat.z *= -1; // these two operations correspond to negating the x and y euler angles\r\n quat.w *= -1;\r\n\r\n Matrix.ComposeToRef(scale, quat, trans, this._textureMatrixRefraction);\r\n }\r\n\r\n /**\r\n * Gets a suitable rotate/transform matrix when the texture is used for refraction.\r\n * There's a separate function from getReflectionTextureMatrix because refraction requires a special configuration of the matrix in right-handed mode.\r\n * @returns The refraction matrix\r\n */\r\n public getRefractionTextureMatrix(): Matrix {\r\n return this.getScene()?.useRightHandedSystem ? this._textureMatrixRefraction : this._textureMatrix;\r\n }\r\n\r\n private _loadTexture(onLoad: Nullable<() => void> = null, onError: Nullable<(message?: string, exception?: any) => void> = null) {\r\n const scene = this.getScene();\r\n const oldTexture = this._texture;\r\n this._texture = this._getFromCache(this.url, this._noMipmap, undefined, undefined, this._useSRGBBuffer, this.isCube);\r\n\r\n const onLoadProcessing = () => {\r\n this.onLoadObservable.notifyObservers(this);\r\n if (oldTexture) {\r\n oldTexture.dispose();\r\n this.getScene()?.markAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n if (onLoad) {\r\n onLoad();\r\n }\r\n };\r\n\r\n const errorHandler = (message?: string, exception?: any) => {\r\n this._loadingError = true;\r\n this._errorObject = { message, exception };\r\n if (onError) {\r\n onError(message, exception);\r\n }\r\n Texture.OnTextureLoadErrorObservable.notifyObservers(this);\r\n };\r\n\r\n if (!this._texture) {\r\n if (this._prefiltered) {\r\n this._texture = this._getEngine()!.createPrefilteredCubeTexture(\r\n this.url,\r\n scene,\r\n this._lodScale,\r\n this._lodOffset,\r\n onLoad,\r\n errorHandler,\r\n this._format,\r\n this._forcedExtension,\r\n this._createPolynomials\r\n );\r\n } else {\r\n this._texture = this._getEngine()!.createCubeTexture(\r\n this.url,\r\n scene,\r\n this._files,\r\n this._noMipmap,\r\n onLoad,\r\n errorHandler,\r\n this._format,\r\n this._forcedExtension,\r\n false,\r\n this._lodScale,\r\n this._lodOffset,\r\n null,\r\n this._loaderOptions,\r\n !!this._useSRGBBuffer\r\n );\r\n }\r\n\r\n this._texture?.onLoadedObservable.add(() => this.onLoadObservable.notifyObservers(this));\r\n } else {\r\n if (this._texture.isReady) {\r\n Tools.SetImmediate(() => onLoadProcessing());\r\n } else {\r\n this._texture.onLoadedObservable.add(() => onLoadProcessing());\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Parses text to create a cube texture\r\n * @param parsedTexture define the serialized text to read from\r\n * @param scene defines the hosting scene\r\n * @param rootUrl defines the root url of the cube texture\r\n * @returns a cube texture\r\n */\r\n public static Parse(parsedTexture: any, scene: Scene, rootUrl: string): CubeTexture {\r\n const texture = SerializationHelper.Parse(\r\n () => {\r\n let prefiltered: boolean = false;\r\n if (parsedTexture.prefiltered) {\r\n prefiltered = parsedTexture.prefiltered;\r\n }\r\n return new CubeTexture(\r\n rootUrl + (parsedTexture.url ?? parsedTexture.name),\r\n scene,\r\n parsedTexture.extensions,\r\n false,\r\n parsedTexture.files || null,\r\n null,\r\n null,\r\n undefined,\r\n prefiltered,\r\n parsedTexture.forcedExtension\r\n );\r\n },\r\n parsedTexture,\r\n scene\r\n );\r\n\r\n // Local Cubemaps\r\n if (parsedTexture.boundingBoxPosition) {\r\n texture.boundingBoxPosition = Vector3.FromArray(parsedTexture.boundingBoxPosition);\r\n }\r\n if (parsedTexture.boundingBoxSize) {\r\n texture.boundingBoxSize = Vector3.FromArray(parsedTexture.boundingBoxSize);\r\n }\r\n\r\n // Animations\r\n if (parsedTexture.animations) {\r\n for (let animationIndex = 0; animationIndex < parsedTexture.animations.length; animationIndex++) {\r\n const parsedAnimation = parsedTexture.animations[animationIndex];\r\n const internalClass = GetClass(\"BABYLON.Animation\");\r\n if (internalClass) {\r\n texture.animations.push(internalClass.Parse(parsedAnimation));\r\n }\r\n }\r\n }\r\n\r\n return texture;\r\n }\r\n\r\n /**\r\n * Makes a clone, or deep copy, of the cube texture\r\n * @returns a new cube texture\r\n */\r\n public clone(): CubeTexture {\r\n let uniqueId = 0;\r\n\r\n const newCubeTexture = SerializationHelper.Clone(() => {\r\n const cubeTexture = new CubeTexture(this.url, this.getScene() || this._getEngine()!, this._extensions, this._noMipmap, this._files);\r\n uniqueId = cubeTexture.uniqueId;\r\n\r\n return cubeTexture;\r\n }, this);\r\n\r\n newCubeTexture.uniqueId = uniqueId;\r\n\r\n return newCubeTexture;\r\n }\r\n}\r\n\r\nTexture._CubeTextureParser = CubeTexture.Parse;\r\n// Some exporters relies on Tools.Instantiate\r\nRegisterClass(\"BABYLON.CubeTexture\", CubeTexture);\r\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"backgroundFragmentDeclaration\";\nconst shader = `uniform vec4 vEyePosition;uniform vec4 vPrimaryColor;\n#ifdef USEHIGHLIGHTANDSHADOWCOLORS\nuniform vec4 vPrimaryColorShadow;\n#endif\nuniform float shadowLevel;uniform float alpha;\n#ifdef DIFFUSE\nuniform vec2 vDiffuseInfos;\n#endif\n#ifdef REFLECTION\nuniform vec2 vReflectionInfos;uniform mat4 reflectionMatrix;uniform vec3 vReflectionMicrosurfaceInfos;\n#endif\n#if defined(REFLECTIONFRESNEL) || defined(OPACITYFRESNEL)\nuniform vec3 vBackgroundCenter;\n#endif\n#ifdef REFLECTIONFRESNEL\nuniform vec4 vReflectionControl;\n#endif\n#if defined(REFLECTIONMAP_SPHERICAL) || defined(REFLECTIONMAP_PROJECTION) || defined(REFRACTION)\nuniform mat4 view;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const backgroundFragmentDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\nimport \"./sceneUboDeclaration\";\n\nconst name = \"backgroundUboDeclaration\";\nconst shader = `layout(std140,column_major) uniform;uniform Material\n{uniform vec4 vPrimaryColor;uniform vec4 vPrimaryColorShadow;uniform vec2 vDiffuseInfos;uniform vec2 vReflectionInfos;uniform mat4 diffuseMatrix;uniform mat4 reflectionMatrix;uniform vec3 vReflectionMicrosurfaceInfos;uniform float fFovMultiplier;uniform float pointSize;uniform float shadowLevel;uniform float alpha;\n#if defined(REFLECTIONFRESNEL) || defined(OPACITYFRESNEL)\nuniform vec3 vBackgroundCenter;\n#endif\n#ifdef REFLECTIONFRESNEL\nuniform vec4 vReflectionControl;\n#endif\n};\n#include\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const backgroundUboDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/backgroundFragmentDeclaration\";\nimport \"./ShadersInclude/backgroundUboDeclaration\";\nimport \"./ShadersInclude/helperFunctions\";\nimport \"./ShadersInclude/reflectionFunction\";\nimport \"./ShadersInclude/imageProcessingDeclaration\";\nimport \"./ShadersInclude/lightFragmentDeclaration\";\nimport \"./ShadersInclude/lightUboDeclaration\";\nimport \"./ShadersInclude/lightsFragmentFunctions\";\nimport \"./ShadersInclude/shadowsFragmentFunctions\";\nimport \"./ShadersInclude/imageProcessingFunctions\";\nimport \"./ShadersInclude/clipPlaneFragmentDeclaration\";\nimport \"./ShadersInclude/fogFragmentDeclaration\";\nimport \"./ShadersInclude/clipPlaneFragment\";\nimport \"./ShadersInclude/lightFragment\";\nimport \"./ShadersInclude/fogFragment\";\n\nconst name = \"backgroundPixelShader\";\nconst shader = `#ifdef TEXTURELODSUPPORT\n#extension GL_EXT_shader_texture_lod : enable\n#endif\nprecision highp float;\n#include<__decl__backgroundFragment>\n#include\n#define RECIPROCAL_PI2 0.15915494\nvarying vec3 vPositionW;\n#ifdef MAINUV1\nvarying vec2 vMainUV1;\n#endif \n#ifdef MAINUV2 \nvarying vec2 vMainUV2; \n#endif \n#ifdef NORMAL\nvarying vec3 vNormalW;\n#endif\n#ifdef DIFFUSE\n#if DIFFUSEDIRECTUV==1\n#define vDiffuseUV vMainUV1\n#elif DIFFUSEDIRECTUV==2\n#define vDiffuseUV vMainUV2\n#else\nvarying vec2 vDiffuseUV;\n#endif\nuniform sampler2D diffuseSampler;\n#endif\n#ifdef REFLECTION\n#ifdef REFLECTIONMAP_3D\n#define sampleReflection(s,c) textureCube(s,c)\nuniform samplerCube reflectionSampler;\n#ifdef TEXTURELODSUPPORT\n#define sampleReflectionLod(s,c,l) textureCubeLodEXT(s,c,l)\n#else\nuniform samplerCube reflectionSamplerLow;uniform samplerCube reflectionSamplerHigh;\n#endif\n#else\n#define sampleReflection(s,c) texture2D(s,c)\nuniform sampler2D reflectionSampler;\n#ifdef TEXTURELODSUPPORT\n#define sampleReflectionLod(s,c,l) texture2DLodEXT(s,c,l)\n#else\nuniform samplerCube reflectionSamplerLow;uniform samplerCube reflectionSamplerHigh;\n#endif\n#endif\n#ifdef REFLECTIONMAP_SKYBOX\nvarying vec3 vPositionUVW;\n#else\n#if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED)\nvarying vec3 vDirectionW;\n#endif\n#endif\n#include\n#endif\n#ifndef FROMLINEARSPACE\n#define FROMLINEARSPACE;\n#endif\n#ifndef SHADOWONLY\n#define SHADOWONLY;\n#endif\n#include\n#include<__decl__lightFragment>[0..maxSimultaneousLights]\n#include\n#include\n#include\n#include\n#include\n#ifdef REFLECTIONFRESNEL\n#define FRESNEL_MAXIMUM_ON_ROUGH 0.25\nvec3 fresnelSchlickEnvironmentGGX(float VdotN,vec3 reflectance0,vec3 reflectance90,float smoothness)\n{float weight=mix(FRESNEL_MAXIMUM_ON_ROUGH,1.0,smoothness);return reflectance0+weight*(reflectance90-reflectance0)*pow5(saturate(1.0-VdotN));}\n#endif\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_FRAGMENT_MAIN_BEGIN\n#include\nvec3 viewDirectionW=normalize(vEyePosition.xyz-vPositionW);\n#ifdef NORMAL\nvec3 normalW=normalize(vNormalW);\n#else\nvec3 normalW=vec3(0.0,1.0,0.0);\n#endif\nfloat shadow=1.;float globalShadow=0.;float shadowLightCount=0.;\n#include[0..maxSimultaneousLights]\n#ifdef SHADOWINUSE\nglobalShadow/=shadowLightCount;\n#else\nglobalShadow=1.0;\n#endif\n#ifndef BACKMAT_SHADOWONLY\nvec4 reflectionColor=vec4(1.,1.,1.,1.);\n#ifdef REFLECTION\nvec3 reflectionVector=computeReflectionCoords(vec4(vPositionW,1.0),normalW);\n#ifdef REFLECTIONMAP_OPPOSITEZ\nreflectionVector.z*=-1.0;\n#endif\n#ifdef REFLECTIONMAP_3D\nvec3 reflectionCoords=reflectionVector;\n#else\nvec2 reflectionCoords=reflectionVector.xy;\n#ifdef REFLECTIONMAP_PROJECTION\nreflectionCoords/=reflectionVector.z;\n#endif\nreflectionCoords.y=1.0-reflectionCoords.y;\n#endif\n#ifdef REFLECTIONBLUR\nfloat reflectionLOD=vReflectionInfos.y;\n#ifdef TEXTURELODSUPPORT\nreflectionLOD=reflectionLOD*log2(vReflectionMicrosurfaceInfos.x)*vReflectionMicrosurfaceInfos.y+vReflectionMicrosurfaceInfos.z;reflectionColor=sampleReflectionLod(reflectionSampler,reflectionCoords,reflectionLOD);\n#else\nfloat lodReflectionNormalized=saturate(reflectionLOD);float lodReflectionNormalizedDoubled=lodReflectionNormalized*2.0;vec4 reflectionSpecularMid=sampleReflection(reflectionSampler,reflectionCoords);if(lodReflectionNormalizedDoubled<1.0){reflectionColor=mix(\nsampleReflection(reflectionSamplerHigh,reflectionCoords),\nreflectionSpecularMid,\nlodReflectionNormalizedDoubled\n);} else {reflectionColor=mix(\nreflectionSpecularMid,\nsampleReflection(reflectionSamplerLow,reflectionCoords),\nlodReflectionNormalizedDoubled-1.0\n);}\n#endif\n#else\nvec4 reflectionSample=sampleReflection(reflectionSampler,reflectionCoords);reflectionColor=reflectionSample;\n#endif\n#ifdef RGBDREFLECTION\nreflectionColor.rgb=fromRGBD(reflectionColor);\n#endif\n#ifdef GAMMAREFLECTION\nreflectionColor.rgb=toLinearSpace(reflectionColor.rgb);\n#endif\n#ifdef REFLECTIONBGR\nreflectionColor.rgb=reflectionColor.bgr;\n#endif\nreflectionColor.rgb*=vReflectionInfos.x;\n#endif\nvec3 diffuseColor=vec3(1.,1.,1.);float finalAlpha=alpha;\n#ifdef DIFFUSE\nvec4 diffuseMap=texture2D(diffuseSampler,vDiffuseUV);\n#ifdef GAMMADIFFUSE\ndiffuseMap.rgb=toLinearSpace(diffuseMap.rgb);\n#endif\ndiffuseMap.rgb*=vDiffuseInfos.y;\n#ifdef DIFFUSEHASALPHA\nfinalAlpha*=diffuseMap.a;\n#endif\ndiffuseColor=diffuseMap.rgb;\n#endif\n#ifdef REFLECTIONFRESNEL\nvec3 colorBase=diffuseColor;\n#else\nvec3 colorBase=reflectionColor.rgb*diffuseColor;\n#endif\ncolorBase=max(colorBase,0.0);\n#ifdef USERGBCOLOR\nvec3 finalColor=colorBase;\n#else\n#ifdef USEHIGHLIGHTANDSHADOWCOLORS\nvec3 mainColor=mix(vPrimaryColorShadow.rgb,vPrimaryColor.rgb,colorBase);\n#else\nvec3 mainColor=vPrimaryColor.rgb;\n#endif\nvec3 finalColor=colorBase*mainColor;\n#endif\n#ifdef REFLECTIONFRESNEL\nvec3 reflectionAmount=vReflectionControl.xxx;vec3 reflectionReflectance0=vReflectionControl.yyy;vec3 reflectionReflectance90=vReflectionControl.zzz;float VdotN=dot(normalize(vEyePosition.xyz),normalW);vec3 planarReflectionFresnel=fresnelSchlickEnvironmentGGX(saturate(VdotN),reflectionReflectance0,reflectionReflectance90,1.0);reflectionAmount*=planarReflectionFresnel;\n#ifdef REFLECTIONFALLOFF\nfloat reflectionDistanceFalloff=1.0-saturate(length(vPositionW.xyz-vBackgroundCenter)*vReflectionControl.w);reflectionDistanceFalloff*=reflectionDistanceFalloff;reflectionAmount*=reflectionDistanceFalloff;\n#endif\nfinalColor=mix(finalColor,reflectionColor.rgb,saturate(reflectionAmount));\n#endif\n#ifdef OPACITYFRESNEL\nfloat viewAngleToFloor=dot(normalW,normalize(vEyePosition.xyz-vBackgroundCenter));const float startAngle=0.1;float fadeFactor=saturate(viewAngleToFloor/startAngle);finalAlpha*=fadeFactor*fadeFactor;\n#endif\n#ifdef SHADOWINUSE\nfinalColor=mix(finalColor*shadowLevel,finalColor,globalShadow);\n#endif\nvec4 color=vec4(finalColor,finalAlpha);\n#else\nvec4 color=vec4(vPrimaryColor.rgb,(1.0-clamp(globalShadow,0.,1.))*alpha);\n#endif\n#include\n#ifdef IMAGEPROCESSINGPOSTPROCESS\n#if !defined(SKIPFINALCOLORCLAMP)\ncolor.rgb=clamp(color.rgb,0.,30.0);\n#endif\n#else\ncolor=applyImageProcessing(color);\n#endif\n#ifdef PREMULTIPLYALPHA\ncolor.rgb*=color.a;\n#endif\n#ifdef NOISE\ncolor.rgb+=dither(vPositionW.xy,0.5);color=max(color,0.0);\n#endif\ngl_FragColor=color;\n#define CUSTOM_FRAGMENT_MAIN_END\n}\n`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const backgroundPixelShader = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"backgroundVertexDeclaration\";\nconst shader = `uniform mat4 view;uniform mat4 viewProjection;uniform float shadowLevel;\n#ifdef DIFFUSE\nuniform mat4 diffuseMatrix;uniform vec2 vDiffuseInfos;\n#endif\n#ifdef REFLECTION\nuniform vec2 vReflectionInfos;uniform mat4 reflectionMatrix;uniform vec3 vReflectionMicrosurfaceInfos;uniform float fFovMultiplier;\n#endif\n#ifdef POINTSIZE\nuniform float pointSize;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const backgroundVertexDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/backgroundVertexDeclaration\";\nimport \"./ShadersInclude/backgroundUboDeclaration\";\nimport \"./ShadersInclude/helperFunctions\";\nimport \"./ShadersInclude/bonesDeclaration\";\nimport \"./ShadersInclude/bakedVertexAnimationDeclaration\";\nimport \"./ShadersInclude/instancesDeclaration\";\nimport \"./ShadersInclude/clipPlaneVertexDeclaration\";\nimport \"./ShadersInclude/fogVertexDeclaration\";\nimport \"./ShadersInclude/lightVxFragmentDeclaration\";\nimport \"./ShadersInclude/lightVxUboDeclaration\";\nimport \"./ShadersInclude/instancesVertex\";\nimport \"./ShadersInclude/bonesVertex\";\nimport \"./ShadersInclude/bakedVertexAnimation\";\nimport \"./ShadersInclude/clipPlaneVertex\";\nimport \"./ShadersInclude/fogVertex\";\nimport \"./ShadersInclude/shadowsVertex\";\n\nconst name = \"backgroundVertexShader\";\nconst shader = `precision highp float;\n#include<__decl__backgroundVertex>\n#include\nattribute vec3 position;\n#ifdef NORMAL\nattribute vec3 normal;\n#endif\n#include\n#include\n#include\nvarying vec3 vPositionW;\n#ifdef NORMAL\nvarying vec3 vNormalW;\n#endif\n#ifdef UV1\nattribute vec2 uv;\n#endif\n#ifdef UV2\nattribute vec2 uv2;\n#endif\n#ifdef MAINUV1\nvarying vec2 vMainUV1;\n#endif\n#ifdef MAINUV2\nvarying vec2 vMainUV2;\n#endif\n#if defined(DIFFUSE) && DIFFUSEDIRECTUV==0\nvarying vec2 vDiffuseUV;\n#endif\n#include\n#include\n#include<__decl__lightVxFragment>[0..maxSimultaneousLights]\n#ifdef REFLECTIONMAP_SKYBOX\nvarying vec3 vPositionUVW;\n#endif\n#if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED)\nvarying vec3 vDirectionW;\n#endif\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\n#ifdef REFLECTIONMAP_SKYBOX\nvPositionUVW=position;\n#endif\n#include\n#include\n#include\n#ifdef MULTIVIEW\nif (gl_ViewID_OVR==0u) {gl_Position=viewProjection*finalWorld*vec4(position,1.0);} else {gl_Position=viewProjectionR*finalWorld*vec4(position,1.0);}\n#else\ngl_Position=viewProjection*finalWorld*vec4(position,1.0);\n#endif\nvec4 worldPos=finalWorld*vec4(position,1.0);vPositionW=vec3(worldPos);\n#ifdef NORMAL\nmat3 normalWorld=mat3(finalWorld);\n#ifdef NONUNIFORMSCALING\nnormalWorld=transposeMat3(inverseMat3(normalWorld));\n#endif\nvNormalW=normalize(normalWorld*normal);\n#endif\n#if defined(REFLECTIONMAP_EQUIRECTANGULAR_FIXED) || defined(REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED)\nvDirectionW=normalize(vec3(finalWorld*vec4(position,0.0)));\n#ifdef EQUIRECTANGULAR_RELFECTION_FOV\nmat3 screenToWorld=inverseMat3(mat3(finalWorld*viewProjection));vec3 segment=mix(vDirectionW,screenToWorld*vec3(0.0,0.0,1.0),abs(fFovMultiplier-1.0));if (fFovMultiplier<=1.0) {vDirectionW=normalize(segment);} else {vDirectionW=normalize(vDirectionW+(vDirectionW-segment));}\n#endif\n#endif\n#ifndef UV1\nvec2 uv=vec2(0.,0.);\n#endif\n#ifndef UV2\nvec2 uv2=vec2(0.,0.);\n#endif\n#ifdef MAINUV1\nvMainUV1=uv;\n#endif\n#ifdef MAINUV2\nvMainUV2=uv2;\n#endif\n#if defined(DIFFUSE) && DIFFUSEDIRECTUV==0\nif (vDiffuseInfos.x==0.)\n{vDiffuseUV=vec2(diffuseMatrix*vec4(uv,1.0,0.0));}\nelse\n{vDiffuseUV=vec2(diffuseMatrix*vec4(uv2,1.0,0.0));}\n#endif\n#include\n#include\n#include[0..maxSimultaneousLights]\n#ifdef VERTEXCOLOR\nvColor=color;\n#endif\n#if defined(POINTSIZE) && !defined(WEBGPU)\ngl_PointSize=pointSize;\n#endif\n#define CUSTOM_VERTEX_MAIN_END\n}\n`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const backgroundVertexShader = { name, shader };\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport {\r\n SerializationHelper,\r\n serialize,\r\n serializeAsColor3,\r\n expandToProperty,\r\n serializeAsTexture,\r\n serializeAsVector3,\r\n serializeAsImageProcessingConfiguration,\r\n} from \"../../Misc/decorators\";\r\nimport { SmartArray } from \"../../Misc/smartArray\";\r\nimport type { Observer } from \"../../Misc/observable\";\r\nimport { Logger } from \"../../Misc/logger\";\r\nimport type { Nullable, int, float } from \"../../types\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { Matrix } from \"../../Maths/math.vector\";\r\nimport { Vector3, Vector4 } from \"../../Maths/math.vector\";\r\nimport { VertexBuffer } from \"../../Buffers/buffer\";\r\nimport type { SubMesh } from \"../../Meshes/subMesh\";\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport type { Mesh } from \"../../Meshes/mesh\";\r\nimport type { IEffectCreationOptions } from \"../../Materials/effect\";\r\nimport { MaterialHelper } from \"../../Materials/materialHelper\";\r\nimport { MaterialDefines } from \"../../Materials/materialDefines\";\r\nimport { PushMaterial } from \"../../Materials/pushMaterial\";\r\nimport type { ColorCurves } from \"../../Materials/colorCurves\";\r\nimport type { IImageProcessingConfigurationDefines } from \"../../Materials/imageProcessingConfiguration\";\r\nimport { ImageProcessingConfiguration } from \"../../Materials/imageProcessingConfiguration\";\r\nimport type { BaseTexture } from \"../../Materials/Textures/baseTexture\";\r\nimport { Texture } from \"../../Materials/Textures/texture\";\r\nimport type { RenderTargetTexture } from \"../../Materials/Textures/renderTargetTexture\";\r\nimport type { IShadowLight } from \"../../Lights/shadowLight\";\r\nimport { Constants } from \"../../Engines/constants\";\r\nimport { RegisterClass } from \"../../Misc/typeStore\";\r\nimport { MaterialFlags } from \"../materialFlags\";\r\nimport { Color3 } from \"../../Maths/math.color\";\r\n\r\nimport \"../../Shaders/background.fragment\";\r\nimport \"../../Shaders/background.vertex\";\r\nimport { EffectFallbacks } from \"../effectFallbacks\";\r\nimport { addClipPlaneUniforms, bindClipPlane } from \"../clipPlaneMaterialHelper\";\r\n\r\n/**\r\n * Background material defines definition.\r\n * @internal Mainly internal Use\r\n */\r\nclass BackgroundMaterialDefines extends MaterialDefines implements IImageProcessingConfigurationDefines {\r\n /**\r\n * True if the diffuse texture is in use.\r\n */\r\n public DIFFUSE = false;\r\n\r\n /**\r\n * The direct UV channel to use.\r\n */\r\n public DIFFUSEDIRECTUV = 0;\r\n\r\n /**\r\n * True if the diffuse texture is in gamma space.\r\n */\r\n public GAMMADIFFUSE = false;\r\n\r\n /**\r\n * True if the diffuse texture has opacity in the alpha channel.\r\n */\r\n public DIFFUSEHASALPHA = false;\r\n\r\n /**\r\n * True if you want the material to fade to transparent at grazing angle.\r\n */\r\n public OPACITYFRESNEL = false;\r\n\r\n /**\r\n * True if an extra blur needs to be added in the reflection.\r\n */\r\n public REFLECTIONBLUR = false;\r\n\r\n /**\r\n * True if you want the material to fade to reflection at grazing angle.\r\n */\r\n public REFLECTIONFRESNEL = false;\r\n\r\n /**\r\n * True if you want the material to falloff as far as you move away from the scene center.\r\n */\r\n public REFLECTIONFALLOFF = false;\r\n\r\n /**\r\n * False if the current Webgl implementation does not support the texture lod extension.\r\n */\r\n public TEXTURELODSUPPORT = false;\r\n\r\n /**\r\n * True to ensure the data are premultiplied.\r\n */\r\n public PREMULTIPLYALPHA = false;\r\n\r\n /**\r\n * True if the texture contains cooked RGB values and not gray scaled multipliers.\r\n */\r\n public USERGBCOLOR = false;\r\n\r\n /**\r\n * True if highlight and shadow levels have been specified. It can help ensuring the main perceived color\r\n * stays aligned with the desired configuration.\r\n */\r\n public USEHIGHLIGHTANDSHADOWCOLORS = false;\r\n\r\n /**\r\n * True if only shadows must be rendered\r\n */\r\n public BACKMAT_SHADOWONLY = false;\r\n\r\n /**\r\n * True to add noise in order to reduce the banding effect.\r\n */\r\n public NOISE = false;\r\n\r\n /**\r\n * is the reflection texture in BGR color scheme?\r\n * Mainly used to solve a bug in ios10 video tag\r\n */\r\n public REFLECTIONBGR = false;\r\n\r\n public IMAGEPROCESSING = false;\r\n public VIGNETTE = false;\r\n public VIGNETTEBLENDMODEMULTIPLY = false;\r\n public VIGNETTEBLENDMODEOPAQUE = false;\r\n public TONEMAPPING = false;\r\n public TONEMAPPING_ACES = false;\r\n public CONTRAST = false;\r\n public COLORCURVES = false;\r\n public COLORGRADING = false;\r\n public COLORGRADING3D = false;\r\n public SAMPLER3DGREENDEPTH = false;\r\n public SAMPLER3DBGRMAP = false;\r\n public DITHER = false;\r\n public IMAGEPROCESSINGPOSTPROCESS = false;\r\n public SKIPFINALCOLORCLAMP = false;\r\n public EXPOSURE = false;\r\n public MULTIVIEW = false;\r\n\r\n // Reflection.\r\n public REFLECTION = false;\r\n public REFLECTIONMAP_3D = false;\r\n public REFLECTIONMAP_SPHERICAL = false;\r\n public REFLECTIONMAP_PLANAR = false;\r\n public REFLECTIONMAP_CUBIC = false;\r\n public REFLECTIONMAP_PROJECTION = false;\r\n public REFLECTIONMAP_SKYBOX = false;\r\n public REFLECTIONMAP_EXPLICIT = false;\r\n public REFLECTIONMAP_EQUIRECTANGULAR = false;\r\n public REFLECTIONMAP_EQUIRECTANGULAR_FIXED = false;\r\n public REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = false;\r\n public INVERTCUBICMAP = false;\r\n public REFLECTIONMAP_OPPOSITEZ = false;\r\n public LODINREFLECTIONALPHA = false;\r\n public GAMMAREFLECTION = false;\r\n public RGBDREFLECTION = false;\r\n public EQUIRECTANGULAR_RELFECTION_FOV = false;\r\n\r\n // Default BJS.\r\n public MAINUV1 = false;\r\n public MAINUV2 = false;\r\n public UV1 = false;\r\n public UV2 = false;\r\n public CLIPPLANE = false;\r\n public CLIPPLANE2 = false;\r\n public CLIPPLANE3 = false;\r\n public CLIPPLANE4 = false;\r\n public CLIPPLANE5 = false;\r\n public CLIPPLANE6 = false;\r\n public POINTSIZE = false;\r\n public FOG = false;\r\n public NORMAL = false;\r\n public NUM_BONE_INFLUENCERS = 0;\r\n public BonesPerMesh = 0;\r\n public INSTANCES = false;\r\n public SHADOWFLOAT = false;\r\n public LOGARITHMICDEPTH = false;\r\n public NONUNIFORMSCALING = false;\r\n public ALPHATEST = false;\r\n\r\n /**\r\n * Constructor of the defines.\r\n */\r\n constructor() {\r\n super();\r\n this.rebuild();\r\n }\r\n}\r\n\r\n/**\r\n * Background material used to create an efficient environment around your scene.\r\n */\r\nexport class BackgroundMaterial extends PushMaterial {\r\n /**\r\n * Standard reflectance value at parallel view angle.\r\n */\r\n public static StandardReflectance0 = 0.05;\r\n\r\n /**\r\n * Standard reflectance value at grazing angle.\r\n */\r\n public static StandardReflectance90 = 0.5;\r\n\r\n @serializeAsColor3()\r\n protected _primaryColor: Color3;\r\n /**\r\n * Key light Color (multiply against the environment texture)\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsLightsDirty\")\r\n public primaryColor = Color3.White();\r\n\r\n @serializeAsColor3()\r\n protected __perceptualColor: Nullable;\r\n /**\r\n * Experimental Internal Use Only.\r\n *\r\n * Key light Color in \"perceptual value\" meaning the color you would like to see on screen.\r\n * This acts as a helper to set the primary color to a more \"human friendly\" value.\r\n * Conversion to linear space as well as exposure and tone mapping correction will be applied to keep the\r\n * output color as close as possible from the chosen value.\r\n * (This does not account for contrast color grading and color curves as they are considered post effect and not directly\r\n * part of lighting setup.)\r\n */\r\n public get _perceptualColor(): Nullable {\r\n return this.__perceptualColor;\r\n }\r\n public set _perceptualColor(value: Nullable) {\r\n this.__perceptualColor = value;\r\n this._computePrimaryColorFromPerceptualColor();\r\n this._markAllSubMeshesAsLightsDirty();\r\n }\r\n\r\n @serialize()\r\n protected _primaryColorShadowLevel: float = 0;\r\n /**\r\n * Defines the level of the shadows (dark area of the reflection map) in order to help scaling the colors.\r\n * The color opposite to the primary color is used at the level chosen to define what the black area would look.\r\n */\r\n public get primaryColorShadowLevel(): float {\r\n return this._primaryColorShadowLevel;\r\n }\r\n public set primaryColorShadowLevel(value: float) {\r\n this._primaryColorShadowLevel = value;\r\n this._computePrimaryColors();\r\n this._markAllSubMeshesAsLightsDirty();\r\n }\r\n\r\n @serialize()\r\n protected _primaryColorHighlightLevel: float = 0;\r\n /**\r\n * Defines the level of the highlights (highlight area of the reflection map) in order to help scaling the colors.\r\n * The primary color is used at the level chosen to define what the white area would look.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsLightsDirty\")\r\n public get primaryColorHighlightLevel(): float {\r\n return this._primaryColorHighlightLevel;\r\n }\r\n public set primaryColorHighlightLevel(value: float) {\r\n this._primaryColorHighlightLevel = value;\r\n this._computePrimaryColors();\r\n this._markAllSubMeshesAsLightsDirty();\r\n }\r\n\r\n @serializeAsTexture()\r\n protected _reflectionTexture: Nullable;\r\n /**\r\n * Reflection Texture used in the material.\r\n * Should be author in a specific way for the best result (refer to the documentation).\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public reflectionTexture: Nullable = null;\r\n\r\n @serialize()\r\n protected _reflectionBlur: float;\r\n /**\r\n * Reflection Texture level of blur.\r\n *\r\n * Can be use to reuse an existing HDR Texture and target a specific LOD to prevent authoring the\r\n * texture twice.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public reflectionBlur: float = 0;\r\n\r\n @serializeAsTexture()\r\n protected _diffuseTexture: Nullable;\r\n /**\r\n * Diffuse Texture used in the material.\r\n * Should be author in a specific way for the best result (refer to the documentation).\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public diffuseTexture: Nullable = null;\r\n\r\n protected _shadowLights: Nullable = null;\r\n /**\r\n * Specify the list of lights casting shadow on the material.\r\n * All scene shadow lights will be included if null.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public shadowLights: Nullable = null;\r\n\r\n @serialize()\r\n protected _shadowLevel: float;\r\n /**\r\n * Helps adjusting the shadow to a softer level if required.\r\n * 0 means black shadows and 1 means no shadows.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public shadowLevel: float = 0;\r\n\r\n @serializeAsVector3()\r\n protected _sceneCenter: Vector3;\r\n /**\r\n * In case of opacity Fresnel or reflection falloff, this is use as a scene center.\r\n * It is usually zero but might be interesting to modify according to your setup.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public sceneCenter: Vector3 = Vector3.Zero();\r\n\r\n @serialize()\r\n protected _opacityFresnel: boolean;\r\n /**\r\n * This helps specifying that the material is falling off to the sky box at grazing angle.\r\n * This helps ensuring a nice transition when the camera goes under the ground.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public opacityFresnel: boolean = true;\r\n\r\n @serialize()\r\n protected _reflectionFresnel: boolean;\r\n /**\r\n * This helps specifying that the material is falling off from diffuse to the reflection texture at grazing angle.\r\n * This helps adding a mirror texture on the ground.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public reflectionFresnel: boolean = false;\r\n\r\n @serialize()\r\n protected _reflectionFalloffDistance: number;\r\n /**\r\n * This helps specifying the falloff radius off the reflection texture from the sceneCenter.\r\n * This helps adding a nice falloff effect to the reflection if used as a mirror for instance.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public reflectionFalloffDistance: number = 0.0;\r\n\r\n @serialize()\r\n protected _reflectionAmount: number;\r\n /**\r\n * This specifies the weight of the reflection against the background in case of reflection Fresnel.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public reflectionAmount: number = 1.0;\r\n\r\n @serialize()\r\n protected _reflectionReflectance0: number;\r\n /**\r\n * This specifies the weight of the reflection at grazing angle.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public reflectionReflectance0: number = 0.05;\r\n\r\n @serialize()\r\n protected _reflectionReflectance90: number;\r\n /**\r\n * This specifies the weight of the reflection at a perpendicular point of view.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public reflectionReflectance90: number = 0.5;\r\n\r\n /**\r\n * Sets the reflection reflectance fresnel values according to the default standard\r\n * empirically know to work well :-)\r\n */\r\n public set reflectionStandardFresnelWeight(value: number) {\r\n let reflectionWeight = value;\r\n\r\n if (reflectionWeight < 0.5) {\r\n reflectionWeight = reflectionWeight * 2.0;\r\n this.reflectionReflectance0 = BackgroundMaterial.StandardReflectance0 * reflectionWeight;\r\n this.reflectionReflectance90 = BackgroundMaterial.StandardReflectance90 * reflectionWeight;\r\n } else {\r\n reflectionWeight = reflectionWeight * 2.0 - 1.0;\r\n this.reflectionReflectance0 = BackgroundMaterial.StandardReflectance0 + (1.0 - BackgroundMaterial.StandardReflectance0) * reflectionWeight;\r\n this.reflectionReflectance90 = BackgroundMaterial.StandardReflectance90 + (1.0 - BackgroundMaterial.StandardReflectance90) * reflectionWeight;\r\n }\r\n }\r\n\r\n @serialize()\r\n protected _useRGBColor: boolean;\r\n /**\r\n * Helps to directly use the maps channels instead of their level.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public useRGBColor: boolean = true;\r\n\r\n @serialize()\r\n protected _enableNoise: boolean;\r\n /**\r\n * This helps reducing the banding effect that could occur on the background.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public enableNoise: boolean = false;\r\n\r\n /**\r\n * The current fov(field of view) multiplier, 0.0 - 2.0. Defaults to 1.0. Lower values \"zoom in\" and higher values \"zoom out\".\r\n * Best used when trying to implement visual zoom effects like fish-eye or binoculars while not adjusting camera fov.\r\n * Recommended to be keep at 1.0 except for special cases.\r\n */\r\n public get fovMultiplier(): number {\r\n return this._fovMultiplier;\r\n }\r\n public set fovMultiplier(value: number) {\r\n if (isNaN(value)) {\r\n value = 1.0;\r\n }\r\n this._fovMultiplier = Math.max(0.0, Math.min(2.0, value));\r\n }\r\n private _fovMultiplier: float = 1.0;\r\n\r\n /**\r\n * Enable the FOV adjustment feature controlled by fovMultiplier.\r\n */\r\n public useEquirectangularFOV: boolean = false;\r\n\r\n @serialize()\r\n private _maxSimultaneousLights: int = 4;\r\n /**\r\n * Number of Simultaneous lights allowed on the material.\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsTexturesDirty\")\r\n public maxSimultaneousLights: int = 4;\r\n\r\n @serialize()\r\n private _shadowOnly: boolean = false;\r\n /**\r\n * Make the material only render shadows\r\n */\r\n @expandToProperty(\"_markAllSubMeshesAsLightsDirty\")\r\n public shadowOnly: boolean = false;\r\n\r\n /**\r\n * Default configuration related to image processing available in the Background Material.\r\n */\r\n @serializeAsImageProcessingConfiguration()\r\n protected _imageProcessingConfiguration: ImageProcessingConfiguration;\r\n\r\n /**\r\n * Keep track of the image processing observer to allow dispose and replace.\r\n */\r\n private _imageProcessingObserver: Nullable> = null;\r\n\r\n /**\r\n * Attaches a new image processing configuration to the PBR Material.\r\n * @param configuration (if null the scene configuration will be use)\r\n */\r\n protected _attachImageProcessingConfiguration(configuration: Nullable): void {\r\n if (configuration === this._imageProcessingConfiguration) {\r\n return;\r\n }\r\n\r\n // Detaches observer.\r\n if (this._imageProcessingConfiguration && this._imageProcessingObserver) {\r\n this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver);\r\n }\r\n\r\n // Pick the scene configuration if needed.\r\n if (!configuration) {\r\n this._imageProcessingConfiguration = this.getScene().imageProcessingConfiguration;\r\n } else {\r\n this._imageProcessingConfiguration = configuration;\r\n }\r\n\r\n // Attaches observer.\r\n if (this._imageProcessingConfiguration) {\r\n this._imageProcessingObserver = this._imageProcessingConfiguration.onUpdateParameters.add(() => {\r\n this._computePrimaryColorFromPerceptualColor();\r\n this._markAllSubMeshesAsImageProcessingDirty();\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Gets the image processing configuration used either in this material.\r\n */\r\n public get imageProcessingConfiguration(): Nullable {\r\n return this._imageProcessingConfiguration;\r\n }\r\n\r\n /**\r\n * Sets the Default image processing configuration used either in the this material.\r\n *\r\n * If sets to null, the scene one is in use.\r\n */\r\n public set imageProcessingConfiguration(value: Nullable) {\r\n this._attachImageProcessingConfiguration(value);\r\n\r\n // Ensure the effect will be rebuilt.\r\n this._markAllSubMeshesAsTexturesDirty();\r\n }\r\n\r\n /**\r\n * Gets whether the color curves effect is enabled.\r\n */\r\n public get cameraColorCurvesEnabled(): boolean {\r\n return (this.imageProcessingConfiguration).colorCurvesEnabled;\r\n }\r\n /**\r\n * Sets whether the color curves effect is enabled.\r\n */\r\n public set cameraColorCurvesEnabled(value: boolean) {\r\n (this.imageProcessingConfiguration).colorCurvesEnabled = value;\r\n }\r\n\r\n /**\r\n * Gets whether the color grading effect is enabled.\r\n */\r\n public get cameraColorGradingEnabled(): boolean {\r\n return (this.imageProcessingConfiguration).colorGradingEnabled;\r\n }\r\n /**\r\n * Gets whether the color grading effect is enabled.\r\n */\r\n public set cameraColorGradingEnabled(value: boolean) {\r\n (this.imageProcessingConfiguration).colorGradingEnabled = value;\r\n }\r\n\r\n /**\r\n * Gets whether tonemapping is enabled or not.\r\n */\r\n public get cameraToneMappingEnabled(): boolean {\r\n return this._imageProcessingConfiguration.toneMappingEnabled;\r\n }\r\n /**\r\n * Sets whether tonemapping is enabled or not\r\n */\r\n public set cameraToneMappingEnabled(value: boolean) {\r\n this._imageProcessingConfiguration.toneMappingEnabled = value;\r\n }\r\n\r\n /**\r\n * The camera exposure used on this material.\r\n * This property is here and not in the camera to allow controlling exposure without full screen post process.\r\n * This corresponds to a photographic exposure.\r\n */\r\n public get cameraExposure(): float {\r\n return this._imageProcessingConfiguration.exposure;\r\n }\r\n /**\r\n * The camera exposure used on this material.\r\n * This property is here and not in the camera to allow controlling exposure without full screen post process.\r\n * This corresponds to a photographic exposure.\r\n */\r\n public set cameraExposure(value: float) {\r\n this._imageProcessingConfiguration.exposure = value;\r\n }\r\n\r\n /**\r\n * Gets The camera contrast used on this material.\r\n */\r\n public get cameraContrast(): float {\r\n return this._imageProcessingConfiguration.contrast;\r\n }\r\n\r\n /**\r\n * Sets The camera contrast used on this material.\r\n */\r\n public set cameraContrast(value: float) {\r\n this._imageProcessingConfiguration.contrast = value;\r\n }\r\n\r\n /**\r\n * Gets the Color Grading 2D Lookup Texture.\r\n */\r\n public get cameraColorGradingTexture(): Nullable {\r\n return this._imageProcessingConfiguration.colorGradingTexture;\r\n }\r\n /**\r\n * Sets the Color Grading 2D Lookup Texture.\r\n */\r\n public set cameraColorGradingTexture(value: Nullable) {\r\n (this.imageProcessingConfiguration).colorGradingTexture = value;\r\n }\r\n\r\n /**\r\n * The color grading curves provide additional color adjustment that is applied after any color grading transform (3D LUT).\r\n * They allow basic adjustment of saturation and small exposure adjustments, along with color filter tinting to provide white balance adjustment or more stylistic effects.\r\n * These are similar to controls found in many professional imaging or colorist software. The global controls are applied to the entire image. For advanced tuning, extra controls are provided to adjust the shadow, midtone and highlight areas of the image;\r\n * corresponding to low luminance, medium luminance, and high luminance areas respectively.\r\n */\r\n public get cameraColorCurves(): Nullable {\r\n return (this.imageProcessingConfiguration).colorCurves;\r\n }\r\n /**\r\n * The color grading curves provide additional color adjustment that is applied after any color grading transform (3D LUT).\r\n * They allow basic adjustment of saturation and small exposure adjustments, along with color filter tinting to provide white balance adjustment or more stylistic effects.\r\n * These are similar to controls found in many professional imaging or colorist software. The global controls are applied to the entire image. For advanced tuning, extra controls are provided to adjust the shadow, midtone and highlight areas of the image;\r\n * corresponding to low luminance, medium luminance, and high luminance areas respectively.\r\n */\r\n public set cameraColorCurves(value: Nullable) {\r\n (this.imageProcessingConfiguration).colorCurves = value;\r\n }\r\n\r\n /**\r\n * Due to a bug in iOS10, video tags (which are using the background material) are in BGR and not RGB.\r\n * Setting this flag to true (not done automatically!) will convert it back to RGB.\r\n */\r\n public switchToBGR: boolean = false;\r\n\r\n // Temp values kept as cache in the material.\r\n private _renderTargets = new SmartArray(16);\r\n private _reflectionControls = Vector4.Zero();\r\n private _white = Color3.White();\r\n private _primaryShadowColor = Color3.Black();\r\n private _primaryHighlightColor = Color3.Black();\r\n\r\n /**\r\n * Instantiates a Background Material in the given scene\r\n * @param name The friendly name of the material\r\n * @param scene The scene to add the material to\r\n */\r\n constructor(name: string, scene?: Scene) {\r\n super(name, scene);\r\n\r\n // Setup the default processing configuration to the scene.\r\n this._attachImageProcessingConfiguration(null);\r\n\r\n this.getRenderTargetTextures = (): SmartArray => {\r\n this._renderTargets.reset();\r\n\r\n if (this._diffuseTexture && this._diffuseTexture.isRenderTarget) {\r\n this._renderTargets.push(this._diffuseTexture as RenderTargetTexture);\r\n }\r\n\r\n if (this._reflectionTexture && this._reflectionTexture.isRenderTarget) {\r\n this._renderTargets.push(this._reflectionTexture as RenderTargetTexture);\r\n }\r\n\r\n return this._renderTargets;\r\n };\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating that current material needs to register RTT\r\n */\r\n public get hasRenderTargetTextures(): boolean {\r\n if (this._diffuseTexture && this._diffuseTexture.isRenderTarget) {\r\n return true;\r\n }\r\n\r\n if (this._reflectionTexture && this._reflectionTexture.isRenderTarget) {\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * The entire material has been created in order to prevent overdraw.\r\n * @returns false\r\n */\r\n public needAlphaTesting(): boolean {\r\n return true;\r\n }\r\n\r\n /**\r\n * The entire material has been created in order to prevent overdraw.\r\n * @returns true if blending is enable\r\n */\r\n public needAlphaBlending(): boolean {\r\n return this.alpha < 1 || (this._diffuseTexture != null && this._diffuseTexture.hasAlpha) || this._shadowOnly;\r\n }\r\n\r\n /**\r\n * Checks whether the material is ready to be rendered for a given mesh.\r\n * @param mesh The mesh to render\r\n * @param subMesh The submesh to check against\r\n * @param useInstances Specify wether or not the material is used with instances\r\n * @returns true if all the dependencies are ready (Textures, Effects...)\r\n */\r\n public isReadyForSubMesh(mesh: AbstractMesh, subMesh: SubMesh, useInstances: boolean = false): boolean {\r\n if (subMesh.effect && this.isFrozen) {\r\n if (subMesh.effect._wasPreviouslyReady && subMesh.effect._wasPreviouslyUsingInstances === useInstances) {\r\n return true;\r\n }\r\n }\r\n\r\n if (!subMesh.materialDefines) {\r\n subMesh.materialDefines = new BackgroundMaterialDefines();\r\n }\r\n\r\n const scene = this.getScene();\r\n const defines = subMesh.materialDefines;\r\n\r\n if (this._isReadyForSubMesh(subMesh)) {\r\n return true;\r\n }\r\n\r\n const engine = scene.getEngine();\r\n\r\n // Lights\r\n MaterialHelper.PrepareDefinesForLights(scene, mesh, defines, false, this._maxSimultaneousLights);\r\n defines._needNormals = true;\r\n\r\n // Multiview\r\n MaterialHelper.PrepareDefinesForMultiview(scene, defines);\r\n\r\n // Textures\r\n if (defines._areTexturesDirty) {\r\n defines._needUVs = false;\r\n if (scene.texturesEnabled) {\r\n if (scene.getEngine().getCaps().textureLOD) {\r\n defines.TEXTURELODSUPPORT = true;\r\n }\r\n\r\n if (this._diffuseTexture && MaterialFlags.DiffuseTextureEnabled) {\r\n if (!this._diffuseTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n\r\n MaterialHelper.PrepareDefinesForMergedUV(this._diffuseTexture, defines, \"DIFFUSE\");\r\n defines.DIFFUSEHASALPHA = this._diffuseTexture.hasAlpha;\r\n defines.GAMMADIFFUSE = this._diffuseTexture.gammaSpace;\r\n defines.OPACITYFRESNEL = this._opacityFresnel;\r\n } else {\r\n defines.DIFFUSE = false;\r\n defines.DIFFUSEDIRECTUV = 0;\r\n defines.DIFFUSEHASALPHA = false;\r\n defines.GAMMADIFFUSE = false;\r\n defines.OPACITYFRESNEL = false;\r\n }\r\n\r\n const reflectionTexture = this._reflectionTexture;\r\n if (reflectionTexture && MaterialFlags.ReflectionTextureEnabled) {\r\n if (!reflectionTexture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n\r\n defines.REFLECTION = true;\r\n defines.GAMMAREFLECTION = reflectionTexture.gammaSpace;\r\n defines.RGBDREFLECTION = reflectionTexture.isRGBD;\r\n defines.REFLECTIONBLUR = this._reflectionBlur > 0;\r\n defines.LODINREFLECTIONALPHA = reflectionTexture.lodLevelInAlpha;\r\n defines.EQUIRECTANGULAR_RELFECTION_FOV = this.useEquirectangularFOV;\r\n defines.REFLECTIONBGR = this.switchToBGR;\r\n\r\n if (reflectionTexture.coordinatesMode === Texture.INVCUBIC_MODE) {\r\n defines.INVERTCUBICMAP = true;\r\n }\r\n\r\n defines.REFLECTIONMAP_3D = reflectionTexture.isCube;\r\n defines.REFLECTIONMAP_OPPOSITEZ = defines.REFLECTIONMAP_3D && this.getScene().useRightHandedSystem ? !reflectionTexture.invertZ : reflectionTexture.invertZ;\r\n\r\n switch (reflectionTexture.coordinatesMode) {\r\n case Texture.EXPLICIT_MODE:\r\n defines.REFLECTIONMAP_EXPLICIT = true;\r\n break;\r\n case Texture.PLANAR_MODE:\r\n defines.REFLECTIONMAP_PLANAR = true;\r\n break;\r\n case Texture.PROJECTION_MODE:\r\n defines.REFLECTIONMAP_PROJECTION = true;\r\n break;\r\n case Texture.SKYBOX_MODE:\r\n defines.REFLECTIONMAP_SKYBOX = true;\r\n break;\r\n case Texture.SPHERICAL_MODE:\r\n defines.REFLECTIONMAP_SPHERICAL = true;\r\n break;\r\n case Texture.EQUIRECTANGULAR_MODE:\r\n defines.REFLECTIONMAP_EQUIRECTANGULAR = true;\r\n break;\r\n case Texture.FIXED_EQUIRECTANGULAR_MODE:\r\n defines.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = true;\r\n break;\r\n case Texture.FIXED_EQUIRECTANGULAR_MIRRORED_MODE:\r\n defines.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = true;\r\n break;\r\n case Texture.CUBIC_MODE:\r\n case Texture.INVCUBIC_MODE:\r\n default:\r\n defines.REFLECTIONMAP_CUBIC = true;\r\n break;\r\n }\r\n\r\n if (this.reflectionFresnel) {\r\n defines.REFLECTIONFRESNEL = true;\r\n defines.REFLECTIONFALLOFF = this.reflectionFalloffDistance > 0;\r\n\r\n this._reflectionControls.x = this.reflectionAmount;\r\n this._reflectionControls.y = this.reflectionReflectance0;\r\n this._reflectionControls.z = this.reflectionReflectance90;\r\n this._reflectionControls.w = 1 / this.reflectionFalloffDistance;\r\n } else {\r\n defines.REFLECTIONFRESNEL = false;\r\n defines.REFLECTIONFALLOFF = false;\r\n }\r\n } else {\r\n defines.REFLECTION = false;\r\n defines.REFLECTIONFRESNEL = false;\r\n defines.REFLECTIONFALLOFF = false;\r\n defines.REFLECTIONBLUR = false;\r\n defines.REFLECTIONMAP_3D = false;\r\n defines.REFLECTIONMAP_SPHERICAL = false;\r\n defines.REFLECTIONMAP_PLANAR = false;\r\n defines.REFLECTIONMAP_CUBIC = false;\r\n defines.REFLECTIONMAP_PROJECTION = false;\r\n defines.REFLECTIONMAP_SKYBOX = false;\r\n defines.REFLECTIONMAP_EXPLICIT = false;\r\n defines.REFLECTIONMAP_EQUIRECTANGULAR = false;\r\n defines.REFLECTIONMAP_EQUIRECTANGULAR_FIXED = false;\r\n defines.REFLECTIONMAP_MIRROREDEQUIRECTANGULAR_FIXED = false;\r\n defines.INVERTCUBICMAP = false;\r\n defines.REFLECTIONMAP_OPPOSITEZ = false;\r\n defines.LODINREFLECTIONALPHA = false;\r\n defines.GAMMAREFLECTION = false;\r\n defines.RGBDREFLECTION = false;\r\n }\r\n }\r\n\r\n defines.PREMULTIPLYALPHA = this.alphaMode === Constants.ALPHA_PREMULTIPLIED || this.alphaMode === Constants.ALPHA_PREMULTIPLIED_PORTERDUFF;\r\n defines.USERGBCOLOR = this._useRGBColor;\r\n defines.NOISE = this._enableNoise;\r\n }\r\n\r\n if (defines._areLightsDirty) {\r\n defines.USEHIGHLIGHTANDSHADOWCOLORS = !this._useRGBColor && (this._primaryColorShadowLevel !== 0 || this._primaryColorHighlightLevel !== 0);\r\n defines.BACKMAT_SHADOWONLY = this._shadowOnly;\r\n }\r\n\r\n if (defines._areImageProcessingDirty && this._imageProcessingConfiguration) {\r\n if (!this._imageProcessingConfiguration.isReady()) {\r\n return false;\r\n }\r\n\r\n this._imageProcessingConfiguration.prepareDefines(defines);\r\n }\r\n\r\n // Misc.\r\n MaterialHelper.PrepareDefinesForMisc(mesh, scene, false, this.pointsCloud, this.fogEnabled, this._shouldTurnAlphaTestOn(mesh), defines);\r\n\r\n // Values that need to be evaluated on every frame\r\n MaterialHelper.PrepareDefinesForFrameBoundValues(scene, engine, this, defines, useInstances, null, subMesh.getRenderingMesh().hasThinInstances);\r\n\r\n // Attribs\r\n if (MaterialHelper.PrepareDefinesForAttributes(mesh, defines, false, true, false)) {\r\n if (mesh) {\r\n if (!scene.getEngine().getCaps().standardDerivatives && !mesh.isVerticesDataPresent(VertexBuffer.NormalKind)) {\r\n mesh.createNormals(true);\r\n Logger.Warn(\"BackgroundMaterial: Normals have been created for the mesh: \" + mesh.name);\r\n }\r\n }\r\n }\r\n\r\n // Get correct effect\r\n if (defines.isDirty) {\r\n defines.markAsProcessed();\r\n scene.resetCachedMaterial();\r\n\r\n // Fallbacks\r\n const fallbacks = new EffectFallbacks();\r\n if (defines.FOG) {\r\n fallbacks.addFallback(0, \"FOG\");\r\n }\r\n\r\n if (defines.POINTSIZE) {\r\n fallbacks.addFallback(1, \"POINTSIZE\");\r\n }\r\n\r\n if (defines.MULTIVIEW) {\r\n fallbacks.addFallback(0, \"MULTIVIEW\");\r\n }\r\n\r\n MaterialHelper.HandleFallbacksForShadows(defines, fallbacks, this._maxSimultaneousLights);\r\n\r\n //Attributes\r\n const attribs = [VertexBuffer.PositionKind];\r\n\r\n if (defines.NORMAL) {\r\n attribs.push(VertexBuffer.NormalKind);\r\n }\r\n\r\n if (defines.UV1) {\r\n attribs.push(VertexBuffer.UVKind);\r\n }\r\n\r\n if (defines.UV2) {\r\n attribs.push(VertexBuffer.UV2Kind);\r\n }\r\n\r\n MaterialHelper.PrepareAttributesForBones(attribs, mesh, defines, fallbacks);\r\n MaterialHelper.PrepareAttributesForInstances(attribs, defines);\r\n\r\n const uniforms = [\r\n \"world\",\r\n \"view\",\r\n \"viewProjection\",\r\n \"vEyePosition\",\r\n \"vLightsType\",\r\n \"vFogInfos\",\r\n \"vFogColor\",\r\n \"pointSize\",\r\n \"mBones\",\r\n\r\n \"vPrimaryColor\",\r\n \"vPrimaryColorShadow\",\r\n \"vReflectionInfos\",\r\n \"reflectionMatrix\",\r\n \"vReflectionMicrosurfaceInfos\",\r\n \"fFovMultiplier\",\r\n\r\n \"shadowLevel\",\r\n \"alpha\",\r\n\r\n \"vBackgroundCenter\",\r\n \"vReflectionControl\",\r\n\r\n \"vDiffuseInfos\",\r\n \"diffuseMatrix\",\r\n ];\r\n\r\n addClipPlaneUniforms(uniforms);\r\n const samplers = [\"diffuseSampler\", \"reflectionSampler\", \"reflectionSamplerLow\", \"reflectionSamplerHigh\"];\r\n const uniformBuffers = [\"Material\", \"Scene\"];\r\n\r\n if (ImageProcessingConfiguration) {\r\n ImageProcessingConfiguration.PrepareUniforms(uniforms, defines);\r\n ImageProcessingConfiguration.PrepareSamplers(samplers, defines);\r\n }\r\n\r\n MaterialHelper.PrepareUniformsAndSamplersList({\r\n uniformsNames: uniforms,\r\n uniformBuffersNames: uniformBuffers,\r\n samplers: samplers,\r\n defines: defines,\r\n maxSimultaneousLights: this._maxSimultaneousLights,\r\n });\r\n\r\n const join = defines.toString();\r\n const effect = scene.getEngine().createEffect(\r\n \"background\",\r\n {\r\n attributes: attribs,\r\n uniformsNames: uniforms,\r\n uniformBuffersNames: uniformBuffers,\r\n samplers: samplers,\r\n defines: join,\r\n fallbacks: fallbacks,\r\n onCompiled: this.onCompiled,\r\n onError: this.onError,\r\n indexParameters: { maxSimultaneousLights: this._maxSimultaneousLights },\r\n },\r\n engine\r\n );\r\n subMesh.setEffect(effect, defines, this._materialContext);\r\n\r\n this.buildUniformLayout();\r\n }\r\n\r\n if (!subMesh.effect || !subMesh.effect.isReady()) {\r\n return false;\r\n }\r\n\r\n defines._renderId = scene.getRenderId();\r\n subMesh.effect._wasPreviouslyReady = true;\r\n subMesh.effect._wasPreviouslyUsingInstances = useInstances;\r\n\r\n this._checkScenePerformancePriority();\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Compute the primary color according to the chosen perceptual color.\r\n */\r\n private _computePrimaryColorFromPerceptualColor(): void {\r\n if (!this.__perceptualColor) {\r\n return;\r\n }\r\n\r\n this._primaryColor.copyFrom(this.__perceptualColor);\r\n\r\n // Revert gamma space.\r\n this._primaryColor.toLinearSpaceToRef(this._primaryColor, this.getScene().getEngine().useExactSrgbConversions);\r\n\r\n // Revert image processing configuration.\r\n if (this._imageProcessingConfiguration) {\r\n // Revert Exposure.\r\n this._primaryColor.scaleToRef(1 / this._imageProcessingConfiguration.exposure, this._primaryColor);\r\n }\r\n\r\n this._computePrimaryColors();\r\n }\r\n\r\n /**\r\n * Compute the highlights and shadow colors according to their chosen levels.\r\n */\r\n private _computePrimaryColors(): void {\r\n if (this._primaryColorShadowLevel === 0 && this._primaryColorHighlightLevel === 0) {\r\n return;\r\n }\r\n\r\n // Find the highlight color based on the configuration.\r\n this._primaryColor.scaleToRef(this._primaryColorShadowLevel, this._primaryShadowColor);\r\n this._primaryColor.subtractToRef(this._primaryShadowColor, this._primaryShadowColor);\r\n\r\n // Find the shadow color based on the configuration.\r\n this._white.subtractToRef(this._primaryColor, this._primaryHighlightColor);\r\n this._primaryHighlightColor.scaleToRef(this._primaryColorHighlightLevel, this._primaryHighlightColor);\r\n this._primaryColor.addToRef(this._primaryHighlightColor, this._primaryHighlightColor);\r\n }\r\n\r\n /**\r\n * Build the uniform buffer used in the material.\r\n */\r\n public buildUniformLayout(): void {\r\n // Order is important !\r\n this._uniformBuffer.addUniform(\"vPrimaryColor\", 4);\r\n this._uniformBuffer.addUniform(\"vPrimaryColorShadow\", 4);\r\n this._uniformBuffer.addUniform(\"vDiffuseInfos\", 2);\r\n this._uniformBuffer.addUniform(\"vReflectionInfos\", 2);\r\n this._uniformBuffer.addUniform(\"diffuseMatrix\", 16);\r\n this._uniformBuffer.addUniform(\"reflectionMatrix\", 16);\r\n this._uniformBuffer.addUniform(\"vReflectionMicrosurfaceInfos\", 3);\r\n this._uniformBuffer.addUniform(\"fFovMultiplier\", 1);\r\n this._uniformBuffer.addUniform(\"pointSize\", 1);\r\n this._uniformBuffer.addUniform(\"shadowLevel\", 1);\r\n this._uniformBuffer.addUniform(\"alpha\", 1);\r\n this._uniformBuffer.addUniform(\"vBackgroundCenter\", 3);\r\n this._uniformBuffer.addUniform(\"vReflectionControl\", 4);\r\n\r\n this._uniformBuffer.create();\r\n }\r\n\r\n /**\r\n * Unbind the material.\r\n */\r\n public unbind(): void {\r\n if (this._diffuseTexture && this._diffuseTexture.isRenderTarget) {\r\n this._uniformBuffer.setTexture(\"diffuseSampler\", null);\r\n }\r\n\r\n if (this._reflectionTexture && this._reflectionTexture.isRenderTarget) {\r\n this._uniformBuffer.setTexture(\"reflectionSampler\", null);\r\n }\r\n\r\n super.unbind();\r\n }\r\n\r\n /**\r\n * Bind only the world matrix to the material.\r\n * @param world The world matrix to bind.\r\n */\r\n public bindOnlyWorldMatrix(world: Matrix): void {\r\n this._activeEffect!.setMatrix(\"world\", world);\r\n }\r\n\r\n /**\r\n * Bind the material for a dedicated submeh (every used meshes will be considered opaque).\r\n * @param world The world matrix to bind.\r\n * @param mesh\r\n * @param subMesh The submesh to bind for.\r\n */\r\n public bindForSubMesh(world: Matrix, mesh: Mesh, subMesh: SubMesh): void {\r\n const scene = this.getScene();\r\n\r\n const defines = subMesh.materialDefines;\r\n if (!defines) {\r\n return;\r\n }\r\n\r\n const effect = subMesh.effect;\r\n if (!effect) {\r\n return;\r\n }\r\n this._activeEffect = effect;\r\n\r\n // Matrices\r\n this.bindOnlyWorldMatrix(world);\r\n\r\n // Bones\r\n MaterialHelper.BindBonesParameters(mesh, this._activeEffect);\r\n\r\n const mustRebind = this._mustRebind(scene, effect, mesh.visibility);\r\n if (mustRebind) {\r\n this._uniformBuffer.bindToEffect(effect, \"Material\");\r\n\r\n this.bindViewProjection(effect);\r\n\r\n const reflectionTexture = this._reflectionTexture;\r\n if (!this._uniformBuffer.useUbo || !this.isFrozen || !this._uniformBuffer.isSync) {\r\n // Texture uniforms\r\n if (scene.texturesEnabled) {\r\n if (this._diffuseTexture && MaterialFlags.DiffuseTextureEnabled) {\r\n this._uniformBuffer.updateFloat2(\"vDiffuseInfos\", this._diffuseTexture.coordinatesIndex, this._diffuseTexture.level);\r\n MaterialHelper.BindTextureMatrix(this._diffuseTexture, this._uniformBuffer, \"diffuse\");\r\n }\r\n\r\n if (reflectionTexture && MaterialFlags.ReflectionTextureEnabled) {\r\n this._uniformBuffer.updateMatrix(\"reflectionMatrix\", reflectionTexture.getReflectionTextureMatrix());\r\n this._uniformBuffer.updateFloat2(\"vReflectionInfos\", reflectionTexture.level, this._reflectionBlur);\r\n\r\n this._uniformBuffer.updateFloat3(\r\n \"vReflectionMicrosurfaceInfos\",\r\n reflectionTexture.getSize().width,\r\n reflectionTexture.lodGenerationScale,\r\n reflectionTexture.lodGenerationOffset\r\n );\r\n }\r\n }\r\n\r\n if (this.shadowLevel > 0) {\r\n this._uniformBuffer.updateFloat(\"shadowLevel\", this.shadowLevel);\r\n }\r\n this._uniformBuffer.updateFloat(\"alpha\", this.alpha);\r\n\r\n // Point size\r\n if (this.pointsCloud) {\r\n this._uniformBuffer.updateFloat(\"pointSize\", this.pointSize);\r\n }\r\n\r\n if (defines.USEHIGHLIGHTANDSHADOWCOLORS) {\r\n this._uniformBuffer.updateColor4(\"vPrimaryColor\", this._primaryHighlightColor, 1.0);\r\n this._uniformBuffer.updateColor4(\"vPrimaryColorShadow\", this._primaryShadowColor, 1.0);\r\n } else {\r\n this._uniformBuffer.updateColor4(\"vPrimaryColor\", this._primaryColor, 1.0);\r\n }\r\n }\r\n\r\n this._uniformBuffer.updateFloat(\"fFovMultiplier\", this._fovMultiplier);\r\n\r\n // Textures\r\n if (scene.texturesEnabled) {\r\n if (this._diffuseTexture && MaterialFlags.DiffuseTextureEnabled) {\r\n this._uniformBuffer.setTexture(\"diffuseSampler\", this._diffuseTexture);\r\n }\r\n\r\n if (reflectionTexture && MaterialFlags.ReflectionTextureEnabled) {\r\n if (defines.REFLECTIONBLUR && defines.TEXTURELODSUPPORT) {\r\n this._uniformBuffer.setTexture(\"reflectionSampler\", reflectionTexture);\r\n } else if (!defines.REFLECTIONBLUR) {\r\n this._uniformBuffer.setTexture(\"reflectionSampler\", reflectionTexture);\r\n } else {\r\n this._uniformBuffer.setTexture(\"reflectionSampler\", reflectionTexture._lodTextureMid || reflectionTexture);\r\n this._uniformBuffer.setTexture(\"reflectionSamplerLow\", reflectionTexture._lodTextureLow || reflectionTexture);\r\n this._uniformBuffer.setTexture(\"reflectionSamplerHigh\", reflectionTexture._lodTextureHigh || reflectionTexture);\r\n }\r\n\r\n if (defines.REFLECTIONFRESNEL) {\r\n this._uniformBuffer.updateFloat3(\"vBackgroundCenter\", this.sceneCenter.x, this.sceneCenter.y, this.sceneCenter.z);\r\n this._uniformBuffer.updateFloat4(\r\n \"vReflectionControl\",\r\n this._reflectionControls.x,\r\n this._reflectionControls.y,\r\n this._reflectionControls.z,\r\n this._reflectionControls.w\r\n );\r\n }\r\n }\r\n }\r\n\r\n // Clip plane\r\n bindClipPlane(this._activeEffect, this, scene);\r\n\r\n scene.bindEyePosition(effect);\r\n } else if (scene.getEngine()._features.needToAlwaysBindUniformBuffers) {\r\n this._uniformBuffer.bindToEffect(effect, \"Material\");\r\n this._needToBindSceneUbo = true;\r\n }\r\n\r\n if (mustRebind || !this.isFrozen) {\r\n if (scene.lightsEnabled) {\r\n MaterialHelper.BindLights(scene, mesh, this._activeEffect, defines, this._maxSimultaneousLights);\r\n }\r\n\r\n // View\r\n this.bindView(effect);\r\n\r\n // Fog\r\n MaterialHelper.BindFogParameters(scene, mesh, this._activeEffect, true);\r\n\r\n // image processing\r\n if (this._imageProcessingConfiguration) {\r\n this._imageProcessingConfiguration.bind(this._activeEffect);\r\n }\r\n }\r\n\r\n this._afterBind(mesh, this._activeEffect);\r\n\r\n this._uniformBuffer.update();\r\n }\r\n\r\n /**\r\n * Checks to see if a texture is used in the material.\r\n * @param texture - Base texture to use.\r\n * @returns - Boolean specifying if a texture is used in the material.\r\n */\r\n public hasTexture(texture: BaseTexture): boolean {\r\n if (super.hasTexture(texture)) {\r\n return true;\r\n }\r\n\r\n if (this._reflectionTexture === texture) {\r\n return true;\r\n }\r\n\r\n if (this._diffuseTexture === texture) {\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Dispose the material.\r\n * @param forceDisposeEffect Force disposal of the associated effect.\r\n * @param forceDisposeTextures Force disposal of the associated textures.\r\n */\r\n public dispose(forceDisposeEffect: boolean = false, forceDisposeTextures: boolean = false): void {\r\n if (forceDisposeTextures) {\r\n if (this.diffuseTexture) {\r\n this.diffuseTexture.dispose();\r\n }\r\n if (this.reflectionTexture) {\r\n this.reflectionTexture.dispose();\r\n }\r\n }\r\n\r\n this._renderTargets.dispose();\r\n\r\n if (this._imageProcessingConfiguration && this._imageProcessingObserver) {\r\n this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver);\r\n }\r\n\r\n super.dispose(forceDisposeEffect);\r\n }\r\n\r\n /**\r\n * Clones the material.\r\n * @param name The cloned name.\r\n * @returns The cloned material.\r\n */\r\n public clone(name: string): BackgroundMaterial {\r\n return SerializationHelper.Clone(() => new BackgroundMaterial(name, this.getScene()), this);\r\n }\r\n\r\n /**\r\n * Serializes the current material to its JSON representation.\r\n * @returns The JSON representation.\r\n */\r\n public serialize(): any {\r\n const serializationObject = super.serialize();\r\n serializationObject.customType = \"BABYLON.BackgroundMaterial\";\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Gets the class name of the material\r\n * @returns \"BackgroundMaterial\"\r\n */\r\n public getClassName(): string {\r\n return \"BackgroundMaterial\";\r\n }\r\n\r\n /**\r\n * Parse a JSON input to create back a background material.\r\n * @param source The JSON data to parse\r\n * @param scene The scene to create the parsed material in\r\n * @param rootUrl The root url of the assets the material depends upon\r\n * @returns the instantiated BackgroundMaterial.\r\n */\r\n public static Parse(source: any, scene: Scene, rootUrl: string): BackgroundMaterial {\r\n return SerializationHelper.Parse(() => new BackgroundMaterial(source.name, scene), source, scene, rootUrl);\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.BackgroundMaterial\", BackgroundMaterial);\r\n","import type { Scene } from \"../../scene\";\r\nimport type { Vector4 } from \"../../Maths/math.vector\";\r\nimport { Mesh } from \"../mesh\";\r\nimport { VertexData } from \"../mesh.vertexData\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { Plane } from \"../../Maths/math.plane\";\r\nimport { CompatibilityOptions } from \"../../Compat/compatibilityOptions\";\r\n\r\n/**\r\n * Creates the VertexData for a Plane\r\n * @param options an object used to set the following optional parameters for the plane, required but can be empty\r\n * * size sets the width and height of the plane to the value of size, optional default 1\r\n * * width sets the width (x direction) of the plane, overwrites the width set by size, optional, default size\r\n * * height sets the height (y direction) of the plane, overwrites the height set by size, optional, default size\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.size\r\n * @param options.width\r\n * @param options.height\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @returns the VertexData of the box\r\n */\r\nexport function CreatePlaneVertexData(options: { size?: number; width?: number; height?: number; sideOrientation?: number; frontUVs?: Vector4; backUVs?: Vector4 }): VertexData {\r\n const indices = [];\r\n const positions = [];\r\n const normals = [];\r\n const uvs = [];\r\n\r\n const width: number = options.width || options.size || 1;\r\n const height: number = options.height || options.size || 1;\r\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\r\n\r\n // Vertices\r\n const halfWidth = width / 2.0;\r\n const halfHeight = height / 2.0;\r\n\r\n positions.push(-halfWidth, -halfHeight, 0);\r\n normals.push(0, 0, -1.0);\r\n uvs.push(0.0, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 : 0.0);\r\n\r\n positions.push(halfWidth, -halfHeight, 0);\r\n normals.push(0, 0, -1.0);\r\n uvs.push(1.0, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 : 0.0);\r\n\r\n positions.push(halfWidth, halfHeight, 0);\r\n normals.push(0, 0, -1.0);\r\n uvs.push(1.0, CompatibilityOptions.UseOpenGLOrientationForUV ? 0.0 : 1.0);\r\n\r\n positions.push(-halfWidth, halfHeight, 0);\r\n normals.push(0, 0, -1.0);\r\n uvs.push(0.0, CompatibilityOptions.UseOpenGLOrientationForUV ? 0.0 : 1.0);\r\n\r\n // Indices\r\n indices.push(0);\r\n indices.push(1);\r\n indices.push(2);\r\n\r\n indices.push(0);\r\n indices.push(2);\r\n indices.push(3);\r\n\r\n // Sides\r\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\r\n\r\n // Result\r\n const vertexData = new VertexData();\r\n\r\n vertexData.indices = indices;\r\n vertexData.positions = positions;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n\r\n return vertexData;\r\n}\r\n\r\n/**\r\n * Creates a plane mesh\r\n * * The parameter `size` sets the size (float) of both sides of the plane at once (default 1)\r\n * * You can set some different plane dimensions by using the parameters `width` and `height` (both by default have the same value of `size`)\r\n * * The parameter `sourcePlane` is a Plane instance. It builds a mesh plane from a Math plane\r\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param options.size\r\n * @param options.width\r\n * @param options.height\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param options.updatable\r\n * @param options.sourcePlane\r\n * @param scene defines the hosting scene\r\n * @returns the plane mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#plane\r\n */\r\nexport function CreatePlane(\r\n name: string,\r\n options: { size?: number; width?: number; height?: number; sideOrientation?: number; frontUVs?: Vector4; backUVs?: Vector4; updatable?: boolean; sourcePlane?: Plane } = {},\r\n scene: Nullable = null\r\n): Mesh {\r\n const plane = new Mesh(name, scene);\r\n\r\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n plane._originalBuilderSideOrientation = options.sideOrientation;\r\n\r\n const vertexData = CreatePlaneVertexData(options);\r\n\r\n vertexData.applyToMesh(plane, options.updatable);\r\n\r\n if (options.sourcePlane) {\r\n plane.translate(options.sourcePlane.normal, -options.sourcePlane.d);\r\n plane.setDirection(options.sourcePlane.normal.scale(-1));\r\n }\r\n\r\n return plane;\r\n}\r\n\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated use the function directly from the module\r\n */\r\nexport const PlaneBuilder = {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreatePlane,\r\n};\r\n\r\nVertexData.CreatePlane = CreatePlaneVertexData;\r\nMesh.CreatePlane = (name: string, size: number, scene: Scene, updatable?: boolean, sideOrientation?: number): Mesh => {\r\n const options = {\r\n size,\r\n width: size,\r\n height: size,\r\n sideOrientation,\r\n updatable,\r\n };\r\n\r\n return CreatePlane(name, options, scene);\r\n};\r\n","import type { Scene } from \"../scene\";\r\nimport { Vector3, Vector2, TmpVectors, Vector4 } from \"../Maths/math.vector\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport { Mesh } from \"../Meshes/mesh\";\r\n\r\nMesh._GroundMeshParser = (parsedMesh: any, scene: Scene): Mesh => {\r\n return GroundMesh.Parse(parsedMesh, scene);\r\n};\r\n\r\n/**\r\n * Mesh representing the ground\r\n */\r\nexport class GroundMesh extends Mesh {\r\n /** If octree should be generated */\r\n public generateOctree = false;\r\n\r\n private _heightQuads: { slope: Vector2; facet1: Vector4; facet2: Vector4 }[];\r\n\r\n /** @internal */\r\n public _subdivisionsX: number;\r\n /** @internal */\r\n public _subdivisionsY: number;\r\n /** @internal */\r\n public _width: number;\r\n /** @internal */\r\n public _height: number;\r\n /** @internal */\r\n public _minX: number;\r\n /** @internal */\r\n public _maxX: number;\r\n /** @internal */\r\n public _minZ: number;\r\n /** @internal */\r\n public _maxZ: number;\r\n\r\n constructor(name: string, scene?: Scene) {\r\n super(name, scene);\r\n }\r\n\r\n /**\r\n * \"GroundMesh\"\r\n * @returns \"GroundMesh\"\r\n */\r\n public getClassName(): string {\r\n return \"GroundMesh\";\r\n }\r\n\r\n /**\r\n * The minimum of x and y subdivisions\r\n */\r\n public get subdivisions(): number {\r\n return Math.min(this._subdivisionsX, this._subdivisionsY);\r\n }\r\n\r\n /**\r\n * X subdivisions\r\n */\r\n public get subdivisionsX(): number {\r\n return this._subdivisionsX;\r\n }\r\n\r\n /**\r\n * Y subdivisions\r\n */\r\n public get subdivisionsY(): number {\r\n return this._subdivisionsY;\r\n }\r\n\r\n /**\r\n * This function will divide the mesh into submeshes and update an octree to help to select the right submeshes\r\n * for rendering, picking and collision computations. Please note that you must have a decent number of submeshes\r\n * to get performance improvements when using an octree.\r\n * @param chunksCount the number of submeshes the mesh will be divided into\r\n * @param octreeBlocksSize the maximum size of the octree blocks (Default: 32)\r\n */\r\n public optimize(chunksCount: number, octreeBlocksSize = 32): void {\r\n this._subdivisionsX = chunksCount;\r\n this._subdivisionsY = chunksCount;\r\n this.subdivide(chunksCount);\r\n\r\n // Call the octree system optimization if it is defined.\r\n const thisAsAny = this as any;\r\n if (thisAsAny.createOrUpdateSubmeshesOctree) {\r\n thisAsAny.createOrUpdateSubmeshesOctree(octreeBlocksSize);\r\n }\r\n }\r\n\r\n /**\r\n * Returns a height (y) value in the World system :\r\n * the ground altitude at the coordinates (x, z) expressed in the World system.\r\n * @param x x coordinate\r\n * @param z z coordinate\r\n * @returns the ground y position if (x, z) are outside the ground surface.\r\n */\r\n public getHeightAtCoordinates(x: number, z: number): number {\r\n const world = this.getWorldMatrix();\r\n const invMat = TmpVectors.Matrix[5];\r\n world.invertToRef(invMat);\r\n const tmpVect = TmpVectors.Vector3[8];\r\n Vector3.TransformCoordinatesFromFloatsToRef(x, 0.0, z, invMat, tmpVect); // transform x,z in the mesh local space\r\n x = tmpVect.x;\r\n z = tmpVect.z;\r\n if (x < this._minX || x >= this._maxX || z <= this._minZ || z > this._maxZ) {\r\n return this.position.y;\r\n }\r\n if (!this._heightQuads || this._heightQuads.length == 0) {\r\n this._initHeightQuads();\r\n this._computeHeightQuads();\r\n }\r\n const facet = this._getFacetAt(x, z);\r\n const y = -(facet.x * x + facet.z * z + facet.w) / facet.y;\r\n // return y in the World system\r\n Vector3.TransformCoordinatesFromFloatsToRef(0.0, y, 0.0, world, tmpVect);\r\n return tmpVect.y;\r\n }\r\n\r\n /**\r\n * Returns a normalized vector (Vector3) orthogonal to the ground\r\n * at the ground coordinates (x, z) expressed in the World system.\r\n * @param x x coordinate\r\n * @param z z coordinate\r\n * @returns Vector3(0.0, 1.0, 0.0) if (x, z) are outside the ground surface.\r\n */\r\n public getNormalAtCoordinates(x: number, z: number): Vector3 {\r\n const normal = new Vector3(0.0, 1.0, 0.0);\r\n this.getNormalAtCoordinatesToRef(x, z, normal);\r\n return normal;\r\n }\r\n\r\n /**\r\n * Updates the Vector3 passed a reference with a normalized vector orthogonal to the ground\r\n * at the ground coordinates (x, z) expressed in the World system.\r\n * Doesn't update the reference Vector3 if (x, z) are outside the ground surface.\r\n * @param x x coordinate\r\n * @param z z coordinate\r\n * @param ref vector to store the result\r\n * @returns the GroundMesh.\r\n */\r\n public getNormalAtCoordinatesToRef(x: number, z: number, ref: Vector3): GroundMesh {\r\n const world = this.getWorldMatrix();\r\n const tmpMat = TmpVectors.Matrix[5];\r\n world.invertToRef(tmpMat);\r\n const tmpVect = TmpVectors.Vector3[8];\r\n Vector3.TransformCoordinatesFromFloatsToRef(x, 0.0, z, tmpMat, tmpVect); // transform x,z in the mesh local space\r\n x = tmpVect.x;\r\n z = tmpVect.z;\r\n if (x < this._minX || x > this._maxX || z < this._minZ || z > this._maxZ) {\r\n return this;\r\n }\r\n if (!this._heightQuads || this._heightQuads.length == 0) {\r\n this._initHeightQuads();\r\n this._computeHeightQuads();\r\n }\r\n const facet = this._getFacetAt(x, z);\r\n Vector3.TransformNormalFromFloatsToRef(facet.x, facet.y, facet.z, world, ref);\r\n return this;\r\n }\r\n\r\n /**\r\n * Force the heights to be recomputed for getHeightAtCoordinates() or getNormalAtCoordinates()\r\n * if the ground has been updated.\r\n * This can be used in the render loop.\r\n * @returns the GroundMesh.\r\n */\r\n public updateCoordinateHeights(): GroundMesh {\r\n if (!this._heightQuads || this._heightQuads.length == 0) {\r\n this._initHeightQuads();\r\n }\r\n this._computeHeightQuads();\r\n return this;\r\n }\r\n\r\n // Returns the element \"facet\" from the heightQuads array relative to (x, z) local coordinates\r\n private _getFacetAt(x: number, z: number): Vector4 {\r\n // retrieve col and row from x, z coordinates in the ground local system\r\n const col = Math.floor(((x + this._maxX) * this._subdivisionsX) / this._width);\r\n const row = Math.floor((-(z + this._maxZ) * this._subdivisionsY) / this._height + this._subdivisionsY);\r\n const quad = this._heightQuads[row * this._subdivisionsX + col];\r\n let facet;\r\n if (z < quad.slope.x * x + quad.slope.y) {\r\n facet = quad.facet1;\r\n } else {\r\n facet = quad.facet2;\r\n }\r\n return facet;\r\n }\r\n\r\n // Creates and populates the heightMap array with \"facet\" elements :\r\n // a quad is two triangular facets separated by a slope, so a \"facet\" element is 1 slope + 2 facets\r\n // slope : Vector2(c, h) = 2D diagonal line equation setting apart two triangular facets in a quad : z = cx + h\r\n // facet1 : Vector4(a, b, c, d) = first facet 3D plane equation : ax + by + cz + d = 0\r\n // facet2 : Vector4(a, b, c, d) = second facet 3D plane equation : ax + by + cz + d = 0\r\n // Returns the GroundMesh.\r\n private _initHeightQuads(): GroundMesh {\r\n const subdivisionsX = this._subdivisionsX;\r\n const subdivisionsY = this._subdivisionsY;\r\n this._heightQuads = new Array();\r\n for (let row = 0; row < subdivisionsY; row++) {\r\n for (let col = 0; col < subdivisionsX; col++) {\r\n const quad = { slope: Vector2.Zero(), facet1: new Vector4(0.0, 0.0, 0.0, 0.0), facet2: new Vector4(0.0, 0.0, 0.0, 0.0) };\r\n this._heightQuads[row * subdivisionsX + col] = quad;\r\n }\r\n }\r\n return this;\r\n }\r\n\r\n // Compute each quad element values and update the the heightMap array :\r\n // slope : Vector2(c, h) = 2D diagonal line equation setting apart two triangular facets in a quad : z = cx + h\r\n // facet1 : Vector4(a, b, c, d) = first facet 3D plane equation : ax + by + cz + d = 0\r\n // facet2 : Vector4(a, b, c, d) = second facet 3D plane equation : ax + by + cz + d = 0\r\n // Returns the GroundMesh.\r\n private _computeHeightQuads(): GroundMesh {\r\n const positions = this.getVerticesData(VertexBuffer.PositionKind);\r\n\r\n if (!positions) {\r\n return this;\r\n }\r\n\r\n const v1 = TmpVectors.Vector3[3];\r\n const v2 = TmpVectors.Vector3[2];\r\n const v3 = TmpVectors.Vector3[1];\r\n const v4 = TmpVectors.Vector3[0];\r\n const v1v2 = TmpVectors.Vector3[4];\r\n const v1v3 = TmpVectors.Vector3[5];\r\n const v1v4 = TmpVectors.Vector3[6];\r\n const norm1 = TmpVectors.Vector3[7];\r\n const norm2 = TmpVectors.Vector3[8];\r\n let i = 0;\r\n let j = 0;\r\n let k = 0;\r\n let cd = 0; // 2D slope coefficient : z = cd * x + h\r\n let h = 0;\r\n let d1 = 0; // facet plane equation : ax + by + cz + d = 0\r\n let d2 = 0;\r\n\r\n const subdivisionsX = this._subdivisionsX;\r\n const subdivisionsY = this._subdivisionsY;\r\n\r\n for (let row = 0; row < subdivisionsY; row++) {\r\n for (let col = 0; col < subdivisionsX; col++) {\r\n i = col * 3;\r\n j = row * (subdivisionsX + 1) * 3;\r\n k = (row + 1) * (subdivisionsX + 1) * 3;\r\n v1.x = positions[j + i];\r\n v1.y = positions[j + i + 1];\r\n v1.z = positions[j + i + 2];\r\n v2.x = positions[j + i + 3];\r\n v2.y = positions[j + i + 4];\r\n v2.z = positions[j + i + 5];\r\n v3.x = positions[k + i];\r\n v3.y = positions[k + i + 1];\r\n v3.z = positions[k + i + 2];\r\n v4.x = positions[k + i + 3];\r\n v4.y = positions[k + i + 4];\r\n v4.z = positions[k + i + 5];\r\n\r\n // 2D slope V1V4\r\n cd = (v4.z - v1.z) / (v4.x - v1.x);\r\n h = v1.z - cd * v1.x; // v1 belongs to the slope\r\n\r\n // facet equations :\r\n // we compute each facet normal vector\r\n // the equation of the facet plane is : norm.x * x + norm.y * y + norm.z * z + d = 0\r\n // we compute the value d by applying the equation to v1 which belongs to the plane\r\n // then we store the facet equation in a Vector4\r\n v2.subtractToRef(v1, v1v2);\r\n v3.subtractToRef(v1, v1v3);\r\n v4.subtractToRef(v1, v1v4);\r\n Vector3.CrossToRef(v1v4, v1v3, norm1); // caution : CrossToRef uses the Tmp class\r\n Vector3.CrossToRef(v1v2, v1v4, norm2);\r\n norm1.normalize();\r\n norm2.normalize();\r\n d1 = -(norm1.x * v1.x + norm1.y * v1.y + norm1.z * v1.z);\r\n d2 = -(norm2.x * v2.x + norm2.y * v2.y + norm2.z * v2.z);\r\n\r\n const quad = this._heightQuads[row * subdivisionsX + col];\r\n quad.slope.copyFromFloats(cd, h);\r\n quad.facet1.copyFromFloats(norm1.x, norm1.y, norm1.z, d1);\r\n quad.facet2.copyFromFloats(norm2.x, norm2.y, norm2.z, d2);\r\n }\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Serializes this ground mesh\r\n * @param serializationObject object to write serialization to\r\n */\r\n public serialize(serializationObject: any): void {\r\n super.serialize(serializationObject);\r\n serializationObject.subdivisionsX = this._subdivisionsX;\r\n serializationObject.subdivisionsY = this._subdivisionsY;\r\n\r\n serializationObject.minX = this._minX;\r\n serializationObject.maxX = this._maxX;\r\n\r\n serializationObject.minZ = this._minZ;\r\n serializationObject.maxZ = this._maxZ;\r\n\r\n serializationObject.width = this._width;\r\n serializationObject.height = this._height;\r\n }\r\n\r\n /**\r\n * Parses a serialized ground mesh\r\n * @param parsedMesh the serialized mesh\r\n * @param scene the scene to create the ground mesh in\r\n * @returns the created ground mesh\r\n */\r\n public static Parse(parsedMesh: any, scene: Scene): GroundMesh {\r\n const result = new GroundMesh(parsedMesh.name, scene);\r\n\r\n result._subdivisionsX = parsedMesh.subdivisionsX || 1;\r\n result._subdivisionsY = parsedMesh.subdivisionsY || 1;\r\n\r\n result._minX = parsedMesh.minX;\r\n result._maxX = parsedMesh.maxX;\r\n\r\n result._minZ = parsedMesh.minZ;\r\n result._maxZ = parsedMesh.maxZ;\r\n\r\n result._width = parsedMesh.width;\r\n result._height = parsedMesh.height;\r\n\r\n return result;\r\n }\r\n}\r\n","import type { Scene } from \"../../scene\";\r\nimport { Vector3 } from \"../../Maths/math.vector\";\r\nimport { Color3 } from \"../../Maths/math.color\";\r\nimport { Mesh } from \"../mesh\";\r\nimport { VertexData } from \"../mesh.vertexData\";\r\nimport { GroundMesh } from \"../groundMesh\";\r\nimport { Tools } from \"../../Misc/tools\";\r\nimport type { Nullable } from \"../../types\";\r\nimport { EngineStore } from \"../../Engines/engineStore\";\r\nimport { Epsilon } from \"../../Maths/math.constants\";\r\nimport { CompatibilityOptions } from \"../../Compat/compatibilityOptions\";\r\n\r\n/**\r\n * Creates the VertexData for a Ground\r\n * @param options an object used to set the following optional parameters for the Ground, required but can be empty\r\n * - width the width (x direction) of the ground, optional, default 1\r\n * - height the height (z direction) of the ground, optional, default 1\r\n * - subdivisions the number of subdivisions per side, optional, default 1\r\n * @param options.width\r\n * @param options.height\r\n * @param options.subdivisions\r\n * @param options.subdivisionsX\r\n * @param options.subdivisionsY\r\n * @returns the VertexData of the Ground\r\n */\r\nexport function CreateGroundVertexData(options: { width?: number; height?: number; subdivisions?: number; subdivisionsX?: number; subdivisionsY?: number }): VertexData {\r\n const indices = [];\r\n const positions = [];\r\n const normals = [];\r\n const uvs = [];\r\n let row: number, col: number;\r\n\r\n const width: number = options.width || 1;\r\n const height: number = options.height || 1;\r\n const subdivisionsX: number = (options.subdivisionsX || options.subdivisions || 1) | 0;\r\n const subdivisionsY: number = (options.subdivisionsY || options.subdivisions || 1) | 0;\r\n\r\n for (row = 0; row <= subdivisionsY; row++) {\r\n for (col = 0; col <= subdivisionsX; col++) {\r\n const position = new Vector3((col * width) / subdivisionsX - width / 2.0, 0, ((subdivisionsY - row) * height) / subdivisionsY - height / 2.0);\r\n const normal = new Vector3(0, 1.0, 0);\r\n\r\n positions.push(position.x, position.y, position.z);\r\n normals.push(normal.x, normal.y, normal.z);\r\n uvs.push(col / subdivisionsX, CompatibilityOptions.UseOpenGLOrientationForUV ? row / subdivisionsY : 1.0 - row / subdivisionsY);\r\n }\r\n }\r\n\r\n for (row = 0; row < subdivisionsY; row++) {\r\n for (col = 0; col < subdivisionsX; col++) {\r\n indices.push(col + 1 + (row + 1) * (subdivisionsX + 1));\r\n indices.push(col + 1 + row * (subdivisionsX + 1));\r\n indices.push(col + row * (subdivisionsX + 1));\r\n\r\n indices.push(col + (row + 1) * (subdivisionsX + 1));\r\n indices.push(col + 1 + (row + 1) * (subdivisionsX + 1));\r\n indices.push(col + row * (subdivisionsX + 1));\r\n }\r\n }\r\n\r\n // Result\r\n const vertexData = new VertexData();\r\n\r\n vertexData.indices = indices;\r\n vertexData.positions = positions;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n\r\n return vertexData;\r\n}\r\n\r\n/**\r\n * Creates the VertexData for a TiledGround by subdividing the ground into tiles\r\n * @param options an object used to set the following optional parameters for the Ground, required but can be empty\r\n * * xmin the ground minimum X coordinate, optional, default -1\r\n * * zmin the ground minimum Z coordinate, optional, default -1\r\n * * xmax the ground maximum X coordinate, optional, default 1\r\n * * zmax the ground maximum Z coordinate, optional, default 1\r\n * * subdivisions a javascript object {w: positive integer, h: positive integer}, `w` and `h` are the numbers of subdivisions on the ground width and height creating 'tiles', default {w: 6, h: 6}\r\n * * precision a javascript object {w: positive integer, h: positive integer}, `w` and `h` are the numbers of subdivisions on the tile width and height, default {w: 2, h: 2}\r\n * @param options.xmin\r\n * @param options.zmin\r\n * @param options.xmax\r\n * @param options.zmax\r\n * @param options.subdivisions\r\n * @param options.subdivisions.w\r\n * @param options.subdivisions.h\r\n * @param options.precision\r\n * @param options.precision.w\r\n * @param options.precision.h\r\n * @returns the VertexData of the TiledGround\r\n */\r\nexport function CreateTiledGroundVertexData(options: {\r\n xmin: number;\r\n zmin: number;\r\n xmax: number;\r\n zmax: number;\r\n subdivisions?: { w: number; h: number };\r\n precision?: { w: number; h: number };\r\n}): VertexData {\r\n const xmin = options.xmin !== undefined && options.xmin !== null ? options.xmin : -1.0;\r\n const zmin = options.zmin !== undefined && options.zmin !== null ? options.zmin : -1.0;\r\n const xmax = options.xmax !== undefined && options.xmax !== null ? options.xmax : 1.0;\r\n const zmax = options.zmax !== undefined && options.zmax !== null ? options.zmax : 1.0;\r\n const subdivisions = options.subdivisions || { w: 1, h: 1 };\r\n const precision = options.precision || { w: 1, h: 1 };\r\n\r\n const indices = new Array();\r\n const positions = new Array();\r\n const normals = new Array();\r\n const uvs = new Array();\r\n let row: number, col: number, tileRow: number, tileCol: number;\r\n\r\n subdivisions.h = subdivisions.h < 1 ? 1 : subdivisions.h;\r\n subdivisions.w = subdivisions.w < 1 ? 1 : subdivisions.w;\r\n precision.w = precision.w < 1 ? 1 : precision.w;\r\n precision.h = precision.h < 1 ? 1 : precision.h;\r\n\r\n const tileSize = {\r\n w: (xmax - xmin) / subdivisions.w,\r\n h: (zmax - zmin) / subdivisions.h,\r\n };\r\n\r\n function applyTile(xTileMin: number, zTileMin: number, xTileMax: number, zTileMax: number) {\r\n // Indices\r\n const base = positions.length / 3;\r\n const rowLength = precision.w + 1;\r\n for (row = 0; row < precision.h; row++) {\r\n for (col = 0; col < precision.w; col++) {\r\n const square = [base + col + row * rowLength, base + (col + 1) + row * rowLength, base + (col + 1) + (row + 1) * rowLength, base + col + (row + 1) * rowLength];\r\n\r\n indices.push(square[1]);\r\n indices.push(square[2]);\r\n indices.push(square[3]);\r\n indices.push(square[0]);\r\n indices.push(square[1]);\r\n indices.push(square[3]);\r\n }\r\n }\r\n\r\n // Position, normals and uvs\r\n const position = Vector3.Zero();\r\n const normal = new Vector3(0, 1.0, 0);\r\n for (row = 0; row <= precision.h; row++) {\r\n position.z = (row * (zTileMax - zTileMin)) / precision.h + zTileMin;\r\n for (col = 0; col <= precision.w; col++) {\r\n position.x = (col * (xTileMax - xTileMin)) / precision.w + xTileMin;\r\n position.y = 0;\r\n\r\n positions.push(position.x, position.y, position.z);\r\n normals.push(normal.x, normal.y, normal.z);\r\n uvs.push(col / precision.w, row / precision.h);\r\n }\r\n }\r\n }\r\n\r\n for (tileRow = 0; tileRow < subdivisions.h; tileRow++) {\r\n for (tileCol = 0; tileCol < subdivisions.w; tileCol++) {\r\n applyTile(xmin + tileCol * tileSize.w, zmin + tileRow * tileSize.h, xmin + (tileCol + 1) * tileSize.w, zmin + (tileRow + 1) * tileSize.h);\r\n }\r\n }\r\n\r\n // Result\r\n const vertexData = new VertexData();\r\n\r\n vertexData.indices = indices;\r\n vertexData.positions = positions;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n\r\n return vertexData;\r\n}\r\n\r\n/**\r\n * Creates the VertexData of the Ground designed from a heightmap\r\n * @param options an object used to set the following parameters for the Ground, required and provided by CreateGroundFromHeightMap\r\n * * width the width (x direction) of the ground\r\n * * height the height (z direction) of the ground\r\n * * subdivisions the number of subdivisions per side\r\n * * minHeight the minimum altitude on the ground, optional, default 0\r\n * * maxHeight the maximum altitude on the ground, optional default 1\r\n * * colorFilter the filter to apply to the image pixel colors to compute the height, optional Color3, default (0.3, 0.59, 0.11)\r\n * * buffer the array holding the image color data\r\n * * bufferWidth the width of image\r\n * * bufferHeight the height of image\r\n * * alphaFilter Remove any data where the alpha channel is below this value, defaults 0 (all data visible)\r\n * @param options.width\r\n * @param options.height\r\n * @param options.subdivisions\r\n * @param options.minHeight\r\n * @param options.maxHeight\r\n * @param options.colorFilter\r\n * @param options.buffer\r\n * @param options.bufferWidth\r\n * @param options.bufferHeight\r\n * @param options.alphaFilter\r\n * @returns the VertexData of the Ground designed from a heightmap\r\n */\r\nexport function CreateGroundFromHeightMapVertexData(options: {\r\n width: number;\r\n height: number;\r\n subdivisions: number;\r\n minHeight: number;\r\n maxHeight: number;\r\n colorFilter: Color3;\r\n buffer: Uint8Array;\r\n bufferWidth: number;\r\n bufferHeight: number;\r\n alphaFilter: number;\r\n}): VertexData {\r\n const indices = [];\r\n const positions = [];\r\n const normals = [];\r\n const uvs = [];\r\n let row, col;\r\n const filter = options.colorFilter || new Color3(0.3, 0.59, 0.11);\r\n const alphaFilter = options.alphaFilter || 0.0;\r\n let invert = false;\r\n\r\n if (options.minHeight > options.maxHeight) {\r\n invert = true;\r\n const temp = options.maxHeight;\r\n options.maxHeight = options.minHeight;\r\n options.minHeight = temp;\r\n }\r\n\r\n // Vertices\r\n for (row = 0; row <= options.subdivisions; row++) {\r\n for (col = 0; col <= options.subdivisions; col++) {\r\n const position = new Vector3(\r\n (col * options.width) / options.subdivisions - options.width / 2.0,\r\n 0,\r\n ((options.subdivisions - row) * options.height) / options.subdivisions - options.height / 2.0\r\n );\r\n\r\n // Compute height\r\n const heightMapX = (((position.x + options.width / 2) / options.width) * (options.bufferWidth - 1)) | 0;\r\n const heightMapY = ((1.0 - (position.z + options.height / 2) / options.height) * (options.bufferHeight - 1)) | 0;\r\n\r\n const pos = (heightMapX + heightMapY * options.bufferWidth) * 4;\r\n let r = options.buffer[pos] / 255.0;\r\n let g = options.buffer[pos + 1] / 255.0;\r\n let b = options.buffer[pos + 2] / 255.0;\r\n const a = options.buffer[pos + 3] / 255.0;\r\n\r\n if (invert) {\r\n r = 1.0 - r;\r\n g = 1.0 - g;\r\n b = 1.0 - b;\r\n }\r\n\r\n const gradient = r * filter.r + g * filter.g + b * filter.b;\r\n\r\n // If our alpha channel is not within our filter then we will assign a 'special' height\r\n // Then when building the indices, we will ignore any vertex that is using the special height\r\n if (a >= alphaFilter) {\r\n position.y = options.minHeight + (options.maxHeight - options.minHeight) * gradient;\r\n } else {\r\n position.y = options.minHeight - Epsilon; // We can't have a height below minHeight, normally.\r\n }\r\n\r\n // Add vertex\r\n positions.push(position.x, position.y, position.z);\r\n normals.push(0, 0, 0);\r\n uvs.push(col / options.subdivisions, 1.0 - row / options.subdivisions);\r\n }\r\n }\r\n\r\n // Indices\r\n for (row = 0; row < options.subdivisions; row++) {\r\n for (col = 0; col < options.subdivisions; col++) {\r\n // Calculate Indices\r\n const idx1 = col + 1 + (row + 1) * (options.subdivisions + 1);\r\n const idx2 = col + 1 + row * (options.subdivisions + 1);\r\n const idx3 = col + row * (options.subdivisions + 1);\r\n const idx4 = col + (row + 1) * (options.subdivisions + 1);\r\n\r\n // Check that all indices are visible (based on our special height)\r\n // Only display the vertex if all Indices are visible\r\n // Positions are stored x,y,z for each vertex, hence the * 3 and + 1 for height\r\n const isVisibleIdx1 = positions[idx1 * 3 + 1] >= options.minHeight;\r\n const isVisibleIdx2 = positions[idx2 * 3 + 1] >= options.minHeight;\r\n const isVisibleIdx3 = positions[idx3 * 3 + 1] >= options.minHeight;\r\n if (isVisibleIdx1 && isVisibleIdx2 && isVisibleIdx3) {\r\n indices.push(idx1);\r\n indices.push(idx2);\r\n indices.push(idx3);\r\n }\r\n\r\n const isVisibleIdx4 = positions[idx4 * 3 + 1] >= options.minHeight;\r\n if (isVisibleIdx4 && isVisibleIdx1 && isVisibleIdx3) {\r\n indices.push(idx4);\r\n indices.push(idx1);\r\n indices.push(idx3);\r\n }\r\n }\r\n }\r\n\r\n // Normals\r\n VertexData.ComputeNormals(positions, indices, normals);\r\n\r\n // Result\r\n const vertexData = new VertexData();\r\n\r\n vertexData.indices = indices;\r\n vertexData.positions = positions;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n\r\n return vertexData;\r\n}\r\n\r\n/**\r\n * Creates a ground mesh\r\n * * The parameters `width` and `height` (floats, default 1) set the width and height sizes of the ground\r\n * * The parameter `subdivisions` (positive integer) sets the number of subdivisions per side\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param options.width\r\n * @param options.height\r\n * @param options.subdivisions\r\n * @param options.subdivisionsX\r\n * @param options.subdivisionsY\r\n * @param options.updatable\r\n * @param scene defines the hosting scene\r\n * @returns the ground mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#ground\r\n */\r\nexport function CreateGround(\r\n name: string,\r\n options: { width?: number; height?: number; subdivisions?: number; subdivisionsX?: number; subdivisionsY?: number; updatable?: boolean } = {},\r\n scene?: Scene\r\n): GroundMesh {\r\n const ground = new GroundMesh(name, scene);\r\n ground._setReady(false);\r\n ground._subdivisionsX = options.subdivisionsX || options.subdivisions || 1;\r\n ground._subdivisionsY = options.subdivisionsY || options.subdivisions || 1;\r\n ground._width = options.width || 1;\r\n ground._height = options.height || 1;\r\n ground._maxX = ground._width / 2;\r\n ground._maxZ = ground._height / 2;\r\n ground._minX = -ground._maxX;\r\n ground._minZ = -ground._maxZ;\r\n\r\n const vertexData = CreateGroundVertexData(options);\r\n\r\n vertexData.applyToMesh(ground, options.updatable);\r\n\r\n ground._setReady(true);\r\n\r\n return ground;\r\n}\r\n\r\n/**\r\n * Creates a tiled ground mesh\r\n * * The parameters `xmin` and `xmax` (floats, default -1 and 1) set the ground minimum and maximum X coordinates\r\n * * The parameters `zmin` and `zmax` (floats, default -1 and 1) set the ground minimum and maximum Z coordinates\r\n * * The parameter `subdivisions` is a javascript object `{w: positive integer, h: positive integer}` (default `{w: 6, h: 6}`). `w` and `h` are the numbers of subdivisions on the ground width and height. Each subdivision is called a tile\r\n * * The parameter `precision` is a javascript object `{w: positive integer, h: positive integer}` (default `{w: 2, h: 2}`). `w` and `h` are the numbers of subdivisions on the ground width and height of each tile\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param options.xmin\r\n * @param options.zmin\r\n * @param options.xmax\r\n * @param options.zmax\r\n * @param options.subdivisions\r\n * @param options.subdivisions.w\r\n * @param options.subdivisions.h\r\n * @param options.precision\r\n * @param options.precision.w\r\n * @param options.precision.h\r\n * @param options.updatable\r\n * @param scene defines the hosting scene\r\n * @returns the tiled ground mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#tiled-ground\r\n */\r\nexport function CreateTiledGround(\r\n name: string,\r\n options: { xmin: number; zmin: number; xmax: number; zmax: number; subdivisions?: { w: number; h: number }; precision?: { w: number; h: number }; updatable?: boolean },\r\n scene: Nullable = null\r\n): Mesh {\r\n const tiledGround = new Mesh(name, scene);\r\n\r\n const vertexData = CreateTiledGroundVertexData(options);\r\n\r\n vertexData.applyToMesh(tiledGround, options.updatable);\r\n\r\n return tiledGround;\r\n}\r\n\r\n/**\r\n * Creates a ground mesh from a height map\r\n * * The parameter `url` sets the URL of the height map image resource.\r\n * * The parameters `width` and `height` (positive floats, default 10) set the ground width and height sizes.\r\n * * The parameter `subdivisions` (positive integer, default 1) sets the number of subdivision per side.\r\n * * The parameter `minHeight` (float, default 0) is the minimum altitude on the ground.\r\n * * The parameter `maxHeight` (float, default 1) is the maximum altitude on the ground.\r\n * * The parameter `colorFilter` (optional Color3, default (0.3, 0.59, 0.11) ) is the filter to apply to the image pixel colors to compute the height.\r\n * * The parameter `onReady` is a javascript callback function that will be called once the mesh is just built (the height map download can last some time).\r\n * * The parameter `alphaFilter` will filter any data where the alpha channel is below this value, defaults 0 (all data visible)\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\r\n * @param name defines the name of the mesh\r\n * @param url defines the url to the height map\r\n * @param options defines the options used to create the mesh\r\n * @param options.width\r\n * @param options.height\r\n * @param options.subdivisions\r\n * @param options.minHeight\r\n * @param options.maxHeight\r\n * @param options.colorFilter\r\n * @param options.alphaFilter\r\n * @param options.updatable\r\n * @param options.onReady\r\n * @param scene defines the hosting scene\r\n * @returns the ground mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set/height_map\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#ground-from-a-height-map\r\n */\r\nexport function CreateGroundFromHeightMap(\r\n name: string,\r\n url: string,\r\n options: {\r\n width?: number;\r\n height?: number;\r\n subdivisions?: number;\r\n minHeight?: number;\r\n maxHeight?: number;\r\n colorFilter?: Color3;\r\n alphaFilter?: number;\r\n updatable?: boolean;\r\n onReady?: (mesh: GroundMesh) => void;\r\n } = {},\r\n scene: Nullable = null\r\n): GroundMesh {\r\n const width = options.width || 10.0;\r\n const height = options.height || 10.0;\r\n const subdivisions = options.subdivisions || 1 | 0;\r\n const minHeight = options.minHeight || 0.0;\r\n const maxHeight = options.maxHeight || 1.0;\r\n const filter = options.colorFilter || new Color3(0.3, 0.59, 0.11);\r\n const alphaFilter = options.alphaFilter || 0.0;\r\n const updatable = options.updatable;\r\n const onReady = options.onReady;\r\n\r\n scene = scene || EngineStore.LastCreatedScene!;\r\n\r\n const ground = new GroundMesh(name, scene);\r\n ground._subdivisionsX = subdivisions;\r\n ground._subdivisionsY = subdivisions;\r\n ground._width = width;\r\n ground._height = height;\r\n ground._maxX = ground._width / 2.0;\r\n ground._maxZ = ground._height / 2.0;\r\n ground._minX = -ground._maxX;\r\n ground._minZ = -ground._maxZ;\r\n\r\n ground._setReady(false);\r\n\r\n const onload = (img: HTMLImageElement | ImageBitmap) => {\r\n const bufferWidth = img.width;\r\n const bufferHeight = img.height;\r\n\r\n if (scene!.isDisposed) {\r\n return;\r\n }\r\n\r\n const buffer = scene?.getEngine().resizeImageBitmap(img, bufferWidth, bufferHeight);\r\n\r\n const vertexData = CreateGroundFromHeightMapVertexData({\r\n width: width,\r\n height: height,\r\n subdivisions: subdivisions,\r\n minHeight: minHeight,\r\n maxHeight: maxHeight,\r\n colorFilter: filter,\r\n buffer: buffer,\r\n bufferWidth: bufferWidth,\r\n bufferHeight: bufferHeight,\r\n alphaFilter: alphaFilter,\r\n });\r\n\r\n vertexData.applyToMesh(ground, updatable);\r\n\r\n //execute ready callback, if set\r\n if (onReady) {\r\n onReady(ground);\r\n }\r\n\r\n ground._setReady(true);\r\n };\r\n\r\n Tools.LoadImage(url, onload, () => {}, scene.offlineProvider);\r\n\r\n return ground;\r\n}\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated use the functions directly from the module\r\n */\r\nexport const GroundBuilder = {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreateGround,\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreateGroundFromHeightMap,\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreateTiledGround,\r\n};\r\n\r\nVertexData.CreateGround = CreateGroundVertexData;\r\nVertexData.CreateTiledGround = CreateTiledGroundVertexData;\r\nVertexData.CreateGroundFromHeightMap = CreateGroundFromHeightMapVertexData;\r\n\r\nMesh.CreateGround = (name: string, width: number, height: number, subdivisions: number, scene?: Scene, updatable?: boolean): Mesh => {\r\n const options = {\r\n width,\r\n height,\r\n subdivisions,\r\n updatable,\r\n };\r\n\r\n return CreateGround(name, options, scene);\r\n};\r\n\r\nMesh.CreateTiledGround = (\r\n name: string,\r\n xmin: number,\r\n zmin: number,\r\n xmax: number,\r\n zmax: number,\r\n subdivisions: { w: number; h: number },\r\n precision: { w: number; h: number },\r\n scene: Scene,\r\n updatable?: boolean\r\n): Mesh => {\r\n const options = {\r\n xmin,\r\n zmin,\r\n xmax,\r\n zmax,\r\n subdivisions,\r\n precision,\r\n updatable,\r\n };\r\n\r\n return CreateTiledGround(name, options, scene);\r\n};\r\n\r\nMesh.CreateGroundFromHeightMap = (\r\n name: string,\r\n url: string,\r\n width: number,\r\n height: number,\r\n subdivisions: number,\r\n minHeight: number,\r\n maxHeight: number,\r\n scene: Scene,\r\n updatable?: boolean,\r\n onReady?: (mesh: GroundMesh) => void,\r\n alphaFilter?: number\r\n): GroundMesh => {\r\n const options = {\r\n width,\r\n height,\r\n subdivisions,\r\n minHeight,\r\n maxHeight,\r\n updatable,\r\n onReady,\r\n alphaFilter,\r\n };\r\n\r\n return CreateGroundFromHeightMap(name, url, options, scene);\r\n};\r\n","import type { Nullable } from \"../../types\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { Matrix, Vector4 } from \"../../Maths/math.vector\";\r\nimport { Color4 } from \"../../Maths/math.color\";\r\nimport { Mesh } from \"../mesh\";\r\nimport { VertexData } from \"../mesh.vertexData\";\r\nimport { CompatibilityOptions } from \"../../Compat/compatibilityOptions\";\r\nimport { CreateGroundVertexData } from \"./groundBuilder\";\r\n\r\n/**\r\n * Creates the VertexData for a box\r\n * @param options an object used to set the following optional parameters for the box, required but can be empty\r\n * * size sets the width, height and depth of the box to the value of size, optional default 1\r\n * * width sets the width (x direction) of the box, overwrites the width set by size, optional, default size\r\n * * height sets the height (y direction) of the box, overwrites the height set by size, optional, default size\r\n * * depth sets the depth (z direction) of the box, overwrites the depth set by size, optional, default size\r\n * * faceUV an array of 6 Vector4 elements used to set different images to each box side\r\n * * faceColors an array of 6 Color3 elements used to set different colors to each box side\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.size\r\n * @param options.width\r\n * @param options.height\r\n * @param options.depth\r\n * @param options.faceUV\r\n * @param options.faceColors\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param options.wrap\r\n * @param options.topBaseAt\r\n * @param options.bottomBaseAt\r\n * @returns the VertexData of the box\r\n */\r\nexport function CreateBoxVertexData(options: {\r\n size?: number;\r\n width?: number;\r\n height?: number;\r\n depth?: number;\r\n faceUV?: Vector4[];\r\n faceColors?: Color4[];\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n wrap?: boolean;\r\n topBaseAt?: number;\r\n bottomBaseAt?: number;\r\n}): VertexData {\r\n const nbFaces = 6;\r\n let indices = [0, 1, 2, 0, 2, 3, 4, 5, 6, 4, 6, 7, 8, 9, 10, 8, 10, 11, 12, 13, 14, 12, 14, 15, 16, 17, 18, 16, 18, 19, 20, 21, 22, 20, 22, 23];\r\n const normals = [\r\n 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, 0, 1, 0, 0, 1, 0, 0,\r\n 1, 0, 0, 1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0,\r\n ];\r\n const uvs = [];\r\n let positions = [];\r\n const width = options.width || options.size || 1;\r\n const height = options.height || options.size || 1;\r\n const depth = options.depth || options.size || 1;\r\n const wrap = options.wrap || false;\r\n let topBaseAt = options.topBaseAt === void 0 ? 1 : options.topBaseAt;\r\n let bottomBaseAt = options.bottomBaseAt === void 0 ? 0 : options.bottomBaseAt;\r\n topBaseAt = (topBaseAt + 4) % 4; // places values as 0 to 3\r\n bottomBaseAt = (bottomBaseAt + 4) % 4; // places values as 0 to 3\r\n const topOrder = [2, 0, 3, 1];\r\n const bottomOrder = [2, 0, 1, 3];\r\n let topIndex = topOrder[topBaseAt];\r\n let bottomIndex = bottomOrder[bottomBaseAt];\r\n let basePositions = [\r\n 1, -1, 1, -1, -1, 1, -1, 1, 1, 1, 1, 1, 1, 1, -1, -1, 1, -1, -1, -1, -1, 1, -1, -1, 1, 1, -1, 1, -1, -1, 1, -1, 1, 1, 1, 1, -1, 1, 1, -1, -1, 1, -1, -1, -1, -1, 1, -1, -1,\r\n 1, 1, -1, 1, -1, 1, 1, -1, 1, 1, 1, 1, -1, 1, 1, -1, -1, -1, -1, -1, -1, -1, 1,\r\n ];\r\n if (wrap) {\r\n indices = [2, 3, 0, 2, 0, 1, 4, 5, 6, 4, 6, 7, 9, 10, 11, 9, 11, 8, 12, 14, 15, 12, 13, 14];\r\n basePositions = [\r\n -1, 1, 1, 1, 1, 1, 1, -1, 1, -1, -1, 1, 1, 1, -1, -1, 1, -1, -1, -1, -1, 1, -1, -1, 1, 1, 1, 1, 1, -1, 1, -1, -1, 1, -1, 1, -1, 1, -1, -1, 1, 1, -1, -1, 1, -1, -1, -1,\r\n ];\r\n let topFaceBase: any = [\r\n [1, 1, 1],\r\n [-1, 1, 1],\r\n [-1, 1, -1],\r\n [1, 1, -1],\r\n ];\r\n let bottomFaceBase: any = [\r\n [-1, -1, 1],\r\n [1, -1, 1],\r\n [1, -1, -1],\r\n [-1, -1, -1],\r\n ];\r\n const topFaceOrder: any = [17, 18, 19, 16];\r\n const bottomFaceOrder: any = [22, 23, 20, 21];\r\n while (topIndex > 0) {\r\n topFaceBase.unshift(topFaceBase.pop());\r\n topFaceOrder.unshift(topFaceOrder.pop());\r\n topIndex--;\r\n }\r\n while (bottomIndex > 0) {\r\n bottomFaceBase.unshift(bottomFaceBase.pop());\r\n bottomFaceOrder.unshift(bottomFaceOrder.pop());\r\n bottomIndex--;\r\n }\r\n topFaceBase = topFaceBase.flat();\r\n bottomFaceBase = bottomFaceBase.flat();\r\n basePositions = basePositions.concat(topFaceBase).concat(bottomFaceBase);\r\n indices.push(topFaceOrder[0], topFaceOrder[2], topFaceOrder[3], topFaceOrder[0], topFaceOrder[1], topFaceOrder[2]);\r\n indices.push(bottomFaceOrder[0], bottomFaceOrder[2], bottomFaceOrder[3], bottomFaceOrder[0], bottomFaceOrder[1], bottomFaceOrder[2]);\r\n }\r\n const scaleArray = [width / 2, height / 2, depth / 2];\r\n positions = basePositions.reduce((accumulator: Array, currentValue, currentIndex) => accumulator.concat(currentValue * scaleArray[currentIndex % 3]), []);\r\n\r\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\r\n\r\n const faceUV: Vector4[] = options.faceUV || new Array(6);\r\n const faceColors = options.faceColors;\r\n const colors = [];\r\n\r\n // default face colors and UV if undefined\r\n for (let f = 0; f < 6; f++) {\r\n if (faceUV[f] === undefined) {\r\n faceUV[f] = new Vector4(0, 0, 1, 1);\r\n }\r\n if (faceColors && faceColors[f] === undefined) {\r\n faceColors[f] = new Color4(1, 1, 1, 1);\r\n }\r\n }\r\n\r\n // Create each face in turn.\r\n for (let index = 0; index < nbFaces; index++) {\r\n uvs.push(faceUV[index].z, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - faceUV[index].w : faceUV[index].w);\r\n uvs.push(faceUV[index].x, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - faceUV[index].w : faceUV[index].w);\r\n uvs.push(faceUV[index].x, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - faceUV[index].y : faceUV[index].y);\r\n uvs.push(faceUV[index].z, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - faceUV[index].y : faceUV[index].y);\r\n if (faceColors) {\r\n for (let c = 0; c < 4; c++) {\r\n colors.push(faceColors[index].r, faceColors[index].g, faceColors[index].b, faceColors[index].a);\r\n }\r\n }\r\n }\r\n\r\n // sides\r\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\r\n\r\n // Result\r\n const vertexData = new VertexData();\r\n\r\n vertexData.indices = indices;\r\n vertexData.positions = positions;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n\r\n if (faceColors) {\r\n const totalColors = sideOrientation === VertexData.DOUBLESIDE ? colors.concat(colors) : colors;\r\n vertexData.colors = totalColors;\r\n }\r\n\r\n return vertexData;\r\n}\r\n\r\n/**\r\n * Creates the VertexData for a segmented box\r\n * @param options an object used to set the following optional parameters for the box, required but can be empty\r\n * * size sets the width, height and depth of the box to the value of size, optional default 1\r\n * * width sets the width (x direction) of the box, overwrites the width set by size, optional, default size\r\n * * height sets the height (y direction) of the box, overwrites the height set by size, optional, default size\r\n * * depth sets the depth (z direction) of the box, overwrites the depth set by size, optional, default size\r\n * * segments sets the number of segments on the all axis (1 by default)\r\n * * widthSegments sets the number of segments on the x axis (1 by default)\r\n * * heightSegments sets the number of segments on the y axis (1 by default)\r\n * * depthSegments sets the number of segments on the z axis (1 by default)\r\n * @param options.size\r\n * @param options.width\r\n * @param options.height\r\n * @param options.depth\r\n * @param options.segments\r\n * @param options.widthSegments\r\n * @param options.heightSegments\r\n * @param options.depthSegments\r\n * @returns the VertexData of the box\r\n */\r\nexport function CreateSegmentedBoxVertexData(options: {\r\n size?: number;\r\n width?: number;\r\n height?: number;\r\n depth?: number;\r\n segments?: number;\r\n widthSegments?: number;\r\n heightSegments?: number;\r\n depthSegments?: number;\r\n}): VertexData {\r\n const width = options.width || options.size || 1;\r\n const height = options.height || options.size || 1;\r\n const depth = options.depth || options.size || 1;\r\n const widthSegments = (options.widthSegments || options.segments || 1) | 0;\r\n const heightSegments = (options.heightSegments || options.segments || 1) | 0;\r\n const depthSegments = (options.depthSegments || options.segments || 1) | 0;\r\n const rotationMatrix = new Matrix();\r\n const translationMatrix = new Matrix();\r\n const transformMatrix = new Matrix();\r\n\r\n const bottomPlane = CreateGroundVertexData({ width: width, height: depth, subdivisionsX: widthSegments, subdivisionsY: depthSegments });\r\n Matrix.TranslationToRef(0, -height / 2, 0, translationMatrix);\r\n Matrix.RotationZToRef(Math.PI, rotationMatrix);\r\n rotationMatrix.multiplyToRef(translationMatrix, transformMatrix);\r\n bottomPlane.transform(transformMatrix);\r\n\r\n const topPlane = CreateGroundVertexData({ width: width, height: depth, subdivisionsX: widthSegments, subdivisionsY: depthSegments });\r\n Matrix.TranslationToRef(0, height / 2, 0, transformMatrix);\r\n topPlane.transform(transformMatrix);\r\n\r\n const negXPlane = CreateGroundVertexData({ width: height, height: depth, subdivisionsX: heightSegments, subdivisionsY: depthSegments });\r\n Matrix.TranslationToRef(-width / 2, 0, 0, translationMatrix);\r\n Matrix.RotationZToRef(Math.PI / 2, rotationMatrix);\r\n rotationMatrix.multiplyToRef(translationMatrix, transformMatrix);\r\n negXPlane.transform(transformMatrix);\r\n\r\n const posXPlane = CreateGroundVertexData({ width: height, height: depth, subdivisionsX: heightSegments, subdivisionsY: depthSegments });\r\n Matrix.TranslationToRef(width / 2, 0, 0, translationMatrix);\r\n Matrix.RotationZToRef(-Math.PI / 2, rotationMatrix);\r\n rotationMatrix.multiplyToRef(translationMatrix, transformMatrix);\r\n posXPlane.transform(transformMatrix);\r\n\r\n const negZPlane = CreateGroundVertexData({ width: width, height: height, subdivisionsX: widthSegments, subdivisionsY: heightSegments });\r\n Matrix.TranslationToRef(0, 0, -depth / 2, translationMatrix);\r\n Matrix.RotationXToRef(-Math.PI / 2, rotationMatrix);\r\n rotationMatrix.multiplyToRef(translationMatrix, transformMatrix);\r\n negZPlane.transform(transformMatrix);\r\n\r\n const posZPlane = CreateGroundVertexData({ width: width, height: height, subdivisionsX: widthSegments, subdivisionsY: heightSegments });\r\n Matrix.TranslationToRef(0, 0, depth / 2, translationMatrix);\r\n Matrix.RotationXToRef(Math.PI / 2, rotationMatrix);\r\n rotationMatrix.multiplyToRef(translationMatrix, transformMatrix);\r\n posZPlane.transform(transformMatrix);\r\n\r\n // Result\r\n bottomPlane.merge([topPlane, posXPlane, negXPlane, negZPlane, posZPlane], true);\r\n\r\n return bottomPlane;\r\n}\r\n\r\n/**\r\n * Creates a box mesh\r\n * * The parameter `size` sets the size (float) of each box side (default 1)\r\n * * You can set some different box dimensions by using the parameters `width`, `height` and `depth` (all by default have the same value of `size`)\r\n * * You can set different colors and different images to each box side by using the parameters `faceColors` (an array of 6 Color3 elements) and `faceUV` (an array of 6 Vector4 elements)\r\n * * Please read this tutorial : https://doc.babylonjs.com/features/featuresDeepDive/materials/using/texturePerBoxFace\r\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#box\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param options.size\r\n * @param options.width\r\n * @param options.height\r\n * @param options.depth\r\n * @param options.faceUV\r\n * @param options.faceColors\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param options.wrap\r\n * @param options.topBaseAt\r\n * @param options.bottomBaseAt\r\n * @param options.updatable\r\n * @param scene defines the hosting scene\r\n * @returns the box mesh\r\n */\r\nexport function CreateBox(\r\n name: string,\r\n options: {\r\n size?: number;\r\n width?: number;\r\n height?: number;\r\n depth?: number;\r\n faceUV?: Vector4[];\r\n faceColors?: Color4[];\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n wrap?: boolean;\r\n topBaseAt?: number;\r\n bottomBaseAt?: number;\r\n updatable?: boolean;\r\n } = {},\r\n scene: Nullable = null\r\n): Mesh {\r\n const box = new Mesh(name, scene);\r\n\r\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n box._originalBuilderSideOrientation = options.sideOrientation;\r\n\r\n const vertexData = CreateBoxVertexData(options);\r\n\r\n vertexData.applyToMesh(box, options.updatable);\r\n\r\n return box;\r\n}\r\n\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated please use CreateBox directly\r\n */\r\nexport const BoxBuilder = {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreateBox,\r\n};\r\n\r\n// Side effects\r\nVertexData.CreateBox = CreateBoxVertexData;\r\n\r\nMesh.CreateBox = (name: string, size: number, scene: Nullable = null, updatable?: boolean, sideOrientation?: number): Mesh => {\r\n const options = {\r\n size,\r\n sideOrientation,\r\n updatable,\r\n };\r\n\r\n return CreateBox(name, options, scene);\r\n};\r\n","import { Observable } from \"../Misc/observable\";\r\nimport type { Nullable } from \"../types\";\r\nimport { ArcRotateCamera } from \"../Cameras/arcRotateCamera\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Vector3 } from \"../Maths/math.vector\";\r\nimport { Color3, Color4 } from \"../Maths/math.color\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport { Mesh } from \"../Meshes/mesh\";\r\nimport { BaseTexture } from \"../Materials/Textures/baseTexture\";\r\nimport { Texture } from \"../Materials/Textures/texture\";\r\nimport { MirrorTexture } from \"../Materials/Textures/mirrorTexture\";\r\nimport { CubeTexture } from \"../Materials/Textures/cubeTexture\";\r\nimport { BackgroundMaterial } from \"../Materials/Background/backgroundMaterial\";\r\nimport { Constants } from \"../Engines/constants\";\r\n\r\nimport { CreatePlane } from \"../Meshes/Builders/planeBuilder\";\r\nimport { CreateBox } from \"../Meshes/Builders/boxBuilder\";\r\nimport { Plane } from \"../Maths/math.plane\";\r\n\r\n/**\r\n * Represents the different options available during the creation of\r\n * a Environment helper.\r\n *\r\n * This can control the default ground, skybox and image processing setup of your scene.\r\n */\r\nexport interface IEnvironmentHelperOptions {\r\n /**\r\n * Specifies whether or not to create a ground.\r\n * True by default.\r\n */\r\n createGround: boolean;\r\n /**\r\n * Specifies the ground size.\r\n * 15 by default.\r\n */\r\n groundSize: number;\r\n /**\r\n * The texture used on the ground for the main color.\r\n * Comes from the BabylonJS CDN by default.\r\n *\r\n * Remarks: Can be either a texture or a url.\r\n */\r\n groundTexture: string | BaseTexture;\r\n /**\r\n * The color mixed in the ground texture by default.\r\n * BabylonJS clearColor by default.\r\n */\r\n groundColor: Color3;\r\n /**\r\n * Specifies the ground opacity.\r\n * 1 by default.\r\n */\r\n groundOpacity: number;\r\n /**\r\n * Enables the ground to receive shadows.\r\n * True by default.\r\n */\r\n enableGroundShadow: boolean;\r\n /**\r\n * Helps preventing the shadow to be fully black on the ground.\r\n * 0.5 by default.\r\n */\r\n groundShadowLevel: number;\r\n /**\r\n * Creates a mirror texture attach to the ground.\r\n * false by default.\r\n */\r\n enableGroundMirror: boolean;\r\n /**\r\n * Specifies the ground mirror size ratio.\r\n * 0.3 by default as the default kernel is 64.\r\n */\r\n groundMirrorSizeRatio: number;\r\n /**\r\n * Specifies the ground mirror blur kernel size.\r\n * 64 by default.\r\n */\r\n groundMirrorBlurKernel: number;\r\n /**\r\n * Specifies the ground mirror visibility amount.\r\n * 1 by default\r\n */\r\n groundMirrorAmount: number;\r\n /**\r\n * Specifies the ground mirror reflectance weight.\r\n * This uses the standard weight of the background material to setup the fresnel effect\r\n * of the mirror.\r\n * 1 by default.\r\n */\r\n groundMirrorFresnelWeight: number;\r\n /**\r\n * Specifies the ground mirror Falloff distance.\r\n * This can helps reducing the size of the reflection.\r\n * 0 by Default.\r\n */\r\n groundMirrorFallOffDistance: number;\r\n /**\r\n * Specifies the ground mirror texture type.\r\n * Unsigned Int by Default.\r\n */\r\n groundMirrorTextureType: number;\r\n /**\r\n * Specifies a bias applied to the ground vertical position to prevent z-fighting with\r\n * the shown objects.\r\n */\r\n groundYBias: number;\r\n\r\n /**\r\n * Specifies whether or not to create a skybox.\r\n * True by default.\r\n */\r\n createSkybox: boolean;\r\n /**\r\n * Specifies the skybox size.\r\n * 20 by default.\r\n */\r\n skyboxSize: number;\r\n /**\r\n * The texture used on the skybox for the main color.\r\n * Comes from the BabylonJS CDN by default.\r\n *\r\n * Remarks: Can be either a texture or a url.\r\n */\r\n skyboxTexture: string | BaseTexture;\r\n /**\r\n * The color mixed in the skybox texture by default.\r\n * BabylonJS clearColor by default.\r\n */\r\n skyboxColor: Color3;\r\n\r\n /**\r\n * The background rotation around the Y axis of the scene.\r\n * This helps aligning the key lights of your scene with the background.\r\n * 0 by default.\r\n */\r\n backgroundYRotation: number;\r\n\r\n /**\r\n * Compute automatically the size of the elements to best fit with the scene.\r\n */\r\n sizeAuto: boolean;\r\n\r\n /**\r\n * Default position of the rootMesh if autoSize is not true.\r\n */\r\n rootPosition: Vector3;\r\n\r\n /**\r\n * Sets up the image processing in the scene.\r\n * true by default.\r\n */\r\n setupImageProcessing: boolean;\r\n\r\n /**\r\n * The texture used as your environment texture in the scene.\r\n * Comes from the BabylonJS CDN by default and in use if setupImageProcessing is true.\r\n *\r\n * Remarks: Can be either a texture or a url.\r\n */\r\n environmentTexture: string | BaseTexture;\r\n\r\n /**\r\n * The value of the exposure to apply to the scene.\r\n * 0.6 by default if setupImageProcessing is true.\r\n */\r\n cameraExposure: number;\r\n\r\n /**\r\n * The value of the contrast to apply to the scene.\r\n * 1.6 by default if setupImageProcessing is true.\r\n */\r\n cameraContrast: number;\r\n\r\n /**\r\n * Specifies whether or not tonemapping should be enabled in the scene.\r\n * true by default if setupImageProcessing is true.\r\n */\r\n toneMappingEnabled: boolean;\r\n}\r\n\r\ninterface ISceneSize {\r\n groundSize: number;\r\n skyboxSize: number;\r\n rootPosition: Vector3;\r\n}\r\n\r\n/**\r\n * The EnvironmentHelper class can be used to add a fully featured non-expensive background to your scene.\r\n * It includes by default a skybox and a ground relying on the BackgroundMaterial.\r\n * It also helps with the default setup of your ImageProcessingConfiguration.\r\n */\r\nexport class EnvironmentHelper {\r\n /**\r\n * Default ground texture URL.\r\n */\r\n private static _GroundTextureCDNUrl = \"https://assets.babylonjs.com/environments/backgroundGround.png\";\r\n\r\n /**\r\n * Default skybox texture URL.\r\n */\r\n private static _SkyboxTextureCDNUrl = \"https://assets.babylonjs.com/environments/backgroundSkybox.dds\";\r\n\r\n /**\r\n * Default environment texture URL.\r\n */\r\n private static _EnvironmentTextureCDNUrl = \"https://assets.babylonjs.com/environments/environmentSpecular.env\";\r\n\r\n /**\r\n * Creates the default options for the helper.\r\n * @param scene The scene the environment helper belongs to.\r\n */\r\n private static _GetDefaultOptions(scene: Scene): IEnvironmentHelperOptions {\r\n return {\r\n createGround: true,\r\n groundSize: 15,\r\n groundTexture: this._GroundTextureCDNUrl,\r\n groundColor: new Color3(0.2, 0.2, 0.3).toLinearSpace(scene.getEngine().useExactSrgbConversions).scale(3),\r\n groundOpacity: 0.9,\r\n enableGroundShadow: true,\r\n groundShadowLevel: 0.5,\r\n\r\n enableGroundMirror: false,\r\n groundMirrorSizeRatio: 0.3,\r\n groundMirrorBlurKernel: 64,\r\n groundMirrorAmount: 1,\r\n groundMirrorFresnelWeight: 1,\r\n groundMirrorFallOffDistance: 0,\r\n groundMirrorTextureType: Constants.TEXTURETYPE_UNSIGNED_INT,\r\n\r\n groundYBias: 0.00001,\r\n\r\n createSkybox: true,\r\n skyboxSize: 20,\r\n skyboxTexture: this._SkyboxTextureCDNUrl,\r\n skyboxColor: new Color3(0.2, 0.2, 0.3).toLinearSpace(scene.getEngine().useExactSrgbConversions).scale(3),\r\n\r\n backgroundYRotation: 0,\r\n sizeAuto: true,\r\n rootPosition: Vector3.Zero(),\r\n\r\n setupImageProcessing: true,\r\n environmentTexture: this._EnvironmentTextureCDNUrl,\r\n cameraExposure: 0.8,\r\n cameraContrast: 1.2,\r\n toneMappingEnabled: true,\r\n };\r\n }\r\n\r\n private _rootMesh: Mesh;\r\n /**\r\n * Gets the root mesh created by the helper.\r\n */\r\n public get rootMesh(): Mesh {\r\n return this._rootMesh;\r\n }\r\n\r\n private _skybox: Nullable;\r\n /**\r\n * Gets the skybox created by the helper.\r\n */\r\n public get skybox(): Nullable {\r\n return this._skybox;\r\n }\r\n\r\n private _skyboxTexture: Nullable;\r\n /**\r\n * Gets the skybox texture created by the helper.\r\n */\r\n public get skyboxTexture(): Nullable {\r\n return this._skyboxTexture;\r\n }\r\n\r\n private _skyboxMaterial: Nullable;\r\n /**\r\n * Gets the skybox material created by the helper.\r\n */\r\n public get skyboxMaterial(): Nullable {\r\n return this._skyboxMaterial;\r\n }\r\n\r\n private _ground: Nullable;\r\n /**\r\n * Gets the ground mesh created by the helper.\r\n */\r\n public get ground(): Nullable {\r\n return this._ground;\r\n }\r\n\r\n private _groundTexture: Nullable;\r\n /**\r\n * Gets the ground texture created by the helper.\r\n */\r\n public get groundTexture(): Nullable {\r\n return this._groundTexture;\r\n }\r\n\r\n private _groundMirror: Nullable;\r\n /**\r\n * Gets the ground mirror created by the helper.\r\n */\r\n public get groundMirror(): Nullable {\r\n return this._groundMirror;\r\n }\r\n\r\n /**\r\n * Gets the ground mirror render list to helps pushing the meshes\r\n * you wish in the ground reflection.\r\n */\r\n public get groundMirrorRenderList(): Nullable {\r\n if (this._groundMirror) {\r\n return this._groundMirror.renderList;\r\n }\r\n return null;\r\n }\r\n\r\n private _groundMaterial: Nullable;\r\n /**\r\n * Gets the ground material created by the helper.\r\n */\r\n public get groundMaterial(): Nullable {\r\n return this._groundMaterial;\r\n }\r\n\r\n /**\r\n * Stores the creation options.\r\n */\r\n private readonly _scene: Scene;\r\n private _options: IEnvironmentHelperOptions;\r\n\r\n /**\r\n * This observable will be notified with any error during the creation of the environment,\r\n * mainly texture creation errors.\r\n */\r\n public onErrorObservable: Observable<{ message?: string; exception?: any }>;\r\n\r\n /**\r\n * constructor\r\n * @param options Defines the options we want to customize the helper\r\n * @param scene The scene to add the material to\r\n */\r\n constructor(options: Partial, scene: Scene) {\r\n this._options = {\r\n ...EnvironmentHelper._GetDefaultOptions(scene),\r\n ...options,\r\n };\r\n this._scene = scene;\r\n this.onErrorObservable = new Observable();\r\n\r\n this._setupBackground();\r\n this._setupImageProcessing();\r\n }\r\n\r\n /**\r\n * Updates the background according to the new options\r\n * @param options\r\n */\r\n public updateOptions(options: Partial) {\r\n const newOptions = {\r\n ...this._options,\r\n ...options,\r\n };\r\n\r\n if (this._ground && !newOptions.createGround) {\r\n this._ground.dispose();\r\n this._ground = null;\r\n }\r\n\r\n if (this._groundMaterial && !newOptions.createGround) {\r\n this._groundMaterial.dispose();\r\n this._groundMaterial = null;\r\n }\r\n\r\n if (this._groundTexture) {\r\n if (this._options.groundTexture != newOptions.groundTexture) {\r\n this._groundTexture.dispose();\r\n this._groundTexture = null;\r\n }\r\n }\r\n\r\n if (this._skybox && !newOptions.createSkybox) {\r\n this._skybox.dispose();\r\n this._skybox = null;\r\n }\r\n\r\n if (this._skyboxMaterial && !newOptions.createSkybox) {\r\n this._skyboxMaterial.dispose();\r\n this._skyboxMaterial = null;\r\n }\r\n\r\n if (this._skyboxTexture) {\r\n if (this._options.skyboxTexture != newOptions.skyboxTexture) {\r\n this._skyboxTexture.dispose();\r\n this._skyboxTexture = null;\r\n }\r\n }\r\n\r\n if (this._groundMirror && !newOptions.enableGroundMirror) {\r\n this._groundMirror.dispose();\r\n this._groundMirror = null;\r\n }\r\n\r\n if (this._scene.environmentTexture) {\r\n if (this._options.environmentTexture != newOptions.environmentTexture) {\r\n this._scene.environmentTexture.dispose();\r\n }\r\n }\r\n\r\n this._options = newOptions;\r\n\r\n this._setupBackground();\r\n this._setupImageProcessing();\r\n }\r\n\r\n /**\r\n * Sets the primary color of all the available elements.\r\n * @param color the main color to affect to the ground and the background\r\n */\r\n public setMainColor(color: Color3): void {\r\n if (this.groundMaterial) {\r\n this.groundMaterial.primaryColor = color;\r\n }\r\n\r\n if (this.skyboxMaterial) {\r\n this.skyboxMaterial.primaryColor = color;\r\n }\r\n\r\n if (this.groundMirror) {\r\n this.groundMirror.clearColor = new Color4(color.r, color.g, color.b, 1.0);\r\n }\r\n }\r\n\r\n /**\r\n * Setup the image processing according to the specified options.\r\n */\r\n private _setupImageProcessing(): void {\r\n if (this._options.setupImageProcessing) {\r\n this._scene.imageProcessingConfiguration.contrast = this._options.cameraContrast;\r\n this._scene.imageProcessingConfiguration.exposure = this._options.cameraExposure;\r\n this._scene.imageProcessingConfiguration.toneMappingEnabled = this._options.toneMappingEnabled;\r\n this._setupEnvironmentTexture();\r\n }\r\n }\r\n\r\n /**\r\n * Setup the environment texture according to the specified options.\r\n */\r\n private _setupEnvironmentTexture(): void {\r\n if (this._scene.environmentTexture) {\r\n return;\r\n }\r\n\r\n if (this._options.environmentTexture instanceof BaseTexture) {\r\n this._scene.environmentTexture = this._options.environmentTexture;\r\n return;\r\n }\r\n\r\n const environmentTexture = CubeTexture.CreateFromPrefilteredData(this._options.environmentTexture, this._scene);\r\n this._scene.environmentTexture = environmentTexture;\r\n }\r\n\r\n /**\r\n * Setup the background according to the specified options.\r\n */\r\n private _setupBackground(): void {\r\n if (!this._rootMesh) {\r\n this._rootMesh = new Mesh(\"BackgroundHelper\", this._scene);\r\n }\r\n this._rootMesh.rotation.y = this._options.backgroundYRotation;\r\n\r\n const sceneSize = this._getSceneSize();\r\n if (this._options.createGround) {\r\n this._setupGround(sceneSize);\r\n this._setupGroundMaterial();\r\n this._setupGroundDiffuseTexture();\r\n\r\n if (this._options.enableGroundMirror) {\r\n this._setupGroundMirrorTexture(sceneSize);\r\n }\r\n this._setupMirrorInGroundMaterial();\r\n }\r\n\r\n if (this._options.createSkybox) {\r\n this._setupSkybox(sceneSize);\r\n this._setupSkyboxMaterial();\r\n this._setupSkyboxReflectionTexture();\r\n }\r\n\r\n this._rootMesh.position.x = sceneSize.rootPosition.x;\r\n this._rootMesh.position.z = sceneSize.rootPosition.z;\r\n this._rootMesh.position.y = sceneSize.rootPosition.y;\r\n }\r\n\r\n /**\r\n * Get the scene sizes according to the setup.\r\n */\r\n private _getSceneSize(): ISceneSize {\r\n let groundSize = this._options.groundSize;\r\n let skyboxSize = this._options.skyboxSize;\r\n let rootPosition = this._options.rootPosition;\r\n if (!this._scene.meshes || this._scene.meshes.length === 1) {\r\n // 1 only means the root of the helper.\r\n return { groundSize, skyboxSize, rootPosition };\r\n }\r\n\r\n const sceneExtends = this._scene.getWorldExtends((mesh) => {\r\n return mesh !== this._ground && mesh !== this._rootMesh && mesh !== this._skybox;\r\n });\r\n const sceneDiagonal = sceneExtends.max.subtract(sceneExtends.min);\r\n\r\n if (this._options.sizeAuto) {\r\n if (this._scene.activeCamera instanceof ArcRotateCamera && this._scene.activeCamera.upperRadiusLimit) {\r\n groundSize = this._scene.activeCamera.upperRadiusLimit * 2;\r\n skyboxSize = groundSize;\r\n }\r\n\r\n const sceneDiagonalLenght = sceneDiagonal.length();\r\n if (sceneDiagonalLenght > groundSize) {\r\n groundSize = sceneDiagonalLenght * 2;\r\n skyboxSize = groundSize;\r\n }\r\n\r\n // 10 % bigger.\r\n groundSize *= 1.1;\r\n skyboxSize *= 1.5;\r\n rootPosition = sceneExtends.min.add(sceneDiagonal.scale(0.5));\r\n rootPosition.y = sceneExtends.min.y - this._options.groundYBias;\r\n }\r\n\r\n return { groundSize, skyboxSize, rootPosition };\r\n }\r\n\r\n /**\r\n * Setup the ground according to the specified options.\r\n * @param sceneSize\r\n */\r\n private _setupGround(sceneSize: ISceneSize): void {\r\n if (!this._ground || this._ground.isDisposed()) {\r\n this._ground = CreatePlane(\"BackgroundPlane\", { size: sceneSize.groundSize }, this._scene);\r\n this._ground.rotation.x = Math.PI / 2; // Face up by default.\r\n this._ground.parent = this._rootMesh;\r\n this._ground.onDisposeObservable.add(() => {\r\n this._ground = null;\r\n });\r\n }\r\n\r\n this._ground.receiveShadows = this._options.enableGroundShadow;\r\n }\r\n\r\n /**\r\n * Setup the ground material according to the specified options.\r\n */\r\n private _setupGroundMaterial(): void {\r\n if (!this._groundMaterial) {\r\n this._groundMaterial = new BackgroundMaterial(\"BackgroundPlaneMaterial\", this._scene);\r\n }\r\n this._groundMaterial.alpha = this._options.groundOpacity;\r\n this._groundMaterial.alphaMode = Constants.ALPHA_PREMULTIPLIED_PORTERDUFF;\r\n this._groundMaterial.shadowLevel = this._options.groundShadowLevel;\r\n this._groundMaterial.primaryColor = this._options.groundColor;\r\n this._groundMaterial.useRGBColor = false;\r\n this._groundMaterial.enableNoise = true;\r\n\r\n if (this._ground) {\r\n this._ground.material = this._groundMaterial;\r\n }\r\n }\r\n\r\n /**\r\n * Setup the ground diffuse texture according to the specified options.\r\n */\r\n private _setupGroundDiffuseTexture(): void {\r\n if (!this._groundMaterial) {\r\n return;\r\n }\r\n\r\n if (this._groundTexture) {\r\n return;\r\n }\r\n\r\n if (this._options.groundTexture instanceof BaseTexture) {\r\n this._groundMaterial.diffuseTexture = this._options.groundTexture;\r\n return;\r\n }\r\n\r\n this._groundTexture = new Texture(this._options.groundTexture, this._scene, undefined, undefined, undefined, undefined, this._errorHandler);\r\n this._groundTexture.gammaSpace = false;\r\n this._groundTexture.hasAlpha = true;\r\n this._groundMaterial.diffuseTexture = this._groundTexture;\r\n }\r\n\r\n /**\r\n * Setup the ground mirror texture according to the specified options.\r\n * @param sceneSize\r\n */\r\n private _setupGroundMirrorTexture(sceneSize: ISceneSize): void {\r\n const wrapping = Texture.CLAMP_ADDRESSMODE;\r\n if (!this._groundMirror) {\r\n this._groundMirror = new MirrorTexture(\r\n \"BackgroundPlaneMirrorTexture\",\r\n { ratio: this._options.groundMirrorSizeRatio },\r\n this._scene,\r\n false,\r\n this._options.groundMirrorTextureType,\r\n Texture.BILINEAR_SAMPLINGMODE,\r\n true\r\n );\r\n this._groundMirror.mirrorPlane = new Plane(0, -1, 0, sceneSize.rootPosition.y);\r\n this._groundMirror.anisotropicFilteringLevel = 1;\r\n this._groundMirror.wrapU = wrapping;\r\n this._groundMirror.wrapV = wrapping;\r\n\r\n if (this._groundMirror.renderList) {\r\n for (let i = 0; i < this._scene.meshes.length; i++) {\r\n const mesh = this._scene.meshes[i];\r\n if (mesh !== this._ground && mesh !== this._skybox && mesh !== this._rootMesh) {\r\n this._groundMirror.renderList.push(mesh);\r\n }\r\n }\r\n }\r\n }\r\n\r\n const gammaGround = this._options.groundColor.toGammaSpace(this._scene.getEngine().useExactSrgbConversions);\r\n this._groundMirror.clearColor = new Color4(gammaGround.r, gammaGround.g, gammaGround.b, 1);\r\n this._groundMirror.adaptiveBlurKernel = this._options.groundMirrorBlurKernel;\r\n }\r\n\r\n /**\r\n * Setup the ground to receive the mirror texture.\r\n */\r\n private _setupMirrorInGroundMaterial(): void {\r\n if (this._groundMaterial) {\r\n this._groundMaterial.reflectionTexture = this._groundMirror;\r\n this._groundMaterial.reflectionFresnel = true;\r\n this._groundMaterial.reflectionAmount = this._options.groundMirrorAmount;\r\n this._groundMaterial.reflectionStandardFresnelWeight = this._options.groundMirrorFresnelWeight;\r\n this._groundMaterial.reflectionFalloffDistance = this._options.groundMirrorFallOffDistance;\r\n }\r\n }\r\n\r\n /**\r\n * Setup the skybox according to the specified options.\r\n * @param sceneSize\r\n */\r\n private _setupSkybox(sceneSize: ISceneSize): void {\r\n if (!this._skybox || this._skybox.isDisposed()) {\r\n this._skybox = CreateBox(\"BackgroundSkybox\", { size: sceneSize.skyboxSize, sideOrientation: Mesh.BACKSIDE }, this._scene);\r\n this._skybox.onDisposeObservable.add(() => {\r\n this._skybox = null;\r\n });\r\n }\r\n this._skybox.parent = this._rootMesh;\r\n }\r\n\r\n /**\r\n * Setup the skybox material according to the specified options.\r\n */\r\n private _setupSkyboxMaterial(): void {\r\n if (!this._skybox) {\r\n return;\r\n }\r\n\r\n if (!this._skyboxMaterial) {\r\n this._skyboxMaterial = new BackgroundMaterial(\"BackgroundSkyboxMaterial\", this._scene);\r\n }\r\n this._skyboxMaterial.useRGBColor = false;\r\n this._skyboxMaterial.primaryColor = this._options.skyboxColor;\r\n this._skyboxMaterial.enableNoise = true;\r\n\r\n this._skybox.material = this._skyboxMaterial;\r\n }\r\n\r\n /**\r\n * Setup the skybox reflection texture according to the specified options.\r\n */\r\n private _setupSkyboxReflectionTexture(): void {\r\n if (!this._skyboxMaterial) {\r\n return;\r\n }\r\n\r\n if (this._skyboxTexture) {\r\n return;\r\n }\r\n\r\n if (this._options.skyboxTexture instanceof BaseTexture) {\r\n this._skyboxMaterial.reflectionTexture = this._options.skyboxTexture;\r\n return;\r\n }\r\n\r\n this._skyboxTexture = new CubeTexture(this._options.skyboxTexture, this._scene, undefined, undefined, undefined, undefined, this._errorHandler);\r\n this._skyboxTexture.coordinatesMode = Texture.SKYBOX_MODE;\r\n this._skyboxTexture.gammaSpace = false;\r\n this._skyboxMaterial.reflectionTexture = this._skyboxTexture;\r\n }\r\n\r\n private _errorHandler = (message?: string, exception?: any) => {\r\n this.onErrorObservable.notifyObservers({ message: message, exception: exception });\r\n };\r\n\r\n /**\r\n * Dispose all the elements created by the Helper.\r\n */\r\n public dispose(): void {\r\n if (this._groundMaterial) {\r\n this._groundMaterial.dispose(true, true);\r\n }\r\n if (this._skyboxMaterial) {\r\n this._skyboxMaterial.dispose(true, true);\r\n }\r\n this._rootMesh.dispose(false);\r\n }\r\n}\r\n","import { serialize } from \"../../Misc/decorators\";\r\nimport type { Observer } from \"../../Misc/observable\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { ICameraInput } from \"../../Cameras/cameraInputsManager\";\r\nimport { CameraInputTypes } from \"../../Cameras/cameraInputsManager\";\r\nimport type { FreeCamera } from \"../../Cameras/freeCamera\";\r\nimport type { KeyboardInfo } from \"../../Events/keyboardEvents\";\r\nimport { KeyboardEventTypes } from \"../../Events/keyboardEvents\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { Vector3 } from \"../../Maths/math.vector\";\r\nimport type { Engine } from \"../../Engines/engine\";\r\nimport { Tools } from \"../../Misc/tools\";\r\n/**\r\n * Manage the keyboard inputs to control the movement of a free camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\r\n */\r\nexport class FreeCameraKeyboardMoveInput implements ICameraInput {\r\n /**\r\n * Defines the camera the input is attached to.\r\n */\r\n public camera: FreeCamera;\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control the forward move of the camera.\r\n */\r\n @serialize()\r\n public keysUp = [38];\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control the upward move of the camera.\r\n */\r\n @serialize()\r\n public keysUpward = [33];\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control the backward move of the camera.\r\n */\r\n @serialize()\r\n public keysDown = [40];\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control the downward move of the camera.\r\n */\r\n @serialize()\r\n public keysDownward = [34];\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control the left strafe move of the camera.\r\n */\r\n @serialize()\r\n public keysLeft = [37];\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control the right strafe move of the camera.\r\n */\r\n @serialize()\r\n public keysRight = [39];\r\n\r\n /**\r\n * Defines the pointer angular sensibility along the X and Y axis or how fast is the camera rotating.\r\n */\r\n @serialize()\r\n public rotationSpeed = 0.5;\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control the left rotation move of the camera.\r\n */\r\n @serialize()\r\n public keysRotateLeft: number[] = [];\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control the right rotation move of the camera.\r\n */\r\n @serialize()\r\n public keysRotateRight: number[] = [];\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control the up rotation move of the camera.\r\n */\r\n @serialize()\r\n public keysRotateUp: number[] = [];\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control the down rotation move of the camera.\r\n */\r\n @serialize()\r\n public keysRotateDown: number[] = [];\r\n\r\n private _keys = new Array();\r\n private _onCanvasBlurObserver: Nullable>;\r\n private _onKeyboardObserver: Nullable>;\r\n private _engine: Engine;\r\n private _scene: Scene;\r\n\r\n /**\r\n * Attach the input controls to a specific dom element to get the input from.\r\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n */\r\n public attachControl(noPreventDefault?: boolean): void {\r\n // eslint-disable-next-line prefer-rest-params\r\n noPreventDefault = Tools.BackCompatCameraNoPreventDefault(arguments);\r\n if (this._onCanvasBlurObserver) {\r\n return;\r\n }\r\n\r\n this._scene = this.camera.getScene();\r\n this._engine = this._scene.getEngine();\r\n\r\n this._onCanvasBlurObserver = this._engine.onCanvasBlurObservable.add(() => {\r\n this._keys.length = 0;\r\n });\r\n\r\n this._onKeyboardObserver = this._scene.onKeyboardObservable.add((info) => {\r\n const evt = info.event;\r\n if (!evt.metaKey) {\r\n if (info.type === KeyboardEventTypes.KEYDOWN) {\r\n if (\r\n this.keysUp.indexOf(evt.keyCode) !== -1 ||\r\n this.keysDown.indexOf(evt.keyCode) !== -1 ||\r\n this.keysLeft.indexOf(evt.keyCode) !== -1 ||\r\n this.keysRight.indexOf(evt.keyCode) !== -1 ||\r\n this.keysUpward.indexOf(evt.keyCode) !== -1 ||\r\n this.keysDownward.indexOf(evt.keyCode) !== -1 ||\r\n this.keysRotateLeft.indexOf(evt.keyCode) !== -1 ||\r\n this.keysRotateRight.indexOf(evt.keyCode) !== -1 ||\r\n this.keysRotateUp.indexOf(evt.keyCode) !== -1 ||\r\n this.keysRotateDown.indexOf(evt.keyCode) !== -1\r\n ) {\r\n const index = this._keys.indexOf(evt.keyCode);\r\n\r\n if (index === -1) {\r\n this._keys.push(evt.keyCode);\r\n }\r\n if (!noPreventDefault) {\r\n evt.preventDefault();\r\n }\r\n }\r\n } else {\r\n if (\r\n this.keysUp.indexOf(evt.keyCode) !== -1 ||\r\n this.keysDown.indexOf(evt.keyCode) !== -1 ||\r\n this.keysLeft.indexOf(evt.keyCode) !== -1 ||\r\n this.keysRight.indexOf(evt.keyCode) !== -1 ||\r\n this.keysUpward.indexOf(evt.keyCode) !== -1 ||\r\n this.keysDownward.indexOf(evt.keyCode) !== -1 ||\r\n this.keysRotateLeft.indexOf(evt.keyCode) !== -1 ||\r\n this.keysRotateRight.indexOf(evt.keyCode) !== -1 ||\r\n this.keysRotateUp.indexOf(evt.keyCode) !== -1 ||\r\n this.keysRotateDown.indexOf(evt.keyCode) !== -1\r\n ) {\r\n const index = this._keys.indexOf(evt.keyCode);\r\n\r\n if (index >= 0) {\r\n this._keys.splice(index, 1);\r\n }\r\n if (!noPreventDefault) {\r\n evt.preventDefault();\r\n }\r\n }\r\n }\r\n }\r\n });\r\n }\r\n /**\r\n * Detach the current controls from the specified dom element.\r\n */\r\n public detachControl(): void {\r\n if (this._scene) {\r\n if (this._onKeyboardObserver) {\r\n this._scene.onKeyboardObservable.remove(this._onKeyboardObserver);\r\n }\r\n\r\n if (this._onCanvasBlurObserver) {\r\n this._engine.onCanvasBlurObservable.remove(this._onCanvasBlurObserver);\r\n }\r\n this._onKeyboardObserver = null;\r\n this._onCanvasBlurObserver = null;\r\n }\r\n this._keys.length = 0;\r\n }\r\n\r\n /**\r\n * Update the current camera state depending on the inputs that have been used this frame.\r\n * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop.\r\n */\r\n public checkInputs(): void {\r\n if (this._onKeyboardObserver) {\r\n const camera = this.camera;\r\n // Keyboard\r\n for (let index = 0; index < this._keys.length; index++) {\r\n const keyCode = this._keys[index];\r\n const speed = camera._computeLocalCameraSpeed();\r\n\r\n if (this.keysLeft.indexOf(keyCode) !== -1) {\r\n camera._localDirection.copyFromFloats(-speed, 0, 0);\r\n } else if (this.keysUp.indexOf(keyCode) !== -1) {\r\n camera._localDirection.copyFromFloats(0, 0, speed);\r\n } else if (this.keysRight.indexOf(keyCode) !== -1) {\r\n camera._localDirection.copyFromFloats(speed, 0, 0);\r\n } else if (this.keysDown.indexOf(keyCode) !== -1) {\r\n camera._localDirection.copyFromFloats(0, 0, -speed);\r\n } else if (this.keysUpward.indexOf(keyCode) !== -1) {\r\n camera._localDirection.copyFromFloats(0, speed, 0);\r\n } else if (this.keysDownward.indexOf(keyCode) !== -1) {\r\n camera._localDirection.copyFromFloats(0, -speed, 0);\r\n } else if (this.keysRotateLeft.indexOf(keyCode) !== -1) {\r\n camera._localDirection.copyFromFloats(0, 0, 0);\r\n camera.cameraRotation.y -= this._getLocalRotation();\r\n } else if (this.keysRotateRight.indexOf(keyCode) !== -1) {\r\n camera._localDirection.copyFromFloats(0, 0, 0);\r\n camera.cameraRotation.y += this._getLocalRotation();\r\n } else if (this.keysRotateUp.indexOf(keyCode) !== -1) {\r\n camera._localDirection.copyFromFloats(0, 0, 0);\r\n camera.cameraRotation.x -= this._getLocalRotation();\r\n } else if (this.keysRotateDown.indexOf(keyCode) !== -1) {\r\n camera._localDirection.copyFromFloats(0, 0, 0);\r\n camera.cameraRotation.x += this._getLocalRotation();\r\n }\r\n\r\n if (camera.getScene().useRightHandedSystem) {\r\n camera._localDirection.z *= -1;\r\n }\r\n\r\n camera.getViewMatrix().invertToRef(camera._cameraTransformMatrix);\r\n Vector3.TransformNormalToRef(camera._localDirection, camera._cameraTransformMatrix, camera._transformedDirection);\r\n camera.cameraDirection.addInPlace(camera._transformedDirection);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Gets the class name of the current input.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"FreeCameraKeyboardMoveInput\";\r\n }\r\n\r\n /** @internal */\r\n public _onLostFocus(): void {\r\n this._keys.length = 0;\r\n }\r\n\r\n /**\r\n * Get the friendly name associated with the input class.\r\n * @returns the input friendly name\r\n */\r\n public getSimpleName(): string {\r\n return \"keyboard\";\r\n }\r\n\r\n private _getLocalRotation(): number {\r\n let rotation = (this.rotationSpeed * this._engine.getDeltaTime()) / 1000;\r\n if (this.camera.getScene().useRightHandedSystem) {\r\n rotation *= -1;\r\n }\r\n if (this.camera.parent && this.camera.parent._getWorldMatrixDeterminant() < 0) {\r\n rotation *= -1;\r\n }\r\n return rotation;\r\n }\r\n}\r\n\r\n(CameraInputTypes)[\"FreeCameraKeyboardMoveInput\"] = FreeCameraKeyboardMoveInput;\r\n","import type { Observer, EventState } from \"../../Misc/observable\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport { serialize } from \"../../Misc/decorators\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { ICameraInput } from \"../../Cameras/cameraInputsManager\";\r\nimport { CameraInputTypes } from \"../../Cameras/cameraInputsManager\";\r\nimport type { FreeCamera } from \"../../Cameras/freeCamera\";\r\nimport type { PointerInfo } from \"../../Events/pointerEvents\";\r\nimport { PointerEventTypes } from \"../../Events/pointerEvents\";\r\nimport { Tools } from \"../../Misc/tools\";\r\nimport type { IMouseEvent, IPointerEvent } from \"../../Events/deviceInputEvents\";\r\n/**\r\n * Manage the mouse inputs to control the movement of a free camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\r\n */\r\nexport class FreeCameraMouseInput implements ICameraInput {\r\n /**\r\n * Defines the camera the input is attached to.\r\n */\r\n public camera: FreeCamera;\r\n\r\n /**\r\n * Defines the buttons associated with the input to handle camera move.\r\n */\r\n @serialize()\r\n public buttons = [0, 1, 2];\r\n\r\n /**\r\n * Defines the pointer angular sensibility along the X and Y axis or how fast is the camera rotating.\r\n */\r\n @serialize()\r\n public angularSensibility = 2000.0;\r\n\r\n private _pointerInput: (p: PointerInfo, s: EventState) => void;\r\n private _onMouseMove: Nullable<(e: IMouseEvent) => any>;\r\n private _observer: Nullable>;\r\n private _previousPosition: Nullable<{ x: number; y: number }> = null;\r\n\r\n /**\r\n * Observable for when a pointer move event occurs containing the move offset\r\n */\r\n public onPointerMovedObservable = new Observable<{ offsetX: number; offsetY: number }>();\r\n /**\r\n * @internal\r\n * If the camera should be rotated automatically based on pointer movement\r\n */\r\n public _allowCameraRotation = true;\r\n\r\n private _currentActiveButton: number = -1;\r\n private _activePointerId: number = -1;\r\n private _contextMenuBind: () => void;\r\n\r\n /**\r\n * Manage the mouse inputs to control the movement of a free camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\r\n * @param touchEnabled Defines if touch is enabled or not\r\n */\r\n constructor(\r\n /**\r\n * Define if touch is enabled in the mouse input\r\n */\r\n public touchEnabled = true\r\n ) {}\r\n\r\n /**\r\n * Attach the input controls to a specific dom element to get the input from.\r\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n */\r\n public attachControl(noPreventDefault?: boolean): void {\r\n // eslint-disable-next-line prefer-rest-params\r\n noPreventDefault = Tools.BackCompatCameraNoPreventDefault(arguments);\r\n const engine = this.camera.getEngine();\r\n const element = engine.getInputElement();\r\n\r\n if (!this._pointerInput) {\r\n this._pointerInput = (p) => {\r\n const evt = p.event;\r\n const isTouch = evt.pointerType === \"touch\";\r\n\r\n if (engine.isInVRExclusivePointerMode) {\r\n return;\r\n }\r\n\r\n if (!this.touchEnabled && isTouch) {\r\n return;\r\n }\r\n\r\n if (p.type !== PointerEventTypes.POINTERMOVE && this.buttons.indexOf(evt.button) === -1) {\r\n return;\r\n }\r\n\r\n const srcElement = evt.target;\r\n\r\n if (p.type === PointerEventTypes.POINTERDOWN) {\r\n // If the input is touch with more than one touch OR if the input is mouse and there is already an active button, return\r\n if ((isTouch && this._activePointerId !== -1) || (!isTouch && this._currentActiveButton !== -1)) {\r\n return;\r\n }\r\n\r\n this._activePointerId = evt.pointerId;\r\n try {\r\n srcElement?.setPointerCapture(evt.pointerId);\r\n } catch (e) {\r\n //Nothing to do with the error. Execution will continue.\r\n }\r\n\r\n if (this._currentActiveButton === -1) {\r\n this._currentActiveButton = evt.button;\r\n }\r\n\r\n this._previousPosition = {\r\n x: evt.clientX,\r\n y: evt.clientY,\r\n };\r\n\r\n if (!noPreventDefault) {\r\n evt.preventDefault();\r\n element && element.focus();\r\n }\r\n\r\n // This is required to move while pointer button is down\r\n if (engine.isPointerLock && this._onMouseMove) {\r\n this._onMouseMove(p.event);\r\n }\r\n } else if (p.type === PointerEventTypes.POINTERUP) {\r\n // If input is touch with a different touch id OR if input is mouse with a different button, return\r\n if ((isTouch && this._activePointerId !== evt.pointerId) || (!isTouch && this._currentActiveButton !== evt.button)) {\r\n return;\r\n }\r\n\r\n try {\r\n srcElement?.releasePointerCapture(evt.pointerId);\r\n } catch (e) {\r\n //Nothing to do with the error.\r\n }\r\n this._currentActiveButton = -1;\r\n\r\n this._previousPosition = null;\r\n if (!noPreventDefault) {\r\n evt.preventDefault();\r\n }\r\n\r\n this._activePointerId = -1;\r\n } else if (p.type === PointerEventTypes.POINTERMOVE && (this._activePointerId === evt.pointerId || !isTouch)) {\r\n if (engine.isPointerLock && this._onMouseMove) {\r\n this._onMouseMove(p.event);\r\n } else if (this._previousPosition) {\r\n let offsetX = evt.clientX - this._previousPosition.x;\r\n const offsetY = evt.clientY - this._previousPosition.y;\r\n if (this.camera.getScene().useRightHandedSystem) {\r\n offsetX *= -1;\r\n }\r\n if (this.camera.parent && this.camera.parent._getWorldMatrixDeterminant() < 0) {\r\n offsetX *= -1;\r\n }\r\n\r\n if (this._allowCameraRotation) {\r\n this.camera.cameraRotation.y += offsetX / this.angularSensibility;\r\n this.camera.cameraRotation.x += offsetY / this.angularSensibility;\r\n }\r\n this.onPointerMovedObservable.notifyObservers({ offsetX: offsetX, offsetY: offsetY });\r\n\r\n this._previousPosition = {\r\n x: evt.clientX,\r\n y: evt.clientY,\r\n };\r\n\r\n if (!noPreventDefault) {\r\n evt.preventDefault();\r\n }\r\n }\r\n }\r\n };\r\n }\r\n\r\n this._onMouseMove = (evt) => {\r\n if (!engine.isPointerLock) {\r\n return;\r\n }\r\n\r\n if (engine.isInVRExclusivePointerMode) {\r\n return;\r\n }\r\n\r\n let offsetX = evt.movementX;\r\n if (this.camera.getScene().useRightHandedSystem) {\r\n offsetX *= -1;\r\n }\r\n if (this.camera.parent && this.camera.parent._getWorldMatrixDeterminant() < 0) {\r\n offsetX *= -1;\r\n }\r\n this.camera.cameraRotation.y += offsetX / this.angularSensibility;\r\n\r\n const offsetY = evt.movementY;\r\n this.camera.cameraRotation.x += offsetY / this.angularSensibility;\r\n\r\n this._previousPosition = null;\r\n\r\n if (!noPreventDefault) {\r\n evt.preventDefault();\r\n }\r\n };\r\n\r\n this._observer = this.camera\r\n .getScene()\r\n ._inputManager._addCameraPointerObserver(this._pointerInput, PointerEventTypes.POINTERDOWN | PointerEventTypes.POINTERUP | PointerEventTypes.POINTERMOVE);\r\n\r\n if (element) {\r\n this._contextMenuBind = this.onContextMenu.bind(this);\r\n element.addEventListener(\"contextmenu\", this._contextMenuBind, false); // TODO: We need to figure out how to handle this for Native\r\n }\r\n }\r\n\r\n /**\r\n * Called on JS contextmenu event.\r\n * Override this method to provide functionality.\r\n * @param evt\r\n */\r\n public onContextMenu(evt: PointerEvent): void {\r\n evt.preventDefault();\r\n }\r\n\r\n /**\r\n * Detach the current controls from the specified dom element.\r\n */\r\n public detachControl(): void {\r\n if (this._observer) {\r\n this.camera.getScene()._inputManager._removeCameraPointerObserver(this._observer);\r\n\r\n if (this._contextMenuBind) {\r\n const engine = this.camera.getEngine();\r\n const element = engine.getInputElement();\r\n element && element.removeEventListener(\"contextmenu\", this._contextMenuBind);\r\n }\r\n\r\n if (this.onPointerMovedObservable) {\r\n this.onPointerMovedObservable.clear();\r\n }\r\n\r\n this._observer = null;\r\n this._onMouseMove = null;\r\n this._previousPosition = null;\r\n }\r\n\r\n this._activePointerId = -1;\r\n this._currentActiveButton = -1;\r\n }\r\n\r\n /**\r\n * Gets the class name of the current input.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"FreeCameraMouseInput\";\r\n }\r\n\r\n /**\r\n * Get the friendly name associated with the input class.\r\n * @returns the input friendly name\r\n */\r\n public getSimpleName(): string {\r\n return \"mouse\";\r\n }\r\n}\r\n\r\n(CameraInputTypes)[\"FreeCameraMouseInput\"] = FreeCameraMouseInput;\r\n","import type { Nullable } from \"../../types\";\r\nimport { serialize } from \"../../Misc/decorators\";\r\nimport type { Observer } from \"../../Misc/observable\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport type { Camera } from \"../../Cameras/camera\";\r\nimport type { ICameraInput } from \"../../Cameras/cameraInputsManager\";\r\nimport type { PointerInfo } from \"../../Events/pointerEvents\";\r\nimport { PointerEventTypes } from \"../../Events/pointerEvents\";\r\nimport type { IWheelEvent } from \"../../Events/deviceInputEvents\";\r\nimport { EventConstants } from \"../../Events/deviceInputEvents\";\r\nimport { Tools } from \"../../Misc/tools\";\r\n\r\n/**\r\n * Base class for mouse wheel input..\r\n * See FollowCameraMouseWheelInput in src/Cameras/Inputs/freeCameraMouseWheelInput.ts\r\n * for example usage.\r\n */\r\nexport abstract class BaseCameraMouseWheelInput implements ICameraInput {\r\n /**\r\n * Defines the camera the input is attached to.\r\n */\r\n public abstract camera: Camera;\r\n\r\n /**\r\n * How fast is the camera moves in relation to X axis mouseWheel events.\r\n * Use negative value to reverse direction.\r\n */\r\n @serialize()\r\n public wheelPrecisionX = 3.0;\r\n\r\n /**\r\n * How fast is the camera moves in relation to Y axis mouseWheel events.\r\n * Use negative value to reverse direction.\r\n */\r\n @serialize()\r\n public wheelPrecisionY = 3.0;\r\n\r\n /**\r\n * How fast is the camera moves in relation to Z axis mouseWheel events.\r\n * Use negative value to reverse direction.\r\n */\r\n @serialize()\r\n public wheelPrecisionZ = 3.0;\r\n\r\n /**\r\n * Observable for when a mouse wheel move event occurs.\r\n */\r\n public onChangedObservable = new Observable<{ wheelDeltaX: number; wheelDeltaY: number; wheelDeltaZ: number }>();\r\n\r\n private _wheel: Nullable<(pointer: PointerInfo) => void>;\r\n private _observer: Nullable>;\r\n\r\n /**\r\n * Attach the input controls to a specific dom element to get the input from.\r\n * @param noPreventDefault Defines whether event caught by the controls\r\n * should call preventdefault().\r\n * (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n */\r\n public attachControl(noPreventDefault?: boolean): void {\r\n noPreventDefault = Tools.BackCompatCameraNoPreventDefault(arguments);\r\n\r\n this._wheel = (pointer) => {\r\n // sanity check - this should be a PointerWheel event.\r\n if (pointer.type !== PointerEventTypes.POINTERWHEEL) {\r\n return;\r\n }\r\n\r\n const event = pointer.event;\r\n\r\n const platformScale = event.deltaMode === EventConstants.DOM_DELTA_LINE ? this._ffMultiplier : 1; // If this happens to be set to DOM_DELTA_LINE, adjust accordingly\r\n\r\n this._wheelDeltaX += (this.wheelPrecisionX * platformScale * event.deltaX) / this._normalize;\r\n this._wheelDeltaY -= (this.wheelPrecisionY * platformScale * event.deltaY) / this._normalize;\r\n this._wheelDeltaZ += (this.wheelPrecisionZ * platformScale * event.deltaZ) / this._normalize;\r\n\r\n if (event.preventDefault) {\r\n if (!noPreventDefault) {\r\n event.preventDefault();\r\n }\r\n }\r\n };\r\n\r\n this._observer = this.camera.getScene()._inputManager._addCameraPointerObserver(this._wheel, PointerEventTypes.POINTERWHEEL);\r\n }\r\n\r\n /**\r\n * Detach the current controls from the specified dom element.\r\n */\r\n public detachControl(): void {\r\n if (this._observer) {\r\n this.camera.getScene()._inputManager._removeCameraPointerObserver(this._observer);\r\n this._observer = null;\r\n this._wheel = null;\r\n }\r\n if (this.onChangedObservable) {\r\n this.onChangedObservable.clear();\r\n }\r\n }\r\n\r\n /**\r\n * Called for each rendered frame.\r\n */\r\n public checkInputs(): void {\r\n this.onChangedObservable.notifyObservers({\r\n wheelDeltaX: this._wheelDeltaX,\r\n wheelDeltaY: this._wheelDeltaY,\r\n wheelDeltaZ: this._wheelDeltaZ,\r\n });\r\n\r\n // Clear deltas.\r\n this._wheelDeltaX = 0;\r\n this._wheelDeltaY = 0;\r\n this._wheelDeltaZ = 0;\r\n }\r\n\r\n /**\r\n * Gets the class name of the current input.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"BaseCameraMouseWheelInput\";\r\n }\r\n\r\n /**\r\n * Get the friendly name associated with the input class.\r\n * @returns the input friendly name\r\n */\r\n public getSimpleName(): string {\r\n return \"mousewheel\";\r\n }\r\n\r\n /**\r\n * Incremental value of multiple mouse wheel movements of the X axis.\r\n * Should be zero-ed when read.\r\n */\r\n protected _wheelDeltaX: number = 0;\r\n\r\n /**\r\n * Incremental value of multiple mouse wheel movements of the Y axis.\r\n * Should be zero-ed when read.\r\n */\r\n protected _wheelDeltaY: number = 0;\r\n\r\n /**\r\n * Incremental value of multiple mouse wheel movements of the Z axis.\r\n * Should be zero-ed when read.\r\n */\r\n protected _wheelDeltaZ: number = 0;\r\n\r\n /**\r\n * Firefox uses a different scheme to report scroll distances to other\r\n * browsers. Rather than use complicated methods to calculate the exact\r\n * multiple we need to apply, let's just cheat and use a constant.\r\n * https://developer.mozilla.org/en-US/docs/Web/API/WheelEvent/deltaMode\r\n * https://stackoverflow.com/questions/20110224/what-is-the-height-of-a-line-in-a-wheel-event-deltamode-dom-delta-line\r\n */\r\n private readonly _ffMultiplier = 12;\r\n\r\n /**\r\n * Different event attributes for wheel data fall into a few set ranges.\r\n * Some relevant but dated date here:\r\n * https://stackoverflow.com/questions/5527601/normalizing-mousewheel-speed-across-browsers\r\n */\r\n private readonly _normalize = 120;\r\n}\r\n","import type { Nullable } from \"../../types\";\r\nimport { serialize } from \"../../Misc/decorators\";\r\nimport type { FreeCamera } from \"../../Cameras/freeCamera\";\r\nimport { CameraInputTypes } from \"../../Cameras/cameraInputsManager\";\r\nimport { BaseCameraMouseWheelInput } from \"../../Cameras/Inputs/BaseCameraMouseWheelInput\";\r\nimport { Matrix, Vector3 } from \"../../Maths/math.vector\";\r\nimport { Coordinate } from \"../../Maths/math.axis\";\r\n\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nenum _CameraProperty {\r\n MoveRelative,\r\n RotateRelative,\r\n MoveScene,\r\n}\r\n\r\n/**\r\n * Manage the mouse wheel inputs to control a free camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\r\n */\r\nexport class FreeCameraMouseWheelInput extends BaseCameraMouseWheelInput {\r\n /**\r\n * Defines the camera the input is attached to.\r\n */\r\n public camera: FreeCamera;\r\n\r\n /**\r\n * Gets the class name of the current input.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"FreeCameraMouseWheelInput\";\r\n }\r\n\r\n /**\r\n * Set which movement axis (relative to camera's orientation) the mouse\r\n * wheel's X axis controls.\r\n * @param axis The axis to be moved. Set null to clear.\r\n */\r\n @serialize()\r\n public set wheelXMoveRelative(axis: Nullable) {\r\n if (axis === null && this._wheelXAction !== _CameraProperty.MoveRelative) {\r\n // Attempting to clear different _wheelXAction.\r\n return;\r\n }\r\n this._wheelXAction = _CameraProperty.MoveRelative;\r\n this._wheelXActionCoordinate = axis;\r\n }\r\n\r\n /**\r\n * Get the configured movement axis (relative to camera's orientation) the\r\n * mouse wheel's X axis controls.\r\n * @returns The configured axis or null if none.\r\n */\r\n public get wheelXMoveRelative(): Nullable {\r\n if (this._wheelXAction !== _CameraProperty.MoveRelative) {\r\n return null;\r\n }\r\n return this._wheelXActionCoordinate;\r\n }\r\n\r\n /**\r\n * Set which movement axis (relative to camera's orientation) the mouse\r\n * wheel's Y axis controls.\r\n * @param axis The axis to be moved. Set null to clear.\r\n */\r\n @serialize()\r\n public set wheelYMoveRelative(axis: Nullable) {\r\n if (axis === null && this._wheelYAction !== _CameraProperty.MoveRelative) {\r\n // Attempting to clear different _wheelYAction.\r\n return;\r\n }\r\n this._wheelYAction = _CameraProperty.MoveRelative;\r\n this._wheelYActionCoordinate = axis;\r\n }\r\n\r\n /**\r\n * Get the configured movement axis (relative to camera's orientation) the\r\n * mouse wheel's Y axis controls.\r\n * @returns The configured axis or null if none.\r\n */\r\n public get wheelYMoveRelative(): Nullable {\r\n if (this._wheelYAction !== _CameraProperty.MoveRelative) {\r\n return null;\r\n }\r\n return this._wheelYActionCoordinate;\r\n }\r\n\r\n /**\r\n * Set which movement axis (relative to camera's orientation) the mouse\r\n * wheel's Z axis controls.\r\n * @param axis The axis to be moved. Set null to clear.\r\n */\r\n @serialize()\r\n public set wheelZMoveRelative(axis: Nullable) {\r\n if (axis === null && this._wheelZAction !== _CameraProperty.MoveRelative) {\r\n // Attempting to clear different _wheelZAction.\r\n return;\r\n }\r\n this._wheelZAction = _CameraProperty.MoveRelative;\r\n this._wheelZActionCoordinate = axis;\r\n }\r\n\r\n /**\r\n * Get the configured movement axis (relative to camera's orientation) the\r\n * mouse wheel's Z axis controls.\r\n * @returns The configured axis or null if none.\r\n */\r\n public get wheelZMoveRelative(): Nullable {\r\n if (this._wheelZAction !== _CameraProperty.MoveRelative) {\r\n return null;\r\n }\r\n return this._wheelZActionCoordinate;\r\n }\r\n\r\n /**\r\n * Set which rotation axis (relative to camera's orientation) the mouse\r\n * wheel's X axis controls.\r\n * @param axis The axis to be moved. Set null to clear.\r\n */\r\n @serialize()\r\n public set wheelXRotateRelative(axis: Nullable) {\r\n if (axis === null && this._wheelXAction !== _CameraProperty.RotateRelative) {\r\n // Attempting to clear different _wheelXAction.\r\n return;\r\n }\r\n this._wheelXAction = _CameraProperty.RotateRelative;\r\n this._wheelXActionCoordinate = axis;\r\n }\r\n\r\n /**\r\n * Get the configured rotation axis (relative to camera's orientation) the\r\n * mouse wheel's X axis controls.\r\n * @returns The configured axis or null if none.\r\n */\r\n public get wheelXRotateRelative(): Nullable {\r\n if (this._wheelXAction !== _CameraProperty.RotateRelative) {\r\n return null;\r\n }\r\n return this._wheelXActionCoordinate;\r\n }\r\n\r\n /**\r\n * Set which rotation axis (relative to camera's orientation) the mouse\r\n * wheel's Y axis controls.\r\n * @param axis The axis to be moved. Set null to clear.\r\n */\r\n @serialize()\r\n public set wheelYRotateRelative(axis: Nullable) {\r\n if (axis === null && this._wheelYAction !== _CameraProperty.RotateRelative) {\r\n // Attempting to clear different _wheelYAction.\r\n return;\r\n }\r\n this._wheelYAction = _CameraProperty.RotateRelative;\r\n this._wheelYActionCoordinate = axis;\r\n }\r\n\r\n /**\r\n * Get the configured rotation axis (relative to camera's orientation) the\r\n * mouse wheel's Y axis controls.\r\n * @returns The configured axis or null if none.\r\n */\r\n public get wheelYRotateRelative(): Nullable {\r\n if (this._wheelYAction !== _CameraProperty.RotateRelative) {\r\n return null;\r\n }\r\n return this._wheelYActionCoordinate;\r\n }\r\n\r\n /**\r\n * Set which rotation axis (relative to camera's orientation) the mouse\r\n * wheel's Z axis controls.\r\n * @param axis The axis to be moved. Set null to clear.\r\n */\r\n @serialize()\r\n public set wheelZRotateRelative(axis: Nullable) {\r\n if (axis === null && this._wheelZAction !== _CameraProperty.RotateRelative) {\r\n // Attempting to clear different _wheelZAction.\r\n return;\r\n }\r\n this._wheelZAction = _CameraProperty.RotateRelative;\r\n this._wheelZActionCoordinate = axis;\r\n }\r\n\r\n /**\r\n * Get the configured rotation axis (relative to camera's orientation) the\r\n * mouse wheel's Z axis controls.\r\n * @returns The configured axis or null if none.\r\n */\r\n public get wheelZRotateRelative(): Nullable {\r\n if (this._wheelZAction !== _CameraProperty.RotateRelative) {\r\n return null;\r\n }\r\n return this._wheelZActionCoordinate;\r\n }\r\n\r\n /**\r\n * Set which movement axis (relative to the scene) the mouse wheel's X axis\r\n * controls.\r\n * @param axis The axis to be moved. Set null to clear.\r\n */\r\n @serialize()\r\n public set wheelXMoveScene(axis: Nullable) {\r\n if (axis === null && this._wheelXAction !== _CameraProperty.MoveScene) {\r\n // Attempting to clear different _wheelXAction.\r\n return;\r\n }\r\n this._wheelXAction = _CameraProperty.MoveScene;\r\n this._wheelXActionCoordinate = axis;\r\n }\r\n\r\n /**\r\n * Get the configured movement axis (relative to the scene) the mouse wheel's\r\n * X axis controls.\r\n * @returns The configured axis or null if none.\r\n */\r\n public get wheelXMoveScene(): Nullable {\r\n if (this._wheelXAction !== _CameraProperty.MoveScene) {\r\n return null;\r\n }\r\n return this._wheelXActionCoordinate;\r\n }\r\n\r\n /**\r\n * Set which movement axis (relative to the scene) the mouse wheel's Y axis\r\n * controls.\r\n * @param axis The axis to be moved. Set null to clear.\r\n */\r\n @serialize()\r\n public set wheelYMoveScene(axis: Nullable) {\r\n if (axis === null && this._wheelYAction !== _CameraProperty.MoveScene) {\r\n // Attempting to clear different _wheelYAction.\r\n return;\r\n }\r\n this._wheelYAction = _CameraProperty.MoveScene;\r\n this._wheelYActionCoordinate = axis;\r\n }\r\n\r\n /**\r\n * Get the configured movement axis (relative to the scene) the mouse wheel's\r\n * Y axis controls.\r\n * @returns The configured axis or null if none.\r\n */\r\n public get wheelYMoveScene(): Nullable {\r\n if (this._wheelYAction !== _CameraProperty.MoveScene) {\r\n return null;\r\n }\r\n return this._wheelYActionCoordinate;\r\n }\r\n\r\n /**\r\n * Set which movement axis (relative to the scene) the mouse wheel's Z axis\r\n * controls.\r\n * @param axis The axis to be moved. Set null to clear.\r\n */\r\n @serialize()\r\n public set wheelZMoveScene(axis: Nullable) {\r\n if (axis === null && this._wheelZAction !== _CameraProperty.MoveScene) {\r\n // Attempting to clear different _wheelZAction.\r\n return;\r\n }\r\n this._wheelZAction = _CameraProperty.MoveScene;\r\n this._wheelZActionCoordinate = axis;\r\n }\r\n\r\n /**\r\n * Get the configured movement axis (relative to the scene) the mouse wheel's\r\n * Z axis controls.\r\n * @returns The configured axis or null if none.\r\n */\r\n public get wheelZMoveScene(): Nullable {\r\n if (this._wheelZAction !== _CameraProperty.MoveScene) {\r\n return null;\r\n }\r\n return this._wheelZActionCoordinate;\r\n }\r\n\r\n /**\r\n * Called for each rendered frame.\r\n */\r\n public checkInputs(): void {\r\n if (this._wheelDeltaX === 0 && this._wheelDeltaY === 0 && this._wheelDeltaZ == 0) {\r\n return;\r\n }\r\n\r\n // Clear the camera properties that we might be updating.\r\n this._moveRelative.setAll(0);\r\n this._rotateRelative.setAll(0);\r\n this._moveScene.setAll(0);\r\n\r\n // Set the camera properties that are to be updated.\r\n this._updateCamera();\r\n\r\n if (this.camera.getScene().useRightHandedSystem) {\r\n // TODO: Does this need done for worldUpdate too?\r\n this._moveRelative.z *= -1;\r\n }\r\n\r\n // Convert updates relative to camera to world position update.\r\n const cameraTransformMatrix = Matrix.Zero();\r\n this.camera.getViewMatrix().invertToRef(cameraTransformMatrix);\r\n\r\n const transformedDirection = Vector3.Zero();\r\n Vector3.TransformNormalToRef(this._moveRelative, cameraTransformMatrix, transformedDirection);\r\n\r\n // Apply updates to camera position.\r\n this.camera.cameraRotation.x += this._rotateRelative.x / 200;\r\n this.camera.cameraRotation.y += this._rotateRelative.y / 200;\r\n this.camera.cameraDirection.addInPlace(transformedDirection);\r\n this.camera.cameraDirection.addInPlace(this._moveScene);\r\n\r\n // Call the base class implementation to handle observers and do cleanup.\r\n super.checkInputs();\r\n }\r\n\r\n private _moveRelative = Vector3.Zero();\r\n private _rotateRelative = Vector3.Zero();\r\n private _moveScene = Vector3.Zero();\r\n\r\n /**\r\n * These are set to the desired default behaviour.\r\n */\r\n private _wheelXAction: Nullable<_CameraProperty> = _CameraProperty.MoveRelative;\r\n private _wheelXActionCoordinate: Nullable = Coordinate.X;\r\n private _wheelYAction: Nullable<_CameraProperty> = _CameraProperty.MoveRelative;\r\n private _wheelYActionCoordinate: Nullable = Coordinate.Z;\r\n private _wheelZAction: Nullable<_CameraProperty> = null;\r\n private _wheelZActionCoordinate: Nullable = null;\r\n\r\n /**\r\n * Update the camera according to any configured properties for the 3\r\n * mouse-wheel axis.\r\n */\r\n private _updateCamera(): void {\r\n // Do the camera updates for each of the 3 touch-wheel axis.\r\n this._updateCameraProperty(this._wheelDeltaX, this._wheelXAction, this._wheelXActionCoordinate);\r\n this._updateCameraProperty(this._wheelDeltaY, this._wheelYAction, this._wheelYActionCoordinate);\r\n this._updateCameraProperty(this._wheelDeltaZ, this._wheelZAction, this._wheelZActionCoordinate);\r\n }\r\n\r\n /**\r\n * Update one property of the camera.\r\n * @param value\r\n * @param cameraProperty\r\n * @param coordinate\r\n */\r\n private _updateCameraProperty(\r\n /* Mouse-wheel delta. */\r\n value: number,\r\n /* Camera property to be changed. */\r\n cameraProperty: Nullable<_CameraProperty>,\r\n /* Axis of Camera property to be changed. */\r\n coordinate: Nullable\r\n ): void {\r\n if (value === 0) {\r\n // Mouse wheel has not moved.\r\n return;\r\n }\r\n if (cameraProperty === null || coordinate === null) {\r\n // Mouse wheel axis not configured.\r\n return;\r\n }\r\n\r\n let action = null;\r\n switch (cameraProperty) {\r\n case _CameraProperty.MoveRelative:\r\n action = this._moveRelative;\r\n break;\r\n case _CameraProperty.RotateRelative:\r\n action = this._rotateRelative;\r\n break;\r\n case _CameraProperty.MoveScene:\r\n action = this._moveScene;\r\n break;\r\n }\r\n\r\n switch (coordinate) {\r\n case Coordinate.X:\r\n action.set(value, 0, 0);\r\n break;\r\n case Coordinate.Y:\r\n action.set(0, value, 0);\r\n break;\r\n case Coordinate.Z:\r\n action.set(0, 0, value);\r\n break;\r\n }\r\n }\r\n}\r\n\r\n(CameraInputTypes)[\"FreeCameraMouseWheelInput\"] = FreeCameraMouseWheelInput;\r\n","import { serialize } from \"../../Misc/decorators\";\r\nimport type { Observer, EventState } from \"../../Misc/observable\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { ICameraInput } from \"../../Cameras/cameraInputsManager\";\r\nimport { CameraInputTypes } from \"../../Cameras/cameraInputsManager\";\r\nimport type { FreeCamera } from \"../../Cameras/freeCamera\";\r\nimport type { PointerInfo } from \"../../Events/pointerEvents\";\r\nimport { PointerEventTypes } from \"../../Events/pointerEvents\";\r\nimport { Matrix, Vector3 } from \"../../Maths/math.vector\";\r\nimport { Tools } from \"../../Misc/tools\";\r\nimport type { IPointerEvent } from \"../../Events/deviceInputEvents\";\r\n/**\r\n * Manage the touch inputs to control the movement of a free camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\r\n */\r\nexport class FreeCameraTouchInput implements ICameraInput {\r\n /**\r\n * Defines the camera the input is attached to.\r\n */\r\n public camera: FreeCamera;\r\n\r\n /**\r\n * Defines the touch sensibility for rotation.\r\n * The lower the faster.\r\n */\r\n @serialize()\r\n public touchAngularSensibility: number = 200000.0;\r\n\r\n /**\r\n * Defines the touch sensibility for move.\r\n * The lower the faster.\r\n */\r\n @serialize()\r\n public touchMoveSensibility: number = 250.0;\r\n\r\n /**\r\n * Swap touch actions so that one touch is used for rotation and multiple for movement\r\n */\r\n public singleFingerRotate: boolean = false;\r\n\r\n private _offsetX: Nullable = null;\r\n private _offsetY: Nullable = null;\r\n\r\n private _pointerPressed = new Array();\r\n private _pointerInput?: (p: PointerInfo, s: EventState) => void;\r\n private _observer: Nullable>;\r\n private _onLostFocus: Nullable<(e: FocusEvent) => any>;\r\n private _isSafari: boolean;\r\n\r\n /**\r\n * Manage the touch inputs to control the movement of a free camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\r\n * @param allowMouse Defines if mouse events can be treated as touch events\r\n */\r\n constructor(\r\n /**\r\n * Define if mouse events can be treated as touch events\r\n */\r\n public allowMouse = false\r\n ) {\r\n this._isSafari = Tools.IsSafari();\r\n }\r\n\r\n /**\r\n * Attach the input controls to a specific dom element to get the input from.\r\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n */\r\n public attachControl(noPreventDefault?: boolean): void {\r\n // eslint-disable-next-line prefer-rest-params\r\n noPreventDefault = Tools.BackCompatCameraNoPreventDefault(arguments);\r\n let previousPosition: Nullable<{ x: number; y: number }> = null;\r\n\r\n if (this._pointerInput === undefined) {\r\n this._onLostFocus = () => {\r\n this._offsetX = null;\r\n this._offsetY = null;\r\n };\r\n\r\n this._pointerInput = (p) => {\r\n const evt = p.event;\r\n\r\n const isMouseEvent = evt.pointerType === \"mouse\" || (this._isSafari && typeof evt.pointerType === \"undefined\");\r\n\r\n if (!this.allowMouse && isMouseEvent) {\r\n return;\r\n }\r\n\r\n if (p.type === PointerEventTypes.POINTERDOWN) {\r\n if (!noPreventDefault) {\r\n evt.preventDefault();\r\n }\r\n\r\n this._pointerPressed.push(evt.pointerId);\r\n\r\n if (this._pointerPressed.length !== 1) {\r\n return;\r\n }\r\n\r\n previousPosition = {\r\n x: evt.clientX,\r\n y: evt.clientY,\r\n };\r\n } else if (p.type === PointerEventTypes.POINTERUP) {\r\n if (!noPreventDefault) {\r\n evt.preventDefault();\r\n }\r\n\r\n const index: number = this._pointerPressed.indexOf(evt.pointerId);\r\n\r\n if (index === -1) {\r\n return;\r\n }\r\n this._pointerPressed.splice(index, 1);\r\n\r\n if (index != 0) {\r\n return;\r\n }\r\n previousPosition = null;\r\n this._offsetX = null;\r\n this._offsetY = null;\r\n } else if (p.type === PointerEventTypes.POINTERMOVE) {\r\n if (!noPreventDefault) {\r\n evt.preventDefault();\r\n }\r\n\r\n if (!previousPosition) {\r\n return;\r\n }\r\n\r\n const index: number = this._pointerPressed.indexOf(evt.pointerId);\r\n\r\n if (index != 0) {\r\n return;\r\n }\r\n\r\n this._offsetX = evt.clientX - previousPosition.x;\r\n this._offsetY = -(evt.clientY - previousPosition.y);\r\n }\r\n };\r\n }\r\n\r\n this._observer = this.camera\r\n .getScene()\r\n ._inputManager._addCameraPointerObserver(this._pointerInput, PointerEventTypes.POINTERDOWN | PointerEventTypes.POINTERUP | PointerEventTypes.POINTERMOVE);\r\n\r\n if (this._onLostFocus) {\r\n const engine = this.camera.getEngine();\r\n const element = engine.getInputElement();\r\n element && element.addEventListener(\"blur\", this._onLostFocus);\r\n }\r\n }\r\n\r\n /**\r\n * Detach the current controls from the specified dom element.\r\n */\r\n public detachControl(): void {\r\n if (this._pointerInput) {\r\n if (this._observer) {\r\n this.camera.getScene()._inputManager._removeCameraPointerObserver(this._observer);\r\n this._observer = null;\r\n }\r\n\r\n if (this._onLostFocus) {\r\n const engine = this.camera.getEngine();\r\n const element = engine.getInputElement();\r\n element && element.removeEventListener(\"blur\", this._onLostFocus);\r\n this._onLostFocus = null;\r\n }\r\n this._pointerPressed.length = 0;\r\n this._offsetX = null;\r\n this._offsetY = null;\r\n }\r\n }\r\n\r\n /**\r\n * Update the current camera state depending on the inputs that have been used this frame.\r\n * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop.\r\n */\r\n public checkInputs(): void {\r\n if (this._offsetX === null || this._offsetY === null) {\r\n return;\r\n }\r\n if (this._offsetX === 0 && this._offsetY === 0) {\r\n return;\r\n }\r\n\r\n const camera = this.camera;\r\n camera.cameraRotation.y = this._offsetX / this.touchAngularSensibility;\r\n\r\n const rotateCamera = (this.singleFingerRotate && this._pointerPressed.length === 1) || (!this.singleFingerRotate && this._pointerPressed.length > 1);\r\n\r\n if (rotateCamera) {\r\n camera.cameraRotation.x = -this._offsetY / this.touchAngularSensibility;\r\n } else {\r\n const speed = camera._computeLocalCameraSpeed();\r\n const direction = new Vector3(0, 0, this.touchMoveSensibility !== 0 ? (speed * this._offsetY) / this.touchMoveSensibility : 0);\r\n\r\n Matrix.RotationYawPitchRollToRef(camera.rotation.y, camera.rotation.x, 0, camera._cameraRotationMatrix);\r\n camera.cameraDirection.addInPlace(Vector3.TransformCoordinates(direction, camera._cameraRotationMatrix));\r\n }\r\n }\r\n\r\n /**\r\n * Gets the class name of the current input.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"FreeCameraTouchInput\";\r\n }\r\n\r\n /**\r\n * Get the friendly name associated with the input class.\r\n * @returns the input friendly name\r\n */\r\n public getSimpleName(): string {\r\n return \"touch\";\r\n }\r\n}\r\n\r\n(CameraInputTypes)[\"FreeCameraTouchInput\"] = FreeCameraTouchInput;\r\n","import type { FreeCamera } from \"./freeCamera\";\r\nimport { CameraInputsManager } from \"./cameraInputsManager\";\r\nimport { FreeCameraKeyboardMoveInput } from \"../Cameras/Inputs/freeCameraKeyboardMoveInput\";\r\nimport { FreeCameraMouseInput } from \"../Cameras/Inputs/freeCameraMouseInput\";\r\nimport { FreeCameraMouseWheelInput } from \"../Cameras/Inputs/freeCameraMouseWheelInput\";\r\nimport { FreeCameraTouchInput } from \"../Cameras/Inputs/freeCameraTouchInput\";\r\nimport type { Nullable } from \"../types\";\r\n\r\n/**\r\n * Default Inputs manager for the FreeCamera.\r\n * It groups all the default supported inputs for ease of use.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\r\n */\r\nexport class FreeCameraInputsManager extends CameraInputsManager {\r\n /**\r\n * @internal\r\n */\r\n public _mouseInput: Nullable = null;\r\n /**\r\n * @internal\r\n */\r\n public _mouseWheelInput: Nullable = null;\r\n /**\r\n * Instantiates a new FreeCameraInputsManager.\r\n * @param camera Defines the camera the inputs belong to\r\n */\r\n constructor(camera: FreeCamera) {\r\n super(camera);\r\n }\r\n\r\n /**\r\n * Add keyboard input support to the input manager.\r\n * @returns the current input manager\r\n */\r\n addKeyboard(): FreeCameraInputsManager {\r\n this.add(new FreeCameraKeyboardMoveInput());\r\n return this;\r\n }\r\n\r\n /**\r\n * Add mouse input support to the input manager.\r\n * @param touchEnabled if the FreeCameraMouseInput should support touch (default: true)\r\n * @returns the current input manager\r\n */\r\n addMouse(touchEnabled = true): FreeCameraInputsManager {\r\n if (!this._mouseInput) {\r\n this._mouseInput = new FreeCameraMouseInput(touchEnabled);\r\n this.add(this._mouseInput);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Removes the mouse input support from the manager\r\n * @returns the current input manager\r\n */\r\n removeMouse(): FreeCameraInputsManager {\r\n if (this._mouseInput) {\r\n this.remove(this._mouseInput);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Add mouse wheel input support to the input manager.\r\n * @returns the current input manager\r\n */\r\n addMouseWheel(): FreeCameraInputsManager {\r\n if (!this._mouseWheelInput) {\r\n this._mouseWheelInput = new FreeCameraMouseWheelInput();\r\n this.add(this._mouseWheelInput);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Removes the mouse wheel input support from the manager\r\n * @returns the current input manager\r\n */\r\n removeMouseWheel(): FreeCameraInputsManager {\r\n if (this._mouseWheelInput) {\r\n this.remove(this._mouseWheelInput);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Add touch input support to the input manager.\r\n * @returns the current input manager\r\n */\r\n addTouch(): FreeCameraInputsManager {\r\n this.add(new FreeCameraTouchInput());\r\n return this;\r\n }\r\n\r\n /**\r\n * Remove all attached input methods from a camera\r\n */\r\n public clear(): void {\r\n super.clear();\r\n this._mouseInput = null;\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\nimport { serializeAsVector3, serialize } from \"../Misc/decorators\";\r\nimport { Vector3, Vector2 } from \"../Maths/math.vector\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Engine } from \"../Engines/engine\";\r\nimport { TargetCamera } from \"./targetCamera\";\r\nimport { FreeCameraInputsManager } from \"./freeCameraInputsManager\";\r\nimport type { FreeCameraMouseInput } from \"../Cameras/Inputs/freeCameraMouseInput\";\r\nimport type { FreeCameraKeyboardMoveInput } from \"../Cameras/Inputs/freeCameraKeyboardMoveInput\";\r\nimport { Tools } from \"../Misc/tools\";\r\n\r\nimport type { Collider } from \"../Collisions/collider\";\r\n\r\n/**\r\n * This represents a free type of camera. It can be useful in First Person Shooter game for instance.\r\n * Please consider using the new UniversalCamera instead as it adds more functionality like the gamepad.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#universal-camera\r\n */\r\nexport class FreeCamera extends TargetCamera {\r\n /**\r\n * Define the collision ellipsoid of the camera.\r\n * This is helpful to simulate a camera body like the player body around the camera\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_collisions#arcrotatecamera\r\n */\r\n @serializeAsVector3()\r\n public ellipsoid = new Vector3(0.5, 1, 0.5);\r\n\r\n /**\r\n * Define an offset for the position of the ellipsoid around the camera.\r\n * This can be helpful to determine the center of the body near the gravity center of the body\r\n * instead of its head.\r\n */\r\n @serializeAsVector3()\r\n public ellipsoidOffset = new Vector3(0, 0, 0);\r\n\r\n /**\r\n * Enable or disable collisions of the camera with the rest of the scene objects.\r\n */\r\n @serialize()\r\n public checkCollisions = false;\r\n\r\n /**\r\n * Enable or disable gravity on the camera.\r\n */\r\n @serialize()\r\n public applyGravity = false;\r\n\r\n /**\r\n * Define the input manager associated to the camera.\r\n */\r\n public inputs: FreeCameraInputsManager;\r\n\r\n /**\r\n * Gets the input sensibility for a mouse input. (default is 2000.0)\r\n * Higher values reduce sensitivity.\r\n */\r\n public get angularSensibility(): number {\r\n const mouse = this.inputs.attached[\"mouse\"];\r\n if (mouse) {\r\n return mouse.angularSensibility;\r\n }\r\n\r\n return 0;\r\n }\r\n\r\n /**\r\n * Sets the input sensibility for a mouse input. (default is 2000.0)\r\n * Higher values reduce sensitivity.\r\n */\r\n public set angularSensibility(value: number) {\r\n const mouse = this.inputs.attached[\"mouse\"];\r\n if (mouse) {\r\n mouse.angularSensibility = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control the forward move of the camera.\r\n */\r\n public get keysUp(): number[] {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n return keyboard.keysUp;\r\n }\r\n\r\n return [];\r\n }\r\n\r\n public set keysUp(value: number[]) {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n keyboard.keysUp = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control the upward move of the camera.\r\n */\r\n public get keysUpward(): number[] {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n return keyboard.keysUpward;\r\n }\r\n\r\n return [];\r\n }\r\n\r\n public set keysUpward(value: number[]) {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n keyboard.keysUpward = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control the backward move of the camera.\r\n */\r\n public get keysDown(): number[] {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n return keyboard.keysDown;\r\n }\r\n\r\n return [];\r\n }\r\n\r\n public set keysDown(value: number[]) {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n keyboard.keysDown = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control the downward move of the camera.\r\n */\r\n public get keysDownward(): number[] {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n return keyboard.keysDownward;\r\n }\r\n\r\n return [];\r\n }\r\n\r\n public set keysDownward(value: number[]) {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n keyboard.keysDownward = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control the left strafe move of the camera.\r\n */\r\n public get keysLeft(): number[] {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n return keyboard.keysLeft;\r\n }\r\n\r\n return [];\r\n }\r\n\r\n public set keysLeft(value: number[]) {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n keyboard.keysLeft = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control the right strafe move of the camera.\r\n */\r\n public get keysRight(): number[] {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n return keyboard.keysRight;\r\n }\r\n\r\n return [];\r\n }\r\n\r\n public set keysRight(value: number[]) {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n keyboard.keysRight = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control the left rotation move of the camera.\r\n */\r\n public get keysRotateLeft(): number[] {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n return keyboard.keysRotateLeft;\r\n }\r\n\r\n return [];\r\n }\r\n\r\n public set keysRotateLeft(value: number[]) {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n keyboard.keysRotateLeft = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control the right rotation move of the camera.\r\n */\r\n public get keysRotateRight(): number[] {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n return keyboard.keysRotateRight;\r\n }\r\n\r\n return [];\r\n }\r\n\r\n public set keysRotateRight(value: number[]) {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n keyboard.keysRotateRight = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control the up rotation move of the camera.\r\n */\r\n public get keysRotateUp(): number[] {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n return keyboard.keysRotateUp;\r\n }\r\n\r\n return [];\r\n }\r\n\r\n public set keysRotateUp(value: number[]) {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n keyboard.keysRotateUp = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets or Set the list of keyboard keys used to control the down rotation move of the camera.\r\n */\r\n public get keysRotateDown(): number[] {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n return keyboard.keysRotateDown;\r\n }\r\n\r\n return [];\r\n }\r\n\r\n public set keysRotateDown(value: number[]) {\r\n const keyboard = this.inputs.attached[\"keyboard\"];\r\n if (keyboard) {\r\n keyboard.keysRotateDown = value;\r\n }\r\n }\r\n\r\n /**\r\n * Event raised when the camera collide with a mesh in the scene.\r\n */\r\n public onCollide: (collidedMesh: AbstractMesh) => void;\r\n\r\n private _collider: Collider;\r\n private _needMoveForGravity = false;\r\n private _oldPosition = Vector3.Zero();\r\n private _diffPosition = Vector3.Zero();\r\n private _newPosition = Vector3.Zero();\r\n\r\n /** @internal */\r\n public _localDirection: Vector3;\r\n /** @internal */\r\n public _transformedDirection: Vector3;\r\n\r\n /**\r\n * Instantiates a Free Camera.\r\n * This represents a free type of camera. It can be useful in First Person Shooter game for instance.\r\n * Please consider using the new UniversalCamera instead as it adds more functionality like touch to this camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#universal-camera\r\n * @param name Define the name of the camera in the scene\r\n * @param position Define the start position of the camera in the scene\r\n * @param scene Define the scene the camera belongs to\r\n * @param setActiveOnSceneIfNoneActive Defines whether the camera should be marked as active if not other active cameras have been defined\r\n */\r\n constructor(name: string, position: Vector3, scene?: Scene, setActiveOnSceneIfNoneActive = true) {\r\n super(name, position, scene, setActiveOnSceneIfNoneActive);\r\n this.inputs = new FreeCameraInputsManager(this);\r\n this.inputs.addKeyboard().addMouse();\r\n }\r\n\r\n /**\r\n * Attach the input controls to a specific dom element to get the input from.\r\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n */\r\n public attachControl(noPreventDefault?: boolean): void;\r\n /**\r\n * Attach the input controls to a specific dom element to get the input from.\r\n * @param ignored defines an ignored parameter kept for backward compatibility.\r\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n * BACK COMPAT SIGNATURE ONLY.\r\n */\r\n public attachControl(ignored: any, noPreventDefault?: boolean): void;\r\n /**\r\n * Attached controls to the current camera.\r\n * @param ignored defines an ignored parameter kept for backward compatibility.\r\n * @param noPreventDefault Defines whether event caught by the controls should call preventdefault() (https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault)\r\n */\r\n public attachControl(ignored?: any, noPreventDefault?: boolean): void {\r\n // eslint-disable-next-line prefer-rest-params\r\n noPreventDefault = Tools.BackCompatCameraNoPreventDefault(arguments);\r\n this.inputs.attachElement(noPreventDefault);\r\n }\r\n\r\n /**\r\n * Detach the current controls from the specified dom element.\r\n */\r\n public detachControl(): void {\r\n this.inputs.detachElement();\r\n\r\n this.cameraDirection = new Vector3(0, 0, 0);\r\n this.cameraRotation = new Vector2(0, 0);\r\n }\r\n\r\n // Collisions\r\n private _collisionMask = -1;\r\n\r\n /**\r\n * Define a collision mask to limit the list of object the camera can collide with\r\n */\r\n public get collisionMask(): number {\r\n return this._collisionMask;\r\n }\r\n\r\n public set collisionMask(mask: number) {\r\n this._collisionMask = !isNaN(mask) ? mask : -1;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _collideWithWorld(displacement: Vector3): void {\r\n let globalPosition: Vector3;\r\n\r\n if (this.parent) {\r\n globalPosition = Vector3.TransformCoordinates(this.position, this.parent.getWorldMatrix());\r\n } else {\r\n globalPosition = this.position;\r\n }\r\n\r\n globalPosition.subtractFromFloatsToRef(0, this.ellipsoid.y, 0, this._oldPosition);\r\n this._oldPosition.addInPlace(this.ellipsoidOffset);\r\n\r\n const coordinator = this.getScene().collisionCoordinator;\r\n if (!this._collider) {\r\n this._collider = coordinator.createCollider();\r\n }\r\n\r\n this._collider._radius = this.ellipsoid;\r\n this._collider.collisionMask = this._collisionMask;\r\n\r\n //no need for clone, as long as gravity is not on.\r\n let actualDisplacement = displacement;\r\n\r\n //add gravity to the direction to prevent the dual-collision checking\r\n if (this.applyGravity) {\r\n //this prevents mending with cameraDirection, a global variable of the free camera class.\r\n actualDisplacement = displacement.add(this.getScene().gravity);\r\n }\r\n\r\n coordinator.getNewPosition(this._oldPosition, actualDisplacement, this._collider, 3, null, this._onCollisionPositionChange, this.uniqueId);\r\n }\r\n\r\n private _onCollisionPositionChange = (collisionId: number, newPosition: Vector3, collidedMesh: Nullable = null) => {\r\n this._newPosition.copyFrom(newPosition);\r\n\r\n this._newPosition.subtractToRef(this._oldPosition, this._diffPosition);\r\n\r\n if (this._diffPosition.length() > Engine.CollisionsEpsilon) {\r\n this.position.addToRef(this._diffPosition, this._deferredPositionUpdate);\r\n if (!this._deferOnly) {\r\n this.position.copyFrom(this._deferredPositionUpdate);\r\n } else {\r\n this._deferredUpdated = true;\r\n }\r\n // call onCollide, if defined. Note that in case of deferred update, the actual position change might happen in the next frame.\r\n if (this.onCollide && collidedMesh) {\r\n this.onCollide(collidedMesh);\r\n }\r\n }\r\n };\r\n\r\n /** @internal */\r\n public _checkInputs(): void {\r\n if (!this._localDirection) {\r\n this._localDirection = Vector3.Zero();\r\n this._transformedDirection = Vector3.Zero();\r\n }\r\n\r\n this.inputs.checkInputs();\r\n\r\n super._checkInputs();\r\n }\r\n\r\n /**\r\n * Enable movement without a user input. This allows gravity to always be applied.\r\n */\r\n public set needMoveForGravity(value: boolean) {\r\n this._needMoveForGravity = value;\r\n }\r\n\r\n /**\r\n * When true, gravity is applied whether there is user input or not.\r\n */\r\n public get needMoveForGravity(): boolean {\r\n return this._needMoveForGravity;\r\n }\r\n\r\n /** @internal */\r\n public _decideIfNeedsToMove(): boolean {\r\n return this._needMoveForGravity || Math.abs(this.cameraDirection.x) > 0 || Math.abs(this.cameraDirection.y) > 0 || Math.abs(this.cameraDirection.z) > 0;\r\n }\r\n\r\n /** @internal */\r\n public _updatePosition(): void {\r\n if (this.checkCollisions && this.getScene().collisionsEnabled) {\r\n this._collideWithWorld(this.cameraDirection);\r\n } else {\r\n super._updatePosition();\r\n }\r\n }\r\n\r\n /**\r\n * Destroy the camera and release the current resources hold by it.\r\n */\r\n public dispose(): void {\r\n this.inputs.clear();\r\n super.dispose();\r\n }\r\n\r\n /**\r\n * Gets the current object class name.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"FreeCamera\";\r\n }\r\n}\r\n","import type { Nullable } from \"../../types\";\r\nimport type { ICameraInput } from \"../../Cameras/cameraInputsManager\";\r\nimport { CameraInputTypes } from \"../../Cameras/cameraInputsManager\";\r\nimport type { FreeCamera } from \"../../Cameras/freeCamera\";\r\nimport { Quaternion } from \"../../Maths/math.vector\";\r\nimport { Tools } from \"../../Misc/tools\";\r\nimport { FreeCameraInputsManager } from \"../../Cameras/freeCameraInputsManager\";\r\nimport { Observable } from \"../../Misc/observable\";\r\n\r\n// Module augmentation to abstract orientation inputs from camera.\r\ndeclare module \"../../Cameras/freeCameraInputsManager\" {\r\n export interface FreeCameraInputsManager {\r\n /**\r\n * @internal\r\n */\r\n _deviceOrientationInput: Nullable;\r\n /**\r\n * Add orientation input support to the input manager.\r\n * @param smoothFactor deviceOrientation smoothing. 0: no smoothing, 1: new data ignored, 0.9 recommended for smoothing\r\n * @returns the current input manager\r\n */\r\n addDeviceOrientation(smoothFactor?: number): FreeCameraInputsManager;\r\n }\r\n}\r\n\r\n/**\r\n * Add orientation input support to the input manager.\r\n * @param smoothFactor deviceOrientation smoothing. 0: no smoothing, 1: new data ignored, 0.9 recommended for smoothing\r\n * @returns the current input manager\r\n */\r\nFreeCameraInputsManager.prototype.addDeviceOrientation = function (smoothFactor?: number): FreeCameraInputsManager {\r\n if (!this._deviceOrientationInput) {\r\n this._deviceOrientationInput = new FreeCameraDeviceOrientationInput();\r\n if (smoothFactor) {\r\n this._deviceOrientationInput.smoothFactor = smoothFactor;\r\n }\r\n this.add(this._deviceOrientationInput);\r\n }\r\n\r\n return this;\r\n};\r\n\r\n/**\r\n * Takes information about the orientation of the device as reported by the deviceorientation event to orient the camera.\r\n * Screen rotation is taken into account.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\r\n */\r\nexport class FreeCameraDeviceOrientationInput implements ICameraInput {\r\n private _camera: FreeCamera;\r\n\r\n private _screenOrientationAngle: number = 0;\r\n\r\n private _constantTranform: Quaternion;\r\n private _screenQuaternion: Quaternion = new Quaternion();\r\n\r\n private _alpha: number = 0;\r\n private _beta: number = 0;\r\n private _gamma: number = 0;\r\n\r\n /** alpha+beta+gamma smoothing. 0: no smoothing, 1: new data ignored, 0.9 recommended for smoothing */\r\n public smoothFactor: number = 0;\r\n\r\n /**\r\n * Can be used to detect if a device orientation sensor is available on a device\r\n * @param timeout amount of time in milliseconds to wait for a response from the sensor (default: infinite)\r\n * @returns a promise that will resolve on orientation change\r\n */\r\n public static WaitForOrientationChangeAsync(timeout?: number): Promise {\r\n return new Promise((res, rej) => {\r\n let gotValue = false;\r\n const eventHandler = () => {\r\n window.removeEventListener(\"deviceorientation\", eventHandler);\r\n gotValue = true;\r\n res();\r\n };\r\n\r\n // If timeout is populated reject the promise\r\n if (timeout) {\r\n setTimeout(() => {\r\n if (!gotValue) {\r\n window.removeEventListener(\"deviceorientation\", eventHandler);\r\n rej(\"WaitForOrientationChangeAsync timed out\");\r\n }\r\n }, timeout);\r\n }\r\n\r\n if (typeof DeviceOrientationEvent !== \"undefined\" && typeof (DeviceOrientationEvent).requestPermission === \"function\") {\r\n (DeviceOrientationEvent)\r\n .requestPermission()\r\n .then((response: string) => {\r\n if (response == \"granted\") {\r\n window.addEventListener(\"deviceorientation\", eventHandler);\r\n } else {\r\n Tools.Warn(\"Permission not granted.\");\r\n }\r\n })\r\n .catch((error: any) => {\r\n Tools.Error(error);\r\n });\r\n } else {\r\n window.addEventListener(\"deviceorientation\", eventHandler);\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _onDeviceOrientationChangedObservable = new Observable();\r\n /**\r\n * Instantiates a new input\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\r\n */\r\n constructor() {\r\n this._constantTranform = new Quaternion(-Math.sqrt(0.5), 0, 0, Math.sqrt(0.5));\r\n this._orientationChanged();\r\n }\r\n\r\n /**\r\n * Define the camera controlled by the input.\r\n */\r\n public get camera(): FreeCamera {\r\n return this._camera;\r\n }\r\n\r\n public set camera(camera: FreeCamera) {\r\n this._camera = camera;\r\n if (this._camera != null && !this._camera.rotationQuaternion) {\r\n this._camera.rotationQuaternion = new Quaternion();\r\n }\r\n if (this._camera) {\r\n this._camera.onDisposeObservable.add(() => {\r\n this._onDeviceOrientationChangedObservable.clear();\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Attach the input controls to a specific dom element to get the input from.\r\n */\r\n public attachControl(): void {\r\n const hostWindow = this.camera.getScene().getEngine().getHostWindow();\r\n\r\n if (hostWindow) {\r\n const eventHandler = () => {\r\n hostWindow!.addEventListener(\"orientationchange\", this._orientationChanged);\r\n hostWindow!.addEventListener(\"deviceorientation\", this._deviceOrientation);\r\n //In certain cases, the attach control is called AFTER orientation was changed,\r\n //So this is needed.\r\n this._orientationChanged();\r\n };\r\n if (typeof DeviceOrientationEvent !== \"undefined\" && typeof (DeviceOrientationEvent).requestPermission === \"function\") {\r\n (DeviceOrientationEvent)\r\n .requestPermission()\r\n .then((response: string) => {\r\n if (response === \"granted\") {\r\n eventHandler();\r\n } else {\r\n Tools.Warn(\"Permission not granted.\");\r\n }\r\n })\r\n .catch((error: any) => {\r\n Tools.Error(error);\r\n });\r\n } else {\r\n eventHandler();\r\n }\r\n }\r\n }\r\n\r\n private _orientationChanged = () => {\r\n this._screenOrientationAngle =\r\n window.orientation !== undefined\r\n ? +(window.orientation)\r\n : (window.screen).orientation && (window.screen).orientation[\"angle\"]\r\n ? (window.screen).orientation.angle\r\n : 0;\r\n this._screenOrientationAngle = -Tools.ToRadians(this._screenOrientationAngle / 2);\r\n this._screenQuaternion.copyFromFloats(0, Math.sin(this._screenOrientationAngle), 0, Math.cos(this._screenOrientationAngle));\r\n };\r\n\r\n private _deviceOrientation = (evt: DeviceOrientationEvent) => {\r\n if (this.smoothFactor) {\r\n this._alpha = evt.alpha !== null ? Tools.SmoothAngleChange(this._alpha, evt.alpha, this.smoothFactor) : 0;\r\n this._beta = evt.beta !== null ? Tools.SmoothAngleChange(this._beta, evt.beta, this.smoothFactor) : 0;\r\n this._gamma = evt.gamma !== null ? Tools.SmoothAngleChange(this._gamma, evt.gamma, this.smoothFactor) : 0;\r\n } else {\r\n this._alpha = evt.alpha !== null ? evt.alpha : 0;\r\n this._beta = evt.beta !== null ? evt.beta : 0;\r\n this._gamma = evt.gamma !== null ? evt.gamma : 0;\r\n }\r\n\r\n if (evt.alpha !== null) {\r\n this._onDeviceOrientationChangedObservable.notifyObservers();\r\n }\r\n };\r\n\r\n /**\r\n * Detach the current controls from the specified dom element.\r\n */\r\n public detachControl(): void {\r\n window.removeEventListener(\"orientationchange\", this._orientationChanged);\r\n window.removeEventListener(\"deviceorientation\", this._deviceOrientation);\r\n this._alpha = 0;\r\n }\r\n\r\n /**\r\n * Update the current camera state depending on the inputs that have been used this frame.\r\n * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop.\r\n */\r\n public checkInputs(): void {\r\n //if no device orientation provided, don't update the rotation.\r\n //Only testing against alpha under the assumption thatnorientation will never be so exact when set.\r\n if (!this._alpha) {\r\n return;\r\n }\r\n Quaternion.RotationYawPitchRollToRef(Tools.ToRadians(this._alpha), Tools.ToRadians(this._beta), -Tools.ToRadians(this._gamma), this.camera.rotationQuaternion);\r\n this._camera.rotationQuaternion.multiplyInPlace(this._screenQuaternion);\r\n this._camera.rotationQuaternion.multiplyInPlace(this._constantTranform);\r\n //Mirror on XY Plane\r\n this._camera.rotationQuaternion.z *= -1;\r\n this._camera.rotationQuaternion.w *= -1;\r\n }\r\n\r\n /**\r\n * Gets the class name of the current input.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"FreeCameraDeviceOrientationInput\";\r\n }\r\n\r\n /**\r\n * Get the friendly name associated with the input class.\r\n * @returns the input friendly name\r\n */\r\n public getSimpleName(): string {\r\n return \"deviceOrientation\";\r\n }\r\n}\r\n\r\n(CameraInputTypes)[\"FreeCameraDeviceOrientationInput\"] = FreeCameraDeviceOrientationInput;\r\n","import { FreeCamera } from \"./freeCamera\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Quaternion, Vector3 } from \"../Maths/math.vector\";\r\nimport { Node } from \"../node\";\r\n\r\nimport \"./Inputs/freeCameraDeviceOrientationInput\";\r\nimport { Axis } from \"../Maths/math.axis\";\r\n\r\nNode.AddNodeConstructor(\"DeviceOrientationCamera\", (name, scene) => {\r\n return () => new DeviceOrientationCamera(name, Vector3.Zero(), scene);\r\n});\r\n\r\n// We're mainly based on the logic defined into the FreeCamera code\r\n/**\r\n * This is a camera specifically designed to react to device orientation events such as a modern mobile device\r\n * being tilted forward or back and left or right.\r\n */\r\nexport class DeviceOrientationCamera extends FreeCamera {\r\n private _initialQuaternion: Quaternion;\r\n private _quaternionCache: Quaternion;\r\n private _tmpDragQuaternion = new Quaternion();\r\n private _disablePointerInputWhenUsingDeviceOrientation = true;\r\n\r\n /**\r\n * Creates a new device orientation camera\r\n * @param name The name of the camera\r\n * @param position The start position camera\r\n * @param scene The scene the camera belongs to\r\n */\r\n constructor(name: string, position: Vector3, scene?: Scene) {\r\n super(name, position, scene);\r\n this._quaternionCache = new Quaternion();\r\n this.inputs.addDeviceOrientation();\r\n\r\n // When the orientation sensor fires it's first event, disable mouse input\r\n if (this.inputs._deviceOrientationInput) {\r\n this.inputs._deviceOrientationInput._onDeviceOrientationChangedObservable.addOnce(() => {\r\n if (this._disablePointerInputWhenUsingDeviceOrientation) {\r\n if (this.inputs._mouseInput) {\r\n this.inputs._mouseInput._allowCameraRotation = false;\r\n this.inputs._mouseInput.onPointerMovedObservable.add((e) => {\r\n if (this._dragFactor != 0) {\r\n if (!this._initialQuaternion) {\r\n this._initialQuaternion = new Quaternion();\r\n }\r\n // Rotate the initial space around the y axis to allow users to \"turn around\" via touch/mouse\r\n Quaternion.FromEulerAnglesToRef(0, e.offsetX * this._dragFactor, 0, this._tmpDragQuaternion);\r\n this._initialQuaternion.multiplyToRef(this._tmpDragQuaternion, this._initialQuaternion);\r\n }\r\n });\r\n }\r\n }\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating that pointer input must be disabled on first orientation sensor update (Default: true)\r\n */\r\n public get disablePointerInputWhenUsingDeviceOrientation() {\r\n return this._disablePointerInputWhenUsingDeviceOrientation;\r\n }\r\n\r\n public set disablePointerInputWhenUsingDeviceOrientation(value: boolean) {\r\n this._disablePointerInputWhenUsingDeviceOrientation = value;\r\n }\r\n\r\n private _dragFactor = 0;\r\n /**\r\n * Enabled turning on the y axis when the orientation sensor is active\r\n * @param dragFactor the factor that controls the turn speed (default: 1/300)\r\n */\r\n public enableHorizontalDragging(dragFactor = 1 / 300) {\r\n this._dragFactor = dragFactor;\r\n }\r\n\r\n /**\r\n * Gets the current instance class name (\"DeviceOrientationCamera\").\r\n * This helps avoiding instanceof at run time.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"DeviceOrientationCamera\";\r\n }\r\n\r\n /**\r\n * @internal\r\n * Checks and applies the current values of the inputs to the camera. (Internal use only)\r\n */\r\n public _checkInputs(): void {\r\n super._checkInputs();\r\n this._quaternionCache.copyFrom(this.rotationQuaternion);\r\n if (this._initialQuaternion) {\r\n this._initialQuaternion.multiplyToRef(this.rotationQuaternion, this.rotationQuaternion);\r\n }\r\n }\r\n\r\n /**\r\n * Reset the camera to its default orientation on the specified axis only.\r\n * @param axis The axis to reset\r\n */\r\n public resetToCurrentRotation(axis: Axis = Axis.Y): void {\r\n //can only work if this camera has a rotation quaternion already.\r\n if (!this.rotationQuaternion) {\r\n return;\r\n }\r\n\r\n if (!this._initialQuaternion) {\r\n this._initialQuaternion = new Quaternion();\r\n }\r\n\r\n this._initialQuaternion.copyFrom(this._quaternionCache || this.rotationQuaternion);\r\n\r\n [\"x\", \"y\", \"z\"].forEach((axisName) => {\r\n if (!(axis)[axisName]) {\r\n (this._initialQuaternion)[axisName] = 0;\r\n } else {\r\n (this._initialQuaternion)[axisName] *= -1;\r\n }\r\n });\r\n this._initialQuaternion.normalize();\r\n //force rotation update\r\n this._initialQuaternion.multiplyToRef(this.rotationQuaternion, this.rotationQuaternion);\r\n }\r\n}\r\n","import { Matrix } from \"../../Maths/math.vector\";\r\n/**\r\n * This represents all the required metrics to create a VR camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#device-orientation-camera\r\n */\r\nexport class VRCameraMetrics {\r\n /**\r\n * Define the horizontal resolution off the screen.\r\n */\r\n public hResolution: number;\r\n /**\r\n * Define the vertical resolution off the screen.\r\n */\r\n public vResolution: number;\r\n /**\r\n * Define the horizontal screen size.\r\n */\r\n public hScreenSize: number;\r\n /**\r\n * Define the vertical screen size.\r\n */\r\n public vScreenSize: number;\r\n /**\r\n * Define the vertical screen center position.\r\n */\r\n public vScreenCenter: number;\r\n /**\r\n * Define the distance of the eyes to the screen.\r\n */\r\n public eyeToScreenDistance: number;\r\n /**\r\n * Define the distance between both lenses\r\n */\r\n public lensSeparationDistance: number;\r\n /**\r\n * Define the distance between both viewer's eyes.\r\n */\r\n public interpupillaryDistance: number;\r\n /**\r\n * Define the distortion factor of the VR postprocess.\r\n * Please, touch with care.\r\n */\r\n public distortionK: number[];\r\n /**\r\n * Define the chromatic aberration correction factors for the VR post process.\r\n */\r\n public chromaAbCorrection: number[];\r\n /**\r\n * Define the scale factor of the post process.\r\n * The smaller the better but the slower.\r\n */\r\n public postProcessScaleFactor: number;\r\n /**\r\n * Define an offset for the lens center.\r\n */\r\n public lensCenterOffset: number;\r\n /**\r\n * Define if the current vr camera should compensate the distortion of the lens or not.\r\n */\r\n public compensateDistortion = true;\r\n\r\n /**\r\n * Defines if multiview should be enabled when rendering (Default: false)\r\n */\r\n public multiviewEnabled = false;\r\n\r\n /**\r\n * Gets the rendering aspect ratio based on the provided resolutions.\r\n */\r\n public get aspectRatio(): number {\r\n return this.hResolution / (2 * this.vResolution);\r\n }\r\n\r\n /**\r\n * Gets the aspect ratio based on the FOV, scale factors, and real screen sizes.\r\n */\r\n public get aspectRatioFov(): number {\r\n return 2 * Math.atan((this.postProcessScaleFactor * this.vScreenSize) / (2 * this.eyeToScreenDistance));\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public get leftHMatrix(): Matrix {\r\n const meters = this.hScreenSize / 4 - this.lensSeparationDistance / 2;\r\n const h = (4 * meters) / this.hScreenSize;\r\n\r\n return Matrix.Translation(h, 0, 0);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public get rightHMatrix(): Matrix {\r\n const meters = this.hScreenSize / 4 - this.lensSeparationDistance / 2;\r\n const h = (4 * meters) / this.hScreenSize;\r\n\r\n return Matrix.Translation(-h, 0, 0);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public get leftPreViewMatrix(): Matrix {\r\n return Matrix.Translation(0.5 * this.interpupillaryDistance, 0, 0);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public get rightPreViewMatrix(): Matrix {\r\n return Matrix.Translation(-0.5 * this.interpupillaryDistance, 0, 0);\r\n }\r\n\r\n /**\r\n * Get the default VRMetrics based on the most generic setup.\r\n * @returns the default vr metrics\r\n */\r\n public static GetDefault(): VRCameraMetrics {\r\n const result = new VRCameraMetrics();\r\n\r\n result.hResolution = 1280;\r\n result.vResolution = 800;\r\n result.hScreenSize = 0.149759993;\r\n result.vScreenSize = 0.0935999975;\r\n result.vScreenCenter = 0.0467999987;\r\n result.eyeToScreenDistance = 0.0410000011;\r\n result.lensSeparationDistance = 0.063500002;\r\n result.interpupillaryDistance = 0.064000003;\r\n result.distortionK = [1.0, 0.219999999, 0.239999995, 0.0];\r\n result.chromaAbCorrection = [0.995999992, -0.00400000019, 1.01400006, 0.0];\r\n result.postProcessScaleFactor = 1.714605507808412;\r\n result.lensCenterOffset = 0.151976421;\r\n\r\n return result;\r\n }\r\n}\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\n\nconst name = \"vrDistortionCorrectionPixelShader\";\nconst shader = `varying vec2 vUV;uniform sampler2D textureSampler;uniform vec2 LensCenter;uniform vec2 Scale;uniform vec2 ScaleIn;uniform vec4 HmdWarpParam;vec2 HmdWarp(vec2 in01) {vec2 theta=(in01-LensCenter)*ScaleIn; \nfloat rSq=theta.x*theta.x+theta.y*theta.y;vec2 rvector=theta*(HmdWarpParam.x+HmdWarpParam.y*rSq+HmdWarpParam.z*rSq*rSq+HmdWarpParam.w*rSq*rSq*rSq);return LensCenter+Scale*rvector;}\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{vec2 tc=HmdWarp(vUV);if (tc.x <0.0 || tc.x>1.0 || tc.y<0.0 || tc.y>1.0)\ngl_FragColor=vec4(0.0,0.0,0.0,0.0);else{gl_FragColor=texture2D(textureSampler,tc);}}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const vrDistortionCorrectionPixelShader = { name, shader };\n","import { Vector2 } from \"../Maths/math.vector\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\nimport type { VRCameraMetrics } from \"../Cameras/VR/vrCameraMetrics\";\r\nimport type { Effect } from \"../Materials/effect\";\r\nimport { Texture } from \"../Materials/Textures/texture\";\r\nimport { PostProcess } from \"./postProcess\";\r\n\r\nimport \"../Shaders/vrDistortionCorrection.fragment\";\r\nimport type { Nullable } from \"../types\";\r\n\r\n/**\r\n * VRDistortionCorrectionPostProcess used for mobile VR\r\n */\r\nexport class VRDistortionCorrectionPostProcess extends PostProcess {\r\n private _isRightEye: boolean;\r\n private _distortionFactors: number[];\r\n private _postProcessScaleFactor: number;\r\n private _lensCenterOffset: number;\r\n private _scaleIn: Vector2;\r\n private _scaleFactor: Vector2;\r\n private _lensCenter: Vector2;\r\n\r\n /**\r\n * Gets a string identifying the name of the class\r\n * @returns \"VRDistortionCorrectionPostProcess\" string\r\n */\r\n public getClassName(): string {\r\n return \"VRDistortionCorrectionPostProcess\";\r\n }\r\n\r\n /**\r\n * Initializes the VRDistortionCorrectionPostProcess\r\n * @param name The name of the effect.\r\n * @param camera The camera to apply the render pass to.\r\n * @param isRightEye If this is for the right eye distortion\r\n * @param vrMetrics All the required metrics for the VR camera\r\n */\r\n constructor(name: string, camera: Nullable, isRightEye: boolean, vrMetrics: VRCameraMetrics) {\r\n super(name, \"vrDistortionCorrection\", [\"LensCenter\", \"Scale\", \"ScaleIn\", \"HmdWarpParam\"], null, vrMetrics.postProcessScaleFactor, camera, Texture.BILINEAR_SAMPLINGMODE);\r\n\r\n this._isRightEye = isRightEye;\r\n this._distortionFactors = vrMetrics.distortionK;\r\n this._postProcessScaleFactor = vrMetrics.postProcessScaleFactor;\r\n this._lensCenterOffset = vrMetrics.lensCenterOffset;\r\n this.adaptScaleToCurrentViewport = true;\r\n\r\n this.onSizeChangedObservable.add(() => {\r\n this._scaleIn = new Vector2(2, 2 / this.aspectRatio);\r\n this._scaleFactor = new Vector2(0.5 * (1 / this._postProcessScaleFactor), 0.5 * (1 / this._postProcessScaleFactor) * this.aspectRatio);\r\n this._lensCenter = new Vector2(this._isRightEye ? 0.5 - this._lensCenterOffset * 0.5 : 0.5 + this._lensCenterOffset * 0.5, 0.5);\r\n });\r\n this.onApplyObservable.add((effect: Effect) => {\r\n effect.setFloat2(\"LensCenter\", this._lensCenter.x, this._lensCenter.y);\r\n effect.setFloat2(\"Scale\", this._scaleFactor.x, this._scaleFactor.y);\r\n effect.setFloat2(\"ScaleIn\", this._scaleIn.x, this._scaleIn.y);\r\n effect.setFloat4(\"HmdWarpParam\", this._distortionFactors[0], this._distortionFactors[1], this._distortionFactors[2], this._distortionFactors[3]);\r\n });\r\n }\r\n}\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\n\nconst name = \"vrMultiviewToSingleviewPixelShader\";\nconst shader = `precision mediump sampler2DArray;varying vec2 vUV;uniform sampler2DArray multiviewSampler;uniform int imageIndex;\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{gl_FragColor=texture2D(multiviewSampler,vec3(vUV,imageIndex));}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const vrMultiviewToSingleviewPixelShader = { name, shader };\n","import { RenderTargetTexture } from \"../Textures/renderTargetTexture\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { Constants } from \"../../Engines/constants\";\r\n\r\n/**\r\n * Renders to multiple views with a single draw call\r\n * @see https://www.khronos.org/registry/webgl/extensions/OVR_multiview2/\r\n */\r\nexport class MultiviewRenderTarget extends RenderTargetTexture {\r\n public set samples(value: number) {\r\n // We override this setter because multisampling is handled by framebufferTextureMultisampleMultiviewOVR\r\n this._samples = value;\r\n }\r\n\r\n public get samples(): number {\r\n return this._samples;\r\n }\r\n\r\n /**\r\n * Creates a multiview render target\r\n * @param scene scene used with the render target\r\n * @param size the size of the render target (used for each view)\r\n */\r\n constructor(scene?: Scene, size: number | { width: number; height: number } | { ratio: number } = 512) {\r\n super(\"multiview rtt\", size, scene, false, true, Constants.TEXTURETYPE_UNSIGNED_INT, false, undefined, false, false, true, undefined, true);\r\n this._renderTarget = this.getScene()!.getEngine().createMultiviewRenderTargetTexture(this.getRenderWidth(), this.getRenderHeight());\r\n this._texture = this._renderTarget.texture!;\r\n this._texture.isMultiview = true;\r\n this._texture.format = Constants.TEXTUREFORMAT_RGBA;\r\n this.samples = this._getEngine()!.getCaps().maxSamples || this.samples;\r\n this._texture.samples = this._samples;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _bindFrameBuffer() {\r\n if (!this._renderTarget) {\r\n return;\r\n }\r\n this.getScene()!.getEngine().bindMultiviewFramebuffer(this._renderTarget);\r\n }\r\n\r\n /**\r\n * Gets the number of views the corresponding to the texture (eg. a MultiviewRenderTarget will have > 1)\r\n * @returns the view count\r\n */\r\n public getViewCount() {\r\n return 2;\r\n }\r\n}\r\n","import { Camera } from \"../../Cameras/camera\";\r\nimport { Engine } from \"../../Engines/engine\";\r\nimport { Scene } from \"../../scene\";\r\nimport { InternalTexture, InternalTextureSource } from \"../../Materials/Textures/internalTexture\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { RenderTargetTexture } from \"../../Materials/Textures/renderTargetTexture\";\r\nimport { Matrix, TmpVectors } from \"../../Maths/math.vector\";\r\nimport { UniformBuffer } from \"../../Materials/uniformBuffer\";\r\nimport { MultiviewRenderTarget } from \"../../Materials/Textures/MultiviewRenderTarget\";\r\nimport { Frustum } from \"../../Maths/math.frustum\";\r\nimport type { WebGLRenderTargetWrapper } from \"../WebGL/webGLRenderTargetWrapper\";\r\nimport type { RenderTargetWrapper } from \"../renderTargetWrapper\";\r\n\r\ndeclare module \"../../Engines/engine\" {\r\n export interface Engine {\r\n /**\r\n * Creates a new multiview render target\r\n * @param width defines the width of the texture\r\n * @param height defines the height of the texture\r\n * @returns the created multiview render target wrapper\r\n */\r\n createMultiviewRenderTargetTexture(width: number, height: number, colorTexture?: WebGLTexture, depthStencilTexture?: WebGLTexture): RenderTargetWrapper;\r\n\r\n /**\r\n * Binds a multiview render target wrapper to be drawn to\r\n * @param multiviewTexture render target wrapper to bind\r\n */\r\n bindMultiviewFramebuffer(multiviewTexture: RenderTargetWrapper): void;\r\n\r\n /**\r\n * Binds a Space Warp render target wrapper to be drawn to\r\n * @param spaceWarpTexture render target wrapper to bind\r\n */\r\n bindSpaceWarpFramebuffer(spaceWarpTexture: RenderTargetWrapper): void;\r\n }\r\n}\r\n\r\nEngine.prototype.createMultiviewRenderTargetTexture = function (width: number, height: number, colorTexture?: WebGLTexture, depthStencilTexture?: WebGLTexture) {\r\n const gl = this._gl;\r\n\r\n if (!this.getCaps().multiview) {\r\n throw \"Multiview is not supported\";\r\n }\r\n\r\n const rtWrapper = this._createHardwareRenderTargetWrapper(false, false, { width, height }) as WebGLRenderTargetWrapper;\r\n\r\n rtWrapper._framebuffer = gl.createFramebuffer();\r\n\r\n const internalTexture = new InternalTexture(this, InternalTextureSource.Unknown, true);\r\n internalTexture.width = width;\r\n internalTexture.height = height;\r\n internalTexture.isMultiview = true;\r\n\r\n if (!colorTexture) {\r\n colorTexture = gl.createTexture() as WebGLTexture;\r\n gl.bindTexture(gl.TEXTURE_2D_ARRAY, colorTexture);\r\n (gl as any).texStorage3D(gl.TEXTURE_2D_ARRAY, 1, gl.RGBA8, width, height, 2);\r\n }\r\n\r\n rtWrapper._colorTextureArray = colorTexture;\r\n\r\n if (!depthStencilTexture) {\r\n depthStencilTexture = gl.createTexture() as WebGLTexture;\r\n gl.bindTexture(gl.TEXTURE_2D_ARRAY, depthStencilTexture);\r\n (gl as any).texStorage3D(gl.TEXTURE_2D_ARRAY, 1, (gl as any).DEPTH24_STENCIL8, width, height, 2);\r\n }\r\n\r\n rtWrapper._depthStencilTextureArray = depthStencilTexture;\r\n\r\n internalTexture.isReady = true;\r\n\r\n rtWrapper.setTextures(internalTexture);\r\n rtWrapper._depthStencilTexture = internalTexture;\r\n\r\n return rtWrapper;\r\n};\r\n\r\nEngine.prototype.bindMultiviewFramebuffer = function (_multiviewTexture: RenderTargetWrapper) {\r\n const multiviewTexture = _multiviewTexture as WebGLRenderTargetWrapper;\r\n\r\n const gl: any = this._gl;\r\n const ext = this.getCaps().oculusMultiview || this.getCaps().multiview;\r\n\r\n this.bindFramebuffer(multiviewTexture, undefined, undefined, undefined, true);\r\n gl.bindFramebuffer(gl.DRAW_FRAMEBUFFER, multiviewTexture._framebuffer);\r\n if (multiviewTexture._colorTextureArray && multiviewTexture._depthStencilTextureArray) {\r\n if (this.getCaps().oculusMultiview) {\r\n ext.framebufferTextureMultisampleMultiviewOVR(gl.DRAW_FRAMEBUFFER, gl.COLOR_ATTACHMENT0, multiviewTexture._colorTextureArray, 0, multiviewTexture.samples, 0, 2);\r\n ext.framebufferTextureMultisampleMultiviewOVR(\r\n gl.DRAW_FRAMEBUFFER,\r\n gl.DEPTH_STENCIL_ATTACHMENT,\r\n multiviewTexture._depthStencilTextureArray,\r\n 0,\r\n multiviewTexture.samples,\r\n 0,\r\n 2\r\n );\r\n } else {\r\n ext.framebufferTextureMultiviewOVR(gl.DRAW_FRAMEBUFFER, gl.COLOR_ATTACHMENT0, multiviewTexture._colorTextureArray, 0, 0, 2);\r\n ext.framebufferTextureMultiviewOVR(gl.DRAW_FRAMEBUFFER, gl.DEPTH_STENCIL_ATTACHMENT, multiviewTexture._depthStencilTextureArray, 0, 0, 2);\r\n }\r\n } else {\r\n throw \"Invalid multiview frame buffer\";\r\n }\r\n};\r\n\r\nEngine.prototype.bindSpaceWarpFramebuffer = function (_spaceWarpTexture: RenderTargetWrapper) {\r\n const spaceWarpTexture = _spaceWarpTexture as WebGLRenderTargetWrapper;\r\n\r\n const gl: any = this._gl;\r\n const ext = this.getCaps().oculusMultiview || this.getCaps().multiview;\r\n\r\n this.bindFramebuffer(spaceWarpTexture, undefined, undefined, undefined, true);\r\n gl.bindFramebuffer(gl.DRAW_FRAMEBUFFER, spaceWarpTexture._framebuffer);\r\n if (spaceWarpTexture._colorTextureArray && spaceWarpTexture._depthStencilTextureArray) {\r\n ext.framebufferTextureMultiviewOVR(gl.DRAW_FRAMEBUFFER, gl.COLOR_ATTACHMENT0, spaceWarpTexture._colorTextureArray, 0, 0, 2);\r\n ext.framebufferTextureMultiviewOVR(gl.DRAW_FRAMEBUFFER, gl.DEPTH_ATTACHMENT, spaceWarpTexture._depthStencilTextureArray, 0, 0, 2);\r\n } else {\r\n throw new Error(\"Invalid Space Warp framebuffer\");\r\n }\r\n};\r\n\r\ndeclare module \"../../Cameras/camera\" {\r\n export interface Camera {\r\n /**\r\n * @internal\r\n * For cameras that cannot use multiview images to display directly. (e.g. webVR camera will render to multiview texture, then copy to each eye texture and go from there)\r\n */\r\n _useMultiviewToSingleView: boolean;\r\n /**\r\n * @internal\r\n * For cameras that cannot use multiview images to display directly. (e.g. webVR camera will render to multiview texture, then copy to each eye texture and go from there)\r\n */\r\n _multiviewTexture: Nullable;\r\n\r\n /**\r\n * @internal\r\n * For WebXR cameras that are rendering to multiview texture arrays.\r\n */\r\n _renderingMultiview: boolean;\r\n\r\n /**\r\n * @internal\r\n * ensures the multiview texture of the camera exists and has the specified width/height\r\n * @param width height to set on the multiview texture\r\n * @param height width to set on the multiview texture\r\n */\r\n _resizeOrCreateMultiviewTexture(width: number, height: number): void;\r\n }\r\n}\r\n\r\nCamera.prototype._useMultiviewToSingleView = false;\r\n\r\nCamera.prototype._multiviewTexture = null;\r\n\r\nCamera.prototype._resizeOrCreateMultiviewTexture = function (width: number, height: number) {\r\n if (!this._multiviewTexture) {\r\n this._multiviewTexture = new MultiviewRenderTarget(this.getScene(), { width: width, height: height });\r\n } else if (this._multiviewTexture.getRenderWidth() != width || this._multiviewTexture.getRenderHeight() != height) {\r\n this._multiviewTexture.dispose();\r\n this._multiviewTexture = new MultiviewRenderTarget(this.getScene(), { width: width, height: height });\r\n }\r\n};\r\n\r\ndeclare module \"../../scene\" {\r\n export interface Scene {\r\n /** @internal */\r\n _transformMatrixR: Matrix;\r\n /** @internal */\r\n _multiviewSceneUbo: Nullable;\r\n /** @internal */\r\n _createMultiviewUbo(): void;\r\n /** @internal */\r\n _updateMultiviewUbo(viewR?: Matrix, projectionR?: Matrix): void;\r\n /** @internal */\r\n _renderMultiviewToSingleView(camera: Camera): void;\r\n }\r\n}\r\n\r\nfunction createMultiviewUbo(engine: Engine, name?: string) {\r\n const ubo = new UniformBuffer(engine, undefined, true, name);\r\n ubo.addUniform(\"viewProjection\", 16);\r\n ubo.addUniform(\"viewProjectionR\", 16);\r\n ubo.addUniform(\"view\", 16);\r\n ubo.addUniform(\"projection\", 16);\r\n ubo.addUniform(\"vEyePosition\", 4);\r\n return ubo;\r\n}\r\n\r\nconst currentCreateSceneUniformBuffer = Scene.prototype.createSceneUniformBuffer;\r\n\r\nScene.prototype._transformMatrixR = Matrix.Zero();\r\nScene.prototype._multiviewSceneUbo = null;\r\nScene.prototype._createMultiviewUbo = function () {\r\n this._multiviewSceneUbo = createMultiviewUbo(this.getEngine(), \"scene_multiview\");\r\n};\r\nScene.prototype.createSceneUniformBuffer = function (name?: string): UniformBuffer {\r\n if (this._multiviewSceneUbo) {\r\n return createMultiviewUbo(this.getEngine(), name);\r\n }\r\n return currentCreateSceneUniformBuffer.bind(this)(name);\r\n};\r\nScene.prototype._updateMultiviewUbo = function (viewR?: Matrix, projectionR?: Matrix) {\r\n if (viewR && projectionR) {\r\n viewR.multiplyToRef(projectionR, this._transformMatrixR);\r\n }\r\n\r\n if (viewR && projectionR) {\r\n viewR.multiplyToRef(projectionR, TmpVectors.Matrix[0]);\r\n Frustum.GetRightPlaneToRef(TmpVectors.Matrix[0], this._frustumPlanes[3]); // Replace right plane by second camera right plane\r\n }\r\n\r\n if (this._multiviewSceneUbo) {\r\n this._multiviewSceneUbo.updateMatrix(\"viewProjection\", this.getTransformMatrix());\r\n this._multiviewSceneUbo.updateMatrix(\"viewProjectionR\", this._transformMatrixR);\r\n this._multiviewSceneUbo.updateMatrix(\"view\", this._viewMatrix);\r\n this._multiviewSceneUbo.updateMatrix(\"projection\", this._projectionMatrix);\r\n }\r\n};\r\nScene.prototype._renderMultiviewToSingleView = function (camera: Camera) {\r\n // Multiview is only able to be displayed directly for API's such as webXR\r\n // This displays a multiview image by rendering to the multiview image and then\r\n // copying the result into the sub cameras instead of rendering them and proceeding as normal from there\r\n\r\n // Render to a multiview texture\r\n camera._resizeOrCreateMultiviewTexture(\r\n camera._rigPostProcess && camera._rigPostProcess && camera._rigPostProcess.width > 0 ? camera._rigPostProcess.width : this.getEngine().getRenderWidth(true),\r\n camera._rigPostProcess && camera._rigPostProcess && camera._rigPostProcess.height > 0 ? camera._rigPostProcess.height : this.getEngine().getRenderHeight(true)\r\n );\r\n if (!this._multiviewSceneUbo) {\r\n this._createMultiviewUbo();\r\n }\r\n camera.outputRenderTarget = camera._multiviewTexture;\r\n this._renderForCamera(camera);\r\n camera.outputRenderTarget = null;\r\n\r\n // Consume the multiview texture through a shader for each eye\r\n for (let index = 0; index < camera._rigCameras.length; index++) {\r\n const engine = this.getEngine();\r\n this._activeCamera = camera._rigCameras[index];\r\n engine.setViewport(this._activeCamera.viewport);\r\n if (this.postProcessManager) {\r\n this.postProcessManager._prepareFrame();\r\n this.postProcessManager._finalizeFrame(this._activeCamera.isIntermediate);\r\n }\r\n }\r\n};\r\n","import type { Camera } from \"../Cameras/camera\";\r\nimport type { Effect } from \"../Materials/effect\";\r\nimport { Texture } from \"../Materials/Textures/texture\";\r\nimport { PostProcess } from \"./postProcess\";\r\n\r\nimport \"../Shaders/vrMultiviewToSingleview.fragment\";\r\nimport \"../Engines/Extensions/engine.multiview\";\r\nimport type { Nullable } from \"../types\";\r\n\r\n/**\r\n * VRMultiviewToSingleview used to convert multiview texture arrays to standard textures for scenarios such as webVR\r\n * This will not be used for webXR as it supports displaying texture arrays directly\r\n */\r\nexport class VRMultiviewToSingleviewPostProcess extends PostProcess {\r\n /**\r\n * Gets a string identifying the name of the class\r\n * @returns \"VRMultiviewToSingleviewPostProcess\" string\r\n */\r\n public getClassName(): string {\r\n return \"VRMultiviewToSingleviewPostProcess\";\r\n }\r\n\r\n /**\r\n * Initializes a VRMultiviewToSingleview\r\n * @param name name of the post process\r\n * @param camera camera to be applied to\r\n * @param scaleFactor scaling factor to the size of the output texture\r\n */\r\n constructor(name: string, camera: Nullable, scaleFactor: number) {\r\n super(name, \"vrMultiviewToSingleview\", [\"imageIndex\"], [\"multiviewSampler\"], scaleFactor, camera, Texture.BILINEAR_SAMPLINGMODE);\r\n\r\n const cam = camera ?? this.getCamera();\r\n this.onSizeChangedObservable.add(() => {});\r\n this.onApplyObservable.add((effect: Effect) => {\r\n if (cam._scene.activeCamera && cam._scene.activeCamera.isLeftCamera) {\r\n effect.setInt(\"imageIndex\", 0);\r\n } else {\r\n effect.setInt(\"imageIndex\", 1);\r\n }\r\n effect.setTexture(\"multiviewSampler\", cam._multiviewTexture);\r\n });\r\n }\r\n}\r\n","import type { Camera } from \"../camera\";\r\nimport { Matrix } from \"../../Maths/math.vector\";\r\nimport { VRDistortionCorrectionPostProcess } from \"../../PostProcesses/vrDistortionCorrectionPostProcess\";\r\nimport { VRMultiviewToSingleviewPostProcess } from \"../../PostProcesses/vrMultiviewToSingleviewPostProcess\";\r\nimport { VRCameraMetrics } from \"../VR/vrCameraMetrics\";\r\nimport { Logger } from \"../../Misc/logger\";\r\nimport { Viewport } from \"../../Maths/math.viewport\";\r\n\r\n/**\r\n * @internal\r\n */\r\nexport function setVRRigMode(camera: Camera, rigParams: any) {\r\n const metrics = rigParams.vrCameraMetrics || VRCameraMetrics.GetDefault();\r\n\r\n camera._rigCameras[0]._cameraRigParams.vrMetrics = metrics;\r\n camera._rigCameras[0].viewport = new Viewport(0, 0, 0.5, 1.0);\r\n camera._rigCameras[0]._cameraRigParams.vrWorkMatrix = new Matrix();\r\n camera._rigCameras[0]._cameraRigParams.vrHMatrix = metrics.leftHMatrix;\r\n camera._rigCameras[0]._cameraRigParams.vrPreViewMatrix = metrics.leftPreViewMatrix;\r\n camera._rigCameras[0].getProjectionMatrix = camera._rigCameras[0]._getVRProjectionMatrix;\r\n\r\n camera._rigCameras[1]._cameraRigParams.vrMetrics = metrics;\r\n camera._rigCameras[1].viewport = new Viewport(0.5, 0, 0.5, 1.0);\r\n camera._rigCameras[1]._cameraRigParams.vrWorkMatrix = new Matrix();\r\n camera._rigCameras[1]._cameraRigParams.vrHMatrix = metrics.rightHMatrix;\r\n camera._rigCameras[1]._cameraRigParams.vrPreViewMatrix = metrics.rightPreViewMatrix;\r\n camera._rigCameras[1].getProjectionMatrix = camera._rigCameras[1]._getVRProjectionMatrix;\r\n\r\n // For multiview on a webVR camera\r\n // First multiview will be rendered to camera._multiviewTexture\r\n // Then this postprocess will run on each eye to copy the right texture to each eye\r\n if (metrics.multiviewEnabled) {\r\n if (!camera.getScene().getEngine().getCaps().multiview) {\r\n Logger.Warn(\"Multiview is not supported, falling back to standard rendering\");\r\n metrics.multiviewEnabled = false;\r\n } else {\r\n camera._useMultiviewToSingleView = true;\r\n camera._rigPostProcess = new VRMultiviewToSingleviewPostProcess(\"VRMultiviewToSingleview\", camera, metrics.postProcessScaleFactor);\r\n }\r\n }\r\n\r\n if (metrics.compensateDistortion) {\r\n camera._rigCameras[0]._rigPostProcess = new VRDistortionCorrectionPostProcess(\"VR_Distort_Compensation_Left\", camera._rigCameras[0], false, metrics);\r\n camera._rigCameras[1]._rigPostProcess = new VRDistortionCorrectionPostProcess(\"VR_Distort_Compensation_Right\", camera._rigCameras[1], true, metrics);\r\n }\r\n}\r\n","import { Camera } from \"../../Cameras/camera\";\r\nimport { DeviceOrientationCamera } from \"../../Cameras/deviceOrientationCamera\";\r\nimport { VRCameraMetrics } from \"./vrCameraMetrics\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { Vector3 } from \"../../Maths/math.vector\";\r\nimport { Node } from \"../../node\";\r\nimport { setVRRigMode } from \"../RigModes/vrRigMode\";\r\n\r\nNode.AddNodeConstructor(\"VRDeviceOrientationFreeCamera\", (name, scene) => {\r\n return () => new VRDeviceOrientationFreeCamera(name, Vector3.Zero(), scene);\r\n});\r\n\r\n/**\r\n * Camera used to simulate VR rendering (based on FreeCamera)\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#vr-device-orientation-cameras\r\n */\r\nexport class VRDeviceOrientationFreeCamera extends DeviceOrientationCamera {\r\n /**\r\n * Creates a new VRDeviceOrientationFreeCamera\r\n * @param name defines camera name\r\n * @param position defines the start position of the camera\r\n * @param scene defines the scene the camera belongs to\r\n * @param compensateDistortion defines if the camera needs to compensate the lens distortion\r\n * @param vrCameraMetrics defines the vr metrics associated to the camera\r\n */\r\n constructor(name: string, position: Vector3, scene?: Scene, compensateDistortion = true, vrCameraMetrics: VRCameraMetrics = VRCameraMetrics.GetDefault()) {\r\n super(name, position, scene);\r\n\r\n vrCameraMetrics.compensateDistortion = compensateDistortion;\r\n this.setCameraRigMode(Camera.RIG_MODE_VR, { vrCameraMetrics: vrCameraMetrics });\r\n }\r\n\r\n /**\r\n * Gets camera class name\r\n * @returns VRDeviceOrientationFreeCamera\r\n */\r\n public getClassName(): string {\r\n return \"VRDeviceOrientationFreeCamera\";\r\n }\r\n\r\n protected _setRigMode = setVRRigMode.bind(null, this);\r\n}\r\n","import { Observable } from \"../Misc/observable\";\r\n\r\n/**\r\n * Represents a gamepad control stick position\r\n */\r\nexport class StickValues {\r\n /**\r\n * Initializes the gamepad x and y control stick values\r\n * @param x The x component of the gamepad control stick value\r\n * @param y The y component of the gamepad control stick value\r\n */\r\n constructor(\r\n /**\r\n * The x component of the control stick\r\n */\r\n public x: number,\r\n /**\r\n * The y component of the control stick\r\n */\r\n public y: number\r\n ) {}\r\n}\r\n\r\n/**\r\n * An interface which manages callbacks for gamepad button changes\r\n */\r\nexport interface GamepadButtonChanges {\r\n /**\r\n * Called when a gamepad has been changed\r\n */\r\n changed: boolean;\r\n /**\r\n * Called when a gamepad press event has been triggered\r\n */\r\n pressChanged: boolean;\r\n /**\r\n * Called when a touch event has been triggered\r\n */\r\n touchChanged: boolean;\r\n /**\r\n * Called when a value has changed\r\n */\r\n valueChanged: boolean;\r\n}\r\n\r\n/**\r\n * Represents a gamepad\r\n */\r\nexport class Gamepad {\r\n /**\r\n * Specifies what type of gamepad this represents\r\n */\r\n public type: number;\r\n\r\n private _leftStick: StickValues = { x: 0, y: 0 };\r\n private _rightStick: StickValues = { x: 0, y: 0 };\r\n\r\n /** @internal */\r\n public _isConnected = true;\r\n\r\n private _leftStickAxisX: number;\r\n private _leftStickAxisY: number;\r\n private _rightStickAxisX: number;\r\n private _rightStickAxisY: number;\r\n\r\n /**\r\n * Triggered when the left control stick has been changed\r\n */\r\n private _onleftstickchanged: (values: StickValues) => void;\r\n\r\n /**\r\n * Triggered when the right control stick has been changed\r\n */\r\n private _onrightstickchanged: (values: StickValues) => void;\r\n\r\n /**\r\n * Represents a gamepad controller\r\n */\r\n public static GAMEPAD = 0;\r\n /**\r\n * Represents a generic controller\r\n */\r\n public static GENERIC = 1;\r\n /**\r\n * Represents an XBox controller\r\n */\r\n public static XBOX = 2;\r\n /**\r\n * Represents a pose-enabled controller\r\n */\r\n public static POSE_ENABLED = 3;\r\n /**\r\n * Represents an Dual Shock controller\r\n */\r\n public static DUALSHOCK = 4;\r\n\r\n /**\r\n * Specifies whether the left control stick should be Y-inverted\r\n */\r\n protected _invertLeftStickY: boolean = false;\r\n\r\n /**\r\n * Specifies if the gamepad has been connected\r\n */\r\n public get isConnected(): boolean {\r\n return this._isConnected;\r\n }\r\n\r\n /**\r\n * Initializes the gamepad\r\n * @param id The id of the gamepad\r\n * @param index The index of the gamepad\r\n * @param browserGamepad The browser gamepad\r\n * @param leftStickX The x component of the left joystick\r\n * @param leftStickY The y component of the left joystick\r\n * @param rightStickX The x component of the right joystick\r\n * @param rightStickY The y component of the right joystick\r\n */\r\n constructor(\r\n /**\r\n * The id of the gamepad\r\n */\r\n public id: string,\r\n /**\r\n * The index of the gamepad\r\n */\r\n public index: number,\r\n /**\r\n * The browser gamepad\r\n */\r\n public browserGamepad: any,\r\n leftStickX: number = 0,\r\n leftStickY: number = 1,\r\n rightStickX: number = 2,\r\n rightStickY: number = 3\r\n ) {\r\n this.type = Gamepad.GAMEPAD;\r\n this._leftStickAxisX = leftStickX;\r\n this._leftStickAxisY = leftStickY;\r\n this._rightStickAxisX = rightStickX;\r\n this._rightStickAxisY = rightStickY;\r\n if (this.browserGamepad.axes.length >= 2) {\r\n this._leftStick = { x: this.browserGamepad.axes[this._leftStickAxisX], y: this.browserGamepad.axes[this._leftStickAxisY] };\r\n }\r\n if (this.browserGamepad.axes.length >= 4) {\r\n this._rightStick = { x: this.browserGamepad.axes[this._rightStickAxisX], y: this.browserGamepad.axes[this._rightStickAxisY] };\r\n }\r\n }\r\n\r\n /**\r\n * Callback triggered when the left joystick has changed\r\n * @param callback\r\n */\r\n public onleftstickchanged(callback: (values: StickValues) => void) {\r\n this._onleftstickchanged = callback;\r\n }\r\n\r\n /**\r\n * Callback triggered when the right joystick has changed\r\n * @param callback\r\n */\r\n public onrightstickchanged(callback: (values: StickValues) => void) {\r\n this._onrightstickchanged = callback;\r\n }\r\n\r\n /**\r\n * Gets the left joystick\r\n */\r\n public get leftStick(): StickValues {\r\n return this._leftStick;\r\n }\r\n /**\r\n * Sets the left joystick values\r\n */\r\n public set leftStick(newValues: StickValues) {\r\n if (this._onleftstickchanged && (this._leftStick.x !== newValues.x || this._leftStick.y !== newValues.y)) {\r\n this._onleftstickchanged(newValues);\r\n }\r\n this._leftStick = newValues;\r\n }\r\n /**\r\n * Gets the right joystick\r\n */\r\n public get rightStick(): StickValues {\r\n return this._rightStick;\r\n }\r\n /**\r\n * Sets the right joystick value\r\n */\r\n public set rightStick(newValues: StickValues) {\r\n if (this._onrightstickchanged && (this._rightStick.x !== newValues.x || this._rightStick.y !== newValues.y)) {\r\n this._onrightstickchanged(newValues);\r\n }\r\n this._rightStick = newValues;\r\n }\r\n\r\n /**\r\n * Updates the gamepad joystick positions\r\n */\r\n\r\n public update() {\r\n if (this._leftStick) {\r\n this.leftStick = { x: this.browserGamepad.axes[this._leftStickAxisX], y: this.browserGamepad.axes[this._leftStickAxisY] };\r\n if (this._invertLeftStickY) {\r\n this.leftStick.y *= -1;\r\n }\r\n }\r\n if (this._rightStick) {\r\n this.rightStick = { x: this.browserGamepad.axes[this._rightStickAxisX], y: this.browserGamepad.axes[this._rightStickAxisY] };\r\n }\r\n }\r\n\r\n /**\r\n * Disposes the gamepad\r\n */\r\n public dispose() {}\r\n}\r\n\r\n/**\r\n * Represents a generic gamepad\r\n */\r\nexport class GenericPad extends Gamepad {\r\n private _buttons: Array;\r\n private _onbuttondown: (buttonPressed: number) => void;\r\n private _onbuttonup: (buttonReleased: number) => void;\r\n\r\n /**\r\n * Observable triggered when a button has been pressed\r\n */\r\n public onButtonDownObservable = new Observable();\r\n /**\r\n * Observable triggered when a button has been released\r\n */\r\n public onButtonUpObservable = new Observable();\r\n\r\n /**\r\n * Callback triggered when a button has been pressed\r\n * @param callback Called when a button has been pressed\r\n */\r\n public onbuttondown(callback: (buttonPressed: number) => void) {\r\n this._onbuttondown = callback;\r\n }\r\n /**\r\n * Callback triggered when a button has been released\r\n * @param callback Called when a button has been released\r\n */\r\n public onbuttonup(callback: (buttonReleased: number) => void) {\r\n this._onbuttonup = callback;\r\n }\r\n\r\n /**\r\n * Initializes the generic gamepad\r\n * @param id The id of the generic gamepad\r\n * @param index The index of the generic gamepad\r\n * @param browserGamepad The browser gamepad\r\n */\r\n constructor(id: string, index: number, browserGamepad: any) {\r\n super(id, index, browserGamepad);\r\n this.type = Gamepad.GENERIC;\r\n this._buttons = new Array(browserGamepad.buttons.length);\r\n }\r\n\r\n private _setButtonValue(newValue: number, currentValue: number, buttonIndex: number): number {\r\n if (newValue !== currentValue) {\r\n if (newValue === 1) {\r\n if (this._onbuttondown) {\r\n this._onbuttondown(buttonIndex);\r\n }\r\n\r\n this.onButtonDownObservable.notifyObservers(buttonIndex);\r\n }\r\n if (newValue === 0) {\r\n if (this._onbuttonup) {\r\n this._onbuttonup(buttonIndex);\r\n }\r\n\r\n this.onButtonUpObservable.notifyObservers(buttonIndex);\r\n }\r\n }\r\n return newValue;\r\n }\r\n\r\n /**\r\n * Updates the generic gamepad\r\n */\r\n public update() {\r\n super.update();\r\n for (let index = 0; index < this._buttons.length; index++) {\r\n this._buttons[index] = this._setButtonValue(this.browserGamepad.buttons[index].value, this._buttons[index], index);\r\n }\r\n }\r\n\r\n /**\r\n * Disposes the generic gamepad\r\n */\r\n public dispose() {\r\n super.dispose();\r\n this.onButtonDownObservable.clear();\r\n this.onButtonUpObservable.clear();\r\n }\r\n}\r\n","import { Observable } from \"../../Misc/observable\";\r\nimport type { Nullable } from \"../../types\";\r\nimport { Quaternion, Matrix, Vector3, TmpVectors } from \"../../Maths/math.vector\";\r\nimport type { Node } from \"../../node\";\r\nimport type { TransformNode } from \"../../Meshes/transformNode\";\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport { Ray } from \"../../Culling/ray\";\r\nimport { EngineStore } from \"../../Engines/engineStore\";\r\n\r\nimport { Gamepad } from \"../../Gamepads/gamepad\";\r\nimport type { WebVRFreeCamera, PoseControlled, DevicePose } from \"../../Cameras/VR/webVRCamera\";\r\nimport type { TargetCamera } from \"../../Cameras/targetCamera\";\r\n\r\n/**\r\n * Defines the types of pose enabled controllers that are supported\r\n */\r\nexport enum PoseEnabledControllerType {\r\n /**\r\n * HTC Vive\r\n */\r\n VIVE,\r\n /**\r\n * Oculus Rift\r\n */\r\n OCULUS,\r\n /**\r\n * Windows mixed reality\r\n */\r\n WINDOWS,\r\n /**\r\n * Samsung gear VR\r\n */\r\n GEAR_VR,\r\n /**\r\n * Google Daydream\r\n */\r\n DAYDREAM,\r\n /**\r\n * Generic\r\n */\r\n GENERIC,\r\n}\r\n\r\n/**\r\n * Defines the MutableGamepadButton interface for the state of a gamepad button\r\n */\r\nexport interface MutableGamepadButton {\r\n /**\r\n * Value of the button/trigger\r\n */\r\n value: number;\r\n /**\r\n * If the button/trigger is currently touched\r\n */\r\n touched: boolean;\r\n /**\r\n * If the button/trigger is currently pressed\r\n */\r\n pressed: boolean;\r\n}\r\n\r\n/**\r\n * Defines the ExtendedGamepadButton interface for a gamepad button which includes state provided by a pose controller\r\n * @internal\r\n */\r\nexport interface ExtendedGamepadButton extends GamepadButton {\r\n /**\r\n * If the button/trigger is currently pressed\r\n */\r\n readonly pressed: boolean;\r\n /**\r\n * If the button/trigger is currently touched\r\n */\r\n readonly touched: boolean;\r\n /**\r\n * Value of the button/trigger\r\n */\r\n readonly value: number;\r\n}\r\n\r\n/** @internal */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport interface _GamePadFactory {\r\n /**\r\n * Returns whether or not the current gamepad can be created for this type of controller.\r\n * @param gamepadInfo Defines the gamepad info as received from the controller APIs.\r\n * @returns true if it can be created, otherwise false\r\n */\r\n canCreate(gamepadInfo: any): boolean;\r\n\r\n /**\r\n * Creates a new instance of the Gamepad.\r\n * @param gamepadInfo Defines the gamepad info as received from the controller APIs.\r\n * @returns the new gamepad instance\r\n */\r\n create(gamepadInfo: any): Gamepad;\r\n}\r\n\r\n/**\r\n * Defines the PoseEnabledControllerHelper object that is used initialize a gamepad as the controller type it is specified as (eg. windows mixed reality controller)\r\n */\r\nexport class PoseEnabledControllerHelper {\r\n /** @internal */\r\n public static _ControllerFactories: _GamePadFactory[] = [];\r\n\r\n /** @internal */\r\n public static _DefaultControllerFactory: Nullable<(gamepadInfo: any) => Gamepad> = null;\r\n\r\n /**\r\n * Initializes a gamepad as the controller type it is specified as (eg. windows mixed reality controller)\r\n * @param vrGamepad the gamepad to initialized\r\n * @returns a vr controller of the type the gamepad identified as\r\n */\r\n public static InitiateController(vrGamepad: any) {\r\n for (const factory of this._ControllerFactories) {\r\n if (factory.canCreate(vrGamepad)) {\r\n return factory.create(vrGamepad);\r\n }\r\n }\r\n\r\n if (this._DefaultControllerFactory) {\r\n return this._DefaultControllerFactory(vrGamepad);\r\n }\r\n\r\n throw \"The type of gamepad you are trying to load needs to be imported first or is not supported.\";\r\n }\r\n}\r\n\r\n/**\r\n * Defines the PoseEnabledController object that contains state of a vr capable controller\r\n */\r\nexport class PoseEnabledController extends Gamepad implements PoseControlled {\r\n /**\r\n * If the controller is used in a webXR session\r\n */\r\n public isXR = false;\r\n // Represents device position and rotation in room space. Should only be used to help calculate babylon space values\r\n private _deviceRoomPosition = Vector3.Zero();\r\n private _deviceRoomRotationQuaternion = new Quaternion();\r\n\r\n /**\r\n * The device position in babylon space\r\n */\r\n public devicePosition = Vector3.Zero();\r\n /**\r\n * The device rotation in babylon space\r\n */\r\n public deviceRotationQuaternion = new Quaternion();\r\n /**\r\n * The scale factor of the device in babylon space\r\n */\r\n public deviceScaleFactor: number = 1;\r\n\r\n /**\r\n * (Likely devicePosition should be used instead) The device position in its room space\r\n */\r\n public position: Vector3;\r\n /**\r\n * (Likely deviceRotationQuaternion should be used instead) The device rotation in its room space\r\n */\r\n public rotationQuaternion: Quaternion;\r\n /**\r\n * The type of controller (Eg. Windows mixed reality)\r\n */\r\n public controllerType: PoseEnabledControllerType;\r\n\r\n protected _calculatedPosition: Vector3;\r\n private _calculatedRotation: Quaternion;\r\n\r\n /**\r\n * The raw pose from the device\r\n */\r\n public rawPose: DevicePose; //GamepadPose;\r\n\r\n // Used to convert 6dof controllers to 3dof\r\n private _trackPosition = true;\r\n private _maxRotationDistFromHeadset = Math.PI / 5;\r\n private _draggedRoomRotation = 0;\r\n /**\r\n * @internal\r\n */\r\n public _disableTrackPosition(fixedPosition: Vector3) {\r\n if (this._trackPosition) {\r\n this._calculatedPosition.copyFrom(fixedPosition);\r\n this._trackPosition = false;\r\n }\r\n }\r\n\r\n /**\r\n * Internal, the mesh attached to the controller\r\n * @internal\r\n */\r\n public _mesh: Nullable; // a node that will be attached to this Gamepad\r\n private _poseControlledCamera: TargetCamera;\r\n\r\n private _leftHandSystemQuaternion: Quaternion = new Quaternion();\r\n\r\n /**\r\n * Internal, matrix used to convert room space to babylon space\r\n * @internal\r\n */\r\n public _deviceToWorld = Matrix.Identity();\r\n\r\n /**\r\n * Node to be used when casting a ray from the controller\r\n * @internal\r\n */\r\n public _pointingPoseNode: Nullable = null;\r\n /**\r\n * Name of the child mesh that can be used to cast a ray from the controller\r\n */\r\n public static readonly POINTING_POSE = \"POINTING_POSE\";\r\n /**\r\n * Creates a new PoseEnabledController from a gamepad\r\n * @param browserGamepad the gamepad that the PoseEnabledController should be created from\r\n */\r\n constructor(browserGamepad: any) {\r\n super(browserGamepad.id, browserGamepad.index, browserGamepad);\r\n this.type = Gamepad.POSE_ENABLED;\r\n this.controllerType = PoseEnabledControllerType.GENERIC;\r\n this.position = Vector3.Zero();\r\n this.rotationQuaternion = new Quaternion();\r\n\r\n this._calculatedPosition = Vector3.Zero();\r\n this._calculatedRotation = new Quaternion();\r\n Quaternion.RotationYawPitchRollToRef(Math.PI, 0, 0, this._leftHandSystemQuaternion);\r\n }\r\n\r\n private _workingMatrix = Matrix.Identity();\r\n /**\r\n * Updates the state of the pose enabled controller and mesh based on the current position and rotation of the controller\r\n */\r\n public update() {\r\n super.update();\r\n this._updatePoseAndMesh();\r\n }\r\n\r\n /**\r\n * Updates only the pose device and mesh without doing any button event checking\r\n */\r\n protected _updatePoseAndMesh() {\r\n if (this.isXR) {\r\n return;\r\n }\r\n const pose: DevicePose = this.browserGamepad.pose;\r\n this.updateFromDevice(pose);\r\n\r\n if (\r\n !this._trackPosition &&\r\n EngineStore.LastCreatedScene &&\r\n EngineStore.LastCreatedScene.activeCamera &&\r\n (EngineStore.LastCreatedScene.activeCamera).devicePosition\r\n ) {\r\n const camera = EngineStore.LastCreatedScene.activeCamera;\r\n camera._computeDevicePosition();\r\n\r\n this._deviceToWorld.setTranslation(camera.devicePosition);\r\n if (camera.deviceRotationQuaternion) {\r\n camera._deviceRoomRotationQuaternion.toEulerAnglesToRef(TmpVectors.Vector3[0]);\r\n\r\n // Find the radian distance away that the headset is from the controllers rotation\r\n const distanceAway = Math.atan2(Math.sin(TmpVectors.Vector3[0].y - this._draggedRoomRotation), Math.cos(TmpVectors.Vector3[0].y - this._draggedRoomRotation));\r\n if (Math.abs(distanceAway) > this._maxRotationDistFromHeadset) {\r\n // Only rotate enouph to be within the _maxRotationDistFromHeadset\r\n const rotationAmount = distanceAway - (distanceAway < 0 ? -this._maxRotationDistFromHeadset : this._maxRotationDistFromHeadset);\r\n this._draggedRoomRotation += rotationAmount;\r\n\r\n // Rotate controller around headset\r\n const sin = Math.sin(-rotationAmount);\r\n const cos = Math.cos(-rotationAmount);\r\n this._calculatedPosition.x = this._calculatedPosition.x * cos - this._calculatedPosition.z * sin;\r\n this._calculatedPosition.z = this._calculatedPosition.x * sin + this._calculatedPosition.z * cos;\r\n }\r\n }\r\n }\r\n\r\n Vector3.TransformCoordinatesToRef(this._calculatedPosition, this._deviceToWorld, this.devicePosition);\r\n this._deviceToWorld.getRotationMatrixToRef(this._workingMatrix);\r\n Quaternion.FromRotationMatrixToRef(this._workingMatrix, this.deviceRotationQuaternion);\r\n this.deviceRotationQuaternion.multiplyInPlace(this._calculatedRotation);\r\n\r\n if (this._mesh) {\r\n this._mesh.position.copyFrom(this.devicePosition);\r\n\r\n if (this._mesh.rotationQuaternion) {\r\n this._mesh.rotationQuaternion.copyFrom(this.deviceRotationQuaternion);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Updates the state of the pose enbaled controller based on the raw pose data from the device\r\n * @param poseData raw pose fromthe device\r\n */\r\n updateFromDevice(poseData: DevicePose) {\r\n if (this.isXR) {\r\n return;\r\n }\r\n if (poseData) {\r\n this.rawPose = poseData;\r\n if (poseData.position) {\r\n this._deviceRoomPosition.copyFromFloats(poseData.position[0], poseData.position[1], -poseData.position[2]);\r\n if (this._mesh && this._mesh.getScene().useRightHandedSystem) {\r\n this._deviceRoomPosition.z *= -1;\r\n }\r\n if (this._trackPosition) {\r\n this._deviceRoomPosition.scaleToRef(this.deviceScaleFactor, this._calculatedPosition);\r\n }\r\n this._calculatedPosition.addInPlace(this.position);\r\n }\r\n const pose = this.rawPose;\r\n if (poseData.orientation && pose.orientation && pose.orientation.length === 4) {\r\n this._deviceRoomRotationQuaternion.copyFromFloats(pose.orientation[0], pose.orientation[1], -pose.orientation[2], -pose.orientation[3]);\r\n if (this._mesh) {\r\n if (this._mesh.getScene().useRightHandedSystem) {\r\n this._deviceRoomRotationQuaternion.z *= -1;\r\n this._deviceRoomRotationQuaternion.w *= -1;\r\n } else {\r\n this._deviceRoomRotationQuaternion.multiplyToRef(this._leftHandSystemQuaternion, this._deviceRoomRotationQuaternion);\r\n }\r\n }\r\n\r\n // if the camera is set, rotate to the camera's rotation\r\n this._deviceRoomRotationQuaternion.multiplyToRef(this.rotationQuaternion, this._calculatedRotation);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _meshAttachedObservable = new Observable();\r\n\r\n /**\r\n * Attaches a mesh to the controller\r\n * @param mesh the mesh to be attached\r\n */\r\n public attachToMesh(mesh: AbstractMesh) {\r\n if (this._mesh) {\r\n this._mesh.parent = null;\r\n }\r\n this._mesh = mesh;\r\n if (this._poseControlledCamera) {\r\n this._mesh.parent = this._poseControlledCamera;\r\n }\r\n if (!this._mesh.rotationQuaternion) {\r\n this._mesh.rotationQuaternion = new Quaternion();\r\n }\r\n\r\n // Sync controller mesh and pointing pose node's state with controller, this is done to avoid a frame where position is 0,0,0 when attaching mesh\r\n if (!this.isXR) {\r\n this._updatePoseAndMesh();\r\n if (this._pointingPoseNode) {\r\n const parents = [];\r\n let obj: Node = this._pointingPoseNode;\r\n while (obj.parent) {\r\n parents.push(obj.parent);\r\n obj = obj.parent;\r\n }\r\n parents.reverse().forEach((p) => {\r\n p.computeWorldMatrix(true);\r\n });\r\n }\r\n }\r\n\r\n this._meshAttachedObservable.notifyObservers(mesh);\r\n }\r\n\r\n /**\r\n * Attaches the controllers mesh to a camera\r\n * @param camera the camera the mesh should be attached to\r\n */\r\n public attachToPoseControlledCamera(camera: TargetCamera) {\r\n this._poseControlledCamera = camera;\r\n if (this._mesh) {\r\n this._mesh.parent = this._poseControlledCamera;\r\n }\r\n }\r\n\r\n /**\r\n * Disposes of the controller\r\n */\r\n public dispose() {\r\n if (this._mesh) {\r\n this._mesh.dispose();\r\n }\r\n this._mesh = null;\r\n\r\n super.dispose();\r\n }\r\n\r\n /**\r\n * The mesh that is attached to the controller\r\n */\r\n public get mesh(): Nullable {\r\n return this._mesh;\r\n }\r\n\r\n /**\r\n * Gets the ray of the controller in the direction the controller is pointing\r\n * @param length the length the resulting ray should be\r\n * @returns a ray in the direction the controller is pointing\r\n */\r\n public getForwardRay(length = 100): Ray {\r\n if (!this.mesh) {\r\n return new Ray(Vector3.Zero(), new Vector3(0, 0, 1), length);\r\n }\r\n\r\n const m = this._pointingPoseNode ? this._pointingPoseNode.getWorldMatrix() : this.mesh.getWorldMatrix();\r\n const origin = m.getTranslation();\r\n\r\n const forward = new Vector3(0, 0, -1);\r\n const forwardWorld = Vector3.TransformNormal(forward, m);\r\n\r\n const direction = Vector3.Normalize(forwardWorld);\r\n\r\n return new Ray(origin, direction, length);\r\n }\r\n}\r\n","import { Observable } from \"../Misc/observable\";\r\nimport { Gamepad } from \"../Gamepads/gamepad\";\r\n/**\r\n * Defines supported buttons for XBox360 compatible gamepads\r\n */\r\nexport enum Xbox360Button {\r\n /** A */\r\n A = 0,\r\n /** B */\r\n B = 1,\r\n /** X */\r\n X = 2,\r\n /** Y */\r\n Y = 3,\r\n /** Left button */\r\n LB = 4,\r\n /** Right button */\r\n RB = 5,\r\n /** Back */\r\n Back = 8,\r\n /** Start */\r\n Start = 9,\r\n /** Left stick */\r\n LeftStick = 10,\r\n /** Right stick */\r\n RightStick = 11,\r\n}\r\n\r\n/** Defines values for XBox360 DPad */\r\nexport enum Xbox360Dpad {\r\n /** Up */\r\n Up = 12,\r\n /** Down */\r\n Down = 13,\r\n /** Left */\r\n Left = 14,\r\n /** Right */\r\n Right = 15,\r\n}\r\n\r\n/**\r\n * Defines a XBox360 gamepad\r\n */\r\nexport class Xbox360Pad extends Gamepad {\r\n private _leftTrigger: number = 0;\r\n private _rightTrigger: number = 0;\r\n\r\n private _onlefttriggerchanged: (value: number) => void;\r\n private _onrighttriggerchanged: (value: number) => void;\r\n\r\n private _onbuttondown: (buttonPressed: Xbox360Button) => void;\r\n private _onbuttonup: (buttonReleased: Xbox360Button) => void;\r\n private _ondpaddown: (dPadPressed: Xbox360Dpad) => void;\r\n private _ondpadup: (dPadReleased: Xbox360Dpad) => void;\r\n\r\n /** Observable raised when a button is pressed */\r\n public onButtonDownObservable = new Observable();\r\n /** Observable raised when a button is released */\r\n public onButtonUpObservable = new Observable();\r\n /** Observable raised when a pad is pressed */\r\n public onPadDownObservable = new Observable();\r\n /** Observable raised when a pad is released */\r\n public onPadUpObservable = new Observable();\r\n\r\n private _buttonA: number = 0;\r\n private _buttonB: number = 0;\r\n private _buttonX: number = 0;\r\n private _buttonY: number = 0;\r\n private _buttonBack: number = 0;\r\n private _buttonStart: number = 0;\r\n private _buttonLB: number = 0;\r\n private _buttonRB: number = 0;\r\n\r\n private _buttonLeftStick: number = 0;\r\n private _buttonRightStick: number = 0;\r\n private _dPadUp: number = 0;\r\n private _dPadDown: number = 0;\r\n private _dPadLeft: number = 0;\r\n private _dPadRight: number = 0;\r\n\r\n private _isXboxOnePad: boolean = false;\r\n\r\n /**\r\n * Creates a new XBox360 gamepad object\r\n * @param id defines the id of this gamepad\r\n * @param index defines its index\r\n * @param gamepad defines the internal HTML gamepad object\r\n * @param xboxOne defines if it is a XBox One gamepad\r\n */\r\n constructor(id: string, index: number, gamepad: any, xboxOne: boolean = false) {\r\n super(id, index, gamepad, 0, 1, 2, 3);\r\n this.type = Gamepad.XBOX;\r\n this._isXboxOnePad = xboxOne;\r\n }\r\n\r\n /**\r\n * Defines the callback to call when left trigger is pressed\r\n * @param callback defines the callback to use\r\n */\r\n public onlefttriggerchanged(callback: (value: number) => void) {\r\n this._onlefttriggerchanged = callback;\r\n }\r\n\r\n /**\r\n * Defines the callback to call when right trigger is pressed\r\n * @param callback defines the callback to use\r\n */\r\n public onrighttriggerchanged(callback: (value: number) => void) {\r\n this._onrighttriggerchanged = callback;\r\n }\r\n\r\n /**\r\n * Gets the left trigger value\r\n */\r\n public get leftTrigger(): number {\r\n return this._leftTrigger;\r\n }\r\n /**\r\n * Sets the left trigger value\r\n */\r\n public set leftTrigger(newValue: number) {\r\n if (this._onlefttriggerchanged && this._leftTrigger !== newValue) {\r\n this._onlefttriggerchanged(newValue);\r\n }\r\n this._leftTrigger = newValue;\r\n }\r\n\r\n /**\r\n * Gets the right trigger value\r\n */\r\n public get rightTrigger(): number {\r\n return this._rightTrigger;\r\n }\r\n /**\r\n * Sets the right trigger value\r\n */\r\n public set rightTrigger(newValue: number) {\r\n if (this._onrighttriggerchanged && this._rightTrigger !== newValue) {\r\n this._onrighttriggerchanged(newValue);\r\n }\r\n this._rightTrigger = newValue;\r\n }\r\n\r\n /**\r\n * Defines the callback to call when a button is pressed\r\n * @param callback defines the callback to use\r\n */\r\n public onbuttondown(callback: (buttonPressed: Xbox360Button) => void) {\r\n this._onbuttondown = callback;\r\n }\r\n\r\n /**\r\n * Defines the callback to call when a button is released\r\n * @param callback defines the callback to use\r\n */\r\n public onbuttonup(callback: (buttonReleased: Xbox360Button) => void) {\r\n this._onbuttonup = callback;\r\n }\r\n\r\n /**\r\n * Defines the callback to call when a pad is pressed\r\n * @param callback defines the callback to use\r\n */\r\n public ondpaddown(callback: (dPadPressed: Xbox360Dpad) => void) {\r\n this._ondpaddown = callback;\r\n }\r\n\r\n /**\r\n * Defines the callback to call when a pad is released\r\n * @param callback defines the callback to use\r\n */\r\n public ondpadup(callback: (dPadReleased: Xbox360Dpad) => void) {\r\n this._ondpadup = callback;\r\n }\r\n\r\n private _setButtonValue(newValue: number, currentValue: number, buttonType: Xbox360Button): number {\r\n if (newValue !== currentValue) {\r\n if (newValue === 1) {\r\n if (this._onbuttondown) {\r\n this._onbuttondown(buttonType);\r\n }\r\n\r\n this.onButtonDownObservable.notifyObservers(buttonType);\r\n }\r\n if (newValue === 0) {\r\n if (this._onbuttonup) {\r\n this._onbuttonup(buttonType);\r\n }\r\n\r\n this.onButtonUpObservable.notifyObservers(buttonType);\r\n }\r\n }\r\n return newValue;\r\n }\r\n\r\n private _setDPadValue(newValue: number, currentValue: number, buttonType: Xbox360Dpad): number {\r\n if (newValue !== currentValue) {\r\n if (newValue === 1) {\r\n if (this._ondpaddown) {\r\n this._ondpaddown(buttonType);\r\n }\r\n\r\n this.onPadDownObservable.notifyObservers(buttonType);\r\n }\r\n if (newValue === 0) {\r\n if (this._ondpadup) {\r\n this._ondpadup(buttonType);\r\n }\r\n\r\n this.onPadUpObservable.notifyObservers(buttonType);\r\n }\r\n }\r\n return newValue;\r\n }\r\n\r\n /**\r\n * Gets the value of the `A` button\r\n */\r\n public get buttonA(): number {\r\n return this._buttonA;\r\n }\r\n /**\r\n * Sets the value of the `A` button\r\n */\r\n public set buttonA(value) {\r\n this._buttonA = this._setButtonValue(value, this._buttonA, Xbox360Button.A);\r\n }\r\n\r\n /**\r\n * Gets the value of the `B` button\r\n */\r\n public get buttonB(): number {\r\n return this._buttonB;\r\n }\r\n /**\r\n * Sets the value of the `B` button\r\n */\r\n public set buttonB(value) {\r\n this._buttonB = this._setButtonValue(value, this._buttonB, Xbox360Button.B);\r\n }\r\n\r\n /**\r\n * Gets the value of the `X` button\r\n */\r\n public get buttonX(): number {\r\n return this._buttonX;\r\n }\r\n /**\r\n * Sets the value of the `X` button\r\n */\r\n public set buttonX(value) {\r\n this._buttonX = this._setButtonValue(value, this._buttonX, Xbox360Button.X);\r\n }\r\n\r\n /**\r\n * Gets the value of the `Y` button\r\n */\r\n public get buttonY(): number {\r\n return this._buttonY;\r\n }\r\n /**\r\n * Sets the value of the `Y` button\r\n */\r\n public set buttonY(value) {\r\n this._buttonY = this._setButtonValue(value, this._buttonY, Xbox360Button.Y);\r\n }\r\n\r\n /**\r\n * Gets the value of the `Start` button\r\n */\r\n public get buttonStart(): number {\r\n return this._buttonStart;\r\n }\r\n /**\r\n * Sets the value of the `Start` button\r\n */\r\n public set buttonStart(value) {\r\n this._buttonStart = this._setButtonValue(value, this._buttonStart, Xbox360Button.Start);\r\n }\r\n\r\n /**\r\n * Gets the value of the `Back` button\r\n */\r\n public get buttonBack(): number {\r\n return this._buttonBack;\r\n }\r\n /**\r\n * Sets the value of the `Back` button\r\n */\r\n public set buttonBack(value) {\r\n this._buttonBack = this._setButtonValue(value, this._buttonBack, Xbox360Button.Back);\r\n }\r\n\r\n /**\r\n * Gets the value of the `Left` button\r\n */\r\n public get buttonLB(): number {\r\n return this._buttonLB;\r\n }\r\n /**\r\n * Sets the value of the `Left` button\r\n */\r\n public set buttonLB(value) {\r\n this._buttonLB = this._setButtonValue(value, this._buttonLB, Xbox360Button.LB);\r\n }\r\n\r\n /**\r\n * Gets the value of the `Right` button\r\n */\r\n public get buttonRB(): number {\r\n return this._buttonRB;\r\n }\r\n /**\r\n * Sets the value of the `Right` button\r\n */\r\n public set buttonRB(value) {\r\n this._buttonRB = this._setButtonValue(value, this._buttonRB, Xbox360Button.RB);\r\n }\r\n\r\n /**\r\n * Gets the value of the Left joystick\r\n */\r\n public get buttonLeftStick(): number {\r\n return this._buttonLeftStick;\r\n }\r\n /**\r\n * Sets the value of the Left joystick\r\n */\r\n public set buttonLeftStick(value) {\r\n this._buttonLeftStick = this._setButtonValue(value, this._buttonLeftStick, Xbox360Button.LeftStick);\r\n }\r\n\r\n /**\r\n * Gets the value of the Right joystick\r\n */\r\n public get buttonRightStick(): number {\r\n return this._buttonRightStick;\r\n }\r\n /**\r\n * Sets the value of the Right joystick\r\n */\r\n public set buttonRightStick(value) {\r\n this._buttonRightStick = this._setButtonValue(value, this._buttonRightStick, Xbox360Button.RightStick);\r\n }\r\n\r\n /**\r\n * Gets the value of D-pad up\r\n */\r\n public get dPadUp(): number {\r\n return this._dPadUp;\r\n }\r\n /**\r\n * Sets the value of D-pad up\r\n */\r\n public set dPadUp(value) {\r\n this._dPadUp = this._setDPadValue(value, this._dPadUp, Xbox360Dpad.Up);\r\n }\r\n\r\n /**\r\n * Gets the value of D-pad down\r\n */\r\n public get dPadDown(): number {\r\n return this._dPadDown;\r\n }\r\n /**\r\n * Sets the value of D-pad down\r\n */\r\n public set dPadDown(value) {\r\n this._dPadDown = this._setDPadValue(value, this._dPadDown, Xbox360Dpad.Down);\r\n }\r\n\r\n /**\r\n * Gets the value of D-pad left\r\n */\r\n public get dPadLeft(): number {\r\n return this._dPadLeft;\r\n }\r\n /**\r\n * Sets the value of D-pad left\r\n */\r\n public set dPadLeft(value) {\r\n this._dPadLeft = this._setDPadValue(value, this._dPadLeft, Xbox360Dpad.Left);\r\n }\r\n\r\n /**\r\n * Gets the value of D-pad right\r\n */\r\n public get dPadRight(): number {\r\n return this._dPadRight;\r\n }\r\n /**\r\n * Sets the value of D-pad right\r\n */\r\n public set dPadRight(value) {\r\n this._dPadRight = this._setDPadValue(value, this._dPadRight, Xbox360Dpad.Right);\r\n }\r\n\r\n /**\r\n * Force the gamepad to synchronize with device values\r\n */\r\n public update() {\r\n super.update();\r\n if (this._isXboxOnePad) {\r\n this.buttonA = this.browserGamepad.buttons[0].value;\r\n this.buttonB = this.browserGamepad.buttons[1].value;\r\n this.buttonX = this.browserGamepad.buttons[2].value;\r\n this.buttonY = this.browserGamepad.buttons[3].value;\r\n this.buttonLB = this.browserGamepad.buttons[4].value;\r\n this.buttonRB = this.browserGamepad.buttons[5].value;\r\n this.leftTrigger = this.browserGamepad.buttons[6].value;\r\n this.rightTrigger = this.browserGamepad.buttons[7].value;\r\n this.buttonBack = this.browserGamepad.buttons[8].value;\r\n this.buttonStart = this.browserGamepad.buttons[9].value;\r\n this.buttonLeftStick = this.browserGamepad.buttons[10].value;\r\n this.buttonRightStick = this.browserGamepad.buttons[11].value;\r\n this.dPadUp = this.browserGamepad.buttons[12].value;\r\n this.dPadDown = this.browserGamepad.buttons[13].value;\r\n this.dPadLeft = this.browserGamepad.buttons[14].value;\r\n this.dPadRight = this.browserGamepad.buttons[15].value;\r\n } else {\r\n this.buttonA = this.browserGamepad.buttons[0].value;\r\n this.buttonB = this.browserGamepad.buttons[1].value;\r\n this.buttonX = this.browserGamepad.buttons[2].value;\r\n this.buttonY = this.browserGamepad.buttons[3].value;\r\n this.buttonLB = this.browserGamepad.buttons[4].value;\r\n this.buttonRB = this.browserGamepad.buttons[5].value;\r\n this.leftTrigger = this.browserGamepad.buttons[6].value;\r\n this.rightTrigger = this.browserGamepad.buttons[7].value;\r\n this.buttonBack = this.browserGamepad.buttons[8].value;\r\n this.buttonStart = this.browserGamepad.buttons[9].value;\r\n this.buttonLeftStick = this.browserGamepad.buttons[10].value;\r\n this.buttonRightStick = this.browserGamepad.buttons[11].value;\r\n this.dPadUp = this.browserGamepad.buttons[12].value;\r\n this.dPadDown = this.browserGamepad.buttons[13].value;\r\n this.dPadLeft = this.browserGamepad.buttons[14].value;\r\n this.dPadRight = this.browserGamepad.buttons[15].value;\r\n }\r\n }\r\n\r\n /**\r\n * Disposes the gamepad\r\n */\r\n public dispose() {\r\n super.dispose();\r\n this.onButtonDownObservable.clear();\r\n this.onButtonUpObservable.clear();\r\n this.onPadDownObservable.clear();\r\n this.onPadUpObservable.clear();\r\n }\r\n}\r\n","import type { Camera } from \"../camera\";\r\nimport { Matrix } from \"../../Maths/math.vector\";\r\nimport { Viewport } from \"../../Maths/math.viewport\";\r\n\r\n/**\r\n * @internal\r\n */\r\nexport function setWebVRRigMode(camera: Camera, rigParams: any) {\r\n if (rigParams.vrDisplay) {\r\n const leftEye = rigParams.vrDisplay.getEyeParameters(\"left\");\r\n const rightEye = rigParams.vrDisplay.getEyeParameters(\"right\");\r\n\r\n //Left eye\r\n camera._rigCameras[0].viewport = new Viewport(0, 0, 0.5, 1.0);\r\n camera._rigCameras[0].setCameraRigParameter(\"left\", true);\r\n //leaving this for future reference\r\n camera._rigCameras[0].setCameraRigParameter(\"specs\", rigParams.specs);\r\n camera._rigCameras[0].setCameraRigParameter(\"eyeParameters\", leftEye);\r\n camera._rigCameras[0].setCameraRigParameter(\"frameData\", rigParams.frameData);\r\n camera._rigCameras[0].setCameraRigParameter(\"parentCamera\", rigParams.parentCamera);\r\n camera._rigCameras[0]._cameraRigParams.vrWorkMatrix = new Matrix();\r\n camera._rigCameras[0].getProjectionMatrix = camera._getWebVRProjectionMatrix;\r\n camera._rigCameras[0].parent = camera;\r\n camera._rigCameras[0]._getViewMatrix = camera._getWebVRViewMatrix;\r\n\r\n //Right eye\r\n camera._rigCameras[1].viewport = new Viewport(0.5, 0, 0.5, 1.0);\r\n camera._rigCameras[1].setCameraRigParameter(\"eyeParameters\", rightEye);\r\n camera._rigCameras[1].setCameraRigParameter(\"specs\", rigParams.specs);\r\n camera._rigCameras[1].setCameraRigParameter(\"frameData\", rigParams.frameData);\r\n camera._rigCameras[1].setCameraRigParameter(\"parentCamera\", rigParams.parentCamera);\r\n camera._rigCameras[1]._cameraRigParams.vrWorkMatrix = new Matrix();\r\n camera._rigCameras[1].getProjectionMatrix = camera._getWebVRProjectionMatrix;\r\n camera._rigCameras[1].parent = camera;\r\n camera._rigCameras[1]._getViewMatrix = camera._getWebVRViewMatrix;\r\n }\r\n}\r\n","import type { Nullable } from \"../../types\";\r\nimport type { IDisplayChangedEventArgs } from \"../../Engines/engine\";\r\nimport { Engine } from \"../../Engines/engine\";\r\nimport { Size } from \"../../Maths/math.size\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport { Tools } from \"../../Misc/tools\";\r\nimport { IsWindowObjectExist } from \"../../Misc/domManagement\";\r\nimport type { WebVROptions } from \"../../Cameras/VR/webVRCamera\";\r\n\r\n/**\r\n * Interface used to define additional presentation attributes\r\n */\r\nexport interface IVRPresentationAttributes {\r\n /**\r\n * Defines a boolean indicating that we want to get 72hz mode on Oculus Browser (default is off eg. 60hz)\r\n */\r\n highRefreshRate: boolean;\r\n /**\r\n * Enables foveation in VR to improve perf. 0 none, 1 low, 2 medium, 3 high (Default is 1)\r\n */\r\n foveationLevel: number;\r\n}\r\n\r\ndeclare module \"../../Engines/engine\" {\r\n export interface Engine {\r\n /** @internal */\r\n _vrDisplay: any;\r\n /** @internal */\r\n _vrSupported: boolean;\r\n /** @internal */\r\n _oldSize: Size;\r\n /** @internal */\r\n _oldHardwareScaleFactor: number;\r\n /** @internal */\r\n _vrExclusivePointerMode: boolean;\r\n /** @internal */\r\n _webVRInitPromise: Promise;\r\n\r\n /** @internal */\r\n _onVRDisplayPointerRestricted: () => void;\r\n /** @internal */\r\n _onVRDisplayPointerUnrestricted: () => void;\r\n\r\n /** @internal */\r\n _onVrDisplayConnect: Nullable<(display: any) => void>;\r\n /** @internal */\r\n _onVrDisplayDisconnect: Nullable<() => void>;\r\n /** @internal */\r\n _onVrDisplayPresentChange: Nullable<() => void>;\r\n\r\n /**\r\n * Observable signaled when VR display mode changes\r\n */\r\n onVRDisplayChangedObservable: Observable;\r\n /**\r\n * Observable signaled when VR request present is complete\r\n */\r\n onVRRequestPresentComplete: Observable;\r\n /**\r\n * Observable signaled when VR request present starts\r\n */\r\n onVRRequestPresentStart: Observable;\r\n\r\n /**\r\n * Gets a boolean indicating that the engine is currently in VR exclusive mode for the pointers\r\n * @see https://docs.microsoft.com/en-us/microsoft-edge/webvr/essentials#mouse-input\r\n */\r\n isInVRExclusivePointerMode: boolean;\r\n\r\n /**\r\n * Gets a boolean indicating if a webVR device was detected\r\n * @returns true if a webVR device was detected\r\n */\r\n isVRDevicePresent(): boolean;\r\n\r\n /**\r\n * Gets the current webVR device\r\n * @returns the current webVR device (or null)\r\n */\r\n getVRDevice(): any;\r\n\r\n /**\r\n * Initializes a webVR display and starts listening to display change events\r\n * The onVRDisplayChangedObservable will be notified upon these changes\r\n * @returns A promise containing a VRDisplay and if vr is supported\r\n */\r\n initWebVRAsync(): Promise;\r\n\r\n /** @internal */\r\n _getVRDisplaysAsync(): Promise;\r\n\r\n /**\r\n * Gets or sets the presentation attributes used to configure VR rendering\r\n */\r\n vrPresentationAttributes?: IVRPresentationAttributes;\r\n\r\n /**\r\n * Call this function to switch to webVR mode\r\n * Will do nothing if webVR is not supported or if there is no webVR device\r\n * @param options the webvr options provided to the camera. mainly used for multiview\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/webVRCamera\r\n */\r\n enableVR(options: WebVROptions): void;\r\n\r\n /** @internal */\r\n _onVRFullScreenTriggered(): void;\r\n }\r\n}\r\n\r\nObject.defineProperty(Engine.prototype, \"isInVRExclusivePointerMode\", {\r\n get: function (this: Engine) {\r\n return this._vrExclusivePointerMode;\r\n },\r\n enumerable: true,\r\n configurable: true,\r\n});\r\n\r\nEngine.prototype._prepareVRComponent = function () {\r\n this._vrSupported = false;\r\n this._vrExclusivePointerMode = false;\r\n this.onVRDisplayChangedObservable = new Observable();\r\n this.onVRRequestPresentComplete = new Observable();\r\n this.onVRRequestPresentStart = new Observable();\r\n};\r\n\r\nEngine.prototype.isVRDevicePresent = function () {\r\n return !!this._vrDisplay;\r\n};\r\n\r\nEngine.prototype.getVRDevice = function (): any {\r\n return this._vrDisplay;\r\n};\r\n\r\nEngine.prototype.initWebVR = function (): Observable {\r\n this.initWebVRAsync();\r\n return this.onVRDisplayChangedObservable;\r\n};\r\n\r\nEngine.prototype.initWebVRAsync = function (): Promise {\r\n const notifyObservers = () => {\r\n const eventArgs = {\r\n vrDisplay: this._vrDisplay,\r\n vrSupported: this._vrSupported,\r\n };\r\n this.onVRDisplayChangedObservable.notifyObservers(eventArgs);\r\n this._webVRInitPromise = new Promise((res) => {\r\n res(eventArgs);\r\n });\r\n };\r\n\r\n if (!this._onVrDisplayConnect) {\r\n this._onVrDisplayConnect = (event) => {\r\n this._vrDisplay = event.display;\r\n notifyObservers();\r\n };\r\n this._onVrDisplayDisconnect = () => {\r\n this._vrDisplay.cancelAnimationFrame(this._frameHandler);\r\n this._vrDisplay = undefined;\r\n this._frameHandler = Engine.QueueNewFrame(this._boundRenderFunction);\r\n notifyObservers();\r\n };\r\n this._onVrDisplayPresentChange = () => {\r\n this._vrExclusivePointerMode = this._vrDisplay && this._vrDisplay.isPresenting;\r\n };\r\n const hostWindow = this.getHostWindow();\r\n if (hostWindow) {\r\n hostWindow.addEventListener(\"vrdisplayconnect\", this._onVrDisplayConnect);\r\n hostWindow.addEventListener(\"vrdisplaydisconnect\", this._onVrDisplayDisconnect);\r\n hostWindow.addEventListener(\"vrdisplaypresentchange\", this._onVrDisplayPresentChange);\r\n }\r\n }\r\n this._webVRInitPromise = this._webVRInitPromise || this._getVRDisplaysAsync();\r\n this._webVRInitPromise.then(notifyObservers);\r\n return this._webVRInitPromise;\r\n};\r\n\r\nEngine.prototype._getVRDisplaysAsync = function (): Promise {\r\n return new Promise((res) => {\r\n if (navigator.getVRDisplays) {\r\n navigator.getVRDisplays().then((devices: Array) => {\r\n this._vrSupported = true;\r\n // note that devices may actually be an empty array. This is fine;\r\n // we expect this._vrDisplay to be undefined in this case.\r\n this._vrDisplay = devices[0];\r\n res({\r\n vrDisplay: this._vrDisplay,\r\n vrSupported: this._vrSupported,\r\n });\r\n });\r\n } else {\r\n this._vrDisplay = undefined;\r\n this._vrSupported = false;\r\n res({\r\n vrDisplay: this._vrDisplay,\r\n vrSupported: this._vrSupported,\r\n });\r\n }\r\n });\r\n};\r\n\r\nEngine.prototype.enableVR = function (options: WebVROptions) {\r\n if (this._vrDisplay && !this._vrDisplay.isPresenting) {\r\n const onResolved = () => {\r\n this.onVRRequestPresentComplete.notifyObservers(true);\r\n this._onVRFullScreenTriggered();\r\n };\r\n const onRejected = () => {\r\n this.onVRRequestPresentComplete.notifyObservers(false);\r\n };\r\n\r\n this.onVRRequestPresentStart.notifyObservers(this);\r\n\r\n const presentationAttributes = {\r\n highRefreshRate: this.vrPresentationAttributes ? this.vrPresentationAttributes.highRefreshRate : false,\r\n foveationLevel: this.vrPresentationAttributes ? this.vrPresentationAttributes.foveationLevel : 1,\r\n multiview: (this.getCaps().multiview || this.getCaps().oculusMultiview) && options.useMultiview,\r\n };\r\n\r\n this._vrDisplay\r\n .requestPresent([\r\n {\r\n source: this.getRenderingCanvas(),\r\n attributes: presentationAttributes,\r\n ...presentationAttributes,\r\n },\r\n ])\r\n .then(onResolved)\r\n .catch(onRejected);\r\n }\r\n};\r\n\r\nEngine.prototype._onVRFullScreenTriggered = function () {\r\n if (this._vrDisplay && this._vrDisplay.isPresenting) {\r\n //get the old size before we change\r\n this._oldSize = new Size(this.getRenderWidth(), this.getRenderHeight());\r\n this._oldHardwareScaleFactor = this.getHardwareScalingLevel();\r\n\r\n //get the width and height, change the render size\r\n const leftEye = this._vrDisplay.getEyeParameters(\"left\");\r\n this.setHardwareScalingLevel(1);\r\n this.setSize(leftEye.renderWidth * 2, leftEye.renderHeight);\r\n } else {\r\n this.setHardwareScalingLevel(this._oldHardwareScaleFactor);\r\n this.setSize(this._oldSize.width, this._oldSize.height);\r\n }\r\n};\r\n\r\nEngine.prototype.disableVR = function () {\r\n if (this._vrDisplay && this._vrDisplay.isPresenting) {\r\n this._vrDisplay\r\n .exitPresent()\r\n .then(() => this._onVRFullScreenTriggered())\r\n .catch(() => this._onVRFullScreenTriggered());\r\n }\r\n\r\n if (IsWindowObjectExist()) {\r\n window.removeEventListener(\"vrdisplaypointerrestricted\", this._onVRDisplayPointerRestricted);\r\n window.removeEventListener(\"vrdisplaypointerunrestricted\", this._onVRDisplayPointerUnrestricted);\r\n\r\n if (this._onVrDisplayConnect) {\r\n window.removeEventListener(\"vrdisplayconnect\", this._onVrDisplayConnect);\r\n if (this._onVrDisplayDisconnect) {\r\n window.removeEventListener(\"vrdisplaydisconnect\", this._onVrDisplayDisconnect);\r\n }\r\n\r\n if (this._onVrDisplayPresentChange) {\r\n window.removeEventListener(\"vrdisplaypresentchange\", this._onVrDisplayPresentChange);\r\n }\r\n this._onVrDisplayConnect = null;\r\n this._onVrDisplayDisconnect = null;\r\n }\r\n }\r\n};\r\n\r\nEngine.prototype._connectVREvents = function (canvas?: HTMLCanvasElement, document?: any) {\r\n this._onVRDisplayPointerRestricted = () => {\r\n if (canvas) {\r\n canvas.requestPointerLock();\r\n }\r\n };\r\n\r\n this._onVRDisplayPointerUnrestricted = () => {\r\n // Edge fix - for some reason document is not present and this is window\r\n if (!document) {\r\n const hostWindow = this.getHostWindow()!;\r\n if (hostWindow.document && hostWindow.document.exitPointerLock) {\r\n hostWindow.document.exitPointerLock();\r\n }\r\n return;\r\n }\r\n if (!document.exitPointerLock) {\r\n return;\r\n }\r\n document.exitPointerLock();\r\n };\r\n\r\n if (IsWindowObjectExist()) {\r\n const hostWindow = this.getHostWindow()!;\r\n hostWindow.addEventListener(\"vrdisplaypointerrestricted\", this._onVRDisplayPointerRestricted, false);\r\n hostWindow.addEventListener(\"vrdisplaypointerunrestricted\", this._onVRDisplayPointerUnrestricted, false);\r\n }\r\n};\r\n\r\nEngine.prototype._submitVRFrame = function () {\r\n // Submit frame to the vr device, if enabled\r\n if (this._vrDisplay && this._vrDisplay.isPresenting) {\r\n // TODO: We should only submit the frame if we read frameData successfully.\r\n try {\r\n this._vrDisplay.submitFrame();\r\n } catch (e) {\r\n Tools.Warn(\"webVR submitFrame has had an unexpected failure: \" + e);\r\n }\r\n }\r\n};\r\n\r\nEngine.prototype.isVRPresenting = function () {\r\n return this._vrDisplay && this._vrDisplay.isPresenting;\r\n};\r\n\r\nEngine.prototype._requestVRFrame = function () {\r\n this._frameHandler = Engine.QueueNewFrame(this._boundRenderFunction, this._vrDisplay);\r\n};\r\n","import type { Nullable } from \"../../types\";\r\nimport type { Observer } from \"../../Misc/observable\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport { FreeCamera } from \"../../Cameras/freeCamera\";\r\nimport type { TargetCamera } from \"../../Cameras/targetCamera\";\r\nimport { Camera } from \"../../Cameras/camera\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { Quaternion, Matrix, Vector3 } from \"../../Maths/math.vector\";\r\nimport { Gamepad } from \"../../Gamepads/gamepad\";\r\nimport { PoseEnabledControllerType } from \"../../Gamepads/Controllers/poseEnabledController\";\r\nimport type { WebVRController } from \"../../Gamepads/Controllers/webVRController\";\r\nimport type { IDisplayChangedEventArgs } from \"../../Engines/engine\";\r\nimport { Node } from \"../../node\";\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport type { Ray } from \"../../Culling/ray\";\r\nimport { HemisphericLight } from \"../../Lights/hemisphericLight\";\r\nimport { Logger } from \"../../Misc/logger\";\r\nimport { VRMultiviewToSingleviewPostProcess } from \"../../PostProcesses/vrMultiviewToSingleviewPostProcess\";\r\nimport { Tools } from \"../../Misc/tools\";\r\nimport { setWebVRRigMode } from \"../RigModes/webVRRigMode\";\r\n\r\n// Side effect import to add webvr support to engine\r\nimport \"../../Engines/Extensions/engine.webVR\";\r\n\r\nNode.AddNodeConstructor(\"WebVRFreeCamera\", (name, scene) => {\r\n return () => new WebVRFreeCamera(name, Vector3.Zero(), scene);\r\n});\r\n\r\nNode.AddNodeConstructor(\"WebVRGamepadCamera\", (name, scene) => {\r\n return () => new WebVRFreeCamera(name, Vector3.Zero(), scene);\r\n});\r\n\r\n/**\r\n * This is a copy of VRPose. See https://developer.mozilla.org/en-US/docs/Web/API/VRPose\r\n * IMPORTANT!! The data is right-hand data.\r\n * @export\r\n * @interface DevicePose\r\n */\r\nexport interface DevicePose {\r\n /**\r\n * The position of the device, values in array are [x,y,z].\r\n */\r\n readonly position: Nullable;\r\n /**\r\n * The linearVelocity of the device, values in array are [x,y,z].\r\n */\r\n readonly linearVelocity: Nullable;\r\n /**\r\n * The linearAcceleration of the device, values in array are [x,y,z].\r\n */\r\n readonly linearAcceleration: Nullable;\r\n\r\n /**\r\n * The orientation of the device in a quaternion array, values in array are [x,y,z,w].\r\n */\r\n readonly orientation: Nullable;\r\n /**\r\n * The angularVelocity of the device, values in array are [x,y,z].\r\n */\r\n readonly angularVelocity: Nullable;\r\n /**\r\n * The angularAcceleration of the device, values in array are [x,y,z].\r\n */\r\n readonly angularAcceleration: Nullable;\r\n}\r\n\r\n/**\r\n * Interface representing a pose controlled object in Babylon.\r\n * A pose controlled object has both regular pose values as well as pose values\r\n * from an external device such as a VR head mounted display\r\n */\r\nexport interface PoseControlled {\r\n /**\r\n * The position of the object in babylon space.\r\n */\r\n position: Vector3;\r\n /**\r\n * The rotation quaternion of the object in babylon space.\r\n */\r\n rotationQuaternion: Quaternion;\r\n /**\r\n * The position of the device in babylon space.\r\n */\r\n devicePosition?: Vector3;\r\n /**\r\n * The rotation quaternion of the device in babylon space.\r\n */\r\n deviceRotationQuaternion: Quaternion;\r\n /**\r\n * The raw pose coming from the device.\r\n */\r\n rawPose: Nullable;\r\n /**\r\n * The scale of the device to be used when translating from device space to babylon space.\r\n */\r\n deviceScaleFactor: number;\r\n /**\r\n * Updates the poseControlled values based on the input device pose.\r\n * @param poseData the pose data to update the object with\r\n */\r\n updateFromDevice(poseData: DevicePose): void;\r\n}\r\n\r\n/**\r\n * Set of options to customize the webVRCamera\r\n */\r\nexport interface WebVROptions {\r\n /**\r\n * Sets if the webVR camera should be tracked to the vrDevice. (default: true)\r\n */\r\n trackPosition?: boolean;\r\n /**\r\n * Sets the scale of the vrDevice in babylon space. (default: 1)\r\n */\r\n positionScale?: number;\r\n /**\r\n * If there are more than one VRDisplays, this will choose the display matching this name. (default: pick first vrDisplay)\r\n */\r\n displayName?: string;\r\n /**\r\n * Should the native controller meshes be initialized. (default: true)\r\n */\r\n controllerMeshes?: boolean;\r\n /**\r\n * Creating a default HemiLight only on controllers. (default: true)\r\n */\r\n defaultLightingOnControllers?: boolean;\r\n /**\r\n * If you don't want to use the default VR button of the helper. (default: false)\r\n */\r\n useCustomVRButton?: boolean;\r\n\r\n /**\r\n * If you'd like to provide your own button to the VRHelper. (default: standard babylon vr button)\r\n */\r\n customVRButton?: HTMLButtonElement;\r\n\r\n /**\r\n * To change the length of the ray for gaze/controllers. Will be scaled by positionScale. (default: 100)\r\n */\r\n rayLength?: number;\r\n\r\n /**\r\n * To change the default offset from the ground to account for user's height in meters. Will be scaled by positionScale. (default: 1.7)\r\n */\r\n defaultHeight?: number;\r\n\r\n /**\r\n * If multiview should be used if available (default: false)\r\n */\r\n useMultiview?: boolean;\r\n}\r\n\r\n/**\r\n * This represents a WebVR camera.\r\n * The WebVR camera is Babylon's simple interface to interaction with Windows Mixed Reality, HTC Vive and Oculus Rift.\r\n * @deprecated Use WebXR instead - https://doc.babylonjs.com/features/featuresDeepDive/webXR\r\n * @example https://doc.babylonjs.com/features/featuresDeepDive/cameras/webVRCamera\r\n */\r\nexport class WebVRFreeCamera extends FreeCamera implements PoseControlled {\r\n /**\r\n * @internal\r\n * The vrDisplay tied to the camera. See https://developer.mozilla.org/en-US/docs/Web/API/VRDisplay\r\n */\r\n public _vrDevice: any = null;\r\n /**\r\n * The rawPose of the vrDevice.\r\n */\r\n public rawPose: Nullable = null;\r\n private _onVREnabled: (success: boolean) => void;\r\n private _specsVersion: string = \"1.1\";\r\n private _attached: boolean = false;\r\n\r\n private _frameData: any;\r\n\r\n protected _descendants: Array = [];\r\n\r\n // Represents device position and rotation in room space. Should only be used to help calculate babylon space values\r\n private _deviceRoomPosition = Vector3.Zero();\r\n /** @internal */\r\n public _deviceRoomRotationQuaternion = Quaternion.Identity();\r\n\r\n private _standingMatrix: Nullable = null;\r\n\r\n /**\r\n * Represents device position in babylon space.\r\n */\r\n public devicePosition = Vector3.Zero();\r\n /**\r\n * Represents device rotation in babylon space.\r\n */\r\n public deviceRotationQuaternion = Quaternion.Identity();\r\n\r\n /**\r\n * The scale of the device to be used when translating from device space to babylon space.\r\n */\r\n public deviceScaleFactor: number = 1;\r\n\r\n private _deviceToWorld = Matrix.Identity();\r\n private _worldToDevice = Matrix.Identity();\r\n\r\n /**\r\n * References to the webVR controllers for the vrDevice.\r\n */\r\n public controllers: Array = [];\r\n /**\r\n * Emits an event when a controller is attached.\r\n */\r\n public onControllersAttachedObservable = new Observable>();\r\n /**\r\n * Emits an event when a controller's mesh has been loaded;\r\n */\r\n public onControllerMeshLoadedObservable = new Observable();\r\n /**\r\n * Emits an event when the HMD's pose has been updated.\r\n */\r\n public onPoseUpdatedFromDeviceObservable = new Observable();\r\n private _poseSet = false;\r\n /**\r\n * If the rig cameras be used as parent instead of this camera.\r\n */\r\n public rigParenting: boolean = true;\r\n\r\n private _lightOnControllers: HemisphericLight;\r\n\r\n private _defaultHeight?: number = undefined;\r\n\r\n /**\r\n * Instantiates a WebVRFreeCamera.\r\n * @param name The name of the WebVRFreeCamera\r\n * @param position The starting anchor position for the camera\r\n * @param scene The scene the camera belongs to\r\n * @param _webVROptions a set of customizable options for the webVRCamera\r\n */\r\n constructor(name: string, position: Vector3, scene?: Scene, private _webVROptions: WebVROptions = {}) {\r\n super(name, position, scene);\r\n this._cache.position = Vector3.Zero();\r\n if (_webVROptions.defaultHeight) {\r\n this._defaultHeight = _webVROptions.defaultHeight;\r\n this.position.y = this._defaultHeight;\r\n }\r\n\r\n this.minZ = 0.1;\r\n\r\n //legacy support - the compensation boolean was removed.\r\n if (arguments.length === 5) {\r\n // eslint-disable-next-line prefer-rest-params\r\n this._webVROptions = arguments[4];\r\n }\r\n\r\n // default webVR options\r\n if (this._webVROptions.trackPosition == undefined) {\r\n this._webVROptions.trackPosition = true;\r\n }\r\n if (this._webVROptions.controllerMeshes == undefined) {\r\n this._webVROptions.controllerMeshes = true;\r\n }\r\n if (this._webVROptions.defaultLightingOnControllers == undefined) {\r\n this._webVROptions.defaultLightingOnControllers = true;\r\n }\r\n\r\n this.rotationQuaternion = new Quaternion();\r\n\r\n if (this._webVROptions && this._webVROptions.positionScale) {\r\n this.deviceScaleFactor = this._webVROptions.positionScale;\r\n }\r\n\r\n //enable VR\r\n const engine = this.getEngine();\r\n this._onVREnabled = (success: boolean) => {\r\n if (success) {\r\n this.initControllers();\r\n }\r\n };\r\n engine.onVRRequestPresentComplete.add(this._onVREnabled);\r\n engine.initWebVR().add((event: IDisplayChangedEventArgs) => {\r\n if (!event.vrDisplay || this._vrDevice === event.vrDisplay) {\r\n return;\r\n }\r\n\r\n this._vrDevice = event.vrDisplay;\r\n\r\n //reset the rig parameters.\r\n this.setCameraRigMode(Camera.RIG_MODE_WEBVR, { parentCamera: this, vrDisplay: this._vrDevice, frameData: this._frameData, specs: this._specsVersion });\r\n\r\n if (this._attached) {\r\n this.getEngine().enableVR(this._webVROptions);\r\n }\r\n });\r\n\r\n if (typeof VRFrameData !== \"undefined\") {\r\n this._frameData = new VRFrameData();\r\n }\r\n\r\n if (_webVROptions.useMultiview) {\r\n if (!this.getScene().getEngine().getCaps().multiview) {\r\n Logger.Warn(\"Multiview is not supported, falling back to standard rendering\");\r\n this._useMultiviewToSingleView = false;\r\n } else {\r\n this._useMultiviewToSingleView = true;\r\n this._rigPostProcess = new VRMultiviewToSingleviewPostProcess(\"VRMultiviewToSingleview\", this, 1.0);\r\n }\r\n }\r\n\r\n /**\r\n * The idea behind the following lines:\r\n * objects that have the camera as parent should actually have the rig cameras as a parent.\r\n * BUT, each of those cameras has a different view matrix, which means that if we set the parent to the first rig camera,\r\n * the second will not show it correctly.\r\n *\r\n * To solve this - each object that has the camera as parent will be added to a protected array.\r\n * When the rig camera renders, it will take this array and set all of those to be its children.\r\n * This way, the right camera will be used as a parent, and the mesh will be rendered correctly.\r\n * Amazing!\r\n */\r\n this.getScene().onBeforeCameraRenderObservable.add((camera) => {\r\n if (camera.parent === this && this.rigParenting) {\r\n this._descendants = this.getDescendants(true, (n) => {\r\n // don't take the cameras or the controllers!\r\n const isController = this.controllers.some((controller) => {\r\n return controller._mesh === n;\r\n });\r\n const isRigCamera = this._rigCameras.indexOf(n) !== -1;\r\n return !isController && !isRigCamera;\r\n });\r\n this._descendants.forEach((node) => {\r\n node.parent = camera;\r\n });\r\n }\r\n });\r\n\r\n this.getScene().onAfterCameraRenderObservable.add((camera) => {\r\n if (camera.parent === this && this.rigParenting) {\r\n this._descendants.forEach((node) => {\r\n node.parent = this;\r\n });\r\n }\r\n });\r\n }\r\n\r\n protected _setRigMode = setWebVRRigMode.bind(null, this);\r\n\r\n /**\r\n * Gets the device distance from the ground in meters.\r\n * @returns the distance in meters from the vrDevice to ground in device space. If standing matrix is not supported for the vrDevice 0 is returned.\r\n */\r\n public deviceDistanceToRoomGround(): number {\r\n if (this._standingMatrix) {\r\n // Add standing matrix offset to get real offset from ground in room\r\n this._standingMatrix.getTranslationToRef(this._workingVector);\r\n return this._deviceRoomPosition.y + this._workingVector.y;\r\n }\r\n //If VRDisplay does not inform stage parameters and no default height is set we fallback to zero.\r\n return this._defaultHeight || 0;\r\n }\r\n\r\n /**\r\n * Enables the standing matrix when supported. This can be used to position the user's view the correct height from the ground.\r\n * @param callback will be called when the standing matrix is set. Callback parameter is if the standing matrix is supported.\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public useStandingMatrix(callback = (bool: boolean) => {}) {\r\n // Use standing matrix if available\r\n this.getEngine()\r\n .initWebVRAsync()\r\n .then((result) => {\r\n if (!result.vrDisplay || !result.vrDisplay.stageParameters || !result.vrDisplay.stageParameters.sittingToStandingTransform || !this._webVROptions.trackPosition) {\r\n callback(false);\r\n } else {\r\n this._standingMatrix = new Matrix();\r\n Matrix.FromFloat32ArrayToRefScaled(result.vrDisplay.stageParameters.sittingToStandingTransform, 0, 1, this._standingMatrix);\r\n if (!this.getScene().useRightHandedSystem) {\r\n if (this._standingMatrix) {\r\n this._standingMatrix.toggleModelMatrixHandInPlace();\r\n }\r\n }\r\n callback(true);\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Enables the standing matrix when supported. This can be used to position the user's view the correct height from the ground.\r\n * @returns A promise with a boolean set to if the standing matrix is supported.\r\n */\r\n public useStandingMatrixAsync(): Promise {\r\n return new Promise((res) => {\r\n this.useStandingMatrix((supported) => {\r\n res(supported);\r\n });\r\n });\r\n }\r\n\r\n /**\r\n * Disposes the camera\r\n */\r\n public dispose(): void {\r\n this._detachIfAttached();\r\n this.getEngine().onVRRequestPresentComplete.removeCallback(this._onVREnabled);\r\n if (this._updateCacheWhenTrackingDisabledObserver) {\r\n this._scene.onBeforeRenderObservable.remove(this._updateCacheWhenTrackingDisabledObserver);\r\n }\r\n super.dispose();\r\n }\r\n\r\n /**\r\n * Gets a vrController by name.\r\n * @param name The name of the controller to retrieve\r\n * @returns the controller matching the name specified or null if not found\r\n */\r\n public getControllerByName(name: string): Nullable {\r\n for (const gp of this.controllers) {\r\n if (gp.hand === name) {\r\n return gp;\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n private _leftController: Nullable;\r\n /**\r\n * The controller corresponding to the users left hand.\r\n */\r\n public get leftController(): Nullable {\r\n if (!this._leftController) {\r\n this._leftController = this.getControllerByName(\"left\");\r\n }\r\n\r\n return this._leftController;\r\n }\r\n\r\n private _rightController: Nullable;\r\n /**\r\n * The controller corresponding to the users right hand.\r\n */\r\n public get rightController(): Nullable {\r\n if (!this._rightController) {\r\n this._rightController = this.getControllerByName(\"right\");\r\n }\r\n\r\n return this._rightController;\r\n }\r\n\r\n /**\r\n * Casts a ray forward from the vrCamera's gaze.\r\n * @param length Length of the ray (default: 100)\r\n * @returns the ray corresponding to the gaze\r\n */\r\n public getForwardRay(length = 100): Ray {\r\n if (this.leftCamera) {\r\n // Use left eye to avoid computation to compute center on every call\r\n return super.getForwardRay(length, this.leftCamera.getWorldMatrix(), this.leftCamera.globalPosition); // Need the actual rendered camera\r\n } else {\r\n return super.getForwardRay(length);\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n * Updates the camera based on device's frame data\r\n */\r\n public _checkInputs(): void {\r\n if (this._vrDevice && this._vrDevice.isPresenting) {\r\n this._vrDevice.getFrameData(this._frameData);\r\n\r\n this.updateFromDevice(this._frameData.pose);\r\n }\r\n\r\n super._checkInputs();\r\n }\r\n\r\n /**\r\n * Updates the poseControlled values based on the input device pose.\r\n * @param poseData Pose coming from the device\r\n */\r\n updateFromDevice(poseData: DevicePose) {\r\n if (poseData && poseData.orientation && poseData.orientation.length === 4) {\r\n this.rawPose = poseData;\r\n this._deviceRoomRotationQuaternion.copyFromFloats(poseData.orientation[0], poseData.orientation[1], -poseData.orientation[2], -poseData.orientation[3]);\r\n\r\n if (this.getScene().useRightHandedSystem) {\r\n this._deviceRoomRotationQuaternion.z *= -1;\r\n this._deviceRoomRotationQuaternion.w *= -1;\r\n }\r\n if (this._webVROptions.trackPosition && this.rawPose.position) {\r\n this._deviceRoomPosition.copyFromFloats(this.rawPose.position[0], this.rawPose.position[1], -this.rawPose.position[2]);\r\n if (this.getScene().useRightHandedSystem) {\r\n this._deviceRoomPosition.z *= -1;\r\n }\r\n }\r\n this._poseSet = true;\r\n }\r\n }\r\n\r\n private _detachIfAttached = () => {\r\n const vrDisplay = this.getEngine().getVRDevice();\r\n if (vrDisplay && !vrDisplay.isPresenting) {\r\n this.detachControl();\r\n }\r\n };\r\n\r\n /**\r\n * WebVR's attach control will start broadcasting frames to the device.\r\n * Note that in certain browsers (chrome for example) this function must be called\r\n * within a user-interaction callback. Example:\r\n *
 scene.onPointerDown = function() { camera.attachControl(canvas); }
\r\n *\r\n * @param noPreventDefault prevent the default html element operation when attaching the vrDevice\r\n */\r\n public attachControl(noPreventDefault?: boolean): void {\r\n // eslint-disable-next-line prefer-rest-params\r\n noPreventDefault = Tools.BackCompatCameraNoPreventDefault(arguments);\r\n super.attachControl(noPreventDefault);\r\n this._attached = true;\r\n\r\n noPreventDefault = Camera.ForceAttachControlToAlwaysPreventDefault ? false : noPreventDefault;\r\n\r\n if (this._vrDevice) {\r\n this.getEngine().enableVR(this._webVROptions);\r\n }\r\n\r\n const hostWindow = this._scene.getEngine().getHostWindow();\r\n\r\n if (hostWindow) {\r\n hostWindow.addEventListener(\"vrdisplaypresentchange\", this._detachIfAttached);\r\n }\r\n }\r\n\r\n /**\r\n * Detach the current controls from the specified dom element.\r\n */\r\n public detachControl(): void {\r\n this.getScene().gamepadManager.onGamepadConnectedObservable.remove(this._onGamepadConnectedObserver);\r\n this.getScene().gamepadManager.onGamepadDisconnectedObservable.remove(this._onGamepadDisconnectedObserver);\r\n\r\n super.detachControl();\r\n this._attached = false;\r\n this.getEngine().disableVR();\r\n window.removeEventListener(\"vrdisplaypresentchange\", this._detachIfAttached);\r\n }\r\n\r\n /**\r\n * @returns the name of this class\r\n */\r\n public getClassName(): string {\r\n return \"WebVRFreeCamera\";\r\n }\r\n\r\n /**\r\n * Calls resetPose on the vrDisplay\r\n * See: https://developer.mozilla.org/en-US/docs/Web/API/VRDisplay/resetPose\r\n */\r\n public resetToCurrentRotation() {\r\n //uses the vrDisplay's \"resetPose()\".\r\n //pitch and roll won't be affected.\r\n this._vrDevice.resetPose();\r\n }\r\n\r\n /**\r\n * @internal\r\n * Updates the rig cameras (left and right eye)\r\n */\r\n public _updateRigCameras() {\r\n const camLeft = this._rigCameras[0];\r\n const camRight = this._rigCameras[1];\r\n camLeft.rotationQuaternion.copyFrom(this._deviceRoomRotationQuaternion);\r\n camRight.rotationQuaternion.copyFrom(this._deviceRoomRotationQuaternion);\r\n\r\n camLeft.position.copyFrom(this._deviceRoomPosition);\r\n camRight.position.copyFrom(this._deviceRoomPosition);\r\n }\r\n\r\n private _workingVector = Vector3.Zero();\r\n private _oneVector = Vector3.One();\r\n private _workingMatrix = Matrix.Identity();\r\n\r\n private _updateCacheCalled: boolean;\r\n\r\n // Remove translation from 6dof headset if trackposition is set to false\r\n private _correctPositionIfNotTrackPosition(matrix: Matrix, isViewMatrix = false) {\r\n if (this.rawPose && this.rawPose.position && !this._webVROptions.trackPosition) {\r\n Matrix.TranslationToRef(this.rawPose.position[0], this.rawPose.position[1], -this.rawPose.position[2], this._tmpMatrix);\r\n if (!isViewMatrix) {\r\n this._tmpMatrix.invert();\r\n }\r\n this._tmpMatrix.multiplyToRef(matrix, matrix);\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n * Updates the cached values of the camera\r\n * @param ignoreParentClass ignores updating the parent class's cache (default: false)\r\n */\r\n public _updateCache(ignoreParentClass?: boolean): void {\r\n if (!this.rotationQuaternion.equals(this._cache.rotationQuaternion) || !this.position.equals(this._cache.position)) {\r\n // Update to ensure devicePosition is up to date with most recent _deviceRoomPosition\r\n if (!this._updateCacheCalled) {\r\n // make sure it is only called once per loop. this.update() might cause an infinite loop.\r\n this._updateCacheCalled = true;\r\n this.update();\r\n }\r\n\r\n // Set working vector to the device position in room space rotated by the new rotation\r\n this.rotationQuaternion.toRotationMatrix(this._workingMatrix);\r\n Vector3.TransformCoordinatesToRef(this._deviceRoomPosition, this._workingMatrix, this._workingVector);\r\n\r\n // Subtract this vector from the current device position in world to get the translation for the device world matrix\r\n this.devicePosition.subtractToRef(this._workingVector, this._workingVector);\r\n Matrix.ComposeToRef(this._oneVector, this.rotationQuaternion, this._workingVector, this._deviceToWorld);\r\n\r\n // Add translation from anchor position\r\n this._deviceToWorld.getTranslationToRef(this._workingVector);\r\n this._workingVector.addInPlace(this.position);\r\n this._workingVector.subtractInPlace(this._cache.position);\r\n this._deviceToWorld.setTranslation(this._workingVector);\r\n\r\n // Set an inverted matrix to be used when updating the camera\r\n this._deviceToWorld.invertToRef(this._worldToDevice);\r\n\r\n // Update the gamepad to ensure the mesh is updated on the same frame as camera\r\n this.controllers.forEach((controller) => {\r\n controller._deviceToWorld.copyFrom(this._deviceToWorld);\r\n this._correctPositionIfNotTrackPosition(controller._deviceToWorld);\r\n controller.update();\r\n });\r\n }\r\n\r\n if (!ignoreParentClass) {\r\n super._updateCache();\r\n }\r\n this._updateCacheCalled = false;\r\n }\r\n\r\n /**\r\n * @internal\r\n * Get current device position in babylon world\r\n */\r\n public _computeDevicePosition() {\r\n Vector3.TransformCoordinatesToRef(this._deviceRoomPosition, this._deviceToWorld, this.devicePosition);\r\n }\r\n\r\n /**\r\n * Updates the current device position and rotation in the babylon world\r\n */\r\n public update() {\r\n this._computeDevicePosition();\r\n\r\n // Get current device rotation in babylon world\r\n Matrix.FromQuaternionToRef(this._deviceRoomRotationQuaternion, this._workingMatrix);\r\n this._workingMatrix.multiplyToRef(this._deviceToWorld, this._workingMatrix);\r\n Quaternion.FromRotationMatrixToRef(this._workingMatrix, this.deviceRotationQuaternion);\r\n\r\n if (this._poseSet) {\r\n this.onPoseUpdatedFromDeviceObservable.notifyObservers(null);\r\n }\r\n super.update();\r\n }\r\n\r\n /**\r\n * @internal\r\n * Gets the view matrix of this camera (Always set to identity as left and right eye cameras contain the actual view matrix)\r\n * @returns an identity matrix\r\n */\r\n public _getViewMatrix(): Matrix {\r\n return Matrix.Identity();\r\n }\r\n\r\n private _tmpMatrix = new Matrix();\r\n /**\r\n * This function is called by the two RIG cameras.\r\n * 'this' is the left or right camera (and NOT (!!!) the WebVRFreeCamera instance)\r\n * @internal\r\n */\r\n public _getWebVRViewMatrix(): Matrix {\r\n // Update the parent camera prior to using a child camera to avoid desynchronization\r\n const parentCamera: WebVRFreeCamera = this._cameraRigParams[\"parentCamera\"];\r\n parentCamera._updateCache();\r\n\r\n //WebVR 1.1\r\n const viewArray = this._cameraRigParams[\"left\"] ? this._cameraRigParams[\"frameData\"].leftViewMatrix : this._cameraRigParams[\"frameData\"].rightViewMatrix;\r\n\r\n Matrix.FromArrayToRef(viewArray, 0, this._webvrViewMatrix);\r\n\r\n if (!this.getScene().useRightHandedSystem) {\r\n this._webvrViewMatrix.toggleModelMatrixHandInPlace();\r\n }\r\n\r\n // update the camera rotation matrix\r\n this._webvrViewMatrix.getRotationMatrixToRef(this._cameraRotationMatrix);\r\n Vector3.TransformCoordinatesToRef(this._referencePoint, this._cameraRotationMatrix, this._transformedReferencePoint);\r\n\r\n // Computing target and final matrix\r\n this.position.addToRef(this._transformedReferencePoint, this._currentTarget);\r\n\r\n // should the view matrix be updated with scale and position offset?\r\n if (parentCamera.deviceScaleFactor !== 1) {\r\n this._webvrViewMatrix.invert();\r\n // scale the position, if set\r\n if (parentCamera.deviceScaleFactor) {\r\n this._webvrViewMatrix.multiplyAtIndex(12, parentCamera.deviceScaleFactor);\r\n this._webvrViewMatrix.multiplyAtIndex(13, parentCamera.deviceScaleFactor);\r\n this._webvrViewMatrix.multiplyAtIndex(14, parentCamera.deviceScaleFactor);\r\n }\r\n\r\n this._webvrViewMatrix.invert();\r\n }\r\n\r\n // Remove translation from 6dof headset if trackposition is set to false\r\n parentCamera._correctPositionIfNotTrackPosition(this._webvrViewMatrix, true);\r\n\r\n parentCamera._worldToDevice.multiplyToRef(this._webvrViewMatrix, this._webvrViewMatrix);\r\n\r\n // Compute global position\r\n this._workingMatrix = this._workingMatrix || Matrix.Identity();\r\n this._webvrViewMatrix.invertToRef(this._workingMatrix);\r\n this._workingMatrix.multiplyToRef(parentCamera.getWorldMatrix(), this._workingMatrix);\r\n this._workingMatrix.getTranslationToRef(this._globalPosition);\r\n this._markSyncedWithParent();\r\n\r\n return this._webvrViewMatrix;\r\n }\r\n\r\n /** @internal */\r\n public _getWebVRProjectionMatrix(): Matrix {\r\n const parentCamera = this.parent;\r\n\r\n parentCamera._vrDevice.depthNear = parentCamera.minZ;\r\n parentCamera._vrDevice.depthFar = parentCamera.maxZ;\r\n\r\n const projectionArray = this._cameraRigParams[\"left\"] ? this._cameraRigParams[\"frameData\"].leftProjectionMatrix : this._cameraRigParams[\"frameData\"].rightProjectionMatrix;\r\n Matrix.FromArrayToRef(projectionArray, 0, this._projectionMatrix);\r\n\r\n //babylon compatible matrix\r\n if (!this.getScene().useRightHandedSystem) {\r\n this._projectionMatrix.toggleProjectionMatrixHandInPlace();\r\n }\r\n\r\n return this._projectionMatrix;\r\n }\r\n\r\n private _onGamepadConnectedObserver: Nullable>;\r\n private _onGamepadDisconnectedObserver: Nullable>;\r\n private _updateCacheWhenTrackingDisabledObserver: Nullable>;\r\n /**\r\n * Initializes the controllers and their meshes\r\n */\r\n public initControllers() {\r\n this.controllers.length = 0;\r\n\r\n const manager = this.getScene().gamepadManager;\r\n this._onGamepadDisconnectedObserver = manager.onGamepadDisconnectedObservable.add((gamepad) => {\r\n if (gamepad.type === Gamepad.POSE_ENABLED) {\r\n const webVrController: WebVRController = gamepad;\r\n\r\n if (webVrController.defaultModel) {\r\n webVrController.defaultModel.setEnabled(false);\r\n }\r\n\r\n if (webVrController.hand === \"right\") {\r\n this._rightController = null;\r\n }\r\n if (webVrController.hand === \"left\") {\r\n this._leftController = null;\r\n }\r\n const controllerIndex = this.controllers.indexOf(webVrController);\r\n if (controllerIndex !== -1) {\r\n this.controllers.splice(controllerIndex, 1);\r\n }\r\n }\r\n });\r\n\r\n this._onGamepadConnectedObserver = manager.onGamepadConnectedObservable.add((gamepad) => {\r\n if (gamepad.type === Gamepad.POSE_ENABLED) {\r\n const webVrController: WebVRController = gamepad;\r\n if (!this._webVROptions.trackPosition) {\r\n webVrController._disableTrackPosition(new Vector3(webVrController.hand == \"left\" ? -0.15 : 0.15, -0.5, 0.25));\r\n // Cache must be updated before rendering controllers to avoid them being one frame behind\r\n if (!this._updateCacheWhenTrackingDisabledObserver) {\r\n this._updateCacheWhenTrackingDisabledObserver = this._scene.onBeforeRenderObservable.add(() => {\r\n this._updateCache();\r\n });\r\n }\r\n }\r\n webVrController.deviceScaleFactor = this.deviceScaleFactor;\r\n webVrController._deviceToWorld.copyFrom(this._deviceToWorld);\r\n this._correctPositionIfNotTrackPosition(webVrController._deviceToWorld);\r\n\r\n if (this._webVROptions.controllerMeshes) {\r\n if (webVrController.defaultModel) {\r\n webVrController.defaultModel.setEnabled(true);\r\n } else {\r\n // Load the meshes\r\n webVrController.initControllerMesh(this.getScene(), (loadedMesh) => {\r\n loadedMesh.scaling.scaleInPlace(this.deviceScaleFactor);\r\n this.onControllerMeshLoadedObservable.notifyObservers(webVrController);\r\n if (this._webVROptions.defaultLightingOnControllers) {\r\n if (!this._lightOnControllers) {\r\n this._lightOnControllers = new HemisphericLight(\"vrControllersLight\", new Vector3(0, 1, 0), this.getScene());\r\n }\r\n const activateLightOnSubMeshes = function (mesh: AbstractMesh, light: HemisphericLight) {\r\n const children = mesh.getChildren();\r\n if (children && children.length !== 0) {\r\n children.forEach((mesh) => {\r\n light.includedOnlyMeshes.push(mesh);\r\n activateLightOnSubMeshes(mesh, light);\r\n });\r\n }\r\n };\r\n this._lightOnControllers.includedOnlyMeshes.push(loadedMesh);\r\n activateLightOnSubMeshes(loadedMesh, this._lightOnControllers);\r\n }\r\n });\r\n }\r\n }\r\n webVrController.attachToPoseControlledCamera(this);\r\n\r\n // since this is async - sanity check. Is the controller already stored?\r\n if (this.controllers.indexOf(webVrController) === -1) {\r\n //add to the controllers array\r\n this.controllers.push(webVrController);\r\n\r\n // Forced to add some control code for Vive as it doesn't always fill properly the \"hand\" property\r\n // Sometimes, both controllers are set correctly (left and right), sometimes none, sometimes only one of them...\r\n // So we're overriding setting left & right manually to be sure\r\n let firstViveWandDetected = false;\r\n\r\n for (let i = 0; i < this.controllers.length; i++) {\r\n if (this.controllers[i].controllerType === PoseEnabledControllerType.VIVE) {\r\n if (!firstViveWandDetected) {\r\n firstViveWandDetected = true;\r\n this.controllers[i].hand = \"left\";\r\n } else {\r\n this.controllers[i].hand = \"right\";\r\n }\r\n }\r\n }\r\n\r\n //did we find enough controllers? Great! let the developer know.\r\n if (this.controllers.length >= 2) {\r\n this.onControllersAttachedObservable.notifyObservers(this.controllers);\r\n }\r\n }\r\n }\r\n });\r\n }\r\n}\r\n","import { Observable } from \"../../Misc/observable\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport type { ExtendedGamepadButton, MutableGamepadButton } from \"./poseEnabledController\";\r\nimport { PoseEnabledController } from \"./poseEnabledController\";\r\nimport type { StickValues, GamepadButtonChanges } from \"../../Gamepads/gamepad\";\r\nimport type { Nullable } from \"../../types\";\r\n\r\n/**\r\n * Defines the WebVRController object that represents controllers tracked in 3D space\r\n * @deprecated Use WebXR instead\r\n */\r\nexport abstract class WebVRController extends PoseEnabledController {\r\n /**\r\n * Internal, the default controller model for the controller\r\n */\r\n protected _defaultModel: Nullable;\r\n\r\n // Observables\r\n /**\r\n * Fired when the trigger state has changed\r\n */\r\n public onTriggerStateChangedObservable = new Observable();\r\n /**\r\n * Fired when the main button state has changed\r\n */\r\n public onMainButtonStateChangedObservable = new Observable();\r\n /**\r\n * Fired when the secondary button state has changed\r\n */\r\n public onSecondaryButtonStateChangedObservable = new Observable();\r\n /**\r\n * Fired when the pad state has changed\r\n */\r\n public onPadStateChangedObservable = new Observable();\r\n /**\r\n * Fired when controllers stick values have changed\r\n */\r\n public onPadValuesChangedObservable = new Observable();\r\n\r\n /**\r\n * Array of button available on the controller\r\n */\r\n protected _buttons: Array;\r\n\r\n private _onButtonStateChange: (controlledIndex: number, buttonIndex: number, state: ExtendedGamepadButton) => void;\r\n\r\n /**\r\n * Fired when a controller button's state has changed\r\n * @param callback the callback containing the button that was modified\r\n */\r\n public onButtonStateChange(callback: (controlledIndex: number, buttonIndex: number, state: ExtendedGamepadButton) => void) {\r\n this._onButtonStateChange = callback;\r\n }\r\n\r\n /**\r\n * X and Y axis corresponding to the controllers joystick\r\n */\r\n public pad: StickValues = { x: 0, y: 0 };\r\n\r\n /**\r\n * 'left' or 'right', see https://w3c.github.io/gamepad/extensions.html#gamepadhand-enum\r\n */\r\n public hand: string;\r\n\r\n /**\r\n * The default controller model for the controller\r\n */\r\n public get defaultModel(): Nullable {\r\n return this._defaultModel;\r\n }\r\n\r\n /**\r\n * Creates a new WebVRController from a gamepad\r\n * @param vrGamepad the gamepad that the WebVRController should be created from\r\n */\r\n constructor(vrGamepad: any) {\r\n super(vrGamepad);\r\n this._buttons = new Array(vrGamepad.buttons.length);\r\n this.hand = vrGamepad.hand;\r\n }\r\n\r\n /**\r\n * Updates the state of the controller and mesh based on the current position and rotation of the controller\r\n */\r\n public update() {\r\n super.update();\r\n for (let index = 0; index < this._buttons.length; index++) {\r\n this._setButtonValue(this.browserGamepad.buttons[index], this._buttons[index], index);\r\n }\r\n if (this.leftStick.x !== this.pad.x || this.leftStick.y !== this.pad.y) {\r\n this.pad.x = this.leftStick.x;\r\n this.pad.y = this.leftStick.y;\r\n this.onPadValuesChangedObservable.notifyObservers(this.pad);\r\n }\r\n }\r\n\r\n /**\r\n * Function to be called when a button is modified\r\n */\r\n protected abstract _handleButtonChange(buttonIdx: number, value: ExtendedGamepadButton, changes: GamepadButtonChanges): void;\r\n\r\n /**\r\n * Loads a mesh and attaches it to the controller\r\n * @param scene the scene the mesh should be added to\r\n * @param meshLoaded callback for when the mesh has been loaded\r\n */\r\n public abstract initControllerMesh(scene: Scene, meshLoaded?: (mesh: AbstractMesh) => void): void;\r\n\r\n private _setButtonValue(newState: ExtendedGamepadButton, currentState: ExtendedGamepadButton, buttonIndex: number) {\r\n if (!newState) {\r\n newState = {\r\n pressed: false,\r\n touched: false,\r\n value: 0,\r\n };\r\n }\r\n if (!currentState) {\r\n this._buttons[buttonIndex] = {\r\n pressed: newState.pressed,\r\n touched: newState.touched,\r\n value: newState.value,\r\n };\r\n return;\r\n }\r\n this._checkChanges(newState, currentState);\r\n if (this._changes.changed) {\r\n this._onButtonStateChange && this._onButtonStateChange(this.index, buttonIndex, newState);\r\n\r\n this._handleButtonChange(buttonIndex, newState, this._changes);\r\n }\r\n this._buttons[buttonIndex].pressed = newState.pressed;\r\n this._buttons[buttonIndex].touched = newState.touched;\r\n // oculus triggers are never 0, thou not touched.\r\n this._buttons[buttonIndex].value = newState.value < 0.00000001 ? 0 : newState.value;\r\n }\r\n\r\n // avoid GC, store state in a tmp object\r\n private _changes: GamepadButtonChanges = {\r\n pressChanged: false,\r\n touchChanged: false,\r\n valueChanged: false,\r\n changed: false,\r\n };\r\n\r\n private _checkChanges(newState: ExtendedGamepadButton, currentState: ExtendedGamepadButton) {\r\n this._changes.pressChanged = newState.pressed !== currentState.pressed;\r\n this._changes.touchChanged = newState.touched !== currentState.touched;\r\n this._changes.valueChanged = newState.value !== currentState.value;\r\n this._changes.changed = this._changes.pressChanged || this._changes.touchChanged || this._changes.valueChanged;\r\n return this._changes;\r\n }\r\n\r\n /**\r\n * Disposes of th webVRController\r\n */\r\n public dispose(): void {\r\n super.dispose();\r\n\r\n this._defaultModel = null;\r\n\r\n this.onTriggerStateChangedObservable.clear();\r\n this.onMainButtonStateChangedObservable.clear();\r\n this.onSecondaryButtonStateChangedObservable.clear();\r\n this.onPadStateChangedObservable.clear();\r\n this.onPadValuesChangedObservable.clear();\r\n }\r\n}\r\n","import { ThinEngine } from \"../../Engines/thinEngine\";\r\nimport { InternalTexture, InternalTextureSource } from \"../../Materials/Textures/internalTexture\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { ICanvas } from \"../ICanvas\";\r\n\r\ndeclare module \"../../Engines/thinEngine\" {\r\n export interface ThinEngine {\r\n /**\r\n * Creates a dynamic texture\r\n * @param width defines the width of the texture\r\n * @param height defines the height of the texture\r\n * @param generateMipMaps defines if the engine should generate the mip levels\r\n * @param samplingMode defines the required sampling mode (Texture.NEAREST_SAMPLINGMODE by default)\r\n * @returns the dynamic texture inside an InternalTexture\r\n */\r\n createDynamicTexture(width: number, height: number, generateMipMaps: boolean, samplingMode: number): InternalTexture;\r\n\r\n /**\r\n * Update the content of a dynamic texture\r\n * @param texture defines the texture to update\r\n * @param source defines the source containing the data\r\n * @param invertY defines if data must be stored with Y axis inverted\r\n * @param premulAlpha defines if alpha is stored as premultiplied\r\n * @param format defines the format of the data\r\n * @param forceBindTexture if the texture should be forced to be bound eg. after a graphics context loss (Default: false)\r\n * @param allowGPUOptimization true to allow some specific GPU optimizations (subject to engine feature \"allowGPUOptimizationsForGUI\" being true)\r\n */\r\n updateDynamicTexture(\r\n texture: Nullable,\r\n source: ImageBitmap | ImageData | HTMLImageElement | HTMLCanvasElement | HTMLVideoElement | OffscreenCanvas | ICanvas,\r\n invertY?: boolean,\r\n premulAlpha?: boolean,\r\n format?: number,\r\n forceBindTexture?: boolean,\r\n allowGPUOptimization?: boolean\r\n ): void;\r\n }\r\n}\r\n\r\nThinEngine.prototype.createDynamicTexture = function (width: number, height: number, generateMipMaps: boolean, samplingMode: number): InternalTexture {\r\n const texture = new InternalTexture(this, InternalTextureSource.Dynamic);\r\n texture.baseWidth = width;\r\n texture.baseHeight = height;\r\n\r\n if (generateMipMaps) {\r\n width = this.needPOTTextures ? ThinEngine.GetExponentOfTwo(width, this._caps.maxTextureSize) : width;\r\n height = this.needPOTTextures ? ThinEngine.GetExponentOfTwo(height, this._caps.maxTextureSize) : height;\r\n }\r\n\r\n // this.resetTextureCache();\r\n texture.width = width;\r\n texture.height = height;\r\n texture.isReady = false;\r\n texture.generateMipMaps = generateMipMaps;\r\n texture.samplingMode = samplingMode;\r\n\r\n this.updateTextureSamplingMode(samplingMode, texture);\r\n\r\n this._internalTexturesCache.push(texture);\r\n\r\n return texture;\r\n};\r\n\r\nThinEngine.prototype.updateDynamicTexture = function (\r\n texture: Nullable,\r\n source: ImageBitmap | ImageData | HTMLImageElement | HTMLCanvasElement | HTMLVideoElement | OffscreenCanvas,\r\n invertY?: boolean,\r\n premulAlpha: boolean = false,\r\n format?: number,\r\n forceBindTexture: boolean = false,\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n allowGPUOptimization: boolean = false\r\n): void {\r\n if (!texture) {\r\n return;\r\n }\r\n\r\n const gl = this._gl;\r\n const target = gl.TEXTURE_2D;\r\n\r\n const wasPreviouslyBound = this._bindTextureDirectly(target, texture, true, forceBindTexture);\r\n\r\n this._unpackFlipY(invertY === undefined ? texture.invertY : invertY);\r\n\r\n if (premulAlpha) {\r\n gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, 1);\r\n }\r\n\r\n const textureType = this._getWebGLTextureType(texture.type);\r\n const glformat = this._getInternalFormat(format ? format : texture.format);\r\n const internalFormat = this._getRGBABufferInternalSizedFormat(texture.type, glformat);\r\n\r\n gl.texImage2D(target, 0, internalFormat, glformat, textureType, source as TexImageSource);\r\n\r\n if (texture.generateMipMaps) {\r\n gl.generateMipmap(target);\r\n }\r\n\r\n if (!wasPreviouslyBound) {\r\n this._bindTextureDirectly(target, null);\r\n }\r\n\r\n if (premulAlpha) {\r\n gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, 0);\r\n }\r\n\r\n texture.isReady = true;\r\n};\r\n","import { Logger } from \"../../Misc/logger\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { ISize } from \"../../Maths/math.size\";\r\nimport { Texture } from \"../../Materials/Textures/texture\";\r\nimport { Constants } from \"../../Engines/constants\";\r\nimport \"../../Engines/Extensions/engine.dynamicTexture\";\r\nimport type { ICanvas, ICanvasRenderingContext } from \"../../Engines/ICanvas\";\r\n\r\n/**\r\n * A class extending Texture allowing drawing on a texture\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/dynamicTexture\r\n */\r\nexport class DynamicTexture extends Texture {\r\n private _generateMipMaps: boolean;\r\n private _canvas: ICanvas;\r\n private _context: ICanvasRenderingContext;\r\n\r\n /**\r\n * Creates a DynamicTexture\r\n * @param name defines the name of the texture\r\n * @param options provides 3 alternatives for width and height of texture, a canvas, object with width and height properties, number for both width and height\r\n * @param scene defines the scene where you want the texture\r\n * @param generateMipMaps defines the use of MinMaps or not (default is false)\r\n * @param samplingMode defines the sampling mode to use (default is Texture.TRILINEAR_SAMPLINGMODE)\r\n * @param format defines the texture format to use (default is Engine.TEXTUREFORMAT_RGBA)\r\n * @param invertY defines if the texture needs to be inverted on the y axis during loading\r\n */\r\n\r\n constructor(\r\n name: string,\r\n options: any,\r\n scene: Nullable = null,\r\n generateMipMaps: boolean = false,\r\n samplingMode: number = Constants.TEXTURE_TRILINEAR_SAMPLINGMODE,\r\n format: number = Constants.TEXTUREFORMAT_RGBA,\r\n invertY?: boolean\r\n ) {\r\n super(null, scene, !generateMipMaps, invertY, samplingMode, undefined, undefined, undefined, undefined, format);\r\n\r\n this.name = name;\r\n this.wrapU = Texture.CLAMP_ADDRESSMODE;\r\n this.wrapV = Texture.CLAMP_ADDRESSMODE;\r\n\r\n this._generateMipMaps = generateMipMaps;\r\n\r\n const engine = this._getEngine();\r\n if (!engine) {\r\n return;\r\n }\r\n\r\n if (options.getContext) {\r\n this._canvas = options;\r\n this._texture = engine.createDynamicTexture(options.width, options.height, generateMipMaps, samplingMode);\r\n } else {\r\n this._canvas = engine.createCanvas(1, 1);\r\n\r\n if (options.width || options.width === 0) {\r\n this._texture = engine.createDynamicTexture(options.width, options.height, generateMipMaps, samplingMode);\r\n } else {\r\n this._texture = engine.createDynamicTexture(options, options, generateMipMaps, samplingMode);\r\n }\r\n }\r\n\r\n const textureSize = this.getSize();\r\n\r\n if (this._canvas.width !== textureSize.width) {\r\n this._canvas.width = textureSize.width;\r\n }\r\n if (this._canvas.height !== textureSize.height) {\r\n this._canvas.height = textureSize.height;\r\n }\r\n this._context = this._canvas.getContext(\"2d\");\r\n }\r\n\r\n /**\r\n * Get the current class name of the texture useful for serialization or dynamic coding.\r\n * @returns \"DynamicTexture\"\r\n */\r\n public getClassName(): string {\r\n return \"DynamicTexture\";\r\n }\r\n\r\n /**\r\n * Gets the current state of canRescale\r\n */\r\n public get canRescale(): boolean {\r\n return true;\r\n }\r\n\r\n private _recreate(textureSize: ISize): void {\r\n this._canvas.width = textureSize.width;\r\n this._canvas.height = textureSize.height;\r\n\r\n this.releaseInternalTexture();\r\n\r\n this._texture = this._getEngine()!.createDynamicTexture(textureSize.width, textureSize.height, this._generateMipMaps, this.samplingMode);\r\n }\r\n\r\n /**\r\n * Scales the texture\r\n * @param ratio the scale factor to apply to both width and height\r\n */\r\n public scale(ratio: number): void {\r\n const textureSize = this.getSize();\r\n\r\n textureSize.width *= ratio;\r\n textureSize.height *= ratio;\r\n\r\n this._recreate(textureSize);\r\n }\r\n\r\n /**\r\n * Resizes the texture\r\n * @param width the new width\r\n * @param height the new height\r\n */\r\n public scaleTo(width: number, height: number): void {\r\n const textureSize = this.getSize();\r\n\r\n textureSize.width = width;\r\n textureSize.height = height;\r\n\r\n this._recreate(textureSize);\r\n }\r\n\r\n /**\r\n * Gets the context of the canvas used by the texture\r\n * @returns the canvas context of the dynamic texture\r\n */\r\n public getContext(): ICanvasRenderingContext {\r\n return this._context;\r\n }\r\n\r\n /**\r\n * Clears the texture\r\n */\r\n public clear(): void {\r\n const size = this.getSize();\r\n this._context.fillRect(0, 0, size.width, size.height);\r\n }\r\n\r\n /**\r\n * Updates the texture\r\n * @param invertY defines the direction for the Y axis (default is true - y increases downwards)\r\n * @param premulAlpha defines if alpha is stored as premultiplied (default is false)\r\n * @param allowGPUOptimization true to allow some specific GPU optimizations (subject to engine feature \"allowGPUOptimizationsForGUI\" being true)\r\n */\r\n public update(invertY?: boolean, premulAlpha = false, allowGPUOptimization = false): void {\r\n this._getEngine()!.updateDynamicTexture(\r\n this._texture,\r\n this._canvas,\r\n invertY === undefined ? true : invertY,\r\n premulAlpha,\r\n this._format || undefined,\r\n undefined,\r\n allowGPUOptimization\r\n );\r\n }\r\n\r\n /**\r\n * Draws text onto the texture\r\n * @param text defines the text to be drawn\r\n * @param x defines the placement of the text from the left\r\n * @param y defines the placement of the text from the top when invertY is true and from the bottom when false\r\n * @param font defines the font to be used with font-style, font-size, font-name\r\n * @param color defines the color used for the text\r\n * @param clearColor defines the color for the canvas, use null to not overwrite canvas\r\n * @param invertY defines the direction for the Y axis (default is true - y increases downwards)\r\n * @param update defines whether texture is immediately update (default is true)\r\n */\r\n public drawText(\r\n text: string,\r\n x: number | null | undefined,\r\n y: number | null | undefined,\r\n font: string,\r\n color: string | null,\r\n clearColor: string | null,\r\n invertY?: boolean,\r\n update = true\r\n ) {\r\n const size = this.getSize();\r\n if (clearColor) {\r\n this._context.fillStyle = clearColor;\r\n this._context.fillRect(0, 0, size.width, size.height);\r\n }\r\n\r\n this._context.font = font;\r\n if (x === null || x === undefined) {\r\n const textSize = this._context.measureText(text);\r\n x = (size.width - textSize.width) / 2;\r\n }\r\n if (y === null || y === undefined) {\r\n const fontSize = parseInt(font.replace(/\\D/g, \"\"));\r\n y = size.height / 2 + fontSize / 3.65;\r\n }\r\n\r\n this._context.fillStyle = color || \"\";\r\n this._context.fillText(text, x, y);\r\n\r\n if (update) {\r\n this.update(invertY);\r\n }\r\n }\r\n\r\n /**\r\n * Clones the texture\r\n * @returns the clone of the texture.\r\n */\r\n public clone(): DynamicTexture {\r\n const scene = this.getScene();\r\n\r\n if (!scene) {\r\n return this;\r\n }\r\n\r\n const textureSize = this.getSize();\r\n const newTexture = new DynamicTexture(this.name, textureSize, scene, this._generateMipMaps);\r\n\r\n // Base texture\r\n newTexture.hasAlpha = this.hasAlpha;\r\n newTexture.level = this.level;\r\n\r\n // Dynamic Texture\r\n newTexture.wrapU = this.wrapU;\r\n newTexture.wrapV = this.wrapV;\r\n\r\n return newTexture;\r\n }\r\n\r\n /**\r\n * Serializes the dynamic texture. The scene should be ready before the dynamic texture is serialized\r\n * @returns a serialized dynamic texture object\r\n */\r\n public serialize(): any {\r\n const scene = this.getScene();\r\n if (scene && !scene.isReady()) {\r\n Logger.Warn(\"The scene must be ready before serializing the dynamic texture\");\r\n }\r\n\r\n const serializationObject = super.serialize();\r\n if (DynamicTexture._IsCanvasElement(this._canvas)) {\r\n serializationObject.base64String = this._canvas.toDataURL();\r\n }\r\n\r\n serializationObject.invertY = this._invertY;\r\n serializationObject.samplingMode = this.samplingMode;\r\n\r\n return serializationObject;\r\n }\r\n\r\n private static _IsCanvasElement(canvas: HTMLCanvasElement | OffscreenCanvas | ICanvas): canvas is HTMLCanvasElement {\r\n return (canvas as HTMLCanvasElement).toDataURL !== undefined;\r\n }\r\n\r\n /** @internal */\r\n public _rebuild(): void {\r\n this.update();\r\n }\r\n}\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/imageProcessingDeclaration\";\nimport \"./ShadersInclude/helperFunctions\";\nimport \"./ShadersInclude/imageProcessingFunctions\";\n\nconst name = \"imageProcessingPixelShader\";\nconst shader = `varying vec2 vUV;uniform sampler2D textureSampler;\n#include\n#include\n#include\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{vec4 result=texture2D(textureSampler,vUV);\n#ifdef IMAGEPROCESSING\n#ifndef FROMLINEARSPACE\nresult.rgb=toLinearSpace(result.rgb);\n#endif\nresult=applyImageProcessing(result);\n#else\n#ifdef FROMLINEARSPACE\nresult=applyImageProcessing(result);\n#endif\n#endif\ngl_FragColor=result;}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const imageProcessingPixelShader = { name, shader };\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport type { Nullable } from \"../types\";\r\nimport type { Observer } from \"../Misc/observable\";\r\nimport { serialize } from \"../Misc/decorators\";\r\nimport type { Color4 } from \"../Maths/math.color\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\nimport type { BaseTexture } from \"../Materials/Textures/baseTexture\";\r\nimport type { Effect } from \"../Materials/effect\";\r\nimport type { ColorCurves } from \"../Materials/colorCurves\";\r\nimport type { IImageProcessingConfigurationDefines } from \"../Materials/imageProcessingConfiguration\";\r\nimport { ImageProcessingConfiguration } from \"../Materials/imageProcessingConfiguration\";\r\nimport type { PostProcessOptions } from \"./postProcess\";\r\nimport { PostProcess } from \"./postProcess\";\r\nimport type { Engine } from \"../Engines/engine\";\r\nimport { EngineStore } from \"../Engines/engineStore\";\r\nimport { Constants } from \"../Engines/constants\";\r\n\r\nimport \"../Shaders/imageProcessing.fragment\";\r\nimport \"../Shaders/postprocess.vertex\";\r\n\r\n/**\r\n * ImageProcessingPostProcess\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/usePostProcesses#imageprocessing\r\n */\r\nexport class ImageProcessingPostProcess extends PostProcess {\r\n /**\r\n * Default configuration related to image processing available in the PBR Material.\r\n */\r\n protected _imageProcessingConfiguration: ImageProcessingConfiguration;\r\n\r\n /**\r\n * Gets the image processing configuration used either in this material.\r\n */\r\n public get imageProcessingConfiguration(): ImageProcessingConfiguration {\r\n return this._imageProcessingConfiguration;\r\n }\r\n\r\n /**\r\n * Sets the Default image processing configuration used either in the this material.\r\n *\r\n * If sets to null, the scene one is in use.\r\n */\r\n public set imageProcessingConfiguration(value: ImageProcessingConfiguration) {\r\n // We are almost sure it is applied by post process as\r\n // We are in the post process :-)\r\n value.applyByPostProcess = true;\r\n this._attachImageProcessingConfiguration(value);\r\n }\r\n\r\n /**\r\n * Keep track of the image processing observer to allow dispose and replace.\r\n */\r\n private _imageProcessingObserver: Nullable>;\r\n\r\n /**\r\n * Attaches a new image processing configuration to the PBR Material.\r\n * @param configuration\r\n * @param doNotBuild\r\n */\r\n protected _attachImageProcessingConfiguration(configuration: Nullable, doNotBuild = false): void {\r\n if (configuration === this._imageProcessingConfiguration) {\r\n return;\r\n }\r\n\r\n // Detaches observer.\r\n if (this._imageProcessingConfiguration && this._imageProcessingObserver) {\r\n this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver);\r\n }\r\n\r\n // Pick the scene configuration if needed.\r\n if (!configuration) {\r\n let scene = null;\r\n const engine = this.getEngine();\r\n const camera = this.getCamera();\r\n\r\n if (camera) {\r\n scene = camera.getScene();\r\n } else if (engine && engine.scenes) {\r\n const scenes = engine.scenes;\r\n scene = scenes[scenes.length - 1];\r\n } else {\r\n scene = EngineStore.LastCreatedScene;\r\n }\r\n\r\n if (scene) {\r\n this._imageProcessingConfiguration = scene.imageProcessingConfiguration;\r\n } else {\r\n this._imageProcessingConfiguration = new ImageProcessingConfiguration();\r\n }\r\n } else {\r\n this._imageProcessingConfiguration = configuration;\r\n }\r\n\r\n // Attaches observer.\r\n if (this._imageProcessingConfiguration) {\r\n this._imageProcessingObserver = this._imageProcessingConfiguration.onUpdateParameters.add(() => {\r\n this._updateParameters();\r\n });\r\n }\r\n\r\n // Ensure the effect will be rebuilt.\r\n if (!doNotBuild) {\r\n this._updateParameters();\r\n }\r\n }\r\n\r\n /**\r\n * If the post process is supported.\r\n */\r\n public get isSupported(): boolean {\r\n const effect = this.getEffect();\r\n return !effect || effect.isSupported;\r\n }\r\n\r\n /**\r\n * Gets Color curves setup used in the effect if colorCurvesEnabled is set to true .\r\n */\r\n public get colorCurves(): Nullable {\r\n return this.imageProcessingConfiguration.colorCurves;\r\n }\r\n /**\r\n * Sets Color curves setup used in the effect if colorCurvesEnabled is set to true .\r\n */\r\n public set colorCurves(value: Nullable) {\r\n this.imageProcessingConfiguration.colorCurves = value;\r\n }\r\n\r\n /**\r\n * Gets whether the color curves effect is enabled.\r\n */\r\n public get colorCurvesEnabled(): boolean {\r\n return this.imageProcessingConfiguration.colorCurvesEnabled;\r\n }\r\n /**\r\n * Sets whether the color curves effect is enabled.\r\n */\r\n public set colorCurvesEnabled(value: boolean) {\r\n this.imageProcessingConfiguration.colorCurvesEnabled = value;\r\n }\r\n\r\n /**\r\n * Gets Color grading LUT texture used in the effect if colorGradingEnabled is set to true.\r\n */\r\n public get colorGradingTexture(): Nullable {\r\n return this.imageProcessingConfiguration.colorGradingTexture;\r\n }\r\n /**\r\n * Sets Color grading LUT texture used in the effect if colorGradingEnabled is set to true.\r\n */\r\n public set colorGradingTexture(value: Nullable) {\r\n this.imageProcessingConfiguration.colorGradingTexture = value;\r\n }\r\n\r\n /**\r\n * Gets whether the color grading effect is enabled.\r\n */\r\n public get colorGradingEnabled(): boolean {\r\n return this.imageProcessingConfiguration.colorGradingEnabled;\r\n }\r\n /**\r\n * Gets whether the color grading effect is enabled.\r\n */\r\n public set colorGradingEnabled(value: boolean) {\r\n this.imageProcessingConfiguration.colorGradingEnabled = value;\r\n }\r\n\r\n /**\r\n * Gets exposure used in the effect.\r\n */\r\n public get exposure(): number {\r\n return this.imageProcessingConfiguration.exposure;\r\n }\r\n /**\r\n * Sets exposure used in the effect.\r\n */\r\n public set exposure(value: number) {\r\n this.imageProcessingConfiguration.exposure = value;\r\n }\r\n\r\n /**\r\n * Gets whether tonemapping is enabled or not.\r\n */\r\n public get toneMappingEnabled(): boolean {\r\n return this._imageProcessingConfiguration.toneMappingEnabled;\r\n }\r\n /**\r\n * Sets whether tonemapping is enabled or not\r\n */\r\n public set toneMappingEnabled(value: boolean) {\r\n this._imageProcessingConfiguration.toneMappingEnabled = value;\r\n }\r\n\r\n /**\r\n * Gets the type of tone mapping effect.\r\n */\r\n public get toneMappingType(): number {\r\n return this._imageProcessingConfiguration.toneMappingType;\r\n }\r\n /**\r\n * Sets the type of tone mapping effect.\r\n */\r\n public set toneMappingType(value: number) {\r\n this._imageProcessingConfiguration.toneMappingType = value;\r\n }\r\n\r\n /**\r\n * Gets contrast used in the effect.\r\n */\r\n public get contrast(): number {\r\n return this.imageProcessingConfiguration.contrast;\r\n }\r\n /**\r\n * Sets contrast used in the effect.\r\n */\r\n public set contrast(value: number) {\r\n this.imageProcessingConfiguration.contrast = value;\r\n }\r\n\r\n /**\r\n * Gets Vignette stretch size.\r\n */\r\n public get vignetteStretch(): number {\r\n return this.imageProcessingConfiguration.vignetteStretch;\r\n }\r\n /**\r\n * Sets Vignette stretch size.\r\n */\r\n public set vignetteStretch(value: number) {\r\n this.imageProcessingConfiguration.vignetteStretch = value;\r\n }\r\n\r\n /**\r\n * Gets Vignette center X Offset.\r\n * @deprecated use vignetteCenterX instead\r\n */\r\n public get vignetteCentreX(): number {\r\n return this.imageProcessingConfiguration.vignetteCenterX;\r\n }\r\n /**\r\n * Sets Vignette center X Offset.\r\n * @deprecated use vignetteCenterX instead\r\n */\r\n public set vignetteCentreX(value: number) {\r\n this.imageProcessingConfiguration.vignetteCenterX = value;\r\n }\r\n\r\n /**\r\n * Gets Vignette center Y Offset.\r\n * @deprecated use vignetteCenterY instead\r\n */\r\n public get vignetteCentreY(): number {\r\n return this.imageProcessingConfiguration.vignetteCenterY;\r\n }\r\n /**\r\n * Sets Vignette center Y Offset.\r\n * @deprecated use vignetteCenterY instead\r\n */\r\n public set vignetteCentreY(value: number) {\r\n this.imageProcessingConfiguration.vignetteCenterY = value;\r\n }\r\n\r\n /**\r\n * Vignette center Y Offset.\r\n */\r\n public get vignetteCenterY(): number {\r\n return this.imageProcessingConfiguration.vignetteCenterY;\r\n }\r\n public set vignetteCenterY(value: number) {\r\n this.imageProcessingConfiguration.vignetteCenterY = value;\r\n }\r\n\r\n /**\r\n * Vignette center X Offset.\r\n */\r\n public get vignetteCenterX(): number {\r\n return this.imageProcessingConfiguration.vignetteCenterX;\r\n }\r\n public set vignetteCenterX(value: number) {\r\n this.imageProcessingConfiguration.vignetteCenterX = value;\r\n }\r\n\r\n /**\r\n * Gets Vignette weight or intensity of the vignette effect.\r\n */\r\n public get vignetteWeight(): number {\r\n return this.imageProcessingConfiguration.vignetteWeight;\r\n }\r\n /**\r\n * Sets Vignette weight or intensity of the vignette effect.\r\n */\r\n public set vignetteWeight(value: number) {\r\n this.imageProcessingConfiguration.vignetteWeight = value;\r\n }\r\n\r\n /**\r\n * Gets Color of the vignette applied on the screen through the chosen blend mode (vignetteBlendMode)\r\n * if vignetteEnabled is set to true.\r\n */\r\n public get vignetteColor(): Color4 {\r\n return this.imageProcessingConfiguration.vignetteColor;\r\n }\r\n /**\r\n * Sets Color of the vignette applied on the screen through the chosen blend mode (vignetteBlendMode)\r\n * if vignetteEnabled is set to true.\r\n */\r\n public set vignetteColor(value: Color4) {\r\n this.imageProcessingConfiguration.vignetteColor = value;\r\n }\r\n\r\n /**\r\n * Gets Camera field of view used by the Vignette effect.\r\n */\r\n public get vignetteCameraFov(): number {\r\n return this.imageProcessingConfiguration.vignetteCameraFov;\r\n }\r\n /**\r\n * Sets Camera field of view used by the Vignette effect.\r\n */\r\n public set vignetteCameraFov(value: number) {\r\n this.imageProcessingConfiguration.vignetteCameraFov = value;\r\n }\r\n\r\n /**\r\n * Gets the vignette blend mode allowing different kind of effect.\r\n */\r\n public get vignetteBlendMode(): number {\r\n return this.imageProcessingConfiguration.vignetteBlendMode;\r\n }\r\n /**\r\n * Sets the vignette blend mode allowing different kind of effect.\r\n */\r\n public set vignetteBlendMode(value: number) {\r\n this.imageProcessingConfiguration.vignetteBlendMode = value;\r\n }\r\n\r\n /**\r\n * Gets whether the vignette effect is enabled.\r\n */\r\n public get vignetteEnabled(): boolean {\r\n return this.imageProcessingConfiguration.vignetteEnabled;\r\n }\r\n /**\r\n * Sets whether the vignette effect is enabled.\r\n */\r\n public set vignetteEnabled(value: boolean) {\r\n this.imageProcessingConfiguration.vignetteEnabled = value;\r\n }\r\n\r\n /**\r\n * Gets intensity of the dithering effect.\r\n */\r\n public get ditheringIntensity(): number {\r\n return this.imageProcessingConfiguration.ditheringIntensity;\r\n }\r\n /**\r\n * Sets intensity of the dithering effect.\r\n */\r\n public set ditheringIntensity(value: number) {\r\n this.imageProcessingConfiguration.ditheringIntensity = value;\r\n }\r\n\r\n /**\r\n * Gets whether the dithering effect is enabled.\r\n */\r\n public get ditheringEnabled(): boolean {\r\n return this.imageProcessingConfiguration.ditheringEnabled;\r\n }\r\n /**\r\n * Sets whether the dithering effect is enabled.\r\n */\r\n public set ditheringEnabled(value: boolean) {\r\n this.imageProcessingConfiguration.ditheringEnabled = value;\r\n }\r\n\r\n @serialize()\r\n private _fromLinearSpace = true;\r\n /**\r\n * Gets whether the input of the processing is in Gamma or Linear Space.\r\n */\r\n public get fromLinearSpace(): boolean {\r\n return this._fromLinearSpace;\r\n }\r\n /**\r\n * Sets whether the input of the processing is in Gamma or Linear Space.\r\n */\r\n public set fromLinearSpace(value: boolean) {\r\n if (this._fromLinearSpace === value) {\r\n return;\r\n }\r\n\r\n this._fromLinearSpace = value;\r\n this._updateParameters();\r\n }\r\n\r\n /**\r\n * Defines cache preventing GC.\r\n */\r\n private _defines: IImageProcessingConfigurationDefines & { FROMLINEARSPACE: boolean } = {\r\n IMAGEPROCESSING: false,\r\n VIGNETTE: false,\r\n VIGNETTEBLENDMODEMULTIPLY: false,\r\n VIGNETTEBLENDMODEOPAQUE: false,\r\n TONEMAPPING: false,\r\n TONEMAPPING_ACES: false,\r\n CONTRAST: false,\r\n COLORCURVES: false,\r\n COLORGRADING: false,\r\n COLORGRADING3D: false,\r\n FROMLINEARSPACE: false,\r\n SAMPLER3DGREENDEPTH: false,\r\n SAMPLER3DBGRMAP: false,\r\n DITHER: false,\r\n IMAGEPROCESSINGPOSTPROCESS: false,\r\n EXPOSURE: false,\r\n SKIPFINALCOLORCLAMP: false,\r\n };\r\n\r\n constructor(\r\n name: string,\r\n options: number | PostProcessOptions,\r\n camera: Nullable = null,\r\n samplingMode?: number,\r\n engine?: Engine,\r\n reusable?: boolean,\r\n textureType: number = Constants.TEXTURETYPE_UNSIGNED_INT,\r\n imageProcessingConfiguration?: ImageProcessingConfiguration\r\n ) {\r\n super(name, \"imageProcessing\", [], [], options, camera, samplingMode, engine, reusable, null, textureType, \"postprocess\", null, true);\r\n\r\n // Setup the configuration as forced by the constructor. This would then not force the\r\n // scene materials output in linear space and let untouched the default forward pass.\r\n if (imageProcessingConfiguration) {\r\n imageProcessingConfiguration.applyByPostProcess = true;\r\n this._attachImageProcessingConfiguration(imageProcessingConfiguration, true);\r\n // This will cause the shader to be compiled\r\n this._updateParameters();\r\n }\r\n // Setup the default processing configuration to the scene.\r\n else {\r\n this._attachImageProcessingConfiguration(null, true);\r\n this.imageProcessingConfiguration.applyByPostProcess = true;\r\n }\r\n\r\n this.onApply = (effect: Effect) => {\r\n this.imageProcessingConfiguration.bind(effect, this.aspectRatio);\r\n };\r\n }\r\n /**\r\n * \"ImageProcessingPostProcess\"\r\n * @returns \"ImageProcessingPostProcess\"\r\n */\r\n public getClassName(): string {\r\n return \"ImageProcessingPostProcess\";\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _updateParameters(): void {\r\n this._defines.FROMLINEARSPACE = this._fromLinearSpace;\r\n this.imageProcessingConfiguration.prepareDefines(this._defines, true);\r\n let defines = \"\";\r\n for (const define in this._defines) {\r\n if ((this._defines)[define]) {\r\n defines += `#define ${define};\\n`;\r\n }\r\n }\r\n\r\n const samplers = [\"textureSampler\"];\r\n const uniforms = [\"scale\"];\r\n\r\n if (ImageProcessingConfiguration) {\r\n ImageProcessingConfiguration.PrepareSamplers(samplers, this._defines);\r\n ImageProcessingConfiguration.PrepareUniforms(uniforms, this._defines);\r\n }\r\n\r\n this.updateEffect(defines, uniforms, samplers);\r\n }\r\n\r\n public dispose(camera?: Camera): void {\r\n super.dispose(camera);\r\n\r\n if (this._imageProcessingConfiguration && this._imageProcessingObserver) {\r\n this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver);\r\n }\r\n\r\n if (this._imageProcessingConfiguration) {\r\n this.imageProcessingConfiguration.applyByPostProcess = false;\r\n }\r\n }\r\n}\r\n","import { Observable } from \"../Misc/observable\";\r\nimport { Gamepad } from \"./gamepad\";\r\n\r\n/**\r\n * Defines supported buttons for DualShock compatible gamepads\r\n */\r\nexport enum DualShockButton {\r\n /** Cross */\r\n Cross = 0,\r\n /** Circle */\r\n Circle = 1,\r\n /** Square */\r\n Square = 2,\r\n /** Triangle */\r\n Triangle = 3,\r\n /** L1 */\r\n L1 = 4,\r\n /** R1 */\r\n R1 = 5,\r\n /** Share */\r\n Share = 8,\r\n /** Options */\r\n Options = 9,\r\n /** Left stick */\r\n LeftStick = 10,\r\n /** Right stick */\r\n RightStick = 11,\r\n}\r\n\r\n/** Defines values for DualShock DPad */\r\nexport enum DualShockDpad {\r\n /** Up */\r\n Up = 12,\r\n /** Down */\r\n Down = 13,\r\n /** Left */\r\n Left = 14,\r\n /** Right */\r\n Right = 15,\r\n}\r\n\r\n/**\r\n * Defines a DualShock gamepad\r\n */\r\nexport class DualShockPad extends Gamepad {\r\n private _leftTrigger: number = 0;\r\n private _rightTrigger: number = 0;\r\n\r\n private _onlefttriggerchanged: (value: number) => void;\r\n private _onrighttriggerchanged: (value: number) => void;\r\n\r\n private _onbuttondown: (buttonPressed: DualShockButton) => void;\r\n private _onbuttonup: (buttonReleased: DualShockButton) => void;\r\n private _ondpaddown: (dPadPressed: DualShockDpad) => void;\r\n private _ondpadup: (dPadReleased: DualShockDpad) => void;\r\n\r\n /** Observable raised when a button is pressed */\r\n public onButtonDownObservable = new Observable();\r\n /** Observable raised when a button is released */\r\n public onButtonUpObservable = new Observable();\r\n /** Observable raised when a pad is pressed */\r\n public onPadDownObservable = new Observable();\r\n /** Observable raised when a pad is released */\r\n public onPadUpObservable = new Observable();\r\n\r\n private _buttonCross: number = 0;\r\n private _buttonCircle: number = 0;\r\n private _buttonSquare: number = 0;\r\n private _buttonTriangle: number = 0;\r\n private _buttonShare: number = 0;\r\n private _buttonOptions: number = 0;\r\n private _buttonL1: number = 0;\r\n private _buttonR1: number = 0;\r\n\r\n private _buttonLeftStick: number = 0;\r\n private _buttonRightStick: number = 0;\r\n private _dPadUp: number = 0;\r\n private _dPadDown: number = 0;\r\n private _dPadLeft: number = 0;\r\n private _dPadRight: number = 0;\r\n\r\n /**\r\n * Creates a new DualShock gamepad object\r\n * @param id defines the id of this gamepad\r\n * @param index defines its index\r\n * @param gamepad defines the internal HTML gamepad object\r\n */\r\n constructor(id: string, index: number, gamepad: any) {\r\n super(id.replace(\"STANDARD GAMEPAD\", \"SONY PLAYSTATION DUALSHOCK\"), index, gamepad, 0, 1, 2, 3);\r\n this.type = Gamepad.DUALSHOCK;\r\n }\r\n\r\n /**\r\n * Defines the callback to call when left trigger is pressed\r\n * @param callback defines the callback to use\r\n */\r\n public onlefttriggerchanged(callback: (value: number) => void) {\r\n this._onlefttriggerchanged = callback;\r\n }\r\n\r\n /**\r\n * Defines the callback to call when right trigger is pressed\r\n * @param callback defines the callback to use\r\n */\r\n public onrighttriggerchanged(callback: (value: number) => void) {\r\n this._onrighttriggerchanged = callback;\r\n }\r\n\r\n /**\r\n * Gets the left trigger value\r\n */\r\n public get leftTrigger(): number {\r\n return this._leftTrigger;\r\n }\r\n /**\r\n * Sets the left trigger value\r\n */\r\n public set leftTrigger(newValue: number) {\r\n if (this._onlefttriggerchanged && this._leftTrigger !== newValue) {\r\n this._onlefttriggerchanged(newValue);\r\n }\r\n this._leftTrigger = newValue;\r\n }\r\n\r\n /**\r\n * Gets the right trigger value\r\n */\r\n public get rightTrigger(): number {\r\n return this._rightTrigger;\r\n }\r\n /**\r\n * Sets the right trigger value\r\n */\r\n public set rightTrigger(newValue: number) {\r\n if (this._onrighttriggerchanged && this._rightTrigger !== newValue) {\r\n this._onrighttriggerchanged(newValue);\r\n }\r\n this._rightTrigger = newValue;\r\n }\r\n\r\n /**\r\n * Defines the callback to call when a button is pressed\r\n * @param callback defines the callback to use\r\n */\r\n public onbuttondown(callback: (buttonPressed: DualShockButton) => void) {\r\n this._onbuttondown = callback;\r\n }\r\n\r\n /**\r\n * Defines the callback to call when a button is released\r\n * @param callback defines the callback to use\r\n */\r\n public onbuttonup(callback: (buttonReleased: DualShockButton) => void) {\r\n this._onbuttonup = callback;\r\n }\r\n\r\n /**\r\n * Defines the callback to call when a pad is pressed\r\n * @param callback defines the callback to use\r\n */\r\n public ondpaddown(callback: (dPadPressed: DualShockDpad) => void) {\r\n this._ondpaddown = callback;\r\n }\r\n\r\n /**\r\n * Defines the callback to call when a pad is released\r\n * @param callback defines the callback to use\r\n */\r\n public ondpadup(callback: (dPadReleased: DualShockDpad) => void) {\r\n this._ondpadup = callback;\r\n }\r\n\r\n private _setButtonValue(newValue: number, currentValue: number, buttonType: DualShockButton): number {\r\n if (newValue !== currentValue) {\r\n if (newValue === 1) {\r\n if (this._onbuttondown) {\r\n this._onbuttondown(buttonType);\r\n }\r\n\r\n this.onButtonDownObservable.notifyObservers(buttonType);\r\n }\r\n if (newValue === 0) {\r\n if (this._onbuttonup) {\r\n this._onbuttonup(buttonType);\r\n }\r\n\r\n this.onButtonUpObservable.notifyObservers(buttonType);\r\n }\r\n }\r\n return newValue;\r\n }\r\n\r\n private _setDPadValue(newValue: number, currentValue: number, buttonType: DualShockDpad): number {\r\n if (newValue !== currentValue) {\r\n if (newValue === 1) {\r\n if (this._ondpaddown) {\r\n this._ondpaddown(buttonType);\r\n }\r\n\r\n this.onPadDownObservable.notifyObservers(buttonType);\r\n }\r\n if (newValue === 0) {\r\n if (this._ondpadup) {\r\n this._ondpadup(buttonType);\r\n }\r\n\r\n this.onPadUpObservable.notifyObservers(buttonType);\r\n }\r\n }\r\n return newValue;\r\n }\r\n\r\n /**\r\n * Gets the value of the `Cross` button\r\n */\r\n public get buttonCross(): number {\r\n return this._buttonCross;\r\n }\r\n /**\r\n * Sets the value of the `Cross` button\r\n */\r\n public set buttonCross(value) {\r\n this._buttonCross = this._setButtonValue(value, this._buttonCross, DualShockButton.Cross);\r\n }\r\n\r\n /**\r\n * Gets the value of the `Circle` button\r\n */\r\n public get buttonCircle(): number {\r\n return this._buttonCircle;\r\n }\r\n /**\r\n * Sets the value of the `Circle` button\r\n */\r\n public set buttonCircle(value) {\r\n this._buttonCircle = this._setButtonValue(value, this._buttonCircle, DualShockButton.Circle);\r\n }\r\n\r\n /**\r\n * Gets the value of the `Square` button\r\n */\r\n public get buttonSquare(): number {\r\n return this._buttonSquare;\r\n }\r\n /**\r\n * Sets the value of the `Square` button\r\n */\r\n public set buttonSquare(value) {\r\n this._buttonSquare = this._setButtonValue(value, this._buttonSquare, DualShockButton.Square);\r\n }\r\n\r\n /**\r\n * Gets the value of the `Triangle` button\r\n */\r\n public get buttonTriangle(): number {\r\n return this._buttonTriangle;\r\n }\r\n /**\r\n * Sets the value of the `Triangle` button\r\n */\r\n public set buttonTriangle(value) {\r\n this._buttonTriangle = this._setButtonValue(value, this._buttonTriangle, DualShockButton.Triangle);\r\n }\r\n\r\n /**\r\n * Gets the value of the `Options` button\r\n */\r\n public get buttonOptions(): number {\r\n return this._buttonOptions;\r\n }\r\n /**\r\n * Sets the value of the `Options` button\r\n */\r\n public set buttonOptions(value) {\r\n this._buttonOptions = this._setButtonValue(value, this._buttonOptions, DualShockButton.Options);\r\n }\r\n\r\n /**\r\n * Gets the value of the `Share` button\r\n */\r\n public get buttonShare(): number {\r\n return this._buttonShare;\r\n }\r\n /**\r\n * Sets the value of the `Share` button\r\n */\r\n public set buttonShare(value) {\r\n this._buttonShare = this._setButtonValue(value, this._buttonShare, DualShockButton.Share);\r\n }\r\n\r\n /**\r\n * Gets the value of the `L1` button\r\n */\r\n public get buttonL1(): number {\r\n return this._buttonL1;\r\n }\r\n /**\r\n * Sets the value of the `L1` button\r\n */\r\n public set buttonL1(value) {\r\n this._buttonL1 = this._setButtonValue(value, this._buttonL1, DualShockButton.L1);\r\n }\r\n\r\n /**\r\n * Gets the value of the `R1` button\r\n */\r\n public get buttonR1(): number {\r\n return this._buttonR1;\r\n }\r\n /**\r\n * Sets the value of the `R1` button\r\n */\r\n public set buttonR1(value) {\r\n this._buttonR1 = this._setButtonValue(value, this._buttonR1, DualShockButton.R1);\r\n }\r\n\r\n /**\r\n * Gets the value of the Left joystick\r\n */\r\n public get buttonLeftStick(): number {\r\n return this._buttonLeftStick;\r\n }\r\n /**\r\n * Sets the value of the Left joystick\r\n */\r\n public set buttonLeftStick(value) {\r\n this._buttonLeftStick = this._setButtonValue(value, this._buttonLeftStick, DualShockButton.LeftStick);\r\n }\r\n\r\n /**\r\n * Gets the value of the Right joystick\r\n */\r\n public get buttonRightStick(): number {\r\n return this._buttonRightStick;\r\n }\r\n /**\r\n * Sets the value of the Right joystick\r\n */\r\n public set buttonRightStick(value) {\r\n this._buttonRightStick = this._setButtonValue(value, this._buttonRightStick, DualShockButton.RightStick);\r\n }\r\n\r\n /**\r\n * Gets the value of D-pad up\r\n */\r\n public get dPadUp(): number {\r\n return this._dPadUp;\r\n }\r\n /**\r\n * Sets the value of D-pad up\r\n */\r\n public set dPadUp(value) {\r\n this._dPadUp = this._setDPadValue(value, this._dPadUp, DualShockDpad.Up);\r\n }\r\n\r\n /**\r\n * Gets the value of D-pad down\r\n */\r\n public get dPadDown(): number {\r\n return this._dPadDown;\r\n }\r\n /**\r\n * Sets the value of D-pad down\r\n */\r\n public set dPadDown(value) {\r\n this._dPadDown = this._setDPadValue(value, this._dPadDown, DualShockDpad.Down);\r\n }\r\n\r\n /**\r\n * Gets the value of D-pad left\r\n */\r\n public get dPadLeft(): number {\r\n return this._dPadLeft;\r\n }\r\n /**\r\n * Sets the value of D-pad left\r\n */\r\n public set dPadLeft(value) {\r\n this._dPadLeft = this._setDPadValue(value, this._dPadLeft, DualShockDpad.Left);\r\n }\r\n\r\n /**\r\n * Gets the value of D-pad right\r\n */\r\n public get dPadRight(): number {\r\n return this._dPadRight;\r\n }\r\n /**\r\n * Sets the value of D-pad right\r\n */\r\n public set dPadRight(value) {\r\n this._dPadRight = this._setDPadValue(value, this._dPadRight, DualShockDpad.Right);\r\n }\r\n\r\n /**\r\n * Force the gamepad to synchronize with device values\r\n */\r\n public update() {\r\n super.update();\r\n this.buttonCross = this.browserGamepad.buttons[0].value;\r\n this.buttonCircle = this.browserGamepad.buttons[1].value;\r\n this.buttonSquare = this.browserGamepad.buttons[2].value;\r\n this.buttonTriangle = this.browserGamepad.buttons[3].value;\r\n this.buttonL1 = this.browserGamepad.buttons[4].value;\r\n this.buttonR1 = this.browserGamepad.buttons[5].value;\r\n this.leftTrigger = this.browserGamepad.buttons[6].value;\r\n this.rightTrigger = this.browserGamepad.buttons[7].value;\r\n this.buttonShare = this.browserGamepad.buttons[8].value;\r\n this.buttonOptions = this.browserGamepad.buttons[9].value;\r\n this.buttonLeftStick = this.browserGamepad.buttons[10].value;\r\n this.buttonRightStick = this.browserGamepad.buttons[11].value;\r\n this.dPadUp = this.browserGamepad.buttons[12].value;\r\n this.dPadDown = this.browserGamepad.buttons[13].value;\r\n this.dPadLeft = this.browserGamepad.buttons[14].value;\r\n this.dPadRight = this.browserGamepad.buttons[15].value;\r\n }\r\n\r\n /**\r\n * Disposes the gamepad\r\n */\r\n public dispose() {\r\n super.dispose();\r\n this.onButtonDownObservable.clear();\r\n this.onButtonUpObservable.clear();\r\n this.onPadDownObservable.clear();\r\n this.onPadUpObservable.clear();\r\n }\r\n}\r\n","import { Observable } from \"../Misc/observable\";\r\nimport { IsWindowObjectExist } from \"../Misc/domManagement\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { Scene } from \"../scene\";\r\nimport { PoseEnabledControllerHelper } from \"../Gamepads/Controllers/poseEnabledController\";\r\nimport { Xbox360Pad } from \"./xboxGamepad\";\r\nimport { Gamepad, GenericPad } from \"./gamepad\";\r\nimport { Engine } from \"../Engines/engine\";\r\nimport { DualShockPad } from \"./dualShockGamepad\";\r\nimport { Tools } from \"../Misc/tools\";\r\n/**\r\n * Manager for handling gamepads\r\n */\r\nexport class GamepadManager {\r\n private _babylonGamepads: Array = [];\r\n private _oneGamepadConnected: boolean = false;\r\n\r\n /** @internal */\r\n public _isMonitoring: boolean = false;\r\n private _gamepadEventSupported: boolean;\r\n private _gamepadSupport?: () => Array;\r\n\r\n /**\r\n * observable to be triggered when the gamepad controller has been connected\r\n */\r\n public onGamepadConnectedObservable: Observable;\r\n\r\n /**\r\n * observable to be triggered when the gamepad controller has been disconnected\r\n */\r\n public onGamepadDisconnectedObservable = new Observable();\r\n\r\n private _onGamepadConnectedEvent: Nullable<(evt: any) => void>;\r\n private _onGamepadDisconnectedEvent: Nullable<(evt: any) => void>;\r\n\r\n /**\r\n * Initializes the gamepad manager\r\n * @param _scene BabylonJS scene\r\n */\r\n constructor(private _scene?: Scene) {\r\n if (!IsWindowObjectExist()) {\r\n this._gamepadEventSupported = false;\r\n } else {\r\n this._gamepadEventSupported = \"GamepadEvent\" in window;\r\n this._gamepadSupport = navigator && navigator.getGamepads;\r\n }\r\n\r\n this.onGamepadConnectedObservable = new Observable((observer) => {\r\n // This will be used to raise the onGamepadConnected for all gamepads ALREADY connected\r\n for (const i in this._babylonGamepads) {\r\n const gamepad = this._babylonGamepads[i];\r\n if (gamepad && gamepad._isConnected) {\r\n this.onGamepadConnectedObservable.notifyObserver(observer, gamepad);\r\n }\r\n }\r\n });\r\n\r\n this._onGamepadConnectedEvent = (evt) => {\r\n const gamepad = evt.gamepad;\r\n\r\n if (gamepad.index in this._babylonGamepads) {\r\n if (this._babylonGamepads[gamepad.index].isConnected) {\r\n return;\r\n }\r\n }\r\n\r\n let newGamepad: Gamepad;\r\n\r\n if (this._babylonGamepads[gamepad.index]) {\r\n newGamepad = this._babylonGamepads[gamepad.index];\r\n newGamepad.browserGamepad = gamepad;\r\n newGamepad._isConnected = true;\r\n } else {\r\n newGamepad = this._addNewGamepad(gamepad);\r\n }\r\n this.onGamepadConnectedObservable.notifyObservers(newGamepad);\r\n this._startMonitoringGamepads();\r\n };\r\n\r\n this._onGamepadDisconnectedEvent = (evt) => {\r\n const gamepad = evt.gamepad;\r\n\r\n // Remove the gamepad from the list of gamepads to monitor.\r\n for (const i in this._babylonGamepads) {\r\n if (this._babylonGamepads[i].index === gamepad.index) {\r\n const disconnectedGamepad = this._babylonGamepads[i];\r\n disconnectedGamepad._isConnected = false;\r\n\r\n this.onGamepadDisconnectedObservable.notifyObservers(disconnectedGamepad);\r\n disconnectedGamepad.dispose && disconnectedGamepad.dispose();\r\n break;\r\n }\r\n }\r\n };\r\n\r\n if (this._gamepadSupport) {\r\n //first add already-connected gamepads\r\n this._updateGamepadObjects();\r\n if (this._babylonGamepads.length) {\r\n this._startMonitoringGamepads();\r\n }\r\n // Checking if the gamepad connected event is supported (like in Firefox)\r\n if (this._gamepadEventSupported) {\r\n const hostWindow = this._scene ? this._scene.getEngine().getHostWindow() : window;\r\n\r\n if (hostWindow) {\r\n hostWindow.addEventListener(\"gamepadconnected\", this._onGamepadConnectedEvent, false);\r\n hostWindow.addEventListener(\"gamepaddisconnected\", this._onGamepadDisconnectedEvent, false);\r\n }\r\n } else {\r\n this._startMonitoringGamepads();\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * The gamepads in the game pad manager\r\n */\r\n public get gamepads(): Gamepad[] {\r\n return this._babylonGamepads;\r\n }\r\n\r\n /**\r\n * Get the gamepad controllers based on type\r\n * @param type The type of gamepad controller\r\n * @returns Nullable gamepad\r\n */\r\n public getGamepadByType(type: number = Gamepad.XBOX): Nullable {\r\n for (const gamepad of this._babylonGamepads) {\r\n if (gamepad && gamepad.type === type) {\r\n return gamepad;\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Disposes the gamepad manager\r\n */\r\n public dispose() {\r\n if (this._gamepadEventSupported) {\r\n if (this._onGamepadConnectedEvent) {\r\n window.removeEventListener(\"gamepadconnected\", this._onGamepadConnectedEvent);\r\n }\r\n\r\n if (this._onGamepadDisconnectedEvent) {\r\n window.removeEventListener(\"gamepaddisconnected\", this._onGamepadDisconnectedEvent);\r\n }\r\n this._onGamepadConnectedEvent = null;\r\n this._onGamepadDisconnectedEvent = null;\r\n }\r\n\r\n this._babylonGamepads.forEach((gamepad) => {\r\n gamepad.dispose();\r\n });\r\n\r\n this.onGamepadConnectedObservable.clear();\r\n this.onGamepadDisconnectedObservable.clear();\r\n\r\n this._oneGamepadConnected = false;\r\n this._stopMonitoringGamepads();\r\n this._babylonGamepads = [];\r\n }\r\n\r\n private _addNewGamepad(gamepad: any): Gamepad {\r\n if (!this._oneGamepadConnected) {\r\n this._oneGamepadConnected = true;\r\n }\r\n\r\n let newGamepad;\r\n const dualShock: boolean = (gamepad.id).search(\"054c\") !== -1 && (gamepad.id).search(\"0ce6\") === -1;\r\n const xboxOne: boolean = (gamepad.id).search(\"Xbox One\") !== -1;\r\n if (\r\n xboxOne ||\r\n (gamepad.id).search(\"Xbox 360\") !== -1 ||\r\n (gamepad.id).search(\"xinput\") !== -1 ||\r\n ((gamepad.id).search(\"045e\") !== -1 && (gamepad.id).search(\"Surface Dock\") === -1)\r\n ) {\r\n // make sure the Surface Dock Extender is not detected as an xbox controller\r\n newGamepad = new Xbox360Pad(gamepad.id, gamepad.index, gamepad, xboxOne);\r\n } else if (dualShock) {\r\n newGamepad = new DualShockPad(gamepad.id, gamepad.index, gamepad);\r\n }\r\n // if pose is supported, use the (WebVR) pose enabled controller\r\n else if (gamepad.pose) {\r\n newGamepad = PoseEnabledControllerHelper.InitiateController(gamepad);\r\n } else {\r\n newGamepad = new GenericPad(gamepad.id, gamepad.index, gamepad);\r\n }\r\n this._babylonGamepads[newGamepad.index] = newGamepad;\r\n return newGamepad;\r\n }\r\n\r\n private _startMonitoringGamepads() {\r\n if (!this._isMonitoring) {\r\n this._isMonitoring = true;\r\n //back-comp\r\n this._checkGamepadsStatus();\r\n }\r\n }\r\n\r\n private _stopMonitoringGamepads() {\r\n this._isMonitoring = false;\r\n }\r\n\r\n private _loggedErrors: number[];\r\n\r\n /** @internal */\r\n public _checkGamepadsStatus() {\r\n // Hack to be compatible Chrome\r\n this._updateGamepadObjects();\r\n\r\n for (const i in this._babylonGamepads) {\r\n const gamepad = this._babylonGamepads[i];\r\n if (!gamepad || !gamepad.isConnected) {\r\n continue;\r\n }\r\n try {\r\n gamepad.update();\r\n } catch {\r\n if (this._loggedErrors.indexOf(gamepad.index) === -1) {\r\n Tools.Warn(`Error updating gamepad ${gamepad.id}`);\r\n this._loggedErrors.push(gamepad.index);\r\n }\r\n }\r\n }\r\n\r\n if (this._isMonitoring) {\r\n Engine.QueueNewFrame(() => {\r\n this._checkGamepadsStatus();\r\n });\r\n }\r\n }\r\n\r\n // This function is called only on Chrome, which does not properly support\r\n // connection/disconnection events and forces you to recopy again the gamepad object\r\n private _updateGamepadObjects() {\r\n const gamepads = navigator.getGamepads ? navigator.getGamepads() : [];\r\n for (let i = 0; i < gamepads.length; i++) {\r\n const gamepad = gamepads[i];\r\n if (gamepad) {\r\n if (!this._babylonGamepads[gamepad.index]) {\r\n const newGamepad = this._addNewGamepad(gamepad);\r\n this.onGamepadConnectedObservable.notifyObservers(newGamepad);\r\n } else {\r\n // Forced to copy again this object for Chrome for unknown reason\r\n this._babylonGamepads[i].browserGamepad = gamepad;\r\n\r\n if (!this._babylonGamepads[i].isConnected) {\r\n this._babylonGamepads[i]._isConnected = true;\r\n this.onGamepadConnectedObservable.notifyObservers(this._babylonGamepads[i]);\r\n }\r\n }\r\n }\r\n }\r\n }\r\n}\r\n","import { serialize } from \"../../Misc/decorators\";\r\nimport type { Observer } from \"../../Misc/observable\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { ICameraInput } from \"../../Cameras/cameraInputsManager\";\r\nimport { CameraInputTypes } from \"../../Cameras/cameraInputsManager\";\r\nimport type { FreeCamera } from \"../../Cameras/freeCamera\";\r\nimport { Matrix, Vector3, Vector2 } from \"../../Maths/math.vector\";\r\nimport { Gamepad } from \"../../Gamepads/gamepad\";\r\n\r\n/**\r\n * Manage the gamepad inputs to control a free camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\r\n */\r\nexport class FreeCameraGamepadInput implements ICameraInput {\r\n /**\r\n * Define the camera the input is attached to.\r\n */\r\n public camera: FreeCamera;\r\n\r\n /**\r\n * Define the Gamepad controlling the input\r\n */\r\n public gamepad: Nullable;\r\n\r\n /**\r\n * Defines the gamepad rotation sensibility.\r\n * This is the threshold from when rotation starts to be accounted for to prevent jittering.\r\n */\r\n @serialize()\r\n public gamepadAngularSensibility = 200;\r\n\r\n /**\r\n * Defines the gamepad move sensibility.\r\n * This is the threshold from when moving starts to be accounted for for to prevent jittering.\r\n */\r\n @serialize()\r\n public gamepadMoveSensibility = 40;\r\n\r\n /**\r\n * Defines the minimum value at which any analog stick input is ignored.\r\n * Note: This value should only be a value between 0 and 1.\r\n */\r\n public deadzoneDelta = 0.1;\r\n\r\n private _yAxisScale = 1.0;\r\n\r\n /**\r\n * Gets or sets a boolean indicating that Yaxis (for right stick) should be inverted\r\n */\r\n public get invertYAxis() {\r\n return this._yAxisScale !== 1.0;\r\n }\r\n\r\n public set invertYAxis(value: boolean) {\r\n this._yAxisScale = value ? -1.0 : 1.0;\r\n }\r\n\r\n // private members\r\n private _onGamepadConnectedObserver: Nullable>;\r\n private _onGamepadDisconnectedObserver: Nullable>;\r\n private _cameraTransform: Matrix = Matrix.Identity();\r\n private _deltaTransform: Vector3 = Vector3.Zero();\r\n private _vector3: Vector3 = Vector3.Zero();\r\n private _vector2: Vector2 = Vector2.Zero();\r\n\r\n /**\r\n * Attach the input controls to a specific dom element to get the input from.\r\n */\r\n public attachControl(): void {\r\n const manager = this.camera.getScene().gamepadManager;\r\n this._onGamepadConnectedObserver = manager.onGamepadConnectedObservable.add((gamepad) => {\r\n if (gamepad.type !== Gamepad.POSE_ENABLED) {\r\n // prioritize XBOX gamepads.\r\n if (!this.gamepad || gamepad.type === Gamepad.XBOX) {\r\n this.gamepad = gamepad;\r\n }\r\n }\r\n });\r\n\r\n this._onGamepadDisconnectedObserver = manager.onGamepadDisconnectedObservable.add((gamepad) => {\r\n if (this.gamepad === gamepad) {\r\n this.gamepad = null;\r\n }\r\n });\r\n\r\n // check if there are already other controllers connected\r\n this.gamepad = manager.getGamepadByType(Gamepad.XBOX);\r\n // if no xbox controller was found, but there are gamepad controllers, take the first one\r\n if (!this.gamepad && manager.gamepads.length) {\r\n this.gamepad = manager.gamepads[0];\r\n }\r\n }\r\n\r\n /**\r\n * Detach the current controls from the specified dom element.\r\n */\r\n public detachControl(): void {\r\n this.camera.getScene().gamepadManager.onGamepadConnectedObservable.remove(this._onGamepadConnectedObserver);\r\n this.camera.getScene().gamepadManager.onGamepadDisconnectedObservable.remove(this._onGamepadDisconnectedObserver);\r\n this.gamepad = null;\r\n }\r\n\r\n /**\r\n * Update the current camera state depending on the inputs that have been used this frame.\r\n * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop.\r\n */\r\n public checkInputs(): void {\r\n if (this.gamepad && this.gamepad.leftStick) {\r\n const camera = this.camera;\r\n const lsValues = this.gamepad.leftStick;\r\n if (this.gamepadMoveSensibility !== 0) {\r\n lsValues.x = Math.abs(lsValues.x) > this.deadzoneDelta ? lsValues.x / this.gamepadMoveSensibility : 0;\r\n lsValues.y = Math.abs(lsValues.y) > this.deadzoneDelta ? lsValues.y / this.gamepadMoveSensibility : 0;\r\n }\r\n\r\n let rsValues = this.gamepad.rightStick;\r\n if (rsValues && this.gamepadAngularSensibility !== 0) {\r\n rsValues.x = Math.abs(rsValues.x) > this.deadzoneDelta ? rsValues.x / this.gamepadAngularSensibility : 0;\r\n rsValues.y = (Math.abs(rsValues.y) > this.deadzoneDelta ? rsValues.y / this.gamepadAngularSensibility : 0) * this._yAxisScale;\r\n } else {\r\n rsValues = { x: 0, y: 0 };\r\n }\r\n\r\n if (!camera.rotationQuaternion) {\r\n Matrix.RotationYawPitchRollToRef(camera.rotation.y, camera.rotation.x, 0, this._cameraTransform);\r\n } else {\r\n camera.rotationQuaternion.toRotationMatrix(this._cameraTransform);\r\n }\r\n\r\n const speed = camera._computeLocalCameraSpeed() * 50.0;\r\n this._vector3.copyFromFloats(lsValues.x * speed, 0, -lsValues.y * speed);\r\n\r\n Vector3.TransformCoordinatesToRef(this._vector3, this._cameraTransform, this._deltaTransform);\r\n camera.cameraDirection.addInPlace(this._deltaTransform);\r\n this._vector2.copyFromFloats(rsValues.y, rsValues.x);\r\n camera.cameraRotation.addInPlace(this._vector2);\r\n }\r\n }\r\n\r\n /**\r\n * Gets the class name of the current input.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"FreeCameraGamepadInput\";\r\n }\r\n\r\n /**\r\n * Get the friendly name associated with the input class.\r\n * @returns the input friendly name\r\n */\r\n public getSimpleName(): string {\r\n return \"gamepad\";\r\n }\r\n}\r\n\r\n(CameraInputTypes)[\"FreeCameraGamepadInput\"] = FreeCameraGamepadInput;\r\n","import type { Nullable } from \"../../types\";\r\nimport { serialize } from \"../../Misc/decorators\";\r\nimport type { Observer } from \"../../Misc/observable\";\r\nimport type { ArcRotateCamera } from \"../../Cameras/arcRotateCamera\";\r\nimport type { ICameraInput } from \"../../Cameras/cameraInputsManager\";\r\nimport { CameraInputTypes } from \"../../Cameras/cameraInputsManager\";\r\nimport { Gamepad } from \"../../Gamepads/gamepad\";\r\n/**\r\n * Manage the gamepad inputs to control an arc rotate camera.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/customizingCameraInputs\r\n */\r\nexport class ArcRotateCameraGamepadInput implements ICameraInput {\r\n /**\r\n * Defines the camera the input is attached to.\r\n */\r\n public camera: ArcRotateCamera;\r\n\r\n /**\r\n * Defines the gamepad the input is gathering event from.\r\n */\r\n public gamepad: Nullable;\r\n\r\n /**\r\n * Defines the gamepad rotation sensibility.\r\n * This is the threshold from when rotation starts to be accounted for to prevent jittering.\r\n */\r\n @serialize()\r\n public gamepadRotationSensibility = 80;\r\n\r\n /**\r\n * Defines the gamepad move sensibility.\r\n * This is the threshold from when moving starts to be accounted for for to prevent jittering.\r\n */\r\n @serialize()\r\n public gamepadMoveSensibility = 40;\r\n\r\n private _yAxisScale = 1.0;\r\n\r\n /**\r\n * Gets or sets a boolean indicating that Yaxis (for right stick) should be inverted\r\n */\r\n public get invertYAxis() {\r\n return this._yAxisScale !== 1.0;\r\n }\r\n\r\n public set invertYAxis(value: boolean) {\r\n this._yAxisScale = value ? -1.0 : 1.0;\r\n }\r\n\r\n private _onGamepadConnectedObserver: Nullable>;\r\n private _onGamepadDisconnectedObserver: Nullable>;\r\n\r\n /**\r\n * Attach the input controls to a specific dom element to get the input from.\r\n */\r\n public attachControl(): void {\r\n const manager = this.camera.getScene().gamepadManager;\r\n this._onGamepadConnectedObserver = manager.onGamepadConnectedObservable.add((gamepad) => {\r\n if (gamepad.type !== Gamepad.POSE_ENABLED) {\r\n // prioritize XBOX gamepads.\r\n if (!this.gamepad || gamepad.type === Gamepad.XBOX) {\r\n this.gamepad = gamepad;\r\n }\r\n }\r\n });\r\n\r\n this._onGamepadDisconnectedObserver = manager.onGamepadDisconnectedObservable.add((gamepad) => {\r\n if (this.gamepad === gamepad) {\r\n this.gamepad = null;\r\n }\r\n });\r\n\r\n this.gamepad = manager.getGamepadByType(Gamepad.XBOX);\r\n }\r\n\r\n /**\r\n * Detach the current controls from the specified dom element.\r\n */\r\n public detachControl(): void {\r\n this.camera.getScene().gamepadManager.onGamepadConnectedObservable.remove(this._onGamepadConnectedObserver);\r\n this.camera.getScene().gamepadManager.onGamepadDisconnectedObservable.remove(this._onGamepadDisconnectedObserver);\r\n this.gamepad = null;\r\n }\r\n\r\n /**\r\n * Update the current camera state depending on the inputs that have been used this frame.\r\n * This is a dynamically created lambda to avoid the performance penalty of looping for inputs in the render loop.\r\n */\r\n public checkInputs(): void {\r\n if (this.gamepad) {\r\n const camera = this.camera;\r\n const rsValues = this.gamepad.rightStick;\r\n\r\n if (rsValues) {\r\n if (rsValues.x != 0) {\r\n const normalizedRX = rsValues.x / this.gamepadRotationSensibility;\r\n if (normalizedRX != 0 && Math.abs(normalizedRX) > 0.005) {\r\n camera.inertialAlphaOffset += normalizedRX;\r\n }\r\n }\r\n\r\n if (rsValues.y != 0) {\r\n const normalizedRY = (rsValues.y / this.gamepadRotationSensibility) * this._yAxisScale;\r\n if (normalizedRY != 0 && Math.abs(normalizedRY) > 0.005) {\r\n camera.inertialBetaOffset += normalizedRY;\r\n }\r\n }\r\n }\r\n\r\n const lsValues = this.gamepad.leftStick;\r\n if (lsValues && lsValues.y != 0) {\r\n const normalizedLY = lsValues.y / this.gamepadMoveSensibility;\r\n if (normalizedLY != 0 && Math.abs(normalizedLY) > 0.005) {\r\n this.camera.inertialRadiusOffset -= normalizedLY;\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Gets the class name of the current intput.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"ArcRotateCameraGamepadInput\";\r\n }\r\n\r\n /**\r\n * Get the friendly name associated with the input class.\r\n * @returns the input friendly name\r\n */\r\n public getSimpleName(): string {\r\n return \"gamepad\";\r\n }\r\n}\r\n\r\n(CameraInputTypes)[\"ArcRotateCameraGamepadInput\"] = ArcRotateCameraGamepadInput;\r\n","import type { Nullable } from \"../types\";\r\nimport { Scene } from \"../scene\";\r\nimport type { ISceneComponent } from \"../sceneComponent\";\r\nimport { SceneComponentConstants } from \"../sceneComponent\";\r\nimport { GamepadManager } from \"./gamepadManager\";\r\n\r\nimport { FreeCameraInputsManager } from \"../Cameras/freeCameraInputsManager\";\r\nimport { FreeCameraGamepadInput } from \"../Cameras/Inputs/freeCameraGamepadInput\";\r\nimport { ArcRotateCameraInputsManager } from \"../Cameras/arcRotateCameraInputsManager\";\r\nimport { ArcRotateCameraGamepadInput } from \"../Cameras/Inputs/arcRotateCameraGamepadInput\";\r\n\r\ndeclare module \"../scene\" {\r\n export interface Scene {\r\n /** @internal */\r\n _gamepadManager: Nullable;\r\n\r\n /**\r\n * Gets the gamepad manager associated with the scene\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/input/gamepads\r\n */\r\n gamepadManager: GamepadManager;\r\n }\r\n}\r\n\r\nObject.defineProperty(Scene.prototype, \"gamepadManager\", {\r\n get: function (this: Scene) {\r\n if (!this._gamepadManager) {\r\n this._gamepadManager = new GamepadManager(this);\r\n let component = this._getComponent(SceneComponentConstants.NAME_GAMEPAD) as GamepadSystemSceneComponent;\r\n if (!component) {\r\n component = new GamepadSystemSceneComponent(this);\r\n this._addComponent(component);\r\n }\r\n }\r\n\r\n return this._gamepadManager;\r\n },\r\n enumerable: true,\r\n configurable: true,\r\n});\r\n\r\ndeclare module \"../Cameras/freeCameraInputsManager\" {\r\n /**\r\n * Interface representing a free camera inputs manager\r\n */\r\n export interface FreeCameraInputsManager {\r\n /**\r\n * Adds gamepad input support to the FreeCameraInputsManager.\r\n * @returns the FreeCameraInputsManager\r\n */\r\n addGamepad(): FreeCameraInputsManager;\r\n }\r\n}\r\n\r\n/**\r\n * Adds a gamepad to the free camera inputs manager\r\n */\r\nFreeCameraInputsManager.prototype.addGamepad = function (): FreeCameraInputsManager {\r\n this.add(new FreeCameraGamepadInput());\r\n return this;\r\n};\r\n\r\ndeclare module \"../Cameras/arcRotateCameraInputsManager\" {\r\n /**\r\n * Interface representing an arc rotate camera inputs manager\r\n */\r\n export interface ArcRotateCameraInputsManager {\r\n /**\r\n * Adds gamepad input support to the ArcRotateCamera InputManager.\r\n * @returns the camera inputs manager\r\n */\r\n addGamepad(): ArcRotateCameraInputsManager;\r\n }\r\n}\r\n\r\n/**\r\n * Adds a gamepad to the arc rotate camera inputs manager\r\n */\r\nArcRotateCameraInputsManager.prototype.addGamepad = function (): ArcRotateCameraInputsManager {\r\n this.add(new ArcRotateCameraGamepadInput());\r\n return this;\r\n};\r\n\r\n/**\r\n * Defines the gamepad scene component responsible to manage gamepads in a given scene\r\n */\r\nexport class GamepadSystemSceneComponent implements ISceneComponent {\r\n /**\r\n * The component name helpfull to identify the component in the list of scene components.\r\n */\r\n public readonly name = SceneComponentConstants.NAME_GAMEPAD;\r\n\r\n /**\r\n * The scene the component belongs to.\r\n */\r\n public scene: Scene;\r\n\r\n /**\r\n * Creates a new instance of the component for the given scene\r\n * @param scene Defines the scene to register the component in\r\n */\r\n constructor(scene: Scene) {\r\n this.scene = scene;\r\n }\r\n\r\n /**\r\n * Registers the component in a given scene\r\n */\r\n public register(): void {\r\n this.scene._beforeCameraUpdateStage.registerStep(SceneComponentConstants.STEP_BEFORECAMERAUPDATE_GAMEPAD, this, this._beforeCameraUpdate);\r\n }\r\n\r\n /**\r\n * Rebuilds the elements related to this component in case of\r\n * context lost for instance.\r\n */\r\n public rebuild(): void {\r\n // Nothing to do for gamepads\r\n }\r\n\r\n /**\r\n * Disposes the component and the associated resources\r\n */\r\n public dispose(): void {\r\n const gamepadManager = this.scene._gamepadManager;\r\n if (gamepadManager) {\r\n gamepadManager.dispose();\r\n this.scene._gamepadManager = null;\r\n }\r\n }\r\n\r\n private _beforeCameraUpdate(): void {\r\n const gamepadManager = this.scene._gamepadManager;\r\n\r\n if (gamepadManager && gamepadManager._isMonitoring) {\r\n gamepadManager._checkGamepadsStatus();\r\n }\r\n }\r\n}\r\n","import type { DeepImmutable, Nullable } from \"../types\";\r\nimport { Quaternion, Vector3, Vector2, Matrix } from \"../Maths/math.vector\";\r\nimport { Color3 } from \"../Maths/math.color\";\r\nimport type { _IAnimationState } from \"./animation\";\r\nimport { Animation } from \"./animation\";\r\nimport type { AnimationEvent } from \"./animationEvent\";\r\n\r\nimport type { Animatable } from \"./animatable\";\r\n\r\nimport type { Scene } from \"../scene\";\r\nimport type { IAnimationKey } from \"./animationKey\";\r\nimport { Size } from \"../Maths/math.size\";\r\n\r\n// Static values to help the garbage collector\r\n\r\n// Quaternion\r\nconst _staticOffsetValueQuaternion: DeepImmutable = Object.freeze(new Quaternion(0, 0, 0, 0));\r\n\r\n// Vector3\r\nconst _staticOffsetValueVector3: DeepImmutable = Object.freeze(Vector3.Zero());\r\n\r\n// Vector2\r\nconst _staticOffsetValueVector2: DeepImmutable = Object.freeze(Vector2.Zero());\r\n\r\n// Size\r\nconst _staticOffsetValueSize: DeepImmutable = Object.freeze(Size.Zero());\r\n\r\n// Color3\r\nconst _staticOffsetValueColor3: DeepImmutable = Object.freeze(Color3.Black());\r\n\r\n/**\r\n * Defines a runtime animation\r\n */\r\nexport class RuntimeAnimation {\r\n private _events = new Array();\r\n\r\n /**\r\n * The current frame of the runtime animation\r\n */\r\n private _currentFrame: number = 0;\r\n\r\n /**\r\n * The animation used by the runtime animation\r\n */\r\n private _animation: Animation;\r\n\r\n /**\r\n * The target of the runtime animation\r\n */\r\n private _target: any;\r\n\r\n /**\r\n * The initiating animatable\r\n */\r\n private _host: Animatable;\r\n\r\n /**\r\n * The original value of the runtime animation\r\n */\r\n private _originalValue = new Array();\r\n\r\n /**\r\n * The original blend value of the runtime animation\r\n */\r\n private _originalBlendValue: Nullable = null;\r\n\r\n /**\r\n * The offsets cache of the runtime animation\r\n */\r\n private _offsetsCache: { [key: string]: any } = {};\r\n\r\n /**\r\n * The high limits cache of the runtime animation\r\n */\r\n private _highLimitsCache: { [key: string]: any } = {};\r\n\r\n /**\r\n * Specifies if the runtime animation has been stopped\r\n */\r\n private _stopped = false;\r\n\r\n /**\r\n * The blending factor of the runtime animation\r\n */\r\n private _blendingFactor = 0;\r\n\r\n /**\r\n * The BabylonJS scene\r\n */\r\n private _scene: Scene;\r\n\r\n /**\r\n * The current value of the runtime animation\r\n */\r\n private _currentValue: Nullable = null;\r\n\r\n /** @internal */\r\n public _animationState: _IAnimationState;\r\n\r\n /**\r\n * The active target of the runtime animation\r\n */\r\n private _activeTargets: any[];\r\n private _currentActiveTarget: Nullable = null;\r\n private _directTarget: Nullable = null;\r\n\r\n /**\r\n * The target path of the runtime animation\r\n */\r\n private _targetPath: string = \"\";\r\n\r\n /**\r\n * The weight of the runtime animation\r\n */\r\n private _weight = 1.0;\r\n\r\n /**\r\n * The absolute frame offset of the runtime animation\r\n */\r\n private _absoluteFrameOffset = 0;\r\n\r\n /**\r\n * The previous elapsed time (since start of animation) of the runtime animation\r\n */\r\n private _previousElapsedTime: number = 0;\r\n\r\n /**\r\n * The previous absolute frame of the runtime animation (meaning, without taking into account the from/to values, only the elapsed time and the fps)\r\n */\r\n private _previousAbsoluteFrame: number = 0;\r\n\r\n private _enableBlending: boolean;\r\n\r\n private _keys: IAnimationKey[];\r\n private _minFrame: number;\r\n private _maxFrame: number;\r\n private _minValue: any;\r\n private _maxValue: any;\r\n private _targetIsArray = false;\r\n\r\n /**\r\n * Gets the current frame of the runtime animation\r\n */\r\n public get currentFrame(): number {\r\n return this._currentFrame;\r\n }\r\n\r\n /**\r\n * Gets the weight of the runtime animation\r\n */\r\n public get weight(): number {\r\n return this._weight;\r\n }\r\n\r\n /**\r\n * Gets the current value of the runtime animation\r\n */\r\n public get currentValue(): any {\r\n return this._currentValue;\r\n }\r\n\r\n /**\r\n * Gets or sets the target path of the runtime animation\r\n */\r\n public get targetPath(): string {\r\n return this._targetPath;\r\n }\r\n\r\n /**\r\n * Gets the actual target of the runtime animation\r\n */\r\n public get target(): any {\r\n return this._currentActiveTarget;\r\n }\r\n\r\n /**\r\n * Gets the additive state of the runtime animation\r\n */\r\n public get isAdditive(): boolean {\r\n return this._host && this._host.isAdditive;\r\n }\r\n\r\n /** @internal */\r\n public _onLoop: () => void;\r\n\r\n /**\r\n * Create a new RuntimeAnimation object\r\n * @param target defines the target of the animation\r\n * @param animation defines the source animation object\r\n * @param scene defines the hosting scene\r\n * @param host defines the initiating Animatable\r\n */\r\n public constructor(target: any, animation: Animation, scene: Scene, host: Animatable) {\r\n this._animation = animation;\r\n this._target = target;\r\n this._scene = scene;\r\n this._host = host;\r\n this._activeTargets = [];\r\n\r\n animation._runtimeAnimations.push(this);\r\n\r\n // State\r\n this._animationState = {\r\n key: 0,\r\n repeatCount: 0,\r\n loopMode: this._getCorrectLoopMode(),\r\n };\r\n\r\n if (this._animation.dataType === Animation.ANIMATIONTYPE_MATRIX) {\r\n this._animationState.workValue = Matrix.Zero();\r\n }\r\n\r\n // Limits\r\n this._keys = this._animation.getKeys();\r\n this._minFrame = this._keys[0].frame;\r\n this._maxFrame = this._keys[this._keys.length - 1].frame;\r\n this._minValue = this._keys[0].value;\r\n this._maxValue = this._keys[this._keys.length - 1].value;\r\n\r\n // Add a start key at frame 0 if missing\r\n if (this._minFrame !== 0) {\r\n const newKey = { frame: 0, value: this._minValue };\r\n this._keys.splice(0, 0, newKey);\r\n }\r\n\r\n // Check data\r\n if (this._target instanceof Array) {\r\n let index = 0;\r\n for (const target of this._target) {\r\n this._preparePath(target, index);\r\n this._getOriginalValues(index);\r\n index++;\r\n }\r\n this._targetIsArray = true;\r\n } else {\r\n this._preparePath(this._target);\r\n this._getOriginalValues();\r\n this._targetIsArray = false;\r\n this._directTarget = this._activeTargets[0];\r\n }\r\n\r\n // Cloning events locally\r\n const events = animation.getEvents();\r\n if (events && events.length > 0) {\r\n events.forEach((e) => {\r\n this._events.push(e._clone());\r\n });\r\n }\r\n\r\n this._enableBlending = target && target.animationPropertiesOverride ? target.animationPropertiesOverride.enableBlending : this._animation.enableBlending;\r\n }\r\n\r\n private _preparePath(target: any, targetIndex = 0) {\r\n const targetPropertyPath = this._animation.targetPropertyPath;\r\n\r\n if (targetPropertyPath.length > 1) {\r\n let property = target[targetPropertyPath[0]];\r\n\r\n for (let index = 1; index < targetPropertyPath.length - 1; index++) {\r\n property = property[targetPropertyPath[index]];\r\n }\r\n\r\n this._targetPath = targetPropertyPath[targetPropertyPath.length - 1];\r\n this._activeTargets[targetIndex] = property;\r\n } else {\r\n this._targetPath = targetPropertyPath[0];\r\n this._activeTargets[targetIndex] = target;\r\n }\r\n }\r\n\r\n /**\r\n * Gets the animation from the runtime animation\r\n */\r\n public get animation(): Animation {\r\n return this._animation;\r\n }\r\n\r\n /**\r\n * Resets the runtime animation to the beginning\r\n * @param restoreOriginal defines whether to restore the target property to the original value\r\n */\r\n public reset(restoreOriginal = false): void {\r\n if (restoreOriginal) {\r\n if (this._target instanceof Array) {\r\n let index = 0;\r\n for (const target of this._target) {\r\n if (this._originalValue[index] !== undefined) {\r\n this._setValue(target, this._activeTargets[index], this._originalValue[index], -1, index);\r\n }\r\n index++;\r\n }\r\n } else {\r\n if (this._originalValue[0] !== undefined) {\r\n this._setValue(this._target, this._directTarget, this._originalValue[0], -1, 0);\r\n }\r\n }\r\n }\r\n\r\n this._offsetsCache = {};\r\n this._highLimitsCache = {};\r\n this._currentFrame = 0;\r\n this._blendingFactor = 0;\r\n\r\n // Events\r\n for (let index = 0; index < this._events.length; index++) {\r\n this._events[index].isDone = false;\r\n }\r\n }\r\n\r\n /**\r\n * Specifies if the runtime animation is stopped\r\n * @returns Boolean specifying if the runtime animation is stopped\r\n */\r\n public isStopped(): boolean {\r\n return this._stopped;\r\n }\r\n\r\n /**\r\n * Disposes of the runtime animation\r\n */\r\n public dispose(): void {\r\n const index = this._animation.runtimeAnimations.indexOf(this);\r\n\r\n if (index > -1) {\r\n this._animation.runtimeAnimations.splice(index, 1);\r\n }\r\n }\r\n\r\n /**\r\n * Apply the interpolated value to the target\r\n * @param currentValue defines the value computed by the animation\r\n * @param weight defines the weight to apply to this value (Defaults to 1.0)\r\n */\r\n public setValue(currentValue: any, weight: number) {\r\n if (this._targetIsArray) {\r\n for (let index = 0; index < this._target.length; index++) {\r\n const target = this._target[index];\r\n this._setValue(target, this._activeTargets[index], currentValue, weight, index);\r\n }\r\n return;\r\n }\r\n this._setValue(this._target, this._directTarget, currentValue, weight, 0);\r\n }\r\n\r\n private _getOriginalValues(targetIndex = 0) {\r\n let originalValue: any;\r\n const target = this._activeTargets[targetIndex];\r\n\r\n if (target.getRestPose && this._targetPath === \"_matrix\") {\r\n // For bones\r\n originalValue = target.getRestPose();\r\n } else {\r\n originalValue = target[this._targetPath];\r\n }\r\n\r\n if (originalValue && originalValue.clone) {\r\n this._originalValue[targetIndex] = originalValue.clone();\r\n } else {\r\n this._originalValue[targetIndex] = originalValue;\r\n }\r\n }\r\n\r\n private _setValue(target: any, destination: any, currentValue: any, weight: number, targetIndex: number): void {\r\n // Set value\r\n this._currentActiveTarget = destination;\r\n\r\n this._weight = weight;\r\n\r\n if (this._enableBlending && this._blendingFactor <= 1.0) {\r\n if (!this._originalBlendValue) {\r\n const originalValue = destination[this._targetPath];\r\n\r\n if (originalValue.clone) {\r\n this._originalBlendValue = originalValue.clone();\r\n } else {\r\n this._originalBlendValue = originalValue;\r\n }\r\n }\r\n\r\n if (this._originalBlendValue.m) {\r\n // Matrix\r\n if (Animation.AllowMatrixDecomposeForInterpolation) {\r\n if (this._currentValue) {\r\n Matrix.DecomposeLerpToRef(this._originalBlendValue, currentValue, this._blendingFactor, this._currentValue);\r\n } else {\r\n this._currentValue = Matrix.DecomposeLerp(this._originalBlendValue, currentValue, this._blendingFactor);\r\n }\r\n } else {\r\n if (this._currentValue) {\r\n Matrix.LerpToRef(this._originalBlendValue, currentValue, this._blendingFactor, this._currentValue);\r\n } else {\r\n this._currentValue = Matrix.Lerp(this._originalBlendValue, currentValue, this._blendingFactor);\r\n }\r\n }\r\n } else {\r\n this._currentValue = Animation._UniversalLerp(this._originalBlendValue, currentValue, this._blendingFactor);\r\n }\r\n\r\n const blendingSpeed = target && target.animationPropertiesOverride ? target.animationPropertiesOverride.blendingSpeed : this._animation.blendingSpeed;\r\n this._blendingFactor += blendingSpeed;\r\n } else {\r\n if (!this._currentValue) {\r\n if (currentValue?.clone) {\r\n this._currentValue = currentValue.clone();\r\n } else {\r\n this._currentValue = currentValue;\r\n }\r\n } else if (this._currentValue.copyFrom) {\r\n this._currentValue.copyFrom(currentValue);\r\n } else {\r\n this._currentValue = currentValue;\r\n }\r\n }\r\n\r\n if (weight !== -1.0) {\r\n this._scene._registerTargetForLateAnimationBinding(this, this._originalValue[targetIndex]);\r\n } else {\r\n destination[this._targetPath] = this._currentValue;\r\n }\r\n\r\n if (target.markAsDirty) {\r\n target.markAsDirty(this._animation.targetProperty);\r\n }\r\n }\r\n\r\n /**\r\n * Gets the loop pmode of the runtime animation\r\n * @returns Loop Mode\r\n */\r\n private _getCorrectLoopMode(): number | undefined {\r\n if (this._target && this._target.animationPropertiesOverride) {\r\n return this._target.animationPropertiesOverride.loopMode;\r\n }\r\n\r\n return this._animation.loopMode;\r\n }\r\n\r\n /**\r\n * Move the current animation to a given frame\r\n * @param frame defines the frame to move to\r\n */\r\n public goToFrame(frame: number): void {\r\n const keys = this._animation.getKeys();\r\n\r\n if (frame < keys[0].frame) {\r\n frame = keys[0].frame;\r\n } else if (frame > keys[keys.length - 1].frame) {\r\n frame = keys[keys.length - 1].frame;\r\n }\r\n\r\n // Need to reset animation events\r\n const events = this._events;\r\n if (events.length) {\r\n for (let index = 0; index < events.length; index++) {\r\n if (!events[index].onlyOnce) {\r\n // reset events in the future\r\n events[index].isDone = events[index].frame < frame;\r\n }\r\n }\r\n }\r\n\r\n this._currentFrame = frame;\r\n const currentValue = this._animation._interpolate(frame, this._animationState);\r\n\r\n this.setValue(currentValue, -1);\r\n }\r\n\r\n /**\r\n * @internal Internal use only\r\n */\r\n public _prepareForSpeedRatioChange(newSpeedRatio: number): void {\r\n const newAbsoluteFrame = (this._previousElapsedTime * (this._animation.framePerSecond * newSpeedRatio)) / 1000.0;\r\n\r\n this._absoluteFrameOffset = this._previousAbsoluteFrame - newAbsoluteFrame;\r\n }\r\n\r\n /**\r\n * Execute the current animation\r\n * @param elapsedTimeSinceAnimationStart defines the elapsed time (in milliseconds) since the animation was started\r\n * @param from defines the lower frame of the animation range\r\n * @param to defines the upper frame of the animation range\r\n * @param loop defines if the current animation must loop\r\n * @param speedRatio defines the current speed ratio\r\n * @param weight defines the weight of the animation (default is -1 so no weight)\r\n * @returns a boolean indicating if the animation is running\r\n */\r\n public animate(elapsedTimeSinceAnimationStart: number, from: number, to: number, loop: boolean, speedRatio: number, weight = -1.0): boolean {\r\n const animation = this._animation;\r\n const targetPropertyPath = animation.targetPropertyPath;\r\n if (!targetPropertyPath || targetPropertyPath.length < 1) {\r\n this._stopped = true;\r\n return false;\r\n }\r\n\r\n let returnValue = true;\r\n\r\n // Check limits\r\n if (from < this._minFrame || from > this._maxFrame) {\r\n from = this._minFrame;\r\n }\r\n if (to < this._minFrame || to > this._maxFrame) {\r\n to = this._maxFrame;\r\n }\r\n\r\n const frameRange = to - from;\r\n let offsetValue: any;\r\n\r\n // Compute the frame according to the elapsed time and the fps of the animation (\"from\" and \"to\" are not factored in!)\r\n let absoluteFrame = (elapsedTimeSinceAnimationStart * (animation.framePerSecond * speedRatio)) / 1000.0 + this._absoluteFrameOffset;\r\n let highLimitValue = 0;\r\n\r\n // Apply the yoyo function if required\r\n if (loop && this._animationState.loopMode === Animation.ANIMATIONLOOPMODE_YOYO) {\r\n const position = (absoluteFrame - from) / frameRange;\r\n\r\n // Apply the yoyo curve\r\n const yoyoPosition = Math.abs(Math.sin(position * Math.PI));\r\n\r\n // Map the yoyo position back to the range\r\n absoluteFrame = yoyoPosition * frameRange + from;\r\n }\r\n\r\n this._previousElapsedTime = elapsedTimeSinceAnimationStart;\r\n this._previousAbsoluteFrame = absoluteFrame;\r\n\r\n if (!loop && to >= from && absoluteFrame >= frameRange) {\r\n // If we are out of range and not looping get back to caller\r\n returnValue = false;\r\n highLimitValue = animation._getKeyValue(this._maxValue);\r\n } else if (!loop && from >= to && absoluteFrame <= frameRange) {\r\n returnValue = false;\r\n highLimitValue = animation._getKeyValue(this._minValue);\r\n } else if (this._animationState.loopMode !== Animation.ANIMATIONLOOPMODE_CYCLE) {\r\n const keyOffset = to.toString() + from.toString();\r\n if (!this._offsetsCache[keyOffset]) {\r\n this._animationState.repeatCount = 0;\r\n this._animationState.loopMode = Animation.ANIMATIONLOOPMODE_CYCLE; // force a specific codepath in animation._interpolate()!\r\n const fromValue = animation._interpolate(from, this._animationState);\r\n const toValue = animation._interpolate(to, this._animationState);\r\n\r\n this._animationState.loopMode = this._getCorrectLoopMode();\r\n switch (animation.dataType) {\r\n // Float\r\n case Animation.ANIMATIONTYPE_FLOAT:\r\n this._offsetsCache[keyOffset] = toValue - fromValue;\r\n break;\r\n // Quaternion\r\n case Animation.ANIMATIONTYPE_QUATERNION:\r\n this._offsetsCache[keyOffset] = toValue.subtract(fromValue);\r\n break;\r\n // Vector3\r\n case Animation.ANIMATIONTYPE_VECTOR3:\r\n this._offsetsCache[keyOffset] = toValue.subtract(fromValue);\r\n break;\r\n // Vector2\r\n case Animation.ANIMATIONTYPE_VECTOR2:\r\n this._offsetsCache[keyOffset] = toValue.subtract(fromValue);\r\n break;\r\n // Size\r\n case Animation.ANIMATIONTYPE_SIZE:\r\n this._offsetsCache[keyOffset] = toValue.subtract(fromValue);\r\n break;\r\n // Color3\r\n case Animation.ANIMATIONTYPE_COLOR3:\r\n this._offsetsCache[keyOffset] = toValue.subtract(fromValue);\r\n break;\r\n default:\r\n break;\r\n }\r\n\r\n this._highLimitsCache[keyOffset] = toValue;\r\n }\r\n\r\n highLimitValue = this._highLimitsCache[keyOffset];\r\n offsetValue = this._offsetsCache[keyOffset];\r\n }\r\n\r\n if (offsetValue === undefined) {\r\n switch (animation.dataType) {\r\n // Float\r\n case Animation.ANIMATIONTYPE_FLOAT:\r\n offsetValue = 0;\r\n break;\r\n // Quaternion\r\n case Animation.ANIMATIONTYPE_QUATERNION:\r\n offsetValue = _staticOffsetValueQuaternion;\r\n break;\r\n // Vector3\r\n case Animation.ANIMATIONTYPE_VECTOR3:\r\n offsetValue = _staticOffsetValueVector3;\r\n break;\r\n // Vector2\r\n case Animation.ANIMATIONTYPE_VECTOR2:\r\n offsetValue = _staticOffsetValueVector2;\r\n break;\r\n // Size\r\n case Animation.ANIMATIONTYPE_SIZE:\r\n offsetValue = _staticOffsetValueSize;\r\n break;\r\n // Color3\r\n case Animation.ANIMATIONTYPE_COLOR3:\r\n offsetValue = _staticOffsetValueColor3;\r\n }\r\n }\r\n\r\n // Compute value\r\n let currentFrame: number;\r\n\r\n if (this._host && this._host.syncRoot) {\r\n // If we must sync with an animatable, calculate the current frame based on the frame of the root animatable\r\n const syncRoot = this._host.syncRoot;\r\n const hostNormalizedFrame = (syncRoot.masterFrame - syncRoot.fromFrame) / (syncRoot.toFrame - syncRoot.fromFrame);\r\n currentFrame = from + frameRange * hostNormalizedFrame;\r\n } else {\r\n if ((absoluteFrame > 0 && from > to) || (absoluteFrame < 0 && from < to)) {\r\n currentFrame = returnValue && frameRange !== 0 ? to + (absoluteFrame % frameRange) : from;\r\n } else {\r\n currentFrame = returnValue && frameRange !== 0 ? from + (absoluteFrame % frameRange) : to;\r\n }\r\n }\r\n\r\n const events = this._events;\r\n\r\n // Reset event/state if looping\r\n if ((speedRatio > 0 && this.currentFrame > currentFrame) || (speedRatio < 0 && this.currentFrame < currentFrame)) {\r\n this._onLoop();\r\n\r\n // Need to reset animation events\r\n for (let index = 0; index < events.length; index++) {\r\n if (!events[index].onlyOnce) {\r\n // reset event, the animation is looping\r\n events[index].isDone = false;\r\n }\r\n }\r\n\r\n this._animationState.key = speedRatio > 0 ? 0 : animation.getKeys().length - 1;\r\n }\r\n this._currentFrame = currentFrame;\r\n this._animationState.repeatCount = frameRange === 0 ? 0 : (absoluteFrame / frameRange) >> 0;\r\n this._animationState.highLimitValue = highLimitValue;\r\n this._animationState.offsetValue = offsetValue;\r\n\r\n const currentValue = animation._interpolate(currentFrame, this._animationState);\r\n\r\n // Set value\r\n this.setValue(currentValue, weight);\r\n\r\n // Check events\r\n if (events.length) {\r\n for (let index = 0; index < events.length; index++) {\r\n // Make sure current frame has passed event frame and that event frame is within the current range\r\n // Also, handle both forward and reverse animations\r\n if (\r\n (frameRange > 0 && currentFrame >= events[index].frame && events[index].frame >= from) ||\r\n (frameRange < 0 && currentFrame <= events[index].frame && events[index].frame <= from)\r\n ) {\r\n const event = events[index];\r\n if (!event.isDone) {\r\n // If event should be done only once, remove it.\r\n if (event.onlyOnce) {\r\n events.splice(index, 1);\r\n index--;\r\n }\r\n event.isDone = true;\r\n event.action(currentFrame);\r\n } // Don't do anything if the event has already been done.\r\n }\r\n }\r\n }\r\n\r\n if (!returnValue) {\r\n this._stopped = true;\r\n }\r\n\r\n return returnValue;\r\n }\r\n}\r\n","import type { Skeleton } from \"./skeleton\";\r\nimport { Vector3, Quaternion, Matrix, TmpVectors } from \"../Maths/math.vector\";\r\nimport { ArrayTools } from \"../Misc/arrayTools\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { TransformNode } from \"../Meshes/transformNode\";\r\nimport { Node } from \"../node\";\r\nimport { Space } from \"../Maths/math.axis\";\r\n\r\nimport type { Animation } from \"../Animations/animation\";\r\nimport type { AnimationPropertiesOverride } from \"../Animations/animationPropertiesOverride\";\r\n\r\n/**\r\n * Class used to store bone information\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/bonesSkeletons\r\n */\r\nexport class Bone extends Node {\r\n private static _TmpVecs: Vector3[] = ArrayTools.BuildArray(2, Vector3.Zero);\r\n private static _TmpQuat = Quaternion.Identity();\r\n private static _TmpMats: Matrix[] = ArrayTools.BuildArray(5, Matrix.Identity);\r\n\r\n /**\r\n * Gets the list of child bones\r\n */\r\n public children = new Array();\r\n\r\n /** Gets the animations associated with this bone */\r\n public animations = new Array();\r\n\r\n /**\r\n * Gets or sets bone length\r\n */\r\n public length: number;\r\n\r\n /**\r\n * @internal Internal only\r\n * Set this value to map this bone to a different index in the transform matrices\r\n * Set this value to -1 to exclude the bone from the transform matrices\r\n */\r\n public _index: Nullable = null;\r\n\r\n private _skeleton: Skeleton;\r\n private _localMatrix: Matrix; // transformation of the bone, in local space\r\n private _absoluteMatrix: Matrix; // transformation of the bone, in world space (relative to the skeleton root)\r\n private _bindMatrix: Matrix; // the bind matrix, in local space\r\n private _absoluteBindMatrix: Matrix; // the bind matrix, in world space (relative to the skeleton root)\r\n private _absoluteInverseBindMatrix: Matrix; // the inverse of the bind matrix, in world space (relative to the skeleton root)\r\n private _finalMatrix: Matrix; // the final matrix used to transform vertices of the mesh according to the bone, in world space (relative to the skeleton root). It is the multiplication of _absoluteInverseBindMatrix with _absoluteMatrix.\r\n private _restMatrix: Matrix; // a matrix for the exclusive use of the end user (not used internally by the framework), in local space\r\n private _scalingDeterminant = 1;\r\n\r\n private _localScaling: Vector3;\r\n private _localRotation: Quaternion;\r\n private _localPosition: Vector3;\r\n private _needToDecompose = true;\r\n private _needToCompose = false;\r\n\r\n /** @internal */\r\n public _linkedTransformNode: Nullable = null;\r\n\r\n /** @internal */\r\n public _waitingTransformNodeId: Nullable = null;\r\n\r\n /** @internal */\r\n get _matrix(): Matrix {\r\n this._compose();\r\n return this._localMatrix;\r\n }\r\n\r\n /** @internal */\r\n set _matrix(value: Matrix) {\r\n // skip if the matrices are the same\r\n if (value.updateFlag === this._localMatrix.updateFlag && !this._needToCompose) {\r\n return;\r\n }\r\n\r\n this._needToCompose = false; // in case there was a pending compose\r\n\r\n this._localMatrix.copyFrom(value);\r\n this._markAsDirtyAndDecompose();\r\n }\r\n\r\n /**\r\n * Create a new bone\r\n * @param name defines the bone name\r\n * @param skeleton defines the parent skeleton\r\n * @param parentBone defines the parent (can be null if the bone is the root)\r\n * @param localMatrix defines the local matrix (default: identity)\r\n * @param restMatrix defines the rest matrix (default: localMatrix)\r\n * @param bindMatrix defines the bind matrix (default: localMatrix)\r\n * @param index defines index of the bone in the hierarchy (default: null)\r\n */\r\n constructor(\r\n /**\r\n * defines the bone name\r\n */\r\n public name: string,\r\n skeleton: Skeleton,\r\n parentBone: Nullable = null,\r\n localMatrix: Nullable = null,\r\n restMatrix: Nullable = null,\r\n bindMatrix: Nullable = null,\r\n index: Nullable = null\r\n ) {\r\n super(name, skeleton.getScene());\r\n this._skeleton = skeleton;\r\n this._localMatrix = localMatrix?.clone() ?? Matrix.Identity();\r\n this._restMatrix = restMatrix ?? this._localMatrix.clone();\r\n this._bindMatrix = bindMatrix ?? this._localMatrix.clone();\r\n this._index = index;\r\n\r\n this._absoluteMatrix = new Matrix();\r\n this._absoluteBindMatrix = new Matrix();\r\n this._absoluteInverseBindMatrix = new Matrix();\r\n this._finalMatrix = new Matrix();\r\n\r\n skeleton.bones.push(this);\r\n\r\n this.setParent(parentBone, false);\r\n\r\n this._updateAbsoluteBindMatrices();\r\n }\r\n\r\n /**\r\n * Gets the current object class name.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"Bone\";\r\n }\r\n\r\n // Members\r\n\r\n /**\r\n * Gets the parent skeleton\r\n * @returns a skeleton\r\n */\r\n public getSkeleton(): Skeleton {\r\n return this._skeleton;\r\n }\r\n\r\n public get parent(): Bone {\r\n return this._parentNode as Bone;\r\n }\r\n\r\n /**\r\n * Gets parent bone\r\n * @returns a bone or null if the bone is the root of the bone hierarchy\r\n */\r\n public getParent(): Nullable {\r\n return this.parent;\r\n }\r\n\r\n /**\r\n * Returns an array containing the children of the bone\r\n * @returns an array containing the children of the bone (can be empty if the bone has no children)\r\n */\r\n public getChildren(): Array {\r\n return this.children;\r\n }\r\n\r\n /**\r\n * Gets the node index in matrix array generated for rendering\r\n * @returns the node index\r\n */\r\n public getIndex(): number {\r\n return this._index === null ? this.getSkeleton().bones.indexOf(this) : this._index;\r\n }\r\n\r\n public set parent(newParent: Nullable) {\r\n this.setParent(newParent);\r\n }\r\n\r\n /**\r\n * Sets the parent bone\r\n * @param parent defines the parent (can be null if the bone is the root)\r\n * @param updateAbsoluteBindMatrices defines if the absolute bind and absolute inverse bind matrices must be updated\r\n */\r\n public setParent(parent: Nullable, updateAbsoluteBindMatrices: boolean = true): void {\r\n if (this.parent === parent) {\r\n return;\r\n }\r\n\r\n if (this.parent) {\r\n const index = this.parent.children.indexOf(this);\r\n if (index !== -1) {\r\n this.parent.children.splice(index, 1);\r\n }\r\n }\r\n\r\n this._parentNode = parent;\r\n\r\n if (this.parent) {\r\n this.parent.children.push(this);\r\n }\r\n\r\n if (updateAbsoluteBindMatrices) {\r\n this._updateAbsoluteBindMatrices();\r\n }\r\n\r\n this.markAsDirty();\r\n }\r\n\r\n /**\r\n * Gets the local matrix\r\n * @returns the local matrix\r\n */\r\n public getLocalMatrix(): Matrix {\r\n this._compose();\r\n return this._localMatrix;\r\n }\r\n\r\n /**\r\n * Gets the bind matrix\r\n * @returns the bind matrix\r\n */\r\n public getBindMatrix(): Matrix {\r\n return this._bindMatrix;\r\n }\r\n\r\n /**\r\n * Gets the bind matrix.\r\n * @returns the bind matrix\r\n * @deprecated Please use getBindMatrix instead\r\n */\r\n public getBaseMatrix(): Matrix {\r\n return this.getBindMatrix();\r\n }\r\n\r\n /**\r\n * Gets the rest matrix\r\n * @returns the rest matrix\r\n */\r\n public getRestMatrix(): Matrix {\r\n return this._restMatrix;\r\n }\r\n\r\n /**\r\n * Gets the rest matrix\r\n * @returns the rest matrix\r\n * @deprecated Please use getRestMatrix instead\r\n */\r\n public getRestPose(): Matrix {\r\n return this.getRestMatrix();\r\n }\r\n\r\n /**\r\n * Sets the rest matrix\r\n * @param matrix the local-space rest matrix to set for this bone\r\n */\r\n public setRestMatrix(matrix: Matrix): void {\r\n this._restMatrix.copyFrom(matrix);\r\n }\r\n\r\n /**\r\n * Sets the rest matrix\r\n * @param matrix the local-space rest to set for this bone\r\n * @deprecated Please use setRestMatrix instead\r\n */\r\n public setRestPose(matrix: Matrix): void {\r\n this.setRestMatrix(matrix);\r\n }\r\n\r\n /**\r\n * Gets the bind matrix\r\n * @returns the bind matrix\r\n * @deprecated Please use getBindMatrix instead\r\n */\r\n public getBindPose(): Matrix {\r\n return this.getBindMatrix();\r\n }\r\n\r\n /**\r\n * Sets the bind matrix\r\n * This will trigger a recomputation of the absolute bind and absolute inverse bind matrices for this bone and its children\r\n * Note that the local matrix will also be set with the matrix passed in parameter!\r\n * @param matrix the local-space bind matrix to set for this bone\r\n */\r\n public setBindMatrix(matrix: Matrix): void {\r\n this.updateMatrix(matrix);\r\n }\r\n\r\n /**\r\n * Sets the bind matrix\r\n * @param matrix the local-space bind to set for this bone\r\n * @deprecated Please use setBindMatrix instead\r\n */\r\n public setBindPose(matrix: Matrix): void {\r\n this.setBindMatrix(matrix);\r\n }\r\n\r\n /**\r\n * Gets the matrix used to store the final world transformation of the bone (ie. the matrix sent to shaders)\r\n */\r\n public getFinalMatrix(): Matrix {\r\n return this._finalMatrix;\r\n }\r\n\r\n /**\r\n * Gets the matrix used to store the final world transformation of the bone (ie. the matrix sent to shaders)\r\n * @deprecated Please use getFinalMatrix instead\r\n */\r\n public getWorldMatrix(): Matrix {\r\n return this.getFinalMatrix();\r\n }\r\n\r\n /**\r\n * Sets the local matrix to the rest matrix\r\n */\r\n public returnToRest(): void {\r\n if (this._linkedTransformNode) {\r\n const localScaling = TmpVectors.Vector3[0];\r\n const localRotation = TmpVectors.Quaternion[0];\r\n const localPosition = TmpVectors.Vector3[1];\r\n\r\n this.getRestMatrix().decompose(localScaling, localRotation, localPosition);\r\n\r\n this._linkedTransformNode.position.copyFrom(localPosition);\r\n this._linkedTransformNode.rotationQuaternion = this._linkedTransformNode.rotationQuaternion ?? Quaternion.Identity();\r\n this._linkedTransformNode.rotationQuaternion.copyFrom(localRotation);\r\n this._linkedTransformNode.scaling.copyFrom(localScaling);\r\n } else {\r\n this._matrix = this._restMatrix;\r\n }\r\n }\r\n\r\n /**\r\n * Gets the inverse of the bind matrix, in world space (relative to the skeleton root)\r\n * @returns the inverse bind matrix, in world space\r\n */\r\n public getAbsoluteInverseBindMatrix(): Matrix {\r\n return this._absoluteInverseBindMatrix;\r\n }\r\n\r\n /**\r\n * Gets the inverse of the bind matrix, in world space (relative to the skeleton root)\r\n * @returns the inverse bind matrix, in world space\r\n * @deprecated Please use getAbsoluteInverseBindMatrix instead\r\n */\r\n public getInvertedAbsoluteTransform(): Matrix {\r\n return this.getAbsoluteInverseBindMatrix();\r\n }\r\n\r\n /**\r\n * Gets the bone matrix, in world space (relative to the skeleton root)\r\n * @returns the bone matrix, in world space\r\n */\r\n public getAbsoluteMatrix(): Matrix {\r\n return this._absoluteMatrix;\r\n }\r\n\r\n /**\r\n * Gets the bone matrix, in world space (relative to the skeleton root)\r\n * @returns the bone matrix, in world space\r\n * @deprecated Please use getAbsoluteMatrix instead\r\n */\r\n public getAbsoluteTransform(): Matrix {\r\n return this._absoluteMatrix;\r\n }\r\n\r\n /**\r\n * Links with the given transform node.\r\n * The local matrix of this bone is overwritten by the transform of the node every frame.\r\n * @param transformNode defines the transform node to link to\r\n */\r\n public linkTransformNode(transformNode: Nullable): void {\r\n if (this._linkedTransformNode) {\r\n this._skeleton._numBonesWithLinkedTransformNode--;\r\n }\r\n\r\n this._linkedTransformNode = transformNode;\r\n\r\n if (this._linkedTransformNode) {\r\n this._skeleton._numBonesWithLinkedTransformNode++;\r\n }\r\n }\r\n\r\n // Properties (matches TransformNode properties)\r\n\r\n /**\r\n * Gets the node used to drive the bone's transformation\r\n * @returns a transform node or null\r\n */\r\n public getTransformNode() {\r\n return this._linkedTransformNode;\r\n }\r\n\r\n /** Gets or sets current position (in local space) */\r\n public get position(): Vector3 {\r\n this._decompose();\r\n return this._localPosition;\r\n }\r\n\r\n public set position(newPosition: Vector3) {\r\n this._decompose();\r\n this._localPosition.copyFrom(newPosition);\r\n\r\n this._markAsDirtyAndCompose();\r\n }\r\n\r\n /** Gets or sets current rotation (in local space) */\r\n public get rotation(): Vector3 {\r\n return this.getRotation();\r\n }\r\n\r\n public set rotation(newRotation: Vector3) {\r\n this.setRotation(newRotation);\r\n }\r\n\r\n /** Gets or sets current rotation quaternion (in local space) */\r\n public get rotationQuaternion() {\r\n this._decompose();\r\n return this._localRotation;\r\n }\r\n\r\n public set rotationQuaternion(newRotation: Quaternion) {\r\n this.setRotationQuaternion(newRotation);\r\n }\r\n\r\n /** Gets or sets current scaling (in local space) */\r\n public get scaling(): Vector3 {\r\n return this.getScale();\r\n }\r\n\r\n public set scaling(newScaling: Vector3) {\r\n this.setScale(newScaling);\r\n }\r\n\r\n /**\r\n * Gets the animation properties override\r\n */\r\n public get animationPropertiesOverride(): Nullable {\r\n return this._skeleton.animationPropertiesOverride;\r\n }\r\n\r\n // Methods\r\n private _decompose() {\r\n if (!this._needToDecompose) {\r\n return;\r\n }\r\n\r\n this._needToDecompose = false;\r\n\r\n if (!this._localScaling) {\r\n this._localScaling = Vector3.Zero();\r\n this._localRotation = Quaternion.Zero();\r\n this._localPosition = Vector3.Zero();\r\n }\r\n this._localMatrix.decompose(this._localScaling, this._localRotation, this._localPosition);\r\n }\r\n\r\n private _compose() {\r\n if (!this._needToCompose) {\r\n return;\r\n }\r\n\r\n if (!this._localScaling) {\r\n this._needToCompose = false;\r\n return;\r\n }\r\n\r\n this._needToCompose = false;\r\n Matrix.ComposeToRef(this._localScaling, this._localRotation, this._localPosition, this._localMatrix);\r\n }\r\n\r\n /**\r\n * Update the bind (and optionally the local) matrix\r\n * @param bindMatrix defines the new matrix to set to the bind/local matrix, in local space\r\n * @param updateAbsoluteBindMatrices defines if the absolute bind and absolute inverse bind matrices must be recomputed (default: true)\r\n * @param updateLocalMatrix defines if the local matrix should also be updated with the matrix passed in parameter (default: true)\r\n */\r\n public updateMatrix(bindMatrix: Matrix, updateAbsoluteBindMatrices = true, updateLocalMatrix = true): void {\r\n this._bindMatrix.copyFrom(bindMatrix);\r\n\r\n if (updateAbsoluteBindMatrices) {\r\n this._updateAbsoluteBindMatrices();\r\n }\r\n\r\n if (updateLocalMatrix) {\r\n this._matrix = bindMatrix;\r\n } else {\r\n this.markAsDirty();\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _updateAbsoluteBindMatrices(bindMatrix?: Matrix, updateChildren = true): void {\r\n if (!bindMatrix) {\r\n bindMatrix = this._bindMatrix;\r\n }\r\n\r\n if (this.parent) {\r\n bindMatrix.multiplyToRef(this.parent._absoluteBindMatrix, this._absoluteBindMatrix);\r\n } else {\r\n this._absoluteBindMatrix.copyFrom(bindMatrix);\r\n }\r\n\r\n this._absoluteBindMatrix.invertToRef(this._absoluteInverseBindMatrix);\r\n\r\n if (updateChildren) {\r\n for (let index = 0; index < this.children.length; index++) {\r\n this.children[index]._updateAbsoluteBindMatrices();\r\n }\r\n }\r\n\r\n this._scalingDeterminant = this._absoluteBindMatrix.determinant() < 0 ? -1 : 1;\r\n }\r\n\r\n /**\r\n * Flag the bone as dirty (Forcing it to update everything)\r\n * @returns this bone\r\n */\r\n public markAsDirty(): Bone {\r\n this._currentRenderId++;\r\n this._childUpdateId++;\r\n this._skeleton._markAsDirty();\r\n return this;\r\n }\r\n\r\n /** @internal */\r\n public _markAsDirtyAndCompose() {\r\n this.markAsDirty();\r\n this._needToCompose = true;\r\n }\r\n\r\n private _markAsDirtyAndDecompose() {\r\n this.markAsDirty();\r\n this._needToDecompose = true;\r\n }\r\n\r\n private _updatePosition(vec: Vector3, space = Space.LOCAL, tNode?: TransformNode, translationMode = true): void {\r\n const lm = this.getLocalMatrix();\r\n\r\n if (space == Space.LOCAL) {\r\n if (translationMode) {\r\n lm.addAtIndex(12, vec.x);\r\n lm.addAtIndex(13, vec.y);\r\n lm.addAtIndex(14, vec.z);\r\n } else {\r\n lm.setTranslationFromFloats(vec.x, vec.y, vec.z);\r\n }\r\n } else {\r\n let wm: Nullable = null;\r\n\r\n //tNode.getWorldMatrix() needs to be called before skeleton.computeAbsoluteMatrices()\r\n if (tNode) {\r\n wm = tNode.getWorldMatrix();\r\n }\r\n\r\n this._skeleton.computeAbsoluteMatrices();\r\n\r\n const tmat = Bone._TmpMats[0];\r\n const tvec = Bone._TmpVecs[0];\r\n\r\n if (this.parent) {\r\n if (tNode && wm) {\r\n tmat.copyFrom(this.parent.getAbsoluteMatrix());\r\n tmat.multiplyToRef(wm, tmat);\r\n } else {\r\n tmat.copyFrom(this.parent.getAbsoluteMatrix());\r\n }\r\n } else {\r\n Matrix.IdentityToRef(tmat);\r\n }\r\n\r\n if (translationMode) {\r\n tmat.setTranslationFromFloats(0, 0, 0);\r\n }\r\n tmat.invert();\r\n Vector3.TransformCoordinatesToRef(vec, tmat, tvec);\r\n\r\n if (translationMode) {\r\n lm.addAtIndex(12, tvec.x);\r\n lm.addAtIndex(13, tvec.y);\r\n lm.addAtIndex(14, tvec.z);\r\n } else {\r\n lm.setTranslationFromFloats(tvec.x, tvec.y, tvec.z);\r\n }\r\n }\r\n\r\n this._markAsDirtyAndDecompose();\r\n }\r\n\r\n /**\r\n * Translate the bone in local or world space\r\n * @param vec The amount to translate the bone\r\n * @param space The space that the translation is in (default: Space.LOCAL)\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n */\r\n public translate(vec: Vector3, space = Space.LOCAL, tNode?: TransformNode): void {\r\n this._updatePosition(vec, space, tNode, true);\r\n }\r\n\r\n /**\r\n * Set the position of the bone in local or world space\r\n * @param position The position to set the bone\r\n * @param space The space that the position is in (default: Space.LOCAL)\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n */\r\n public setPosition(position: Vector3, space = Space.LOCAL, tNode?: TransformNode): void {\r\n this._updatePosition(position, space, tNode, false);\r\n }\r\n\r\n /**\r\n * Set the absolute position of the bone (world space)\r\n * @param position The position to set the bone\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n */\r\n public setAbsolutePosition(position: Vector3, tNode?: TransformNode) {\r\n this.setPosition(position, Space.WORLD, tNode);\r\n }\r\n\r\n /**\r\n * Scale the bone on the x, y and z axes (in local space)\r\n * @param x The amount to scale the bone on the x axis\r\n * @param y The amount to scale the bone on the y axis\r\n * @param z The amount to scale the bone on the z axis\r\n * @param scaleChildren sets this to true if children of the bone should be scaled as well (false by default)\r\n */\r\n public scale(x: number, y: number, z: number, scaleChildren = false): void {\r\n const locMat = this.getLocalMatrix();\r\n\r\n // Apply new scaling on top of current local matrix\r\n const scaleMat = Bone._TmpMats[0];\r\n Matrix.ScalingToRef(x, y, z, scaleMat);\r\n scaleMat.multiplyToRef(locMat, locMat);\r\n\r\n // Invert scaling matrix and apply the inverse to all children\r\n scaleMat.invert();\r\n\r\n for (const child of this.children) {\r\n const cm = child.getLocalMatrix();\r\n cm.multiplyToRef(scaleMat, cm);\r\n cm.multiplyAtIndex(12, x);\r\n cm.multiplyAtIndex(13, y);\r\n cm.multiplyAtIndex(14, z);\r\n\r\n child._markAsDirtyAndDecompose();\r\n }\r\n\r\n this._markAsDirtyAndDecompose();\r\n\r\n if (scaleChildren) {\r\n for (const child of this.children) {\r\n child.scale(x, y, z, scaleChildren);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Set the bone scaling in local space\r\n * @param scale defines the scaling vector\r\n */\r\n public setScale(scale: Vector3): void {\r\n this._decompose();\r\n this._localScaling.copyFrom(scale);\r\n this._markAsDirtyAndCompose();\r\n }\r\n\r\n /**\r\n * Gets the current scaling in local space\r\n * @returns the current scaling vector\r\n */\r\n public getScale(): Vector3 {\r\n this._decompose();\r\n return this._localScaling;\r\n }\r\n\r\n /**\r\n * Gets the current scaling in local space and stores it in a target vector\r\n * @param result defines the target vector\r\n */\r\n public getScaleToRef(result: Vector3) {\r\n this._decompose();\r\n result.copyFrom(this._localScaling);\r\n }\r\n\r\n /**\r\n * Set the yaw, pitch, and roll of the bone in local or world space\r\n * @param yaw The rotation of the bone on the y axis\r\n * @param pitch The rotation of the bone on the x axis\r\n * @param roll The rotation of the bone on the z axis\r\n * @param space The space that the axes of rotation are in\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n */\r\n public setYawPitchRoll(yaw: number, pitch: number, roll: number, space = Space.LOCAL, tNode?: TransformNode): void {\r\n if (space === Space.LOCAL) {\r\n const quat = Bone._TmpQuat;\r\n Quaternion.RotationYawPitchRollToRef(yaw, pitch, roll, quat);\r\n this.setRotationQuaternion(quat, space, tNode);\r\n return;\r\n }\r\n\r\n const rotMatInv = Bone._TmpMats[0];\r\n if (!this._getAbsoluteInverseMatrixUnscaledToRef(rotMatInv, tNode)) {\r\n return;\r\n }\r\n\r\n const rotMat = Bone._TmpMats[1];\r\n Matrix.RotationYawPitchRollToRef(yaw, pitch, roll, rotMat);\r\n\r\n rotMatInv.multiplyToRef(rotMat, rotMat);\r\n this._rotateWithMatrix(rotMat, space, tNode);\r\n }\r\n\r\n /**\r\n * Add a rotation to the bone on an axis in local or world space\r\n * @param axis The axis to rotate the bone on\r\n * @param amount The amount to rotate the bone\r\n * @param space The space that the axis is in\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n */\r\n public rotate(axis: Vector3, amount: number, space = Space.LOCAL, tNode?: TransformNode): void {\r\n const rmat = Bone._TmpMats[0];\r\n rmat.setTranslationFromFloats(0, 0, 0);\r\n Matrix.RotationAxisToRef(axis, amount, rmat);\r\n this._rotateWithMatrix(rmat, space, tNode);\r\n }\r\n\r\n /**\r\n * Set the rotation of the bone to a particular axis angle in local or world space\r\n * @param axis The axis to rotate the bone on\r\n * @param angle The angle that the bone should be rotated to\r\n * @param space The space that the axis is in\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n */\r\n public setAxisAngle(axis: Vector3, angle: number, space = Space.LOCAL, tNode?: TransformNode): void {\r\n if (space === Space.LOCAL) {\r\n const quat = Bone._TmpQuat;\r\n Quaternion.RotationAxisToRef(axis, angle, quat);\r\n\r\n this.setRotationQuaternion(quat, space, tNode);\r\n return;\r\n }\r\n\r\n const rotMatInv = Bone._TmpMats[0];\r\n if (!this._getAbsoluteInverseMatrixUnscaledToRef(rotMatInv, tNode)) {\r\n return;\r\n }\r\n\r\n const rotMat = Bone._TmpMats[1];\r\n Matrix.RotationAxisToRef(axis, angle, rotMat);\r\n\r\n rotMatInv.multiplyToRef(rotMat, rotMat);\r\n this._rotateWithMatrix(rotMat, space, tNode);\r\n }\r\n\r\n /**\r\n * Set the euler rotation of the bone in local or world space\r\n * @param rotation The euler rotation that the bone should be set to\r\n * @param space The space that the rotation is in\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n */\r\n public setRotation(rotation: Vector3, space = Space.LOCAL, tNode?: TransformNode): void {\r\n this.setYawPitchRoll(rotation.y, rotation.x, rotation.z, space, tNode);\r\n }\r\n\r\n /**\r\n * Set the quaternion rotation of the bone in local or world space\r\n * @param quat The quaternion rotation that the bone should be set to\r\n * @param space The space that the rotation is in\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n */\r\n public setRotationQuaternion(quat: Quaternion, space = Space.LOCAL, tNode?: TransformNode): void {\r\n if (space === Space.LOCAL) {\r\n this._decompose();\r\n this._localRotation.copyFrom(quat);\r\n\r\n this._markAsDirtyAndCompose();\r\n\r\n return;\r\n }\r\n\r\n const rotMatInv = Bone._TmpMats[0];\r\n if (!this._getAbsoluteInverseMatrixUnscaledToRef(rotMatInv, tNode)) {\r\n return;\r\n }\r\n\r\n const rotMat = Bone._TmpMats[1];\r\n Matrix.FromQuaternionToRef(quat, rotMat);\r\n\r\n rotMatInv.multiplyToRef(rotMat, rotMat);\r\n\r\n this._rotateWithMatrix(rotMat, space, tNode);\r\n }\r\n\r\n /**\r\n * Set the rotation matrix of the bone in local or world space\r\n * @param rotMat The rotation matrix that the bone should be set to\r\n * @param space The space that the rotation is in\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n */\r\n public setRotationMatrix(rotMat: Matrix, space = Space.LOCAL, tNode?: TransformNode): void {\r\n if (space === Space.LOCAL) {\r\n const quat = Bone._TmpQuat;\r\n Quaternion.FromRotationMatrixToRef(rotMat, quat);\r\n this.setRotationQuaternion(quat, space, tNode);\r\n return;\r\n }\r\n\r\n const rotMatInv = Bone._TmpMats[0];\r\n if (!this._getAbsoluteInverseMatrixUnscaledToRef(rotMatInv, tNode)) {\r\n return;\r\n }\r\n\r\n const rotMat2 = Bone._TmpMats[1];\r\n rotMat2.copyFrom(rotMat);\r\n\r\n rotMatInv.multiplyToRef(rotMat, rotMat2);\r\n\r\n this._rotateWithMatrix(rotMat2, space, tNode);\r\n }\r\n\r\n private _rotateWithMatrix(rmat: Matrix, space = Space.LOCAL, tNode?: TransformNode): void {\r\n const lmat = this.getLocalMatrix();\r\n const lx = lmat.m[12];\r\n const ly = lmat.m[13];\r\n const lz = lmat.m[14];\r\n const parent = this.getParent();\r\n const parentScale = Bone._TmpMats[3];\r\n const parentScaleInv = Bone._TmpMats[4];\r\n\r\n if (parent && space == Space.WORLD) {\r\n if (tNode) {\r\n parentScale.copyFrom(tNode.getWorldMatrix());\r\n parent.getAbsoluteMatrix().multiplyToRef(parentScale, parentScale);\r\n } else {\r\n parentScale.copyFrom(parent.getAbsoluteMatrix());\r\n }\r\n parentScaleInv.copyFrom(parentScale);\r\n parentScaleInv.invert();\r\n lmat.multiplyToRef(parentScale, lmat);\r\n lmat.multiplyToRef(rmat, lmat);\r\n lmat.multiplyToRef(parentScaleInv, lmat);\r\n } else {\r\n if (space == Space.WORLD && tNode) {\r\n parentScale.copyFrom(tNode.getWorldMatrix());\r\n parentScaleInv.copyFrom(parentScale);\r\n parentScaleInv.invert();\r\n lmat.multiplyToRef(parentScale, lmat);\r\n lmat.multiplyToRef(rmat, lmat);\r\n lmat.multiplyToRef(parentScaleInv, lmat);\r\n } else {\r\n lmat.multiplyToRef(rmat, lmat);\r\n }\r\n }\r\n\r\n lmat.setTranslationFromFloats(lx, ly, lz);\r\n\r\n this.computeAbsoluteMatrices();\r\n this._markAsDirtyAndDecompose();\r\n }\r\n\r\n private _getAbsoluteInverseMatrixUnscaledToRef(rotMatInv: Matrix, tNode?: TransformNode): boolean {\r\n const scaleMatrix = Bone._TmpMats[2];\r\n rotMatInv.copyFrom(this.getAbsoluteMatrix());\r\n\r\n if (tNode) {\r\n rotMatInv.multiplyToRef(tNode.getWorldMatrix(), rotMatInv);\r\n Matrix.ScalingToRef(tNode.scaling.x, tNode.scaling.y, tNode.scaling.z, scaleMatrix);\r\n } else {\r\n Matrix.IdentityToRef(scaleMatrix);\r\n }\r\n\r\n rotMatInv.invert();\r\n if (isNaN(rotMatInv.m[0])) {\r\n // Matrix failed to invert.\r\n // This can happen if scale is zero for example.\r\n return false;\r\n }\r\n\r\n scaleMatrix.multiplyAtIndex(0, this._scalingDeterminant);\r\n rotMatInv.multiplyToRef(scaleMatrix, rotMatInv);\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Get the position of the bone in local or world space\r\n * @param space The space that the returned position is in\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n * @returns The position of the bone\r\n */\r\n public getPosition(space = Space.LOCAL, tNode: Nullable = null): Vector3 {\r\n const pos = Vector3.Zero();\r\n\r\n this.getPositionToRef(space, tNode, pos);\r\n\r\n return pos;\r\n }\r\n\r\n /**\r\n * Copy the position of the bone to a vector3 in local or world space\r\n * @param space The space that the returned position is in\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n * @param result The vector3 to copy the position to\r\n */\r\n public getPositionToRef(space = Space.LOCAL, tNode: Nullable, result: Vector3): void {\r\n if (space == Space.LOCAL) {\r\n const lm = this.getLocalMatrix();\r\n\r\n result.x = lm.m[12];\r\n result.y = lm.m[13];\r\n result.z = lm.m[14];\r\n } else {\r\n let wm: Nullable = null;\r\n\r\n //tNode.getWorldMatrix() needs to be called before skeleton.computeAbsoluteMatrices()\r\n if (tNode) {\r\n wm = tNode.getWorldMatrix();\r\n }\r\n\r\n this._skeleton.computeAbsoluteMatrices();\r\n\r\n let tmat = Bone._TmpMats[0];\r\n\r\n if (tNode && wm) {\r\n tmat.copyFrom(this.getAbsoluteMatrix());\r\n tmat.multiplyToRef(wm, tmat);\r\n } else {\r\n tmat = this.getAbsoluteMatrix();\r\n }\r\n\r\n result.x = tmat.m[12];\r\n result.y = tmat.m[13];\r\n result.z = tmat.m[14];\r\n }\r\n }\r\n\r\n /**\r\n * Get the absolute position of the bone (world space)\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n * @returns The absolute position of the bone\r\n */\r\n public getAbsolutePosition(tNode: Nullable = null): Vector3 {\r\n const pos = Vector3.Zero();\r\n\r\n this.getPositionToRef(Space.WORLD, tNode, pos);\r\n\r\n return pos;\r\n }\r\n\r\n /**\r\n * Copy the absolute position of the bone (world space) to the result param\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n * @param result The vector3 to copy the absolute position to\r\n */\r\n public getAbsolutePositionToRef(tNode: TransformNode, result: Vector3) {\r\n this.getPositionToRef(Space.WORLD, tNode, result);\r\n }\r\n\r\n /**\r\n * Compute the absolute matrices of this bone and its children\r\n */\r\n public computeAbsoluteMatrices(): void {\r\n this._compose();\r\n\r\n if (this.parent) {\r\n this._localMatrix.multiplyToRef(this.parent._absoluteMatrix, this._absoluteMatrix);\r\n } else {\r\n this._absoluteMatrix.copyFrom(this._localMatrix);\r\n\r\n const poseMatrix = this._skeleton.getPoseMatrix();\r\n\r\n if (poseMatrix) {\r\n this._absoluteMatrix.multiplyToRef(poseMatrix, this._absoluteMatrix);\r\n }\r\n }\r\n\r\n const children = this.children;\r\n const len = children.length;\r\n\r\n for (let i = 0; i < len; i++) {\r\n children[i].computeAbsoluteMatrices();\r\n }\r\n }\r\n\r\n /**\r\n * Compute the absolute matrices of this bone and its children\r\n * @deprecated Please use computeAbsoluteMatrices instead\r\n */\r\n public computeAbsoluteTransforms(): void {\r\n this.computeAbsoluteMatrices();\r\n }\r\n\r\n /**\r\n * Get the world direction from an axis that is in the local space of the bone\r\n * @param localAxis The local direction that is used to compute the world direction\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n * @returns The world direction\r\n */\r\n public getDirection(localAxis: Vector3, tNode: Nullable = null): Vector3 {\r\n const result = Vector3.Zero();\r\n\r\n this.getDirectionToRef(localAxis, tNode, result);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Copy the world direction to a vector3 from an axis that is in the local space of the bone\r\n * @param localAxis The local direction that is used to compute the world direction\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n * @param result The vector3 that the world direction will be copied to\r\n */\r\n public getDirectionToRef(localAxis: Vector3, tNode: Nullable = null, result: Vector3): void {\r\n let wm: Nullable = null;\r\n\r\n //tNode.getWorldMatrix() needs to be called before skeleton.computeAbsoluteMatrices()\r\n if (tNode) {\r\n wm = tNode.getWorldMatrix();\r\n }\r\n\r\n this._skeleton.computeAbsoluteMatrices();\r\n\r\n const mat = Bone._TmpMats[0];\r\n\r\n mat.copyFrom(this.getAbsoluteMatrix());\r\n\r\n if (tNode && wm) {\r\n mat.multiplyToRef(wm, mat);\r\n }\r\n\r\n Vector3.TransformNormalToRef(localAxis, mat, result);\r\n\r\n result.normalize();\r\n }\r\n\r\n /**\r\n * Get the euler rotation of the bone in local or world space\r\n * @param space The space that the rotation should be in\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n * @returns The euler rotation\r\n */\r\n public getRotation(space = Space.LOCAL, tNode: Nullable = null): Vector3 {\r\n const result = Vector3.Zero();\r\n\r\n this.getRotationToRef(space, tNode, result);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Copy the euler rotation of the bone to a vector3. The rotation can be in either local or world space\r\n * @param space The space that the rotation should be in\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n * @param result The vector3 that the rotation should be copied to\r\n */\r\n public getRotationToRef(space = Space.LOCAL, tNode: Nullable = null, result: Vector3): void {\r\n const quat = Bone._TmpQuat;\r\n\r\n this.getRotationQuaternionToRef(space, tNode, quat);\r\n\r\n quat.toEulerAnglesToRef(result);\r\n }\r\n\r\n /**\r\n * Get the quaternion rotation of the bone in either local or world space\r\n * @param space The space that the rotation should be in\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n * @returns The quaternion rotation\r\n */\r\n public getRotationQuaternion(space = Space.LOCAL, tNode: Nullable = null): Quaternion {\r\n const result = Quaternion.Identity();\r\n\r\n this.getRotationQuaternionToRef(space, tNode, result);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Copy the quaternion rotation of the bone to a quaternion. The rotation can be in either local or world space\r\n * @param space The space that the rotation should be in\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n * @param result The quaternion that the rotation should be copied to\r\n */\r\n public getRotationQuaternionToRef(space = Space.LOCAL, tNode: Nullable = null, result: Quaternion): void {\r\n if (space == Space.LOCAL) {\r\n this._decompose();\r\n result.copyFrom(this._localRotation);\r\n } else {\r\n const mat = Bone._TmpMats[0];\r\n const amat = this.getAbsoluteMatrix();\r\n\r\n if (tNode) {\r\n amat.multiplyToRef(tNode.getWorldMatrix(), mat);\r\n } else {\r\n mat.copyFrom(amat);\r\n }\r\n\r\n mat.multiplyAtIndex(0, this._scalingDeterminant);\r\n mat.multiplyAtIndex(1, this._scalingDeterminant);\r\n mat.multiplyAtIndex(2, this._scalingDeterminant);\r\n\r\n mat.decompose(undefined, result, undefined);\r\n }\r\n }\r\n\r\n /**\r\n * Get the rotation matrix of the bone in local or world space\r\n * @param space The space that the rotation should be in\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n * @returns The rotation matrix\r\n */\r\n public getRotationMatrix(space = Space.LOCAL, tNode: TransformNode): Matrix {\r\n const result = Matrix.Identity();\r\n\r\n this.getRotationMatrixToRef(space, tNode, result);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Copy the rotation matrix of the bone to a matrix. The rotation can be in either local or world space\r\n * @param space The space that the rotation should be in\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n * @param result The quaternion that the rotation should be copied to\r\n */\r\n public getRotationMatrixToRef(space = Space.LOCAL, tNode: TransformNode, result: Matrix): void {\r\n if (space == Space.LOCAL) {\r\n this.getLocalMatrix().getRotationMatrixToRef(result);\r\n } else {\r\n const mat = Bone._TmpMats[0];\r\n const amat = this.getAbsoluteMatrix();\r\n\r\n if (tNode) {\r\n amat.multiplyToRef(tNode.getWorldMatrix(), mat);\r\n } else {\r\n mat.copyFrom(amat);\r\n }\r\n\r\n mat.multiplyAtIndex(0, this._scalingDeterminant);\r\n mat.multiplyAtIndex(1, this._scalingDeterminant);\r\n mat.multiplyAtIndex(2, this._scalingDeterminant);\r\n\r\n mat.getRotationMatrixToRef(result);\r\n }\r\n }\r\n\r\n /**\r\n * Get the world position of a point that is in the local space of the bone\r\n * @param position The local position\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n * @returns The world position\r\n */\r\n public getAbsolutePositionFromLocal(position: Vector3, tNode: Nullable = null): Vector3 {\r\n const result = Vector3.Zero();\r\n\r\n this.getAbsolutePositionFromLocalToRef(position, tNode, result);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Get the world position of a point that is in the local space of the bone and copy it to the result param\r\n * @param position The local position\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n * @param result The vector3 that the world position should be copied to\r\n */\r\n public getAbsolutePositionFromLocalToRef(position: Vector3, tNode: Nullable = null, result: Vector3): void {\r\n let wm: Nullable = null;\r\n\r\n //tNode.getWorldMatrix() needs to be called before skeleton.computeAbsoluteMatrices()\r\n if (tNode) {\r\n wm = tNode.getWorldMatrix();\r\n }\r\n\r\n this._skeleton.computeAbsoluteMatrices();\r\n\r\n const tmat = Bone._TmpMats[0];\r\n\r\n tmat.copyFrom(this.getAbsoluteMatrix());\r\n\r\n if (tNode && wm) {\r\n tmat.multiplyToRef(wm, tmat);\r\n }\r\n\r\n Vector3.TransformCoordinatesToRef(position, tmat, result);\r\n }\r\n\r\n /**\r\n * Get the local position of a point that is in world space\r\n * @param position The world position\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n * @returns The local position\r\n */\r\n public getLocalPositionFromAbsolute(position: Vector3, tNode: Nullable = null): Vector3 {\r\n const result = Vector3.Zero();\r\n\r\n this.getLocalPositionFromAbsoluteToRef(position, tNode, result);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Get the local position of a point that is in world space and copy it to the result param\r\n * @param position The world position\r\n * @param tNode A TransformNode whose world matrix is to be applied to the calculated absolute matrix. In most cases, you'll want to pass the mesh associated with the skeleton from which this bone comes. Used only when space=Space.WORLD\r\n * @param result The vector3 that the local position should be copied to\r\n */\r\n public getLocalPositionFromAbsoluteToRef(position: Vector3, tNode: Nullable = null, result: Vector3): void {\r\n let wm: Nullable = null;\r\n\r\n //tNode.getWorldMatrix() needs to be called before skeleton.computeAbsoluteMatrices()\r\n if (tNode) {\r\n wm = tNode.getWorldMatrix();\r\n }\r\n\r\n this._skeleton.computeAbsoluteMatrices();\r\n\r\n const tmat = Bone._TmpMats[0];\r\n\r\n tmat.copyFrom(this.getAbsoluteMatrix());\r\n\r\n if (tNode && wm) {\r\n tmat.multiplyToRef(wm, tmat);\r\n }\r\n\r\n tmat.invert();\r\n\r\n Vector3.TransformCoordinatesToRef(position, tmat, result);\r\n }\r\n\r\n /**\r\n * Set the current local matrix as the restMatrix for this bone.\r\n */\r\n public setCurrentPoseAsRest(): void {\r\n this.setRestMatrix(this.getLocalMatrix());\r\n }\r\n}\r\n","import { Animation } from \"./animation\";\r\nimport { RuntimeAnimation } from \"./runtimeAnimation\";\r\n\r\nimport type { Nullable } from \"../types\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport { Scene } from \"../scene\";\r\nimport { Matrix, Quaternion, Vector3, TmpVectors } from \"../Maths/math.vector\";\r\nimport { PrecisionDate } from \"../Misc/precisionDate\";\r\nimport { Bone } from \"../Bones/bone\";\r\nimport type { Node } from \"../node\";\r\n\r\n/**\r\n * Class used to store an actual running animation\r\n */\r\nexport class Animatable {\r\n private _localDelayOffset: Nullable = null;\r\n private _pausedDelay: Nullable = null;\r\n private _manualJumpDelay: Nullable = null;\r\n /** @hidden */\r\n public _runtimeAnimations = new Array();\r\n private _paused = false;\r\n private _scene: Scene;\r\n private _speedRatio = 1;\r\n private _weight = -1.0;\r\n private _syncRoot: Nullable = null;\r\n private _frameToSyncFromJump: Nullable = null;\r\n private _goToFrame: Nullable = null;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if the animatable must be disposed and removed at the end of the animation.\r\n * This will only apply for non looping animation (default is true)\r\n */\r\n public disposeOnEnd = true;\r\n\r\n /**\r\n * Gets a boolean indicating if the animation has started\r\n */\r\n public animationStarted = false;\r\n\r\n /**\r\n * Observer raised when the animation ends\r\n */\r\n public onAnimationEndObservable = new Observable();\r\n\r\n /**\r\n * Observer raised when the animation loops\r\n */\r\n public onAnimationLoopObservable = new Observable();\r\n\r\n /**\r\n * Gets the root Animatable used to synchronize and normalize animations\r\n */\r\n public get syncRoot(): Nullable {\r\n return this._syncRoot;\r\n }\r\n\r\n /**\r\n * Gets the current frame of the first RuntimeAnimation\r\n * Used to synchronize Animatables\r\n */\r\n public get masterFrame(): number {\r\n if (this._runtimeAnimations.length === 0) {\r\n return 0;\r\n }\r\n\r\n return this._runtimeAnimations[0].currentFrame;\r\n }\r\n\r\n /**\r\n * Gets or sets the animatable weight (-1.0 by default meaning not weighted)\r\n */\r\n public get weight(): number {\r\n return this._weight;\r\n }\r\n\r\n public set weight(value: number) {\r\n if (value === -1) {\r\n // -1 is ok and means no weight\r\n this._weight = -1;\r\n return;\r\n }\r\n\r\n // Else weight must be in [0, 1] range\r\n this._weight = Math.min(Math.max(value, 0), 1.0);\r\n }\r\n\r\n /**\r\n * Gets or sets the speed ratio to apply to the animatable (1.0 by default)\r\n */\r\n public get speedRatio(): number {\r\n return this._speedRatio;\r\n }\r\n\r\n public set speedRatio(value: number) {\r\n for (let index = 0; index < this._runtimeAnimations.length; index++) {\r\n const animation = this._runtimeAnimations[index];\r\n\r\n animation._prepareForSpeedRatioChange(value);\r\n }\r\n this._speedRatio = value;\r\n\r\n // Resync _manualJumpDelay in case goToFrame was called before speedRatio was set.\r\n if (this._goToFrame !== null) {\r\n this.goToFrame(this._goToFrame);\r\n }\r\n }\r\n\r\n /**\r\n * Creates a new Animatable\r\n * @param scene defines the hosting scene\r\n * @param target defines the target object\r\n * @param fromFrame defines the starting frame number (default is 0)\r\n * @param toFrame defines the ending frame number (default is 100)\r\n * @param loopAnimation defines if the animation must loop (default is false)\r\n * @param speedRatio defines the factor to apply to animation speed (default is 1)\r\n * @param onAnimationEnd defines a callback to call when animation ends if it is not looping\r\n * @param animations defines a group of animation to add to the new Animatable\r\n * @param onAnimationLoop defines a callback to call when animation loops\r\n * @param isAdditive defines whether the animation should be evaluated additively\r\n * @param playOrder defines the order in which this animatable should be processed in the list of active animatables (default: 0)\r\n */\r\n constructor(\r\n scene: Scene,\r\n /** defines the target object */\r\n public target: any,\r\n /** defines the starting frame number (default is 0) */\r\n public fromFrame: number = 0,\r\n /** defines the ending frame number (default is 100) */\r\n public toFrame: number = 100,\r\n /** defines if the animation must loop (default is false) */\r\n public loopAnimation: boolean = false,\r\n speedRatio: number = 1.0,\r\n /** defines a callback to call when animation ends if it is not looping */\r\n public onAnimationEnd?: Nullable<() => void>,\r\n animations?: Animation[],\r\n /** defines a callback to call when animation loops */\r\n public onAnimationLoop?: Nullable<() => void>,\r\n /** defines whether the animation should be evaluated additively */\r\n public isAdditive: boolean = false,\r\n /** defines the order in which this animatable should be processed in the list of active animatables (default: 0) */\r\n public playOrder = 0\r\n ) {\r\n this._scene = scene;\r\n if (animations) {\r\n this.appendAnimations(target, animations);\r\n }\r\n\r\n this._speedRatio = speedRatio;\r\n scene._activeAnimatables.push(this);\r\n }\r\n\r\n // Methods\r\n /**\r\n * Synchronize and normalize current Animatable with a source Animatable\r\n * This is useful when using animation weights and when animations are not of the same length\r\n * @param root defines the root Animatable to synchronize with (null to stop synchronizing)\r\n * @returns the current Animatable\r\n */\r\n public syncWith(root: Nullable): Animatable {\r\n this._syncRoot = root;\r\n\r\n if (root) {\r\n // Make sure this animatable will animate after the root\r\n const index = this._scene._activeAnimatables.indexOf(this);\r\n if (index > -1) {\r\n this._scene._activeAnimatables.splice(index, 1);\r\n this._scene._activeAnimatables.push(this);\r\n }\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Gets the list of runtime animations\r\n * @returns an array of RuntimeAnimation\r\n */\r\n public getAnimations(): RuntimeAnimation[] {\r\n return this._runtimeAnimations;\r\n }\r\n\r\n /**\r\n * Adds more animations to the current animatable\r\n * @param target defines the target of the animations\r\n * @param animations defines the new animations to add\r\n */\r\n public appendAnimations(target: any, animations: Animation[]): void {\r\n for (let index = 0; index < animations.length; index++) {\r\n const animation = animations[index];\r\n\r\n const newRuntimeAnimation = new RuntimeAnimation(target, animation, this._scene, this);\r\n newRuntimeAnimation._onLoop = () => {\r\n this.onAnimationLoopObservable.notifyObservers(this);\r\n if (this.onAnimationLoop) {\r\n this.onAnimationLoop();\r\n }\r\n };\r\n\r\n this._runtimeAnimations.push(newRuntimeAnimation);\r\n }\r\n }\r\n\r\n /**\r\n * Gets the source animation for a specific property\r\n * @param property defines the property to look for\r\n * @returns null or the source animation for the given property\r\n */\r\n public getAnimationByTargetProperty(property: string): Nullable {\r\n const runtimeAnimations = this._runtimeAnimations;\r\n\r\n for (let index = 0; index < runtimeAnimations.length; index++) {\r\n if (runtimeAnimations[index].animation.targetProperty === property) {\r\n return runtimeAnimations[index].animation;\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets the runtime animation for a specific property\r\n * @param property defines the property to look for\r\n * @returns null or the runtime animation for the given property\r\n */\r\n public getRuntimeAnimationByTargetProperty(property: string): Nullable {\r\n const runtimeAnimations = this._runtimeAnimations;\r\n\r\n for (let index = 0; index < runtimeAnimations.length; index++) {\r\n if (runtimeAnimations[index].animation.targetProperty === property) {\r\n return runtimeAnimations[index];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Resets the animatable to its original state\r\n */\r\n public reset(): void {\r\n const runtimeAnimations = this._runtimeAnimations;\r\n\r\n for (let index = 0; index < runtimeAnimations.length; index++) {\r\n runtimeAnimations[index].reset(true);\r\n }\r\n\r\n this._localDelayOffset = null;\r\n this._pausedDelay = null;\r\n }\r\n\r\n /**\r\n * Allows the animatable to blend with current running animations\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#animation-blending\r\n * @param blendingSpeed defines the blending speed to use\r\n */\r\n public enableBlending(blendingSpeed: number): void {\r\n const runtimeAnimations = this._runtimeAnimations;\r\n\r\n for (let index = 0; index < runtimeAnimations.length; index++) {\r\n runtimeAnimations[index].animation.enableBlending = true;\r\n runtimeAnimations[index].animation.blendingSpeed = blendingSpeed;\r\n }\r\n }\r\n\r\n /**\r\n * Disable animation blending\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/animation/advanced_animations#animation-blending\r\n */\r\n public disableBlending(): void {\r\n const runtimeAnimations = this._runtimeAnimations;\r\n\r\n for (let index = 0; index < runtimeAnimations.length; index++) {\r\n runtimeAnimations[index].animation.enableBlending = false;\r\n }\r\n }\r\n\r\n /**\r\n * Jump directly to a given frame\r\n * @param frame defines the frame to jump to\r\n */\r\n public goToFrame(frame: number): void {\r\n const runtimeAnimations = this._runtimeAnimations;\r\n\r\n if (runtimeAnimations[0]) {\r\n const fps = runtimeAnimations[0].animation.framePerSecond;\r\n this._frameToSyncFromJump = this._frameToSyncFromJump ?? runtimeAnimations[0].currentFrame;\r\n const delay = this.speedRatio === 0 ? 0 : (((frame - this._frameToSyncFromJump) / fps) * 1000) / this.speedRatio;\r\n this._manualJumpDelay = -delay;\r\n }\r\n\r\n for (let index = 0; index < runtimeAnimations.length; index++) {\r\n runtimeAnimations[index].goToFrame(frame);\r\n }\r\n\r\n this._goToFrame = frame;\r\n }\r\n\r\n /**\r\n * Pause the animation\r\n */\r\n public pause(): void {\r\n if (this._paused) {\r\n return;\r\n }\r\n this._paused = true;\r\n }\r\n\r\n /**\r\n * Restart the animation\r\n */\r\n public restart(): void {\r\n this._paused = false;\r\n }\r\n\r\n private _raiseOnAnimationEnd() {\r\n if (this.onAnimationEnd) {\r\n this.onAnimationEnd();\r\n }\r\n\r\n this.onAnimationEndObservable.notifyObservers(this);\r\n }\r\n\r\n /**\r\n * Stop and delete the current animation\r\n * @param animationName defines a string used to only stop some of the runtime animations instead of all\r\n * @param targetMask a function that determines if the animation should be stopped based on its target (all animations will be stopped if both this and animationName are empty)\r\n * @param useGlobalSplice if true, the animatables will be removed by the caller of this function (false by default)\r\n */\r\n public stop(animationName?: string, targetMask?: (target: any) => boolean, useGlobalSplice = false): void {\r\n if (animationName || targetMask) {\r\n const idx = this._scene._activeAnimatables.indexOf(this);\r\n\r\n if (idx > -1) {\r\n const runtimeAnimations = this._runtimeAnimations;\r\n\r\n for (let index = runtimeAnimations.length - 1; index >= 0; index--) {\r\n const runtimeAnimation = runtimeAnimations[index];\r\n if (animationName && runtimeAnimation.animation.name != animationName) {\r\n continue;\r\n }\r\n if (targetMask && !targetMask(runtimeAnimation.target)) {\r\n continue;\r\n }\r\n\r\n runtimeAnimation.dispose();\r\n runtimeAnimations.splice(index, 1);\r\n }\r\n\r\n if (runtimeAnimations.length == 0) {\r\n if (!useGlobalSplice) {\r\n this._scene._activeAnimatables.splice(idx, 1);\r\n }\r\n this._raiseOnAnimationEnd();\r\n }\r\n }\r\n } else {\r\n const index = this._scene._activeAnimatables.indexOf(this);\r\n\r\n if (index > -1) {\r\n if (!useGlobalSplice) {\r\n this._scene._activeAnimatables.splice(index, 1);\r\n }\r\n const runtimeAnimations = this._runtimeAnimations;\r\n\r\n for (let index = 0; index < runtimeAnimations.length; index++) {\r\n runtimeAnimations[index].dispose();\r\n }\r\n\r\n this._runtimeAnimations.length = 0;\r\n\r\n this._raiseOnAnimationEnd();\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Wait asynchronously for the animation to end\r\n * @returns a promise which will be fulfilled when the animation ends\r\n */\r\n public waitAsync(): Promise {\r\n return new Promise((resolve) => {\r\n this.onAnimationEndObservable.add(\r\n () => {\r\n resolve(this);\r\n },\r\n undefined,\r\n undefined,\r\n this,\r\n true\r\n );\r\n });\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _animate(delay: number): boolean {\r\n if (this._paused) {\r\n this.animationStarted = false;\r\n if (this._pausedDelay === null) {\r\n this._pausedDelay = delay;\r\n }\r\n return true;\r\n }\r\n\r\n if (this._localDelayOffset === null) {\r\n this._localDelayOffset = delay;\r\n this._pausedDelay = null;\r\n } else if (this._pausedDelay !== null) {\r\n this._localDelayOffset += delay - this._pausedDelay;\r\n this._pausedDelay = null;\r\n }\r\n\r\n if (this._manualJumpDelay !== null) {\r\n this._localDelayOffset += this._manualJumpDelay;\r\n this._manualJumpDelay = null;\r\n this._frameToSyncFromJump = null;\r\n }\r\n\r\n this._goToFrame = null;\r\n\r\n if (this._weight === 0) {\r\n // We consider that an animation with a weight === 0 is \"actively\" paused\r\n return true;\r\n }\r\n\r\n // Animating\r\n let running = false;\r\n const runtimeAnimations = this._runtimeAnimations;\r\n let index: number;\r\n\r\n for (index = 0; index < runtimeAnimations.length; index++) {\r\n const animation = runtimeAnimations[index];\r\n const isRunning = animation.animate(delay - this._localDelayOffset, this.fromFrame, this.toFrame, this.loopAnimation, this._speedRatio, this._weight);\r\n running = running || isRunning;\r\n }\r\n\r\n this.animationStarted = running;\r\n\r\n if (!running) {\r\n if (this.disposeOnEnd) {\r\n // Remove from active animatables\r\n index = this._scene._activeAnimatables.indexOf(this);\r\n this._scene._activeAnimatables.splice(index, 1);\r\n\r\n // Dispose all runtime animations\r\n for (index = 0; index < runtimeAnimations.length; index++) {\r\n runtimeAnimations[index].dispose();\r\n }\r\n }\r\n\r\n this._raiseOnAnimationEnd();\r\n\r\n if (this.disposeOnEnd) {\r\n this.onAnimationEnd = null;\r\n this.onAnimationLoop = null;\r\n this.onAnimationLoopObservable.clear();\r\n this.onAnimationEndObservable.clear();\r\n }\r\n }\r\n\r\n return running;\r\n }\r\n}\r\n\r\ndeclare module \"../scene\" {\r\n export interface Scene {\r\n /** @internal */\r\n _registerTargetForLateAnimationBinding(runtimeAnimation: RuntimeAnimation, originalValue: any): void;\r\n\r\n /** @internal */\r\n _processLateAnimationBindingsForMatrices(holder: {\r\n totalWeight: number;\r\n totalAdditiveWeight: number;\r\n animations: RuntimeAnimation[];\r\n additiveAnimations: RuntimeAnimation[];\r\n originalValue: Matrix;\r\n }): any;\r\n\r\n /** @internal */\r\n _processLateAnimationBindingsForQuaternions(\r\n holder: {\r\n totalWeight: number;\r\n totalAdditiveWeight: number;\r\n animations: RuntimeAnimation[];\r\n additiveAnimations: RuntimeAnimation[];\r\n originalValue: Quaternion;\r\n },\r\n refQuaternion: Quaternion\r\n ): Quaternion;\r\n\r\n /** @internal */\r\n _processLateAnimationBindings(): void;\r\n\r\n /**\r\n * Sort active animatables based on their playOrder property\r\n */\r\n sortActiveAnimatables(): void;\r\n\r\n /**\r\n * Will start the animation sequence of a given target\r\n * @param target defines the target\r\n * @param from defines from which frame should animation start\r\n * @param to defines until which frame should animation run.\r\n * @param weight defines the weight to apply to the animation (1.0 by default)\r\n * @param loop defines if the animation loops\r\n * @param speedRatio defines the speed in which to run the animation (1.0 by default)\r\n * @param onAnimationEnd defines the function to be executed when the animation ends\r\n * @param animatable defines an animatable object. If not provided a new one will be created from the given params\r\n * @param targetMask defines if the target should be animated if animations are present (this is called recursively on descendant animatables regardless of return value)\r\n * @param onAnimationLoop defines the callback to call when an animation loops\r\n * @param isAdditive defines whether the animation should be evaluated additively (false by default)\r\n * @returns the animatable object created for this animation\r\n */\r\n beginWeightedAnimation(\r\n target: any,\r\n from: number,\r\n to: number,\r\n weight: number,\r\n loop?: boolean,\r\n speedRatio?: number,\r\n onAnimationEnd?: () => void,\r\n animatable?: Animatable,\r\n targetMask?: (target: any) => boolean,\r\n onAnimationLoop?: () => void,\r\n isAdditive?: boolean\r\n ): Animatable;\r\n\r\n /**\r\n * Will start the animation sequence of a given target\r\n * @param target defines the target\r\n * @param from defines from which frame should animation start\r\n * @param to defines until which frame should animation run.\r\n * @param loop defines if the animation loops\r\n * @param speedRatio defines the speed in which to run the animation (1.0 by default)\r\n * @param onAnimationEnd defines the function to be executed when the animation ends\r\n * @param animatable defines an animatable object. If not provided a new one will be created from the given params\r\n * @param stopCurrent defines if the current animations must be stopped first (true by default)\r\n * @param targetMask defines if the target should be animate if animations are present (this is called recursively on descendant animatables regardless of return value)\r\n * @param onAnimationLoop defines the callback to call when an animation loops\r\n * @param isAdditive defines whether the animation should be evaluated additively (false by default)\r\n * @returns the animatable object created for this animation\r\n */\r\n beginAnimation(\r\n target: any,\r\n from: number,\r\n to: number,\r\n loop?: boolean,\r\n speedRatio?: number,\r\n onAnimationEnd?: () => void,\r\n animatable?: Animatable,\r\n stopCurrent?: boolean,\r\n targetMask?: (target: any) => boolean,\r\n onAnimationLoop?: () => void,\r\n isAdditive?: boolean\r\n ): Animatable;\r\n\r\n /**\r\n * Will start the animation sequence of a given target and its hierarchy\r\n * @param target defines the target\r\n * @param directDescendantsOnly if true only direct descendants will be used, if false direct and also indirect (children of children, an so on in a recursive manner) descendants will be used.\r\n * @param from defines from which frame should animation start\r\n * @param to defines until which frame should animation run.\r\n * @param loop defines if the animation loops\r\n * @param speedRatio defines the speed in which to run the animation (1.0 by default)\r\n * @param onAnimationEnd defines the function to be executed when the animation ends\r\n * @param animatable defines an animatable object. If not provided a new one will be created from the given params\r\n * @param stopCurrent defines if the current animations must be stopped first (true by default)\r\n * @param targetMask defines if the target should be animated if animations are present (this is called recursively on descendant animatables regardless of return value)\r\n * @param onAnimationLoop defines the callback to call when an animation loops\r\n * @param isAdditive defines whether the animation should be evaluated additively (false by default)\r\n * @returns the list of created animatables\r\n */\r\n beginHierarchyAnimation(\r\n target: any,\r\n directDescendantsOnly: boolean,\r\n from: number,\r\n to: number,\r\n loop?: boolean,\r\n speedRatio?: number,\r\n onAnimationEnd?: () => void,\r\n animatable?: Animatable,\r\n stopCurrent?: boolean,\r\n targetMask?: (target: any) => boolean,\r\n onAnimationLoop?: () => void,\r\n isAdditive?: boolean\r\n ): Animatable[];\r\n\r\n /**\r\n * Begin a new animation on a given node\r\n * @param target defines the target where the animation will take place\r\n * @param animations defines the list of animations to start\r\n * @param from defines the initial value\r\n * @param to defines the final value\r\n * @param loop defines if you want animation to loop (off by default)\r\n * @param speedRatio defines the speed ratio to apply to all animations\r\n * @param onAnimationEnd defines the callback to call when an animation ends (will be called once per node)\r\n * @param onAnimationLoop defines the callback to call when an animation loops\r\n * @param isAdditive defines whether the animation should be evaluated additively (false by default)\r\n * @returns the list of created animatables\r\n */\r\n beginDirectAnimation(\r\n target: any,\r\n animations: Animation[],\r\n from: number,\r\n to: number,\r\n loop?: boolean,\r\n speedRatio?: number,\r\n onAnimationEnd?: () => void,\r\n onAnimationLoop?: () => void,\r\n isAdditive?: boolean\r\n ): Animatable;\r\n\r\n /**\r\n * Begin a new animation on a given node and its hierarchy\r\n * @param target defines the root node where the animation will take place\r\n * @param directDescendantsOnly if true only direct descendants will be used, if false direct and also indirect (children of children, an so on in a recursive manner) descendants will be used.\r\n * @param animations defines the list of animations to start\r\n * @param from defines the initial value\r\n * @param to defines the final value\r\n * @param loop defines if you want animation to loop (off by default)\r\n * @param speedRatio defines the speed ratio to apply to all animations\r\n * @param onAnimationEnd defines the callback to call when an animation ends (will be called once per node)\r\n * @param onAnimationLoop defines the callback to call when an animation loops\r\n * @param isAdditive defines whether the animation should be evaluated additively (false by default)\r\n * @returns the list of animatables created for all nodes\r\n */\r\n beginDirectHierarchyAnimation(\r\n target: Node,\r\n directDescendantsOnly: boolean,\r\n animations: Animation[],\r\n from: number,\r\n to: number,\r\n loop?: boolean,\r\n speedRatio?: number,\r\n onAnimationEnd?: () => void,\r\n onAnimationLoop?: () => void,\r\n isAdditive?: boolean\r\n ): Animatable[];\r\n\r\n /**\r\n * Gets the animatable associated with a specific target\r\n * @param target defines the target of the animatable\r\n * @returns the required animatable if found\r\n */\r\n getAnimatableByTarget(target: any): Nullable;\r\n\r\n /**\r\n * Gets all animatables associated with a given target\r\n * @param target defines the target to look animatables for\r\n * @returns an array of Animatables\r\n */\r\n getAllAnimatablesByTarget(target: any): Array;\r\n\r\n /**\r\n * Stops and removes all animations that have been applied to the scene\r\n */\r\n stopAllAnimations(): void;\r\n\r\n /**\r\n * Gets the current delta time used by animation engine\r\n */\r\n deltaTime: number;\r\n }\r\n}\r\n\r\nScene.prototype._animate = function (): void {\r\n if (!this.animationsEnabled) {\r\n return;\r\n }\r\n\r\n // Getting time\r\n const now = PrecisionDate.Now;\r\n if (!this._animationTimeLast) {\r\n if (this._pendingData.length > 0) {\r\n return;\r\n }\r\n this._animationTimeLast = now;\r\n }\r\n\r\n this.deltaTime = this.useConstantAnimationDeltaTime ? 16.0 : (now - this._animationTimeLast) * this.animationTimeScale;\r\n this._animationTimeLast = now;\r\n\r\n const animatables = this._activeAnimatables;\r\n if (animatables.length === 0) {\r\n return;\r\n }\r\n\r\n this._animationTime += this.deltaTime;\r\n const animationTime = this._animationTime;\r\n\r\n for (let index = 0; index < animatables.length; index++) {\r\n const animatable = animatables[index];\r\n\r\n if (!animatable._animate(animationTime) && animatable.disposeOnEnd) {\r\n index--; // Array was updated\r\n }\r\n }\r\n\r\n // Late animation bindings\r\n this._processLateAnimationBindings();\r\n};\r\n\r\nScene.prototype.sortActiveAnimatables = function (): void {\r\n this._activeAnimatables.sort((a, b) => {\r\n return a.playOrder - b.playOrder;\r\n });\r\n};\r\n\r\nScene.prototype.beginWeightedAnimation = function (\r\n target: any,\r\n from: number,\r\n to: number,\r\n weight = 1.0,\r\n loop?: boolean,\r\n speedRatio: number = 1.0,\r\n onAnimationEnd?: () => void,\r\n animatable?: Animatable,\r\n targetMask?: (target: any) => boolean,\r\n onAnimationLoop?: () => void,\r\n isAdditive = false\r\n): Animatable {\r\n const returnedAnimatable = this.beginAnimation(target, from, to, loop, speedRatio, onAnimationEnd, animatable, false, targetMask, onAnimationLoop, isAdditive);\r\n returnedAnimatable.weight = weight;\r\n\r\n return returnedAnimatable;\r\n};\r\n\r\nScene.prototype.beginAnimation = function (\r\n target: any,\r\n from: number,\r\n to: number,\r\n loop?: boolean,\r\n speedRatio: number = 1.0,\r\n onAnimationEnd?: () => void,\r\n animatable?: Animatable,\r\n stopCurrent = true,\r\n targetMask?: (target: any) => boolean,\r\n onAnimationLoop?: () => void,\r\n isAdditive = false\r\n): Animatable {\r\n if (from > to && speedRatio > 0) {\r\n speedRatio *= -1;\r\n }\r\n\r\n if (stopCurrent) {\r\n this.stopAnimation(target, undefined, targetMask);\r\n }\r\n\r\n if (!animatable) {\r\n animatable = new Animatable(this, target, from, to, loop, speedRatio, onAnimationEnd, undefined, onAnimationLoop, isAdditive);\r\n }\r\n\r\n const shouldRunTargetAnimations = targetMask ? targetMask(target) : true;\r\n // Local animations\r\n if (target.animations && shouldRunTargetAnimations) {\r\n animatable.appendAnimations(target, target.animations);\r\n }\r\n\r\n // Children animations\r\n if (target.getAnimatables) {\r\n const animatables = target.getAnimatables();\r\n for (let index = 0; index < animatables.length; index++) {\r\n this.beginAnimation(animatables[index], from, to, loop, speedRatio, onAnimationEnd, animatable, stopCurrent, targetMask, onAnimationLoop);\r\n }\r\n }\r\n\r\n animatable.reset();\r\n\r\n return animatable;\r\n};\r\n\r\nScene.prototype.beginHierarchyAnimation = function (\r\n target: any,\r\n directDescendantsOnly: boolean,\r\n from: number,\r\n to: number,\r\n loop?: boolean,\r\n speedRatio: number = 1.0,\r\n onAnimationEnd?: () => void,\r\n animatable?: Animatable,\r\n stopCurrent = true,\r\n targetMask?: (target: any) => boolean,\r\n onAnimationLoop?: () => void,\r\n isAdditive = false\r\n): Animatable[] {\r\n const children = target.getDescendants(directDescendantsOnly);\r\n\r\n const result = [];\r\n result.push(this.beginAnimation(target, from, to, loop, speedRatio, onAnimationEnd, animatable, stopCurrent, targetMask, undefined, isAdditive));\r\n for (const child of children) {\r\n result.push(this.beginAnimation(child, from, to, loop, speedRatio, onAnimationEnd, animatable, stopCurrent, targetMask, undefined, isAdditive));\r\n }\r\n\r\n return result;\r\n};\r\n\r\nScene.prototype.beginDirectAnimation = function (\r\n target: any,\r\n animations: Animation[],\r\n from: number,\r\n to: number,\r\n loop?: boolean,\r\n speedRatio?: number,\r\n onAnimationEnd?: () => void,\r\n onAnimationLoop?: () => void,\r\n isAdditive = false\r\n): Animatable {\r\n if (speedRatio === undefined) {\r\n speedRatio = 1.0;\r\n }\r\n\r\n if (from > to && speedRatio > 0) {\r\n speedRatio *= -1;\r\n } else if (to > from && speedRatio < 0) {\r\n const temp = to;\r\n to = from;\r\n from = temp;\r\n }\r\n\r\n const animatable = new Animatable(this, target, from, to, loop, speedRatio, onAnimationEnd, animations, onAnimationLoop, isAdditive);\r\n\r\n return animatable;\r\n};\r\n\r\nScene.prototype.beginDirectHierarchyAnimation = function (\r\n target: Node,\r\n directDescendantsOnly: boolean,\r\n animations: Animation[],\r\n from: number,\r\n to: number,\r\n loop?: boolean,\r\n speedRatio?: number,\r\n onAnimationEnd?: () => void,\r\n onAnimationLoop?: () => void,\r\n isAdditive = false\r\n): Animatable[] {\r\n const children = target.getDescendants(directDescendantsOnly);\r\n\r\n const result = [];\r\n result.push(this.beginDirectAnimation(target, animations, from, to, loop, speedRatio, onAnimationEnd, onAnimationLoop, isAdditive));\r\n for (const child of children) {\r\n result.push(this.beginDirectAnimation(child, animations, from, to, loop, speedRatio, onAnimationEnd, onAnimationLoop, isAdditive));\r\n }\r\n\r\n return result;\r\n};\r\n\r\nScene.prototype.getAnimatableByTarget = function (target: any): Nullable {\r\n for (let index = 0; index < this._activeAnimatables.length; index++) {\r\n if (this._activeAnimatables[index].target === target) {\r\n return this._activeAnimatables[index];\r\n }\r\n }\r\n\r\n return null;\r\n};\r\n\r\nScene.prototype.getAllAnimatablesByTarget = function (target: any): Array {\r\n const result = [];\r\n for (let index = 0; index < this._activeAnimatables.length; index++) {\r\n if (this._activeAnimatables[index].target === target) {\r\n result.push(this._activeAnimatables[index]);\r\n }\r\n }\r\n\r\n return result;\r\n};\r\n\r\n/**\r\n * Will stop the animation of the given target\r\n * @param target - the target\r\n * @param animationName - the name of the animation to stop (all animations will be stopped if both this and targetMask are empty)\r\n * @param targetMask - a function that determines if the animation should be stopped based on its target (all animations will be stopped if both this and animationName are empty)\r\n */\r\nScene.prototype.stopAnimation = function (target: any, animationName?: string, targetMask?: (target: any) => boolean): void {\r\n const animatables = this.getAllAnimatablesByTarget(target);\r\n\r\n for (const animatable of animatables) {\r\n animatable.stop(animationName, targetMask);\r\n }\r\n};\r\n\r\n/**\r\n * Stops and removes all animations that have been applied to the scene\r\n */\r\nScene.prototype.stopAllAnimations = function (): void {\r\n if (this._activeAnimatables) {\r\n for (let i = 0; i < this._activeAnimatables.length; i++) {\r\n this._activeAnimatables[i].stop(undefined, undefined, true);\r\n }\r\n this._activeAnimatables.length = 0;\r\n }\r\n\r\n for (const group of this.animationGroups) {\r\n group.stop();\r\n }\r\n};\r\n\r\nScene.prototype._registerTargetForLateAnimationBinding = function (runtimeAnimation: RuntimeAnimation, originalValue: any): void {\r\n const target = runtimeAnimation.target;\r\n this._registeredForLateAnimationBindings.pushNoDuplicate(target);\r\n\r\n if (!target._lateAnimationHolders) {\r\n target._lateAnimationHolders = {};\r\n }\r\n\r\n if (!target._lateAnimationHolders[runtimeAnimation.targetPath]) {\r\n target._lateAnimationHolders[runtimeAnimation.targetPath] = {\r\n totalWeight: 0,\r\n totalAdditiveWeight: 0,\r\n animations: [],\r\n additiveAnimations: [],\r\n originalValue: originalValue,\r\n };\r\n }\r\n\r\n if (runtimeAnimation.isAdditive) {\r\n target._lateAnimationHolders[runtimeAnimation.targetPath].additiveAnimations.push(runtimeAnimation);\r\n target._lateAnimationHolders[runtimeAnimation.targetPath].totalAdditiveWeight += runtimeAnimation.weight;\r\n } else {\r\n target._lateAnimationHolders[runtimeAnimation.targetPath].animations.push(runtimeAnimation);\r\n target._lateAnimationHolders[runtimeAnimation.targetPath].totalWeight += runtimeAnimation.weight;\r\n }\r\n};\r\n\r\nScene.prototype._processLateAnimationBindingsForMatrices = function (holder: {\r\n totalWeight: number;\r\n totalAdditiveWeight: number;\r\n animations: RuntimeAnimation[];\r\n additiveAnimations: RuntimeAnimation[];\r\n originalValue: Matrix;\r\n}): any {\r\n if (holder.totalWeight === 0 && holder.totalAdditiveWeight === 0) {\r\n return holder.originalValue;\r\n }\r\n\r\n let normalizer = 1.0;\r\n const finalPosition = TmpVectors.Vector3[0];\r\n const finalScaling = TmpVectors.Vector3[1];\r\n const finalQuaternion = TmpVectors.Quaternion[0];\r\n let startIndex = 0;\r\n const originalAnimation = holder.animations[0];\r\n const originalValue = holder.originalValue;\r\n\r\n let scale = 1;\r\n let skipOverride = false;\r\n if (holder.totalWeight < 1.0) {\r\n // We need to mix the original value in\r\n scale = 1.0 - holder.totalWeight;\r\n originalValue.decompose(finalScaling, finalQuaternion, finalPosition);\r\n } else {\r\n startIndex = 1;\r\n // We need to normalize the weights\r\n normalizer = holder.totalWeight;\r\n scale = originalAnimation.weight / normalizer;\r\n if (scale == 1) {\r\n if (holder.totalAdditiveWeight) {\r\n skipOverride = true;\r\n } else {\r\n return originalAnimation.currentValue;\r\n }\r\n }\r\n\r\n originalAnimation.currentValue.decompose(finalScaling, finalQuaternion, finalPosition);\r\n }\r\n\r\n // Add up the override animations\r\n if (!skipOverride) {\r\n finalScaling.scaleInPlace(scale);\r\n finalPosition.scaleInPlace(scale);\r\n finalQuaternion.scaleInPlace(scale);\r\n\r\n for (let animIndex = startIndex; animIndex < holder.animations.length; animIndex++) {\r\n const runtimeAnimation = holder.animations[animIndex];\r\n if (runtimeAnimation.weight === 0) {\r\n continue;\r\n }\r\n\r\n scale = runtimeAnimation.weight / normalizer;\r\n const currentPosition = TmpVectors.Vector3[2];\r\n const currentScaling = TmpVectors.Vector3[3];\r\n const currentQuaternion = TmpVectors.Quaternion[1];\r\n\r\n runtimeAnimation.currentValue.decompose(currentScaling, currentQuaternion, currentPosition);\r\n\r\n currentScaling.scaleAndAddToRef(scale, finalScaling);\r\n currentQuaternion.scaleAndAddToRef(Quaternion.Dot(finalQuaternion, currentQuaternion) > 0 ? scale : -scale, finalQuaternion);\r\n currentPosition.scaleAndAddToRef(scale, finalPosition);\r\n }\r\n\r\n finalQuaternion.normalize();\r\n }\r\n\r\n // Add up the additive animations\r\n for (let animIndex = 0; animIndex < holder.additiveAnimations.length; animIndex++) {\r\n const runtimeAnimation = holder.additiveAnimations[animIndex];\r\n if (runtimeAnimation.weight === 0) {\r\n continue;\r\n }\r\n\r\n const currentPosition = TmpVectors.Vector3[2];\r\n const currentScaling = TmpVectors.Vector3[3];\r\n const currentQuaternion = TmpVectors.Quaternion[1];\r\n\r\n runtimeAnimation.currentValue.decompose(currentScaling, currentQuaternion, currentPosition);\r\n currentScaling.multiplyToRef(finalScaling, currentScaling);\r\n Vector3.LerpToRef(finalScaling, currentScaling, runtimeAnimation.weight, finalScaling);\r\n finalQuaternion.multiplyToRef(currentQuaternion, currentQuaternion);\r\n Quaternion.SlerpToRef(finalQuaternion, currentQuaternion, runtimeAnimation.weight, finalQuaternion);\r\n currentPosition.scaleAndAddToRef(runtimeAnimation.weight, finalPosition);\r\n }\r\n\r\n const workValue = originalAnimation ? originalAnimation._animationState.workValue : TmpVectors.Matrix[0].clone();\r\n Matrix.ComposeToRef(finalScaling, finalQuaternion, finalPosition, workValue);\r\n return workValue;\r\n};\r\n\r\nScene.prototype._processLateAnimationBindingsForQuaternions = function (\r\n holder: {\r\n totalWeight: number;\r\n totalAdditiveWeight: number;\r\n animations: RuntimeAnimation[];\r\n additiveAnimations: RuntimeAnimation[];\r\n originalValue: Quaternion;\r\n },\r\n refQuaternion: Quaternion\r\n): Quaternion {\r\n if (holder.totalWeight === 0 && holder.totalAdditiveWeight === 0) {\r\n return refQuaternion;\r\n }\r\n\r\n const originalAnimation = holder.animations[0];\r\n const originalValue = holder.originalValue;\r\n let cumulativeQuaternion = refQuaternion;\r\n\r\n if (holder.totalWeight === 0 && holder.totalAdditiveWeight > 0) {\r\n cumulativeQuaternion.copyFrom(originalValue);\r\n } else if (holder.animations.length === 1) {\r\n Quaternion.SlerpToRef(originalValue, originalAnimation.currentValue, Math.min(1.0, holder.totalWeight), cumulativeQuaternion);\r\n\r\n if (holder.totalAdditiveWeight === 0) {\r\n return cumulativeQuaternion;\r\n }\r\n } else if (holder.animations.length > 1) {\r\n // Add up the override animations\r\n let normalizer = 1.0;\r\n let quaternions: Array;\r\n let weights: Array;\r\n\r\n if (holder.totalWeight < 1.0) {\r\n const scale = 1.0 - holder.totalWeight;\r\n\r\n quaternions = [];\r\n weights = [];\r\n\r\n quaternions.push(originalValue);\r\n weights.push(scale);\r\n } else {\r\n if (holder.animations.length === 2) {\r\n // Slerp as soon as we can\r\n Quaternion.SlerpToRef(holder.animations[0].currentValue, holder.animations[1].currentValue, holder.animations[1].weight / holder.totalWeight, refQuaternion);\r\n\r\n if (holder.totalAdditiveWeight === 0) {\r\n return refQuaternion;\r\n }\r\n }\r\n\r\n quaternions = [];\r\n weights = [];\r\n normalizer = holder.totalWeight;\r\n }\r\n\r\n for (let animIndex = 0; animIndex < holder.animations.length; animIndex++) {\r\n const runtimeAnimation = holder.animations[animIndex];\r\n quaternions.push(runtimeAnimation.currentValue);\r\n weights.push(runtimeAnimation.weight / normalizer);\r\n }\r\n\r\n // https://gamedev.stackexchange.com/questions/62354/method-for-interpolation-between-3-quaternions\r\n\r\n let cumulativeAmount = 0;\r\n for (let index = 0; index < quaternions.length; ) {\r\n if (!index) {\r\n Quaternion.SlerpToRef(quaternions[index], quaternions[index + 1], weights[index + 1] / (weights[index] + weights[index + 1]), refQuaternion);\r\n cumulativeQuaternion = refQuaternion;\r\n cumulativeAmount = weights[index] + weights[index + 1];\r\n index += 2;\r\n continue;\r\n }\r\n cumulativeAmount += weights[index];\r\n Quaternion.SlerpToRef(cumulativeQuaternion, quaternions[index], weights[index] / cumulativeAmount, cumulativeQuaternion);\r\n index++;\r\n }\r\n }\r\n\r\n // Add up the additive animations\r\n for (let animIndex = 0; animIndex < holder.additiveAnimations.length; animIndex++) {\r\n const runtimeAnimation = holder.additiveAnimations[animIndex];\r\n if (runtimeAnimation.weight === 0) {\r\n continue;\r\n }\r\n\r\n cumulativeQuaternion.multiplyToRef(runtimeAnimation.currentValue, TmpVectors.Quaternion[0]);\r\n Quaternion.SlerpToRef(cumulativeQuaternion, TmpVectors.Quaternion[0], runtimeAnimation.weight, cumulativeQuaternion);\r\n }\r\n\r\n return cumulativeQuaternion!;\r\n};\r\n\r\nScene.prototype._processLateAnimationBindings = function (): void {\r\n if (!this._registeredForLateAnimationBindings.length) {\r\n return;\r\n }\r\n for (let index = 0; index < this._registeredForLateAnimationBindings.length; index++) {\r\n const target = this._registeredForLateAnimationBindings.data[index];\r\n\r\n for (const path in target._lateAnimationHolders) {\r\n const holder = target._lateAnimationHolders[path];\r\n const originalAnimation: RuntimeAnimation = holder.animations[0];\r\n const originalValue = holder.originalValue;\r\n if (originalValue === undefined || originalValue === null) {\r\n continue;\r\n }\r\n const matrixDecomposeMode = Animation.AllowMatrixDecomposeForInterpolation && originalValue.m; // ie. data is matrix\r\n\r\n let finalValue: any = target[path];\r\n if (matrixDecomposeMode) {\r\n finalValue = this._processLateAnimationBindingsForMatrices(holder);\r\n } else {\r\n const quaternionMode = originalValue.w !== undefined;\r\n if (quaternionMode) {\r\n finalValue = this._processLateAnimationBindingsForQuaternions(holder, finalValue || Quaternion.Identity());\r\n } else {\r\n let startIndex = 0;\r\n let normalizer = 1.0;\r\n\r\n if (holder.totalWeight < 1.0) {\r\n // We need to mix the original value in\r\n if (originalAnimation && originalValue.scale) {\r\n finalValue = originalValue.scale(1.0 - holder.totalWeight);\r\n } else if (originalAnimation) {\r\n finalValue = originalValue * (1.0 - holder.totalWeight);\r\n } else if (originalValue.clone) {\r\n finalValue = originalValue.clone();\r\n } else {\r\n finalValue = originalValue;\r\n }\r\n } else if (originalAnimation) {\r\n // We need to normalize the weights\r\n normalizer = holder.totalWeight;\r\n const scale = originalAnimation.weight / normalizer;\r\n if (scale !== 1) {\r\n if (originalAnimation.currentValue.scale) {\r\n finalValue = originalAnimation.currentValue.scale(scale);\r\n } else {\r\n finalValue = originalAnimation.currentValue * scale;\r\n }\r\n } else {\r\n finalValue = originalAnimation.currentValue;\r\n }\r\n\r\n startIndex = 1;\r\n }\r\n\r\n // Add up the override animations\r\n for (let animIndex = startIndex; animIndex < holder.animations.length; animIndex++) {\r\n const runtimeAnimation = holder.animations[animIndex];\r\n const scale = runtimeAnimation.weight / normalizer;\r\n\r\n if (!scale) {\r\n continue;\r\n } else if (runtimeAnimation.currentValue.scaleAndAddToRef) {\r\n runtimeAnimation.currentValue.scaleAndAddToRef(scale, finalValue);\r\n } else {\r\n finalValue += runtimeAnimation.currentValue * scale;\r\n }\r\n }\r\n\r\n // Add up the additive animations\r\n for (let animIndex = 0; animIndex < holder.additiveAnimations.length; animIndex++) {\r\n const runtimeAnimation = holder.additiveAnimations[animIndex];\r\n const scale: number = runtimeAnimation.weight;\r\n\r\n if (!scale) {\r\n continue;\r\n } else if (runtimeAnimation.currentValue.scaleAndAddToRef) {\r\n runtimeAnimation.currentValue.scaleAndAddToRef(scale, finalValue);\r\n } else {\r\n finalValue += runtimeAnimation.currentValue * scale;\r\n }\r\n }\r\n }\r\n }\r\n target[path] = finalValue;\r\n }\r\n\r\n target._lateAnimationHolders = {};\r\n }\r\n this._registeredForLateAnimationBindings.reset();\r\n};\r\n\r\ndeclare module \"../Bones/bone\" {\r\n export interface Bone {\r\n /**\r\n * Copy an animation range from another bone\r\n * @param source defines the source bone\r\n * @param rangeName defines the range name to copy\r\n * @param frameOffset defines the frame offset\r\n * @param rescaleAsRequired defines if rescaling must be applied if required\r\n * @param skelDimensionsRatio defines the scaling ratio\r\n * @returns true if operation was successful\r\n */\r\n copyAnimationRange(source: Bone, rangeName: string, frameOffset: number, rescaleAsRequired: boolean, skelDimensionsRatio: Nullable): boolean;\r\n }\r\n}\r\n\r\nBone.prototype.copyAnimationRange = function (\r\n source: Bone,\r\n rangeName: string,\r\n frameOffset: number,\r\n rescaleAsRequired = false,\r\n skelDimensionsRatio: Nullable = null\r\n): boolean {\r\n // all animation may be coming from a library skeleton, so may need to create animation\r\n if (this.animations.length === 0) {\r\n this.animations.push(new Animation(this.name, \"_matrix\", source.animations[0].framePerSecond, Animation.ANIMATIONTYPE_MATRIX, 0));\r\n this.animations[0].setKeys([]);\r\n }\r\n\r\n // get animation info / verify there is such a range from the source bone\r\n const sourceRange = source.animations[0].getRange(rangeName);\r\n if (!sourceRange) {\r\n return false;\r\n }\r\n const from = sourceRange.from;\r\n const to = sourceRange.to;\r\n const sourceKeys = source.animations[0].getKeys();\r\n\r\n // rescaling prep\r\n const sourceBoneLength = source.length;\r\n const sourceParent = source.getParent();\r\n const parent = this.getParent();\r\n const parentScalingReqd = rescaleAsRequired && sourceParent && sourceBoneLength && this.length && sourceBoneLength !== this.length;\r\n const parentRatio = parentScalingReqd && parent && sourceParent ? parent.length / sourceParent.length : 1;\r\n\r\n const dimensionsScalingReqd =\r\n rescaleAsRequired && !parent && skelDimensionsRatio && (skelDimensionsRatio.x !== 1 || skelDimensionsRatio.y !== 1 || skelDimensionsRatio.z !== 1);\r\n\r\n const destKeys = this.animations[0].getKeys();\r\n\r\n // loop vars declaration\r\n let orig: { frame: number; value: Matrix };\r\n let origTranslation: Vector3;\r\n let mat: Matrix;\r\n\r\n for (let key = 0, nKeys = sourceKeys.length; key < nKeys; key++) {\r\n orig = sourceKeys[key];\r\n if (orig.frame >= from && orig.frame <= to) {\r\n if (rescaleAsRequired) {\r\n mat = orig.value.clone();\r\n\r\n // scale based on parent ratio, when bone has parent\r\n if (parentScalingReqd) {\r\n origTranslation = mat.getTranslation();\r\n mat.setTranslation(origTranslation.scaleInPlace(parentRatio));\r\n\r\n // scale based on skeleton dimension ratio when root bone, and value is passed\r\n } else if (dimensionsScalingReqd && skelDimensionsRatio) {\r\n origTranslation = mat.getTranslation();\r\n mat.setTranslation(origTranslation.multiplyInPlace(skelDimensionsRatio));\r\n\r\n // use original when root bone, and no data for skelDimensionsRatio\r\n } else {\r\n mat = orig.value;\r\n }\r\n } else {\r\n mat = orig.value;\r\n }\r\n destKeys.push({ frame: orig.frame + frameOffset, value: mat });\r\n }\r\n }\r\n this.animations[0].createRange(rangeName, from + frameOffset, to + frameOffset);\r\n return true;\r\n};\r\n","import type { Nullable } from \"../types\";\r\nimport type { WebXRLayerRenderTargetTextureProvider } from \"./webXRRenderTargetTextureProvider\";\r\nimport type { WebXRSessionManager } from \"./webXRSessionManager\";\r\n\r\n/** Covers all supported subclasses of WebXR's XRCompositionLayer */\r\n// TODO (rgerd): Extend for all other subclasses of XRCompositionLayer.\r\nexport type WebXRCompositionLayerType = \"XRProjectionLayer\";\r\n\r\n/** Covers all supported subclasses of WebXR's XRLayer */\r\nexport type WebXRLayerType = \"XRWebGLLayer\" | WebXRCompositionLayerType;\r\n\r\n/**\r\n * Wrapper over subclasses of XRLayer.\r\n * @internal\r\n */\r\nexport class WebXRLayerWrapper {\r\n /**\r\n * Check if fixed foveation is supported on this device\r\n */\r\n public get isFixedFoveationSupported(): boolean {\r\n return this.layerType == \"XRWebGLLayer\" && typeof (this.layer as XRWebGLLayer).fixedFoveation == \"number\";\r\n }\r\n\r\n /**\r\n * Get the fixed foveation currently set, as specified by the webxr specs\r\n * If this returns null, then fixed foveation is not supported\r\n */\r\n public get fixedFoveation(): Nullable {\r\n if (this.isFixedFoveationSupported) {\r\n return (this.layer as XRWebGLLayer).fixedFoveation!;\r\n }\r\n return null;\r\n }\r\n\r\n /**\r\n * Set the fixed foveation to the specified value, as specified by the webxr specs\r\n * This value will be normalized to be between 0 and 1, 1 being max foveation, 0 being no foveation\r\n */\r\n public set fixedFoveation(value: Nullable) {\r\n if (this.isFixedFoveationSupported) {\r\n const val = Math.max(0, Math.min(1, value || 0));\r\n (this.layer as XRWebGLLayer).fixedFoveation = val;\r\n }\r\n }\r\n\r\n protected constructor(\r\n /** The width of the layer's framebuffer. */\r\n public getWidth: () => number,\r\n /** The height of the layer's framebuffer. */\r\n public getHeight: () => number,\r\n /** The XR layer that this WebXRLayerWrapper wraps. */\r\n public readonly layer: XRLayer,\r\n /** The type of XR layer that is being wrapped. */\r\n public readonly layerType: WebXRLayerType,\r\n /** Create a render target provider for the wrapped layer. */\r\n public createRenderTargetTextureProvider: (xrSessionManager: WebXRSessionManager) => WebXRLayerRenderTargetTextureProvider\r\n ) {}\r\n}\r\n","import type { Engine } from \"../Engines/engine\";\r\nimport { WebGLHardwareTexture } from \"../Engines/WebGL/webGLHardwareTexture\";\r\nimport type { WebGLRenderTargetWrapper } from \"../Engines/WebGL/webGLRenderTargetWrapper\";\r\nimport { InternalTexture, InternalTextureSource } from \"../Materials/Textures/internalTexture\";\r\nimport { MultiviewRenderTarget } from \"../Materials/Textures/MultiviewRenderTarget\";\r\nimport { RenderTargetTexture } from \"../Materials/Textures/renderTargetTexture\";\r\nimport type { Viewport } from \"../Maths/math.viewport\";\r\nimport type { IDisposable, Scene } from \"../scene\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { WebXRLayerWrapper } from \"./webXRLayerWrapper\";\r\n\r\n/**\r\n * An interface for objects that provide render target textures for XR rendering.\r\n */\r\nexport interface IWebXRRenderTargetTextureProvider extends IDisposable {\r\n /**\r\n * Attempts to set the framebuffer-size-normalized viewport to be rendered this frame for this view.\r\n * In the event of a failure, the supplied viewport is not updated.\r\n * @param viewport the viewport to which the view will be rendered\r\n * @param view the view for which to set the viewport\r\n * @returns whether the operation was successful\r\n */\r\n trySetViewportForView(viewport: Viewport, view: XRView): boolean;\r\n /**\r\n * Gets the correct render target texture to be rendered this frame for this eye\r\n * @param eye the eye for which to get the render target\r\n * @returns the render target for the specified eye or null if not available\r\n */\r\n getRenderTargetTextureForEye(eye: XREye): Nullable;\r\n /**\r\n * Gets the correct render target texture to be rendered this frame for this view\r\n * @param view the view for which to get the render target\r\n * @returns the render target for the specified view or null if not available\r\n */\r\n getRenderTargetTextureForView(view: XRView): Nullable;\r\n}\r\n\r\n/**\r\n * Provides render target textures and other important rendering information for a given XRLayer.\r\n * @internal\r\n */\r\nexport abstract class WebXRLayerRenderTargetTextureProvider implements IWebXRRenderTargetTextureProvider {\r\n public abstract trySetViewportForView(viewport: Viewport, view: XRView): boolean;\r\n public abstract getRenderTargetTextureForEye(eye: XREye): Nullable;\r\n public abstract getRenderTargetTextureForView(view: XRView): Nullable;\r\n\r\n protected _renderTargetTextures = new Array();\r\n protected _framebufferDimensions: Nullable<{ framebufferWidth: number; framebufferHeight: number }>;\r\n\r\n private _engine: Engine;\r\n\r\n constructor(private readonly _scene: Scene, public readonly layerWrapper: WebXRLayerWrapper) {\r\n this._engine = _scene.getEngine();\r\n }\r\n\r\n private _createInternalTexture(textureSize: { width: number; height: number }, texture: WebGLTexture): InternalTexture {\r\n const internalTexture = new InternalTexture(this._engine, InternalTextureSource.Unknown, true);\r\n internalTexture.width = textureSize.width;\r\n internalTexture.height = textureSize.height;\r\n internalTexture._hardwareTexture = new WebGLHardwareTexture(texture, this._engine._gl);\r\n internalTexture.isReady = true;\r\n return internalTexture;\r\n }\r\n\r\n protected _createRenderTargetTexture(\r\n width: number,\r\n height: number,\r\n framebuffer: Nullable,\r\n colorTexture?: WebGLTexture,\r\n depthStencilTexture?: WebGLTexture,\r\n multiview?: boolean\r\n ): RenderTargetTexture {\r\n if (!this._engine) {\r\n throw new Error(\"Engine is disposed\");\r\n }\r\n\r\n const textureSize = { width, height };\r\n\r\n // Create render target texture from the internal texture\r\n const renderTargetTexture = multiview ? new MultiviewRenderTarget(this._scene, textureSize) : new RenderTargetTexture(\"XR renderTargetTexture\", textureSize, this._scene);\r\n const renderTargetWrapper = renderTargetTexture.renderTarget as WebGLRenderTargetWrapper;\r\n renderTargetWrapper._samples = renderTargetTexture.samples;\r\n // Set the framebuffer, make sure it works in all scenarios - emulator, no layers and layers\r\n if (framebuffer || !colorTexture) {\r\n renderTargetWrapper._framebuffer = framebuffer;\r\n }\r\n\r\n // Create internal texture\r\n if (colorTexture) {\r\n if (multiview) {\r\n renderTargetWrapper._colorTextureArray = colorTexture;\r\n } else {\r\n const internalTexture = this._createInternalTexture(textureSize, colorTexture);\r\n renderTargetWrapper.setTexture(internalTexture, 0);\r\n renderTargetTexture._texture = internalTexture;\r\n }\r\n }\r\n\r\n if (depthStencilTexture) {\r\n if (multiview) {\r\n renderTargetWrapper._depthStencilTextureArray = depthStencilTexture;\r\n } else {\r\n renderTargetWrapper._depthStencilTexture = this._createInternalTexture(textureSize, depthStencilTexture);\r\n }\r\n }\r\n\r\n renderTargetTexture.disableRescaling();\r\n // Firefox reality fails if skipInitialClear is set to true, so make sure only modern XR implementations set it.\r\n if (typeof XRWebGLBinding !== \"undefined\") {\r\n // WebXR pre-clears textures\r\n renderTargetTexture.skipInitialClear = true;\r\n }\r\n\r\n this._renderTargetTextures.push(renderTargetTexture);\r\n\r\n return renderTargetTexture;\r\n }\r\n\r\n protected _destroyRenderTargetTexture(renderTargetTexture: RenderTargetTexture) {\r\n this._renderTargetTextures.splice(this._renderTargetTextures.indexOf(renderTargetTexture), 1);\r\n renderTargetTexture.dispose();\r\n }\r\n\r\n public getFramebufferDimensions(): Nullable<{ framebufferWidth: number; framebufferHeight: number }> {\r\n return this._framebufferDimensions;\r\n }\r\n\r\n public dispose() {\r\n this._renderTargetTextures.forEach((rtt) => rtt.dispose());\r\n this._renderTargetTextures.length = 0;\r\n }\r\n}\r\n","import type { RenderTargetTexture } from \"../Materials/Textures/renderTargetTexture\";\r\nimport type { Viewport } from \"../Maths/math.viewport\";\r\nimport type { Scene } from \"../scene\";\r\nimport type { Nullable } from \"../types\";\r\nimport { WebXRLayerWrapper } from \"./webXRLayerWrapper\";\r\nimport { WebXRLayerRenderTargetTextureProvider } from \"./webXRRenderTargetTextureProvider\";\r\n\r\n/**\r\n * Wraps xr webgl layers.\r\n * @internal\r\n */\r\nexport class WebXRWebGLLayerWrapper extends WebXRLayerWrapper {\r\n /**\r\n * @param layer is the layer to be wrapped.\r\n * @returns a new WebXRLayerWrapper wrapping the provided XRWebGLLayer.\r\n */\r\n constructor(public readonly layer: XRWebGLLayer) {\r\n super(\r\n () => layer.framebufferWidth,\r\n () => layer.framebufferHeight,\r\n layer,\r\n \"XRWebGLLayer\",\r\n (sessionManager) => new WebXRWebGLLayerRenderTargetTextureProvider(sessionManager.scene, this)\r\n );\r\n }\r\n}\r\n\r\n/**\r\n * Provides render target textures and other important rendering information for a given XRWebGLLayer.\r\n * @internal\r\n */\r\nexport class WebXRWebGLLayerRenderTargetTextureProvider extends WebXRLayerRenderTargetTextureProvider {\r\n // The dimensions will always be defined in this class.\r\n protected _framebufferDimensions: { framebufferWidth: number; framebufferHeight: number };\r\n private _rtt: Nullable;\r\n private _framebuffer: WebGLFramebuffer;\r\n private _layer: XRWebGLLayer;\r\n\r\n constructor(scene: Scene, public readonly layerWrapper: WebXRWebGLLayerWrapper) {\r\n super(scene, layerWrapper);\r\n this._layer = layerWrapper.layer;\r\n this._framebufferDimensions = {\r\n framebufferWidth: this._layer.framebufferWidth,\r\n framebufferHeight: this._layer.framebufferHeight,\r\n };\r\n }\r\n\r\n public trySetViewportForView(viewport: Viewport, view: XRView): boolean {\r\n const xrViewport = this._layer.getViewport(view);\r\n if (!xrViewport) {\r\n return false;\r\n }\r\n const framebufferWidth = this._framebufferDimensions.framebufferWidth;\r\n const framebufferHeight = this._framebufferDimensions.framebufferHeight;\r\n viewport.x = xrViewport.x / framebufferWidth;\r\n viewport.y = xrViewport.y / framebufferHeight;\r\n viewport.width = xrViewport.width / framebufferWidth;\r\n viewport.height = xrViewport.height / framebufferHeight;\r\n return true;\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public getRenderTargetTextureForEye(eye: XREye): Nullable {\r\n const layerWidth = this._layer.framebufferWidth;\r\n const layerHeight = this._layer.framebufferHeight;\r\n const framebuffer = this._layer.framebuffer;\r\n\r\n if (\r\n !this._rtt ||\r\n layerWidth !== this._framebufferDimensions.framebufferWidth ||\r\n layerHeight !== this._framebufferDimensions.framebufferHeight ||\r\n framebuffer !== this._framebuffer\r\n ) {\r\n this._rtt = this._createRenderTargetTexture(layerWidth, layerHeight, framebuffer);\r\n this._framebufferDimensions.framebufferWidth = layerWidth;\r\n this._framebufferDimensions.framebufferHeight = layerHeight;\r\n this._framebuffer = framebuffer;\r\n }\r\n\r\n return this._rtt;\r\n }\r\n\r\n public getRenderTargetTextureForView(view: XRView): Nullable {\r\n return this.getRenderTargetTextureForEye(view.eye);\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\nimport type { ThinEngine } from \"../Engines/thinEngine\";\r\nimport type { WebXRRenderTarget } from \"./webXRTypes\";\r\nimport type { WebXRSessionManager } from \"./webXRSessionManager\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport { Tools } from \"../Misc/tools\";\r\nimport type { WebXRLayerWrapper } from \"./webXRLayerWrapper\";\r\nimport { WebXRWebGLLayerWrapper } from \"./webXRWebGLLayer\";\r\n\r\n/**\r\n * Configuration object for WebXR output canvas\r\n */\r\nexport class WebXRManagedOutputCanvasOptions {\r\n /**\r\n * An optional canvas in case you wish to create it yourself and provide it here.\r\n * If not provided, a new canvas will be created\r\n */\r\n public canvasElement?: HTMLCanvasElement;\r\n /**\r\n * Options for this XR Layer output\r\n */\r\n public canvasOptions?: XRWebGLLayerInit;\r\n /**\r\n * CSS styling for a newly created canvas (if not provided)\r\n */\r\n public newCanvasCssStyle?: string;\r\n\r\n /**\r\n * Get the default values of the configuration object\r\n * @param engine defines the engine to use (can be null)\r\n * @returns default values of this configuration object\r\n */\r\n public static GetDefaults(engine?: ThinEngine): WebXRManagedOutputCanvasOptions {\r\n const defaults = new WebXRManagedOutputCanvasOptions();\r\n defaults.canvasOptions = {\r\n antialias: true,\r\n depth: true,\r\n stencil: engine ? engine.isStencilEnable : true,\r\n alpha: true,\r\n framebufferScaleFactor: 1,\r\n };\r\n\r\n defaults.newCanvasCssStyle = \"position:absolute; bottom:0px;right:0px;z-index:10;width:90%;height:100%;background-color: #000000;\";\r\n\r\n return defaults;\r\n }\r\n}\r\n/**\r\n * Creates a canvas that is added/removed from the webpage when entering/exiting XR\r\n */\r\nexport class WebXRManagedOutputCanvas implements WebXRRenderTarget {\r\n private _canvas: Nullable = null;\r\n private _engine: Nullable = null;\r\n private _originalCanvasSize: {\r\n width: number;\r\n height: number;\r\n };\r\n\r\n /**\r\n * Rendering context of the canvas which can be used to display/mirror xr content\r\n */\r\n public canvasContext: WebGLRenderingContext;\r\n\r\n /**\r\n * xr layer for the canvas\r\n */\r\n public xrLayer: Nullable = null;\r\n\r\n private _xrLayerWrapper: Nullable = null;\r\n\r\n /**\r\n * Observers registered here will be triggered when the xr layer was initialized\r\n */\r\n public onXRLayerInitObservable: Observable = new Observable();\r\n\r\n /**\r\n * Initializes the canvas to be added/removed upon entering/exiting xr\r\n * @param _xrSessionManager The XR Session manager\r\n * @param _options optional configuration for this canvas output. defaults will be used if not provided\r\n */\r\n constructor(_xrSessionManager: WebXRSessionManager, private _options: WebXRManagedOutputCanvasOptions = WebXRManagedOutputCanvasOptions.GetDefaults()) {\r\n this._engine = _xrSessionManager.scene.getEngine();\r\n this._engine.onDisposeObservable.addOnce(() => {\r\n this._engine = null;\r\n });\r\n\r\n if (!_options.canvasElement) {\r\n const canvas = document.createElement(\"canvas\");\r\n canvas.style.cssText = this._options.newCanvasCssStyle || \"position:absolute; bottom:0px;right:0px;\";\r\n this._setManagedOutputCanvas(canvas);\r\n } else {\r\n this._setManagedOutputCanvas(_options.canvasElement);\r\n }\r\n\r\n _xrSessionManager.onXRSessionInit.add(() => {\r\n this._addCanvas();\r\n });\r\n\r\n _xrSessionManager.onXRSessionEnded.add(() => {\r\n this._removeCanvas();\r\n });\r\n }\r\n\r\n /**\r\n * Disposes of the object\r\n */\r\n public dispose() {\r\n this._removeCanvas();\r\n this._setManagedOutputCanvas(null);\r\n }\r\n\r\n /**\r\n * Initializes a XRWebGLLayer to be used as the session's baseLayer.\r\n * @param xrSession xr session\r\n * @returns a promise that will resolve once the XR Layer has been created\r\n */\r\n public async initializeXRLayerAsync(xrSession: XRSession): Promise {\r\n const createLayer = () => {\r\n this.xrLayer = new XRWebGLLayer(xrSession, this.canvasContext, this._options.canvasOptions);\r\n this._xrLayerWrapper = new WebXRWebGLLayerWrapper(this.xrLayer);\r\n this.onXRLayerInitObservable.notifyObservers(this.xrLayer);\r\n return this.xrLayer;\r\n };\r\n\r\n // support canvases without makeXRCompatible\r\n if (!(this.canvasContext as any).makeXRCompatible) {\r\n return Promise.resolve(createLayer());\r\n }\r\n\r\n return (this.canvasContext as any)\r\n .makeXRCompatible()\r\n .then(\r\n // catch any error and continue. When using the emulator is throws this error for no apparent reason.\r\n () => {},\r\n () => {\r\n // log the error, continue nonetheless!\r\n Tools.Warn(\"Error executing makeXRCompatible. This does not mean that the session will work incorrectly.\");\r\n }\r\n )\r\n .then(() => {\r\n return createLayer();\r\n });\r\n }\r\n\r\n private _addCanvas() {\r\n if (this._canvas && this._engine && this._canvas !== this._engine.getRenderingCanvas()) {\r\n document.body.appendChild(this._canvas);\r\n }\r\n if (this.xrLayer) {\r\n this._setCanvasSize(true);\r\n } else {\r\n this.onXRLayerInitObservable.addOnce(() => {\r\n this._setCanvasSize(true);\r\n });\r\n }\r\n }\r\n\r\n private _removeCanvas() {\r\n if (this._canvas && this._engine && document.body.contains(this._canvas) && this._canvas !== this._engine.getRenderingCanvas()) {\r\n document.body.removeChild(this._canvas);\r\n }\r\n this._setCanvasSize(false);\r\n }\r\n\r\n private _setCanvasSize(init: boolean = true, xrLayer = this._xrLayerWrapper) {\r\n if (!this._canvas || !this._engine) {\r\n return;\r\n }\r\n if (init) {\r\n if (xrLayer) {\r\n if (this._canvas !== this._engine.getRenderingCanvas()) {\r\n this._canvas.style.width = xrLayer.getWidth() + \"px\";\r\n this._canvas.style.height = xrLayer.getHeight() + \"px\";\r\n } else {\r\n this._engine.setSize(xrLayer.getWidth(), xrLayer.getHeight());\r\n }\r\n }\r\n } else {\r\n if (this._originalCanvasSize) {\r\n if (this._canvas !== this._engine.getRenderingCanvas()) {\r\n this._canvas.style.width = this._originalCanvasSize.width + \"px\";\r\n this._canvas.style.height = this._originalCanvasSize.height + \"px\";\r\n } else {\r\n this._engine.setSize(this._originalCanvasSize.width, this._originalCanvasSize.height);\r\n }\r\n }\r\n }\r\n }\r\n\r\n private _setManagedOutputCanvas(canvas: Nullable) {\r\n this._removeCanvas();\r\n if (!canvas) {\r\n this._canvas = null;\r\n (this.canvasContext as any) = null;\r\n } else {\r\n this._originalCanvasSize = {\r\n width: canvas.offsetWidth,\r\n height: canvas.offsetHeight,\r\n };\r\n this._canvas = canvas;\r\n this.canvasContext = this._canvas.getContext(\"webgl2\");\r\n if (!this.canvasContext) {\r\n this.canvasContext = this._canvas.getContext(\"webgl\");\r\n }\r\n }\r\n }\r\n}\r\n","import type { RenderTargetTexture } from \"../../Materials/Textures/renderTargetTexture\";\r\nimport type { Viewport } from \"../../Maths/math.viewport\";\r\nimport type { Nullable } from \"../../types\";\r\nimport { WebXRLayerWrapper } from \"../webXRLayerWrapper\";\r\nimport { WebXRLayerRenderTargetTextureProvider } from \"../webXRRenderTargetTextureProvider\";\r\nimport type { WebXRSessionManager } from \"../webXRSessionManager\";\r\nimport type { WebXRRenderTarget } from \"../webXRTypes\";\r\n\r\n/**\r\n * Wraps XRWebGLLayer's created by Babylon Native.\r\n * @internal\r\n */\r\nexport class NativeXRLayerWrapper extends WebXRLayerWrapper {\r\n constructor(public readonly layer: XRWebGLLayer) {\r\n super(\r\n () => layer.framebufferWidth,\r\n () => layer.framebufferHeight,\r\n layer,\r\n \"XRWebGLLayer\",\r\n (sessionManager) => new NativeXRLayerRenderTargetTextureProvider(sessionManager, this)\r\n );\r\n }\r\n}\r\n\r\n/**\r\n * Provides render target textures for layers created by Babylon Native.\r\n * @internal\r\n */\r\nexport class NativeXRLayerRenderTargetTextureProvider extends WebXRLayerRenderTargetTextureProvider {\r\n private _nativeRTTProvider: WebXRLayerRenderTargetTextureProvider;\r\n private _nativeLayer: XRWebGLLayer;\r\n\r\n constructor(sessionManager: WebXRSessionManager, public readonly layerWrapper: NativeXRLayerWrapper) {\r\n super(sessionManager.scene, layerWrapper);\r\n this._nativeRTTProvider = (navigator as any).xr.getNativeRenderTargetProvider(\r\n sessionManager.session,\r\n this._createRenderTargetTexture.bind(this),\r\n this._destroyRenderTargetTexture.bind(this)\r\n );\r\n this._nativeLayer = layerWrapper.layer;\r\n }\r\n\r\n public trySetViewportForView(viewport: Viewport): boolean {\r\n viewport.x = 0;\r\n viewport.y = 0;\r\n viewport.width = 1;\r\n viewport.height = 1;\r\n return true;\r\n }\r\n\r\n public getRenderTargetTextureForEye(eye: XREye): Nullable {\r\n // TODO (rgerd): Update the contract on the BabylonNative side to call this \"getRenderTargetTextureForEye\"\r\n return (this._nativeRTTProvider as any).getRenderTargetForEye(eye);\r\n }\r\n\r\n public getRenderTargetTextureForView(view: XRView): Nullable {\r\n return (this._nativeRTTProvider as any).getRenderTargetForEye(view.eye);\r\n }\r\n\r\n public getFramebufferDimensions(): Nullable<{ framebufferWidth: number; framebufferHeight: number }> {\r\n return {\r\n framebufferWidth: this._nativeLayer.framebufferWidth,\r\n framebufferHeight: this._nativeLayer.framebufferHeight,\r\n };\r\n }\r\n}\r\n\r\n/**\r\n * Creates the xr layer that will be used as the xr session's base layer.\r\n * @internal\r\n */\r\nexport class NativeXRRenderTarget implements WebXRRenderTarget {\r\n public canvasContext: WebGLRenderingContext;\r\n public xrLayer: Nullable;\r\n\r\n private _nativeRenderTarget: WebXRRenderTarget;\r\n\r\n constructor(_xrSessionManager: WebXRSessionManager) {\r\n this._nativeRenderTarget = (navigator as any).xr.getWebXRRenderTarget(_xrSessionManager.scene.getEngine());\r\n }\r\n\r\n public async initializeXRLayerAsync(xrSession: XRSession): Promise {\r\n await this._nativeRenderTarget.initializeXRLayerAsync(xrSession);\r\n this.xrLayer = this._nativeRenderTarget.xrLayer!;\r\n return this.xrLayer;\r\n }\r\n\r\n dispose(): void {\r\n /* empty */\r\n }\r\n}\r\n","import { Logger } from \"../Misc/logger\";\r\nimport type { Observer } from \"../Misc/observable\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { IDisposable, Scene } from \"../scene\";\r\nimport type { RenderTargetTexture } from \"../Materials/Textures/renderTargetTexture\";\r\nimport type { WebXRRenderTarget } from \"./webXRTypes\";\r\nimport { WebXRManagedOutputCanvas, WebXRManagedOutputCanvasOptions } from \"./webXRManagedOutputCanvas\";\r\nimport type { Engine } from \"../Engines/engine\";\r\nimport type { IWebXRRenderTargetTextureProvider, WebXRLayerRenderTargetTextureProvider } from \"./webXRRenderTargetTextureProvider\";\r\nimport type { Viewport } from \"../Maths/math.viewport\";\r\nimport type { WebXRLayerWrapper } from \"./webXRLayerWrapper\";\r\nimport { NativeXRLayerWrapper, NativeXRRenderTarget } from \"./native/nativeXRRenderTarget\";\r\nimport { WebXRWebGLLayerWrapper } from \"./webXRWebGLLayer\";\r\nimport type { ThinEngine } from \"../Engines/thinEngine\";\r\n\r\n/**\r\n * Manages an XRSession to work with Babylon's engine\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/webXR/webXRSessionManagers\r\n */\r\nexport class WebXRSessionManager implements IDisposable, IWebXRRenderTargetTextureProvider {\r\n private _engine: Nullable;\r\n private _referenceSpace: XRReferenceSpace;\r\n private _baseLayerWrapper: Nullable;\r\n private _baseLayerRTTProvider: Nullable;\r\n private _xrNavigator: any;\r\n private _sessionMode: XRSessionMode;\r\n private _onEngineDisposedObserver: Nullable>;\r\n\r\n /**\r\n * The base reference space from which the session started. good if you want to reset your\r\n * reference space\r\n */\r\n public baseReferenceSpace: XRReferenceSpace;\r\n /**\r\n * Current XR frame\r\n */\r\n public currentFrame: Nullable;\r\n /** WebXR timestamp updated every frame */\r\n public currentTimestamp: number = -1;\r\n /**\r\n * Used just in case of a failure to initialize an immersive session.\r\n * The viewer reference space is compensated using this height, creating a kind of \"viewer-floor\" reference space\r\n */\r\n public defaultHeightCompensation = 1.7;\r\n /**\r\n * Fires every time a new xrFrame arrives which can be used to update the camera\r\n */\r\n public onXRFrameObservable: Observable = new Observable();\r\n /**\r\n * Fires when the reference space changed\r\n */\r\n public onXRReferenceSpaceChanged: Observable = new Observable();\r\n /**\r\n * Fires when the xr session is ended either by the device or manually done\r\n */\r\n public onXRSessionEnded: Observable = new Observable();\r\n /**\r\n * Fires when the xr session is initialized: right after requestSession was called and returned with a successful result\r\n */\r\n public onXRSessionInit: Observable = new Observable();\r\n /**\r\n * Underlying xr session\r\n */\r\n public session: XRSession;\r\n /**\r\n * The viewer (head position) reference space. This can be used to get the XR world coordinates\r\n * or get the offset the player is currently at.\r\n */\r\n public viewerReferenceSpace: XRReferenceSpace;\r\n /**\r\n * Are we currently in the XR loop?\r\n */\r\n public inXRFrameLoop: boolean = false;\r\n /**\r\n * Are we in an XR session?\r\n */\r\n public inXRSession: boolean = false;\r\n\r\n /**\r\n * Constructs a WebXRSessionManager, this must be initialized within a user action before usage\r\n * @param scene The scene which the session should be created for\r\n */\r\n constructor(\r\n /** The scene which the session should be created for */\r\n public scene: Scene\r\n ) {\r\n this._engine = scene.getEngine();\r\n this._onEngineDisposedObserver = this._engine.onDisposeObservable.addOnce(() => {\r\n this._engine = null;\r\n });\r\n scene.onDisposeObservable.addOnce(() => {\r\n this.dispose();\r\n });\r\n }\r\n\r\n /**\r\n * The current reference space used in this session. This reference space can constantly change!\r\n * It is mainly used to offset the camera's position.\r\n */\r\n public get referenceSpace(): XRReferenceSpace {\r\n return this._referenceSpace;\r\n }\r\n\r\n /**\r\n * Set a new reference space and triggers the observable\r\n */\r\n public set referenceSpace(newReferenceSpace: XRReferenceSpace) {\r\n this._referenceSpace = newReferenceSpace;\r\n this.onXRReferenceSpaceChanged.notifyObservers(this._referenceSpace);\r\n }\r\n\r\n /**\r\n * The mode for the managed XR session\r\n */\r\n public get sessionMode(): XRSessionMode {\r\n return this._sessionMode;\r\n }\r\n\r\n /**\r\n * Disposes of the session manager\r\n * This should be called explicitly by the dev, if required.\r\n */\r\n public dispose() {\r\n // disposing without leaving XR? Exit XR first\r\n if (this.inXRSession) {\r\n this.exitXRAsync();\r\n }\r\n this.onXRFrameObservable.clear();\r\n this.onXRSessionEnded.clear();\r\n this.onXRReferenceSpaceChanged.clear();\r\n this.onXRSessionInit.clear();\r\n this._engine?.onDisposeObservable.remove(this._onEngineDisposedObserver);\r\n this._engine = null;\r\n }\r\n\r\n /**\r\n * Stops the xrSession and restores the render loop\r\n * @returns Promise which resolves after it exits XR\r\n */\r\n public exitXRAsync() {\r\n if (this.session && this.inXRSession) {\r\n this.inXRSession = false;\r\n return this.session.end().catch(() => {\r\n Logger.Warn(\"Could not end XR session.\");\r\n });\r\n }\r\n return Promise.resolve();\r\n }\r\n\r\n /**\r\n * Attempts to set the framebuffer-size-normalized viewport to be rendered this frame for this view.\r\n * In the event of a failure, the supplied viewport is not updated.\r\n * @param viewport the viewport to which the view will be rendered\r\n * @param view the view for which to set the viewport\r\n * @returns whether the operation was successful\r\n */\r\n public trySetViewportForView(viewport: Viewport, view: XRView): boolean {\r\n return this._baseLayerRTTProvider?.trySetViewportForView(viewport, view) || false;\r\n }\r\n\r\n /**\r\n * Gets the correct render target texture to be rendered this frame for this eye\r\n * @param eye the eye for which to get the render target\r\n * @returns the render target for the specified eye or null if not available\r\n */\r\n public getRenderTargetTextureForEye(eye: XREye): Nullable {\r\n return this._baseLayerRTTProvider?.getRenderTargetTextureForEye(eye) || null;\r\n }\r\n\r\n /**\r\n * Gets the correct render target texture to be rendered this frame for this view\r\n * @param view the view for which to get the render target\r\n * @returns the render target for the specified view or null if not available\r\n */\r\n public getRenderTargetTextureForView(view: XRView): Nullable {\r\n return this._baseLayerRTTProvider?.getRenderTargetTextureForView(view) || null;\r\n }\r\n\r\n /**\r\n * Creates a WebXRRenderTarget object for the XR session\r\n * @param options optional options to provide when creating a new render target\r\n * @returns a WebXR render target to which the session can render\r\n */\r\n public getWebXRRenderTarget(options?: WebXRManagedOutputCanvasOptions): WebXRRenderTarget {\r\n const engine = this.scene.getEngine();\r\n if (this._xrNavigator.xr.native) {\r\n return new NativeXRRenderTarget(this);\r\n } else {\r\n options = options || WebXRManagedOutputCanvasOptions.GetDefaults(engine);\r\n options.canvasElement = options.canvasElement || engine.getRenderingCanvas() || undefined;\r\n return new WebXRManagedOutputCanvas(this, options);\r\n }\r\n }\r\n\r\n /**\r\n * Initializes the manager\r\n * After initialization enterXR can be called to start an XR session\r\n * @returns Promise which resolves after it is initialized\r\n */\r\n public initializeAsync(): Promise {\r\n // Check if the browser supports webXR\r\n this._xrNavigator = navigator;\r\n if (!this._xrNavigator.xr) {\r\n return Promise.reject(\"WebXR not available\");\r\n }\r\n return Promise.resolve();\r\n }\r\n\r\n /**\r\n * Initializes an xr session\r\n * @param xrSessionMode mode to initialize\r\n * @param xrSessionInit defines optional and required values to pass to the session builder\r\n * @returns a promise which will resolve once the session has been initialized\r\n */\r\n public initializeSessionAsync(xrSessionMode: XRSessionMode = \"immersive-vr\", xrSessionInit: XRSessionInit = {}): Promise {\r\n return this._xrNavigator.xr.requestSession(xrSessionMode, xrSessionInit).then((session: XRSession) => {\r\n this.session = session;\r\n this._sessionMode = xrSessionMode;\r\n this.onXRSessionInit.notifyObservers(session);\r\n this.inXRSession = true;\r\n\r\n // handle when the session is ended (By calling session.end or device ends its own session eg. pressing home button on phone)\r\n this.session.addEventListener(\r\n \"end\",\r\n () => {\r\n this.inXRSession = false;\r\n\r\n // Notify frame observers\r\n this.onXRSessionEnded.notifyObservers(null);\r\n\r\n if (this._engine) {\r\n // make sure dimensions object is restored\r\n this._engine.framebufferDimensionsObject = null;\r\n\r\n // Restore frame buffer to avoid clear on xr framebuffer after session end\r\n this._engine.restoreDefaultFramebuffer();\r\n\r\n // Need to restart render loop as after the session is ended the last request for new frame will never call callback\r\n this._engine.customAnimationFrameRequester = null;\r\n this._engine._renderLoop();\r\n }\r\n\r\n // Dispose render target textures.\r\n // Only dispose on native because we can't destroy opaque textures on browser.\r\n if (this.isNative) {\r\n this._baseLayerRTTProvider?.dispose();\r\n }\r\n this._baseLayerRTTProvider = null;\r\n this._baseLayerWrapper = null;\r\n },\r\n { once: true }\r\n );\r\n\r\n return this.session;\r\n });\r\n }\r\n\r\n /**\r\n * Checks if a session would be supported for the creation options specified\r\n * @param sessionMode session mode to check if supported eg. immersive-vr\r\n * @returns A Promise that resolves to true if supported and false if not\r\n */\r\n public isSessionSupportedAsync(sessionMode: XRSessionMode): Promise {\r\n return WebXRSessionManager.IsSessionSupportedAsync(sessionMode);\r\n }\r\n\r\n /**\r\n * Resets the reference space to the one started the session\r\n */\r\n public resetReferenceSpace() {\r\n this.referenceSpace = this.baseReferenceSpace;\r\n }\r\n\r\n /**\r\n * Starts rendering to the xr layer\r\n */\r\n public runXRRenderLoop() {\r\n if (!this.inXRSession || !this._engine) {\r\n return;\r\n }\r\n\r\n // Tell the engine's render loop to be driven by the xr session's refresh rate and provide xr pose information\r\n this._engine.customAnimationFrameRequester = {\r\n requestAnimationFrame: this.session.requestAnimationFrame.bind(this.session),\r\n renderFunction: (timestamp: number, xrFrame: Nullable) => {\r\n if (!this.inXRSession || !this._engine) {\r\n return;\r\n }\r\n // Store the XR frame and timestamp in the session manager\r\n this.currentFrame = xrFrame;\r\n this.currentTimestamp = timestamp;\r\n if (xrFrame) {\r\n this.inXRFrameLoop = true;\r\n this._engine.framebufferDimensionsObject = this._baseLayerRTTProvider?.getFramebufferDimensions() || null;\r\n this.onXRFrameObservable.notifyObservers(xrFrame);\r\n this._engine._renderLoop();\r\n this._engine.framebufferDimensionsObject = null;\r\n this.inXRFrameLoop = false;\r\n }\r\n },\r\n };\r\n\r\n this._engine.framebufferDimensionsObject = this._baseLayerRTTProvider?.getFramebufferDimensions() || null;\r\n\r\n // Stop window's animation frame and trigger sessions animation frame\r\n if (typeof window !== \"undefined\" && window.cancelAnimationFrame) {\r\n window.cancelAnimationFrame(this._engine._frameHandler);\r\n }\r\n this._engine._renderLoop();\r\n }\r\n\r\n /**\r\n * Sets the reference space on the xr session\r\n * @param referenceSpaceType space to set\r\n * @returns a promise that will resolve once the reference space has been set\r\n */\r\n public setReferenceSpaceTypeAsync(referenceSpaceType: XRReferenceSpaceType = \"local-floor\"): Promise {\r\n return this.session\r\n .requestReferenceSpace(referenceSpaceType)\r\n .then(\r\n (referenceSpace) => {\r\n return referenceSpace as XRReferenceSpace;\r\n },\r\n (rejectionReason) => {\r\n Logger.Error(\"XR.requestReferenceSpace failed for the following reason: \");\r\n Logger.Error(rejectionReason);\r\n Logger.Log('Defaulting to universally-supported \"viewer\" reference space type.');\r\n\r\n return this.session.requestReferenceSpace(\"viewer\").then(\r\n (referenceSpace) => {\r\n const heightCompensation = new XRRigidTransform({ x: 0, y: -this.defaultHeightCompensation, z: 0 });\r\n return (referenceSpace as XRReferenceSpace).getOffsetReferenceSpace(heightCompensation);\r\n },\r\n (rejectionReason) => {\r\n Logger.Error(rejectionReason);\r\n throw 'XR initialization failed: required \"viewer\" reference space type not supported.';\r\n }\r\n );\r\n }\r\n )\r\n .then((referenceSpace) => {\r\n // create viewer reference space before setting the first reference space\r\n return this.session.requestReferenceSpace(\"viewer\").then((viewerReferenceSpace) => {\r\n this.viewerReferenceSpace = viewerReferenceSpace as XRReferenceSpace;\r\n return referenceSpace;\r\n });\r\n })\r\n .then((referenceSpace) => {\r\n // initialize the base and offset (currently the same)\r\n this.referenceSpace = this.baseReferenceSpace = referenceSpace;\r\n return this.referenceSpace;\r\n });\r\n }\r\n\r\n /**\r\n * Updates the render state of the session.\r\n * Note that this is deprecated in favor of WebXRSessionManager.updateRenderState().\r\n * @param state state to set\r\n * @returns a promise that resolves once the render state has been updated\r\n * @deprecated\r\n */\r\n public updateRenderStateAsync(state: XRRenderState): Promise {\r\n return Promise.resolve(this.session.updateRenderState(state));\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _setBaseLayerWrapper(baseLayerWrapper: Nullable): void {\r\n if (this.isNative) {\r\n this._baseLayerRTTProvider?.dispose();\r\n }\r\n this._baseLayerWrapper = baseLayerWrapper;\r\n this._baseLayerRTTProvider = this._baseLayerWrapper?.createRenderTargetTextureProvider(this) || null;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getBaseLayerWrapper(): Nullable {\r\n return this._baseLayerWrapper;\r\n }\r\n\r\n /**\r\n * Updates the render state of the session\r\n * @param state state to set\r\n */\r\n public updateRenderState(state: XRRenderStateInit): void {\r\n if (state.baseLayer) {\r\n this._setBaseLayerWrapper(this.isNative ? new NativeXRLayerWrapper(state.baseLayer) : new WebXRWebGLLayerWrapper(state.baseLayer));\r\n }\r\n\r\n this.session.updateRenderState(state);\r\n }\r\n\r\n /**\r\n * Returns a promise that resolves with a boolean indicating if the provided session mode is supported by this browser\r\n * @param sessionMode defines the session to test\r\n * @returns a promise with boolean as final value\r\n */\r\n public static IsSessionSupportedAsync(sessionMode: XRSessionMode): Promise {\r\n if (!(navigator as any).xr) {\r\n return Promise.resolve(false);\r\n }\r\n // When the specs are final, remove supportsSession!\r\n const functionToUse = (navigator as any).xr.isSessionSupported || (navigator as any).xr.supportsSession;\r\n if (!functionToUse) {\r\n return Promise.resolve(false);\r\n } else {\r\n return functionToUse\r\n .call((navigator as any).xr, sessionMode)\r\n .then((result: boolean) => {\r\n const returnValue = typeof result === \"undefined\" ? true : result;\r\n return Promise.resolve(returnValue);\r\n })\r\n .catch((e: any) => {\r\n Logger.Warn(e);\r\n return Promise.resolve(false);\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Returns true if Babylon.js is using the BabylonNative backend, otherwise false\r\n */\r\n public get isNative(): boolean {\r\n return this._xrNavigator.xr.native ?? false;\r\n }\r\n\r\n /**\r\n * The current frame rate as reported by the device\r\n */\r\n public get currentFrameRate(): number | undefined {\r\n return this.session?.frameRate;\r\n }\r\n\r\n /**\r\n * A list of supported frame rates (only available in-session!\r\n */\r\n public get supportedFrameRates(): Float32Array | undefined {\r\n return this.session?.supportedFrameRates;\r\n }\r\n\r\n /**\r\n * Set the framerate of the session.\r\n * @param rate the new framerate. This value needs to be in the supportedFrameRates array\r\n * @returns a promise that resolves once the framerate has been set\r\n */\r\n public updateTargetFrameRate(rate: number): Promise {\r\n return this.session.updateTargetFrameRate(rate);\r\n }\r\n\r\n /**\r\n * Run a callback in the xr render loop\r\n * @param callback the callback to call when in XR Frame\r\n * @param ignoreIfNotInSession if no session is currently running, run it first thing on the next session\r\n */\r\n public runInXRFrame(callback: () => void, ignoreIfNotInSession = true): void {\r\n if (this.inXRFrameLoop) {\r\n callback();\r\n } else if (this.inXRSession || !ignoreIfNotInSession) {\r\n this.onXRFrameObservable.addOnce(callback);\r\n }\r\n }\r\n\r\n /**\r\n * Check if fixed foveation is supported on this device\r\n */\r\n public get isFixedFoveationSupported(): boolean {\r\n return this._baseLayerWrapper?.isFixedFoveationSupported || false;\r\n }\r\n\r\n /**\r\n * Get the fixed foveation currently set, as specified by the webxr specs\r\n * If this returns null, then fixed foveation is not supported\r\n */\r\n public get fixedFoveation(): Nullable {\r\n return this._baseLayerWrapper?.fixedFoveation || null;\r\n }\r\n\r\n /**\r\n * Set the fixed foveation to the specified value, as specified by the webxr specs\r\n * This value will be normalized to be between 0 and 1, 1 being max foveation, 0 being no foveation\r\n */\r\n public set fixedFoveation(value: Nullable) {\r\n const val = Math.max(0, Math.min(1, value || 0));\r\n if (this._baseLayerWrapper) {\r\n this._baseLayerWrapper.fixedFoveation = val;\r\n }\r\n }\r\n\r\n /**\r\n * Get the features enabled on the current session\r\n * This is only available in-session!\r\n * @see https://www.w3.org/TR/webxr/#dom-xrsession-enabledfeatures\r\n */\r\n public get enabledFeatures(): Nullable {\r\n return this.session?.enabledFeatures ?? null;\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\nimport type { IDisposable } from \"../scene\";\r\n\r\n/**\r\n * States of the webXR experience\r\n */\r\nexport enum WebXRState {\r\n /**\r\n * Transitioning to being in XR mode\r\n */\r\n ENTERING_XR,\r\n /**\r\n * Transitioning to non XR mode\r\n */\r\n EXITING_XR,\r\n /**\r\n * In XR mode and presenting\r\n */\r\n IN_XR,\r\n /**\r\n * Not entered XR mode\r\n */\r\n NOT_IN_XR,\r\n}\r\n\r\n/**\r\n * The state of the XR camera's tracking\r\n */\r\nexport enum WebXRTrackingState {\r\n /**\r\n * No transformation received, device is not being tracked\r\n */\r\n NOT_TRACKING,\r\n /**\r\n * Tracking lost - using emulated position\r\n */\r\n TRACKING_LOST,\r\n /**\r\n * Transformation tracking works normally\r\n */\r\n TRACKING,\r\n}\r\n\r\n/**\r\n * Abstraction of the XR render target\r\n */\r\nexport interface WebXRRenderTarget extends IDisposable {\r\n /**\r\n * xrpresent context of the canvas which can be used to display/mirror xr content\r\n */\r\n canvasContext: WebGLRenderingContext;\r\n\r\n /**\r\n * xr layer for the canvas\r\n */\r\n xrLayer: Nullable;\r\n\r\n /**\r\n * Initializes a XRWebGLLayer to be used as the session's baseLayer.\r\n * @param xrSession xr session\r\n * @returns a promise that will resolve once the XR Layer has been created\r\n */\r\n initializeXRLayerAsync(xrSession: XRSession): Promise;\r\n}\r\n","import { Vector4, Vector3, Vector2 } from \"../../Maths/math.vector\";\r\nimport { Color4 } from \"../../Maths/math.color\";\r\nimport { Mesh } from \"../mesh\";\r\nimport { VertexData } from \"../mesh.vertexData\";\r\nimport { Scene } from \"../../scene\";\r\nimport type { Nullable } from \"../../types\";\r\nimport { Axis } from \"../../Maths/math.axis\";\r\nimport { CompatibilityOptions } from \"../../Compat/compatibilityOptions\";\r\n\r\n/**\r\n * Creates the VertexData for a cylinder, cone or prism\r\n * @param options an object used to set the following optional parameters for the box, required but can be empty\r\n * * height sets the height (y direction) of the cylinder, optional, default 2\r\n * * diameterTop sets the diameter of the top of the cone, overwrites diameter, optional, default diameter\r\n * * diameterBottom sets the diameter of the bottom of the cone, overwrites diameter, optional, default diameter\r\n * * diameter sets the diameter of the top and bottom of the cone, optional default 1\r\n * * tessellation the number of prism sides, 3 for a triangular prism, optional, default 24\r\n * * subdivisions` the number of rings along the cylinder height, optional, default 1\r\n * * arc a number from 0 to 1, to create an unclosed cylinder based on the fraction of the circumference given by the arc value, optional, default 1\r\n * * faceColors an array of Color3 elements used to set different colors to the top, rings and bottom respectively\r\n * * faceUV an array of Vector4 elements used to set different images to the top, rings and bottom respectively\r\n * * hasRings when true makes each subdivision independently treated as a face for faceUV and faceColors, optional, default false\r\n * * enclose when true closes an open cylinder by adding extra flat faces between the height axis and vertical edges, think cut cake\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.height\r\n * @param options.diameterTop\r\n * @param options.diameterBottom\r\n * @param options.diameter\r\n * @param options.tessellation\r\n * @param options.subdivisions\r\n * @param options.arc\r\n * @param options.faceColors\r\n * @param options.faceUV\r\n * @param options.hasRings\r\n * @param options.enclose\r\n * @param options.cap\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @returns the VertexData of the cylinder, cone or prism\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport function CreateCylinderVertexData(options: {\r\n height?: number;\r\n diameterTop?: number;\r\n diameterBottom?: number;\r\n diameter?: number;\r\n tessellation?: number;\r\n subdivisions?: number;\r\n arc?: number;\r\n faceColors?: Color4[];\r\n faceUV?: Vector4[];\r\n hasRings?: boolean;\r\n enclose?: boolean;\r\n cap?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n}): VertexData {\r\n const height: number = options.height || 2;\r\n let diameterTop: number = options.diameterTop === 0 ? 0 : options.diameterTop || options.diameter || 1;\r\n let diameterBottom: number = options.diameterBottom === 0 ? 0 : options.diameterBottom || options.diameter || 1;\r\n diameterTop = diameterTop || 0.00001; // Prevent broken normals\r\n diameterBottom = diameterBottom || 0.00001; // Prevent broken normals\r\n const tessellation: number = (options.tessellation || 24) | 0;\r\n const subdivisions: number = (options.subdivisions || 1) | 0;\r\n const hasRings: boolean = options.hasRings ? true : false;\r\n const enclose: boolean = options.enclose ? true : false;\r\n const cap = options.cap === 0 ? 0 : options.cap || Mesh.CAP_ALL;\r\n const arc: number = options.arc && (options.arc <= 0 || options.arc > 1) ? 1.0 : options.arc || 1.0;\r\n const sideOrientation: number = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\r\n const faceUV: Vector4[] = options.faceUV || new Array(3);\r\n const faceColors = options.faceColors;\r\n // default face colors and UV if undefined\r\n const quadNb: number = arc !== 1 && enclose ? 2 : 0;\r\n const ringNb: number = hasRings ? subdivisions : 1;\r\n const surfaceNb: number = 2 + (1 + quadNb) * ringNb;\r\n let f: number;\r\n\r\n for (f = 0; f < surfaceNb; f++) {\r\n if (faceColors && faceColors[f] === undefined) {\r\n faceColors[f] = new Color4(1, 1, 1, 1);\r\n }\r\n }\r\n for (f = 0; f < surfaceNb; f++) {\r\n if (faceUV && faceUV[f] === undefined) {\r\n faceUV[f] = new Vector4(0, 0, 1, 1);\r\n }\r\n }\r\n\r\n const indices = new Array();\r\n const positions = new Array();\r\n const normals = new Array();\r\n const uvs = new Array();\r\n const colors = new Array();\r\n\r\n const angleStep = (Math.PI * 2 * arc) / tessellation;\r\n let angle: number;\r\n let h: number;\r\n let radius: number;\r\n const tan = (diameterBottom - diameterTop) / 2 / height;\r\n const ringVertex: Vector3 = Vector3.Zero();\r\n const ringNormal: Vector3 = Vector3.Zero();\r\n const ringFirstVertex: Vector3 = Vector3.Zero();\r\n const ringFirstNormal: Vector3 = Vector3.Zero();\r\n const quadNormal: Vector3 = Vector3.Zero();\r\n const Y: Vector3 = Axis.Y;\r\n\r\n // positions, normals, uvs\r\n let i: number;\r\n let j: number;\r\n let r: number;\r\n let ringIdx: number = 1;\r\n let s: number = 1; // surface index\r\n let cs: number = 0;\r\n let v: number = 0;\r\n\r\n for (i = 0; i <= subdivisions; i++) {\r\n h = i / subdivisions;\r\n radius = (h * (diameterTop - diameterBottom) + diameterBottom) / 2;\r\n ringIdx = hasRings && i !== 0 && i !== subdivisions ? 2 : 1;\r\n for (r = 0; r < ringIdx; r++) {\r\n if (hasRings) {\r\n s += r;\r\n }\r\n if (enclose) {\r\n s += 2 * r;\r\n }\r\n for (j = 0; j <= tessellation; j++) {\r\n angle = j * angleStep;\r\n\r\n // position\r\n ringVertex.x = Math.cos(-angle) * radius;\r\n ringVertex.y = -height / 2 + h * height;\r\n ringVertex.z = Math.sin(-angle) * radius;\r\n\r\n // normal\r\n if (diameterTop === 0 && i === subdivisions) {\r\n // if no top cap, reuse former normals\r\n ringNormal.x = normals[normals.length - (tessellation + 1) * 3];\r\n ringNormal.y = normals[normals.length - (tessellation + 1) * 3 + 1];\r\n ringNormal.z = normals[normals.length - (tessellation + 1) * 3 + 2];\r\n } else {\r\n ringNormal.x = ringVertex.x;\r\n ringNormal.z = ringVertex.z;\r\n ringNormal.y = Math.sqrt(ringNormal.x * ringNormal.x + ringNormal.z * ringNormal.z) * tan;\r\n ringNormal.normalize();\r\n }\r\n\r\n // keep first ring vertex values for enclose\r\n if (j === 0) {\r\n ringFirstVertex.copyFrom(ringVertex);\r\n ringFirstNormal.copyFrom(ringNormal);\r\n }\r\n\r\n positions.push(ringVertex.x, ringVertex.y, ringVertex.z);\r\n normals.push(ringNormal.x, ringNormal.y, ringNormal.z);\r\n if (hasRings) {\r\n v = cs !== s ? faceUV[s].y : faceUV[s].w;\r\n } else {\r\n v = faceUV[s].y + (faceUV[s].w - faceUV[s].y) * h;\r\n }\r\n uvs.push(faceUV[s].x + ((faceUV[s].z - faceUV[s].x) * j) / tessellation, CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - v : v);\r\n if (faceColors) {\r\n colors.push(faceColors[s].r, faceColors[s].g, faceColors[s].b, faceColors[s].a);\r\n }\r\n }\r\n\r\n // if enclose, add four vertices and their dedicated normals\r\n if (arc !== 1 && enclose) {\r\n positions.push(ringVertex.x, ringVertex.y, ringVertex.z);\r\n positions.push(0, ringVertex.y, 0);\r\n positions.push(0, ringVertex.y, 0);\r\n positions.push(ringFirstVertex.x, ringFirstVertex.y, ringFirstVertex.z);\r\n Vector3.CrossToRef(Y, ringNormal, quadNormal);\r\n quadNormal.normalize();\r\n normals.push(quadNormal.x, quadNormal.y, quadNormal.z, quadNormal.x, quadNormal.y, quadNormal.z);\r\n Vector3.CrossToRef(ringFirstNormal, Y, quadNormal);\r\n quadNormal.normalize();\r\n normals.push(quadNormal.x, quadNormal.y, quadNormal.z, quadNormal.x, quadNormal.y, quadNormal.z);\r\n if (hasRings) {\r\n v = cs !== s ? faceUV[s + 1].y : faceUV[s + 1].w;\r\n } else {\r\n v = faceUV[s + 1].y + (faceUV[s + 1].w - faceUV[s + 1].y) * h;\r\n }\r\n uvs.push(faceUV[s + 1].x, CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - v : v);\r\n uvs.push(faceUV[s + 1].z, CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - v : v);\r\n if (hasRings) {\r\n v = cs !== s ? faceUV[s + 2].y : faceUV[s + 2].w;\r\n } else {\r\n v = faceUV[s + 2].y + (faceUV[s + 2].w - faceUV[s + 2].y) * h;\r\n }\r\n uvs.push(faceUV[s + 2].x, CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - v : v);\r\n uvs.push(faceUV[s + 2].z, CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - v : v);\r\n if (faceColors) {\r\n colors.push(faceColors[s + 1].r, faceColors[s + 1].g, faceColors[s + 1].b, faceColors[s + 1].a);\r\n colors.push(faceColors[s + 1].r, faceColors[s + 1].g, faceColors[s + 1].b, faceColors[s + 1].a);\r\n colors.push(faceColors[s + 2].r, faceColors[s + 2].g, faceColors[s + 2].b, faceColors[s + 2].a);\r\n colors.push(faceColors[s + 2].r, faceColors[s + 2].g, faceColors[s + 2].b, faceColors[s + 2].a);\r\n }\r\n }\r\n if (cs !== s) {\r\n cs = s;\r\n }\r\n }\r\n }\r\n\r\n // indices\r\n const e: number = arc !== 1 && enclose ? tessellation + 4 : tessellation; // correction of number of iteration if enclose\r\n i = 0;\r\n for (s = 0; s < subdivisions; s++) {\r\n let i0: number = 0;\r\n let i1: number = 0;\r\n let i2: number = 0;\r\n let i3: number = 0;\r\n for (j = 0; j < tessellation; j++) {\r\n i0 = i * (e + 1) + j;\r\n i1 = (i + 1) * (e + 1) + j;\r\n i2 = i * (e + 1) + (j + 1);\r\n i3 = (i + 1) * (e + 1) + (j + 1);\r\n indices.push(i0, i1, i2);\r\n indices.push(i3, i2, i1);\r\n }\r\n if (arc !== 1 && enclose) {\r\n // if enclose, add two quads\r\n indices.push(i0 + 2, i1 + 2, i2 + 2);\r\n indices.push(i3 + 2, i2 + 2, i1 + 2);\r\n indices.push(i0 + 4, i1 + 4, i2 + 4);\r\n indices.push(i3 + 4, i2 + 4, i1 + 4);\r\n }\r\n i = hasRings ? i + 2 : i + 1;\r\n }\r\n\r\n // Caps\r\n const createCylinderCap = (isTop: boolean) => {\r\n const radius = isTop ? diameterTop / 2 : diameterBottom / 2;\r\n if (radius === 0) {\r\n return;\r\n }\r\n\r\n // Cap positions, normals & uvs\r\n let angle;\r\n let circleVector;\r\n let i: number;\r\n const u: Vector4 = isTop ? faceUV[surfaceNb - 1] : faceUV[0];\r\n let c: Nullable = null;\r\n if (faceColors) {\r\n c = isTop ? faceColors[surfaceNb - 1] : faceColors[0];\r\n }\r\n // cap center\r\n const vbase = positions.length / 3;\r\n const offset = isTop ? height / 2 : -height / 2;\r\n const center = new Vector3(0, offset, 0);\r\n positions.push(center.x, center.y, center.z);\r\n normals.push(0, isTop ? 1 : -1, 0);\r\n const v = u.y + (u.w - u.y) * 0.5;\r\n uvs.push(u.x + (u.z - u.x) * 0.5, CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - v : v);\r\n if (c) {\r\n colors.push(c.r, c.g, c.b, c.a);\r\n }\r\n\r\n const textureScale = new Vector2(0.5, 0.5);\r\n for (i = 0; i <= tessellation; i++) {\r\n angle = (Math.PI * 2 * i * arc) / tessellation;\r\n const cos = Math.cos(-angle);\r\n const sin = Math.sin(-angle);\r\n circleVector = new Vector3(cos * radius, offset, sin * radius);\r\n const textureCoordinate = new Vector2(cos * textureScale.x + 0.5, sin * textureScale.y + 0.5);\r\n positions.push(circleVector.x, circleVector.y, circleVector.z);\r\n normals.push(0, isTop ? 1 : -1, 0);\r\n const v = u.y + (u.w - u.y) * textureCoordinate.y;\r\n uvs.push(u.x + (u.z - u.x) * textureCoordinate.x, CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - v : v);\r\n if (c) {\r\n colors.push(c.r, c.g, c.b, c.a);\r\n }\r\n }\r\n // Cap indices\r\n for (i = 0; i < tessellation; i++) {\r\n if (!isTop) {\r\n indices.push(vbase);\r\n indices.push(vbase + (i + 1));\r\n indices.push(vbase + (i + 2));\r\n } else {\r\n indices.push(vbase);\r\n indices.push(vbase + (i + 2));\r\n indices.push(vbase + (i + 1));\r\n }\r\n }\r\n };\r\n\r\n // add caps to geometry based on cap parameter\r\n if (cap === Mesh.CAP_START || cap === Mesh.CAP_ALL) {\r\n createCylinderCap(false);\r\n }\r\n if (cap === Mesh.CAP_END || cap === Mesh.CAP_ALL) {\r\n createCylinderCap(true);\r\n }\r\n\r\n // Sides\r\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\r\n\r\n const vertexData = new VertexData();\r\n\r\n vertexData.indices = indices;\r\n vertexData.positions = positions;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n if (faceColors) {\r\n vertexData.colors = colors;\r\n }\r\n\r\n return vertexData;\r\n}\r\n\r\n/**\r\n * Creates a cylinder or a cone mesh\r\n * * The parameter `height` sets the height size (float) of the cylinder/cone (float, default 2).\r\n * * The parameter `diameter` sets the diameter of the top and bottom cap at once (float, default 1).\r\n * * The parameters `diameterTop` and `diameterBottom` overwrite the parameter `diameter` and set respectively the top cap and bottom cap diameter (floats, default 1). The parameter \"diameterBottom\" can't be zero.\r\n * * The parameter `tessellation` sets the number of cylinder sides (positive integer, default 24). Set it to 3 to get a prism for instance.\r\n * * The parameter `subdivisions` sets the number of rings along the cylinder height (positive integer, default 1).\r\n * * The parameter `hasRings` (boolean, default false) makes the subdivisions independent from each other, so they become different faces.\r\n * * The parameter `enclose` (boolean, default false) adds two extra faces per subdivision to a sliced cylinder to close it around its height axis.\r\n * * The parameter `cap` sets the way the cylinder is capped. Possible values : BABYLON.Mesh.NO_CAP, BABYLON.Mesh.CAP_START, BABYLON.Mesh.CAP_END, BABYLON.Mesh.CAP_ALL (default).\r\n * * The parameter `arc` (float, default 1) is the ratio (max 1) to apply to the circumference to slice the cylinder.\r\n * * You can set different colors and different images to each box side by using the parameters `faceColors` (an array of n Color3 elements) and `faceUV` (an array of n Vector4 elements).\r\n * * The value of n is the number of cylinder faces. If the cylinder has only 1 subdivisions, n equals : top face + cylinder surface + bottom face = 3\r\n * * Now, if the cylinder has 5 independent subdivisions (hasRings = true), n equals : top face + 5 stripe surfaces + bottom face = 2 + 5 = 7\r\n * * Finally, if the cylinder has 5 independent subdivisions and is enclose, n equals : top face + 5 x (stripe surface + 2 closing faces) + bottom face = 2 + 5 * 3 = 17\r\n * * Each array (color or UVs) is always ordered the same way : the first element is the bottom cap, the last element is the top cap. The other elements are each a ring surface.\r\n * * If `enclose` is false, a ring surface is one element.\r\n * * If `enclose` is true, a ring surface is 3 successive elements in the array : the tubular surface, then the two closing faces.\r\n * * Example how to set colors and textures on a sliced cylinder : https://www.html5gamedevs.com/topic/17945-creating-a-closed-slice-of-a-cylinder/#comment-106379\r\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param options.height\r\n * @param options.diameterTop\r\n * @param options.diameterBottom\r\n * @param options.diameter\r\n * @param options.tessellation\r\n * @param options.subdivisions\r\n * @param options.arc\r\n * @param options.faceColors\r\n * @param options.faceUV\r\n * @param options.updatable\r\n * @param options.hasRings\r\n * @param options.enclose\r\n * @param options.cap\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param scene defines the hosting scene\r\n * @returns the cylinder mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#cylinder-or-cone\r\n */\r\nexport function CreateCylinder(\r\n name: string,\r\n options: {\r\n height?: number;\r\n diameterTop?: number;\r\n diameterBottom?: number;\r\n diameter?: number;\r\n tessellation?: number;\r\n subdivisions?: number;\r\n arc?: number;\r\n faceColors?: Color4[];\r\n faceUV?: Vector4[];\r\n updatable?: boolean;\r\n hasRings?: boolean;\r\n enclose?: boolean;\r\n cap?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n } = {},\r\n scene?: Nullable\r\n): Mesh {\r\n const cylinder = new Mesh(name, scene);\r\n\r\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n cylinder._originalBuilderSideOrientation = options.sideOrientation;\r\n\r\n const vertexData = CreateCylinderVertexData(options);\r\n\r\n vertexData.applyToMesh(cylinder, options.updatable);\r\n\r\n return cylinder;\r\n}\r\n\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated Please use CreateCylinder directly\r\n */\r\nexport const CylinderBuilder = {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreateCylinder,\r\n};\r\n\r\nVertexData.CreateCylinder = CreateCylinderVertexData;\r\n\r\nMesh.CreateCylinder = (\r\n name: string,\r\n height: number,\r\n diameterTop: number,\r\n diameterBottom: number,\r\n tessellation: number,\r\n subdivisions: any,\r\n scene?: Scene,\r\n updatable?: any,\r\n sideOrientation?: number\r\n): Mesh => {\r\n if (scene === undefined || !(scene instanceof Scene)) {\r\n if (scene !== undefined) {\r\n sideOrientation = updatable || Mesh.DEFAULTSIDE;\r\n updatable = scene;\r\n }\r\n scene = subdivisions;\r\n subdivisions = 1;\r\n }\r\n\r\n const options = {\r\n height,\r\n diameterTop,\r\n diameterBottom,\r\n tessellation,\r\n subdivisions,\r\n sideOrientation,\r\n updatable,\r\n };\r\n\r\n return CreateCylinder(name, options, scene);\r\n};\r\n","import type { Vector4 } from \"../../Maths/math.vector\";\r\nimport { Matrix, Vector3, Vector2 } from \"../../Maths/math.vector\";\r\nimport { Mesh } from \"../mesh\";\r\nimport { VertexData } from \"../mesh.vertexData\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { CompatibilityOptions } from \"../../Compat/compatibilityOptions\";\r\n\r\n/**\r\n * Creates the VertexData for a torus\r\n * @param options an object used to set the following optional parameters for the box, required but can be empty\r\n * * diameter the diameter of the torus, optional default 1\r\n * * thickness the diameter of the tube forming the torus, optional default 0.5\r\n * * tessellation the number of prism sides, 3 for a triangular prism, optional, default 24\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.diameter\r\n * @param options.thickness\r\n * @param options.tessellation\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @returns the VertexData of the torus\r\n */\r\nexport function CreateTorusVertexData(options: { diameter?: number; thickness?: number; tessellation?: number; sideOrientation?: number; frontUVs?: Vector4; backUVs?: Vector4 }) {\r\n const indices = [];\r\n const positions = [];\r\n const normals = [];\r\n const uvs = [];\r\n\r\n const diameter = options.diameter || 1;\r\n const thickness = options.thickness || 0.5;\r\n const tessellation = (options.tessellation || 16) | 0;\r\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\r\n\r\n const stride = tessellation + 1;\r\n\r\n for (let i = 0; i <= tessellation; i++) {\r\n const u = i / tessellation;\r\n\r\n const outerAngle = (i * Math.PI * 2.0) / tessellation - Math.PI / 2.0;\r\n\r\n const transform = Matrix.Translation(diameter / 2.0, 0, 0).multiply(Matrix.RotationY(outerAngle));\r\n\r\n for (let j = 0; j <= tessellation; j++) {\r\n const v = 1 - j / tessellation;\r\n\r\n const innerAngle = (j * Math.PI * 2.0) / tessellation + Math.PI;\r\n const dx = Math.cos(innerAngle);\r\n const dy = Math.sin(innerAngle);\r\n\r\n // Create a vertex.\r\n let normal = new Vector3(dx, dy, 0);\r\n let position = normal.scale(thickness / 2);\r\n const textureCoordinate = new Vector2(u, v);\r\n\r\n position = Vector3.TransformCoordinates(position, transform);\r\n normal = Vector3.TransformNormal(normal, transform);\r\n\r\n positions.push(position.x, position.y, position.z);\r\n normals.push(normal.x, normal.y, normal.z);\r\n uvs.push(textureCoordinate.x, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - textureCoordinate.y : textureCoordinate.y);\r\n\r\n // And create indices for two triangles.\r\n const nextI = (i + 1) % stride;\r\n const nextJ = (j + 1) % stride;\r\n\r\n indices.push(i * stride + j);\r\n indices.push(i * stride + nextJ);\r\n indices.push(nextI * stride + j);\r\n\r\n indices.push(i * stride + nextJ);\r\n indices.push(nextI * stride + nextJ);\r\n indices.push(nextI * stride + j);\r\n }\r\n }\r\n\r\n // Sides\r\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\r\n\r\n // Result\r\n const vertexData = new VertexData();\r\n\r\n vertexData.indices = indices;\r\n vertexData.positions = positions;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n\r\n return vertexData;\r\n}\r\n\r\n/**\r\n * Creates a torus mesh\r\n * * The parameter `diameter` sets the diameter size (float) of the torus (default 1)\r\n * * The parameter `thickness` sets the diameter size of the tube of the torus (float, default 0.5)\r\n * * The parameter `tessellation` sets the number of torus sides (positive integer, default 16)\r\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param options.diameter\r\n * @param options.thickness\r\n * @param options.tessellation\r\n * @param options.updatable\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param scene defines the hosting scene\r\n * @returns the torus mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#torus\r\n */\r\nexport function CreateTorus(\r\n name: string,\r\n options: { diameter?: number; thickness?: number; tessellation?: number; updatable?: boolean; sideOrientation?: number; frontUVs?: Vector4; backUVs?: Vector4 } = {},\r\n scene?: Scene\r\n): Mesh {\r\n const torus = new Mesh(name, scene);\r\n\r\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n torus._originalBuilderSideOrientation = options.sideOrientation;\r\n\r\n const vertexData = CreateTorusVertexData(options);\r\n\r\n vertexData.applyToMesh(torus, options.updatable);\r\n\r\n return torus;\r\n}\r\n\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated use CreateTorus instead\r\n */\r\nexport const TorusBuilder = {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreateTorus,\r\n};\r\n\r\nVertexData.CreateTorus = CreateTorusVertexData;\r\n\r\nMesh.CreateTorus = (name: string, diameter: number, thickness: number, tessellation: number, scene?: Scene, updatable?: boolean, sideOrientation?: number): Mesh => {\r\n const options = {\r\n diameter,\r\n thickness,\r\n tessellation,\r\n sideOrientation,\r\n updatable,\r\n };\r\n\r\n return CreateTorus(name, options, scene);\r\n};\r\n","import { Logger } from \"../../Misc/logger\";\r\nimport type { Observer } from \"../../Misc/observable\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { Camera } from \"../../Cameras/camera\";\r\nimport { FreeCamera } from \"../../Cameras/freeCamera\";\r\nimport { TargetCamera } from \"../../Cameras/targetCamera\";\r\nimport { DeviceOrientationCamera } from \"../../Cameras/deviceOrientationCamera\";\r\nimport { VRDeviceOrientationFreeCamera } from \"../../Cameras/VR/vrDeviceOrientationFreeCamera\";\r\nimport type { WebVROptions } from \"../../Cameras/VR/webVRCamera\";\r\nimport { WebVRFreeCamera } from \"../../Cameras/VR/webVRCamera\";\r\nimport { PointerEventTypes } from \"../../Events/pointerEvents\";\r\nimport type { Scene, IDisposable } from \"../../scene\";\r\nimport { Quaternion, Matrix, Vector3 } from \"../../Maths/math.vector\";\r\nimport { Color3, Color4 } from \"../../Maths/math.color\";\r\nimport type { StickValues } from \"../../Gamepads/gamepad\";\r\nimport { Gamepad } from \"../../Gamepads/gamepad\";\r\nimport { PoseEnabledController, PoseEnabledControllerType } from \"../../Gamepads/Controllers/poseEnabledController\";\r\nimport { WebVRController } from \"../../Gamepads/Controllers/webVRController\";\r\nimport type { Xbox360Pad } from \"../../Gamepads/xboxGamepad\";\r\nimport { Xbox360Button } from \"../../Gamepads/xboxGamepad\";\r\nimport type { IDisplayChangedEventArgs } from \"../../Engines/engine\";\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport type { TransformNode } from \"../../Meshes/transformNode\";\r\nimport { Mesh } from \"../../Meshes/mesh\";\r\nimport type { PickingInfo } from \"../../Collisions/pickingInfo\";\r\nimport { Ray } from \"../../Culling/ray\";\r\nimport { ImageProcessingConfiguration } from \"../../Materials/imageProcessingConfiguration\";\r\nimport { StandardMaterial } from \"../../Materials/standardMaterial\";\r\nimport { DynamicTexture } from \"../../Materials/Textures/dynamicTexture\";\r\nimport { ImageProcessingPostProcess } from \"../../PostProcesses/imageProcessingPostProcess\";\r\nimport { SineEase, EasingFunction, CircleEase } from \"../../Animations/easing\";\r\nimport { Animation } from \"../../Animations/animation\";\r\nimport { VRCameraMetrics } from \"../../Cameras/VR/vrCameraMetrics\";\r\nimport \"../../Gamepads/gamepadSceneComponent\";\r\nimport \"../../Animations/animatable\";\r\nimport { Axis } from \"../../Maths/math.axis\";\r\nimport { WebXRSessionManager } from \"../../XR/webXRSessionManager\";\r\nimport type { WebXRDefaultExperience } from \"../../XR/webXRDefaultExperience\";\r\nimport { WebXRState } from \"../../XR/webXRTypes\";\r\nimport { CreateCylinder } from \"../../Meshes/Builders/cylinderBuilder\";\r\nimport { CreateTorus } from \"../../Meshes/Builders/torusBuilder\";\r\nimport { CreateGround } from \"../../Meshes/Builders/groundBuilder\";\r\nimport type { IPointerEvent } from \"../../Events/deviceInputEvents\";\r\n\r\n/**\r\n * Options to modify the vr teleportation behavior.\r\n */\r\nexport interface VRTeleportationOptions {\r\n /**\r\n * The name of the mesh which should be used as the teleportation floor. (default: null)\r\n */\r\n floorMeshName?: string;\r\n /**\r\n * A list of meshes to be used as the teleportation floor. (default: empty)\r\n */\r\n floorMeshes?: Mesh[];\r\n /**\r\n * The teleportation mode. (default: TELEPORTATIONMODE_CONSTANTTIME)\r\n */\r\n teleportationMode?: number;\r\n /**\r\n * The duration of the animation in ms, apply when animationMode is TELEPORTATIONMODE_CONSTANTTIME. (default 122ms)\r\n */\r\n teleportationTime?: number;\r\n /**\r\n * The speed of the animation in distance/sec, apply when animationMode is TELEPORTATIONMODE_CONSTANTSPEED. (default 20 units / sec)\r\n */\r\n teleportationSpeed?: number;\r\n /**\r\n * The easing function used in the animation or null for Linear. (default CircleEase)\r\n */\r\n easingFunction?: EasingFunction;\r\n}\r\n\r\n/**\r\n * Options to modify the vr experience helper's behavior.\r\n */\r\nexport interface VRExperienceHelperOptions extends WebVROptions {\r\n /**\r\n * Create a DeviceOrientationCamera to be used as your out of vr camera. (default: true)\r\n */\r\n createDeviceOrientationCamera?: boolean;\r\n /**\r\n * Create a VRDeviceOrientationFreeCamera to be used for VR when no external HMD is found. (default: true)\r\n */\r\n createFallbackVRDeviceOrientationFreeCamera?: boolean;\r\n /**\r\n * Uses the main button on the controller to toggle the laser casted. (default: true)\r\n */\r\n laserToggle?: boolean;\r\n /**\r\n * A list of meshes to be used as the teleportation floor. If specified, teleportation will be enabled (default: undefined)\r\n */\r\n floorMeshes?: Mesh[];\r\n /**\r\n * Distortion metrics for the fallback vrDeviceOrientationCamera (default: VRCameraMetrics.Default)\r\n */\r\n vrDeviceOrientationCameraMetrics?: VRCameraMetrics;\r\n /**\r\n * Defines if WebXR should be used instead of WebVR (if available)\r\n */\r\n useXR?: boolean;\r\n}\r\n\r\nclass VRExperienceHelperGazer implements IDisposable {\r\n /** @internal */\r\n public _gazeTracker: Mesh;\r\n\r\n /** @internal */\r\n public _currentMeshSelected: Nullable;\r\n /** @internal */\r\n public _currentHit: Nullable;\r\n\r\n public static _IdCounter = 0;\r\n /** @internal */\r\n public _id: number;\r\n\r\n /** @internal */\r\n public _pointerDownOnMeshAsked: boolean = false;\r\n /** @internal */\r\n public _isActionableMesh: boolean = false;\r\n\r\n /** @internal */\r\n public _interactionsEnabled: boolean;\r\n /** @internal */\r\n public _teleportationEnabled: boolean;\r\n /** @internal */\r\n public _teleportationRequestInitiated = false;\r\n /** @internal */\r\n public _teleportationBackRequestInitiated = false;\r\n /** @internal */\r\n public _rotationRightAsked = false;\r\n /** @internal */\r\n public _rotationLeftAsked = false;\r\n /** @internal */\r\n public _dpadPressed = true;\r\n\r\n /** @internal */\r\n public _activePointer = false;\r\n\r\n constructor(public scene: Scene, gazeTrackerToClone: Nullable = null) {\r\n this._id = VRExperienceHelperGazer._IdCounter++;\r\n\r\n // Gaze tracker\r\n if (!gazeTrackerToClone) {\r\n this._gazeTracker = CreateTorus(\r\n \"gazeTracker\",\r\n {\r\n diameter: 0.0035,\r\n thickness: 0.0025,\r\n tessellation: 20,\r\n updatable: false,\r\n },\r\n scene\r\n );\r\n this._gazeTracker.bakeCurrentTransformIntoVertices();\r\n this._gazeTracker.isPickable = false;\r\n this._gazeTracker.isVisible = false;\r\n const targetMat = new StandardMaterial(\"targetMat\", scene);\r\n targetMat.specularColor = Color3.Black();\r\n targetMat.emissiveColor = new Color3(0.7, 0.7, 0.7);\r\n targetMat.backFaceCulling = false;\r\n this._gazeTracker.material = targetMat;\r\n } else {\r\n this._gazeTracker = gazeTrackerToClone.clone(\"gazeTracker\") as Mesh;\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getForwardRay(length: number): Ray {\r\n return new Ray(Vector3.Zero(), new Vector3(0, 0, length));\r\n }\r\n\r\n /** @internal */\r\n public _selectionPointerDown() {\r\n this._pointerDownOnMeshAsked = true;\r\n if (this._currentHit) {\r\n this.scene.simulatePointerDown(this._currentHit, { pointerId: this._id });\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _selectionPointerUp() {\r\n if (this._currentHit) {\r\n this.scene.simulatePointerUp(this._currentHit, { pointerId: this._id });\r\n }\r\n this._pointerDownOnMeshAsked = false;\r\n }\r\n\r\n /** @internal */\r\n public _activatePointer() {\r\n this._activePointer = true;\r\n }\r\n\r\n /** @internal */\r\n public _deactivatePointer() {\r\n this._activePointer = false;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public _updatePointerDistance(distance: number = 100) {}\r\n\r\n public dispose() {\r\n this._interactionsEnabled = false;\r\n this._teleportationEnabled = false;\r\n if (this._gazeTracker) {\r\n this._gazeTracker.dispose();\r\n }\r\n }\r\n}\r\n\r\nclass VRExperienceHelperControllerGazer extends VRExperienceHelperGazer {\r\n private _laserPointer: Mesh;\r\n private _meshAttachedObserver: Nullable>;\r\n constructor(public webVRController: WebVRController, scene: Scene, gazeTrackerToClone: Mesh) {\r\n super(scene, gazeTrackerToClone);\r\n // Laser pointer\r\n this._laserPointer = CreateCylinder(\r\n \"laserPointer\",\r\n {\r\n updatable: false,\r\n height: 1,\r\n diameterTop: 0.004,\r\n diameterBottom: 0.0002,\r\n tessellation: 20,\r\n subdivisions: 1,\r\n },\r\n scene\r\n );\r\n const laserPointerMaterial = new StandardMaterial(\"laserPointerMat\", scene);\r\n laserPointerMaterial.emissiveColor = new Color3(0.7, 0.7, 0.7);\r\n laserPointerMaterial.alpha = 0.6;\r\n this._laserPointer.material = laserPointerMaterial;\r\n this._laserPointer.rotation.x = Math.PI / 2;\r\n this._laserPointer.position.z = -0.5;\r\n this._laserPointer.isVisible = false;\r\n this._laserPointer.isPickable = false;\r\n\r\n if (!webVRController.mesh) {\r\n // Create an empty mesh that is used prior to loading the high quality model\r\n const preloadMesh = new Mesh(\"preloadControllerMesh\", scene);\r\n const preloadPointerPose = new Mesh(PoseEnabledController.POINTING_POSE, scene);\r\n preloadPointerPose.rotation.x = -0.7;\r\n preloadMesh.addChild(preloadPointerPose);\r\n webVRController.attachToMesh(preloadMesh);\r\n }\r\n\r\n this._setLaserPointerParent(webVRController.mesh!);\r\n\r\n this._meshAttachedObserver = webVRController._meshAttachedObservable.add((mesh) => {\r\n this._setLaserPointerParent(mesh);\r\n });\r\n }\r\n\r\n _getForwardRay(length: number): Ray {\r\n return this.webVRController.getForwardRay(length);\r\n }\r\n\r\n /** @internal */\r\n public _activatePointer() {\r\n super._activatePointer();\r\n this._laserPointer.isVisible = true;\r\n }\r\n\r\n /** @internal */\r\n public _deactivatePointer() {\r\n super._deactivatePointer();\r\n this._laserPointer.isVisible = false;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _setLaserPointerColor(color: Color3) {\r\n (this._laserPointer.material).emissiveColor = color;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _setLaserPointerLightingDisabled(disabled: boolean) {\r\n (this._laserPointer.material).disableLighting = disabled;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _setLaserPointerParent(mesh: AbstractMesh) {\r\n const makeNotPick = (root: AbstractMesh) => {\r\n root.isPickable = false;\r\n root.getChildMeshes().forEach((c) => {\r\n makeNotPick(c);\r\n });\r\n };\r\n makeNotPick(mesh);\r\n const meshChildren = mesh.getChildren(undefined, false);\r\n\r\n let laserParent: TransformNode = mesh;\r\n this.webVRController._pointingPoseNode = null;\r\n for (let i = 0; i < meshChildren.length; i++) {\r\n if (meshChildren[i].name && meshChildren[i].name.indexOf(PoseEnabledController.POINTING_POSE) >= 0) {\r\n laserParent = meshChildren[i];\r\n this.webVRController._pointingPoseNode = laserParent;\r\n break;\r\n }\r\n }\r\n this._laserPointer.parent = laserParent;\r\n }\r\n\r\n public _updatePointerDistance(distance: number = 100) {\r\n this._laserPointer.scaling.y = distance;\r\n this._laserPointer.position.z = -distance / 2;\r\n }\r\n\r\n dispose() {\r\n super.dispose();\r\n this._laserPointer.dispose();\r\n if (this._meshAttachedObserver) {\r\n this.webVRController._meshAttachedObservable.remove(this._meshAttachedObserver);\r\n }\r\n }\r\n}\r\n\r\nclass VRExperienceHelperCameraGazer extends VRExperienceHelperGazer {\r\n constructor(private _getCamera: () => Nullable, scene: Scene) {\r\n super(scene);\r\n }\r\n\r\n _getForwardRay(length: number): Ray {\r\n const camera = this._getCamera();\r\n if (camera) {\r\n return camera.getForwardRay(length);\r\n } else {\r\n return new Ray(Vector3.Zero(), Vector3.Forward());\r\n }\r\n }\r\n}\r\n\r\n/**\r\n * Event containing information after VR has been entered\r\n */\r\nexport class OnAfterEnteringVRObservableEvent {\r\n /**\r\n * If entering vr was successful\r\n */\r\n public success: boolean;\r\n}\r\n\r\n/**\r\n * Helps to quickly add VR support to an existing scene.\r\n * See https://doc.babylonjs.com/features/featuresDeepDive/cameras/webVRHelper\r\n * @deprecated\r\n */\r\nexport class VRExperienceHelper {\r\n private _scene: Scene;\r\n private _position: Vector3;\r\n private _btnVR: Nullable;\r\n private _btnVRDisplayed: boolean;\r\n\r\n // Can the system support WebVR, even if a headset isn't plugged in?\r\n private _webVRsupported = false;\r\n // If WebVR is supported, is a headset plugged in and are we ready to present?\r\n private _webVRready = false;\r\n // Are we waiting for the requestPresent callback to complete?\r\n private _webVRrequesting = false;\r\n // Are we presenting to the headset right now? (this is the vrDevice state)\r\n private _webVRpresenting = false;\r\n // Have we entered VR? (this is the VRExperienceHelper state)\r\n private _hasEnteredVR: boolean;\r\n\r\n // Are we presenting in the fullscreen fallback?\r\n private _fullscreenVRpresenting = false;\r\n\r\n private _inputElement: Nullable;\r\n private _webVRCamera: WebVRFreeCamera;\r\n private _vrDeviceOrientationCamera: Nullable;\r\n private _deviceOrientationCamera: Nullable;\r\n private _existingCamera: Camera;\r\n\r\n private _onKeyDown: (event: KeyboardEvent) => void;\r\n private _onVrDisplayPresentChangeBind: any;\r\n private _onVRDisplayChangedBind: (eventArgs: IDisplayChangedEventArgs) => void;\r\n private _onVRRequestPresentStart: () => void;\r\n private _onVRRequestPresentComplete: (success: boolean) => void;\r\n\r\n /**\r\n * Gets or sets a boolean indicating that gaze can be enabled even if pointer lock is not engage (useful on iOS where fullscreen mode and pointer lock are not supported)\r\n */\r\n public enableGazeEvenWhenNoPointerLock = false;\r\n\r\n /**\r\n * Gets or sets a boolean indicating that the VREXperienceHelper will exit VR if double tap is detected\r\n */\r\n public exitVROnDoubleTap = true;\r\n\r\n /**\r\n * Observable raised right before entering VR.\r\n */\r\n public onEnteringVRObservable = new Observable();\r\n\r\n /**\r\n * Observable raised when entering VR has completed.\r\n */\r\n public onAfterEnteringVRObservable = new Observable();\r\n\r\n /**\r\n * Observable raised when exiting VR.\r\n */\r\n public onExitingVRObservable = new Observable();\r\n\r\n /**\r\n * Observable raised when controller mesh is loaded.\r\n */\r\n public onControllerMeshLoadedObservable = new Observable();\r\n\r\n /** Return this.onEnteringVRObservable\r\n * Note: This one is for backward compatibility. Please use onEnteringVRObservable directly\r\n */\r\n public get onEnteringVR(): Observable {\r\n return this.onEnteringVRObservable;\r\n }\r\n\r\n /** Return this.onExitingVRObservable\r\n * Note: This one is for backward compatibility. Please use onExitingVRObservable directly\r\n */\r\n public get onExitingVR(): Observable {\r\n return this.onExitingVRObservable;\r\n }\r\n\r\n /** Return this.onControllerMeshLoadedObservable\r\n * Note: This one is for backward compatibility. Please use onControllerMeshLoadedObservable directly\r\n */\r\n public get onControllerMeshLoaded(): Observable {\r\n return this.onControllerMeshLoadedObservable;\r\n }\r\n\r\n private _rayLength: number;\r\n private _useCustomVRButton: boolean = false;\r\n private _teleportationRequested: boolean = false;\r\n private _teleportActive = false;\r\n private _floorMeshName: string;\r\n private _floorMeshesCollection: Mesh[] = [];\r\n private _teleportationMode: number = VRExperienceHelper.TELEPORTATIONMODE_CONSTANTTIME;\r\n private _teleportationTime: number = 122;\r\n private _teleportationSpeed: number = 20;\r\n private _teleportationEasing: EasingFunction;\r\n private _rotationAllowed: boolean = true;\r\n private _teleportBackwardsVector = new Vector3(0, -1, -1);\r\n private _teleportationTarget: Mesh;\r\n private _isDefaultTeleportationTarget = true;\r\n private _postProcessMove: ImageProcessingPostProcess;\r\n private _teleportationFillColor: string = \"#444444\";\r\n private _teleportationBorderColor: string = \"#FFFFFF\";\r\n private _rotationAngle: number = 0;\r\n private _haloCenter = new Vector3(0, 0, 0);\r\n private _cameraGazer: VRExperienceHelperCameraGazer;\r\n private _padSensibilityUp = 0.65;\r\n private _padSensibilityDown = 0.35;\r\n\r\n private _leftController: Nullable = null;\r\n private _rightController: Nullable = null;\r\n\r\n private _gazeColor: Color3 = new Color3(0.7, 0.7, 0.7);\r\n private _laserColor: Color3 = new Color3(0.7, 0.7, 0.7);\r\n private _pickedLaserColor: Color3 = new Color3(0.2, 0.2, 1);\r\n private _pickedGazeColor: Color3 = new Color3(0, 0, 1);\r\n\r\n /**\r\n * Observable raised when a new mesh is selected based on meshSelectionPredicate\r\n */\r\n public onNewMeshSelected = new Observable();\r\n\r\n /**\r\n * Observable raised when a new mesh is selected based on meshSelectionPredicate.\r\n * This observable will provide the mesh and the controller used to select the mesh\r\n */\r\n public onMeshSelectedWithController = new Observable<{ mesh: AbstractMesh; controller: WebVRController }>();\r\n\r\n /**\r\n * Observable raised when a new mesh is picked based on meshSelectionPredicate\r\n */\r\n public onNewMeshPicked = new Observable();\r\n\r\n private _circleEase: CircleEase;\r\n\r\n /**\r\n * Observable raised before camera teleportation\r\n */\r\n public onBeforeCameraTeleport = new Observable();\r\n\r\n /**\r\n * Observable raised after camera teleportation\r\n */\r\n public onAfterCameraTeleport = new Observable();\r\n\r\n /**\r\n * Observable raised when current selected mesh gets unselected\r\n */\r\n public onSelectedMeshUnselected = new Observable();\r\n\r\n private _raySelectionPredicate: (mesh: AbstractMesh) => boolean;\r\n\r\n /**\r\n * To be optionally changed by user to define custom ray selection\r\n */\r\n public raySelectionPredicate: (mesh: AbstractMesh) => boolean;\r\n\r\n /**\r\n * To be optionally changed by user to define custom selection logic (after ray selection)\r\n */\r\n public meshSelectionPredicate: (mesh: AbstractMesh) => boolean;\r\n\r\n /**\r\n * Set teleportation enabled. If set to false camera teleportation will be disabled but camera rotation will be kept.\r\n */\r\n public teleportationEnabled: boolean = true;\r\n\r\n private _defaultHeight: number;\r\n private _teleportationInitialized = false;\r\n private _interactionsEnabled = false;\r\n private _interactionsRequested = false;\r\n private _displayGaze = true;\r\n private _displayLaserPointer = true;\r\n\r\n /**\r\n * The mesh used to display where the user is going to teleport.\r\n */\r\n public get teleportationTarget(): Mesh {\r\n return this._teleportationTarget;\r\n }\r\n\r\n /**\r\n * Sets the mesh to be used to display where the user is going to teleport.\r\n */\r\n public set teleportationTarget(value: Mesh) {\r\n if (value) {\r\n value.name = \"teleportationTarget\";\r\n this._isDefaultTeleportationTarget = false;\r\n this._teleportationTarget = value;\r\n }\r\n }\r\n\r\n /**\r\n * The mesh used to display where the user is selecting, this mesh will be cloned and set as the gazeTracker for the left and right controller\r\n * when set bakeCurrentTransformIntoVertices will be called on the mesh.\r\n * See https://doc.babylonjs.com/features/featuresDeepDive/mesh/transforms/center_origin/bakingTransforms\r\n */\r\n public get gazeTrackerMesh(): Mesh {\r\n return this._cameraGazer._gazeTracker;\r\n }\r\n\r\n public set gazeTrackerMesh(value: Mesh) {\r\n if (value) {\r\n // Dispose of existing meshes\r\n if (this._cameraGazer._gazeTracker) {\r\n this._cameraGazer._gazeTracker.dispose();\r\n }\r\n if (this._leftController && this._leftController._gazeTracker) {\r\n this._leftController._gazeTracker.dispose();\r\n }\r\n if (this._rightController && this._rightController._gazeTracker) {\r\n this._rightController._gazeTracker.dispose();\r\n }\r\n\r\n // Set and create gaze trackers on head and controllers\r\n this._cameraGazer._gazeTracker = value;\r\n this._cameraGazer._gazeTracker.bakeCurrentTransformIntoVertices();\r\n this._cameraGazer._gazeTracker.isPickable = false;\r\n this._cameraGazer._gazeTracker.isVisible = false;\r\n this._cameraGazer._gazeTracker.name = \"gazeTracker\";\r\n if (this._leftController) {\r\n this._leftController._gazeTracker = this._cameraGazer._gazeTracker.clone(\"gazeTracker\") as Mesh;\r\n }\r\n\r\n if (this._rightController) {\r\n this._rightController._gazeTracker = this._cameraGazer._gazeTracker.clone(\"gazeTracker\") as Mesh;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * If the gaze trackers scale should be updated to be constant size when pointing at near/far meshes\r\n */\r\n public updateGazeTrackerScale = true;\r\n /**\r\n * If the gaze trackers color should be updated when selecting meshes\r\n */\r\n public updateGazeTrackerColor = true;\r\n /**\r\n * If the controller laser color should be updated when selecting meshes\r\n */\r\n public updateControllerLaserColor = true;\r\n\r\n /**\r\n * The gaze tracking mesh corresponding to the left controller\r\n */\r\n public get leftControllerGazeTrackerMesh(): Nullable {\r\n if (this._leftController) {\r\n return this._leftController._gazeTracker;\r\n }\r\n return null;\r\n }\r\n\r\n /**\r\n * The gaze tracking mesh corresponding to the right controller\r\n */\r\n public get rightControllerGazeTrackerMesh(): Nullable {\r\n if (this._rightController) {\r\n return this._rightController._gazeTracker;\r\n }\r\n return null;\r\n }\r\n\r\n /**\r\n * If the ray of the gaze should be displayed.\r\n */\r\n public get displayGaze(): boolean {\r\n return this._displayGaze;\r\n }\r\n\r\n /**\r\n * Sets if the ray of the gaze should be displayed.\r\n */\r\n public set displayGaze(value: boolean) {\r\n this._displayGaze = value;\r\n if (!value) {\r\n this._cameraGazer._gazeTracker.isVisible = false;\r\n\r\n if (this._leftController) {\r\n this._leftController._gazeTracker.isVisible = false;\r\n }\r\n\r\n if (this._rightController) {\r\n this._rightController._gazeTracker.isVisible = false;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * If the ray of the LaserPointer should be displayed.\r\n */\r\n public get displayLaserPointer(): boolean {\r\n return this._displayLaserPointer;\r\n }\r\n\r\n /**\r\n * Sets if the ray of the LaserPointer should be displayed.\r\n */\r\n public set displayLaserPointer(value: boolean) {\r\n this._displayLaserPointer = value;\r\n if (!value) {\r\n if (this._rightController) {\r\n this._rightController._deactivatePointer();\r\n this._rightController._gazeTracker.isVisible = false;\r\n }\r\n if (this._leftController) {\r\n this._leftController._deactivatePointer();\r\n this._leftController._gazeTracker.isVisible = false;\r\n }\r\n } else {\r\n if (this._rightController) {\r\n this._rightController._activatePointer();\r\n }\r\n if (this._leftController) {\r\n this._leftController._activatePointer();\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * The deviceOrientationCamera used as the camera when not in VR.\r\n */\r\n public get deviceOrientationCamera(): Nullable {\r\n return this._deviceOrientationCamera;\r\n }\r\n\r\n /**\r\n * Based on the current WebVR support, returns the current VR camera used.\r\n */\r\n public get currentVRCamera(): Nullable {\r\n if (this._webVRready) {\r\n return this._webVRCamera;\r\n } else {\r\n return this._scene.activeCamera;\r\n }\r\n }\r\n\r\n /**\r\n * The webVRCamera which is used when in VR.\r\n */\r\n public get webVRCamera(): WebVRFreeCamera {\r\n return this._webVRCamera;\r\n }\r\n\r\n /**\r\n * The deviceOrientationCamera that is used as a fallback when vr device is not connected.\r\n */\r\n public get vrDeviceOrientationCamera(): Nullable {\r\n return this._vrDeviceOrientationCamera;\r\n }\r\n\r\n /**\r\n * The html button that is used to trigger entering into VR.\r\n */\r\n public get vrButton(): Nullable {\r\n return this._btnVR;\r\n }\r\n\r\n private get _teleportationRequestInitiated(): boolean {\r\n const result =\r\n this._cameraGazer._teleportationRequestInitiated ||\r\n (this._leftController !== null && this._leftController._teleportationRequestInitiated) ||\r\n (this._rightController !== null && this._rightController._teleportationRequestInitiated);\r\n return result;\r\n }\r\n\r\n /**\r\n * Defines whether or not Pointer lock should be requested when switching to\r\n * full screen.\r\n */\r\n public requestPointerLockOnFullScreen = true;\r\n\r\n // XR\r\n\r\n /**\r\n * If asking to force XR, this will be populated with the default xr experience\r\n */\r\n public xr: WebXRDefaultExperience;\r\n\r\n /**\r\n * Was the XR test done already. If this is true AND this.xr exists, xr is initialized.\r\n * If this is true and no this.xr, xr exists but is not supported, using WebVR.\r\n */\r\n public xrTestDone: boolean = false;\r\n\r\n /**\r\n * Instantiates a VRExperienceHelper.\r\n * Helps to quickly add VR support to an existing scene.\r\n * @param scene The scene the VRExperienceHelper belongs to.\r\n * @param webVROptions Options to modify the vr experience helper's behavior.\r\n */\r\n constructor(\r\n scene: Scene,\r\n /** Options to modify the vr experience helper's behavior. */\r\n public webVROptions: VRExperienceHelperOptions = {}\r\n ) {\r\n Logger.Warn(\"WebVR is deprecated. Please avoid using this experience helper and use the WebXR experience helper instead\");\r\n this._scene = scene;\r\n this._inputElement = scene.getEngine().getInputElement();\r\n\r\n // check for VR support:\r\n\r\n const vrSupported = \"getVRDisplays\" in navigator;\r\n // no VR support? force XR but only when it is not set because web vr can work without the getVRDisplays\r\n if (!vrSupported && webVROptions.useXR === undefined) {\r\n webVROptions.useXR = true;\r\n }\r\n\r\n // Parse options\r\n if (webVROptions.createFallbackVRDeviceOrientationFreeCamera === undefined) {\r\n webVROptions.createFallbackVRDeviceOrientationFreeCamera = true;\r\n }\r\n if (webVROptions.createDeviceOrientationCamera === undefined) {\r\n webVROptions.createDeviceOrientationCamera = true;\r\n }\r\n if (webVROptions.laserToggle === undefined) {\r\n webVROptions.laserToggle = true;\r\n }\r\n if (webVROptions.defaultHeight === undefined) {\r\n webVROptions.defaultHeight = 1.7;\r\n }\r\n if (webVROptions.useCustomVRButton) {\r\n this._useCustomVRButton = true;\r\n if (webVROptions.customVRButton) {\r\n this._btnVR = webVROptions.customVRButton;\r\n }\r\n }\r\n if (webVROptions.rayLength) {\r\n this._rayLength = webVROptions.rayLength;\r\n }\r\n this._defaultHeight = webVROptions.defaultHeight;\r\n\r\n if (webVROptions.positionScale) {\r\n this._rayLength *= webVROptions.positionScale;\r\n this._defaultHeight *= webVROptions.positionScale;\r\n }\r\n\r\n this._hasEnteredVR = false;\r\n\r\n // Set position\r\n if (this._scene.activeCamera) {\r\n this._position = this._scene.activeCamera.position.clone();\r\n } else {\r\n this._position = new Vector3(0, this._defaultHeight, 0);\r\n }\r\n\r\n // Set non-vr camera\r\n if (webVROptions.createDeviceOrientationCamera || !this._scene.activeCamera) {\r\n this._deviceOrientationCamera = new DeviceOrientationCamera(\"deviceOrientationVRHelper\", this._position.clone(), scene);\r\n\r\n // Copy data from existing camera\r\n if (this._scene.activeCamera) {\r\n this._deviceOrientationCamera.minZ = this._scene.activeCamera.minZ;\r\n this._deviceOrientationCamera.maxZ = this._scene.activeCamera.maxZ;\r\n // Set rotation from previous camera\r\n if (this._scene.activeCamera instanceof TargetCamera && this._scene.activeCamera.rotation) {\r\n const targetCamera = this._scene.activeCamera;\r\n if (targetCamera.rotationQuaternion) {\r\n this._deviceOrientationCamera.rotationQuaternion.copyFrom(targetCamera.rotationQuaternion);\r\n } else {\r\n this._deviceOrientationCamera.rotationQuaternion.copyFrom(\r\n Quaternion.RotationYawPitchRoll(targetCamera.rotation.y, targetCamera.rotation.x, targetCamera.rotation.z)\r\n );\r\n }\r\n this._deviceOrientationCamera.rotation = targetCamera.rotation.clone();\r\n }\r\n }\r\n\r\n this._scene.activeCamera = this._deviceOrientationCamera;\r\n if (this._inputElement) {\r\n this._scene.activeCamera.attachControl();\r\n }\r\n } else {\r\n this._existingCamera = this._scene.activeCamera;\r\n }\r\n\r\n if (this.webVROptions.useXR && (navigator as any).xr) {\r\n // force-check XR session support\r\n WebXRSessionManager.IsSessionSupportedAsync(\"immersive-vr\").then((supported) => {\r\n if (supported) {\r\n Logger.Log(\"Using WebXR. It is recommended to use the WebXRDefaultExperience directly\");\r\n // it is possible to use XR, let's do it!\r\n scene\r\n .createDefaultXRExperienceAsync({\r\n floorMeshes: webVROptions.floorMeshes || [],\r\n })\r\n .then((xr) => {\r\n this.xr = xr;\r\n // connect observables\r\n this.xrTestDone = true;\r\n\r\n this._cameraGazer = new VRExperienceHelperCameraGazer(() => {\r\n return this.xr.baseExperience.camera;\r\n }, scene);\r\n\r\n this.xr.baseExperience.onStateChangedObservable.add((state) => {\r\n // support for entering / exiting\r\n switch (state) {\r\n case WebXRState.ENTERING_XR:\r\n this.onEnteringVRObservable.notifyObservers(this);\r\n if (!this._interactionsEnabled) {\r\n this.xr.pointerSelection.detach();\r\n }\r\n this.xr.pointerSelection.displayLaserPointer = this._displayLaserPointer;\r\n break;\r\n case WebXRState.EXITING_XR:\r\n this.onExitingVRObservable.notifyObservers(this);\r\n\r\n // resize to update width and height when exiting vr exits fullscreen\r\n this._scene.getEngine().resize();\r\n break;\r\n case WebXRState.IN_XR:\r\n this._hasEnteredVR = true;\r\n break;\r\n case WebXRState.NOT_IN_XR:\r\n this._hasEnteredVR = false;\r\n break;\r\n }\r\n });\r\n });\r\n } else {\r\n // XR not supported (thou exists), continue WebVR init\r\n this._completeVRInit(scene, webVROptions);\r\n }\r\n });\r\n } else {\r\n // no XR, continue init synchronous\r\n this._completeVRInit(scene, webVROptions);\r\n }\r\n }\r\n\r\n private _completeVRInit(scene: Scene, webVROptions: VRExperienceHelperOptions): void {\r\n this.xrTestDone = true;\r\n // Create VR cameras\r\n if (webVROptions.createFallbackVRDeviceOrientationFreeCamera) {\r\n if (webVROptions.useMultiview) {\r\n if (!webVROptions.vrDeviceOrientationCameraMetrics) {\r\n webVROptions.vrDeviceOrientationCameraMetrics = VRCameraMetrics.GetDefault();\r\n }\r\n webVROptions.vrDeviceOrientationCameraMetrics.multiviewEnabled = true;\r\n }\r\n this._vrDeviceOrientationCamera = new VRDeviceOrientationFreeCamera(\r\n \"VRDeviceOrientationVRHelper\",\r\n this._position,\r\n this._scene,\r\n true,\r\n webVROptions.vrDeviceOrientationCameraMetrics\r\n );\r\n this._vrDeviceOrientationCamera.angularSensibility = Number.MAX_VALUE;\r\n }\r\n this._webVRCamera = new WebVRFreeCamera(\"WebVRHelper\", this._position, this._scene, webVROptions);\r\n this._webVRCamera.useStandingMatrix();\r\n\r\n this._cameraGazer = new VRExperienceHelperCameraGazer(() => {\r\n return this.currentVRCamera;\r\n }, scene);\r\n // Create default button\r\n if (!this._useCustomVRButton) {\r\n this._btnVR = document.createElement(\"BUTTON\");\r\n this._btnVR.className = \"babylonVRicon\";\r\n this._btnVR.id = \"babylonVRiconbtn\";\r\n this._btnVR.title = \"Click to switch to VR\";\r\n const url = !window.SVGSVGElement\r\n ? \"https://cdn.babylonjs.com/Assets/vrButton.png\"\r\n : \"data:image/svg+xml;charset=UTF-8,%3Csvg%20xmlns%3D%22http%3A//www.w3.org/2000/svg%22%20width%3D%222048%22%20height%3D%221152%22%20viewBox%3D%220%200%202048%201152%22%20version%3D%221.1%22%3E%3Cpath%20transform%3D%22rotate%28180%201024%2C576.0000000000001%29%22%20d%3D%22m1109%2C896q17%2C0%2030%2C-12t13%2C-30t-12.5%2C-30.5t-30.5%2C-12.5l-170%2C0q-18%2C0%20-30.5%2C12.5t-12.5%2C30.5t13%2C30t30%2C12l170%2C0zm-85%2C256q59%2C0%20132.5%2C-1.5t154.5%2C-5.5t164.5%2C-11.5t163%2C-20t150%2C-30t124.5%2C-41.5q23%2C-11%2042%2C-24t38%2C-30q27%2C-25%2041%2C-61.5t14%2C-72.5l0%2C-257q0%2C-123%20-47%2C-232t-128%2C-190t-190%2C-128t-232%2C-47l-81%2C0q-37%2C0%20-68.5%2C14t-60.5%2C34.5t-55.5%2C45t-53%2C45t-53%2C34.5t-55.5%2C14t-55.5%2C-14t-53%2C-34.5t-53%2C-45t-55.5%2C-45t-60.5%2C-34.5t-68.5%2C-14l-81%2C0q-123%2C0%20-232%2C47t-190%2C128t-128%2C190t-47%2C232l0%2C257q0%2C68%2038%2C115t97%2C73q54%2C24%20124.5%2C41.5t150%2C30t163%2C20t164.5%2C11.5t154.5%2C5.5t132.5%2C1.5zm939%2C-298q0%2C39%20-24.5%2C67t-58.5%2C42q-54%2C23%20-122%2C39.5t-143.5%2C28t-155.5%2C19t-157%2C11t-148.5%2C5t-129.5%2C1.5q-59%2C0%20-130%2C-1.5t-148%2C-5t-157%2C-11t-155.5%2C-19t-143.5%2C-28t-122%2C-39.5q-34%2C-14%20-58.5%2C-42t-24.5%2C-67l0%2C-257q0%2C-106%2040.5%2C-199t110%2C-162.5t162.5%2C-109.5t199%2C-40l81%2C0q27%2C0%2052%2C14t50%2C34.5t51%2C44.5t55.5%2C44.5t63.5%2C34.5t74%2C14t74%2C-14t63.5%2C-34.5t55.5%2C-44.5t51%2C-44.5t50%2C-34.5t52%2C-14l14%2C0q37%2C0%2070%2C0.5t64.5%2C4.5t63.5%2C12t68%2C23q71%2C30%20128.5%2C78.5t98.5%2C110t63.5%2C133.5t22.5%2C149l0%2C257z%22%20fill%3D%22white%22%20/%3E%3C/svg%3E%0A\";\r\n let css =\r\n \".babylonVRicon { position: absolute; right: 20px; height: 50px; width: 80px; background-color: rgba(51,51,51,0.7); background-image: url(\" +\r\n url +\r\n \"); background-size: 80%; background-repeat:no-repeat; background-position: center; border: none; outline: none; transition: transform 0.125s ease-out } .babylonVRicon:hover { transform: scale(1.05) } .babylonVRicon:active {background-color: rgba(51,51,51,1) } .babylonVRicon:focus {background-color: rgba(51,51,51,1) }\";\r\n css += \".babylonVRicon.vrdisplaypresenting { display: none; }\";\r\n // TODO: Add user feedback so that they know what state the VRDisplay is in (disconnected, connected, entering-VR)\r\n // css += \".babylonVRicon.vrdisplaysupported { }\";\r\n // css += \".babylonVRicon.vrdisplayready { }\";\r\n // css += \".babylonVRicon.vrdisplayrequesting { }\";\r\n\r\n const style = document.createElement(\"style\");\r\n style.appendChild(document.createTextNode(css));\r\n document.getElementsByTagName(\"head\")[0].appendChild(style);\r\n\r\n this._moveButtonToBottomRight();\r\n }\r\n\r\n // VR button click event\r\n if (this._btnVR) {\r\n this._btnVR.addEventListener(\"click\", () => {\r\n if (!this.isInVRMode) {\r\n this.enterVR();\r\n } else {\r\n this._scene.getEngine().disableVR();\r\n }\r\n });\r\n }\r\n\r\n // Window events\r\n\r\n const hostWindow = this._scene.getEngine().getHostWindow();\r\n if (!hostWindow) {\r\n return;\r\n }\r\n\r\n hostWindow.addEventListener(\"resize\", this._onResize);\r\n document.addEventListener(\"fullscreenchange\", this._onFullscreenChange, false);\r\n\r\n // Display vr button when headset is connected\r\n if (webVROptions.createFallbackVRDeviceOrientationFreeCamera) {\r\n this._displayVRButton();\r\n } else {\r\n this._scene.getEngine().onVRDisplayChangedObservable.add((e) => {\r\n if (e.vrDisplay) {\r\n this._displayVRButton();\r\n }\r\n });\r\n }\r\n\r\n // Exiting VR mode using 'ESC' key on desktop\r\n this._onKeyDown = (event: KeyboardEvent) => {\r\n if (event.keyCode === 27 && this.isInVRMode) {\r\n this.exitVR();\r\n }\r\n };\r\n document.addEventListener(\"keydown\", this._onKeyDown);\r\n\r\n // Exiting VR mode double tapping the touch screen\r\n this._scene.onPrePointerObservable.add(\r\n () => {\r\n if (this._hasEnteredVR && this.exitVROnDoubleTap) {\r\n this.exitVR();\r\n if (this._fullscreenVRpresenting) {\r\n this._scene.getEngine().exitFullscreen();\r\n }\r\n }\r\n },\r\n PointerEventTypes.POINTERDOUBLETAP,\r\n false\r\n );\r\n\r\n // Listen for WebVR display changes\r\n this._onVRDisplayChangedBind = (eventArgs: IDisplayChangedEventArgs) => this._onVRDisplayChanged(eventArgs);\r\n this._onVrDisplayPresentChangeBind = () => this._onVrDisplayPresentChange();\r\n this._onVRRequestPresentStart = () => {\r\n this._webVRrequesting = true;\r\n this._updateButtonVisibility();\r\n };\r\n this._onVRRequestPresentComplete = () => {\r\n this._webVRrequesting = false;\r\n this._updateButtonVisibility();\r\n };\r\n scene.getEngine().onVRDisplayChangedObservable.add(this._onVRDisplayChangedBind);\r\n scene.getEngine().onVRRequestPresentStart.add(this._onVRRequestPresentStart);\r\n scene.getEngine().onVRRequestPresentComplete.add(this._onVRRequestPresentComplete);\r\n hostWindow.addEventListener(\"vrdisplaypresentchange\", this._onVrDisplayPresentChangeBind);\r\n\r\n scene.onDisposeObservable.add(() => {\r\n this.dispose();\r\n });\r\n\r\n // Gamepad connection events\r\n this._webVRCamera.onControllerMeshLoadedObservable.add((webVRController) => this._onDefaultMeshLoaded(webVRController));\r\n this._scene.gamepadManager.onGamepadConnectedObservable.add(this._onNewGamepadConnected);\r\n this._scene.gamepadManager.onGamepadDisconnectedObservable.add(this._onNewGamepadDisconnected);\r\n\r\n this._updateButtonVisibility();\r\n\r\n //create easing functions\r\n this._circleEase = new CircleEase();\r\n this._circleEase.setEasingMode(EasingFunction.EASINGMODE_EASEINOUT);\r\n this._teleportationEasing = this._circleEase;\r\n\r\n // Allow clicking in the vrDeviceOrientationCamera\r\n scene.onPointerObservable.add((e) => {\r\n if (this._interactionsEnabled) {\r\n if (scene.activeCamera === this.vrDeviceOrientationCamera && (e.event as IPointerEvent).pointerType === \"mouse\") {\r\n if (e.type === PointerEventTypes.POINTERDOWN) {\r\n this._cameraGazer._selectionPointerDown();\r\n } else if (e.type === PointerEventTypes.POINTERUP) {\r\n this._cameraGazer._selectionPointerUp();\r\n }\r\n }\r\n }\r\n });\r\n\r\n if (this.webVROptions.floorMeshes) {\r\n this.enableTeleportation({ floorMeshes: this.webVROptions.floorMeshes });\r\n }\r\n }\r\n\r\n // Raised when one of the controller has loaded successfully its associated default mesh\r\n private _onDefaultMeshLoaded(webVRController: WebVRController) {\r\n if (this._leftController && this._leftController.webVRController == webVRController) {\r\n if (webVRController.mesh) {\r\n this._leftController._setLaserPointerParent(webVRController.mesh);\r\n }\r\n }\r\n if (this._rightController && this._rightController.webVRController == webVRController) {\r\n if (webVRController.mesh) {\r\n this._rightController._setLaserPointerParent(webVRController.mesh);\r\n }\r\n }\r\n\r\n try {\r\n this.onControllerMeshLoadedObservable.notifyObservers(webVRController);\r\n } catch (err) {\r\n Logger.Warn(\"Error in your custom logic onControllerMeshLoaded: \" + err);\r\n }\r\n }\r\n\r\n private _onResize = () => {\r\n this._moveButtonToBottomRight();\r\n if (this._fullscreenVRpresenting && this._webVRready) {\r\n this.exitVR();\r\n }\r\n };\r\n\r\n private _onFullscreenChange = () => {\r\n this._fullscreenVRpresenting = !!document.fullscreenElement;\r\n if (!this._fullscreenVRpresenting && this._inputElement) {\r\n this.exitVR();\r\n if (!this._useCustomVRButton && this._btnVR) {\r\n this._btnVR.style.top = this._inputElement.offsetTop + this._inputElement.offsetHeight - 70 + \"px\";\r\n this._btnVR.style.left = this._inputElement.offsetLeft + this._inputElement.offsetWidth - 100 + \"px\";\r\n // make sure the button is visible after setting its position\r\n this._updateButtonVisibility();\r\n }\r\n }\r\n };\r\n\r\n /**\r\n * Gets a value indicating if we are currently in VR mode.\r\n */\r\n public get isInVRMode(): boolean {\r\n return (this.xr && this.webVROptions.useXR && this.xr.baseExperience.state === WebXRState.IN_XR) || this._webVRpresenting || this._fullscreenVRpresenting;\r\n }\r\n\r\n private _onVrDisplayPresentChange() {\r\n const vrDisplay = this._scene.getEngine().getVRDevice();\r\n if (vrDisplay) {\r\n const wasPresenting = this._webVRpresenting;\r\n this._webVRpresenting = vrDisplay.isPresenting;\r\n\r\n if (wasPresenting && !this._webVRpresenting) {\r\n this.exitVR();\r\n }\r\n } else {\r\n Logger.Warn(\"Detected VRDisplayPresentChange on an unknown VRDisplay. Did you can enterVR on the vrExperienceHelper?\");\r\n }\r\n\r\n this._updateButtonVisibility();\r\n }\r\n\r\n private _onVRDisplayChanged(eventArgs: IDisplayChangedEventArgs) {\r\n this._webVRsupported = eventArgs.vrSupported;\r\n this._webVRready = !!eventArgs.vrDisplay;\r\n this._webVRpresenting = eventArgs.vrDisplay && eventArgs.vrDisplay.isPresenting;\r\n\r\n this._updateButtonVisibility();\r\n }\r\n\r\n private _moveButtonToBottomRight() {\r\n if (this._inputElement && !this._useCustomVRButton && this._btnVR) {\r\n const rect: ClientRect = this._inputElement.getBoundingClientRect();\r\n this._btnVR.style.top = rect.top + rect.height - 70 + \"px\";\r\n this._btnVR.style.left = rect.left + rect.width - 100 + \"px\";\r\n }\r\n }\r\n\r\n private _displayVRButton() {\r\n if (!this._useCustomVRButton && !this._btnVRDisplayed && this._btnVR) {\r\n document.body.appendChild(this._btnVR);\r\n this._btnVRDisplayed = true;\r\n }\r\n }\r\n\r\n private _updateButtonVisibility() {\r\n if (!this._btnVR || this._useCustomVRButton) {\r\n return;\r\n }\r\n this._btnVR.className = \"babylonVRicon\";\r\n if (this.isInVRMode) {\r\n this._btnVR.className += \" vrdisplaypresenting\";\r\n } else {\r\n if (this._webVRready) {\r\n this._btnVR.className += \" vrdisplayready\";\r\n }\r\n if (this._webVRsupported) {\r\n this._btnVR.className += \" vrdisplaysupported\";\r\n }\r\n if (this._webVRrequesting) {\r\n this._btnVR.className += \" vrdisplayrequesting\";\r\n }\r\n }\r\n }\r\n\r\n private _cachedAngularSensibility = { angularSensibilityX: null, angularSensibilityY: null, angularSensibility: null };\r\n /**\r\n * Attempt to enter VR. If a headset is connected and ready, will request present on that.\r\n * Otherwise, will use the fullscreen API.\r\n */\r\n public enterVR() {\r\n if (this.xr) {\r\n this.xr.baseExperience.enterXRAsync(\"immersive-vr\", \"local-floor\", this.xr.renderTarget);\r\n return;\r\n }\r\n if (this.onEnteringVRObservable) {\r\n try {\r\n this.onEnteringVRObservable.notifyObservers(this);\r\n } catch (err) {\r\n Logger.Warn(\"Error in your custom logic onEnteringVR: \" + err);\r\n }\r\n }\r\n\r\n if (this._scene.activeCamera) {\r\n this._position = this._scene.activeCamera.position.clone();\r\n\r\n if (this.vrDeviceOrientationCamera) {\r\n this.vrDeviceOrientationCamera.rotation = Quaternion.FromRotationMatrix(this._scene.activeCamera.getWorldMatrix().getRotationMatrix()).toEulerAngles();\r\n this.vrDeviceOrientationCamera.angularSensibility = 2000;\r\n }\r\n if (this.webVRCamera) {\r\n const currentYRotation = this.webVRCamera.deviceRotationQuaternion.toEulerAngles().y;\r\n const desiredYRotation = Quaternion.FromRotationMatrix(this._scene.activeCamera.getWorldMatrix().getRotationMatrix()).toEulerAngles().y;\r\n const delta = desiredYRotation - currentYRotation;\r\n const currentGlobalRotation = this.webVRCamera.rotationQuaternion.toEulerAngles().y;\r\n this.webVRCamera.rotationQuaternion = Quaternion.FromEulerAngles(0, currentGlobalRotation + delta, 0);\r\n }\r\n\r\n // make sure that we return to the last active camera\r\n this._existingCamera = this._scene.activeCamera;\r\n\r\n // Remove and cache angular sensability to avoid camera rotation when in VR\r\n if ((this._existingCamera).angularSensibilityX) {\r\n this._cachedAngularSensibility.angularSensibilityX = (this._existingCamera).angularSensibilityX;\r\n (this._existingCamera).angularSensibilityX = Number.MAX_VALUE;\r\n }\r\n if ((this._existingCamera).angularSensibilityY) {\r\n this._cachedAngularSensibility.angularSensibilityY = (this._existingCamera).angularSensibilityY;\r\n (this._existingCamera).angularSensibilityY = Number.MAX_VALUE;\r\n }\r\n if ((this._existingCamera).angularSensibility) {\r\n this._cachedAngularSensibility.angularSensibility = (this._existingCamera).angularSensibility;\r\n (this._existingCamera).angularSensibility = Number.MAX_VALUE;\r\n }\r\n }\r\n\r\n if (this._webVRrequesting) {\r\n return;\r\n }\r\n\r\n // If WebVR is supported and a headset is connected\r\n if (this._webVRready) {\r\n if (!this._webVRpresenting) {\r\n this._scene.getEngine().onVRRequestPresentComplete.addOnce((result) => {\r\n this.onAfterEnteringVRObservable.notifyObservers({ success: result });\r\n });\r\n this._webVRCamera.position = this._position;\r\n this._scene.activeCamera = this._webVRCamera;\r\n }\r\n } else if (this._vrDeviceOrientationCamera) {\r\n this._vrDeviceOrientationCamera.position = this._position;\r\n if (this._scene.activeCamera) {\r\n this._vrDeviceOrientationCamera.minZ = this._scene.activeCamera.minZ;\r\n }\r\n this._scene.activeCamera = this._vrDeviceOrientationCamera;\r\n this._scene.getEngine().enterFullscreen(this.requestPointerLockOnFullScreen);\r\n this._updateButtonVisibility();\r\n this._vrDeviceOrientationCamera.onViewMatrixChangedObservable.addOnce(() => {\r\n this.onAfterEnteringVRObservable.notifyObservers({ success: true });\r\n });\r\n }\r\n\r\n if (this._scene.activeCamera && this._inputElement) {\r\n this._scene.activeCamera.attachControl();\r\n }\r\n\r\n if (this._interactionsEnabled) {\r\n this._scene.registerBeforeRender(this._beforeRender);\r\n }\r\n\r\n if (this._displayLaserPointer) {\r\n [this._leftController, this._rightController].forEach((controller) => {\r\n if (controller) {\r\n controller._activatePointer();\r\n }\r\n });\r\n }\r\n\r\n this._hasEnteredVR = true;\r\n }\r\n\r\n /**\r\n * Attempt to exit VR, or fullscreen.\r\n */\r\n public exitVR() {\r\n if (this.xr) {\r\n this.xr.baseExperience.exitXRAsync();\r\n return;\r\n }\r\n if (this._hasEnteredVR) {\r\n if (this.onExitingVRObservable) {\r\n try {\r\n this.onExitingVRObservable.notifyObservers(this);\r\n } catch (err) {\r\n Logger.Warn(\"Error in your custom logic onExitingVR: \" + err);\r\n }\r\n }\r\n if (this._webVRpresenting) {\r\n this._scene.getEngine().disableVR();\r\n }\r\n if (this._scene.activeCamera) {\r\n this._position = this._scene.activeCamera.position.clone();\r\n }\r\n\r\n if (this.vrDeviceOrientationCamera) {\r\n this.vrDeviceOrientationCamera.angularSensibility = Number.MAX_VALUE;\r\n }\r\n\r\n if (this._deviceOrientationCamera) {\r\n this._deviceOrientationCamera.position = this._position;\r\n this._scene.activeCamera = this._deviceOrientationCamera;\r\n\r\n // Restore angular sensibility\r\n if (this._cachedAngularSensibility.angularSensibilityX) {\r\n (this._deviceOrientationCamera).angularSensibilityX = this._cachedAngularSensibility.angularSensibilityX;\r\n this._cachedAngularSensibility.angularSensibilityX = null;\r\n }\r\n if (this._cachedAngularSensibility.angularSensibilityY) {\r\n (this._deviceOrientationCamera).angularSensibilityY = this._cachedAngularSensibility.angularSensibilityY;\r\n this._cachedAngularSensibility.angularSensibilityY = null;\r\n }\r\n if (this._cachedAngularSensibility.angularSensibility) {\r\n (this._deviceOrientationCamera).angularSensibility = this._cachedAngularSensibility.angularSensibility;\r\n this._cachedAngularSensibility.angularSensibility = null;\r\n }\r\n } else if (this._existingCamera) {\r\n this._existingCamera.position = this._position;\r\n this._scene.activeCamera = this._existingCamera;\r\n if (this._inputElement) {\r\n this._scene.activeCamera.attachControl();\r\n }\r\n\r\n // Restore angular sensibility\r\n if (this._cachedAngularSensibility.angularSensibilityX) {\r\n (this._existingCamera).angularSensibilityX = this._cachedAngularSensibility.angularSensibilityX;\r\n this._cachedAngularSensibility.angularSensibilityX = null;\r\n }\r\n if (this._cachedAngularSensibility.angularSensibilityY) {\r\n (this._existingCamera).angularSensibilityY = this._cachedAngularSensibility.angularSensibilityY;\r\n this._cachedAngularSensibility.angularSensibilityY = null;\r\n }\r\n if (this._cachedAngularSensibility.angularSensibility) {\r\n (this._existingCamera).angularSensibility = this._cachedAngularSensibility.angularSensibility;\r\n this._cachedAngularSensibility.angularSensibility = null;\r\n }\r\n }\r\n\r\n this._updateButtonVisibility();\r\n\r\n if (this._interactionsEnabled) {\r\n this._scene.unregisterBeforeRender(this._beforeRender);\r\n this._cameraGazer._gazeTracker.isVisible = false;\r\n if (this._leftController) {\r\n this._leftController._gazeTracker.isVisible = false;\r\n }\r\n if (this._rightController) {\r\n this._rightController._gazeTracker.isVisible = false;\r\n }\r\n }\r\n\r\n // resize to update width and height when exiting vr exits fullscreen\r\n this._scene.getEngine().resize();\r\n\r\n [this._leftController, this._rightController].forEach((controller) => {\r\n if (controller) {\r\n controller._deactivatePointer();\r\n }\r\n });\r\n\r\n this._hasEnteredVR = false;\r\n\r\n // Update engine state to re enable non-vr camera input\r\n const engine = this._scene.getEngine();\r\n if (engine._onVrDisplayPresentChange) {\r\n engine._onVrDisplayPresentChange();\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * The position of the vr experience helper.\r\n */\r\n public get position(): Vector3 {\r\n return this._position;\r\n }\r\n\r\n /**\r\n * Sets the position of the vr experience helper.\r\n */\r\n public set position(value: Vector3) {\r\n this._position = value;\r\n\r\n if (this._scene.activeCamera) {\r\n this._scene.activeCamera.position = value;\r\n }\r\n }\r\n\r\n /**\r\n * Enables controllers and user interactions such as selecting and object or clicking on an object.\r\n */\r\n public enableInteractions() {\r\n if (!this._interactionsEnabled) {\r\n this._interactionsRequested = true;\r\n\r\n // in XR it is enabled by default, but just to make sure, re-attach\r\n if (this.xr) {\r\n if (this.xr.baseExperience.state === WebXRState.IN_XR) {\r\n this.xr.pointerSelection.attach();\r\n }\r\n return;\r\n }\r\n\r\n if (this._leftController) {\r\n this._enableInteractionOnController(this._leftController);\r\n }\r\n\r\n if (this._rightController) {\r\n this._enableInteractionOnController(this._rightController);\r\n }\r\n\r\n this.raySelectionPredicate = (mesh) => {\r\n return mesh.isVisible && (mesh.isPickable || mesh.name === this._floorMeshName);\r\n };\r\n\r\n this.meshSelectionPredicate = () => {\r\n return true;\r\n };\r\n\r\n this._raySelectionPredicate = (mesh) => {\r\n if (\r\n this._isTeleportationFloor(mesh) ||\r\n (mesh.name.indexOf(\"gazeTracker\") === -1 && mesh.name.indexOf(\"teleportationTarget\") === -1 && mesh.name.indexOf(\"torusTeleportation\") === -1)\r\n ) {\r\n return this.raySelectionPredicate(mesh);\r\n }\r\n return false;\r\n };\r\n\r\n this._interactionsEnabled = true;\r\n }\r\n }\r\n\r\n private get _noControllerIsActive() {\r\n return !(this._leftController && this._leftController._activePointer) && !(this._rightController && this._rightController._activePointer);\r\n }\r\n\r\n private _beforeRender = () => {\r\n if (this._leftController && this._leftController._activePointer) {\r\n this._castRayAndSelectObject(this._leftController);\r\n }\r\n\r\n if (this._rightController && this._rightController._activePointer) {\r\n this._castRayAndSelectObject(this._rightController);\r\n }\r\n\r\n if (this._noControllerIsActive && (this._scene.getEngine().isPointerLock || this.enableGazeEvenWhenNoPointerLock)) {\r\n this._castRayAndSelectObject(this._cameraGazer);\r\n } else {\r\n this._cameraGazer._gazeTracker.isVisible = false;\r\n }\r\n };\r\n\r\n private _isTeleportationFloor(mesh: AbstractMesh): boolean {\r\n for (let i = 0; i < this._floorMeshesCollection.length; i++) {\r\n if (this._floorMeshesCollection[i].id === mesh.id) {\r\n return true;\r\n }\r\n }\r\n if (this._floorMeshName && mesh.name === this._floorMeshName) {\r\n return true;\r\n }\r\n return false;\r\n }\r\n\r\n /**\r\n * Adds a floor mesh to be used for teleportation.\r\n * @param floorMesh the mesh to be used for teleportation.\r\n */\r\n public addFloorMesh(floorMesh: Mesh): void {\r\n if (!this._floorMeshesCollection) {\r\n return;\r\n }\r\n\r\n if (this._floorMeshesCollection.indexOf(floorMesh) > -1) {\r\n return;\r\n }\r\n\r\n this._floorMeshesCollection.push(floorMesh);\r\n }\r\n\r\n /**\r\n * Removes a floor mesh from being used for teleportation.\r\n * @param floorMesh the mesh to be removed.\r\n */\r\n public removeFloorMesh(floorMesh: Mesh): void {\r\n if (!this._floorMeshesCollection) {\r\n return;\r\n }\r\n\r\n const meshIndex = this._floorMeshesCollection.indexOf(floorMesh);\r\n if (meshIndex !== -1) {\r\n this._floorMeshesCollection.splice(meshIndex, 1);\r\n }\r\n }\r\n\r\n /**\r\n * Enables interactions and teleportation using the VR controllers and gaze.\r\n * @param vrTeleportationOptions options to modify teleportation behavior.\r\n */\r\n public enableTeleportation(vrTeleportationOptions: VRTeleportationOptions = {}) {\r\n if (!this._teleportationInitialized) {\r\n this._teleportationRequested = true;\r\n\r\n this.enableInteractions();\r\n\r\n if (this.webVROptions.useXR && (vrTeleportationOptions.floorMeshes || vrTeleportationOptions.floorMeshName)) {\r\n const floorMeshes: AbstractMesh[] = vrTeleportationOptions.floorMeshes || [];\r\n if (!floorMeshes.length) {\r\n const floorMesh = this._scene.getMeshByName(vrTeleportationOptions.floorMeshName!);\r\n if (floorMesh) {\r\n floorMeshes.push(floorMesh);\r\n }\r\n }\r\n if (this.xr) {\r\n floorMeshes.forEach((mesh) => {\r\n this.xr.teleportation.addFloorMesh(mesh);\r\n });\r\n if (!this.xr.teleportation.attached) {\r\n this.xr.teleportation.attach();\r\n }\r\n return;\r\n } else if (!this.xrTestDone) {\r\n const waitForXr = () => {\r\n if (this.xrTestDone) {\r\n this._scene.unregisterBeforeRender(waitForXr);\r\n if (this.xr) {\r\n if (!this.xr.teleportation.attached) {\r\n this.xr.teleportation.attach();\r\n }\r\n } else {\r\n this.enableTeleportation(vrTeleportationOptions);\r\n }\r\n }\r\n };\r\n this._scene.registerBeforeRender(waitForXr);\r\n return;\r\n }\r\n }\r\n\r\n if (vrTeleportationOptions.floorMeshName) {\r\n this._floorMeshName = vrTeleportationOptions.floorMeshName;\r\n }\r\n if (vrTeleportationOptions.floorMeshes) {\r\n this._floorMeshesCollection = vrTeleportationOptions.floorMeshes;\r\n }\r\n\r\n if (vrTeleportationOptions.teleportationMode) {\r\n this._teleportationMode = vrTeleportationOptions.teleportationMode;\r\n }\r\n if (vrTeleportationOptions.teleportationTime && vrTeleportationOptions.teleportationTime > 0) {\r\n this._teleportationTime = vrTeleportationOptions.teleportationTime;\r\n }\r\n if (vrTeleportationOptions.teleportationSpeed && vrTeleportationOptions.teleportationSpeed > 0) {\r\n this._teleportationSpeed = vrTeleportationOptions.teleportationSpeed;\r\n }\r\n if (vrTeleportationOptions.easingFunction !== undefined) {\r\n this._teleportationEasing = vrTeleportationOptions.easingFunction;\r\n }\r\n\r\n if (this._leftController != null) {\r\n this._enableTeleportationOnController(this._leftController);\r\n }\r\n if (this._rightController != null) {\r\n this._enableTeleportationOnController(this._rightController);\r\n }\r\n\r\n // Creates an image processing post process for the vignette not relying\r\n // on the main scene configuration for image processing to reduce setup and spaces\r\n // (gamma/linear) conflicts.\r\n const imageProcessingConfiguration = new ImageProcessingConfiguration();\r\n imageProcessingConfiguration.vignetteColor = new Color4(0, 0, 0, 0);\r\n imageProcessingConfiguration.vignetteEnabled = true;\r\n this._postProcessMove = new ImageProcessingPostProcess(\r\n \"postProcessMove\",\r\n 1.0,\r\n this._webVRCamera,\r\n undefined,\r\n undefined,\r\n undefined,\r\n undefined,\r\n imageProcessingConfiguration\r\n );\r\n\r\n this._webVRCamera.detachPostProcess(this._postProcessMove);\r\n this._teleportationInitialized = true;\r\n if (this._isDefaultTeleportationTarget) {\r\n this._createTeleportationCircles();\r\n this._teleportationTarget.scaling.scaleInPlace(this._webVRCamera.deviceScaleFactor);\r\n }\r\n }\r\n }\r\n\r\n private _onNewGamepadConnected = (gamepad: Gamepad) => {\r\n if (gamepad.type !== Gamepad.POSE_ENABLED) {\r\n if (gamepad.leftStick) {\r\n gamepad.onleftstickchanged((stickValues) => {\r\n if (this._teleportationInitialized && this.teleportationEnabled) {\r\n // Listening to classic/xbox gamepad only if no VR controller is active\r\n if (\r\n (!this._leftController && !this._rightController) ||\r\n (this._leftController && !this._leftController._activePointer && this._rightController && !this._rightController._activePointer)\r\n ) {\r\n this._checkTeleportWithRay(stickValues, this._cameraGazer);\r\n this._checkTeleportBackwards(stickValues, this._cameraGazer);\r\n }\r\n }\r\n });\r\n }\r\n if (gamepad.rightStick) {\r\n gamepad.onrightstickchanged((stickValues) => {\r\n if (this._teleportationInitialized) {\r\n this._checkRotate(stickValues, this._cameraGazer);\r\n }\r\n });\r\n }\r\n if (gamepad.type === Gamepad.XBOX) {\r\n (gamepad).onbuttondown((buttonPressed: Xbox360Button) => {\r\n if (this._interactionsEnabled && buttonPressed === Xbox360Button.A) {\r\n this._cameraGazer._selectionPointerDown();\r\n }\r\n });\r\n (gamepad).onbuttonup((buttonPressed: Xbox360Button) => {\r\n if (this._interactionsEnabled && buttonPressed === Xbox360Button.A) {\r\n this._cameraGazer._selectionPointerUp();\r\n }\r\n });\r\n }\r\n } else {\r\n const webVRController = gamepad;\r\n const controller = new VRExperienceHelperControllerGazer(webVRController, this._scene, this._cameraGazer._gazeTracker);\r\n\r\n if (webVRController.hand === \"right\" || (this._leftController && this._leftController.webVRController != webVRController)) {\r\n this._rightController = controller;\r\n } else {\r\n this._leftController = controller;\r\n }\r\n\r\n this._tryEnableInteractionOnController(controller);\r\n }\r\n };\r\n\r\n // This only succeeds if the controller's mesh exists for the controller so this must be called whenever new controller is connected or when mesh is loaded\r\n private _tryEnableInteractionOnController = (controller: VRExperienceHelperControllerGazer) => {\r\n if (this._interactionsRequested && !controller._interactionsEnabled) {\r\n this._enableInteractionOnController(controller);\r\n }\r\n if (this._teleportationRequested && !controller._teleportationEnabled) {\r\n this._enableTeleportationOnController(controller);\r\n }\r\n };\r\n\r\n private _onNewGamepadDisconnected = (gamepad: Gamepad) => {\r\n if (gamepad instanceof WebVRController) {\r\n if (gamepad.hand === \"left\" && this._leftController != null) {\r\n this._leftController.dispose();\r\n this._leftController = null;\r\n }\r\n if (gamepad.hand === \"right\" && this._rightController != null) {\r\n this._rightController.dispose();\r\n this._rightController = null;\r\n }\r\n }\r\n };\r\n\r\n private _enableInteractionOnController(controller: VRExperienceHelperControllerGazer) {\r\n const controllerMesh = controller.webVRController.mesh;\r\n if (controllerMesh) {\r\n controller._interactionsEnabled = true;\r\n if (this.isInVRMode && this._displayLaserPointer) {\r\n controller._activatePointer();\r\n }\r\n if (this.webVROptions.laserToggle) {\r\n controller.webVRController.onMainButtonStateChangedObservable.add((stateObject) => {\r\n // Enabling / disabling laserPointer\r\n if (this._displayLaserPointer && stateObject.value === 1) {\r\n if (controller._activePointer) {\r\n controller._deactivatePointer();\r\n } else {\r\n controller._activatePointer();\r\n }\r\n if (this.displayGaze) {\r\n controller._gazeTracker.isVisible = controller._activePointer;\r\n }\r\n }\r\n });\r\n }\r\n controller.webVRController.onTriggerStateChangedObservable.add((stateObject) => {\r\n let gazer: VRExperienceHelperGazer = controller;\r\n if (this._noControllerIsActive) {\r\n gazer = this._cameraGazer;\r\n }\r\n if (!gazer._pointerDownOnMeshAsked) {\r\n if (stateObject.value > this._padSensibilityUp) {\r\n gazer._selectionPointerDown();\r\n }\r\n } else if (stateObject.value < this._padSensibilityDown) {\r\n gazer._selectionPointerUp();\r\n }\r\n });\r\n }\r\n }\r\n\r\n private _checkTeleportWithRay(stateObject: StickValues, gazer: VRExperienceHelperGazer) {\r\n // Dont teleport if another gaze already requested teleportation\r\n if (this._teleportationRequestInitiated && !gazer._teleportationRequestInitiated) {\r\n return;\r\n }\r\n if (!gazer._teleportationRequestInitiated) {\r\n if (stateObject.y < -this._padSensibilityUp && gazer._dpadPressed) {\r\n gazer._activatePointer();\r\n gazer._teleportationRequestInitiated = true;\r\n }\r\n } else {\r\n // Listening to the proper controller values changes to confirm teleportation\r\n if (Math.sqrt(stateObject.y * stateObject.y + stateObject.x * stateObject.x) < this._padSensibilityDown) {\r\n if (this._teleportActive) {\r\n this.teleportCamera(this._haloCenter);\r\n }\r\n\r\n gazer._teleportationRequestInitiated = false;\r\n }\r\n }\r\n }\r\n private _checkRotate(stateObject: StickValues, gazer: VRExperienceHelperGazer) {\r\n // Only rotate when user is not currently selecting a teleportation location\r\n if (gazer._teleportationRequestInitiated) {\r\n return;\r\n }\r\n\r\n if (!gazer._rotationLeftAsked) {\r\n if (stateObject.x < -this._padSensibilityUp && gazer._dpadPressed) {\r\n gazer._rotationLeftAsked = true;\r\n if (this._rotationAllowed) {\r\n this._rotateCamera(false);\r\n }\r\n }\r\n } else {\r\n if (stateObject.x > -this._padSensibilityDown) {\r\n gazer._rotationLeftAsked = false;\r\n }\r\n }\r\n\r\n if (!gazer._rotationRightAsked) {\r\n if (stateObject.x > this._padSensibilityUp && gazer._dpadPressed) {\r\n gazer._rotationRightAsked = true;\r\n if (this._rotationAllowed) {\r\n this._rotateCamera(true);\r\n }\r\n }\r\n } else {\r\n if (stateObject.x < this._padSensibilityDown) {\r\n gazer._rotationRightAsked = false;\r\n }\r\n }\r\n }\r\n private _checkTeleportBackwards(stateObject: StickValues, gazer: VRExperienceHelperGazer) {\r\n // Only teleport backwards when user is not currently selecting a teleportation location\r\n if (gazer._teleportationRequestInitiated) {\r\n return;\r\n }\r\n // Teleport backwards\r\n if (stateObject.y > this._padSensibilityUp && gazer._dpadPressed) {\r\n if (!gazer._teleportationBackRequestInitiated) {\r\n if (!this.currentVRCamera) {\r\n return;\r\n }\r\n\r\n // Get rotation and position of the current camera\r\n let rotation = Quaternion.FromRotationMatrix(this.currentVRCamera.getWorldMatrix().getRotationMatrix());\r\n let position = this.currentVRCamera.position;\r\n\r\n // If the camera has device position, use that instead\r\n if ((this.currentVRCamera).devicePosition && (this.currentVRCamera).deviceRotationQuaternion) {\r\n rotation = (this.currentVRCamera).deviceRotationQuaternion;\r\n position = (this.currentVRCamera).devicePosition;\r\n }\r\n\r\n // Get matrix with only the y rotation of the device rotation\r\n rotation.toEulerAnglesToRef(this._workingVector);\r\n this._workingVector.z = 0;\r\n this._workingVector.x = 0;\r\n Quaternion.RotationYawPitchRollToRef(this._workingVector.y, this._workingVector.x, this._workingVector.z, this._workingQuaternion);\r\n this._workingQuaternion.toRotationMatrix(this._workingMatrix);\r\n\r\n // Rotate backwards ray by device rotation to cast at the ground behind the user\r\n Vector3.TransformCoordinatesToRef(this._teleportBackwardsVector, this._workingMatrix, this._workingVector);\r\n\r\n // Teleport if ray hit the ground and is not to far away eg. backwards off a cliff\r\n const ray = new Ray(position, this._workingVector);\r\n const hit = this._scene.pickWithRay(ray, this._raySelectionPredicate);\r\n if (hit && hit.pickedPoint && hit.pickedMesh && this._isTeleportationFloor(hit.pickedMesh) && hit.distance < 5) {\r\n this.teleportCamera(hit.pickedPoint);\r\n }\r\n\r\n gazer._teleportationBackRequestInitiated = true;\r\n }\r\n } else {\r\n gazer._teleportationBackRequestInitiated = false;\r\n }\r\n }\r\n\r\n private _enableTeleportationOnController(controller: VRExperienceHelperControllerGazer) {\r\n const controllerMesh = controller.webVRController.mesh;\r\n if (controllerMesh) {\r\n if (!controller._interactionsEnabled) {\r\n this._enableInteractionOnController(controller);\r\n }\r\n controller._interactionsEnabled = true;\r\n controller._teleportationEnabled = true;\r\n if (controller.webVRController.controllerType === PoseEnabledControllerType.VIVE) {\r\n controller._dpadPressed = false;\r\n controller.webVRController.onPadStateChangedObservable.add((stateObject) => {\r\n controller._dpadPressed = stateObject.pressed;\r\n if (!controller._dpadPressed) {\r\n controller._rotationLeftAsked = false;\r\n controller._rotationRightAsked = false;\r\n controller._teleportationBackRequestInitiated = false;\r\n }\r\n });\r\n }\r\n controller.webVRController.onPadValuesChangedObservable.add((stateObject) => {\r\n if (this.teleportationEnabled) {\r\n this._checkTeleportBackwards(stateObject, controller);\r\n this._checkTeleportWithRay(stateObject, controller);\r\n }\r\n this._checkRotate(stateObject, controller);\r\n });\r\n }\r\n }\r\n\r\n private _createTeleportationCircles() {\r\n this._teleportationTarget = CreateGround(\"teleportationTarget\", { width: 2, height: 2, subdivisions: 2 }, this._scene);\r\n this._teleportationTarget.isPickable = false;\r\n\r\n const length = 512;\r\n const dynamicTexture = new DynamicTexture(\"DynamicTexture\", length, this._scene, true);\r\n dynamicTexture.hasAlpha = true;\r\n const context = dynamicTexture.getContext();\r\n\r\n const centerX = length / 2;\r\n const centerY = length / 2;\r\n const radius = 200;\r\n\r\n context.beginPath();\r\n context.arc(centerX, centerY, radius, 0, 2 * Math.PI, false);\r\n context.fillStyle = this._teleportationFillColor;\r\n context.fill();\r\n context.lineWidth = 10;\r\n context.strokeStyle = this._teleportationBorderColor;\r\n context.stroke();\r\n context.closePath();\r\n dynamicTexture.update();\r\n\r\n const teleportationCircleMaterial = new StandardMaterial(\"TextPlaneMaterial\", this._scene);\r\n teleportationCircleMaterial.diffuseTexture = dynamicTexture;\r\n this._teleportationTarget.material = teleportationCircleMaterial;\r\n\r\n const torus = CreateTorus(\r\n \"torusTeleportation\",\r\n {\r\n diameter: 0.75,\r\n thickness: 0.1,\r\n tessellation: 25,\r\n updatable: false,\r\n },\r\n this._scene\r\n );\r\n torus.isPickable = false;\r\n torus.parent = this._teleportationTarget;\r\n\r\n const animationInnerCircle = new Animation(\"animationInnerCircle\", \"position.y\", 30, Animation.ANIMATIONTYPE_FLOAT, Animation.ANIMATIONLOOPMODE_CYCLE);\r\n\r\n const keys = [];\r\n keys.push({\r\n frame: 0,\r\n value: 0,\r\n });\r\n keys.push({\r\n frame: 30,\r\n value: 0.4,\r\n });\r\n keys.push({\r\n frame: 60,\r\n value: 0,\r\n });\r\n\r\n animationInnerCircle.setKeys(keys);\r\n\r\n const easingFunction = new SineEase();\r\n easingFunction.setEasingMode(EasingFunction.EASINGMODE_EASEINOUT);\r\n animationInnerCircle.setEasingFunction(easingFunction);\r\n\r\n torus.animations = [];\r\n torus.animations.push(animationInnerCircle);\r\n\r\n this._scene.beginAnimation(torus, 0, 60, true);\r\n\r\n this._hideTeleportationTarget();\r\n }\r\n\r\n private _displayTeleportationTarget() {\r\n this._teleportActive = true;\r\n if (this._teleportationInitialized) {\r\n this._teleportationTarget.isVisible = true;\r\n if (this._isDefaultTeleportationTarget) {\r\n (this._teleportationTarget.getChildren()[0]).isVisible = true;\r\n }\r\n }\r\n }\r\n\r\n private _hideTeleportationTarget() {\r\n this._teleportActive = false;\r\n if (this._teleportationInitialized) {\r\n this._teleportationTarget.isVisible = false;\r\n if (this._isDefaultTeleportationTarget) {\r\n (this._teleportationTarget.getChildren()[0]).isVisible = false;\r\n }\r\n }\r\n }\r\n\r\n private _rotateCamera(right: boolean) {\r\n if (!(this.currentVRCamera instanceof FreeCamera)) {\r\n return;\r\n }\r\n\r\n if (right) {\r\n this._rotationAngle++;\r\n } else {\r\n this._rotationAngle--;\r\n }\r\n\r\n this.currentVRCamera.animations = [];\r\n\r\n const target = Quaternion.FromRotationMatrix(Matrix.RotationY((Math.PI / 4) * this._rotationAngle));\r\n\r\n const animationRotation = new Animation(\"animationRotation\", \"rotationQuaternion\", 90, Animation.ANIMATIONTYPE_QUATERNION, Animation.ANIMATIONLOOPMODE_CONSTANT);\r\n\r\n const animationRotationKeys = [];\r\n animationRotationKeys.push({\r\n frame: 0,\r\n value: this.currentVRCamera.rotationQuaternion,\r\n });\r\n animationRotationKeys.push({\r\n frame: 6,\r\n value: target,\r\n });\r\n\r\n animationRotation.setKeys(animationRotationKeys);\r\n\r\n animationRotation.setEasingFunction(this._circleEase);\r\n\r\n this.currentVRCamera.animations.push(animationRotation);\r\n\r\n this._postProcessMove.animations = [];\r\n\r\n const animationPP = new Animation(\"animationPP\", \"vignetteWeight\", 90, Animation.ANIMATIONTYPE_FLOAT, Animation.ANIMATIONLOOPMODE_CONSTANT);\r\n\r\n const vignetteWeightKeys = [];\r\n vignetteWeightKeys.push({\r\n frame: 0,\r\n value: 0,\r\n });\r\n vignetteWeightKeys.push({\r\n frame: 3,\r\n value: 4,\r\n });\r\n vignetteWeightKeys.push({\r\n frame: 6,\r\n value: 0,\r\n });\r\n\r\n animationPP.setKeys(vignetteWeightKeys);\r\n animationPP.setEasingFunction(this._circleEase);\r\n this._postProcessMove.animations.push(animationPP);\r\n\r\n const animationPP2 = new Animation(\"animationPP2\", \"vignetteStretch\", 90, Animation.ANIMATIONTYPE_FLOAT, Animation.ANIMATIONLOOPMODE_CONSTANT);\r\n\r\n const vignetteStretchKeys = [];\r\n vignetteStretchKeys.push({\r\n frame: 0,\r\n value: 0,\r\n });\r\n vignetteStretchKeys.push({\r\n frame: 3,\r\n value: 10,\r\n });\r\n vignetteStretchKeys.push({\r\n frame: 6,\r\n value: 0,\r\n });\r\n\r\n animationPP2.setKeys(vignetteStretchKeys);\r\n animationPP2.setEasingFunction(this._circleEase);\r\n this._postProcessMove.animations.push(animationPP2);\r\n\r\n this._postProcessMove.imageProcessingConfiguration.vignetteWeight = 0;\r\n this._postProcessMove.imageProcessingConfiguration.vignetteStretch = 0;\r\n this._postProcessMove.samples = 4;\r\n this._webVRCamera.attachPostProcess(this._postProcessMove);\r\n this._scene.beginAnimation(this._postProcessMove, 0, 6, false, 1, () => {\r\n this._webVRCamera.detachPostProcess(this._postProcessMove);\r\n });\r\n this._scene.beginAnimation(this.currentVRCamera, 0, 6, false, 1);\r\n }\r\n\r\n private _moveTeleportationSelectorTo(hit: PickingInfo, gazer: VRExperienceHelperGazer, ray: Ray) {\r\n if (hit.pickedPoint) {\r\n if (gazer._teleportationRequestInitiated) {\r\n this._displayTeleportationTarget();\r\n this._haloCenter.copyFrom(hit.pickedPoint);\r\n this._teleportationTarget.position.copyFrom(hit.pickedPoint);\r\n }\r\n\r\n const pickNormal = this._convertNormalToDirectionOfRay(hit.getNormal(true, false), ray);\r\n if (pickNormal) {\r\n const axis1 = Vector3.Cross(Axis.Y, pickNormal);\r\n const axis2 = Vector3.Cross(pickNormal, axis1);\r\n Vector3.RotationFromAxisToRef(axis2, pickNormal, axis1, this._teleportationTarget.rotation);\r\n }\r\n this._teleportationTarget.position.y += 0.1;\r\n }\r\n }\r\n private _workingVector = Vector3.Zero();\r\n private _workingQuaternion = Quaternion.Identity();\r\n private _workingMatrix = Matrix.Identity();\r\n\r\n /**\r\n * Time Constant Teleportation Mode\r\n */\r\n public static readonly TELEPORTATIONMODE_CONSTANTTIME = 0;\r\n /**\r\n * Speed Constant Teleportation Mode\r\n */\r\n public static readonly TELEPORTATIONMODE_CONSTANTSPEED = 1;\r\n\r\n /**\r\n * Teleports the users feet to the desired location\r\n * @param location The location where the user's feet should be placed\r\n */\r\n public teleportCamera(location: Vector3) {\r\n if (!(this.currentVRCamera instanceof FreeCamera)) {\r\n return;\r\n }\r\n // Teleport the hmd to where the user is looking by moving the anchor to where they are looking minus the\r\n // offset of the headset from the anchor.\r\n if (this.webVRCamera.leftCamera) {\r\n this._workingVector.copyFrom(this.webVRCamera.leftCamera.globalPosition);\r\n this._workingVector.subtractInPlace(this.webVRCamera.position);\r\n location.subtractToRef(this._workingVector, this._workingVector);\r\n } else {\r\n this._workingVector.copyFrom(location);\r\n }\r\n // Add height to account for user's height offset\r\n if (this.isInVRMode) {\r\n this._workingVector.y += this.webVRCamera.deviceDistanceToRoomGround() * this._webVRCamera.deviceScaleFactor;\r\n } else {\r\n this._workingVector.y += this._defaultHeight;\r\n }\r\n\r\n this.onBeforeCameraTeleport.notifyObservers(this._workingVector);\r\n\r\n // Animations FPS\r\n const FPS = 90;\r\n let speedRatio, lastFrame;\r\n if (this._teleportationMode == VRExperienceHelper.TELEPORTATIONMODE_CONSTANTSPEED) {\r\n lastFrame = FPS;\r\n const dist = Vector3.Distance(this.currentVRCamera.position, this._workingVector);\r\n speedRatio = this._teleportationSpeed / dist;\r\n } else {\r\n // teleportationMode is TELEPORTATIONMODE_CONSTANTTIME\r\n lastFrame = Math.round((this._teleportationTime * FPS) / 1000);\r\n speedRatio = 1;\r\n }\r\n\r\n // Create animation from the camera's position to the new location\r\n this.currentVRCamera.animations = [];\r\n const animationCameraTeleportation = new Animation(\"animationCameraTeleportation\", \"position\", FPS, Animation.ANIMATIONTYPE_VECTOR3, Animation.ANIMATIONLOOPMODE_CONSTANT);\r\n const animationCameraTeleportationKeys = [\r\n {\r\n frame: 0,\r\n value: this.currentVRCamera.position,\r\n },\r\n {\r\n frame: lastFrame,\r\n value: this._workingVector,\r\n },\r\n ];\r\n\r\n animationCameraTeleportation.setKeys(animationCameraTeleportationKeys);\r\n animationCameraTeleportation.setEasingFunction(this._teleportationEasing);\r\n this.currentVRCamera.animations.push(animationCameraTeleportation);\r\n\r\n this._postProcessMove.animations = [];\r\n\r\n // Calculate the mid frame for vignette animations\r\n const midFrame = Math.round(lastFrame / 2);\r\n\r\n const animationPP = new Animation(\"animationPP\", \"vignetteWeight\", FPS, Animation.ANIMATIONTYPE_FLOAT, Animation.ANIMATIONLOOPMODE_CONSTANT);\r\n\r\n const vignetteWeightKeys = [];\r\n vignetteWeightKeys.push({\r\n frame: 0,\r\n value: 0,\r\n });\r\n vignetteWeightKeys.push({\r\n frame: midFrame,\r\n value: 8,\r\n });\r\n vignetteWeightKeys.push({\r\n frame: lastFrame,\r\n value: 0,\r\n });\r\n\r\n animationPP.setKeys(vignetteWeightKeys);\r\n this._postProcessMove.animations.push(animationPP);\r\n\r\n const animationPP2 = new Animation(\"animationPP2\", \"vignetteStretch\", FPS, Animation.ANIMATIONTYPE_FLOAT, Animation.ANIMATIONLOOPMODE_CONSTANT);\r\n\r\n const vignetteStretchKeys = [];\r\n vignetteStretchKeys.push({\r\n frame: 0,\r\n value: 0,\r\n });\r\n vignetteStretchKeys.push({\r\n frame: midFrame,\r\n value: 10,\r\n });\r\n vignetteStretchKeys.push({\r\n frame: lastFrame,\r\n value: 0,\r\n });\r\n\r\n animationPP2.setKeys(vignetteStretchKeys);\r\n this._postProcessMove.animations.push(animationPP2);\r\n\r\n this._postProcessMove.imageProcessingConfiguration.vignetteWeight = 0;\r\n this._postProcessMove.imageProcessingConfiguration.vignetteStretch = 0;\r\n\r\n this._webVRCamera.attachPostProcess(this._postProcessMove);\r\n this._scene.beginAnimation(this._postProcessMove, 0, lastFrame, false, speedRatio, () => {\r\n this._webVRCamera.detachPostProcess(this._postProcessMove);\r\n });\r\n this._scene.beginAnimation(this.currentVRCamera, 0, lastFrame, false, speedRatio, () => {\r\n this.onAfterCameraTeleport.notifyObservers(this._workingVector);\r\n });\r\n\r\n this._hideTeleportationTarget();\r\n }\r\n\r\n private _convertNormalToDirectionOfRay(normal: Nullable, ray: Ray) {\r\n if (normal) {\r\n const angle = Math.acos(Vector3.Dot(normal, ray.direction));\r\n if (angle < Math.PI / 2) {\r\n normal.scaleInPlace(-1);\r\n }\r\n }\r\n return normal;\r\n }\r\n\r\n private _castRayAndSelectObject(gazer: VRExperienceHelperGazer) {\r\n if (!(this.currentVRCamera instanceof FreeCamera)) {\r\n return;\r\n }\r\n\r\n const ray = gazer._getForwardRay(this._rayLength);\r\n const hit = this._scene.pickWithRay(ray, this._raySelectionPredicate);\r\n\r\n if (hit) {\r\n this._scene.simulatePointerMove(hit, { pointerId: gazer._id });\r\n }\r\n gazer._currentHit = hit;\r\n\r\n // Moving the gazeTracker on the mesh face targetted\r\n if (hit && hit.pickedPoint) {\r\n if (this._displayGaze) {\r\n let multiplier = 1;\r\n\r\n gazer._gazeTracker.isVisible = true;\r\n\r\n if (gazer._isActionableMesh) {\r\n multiplier = 3;\r\n }\r\n if (this.updateGazeTrackerScale) {\r\n gazer._gazeTracker.scaling.x = hit.distance * multiplier;\r\n gazer._gazeTracker.scaling.y = hit.distance * multiplier;\r\n gazer._gazeTracker.scaling.z = hit.distance * multiplier;\r\n }\r\n\r\n const pickNormal = this._convertNormalToDirectionOfRay(hit.getNormal(), ray);\r\n // To avoid z-fighting\r\n const deltaFighting = 0.002;\r\n\r\n if (pickNormal) {\r\n const axis1 = Vector3.Cross(Axis.Y, pickNormal);\r\n const axis2 = Vector3.Cross(pickNormal, axis1);\r\n Vector3.RotationFromAxisToRef(axis2, pickNormal, axis1, gazer._gazeTracker.rotation);\r\n }\r\n gazer._gazeTracker.position.copyFrom(hit.pickedPoint);\r\n\r\n if (gazer._gazeTracker.position.x < 0) {\r\n gazer._gazeTracker.position.x += deltaFighting;\r\n } else {\r\n gazer._gazeTracker.position.x -= deltaFighting;\r\n }\r\n if (gazer._gazeTracker.position.y < 0) {\r\n gazer._gazeTracker.position.y += deltaFighting;\r\n } else {\r\n gazer._gazeTracker.position.y -= deltaFighting;\r\n }\r\n if (gazer._gazeTracker.position.z < 0) {\r\n gazer._gazeTracker.position.z += deltaFighting;\r\n } else {\r\n gazer._gazeTracker.position.z -= deltaFighting;\r\n }\r\n }\r\n\r\n // Changing the size of the laser pointer based on the distance from the targetted point\r\n gazer._updatePointerDistance(hit.distance);\r\n } else {\r\n gazer._updatePointerDistance();\r\n gazer._gazeTracker.isVisible = false;\r\n }\r\n\r\n if (hit && hit.pickedMesh) {\r\n // The object selected is the floor, we're in a teleportation scenario\r\n if (this._teleportationInitialized && this._isTeleportationFloor(hit.pickedMesh) && hit.pickedPoint) {\r\n // Moving the teleportation area to this targetted point\r\n\r\n //Raise onSelectedMeshUnselected observable if ray collided floor mesh/meshes and a non floor mesh was previously selected\r\n if (gazer._currentMeshSelected && !this._isTeleportationFloor(gazer._currentMeshSelected)) {\r\n this._notifySelectedMeshUnselected(gazer._currentMeshSelected);\r\n }\r\n\r\n gazer._currentMeshSelected = null;\r\n if (gazer._teleportationRequestInitiated) {\r\n this._moveTeleportationSelectorTo(hit, gazer, ray);\r\n }\r\n return;\r\n }\r\n // If not, we're in a selection scenario\r\n //this._teleportationAllowed = false;\r\n if (hit.pickedMesh !== gazer._currentMeshSelected) {\r\n if (this.meshSelectionPredicate(hit.pickedMesh)) {\r\n this.onNewMeshPicked.notifyObservers(hit);\r\n gazer._currentMeshSelected = hit.pickedMesh;\r\n if (hit.pickedMesh.isPickable && hit.pickedMesh.actionManager) {\r\n this.changeGazeColor(this._pickedGazeColor);\r\n this.changeLaserColor(this._pickedLaserColor);\r\n gazer._isActionableMesh = true;\r\n } else {\r\n this.changeGazeColor(this._gazeColor);\r\n this.changeLaserColor(this._laserColor);\r\n gazer._isActionableMesh = false;\r\n }\r\n try {\r\n this.onNewMeshSelected.notifyObservers(hit.pickedMesh);\r\n const gazerAsControllerGazer = gazer as VRExperienceHelperControllerGazer;\r\n if (gazerAsControllerGazer.webVRController) {\r\n this.onMeshSelectedWithController.notifyObservers({ mesh: hit.pickedMesh, controller: gazerAsControllerGazer.webVRController });\r\n }\r\n } catch (err) {\r\n Logger.Warn(\"Error while raising onNewMeshSelected or onMeshSelectedWithController: \" + err);\r\n }\r\n } else {\r\n this._notifySelectedMeshUnselected(gazer._currentMeshSelected);\r\n gazer._currentMeshSelected = null;\r\n this.changeGazeColor(this._gazeColor);\r\n this.changeLaserColor(this._laserColor);\r\n }\r\n }\r\n } else {\r\n this._notifySelectedMeshUnselected(gazer._currentMeshSelected);\r\n gazer._currentMeshSelected = null;\r\n //this._teleportationAllowed = false;\r\n this.changeGazeColor(this._gazeColor);\r\n this.changeLaserColor(this._laserColor);\r\n }\r\n }\r\n\r\n private _notifySelectedMeshUnselected(mesh: Nullable) {\r\n if (mesh) {\r\n this.onSelectedMeshUnselected.notifyObservers(mesh);\r\n }\r\n }\r\n\r\n /**\r\n * Permanently set new colors for the laser pointer\r\n * @param color the new laser color\r\n * @param pickedColor the new laser color when picked mesh detected\r\n */\r\n public setLaserColor(color: Color3, pickedColor: Color3 = this._pickedLaserColor) {\r\n this._laserColor = color;\r\n this._pickedLaserColor = pickedColor;\r\n }\r\n\r\n /**\r\n * Set lighting enabled / disabled on the laser pointer of both controllers\r\n * @param enabled should the lighting be enabled on the laser pointer\r\n */\r\n public setLaserLightingState(enabled: boolean = true) {\r\n if (this._leftController) {\r\n this._leftController._setLaserPointerLightingDisabled(!enabled);\r\n }\r\n if (this._rightController) {\r\n this._rightController._setLaserPointerLightingDisabled(!enabled);\r\n }\r\n }\r\n\r\n /**\r\n * Permanently set new colors for the gaze pointer\r\n * @param color the new gaze color\r\n * @param pickedColor the new gaze color when picked mesh detected\r\n */\r\n public setGazeColor(color: Color3, pickedColor: Color3 = this._pickedGazeColor) {\r\n this._gazeColor = color;\r\n this._pickedGazeColor = pickedColor;\r\n }\r\n\r\n /**\r\n * Sets the color of the laser ray from the vr controllers.\r\n * @param color new color for the ray.\r\n */\r\n public changeLaserColor(color: Color3) {\r\n if (!this.updateControllerLaserColor) {\r\n return;\r\n }\r\n\r\n if (this._leftController) {\r\n this._leftController._setLaserPointerColor(color);\r\n }\r\n if (this._rightController) {\r\n this._rightController._setLaserPointerColor(color);\r\n }\r\n }\r\n\r\n /**\r\n * Sets the color of the ray from the vr headsets gaze.\r\n * @param color new color for the ray.\r\n */\r\n public changeGazeColor(color: Color3) {\r\n if (!this.updateGazeTrackerColor) {\r\n return;\r\n }\r\n if (!(this._cameraGazer._gazeTracker.material)) {\r\n return;\r\n }\r\n (this._cameraGazer._gazeTracker.material).emissiveColor = color;\r\n if (this._leftController) {\r\n (this._leftController._gazeTracker.material).emissiveColor = color;\r\n }\r\n if (this._rightController) {\r\n (this._rightController._gazeTracker.material).emissiveColor = color;\r\n }\r\n }\r\n\r\n /**\r\n * Exits VR and disposes of the vr experience helper\r\n */\r\n public dispose() {\r\n if (this.isInVRMode) {\r\n this.exitVR();\r\n }\r\n\r\n if (this._postProcessMove) {\r\n this._postProcessMove.dispose();\r\n }\r\n\r\n if (this._webVRCamera) {\r\n this._webVRCamera.dispose();\r\n }\r\n if (this._vrDeviceOrientationCamera) {\r\n this._vrDeviceOrientationCamera.dispose();\r\n }\r\n if (!this._useCustomVRButton && this._btnVR && this._btnVR.parentNode) {\r\n document.body.removeChild(this._btnVR);\r\n }\r\n\r\n if (this._deviceOrientationCamera && this._scene.activeCamera != this._deviceOrientationCamera) {\r\n this._deviceOrientationCamera.dispose();\r\n }\r\n\r\n if (this._cameraGazer) {\r\n this._cameraGazer.dispose();\r\n }\r\n if (this._leftController) {\r\n this._leftController.dispose();\r\n }\r\n if (this._rightController) {\r\n this._rightController.dispose();\r\n }\r\n\r\n if (this._teleportationTarget) {\r\n this._teleportationTarget.dispose();\r\n }\r\n\r\n if (this.xr) {\r\n this.xr.dispose();\r\n }\r\n\r\n this._floorMeshesCollection.length = 0;\r\n\r\n document.removeEventListener(\"keydown\", this._onKeyDown);\r\n window.removeEventListener(\"vrdisplaypresentchange\", this._onVrDisplayPresentChangeBind);\r\n\r\n window.removeEventListener(\"resize\", this._onResize);\r\n document.removeEventListener(\"fullscreenchange\", this._onFullscreenChange);\r\n\r\n this._scene.getEngine().onVRDisplayChangedObservable.removeCallback(this._onVRDisplayChangedBind);\r\n this._scene.getEngine().onVRRequestPresentStart.removeCallback(this._onVRRequestPresentStart);\r\n this._scene.getEngine().onVRRequestPresentComplete.removeCallback(this._onVRRequestPresentComplete);\r\n\r\n this._scene.gamepadManager.onGamepadConnectedObservable.removeCallback(this._onNewGamepadConnected);\r\n this._scene.gamepadManager.onGamepadDisconnectedObservable.removeCallback(this._onNewGamepadDisconnected);\r\n\r\n this._scene.unregisterBeforeRender(this._beforeRender);\r\n }\r\n\r\n /**\r\n * Gets the name of the VRExperienceHelper class\r\n * @returns \"VRExperienceHelper\"\r\n */\r\n public getClassName(): string {\r\n return \"VRExperienceHelper\";\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport { Scalar } from \"../Maths/math.scalar\";\r\nimport { SphericalPolynomial } from \"../Maths/sphericalPolynomial\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport { InternalTexture, InternalTextureSource } from \"../Materials/Textures/internalTexture\";\r\nimport type { Nullable } from \"../types\";\r\nimport { Logger } from \"../Misc/logger\";\r\nimport { CubeMapToSphericalPolynomialTools } from \"../Misc/HighDynamicRange/cubemapToSphericalPolynomial\";\r\nimport type { Scene } from \"../scene\";\r\nimport { BaseTexture } from \"../Materials/Textures/baseTexture\";\r\nimport { ThinEngine } from \"../Engines/thinEngine\";\r\nimport { FromHalfFloat, ToHalfFloat } from \"./textureTools\";\r\n\r\nimport \"../Engines/Extensions/engine.cubeTexture\";\r\n\r\n// Based on demo done by Brandon Jones - http://media.tojicode.com/webgl-samples/dds.html\r\n// All values and structures referenced from:\r\n// http://msdn.microsoft.com/en-us/library/bb943991.aspx/\r\nconst DDS_MAGIC = 0x20534444;\r\n\r\nconst //DDSD_CAPS = 0x1,\r\n //DDSD_HEIGHT = 0x2,\r\n //DDSD_WIDTH = 0x4,\r\n //DDSD_PITCH = 0x8,\r\n //DDSD_PIXELFORMAT = 0x1000,\r\n DDSD_MIPMAPCOUNT = 0x20000;\r\n//DDSD_LINEARSIZE = 0x80000,\r\n//DDSD_DEPTH = 0x800000;\r\n\r\n// var DDSCAPS_COMPLEX = 0x8,\r\n// DDSCAPS_MIPMAP = 0x400000,\r\n// DDSCAPS_TEXTURE = 0x1000;\r\n\r\nconst DDSCAPS2_CUBEMAP = 0x200;\r\n// DDSCAPS2_CUBEMAP_POSITIVEX = 0x400,\r\n// DDSCAPS2_CUBEMAP_NEGATIVEX = 0x800,\r\n// DDSCAPS2_CUBEMAP_POSITIVEY = 0x1000,\r\n// DDSCAPS2_CUBEMAP_NEGATIVEY = 0x2000,\r\n// DDSCAPS2_CUBEMAP_POSITIVEZ = 0x4000,\r\n// DDSCAPS2_CUBEMAP_NEGATIVEZ = 0x8000,\r\n// DDSCAPS2_VOLUME = 0x200000;\r\n\r\nconst //DDPF_ALPHAPIXELS = 0x1,\r\n //DDPF_ALPHA = 0x2,\r\n DDPF_FOURCC = 0x4,\r\n DDPF_RGB = 0x40,\r\n //DDPF_YUV = 0x200,\r\n DDPF_LUMINANCE = 0x20000;\r\n\r\nfunction FourCCToInt32(value: string) {\r\n return value.charCodeAt(0) + (value.charCodeAt(1) << 8) + (value.charCodeAt(2) << 16) + (value.charCodeAt(3) << 24);\r\n}\r\n\r\nfunction Int32ToFourCC(value: number) {\r\n return String.fromCharCode(value & 0xff, (value >> 8) & 0xff, (value >> 16) & 0xff, (value >> 24) & 0xff);\r\n}\r\n\r\nconst FOURCC_DXT1 = FourCCToInt32(\"DXT1\");\r\nconst FOURCC_DXT3 = FourCCToInt32(\"DXT3\");\r\nconst FOURCC_DXT5 = FourCCToInt32(\"DXT5\");\r\nconst FOURCC_DX10 = FourCCToInt32(\"DX10\");\r\nconst FOURCC_D3DFMT_R16G16B16A16F = 113;\r\nconst FOURCC_D3DFMT_R32G32B32A32F = 116;\r\n\r\nconst DXGI_FORMAT_R32G32B32A32_FLOAT = 2;\r\nconst DXGI_FORMAT_R16G16B16A16_FLOAT = 10;\r\nconst DXGI_FORMAT_B8G8R8X8_UNORM = 88;\r\n\r\nconst headerLengthInt = 31; // The header length in 32 bit ints\r\n\r\n// Offsets into the header array\r\nconst off_magic = 0;\r\n\r\nconst off_size = 1;\r\nconst off_flags = 2;\r\nconst off_height = 3;\r\nconst off_width = 4;\r\n\r\nconst off_mipmapCount = 7;\r\n\r\nconst off_pfFlags = 20;\r\nconst off_pfFourCC = 21;\r\nconst off_RGBbpp = 22;\r\nconst off_RMask = 23;\r\nconst off_GMask = 24;\r\nconst off_BMask = 25;\r\nconst off_AMask = 26;\r\n// var off_caps1 = 27;\r\nconst off_caps2 = 28;\r\n// var off_caps3 = 29;\r\n// var off_caps4 = 30;\r\nconst off_dxgiFormat = 32;\r\n\r\n/**\r\n * Direct draw surface info\r\n * @see https://docs.microsoft.com/en-us/windows/desktop/direct3ddds/dx-graphics-dds-pguide\r\n */\r\nexport interface DDSInfo {\r\n /**\r\n * Width of the texture\r\n */\r\n width: number;\r\n /**\r\n * Width of the texture\r\n */\r\n height: number;\r\n /**\r\n * Number of Mipmaps for the texture\r\n * @see https://en.wikipedia.org/wiki/Mipmap\r\n */\r\n mipmapCount: number;\r\n /**\r\n * If the textures format is a known fourCC format\r\n * @see https://www.fourcc.org/\r\n */\r\n isFourCC: boolean;\r\n /**\r\n * If the texture is an RGB format eg. DXGI_FORMAT_B8G8R8X8_UNORM format\r\n */\r\n isRGB: boolean;\r\n /**\r\n * If the texture is a lumincance format\r\n */\r\n isLuminance: boolean;\r\n /**\r\n * If this is a cube texture\r\n * @see https://docs.microsoft.com/en-us/windows/desktop/direct3ddds/dds-file-layout-for-cubic-environment-maps\r\n */\r\n isCube: boolean;\r\n /**\r\n * If the texture is a compressed format eg. FOURCC_DXT1\r\n */\r\n isCompressed: boolean;\r\n /**\r\n * The dxgiFormat of the texture\r\n * @see https://docs.microsoft.com/en-us/windows/desktop/api/dxgiformat/ne-dxgiformat-dxgi_format\r\n */\r\n dxgiFormat: number;\r\n /**\r\n * Texture type eg. Engine.TEXTURETYPE_UNSIGNED_INT, Engine.TEXTURETYPE_FLOAT\r\n */\r\n textureType: number;\r\n /**\r\n * Sphericle polynomial created for the dds texture\r\n */\r\n sphericalPolynomial?: SphericalPolynomial;\r\n}\r\n\r\n/**\r\n * Class used to provide DDS decompression tools\r\n */\r\nexport class DDSTools {\r\n /**\r\n * Gets or sets a boolean indicating that LOD info is stored in alpha channel (false by default)\r\n */\r\n public static StoreLODInAlphaChannel = false;\r\n\r\n /**\r\n * Gets DDS information from an array buffer\r\n * @param data defines the array buffer view to read data from\r\n * @returns the DDS information\r\n */\r\n public static GetDDSInfo(data: ArrayBufferView): DDSInfo {\r\n const header = new Int32Array(data.buffer, data.byteOffset, headerLengthInt);\r\n const extendedHeader = new Int32Array(data.buffer, data.byteOffset, headerLengthInt + 4);\r\n\r\n let mipmapCount = 1;\r\n if (header[off_flags] & DDSD_MIPMAPCOUNT) {\r\n mipmapCount = Math.max(1, header[off_mipmapCount]);\r\n }\r\n\r\n const fourCC = header[off_pfFourCC];\r\n const dxgiFormat = fourCC === FOURCC_DX10 ? extendedHeader[off_dxgiFormat] : 0;\r\n let textureType = Constants.TEXTURETYPE_UNSIGNED_INT;\r\n\r\n switch (fourCC) {\r\n case FOURCC_D3DFMT_R16G16B16A16F:\r\n textureType = Constants.TEXTURETYPE_HALF_FLOAT;\r\n break;\r\n case FOURCC_D3DFMT_R32G32B32A32F:\r\n textureType = Constants.TEXTURETYPE_FLOAT;\r\n break;\r\n case FOURCC_DX10:\r\n if (dxgiFormat === DXGI_FORMAT_R16G16B16A16_FLOAT) {\r\n textureType = Constants.TEXTURETYPE_HALF_FLOAT;\r\n break;\r\n }\r\n if (dxgiFormat === DXGI_FORMAT_R32G32B32A32_FLOAT) {\r\n textureType = Constants.TEXTURETYPE_FLOAT;\r\n break;\r\n }\r\n }\r\n\r\n return {\r\n width: header[off_width],\r\n height: header[off_height],\r\n mipmapCount: mipmapCount,\r\n isFourCC: (header[off_pfFlags] & DDPF_FOURCC) === DDPF_FOURCC,\r\n isRGB: (header[off_pfFlags] & DDPF_RGB) === DDPF_RGB,\r\n isLuminance: (header[off_pfFlags] & DDPF_LUMINANCE) === DDPF_LUMINANCE,\r\n isCube: (header[off_caps2] & DDSCAPS2_CUBEMAP) === DDSCAPS2_CUBEMAP,\r\n isCompressed: fourCC === FOURCC_DXT1 || fourCC === FOURCC_DXT3 || fourCC === FOURCC_DXT5,\r\n dxgiFormat: dxgiFormat,\r\n textureType: textureType,\r\n };\r\n }\r\n\r\n private static _GetHalfFloatAsFloatRGBAArrayBuffer(width: number, height: number, dataOffset: number, dataLength: number, arrayBuffer: ArrayBuffer, lod: number): Float32Array {\r\n const destArray = new Float32Array(dataLength);\r\n const srcData = new Uint16Array(arrayBuffer, dataOffset);\r\n let index = 0;\r\n for (let y = 0; y < height; y++) {\r\n for (let x = 0; x < width; x++) {\r\n const srcPos = (x + y * width) * 4;\r\n destArray[index] = FromHalfFloat(srcData[srcPos]);\r\n destArray[index + 1] = FromHalfFloat(srcData[srcPos + 1]);\r\n destArray[index + 2] = FromHalfFloat(srcData[srcPos + 2]);\r\n if (DDSTools.StoreLODInAlphaChannel) {\r\n destArray[index + 3] = lod;\r\n } else {\r\n destArray[index + 3] = FromHalfFloat(srcData[srcPos + 3]);\r\n }\r\n index += 4;\r\n }\r\n }\r\n\r\n return destArray;\r\n }\r\n\r\n private static _GetHalfFloatRGBAArrayBuffer(width: number, height: number, dataOffset: number, dataLength: number, arrayBuffer: ArrayBuffer, lod: number): Uint16Array {\r\n if (DDSTools.StoreLODInAlphaChannel) {\r\n const destArray = new Uint16Array(dataLength);\r\n const srcData = new Uint16Array(arrayBuffer, dataOffset);\r\n let index = 0;\r\n for (let y = 0; y < height; y++) {\r\n for (let x = 0; x < width; x++) {\r\n const srcPos = (x + y * width) * 4;\r\n destArray[index] = srcData[srcPos];\r\n destArray[index + 1] = srcData[srcPos + 1];\r\n destArray[index + 2] = srcData[srcPos + 2];\r\n destArray[index + 3] = ToHalfFloat(lod);\r\n index += 4;\r\n }\r\n }\r\n\r\n return destArray;\r\n }\r\n\r\n return new Uint16Array(arrayBuffer, dataOffset, dataLength);\r\n }\r\n\r\n private static _GetFloatRGBAArrayBuffer(width: number, height: number, dataOffset: number, dataLength: number, arrayBuffer: ArrayBuffer, lod: number): Float32Array {\r\n if (DDSTools.StoreLODInAlphaChannel) {\r\n const destArray = new Float32Array(dataLength);\r\n const srcData = new Float32Array(arrayBuffer, dataOffset);\r\n let index = 0;\r\n for (let y = 0; y < height; y++) {\r\n for (let x = 0; x < width; x++) {\r\n const srcPos = (x + y * width) * 4;\r\n destArray[index] = srcData[srcPos];\r\n destArray[index + 1] = srcData[srcPos + 1];\r\n destArray[index + 2] = srcData[srcPos + 2];\r\n destArray[index + 3] = lod;\r\n index += 4;\r\n }\r\n }\r\n\r\n return destArray;\r\n }\r\n return new Float32Array(arrayBuffer, dataOffset, dataLength);\r\n }\r\n\r\n private static _GetFloatAsHalfFloatRGBAArrayBuffer(width: number, height: number, dataOffset: number, dataLength: number, arrayBuffer: ArrayBuffer, lod: number): Uint16Array {\r\n const destArray = new Uint16Array(dataLength);\r\n const srcData = new Float32Array(arrayBuffer, dataOffset);\r\n let index = 0;\r\n for (let y = 0; y < height; y++) {\r\n for (let x = 0; x < width; x++) {\r\n destArray[index] = ToHalfFloat(srcData[index]);\r\n destArray[index + 1] = ToHalfFloat(srcData[index + 1]);\r\n destArray[index + 2] = ToHalfFloat(srcData[index + 2]);\r\n if (DDSTools.StoreLODInAlphaChannel) {\r\n destArray[index + 3] = ToHalfFloat(lod);\r\n } else {\r\n destArray[index + 3] = ToHalfFloat(srcData[index + 3]);\r\n }\r\n index += 4;\r\n }\r\n }\r\n\r\n return destArray;\r\n }\r\n\r\n private static _GetFloatAsUIntRGBAArrayBuffer(width: number, height: number, dataOffset: number, dataLength: number, arrayBuffer: ArrayBuffer, lod: number): Uint8Array {\r\n const destArray = new Uint8Array(dataLength);\r\n const srcData = new Float32Array(arrayBuffer, dataOffset);\r\n let index = 0;\r\n for (let y = 0; y < height; y++) {\r\n for (let x = 0; x < width; x++) {\r\n const srcPos = (x + y * width) * 4;\r\n destArray[index] = Scalar.Clamp(srcData[srcPos]) * 255;\r\n destArray[index + 1] = Scalar.Clamp(srcData[srcPos + 1]) * 255;\r\n destArray[index + 2] = Scalar.Clamp(srcData[srcPos + 2]) * 255;\r\n if (DDSTools.StoreLODInAlphaChannel) {\r\n destArray[index + 3] = lod;\r\n } else {\r\n destArray[index + 3] = Scalar.Clamp(srcData[srcPos + 3]) * 255;\r\n }\r\n index += 4;\r\n }\r\n }\r\n\r\n return destArray;\r\n }\r\n\r\n private static _GetHalfFloatAsUIntRGBAArrayBuffer(width: number, height: number, dataOffset: number, dataLength: number, arrayBuffer: ArrayBuffer, lod: number): Uint8Array {\r\n const destArray = new Uint8Array(dataLength);\r\n const srcData = new Uint16Array(arrayBuffer, dataOffset);\r\n let index = 0;\r\n for (let y = 0; y < height; y++) {\r\n for (let x = 0; x < width; x++) {\r\n const srcPos = (x + y * width) * 4;\r\n destArray[index] = Scalar.Clamp(FromHalfFloat(srcData[srcPos])) * 255;\r\n destArray[index + 1] = Scalar.Clamp(FromHalfFloat(srcData[srcPos + 1])) * 255;\r\n destArray[index + 2] = Scalar.Clamp(FromHalfFloat(srcData[srcPos + 2])) * 255;\r\n if (DDSTools.StoreLODInAlphaChannel) {\r\n destArray[index + 3] = lod;\r\n } else {\r\n destArray[index + 3] = Scalar.Clamp(FromHalfFloat(srcData[srcPos + 3])) * 255;\r\n }\r\n index += 4;\r\n }\r\n }\r\n\r\n return destArray;\r\n }\r\n\r\n private static _GetRGBAArrayBuffer(\r\n width: number,\r\n height: number,\r\n dataOffset: number,\r\n dataLength: number,\r\n arrayBuffer: ArrayBuffer,\r\n rOffset: number,\r\n gOffset: number,\r\n bOffset: number,\r\n aOffset: number\r\n ): Uint8Array {\r\n const byteArray = new Uint8Array(dataLength);\r\n const srcData = new Uint8Array(arrayBuffer, dataOffset);\r\n let index = 0;\r\n for (let y = 0; y < height; y++) {\r\n for (let x = 0; x < width; x++) {\r\n const srcPos = (x + y * width) * 4;\r\n\r\n byteArray[index] = srcData[srcPos + rOffset];\r\n byteArray[index + 1] = srcData[srcPos + gOffset];\r\n byteArray[index + 2] = srcData[srcPos + bOffset];\r\n byteArray[index + 3] = srcData[srcPos + aOffset];\r\n index += 4;\r\n }\r\n }\r\n\r\n return byteArray;\r\n }\r\n\r\n private static _ExtractLongWordOrder(value: number): number {\r\n if (value === 0 || value === 255 || value === -16777216) {\r\n return 0;\r\n }\r\n\r\n return 1 + DDSTools._ExtractLongWordOrder(value >> 8);\r\n }\r\n\r\n private static _GetRGBArrayBuffer(\r\n width: number,\r\n height: number,\r\n dataOffset: number,\r\n dataLength: number,\r\n arrayBuffer: ArrayBuffer,\r\n rOffset: number,\r\n gOffset: number,\r\n bOffset: number\r\n ): Uint8Array {\r\n const byteArray = new Uint8Array(dataLength);\r\n const srcData = new Uint8Array(arrayBuffer, dataOffset);\r\n let index = 0;\r\n for (let y = 0; y < height; y++) {\r\n for (let x = 0; x < width; x++) {\r\n const srcPos = (x + y * width) * 3;\r\n\r\n byteArray[index] = srcData[srcPos + rOffset];\r\n byteArray[index + 1] = srcData[srcPos + gOffset];\r\n byteArray[index + 2] = srcData[srcPos + bOffset];\r\n index += 3;\r\n }\r\n }\r\n\r\n return byteArray;\r\n }\r\n\r\n private static _GetLuminanceArrayBuffer(width: number, height: number, dataOffset: number, dataLength: number, arrayBuffer: ArrayBuffer): Uint8Array {\r\n const byteArray = new Uint8Array(dataLength);\r\n const srcData = new Uint8Array(arrayBuffer, dataOffset);\r\n let index = 0;\r\n for (let y = 0; y < height; y++) {\r\n for (let x = 0; x < width; x++) {\r\n const srcPos = x + y * width;\r\n byteArray[index] = srcData[srcPos];\r\n index++;\r\n }\r\n }\r\n\r\n return byteArray;\r\n }\r\n\r\n /**\r\n * Uploads DDS Levels to a Babylon Texture\r\n * @internal\r\n */\r\n public static UploadDDSLevels(\r\n engine: ThinEngine,\r\n texture: InternalTexture,\r\n data: ArrayBufferView,\r\n info: DDSInfo,\r\n loadMipmaps: boolean,\r\n faces: number,\r\n lodIndex = -1,\r\n currentFace?: number,\r\n destTypeMustBeFilterable = true\r\n ) {\r\n let sphericalPolynomialFaces: Nullable> = null;\r\n if (info.sphericalPolynomial) {\r\n sphericalPolynomialFaces = new Array();\r\n }\r\n const ext = !!engine.getCaps().s3tc;\r\n\r\n // TODO WEBGPU Once generateMipMaps is split into generateMipMaps + hasMipMaps in InternalTexture this line can be removed\r\n texture.generateMipMaps = loadMipmaps;\r\n\r\n const header = new Int32Array(data.buffer, data.byteOffset, headerLengthInt);\r\n let fourCC: number,\r\n width: number,\r\n height: number,\r\n dataLength: number = 0,\r\n dataOffset: number;\r\n let byteArray: Uint8Array, mipmapCount: number, mip: number;\r\n let internalCompressedFormat = 0;\r\n let blockBytes = 1;\r\n\r\n if (header[off_magic] !== DDS_MAGIC) {\r\n Logger.Error(\"Invalid magic number in DDS header\");\r\n return;\r\n }\r\n\r\n if (!info.isFourCC && !info.isRGB && !info.isLuminance) {\r\n Logger.Error(\"Unsupported format, must contain a FourCC, RGB or LUMINANCE code\");\r\n return;\r\n }\r\n\r\n if (info.isCompressed && !ext) {\r\n Logger.Error(\"Compressed textures are not supported on this platform.\");\r\n return;\r\n }\r\n\r\n let bpp = header[off_RGBbpp];\r\n dataOffset = header[off_size] + 4;\r\n\r\n let computeFormats = false;\r\n\r\n if (info.isFourCC) {\r\n fourCC = header[off_pfFourCC];\r\n switch (fourCC) {\r\n case FOURCC_DXT1:\r\n blockBytes = 8;\r\n internalCompressedFormat = Constants.TEXTUREFORMAT_COMPRESSED_RGBA_S3TC_DXT1;\r\n break;\r\n case FOURCC_DXT3:\r\n blockBytes = 16;\r\n internalCompressedFormat = Constants.TEXTUREFORMAT_COMPRESSED_RGBA_S3TC_DXT3;\r\n break;\r\n case FOURCC_DXT5:\r\n blockBytes = 16;\r\n internalCompressedFormat = Constants.TEXTUREFORMAT_COMPRESSED_RGBA_S3TC_DXT5;\r\n break;\r\n case FOURCC_D3DFMT_R16G16B16A16F:\r\n computeFormats = true;\r\n bpp = 64;\r\n break;\r\n case FOURCC_D3DFMT_R32G32B32A32F:\r\n computeFormats = true;\r\n bpp = 128;\r\n break;\r\n case FOURCC_DX10: {\r\n // There is an additionnal header so dataOffset need to be changed\r\n dataOffset += 5 * 4; // 5 uints\r\n\r\n let supported = false;\r\n switch (info.dxgiFormat) {\r\n case DXGI_FORMAT_R16G16B16A16_FLOAT:\r\n computeFormats = true;\r\n bpp = 64;\r\n supported = true;\r\n break;\r\n case DXGI_FORMAT_R32G32B32A32_FLOAT:\r\n computeFormats = true;\r\n bpp = 128;\r\n supported = true;\r\n break;\r\n case DXGI_FORMAT_B8G8R8X8_UNORM:\r\n info.isRGB = true;\r\n info.isFourCC = false;\r\n bpp = 32;\r\n supported = true;\r\n break;\r\n }\r\n\r\n if (supported) {\r\n break;\r\n }\r\n }\r\n // eslint-disable-next-line no-fallthrough\r\n default:\r\n console.error(\"Unsupported FourCC code:\", Int32ToFourCC(fourCC));\r\n return;\r\n }\r\n }\r\n\r\n const rOffset = DDSTools._ExtractLongWordOrder(header[off_RMask]);\r\n const gOffset = DDSTools._ExtractLongWordOrder(header[off_GMask]);\r\n const bOffset = DDSTools._ExtractLongWordOrder(header[off_BMask]);\r\n const aOffset = DDSTools._ExtractLongWordOrder(header[off_AMask]);\r\n\r\n if (computeFormats) {\r\n internalCompressedFormat = engine._getRGBABufferInternalSizedFormat(info.textureType);\r\n }\r\n\r\n mipmapCount = 1;\r\n if (header[off_flags] & DDSD_MIPMAPCOUNT && loadMipmaps !== false) {\r\n mipmapCount = Math.max(1, header[off_mipmapCount]);\r\n }\r\n\r\n const startFace = currentFace || 0;\r\n const caps = engine.getCaps();\r\n for (let face = startFace; face < faces; face++) {\r\n width = header[off_width];\r\n height = header[off_height];\r\n\r\n for (mip = 0; mip < mipmapCount; ++mip) {\r\n if (lodIndex === -1 || lodIndex === mip) {\r\n // In case of fixed LOD, if the lod has just been uploaded, early exit.\r\n const i = lodIndex === -1 ? mip : 0;\r\n\r\n if (!info.isCompressed && info.isFourCC) {\r\n texture.format = Constants.TEXTUREFORMAT_RGBA;\r\n dataLength = width * height * 4;\r\n let floatArray: Nullable = null;\r\n\r\n if (engine._badOS || engine._badDesktopOS || (!caps.textureHalfFloat && !caps.textureFloat)) {\r\n // Required because iOS has many issues with float and half float generation\r\n if (bpp === 128) {\r\n floatArray = DDSTools._GetFloatAsUIntRGBAArrayBuffer(width, height, data.byteOffset + dataOffset, dataLength, data.buffer, i);\r\n if (sphericalPolynomialFaces && i == 0) {\r\n sphericalPolynomialFaces.push(DDSTools._GetFloatRGBAArrayBuffer(width, height, data.byteOffset + dataOffset, dataLength, data.buffer, i));\r\n }\r\n } else if (bpp === 64) {\r\n floatArray = DDSTools._GetHalfFloatAsUIntRGBAArrayBuffer(width, height, data.byteOffset + dataOffset, dataLength, data.buffer, i);\r\n if (sphericalPolynomialFaces && i == 0) {\r\n sphericalPolynomialFaces.push(\r\n DDSTools._GetHalfFloatAsFloatRGBAArrayBuffer(width, height, data.byteOffset + dataOffset, dataLength, data.buffer, i)\r\n );\r\n }\r\n }\r\n\r\n texture.type = Constants.TEXTURETYPE_UNSIGNED_INT;\r\n } else {\r\n const floatAvailable = caps.textureFloat && ((destTypeMustBeFilterable && caps.textureFloatLinearFiltering) || !destTypeMustBeFilterable);\r\n const halfFloatAvailable = caps.textureHalfFloat && ((destTypeMustBeFilterable && caps.textureHalfFloatLinearFiltering) || !destTypeMustBeFilterable);\r\n\r\n const destType =\r\n (bpp === 128 || (bpp === 64 && !halfFloatAvailable)) && floatAvailable\r\n ? Constants.TEXTURETYPE_FLOAT\r\n : (bpp === 64 || (bpp === 128 && !floatAvailable)) && halfFloatAvailable\r\n ? Constants.TEXTURETYPE_HALF_FLOAT\r\n : Constants.TEXTURETYPE_UNSIGNED_BYTE;\r\n\r\n let dataGetter: (width: number, height: number, dataOffset: number, dataLength: number, arrayBuffer: ArrayBuffer, lod: number) => ArrayBufferView;\r\n let dataGetterPolynomial: Nullable<\r\n (width: number, height: number, dataOffset: number, dataLength: number, arrayBuffer: ArrayBuffer, lod: number) => ArrayBufferView\r\n > = null;\r\n\r\n switch (bpp) {\r\n case 128: {\r\n switch (destType) {\r\n case Constants.TEXTURETYPE_FLOAT:\r\n dataGetter = DDSTools._GetFloatRGBAArrayBuffer;\r\n dataGetterPolynomial = null;\r\n break;\r\n case Constants.TEXTURETYPE_HALF_FLOAT:\r\n dataGetter = DDSTools._GetFloatAsHalfFloatRGBAArrayBuffer;\r\n dataGetterPolynomial = DDSTools._GetFloatRGBAArrayBuffer;\r\n break;\r\n case Constants.TEXTURETYPE_UNSIGNED_BYTE:\r\n dataGetter = DDSTools._GetFloatAsUIntRGBAArrayBuffer;\r\n dataGetterPolynomial = DDSTools._GetFloatRGBAArrayBuffer;\r\n break;\r\n }\r\n break;\r\n }\r\n default: {\r\n // 64 bpp\r\n switch (destType) {\r\n case Constants.TEXTURETYPE_FLOAT:\r\n dataGetter = DDSTools._GetHalfFloatAsFloatRGBAArrayBuffer;\r\n dataGetterPolynomial = null;\r\n break;\r\n case Constants.TEXTURETYPE_HALF_FLOAT:\r\n dataGetter = DDSTools._GetHalfFloatRGBAArrayBuffer;\r\n dataGetterPolynomial = DDSTools._GetHalfFloatAsFloatRGBAArrayBuffer;\r\n break;\r\n case Constants.TEXTURETYPE_UNSIGNED_BYTE:\r\n dataGetter = DDSTools._GetHalfFloatAsUIntRGBAArrayBuffer;\r\n dataGetterPolynomial = DDSTools._GetHalfFloatAsFloatRGBAArrayBuffer;\r\n break;\r\n }\r\n break;\r\n }\r\n }\r\n\r\n texture.type = destType;\r\n\r\n floatArray = dataGetter(width, height, data.byteOffset + dataOffset, dataLength, data.buffer, i);\r\n\r\n if (sphericalPolynomialFaces && i == 0) {\r\n sphericalPolynomialFaces.push(\r\n dataGetterPolynomial ? dataGetterPolynomial(width, height, data.byteOffset + dataOffset, dataLength, data.buffer, i) : floatArray\r\n );\r\n }\r\n }\r\n\r\n if (floatArray) {\r\n engine._uploadDataToTextureDirectly(texture, floatArray, face, i);\r\n }\r\n } else if (info.isRGB) {\r\n texture.type = Constants.TEXTURETYPE_UNSIGNED_INT;\r\n if (bpp === 24) {\r\n texture.format = Constants.TEXTUREFORMAT_RGB;\r\n dataLength = width * height * 3;\r\n byteArray = DDSTools._GetRGBArrayBuffer(width, height, data.byteOffset + dataOffset, dataLength, data.buffer, rOffset, gOffset, bOffset);\r\n engine._uploadDataToTextureDirectly(texture, byteArray, face, i);\r\n } else {\r\n // 32\r\n texture.format = Constants.TEXTUREFORMAT_RGBA;\r\n dataLength = width * height * 4;\r\n byteArray = DDSTools._GetRGBAArrayBuffer(width, height, data.byteOffset + dataOffset, dataLength, data.buffer, rOffset, gOffset, bOffset, aOffset);\r\n engine._uploadDataToTextureDirectly(texture, byteArray, face, i);\r\n }\r\n } else if (info.isLuminance) {\r\n const unpackAlignment = engine._getUnpackAlignement();\r\n const unpaddedRowSize = width;\r\n const paddedRowSize = Math.floor((width + unpackAlignment - 1) / unpackAlignment) * unpackAlignment;\r\n dataLength = paddedRowSize * (height - 1) + unpaddedRowSize;\r\n\r\n byteArray = DDSTools._GetLuminanceArrayBuffer(width, height, data.byteOffset + dataOffset, dataLength, data.buffer);\r\n texture.format = Constants.TEXTUREFORMAT_LUMINANCE;\r\n texture.type = Constants.TEXTURETYPE_UNSIGNED_INT;\r\n\r\n engine._uploadDataToTextureDirectly(texture, byteArray, face, i);\r\n } else {\r\n dataLength = (((Math.max(4, width) / 4) * Math.max(4, height)) / 4) * blockBytes;\r\n byteArray = new Uint8Array(data.buffer, data.byteOffset + dataOffset, dataLength);\r\n\r\n texture.type = Constants.TEXTURETYPE_UNSIGNED_INT;\r\n engine._uploadCompressedDataToTextureDirectly(texture, internalCompressedFormat, width, height, byteArray, face, i);\r\n }\r\n }\r\n dataOffset += bpp ? width * height * (bpp / 8) : dataLength;\r\n width *= 0.5;\r\n height *= 0.5;\r\n\r\n width = Math.max(1.0, width);\r\n height = Math.max(1.0, height);\r\n }\r\n\r\n if (currentFace !== undefined) {\r\n // Loading a single face\r\n break;\r\n }\r\n }\r\n if (sphericalPolynomialFaces && sphericalPolynomialFaces.length > 0) {\r\n info.sphericalPolynomial = CubeMapToSphericalPolynomialTools.ConvertCubeMapToSphericalPolynomial({\r\n size: header[off_width],\r\n right: sphericalPolynomialFaces[0],\r\n left: sphericalPolynomialFaces[1],\r\n up: sphericalPolynomialFaces[2],\r\n down: sphericalPolynomialFaces[3],\r\n front: sphericalPolynomialFaces[4],\r\n back: sphericalPolynomialFaces[5],\r\n format: Constants.TEXTUREFORMAT_RGBA,\r\n type: Constants.TEXTURETYPE_FLOAT,\r\n gammaSpace: false,\r\n });\r\n } else {\r\n info.sphericalPolynomial = undefined;\r\n }\r\n }\r\n}\r\n\r\ndeclare module \"../Engines/thinEngine\" {\r\n export interface ThinEngine {\r\n /**\r\n * Create a cube texture from prefiltered data (ie. the mipmaps contain ready to use data for PBR reflection)\r\n * @param rootUrl defines the url where the file to load is located\r\n * @param scene defines the current scene\r\n * @param lodScale defines scale to apply to the mip map selection\r\n * @param lodOffset defines offset to apply to the mip map selection\r\n * @param onLoad defines an optional callback raised when the texture is loaded\r\n * @param onError defines an optional callback raised if there is an issue to load the texture\r\n * @param format defines the format of the data\r\n * @param forcedExtension defines the extension to use to pick the right loader\r\n * @param createPolynomials defines wheter or not to create polynomails harmonics for the texture\r\n * @returns the cube texture as an InternalTexture\r\n */\r\n createPrefilteredCubeTexture(\r\n rootUrl: string,\r\n scene: Nullable,\r\n lodScale: number,\r\n lodOffset: number,\r\n onLoad?: Nullable<(internalTexture: Nullable) => void>,\r\n onError?: Nullable<(message?: string, exception?: any) => void>,\r\n format?: number,\r\n forcedExtension?: any,\r\n createPolynomials?: boolean\r\n ): InternalTexture;\r\n }\r\n}\r\n\r\n/**\r\n * Create a cube texture from prefiltered data (ie. the mipmaps contain ready to use data for PBR reflection)\r\n * @param rootUrl defines the url where the file to load is located\r\n * @param scene defines the current scene\r\n * @param lodScale defines scale to apply to the mip map selection\r\n * @param lodOffset defines offset to apply to the mip map selection\r\n * @param onLoad defines an optional callback raised when the texture is loaded\r\n * @param onError defines an optional callback raised if there is an issue to load the texture\r\n * @param format defines the format of the data\r\n * @param forcedExtension defines the extension to use to pick the right loader\r\n * @param createPolynomials defines wheter or not to create polynomails harmonics for the texture\r\n * @returns the cube texture as an InternalTexture\r\n */\r\nThinEngine.prototype.createPrefilteredCubeTexture = function (\r\n rootUrl: string,\r\n scene: Nullable,\r\n lodScale: number,\r\n lodOffset: number,\r\n onLoad: Nullable<(internalTexture: Nullable) => void> = null,\r\n onError: Nullable<(message?: string, exception?: any) => void> = null,\r\n format?: number,\r\n forcedExtension: any = null,\r\n createPolynomials: boolean = true\r\n): InternalTexture {\r\n const callback = (loadData: any) => {\r\n if (!loadData) {\r\n if (onLoad) {\r\n onLoad(null);\r\n }\r\n return;\r\n }\r\n\r\n const texture = loadData.texture as InternalTexture;\r\n if (!createPolynomials) {\r\n texture._sphericalPolynomial = new SphericalPolynomial();\r\n } else if (loadData.info.sphericalPolynomial) {\r\n texture._sphericalPolynomial = loadData.info.sphericalPolynomial;\r\n }\r\n texture._source = InternalTextureSource.CubePrefiltered;\r\n\r\n if (this.getCaps().textureLOD) {\r\n // Do not add extra process if texture lod is supported.\r\n if (onLoad) {\r\n onLoad(texture);\r\n }\r\n return;\r\n }\r\n\r\n const mipSlices = 3;\r\n\r\n const gl = this._gl;\r\n const width = loadData.width;\r\n if (!width) {\r\n return;\r\n }\r\n\r\n const textures: BaseTexture[] = [];\r\n for (let i = 0; i < mipSlices; i++) {\r\n //compute LOD from even spacing in smoothness (matching shader calculation)\r\n const smoothness = i / (mipSlices - 1);\r\n const roughness = 1 - smoothness;\r\n\r\n const minLODIndex = lodOffset; // roughness = 0\r\n const maxLODIndex = Scalar.Log2(width) * lodScale + lodOffset; // roughness = 1\r\n\r\n const lodIndex = minLODIndex + (maxLODIndex - minLODIndex) * roughness;\r\n const mipmapIndex = Math.round(Math.min(Math.max(lodIndex, 0), maxLODIndex));\r\n\r\n const glTextureFromLod = new InternalTexture(this, InternalTextureSource.Temp);\r\n glTextureFromLod.type = texture.type;\r\n glTextureFromLod.format = texture.format;\r\n glTextureFromLod.width = Math.pow(2, Math.max(Scalar.Log2(width) - mipmapIndex, 0));\r\n glTextureFromLod.height = glTextureFromLod.width;\r\n glTextureFromLod.isCube = true;\r\n glTextureFromLod._cachedWrapU = Constants.TEXTURE_CLAMP_ADDRESSMODE;\r\n glTextureFromLod._cachedWrapV = Constants.TEXTURE_CLAMP_ADDRESSMODE;\r\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, glTextureFromLod, true);\r\n\r\n glTextureFromLod.samplingMode = Constants.TEXTURE_LINEAR_LINEAR;\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_MAG_FILTER, gl.LINEAR);\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_MIN_FILTER, gl.LINEAR);\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\r\n\r\n if (loadData.isDDS) {\r\n const info: DDSInfo = loadData.info;\r\n const data: any = loadData.data;\r\n this._unpackFlipY(info.isCompressed);\r\n\r\n DDSTools.UploadDDSLevels(this, glTextureFromLod, data, info, true, 6, mipmapIndex);\r\n } else {\r\n Logger.Warn(\"DDS is the only prefiltered cube map supported so far.\");\r\n }\r\n\r\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, null);\r\n\r\n // Wrap in a base texture for easy binding.\r\n const lodTexture = new BaseTexture(scene);\r\n lodTexture._isCube = true;\r\n lodTexture._texture = glTextureFromLod;\r\n\r\n glTextureFromLod.isReady = true;\r\n textures.push(lodTexture);\r\n }\r\n\r\n texture._lodTextureHigh = textures[2];\r\n texture._lodTextureMid = textures[1];\r\n texture._lodTextureLow = textures[0];\r\n\r\n if (onLoad) {\r\n onLoad(texture);\r\n }\r\n };\r\n\r\n return this.createCubeTexture(rootUrl, scene, null, false, callback, onError, format, forcedExtension, createPolynomials, lodScale, lodOffset);\r\n};\r\n","import type { Nullable } from \"../../../types\";\r\nimport { SphericalPolynomial } from \"../../../Maths/sphericalPolynomial\";\r\nimport { Engine } from \"../../../Engines/engine\";\r\nimport type { InternalTexture } from \"../../../Materials/Textures/internalTexture\";\r\nimport type { IInternalTextureLoader } from \"../../../Materials/Textures/internalTextureLoader\";\r\nimport type { DDSInfo } from \"../../../Misc/dds\";\r\nimport { DDSTools } from \"../../../Misc/dds\";\r\n\r\n/**\r\n * Implementation of the DDS Texture Loader.\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class _DDSTextureLoader implements IInternalTextureLoader {\r\n /**\r\n * Defines whether the loader supports cascade loading the different faces.\r\n */\r\n public readonly supportCascades = true;\r\n\r\n /**\r\n * This returns if the loader support the current file information.\r\n * @param extension defines the file extension of the file being loaded\r\n * @returns true if the loader can load the specified file\r\n */\r\n public canLoad(extension: string): boolean {\r\n return extension.endsWith(\".dds\");\r\n }\r\n\r\n /**\r\n * Uploads the cube texture data to the WebGL texture. It has already been bound.\r\n * @param imgs contains the cube maps\r\n * @param texture defines the BabylonJS internal texture\r\n * @param createPolynomials will be true if polynomials have been requested\r\n * @param onLoad defines the callback to trigger once the texture is ready\r\n */\r\n public loadCubeData(imgs: ArrayBufferView | ArrayBufferView[], texture: InternalTexture, createPolynomials: boolean, onLoad: Nullable<(data?: any) => void>): void {\r\n const engine = texture.getEngine() as Engine;\r\n let info: DDSInfo | undefined;\r\n let loadMipmap: boolean = false;\r\n let maxLevel: number = 1000;\r\n if (Array.isArray(imgs)) {\r\n for (let index = 0; index < imgs.length; index++) {\r\n const data = imgs[index];\r\n info = DDSTools.GetDDSInfo(data);\r\n\r\n texture.width = info.width;\r\n texture.height = info.height;\r\n\r\n loadMipmap = (info.isRGB || info.isLuminance || info.mipmapCount > 1) && texture.generateMipMaps;\r\n\r\n engine._unpackFlipY(info.isCompressed);\r\n\r\n DDSTools.UploadDDSLevels(engine, texture, data, info, loadMipmap, 6, -1, index);\r\n\r\n if (!info.isFourCC && info.mipmapCount === 1) {\r\n engine.generateMipMapsForCubemap(texture);\r\n } else {\r\n maxLevel = info.mipmapCount - 1;\r\n }\r\n }\r\n } else {\r\n const data = imgs;\r\n info = DDSTools.GetDDSInfo(data);\r\n\r\n texture.width = info.width;\r\n texture.height = info.height;\r\n\r\n if (createPolynomials) {\r\n info.sphericalPolynomial = new SphericalPolynomial();\r\n }\r\n\r\n loadMipmap = (info.isRGB || info.isLuminance || info.mipmapCount > 1) && texture.generateMipMaps;\r\n engine._unpackFlipY(info.isCompressed);\r\n\r\n DDSTools.UploadDDSLevels(engine, texture, data, info, loadMipmap, 6);\r\n\r\n if (!info.isFourCC && info.mipmapCount === 1) {\r\n // Do not unbind as we still need to set the parameters.\r\n engine.generateMipMapsForCubemap(texture, false);\r\n } else {\r\n maxLevel = info.mipmapCount - 1;\r\n }\r\n }\r\n engine._setCubeMapTextureParams(texture, loadMipmap, maxLevel);\r\n texture.isReady = true;\r\n texture.onLoadedObservable.notifyObservers(texture);\r\n texture.onLoadedObservable.clear();\r\n\r\n if (onLoad) {\r\n onLoad({ isDDS: true, width: texture.width, info, data: imgs, texture });\r\n }\r\n }\r\n\r\n /**\r\n * Uploads the 2D texture data to the WebGL texture. It has already been bound once in the callback.\r\n * @param data contains the texture data\r\n * @param texture defines the BabylonJS internal texture\r\n * @param callback defines the method to call once ready to upload\r\n */\r\n public loadData(\r\n data: ArrayBufferView,\r\n texture: InternalTexture,\r\n callback: (width: number, height: number, loadMipmap: boolean, isCompressed: boolean, done: () => void) => void\r\n ): void {\r\n const info = DDSTools.GetDDSInfo(data);\r\n\r\n const loadMipmap = (info.isRGB || info.isLuminance || info.mipmapCount > 1) && texture.generateMipMaps && info.width >> (info.mipmapCount - 1) === 1;\r\n callback(info.width, info.height, loadMipmap, info.isFourCC, () => {\r\n DDSTools.UploadDDSLevels(texture.getEngine(), texture, data, info, loadMipmap, 1);\r\n });\r\n }\r\n}\r\n\r\n// Register the loader.\r\nEngine._TextureLoaders.push(new _DDSTextureLoader());\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/helperFunctions\";\n\nconst name = \"rgbdEncodePixelShader\";\nconst shader = `varying vec2 vUV;uniform sampler2D textureSampler;\n#include\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) \n{gl_FragColor=toRGBD(texture2D(textureSampler,vUV).rgb);}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const rgbdEncodePixelShader = { name, shader };\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport type { Nullable } from \"../types\";\r\nimport { Tools } from \"./tools\";\r\nimport { Vector3 } from \"../Maths/math.vector\";\r\nimport { Scalar } from \"../Maths/math.scalar\";\r\nimport { SphericalPolynomial } from \"../Maths/sphericalPolynomial\";\r\nimport { InternalTexture, InternalTextureSource } from \"../Materials/Textures/internalTexture\";\r\nimport { BaseTexture } from \"../Materials/Textures/baseTexture\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport { Scene } from \"../scene\";\r\nimport { PostProcess } from \"../PostProcesses/postProcess\";\r\nimport { Logger } from \"../Misc/logger\";\r\nimport type { Engine } from \"../Engines/engine\";\r\nimport { RGBDTextureTools } from \"./rgbdTextureTools\";\r\nimport type { RenderTargetWrapper } from \"../Engines/renderTargetWrapper\";\r\n\r\nimport \"../Engines/Extensions/engine.renderTargetCube\";\r\nimport \"../Engines/Extensions/engine.readTexture\";\r\nimport \"../Materials/Textures/baseTexture.polynomial\";\r\n\r\nimport \"../Shaders/rgbdEncode.fragment\";\r\nimport \"../Shaders/rgbdDecode.fragment\";\r\nimport { DumpTools } from \"../Misc/dumpTools\";\r\n\r\nconst DefaultEnvironmentTextureImageType = \"image/png\";\r\nconst CurrentVersion = 2;\r\n\r\n/**\r\n * Raw texture data and descriptor sufficient for WebGL texture upload\r\n */\r\nexport type EnvironmentTextureInfo = EnvironmentTextureInfoV1 | EnvironmentTextureInfoV2;\r\n\r\n/**\r\n * v1 of EnvironmentTextureInfo\r\n */\r\ninterface EnvironmentTextureInfoV1 {\r\n /**\r\n * Version of the environment map\r\n */\r\n version: 1;\r\n\r\n /**\r\n * Width of image\r\n */\r\n width: number;\r\n\r\n /**\r\n * Irradiance information stored in the file.\r\n */\r\n irradiance: any;\r\n\r\n /**\r\n * Specular information stored in the file.\r\n */\r\n specular: any;\r\n}\r\n\r\n/**\r\n * v2 of EnvironmentTextureInfo\r\n */\r\ninterface EnvironmentTextureInfoV2 {\r\n /**\r\n * Version of the environment map\r\n */\r\n version: 2;\r\n\r\n /**\r\n * Width of image\r\n */\r\n width: number;\r\n\r\n /**\r\n * Irradiance information stored in the file.\r\n */\r\n irradiance: any;\r\n\r\n /**\r\n * Specular information stored in the file.\r\n */\r\n specular: any;\r\n\r\n /**\r\n * The mime type used to encode the image data.\r\n */\r\n imageType: string;\r\n}\r\n\r\n/**\r\n * Defines One Image in the file. It requires only the position in the file\r\n * as well as the length.\r\n */\r\ninterface BufferImageData {\r\n /**\r\n * Length of the image data.\r\n */\r\n length: number;\r\n /**\r\n * Position of the data from the null terminator delimiting the end of the JSON.\r\n */\r\n position: number;\r\n}\r\n\r\n/**\r\n * Defines the specular data enclosed in the file.\r\n * This corresponds to the version 1 of the data.\r\n */\r\nexport interface EnvironmentTextureSpecularInfoV1 {\r\n /**\r\n * Defines where the specular Payload is located. It is a runtime value only not stored in the file.\r\n */\r\n specularDataPosition?: number;\r\n /**\r\n * This contains all the images data needed to reconstruct the cubemap.\r\n */\r\n mipmaps: Array;\r\n /**\r\n * Defines the scale applied to environment texture. This manages the range of LOD level used for IBL according to the roughness.\r\n */\r\n lodGenerationScale: number;\r\n}\r\n\r\n/**\r\n * Defines the required storage to save the environment irradiance information.\r\n */\r\ninterface EnvironmentTextureIrradianceInfoV1 {\r\n x: Array;\r\n y: Array;\r\n z: Array;\r\n\r\n xx: Array;\r\n yy: Array;\r\n zz: Array;\r\n\r\n yz: Array;\r\n zx: Array;\r\n xy: Array;\r\n}\r\n\r\n/**\r\n * Options for creating environment textures\r\n */\r\nexport interface CreateEnvTextureOptions {\r\n /**\r\n * The mime type of encoded images.\r\n */\r\n imageType?: string;\r\n\r\n /**\r\n * the image quality of encoded WebP images.\r\n */\r\n imageQuality?: number;\r\n}\r\n\r\n/**\r\n * Magic number identifying the env file.\r\n */\r\nconst MagicBytes = [0x86, 0x16, 0x87, 0x96, 0xf6, 0xd6, 0x96, 0x36];\r\n\r\n/**\r\n * Gets the environment info from an env file.\r\n * @param data The array buffer containing the .env bytes.\r\n * @returns the environment file info (the json header) if successfully parsed, normalized to the latest supported version.\r\n */\r\nexport function GetEnvInfo(data: ArrayBufferView): Nullable {\r\n const dataView = new DataView(data.buffer, data.byteOffset, data.byteLength);\r\n let pos = 0;\r\n\r\n for (let i = 0; i < MagicBytes.length; i++) {\r\n if (dataView.getUint8(pos++) !== MagicBytes[i]) {\r\n Logger.Error(\"Not a babylon environment map\");\r\n return null;\r\n }\r\n }\r\n\r\n // Read json manifest - collect characters up to null terminator\r\n let manifestString = \"\";\r\n let charCode = 0x00;\r\n while ((charCode = dataView.getUint8(pos++))) {\r\n manifestString += String.fromCharCode(charCode);\r\n }\r\n\r\n let manifest: EnvironmentTextureInfo = JSON.parse(manifestString);\r\n manifest = normalizeEnvInfo(manifest);\r\n if (manifest.specular) {\r\n // Extend the header with the position of the payload.\r\n manifest.specular.specularDataPosition = pos;\r\n // Fallback to 0.8 exactly if lodGenerationScale is not defined for backward compatibility.\r\n manifest.specular.lodGenerationScale = manifest.specular.lodGenerationScale || 0.8;\r\n }\r\n\r\n return manifest;\r\n}\r\n\r\n/**\r\n * Normalizes any supported version of the environment file info to the latest version\r\n * @param info environment file info on any supported version\r\n * @returns environment file info in the latest supported version\r\n * @private\r\n */\r\nexport function normalizeEnvInfo(info: EnvironmentTextureInfo): EnvironmentTextureInfoV2 {\r\n if (info.version > CurrentVersion) {\r\n throw new Error(`Unsupported babylon environment map version \"${info.version}\". Latest supported version is \"${CurrentVersion}\".`);\r\n }\r\n\r\n if (info.version === 2) {\r\n return info;\r\n }\r\n\r\n // Migrate a v1 info to v2\r\n info = { ...info, version: 2, imageType: DefaultEnvironmentTextureImageType };\r\n\r\n return info;\r\n}\r\n\r\n/**\r\n * Creates an environment texture from a loaded cube texture.\r\n * @param texture defines the cube texture to convert in env file\r\n * @param options options for the conversion process\r\n * @param options.imageType the mime type for the encoded images, with support for \"image/png\" (default) and \"image/webp\"\r\n * @param options.imageQuality the image quality of encoded WebP images.\r\n * @returns a promise containing the environment data if successful.\r\n */\r\nexport async function CreateEnvTextureAsync(texture: BaseTexture, options: CreateEnvTextureOptions = {}): Promise {\r\n const internalTexture = texture.getInternalTexture();\r\n if (!internalTexture) {\r\n return Promise.reject(\"The cube texture is invalid.\");\r\n }\r\n\r\n const imageType = options.imageType ?? DefaultEnvironmentTextureImageType;\r\n\r\n const engine = internalTexture.getEngine() as Engine;\r\n\r\n if (\r\n texture.textureType !== Constants.TEXTURETYPE_HALF_FLOAT &&\r\n texture.textureType !== Constants.TEXTURETYPE_FLOAT &&\r\n texture.textureType !== Constants.TEXTURETYPE_UNSIGNED_BYTE &&\r\n texture.textureType !== Constants.TEXTURETYPE_UNSIGNED_INT &&\r\n texture.textureType !== Constants.TEXTURETYPE_UNSIGNED_INTEGER &&\r\n texture.textureType !== -1\r\n ) {\r\n return Promise.reject(\"The cube texture should allow HDR (Full Float or Half Float).\");\r\n }\r\n\r\n let textureType = Constants.TEXTURETYPE_FLOAT;\r\n if (!engine.getCaps().textureFloatRender) {\r\n textureType = Constants.TEXTURETYPE_HALF_FLOAT;\r\n if (!engine.getCaps().textureHalfFloatRender) {\r\n return Promise.reject(\"Env texture can only be created when the browser supports half float or full float rendering.\");\r\n }\r\n }\r\n\r\n // sphericalPolynomial is lazy loaded so simply accessing it should trigger the computation.\r\n texture.sphericalPolynomial;\r\n\r\n // Lets keep track of the polynomial promise so we can wait for it to be ready before generating the pixels.\r\n const sphericalPolynomialPromise = texture.getInternalTexture()?._sphericalPolynomialPromise;\r\n\r\n const cubeWidth = internalTexture.width;\r\n const hostingScene = new Scene(engine);\r\n const specularTextures: { [key: number]: ArrayBuffer } = {};\r\n\r\n // As we are going to readPixels the faces of the cube, make sure the drawing/update commands for the cube texture are fully sent to the GPU in case it is drawn for the first time in this very frame!\r\n engine.flushFramebuffer();\r\n\r\n // Read and collect all mipmaps data from the cube.\r\n const mipmapsCount = Scalar.ILog2(internalTexture.width);\r\n for (let i = 0; i <= mipmapsCount; i++) {\r\n const faceWidth = Math.pow(2, mipmapsCount - i);\r\n\r\n // All faces of the cube.\r\n for (let face = 0; face < 6; face++) {\r\n let faceData = await texture.readPixels(face, i, undefined, false);\r\n if (faceData && faceData.byteLength === (faceData as Uint8Array).length) {\r\n const faceDataFloat = new Float32Array(faceData!.byteLength * 4);\r\n for (let i = 0; i < faceData.byteLength; i++) {\r\n faceDataFloat[i] = (faceData as Uint8Array)[i] / 255;\r\n // Gamma to linear\r\n faceDataFloat[i] = Math.pow(faceDataFloat[i], 2.2);\r\n }\r\n faceData = faceDataFloat;\r\n } else if (faceData && texture.gammaSpace) {\r\n const floatData = faceData as Float32Array;\r\n for (let i = 0; i < floatData.length; i++) {\r\n // Gamma to linear\r\n floatData[i] = Math.pow(floatData[i], 2.2);\r\n }\r\n }\r\n\r\n const tempTexture = engine.createRawTexture(\r\n faceData,\r\n faceWidth,\r\n faceWidth,\r\n Constants.TEXTUREFORMAT_RGBA,\r\n false,\r\n true,\r\n Constants.TEXTURE_NEAREST_SAMPLINGMODE,\r\n null,\r\n textureType\r\n );\r\n\r\n await RGBDTextureTools.EncodeTextureToRGBD(tempTexture, hostingScene, textureType);\r\n\r\n const rgbdEncodedData = await engine._readTexturePixels(tempTexture, faceWidth, faceWidth);\r\n\r\n const imageEncodedData = await DumpTools.DumpDataAsync(faceWidth, faceWidth, rgbdEncodedData, imageType, undefined, false, true, options.imageQuality);\r\n\r\n specularTextures[i * 6 + face] = imageEncodedData as ArrayBuffer;\r\n\r\n tempTexture.dispose();\r\n }\r\n }\r\n\r\n // We can delete the hosting scene keeping track of all the creation objects\r\n hostingScene.dispose();\r\n\r\n // Ensure completion of the polynomial creation promise.\r\n if (sphericalPolynomialPromise) {\r\n await sphericalPolynomialPromise;\r\n }\r\n\r\n // Creates the json header for the env texture\r\n const info: EnvironmentTextureInfo = {\r\n version: CurrentVersion,\r\n width: cubeWidth,\r\n imageType,\r\n irradiance: _CreateEnvTextureIrradiance(texture),\r\n specular: {\r\n mipmaps: [],\r\n lodGenerationScale: texture.lodGenerationScale,\r\n },\r\n };\r\n\r\n // Sets the specular image data information\r\n let position = 0;\r\n for (let i = 0; i <= mipmapsCount; i++) {\r\n for (let face = 0; face < 6; face++) {\r\n const byteLength = specularTextures[i * 6 + face].byteLength;\r\n info.specular.mipmaps.push({\r\n length: byteLength,\r\n position: position,\r\n });\r\n position += byteLength;\r\n }\r\n }\r\n\r\n // Encode the JSON as an array buffer\r\n const infoString = JSON.stringify(info);\r\n const infoBuffer = new ArrayBuffer(infoString.length + 1);\r\n const infoView = new Uint8Array(infoBuffer); // Limited to ascii subset matching unicode.\r\n for (let i = 0, strLen = infoString.length; i < strLen; i++) {\r\n infoView[i] = infoString.charCodeAt(i);\r\n }\r\n // Ends up with a null terminator for easier parsing\r\n infoView[infoString.length] = 0x00;\r\n\r\n // Computes the final required size and creates the storage\r\n const totalSize = MagicBytes.length + position + infoBuffer.byteLength;\r\n const finalBuffer = new ArrayBuffer(totalSize);\r\n const finalBufferView = new Uint8Array(finalBuffer);\r\n const dataView = new DataView(finalBuffer);\r\n\r\n // Copy the magic bytes identifying the file in\r\n let pos = 0;\r\n for (let i = 0; i < MagicBytes.length; i++) {\r\n dataView.setUint8(pos++, MagicBytes[i]);\r\n }\r\n\r\n // Add the json info\r\n finalBufferView.set(new Uint8Array(infoBuffer), pos);\r\n pos += infoBuffer.byteLength;\r\n\r\n // Finally inserts the texture data\r\n for (let i = 0; i <= mipmapsCount; i++) {\r\n for (let face = 0; face < 6; face++) {\r\n const dataBuffer = specularTextures[i * 6 + face];\r\n finalBufferView.set(new Uint8Array(dataBuffer), pos);\r\n pos += dataBuffer.byteLength;\r\n }\r\n }\r\n\r\n // Voila\r\n return finalBuffer;\r\n}\r\n\r\n/**\r\n * Creates a JSON representation of the spherical data.\r\n * @param texture defines the texture containing the polynomials\r\n * @returns the JSON representation of the spherical info\r\n */\r\nfunction _CreateEnvTextureIrradiance(texture: BaseTexture): Nullable {\r\n const polynmials = texture.sphericalPolynomial;\r\n if (polynmials == null) {\r\n return null;\r\n }\r\n\r\n return {\r\n x: [polynmials.x.x, polynmials.x.y, polynmials.x.z],\r\n y: [polynmials.y.x, polynmials.y.y, polynmials.y.z],\r\n z: [polynmials.z.x, polynmials.z.y, polynmials.z.z],\r\n\r\n xx: [polynmials.xx.x, polynmials.xx.y, polynmials.xx.z],\r\n yy: [polynmials.yy.x, polynmials.yy.y, polynmials.yy.z],\r\n zz: [polynmials.zz.x, polynmials.zz.y, polynmials.zz.z],\r\n\r\n yz: [polynmials.yz.x, polynmials.yz.y, polynmials.yz.z],\r\n zx: [polynmials.zx.x, polynmials.zx.y, polynmials.zx.z],\r\n xy: [polynmials.xy.x, polynmials.xy.y, polynmials.xy.z],\r\n } as any;\r\n}\r\n\r\n/**\r\n * Creates the ArrayBufferViews used for initializing environment texture image data.\r\n * @param data the image data\r\n * @param info parameters that determine what views will be created for accessing the underlying buffer\r\n * @returns the views described by info providing access to the underlying buffer\r\n */\r\nexport function CreateImageDataArrayBufferViews(data: ArrayBufferView, info: EnvironmentTextureInfo): Array> {\r\n info = normalizeEnvInfo(info);\r\n\r\n const specularInfo = info.specular as EnvironmentTextureSpecularInfoV1;\r\n\r\n // Double checks the enclosed info\r\n let mipmapsCount = Scalar.Log2(info.width);\r\n mipmapsCount = Math.round(mipmapsCount) + 1;\r\n if (specularInfo.mipmaps.length !== 6 * mipmapsCount) {\r\n throw new Error(`Unsupported specular mipmaps number \"${specularInfo.mipmaps.length}\"`);\r\n }\r\n\r\n const imageData = new Array>(mipmapsCount);\r\n for (let i = 0; i < mipmapsCount; i++) {\r\n imageData[i] = new Array(6);\r\n for (let face = 0; face < 6; face++) {\r\n const imageInfo = specularInfo.mipmaps[i * 6 + face];\r\n imageData[i][face] = new Uint8Array(data.buffer, data.byteOffset + specularInfo.specularDataPosition! + imageInfo.position, imageInfo.length);\r\n }\r\n }\r\n\r\n return imageData;\r\n}\r\n\r\n/**\r\n * Uploads the texture info contained in the env file to the GPU.\r\n * @param texture defines the internal texture to upload to\r\n * @param data defines the data to load\r\n * @param info defines the texture info retrieved through the GetEnvInfo method\r\n * @returns a promise\r\n */\r\nexport function UploadEnvLevelsAsync(texture: InternalTexture, data: ArrayBufferView, info: EnvironmentTextureInfo): Promise {\r\n info = normalizeEnvInfo(info);\r\n\r\n const specularInfo = info.specular as EnvironmentTextureSpecularInfoV1;\r\n if (!specularInfo) {\r\n // Nothing else parsed so far\r\n return Promise.resolve();\r\n }\r\n\r\n texture._lodGenerationScale = specularInfo.lodGenerationScale;\r\n\r\n const imageData = CreateImageDataArrayBufferViews(data, info);\r\n\r\n return UploadLevelsAsync(texture, imageData, info.imageType);\r\n}\r\n\r\nfunction _OnImageReadyAsync(\r\n image: HTMLImageElement | ImageBitmap,\r\n engine: Engine,\r\n expandTexture: boolean,\r\n rgbdPostProcess: Nullable,\r\n url: string,\r\n face: number,\r\n i: number,\r\n generateNonLODTextures: boolean,\r\n lodTextures: Nullable<{ [lod: number]: BaseTexture }>,\r\n cubeRtt: Nullable,\r\n texture: InternalTexture\r\n): Promise {\r\n return new Promise((resolve, reject) => {\r\n if (expandTexture) {\r\n const tempTexture = engine.createTexture(\r\n null,\r\n true,\r\n true,\r\n null,\r\n Constants.TEXTURE_NEAREST_SAMPLINGMODE,\r\n null,\r\n (message) => {\r\n reject(message);\r\n },\r\n image\r\n );\r\n\r\n rgbdPostProcess!.getEffect().executeWhenCompiled(() => {\r\n // Uncompress the data to a RTT\r\n rgbdPostProcess!.externalTextureSamplerBinding = true;\r\n rgbdPostProcess!.onApply = (effect) => {\r\n effect._bindTexture(\"textureSampler\", tempTexture);\r\n effect.setFloat2(\"scale\", 1, engine._features.needsInvertingBitmap && image instanceof ImageBitmap ? -1 : 1);\r\n };\r\n\r\n if (!engine.scenes.length) {\r\n return;\r\n }\r\n\r\n engine.scenes[0].postProcessManager.directRender([rgbdPostProcess!], cubeRtt, true, face, i);\r\n\r\n // Cleanup\r\n engine.restoreDefaultFramebuffer();\r\n tempTexture.dispose();\r\n URL.revokeObjectURL(url);\r\n resolve();\r\n });\r\n } else {\r\n engine._uploadImageToTexture(texture, image, face, i);\r\n\r\n // Upload the face to the non lod texture support\r\n if (generateNonLODTextures) {\r\n const lodTexture = lodTextures![i];\r\n if (lodTexture) {\r\n engine._uploadImageToTexture(lodTexture._texture!, image, face, 0);\r\n }\r\n }\r\n resolve();\r\n }\r\n });\r\n}\r\n\r\n/**\r\n * Uploads the levels of image data to the GPU.\r\n * @param texture defines the internal texture to upload to\r\n * @param imageData defines the array buffer views of image data [mipmap][face]\r\n * @param imageType the mime type of the image data\r\n * @returns a promise\r\n */\r\nexport function UploadLevelsAsync(texture: InternalTexture, imageData: ArrayBufferView[][], imageType: string = DefaultEnvironmentTextureImageType): Promise {\r\n if (!Tools.IsExponentOfTwo(texture.width)) {\r\n throw new Error(\"Texture size must be a power of two\");\r\n }\r\n\r\n const mipmapsCount = Scalar.ILog2(texture.width) + 1;\r\n\r\n // Gets everything ready.\r\n const engine = texture.getEngine() as Engine;\r\n let expandTexture = false;\r\n let generateNonLODTextures = false;\r\n let rgbdPostProcess: Nullable = null;\r\n let cubeRtt: Nullable = null;\r\n let lodTextures: Nullable<{ [lod: number]: BaseTexture }> = null;\r\n const caps = engine.getCaps();\r\n\r\n texture.format = Constants.TEXTUREFORMAT_RGBA;\r\n texture.type = Constants.TEXTURETYPE_UNSIGNED_INT;\r\n texture.generateMipMaps = true;\r\n texture._cachedAnisotropicFilteringLevel = null;\r\n engine.updateTextureSamplingMode(Constants.TEXTURE_TRILINEAR_SAMPLINGMODE, texture);\r\n\r\n // Add extra process if texture lod is not supported\r\n if (!caps.textureLOD) {\r\n expandTexture = false;\r\n generateNonLODTextures = true;\r\n lodTextures = {};\r\n }\r\n // in webgl 1 there are no ways to either render or copy lod level information for float textures.\r\n else if (!engine._features.supportRenderAndCopyToLodForFloatTextures) {\r\n expandTexture = false;\r\n }\r\n // If half float available we can uncompress the texture\r\n else if (caps.textureHalfFloatRender && caps.textureHalfFloatLinearFiltering) {\r\n expandTexture = true;\r\n texture.type = Constants.TEXTURETYPE_HALF_FLOAT;\r\n }\r\n // If full float available we can uncompress the texture\r\n else if (caps.textureFloatRender && caps.textureFloatLinearFiltering) {\r\n expandTexture = true;\r\n texture.type = Constants.TEXTURETYPE_FLOAT;\r\n }\r\n\r\n // Expand the texture if possible\r\n if (expandTexture) {\r\n // Simply run through the decode PP\r\n rgbdPostProcess = new PostProcess(\r\n \"rgbdDecode\",\r\n \"rgbdDecode\",\r\n null,\r\n null,\r\n 1,\r\n null,\r\n Constants.TEXTURE_TRILINEAR_SAMPLINGMODE,\r\n engine,\r\n false,\r\n undefined,\r\n texture.type,\r\n undefined,\r\n null,\r\n false\r\n );\r\n\r\n texture._isRGBD = false;\r\n texture.invertY = false;\r\n cubeRtt = engine.createRenderTargetCubeTexture(texture.width, {\r\n generateDepthBuffer: false,\r\n generateMipMaps: true,\r\n generateStencilBuffer: false,\r\n samplingMode: Constants.TEXTURE_TRILINEAR_SAMPLINGMODE,\r\n type: texture.type,\r\n format: Constants.TEXTUREFORMAT_RGBA,\r\n });\r\n } else {\r\n texture._isRGBD = true;\r\n texture.invertY = true;\r\n\r\n // In case of missing support, applies the same patch than DDS files.\r\n if (generateNonLODTextures) {\r\n const mipSlices = 3;\r\n const scale = texture._lodGenerationScale;\r\n const offset = texture._lodGenerationOffset;\r\n\r\n for (let i = 0; i < mipSlices; i++) {\r\n //compute LOD from even spacing in smoothness (matching shader calculation)\r\n const smoothness = i / (mipSlices - 1);\r\n const roughness = 1 - smoothness;\r\n\r\n const minLODIndex = offset; // roughness = 0\r\n const maxLODIndex = (mipmapsCount - 1) * scale + offset; // roughness = 1 (mipmaps start from 0)\r\n\r\n const lodIndex = minLODIndex + (maxLODIndex - minLODIndex) * roughness;\r\n const mipmapIndex = Math.round(Math.min(Math.max(lodIndex, 0), maxLODIndex));\r\n\r\n const glTextureFromLod = new InternalTexture(engine, InternalTextureSource.Temp);\r\n glTextureFromLod.isCube = true;\r\n glTextureFromLod.invertY = true;\r\n glTextureFromLod.generateMipMaps = false;\r\n engine.updateTextureSamplingMode(Constants.TEXTURE_LINEAR_LINEAR, glTextureFromLod);\r\n\r\n // Wrap in a base texture for easy binding.\r\n const lodTexture = new BaseTexture(null);\r\n lodTexture._isCube = true;\r\n lodTexture._texture = glTextureFromLod;\r\n lodTextures![mipmapIndex] = lodTexture;\r\n\r\n switch (i) {\r\n case 0:\r\n texture._lodTextureLow = lodTexture;\r\n break;\r\n case 1:\r\n texture._lodTextureMid = lodTexture;\r\n break;\r\n case 2:\r\n texture._lodTextureHigh = lodTexture;\r\n break;\r\n }\r\n }\r\n }\r\n }\r\n\r\n const promises: Promise[] = [];\r\n // All mipmaps up to provided number of images\r\n for (let i = 0; i < imageData.length; i++) {\r\n // All faces\r\n for (let face = 0; face < 6; face++) {\r\n // Constructs an image element from image data\r\n const bytes = imageData[i][face];\r\n const blob = new Blob([bytes], { type: imageType });\r\n const url = URL.createObjectURL(blob);\r\n let promise: Promise;\r\n\r\n if (typeof Image === \"undefined\" || engine._features.forceBitmapOverHTMLImageElement) {\r\n promise = engine.createImageBitmap(blob, { premultiplyAlpha: \"none\" }).then((img) => {\r\n return _OnImageReadyAsync(img, engine, expandTexture, rgbdPostProcess, url, face, i, generateNonLODTextures, lodTextures, cubeRtt, texture);\r\n });\r\n } else {\r\n const image = new Image();\r\n image.src = url;\r\n\r\n // Enqueue promise to upload to the texture.\r\n promise = new Promise((resolve, reject) => {\r\n image.onload = () => {\r\n _OnImageReadyAsync(image, engine, expandTexture, rgbdPostProcess, url, face, i, generateNonLODTextures, lodTextures, cubeRtt, texture)\r\n .then(() => resolve())\r\n .catch((reason) => {\r\n reject(reason);\r\n });\r\n };\r\n image.onerror = (error) => {\r\n reject(error);\r\n };\r\n });\r\n }\r\n promises.push(promise);\r\n }\r\n }\r\n\r\n // Fill remaining mipmaps with black textures.\r\n if (imageData.length < mipmapsCount) {\r\n let data: ArrayBufferView;\r\n const size = Math.pow(2, mipmapsCount - 1 - imageData.length);\r\n const dataLength = size * size * 4;\r\n switch (texture.type) {\r\n case Constants.TEXTURETYPE_UNSIGNED_INT: {\r\n data = new Uint8Array(dataLength);\r\n break;\r\n }\r\n case Constants.TEXTURETYPE_HALF_FLOAT: {\r\n data = new Uint16Array(dataLength);\r\n break;\r\n }\r\n case Constants.TEXTURETYPE_FLOAT: {\r\n data = new Float32Array(dataLength);\r\n break;\r\n }\r\n }\r\n for (let i = imageData.length; i < mipmapsCount; i++) {\r\n for (let face = 0; face < 6; face++) {\r\n engine._uploadArrayBufferViewToTexture(texture, data!, face, i);\r\n }\r\n }\r\n }\r\n\r\n // Once all done, finishes the cleanup and return\r\n return Promise.all(promises).then(() => {\r\n // Release temp RTT.\r\n if (cubeRtt) {\r\n engine._releaseTexture(texture);\r\n cubeRtt._swapAndDie(texture);\r\n }\r\n // Release temp Post Process.\r\n if (rgbdPostProcess) {\r\n rgbdPostProcess.dispose();\r\n }\r\n // Flag internal texture as ready in case they are in use.\r\n if (generateNonLODTextures) {\r\n if (texture._lodTextureHigh && texture._lodTextureHigh._texture) {\r\n texture._lodTextureHigh._texture.isReady = true;\r\n }\r\n if (texture._lodTextureMid && texture._lodTextureMid._texture) {\r\n texture._lodTextureMid._texture.isReady = true;\r\n }\r\n if (texture._lodTextureLow && texture._lodTextureLow._texture) {\r\n texture._lodTextureLow._texture.isReady = true;\r\n }\r\n }\r\n });\r\n}\r\n\r\n/**\r\n * Uploads spherical polynomials information to the texture.\r\n * @param texture defines the texture we are trying to upload the information to\r\n * @param info defines the environment texture info retrieved through the GetEnvInfo method\r\n */\r\nexport function UploadEnvSpherical(texture: InternalTexture, info: EnvironmentTextureInfo): void {\r\n info = normalizeEnvInfo(info);\r\n\r\n const irradianceInfo = info.irradiance as EnvironmentTextureIrradianceInfoV1;\r\n if (!irradianceInfo) {\r\n return;\r\n }\r\n\r\n const sp = new SphericalPolynomial();\r\n Vector3.FromArrayToRef(irradianceInfo.x, 0, sp.x);\r\n Vector3.FromArrayToRef(irradianceInfo.y, 0, sp.y);\r\n Vector3.FromArrayToRef(irradianceInfo.z, 0, sp.z);\r\n Vector3.FromArrayToRef(irradianceInfo.xx, 0, sp.xx);\r\n Vector3.FromArrayToRef(irradianceInfo.yy, 0, sp.yy);\r\n Vector3.FromArrayToRef(irradianceInfo.zz, 0, sp.zz);\r\n Vector3.FromArrayToRef(irradianceInfo.yz, 0, sp.yz);\r\n Vector3.FromArrayToRef(irradianceInfo.zx, 0, sp.zx);\r\n Vector3.FromArrayToRef(irradianceInfo.xy, 0, sp.xy);\r\n texture._sphericalPolynomial = sp;\r\n}\r\n\r\n/**\r\n * @internal\r\n */\r\nexport function _UpdateRGBDAsync(\r\n internalTexture: InternalTexture,\r\n data: ArrayBufferView[][],\r\n sphericalPolynomial: Nullable,\r\n lodScale: number,\r\n lodOffset: number\r\n): Promise {\r\n const proxy = internalTexture\r\n .getEngine()\r\n .createRawCubeTexture(\r\n null,\r\n internalTexture.width,\r\n internalTexture.format,\r\n internalTexture.type,\r\n internalTexture.generateMipMaps,\r\n internalTexture.invertY,\r\n internalTexture.samplingMode,\r\n internalTexture._compression\r\n );\r\n const proxyPromise = UploadLevelsAsync(proxy, data).then(() => internalTexture);\r\n internalTexture.onRebuildCallback = (_internalTexture) => {\r\n return {\r\n proxy: proxyPromise,\r\n isReady: true,\r\n isAsync: true,\r\n };\r\n };\r\n internalTexture._source = InternalTextureSource.CubeRawRGBD;\r\n internalTexture._bufferViewArrayArray = data;\r\n internalTexture._lodGenerationScale = lodScale;\r\n internalTexture._lodGenerationOffset = lodOffset;\r\n internalTexture._sphericalPolynomial = sphericalPolynomial;\r\n\r\n return UploadLevelsAsync(internalTexture, data).then(() => {\r\n internalTexture.isReady = true;\r\n return internalTexture;\r\n });\r\n}\r\n\r\n/**\r\n * Sets of helpers addressing the serialization and deserialization of environment texture\r\n * stored in a BabylonJS env file.\r\n * Those files are usually stored as .env files.\r\n */\r\nexport const EnvironmentTextureTools = {\r\n /**\r\n * Gets the environment info from an env file.\r\n * @param data The array buffer containing the .env bytes.\r\n * @returns the environment file info (the json header) if successfully parsed, normalized to the latest supported version.\r\n */\r\n GetEnvInfo,\r\n\r\n /**\r\n * Creates an environment texture from a loaded cube texture.\r\n * @param texture defines the cube texture to convert in env file\r\n * @param options options for the conversion process\r\n * @param options.imageType the mime type for the encoded images, with support for \"image/png\" (default) and \"image/webp\"\r\n * @param options.imageQuality the image quality of encoded WebP images.\r\n * @returns a promise containing the environment data if successful.\r\n */\r\n CreateEnvTextureAsync,\r\n\r\n /**\r\n * Creates the ArrayBufferViews used for initializing environment texture image data.\r\n * @param data the image data\r\n * @param info parameters that determine what views will be created for accessing the underlying buffer\r\n * @returns the views described by info providing access to the underlying buffer\r\n */\r\n CreateImageDataArrayBufferViews,\r\n\r\n /**\r\n * Uploads the texture info contained in the env file to the GPU.\r\n * @param texture defines the internal texture to upload to\r\n * @param data defines the data to load\r\n * @param info defines the texture info retrieved through the GetEnvInfo method\r\n * @returns a promise\r\n */\r\n UploadEnvLevelsAsync,\r\n\r\n /**\r\n * Uploads the levels of image data to the GPU.\r\n * @param texture defines the internal texture to upload to\r\n * @param imageData defines the array buffer views of image data [mipmap][face]\r\n * @param imageType the mime type of the image data\r\n * @returns a promise\r\n */\r\n UploadLevelsAsync,\r\n\r\n /**\r\n * Uploads spherical polynomials information to the texture.\r\n * @param texture defines the texture we are trying to upload the information to\r\n * @param info defines the environment texture info retrieved through the GetEnvInfo method\r\n */\r\n UploadEnvSpherical,\r\n};\r\n","import { GetEnvInfo, UploadEnvLevelsAsync, UploadEnvSpherical } from \"../../../Misc/environmentTextureTools\";\r\nimport type { Nullable } from \"../../../types\";\r\nimport { Engine } from \"../../../Engines/engine\";\r\nimport type { InternalTexture } from \"../../../Materials/Textures/internalTexture\";\r\nimport type { IInternalTextureLoader } from \"../../../Materials/Textures/internalTextureLoader\";\r\n\r\n/**\r\n * Implementation of the ENV Texture Loader.\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class _ENVTextureLoader implements IInternalTextureLoader {\r\n /**\r\n * Defines whether the loader supports cascade loading the different faces.\r\n */\r\n public readonly supportCascades = false;\r\n\r\n /**\r\n * This returns if the loader support the current file information.\r\n * @param extension defines the file extension of the file being loaded\r\n * @returns true if the loader can load the specified file\r\n */\r\n public canLoad(extension: string): boolean {\r\n return extension.endsWith(\".env\");\r\n }\r\n\r\n /**\r\n * Uploads the cube texture data to the WebGL texture. It has already been bound.\r\n * @param data contains the texture data\r\n * @param texture defines the BabylonJS internal texture\r\n * @param createPolynomials will be true if polynomials have been requested\r\n * @param onLoad defines the callback to trigger once the texture is ready\r\n * @param onError defines the callback to trigger in case of error\r\n */\r\n public loadCubeData(\r\n data: ArrayBufferView | ArrayBufferView[],\r\n texture: InternalTexture,\r\n createPolynomials: boolean,\r\n onLoad: Nullable<(data?: any) => void>,\r\n onError: Nullable<(message?: string, exception?: any) => void>\r\n ): void {\r\n if (Array.isArray(data)) {\r\n return;\r\n }\r\n\r\n const info = GetEnvInfo(data);\r\n if (info) {\r\n texture.width = info.width;\r\n texture.height = info.width;\r\n\r\n try {\r\n UploadEnvSpherical(texture, info);\r\n UploadEnvLevelsAsync(texture, data, info).then(\r\n () => {\r\n texture.isReady = true;\r\n texture.onLoadedObservable.notifyObservers(texture);\r\n texture.onLoadedObservable.clear();\r\n if (onLoad) {\r\n onLoad();\r\n }\r\n },\r\n (reason) => {\r\n onError?.(\"Can not upload environment levels\", reason);\r\n }\r\n );\r\n } catch (e) {\r\n onError?.(\"Can not upload environment file\", e);\r\n }\r\n } else if (onError) {\r\n onError(\"Can not parse the environment file\", null);\r\n }\r\n }\r\n\r\n /**\r\n * Uploads the 2D texture data to the WebGL texture. It has already been bound once in the callback.\r\n */\r\n public loadData(): void {\r\n throw \".env not supported in 2d.\";\r\n }\r\n}\r\n\r\n// Register the loader.\r\nEngine._TextureLoaders.push(new _ENVTextureLoader());\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport { Logger } from \"../Misc/logger\";\r\nimport type { InternalTexture } from \"../Materials/Textures/internalTexture\";\r\n\r\n/**\r\n * for description see https://www.khronos.org/opengles/sdk/tools/KTX/\r\n * for file layout see https://www.khronos.org/opengles/sdk/tools/KTX/file_format_spec/\r\n */\r\nexport class KhronosTextureContainer {\r\n private static HEADER_LEN = 12 + 13 * 4; // identifier + header elements (not including key value meta-data pairs)\r\n\r\n // load types\r\n private static COMPRESSED_2D = 0; // uses a gl.compressedTexImage2D()\r\n private static COMPRESSED_3D = 1; // uses a gl.compressedTexImage3D()\r\n private static TEX_2D = 2; // uses a gl.texImage2D()\r\n private static TEX_3D = 3; // uses a gl.texImage3D()\r\n\r\n // elements of the header\r\n /**\r\n * Gets the openGL type\r\n */\r\n public glType: number;\r\n /**\r\n * Gets the openGL type size\r\n */\r\n public glTypeSize: number;\r\n /**\r\n * Gets the openGL format\r\n */\r\n public glFormat: number;\r\n /**\r\n * Gets the openGL internal format\r\n */\r\n public glInternalFormat: number;\r\n /**\r\n * Gets the base internal format\r\n */\r\n public glBaseInternalFormat: number;\r\n /**\r\n * Gets image width in pixel\r\n */\r\n public pixelWidth: number;\r\n /**\r\n * Gets image height in pixel\r\n */\r\n public pixelHeight: number;\r\n /**\r\n * Gets image depth in pixels\r\n */\r\n public pixelDepth: number;\r\n /**\r\n * Gets the number of array elements\r\n */\r\n public numberOfArrayElements: number;\r\n /**\r\n * Gets the number of faces\r\n */\r\n public numberOfFaces: number;\r\n /**\r\n * Gets the number of mipmap levels\r\n */\r\n public numberOfMipmapLevels: number;\r\n /**\r\n * Gets the bytes of key value data\r\n */\r\n public bytesOfKeyValueData: number;\r\n /**\r\n * Gets the load type\r\n */\r\n public loadType: number;\r\n /**\r\n * If the container has been made invalid (eg. constructor failed to correctly load array buffer)\r\n */\r\n public isInvalid = false;\r\n\r\n /**\r\n * Creates a new KhronosTextureContainer\r\n * @param data contents of the KTX container file\r\n * @param facesExpected should be either 1 or 6, based whether a cube texture or or\r\n */\r\n public constructor(\r\n /** contents of the KTX container file */\r\n public data: ArrayBufferView,\r\n facesExpected: number\r\n ) {\r\n if (!KhronosTextureContainer.IsValid(data)) {\r\n this.isInvalid = true;\r\n Logger.Error(\"texture missing KTX identifier\");\r\n return;\r\n }\r\n\r\n // load the reset of the header in native 32 bit uint\r\n const dataSize = Uint32Array.BYTES_PER_ELEMENT;\r\n const headerDataView = new DataView(this.data.buffer, this.data.byteOffset + 12, 13 * dataSize);\r\n const endianness = headerDataView.getUint32(0, true);\r\n const littleEndian = endianness === 0x04030201;\r\n\r\n this.glType = headerDataView.getUint32(1 * dataSize, littleEndian); // must be 0 for compressed textures\r\n this.glTypeSize = headerDataView.getUint32(2 * dataSize, littleEndian); // must be 1 for compressed textures\r\n this.glFormat = headerDataView.getUint32(3 * dataSize, littleEndian); // must be 0 for compressed textures\r\n this.glInternalFormat = headerDataView.getUint32(4 * dataSize, littleEndian); // the value of arg passed to gl.compressedTexImage2D(,,x,,,,)\r\n this.glBaseInternalFormat = headerDataView.getUint32(5 * dataSize, littleEndian); // specify GL_RGB, GL_RGBA, GL_ALPHA, etc (un-compressed only)\r\n this.pixelWidth = headerDataView.getUint32(6 * dataSize, littleEndian); // level 0 value of arg passed to gl.compressedTexImage2D(,,,x,,,)\r\n this.pixelHeight = headerDataView.getUint32(7 * dataSize, littleEndian); // level 0 value of arg passed to gl.compressedTexImage2D(,,,,x,,)\r\n this.pixelDepth = headerDataView.getUint32(8 * dataSize, littleEndian); // level 0 value of arg passed to gl.compressedTexImage3D(,,,,,x,,)\r\n this.numberOfArrayElements = headerDataView.getUint32(9 * dataSize, littleEndian); // used for texture arrays\r\n this.numberOfFaces = headerDataView.getUint32(10 * dataSize, littleEndian); // used for cubemap textures, should either be 1 or 6\r\n this.numberOfMipmapLevels = headerDataView.getUint32(11 * dataSize, littleEndian); // number of levels; disregard possibility of 0 for compressed textures\r\n this.bytesOfKeyValueData = headerDataView.getUint32(12 * dataSize, littleEndian); // the amount of space after the header for meta-data\r\n\r\n // Make sure we have a compressed type. Not only reduces work, but probably better to let dev know they are not compressing.\r\n if (this.glType !== 0) {\r\n Logger.Error(\"only compressed formats currently supported\");\r\n this.isInvalid = true;\r\n return;\r\n } else {\r\n // value of zero is an indication to generate mipmaps @ runtime. Not usually allowed for compressed, so disregard.\r\n this.numberOfMipmapLevels = Math.max(1, this.numberOfMipmapLevels);\r\n }\r\n\r\n if (this.pixelHeight === 0 || this.pixelDepth !== 0) {\r\n Logger.Error(\"only 2D textures currently supported\");\r\n this.isInvalid = true;\r\n return;\r\n }\r\n\r\n if (this.numberOfArrayElements !== 0) {\r\n Logger.Error(\"texture arrays not currently supported\");\r\n this.isInvalid = true;\r\n return;\r\n }\r\n\r\n if (this.numberOfFaces !== facesExpected) {\r\n Logger.Error(\"number of faces expected\" + facesExpected + \", but found \" + this.numberOfFaces);\r\n this.isInvalid = true;\r\n return;\r\n }\r\n\r\n // we now have a completely validated file, so could use existence of loadType as success\r\n // would need to make this more elaborate & adjust checks above to support more than one load type\r\n this.loadType = KhronosTextureContainer.COMPRESSED_2D;\r\n }\r\n\r\n /**\r\n * Uploads KTX content to a Babylon Texture.\r\n * It is assumed that the texture has already been created & is currently bound\r\n * @internal\r\n */\r\n public uploadLevels(texture: InternalTexture, loadMipmaps: boolean): void {\r\n switch (this.loadType) {\r\n case KhronosTextureContainer.COMPRESSED_2D:\r\n this._upload2DCompressedLevels(texture, loadMipmaps);\r\n break;\r\n\r\n case KhronosTextureContainer.TEX_2D:\r\n case KhronosTextureContainer.COMPRESSED_3D:\r\n case KhronosTextureContainer.TEX_3D:\r\n }\r\n }\r\n\r\n private _upload2DCompressedLevels(texture: InternalTexture, loadMipmaps: boolean): void {\r\n // initialize width & height for level 1\r\n let dataOffset = KhronosTextureContainer.HEADER_LEN + this.bytesOfKeyValueData;\r\n let width = this.pixelWidth;\r\n let height = this.pixelHeight;\r\n\r\n const mipmapCount = loadMipmaps ? this.numberOfMipmapLevels : 1;\r\n for (let level = 0; level < mipmapCount; level++) {\r\n const imageSize = new Int32Array(this.data.buffer, this.data.byteOffset + dataOffset, 1)[0]; // size per face, since not supporting array cubemaps\r\n dataOffset += 4; //image data starts from next multiple of 4 offset. Each face refers to same imagesize field above.\r\n for (let face = 0; face < this.numberOfFaces; face++) {\r\n const byteArray = new Uint8Array(this.data.buffer, this.data.byteOffset + dataOffset, imageSize);\r\n\r\n const engine = texture.getEngine();\r\n engine._uploadCompressedDataToTextureDirectly(texture, texture.format, width, height, byteArray, face, level);\r\n\r\n dataOffset += imageSize; // add size of the image for the next face/mipmap\r\n dataOffset += 3 - ((imageSize + 3) % 4); // add padding for odd sized image\r\n }\r\n width = Math.max(1.0, width * 0.5);\r\n height = Math.max(1.0, height * 0.5);\r\n }\r\n }\r\n\r\n /**\r\n * Checks if the given data starts with a KTX file identifier.\r\n * @param data the data to check\r\n * @returns true if the data is a KTX file or false otherwise\r\n */\r\n public static IsValid(data: ArrayBufferView): boolean {\r\n if (data.byteLength >= 12) {\r\n // '«', 'K', 'T', 'X', ' ', '1', '1', '»', '\\r', '\\n', '\\x1A', '\\n'\r\n const identifier = new Uint8Array(data.buffer, data.byteOffset, 12);\r\n if (\r\n identifier[0] === 0xab &&\r\n identifier[1] === 0x4b &&\r\n identifier[2] === 0x54 &&\r\n identifier[3] === 0x58 &&\r\n identifier[4] === 0x20 &&\r\n identifier[5] === 0x31 &&\r\n identifier[6] === 0x31 &&\r\n identifier[7] === 0xbb &&\r\n identifier[8] === 0x0d &&\r\n identifier[9] === 0x0a &&\r\n identifier[10] === 0x1a &&\r\n identifier[11] === 0x0a\r\n ) {\r\n return true;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n}\r\n","import type { IDisposable } from \"../scene\";\r\n\r\n/** @ignore */\r\ninterface WorkerInfo {\r\n workerPromise: Promise;\r\n idle: boolean;\r\n timeoutId?: ReturnType;\r\n}\r\n\r\n/**\r\n * Helper class to push actions to a pool of workers.\r\n */\r\nexport class WorkerPool implements IDisposable {\r\n protected _workerInfos: Array;\r\n protected _pendingActions = new Array<(worker: Worker, onComplete: () => void) => void>();\r\n\r\n /**\r\n * Constructor\r\n * @param workers Array of workers to use for actions\r\n */\r\n constructor(workers: Array) {\r\n this._workerInfos = workers.map((worker) => ({\r\n workerPromise: Promise.resolve(worker),\r\n idle: true,\r\n }));\r\n }\r\n\r\n /**\r\n * Terminates all workers and clears any pending actions.\r\n */\r\n public dispose(): void {\r\n for (const workerInfo of this._workerInfos) {\r\n workerInfo.workerPromise.then((worker) => {\r\n worker.terminate();\r\n });\r\n }\r\n\r\n this._workerInfos.length = 0;\r\n this._pendingActions.length = 0;\r\n }\r\n\r\n /**\r\n * Pushes an action to the worker pool. If all the workers are active, the action will be\r\n * pended until a worker has completed its action.\r\n * @param action The action to perform. Call onComplete when the action is complete.\r\n */\r\n public push(action: (worker: Worker, onComplete: () => void) => void): void {\r\n if (!this._executeOnIdleWorker(action)) {\r\n this._pendingActions.push(action);\r\n }\r\n }\r\n\r\n protected _executeOnIdleWorker(action: (worker: Worker, onComplete: () => void) => void): boolean {\r\n for (const workerInfo of this._workerInfos) {\r\n if (workerInfo.idle) {\r\n this._execute(workerInfo, action);\r\n return true;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n protected _execute(workerInfo: WorkerInfo, action: (worker: Worker, onComplete: () => void) => void): void {\r\n workerInfo.idle = false;\r\n workerInfo.workerPromise.then((worker) => {\r\n action(worker, () => {\r\n const nextAction = this._pendingActions.shift();\r\n if (nextAction) {\r\n this._execute(workerInfo, nextAction);\r\n } else {\r\n workerInfo.idle = true;\r\n }\r\n });\r\n });\r\n }\r\n}\r\n\r\n/**\r\n * Options for AutoReleaseWorkerPool\r\n */\r\nexport interface AutoReleaseWorkerPoolOptions {\r\n /**\r\n * Idle time elapsed before workers are terminated.\r\n */\r\n idleTimeElapsedBeforeRelease: number;\r\n}\r\n\r\n/**\r\n * Similar to the WorkerPool class except it creates and destroys workers automatically with a maximum of `maxWorkers` workers.\r\n * Workers are terminated when it is idle for at least `idleTimeElapsedBeforeRelease` milliseconds.\r\n */\r\nexport class AutoReleaseWorkerPool extends WorkerPool {\r\n /**\r\n * Default options for the constructor.\r\n * Override to change the defaults.\r\n */\r\n public static DefaultOptions: AutoReleaseWorkerPoolOptions = {\r\n idleTimeElapsedBeforeRelease: 1000,\r\n };\r\n\r\n private readonly _maxWorkers: number;\r\n private readonly _createWorkerAsync: () => Promise;\r\n private readonly _options: AutoReleaseWorkerPoolOptions;\r\n\r\n constructor(maxWorkers: number, createWorkerAsync: () => Promise, options = AutoReleaseWorkerPool.DefaultOptions) {\r\n super([]);\r\n\r\n this._maxWorkers = maxWorkers;\r\n this._createWorkerAsync = createWorkerAsync;\r\n this._options = options;\r\n }\r\n\r\n public push(action: (worker: Worker, onComplete: () => void) => void): void {\r\n if (!this._executeOnIdleWorker(action)) {\r\n if (this._workerInfos.length < this._maxWorkers) {\r\n const workerInfo: WorkerInfo = {\r\n workerPromise: this._createWorkerAsync(),\r\n idle: false,\r\n };\r\n this._workerInfos.push(workerInfo);\r\n this._execute(workerInfo, action);\r\n } else {\r\n this._pendingActions.push(action);\r\n }\r\n }\r\n }\r\n\r\n protected _execute(workerInfo: WorkerInfo, action: (worker: Worker, onComplete: () => void) => void): void {\r\n // Reset the idle timeout.\r\n if (workerInfo.timeoutId) {\r\n clearTimeout(workerInfo.timeoutId);\r\n delete workerInfo.timeoutId;\r\n }\r\n\r\n super._execute(workerInfo, (worker, onComplete) => {\r\n action(worker, () => {\r\n onComplete();\r\n\r\n if (workerInfo.idle) {\r\n // Schedule the worker to be terminated after the elapsed time.\r\n workerInfo.timeoutId = setTimeout(() => {\r\n workerInfo.workerPromise.then((worker) => {\r\n worker.terminate();\r\n });\r\n\r\n const indexOf = this._workerInfos.indexOf(workerInfo);\r\n if (indexOf !== -1) {\r\n this._workerInfos.splice(indexOf, 1);\r\n }\r\n }, this._options.idleTimeElapsedBeforeRelease);\r\n }\r\n });\r\n });\r\n }\r\n}\r\n","export enum SourceTextureFormat {\r\n ETC1S,\r\n UASTC4x4,\r\n}\r\n\r\nexport enum TranscodeTarget {\r\n ASTC_4X4_RGBA,\r\n BC7_RGBA,\r\n BC3_RGBA,\r\n BC1_RGB,\r\n PVRTC1_4_RGBA,\r\n PVRTC1_4_RGB,\r\n ETC2_RGBA,\r\n ETC1_RGB,\r\n RGBA32,\r\n R8,\r\n RG8,\r\n}\r\n\r\nexport enum EngineFormat {\r\n COMPRESSED_RGBA_BPTC_UNORM_EXT = 0x8e8c,\r\n COMPRESSED_RGBA_ASTC_4X4_KHR = 0x93b0,\r\n COMPRESSED_RGB_S3TC_DXT1_EXT = 0x83f0,\r\n COMPRESSED_RGBA_S3TC_DXT5_EXT = 0x83f3,\r\n COMPRESSED_RGBA_PVRTC_4BPPV1_IMG = 0x8c02,\r\n COMPRESSED_RGB_PVRTC_4BPPV1_IMG = 0x8c00,\r\n COMPRESSED_RGBA8_ETC2_EAC = 0x9278,\r\n COMPRESSED_RGB8_ETC2 = 0x9274,\r\n COMPRESSED_RGB_ETC1_WEBGL = 0x8d64,\r\n RGBA8Format = 0x8058,\r\n R8Format = 0x8229,\r\n RG8Format = 0x822b,\r\n}\r\n\r\n/**\r\n * Leaf node of a decision tree\r\n * It defines the transcoding format to use to transcode the texture as well as the corresponding format to use at the engine level when creating the texture\r\n */\r\nexport interface ILeaf {\r\n /**\r\n * The format to transcode to\r\n */\r\n transcodeFormat: TranscodeTarget;\r\n\r\n /**\r\n * The format to use when creating the texture at the engine level after it has been transcoded to transcodeFormat\r\n */\r\n engineFormat: EngineFormat;\r\n\r\n /**\r\n * Whether the texture must be rounded to a multiple of 4 (should normally be the case for all compressed formats). Default: true\r\n */\r\n roundToMultiple4?: boolean;\r\n}\r\n\r\n/**\r\n * Regular node of a decision tree\r\n *\r\n * Each property (except for \"yes\" and \"no\"), if not empty, will be checked in order to determine the next node to select.\r\n * If all checks are successful, the \"yes\" node will be selected, else the \"no\" node will be selected.\r\n */\r\nexport interface INode {\r\n /**\r\n * The name of the capability to check. Can be one of the following:\r\n * astc\r\n * bptc\r\n * s3tc\r\n * pvrtc\r\n * etc2\r\n * etc1\r\n */\r\n cap?: string;\r\n\r\n /**\r\n * The name of the option to check from the options object passed to the KTX2 decode function. {@link IKTX2DecoderOptions}\r\n */\r\n option?: string;\r\n\r\n /**\r\n * Checks if alpha is present in the texture\r\n */\r\n alpha?: boolean;\r\n\r\n /**\r\n * Checks the currently selected transcoding format.\r\n */\r\n transcodeFormat?: TranscodeTarget | TranscodeTarget[];\r\n\r\n /**\r\n * Checks that the texture is a power of two\r\n */\r\n needsPowerOfTwo?: boolean;\r\n\r\n /**\r\n * The node to select if all checks are successful\r\n */\r\n yes?: INode | ILeaf;\r\n\r\n /**\r\n * The node to select if at least one check is not successful\r\n */\r\n no?: INode | ILeaf;\r\n}\r\n\r\n/**\r\n * Decision tree used to determine the transcoding format to use for a given source texture format\r\n */\r\nexport interface IDecisionTree {\r\n /**\r\n * textureFormat can be either UASTC or ETC1S\r\n */\r\n [textureFormat: string]: INode;\r\n}\r\n\r\n/**\r\n * Result of the KTX2 decode function\r\n */\r\nexport interface IDecodedData {\r\n /**\r\n * Width of the texture\r\n */\r\n width: number;\r\n\r\n /**\r\n * Height of the texture\r\n */\r\n height: number;\r\n\r\n /**\r\n * The format to use when creating the texture at the engine level\r\n * This corresponds to the engineFormat property of the leaf node of the decision tree\r\n */\r\n transcodedFormat: number;\r\n\r\n /**\r\n * List of mipmap levels.\r\n * The first element is the base level, the last element is the smallest mipmap level (if more than one mipmap level is present)\r\n */\r\n mipmaps: Array;\r\n\r\n /**\r\n * Whether the texture data is in gamma space or not\r\n */\r\n isInGammaSpace: boolean;\r\n\r\n /**\r\n * Whether the texture has an alpha channel or not\r\n */\r\n hasAlpha: boolean;\r\n\r\n /**\r\n * The name of the transcoder used to transcode the texture\r\n */\r\n transcoderName: string;\r\n\r\n /**\r\n * The errors (if any) encountered during the decoding process\r\n */\r\n errors?: string;\r\n}\r\n\r\n/**\r\n * Defines a mipmap level\r\n */\r\nexport interface IMipmap {\r\n /**\r\n * The data of the mipmap level\r\n */\r\n data: Uint8Array | null;\r\n\r\n /**\r\n * The width of the mipmap level\r\n */\r\n width: number;\r\n\r\n /**\r\n * The height of the mipmap level\r\n */\r\n height: number;\r\n}\r\n\r\n/**\r\n * The compressed texture formats supported by the browser\r\n */\r\nexport interface ICompressedFormatCapabilities {\r\n /**\r\n * Whether the browser supports ASTC\r\n */\r\n astc?: boolean;\r\n\r\n /**\r\n * Whether the browser supports BPTC\r\n */\r\n bptc?: boolean;\r\n\r\n /**\r\n * Whether the browser supports S3TC\r\n */\r\n s3tc?: boolean;\r\n\r\n /**\r\n * Whether the browser supports PVRTC\r\n */\r\n pvrtc?: boolean;\r\n\r\n /**\r\n * Whether the browser supports ETC2\r\n */\r\n etc2?: boolean;\r\n\r\n /**\r\n * Whether the browser supports ETC1\r\n */\r\n etc1?: boolean;\r\n}\r\n\r\n/**\r\n * Options passed to the KTX2 decode function\r\n */\r\nexport interface IKTX2DecoderOptions {\r\n /** use RGBA format if ASTC and BC7 are not available as transcoded format */\r\n useRGBAIfASTCBC7NotAvailableWhenUASTC?: boolean;\r\n\r\n /** force to always use (uncompressed) RGBA for transcoded format */\r\n forceRGBA?: boolean;\r\n\r\n /** force to always use (uncompressed) R8 for transcoded format */\r\n forceR8?: boolean;\r\n\r\n /** force to always use (uncompressed) RG8 for transcoded format */\r\n forceRG8?: boolean;\r\n\r\n /**\r\n * list of transcoders to bypass when looking for a suitable transcoder. The available transcoders are:\r\n * UniversalTranscoder_UASTC_ASTC\r\n * UniversalTranscoder_UASTC_BC7\r\n * UniversalTranscoder_UASTC_RGBA_UNORM\r\n * UniversalTranscoder_UASTC_RGBA_SRGB\r\n * UniversalTranscoder_UASTC_R8_UNORM\r\n * UniversalTranscoder_UASTC_RG8_UNORM\r\n * MSCTranscoder\r\n */\r\n bypassTranscoders?: string[];\r\n\r\n /**\r\n * Custom decision tree to apply after the default decision tree has selected a transcoding format.\r\n * Allows the user to override the default decision tree selection.\r\n * The decision tree can use the INode.transcodeFormat property to base its decision on the transcoding format selected by the default decision tree.\r\n */\r\n transcodeFormatDecisionTree?: IDecisionTree;\r\n}\r\n","import { Tools } from \"../Misc/tools\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport type { Nullable } from \"../types\";\r\nimport { Scene } from \"../scene\";\r\nimport { Engine } from \"../Engines/engine\";\r\nimport { EngineStore } from \"../Engines/engineStore\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { AnimationGroup } from \"../Animations/animationGroup\";\r\nimport type { AssetContainer } from \"../assetContainer\";\r\nimport type { IParticleSystem } from \"../Particles/IParticleSystem\";\r\nimport type { Skeleton } from \"../Bones/skeleton\";\r\nimport { Logger } from \"../Misc/logger\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport { SceneLoaderFlags } from \"./sceneLoaderFlags\";\r\nimport type { IFileRequest } from \"../Misc/fileRequest\";\r\nimport type { WebRequest } from \"../Misc/webRequest\";\r\nimport type { LoadFileError } from \"../Misc/fileTools\";\r\nimport { IsBase64DataUrl } from \"../Misc/fileTools\";\r\nimport type { TransformNode } from \"../Meshes/transformNode\";\r\nimport type { Geometry } from \"../Meshes/geometry\";\r\nimport type { Light } from \"../Lights/light\";\r\nimport { RuntimeError, ErrorCodes } from \"../Misc/error\";\r\n\r\n/**\r\n * Type used for the success callback of ImportMesh\r\n */\r\nexport type SceneLoaderSuccessCallback = (\r\n meshes: AbstractMesh[],\r\n particleSystems: IParticleSystem[],\r\n skeletons: Skeleton[],\r\n animationGroups: AnimationGroup[],\r\n transformNodes: TransformNode[],\r\n geometries: Geometry[],\r\n lights: Light[]\r\n) => void;\r\n\r\n/**\r\n * Interface used for the result of ImportMeshAsync\r\n */\r\nexport interface ISceneLoaderAsyncResult {\r\n /**\r\n * The array of loaded meshes\r\n */\r\n readonly meshes: AbstractMesh[];\r\n\r\n /**\r\n * The array of loaded particle systems\r\n */\r\n readonly particleSystems: IParticleSystem[];\r\n\r\n /**\r\n * The array of loaded skeletons\r\n */\r\n readonly skeletons: Skeleton[];\r\n\r\n /**\r\n * The array of loaded animation groups\r\n */\r\n readonly animationGroups: AnimationGroup[];\r\n\r\n /**\r\n * The array of loaded transform nodes\r\n */\r\n readonly transformNodes: TransformNode[];\r\n\r\n /**\r\n * The array of loaded geometries\r\n */\r\n readonly geometries: Geometry[];\r\n\r\n /**\r\n * The array of loaded lights\r\n */\r\n readonly lights: Light[];\r\n}\r\n\r\n/**\r\n * Interface used to represent data loading progression\r\n */\r\nexport interface ISceneLoaderProgressEvent {\r\n /**\r\n * Defines if data length to load can be evaluated\r\n */\r\n readonly lengthComputable: boolean;\r\n\r\n /**\r\n * Defines the loaded data length\r\n */\r\n readonly loaded: number;\r\n\r\n /**\r\n * Defines the data length to load\r\n */\r\n readonly total: number;\r\n}\r\n\r\n/**\r\n * Interface used by SceneLoader plugins to define supported file extensions\r\n */\r\nexport interface ISceneLoaderPluginExtensions {\r\n /**\r\n * Defines the list of supported extensions\r\n */\r\n [extension: string]: {\r\n isBinary: boolean;\r\n };\r\n}\r\n\r\n/**\r\n * Interface used by SceneLoader plugin factory\r\n */\r\nexport interface ISceneLoaderPluginFactory {\r\n /**\r\n * Defines the name of the factory\r\n */\r\n name: string;\r\n\r\n /**\r\n * Function called to create a new plugin\r\n * @returns the new plugin\r\n */\r\n createPlugin(): ISceneLoaderPlugin | ISceneLoaderPluginAsync;\r\n\r\n /**\r\n * The callback that returns true if the data can be directly loaded.\r\n * @param data string containing the file data\r\n * @returns if the data can be loaded directly\r\n */\r\n canDirectLoad?(data: string): boolean;\r\n}\r\n\r\n/**\r\n * Interface used to define the base of ISceneLoaderPlugin and ISceneLoaderPluginAsync\r\n */\r\nexport interface ISceneLoaderPluginBase {\r\n /**\r\n * The friendly name of this plugin.\r\n */\r\n name: string;\r\n\r\n /**\r\n * The file extensions supported by this plugin.\r\n */\r\n extensions: string | ISceneLoaderPluginExtensions;\r\n\r\n /**\r\n * The callback called when loading from a url.\r\n * @param scene scene loading this url\r\n * @param fileOrUrl file or url to load\r\n * @param onSuccess callback called when the file successfully loads\r\n * @param onProgress callback called while file is loading (if the server supports this mode)\r\n * @param useArrayBuffer defines a boolean indicating that date must be returned as ArrayBuffer\r\n * @param onError callback called when the file fails to load\r\n * @returns a file request object\r\n */\r\n loadFile?(\r\n scene: Scene,\r\n fileOrUrl: File | string | ArrayBufferView,\r\n rootUrl: string,\r\n onSuccess: (data: any, responseURL?: string) => void,\r\n onProgress?: (ev: ISceneLoaderProgressEvent) => void,\r\n useArrayBuffer?: boolean,\r\n onError?: (request?: WebRequest, exception?: LoadFileError) => void,\r\n name?: string\r\n ): Nullable;\r\n\r\n /**\r\n * The callback that returns true if the data can be directly loaded.\r\n * @param data string containing the file data\r\n * @returns if the data can be loaded directly\r\n */\r\n canDirectLoad?(data: string): boolean;\r\n\r\n /**\r\n * The callback that returns the data to pass to the plugin if the data can be directly loaded.\r\n * @param scene scene loading this data\r\n * @param data string containing the data\r\n * @returns data to pass to the plugin\r\n */\r\n directLoad?(scene: Scene, data: string): any;\r\n\r\n /**\r\n * The callback that allows custom handling of the root url based on the response url.\r\n * @param rootUrl the original root url\r\n * @param responseURL the response url if available\r\n * @returns the new root url\r\n */\r\n rewriteRootURL?(rootUrl: string, responseURL?: string): string;\r\n}\r\n\r\n/**\r\n * Interface used to define a SceneLoader plugin\r\n */\r\nexport interface ISceneLoaderPlugin extends ISceneLoaderPluginBase {\r\n /**\r\n * Import meshes into a scene.\r\n * @param meshesNames An array of mesh names, a single mesh name, or empty string for all meshes that filter what meshes are imported\r\n * @param scene The scene to import into\r\n * @param data The data to import\r\n * @param rootUrl The root url for scene and resources\r\n * @param meshes The meshes array to import into\r\n * @param particleSystems The particle systems array to import into\r\n * @param skeletons The skeletons array to import into\r\n * @param onError The callback when import fails\r\n * @returns True if successful or false otherwise\r\n */\r\n importMesh(\r\n meshesNames: any,\r\n scene: Scene,\r\n data: any,\r\n rootUrl: string,\r\n meshes: AbstractMesh[],\r\n particleSystems: IParticleSystem[],\r\n skeletons: Skeleton[],\r\n onError?: (message: string, exception?: any) => void\r\n ): boolean;\r\n\r\n /**\r\n * Load into a scene.\r\n * @param scene The scene to load into\r\n * @param data The data to import\r\n * @param rootUrl The root url for scene and resources\r\n * @param onError The callback when import fails\r\n * @returns True if successful or false otherwise\r\n */\r\n load(scene: Scene, data: any, rootUrl: string, onError?: (message: string, exception?: any) => void): boolean;\r\n\r\n /**\r\n * Load into an asset container.\r\n * @param scene The scene to load into\r\n * @param data The data to import\r\n * @param rootUrl The root url for scene and resources\r\n * @param onError The callback when import fails\r\n * @returns The loaded asset container\r\n */\r\n loadAssetContainer(scene: Scene, data: any, rootUrl: string, onError?: (message: string, exception?: any) => void): AssetContainer;\r\n}\r\n\r\n/**\r\n * Interface used to define an async SceneLoader plugin\r\n */\r\nexport interface ISceneLoaderPluginAsync extends ISceneLoaderPluginBase {\r\n /**\r\n * Import meshes into a scene.\r\n * @param meshesNames An array of mesh names, a single mesh name, or empty string for all meshes that filter what meshes are imported\r\n * @param scene The scene to import into\r\n * @param data The data to import\r\n * @param rootUrl The root url for scene and resources\r\n * @param onProgress The callback when the load progresses\r\n * @param fileName Defines the name of the file to load\r\n * @returns The loaded objects (e.g. meshes, particle systems, skeletons, animation groups, etc.)\r\n */\r\n importMeshAsync(\r\n meshesNames: any,\r\n scene: Scene,\r\n data: any,\r\n rootUrl: string,\r\n onProgress?: (event: ISceneLoaderProgressEvent) => void,\r\n fileName?: string\r\n ): Promise;\r\n\r\n /**\r\n * Load into a scene.\r\n * @param scene The scene to load into\r\n * @param data The data to import\r\n * @param rootUrl The root url for scene and resources\r\n * @param onProgress The callback when the load progresses\r\n * @param fileName Defines the name of the file to load\r\n * @returns Nothing\r\n */\r\n loadAsync(scene: Scene, data: any, rootUrl: string, onProgress?: (event: ISceneLoaderProgressEvent) => void, fileName?: string): Promise;\r\n\r\n /**\r\n * Load into an asset container.\r\n * @param scene The scene to load into\r\n * @param data The data to import\r\n * @param rootUrl The root url for scene and resources\r\n * @param onProgress The callback when the load progresses\r\n * @param fileName Defines the name of the file to load\r\n * @returns The loaded asset container\r\n */\r\n loadAssetContainerAsync(scene: Scene, data: any, rootUrl: string, onProgress?: (event: ISceneLoaderProgressEvent) => void, fileName?: string): Promise;\r\n}\r\n\r\n/**\r\n * Mode that determines how to handle old animation groups before loading new ones.\r\n */\r\nexport enum SceneLoaderAnimationGroupLoadingMode {\r\n /**\r\n * Reset all old animations to initial state then dispose them.\r\n */\r\n Clean = 0,\r\n\r\n /**\r\n * Stop all old animations.\r\n */\r\n Stop = 1,\r\n\r\n /**\r\n * Restart old animations from first frame.\r\n */\r\n Sync = 2,\r\n\r\n /**\r\n * Old animations remains untouched.\r\n */\r\n NoSync = 3,\r\n}\r\n\r\n/**\r\n * Defines a plugin registered by the SceneLoader\r\n */\r\ninterface IRegisteredPlugin {\r\n /**\r\n * Defines the plugin to use\r\n */\r\n plugin: ISceneLoaderPlugin | ISceneLoaderPluginAsync | ISceneLoaderPluginFactory;\r\n /**\r\n * Defines if the plugin supports binary data\r\n */\r\n isBinary: boolean;\r\n}\r\n\r\n/**\r\n * Defines file information\r\n */\r\ninterface IFileInfo {\r\n /**\r\n * Gets the file url\r\n */\r\n url: string;\r\n /**\r\n * Gets the root url\r\n */\r\n rootUrl: string;\r\n /**\r\n * Gets filename\r\n */\r\n name: string;\r\n /**\r\n * Gets the file\r\n */\r\n file: Nullable;\r\n\r\n /**\r\n * Gets raw binary data.\r\n */\r\n rawData: Nullable;\r\n}\r\n\r\n/**\r\n * Class used to load scene from various file formats using registered plugins\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/importers/loadingFileTypes\r\n */\r\nexport class SceneLoader {\r\n /**\r\n * No logging while loading\r\n */\r\n public static readonly NO_LOGGING = Constants.SCENELOADER_NO_LOGGING;\r\n\r\n /**\r\n * Minimal logging while loading\r\n */\r\n public static readonly MINIMAL_LOGGING = Constants.SCENELOADER_MINIMAL_LOGGING;\r\n\r\n /**\r\n * Summary logging while loading\r\n */\r\n public static readonly SUMMARY_LOGGING = Constants.SCENELOADER_SUMMARY_LOGGING;\r\n\r\n /**\r\n * Detailed logging while loading\r\n */\r\n public static readonly DETAILED_LOGGING = Constants.SCENELOADER_DETAILED_LOGGING;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if entire scene must be loaded even if scene contains incremental data\r\n */\r\n public static get ForceFullSceneLoadingForIncremental() {\r\n return SceneLoaderFlags.ForceFullSceneLoadingForIncremental;\r\n }\r\n\r\n public static set ForceFullSceneLoadingForIncremental(value: boolean) {\r\n SceneLoaderFlags.ForceFullSceneLoadingForIncremental = value;\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating if loading screen must be displayed while loading a scene\r\n */\r\n public static get ShowLoadingScreen(): boolean {\r\n return SceneLoaderFlags.ShowLoadingScreen;\r\n }\r\n\r\n public static set ShowLoadingScreen(value: boolean) {\r\n SceneLoaderFlags.ShowLoadingScreen = value;\r\n }\r\n\r\n /**\r\n * Defines the current logging level (while loading the scene)\r\n * @ignorenaming\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public static get loggingLevel(): number {\r\n return SceneLoaderFlags.loggingLevel;\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public static set loggingLevel(value: number) {\r\n SceneLoaderFlags.loggingLevel = value;\r\n }\r\n\r\n /**\r\n * Gets or set a boolean indicating if matrix weights must be cleaned upon loading\r\n */\r\n public static get CleanBoneMatrixWeights(): boolean {\r\n return SceneLoaderFlags.CleanBoneMatrixWeights;\r\n }\r\n\r\n public static set CleanBoneMatrixWeights(value: boolean) {\r\n SceneLoaderFlags.CleanBoneMatrixWeights = value;\r\n }\r\n\r\n // Members\r\n\r\n /**\r\n * Event raised when a plugin is used to load a scene\r\n */\r\n public static OnPluginActivatedObservable = new Observable();\r\n\r\n private static _RegisteredPlugins: { [extension: string]: IRegisteredPlugin } = {};\r\n\r\n private static _ShowingLoadingScreen = false;\r\n\r\n /**\r\n * Gets the default plugin (used to load Babylon files)\r\n * @returns the .babylon plugin\r\n */\r\n public static GetDefaultPlugin(): IRegisteredPlugin {\r\n return SceneLoader._RegisteredPlugins[\".babylon\"];\r\n }\r\n\r\n private static _GetPluginForExtension(extension: string): IRegisteredPlugin {\r\n const registeredPlugin = SceneLoader._RegisteredPlugins[extension];\r\n if (registeredPlugin) {\r\n return registeredPlugin;\r\n }\r\n Logger.Warn(\r\n \"Unable to find a plugin to load \" +\r\n extension +\r\n \" files. Trying to use .babylon default plugin. To load from a specific filetype (eg. gltf) see: https://doc.babylonjs.com/features/featuresDeepDive/importers/loadingFileTypes\"\r\n );\r\n return SceneLoader.GetDefaultPlugin();\r\n }\r\n\r\n private static _GetPluginForDirectLoad(data: string): IRegisteredPlugin {\r\n for (const extension in SceneLoader._RegisteredPlugins) {\r\n const plugin = SceneLoader._RegisteredPlugins[extension].plugin;\r\n\r\n if (plugin.canDirectLoad && plugin.canDirectLoad(data)) {\r\n return SceneLoader._RegisteredPlugins[extension];\r\n }\r\n }\r\n\r\n return SceneLoader.GetDefaultPlugin();\r\n }\r\n\r\n private static _GetPluginForFilename(sceneFilename: string): IRegisteredPlugin {\r\n const queryStringPosition = sceneFilename.indexOf(\"?\");\r\n\r\n if (queryStringPosition !== -1) {\r\n sceneFilename = sceneFilename.substring(0, queryStringPosition);\r\n }\r\n\r\n const dotPosition = sceneFilename.lastIndexOf(\".\");\r\n\r\n const extension = sceneFilename.substring(dotPosition, sceneFilename.length).toLowerCase();\r\n return SceneLoader._GetPluginForExtension(extension);\r\n }\r\n\r\n private static _GetDirectLoad(sceneFilename: string): Nullable {\r\n if (sceneFilename.substr(0, 5) === \"data:\") {\r\n return sceneFilename.substr(5);\r\n }\r\n\r\n return null;\r\n }\r\n\r\n private static _FormatErrorMessage(fileInfo: IFileInfo, message?: string, exception?: any): string {\r\n const fromLoad = fileInfo.rawData ? \"binary data\" : fileInfo.url;\r\n let errorMessage = \"Unable to load from \" + fromLoad;\r\n\r\n if (message) {\r\n errorMessage += `: ${message}`;\r\n } else if (exception) {\r\n errorMessage += `: ${exception}`;\r\n }\r\n\r\n return errorMessage;\r\n }\r\n\r\n private static _LoadData(\r\n fileInfo: IFileInfo,\r\n scene: Scene,\r\n onSuccess: (plugin: ISceneLoaderPlugin | ISceneLoaderPluginAsync, data: any, responseURL?: string) => void,\r\n onProgress: ((event: ISceneLoaderProgressEvent) => void) | undefined,\r\n onError: (message?: string, exception?: any) => void,\r\n onDispose: () => void,\r\n pluginExtension: Nullable,\r\n name: string\r\n ): Nullable {\r\n const directLoad = SceneLoader._GetDirectLoad(fileInfo.url);\r\n\r\n if (fileInfo.rawData && !pluginExtension) {\r\n throw \"When using ArrayBufferView to load data the file extension must be provided.\";\r\n }\r\n\r\n const registeredPlugin = pluginExtension\r\n ? SceneLoader._GetPluginForExtension(pluginExtension)\r\n : directLoad\r\n ? SceneLoader._GetPluginForDirectLoad(fileInfo.url)\r\n : SceneLoader._GetPluginForFilename(fileInfo.url);\r\n\r\n if (fileInfo.rawData && !registeredPlugin.isBinary) {\r\n throw \"Loading from ArrayBufferView can not be used with plugins that don't support binary loading.\";\r\n }\r\n\r\n let plugin: ISceneLoaderPlugin | ISceneLoaderPluginAsync;\r\n\r\n if ((registeredPlugin.plugin as ISceneLoaderPluginFactory).createPlugin !== undefined) {\r\n plugin = (registeredPlugin.plugin as ISceneLoaderPluginFactory).createPlugin();\r\n } else {\r\n plugin = registeredPlugin.plugin;\r\n }\r\n\r\n if (!plugin) {\r\n throw \"The loader plugin corresponding to the file type you are trying to load has not been found. If using es6, please import the plugin you wish to use before.\";\r\n }\r\n\r\n SceneLoader.OnPluginActivatedObservable.notifyObservers(plugin);\r\n\r\n // Check if we have a direct load url. If the plugin is registered to handle\r\n // it or it's not a base64 data url, then pass it through the direct load path.\r\n if (directLoad && ((plugin.canDirectLoad && plugin.canDirectLoad(fileInfo.url)) || !IsBase64DataUrl(fileInfo.url))) {\r\n if (plugin.directLoad) {\r\n const result = plugin.directLoad(scene, directLoad);\r\n if (result.then) {\r\n result\r\n .then((data: any) => {\r\n onSuccess(plugin, data);\r\n })\r\n .catch((error: any) => {\r\n onError(\"Error in directLoad of _loadData: \" + error, error);\r\n });\r\n } else {\r\n onSuccess(plugin, result);\r\n }\r\n } else {\r\n onSuccess(plugin, directLoad);\r\n }\r\n return plugin;\r\n }\r\n\r\n const useArrayBuffer = registeredPlugin.isBinary;\r\n\r\n const dataCallback = (data: any, responseURL?: string) => {\r\n if (scene.isDisposed) {\r\n onError(\"Scene has been disposed\");\r\n return;\r\n }\r\n\r\n onSuccess(plugin, data, responseURL);\r\n };\r\n\r\n let request: Nullable = null;\r\n let pluginDisposed = false;\r\n const onDisposeObservable = (plugin as any).onDisposeObservable as Observable;\r\n if (onDisposeObservable) {\r\n onDisposeObservable.add(() => {\r\n pluginDisposed = true;\r\n\r\n if (request) {\r\n request.abort();\r\n request = null;\r\n }\r\n\r\n onDispose();\r\n });\r\n }\r\n\r\n const manifestChecked = () => {\r\n if (pluginDisposed) {\r\n return;\r\n }\r\n\r\n const errorCallback = (request?: WebRequest, exception?: LoadFileError) => {\r\n onError(request?.statusText, exception);\r\n };\r\n\r\n if (!plugin.loadFile && fileInfo.rawData) {\r\n throw \"Plugin does not support loading ArrayBufferView.\";\r\n }\r\n\r\n request = plugin.loadFile\r\n ? plugin.loadFile(scene, fileInfo.rawData || fileInfo.file || fileInfo.url, fileInfo.rootUrl, dataCallback, onProgress, useArrayBuffer, errorCallback, name)\r\n : scene._loadFile(fileInfo.file || fileInfo.url, dataCallback, onProgress, true, useArrayBuffer, errorCallback);\r\n };\r\n\r\n const engine = scene.getEngine();\r\n let canUseOfflineSupport = engine.enableOfflineSupport;\r\n if (canUseOfflineSupport) {\r\n // Also check for exceptions\r\n let exceptionFound = false;\r\n for (const regex of scene.disableOfflineSupportExceptionRules) {\r\n if (regex.test(fileInfo.url)) {\r\n exceptionFound = true;\r\n break;\r\n }\r\n }\r\n\r\n canUseOfflineSupport = !exceptionFound;\r\n }\r\n\r\n if (canUseOfflineSupport && Engine.OfflineProviderFactory) {\r\n // Checking if a manifest file has been set for this scene and if offline mode has been requested\r\n scene.offlineProvider = Engine.OfflineProviderFactory(fileInfo.url, manifestChecked, engine.disableManifestCheck);\r\n } else {\r\n manifestChecked();\r\n }\r\n\r\n return plugin;\r\n }\r\n\r\n private static _GetFileInfo(rootUrl: string, sceneFilename: string | File | ArrayBufferView): Nullable {\r\n let url: string;\r\n let name: string;\r\n let file: Nullable = null;\r\n let rawData: Nullable = null;\r\n\r\n if (!sceneFilename) {\r\n url = rootUrl;\r\n name = Tools.GetFilename(rootUrl);\r\n rootUrl = Tools.GetFolderPath(rootUrl);\r\n } else if ((sceneFilename as File).name) {\r\n const sceneFile = sceneFilename as File;\r\n url = `file:${sceneFile.name}`;\r\n name = sceneFile.name;\r\n file = sceneFile;\r\n } else if (ArrayBuffer.isView(sceneFilename)) {\r\n url = \"\";\r\n name = \"arrayBuffer\";\r\n rawData = sceneFilename as ArrayBufferView;\r\n } else if (typeof sceneFilename === \"string\" && sceneFilename.startsWith(\"data:\")) {\r\n url = sceneFilename;\r\n name = \"\";\r\n } else {\r\n const filename = sceneFilename as string;\r\n if (filename.substr(0, 1) === \"/\") {\r\n Tools.Error(\"Wrong sceneFilename parameter\");\r\n return null;\r\n }\r\n\r\n url = rootUrl + filename;\r\n name = filename;\r\n }\r\n\r\n return {\r\n url: url,\r\n rootUrl: rootUrl,\r\n name: name,\r\n file: file,\r\n rawData,\r\n };\r\n }\r\n\r\n // Public functions\r\n\r\n /**\r\n * Gets a plugin that can load the given extension\r\n * @param extension defines the extension to load\r\n * @returns a plugin or null if none works\r\n */\r\n public static GetPluginForExtension(extension: string): ISceneLoaderPlugin | ISceneLoaderPluginAsync | ISceneLoaderPluginFactory {\r\n return SceneLoader._GetPluginForExtension(extension).plugin;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating that the given extension can be loaded\r\n * @param extension defines the extension to load\r\n * @returns true if the extension is supported\r\n */\r\n public static IsPluginForExtensionAvailable(extension: string): boolean {\r\n return !!SceneLoader._RegisteredPlugins[extension];\r\n }\r\n\r\n /**\r\n * Adds a new plugin to the list of registered plugins\r\n * @param plugin defines the plugin to add\r\n */\r\n public static RegisterPlugin(plugin: ISceneLoaderPlugin | ISceneLoaderPluginAsync): void {\r\n if (typeof plugin.extensions === \"string\") {\r\n const extension = plugin.extensions;\r\n SceneLoader._RegisteredPlugins[extension.toLowerCase()] = {\r\n plugin: plugin,\r\n isBinary: false,\r\n };\r\n } else {\r\n const extensions = plugin.extensions;\r\n Object.keys(extensions).forEach((extension) => {\r\n SceneLoader._RegisteredPlugins[extension.toLowerCase()] = {\r\n plugin: plugin,\r\n isBinary: extensions[extension].isBinary,\r\n };\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Import meshes into a scene\r\n * @param meshNames an array of mesh names, a single mesh name, or empty string for all meshes that filter what meshes are imported\r\n * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb)\r\n * @param sceneFilename a string that defines the name of the scene file or starts with \"data:\" following by the stringified version of the scene or a File object (default: empty string)\r\n * @param scene the instance of BABYLON.Scene to append to\r\n * @param onSuccess a callback with a list of imported meshes, particleSystems, skeletons, and animationGroups when import succeeds\r\n * @param onProgress a callback with a progress event for each file being loaded\r\n * @param onError a callback with the scene, a message, and possibly an exception when import fails\r\n * @param pluginExtension the extension used to determine the plugin\r\n * @returns The loaded plugin\r\n */\r\n public static ImportMesh(\r\n meshNames: any,\r\n rootUrl: string,\r\n sceneFilename: string | File | ArrayBufferView = \"\",\r\n scene: Nullable = EngineStore.LastCreatedScene,\r\n onSuccess: Nullable = null,\r\n onProgress: Nullable<(event: ISceneLoaderProgressEvent) => void> = null,\r\n onError: Nullable<(scene: Scene, message: string, exception?: any) => void> = null,\r\n pluginExtension: Nullable = null,\r\n name: string = \"\"\r\n ): Nullable {\r\n if (!scene) {\r\n Logger.Error(\"No scene available to import mesh to\");\r\n return null;\r\n }\r\n\r\n const fileInfo = SceneLoader._GetFileInfo(rootUrl, sceneFilename);\r\n if (!fileInfo) {\r\n return null;\r\n }\r\n\r\n const loadingToken = {};\r\n scene.addPendingData(loadingToken);\r\n\r\n const disposeHandler = () => {\r\n scene.removePendingData(loadingToken);\r\n };\r\n\r\n const errorHandler = (message?: string, exception?: any) => {\r\n const errorMessage = SceneLoader._FormatErrorMessage(fileInfo, message, exception);\r\n\r\n if (onError) {\r\n onError(scene, errorMessage, new RuntimeError(errorMessage, ErrorCodes.SceneLoaderError, exception));\r\n } else {\r\n Logger.Error(errorMessage);\r\n // should the exception be thrown?\r\n }\r\n\r\n disposeHandler();\r\n };\r\n\r\n const progressHandler = onProgress\r\n ? (event: ISceneLoaderProgressEvent) => {\r\n try {\r\n onProgress(event);\r\n } catch (e) {\r\n errorHandler(\"Error in onProgress callback: \" + e, e);\r\n }\r\n }\r\n : undefined;\r\n\r\n const successHandler: SceneLoaderSuccessCallback = (meshes, particleSystems, skeletons, animationGroups, transformNodes, geometries, lights) => {\r\n scene.importedMeshesFiles.push(fileInfo.url);\r\n\r\n if (onSuccess) {\r\n try {\r\n onSuccess(meshes, particleSystems, skeletons, animationGroups, transformNodes, geometries, lights);\r\n } catch (e) {\r\n errorHandler(\"Error in onSuccess callback: \" + e, e);\r\n }\r\n }\r\n\r\n scene.removePendingData(loadingToken);\r\n };\r\n\r\n return SceneLoader._LoadData(\r\n fileInfo,\r\n scene,\r\n (plugin, data, responseURL) => {\r\n if (plugin.rewriteRootURL) {\r\n fileInfo.rootUrl = plugin.rewriteRootURL(fileInfo.rootUrl, responseURL);\r\n }\r\n\r\n if ((plugin).importMesh) {\r\n const syncedPlugin = plugin;\r\n const meshes = new Array();\r\n const particleSystems = new Array();\r\n const skeletons = new Array();\r\n\r\n if (!syncedPlugin.importMesh(meshNames, scene, data, fileInfo.rootUrl, meshes, particleSystems, skeletons, errorHandler)) {\r\n return;\r\n }\r\n\r\n scene.loadingPluginName = plugin.name;\r\n successHandler(meshes, particleSystems, skeletons, [], [], [], []);\r\n } else {\r\n const asyncedPlugin = plugin;\r\n asyncedPlugin\r\n .importMeshAsync(meshNames, scene, data, fileInfo.rootUrl, progressHandler, fileInfo.name)\r\n .then((result) => {\r\n scene.loadingPluginName = plugin.name;\r\n successHandler(\r\n result.meshes,\r\n result.particleSystems,\r\n result.skeletons,\r\n result.animationGroups,\r\n result.transformNodes,\r\n result.geometries,\r\n result.lights\r\n );\r\n })\r\n .catch((error) => {\r\n errorHandler(error.message, error);\r\n });\r\n }\r\n },\r\n progressHandler,\r\n errorHandler,\r\n disposeHandler,\r\n pluginExtension,\r\n name\r\n );\r\n }\r\n\r\n /**\r\n * Import meshes into a scene\r\n * @param meshNames an array of mesh names, a single mesh name, or empty string for all meshes that filter what meshes are imported\r\n * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb)\r\n * @param sceneFilename a string that defines the name of the scene file or starts with \"data:\" following by the stringified version of the scene or a File object (default: empty string)\r\n * @param scene the instance of BABYLON.Scene to append to\r\n * @param onProgress a callback with a progress event for each file being loaded\r\n * @param pluginExtension the extension used to determine the plugin\r\n * @returns The loaded list of imported meshes, particle systems, skeletons, and animation groups\r\n */\r\n public static ImportMeshAsync(\r\n meshNames: any,\r\n rootUrl: string,\r\n sceneFilename: string | File | ArrayBufferView = \"\",\r\n scene: Nullable = EngineStore.LastCreatedScene,\r\n onProgress: Nullable<(event: ISceneLoaderProgressEvent) => void> = null,\r\n pluginExtension: Nullable = null,\r\n name: string = \"\"\r\n ): Promise {\r\n return new Promise((resolve, reject) => {\r\n SceneLoader.ImportMesh(\r\n meshNames,\r\n rootUrl,\r\n sceneFilename,\r\n scene,\r\n (meshes, particleSystems, skeletons, animationGroups, transformNodes, geometries, lights) => {\r\n resolve({\r\n meshes: meshes,\r\n particleSystems: particleSystems,\r\n skeletons: skeletons,\r\n animationGroups: animationGroups,\r\n transformNodes: transformNodes,\r\n geometries: geometries,\r\n lights: lights,\r\n });\r\n },\r\n onProgress,\r\n (scene, message, exception) => {\r\n reject(exception || new Error(message));\r\n },\r\n pluginExtension,\r\n name\r\n );\r\n });\r\n }\r\n\r\n /**\r\n * Load a scene\r\n * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb)\r\n * @param sceneFilename a string that defines the name of the scene file or starts with \"data:\" following by the stringified version of the scene or a File object (default: empty string)\r\n * @param engine is the instance of BABYLON.Engine to use to create the scene\r\n * @param onSuccess a callback with the scene when import succeeds\r\n * @param onProgress a callback with a progress event for each file being loaded\r\n * @param onError a callback with the scene, a message, and possibly an exception when import fails\r\n * @param pluginExtension the extension used to determine the plugin\r\n * @returns The loaded plugin\r\n */\r\n public static Load(\r\n rootUrl: string,\r\n sceneFilename: string | File | ArrayBufferView = \"\",\r\n engine: Nullable = EngineStore.LastCreatedEngine,\r\n onSuccess: Nullable<(scene: Scene) => void> = null,\r\n onProgress: Nullable<(event: ISceneLoaderProgressEvent) => void> = null,\r\n onError: Nullable<(scene: Scene, message: string, exception?: any) => void> = null,\r\n pluginExtension: Nullable = null,\r\n name: string = \"\"\r\n ): Nullable {\r\n if (!engine) {\r\n Tools.Error(\"No engine available\");\r\n return null;\r\n }\r\n\r\n return SceneLoader.Append(rootUrl, sceneFilename, new Scene(engine), onSuccess, onProgress, onError, pluginExtension, name);\r\n }\r\n\r\n /**\r\n * Load a scene\r\n * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb)\r\n * @param sceneFilename a string that defines the name of the scene file or starts with \"data:\" following by the stringified version of the scene or a File object (default: empty string)\r\n * @param engine is the instance of BABYLON.Engine to use to create the scene\r\n * @param onProgress a callback with a progress event for each file being loaded\r\n * @param pluginExtension the extension used to determine the plugin\r\n * @returns The loaded scene\r\n */\r\n public static LoadAsync(\r\n rootUrl: string,\r\n sceneFilename: string | File | ArrayBufferView = \"\",\r\n engine: Nullable = EngineStore.LastCreatedEngine,\r\n onProgress: Nullable<(event: ISceneLoaderProgressEvent) => void> = null,\r\n pluginExtension: Nullable = null,\r\n name: string = \"\"\r\n ): Promise {\r\n return new Promise((resolve, reject) => {\r\n SceneLoader.Load(\r\n rootUrl,\r\n sceneFilename,\r\n engine,\r\n (scene) => {\r\n resolve(scene);\r\n },\r\n onProgress,\r\n (scene, message, exception) => {\r\n reject(exception || new Error(message));\r\n },\r\n pluginExtension,\r\n name\r\n );\r\n });\r\n }\r\n\r\n /**\r\n * Append a scene\r\n * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb)\r\n * @param sceneFilename a string that defines the name of the scene file or starts with \"data:\" following by the stringified version of the scene or a File object (default: empty string)\r\n * @param scene is the instance of BABYLON.Scene to append to\r\n * @param onSuccess a callback with the scene when import succeeds\r\n * @param onProgress a callback with a progress event for each file being loaded\r\n * @param onError a callback with the scene, a message, and possibly an exception when import fails\r\n * @param pluginExtension the extension used to determine the plugin\r\n * @returns The loaded plugin\r\n */\r\n public static Append(\r\n rootUrl: string,\r\n sceneFilename: string | File | ArrayBufferView = \"\",\r\n scene: Nullable = EngineStore.LastCreatedScene,\r\n onSuccess: Nullable<(scene: Scene) => void> = null,\r\n onProgress: Nullable<(event: ISceneLoaderProgressEvent) => void> = null,\r\n onError: Nullable<(scene: Scene, message: string, exception?: any) => void> = null,\r\n pluginExtension: Nullable = null,\r\n name: string = \"\"\r\n ): Nullable {\r\n if (!scene) {\r\n Logger.Error(\"No scene available to append to\");\r\n return null;\r\n }\r\n\r\n const fileInfo = SceneLoader._GetFileInfo(rootUrl, sceneFilename);\r\n if (!fileInfo) {\r\n return null;\r\n }\r\n\r\n const loadingToken = {};\r\n scene.addPendingData(loadingToken);\r\n\r\n const disposeHandler = () => {\r\n scene.removePendingData(loadingToken);\r\n };\r\n\r\n if (SceneLoader.ShowLoadingScreen && !this._ShowingLoadingScreen) {\r\n this._ShowingLoadingScreen = true;\r\n scene.getEngine().displayLoadingUI();\r\n scene.executeWhenReady(() => {\r\n scene.getEngine().hideLoadingUI();\r\n this._ShowingLoadingScreen = false;\r\n });\r\n }\r\n\r\n const errorHandler = (message?: string, exception?: any) => {\r\n const errorMessage = SceneLoader._FormatErrorMessage(fileInfo, message, exception);\r\n\r\n if (onError) {\r\n onError(scene, errorMessage, new RuntimeError(errorMessage, ErrorCodes.SceneLoaderError, exception));\r\n } else {\r\n Logger.Error(errorMessage);\r\n // should the exception be thrown?\r\n }\r\n\r\n disposeHandler();\r\n };\r\n\r\n const progressHandler = onProgress\r\n ? (event: ISceneLoaderProgressEvent) => {\r\n try {\r\n onProgress(event);\r\n } catch (e) {\r\n errorHandler(\"Error in onProgress callback\", e);\r\n }\r\n }\r\n : undefined;\r\n\r\n const successHandler = () => {\r\n if (onSuccess) {\r\n try {\r\n onSuccess(scene);\r\n } catch (e) {\r\n errorHandler(\"Error in onSuccess callback\", e);\r\n }\r\n }\r\n\r\n scene.removePendingData(loadingToken);\r\n };\r\n\r\n return SceneLoader._LoadData(\r\n fileInfo,\r\n scene,\r\n (plugin, data) => {\r\n if ((plugin).load) {\r\n const syncedPlugin = plugin;\r\n if (!syncedPlugin.load(scene, data, fileInfo.rootUrl, errorHandler)) {\r\n return;\r\n }\r\n\r\n scene.loadingPluginName = plugin.name;\r\n successHandler();\r\n } else {\r\n const asyncedPlugin = plugin;\r\n asyncedPlugin\r\n .loadAsync(scene, data, fileInfo.rootUrl, progressHandler, fileInfo.name)\r\n .then(() => {\r\n scene.loadingPluginName = plugin.name;\r\n successHandler();\r\n })\r\n .catch((error) => {\r\n errorHandler(error.message, error);\r\n });\r\n }\r\n },\r\n progressHandler,\r\n errorHandler,\r\n disposeHandler,\r\n pluginExtension,\r\n name\r\n );\r\n }\r\n\r\n /**\r\n * Append a scene\r\n * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb)\r\n * @param sceneFilename a string that defines the name of the scene file or starts with \"data:\" following by the stringified version of the scene or a File object (default: empty string)\r\n * @param scene is the instance of BABYLON.Scene to append to\r\n * @param onProgress a callback with a progress event for each file being loaded\r\n * @param pluginExtension the extension used to determine the plugin\r\n * @returns The given scene\r\n */\r\n public static AppendAsync(\r\n rootUrl: string,\r\n sceneFilename: string | File | ArrayBufferView = \"\",\r\n scene: Nullable = EngineStore.LastCreatedScene,\r\n onProgress: Nullable<(event: ISceneLoaderProgressEvent) => void> = null,\r\n pluginExtension: Nullable = null,\r\n name: string = \"\"\r\n ): Promise {\r\n return new Promise((resolve, reject) => {\r\n SceneLoader.Append(\r\n rootUrl,\r\n sceneFilename,\r\n scene,\r\n (scene) => {\r\n resolve(scene);\r\n },\r\n onProgress,\r\n (scene, message, exception) => {\r\n reject(exception || new Error(message));\r\n },\r\n pluginExtension,\r\n name\r\n );\r\n });\r\n }\r\n\r\n /**\r\n * Load a scene into an asset container\r\n * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb)\r\n * @param sceneFilename a string that defines the name of the scene file or starts with \"data:\" following by the stringified version of the scene or a File object (default: empty string)\r\n * @param scene is the instance of BABYLON.Scene to append to (default: last created scene)\r\n * @param onSuccess a callback with the scene when import succeeds\r\n * @param onProgress a callback with a progress event for each file being loaded\r\n * @param onError a callback with the scene, a message, and possibly an exception when import fails\r\n * @param pluginExtension the extension used to determine the plugin\r\n * @returns The loaded plugin\r\n */\r\n public static LoadAssetContainer(\r\n rootUrl: string,\r\n sceneFilename: string | File | ArrayBufferView = \"\",\r\n scene: Nullable = EngineStore.LastCreatedScene,\r\n onSuccess: Nullable<(assets: AssetContainer) => void> = null,\r\n onProgress: Nullable<(event: ISceneLoaderProgressEvent) => void> = null,\r\n onError: Nullable<(scene: Scene, message: string, exception?: any) => void> = null,\r\n pluginExtension: Nullable = null,\r\n name: string = \"\"\r\n ): Nullable {\r\n if (!scene) {\r\n Logger.Error(\"No scene available to load asset container to\");\r\n return null;\r\n }\r\n\r\n const fileInfo = SceneLoader._GetFileInfo(rootUrl, sceneFilename);\r\n if (!fileInfo) {\r\n return null;\r\n }\r\n\r\n const loadingToken = {};\r\n scene.addPendingData(loadingToken);\r\n\r\n const disposeHandler = () => {\r\n scene.removePendingData(loadingToken);\r\n };\r\n\r\n const errorHandler = (message?: string, exception?: any) => {\r\n const errorMessage = SceneLoader._FormatErrorMessage(fileInfo, message, exception);\r\n\r\n if (onError) {\r\n onError(scene, errorMessage, new RuntimeError(errorMessage, ErrorCodes.SceneLoaderError, exception));\r\n } else {\r\n Logger.Error(errorMessage);\r\n // should the exception be thrown?\r\n }\r\n\r\n disposeHandler();\r\n };\r\n\r\n const progressHandler = onProgress\r\n ? (event: ISceneLoaderProgressEvent) => {\r\n try {\r\n onProgress(event);\r\n } catch (e) {\r\n errorHandler(\"Error in onProgress callback\", e);\r\n }\r\n }\r\n : undefined;\r\n\r\n const successHandler = (assets: AssetContainer) => {\r\n if (onSuccess) {\r\n try {\r\n onSuccess(assets);\r\n } catch (e) {\r\n errorHandler(\"Error in onSuccess callback\", e);\r\n }\r\n }\r\n\r\n scene.removePendingData(loadingToken);\r\n };\r\n\r\n return SceneLoader._LoadData(\r\n fileInfo,\r\n scene,\r\n (plugin, data) => {\r\n if ((plugin).loadAssetContainer) {\r\n const syncedPlugin = plugin;\r\n const assetContainer = syncedPlugin.loadAssetContainer(scene, data, fileInfo.rootUrl, errorHandler);\r\n if (!assetContainer) {\r\n return;\r\n }\r\n assetContainer.populateRootNodes();\r\n scene.loadingPluginName = plugin.name;\r\n successHandler(assetContainer);\r\n } else if ((plugin).loadAssetContainerAsync) {\r\n const asyncedPlugin = plugin;\r\n asyncedPlugin\r\n .loadAssetContainerAsync(scene, data, fileInfo.rootUrl, progressHandler, fileInfo.name)\r\n .then((assetContainer) => {\r\n assetContainer.populateRootNodes();\r\n scene.loadingPluginName = plugin.name;\r\n successHandler(assetContainer);\r\n })\r\n .catch((error) => {\r\n errorHandler(error.message, error);\r\n });\r\n } else {\r\n errorHandler(\"LoadAssetContainer is not supported by this plugin. Plugin did not provide a loadAssetContainer or loadAssetContainerAsync method.\");\r\n }\r\n },\r\n progressHandler,\r\n errorHandler,\r\n disposeHandler,\r\n pluginExtension,\r\n name\r\n );\r\n }\r\n\r\n /**\r\n * Load a scene into an asset container\r\n * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb)\r\n * @param sceneFilename a string that defines the name of the scene file or starts with \"data:\" following by the stringified version of the scene (default: empty string)\r\n * @param scene is the instance of Scene to append to\r\n * @param onProgress a callback with a progress event for each file being loaded\r\n * @param pluginExtension the extension used to determine the plugin\r\n * @returns The loaded asset container\r\n */\r\n public static LoadAssetContainerAsync(\r\n rootUrl: string,\r\n sceneFilename: string | File = \"\",\r\n scene: Nullable = EngineStore.LastCreatedScene,\r\n onProgress: Nullable<(event: ISceneLoaderProgressEvent) => void> = null,\r\n pluginExtension: Nullable = null\r\n ): Promise {\r\n return new Promise((resolve, reject) => {\r\n SceneLoader.LoadAssetContainer(\r\n rootUrl,\r\n sceneFilename,\r\n scene,\r\n (assetContainer) => {\r\n resolve(assetContainer);\r\n },\r\n onProgress,\r\n (scene, message, exception) => {\r\n reject(exception || new Error(message));\r\n },\r\n pluginExtension\r\n );\r\n });\r\n }\r\n\r\n /**\r\n * Import animations from a file into a scene\r\n * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb)\r\n * @param sceneFilename a string that defines the name of the scene file or starts with \"data:\" following by the stringified version of the scene or a File object (default: empty string)\r\n * @param scene is the instance of BABYLON.Scene to append to (default: last created scene)\r\n * @param overwriteAnimations when true, animations are cleaned before importing new ones. Animations are appended otherwise\r\n * @param animationGroupLoadingMode defines how to handle old animations groups before importing new ones\r\n * @param targetConverter defines a function used to convert animation targets from loaded scene to current scene (default: search node by name)\r\n * @param onSuccess a callback with the scene when import succeeds\r\n * @param onProgress a callback with a progress event for each file being loaded\r\n * @param onError a callback with the scene, a message, and possibly an exception when import fails\r\n * @param pluginExtension the extension used to determine the plugin\r\n */\r\n public static ImportAnimations(\r\n rootUrl: string,\r\n sceneFilename: string | File = \"\",\r\n scene: Nullable = EngineStore.LastCreatedScene,\r\n overwriteAnimations = true,\r\n animationGroupLoadingMode = SceneLoaderAnimationGroupLoadingMode.Clean,\r\n targetConverter: Nullable<(target: any) => any> = null,\r\n onSuccess: Nullable<(scene: Scene) => void> = null,\r\n onProgress: Nullable<(event: ISceneLoaderProgressEvent) => void> = null,\r\n onError: Nullable<(scene: Scene, message: string, exception?: any) => void> = null,\r\n pluginExtension: Nullable = null\r\n ): void {\r\n if (!scene) {\r\n Logger.Error(\"No scene available to load animations to\");\r\n return;\r\n }\r\n\r\n if (overwriteAnimations) {\r\n // Reset, stop and dispose all animations before loading new ones\r\n for (const animatable of scene.animatables) {\r\n animatable.reset();\r\n }\r\n scene.stopAllAnimations();\r\n scene.animationGroups.slice().forEach((animationGroup) => {\r\n animationGroup.dispose();\r\n });\r\n const nodes = scene.getNodes();\r\n nodes.forEach((node) => {\r\n if (node.animations) {\r\n node.animations = [];\r\n }\r\n });\r\n } else {\r\n switch (animationGroupLoadingMode) {\r\n case SceneLoaderAnimationGroupLoadingMode.Clean:\r\n scene.animationGroups.slice().forEach((animationGroup) => {\r\n animationGroup.dispose();\r\n });\r\n break;\r\n case SceneLoaderAnimationGroupLoadingMode.Stop:\r\n scene.animationGroups.forEach((animationGroup) => {\r\n animationGroup.stop();\r\n });\r\n break;\r\n case SceneLoaderAnimationGroupLoadingMode.Sync:\r\n scene.animationGroups.forEach((animationGroup) => {\r\n animationGroup.reset();\r\n animationGroup.restart();\r\n });\r\n break;\r\n case SceneLoaderAnimationGroupLoadingMode.NoSync:\r\n // nothing to do\r\n break;\r\n default:\r\n Logger.Error(\"Unknown animation group loading mode value '\" + animationGroupLoadingMode + \"'\");\r\n return;\r\n }\r\n }\r\n\r\n const startingIndexForNewAnimatables = scene.animatables.length;\r\n\r\n const onAssetContainerLoaded = (container: AssetContainer) => {\r\n container.mergeAnimationsTo(scene, scene.animatables.slice(startingIndexForNewAnimatables), targetConverter);\r\n\r\n container.dispose();\r\n\r\n scene.onAnimationFileImportedObservable.notifyObservers(scene);\r\n\r\n if (onSuccess) {\r\n onSuccess(scene);\r\n }\r\n };\r\n\r\n this.LoadAssetContainer(rootUrl, sceneFilename, scene, onAssetContainerLoaded, onProgress, onError, pluginExtension);\r\n }\r\n\r\n /**\r\n * Import animations from a file into a scene\r\n * @param rootUrl a string that defines the root url for the scene and resources or the concatenation of rootURL and filename (e.g. http://example.com/test.glb)\r\n * @param sceneFilename a string that defines the name of the scene file or starts with \"data:\" following by the stringified version of the scene or a File object (default: empty string)\r\n * @param scene is the instance of BABYLON.Scene to append to (default: last created scene)\r\n * @param overwriteAnimations when true, animations are cleaned before importing new ones. Animations are appended otherwise\r\n * @param animationGroupLoadingMode defines how to handle old animations groups before importing new ones\r\n * @param targetConverter defines a function used to convert animation targets from loaded scene to current scene (default: search node by name)\r\n * @param onSuccess a callback with the scene when import succeeds\r\n * @param onProgress a callback with a progress event for each file being loaded\r\n * @param onError a callback with the scene, a message, and possibly an exception when import fails\r\n * @param pluginExtension the extension used to determine the plugin\r\n * @returns the updated scene with imported animations\r\n */\r\n public static ImportAnimationsAsync(\r\n rootUrl: string,\r\n sceneFilename: string | File = \"\",\r\n scene: Nullable = EngineStore.LastCreatedScene,\r\n overwriteAnimations = true,\r\n animationGroupLoadingMode = SceneLoaderAnimationGroupLoadingMode.Clean,\r\n targetConverter: Nullable<(target: any) => any> = null,\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n onSuccess: Nullable<(scene: Scene) => void> = null,\r\n onProgress: Nullable<(event: ISceneLoaderProgressEvent) => void> = null,\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n onError: Nullable<(scene: Scene, message: string, exception?: any) => void> = null,\r\n pluginExtension: Nullable = null\r\n ): Promise {\r\n return new Promise((resolve, reject) => {\r\n SceneLoader.ImportAnimations(\r\n rootUrl,\r\n sceneFilename,\r\n scene,\r\n overwriteAnimations,\r\n animationGroupLoadingMode,\r\n targetConverter,\r\n (_scene: Scene) => {\r\n resolve(_scene);\r\n },\r\n onProgress,\r\n (_scene: Scene, message: string, exception: any) => {\r\n reject(exception || new Error(message));\r\n },\r\n pluginExtension\r\n );\r\n });\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport type { InternalTexture } from \"../Materials/Textures/internalTexture\";\r\nimport type { ThinEngine } from \"../Engines/thinEngine\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport { AutoReleaseWorkerPool } from \"./workerPool\";\r\nimport { Tools } from \"./tools\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { ICompressedFormatCapabilities, IDecodedData, IKTX2DecoderOptions } from \"core/Materials/Textures/ktx2decoderTypes\";\r\nimport { EngineFormat, TranscodeTarget } from \"core/Materials/Textures/ktx2decoderTypes\";\r\n\r\ndeclare let KTX2DECODER: any;\r\n\r\nfunction getAbsoluteUrlOrNull(url: Nullable): Nullable {\r\n return url ? Tools.GetAbsoluteUrl(url) : null;\r\n}\r\n\r\nfunction applyConfig(urls: typeof KhronosTextureContainer2.URLConfig): void {\r\n if (urls.wasmUASTCToASTC !== null) {\r\n KTX2DECODER.LiteTranscoder_UASTC_ASTC.WasmModuleURL = urls.wasmUASTCToASTC;\r\n }\r\n\r\n if (urls.wasmUASTCToBC7 !== null) {\r\n KTX2DECODER.LiteTranscoder_UASTC_BC7.WasmModuleURL = urls.wasmUASTCToBC7;\r\n }\r\n\r\n if (urls.wasmUASTCToRGBA_UNORM !== null) {\r\n KTX2DECODER.LiteTranscoder_UASTC_RGBA_UNORM.WasmModuleURL = urls.wasmUASTCToRGBA_UNORM;\r\n }\r\n\r\n if (urls.wasmUASTCToRGBA_SRGB !== null) {\r\n KTX2DECODER.LiteTranscoder_UASTC_RGBA_SRGB.WasmModuleURL = urls.wasmUASTCToRGBA_SRGB;\r\n }\r\n\r\n if (urls.wasmUASTCToR8_UNORM !== null) {\r\n KTX2DECODER.LiteTranscoder_UASTC_R8_UNORM.WasmModuleURL = urls.wasmUASTCToR8_UNORM;\r\n }\r\n\r\n if (urls.wasmUASTCToRG8_UNORM !== null) {\r\n KTX2DECODER.LiteTranscoder_UASTC_RG8_UNORM.WasmModuleURL = urls.wasmUASTCToRG8_UNORM;\r\n }\r\n\r\n if (urls.jsMSCTranscoder !== null) {\r\n KTX2DECODER.MSCTranscoder.JSModuleURL = urls.jsMSCTranscoder;\r\n }\r\n\r\n if (urls.wasmMSCTranscoder !== null) {\r\n KTX2DECODER.MSCTranscoder.WasmModuleURL = urls.wasmMSCTranscoder;\r\n }\r\n\r\n if (urls.wasmZSTDDecoder !== null) {\r\n KTX2DECODER.ZSTDDecoder.WasmModuleURL = urls.wasmZSTDDecoder;\r\n }\r\n}\r\n\r\n/**\r\n * Class that defines the default KTX2 decoder options.\r\n *\r\n * This class is useful for providing options to the KTX2 decoder to control how the source data is transcoded.\r\n */\r\nexport class DefaultKTX2DecoderOptions {\r\n private _isDirty = true;\r\n\r\n /**\r\n * Gets the dirty flag\r\n */\r\n public get isDirty() {\r\n return this._isDirty;\r\n }\r\n\r\n private _useRGBAIfASTCBC7NotAvailableWhenUASTC?: boolean;\r\n /**\r\n * force a (uncompressed) RGBA transcoded format if transcoding a UASTC source format and ASTC + BC7 are not available as a compressed transcoded format\r\n */\r\n public get useRGBAIfASTCBC7NotAvailableWhenUASTC() {\r\n return this._useRGBAIfASTCBC7NotAvailableWhenUASTC;\r\n }\r\n\r\n public set useRGBAIfASTCBC7NotAvailableWhenUASTC(value: boolean | undefined) {\r\n if (this._useRGBAIfASTCBC7NotAvailableWhenUASTC === value) {\r\n return;\r\n }\r\n this._useRGBAIfASTCBC7NotAvailableWhenUASTC = value;\r\n this._isDirty = true;\r\n }\r\n\r\n private _useRGBAIfOnlyBC1BC3AvailableWhenUASTC?: boolean = true;\r\n /**\r\n * force a (uncompressed) RGBA transcoded format if transcoding a UASTC source format and only BC1 or BC3 are available as a compressed transcoded format.\r\n * This property is true by default to favor speed over memory, because currently transcoding from UASTC to BC1/3 is slow because the transcoder transcodes\r\n * to uncompressed and then recompresses the texture\r\n */\r\n public get useRGBAIfOnlyBC1BC3AvailableWhenUASTC() {\r\n return this._useRGBAIfOnlyBC1BC3AvailableWhenUASTC;\r\n }\r\n\r\n public set useRGBAIfOnlyBC1BC3AvailableWhenUASTC(value: boolean | undefined) {\r\n if (this._useRGBAIfOnlyBC1BC3AvailableWhenUASTC === value) {\r\n return;\r\n }\r\n this._useRGBAIfOnlyBC1BC3AvailableWhenUASTC = value;\r\n this._isDirty = true;\r\n }\r\n\r\n private _forceRGBA?: boolean;\r\n /**\r\n * force to always use (uncompressed) RGBA for transcoded format\r\n */\r\n public get forceRGBA() {\r\n return this._forceRGBA;\r\n }\r\n\r\n public set forceRGBA(value: boolean | undefined) {\r\n if (this._forceRGBA === value) {\r\n return;\r\n }\r\n this._forceRGBA = value;\r\n this._isDirty = true;\r\n }\r\n\r\n private _forceR8?: boolean;\r\n /**\r\n * force to always use (uncompressed) R8 for transcoded format\r\n */\r\n public get forceR8() {\r\n return this._forceR8;\r\n }\r\n\r\n public set forceR8(value: boolean | undefined) {\r\n if (this._forceR8 === value) {\r\n return;\r\n }\r\n this._forceR8 = value;\r\n this._isDirty = true;\r\n }\r\n\r\n private _forceRG8?: boolean;\r\n /**\r\n * force to always use (uncompressed) RG8 for transcoded format\r\n */\r\n public get forceRG8() {\r\n return this._forceRG8;\r\n }\r\n\r\n public set forceRG8(value: boolean | undefined) {\r\n if (this._forceRG8 === value) {\r\n return;\r\n }\r\n this._forceRG8 = value;\r\n this._isDirty = true;\r\n }\r\n\r\n private _bypassTranscoders?: string[];\r\n /**\r\n * list of transcoders to bypass when looking for a suitable transcoder. The available transcoders are:\r\n * UniversalTranscoder_UASTC_ASTC\r\n * UniversalTranscoder_UASTC_BC7\r\n * UniversalTranscoder_UASTC_RGBA_UNORM\r\n * UniversalTranscoder_UASTC_RGBA_SRGB\r\n * UniversalTranscoder_UASTC_R8_UNORM\r\n * UniversalTranscoder_UASTC_RG8_UNORM\r\n * MSCTranscoder\r\n */\r\n public get bypassTranscoders() {\r\n return this._bypassTranscoders;\r\n }\r\n\r\n public set bypassTranscoders(value: string[] | undefined) {\r\n if (this._bypassTranscoders === value) {\r\n return;\r\n }\r\n this._bypassTranscoders = value;\r\n this._isDirty = true;\r\n }\r\n\r\n private _ktx2DecoderOptions: IKTX2DecoderOptions = {};\r\n\r\n /** @internal */\r\n public _getKTX2DecoderOptions(): IKTX2DecoderOptions {\r\n if (!this._isDirty) {\r\n return this._ktx2DecoderOptions;\r\n }\r\n\r\n this._isDirty = false;\r\n\r\n const options: IKTX2DecoderOptions = {\r\n useRGBAIfASTCBC7NotAvailableWhenUASTC: this._useRGBAIfASTCBC7NotAvailableWhenUASTC,\r\n forceRGBA: this._forceRGBA,\r\n forceR8: this._forceR8,\r\n forceRG8: this._forceRG8,\r\n bypassTranscoders: this._bypassTranscoders,\r\n };\r\n\r\n if (this.useRGBAIfOnlyBC1BC3AvailableWhenUASTC) {\r\n options.transcodeFormatDecisionTree = {\r\n UASTC: {\r\n transcodeFormat: [TranscodeTarget.BC1_RGB, TranscodeTarget.BC3_RGBA],\r\n yes: {\r\n transcodeFormat: TranscodeTarget.RGBA32,\r\n engineFormat: EngineFormat.RGBA8Format,\r\n roundToMultiple4: false,\r\n },\r\n },\r\n };\r\n }\r\n\r\n this._ktx2DecoderOptions = options;\r\n\r\n return options;\r\n }\r\n}\r\n\r\n/**\r\n * Class for loading KTX2 files\r\n */\r\nexport class KhronosTextureContainer2 {\r\n private static _WorkerPoolPromise?: Promise;\r\n private static _DecoderModulePromise?: Promise;\r\n\r\n /**\r\n * URLs to use when loading the KTX2 decoder module as well as its dependencies\r\n * If a url is null, the default url is used (pointing to https://preview.babylonjs.com)\r\n * Note that jsDecoderModule can't be null and that the other dependencies will only be loaded if necessary\r\n * Urls you can change:\r\n * URLConfig.jsDecoderModule\r\n * URLConfig.wasmUASTCToASTC\r\n * URLConfig.wasmUASTCToBC7\r\n * URLConfig.wasmUASTCToRGBA_UNORM\r\n * URLConfig.wasmUASTCToRGBA_SRGB\r\n * URLConfig.wasmUASTCToR8_UNORM\r\n * URLConfig.wasmUASTCToRG8_UNORM\r\n * URLConfig.jsMSCTranscoder\r\n * URLConfig.wasmMSCTranscoder\r\n * URLConfig.wasmZSTDDecoder\r\n * You can see their default values in this PG: https://playground.babylonjs.com/#EIJH8L#29\r\n */\r\n public static URLConfig: {\r\n jsDecoderModule: string;\r\n wasmUASTCToASTC: Nullable;\r\n wasmUASTCToBC7: Nullable;\r\n wasmUASTCToRGBA_UNORM: Nullable;\r\n wasmUASTCToRGBA_SRGB: Nullable;\r\n wasmUASTCToR8_UNORM: Nullable;\r\n wasmUASTCToRG8_UNORM: Nullable;\r\n jsMSCTranscoder: Nullable;\r\n wasmMSCTranscoder: Nullable;\r\n wasmZSTDDecoder: Nullable;\r\n } = {\r\n jsDecoderModule: \"https://preview.babylonjs.com/babylon.ktx2Decoder.js\",\r\n wasmUASTCToASTC: null,\r\n wasmUASTCToBC7: null,\r\n wasmUASTCToRGBA_UNORM: null,\r\n wasmUASTCToRGBA_SRGB: null,\r\n wasmUASTCToR8_UNORM: null,\r\n wasmUASTCToRG8_UNORM: null,\r\n jsMSCTranscoder: null,\r\n wasmMSCTranscoder: null,\r\n wasmZSTDDecoder: null,\r\n };\r\n\r\n /**\r\n * Default number of workers used to handle data decoding\r\n */\r\n public static DefaultNumWorkers = KhronosTextureContainer2.GetDefaultNumWorkers();\r\n\r\n /**\r\n * Default configuration for the KTX2 decoder.\r\n * The options defined in this way have priority over those passed when creating a KTX2 texture with new Texture(...).\r\n */\r\n public static DefaultDecoderOptions = new DefaultKTX2DecoderOptions();\r\n\r\n private static GetDefaultNumWorkers(): number {\r\n if (typeof navigator !== \"object\" || !navigator.hardwareConcurrency) {\r\n return 1;\r\n }\r\n\r\n // Use 50% of the available logical processors but capped at 4.\r\n return Math.min(Math.floor(navigator.hardwareConcurrency * 0.5), 4);\r\n }\r\n\r\n private _engine: ThinEngine;\r\n\r\n private static _Initialize(numWorkers: number): void {\r\n if (KhronosTextureContainer2._WorkerPoolPromise || KhronosTextureContainer2._DecoderModulePromise) {\r\n return;\r\n }\r\n\r\n const urls = {\r\n jsDecoderModule: Tools.GetAbsoluteUrl(this.URLConfig.jsDecoderModule),\r\n wasmUASTCToASTC: getAbsoluteUrlOrNull(this.URLConfig.wasmUASTCToASTC),\r\n wasmUASTCToBC7: getAbsoluteUrlOrNull(this.URLConfig.wasmUASTCToBC7),\r\n wasmUASTCToRGBA_UNORM: getAbsoluteUrlOrNull(this.URLConfig.wasmUASTCToRGBA_UNORM),\r\n wasmUASTCToRGBA_SRGB: getAbsoluteUrlOrNull(this.URLConfig.wasmUASTCToRGBA_SRGB),\r\n wasmUASTCToR8_UNORM: getAbsoluteUrlOrNull(this.URLConfig.wasmUASTCToR8_UNORM),\r\n wasmUASTCToRG8_UNORM: getAbsoluteUrlOrNull(this.URLConfig.wasmUASTCToRG8_UNORM),\r\n jsMSCTranscoder: getAbsoluteUrlOrNull(this.URLConfig.jsMSCTranscoder),\r\n wasmMSCTranscoder: getAbsoluteUrlOrNull(this.URLConfig.wasmMSCTranscoder),\r\n wasmZSTDDecoder: getAbsoluteUrlOrNull(this.URLConfig.wasmZSTDDecoder),\r\n };\r\n\r\n if (numWorkers && typeof Worker === \"function\" && typeof URL !== \"undefined\") {\r\n KhronosTextureContainer2._WorkerPoolPromise = new Promise((resolve) => {\r\n const workerContent = `${applyConfig}(${workerFunc})()`;\r\n const workerBlobUrl = URL.createObjectURL(new Blob([workerContent], { type: \"application/javascript\" }));\r\n resolve(\r\n new AutoReleaseWorkerPool(\r\n numWorkers,\r\n () =>\r\n new Promise((resolve, reject) => {\r\n const worker = new Worker(workerBlobUrl);\r\n\r\n const onError = (error: ErrorEvent) => {\r\n worker.removeEventListener(\"error\", onError);\r\n worker.removeEventListener(\"message\", onMessage);\r\n reject(error);\r\n };\r\n\r\n const onMessage = (message: MessageEvent) => {\r\n if (message.data.action === \"init\") {\r\n worker.removeEventListener(\"error\", onError);\r\n worker.removeEventListener(\"message\", onMessage);\r\n resolve(worker);\r\n }\r\n };\r\n\r\n worker.addEventListener(\"error\", onError);\r\n worker.addEventListener(\"message\", onMessage);\r\n\r\n worker.postMessage({\r\n action: \"init\",\r\n urls: urls,\r\n });\r\n })\r\n )\r\n );\r\n });\r\n } else if (typeof KTX2DECODER === \"undefined\") {\r\n KhronosTextureContainer2._DecoderModulePromise = Tools.LoadScriptAsync(urls.jsDecoderModule).then(() => {\r\n KTX2DECODER.MSCTranscoder.UseFromWorkerThread = false;\r\n KTX2DECODER.WASMMemoryManager.LoadBinariesFromCurrentThread = true;\r\n applyConfig(urls);\r\n return new KTX2DECODER.KTX2Decoder();\r\n });\r\n } else {\r\n KTX2DECODER.MSCTranscoder.UseFromWorkerThread = false;\r\n KTX2DECODER.WASMMemoryManager.LoadBinariesFromCurrentThread = true;\r\n KhronosTextureContainer2._DecoderModulePromise = Promise.resolve(new KTX2DECODER.KTX2Decoder());\r\n }\r\n }\r\n\r\n /**\r\n * Constructor\r\n * @param engine The engine to use\r\n * @param numWorkers The number of workers for async operations. Specify `0` to disable web workers and run synchronously in the current context.\r\n */\r\n public constructor(engine: ThinEngine, numWorkers = KhronosTextureContainer2.DefaultNumWorkers) {\r\n this._engine = engine;\r\n\r\n KhronosTextureContainer2._Initialize(numWorkers);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public uploadAsync(data: ArrayBufferView, internalTexture: InternalTexture, options?: IKTX2DecoderOptions & IDecodedData): Promise {\r\n const caps = this._engine.getCaps();\r\n\r\n const compressedTexturesCaps: ICompressedFormatCapabilities = {\r\n astc: !!caps.astc,\r\n bptc: !!caps.bptc,\r\n s3tc: !!caps.s3tc,\r\n pvrtc: !!caps.pvrtc,\r\n etc2: !!caps.etc2,\r\n etc1: !!caps.etc1,\r\n };\r\n\r\n if (KhronosTextureContainer2._WorkerPoolPromise) {\r\n return KhronosTextureContainer2._WorkerPoolPromise.then((workerPool) => {\r\n return new Promise((resolve, reject) => {\r\n workerPool.push((worker, onComplete) => {\r\n const onError = (error: ErrorEvent) => {\r\n worker.removeEventListener(\"error\", onError);\r\n worker.removeEventListener(\"message\", onMessage);\r\n reject(error);\r\n onComplete();\r\n };\r\n\r\n const onMessage = (message: MessageEvent) => {\r\n if (message.data.action === \"decoded\") {\r\n worker.removeEventListener(\"error\", onError);\r\n worker.removeEventListener(\"message\", onMessage);\r\n if (!message.data.success) {\r\n reject({ message: message.data.msg });\r\n } else {\r\n try {\r\n this._createTexture(message.data.decodedData, internalTexture, options);\r\n resolve();\r\n } catch (err) {\r\n reject({ message: err });\r\n }\r\n }\r\n onComplete();\r\n }\r\n };\r\n\r\n worker.addEventListener(\"error\", onError);\r\n worker.addEventListener(\"message\", onMessage);\r\n worker.postMessage({ action: \"setDefaultDecoderOptions\", options: KhronosTextureContainer2.DefaultDecoderOptions._getKTX2DecoderOptions() });\r\n\r\n const dataCopy = new Uint8Array(data.byteLength);\r\n dataCopy.set(new Uint8Array(data.buffer, data.byteOffset, data.byteLength));\r\n\r\n worker.postMessage({ action: \"decode\", data: dataCopy, caps: compressedTexturesCaps, options }, [dataCopy.buffer]);\r\n });\r\n });\r\n });\r\n } else if (KhronosTextureContainer2._DecoderModulePromise) {\r\n return KhronosTextureContainer2._DecoderModulePromise.then((decoder) => {\r\n if (KhronosTextureContainer2.DefaultDecoderOptions.isDirty) {\r\n KTX2DECODER.KTX2Decoder.DefaultDecoderOptions = KhronosTextureContainer2.DefaultDecoderOptions._getKTX2DecoderOptions();\r\n }\r\n return new Promise((resolve, reject) => {\r\n decoder\r\n .decode(data, caps)\r\n .then((data: IDecodedData) => {\r\n this._createTexture(data, internalTexture);\r\n resolve();\r\n })\r\n .catch((reason: any) => {\r\n reject({ message: reason });\r\n });\r\n });\r\n });\r\n }\r\n\r\n throw new Error(\"KTX2 decoder module is not available\");\r\n }\r\n\r\n protected _createTexture(data: IDecodedData, internalTexture: InternalTexture, options?: IKTX2DecoderOptions & IDecodedData): void {\r\n const oglTexture2D = 3553; // gl.TEXTURE_2D\r\n\r\n this._engine._bindTextureDirectly(oglTexture2D, internalTexture);\r\n\r\n if (options) {\r\n // return back some information about the decoded data\r\n options.transcodedFormat = data.transcodedFormat;\r\n options.isInGammaSpace = data.isInGammaSpace;\r\n options.hasAlpha = data.hasAlpha;\r\n options.transcoderName = data.transcoderName;\r\n }\r\n\r\n let isUncompressedFormat = true;\r\n\r\n switch (data.transcodedFormat) {\r\n case 0x8058 /* RGBA8 */:\r\n internalTexture.type = Constants.TEXTURETYPE_UNSIGNED_BYTE;\r\n internalTexture.format = Constants.TEXTUREFORMAT_RGBA;\r\n break;\r\n case 0x8229 /* R8 */:\r\n internalTexture.type = Constants.TEXTURETYPE_UNSIGNED_BYTE;\r\n internalTexture.format = Constants.TEXTUREFORMAT_R;\r\n break;\r\n case 0x822b /* RG8 */:\r\n internalTexture.type = Constants.TEXTURETYPE_UNSIGNED_BYTE;\r\n internalTexture.format = Constants.TEXTUREFORMAT_RG;\r\n break;\r\n default:\r\n internalTexture.format = data.transcodedFormat;\r\n isUncompressedFormat = false;\r\n break;\r\n }\r\n\r\n internalTexture._gammaSpace = data.isInGammaSpace;\r\n internalTexture.generateMipMaps = data.mipmaps.length > 1;\r\n\r\n if (data.errors) {\r\n throw new Error(\"KTX2 container - could not transcode the data. \" + data.errors);\r\n }\r\n\r\n for (let t = 0; t < data.mipmaps.length; ++t) {\r\n const mipmap = data.mipmaps[t];\r\n\r\n if (!mipmap || !mipmap.data) {\r\n throw new Error(\"KTX2 container - could not transcode one of the image\");\r\n }\r\n\r\n if (isUncompressedFormat) {\r\n // uncompressed RGBA / R8 / RG8\r\n internalTexture.width = mipmap.width; // need to set width/height so that the call to _uploadDataToTextureDirectly uses the right dimensions\r\n internalTexture.height = mipmap.height;\r\n\r\n this._engine._uploadDataToTextureDirectly(internalTexture, mipmap.data, 0, t, undefined, true);\r\n } else {\r\n this._engine._uploadCompressedDataToTextureDirectly(internalTexture, data.transcodedFormat, mipmap.width, mipmap.height, mipmap.data, 0, t);\r\n }\r\n }\r\n\r\n internalTexture._extension = \".ktx2\";\r\n internalTexture.width = data.mipmaps[0].width;\r\n internalTexture.height = data.mipmaps[0].height;\r\n internalTexture.isReady = true;\r\n\r\n this._engine._bindTextureDirectly(oglTexture2D, null);\r\n }\r\n\r\n /**\r\n * Checks if the given data starts with a KTX2 file identifier.\r\n * @param data the data to check\r\n * @returns true if the data is a KTX2 file or false otherwise\r\n */\r\n public static IsValid(data: ArrayBufferView): boolean {\r\n if (data.byteLength >= 12) {\r\n // '«', 'K', 'T', 'X', ' ', '2', '0', '»', '\\r', '\\n', '\\x1A', '\\n'\r\n const identifier = new Uint8Array(data.buffer, data.byteOffset, 12);\r\n if (\r\n identifier[0] === 0xab &&\r\n identifier[1] === 0x4b &&\r\n identifier[2] === 0x54 &&\r\n identifier[3] === 0x58 &&\r\n identifier[4] === 0x20 &&\r\n identifier[5] === 0x32 &&\r\n identifier[6] === 0x30 &&\r\n identifier[7] === 0xbb &&\r\n identifier[8] === 0x0d &&\r\n identifier[9] === 0x0a &&\r\n identifier[10] === 0x1a &&\r\n identifier[11] === 0x0a\r\n ) {\r\n return true;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n}\r\n\r\ndeclare function importScripts(...urls: string[]): void;\r\ndeclare function postMessage(message: any, transfer?: any[]): void;\r\n\r\nfunction workerFunc(): void {\r\n let ktx2Decoder: any;\r\n\r\n onmessage = (event) => {\r\n if (!event.data) {\r\n return;\r\n }\r\n switch (event.data.action) {\r\n case \"init\": {\r\n const urls = event.data.urls;\r\n importScripts(urls.jsDecoderModule);\r\n applyConfig(urls);\r\n ktx2Decoder = new KTX2DECODER.KTX2Decoder();\r\n postMessage({ action: \"init\" });\r\n break;\r\n }\r\n case \"setDefaultDecoderOptions\": {\r\n KTX2DECODER.KTX2Decoder.DefaultDecoderOptions = event.data.options;\r\n break;\r\n }\r\n case \"decode\":\r\n ktx2Decoder\r\n .decode(event.data.data, event.data.caps, event.data.options)\r\n .then((data: IDecodedData) => {\r\n const buffers = [];\r\n for (let mip = 0; mip < data.mipmaps.length; ++mip) {\r\n const mipmap = data.mipmaps[mip];\r\n if (mipmap && mipmap.data) {\r\n buffers.push(mipmap.data.buffer);\r\n }\r\n }\r\n postMessage({ action: \"decoded\", success: true, decodedData: data }, buffers);\r\n })\r\n .catch((reason: any) => {\r\n postMessage({ action: \"decoded\", success: false, msg: reason });\r\n });\r\n break;\r\n }\r\n };\r\n}\r\n","import { KhronosTextureContainer } from \"../../../Misc/khronosTextureContainer\";\r\nimport { KhronosTextureContainer2 } from \"../../../Misc/khronosTextureContainer2\";\r\nimport type { Nullable } from \"../../../types\";\r\nimport { Engine } from \"../../../Engines/engine\";\r\nimport type { InternalTexture } from \"../../../Materials/Textures/internalTexture\";\r\nimport type { IInternalTextureLoader } from \"../../../Materials/Textures/internalTextureLoader\";\r\nimport { Logger } from \"../../../Misc/logger\";\r\nimport { Constants } from \"../../../Engines/constants\";\r\n\r\nfunction mapSRGBToLinear(format: number): Nullable {\r\n switch (format) {\r\n case Constants.TEXTUREFORMAT_COMPRESSED_SRGB_S3TC_DXT1_EXT:\r\n return Constants.TEXTUREFORMAT_COMPRESSED_RGB_S3TC_DXT1;\r\n case Constants.TEXTUREFORMAT_COMPRESSED_SRGB_ALPHA_S3TC_DXT3_EXT:\r\n return Constants.TEXTUREFORMAT_COMPRESSED_RGBA_S3TC_DXT3;\r\n case Constants.TEXTUREFORMAT_COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT:\r\n return Constants.TEXTUREFORMAT_COMPRESSED_RGBA_S3TC_DXT5;\r\n case Constants.TEXTUREFORMAT_COMPRESSED_SRGB8_ETC2:\r\n return Constants.TEXTUREFORMAT_COMPRESSED_RGB8_ETC2;\r\n case Constants.TEXTUREFORMAT_COMPRESSED_SRGB8_ALPHA8_ETC2_EAC:\r\n return Constants.TEXTUREFORMAT_COMPRESSED_RGBA8_ETC2_EAC;\r\n case Constants.TEXTUREFORMAT_COMPRESSED_SRGB8_PUNCHTHROUGH_ALPHA1_ETC2:\r\n return Constants.TEXTUREFORMAT_COMPRESSED_RGB8_PUNCHTHROUGH_ALPHA1_ETC2;\r\n case Constants.TEXTUREFORMAT_COMPRESSED_SRGB8_ALPHA8_ASTC_4x4_KHR:\r\n return Constants.TEXTUREFORMAT_COMPRESSED_RGBA_ASTC_4x4;\r\n case Constants.TEXTUREFORMAT_COMPRESSED_SRGB_ALPHA_BPTC_UNORM:\r\n return Constants.TEXTUREFORMAT_COMPRESSED_RGBA_BPTC_UNORM;\r\n }\r\n\r\n return null;\r\n}\r\n\r\n/**\r\n * Implementation of the KTX Texture Loader.\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class _KTXTextureLoader implements IInternalTextureLoader {\r\n /**\r\n * Defines whether the loader supports cascade loading the different faces.\r\n */\r\n public readonly supportCascades = false;\r\n\r\n /**\r\n * This returns if the loader support the current file information.\r\n * @param extension defines the file extension of the file being loaded\r\n * @param mimeType defines the optional mime type of the file being loaded\r\n * @returns true if the loader can load the specified file\r\n */\r\n public canLoad(extension: string, mimeType?: string): boolean {\r\n // The \".ktx2\" file extension is still up for debate: https://github.com/KhronosGroup/KTX-Specification/issues/18\r\n return extension.endsWith(\".ktx\") || extension.endsWith(\".ktx2\") || mimeType === \"image/ktx\" || mimeType === \"image/ktx2\";\r\n }\r\n\r\n /**\r\n * Uploads the cube texture data to the WebGL texture. It has already been bound.\r\n * @param data contains the texture data\r\n * @param texture defines the BabylonJS internal texture\r\n * @param createPolynomials will be true if polynomials have been requested\r\n * @param onLoad defines the callback to trigger once the texture is ready\r\n */\r\n public loadCubeData(data: ArrayBufferView | ArrayBufferView[], texture: InternalTexture, createPolynomials: boolean, onLoad: Nullable<(data?: any) => void>): void {\r\n if (Array.isArray(data)) {\r\n return;\r\n }\r\n\r\n // Need to invert vScale as invertY via UNPACK_FLIP_Y_WEBGL is not supported by compressed texture\r\n texture._invertVScale = !texture.invertY;\r\n const engine = texture.getEngine() as Engine;\r\n const ktx = new KhronosTextureContainer(data, 6);\r\n\r\n const loadMipmap = ktx.numberOfMipmapLevels > 1 && texture.generateMipMaps;\r\n\r\n engine._unpackFlipY(true);\r\n\r\n ktx.uploadLevels(texture, texture.generateMipMaps);\r\n\r\n texture.width = ktx.pixelWidth;\r\n texture.height = ktx.pixelHeight;\r\n\r\n engine._setCubeMapTextureParams(texture, loadMipmap, ktx.numberOfMipmapLevels - 1);\r\n texture.isReady = true;\r\n texture.onLoadedObservable.notifyObservers(texture);\r\n texture.onLoadedObservable.clear();\r\n\r\n if (onLoad) {\r\n onLoad();\r\n }\r\n }\r\n\r\n /**\r\n * Uploads the 2D texture data to the WebGL texture. It has already been bound once in the callback.\r\n * @param data contains the texture data\r\n * @param texture defines the BabylonJS internal texture\r\n * @param callback defines the method to call once ready to upload\r\n * @param options\r\n */\r\n public loadData(\r\n data: ArrayBufferView,\r\n texture: InternalTexture,\r\n callback: (width: number, height: number, loadMipmap: boolean, isCompressed: boolean, done: () => void, loadFailed: boolean) => void,\r\n options?: any\r\n ): void {\r\n if (KhronosTextureContainer.IsValid(data)) {\r\n // Need to invert vScale as invertY via UNPACK_FLIP_Y_WEBGL is not supported by compressed texture\r\n texture._invertVScale = !texture.invertY;\r\n const ktx = new KhronosTextureContainer(data, 1);\r\n\r\n const mappedFormat = mapSRGBToLinear(ktx.glInternalFormat);\r\n if (mappedFormat) {\r\n texture.format = mappedFormat;\r\n texture._useSRGBBuffer = texture.getEngine()._getUseSRGBBuffer(true, texture.generateMipMaps);\r\n texture._gammaSpace = true;\r\n } else {\r\n texture.format = ktx.glInternalFormat;\r\n }\r\n\r\n callback(\r\n ktx.pixelWidth,\r\n ktx.pixelHeight,\r\n texture.generateMipMaps,\r\n true,\r\n () => {\r\n ktx.uploadLevels(texture, texture.generateMipMaps);\r\n },\r\n ktx.isInvalid\r\n );\r\n } else if (KhronosTextureContainer2.IsValid(data)) {\r\n const ktx2 = new KhronosTextureContainer2(texture.getEngine());\r\n ktx2.uploadAsync(data, texture, options).then(\r\n () => {\r\n callback(texture.width, texture.height, texture.generateMipMaps, true, () => {}, false);\r\n },\r\n (error) => {\r\n Logger.Warn(`Failed to load KTX2 texture data: ${error.message}`);\r\n callback(0, 0, false, false, () => {}, true);\r\n }\r\n );\r\n } else {\r\n Logger.Error(\"texture missing KTX identifier\");\r\n callback(0, 0, false, false, () => {}, true);\r\n }\r\n }\r\n}\r\n\r\n// Register the loader.\r\nEngine._TextureLoaders.unshift(new _KTXTextureLoader());\r\n","import { Vector3, Matrix, Quaternion, TmpVectors } from \"../Maths/math.vector\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Camera } from \"../Cameras/camera\";\r\nimport { FreeCamera } from \"../Cameras/freeCamera\";\r\nimport { TargetCamera } from \"../Cameras/targetCamera\";\r\nimport type { WebXRSessionManager } from \"./webXRSessionManager\";\r\nimport { Viewport } from \"../Maths/math.viewport\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport { WebXRTrackingState } from \"./webXRTypes\";\r\n\r\n/**\r\n * WebXR Camera which holds the views for the xrSession\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/webXR/webXRCamera\r\n */\r\nexport class WebXRCamera extends FreeCamera {\r\n private static _ScaleReadOnly = Vector3.One();\r\n\r\n private _firstFrame = false;\r\n private _referenceQuaternion: Quaternion = Quaternion.Identity();\r\n private _referencedPosition: Vector3 = new Vector3();\r\n private _trackingState: WebXRTrackingState = WebXRTrackingState.NOT_TRACKING;\r\n\r\n /**\r\n * Observable raised before camera teleportation\r\n */\r\n public onBeforeCameraTeleport = new Observable();\r\n\r\n /**\r\n * Observable raised after camera teleportation\r\n */\r\n public onAfterCameraTeleport = new Observable();\r\n\r\n /**\r\n * Notifies when the camera's tracking state has changed.\r\n * Notice - will also be triggered when tracking has started (at the beginning of the session)\r\n */\r\n public onTrackingStateChanged = new Observable();\r\n /**\r\n * Should position compensation execute on first frame.\r\n * This is used when copying the position from a native (non XR) camera\r\n */\r\n public compensateOnFirstFrame: boolean = true;\r\n\r\n /**\r\n * The last XRViewerPose from the current XRFrame\r\n * @internal\r\n */\r\n public _lastXRViewerPose?: XRViewerPose;\r\n\r\n /**\r\n * Creates a new webXRCamera, this should only be set at the camera after it has been updated by the xrSessionManager\r\n * @param name the name of the camera\r\n * @param scene the scene to add the camera to\r\n * @param _xrSessionManager a constructed xr session manager\r\n */\r\n constructor(name: string, scene: Scene, private _xrSessionManager: WebXRSessionManager) {\r\n super(name, Vector3.Zero(), scene);\r\n\r\n // Initial camera configuration\r\n this.minZ = 0.1;\r\n this.rotationQuaternion = new Quaternion();\r\n this.cameraRigMode = Camera.RIG_MODE_CUSTOM;\r\n this.updateUpVectorFromRotation = true;\r\n this._updateNumberOfRigCameras(1);\r\n // freeze projection matrix, which will be copied later\r\n this.freezeProjectionMatrix();\r\n this._deferOnly = true;\r\n\r\n this._xrSessionManager.onXRSessionInit.add(() => {\r\n this._referencedPosition.copyFromFloats(0, 0, 0);\r\n this._referenceQuaternion.copyFromFloats(0, 0, 0, 1);\r\n // first frame - camera's y position should be 0 for the correct offset\r\n this._firstFrame = this.compensateOnFirstFrame;\r\n });\r\n\r\n // Check transformation changes on each frame. Callback is added to be first so that the transformation will be\r\n // applied to the rest of the elements using the referenceSpace object\r\n this._xrSessionManager.onXRFrameObservable.add(\r\n () => {\r\n if (this._firstFrame) {\r\n this._updateFromXRSession();\r\n }\r\n\r\n if (this._deferredUpdated) {\r\n this.position.copyFrom(this._deferredPositionUpdate);\r\n this.rotationQuaternion.copyFrom(this._deferredRotationQuaternionUpdate);\r\n }\r\n\r\n this._updateReferenceSpace();\r\n this._updateFromXRSession();\r\n },\r\n undefined,\r\n true\r\n );\r\n }\r\n\r\n /**\r\n * Get the current XR tracking state of the camera\r\n */\r\n public get trackingState(): WebXRTrackingState {\r\n return this._trackingState;\r\n }\r\n\r\n private _setTrackingState(newState: WebXRTrackingState) {\r\n if (this._trackingState !== newState) {\r\n this._trackingState = newState;\r\n this.onTrackingStateChanged.notifyObservers(newState);\r\n }\r\n }\r\n\r\n /**\r\n * Return the user's height, unrelated to the current ground.\r\n * This will be the y position of this camera, when ground level is 0.\r\n */\r\n public get realWorldHeight(): number {\r\n const basePose = this._xrSessionManager.currentFrame && this._xrSessionManager.currentFrame.getViewerPose(this._xrSessionManager.baseReferenceSpace);\r\n if (basePose && basePose.transform) {\r\n return basePose.transform.position.y;\r\n } else {\r\n return 0;\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _updateForDualEyeDebugging(/*pupilDistance = 0.01*/) {\r\n // Create initial camera rigs\r\n this._updateNumberOfRigCameras(2);\r\n this.rigCameras[0].viewport = new Viewport(0, 0, 0.5, 1.0);\r\n // this.rigCameras[0].position.x = -pupilDistance / 2;\r\n this.rigCameras[0].outputRenderTarget = null;\r\n this.rigCameras[1].viewport = new Viewport(0.5, 0, 0.5, 1.0);\r\n // this.rigCameras[1].position.x = pupilDistance / 2;\r\n this.rigCameras[1].outputRenderTarget = null;\r\n }\r\n\r\n /**\r\n * Sets this camera's transformation based on a non-vr camera\r\n * @param otherCamera the non-vr camera to copy the transformation from\r\n * @param resetToBaseReferenceSpace should XR reset to the base reference space\r\n */\r\n public setTransformationFromNonVRCamera(otherCamera: Camera = this.getScene().activeCamera!, resetToBaseReferenceSpace: boolean = true) {\r\n if (!otherCamera || otherCamera === this) {\r\n return;\r\n }\r\n const mat = otherCamera.computeWorldMatrix();\r\n mat.decompose(undefined, this.rotationQuaternion, this.position);\r\n // set the ground level\r\n this.position.y = 0;\r\n Quaternion.FromEulerAnglesToRef(0, this.rotationQuaternion.toEulerAngles().y, 0, this.rotationQuaternion);\r\n this._firstFrame = true;\r\n if (resetToBaseReferenceSpace) {\r\n this._xrSessionManager.resetReferenceSpace();\r\n }\r\n }\r\n\r\n /**\r\n * Gets the current instance class name (\"WebXRCamera\").\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"WebXRCamera\";\r\n }\r\n\r\n /**\r\n * Set the target for the camera to look at.\r\n * Note that this only rotates around the Y axis, as opposed to the default behavior of other cameras\r\n * @param target the target to set the camera to look at\r\n */\r\n public setTarget(target: Vector3): void {\r\n // only rotate around the y axis!\r\n const tmpVector = TmpVectors.Vector3[1];\r\n target.subtractToRef(this.position, tmpVector);\r\n tmpVector.y = 0;\r\n tmpVector.normalize();\r\n const yRotation = Math.atan2(tmpVector.x, tmpVector.z);\r\n this.rotationQuaternion.toEulerAnglesToRef(tmpVector);\r\n Quaternion.FromEulerAnglesToRef(tmpVector.x, yRotation, tmpVector.z, this.rotationQuaternion);\r\n }\r\n\r\n public dispose() {\r\n super.dispose();\r\n this._lastXRViewerPose = undefined;\r\n }\r\n\r\n private _rotate180 = new Quaternion(0, 1, 0, 0);\r\n\r\n private _updateFromXRSession() {\r\n const pose = this._xrSessionManager.currentFrame && this._xrSessionManager.currentFrame.getViewerPose(this._xrSessionManager.referenceSpace);\r\n this._lastXRViewerPose = pose || undefined;\r\n if (!pose) {\r\n this._setTrackingState(WebXRTrackingState.NOT_TRACKING);\r\n return;\r\n }\r\n\r\n // Set the tracking state. if it didn't change it is a no-op\r\n const trackingState = pose.emulatedPosition ? WebXRTrackingState.TRACKING_LOST : WebXRTrackingState.TRACKING;\r\n this._setTrackingState(trackingState);\r\n\r\n // check min/max Z and update if not the same as in cache\r\n if (this.minZ !== this._cache.minZ || this.maxZ !== this._cache.maxZ) {\r\n const xrRenderState: XRRenderStateInit = {\r\n // if maxZ is 0 it should be \"Infinity\", but it doesn't work with the WebXR API. Setting to a large number.\r\n depthFar: this.maxZ || 10000,\r\n depthNear: this.minZ,\r\n };\r\n\r\n this._xrSessionManager.updateRenderState(xrRenderState);\r\n this._cache.minZ = this.minZ;\r\n this._cache.maxZ = this.maxZ;\r\n }\r\n\r\n if (pose.transform) {\r\n const orientation = pose.transform.orientation;\r\n if (pose.transform.orientation.x === undefined) {\r\n // Babylon native polyfill can return an undefined orientation value\r\n // When not initialized\r\n return;\r\n }\r\n const pos = pose.transform.position;\r\n this._referencedPosition.set(pos.x, pos.y, pos.z);\r\n\r\n this._referenceQuaternion.set(orientation.x, orientation.y, orientation.z, orientation.w);\r\n if (!this._scene.useRightHandedSystem) {\r\n this._referencedPosition.z *= -1;\r\n this._referenceQuaternion.z *= -1;\r\n this._referenceQuaternion.w *= -1;\r\n }\r\n\r\n if (this._firstFrame) {\r\n this._firstFrame = false;\r\n // we have the XR reference, now use this to find the offset to get the camera to be\r\n // in the right position\r\n\r\n // set the height to correlate to the current height\r\n this.position.y += this._referencedPosition.y;\r\n // avoid using the head rotation on the first frame.\r\n this._referenceQuaternion.copyFromFloats(0, 0, 0, 1);\r\n } else {\r\n // update position and rotation as reference\r\n this.rotationQuaternion.copyFrom(this._referenceQuaternion);\r\n this.position.copyFrom(this._referencedPosition);\r\n }\r\n }\r\n\r\n // Update camera rigs\r\n if (this.rigCameras.length !== pose.views.length) {\r\n this._updateNumberOfRigCameras(pose.views.length);\r\n }\r\n\r\n pose.views.forEach((view: XRView, i: number) => {\r\n const currentRig = this.rigCameras[i];\r\n // update right and left, where applicable\r\n if (!currentRig.isLeftCamera && !currentRig.isRightCamera) {\r\n if (view.eye === \"right\") {\r\n currentRig._isRightCamera = true;\r\n } else if (view.eye === \"left\") {\r\n currentRig._isLeftCamera = true;\r\n }\r\n }\r\n // Update view/projection matrix\r\n const pos = view.transform.position;\r\n const orientation = view.transform.orientation;\r\n\r\n currentRig.parent = this.parent;\r\n\r\n currentRig.position.set(pos.x, pos.y, pos.z);\r\n currentRig.rotationQuaternion.set(orientation.x, orientation.y, orientation.z, orientation.w);\r\n if (!this._scene.useRightHandedSystem) {\r\n currentRig.position.z *= -1;\r\n currentRig.rotationQuaternion.z *= -1;\r\n currentRig.rotationQuaternion.w *= -1;\r\n } else {\r\n currentRig.rotationQuaternion.multiplyInPlace(this._rotate180);\r\n }\r\n Matrix.FromFloat32ArrayToRefScaled(view.projectionMatrix, 0, 1, currentRig._projectionMatrix);\r\n\r\n if (!this._scene.useRightHandedSystem) {\r\n currentRig._projectionMatrix.toggleProjectionMatrixHandInPlace();\r\n }\r\n\r\n // first camera?\r\n if (i === 0) {\r\n this._projectionMatrix.copyFrom(currentRig._projectionMatrix);\r\n }\r\n\r\n const renderTargetTexture = this._xrSessionManager.getRenderTargetTextureForView(view);\r\n this._renderingMultiview = renderTargetTexture?._texture?.isMultiview || false;\r\n if (this._renderingMultiview) {\r\n // For multiview, the render target texture is the same per-view (just the slice index is different),\r\n // so we only need to set the output render target once for the rig parent.\r\n if (i == 0) {\r\n this._xrSessionManager.trySetViewportForView(this.viewport, view);\r\n this.outputRenderTarget = renderTargetTexture;\r\n }\r\n } else {\r\n // Update viewport\r\n this._xrSessionManager.trySetViewportForView(currentRig.viewport, view);\r\n\r\n // Set cameras to render to the session's render target\r\n currentRig.outputRenderTarget = renderTargetTexture || this._xrSessionManager.getRenderTargetTextureForView(view);\r\n }\r\n\r\n // Replicate parent rig camera behavior\r\n currentRig.layerMask = this.layerMask;\r\n });\r\n }\r\n\r\n private _updateNumberOfRigCameras(viewCount = 1) {\r\n while (this.rigCameras.length < viewCount) {\r\n const newCamera = new TargetCamera(\"XR-RigCamera: \" + this.rigCameras.length, Vector3.Zero(), this.getScene());\r\n newCamera.minZ = 0.1;\r\n newCamera.rotationQuaternion = new Quaternion();\r\n newCamera.updateUpVectorFromRotation = true;\r\n newCamera.isRigCamera = true;\r\n newCamera.rigParent = this;\r\n // do not compute projection matrix, provided by XR\r\n newCamera.freezeProjectionMatrix();\r\n this.rigCameras.push(newCamera);\r\n }\r\n while (this.rigCameras.length > viewCount) {\r\n const removedCamera = this.rigCameras.pop();\r\n if (removedCamera) {\r\n removedCamera.dispose();\r\n }\r\n }\r\n }\r\n\r\n private _updateReferenceSpace() {\r\n // were position & rotation updated OUTSIDE of the xr update loop\r\n if (!this.position.equals(this._referencedPosition) || !this.rotationQuaternion.equals(this._referenceQuaternion)) {\r\n const referencedMat = TmpVectors.Matrix[0];\r\n const poseMat = TmpVectors.Matrix[1];\r\n const transformMat = TmpVectors.Matrix[2];\r\n\r\n Matrix.ComposeToRef(WebXRCamera._ScaleReadOnly, this._referenceQuaternion, this._referencedPosition, referencedMat);\r\n Matrix.ComposeToRef(WebXRCamera._ScaleReadOnly, this.rotationQuaternion, this.position, poseMat);\r\n referencedMat.invert().multiplyToRef(poseMat, transformMat);\r\n transformMat.invert();\r\n\r\n if (!this._scene.useRightHandedSystem) {\r\n transformMat.toggleModelMatrixHandInPlace();\r\n }\r\n\r\n transformMat.decompose(undefined, this._referenceQuaternion, this._referencedPosition);\r\n const transform = new XRRigidTransform(\r\n {\r\n x: this._referencedPosition.x,\r\n y: this._referencedPosition.y,\r\n z: this._referencedPosition.z,\r\n },\r\n {\r\n x: this._referenceQuaternion.x,\r\n y: this._referenceQuaternion.y,\r\n z: this._referenceQuaternion.z,\r\n w: this._referenceQuaternion.w,\r\n }\r\n );\r\n this._xrSessionManager.referenceSpace = this._xrSessionManager.referenceSpace.getOffsetReferenceSpace(transform);\r\n }\r\n }\r\n}\r\n","import type { WebXRSessionManager } from \"./webXRSessionManager\";\r\nimport type { IDisposable } from \"../scene\";\r\nimport { Tools } from \"../Misc/tools\";\r\n\r\n/**\r\n * Defining the interface required for a (webxr) feature\r\n */\r\nexport interface IWebXRFeature extends IDisposable {\r\n /**\r\n * Is this feature attached\r\n */\r\n attached: boolean;\r\n /**\r\n * Should auto-attach be disabled?\r\n */\r\n disableAutoAttach: boolean;\r\n\r\n /**\r\n * Attach the feature to the session\r\n * Will usually be called by the features manager\r\n *\r\n * @param force should attachment be forced (even when already attached)\r\n * @returns true if successful.\r\n */\r\n attach(force?: boolean): boolean;\r\n /**\r\n * Detach the feature from the session\r\n * Will usually be called by the features manager\r\n *\r\n * @returns true if successful.\r\n */\r\n detach(): boolean;\r\n\r\n /**\r\n * This function will be executed during before enabling the feature and can be used to not-allow enabling it.\r\n * Note that at this point the session has NOT started, so this is purely checking if the browser supports it\r\n *\r\n * @returns whether or not the feature is compatible in this environment\r\n */\r\n isCompatible(): boolean;\r\n\r\n /**\r\n * Was this feature disposed;\r\n */\r\n isDisposed: boolean;\r\n\r\n /**\r\n * The name of the native xr feature name, if applicable (like anchor, hit-test, or hand-tracking)\r\n */\r\n xrNativeFeatureName?: string;\r\n\r\n /**\r\n * A list of (Babylon WebXR) features this feature depends on\r\n */\r\n dependsOn?: string[];\r\n\r\n /**\r\n * If this feature requires to extend the XRSessionInit object, this function will return the partial XR session init object\r\n */\r\n getXRSessionInitExtension?: () => Promise>;\r\n}\r\n\r\n/**\r\n * A list of the currently available features without referencing them\r\n */\r\nexport class WebXRFeatureName {\r\n /**\r\n * The name of the anchor system feature\r\n */\r\n public static readonly ANCHOR_SYSTEM = \"xr-anchor-system\";\r\n /**\r\n * The name of the background remover feature\r\n */\r\n public static readonly BACKGROUND_REMOVER = \"xr-background-remover\";\r\n /**\r\n * The name of the hit test feature\r\n */\r\n public static readonly HIT_TEST = \"xr-hit-test\";\r\n /**\r\n * The name of the mesh detection feature\r\n */\r\n public static readonly MESH_DETECTION = \"xr-mesh-detection\";\r\n /**\r\n * physics impostors for xr controllers feature\r\n */\r\n public static readonly PHYSICS_CONTROLLERS = \"xr-physics-controller\";\r\n /**\r\n * The name of the plane detection feature\r\n */\r\n public static readonly PLANE_DETECTION = \"xr-plane-detection\";\r\n /**\r\n * The name of the pointer selection feature\r\n */\r\n public static readonly POINTER_SELECTION = \"xr-controller-pointer-selection\";\r\n /**\r\n * The name of the teleportation feature\r\n */\r\n public static readonly TELEPORTATION = \"xr-controller-teleportation\";\r\n /**\r\n * The name of the feature points feature.\r\n */\r\n public static readonly FEATURE_POINTS = \"xr-feature-points\";\r\n /**\r\n * The name of the hand tracking feature.\r\n */\r\n public static readonly HAND_TRACKING = \"xr-hand-tracking\";\r\n /**\r\n * The name of the image tracking feature\r\n */\r\n public static readonly IMAGE_TRACKING = \"xr-image-tracking\";\r\n /**\r\n * The name of the near interaction feature\r\n */\r\n public static readonly NEAR_INTERACTION = \"xr-near-interaction\";\r\n /**\r\n * The name of the DOM overlay feature\r\n */\r\n public static readonly DOM_OVERLAY = \"xr-dom-overlay\";\r\n /**\r\n * The name of the movement feature\r\n */\r\n public static readonly MOVEMENT = \"xr-controller-movement\";\r\n /**\r\n * The name of the light estimation feature\r\n */\r\n public static readonly LIGHT_ESTIMATION = \"xr-light-estimation\";\r\n /**\r\n * The name of the eye tracking feature\r\n */\r\n public static readonly EYE_TRACKING = \"xr-eye-tracking\";\r\n /**\r\n * The name of the walking locomotion feature\r\n */\r\n public static readonly WALKING_LOCOMOTION = \"xr-walking-locomotion\";\r\n /**\r\n * The name of the composition layers feature\r\n */\r\n public static readonly LAYERS = \"xr-layers\";\r\n /**\r\n * The name of the depth sensing feature\r\n */\r\n public static readonly DEPTH_SENSING = \"xr-depth-sensing\";\r\n /**\r\n * The name of the WebXR Space Warp feature\r\n */\r\n public static readonly SPACE_WARP = \"xr-space-warp\";\r\n}\r\n\r\n/**\r\n * Defining the constructor of a feature. Used to register the modules.\r\n */\r\nexport type WebXRFeatureConstructor = (xrSessionManager: WebXRSessionManager, options?: any) => () => IWebXRFeature;\r\n\r\n/**\r\n * The WebXR features manager is responsible of enabling or disabling features required for the current XR session.\r\n * It is mainly used in AR sessions.\r\n *\r\n * A feature can have a version that is defined by Babylon (and does not correspond with the webxr version).\r\n */\r\nexport class WebXRFeaturesManager implements IDisposable {\r\n private static readonly _AvailableFeatures: {\r\n [name: string]: {\r\n stable: number;\r\n latest: number;\r\n [version: number]: WebXRFeatureConstructor;\r\n };\r\n } = {};\r\n\r\n private _features: {\r\n [name: string]: {\r\n featureImplementation: IWebXRFeature;\r\n version: number;\r\n enabled: boolean;\r\n required: boolean;\r\n };\r\n } = {};\r\n\r\n /**\r\n * The key is the feature to check and the value is the feature that conflicts.\r\n */\r\n private static readonly _ConflictingFeatures: { [key: string]: string } = {\r\n [WebXRFeatureName.TELEPORTATION]: WebXRFeatureName.MOVEMENT,\r\n [WebXRFeatureName.MOVEMENT]: WebXRFeatureName.TELEPORTATION,\r\n };\r\n\r\n /**\r\n * constructs a new features manages.\r\n *\r\n * @param _xrSessionManager an instance of WebXRSessionManager\r\n */\r\n constructor(private _xrSessionManager: WebXRSessionManager) {\r\n // when session starts / initialized - attach\r\n this._xrSessionManager.onXRSessionInit.add(() => {\r\n this.getEnabledFeatures().forEach((featureName) => {\r\n const feature = this._features[featureName];\r\n if (feature.enabled && !feature.featureImplementation.attached && !feature.featureImplementation.disableAutoAttach) {\r\n this.attachFeature(featureName);\r\n }\r\n });\r\n });\r\n\r\n // when session ends - detach\r\n this._xrSessionManager.onXRSessionEnded.add(() => {\r\n this.getEnabledFeatures().forEach((featureName) => {\r\n const feature = this._features[featureName];\r\n if (feature.enabled && feature.featureImplementation.attached) {\r\n // detach, but don't disable!\r\n this.detachFeature(featureName);\r\n }\r\n });\r\n });\r\n }\r\n\r\n /**\r\n * Used to register a module. After calling this function a developer can use this feature in the scene.\r\n * Mainly used internally.\r\n *\r\n * @param featureName the name of the feature to register\r\n * @param constructorFunction the function used to construct the module\r\n * @param version the (babylon) version of the module\r\n * @param stable is that a stable version of this module\r\n */\r\n public static AddWebXRFeature(featureName: string, constructorFunction: WebXRFeatureConstructor, version: number = 1, stable: boolean = false) {\r\n this._AvailableFeatures[featureName] = this._AvailableFeatures[featureName] || { latest: version };\r\n if (version > this._AvailableFeatures[featureName].latest) {\r\n this._AvailableFeatures[featureName].latest = version;\r\n }\r\n if (stable) {\r\n this._AvailableFeatures[featureName].stable = version;\r\n }\r\n this._AvailableFeatures[featureName][version] = constructorFunction;\r\n }\r\n\r\n /**\r\n * Returns a constructor of a specific feature.\r\n *\r\n * @param featureName the name of the feature to construct\r\n * @param version the version of the feature to load\r\n * @param xrSessionManager the xrSessionManager. Used to construct the module\r\n * @param options optional options provided to the module.\r\n * @returns a function that, when called, will return a new instance of this feature\r\n */\r\n public static ConstructFeature(featureName: string, version: number = 1, xrSessionManager: WebXRSessionManager, options?: any): () => IWebXRFeature {\r\n const constructorFunction = this._AvailableFeatures[featureName][version];\r\n if (!constructorFunction) {\r\n // throw an error? return nothing?\r\n throw new Error(\"feature not found\");\r\n }\r\n\r\n return constructorFunction(xrSessionManager, options);\r\n }\r\n\r\n /**\r\n * Can be used to return the list of features currently registered\r\n *\r\n * @returns an Array of available features\r\n */\r\n public static GetAvailableFeatures() {\r\n return Object.keys(this._AvailableFeatures);\r\n }\r\n\r\n /**\r\n * Gets the versions available for a specific feature\r\n * @param featureName the name of the feature\r\n * @returns an array with the available versions\r\n */\r\n public static GetAvailableVersions(featureName: string) {\r\n return Object.keys(this._AvailableFeatures[featureName]);\r\n }\r\n\r\n /**\r\n * Return the latest unstable version of this feature\r\n * @param featureName the name of the feature to search\r\n * @returns the version number. if not found will return -1\r\n */\r\n public static GetLatestVersionOfFeature(featureName: string): number {\r\n return (this._AvailableFeatures[featureName] && this._AvailableFeatures[featureName].latest) || -1;\r\n }\r\n\r\n /**\r\n * Return the latest stable version of this feature\r\n * @param featureName the name of the feature to search\r\n * @returns the version number. if not found will return -1\r\n */\r\n public static GetStableVersionOfFeature(featureName: string): number {\r\n return (this._AvailableFeatures[featureName] && this._AvailableFeatures[featureName].stable) || -1;\r\n }\r\n\r\n /**\r\n * Attach a feature to the current session. Mainly used when session started to start the feature effect.\r\n * Can be used during a session to start a feature\r\n * @param featureName the name of feature to attach\r\n */\r\n public attachFeature(featureName: string) {\r\n const feature = this._features[featureName];\r\n if (feature && feature.enabled && !feature.featureImplementation.attached) {\r\n feature.featureImplementation.attach();\r\n }\r\n }\r\n\r\n /**\r\n * Can be used inside a session or when the session ends to detach a specific feature\r\n * @param featureName the name of the feature to detach\r\n */\r\n public detachFeature(featureName: string) {\r\n const feature = this._features[featureName];\r\n if (feature && feature.featureImplementation.attached) {\r\n feature.featureImplementation.detach();\r\n }\r\n }\r\n\r\n /**\r\n * Used to disable an already-enabled feature\r\n * The feature will be disposed and will be recreated once enabled.\r\n * @param featureName the feature to disable\r\n * @returns true if disable was successful\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public disableFeature(featureName: string | { Name: string }): boolean {\r\n const name = typeof featureName === \"string\" ? featureName : featureName.Name;\r\n const feature = this._features[name];\r\n if (feature && feature.enabled) {\r\n feature.enabled = false;\r\n this.detachFeature(name);\r\n feature.featureImplementation.dispose();\r\n delete this._features[name];\r\n return true;\r\n }\r\n return false;\r\n }\r\n\r\n /**\r\n * dispose this features manager\r\n */\r\n public dispose(): void {\r\n this.getEnabledFeatures().forEach((feature) => {\r\n this.disableFeature(feature);\r\n });\r\n }\r\n\r\n /**\r\n * Enable a feature using its name and a version. This will enable it in the scene, and will be responsible to attach it when the session starts.\r\n * If used twice, the old version will be disposed and a new one will be constructed. This way you can re-enable with different configuration.\r\n *\r\n * @param featureName the name of the feature to load or the class of the feature\r\n * @param version optional version to load. if not provided the latest version will be enabled\r\n * @param moduleOptions options provided to the module. Ses the module documentation / constructor\r\n * @param attachIfPossible if set to true (default) the feature will be automatically attached, if it is currently possible\r\n * @param required is this feature required to the app. If set to true the session init will fail if the feature is not available.\r\n * @returns a new constructed feature or throws an error if feature not found or conflicts with another enabled feature.\r\n */\r\n public enableFeature(\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n featureName: string | { Name: string },\r\n version: number | string = \"latest\",\r\n moduleOptions: any = {},\r\n attachIfPossible: boolean = true,\r\n required: boolean = true\r\n ): IWebXRFeature {\r\n const name = typeof featureName === \"string\" ? featureName : featureName.Name;\r\n let versionToLoad = 0;\r\n if (typeof version === \"string\") {\r\n if (!version) {\r\n throw new Error(`Error in provided version - ${name} (${version})`);\r\n }\r\n if (version === \"stable\") {\r\n versionToLoad = WebXRFeaturesManager.GetStableVersionOfFeature(name);\r\n } else if (version === \"latest\") {\r\n versionToLoad = WebXRFeaturesManager.GetLatestVersionOfFeature(name);\r\n } else {\r\n // try loading the number the string represents\r\n versionToLoad = +version;\r\n }\r\n if (versionToLoad === -1 || isNaN(versionToLoad)) {\r\n throw new Error(`feature not found - ${name} (${version})`);\r\n }\r\n } else {\r\n versionToLoad = version;\r\n }\r\n\r\n // check if there is a feature conflict\r\n const conflictingFeature = WebXRFeaturesManager._ConflictingFeatures[name];\r\n if (conflictingFeature !== undefined && this.getEnabledFeatures().indexOf(conflictingFeature) !== -1) {\r\n throw new Error(`Feature ${name} cannot be enabled while ${conflictingFeature} is enabled.`);\r\n }\r\n\r\n // check if already initialized\r\n const feature = this._features[name];\r\n const constructFunction = WebXRFeaturesManager.ConstructFeature(name, versionToLoad, this._xrSessionManager, moduleOptions);\r\n if (!constructFunction) {\r\n // report error?\r\n throw new Error(`feature not found - ${name}`);\r\n }\r\n\r\n /* If the feature is already enabled, detach and dispose it, and create a new one */\r\n if (feature) {\r\n this.disableFeature(name);\r\n }\r\n\r\n const constructed = constructFunction();\r\n if (constructed.dependsOn) {\r\n const dependentsFound = constructed.dependsOn.every((featureName) => !!this._features[featureName]);\r\n if (!dependentsFound) {\r\n throw new Error(`Dependant features missing. Make sure the following features are enabled - ${constructed.dependsOn.join(\", \")}`);\r\n }\r\n }\r\n if (constructed.isCompatible()) {\r\n this._features[name] = {\r\n featureImplementation: constructed,\r\n enabled: true,\r\n version: versionToLoad,\r\n required,\r\n };\r\n\r\n if (attachIfPossible) {\r\n // if session started already, request and enable\r\n if (this._xrSessionManager.session && !this._features[name].featureImplementation.attached) {\r\n // enable feature\r\n this.attachFeature(name);\r\n }\r\n } else {\r\n // disable auto-attach when session starts\r\n this._features[name].featureImplementation.disableAutoAttach = true;\r\n }\r\n\r\n return this._features[name].featureImplementation;\r\n } else {\r\n if (required) {\r\n throw new Error(\"required feature not compatible\");\r\n } else {\r\n Tools.Warn(`Feature ${name} not compatible with the current environment/browser and was not enabled.`);\r\n return constructed;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * get the implementation of an enabled feature.\r\n * @param featureName the name of the feature to load\r\n * @returns the feature class, if found\r\n */\r\n public getEnabledFeature(featureName: string): IWebXRFeature {\r\n return this._features[featureName] && this._features[featureName].featureImplementation;\r\n }\r\n\r\n /**\r\n * Get the list of enabled features\r\n * @returns an array of enabled features\r\n */\r\n public getEnabledFeatures() {\r\n return Object.keys(this._features);\r\n }\r\n\r\n /**\r\n * This function will extend the session creation configuration object with enabled features.\r\n * If, for example, the anchors feature is enabled, it will be automatically added to the optional or required features list,\r\n * according to the defined \"required\" variable, provided during enableFeature call\r\n * @param xrSessionInit the xr Session init object to extend\r\n *\r\n * @returns an extended XRSessionInit object\r\n */\r\n public async _extendXRSessionInitObject(xrSessionInit: XRSessionInit): Promise {\r\n const enabledFeatures = this.getEnabledFeatures();\r\n for (const featureName of enabledFeatures) {\r\n const feature = this._features[featureName];\r\n const nativeName = feature.featureImplementation.xrNativeFeatureName;\r\n if (nativeName) {\r\n if (feature.required) {\r\n xrSessionInit.requiredFeatures = xrSessionInit.requiredFeatures || [];\r\n if (xrSessionInit.requiredFeatures.indexOf(nativeName) === -1) {\r\n xrSessionInit.requiredFeatures.push(nativeName);\r\n }\r\n } else {\r\n xrSessionInit.optionalFeatures = xrSessionInit.optionalFeatures || [];\r\n if (xrSessionInit.optionalFeatures.indexOf(nativeName) === -1) {\r\n xrSessionInit.optionalFeatures.push(nativeName);\r\n }\r\n }\r\n }\r\n if (feature.featureImplementation.getXRSessionInitExtension) {\r\n const extended = await feature.featureImplementation.getXRSessionInitExtension();\r\n xrSessionInit = {\r\n ...xrSessionInit,\r\n ...extended,\r\n };\r\n }\r\n }\r\n return xrSessionInit;\r\n }\r\n}\r\n","import { FreeCamera } from \"./freeCamera\";\r\nimport type { FreeCameraTouchInput } from \"../Cameras/Inputs/freeCameraTouchInput\";\r\nimport type { FreeCameraMouseInput } from \"../Cameras/Inputs/freeCameraMouseInput\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Vector3 } from \"../Maths/math.vector\";\r\nimport { Node } from \"../node\";\r\n\r\nNode.AddNodeConstructor(\"TouchCamera\", (name, scene) => {\r\n return () => new TouchCamera(name, Vector3.Zero(), scene);\r\n});\r\n\r\n/**\r\n * This represents a FPS type of camera controlled by touch.\r\n * This is like a universal camera minus the Gamepad controls.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#universal-camera\r\n */\r\nexport class TouchCamera extends FreeCamera {\r\n /**\r\n * Defines the touch sensibility for rotation.\r\n * The higher the faster.\r\n */\r\n public get touchAngularSensibility(): number {\r\n const touch = this.inputs.attached[\"touch\"];\r\n if (touch) {\r\n return touch.touchAngularSensibility;\r\n }\r\n\r\n return 0;\r\n }\r\n\r\n public set touchAngularSensibility(value: number) {\r\n const touch = this.inputs.attached[\"touch\"];\r\n if (touch) {\r\n touch.touchAngularSensibility = value;\r\n }\r\n }\r\n\r\n /**\r\n * Defines the touch sensibility for move.\r\n * The higher the faster.\r\n */\r\n public get touchMoveSensibility(): number {\r\n const touch = this.inputs.attached[\"touch\"];\r\n if (touch) {\r\n return touch.touchMoveSensibility;\r\n }\r\n\r\n return 0;\r\n }\r\n\r\n public set touchMoveSensibility(value: number) {\r\n const touch = this.inputs.attached[\"touch\"];\r\n if (touch) {\r\n touch.touchMoveSensibility = value;\r\n }\r\n }\r\n\r\n /**\r\n * Instantiates a new touch camera.\r\n * This represents a FPS type of camera controlled by touch.\r\n * This is like a universal camera minus the Gamepad controls.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#universal-camera\r\n * @param name Define the name of the camera in the scene\r\n * @param position Define the start position of the camera in the scene\r\n * @param scene Define the scene the camera belongs to\r\n */\r\n constructor(name: string, position: Vector3, scene?: Scene) {\r\n super(name, position, scene);\r\n this.inputs.addTouch();\r\n\r\n this._setupInputs();\r\n }\r\n\r\n /**\r\n * Gets the current object class name.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"TouchCamera\";\r\n }\r\n\r\n /** @internal */\r\n public _setupInputs() {\r\n const touch = this.inputs.attached[\"touch\"];\r\n const mouse = this.inputs.attached[\"mouse\"];\r\n if (mouse) {\r\n mouse.touchEnabled = false;\r\n } else {\r\n touch.allowMouse = true;\r\n }\r\n }\r\n}\r\n","import { TouchCamera } from \"./touchCamera\";\r\nimport { Node } from \"../node\";\r\nimport type { FreeCameraGamepadInput } from \"../Cameras/Inputs/freeCameraGamepadInput\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Vector3 } from \"../Maths/math.vector\";\r\nimport { Camera } from \"./camera\";\r\n\r\nimport \"../Gamepads/gamepadSceneComponent\";\r\n\r\nNode.AddNodeConstructor(\"FreeCamera\", (name, scene) => {\r\n // Forcing to use the Universal camera\r\n return () => new UniversalCamera(name, Vector3.Zero(), scene);\r\n});\r\n\r\n/**\r\n * The Universal Camera is the one to choose for first person shooter type games, and works with all the keyboard, mouse, touch and gamepads. This replaces the earlier Free Camera,\r\n * which still works and will still be found in many Playgrounds.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#universal-camera\r\n */\r\nexport class UniversalCamera extends TouchCamera {\r\n /**\r\n * Defines the gamepad rotation sensibility.\r\n * This is the threshold from when rotation starts to be accounted for to prevent jittering.\r\n */\r\n public get gamepadAngularSensibility(): number {\r\n const gamepad = this.inputs.attached[\"gamepad\"];\r\n if (gamepad) {\r\n return gamepad.gamepadAngularSensibility;\r\n }\r\n\r\n return 0;\r\n }\r\n\r\n public set gamepadAngularSensibility(value: number) {\r\n const gamepad = this.inputs.attached[\"gamepad\"];\r\n if (gamepad) {\r\n gamepad.gamepadAngularSensibility = value;\r\n }\r\n }\r\n\r\n /**\r\n * Defines the gamepad move sensibility.\r\n * This is the threshold from when moving starts to be accounted for to prevent jittering.\r\n */\r\n public get gamepadMoveSensibility(): number {\r\n const gamepad = this.inputs.attached[\"gamepad\"];\r\n if (gamepad) {\r\n return gamepad.gamepadMoveSensibility;\r\n }\r\n\r\n return 0;\r\n }\r\n\r\n public set gamepadMoveSensibility(value: number) {\r\n const gamepad = this.inputs.attached[\"gamepad\"];\r\n if (gamepad) {\r\n gamepad.gamepadMoveSensibility = value;\r\n }\r\n }\r\n\r\n /**\r\n * The Universal Camera is the one to choose for first person shooter type games, and works with all the keyboard, mouse, touch and gamepads. This replaces the earlier Free Camera,\r\n * which still works and will still be found in many Playgrounds.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/camera_introduction#universal-camera\r\n * @param name Define the name of the camera in the scene\r\n * @param position Define the start position of the camera in the scene\r\n * @param scene Define the scene the camera belongs to\r\n */\r\n constructor(name: string, position: Vector3, scene?: Scene) {\r\n super(name, position, scene);\r\n this.inputs.addGamepad();\r\n }\r\n\r\n /**\r\n * Gets the current object class name.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"UniversalCamera\";\r\n }\r\n}\r\n\r\nCamera._CreateDefaultParsedCamera = (name: string, scene: Scene) => {\r\n return new UniversalCamera(name, Vector3.Zero(), scene);\r\n};\r\n","import type { Nullable } from \"../types\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport type { IDisposable, Scene } from \"../scene\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\nimport { WebXRSessionManager } from \"./webXRSessionManager\";\r\nimport { WebXRCamera } from \"./webXRCamera\";\r\nimport type { WebXRRenderTarget } from \"./webXRTypes\";\r\nimport { WebXRState } from \"./webXRTypes\";\r\nimport { WebXRFeatureName, WebXRFeaturesManager } from \"./webXRFeaturesManager\";\r\nimport { Logger } from \"../Misc/logger\";\r\nimport { UniversalCamera } from \"../Cameras/universalCamera\";\r\nimport { Quaternion, Vector3 } from \"../Maths/math.vector\";\r\n\r\n/**\r\n * Options for setting up XR spectator camera.\r\n */\r\nexport interface WebXRSpectatorModeOption {\r\n /**\r\n * Expected refresh rate (frames per sec) for a spectator camera.\r\n */\r\n fps?: number;\r\n /**\r\n * The index of rigCameras array in a WebXR camera.\r\n */\r\n preferredCameraIndex?: number;\r\n}\r\n\r\n/**\r\n * Base set of functionality needed to create an XR experience (WebXRSessionManager, Camera, StateManagement, etc.)\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/webXR/webXRExperienceHelpers\r\n */\r\nexport class WebXRExperienceHelper implements IDisposable {\r\n private _nonVRCamera: Nullable = null;\r\n private _attachedToElement: boolean = false;\r\n private _spectatorCamera: Nullable = null;\r\n private _originalSceneAutoClear = true;\r\n private _supported = false;\r\n private _spectatorMode = false;\r\n private _lastTimestamp = 0;\r\n\r\n /**\r\n * Camera used to render xr content\r\n */\r\n public camera: WebXRCamera;\r\n /** A features manager for this xr session */\r\n public featuresManager: WebXRFeaturesManager;\r\n /**\r\n * Observers registered here will be triggered after the camera's initial transformation is set\r\n * This can be used to set a different ground level or an extra rotation.\r\n *\r\n * Note that ground level is considered to be at 0. The height defined by the XR camera will be added\r\n * to the position set after this observable is done executing.\r\n */\r\n public onInitialXRPoseSetObservable = new Observable();\r\n /**\r\n * Fires when the state of the experience helper has changed\r\n */\r\n public onStateChangedObservable = new Observable();\r\n /** Session manager used to keep track of xr session */\r\n public sessionManager: WebXRSessionManager;\r\n /**\r\n * The current state of the XR experience (eg. transitioning, in XR or not in XR)\r\n */\r\n public state: WebXRState = WebXRState.NOT_IN_XR;\r\n\r\n /**\r\n * Creates a WebXRExperienceHelper\r\n * @param _scene The scene the helper should be created in\r\n */\r\n private constructor(private _scene: Scene) {\r\n this.sessionManager = new WebXRSessionManager(_scene);\r\n this.camera = new WebXRCamera(\"webxr\", _scene, this.sessionManager);\r\n this.featuresManager = new WebXRFeaturesManager(this.sessionManager);\r\n\r\n _scene.onDisposeObservable.addOnce(() => {\r\n this.dispose();\r\n });\r\n }\r\n\r\n /**\r\n * Creates the experience helper\r\n * @param scene the scene to attach the experience helper to\r\n * @returns a promise for the experience helper\r\n */\r\n public static CreateAsync(scene: Scene): Promise {\r\n const helper = new WebXRExperienceHelper(scene);\r\n return helper.sessionManager\r\n .initializeAsync()\r\n .then(() => {\r\n helper._supported = true;\r\n return helper;\r\n })\r\n .catch((e) => {\r\n helper._setState(WebXRState.NOT_IN_XR);\r\n helper.dispose();\r\n throw e;\r\n });\r\n }\r\n\r\n /**\r\n * Disposes of the experience helper\r\n */\r\n public dispose() {\r\n this.exitXRAsync();\r\n this.camera.dispose();\r\n this.onStateChangedObservable.clear();\r\n this.onInitialXRPoseSetObservable.clear();\r\n this.sessionManager.dispose();\r\n this._spectatorCamera?.dispose();\r\n if (this._nonVRCamera) {\r\n this._scene.activeCamera = this._nonVRCamera;\r\n }\r\n }\r\n\r\n /**\r\n * Enters XR mode (This must be done within a user interaction in most browsers eg. button click)\r\n * @param sessionMode options for the XR session\r\n * @param referenceSpaceType frame of reference of the XR session\r\n * @param renderTarget the output canvas that will be used to enter XR mode\r\n * @param sessionCreationOptions optional XRSessionInit object to init the session with\r\n * @returns promise that resolves after xr mode has entered\r\n */\r\n public async enterXRAsync(\r\n sessionMode: XRSessionMode,\r\n referenceSpaceType: XRReferenceSpaceType,\r\n renderTarget: WebXRRenderTarget = this.sessionManager.getWebXRRenderTarget(),\r\n sessionCreationOptions: XRSessionInit = {}\r\n ): Promise {\r\n if (!this._supported) {\r\n throw \"WebXR not supported in this browser or environment\";\r\n }\r\n this._setState(WebXRState.ENTERING_XR);\r\n if (referenceSpaceType !== \"viewer\" && referenceSpaceType !== \"local\") {\r\n sessionCreationOptions.optionalFeatures = sessionCreationOptions.optionalFeatures || [];\r\n sessionCreationOptions.optionalFeatures.push(referenceSpaceType);\r\n }\r\n sessionCreationOptions = await this.featuresManager._extendXRSessionInitObject(sessionCreationOptions);\r\n // we currently recommend \"unbounded\" space in AR (#7959)\r\n if (sessionMode === \"immersive-ar\" && referenceSpaceType !== \"unbounded\") {\r\n Logger.Warn(\"We recommend using 'unbounded' reference space type when using 'immersive-ar' session mode\");\r\n }\r\n // make sure that the session mode is supported\r\n try {\r\n await this.sessionManager.initializeSessionAsync(sessionMode, sessionCreationOptions);\r\n await this.sessionManager.setReferenceSpaceTypeAsync(referenceSpaceType);\r\n const baseLayer = await renderTarget.initializeXRLayerAsync(this.sessionManager.session);\r\n\r\n const xrRenderState: XRRenderStateInit = {\r\n // if maxZ is 0 it should be \"Infinity\", but it doesn't work with the WebXR API. Setting to a large number.\r\n depthFar: this.camera.maxZ || 10000,\r\n depthNear: this.camera.minZ,\r\n };\r\n\r\n // The layers feature will have already initialized the xr session's layers on session init.\r\n if (!this.featuresManager.getEnabledFeature(WebXRFeatureName.LAYERS)) {\r\n xrRenderState.baseLayer = baseLayer;\r\n }\r\n\r\n this.sessionManager.updateRenderState(xrRenderState);\r\n // run the render loop\r\n this.sessionManager.runXRRenderLoop();\r\n // Cache pre xr scene settings\r\n this._originalSceneAutoClear = this._scene.autoClear;\r\n this._nonVRCamera = this._scene.activeCamera;\r\n this._attachedToElement = !!this._nonVRCamera?.inputs?.attachedToElement;\r\n this._nonVRCamera?.detachControl();\r\n\r\n this._scene.activeCamera = this.camera;\r\n // do not compensate when AR session is used\r\n if (sessionMode !== \"immersive-ar\") {\r\n this._nonXRToXRCamera();\r\n } else {\r\n // Kept here, TODO - check if needed\r\n this._scene.autoClear = false;\r\n this.camera.compensateOnFirstFrame = false;\r\n // reset the camera's position to the origin\r\n this.camera.position.set(0, 0, 0);\r\n this.camera.rotationQuaternion.set(0, 0, 0, 1);\r\n this.onInitialXRPoseSetObservable.notifyObservers(this.camera);\r\n }\r\n\r\n this.sessionManager.onXRSessionEnded.addOnce(() => {\r\n // when using the back button and not the exit button (default on mobile), the session is ending but the EXITING state was not set\r\n if (this.state !== WebXRState.EXITING_XR) {\r\n this._setState(WebXRState.EXITING_XR);\r\n }\r\n // Reset camera rigs output render target to ensure sessions render target is not drawn after it ends\r\n this.camera.rigCameras.forEach((c) => {\r\n c.outputRenderTarget = null;\r\n });\r\n\r\n // Restore scene settings\r\n this._scene.autoClear = this._originalSceneAutoClear;\r\n this._scene.activeCamera = this._nonVRCamera;\r\n if (this._attachedToElement && this._nonVRCamera) {\r\n this._nonVRCamera.attachControl(!!this._nonVRCamera.inputs.noPreventDefault);\r\n }\r\n if (sessionMode !== \"immersive-ar\" && this.camera.compensateOnFirstFrame) {\r\n if ((this._nonVRCamera).setPosition) {\r\n (this._nonVRCamera).setPosition(this.camera.position);\r\n } else {\r\n this._nonVRCamera!.position.copyFrom(this.camera.position);\r\n }\r\n }\r\n\r\n this._setState(WebXRState.NOT_IN_XR);\r\n });\r\n\r\n // Wait until the first frame arrives before setting state to in xr\r\n this.sessionManager.onXRFrameObservable.addOnce(() => {\r\n this._setState(WebXRState.IN_XR);\r\n });\r\n return this.sessionManager;\r\n } catch (e) {\r\n console.log(e);\r\n console.log(e.message);\r\n this._setState(WebXRState.NOT_IN_XR);\r\n throw e;\r\n }\r\n }\r\n\r\n /**\r\n * Exits XR mode and returns the scene to its original state\r\n * @returns promise that resolves after xr mode has exited\r\n */\r\n public exitXRAsync() {\r\n // only exit if state is IN_XR\r\n if (this.state !== WebXRState.IN_XR) {\r\n return Promise.resolve();\r\n }\r\n this._setState(WebXRState.EXITING_XR);\r\n return this.sessionManager.exitXRAsync();\r\n }\r\n\r\n /**\r\n * Enable spectator mode for desktop VR experiences.\r\n * When spectator mode is enabled a camera will be attached to the desktop canvas and will\r\n * display the first rig camera's view on the desktop canvas.\r\n * Please note that this will degrade performance, as it requires another camera render.\r\n * It is also not recommended to enable this in devices like the quest, as it brings no benefit there.\r\n * @param options giving WebXRSpectatorModeOption for specutator camera to setup when the spectator mode is enabled.\r\n */\r\n public enableSpectatorMode(options?: WebXRSpectatorModeOption): void {\r\n if (!this._spectatorMode) {\r\n this._spectatorMode = true;\r\n this._switchSpectatorMode(options);\r\n }\r\n }\r\n\r\n /**\r\n * Disable spectator mode for desktop VR experiences.\r\n */\r\n public disableSpecatatorMode(): void {\r\n if (this._spectatorMode) {\r\n this._spectatorMode = false;\r\n this._switchSpectatorMode();\r\n }\r\n }\r\n\r\n private _switchSpectatorMode(options?: WebXRSpectatorModeOption): void {\r\n const fps = options?.fps ? options.fps : 1000.0;\r\n const refreshRate = (1.0 / fps) * 1000.0;\r\n const cameraIndex = options?.preferredCameraIndex ? options?.preferredCameraIndex : 0;\r\n\r\n const updateSpectatorCamera = () => {\r\n if (this._spectatorCamera) {\r\n const delta = this.sessionManager.currentTimestamp - this._lastTimestamp;\r\n if (delta >= refreshRate) {\r\n this._lastTimestamp = this.sessionManager.currentTimestamp;\r\n this._spectatorCamera.position.copyFrom(this.camera.rigCameras[cameraIndex].globalPosition);\r\n this._spectatorCamera.rotationQuaternion.copyFrom(this.camera.rigCameras[cameraIndex].absoluteRotation);\r\n }\r\n }\r\n };\r\n if (this._spectatorMode) {\r\n if (cameraIndex >= this.camera.rigCameras.length) {\r\n throw new Error(\"the preferred camera index is beyond the length of rig camera array.\");\r\n }\r\n const onStateChanged = () => {\r\n if (this.state === WebXRState.IN_XR) {\r\n this._spectatorCamera = new UniversalCamera(\"webxr-spectator\", Vector3.Zero(), this._scene);\r\n this._spectatorCamera.rotationQuaternion = new Quaternion();\r\n this._scene.activeCameras = [this.camera, this._spectatorCamera];\r\n this.sessionManager.onXRFrameObservable.add(updateSpectatorCamera);\r\n this._scene.onAfterRenderCameraObservable.add((camera) => {\r\n if (camera === this.camera) {\r\n // reset the dimensions object for correct resizing\r\n this._scene.getEngine().framebufferDimensionsObject = null;\r\n }\r\n });\r\n } else if (this.state === WebXRState.EXITING_XR) {\r\n this.sessionManager.onXRFrameObservable.removeCallback(updateSpectatorCamera);\r\n this._scene.activeCameras = null;\r\n }\r\n };\r\n this.onStateChangedObservable.add(onStateChanged);\r\n onStateChanged();\r\n } else {\r\n this.sessionManager.onXRFrameObservable.removeCallback(updateSpectatorCamera);\r\n this._scene.activeCameras = [this.camera];\r\n }\r\n }\r\n\r\n private _nonXRToXRCamera() {\r\n this.camera.setTransformationFromNonVRCamera(this._nonVRCamera!);\r\n this.onInitialXRPoseSetObservable.notifyObservers(this.camera);\r\n }\r\n\r\n private _setState(val: WebXRState) {\r\n if (this.state === val) {\r\n return;\r\n }\r\n this.state = val;\r\n this.onStateChangedObservable.notifyObservers(this.state);\r\n }\r\n}\r\n","import type { IMinimalMotionControllerObject, MotionControllerComponentType } from \"./webXRAbstractMotionController\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport type { IDisposable } from \"../../scene\";\r\n\r\n/**\r\n * X-Y values for axes in WebXR\r\n */\r\nexport interface IWebXRMotionControllerAxesValue {\r\n /**\r\n * The value of the x axis\r\n */\r\n x: number;\r\n /**\r\n * The value of the y-axis\r\n */\r\n y: number;\r\n}\r\n\r\n/**\r\n * changed / previous values for the values of this component\r\n */\r\nexport interface IWebXRMotionControllerComponentChangesValues {\r\n /**\r\n * current (this frame) value\r\n */\r\n current: T;\r\n /**\r\n * previous (last change) value\r\n */\r\n previous: T;\r\n}\r\n\r\n/**\r\n * Represents changes in the component between current frame and last values recorded\r\n */\r\nexport interface IWebXRMotionControllerComponentChanges {\r\n /**\r\n * will be populated with previous and current values if axes changed\r\n */\r\n axes?: IWebXRMotionControllerComponentChangesValues;\r\n /**\r\n * will be populated with previous and current values if pressed changed\r\n */\r\n pressed?: IWebXRMotionControllerComponentChangesValues;\r\n /**\r\n * will be populated with previous and current values if touched changed\r\n */\r\n touched?: IWebXRMotionControllerComponentChangesValues;\r\n /**\r\n * will be populated with previous and current values if value changed\r\n */\r\n value?: IWebXRMotionControllerComponentChangesValues;\r\n}\r\n/**\r\n * This class represents a single component (for example button or thumbstick) of a motion controller\r\n */\r\nexport class WebXRControllerComponent implements IDisposable {\r\n private _axes: IWebXRMotionControllerAxesValue = {\r\n x: 0,\r\n y: 0,\r\n };\r\n private _changes: IWebXRMotionControllerComponentChanges = {};\r\n private _currentValue: number = 0;\r\n private _hasChanges: boolean = false;\r\n private _pressed: boolean = false;\r\n private _touched: boolean = false;\r\n\r\n /**\r\n * button component type\r\n */\r\n public static BUTTON_TYPE: MotionControllerComponentType = \"button\";\r\n /**\r\n * squeeze component type\r\n */\r\n public static SQUEEZE_TYPE: MotionControllerComponentType = \"squeeze\";\r\n /**\r\n * Thumbstick component type\r\n */\r\n public static THUMBSTICK_TYPE: MotionControllerComponentType = \"thumbstick\";\r\n /**\r\n * Touchpad component type\r\n */\r\n public static TOUCHPAD_TYPE: MotionControllerComponentType = \"touchpad\";\r\n /**\r\n * trigger component type\r\n */\r\n public static TRIGGER_TYPE: MotionControllerComponentType = \"trigger\";\r\n\r\n /**\r\n * If axes are available for this component (like a touchpad or thumbstick) the observers will be notified when\r\n * the axes data changes\r\n */\r\n public onAxisValueChangedObservable: Observable<{ x: number; y: number }> = new Observable();\r\n /**\r\n * Observers registered here will be triggered when the state of a button changes\r\n * State change is either pressed / touched / value\r\n */\r\n public onButtonStateChangedObservable: Observable = new Observable();\r\n\r\n /**\r\n * Creates a new component for a motion controller.\r\n * It is created by the motion controller itself\r\n *\r\n * @param id the id of this component\r\n * @param type the type of the component\r\n * @param _buttonIndex index in the buttons array of the gamepad\r\n * @param _axesIndices indices of the values in the axes array of the gamepad\r\n */\r\n constructor(\r\n /**\r\n * the id of this component\r\n */\r\n public id: string,\r\n /**\r\n * the type of the component\r\n */\r\n public type: MotionControllerComponentType,\r\n private _buttonIndex: number = -1,\r\n private _axesIndices: number[] = []\r\n ) {}\r\n\r\n /**\r\n * The current axes data. If this component has no axes it will still return an object { x: 0, y: 0 }\r\n */\r\n public get axes(): IWebXRMotionControllerAxesValue {\r\n return this._axes;\r\n }\r\n\r\n /**\r\n * Get the changes. Elements will be populated only if they changed with their previous and current value\r\n */\r\n public get changes(): IWebXRMotionControllerComponentChanges {\r\n return this._changes;\r\n }\r\n\r\n /**\r\n * Return whether or not the component changed the last frame\r\n */\r\n public get hasChanges(): boolean {\r\n return this._hasChanges;\r\n }\r\n\r\n /**\r\n * is the button currently pressed\r\n */\r\n public get pressed(): boolean {\r\n return this._pressed;\r\n }\r\n\r\n /**\r\n * is the button currently touched\r\n */\r\n public get touched(): boolean {\r\n return this._touched;\r\n }\r\n\r\n /**\r\n * Get the current value of this component\r\n */\r\n public get value(): number {\r\n return this._currentValue;\r\n }\r\n\r\n /**\r\n * Dispose this component\r\n */\r\n public dispose(): void {\r\n this.onAxisValueChangedObservable.clear();\r\n this.onButtonStateChangedObservable.clear();\r\n }\r\n\r\n /**\r\n * Are there axes correlating to this component\r\n * @returns true is axes data is available\r\n */\r\n public isAxes(): boolean {\r\n return this._axesIndices.length !== 0;\r\n }\r\n\r\n /**\r\n * Is this component a button (hence - pressable)\r\n * @returns true if can be pressed\r\n */\r\n public isButton(): boolean {\r\n return this._buttonIndex !== -1;\r\n }\r\n\r\n /**\r\n * update this component using the gamepad object it is in. Called on every frame\r\n * @param nativeController the native gamepad controller object\r\n */\r\n public update(nativeController: IMinimalMotionControllerObject) {\r\n let buttonUpdated = false;\r\n let axesUpdate = false;\r\n this._hasChanges = false;\r\n this._changes = {};\r\n\r\n if (this.isButton()) {\r\n const button = nativeController.buttons[this._buttonIndex];\r\n // defensive, in case a profile was forced\r\n if (!button) {\r\n return;\r\n }\r\n if (this._currentValue !== button.value) {\r\n this.changes.value = {\r\n current: button.value,\r\n previous: this._currentValue,\r\n };\r\n buttonUpdated = true;\r\n this._currentValue = button.value;\r\n }\r\n if (this._touched !== button.touched) {\r\n this.changes.touched = {\r\n current: button.touched,\r\n previous: this._touched,\r\n };\r\n buttonUpdated = true;\r\n this._touched = button.touched;\r\n }\r\n if (this._pressed !== button.pressed) {\r\n this.changes.pressed = {\r\n current: button.pressed,\r\n previous: this._pressed,\r\n };\r\n buttonUpdated = true;\r\n this._pressed = button.pressed;\r\n }\r\n }\r\n\r\n if (this.isAxes()) {\r\n if (this._axes.x !== nativeController.axes[this._axesIndices[0]]) {\r\n this.changes.axes = {\r\n current: {\r\n x: nativeController.axes[this._axesIndices[0]],\r\n y: this._axes.y,\r\n },\r\n previous: {\r\n x: this._axes.x,\r\n y: this._axes.y,\r\n },\r\n };\r\n this._axes.x = nativeController.axes[this._axesIndices[0]];\r\n axesUpdate = true;\r\n }\r\n\r\n if (this._axes.y !== nativeController.axes[this._axesIndices[1]]) {\r\n if (this.changes.axes) {\r\n this.changes.axes.current.y = nativeController.axes[this._axesIndices[1]];\r\n } else {\r\n this.changes.axes = {\r\n current: {\r\n x: this._axes.x,\r\n y: nativeController.axes[this._axesIndices[1]],\r\n },\r\n previous: {\r\n x: this._axes.x,\r\n y: this._axes.y,\r\n },\r\n };\r\n }\r\n this._axes.y = nativeController.axes[this._axesIndices[1]];\r\n axesUpdate = true;\r\n }\r\n }\r\n\r\n if (buttonUpdated) {\r\n this._hasChanges = true;\r\n this.onButtonStateChangedObservable.notifyObservers(this);\r\n }\r\n if (axesUpdate) {\r\n this._hasChanges = true;\r\n this.onAxisValueChangedObservable.notifyObservers(this._axes);\r\n }\r\n }\r\n}\r\n","import type { IDisposable, Scene } from \"../../scene\";\r\nimport { WebXRControllerComponent } from \"./webXRControllerComponent\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport { Logger } from \"../../Misc/logger\";\r\nimport { SceneLoader } from \"../../Loading/sceneLoader\";\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport type { Nullable } from \"../../types\";\r\nimport { Quaternion, Vector3 } from \"../../Maths/math.vector\";\r\nimport { Mesh } from \"../../Meshes/mesh\";\r\n\r\n/**\r\n * Handedness type in xrInput profiles. These can be used to define layouts in the Layout Map.\r\n */\r\nexport type MotionControllerHandedness = \"none\" | \"left\" | \"right\";\r\n/**\r\n * The type of components available in motion controllers.\r\n * This is not the name of the component.\r\n */\r\nexport type MotionControllerComponentType = \"trigger\" | \"squeeze\" | \"touchpad\" | \"thumbstick\" | \"button\";\r\n\r\n/**\r\n * The state of a controller component\r\n */\r\nexport type MotionControllerComponentStateType = \"default\" | \"touched\" | \"pressed\";\r\n\r\n/**\r\n * The schema of motion controller layout.\r\n * No object will be initialized using this interface\r\n * This is used just to define the profile.\r\n */\r\nexport interface IMotionControllerLayout {\r\n /**\r\n * Path to load the assets. Usually relative to the base path\r\n */\r\n assetPath: string;\r\n /**\r\n * Available components (unsorted)\r\n */\r\n components: {\r\n /**\r\n * A map of component Ids\r\n */\r\n [componentId: string]: {\r\n /**\r\n * The type of input the component outputs\r\n */\r\n type: MotionControllerComponentType;\r\n /**\r\n * The indices of this component in the gamepad object\r\n */\r\n gamepadIndices: {\r\n /**\r\n * Index of button\r\n */\r\n button?: number;\r\n /**\r\n * If available, index of x-axis\r\n */\r\n xAxis?: number;\r\n /**\r\n * If available, index of y-axis\r\n */\r\n yAxis?: number;\r\n };\r\n /**\r\n * The mesh's root node name\r\n */\r\n rootNodeName: string;\r\n /**\r\n * Animation definitions for this model\r\n */\r\n visualResponses: {\r\n [stateKey: string]: {\r\n /**\r\n * What property will be animated\r\n */\r\n componentProperty: \"xAxis\" | \"yAxis\" | \"button\" | \"state\";\r\n /**\r\n * What states influence this visual response\r\n */\r\n states: MotionControllerComponentStateType[];\r\n /**\r\n * Type of animation - movement or visibility\r\n */\r\n valueNodeProperty: \"transform\" | \"visibility\";\r\n /**\r\n * Base node name to move. Its position will be calculated according to the min and max nodes\r\n */\r\n valueNodeName?: string;\r\n /**\r\n * Minimum movement node\r\n */\r\n minNodeName?: string;\r\n /**\r\n * Max movement node\r\n */\r\n maxNodeName?: string;\r\n };\r\n };\r\n /**\r\n * If touch enabled, what is the name of node to display user feedback\r\n */\r\n touchPointNodeName?: string;\r\n };\r\n };\r\n /**\r\n * Is it xr standard mapping or not\r\n */\r\n gamepadMapping: \"\" | \"xr-standard\";\r\n /**\r\n * Base root node of this entire model\r\n */\r\n rootNodeName: string;\r\n /**\r\n * Defines the main button component id\r\n */\r\n selectComponentId: string;\r\n}\r\n\r\n/**\r\n * A definition for the layout map in the input profile\r\n */\r\nexport interface IMotionControllerLayoutMap {\r\n /**\r\n * Layouts with handedness type as a key\r\n */\r\n [handedness: string /* handedness */]: IMotionControllerLayout;\r\n}\r\n\r\n/**\r\n * The XR Input profile schema\r\n * Profiles can be found here:\r\n * https://github.com/immersive-web/webxr-input-profiles/tree/master/packages/registry/profiles\r\n */\r\nexport interface IMotionControllerProfile {\r\n /**\r\n * fallback profiles for this profileId\r\n */\r\n fallbackProfileIds: string[];\r\n /**\r\n * The layout map, with handedness as key\r\n */\r\n layouts: IMotionControllerLayoutMap;\r\n /**\r\n * The id of this profile\r\n * correlates to the profile(s) in the xrInput.profiles array\r\n */\r\n profileId: string;\r\n}\r\n\r\n/**\r\n * A helper-interface for the 3 meshes needed for controller button animation\r\n * The meshes are provided to the _lerpButtonTransform function to calculate the current position of the value mesh\r\n */\r\nexport interface IMotionControllerButtonMeshMap {\r\n /**\r\n * the mesh that defines the pressed value mesh position.\r\n * This is used to find the max-position of this button\r\n */\r\n pressedMesh: AbstractMesh;\r\n /**\r\n * the mesh that defines the unpressed value mesh position.\r\n * This is used to find the min (or initial) position of this button\r\n */\r\n unpressedMesh: AbstractMesh;\r\n /**\r\n * The mesh that will be changed when value changes\r\n */\r\n valueMesh: AbstractMesh;\r\n}\r\n\r\n/**\r\n * A helper-interface for the 3 meshes needed for controller axis animation.\r\n * This will be expanded when touchpad animations are fully supported\r\n * The meshes are provided to the _lerpAxisTransform function to calculate the current position of the value mesh\r\n */\r\nexport interface IMotionControllerMeshMap {\r\n /**\r\n * the mesh that defines the maximum value mesh position.\r\n */\r\n maxMesh?: AbstractMesh;\r\n /**\r\n * the mesh that defines the minimum value mesh position.\r\n */\r\n minMesh?: AbstractMesh;\r\n /**\r\n * The mesh that will be changed when axis value changes\r\n */\r\n valueMesh?: AbstractMesh;\r\n}\r\n\r\n/**\r\n * The elements needed for change-detection of the gamepad objects in motion controllers\r\n */\r\nexport interface IMinimalMotionControllerObject {\r\n /**\r\n * Available axes of this controller\r\n */\r\n axes: number[];\r\n /**\r\n * An array of available buttons\r\n */\r\n buttons: Array<{\r\n /**\r\n * Value of the button/trigger\r\n */\r\n value: number;\r\n /**\r\n * If the button/trigger is currently touched\r\n */\r\n touched: boolean;\r\n /**\r\n * If the button/trigger is currently pressed\r\n */\r\n pressed: boolean;\r\n }>;\r\n\r\n /**\r\n * EXPERIMENTAL haptic support.\r\n */\r\n hapticActuators?: Array<{\r\n pulse: (value: number, duration: number) => Promise;\r\n }>;\r\n}\r\n\r\n/**\r\n * An Abstract Motion controller\r\n * This class receives an xrInput and a profile layout and uses those to initialize the components\r\n * Each component has an observable to check for changes in value and state\r\n */\r\nexport abstract class WebXRAbstractMotionController implements IDisposable {\r\n private _initComponent = (id: string) => {\r\n if (!id) {\r\n return;\r\n }\r\n const componentDef = this.layout.components[id];\r\n const type = componentDef.type;\r\n const buttonIndex = componentDef.gamepadIndices.button;\r\n // search for axes\r\n const axes: number[] = [];\r\n if (componentDef.gamepadIndices.xAxis !== undefined && componentDef.gamepadIndices.yAxis !== undefined) {\r\n axes.push(componentDef.gamepadIndices.xAxis, componentDef.gamepadIndices.yAxis);\r\n }\r\n\r\n this.components[id] = new WebXRControllerComponent(id, type, buttonIndex, axes);\r\n };\r\n\r\n private _modelReady: boolean = false;\r\n\r\n /**\r\n * A map of components (WebXRControllerComponent) in this motion controller\r\n * Components have a ComponentType and can also have both button and axis definitions\r\n */\r\n public readonly components: {\r\n [id: string]: WebXRControllerComponent;\r\n } = {};\r\n\r\n /**\r\n * Disable the model's animation. Can be set at any time.\r\n */\r\n public disableAnimation: boolean = false;\r\n /**\r\n * Observers registered here will be triggered when the model of this controller is done loading\r\n */\r\n public onModelLoadedObservable: Observable = new Observable();\r\n /**\r\n * The profile id of this motion controller\r\n */\r\n public abstract profileId: string;\r\n /**\r\n * The root mesh of the model. It is null if the model was not yet initialized\r\n */\r\n public rootMesh: Nullable;\r\n\r\n /**\r\n * constructs a new abstract motion controller\r\n * @param scene the scene to which the model of the controller will be added\r\n * @param layout The profile layout to load\r\n * @param gamepadObject The gamepad object correlating to this controller\r\n * @param handedness handedness (left/right/none) of this controller\r\n * @param _doNotLoadControllerMesh set this flag to ignore the mesh loading\r\n * @param _controllerCache a cache holding controller models already loaded in this session\r\n */\r\n constructor(\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n protected scene: Scene,\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n protected layout: IMotionControllerLayout,\r\n /**\r\n * The gamepad object correlating to this controller\r\n */\r\n public gamepadObject: IMinimalMotionControllerObject,\r\n /**\r\n * handedness (left/right/none) of this controller\r\n */\r\n public handedness: MotionControllerHandedness,\r\n /**\r\n * @internal\r\n */\r\n public _doNotLoadControllerMesh: boolean = false,\r\n private _controllerCache?: Array<{\r\n filename: string;\r\n path: string;\r\n meshes: AbstractMesh[];\r\n }>\r\n ) {\r\n // initialize the components\r\n if (layout.components) {\r\n Object.keys(layout.components).forEach(this._initComponent);\r\n }\r\n // Model is loaded in WebXRInput\r\n }\r\n\r\n /**\r\n * Dispose this controller, the model mesh and all its components\r\n */\r\n public dispose(): void {\r\n this.getComponentIds().forEach((id) => this.getComponent(id).dispose());\r\n if (this.rootMesh) {\r\n this.rootMesh.getChildren(undefined, true).forEach((node) => {\r\n node.setEnabled(false);\r\n });\r\n this.rootMesh.dispose(!!this._controllerCache, !this._controllerCache);\r\n }\r\n }\r\n\r\n /**\r\n * Returns all components of specific type\r\n * @param type the type to search for\r\n * @returns an array of components with this type\r\n */\r\n public getAllComponentsOfType(type: MotionControllerComponentType): WebXRControllerComponent[] {\r\n return this.getComponentIds()\r\n .map((id) => this.components[id])\r\n .filter((component) => component.type === type);\r\n }\r\n\r\n /**\r\n * get a component based an its component id as defined in layout.components\r\n * @param id the id of the component\r\n * @returns the component correlates to the id or undefined if not found\r\n */\r\n public getComponent(id: string): WebXRControllerComponent {\r\n return this.components[id];\r\n }\r\n\r\n /**\r\n * Get the list of components available in this motion controller\r\n * @returns an array of strings correlating to available components\r\n */\r\n public getComponentIds(): string[] {\r\n return Object.keys(this.components);\r\n }\r\n\r\n /**\r\n * Get the first component of specific type\r\n * @param type type of component to find\r\n * @returns a controller component or null if not found\r\n */\r\n public getComponentOfType(type: MotionControllerComponentType): Nullable {\r\n return this.getAllComponentsOfType(type)[0] || null;\r\n }\r\n\r\n /**\r\n * Get the main (Select) component of this controller as defined in the layout\r\n * @returns the main component of this controller\r\n */\r\n public getMainComponent(): WebXRControllerComponent {\r\n return this.getComponent(this.layout.selectComponentId);\r\n }\r\n\r\n /**\r\n * Loads the model correlating to this controller\r\n * When the mesh is loaded, the onModelLoadedObservable will be triggered\r\n * @returns A promise fulfilled with the result of the model loading\r\n */\r\n public async loadModel(): Promise {\r\n const useGeneric = !this._getModelLoadingConstraints();\r\n let loadingParams = this._getGenericFilenameAndPath();\r\n // Checking if GLB loader is present\r\n if (useGeneric) {\r\n Logger.Warn(\"Falling back to generic models\");\r\n } else {\r\n loadingParams = this._getFilenameAndPath();\r\n }\r\n return new Promise((resolve, reject) => {\r\n const meshesLoaded = (meshes: AbstractMesh[]) => {\r\n if (useGeneric) {\r\n this._getGenericParentMesh(meshes);\r\n } else {\r\n this._setRootMesh(meshes);\r\n }\r\n this._processLoadedModel(meshes);\r\n this._modelReady = true;\r\n this.onModelLoadedObservable.notifyObservers(this);\r\n resolve(true);\r\n };\r\n if (this._controllerCache) {\r\n // look for it in the cache\r\n const found = this._controllerCache.filter((c) => {\r\n return c.filename === loadingParams.filename && c.path === loadingParams.path;\r\n });\r\n if (found[0]) {\r\n found[0].meshes.forEach((mesh) => mesh.setEnabled(true));\r\n meshesLoaded(found[0].meshes);\r\n return;\r\n // found, don't continue to load\r\n }\r\n }\r\n SceneLoader.ImportMesh(\r\n \"\",\r\n loadingParams.path,\r\n loadingParams.filename,\r\n this.scene,\r\n (meshes) => {\r\n if (this._controllerCache) {\r\n this._controllerCache.push({\r\n ...loadingParams,\r\n meshes,\r\n });\r\n }\r\n meshesLoaded(meshes);\r\n },\r\n null,\r\n (_scene: Scene, message: string) => {\r\n Logger.Log(message);\r\n Logger.Warn(`Failed to retrieve controller model of type ${this.profileId} from the remote server: ${loadingParams.path}${loadingParams.filename}`);\r\n reject(message);\r\n }\r\n );\r\n });\r\n }\r\n\r\n /**\r\n * Update this model using the current XRFrame\r\n * @param xrFrame the current xr frame to use and update the model\r\n */\r\n public updateFromXRFrame(xrFrame: XRFrame): void {\r\n this.getComponentIds().forEach((id) => this.getComponent(id).update(this.gamepadObject));\r\n this.updateModel(xrFrame);\r\n }\r\n\r\n /**\r\n * Backwards compatibility due to a deeply-integrated typo\r\n */\r\n public get handness() {\r\n return this.handedness;\r\n }\r\n\r\n /**\r\n * Pulse (vibrate) this controller\r\n * If the controller does not support pulses, this function will fail silently and return Promise directly after called\r\n * Consecutive calls to this function will cancel the last pulse call\r\n *\r\n * @param value the strength of the pulse in 0.0...1.0 range\r\n * @param duration Duration of the pulse in milliseconds\r\n * @param hapticActuatorIndex optional index of actuator (will usually be 0)\r\n * @returns a promise that will send true when the pulse has ended and false if the device doesn't support pulse or an error accrued\r\n */\r\n public pulse(value: number, duration: number, hapticActuatorIndex: number = 0): Promise {\r\n if (this.gamepadObject.hapticActuators && this.gamepadObject.hapticActuators[hapticActuatorIndex]) {\r\n return this.gamepadObject.hapticActuators[hapticActuatorIndex].pulse(value, duration);\r\n } else {\r\n return Promise.resolve(false);\r\n }\r\n }\r\n\r\n // Look through all children recursively. This will return null if no mesh exists with the given name.\r\n protected _getChildByName(node: AbstractMesh, name: string): AbstractMesh | undefined {\r\n return node.getChildren((n) => n.name === name, false)[0];\r\n }\r\n\r\n // Look through only immediate children. This will return null if no mesh exists with the given name.\r\n protected _getImmediateChildByName(node: AbstractMesh, name: string): AbstractMesh | undefined {\r\n return node.getChildren((n) => n.name == name, true)[0];\r\n }\r\n\r\n /**\r\n * Moves the axis on the controller mesh based on its current state\r\n * @param axisMap\r\n * @param axisValue the value of the axis which determines the meshes new position\r\n * @internal\r\n */\r\n protected _lerpTransform(axisMap: IMotionControllerMeshMap, axisValue: number, fixValueCoordinates?: boolean): void {\r\n if (!axisMap.minMesh || !axisMap.maxMesh || !axisMap.valueMesh) {\r\n return;\r\n }\r\n\r\n if (!axisMap.minMesh.rotationQuaternion || !axisMap.maxMesh.rotationQuaternion || !axisMap.valueMesh.rotationQuaternion) {\r\n return;\r\n }\r\n\r\n // Convert from gamepad value range (-1 to +1) to lerp range (0 to 1)\r\n const lerpValue = fixValueCoordinates ? axisValue * 0.5 + 0.5 : axisValue;\r\n Quaternion.SlerpToRef(axisMap.minMesh.rotationQuaternion, axisMap.maxMesh.rotationQuaternion, lerpValue, axisMap.valueMesh.rotationQuaternion);\r\n Vector3.LerpToRef(axisMap.minMesh.position, axisMap.maxMesh.position, lerpValue, axisMap.valueMesh.position);\r\n }\r\n\r\n /**\r\n * Update the model itself with the current frame data\r\n * @param xrFrame the frame to use for updating the model mesh\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n protected updateModel(xrFrame: XRFrame): void {\r\n if (!this._modelReady) {\r\n return;\r\n }\r\n this._updateModel(xrFrame);\r\n }\r\n\r\n /**\r\n * Get the filename and path for this controller's model\r\n * @returns a map of filename and path\r\n */\r\n protected abstract _getFilenameAndPath(): { filename: string; path: string };\r\n /**\r\n * This function is called before the mesh is loaded. It checks for loading constraints.\r\n * For example, this function can check if the GLB loader is available\r\n * If this function returns false, the generic controller will be loaded instead\r\n * @returns Is the client ready to load the mesh\r\n */\r\n protected abstract _getModelLoadingConstraints(): boolean;\r\n /**\r\n * This function will be called after the model was successfully loaded and can be used\r\n * for mesh transformations before it is available for the user\r\n * @param meshes the loaded meshes\r\n */\r\n protected abstract _processLoadedModel(meshes: AbstractMesh[]): void;\r\n /**\r\n * Set the root mesh for this controller. Important for the WebXR controller class\r\n * @param meshes the loaded meshes\r\n */\r\n protected abstract _setRootMesh(meshes: AbstractMesh[]): void;\r\n /**\r\n * A function executed each frame that updates the mesh (if needed)\r\n * @param xrFrame the current xrFrame\r\n */\r\n protected abstract _updateModel(xrFrame: XRFrame): void;\r\n\r\n private _getGenericFilenameAndPath(): { filename: string; path: string } {\r\n return {\r\n filename: \"generic.babylon\",\r\n path: \"https://controllers.babylonjs.com/generic/\",\r\n };\r\n }\r\n\r\n private _getGenericParentMesh(meshes: AbstractMesh[]): void {\r\n this.rootMesh = new Mesh(this.profileId + \" \" + this.handedness, this.scene);\r\n\r\n meshes.forEach((mesh) => {\r\n if (!mesh.parent) {\r\n mesh.isPickable = false;\r\n mesh.setParent(this.rootMesh);\r\n }\r\n });\r\n\r\n this.rootMesh.rotationQuaternion = Quaternion.FromEulerAngles(0, Math.PI, 0);\r\n }\r\n}\r\n","import type { IMinimalMotionControllerObject, MotionControllerHandedness, IMotionControllerLayoutMap } from \"./webXRAbstractMotionController\";\r\nimport { WebXRAbstractMotionController } from \"./webXRAbstractMotionController\";\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { Mesh } from \"../../Meshes/mesh\";\r\nimport { Quaternion } from \"../../Maths/math.vector\";\r\n\r\n/**\r\n * A generic trigger-only motion controller for WebXR\r\n */\r\nexport class WebXRGenericTriggerMotionController extends WebXRAbstractMotionController {\r\n /**\r\n * Static version of the profile id of this controller\r\n */\r\n public static ProfileId = \"generic-trigger\";\r\n\r\n public profileId = WebXRGenericTriggerMotionController.ProfileId;\r\n\r\n constructor(scene: Scene, gamepadObject: IMinimalMotionControllerObject, handedness: MotionControllerHandedness) {\r\n super(scene, GenericTriggerLayout[handedness], gamepadObject, handedness);\r\n }\r\n\r\n protected _getFilenameAndPath(): { filename: string; path: string } {\r\n return {\r\n filename: \"generic.babylon\",\r\n path: \"https://controllers.babylonjs.com/generic/\",\r\n };\r\n }\r\n\r\n protected _getModelLoadingConstraints(): boolean {\r\n return true;\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n protected _processLoadedModel(meshes: AbstractMesh[]): void {\r\n // nothing to do\r\n }\r\n\r\n protected _setRootMesh(meshes: AbstractMesh[]): void {\r\n this.rootMesh = new Mesh(this.profileId + \" \" + this.handedness, this.scene);\r\n\r\n meshes.forEach((mesh) => {\r\n mesh.isPickable = false;\r\n if (!mesh.parent) {\r\n mesh.setParent(this.rootMesh);\r\n }\r\n });\r\n\r\n this.rootMesh.rotationQuaternion = Quaternion.FromEulerAngles(0, Math.PI, 0);\r\n }\r\n\r\n protected _updateModel(): void {\r\n // no-op\r\n }\r\n}\r\n\r\n// https://github.com/immersive-web/webxr-input-profiles/blob/master/packages/registry/profiles/generic/generic-trigger-touchpad-thumbstick.json\r\nconst GenericTriggerLayout: IMotionControllerLayoutMap = {\r\n left: {\r\n selectComponentId: \"xr-standard-trigger\",\r\n components: {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n \"xr-standard-trigger\": {\r\n type: \"trigger\",\r\n gamepadIndices: {\r\n button: 0,\r\n },\r\n rootNodeName: \"xr_standard_trigger\",\r\n visualResponses: {},\r\n },\r\n },\r\n gamepadMapping: \"xr-standard\",\r\n rootNodeName: \"generic-trigger-left\",\r\n assetPath: \"left.glb\",\r\n },\r\n right: {\r\n selectComponentId: \"xr-standard-trigger\",\r\n components: {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n \"xr-standard-trigger\": {\r\n type: \"trigger\",\r\n gamepadIndices: {\r\n button: 0,\r\n },\r\n rootNodeName: \"xr_standard_trigger\",\r\n visualResponses: {},\r\n },\r\n },\r\n gamepadMapping: \"xr-standard\",\r\n rootNodeName: \"generic-trigger-right\",\r\n assetPath: \"right.glb\",\r\n },\r\n none: {\r\n selectComponentId: \"xr-standard-trigger\",\r\n components: {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n \"xr-standard-trigger\": {\r\n type: \"trigger\",\r\n gamepadIndices: {\r\n button: 0,\r\n },\r\n rootNodeName: \"xr_standard_trigger\",\r\n visualResponses: {},\r\n },\r\n },\r\n gamepadMapping: \"xr-standard\",\r\n rootNodeName: \"generic-trigger-none\",\r\n assetPath: \"none.glb\",\r\n },\r\n};\r\n","import type { Vector4 } from \"../../Maths/math.vector\";\r\nimport { Vector3, Matrix } from \"../../Maths/math.vector\";\r\nimport { Mesh } from \"../mesh\";\r\nimport { VertexData } from \"../mesh.vertexData\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { Nullable } from \"../../types\";\r\nimport { CompatibilityOptions } from \"../../Compat/compatibilityOptions\";\r\n\r\n/**\r\n * Creates the VertexData for an ellipsoid, defaults to a sphere\r\n * @param options an object used to set the following optional parameters for the box, required but can be empty\r\n * * segments sets the number of horizontal strips optional, default 32\r\n * * diameter sets the axes dimensions, diameterX, diameterY and diameterZ to the value of diameter, optional default 1\r\n * * diameterX sets the diameterX (x direction) of the ellipsoid, overwrites the diameterX set by diameter, optional, default diameter\r\n * * diameterY sets the diameterY (y direction) of the ellipsoid, overwrites the diameterY set by diameter, optional, default diameter\r\n * * diameterZ sets the diameterZ (z direction) of the ellipsoid, overwrites the diameterZ set by diameter, optional, default diameter\r\n * * arc a number from 0 to 1, to create an unclosed ellipsoid based on the fraction of the circumference (latitude) given by the arc value, optional, default 1\r\n * * slice a number from 0 to 1, to create an unclosed ellipsoid based on the fraction of the height (latitude) given by the arc value, optional, default 1\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.segments\r\n * @param options.diameter\r\n * @param options.diameterX\r\n * @param options.diameterY\r\n * @param options.diameterZ\r\n * @param options.arc\r\n * @param options.slice\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param options.dedupTopBottomIndices\r\n * @returns the VertexData of the ellipsoid\r\n */\r\nexport function CreateSphereVertexData(options: {\r\n segments?: number;\r\n diameter?: number;\r\n diameterX?: number;\r\n diameterY?: number;\r\n diameterZ?: number;\r\n arc?: number;\r\n slice?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n dedupTopBottomIndices?: boolean;\r\n}): VertexData {\r\n const segments: number = (options.segments || 32) | 0;\r\n const diameterX: number = options.diameterX || options.diameter || 1;\r\n const diameterY: number = options.diameterY || options.diameter || 1;\r\n const diameterZ: number = options.diameterZ || options.diameter || 1;\r\n const arc: number = options.arc && (options.arc <= 0 || options.arc > 1) ? 1.0 : options.arc || 1.0;\r\n const slice: number = options.slice && options.slice <= 0 ? 1.0 : options.slice || 1.0;\r\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\r\n const dedupTopBottomIndices = !!options.dedupTopBottomIndices;\r\n\r\n const radius = new Vector3(diameterX / 2, diameterY / 2, diameterZ / 2);\r\n\r\n const totalZRotationSteps = 2 + segments;\r\n const totalYRotationSteps = 2 * totalZRotationSteps;\r\n\r\n const indices = [];\r\n const positions = [];\r\n const normals = [];\r\n const uvs = [];\r\n\r\n for (let zRotationStep = 0; zRotationStep <= totalZRotationSteps; zRotationStep++) {\r\n const normalizedZ = zRotationStep / totalZRotationSteps;\r\n const angleZ = normalizedZ * Math.PI * slice;\r\n\r\n for (let yRotationStep = 0; yRotationStep <= totalYRotationSteps; yRotationStep++) {\r\n const normalizedY = yRotationStep / totalYRotationSteps;\r\n\r\n const angleY = normalizedY * Math.PI * 2 * arc;\r\n\r\n const rotationZ = Matrix.RotationZ(-angleZ);\r\n const rotationY = Matrix.RotationY(angleY);\r\n const afterRotZ = Vector3.TransformCoordinates(Vector3.Up(), rotationZ);\r\n const complete = Vector3.TransformCoordinates(afterRotZ, rotationY);\r\n\r\n const vertex = complete.multiply(radius);\r\n const normal = complete.divide(radius).normalize();\r\n\r\n positions.push(vertex.x, vertex.y, vertex.z);\r\n normals.push(normal.x, normal.y, normal.z);\r\n uvs.push(normalizedY, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - normalizedZ : normalizedZ);\r\n }\r\n\r\n if (zRotationStep > 0) {\r\n const verticesCount = positions.length / 3;\r\n for (let firstIndex = verticesCount - 2 * (totalYRotationSteps + 1); firstIndex + totalYRotationSteps + 2 < verticesCount; firstIndex++) {\r\n if (dedupTopBottomIndices) {\r\n if (zRotationStep > 1) {\r\n indices.push(firstIndex);\r\n indices.push(firstIndex + 1);\r\n indices.push(firstIndex + totalYRotationSteps + 1);\r\n }\r\n if (zRotationStep < totalZRotationSteps || slice < 1.0) {\r\n indices.push(firstIndex + totalYRotationSteps + 1);\r\n indices.push(firstIndex + 1);\r\n indices.push(firstIndex + totalYRotationSteps + 2);\r\n }\r\n } else {\r\n indices.push(firstIndex);\r\n indices.push(firstIndex + 1);\r\n indices.push(firstIndex + totalYRotationSteps + 1);\r\n\r\n indices.push(firstIndex + totalYRotationSteps + 1);\r\n indices.push(firstIndex + 1);\r\n indices.push(firstIndex + totalYRotationSteps + 2);\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Sides\r\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\r\n\r\n // Result\r\n const vertexData = new VertexData();\r\n\r\n vertexData.indices = indices;\r\n vertexData.positions = positions;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n\r\n return vertexData;\r\n}\r\n\r\n/**\r\n * Creates a sphere mesh\r\n * * The parameter `diameter` sets the diameter size (float) of the sphere (default 1)\r\n * * You can set some different sphere dimensions, for instance to build an ellipsoid, by using the parameters `diameterX`, `diameterY` and `diameterZ` (all by default have the same value of `diameter`)\r\n * * The parameter `segments` sets the sphere number of horizontal stripes (positive integer, default 32)\r\n * * You can create an unclosed sphere with the parameter `arc` (positive float, default 1), valued between 0 and 1, what is the ratio of the circumference (latitude) : 2 x PI x ratio\r\n * * You can create an unclosed sphere on its height with the parameter `slice` (positive float, default1), valued between 0 and 1, what is the height ratio (longitude)\r\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param options.segments\r\n * @param options.diameter\r\n * @param options.diameterX\r\n * @param options.diameterY\r\n * @param options.diameterZ\r\n * @param options.arc\r\n * @param options.slice\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param options.updatable\r\n * @param scene defines the hosting scene\r\n * @returns the sphere mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#sphere\r\n */\r\nexport function CreateSphere(\r\n name: string,\r\n options: {\r\n segments?: number;\r\n diameter?: number;\r\n diameterX?: number;\r\n diameterY?: number;\r\n diameterZ?: number;\r\n arc?: number;\r\n slice?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n updatable?: boolean;\r\n } = {},\r\n scene: Nullable = null\r\n): Mesh {\r\n const sphere = new Mesh(name, scene);\r\n\r\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n sphere._originalBuilderSideOrientation = options.sideOrientation;\r\n\r\n const vertexData = CreateSphereVertexData(options);\r\n\r\n vertexData.applyToMesh(sphere, options.updatable);\r\n\r\n return sphere;\r\n}\r\n\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated use CreateSphere directly\r\n */\r\nexport const SphereBuilder = {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreateSphere,\r\n};\r\n\r\nVertexData.CreateSphere = CreateSphereVertexData;\r\n\r\nMesh.CreateSphere = (name: string, segments: number, diameter: number, scene?: Scene, updatable?: boolean, sideOrientation?: number): Mesh => {\r\n const options = {\r\n segments: segments,\r\n diameterX: diameter,\r\n diameterY: diameter,\r\n diameterZ: diameter,\r\n sideOrientation: sideOrientation,\r\n updatable: updatable,\r\n };\r\n\r\n return CreateSphere(name, options, scene);\r\n};\r\n","import type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport type { IMotionControllerProfile, IMotionControllerMeshMap } from \"./webXRAbstractMotionController\";\r\nimport { WebXRAbstractMotionController } from \"./webXRAbstractMotionController\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { SceneLoader } from \"../../Loading/sceneLoader\";\r\nimport { Mesh } from \"../../Meshes/mesh\";\r\nimport { Axis, Space } from \"../../Maths/math.axis\";\r\nimport { Color3 } from \"../../Maths/math.color\";\r\nimport { WebXRControllerComponent } from \"./webXRControllerComponent\";\r\nimport { CreateSphere } from \"../../Meshes/Builders/sphereBuilder\";\r\nimport { StandardMaterial } from \"../../Materials/standardMaterial\";\r\nimport { Logger } from \"../../Misc/logger\";\r\n\r\n/**\r\n * A profiled motion controller has its profile loaded from an online repository.\r\n * The class is responsible of loading the model, mapping the keys and enabling model-animations\r\n */\r\nexport class WebXRProfiledMotionController extends WebXRAbstractMotionController {\r\n private _buttonMeshMapping: {\r\n [buttonName: string]: {\r\n mainMesh?: AbstractMesh;\r\n states: {\r\n [state: string]: IMotionControllerMeshMap;\r\n };\r\n };\r\n } = {};\r\n private _touchDots: { [visKey: string]: AbstractMesh } = {};\r\n\r\n /**\r\n * The profile ID of this controller. Will be populated when the controller initializes.\r\n */\r\n public profileId: string;\r\n\r\n constructor(\r\n scene: Scene,\r\n xrInput: XRInputSource,\r\n _profile: IMotionControllerProfile,\r\n private _repositoryUrl: string,\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n private controllerCache?: Array<{\r\n filename: string;\r\n path: string;\r\n meshes: AbstractMesh[];\r\n }>\r\n ) {\r\n super(scene, _profile.layouts[xrInput.handedness || \"none\"], xrInput.gamepad as any, xrInput.handedness, undefined, controllerCache);\r\n this.profileId = _profile.profileId;\r\n }\r\n\r\n public dispose() {\r\n super.dispose();\r\n if (!this.controllerCache) {\r\n Object.keys(this._touchDots).forEach((visResKey) => {\r\n this._touchDots[visResKey].dispose();\r\n });\r\n }\r\n }\r\n\r\n protected _getFilenameAndPath(): { filename: string; path: string } {\r\n return {\r\n filename: this.layout.assetPath,\r\n path: `${this._repositoryUrl}/profiles/${this.profileId}/`,\r\n };\r\n }\r\n\r\n protected _getModelLoadingConstraints(): boolean {\r\n const glbLoaded = SceneLoader.IsPluginForExtensionAvailable(\".glb\");\r\n if (!glbLoaded) {\r\n Logger.Warn(\"glTF / glb loader was not registered, using generic controller instead\");\r\n }\r\n return glbLoaded;\r\n }\r\n\r\n protected _processLoadedModel(_meshes: AbstractMesh[]): void {\r\n this.getComponentIds().forEach((type) => {\r\n const componentInLayout = this.layout.components[type];\r\n this._buttonMeshMapping[type] = {\r\n mainMesh: this._getChildByName(this.rootMesh!, componentInLayout.rootNodeName),\r\n states: {},\r\n };\r\n Object.keys(componentInLayout.visualResponses).forEach((visualResponseKey) => {\r\n const visResponse = componentInLayout.visualResponses[visualResponseKey];\r\n if (visResponse.valueNodeProperty === \"transform\") {\r\n this._buttonMeshMapping[type].states[visualResponseKey] = {\r\n valueMesh: this._getChildByName(this.rootMesh!, visResponse.valueNodeName!),\r\n minMesh: this._getChildByName(this.rootMesh!, visResponse.minNodeName!),\r\n maxMesh: this._getChildByName(this.rootMesh!, visResponse.maxNodeName!),\r\n };\r\n } else {\r\n // visibility, usually for touchpads\r\n const nameOfMesh =\r\n componentInLayout.type === WebXRControllerComponent.TOUCHPAD_TYPE && componentInLayout.touchPointNodeName\r\n ? componentInLayout.touchPointNodeName\r\n : visResponse.valueNodeName!;\r\n this._buttonMeshMapping[type].states[visualResponseKey] = {\r\n valueMesh: this._getChildByName(this.rootMesh!, nameOfMesh),\r\n };\r\n if (componentInLayout.type === WebXRControllerComponent.TOUCHPAD_TYPE && !this._touchDots[visualResponseKey]) {\r\n const dot = CreateSphere(\r\n visualResponseKey + \"dot\",\r\n {\r\n diameter: 0.0015,\r\n segments: 8,\r\n },\r\n this.scene\r\n );\r\n dot.material = new StandardMaterial(visualResponseKey + \"mat\", this.scene);\r\n (dot.material).diffuseColor = Color3.Red();\r\n dot.parent = this._buttonMeshMapping[type].states[visualResponseKey].valueMesh || null;\r\n dot.isVisible = false;\r\n this._touchDots[visualResponseKey] = dot;\r\n }\r\n }\r\n });\r\n });\r\n }\r\n\r\n protected _setRootMesh(meshes: AbstractMesh[]): void {\r\n this.rootMesh = new Mesh(this.profileId + \"-\" + this.handedness, this.scene);\r\n this.rootMesh.isPickable = false;\r\n let rootMesh;\r\n // Find the root node in the loaded glTF scene, and attach it as a child of 'parentMesh'\r\n for (let i = 0; i < meshes.length; i++) {\r\n const mesh = meshes[i];\r\n\r\n mesh.isPickable = false;\r\n\r\n if (!mesh.parent) {\r\n // Handle root node, attach to the new parentMesh\r\n rootMesh = mesh;\r\n }\r\n }\r\n\r\n if (rootMesh) {\r\n rootMesh.setParent(this.rootMesh);\r\n }\r\n if (!this.scene.useRightHandedSystem) {\r\n this.rootMesh.rotate(Axis.Y, Math.PI, Space.WORLD);\r\n }\r\n }\r\n\r\n protected _updateModel(_xrFrame: XRFrame): void {\r\n if (this.disableAnimation) {\r\n return;\r\n }\r\n this.getComponentIds().forEach((id) => {\r\n const component = this.getComponent(id);\r\n if (!component.hasChanges) {\r\n return;\r\n }\r\n const meshes = this._buttonMeshMapping[id];\r\n const componentInLayout = this.layout.components[id];\r\n Object.keys(componentInLayout.visualResponses).forEach((visualResponseKey) => {\r\n const visResponse = componentInLayout.visualResponses[visualResponseKey];\r\n let value = component.value;\r\n if (visResponse.componentProperty === \"xAxis\") {\r\n value = component.axes.x;\r\n } else if (visResponse.componentProperty === \"yAxis\") {\r\n value = component.axes.y;\r\n }\r\n if (visResponse.valueNodeProperty === \"transform\") {\r\n this._lerpTransform(meshes.states[visualResponseKey], value, visResponse.componentProperty !== \"button\");\r\n } else {\r\n // visibility\r\n const valueMesh = meshes.states[visualResponseKey].valueMesh;\r\n if (valueMesh) {\r\n valueMesh.isVisible = component.touched || component.pressed;\r\n }\r\n if (this._touchDots[visualResponseKey]) {\r\n this._touchDots[visualResponseKey].isVisible = component.touched || component.pressed;\r\n }\r\n }\r\n });\r\n });\r\n }\r\n}\r\n","import type { WebXRAbstractMotionController, IMotionControllerProfile } from \"./webXRAbstractMotionController\";\r\nimport { WebXRGenericTriggerMotionController } from \"./webXRGenericMotionController\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { Tools } from \"../../Misc/tools\";\r\nimport { WebXRProfiledMotionController } from \"./webXRProfiledMotionController\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\n\r\n/**\r\n * A construction function type to create a new controller based on an xrInput object\r\n */\r\nexport type MotionControllerConstructor = (xrInput: XRInputSource, scene: Scene) => WebXRAbstractMotionController;\r\n\r\n/**\r\n * The MotionController Manager manages all registered motion controllers and loads the right one when needed.\r\n *\r\n * When this repository is complete: https://github.com/immersive-web/webxr-input-profiles/tree/master/packages/assets\r\n * it should be replaced with auto-loaded controllers.\r\n *\r\n * When using a model try to stay as generic as possible. Eventually there will be no need in any of the controller classes\r\n */\r\n\r\nconst controllerCache: Array<{\r\n filename: string;\r\n path: string;\r\n meshes: AbstractMesh[];\r\n}> = [];\r\n\r\n/**\r\n * Motion controller manager is managing the different webxr profiles and makes sure the right\r\n * controller is being loaded.\r\n */\r\nexport class WebXRMotionControllerManager {\r\n private static _AvailableControllers: { [type: string]: MotionControllerConstructor } = {};\r\n private static _Fallbacks: { [profileId: string]: string[] } = {};\r\n // cache for loading\r\n private static _ProfileLoadingPromises: { [profileName: string]: Promise } = {};\r\n private static _ProfilesList: Nullable>;\r\n\r\n /**\r\n * The base URL of the online controller repository. Can be changed at any time.\r\n */\r\n public static BaseRepositoryUrl = \"https://immersive-web.github.io/webxr-input-profiles/packages/viewer/dist\";\r\n /**\r\n * Which repository gets priority - local or online\r\n */\r\n public static PrioritizeOnlineRepository: boolean = true;\r\n /**\r\n * Use the online repository, or use only locally-defined controllers\r\n */\r\n public static UseOnlineRepository: boolean = true;\r\n\r\n /**\r\n * Disable the controller cache and load the models each time a new WebXRProfileMotionController is loaded.\r\n * Defaults to true.\r\n */\r\n public static DisableControllerCache: boolean = true;\r\n\r\n /**\r\n * Clear the cache used for profile loading and reload when requested again\r\n */\r\n public static ClearProfilesCache() {\r\n this._ProfilesList = null;\r\n this._ProfileLoadingPromises = {};\r\n }\r\n\r\n /**\r\n * Register the default fallbacks.\r\n * This function is called automatically when this file is imported.\r\n */\r\n public static DefaultFallbacks() {\r\n this.RegisterFallbacksForProfileId(\"google-daydream\", [\"generic-touchpad\"]);\r\n this.RegisterFallbacksForProfileId(\"htc-vive-focus\", [\"generic-trigger-touchpad\"]);\r\n this.RegisterFallbacksForProfileId(\"htc-vive\", [\"generic-trigger-squeeze-touchpad\"]);\r\n this.RegisterFallbacksForProfileId(\"magicleap-one\", [\"generic-trigger-squeeze-touchpad\"]);\r\n this.RegisterFallbacksForProfileId(\"windows-mixed-reality\", [\"generic-trigger-squeeze-touchpad-thumbstick\"]);\r\n this.RegisterFallbacksForProfileId(\"microsoft-mixed-reality\", [\"windows-mixed-reality\", \"generic-trigger-squeeze-touchpad-thumbstick\"]);\r\n this.RegisterFallbacksForProfileId(\"oculus-go\", [\"generic-trigger-touchpad\"]);\r\n this.RegisterFallbacksForProfileId(\"oculus-touch-v2\", [\"oculus-touch\", \"generic-trigger-squeeze-thumbstick\"]);\r\n this.RegisterFallbacksForProfileId(\"oculus-touch\", [\"generic-trigger-squeeze-thumbstick\"]);\r\n this.RegisterFallbacksForProfileId(\"samsung-gearvr\", [\"windows-mixed-reality\", \"generic-trigger-squeeze-touchpad-thumbstick\"]);\r\n this.RegisterFallbacksForProfileId(\"samsung-odyssey\", [\"generic-touchpad\"]);\r\n this.RegisterFallbacksForProfileId(\"valve-index\", [\"generic-trigger-squeeze-touchpad-thumbstick\"]);\r\n this.RegisterFallbacksForProfileId(\"generic-hand-select\", [\"generic-trigger\"]);\r\n }\r\n\r\n /**\r\n * Find a fallback profile if the profile was not found. There are a few predefined generic profiles.\r\n * @param profileId the profile to which a fallback needs to be found\r\n * @returns an array with corresponding fallback profiles\r\n */\r\n public static FindFallbackWithProfileId(profileId: string): string[] {\r\n const returnArray = this._Fallbacks[profileId] || [];\r\n\r\n returnArray.unshift(profileId);\r\n return returnArray;\r\n }\r\n\r\n /**\r\n * When acquiring a new xrInput object (usually by the WebXRInput class), match it with the correct profile.\r\n * The order of search:\r\n *\r\n * 1) Iterate the profiles array of the xr input and try finding a corresponding motion controller\r\n * 2) (If not found) search in the gamepad id and try using it (legacy versions only)\r\n * 3) search for registered fallbacks (should be redundant, nonetheless it makes sense to check)\r\n * 4) return the generic trigger controller if none were found\r\n *\r\n * @param xrInput the xrInput to which a new controller is initialized\r\n * @param scene the scene to which the model will be added\r\n * @param forceProfile force a certain profile for this controller\r\n * @returns A promise that fulfils with the motion controller class for this profile id or the generic standard class if none was found\r\n */\r\n public static GetMotionControllerWithXRInput(xrInput: XRInputSource, scene: Scene, forceProfile?: string): Promise {\r\n const profileArray: string[] = [];\r\n if (forceProfile) {\r\n profileArray.push(forceProfile);\r\n }\r\n profileArray.push(...(xrInput.profiles || []));\r\n\r\n // emulator support\r\n if (profileArray.length && !profileArray[0]) {\r\n // remove the first \"undefined\" that the emulator is adding\r\n profileArray.pop();\r\n }\r\n\r\n // legacy support - try using the gamepad id\r\n if (xrInput.gamepad && xrInput.gamepad.id) {\r\n switch (xrInput.gamepad.id) {\r\n case xrInput.gamepad.id.match(/oculus touch/gi) ? xrInput.gamepad.id : undefined:\r\n // oculus in gamepad id\r\n profileArray.push(\"oculus-touch-v2\");\r\n break;\r\n }\r\n }\r\n\r\n // make sure microsoft/windows mixed reality works correctly\r\n const windowsMRIdx = profileArray.indexOf(\"windows-mixed-reality\");\r\n if (windowsMRIdx !== -1) {\r\n profileArray.splice(windowsMRIdx, 0, \"microsoft-mixed-reality\");\r\n }\r\n\r\n if (!profileArray.length) {\r\n profileArray.push(\"generic-trigger\");\r\n }\r\n\r\n if (this.UseOnlineRepository) {\r\n const firstFunction = this.PrioritizeOnlineRepository ? this._LoadProfileFromRepository : this._LoadProfilesFromAvailableControllers;\r\n const secondFunction = this.PrioritizeOnlineRepository ? this._LoadProfilesFromAvailableControllers : this._LoadProfileFromRepository;\r\n\r\n return firstFunction.call(this, profileArray, xrInput, scene).catch(() => {\r\n return secondFunction.call(this, profileArray, xrInput, scene);\r\n });\r\n } else {\r\n // use only available functions\r\n return this._LoadProfilesFromAvailableControllers(profileArray, xrInput, scene);\r\n }\r\n }\r\n\r\n /**\r\n * Register a new controller based on its profile. This function will be called by the controller classes themselves.\r\n *\r\n * If you are missing a profile, make sure it is imported in your source, otherwise it will not register.\r\n *\r\n * @param type the profile type to register\r\n * @param constructFunction the function to be called when loading this profile\r\n */\r\n public static RegisterController(type: string, constructFunction: MotionControllerConstructor) {\r\n this._AvailableControllers[type] = constructFunction;\r\n }\r\n\r\n /**\r\n * Register a fallback to a specific profile.\r\n * @param profileId the profileId that will receive the fallbacks\r\n * @param fallbacks A list of fallback profiles\r\n */\r\n public static RegisterFallbacksForProfileId(profileId: string, fallbacks: string[]): void {\r\n if (this._Fallbacks[profileId]) {\r\n this._Fallbacks[profileId].push(...fallbacks);\r\n } else {\r\n this._Fallbacks[profileId] = fallbacks;\r\n }\r\n }\r\n\r\n /**\r\n * Will update the list of profiles available in the repository\r\n * @returns a promise that resolves to a map of profiles available online\r\n */\r\n public static UpdateProfilesList() {\r\n this._ProfilesList = Tools.LoadFileAsync(this.BaseRepositoryUrl + \"/profiles/profilesList.json\", false).then((data) => {\r\n return JSON.parse(data.toString());\r\n });\r\n return this._ProfilesList;\r\n }\r\n\r\n /**\r\n * Clear the controller's cache (usually happens at the end of a session)\r\n */\r\n public static ClearControllerCache() {\r\n controllerCache.forEach((cacheItem) => {\r\n cacheItem.meshes.forEach((mesh) => {\r\n mesh.dispose(false, true);\r\n });\r\n });\r\n controllerCache.length = 0;\r\n }\r\n\r\n private static _LoadProfileFromRepository(profileArray: string[], xrInput: XRInputSource, scene: Scene): Promise {\r\n return Promise.resolve()\r\n .then(() => {\r\n if (!this._ProfilesList) {\r\n return this.UpdateProfilesList();\r\n } else {\r\n return this._ProfilesList;\r\n }\r\n })\r\n .then((profilesList: { [profile: string]: string }) => {\r\n // load the right profile\r\n for (let i = 0; i < profileArray.length; ++i) {\r\n // defensive\r\n if (!profileArray[i]) {\r\n continue;\r\n }\r\n if (profilesList[profileArray[i]]) {\r\n return profileArray[i];\r\n }\r\n }\r\n\r\n throw new Error(`neither controller ${profileArray[0]} nor all fallbacks were found in the repository,`);\r\n })\r\n .then((profileToLoad: string) => {\r\n // load the profile\r\n if (!this._ProfileLoadingPromises[profileToLoad]) {\r\n this._ProfileLoadingPromises[profileToLoad] = Tools.LoadFileAsync(`${this.BaseRepositoryUrl}/profiles/${profileToLoad}/profile.json`, false).then(\r\n (data) => JSON.parse(data as string)\r\n );\r\n }\r\n return this._ProfileLoadingPromises[profileToLoad];\r\n })\r\n .then((profile: IMotionControllerProfile) => {\r\n return new WebXRProfiledMotionController(scene, xrInput, profile, this.BaseRepositoryUrl, this.DisableControllerCache ? undefined : controllerCache);\r\n });\r\n }\r\n\r\n private static _LoadProfilesFromAvailableControllers(profileArray: string[], xrInput: XRInputSource, scene: Scene) {\r\n // check fallbacks\r\n for (let i = 0; i < profileArray.length; ++i) {\r\n // defensive\r\n if (!profileArray[i]) {\r\n continue;\r\n }\r\n const fallbacks = this.FindFallbackWithProfileId(profileArray[i]);\r\n for (let j = 0; j < fallbacks.length; ++j) {\r\n const constructionFunction = this._AvailableControllers[fallbacks[j]];\r\n if (constructionFunction) {\r\n return Promise.resolve(constructionFunction(xrInput, scene));\r\n }\r\n }\r\n }\r\n\r\n throw new Error(`no controller requested was found in the available controllers list`);\r\n }\r\n}\r\n\r\n// register the generic profile(s) here so we will at least have them\r\nWebXRMotionControllerManager.RegisterController(WebXRGenericTriggerMotionController.ProfileId, (xrInput: XRInputSource, scene: Scene) => {\r\n return new WebXRGenericTriggerMotionController(scene, xrInput.gamepad, xrInput.handedness);\r\n});\r\n\r\n// register fallbacks\r\nWebXRMotionControllerManager.DefaultFallbacks();\r\n","import { Observable } from \"../Misc/observable\";\r\nimport { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport { Quaternion, Vector3 } from \"../Maths/math.vector\";\r\nimport type { Ray } from \"../Culling/ray\";\r\nimport type { Scene } from \"../scene\";\r\nimport type { WebXRAbstractMotionController } from \"./motionController/webXRAbstractMotionController\";\r\nimport { WebXRMotionControllerManager } from \"./motionController/webXRMotionControllerManager\";\r\nimport { Tools } from \"../Misc/tools\";\r\nimport type { WebXRCamera } from \"./webXRCamera\";\r\n\r\nlet idCount = 0;\r\n\r\n/**\r\n * Configuration options for the WebXR controller creation\r\n */\r\nexport interface IWebXRControllerOptions {\r\n /**\r\n * Should the controller mesh be animated when a user interacts with it\r\n * The pressed buttons / thumbstick and touchpad animations will be disabled\r\n */\r\n disableMotionControllerAnimation?: boolean;\r\n /**\r\n * Do not load the controller mesh, in case a different mesh needs to be loaded.\r\n */\r\n doNotLoadControllerMesh?: boolean;\r\n /**\r\n * Force a specific controller type for this controller.\r\n * This can be used when creating your own profile or when testing different controllers\r\n */\r\n forceControllerProfile?: string;\r\n /**\r\n * Defines a rendering group ID for meshes that will be loaded.\r\n * This is for the default controllers only.\r\n */\r\n renderingGroupId?: number;\r\n}\r\n\r\n/**\r\n * Represents an XR controller\r\n */\r\nexport class WebXRInputSource {\r\n private _tmpVector = new Vector3();\r\n private _uniqueId: string;\r\n private _disposed = false;\r\n\r\n /**\r\n * Represents the part of the controller that is held. This may not exist if the controller is the head mounted display itself, if that's the case only the pointer from the head will be available\r\n */\r\n public grip?: AbstractMesh;\r\n /**\r\n * If available, this is the gamepad object related to this controller.\r\n * Using this object it is possible to get click events and trackpad changes of the\r\n * webxr controller that is currently being used.\r\n */\r\n public motionController?: WebXRAbstractMotionController;\r\n /**\r\n * Event that fires when the controller is removed/disposed.\r\n * The object provided as event data is this controller, after associated assets were disposed.\r\n * uniqueId is still available.\r\n */\r\n public onDisposeObservable = new Observable();\r\n /**\r\n * Will be triggered when the mesh associated with the motion controller is done loading.\r\n * It is also possible that this will never trigger (!) if no mesh was loaded, or if the developer decides to load a different mesh\r\n * A shortened version of controller -> motion controller -> on mesh loaded.\r\n */\r\n public onMeshLoadedObservable = new Observable();\r\n /**\r\n * Observers registered here will trigger when a motion controller profile was assigned to this xr controller\r\n */\r\n public onMotionControllerInitObservable = new Observable();\r\n /**\r\n * Pointer which can be used to select objects or attach a visible laser to\r\n */\r\n public pointer: AbstractMesh;\r\n\r\n /**\r\n * The last XRPose the was calculated on the current XRFrame\r\n * @internal\r\n */\r\n public _lastXRPose?: XRPose;\r\n\r\n /**\r\n * Creates the input source object\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/webXR/webXRInputControllerSupport\r\n * @param _scene the scene which the controller should be associated to\r\n * @param inputSource the underlying input source for the controller\r\n * @param _options options for this controller creation\r\n */\r\n constructor(\r\n private _scene: Scene,\r\n /** The underlying input source for the controller */\r\n public inputSource: XRInputSource,\r\n private _options: IWebXRControllerOptions = {}\r\n ) {\r\n this._uniqueId = `controller-${idCount++}-${inputSource.targetRayMode}-${inputSource.handedness}`;\r\n\r\n this.pointer = new AbstractMesh(`${this._uniqueId}-pointer`, _scene);\r\n this.pointer.rotationQuaternion = new Quaternion();\r\n\r\n if (this.inputSource.gripSpace) {\r\n this.grip = new AbstractMesh(`${this._uniqueId}-grip`, this._scene);\r\n this.grip.rotationQuaternion = new Quaternion();\r\n }\r\n\r\n this._tmpVector.set(0, 0, this._scene.useRightHandedSystem ? -1.0 : 1.0);\r\n\r\n // for now only load motion controllers if gamepad object available\r\n if (this.inputSource.gamepad && this.inputSource.targetRayMode === \"tracked-pointer\") {\r\n WebXRMotionControllerManager.GetMotionControllerWithXRInput(inputSource, _scene, this._options.forceControllerProfile).then(\r\n (motionController) => {\r\n this.motionController = motionController;\r\n this.onMotionControllerInitObservable.notifyObservers(motionController);\r\n // should the model be loaded?\r\n if (!this._options.doNotLoadControllerMesh && !this.motionController._doNotLoadControllerMesh) {\r\n this.motionController.loadModel().then((success) => {\r\n if (success && this.motionController && this.motionController.rootMesh) {\r\n if (this._options.renderingGroupId) {\r\n // anything other than 0?\r\n this.motionController.rootMesh.renderingGroupId = this._options.renderingGroupId;\r\n this.motionController.rootMesh.getChildMeshes(false).forEach((mesh) => (mesh.renderingGroupId = this._options.renderingGroupId!));\r\n }\r\n this.onMeshLoadedObservable.notifyObservers(this.motionController.rootMesh);\r\n this.motionController.rootMesh.parent = this.grip || this.pointer;\r\n this.motionController.disableAnimation = !!this._options.disableMotionControllerAnimation;\r\n }\r\n // make sure to dispose is the controller is already disposed\r\n if (this._disposed) {\r\n this.motionController?.dispose();\r\n }\r\n });\r\n }\r\n },\r\n () => {\r\n Tools.Warn(`Could not find a matching motion controller for the registered input source`);\r\n }\r\n );\r\n }\r\n }\r\n\r\n /**\r\n * Get this controllers unique id\r\n */\r\n public get uniqueId() {\r\n return this._uniqueId;\r\n }\r\n\r\n /**\r\n * Disposes of the object\r\n */\r\n public dispose() {\r\n if (this.grip) {\r\n this.grip.dispose(true);\r\n }\r\n if (this.motionController) {\r\n this.motionController.dispose();\r\n }\r\n this.pointer.dispose(true);\r\n this.onMotionControllerInitObservable.clear();\r\n this.onMeshLoadedObservable.clear();\r\n this.onDisposeObservable.notifyObservers(this);\r\n this.onDisposeObservable.clear();\r\n this._disposed = true;\r\n }\r\n\r\n /**\r\n * Gets a world space ray coming from the pointer or grip\r\n * @param result the resulting ray\r\n * @param gripIfAvailable use the grip mesh instead of the pointer, if available\r\n */\r\n public getWorldPointerRayToRef(result: Ray, gripIfAvailable: boolean = false) {\r\n const object = gripIfAvailable && this.grip ? this.grip : this.pointer;\r\n Vector3.TransformNormalToRef(this._tmpVector, object.getWorldMatrix(), result.direction);\r\n result.direction.normalize();\r\n result.origin.copyFrom(object.absolutePosition);\r\n result.length = 1000;\r\n }\r\n\r\n /**\r\n * Updates the controller pose based on the given XRFrame\r\n * @param xrFrame xr frame to update the pose with\r\n * @param referenceSpace reference space to use\r\n * @param xrCamera the xr camera, used for parenting\r\n */\r\n public updateFromXRFrame(xrFrame: XRFrame, referenceSpace: XRReferenceSpace, xrCamera: WebXRCamera) {\r\n const pose = xrFrame.getPose(this.inputSource.targetRaySpace, referenceSpace);\r\n this._lastXRPose = pose;\r\n\r\n // Update the pointer mesh\r\n if (pose) {\r\n const pos = pose.transform.position;\r\n this.pointer.position.set(pos.x, pos.y, pos.z);\r\n const orientation = pose.transform.orientation;\r\n this.pointer.rotationQuaternion!.set(orientation.x, orientation.y, orientation.z, orientation.w);\r\n if (!this._scene.useRightHandedSystem) {\r\n this.pointer.position.z *= -1;\r\n this.pointer.rotationQuaternion!.z *= -1;\r\n this.pointer.rotationQuaternion!.w *= -1;\r\n }\r\n this.pointer.parent = xrCamera.parent;\r\n }\r\n\r\n // Update the grip mesh if it exists\r\n if (this.inputSource.gripSpace && this.grip) {\r\n const pose = xrFrame.getPose(this.inputSource.gripSpace, referenceSpace);\r\n if (pose) {\r\n const pos = pose.transform.position;\r\n const orientation = pose.transform.orientation;\r\n this.grip.position.set(pos.x, pos.y, pos.z);\r\n this.grip.rotationQuaternion!.set(orientation.x, orientation.y, orientation.z, orientation.w);\r\n if (!this._scene.useRightHandedSystem) {\r\n this.grip.position.z *= -1;\r\n this.grip.rotationQuaternion!.z *= -1;\r\n this.grip.rotationQuaternion!.w *= -1;\r\n }\r\n }\r\n this.grip.parent = xrCamera.parent;\r\n }\r\n if (this.motionController) {\r\n // either update buttons only or also position, if in gamepad mode\r\n this.motionController.updateFromXRFrame(xrFrame);\r\n }\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\nimport type { Observer } from \"../Misc/observable\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport type { IDisposable } from \"../scene\";\r\nimport type { IWebXRControllerOptions } from \"./webXRInputSource\";\r\nimport { WebXRInputSource } from \"./webXRInputSource\";\r\nimport type { WebXRSessionManager } from \"./webXRSessionManager\";\r\nimport type { WebXRCamera } from \"./webXRCamera\";\r\nimport { WebXRMotionControllerManager } from \"./motionController/webXRMotionControllerManager\";\r\n\r\n/**\r\n * The schema for initialization options of the XR Input class\r\n */\r\nexport interface IWebXRInputOptions {\r\n /**\r\n * If set to true no model will be automatically loaded\r\n */\r\n doNotLoadControllerMeshes?: boolean;\r\n\r\n /**\r\n * If set, this profile will be used for all controllers loaded (for example \"microsoft-mixed-reality\")\r\n * If not found, the xr input profile data will be used.\r\n * Profiles are defined here - https://github.com/immersive-web/webxr-input-profiles/\r\n */\r\n forceInputProfile?: string;\r\n\r\n /**\r\n * Do not send a request to the controller repository to load the profile.\r\n *\r\n * Instead, use the controllers available in babylon itself.\r\n */\r\n disableOnlineControllerRepository?: boolean;\r\n\r\n /**\r\n * A custom URL for the controllers repository\r\n */\r\n customControllersRepositoryURL?: string;\r\n\r\n /**\r\n * Should the controller model's components not move according to the user input\r\n */\r\n disableControllerAnimation?: boolean;\r\n\r\n /**\r\n * Optional options to pass to the controller. Will be overridden by the Input options where applicable\r\n */\r\n controllerOptions?: IWebXRControllerOptions;\r\n}\r\n/**\r\n * XR input used to track XR inputs such as controllers/rays\r\n */\r\nexport class WebXRInput implements IDisposable {\r\n /**\r\n * XR controllers being tracked\r\n */\r\n public controllers: Array = [];\r\n private _frameObserver: Nullable>;\r\n private _sessionEndedObserver: Nullable>;\r\n private _sessionInitObserver: Nullable>;\r\n /**\r\n * Event when a controller has been connected/added\r\n */\r\n public onControllerAddedObservable = new Observable();\r\n /**\r\n * Event when a controller has been removed/disconnected\r\n */\r\n public onControllerRemovedObservable = new Observable();\r\n\r\n /**\r\n * Initializes the WebXRInput\r\n * @param xrSessionManager the xr session manager for this session\r\n * @param xrCamera the WebXR camera for this session. Mainly used for teleportation\r\n * @param _options = initialization options for this xr input\r\n */\r\n public constructor(\r\n /**\r\n * the xr session manager for this session\r\n */\r\n public xrSessionManager: WebXRSessionManager,\r\n /**\r\n * the WebXR camera for this session. Mainly used for teleportation\r\n */\r\n public xrCamera: WebXRCamera,\r\n private readonly _options: IWebXRInputOptions = {}\r\n ) {\r\n // Remove controllers when exiting XR\r\n this._sessionEndedObserver = this.xrSessionManager.onXRSessionEnded.add(() => {\r\n this._addAndRemoveControllers(\r\n [],\r\n this.controllers.map((c) => {\r\n return c.inputSource;\r\n })\r\n );\r\n });\r\n\r\n this._sessionInitObserver = this.xrSessionManager.onXRSessionInit.add((session) => {\r\n session.addEventListener(\"inputsourceschange\", this._onInputSourcesChange);\r\n });\r\n\r\n this._frameObserver = this.xrSessionManager.onXRFrameObservable.add((frame) => {\r\n // Update controller pose info\r\n this.controllers.forEach((controller) => {\r\n controller.updateFromXRFrame(frame, this.xrSessionManager.referenceSpace, this.xrCamera);\r\n });\r\n });\r\n\r\n if (this._options.customControllersRepositoryURL) {\r\n WebXRMotionControllerManager.BaseRepositoryUrl = this._options.customControllersRepositoryURL;\r\n }\r\n\r\n WebXRMotionControllerManager.UseOnlineRepository = !this._options.disableOnlineControllerRepository;\r\n if (WebXRMotionControllerManager.UseOnlineRepository) {\r\n // pre-load the profiles list to load the controllers quicker afterwards\r\n try {\r\n WebXRMotionControllerManager.UpdateProfilesList().catch(() => {\r\n WebXRMotionControllerManager.UseOnlineRepository = false;\r\n });\r\n } catch (e) {\r\n WebXRMotionControllerManager.UseOnlineRepository = false;\r\n }\r\n }\r\n }\r\n\r\n private _onInputSourcesChange = (event: XRInputSourceChangeEvent) => {\r\n this._addAndRemoveControllers(event.added, event.removed);\r\n };\r\n\r\n private _addAndRemoveControllers(addInputs: readonly XRInputSource[], removeInputs: readonly XRInputSource[]) {\r\n // Add controllers if they don't already exist\r\n const sources = this.controllers.map((c) => {\r\n return c.inputSource;\r\n });\r\n for (const input of addInputs) {\r\n if (sources.indexOf(input) === -1) {\r\n const controller = new WebXRInputSource(this.xrSessionManager.scene, input, {\r\n ...(this._options.controllerOptions || {}),\r\n forceControllerProfile: this._options.forceInputProfile,\r\n doNotLoadControllerMesh: this._options.doNotLoadControllerMeshes,\r\n disableMotionControllerAnimation: this._options.disableControllerAnimation,\r\n });\r\n this.controllers.push(controller);\r\n this.onControllerAddedObservable.notifyObservers(controller);\r\n }\r\n }\r\n\r\n // Remove and dispose of controllers to be disposed\r\n const keepControllers: Array = [];\r\n const removedControllers: Array = [];\r\n this.controllers.forEach((c) => {\r\n if (removeInputs.indexOf(c.inputSource) === -1) {\r\n keepControllers.push(c);\r\n } else {\r\n removedControllers.push(c);\r\n }\r\n });\r\n this.controllers = keepControllers;\r\n removedControllers.forEach((c) => {\r\n this.onControllerRemovedObservable.notifyObservers(c);\r\n c.dispose();\r\n });\r\n }\r\n\r\n /**\r\n * Disposes of the object\r\n */\r\n public dispose() {\r\n this.controllers.forEach((c) => {\r\n c.dispose();\r\n });\r\n this.xrSessionManager.onXRFrameObservable.remove(this._frameObserver);\r\n this.xrSessionManager.onXRSessionInit.remove(this._sessionInitObserver);\r\n this.xrSessionManager.onXRSessionEnded.remove(this._sessionEndedObserver);\r\n this.onControllerAddedObservable.clear();\r\n this.onControllerRemovedObservable.clear();\r\n\r\n // clear the controller cache\r\n WebXRMotionControllerManager.ClearControllerCache();\r\n }\r\n}\r\n","import type { IWebXRFeature } from \"../webXRFeaturesManager\";\r\nimport type { Observer, Observable, EventState } from \"../../Misc/observable\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { WebXRSessionManager } from \"../webXRSessionManager\";\r\n\r\n/**\r\n * This is the base class for all WebXR features.\r\n * Since most features require almost the same resources and callbacks, this class can be used to simplify the development\r\n * Note that since the features manager is using the `IWebXRFeature` you are in no way obligated to use this class\r\n */\r\nexport abstract class WebXRAbstractFeature implements IWebXRFeature {\r\n private _attached: boolean = false;\r\n private _removeOnDetach: {\r\n observer: Nullable>;\r\n observable: Observable;\r\n }[] = [];\r\n\r\n /**\r\n * Is this feature disposed?\r\n */\r\n public isDisposed: boolean = false;\r\n\r\n /**\r\n * Should auto-attach be disabled?\r\n */\r\n public disableAutoAttach: boolean = false;\r\n\r\n /**\r\n * The name of the native xr feature name (like anchor, hit-test, or hand-tracking)\r\n */\r\n public xrNativeFeatureName: string = \"\";\r\n\r\n /**\r\n * Construct a new (abstract) WebXR feature\r\n * @param _xrSessionManager the xr session manager for this feature\r\n */\r\n constructor(protected _xrSessionManager: WebXRSessionManager) {}\r\n\r\n /**\r\n * Is this feature attached\r\n */\r\n public get attached() {\r\n return this._attached;\r\n }\r\n\r\n /**\r\n * attach this feature\r\n *\r\n * @param force should attachment be forced (even when already attached)\r\n * @returns true if successful, false is failed or already attached\r\n */\r\n public attach(force?: boolean): boolean {\r\n // do not attach a disposed feature\r\n if (this.isDisposed) {\r\n return false;\r\n }\r\n if (!force) {\r\n if (this.attached) {\r\n return false;\r\n }\r\n } else {\r\n if (this.attached) {\r\n // detach first, to be sure\r\n this.detach();\r\n }\r\n }\r\n\r\n this._attached = true;\r\n this._addNewAttachObserver(this._xrSessionManager.onXRFrameObservable, (frame) => this._onXRFrame(frame));\r\n return true;\r\n }\r\n\r\n /**\r\n * detach this feature.\r\n *\r\n * @returns true if successful, false if failed or already detached\r\n */\r\n public detach(): boolean {\r\n if (!this._attached) {\r\n this.disableAutoAttach = true;\r\n return false;\r\n }\r\n this._attached = false;\r\n this._removeOnDetach.forEach((toRemove) => {\r\n toRemove.observable.remove(toRemove.observer);\r\n });\r\n return true;\r\n }\r\n\r\n /**\r\n * Dispose this feature and all of the resources attached\r\n */\r\n public dispose(): void {\r\n this.detach();\r\n this.isDisposed = true;\r\n }\r\n\r\n /**\r\n * This function will be executed during before enabling the feature and can be used to not-allow enabling it.\r\n * Note that at this point the session has NOT started, so this is purely checking if the browser supports it\r\n *\r\n * @returns whether or not the feature is compatible in this environment\r\n */\r\n public isCompatible(): boolean {\r\n return true;\r\n }\r\n\r\n /**\r\n * This is used to register callbacks that will automatically be removed when detach is called.\r\n * @param observable the observable to which the observer will be attached\r\n * @param callback the callback to register\r\n */\r\n protected _addNewAttachObserver(observable: Observable, callback: (eventData: T, eventState: EventState) => void) {\r\n this._removeOnDetach.push({\r\n observable,\r\n observer: observable.add(callback),\r\n });\r\n }\r\n\r\n /**\r\n * Code in this function will be executed on each xrFrame received from the browser.\r\n * This function will not execute after the feature is detached.\r\n * @param _xrFrame the current frame\r\n */\r\n protected abstract _onXRFrame(_xrFrame: XRFrame): void;\r\n}\r\n","import type { IDisposable } from \"../scene\";\r\nimport { Scene } from \"../scene\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { Observer } from \"../Misc/observable\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport type { PointerInfoPre } from \"../Events/pointerEvents\";\r\nimport { PointerInfo, PointerEventTypes } from \"../Events/pointerEvents\";\r\nimport { PickingInfo } from \"../Collisions/pickingInfo\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport { EngineStore } from \"../Engines/engineStore\";\r\nimport { HemisphericLight } from \"../Lights/hemisphericLight\";\r\nimport { Vector3 } from \"../Maths/math.vector\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\nimport { Color3 } from \"../Maths/math.color\";\r\nimport type { IPointerEvent } from \"../Events/deviceInputEvents\";\r\n\r\n/**\r\n * Renders a layer on top of an existing scene\r\n */\r\nexport class UtilityLayerRenderer implements IDisposable {\r\n private _pointerCaptures: { [pointerId: number]: boolean } = {};\r\n private _lastPointerEvents: { [pointerId: number]: boolean } = {};\r\n /** @internal */\r\n public static _DefaultUtilityLayer: Nullable = null;\r\n /** @internal */\r\n public static _DefaultKeepDepthUtilityLayer: Nullable = null;\r\n private _sharedGizmoLight: Nullable = null;\r\n\r\n private _renderCamera: Nullable = null;\r\n\r\n /**\r\n * Gets the camera that is used to render the utility layer (when not set, this will be the last active camera)\r\n * @param getRigParentIfPossible if the current active camera is a rig camera, should its parent camera be returned\r\n * @returns the camera that is used when rendering the utility layer\r\n */\r\n public getRenderCamera(getRigParentIfPossible?: boolean) {\r\n if (this._renderCamera) {\r\n return this._renderCamera;\r\n } else {\r\n let activeCam: Camera;\r\n if (this.originalScene.activeCameras && this.originalScene.activeCameras.length > 1) {\r\n activeCam = this.originalScene.activeCameras[this.originalScene.activeCameras.length - 1];\r\n } else {\r\n activeCam = this.originalScene.activeCamera!;\r\n }\r\n\r\n if (getRigParentIfPossible && activeCam && activeCam.isRigCamera) {\r\n return activeCam.rigParent!;\r\n }\r\n return activeCam;\r\n }\r\n }\r\n /**\r\n * Sets the camera that should be used when rendering the utility layer (If set to null the last active camera will be used)\r\n * @param cam the camera that should be used when rendering the utility layer\r\n */\r\n public setRenderCamera(cam: Nullable) {\r\n this._renderCamera = cam;\r\n }\r\n\r\n /**\r\n * @internal\r\n * Light which used by gizmos to get light shading\r\n */\r\n public _getSharedGizmoLight(): HemisphericLight {\r\n if (!this._sharedGizmoLight) {\r\n this._sharedGizmoLight = new HemisphericLight(\"shared gizmo light\", new Vector3(0, 1, 0), this.utilityLayerScene);\r\n this._sharedGizmoLight.intensity = 2;\r\n this._sharedGizmoLight.groundColor = Color3.Gray();\r\n }\r\n return this._sharedGizmoLight;\r\n }\r\n\r\n /**\r\n * If the picking should be done on the utility layer prior to the actual scene (Default: true)\r\n */\r\n public pickUtilitySceneFirst = true;\r\n /**\r\n * A shared utility layer that can be used to overlay objects into a scene (Depth map of the previous scene is cleared before drawing on top of it)\r\n */\r\n public static get DefaultUtilityLayer(): UtilityLayerRenderer {\r\n if (UtilityLayerRenderer._DefaultUtilityLayer == null) {\r\n return UtilityLayerRenderer._CreateDefaultUtilityLayerFromScene(EngineStore.LastCreatedScene!);\r\n }\r\n\r\n return UtilityLayerRenderer._DefaultUtilityLayer;\r\n }\r\n\r\n /**\r\n * Creates an utility layer, and set it as a default utility layer\r\n * @param scene associated scene\r\n * @internal\r\n */\r\n public static _CreateDefaultUtilityLayerFromScene(scene: Scene): UtilityLayerRenderer {\r\n UtilityLayerRenderer._DefaultUtilityLayer = new UtilityLayerRenderer(scene);\r\n UtilityLayerRenderer._DefaultUtilityLayer.originalScene.onDisposeObservable.addOnce(() => {\r\n UtilityLayerRenderer._DefaultUtilityLayer = null;\r\n });\r\n\r\n return UtilityLayerRenderer._DefaultUtilityLayer;\r\n }\r\n /**\r\n * A shared utility layer that can be used to embed objects into a scene (Depth map of the previous scene is not cleared before drawing on top of it)\r\n */\r\n public static get DefaultKeepDepthUtilityLayer(): UtilityLayerRenderer {\r\n if (UtilityLayerRenderer._DefaultKeepDepthUtilityLayer == null) {\r\n UtilityLayerRenderer._DefaultKeepDepthUtilityLayer = new UtilityLayerRenderer(EngineStore.LastCreatedScene!);\r\n UtilityLayerRenderer._DefaultKeepDepthUtilityLayer.utilityLayerScene.autoClearDepthAndStencil = false;\r\n UtilityLayerRenderer._DefaultKeepDepthUtilityLayer.originalScene.onDisposeObservable.addOnce(() => {\r\n UtilityLayerRenderer._DefaultKeepDepthUtilityLayer = null;\r\n });\r\n }\r\n return UtilityLayerRenderer._DefaultKeepDepthUtilityLayer;\r\n }\r\n\r\n /**\r\n * The scene that is rendered on top of the original scene\r\n */\r\n public utilityLayerScene: Scene;\r\n\r\n /**\r\n * If the utility layer should automatically be rendered on top of existing scene\r\n */\r\n public shouldRender: boolean = true;\r\n /**\r\n * If set to true, only pointer down onPointerObservable events will be blocked when picking is occluded by original scene\r\n */\r\n public onlyCheckPointerDownEvents = true;\r\n\r\n /**\r\n * If set to false, only pointerUp, pointerDown and pointerMove will be sent to the utilityLayerScene (false by default)\r\n */\r\n public processAllEvents = false;\r\n\r\n /**\r\n * Set to false to disable picking\r\n */\r\n public pickingEnabled = true;\r\n\r\n /**\r\n * Observable raised when the pointer moves from the utility layer scene to the main scene\r\n */\r\n public onPointerOutObservable = new Observable();\r\n\r\n /** Gets or sets a predicate that will be used to indicate utility meshes present in the main scene */\r\n public mainSceneTrackerPredicate: (mesh: Nullable) => boolean;\r\n\r\n private _afterRenderObserver: Nullable>;\r\n private _sceneDisposeObserver: Nullable>;\r\n private _originalPointerObserver: Nullable>;\r\n /**\r\n * Instantiates a UtilityLayerRenderer\r\n * @param originalScene the original scene that will be rendered on top of\r\n * @param handleEvents boolean indicating if the utility layer should handle events\r\n */\r\n constructor(\r\n /** the original scene that will be rendered on top of */\r\n public originalScene: Scene,\r\n handleEvents: boolean = true\r\n ) {\r\n // Create scene which will be rendered in the foreground and remove it from being referenced by engine to avoid interfering with existing app\r\n this.utilityLayerScene = new Scene(originalScene.getEngine(), { virtual: true });\r\n this.utilityLayerScene.useRightHandedSystem = originalScene.useRightHandedSystem;\r\n this.utilityLayerScene._allowPostProcessClearColor = false;\r\n\r\n // Deactivate post processes\r\n this.utilityLayerScene.postProcessesEnabled = false;\r\n\r\n // Detach controls on utility scene, events will be fired by logic below to handle picking priority\r\n this.utilityLayerScene.detachControl();\r\n\r\n if (handleEvents) {\r\n this._originalPointerObserver = originalScene.onPrePointerObservable.add((prePointerInfo) => {\r\n if (!this.utilityLayerScene.activeCamera) {\r\n return;\r\n }\r\n if (!this.pickingEnabled) {\r\n return;\r\n }\r\n\r\n if (!this.processAllEvents) {\r\n if (\r\n prePointerInfo.type !== PointerEventTypes.POINTERMOVE &&\r\n prePointerInfo.type !== PointerEventTypes.POINTERUP &&\r\n prePointerInfo.type !== PointerEventTypes.POINTERDOWN &&\r\n prePointerInfo.type !== PointerEventTypes.POINTERDOUBLETAP\r\n ) {\r\n return;\r\n }\r\n }\r\n this.utilityLayerScene.pointerX = originalScene.pointerX;\r\n this.utilityLayerScene.pointerY = originalScene.pointerY;\r\n const pointerEvent = prePointerInfo.event;\r\n if (originalScene!.isPointerCaptured(pointerEvent.pointerId)) {\r\n this._pointerCaptures[pointerEvent.pointerId] = false;\r\n return;\r\n }\r\n\r\n const getNearPickDataForScene = (scene: Scene) => {\r\n let scenePick = null;\r\n\r\n if (prePointerInfo.nearInteractionPickingInfo) {\r\n if (prePointerInfo.nearInteractionPickingInfo.pickedMesh!.getScene() == scene) {\r\n scenePick = prePointerInfo.nearInteractionPickingInfo;\r\n } else {\r\n scenePick = new PickingInfo();\r\n }\r\n } else if (scene !== this.utilityLayerScene && prePointerInfo.originalPickingInfo) {\r\n scenePick = prePointerInfo.originalPickingInfo;\r\n } else {\r\n let previousActiveCamera: Nullable = null;\r\n // If a camera is set for rendering with this layer\r\n // it will also be used for the ray computation\r\n // To preserve back compat and because scene.pick always use activeCamera\r\n // it's substituted temporarily and a new scenePick is forced.\r\n // otherwise, the ray with previously active camera is always used.\r\n // It's set back to previous activeCamera after operation.\r\n if (this._renderCamera) {\r\n previousActiveCamera = scene._activeCamera;\r\n scene._activeCamera = this._renderCamera;\r\n prePointerInfo.ray = null;\r\n }\r\n scenePick = prePointerInfo.ray ? scene.pickWithRay(prePointerInfo.ray) : scene.pick(originalScene.pointerX, originalScene.pointerY);\r\n if (previousActiveCamera) {\r\n scene._activeCamera = previousActiveCamera;\r\n }\r\n }\r\n\r\n return scenePick;\r\n };\r\n\r\n const utilityScenePick = getNearPickDataForScene(this.utilityLayerScene);\r\n\r\n if (!prePointerInfo.ray && utilityScenePick) {\r\n prePointerInfo.ray = utilityScenePick.ray;\r\n }\r\n\r\n // always fire the prepointer observable\r\n this.utilityLayerScene.onPrePointerObservable.notifyObservers(prePointerInfo);\r\n\r\n // allow every non pointer down event to flow to the utility layer\r\n if (this.onlyCheckPointerDownEvents && prePointerInfo.type != PointerEventTypes.POINTERDOWN) {\r\n if (!prePointerInfo.skipOnPointerObservable) {\r\n this.utilityLayerScene.onPointerObservable.notifyObservers(\r\n new PointerInfo(prePointerInfo.type, prePointerInfo.event, utilityScenePick),\r\n prePointerInfo.type\r\n );\r\n }\r\n if (prePointerInfo.type === PointerEventTypes.POINTERUP && this._pointerCaptures[pointerEvent.pointerId]) {\r\n this._pointerCaptures[pointerEvent.pointerId] = false;\r\n }\r\n return;\r\n }\r\n\r\n if (this.utilityLayerScene.autoClearDepthAndStencil || this.pickUtilitySceneFirst) {\r\n // If this layer is an overlay, check if this layer was hit and if so, skip pointer events for the main scene\r\n if (utilityScenePick && utilityScenePick.hit) {\r\n if (!prePointerInfo.skipOnPointerObservable) {\r\n this.utilityLayerScene.onPointerObservable.notifyObservers(\r\n new PointerInfo(prePointerInfo.type, prePointerInfo.event, utilityScenePick),\r\n prePointerInfo.type\r\n );\r\n }\r\n prePointerInfo.skipOnPointerObservable = true;\r\n }\r\n } else {\r\n const originalScenePick = getNearPickDataForScene(originalScene);\r\n const pointerEvent = prePointerInfo.event;\r\n\r\n // If the layer can be occluded by the original scene, only fire pointer events to the first layer that hit they ray\r\n if (originalScenePick && utilityScenePick) {\r\n // No pick in utility scene\r\n if (utilityScenePick.distance === 0 && originalScenePick.pickedMesh) {\r\n if (this.mainSceneTrackerPredicate && this.mainSceneTrackerPredicate(originalScenePick.pickedMesh)) {\r\n // We touched an utility mesh present in the main scene\r\n this._notifyObservers(prePointerInfo, originalScenePick, pointerEvent);\r\n prePointerInfo.skipOnPointerObservable = true;\r\n } else if (prePointerInfo.type === PointerEventTypes.POINTERDOWN) {\r\n this._pointerCaptures[pointerEvent.pointerId] = true;\r\n } else if (prePointerInfo.type === PointerEventTypes.POINTERMOVE || prePointerInfo.type === PointerEventTypes.POINTERUP) {\r\n if (this._lastPointerEvents[pointerEvent.pointerId]) {\r\n // We need to send a last pointerup to the utilityLayerScene to make sure animations can complete\r\n this.onPointerOutObservable.notifyObservers(pointerEvent.pointerId);\r\n delete this._lastPointerEvents[pointerEvent.pointerId];\r\n }\r\n this._notifyObservers(prePointerInfo, originalScenePick, pointerEvent);\r\n }\r\n } else if (!this._pointerCaptures[pointerEvent.pointerId] && (utilityScenePick.distance < originalScenePick.distance || originalScenePick.distance === 0)) {\r\n // We pick something in utility scene or the pick in utility is closer than the one in main scene\r\n this._notifyObservers(prePointerInfo, utilityScenePick, pointerEvent);\r\n // If a previous utility layer set this, do not unset this\r\n if (!prePointerInfo.skipOnPointerObservable) {\r\n prePointerInfo.skipOnPointerObservable = utilityScenePick.distance > 0;\r\n }\r\n } else if (!this._pointerCaptures[pointerEvent.pointerId] && utilityScenePick.distance >= originalScenePick.distance) {\r\n // We have a pick in both scenes but main is closer than utility\r\n\r\n // We touched an utility mesh present in the main scene\r\n if (this.mainSceneTrackerPredicate && this.mainSceneTrackerPredicate(originalScenePick.pickedMesh)) {\r\n this._notifyObservers(prePointerInfo, originalScenePick, pointerEvent);\r\n prePointerInfo.skipOnPointerObservable = true;\r\n } else {\r\n if (prePointerInfo.type === PointerEventTypes.POINTERMOVE || prePointerInfo.type === PointerEventTypes.POINTERUP) {\r\n if (this._lastPointerEvents[pointerEvent.pointerId]) {\r\n // We need to send a last pointerup to the utilityLayerScene to make sure animations can complete\r\n this.onPointerOutObservable.notifyObservers(pointerEvent.pointerId);\r\n delete this._lastPointerEvents[pointerEvent.pointerId];\r\n }\r\n }\r\n this._notifyObservers(prePointerInfo, utilityScenePick, pointerEvent);\r\n }\r\n }\r\n\r\n if (prePointerInfo.type === PointerEventTypes.POINTERUP && this._pointerCaptures[pointerEvent.pointerId]) {\r\n this._pointerCaptures[pointerEvent.pointerId] = false;\r\n }\r\n }\r\n }\r\n });\r\n\r\n // As a newly added utility layer will be rendered over the screen last, it's pointer events should be processed first\r\n if (this._originalPointerObserver) {\r\n originalScene.onPrePointerObservable.makeObserverTopPriority(this._originalPointerObserver);\r\n }\r\n }\r\n\r\n // Render directly on top of existing scene without clearing\r\n this.utilityLayerScene.autoClear = false;\r\n\r\n this._afterRenderObserver = this.originalScene.onAfterRenderCameraObservable.add((camera) => {\r\n // Only render when the render camera finishes rendering\r\n if (this.shouldRender && camera == this.getRenderCamera()) {\r\n this.render();\r\n }\r\n });\r\n\r\n this._sceneDisposeObserver = this.originalScene.onDisposeObservable.add(() => {\r\n this.dispose();\r\n });\r\n\r\n this._updateCamera();\r\n }\r\n\r\n private _notifyObservers(prePointerInfo: PointerInfoPre, pickInfo: PickingInfo, pointerEvent: IPointerEvent) {\r\n if (!prePointerInfo.skipOnPointerObservable) {\r\n this.utilityLayerScene.onPointerObservable.notifyObservers(new PointerInfo(prePointerInfo.type, prePointerInfo.event, pickInfo), prePointerInfo.type);\r\n this._lastPointerEvents[pointerEvent.pointerId] = true;\r\n }\r\n }\r\n\r\n /**\r\n * Renders the utility layers scene on top of the original scene\r\n */\r\n public render() {\r\n this._updateCamera();\r\n if (this.utilityLayerScene.activeCamera) {\r\n // Set the camera's scene to utility layers scene\r\n const oldScene = this.utilityLayerScene.activeCamera.getScene();\r\n const camera = this.utilityLayerScene.activeCamera;\r\n camera._scene = this.utilityLayerScene;\r\n if (camera.leftCamera) {\r\n camera.leftCamera._scene = this.utilityLayerScene;\r\n }\r\n if (camera.rightCamera) {\r\n camera.rightCamera._scene = this.utilityLayerScene;\r\n }\r\n\r\n this.utilityLayerScene.render(false);\r\n\r\n // Reset camera's scene back to original\r\n camera._scene = oldScene;\r\n if (camera.leftCamera) {\r\n camera.leftCamera._scene = oldScene;\r\n }\r\n if (camera.rightCamera) {\r\n camera.rightCamera._scene = oldScene;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Disposes of the renderer\r\n */\r\n public dispose() {\r\n this.onPointerOutObservable.clear();\r\n\r\n if (this._afterRenderObserver) {\r\n this.originalScene.onAfterCameraRenderObservable.remove(this._afterRenderObserver);\r\n }\r\n if (this._sceneDisposeObserver) {\r\n this.originalScene.onDisposeObservable.remove(this._sceneDisposeObserver);\r\n }\r\n if (this._originalPointerObserver) {\r\n this.originalScene.onPrePointerObservable.remove(this._originalPointerObserver);\r\n }\r\n this.utilityLayerScene.dispose();\r\n }\r\n\r\n private _updateCamera() {\r\n this.utilityLayerScene.cameraToUseForPointers = this.getRenderCamera();\r\n this.utilityLayerScene.activeCamera = this.getRenderCamera();\r\n }\r\n}\r\n","import { WebXRFeaturesManager, WebXRFeatureName } from \"../webXRFeaturesManager\";\r\nimport type { WebXRSessionManager } from \"../webXRSessionManager\";\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport type { Observer } from \"../../Misc/observable\";\r\nimport type { WebXRInput } from \"../webXRInput\";\r\nimport type { WebXRInputSource } from \"../webXRInputSource\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { WebXRControllerComponent } from \"../motionController/webXRControllerComponent\";\r\nimport type { Nullable } from \"../../types\";\r\nimport { Matrix, Vector3 } from \"../../Maths/math.vector\";\r\nimport { Color3 } from \"../../Maths/math.color\";\r\nimport { Axis } from \"../../Maths/math.axis\";\r\nimport { StandardMaterial } from \"../../Materials/standardMaterial\";\r\nimport { CreateCylinder } from \"../../Meshes/Builders/cylinderBuilder\";\r\nimport { CreateTorus } from \"../../Meshes/Builders/torusBuilder\";\r\nimport { Ray } from \"../../Culling/ray\";\r\nimport { PickingInfo } from \"../../Collisions/pickingInfo\";\r\nimport { WebXRAbstractFeature } from \"./WebXRAbstractFeature\";\r\nimport { UtilityLayerRenderer } from \"../../Rendering/utilityLayerRenderer\";\r\nimport type { WebXRAbstractMotionController } from \"../motionController/webXRAbstractMotionController\";\r\nimport type { WebXRCamera } from \"../webXRCamera\";\r\nimport type { Node } from \"../../node\";\r\nimport { Viewport } from \"../../Maths/math.viewport\";\r\nimport type { Mesh } from \"../../Meshes/mesh\";\r\nimport { Tools } from \"../../Misc/tools\";\r\n\r\n/**\r\n * Options interface for the pointer selection module\r\n */\r\nexport interface IWebXRControllerPointerSelectionOptions {\r\n /**\r\n * if provided, this scene will be used to render meshes.\r\n */\r\n customUtilityLayerScene?: Scene;\r\n /**\r\n * Disable the pointer up event when the xr controller in screen and gaze mode is disposed (meaning - when the user removed the finger from the screen)\r\n * If not disabled, the last picked point will be used to execute a pointer up event\r\n * If disabled, pointer up event will be triggered right after the pointer down event.\r\n * Used in screen and gaze target ray mode only\r\n */\r\n disablePointerUpOnTouchOut: boolean;\r\n /**\r\n * For gaze mode for tracked-pointer / controllers (time to select instead of button press)\r\n */\r\n forceGazeMode: boolean;\r\n /**\r\n * Factor to be applied to the pointer-moved function in the gaze mode. How sensitive should the gaze mode be when checking if the pointer moved\r\n * to start a new countdown to the pointer down event.\r\n * Defaults to 1.\r\n */\r\n gazeModePointerMovedFactor?: number;\r\n /**\r\n * Different button type to use instead of the main component\r\n */\r\n overrideButtonId?: string;\r\n /**\r\n * use this rendering group id for the meshes (optional)\r\n */\r\n renderingGroupId?: number;\r\n /**\r\n * The amount of time in milliseconds it takes between pick found something to a pointer down event.\r\n * Used in gaze modes. Tracked pointer uses the trigger, screen uses touch events\r\n * 3000 means 3 seconds between pointing at something and selecting it\r\n */\r\n timeToSelect?: number;\r\n /**\r\n * Should meshes created here be added to a utility layer or the main scene\r\n */\r\n useUtilityLayer?: boolean;\r\n /**\r\n * Optional WebXR camera to be used for gaze selection\r\n */\r\n gazeCamera?: WebXRCamera;\r\n /**\r\n * the xr input to use with this pointer selection\r\n */\r\n xrInput: WebXRInput;\r\n\r\n /**\r\n * Should the scene pointerX and pointerY update be disabled\r\n * This is required for fullscreen AR GUI, but might slow down other experiences.\r\n * Disable in VR, if not needed.\r\n * The first rig camera (left eye) will be used to calculate the projection\r\n */\r\n disableScenePointerVectorUpdate: boolean;\r\n\r\n /**\r\n * Enable pointer selection on all controllers instead of switching between them\r\n */\r\n enablePointerSelectionOnAllControllers?: boolean;\r\n\r\n /**\r\n * The preferred hand to give the pointer selection to. This will be prioritized when the controller initialize.\r\n * If switch is enabled, it will still allow the user to switch between the different controllers\r\n */\r\n preferredHandedness?: XRHandedness;\r\n\r\n /**\r\n * Disable switching the pointer selection from one controller to the other.\r\n * If the preferred hand is set it will be fixed on this hand, and if not it will be fixed on the first controller added to the scene\r\n */\r\n disableSwitchOnClick?: boolean;\r\n\r\n /**\r\n * The maximum distance of the pointer selection feature. Defaults to 100.\r\n */\r\n maxPointerDistance?: number;\r\n\r\n /**\r\n * A function that will be called when a new selection mesh is generated.\r\n * This function should return a mesh that will be used as the selection mesh.\r\n * The default is a torus with a 0.01 diameter and 0.0075 thickness .\r\n */\r\n customSelectionMeshGenerator?: () => Mesh;\r\n\r\n /**\r\n * A function that will be called when a new laser pointer mesh is generated.\r\n * This function should return a mesh that will be used as the laser pointer mesh.\r\n * The height (y) of the mesh must be 1.\r\n */\r\n customLasterPointerMeshGenerator?: () => AbstractMesh;\r\n}\r\n\r\n/**\r\n * A module that will enable pointer selection for motion controllers of XR Input Sources\r\n */\r\nexport class WebXRControllerPointerSelection extends WebXRAbstractFeature {\r\n private static _IdCounter = 200;\r\n\r\n private _attachController = (xrController: WebXRInputSource) => {\r\n if (this._controllers[xrController.uniqueId]) {\r\n // already attached\r\n return;\r\n }\r\n\r\n const { laserPointer, selectionMesh } = this._generateNewMeshPair(xrController.pointer);\r\n\r\n // get two new meshes\r\n this._controllers[xrController.uniqueId] = {\r\n xrController,\r\n laserPointer,\r\n selectionMesh,\r\n meshUnderPointer: null,\r\n pick: null,\r\n tmpRay: new Ray(new Vector3(), new Vector3()),\r\n disabledByNearInteraction: false,\r\n id: WebXRControllerPointerSelection._IdCounter++,\r\n };\r\n\r\n if (this._attachedController) {\r\n if (\r\n !this._options.enablePointerSelectionOnAllControllers &&\r\n this._options.preferredHandedness &&\r\n xrController.inputSource.handedness === this._options.preferredHandedness\r\n ) {\r\n this._attachedController = xrController.uniqueId;\r\n }\r\n } else {\r\n if (!this._options.enablePointerSelectionOnAllControllers) {\r\n this._attachedController = xrController.uniqueId;\r\n }\r\n }\r\n\r\n switch (xrController.inputSource.targetRayMode) {\r\n case \"tracked-pointer\":\r\n return this._attachTrackedPointerRayMode(xrController);\r\n case \"gaze\":\r\n return this._attachGazeMode(xrController);\r\n case \"screen\":\r\n return this._attachScreenRayMode(xrController);\r\n }\r\n };\r\n\r\n private _controllers: {\r\n [controllerUniqueId: string]: {\r\n xrController?: WebXRInputSource;\r\n webXRCamera?: WebXRCamera;\r\n selectionComponent?: WebXRControllerComponent;\r\n onButtonChangedObserver?: Nullable>;\r\n onFrameObserver?: Nullable>;\r\n laserPointer: AbstractMesh;\r\n selectionMesh: AbstractMesh;\r\n meshUnderPointer: Nullable;\r\n pick: Nullable;\r\n id: number;\r\n tmpRay: Ray;\r\n disabledByNearInteraction: boolean;\r\n // event support\r\n eventListeners?: { [event in XREventType]?: (event: XRInputSourceEvent) => void };\r\n screenCoordinates?: { x: number; y: number };\r\n pointerDownTriggered?: boolean;\r\n finalPointerUpTriggered?: boolean;\r\n };\r\n } = {};\r\n private _scene: Scene;\r\n private _tmpVectorForPickCompare = new Vector3();\r\n\r\n private _attachedController: string;\r\n\r\n /**\r\n * The module's name\r\n */\r\n public static readonly Name = WebXRFeatureName.POINTER_SELECTION;\r\n /**\r\n * The (Babylon) version of this module.\r\n * This is an integer representing the implementation version.\r\n * This number does not correspond to the WebXR specs version\r\n */\r\n public static readonly Version = 1;\r\n\r\n /**\r\n * Disable lighting on the laser pointer (so it will always be visible)\r\n */\r\n public disablePointerLighting: boolean = true;\r\n /**\r\n * Disable lighting on the selection mesh (so it will always be visible)\r\n */\r\n public disableSelectionMeshLighting: boolean = true;\r\n /**\r\n * Should the laser pointer be displayed\r\n */\r\n public displayLaserPointer: boolean = true;\r\n /**\r\n * Should the selection mesh be displayed (The ring at the end of the laser pointer)\r\n */\r\n public displaySelectionMesh: boolean = true;\r\n /**\r\n * This color will be set to the laser pointer when selection is triggered\r\n */\r\n public laserPointerPickedColor: Color3 = new Color3(0.9, 0.9, 0.9);\r\n /**\r\n * Default color of the laser pointer\r\n */\r\n public laserPointerDefaultColor: Color3 = new Color3(0.7, 0.7, 0.7);\r\n /**\r\n * default color of the selection ring\r\n */\r\n public selectionMeshDefaultColor: Color3 = new Color3(0.8, 0.8, 0.8);\r\n /**\r\n * This color will be applied to the selection ring when selection is triggered\r\n */\r\n public selectionMeshPickedColor: Color3 = new Color3(0.3, 0.3, 1.0);\r\n\r\n /**\r\n * Optional filter to be used for ray selection. This predicate shares behavior with\r\n * scene.pointerMovePredicate which takes priority if it is also assigned.\r\n */\r\n public raySelectionPredicate: (mesh: AbstractMesh) => boolean;\r\n\r\n /**\r\n * constructs a new background remover module\r\n * @param _xrSessionManager the session manager for this module\r\n * @param _options read-only options to be used in this module\r\n */\r\n constructor(_xrSessionManager: WebXRSessionManager, private readonly _options: IWebXRControllerPointerSelectionOptions) {\r\n super(_xrSessionManager);\r\n this._scene = this._xrSessionManager.scene;\r\n }\r\n\r\n /**\r\n * attach this feature\r\n * Will usually be called by the features manager\r\n *\r\n * @returns true if successful.\r\n */\r\n public attach(): boolean {\r\n if (!super.attach()) {\r\n return false;\r\n }\r\n\r\n this._options.xrInput.controllers.forEach(this._attachController);\r\n this._addNewAttachObserver(this._options.xrInput.onControllerAddedObservable, this._attachController);\r\n this._addNewAttachObserver(this._options.xrInput.onControllerRemovedObservable, (controller) => {\r\n // REMOVE the controller\r\n this._detachController(controller.uniqueId);\r\n });\r\n\r\n this._scene.constantlyUpdateMeshUnderPointer = true;\r\n\r\n if (this._options.gazeCamera) {\r\n const webXRCamera = this._options.gazeCamera;\r\n\r\n const { laserPointer, selectionMesh } = this._generateNewMeshPair(webXRCamera);\r\n\r\n this._controllers[\"camera\"] = {\r\n webXRCamera,\r\n laserPointer,\r\n selectionMesh,\r\n meshUnderPointer: null,\r\n pick: null,\r\n tmpRay: new Ray(new Vector3(), new Vector3()),\r\n disabledByNearInteraction: false,\r\n id: WebXRControllerPointerSelection._IdCounter++,\r\n };\r\n this._attachGazeMode();\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * detach this feature.\r\n * Will usually be called by the features manager\r\n *\r\n * @returns true if successful.\r\n */\r\n public detach(): boolean {\r\n if (!super.detach()) {\r\n return false;\r\n }\r\n\r\n Object.keys(this._controllers).forEach((controllerId) => {\r\n this._detachController(controllerId);\r\n });\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Will get the mesh under a specific pointer.\r\n * `scene.meshUnderPointer` will only return one mesh - either left or right.\r\n * @param controllerId the controllerId to check\r\n * @returns The mesh under pointer or null if no mesh is under the pointer\r\n */\r\n public getMeshUnderPointer(controllerId: string): Nullable {\r\n if (this._controllers[controllerId]) {\r\n return this._controllers[controllerId].meshUnderPointer;\r\n } else {\r\n return null;\r\n }\r\n }\r\n\r\n /**\r\n * Get the xr controller that correlates to the pointer id in the pointer event\r\n *\r\n * @param id the pointer id to search for\r\n * @returns the controller that correlates to this id or null if not found\r\n */\r\n public getXRControllerByPointerId(id: number): Nullable {\r\n const keys = Object.keys(this._controllers);\r\n\r\n for (let i = 0; i < keys.length; ++i) {\r\n if (this._controllers[keys[i]].id === id) {\r\n return this._controllers[keys[i]].xrController || null;\r\n }\r\n }\r\n return null;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getPointerSelectionDisabledByPointerId(id: number): boolean {\r\n const keys = Object.keys(this._controllers);\r\n\r\n for (let i = 0; i < keys.length; ++i) {\r\n if (this._controllers[keys[i]].id === id) {\r\n return this._controllers[keys[i]].disabledByNearInteraction;\r\n }\r\n }\r\n return true;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _setPointerSelectionDisabledByPointerId(id: number, state: boolean) {\r\n const keys = Object.keys(this._controllers);\r\n\r\n for (let i = 0; i < keys.length; ++i) {\r\n if (this._controllers[keys[i]].id === id) {\r\n this._controllers[keys[i]].disabledByNearInteraction = state;\r\n return;\r\n }\r\n }\r\n }\r\n\r\n private _identityMatrix = Matrix.Identity();\r\n private _screenCoordinatesRef = Vector3.Zero();\r\n private _viewportRef = new Viewport(0, 0, 0, 0);\r\n\r\n protected _onXRFrame(_xrFrame: XRFrame) {\r\n Object.keys(this._controllers).forEach((id) => {\r\n // only do this for the selected pointer\r\n const controllerData = this._controllers[id];\r\n if ((!this._options.enablePointerSelectionOnAllControllers && id !== this._attachedController) || controllerData.disabledByNearInteraction) {\r\n controllerData.selectionMesh.isVisible = false;\r\n controllerData.laserPointer.isVisible = false;\r\n controllerData.pick = null;\r\n return;\r\n }\r\n\r\n controllerData.laserPointer.isVisible = this.displayLaserPointer;\r\n\r\n let controllerGlobalPosition: Vector3;\r\n\r\n // Every frame check collisions/input\r\n if (controllerData.xrController) {\r\n controllerGlobalPosition = controllerData.xrController.pointer.position;\r\n controllerData.xrController.getWorldPointerRayToRef(controllerData.tmpRay);\r\n } else if (controllerData.webXRCamera) {\r\n controllerGlobalPosition = controllerData.webXRCamera.position;\r\n controllerData.webXRCamera.getForwardRayToRef(controllerData.tmpRay);\r\n } else {\r\n return;\r\n }\r\n\r\n if (this._options.maxPointerDistance) {\r\n controllerData.tmpRay.length = this._options.maxPointerDistance;\r\n }\r\n // update pointerX and pointerY of the scene. Only if the flag is set to true!\r\n if (!this._options.disableScenePointerVectorUpdate && controllerGlobalPosition) {\r\n const scene = this._xrSessionManager.scene;\r\n const camera = this._options.xrInput.xrCamera;\r\n if (camera) {\r\n camera.viewport.toGlobalToRef(scene.getEngine().getRenderWidth(), scene.getEngine().getRenderHeight(), this._viewportRef);\r\n Vector3.ProjectToRef(controllerGlobalPosition, this._identityMatrix, scene.getTransformMatrix(), this._viewportRef, this._screenCoordinatesRef);\r\n // stay safe\r\n if (\r\n typeof this._screenCoordinatesRef.x === \"number\" &&\r\n typeof this._screenCoordinatesRef.y === \"number\" &&\r\n !isNaN(this._screenCoordinatesRef.x) &&\r\n !isNaN(this._screenCoordinatesRef.y)\r\n ) {\r\n scene.pointerX = this._screenCoordinatesRef.x;\r\n scene.pointerY = this._screenCoordinatesRef.y;\r\n\r\n controllerData.screenCoordinates = {\r\n x: this._screenCoordinatesRef.x,\r\n y: this._screenCoordinatesRef.y,\r\n };\r\n }\r\n }\r\n }\r\n\r\n let utilityScenePick = null;\r\n if (this._utilityLayerScene) {\r\n utilityScenePick = this._utilityLayerScene.pickWithRay(controllerData.tmpRay, this._utilityLayerScene.pointerMovePredicate || this.raySelectionPredicate);\r\n }\r\n\r\n const originalScenePick = this._scene.pickWithRay(controllerData.tmpRay, this._scene.pointerMovePredicate || this.raySelectionPredicate);\r\n if (!utilityScenePick || !utilityScenePick.hit) {\r\n // No hit in utility scene\r\n controllerData.pick = originalScenePick;\r\n } else if (!originalScenePick || !originalScenePick.hit) {\r\n // No hit in original scene\r\n controllerData.pick = utilityScenePick;\r\n } else if (utilityScenePick.distance < originalScenePick.distance) {\r\n // Hit is closer in utility scene\r\n controllerData.pick = utilityScenePick;\r\n } else {\r\n // Hit is closer in original scene\r\n controllerData.pick = originalScenePick;\r\n }\r\n\r\n if (controllerData.pick && controllerData.xrController) {\r\n controllerData.pick.aimTransform = controllerData.xrController.pointer;\r\n controllerData.pick.gripTransform = controllerData.xrController.grip || null;\r\n }\r\n\r\n const pick = controllerData.pick;\r\n\r\n if (pick && pick.pickedPoint && pick.hit) {\r\n // Update laser state\r\n this._updatePointerDistance(controllerData.laserPointer, pick.distance);\r\n\r\n // Update cursor state\r\n controllerData.selectionMesh.position.copyFrom(pick.pickedPoint);\r\n controllerData.selectionMesh.scaling.x = Math.sqrt(pick.distance);\r\n controllerData.selectionMesh.scaling.y = Math.sqrt(pick.distance);\r\n controllerData.selectionMesh.scaling.z = Math.sqrt(pick.distance);\r\n\r\n // To avoid z-fighting\r\n const pickNormal = this._convertNormalToDirectionOfRay(pick.getNormal(true), controllerData.tmpRay);\r\n const deltaFighting = 0.001;\r\n controllerData.selectionMesh.position.copyFrom(pick.pickedPoint);\r\n if (pickNormal) {\r\n const axis1 = Vector3.Cross(Axis.Y, pickNormal);\r\n const axis2 = Vector3.Cross(pickNormal, axis1);\r\n Vector3.RotationFromAxisToRef(axis2, pickNormal, axis1, controllerData.selectionMesh.rotation);\r\n controllerData.selectionMesh.position.addInPlace(pickNormal.scale(deltaFighting));\r\n }\r\n controllerData.selectionMesh.isVisible = true && this.displaySelectionMesh;\r\n controllerData.meshUnderPointer = pick.pickedMesh;\r\n } else {\r\n controllerData.selectionMesh.isVisible = false;\r\n this._updatePointerDistance(controllerData.laserPointer, 1);\r\n controllerData.meshUnderPointer = null;\r\n }\r\n });\r\n }\r\n\r\n private get _utilityLayerScene() {\r\n return this._options.customUtilityLayerScene || UtilityLayerRenderer.DefaultUtilityLayer.utilityLayerScene;\r\n }\r\n\r\n private _attachGazeMode(xrController?: WebXRInputSource) {\r\n const controllerData = this._controllers[(xrController && xrController.uniqueId) || \"camera\"];\r\n // attached when touched, detaches when raised\r\n const timeToSelect = this._options.timeToSelect || 3000;\r\n const sceneToRenderTo = this._options.useUtilityLayer ? this._utilityLayerScene : this._scene;\r\n let oldPick = new PickingInfo();\r\n const discMesh = CreateTorus(\r\n \"selection\",\r\n {\r\n diameter: 0.0035 * 15,\r\n thickness: 0.0025 * 6,\r\n tessellation: 20,\r\n },\r\n sceneToRenderTo\r\n );\r\n discMesh.isVisible = false;\r\n discMesh.isPickable = false;\r\n discMesh.parent = controllerData.selectionMesh;\r\n let timer = 0;\r\n let downTriggered = false;\r\n const pointerEventInit: PointerEventInit = {\r\n pointerId: controllerData.id,\r\n pointerType: \"xr\",\r\n };\r\n controllerData.onFrameObserver = this._xrSessionManager.onXRFrameObservable.add(() => {\r\n if (!controllerData.pick) {\r\n return;\r\n }\r\n this._augmentPointerInit(pointerEventInit, controllerData.id, controllerData.screenCoordinates);\r\n controllerData.laserPointer.material!.alpha = 0;\r\n discMesh.isVisible = false;\r\n if (controllerData.pick.hit) {\r\n if (!this._pickingMoved(oldPick, controllerData.pick)) {\r\n if (timer > timeToSelect / 10) {\r\n discMesh.isVisible = true;\r\n }\r\n\r\n timer += this._scene.getEngine().getDeltaTime();\r\n if (timer >= timeToSelect) {\r\n this._scene.simulatePointerDown(controllerData.pick, pointerEventInit);\r\n // this pointerdown event is not setting the controllerData.pointerDownTriggered to avoid a pointerUp event when this feature is detached\r\n downTriggered = true;\r\n // pointer up right after down, if disable on touch out\r\n if (this._options.disablePointerUpOnTouchOut) {\r\n this._scene.simulatePointerUp(controllerData.pick, pointerEventInit);\r\n }\r\n discMesh.isVisible = false;\r\n } else {\r\n const scaleFactor = 1 - timer / timeToSelect;\r\n discMesh.scaling.set(scaleFactor, scaleFactor, scaleFactor);\r\n }\r\n } else {\r\n if (downTriggered) {\r\n if (!this._options.disablePointerUpOnTouchOut) {\r\n this._scene.simulatePointerUp(controllerData.pick, pointerEventInit);\r\n }\r\n }\r\n downTriggered = false;\r\n timer = 0;\r\n }\r\n } else {\r\n downTriggered = false;\r\n timer = 0;\r\n }\r\n\r\n this._scene.simulatePointerMove(controllerData.pick, pointerEventInit);\r\n\r\n oldPick = controllerData.pick;\r\n });\r\n\r\n if (this._options.renderingGroupId !== undefined) {\r\n discMesh.renderingGroupId = this._options.renderingGroupId;\r\n }\r\n if (xrController) {\r\n xrController.onDisposeObservable.addOnce(() => {\r\n if (controllerData.pick && !this._options.disablePointerUpOnTouchOut && downTriggered) {\r\n this._scene.simulatePointerUp(controllerData.pick, pointerEventInit);\r\n controllerData.finalPointerUpTriggered = true;\r\n }\r\n discMesh.dispose();\r\n });\r\n }\r\n }\r\n\r\n private _attachScreenRayMode(xrController: WebXRInputSource) {\r\n const controllerData = this._controllers[xrController.uniqueId];\r\n let downTriggered = false;\r\n const pointerEventInit: PointerEventInit = {\r\n pointerId: controllerData.id,\r\n pointerType: \"xr\",\r\n };\r\n controllerData.onFrameObserver = this._xrSessionManager.onXRFrameObservable.add(() => {\r\n this._augmentPointerInit(pointerEventInit, controllerData.id, controllerData.screenCoordinates);\r\n if (!controllerData.pick || (this._options.disablePointerUpOnTouchOut && downTriggered)) {\r\n return;\r\n }\r\n if (!downTriggered) {\r\n this._scene.simulatePointerDown(controllerData.pick, pointerEventInit);\r\n controllerData.pointerDownTriggered = true;\r\n downTriggered = true;\r\n if (this._options.disablePointerUpOnTouchOut) {\r\n this._scene.simulatePointerUp(controllerData.pick, pointerEventInit);\r\n }\r\n } else {\r\n this._scene.simulatePointerMove(controllerData.pick, pointerEventInit);\r\n }\r\n });\r\n xrController.onDisposeObservable.addOnce(() => {\r\n this._augmentPointerInit(pointerEventInit, controllerData.id, controllerData.screenCoordinates);\r\n this._xrSessionManager.runInXRFrame(() => {\r\n if (controllerData.pick && !controllerData.finalPointerUpTriggered && downTriggered && !this._options.disablePointerUpOnTouchOut) {\r\n this._scene.simulatePointerUp(controllerData.pick, pointerEventInit);\r\n controllerData.finalPointerUpTriggered = true;\r\n }\r\n });\r\n });\r\n }\r\n\r\n private _attachTrackedPointerRayMode(xrController: WebXRInputSource) {\r\n const controllerData = this._controllers[xrController.uniqueId];\r\n if (this._options.forceGazeMode) {\r\n return this._attachGazeMode(xrController);\r\n }\r\n const pointerEventInit: PointerEventInit = {\r\n pointerId: controllerData.id,\r\n pointerType: \"xr\",\r\n };\r\n controllerData.onFrameObserver = this._xrSessionManager.onXRFrameObservable.add(() => {\r\n (controllerData.laserPointer.material).disableLighting = this.disablePointerLighting;\r\n (controllerData.selectionMesh.material).disableLighting = this.disableSelectionMeshLighting;\r\n\r\n if (controllerData.pick) {\r\n this._augmentPointerInit(pointerEventInit, controllerData.id, controllerData.screenCoordinates);\r\n this._scene.simulatePointerMove(controllerData.pick, pointerEventInit);\r\n }\r\n });\r\n if (xrController.inputSource.gamepad) {\r\n const init = (motionController: WebXRAbstractMotionController) => {\r\n if (this._options.overrideButtonId) {\r\n controllerData.selectionComponent = motionController.getComponent(this._options.overrideButtonId);\r\n }\r\n if (!controllerData.selectionComponent) {\r\n controllerData.selectionComponent = motionController.getMainComponent();\r\n }\r\n\r\n controllerData.onButtonChangedObserver = controllerData.selectionComponent.onButtonStateChangedObservable.add((component) => {\r\n if (component.changes.pressed) {\r\n const pressed = component.changes.pressed.current;\r\n if (controllerData.pick) {\r\n if (this._options.enablePointerSelectionOnAllControllers || xrController.uniqueId === this._attachedController) {\r\n this._augmentPointerInit(pointerEventInit, controllerData.id, controllerData.screenCoordinates);\r\n if (pressed) {\r\n this._scene.simulatePointerDown(controllerData.pick, pointerEventInit);\r\n controllerData.pointerDownTriggered = true;\r\n (controllerData.selectionMesh.material).emissiveColor = this.selectionMeshPickedColor;\r\n (controllerData.laserPointer.material).emissiveColor = this.laserPointerPickedColor;\r\n } else {\r\n this._scene.simulatePointerUp(controllerData.pick, pointerEventInit);\r\n (controllerData.selectionMesh.material).emissiveColor = this.selectionMeshDefaultColor;\r\n (controllerData.laserPointer.material).emissiveColor = this.laserPointerDefaultColor;\r\n }\r\n }\r\n } else {\r\n if (pressed && !this._options.enablePointerSelectionOnAllControllers && !this._options.disableSwitchOnClick) {\r\n this._attachedController = xrController.uniqueId;\r\n }\r\n }\r\n }\r\n });\r\n };\r\n if (xrController.motionController) {\r\n init(xrController.motionController);\r\n } else {\r\n xrController.onMotionControllerInitObservable.add(init);\r\n }\r\n } else {\r\n // use the select and squeeze events\r\n const selectStartListener = (event: XRInputSourceEvent) => {\r\n this._augmentPointerInit(pointerEventInit, controllerData.id, controllerData.screenCoordinates);\r\n if (controllerData.xrController && event.inputSource === controllerData.xrController.inputSource && controllerData.pick) {\r\n this._scene.simulatePointerDown(controllerData.pick, pointerEventInit);\r\n controllerData.pointerDownTriggered = true;\r\n (controllerData.selectionMesh.material).emissiveColor = this.selectionMeshPickedColor;\r\n (controllerData.laserPointer.material).emissiveColor = this.laserPointerPickedColor;\r\n }\r\n };\r\n\r\n const selectEndListener = (event: XRInputSourceEvent) => {\r\n this._augmentPointerInit(pointerEventInit, controllerData.id, controllerData.screenCoordinates);\r\n if (controllerData.xrController && event.inputSource === controllerData.xrController.inputSource && controllerData.pick) {\r\n this._scene.simulatePointerUp(controllerData.pick, pointerEventInit);\r\n (controllerData.selectionMesh.material).emissiveColor = this.selectionMeshDefaultColor;\r\n (controllerData.laserPointer.material).emissiveColor = this.laserPointerDefaultColor;\r\n }\r\n };\r\n\r\n controllerData.eventListeners = {\r\n selectend: selectEndListener,\r\n selectstart: selectStartListener,\r\n };\r\n\r\n this._xrSessionManager.session.addEventListener(\"selectstart\", selectStartListener);\r\n this._xrSessionManager.session.addEventListener(\"selectend\", selectEndListener);\r\n }\r\n }\r\n\r\n private _convertNormalToDirectionOfRay(normal: Nullable, ray: Ray) {\r\n if (normal) {\r\n const angle = Math.acos(Vector3.Dot(normal, ray.direction));\r\n if (angle < Math.PI / 2) {\r\n normal.scaleInPlace(-1);\r\n }\r\n }\r\n return normal;\r\n }\r\n\r\n private _detachController(xrControllerUniqueId: string) {\r\n const controllerData = this._controllers[xrControllerUniqueId];\r\n if (!controllerData) {\r\n return;\r\n }\r\n if (controllerData.selectionComponent) {\r\n if (controllerData.onButtonChangedObserver) {\r\n controllerData.selectionComponent.onButtonStateChangedObservable.remove(controllerData.onButtonChangedObserver);\r\n }\r\n }\r\n if (controllerData.onFrameObserver) {\r\n this._xrSessionManager.onXRFrameObservable.remove(controllerData.onFrameObserver);\r\n }\r\n if (controllerData.eventListeners) {\r\n Object.keys(controllerData.eventListeners).forEach((eventName: string) => {\r\n const func = controllerData.eventListeners && controllerData.eventListeners[eventName as XREventType];\r\n if (func) {\r\n // For future reference - this is an issue in the WebXR typings.\r\n this._xrSessionManager.session.removeEventListener(eventName as XREventType, func as any);\r\n }\r\n });\r\n }\r\n\r\n if (!controllerData.finalPointerUpTriggered && controllerData.pointerDownTriggered) {\r\n // Stay safe and fire a pointerup, in case it wasn't already triggered\r\n const pointerEventInit: PointerEventInit = {\r\n pointerId: controllerData.id,\r\n pointerType: \"xr\",\r\n };\r\n this._xrSessionManager.runInXRFrame(() => {\r\n this._augmentPointerInit(pointerEventInit, controllerData.id, controllerData.screenCoordinates);\r\n this._scene.simulatePointerUp(controllerData.pick || new PickingInfo(), pointerEventInit);\r\n controllerData.finalPointerUpTriggered = true;\r\n });\r\n }\r\n this._xrSessionManager.scene.onBeforeRenderObservable.addOnce(() => {\r\n try {\r\n controllerData.selectionMesh.dispose();\r\n controllerData.laserPointer.dispose();\r\n // remove from the map\r\n delete this._controllers[xrControllerUniqueId];\r\n if (this._attachedController === xrControllerUniqueId) {\r\n // check for other controllers\r\n const keys = Object.keys(this._controllers);\r\n if (keys.length) {\r\n this._attachedController = keys[0];\r\n } else {\r\n this._attachedController = \"\";\r\n }\r\n }\r\n } catch (e) {\r\n Tools.Warn(\"controller already detached.\");\r\n }\r\n });\r\n }\r\n\r\n private _generateNewMeshPair(meshParent: Node) {\r\n const sceneToRenderTo = this._options.useUtilityLayer ? this._options.customUtilityLayerScene || UtilityLayerRenderer.DefaultUtilityLayer.utilityLayerScene : this._scene;\r\n const laserPointer = this._options.customLasterPointerMeshGenerator\r\n ? this._options.customLasterPointerMeshGenerator()\r\n : CreateCylinder(\r\n \"laserPointer\",\r\n {\r\n height: 1,\r\n diameterTop: 0.0002,\r\n diameterBottom: 0.004,\r\n tessellation: 20,\r\n subdivisions: 1,\r\n },\r\n sceneToRenderTo\r\n );\r\n laserPointer.parent = meshParent;\r\n const laserPointerMaterial = new StandardMaterial(\"laserPointerMat\", sceneToRenderTo);\r\n laserPointerMaterial.emissiveColor = this.laserPointerDefaultColor;\r\n laserPointerMaterial.alpha = 0.7;\r\n laserPointer.material = laserPointerMaterial;\r\n laserPointer.rotation.x = Math.PI / 2;\r\n this._updatePointerDistance(laserPointer, 1);\r\n laserPointer.isPickable = false;\r\n laserPointer.isVisible = false;\r\n\r\n // Create a gaze tracker for the XR controller\r\n const selectionMesh = this._options.customSelectionMeshGenerator\r\n ? this._options.customSelectionMeshGenerator()\r\n : CreateTorus(\r\n \"gazeTracker\",\r\n {\r\n diameter: 0.0035 * 3,\r\n thickness: 0.0025 * 3,\r\n tessellation: 20,\r\n },\r\n sceneToRenderTo\r\n );\r\n selectionMesh.bakeCurrentTransformIntoVertices();\r\n selectionMesh.isPickable = false;\r\n selectionMesh.isVisible = false;\r\n const targetMat = new StandardMaterial(\"targetMat\", sceneToRenderTo);\r\n targetMat.specularColor = Color3.Black();\r\n targetMat.emissiveColor = this.selectionMeshDefaultColor;\r\n targetMat.backFaceCulling = false;\r\n selectionMesh.material = targetMat;\r\n\r\n if (this._options.renderingGroupId !== undefined) {\r\n laserPointer.renderingGroupId = this._options.renderingGroupId;\r\n selectionMesh.renderingGroupId = this._options.renderingGroupId;\r\n }\r\n\r\n return {\r\n laserPointer,\r\n selectionMesh,\r\n };\r\n }\r\n\r\n private _pickingMoved(oldPick: PickingInfo, newPick: PickingInfo) {\r\n if (!oldPick.hit || !newPick.hit) {\r\n return true;\r\n }\r\n if (!oldPick.pickedMesh || !oldPick.pickedPoint || !newPick.pickedMesh || !newPick.pickedPoint) {\r\n return true;\r\n }\r\n if (oldPick.pickedMesh !== newPick.pickedMesh) {\r\n return true;\r\n }\r\n oldPick.pickedPoint?.subtractToRef(newPick.pickedPoint, this._tmpVectorForPickCompare);\r\n this._tmpVectorForPickCompare.set(Math.abs(this._tmpVectorForPickCompare.x), Math.abs(this._tmpVectorForPickCompare.y), Math.abs(this._tmpVectorForPickCompare.z));\r\n const delta = (this._options.gazeModePointerMovedFactor || 1) * 0.01 * newPick.distance;\r\n const length = this._tmpVectorForPickCompare.length();\r\n if (length > delta) {\r\n return true;\r\n }\r\n return false;\r\n }\r\n\r\n private _updatePointerDistance(_laserPointer: AbstractMesh, distance: number = 100) {\r\n _laserPointer.scaling.y = distance;\r\n // a bit of distance from the controller\r\n if (this._scene.useRightHandedSystem) {\r\n distance *= -1;\r\n }\r\n _laserPointer.position.z = distance / 2 + 0.05;\r\n }\r\n\r\n private _augmentPointerInit(pointerEventInit: PointerEventInit, id: number, screenCoordinates?: { x: number; y: number }): void {\r\n pointerEventInit.pointerId = id;\r\n pointerEventInit.pointerType = \"xr\";\r\n if (screenCoordinates) {\r\n pointerEventInit.screenX = screenCoordinates.x;\r\n pointerEventInit.screenY = screenCoordinates.y;\r\n }\r\n }\r\n\r\n /** @internal */\r\n public get lasterPointerDefaultColor(): Color3 {\r\n // here due to a typo\r\n return this.laserPointerDefaultColor;\r\n }\r\n}\r\n\r\n//register the plugin\r\nWebXRFeaturesManager.AddWebXRFeature(\r\n WebXRControllerPointerSelection.Name,\r\n (xrSessionManager, options) => {\r\n return () => new WebXRControllerPointerSelection(xrSessionManager, options);\r\n },\r\n WebXRControllerPointerSelection.Version,\r\n true\r\n);\r\n","/**\r\n * Defines the kind of connection point for node based material\r\n */\r\nexport enum NodeMaterialBlockConnectionPointTypes {\r\n /** Float */\r\n Float = 0x0001,\r\n /** Int */\r\n Int = 0x0002,\r\n /** Vector2 */\r\n Vector2 = 0x0004,\r\n /** Vector3 */\r\n Vector3 = 0x0008,\r\n /** Vector4 */\r\n Vector4 = 0x0010,\r\n /** Color3 */\r\n Color3 = 0x0020,\r\n /** Color4 */\r\n Color4 = 0x0040,\r\n /** Matrix */\r\n Matrix = 0x0080,\r\n /** Custom object */\r\n Object = 0x0100,\r\n /** Detect type based on connection */\r\n AutoDetect = 0x0400,\r\n /** Output type that will be defined by input type */\r\n BasedOnInput = 0x0800,\r\n /** Bitmask of all types */\r\n All = 0x0fff,\r\n}\r\n","/**\r\n * Enum used to define the target of a block\r\n */\r\nexport enum NodeMaterialBlockTargets {\r\n /** Vertex shader */\r\n Vertex = 1,\r\n /** Fragment shader */\r\n Fragment = 2,\r\n /** Neutral */\r\n Neutral = 4,\r\n /** Vertex and Fragment */\r\n VertexAndFragment = Vertex | Fragment,\r\n}\r\n","import { NodeMaterialBlockConnectionPointTypes } from \"./Enums/nodeMaterialBlockConnectionPointTypes\";\r\nimport { NodeMaterialBlockTargets } from \"./Enums/nodeMaterialBlockTargets\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { InputBlock } from \"./Blocks/Input/inputBlock\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport type { NodeMaterialBlock } from \"./nodeMaterialBlock\";\r\n\r\n/**\r\n * Enum used to define the compatibility state between two connection points\r\n */\r\nexport enum NodeMaterialConnectionPointCompatibilityStates {\r\n /** Points are compatibles */\r\n Compatible,\r\n /** Points are incompatible because of their types */\r\n TypeIncompatible,\r\n /** Points are incompatible because of their targets (vertex vs fragment) */\r\n TargetIncompatible,\r\n /** Points are incompatible because they are in the same hierarchy **/\r\n HierarchyIssue,\r\n}\r\n\r\n/**\r\n * Defines the direction of a connection point\r\n */\r\nexport enum NodeMaterialConnectionPointDirection {\r\n /** Input */\r\n Input,\r\n /** Output */\r\n Output,\r\n}\r\n\r\n/**\r\n * Defines a connection point for a block\r\n */\r\nexport class NodeMaterialConnectionPoint {\r\n /**\r\n * Checks if two types are equivalent\r\n * @param type1 type 1 to check\r\n * @param type2 type 2 to check\r\n * @returns true if both types are equivalent, else false\r\n */\r\n public static AreEquivalentTypes(type1: number, type2: number): boolean {\r\n switch (type1) {\r\n case NodeMaterialBlockConnectionPointTypes.Vector3: {\r\n if (type2 === NodeMaterialBlockConnectionPointTypes.Color3) {\r\n return true;\r\n }\r\n break;\r\n }\r\n case NodeMaterialBlockConnectionPointTypes.Vector4: {\r\n if (type2 === NodeMaterialBlockConnectionPointTypes.Color4) {\r\n return true;\r\n }\r\n break;\r\n }\r\n case NodeMaterialBlockConnectionPointTypes.Color3: {\r\n if (type2 === NodeMaterialBlockConnectionPointTypes.Vector3) {\r\n return true;\r\n }\r\n break;\r\n }\r\n case NodeMaterialBlockConnectionPointTypes.Color4: {\r\n if (type2 === NodeMaterialBlockConnectionPointTypes.Vector4) {\r\n return true;\r\n }\r\n break;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /** @internal */\r\n public _ownerBlock: NodeMaterialBlock;\r\n /** @internal */\r\n public _connectedPoint: Nullable = null;\r\n\r\n private _endpoints = new Array();\r\n private _associatedVariableName: string;\r\n private _direction: NodeMaterialConnectionPointDirection;\r\n\r\n /** @internal */\r\n public _typeConnectionSource: Nullable = null;\r\n\r\n /** @internal */\r\n public _defaultConnectionPointType: Nullable = null;\r\n\r\n /** @internal */\r\n public _linkedConnectionSource: Nullable = null;\r\n\r\n /** @internal */\r\n public _acceptedConnectionPointType: Nullable = null;\r\n\r\n private _type = NodeMaterialBlockConnectionPointTypes.Float;\r\n\r\n /** @internal */\r\n public _enforceAssociatedVariableName = false;\r\n\r\n /** Gets the direction of the point */\r\n public get direction() {\r\n return this._direction;\r\n }\r\n\r\n /** Indicates that this connection point needs dual validation before being connected to another point */\r\n public needDualDirectionValidation: boolean = false;\r\n\r\n /**\r\n * Gets or sets the additional types supported by this connection point\r\n */\r\n public acceptedConnectionPointTypes = new Array();\r\n\r\n /**\r\n * Gets or sets the additional types excluded by this connection point\r\n */\r\n public excludedConnectionPointTypes = new Array();\r\n\r\n /**\r\n * Observable triggered when this point is connected\r\n */\r\n public onConnectionObservable = new Observable();\r\n\r\n /**\r\n * Gets or sets the associated variable name in the shader\r\n */\r\n public get associatedVariableName(): string {\r\n if (this._ownerBlock.isInput) {\r\n return (this._ownerBlock as InputBlock).associatedVariableName;\r\n }\r\n\r\n if ((!this._enforceAssociatedVariableName || !this._associatedVariableName) && this._connectedPoint) {\r\n return this._connectedPoint.associatedVariableName;\r\n }\r\n\r\n return this._associatedVariableName;\r\n }\r\n\r\n public set associatedVariableName(value: string) {\r\n this._associatedVariableName = value;\r\n }\r\n\r\n /** Get the inner type (ie AutoDetect for instance instead of the inferred one) */\r\n public get innerType() {\r\n if (this._linkedConnectionSource && this._linkedConnectionSource.isConnected) {\r\n return this.type;\r\n }\r\n return this._type;\r\n }\r\n\r\n /**\r\n * Gets or sets the connection point type (default is float)\r\n */\r\n public get type(): NodeMaterialBlockConnectionPointTypes {\r\n if (this._type === NodeMaterialBlockConnectionPointTypes.AutoDetect) {\r\n if (this._ownerBlock.isInput) {\r\n return (this._ownerBlock as InputBlock).type;\r\n }\r\n\r\n if (this._connectedPoint) {\r\n return this._connectedPoint.type;\r\n }\r\n\r\n if (this._linkedConnectionSource && this._linkedConnectionSource.isConnected) {\r\n return this._linkedConnectionSource.type;\r\n }\r\n }\r\n\r\n if (this._type === NodeMaterialBlockConnectionPointTypes.BasedOnInput) {\r\n if (this._typeConnectionSource) {\r\n if (!this._typeConnectionSource.isConnected && this._defaultConnectionPointType) {\r\n return this._defaultConnectionPointType;\r\n }\r\n return this._typeConnectionSource.type;\r\n } else if (this._defaultConnectionPointType) {\r\n return this._defaultConnectionPointType;\r\n }\r\n }\r\n\r\n return this._type;\r\n }\r\n\r\n public set type(value: NodeMaterialBlockConnectionPointTypes) {\r\n this._type = value;\r\n }\r\n\r\n /**\r\n * Gets or sets the connection point name\r\n */\r\n public name: string;\r\n\r\n /**\r\n * Gets or sets the connection point name\r\n */\r\n public displayName: string;\r\n\r\n /**\r\n * Gets or sets a boolean indicating that this connection point can be omitted\r\n */\r\n public isOptional: boolean;\r\n\r\n /**\r\n * Gets or sets a boolean indicating that this connection point is exposed on a frame\r\n */\r\n public isExposedOnFrame: boolean = false;\r\n\r\n /**\r\n * Gets or sets number indicating the position that the port is exposed to on a frame\r\n */\r\n public exposedPortPosition: number = -1;\r\n\r\n /**\r\n * Gets or sets a string indicating that this uniform must be defined under a #ifdef\r\n */\r\n public define: string;\r\n\r\n /** @internal */\r\n public _prioritizeVertex = false;\r\n\r\n private _target: NodeMaterialBlockTargets = NodeMaterialBlockTargets.VertexAndFragment;\r\n\r\n /** Gets or sets the target of that connection point */\r\n public get target(): NodeMaterialBlockTargets {\r\n if (!this._prioritizeVertex || !this._ownerBlock) {\r\n return this._target;\r\n }\r\n\r\n if (this._target !== NodeMaterialBlockTargets.VertexAndFragment) {\r\n return this._target;\r\n }\r\n\r\n if (this._ownerBlock.target === NodeMaterialBlockTargets.Fragment) {\r\n return NodeMaterialBlockTargets.Fragment;\r\n }\r\n\r\n return NodeMaterialBlockTargets.Vertex;\r\n }\r\n\r\n public set target(value: NodeMaterialBlockTargets) {\r\n this._target = value;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating that the current point is connected to another NodeMaterialBlock\r\n */\r\n public get isConnected(): boolean {\r\n return this.connectedPoint !== null || this.hasEndpoints;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating that the current point is connected to an input block\r\n */\r\n public get isConnectedToInputBlock(): boolean {\r\n return this.connectedPoint !== null && this.connectedPoint.ownerBlock.isInput;\r\n }\r\n\r\n /**\r\n * Gets a the connected input block (if any)\r\n */\r\n public get connectInputBlock(): Nullable {\r\n if (!this.isConnectedToInputBlock) {\r\n return null;\r\n }\r\n\r\n return this.connectedPoint!.ownerBlock as InputBlock;\r\n }\r\n\r\n /** Get the other side of the connection (if any) */\r\n public get connectedPoint(): Nullable {\r\n return this._connectedPoint;\r\n }\r\n\r\n /** Get the block that owns this connection point */\r\n public get ownerBlock(): NodeMaterialBlock {\r\n return this._ownerBlock;\r\n }\r\n\r\n /** Get the block connected on the other side of this connection (if any) */\r\n public get sourceBlock(): Nullable {\r\n if (!this._connectedPoint) {\r\n return null;\r\n }\r\n\r\n return this._connectedPoint.ownerBlock;\r\n }\r\n\r\n /** Get the block connected on the endpoints of this connection (if any) */\r\n public get connectedBlocks(): Array {\r\n if (this._endpoints.length === 0) {\r\n return [];\r\n }\r\n\r\n return this._endpoints.map((e) => e.ownerBlock);\r\n }\r\n\r\n /** Gets the list of connected endpoints */\r\n public get endpoints() {\r\n return this._endpoints;\r\n }\r\n\r\n /** Gets a boolean indicating if that output point is connected to at least one input */\r\n public get hasEndpoints(): boolean {\r\n return this._endpoints && this._endpoints.length > 0;\r\n }\r\n\r\n /** Gets a boolean indicating that this connection has a path to the vertex output*/\r\n public get isDirectlyConnectedToVertexOutput(): boolean {\r\n if (!this.hasEndpoints) {\r\n return false;\r\n }\r\n\r\n for (const endpoint of this._endpoints) {\r\n if (endpoint.ownerBlock.target === NodeMaterialBlockTargets.Vertex) {\r\n return true;\r\n }\r\n\r\n if (endpoint.ownerBlock.target === NodeMaterialBlockTargets.Neutral || endpoint.ownerBlock.target === NodeMaterialBlockTargets.VertexAndFragment) {\r\n if (endpoint.ownerBlock.outputs.some((o) => o.isDirectlyConnectedToVertexOutput)) {\r\n return true;\r\n }\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /** Gets a boolean indicating that this connection will be used in the vertex shader */\r\n public get isConnectedInVertexShader(): boolean {\r\n if (this.target === NodeMaterialBlockTargets.Vertex) {\r\n return true;\r\n }\r\n\r\n if (!this.hasEndpoints) {\r\n return false;\r\n }\r\n\r\n for (const endpoint of this._endpoints) {\r\n if (endpoint.ownerBlock.target === NodeMaterialBlockTargets.Vertex) {\r\n return true;\r\n }\r\n\r\n if (endpoint.target === NodeMaterialBlockTargets.Vertex) {\r\n return true;\r\n }\r\n\r\n if (endpoint.ownerBlock.target === NodeMaterialBlockTargets.Neutral || endpoint.ownerBlock.target === NodeMaterialBlockTargets.VertexAndFragment) {\r\n if (endpoint.ownerBlock.outputs.some((o) => o.isConnectedInVertexShader)) {\r\n return true;\r\n }\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /** Gets a boolean indicating that this connection will be used in the fragment shader */\r\n public get isConnectedInFragmentShader(): boolean {\r\n if (this.target === NodeMaterialBlockTargets.Fragment) {\r\n return true;\r\n }\r\n\r\n if (!this.hasEndpoints) {\r\n return false;\r\n }\r\n\r\n for (const endpoint of this._endpoints) {\r\n if (endpoint.ownerBlock.target === NodeMaterialBlockTargets.Fragment) {\r\n return true;\r\n }\r\n\r\n if (endpoint.ownerBlock.target === NodeMaterialBlockTargets.Neutral || endpoint.ownerBlock.target === NodeMaterialBlockTargets.VertexAndFragment) {\r\n if (endpoint.ownerBlock.isConnectedInFragmentShader()) {\r\n return true;\r\n }\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Creates a block suitable to be used as an input for this input point.\r\n * If null is returned, a block based on the point type will be created.\r\n * @returns The returned string parameter is the name of the output point of NodeMaterialBlock (first parameter of the returned array) that can be connected to the input\r\n */\r\n public createCustomInputBlock(): Nullable<[NodeMaterialBlock, string]> {\r\n return null;\r\n }\r\n\r\n /**\r\n * Creates a new connection point\r\n * @param name defines the connection point name\r\n * @param ownerBlock defines the block hosting this connection point\r\n * @param direction defines the direction of the connection point\r\n */\r\n public constructor(name: string, ownerBlock: NodeMaterialBlock, direction: NodeMaterialConnectionPointDirection) {\r\n this._ownerBlock = ownerBlock;\r\n this.name = name;\r\n this._direction = direction;\r\n }\r\n\r\n /**\r\n * Gets the current class name e.g. \"NodeMaterialConnectionPoint\"\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"NodeMaterialConnectionPoint\";\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if the current point can be connected to another point\r\n * @param connectionPoint defines the other connection point\r\n * @returns a boolean\r\n */\r\n public canConnectTo(connectionPoint: NodeMaterialConnectionPoint) {\r\n return this.checkCompatibilityState(connectionPoint) === NodeMaterialConnectionPointCompatibilityStates.Compatible;\r\n }\r\n\r\n /**\r\n * Gets a number indicating if the current point can be connected to another point\r\n * @param connectionPoint defines the other connection point\r\n * @returns a number defining the compatibility state\r\n */\r\n public checkCompatibilityState(connectionPoint: NodeMaterialConnectionPoint): NodeMaterialConnectionPointCompatibilityStates {\r\n const ownerBlock = this._ownerBlock;\r\n const otherBlock = connectionPoint.ownerBlock;\r\n\r\n if (ownerBlock.target === NodeMaterialBlockTargets.Fragment) {\r\n // Let's check we are not going reverse\r\n\r\n if (otherBlock.target === NodeMaterialBlockTargets.Vertex) {\r\n return NodeMaterialConnectionPointCompatibilityStates.TargetIncompatible;\r\n }\r\n\r\n for (const output of otherBlock.outputs) {\r\n if (output.ownerBlock.target != NodeMaterialBlockTargets.Neutral && output.isConnectedInVertexShader) {\r\n return NodeMaterialConnectionPointCompatibilityStates.TargetIncompatible;\r\n }\r\n }\r\n }\r\n\r\n if (this.type !== connectionPoint.type && connectionPoint.innerType !== NodeMaterialBlockConnectionPointTypes.AutoDetect) {\r\n // Equivalents\r\n if (NodeMaterialConnectionPoint.AreEquivalentTypes(this.type, connectionPoint.type)) {\r\n return NodeMaterialConnectionPointCompatibilityStates.Compatible;\r\n }\r\n\r\n // Accepted types\r\n if (\r\n (connectionPoint.acceptedConnectionPointTypes && connectionPoint.acceptedConnectionPointTypes.indexOf(this.type) !== -1) ||\r\n (connectionPoint._acceptedConnectionPointType && NodeMaterialConnectionPoint.AreEquivalentTypes(connectionPoint._acceptedConnectionPointType.type, this.type))\r\n ) {\r\n return NodeMaterialConnectionPointCompatibilityStates.Compatible;\r\n } else {\r\n return NodeMaterialConnectionPointCompatibilityStates.TypeIncompatible;\r\n }\r\n }\r\n\r\n // Excluded\r\n if (connectionPoint.excludedConnectionPointTypes && connectionPoint.excludedConnectionPointTypes.indexOf(this.type) !== -1) {\r\n return NodeMaterialConnectionPointCompatibilityStates.TypeIncompatible;\r\n }\r\n\r\n // Check hierarchy\r\n let targetBlock = otherBlock;\r\n let sourceBlock = ownerBlock;\r\n if (this.direction === NodeMaterialConnectionPointDirection.Input) {\r\n targetBlock = ownerBlock;\r\n sourceBlock = otherBlock;\r\n }\r\n\r\n if (targetBlock.isAnAncestorOf(sourceBlock)) {\r\n return NodeMaterialConnectionPointCompatibilityStates.HierarchyIssue;\r\n }\r\n\r\n return NodeMaterialConnectionPointCompatibilityStates.Compatible;\r\n }\r\n\r\n /**\r\n * Connect this point to another connection point\r\n * @param connectionPoint defines the other connection point\r\n * @param ignoreConstraints defines if the system will ignore connection type constraints (default is false)\r\n * @returns the current connection point\r\n */\r\n public connectTo(connectionPoint: NodeMaterialConnectionPoint, ignoreConstraints = false): NodeMaterialConnectionPoint {\r\n if (!ignoreConstraints && !this.canConnectTo(connectionPoint)) {\r\n throw \"Cannot connect these two connectors.\";\r\n }\r\n\r\n this._endpoints.push(connectionPoint);\r\n connectionPoint._connectedPoint = this;\r\n\r\n this._enforceAssociatedVariableName = false;\r\n\r\n this.onConnectionObservable.notifyObservers(connectionPoint);\r\n connectionPoint.onConnectionObservable.notifyObservers(this);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Disconnect this point from one of his endpoint\r\n * @param endpoint defines the other connection point\r\n * @returns the current connection point\r\n */\r\n public disconnectFrom(endpoint: NodeMaterialConnectionPoint): NodeMaterialConnectionPoint {\r\n const index = this._endpoints.indexOf(endpoint);\r\n\r\n if (index === -1) {\r\n return this;\r\n }\r\n\r\n this._endpoints.splice(index, 1);\r\n endpoint._connectedPoint = null;\r\n this._enforceAssociatedVariableName = false;\r\n endpoint._enforceAssociatedVariableName = false;\r\n return this;\r\n }\r\n\r\n /**\r\n * Fill the list of excluded connection point types with all types other than those passed in the parameter\r\n * @param mask Types (ORed values of NodeMaterialBlockConnectionPointTypes) that are allowed, and thus will not be pushed to the excluded list\r\n */\r\n public addExcludedConnectionPointFromAllowedTypes(mask: number): void {\r\n let bitmask = 1;\r\n while (bitmask < NodeMaterialBlockConnectionPointTypes.All) {\r\n if (!(mask & bitmask)) {\r\n this.excludedConnectionPointTypes.push(bitmask);\r\n }\r\n bitmask = bitmask << 1;\r\n }\r\n }\r\n\r\n /**\r\n * Serializes this point in a JSON representation\r\n * @param isInput defines if the connection point is an input (default is true)\r\n * @returns the serialized point object\r\n */\r\n public serialize(isInput = true): any {\r\n const serializationObject: any = {};\r\n\r\n serializationObject.name = this.name;\r\n serializationObject.displayName = this.displayName;\r\n\r\n if (isInput && this.connectedPoint) {\r\n serializationObject.inputName = this.name;\r\n serializationObject.targetBlockId = this.connectedPoint.ownerBlock.uniqueId;\r\n serializationObject.targetConnectionName = this.connectedPoint.name;\r\n serializationObject.isExposedOnFrame = true;\r\n serializationObject.exposedPortPosition = this.exposedPortPosition;\r\n }\r\n\r\n if (this.isExposedOnFrame || this.exposedPortPosition >= 0) {\r\n serializationObject.isExposedOnFrame = true;\r\n serializationObject.exposedPortPosition = this.exposedPortPosition;\r\n }\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Release resources\r\n */\r\n public dispose() {\r\n this.onConnectionObservable.clear();\r\n }\r\n}\r\n","import type { Scene } from \"../scene\";\r\n\r\n/**\r\n * Enum defining the type of properties that can be edited in the property pages in the node editor\r\n */\r\nexport enum PropertyTypeForEdition {\r\n /** property is a boolean */\r\n Boolean,\r\n /** property is a float */\r\n Float,\r\n /** property is a int */\r\n Int,\r\n /** property is a Vector2 */\r\n Vector2,\r\n /** property is a list of values */\r\n List,\r\n}\r\n\r\n/**\r\n * Interface that defines an option in a variable of type list\r\n */\r\nexport interface IEditablePropertyListOption {\r\n /** label of the option */\r\n label: string;\r\n /** value of the option */\r\n value: number;\r\n}\r\n\r\n/**\r\n * Interface that defines the options available for an editable property\r\n */\r\nexport interface IEditablePropertyOption {\r\n /** min value */\r\n min?: number;\r\n /** max value */\r\n max?: number;\r\n /** notifiers: indicates which actions to take when the property is changed */\r\n notifiers?: {\r\n /** the entity should be rebuilt */\r\n rebuild?: boolean;\r\n /** the preview should be updated */\r\n update?: boolean;\r\n /** the onPreviewCommandActivated observer of the preview manager should be triggered */\r\n activatePreviewCommand?: boolean;\r\n /** a callback to trigger */\r\n callback?: (scene: Scene, block: any) => boolean | undefined | void;\r\n /** a callback to validate the property. Returns true if the property is ok, else false. If false, the rebuild/update/callback events won't be called */\r\n onValidation?: (block: any, propertyName: string) => boolean;\r\n };\r\n /** list of the options for a variable of type list */\r\n options?: IEditablePropertyListOption[];\r\n}\r\n\r\n/**\r\n * Interface that describes an editable property\r\n */\r\nexport interface IPropertyDescriptionForEdition {\r\n /** name of the property */\r\n propertyName: string;\r\n /** display name of the property */\r\n displayName: string;\r\n /** type of the property */\r\n type: PropertyTypeForEdition;\r\n /** group of the property - all properties with the same group value will be displayed in a specific section */\r\n groupName: string;\r\n /** options for the property */\r\n options: IEditablePropertyOption;\r\n}\r\n\r\n/**\r\n * Decorator that flags a property in a node block as being editable\r\n * @param displayName\r\n * @param propertyType\r\n * @param groupName\r\n * @param options\r\n */\r\nexport function editableInPropertyPage(\r\n displayName: string,\r\n propertyType: PropertyTypeForEdition = PropertyTypeForEdition.Boolean,\r\n groupName: string = \"PROPERTIES\",\r\n options?: IEditablePropertyOption\r\n) {\r\n return (target: any, propertyKey: string) => {\r\n let propStore: IPropertyDescriptionForEdition[] = target._propStore;\r\n if (!propStore) {\r\n propStore = [];\r\n target._propStore = propStore;\r\n }\r\n propStore.push({\r\n propertyName: propertyKey,\r\n displayName: displayName,\r\n type: propertyType,\r\n groupName: groupName,\r\n options: options ?? {},\r\n });\r\n };\r\n}\r\n","/**\r\n * Enum defining the mode of a NodeMaterialBlockConnectionPoint\r\n */\r\nexport enum NodeMaterialBlockConnectionPointMode {\r\n /** Value is an uniform */\r\n Uniform,\r\n /** Value is a mesh attribute */\r\n Attribute,\r\n /** Value is a varying between vertex and fragment shaders */\r\n Varying,\r\n /** Mode is undefined */\r\n Undefined,\r\n}\r\n","/**\r\n * Enum used to define system values e.g. values automatically provided by the system\r\n */\r\nexport enum NodeMaterialSystemValues {\r\n /** World */\r\n World = 1,\r\n /** View */\r\n View = 2,\r\n /** Projection */\r\n Projection = 3,\r\n /** ViewProjection */\r\n ViewProjection = 4,\r\n /** WorldView */\r\n WorldView = 5,\r\n /** WorldViewProjection */\r\n WorldViewProjection = 6,\r\n /** CameraPosition */\r\n CameraPosition = 7,\r\n /** Fog Color */\r\n FogColor = 8,\r\n /** Delta time */\r\n DeltaTime = 9,\r\n /** Camera parameters */\r\n CameraParameters = 10,\r\n /** Material alpha */\r\n MaterialAlpha = 11,\r\n}\r\n","/**\r\n * Enum defining the type of animations supported by InputBlock\r\n */\r\nexport enum AnimatedInputBlockTypes {\r\n /** No animation */\r\n None,\r\n /** Time based animation (is incremented by 0.6 each second). Will only work for floats */\r\n Time,\r\n /** Time elapsed (in seconds) since the engine was initialized. Will only work for floats */\r\n RealTime,\r\n}\r\n","import { NodeMaterialBlockConnectionPointTypes } from \"./Enums/nodeMaterialBlockConnectionPointTypes\";\r\nimport { NodeMaterialBlockTargets } from \"./Enums/nodeMaterialBlockTargets\";\r\nimport type { NodeMaterialBuildStateSharedData } from \"./nodeMaterialBuildStateSharedData\";\r\nimport { Effect } from \"../effect\";\r\n\r\n/**\r\n * Class used to store node based material build state\r\n */\r\nexport class NodeMaterialBuildState {\r\n /** Gets or sets a boolean indicating if the current state can emit uniform buffers */\r\n public supportUniformBuffers = false;\r\n /**\r\n * Gets the list of emitted attributes\r\n */\r\n public attributes = new Array();\r\n /**\r\n * Gets the list of emitted uniforms\r\n */\r\n public uniforms = new Array();\r\n /**\r\n * Gets the list of emitted constants\r\n */\r\n public constants = new Array();\r\n /**\r\n * Gets the list of emitted samplers\r\n */\r\n public samplers = new Array();\r\n /**\r\n * Gets the list of emitted functions\r\n */\r\n public functions: { [key: string]: string } = {};\r\n /**\r\n * Gets the list of emitted extensions\r\n */\r\n public extensions: { [key: string]: string } = {};\r\n\r\n /**\r\n * Gets the target of the compilation state\r\n */\r\n public target: NodeMaterialBlockTargets;\r\n /**\r\n * Gets the list of emitted counters\r\n */\r\n public counters: { [key: string]: number } = {};\r\n\r\n /**\r\n * Shared data between multiple NodeMaterialBuildState instances\r\n */\r\n public sharedData: NodeMaterialBuildStateSharedData;\r\n\r\n /** @internal */\r\n public _vertexState: NodeMaterialBuildState;\r\n\r\n /** @internal */\r\n public _attributeDeclaration = \"\";\r\n /** @internal */\r\n public _uniformDeclaration = \"\";\r\n /** @internal */\r\n public _constantDeclaration = \"\";\r\n /** @internal */\r\n public _samplerDeclaration = \"\";\r\n /** @internal */\r\n public _varyingTransfer = \"\";\r\n /** @internal */\r\n public _injectAtEnd = \"\";\r\n\r\n private _repeatableContentAnchorIndex = 0;\r\n /** @internal */\r\n public _builtCompilationString = \"\";\r\n\r\n /**\r\n * Gets the emitted compilation strings\r\n */\r\n public compilationString = \"\";\r\n\r\n /**\r\n * Finalize the compilation strings\r\n * @param state defines the current compilation state\r\n */\r\n public finalize(state: NodeMaterialBuildState) {\r\n const emitComments = state.sharedData.emitComments;\r\n const isFragmentMode = this.target === NodeMaterialBlockTargets.Fragment;\r\n\r\n this.compilationString = `\\n${emitComments ? \"//Entry point\\n\" : \"\"}void main(void) {\\n${this.compilationString}`;\r\n\r\n if (this._constantDeclaration) {\r\n this.compilationString = `\\n${emitComments ? \"//Constants\\n\" : \"\"}${this._constantDeclaration}\\n${this.compilationString}`;\r\n }\r\n\r\n let functionCode = \"\";\r\n for (const functionName in this.functions) {\r\n functionCode += this.functions[functionName] + `\\n`;\r\n }\r\n this.compilationString = `\\n${functionCode}\\n${this.compilationString}`;\r\n\r\n if (!isFragmentMode && this._varyingTransfer) {\r\n this.compilationString = `${this.compilationString}\\n${this._varyingTransfer}`;\r\n }\r\n\r\n if (this._injectAtEnd) {\r\n this.compilationString = `${this.compilationString}\\n${this._injectAtEnd}`;\r\n }\r\n\r\n this.compilationString = `${this.compilationString}\\n}`;\r\n\r\n if (this.sharedData.varyingDeclaration) {\r\n this.compilationString = `\\n${emitComments ? \"//Varyings\\n\" : \"\"}${this.sharedData.varyingDeclaration}\\n${this.compilationString}`;\r\n }\r\n\r\n if (this._samplerDeclaration) {\r\n this.compilationString = `\\n${emitComments ? \"//Samplers\\n\" : \"\"}${this._samplerDeclaration}\\n${this.compilationString}`;\r\n }\r\n\r\n if (this._uniformDeclaration) {\r\n this.compilationString = `\\n${emitComments ? \"//Uniforms\\n\" : \"\"}${this._uniformDeclaration}\\n${this.compilationString}`;\r\n }\r\n\r\n if (this._attributeDeclaration && !isFragmentMode) {\r\n this.compilationString = `\\n${emitComments ? \"//Attributes\\n\" : \"\"}${this._attributeDeclaration}\\n${this.compilationString}`;\r\n }\r\n\r\n this.compilationString = \"precision highp float;\\n\" + this.compilationString;\r\n this.compilationString = \"#if defined(WEBGL2) || defines(WEBGPU)\\nprecision highp sampler2DArray;\\n#endif\\n\" + this.compilationString;\r\n\r\n for (const extensionName in this.extensions) {\r\n const extension = this.extensions[extensionName];\r\n this.compilationString = `\\n${extension}\\n${this.compilationString}`;\r\n }\r\n\r\n this._builtCompilationString = this.compilationString;\r\n }\r\n\r\n /** @internal */\r\n public get _repeatableContentAnchor(): string {\r\n return `###___ANCHOR${this._repeatableContentAnchorIndex++}___###`;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getFreeVariableName(prefix: string): string {\r\n prefix = prefix.replace(/[^a-zA-Z_]+/g, \"\");\r\n\r\n if (this.sharedData.variableNames[prefix] === undefined) {\r\n this.sharedData.variableNames[prefix] = 0;\r\n\r\n // Check reserved words\r\n if (prefix === \"output\" || prefix === \"texture\") {\r\n return prefix + this.sharedData.variableNames[prefix];\r\n }\r\n\r\n return prefix;\r\n } else {\r\n this.sharedData.variableNames[prefix]++;\r\n }\r\n\r\n return prefix + this.sharedData.variableNames[prefix];\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getFreeDefineName(prefix: string): string {\r\n if (this.sharedData.defineNames[prefix] === undefined) {\r\n this.sharedData.defineNames[prefix] = 0;\r\n } else {\r\n this.sharedData.defineNames[prefix]++;\r\n }\r\n\r\n return prefix + this.sharedData.defineNames[prefix];\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _excludeVariableName(name: string) {\r\n this.sharedData.variableNames[name] = 0;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _emit2DSampler(name: string) {\r\n if (this.samplers.indexOf(name) < 0) {\r\n this._samplerDeclaration += `uniform sampler2D ${name};\\n`;\r\n this.samplers.push(name);\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _emit2DArraySampler(name: string) {\r\n if (this.samplers.indexOf(name) < 0) {\r\n this._samplerDeclaration += `uniform sampler2DArray ${name};\\n`;\r\n this.samplers.push(name);\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _getGLType(type: NodeMaterialBlockConnectionPointTypes): string {\r\n switch (type) {\r\n case NodeMaterialBlockConnectionPointTypes.Float:\r\n return \"float\";\r\n case NodeMaterialBlockConnectionPointTypes.Int:\r\n return \"int\";\r\n case NodeMaterialBlockConnectionPointTypes.Vector2:\r\n return \"vec2\";\r\n case NodeMaterialBlockConnectionPointTypes.Color3:\r\n case NodeMaterialBlockConnectionPointTypes.Vector3:\r\n return \"vec3\";\r\n case NodeMaterialBlockConnectionPointTypes.Color4:\r\n case NodeMaterialBlockConnectionPointTypes.Vector4:\r\n return \"vec4\";\r\n case NodeMaterialBlockConnectionPointTypes.Matrix:\r\n return \"mat4\";\r\n }\r\n\r\n return \"\";\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _emitExtension(name: string, extension: string, define: string = \"\") {\r\n if (this.extensions[name]) {\r\n return;\r\n }\r\n\r\n if (define) {\r\n extension = `#if ${define}\\n${extension}\\n#endif`;\r\n }\r\n this.extensions[name] = extension;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _emitFunction(name: string, code: string, comments: string) {\r\n if (this.functions[name]) {\r\n return;\r\n }\r\n\r\n if (this.sharedData.emitComments) {\r\n code = comments + `\\n` + code;\r\n }\r\n\r\n this.functions[name] = code;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _emitCodeFromInclude(\r\n includeName: string,\r\n comments: string,\r\n options?: {\r\n replaceStrings?: { search: RegExp; replace: string }[];\r\n repeatKey?: string;\r\n substitutionVars?: string;\r\n }\r\n ) {\r\n if (options && options.repeatKey) {\r\n return `#include<${includeName}>${options.substitutionVars ? \"(\" + options.substitutionVars + \")\" : \"\"}[0..${options.repeatKey}]\\n`;\r\n }\r\n\r\n let code = Effect.IncludesShadersStore[includeName] + \"\\n\";\r\n\r\n if (this.sharedData.emitComments) {\r\n code = comments + `\\n` + code;\r\n }\r\n\r\n if (!options) {\r\n return code;\r\n }\r\n\r\n if (options.replaceStrings) {\r\n for (let index = 0; index < options.replaceStrings.length; index++) {\r\n const replaceString = options.replaceStrings[index];\r\n code = code.replace(replaceString.search, replaceString.replace);\r\n }\r\n }\r\n\r\n return code;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _emitFunctionFromInclude(\r\n includeName: string,\r\n comments: string,\r\n options?: {\r\n repeatKey?: string;\r\n substitutionVars?: string;\r\n removeAttributes?: boolean;\r\n removeUniforms?: boolean;\r\n removeVaryings?: boolean;\r\n removeIfDef?: boolean;\r\n replaceStrings?: { search: RegExp; replace: string }[];\r\n },\r\n storeKey: string = \"\"\r\n ) {\r\n const key = includeName + storeKey;\r\n if (this.functions[key]) {\r\n return;\r\n }\r\n\r\n if (!options || (!options.removeAttributes && !options.removeUniforms && !options.removeVaryings && !options.removeIfDef && !options.replaceStrings)) {\r\n if (options && options.repeatKey) {\r\n this.functions[key] = `#include<${includeName}>${options.substitutionVars ? \"(\" + options.substitutionVars + \")\" : \"\"}[0..${options.repeatKey}]\\n`;\r\n } else {\r\n this.functions[key] = `#include<${includeName}>${options?.substitutionVars ? \"(\" + options?.substitutionVars + \")\" : \"\"}\\n`;\r\n }\r\n\r\n if (this.sharedData.emitComments) {\r\n this.functions[key] = comments + `\\n` + this.functions[key];\r\n }\r\n\r\n return;\r\n }\r\n\r\n this.functions[key] = Effect.IncludesShadersStore[includeName];\r\n\r\n if (this.sharedData.emitComments) {\r\n this.functions[key] = comments + `\\n` + this.functions[key];\r\n }\r\n\r\n if (options.removeIfDef) {\r\n this.functions[key] = this.functions[key].replace(/^\\s*?#ifdef.+$/gm, \"\");\r\n this.functions[key] = this.functions[key].replace(/^\\s*?#endif.*$/gm, \"\");\r\n this.functions[key] = this.functions[key].replace(/^\\s*?#else.*$/gm, \"\");\r\n this.functions[key] = this.functions[key].replace(/^\\s*?#elif.*$/gm, \"\");\r\n }\r\n\r\n if (options.removeAttributes) {\r\n this.functions[key] = this.functions[key].replace(/\\s*?attribute .+?;/g, \"\\n\");\r\n }\r\n\r\n if (options.removeUniforms) {\r\n this.functions[key] = this.functions[key].replace(/\\s*?uniform .*?;/g, \"\\n\");\r\n }\r\n\r\n if (options.removeVaryings) {\r\n this.functions[key] = this.functions[key].replace(/\\s*?(varying|in) .+?;/g, \"\\n\");\r\n }\r\n\r\n if (options.replaceStrings) {\r\n for (let index = 0; index < options.replaceStrings.length; index++) {\r\n const replaceString = options.replaceStrings[index];\r\n this.functions[key] = this.functions[key].replace(replaceString.search, replaceString.replace);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _registerTempVariable(name: string) {\r\n if (this.sharedData.temps.indexOf(name) !== -1) {\r\n return false;\r\n }\r\n\r\n this.sharedData.temps.push(name);\r\n return true;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _emitVaryingFromString(name: string, type: string, define: string = \"\", notDefine = false) {\r\n if (this.sharedData.varyings.indexOf(name) !== -1) {\r\n return false;\r\n }\r\n\r\n this.sharedData.varyings.push(name);\r\n\r\n if (define) {\r\n if (define.startsWith(\"defined(\")) {\r\n this.sharedData.varyingDeclaration += `#if ${define}\\n`;\r\n } else {\r\n this.sharedData.varyingDeclaration += `${notDefine ? \"#ifndef\" : \"#ifdef\"} ${define}\\n`;\r\n }\r\n }\r\n this.sharedData.varyingDeclaration += `varying ${type} ${name};\\n`;\r\n if (define) {\r\n this.sharedData.varyingDeclaration += `#endif\\n`;\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _emitUniformFromString(name: string, type: string, define: string = \"\", notDefine = false) {\r\n if (this.uniforms.indexOf(name) !== -1) {\r\n return;\r\n }\r\n\r\n this.uniforms.push(name);\r\n\r\n if (define) {\r\n if (define.startsWith(\"defined(\")) {\r\n this._uniformDeclaration += `#if ${define}\\n`;\r\n } else {\r\n this._uniformDeclaration += `${notDefine ? \"#ifndef\" : \"#ifdef\"} ${define}\\n`;\r\n }\r\n }\r\n this._uniformDeclaration += `uniform ${type} ${name};\\n`;\r\n if (define) {\r\n this._uniformDeclaration += `#endif\\n`;\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _emitFloat(value: number) {\r\n if (value.toString() === value.toFixed(0)) {\r\n return `${value}.0`;\r\n }\r\n\r\n return value.toString();\r\n }\r\n}\r\n","import type { NodeMaterialConnectionPoint } from \"./nodeMaterialBlockConnectionPoint\";\r\nimport type { NodeMaterialBlock } from \"./nodeMaterialBlock\";\r\nimport type { InputBlock } from \"./Blocks/Input/inputBlock\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { Immutable } from \"../../types\";\r\nimport type { NodeMaterialTextureBlocks } from \"./nodeMaterial\";\r\n\r\n/**\r\n * Class used to store shared data between 2 NodeMaterialBuildState\r\n */\r\nexport class NodeMaterialBuildStateSharedData {\r\n /**\r\n * Gets the list of emitted varyings\r\n */\r\n public temps = new Array();\r\n\r\n /**\r\n * Gets the list of emitted varyings\r\n */\r\n public varyings = new Array();\r\n\r\n /**\r\n * Gets the varying declaration string\r\n */\r\n public varyingDeclaration = \"\";\r\n\r\n /**\r\n * List of the fragment output nodes\r\n */\r\n public fragmentOutputNodes: Immutable>;\r\n\r\n /**\r\n * Input blocks\r\n */\r\n public inputBlocks = new Array();\r\n\r\n /**\r\n * Input blocks\r\n */\r\n public textureBlocks = new Array();\r\n\r\n /**\r\n * Bindable blocks (Blocks that need to set data to the effect)\r\n */\r\n public bindableBlocks = new Array();\r\n\r\n /**\r\n * Bindable blocks (Blocks that need to set data to the effect) that will always be called (by bindForSubMesh), contrary to bindableBlocks that won't be called if _mustRebind() returns false\r\n */\r\n public forcedBindableBlocks = new Array();\r\n\r\n /**\r\n * List of blocks that can provide a compilation fallback\r\n */\r\n public blocksWithFallbacks = new Array();\r\n\r\n /**\r\n * List of blocks that can provide a define update\r\n */\r\n public blocksWithDefines = new Array();\r\n\r\n /**\r\n * List of blocks that can provide a repeatable content\r\n */\r\n public repeatableContentBlocks = new Array();\r\n\r\n /**\r\n * List of blocks that can provide a dynamic list of uniforms\r\n */\r\n public dynamicUniformBlocks = new Array();\r\n\r\n /**\r\n * List of blocks that can block the isReady function for the material\r\n */\r\n public blockingBlocks = new Array();\r\n\r\n /**\r\n * Gets the list of animated inputs\r\n */\r\n public animatedInputs = new Array();\r\n\r\n /**\r\n * Build Id used to avoid multiple recompilations\r\n */\r\n public buildId: number;\r\n\r\n /** List of emitted variables */\r\n public variableNames: { [key: string]: number } = {};\r\n\r\n /** List of emitted defines */\r\n public defineNames: { [key: string]: number } = {};\r\n\r\n /** Should emit comments? */\r\n public emitComments: boolean;\r\n\r\n /** Emit build activity */\r\n public verbose: boolean;\r\n\r\n /** Gets or sets the hosting scene */\r\n public scene: Scene;\r\n\r\n /**\r\n * Gets the compilation hints emitted at compilation time\r\n */\r\n public hints = {\r\n needWorldViewMatrix: false,\r\n needWorldViewProjectionMatrix: false,\r\n needAlphaBlending: false,\r\n needAlphaTesting: false,\r\n };\r\n\r\n /**\r\n * List of compilation checks\r\n */\r\n public checks = {\r\n emitVertex: false,\r\n emitFragment: false,\r\n notConnectedNonOptionalInputs: new Array(),\r\n };\r\n\r\n /**\r\n * Is vertex program allowed to be empty?\r\n */\r\n public allowEmptyVertexProgram: boolean = false;\r\n\r\n /** Creates a new shared data */\r\n public constructor() {\r\n // Exclude usual attributes from free variable names\r\n this.variableNames[\"position\"] = 0;\r\n this.variableNames[\"normal\"] = 0;\r\n this.variableNames[\"tangent\"] = 0;\r\n this.variableNames[\"uv\"] = 0;\r\n this.variableNames[\"uv2\"] = 0;\r\n this.variableNames[\"uv3\"] = 0;\r\n this.variableNames[\"uv4\"] = 0;\r\n this.variableNames[\"uv5\"] = 0;\r\n this.variableNames[\"uv6\"] = 0;\r\n this.variableNames[\"color\"] = 0;\r\n this.variableNames[\"matricesIndices\"] = 0;\r\n this.variableNames[\"matricesWeights\"] = 0;\r\n this.variableNames[\"matricesIndicesExtra\"] = 0;\r\n this.variableNames[\"matricesWeightsExtra\"] = 0;\r\n this.variableNames[\"diffuseBase\"] = 0;\r\n this.variableNames[\"specularBase\"] = 0;\r\n this.variableNames[\"worldPos\"] = 0;\r\n this.variableNames[\"shadow\"] = 0;\r\n this.variableNames[\"view\"] = 0;\r\n\r\n // Exclude known varyings\r\n this.variableNames[\"vTBN\"] = 0;\r\n\r\n // Exclude defines\r\n this.defineNames[\"MAINUV0\"] = 0;\r\n this.defineNames[\"MAINUV1\"] = 0;\r\n this.defineNames[\"MAINUV2\"] = 0;\r\n this.defineNames[\"MAINUV3\"] = 0;\r\n this.defineNames[\"MAINUV4\"] = 0;\r\n this.defineNames[\"MAINUV5\"] = 0;\r\n this.defineNames[\"MAINUV6\"] = 0;\r\n this.defineNames[\"MAINUV7\"] = 0;\r\n }\r\n\r\n /**\r\n * Emits console errors and exceptions if there is a failing check\r\n */\r\n public emitErrors() {\r\n let errorMessage = \"\";\r\n\r\n if (!this.checks.emitVertex && !this.allowEmptyVertexProgram) {\r\n errorMessage += \"NodeMaterial does not have a vertex output. You need to at least add a block that generates a glPosition value.\\n\";\r\n }\r\n if (!this.checks.emitFragment) {\r\n errorMessage += \"NodeMaterial does not have a fragment output. You need to at least add a block that generates a glFragColor value.\\n\";\r\n }\r\n for (const notConnectedInput of this.checks.notConnectedNonOptionalInputs) {\r\n errorMessage += `input ${notConnectedInput.name} from block ${\r\n notConnectedInput.ownerBlock.name\r\n }[${notConnectedInput.ownerBlock.getClassName()}] is not connected and is not optional.\\n`;\r\n }\r\n\r\n if (errorMessage) {\r\n throw \"Build of NodeMaterial failed:\\n\" + errorMessage;\r\n }\r\n }\r\n}\r\n","import { NodeMaterialBlockConnectionPointTypes } from \"./Enums/nodeMaterialBlockConnectionPointTypes\";\r\nimport type { NodeMaterialBuildState } from \"./nodeMaterialBuildState\";\r\nimport type { Nullable } from \"../../types\";\r\nimport { NodeMaterialConnectionPoint, NodeMaterialConnectionPointDirection } from \"./nodeMaterialBlockConnectionPoint\";\r\nimport { NodeMaterialBlockTargets } from \"./Enums/nodeMaterialBlockTargets\";\r\nimport type { Effect } from \"../effect\";\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport type { Mesh } from \"../../Meshes/mesh\";\r\nimport type { SubMesh } from \"../../Meshes/subMesh\";\r\nimport type { NodeMaterial, NodeMaterialDefines } from \"./nodeMaterial\";\r\nimport type { InputBlock } from \"./Blocks/Input/inputBlock\";\r\nimport { UniqueIdGenerator } from \"../../Misc/uniqueIdGenerator\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { GetClass } from \"../../Misc/typeStore\";\r\nimport type { EffectFallbacks } from \"../effectFallbacks\";\r\n\r\n/**\r\n * Defines a block that can be used inside a node based material\r\n */\r\nexport class NodeMaterialBlock {\r\n private _buildId: number;\r\n private _buildTarget: NodeMaterialBlockTargets;\r\n protected _target: NodeMaterialBlockTargets;\r\n private _isFinalMerger = false;\r\n private _isInput = false;\r\n private _isTeleportOut = false;\r\n private _isTeleportIn = false;\r\n private _name = \"\";\r\n protected _isUnique = false;\r\n\r\n /** Gets or sets a boolean indicating that only one input can be connected at a time */\r\n public inputsAreExclusive = false;\r\n\r\n /** @internal */\r\n public _codeVariableName = \"\";\r\n\r\n /** @internal */\r\n public _inputs = new Array();\r\n /** @internal */\r\n public _outputs = new Array();\r\n\r\n /** @internal */\r\n public _preparationId: number;\r\n\r\n /** @internal */\r\n public readonly _originalTargetIsNeutral: boolean;\r\n\r\n /**\r\n * Gets the name of the block\r\n */\r\n public get name(): string {\r\n return this._name;\r\n }\r\n\r\n /**\r\n * Sets the name of the block. Will check if the name is valid.\r\n */\r\n public set name(newName: string) {\r\n if (!this.validateBlockName(newName)) {\r\n return;\r\n }\r\n\r\n this._name = newName;\r\n }\r\n\r\n /**\r\n * Gets or sets the unique id of the node\r\n */\r\n public uniqueId: number;\r\n\r\n /**\r\n * Gets or sets the comments associated with this block\r\n */\r\n public comments: string = \"\";\r\n\r\n /**\r\n * Gets a boolean indicating that this block can only be used once per NodeMaterial\r\n */\r\n public get isUnique() {\r\n return this._isUnique;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating that this block is an end block (e.g. it is generating a system value)\r\n */\r\n public get isFinalMerger(): boolean {\r\n return this._isFinalMerger;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating that this block is an input (e.g. it sends data to the shader)\r\n */\r\n public get isInput(): boolean {\r\n return this._isInput;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if this block is a teleport out\r\n */\r\n public get isTeleportOut(): boolean {\r\n return this._isTeleportOut;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if this block is a teleport in\r\n */\r\n public get isTeleportIn(): boolean {\r\n return this._isTeleportIn;\r\n }\r\n\r\n /**\r\n * Gets or sets the build Id\r\n */\r\n public get buildId(): number {\r\n return this._buildId;\r\n }\r\n\r\n public set buildId(value: number) {\r\n this._buildId = value;\r\n }\r\n\r\n /**\r\n * Gets or sets the target of the block\r\n */\r\n public get target() {\r\n return this._target;\r\n }\r\n\r\n public set target(value: NodeMaterialBlockTargets) {\r\n if ((this._target & value) !== 0) {\r\n return;\r\n }\r\n this._target = value;\r\n }\r\n\r\n /**\r\n * Gets the list of input points\r\n */\r\n public get inputs(): NodeMaterialConnectionPoint[] {\r\n return this._inputs;\r\n }\r\n\r\n /** Gets the list of output points */\r\n public get outputs(): NodeMaterialConnectionPoint[] {\r\n return this._outputs;\r\n }\r\n\r\n /**\r\n * Find an input by its name\r\n * @param name defines the name of the input to look for\r\n * @returns the input or null if not found\r\n */\r\n public getInputByName(name: string) {\r\n const filter = this._inputs.filter((e) => e.name === name);\r\n\r\n if (filter.length) {\r\n return filter[0];\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Find an output by its name\r\n * @param name defines the name of the output to look for\r\n * @returns the output or null if not found\r\n */\r\n public getOutputByName(name: string) {\r\n const filter = this._outputs.filter((e) => e.name === name);\r\n\r\n if (filter.length) {\r\n return filter[0];\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /** Gets or sets a boolean indicating that this input can be edited in the Inspector (false by default) */\r\n public visibleInInspector = false;\r\n\r\n /** Gets or sets a boolean indicating that this input can be edited from a collapsed frame */\r\n public visibleOnFrame = false;\r\n\r\n /**\r\n * Creates a new NodeMaterialBlock\r\n * @param name defines the block name\r\n * @param target defines the target of that block (Vertex by default)\r\n * @param isFinalMerger defines a boolean indicating that this block is an end block (e.g. it is generating a system value). Default is false\r\n */\r\n public constructor(name: string, target = NodeMaterialBlockTargets.Vertex, isFinalMerger = false) {\r\n this._target = target;\r\n this._originalTargetIsNeutral = target === NodeMaterialBlockTargets.Neutral;\r\n this._isFinalMerger = isFinalMerger;\r\n this._isInput = this.getClassName() === \"InputBlock\";\r\n this._isTeleportOut = this.getClassName() === \"NodeMaterialTeleportOutBlock\";\r\n this._isTeleportIn = this.getClassName() === \"NodeMaterialTeleportInBlock\";\r\n this._name = name;\r\n this.uniqueId = UniqueIdGenerator.UniqueId;\r\n }\r\n\r\n /** @internal */\r\n public _setInitialTarget(target: NodeMaterialBlockTargets): void {\r\n this._target = target;\r\n (this._originalTargetIsNeutral as boolean) = target === NodeMaterialBlockTargets.Neutral;\r\n }\r\n\r\n /**\r\n * Initialize the block and prepare the context for build\r\n * @param state defines the state that will be used for the build\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public initialize(state: NodeMaterialBuildState) {\r\n // Do nothing\r\n }\r\n\r\n /**\r\n * Bind data to effect. Will only be called for blocks with isBindable === true\r\n * @param effect defines the effect to bind data to\r\n * @param nodeMaterial defines the hosting NodeMaterial\r\n * @param mesh defines the mesh that will be rendered\r\n * @param subMesh defines the submesh that will be rendered\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public bind(effect: Effect, nodeMaterial: NodeMaterial, mesh?: Mesh, subMesh?: SubMesh) {\r\n // Do nothing\r\n }\r\n\r\n protected _declareOutput(output: NodeMaterialConnectionPoint, state: NodeMaterialBuildState): string {\r\n return `${state._getGLType(output.type)} ${output.associatedVariableName}`;\r\n }\r\n\r\n protected _writeVariable(currentPoint: NodeMaterialConnectionPoint): string {\r\n const connectionPoint = currentPoint.connectedPoint;\r\n\r\n if (connectionPoint) {\r\n return `${currentPoint.associatedVariableName}`;\r\n }\r\n\r\n return `0.`;\r\n }\r\n\r\n protected _writeFloat(value: number) {\r\n let stringVersion = value.toString();\r\n\r\n if (stringVersion.indexOf(\".\") === -1) {\r\n stringVersion += \".0\";\r\n }\r\n return `${stringVersion}`;\r\n }\r\n\r\n /**\r\n * Gets the current class name e.g. \"NodeMaterialBlock\"\r\n * @returns the class name\r\n */\r\n public getClassName() {\r\n return \"NodeMaterialBlock\";\r\n }\r\n\r\n /** Gets a boolean indicating that this connection will be used in the fragment shader */\r\n public isConnectedInFragmentShader() {\r\n return this.outputs.some((o) => o.isConnectedInFragmentShader);\r\n }\r\n\r\n /**\r\n * Register a new input. Must be called inside a block constructor\r\n * @param name defines the connection point name\r\n * @param type defines the connection point type\r\n * @param isOptional defines a boolean indicating that this input can be omitted\r\n * @param target defines the target to use to limit the connection point (will be VertexAndFragment by default)\r\n * @param point an already created connection point. If not provided, create a new one\r\n * @returns the current block\r\n */\r\n public registerInput(\r\n name: string,\r\n type: NodeMaterialBlockConnectionPointTypes,\r\n isOptional: boolean = false,\r\n target?: NodeMaterialBlockTargets,\r\n point?: NodeMaterialConnectionPoint\r\n ) {\r\n point = point ?? new NodeMaterialConnectionPoint(name, this, NodeMaterialConnectionPointDirection.Input);\r\n point.type = type;\r\n point.isOptional = isOptional;\r\n if (target) {\r\n point.target = target;\r\n }\r\n\r\n this._inputs.push(point);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Register a new output. Must be called inside a block constructor\r\n * @param name defines the connection point name\r\n * @param type defines the connection point type\r\n * @param target defines the target to use to limit the connection point (will be VertexAndFragment by default)\r\n * @param point an already created connection point. If not provided, create a new one\r\n * @returns the current block\r\n */\r\n public registerOutput(name: string, type: NodeMaterialBlockConnectionPointTypes, target?: NodeMaterialBlockTargets, point?: NodeMaterialConnectionPoint) {\r\n point = point ?? new NodeMaterialConnectionPoint(name, this, NodeMaterialConnectionPointDirection.Output);\r\n point.type = type;\r\n if (target) {\r\n point.target = target;\r\n }\r\n\r\n this._outputs.push(point);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Will return the first available input e.g. the first one which is not an uniform or an attribute\r\n * @param forOutput defines an optional connection point to check compatibility with\r\n * @returns the first available input or null\r\n */\r\n public getFirstAvailableInput(forOutput: Nullable = null) {\r\n for (const input of this._inputs) {\r\n if (!input.connectedPoint) {\r\n if (!forOutput || forOutput.type === input.type || input.type === NodeMaterialBlockConnectionPointTypes.AutoDetect) {\r\n return input;\r\n }\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Will return the first available output e.g. the first one which is not yet connected and not a varying\r\n * @param forBlock defines an optional block to check compatibility with\r\n * @returns the first available input or null\r\n */\r\n public getFirstAvailableOutput(forBlock: Nullable = null) {\r\n for (const output of this._outputs) {\r\n if (!forBlock || !forBlock.target || forBlock.target === NodeMaterialBlockTargets.Neutral || (forBlock.target & output.target) !== 0) {\r\n return output;\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets the sibling of the given output\r\n * @param current defines the current output\r\n * @returns the next output in the list or null\r\n */\r\n public getSiblingOutput(current: NodeMaterialConnectionPoint) {\r\n const index = this._outputs.indexOf(current);\r\n\r\n if (index === -1 || index >= this._outputs.length) {\r\n return null;\r\n }\r\n\r\n return this._outputs[index + 1];\r\n }\r\n\r\n /**\r\n * Checks if the current block is an ancestor of a given block\r\n * @param block defines the potential descendant block to check\r\n * @returns true if block is a descendant\r\n */\r\n public isAnAncestorOf(block: NodeMaterialBlock): boolean {\r\n for (const output of this._outputs) {\r\n if (!output.hasEndpoints) {\r\n continue;\r\n }\r\n\r\n for (const endpoint of output.endpoints) {\r\n if (endpoint.ownerBlock === block) {\r\n return true;\r\n }\r\n if (endpoint.ownerBlock.isAnAncestorOf(block)) {\r\n return true;\r\n }\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Connect current block with another block\r\n * @param other defines the block to connect with\r\n * @param options define the various options to help pick the right connections\r\n * @param options.input\r\n * @param options.output\r\n * @param options.outputSwizzle\r\n * @returns the current block\r\n */\r\n public connectTo(\r\n other: NodeMaterialBlock,\r\n options?: {\r\n input?: string;\r\n output?: string;\r\n outputSwizzle?: string;\r\n }\r\n ) {\r\n if (this._outputs.length === 0) {\r\n return;\r\n }\r\n\r\n let output = options && options.output ? this.getOutputByName(options.output) : this.getFirstAvailableOutput(other);\r\n\r\n let notFound = true;\r\n while (notFound) {\r\n const input = options && options.input ? other.getInputByName(options.input) : other.getFirstAvailableInput(output);\r\n\r\n if (output && input && output.canConnectTo(input)) {\r\n output.connectTo(input);\r\n notFound = false;\r\n } else if (!output) {\r\n throw \"Unable to find a compatible match\";\r\n } else {\r\n output = this.getSiblingOutput(output);\r\n }\r\n }\r\n\r\n return this;\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n protected _buildBlock(state: NodeMaterialBuildState) {\r\n // Empty. Must be defined by child nodes\r\n }\r\n\r\n /**\r\n * Add uniforms, samplers and uniform buffers at compilation time\r\n * @param state defines the state to update\r\n * @param nodeMaterial defines the node material requesting the update\r\n * @param defines defines the material defines to update\r\n * @param uniformBuffers defines the list of uniform buffer names\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public updateUniformsAndSamples(state: NodeMaterialBuildState, nodeMaterial: NodeMaterial, defines: NodeMaterialDefines, uniformBuffers: string[]) {\r\n // Do nothing\r\n }\r\n\r\n /**\r\n * Add potential fallbacks if shader compilation fails\r\n * @param mesh defines the mesh to be rendered\r\n * @param fallbacks defines the current prioritized list of fallbacks\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public provideFallbacks(mesh: AbstractMesh, fallbacks: EffectFallbacks) {\r\n // Do nothing\r\n }\r\n\r\n /**\r\n * Initialize defines for shader compilation\r\n * @param mesh defines the mesh to be rendered\r\n * @param nodeMaterial defines the node material requesting the update\r\n * @param defines defines the material defines to update\r\n * @param useInstances specifies that instances should be used\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public initializeDefines(mesh: AbstractMesh, nodeMaterial: NodeMaterial, defines: NodeMaterialDefines, useInstances: boolean = false) {}\r\n\r\n /**\r\n * Update defines for shader compilation\r\n * @param mesh defines the mesh to be rendered\r\n * @param nodeMaterial defines the node material requesting the update\r\n * @param defines defines the material defines to update\r\n * @param useInstances specifies that instances should be used\r\n * @param subMesh defines which submesh to render\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public prepareDefines(mesh: AbstractMesh, nodeMaterial: NodeMaterial, defines: NodeMaterialDefines, useInstances: boolean = false, subMesh?: SubMesh) {\r\n // Do nothing\r\n }\r\n\r\n /**\r\n * Lets the block try to connect some inputs automatically\r\n * @param material defines the hosting NodeMaterial\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public autoConfigure(material: NodeMaterial) {\r\n // Do nothing\r\n }\r\n\r\n /**\r\n * Function called when a block is declared as repeatable content generator\r\n * @param vertexShaderState defines the current compilation state for the vertex shader\r\n * @param fragmentShaderState defines the current compilation state for the fragment shader\r\n * @param mesh defines the mesh to be rendered\r\n * @param defines defines the material defines to update\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public replaceRepeatableContent(vertexShaderState: NodeMaterialBuildState, fragmentShaderState: NodeMaterialBuildState, mesh: AbstractMesh, defines: NodeMaterialDefines) {\r\n // Do nothing\r\n }\r\n\r\n /** Gets a boolean indicating that the code of this block will be promoted to vertex shader even if connected to fragment output */\r\n public get willBeGeneratedIntoVertexShaderFromFragmentShader(): boolean {\r\n if (this.isInput || this.isFinalMerger) {\r\n return false;\r\n }\r\n\r\n if (this._outputs.some((o) => o.isDirectlyConnectedToVertexOutput)) {\r\n return false;\r\n }\r\n\r\n if (this.target === NodeMaterialBlockTargets.Vertex) {\r\n return false;\r\n }\r\n\r\n if (this.target === NodeMaterialBlockTargets.VertexAndFragment || this.target === NodeMaterialBlockTargets.Neutral) {\r\n if (this._outputs.some((o) => o.isConnectedInVertexShader)) {\r\n return true;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Checks if the block is ready\r\n * @param mesh defines the mesh to be rendered\r\n * @param nodeMaterial defines the node material requesting the update\r\n * @param defines defines the material defines to update\r\n * @param useInstances specifies that instances should be used\r\n * @returns true if the block is ready\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public isReady(mesh: AbstractMesh, nodeMaterial: NodeMaterial, defines: NodeMaterialDefines, useInstances: boolean = false) {\r\n return true;\r\n }\r\n\r\n protected _linkConnectionTypes(inputIndex0: number, inputIndex1: number, looseCoupling = false) {\r\n if (looseCoupling) {\r\n this._inputs[inputIndex1]._acceptedConnectionPointType = this._inputs[inputIndex0];\r\n } else {\r\n this._inputs[inputIndex0]._linkedConnectionSource = this._inputs[inputIndex1];\r\n }\r\n this._inputs[inputIndex1]._linkedConnectionSource = this._inputs[inputIndex0];\r\n }\r\n\r\n private _processBuild(block: NodeMaterialBlock, state: NodeMaterialBuildState, input: NodeMaterialConnectionPoint, activeBlocks: NodeMaterialBlock[]) {\r\n block.build(state, activeBlocks);\r\n\r\n const localBlockIsFragment = state._vertexState != null;\r\n const otherBlockWasGeneratedInVertexShader = block._buildTarget === NodeMaterialBlockTargets.Vertex && block.target !== NodeMaterialBlockTargets.VertexAndFragment;\r\n\r\n if (\r\n localBlockIsFragment &&\r\n ((block.target & block._buildTarget) === 0 ||\r\n (block.target & input.target) === 0 ||\r\n (this.target !== NodeMaterialBlockTargets.VertexAndFragment && otherBlockWasGeneratedInVertexShader))\r\n ) {\r\n // context switch! We need a varying\r\n if (\r\n (!block.isInput && state.target !== block._buildTarget) || // block was already emitted by vertex shader\r\n (block.isInput && (block as InputBlock).isAttribute && !(block as InputBlock)._noContextSwitch) // block is an attribute\r\n ) {\r\n const connectedPoint = input.connectedPoint!;\r\n if (state._vertexState._emitVaryingFromString(\"v_\" + connectedPoint.associatedVariableName, state._getGLType(connectedPoint.type))) {\r\n state._vertexState.compilationString += `${\"v_\" + connectedPoint.associatedVariableName} = ${connectedPoint.associatedVariableName};\\n`;\r\n }\r\n input.associatedVariableName = \"v_\" + connectedPoint.associatedVariableName;\r\n input._enforceAssociatedVariableName = true;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Validates the new name for the block node.\r\n * @param newName the new name to be given to the node.\r\n * @returns false if the name is a reserve word, else true.\r\n */\r\n public validateBlockName(newName: string) {\r\n const reservedNames: Array = [\r\n \"position\",\r\n \"normal\",\r\n \"tangent\",\r\n \"particle_positionw\",\r\n \"uv\",\r\n \"uv2\",\r\n \"uv3\",\r\n \"uv4\",\r\n \"uv5\",\r\n \"uv6\",\r\n \"position2d\",\r\n \"particle_uv\",\r\n \"matricesIndices\",\r\n \"matricesWeights\",\r\n \"world0\",\r\n \"world1\",\r\n \"world2\",\r\n \"world3\",\r\n \"particle_color\",\r\n \"particle_texturemask\",\r\n ];\r\n for (const reservedName of reservedNames) {\r\n if (newName === reservedName) {\r\n return false;\r\n }\r\n }\r\n return true;\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n protected _customBuildStep(state: NodeMaterialBuildState, activeBlocks: NodeMaterialBlock[]): void {\r\n // Must be implemented by children\r\n }\r\n\r\n /**\r\n * Compile the current node and generate the shader code\r\n * @param state defines the current compilation state (uniforms, samplers, current string)\r\n * @param activeBlocks defines the list of active blocks (i.e. blocks to compile)\r\n * @returns true if already built\r\n */\r\n public build(state: NodeMaterialBuildState, activeBlocks: NodeMaterialBlock[]): boolean {\r\n if (this._buildId === state.sharedData.buildId) {\r\n return true;\r\n }\r\n\r\n if (!this.isInput) {\r\n /** Prepare outputs */\r\n for (const output of this._outputs) {\r\n if (!output.associatedVariableName) {\r\n output.associatedVariableName = state._getFreeVariableName(output.name);\r\n }\r\n }\r\n }\r\n\r\n // Check if \"parent\" blocks are compiled\r\n for (const input of this._inputs) {\r\n if (!input.connectedPoint) {\r\n if (!input.isOptional) {\r\n // Emit a warning\r\n state.sharedData.checks.notConnectedNonOptionalInputs.push(input);\r\n }\r\n continue;\r\n }\r\n\r\n if (this.target !== NodeMaterialBlockTargets.Neutral) {\r\n if ((input.target & this.target!) === 0) {\r\n continue;\r\n }\r\n\r\n if ((input.target & state.target!) === 0) {\r\n continue;\r\n }\r\n }\r\n\r\n const block = input.connectedPoint.ownerBlock;\r\n if (block && block !== this) {\r\n this._processBuild(block, state, input, activeBlocks);\r\n }\r\n }\r\n\r\n this._customBuildStep(state, activeBlocks);\r\n\r\n if (this._buildId === state.sharedData.buildId) {\r\n return true; // Need to check again as inputs can be connected multiple time to this endpoint\r\n }\r\n\r\n // Logs\r\n if (state.sharedData.verbose) {\r\n console.log(`${state.target === NodeMaterialBlockTargets.Vertex ? \"Vertex shader\" : \"Fragment shader\"}: Building ${this.name} [${this.getClassName()}]`);\r\n }\r\n\r\n // Checks final outputs\r\n if (this.isFinalMerger) {\r\n switch (state.target) {\r\n case NodeMaterialBlockTargets.Vertex:\r\n state.sharedData.checks.emitVertex = true;\r\n break;\r\n case NodeMaterialBlockTargets.Fragment:\r\n state.sharedData.checks.emitFragment = true;\r\n break;\r\n }\r\n }\r\n\r\n if (!this.isInput && state.sharedData.emitComments) {\r\n state.compilationString += `\\n//${this.name}\\n`;\r\n }\r\n\r\n this._buildBlock(state);\r\n\r\n this._buildId = state.sharedData.buildId;\r\n this._buildTarget = state.target;\r\n\r\n // Compile connected blocks\r\n for (const output of this._outputs) {\r\n if ((output.target & state.target) === 0) {\r\n continue;\r\n }\r\n\r\n for (const endpoint of output.endpoints) {\r\n const block = endpoint.ownerBlock;\r\n\r\n if (block && (block.target & state.target) !== 0 && activeBlocks.indexOf(block) !== -1) {\r\n this._processBuild(block, state, endpoint, activeBlocks);\r\n }\r\n }\r\n }\r\n return false;\r\n }\r\n\r\n protected _inputRename(name: string) {\r\n return name;\r\n }\r\n\r\n protected _outputRename(name: string) {\r\n return name;\r\n }\r\n\r\n protected _dumpPropertiesCode() {\r\n const variableName = this._codeVariableName;\r\n return `${variableName}.visibleInInspector = ${this.visibleInInspector};\\n${variableName}.visibleOnFrame = ${this.visibleOnFrame};\\n${variableName}.target = ${this.target};\\n`;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _dumpCode(uniqueNames: string[], alreadyDumped: NodeMaterialBlock[]) {\r\n alreadyDumped.push(this);\r\n\r\n // Get unique name\r\n const nameAsVariableName = this.name.replace(/[^A-Za-z_]+/g, \"\");\r\n this._codeVariableName = nameAsVariableName || `${this.getClassName()}_${this.uniqueId}`;\r\n\r\n if (uniqueNames.indexOf(this._codeVariableName) !== -1) {\r\n let index = 0;\r\n do {\r\n index++;\r\n this._codeVariableName = nameAsVariableName + index;\r\n } while (uniqueNames.indexOf(this._codeVariableName) !== -1);\r\n }\r\n\r\n uniqueNames.push(this._codeVariableName);\r\n\r\n // Declaration\r\n let codeString = `\\n// ${this.getClassName()}\\n`;\r\n if (this.comments) {\r\n codeString += `// ${this.comments}\\n`;\r\n }\r\n codeString += `var ${this._codeVariableName} = new BABYLON.${this.getClassName()}(\"${this.name}\");\\n`;\r\n\r\n // Properties\r\n codeString += this._dumpPropertiesCode();\r\n\r\n // Inputs\r\n for (const input of this.inputs) {\r\n if (!input.isConnected) {\r\n continue;\r\n }\r\n\r\n const connectedOutput = input.connectedPoint!;\r\n const connectedBlock = connectedOutput.ownerBlock;\r\n\r\n if (alreadyDumped.indexOf(connectedBlock) === -1) {\r\n codeString += connectedBlock._dumpCode(uniqueNames, alreadyDumped);\r\n }\r\n }\r\n\r\n // Outputs\r\n for (const output of this.outputs) {\r\n if (!output.hasEndpoints) {\r\n continue;\r\n }\r\n\r\n for (const endpoint of output.endpoints) {\r\n const connectedBlock = endpoint.ownerBlock;\r\n if (connectedBlock && alreadyDumped.indexOf(connectedBlock) === -1) {\r\n codeString += connectedBlock._dumpCode(uniqueNames, alreadyDumped);\r\n }\r\n }\r\n }\r\n\r\n return codeString;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _dumpCodeForOutputConnections(alreadyDumped: NodeMaterialBlock[]) {\r\n let codeString = \"\";\r\n\r\n if (alreadyDumped.indexOf(this) !== -1) {\r\n return codeString;\r\n }\r\n\r\n alreadyDumped.push(this);\r\n\r\n for (const input of this.inputs) {\r\n if (!input.isConnected) {\r\n continue;\r\n }\r\n\r\n const connectedOutput = input.connectedPoint!;\r\n const connectedBlock = connectedOutput.ownerBlock;\r\n\r\n codeString += connectedBlock._dumpCodeForOutputConnections(alreadyDumped);\r\n codeString += `${connectedBlock._codeVariableName}.${connectedBlock._outputRename(connectedOutput.name)}.connectTo(${this._codeVariableName}.${this._inputRename(\r\n input.name\r\n )});\\n`;\r\n }\r\n\r\n return codeString;\r\n }\r\n\r\n /**\r\n * Clone the current block to a new identical block\r\n * @param scene defines the hosting scene\r\n * @param rootUrl defines the root URL to use to load textures and relative dependencies\r\n * @returns a copy of the current block\r\n */\r\n public clone(scene: Scene, rootUrl: string = \"\") {\r\n const serializationObject = this.serialize();\r\n\r\n const blockType = GetClass(serializationObject.customType);\r\n if (blockType) {\r\n const block: NodeMaterialBlock = new blockType();\r\n block._deserialize(serializationObject, scene, rootUrl);\r\n\r\n return block;\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Serializes this block in a JSON representation\r\n * @returns the serialized block object\r\n */\r\n public serialize(): any {\r\n const serializationObject: any = {};\r\n serializationObject.customType = \"BABYLON.\" + this.getClassName();\r\n serializationObject.id = this.uniqueId;\r\n serializationObject.name = this.name;\r\n serializationObject.comments = this.comments;\r\n serializationObject.visibleInInspector = this.visibleInInspector;\r\n serializationObject.visibleOnFrame = this.visibleOnFrame;\r\n serializationObject.target = this.target;\r\n\r\n serializationObject.inputs = [];\r\n serializationObject.outputs = [];\r\n\r\n for (const input of this.inputs) {\r\n serializationObject.inputs.push(input.serialize());\r\n }\r\n\r\n for (const output of this.outputs) {\r\n serializationObject.outputs.push(output.serialize(false));\r\n }\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public _deserialize(serializationObject: any, scene: Scene, rootUrl: string) {\r\n this.name = serializationObject.name;\r\n this.comments = serializationObject.comments;\r\n this.visibleInInspector = !!serializationObject.visibleInInspector;\r\n this.visibleOnFrame = !!serializationObject.visibleOnFrame;\r\n this._target = serializationObject.target ?? this.target;\r\n this._deserializePortDisplayNamesAndExposedOnFrame(serializationObject);\r\n }\r\n\r\n private _deserializePortDisplayNamesAndExposedOnFrame(serializationObject: any) {\r\n const serializedInputs = serializationObject.inputs;\r\n const serializedOutputs = serializationObject.outputs;\r\n if (serializedInputs) {\r\n serializedInputs.forEach((port: any, i: number) => {\r\n if (port.displayName) {\r\n this.inputs[i].displayName = port.displayName;\r\n }\r\n if (port.isExposedOnFrame) {\r\n this.inputs[i].isExposedOnFrame = port.isExposedOnFrame;\r\n this.inputs[i].exposedPortPosition = port.exposedPortPosition;\r\n }\r\n });\r\n }\r\n if (serializedOutputs) {\r\n serializedOutputs.forEach((port: any, i: number) => {\r\n if (port.displayName) {\r\n this.outputs[i].displayName = port.displayName;\r\n }\r\n if (port.isExposedOnFrame) {\r\n this.outputs[i].isExposedOnFrame = port.isExposedOnFrame;\r\n this.outputs[i].exposedPortPosition = port.exposedPortPosition;\r\n }\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Release resources\r\n */\r\n public dispose() {\r\n for (const input of this.inputs) {\r\n input.dispose();\r\n }\r\n\r\n for (const output of this.outputs) {\r\n output.dispose();\r\n }\r\n }\r\n}\r\n","import { NodeMaterialBlock } from \"../nodeMaterialBlock\";\r\nimport { NodeMaterialBlockConnectionPointTypes } from \"../Enums/nodeMaterialBlockConnectionPointTypes\";\r\nimport type { NodeMaterialBuildState } from \"../nodeMaterialBuildState\";\r\nimport { NodeMaterialBlockTargets } from \"../Enums/nodeMaterialBlockTargets\";\r\nimport type { NodeMaterialConnectionPoint } from \"../nodeMaterialBlockConnectionPoint\";\r\nimport { RegisterClass } from \"../../../Misc/typeStore\";\r\nimport type { Scene } from \"../../../scene\";\r\nimport type { InputBlock } from \"./Input/inputBlock\";\r\nimport type { AbstractMesh } from \"../../../Meshes/abstractMesh\";\r\nimport type { NodeMaterial, NodeMaterialDefines } from \"../nodeMaterial\";\r\n\r\n/**\r\n * Block used to transform a vector (2, 3 or 4) with a matrix. It will generate a Vector4\r\n */\r\nexport class TransformBlock extends NodeMaterialBlock {\r\n /**\r\n * Defines the value to use to complement W value to transform it to a Vector4\r\n */\r\n public complementW = 1;\r\n\r\n /**\r\n * Defines the value to use to complement z value to transform it to a Vector4\r\n */\r\n public complementZ = 0;\r\n\r\n /**\r\n * Creates a new TransformBlock\r\n * @param name defines the block name\r\n */\r\n public constructor(name: string) {\r\n super(name, NodeMaterialBlockTargets.Neutral);\r\n\r\n this.target = NodeMaterialBlockTargets.Vertex;\r\n\r\n this.registerInput(\"vector\", NodeMaterialBlockConnectionPointTypes.AutoDetect);\r\n this.registerInput(\"transform\", NodeMaterialBlockConnectionPointTypes.Matrix);\r\n this.registerOutput(\"output\", NodeMaterialBlockConnectionPointTypes.Vector4);\r\n this.registerOutput(\"xyz\", NodeMaterialBlockConnectionPointTypes.Vector3);\r\n\r\n this._inputs[0].onConnectionObservable.add((other) => {\r\n if (other.ownerBlock.isInput) {\r\n const otherAsInput = other.ownerBlock as InputBlock;\r\n\r\n if (otherAsInput.name === \"normal\" || otherAsInput.name === \"tangent\") {\r\n this.complementW = 0;\r\n }\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Gets the current class name\r\n * @returns the class name\r\n */\r\n public getClassName() {\r\n return \"TransformBlock\";\r\n }\r\n\r\n /**\r\n * Gets the vector input\r\n */\r\n public get vector(): NodeMaterialConnectionPoint {\r\n return this._inputs[0];\r\n }\r\n\r\n /**\r\n * Gets the output component\r\n */\r\n public get output(): NodeMaterialConnectionPoint {\r\n return this._outputs[0];\r\n }\r\n\r\n /**\r\n * Gets the xyz output component\r\n */\r\n public get xyz(): NodeMaterialConnectionPoint {\r\n return this._outputs[1];\r\n }\r\n\r\n /**\r\n * Gets the matrix transform input\r\n */\r\n public get transform(): NodeMaterialConnectionPoint {\r\n return this._inputs[1];\r\n }\r\n\r\n protected _buildBlock(state: NodeMaterialBuildState) {\r\n super._buildBlock(state);\r\n\r\n const vector = this.vector;\r\n const transform = this.transform;\r\n\r\n if (vector.connectedPoint) {\r\n // None uniform scaling case.\r\n if (this.complementW === 0) {\r\n const comments = `//${this.name}`;\r\n state._emitFunctionFromInclude(\"helperFunctions\", comments);\r\n state.sharedData.blocksWithDefines.push(this);\r\n\r\n const transformName = state._getFreeVariableName(`${transform.associatedVariableName}_NUS`);\r\n state.compilationString += `mat3 ${transformName} = mat3(${transform.associatedVariableName});\\n`;\r\n state.compilationString += `#ifdef NONUNIFORMSCALING\\n`;\r\n state.compilationString += `${transformName} = transposeMat3(inverseMat3(${transformName}));\\n`;\r\n state.compilationString += `#endif\\n`;\r\n switch (vector.connectedPoint.type) {\r\n case NodeMaterialBlockConnectionPointTypes.Vector2:\r\n state.compilationString +=\r\n this._declareOutput(this.output, state) +\r\n ` = vec4(${transformName} * vec3(${vector.associatedVariableName}, ${this._writeFloat(this.complementZ)}), ${this._writeFloat(this.complementW)});\\n`;\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Vector3:\r\n case NodeMaterialBlockConnectionPointTypes.Color3:\r\n state.compilationString +=\r\n this._declareOutput(this.output, state) + ` = vec4(${transformName} * ${vector.associatedVariableName}, ${this._writeFloat(this.complementW)});\\n`;\r\n break;\r\n default:\r\n state.compilationString +=\r\n this._declareOutput(this.output, state) + ` = vec4(${transformName} * ${vector.associatedVariableName}.xyz, ${this._writeFloat(this.complementW)});\\n`;\r\n break;\r\n }\r\n } else {\r\n const transformName = transform.associatedVariableName;\r\n switch (vector.connectedPoint.type) {\r\n case NodeMaterialBlockConnectionPointTypes.Vector2:\r\n state.compilationString +=\r\n this._declareOutput(this.output, state) +\r\n ` = ${transformName} * vec4(${vector.associatedVariableName}, ${this._writeFloat(this.complementZ)}, ${this._writeFloat(this.complementW)});\\n`;\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Vector3:\r\n case NodeMaterialBlockConnectionPointTypes.Color3:\r\n state.compilationString +=\r\n this._declareOutput(this.output, state) + ` = ${transformName} * vec4(${vector.associatedVariableName}, ${this._writeFloat(this.complementW)});\\n`;\r\n break;\r\n default:\r\n state.compilationString += this._declareOutput(this.output, state) + ` = ${transformName} * ${vector.associatedVariableName};\\n`;\r\n break;\r\n }\r\n }\r\n\r\n if (this.xyz.hasEndpoints) {\r\n state.compilationString += this._declareOutput(this.xyz, state) + ` = ${this.output.associatedVariableName}.xyz;\\n`;\r\n }\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Update defines for shader compilation\r\n * @param mesh defines the mesh to be rendered\r\n * @param nodeMaterial defines the node material requesting the update\r\n * @param defines defines the material defines to update\r\n */\r\n public prepareDefines(mesh: AbstractMesh, nodeMaterial: NodeMaterial, defines: NodeMaterialDefines) {\r\n // Do nothing\r\n if (mesh.nonUniformScaling) {\r\n defines.setValue(\"NONUNIFORMSCALING\", true);\r\n }\r\n }\r\n\r\n public serialize(): any {\r\n const serializationObject = super.serialize();\r\n\r\n serializationObject.complementZ = this.complementZ;\r\n serializationObject.complementW = this.complementW;\r\n\r\n return serializationObject;\r\n }\r\n\r\n public _deserialize(serializationObject: any, scene: Scene, rootUrl: string) {\r\n super._deserialize(serializationObject, scene, rootUrl);\r\n\r\n this.complementZ = serializationObject.complementZ !== undefined ? serializationObject.complementZ : 0.0;\r\n this.complementW = serializationObject.complementW !== undefined ? serializationObject.complementW : 1.0;\r\n }\r\n\r\n protected _dumpPropertiesCode() {\r\n let codeString = super._dumpPropertiesCode() + `${this._codeVariableName}.complementZ = ${this.complementZ};\\n`;\r\n\r\n codeString += `${this._codeVariableName}.complementW = ${this.complementW};\\n`;\r\n\r\n return codeString;\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.TransformBlock\", TransformBlock);\r\n","import { NodeMaterialBlock } from \"../../nodeMaterialBlock\";\r\nimport { NodeMaterialBlockConnectionPointTypes } from \"../../Enums/nodeMaterialBlockConnectionPointTypes\";\r\nimport type { NodeMaterialBuildState } from \"../../nodeMaterialBuildState\";\r\nimport { NodeMaterialBlockTargets } from \"../../Enums/nodeMaterialBlockTargets\";\r\nimport type { NodeMaterialConnectionPoint } from \"../../nodeMaterialBlockConnectionPoint\";\r\nimport { RegisterClass } from \"../../../../Misc/typeStore\";\r\nimport type { Immutable } from \"../../../../types\";\r\n\r\nimport type { FragmentOutputBlock } from \"../Fragment/fragmentOutputBlock\";\r\n\r\n/**\r\n * Block used to output the vertex position\r\n */\r\nexport class VertexOutputBlock extends NodeMaterialBlock {\r\n /**\r\n * Creates a new VertexOutputBlock\r\n * @param name defines the block name\r\n */\r\n public constructor(name: string) {\r\n super(name, NodeMaterialBlockTargets.Vertex, true);\r\n\r\n this.registerInput(\"vector\", NodeMaterialBlockConnectionPointTypes.Vector4);\r\n }\r\n\r\n /**\r\n * Gets the current class name\r\n * @returns the class name\r\n */\r\n public getClassName() {\r\n return \"VertexOutputBlock\";\r\n }\r\n\r\n /**\r\n * Gets the vector input component\r\n */\r\n public get vector(): NodeMaterialConnectionPoint {\r\n return this._inputs[0];\r\n }\r\n\r\n private _isLogarithmicDepthEnabled(nodeList: Immutable): boolean {\r\n for (const node of nodeList) {\r\n if ((node as FragmentOutputBlock).useLogarithmicDepth) {\r\n return true;\r\n }\r\n }\r\n return false;\r\n }\r\n\r\n protected _buildBlock(state: NodeMaterialBuildState) {\r\n super._buildBlock(state);\r\n\r\n const input = this.vector;\r\n\r\n state.compilationString += `gl_Position = ${input.associatedVariableName};\\n`;\r\n\r\n if (this._isLogarithmicDepthEnabled(state.sharedData.fragmentOutputNodes)) {\r\n state._emitUniformFromString(\"logarithmicDepthConstant\", \"float\");\r\n state._emitVaryingFromString(\"vFragmentDepth\", \"float\");\r\n\r\n state.compilationString += `vFragmentDepth = 1.0 + gl_Position.w;\\n`;\r\n state.compilationString += `gl_Position.z = log2(max(0.000001, vFragmentDepth)) * logarithmicDepthConstant;\\n`;\r\n }\r\n\r\n return this;\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.VertexOutputBlock\", VertexOutputBlock);\r\n","import { NodeMaterialBlock } from \"../../nodeMaterialBlock\";\r\nimport { NodeMaterialBlockConnectionPointTypes } from \"../../Enums/nodeMaterialBlockConnectionPointTypes\";\r\nimport type { NodeMaterialBuildState } from \"../../nodeMaterialBuildState\";\r\nimport { NodeMaterialBlockTargets } from \"../../Enums/nodeMaterialBlockTargets\";\r\nimport type { NodeMaterialConnectionPoint } from \"../../nodeMaterialBlockConnectionPoint\";\r\nimport { RegisterClass } from \"../../../../Misc/typeStore\";\r\nimport type { Scene } from \"../../../../scene\";\r\nimport type { AbstractMesh } from \"../../../../Meshes/abstractMesh\";\r\nimport type { NodeMaterialDefines } from \"../../nodeMaterial\";\r\nimport { editableInPropertyPage, PropertyTypeForEdition } from \"../../../../Decorators/nodeDecorator\";\r\nimport { MaterialHelper } from \"../../../materialHelper\";\r\n\r\nimport type { NodeMaterial } from \"../../nodeMaterial\";\r\nimport type { Effect } from \"../../../effect\";\r\nimport type { Mesh } from \"../../../../Meshes/mesh\";\r\n\r\n/**\r\n * Block used to output the final color\r\n */\r\nexport class FragmentOutputBlock extends NodeMaterialBlock {\r\n private _linearDefineName: string;\r\n private _gammaDefineName: string;\r\n\r\n /**\r\n * Create a new FragmentOutputBlock\r\n * @param name defines the block name\r\n */\r\n public constructor(name: string) {\r\n super(name, NodeMaterialBlockTargets.Fragment, true);\r\n\r\n this.registerInput(\"rgba\", NodeMaterialBlockConnectionPointTypes.Color4, true);\r\n this.registerInput(\"rgb\", NodeMaterialBlockConnectionPointTypes.AutoDetect, true);\r\n this.registerInput(\"a\", NodeMaterialBlockConnectionPointTypes.Float, true);\r\n\r\n this.rgb.addExcludedConnectionPointFromAllowedTypes(\r\n NodeMaterialBlockConnectionPointTypes.Color3 | NodeMaterialBlockConnectionPointTypes.Vector3 | NodeMaterialBlockConnectionPointTypes.Float\r\n );\r\n }\r\n\r\n /** Gets or sets a boolean indicating if content needs to be converted to gamma space */\r\n @editableInPropertyPage(\"Convert to gamma space\", PropertyTypeForEdition.Boolean, \"PROPERTIES\", { notifiers: { update: true } })\r\n public convertToGammaSpace = false;\r\n\r\n /** Gets or sets a boolean indicating if content needs to be converted to linear space */\r\n @editableInPropertyPage(\"Convert to linear space\", PropertyTypeForEdition.Boolean, \"PROPERTIES\", { notifiers: { update: true } })\r\n public convertToLinearSpace = false;\r\n\r\n /** Gets or sets a boolean indicating if logarithmic depth should be used */\r\n @editableInPropertyPage(\"Use logarithmic depth\", PropertyTypeForEdition.Boolean, \"PROPERTIES\")\r\n public useLogarithmicDepth = false;\r\n\r\n /**\r\n * Gets the current class name\r\n * @returns the class name\r\n */\r\n public getClassName() {\r\n return \"FragmentOutputBlock\";\r\n }\r\n\r\n /**\r\n * Initialize the block and prepare the context for build\r\n * @param state defines the state that will be used for the build\r\n */\r\n public initialize(state: NodeMaterialBuildState) {\r\n state._excludeVariableName(\"logarithmicDepthConstant\");\r\n state._excludeVariableName(\"vFragmentDepth\");\r\n }\r\n\r\n /**\r\n * Gets the rgba input component\r\n */\r\n public get rgba(): NodeMaterialConnectionPoint {\r\n return this._inputs[0];\r\n }\r\n\r\n /**\r\n * Gets the rgb input component\r\n */\r\n public get rgb(): NodeMaterialConnectionPoint {\r\n return this._inputs[1];\r\n }\r\n\r\n /**\r\n * Gets the a input component\r\n */\r\n public get a(): NodeMaterialConnectionPoint {\r\n return this._inputs[2];\r\n }\r\n\r\n public prepareDefines(mesh: AbstractMesh, nodeMaterial: NodeMaterial, defines: NodeMaterialDefines) {\r\n defines.setValue(this._linearDefineName, this.convertToLinearSpace, true);\r\n defines.setValue(this._gammaDefineName, this.convertToGammaSpace, true);\r\n }\r\n\r\n public bind(effect: Effect, nodeMaterial: NodeMaterial, mesh?: Mesh) {\r\n if (this.useLogarithmicDepth && mesh) {\r\n MaterialHelper.BindLogDepth(undefined, effect, mesh.getScene());\r\n }\r\n }\r\n\r\n protected _buildBlock(state: NodeMaterialBuildState) {\r\n super._buildBlock(state);\r\n\r\n const rgba = this.rgba;\r\n const rgb = this.rgb;\r\n const a = this.a;\r\n\r\n state.sharedData.hints.needAlphaBlending = rgba.isConnected || a.isConnected;\r\n state.sharedData.blocksWithDefines.push(this);\r\n if (this.useLogarithmicDepth) {\r\n state._emitUniformFromString(\"logarithmicDepthConstant\", \"float\");\r\n state._emitVaryingFromString(\"vFragmentDepth\", \"float\");\r\n state.sharedData.bindableBlocks.push(this);\r\n }\r\n this._linearDefineName = state._getFreeDefineName(\"CONVERTTOLINEAR\");\r\n this._gammaDefineName = state._getFreeDefineName(\"CONVERTTOGAMMA\");\r\n\r\n const comments = `//${this.name}`;\r\n state._emitFunctionFromInclude(\"helperFunctions\", comments);\r\n\r\n if (rgba.connectedPoint) {\r\n if (a.isConnected) {\r\n state.compilationString += `gl_FragColor = vec4(${rgba.associatedVariableName}.rgb, ${a.associatedVariableName});\\n`;\r\n } else {\r\n state.compilationString += `gl_FragColor = ${rgba.associatedVariableName};\\n`;\r\n }\r\n } else if (rgb.connectedPoint) {\r\n let aValue = \"1.0\";\r\n\r\n if (a.connectedPoint) {\r\n aValue = a.associatedVariableName;\r\n }\r\n\r\n if (rgb.connectedPoint.type === NodeMaterialBlockConnectionPointTypes.Float) {\r\n state.compilationString += `gl_FragColor = vec4(${rgb.associatedVariableName}, ${rgb.associatedVariableName}, ${rgb.associatedVariableName}, ${aValue});\\n`;\r\n } else {\r\n state.compilationString += `gl_FragColor = vec4(${rgb.associatedVariableName}, ${aValue});\\n`;\r\n }\r\n } else {\r\n state.sharedData.checks.notConnectedNonOptionalInputs.push(rgba);\r\n }\r\n\r\n state.compilationString += `#ifdef ${this._linearDefineName}\\n`;\r\n state.compilationString += `gl_FragColor = toLinearSpace(gl_FragColor);\\n`;\r\n state.compilationString += `#endif\\n`;\r\n\r\n state.compilationString += `#ifdef ${this._gammaDefineName}\\n`;\r\n state.compilationString += `gl_FragColor = toGammaSpace(gl_FragColor);\\n`;\r\n state.compilationString += `#endif\\n`;\r\n\r\n if (this.useLogarithmicDepth) {\r\n state.compilationString += `gl_FragDepthEXT = log2(vFragmentDepth) * logarithmicDepthConstant * 0.5;\\n`;\r\n }\r\n\r\n return this;\r\n }\r\n\r\n protected _dumpPropertiesCode() {\r\n let codeString = super._dumpPropertiesCode();\r\n codeString += `${this._codeVariableName}.convertToGammaSpace = ${this.convertToGammaSpace};\\n`;\r\n codeString += `${this._codeVariableName}.convertToLinearSpace = ${this.convertToLinearSpace};\\n`;\r\n codeString += `${this._codeVariableName}.useLogarithmicDepth = ${this.useLogarithmicDepth};\\n`;\r\n\r\n return codeString;\r\n }\r\n\r\n public serialize(): any {\r\n const serializationObject = super.serialize();\r\n\r\n serializationObject.convertToGammaSpace = this.convertToGammaSpace;\r\n serializationObject.convertToLinearSpace = this.convertToLinearSpace;\r\n serializationObject.useLogarithmicDepth = this.useLogarithmicDepth;\r\n\r\n return serializationObject;\r\n }\r\n\r\n public _deserialize(serializationObject: any, scene: Scene, rootUrl: string) {\r\n super._deserialize(serializationObject, scene, rootUrl);\r\n\r\n this.convertToGammaSpace = serializationObject.convertToGammaSpace;\r\n this.convertToLinearSpace = serializationObject.convertToLinearSpace;\r\n this.useLogarithmicDepth = serializationObject.useLogarithmicDepth ?? false;\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.FragmentOutputBlock\", FragmentOutputBlock);\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport { NodeMaterialBlock } from \"../../nodeMaterialBlock\";\r\nimport { NodeMaterialBlockConnectionPointTypes } from \"../../Enums/nodeMaterialBlockConnectionPointTypes\";\r\nimport { NodeMaterialBlockConnectionPointMode } from \"../../Enums/nodeMaterialBlockConnectionPointMode\";\r\nimport { NodeMaterialSystemValues } from \"../../Enums/nodeMaterialSystemValues\";\r\nimport type { Nullable } from \"../../../../types\";\r\nimport type { Effect } from \"../../../../Materials/effect\";\r\nimport { Matrix, Vector2, Vector3, Vector4 } from \"../../../../Maths/math.vector\";\r\nimport type { Scene } from \"../../../../scene\";\r\nimport type { NodeMaterialConnectionPoint } from \"../../nodeMaterialBlockConnectionPoint\";\r\nimport type { NodeMaterialBuildState } from \"../../nodeMaterialBuildState\";\r\nimport { NodeMaterialBlockTargets } from \"../../Enums/nodeMaterialBlockTargets\";\r\nimport { GetClass, RegisterClass } from \"../../../../Misc/typeStore\";\r\nimport { Color3, Color4, TmpColors } from \"../../../../Maths/math\";\r\nimport { AnimatedInputBlockTypes } from \"./animatedInputBlockTypes\";\r\nimport { Observable } from \"../../../../Misc/observable\";\r\nimport type { NodeMaterial } from \"../../nodeMaterial\";\r\nimport { PrecisionDate } from \"core/Misc/precisionDate\";\r\n\r\nconst remapAttributeName: { [name: string]: string } = {\r\n position2d: \"position\",\r\n particle_uv: \"vUV\",\r\n particle_color: \"vColor\",\r\n particle_texturemask: \"textureMask\",\r\n particle_positionw: \"vPositionW\",\r\n};\r\n\r\nconst attributeInFragmentOnly: { [name: string]: boolean } = {\r\n particle_uv: true,\r\n particle_color: true,\r\n particle_texturemask: true,\r\n particle_positionw: true,\r\n};\r\n\r\nconst attributeAsUniform: { [name: string]: boolean } = {\r\n particle_texturemask: true,\r\n};\r\n\r\n/**\r\n * Block used to expose an input value\r\n */\r\nexport class InputBlock extends NodeMaterialBlock {\r\n private _mode = NodeMaterialBlockConnectionPointMode.Undefined;\r\n private _associatedVariableName: string;\r\n private _storedValue: any;\r\n private _valueCallback: () => any;\r\n private _type: NodeMaterialBlockConnectionPointTypes;\r\n private _animationType = AnimatedInputBlockTypes.None;\r\n\r\n /** Gets or set a value used to limit the range of float values */\r\n public min: number = 0;\r\n\r\n /** Gets or set a value used to limit the range of float values */\r\n public max: number = 0;\r\n\r\n /** Gets or set a value indicating that this input can only get 0 and 1 values */\r\n public isBoolean: boolean = false;\r\n\r\n /** Gets or sets a value used by the Node Material editor to determine how to configure the current value if it is a matrix */\r\n public matrixMode: number = 0;\r\n\r\n /** @internal */\r\n public _systemValue: Nullable = null;\r\n\r\n /** Gets or sets a boolean indicating that the value of this input will not change after a build */\r\n public isConstant = false;\r\n\r\n /** Gets or sets the group to use to display this block in the Inspector */\r\n public groupInInspector = \"\";\r\n\r\n /** Gets an observable raised when the value is changed */\r\n public onValueChangedObservable = new Observable();\r\n\r\n /** Gets or sets a boolean indicating if content needs to be converted to gamma space (for color3/4 only) */\r\n public convertToGammaSpace = false;\r\n\r\n /** Gets or sets a boolean indicating if content needs to be converted to linear space (for color3/4 only) */\r\n public convertToLinearSpace = false;\r\n\r\n /**\r\n * Gets or sets the connection point type (default is float)\r\n */\r\n public get type(): NodeMaterialBlockConnectionPointTypes {\r\n if (this._type === NodeMaterialBlockConnectionPointTypes.AutoDetect) {\r\n if (this.isUniform && this.value != null) {\r\n if (!isNaN(this.value)) {\r\n this._type = NodeMaterialBlockConnectionPointTypes.Float;\r\n return this._type;\r\n }\r\n\r\n switch (this.value.getClassName()) {\r\n case \"Vector2\":\r\n this._type = NodeMaterialBlockConnectionPointTypes.Vector2;\r\n return this._type;\r\n case \"Vector3\":\r\n this._type = NodeMaterialBlockConnectionPointTypes.Vector3;\r\n return this._type;\r\n case \"Vector4\":\r\n this._type = NodeMaterialBlockConnectionPointTypes.Vector4;\r\n return this._type;\r\n case \"Color3\":\r\n this._type = NodeMaterialBlockConnectionPointTypes.Color3;\r\n return this._type;\r\n case \"Color4\":\r\n this._type = NodeMaterialBlockConnectionPointTypes.Color4;\r\n return this._type;\r\n case \"Matrix\":\r\n this._type = NodeMaterialBlockConnectionPointTypes.Matrix;\r\n return this._type;\r\n }\r\n }\r\n\r\n if (this.isAttribute) {\r\n switch (this.name) {\r\n case \"position\":\r\n case \"normal\":\r\n case \"particle_positionw\":\r\n this._type = NodeMaterialBlockConnectionPointTypes.Vector3;\r\n return this._type;\r\n case \"uv\":\r\n case \"uv2\":\r\n case \"uv3\":\r\n case \"uv4\":\r\n case \"uv5\":\r\n case \"uv6\":\r\n case \"position2d\":\r\n case \"particle_uv\":\r\n this._type = NodeMaterialBlockConnectionPointTypes.Vector2;\r\n return this._type;\r\n case \"matricesIndices\":\r\n case \"matricesWeights\":\r\n case \"matricesIndicesExtra\":\r\n case \"matricesWeightsExtra\":\r\n case \"world0\":\r\n case \"world1\":\r\n case \"world2\":\r\n case \"world3\":\r\n case \"tangent\":\r\n this._type = NodeMaterialBlockConnectionPointTypes.Vector4;\r\n return this._type;\r\n case \"color\":\r\n case \"instanceColor\":\r\n case \"particle_color\":\r\n case \"particle_texturemask\":\r\n this._type = NodeMaterialBlockConnectionPointTypes.Color4;\r\n return this._type;\r\n }\r\n }\r\n\r\n if (this.isSystemValue) {\r\n switch (this._systemValue) {\r\n case NodeMaterialSystemValues.World:\r\n case NodeMaterialSystemValues.WorldView:\r\n case NodeMaterialSystemValues.WorldViewProjection:\r\n case NodeMaterialSystemValues.View:\r\n case NodeMaterialSystemValues.ViewProjection:\r\n case NodeMaterialSystemValues.Projection:\r\n this._type = NodeMaterialBlockConnectionPointTypes.Matrix;\r\n return this._type;\r\n case NodeMaterialSystemValues.CameraPosition:\r\n this._type = NodeMaterialBlockConnectionPointTypes.Vector3;\r\n return this._type;\r\n case NodeMaterialSystemValues.FogColor:\r\n this._type = NodeMaterialBlockConnectionPointTypes.Color3;\r\n return this._type;\r\n case NodeMaterialSystemValues.DeltaTime:\r\n case NodeMaterialSystemValues.MaterialAlpha:\r\n this._type = NodeMaterialBlockConnectionPointTypes.Float;\r\n return this._type;\r\n case NodeMaterialSystemValues.CameraParameters:\r\n this._type = NodeMaterialBlockConnectionPointTypes.Vector4;\r\n return this._type;\r\n }\r\n }\r\n }\r\n\r\n return this._type;\r\n }\r\n\r\n /**\r\n * Creates a new InputBlock\r\n * @param name defines the block name\r\n * @param target defines the target of that block (Vertex by default)\r\n * @param type defines the type of the input (can be set to NodeMaterialBlockConnectionPointTypes.AutoDetect)\r\n */\r\n public constructor(name: string, target = NodeMaterialBlockTargets.Vertex, type: NodeMaterialBlockConnectionPointTypes = NodeMaterialBlockConnectionPointTypes.AutoDetect) {\r\n super(name, target, false);\r\n\r\n this._type = type;\r\n\r\n this.setDefaultValue();\r\n\r\n this.registerOutput(\"output\", type);\r\n }\r\n\r\n /**\r\n * Validates if a name is a reserve word.\r\n * @param newName the new name to be given to the node.\r\n * @returns false if the name is a reserve word, else true.\r\n */\r\n public validateBlockName(newName: string) {\r\n if (!this.isAttribute) {\r\n return super.validateBlockName(newName);\r\n }\r\n return true;\r\n }\r\n\r\n /**\r\n * Gets the output component\r\n */\r\n public get output(): NodeMaterialConnectionPoint {\r\n return this._outputs[0];\r\n }\r\n\r\n /**\r\n * Set the source of this connection point to a vertex attribute\r\n * @param attributeName defines the attribute name (position, uv, normal, etc...). If not specified it will take the connection point name\r\n * @returns the current connection point\r\n */\r\n public setAsAttribute(attributeName?: string): InputBlock {\r\n this._mode = NodeMaterialBlockConnectionPointMode.Attribute;\r\n if (attributeName) {\r\n this.name = attributeName;\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Set the source of this connection point to a system value\r\n * @param value define the system value to use (world, view, etc...) or null to switch to manual value\r\n * @returns the current connection point\r\n */\r\n public setAsSystemValue(value: Nullable): InputBlock {\r\n this.systemValue = value;\r\n return this;\r\n }\r\n\r\n /**\r\n * Gets or sets the value of that point.\r\n * Please note that this value will be ignored if valueCallback is defined\r\n */\r\n public get value(): any {\r\n return this._storedValue;\r\n }\r\n\r\n public set value(value: any) {\r\n if (this.type === NodeMaterialBlockConnectionPointTypes.Float) {\r\n if (this.isBoolean) {\r\n value = value ? 1 : 0;\r\n } else if (this.min !== this.max) {\r\n value = Math.max(this.min, value);\r\n value = Math.min(this.max, value);\r\n }\r\n }\r\n\r\n this._storedValue = value;\r\n this._mode = NodeMaterialBlockConnectionPointMode.Uniform;\r\n\r\n this.onValueChangedObservable.notifyObservers(this);\r\n }\r\n\r\n /**\r\n * Gets or sets a callback used to get the value of that point.\r\n * Please note that setting this value will force the connection point to ignore the value property\r\n */\r\n public get valueCallback(): () => any {\r\n return this._valueCallback;\r\n }\r\n\r\n public set valueCallback(value: () => any) {\r\n this._valueCallback = value;\r\n this._mode = NodeMaterialBlockConnectionPointMode.Uniform;\r\n }\r\n\r\n /**\r\n * Gets or sets the associated variable name in the shader\r\n */\r\n public get associatedVariableName(): string {\r\n return this._associatedVariableName;\r\n }\r\n\r\n public set associatedVariableName(value: string) {\r\n this._associatedVariableName = value;\r\n }\r\n\r\n /** Gets or sets the type of animation applied to the input */\r\n public get animationType() {\r\n return this._animationType;\r\n }\r\n\r\n public set animationType(value: AnimatedInputBlockTypes) {\r\n this._animationType = value;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating that this connection point not defined yet\r\n */\r\n public get isUndefined(): boolean {\r\n return this._mode === NodeMaterialBlockConnectionPointMode.Undefined;\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating that this connection point is coming from an uniform.\r\n * In this case the connection point name must be the name of the uniform to use.\r\n * Can only be set on inputs\r\n */\r\n public get isUniform(): boolean {\r\n return this._mode === NodeMaterialBlockConnectionPointMode.Uniform;\r\n }\r\n\r\n public set isUniform(value: boolean) {\r\n this._mode = value ? NodeMaterialBlockConnectionPointMode.Uniform : NodeMaterialBlockConnectionPointMode.Undefined;\r\n this.associatedVariableName = \"\";\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating that this connection point is coming from an attribute.\r\n * In this case the connection point name must be the name of the attribute to use\r\n * Can only be set on inputs\r\n */\r\n public get isAttribute(): boolean {\r\n return this._mode === NodeMaterialBlockConnectionPointMode.Attribute;\r\n }\r\n\r\n public set isAttribute(value: boolean) {\r\n this._mode = value ? NodeMaterialBlockConnectionPointMode.Attribute : NodeMaterialBlockConnectionPointMode.Undefined;\r\n this.associatedVariableName = \"\";\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating that this connection point is generating a varying variable.\r\n * Can only be set on exit points\r\n */\r\n public get isVarying(): boolean {\r\n return this._mode === NodeMaterialBlockConnectionPointMode.Varying;\r\n }\r\n\r\n public set isVarying(value: boolean) {\r\n this._mode = value ? NodeMaterialBlockConnectionPointMode.Varying : NodeMaterialBlockConnectionPointMode.Undefined;\r\n this.associatedVariableName = \"\";\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating that the current connection point is a system value\r\n */\r\n public get isSystemValue(): boolean {\r\n return this._systemValue != null;\r\n }\r\n\r\n /**\r\n * Gets or sets the current well known value or null if not defined as a system value\r\n */\r\n public get systemValue(): Nullable {\r\n return this._systemValue;\r\n }\r\n\r\n public set systemValue(value: Nullable) {\r\n this._mode = NodeMaterialBlockConnectionPointMode.Uniform;\r\n this.associatedVariableName = \"\";\r\n this._systemValue = value;\r\n }\r\n\r\n /**\r\n * Gets the current class name\r\n * @returns the class name\r\n */\r\n public getClassName() {\r\n return \"InputBlock\";\r\n }\r\n\r\n /**\r\n * Animate the input if animationType !== None\r\n * @param scene defines the rendering scene\r\n */\r\n public animate(scene: Scene) {\r\n switch (this._animationType) {\r\n case AnimatedInputBlockTypes.Time: {\r\n if (this.type === NodeMaterialBlockConnectionPointTypes.Float) {\r\n this.value += scene.getAnimationRatio() * 0.01;\r\n }\r\n break;\r\n }\r\n case AnimatedInputBlockTypes.RealTime: {\r\n if (this.type === NodeMaterialBlockConnectionPointTypes.Float) {\r\n this.value = (PrecisionDate.Now - scene.getEngine().startTime) / 1000;\r\n }\r\n break;\r\n }\r\n }\r\n }\r\n\r\n private _emitDefine(define: string): string {\r\n if (define[0] === \"!\") {\r\n return `#ifndef ${define.substring(1)}\\n`;\r\n }\r\n\r\n return `#ifdef ${define}\\n`;\r\n }\r\n\r\n public initialize() {\r\n this.associatedVariableName = \"\";\r\n }\r\n\r\n /**\r\n * Set the input block to its default value (based on its type)\r\n */\r\n public setDefaultValue() {\r\n switch (this.type) {\r\n case NodeMaterialBlockConnectionPointTypes.Float:\r\n this.value = 0;\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Vector2:\r\n this.value = Vector2.Zero();\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Vector3:\r\n this.value = Vector3.Zero();\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Vector4:\r\n this.value = Vector4.Zero();\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Color3:\r\n this.value = Color3.White();\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Color4:\r\n this.value = new Color4(1, 1, 1, 1);\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Matrix:\r\n this.value = Matrix.Identity();\r\n break;\r\n }\r\n }\r\n\r\n private _emitConstant(state: NodeMaterialBuildState) {\r\n switch (this.type) {\r\n case NodeMaterialBlockConnectionPointTypes.Float:\r\n return `${state._emitFloat(this.value)}`;\r\n case NodeMaterialBlockConnectionPointTypes.Vector2:\r\n return `vec2(${this.value.x}, ${this.value.y})`;\r\n case NodeMaterialBlockConnectionPointTypes.Vector3:\r\n return `vec3(${this.value.x}, ${this.value.y}, ${this.value.z})`;\r\n case NodeMaterialBlockConnectionPointTypes.Vector4:\r\n return `vec4(${this.value.x}, ${this.value.y}, ${this.value.z}, ${this.value.w})`;\r\n case NodeMaterialBlockConnectionPointTypes.Color3:\r\n TmpColors.Color3[0].set(this.value.r, this.value.g, this.value.b);\r\n if (this.convertToGammaSpace) {\r\n TmpColors.Color3[0].toGammaSpaceToRef(TmpColors.Color3[0], state.sharedData.scene.getEngine().useExactSrgbConversions);\r\n }\r\n if (this.convertToLinearSpace) {\r\n TmpColors.Color3[0].toLinearSpaceToRef(TmpColors.Color3[0], state.sharedData.scene.getEngine().useExactSrgbConversions);\r\n }\r\n return `vec3(${TmpColors.Color3[0].r}, ${TmpColors.Color3[0].g}, ${TmpColors.Color3[0].b})`;\r\n case NodeMaterialBlockConnectionPointTypes.Color4:\r\n TmpColors.Color4[0].set(this.value.r, this.value.g, this.value.b, this.value.a);\r\n if (this.convertToGammaSpace) {\r\n TmpColors.Color4[0].toGammaSpaceToRef(TmpColors.Color4[0], state.sharedData.scene.getEngine().useExactSrgbConversions);\r\n }\r\n if (this.convertToLinearSpace) {\r\n TmpColors.Color4[0].toLinearSpaceToRef(TmpColors.Color4[0], state.sharedData.scene.getEngine().useExactSrgbConversions);\r\n }\r\n return `vec4(${TmpColors.Color4[0].r}, ${TmpColors.Color4[0].g}, ${TmpColors.Color4[0].b}, ${TmpColors.Color4[0].a})`;\r\n }\r\n\r\n return \"\";\r\n }\r\n\r\n /** @internal */\r\n public get _noContextSwitch(): boolean {\r\n return attributeInFragmentOnly[this.name];\r\n }\r\n\r\n private _emit(state: NodeMaterialBuildState, define?: string) {\r\n // Uniforms\r\n if (this.isUniform) {\r\n if (!this.associatedVariableName) {\r\n this.associatedVariableName = state._getFreeVariableName(\"u_\" + this.name);\r\n }\r\n\r\n if (this.isConstant) {\r\n if (state.constants.indexOf(this.associatedVariableName) !== -1) {\r\n return;\r\n }\r\n state.constants.push(this.associatedVariableName);\r\n state._constantDeclaration += this._declareOutput(this.output, state) + ` = ${this._emitConstant(state)};\\n`;\r\n return;\r\n }\r\n\r\n if (state.uniforms.indexOf(this.associatedVariableName) !== -1) {\r\n return;\r\n }\r\n\r\n state.uniforms.push(this.associatedVariableName);\r\n if (define) {\r\n state._uniformDeclaration += this._emitDefine(define);\r\n }\r\n state._uniformDeclaration += `uniform ${state._getGLType(this.type)} ${this.associatedVariableName};\\n`;\r\n if (define) {\r\n state._uniformDeclaration += `#endif\\n`;\r\n }\r\n\r\n // well known\r\n const hints = state.sharedData.hints;\r\n if (this._systemValue !== null && this._systemValue !== undefined) {\r\n switch (this._systemValue) {\r\n case NodeMaterialSystemValues.WorldView:\r\n hints.needWorldViewMatrix = true;\r\n break;\r\n case NodeMaterialSystemValues.WorldViewProjection:\r\n hints.needWorldViewProjectionMatrix = true;\r\n break;\r\n }\r\n } else {\r\n if (this._animationType !== AnimatedInputBlockTypes.None) {\r\n state.sharedData.animatedInputs.push(this);\r\n }\r\n }\r\n\r\n return;\r\n }\r\n\r\n // Attribute\r\n if (this.isAttribute) {\r\n this.associatedVariableName = remapAttributeName[this.name] ?? this.name;\r\n\r\n if (this.target === NodeMaterialBlockTargets.Vertex && state._vertexState) {\r\n // Attribute for fragment need to be carried over by varyings\r\n if (attributeInFragmentOnly[this.name]) {\r\n if (attributeAsUniform[this.name]) {\r\n state._emitUniformFromString(this.associatedVariableName, state._getGLType(this.type), define);\r\n } else {\r\n state._emitVaryingFromString(this.associatedVariableName, state._getGLType(this.type), define);\r\n }\r\n } else {\r\n this._emit(state._vertexState, define);\r\n }\r\n return;\r\n }\r\n\r\n if (state.attributes.indexOf(this.associatedVariableName) !== -1) {\r\n return;\r\n }\r\n\r\n state.attributes.push(this.associatedVariableName);\r\n\r\n if (attributeInFragmentOnly[this.name]) {\r\n if (attributeAsUniform[this.name]) {\r\n state._emitUniformFromString(this.associatedVariableName, state._getGLType(this.type), define);\r\n } else {\r\n state._emitVaryingFromString(this.associatedVariableName, state._getGLType(this.type), define);\r\n }\r\n } else {\r\n if (define) {\r\n state._attributeDeclaration += this._emitDefine(define);\r\n }\r\n state._attributeDeclaration += `attribute ${state._getGLType(this.type)} ${this.associatedVariableName};\\n`;\r\n if (define) {\r\n state._attributeDeclaration += `#endif\\n`;\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _transmitWorld(effect: Effect, world: Matrix, worldView: Matrix, worldViewProjection: Matrix) {\r\n if (!this._systemValue) {\r\n return;\r\n }\r\n\r\n const variableName = this.associatedVariableName;\r\n switch (this._systemValue) {\r\n case NodeMaterialSystemValues.World:\r\n effect.setMatrix(variableName, world);\r\n break;\r\n case NodeMaterialSystemValues.WorldView:\r\n effect.setMatrix(variableName, worldView);\r\n break;\r\n case NodeMaterialSystemValues.WorldViewProjection:\r\n effect.setMatrix(variableName, worldViewProjection);\r\n break;\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _transmit(effect: Effect, scene: Scene, material: NodeMaterial) {\r\n if (this.isAttribute) {\r\n return;\r\n }\r\n\r\n const variableName = this.associatedVariableName;\r\n if (this._systemValue) {\r\n switch (this._systemValue) {\r\n case NodeMaterialSystemValues.World:\r\n case NodeMaterialSystemValues.WorldView:\r\n case NodeMaterialSystemValues.WorldViewProjection:\r\n return;\r\n case NodeMaterialSystemValues.View:\r\n effect.setMatrix(variableName, scene.getViewMatrix());\r\n break;\r\n case NodeMaterialSystemValues.Projection:\r\n effect.setMatrix(variableName, scene.getProjectionMatrix());\r\n break;\r\n case NodeMaterialSystemValues.ViewProjection:\r\n effect.setMatrix(variableName, scene.getTransformMatrix());\r\n break;\r\n case NodeMaterialSystemValues.CameraPosition:\r\n scene.bindEyePosition(effect, variableName, true);\r\n break;\r\n case NodeMaterialSystemValues.FogColor:\r\n effect.setColor3(variableName, scene.fogColor);\r\n break;\r\n case NodeMaterialSystemValues.DeltaTime:\r\n effect.setFloat(variableName, scene.deltaTime / 1000.0);\r\n break;\r\n case NodeMaterialSystemValues.CameraParameters:\r\n if (scene.activeCamera) {\r\n effect.setFloat4(\r\n variableName,\r\n scene.getEngine().hasOriginBottomLeft ? -1 : 1,\r\n scene.activeCamera.minZ,\r\n scene.activeCamera.maxZ,\r\n 1 / scene.activeCamera.maxZ\r\n );\r\n }\r\n break;\r\n case NodeMaterialSystemValues.MaterialAlpha:\r\n effect.setFloat(variableName, material.alpha);\r\n break;\r\n }\r\n return;\r\n }\r\n\r\n const value = this._valueCallback ? this._valueCallback() : this._storedValue;\r\n\r\n if (value === null) {\r\n return;\r\n }\r\n\r\n switch (this.type) {\r\n case NodeMaterialBlockConnectionPointTypes.Float:\r\n effect.setFloat(variableName, value);\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Int:\r\n effect.setInt(variableName, value);\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Color3:\r\n TmpColors.Color3[0].set(this.value.r, this.value.g, this.value.b);\r\n if (this.convertToGammaSpace) {\r\n TmpColors.Color3[0].toGammaSpaceToRef(TmpColors.Color3[0], scene.getEngine().useExactSrgbConversions);\r\n }\r\n if (this.convertToLinearSpace) {\r\n TmpColors.Color3[0].toLinearSpaceToRef(TmpColors.Color3[0], scene.getEngine().useExactSrgbConversions);\r\n }\r\n effect.setColor3(variableName, TmpColors.Color3[0]);\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Color4:\r\n TmpColors.Color4[0].set(this.value.r, this.value.g, this.value.b, this.value.a);\r\n if (this.convertToGammaSpace) {\r\n TmpColors.Color4[0].toGammaSpaceToRef(TmpColors.Color4[0], scene.getEngine().useExactSrgbConversions);\r\n }\r\n if (this.convertToLinearSpace) {\r\n TmpColors.Color4[0].toLinearSpaceToRef(TmpColors.Color4[0], scene.getEngine().useExactSrgbConversions);\r\n }\r\n effect.setDirectColor4(variableName, TmpColors.Color4[0]);\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Vector2:\r\n effect.setVector2(variableName, value);\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Vector3:\r\n effect.setVector3(variableName, value);\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Vector4:\r\n effect.setVector4(variableName, value);\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Matrix:\r\n effect.setMatrix(variableName, value);\r\n break;\r\n }\r\n }\r\n\r\n protected _buildBlock(state: NodeMaterialBuildState) {\r\n super._buildBlock(state);\r\n\r\n if (this.isUniform || this.isSystemValue) {\r\n state.sharedData.inputBlocks.push(this);\r\n }\r\n\r\n this._emit(state);\r\n }\r\n\r\n protected _dumpPropertiesCode() {\r\n const variableName = this._codeVariableName;\r\n\r\n if (this.isAttribute) {\r\n return super._dumpPropertiesCode() + `${variableName}.setAsAttribute(\"${this.name}\");\\n`;\r\n }\r\n if (this.isSystemValue) {\r\n return super._dumpPropertiesCode() + `${variableName}.setAsSystemValue(BABYLON.NodeMaterialSystemValues.${NodeMaterialSystemValues[this._systemValue!]});\\n`;\r\n }\r\n if (this.isUniform) {\r\n const codes: string[] = [];\r\n\r\n let valueString = \"\";\r\n\r\n switch (this.type) {\r\n case NodeMaterialBlockConnectionPointTypes.Float:\r\n valueString = `${this.value}`;\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Vector2:\r\n valueString = `new BABYLON.Vector2(${this.value.x}, ${this.value.y})`;\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Vector3:\r\n valueString = `new BABYLON.Vector3(${this.value.x}, ${this.value.y}, ${this.value.z})`;\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Vector4:\r\n valueString = `new BABYLON.Vector4(${this.value.x}, ${this.value.y}, ${this.value.z}, ${this.value.w})`;\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Color3:\r\n valueString = `new BABYLON.Color3(${this.value.r}, ${this.value.g}, ${this.value.b})`;\r\n if (this.convertToGammaSpace) {\r\n valueString += \".toGammaSpace()\";\r\n }\r\n if (this.convertToLinearSpace) {\r\n valueString += \".toLinearSpace()\";\r\n }\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Color4:\r\n valueString = `new BABYLON.Color4(${this.value.r}, ${this.value.g}, ${this.value.b}, ${this.value.a})`;\r\n if (this.convertToGammaSpace) {\r\n valueString += \".toGammaSpace()\";\r\n }\r\n if (this.convertToLinearSpace) {\r\n valueString += \".toLinearSpace()\";\r\n }\r\n break;\r\n case NodeMaterialBlockConnectionPointTypes.Matrix:\r\n valueString = `BABYLON.Matrix.FromArray([${(this.value as Matrix).m}])`;\r\n break;\r\n }\r\n\r\n // Common Property \"Value\"\r\n codes.push(`${variableName}.value = ${valueString}`);\r\n\r\n // Float-Value-Specific Properties\r\n if (this.type === NodeMaterialBlockConnectionPointTypes.Float) {\r\n codes.push(\r\n `${variableName}.min = ${this.min}`,\r\n `${variableName}.max = ${this.max}`,\r\n `${variableName}.isBoolean = ${this.isBoolean}`,\r\n `${variableName}.matrixMode = ${this.matrixMode}`,\r\n `${variableName}.animationType = BABYLON.AnimatedInputBlockTypes.${AnimatedInputBlockTypes[this.animationType]}`\r\n );\r\n }\r\n\r\n // Common Property \"Type\"\r\n codes.push(`${variableName}.isConstant = ${this.isConstant}`);\r\n\r\n codes.push(\"\");\r\n\r\n return super._dumpPropertiesCode() + codes.join(\";\\n\");\r\n }\r\n return super._dumpPropertiesCode();\r\n }\r\n\r\n public dispose() {\r\n this.onValueChangedObservable.clear();\r\n\r\n super.dispose();\r\n }\r\n\r\n public serialize(): any {\r\n const serializationObject = super.serialize();\r\n\r\n serializationObject.type = this.type;\r\n serializationObject.mode = this._mode;\r\n serializationObject.systemValue = this._systemValue;\r\n serializationObject.animationType = this._animationType;\r\n serializationObject.min = this.min;\r\n serializationObject.max = this.max;\r\n serializationObject.isBoolean = this.isBoolean;\r\n serializationObject.matrixMode = this.matrixMode;\r\n serializationObject.isConstant = this.isConstant;\r\n serializationObject.groupInInspector = this.groupInInspector;\r\n serializationObject.convertToGammaSpace = this.convertToGammaSpace;\r\n serializationObject.convertToLinearSpace = this.convertToLinearSpace;\r\n\r\n if (this._storedValue != null && this._mode === NodeMaterialBlockConnectionPointMode.Uniform) {\r\n if (this._storedValue.asArray) {\r\n serializationObject.valueType = \"BABYLON.\" + this._storedValue.getClassName();\r\n serializationObject.value = this._storedValue.asArray();\r\n } else {\r\n serializationObject.valueType = \"number\";\r\n serializationObject.value = this._storedValue;\r\n }\r\n }\r\n\r\n return serializationObject;\r\n }\r\n\r\n public _deserialize(serializationObject: any, scene: Scene, rootUrl: string) {\r\n this._mode = serializationObject.mode;\r\n super._deserialize(serializationObject, scene, rootUrl);\r\n\r\n this._type = serializationObject.type;\r\n\r\n this._systemValue = serializationObject.systemValue || serializationObject.wellKnownValue;\r\n this._animationType = serializationObject.animationType;\r\n this.min = serializationObject.min || 0;\r\n this.max = serializationObject.max || 0;\r\n this.isBoolean = !!serializationObject.isBoolean;\r\n this.matrixMode = serializationObject.matrixMode || 0;\r\n this.isConstant = !!serializationObject.isConstant;\r\n this.groupInInspector = serializationObject.groupInInspector || \"\";\r\n this.convertToGammaSpace = !!serializationObject.convertToGammaSpace;\r\n this.convertToLinearSpace = !!serializationObject.convertToLinearSpace;\r\n\r\n // Tangents back compat\r\n if (\r\n serializationObject.name === \"tangent\" &&\r\n serializationObject.mode === NodeMaterialBlockConnectionPointMode.Attribute &&\r\n serializationObject.type === NodeMaterialBlockConnectionPointTypes.Vector3\r\n ) {\r\n this._type = NodeMaterialBlockConnectionPointTypes.Vector4;\r\n }\r\n\r\n if (!serializationObject.valueType) {\r\n return;\r\n }\r\n\r\n if (serializationObject.valueType === \"number\") {\r\n this._storedValue = serializationObject.value;\r\n } else {\r\n const valueType = GetClass(serializationObject.valueType);\r\n\r\n if (valueType) {\r\n this._storedValue = valueType.FromArray(serializationObject.value);\r\n }\r\n }\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.InputBlock\", InputBlock);\r\n","import { NodeMaterialBlock } from \"../../nodeMaterialBlock\";\r\nimport { NodeMaterialBlockConnectionPointTypes } from \"../../Enums/nodeMaterialBlockConnectionPointTypes\";\r\nimport type { NodeMaterialBuildState } from \"../../nodeMaterialBuildState\";\r\nimport { NodeMaterialBlockTargets } from \"../../Enums/nodeMaterialBlockTargets\";\r\nimport type { NodeMaterialConnectionPoint } from \"../../nodeMaterialBlockConnectionPoint\";\r\nimport type { AbstractMesh } from \"../../../../Meshes/abstractMesh\";\r\nimport type { NodeMaterialDefines } from \"../../nodeMaterial\";\r\nimport type { BaseTexture } from \"../../../Textures/baseTexture\";\r\nimport type { Nullable } from \"../../../../types\";\r\nimport { RegisterClass } from \"../../../../Misc/typeStore\";\r\nimport { Texture } from \"../../../Textures/texture\";\r\nimport type { Scene } from \"../../../../scene\";\r\nimport type { InputBlock } from \"../Input/inputBlock\";\r\n\r\nimport type { NodeMaterial } from \"../../nodeMaterial\";\r\n\r\n/**\r\n * Base block used as input for post process\r\n */\r\nexport class CurrentScreenBlock extends NodeMaterialBlock {\r\n private _samplerName = \"textureSampler\";\r\n private _linearDefineName: string;\r\n private _gammaDefineName: string;\r\n private _mainUVName: string;\r\n private _tempTextureRead: string;\r\n\r\n /**\r\n * Gets or sets the texture associated with the node\r\n */\r\n public texture: Nullable;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if content needs to be converted to gamma space\r\n */\r\n public convertToGammaSpace = false;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if content needs to be converted to linear space\r\n */\r\n public convertToLinearSpace = false;\r\n\r\n /**\r\n * Create a new CurrentScreenBlock\r\n * @param name defines the block name\r\n */\r\n public constructor(name: string) {\r\n super(name, NodeMaterialBlockTargets.VertexAndFragment);\r\n\r\n this._isUnique = false;\r\n\r\n this.registerInput(\"uv\", NodeMaterialBlockConnectionPointTypes.AutoDetect, false, NodeMaterialBlockTargets.VertexAndFragment);\r\n\r\n this.registerOutput(\"rgba\", NodeMaterialBlockConnectionPointTypes.Color4, NodeMaterialBlockTargets.Neutral);\r\n this.registerOutput(\"rgb\", NodeMaterialBlockConnectionPointTypes.Color3, NodeMaterialBlockTargets.Neutral);\r\n this.registerOutput(\"r\", NodeMaterialBlockConnectionPointTypes.Float, NodeMaterialBlockTargets.Neutral);\r\n this.registerOutput(\"g\", NodeMaterialBlockConnectionPointTypes.Float, NodeMaterialBlockTargets.Neutral);\r\n this.registerOutput(\"b\", NodeMaterialBlockConnectionPointTypes.Float, NodeMaterialBlockTargets.Neutral);\r\n this.registerOutput(\"a\", NodeMaterialBlockConnectionPointTypes.Float, NodeMaterialBlockTargets.Neutral);\r\n\r\n this._inputs[0].addExcludedConnectionPointFromAllowedTypes(\r\n NodeMaterialBlockConnectionPointTypes.Vector2 | NodeMaterialBlockConnectionPointTypes.Vector3 | NodeMaterialBlockConnectionPointTypes.Vector4\r\n );\r\n\r\n this._inputs[0]._prioritizeVertex = false;\r\n }\r\n\r\n /**\r\n * Gets the current class name\r\n * @returns the class name\r\n */\r\n public getClassName() {\r\n return \"CurrentScreenBlock\";\r\n }\r\n\r\n /**\r\n * Gets the uv input component\r\n */\r\n public get uv(): NodeMaterialConnectionPoint {\r\n return this._inputs[0];\r\n }\r\n\r\n /**\r\n * Gets the rgba output component\r\n */\r\n public get rgba(): NodeMaterialConnectionPoint {\r\n return this._outputs[0];\r\n }\r\n\r\n /**\r\n * Gets the rgb output component\r\n */\r\n public get rgb(): NodeMaterialConnectionPoint {\r\n return this._outputs[1];\r\n }\r\n\r\n /**\r\n * Gets the r output component\r\n */\r\n public get r(): NodeMaterialConnectionPoint {\r\n return this._outputs[2];\r\n }\r\n\r\n /**\r\n * Gets the g output component\r\n */\r\n public get g(): NodeMaterialConnectionPoint {\r\n return this._outputs[3];\r\n }\r\n\r\n /**\r\n * Gets the b output component\r\n */\r\n public get b(): NodeMaterialConnectionPoint {\r\n return this._outputs[4];\r\n }\r\n\r\n /**\r\n * Gets the a output component\r\n */\r\n public get a(): NodeMaterialConnectionPoint {\r\n return this._outputs[5];\r\n }\r\n\r\n /**\r\n * Initialize the block and prepare the context for build\r\n * @param state defines the state that will be used for the build\r\n */\r\n public initialize(state: NodeMaterialBuildState) {\r\n state._excludeVariableName(\"textureSampler\");\r\n }\r\n\r\n public get target() {\r\n if (!this.uv.isConnected) {\r\n return NodeMaterialBlockTargets.VertexAndFragment;\r\n }\r\n\r\n if (this.uv.sourceBlock!.isInput) {\r\n return NodeMaterialBlockTargets.VertexAndFragment;\r\n }\r\n\r\n return NodeMaterialBlockTargets.Fragment;\r\n }\r\n\r\n public prepareDefines(mesh: AbstractMesh, nodeMaterial: NodeMaterial, defines: NodeMaterialDefines) {\r\n defines.setValue(this._linearDefineName, this.convertToGammaSpace, true);\r\n defines.setValue(this._gammaDefineName, this.convertToLinearSpace, true);\r\n }\r\n\r\n public isReady() {\r\n if (this.texture && !this.texture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n\r\n return true;\r\n }\r\n\r\n private _injectVertexCode(state: NodeMaterialBuildState) {\r\n const uvInput = this.uv;\r\n\r\n if (uvInput.connectedPoint!.ownerBlock.isInput) {\r\n const uvInputOwnerBlock = uvInput.connectedPoint!.ownerBlock as InputBlock;\r\n\r\n if (!uvInputOwnerBlock.isAttribute) {\r\n state._emitUniformFromString(uvInput.associatedVariableName, \"vec2\");\r\n }\r\n }\r\n\r\n this._mainUVName = \"vMain\" + uvInput.associatedVariableName;\r\n\r\n state._emitVaryingFromString(this._mainUVName, \"vec2\");\r\n\r\n state.compilationString += `${this._mainUVName} = ${uvInput.associatedVariableName}.xy;\\n`;\r\n\r\n if (!this._outputs.some((o) => o.isConnectedInVertexShader)) {\r\n return;\r\n }\r\n\r\n this._writeTextureRead(state, true);\r\n\r\n for (const output of this._outputs) {\r\n if (output.hasEndpoints) {\r\n this._writeOutput(state, output, output.name, true);\r\n }\r\n }\r\n }\r\n\r\n private _writeTextureRead(state: NodeMaterialBuildState, vertexMode = false) {\r\n const uvInput = this.uv;\r\n\r\n if (vertexMode) {\r\n if (state.target === NodeMaterialBlockTargets.Fragment) {\r\n return;\r\n }\r\n\r\n state.compilationString += `vec4 ${this._tempTextureRead} = texture2D(${this._samplerName}, ${uvInput.associatedVariableName});\\n`;\r\n return;\r\n }\r\n\r\n if (this.uv.ownerBlock.target === NodeMaterialBlockTargets.Fragment) {\r\n state.compilationString += `vec4 ${this._tempTextureRead} = texture2D(${this._samplerName}, ${uvInput.associatedVariableName});\\n`;\r\n return;\r\n }\r\n\r\n state.compilationString += `vec4 ${this._tempTextureRead} = texture2D(${this._samplerName}, ${this._mainUVName});\\n`;\r\n }\r\n\r\n private _writeOutput(state: NodeMaterialBuildState, output: NodeMaterialConnectionPoint, swizzle: string, vertexMode = false) {\r\n if (vertexMode) {\r\n if (state.target === NodeMaterialBlockTargets.Fragment) {\r\n return;\r\n }\r\n\r\n state.compilationString += `${this._declareOutput(output, state)} = ${this._tempTextureRead}.${swizzle};\\n`;\r\n\r\n return;\r\n }\r\n\r\n if (this.uv.ownerBlock.target === NodeMaterialBlockTargets.Fragment) {\r\n state.compilationString += `${this._declareOutput(output, state)} = ${this._tempTextureRead}.${swizzle};\\n`;\r\n return;\r\n }\r\n\r\n state.compilationString += `${this._declareOutput(output, state)} = ${this._tempTextureRead}.${swizzle};\\n`;\r\n\r\n state.compilationString += `#ifdef ${this._linearDefineName}\\n`;\r\n state.compilationString += `${output.associatedVariableName} = toGammaSpace(${output.associatedVariableName});\\n`;\r\n state.compilationString += `#endif\\n`;\r\n\r\n state.compilationString += `#ifdef ${this._gammaDefineName}\\n`;\r\n state.compilationString += `${output.associatedVariableName} = toLinearSpace(${output.associatedVariableName});\\n`;\r\n state.compilationString += `#endif\\n`;\r\n }\r\n\r\n protected _buildBlock(state: NodeMaterialBuildState) {\r\n super._buildBlock(state);\r\n\r\n this._tempTextureRead = state._getFreeVariableName(\"tempTextureRead\");\r\n\r\n if (state.sharedData.blockingBlocks.indexOf(this) < 0) {\r\n state.sharedData.blockingBlocks.push(this);\r\n }\r\n if (state.sharedData.textureBlocks.indexOf(this) < 0) {\r\n state.sharedData.textureBlocks.push(this);\r\n }\r\n if (state.sharedData.blocksWithDefines.indexOf(this) < 0) {\r\n state.sharedData.blocksWithDefines.push(this);\r\n }\r\n\r\n if (state.target !== NodeMaterialBlockTargets.Fragment) {\r\n // Vertex\r\n state._emit2DSampler(this._samplerName);\r\n this._injectVertexCode(state);\r\n return;\r\n }\r\n\r\n // Fragment\r\n if (!this._outputs.some((o) => o.isConnectedInFragmentShader)) {\r\n return;\r\n }\r\n\r\n state._emit2DSampler(this._samplerName);\r\n\r\n this._linearDefineName = state._getFreeDefineName(\"ISLINEAR\");\r\n this._gammaDefineName = state._getFreeDefineName(\"ISGAMMA\");\r\n\r\n const comments = `//${this.name}`;\r\n state._emitFunctionFromInclude(\"helperFunctions\", comments);\r\n\r\n this._writeTextureRead(state);\r\n\r\n for (const output of this._outputs) {\r\n if (output.hasEndpoints) {\r\n this._writeOutput(state, output, output.name);\r\n }\r\n }\r\n\r\n return this;\r\n }\r\n\r\n public serialize(): any {\r\n const serializationObject = super.serialize();\r\n\r\n serializationObject.convertToGammaSpace = this.convertToGammaSpace;\r\n serializationObject.convertToLinearSpace = this.convertToLinearSpace;\r\n if (this.texture && !this.texture.isRenderTarget) {\r\n serializationObject.texture = this.texture.serialize();\r\n }\r\n\r\n return serializationObject;\r\n }\r\n\r\n public _deserialize(serializationObject: any, scene: Scene, rootUrl: string) {\r\n super._deserialize(serializationObject, scene, rootUrl);\r\n\r\n this.convertToGammaSpace = serializationObject.convertToGammaSpace;\r\n this.convertToLinearSpace = !!serializationObject.convertToLinearSpace;\r\n\r\n if (serializationObject.texture) {\r\n rootUrl = serializationObject.texture.url.indexOf(\"data:\") === 0 ? \"\" : rootUrl;\r\n this.texture = Texture.Parse(serializationObject.texture, scene, rootUrl) as Texture;\r\n }\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.CurrentScreenBlock\", CurrentScreenBlock);\r\n","import { NodeMaterialBlock } from \"../../nodeMaterialBlock\";\r\nimport { NodeMaterialBlockConnectionPointTypes } from \"../../Enums/nodeMaterialBlockConnectionPointTypes\";\r\nimport type { NodeMaterialBuildState } from \"../../nodeMaterialBuildState\";\r\nimport { NodeMaterialBlockTargets } from \"../../Enums/nodeMaterialBlockTargets\";\r\nimport type { NodeMaterialConnectionPoint } from \"../../nodeMaterialBlockConnectionPoint\";\r\nimport type { AbstractMesh } from \"../../../../Meshes/abstractMesh\";\r\nimport type { NodeMaterialDefines } from \"../../nodeMaterial\";\r\nimport { InputBlock } from \"../Input/inputBlock\";\r\nimport type { BaseTexture } from \"../../../Textures/baseTexture\";\r\nimport type { Nullable } from \"../../../../types\";\r\nimport { RegisterClass } from \"../../../../Misc/typeStore\";\r\nimport { Texture } from \"../../../Textures/texture\";\r\nimport type { Scene } from \"../../../../scene\";\r\n\r\nimport type { NodeMaterial } from \"../../nodeMaterial\";\r\n\r\n/**\r\n * Base block used for the particle texture\r\n */\r\nexport class ParticleTextureBlock extends NodeMaterialBlock {\r\n private _samplerName = \"diffuseSampler\";\r\n private _linearDefineName: string;\r\n private _gammaDefineName: string;\r\n private _tempTextureRead: string;\r\n\r\n /**\r\n * Gets or sets the texture associated with the node\r\n */\r\n public texture: Nullable;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if content needs to be converted to gamma space\r\n */\r\n public convertToGammaSpace = false;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if content needs to be converted to linear space\r\n */\r\n public convertToLinearSpace = false;\r\n\r\n /**\r\n * Create a new ParticleTextureBlock\r\n * @param name defines the block name\r\n */\r\n public constructor(name: string) {\r\n super(name, NodeMaterialBlockTargets.Fragment);\r\n\r\n this._isUnique = false;\r\n\r\n this.registerInput(\"uv\", NodeMaterialBlockConnectionPointTypes.AutoDetect, false, NodeMaterialBlockTargets.VertexAndFragment);\r\n\r\n this.registerOutput(\"rgba\", NodeMaterialBlockConnectionPointTypes.Color4, NodeMaterialBlockTargets.Neutral);\r\n this.registerOutput(\"rgb\", NodeMaterialBlockConnectionPointTypes.Color3, NodeMaterialBlockTargets.Neutral);\r\n this.registerOutput(\"r\", NodeMaterialBlockConnectionPointTypes.Float, NodeMaterialBlockTargets.Neutral);\r\n this.registerOutput(\"g\", NodeMaterialBlockConnectionPointTypes.Float, NodeMaterialBlockTargets.Neutral);\r\n this.registerOutput(\"b\", NodeMaterialBlockConnectionPointTypes.Float, NodeMaterialBlockTargets.Neutral);\r\n this.registerOutput(\"a\", NodeMaterialBlockConnectionPointTypes.Float, NodeMaterialBlockTargets.Neutral);\r\n\r\n this._inputs[0].addExcludedConnectionPointFromAllowedTypes(\r\n NodeMaterialBlockConnectionPointTypes.Vector2 | NodeMaterialBlockConnectionPointTypes.Vector3 | NodeMaterialBlockConnectionPointTypes.Vector4\r\n );\r\n }\r\n\r\n /**\r\n * Gets the current class name\r\n * @returns the class name\r\n */\r\n public getClassName() {\r\n return \"ParticleTextureBlock\";\r\n }\r\n\r\n /**\r\n * Gets the uv input component\r\n */\r\n public get uv(): NodeMaterialConnectionPoint {\r\n return this._inputs[0];\r\n }\r\n\r\n /**\r\n * Gets the rgba output component\r\n */\r\n public get rgba(): NodeMaterialConnectionPoint {\r\n return this._outputs[0];\r\n }\r\n\r\n /**\r\n * Gets the rgb output component\r\n */\r\n public get rgb(): NodeMaterialConnectionPoint {\r\n return this._outputs[1];\r\n }\r\n\r\n /**\r\n * Gets the r output component\r\n */\r\n public get r(): NodeMaterialConnectionPoint {\r\n return this._outputs[2];\r\n }\r\n\r\n /**\r\n * Gets the g output component\r\n */\r\n public get g(): NodeMaterialConnectionPoint {\r\n return this._outputs[3];\r\n }\r\n\r\n /**\r\n * Gets the b output component\r\n */\r\n public get b(): NodeMaterialConnectionPoint {\r\n return this._outputs[4];\r\n }\r\n\r\n /**\r\n * Gets the a output component\r\n */\r\n public get a(): NodeMaterialConnectionPoint {\r\n return this._outputs[5];\r\n }\r\n\r\n /**\r\n * Initialize the block and prepare the context for build\r\n * @param state defines the state that will be used for the build\r\n */\r\n public initialize(state: NodeMaterialBuildState) {\r\n state._excludeVariableName(\"diffuseSampler\");\r\n }\r\n\r\n public autoConfigure(material: NodeMaterial) {\r\n if (!this.uv.isConnected) {\r\n let uvInput = material.getInputBlockByPredicate((b) => b.isAttribute && b.name === \"particle_uv\");\r\n\r\n if (!uvInput) {\r\n uvInput = new InputBlock(\"uv\");\r\n uvInput.setAsAttribute(\"particle_uv\");\r\n }\r\n uvInput.output.connectTo(this.uv);\r\n }\r\n }\r\n\r\n public prepareDefines(mesh: AbstractMesh, nodeMaterial: NodeMaterial, defines: NodeMaterialDefines) {\r\n defines.setValue(this._linearDefineName, this.convertToGammaSpace, true);\r\n defines.setValue(this._gammaDefineName, this.convertToLinearSpace, true);\r\n }\r\n\r\n public isReady() {\r\n if (this.texture && !this.texture.isReadyOrNotBlocking()) {\r\n return false;\r\n }\r\n\r\n return true;\r\n }\r\n\r\n private _writeOutput(state: NodeMaterialBuildState, output: NodeMaterialConnectionPoint, swizzle: string) {\r\n state.compilationString += `${this._declareOutput(output, state)} = ${this._tempTextureRead}.${swizzle};\\n`;\r\n\r\n state.compilationString += `#ifdef ${this._linearDefineName}\\n`;\r\n state.compilationString += `${output.associatedVariableName} = toGammaSpace(${output.associatedVariableName});\\n`;\r\n state.compilationString += `#endif\\n`;\r\n\r\n state.compilationString += `#ifdef ${this._gammaDefineName}\\n`;\r\n state.compilationString += `${output.associatedVariableName} = toLinearSpace(${output.associatedVariableName});\\n`;\r\n state.compilationString += `#endif\\n`;\r\n }\r\n\r\n protected _buildBlock(state: NodeMaterialBuildState) {\r\n super._buildBlock(state);\r\n\r\n if (state.target === NodeMaterialBlockTargets.Vertex) {\r\n return;\r\n }\r\n\r\n this._tempTextureRead = state._getFreeVariableName(\"tempTextureRead\");\r\n\r\n state._emit2DSampler(this._samplerName);\r\n\r\n state.sharedData.blockingBlocks.push(this);\r\n state.sharedData.textureBlocks.push(this);\r\n state.sharedData.blocksWithDefines.push(this);\r\n\r\n this._linearDefineName = state._getFreeDefineName(\"ISLINEAR\");\r\n this._gammaDefineName = state._getFreeDefineName(\"ISGAMMA\");\r\n\r\n const comments = `//${this.name}`;\r\n state._emitFunctionFromInclude(\"helperFunctions\", comments);\r\n\r\n state.compilationString += `vec4 ${this._tempTextureRead} = texture2D(${this._samplerName}, ${this.uv.associatedVariableName});\\n`;\r\n\r\n for (const output of this._outputs) {\r\n if (output.hasEndpoints) {\r\n this._writeOutput(state, output, output.name);\r\n }\r\n }\r\n\r\n return this;\r\n }\r\n\r\n public serialize(): any {\r\n const serializationObject = super.serialize();\r\n\r\n serializationObject.convertToGammaSpace = this.convertToGammaSpace;\r\n serializationObject.convertToLinearSpace = this.convertToLinearSpace;\r\n if (this.texture && !this.texture.isRenderTarget) {\r\n serializationObject.texture = this.texture.serialize();\r\n }\r\n\r\n return serializationObject;\r\n }\r\n\r\n public _deserialize(serializationObject: any, scene: Scene, rootUrl: string) {\r\n super._deserialize(serializationObject, scene, rootUrl);\r\n\r\n this.convertToGammaSpace = serializationObject.convertToGammaSpace;\r\n this.convertToLinearSpace = !!serializationObject.convertToLinearSpace;\r\n\r\n if (serializationObject.texture) {\r\n rootUrl = serializationObject.texture.url.indexOf(\"data:\") === 0 ? \"\" : rootUrl;\r\n this.texture = Texture.Parse(serializationObject.texture, scene, rootUrl) as Texture;\r\n }\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.ParticleTextureBlock\", ParticleTextureBlock);\r\n","import { NodeMaterialBlock } from \"../../nodeMaterialBlock\";\r\nimport { NodeMaterialBlockConnectionPointTypes } from \"../../Enums/nodeMaterialBlockConnectionPointTypes\";\r\nimport type { NodeMaterialBuildState } from \"../../nodeMaterialBuildState\";\r\nimport { NodeMaterialBlockTargets } from \"../../Enums/nodeMaterialBlockTargets\";\r\nimport type { NodeMaterialConnectionPoint } from \"../../nodeMaterialBlockConnectionPoint\";\r\nimport { RegisterClass } from \"../../../../Misc/typeStore\";\r\n\r\n/**\r\n * Block used for the particle ramp gradient section\r\n */\r\nexport class ParticleRampGradientBlock extends NodeMaterialBlock {\r\n /**\r\n * Create a new ParticleRampGradientBlock\r\n * @param name defines the block name\r\n */\r\n public constructor(name: string) {\r\n super(name, NodeMaterialBlockTargets.Fragment);\r\n\r\n this._isUnique = true;\r\n\r\n this.registerInput(\"color\", NodeMaterialBlockConnectionPointTypes.Color4, false, NodeMaterialBlockTargets.Fragment);\r\n\r\n this.registerOutput(\"rampColor\", NodeMaterialBlockConnectionPointTypes.Color4, NodeMaterialBlockTargets.Fragment);\r\n }\r\n\r\n /**\r\n * Gets the current class name\r\n * @returns the class name\r\n */\r\n public getClassName() {\r\n return \"ParticleRampGradientBlock\";\r\n }\r\n\r\n /**\r\n * Gets the color input component\r\n */\r\n public get color(): NodeMaterialConnectionPoint {\r\n return this._inputs[0];\r\n }\r\n\r\n /**\r\n * Gets the rampColor output component\r\n */\r\n public get rampColor(): NodeMaterialConnectionPoint {\r\n return this._outputs[0];\r\n }\r\n\r\n /**\r\n * Initialize the block and prepare the context for build\r\n * @param state defines the state that will be used for the build\r\n */\r\n public initialize(state: NodeMaterialBuildState) {\r\n state._excludeVariableName(\"remapRanges\");\r\n state._excludeVariableName(\"rampSampler\");\r\n state._excludeVariableName(\"baseColor\");\r\n state._excludeVariableName(\"alpha\");\r\n state._excludeVariableName(\"remappedColorIndex\");\r\n state._excludeVariableName(\"rampColor\");\r\n state._excludeVariableName(\"finalAlpha\");\r\n }\r\n\r\n protected _buildBlock(state: NodeMaterialBuildState) {\r\n super._buildBlock(state);\r\n\r\n if (state.target === NodeMaterialBlockTargets.Vertex) {\r\n return;\r\n }\r\n\r\n state._emit2DSampler(\"rampSampler\");\r\n state._emitVaryingFromString(\"remapRanges\", \"vec4\", \"RAMPGRADIENT\");\r\n\r\n state.compilationString += `\r\n #ifdef RAMPGRADIENT\r\n vec4 baseColor = ${this.color.associatedVariableName};\r\n float alpha = ${this.color.associatedVariableName}.a;\r\n\r\n float remappedColorIndex = clamp((alpha - remapRanges.x) / remapRanges.y, 0.0, 1.0);\r\n\r\n vec4 rampColor = texture2D(rampSampler, vec2(1.0 - remappedColorIndex, 0.));\r\n baseColor.rgb *= rampColor.rgb;\r\n\r\n // Remapped alpha\r\n float finalAlpha = baseColor.a;\r\n baseColor.a = clamp((alpha * rampColor.a - remapRanges.z) / remapRanges.w, 0.0, 1.0);\r\n\r\n ${this._declareOutput(this.rampColor, state)} = baseColor;\r\n #else\r\n ${this._declareOutput(this.rampColor, state)} = ${this.color.associatedVariableName};\r\n #endif\r\n `;\r\n\r\n return this;\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.ParticleRampGradientBlock\", ParticleRampGradientBlock);\r\n","import { NodeMaterialBlock } from \"../../nodeMaterialBlock\";\r\nimport { NodeMaterialBlockConnectionPointTypes } from \"../../Enums/nodeMaterialBlockConnectionPointTypes\";\r\nimport type { NodeMaterialBuildState } from \"../../nodeMaterialBuildState\";\r\nimport { NodeMaterialBlockTargets } from \"../../Enums/nodeMaterialBlockTargets\";\r\nimport type { NodeMaterialConnectionPoint } from \"../../nodeMaterialBlockConnectionPoint\";\r\nimport { RegisterClass } from \"../../../../Misc/typeStore\";\r\n\r\n/**\r\n * Block used for the particle blend multiply section\r\n */\r\nexport class ParticleBlendMultiplyBlock extends NodeMaterialBlock {\r\n /**\r\n * Create a new ParticleBlendMultiplyBlock\r\n * @param name defines the block name\r\n */\r\n public constructor(name: string) {\r\n super(name, NodeMaterialBlockTargets.Fragment);\r\n\r\n this._isUnique = true;\r\n\r\n this.registerInput(\"color\", NodeMaterialBlockConnectionPointTypes.Color4, false, NodeMaterialBlockTargets.Fragment);\r\n this.registerInput(\"alphaTexture\", NodeMaterialBlockConnectionPointTypes.Float, false, NodeMaterialBlockTargets.Fragment);\r\n this.registerInput(\"alphaColor\", NodeMaterialBlockConnectionPointTypes.Float, false, NodeMaterialBlockTargets.Fragment);\r\n\r\n this.registerOutput(\"blendColor\", NodeMaterialBlockConnectionPointTypes.Color4, NodeMaterialBlockTargets.Fragment);\r\n }\r\n\r\n /**\r\n * Gets the current class name\r\n * @returns the class name\r\n */\r\n public getClassName() {\r\n return \"ParticleBlendMultiplyBlock\";\r\n }\r\n\r\n /**\r\n * Gets the color input component\r\n */\r\n public get color(): NodeMaterialConnectionPoint {\r\n return this._inputs[0];\r\n }\r\n\r\n /**\r\n * Gets the alphaTexture input component\r\n */\r\n public get alphaTexture(): NodeMaterialConnectionPoint {\r\n return this._inputs[1];\r\n }\r\n\r\n /**\r\n * Gets the alphaColor input component\r\n */\r\n public get alphaColor(): NodeMaterialConnectionPoint {\r\n return this._inputs[2];\r\n }\r\n\r\n /**\r\n * Gets the blendColor output component\r\n */\r\n public get blendColor(): NodeMaterialConnectionPoint {\r\n return this._outputs[0];\r\n }\r\n\r\n /**\r\n * Initialize the block and prepare the context for build\r\n * @param state defines the state that will be used for the build\r\n */\r\n public initialize(state: NodeMaterialBuildState) {\r\n state._excludeVariableName(\"sourceAlpha\");\r\n }\r\n\r\n protected _buildBlock(state: NodeMaterialBuildState) {\r\n super._buildBlock(state);\r\n\r\n if (state.target === NodeMaterialBlockTargets.Vertex) {\r\n return;\r\n }\r\n\r\n state.compilationString += `\r\n #ifdef BLENDMULTIPLYMODE\r\n ${this._declareOutput(this.blendColor, state)};\r\n float sourceAlpha = ${this.alphaColor.associatedVariableName} * ${this.alphaTexture.associatedVariableName};\r\n ${this.blendColor.associatedVariableName}.rgb = ${this.color.associatedVariableName}.rgb * sourceAlpha + vec3(1.0) * (1.0 - sourceAlpha);\r\n ${this.blendColor.associatedVariableName}.a = ${this.color.associatedVariableName}.a;\r\n #else\r\n ${this._declareOutput(this.blendColor, state)} = ${this.color.associatedVariableName};\r\n #endif\r\n `;\r\n\r\n return this;\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.ParticleBlendMultiplyBlock\", ParticleBlendMultiplyBlock);\r\n","import { NodeMaterialBlock } from \"../nodeMaterialBlock\";\r\nimport { NodeMaterialBlockConnectionPointTypes } from \"../Enums/nodeMaterialBlockConnectionPointTypes\";\r\nimport type { NodeMaterialBuildState } from \"../nodeMaterialBuildState\";\r\nimport { NodeMaterialBlockTargets } from \"../Enums/nodeMaterialBlockTargets\";\r\nimport type { NodeMaterialConnectionPoint } from \"../nodeMaterialBlockConnectionPoint\";\r\nimport { RegisterClass } from \"../../../Misc/typeStore\";\r\nimport type { Scene } from \"../../../scene\";\r\n\r\n/**\r\n * Block used to create a Vector2/3/4 out of individual inputs (one for each component)\r\n */\r\nexport class VectorMergerBlock extends NodeMaterialBlock {\r\n /**\r\n * Gets or sets the swizzle for x (meaning which component to affect to the output.x)\r\n */\r\n public xSwizzle: \"x\" | \"y\" | \"z\" | \"w\" = \"x\";\r\n /**\r\n * Gets or sets the swizzle for y (meaning which component to affect to the output.y)\r\n */\r\n public ySwizzle: \"x\" | \"y\" | \"z\" | \"w\" = \"y\";\r\n /**\r\n * Gets or sets the swizzle for z (meaning which component to affect to the output.z)\r\n */\r\n public zSwizzle: \"x\" | \"y\" | \"z\" | \"w\" = \"z\";\r\n /**\r\n * Gets or sets the swizzle for w (meaning which component to affect to the output.w)\r\n */\r\n public wSwizzle: \"x\" | \"y\" | \"z\" | \"w\" = \"w\";\r\n\r\n /**\r\n * Create a new VectorMergerBlock\r\n * @param name defines the block name\r\n */\r\n public constructor(name: string) {\r\n super(name, NodeMaterialBlockTargets.Neutral);\r\n\r\n this.registerInput(\"xyzw \", NodeMaterialBlockConnectionPointTypes.Vector4, true);\r\n this.registerInput(\"xyz \", NodeMaterialBlockConnectionPointTypes.Vector3, true);\r\n this.registerInput(\"xy \", NodeMaterialBlockConnectionPointTypes.Vector2, true);\r\n this.registerInput(\"zw \", NodeMaterialBlockConnectionPointTypes.Vector2, true);\r\n this.registerInput(\"x\", NodeMaterialBlockConnectionPointTypes.Float, true);\r\n this.registerInput(\"y\", NodeMaterialBlockConnectionPointTypes.Float, true);\r\n this.registerInput(\"z\", NodeMaterialBlockConnectionPointTypes.Float, true);\r\n this.registerInput(\"w\", NodeMaterialBlockConnectionPointTypes.Float, true);\r\n\r\n this.registerOutput(\"xyzw\", NodeMaterialBlockConnectionPointTypes.Vector4);\r\n this.registerOutput(\"xyz\", NodeMaterialBlockConnectionPointTypes.Vector3);\r\n this.registerOutput(\"xy\", NodeMaterialBlockConnectionPointTypes.Vector2);\r\n this.registerOutput(\"zw\", NodeMaterialBlockConnectionPointTypes.Vector2);\r\n }\r\n\r\n /**\r\n * Gets the current class name\r\n * @returns the class name\r\n */\r\n public getClassName() {\r\n return \"VectorMergerBlock\";\r\n }\r\n\r\n /**\r\n * Gets the xyzw component (input)\r\n */\r\n public get xyzwIn(): NodeMaterialConnectionPoint {\r\n return this._inputs[0];\r\n }\r\n\r\n /**\r\n * Gets the xyz component (input)\r\n */\r\n public get xyzIn(): NodeMaterialConnectionPoint {\r\n return this._inputs[1];\r\n }\r\n\r\n /**\r\n * Gets the xy component (input)\r\n */\r\n public get xyIn(): NodeMaterialConnectionPoint {\r\n return this._inputs[2];\r\n }\r\n\r\n /**\r\n * Gets the zw component (input)\r\n */\r\n public get zwIn(): NodeMaterialConnectionPoint {\r\n return this._inputs[3];\r\n }\r\n\r\n /**\r\n * Gets the x component (input)\r\n */\r\n public get x(): NodeMaterialConnectionPoint {\r\n return this._inputs[4];\r\n }\r\n\r\n /**\r\n * Gets the y component (input)\r\n */\r\n public get y(): NodeMaterialConnectionPoint {\r\n return this._inputs[5];\r\n }\r\n\r\n /**\r\n * Gets the z component (input)\r\n */\r\n public get z(): NodeMaterialConnectionPoint {\r\n return this._inputs[6];\r\n }\r\n\r\n /**\r\n * Gets the w component (input)\r\n */\r\n public get w(): NodeMaterialConnectionPoint {\r\n return this._inputs[7];\r\n }\r\n\r\n /**\r\n * Gets the xyzw component (output)\r\n */\r\n public get xyzw(): NodeMaterialConnectionPoint {\r\n return this._outputs[0];\r\n }\r\n\r\n /**\r\n * Gets the xyz component (output)\r\n */\r\n public get xyzOut(): NodeMaterialConnectionPoint {\r\n return this._outputs[1];\r\n }\r\n\r\n /**\r\n * Gets the xy component (output)\r\n */\r\n public get xyOut(): NodeMaterialConnectionPoint {\r\n return this._outputs[2];\r\n }\r\n\r\n /**\r\n * Gets the zw component (output)\r\n */\r\n public get zwOut(): NodeMaterialConnectionPoint {\r\n return this._outputs[3];\r\n }\r\n\r\n /**\r\n * Gets the xy component (output)\r\n * @deprecated Please use xyOut instead.\r\n */\r\n public get xy(): NodeMaterialConnectionPoint {\r\n return this.xyOut;\r\n }\r\n\r\n /**\r\n * Gets the xyz component (output)\r\n * @deprecated Please use xyzOut instead.\r\n */\r\n public get xyz(): NodeMaterialConnectionPoint {\r\n return this.xyzOut;\r\n }\r\n\r\n protected _inputRename(name: string) {\r\n if (name === \"xyzw \") {\r\n return \"xyzwIn\";\r\n }\r\n if (name === \"xyz \") {\r\n return \"xyzIn\";\r\n }\r\n if (name === \"xy \") {\r\n return \"xyIn\";\r\n }\r\n if (name === \"zw \") {\r\n return \"zwIn\";\r\n }\r\n return name;\r\n }\r\n\r\n private _buildSwizzle(len: number) {\r\n const swizzle = this.xSwizzle + this.ySwizzle + this.zSwizzle + this.wSwizzle;\r\n\r\n return \".\" + swizzle.substr(0, len);\r\n }\r\n\r\n protected _buildBlock(state: NodeMaterialBuildState) {\r\n super._buildBlock(state);\r\n\r\n const xInput = this.x;\r\n const yInput = this.y;\r\n const zInput = this.z;\r\n const wInput = this.w;\r\n const xyInput = this.xyIn;\r\n const zwInput = this.zwIn;\r\n const xyzInput = this.xyzIn;\r\n const xyzwInput = this.xyzwIn;\r\n\r\n const v4Output = this._outputs[0];\r\n const v3Output = this._outputs[1];\r\n const v2Output = this._outputs[2];\r\n const v2CompOutput = this._outputs[3];\r\n\r\n if (xyzwInput.isConnected) {\r\n if (v4Output.hasEndpoints) {\r\n state.compilationString += this._declareOutput(v4Output, state) + ` = ${xyzwInput.associatedVariableName}${this._buildSwizzle(4)};\\n`;\r\n }\r\n\r\n if (v3Output.hasEndpoints) {\r\n state.compilationString += this._declareOutput(v3Output, state) + ` = ${xyzwInput.associatedVariableName}${this._buildSwizzle(3)};\\n`;\r\n }\r\n\r\n if (v2Output.hasEndpoints) {\r\n state.compilationString += this._declareOutput(v2Output, state) + ` = ${xyzwInput.associatedVariableName}${this._buildSwizzle(2)};\\n`;\r\n }\r\n } else if (xyzInput.isConnected) {\r\n if (v4Output.hasEndpoints) {\r\n state.compilationString +=\r\n this._declareOutput(v4Output, state) +\r\n ` = vec4(${xyzInput.associatedVariableName}, ${wInput.isConnected ? this._writeVariable(wInput) : \"0.0\"})${this._buildSwizzle(4)};\\n`;\r\n }\r\n\r\n if (v3Output.hasEndpoints) {\r\n state.compilationString += this._declareOutput(v3Output, state) + ` = ${xyzInput.associatedVariableName}${this._buildSwizzle(3)};\\n`;\r\n }\r\n\r\n if (v2Output.hasEndpoints) {\r\n state.compilationString += this._declareOutput(v2Output, state) + ` = ${xyzInput.associatedVariableName}${this._buildSwizzle(2)};\\n`;\r\n }\r\n } else if (xyInput.isConnected) {\r\n if (v4Output.hasEndpoints) {\r\n if (zwInput.isConnected) {\r\n state.compilationString +=\r\n this._declareOutput(v4Output, state) + ` = vec4(${xyInput.associatedVariableName}, ${zwInput.associatedVariableName})${this._buildSwizzle(4)};\\n`;\r\n } else {\r\n state.compilationString +=\r\n this._declareOutput(v4Output, state) +\r\n ` = vec4(${xyInput.associatedVariableName}, ${zInput.isConnected ? this._writeVariable(zInput) : \"0.0\"}, ${\r\n wInput.isConnected ? this._writeVariable(wInput) : \"0.0\"\r\n })${this._buildSwizzle(4)};\\n`;\r\n }\r\n }\r\n\r\n if (v3Output.hasEndpoints) {\r\n state.compilationString +=\r\n this._declareOutput(v3Output, state) +\r\n ` = vec3(${xyInput.associatedVariableName}, ${zInput.isConnected ? this._writeVariable(zInput) : \"0.0\"})${this._buildSwizzle(3)};\\n`;\r\n }\r\n\r\n if (v2Output.hasEndpoints) {\r\n state.compilationString += this._declareOutput(v2Output, state) + ` = ${xyInput.associatedVariableName}${this._buildSwizzle(2)};\\n`;\r\n }\r\n\r\n if (v2CompOutput.hasEndpoints) {\r\n if (zwInput.isConnected) {\r\n state.compilationString += this._declareOutput(v2CompOutput, state) + ` = ${zwInput.associatedVariableName}${this._buildSwizzle(2)};\\n`;\r\n } else {\r\n state.compilationString +=\r\n this._declareOutput(v2CompOutput, state) +\r\n ` = vec2(${zInput.isConnected ? this._writeVariable(zInput) : \"0.0\"}, ${wInput.isConnected ? this._writeVariable(wInput) : \"0.0\"})${this._buildSwizzle(\r\n 2\r\n )};\\n`;\r\n }\r\n }\r\n } else {\r\n if (v4Output.hasEndpoints) {\r\n if (zwInput.isConnected) {\r\n state.compilationString +=\r\n this._declareOutput(v4Output, state) +\r\n ` = vec4(${xInput.isConnected ? this._writeVariable(xInput) : \"0.0\"}, ${yInput.isConnected ? this._writeVariable(yInput) : \"0.0\"}, ${\r\n zwInput.associatedVariableName\r\n })${this._buildSwizzle(4)};\\n`;\r\n } else {\r\n state.compilationString +=\r\n this._declareOutput(v4Output, state) +\r\n ` = vec4(${xInput.isConnected ? this._writeVariable(xInput) : \"0.0\"}, ${yInput.isConnected ? this._writeVariable(yInput) : \"0.0\"}, ${\r\n zInput.isConnected ? this._writeVariable(zInput) : \"0.0\"\r\n }, ${wInput.isConnected ? this._writeVariable(wInput) : \"0.0\"})${this._buildSwizzle(4)};\\n`;\r\n }\r\n }\r\n\r\n if (v3Output.hasEndpoints) {\r\n state.compilationString +=\r\n this._declareOutput(v3Output, state) +\r\n ` = vec3(${xInput.isConnected ? this._writeVariable(xInput) : \"0.0\"}, ${yInput.isConnected ? this._writeVariable(yInput) : \"0.0\"}, ${\r\n zInput.isConnected ? this._writeVariable(zInput) : \"0.0\"\r\n })${this._buildSwizzle(3)};\\n`;\r\n }\r\n\r\n if (v2Output.hasEndpoints) {\r\n state.compilationString +=\r\n this._declareOutput(v2Output, state) +\r\n ` = vec2(${xInput.isConnected ? this._writeVariable(xInput) : \"0.0\"}, ${yInput.isConnected ? this._writeVariable(yInput) : \"0.0\"})${this._buildSwizzle(2)};\\n`;\r\n }\r\n\r\n if (v2CompOutput.hasEndpoints) {\r\n if (zwInput.isConnected) {\r\n state.compilationString += this._declareOutput(v2CompOutput, state) + ` = ${zwInput.associatedVariableName}${this._buildSwizzle(2)};\\n`;\r\n } else {\r\n state.compilationString +=\r\n this._declareOutput(v2CompOutput, state) +\r\n ` = vec2(${zInput.isConnected ? this._writeVariable(zInput) : \"0.0\"}, ${wInput.isConnected ? this._writeVariable(wInput) : \"0.0\"})${this._buildSwizzle(\r\n 2\r\n )};\\n`;\r\n }\r\n }\r\n }\r\n\r\n return this;\r\n }\r\n\r\n public serialize(): any {\r\n const serializationObject = super.serialize();\r\n\r\n serializationObject.xSwizzle = this.xSwizzle;\r\n serializationObject.ySwizzle = this.ySwizzle;\r\n serializationObject.zSwizzle = this.zSwizzle;\r\n serializationObject.wSwizzle = this.wSwizzle;\r\n\r\n return serializationObject;\r\n }\r\n\r\n public _deserialize(serializationObject: any, scene: Scene, rootUrl: string) {\r\n super._deserialize(serializationObject, scene, rootUrl);\r\n\r\n this.xSwizzle = serializationObject.xSwizzle ?? \"x\";\r\n this.ySwizzle = serializationObject.ySwizzle ?? \"y\";\r\n this.zSwizzle = serializationObject.zSwizzle ?? \"z\";\r\n this.wSwizzle = serializationObject.wSwizzle ?? \"w\";\r\n }\r\n\r\n protected _dumpPropertiesCode() {\r\n let codeString = super._dumpPropertiesCode();\r\n codeString += `${this._codeVariableName}.xSwizzle = \"${this.xSwizzle}\";\\n`;\r\n codeString += `${this._codeVariableName}.ySwizzle = \"${this.ySwizzle}\";\\n`;\r\n codeString += `${this._codeVariableName}.zSwizzle = \"${this.zSwizzle}\";\\n`;\r\n codeString += `${this._codeVariableName}.wSwizzle = \"${this.wSwizzle}\";\\n`;\r\n\r\n return codeString;\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.VectorMergerBlock\", VectorMergerBlock);\r\n","import { NodeMaterialBlock } from \"../nodeMaterialBlock\";\r\nimport { NodeMaterialBlockConnectionPointTypes } from \"../Enums/nodeMaterialBlockConnectionPointTypes\";\r\nimport type { NodeMaterialBuildState } from \"../nodeMaterialBuildState\";\r\nimport type { NodeMaterialConnectionPoint } from \"../nodeMaterialBlockConnectionPoint\";\r\nimport { NodeMaterialBlockTargets } from \"../Enums/nodeMaterialBlockTargets\";\r\nimport { RegisterClass } from \"../../../Misc/typeStore\";\r\nimport { Vector2 } from \"../../../Maths/math.vector\";\r\nimport type { Scene } from \"../../../scene\";\r\nimport { editableInPropertyPage, PropertyTypeForEdition } from \"../../../Decorators/nodeDecorator\";\r\n/**\r\n * Block used to remap a float from a range to a new one\r\n */\r\nexport class RemapBlock extends NodeMaterialBlock {\r\n /**\r\n * Gets or sets the source range\r\n */\r\n @editableInPropertyPage(\"From\", PropertyTypeForEdition.Vector2)\r\n public sourceRange = new Vector2(-1, 1);\r\n\r\n /**\r\n * Gets or sets the target range\r\n */\r\n @editableInPropertyPage(\"To\", PropertyTypeForEdition.Vector2)\r\n public targetRange = new Vector2(0, 1);\r\n\r\n /**\r\n * Creates a new RemapBlock\r\n * @param name defines the block name\r\n */\r\n public constructor(name: string) {\r\n super(name, NodeMaterialBlockTargets.Neutral);\r\n\r\n this.registerInput(\"input\", NodeMaterialBlockConnectionPointTypes.AutoDetect);\r\n this.registerInput(\"sourceMin\", NodeMaterialBlockConnectionPointTypes.Float, true);\r\n this.registerInput(\"sourceMax\", NodeMaterialBlockConnectionPointTypes.Float, true);\r\n this.registerInput(\"targetMin\", NodeMaterialBlockConnectionPointTypes.Float, true);\r\n this.registerInput(\"targetMax\", NodeMaterialBlockConnectionPointTypes.Float, true);\r\n\r\n this.registerOutput(\"output\", NodeMaterialBlockConnectionPointTypes.BasedOnInput);\r\n\r\n this._outputs[0]._typeConnectionSource = this._inputs[0];\r\n }\r\n\r\n /**\r\n * Gets the current class name\r\n * @returns the class name\r\n */\r\n public getClassName() {\r\n return \"RemapBlock\";\r\n }\r\n\r\n /**\r\n * Gets the input component\r\n */\r\n public get input(): NodeMaterialConnectionPoint {\r\n return this._inputs[0];\r\n }\r\n\r\n /**\r\n * Gets the source min input component\r\n */\r\n public get sourceMin(): NodeMaterialConnectionPoint {\r\n return this._inputs[1];\r\n }\r\n\r\n /**\r\n * Gets the source max input component\r\n */\r\n public get sourceMax(): NodeMaterialConnectionPoint {\r\n return this._inputs[2];\r\n }\r\n\r\n /**\r\n * Gets the target min input component\r\n */\r\n public get targetMin(): NodeMaterialConnectionPoint {\r\n return this._inputs[3];\r\n }\r\n\r\n /**\r\n * Gets the target max input component\r\n */\r\n public get targetMax(): NodeMaterialConnectionPoint {\r\n return this._inputs[4];\r\n }\r\n\r\n /**\r\n * Gets the output component\r\n */\r\n public get output(): NodeMaterialConnectionPoint {\r\n return this._outputs[0];\r\n }\r\n\r\n protected _buildBlock(state: NodeMaterialBuildState) {\r\n super._buildBlock(state);\r\n\r\n const output = this._outputs[0];\r\n\r\n const sourceMin = this.sourceMin.isConnected ? this.sourceMin.associatedVariableName : this._writeFloat(this.sourceRange.x);\r\n const sourceMax = this.sourceMax.isConnected ? this.sourceMax.associatedVariableName : this._writeFloat(this.sourceRange.y);\r\n\r\n const targetMin = this.targetMin.isConnected ? this.targetMin.associatedVariableName : this._writeFloat(this.targetRange.x);\r\n const targetMax = this.targetMax.isConnected ? this.targetMax.associatedVariableName : this._writeFloat(this.targetRange.y);\r\n\r\n state.compilationString +=\r\n this._declareOutput(output, state) +\r\n ` = ${targetMin} + (${this._inputs[0].associatedVariableName} - ${sourceMin}) * (${targetMax} - ${targetMin}) / (${sourceMax} - ${sourceMin});\\n`;\r\n\r\n return this;\r\n }\r\n\r\n protected _dumpPropertiesCode() {\r\n let codeString = super._dumpPropertiesCode() + `${this._codeVariableName}.sourceRange = new BABYLON.Vector2(${this.sourceRange.x}, ${this.sourceRange.y});\\n`;\r\n\r\n codeString += `${this._codeVariableName}.targetRange = new BABYLON.Vector2(${this.targetRange.x}, ${this.targetRange.y});\\n`;\r\n\r\n return codeString;\r\n }\r\n\r\n public serialize(): any {\r\n const serializationObject = super.serialize();\r\n\r\n serializationObject.sourceRange = this.sourceRange.asArray();\r\n serializationObject.targetRange = this.targetRange.asArray();\r\n\r\n return serializationObject;\r\n }\r\n\r\n public _deserialize(serializationObject: any, scene: Scene, rootUrl: string) {\r\n super._deserialize(serializationObject, scene, rootUrl);\r\n\r\n this.sourceRange = Vector2.FromArray(serializationObject.sourceRange);\r\n this.targetRange = Vector2.FromArray(serializationObject.targetRange);\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.RemapBlock\", RemapBlock);\r\n","import { NodeMaterialBlock } from \"../nodeMaterialBlock\";\r\nimport { NodeMaterialBlockConnectionPointTypes } from \"../Enums/nodeMaterialBlockConnectionPointTypes\";\r\nimport type { NodeMaterialBuildState } from \"../nodeMaterialBuildState\";\r\nimport type { NodeMaterialConnectionPoint } from \"../nodeMaterialBlockConnectionPoint\";\r\nimport { NodeMaterialBlockTargets } from \"../Enums/nodeMaterialBlockTargets\";\r\nimport { RegisterClass } from \"../../../Misc/typeStore\";\r\n/**\r\n * Block used to multiply 2 values\r\n */\r\nexport class MultiplyBlock extends NodeMaterialBlock {\r\n /**\r\n * Creates a new MultiplyBlock\r\n * @param name defines the block name\r\n */\r\n public constructor(name: string) {\r\n super(name, NodeMaterialBlockTargets.Neutral);\r\n\r\n this.registerInput(\"left\", NodeMaterialBlockConnectionPointTypes.AutoDetect);\r\n this.registerInput(\"right\", NodeMaterialBlockConnectionPointTypes.AutoDetect);\r\n this.registerOutput(\"output\", NodeMaterialBlockConnectionPointTypes.BasedOnInput);\r\n\r\n this._outputs[0]._typeConnectionSource = this._inputs[0];\r\n this._linkConnectionTypes(0, 1);\r\n\r\n this._inputs[0].acceptedConnectionPointTypes.push(NodeMaterialBlockConnectionPointTypes.Float);\r\n this._inputs[1].acceptedConnectionPointTypes.push(NodeMaterialBlockConnectionPointTypes.Float);\r\n }\r\n\r\n /**\r\n * Gets the current class name\r\n * @returns the class name\r\n */\r\n public getClassName() {\r\n return \"MultiplyBlock\";\r\n }\r\n\r\n /**\r\n * Gets the left operand input component\r\n */\r\n public get left(): NodeMaterialConnectionPoint {\r\n return this._inputs[0];\r\n }\r\n\r\n /**\r\n * Gets the right operand input component\r\n */\r\n public get right(): NodeMaterialConnectionPoint {\r\n return this._inputs[1];\r\n }\r\n\r\n /**\r\n * Gets the output component\r\n */\r\n public get output(): NodeMaterialConnectionPoint {\r\n return this._outputs[0];\r\n }\r\n\r\n protected _buildBlock(state: NodeMaterialBuildState) {\r\n super._buildBlock(state);\r\n\r\n const output = this._outputs[0];\r\n\r\n state.compilationString += this._declareOutput(output, state) + ` = ${this.left.associatedVariableName} * ${this.right.associatedVariableName};\\n`;\r\n\r\n return this;\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.MultiplyBlock\", MultiplyBlock);\r\n","/**\r\n * Enum used to define the material modes\r\n */\r\nexport enum NodeMaterialModes {\r\n /** Regular material */\r\n Material = 0,\r\n /** For post process */\r\n PostProcess = 1,\r\n /** For particle system */\r\n Particle = 2,\r\n /** For procedural texture */\r\n ProceduralTexture = 3,\r\n}\r\n","import { DeepCopier } from \"../../Misc/deepCopier\";\r\nimport type { Matrix } from \"../../Maths/math.vector\";\r\nimport { Vector3 } from \"../../Maths/math.vector\";\r\nimport { Scalar } from \"../../Maths/math.scalar\";\r\nimport type { Particle } from \"../../Particles/particle\";\r\nimport type { IParticleEmitterType } from \"./IParticleEmitterType\";\r\nimport type { UniformBufferEffectCommonAccessor } from \"../../Materials/uniformBufferEffectCommonAccessor\";\r\nimport type { UniformBuffer } from \"../../Materials/uniformBuffer\";\r\n/**\r\n * Particle emitter emitting particles from the inside of a box.\r\n * It emits the particles randomly between 2 given directions.\r\n */\r\nexport class BoxParticleEmitter implements IParticleEmitterType {\r\n /**\r\n * Random direction of each particle after it has been emitted, between direction1 and direction2 vectors.\r\n */\r\n public direction1 = new Vector3(0, 1.0, 0);\r\n /**\r\n * Random direction of each particle after it has been emitted, between direction1 and direction2 vectors.\r\n */\r\n public direction2 = new Vector3(0, 1.0, 0);\r\n\r\n /**\r\n * Minimum box point around our emitter. Our emitter is the center of particles source, but if you want your particles to emit from more than one point, then you can tell it to do so.\r\n */\r\n public minEmitBox = new Vector3(-0.5, -0.5, -0.5);\r\n /**\r\n * Maximum box point around our emitter. Our emitter is the center of particles source, but if you want your particles to emit from more than one point, then you can tell it to do so.\r\n */\r\n public maxEmitBox = new Vector3(0.5, 0.5, 0.5);\r\n\r\n /**\r\n * Creates a new instance BoxParticleEmitter\r\n */\r\n constructor() {}\r\n\r\n /**\r\n * Called by the particle System when the direction is computed for the created particle.\r\n * @param worldMatrix is the world matrix of the particle system\r\n * @param directionToUpdate is the direction vector to update with the result\r\n * @param particle is the particle we are computed the direction for\r\n * @param isLocal defines if the direction should be set in local space\r\n */\r\n public startDirectionFunction(worldMatrix: Matrix, directionToUpdate: Vector3, particle: Particle, isLocal: boolean): void {\r\n const randX = Scalar.RandomRange(this.direction1.x, this.direction2.x);\r\n const randY = Scalar.RandomRange(this.direction1.y, this.direction2.y);\r\n const randZ = Scalar.RandomRange(this.direction1.z, this.direction2.z);\r\n\r\n if (isLocal) {\r\n directionToUpdate.x = randX;\r\n directionToUpdate.y = randY;\r\n directionToUpdate.z = randZ;\r\n return;\r\n }\r\n\r\n Vector3.TransformNormalFromFloatsToRef(randX, randY, randZ, worldMatrix, directionToUpdate);\r\n }\r\n\r\n /**\r\n * Called by the particle System when the position is computed for the created particle.\r\n * @param worldMatrix is the world matrix of the particle system\r\n * @param positionToUpdate is the position vector to update with the result\r\n * @param particle is the particle we are computed the position for\r\n * @param isLocal defines if the position should be set in local space\r\n */\r\n public startPositionFunction(worldMatrix: Matrix, positionToUpdate: Vector3, particle: Particle, isLocal: boolean): void {\r\n const randX = Scalar.RandomRange(this.minEmitBox.x, this.maxEmitBox.x);\r\n const randY = Scalar.RandomRange(this.minEmitBox.y, this.maxEmitBox.y);\r\n const randZ = Scalar.RandomRange(this.minEmitBox.z, this.maxEmitBox.z);\r\n\r\n if (isLocal) {\r\n positionToUpdate.x = randX;\r\n positionToUpdate.y = randY;\r\n positionToUpdate.z = randZ;\r\n return;\r\n }\r\n\r\n Vector3.TransformCoordinatesFromFloatsToRef(randX, randY, randZ, worldMatrix, positionToUpdate);\r\n }\r\n\r\n /**\r\n * Clones the current emitter and returns a copy of it\r\n * @returns the new emitter\r\n */\r\n public clone(): BoxParticleEmitter {\r\n const newOne = new BoxParticleEmitter();\r\n\r\n DeepCopier.DeepCopy(this, newOne);\r\n\r\n return newOne;\r\n }\r\n\r\n /**\r\n * Called by the GPUParticleSystem to setup the update shader\r\n * @param uboOrEffect defines the update shader\r\n */\r\n public applyToShader(uboOrEffect: UniformBufferEffectCommonAccessor): void {\r\n uboOrEffect.setVector3(\"direction1\", this.direction1);\r\n uboOrEffect.setVector3(\"direction2\", this.direction2);\r\n uboOrEffect.setVector3(\"minEmitBox\", this.minEmitBox);\r\n uboOrEffect.setVector3(\"maxEmitBox\", this.maxEmitBox);\r\n }\r\n\r\n /**\r\n * Creates the structure of the ubo for this particle emitter\r\n * @param ubo ubo to create the structure for\r\n */\r\n public buildUniformLayout(ubo: UniformBuffer): void {\r\n ubo.addUniform(\"direction1\", 3);\r\n ubo.addUniform(\"direction2\", 3);\r\n ubo.addUniform(\"minEmitBox\", 3);\r\n ubo.addUniform(\"maxEmitBox\", 3);\r\n }\r\n\r\n /**\r\n * Returns a string to use to update the GPU particles update shader\r\n * @returns a string containing the defines string\r\n */\r\n public getEffectDefines(): string {\r\n return \"#define BOXEMITTER\";\r\n }\r\n\r\n /**\r\n * Returns the string \"BoxParticleEmitter\"\r\n * @returns a string containing the class name\r\n */\r\n public getClassName(): string {\r\n return \"BoxParticleEmitter\";\r\n }\r\n\r\n /**\r\n * Serializes the particle system to a JSON object.\r\n * @returns the JSON object\r\n */\r\n public serialize(): any {\r\n const serializationObject: any = {};\r\n\r\n serializationObject.type = this.getClassName();\r\n serializationObject.direction1 = this.direction1.asArray();\r\n serializationObject.direction2 = this.direction2.asArray();\r\n serializationObject.minEmitBox = this.minEmitBox.asArray();\r\n serializationObject.maxEmitBox = this.maxEmitBox.asArray();\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Parse properties from a JSON object\r\n * @param serializationObject defines the JSON object\r\n */\r\n public parse(serializationObject: any): void {\r\n Vector3.FromArrayToRef(serializationObject.direction1, 0, this.direction1);\r\n Vector3.FromArrayToRef(serializationObject.direction2, 0, this.direction2);\r\n Vector3.FromArrayToRef(serializationObject.minEmitBox, 0, this.minEmitBox);\r\n Vector3.FromArrayToRef(serializationObject.maxEmitBox, 0, this.maxEmitBox);\r\n }\r\n}\r\n","import { DeepCopier } from \"../../Misc/deepCopier\";\r\nimport type { Matrix } from \"../../Maths/math.vector\";\r\nimport { Vector3, TmpVectors } from \"../../Maths/math.vector\";\r\nimport { Scalar } from \"../../Maths/math.scalar\";\r\nimport type { Particle } from \"../../Particles/particle\";\r\nimport type { IParticleEmitterType } from \"./IParticleEmitterType\";\r\nimport type { UniformBufferEffectCommonAccessor } from \"../../Materials/uniformBufferEffectCommonAccessor\";\r\nimport type { UniformBuffer } from \"../../Materials/uniformBuffer\";\r\n/**\r\n * Particle emitter emitting particles from the inside of a cone.\r\n * It emits the particles alongside the cone volume from the base to the particle.\r\n * The emission direction might be randomized.\r\n */\r\nexport class ConeParticleEmitter implements IParticleEmitterType {\r\n private _radius: number;\r\n private _angle: number;\r\n private _height: number;\r\n\r\n /**\r\n * Gets or sets a value indicating where on the radius the start position should be picked (1 = everywhere, 0 = only surface)\r\n */\r\n public radiusRange = 1;\r\n\r\n /**\r\n * Gets or sets a value indicating where on the height the start position should be picked (1 = everywhere, 0 = only surface)\r\n */\r\n public heightRange = 1;\r\n\r\n /**\r\n * Gets or sets a value indicating if all the particles should be emitted from the spawn point only (the base of the cone)\r\n */\r\n public emitFromSpawnPointOnly = false;\r\n\r\n /**\r\n * Gets or sets the radius of the emission cone\r\n */\r\n public get radius(): number {\r\n return this._radius;\r\n }\r\n\r\n public set radius(value: number) {\r\n this._radius = value;\r\n this._buildHeight();\r\n }\r\n\r\n /**\r\n * Gets or sets the angle of the emission cone\r\n */\r\n public get angle(): number {\r\n return this._angle;\r\n }\r\n\r\n public set angle(value: number) {\r\n this._angle = value;\r\n this._buildHeight();\r\n }\r\n\r\n private _buildHeight() {\r\n if (this._angle !== 0) {\r\n this._height = this._radius / Math.tan(this._angle / 2);\r\n } else {\r\n this._height = 1;\r\n }\r\n }\r\n\r\n /**\r\n * Creates a new instance ConeParticleEmitter\r\n * @param radius the radius of the emission cone (1 by default)\r\n * @param angle the cone base angle (PI by default)\r\n * @param directionRandomizer defines how much to randomize the particle direction [0-1] (default is 0)\r\n */\r\n constructor(\r\n radius = 1,\r\n angle = Math.PI,\r\n /** defines how much to randomize the particle direction [0-1] (default is 0) */\r\n public directionRandomizer = 0\r\n ) {\r\n this.angle = angle;\r\n this.radius = radius;\r\n }\r\n\r\n /**\r\n * Called by the particle System when the direction is computed for the created particle.\r\n * @param worldMatrix is the world matrix of the particle system\r\n * @param directionToUpdate is the direction vector to update with the result\r\n * @param particle is the particle we are computed the direction for\r\n * @param isLocal defines if the direction should be set in local space\r\n */\r\n public startDirectionFunction(worldMatrix: Matrix, directionToUpdate: Vector3, particle: Particle, isLocal: boolean): void {\r\n if (isLocal) {\r\n TmpVectors.Vector3[0].copyFrom(particle._localPosition!).normalize();\r\n } else {\r\n particle.position.subtractToRef(worldMatrix.getTranslation(), TmpVectors.Vector3[0]).normalize();\r\n }\r\n\r\n const randX = Scalar.RandomRange(0, this.directionRandomizer);\r\n const randY = Scalar.RandomRange(0, this.directionRandomizer);\r\n const randZ = Scalar.RandomRange(0, this.directionRandomizer);\r\n directionToUpdate.x = TmpVectors.Vector3[0].x + randX;\r\n directionToUpdate.y = TmpVectors.Vector3[0].y + randY;\r\n directionToUpdate.z = TmpVectors.Vector3[0].z + randZ;\r\n directionToUpdate.normalize();\r\n }\r\n\r\n /**\r\n * Called by the particle System when the position is computed for the created particle.\r\n * @param worldMatrix is the world matrix of the particle system\r\n * @param positionToUpdate is the position vector to update with the result\r\n * @param particle is the particle we are computed the position for\r\n * @param isLocal defines if the position should be set in local space\r\n */\r\n startPositionFunction(worldMatrix: Matrix, positionToUpdate: Vector3, particle: Particle, isLocal: boolean): void {\r\n const s = Scalar.RandomRange(0, Math.PI * 2);\r\n let h: number;\r\n\r\n if (!this.emitFromSpawnPointOnly) {\r\n h = Scalar.RandomRange(0, this.heightRange);\r\n // Better distribution in a cone at normal angles.\r\n h = 1 - h * h;\r\n } else {\r\n h = 0.0001;\r\n }\r\n let radius = this._radius - Scalar.RandomRange(0, this._radius * this.radiusRange);\r\n radius = radius * h;\r\n\r\n const randX = radius * Math.sin(s);\r\n const randZ = radius * Math.cos(s);\r\n const randY = h * this._height;\r\n\r\n if (isLocal) {\r\n positionToUpdate.x = randX;\r\n positionToUpdate.y = randY;\r\n positionToUpdate.z = randZ;\r\n return;\r\n }\r\n\r\n Vector3.TransformCoordinatesFromFloatsToRef(randX, randY, randZ, worldMatrix, positionToUpdate);\r\n }\r\n\r\n /**\r\n * Clones the current emitter and returns a copy of it\r\n * @returns the new emitter\r\n */\r\n public clone(): ConeParticleEmitter {\r\n const newOne = new ConeParticleEmitter(this._radius, this._angle, this.directionRandomizer);\r\n\r\n DeepCopier.DeepCopy(this, newOne);\r\n\r\n return newOne;\r\n }\r\n\r\n /**\r\n * Called by the GPUParticleSystem to setup the update shader\r\n * @param uboOrEffect defines the update shader\r\n */\r\n public applyToShader(uboOrEffect: UniformBufferEffectCommonAccessor): void {\r\n uboOrEffect.setFloat2(\"radius\", this._radius, this.radiusRange);\r\n uboOrEffect.setFloat(\"coneAngle\", this._angle);\r\n uboOrEffect.setFloat2(\"height\", this._height, this.heightRange);\r\n uboOrEffect.setFloat(\"directionRandomizer\", this.directionRandomizer);\r\n }\r\n\r\n /**\r\n * Creates the structure of the ubo for this particle emitter\r\n * @param ubo ubo to create the structure for\r\n */\r\n public buildUniformLayout(ubo: UniformBuffer): void {\r\n ubo.addUniform(\"radius\", 2);\r\n ubo.addUniform(\"coneAngle\", 1);\r\n ubo.addUniform(\"height\", 2);\r\n ubo.addUniform(\"directionRandomizer\", 1);\r\n }\r\n\r\n /**\r\n * Returns a string to use to update the GPU particles update shader\r\n * @returns a string containing the defines string\r\n */\r\n public getEffectDefines(): string {\r\n let defines = \"#define CONEEMITTER\";\r\n\r\n if (this.emitFromSpawnPointOnly) {\r\n defines += \"\\n#define CONEEMITTERSPAWNPOINT\";\r\n }\r\n\r\n return defines;\r\n }\r\n\r\n /**\r\n * Returns the string \"ConeParticleEmitter\"\r\n * @returns a string containing the class name\r\n */\r\n public getClassName(): string {\r\n return \"ConeParticleEmitter\";\r\n }\r\n\r\n /**\r\n * Serializes the particle system to a JSON object.\r\n * @returns the JSON object\r\n */\r\n public serialize(): any {\r\n const serializationObject: any = {};\r\n\r\n serializationObject.type = this.getClassName();\r\n serializationObject.radius = this._radius;\r\n serializationObject.angle = this._angle;\r\n serializationObject.directionRandomizer = this.directionRandomizer;\r\n serializationObject.radiusRange = this.radiusRange;\r\n serializationObject.heightRange = this.heightRange;\r\n serializationObject.emitFromSpawnPointOnly = this.emitFromSpawnPointOnly;\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Parse properties from a JSON object\r\n * @param serializationObject defines the JSON object\r\n */\r\n public parse(serializationObject: any): void {\r\n this.radius = serializationObject.radius;\r\n this.angle = serializationObject.angle;\r\n this.directionRandomizer = serializationObject.directionRandomizer;\r\n\r\n this.radiusRange = serializationObject.radiusRange !== undefined ? serializationObject.radiusRange : 1;\r\n this.heightRange = serializationObject.radiusRange !== undefined ? serializationObject.heightRange : 1;\r\n this.emitFromSpawnPointOnly = serializationObject.emitFromSpawnPointOnly !== undefined ? serializationObject.emitFromSpawnPointOnly : false;\r\n }\r\n}\r\n","import type { Matrix } from \"../../Maths/math.vector\";\r\nimport { Vector3 } from \"../../Maths/math.vector\";\r\nimport { Scalar } from \"../../Maths/math.scalar\";\r\nimport type { Particle } from \"../../Particles/particle\";\r\nimport type { IParticleEmitterType } from \"./IParticleEmitterType\";\r\nimport { DeepCopier } from \"../../Misc/deepCopier\";\r\nimport type { UniformBufferEffectCommonAccessor } from \"../../Materials/uniformBufferEffectCommonAccessor\";\r\nimport type { UniformBuffer } from \"../../Materials/uniformBuffer\";\r\n/**\r\n * Particle emitter emitting particles from the inside of a cylinder.\r\n * It emits the particles alongside the cylinder radius. The emission direction might be randomized.\r\n */\r\nexport class CylinderParticleEmitter implements IParticleEmitterType {\r\n private _tempVector = Vector3.Zero();\r\n\r\n /**\r\n * Creates a new instance CylinderParticleEmitter\r\n * @param radius the radius of the emission cylinder (1 by default)\r\n * @param height the height of the emission cylinder (1 by default)\r\n * @param radiusRange the range of the emission cylinder [0-1] 0 Surface only, 1 Entire Radius (1 by default)\r\n * @param directionRandomizer defines how much to randomize the particle direction [0-1]\r\n */\r\n constructor(\r\n /**\r\n * The radius of the emission cylinder.\r\n */\r\n public radius = 1,\r\n /**\r\n * The height of the emission cylinder.\r\n */\r\n public height = 1,\r\n /**\r\n * The range of emission [0-1] 0 Surface only, 1 Entire Radius.\r\n */\r\n public radiusRange = 1,\r\n /**\r\n * How much to randomize the particle direction [0-1].\r\n */\r\n public directionRandomizer = 0\r\n ) {}\r\n\r\n /**\r\n * Called by the particle System when the direction is computed for the created particle.\r\n * @param worldMatrix is the world matrix of the particle system\r\n * @param directionToUpdate is the direction vector to update with the result\r\n * @param particle is the particle we are computed the direction for\r\n * @param isLocal defines if the direction should be set in local space\r\n * @param inverseWorldMatrix defines the inverted world matrix to use if isLocal is false\r\n */\r\n public startDirectionFunction(worldMatrix: Matrix, directionToUpdate: Vector3, particle: Particle, isLocal: boolean, inverseWorldMatrix: Matrix): void {\r\n particle.position.subtractToRef(worldMatrix.getTranslation(), this._tempVector);\r\n\r\n this._tempVector.normalize();\r\n\r\n Vector3.TransformNormalToRef(this._tempVector, inverseWorldMatrix, this._tempVector);\r\n\r\n const randY = Scalar.RandomRange(-this.directionRandomizer / 2, this.directionRandomizer / 2);\r\n\r\n let angle = Math.atan2(this._tempVector.x, this._tempVector.z);\r\n angle += Scalar.RandomRange(-Math.PI / 2, Math.PI / 2) * this.directionRandomizer;\r\n\r\n this._tempVector.y = randY; // set direction y to rand y to mirror normal of cylinder surface\r\n this._tempVector.x = Math.sin(angle);\r\n this._tempVector.z = Math.cos(angle);\r\n this._tempVector.normalize();\r\n\r\n if (isLocal) {\r\n directionToUpdate.copyFrom(this._tempVector);\r\n return;\r\n }\r\n\r\n Vector3.TransformNormalFromFloatsToRef(this._tempVector.x, this._tempVector.y, this._tempVector.z, worldMatrix, directionToUpdate);\r\n }\r\n\r\n /**\r\n * Called by the particle System when the position is computed for the created particle.\r\n * @param worldMatrix is the world matrix of the particle system\r\n * @param positionToUpdate is the position vector to update with the result\r\n * @param particle is the particle we are computed the position for\r\n * @param isLocal defines if the position should be set in local space\r\n */\r\n public startPositionFunction(worldMatrix: Matrix, positionToUpdate: Vector3, particle: Particle, isLocal: boolean): void {\r\n const yPos = Scalar.RandomRange(-this.height / 2, this.height / 2);\r\n const angle = Scalar.RandomRange(0, 2 * Math.PI);\r\n\r\n // Pick a properly distributed point within the circle https://programming.guide/random-point-within-circle.html\r\n const radiusDistribution = Scalar.RandomRange((1 - this.radiusRange) * (1 - this.radiusRange), 1);\r\n const positionRadius = Math.sqrt(radiusDistribution) * this.radius;\r\n const xPos = positionRadius * Math.cos(angle);\r\n const zPos = positionRadius * Math.sin(angle);\r\n\r\n if (isLocal) {\r\n positionToUpdate.copyFromFloats(xPos, yPos, zPos);\r\n return;\r\n }\r\n\r\n Vector3.TransformCoordinatesFromFloatsToRef(xPos, yPos, zPos, worldMatrix, positionToUpdate);\r\n }\r\n\r\n /**\r\n * Clones the current emitter and returns a copy of it\r\n * @returns the new emitter\r\n */\r\n public clone(): CylinderParticleEmitter {\r\n const newOne = new CylinderParticleEmitter(this.radius, this.directionRandomizer);\r\n\r\n DeepCopier.DeepCopy(this, newOne);\r\n\r\n return newOne;\r\n }\r\n\r\n /**\r\n * Called by the GPUParticleSystem to setup the update shader\r\n * @param uboOrEffect defines the update shader\r\n */\r\n public applyToShader(uboOrEffect: UniformBufferEffectCommonAccessor): void {\r\n uboOrEffect.setFloat(\"radius\", this.radius);\r\n uboOrEffect.setFloat(\"height\", this.height);\r\n uboOrEffect.setFloat(\"radiusRange\", this.radiusRange);\r\n uboOrEffect.setFloat(\"directionRandomizer\", this.directionRandomizer);\r\n }\r\n\r\n /**\r\n * Creates the structure of the ubo for this particle emitter\r\n * @param ubo ubo to create the structure for\r\n */\r\n public buildUniformLayout(ubo: UniformBuffer): void {\r\n ubo.addUniform(\"radius\", 1);\r\n ubo.addUniform(\"height\", 1);\r\n ubo.addUniform(\"radiusRange\", 1);\r\n ubo.addUniform(\"directionRandomizer\", 1);\r\n }\r\n\r\n /**\r\n * Returns a string to use to update the GPU particles update shader\r\n * @returns a string containing the defines string\r\n */\r\n public getEffectDefines(): string {\r\n return \"#define CYLINDEREMITTER\";\r\n }\r\n\r\n /**\r\n * Returns the string \"CylinderParticleEmitter\"\r\n * @returns a string containing the class name\r\n */\r\n public getClassName(): string {\r\n return \"CylinderParticleEmitter\";\r\n }\r\n\r\n /**\r\n * Serializes the particle system to a JSON object.\r\n * @returns the JSON object\r\n */\r\n public serialize(): any {\r\n const serializationObject: any = {};\r\n serializationObject.type = this.getClassName();\r\n serializationObject.radius = this.radius;\r\n serializationObject.height = this.height;\r\n serializationObject.radiusRange = this.radiusRange;\r\n serializationObject.directionRandomizer = this.directionRandomizer;\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Parse properties from a JSON object\r\n * @param serializationObject defines the JSON object\r\n */\r\n public parse(serializationObject: any): void {\r\n this.radius = serializationObject.radius;\r\n this.height = serializationObject.height;\r\n this.radiusRange = serializationObject.radiusRange;\r\n this.directionRandomizer = serializationObject.directionRandomizer;\r\n }\r\n}\r\n\r\n/**\r\n * Particle emitter emitting particles from the inside of a cylinder.\r\n * It emits the particles randomly between two vectors.\r\n */\r\nexport class CylinderDirectedParticleEmitter extends CylinderParticleEmitter {\r\n /**\r\n * Creates a new instance CylinderDirectedParticleEmitter\r\n * @param radius the radius of the emission cylinder (1 by default)\r\n * @param height the height of the emission cylinder (1 by default)\r\n * @param radiusRange the range of the emission cylinder [0-1] 0 Surface only, 1 Entire Radius (1 by default)\r\n * @param direction1 the min limit of the emission direction (up vector by default)\r\n * @param direction2 the max limit of the emission direction (up vector by default)\r\n */\r\n constructor(\r\n radius = 1,\r\n height = 1,\r\n radiusRange = 1,\r\n /**\r\n * The min limit of the emission direction.\r\n */\r\n public direction1 = new Vector3(0, 1, 0),\r\n /**\r\n * The max limit of the emission direction.\r\n */\r\n public direction2 = new Vector3(0, 1, 0)\r\n ) {\r\n super(radius, height, radiusRange);\r\n }\r\n\r\n /**\r\n * Called by the particle System when the direction is computed for the created particle.\r\n * @param worldMatrix is the world matrix of the particle system\r\n * @param directionToUpdate is the direction vector to update with the result\r\n */\r\n public startDirectionFunction(worldMatrix: Matrix, directionToUpdate: Vector3): void {\r\n const randX = Scalar.RandomRange(this.direction1.x, this.direction2.x);\r\n const randY = Scalar.RandomRange(this.direction1.y, this.direction2.y);\r\n const randZ = Scalar.RandomRange(this.direction1.z, this.direction2.z);\r\n Vector3.TransformNormalFromFloatsToRef(randX, randY, randZ, worldMatrix, directionToUpdate);\r\n }\r\n\r\n /**\r\n * Clones the current emitter and returns a copy of it\r\n * @returns the new emitter\r\n */\r\n public clone(): CylinderDirectedParticleEmitter {\r\n const newOne = new CylinderDirectedParticleEmitter(this.radius, this.height, this.radiusRange, this.direction1, this.direction2);\r\n\r\n DeepCopier.DeepCopy(this, newOne);\r\n\r\n return newOne;\r\n }\r\n\r\n /**\r\n * Called by the GPUParticleSystem to setup the update shader\r\n * @param uboOrEffect defines the update shader\r\n */\r\n public applyToShader(uboOrEffect: UniformBufferEffectCommonAccessor): void {\r\n uboOrEffect.setFloat(\"radius\", this.radius);\r\n uboOrEffect.setFloat(\"height\", this.height);\r\n uboOrEffect.setFloat(\"radiusRange\", this.radiusRange);\r\n uboOrEffect.setVector3(\"direction1\", this.direction1);\r\n uboOrEffect.setVector3(\"direction2\", this.direction2);\r\n }\r\n\r\n /**\r\n * Creates the structure of the ubo for this particle emitter\r\n * @param ubo ubo to create the structure for\r\n */\r\n public buildUniformLayout(ubo: UniformBuffer): void {\r\n ubo.addUniform(\"radius\", 1);\r\n ubo.addUniform(\"height\", 1);\r\n ubo.addUniform(\"radiusRange\", 1);\r\n ubo.addUniform(\"direction1\", 3);\r\n ubo.addUniform(\"direction2\", 3);\r\n }\r\n\r\n /**\r\n * Returns a string to use to update the GPU particles update shader\r\n * @returns a string containing the defines string\r\n */\r\n public getEffectDefines(): string {\r\n return \"#define CYLINDEREMITTER\\n#define DIRECTEDCYLINDEREMITTER\";\r\n }\r\n\r\n /**\r\n * Returns the string \"CylinderDirectedParticleEmitter\"\r\n * @returns a string containing the class name\r\n */\r\n public getClassName(): string {\r\n return \"CylinderDirectedParticleEmitter\";\r\n }\r\n\r\n /**\r\n * Serializes the particle system to a JSON object.\r\n * @returns the JSON object\r\n */\r\n public serialize(): any {\r\n const serializationObject = super.serialize();\r\n\r\n serializationObject.direction1 = this.direction1.asArray();\r\n serializationObject.direction2 = this.direction2.asArray();\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Parse properties from a JSON object\r\n * @param serializationObject defines the JSON object\r\n */\r\n public parse(serializationObject: any): void {\r\n super.parse(serializationObject);\r\n this.direction1.copyFrom(serializationObject.direction1);\r\n this.direction2.copyFrom(serializationObject.direction2);\r\n }\r\n}\r\n","import { DeepCopier } from \"../../Misc/deepCopier\";\r\nimport type { Matrix } from \"../../Maths/math.vector\";\r\nimport { Vector3 } from \"../../Maths/math.vector\";\r\nimport { Scalar } from \"../../Maths/math.scalar\";\r\nimport type { Particle } from \"../../Particles/particle\";\r\nimport type { IParticleEmitterType } from \"./IParticleEmitterType\";\r\nimport type { UniformBufferEffectCommonAccessor } from \"../../Materials/uniformBufferEffectCommonAccessor\";\r\nimport type { UniformBuffer } from \"../../Materials/uniformBuffer\";\r\n/**\r\n * Particle emitter emitting particles from the inside of a hemisphere.\r\n * It emits the particles alongside the hemisphere radius. The emission direction might be randomized.\r\n */\r\nexport class HemisphericParticleEmitter implements IParticleEmitterType {\r\n /**\r\n * Creates a new instance HemisphericParticleEmitter\r\n * @param radius the radius of the emission hemisphere (1 by default)\r\n * @param radiusRange the range of the emission hemisphere [0-1] 0 Surface only, 1 Entire Radius (1 by default)\r\n * @param directionRandomizer defines how much to randomize the particle direction [0-1]\r\n */\r\n constructor(\r\n /**\r\n * The radius of the emission hemisphere.\r\n */\r\n public radius = 1,\r\n /**\r\n * The range of emission [0-1] 0 Surface only, 1 Entire Radius.\r\n */\r\n public radiusRange = 1,\r\n /**\r\n * How much to randomize the particle direction [0-1].\r\n */\r\n public directionRandomizer = 0\r\n ) {}\r\n\r\n /**\r\n * Called by the particle System when the direction is computed for the created particle.\r\n * @param worldMatrix is the world matrix of the particle system\r\n * @param directionToUpdate is the direction vector to update with the result\r\n * @param particle is the particle we are computed the direction for\r\n * @param isLocal defines if the direction should be set in local space\r\n */\r\n public startDirectionFunction(worldMatrix: Matrix, directionToUpdate: Vector3, particle: Particle, isLocal: boolean): void {\r\n const direction = particle.position.subtract(worldMatrix.getTranslation()).normalize();\r\n const randX = Scalar.RandomRange(0, this.directionRandomizer);\r\n const randY = Scalar.RandomRange(0, this.directionRandomizer);\r\n const randZ = Scalar.RandomRange(0, this.directionRandomizer);\r\n direction.x += randX;\r\n direction.y += randY;\r\n direction.z += randZ;\r\n direction.normalize();\r\n\r\n if (isLocal) {\r\n directionToUpdate.copyFrom(direction);\r\n return;\r\n }\r\n\r\n Vector3.TransformNormalFromFloatsToRef(direction.x, direction.y, direction.z, worldMatrix, directionToUpdate);\r\n }\r\n\r\n /**\r\n * Called by the particle System when the position is computed for the created particle.\r\n * @param worldMatrix is the world matrix of the particle system\r\n * @param positionToUpdate is the position vector to update with the result\r\n * @param particle is the particle we are computed the position for\r\n * @param isLocal defines if the position should be set in local space\r\n */\r\n public startPositionFunction(worldMatrix: Matrix, positionToUpdate: Vector3, particle: Particle, isLocal: boolean): void {\r\n const randRadius = this.radius - Scalar.RandomRange(0, this.radius * this.radiusRange);\r\n const v = Scalar.RandomRange(0, 1.0);\r\n const phi = Scalar.RandomRange(0, 2 * Math.PI);\r\n const theta = Math.acos(2 * v - 1);\r\n const randX = randRadius * Math.cos(phi) * Math.sin(theta);\r\n const randY = randRadius * Math.cos(theta);\r\n const randZ = randRadius * Math.sin(phi) * Math.sin(theta);\r\n\r\n if (isLocal) {\r\n positionToUpdate.copyFromFloats(randX, Math.abs(randY), randZ);\r\n return;\r\n }\r\n\r\n Vector3.TransformCoordinatesFromFloatsToRef(randX, Math.abs(randY), randZ, worldMatrix, positionToUpdate);\r\n }\r\n\r\n /**\r\n * Clones the current emitter and returns a copy of it\r\n * @returns the new emitter\r\n */\r\n public clone(): HemisphericParticleEmitter {\r\n const newOne = new HemisphericParticleEmitter(this.radius, this.directionRandomizer);\r\n\r\n DeepCopier.DeepCopy(this, newOne);\r\n\r\n return newOne;\r\n }\r\n\r\n /**\r\n * Called by the GPUParticleSystem to setup the update shader\r\n * @param uboOrEffect defines the update shader\r\n */\r\n public applyToShader(uboOrEffect: UniformBufferEffectCommonAccessor): void {\r\n uboOrEffect.setFloat(\"radius\", this.radius);\r\n uboOrEffect.setFloat(\"radiusRange\", this.radiusRange);\r\n uboOrEffect.setFloat(\"directionRandomizer\", this.directionRandomizer);\r\n }\r\n\r\n /**\r\n * Creates the structure of the ubo for this particle emitter\r\n * @param ubo ubo to create the structure for\r\n */\r\n public buildUniformLayout(ubo: UniformBuffer): void {\r\n ubo.addUniform(\"radius\", 1);\r\n ubo.addUniform(\"radiusRange\", 1);\r\n ubo.addUniform(\"directionRandomizer\", 1);\r\n }\r\n\r\n /**\r\n * Returns a string to use to update the GPU particles update shader\r\n * @returns a string containing the defines string\r\n */\r\n public getEffectDefines(): string {\r\n return \"#define HEMISPHERICEMITTER\";\r\n }\r\n\r\n /**\r\n * Returns the string \"HemisphericParticleEmitter\"\r\n * @returns a string containing the class name\r\n */\r\n public getClassName(): string {\r\n return \"HemisphericParticleEmitter\";\r\n }\r\n\r\n /**\r\n * Serializes the particle system to a JSON object.\r\n * @returns the JSON object\r\n */\r\n public serialize(): any {\r\n const serializationObject: any = {};\r\n serializationObject.type = this.getClassName();\r\n serializationObject.radius = this.radius;\r\n serializationObject.radiusRange = this.radiusRange;\r\n serializationObject.directionRandomizer = this.directionRandomizer;\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Parse properties from a JSON object\r\n * @param serializationObject defines the JSON object\r\n */\r\n public parse(serializationObject: any): void {\r\n this.radius = serializationObject.radius;\r\n this.radiusRange = serializationObject.radiusRange;\r\n this.directionRandomizer = serializationObject.directionRandomizer;\r\n }\r\n}\r\n","import { DeepCopier } from \"../../Misc/deepCopier\";\r\nimport type { Matrix } from \"../../Maths/math.vector\";\r\nimport { Vector3 } from \"../../Maths/math.vector\";\r\nimport { Scalar } from \"../../Maths/math.scalar\";\r\nimport type { Particle } from \"../../Particles/particle\";\r\nimport type { IParticleEmitterType } from \"./IParticleEmitterType\";\r\nimport type { UniformBufferEffectCommonAccessor } from \"../../Materials/uniformBufferEffectCommonAccessor\";\r\nimport type { UniformBuffer } from \"../../Materials/uniformBuffer\";\r\n/**\r\n * Particle emitter emitting particles from a point.\r\n * It emits the particles randomly between 2 given directions.\r\n */\r\nexport class PointParticleEmitter implements IParticleEmitterType {\r\n /**\r\n * Random direction of each particle after it has been emitted, between direction1 and direction2 vectors.\r\n */\r\n public direction1 = new Vector3(0, 1.0, 0);\r\n /**\r\n * Random direction of each particle after it has been emitted, between direction1 and direction2 vectors.\r\n */\r\n public direction2 = new Vector3(0, 1.0, 0);\r\n\r\n /**\r\n * Creates a new instance PointParticleEmitter\r\n */\r\n constructor() {}\r\n\r\n /**\r\n * Called by the particle System when the direction is computed for the created particle.\r\n * @param worldMatrix is the world matrix of the particle system\r\n * @param directionToUpdate is the direction vector to update with the result\r\n * @param particle is the particle we are computed the direction for\r\n * @param isLocal defines if the direction should be set in local space\r\n */\r\n public startDirectionFunction(worldMatrix: Matrix, directionToUpdate: Vector3, particle: Particle, isLocal: boolean): void {\r\n const randX = Scalar.RandomRange(this.direction1.x, this.direction2.x);\r\n const randY = Scalar.RandomRange(this.direction1.y, this.direction2.y);\r\n const randZ = Scalar.RandomRange(this.direction1.z, this.direction2.z);\r\n\r\n if (isLocal) {\r\n directionToUpdate.copyFromFloats(randX, randY, randZ);\r\n return;\r\n }\r\n\r\n Vector3.TransformNormalFromFloatsToRef(randX, randY, randZ, worldMatrix, directionToUpdate);\r\n }\r\n\r\n /**\r\n * Called by the particle System when the position is computed for the created particle.\r\n * @param worldMatrix is the world matrix of the particle system\r\n * @param positionToUpdate is the position vector to update with the result\r\n * @param particle is the particle we are computed the position for\r\n * @param isLocal defines if the position should be set in local space\r\n */\r\n public startPositionFunction(worldMatrix: Matrix, positionToUpdate: Vector3, particle: Particle, isLocal: boolean): void {\r\n if (isLocal) {\r\n positionToUpdate.copyFromFloats(0, 0, 0);\r\n return;\r\n }\r\n Vector3.TransformCoordinatesFromFloatsToRef(0, 0, 0, worldMatrix, positionToUpdate);\r\n }\r\n\r\n /**\r\n * Clones the current emitter and returns a copy of it\r\n * @returns the new emitter\r\n */\r\n public clone(): PointParticleEmitter {\r\n const newOne = new PointParticleEmitter();\r\n\r\n DeepCopier.DeepCopy(this, newOne);\r\n\r\n return newOne;\r\n }\r\n\r\n /**\r\n * Called by the GPUParticleSystem to setup the update shader\r\n * @param uboOrEffect defines the update shader\r\n */\r\n public applyToShader(uboOrEffect: UniformBufferEffectCommonAccessor): void {\r\n uboOrEffect.setVector3(\"direction1\", this.direction1);\r\n uboOrEffect.setVector3(\"direction2\", this.direction2);\r\n }\r\n\r\n /**\r\n * Creates the structure of the ubo for this particle emitter\r\n * @param ubo ubo to create the structure for\r\n */\r\n public buildUniformLayout(ubo: UniformBuffer): void {\r\n ubo.addUniform(\"direction1\", 3);\r\n ubo.addUniform(\"direction2\", 3);\r\n }\r\n\r\n /**\r\n * Returns a string to use to update the GPU particles update shader\r\n * @returns a string containing the defines string\r\n */\r\n public getEffectDefines(): string {\r\n return \"#define POINTEMITTER\";\r\n }\r\n\r\n /**\r\n * Returns the string \"PointParticleEmitter\"\r\n * @returns a string containing the class name\r\n */\r\n public getClassName(): string {\r\n return \"PointParticleEmitter\";\r\n }\r\n\r\n /**\r\n * Serializes the particle system to a JSON object.\r\n * @returns the JSON object\r\n */\r\n public serialize(): any {\r\n const serializationObject: any = {};\r\n\r\n serializationObject.type = this.getClassName();\r\n serializationObject.direction1 = this.direction1.asArray();\r\n serializationObject.direction2 = this.direction2.asArray();\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Parse properties from a JSON object\r\n * @param serializationObject defines the JSON object\r\n */\r\n public parse(serializationObject: any): void {\r\n Vector3.FromArrayToRef(serializationObject.direction1, 0, this.direction1);\r\n Vector3.FromArrayToRef(serializationObject.direction2, 0, this.direction2);\r\n }\r\n}\r\n","import type { Matrix } from \"../../Maths/math.vector\";\r\nimport { Vector3 } from \"../../Maths/math.vector\";\r\nimport { Scalar } from \"../../Maths/math.scalar\";\r\nimport type { Particle } from \"../../Particles/particle\";\r\nimport type { IParticleEmitterType } from \"./IParticleEmitterType\";\r\nimport { DeepCopier } from \"../../Misc/deepCopier\";\r\nimport type { UniformBufferEffectCommonAccessor } from \"../../Materials/uniformBufferEffectCommonAccessor\";\r\nimport type { UniformBuffer } from \"../../Materials/uniformBuffer\";\r\n/**\r\n * Particle emitter emitting particles from the inside of a sphere.\r\n * It emits the particles alongside the sphere radius. The emission direction might be randomized.\r\n */\r\nexport class SphereParticleEmitter implements IParticleEmitterType {\r\n /**\r\n * Creates a new instance SphereParticleEmitter\r\n * @param radius the radius of the emission sphere (1 by default)\r\n * @param radiusRange the range of the emission sphere [0-1] 0 Surface only, 1 Entire Radius (1 by default)\r\n * @param directionRandomizer defines how much to randomize the particle direction [0-1]\r\n */\r\n constructor(\r\n /**\r\n * The radius of the emission sphere.\r\n */\r\n public radius = 1,\r\n /**\r\n * The range of emission [0-1] 0 Surface only, 1 Entire Radius.\r\n */\r\n public radiusRange = 1,\r\n /**\r\n * How much to randomize the particle direction [0-1].\r\n */\r\n public directionRandomizer = 0\r\n ) {}\r\n\r\n /**\r\n * Called by the particle System when the direction is computed for the created particle.\r\n * @param worldMatrix is the world matrix of the particle system\r\n * @param directionToUpdate is the direction vector to update with the result\r\n * @param particle is the particle we are computed the direction for\r\n * @param isLocal defines if the direction should be set in local space\r\n */\r\n public startDirectionFunction(worldMatrix: Matrix, directionToUpdate: Vector3, particle: Particle, isLocal: boolean): void {\r\n const direction = particle.position.subtract(worldMatrix.getTranslation()).normalize();\r\n const randX = Scalar.RandomRange(0, this.directionRandomizer);\r\n const randY = Scalar.RandomRange(0, this.directionRandomizer);\r\n const randZ = Scalar.RandomRange(0, this.directionRandomizer);\r\n direction.x += randX;\r\n direction.y += randY;\r\n direction.z += randZ;\r\n direction.normalize();\r\n\r\n if (isLocal) {\r\n directionToUpdate.copyFrom(direction);\r\n return;\r\n }\r\n\r\n Vector3.TransformNormalFromFloatsToRef(direction.x, direction.y, direction.z, worldMatrix, directionToUpdate);\r\n }\r\n\r\n /**\r\n * Called by the particle System when the position is computed for the created particle.\r\n * @param worldMatrix is the world matrix of the particle system\r\n * @param positionToUpdate is the position vector to update with the result\r\n * @param particle is the particle we are computed the position for\r\n * @param isLocal defines if the position should be set in local space\r\n */\r\n public startPositionFunction(worldMatrix: Matrix, positionToUpdate: Vector3, particle: Particle, isLocal: boolean): void {\r\n const randRadius = this.radius - Scalar.RandomRange(0, this.radius * this.radiusRange);\r\n const v = Scalar.RandomRange(0, 1.0);\r\n const phi = Scalar.RandomRange(0, 2 * Math.PI);\r\n const theta = Math.acos(2 * v - 1);\r\n const randX = randRadius * Math.cos(phi) * Math.sin(theta);\r\n const randY = randRadius * Math.cos(theta);\r\n const randZ = randRadius * Math.sin(phi) * Math.sin(theta);\r\n\r\n if (isLocal) {\r\n positionToUpdate.copyFromFloats(randX, randY, randZ);\r\n return;\r\n }\r\n\r\n Vector3.TransformCoordinatesFromFloatsToRef(randX, randY, randZ, worldMatrix, positionToUpdate);\r\n }\r\n\r\n /**\r\n * Clones the current emitter and returns a copy of it\r\n * @returns the new emitter\r\n */\r\n public clone(): SphereParticleEmitter {\r\n const newOne = new SphereParticleEmitter(this.radius, this.directionRandomizer);\r\n\r\n DeepCopier.DeepCopy(this, newOne);\r\n\r\n return newOne;\r\n }\r\n\r\n /**\r\n * Called by the GPUParticleSystem to setup the update shader\r\n * @param uboOrEffect defines the update shader\r\n */\r\n public applyToShader(uboOrEffect: UniformBufferEffectCommonAccessor): void {\r\n uboOrEffect.setFloat(\"radius\", this.radius);\r\n uboOrEffect.setFloat(\"radiusRange\", this.radiusRange);\r\n uboOrEffect.setFloat(\"directionRandomizer\", this.directionRandomizer);\r\n }\r\n\r\n /**\r\n * Creates the structure of the ubo for this particle emitter\r\n * @param ubo ubo to create the structure for\r\n */\r\n public buildUniformLayout(ubo: UniformBuffer): void {\r\n ubo.addUniform(\"radius\", 1);\r\n ubo.addUniform(\"radiusRange\", 1);\r\n ubo.addUniform(\"directionRandomizer\", 1);\r\n }\r\n\r\n /**\r\n * Returns a string to use to update the GPU particles update shader\r\n * @returns a string containing the defines string\r\n */\r\n public getEffectDefines(): string {\r\n return \"#define SPHEREEMITTER\";\r\n }\r\n\r\n /**\r\n * Returns the string \"SphereParticleEmitter\"\r\n * @returns a string containing the class name\r\n */\r\n public getClassName(): string {\r\n return \"SphereParticleEmitter\";\r\n }\r\n\r\n /**\r\n * Serializes the particle system to a JSON object.\r\n * @returns the JSON object\r\n */\r\n public serialize(): any {\r\n const serializationObject: any = {};\r\n serializationObject.type = this.getClassName();\r\n serializationObject.radius = this.radius;\r\n serializationObject.radiusRange = this.radiusRange;\r\n serializationObject.directionRandomizer = this.directionRandomizer;\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Parse properties from a JSON object\r\n * @param serializationObject defines the JSON object\r\n */\r\n public parse(serializationObject: any): void {\r\n this.radius = serializationObject.radius;\r\n this.radiusRange = serializationObject.radiusRange;\r\n this.directionRandomizer = serializationObject.directionRandomizer;\r\n }\r\n}\r\n\r\n/**\r\n * Particle emitter emitting particles from the inside of a sphere.\r\n * It emits the particles randomly between two vectors.\r\n */\r\nexport class SphereDirectedParticleEmitter extends SphereParticleEmitter {\r\n /**\r\n * Creates a new instance SphereDirectedParticleEmitter\r\n * @param radius the radius of the emission sphere (1 by default)\r\n * @param direction1 the min limit of the emission direction (up vector by default)\r\n * @param direction2 the max limit of the emission direction (up vector by default)\r\n */\r\n constructor(\r\n radius = 1,\r\n /**\r\n * The min limit of the emission direction.\r\n */\r\n public direction1 = new Vector3(0, 1, 0),\r\n /**\r\n * The max limit of the emission direction.\r\n */\r\n public direction2 = new Vector3(0, 1, 0)\r\n ) {\r\n super(radius);\r\n }\r\n\r\n /**\r\n * Called by the particle System when the direction is computed for the created particle.\r\n * @param worldMatrix is the world matrix of the particle system\r\n * @param directionToUpdate is the direction vector to update with the result\r\n */\r\n public startDirectionFunction(worldMatrix: Matrix, directionToUpdate: Vector3): void {\r\n const randX = Scalar.RandomRange(this.direction1.x, this.direction2.x);\r\n const randY = Scalar.RandomRange(this.direction1.y, this.direction2.y);\r\n const randZ = Scalar.RandomRange(this.direction1.z, this.direction2.z);\r\n Vector3.TransformNormalFromFloatsToRef(randX, randY, randZ, worldMatrix, directionToUpdate);\r\n }\r\n\r\n /**\r\n * Clones the current emitter and returns a copy of it\r\n * @returns the new emitter\r\n */\r\n public clone(): SphereDirectedParticleEmitter {\r\n const newOne = new SphereDirectedParticleEmitter(this.radius, this.direction1, this.direction2);\r\n\r\n DeepCopier.DeepCopy(this, newOne);\r\n\r\n return newOne;\r\n }\r\n\r\n /**\r\n * Called by the GPUParticleSystem to setup the update shader\r\n * @param uboOrEffect defines the update shader\r\n */\r\n public applyToShader(uboOrEffect: UniformBufferEffectCommonAccessor): void {\r\n uboOrEffect.setFloat(\"radius\", this.radius);\r\n uboOrEffect.setFloat(\"radiusRange\", this.radiusRange);\r\n uboOrEffect.setVector3(\"direction1\", this.direction1);\r\n uboOrEffect.setVector3(\"direction2\", this.direction2);\r\n }\r\n\r\n /**\r\n * Creates the structure of the ubo for this particle emitter\r\n * @param ubo ubo to create the structure for\r\n */\r\n public buildUniformLayout(ubo: UniformBuffer): void {\r\n ubo.addUniform(\"radius\", 1);\r\n ubo.addUniform(\"radiusRange\", 1);\r\n ubo.addUniform(\"direction1\", 3);\r\n ubo.addUniform(\"direction2\", 3);\r\n }\r\n\r\n /**\r\n * Returns a string to use to update the GPU particles update shader\r\n * @returns a string containing the defines string\r\n */\r\n public getEffectDefines(): string {\r\n return \"#define SPHEREEMITTER\\n#define DIRECTEDSPHEREEMITTER\";\r\n }\r\n\r\n /**\r\n * Returns the string \"SphereDirectedParticleEmitter\"\r\n * @returns a string containing the class name\r\n */\r\n public getClassName(): string {\r\n return \"SphereDirectedParticleEmitter\";\r\n }\r\n\r\n /**\r\n * Serializes the particle system to a JSON object.\r\n * @returns the JSON object\r\n */\r\n public serialize(): any {\r\n const serializationObject = super.serialize();\r\n\r\n serializationObject.direction1 = this.direction1.asArray();\r\n serializationObject.direction2 = this.direction2.asArray();\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Parse properties from a JSON object\r\n * @param serializationObject defines the JSON object\r\n */\r\n public parse(serializationObject: any): void {\r\n super.parse(serializationObject);\r\n this.direction1.copyFrom(serializationObject.direction1);\r\n this.direction2.copyFrom(serializationObject.direction2);\r\n }\r\n}\r\n","import { DeepCopier } from \"../../Misc/deepCopier\";\r\nimport type { Matrix } from \"../../Maths/math.vector\";\r\nimport { Vector3, TmpVectors } from \"../../Maths/math.vector\";\r\nimport { Scalar } from \"../../Maths/math.scalar\";\r\nimport type { Particle } from \"../../Particles/particle\";\r\nimport type { IParticleEmitterType } from \"./IParticleEmitterType\";\r\nimport type { IndicesArray, Nullable, FloatArray } from \"../../types\";\r\nimport { VertexBuffer } from \"../../Buffers/buffer\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport type { UniformBufferEffectCommonAccessor } from \"../../Materials/uniformBufferEffectCommonAccessor\";\r\nimport type { UniformBuffer } from \"../../Materials/uniformBuffer\";\r\n/**\r\n * Particle emitter emitting particles from the inside of a box.\r\n * It emits the particles randomly between 2 given directions.\r\n */\r\nexport class MeshParticleEmitter implements IParticleEmitterType {\r\n private _indices: Nullable = null;\r\n private _positions: Nullable = null;\r\n private _normals: Nullable = null;\r\n private _storedNormal = Vector3.Zero();\r\n private _mesh: Nullable = null;\r\n\r\n /**\r\n * Random direction of each particle after it has been emitted, between direction1 and direction2 vectors.\r\n */\r\n public direction1 = new Vector3(0, 1.0, 0);\r\n /**\r\n * Random direction of each particle after it has been emitted, between direction1 and direction2 vectors.\r\n */\r\n public direction2 = new Vector3(0, 1.0, 0);\r\n\r\n /**\r\n * Gets or sets a boolean indicating that particle directions must be built from mesh face normals\r\n */\r\n public useMeshNormalsForDirection = true;\r\n\r\n /** Defines the mesh to use as source */\r\n public get mesh(): Nullable {\r\n return this._mesh;\r\n }\r\n\r\n public set mesh(value: Nullable) {\r\n if (this._mesh === value) {\r\n return;\r\n }\r\n\r\n this._mesh = value;\r\n\r\n if (value) {\r\n this._indices = value.getIndices();\r\n this._positions = value.getVerticesData(VertexBuffer.PositionKind);\r\n this._normals = value.getVerticesData(VertexBuffer.NormalKind);\r\n } else {\r\n this._indices = null;\r\n this._positions = null;\r\n this._normals = null;\r\n }\r\n }\r\n\r\n /**\r\n * Creates a new instance MeshParticleEmitter\r\n * @param mesh defines the mesh to use as source\r\n */\r\n constructor(mesh: Nullable = null) {\r\n this.mesh = mesh;\r\n }\r\n\r\n /**\r\n * Called by the particle System when the direction is computed for the created particle.\r\n * @param worldMatrix is the world matrix of the particle system\r\n * @param directionToUpdate is the direction vector to update with the result\r\n * @param particle is the particle we are computed the direction for\r\n * @param isLocal defines if the direction should be set in local space\r\n */\r\n public startDirectionFunction(worldMatrix: Matrix, directionToUpdate: Vector3, particle: Particle, isLocal: boolean): void {\r\n if (this.useMeshNormalsForDirection && this._normals) {\r\n Vector3.TransformNormalToRef(this._storedNormal, worldMatrix, directionToUpdate);\r\n return;\r\n }\r\n\r\n const randX = Scalar.RandomRange(this.direction1.x, this.direction2.x);\r\n const randY = Scalar.RandomRange(this.direction1.y, this.direction2.y);\r\n const randZ = Scalar.RandomRange(this.direction1.z, this.direction2.z);\r\n\r\n if (isLocal) {\r\n directionToUpdate.copyFromFloats(randX, randY, randZ);\r\n return;\r\n }\r\n\r\n Vector3.TransformNormalFromFloatsToRef(randX, randY, randZ, worldMatrix, directionToUpdate);\r\n }\r\n\r\n /**\r\n * Called by the particle System when the position is computed for the created particle.\r\n * @param worldMatrix is the world matrix of the particle system\r\n * @param positionToUpdate is the position vector to update with the result\r\n * @param particle is the particle we are computed the position for\r\n * @param isLocal defines if the position should be set in local space\r\n */\r\n public startPositionFunction(worldMatrix: Matrix, positionToUpdate: Vector3, particle: Particle, isLocal: boolean): void {\r\n if (!this._indices || !this._positions) {\r\n return;\r\n }\r\n\r\n const randomFaceIndex = (3 * Math.random() * (this._indices.length / 3)) | 0;\r\n const bu = Math.random();\r\n const bv = Math.random() * (1.0 - bu);\r\n const bw = 1.0 - bu - bv;\r\n\r\n const faceIndexA = this._indices[randomFaceIndex];\r\n const faceIndexB = this._indices[randomFaceIndex + 1];\r\n const faceIndexC = this._indices[randomFaceIndex + 2];\r\n const vertexA = TmpVectors.Vector3[0];\r\n const vertexB = TmpVectors.Vector3[1];\r\n const vertexC = TmpVectors.Vector3[2];\r\n const randomVertex = TmpVectors.Vector3[3];\r\n\r\n Vector3.FromArrayToRef(this._positions, faceIndexA * 3, vertexA);\r\n Vector3.FromArrayToRef(this._positions, faceIndexB * 3, vertexB);\r\n Vector3.FromArrayToRef(this._positions, faceIndexC * 3, vertexC);\r\n\r\n randomVertex.x = bu * vertexA.x + bv * vertexB.x + bw * vertexC.x;\r\n randomVertex.y = bu * vertexA.y + bv * vertexB.y + bw * vertexC.y;\r\n randomVertex.z = bu * vertexA.z + bv * vertexB.z + bw * vertexC.z;\r\n\r\n if (isLocal) {\r\n positionToUpdate.copyFromFloats(randomVertex.x, randomVertex.y, randomVertex.z);\r\n } else {\r\n Vector3.TransformCoordinatesFromFloatsToRef(randomVertex.x, randomVertex.y, randomVertex.z, worldMatrix, positionToUpdate);\r\n }\r\n\r\n if (this.useMeshNormalsForDirection && this._normals) {\r\n Vector3.FromArrayToRef(this._normals, faceIndexA * 3, vertexA);\r\n Vector3.FromArrayToRef(this._normals, faceIndexB * 3, vertexB);\r\n Vector3.FromArrayToRef(this._normals, faceIndexC * 3, vertexC);\r\n\r\n this._storedNormal.x = bu * vertexA.x + bv * vertexB.x + bw * vertexC.x;\r\n this._storedNormal.y = bu * vertexA.y + bv * vertexB.y + bw * vertexC.y;\r\n this._storedNormal.z = bu * vertexA.z + bv * vertexB.z + bw * vertexC.z;\r\n }\r\n }\r\n\r\n /**\r\n * Clones the current emitter and returns a copy of it\r\n * @returns the new emitter\r\n */\r\n public clone(): MeshParticleEmitter {\r\n const newOne = new MeshParticleEmitter(this.mesh);\r\n\r\n DeepCopier.DeepCopy(this, newOne);\r\n\r\n return newOne;\r\n }\r\n\r\n /**\r\n * Called by the GPUParticleSystem to setup the update shader\r\n * @param uboOrEffect defines the update shader\r\n */\r\n public applyToShader(uboOrEffect: UniformBufferEffectCommonAccessor): void {\r\n uboOrEffect.setVector3(\"direction1\", this.direction1);\r\n uboOrEffect.setVector3(\"direction2\", this.direction2);\r\n }\r\n\r\n /**\r\n * Creates the structure of the ubo for this particle emitter\r\n * @param ubo ubo to create the structure for\r\n */\r\n public buildUniformLayout(ubo: UniformBuffer): void {\r\n ubo.addUniform(\"direction1\", 3);\r\n ubo.addUniform(\"direction2\", 3);\r\n }\r\n\r\n /**\r\n * Returns a string to use to update the GPU particles update shader\r\n * @returns a string containing the defines string\r\n */\r\n public getEffectDefines(): string {\r\n return \"\";\r\n }\r\n\r\n /**\r\n * Returns the string \"BoxParticleEmitter\"\r\n * @returns a string containing the class name\r\n */\r\n public getClassName(): string {\r\n return \"MeshParticleEmitter\";\r\n }\r\n\r\n /**\r\n * Serializes the particle system to a JSON object.\r\n * @returns the JSON object\r\n */\r\n public serialize(): any {\r\n const serializationObject: any = {};\r\n\r\n serializationObject.type = this.getClassName();\r\n serializationObject.direction1 = this.direction1.asArray();\r\n serializationObject.direction2 = this.direction2.asArray();\r\n serializationObject.meshId = this.mesh?.id;\r\n serializationObject.useMeshNormalsForDirection = this.useMeshNormalsForDirection;\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Parse properties from a JSON object\r\n * @param serializationObject defines the JSON object\r\n * @param scene defines the hosting scene\r\n */\r\n public parse(serializationObject: any, scene: Nullable): void {\r\n Vector3.FromArrayToRef(serializationObject.direction1, 0, this.direction1);\r\n Vector3.FromArrayToRef(serializationObject.direction2, 0, this.direction2);\r\n\r\n if (serializationObject.meshId && scene) {\r\n this.mesh = scene.getLastMeshById(serializationObject.meshId);\r\n }\r\n\r\n this.useMeshNormalsForDirection = serializationObject.useMeshNormalsForDirection;\r\n }\r\n}\r\n","/* eslint-disable import/no-internal-modules */\r\nimport type { Nullable } from \"../types\";\r\nimport { Vector2, Vector3 } from \"../Maths/math.vector\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { ImageProcessingConfiguration } from \"../Materials/imageProcessingConfiguration\";\r\nimport { ImageProcessingConfigurationDefines } from \"../Materials/imageProcessingConfiguration\";\r\nimport type { ColorGradient, FactorGradient, Color3Gradient, IValueGradient } from \"../Misc/gradients\";\r\nimport type { IParticleEmitterType } from \"../Particles/EmitterTypes/index\";\r\nimport {\r\n BoxParticleEmitter,\r\n PointParticleEmitter,\r\n HemisphericParticleEmitter,\r\n SphereParticleEmitter,\r\n SphereDirectedParticleEmitter,\r\n CylinderParticleEmitter,\r\n CylinderDirectedParticleEmitter,\r\n ConeParticleEmitter,\r\n} from \"../Particles/EmitterTypes/index\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport type { BaseTexture } from \"../Materials/Textures/baseTexture\";\r\nimport { Color4 } from \"../Maths/math.color\";\r\nimport type { ThinEngine } from \"../Engines/thinEngine\";\r\n\r\nimport \"../Engines/Extensions/engine.dynamicBuffer\";\r\nimport type { IClipPlanesHolder } from \"../Misc/interfaces/iClipPlanesHolder\";\r\nimport type { Plane } from \"../Maths/math.plane\";\r\nimport type { Animation } from \"../Animations/animation\";\r\nimport type { Scene } from \"../scene\";\r\nimport type { ProceduralTexture } from \"../Materials/Textures/Procedurals/proceduralTexture\";\r\nimport type { RawTexture } from \"../Materials/Textures/rawTexture\";\r\n\r\n/**\r\n * This represents the base class for particle system in Babylon.\r\n * Particles are often small sprites used to simulate hard-to-reproduce phenomena like fire, smoke, water, or abstract visual effects like magic glitter and faery dust.\r\n * Particles can take different shapes while emitted like box, sphere, cone or you can write your custom function.\r\n * @example https://doc.babylonjs.com/features/featuresDeepDive/particles/particle_system/particle_system_intro\r\n */\r\nexport class BaseParticleSystem implements IClipPlanesHolder {\r\n /**\r\n * Source color is added to the destination color without alpha affecting the result\r\n */\r\n public static BLENDMODE_ONEONE = 0;\r\n /**\r\n * Blend current color and particle color using particle’s alpha\r\n */\r\n public static BLENDMODE_STANDARD = 1;\r\n /**\r\n * Add current color and particle color multiplied by particle’s alpha\r\n */\r\n public static BLENDMODE_ADD = 2;\r\n /**\r\n * Multiply current color with particle color\r\n */\r\n public static BLENDMODE_MULTIPLY = 3;\r\n\r\n /**\r\n * Multiply current color with particle color then add current color and particle color multiplied by particle’s alpha\r\n */\r\n public static BLENDMODE_MULTIPLYADD = 4;\r\n\r\n /**\r\n * List of animations used by the particle system.\r\n */\r\n public animations: Animation[] = [];\r\n\r\n /**\r\n * Gets or sets the unique id of the particle system\r\n */\r\n public uniqueId: number;\r\n\r\n /**\r\n * The id of the Particle system.\r\n */\r\n public id: string;\r\n\r\n /**\r\n * The friendly name of the Particle system.\r\n */\r\n public name: string;\r\n\r\n /**\r\n * Snippet ID if the particle system was created from the snippet server\r\n */\r\n public snippetId: string;\r\n\r\n /**\r\n * The rendering group used by the Particle system to chose when to render.\r\n */\r\n public renderingGroupId = 0;\r\n\r\n /**\r\n * The emitter represents the Mesh or position we are attaching the particle system to.\r\n */\r\n public emitter: Nullable = Vector3.Zero();\r\n\r\n /**\r\n * The maximum number of particles to emit per frame\r\n */\r\n public emitRate = 10;\r\n\r\n /**\r\n * If you want to launch only a few particles at once, that can be done, as well.\r\n */\r\n public manualEmitCount = -1;\r\n\r\n /**\r\n * The overall motion speed (0.01 is default update speed, faster updates = faster animation)\r\n */\r\n public updateSpeed = 0.01;\r\n\r\n /**\r\n * The amount of time the particle system is running (depends of the overall update speed).\r\n */\r\n public targetStopDuration = 0;\r\n\r\n /**\r\n * Specifies whether the particle system will be disposed once it reaches the end of the animation.\r\n */\r\n public disposeOnStop = false;\r\n\r\n /**\r\n * Minimum power of emitting particles.\r\n */\r\n public minEmitPower = 1;\r\n /**\r\n * Maximum power of emitting particles.\r\n */\r\n public maxEmitPower = 1;\r\n\r\n /**\r\n * Minimum life time of emitting particles.\r\n */\r\n public minLifeTime = 1;\r\n /**\r\n * Maximum life time of emitting particles.\r\n */\r\n public maxLifeTime = 1;\r\n\r\n /**\r\n * Minimum Size of emitting particles.\r\n */\r\n public minSize = 1;\r\n /**\r\n * Maximum Size of emitting particles.\r\n */\r\n public maxSize = 1;\r\n\r\n /**\r\n * Minimum scale of emitting particles on X axis.\r\n */\r\n public minScaleX = 1;\r\n /**\r\n * Maximum scale of emitting particles on X axis.\r\n */\r\n public maxScaleX = 1;\r\n\r\n /**\r\n * Minimum scale of emitting particles on Y axis.\r\n */\r\n public minScaleY = 1;\r\n /**\r\n * Maximum scale of emitting particles on Y axis.\r\n */\r\n public maxScaleY = 1;\r\n\r\n /**\r\n * Gets or sets the minimal initial rotation in radians.\r\n */\r\n public minInitialRotation = 0;\r\n /**\r\n * Gets or sets the maximal initial rotation in radians.\r\n */\r\n public maxInitialRotation = 0;\r\n\r\n /**\r\n * Minimum angular speed of emitting particles (Z-axis rotation for each particle).\r\n */\r\n public minAngularSpeed = 0;\r\n /**\r\n * Maximum angular speed of emitting particles (Z-axis rotation for each particle).\r\n */\r\n public maxAngularSpeed = 0;\r\n\r\n /**\r\n * The texture used to render each particle. (this can be a spritesheet)\r\n */\r\n public particleTexture: Nullable;\r\n\r\n /**\r\n * The layer mask we are rendering the particles through.\r\n */\r\n public layerMask: number = 0x0fffffff;\r\n\r\n /**\r\n * This can help using your own shader to render the particle system.\r\n * The according effect will be created\r\n */\r\n public customShader: any = null;\r\n\r\n /**\r\n * By default particle system starts as soon as they are created. This prevents the\r\n * automatic start to happen and let you decide when to start emitting particles.\r\n */\r\n public preventAutoStart: boolean = false;\r\n\r\n /** @internal */\r\n _wasDispatched = false;\r\n\r\n protected _rootUrl = \"\";\r\n private _noiseTexture: Nullable;\r\n\r\n /**\r\n * Gets or sets a texture used to add random noise to particle positions\r\n */\r\n public get noiseTexture(): Nullable {\r\n return this._noiseTexture;\r\n }\r\n\r\n public set noiseTexture(value: Nullable) {\r\n if (this._noiseTexture === value) {\r\n return;\r\n }\r\n\r\n this._noiseTexture = value;\r\n this._reset();\r\n }\r\n\r\n /** Gets or sets the strength to apply to the noise value (default is (10, 10, 10)) */\r\n public noiseStrength = new Vector3(10, 10, 10);\r\n\r\n /**\r\n * Callback triggered when the particle animation is ending.\r\n */\r\n public onAnimationEnd: Nullable<() => void> = null;\r\n\r\n /**\r\n * Blend mode use to render the particle, it can be either ParticleSystem.BLENDMODE_ONEONE or ParticleSystem.BLENDMODE_STANDARD.\r\n */\r\n public blendMode = BaseParticleSystem.BLENDMODE_ONEONE;\r\n\r\n /**\r\n * Forces the particle to write their depth information to the depth buffer. This can help preventing other draw calls\r\n * to override the particles.\r\n */\r\n public forceDepthWrite = false;\r\n\r\n /** Gets or sets a value indicating how many cycles (or frames) must be executed before first rendering (this value has to be set before starting the system). Default is 0 */\r\n public preWarmCycles = 0;\r\n\r\n /** Gets or sets a value indicating the time step multiplier to use in pre-warm mode (default is 1) */\r\n public preWarmStepOffset = 1;\r\n\r\n /**\r\n * If using a spritesheet (isAnimationSheetEnabled) defines the speed of the sprite loop (default is 1 meaning the animation will play once during the entire particle lifetime)\r\n */\r\n public spriteCellChangeSpeed = 1;\r\n /**\r\n * If using a spritesheet (isAnimationSheetEnabled) defines the first sprite cell to display\r\n */\r\n public startSpriteCellID = 0;\r\n /**\r\n * If using a spritesheet (isAnimationSheetEnabled) defines the last sprite cell to display\r\n */\r\n public endSpriteCellID = 0;\r\n /**\r\n * If using a spritesheet (isAnimationSheetEnabled), defines the sprite cell width to use\r\n */\r\n public spriteCellWidth = 0;\r\n /**\r\n * If using a spritesheet (isAnimationSheetEnabled), defines the sprite cell height to use\r\n */\r\n public spriteCellHeight = 0;\r\n /**\r\n * If using a spritesheet (isAnimationSheetEnabled), defines wether the sprite animation is looping\r\n */\r\n public spriteCellLoop = true;\r\n /**\r\n * This allows the system to random pick the start cell ID between startSpriteCellID and endSpriteCellID\r\n */\r\n public spriteRandomStartCell = false;\r\n\r\n /** Gets or sets a Vector2 used to move the pivot (by default (0,0)) */\r\n public translationPivot = new Vector2(0, 0);\r\n\r\n /** @internal */\r\n public _isAnimationSheetEnabled: boolean;\r\n\r\n /**\r\n * Gets or sets a boolean indicating that hosted animations (in the system.animations array) must be started when system.start() is called\r\n */\r\n public beginAnimationOnStart = false;\r\n\r\n /**\r\n * Gets or sets the frame to start the animation from when beginAnimationOnStart is true\r\n */\r\n public beginAnimationFrom = 0;\r\n\r\n /**\r\n * Gets or sets the frame to end the animation on when beginAnimationOnStart is true\r\n */\r\n public beginAnimationTo = 60;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if animations must loop when beginAnimationOnStart is true\r\n */\r\n public beginAnimationLoop = false;\r\n\r\n /**\r\n * Gets or sets a world offset applied to all particles\r\n */\r\n public worldOffset = new Vector3(0, 0, 0);\r\n\r\n /**\r\n * Gets or sets the active clipplane 1\r\n */\r\n public clipPlane: Nullable;\r\n\r\n /**\r\n * Gets or sets the active clipplane 2\r\n */\r\n public clipPlane2: Nullable;\r\n\r\n /**\r\n * Gets or sets the active clipplane 3\r\n */\r\n public clipPlane3: Nullable;\r\n\r\n /**\r\n * Gets or sets the active clipplane 4\r\n */\r\n public clipPlane4: Nullable;\r\n\r\n /**\r\n * Gets or sets the active clipplane 5\r\n */\r\n public clipPlane5: Nullable;\r\n\r\n /**\r\n * Gets or sets the active clipplane 6\r\n */\r\n public clipPlane6: Nullable;\r\n\r\n /**\r\n * Gets or sets whether an animation sprite sheet is enabled or not on the particle system\r\n */\r\n public get isAnimationSheetEnabled(): boolean {\r\n return this._isAnimationSheetEnabled;\r\n }\r\n\r\n public set isAnimationSheetEnabled(value: boolean) {\r\n if (this._isAnimationSheetEnabled == value) {\r\n return;\r\n }\r\n\r\n this._isAnimationSheetEnabled = value;\r\n\r\n this._reset();\r\n }\r\n\r\n private _useLogarithmicDepth: boolean = false;\r\n\r\n /**\r\n * Gets or sets a boolean enabling the use of logarithmic depth buffers, which is good for wide depth buffers.\r\n */\r\n public get useLogarithmicDepth(): boolean {\r\n return this._useLogarithmicDepth;\r\n }\r\n\r\n public set useLogarithmicDepth(value: boolean) {\r\n this._useLogarithmicDepth = value && this.getScene()!.getEngine().getCaps().fragmentDepthSupported;\r\n }\r\n\r\n /**\r\n * Get hosting scene\r\n * @returns the scene\r\n */\r\n public getScene(): Nullable {\r\n return this._scene;\r\n }\r\n\r\n /**\r\n * You can use gravity if you want to give an orientation to your particles.\r\n */\r\n public gravity = Vector3.Zero();\r\n\r\n protected _colorGradients: Nullable> = null;\r\n protected _sizeGradients: Nullable> = null;\r\n protected _lifeTimeGradients: Nullable> = null;\r\n protected _angularSpeedGradients: Nullable> = null;\r\n protected _velocityGradients: Nullable> = null;\r\n protected _limitVelocityGradients: Nullable> = null;\r\n protected _dragGradients: Nullable> = null;\r\n protected _emitRateGradients: Nullable> = null;\r\n protected _startSizeGradients: Nullable> = null;\r\n protected _rampGradients: Nullable> = null;\r\n protected _colorRemapGradients: Nullable> = null;\r\n protected _alphaRemapGradients: Nullable> = null;\r\n\r\n protected _hasTargetStopDurationDependantGradient() {\r\n return (\r\n (this._startSizeGradients && this._startSizeGradients.length > 0) ||\r\n (this._emitRateGradients && this._emitRateGradients.length > 0) ||\r\n (this._lifeTimeGradients && this._lifeTimeGradients.length > 0)\r\n );\r\n }\r\n\r\n /**\r\n * Defines the delay in milliseconds before starting the system (0 by default)\r\n */\r\n public startDelay = 0;\r\n\r\n /**\r\n * Gets the current list of drag gradients.\r\n * You must use addDragGradient and removeDragGradient to update this list\r\n * @returns the list of drag gradients\r\n */\r\n public getDragGradients(): Nullable> {\r\n return this._dragGradients;\r\n }\r\n\r\n /** Gets or sets a value indicating the damping to apply if the limit velocity factor is reached */\r\n public limitVelocityDamping = 0.4;\r\n\r\n /**\r\n * Gets the current list of limit velocity gradients.\r\n * You must use addLimitVelocityGradient and removeLimitVelocityGradient to update this list\r\n * @returns the list of limit velocity gradients\r\n */\r\n public getLimitVelocityGradients(): Nullable> {\r\n return this._limitVelocityGradients;\r\n }\r\n\r\n /**\r\n * Gets the current list of color gradients.\r\n * You must use addColorGradient and removeColorGradient to update this list\r\n * @returns the list of color gradients\r\n */\r\n public getColorGradients(): Nullable> {\r\n return this._colorGradients;\r\n }\r\n\r\n /**\r\n * Gets the current list of size gradients.\r\n * You must use addSizeGradient and removeSizeGradient to update this list\r\n * @returns the list of size gradients\r\n */\r\n public getSizeGradients(): Nullable> {\r\n return this._sizeGradients;\r\n }\r\n\r\n /**\r\n * Gets the current list of color remap gradients.\r\n * You must use addColorRemapGradient and removeColorRemapGradient to update this list\r\n * @returns the list of color remap gradients\r\n */\r\n public getColorRemapGradients(): Nullable> {\r\n return this._colorRemapGradients;\r\n }\r\n\r\n /**\r\n * Gets the current list of alpha remap gradients.\r\n * You must use addAlphaRemapGradient and removeAlphaRemapGradient to update this list\r\n * @returns the list of alpha remap gradients\r\n */\r\n public getAlphaRemapGradients(): Nullable> {\r\n return this._alphaRemapGradients;\r\n }\r\n\r\n /**\r\n * Gets the current list of life time gradients.\r\n * You must use addLifeTimeGradient and removeLifeTimeGradient to update this list\r\n * @returns the list of life time gradients\r\n */\r\n public getLifeTimeGradients(): Nullable> {\r\n return this._lifeTimeGradients;\r\n }\r\n\r\n /**\r\n * Gets the current list of angular speed gradients.\r\n * You must use addAngularSpeedGradient and removeAngularSpeedGradient to update this list\r\n * @returns the list of angular speed gradients\r\n */\r\n public getAngularSpeedGradients(): Nullable> {\r\n return this._angularSpeedGradients;\r\n }\r\n\r\n /**\r\n * Gets the current list of velocity gradients.\r\n * You must use addVelocityGradient and removeVelocityGradient to update this list\r\n * @returns the list of velocity gradients\r\n */\r\n public getVelocityGradients(): Nullable> {\r\n return this._velocityGradients;\r\n }\r\n\r\n /**\r\n * Gets the current list of start size gradients.\r\n * You must use addStartSizeGradient and removeStartSizeGradient to update this list\r\n * @returns the list of start size gradients\r\n */\r\n public getStartSizeGradients(): Nullable> {\r\n return this._startSizeGradients;\r\n }\r\n\r\n /**\r\n * Gets the current list of emit rate gradients.\r\n * You must use addEmitRateGradient and removeEmitRateGradient to update this list\r\n * @returns the list of emit rate gradients\r\n */\r\n public getEmitRateGradients(): Nullable> {\r\n return this._emitRateGradients;\r\n }\r\n\r\n /**\r\n * Random direction of each particle after it has been emitted, between direction1 and direction2 vectors.\r\n * This only works when particleEmitterTyps is a BoxParticleEmitter\r\n */\r\n public get direction1(): Vector3 {\r\n if ((this.particleEmitterType).direction1) {\r\n return (this.particleEmitterType).direction1;\r\n }\r\n\r\n return Vector3.Zero();\r\n }\r\n\r\n public set direction1(value: Vector3) {\r\n if ((this.particleEmitterType).direction1) {\r\n (this.particleEmitterType).direction1 = value;\r\n }\r\n }\r\n\r\n /**\r\n * Random direction of each particle after it has been emitted, between direction1 and direction2 vectors.\r\n * This only works when particleEmitterTyps is a BoxParticleEmitter\r\n */\r\n public get direction2(): Vector3 {\r\n if ((this.particleEmitterType).direction2) {\r\n return (this.particleEmitterType).direction2;\r\n }\r\n\r\n return Vector3.Zero();\r\n }\r\n\r\n public set direction2(value: Vector3) {\r\n if ((this.particleEmitterType).direction2) {\r\n (this.particleEmitterType).direction2 = value;\r\n }\r\n }\r\n\r\n /**\r\n * Minimum box point around our emitter. Our emitter is the center of particles source, but if you want your particles to emit from more than one point, then you can tell it to do so.\r\n * This only works when particleEmitterTyps is a BoxParticleEmitter\r\n */\r\n public get minEmitBox(): Vector3 {\r\n if ((this.particleEmitterType).minEmitBox) {\r\n return (this.particleEmitterType).minEmitBox;\r\n }\r\n\r\n return Vector3.Zero();\r\n }\r\n\r\n public set minEmitBox(value: Vector3) {\r\n if ((this.particleEmitterType).minEmitBox) {\r\n (this.particleEmitterType).minEmitBox = value;\r\n }\r\n }\r\n\r\n /**\r\n * Maximum box point around our emitter. Our emitter is the center of particles source, but if you want your particles to emit from more than one point, then you can tell it to do so.\r\n * This only works when particleEmitterTyps is a BoxParticleEmitter\r\n */\r\n public get maxEmitBox(): Vector3 {\r\n if ((this.particleEmitterType).maxEmitBox) {\r\n return (this.particleEmitterType).maxEmitBox;\r\n }\r\n\r\n return Vector3.Zero();\r\n }\r\n\r\n public set maxEmitBox(value: Vector3) {\r\n if ((this.particleEmitterType).maxEmitBox) {\r\n (this.particleEmitterType).maxEmitBox = value;\r\n }\r\n }\r\n\r\n /**\r\n * Random color of each particle after it has been emitted, between color1 and color2 vectors\r\n */\r\n public color1 = new Color4(1.0, 1.0, 1.0, 1.0);\r\n /**\r\n * Random color of each particle after it has been emitted, between color1 and color2 vectors\r\n */\r\n public color2 = new Color4(1.0, 1.0, 1.0, 1.0);\r\n /**\r\n * Color the particle will have at the end of its lifetime\r\n */\r\n public colorDead = new Color4(0, 0, 0, 1.0);\r\n\r\n /**\r\n * An optional mask to filter some colors out of the texture, or filter a part of the alpha channel\r\n */\r\n public textureMask = new Color4(1.0, 1.0, 1.0, 1.0);\r\n\r\n /**\r\n * The particle emitter type defines the emitter used by the particle system.\r\n * It can be for example box, sphere, or cone...\r\n */\r\n public particleEmitterType: IParticleEmitterType;\r\n\r\n /** @internal */\r\n public _isSubEmitter = false;\r\n\r\n /** @internal */\r\n public _billboardMode = Constants.PARTICLES_BILLBOARDMODE_ALL;\r\n /**\r\n * Gets or sets the billboard mode to use when isBillboardBased = true.\r\n * Value can be: ParticleSystem.BILLBOARDMODE_ALL, ParticleSystem.BILLBOARDMODE_Y, ParticleSystem.BILLBOARDMODE_STRETCHED\r\n */\r\n public get billboardMode(): number {\r\n return this._billboardMode;\r\n }\r\n\r\n public set billboardMode(value: number) {\r\n if (this._billboardMode === value) {\r\n return;\r\n }\r\n\r\n this._billboardMode = value;\r\n this._reset();\r\n }\r\n\r\n /** @internal */\r\n public _isBillboardBased = true;\r\n /**\r\n * Gets or sets a boolean indicating if the particles must be rendered as billboard or aligned with the direction\r\n */\r\n public get isBillboardBased(): boolean {\r\n return this._isBillboardBased;\r\n }\r\n\r\n public set isBillboardBased(value: boolean) {\r\n if (this._isBillboardBased === value) {\r\n return;\r\n }\r\n\r\n this._isBillboardBased = value;\r\n this._reset();\r\n }\r\n\r\n /**\r\n * The scene the particle system belongs to.\r\n */\r\n protected _scene: Nullable;\r\n\r\n /**\r\n * The engine the particle system belongs to.\r\n */\r\n protected _engine: ThinEngine;\r\n\r\n /**\r\n * Local cache of defines for image processing.\r\n */\r\n protected _imageProcessingConfigurationDefines = new ImageProcessingConfigurationDefines();\r\n\r\n /**\r\n * Default configuration related to image processing available in the standard Material.\r\n */\r\n protected _imageProcessingConfiguration: Nullable;\r\n\r\n /**\r\n * Gets the image processing configuration used either in this material.\r\n */\r\n public get imageProcessingConfiguration(): Nullable {\r\n return this._imageProcessingConfiguration;\r\n }\r\n\r\n /**\r\n * Sets the Default image processing configuration used either in the this material.\r\n *\r\n * If sets to null, the scene one is in use.\r\n */\r\n public set imageProcessingConfiguration(value: Nullable) {\r\n this._attachImageProcessingConfiguration(value);\r\n }\r\n\r\n /**\r\n * Attaches a new image processing configuration to the Standard Material.\r\n * @param configuration\r\n */\r\n protected _attachImageProcessingConfiguration(configuration: Nullable): void {\r\n if (configuration === this._imageProcessingConfiguration) {\r\n return;\r\n }\r\n\r\n // Pick the scene configuration if needed.\r\n if (!configuration && this._scene) {\r\n this._imageProcessingConfiguration = this._scene.imageProcessingConfiguration;\r\n } else {\r\n this._imageProcessingConfiguration = configuration;\r\n }\r\n }\r\n\r\n /** @internal */\r\n protected _reset() {}\r\n\r\n /**\r\n * @internal\r\n */\r\n protected _removeGradientAndTexture(gradient: number, gradients: Nullable, texture: Nullable): BaseParticleSystem {\r\n if (!gradients) {\r\n return this;\r\n }\r\n\r\n let index = 0;\r\n for (const valueGradient of gradients) {\r\n if (valueGradient.gradient === gradient) {\r\n gradients.splice(index, 1);\r\n break;\r\n }\r\n index++;\r\n }\r\n\r\n if (texture) {\r\n texture.dispose();\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Instantiates a particle system.\r\n * Particles are often small sprites used to simulate hard-to-reproduce phenomena like fire, smoke, water, or abstract visual effects like magic glitter and faery dust.\r\n * @param name The name of the particle system\r\n */\r\n public constructor(name: string) {\r\n this.id = name;\r\n this.name = name;\r\n }\r\n\r\n /**\r\n * Creates a Point Emitter for the particle system (emits directly from the emitter position)\r\n * @param direction1 Particles are emitted between the direction1 and direction2 from within the box\r\n * @param direction2 Particles are emitted between the direction1 and direction2 from within the box\r\n * @returns the emitter\r\n */\r\n public createPointEmitter(direction1: Vector3, direction2: Vector3): PointParticleEmitter {\r\n const particleEmitter = new PointParticleEmitter();\r\n particleEmitter.direction1 = direction1;\r\n particleEmitter.direction2 = direction2;\r\n\r\n this.particleEmitterType = particleEmitter;\r\n return particleEmitter;\r\n }\r\n\r\n /**\r\n * Creates a Hemisphere Emitter for the particle system (emits along the hemisphere radius)\r\n * @param radius The radius of the hemisphere to emit from\r\n * @param radiusRange The range of the hemisphere to emit from [0-1] 0 Surface Only, 1 Entire Radius\r\n * @returns the emitter\r\n */\r\n public createHemisphericEmitter(radius = 1, radiusRange = 1): HemisphericParticleEmitter {\r\n const particleEmitter = new HemisphericParticleEmitter(radius, radiusRange);\r\n this.particleEmitterType = particleEmitter;\r\n return particleEmitter;\r\n }\r\n\r\n /**\r\n * Creates a Sphere Emitter for the particle system (emits along the sphere radius)\r\n * @param radius The radius of the sphere to emit from\r\n * @param radiusRange The range of the sphere to emit from [0-1] 0 Surface Only, 1 Entire Radius\r\n * @returns the emitter\r\n */\r\n public createSphereEmitter(radius = 1, radiusRange = 1): SphereParticleEmitter {\r\n const particleEmitter = new SphereParticleEmitter(radius, radiusRange);\r\n this.particleEmitterType = particleEmitter;\r\n return particleEmitter;\r\n }\r\n\r\n /**\r\n * Creates a Directed Sphere Emitter for the particle system (emits between direction1 and direction2)\r\n * @param radius The radius of the sphere to emit from\r\n * @param direction1 Particles are emitted between the direction1 and direction2 from within the sphere\r\n * @param direction2 Particles are emitted between the direction1 and direction2 from within the sphere\r\n * @returns the emitter\r\n */\r\n public createDirectedSphereEmitter(radius = 1, direction1 = new Vector3(0, 1.0, 0), direction2 = new Vector3(0, 1.0, 0)): SphereDirectedParticleEmitter {\r\n const particleEmitter = new SphereDirectedParticleEmitter(radius, direction1, direction2);\r\n this.particleEmitterType = particleEmitter;\r\n return particleEmitter;\r\n }\r\n\r\n /**\r\n * Creates a Cylinder Emitter for the particle system (emits from the cylinder to the particle position)\r\n * @param radius The radius of the emission cylinder\r\n * @param height The height of the emission cylinder\r\n * @param radiusRange The range of emission [0-1] 0 Surface only, 1 Entire Radius\r\n * @param directionRandomizer How much to randomize the particle direction [0-1]\r\n * @returns the emitter\r\n */\r\n public createCylinderEmitter(radius = 1, height = 1, radiusRange = 1, directionRandomizer = 0): CylinderParticleEmitter {\r\n const particleEmitter = new CylinderParticleEmitter(radius, height, radiusRange, directionRandomizer);\r\n this.particleEmitterType = particleEmitter;\r\n return particleEmitter;\r\n }\r\n\r\n /**\r\n * Creates a Directed Cylinder Emitter for the particle system (emits between direction1 and direction2)\r\n * @param radius The radius of the cylinder to emit from\r\n * @param height The height of the emission cylinder\r\n * @param radiusRange the range of the emission cylinder [0-1] 0 Surface only, 1 Entire Radius (1 by default)\r\n * @param direction1 Particles are emitted between the direction1 and direction2 from within the cylinder\r\n * @param direction2 Particles are emitted between the direction1 and direction2 from within the cylinder\r\n * @returns the emitter\r\n */\r\n public createDirectedCylinderEmitter(\r\n radius = 1,\r\n height = 1,\r\n radiusRange = 1,\r\n direction1 = new Vector3(0, 1.0, 0),\r\n direction2 = new Vector3(0, 1.0, 0)\r\n ): CylinderDirectedParticleEmitter {\r\n const particleEmitter = new CylinderDirectedParticleEmitter(radius, height, radiusRange, direction1, direction2);\r\n this.particleEmitterType = particleEmitter;\r\n return particleEmitter;\r\n }\r\n\r\n /**\r\n * Creates a Cone Emitter for the particle system (emits from the cone to the particle position)\r\n * @param radius The radius of the cone to emit from\r\n * @param angle The base angle of the cone\r\n * @returns the emitter\r\n */\r\n public createConeEmitter(radius = 1, angle = Math.PI / 4): ConeParticleEmitter {\r\n const particleEmitter = new ConeParticleEmitter(radius, angle);\r\n this.particleEmitterType = particleEmitter;\r\n return particleEmitter;\r\n }\r\n\r\n /**\r\n * Creates a Box Emitter for the particle system. (emits between direction1 and direction2 from withing the box defined by minEmitBox and maxEmitBox)\r\n * @param direction1 Particles are emitted between the direction1 and direction2 from within the box\r\n * @param direction2 Particles are emitted between the direction1 and direction2 from within the box\r\n * @param minEmitBox Particles are emitted from the box between minEmitBox and maxEmitBox\r\n * @param maxEmitBox Particles are emitted from the box between minEmitBox and maxEmitBox\r\n * @returns the emitter\r\n */\r\n public createBoxEmitter(direction1: Vector3, direction2: Vector3, minEmitBox: Vector3, maxEmitBox: Vector3): BoxParticleEmitter {\r\n const particleEmitter = new BoxParticleEmitter();\r\n this.particleEmitterType = particleEmitter;\r\n this.direction1 = direction1;\r\n this.direction2 = direction2;\r\n this.minEmitBox = minEmitBox;\r\n this.maxEmitBox = maxEmitBox;\r\n return particleEmitter;\r\n }\r\n}\r\n","import { NodeMaterialBlock } from \"../nodeMaterialBlock\";\r\nimport { NodeMaterialBlockConnectionPointTypes } from \"../Enums/nodeMaterialBlockConnectionPointTypes\";\r\nimport type { NodeMaterialBuildState } from \"../nodeMaterialBuildState\";\r\nimport { NodeMaterialBlockTargets } from \"../Enums/nodeMaterialBlockTargets\";\r\nimport type { NodeMaterialConnectionPoint } from \"../nodeMaterialBlockConnectionPoint\";\r\nimport { RegisterClass } from \"../../../Misc/typeStore\";\r\n\r\n/**\r\n * Block used to expand a Color3/4 into 4 outputs (one for each component)\r\n */\r\nexport class ColorSplitterBlock extends NodeMaterialBlock {\r\n /**\r\n * Create a new ColorSplitterBlock\r\n * @param name defines the block name\r\n */\r\n public constructor(name: string) {\r\n super(name, NodeMaterialBlockTargets.Neutral);\r\n\r\n this.registerInput(\"rgba\", NodeMaterialBlockConnectionPointTypes.Color4, true);\r\n this.registerInput(\"rgb \", NodeMaterialBlockConnectionPointTypes.Color3, true);\r\n\r\n this.registerOutput(\"rgb\", NodeMaterialBlockConnectionPointTypes.Color3);\r\n this.registerOutput(\"r\", NodeMaterialBlockConnectionPointTypes.Float);\r\n this.registerOutput(\"g\", NodeMaterialBlockConnectionPointTypes.Float);\r\n this.registerOutput(\"b\", NodeMaterialBlockConnectionPointTypes.Float);\r\n this.registerOutput(\"a\", NodeMaterialBlockConnectionPointTypes.Float);\r\n\r\n this.inputsAreExclusive = true;\r\n }\r\n\r\n /**\r\n * Gets the current class name\r\n * @returns the class name\r\n */\r\n public getClassName() {\r\n return \"ColorSplitterBlock\";\r\n }\r\n\r\n /**\r\n * Gets the rgba component (input)\r\n */\r\n public get rgba(): NodeMaterialConnectionPoint {\r\n return this._inputs[0];\r\n }\r\n\r\n /**\r\n * Gets the rgb component (input)\r\n */\r\n public get rgbIn(): NodeMaterialConnectionPoint {\r\n return this._inputs[1];\r\n }\r\n\r\n /**\r\n * Gets the rgb component (output)\r\n */\r\n public get rgbOut(): NodeMaterialConnectionPoint {\r\n return this._outputs[0];\r\n }\r\n\r\n /**\r\n * Gets the r component (output)\r\n */\r\n public get r(): NodeMaterialConnectionPoint {\r\n return this._outputs[1];\r\n }\r\n\r\n /**\r\n * Gets the g component (output)\r\n */\r\n public get g(): NodeMaterialConnectionPoint {\r\n return this._outputs[2];\r\n }\r\n /**\r\n * Gets the b component (output)\r\n */\r\n public get b(): NodeMaterialConnectionPoint {\r\n return this._outputs[3];\r\n }\r\n /**\r\n * Gets the a component (output)\r\n */\r\n public get a(): NodeMaterialConnectionPoint {\r\n return this._outputs[4];\r\n }\r\n\r\n protected _inputRename(name: string) {\r\n if (name === \"rgb \") {\r\n return \"rgbIn\";\r\n }\r\n return name;\r\n }\r\n\r\n protected _outputRename(name: string) {\r\n if (name === \"rgb\") {\r\n return \"rgbOut\";\r\n }\r\n return name;\r\n }\r\n\r\n protected _buildBlock(state: NodeMaterialBuildState) {\r\n super._buildBlock(state);\r\n\r\n const input = this.rgba.isConnected ? this.rgba : this.rgbIn;\r\n\r\n if (!input.isConnected) {\r\n return;\r\n }\r\n\r\n const rgbOutput = this._outputs[0];\r\n const rOutput = this._outputs[1];\r\n const gOutput = this._outputs[2];\r\n const bOutput = this._outputs[3];\r\n const aOutput = this._outputs[4];\r\n\r\n if (rgbOutput.hasEndpoints) {\r\n state.compilationString += this._declareOutput(rgbOutput, state) + ` = ${input.associatedVariableName}.rgb;\\n`;\r\n }\r\n if (rOutput.hasEndpoints) {\r\n state.compilationString += this._declareOutput(rOutput, state) + ` = ${input.associatedVariableName}.r;\\n`;\r\n }\r\n if (gOutput.hasEndpoints) {\r\n state.compilationString += this._declareOutput(gOutput, state) + ` = ${input.associatedVariableName}.g;\\n`;\r\n }\r\n if (bOutput.hasEndpoints) {\r\n state.compilationString += this._declareOutput(bOutput, state) + ` = ${input.associatedVariableName}.b;\\n`;\r\n }\r\n if (aOutput.hasEndpoints) {\r\n state.compilationString += this._declareOutput(aOutput, state) + ` = ${input.associatedVariableName}.a;\\n`;\r\n }\r\n\r\n return this;\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.ColorSplitterBlock\", ColorSplitterBlock);\r\n","import { Tools } from \"../../../Misc/tools\";\r\nimport type { Scene } from \"../../../scene\";\r\nimport type { ISceneComponent } from \"../../../sceneComponent\";\r\nimport { SceneComponentConstants } from \"../../../sceneComponent\";\r\n\r\nimport type { ProceduralTexture } from \"./proceduralTexture\";\r\n\r\ndeclare module \"../../../abstractScene\" {\r\n export interface AbstractScene {\r\n /**\r\n * The list of procedural textures added to the scene\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/proceduralTextures\r\n */\r\n proceduralTextures: Array;\r\n }\r\n}\r\n/**\r\n * Defines the Procedural Texture scene component responsible to manage any Procedural Texture\r\n * in a given scene.\r\n */\r\nexport class ProceduralTextureSceneComponent implements ISceneComponent {\r\n /**\r\n * The component name helpful to identify the component in the list of scene components.\r\n */\r\n public readonly name = SceneComponentConstants.NAME_PROCEDURALTEXTURE;\r\n\r\n /**\r\n * The scene the component belongs to.\r\n */\r\n public scene: Scene;\r\n\r\n /**\r\n * Creates a new instance of the component for the given scene\r\n * @param scene Defines the scene to register the component in\r\n */\r\n constructor(scene: Scene) {\r\n this.scene = scene;\r\n this.scene.proceduralTextures = new Array();\r\n }\r\n\r\n /**\r\n * Registers the component in a given scene\r\n */\r\n public register(): void {\r\n this.scene._beforeClearStage.registerStep(SceneComponentConstants.STEP_BEFORECLEAR_PROCEDURALTEXTURE, this, this._beforeClear);\r\n }\r\n\r\n /**\r\n * Rebuilds the elements related to this component in case of\r\n * context lost for instance.\r\n */\r\n public rebuild(): void {\r\n // Nothing to do here.\r\n }\r\n\r\n /**\r\n * Disposes the component and the associated resources.\r\n */\r\n public dispose(): void {\r\n // Nothing to do here.\r\n }\r\n\r\n private _beforeClear(): void {\r\n if (this.scene.proceduralTexturesEnabled) {\r\n Tools.StartPerformanceCounter(\"Procedural textures\", this.scene.proceduralTextures.length > 0);\r\n for (let proceduralIndex = 0; proceduralIndex < this.scene.proceduralTextures.length; proceduralIndex++) {\r\n const proceduralTexture = this.scene.proceduralTextures[proceduralIndex];\r\n if (proceduralTexture._shouldRender()) {\r\n proceduralTexture.render();\r\n }\r\n }\r\n Tools.EndPerformanceCounter(\"Procedural textures\", this.scene.proceduralTextures.length > 0);\r\n }\r\n }\r\n}\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\n\nconst name = \"proceduralVertexShader\";\nconst shader = `attribute vec2 position;varying vec2 vPosition;varying vec2 vUV;const vec2 madd=vec2(0.5,0.5);\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\nvPosition=position;vUV=position*madd+madd;gl_Position=vec4(position,0.0,1.0);\n#define CUSTOM_VERTEX_MAIN_END\n}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const proceduralVertexShader = { name, shader };\n","import { serialize } from \"../../../Misc/decorators\";\r\nimport { Observable } from \"../../../Misc/observable\";\r\nimport type { Nullable } from \"../../../types\";\r\nimport type { Scene } from \"../../../scene\";\r\nimport type { Matrix, Vector3, Vector2 } from \"../../../Maths/math.vector\";\r\nimport type { Color4, Color3 } from \"../../../Maths/math.color\";\r\nimport type { Engine } from \"../../../Engines/engine\";\r\nimport { VertexBuffer } from \"../../../Buffers/buffer\";\r\nimport { SceneComponentConstants } from \"../../../sceneComponent\";\r\n\r\nimport { Material } from \"../../../Materials/material\";\r\nimport type { Effect } from \"../../../Materials/effect\";\r\nimport { Texture } from \"../../../Materials/Textures/texture\";\r\nimport { RenderTargetTexture } from \"../../../Materials/Textures/renderTargetTexture\";\r\nimport { ProceduralTextureSceneComponent } from \"./proceduralTextureSceneComponent\";\r\n\r\nimport \"../../../Engines/Extensions/engine.renderTarget\";\r\nimport \"../../../Engines/Extensions/engine.renderTargetCube\";\r\nimport \"../../../Shaders/procedural.vertex\";\r\nimport type { DataBuffer } from \"../../../Buffers/dataBuffer\";\r\nimport { RegisterClass } from \"../../../Misc/typeStore\";\r\nimport type { NodeMaterial } from \"../../Node/nodeMaterial\";\r\nimport type { TextureSize } from \"../../../Materials/Textures/textureCreationOptions\";\r\nimport { EngineStore } from \"../../../Engines/engineStore\";\r\nimport { Constants } from \"../../../Engines/constants\";\r\nimport { DrawWrapper } from \"../../drawWrapper\";\r\nimport type { RenderTargetWrapper } from \"../../../Engines/renderTargetWrapper\";\r\n\r\n/**\r\n * Procedural texturing is a way to programmatically create a texture. There are 2 types of procedural textures: code-only, and code that references some classic 2D images, sometimes calmpler' images.\r\n * This is the base class of any Procedural texture and contains most of the shareable code.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/proceduralTextures\r\n */\r\nexport class ProceduralTexture extends Texture {\r\n /**\r\n * Define if the texture is enabled or not (disabled texture will not render)\r\n */\r\n @serialize()\r\n public isEnabled = true;\r\n\r\n /**\r\n * Define if the texture must be cleared before rendering (default is true)\r\n */\r\n @serialize()\r\n public autoClear = true;\r\n\r\n /**\r\n * Callback called when the texture is generated\r\n */\r\n public onGenerated: () => void;\r\n\r\n /**\r\n * Event raised when the texture is generated\r\n */\r\n public onGeneratedObservable = new Observable();\r\n\r\n /**\r\n * Event raised before the texture is generated\r\n */\r\n public onBeforeGenerationObservable = new Observable();\r\n\r\n /**\r\n * Gets or sets the node material used to create this texture (null if the texture was manually created)\r\n */\r\n public nodeMaterialSource: Nullable = null;\r\n\r\n /** @internal */\r\n @serialize()\r\n public _generateMipMaps: boolean;\r\n\r\n private _drawWrapper: DrawWrapper;\r\n\r\n /** @internal */\r\n public _textures: { [key: string]: Texture } = {};\r\n\r\n /** @internal */\r\n protected _fallbackTexture: Nullable;\r\n\r\n @serialize()\r\n private _size: TextureSize;\r\n private _textureType: number;\r\n private _currentRefreshId = -1;\r\n private _frameId = -1;\r\n private _refreshRate = 1;\r\n private _vertexBuffers: { [key: string]: Nullable } = {};\r\n private _indexBuffer: Nullable;\r\n private _uniforms = new Array();\r\n private _samplers = new Array();\r\n private _fragment: any;\r\n\r\n private _floats: { [key: string]: number } = {};\r\n private _ints: { [key: string]: number } = {};\r\n private _floatsArrays: { [key: string]: number[] } = {};\r\n private _colors3: { [key: string]: Color3 } = {};\r\n private _colors4: { [key: string]: Color4 } = {};\r\n private _vectors2: { [key: string]: Vector2 } = {};\r\n private _vectors3: { [key: string]: Vector3 } = {};\r\n private _matrices: { [key: string]: Matrix } = {};\r\n\r\n private _fallbackTextureUsed = false;\r\n private _fullEngine: Engine;\r\n\r\n private _cachedDefines: Nullable = null;\r\n\r\n private _contentUpdateId = -1;\r\n private _contentData: Nullable>;\r\n\r\n private _rtWrapper: Nullable = null;\r\n\r\n /**\r\n * Instantiates a new procedural texture.\r\n * Procedural texturing is a way to programmatically create a texture. There are 2 types of procedural textures: code-only, and code that references some classic 2D images, sometimes called 'refMaps' or 'sampler' images.\r\n * This is the base class of any Procedural texture and contains most of the shareable code.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/proceduralTextures\r\n * @param name Define the name of the texture\r\n * @param size Define the size of the texture to create\r\n * @param fragment Define the fragment shader to use to generate the texture or null if it is defined later\r\n * @param scene Define the scene the texture belongs to\r\n * @param fallbackTexture Define a fallback texture in case there were issues to create the custom texture\r\n * @param generateMipMaps Define if the texture should creates mip maps or not\r\n * @param isCube Define if the texture is a cube texture or not (this will render each faces of the cube)\r\n * @param textureType The FBO internal texture type\r\n */\r\n constructor(\r\n name: string,\r\n size: TextureSize,\r\n fragment: any,\r\n scene: Nullable,\r\n fallbackTexture: Nullable = null,\r\n generateMipMaps = true,\r\n isCube = false,\r\n textureType = Constants.TEXTURETYPE_UNSIGNED_INT\r\n ) {\r\n super(null, scene, !generateMipMaps);\r\n\r\n scene = this.getScene() || EngineStore.LastCreatedScene!;\r\n let component = scene._getComponent(SceneComponentConstants.NAME_PROCEDURALTEXTURE);\r\n if (!component) {\r\n component = new ProceduralTextureSceneComponent(scene);\r\n scene._addComponent(component);\r\n }\r\n scene.proceduralTextures.push(this);\r\n\r\n this._fullEngine = scene.getEngine();\r\n\r\n this.name = name;\r\n this.isRenderTarget = true;\r\n this._size = size;\r\n this._textureType = textureType;\r\n this._generateMipMaps = generateMipMaps;\r\n this._drawWrapper = new DrawWrapper(this._fullEngine);\r\n\r\n this.setFragment(fragment);\r\n\r\n this._fallbackTexture = fallbackTexture;\r\n\r\n const rtWrapper = this._createRtWrapper(isCube, size, generateMipMaps, textureType);\r\n this._texture = rtWrapper.texture;\r\n\r\n // VBO\r\n const vertices = [];\r\n vertices.push(1, 1);\r\n vertices.push(-1, 1);\r\n vertices.push(-1, -1);\r\n vertices.push(1, -1);\r\n\r\n this._vertexBuffers[VertexBuffer.PositionKind] = new VertexBuffer(this._fullEngine, vertices, VertexBuffer.PositionKind, false, false, 2);\r\n\r\n this._createIndexBuffer();\r\n }\r\n\r\n private _createRtWrapper(isCube: boolean, size: TextureSize, generateMipMaps: boolean, textureType: number) {\r\n if (isCube) {\r\n this._rtWrapper = this._fullEngine.createRenderTargetCubeTexture(size as number, {\r\n generateMipMaps: generateMipMaps,\r\n generateDepthBuffer: false,\r\n generateStencilBuffer: false,\r\n type: textureType,\r\n });\r\n this.setFloat(\"face\", 0);\r\n } else {\r\n this._rtWrapper = this._fullEngine.createRenderTargetTexture(size, {\r\n generateMipMaps: generateMipMaps,\r\n generateDepthBuffer: false,\r\n generateStencilBuffer: false,\r\n type: textureType,\r\n });\r\n }\r\n return this._rtWrapper;\r\n }\r\n\r\n /**\r\n * The effect that is created when initializing the post process.\r\n * @returns The created effect corresponding the the postprocess.\r\n */\r\n public getEffect(): Effect {\r\n return this._drawWrapper.effect!;\r\n }\r\n\r\n /**\r\n * @internal*\r\n */\r\n public _setEffect(effect: Effect) {\r\n this._drawWrapper.effect = effect;\r\n }\r\n\r\n /**\r\n * Gets texture content (Use this function wisely as reading from a texture can be slow)\r\n * @returns an ArrayBufferView promise (Uint8Array or Float32Array)\r\n */\r\n public getContent(): Nullable> {\r\n if (this._contentData && this._frameId === this._contentUpdateId) {\r\n return this._contentData;\r\n }\r\n\r\n if (this._contentData) {\r\n this._contentData.then((buffer) => {\r\n this._contentData = this.readPixels(0, 0, buffer);\r\n this._contentUpdateId = this._frameId;\r\n });\r\n } else {\r\n this._contentData = this.readPixels(0, 0);\r\n this._contentUpdateId = this._frameId;\r\n }\r\n\r\n return this._contentData;\r\n }\r\n\r\n private _createIndexBuffer(): void {\r\n const engine = this._fullEngine;\r\n\r\n // Indices\r\n const indices = [];\r\n indices.push(0);\r\n indices.push(1);\r\n indices.push(2);\r\n\r\n indices.push(0);\r\n indices.push(2);\r\n indices.push(3);\r\n\r\n this._indexBuffer = engine.createIndexBuffer(indices);\r\n }\r\n\r\n /** @internal */\r\n public _rebuild(): void {\r\n const vb = this._vertexBuffers[VertexBuffer.PositionKind];\r\n\r\n if (vb) {\r\n vb._rebuild();\r\n }\r\n\r\n this._createIndexBuffer();\r\n\r\n if (this.refreshRate === RenderTargetTexture.REFRESHRATE_RENDER_ONCE) {\r\n this.refreshRate = RenderTargetTexture.REFRESHRATE_RENDER_ONCE;\r\n }\r\n }\r\n\r\n /**\r\n * Resets the texture in order to recreate its associated resources.\r\n * This can be called in case of context loss\r\n */\r\n public reset(): void {\r\n this._drawWrapper.effect?.dispose();\r\n }\r\n\r\n protected _getDefines(): string {\r\n return \"\";\r\n }\r\n\r\n /**\r\n * Is the texture ready to be used ? (rendered at least once)\r\n * @returns true if ready, otherwise, false.\r\n */\r\n public isReady(): boolean {\r\n const engine = this._fullEngine;\r\n let shaders;\r\n\r\n if (this.nodeMaterialSource) {\r\n return this._drawWrapper.effect!.isReady();\r\n }\r\n\r\n if (!this._fragment) {\r\n return false;\r\n }\r\n\r\n if (this._fallbackTextureUsed) {\r\n return true;\r\n }\r\n\r\n if (!this._texture) {\r\n return false;\r\n }\r\n\r\n const defines = this._getDefines();\r\n if (this._drawWrapper.effect && defines === this._cachedDefines && this._drawWrapper.effect.isReady()) {\r\n return true;\r\n }\r\n\r\n if (this._fragment.fragmentElement !== undefined) {\r\n shaders = { vertex: \"procedural\", fragmentElement: this._fragment.fragmentElement };\r\n } else {\r\n shaders = { vertex: \"procedural\", fragment: this._fragment };\r\n }\r\n\r\n if (this._cachedDefines !== defines) {\r\n this._cachedDefines = defines;\r\n\r\n this._drawWrapper.effect = engine.createEffect(shaders, [VertexBuffer.PositionKind], this._uniforms, this._samplers, defines, undefined, undefined, () => {\r\n this._rtWrapper?.dispose();\r\n this._rtWrapper = this._texture = null;\r\n\r\n if (this._fallbackTexture) {\r\n this._texture = this._fallbackTexture._texture;\r\n\r\n if (this._texture) {\r\n this._texture.incrementReferences();\r\n }\r\n }\r\n\r\n this._fallbackTextureUsed = true;\r\n });\r\n }\r\n\r\n return this._drawWrapper.effect!.isReady();\r\n }\r\n\r\n /**\r\n * Resets the refresh counter of the texture and start bak from scratch.\r\n * Could be useful to regenerate the texture if it is setup to render only once.\r\n */\r\n public resetRefreshCounter(): void {\r\n this._currentRefreshId = -1;\r\n }\r\n\r\n /**\r\n * Set the fragment shader to use in order to render the texture.\r\n * @param fragment This can be set to a path (into the shader store) or to a json object containing a fragmentElement property.\r\n */\r\n public setFragment(fragment: any) {\r\n this._fragment = fragment;\r\n }\r\n\r\n /**\r\n * Define the refresh rate of the texture or the rendering frequency.\r\n * Use 0 to render just once, 1 to render on every frame, 2 to render every two frames and so on...\r\n */\r\n @serialize()\r\n public get refreshRate(): number {\r\n return this._refreshRate;\r\n }\r\n\r\n public set refreshRate(value: number) {\r\n this._refreshRate = value;\r\n this.resetRefreshCounter();\r\n }\r\n\r\n /** @internal */\r\n public _shouldRender(): boolean {\r\n if (!this.isEnabled || !this.isReady() || !this._texture) {\r\n if (this._texture) {\r\n this._texture.isReady = false;\r\n }\r\n return false;\r\n }\r\n\r\n if (this._fallbackTextureUsed) {\r\n return false;\r\n }\r\n\r\n if (this._currentRefreshId === -1) {\r\n // At least render once\r\n this._currentRefreshId = 1;\r\n this._frameId++;\r\n return true;\r\n }\r\n\r\n if (this.refreshRate === this._currentRefreshId) {\r\n this._currentRefreshId = 1;\r\n this._frameId++;\r\n return true;\r\n }\r\n\r\n this._currentRefreshId++;\r\n return false;\r\n }\r\n\r\n /**\r\n * Get the size the texture is rendering at.\r\n * @returns the size (on cube texture it is always squared)\r\n */\r\n public getRenderSize(): TextureSize {\r\n return this._size;\r\n }\r\n\r\n /**\r\n * Resize the texture to new value.\r\n * @param size Define the new size the texture should have\r\n * @param generateMipMaps Define whether the new texture should create mip maps\r\n */\r\n public resize(size: TextureSize, generateMipMaps: boolean): void {\r\n if (this._fallbackTextureUsed || !this._rtWrapper || !this._texture) {\r\n return;\r\n }\r\n\r\n const isCube = this._texture.isCube;\r\n this._rtWrapper.dispose();\r\n\r\n const rtWrapper = this._createRtWrapper(isCube, size, generateMipMaps, this._textureType);\r\n this._texture = rtWrapper.texture;\r\n\r\n // Update properties\r\n this._size = size;\r\n this._generateMipMaps = generateMipMaps;\r\n }\r\n\r\n private _checkUniform(uniformName: string): void {\r\n if (this._uniforms.indexOf(uniformName) === -1) {\r\n this._uniforms.push(uniformName);\r\n }\r\n }\r\n\r\n /**\r\n * Set a texture in the shader program used to render.\r\n * @param name Define the name of the uniform samplers as defined in the shader\r\n * @param texture Define the texture to bind to this sampler\r\n * @returns the texture itself allowing \"fluent\" like uniform updates\r\n */\r\n public setTexture(name: string, texture: Texture): ProceduralTexture {\r\n if (this._samplers.indexOf(name) === -1) {\r\n this._samplers.push(name);\r\n }\r\n this._textures[name] = texture;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a float in the shader.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the texture itself allowing \"fluent\" like uniform updates\r\n */\r\n public setFloat(name: string, value: number): ProceduralTexture {\r\n this._checkUniform(name);\r\n this._floats[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a int in the shader.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the texture itself allowing \"fluent\" like uniform updates\r\n */\r\n public setInt(name: string, value: number): ProceduralTexture {\r\n this._checkUniform(name);\r\n this._ints[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set an array of floats in the shader.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the texture itself allowing \"fluent\" like uniform updates\r\n */\r\n public setFloats(name: string, value: number[]): ProceduralTexture {\r\n this._checkUniform(name);\r\n this._floatsArrays[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a vec3 in the shader from a Color3.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the texture itself allowing \"fluent\" like uniform updates\r\n */\r\n public setColor3(name: string, value: Color3): ProceduralTexture {\r\n this._checkUniform(name);\r\n this._colors3[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a vec4 in the shader from a Color4.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the texture itself allowing \"fluent\" like uniform updates\r\n */\r\n public setColor4(name: string, value: Color4): ProceduralTexture {\r\n this._checkUniform(name);\r\n this._colors4[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a vec2 in the shader from a Vector2.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the texture itself allowing \"fluent\" like uniform updates\r\n */\r\n public setVector2(name: string, value: Vector2): ProceduralTexture {\r\n this._checkUniform(name);\r\n this._vectors2[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a vec3 in the shader from a Vector3.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the texture itself allowing \"fluent\" like uniform updates\r\n */\r\n public setVector3(name: string, value: Vector3): ProceduralTexture {\r\n this._checkUniform(name);\r\n this._vectors3[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a mat4 in the shader from a MAtrix.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the texture itself allowing \"fluent\" like uniform updates\r\n */\r\n public setMatrix(name: string, value: Matrix): ProceduralTexture {\r\n this._checkUniform(name);\r\n this._matrices[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Render the texture to its associated render target.\r\n * @param useCameraPostProcess Define if camera post process should be applied to the texture\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public render(useCameraPostProcess?: boolean): void {\r\n const scene = this.getScene();\r\n\r\n if (!scene) {\r\n return;\r\n }\r\n\r\n const engine = this._fullEngine;\r\n\r\n // Render\r\n engine.enableEffect(this._drawWrapper);\r\n this.onBeforeGenerationObservable.notifyObservers(this);\r\n engine.setState(false);\r\n\r\n if (!this.nodeMaterialSource) {\r\n // Texture\r\n for (const name in this._textures) {\r\n this._drawWrapper.effect!.setTexture(name, this._textures[name]);\r\n }\r\n\r\n // Float\r\n for (const name in this._ints) {\r\n this._drawWrapper.effect!.setInt(name, this._ints[name]);\r\n }\r\n\r\n // Float\r\n for (const name in this._floats) {\r\n this._drawWrapper.effect!.setFloat(name, this._floats[name]);\r\n }\r\n\r\n // Floats\r\n for (const name in this._floatsArrays) {\r\n this._drawWrapper.effect!.setArray(name, this._floatsArrays[name]);\r\n }\r\n\r\n // Color3\r\n for (const name in this._colors3) {\r\n this._drawWrapper.effect!.setColor3(name, this._colors3[name]);\r\n }\r\n\r\n // Color4\r\n for (const name in this._colors4) {\r\n const color = this._colors4[name];\r\n this._drawWrapper.effect!.setFloat4(name, color.r, color.g, color.b, color.a);\r\n }\r\n\r\n // Vector2\r\n for (const name in this._vectors2) {\r\n this._drawWrapper.effect!.setVector2(name, this._vectors2[name]);\r\n }\r\n\r\n // Vector3\r\n for (const name in this._vectors3) {\r\n this._drawWrapper.effect!.setVector3(name, this._vectors3[name]);\r\n }\r\n\r\n // Matrix\r\n for (const name in this._matrices) {\r\n this._drawWrapper.effect!.setMatrix(name, this._matrices[name]);\r\n }\r\n }\r\n\r\n if (!this._texture || !this._rtWrapper) {\r\n return;\r\n }\r\n\r\n engine._debugPushGroup?.(`procedural texture generation for ${this.name}`, 1);\r\n\r\n const viewPort = engine.currentViewport;\r\n if (this.isCube) {\r\n for (let face = 0; face < 6; face++) {\r\n engine.bindFramebuffer(this._rtWrapper, face, undefined, undefined, true);\r\n\r\n // VBOs\r\n engine.bindBuffers(this._vertexBuffers, this._indexBuffer, this._drawWrapper.effect!);\r\n\r\n this._drawWrapper.effect!.setFloat(\"face\", face);\r\n\r\n // Clear\r\n if (this.autoClear) {\r\n engine.clear(scene.clearColor, true, false, false);\r\n }\r\n\r\n // Draw order\r\n engine.drawElementsType(Material.TriangleFillMode, 0, 6);\r\n }\r\n } else {\r\n engine.bindFramebuffer(this._rtWrapper, 0, undefined, undefined, true);\r\n\r\n // VBOs\r\n engine.bindBuffers(this._vertexBuffers, this._indexBuffer, this._drawWrapper.effect!);\r\n\r\n // Clear\r\n if (this.autoClear) {\r\n engine.clear(scene.clearColor, true, false, false);\r\n }\r\n\r\n // Draw order\r\n engine.drawElementsType(Material.TriangleFillMode, 0, 6);\r\n }\r\n\r\n // Unbind and restore viewport\r\n engine.unBindFramebuffer(this._rtWrapper, this.isCube);\r\n if (viewPort) {\r\n engine.setViewport(viewPort);\r\n }\r\n\r\n // Mipmaps\r\n if (this.isCube) {\r\n engine.generateMipMapsForCubemap(this._texture);\r\n }\r\n\r\n engine._debugPopGroup?.(1);\r\n\r\n if (this.onGenerated) {\r\n this.onGenerated();\r\n }\r\n\r\n this.onGeneratedObservable.notifyObservers(this);\r\n }\r\n\r\n /**\r\n * Clone the texture.\r\n * @returns the cloned texture\r\n */\r\n public clone(): ProceduralTexture {\r\n const textureSize = this.getSize();\r\n const newTexture = new ProceduralTexture(this.name, textureSize.width, this._fragment, this.getScene(), this._fallbackTexture, this._generateMipMaps);\r\n\r\n // Base texture\r\n newTexture.hasAlpha = this.hasAlpha;\r\n newTexture.level = this.level;\r\n\r\n // RenderTarget Texture\r\n newTexture.coordinatesMode = this.coordinatesMode;\r\n\r\n return newTexture;\r\n }\r\n\r\n /**\r\n * Dispose the texture and release its associated resources.\r\n */\r\n public dispose(): void {\r\n const scene = this.getScene();\r\n\r\n if (!scene) {\r\n return;\r\n }\r\n\r\n const index = scene.proceduralTextures.indexOf(this);\r\n\r\n if (index >= 0) {\r\n scene.proceduralTextures.splice(index, 1);\r\n }\r\n\r\n const vertexBuffer = this._vertexBuffers[VertexBuffer.PositionKind];\r\n if (vertexBuffer) {\r\n vertexBuffer.dispose();\r\n this._vertexBuffers[VertexBuffer.PositionKind] = null;\r\n }\r\n\r\n if (this._indexBuffer && this._fullEngine._releaseBuffer(this._indexBuffer)) {\r\n this._indexBuffer = null;\r\n }\r\n\r\n this.onGeneratedObservable.clear();\r\n this.onBeforeGenerationObservable.clear();\r\n\r\n super.dispose();\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.ProceduralTexture\", ProceduralTexture);\r\n","import { NodeMaterialBlock } from \"../nodeMaterialBlock\";\r\nimport { NodeMaterialBlockConnectionPointTypes } from \"../Enums/nodeMaterialBlockConnectionPointTypes\";\r\nimport type { NodeMaterialBuildState } from \"../nodeMaterialBuildState\";\r\nimport type { NodeMaterialConnectionPoint } from \"../nodeMaterialBlockConnectionPoint\";\r\nimport { NodeMaterialBlockTargets } from \"../Enums/nodeMaterialBlockTargets\";\r\nimport { RegisterClass } from \"../../../Misc/typeStore\";\r\nimport type { Scene } from \"../../../scene\";\r\n\r\n/**\r\n * Operations supported by the Trigonometry block\r\n */\r\nexport enum TrigonometryBlockOperations {\r\n /** Cos */\r\n Cos,\r\n /** Sin */\r\n Sin,\r\n /** Abs */\r\n Abs,\r\n /** Exp */\r\n Exp,\r\n /** Exp2 */\r\n Exp2,\r\n /** Round */\r\n Round,\r\n /** Floor */\r\n Floor,\r\n /** Ceiling */\r\n Ceiling,\r\n /** Square root */\r\n Sqrt,\r\n /** Log */\r\n Log,\r\n /** Tangent */\r\n Tan,\r\n /** Arc tangent */\r\n ArcTan,\r\n /** Arc cosinus */\r\n ArcCos,\r\n /** Arc sinus */\r\n ArcSin,\r\n /** Fraction */\r\n Fract,\r\n /** Sign */\r\n Sign,\r\n /** To radians (from degrees) */\r\n Radians,\r\n /** To degrees (from radians) */\r\n Degrees,\r\n}\r\n\r\n/**\r\n * Block used to apply trigonometry operation to floats\r\n */\r\nexport class TrigonometryBlock extends NodeMaterialBlock {\r\n /**\r\n * Gets or sets the operation applied by the block\r\n */\r\n public operation = TrigonometryBlockOperations.Cos;\r\n\r\n /**\r\n * Creates a new TrigonometryBlock\r\n * @param name defines the block name\r\n */\r\n public constructor(name: string) {\r\n super(name, NodeMaterialBlockTargets.Neutral);\r\n\r\n this.registerInput(\"input\", NodeMaterialBlockConnectionPointTypes.AutoDetect);\r\n this.registerOutput(\"output\", NodeMaterialBlockConnectionPointTypes.BasedOnInput);\r\n\r\n this._outputs[0]._typeConnectionSource = this._inputs[0];\r\n }\r\n\r\n /**\r\n * Gets the current class name\r\n * @returns the class name\r\n */\r\n public getClassName() {\r\n return \"TrigonometryBlock\";\r\n }\r\n\r\n /**\r\n * Gets the input component\r\n */\r\n public get input(): NodeMaterialConnectionPoint {\r\n return this._inputs[0];\r\n }\r\n\r\n /**\r\n * Gets the output component\r\n */\r\n public get output(): NodeMaterialConnectionPoint {\r\n return this._outputs[0];\r\n }\r\n\r\n protected _buildBlock(state: NodeMaterialBuildState) {\r\n super._buildBlock(state);\r\n\r\n const output = this._outputs[0];\r\n let operation = \"\";\r\n\r\n switch (this.operation) {\r\n case TrigonometryBlockOperations.Cos: {\r\n operation = \"cos\";\r\n break;\r\n }\r\n case TrigonometryBlockOperations.Sin: {\r\n operation = \"sin\";\r\n break;\r\n }\r\n case TrigonometryBlockOperations.Abs: {\r\n operation = \"abs\";\r\n break;\r\n }\r\n case TrigonometryBlockOperations.Exp: {\r\n operation = \"exp\";\r\n break;\r\n }\r\n case TrigonometryBlockOperations.Exp2: {\r\n operation = \"exp2\";\r\n break;\r\n }\r\n case TrigonometryBlockOperations.Round: {\r\n operation = \"round\";\r\n break;\r\n }\r\n case TrigonometryBlockOperations.Floor: {\r\n operation = \"floor\";\r\n break;\r\n }\r\n case TrigonometryBlockOperations.Ceiling: {\r\n operation = \"ceil\";\r\n break;\r\n }\r\n case TrigonometryBlockOperations.Sqrt: {\r\n operation = \"sqrt\";\r\n break;\r\n }\r\n case TrigonometryBlockOperations.Log: {\r\n operation = \"log\";\r\n break;\r\n }\r\n case TrigonometryBlockOperations.Tan: {\r\n operation = \"tan\";\r\n break;\r\n }\r\n case TrigonometryBlockOperations.ArcTan: {\r\n operation = \"atan\";\r\n break;\r\n }\r\n case TrigonometryBlockOperations.ArcCos: {\r\n operation = \"acos\";\r\n break;\r\n }\r\n case TrigonometryBlockOperations.ArcSin: {\r\n operation = \"asin\";\r\n break;\r\n }\r\n case TrigonometryBlockOperations.Fract: {\r\n operation = \"fract\";\r\n break;\r\n }\r\n case TrigonometryBlockOperations.Sign: {\r\n operation = \"sign\";\r\n break;\r\n }\r\n case TrigonometryBlockOperations.Radians: {\r\n operation = \"radians\";\r\n break;\r\n }\r\n case TrigonometryBlockOperations.Degrees: {\r\n operation = \"degrees\";\r\n break;\r\n }\r\n }\r\n\r\n state.compilationString += this._declareOutput(output, state) + ` = ${operation}(${this.input.associatedVariableName});\\n`;\r\n\r\n return this;\r\n }\r\n\r\n public serialize(): any {\r\n const serializationObject = super.serialize();\r\n\r\n serializationObject.operation = this.operation;\r\n\r\n return serializationObject;\r\n }\r\n\r\n public _deserialize(serializationObject: any, scene: Scene, rootUrl: string) {\r\n super._deserialize(serializationObject, scene, rootUrl);\r\n\r\n this.operation = serializationObject.operation;\r\n }\r\n\r\n protected _dumpPropertiesCode() {\r\n const codeString =\r\n super._dumpPropertiesCode() + `${this._codeVariableName}.operation = BABYLON.TrigonometryBlockOperations.${TrigonometryBlockOperations[this.operation]};\\n`;\r\n return codeString;\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.TrigonometryBlock\", TrigonometryBlock);\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport type { NodeMaterialBlock } from \"./nodeMaterialBlock\";\r\nimport { PushMaterial } from \"../pushMaterial\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport { Matrix, Vector2 } from \"../../Maths/math.vector\";\r\nimport { Color3, Color4 } from \"../../Maths/math.color\";\r\nimport type { Mesh } from \"../../Meshes/mesh\";\r\nimport { Engine } from \"../../Engines/engine\";\r\nimport { NodeMaterialBuildState } from \"./nodeMaterialBuildState\";\r\nimport type { IEffectCreationOptions } from \"../effect\";\r\nimport { Effect } from \"../effect\";\r\nimport type { BaseTexture } from \"../../Materials/Textures/baseTexture\";\r\nimport type { Observer } from \"../../Misc/observable\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport { NodeMaterialBlockTargets } from \"./Enums/nodeMaterialBlockTargets\";\r\nimport { NodeMaterialBuildStateSharedData } from \"./nodeMaterialBuildStateSharedData\";\r\nimport type { SubMesh } from \"../../Meshes/subMesh\";\r\nimport { MaterialDefines } from \"../../Materials/materialDefines\";\r\nimport type { NodeMaterialOptimizer } from \"./Optimizers/nodeMaterialOptimizer\";\r\nimport type { ImageProcessingConfiguration, IImageProcessingConfigurationDefines } from \"../imageProcessingConfiguration\";\r\nimport type { Nullable } from \"../../types\";\r\nimport { VertexBuffer } from \"../../Buffers/buffer\";\r\nimport { Tools } from \"../../Misc/tools\";\r\nimport { TransformBlock } from \"./Blocks/transformBlock\";\r\nimport { VertexOutputBlock } from \"./Blocks/Vertex/vertexOutputBlock\";\r\nimport { FragmentOutputBlock } from \"./Blocks/Fragment/fragmentOutputBlock\";\r\nimport { InputBlock } from \"./Blocks/Input/inputBlock\";\r\nimport { GetClass, RegisterClass } from \"../../Misc/typeStore\";\r\nimport { serialize, SerializationHelper } from \"../../Misc/decorators\";\r\nimport type { TextureBlock } from \"./Blocks/Dual/textureBlock\";\r\nimport type { ReflectionTextureBaseBlock } from \"./Blocks/Dual/reflectionTextureBaseBlock\";\r\nimport type { RefractionBlock } from \"./Blocks/PBR/refractionBlock\";\r\nimport { CurrentScreenBlock } from \"./Blocks/Dual/currentScreenBlock\";\r\nimport { ParticleTextureBlock } from \"./Blocks/Particle/particleTextureBlock\";\r\nimport { ParticleRampGradientBlock } from \"./Blocks/Particle/particleRampGradientBlock\";\r\nimport { ParticleBlendMultiplyBlock } from \"./Blocks/Particle/particleBlendMultiplyBlock\";\r\nimport { EffectFallbacks } from \"../effectFallbacks\";\r\nimport { WebRequest } from \"../../Misc/webRequest\";\r\n\r\nimport type { PostProcessOptions } from \"../../PostProcesses/postProcess\";\r\nimport { PostProcess } from \"../../PostProcesses/postProcess\";\r\nimport { Constants } from \"../../Engines/constants\";\r\nimport type { Camera } from \"../../Cameras/camera\";\r\nimport { VectorMergerBlock } from \"./Blocks/vectorMergerBlock\";\r\nimport { RemapBlock } from \"./Blocks/remapBlock\";\r\nimport { MultiplyBlock } from \"./Blocks/multiplyBlock\";\r\nimport { NodeMaterialModes } from \"./Enums/nodeMaterialModes\";\r\nimport { Texture } from \"../Textures/texture\";\r\nimport type { IParticleSystem } from \"../../Particles/IParticleSystem\";\r\nimport { BaseParticleSystem } from \"../../Particles/baseParticleSystem\";\r\nimport { ColorSplitterBlock } from \"./Blocks/colorSplitterBlock\";\r\nimport { TimingTools } from \"../../Misc/timingTools\";\r\nimport { ProceduralTexture } from \"../Textures/Procedurals/proceduralTexture\";\r\nimport { AnimatedInputBlockTypes } from \"./Blocks/Input/animatedInputBlockTypes\";\r\nimport { TrigonometryBlock, TrigonometryBlockOperations } from \"./Blocks/trigonometryBlock\";\r\nimport { NodeMaterialSystemValues } from \"./Enums/nodeMaterialSystemValues\";\r\nimport type { ImageSourceBlock } from \"./Blocks/Dual/imageSourceBlock\";\r\nimport { EngineStore } from \"../../Engines/engineStore\";\r\nimport type { Material } from \"../material\";\r\nimport { MaterialHelper } from \"../materialHelper\";\r\nimport type { TriPlanarBlock } from \"./Blocks/triPlanarBlock\";\r\nimport type { BiPlanarBlock } from \"./Blocks/biPlanarBlock\";\r\nimport type { NodeMaterialTeleportOutBlock } from \"./Blocks/Teleport/teleportOutBlock\";\r\nimport type { NodeMaterialTeleportInBlock } from \"./Blocks/Teleport/teleportInBlock\";\r\n\r\nconst onCreatedEffectParameters = { effect: null as unknown as Effect, subMesh: null as unknown as Nullable };\r\n\r\n// declare NODEEDITOR namespace for compilation issue\r\ndeclare let NODEEDITOR: any;\r\ndeclare let BABYLON: any;\r\n\r\n/**\r\n * Interface used to configure the node material editor\r\n */\r\nexport interface INodeMaterialEditorOptions {\r\n /** Define the URL to load node editor script from */\r\n editorURL?: string;\r\n /** Additional configuration for the NME */\r\n nodeEditorConfig?: {\r\n backgroundColor?: Color4;\r\n };\r\n}\r\n\r\n/** @internal */\r\nexport class NodeMaterialDefines extends MaterialDefines implements IImageProcessingConfigurationDefines {\r\n public NORMAL = false;\r\n public TANGENT = false;\r\n public VERTEXCOLOR_NME = false;\r\n public UV1 = false;\r\n public UV2 = false;\r\n public UV3 = false;\r\n public UV4 = false;\r\n public UV5 = false;\r\n public UV6 = false;\r\n\r\n /** BONES */\r\n public NUM_BONE_INFLUENCERS = 0;\r\n public BonesPerMesh = 0;\r\n public BONETEXTURE = false;\r\n\r\n /** MORPH TARGETS */\r\n public MORPHTARGETS = false;\r\n public MORPHTARGETS_NORMAL = false;\r\n public MORPHTARGETS_TANGENT = false;\r\n public MORPHTARGETS_UV = false;\r\n public NUM_MORPH_INFLUENCERS = 0;\r\n public MORPHTARGETS_TEXTURE = false;\r\n\r\n /** IMAGE PROCESSING */\r\n public IMAGEPROCESSING = false;\r\n public VIGNETTE = false;\r\n public VIGNETTEBLENDMODEMULTIPLY = false;\r\n public VIGNETTEBLENDMODEOPAQUE = false;\r\n public TONEMAPPING = false;\r\n public TONEMAPPING_ACES = false;\r\n public CONTRAST = false;\r\n public EXPOSURE = false;\r\n public COLORCURVES = false;\r\n public COLORGRADING = false;\r\n public COLORGRADING3D = false;\r\n public SAMPLER3DGREENDEPTH = false;\r\n public SAMPLER3DBGRMAP = false;\r\n public DITHER = false;\r\n public IMAGEPROCESSINGPOSTPROCESS = false;\r\n public SKIPFINALCOLORCLAMP = false;\r\n\r\n /** MISC. */\r\n public BUMPDIRECTUV = 0;\r\n public CAMERA_ORTHOGRAPHIC = false;\r\n public CAMERA_PERSPECTIVE = false;\r\n\r\n constructor() {\r\n super();\r\n this.rebuild();\r\n }\r\n\r\n public setValue(name: string, value: any, markAsUnprocessedIfDirty = false) {\r\n if (this[name] === undefined) {\r\n this._keys.push(name);\r\n }\r\n\r\n if (markAsUnprocessedIfDirty && this[name] !== value) {\r\n this.markAsUnprocessed();\r\n }\r\n\r\n this[name] = value;\r\n }\r\n}\r\n\r\n/**\r\n * Class used to configure NodeMaterial\r\n */\r\nexport interface INodeMaterialOptions {\r\n /**\r\n * Defines if blocks should emit comments\r\n */\r\n emitComments: boolean;\r\n}\r\n\r\n/**\r\n * Blocks that manage a texture\r\n */\r\nexport type NodeMaterialTextureBlocks =\r\n | TextureBlock\r\n | ReflectionTextureBaseBlock\r\n | RefractionBlock\r\n | CurrentScreenBlock\r\n | ParticleTextureBlock\r\n | ImageSourceBlock\r\n | TriPlanarBlock\r\n | BiPlanarBlock;\r\n\r\n/**\r\n * Class used to create a node based material built by assembling shader blocks\r\n */\r\nexport class NodeMaterial extends PushMaterial {\r\n private static _BuildIdGenerator: number = 0;\r\n private _options: INodeMaterialOptions;\r\n private _vertexCompilationState: NodeMaterialBuildState;\r\n private _fragmentCompilationState: NodeMaterialBuildState;\r\n private _sharedData: NodeMaterialBuildStateSharedData;\r\n private _buildId: number = NodeMaterial._BuildIdGenerator++;\r\n private _buildWasSuccessful = false;\r\n private _cachedWorldViewMatrix = new Matrix();\r\n private _cachedWorldViewProjectionMatrix = new Matrix();\r\n private _optimizers = new Array();\r\n private _animationFrame = -1;\r\n\r\n /** Define the Url to load node editor script */\r\n public static EditorURL = `https://unpkg.com/babylonjs-node-editor@${Engine.Version}/babylon.nodeEditor.js`;\r\n\r\n /** Define the Url to load snippets */\r\n public static SnippetUrl = Constants.SnippetUrl;\r\n\r\n /** Gets or sets a boolean indicating that node materials should not deserialize textures from json / snippet content */\r\n public static IgnoreTexturesAtLoadTime = false;\r\n\r\n /**\r\n * Checks if a block is a texture block\r\n * @param block The block to check\r\n * @returns True if the block is a texture block\r\n */\r\n public static _BlockIsTextureBlock(block: NodeMaterialBlock): block is NodeMaterialTextureBlocks {\r\n return (\r\n block.getClassName() === \"TextureBlock\" ||\r\n block.getClassName() === \"ReflectionTextureBaseBlock\" ||\r\n block.getClassName() === \"RefractionBlock\" ||\r\n block.getClassName() === \"CurrentScreenBlock\" ||\r\n block.getClassName() === \"ParticleTextureBlock\" ||\r\n block.getClassName() === \"ImageSourceBlock\" ||\r\n block.getClassName() === \"TriPlanarBlock\" ||\r\n block.getClassName() === \"BiPlanarBlock\"\r\n );\r\n }\r\n\r\n private BJSNODEMATERIALEDITOR = this._getGlobalNodeMaterialEditor();\r\n\r\n /** Get the inspector from bundle or global */\r\n private _getGlobalNodeMaterialEditor(): any {\r\n // UMD Global name detection from Webpack Bundle UMD Name.\r\n if (typeof NODEEDITOR !== \"undefined\") {\r\n return NODEEDITOR;\r\n }\r\n\r\n // In case of module let's check the global emitted from the editor entry point.\r\n if (typeof BABYLON !== \"undefined\" && typeof BABYLON.NodeEditor !== \"undefined\") {\r\n return BABYLON;\r\n }\r\n\r\n return undefined;\r\n }\r\n\r\n /**\r\n * Snippet ID if the material was created from the snippet server\r\n */\r\n public snippetId: string;\r\n\r\n /**\r\n * Gets or sets data used by visual editor\r\n * @see https://nme.babylonjs.com\r\n */\r\n public editorData: any = null;\r\n\r\n /**\r\n * Gets or sets a boolean indicating that alpha value must be ignored (This will turn alpha blending off even if an alpha value is produced by the material)\r\n */\r\n @serialize()\r\n public ignoreAlpha = false;\r\n\r\n /**\r\n * Defines the maximum number of lights that can be used in the material\r\n */\r\n @serialize()\r\n public maxSimultaneousLights = 4;\r\n\r\n /**\r\n * Observable raised when the material is built\r\n */\r\n public onBuildObservable = new Observable();\r\n\r\n /**\r\n * Gets or sets the root nodes of the material vertex shader\r\n */\r\n public _vertexOutputNodes = new Array();\r\n\r\n /**\r\n * Gets or sets the root nodes of the material fragment (pixel) shader\r\n */\r\n public _fragmentOutputNodes = new Array();\r\n\r\n /** Gets or sets options to control the node material overall behavior */\r\n public get options() {\r\n return this._options;\r\n }\r\n\r\n public set options(options: INodeMaterialOptions) {\r\n this._options = options;\r\n }\r\n\r\n /**\r\n * Default configuration related to image processing available in the standard Material.\r\n */\r\n protected _imageProcessingConfiguration: ImageProcessingConfiguration;\r\n\r\n /**\r\n * Gets the image processing configuration used either in this material.\r\n */\r\n public get imageProcessingConfiguration(): ImageProcessingConfiguration {\r\n return this._imageProcessingConfiguration;\r\n }\r\n\r\n /**\r\n * Sets the Default image processing configuration used either in the this material.\r\n *\r\n * If sets to null, the scene one is in use.\r\n */\r\n public set imageProcessingConfiguration(value: ImageProcessingConfiguration) {\r\n this._attachImageProcessingConfiguration(value);\r\n\r\n // Ensure the effect will be rebuilt.\r\n this._markAllSubMeshesAsTexturesDirty();\r\n }\r\n\r\n /**\r\n * Gets an array of blocks that needs to be serialized even if they are not yet connected\r\n */\r\n public attachedBlocks = new Array();\r\n\r\n /**\r\n * Specifies the mode of the node material\r\n * @internal\r\n */\r\n @serialize(\"mode\")\r\n public _mode: NodeMaterialModes = NodeMaterialModes.Material;\r\n\r\n /**\r\n * Gets or sets the mode property\r\n */\r\n public get mode(): NodeMaterialModes {\r\n return this._mode;\r\n }\r\n\r\n public set mode(value: NodeMaterialModes) {\r\n this._mode = value;\r\n }\r\n\r\n /** Gets or sets the unique identifier used to identified the effect associated with the material */\r\n public get buildId() {\r\n return this._buildId;\r\n }\r\n\r\n public set buildId(value: number) {\r\n this._buildId = value;\r\n }\r\n\r\n /**\r\n * A free comment about the material\r\n */\r\n @serialize(\"comment\")\r\n public comment: string;\r\n\r\n /**\r\n * Create a new node based material\r\n * @param name defines the material name\r\n * @param scene defines the hosting scene\r\n * @param options defines creation option\r\n */\r\n constructor(name: string, scene?: Scene, options: Partial = {}) {\r\n super(name, scene || EngineStore.LastCreatedScene!);\r\n\r\n this._options = {\r\n emitComments: false,\r\n ...options,\r\n };\r\n\r\n // Setup the default processing configuration to the scene.\r\n this._attachImageProcessingConfiguration(null);\r\n }\r\n\r\n /**\r\n * Gets the current class name of the material e.g. \"NodeMaterial\"\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"NodeMaterial\";\r\n }\r\n\r\n /**\r\n * Keep track of the image processing observer to allow dispose and replace.\r\n */\r\n private _imageProcessingObserver: Nullable>;\r\n\r\n /**\r\n * Attaches a new image processing configuration to the Standard Material.\r\n * @param configuration\r\n */\r\n protected _attachImageProcessingConfiguration(configuration: Nullable): void {\r\n if (configuration === this._imageProcessingConfiguration) {\r\n return;\r\n }\r\n\r\n // Detaches observer.\r\n if (this._imageProcessingConfiguration && this._imageProcessingObserver) {\r\n this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver);\r\n }\r\n\r\n // Pick the scene configuration if needed.\r\n if (!configuration) {\r\n this._imageProcessingConfiguration = this.getScene().imageProcessingConfiguration;\r\n } else {\r\n this._imageProcessingConfiguration = configuration;\r\n }\r\n\r\n // Attaches observer.\r\n if (this._imageProcessingConfiguration) {\r\n this._imageProcessingObserver = this._imageProcessingConfiguration.onUpdateParameters.add(() => {\r\n this._markAllSubMeshesAsImageProcessingDirty();\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Get a block by its name\r\n * @param name defines the name of the block to retrieve\r\n * @returns the required block or null if not found\r\n */\r\n public getBlockByName(name: string) {\r\n let result = null;\r\n for (const block of this.attachedBlocks) {\r\n if (block.name === name) {\r\n if (!result) {\r\n result = block;\r\n } else {\r\n Tools.Warn(\"More than one block was found with the name `\" + name + \"`\");\r\n return result;\r\n }\r\n }\r\n }\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Get a block by its name\r\n * @param predicate defines the predicate used to find the good candidate\r\n * @returns the required block or null if not found\r\n */\r\n public getBlockByPredicate(predicate: (block: NodeMaterialBlock) => boolean) {\r\n for (const block of this.attachedBlocks) {\r\n if (predicate(block)) {\r\n return block;\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Get an input block by its name\r\n * @param predicate defines the predicate used to find the good candidate\r\n * @returns the required input block or null if not found\r\n */\r\n public getInputBlockByPredicate(predicate: (block: InputBlock) => boolean): Nullable {\r\n for (const block of this.attachedBlocks) {\r\n if (block.isInput && predicate(block as InputBlock)) {\r\n return block as InputBlock;\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Gets the list of input blocks attached to this material\r\n * @returns an array of InputBlocks\r\n */\r\n public getInputBlocks() {\r\n const blocks: InputBlock[] = [];\r\n for (const block of this.attachedBlocks) {\r\n if (block.isInput) {\r\n blocks.push(block as InputBlock);\r\n }\r\n }\r\n\r\n return blocks;\r\n }\r\n\r\n /**\r\n * Adds a new optimizer to the list of optimizers\r\n * @param optimizer defines the optimizers to add\r\n * @returns the current material\r\n */\r\n public registerOptimizer(optimizer: NodeMaterialOptimizer) {\r\n const index = this._optimizers.indexOf(optimizer);\r\n\r\n if (index > -1) {\r\n return;\r\n }\r\n\r\n this._optimizers.push(optimizer);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Remove an optimizer from the list of optimizers\r\n * @param optimizer defines the optimizers to remove\r\n * @returns the current material\r\n */\r\n public unregisterOptimizer(optimizer: NodeMaterialOptimizer) {\r\n const index = this._optimizers.indexOf(optimizer);\r\n\r\n if (index === -1) {\r\n return;\r\n }\r\n\r\n this._optimizers.splice(index, 1);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Add a new block to the list of output nodes\r\n * @param node defines the node to add\r\n * @returns the current material\r\n */\r\n public addOutputNode(node: NodeMaterialBlock) {\r\n if (node.target === null) {\r\n throw \"This node is not meant to be an output node. You may want to explicitly set its target value.\";\r\n }\r\n\r\n if ((node.target & NodeMaterialBlockTargets.Vertex) !== 0) {\r\n this._addVertexOutputNode(node);\r\n }\r\n\r\n if ((node.target & NodeMaterialBlockTargets.Fragment) !== 0) {\r\n this._addFragmentOutputNode(node);\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Remove a block from the list of root nodes\r\n * @param node defines the node to remove\r\n * @returns the current material\r\n */\r\n public removeOutputNode(node: NodeMaterialBlock) {\r\n if (node.target === null) {\r\n return this;\r\n }\r\n\r\n if ((node.target & NodeMaterialBlockTargets.Vertex) !== 0) {\r\n this._removeVertexOutputNode(node);\r\n }\r\n\r\n if ((node.target & NodeMaterialBlockTargets.Fragment) !== 0) {\r\n this._removeFragmentOutputNode(node);\r\n }\r\n\r\n return this;\r\n }\r\n\r\n private _addVertexOutputNode(node: NodeMaterialBlock) {\r\n if (this._vertexOutputNodes.indexOf(node) !== -1) {\r\n return;\r\n }\r\n\r\n node.target = NodeMaterialBlockTargets.Vertex;\r\n this._vertexOutputNodes.push(node);\r\n\r\n return this;\r\n }\r\n\r\n private _removeVertexOutputNode(node: NodeMaterialBlock) {\r\n const index = this._vertexOutputNodes.indexOf(node);\r\n if (index === -1) {\r\n return;\r\n }\r\n\r\n this._vertexOutputNodes.splice(index, 1);\r\n\r\n return this;\r\n }\r\n\r\n private _addFragmentOutputNode(node: NodeMaterialBlock) {\r\n if (this._fragmentOutputNodes.indexOf(node) !== -1) {\r\n return;\r\n }\r\n\r\n node.target = NodeMaterialBlockTargets.Fragment;\r\n this._fragmentOutputNodes.push(node);\r\n\r\n return this;\r\n }\r\n\r\n private _removeFragmentOutputNode(node: NodeMaterialBlock) {\r\n const index = this._fragmentOutputNodes.indexOf(node);\r\n if (index === -1) {\r\n return;\r\n }\r\n\r\n this._fragmentOutputNodes.splice(index, 1);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating that alpha blending must be enabled no matter what alpha value or alpha channel of the FragmentBlock are\r\n */\r\n @serialize()\r\n public forceAlphaBlending = false;\r\n\r\n /**\r\n * Specifies if the material will require alpha blending\r\n * @returns a boolean specifying if alpha blending is needed\r\n */\r\n public needAlphaBlending(): boolean {\r\n if (this.ignoreAlpha) {\r\n return false;\r\n }\r\n return this.forceAlphaBlending || this.alpha < 1.0 || (this._sharedData && this._sharedData.hints.needAlphaBlending);\r\n }\r\n\r\n /**\r\n * Specifies if this material should be rendered in alpha test mode\r\n * @returns a boolean specifying if an alpha test is needed.\r\n */\r\n public needAlphaTesting(): boolean {\r\n return this._sharedData && this._sharedData.hints.needAlphaTesting;\r\n }\r\n\r\n private _processInitializeOnLink(block: NodeMaterialBlock, state: NodeMaterialBuildState, nodesToProcessForOtherBuildState: NodeMaterialBlock[], autoConfigure = true) {\r\n if (block.target === NodeMaterialBlockTargets.VertexAndFragment) {\r\n nodesToProcessForOtherBuildState.push(block);\r\n } else if (state.target === NodeMaterialBlockTargets.Fragment && block.target === NodeMaterialBlockTargets.Vertex && block._preparationId !== this._buildId) {\r\n nodesToProcessForOtherBuildState.push(block);\r\n }\r\n this._initializeBlock(block, state, nodesToProcessForOtherBuildState, autoConfigure);\r\n }\r\n\r\n private _initializeBlock(node: NodeMaterialBlock, state: NodeMaterialBuildState, nodesToProcessForOtherBuildState: NodeMaterialBlock[], autoConfigure = true) {\r\n node.initialize(state);\r\n if (autoConfigure) {\r\n node.autoConfigure(this);\r\n }\r\n node._preparationId = this._buildId;\r\n\r\n if (this.attachedBlocks.indexOf(node) === -1) {\r\n if (node.isUnique) {\r\n const className = node.getClassName();\r\n\r\n for (const other of this.attachedBlocks) {\r\n if (other.getClassName() === className) {\r\n throw `Cannot have multiple blocks of type ${className} in the same NodeMaterial`;\r\n }\r\n }\r\n }\r\n this.attachedBlocks.push(node);\r\n }\r\n\r\n for (const input of node.inputs) {\r\n input.associatedVariableName = \"\";\r\n\r\n const connectedPoint = input.connectedPoint;\r\n if (connectedPoint) {\r\n const block = connectedPoint.ownerBlock;\r\n if (block !== node) {\r\n this._processInitializeOnLink(block, state, nodesToProcessForOtherBuildState, autoConfigure);\r\n }\r\n }\r\n }\r\n\r\n // Teleportation\r\n if (node.isTeleportOut) {\r\n const teleport = node as NodeMaterialTeleportOutBlock;\r\n if (teleport.entryPoint) {\r\n this._processInitializeOnLink(teleport.entryPoint, state, nodesToProcessForOtherBuildState, autoConfigure);\r\n }\r\n }\r\n\r\n for (const output of node.outputs) {\r\n output.associatedVariableName = \"\";\r\n }\r\n }\r\n\r\n private _resetDualBlocks(node: NodeMaterialBlock, id: number) {\r\n if (node.target === NodeMaterialBlockTargets.VertexAndFragment) {\r\n node.buildId = id;\r\n }\r\n\r\n for (const inputs of node.inputs) {\r\n const connectedPoint = inputs.connectedPoint;\r\n if (connectedPoint) {\r\n const block = connectedPoint.ownerBlock;\r\n if (block !== node) {\r\n this._resetDualBlocks(block, id);\r\n }\r\n }\r\n }\r\n\r\n // If this is a teleport out, we need to reset the connected block\r\n if (node.isTeleportOut) {\r\n const teleportOut = node as NodeMaterialTeleportOutBlock;\r\n if (teleportOut.entryPoint) {\r\n this._resetDualBlocks(teleportOut.entryPoint, id);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Remove a block from the current node material\r\n * @param block defines the block to remove\r\n */\r\n public removeBlock(block: NodeMaterialBlock) {\r\n const attachedBlockIndex = this.attachedBlocks.indexOf(block);\r\n if (attachedBlockIndex > -1) {\r\n this.attachedBlocks.splice(attachedBlockIndex, 1);\r\n }\r\n\r\n if (block.isFinalMerger) {\r\n this.removeOutputNode(block);\r\n }\r\n }\r\n\r\n /**\r\n * Build the material and generates the inner effect\r\n * @param verbose defines if the build should log activity\r\n * @param updateBuildId defines if the internal build Id should be updated (default is true)\r\n * @param autoConfigure defines if the autoConfigure method should be called when initializing blocks (default is true)\r\n */\r\n public build(verbose: boolean = false, updateBuildId = true, autoConfigure = true) {\r\n this._buildWasSuccessful = false;\r\n const engine = this.getScene().getEngine();\r\n\r\n const allowEmptyVertexProgram = this._mode === NodeMaterialModes.Particle;\r\n\r\n if (this._vertexOutputNodes.length === 0 && !allowEmptyVertexProgram) {\r\n throw \"You must define at least one vertexOutputNode\";\r\n }\r\n\r\n if (this._fragmentOutputNodes.length === 0) {\r\n throw \"You must define at least one fragmentOutputNode\";\r\n }\r\n\r\n // Compilation state\r\n this._vertexCompilationState = new NodeMaterialBuildState();\r\n this._vertexCompilationState.supportUniformBuffers = engine.supportsUniformBuffers;\r\n this._vertexCompilationState.target = NodeMaterialBlockTargets.Vertex;\r\n this._fragmentCompilationState = new NodeMaterialBuildState();\r\n this._fragmentCompilationState.supportUniformBuffers = engine.supportsUniformBuffers;\r\n this._fragmentCompilationState.target = NodeMaterialBlockTargets.Fragment;\r\n\r\n // Shared data\r\n this._sharedData = new NodeMaterialBuildStateSharedData();\r\n this._sharedData.fragmentOutputNodes = this._fragmentOutputNodes;\r\n this._vertexCompilationState.sharedData = this._sharedData;\r\n this._fragmentCompilationState.sharedData = this._sharedData;\r\n this._sharedData.buildId = this._buildId;\r\n this._sharedData.emitComments = this._options.emitComments;\r\n this._sharedData.verbose = verbose;\r\n this._sharedData.scene = this.getScene();\r\n this._sharedData.allowEmptyVertexProgram = allowEmptyVertexProgram;\r\n\r\n // Initialize blocks\r\n const vertexNodes: NodeMaterialBlock[] = [];\r\n const fragmentNodes: NodeMaterialBlock[] = [];\r\n\r\n for (const vertexOutputNode of this._vertexOutputNodes) {\r\n vertexNodes.push(vertexOutputNode);\r\n this._initializeBlock(vertexOutputNode, this._vertexCompilationState, fragmentNodes, autoConfigure);\r\n }\r\n\r\n for (const fragmentOutputNode of this._fragmentOutputNodes) {\r\n fragmentNodes.push(fragmentOutputNode);\r\n this._initializeBlock(fragmentOutputNode, this._fragmentCompilationState, vertexNodes, autoConfigure);\r\n }\r\n\r\n // Optimize\r\n this.optimize();\r\n\r\n // Vertex\r\n for (const vertexOutputNode of vertexNodes) {\r\n vertexOutputNode.build(this._vertexCompilationState, vertexNodes);\r\n }\r\n\r\n // Fragment\r\n this._fragmentCompilationState.uniforms = this._vertexCompilationState.uniforms.slice(0);\r\n this._fragmentCompilationState._uniformDeclaration = this._vertexCompilationState._uniformDeclaration;\r\n this._fragmentCompilationState._constantDeclaration = this._vertexCompilationState._constantDeclaration;\r\n this._fragmentCompilationState._vertexState = this._vertexCompilationState;\r\n\r\n for (const fragmentOutputNode of fragmentNodes) {\r\n this._resetDualBlocks(fragmentOutputNode, this._buildId - 1);\r\n }\r\n\r\n for (const fragmentOutputNode of fragmentNodes) {\r\n fragmentOutputNode.build(this._fragmentCompilationState, fragmentNodes);\r\n }\r\n\r\n // Finalize\r\n this._vertexCompilationState.finalize(this._vertexCompilationState);\r\n this._fragmentCompilationState.finalize(this._fragmentCompilationState);\r\n\r\n if (updateBuildId) {\r\n this._buildId = NodeMaterial._BuildIdGenerator++;\r\n }\r\n\r\n // Errors\r\n this._sharedData.emitErrors();\r\n\r\n if (verbose) {\r\n console.log(\"Vertex shader:\");\r\n console.log(this._vertexCompilationState.compilationString);\r\n console.log(\"Fragment shader:\");\r\n console.log(this._fragmentCompilationState.compilationString);\r\n }\r\n\r\n this._buildWasSuccessful = true;\r\n this.onBuildObservable.notifyObservers(this);\r\n\r\n // Wipe defines\r\n const meshes = this.getScene().meshes;\r\n for (const mesh of meshes) {\r\n if (!mesh.subMeshes) {\r\n continue;\r\n }\r\n for (const subMesh of mesh.subMeshes) {\r\n if (subMesh.getMaterial() !== this) {\r\n continue;\r\n }\r\n\r\n if (!subMesh.materialDefines) {\r\n continue;\r\n }\r\n\r\n const defines = subMesh.materialDefines;\r\n defines.markAllAsDirty();\r\n defines.reset();\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Runs an otpimization phase to try to improve the shader code\r\n */\r\n public optimize() {\r\n for (const optimizer of this._optimizers) {\r\n optimizer.optimize(this._vertexOutputNodes, this._fragmentOutputNodes);\r\n }\r\n }\r\n\r\n private _prepareDefinesForAttributes(mesh: AbstractMesh, defines: NodeMaterialDefines) {\r\n const oldNormal = defines[\"NORMAL\"];\r\n const oldTangent = defines[\"TANGENT\"];\r\n const oldColor = defines[\"VERTEXCOLOR_NME\"];\r\n\r\n defines[\"NORMAL\"] = mesh.isVerticesDataPresent(VertexBuffer.NormalKind);\r\n defines[\"TANGENT\"] = mesh.isVerticesDataPresent(VertexBuffer.TangentKind);\r\n\r\n const hasVertexColors = mesh.useVertexColors && mesh.isVerticesDataPresent(VertexBuffer.ColorKind);\r\n defines[\"VERTEXCOLOR_NME\"] = hasVertexColors;\r\n\r\n let uvChanged = false;\r\n for (let i = 1; i <= Constants.MAX_SUPPORTED_UV_SETS; ++i) {\r\n const oldUV = defines[\"UV\" + i];\r\n defines[\"UV\" + i] = mesh.isVerticesDataPresent(`uv${i === 1 ? \"\" : i}`);\r\n uvChanged = uvChanged || defines[\"UV\" + i] !== oldUV;\r\n }\r\n\r\n if (oldNormal !== defines[\"NORMAL\"] || oldTangent !== defines[\"TANGENT\"] || oldColor !== defines[\"VERTEXCOLOR_NME\"] || uvChanged) {\r\n defines.markAsAttributesDirty();\r\n }\r\n }\r\n\r\n /**\r\n * Create a post process from the material\r\n * @param camera The camera to apply the render pass to.\r\n * @param options The required width/height ratio to downsize to before computing the render pass. (Use 1.0 for full size)\r\n * @param samplingMode The sampling mode to be used when computing the pass. (default: 0)\r\n * @param engine The engine which the post process will be applied. (default: current engine)\r\n * @param reusable If the post process can be reused on the same frame. (default: false)\r\n * @param textureType Type of textures used when performing the post process. (default: 0)\r\n * @param textureFormat Format of textures used when performing the post process. (default: TEXTUREFORMAT_RGBA)\r\n * @returns the post process created\r\n */\r\n public createPostProcess(\r\n camera: Nullable,\r\n options: number | PostProcessOptions = 1,\r\n samplingMode: number = Constants.TEXTURE_NEAREST_SAMPLINGMODE,\r\n engine?: Engine,\r\n reusable?: boolean,\r\n textureType: number = Constants.TEXTURETYPE_UNSIGNED_INT,\r\n textureFormat = Constants.TEXTUREFORMAT_RGBA\r\n ): Nullable {\r\n if (this.mode !== NodeMaterialModes.PostProcess) {\r\n console.log(\"Incompatible material mode\");\r\n return null;\r\n }\r\n return this._createEffectForPostProcess(null, camera, options, samplingMode, engine, reusable, textureType, textureFormat);\r\n }\r\n\r\n /**\r\n * Create the post process effect from the material\r\n * @param postProcess The post process to create the effect for\r\n */\r\n public createEffectForPostProcess(postProcess: PostProcess) {\r\n this._createEffectForPostProcess(postProcess);\r\n }\r\n\r\n private _createEffectForPostProcess(\r\n postProcess: Nullable,\r\n camera?: Nullable,\r\n options: number | PostProcessOptions = 1,\r\n samplingMode: number = Constants.TEXTURE_NEAREST_SAMPLINGMODE,\r\n engine?: Engine,\r\n reusable?: boolean,\r\n textureType: number = Constants.TEXTURETYPE_UNSIGNED_INT,\r\n textureFormat = Constants.TEXTUREFORMAT_RGBA\r\n ): PostProcess {\r\n let tempName = this.name + this._buildId;\r\n\r\n const defines = new NodeMaterialDefines();\r\n\r\n const dummyMesh = new AbstractMesh(tempName + \"PostProcess\", this.getScene());\r\n\r\n let buildId = this._buildId;\r\n\r\n this._processDefines(dummyMesh, defines);\r\n\r\n Effect.RegisterShader(tempName, this._fragmentCompilationState._builtCompilationString, this._vertexCompilationState._builtCompilationString);\r\n\r\n if (!postProcess) {\r\n postProcess = new PostProcess(\r\n this.name + \"PostProcess\",\r\n tempName,\r\n this._fragmentCompilationState.uniforms,\r\n this._fragmentCompilationState.samplers,\r\n options,\r\n camera!,\r\n samplingMode,\r\n engine,\r\n reusable,\r\n defines.toString(),\r\n textureType,\r\n tempName,\r\n { maxSimultaneousLights: this.maxSimultaneousLights },\r\n false,\r\n textureFormat\r\n );\r\n } else {\r\n postProcess.updateEffect(\r\n defines.toString(),\r\n this._fragmentCompilationState.uniforms,\r\n this._fragmentCompilationState.samplers,\r\n { maxSimultaneousLights: this.maxSimultaneousLights },\r\n undefined,\r\n undefined,\r\n tempName,\r\n tempName\r\n );\r\n }\r\n\r\n postProcess.nodeMaterialSource = this;\r\n\r\n postProcess.onApplyObservable.add((effect) => {\r\n if (buildId !== this._buildId) {\r\n delete Effect.ShadersStore[tempName + \"VertexShader\"];\r\n delete Effect.ShadersStore[tempName + \"PixelShader\"];\r\n\r\n tempName = this.name + this._buildId;\r\n\r\n defines.markAllAsDirty();\r\n\r\n buildId = this._buildId;\r\n }\r\n\r\n const result = this._processDefines(dummyMesh, defines);\r\n\r\n if (result) {\r\n Effect.RegisterShader(tempName, this._fragmentCompilationState._builtCompilationString, this._vertexCompilationState._builtCompilationString);\r\n\r\n TimingTools.SetImmediate(() =>\r\n postProcess!.updateEffect(\r\n defines.toString(),\r\n this._fragmentCompilationState.uniforms,\r\n this._fragmentCompilationState.samplers,\r\n { maxSimultaneousLights: this.maxSimultaneousLights },\r\n undefined,\r\n undefined,\r\n tempName,\r\n tempName\r\n )\r\n );\r\n }\r\n\r\n this._checkInternals(effect);\r\n });\r\n\r\n return postProcess;\r\n }\r\n\r\n /**\r\n * Create a new procedural texture based on this node material\r\n * @param size defines the size of the texture\r\n * @param scene defines the hosting scene\r\n * @returns the new procedural texture attached to this node material\r\n */\r\n public createProceduralTexture(size: number | { width: number; height: number; layers?: number }, scene: Scene): Nullable {\r\n if (this.mode !== NodeMaterialModes.ProceduralTexture) {\r\n console.log(\"Incompatible material mode\");\r\n return null;\r\n }\r\n\r\n let tempName = this.name + this._buildId;\r\n\r\n const proceduralTexture = new ProceduralTexture(tempName, size, null, scene);\r\n\r\n const dummyMesh = new AbstractMesh(tempName + \"Procedural\", this.getScene());\r\n dummyMesh.reservedDataStore = {\r\n hidden: true,\r\n };\r\n\r\n const defines = new NodeMaterialDefines();\r\n const result = this._processDefines(dummyMesh, defines);\r\n Effect.RegisterShader(tempName, this._fragmentCompilationState._builtCompilationString, this._vertexCompilationState._builtCompilationString);\r\n\r\n let effect = this.getScene().getEngine().createEffect(\r\n {\r\n vertexElement: tempName,\r\n fragmentElement: tempName,\r\n },\r\n [VertexBuffer.PositionKind],\r\n this._fragmentCompilationState.uniforms,\r\n this._fragmentCompilationState.samplers,\r\n defines.toString(),\r\n result?.fallbacks,\r\n undefined\r\n );\r\n\r\n proceduralTexture.nodeMaterialSource = this;\r\n proceduralTexture._setEffect(effect);\r\n\r\n let buildId = this._buildId;\r\n proceduralTexture.onBeforeGenerationObservable.add(() => {\r\n if (buildId !== this._buildId) {\r\n delete Effect.ShadersStore[tempName + \"VertexShader\"];\r\n delete Effect.ShadersStore[tempName + \"PixelShader\"];\r\n\r\n tempName = this.name + this._buildId;\r\n\r\n defines.markAllAsDirty();\r\n\r\n buildId = this._buildId;\r\n }\r\n\r\n const result = this._processDefines(dummyMesh, defines);\r\n\r\n if (result) {\r\n Effect.RegisterShader(tempName, this._fragmentCompilationState._builtCompilationString, this._vertexCompilationState._builtCompilationString);\r\n\r\n TimingTools.SetImmediate(() => {\r\n effect = this.getScene().getEngine().createEffect(\r\n {\r\n vertexElement: tempName,\r\n fragmentElement: tempName,\r\n },\r\n [VertexBuffer.PositionKind],\r\n this._fragmentCompilationState.uniforms,\r\n this._fragmentCompilationState.samplers,\r\n defines.toString(),\r\n result?.fallbacks,\r\n undefined\r\n );\r\n\r\n proceduralTexture._setEffect(effect);\r\n });\r\n }\r\n\r\n this._checkInternals(effect);\r\n });\r\n\r\n return proceduralTexture;\r\n }\r\n\r\n private _createEffectForParticles(\r\n particleSystem: IParticleSystem,\r\n blendMode: number,\r\n onCompiled?: (effect: Effect) => void,\r\n onError?: (effect: Effect, errors: string) => void,\r\n effect?: Effect,\r\n defines?: NodeMaterialDefines,\r\n dummyMesh?: Nullable,\r\n particleSystemDefinesJoined = \"\"\r\n ) {\r\n let tempName = this.name + this._buildId + \"_\" + blendMode;\r\n\r\n if (!defines) {\r\n defines = new NodeMaterialDefines();\r\n }\r\n\r\n if (!dummyMesh) {\r\n dummyMesh = this.getScene().getMeshByName(this.name + \"Particle\");\r\n if (!dummyMesh) {\r\n dummyMesh = new AbstractMesh(this.name + \"Particle\", this.getScene());\r\n dummyMesh.reservedDataStore = {\r\n hidden: true,\r\n };\r\n }\r\n }\r\n\r\n let buildId = this._buildId;\r\n\r\n const particleSystemDefines: Array = [];\r\n let join = particleSystemDefinesJoined;\r\n\r\n if (!effect) {\r\n const result = this._processDefines(dummyMesh, defines);\r\n\r\n Effect.RegisterShader(tempName, this._fragmentCompilationState._builtCompilationString);\r\n\r\n particleSystem.fillDefines(particleSystemDefines, blendMode);\r\n\r\n join = particleSystemDefines.join(\"\\n\");\r\n\r\n effect = this.getScene()\r\n .getEngine()\r\n .createEffectForParticles(\r\n tempName,\r\n this._fragmentCompilationState.uniforms,\r\n this._fragmentCompilationState.samplers,\r\n defines.toString() + \"\\n\" + join,\r\n result?.fallbacks,\r\n onCompiled,\r\n onError,\r\n particleSystem\r\n );\r\n\r\n particleSystem.setCustomEffect(effect, blendMode);\r\n }\r\n\r\n effect.onBindObservable.add((effect) => {\r\n if (buildId !== this._buildId) {\r\n delete Effect.ShadersStore[tempName + \"PixelShader\"];\r\n\r\n tempName = this.name + this._buildId + \"_\" + blendMode;\r\n\r\n defines!.markAllAsDirty();\r\n\r\n buildId = this._buildId;\r\n }\r\n\r\n particleSystemDefines.length = 0;\r\n\r\n particleSystem.fillDefines(particleSystemDefines, blendMode);\r\n\r\n const particleSystemDefinesJoinedCurrent = particleSystemDefines.join(\"\\n\");\r\n\r\n if (particleSystemDefinesJoinedCurrent !== join) {\r\n defines!.markAllAsDirty();\r\n join = particleSystemDefinesJoinedCurrent;\r\n }\r\n\r\n const result = this._processDefines(dummyMesh!, defines!);\r\n\r\n if (result) {\r\n Effect.RegisterShader(tempName, this._fragmentCompilationState._builtCompilationString);\r\n\r\n effect = this.getScene()\r\n .getEngine()\r\n .createEffectForParticles(\r\n tempName,\r\n this._fragmentCompilationState.uniforms,\r\n this._fragmentCompilationState.samplers,\r\n defines!.toString() + \"\\n\" + join,\r\n result?.fallbacks,\r\n onCompiled,\r\n onError,\r\n particleSystem\r\n );\r\n particleSystem.setCustomEffect(effect, blendMode);\r\n this._createEffectForParticles(particleSystem, blendMode, onCompiled, onError, effect, defines, dummyMesh, particleSystemDefinesJoined); // add the effect.onBindObservable observer\r\n return;\r\n }\r\n\r\n this._checkInternals(effect);\r\n });\r\n }\r\n\r\n private _checkInternals(effect: Effect) {\r\n // Animated blocks\r\n if (this._sharedData.animatedInputs) {\r\n const scene = this.getScene();\r\n\r\n const frameId = scene.getFrameId();\r\n\r\n if (this._animationFrame !== frameId) {\r\n for (const input of this._sharedData.animatedInputs) {\r\n input.animate(scene);\r\n }\r\n\r\n this._animationFrame = frameId;\r\n }\r\n }\r\n\r\n // Bindable blocks\r\n for (const block of this._sharedData.bindableBlocks) {\r\n block.bind(effect, this);\r\n }\r\n\r\n // Connection points\r\n for (const inputBlock of this._sharedData.inputBlocks) {\r\n inputBlock._transmit(effect, this.getScene(), this);\r\n }\r\n }\r\n\r\n /**\r\n * Create the effect to be used as the custom effect for a particle system\r\n * @param particleSystem Particle system to create the effect for\r\n * @param onCompiled defines a function to call when the effect creation is successful\r\n * @param onError defines a function to call when the effect creation has failed\r\n */\r\n public createEffectForParticles(particleSystem: IParticleSystem, onCompiled?: (effect: Effect) => void, onError?: (effect: Effect, errors: string) => void) {\r\n if (this.mode !== NodeMaterialModes.Particle) {\r\n console.log(\"Incompatible material mode\");\r\n return;\r\n }\r\n\r\n this._createEffectForParticles(particleSystem, BaseParticleSystem.BLENDMODE_ONEONE, onCompiled, onError);\r\n this._createEffectForParticles(particleSystem, BaseParticleSystem.BLENDMODE_MULTIPLY, onCompiled, onError);\r\n }\r\n\r\n /**\r\n * Use this material as the shadow depth wrapper of a target material\r\n * @param targetMaterial defines the target material\r\n */\r\n public createAsShadowDepthWrapper(targetMaterial: Material) {\r\n if (this.mode !== NodeMaterialModes.Material) {\r\n console.log(\"Incompatible material mode\");\r\n return;\r\n }\r\n\r\n targetMaterial.shadowDepthWrapper = new BABYLON.ShadowDepthWrapper(this, this.getScene());\r\n }\r\n\r\n private _processDefines(\r\n mesh: AbstractMesh,\r\n defines: NodeMaterialDefines,\r\n useInstances = false,\r\n subMesh?: SubMesh\r\n ): Nullable<{\r\n lightDisposed: boolean;\r\n uniformBuffers: string[];\r\n mergedUniforms: string[];\r\n mergedSamplers: string[];\r\n fallbacks: EffectFallbacks;\r\n }> {\r\n let result = null;\r\n\r\n // Global defines\r\n const scene = this.getScene();\r\n if (MaterialHelper.PrepareDefinesForCamera(scene, defines)) {\r\n defines.markAsMiscDirty();\r\n }\r\n\r\n // Shared defines\r\n this._sharedData.blocksWithDefines.forEach((b) => {\r\n b.initializeDefines(mesh, this, defines, useInstances);\r\n });\r\n\r\n this._sharedData.blocksWithDefines.forEach((b) => {\r\n b.prepareDefines(mesh, this, defines, useInstances, subMesh);\r\n });\r\n\r\n // Need to recompile?\r\n if (defines.isDirty) {\r\n const lightDisposed = defines._areLightsDisposed;\r\n defines.markAsProcessed();\r\n\r\n // Repeatable content generators\r\n this._vertexCompilationState.compilationString = this._vertexCompilationState._builtCompilationString;\r\n this._fragmentCompilationState.compilationString = this._fragmentCompilationState._builtCompilationString;\r\n\r\n this._sharedData.repeatableContentBlocks.forEach((b) => {\r\n b.replaceRepeatableContent(this._vertexCompilationState, this._fragmentCompilationState, mesh, defines);\r\n });\r\n\r\n // Uniforms\r\n const uniformBuffers: string[] = [];\r\n this._sharedData.dynamicUniformBlocks.forEach((b) => {\r\n b.updateUniformsAndSamples(this._vertexCompilationState, this, defines, uniformBuffers);\r\n });\r\n\r\n const mergedUniforms = this._vertexCompilationState.uniforms;\r\n\r\n this._fragmentCompilationState.uniforms.forEach((u) => {\r\n const index = mergedUniforms.indexOf(u);\r\n\r\n if (index === -1) {\r\n mergedUniforms.push(u);\r\n }\r\n });\r\n\r\n // Samplers\r\n const mergedSamplers = this._vertexCompilationState.samplers;\r\n\r\n this._fragmentCompilationState.samplers.forEach((s) => {\r\n const index = mergedSamplers.indexOf(s);\r\n\r\n if (index === -1) {\r\n mergedSamplers.push(s);\r\n }\r\n });\r\n\r\n const fallbacks = new EffectFallbacks();\r\n\r\n this._sharedData.blocksWithFallbacks.forEach((b) => {\r\n b.provideFallbacks(mesh, fallbacks);\r\n });\r\n\r\n result = {\r\n lightDisposed,\r\n uniformBuffers,\r\n mergedUniforms,\r\n mergedSamplers,\r\n fallbacks,\r\n };\r\n }\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Get if the submesh is ready to be used and all its information available.\r\n * Child classes can use it to update shaders\r\n * @param mesh defines the mesh to check\r\n * @param subMesh defines which submesh to check\r\n * @param useInstances specifies that instances should be used\r\n * @returns a boolean indicating that the submesh is ready or not\r\n */\r\n public isReadyForSubMesh(mesh: AbstractMesh, subMesh: SubMesh, useInstances: boolean = false): boolean {\r\n if (!this._buildWasSuccessful) {\r\n return false;\r\n }\r\n\r\n const scene = this.getScene();\r\n if (this._sharedData.animatedInputs) {\r\n const frameId = scene.getFrameId();\r\n\r\n if (this._animationFrame !== frameId) {\r\n for (const input of this._sharedData.animatedInputs) {\r\n input.animate(scene);\r\n }\r\n\r\n this._animationFrame = frameId;\r\n }\r\n }\r\n\r\n if (subMesh.effect && this.isFrozen) {\r\n if (subMesh.effect._wasPreviouslyReady && subMesh.effect._wasPreviouslyUsingInstances === useInstances) {\r\n return true;\r\n }\r\n }\r\n\r\n if (!subMesh.materialDefines) {\r\n subMesh.materialDefines = new NodeMaterialDefines();\r\n }\r\n\r\n const defines = subMesh.materialDefines;\r\n if (this._isReadyForSubMesh(subMesh)) {\r\n return true;\r\n }\r\n\r\n const engine = scene.getEngine();\r\n\r\n this._prepareDefinesForAttributes(mesh, defines);\r\n\r\n // Check if blocks are ready\r\n if (this._sharedData.blockingBlocks.some((b) => !b.isReady(mesh, this, defines, useInstances))) {\r\n return false;\r\n }\r\n\r\n const result = this._processDefines(mesh, defines, useInstances, subMesh);\r\n\r\n if (result) {\r\n const previousEffect = subMesh.effect;\r\n // Compilation\r\n const join = defines.toString();\r\n let effect = engine.createEffect(\r\n {\r\n vertex: \"nodeMaterial\" + this._buildId,\r\n fragment: \"nodeMaterial\" + this._buildId,\r\n vertexSource: this._vertexCompilationState.compilationString,\r\n fragmentSource: this._fragmentCompilationState.compilationString,\r\n },\r\n {\r\n attributes: this._vertexCompilationState.attributes,\r\n uniformsNames: result.mergedUniforms,\r\n uniformBuffersNames: result.uniformBuffers,\r\n samplers: result.mergedSamplers,\r\n defines: join,\r\n fallbacks: result.fallbacks,\r\n onCompiled: this.onCompiled,\r\n onError: this.onError,\r\n indexParameters: { maxSimultaneousLights: this.maxSimultaneousLights, maxSimultaneousMorphTargets: defines.NUM_MORPH_INFLUENCERS },\r\n },\r\n engine\r\n );\r\n\r\n if (effect) {\r\n if (this._onEffectCreatedObservable) {\r\n onCreatedEffectParameters.effect = effect;\r\n onCreatedEffectParameters.subMesh = subMesh;\r\n this._onEffectCreatedObservable.notifyObservers(onCreatedEffectParameters);\r\n }\r\n\r\n // Use previous effect while new one is compiling\r\n if (this.allowShaderHotSwapping && previousEffect && !effect.isReady()) {\r\n effect = previousEffect;\r\n defines.markAsUnprocessed();\r\n\r\n if (result.lightDisposed) {\r\n // re register in case it takes more than one frame.\r\n defines._areLightsDisposed = true;\r\n return false;\r\n }\r\n } else {\r\n scene.resetCachedMaterial();\r\n subMesh.setEffect(effect, defines, this._materialContext);\r\n }\r\n }\r\n }\r\n\r\n if (!subMesh.effect || !subMesh.effect.isReady()) {\r\n return false;\r\n }\r\n\r\n defines._renderId = scene.getRenderId();\r\n subMesh.effect._wasPreviouslyReady = true;\r\n subMesh.effect._wasPreviouslyUsingInstances = useInstances;\r\n\r\n this._checkScenePerformancePriority();\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Get a string representing the shaders built by the current node graph\r\n */\r\n public get compiledShaders() {\r\n return `// Vertex shader\\n${this._vertexCompilationState.compilationString}\\n\\n// Fragment shader\\n${this._fragmentCompilationState.compilationString}`;\r\n }\r\n\r\n /**\r\n * Binds the world matrix to the material\r\n * @param world defines the world transformation matrix\r\n */\r\n public bindOnlyWorldMatrix(world: Matrix): void {\r\n const scene = this.getScene();\r\n\r\n if (!this._activeEffect) {\r\n return;\r\n }\r\n\r\n const hints = this._sharedData.hints;\r\n\r\n if (hints.needWorldViewMatrix) {\r\n world.multiplyToRef(scene.getViewMatrix(), this._cachedWorldViewMatrix);\r\n }\r\n\r\n if (hints.needWorldViewProjectionMatrix) {\r\n world.multiplyToRef(scene.getTransformMatrix(), this._cachedWorldViewProjectionMatrix);\r\n }\r\n\r\n // Connection points\r\n for (const inputBlock of this._sharedData.inputBlocks) {\r\n inputBlock._transmitWorld(this._activeEffect, world, this._cachedWorldViewMatrix, this._cachedWorldViewProjectionMatrix);\r\n }\r\n }\r\n\r\n /**\r\n * Binds the submesh to this material by preparing the effect and shader to draw\r\n * @param world defines the world transformation matrix\r\n * @param mesh defines the mesh containing the submesh\r\n * @param subMesh defines the submesh to bind the material to\r\n */\r\n public bindForSubMesh(world: Matrix, mesh: Mesh, subMesh: SubMesh): void {\r\n const scene = this.getScene();\r\n const effect = subMesh.effect;\r\n if (!effect) {\r\n return;\r\n }\r\n this._activeEffect = effect;\r\n\r\n // Matrices\r\n this.bindOnlyWorldMatrix(world);\r\n\r\n const mustRebind = this._mustRebind(scene, effect, mesh.visibility);\r\n const sharedData = this._sharedData;\r\n\r\n if (mustRebind) {\r\n // Bindable blocks\r\n for (const block of sharedData.bindableBlocks) {\r\n block.bind(effect, this, mesh, subMesh);\r\n }\r\n\r\n for (const block of sharedData.forcedBindableBlocks) {\r\n block.bind(effect, this, mesh, subMesh);\r\n }\r\n\r\n // Connection points\r\n for (const inputBlock of sharedData.inputBlocks) {\r\n inputBlock._transmit(effect, scene, this);\r\n }\r\n } else if (!this.isFrozen) {\r\n for (const block of sharedData.forcedBindableBlocks) {\r\n block.bind(effect, this, mesh, subMesh);\r\n }\r\n }\r\n\r\n this._afterBind(mesh, this._activeEffect);\r\n }\r\n\r\n /**\r\n * Gets the active textures from the material\r\n * @returns an array of textures\r\n */\r\n public getActiveTextures(): BaseTexture[] {\r\n const activeTextures = super.getActiveTextures();\r\n\r\n if (this._sharedData) {\r\n activeTextures.push(...this._sharedData.textureBlocks.filter((tb) => tb.texture).map((tb) => tb.texture!));\r\n }\r\n\r\n return activeTextures;\r\n }\r\n\r\n /**\r\n * Gets the list of texture blocks\r\n * Note that this method will only return blocks that are reachable from the final block(s) and only after the material has been built!\r\n * @returns an array of texture blocks\r\n */\r\n public getTextureBlocks(): NodeMaterialTextureBlocks[] {\r\n if (!this._sharedData) {\r\n return [];\r\n }\r\n\r\n return this._sharedData.textureBlocks;\r\n }\r\n\r\n /**\r\n * Gets the list of all texture blocks\r\n * Note that this method will scan all attachedBlocks and return blocks that are texture blocks\r\n * @returns\r\n */\r\n public getAllTextureBlocks(): NodeMaterialTextureBlocks[] {\r\n const textureBlocks: NodeMaterialTextureBlocks[] = [];\r\n\r\n for (const block of this.attachedBlocks) {\r\n if (NodeMaterial._BlockIsTextureBlock(block)) {\r\n textureBlocks.push(block);\r\n }\r\n }\r\n\r\n return textureBlocks;\r\n }\r\n\r\n /**\r\n * Specifies if the material uses a texture\r\n * @param texture defines the texture to check against the material\r\n * @returns a boolean specifying if the material uses the texture\r\n */\r\n public hasTexture(texture: BaseTexture): boolean {\r\n if (super.hasTexture(texture)) {\r\n return true;\r\n }\r\n\r\n if (!this._sharedData) {\r\n return false;\r\n }\r\n\r\n for (const t of this._sharedData.textureBlocks) {\r\n if (t.texture === texture) {\r\n return true;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Disposes the material\r\n * @param forceDisposeEffect specifies if effects should be forcefully disposed\r\n * @param forceDisposeTextures specifies if textures should be forcefully disposed\r\n * @param notBoundToMesh specifies if the material that is being disposed is known to be not bound to any mesh\r\n */\r\n public dispose(forceDisposeEffect?: boolean, forceDisposeTextures?: boolean, notBoundToMesh?: boolean): void {\r\n if (forceDisposeTextures) {\r\n for (const texture of this.getTextureBlocks()\r\n .filter((tb) => tb.texture)\r\n .map((tb) => tb.texture!)) {\r\n texture.dispose();\r\n }\r\n }\r\n\r\n for (const block of this.attachedBlocks) {\r\n block.dispose();\r\n }\r\n\r\n this.attachedBlocks.length = 0;\r\n (this._sharedData as any) = null;\r\n (this._vertexCompilationState as any) = null;\r\n (this._fragmentCompilationState as any) = null;\r\n\r\n this.onBuildObservable.clear();\r\n\r\n if (this._imageProcessingObserver) {\r\n this._imageProcessingConfiguration.onUpdateParameters.remove(this._imageProcessingObserver);\r\n this._imageProcessingObserver = null;\r\n }\r\n\r\n super.dispose(forceDisposeEffect, forceDisposeTextures, notBoundToMesh);\r\n }\r\n\r\n /** Creates the node editor window. */\r\n private _createNodeEditor(additionalConfig?: any) {\r\n const nodeEditorConfig: any = {\r\n nodeMaterial: this,\r\n ...additionalConfig,\r\n };\r\n this.BJSNODEMATERIALEDITOR.NodeEditor.Show(nodeEditorConfig);\r\n }\r\n\r\n /**\r\n * Launch the node material editor\r\n * @param config Define the configuration of the editor\r\n * @returns a promise fulfilled when the node editor is visible\r\n */\r\n public edit(config?: INodeMaterialEditorOptions): Promise {\r\n return new Promise((resolve) => {\r\n this.BJSNODEMATERIALEDITOR = this.BJSNODEMATERIALEDITOR || this._getGlobalNodeMaterialEditor();\r\n if (typeof this.BJSNODEMATERIALEDITOR == \"undefined\") {\r\n const editorUrl = config && config.editorURL ? config.editorURL : NodeMaterial.EditorURL;\r\n\r\n // Load editor and add it to the DOM\r\n Tools.LoadScript(editorUrl, () => {\r\n this.BJSNODEMATERIALEDITOR = this.BJSNODEMATERIALEDITOR || this._getGlobalNodeMaterialEditor();\r\n this._createNodeEditor(config?.nodeEditorConfig);\r\n resolve();\r\n });\r\n } else {\r\n // Otherwise creates the editor\r\n this._createNodeEditor(config?.nodeEditorConfig);\r\n resolve();\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Clear the current material\r\n */\r\n public clear() {\r\n this._vertexOutputNodes.length = 0;\r\n this._fragmentOutputNodes.length = 0;\r\n this.attachedBlocks.length = 0;\r\n }\r\n\r\n /**\r\n * Clear the current material and set it to a default state\r\n */\r\n public setToDefault() {\r\n this.clear();\r\n\r\n this.editorData = null;\r\n\r\n const positionInput = new InputBlock(\"Position\");\r\n positionInput.setAsAttribute(\"position\");\r\n\r\n const worldInput = new InputBlock(\"World\");\r\n worldInput.setAsSystemValue(NodeMaterialSystemValues.World);\r\n\r\n const worldPos = new TransformBlock(\"WorldPos\");\r\n positionInput.connectTo(worldPos);\r\n worldInput.connectTo(worldPos);\r\n\r\n const viewProjectionInput = new InputBlock(\"ViewProjection\");\r\n viewProjectionInput.setAsSystemValue(NodeMaterialSystemValues.ViewProjection);\r\n\r\n const worldPosdMultipliedByViewProjection = new TransformBlock(\"WorldPos * ViewProjectionTransform\");\r\n worldPos.connectTo(worldPosdMultipliedByViewProjection);\r\n viewProjectionInput.connectTo(worldPosdMultipliedByViewProjection);\r\n\r\n const vertexOutput = new VertexOutputBlock(\"VertexOutput\");\r\n worldPosdMultipliedByViewProjection.connectTo(vertexOutput);\r\n\r\n // Pixel\r\n const pixelColor = new InputBlock(\"color\");\r\n pixelColor.value = new Color4(0.8, 0.8, 0.8, 1);\r\n\r\n const fragmentOutput = new FragmentOutputBlock(\"FragmentOutput\");\r\n pixelColor.connectTo(fragmentOutput);\r\n\r\n // Add to nodes\r\n this.addOutputNode(vertexOutput);\r\n this.addOutputNode(fragmentOutput);\r\n\r\n this._mode = NodeMaterialModes.Material;\r\n }\r\n\r\n /**\r\n * Clear the current material and set it to a default state for post process\r\n */\r\n public setToDefaultPostProcess() {\r\n this.clear();\r\n\r\n this.editorData = null;\r\n\r\n const position = new InputBlock(\"Position\");\r\n position.setAsAttribute(\"position2d\");\r\n\r\n const const1 = new InputBlock(\"Constant1\");\r\n const1.isConstant = true;\r\n const1.value = 1;\r\n\r\n const vmerger = new VectorMergerBlock(\"Position3D\");\r\n\r\n position.connectTo(vmerger);\r\n const1.connectTo(vmerger, { input: \"w\" });\r\n\r\n const vertexOutput = new VertexOutputBlock(\"VertexOutput\");\r\n vmerger.connectTo(vertexOutput);\r\n\r\n // Pixel\r\n const scale = new InputBlock(\"Scale\");\r\n scale.visibleInInspector = true;\r\n scale.value = new Vector2(1, 1);\r\n\r\n const uv0 = new RemapBlock(\"uv0\");\r\n position.connectTo(uv0);\r\n\r\n const uv = new MultiplyBlock(\"UV scale\");\r\n uv0.connectTo(uv);\r\n scale.connectTo(uv);\r\n\r\n const currentScreen = new CurrentScreenBlock(\"CurrentScreen\");\r\n uv.connectTo(currentScreen);\r\n\r\n currentScreen.texture = new Texture(\"https://assets.babylonjs.com/nme/currentScreenPostProcess.png\", this.getScene());\r\n\r\n const fragmentOutput = new FragmentOutputBlock(\"FragmentOutput\");\r\n currentScreen.connectTo(fragmentOutput, { output: \"rgba\" });\r\n\r\n // Add to nodes\r\n this.addOutputNode(vertexOutput);\r\n this.addOutputNode(fragmentOutput);\r\n\r\n this._mode = NodeMaterialModes.PostProcess;\r\n }\r\n\r\n /**\r\n * Clear the current material and set it to a default state for procedural texture\r\n */\r\n public setToDefaultProceduralTexture() {\r\n this.clear();\r\n\r\n this.editorData = null;\r\n\r\n const position = new InputBlock(\"Position\");\r\n position.setAsAttribute(\"position2d\");\r\n\r\n const const1 = new InputBlock(\"Constant1\");\r\n const1.isConstant = true;\r\n const1.value = 1;\r\n\r\n const vmerger = new VectorMergerBlock(\"Position3D\");\r\n\r\n position.connectTo(vmerger);\r\n const1.connectTo(vmerger, { input: \"w\" });\r\n\r\n const vertexOutput = new VertexOutputBlock(\"VertexOutput\");\r\n vmerger.connectTo(vertexOutput);\r\n\r\n // Pixel\r\n const time = new InputBlock(\"Time\");\r\n time.value = 0;\r\n time.min = 0;\r\n time.max = 0;\r\n time.isBoolean = false;\r\n time.matrixMode = 0;\r\n time.animationType = AnimatedInputBlockTypes.Time;\r\n time.isConstant = false;\r\n\r\n const color = new InputBlock(\"Color3\");\r\n color.value = new Color3(1, 1, 1);\r\n color.isConstant = false;\r\n const fragmentOutput = new FragmentOutputBlock(\"FragmentOutput\");\r\n\r\n const vectorMerger = new VectorMergerBlock(\"VectorMerger\");\r\n vectorMerger.visibleInInspector = false;\r\n\r\n const cos = new TrigonometryBlock(\"Cos\");\r\n cos.operation = TrigonometryBlockOperations.Cos;\r\n\r\n position.connectTo(vectorMerger);\r\n time.output.connectTo(cos.input);\r\n cos.output.connectTo(vectorMerger.z);\r\n vectorMerger.xyzOut.connectTo(fragmentOutput.rgb);\r\n\r\n // Add to nodes\r\n this.addOutputNode(vertexOutput);\r\n this.addOutputNode(fragmentOutput);\r\n\r\n this._mode = NodeMaterialModes.ProceduralTexture;\r\n }\r\n\r\n /**\r\n * Clear the current material and set it to a default state for particle\r\n */\r\n public setToDefaultParticle() {\r\n this.clear();\r\n\r\n this.editorData = null;\r\n\r\n // Pixel\r\n const uv = new InputBlock(\"uv\");\r\n uv.setAsAttribute(\"particle_uv\");\r\n\r\n const texture = new ParticleTextureBlock(\"ParticleTexture\");\r\n uv.connectTo(texture);\r\n\r\n const color = new InputBlock(\"Color\");\r\n color.setAsAttribute(\"particle_color\");\r\n\r\n const multiply = new MultiplyBlock(\"Texture * Color\");\r\n texture.connectTo(multiply);\r\n color.connectTo(multiply);\r\n\r\n const rampGradient = new ParticleRampGradientBlock(\"ParticleRampGradient\");\r\n multiply.connectTo(rampGradient);\r\n\r\n const cSplitter = new ColorSplitterBlock(\"ColorSplitter\");\r\n color.connectTo(cSplitter);\r\n\r\n const blendMultiply = new ParticleBlendMultiplyBlock(\"ParticleBlendMultiply\");\r\n rampGradient.connectTo(blendMultiply);\r\n texture.connectTo(blendMultiply, { output: \"a\" });\r\n cSplitter.connectTo(blendMultiply, { output: \"a\" });\r\n\r\n const fragmentOutput = new FragmentOutputBlock(\"FragmentOutput\");\r\n blendMultiply.connectTo(fragmentOutput);\r\n\r\n // Add to nodes\r\n this.addOutputNode(fragmentOutput);\r\n\r\n this._mode = NodeMaterialModes.Particle;\r\n }\r\n\r\n /**\r\n * Loads the current Node Material from a url pointing to a file save by the Node Material Editor\r\n * @deprecated Please use NodeMaterial.ParseFromFileAsync instead\r\n * @param url defines the url to load from\r\n * @param rootUrl defines the root URL for nested url in the node material\r\n * @returns a promise that will fulfil when the material is fully loaded\r\n */\r\n public async loadAsync(url: string, rootUrl: string = \"\") {\r\n return NodeMaterial.ParseFromFileAsync(\"\", url, this.getScene(), rootUrl, true, this);\r\n }\r\n\r\n private _gatherBlocks(rootNode: NodeMaterialBlock, list: NodeMaterialBlock[]) {\r\n if (list.indexOf(rootNode) !== -1) {\r\n return;\r\n }\r\n list.push(rootNode);\r\n\r\n for (const input of rootNode.inputs) {\r\n const connectedPoint = input.connectedPoint;\r\n if (connectedPoint) {\r\n const block = connectedPoint.ownerBlock;\r\n if (block !== rootNode) {\r\n this._gatherBlocks(block, list);\r\n }\r\n }\r\n }\r\n\r\n // Teleportation\r\n if (rootNode.isTeleportOut) {\r\n const block = rootNode as NodeMaterialTeleportOutBlock;\r\n if (block.entryPoint) {\r\n this._gatherBlocks(block.entryPoint, list);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Generate a string containing the code declaration required to create an equivalent of this material\r\n * @returns a string\r\n */\r\n public generateCode() {\r\n let alreadyDumped: NodeMaterialBlock[] = [];\r\n const vertexBlocks: NodeMaterialBlock[] = [];\r\n const uniqueNames: string[] = [\"const\", \"var\", \"let\"];\r\n // Gets active blocks\r\n for (const outputNode of this._vertexOutputNodes) {\r\n this._gatherBlocks(outputNode, vertexBlocks);\r\n }\r\n\r\n const fragmentBlocks: NodeMaterialBlock[] = [];\r\n for (const outputNode of this._fragmentOutputNodes) {\r\n this._gatherBlocks(outputNode, fragmentBlocks);\r\n }\r\n\r\n // Generate vertex shader\r\n let codeString = `var nodeMaterial = new BABYLON.NodeMaterial(\"${this.name || \"node material\"}\");\\n`;\r\n codeString += `nodeMaterial.mode = BABYLON.NodeMaterialModes.${NodeMaterialModes[this.mode]};\\n`;\r\n for (const node of vertexBlocks) {\r\n if (node.isInput && alreadyDumped.indexOf(node) === -1) {\r\n codeString += node._dumpCode(uniqueNames, alreadyDumped);\r\n }\r\n }\r\n\r\n // Generate fragment shader\r\n for (const node of fragmentBlocks) {\r\n if (node.isInput && alreadyDumped.indexOf(node) === -1) {\r\n codeString += node._dumpCode(uniqueNames, alreadyDumped);\r\n }\r\n }\r\n\r\n // Connections\r\n alreadyDumped = [];\r\n codeString += \"\\n// Connections\\n\";\r\n for (const node of this._vertexOutputNodes) {\r\n codeString += node._dumpCodeForOutputConnections(alreadyDumped);\r\n }\r\n for (const node of this._fragmentOutputNodes) {\r\n codeString += node._dumpCodeForOutputConnections(alreadyDumped);\r\n }\r\n\r\n // Output nodes\r\n codeString += \"\\n// Output nodes\\n\";\r\n for (const node of this._vertexOutputNodes) {\r\n codeString += `nodeMaterial.addOutputNode(${node._codeVariableName});\\n`;\r\n }\r\n\r\n for (const node of this._fragmentOutputNodes) {\r\n codeString += `nodeMaterial.addOutputNode(${node._codeVariableName});\\n`;\r\n }\r\n\r\n codeString += `nodeMaterial.build();\\n`;\r\n\r\n return codeString;\r\n }\r\n\r\n /**\r\n * Serializes this material in a JSON representation\r\n * @param selectedBlocks\r\n * @returns the serialized material object\r\n */\r\n public serialize(selectedBlocks?: NodeMaterialBlock[]): any {\r\n const serializationObject = selectedBlocks ? {} : SerializationHelper.Serialize(this);\r\n serializationObject.editorData = JSON.parse(JSON.stringify(this.editorData)); // Copy\r\n\r\n let blocks: NodeMaterialBlock[] = [];\r\n\r\n if (selectedBlocks) {\r\n blocks = selectedBlocks;\r\n } else {\r\n serializationObject.customType = \"BABYLON.NodeMaterial\";\r\n serializationObject.outputNodes = [];\r\n\r\n // Outputs\r\n for (const outputNode of this._vertexOutputNodes) {\r\n this._gatherBlocks(outputNode, blocks);\r\n serializationObject.outputNodes.push(outputNode.uniqueId);\r\n }\r\n\r\n for (const outputNode of this._fragmentOutputNodes) {\r\n this._gatherBlocks(outputNode, blocks);\r\n\r\n if (serializationObject.outputNodes.indexOf(outputNode.uniqueId) === -1) {\r\n serializationObject.outputNodes.push(outputNode.uniqueId);\r\n }\r\n }\r\n }\r\n\r\n // Blocks\r\n serializationObject.blocks = [];\r\n\r\n for (const block of blocks) {\r\n serializationObject.blocks.push(block.serialize());\r\n }\r\n\r\n if (!selectedBlocks) {\r\n for (const block of this.attachedBlocks) {\r\n if (blocks.indexOf(block) !== -1) {\r\n continue;\r\n }\r\n serializationObject.blocks.push(block.serialize());\r\n }\r\n }\r\n\r\n return serializationObject;\r\n }\r\n\r\n private _restoreConnections(block: NodeMaterialBlock, source: any, map: { [key: number]: NodeMaterialBlock }) {\r\n for (const outputPoint of block.outputs) {\r\n for (const candidate of source.blocks) {\r\n const target = map[candidate.id];\r\n\r\n if (!target) {\r\n continue;\r\n }\r\n\r\n for (const input of candidate.inputs) {\r\n if (map[input.targetBlockId] === block && input.targetConnectionName === outputPoint.name) {\r\n const inputPoint = target.getInputByName(input.inputName);\r\n if (!inputPoint || inputPoint.isConnected) {\r\n continue;\r\n }\r\n\r\n outputPoint.connectTo(inputPoint, true);\r\n this._restoreConnections(target, source, map);\r\n continue;\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Clear the current graph and load a new one from a serialization object\r\n * @param source defines the JSON representation of the material\r\n * @param rootUrl defines the root URL to use to load textures and relative dependencies\r\n * @param merge defines whether or not the source must be merged or replace the current content\r\n */\r\n public parseSerializedObject(source: any, rootUrl: string = \"\", merge = false) {\r\n if (!merge) {\r\n this.clear();\r\n }\r\n\r\n const map: { [key: number]: NodeMaterialBlock } = {};\r\n\r\n // Create blocks\r\n for (const parsedBlock of source.blocks) {\r\n const blockType = GetClass(parsedBlock.customType);\r\n if (blockType) {\r\n const block: NodeMaterialBlock = new blockType();\r\n block._deserialize(parsedBlock, this.getScene(), rootUrl);\r\n map[parsedBlock.id] = block;\r\n\r\n this.attachedBlocks.push(block);\r\n }\r\n }\r\n\r\n // Reconnect teleportation\r\n for (const block of this.attachedBlocks) {\r\n if (block.isTeleportOut) {\r\n const teleportOut = block as NodeMaterialTeleportOutBlock;\r\n const id = teleportOut._tempEntryPointUniqueId;\r\n if (id) {\r\n const source = map[id] as NodeMaterialTeleportInBlock;\r\n source.attachToEndpoint(teleportOut);\r\n }\r\n }\r\n }\r\n\r\n // Connections - Starts with input blocks only (except if in \"merge\" mode where we scan all blocks)\r\n for (let blockIndex = 0; blockIndex < source.blocks.length; blockIndex++) {\r\n const parsedBlock = source.blocks[blockIndex];\r\n const block = map[parsedBlock.id];\r\n\r\n if (!block) {\r\n continue;\r\n }\r\n\r\n if (block.inputs.length && !merge) {\r\n continue;\r\n }\r\n this._restoreConnections(block, source, map);\r\n }\r\n\r\n // Outputs\r\n if (source.outputNodes) {\r\n for (const outputNodeId of source.outputNodes) {\r\n this.addOutputNode(map[outputNodeId]);\r\n }\r\n }\r\n\r\n // UI related info\r\n if (source.locations || (source.editorData && source.editorData.locations)) {\r\n const locations: {\r\n blockId: number;\r\n x: number;\r\n y: number;\r\n }[] = source.locations || source.editorData.locations;\r\n\r\n for (const location of locations) {\r\n if (map[location.blockId]) {\r\n location.blockId = map[location.blockId].uniqueId;\r\n }\r\n }\r\n\r\n if (merge && this.editorData && this.editorData.locations) {\r\n locations.concat(this.editorData.locations);\r\n }\r\n\r\n if (source.locations) {\r\n this.editorData = {\r\n locations: locations,\r\n };\r\n } else {\r\n this.editorData = source.editorData;\r\n this.editorData.locations = locations;\r\n }\r\n\r\n const blockMap: number[] = [];\r\n\r\n for (const key in map) {\r\n blockMap[key] = map[key].uniqueId;\r\n }\r\n\r\n this.editorData.map = blockMap;\r\n }\r\n\r\n this.comment = source.comment;\r\n\r\n if (source.forceAlphaBlending !== undefined) {\r\n this.forceAlphaBlending = source.forceAlphaBlending;\r\n }\r\n\r\n if (!merge) {\r\n this._mode = source.mode ?? NodeMaterialModes.Material;\r\n }\r\n }\r\n\r\n /**\r\n * Clear the current graph and load a new one from a serialization object\r\n * @param source defines the JSON representation of the material\r\n * @param rootUrl defines the root URL to use to load textures and relative dependencies\r\n * @param merge defines whether or not the source must be merged or replace the current content\r\n * @deprecated Please use the parseSerializedObject method instead\r\n */\r\n public loadFromSerialization(source: any, rootUrl: string = \"\", merge = false) {\r\n this.parseSerializedObject(source, rootUrl, merge);\r\n }\r\n\r\n /**\r\n * Makes a duplicate of the current material.\r\n * @param name defines the name to use for the new material\r\n * @param shareEffect defines if the clone material should share the same effect (default is false)\r\n */\r\n public clone(name: string, shareEffect: boolean = false): NodeMaterial {\r\n const serializationObject = this.serialize();\r\n\r\n const clone = SerializationHelper.Clone(() => new NodeMaterial(name, this.getScene(), this.options), this);\r\n clone.id = name;\r\n clone.name = name;\r\n\r\n clone.parseSerializedObject(serializationObject);\r\n clone._buildId = this._buildId;\r\n clone.build(false, !shareEffect);\r\n\r\n return clone;\r\n }\r\n\r\n /**\r\n * Creates a node material from parsed material data\r\n * @param source defines the JSON representation of the material\r\n * @param scene defines the hosting scene\r\n * @param rootUrl defines the root URL to use to load textures and relative dependencies\r\n * @returns a new node material\r\n */\r\n public static Parse(source: any, scene: Scene, rootUrl: string = \"\"): NodeMaterial {\r\n const nodeMaterial = SerializationHelper.Parse(() => new NodeMaterial(source.name, scene), source, scene, rootUrl);\r\n\r\n nodeMaterial.parseSerializedObject(source, rootUrl);\r\n nodeMaterial.build();\r\n\r\n return nodeMaterial;\r\n }\r\n\r\n /**\r\n * Creates a node material from a snippet saved in a remote file\r\n * @param name defines the name of the material to create\r\n * @param url defines the url to load from\r\n * @param scene defines the hosting scene\r\n * @param rootUrl defines the root URL for nested url in the node material\r\n * @param skipBuild defines whether to build the node material\r\n * @param targetMaterial defines a material to use instead of creating a new one\r\n * @returns a promise that will resolve to the new node material\r\n */\r\n public static async ParseFromFileAsync(\r\n name: string,\r\n url: string,\r\n scene: Scene,\r\n rootUrl: string = \"\",\r\n skipBuild: boolean = false,\r\n targetMaterial?: NodeMaterial\r\n ): Promise {\r\n const material = targetMaterial ?? new NodeMaterial(name, scene);\r\n\r\n const data = await scene._loadFileAsync(url);\r\n const serializationObject = JSON.parse(data as string);\r\n material.parseSerializedObject(serializationObject, rootUrl);\r\n if (!skipBuild) {\r\n material.build();\r\n }\r\n return material;\r\n }\r\n\r\n /**\r\n * Creates a node material from a snippet saved by the node material editor\r\n * @param snippetId defines the snippet to load\r\n * @param scene defines the hosting scene\r\n * @param rootUrl defines the root URL to use to load textures and relative dependencies\r\n * @param nodeMaterial defines a node material to update (instead of creating a new one)\r\n * @param skipBuild defines whether to build the node material\r\n * @returns a promise that will resolve to the new node material\r\n */\r\n public static ParseFromSnippetAsync(\r\n snippetId: string,\r\n scene: Scene = EngineStore.LastCreatedScene!,\r\n rootUrl: string = \"\",\r\n nodeMaterial?: NodeMaterial,\r\n skipBuild: boolean = false\r\n ): Promise {\r\n if (snippetId === \"_BLANK\") {\r\n return Promise.resolve(NodeMaterial.CreateDefault(\"blank\", scene));\r\n }\r\n\r\n return new Promise((resolve, reject) => {\r\n const request = new WebRequest();\r\n request.addEventListener(\"readystatechange\", () => {\r\n if (request.readyState == 4) {\r\n if (request.status == 200) {\r\n const snippet = JSON.parse(JSON.parse(request.responseText).jsonPayload);\r\n const serializationObject = JSON.parse(snippet.nodeMaterial);\r\n\r\n if (!nodeMaterial) {\r\n nodeMaterial = SerializationHelper.Parse(() => new NodeMaterial(snippetId, scene), serializationObject, scene, rootUrl);\r\n nodeMaterial.uniqueId = scene.getUniqueId();\r\n }\r\n\r\n nodeMaterial.parseSerializedObject(serializationObject);\r\n nodeMaterial.snippetId = snippetId;\r\n\r\n try {\r\n if (!skipBuild) {\r\n nodeMaterial.build();\r\n }\r\n resolve(nodeMaterial);\r\n } catch (err) {\r\n reject(err);\r\n }\r\n } else {\r\n reject(\"Unable to load the snippet \" + snippetId);\r\n }\r\n }\r\n });\r\n\r\n request.open(\"GET\", this.SnippetUrl + \"/\" + snippetId.replace(/#/g, \"/\"));\r\n request.send();\r\n });\r\n }\r\n\r\n /**\r\n * Creates a new node material set to default basic configuration\r\n * @param name defines the name of the material\r\n * @param scene defines the hosting scene\r\n * @returns a new NodeMaterial\r\n */\r\n public static CreateDefault(name: string, scene?: Scene) {\r\n const newMaterial = new NodeMaterial(name, scene);\r\n\r\n newMaterial.setToDefault();\r\n newMaterial.build();\r\n\r\n return newMaterial;\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.NodeMaterial\", NodeMaterial);\r\n","import { WebXRFeaturesManager, WebXRFeatureName } from \"../webXRFeaturesManager\";\r\nimport type { WebXRControllerPointerSelection } from \"./WebXRControllerPointerSelection\";\r\nimport type { WebXRSessionManager } from \"../webXRSessionManager\";\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport { CreateSphere } from \"../../Meshes/Builders/sphereBuilder\";\r\nimport type { Observer } from \"../../Misc/observable\";\r\nimport type { WebXRInput } from \"../webXRInput\";\r\nimport type { WebXRInputSource } from \"../webXRInputSource\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { WebXRControllerComponent } from \"../motionController/webXRControllerComponent\";\r\nimport type { IndicesArray, Nullable } from \"../../types\";\r\nimport { Vector3, Quaternion, TmpVectors } from \"../../Maths/math.vector\";\r\nimport { Ray } from \"../../Culling/ray\";\r\nimport { PickingInfo } from \"../../Collisions/pickingInfo\";\r\nimport { WebXRAbstractFeature } from \"./WebXRAbstractFeature\";\r\nimport { UtilityLayerRenderer } from \"../../Rendering/utilityLayerRenderer\";\r\nimport type { WebXRAbstractMotionController } from \"../motionController/webXRAbstractMotionController\";\r\nimport { BoundingSphere } from \"../../Culling/boundingSphere\";\r\nimport type { TransformNode } from \"../../Meshes/transformNode\";\r\nimport { StandardMaterial } from \"../../Materials/standardMaterial\";\r\nimport { Color3 } from \"../../Maths/math.color\";\r\nimport { NodeMaterial } from \"../../Materials/Node/nodeMaterial\";\r\nimport type { Material } from \"../../Materials/material\";\r\nimport { Animation } from \"../../Animations/animation\";\r\nimport { QuadraticEase, EasingFunction } from \"../../Animations/easing\";\r\n// side effects\r\nimport \"../../Meshes/subMesh.project\";\r\n\r\ntype ControllerData = {\r\n xrController?: WebXRInputSource;\r\n squeezeComponent?: WebXRControllerComponent;\r\n selectionComponent?: WebXRControllerComponent;\r\n onButtonChangedObserver?: Nullable>;\r\n onSqueezeButtonChangedObserver?: Nullable>;\r\n onFrameObserver?: Nullable>;\r\n meshUnderPointer: Nullable;\r\n nearInteractionTargetMesh: Nullable;\r\n pick: Nullable;\r\n stalePick: Nullable;\r\n id: number;\r\n touchCollisionMesh: AbstractMesh;\r\n touchCollisionMeshFunction: (isTouch: boolean) => void;\r\n hydrateCollisionMeshFunction: (isHydration: boolean) => void;\r\n currentAnimationState: ControllerOrbAnimationState;\r\n grabRay: Ray;\r\n nearInteraction: boolean;\r\n hoverInteraction: boolean;\r\n grabInteraction: boolean;\r\n // event support\r\n eventListeners?: { [event in XREventType]?: (event: XRInputSourceEvent) => void };\r\n pickedPointVisualCue: AbstractMesh;\r\n};\r\n\r\n// Tracks the interaction animation state when using a motion controller with a near interaction orb\r\nenum ControllerOrbAnimationState {\r\n /**\r\n * Orb is invisible\r\n */\r\n DEHYDRATED,\r\n /**\r\n * Orb is visible and inside the hover range\r\n */\r\n HOVER,\r\n /**\r\n * Orb is visible and touching a near interaction target\r\n */\r\n TOUCH,\r\n}\r\n\r\n/**\r\n * Where should the near interaction mesh be attached to when using a motion controller for near interaction\r\n */\r\nexport enum WebXRNearControllerMode {\r\n /**\r\n * Motion controllers will not support near interaction\r\n */\r\n DISABLED = 0,\r\n /**\r\n * The interaction point for motion controllers will be inside of them\r\n */\r\n CENTERED_ON_CONTROLLER = 1,\r\n /**\r\n * The interaction point for motion controllers will be in front of the controller\r\n */\r\n CENTERED_IN_FRONT = 2,\r\n}\r\n\r\n/**\r\n * Options interface for the near interaction module\r\n */\r\nexport interface IWebXRNearInteractionOptions {\r\n /**\r\n * If provided, this scene will be used to render meshes.\r\n */\r\n customUtilityLayerScene?: Scene;\r\n /**\r\n * Should meshes created here be added to a utility layer or the main scene\r\n */\r\n useUtilityLayer?: boolean;\r\n /**\r\n * The xr input to use with this near interaction\r\n */\r\n xrInput: WebXRInput;\r\n /**\r\n * Enable near interaction on all controllers instead of switching between them\r\n */\r\n enableNearInteractionOnAllControllers?: boolean;\r\n /**\r\n * The preferred hand to give the near interaction to. This will be prioritized when the controller initialize.\r\n * If switch is enabled, it will still allow the user to switch between the different controllers\r\n */\r\n preferredHandedness?: XRHandedness;\r\n /**\r\n * Disable switching the near interaction from one controller to the other.\r\n * If the preferred hand is set it will be fixed on this hand, and if not it will be fixed on the first controller added to the scene\r\n */\r\n disableSwitchOnClick?: boolean;\r\n\r\n /**\r\n * Far interaction feature to toggle when near interaction takes precedence\r\n */\r\n farInteractionFeature?: WebXRControllerPointerSelection;\r\n\r\n /**\r\n * Near interaction mode for motion controllers\r\n */\r\n nearInteractionControllerMode?: WebXRNearControllerMode;\r\n\r\n /**\r\n * Optional material for the motion controller orb, if enabled\r\n */\r\n motionControllerOrbMaterial?: Material;\r\n}\r\n\r\n/**\r\n * A module that will enable near interaction near interaction for hands and motion controllers of XR Input Sources\r\n */\r\nexport class WebXRNearInteraction extends WebXRAbstractFeature {\r\n private static _IdCounter = 200;\r\n\r\n private _tmpRay: Ray = new Ray(new Vector3(), new Vector3());\r\n\r\n private _attachController = (xrController: WebXRInputSource) => {\r\n if (this._controllers[xrController.uniqueId]) {\r\n // already attached\r\n return;\r\n }\r\n // get two new meshes\r\n const { touchCollisionMesh, touchCollisionMeshFunction, hydrateCollisionMeshFunction } = this._generateNewTouchPointMesh();\r\n const selectionMesh = this._generateVisualCue();\r\n\r\n this._controllers[xrController.uniqueId] = {\r\n xrController,\r\n meshUnderPointer: null,\r\n nearInteractionTargetMesh: null,\r\n pick: null,\r\n stalePick: null,\r\n touchCollisionMesh,\r\n touchCollisionMeshFunction: touchCollisionMeshFunction,\r\n hydrateCollisionMeshFunction: hydrateCollisionMeshFunction,\r\n currentAnimationState: ControllerOrbAnimationState.DEHYDRATED,\r\n grabRay: new Ray(new Vector3(), new Vector3()),\r\n hoverInteraction: false,\r\n nearInteraction: false,\r\n grabInteraction: false,\r\n id: WebXRNearInteraction._IdCounter++,\r\n pickedPointVisualCue: selectionMesh,\r\n };\r\n\r\n if (this._attachedController) {\r\n if (\r\n !this._options.enableNearInteractionOnAllControllers &&\r\n this._options.preferredHandedness &&\r\n xrController.inputSource.handedness === this._options.preferredHandedness\r\n ) {\r\n this._attachedController = xrController.uniqueId;\r\n }\r\n } else {\r\n if (!this._options.enableNearInteractionOnAllControllers) {\r\n this._attachedController = xrController.uniqueId;\r\n }\r\n }\r\n switch (xrController.inputSource.targetRayMode) {\r\n case \"tracked-pointer\":\r\n return this._attachNearInteractionMode(xrController);\r\n case \"gaze\":\r\n return null;\r\n case \"screen\":\r\n return null;\r\n }\r\n };\r\n\r\n private _controllers: {\r\n [controllerUniqueId: string]: ControllerData;\r\n } = {};\r\n private _scene: Scene;\r\n\r\n private _attachedController: string;\r\n\r\n private _farInteractionFeature: Nullable = null;\r\n\r\n /**\r\n * The module's name\r\n */\r\n public static readonly Name = WebXRFeatureName.NEAR_INTERACTION;\r\n /**\r\n * The (Babylon) version of this module.\r\n * This is an integer representing the implementation version.\r\n * This number does not correspond to the WebXR specs version\r\n */\r\n public static readonly Version = 1;\r\n\r\n /**\r\n * default color of the selection ring\r\n */\r\n public selectionMeshDefaultColor: Color3 = new Color3(0.8, 0.8, 0.8);\r\n /**\r\n * This color will be applied to the selection ring when selection is triggered\r\n */\r\n public selectionMeshPickedColor: Color3 = new Color3(0.3, 0.3, 1.0);\r\n\r\n /**\r\n * constructs a new background remover module\r\n * @param _xrSessionManager the session manager for this module\r\n * @param _options read-only options to be used in this module\r\n */\r\n constructor(_xrSessionManager: WebXRSessionManager, private readonly _options: IWebXRNearInteractionOptions) {\r\n super(_xrSessionManager);\r\n this._scene = this._xrSessionManager.scene;\r\n if (this._options.nearInteractionControllerMode === undefined) {\r\n this._options.nearInteractionControllerMode = WebXRNearControllerMode.CENTERED_IN_FRONT;\r\n }\r\n\r\n if (this._options.farInteractionFeature) {\r\n this._farInteractionFeature = this._options.farInteractionFeature;\r\n }\r\n }\r\n\r\n /**\r\n * Attach this feature\r\n * Will usually be called by the features manager\r\n *\r\n * @returns true if successful.\r\n */\r\n public attach(): boolean {\r\n if (!super.attach()) {\r\n return false;\r\n }\r\n\r\n this._options.xrInput.controllers.forEach(this._attachController);\r\n this._addNewAttachObserver(this._options.xrInput.onControllerAddedObservable, this._attachController);\r\n this._addNewAttachObserver(this._options.xrInput.onControllerRemovedObservable, (controller) => {\r\n // REMOVE the controller\r\n this._detachController(controller.uniqueId);\r\n });\r\n\r\n this._scene.constantlyUpdateMeshUnderPointer = true;\r\n return true;\r\n }\r\n\r\n /**\r\n * Detach this feature.\r\n * Will usually be called by the features manager\r\n *\r\n * @returns true if successful.\r\n */\r\n public detach(): boolean {\r\n if (!super.detach()) {\r\n return false;\r\n }\r\n\r\n Object.keys(this._controllers).forEach((controllerId) => {\r\n this._detachController(controllerId);\r\n });\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Will get the mesh under a specific pointer.\r\n * `scene.meshUnderPointer` will only return one mesh - either left or right.\r\n * @param controllerId the controllerId to check\r\n * @returns The mesh under pointer or null if no mesh is under the pointer\r\n */\r\n public getMeshUnderPointer(controllerId: string): Nullable {\r\n if (this._controllers[controllerId]) {\r\n return this._controllers[controllerId].meshUnderPointer;\r\n } else {\r\n return null;\r\n }\r\n }\r\n\r\n /**\r\n * Get the xr controller that correlates to the pointer id in the pointer event\r\n *\r\n * @param id the pointer id to search for\r\n * @returns the controller that correlates to this id or null if not found\r\n */\r\n public getXRControllerByPointerId(id: number): Nullable {\r\n const keys = Object.keys(this._controllers);\r\n\r\n for (let i = 0; i < keys.length; ++i) {\r\n if (this._controllers[keys[i]].id === id) {\r\n return this._controllers[keys[i]].xrController || null;\r\n }\r\n }\r\n return null;\r\n }\r\n\r\n /**\r\n * This function sets webXRControllerPointerSelection feature that will be disabled when\r\n * the hover range is reached for a mesh and will be reattached when not in hover range.\r\n * This is used to remove the selection rays when moving.\r\n * @param farInteractionFeature the feature to disable when finger is in hover range for a mesh\r\n */\r\n public setFarInteractionFeature(farInteractionFeature: Nullable) {\r\n this._farInteractionFeature = farInteractionFeature;\r\n }\r\n\r\n /**\r\n * Filter used for near interaction pick and hover\r\n * @param mesh\r\n */\r\n private _nearPickPredicate(mesh: AbstractMesh): boolean {\r\n return mesh.isEnabled() && mesh.isVisible && mesh.isPickable && mesh.isNearPickable;\r\n }\r\n\r\n /**\r\n * Filter used for near interaction grab\r\n * @param mesh\r\n */\r\n private _nearGrabPredicate(mesh: AbstractMesh): boolean {\r\n return mesh.isEnabled() && mesh.isVisible && mesh.isPickable && mesh.isNearGrabbable;\r\n }\r\n\r\n /**\r\n * Filter used for any near interaction\r\n * @param mesh\r\n */\r\n private _nearInteractionPredicate(mesh: AbstractMesh): boolean {\r\n return mesh.isEnabled() && mesh.isVisible && mesh.isPickable && (mesh.isNearPickable || mesh.isNearGrabbable);\r\n }\r\n\r\n private _controllerAvailablePredicate(mesh: AbstractMesh, controllerId: string): boolean {\r\n let parent: TransformNode = mesh;\r\n\r\n while (parent) {\r\n if (parent.reservedDataStore && parent.reservedDataStore.nearInteraction && parent.reservedDataStore.nearInteraction.excludedControllerId === controllerId) {\r\n return false;\r\n }\r\n parent = parent.parent as TransformNode;\r\n }\r\n\r\n return true;\r\n }\r\n\r\n private _handleTransitionAnimation(controllerData: ControllerData, newState: ControllerOrbAnimationState) {\r\n if (\r\n controllerData.currentAnimationState === newState ||\r\n this._options.nearInteractionControllerMode !== WebXRNearControllerMode.CENTERED_IN_FRONT ||\r\n !!controllerData.xrController?.inputSource.hand\r\n ) {\r\n return;\r\n }\r\n\r\n // Don't always break to allow for animation fallthrough on rare cases of multi-transitions\r\n if (newState > controllerData.currentAnimationState) {\r\n switch (controllerData.currentAnimationState) {\r\n case ControllerOrbAnimationState.DEHYDRATED: {\r\n controllerData.hydrateCollisionMeshFunction(true);\r\n if (newState === ControllerOrbAnimationState.HOVER) {\r\n break;\r\n }\r\n }\r\n // eslint-disable-next-line no-fallthrough\r\n case ControllerOrbAnimationState.HOVER: {\r\n controllerData.touchCollisionMeshFunction(true);\r\n if (newState === ControllerOrbAnimationState.TOUCH) {\r\n break;\r\n }\r\n }\r\n }\r\n } else {\r\n switch (controllerData.currentAnimationState) {\r\n case ControllerOrbAnimationState.TOUCH: {\r\n controllerData.touchCollisionMeshFunction(false);\r\n if (newState === ControllerOrbAnimationState.HOVER) {\r\n break;\r\n }\r\n }\r\n // eslint-disable-next-line no-fallthrough\r\n case ControllerOrbAnimationState.HOVER: {\r\n controllerData.hydrateCollisionMeshFunction(false);\r\n if (newState === ControllerOrbAnimationState.DEHYDRATED) {\r\n break;\r\n }\r\n }\r\n }\r\n }\r\n\r\n controllerData.currentAnimationState = newState;\r\n }\r\n\r\n private readonly _hoverRadius = 0.1;\r\n private readonly _pickRadius = 0.02;\r\n private readonly _controllerPickRadius = 0.03; // The radius is slightly larger here to make it easier to manipulate since it's not tied to the hand position\r\n private readonly _nearGrabLengthScale = 5;\r\n\r\n private _processTouchPoint(id: string, position: Vector3, orientation: Quaternion) {\r\n const controllerData = this._controllers[id];\r\n\r\n // Position and orientation could be temporary values, se we take care of them before calling any functions that use temporary vectors/quaternions\r\n controllerData.grabRay.origin.copyFrom(position);\r\n orientation.toEulerAnglesToRef(TmpVectors.Vector3[0]);\r\n controllerData.grabRay.direction.copyFrom(TmpVectors.Vector3[0]);\r\n\r\n if (this._options.nearInteractionControllerMode === WebXRNearControllerMode.CENTERED_IN_FRONT && !controllerData.xrController?.inputSource.hand) {\r\n // offset the touch point in the direction the transform is facing\r\n controllerData.xrController!.getWorldPointerRayToRef(this._tmpRay);\r\n controllerData.grabRay.origin.addInPlace(this._tmpRay.direction.scale(0.05));\r\n }\r\n\r\n controllerData.grabRay.length = this._nearGrabLengthScale * this._hoverRadius;\r\n controllerData.touchCollisionMesh.position.copyFrom(controllerData.grabRay.origin);\r\n }\r\n\r\n protected _onXRFrame(_xrFrame: XRFrame) {\r\n Object.keys(this._controllers).forEach((id) => {\r\n // only do this for the selected pointer\r\n const controllerData = this._controllers[id];\r\n const handData = controllerData.xrController?.inputSource.hand;\r\n // If near interaction is not enabled/available for this controller, return early\r\n if (\r\n (!this._options.enableNearInteractionOnAllControllers && id !== this._attachedController) ||\r\n !controllerData.xrController ||\r\n (!handData && (!this._options.nearInteractionControllerMode || !controllerData.xrController.inputSource.gamepad))\r\n ) {\r\n controllerData.pick = null;\r\n return;\r\n }\r\n controllerData.hoverInteraction = false;\r\n controllerData.nearInteraction = false;\r\n\r\n // Every frame check collisions/input\r\n if (controllerData.xrController) {\r\n if (handData) {\r\n const xrIndexTip = handData.get(\"index-finger-tip\");\r\n if (xrIndexTip) {\r\n const indexTipPose = _xrFrame.getJointPose!(xrIndexTip, this._xrSessionManager.referenceSpace);\r\n if (indexTipPose && indexTipPose.transform) {\r\n const axisRHSMultiplier = this._scene.useRightHandedSystem ? 1 : -1;\r\n TmpVectors.Vector3[0].set(indexTipPose.transform.position.x, indexTipPose.transform.position.y, indexTipPose.transform.position.z * axisRHSMultiplier);\r\n TmpVectors.Quaternion[0].set(\r\n indexTipPose.transform.orientation.x,\r\n indexTipPose.transform.orientation.y,\r\n indexTipPose.transform.orientation.z * axisRHSMultiplier,\r\n indexTipPose.transform.orientation.w * axisRHSMultiplier\r\n );\r\n\r\n this._processTouchPoint(id, TmpVectors.Vector3[0], TmpVectors.Quaternion[0]);\r\n }\r\n }\r\n } else if (controllerData.xrController.inputSource.gamepad && this._options.nearInteractionControllerMode !== WebXRNearControllerMode.DISABLED) {\r\n let controllerPose = controllerData.xrController.pointer;\r\n if (controllerData.xrController.grip && this._options.nearInteractionControllerMode === WebXRNearControllerMode.CENTERED_ON_CONTROLLER) {\r\n controllerPose = controllerData.xrController.grip;\r\n }\r\n\r\n this._processTouchPoint(id, controllerPose.position, controllerPose.rotationQuaternion!);\r\n }\r\n } else {\r\n return;\r\n }\r\n\r\n const accuratePickInfo = (originalScenePick: Nullable, utilityScenePick: Nullable): Nullable => {\r\n let pick = null;\r\n if (!utilityScenePick || !utilityScenePick.hit) {\r\n // No hit in utility scene\r\n pick = originalScenePick;\r\n } else if (!originalScenePick || !originalScenePick.hit) {\r\n // No hit in original scene\r\n pick = utilityScenePick;\r\n } else if (utilityScenePick.distance < originalScenePick.distance) {\r\n // Hit is closer in utility scene\r\n pick = utilityScenePick;\r\n } else {\r\n // Hit is closer in original scene\r\n pick = originalScenePick;\r\n }\r\n return pick;\r\n };\r\n const populateNearInteractionInfo = (nearInteractionInfo: Nullable): PickingInfo => {\r\n let result = new PickingInfo();\r\n\r\n let nearInteractionAtOrigin = false;\r\n const nearInteraction = nearInteractionInfo && nearInteractionInfo.pickedPoint && nearInteractionInfo.hit;\r\n if (nearInteractionInfo?.pickedPoint) {\r\n nearInteractionAtOrigin = nearInteractionInfo.pickedPoint.x === 0 && nearInteractionInfo.pickedPoint.y === 0 && nearInteractionInfo.pickedPoint.z === 0;\r\n }\r\n if (nearInteraction && !nearInteractionAtOrigin) {\r\n result = nearInteractionInfo!;\r\n }\r\n return result;\r\n };\r\n\r\n // Don't perform touch logic while grabbing, to prevent triggering touch interactions while in the middle of a grab interaction\r\n // Dont update cursor logic either - the cursor should already be visible for the grab to be in range,\r\n // and in order to maintain its position on the target mesh it is parented for the duration of the grab.\r\n if (!controllerData.grabInteraction) {\r\n let pick = null;\r\n\r\n // near interaction hover\r\n let utilitySceneHoverPick = null;\r\n if (this._options.useUtilityLayer && this._utilityLayerScene) {\r\n utilitySceneHoverPick = this._pickWithSphere(controllerData, this._hoverRadius, this._utilityLayerScene, (mesh: AbstractMesh) =>\r\n this._nearInteractionPredicate(mesh)\r\n );\r\n }\r\n const originalSceneHoverPick = this._pickWithSphere(controllerData, this._hoverRadius, this._scene, (mesh: AbstractMesh) => this._nearInteractionPredicate(mesh));\r\n\r\n const hoverPickInfo = accuratePickInfo(originalSceneHoverPick, utilitySceneHoverPick);\r\n if (hoverPickInfo && hoverPickInfo.hit) {\r\n pick = populateNearInteractionInfo(hoverPickInfo);\r\n if (pick.hit) {\r\n controllerData.hoverInteraction = true;\r\n }\r\n }\r\n\r\n // near interaction pick\r\n if (controllerData.hoverInteraction) {\r\n let utilitySceneNearPick = null;\r\n const radius = handData ? this._pickRadius : this._controllerPickRadius;\r\n if (this._options.useUtilityLayer && this._utilityLayerScene) {\r\n utilitySceneNearPick = this._pickWithSphere(controllerData, radius, this._utilityLayerScene, (mesh: AbstractMesh) => this._nearPickPredicate(mesh));\r\n }\r\n const originalSceneNearPick = this._pickWithSphere(controllerData, radius, this._scene, (mesh: AbstractMesh) => this._nearPickPredicate(mesh));\r\n const pickInfo = accuratePickInfo(originalSceneNearPick, utilitySceneNearPick);\r\n const nearPick = populateNearInteractionInfo(pickInfo);\r\n if (nearPick.hit) {\r\n // Near pick takes precedence over hover interaction\r\n pick = nearPick;\r\n controllerData.nearInteraction = true;\r\n }\r\n }\r\n\r\n controllerData.stalePick = controllerData.pick;\r\n controllerData.pick = pick;\r\n\r\n // Update mesh under pointer\r\n if (controllerData.pick && controllerData.pick.pickedPoint && controllerData.pick.hit) {\r\n controllerData.meshUnderPointer = controllerData.pick.pickedMesh;\r\n controllerData.pickedPointVisualCue.position.copyFrom(controllerData.pick.pickedPoint);\r\n controllerData.pickedPointVisualCue.isVisible = true;\r\n\r\n if (this._farInteractionFeature && this._farInteractionFeature.attached) {\r\n this._farInteractionFeature._setPointerSelectionDisabledByPointerId(controllerData.id, true);\r\n }\r\n } else {\r\n controllerData.meshUnderPointer = null;\r\n controllerData.pickedPointVisualCue.isVisible = false;\r\n\r\n if (this._farInteractionFeature && this._farInteractionFeature.attached) {\r\n this._farInteractionFeature._setPointerSelectionDisabledByPointerId(controllerData.id, false);\r\n }\r\n }\r\n }\r\n\r\n // Update the interaction animation. Only updates if the visible touch mesh is active\r\n let state = ControllerOrbAnimationState.DEHYDRATED;\r\n if (controllerData.grabInteraction || controllerData.nearInteraction) {\r\n state = ControllerOrbAnimationState.TOUCH;\r\n } else if (controllerData.hoverInteraction) {\r\n state = ControllerOrbAnimationState.HOVER;\r\n }\r\n this._handleTransitionAnimation(controllerData, state);\r\n });\r\n }\r\n\r\n private get _utilityLayerScene() {\r\n return this._options.customUtilityLayerScene || UtilityLayerRenderer.DefaultUtilityLayer.utilityLayerScene;\r\n }\r\n\r\n private _generateVisualCue() {\r\n const sceneToRenderTo = this._options.useUtilityLayer ? this._options.customUtilityLayerScene || UtilityLayerRenderer.DefaultUtilityLayer.utilityLayerScene : this._scene;\r\n const selectionMesh = CreateSphere(\r\n \"nearInteraction\",\r\n {\r\n diameter: 0.0035 * 3,\r\n },\r\n sceneToRenderTo\r\n );\r\n selectionMesh.bakeCurrentTransformIntoVertices();\r\n selectionMesh.isPickable = false;\r\n selectionMesh.isVisible = false;\r\n selectionMesh.rotationQuaternion = Quaternion.Identity();\r\n const targetMat = new StandardMaterial(\"targetMat\", sceneToRenderTo);\r\n targetMat.specularColor = Color3.Black();\r\n targetMat.emissiveColor = this.selectionMeshDefaultColor;\r\n targetMat.backFaceCulling = false;\r\n selectionMesh.material = targetMat;\r\n\r\n return selectionMesh;\r\n }\r\n\r\n private _isControllerReadyForNearInteraction(id: number) {\r\n if (this._farInteractionFeature) {\r\n return this._farInteractionFeature._getPointerSelectionDisabledByPointerId(id);\r\n }\r\n\r\n return true;\r\n }\r\n\r\n private _attachNearInteractionMode(xrController: WebXRInputSource) {\r\n const controllerData = this._controllers[xrController.uniqueId];\r\n const pointerEventInit: PointerEventInit = {\r\n pointerId: controllerData.id,\r\n pointerType: \"xr-near\",\r\n };\r\n controllerData.onFrameObserver = this._xrSessionManager.onXRFrameObservable.add(() => {\r\n if (\r\n (!this._options.enableNearInteractionOnAllControllers && xrController.uniqueId !== this._attachedController) ||\r\n !controllerData.xrController ||\r\n (!controllerData.xrController.inputSource.hand && (!this._options.nearInteractionControllerMode || !controllerData.xrController.inputSource.gamepad))\r\n ) {\r\n return;\r\n }\r\n if (controllerData.pick) {\r\n controllerData.pick.ray = controllerData.grabRay;\r\n }\r\n\r\n if (controllerData.pick && this._isControllerReadyForNearInteraction(controllerData.id)) {\r\n this._scene.simulatePointerMove(controllerData.pick, pointerEventInit);\r\n }\r\n\r\n // Near pick pointer event\r\n if (controllerData.nearInteraction && controllerData.pick && controllerData.pick.hit) {\r\n if (!controllerData.nearInteractionTargetMesh) {\r\n this._scene.simulatePointerDown(controllerData.pick, pointerEventInit);\r\n controllerData.nearInteractionTargetMesh = controllerData.meshUnderPointer;\r\n }\r\n } else if (controllerData.nearInteractionTargetMesh && controllerData.stalePick) {\r\n this._scene.simulatePointerUp(controllerData.stalePick, pointerEventInit);\r\n controllerData.nearInteractionTargetMesh = null;\r\n }\r\n });\r\n\r\n const grabCheck = (pressed: boolean) => {\r\n if (\r\n this._options.enableNearInteractionOnAllControllers ||\r\n (xrController.uniqueId === this._attachedController && this._isControllerReadyForNearInteraction(controllerData.id))\r\n ) {\r\n if (controllerData.pick) {\r\n controllerData.pick.ray = controllerData.grabRay;\r\n }\r\n if (pressed && controllerData.pick && controllerData.meshUnderPointer && this._nearGrabPredicate(controllerData.meshUnderPointer)) {\r\n controllerData.grabInteraction = true;\r\n controllerData.pickedPointVisualCue.isVisible = false;\r\n this._scene.simulatePointerDown(controllerData.pick, pointerEventInit);\r\n } else if (!pressed && controllerData.pick && controllerData.grabInteraction) {\r\n this._scene.simulatePointerUp(controllerData.pick, pointerEventInit);\r\n controllerData.grabInteraction = false;\r\n controllerData.pickedPointVisualCue.isVisible = true;\r\n }\r\n } else {\r\n if (pressed && !this._options.enableNearInteractionOnAllControllers && !this._options.disableSwitchOnClick) {\r\n this._attachedController = xrController.uniqueId;\r\n }\r\n }\r\n };\r\n\r\n if (xrController.inputSource.gamepad) {\r\n const init = (motionController: WebXRAbstractMotionController) => {\r\n controllerData.squeezeComponent = motionController.getComponent(\"grasp\");\r\n if (controllerData.squeezeComponent) {\r\n controllerData.onSqueezeButtonChangedObserver = controllerData.squeezeComponent.onButtonStateChangedObservable.add((component) => {\r\n if (component.changes.pressed) {\r\n const pressed = component.changes.pressed.current;\r\n grabCheck(pressed);\r\n }\r\n });\r\n } else {\r\n controllerData.selectionComponent = motionController.getMainComponent();\r\n controllerData.onButtonChangedObserver = controllerData.selectionComponent.onButtonStateChangedObservable.add((component) => {\r\n if (component.changes.pressed) {\r\n const pressed = component.changes.pressed.current;\r\n grabCheck(pressed);\r\n }\r\n });\r\n }\r\n };\r\n if (xrController.motionController) {\r\n init(xrController.motionController);\r\n } else {\r\n xrController.onMotionControllerInitObservable.add(init);\r\n }\r\n } else {\r\n // use the select and squeeze events\r\n const selectStartListener = (event: XRInputSourceEvent) => {\r\n if (\r\n controllerData.xrController &&\r\n event.inputSource === controllerData.xrController.inputSource &&\r\n controllerData.pick &&\r\n this._isControllerReadyForNearInteraction(controllerData.id) &&\r\n controllerData.meshUnderPointer &&\r\n this._nearGrabPredicate(controllerData.meshUnderPointer)\r\n ) {\r\n controllerData.grabInteraction = true;\r\n controllerData.pickedPointVisualCue.isVisible = false;\r\n this._scene.simulatePointerDown(controllerData.pick, pointerEventInit);\r\n }\r\n };\r\n\r\n const selectEndListener = (event: XRInputSourceEvent) => {\r\n if (\r\n controllerData.xrController &&\r\n event.inputSource === controllerData.xrController.inputSource &&\r\n controllerData.pick &&\r\n this._isControllerReadyForNearInteraction(controllerData.id)\r\n ) {\r\n this._scene.simulatePointerUp(controllerData.pick, pointerEventInit);\r\n controllerData.grabInteraction = false;\r\n controllerData.pickedPointVisualCue.isVisible = true;\r\n }\r\n };\r\n\r\n controllerData.eventListeners = {\r\n selectend: selectEndListener,\r\n selectstart: selectStartListener,\r\n };\r\n\r\n this._xrSessionManager.session.addEventListener(\"selectstart\", selectStartListener);\r\n this._xrSessionManager.session.addEventListener(\"selectend\", selectEndListener);\r\n }\r\n }\r\n\r\n private _detachController(xrControllerUniqueId: string) {\r\n const controllerData = this._controllers[xrControllerUniqueId];\r\n if (!controllerData) {\r\n return;\r\n }\r\n if (controllerData.squeezeComponent) {\r\n if (controllerData.onSqueezeButtonChangedObserver) {\r\n controllerData.squeezeComponent.onButtonStateChangedObservable.remove(controllerData.onSqueezeButtonChangedObserver);\r\n }\r\n }\r\n if (controllerData.selectionComponent) {\r\n if (controllerData.onButtonChangedObserver) {\r\n controllerData.selectionComponent.onButtonStateChangedObservable.remove(controllerData.onButtonChangedObserver);\r\n }\r\n }\r\n if (controllerData.onFrameObserver) {\r\n this._xrSessionManager.onXRFrameObservable.remove(controllerData.onFrameObserver);\r\n }\r\n if (controllerData.eventListeners) {\r\n Object.keys(controllerData.eventListeners).forEach((eventName: string) => {\r\n const func = controllerData.eventListeners && controllerData.eventListeners[eventName as XREventType];\r\n if (func) {\r\n this._xrSessionManager.session.removeEventListener(eventName as XREventType, func as any);\r\n }\r\n });\r\n }\r\n controllerData.touchCollisionMesh.dispose();\r\n controllerData.pickedPointVisualCue.dispose();\r\n\r\n this._xrSessionManager.runInXRFrame(() => {\r\n // Fire a pointerup\r\n const pointerEventInit: PointerEventInit = {\r\n pointerId: controllerData.id,\r\n pointerType: \"xr-near\",\r\n };\r\n this._scene.simulatePointerUp(new PickingInfo(), pointerEventInit);\r\n });\r\n\r\n // remove from the map\r\n delete this._controllers[xrControllerUniqueId];\r\n if (this._attachedController === xrControllerUniqueId) {\r\n // check for other controllers\r\n const keys = Object.keys(this._controllers);\r\n if (keys.length) {\r\n this._attachedController = keys[0];\r\n } else {\r\n this._attachedController = \"\";\r\n }\r\n }\r\n }\r\n\r\n private _generateNewTouchPointMesh() {\r\n // populate information for near hover, pick and pinch\r\n const meshCreationScene = this._options.useUtilityLayer ? this._options.customUtilityLayerScene || UtilityLayerRenderer.DefaultUtilityLayer.utilityLayerScene : this._scene;\r\n\r\n const touchCollisionMesh = CreateSphere(\"PickSphere\", { diameter: 1 }, meshCreationScene);\r\n touchCollisionMesh.isVisible = false;\r\n\r\n // Generate the material for the touch mesh visuals\r\n if (this._options.motionControllerOrbMaterial) {\r\n touchCollisionMesh.material = this._options.motionControllerOrbMaterial;\r\n } else {\r\n NodeMaterial.ParseFromSnippetAsync(\"8RUNKL#3\", meshCreationScene).then((nodeMaterial) => {\r\n touchCollisionMesh.material = nodeMaterial;\r\n });\r\n }\r\n\r\n const easingFunction = new QuadraticEase();\r\n easingFunction.setEasingMode(EasingFunction.EASINGMODE_EASEINOUT);\r\n\r\n // Adjust the visual size based off of the size of the touch collision orb.\r\n // Having the size perfectly match for hover gives a more accurate tell for when the user will start interacting with the target\r\n // Sizes for other states are somewhat arbitrary, as they are based on what feels nice during an interaction\r\n const hoverSizeVec = new Vector3(this._controllerPickRadius, this._controllerPickRadius, this._controllerPickRadius);\r\n const touchSize = this._controllerPickRadius * (4 / 3);\r\n const touchSizeVec = new Vector3(touchSize, touchSize, touchSize);\r\n const hydrateTransitionSize = this._controllerPickRadius * (7 / 6);\r\n const hydrateTransitionSizeVec = new Vector3(hydrateTransitionSize, hydrateTransitionSize, hydrateTransitionSize);\r\n const touchHoverTransitionSize = this._controllerPickRadius * (4 / 5);\r\n const touchHoverTransitionSizeVec = new Vector3(touchHoverTransitionSize, touchHoverTransitionSize, touchHoverTransitionSize);\r\n const hoverTouchTransitionSize = this._controllerPickRadius * (3 / 2);\r\n const hoverTouchTransitionSizeVec = new Vector3(hoverTouchTransitionSize, hoverTouchTransitionSize, hoverTouchTransitionSize);\r\n\r\n const touchKeys = [\r\n { frame: 0, value: hoverSizeVec },\r\n { frame: 10, value: hoverTouchTransitionSizeVec },\r\n { frame: 18, value: touchSizeVec },\r\n ];\r\n const releaseKeys = [\r\n { frame: 0, value: touchSizeVec },\r\n { frame: 10, value: touchHoverTransitionSizeVec },\r\n { frame: 18, value: hoverSizeVec },\r\n ];\r\n const hydrateKeys = [\r\n { frame: 0, value: Vector3.ZeroReadOnly },\r\n { frame: 12, value: hydrateTransitionSizeVec },\r\n { frame: 15, value: hoverSizeVec },\r\n ];\r\n const dehydrateKeys = [\r\n { frame: 0, value: hoverSizeVec },\r\n { frame: 10, value: Vector3.ZeroReadOnly },\r\n { frame: 15, value: Vector3.ZeroReadOnly },\r\n ];\r\n\r\n const touchAction = new Animation(\"touch\", \"scaling\", 60, Animation.ANIMATIONTYPE_VECTOR3, Animation.ANIMATIONLOOPMODE_CONSTANT);\r\n const releaseAction = new Animation(\"release\", \"scaling\", 60, Animation.ANIMATIONTYPE_VECTOR3, Animation.ANIMATIONLOOPMODE_CONSTANT);\r\n const hydrateAction = new Animation(\"hydrate\", \"scaling\", 60, Animation.ANIMATIONTYPE_VECTOR3, Animation.ANIMATIONLOOPMODE_CONSTANT);\r\n const dehydrateAction = new Animation(\"dehydrate\", \"scaling\", 60, Animation.ANIMATIONTYPE_VECTOR3, Animation.ANIMATIONLOOPMODE_CONSTANT);\r\n\r\n touchAction.setEasingFunction(easingFunction);\r\n releaseAction.setEasingFunction(easingFunction);\r\n hydrateAction.setEasingFunction(easingFunction);\r\n dehydrateAction.setEasingFunction(easingFunction);\r\n\r\n touchAction.setKeys(touchKeys);\r\n releaseAction.setKeys(releaseKeys);\r\n hydrateAction.setKeys(hydrateKeys);\r\n dehydrateAction.setKeys(dehydrateKeys);\r\n\r\n const touchCollisionMeshFunction = (isTouch: boolean) => {\r\n const action = isTouch ? touchAction : releaseAction;\r\n meshCreationScene.beginDirectAnimation(touchCollisionMesh, [action], 0, 18, false, 1);\r\n };\r\n\r\n const hydrateCollisionMeshFunction = (isHydration: boolean) => {\r\n const action = isHydration ? hydrateAction : dehydrateAction;\r\n if (isHydration) {\r\n touchCollisionMesh.isVisible = true;\r\n }\r\n meshCreationScene.beginDirectAnimation(touchCollisionMesh, [action], 0, 15, false, 1, () => {\r\n if (!isHydration) {\r\n touchCollisionMesh.isVisible = false;\r\n }\r\n });\r\n };\r\n\r\n return { touchCollisionMesh, touchCollisionMeshFunction, hydrateCollisionMeshFunction };\r\n }\r\n\r\n private _pickWithSphere(controllerData: ControllerData, radius: number, sceneToUse: Scene, predicate: (mesh: AbstractMesh) => boolean): Nullable {\r\n const pickingInfo = new PickingInfo();\r\n pickingInfo.distance = +Infinity;\r\n\r\n if (controllerData.touchCollisionMesh && controllerData.xrController) {\r\n const position = controllerData.touchCollisionMesh.position;\r\n const sphere = BoundingSphere.CreateFromCenterAndRadius(position, radius);\r\n\r\n for (let meshIndex = 0; meshIndex < sceneToUse.meshes.length; meshIndex++) {\r\n const mesh = sceneToUse.meshes[meshIndex];\r\n if (!predicate(mesh) || !this._controllerAvailablePredicate(mesh, controllerData.xrController.uniqueId)) {\r\n continue;\r\n }\r\n const result = WebXRNearInteraction.PickMeshWithSphere(mesh, sphere);\r\n\r\n if (result && result.hit && result.distance < pickingInfo.distance) {\r\n pickingInfo.hit = result.hit;\r\n pickingInfo.pickedMesh = mesh;\r\n pickingInfo.pickedPoint = result.pickedPoint;\r\n pickingInfo.aimTransform = controllerData.xrController.pointer;\r\n pickingInfo.gripTransform = controllerData.xrController.grip || null;\r\n pickingInfo.originMesh = controllerData.touchCollisionMesh;\r\n pickingInfo.distance = result.distance;\r\n }\r\n }\r\n }\r\n return pickingInfo;\r\n }\r\n\r\n /**\r\n * Picks a mesh with a sphere\r\n * @param mesh the mesh to pick\r\n * @param sphere picking sphere in world coordinates\r\n * @param skipBoundingInfo a boolean indicating if we should skip the bounding info check\r\n * @returns the picking info\r\n */\r\n public static PickMeshWithSphere(mesh: AbstractMesh, sphere: BoundingSphere, skipBoundingInfo = false): PickingInfo {\r\n const subMeshes = mesh.subMeshes;\r\n const pi = new PickingInfo();\r\n const boundingInfo = mesh.getBoundingInfo();\r\n\r\n if (!mesh._generatePointsArray()) {\r\n return pi;\r\n }\r\n\r\n if (!mesh.subMeshes || !boundingInfo) {\r\n return pi;\r\n }\r\n\r\n if (!skipBoundingInfo && !BoundingSphere.Intersects(boundingInfo.boundingSphere, sphere)) {\r\n return pi;\r\n }\r\n\r\n const result = TmpVectors.Vector3[0];\r\n const tmpVec = TmpVectors.Vector3[1];\r\n\r\n let distance = +Infinity;\r\n let tmp, tmpDistanceSphereToCenter, tmpDistanceSurfaceToCenter;\r\n const center = TmpVectors.Vector3[2];\r\n const worldToMesh = TmpVectors.Matrix[0];\r\n worldToMesh.copyFrom(mesh.getWorldMatrix());\r\n worldToMesh.invert();\r\n Vector3.TransformCoordinatesToRef(sphere.center, worldToMesh, center);\r\n\r\n for (let index = 0; index < subMeshes.length; index++) {\r\n const subMesh = subMeshes[index];\r\n\r\n subMesh.projectToRef(center, mesh._positions, mesh.getIndices(), tmpVec);\r\n\r\n Vector3.TransformCoordinatesToRef(tmpVec, mesh.getWorldMatrix(), tmpVec);\r\n tmp = Vector3.Distance(tmpVec, sphere.center);\r\n\r\n // Check for finger inside of mesh\r\n tmpDistanceSurfaceToCenter = Vector3.Distance(tmpVec, mesh.getAbsolutePosition());\r\n tmpDistanceSphereToCenter = Vector3.Distance(sphere.center, mesh.getAbsolutePosition());\r\n if (tmpDistanceSphereToCenter !== -1 && tmpDistanceSurfaceToCenter !== -1 && tmpDistanceSurfaceToCenter > tmpDistanceSphereToCenter) {\r\n tmp = 0;\r\n tmpVec.copyFrom(sphere.center);\r\n }\r\n\r\n if (tmp !== -1 && tmp < distance) {\r\n distance = tmp;\r\n result.copyFrom(tmpVec);\r\n }\r\n }\r\n\r\n if (distance < sphere.radius) {\r\n pi.hit = true;\r\n pi.distance = distance;\r\n pi.pickedMesh = mesh;\r\n pi.pickedPoint = result.clone();\r\n }\r\n\r\n return pi;\r\n }\r\n}\r\n\r\n//Register the plugin\r\nWebXRFeaturesManager.AddWebXRFeature(\r\n WebXRNearInteraction.Name,\r\n (xrSessionManager, options) => {\r\n return () => new WebXRNearInteraction(xrSessionManager, options);\r\n },\r\n WebXRNearInteraction.Version,\r\n true\r\n);\r\n","import { Constants } from \"../Engines/constants\";\r\nimport { TmpVectors, Vector3 } from \"../Maths/math.vector\";\r\nimport type { IndicesArray } from \"../types\";\r\nimport { SubMesh } from \"./subMesh\";\r\n\r\ndeclare module \"./subMesh\" {\r\n export interface SubMesh {\r\n /** @internal */\r\n _projectOnTrianglesToRef(vector: Vector3, positions: Vector3[], indices: IndicesArray, step: number, checkStopper: boolean, ref: Vector3): number;\r\n /** @internal */\r\n _projectOnUnIndexedTrianglesToRef(vector: Vector3, positions: Vector3[], indices: IndicesArray, ref: Vector3): number;\r\n /**\r\n * Projects a point on this submesh and stores the result in \"ref\"\r\n *\r\n * @param vector point to project\r\n * @param positions defines mesh's positions array\r\n * @param indices defines mesh's indices array\r\n * @param ref vector that will store the result\r\n * @returns distance from the point and the submesh, or -1 if the mesh rendering mode doesn't support projections\r\n */\r\n projectToRef(vector: Vector3, positions: Vector3[], indices: IndicesArray, ref: Vector3): number;\r\n }\r\n}\r\n\r\n/**\r\n * @internal\r\n */\r\nSubMesh.prototype._projectOnTrianglesToRef = function (vector: Vector3, positions: Vector3[], indices: IndicesArray, step: number, checkStopper: boolean, ref: Vector3): number {\r\n // Triangles test\r\n const proj = TmpVectors.Vector3[0];\r\n const tmp = TmpVectors.Vector3[1];\r\n let distance = +Infinity;\r\n\r\n for (let index = this.indexStart; index < this.indexStart + this.indexCount - (3 - step); index += step) {\r\n const indexA = indices[index];\r\n const indexB = indices[index + 1];\r\n const indexC = indices[index + 2];\r\n\r\n if (checkStopper && indexC === 0xffffffff) {\r\n index += 2;\r\n continue;\r\n }\r\n\r\n const p0 = positions[indexA];\r\n const p1 = positions[indexB];\r\n const p2 = positions[indexC];\r\n\r\n // stay defensive and don't check against undefined positions.\r\n if (!p0 || !p1 || !p2) {\r\n continue;\r\n }\r\n\r\n const tmpDist = Vector3.ProjectOnTriangleToRef(vector, p0, p1, p2, tmp);\r\n if (tmpDist < distance) {\r\n proj.copyFrom(tmp);\r\n distance = tmpDist;\r\n }\r\n }\r\n\r\n ref.copyFrom(proj);\r\n\r\n return distance;\r\n};\r\n\r\n/**\r\n * @internal\r\n */\r\nSubMesh.prototype._projectOnUnIndexedTrianglesToRef = function (vector: Vector3, positions: Vector3[], indices: IndicesArray, ref: Vector3): number {\r\n // Triangles test\r\n const proj = TmpVectors.Vector3[0];\r\n const tmp = TmpVectors.Vector3[1];\r\n let distance = +Infinity;\r\n\r\n for (let index = this.verticesStart; index < this.verticesStart + this.verticesCount; index += 3) {\r\n const p0 = positions[index];\r\n const p1 = positions[index + 1];\r\n const p2 = positions[index + 2];\r\n\r\n const tmpDist = Vector3.ProjectOnTriangleToRef(vector, p0, p1, p2, tmp);\r\n if (tmpDist < distance) {\r\n proj.copyFrom(tmp);\r\n distance = tmpDist;\r\n }\r\n }\r\n\r\n ref.copyFrom(proj);\r\n\r\n return distance;\r\n};\r\n\r\nSubMesh.prototype.projectToRef = function (vector: Vector3, positions: Vector3[], indices: IndicesArray, ref: Vector3): number {\r\n const material = this.getMaterial();\r\n if (!material) {\r\n return -1;\r\n }\r\n let step = 3;\r\n let checkStopper = false;\r\n\r\n switch (material.fillMode) {\r\n case Constants.MATERIAL_PointListDrawMode:\r\n case Constants.MATERIAL_LineLoopDrawMode:\r\n case Constants.MATERIAL_LineStripDrawMode:\r\n case Constants.MATERIAL_TriangleFanDrawMode:\r\n return -1;\r\n case Constants.MATERIAL_TriangleStripDrawMode:\r\n step = 1;\r\n checkStopper = true;\r\n break;\r\n default:\r\n break;\r\n }\r\n\r\n // LineMesh first as it's also a Mesh...\r\n if (material.fillMode === Constants.MATERIAL_LineListDrawMode) {\r\n return -1;\r\n } else {\r\n // Check if mesh is unindexed\r\n if (!indices.length && (this as any)._mesh._unIndexed) {\r\n return this._projectOnUnIndexedTrianglesToRef(vector, positions, indices, ref);\r\n }\r\n\r\n return this._projectOnTrianglesToRef(vector, positions, indices, step, checkStopper, ref);\r\n }\r\n};\r\n","import type { Nullable } from \"../types\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport type { IDisposable, Scene } from \"../scene\";\r\nimport type { WebXRExperienceHelper } from \"./webXRExperienceHelper\";\r\nimport type { WebXRRenderTarget } from \"./webXRTypes\";\r\nimport { WebXRState } from \"./webXRTypes\";\r\nimport { Tools } from \"../Misc/tools\";\r\n/**\r\n * Button which can be used to enter a different mode of XR\r\n */\r\nexport class WebXREnterExitUIButton {\r\n /**\r\n * Creates a WebXREnterExitUIButton\r\n * @param element button element\r\n * @param sessionMode XR initialization session mode\r\n * @param referenceSpaceType the type of reference space to be used\r\n */\r\n constructor(\r\n /** button element */\r\n public element: HTMLElement,\r\n /** XR initialization options for the button */\r\n public sessionMode: XRSessionMode,\r\n /** Reference space type */\r\n public referenceSpaceType: XRReferenceSpaceType\r\n ) {}\r\n\r\n /**\r\n * Extendable function which can be used to update the button's visuals when the state changes\r\n * @param activeButton the current active button in the UI\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public update(activeButton: Nullable) {}\r\n}\r\n\r\n/**\r\n * Options to create the webXR UI\r\n */\r\nexport class WebXREnterExitUIOptions {\r\n /**\r\n * User provided buttons to enable/disable WebXR. The system will provide default if not set\r\n */\r\n customButtons?: Array;\r\n /**\r\n * A reference space type to use when creating the default button.\r\n * Default is local-floor\r\n */\r\n referenceSpaceType?: XRReferenceSpaceType;\r\n /**\r\n * Context to enter xr with\r\n */\r\n renderTarget?: Nullable;\r\n /**\r\n * A session mode to use when creating the default button.\r\n * Default is immersive-vr\r\n */\r\n sessionMode?: XRSessionMode;\r\n\r\n /**\r\n * A list of optional features to init the session with\r\n */\r\n optionalFeatures?: string[];\r\n\r\n /**\r\n * A list of optional features to init the session with\r\n */\r\n requiredFeatures?: string[];\r\n\r\n /**\r\n * If set, the `sessiongranted` event will not be registered. `sessiongranted` is used to move seamlessly between WebXR experiences.\r\n * If set to true the user will be forced to press the \"enter XR\" button even if sessiongranted event was triggered.\r\n * If not set and a sessiongranted event was triggered, the XR session will start automatically.\r\n */\r\n ignoreSessionGrantedEvent?: boolean;\r\n\r\n /**\r\n * If defined, this function will be executed if the UI encounters an error when entering XR\r\n */\r\n onError?: (error: any) => void;\r\n}\r\n/**\r\n * UI to allow the user to enter/exit XR mode\r\n */\r\nexport class WebXREnterExitUI implements IDisposable {\r\n private _activeButton: Nullable = null;\r\n private _buttons: Array = [];\r\n private _helper: WebXRExperienceHelper;\r\n private _renderTarget?: WebXRRenderTarget;\r\n /**\r\n * The HTML Div Element to which buttons are added.\r\n */\r\n public readonly overlay: HTMLDivElement;\r\n\r\n /**\r\n * Fired every time the active button is changed.\r\n *\r\n * When xr is entered via a button that launches xr that button will be the callback parameter\r\n *\r\n * When exiting xr the callback parameter will be null)\r\n */\r\n public activeButtonChangedObservable = new Observable>();\r\n\r\n /**\r\n * Construct a new EnterExit UI class\r\n *\r\n * @param _scene babylon scene object to use\r\n * @param options (read-only) version of the options passed to this UI\r\n */\r\n public constructor(\r\n private _scene: Scene,\r\n /** version of the options passed to this UI */\r\n public options: WebXREnterExitUIOptions\r\n ) {\r\n this.overlay = document.createElement(\"div\");\r\n this.overlay.classList.add(\"xr-button-overlay\");\r\n\r\n // prepare for session granted event\r\n if (!options.ignoreSessionGrantedEvent && (navigator as any).xr) {\r\n (navigator as any).xr.addEventListener(\"sessiongranted\", this._onSessionGranted);\r\n }\r\n\r\n // if served over HTTP, warn people.\r\n // Hopefully the browsers will catch up\r\n if (typeof window !== \"undefined\") {\r\n if (window.location && window.location.protocol === \"http:\" && window.location.hostname !== \"localhost\") {\r\n Tools.Warn(\"WebXR can only be served over HTTPS\");\r\n throw new Error(\"WebXR can only be served over HTTPS\");\r\n }\r\n }\r\n\r\n if (options.customButtons) {\r\n this._buttons = options.customButtons;\r\n } else {\r\n this.overlay.style.cssText = \"z-index:11;position: absolute; right: 20px;bottom: 50px;\";\r\n const sessionMode = options.sessionMode || \"immersive-vr\";\r\n const referenceSpaceType = options.referenceSpaceType || \"local-floor\";\r\n const url =\r\n typeof SVGSVGElement === \"undefined\"\r\n ? \"https://cdn.babylonjs.com/Assets/vrButton.png\"\r\n : \"data:image/svg+xml;charset=UTF-8,%3Csvg%20xmlns%3D%22http%3A//www.w3.org/2000/svg%22%20width%3D%222048%22%20height%3D%221152%22%20viewBox%3D%220%200%202048%201152%22%20version%3D%221.1%22%3E%3Cpath%20transform%3D%22rotate%28180%201024%2C576.0000000000001%29%22%20d%3D%22m1109%2C896q17%2C0%2030%2C-12t13%2C-30t-12.5%2C-30.5t-30.5%2C-12.5l-170%2C0q-18%2C0%20-30.5%2C12.5t-12.5%2C30.5t13%2C30t30%2C12l170%2C0zm-85%2C256q59%2C0%20132.5%2C-1.5t154.5%2C-5.5t164.5%2C-11.5t163%2C-20t150%2C-30t124.5%2C-41.5q23%2C-11%2042%2C-24t38%2C-30q27%2C-25%2041%2C-61.5t14%2C-72.5l0%2C-257q0%2C-123%20-47%2C-232t-128%2C-190t-190%2C-128t-232%2C-47l-81%2C0q-37%2C0%20-68.5%2C14t-60.5%2C34.5t-55.5%2C45t-53%2C45t-53%2C34.5t-55.5%2C14t-55.5%2C-14t-53%2C-34.5t-53%2C-45t-55.5%2C-45t-60.5%2C-34.5t-68.5%2C-14l-81%2C0q-123%2C0%20-232%2C47t-190%2C128t-128%2C190t-47%2C232l0%2C257q0%2C68%2038%2C115t97%2C73q54%2C24%20124.5%2C41.5t150%2C30t163%2C20t164.5%2C11.5t154.5%2C5.5t132.5%2C1.5zm939%2C-298q0%2C39%20-24.5%2C67t-58.5%2C42q-54%2C23%20-122%2C39.5t-143.5%2C28t-155.5%2C19t-157%2C11t-148.5%2C5t-129.5%2C1.5q-59%2C0%20-130%2C-1.5t-148%2C-5t-157%2C-11t-155.5%2C-19t-143.5%2C-28t-122%2C-39.5q-34%2C-14%20-58.5%2C-42t-24.5%2C-67l0%2C-257q0%2C-106%2040.5%2C-199t110%2C-162.5t162.5%2C-109.5t199%2C-40l81%2C0q27%2C0%2052%2C14t50%2C34.5t51%2C44.5t55.5%2C44.5t63.5%2C34.5t74%2C14t74%2C-14t63.5%2C-34.5t55.5%2C-44.5t51%2C-44.5t50%2C-34.5t52%2C-14l14%2C0q37%2C0%2070%2C0.5t64.5%2C4.5t63.5%2C12t68%2C23q71%2C30%20128.5%2C78.5t98.5%2C110t63.5%2C133.5t22.5%2C149l0%2C257z%22%20fill%3D%22white%22%20/%3E%3C/svg%3E%0A\";\r\n let css =\r\n \".babylonVRicon { color: #868686; border-color: #868686; border-style: solid; margin-left: 10px; height: 50px; width: 80px; background-color: rgba(51,51,51,0.7); background-image: url(\" +\r\n url +\r\n \"); background-size: 80%; background-repeat:no-repeat; background-position: center; border: none; outline: none; transition: transform 0.125s ease-out } .babylonVRicon:hover { transform: scale(1.05) } .babylonVRicon:active {background-color: rgba(51,51,51,1) } .babylonVRicon:focus {background-color: rgba(51,51,51,1) }\";\r\n css += '.babylonVRicon.vrdisplaypresenting { background-image: none;} .vrdisplaypresenting::after { content: \"EXIT\"} .xr-error::after { content: \"ERROR\"}';\r\n\r\n const style = document.createElement(\"style\");\r\n style.appendChild(document.createTextNode(css));\r\n document.getElementsByTagName(\"head\")[0].appendChild(style);\r\n const hmdBtn = document.createElement(\"button\");\r\n hmdBtn.className = \"babylonVRicon\";\r\n hmdBtn.title = `${sessionMode} - ${referenceSpaceType}`;\r\n this._buttons.push(new WebXREnterExitUIButton(hmdBtn, sessionMode, referenceSpaceType));\r\n this._buttons[this._buttons.length - 1].update = function (activeButton: WebXREnterExitUIButton) {\r\n this.element.style.display = activeButton === null || activeButton === this ? \"\" : \"none\";\r\n hmdBtn.className = \"babylonVRicon\" + (activeButton === this ? \" vrdisplaypresenting\" : \"\");\r\n };\r\n this._updateButtons(null);\r\n }\r\n\r\n const renderCanvas = _scene.getEngine().getInputElement();\r\n if (renderCanvas && renderCanvas.parentNode) {\r\n renderCanvas.parentNode.appendChild(this.overlay);\r\n _scene.onDisposeObservable.addOnce(() => {\r\n this.dispose();\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Set the helper to be used with this UI component.\r\n * The UI is bound to an experience helper. If not provided the UI can still be used but the events should be registered by the developer.\r\n *\r\n * @param helper the experience helper to attach\r\n * @param renderTarget an optional render target (in case it is created outside of the helper scope)\r\n * @returns a promise that resolves when the ui is ready\r\n */\r\n public async setHelperAsync(helper: WebXRExperienceHelper, renderTarget?: WebXRRenderTarget): Promise {\r\n this._helper = helper;\r\n this._renderTarget = renderTarget;\r\n const supportedPromises = this._buttons.map((btn) => {\r\n return helper.sessionManager.isSessionSupportedAsync(btn.sessionMode);\r\n });\r\n helper.onStateChangedObservable.add((state) => {\r\n if (state == WebXRState.NOT_IN_XR) {\r\n this._updateButtons(null);\r\n }\r\n });\r\n const results = await Promise.all(supportedPromises);\r\n results.forEach((supported, i) => {\r\n if (supported) {\r\n this.overlay.appendChild(this._buttons[i].element);\r\n this._buttons[i].element.onclick = this._enterXRWithButtonIndex.bind(this, i);\r\n } else {\r\n Tools.Warn(`Session mode \"${this._buttons[i].sessionMode}\" not supported in browser`);\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Creates UI to allow the user to enter/exit XR mode\r\n * @param scene the scene to add the ui to\r\n * @param helper the xr experience helper to enter/exit xr with\r\n * @param options options to configure the UI\r\n * @returns the created ui\r\n */\r\n public static async CreateAsync(scene: Scene, helper: WebXRExperienceHelper, options: WebXREnterExitUIOptions): Promise {\r\n const ui = new WebXREnterExitUI(scene, options);\r\n await ui.setHelperAsync(helper, options.renderTarget || undefined);\r\n return ui;\r\n }\r\n\r\n private async _enterXRWithButtonIndex(idx: number = 0) {\r\n if (this._helper.state == WebXRState.IN_XR) {\r\n await this._helper.exitXRAsync();\r\n this._updateButtons(null);\r\n } else if (this._helper.state == WebXRState.NOT_IN_XR) {\r\n try {\r\n await this._helper.enterXRAsync(this._buttons[idx].sessionMode, this._buttons[idx].referenceSpaceType, this._renderTarget, {\r\n optionalFeatures: this.options.optionalFeatures,\r\n requiredFeatures: this.options.requiredFeatures,\r\n });\r\n this._updateButtons(this._buttons[idx]);\r\n } catch (e) {\r\n // make sure button is visible\r\n this._updateButtons(null);\r\n const element = this._buttons[idx].element;\r\n const prevTitle = element.title;\r\n element.title = \"Error entering XR session : \" + prevTitle;\r\n element.classList.add(\"xr-error\");\r\n if (this.options.onError) {\r\n this.options.onError(e);\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Disposes of the XR UI component\r\n */\r\n public dispose() {\r\n const renderCanvas = this._scene.getEngine().getInputElement();\r\n if (renderCanvas && renderCanvas.parentNode && renderCanvas.parentNode.contains(this.overlay)) {\r\n renderCanvas.parentNode.removeChild(this.overlay);\r\n }\r\n this.activeButtonChangedObservable.clear();\r\n (navigator as any).xr.removeEventListener(\"sessiongranted\", this._onSessionGranted);\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n private _onSessionGranted = (evt: { session: XRSession }) => {\r\n // This section is for future reference.\r\n // As per specs, evt.session.mode should have the supported session mode, but no browser supports it for now.\r\n\r\n // // check if the session granted is the same as the one requested\r\n // const grantedMode = (evt.session as any).mode;\r\n // if (grantedMode) {\r\n // this._buttons.some((btn, idx) => {\r\n // if (btn.sessionMode === grantedMode) {\r\n // this._enterXRWithButtonIndex(idx);\r\n // return true;\r\n // }\r\n // return false;\r\n // });\r\n // } else\r\n\r\n if (this._helper) {\r\n this._enterXRWithButtonIndex(0);\r\n }\r\n };\r\n\r\n private _updateButtons(activeButton: Nullable) {\r\n this._activeButton = activeButton;\r\n this._buttons.forEach((b) => {\r\n b.update(this._activeButton);\r\n });\r\n this.activeButtonChangedObservable.notifyObservers(this._activeButton);\r\n }\r\n}\r\n","import type { Nullable, FloatArray, IndicesArray } from \"../types\";\r\nimport type { Vector3 } from \"../Maths/math.vector\";\r\nimport { Matrix, TmpVectors } from \"../Maths/math.vector\";\r\nimport { Logger } from \"../Misc/logger\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\nimport type { Node } from \"../node\";\r\nimport { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport { Mesh } from \"../Meshes/mesh\";\r\nimport type { Material } from \"../Materials/material\";\r\nimport type { Skeleton } from \"../Bones/skeleton\";\r\nimport { DeepCopier } from \"../Misc/deepCopier\";\r\nimport { TransformNode } from \"./transformNode\";\r\nimport type { Light } from \"../Lights/light\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport { Tools } from \"../Misc/tools\";\r\n\r\nMesh._instancedMeshFactory = (name: string, mesh: Mesh): InstancedMesh => {\r\n const instance = new InstancedMesh(name, mesh);\r\n\r\n if (mesh.instancedBuffers) {\r\n instance.instancedBuffers = {};\r\n\r\n for (const key in mesh.instancedBuffers) {\r\n instance.instancedBuffers[key] = mesh.instancedBuffers[key];\r\n }\r\n }\r\n\r\n return instance;\r\n};\r\n\r\n/**\r\n * Creates an instance based on a source mesh.\r\n */\r\nexport class InstancedMesh extends AbstractMesh {\r\n private _sourceMesh: Mesh;\r\n private _currentLOD: Mesh;\r\n private _billboardWorldMatrix: Matrix;\r\n\r\n /** @internal */\r\n public _indexInSourceMeshInstanceArray = -1;\r\n /** @internal */\r\n public _distanceToCamera: number = 0;\r\n /** @internal */\r\n public _previousWorldMatrix: Nullable;\r\n\r\n /**\r\n * Creates a new InstancedMesh object from the mesh source.\r\n * @param name defines the name of the instance\r\n * @param source the mesh to create the instance from\r\n */\r\n constructor(name: string, source: Mesh) {\r\n super(name, source.getScene());\r\n\r\n source.addInstance(this);\r\n\r\n this._sourceMesh = source;\r\n\r\n this._unIndexed = source._unIndexed;\r\n\r\n this.position.copyFrom(source.position);\r\n this.rotation.copyFrom(source.rotation);\r\n this.scaling.copyFrom(source.scaling);\r\n\r\n if (source.rotationQuaternion) {\r\n this.rotationQuaternion = source.rotationQuaternion.clone();\r\n }\r\n\r\n this.animations = source.animations.slice();\r\n for (const range of source.getAnimationRanges()) {\r\n if (range != null) {\r\n this.createAnimationRange(range.name, range.from, range.to);\r\n }\r\n }\r\n\r\n this.infiniteDistance = source.infiniteDistance;\r\n\r\n this.setPivotMatrix(source.getPivotMatrix());\r\n\r\n this.refreshBoundingInfo(true, true);\r\n this._syncSubMeshes();\r\n }\r\n\r\n /**\r\n * Returns the string \"InstancedMesh\".\r\n */\r\n public getClassName(): string {\r\n return \"InstancedMesh\";\r\n }\r\n\r\n /** Gets the list of lights affecting that mesh */\r\n public get lightSources(): Light[] {\r\n return this._sourceMesh._lightSources;\r\n }\r\n\r\n public _resyncLightSources(): void {\r\n // Do nothing as all the work will be done by source mesh\r\n }\r\n\r\n public _resyncLightSource(): void {\r\n // Do nothing as all the work will be done by source mesh\r\n }\r\n\r\n public _removeLightSource(): void {\r\n // Do nothing as all the work will be done by source mesh\r\n }\r\n\r\n // Methods\r\n /**\r\n * If the source mesh receives shadows\r\n */\r\n public get receiveShadows(): boolean {\r\n return this._sourceMesh.receiveShadows;\r\n }\r\n\r\n public set receiveShadows(_value: boolean) {\r\n if (this._sourceMesh?.receiveShadows !== _value) {\r\n Tools.Warn(\"Setting receiveShadows on an instanced mesh has no effect\");\r\n }\r\n }\r\n\r\n /**\r\n * The material of the source mesh\r\n */\r\n public get material(): Nullable {\r\n return this._sourceMesh.material;\r\n }\r\n\r\n public set material(_value: Nullable) {\r\n if (this._sourceMesh?.material !== _value) {\r\n Tools.Warn(\"Setting material on an instanced mesh has no effect\");\r\n }\r\n }\r\n\r\n /**\r\n * Visibility of the source mesh\r\n */\r\n public get visibility(): number {\r\n return this._sourceMesh.visibility;\r\n }\r\n\r\n public set visibility(_value: number) {\r\n if (this._sourceMesh?.visibility !== _value) {\r\n Tools.Warn(\"Setting visibility on an instanced mesh has no effect\");\r\n }\r\n }\r\n\r\n /**\r\n * Skeleton of the source mesh\r\n */\r\n public get skeleton(): Nullable {\r\n return this._sourceMesh.skeleton;\r\n }\r\n\r\n public set skeleton(_value: Nullable) {\r\n if (this._sourceMesh?.skeleton !== _value) {\r\n Tools.Warn(\"Setting skeleton on an instanced mesh has no effect\");\r\n }\r\n }\r\n\r\n /**\r\n * Rendering ground id of the source mesh\r\n */\r\n public get renderingGroupId(): number {\r\n return this._sourceMesh.renderingGroupId;\r\n }\r\n\r\n public set renderingGroupId(value: number) {\r\n if (!this._sourceMesh || value === this._sourceMesh.renderingGroupId) {\r\n return;\r\n }\r\n\r\n //no-op with warning\r\n Logger.Warn(\"Note - setting renderingGroupId of an instanced mesh has no effect on the scene\");\r\n }\r\n\r\n /**\r\n * Returns the total number of vertices (integer).\r\n */\r\n public getTotalVertices(): number {\r\n return this._sourceMesh ? this._sourceMesh.getTotalVertices() : 0;\r\n }\r\n\r\n /**\r\n * Returns a positive integer : the total number of indices in this mesh geometry.\r\n * @returns the number of indices or zero if the mesh has no geometry.\r\n */\r\n public getTotalIndices(): number {\r\n return this._sourceMesh.getTotalIndices();\r\n }\r\n\r\n /**\r\n * The source mesh of the instance\r\n */\r\n public get sourceMesh(): Mesh {\r\n return this._sourceMesh;\r\n }\r\n\r\n /**\r\n * Creates a new InstancedMesh object from the mesh model.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/copies/instances\r\n * @param name defines the name of the new instance\r\n * @returns a new InstancedMesh\r\n */\r\n public createInstance(name: string): InstancedMesh {\r\n return this._sourceMesh.createInstance(name);\r\n }\r\n\r\n /**\r\n * Is this node ready to be used/rendered\r\n * @param completeCheck defines if a complete check (including materials and lights) has to be done (false by default)\r\n * @returns {boolean} is it ready\r\n */\r\n public isReady(completeCheck = false): boolean {\r\n return this._sourceMesh.isReady(completeCheck, true);\r\n }\r\n\r\n /**\r\n * Returns an array of integers or a typed array (Int32Array, Uint32Array, Uint16Array) populated with the mesh indices.\r\n * @param kind kind of verticies to retrieve (eg. positions, normals, uvs, etc.)\r\n * @param copyWhenShared If true (default false) and and if the mesh geometry is shared among some other meshes, the returned array is a copy of the internal one.\r\n * @param forceCopy defines a boolean forcing the copy of the buffer no matter what the value of copyWhenShared is\r\n * @returns a float array or a Float32Array of the requested kind of data : positions, normals, uvs, etc.\r\n */\r\n public getVerticesData(kind: string, copyWhenShared?: boolean, forceCopy?: boolean): Nullable {\r\n return this._sourceMesh.getVerticesData(kind, copyWhenShared, forceCopy);\r\n }\r\n\r\n /**\r\n * Sets the vertex data of the mesh geometry for the requested `kind`.\r\n * If the mesh has no geometry, a new Geometry object is set to the mesh and then passed this vertex data.\r\n * The `data` are either a numeric array either a Float32Array.\r\n * The parameter `updatable` is passed as is to the underlying Geometry object constructor (if initially none) or updater.\r\n * The parameter `stride` is an optional positive integer, it is usually automatically deducted from the `kind` (3 for positions or normals, 2 for UV, etc).\r\n * Note that a new underlying VertexBuffer object is created each call.\r\n * If the `kind` is the `PositionKind`, the mesh BoundingInfo is renewed, so the bounding box and sphere, and the mesh World Matrix is recomputed.\r\n *\r\n * Possible `kind` values :\r\n * - VertexBuffer.PositionKind\r\n * - VertexBuffer.UVKind\r\n * - VertexBuffer.UV2Kind\r\n * - VertexBuffer.UV3Kind\r\n * - VertexBuffer.UV4Kind\r\n * - VertexBuffer.UV5Kind\r\n * - VertexBuffer.UV6Kind\r\n * - VertexBuffer.ColorKind\r\n * - VertexBuffer.MatricesIndicesKind\r\n * - VertexBuffer.MatricesIndicesExtraKind\r\n * - VertexBuffer.MatricesWeightsKind\r\n * - VertexBuffer.MatricesWeightsExtraKind\r\n *\r\n * Returns the Mesh.\r\n * @param kind\r\n * @param data\r\n * @param updatable\r\n * @param stride\r\n */\r\n public setVerticesData(kind: string, data: FloatArray, updatable?: boolean, stride?: number): AbstractMesh {\r\n if (this.sourceMesh) {\r\n this.sourceMesh.setVerticesData(kind, data, updatable, stride);\r\n }\r\n return this.sourceMesh;\r\n }\r\n\r\n /**\r\n * Updates the existing vertex data of the mesh geometry for the requested `kind`.\r\n * If the mesh has no geometry, it is simply returned as it is.\r\n * The `data` are either a numeric array either a Float32Array.\r\n * No new underlying VertexBuffer object is created.\r\n * If the `kind` is the `PositionKind` and if `updateExtends` is true, the mesh BoundingInfo is renewed, so the bounding box and sphere, and the mesh World Matrix is recomputed.\r\n * If the parameter `makeItUnique` is true, a new global geometry is created from this positions and is set to the mesh.\r\n *\r\n * Possible `kind` values :\r\n * - VertexBuffer.PositionKind\r\n * - VertexBuffer.UVKind\r\n * - VertexBuffer.UV2Kind\r\n * - VertexBuffer.UV3Kind\r\n * - VertexBuffer.UV4Kind\r\n * - VertexBuffer.UV5Kind\r\n * - VertexBuffer.UV6Kind\r\n * - VertexBuffer.ColorKind\r\n * - VertexBuffer.MatricesIndicesKind\r\n * - VertexBuffer.MatricesIndicesExtraKind\r\n * - VertexBuffer.MatricesWeightsKind\r\n * - VertexBuffer.MatricesWeightsExtraKind\r\n *\r\n * Returns the Mesh.\r\n * @param kind\r\n * @param data\r\n * @param updateExtends\r\n * @param makeItUnique\r\n */\r\n public updateVerticesData(kind: string, data: FloatArray, updateExtends?: boolean, makeItUnique?: boolean): Mesh {\r\n if (this.sourceMesh) {\r\n this.sourceMesh.updateVerticesData(kind, data, updateExtends, makeItUnique);\r\n }\r\n return this.sourceMesh;\r\n }\r\n\r\n /**\r\n * Sets the mesh indices.\r\n * Expects an array populated with integers or a typed array (Int32Array, Uint32Array, Uint16Array).\r\n * If the mesh has no geometry, a new Geometry object is created and set to the mesh.\r\n * This method creates a new index buffer each call.\r\n * Returns the Mesh.\r\n * @param indices\r\n * @param totalVertices\r\n */\r\n public setIndices(indices: IndicesArray, totalVertices: Nullable = null): Mesh {\r\n if (this.sourceMesh) {\r\n this.sourceMesh.setIndices(indices, totalVertices);\r\n }\r\n return this.sourceMesh;\r\n }\r\n\r\n /**\r\n * Boolean : True if the mesh owns the requested kind of data.\r\n * @param kind\r\n */\r\n public isVerticesDataPresent(kind: string): boolean {\r\n return this._sourceMesh.isVerticesDataPresent(kind);\r\n }\r\n\r\n /**\r\n * Returns an array of indices (IndicesArray).\r\n */\r\n public getIndices(): Nullable {\r\n return this._sourceMesh.getIndices();\r\n }\r\n\r\n public get _positions(): Nullable {\r\n return this._sourceMesh._positions;\r\n }\r\n\r\n /**\r\n * This method recomputes and sets a new BoundingInfo to the mesh unless it is locked.\r\n * This means the mesh underlying bounding box and sphere are recomputed.\r\n * @param applySkeleton defines whether to apply the skeleton before computing the bounding info\r\n * @param applyMorph defines whether to apply the morph target before computing the bounding info\r\n * @returns the current mesh\r\n */\r\n public refreshBoundingInfo(applySkeleton: boolean = false, applyMorph: boolean = false): InstancedMesh {\r\n if (this.hasBoundingInfo && this.getBoundingInfo().isLocked) {\r\n return this;\r\n }\r\n\r\n const bias = this._sourceMesh.geometry ? this._sourceMesh.geometry.boundingBias : null;\r\n this._refreshBoundingInfo(this._sourceMesh._getPositionData(applySkeleton, applyMorph), bias);\r\n return this;\r\n }\r\n\r\n /** @internal */\r\n public _preActivate(): InstancedMesh {\r\n if (this._currentLOD) {\r\n this._currentLOD._preActivate();\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _activate(renderId: number, intermediateRendering: boolean): boolean {\r\n super._activate(renderId, intermediateRendering);\r\n\r\n if (!this._sourceMesh.subMeshes) {\r\n Logger.Warn(\"Instances should only be created for meshes with geometry.\");\r\n }\r\n\r\n if (this._currentLOD) {\r\n const differentSign = this._currentLOD._getWorldMatrixDeterminant() >= 0 !== this._getWorldMatrixDeterminant() >= 0;\r\n if (differentSign) {\r\n this._internalAbstractMeshDataInfo._actAsRegularMesh = true;\r\n return true;\r\n }\r\n this._internalAbstractMeshDataInfo._actAsRegularMesh = false;\r\n\r\n this._currentLOD._registerInstanceForRenderId(this, renderId);\r\n\r\n if (intermediateRendering) {\r\n if (!this._currentLOD._internalAbstractMeshDataInfo._isActiveIntermediate) {\r\n this._currentLOD._internalAbstractMeshDataInfo._onlyForInstancesIntermediate = true;\r\n return true;\r\n }\r\n } else {\r\n if (!this._currentLOD._internalAbstractMeshDataInfo._isActive) {\r\n this._currentLOD._internalAbstractMeshDataInfo._onlyForInstances = true;\r\n return true;\r\n }\r\n }\r\n }\r\n return false;\r\n }\r\n\r\n /** @internal */\r\n public _postActivate(): void {\r\n if (this._sourceMesh.edgesShareWithInstances && this._sourceMesh._edgesRenderer && this._sourceMesh._edgesRenderer.isEnabled && this._sourceMesh._renderingGroup) {\r\n // we are using the edge renderer of the source mesh\r\n this._sourceMesh._renderingGroup._edgesRenderers.pushNoDuplicate(this._sourceMesh._edgesRenderer);\r\n this._sourceMesh._edgesRenderer.customInstances.push(this.getWorldMatrix());\r\n } else if (this._edgesRenderer && this._edgesRenderer.isEnabled && this._sourceMesh._renderingGroup) {\r\n // we are using the edge renderer defined for this instance\r\n this._sourceMesh._renderingGroup._edgesRenderers.push(this._edgesRenderer);\r\n }\r\n }\r\n\r\n public getWorldMatrix(): Matrix {\r\n if (this._currentLOD && this._currentLOD.billboardMode !== TransformNode.BILLBOARDMODE_NONE && this._currentLOD._masterMesh !== this) {\r\n if (!this._billboardWorldMatrix) {\r\n this._billboardWorldMatrix = new Matrix();\r\n }\r\n const tempMaster = this._currentLOD._masterMesh;\r\n this._currentLOD._masterMesh = this;\r\n TmpVectors.Vector3[7].copyFrom(this._currentLOD.position);\r\n this._currentLOD.position.set(0, 0, 0);\r\n this._billboardWorldMatrix.copyFrom(this._currentLOD.computeWorldMatrix(true));\r\n this._currentLOD.position.copyFrom(TmpVectors.Vector3[7]);\r\n this._currentLOD._masterMesh = tempMaster;\r\n return this._billboardWorldMatrix;\r\n }\r\n\r\n return super.getWorldMatrix();\r\n }\r\n\r\n public get isAnInstance(): boolean {\r\n return true;\r\n }\r\n\r\n /**\r\n * Returns the current associated LOD AbstractMesh.\r\n * @param camera\r\n */\r\n public getLOD(camera: Camera): AbstractMesh {\r\n if (!camera) {\r\n return this;\r\n }\r\n\r\n const sourceMeshLODLevels = this.sourceMesh.getLODLevels();\r\n if (!sourceMeshLODLevels || sourceMeshLODLevels.length === 0) {\r\n this._currentLOD = this.sourceMesh;\r\n } else {\r\n const boundingInfo = this.getBoundingInfo();\r\n this._currentLOD = this.sourceMesh.getLOD(camera, boundingInfo.boundingSphere);\r\n }\r\n\r\n return this._currentLOD;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _preActivateForIntermediateRendering(renderId: number): Mesh {\r\n return this.sourceMesh._preActivateForIntermediateRendering(renderId);\r\n }\r\n\r\n /** @internal */\r\n public _syncSubMeshes(): InstancedMesh {\r\n this.releaseSubMeshes();\r\n if (this._sourceMesh.subMeshes) {\r\n for (let index = 0; index < this._sourceMesh.subMeshes.length; index++) {\r\n this._sourceMesh.subMeshes[index].clone(this, this._sourceMesh);\r\n }\r\n }\r\n return this;\r\n }\r\n\r\n /** @internal */\r\n public _generatePointsArray(): boolean {\r\n return this._sourceMesh._generatePointsArray();\r\n }\r\n\r\n /** @internal */\r\n public _updateBoundingInfo(): AbstractMesh {\r\n if (this.hasBoundingInfo) {\r\n this.getBoundingInfo().update(this.worldMatrixFromCache);\r\n } else {\r\n this.buildBoundingInfo(this.absolutePosition, this.absolutePosition, this.worldMatrixFromCache);\r\n }\r\n this._updateSubMeshesBoundingInfo(this.worldMatrixFromCache);\r\n return this;\r\n }\r\n\r\n /**\r\n * Creates a new InstancedMesh from the current mesh.\r\n *\r\n * Returns the clone.\r\n * @param name the cloned mesh name\r\n * @param newParent the optional Node to parent the clone to.\r\n * @param doNotCloneChildren if `true` the model children aren't cloned.\r\n * @param newSourceMesh if set this mesh will be used as the source mesh instead of ths instance's one\r\n * @returns the clone\r\n */\r\n public clone(name: string, newParent: Nullable = null, doNotCloneChildren?: boolean, newSourceMesh?: Mesh): InstancedMesh {\r\n const result = (newSourceMesh || this._sourceMesh).createInstance(name);\r\n\r\n // Deep copy\r\n DeepCopier.DeepCopy(\r\n this,\r\n result,\r\n [\r\n \"name\",\r\n \"subMeshes\",\r\n \"uniqueId\",\r\n \"parent\",\r\n \"lightSources\",\r\n \"receiveShadows\",\r\n \"material\",\r\n \"visibility\",\r\n \"skeleton\",\r\n \"sourceMesh\",\r\n \"isAnInstance\",\r\n \"facetNb\",\r\n \"isFacetDataEnabled\",\r\n \"isBlocked\",\r\n \"useBones\",\r\n \"hasInstances\",\r\n \"collider\",\r\n \"edgesRenderer\",\r\n \"forward\",\r\n \"up\",\r\n \"right\",\r\n \"absolutePosition\",\r\n \"absoluteScaling\",\r\n \"absoluteRotationQuaternion\",\r\n \"isWorldMatrixFrozen\",\r\n \"nonUniformScaling\",\r\n \"behaviors\",\r\n \"worldMatrixFromCache\",\r\n \"hasThinInstances\",\r\n \"hasBoundingInfo\",\r\n ],\r\n []\r\n );\r\n\r\n // Bounding info\r\n this.refreshBoundingInfo();\r\n\r\n // Parent\r\n if (newParent) {\r\n result.parent = newParent;\r\n }\r\n\r\n if (!doNotCloneChildren) {\r\n // Children\r\n for (let index = 0; index < this.getScene().meshes.length; index++) {\r\n const mesh = this.getScene().meshes[index];\r\n\r\n if (mesh.parent === this) {\r\n mesh.clone(mesh.name, result);\r\n }\r\n }\r\n }\r\n\r\n result.computeWorldMatrix(true);\r\n\r\n this.onClonedObservable.notifyObservers(result);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Disposes the InstancedMesh.\r\n * Returns nothing.\r\n * @param doNotRecurse\r\n * @param disposeMaterialAndTextures\r\n */\r\n public dispose(doNotRecurse?: boolean, disposeMaterialAndTextures = false): void {\r\n // Remove from mesh\r\n this._sourceMesh.removeInstance(this);\r\n super.dispose(doNotRecurse, disposeMaterialAndTextures);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _serializeAsParent(serializationObject: any) {\r\n super._serializeAsParent(serializationObject);\r\n\r\n serializationObject.parentId = this._sourceMesh.uniqueId;\r\n serializationObject.parentInstanceIndex = this._indexInSourceMeshInstanceArray;\r\n }\r\n\r\n /**\r\n * Instantiate (when possible) or clone that node with its hierarchy\r\n * @param newParent defines the new parent to use for the instance (or clone)\r\n * @param options defines options to configure how copy is done\r\n * @param options.doNotInstantiate defines if the model must be instantiated or just cloned\r\n * @param options.newSourcedMesh newSourcedMesh the new source mesh for the instance (or clone)\r\n * @param onNewNodeCreated defines an option callback to call when a clone or an instance is created\r\n * @returns an instance (or a clone) of the current node with its hierarchy\r\n */\r\n public instantiateHierarchy(\r\n newParent: Nullable = null,\r\n options?: { doNotInstantiate: boolean | ((node: TransformNode) => boolean); newSourcedMesh?: Mesh },\r\n onNewNodeCreated?: (source: TransformNode, clone: TransformNode) => void\r\n ): Nullable {\r\n const clone = this.clone(\"Clone of \" + (this.name || this.id), newParent || this.parent, true, options && options.newSourcedMesh);\r\n\r\n if (clone) {\r\n if (onNewNodeCreated) {\r\n onNewNodeCreated(this, clone);\r\n }\r\n }\r\n\r\n for (const child of this.getChildTransformNodes(true)) {\r\n child.instantiateHierarchy(clone, options, onNewNodeCreated);\r\n }\r\n\r\n return clone;\r\n }\r\n}\r\n\r\ndeclare module \"./mesh\" {\r\n export interface Mesh {\r\n /**\r\n * Register a custom buffer that will be instanced\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/copies/instances#custom-buffers\r\n * @param kind defines the buffer kind\r\n * @param stride defines the stride in floats\r\n */\r\n registerInstancedBuffer(kind: string, stride: number): void;\r\n\r\n /**\r\n * Invalidate VertexArrayObjects belonging to the mesh (but not to the Geometry of the mesh).\r\n */\r\n _invalidateInstanceVertexArrayObject(): void;\r\n\r\n /**\r\n * true to use the edge renderer for all instances of this mesh\r\n */\r\n edgesShareWithInstances: boolean;\r\n\r\n /** @internal */\r\n _userInstancedBuffersStorage: {\r\n data: { [key: string]: Float32Array };\r\n sizes: { [key: string]: number };\r\n vertexBuffers: { [key: string]: Nullable };\r\n strides: { [key: string]: number };\r\n vertexArrayObjects?: { [key: string]: WebGLVertexArrayObject };\r\n };\r\n }\r\n}\r\n\r\ndeclare module \"./abstractMesh\" {\r\n export interface AbstractMesh {\r\n /**\r\n * Object used to store instanced buffers defined by user\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/copies/instances#custom-buffers\r\n */\r\n instancedBuffers: { [key: string]: any };\r\n }\r\n}\r\n\r\nMesh.prototype.registerInstancedBuffer = function (kind: string, stride: number): void {\r\n // Remove existing one\r\n this._userInstancedBuffersStorage?.vertexBuffers[kind]?.dispose();\r\n\r\n // Creates the instancedBuffer field if not present\r\n if (!this.instancedBuffers) {\r\n this.instancedBuffers = {};\r\n\r\n for (const instance of this.instances) {\r\n instance.instancedBuffers = {};\r\n }\r\n\r\n if (!this._userInstancedBuffersStorage) {\r\n this._userInstancedBuffersStorage = {\r\n data: {},\r\n vertexBuffers: {},\r\n strides: {},\r\n sizes: {},\r\n vertexArrayObjects: this.getEngine().getCaps().vertexArrayObject ? {} : undefined,\r\n };\r\n }\r\n }\r\n\r\n // Creates an empty property for this kind\r\n this.instancedBuffers[kind] = null;\r\n\r\n this._userInstancedBuffersStorage.strides[kind] = stride;\r\n this._userInstancedBuffersStorage.sizes[kind] = stride * 32; // Initial size\r\n this._userInstancedBuffersStorage.data[kind] = new Float32Array(this._userInstancedBuffersStorage.sizes[kind]);\r\n this._userInstancedBuffersStorage.vertexBuffers[kind] = new VertexBuffer(this.getEngine(), this._userInstancedBuffersStorage.data[kind], kind, true, false, stride, true);\r\n\r\n for (const instance of this.instances) {\r\n instance.instancedBuffers[kind] = null;\r\n }\r\n\r\n this._invalidateInstanceVertexArrayObject();\r\n\r\n this._markSubMeshesAsAttributesDirty();\r\n};\r\n\r\nMesh.prototype._processInstancedBuffers = function (visibleInstances: Nullable, renderSelf: boolean) {\r\n const instanceCount = visibleInstances ? visibleInstances.length : 0;\r\n\r\n for (const kind in this.instancedBuffers) {\r\n let size = this._userInstancedBuffersStorage.sizes[kind];\r\n const stride = this._userInstancedBuffersStorage.strides[kind];\r\n\r\n // Resize if required\r\n const expectedSize = (instanceCount + 1) * stride;\r\n\r\n while (size < expectedSize) {\r\n size *= 2;\r\n }\r\n\r\n if (this._userInstancedBuffersStorage.data[kind].length != size) {\r\n this._userInstancedBuffersStorage.data[kind] = new Float32Array(size);\r\n this._userInstancedBuffersStorage.sizes[kind] = size;\r\n if (this._userInstancedBuffersStorage.vertexBuffers[kind]) {\r\n this._userInstancedBuffersStorage.vertexBuffers[kind]!.dispose();\r\n this._userInstancedBuffersStorage.vertexBuffers[kind] = null;\r\n }\r\n }\r\n\r\n const data = this._userInstancedBuffersStorage.data[kind];\r\n\r\n // Update data buffer\r\n let offset = 0;\r\n if (renderSelf) {\r\n const value = this.instancedBuffers[kind];\r\n\r\n if (value.toArray) {\r\n value.toArray(data, offset);\r\n } else if (value.copyToArray) {\r\n value.copyToArray(data, offset);\r\n } else {\r\n data[offset] = value;\r\n }\r\n\r\n offset += stride;\r\n }\r\n\r\n for (let instanceIndex = 0; instanceIndex < instanceCount; instanceIndex++) {\r\n const instance = visibleInstances![instanceIndex]!;\r\n\r\n const value = instance.instancedBuffers[kind];\r\n\r\n if (value.toArray) {\r\n value.toArray(data, offset);\r\n } else if (value.copyToArray) {\r\n value.copyToArray(data, offset);\r\n } else {\r\n data[offset] = value;\r\n }\r\n\r\n offset += stride;\r\n }\r\n\r\n // Update vertex buffer\r\n if (!this._userInstancedBuffersStorage.vertexBuffers[kind]) {\r\n this._userInstancedBuffersStorage.vertexBuffers[kind] = new VertexBuffer(\r\n this.getEngine(),\r\n this._userInstancedBuffersStorage.data[kind],\r\n kind,\r\n true,\r\n false,\r\n stride,\r\n true\r\n );\r\n this._invalidateInstanceVertexArrayObject();\r\n } else {\r\n this._userInstancedBuffersStorage.vertexBuffers[kind]!.updateDirectly(data, 0);\r\n }\r\n }\r\n};\r\n\r\nMesh.prototype._invalidateInstanceVertexArrayObject = function () {\r\n if (!this._userInstancedBuffersStorage || this._userInstancedBuffersStorage.vertexArrayObjects === undefined) {\r\n return;\r\n }\r\n\r\n for (const kind in this._userInstancedBuffersStorage.vertexArrayObjects) {\r\n this.getEngine().releaseVertexArrayObject(this._userInstancedBuffersStorage.vertexArrayObjects[kind]);\r\n }\r\n\r\n this._userInstancedBuffersStorage.vertexArrayObjects = {};\r\n};\r\n\r\nMesh.prototype._disposeInstanceSpecificData = function () {\r\n if (this._instanceDataStorage.instancesBuffer) {\r\n this._instanceDataStorage.instancesBuffer.dispose();\r\n this._instanceDataStorage.instancesBuffer = null;\r\n }\r\n\r\n while (this.instances.length) {\r\n this.instances[0].dispose();\r\n }\r\n\r\n for (const kind in this.instancedBuffers) {\r\n if (this._userInstancedBuffersStorage.vertexBuffers[kind]) {\r\n this._userInstancedBuffersStorage.vertexBuffers[kind]!.dispose();\r\n }\r\n }\r\n\r\n this._invalidateInstanceVertexArrayObject();\r\n\r\n this.instancedBuffers = {};\r\n};\r\n","import { SerializationHelper } from \"../Misc/decorators\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Matrix, Vector3, Vector2, Vector4, Quaternion } from \"../Maths/math.vector\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { Mesh } from \"../Meshes/mesh\";\r\nimport type { SubMesh } from \"../Meshes/subMesh\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport type { BaseTexture } from \"../Materials/Textures/baseTexture\";\r\nimport { Texture } from \"../Materials/Textures/texture\";\r\nimport { MaterialHelper } from \"./materialHelper\";\r\nimport type { Effect, IEffectCreationOptions } from \"./effect\";\r\nimport { RegisterClass } from \"../Misc/typeStore\";\r\nimport { Color3, Color4 } from \"../Maths/math.color\";\r\nimport { EffectFallbacks } from \"./effectFallbacks\";\r\nimport { WebRequest } from \"../Misc/webRequest\";\r\nimport type { ShaderLanguage } from \"./shaderLanguage\";\r\nimport type { UniformBuffer } from \"./uniformBuffer\";\r\nimport type { TextureSampler } from \"./Textures/textureSampler\";\r\nimport type { StorageBuffer } from \"../Buffers/storageBuffer\";\r\nimport { PushMaterial } from \"./pushMaterial\";\r\nimport { EngineStore } from \"../Engines/engineStore\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport { addClipPlaneUniforms, bindClipPlane, prepareStringDefinesForClipPlanes } from \"./clipPlaneMaterialHelper\";\r\n\r\nimport type { ExternalTexture } from \"./Textures/externalTexture\";\r\n\r\nconst onCreatedEffectParameters = { effect: null as unknown as Effect, subMesh: null as unknown as Nullable };\r\n\r\n/**\r\n * Defines the options associated with the creation of a shader material.\r\n */\r\nexport interface IShaderMaterialOptions {\r\n /**\r\n * Does the material work in alpha blend mode\r\n */\r\n needAlphaBlending: boolean;\r\n\r\n /**\r\n * Does the material work in alpha test mode\r\n */\r\n needAlphaTesting: boolean;\r\n\r\n /**\r\n * The list of attribute names used in the shader\r\n */\r\n attributes: string[];\r\n\r\n /**\r\n * The list of uniform names used in the shader\r\n */\r\n uniforms: string[];\r\n\r\n /**\r\n * The list of UBO names used in the shader\r\n */\r\n uniformBuffers: string[];\r\n\r\n /**\r\n * The list of sampler (texture) names used in the shader\r\n */\r\n samplers: string[];\r\n\r\n /**\r\n * The list of external texture names used in the shader\r\n */\r\n externalTextures: string[];\r\n\r\n /**\r\n * The list of sampler object names used in the shader\r\n */\r\n samplerObjects: string[];\r\n\r\n /**\r\n * The list of storage buffer names used in the shader\r\n */\r\n storageBuffers: string[];\r\n\r\n /**\r\n * The list of defines used in the shader\r\n */\r\n defines: string[];\r\n\r\n /**\r\n * Defines if clip planes have to be turned on: true to turn them on, false to turn them off and null to turn them on/off depending on the scene configuration (scene.clipPlaneX)\r\n */\r\n useClipPlane: Nullable;\r\n\r\n /**\r\n * The language the shader is written in (default: GLSL)\r\n */\r\n shaderLanguage?: ShaderLanguage;\r\n}\r\n\r\n/**\r\n * The ShaderMaterial object has the necessary methods to pass data from your scene to the Vertex and Fragment Shaders and returns a material that can be applied to any mesh.\r\n *\r\n * This returned material effects how the mesh will look based on the code in the shaders.\r\n *\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/shaders/shaderMaterial\r\n */\r\nexport class ShaderMaterial extends PushMaterial {\r\n private _shaderPath: any;\r\n private _options: IShaderMaterialOptions;\r\n private _textures: { [name: string]: BaseTexture } = {};\r\n private _textureArrays: { [name: string]: BaseTexture[] } = {};\r\n private _externalTextures: { [name: string]: ExternalTexture } = {};\r\n private _floats: { [name: string]: number } = {};\r\n private _ints: { [name: string]: number } = {};\r\n private _uints: { [name: string]: number } = {};\r\n private _floatsArrays: { [name: string]: number[] } = {};\r\n private _colors3: { [name: string]: Color3 } = {};\r\n private _colors3Arrays: { [name: string]: number[] } = {};\r\n private _colors4: { [name: string]: Color4 } = {};\r\n private _colors4Arrays: { [name: string]: number[] } = {};\r\n private _vectors2: { [name: string]: Vector2 } = {};\r\n private _vectors3: { [name: string]: Vector3 } = {};\r\n private _vectors4: { [name: string]: Vector4 } = {};\r\n private _quaternions: { [name: string]: Quaternion } = {};\r\n private _quaternionsArrays: { [name: string]: number[] } = {};\r\n private _matrices: { [name: string]: Matrix } = {};\r\n private _matrixArrays: { [name: string]: Float32Array | Array } = {};\r\n private _matrices3x3: { [name: string]: Float32Array | Array } = {};\r\n private _matrices2x2: { [name: string]: Float32Array | Array } = {};\r\n private _vectors2Arrays: { [name: string]: number[] } = {};\r\n private _vectors3Arrays: { [name: string]: number[] } = {};\r\n private _vectors4Arrays: { [name: string]: number[] } = {};\r\n private _uniformBuffers: { [name: string]: UniformBuffer } = {};\r\n private _textureSamplers: { [name: string]: TextureSampler } = {};\r\n private _storageBuffers: { [name: string]: StorageBuffer } = {};\r\n private _cachedWorldViewMatrix = new Matrix();\r\n private _cachedWorldViewProjectionMatrix = new Matrix();\r\n private _multiview = false;\r\n\r\n /**\r\n * @internal\r\n */\r\n public _materialHelperNeedsPreviousMatrices = false;\r\n\r\n /** Define the Url to load snippets */\r\n public static SnippetUrl = Constants.SnippetUrl;\r\n\r\n /** Snippet ID if the material was created from the snippet server */\r\n public snippetId: string;\r\n\r\n /**\r\n * Instantiate a new shader material.\r\n * The ShaderMaterial object has the necessary methods to pass data from your scene to the Vertex and Fragment Shaders and returns a material that can be applied to any mesh.\r\n * This returned material effects how the mesh will look based on the code in the shaders.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/shaders/shaderMaterial\r\n * @param name Define the name of the material in the scene\r\n * @param scene Define the scene the material belongs to\r\n * @param shaderPath Defines the route to the shader code in one of three ways:\r\n * * object: \\{ vertex: \"custom\", fragment: \"custom\" \\}, used with Effect.ShadersStore[\"customVertexShader\"] and Effect.ShadersStore[\"customFragmentShader\"]\r\n * * object: \\{ vertexElement: \"vertexShaderCode\", fragmentElement: \"fragmentShaderCode\" \\}, used with shader code in script tags\r\n * * object: \\{ vertexSource: \"vertex shader code string\", fragmentSource: \"fragment shader code string\" \\} using with strings containing the shaders code\r\n * * string: \"./COMMON_NAME\", used with external files COMMON_NAME.vertex.fx and COMMON_NAME.fragment.fx in index.html folder.\r\n * @param options Define the options used to create the shader\r\n * @param storeEffectOnSubMeshes true to store effect on submeshes, false to store the effect directly in the material class.\r\n */\r\n constructor(name: string, scene: Scene, shaderPath: any, options: Partial = {}, storeEffectOnSubMeshes = true) {\r\n super(name, scene, storeEffectOnSubMeshes);\r\n this._shaderPath = shaderPath;\r\n\r\n this._options = {\r\n needAlphaBlending: false,\r\n needAlphaTesting: false,\r\n attributes: [\"position\", \"normal\", \"uv\"],\r\n uniforms: [\"worldViewProjection\"],\r\n uniformBuffers: [],\r\n samplers: [],\r\n externalTextures: [],\r\n samplerObjects: [],\r\n storageBuffers: [],\r\n defines: [],\r\n useClipPlane: false,\r\n ...options,\r\n };\r\n }\r\n\r\n /**\r\n * Gets the shader path used to define the shader code\r\n * It can be modified to trigger a new compilation\r\n */\r\n public get shaderPath(): any {\r\n return this._shaderPath;\r\n }\r\n\r\n /**\r\n * Sets the shader path used to define the shader code\r\n * It can be modified to trigger a new compilation\r\n */\r\n public set shaderPath(shaderPath: any) {\r\n this._shaderPath = shaderPath;\r\n }\r\n\r\n /**\r\n * Gets the options used to compile the shader.\r\n * They can be modified to trigger a new compilation\r\n */\r\n public get options(): IShaderMaterialOptions {\r\n return this._options;\r\n }\r\n\r\n /**\r\n * is multiview set to true?\r\n */\r\n public get isMultiview(): boolean {\r\n return this._multiview;\r\n }\r\n\r\n /**\r\n * Gets the current class name of the material e.g. \"ShaderMaterial\"\r\n * Mainly use in serialization.\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"ShaderMaterial\";\r\n }\r\n\r\n /**\r\n * Specifies if the material will require alpha blending\r\n * @returns a boolean specifying if alpha blending is needed\r\n */\r\n public needAlphaBlending(): boolean {\r\n return this.alpha < 1.0 || this._options.needAlphaBlending;\r\n }\r\n\r\n /**\r\n * Specifies if this material should be rendered in alpha test mode\r\n * @returns a boolean specifying if an alpha test is needed.\r\n */\r\n public needAlphaTesting(): boolean {\r\n return this._options.needAlphaTesting;\r\n }\r\n\r\n private _checkUniform(uniformName: string): void {\r\n if (this._options.uniforms.indexOf(uniformName) === -1) {\r\n this._options.uniforms.push(uniformName);\r\n }\r\n }\r\n\r\n /**\r\n * Set a texture in the shader.\r\n * @param name Define the name of the uniform samplers as defined in the shader\r\n * @param texture Define the texture to bind to this sampler\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setTexture(name: string, texture: BaseTexture): ShaderMaterial {\r\n if (this._options.samplers.indexOf(name) === -1) {\r\n this._options.samplers.push(name);\r\n }\r\n this._textures[name] = texture;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a texture array in the shader.\r\n * @param name Define the name of the uniform sampler array as defined in the shader\r\n * @param textures Define the list of textures to bind to this sampler\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setTextureArray(name: string, textures: BaseTexture[]): ShaderMaterial {\r\n if (this._options.samplers.indexOf(name) === -1) {\r\n this._options.samplers.push(name);\r\n }\r\n\r\n this._checkUniform(name);\r\n\r\n this._textureArrays[name] = textures;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set an internal texture in the shader.\r\n * @param name Define the name of the uniform samplers as defined in the shader\r\n * @param texture Define the texture to bind to this sampler\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setExternalTexture(name: string, texture: ExternalTexture): ShaderMaterial {\r\n if (this._options.externalTextures.indexOf(name) === -1) {\r\n this._options.externalTextures.push(name);\r\n }\r\n this._externalTextures[name] = texture;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a float in the shader.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setFloat(name: string, value: number): ShaderMaterial {\r\n this._checkUniform(name);\r\n this._floats[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a int in the shader.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setInt(name: string, value: number): ShaderMaterial {\r\n this._checkUniform(name);\r\n this._ints[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a unsigned int in the shader.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setUInt(name: string, value: number): ShaderMaterial {\r\n this._checkUniform(name);\r\n this._uints[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set an array of floats in the shader.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setFloats(name: string, value: number[]): ShaderMaterial {\r\n this._checkUniform(name);\r\n this._floatsArrays[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a vec3 in the shader from a Color3.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setColor3(name: string, value: Color3): ShaderMaterial {\r\n this._checkUniform(name);\r\n this._colors3[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a vec3 array in the shader from a Color3 array.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setColor3Array(name: string, value: Color3[]): ShaderMaterial {\r\n this._checkUniform(name);\r\n this._colors3Arrays[name] = value.reduce((arr, color) => {\r\n color.toArray(arr, arr.length);\r\n return arr;\r\n }, []);\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a vec4 in the shader from a Color4.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setColor4(name: string, value: Color4): ShaderMaterial {\r\n this._checkUniform(name);\r\n this._colors4[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a vec4 array in the shader from a Color4 array.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setColor4Array(name: string, value: Color4[]): ShaderMaterial {\r\n this._checkUniform(name);\r\n this._colors4Arrays[name] = value.reduce((arr, color) => {\r\n color.toArray(arr, arr.length);\r\n return arr;\r\n }, []);\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a vec2 in the shader from a Vector2.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setVector2(name: string, value: Vector2): ShaderMaterial {\r\n this._checkUniform(name);\r\n this._vectors2[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a vec3 in the shader from a Vector3.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setVector3(name: string, value: Vector3): ShaderMaterial {\r\n this._checkUniform(name);\r\n this._vectors3[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a vec4 in the shader from a Vector4.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setVector4(name: string, value: Vector4): ShaderMaterial {\r\n this._checkUniform(name);\r\n this._vectors4[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a vec4 in the shader from a Quaternion.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setQuaternion(name: string, value: Quaternion): ShaderMaterial {\r\n this._checkUniform(name);\r\n this._quaternions[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a vec4 array in the shader from a Quaternion array.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setQuaternionArray(name: string, value: Quaternion[]): ShaderMaterial {\r\n this._checkUniform(name);\r\n this._quaternionsArrays[name] = value.reduce((arr, quaternion) => {\r\n quaternion.toArray(arr, arr.length);\r\n return arr;\r\n }, []);\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a mat4 in the shader from a Matrix.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setMatrix(name: string, value: Matrix): ShaderMaterial {\r\n this._checkUniform(name);\r\n this._matrices[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a float32Array in the shader from a matrix array.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setMatrices(name: string, value: Matrix[]): ShaderMaterial {\r\n this._checkUniform(name);\r\n\r\n const float32Array = new Float32Array(value.length * 16);\r\n\r\n for (let index = 0; index < value.length; index++) {\r\n const matrix = value[index];\r\n\r\n matrix.copyToArray(float32Array, index * 16);\r\n }\r\n\r\n this._matrixArrays[name] = float32Array;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a mat3 in the shader from a Float32Array.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setMatrix3x3(name: string, value: Float32Array | Array): ShaderMaterial {\r\n this._checkUniform(name);\r\n this._matrices3x3[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a mat2 in the shader from a Float32Array.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setMatrix2x2(name: string, value: Float32Array | Array): ShaderMaterial {\r\n this._checkUniform(name);\r\n this._matrices2x2[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a vec2 array in the shader from a number array.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setArray2(name: string, value: number[]): ShaderMaterial {\r\n this._checkUniform(name);\r\n this._vectors2Arrays[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a vec3 array in the shader from a number array.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setArray3(name: string, value: number[]): ShaderMaterial {\r\n this._checkUniform(name);\r\n this._vectors3Arrays[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a vec4 array in the shader from a number array.\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param value Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setArray4(name: string, value: number[]): ShaderMaterial {\r\n this._checkUniform(name);\r\n this._vectors4Arrays[name] = value;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a uniform buffer in the shader\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param buffer Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setUniformBuffer(name: string, buffer: UniformBuffer): ShaderMaterial {\r\n if (this._options.uniformBuffers.indexOf(name) === -1) {\r\n this._options.uniformBuffers.push(name);\r\n }\r\n this._uniformBuffers[name] = buffer;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a texture sampler in the shader\r\n * @param name Define the name of the uniform as defined in the shader\r\n * @param sampler Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setTextureSampler(name: string, sampler: TextureSampler): ShaderMaterial {\r\n if (this._options.samplerObjects.indexOf(name) === -1) {\r\n this._options.samplerObjects.push(name);\r\n }\r\n this._textureSamplers[name] = sampler;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set a storage buffer in the shader\r\n * @param name Define the name of the storage buffer as defined in the shader\r\n * @param buffer Define the value to give to the uniform\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setStorageBuffer(name: string, buffer: StorageBuffer): ShaderMaterial {\r\n if (this._options.storageBuffers.indexOf(name) === -1) {\r\n this._options.storageBuffers.push(name);\r\n }\r\n this._storageBuffers[name] = buffer;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds, removes, or replaces the specified shader define and value.\r\n * * setDefine(\"MY_DEFINE\", true); // enables a boolean define\r\n * * setDefine(\"MY_DEFINE\", \"0.5\"); // adds \"#define MY_DEFINE 0.5\" to the shader (or sets and replaces the value of any existing define with that name)\r\n * * setDefine(\"MY_DEFINE\", false); // disables and removes the define\r\n * Note if the active defines do change, the shader will be recompiled and this can be expensive.\r\n * @param define the define name e.g., \"OUTPUT_TO_SRGB\" or \"#define OUTPUT_TO_SRGB\". If the define was passed into the constructor already, the version used should match that, and in either case, it should not include any appended value.\r\n * @param value either the value of the define (e.g. a numerical value) or for booleans, true if the define should be enabled or false if it should be disabled\r\n * @returns the material itself allowing \"fluent\" like uniform updates\r\n */\r\n public setDefine(define: string, value: boolean | string): ShaderMaterial {\r\n // First remove any existing define with this name.\r\n const defineName = define.trimEnd() + \" \";\r\n const existingDefineIdx = this.options.defines.findIndex((x) => x === define || x.startsWith(defineName));\r\n if (existingDefineIdx >= 0) {\r\n this.options.defines.splice(existingDefineIdx, 1);\r\n }\r\n\r\n // Then add the new define value. (If it's a boolean value and false, don't add it.)\r\n if (typeof value !== \"boolean\" || value) {\r\n this.options.defines.push(defineName + value);\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Specifies that the submesh is ready to be used\r\n * @param mesh defines the mesh to check\r\n * @param subMesh defines which submesh to check\r\n * @param useInstances specifies that instances should be used\r\n * @returns a boolean indicating that the submesh is ready or not\r\n */\r\n public isReadyForSubMesh(mesh: AbstractMesh, subMesh: SubMesh, useInstances?: boolean): boolean {\r\n return this.isReady(mesh, useInstances, subMesh);\r\n }\r\n\r\n /**\r\n * Checks if the material is ready to render the requested mesh\r\n * @param mesh Define the mesh to render\r\n * @param useInstances Define whether or not the material is used with instances\r\n * @param subMesh defines which submesh to render\r\n * @returns true if ready, otherwise false\r\n */\r\n public isReady(mesh?: AbstractMesh, useInstances?: boolean, subMesh?: SubMesh): boolean {\r\n const storeEffectOnSubMeshes = subMesh && this._storeEffectOnSubMeshes;\r\n\r\n if (this.isFrozen) {\r\n if (storeEffectOnSubMeshes) {\r\n if (subMesh.effect && subMesh.effect._wasPreviouslyReady) {\r\n return true;\r\n }\r\n } else {\r\n const effect = this._drawWrapper.effect;\r\n if (effect && effect._wasPreviouslyReady && effect._wasPreviouslyUsingInstances === useInstances) {\r\n return true;\r\n }\r\n }\r\n }\r\n\r\n const scene = this.getScene();\r\n const engine = scene.getEngine();\r\n\r\n // Instances\r\n const defines = [];\r\n const attribs = [];\r\n const fallbacks = new EffectFallbacks();\r\n\r\n let shaderName = this._shaderPath,\r\n uniforms = this._options.uniforms,\r\n uniformBuffers = this._options.uniformBuffers,\r\n samplers = this._options.samplers;\r\n\r\n // global multiview\r\n if (engine.getCaps().multiview && scene.activeCamera && scene.activeCamera.outputRenderTarget && scene.activeCamera.outputRenderTarget.getViewCount() > 1) {\r\n this._multiview = true;\r\n defines.push(\"#define MULTIVIEW\");\r\n if (this._options.uniforms.indexOf(\"viewProjection\") !== -1 && this._options.uniforms.indexOf(\"viewProjectionR\") === -1) {\r\n this._options.uniforms.push(\"viewProjectionR\");\r\n }\r\n }\r\n\r\n for (let index = 0; index < this._options.defines.length; index++) {\r\n const defineToAdd = this._options.defines[index].indexOf(\"#define\") === 0 ? this._options.defines[index] : `#define ${this._options.defines[index]}`;\r\n defines.push(defineToAdd);\r\n }\r\n\r\n for (let index = 0; index < this._options.attributes.length; index++) {\r\n attribs.push(this._options.attributes[index]);\r\n }\r\n\r\n if (mesh && mesh.isVerticesDataPresent(VertexBuffer.ColorKind)) {\r\n attribs.push(VertexBuffer.ColorKind);\r\n defines.push(\"#define VERTEXCOLOR\");\r\n }\r\n\r\n if (useInstances) {\r\n defines.push(\"#define INSTANCES\");\r\n MaterialHelper.PushAttributesForInstances(attribs, this._materialHelperNeedsPreviousMatrices);\r\n if (mesh?.hasThinInstances) {\r\n defines.push(\"#define THIN_INSTANCES\");\r\n if (mesh && mesh.isVerticesDataPresent(VertexBuffer.ColorInstanceKind)) {\r\n attribs.push(VertexBuffer.ColorInstanceKind);\r\n defines.push(\"#define INSTANCESCOLOR\");\r\n }\r\n }\r\n }\r\n\r\n // Bones\r\n if (mesh && mesh.useBones && mesh.computeBonesUsingShaders && mesh.skeleton) {\r\n attribs.push(VertexBuffer.MatricesIndicesKind);\r\n attribs.push(VertexBuffer.MatricesWeightsKind);\r\n if (mesh.numBoneInfluencers > 4) {\r\n attribs.push(VertexBuffer.MatricesIndicesExtraKind);\r\n attribs.push(VertexBuffer.MatricesWeightsExtraKind);\r\n }\r\n\r\n const skeleton = mesh.skeleton;\r\n\r\n defines.push(\"#define NUM_BONE_INFLUENCERS \" + mesh.numBoneInfluencers);\r\n fallbacks.addCPUSkinningFallback(0, mesh);\r\n\r\n if (skeleton.isUsingTextureForMatrices) {\r\n defines.push(\"#define BONETEXTURE\");\r\n\r\n if (this._options.uniforms.indexOf(\"boneTextureWidth\") === -1) {\r\n this._options.uniforms.push(\"boneTextureWidth\");\r\n }\r\n\r\n if (this._options.samplers.indexOf(\"boneSampler\") === -1) {\r\n this._options.samplers.push(\"boneSampler\");\r\n }\r\n } else {\r\n defines.push(\"#define BonesPerMesh \" + (skeleton.bones.length + 1));\r\n\r\n if (this._options.uniforms.indexOf(\"mBones\") === -1) {\r\n this._options.uniforms.push(\"mBones\");\r\n }\r\n }\r\n } else {\r\n defines.push(\"#define NUM_BONE_INFLUENCERS 0\");\r\n }\r\n\r\n // Morph\r\n let numInfluencers = 0;\r\n const manager = mesh ? (mesh).morphTargetManager : null;\r\n if (manager) {\r\n const uv = manager.supportsUVs && defines.indexOf(\"#define UV1\") !== -1;\r\n const tangent = manager.supportsTangents && defines.indexOf(\"#define TANGENT\") !== -1;\r\n const normal = manager.supportsNormals && defines.indexOf(\"#define NORMAL\") !== -1;\r\n numInfluencers = manager.numInfluencers;\r\n if (uv) {\r\n defines.push(\"#define MORPHTARGETS_UV\");\r\n }\r\n if (tangent) {\r\n defines.push(\"#define MORPHTARGETS_TANGENT\");\r\n }\r\n if (normal) {\r\n defines.push(\"#define MORPHTARGETS_NORMAL\");\r\n }\r\n if (numInfluencers > 0) {\r\n defines.push(\"#define MORPHTARGETS\");\r\n }\r\n if (manager.isUsingTextureForTargets) {\r\n defines.push(\"#define MORPHTARGETS_TEXTURE\");\r\n\r\n if (this._options.uniforms.indexOf(\"morphTargetTextureIndices\") === -1) {\r\n this._options.uniforms.push(\"morphTargetTextureIndices\");\r\n }\r\n\r\n if (this._options.samplers.indexOf(\"morphTargets\") === -1) {\r\n this._options.samplers.push(\"morphTargets\");\r\n }\r\n }\r\n defines.push(\"#define NUM_MORPH_INFLUENCERS \" + numInfluencers);\r\n for (let index = 0; index < numInfluencers; index++) {\r\n attribs.push(VertexBuffer.PositionKind + index);\r\n\r\n if (normal) {\r\n attribs.push(VertexBuffer.NormalKind + index);\r\n }\r\n\r\n if (tangent) {\r\n attribs.push(VertexBuffer.TangentKind + index);\r\n }\r\n\r\n if (uv) {\r\n attribs.push(VertexBuffer.UVKind + \"_\" + index);\r\n }\r\n }\r\n if (numInfluencers > 0) {\r\n uniforms = uniforms.slice();\r\n uniforms.push(\"morphTargetInfluences\");\r\n uniforms.push(\"morphTargetTextureInfo\");\r\n uniforms.push(\"morphTargetTextureIndices\");\r\n }\r\n } else {\r\n defines.push(\"#define NUM_MORPH_INFLUENCERS 0\");\r\n }\r\n\r\n // Baked Vertex Animation\r\n if (mesh) {\r\n const bvaManager = (mesh).bakedVertexAnimationManager;\r\n\r\n if (bvaManager && bvaManager.isEnabled) {\r\n defines.push(\"#define BAKED_VERTEX_ANIMATION_TEXTURE\");\r\n if (this._options.uniforms.indexOf(\"bakedVertexAnimationSettings\") === -1) {\r\n this._options.uniforms.push(\"bakedVertexAnimationSettings\");\r\n }\r\n if (this._options.uniforms.indexOf(\"bakedVertexAnimationTextureSizeInverted\") === -1) {\r\n this._options.uniforms.push(\"bakedVertexAnimationTextureSizeInverted\");\r\n }\r\n if (this._options.uniforms.indexOf(\"bakedVertexAnimationTime\") === -1) {\r\n this._options.uniforms.push(\"bakedVertexAnimationTime\");\r\n }\r\n\r\n if (this._options.samplers.indexOf(\"bakedVertexAnimationTexture\") === -1) {\r\n this._options.samplers.push(\"bakedVertexAnimationTexture\");\r\n }\r\n }\r\n\r\n MaterialHelper.PrepareAttributesForBakedVertexAnimation(attribs, mesh, defines);\r\n }\r\n\r\n // Textures\r\n for (const name in this._textures) {\r\n if (!this._textures[name].isReady()) {\r\n return false;\r\n }\r\n }\r\n\r\n // Alpha test\r\n if (mesh && this._shouldTurnAlphaTestOn(mesh)) {\r\n defines.push(\"#define ALPHATEST\");\r\n }\r\n\r\n // Clip planes\r\n if (this._options.useClipPlane !== false) {\r\n addClipPlaneUniforms(uniforms);\r\n\r\n prepareStringDefinesForClipPlanes(this, scene, defines);\r\n }\r\n\r\n if (this.customShaderNameResolve) {\r\n uniforms = uniforms.slice();\r\n uniformBuffers = uniformBuffers.slice();\r\n samplers = samplers.slice();\r\n shaderName = this.customShaderNameResolve(shaderName, uniforms, uniformBuffers, samplers, defines, attribs);\r\n }\r\n\r\n const drawWrapper = storeEffectOnSubMeshes ? subMesh._getDrawWrapper() : this._drawWrapper;\r\n const previousEffect = drawWrapper?.effect ?? null;\r\n const previousDefines = drawWrapper?.defines ?? null;\r\n const join = defines.join(\"\\n\");\r\n\r\n let effect = previousEffect;\r\n if (previousDefines !== join) {\r\n effect = engine.createEffect(\r\n shaderName,\r\n {\r\n attributes: attribs,\r\n uniformsNames: uniforms,\r\n uniformBuffersNames: uniformBuffers,\r\n samplers: samplers,\r\n defines: join,\r\n fallbacks: fallbacks,\r\n onCompiled: this.onCompiled,\r\n onError: this.onError,\r\n indexParameters: { maxSimultaneousMorphTargets: numInfluencers },\r\n shaderLanguage: this._options.shaderLanguage,\r\n },\r\n engine\r\n );\r\n\r\n if (storeEffectOnSubMeshes) {\r\n subMesh.setEffect(effect, join, this._materialContext);\r\n } else if (drawWrapper) {\r\n drawWrapper.setEffect(effect, join);\r\n }\r\n\r\n if (this._onEffectCreatedObservable) {\r\n onCreatedEffectParameters.effect = effect;\r\n onCreatedEffectParameters.subMesh = subMesh ?? mesh?.subMeshes[0] ?? null;\r\n this._onEffectCreatedObservable.notifyObservers(onCreatedEffectParameters);\r\n }\r\n }\r\n\r\n effect!._wasPreviouslyUsingInstances = !!useInstances;\r\n\r\n if (!effect?.isReady() ?? true) {\r\n return false;\r\n }\r\n\r\n if (previousEffect !== effect) {\r\n scene.resetCachedMaterial();\r\n }\r\n\r\n effect._wasPreviouslyReady = true;\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Binds the world matrix to the material\r\n * @param world defines the world transformation matrix\r\n * @param effectOverride - If provided, use this effect instead of internal effect\r\n */\r\n public bindOnlyWorldMatrix(world: Matrix, effectOverride?: Nullable): void {\r\n const scene = this.getScene();\r\n\r\n const effect = effectOverride ?? this.getEffect();\r\n\r\n if (!effect) {\r\n return;\r\n }\r\n\r\n if (this._options.uniforms.indexOf(\"world\") !== -1) {\r\n effect.setMatrix(\"world\", world);\r\n }\r\n\r\n if (this._options.uniforms.indexOf(\"worldView\") !== -1) {\r\n world.multiplyToRef(scene.getViewMatrix(), this._cachedWorldViewMatrix);\r\n effect.setMatrix(\"worldView\", this._cachedWorldViewMatrix);\r\n }\r\n\r\n if (this._options.uniforms.indexOf(\"worldViewProjection\") !== -1) {\r\n world.multiplyToRef(scene.getTransformMatrix(), this._cachedWorldViewProjectionMatrix);\r\n effect.setMatrix(\"worldViewProjection\", this._cachedWorldViewProjectionMatrix);\r\n }\r\n }\r\n\r\n /**\r\n * Binds the submesh to this material by preparing the effect and shader to draw\r\n * @param world defines the world transformation matrix\r\n * @param mesh defines the mesh containing the submesh\r\n * @param subMesh defines the submesh to bind the material to\r\n */\r\n public bindForSubMesh(world: Matrix, mesh: Mesh, subMesh: SubMesh): void {\r\n this.bind(world, mesh, subMesh._drawWrapperOverride?.effect, subMesh);\r\n }\r\n\r\n /**\r\n * Binds the material to the mesh\r\n * @param world defines the world transformation matrix\r\n * @param mesh defines the mesh to bind the material to\r\n * @param effectOverride - If provided, use this effect instead of internal effect\r\n * @param subMesh defines the submesh to bind the material to\r\n */\r\n public bind(world: Matrix, mesh?: Mesh, effectOverride?: Nullable, subMesh?: SubMesh): void {\r\n // Std values\r\n const storeEffectOnSubMeshes = subMesh && this._storeEffectOnSubMeshes;\r\n const effect = effectOverride ?? (storeEffectOnSubMeshes ? subMesh.effect : this.getEffect());\r\n\r\n if (!effect) {\r\n return;\r\n }\r\n\r\n this._activeEffect = effect;\r\n\r\n this.bindOnlyWorldMatrix(world, effectOverride);\r\n\r\n const uniformBuffers = this._options.uniformBuffers;\r\n\r\n let useSceneUBO = false;\r\n\r\n if (effect && uniformBuffers && uniformBuffers.length > 0 && this.getScene().getEngine().supportsUniformBuffers) {\r\n for (let i = 0; i < uniformBuffers.length; ++i) {\r\n const bufferName = uniformBuffers[i];\r\n switch (bufferName) {\r\n case \"Mesh\":\r\n if (mesh) {\r\n mesh.getMeshUniformBuffer().bindToEffect(effect, \"Mesh\");\r\n mesh.transferToEffect(world);\r\n }\r\n break;\r\n case \"Scene\":\r\n MaterialHelper.BindSceneUniformBuffer(effect, this.getScene().getSceneUniformBuffer());\r\n this.getScene().finalizeSceneUbo();\r\n useSceneUBO = true;\r\n break;\r\n }\r\n }\r\n }\r\n\r\n const mustRebind = mesh && storeEffectOnSubMeshes ? this._mustRebind(this.getScene(), effect, mesh.visibility) : this.getScene().getCachedMaterial() !== this;\r\n\r\n if (effect && mustRebind) {\r\n if (!useSceneUBO && this._options.uniforms.indexOf(\"view\") !== -1) {\r\n effect.setMatrix(\"view\", this.getScene().getViewMatrix());\r\n }\r\n\r\n if (!useSceneUBO && this._options.uniforms.indexOf(\"projection\") !== -1) {\r\n effect.setMatrix(\"projection\", this.getScene().getProjectionMatrix());\r\n }\r\n\r\n if (!useSceneUBO && this._options.uniforms.indexOf(\"viewProjection\") !== -1) {\r\n effect.setMatrix(\"viewProjection\", this.getScene().getTransformMatrix());\r\n if (this._multiview) {\r\n effect.setMatrix(\"viewProjectionR\", this.getScene()._transformMatrixR);\r\n }\r\n }\r\n\r\n if (this.getScene().activeCamera && this._options.uniforms.indexOf(\"cameraPosition\") !== -1) {\r\n effect.setVector3(\"cameraPosition\", this.getScene().activeCamera!.globalPosition);\r\n }\r\n\r\n // Bones\r\n MaterialHelper.BindBonesParameters(mesh, effect);\r\n\r\n // Clip plane\r\n bindClipPlane(effect, this, this.getScene());\r\n\r\n let name: string;\r\n // Texture\r\n for (name in this._textures) {\r\n effect.setTexture(name, this._textures[name]);\r\n }\r\n\r\n // Texture arrays\r\n for (name in this._textureArrays) {\r\n effect.setTextureArray(name, this._textureArrays[name]);\r\n }\r\n\r\n // External texture\r\n for (name in this._externalTextures) {\r\n effect.setExternalTexture(name, this._externalTextures[name]);\r\n }\r\n\r\n // Int\r\n for (name in this._ints) {\r\n effect.setInt(name, this._ints[name]);\r\n }\r\n\r\n // UInt\r\n for (name in this._uints) {\r\n effect.setUInt(name, this._uints[name]);\r\n }\r\n\r\n // Float\r\n for (name in this._floats) {\r\n effect.setFloat(name, this._floats[name]);\r\n }\r\n\r\n // Floats\r\n for (name in this._floatsArrays) {\r\n effect.setArray(name, this._floatsArrays[name]);\r\n }\r\n\r\n // Color3\r\n for (name in this._colors3) {\r\n effect.setColor3(name, this._colors3[name]);\r\n }\r\n\r\n // Color3Array\r\n for (name in this._colors3Arrays) {\r\n effect.setArray3(name, this._colors3Arrays[name]);\r\n }\r\n\r\n // Color4\r\n for (name in this._colors4) {\r\n const color = this._colors4[name];\r\n effect.setFloat4(name, color.r, color.g, color.b, color.a);\r\n }\r\n\r\n // Color4Array\r\n for (name in this._colors4Arrays) {\r\n effect.setArray4(name, this._colors4Arrays[name]);\r\n }\r\n\r\n // Vector2\r\n for (name in this._vectors2) {\r\n effect.setVector2(name, this._vectors2[name]);\r\n }\r\n\r\n // Vector3\r\n for (name in this._vectors3) {\r\n effect.setVector3(name, this._vectors3[name]);\r\n }\r\n\r\n // Vector4\r\n for (name in this._vectors4) {\r\n effect.setVector4(name, this._vectors4[name]);\r\n }\r\n\r\n // Quaternion\r\n for (name in this._quaternions) {\r\n effect.setQuaternion(name, this._quaternions[name]);\r\n }\r\n\r\n // Matrix\r\n for (name in this._matrices) {\r\n effect.setMatrix(name, this._matrices[name]);\r\n }\r\n\r\n // MatrixArray\r\n for (name in this._matrixArrays) {\r\n effect.setMatrices(name, this._matrixArrays[name]);\r\n }\r\n\r\n // Matrix 3x3\r\n for (name in this._matrices3x3) {\r\n effect.setMatrix3x3(name, this._matrices3x3[name]);\r\n }\r\n\r\n // Matrix 2x2\r\n for (name in this._matrices2x2) {\r\n effect.setMatrix2x2(name, this._matrices2x2[name]);\r\n }\r\n\r\n // Vector2Array\r\n for (name in this._vectors2Arrays) {\r\n effect.setArray2(name, this._vectors2Arrays[name]);\r\n }\r\n\r\n // Vector3Array\r\n for (name in this._vectors3Arrays) {\r\n effect.setArray3(name, this._vectors3Arrays[name]);\r\n }\r\n\r\n // Vector4Array\r\n for (name in this._vectors4Arrays) {\r\n effect.setArray4(name, this._vectors4Arrays[name]);\r\n }\r\n\r\n // QuaternionArray\r\n for (name in this._quaternionsArrays) {\r\n effect.setArray4(name, this._quaternionsArrays[name]);\r\n }\r\n\r\n // Uniform buffers\r\n for (name in this._uniformBuffers) {\r\n const buffer = this._uniformBuffers[name].getBuffer();\r\n if (buffer) {\r\n effect.bindUniformBuffer(buffer, name);\r\n }\r\n }\r\n\r\n // Samplers\r\n for (name in this._textureSamplers) {\r\n effect.setTextureSampler(name, this._textureSamplers[name]);\r\n }\r\n\r\n // Storage buffers\r\n for (name in this._storageBuffers) {\r\n effect.setStorageBuffer(name, this._storageBuffers[name]);\r\n }\r\n }\r\n\r\n if (effect && mesh && (mustRebind || !this.isFrozen)) {\r\n // Morph targets\r\n const manager = (mesh).morphTargetManager;\r\n if (manager && manager.numInfluencers > 0) {\r\n MaterialHelper.BindMorphTargetParameters(mesh, effect);\r\n }\r\n\r\n const bvaManager = (mesh).bakedVertexAnimationManager;\r\n\r\n if (bvaManager && bvaManager.isEnabled) {\r\n mesh.bakedVertexAnimationManager?.bind(effect, !!effect._wasPreviouslyUsingInstances);\r\n }\r\n }\r\n\r\n this._afterBind(mesh, effect);\r\n }\r\n\r\n /**\r\n * Gets the active textures from the material\r\n * @returns an array of textures\r\n */\r\n public getActiveTextures(): BaseTexture[] {\r\n const activeTextures = super.getActiveTextures();\r\n\r\n for (const name in this._textures) {\r\n activeTextures.push(this._textures[name]);\r\n }\r\n\r\n for (const name in this._textureArrays) {\r\n const array = this._textureArrays[name];\r\n for (let index = 0; index < array.length; index++) {\r\n activeTextures.push(array[index]);\r\n }\r\n }\r\n\r\n return activeTextures;\r\n }\r\n\r\n /**\r\n * Specifies if the material uses a texture\r\n * @param texture defines the texture to check against the material\r\n * @returns a boolean specifying if the material uses the texture\r\n */\r\n public hasTexture(texture: BaseTexture): boolean {\r\n if (super.hasTexture(texture)) {\r\n return true;\r\n }\r\n\r\n for (const name in this._textures) {\r\n if (this._textures[name] === texture) {\r\n return true;\r\n }\r\n }\r\n\r\n for (const name in this._textureArrays) {\r\n const array = this._textureArrays[name];\r\n for (let index = 0; index < array.length; index++) {\r\n if (array[index] === texture) {\r\n return true;\r\n }\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Makes a duplicate of the material, and gives it a new name\r\n * @param name defines the new name for the duplicated material\r\n * @returns the cloned material\r\n */\r\n public clone(name: string): ShaderMaterial {\r\n const result = SerializationHelper.Clone(() => new ShaderMaterial(name, this.getScene(), this._shaderPath, this._options, this._storeEffectOnSubMeshes), this);\r\n\r\n result.name = name;\r\n result.id = name;\r\n\r\n // Shader code path\r\n if (typeof result._shaderPath === \"object\") {\r\n result._shaderPath = { ...result._shaderPath };\r\n }\r\n\r\n // Options\r\n this._options = { ...this._options };\r\n\r\n (Object.keys(this._options) as Array).forEach((propName) => {\r\n const propValue = this._options[propName];\r\n if (Array.isArray(propValue)) {\r\n (this._options[propName]) = propValue.slice(0);\r\n }\r\n });\r\n\r\n // Stencil\r\n this.stencil.copyTo(result.stencil);\r\n\r\n // Texture\r\n for (const key in this._textures) {\r\n result.setTexture(key, this._textures[key]);\r\n }\r\n\r\n // TextureArray\r\n for (const key in this._textureArrays) {\r\n result.setTextureArray(key, this._textureArrays[key]);\r\n }\r\n\r\n // External texture\r\n for (const key in this._externalTextures) {\r\n result.setExternalTexture(key, this._externalTextures[key]);\r\n }\r\n\r\n // Int\r\n for (const key in this._ints) {\r\n result.setInt(key, this._ints[key]);\r\n }\r\n\r\n // UInt\r\n for (const key in this._uints) {\r\n result.setUInt(key, this._uints[key]);\r\n }\r\n\r\n // Float\r\n for (const key in this._floats) {\r\n result.setFloat(key, this._floats[key]);\r\n }\r\n\r\n // Floats\r\n for (const key in this._floatsArrays) {\r\n result.setFloats(key, this._floatsArrays[key]);\r\n }\r\n\r\n // Color3\r\n for (const key in this._colors3) {\r\n result.setColor3(key, this._colors3[key]);\r\n }\r\n\r\n // Color3Array\r\n for (const key in this._colors3Arrays) {\r\n result._colors3Arrays[key] = this._colors3Arrays[key];\r\n }\r\n\r\n // Color4\r\n for (const key in this._colors4) {\r\n result.setColor4(key, this._colors4[key]);\r\n }\r\n\r\n // Color4Array\r\n for (const key in this._colors4Arrays) {\r\n result._colors4Arrays[key] = this._colors4Arrays[key];\r\n }\r\n\r\n // Vector2\r\n for (const key in this._vectors2) {\r\n result.setVector2(key, this._vectors2[key]);\r\n }\r\n\r\n // Vector3\r\n for (const key in this._vectors3) {\r\n result.setVector3(key, this._vectors3[key]);\r\n }\r\n\r\n // Vector4\r\n for (const key in this._vectors4) {\r\n result.setVector4(key, this._vectors4[key]);\r\n }\r\n\r\n // Quaternion\r\n for (const key in this._quaternions) {\r\n result.setQuaternion(key, this._quaternions[key]);\r\n }\r\n\r\n // QuaternionArray\r\n for (const key in this._quaternionsArrays) {\r\n result._quaternionsArrays[key] = this._quaternionsArrays[key];\r\n }\r\n\r\n // Matrix\r\n for (const key in this._matrices) {\r\n result.setMatrix(key, this._matrices[key]);\r\n }\r\n\r\n // MatrixArray\r\n for (const key in this._matrixArrays) {\r\n result._matrixArrays[key] = this._matrixArrays[key].slice();\r\n }\r\n\r\n // Matrix 3x3\r\n for (const key in this._matrices3x3) {\r\n result.setMatrix3x3(key, this._matrices3x3[key]);\r\n }\r\n\r\n // Matrix 2x2\r\n for (const key in this._matrices2x2) {\r\n result.setMatrix2x2(key, this._matrices2x2[key]);\r\n }\r\n\r\n // Vector2Array\r\n for (const key in this._vectors2Arrays) {\r\n result.setArray2(key, this._vectors2Arrays[key]);\r\n }\r\n\r\n // Vector3Array\r\n for (const key in this._vectors3Arrays) {\r\n result.setArray3(key, this._vectors3Arrays[key]);\r\n }\r\n\r\n // Vector4Array\r\n for (const key in this._vectors4Arrays) {\r\n result.setArray4(key, this._vectors4Arrays[key]);\r\n }\r\n\r\n // Uniform buffers\r\n for (const key in this._uniformBuffers) {\r\n result.setUniformBuffer(key, this._uniformBuffers[key]);\r\n }\r\n\r\n // Samplers\r\n for (const key in this._textureSamplers) {\r\n result.setTextureSampler(key, this._textureSamplers[key]);\r\n }\r\n\r\n // Storag buffers\r\n for (const key in this._storageBuffers) {\r\n result.setStorageBuffer(key, this._storageBuffers[key]);\r\n }\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Disposes the material\r\n * @param forceDisposeEffect specifies if effects should be forcefully disposed\r\n * @param forceDisposeTextures specifies if textures should be forcefully disposed\r\n * @param notBoundToMesh specifies if the material that is being disposed is known to be not bound to any mesh\r\n */\r\n public dispose(forceDisposeEffect?: boolean, forceDisposeTextures?: boolean, notBoundToMesh?: boolean): void {\r\n if (forceDisposeTextures) {\r\n let name: string;\r\n for (name in this._textures) {\r\n this._textures[name].dispose();\r\n }\r\n\r\n for (name in this._textureArrays) {\r\n const array = this._textureArrays[name];\r\n for (let index = 0; index < array.length; index++) {\r\n array[index].dispose();\r\n }\r\n }\r\n }\r\n\r\n this._textures = {};\r\n\r\n super.dispose(forceDisposeEffect, forceDisposeTextures, notBoundToMesh);\r\n }\r\n\r\n /**\r\n * Serializes this material in a JSON representation\r\n * @returns the serialized material object\r\n */\r\n public serialize(): any {\r\n const serializationObject = SerializationHelper.Serialize(this);\r\n serializationObject.customType = \"BABYLON.ShaderMaterial\";\r\n serializationObject.uniqueId = this.uniqueId;\r\n\r\n serializationObject.options = this._options;\r\n serializationObject.shaderPath = this._shaderPath;\r\n serializationObject.storeEffectOnSubMeshes = this._storeEffectOnSubMeshes;\r\n\r\n let name: string;\r\n\r\n // Stencil\r\n serializationObject.stencil = this.stencil.serialize();\r\n\r\n // Texture\r\n serializationObject.textures = {};\r\n for (name in this._textures) {\r\n serializationObject.textures[name] = this._textures[name].serialize();\r\n }\r\n\r\n // Texture arrays\r\n serializationObject.textureArrays = {};\r\n for (name in this._textureArrays) {\r\n serializationObject.textureArrays[name] = [];\r\n const array = this._textureArrays[name];\r\n for (let index = 0; index < array.length; index++) {\r\n serializationObject.textureArrays[name].push(array[index].serialize());\r\n }\r\n }\r\n\r\n // Int\r\n serializationObject.ints = {};\r\n for (name in this._ints) {\r\n serializationObject.ints[name] = this._ints[name];\r\n }\r\n\r\n // UInt\r\n serializationObject.uints = {};\r\n for (name in this._uints) {\r\n serializationObject.uints[name] = this._uints[name];\r\n }\r\n\r\n // Float\r\n serializationObject.floats = {};\r\n for (name in this._floats) {\r\n serializationObject.floats[name] = this._floats[name];\r\n }\r\n\r\n // Floats\r\n serializationObject.FloatArrays = {};\r\n for (name in this._floatsArrays) {\r\n serializationObject.FloatArrays[name] = this._floatsArrays[name];\r\n }\r\n\r\n // Color3\r\n serializationObject.colors3 = {};\r\n for (name in this._colors3) {\r\n serializationObject.colors3[name] = this._colors3[name].asArray();\r\n }\r\n\r\n // Color3 array\r\n serializationObject.colors3Arrays = {};\r\n for (name in this._colors3Arrays) {\r\n serializationObject.colors3Arrays[name] = this._colors3Arrays[name];\r\n }\r\n\r\n // Color4\r\n serializationObject.colors4 = {};\r\n for (name in this._colors4) {\r\n serializationObject.colors4[name] = this._colors4[name].asArray();\r\n }\r\n\r\n // Color4 array\r\n serializationObject.colors4Arrays = {};\r\n for (name in this._colors4Arrays) {\r\n serializationObject.colors4Arrays[name] = this._colors4Arrays[name];\r\n }\r\n\r\n // Vector2\r\n serializationObject.vectors2 = {};\r\n for (name in this._vectors2) {\r\n serializationObject.vectors2[name] = this._vectors2[name].asArray();\r\n }\r\n\r\n // Vector3\r\n serializationObject.vectors3 = {};\r\n for (name in this._vectors3) {\r\n serializationObject.vectors3[name] = this._vectors3[name].asArray();\r\n }\r\n\r\n // Vector4\r\n serializationObject.vectors4 = {};\r\n for (name in this._vectors4) {\r\n serializationObject.vectors4[name] = this._vectors4[name].asArray();\r\n }\r\n\r\n // Quaternion\r\n serializationObject.quaternions = {};\r\n for (name in this._quaternions) {\r\n serializationObject.quaternions[name] = this._quaternions[name].asArray();\r\n }\r\n\r\n // Matrix\r\n serializationObject.matrices = {};\r\n for (name in this._matrices) {\r\n serializationObject.matrices[name] = this._matrices[name].asArray();\r\n }\r\n\r\n // MatrixArray\r\n serializationObject.matrixArray = {};\r\n for (name in this._matrixArrays) {\r\n serializationObject.matrixArray[name] = this._matrixArrays[name];\r\n }\r\n\r\n // Matrix 3x3\r\n serializationObject.matrices3x3 = {};\r\n for (name in this._matrices3x3) {\r\n serializationObject.matrices3x3[name] = this._matrices3x3[name];\r\n }\r\n\r\n // Matrix 2x2\r\n serializationObject.matrices2x2 = {};\r\n for (name in this._matrices2x2) {\r\n serializationObject.matrices2x2[name] = this._matrices2x2[name];\r\n }\r\n\r\n // Vector2Array\r\n serializationObject.vectors2Arrays = {};\r\n for (name in this._vectors2Arrays) {\r\n serializationObject.vectors2Arrays[name] = this._vectors2Arrays[name];\r\n }\r\n\r\n // Vector3Array\r\n serializationObject.vectors3Arrays = {};\r\n for (name in this._vectors3Arrays) {\r\n serializationObject.vectors3Arrays[name] = this._vectors3Arrays[name];\r\n }\r\n\r\n // Vector4Array\r\n serializationObject.vectors4Arrays = {};\r\n for (name in this._vectors4Arrays) {\r\n serializationObject.vectors4Arrays[name] = this._vectors4Arrays[name];\r\n }\r\n\r\n // QuaternionArray\r\n serializationObject.quaternionsArrays = {};\r\n for (name in this._quaternionsArrays) {\r\n serializationObject.quaternionsArrays[name] = this._quaternionsArrays[name];\r\n }\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Creates a shader material from parsed shader material data\r\n * @param source defines the JSON representation of the material\r\n * @param scene defines the hosting scene\r\n * @param rootUrl defines the root URL to use to load textures and relative dependencies\r\n * @returns a new material\r\n */\r\n public static Parse(source: any, scene: Scene, rootUrl: string): ShaderMaterial {\r\n const material = SerializationHelper.Parse(\r\n () => new ShaderMaterial(source.name, scene, source.shaderPath, source.options, source.storeEffectOnSubMeshes),\r\n source,\r\n scene,\r\n rootUrl\r\n );\r\n\r\n let name: string;\r\n\r\n // Stencil\r\n if (source.stencil) {\r\n material.stencil.parse(source.stencil, scene, rootUrl);\r\n }\r\n\r\n // Texture\r\n for (name in source.textures) {\r\n material.setTexture(name, Texture.Parse(source.textures[name], scene, rootUrl));\r\n }\r\n\r\n // Texture arrays\r\n for (name in source.textureArrays) {\r\n const array = source.textureArrays[name];\r\n const textureArray = new Array();\r\n\r\n for (let index = 0; index < array.length; index++) {\r\n textureArray.push(Texture.Parse(array[index], scene, rootUrl));\r\n }\r\n material.setTextureArray(name, textureArray);\r\n }\r\n\r\n // Int\r\n for (name in source.ints) {\r\n material.setInt(name, source.ints[name]);\r\n }\r\n\r\n // UInt\r\n for (name in source.uints) {\r\n material.setUInt(name, source.uints[name]);\r\n }\r\n\r\n // Float\r\n for (name in source.floats) {\r\n material.setFloat(name, source.floats[name]);\r\n }\r\n\r\n // Floats\r\n for (name in source.floatsArrays) {\r\n material.setFloats(name, source.floatsArrays[name]);\r\n }\r\n\r\n // Color3\r\n for (name in source.colors3) {\r\n material.setColor3(name, Color3.FromArray(source.colors3[name]));\r\n }\r\n\r\n // Color3 arrays\r\n for (name in source.colors3Arrays) {\r\n const colors: Color3[] = source.colors3Arrays[name]\r\n .reduce((arr: Array>, num: number, i: number) => {\r\n if (i % 3 === 0) {\r\n arr.push([num]);\r\n } else {\r\n arr[arr.length - 1].push(num);\r\n }\r\n return arr;\r\n }, [])\r\n .map((color: ArrayLike) => Color3.FromArray(color));\r\n material.setColor3Array(name, colors);\r\n }\r\n\r\n // Color4\r\n for (name in source.colors4) {\r\n material.setColor4(name, Color4.FromArray(source.colors4[name]));\r\n }\r\n\r\n // Color4 arrays\r\n for (name in source.colors4Arrays) {\r\n const colors: Color4[] = source.colors4Arrays[name]\r\n .reduce((arr: Array>, num: number, i: number) => {\r\n if (i % 4 === 0) {\r\n arr.push([num]);\r\n } else {\r\n arr[arr.length - 1].push(num);\r\n }\r\n return arr;\r\n }, [])\r\n .map((color: ArrayLike) => Color4.FromArray(color));\r\n material.setColor4Array(name, colors);\r\n }\r\n\r\n // Vector2\r\n for (name in source.vectors2) {\r\n material.setVector2(name, Vector2.FromArray(source.vectors2[name]));\r\n }\r\n\r\n // Vector3\r\n for (name in source.vectors3) {\r\n material.setVector3(name, Vector3.FromArray(source.vectors3[name]));\r\n }\r\n\r\n // Vector4\r\n for (name in source.vectors4) {\r\n material.setVector4(name, Vector4.FromArray(source.vectors4[name]));\r\n }\r\n\r\n // Quaternion\r\n for (name in source.quaternions) {\r\n material.setQuaternion(name, Quaternion.FromArray(source.quaternions[name]));\r\n }\r\n\r\n // Matrix\r\n for (name in source.matrices) {\r\n material.setMatrix(name, Matrix.FromArray(source.matrices[name]));\r\n }\r\n\r\n // MatrixArray\r\n for (name in source.matrixArray) {\r\n material._matrixArrays[name] = new Float32Array(source.matrixArray[name]);\r\n }\r\n\r\n // Matrix 3x3\r\n for (name in source.matrices3x3) {\r\n material.setMatrix3x3(name, source.matrices3x3[name]);\r\n }\r\n\r\n // Matrix 2x2\r\n for (name in source.matrices2x2) {\r\n material.setMatrix2x2(name, source.matrices2x2[name]);\r\n }\r\n\r\n // Vector2Array\r\n for (name in source.vectors2Arrays) {\r\n material.setArray2(name, source.vectors2Arrays[name]);\r\n }\r\n\r\n // Vector3Array\r\n for (name in source.vectors3Arrays) {\r\n material.setArray3(name, source.vectors3Arrays[name]);\r\n }\r\n\r\n // Vector4Array\r\n for (name in source.vectors4Arrays) {\r\n material.setArray4(name, source.vectors4Arrays[name]);\r\n }\r\n\r\n // QuaternionArray\r\n for (name in source.quaternionsArrays) {\r\n material.setArray4(name, source.quaternionsArrays[name]);\r\n }\r\n\r\n return material;\r\n }\r\n\r\n /**\r\n * Creates a new ShaderMaterial from a snippet saved in a remote file\r\n * @param name defines the name of the ShaderMaterial to create (can be null or empty to use the one from the json data)\r\n * @param url defines the url to load from\r\n * @param scene defines the hosting scene\r\n * @param rootUrl defines the root URL to use to load textures and relative dependencies\r\n * @returns a promise that will resolve to the new ShaderMaterial\r\n */\r\n public static ParseFromFileAsync(name: Nullable, url: string, scene: Scene, rootUrl = \"\"): Promise {\r\n return new Promise((resolve, reject) => {\r\n const request = new WebRequest();\r\n request.addEventListener(\"readystatechange\", () => {\r\n if (request.readyState == 4) {\r\n if (request.status == 200) {\r\n const serializationObject = JSON.parse(request.responseText);\r\n const output = this.Parse(serializationObject, scene || EngineStore.LastCreatedScene, rootUrl);\r\n\r\n if (name) {\r\n output.name = name;\r\n }\r\n\r\n resolve(output);\r\n } else {\r\n reject(\"Unable to load the ShaderMaterial\");\r\n }\r\n }\r\n });\r\n\r\n request.open(\"GET\", url);\r\n request.send();\r\n });\r\n }\r\n\r\n /**\r\n * Creates a ShaderMaterial from a snippet saved by the Inspector\r\n * @param snippetId defines the snippet to load\r\n * @param scene defines the hosting scene\r\n * @param rootUrl defines the root URL to use to load textures and relative dependencies\r\n * @returns a promise that will resolve to the new ShaderMaterial\r\n */\r\n public static ParseFromSnippetAsync(snippetId: string, scene: Scene, rootUrl = \"\"): Promise {\r\n return new Promise((resolve, reject) => {\r\n const request = new WebRequest();\r\n request.addEventListener(\"readystatechange\", () => {\r\n if (request.readyState == 4) {\r\n if (request.status == 200) {\r\n const snippet = JSON.parse(JSON.parse(request.responseText).jsonPayload);\r\n const serializationObject = JSON.parse(snippet.shaderMaterial);\r\n const output = this.Parse(serializationObject, scene || EngineStore.LastCreatedScene, rootUrl);\r\n\r\n output.snippetId = snippetId;\r\n\r\n resolve(output);\r\n } else {\r\n reject(\"Unable to load the snippet \" + snippetId);\r\n }\r\n }\r\n });\r\n\r\n request.open(\"GET\", this.SnippetUrl + \"/\" + snippetId.replace(/#/g, \"/\"));\r\n request.send();\r\n });\r\n }\r\n\r\n /**\r\n * Creates a ShaderMaterial from a snippet saved by the Inspector\r\n * @deprecated Please use ParseFromSnippetAsync instead\r\n * @param snippetId defines the snippet to load\r\n * @param scene defines the hosting scene\r\n * @param rootUrl defines the root URL to use to load textures and relative dependencies\r\n * @returns a promise that will resolve to the new ShaderMaterial\r\n */\r\n public static CreateFromSnippetAsync = ShaderMaterial.ParseFromSnippetAsync;\r\n}\r\n\r\nRegisterClass(\"BABYLON.ShaderMaterial\", ShaderMaterial);\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/clipPlaneFragmentDeclaration\";\nimport \"./ShadersInclude/clipPlaneFragment\";\n\nconst name = \"colorPixelShader\";\nconst shader = `#if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES)\n#define VERTEXCOLOR\nvarying vec4 vColor;\n#else\nuniform vec4 color;\n#endif\n#include\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_FRAGMENT_MAIN_BEGIN\n#include\n#if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES)\ngl_FragColor=vColor;\n#else\ngl_FragColor=color;\n#endif\n#define CUSTOM_FRAGMENT_MAIN_END\n}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const colorPixelShader = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/bonesDeclaration\";\nimport \"./ShadersInclude/bakedVertexAnimationDeclaration\";\nimport \"./ShadersInclude/clipPlaneVertexDeclaration\";\nimport \"./ShadersInclude/instancesDeclaration\";\nimport \"./ShadersInclude/instancesVertex\";\nimport \"./ShadersInclude/bonesVertex\";\nimport \"./ShadersInclude/bakedVertexAnimation\";\nimport \"./ShadersInclude/clipPlaneVertex\";\nimport \"./ShadersInclude/vertexColorMixing\";\n\nconst name = \"colorVertexShader\";\nconst shader = `attribute vec3 position;\n#ifdef VERTEXCOLOR\nattribute vec4 color;\n#endif\n#include\n#include\n#include\n#include\nuniform mat4 viewProjection;\n#ifdef MULTIVIEW\nuniform mat4 viewProjectionR;\n#endif\n#if defined(VERTEXCOLOR) || defined(INSTANCESCOLOR) && defined(INSTANCES)\nvarying vec4 vColor;\n#endif\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\n#include\n#include\n#include\nvec4 worldPos=finalWorld*vec4(position,1.0);\n#ifdef MULTIVIEW\nif (gl_ViewID_OVR==0u) {gl_Position=viewProjection*worldPos;} else {gl_Position=viewProjectionR*worldPos;}\n#else\ngl_Position=viewProjection*worldPos;\n#endif\n#include\n#include\n#define CUSTOM_VERTEX_MAIN_END\n}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const colorVertexShader = { name, shader };\n","import type { Nullable } from \"../types\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Color3, Color4 } from \"../Maths/math.color\";\r\nimport type { Node } from \"../node\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport type { SubMesh } from \"../Meshes/subMesh\";\r\nimport { Mesh } from \"../Meshes/mesh\";\r\nimport { InstancedMesh } from \"../Meshes/instancedMesh\";\r\nimport { Material } from \"../Materials/material\";\r\nimport { ShaderMaterial } from \"../Materials/shaderMaterial\";\r\nimport type { Effect } from \"../Materials/effect\";\r\n\r\nimport \"../Shaders/color.fragment\";\r\nimport \"../Shaders/color.vertex\";\r\n\r\nMesh._LinesMeshParser = (parsedMesh: any, scene: Scene): Mesh => {\r\n return LinesMesh.Parse(parsedMesh, scene);\r\n};\r\n\r\n/**\r\n * Line mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param\r\n */\r\nexport class LinesMesh extends Mesh {\r\n /**\r\n * Color of the line (Default: White)\r\n */\r\n public color = new Color3(1, 1, 1);\r\n\r\n /**\r\n * Alpha of the line (Default: 1)\r\n */\r\n public alpha = 1;\r\n\r\n /**\r\n * The intersection Threshold is the margin applied when intersection a segment of the LinesMesh with a Ray.\r\n * This margin is expressed in world space coordinates, so its value may vary.\r\n * Default value is 0.1\r\n */\r\n public intersectionThreshold: number;\r\n\r\n private _lineMaterial: Material;\r\n\r\n private _isShaderMaterial(shader: Material): shader is ShaderMaterial {\r\n return shader.getClassName() === \"ShaderMaterial\";\r\n }\r\n\r\n private _color4: Color4;\r\n\r\n /**\r\n * Creates a new LinesMesh\r\n * @param name defines the name\r\n * @param scene defines the hosting scene\r\n * @param parent defines the parent mesh if any\r\n * @param source defines the optional source LinesMesh used to clone data from\r\n * @param doNotCloneChildren When cloning, skip cloning child meshes of source, default False.\r\n * When false, achieved by calling a clone(), also passing False.\r\n * This will make creation of children, recursive.\r\n * @param useVertexColor defines if this LinesMesh supports vertex color\r\n * @param useVertexAlpha defines if this LinesMesh supports vertex alpha\r\n * @param material material to use to draw the line. If not provided, will create a new one\r\n */\r\n constructor(\r\n name: string,\r\n scene: Nullable = null,\r\n parent: Nullable = null,\r\n source: Nullable = null,\r\n doNotCloneChildren?: boolean,\r\n /**\r\n * If vertex color should be applied to the mesh\r\n */\r\n public readonly useVertexColor?: boolean,\r\n /**\r\n * If vertex alpha should be applied to the mesh\r\n */\r\n public readonly useVertexAlpha?: boolean,\r\n material?: Material\r\n ) {\r\n super(name, scene, parent, source, doNotCloneChildren);\r\n\r\n if (source) {\r\n this.color = source.color.clone();\r\n this.alpha = source.alpha;\r\n this.useVertexColor = source.useVertexColor;\r\n this.useVertexAlpha = source.useVertexAlpha;\r\n }\r\n\r\n this.intersectionThreshold = 0.1;\r\n\r\n const defines: string[] = [];\r\n const options = {\r\n attributes: [VertexBuffer.PositionKind],\r\n uniforms: [\"world\", \"viewProjection\"],\r\n needAlphaBlending: true,\r\n defines: defines,\r\n useClipPlane: null,\r\n };\r\n\r\n if (useVertexAlpha === false) {\r\n options.needAlphaBlending = false;\r\n } else {\r\n options.defines.push(\"#define VERTEXALPHA\");\r\n }\r\n\r\n if (!useVertexColor) {\r\n options.uniforms.push(\"color\");\r\n this._color4 = new Color4();\r\n } else {\r\n options.defines.push(\"#define VERTEXCOLOR\");\r\n options.attributes.push(VertexBuffer.ColorKind);\r\n }\r\n\r\n if (material) {\r\n this.material = material;\r\n } else {\r\n this.material = new ShaderMaterial(\"colorShader\", this.getScene(), \"color\", options, false);\r\n this.material.doNotSerialize = true;\r\n }\r\n }\r\n\r\n public isReady() {\r\n if (!this._lineMaterial.isReady(this, !!this._userInstancedBuffersStorage)) {\r\n return false;\r\n }\r\n\r\n return super.isReady();\r\n }\r\n\r\n /**\r\n * Returns the string \"LineMesh\"\r\n */\r\n public getClassName(): string {\r\n return \"LinesMesh\";\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public get material(): Material {\r\n return this._lineMaterial;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public set material(value: Material) {\r\n this._lineMaterial = value;\r\n this._lineMaterial.fillMode = Material.LineListDrawMode;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public get checkCollisions(): boolean {\r\n return false;\r\n }\r\n\r\n public set checkCollisions(value: boolean) {\r\n // Just ignore it\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _bind(_subMesh: SubMesh, colorEffect: Effect): Mesh {\r\n if (!this._geometry) {\r\n return this;\r\n }\r\n\r\n // VBOs\r\n const indexToBind = this.isUnIndexed ? null : this._geometry.getIndexBuffer();\r\n if (!this._userInstancedBuffersStorage) {\r\n this._geometry._bind(colorEffect, indexToBind);\r\n } else {\r\n this._geometry._bind(colorEffect, indexToBind, this._userInstancedBuffersStorage.vertexBuffers, this._userInstancedBuffersStorage.vertexArrayObjects);\r\n }\r\n\r\n // Color\r\n if (!this.useVertexColor && this._isShaderMaterial(this._lineMaterial)) {\r\n const { r, g, b } = this.color;\r\n this._color4.set(r, g, b, this.alpha);\r\n this._lineMaterial.setColor4(\"color\", this._color4);\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _draw(subMesh: SubMesh, fillMode: number, instancesCount?: number): Mesh {\r\n if (!this._geometry || !this._geometry.getVertexBuffers() || (!this._unIndexed && !this._geometry.getIndexBuffer())) {\r\n return this;\r\n }\r\n\r\n const engine = this.getScene().getEngine();\r\n\r\n // Draw order\r\n\r\n if (this._unIndexed) {\r\n engine.drawArraysType(Material.LineListDrawMode, subMesh.verticesStart, subMesh.verticesCount, instancesCount);\r\n } else {\r\n engine.drawElementsType(Material.LineListDrawMode, subMesh.indexStart, subMesh.indexCount, instancesCount);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Disposes of the line mesh\r\n * @param doNotRecurse If children should be disposed\r\n * @param disposeMaterialAndTextures This parameter is not used by the LineMesh class\r\n * @param doNotDisposeMaterial If the material should not be disposed (default: false, meaning the material is disposed)\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public dispose(doNotRecurse?: boolean, disposeMaterialAndTextures = false, doNotDisposeMaterial?: boolean): void {\r\n if (!doNotDisposeMaterial) {\r\n this._lineMaterial.dispose(false, false, true);\r\n }\r\n super.dispose(doNotRecurse);\r\n }\r\n\r\n /**\r\n * Returns a new LineMesh object cloned from the current one.\r\n * @param name\r\n * @param newParent\r\n * @param doNotCloneChildren\r\n */\r\n public clone(name: string, newParent: Nullable = null, doNotCloneChildren?: boolean): LinesMesh {\r\n return new LinesMesh(name, this.getScene(), newParent, this, doNotCloneChildren);\r\n }\r\n\r\n /**\r\n * Creates a new InstancedLinesMesh object from the mesh model.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/copies/instances\r\n * @param name defines the name of the new instance\r\n * @returns a new InstancedLinesMesh\r\n */\r\n public createInstance(name: string): InstancedLinesMesh {\r\n const instance = new InstancedLinesMesh(name, this);\r\n\r\n if (this.instancedBuffers) {\r\n instance.instancedBuffers = {};\r\n\r\n for (const key in this.instancedBuffers) {\r\n instance.instancedBuffers[key] = this.instancedBuffers[key];\r\n }\r\n }\r\n\r\n return instance;\r\n }\r\n\r\n /**\r\n * Serializes this ground mesh\r\n * @param serializationObject object to write serialization to\r\n */\r\n public serialize(serializationObject: any): void {\r\n super.serialize(serializationObject);\r\n serializationObject.color = this.color.asArray();\r\n serializationObject.alpha = this.alpha;\r\n }\r\n\r\n /**\r\n * Parses a serialized ground mesh\r\n * @param parsedMesh the serialized mesh\r\n * @param scene the scene to create the ground mesh in\r\n * @returns the created ground mesh\r\n */\r\n public static Parse(parsedMesh: any, scene: Scene): LinesMesh {\r\n const result = new LinesMesh(parsedMesh.name, scene);\r\n\r\n result.color = Color3.FromArray(parsedMesh.color);\r\n result.alpha = parsedMesh.alpha;\r\n\r\n return result;\r\n }\r\n}\r\n\r\n/**\r\n * Creates an instance based on a source LinesMesh\r\n */\r\nexport class InstancedLinesMesh extends InstancedMesh {\r\n /**\r\n * The intersection Threshold is the margin applied when intersection a segment of the LinesMesh with a Ray.\r\n * This margin is expressed in world space coordinates, so its value may vary.\r\n * Initialized with the intersectionThreshold value of the source LinesMesh\r\n */\r\n public intersectionThreshold: number;\r\n\r\n constructor(name: string, source: LinesMesh) {\r\n super(name, source);\r\n this.intersectionThreshold = source.intersectionThreshold;\r\n }\r\n\r\n /**\r\n * Returns the string \"InstancedLinesMesh\".\r\n */\r\n public getClassName(): string {\r\n return \"InstancedLinesMesh\";\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport { Vector3 } from \"../../Maths/math.vector\";\r\nimport type { Color4 } from \"../../Maths/math.color\";\r\nimport { _CreationDataStorage, Mesh } from \"../mesh\";\r\nimport { VertexData } from \"../mesh.vertexData\";\r\nimport type { FloatArray, Nullable } from \"../../types\";\r\nimport { LinesMesh } from \"../../Meshes/linesMesh\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { VertexBuffer } from \"../../Buffers/buffer\";\r\nimport { Logger } from \"../../Misc/logger\";\r\n\r\nimport type { Material } from \"../../Materials/material\";\r\n\r\n/**\r\n * Creates the VertexData of the LineSystem\r\n * @param options an object used to set the following optional parameters for the LineSystem, required but can be empty\r\n * - lines an array of lines, each line being an array of successive Vector3\r\n * - colors an array of line colors, each of the line colors being an array of successive Color4, one per line point\r\n * @param options.lines\r\n * @param options.colors\r\n * @returns the VertexData of the LineSystem\r\n */\r\nexport function CreateLineSystemVertexData(options: { lines: Vector3[][]; colors?: Nullable }): VertexData {\r\n const indices = [];\r\n const positions = [];\r\n const lines = options.lines;\r\n const colors = options.colors;\r\n const vertexColors = [];\r\n let idx = 0;\r\n\r\n for (let l = 0; l < lines.length; l++) {\r\n const points = lines[l];\r\n for (let index = 0; index < points.length; index++) {\r\n positions.push(points[index].x, points[index].y, points[index].z);\r\n if (colors) {\r\n const color = colors[l];\r\n vertexColors.push(color[index].r, color[index].g, color[index].b, color[index].a);\r\n }\r\n if (index > 0) {\r\n indices.push(idx - 1);\r\n indices.push(idx);\r\n }\r\n idx++;\r\n }\r\n }\r\n const vertexData = new VertexData();\r\n vertexData.indices = indices;\r\n vertexData.positions = positions;\r\n if (colors) {\r\n vertexData.colors = vertexColors;\r\n }\r\n return vertexData;\r\n}\r\n\r\n/**\r\n * Create the VertexData for a DashedLines\r\n * @param options an object used to set the following optional parameters for the DashedLines, required but can be empty\r\n * - points an array successive Vector3\r\n * - dashSize the size of the dashes relative to the dash number, optional, default 3\r\n * - gapSize the size of the gap between two successive dashes relative to the dash number, optional, default 1\r\n * - dashNb the intended total number of dashes, optional, default 200\r\n * @param options.points\r\n * @param options.dashSize\r\n * @param options.gapSize\r\n * @param options.dashNb\r\n * @returns the VertexData for the DashedLines\r\n */\r\nexport function CreateDashedLinesVertexData(options: { points: Vector3[]; dashSize?: number; gapSize?: number; dashNb?: number }): VertexData {\r\n const dashSize = options.dashSize || 3;\r\n const gapSize = options.gapSize || 1;\r\n const dashNb = options.dashNb || 200;\r\n const points = options.points;\r\n\r\n const positions = new Array();\r\n const indices = new Array();\r\n\r\n const curvect = Vector3.Zero();\r\n let lg = 0;\r\n let nb = 0;\r\n let shft = 0;\r\n let dashshft = 0;\r\n let curshft = 0;\r\n let idx = 0;\r\n let i = 0;\r\n for (i = 0; i < points.length - 1; i++) {\r\n points[i + 1].subtractToRef(points[i], curvect);\r\n lg += curvect.length();\r\n }\r\n shft = lg / dashNb;\r\n dashshft = (dashSize * shft) / (dashSize + gapSize);\r\n for (i = 0; i < points.length - 1; i++) {\r\n points[i + 1].subtractToRef(points[i], curvect);\r\n nb = Math.floor(curvect.length() / shft);\r\n curvect.normalize();\r\n for (let j = 0; j < nb; j++) {\r\n curshft = shft * j;\r\n positions.push(points[i].x + curshft * curvect.x, points[i].y + curshft * curvect.y, points[i].z + curshft * curvect.z);\r\n positions.push(points[i].x + (curshft + dashshft) * curvect.x, points[i].y + (curshft + dashshft) * curvect.y, points[i].z + (curshft + dashshft) * curvect.z);\r\n indices.push(idx, idx + 1);\r\n idx += 2;\r\n }\r\n }\r\n\r\n // Result\r\n const vertexData = new VertexData();\r\n vertexData.positions = positions;\r\n vertexData.indices = indices;\r\n\r\n return vertexData;\r\n}\r\n\r\n/**\r\n * Creates a line system mesh. A line system is a pool of many lines gathered in a single mesh\r\n * * A line system mesh is considered as a parametric shape since it has no predefined original shape. Its shape is determined by the passed array of lines as an input parameter\r\n * * Like every other parametric shape, it is dynamically updatable by passing an existing instance of LineSystem to this static function\r\n * * The parameter `lines` is an array of lines, each line being an array of successive Vector3\r\n * * The optional parameter `instance` is an instance of an existing LineSystem object to be updated with the passed `lines` parameter\r\n * * The optional parameter `colors` is an array of line colors, each line colors being an array of successive Color4, one per line point\r\n * * The optional parameter `useVertexAlpha` is to be set to `false` (default `true`) when you don't need the alpha blending (faster)\r\n * * The optional parameter `material` is the material to use to draw the lines if provided. If not, a default material will be created\r\n * * Updating a simple Line mesh, you just need to update every line in the `lines` array : https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#lines-and-dashedlines\r\n * * When updating an instance, remember that only line point positions can change, not the number of points, neither the number of lines\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#line-system\r\n * @param name defines the name of the new line system\r\n * @param options defines the options used to create the line system\r\n * @param options.lines\r\n * @param options.updatable\r\n * @param options.instance\r\n * @param options.colors\r\n * @param options.useVertexAlpha\r\n * @param options.material\r\n * @param scene defines the hosting scene\r\n * @returns a new line system mesh\r\n */\r\nexport function CreateLineSystem(\r\n name: string,\r\n options: { lines: Vector3[][]; updatable?: boolean; instance?: Nullable; colors?: Nullable; useVertexAlpha?: boolean; material?: Material },\r\n scene: Nullable = null\r\n): LinesMesh {\r\n const instance = options.instance;\r\n const lines = options.lines;\r\n const colors = options.colors;\r\n\r\n if (instance) {\r\n // lines update\r\n const positions = instance.getVerticesData(VertexBuffer.PositionKind)!;\r\n let vertexColor;\r\n let lineColors;\r\n if (colors) {\r\n vertexColor = instance.getVerticesData(VertexBuffer.ColorKind)!;\r\n }\r\n let i = 0;\r\n let c = 0;\r\n for (let l = 0; l < lines.length; l++) {\r\n const points = lines[l];\r\n for (let p = 0; p < points.length; p++) {\r\n positions[i] = points[p].x;\r\n positions[i + 1] = points[p].y;\r\n positions[i + 2] = points[p].z;\r\n if (colors && vertexColor) {\r\n lineColors = colors[l];\r\n vertexColor[c] = lineColors[p].r;\r\n vertexColor[c + 1] = lineColors[p].g;\r\n vertexColor[c + 2] = lineColors[p].b;\r\n vertexColor[c + 3] = lineColors[p].a;\r\n c += 4;\r\n }\r\n i += 3;\r\n }\r\n }\r\n instance.updateVerticesData(VertexBuffer.PositionKind, positions, false, false);\r\n if (colors && vertexColor) {\r\n instance.updateVerticesData(VertexBuffer.ColorKind, vertexColor, false, false);\r\n }\r\n return instance;\r\n }\r\n\r\n // line system creation\r\n const useVertexColor = colors ? true : false;\r\n const lineSystem = new LinesMesh(name, scene, null, undefined, undefined, useVertexColor, options.useVertexAlpha, options.material);\r\n const vertexData = CreateLineSystemVertexData(options);\r\n vertexData.applyToMesh(lineSystem, options.updatable);\r\n return lineSystem;\r\n}\r\n\r\n/**\r\n * Creates a line mesh\r\n * A line mesh is considered as a parametric shape since it has no predefined original shape. Its shape is determined by the passed array of points as an input parameter\r\n * * Like every other parametric shape, it is dynamically updatable by passing an existing instance of LineMesh to this static function\r\n * * The parameter `points` is an array successive Vector3\r\n * * The optional parameter `instance` is an instance of an existing LineMesh object to be updated with the passed `points` parameter : https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#lines-and-dashedlines\r\n * * The optional parameter `colors` is an array of successive Color4, one per line point\r\n * * The optional parameter `useVertexAlpha` is to be set to `false` (default `true`) when you don't need alpha blending (faster)\r\n * * The optional parameter `material` is the material to use to draw the lines if provided. If not, a default material will be created\r\n * * When updating an instance, remember that only point positions can change, not the number of points\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#lines\r\n * @param name defines the name of the new line system\r\n * @param options defines the options used to create the line system\r\n * @param options.points\r\n * @param options.updatable\r\n * @param options.instance\r\n * @param options.colors\r\n * @param options.useVertexAlpha\r\n * @param options.material\r\n * @param scene defines the hosting scene\r\n * @returns a new line mesh\r\n */\r\nexport function CreateLines(\r\n name: string,\r\n options: { points: Vector3[]; updatable?: boolean; instance?: Nullable; colors?: Color4[]; useVertexAlpha?: boolean; material?: Material },\r\n scene: Nullable = null\r\n): LinesMesh {\r\n const colors = options.colors ? [options.colors] : null;\r\n const lines = CreateLineSystem(\r\n name,\r\n { lines: [options.points], updatable: options.updatable, instance: options.instance, colors: colors, useVertexAlpha: options.useVertexAlpha, material: options.material },\r\n scene\r\n );\r\n return lines;\r\n}\r\n\r\n/**\r\n * Creates a dashed line mesh\r\n * * A dashed line mesh is considered as a parametric shape since it has no predefined original shape. Its shape is determined by the passed array of points as an input parameter\r\n * * Like every other parametric shape, it is dynamically updatable by passing an existing instance of LineMesh to this static function\r\n * * The parameter `points` is an array successive Vector3\r\n * * The parameter `dashNb` is the intended total number of dashes (positive integer, default 200)\r\n * * The parameter `dashSize` is the size of the dashes relatively the dash number (positive float, default 3)\r\n * * The parameter `gapSize` is the size of the gap between two successive dashes relatively the dash number (positive float, default 1)\r\n * * The optional parameter `instance` is an instance of an existing LineMesh object to be updated with the passed `points` parameter : https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#lines-and-dashedlines\r\n * * The optional parameter `useVertexAlpha` is to be set to `false` (default `true`) when you don't need the alpha blending (faster)\r\n * * The optional parameter `material` is the material to use to draw the lines if provided. If not, a default material will be created\r\n * * When updating an instance, remember that only point positions can change, not the number of points\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param options.points\r\n * @param options.dashSize\r\n * @param options.gapSize\r\n * @param options.dashNb\r\n * @param options.updatable\r\n * @param options.instance\r\n * @param options.useVertexAlpha\r\n * @param options.material\r\n * @param scene defines the hosting scene\r\n * @returns the dashed line mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#dashed-lines\r\n */\r\nexport function CreateDashedLines(\r\n name: string,\r\n options: { points: Vector3[]; dashSize?: number; gapSize?: number; dashNb?: number; updatable?: boolean; instance?: LinesMesh; useVertexAlpha?: boolean; material?: Material },\r\n scene: Nullable = null\r\n): LinesMesh {\r\n const points = options.points;\r\n const instance = options.instance;\r\n const gapSize = options.gapSize || 1;\r\n const dashSize = options.dashSize || 3;\r\n\r\n if (instance) {\r\n // dashed lines update\r\n const positionFunction = (positions: FloatArray): void => {\r\n const curvect = Vector3.Zero();\r\n const nbSeg = positions.length / 6;\r\n let lg = 0;\r\n let nb = 0;\r\n let shft = 0;\r\n let dashshft = 0;\r\n let curshft = 0;\r\n let p = 0;\r\n let i = 0;\r\n let j = 0;\r\n for (i = 0; i < points.length - 1; i++) {\r\n points[i + 1].subtractToRef(points[i], curvect);\r\n lg += curvect.length();\r\n }\r\n shft = lg / nbSeg;\r\n const dashSize = instance!._creationDataStorage!.dashSize;\r\n const gapSize = instance!._creationDataStorage!.gapSize;\r\n dashshft = (dashSize * shft) / (dashSize + gapSize);\r\n for (i = 0; i < points.length - 1; i++) {\r\n points[i + 1].subtractToRef(points[i], curvect);\r\n nb = Math.floor(curvect.length() / shft);\r\n curvect.normalize();\r\n j = 0;\r\n while (j < nb && p < positions.length) {\r\n curshft = shft * j;\r\n positions[p] = points[i].x + curshft * curvect.x;\r\n positions[p + 1] = points[i].y + curshft * curvect.y;\r\n positions[p + 2] = points[i].z + curshft * curvect.z;\r\n positions[p + 3] = points[i].x + (curshft + dashshft) * curvect.x;\r\n positions[p + 4] = points[i].y + (curshft + dashshft) * curvect.y;\r\n positions[p + 5] = points[i].z + (curshft + dashshft) * curvect.z;\r\n p += 6;\r\n j++;\r\n }\r\n }\r\n while (p < positions.length) {\r\n positions[p] = points[i].x;\r\n positions[p + 1] = points[i].y;\r\n positions[p + 2] = points[i].z;\r\n p += 3;\r\n }\r\n };\r\n if (options.dashNb || options.dashSize || options.gapSize || options.useVertexAlpha || options.material) {\r\n Logger.Warn(\"You have used an option other than points with the instance option. Please be aware that these other options will be ignored.\");\r\n }\r\n instance.updateMeshPositions(positionFunction, false);\r\n return instance;\r\n }\r\n // dashed lines creation\r\n const dashedLines = new LinesMesh(name, scene, null, undefined, undefined, undefined, options.useVertexAlpha, options.material);\r\n const vertexData = CreateDashedLinesVertexData(options);\r\n vertexData.applyToMesh(dashedLines, options.updatable);\r\n\r\n dashedLines._creationDataStorage = new _CreationDataStorage();\r\n dashedLines._creationDataStorage.dashSize = dashSize;\r\n dashedLines._creationDataStorage.gapSize = gapSize;\r\n return dashedLines;\r\n}\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated use the functions directly from the module\r\n */\r\nexport const LinesBuilder = {\r\n CreateDashedLines,\r\n CreateLineSystem,\r\n CreateLines,\r\n};\r\n\r\nVertexData.CreateLineSystem = CreateLineSystemVertexData;\r\nVertexData.CreateDashedLines = CreateDashedLinesVertexData;\r\n\r\nMesh.CreateLines = (name: string, points: Vector3[], scene: Nullable = null, updatable: boolean = false, instance: Nullable = null): LinesMesh => {\r\n const options = {\r\n points,\r\n updatable,\r\n instance,\r\n };\r\n return CreateLines(name, options, scene);\r\n};\r\n\r\nMesh.CreateDashedLines = (\r\n name: string,\r\n points: Vector3[],\r\n dashSize: number,\r\n gapSize: number,\r\n dashNb: number,\r\n scene: Nullable = null,\r\n updatable?: boolean,\r\n instance?: LinesMesh\r\n): LinesMesh => {\r\n const options = {\r\n points,\r\n dashSize,\r\n gapSize,\r\n dashNb,\r\n updatable,\r\n instance,\r\n };\r\n return CreateDashedLines(name, options, scene);\r\n};\r\n","import type { Observer } from \"../Misc/observable\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { IDisposable } from \"../scene\";\r\n\r\n/**\r\n * Construction options for a timer\r\n */\r\nexport interface ITimerOptions {\r\n /**\r\n * Time-to-end\r\n */\r\n timeout: number;\r\n /**\r\n * The context observable is used to calculate time deltas and provides the context of the timer's callbacks. Will usually be OnBeforeRenderObservable.\r\n * Countdown calculation is done ONLY when the observable is notifying its observers, meaning that if\r\n * you choose an observable that doesn't trigger too often, the wait time might extend further than the requested max time\r\n */\r\n contextObservable: Observable;\r\n /**\r\n * Optional parameters when adding an observer to the observable\r\n */\r\n observableParameters?: {\r\n mask?: number;\r\n insertFirst?: boolean;\r\n scope?: any;\r\n };\r\n /**\r\n * An optional break condition that will stop the times prematurely. In this case onEnded will not be triggered!\r\n */\r\n breakCondition?: (data?: ITimerData) => boolean;\r\n /**\r\n * Will be triggered when the time condition has met\r\n */\r\n onEnded?: (data: ITimerData) => void;\r\n /**\r\n * Will be triggered when the break condition has met (prematurely ended)\r\n */\r\n onAborted?: (data: ITimerData) => void;\r\n /**\r\n * Optional function to execute on each tick (or count)\r\n */\r\n onTick?: (data: ITimerData) => void;\r\n}\r\n\r\n/**\r\n * An interface defining the data sent by the timer\r\n */\r\nexport interface ITimerData {\r\n /**\r\n * When did it start\r\n */\r\n startTime: number;\r\n /**\r\n * Time now\r\n */\r\n currentTime: number;\r\n /**\r\n * Time passed since started\r\n */\r\n deltaTime: number;\r\n /**\r\n * How much is completed, in [0.0...1.0].\r\n * Note that this CAN be higher than 1 due to the fact that we don't actually measure time but delta between observable calls\r\n */\r\n completeRate: number;\r\n /**\r\n * What the registered observable sent in the last count\r\n */\r\n payload: T;\r\n}\r\n\r\n/**\r\n * The current state of the timer\r\n */\r\nexport enum TimerState {\r\n /**\r\n * Timer initialized, not yet started\r\n */\r\n INIT,\r\n /**\r\n * Timer started and counting\r\n */\r\n STARTED,\r\n /**\r\n * Timer ended (whether aborted or time reached)\r\n */\r\n ENDED,\r\n}\r\n\r\n/**\r\n * A simple version of the timer. Will take options and start the timer immediately after calling it\r\n *\r\n * @param options options with which to initialize this timer\r\n */\r\nexport function setAndStartTimer(options: ITimerOptions): Nullable> {\r\n let timer = 0;\r\n const startTime = Date.now();\r\n options.observableParameters = options.observableParameters ?? {};\r\n const observer = options.contextObservable.add(\r\n (payload: any) => {\r\n const now = Date.now();\r\n timer = now - startTime;\r\n const data: ITimerData = {\r\n startTime,\r\n currentTime: now,\r\n deltaTime: timer,\r\n completeRate: timer / options.timeout,\r\n payload,\r\n };\r\n options.onTick && options.onTick(data);\r\n if (options.breakCondition && options.breakCondition()) {\r\n options.contextObservable.remove(observer);\r\n options.onAborted && options.onAborted(data);\r\n }\r\n if (timer >= options.timeout) {\r\n options.contextObservable.remove(observer);\r\n options.onEnded && options.onEnded(data);\r\n }\r\n },\r\n options.observableParameters.mask,\r\n options.observableParameters.insertFirst,\r\n options.observableParameters.scope\r\n );\r\n return observer;\r\n}\r\n\r\n/**\r\n * An advanced implementation of a timer class\r\n */\r\nexport class AdvancedTimer implements IDisposable {\r\n /**\r\n * Will notify each time the timer calculates the remaining time\r\n */\r\n public onEachCountObservable: Observable> = new Observable();\r\n /**\r\n * Will trigger when the timer was aborted due to the break condition\r\n */\r\n public onTimerAbortedObservable: Observable> = new Observable();\r\n /**\r\n * Will trigger when the timer ended successfully\r\n */\r\n public onTimerEndedObservable: Observable> = new Observable();\r\n /**\r\n * Will trigger when the timer state has changed\r\n */\r\n public onStateChangedObservable: Observable = new Observable();\r\n\r\n private _observer: Nullable> = null;\r\n private _contextObservable: Observable;\r\n private _observableParameters: {\r\n mask?: number;\r\n insertFirst?: boolean;\r\n scope?: any;\r\n };\r\n private _startTime: number;\r\n private _timer: number;\r\n private _state: TimerState;\r\n private _breakCondition: (data: ITimerData) => boolean;\r\n private _timeToEnd: number;\r\n private _breakOnNextTick: boolean = false;\r\n\r\n /**\r\n * Will construct a new advanced timer based on the options provided. Timer will not start until start() is called.\r\n * @param options construction options for this advanced timer\r\n */\r\n constructor(options: ITimerOptions) {\r\n this._setState(TimerState.INIT);\r\n this._contextObservable = options.contextObservable;\r\n this._observableParameters = options.observableParameters ?? {};\r\n this._breakCondition = options.breakCondition ?? (() => false);\r\n this._timeToEnd = options.timeout;\r\n if (options.onEnded) {\r\n this.onTimerEndedObservable.add(options.onEnded);\r\n }\r\n if (options.onTick) {\r\n this.onEachCountObservable.add(options.onTick);\r\n }\r\n if (options.onAborted) {\r\n this.onTimerAbortedObservable.add(options.onAborted);\r\n }\r\n }\r\n\r\n /**\r\n * set a breaking condition for this timer. Default is to never break during count\r\n * @param predicate the new break condition. Returns true to break, false otherwise\r\n */\r\n public set breakCondition(predicate: (data: ITimerData) => boolean) {\r\n this._breakCondition = predicate;\r\n }\r\n\r\n /**\r\n * Reset ALL associated observables in this advanced timer\r\n */\r\n public clearObservables() {\r\n this.onEachCountObservable.clear();\r\n this.onTimerAbortedObservable.clear();\r\n this.onTimerEndedObservable.clear();\r\n this.onStateChangedObservable.clear();\r\n }\r\n\r\n /**\r\n * Will start a new iteration of this timer. Only one instance of this timer can run at a time.\r\n *\r\n * @param timeToEnd how much time to measure until timer ended\r\n */\r\n public start(timeToEnd: number = this._timeToEnd) {\r\n if (this._state === TimerState.STARTED) {\r\n throw new Error(\"Timer already started. Please stop it before starting again\");\r\n }\r\n this._timeToEnd = timeToEnd;\r\n this._startTime = Date.now();\r\n this._timer = 0;\r\n this._observer = this._contextObservable.add(this._tick, this._observableParameters.mask, this._observableParameters.insertFirst, this._observableParameters.scope);\r\n this._setState(TimerState.STARTED);\r\n }\r\n\r\n /**\r\n * Will force a stop on the next tick.\r\n */\r\n public stop() {\r\n if (this._state !== TimerState.STARTED) {\r\n return;\r\n }\r\n this._breakOnNextTick = true;\r\n }\r\n\r\n /**\r\n * Dispose this timer, clearing all resources\r\n */\r\n public dispose() {\r\n if (this._observer) {\r\n this._contextObservable.remove(this._observer);\r\n }\r\n this.clearObservables();\r\n }\r\n\r\n private _setState(newState: TimerState) {\r\n this._state = newState;\r\n this.onStateChangedObservable.notifyObservers(this._state);\r\n }\r\n\r\n private _tick = (payload: T) => {\r\n const now = Date.now();\r\n this._timer = now - this._startTime;\r\n const data: ITimerData = {\r\n startTime: this._startTime,\r\n currentTime: now,\r\n deltaTime: this._timer,\r\n completeRate: this._timer / this._timeToEnd,\r\n payload,\r\n };\r\n const shouldBreak = this._breakOnNextTick || this._breakCondition(data);\r\n if (shouldBreak || this._timer >= this._timeToEnd) {\r\n this._stop(data, shouldBreak);\r\n } else {\r\n this.onEachCountObservable.notifyObservers(data);\r\n }\r\n };\r\n\r\n private _stop(data: ITimerData, aborted: boolean = false) {\r\n this._contextObservable.remove(this._observer);\r\n this._setState(TimerState.ENDED);\r\n if (aborted) {\r\n this.onTimerAbortedObservable.notifyObservers(data);\r\n } else {\r\n this.onTimerEndedObservable.notifyObservers(data);\r\n }\r\n }\r\n}\r\n","import type { IWebXRFeature } from \"../webXRFeaturesManager\";\r\nimport { WebXRFeaturesManager, WebXRFeatureName } from \"../webXRFeaturesManager\";\r\nimport type { Observer } from \"../../Misc/observable\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport type { WebXRSessionManager } from \"../webXRSessionManager\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { WebXRInput } from \"../webXRInput\";\r\nimport type { WebXRInputSource } from \"../webXRInputSource\";\r\nimport type { IWebXRMotionControllerAxesValue } from \"../motionController/webXRControllerComponent\";\r\nimport { WebXRControllerComponent } from \"../motionController/webXRControllerComponent\";\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport { Vector3, Quaternion } from \"../../Maths/math.vector\";\r\nimport { Ray } from \"../../Culling/ray\";\r\nimport type { Material } from \"../../Materials/material\";\r\nimport { DynamicTexture } from \"../../Materials/Textures/dynamicTexture\";\r\nimport { CreateCylinder } from \"../../Meshes/Builders/cylinderBuilder\";\r\nimport { SineEase, EasingFunction } from \"../../Animations/easing\";\r\nimport { Animation } from \"../../Animations/animation\";\r\nimport { Axis } from \"../../Maths/math.axis\";\r\nimport { StandardMaterial } from \"../../Materials/standardMaterial\";\r\nimport { CreateGround } from \"../../Meshes/Builders/groundBuilder\";\r\nimport { CreateTorus } from \"../../Meshes/Builders/torusBuilder\";\r\nimport type { PickingInfo } from \"../../Collisions/pickingInfo\";\r\nimport { Curve3 } from \"../../Maths/math.path\";\r\nimport { CreateLines } from \"../../Meshes/Builders/linesBuilder\";\r\nimport { WebXRAbstractFeature } from \"./WebXRAbstractFeature\";\r\nimport { Color3, Color4 } from \"../../Maths/math.color\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { UtilityLayerRenderer } from \"../../Rendering/utilityLayerRenderer\";\r\nimport { PointerEventTypes } from \"../../Events/pointerEvents\";\r\nimport { setAndStartTimer } from \"../../Misc/timer\";\r\nimport type { LinesMesh } from \"../../Meshes/linesMesh\";\r\n\r\n/**\r\n * The options container for the teleportation module\r\n */\r\nexport interface IWebXRTeleportationOptions {\r\n /**\r\n * if provided, this scene will be used to render meshes.\r\n */\r\n customUtilityLayerScene?: Scene;\r\n /**\r\n * Values to configure the default target mesh\r\n */\r\n defaultTargetMeshOptions?: {\r\n /**\r\n * Fill color of the teleportation area\r\n */\r\n teleportationFillColor?: string;\r\n /**\r\n * Border color for the teleportation area\r\n */\r\n teleportationBorderColor?: string;\r\n /**\r\n * Disable the mesh's animation sequence\r\n */\r\n disableAnimation?: boolean;\r\n /**\r\n * Disable lighting on the material or the ring and arrow\r\n */\r\n disableLighting?: boolean;\r\n /**\r\n * Override the default material of the torus and arrow\r\n */\r\n torusArrowMaterial?: Material;\r\n /**\r\n * Override the default material of the Landing Zone\r\n */\r\n teleportationCircleMaterial?: Material;\r\n };\r\n /**\r\n * A list of meshes to use as floor meshes.\r\n * Meshes can be added and removed after initializing the feature using the\r\n * addFloorMesh and removeFloorMesh functions\r\n * If empty, rotation will still work\r\n */\r\n floorMeshes?: AbstractMesh[];\r\n /**\r\n * use this rendering group id for the meshes (optional)\r\n */\r\n renderingGroupId?: number;\r\n /**\r\n * Should teleportation move only to snap points\r\n */\r\n snapPointsOnly?: boolean;\r\n /**\r\n * An array of points to which the teleportation will snap to.\r\n * If the teleportation ray is in the proximity of one of those points, it will be corrected to this point.\r\n */\r\n snapPositions?: Vector3[];\r\n /**\r\n * How close should the teleportation ray be in order to snap to position.\r\n * Default to 0.8 units (meters)\r\n */\r\n snapToPositionRadius?: number;\r\n /**\r\n * Provide your own teleportation mesh instead of babylon's wonderful doughnut.\r\n * If you want to support rotation, make sure your mesh has a direction indicator.\r\n *\r\n * When left untouched, the default mesh will be initialized.\r\n */\r\n teleportationTargetMesh?: AbstractMesh;\r\n /**\r\n * If main component is used (no thumbstick), how long should the \"long press\" take before teleport\r\n */\r\n timeToTeleport?: number;\r\n /**\r\n * Disable using the thumbstick and use the main component (usually trigger) on long press.\r\n * This will be automatically true if the controller doesn't have a thumbstick or touchpad.\r\n */\r\n useMainComponentOnly?: boolean;\r\n /**\r\n * Should meshes created here be added to a utility layer or the main scene\r\n */\r\n useUtilityLayer?: boolean;\r\n /**\r\n * Babylon XR Input class for controller\r\n */\r\n xrInput: WebXRInput;\r\n\r\n /**\r\n * Meshes that the teleportation ray cannot go through\r\n */\r\n pickBlockerMeshes?: AbstractMesh[];\r\n\r\n /**\r\n * Color of the teleportation ray when it is blocked by a mesh in the pickBlockerMeshes array\r\n * Defaults to red.\r\n */\r\n blockedRayColor?: Color4;\r\n\r\n /**\r\n * Should teleport work only on a specific hand?\r\n */\r\n forceHandedness?: XRHandedness;\r\n\r\n /**\r\n * If provided, this function will be used to generate the ray mesh instead of the lines mesh being used per default\r\n */\r\n generateRayPathMesh?: (points: Vector3[], pickingInfo: PickingInfo) => AbstractMesh;\r\n}\r\n\r\n/**\r\n * This is a teleportation feature to be used with WebXR-enabled motion controllers.\r\n * When enabled and attached, the feature will allow a user to move around and rotate in the scene using\r\n * the input of the attached controllers.\r\n */\r\nexport class WebXRMotionControllerTeleportation extends WebXRAbstractFeature {\r\n private _controllers: {\r\n [controllerUniqueId: string]: {\r\n xrController: WebXRInputSource;\r\n teleportationComponent?: WebXRControllerComponent;\r\n teleportationState: {\r\n forward: boolean;\r\n backwards: boolean;\r\n currentRotation: number;\r\n baseRotation: number;\r\n rotating: boolean;\r\n blocked: boolean;\r\n };\r\n onAxisChangedObserver?: Nullable>;\r\n onButtonChangedObserver?: Nullable>;\r\n };\r\n } = {};\r\n private _currentTeleportationControllerId: string;\r\n private _floorMeshes: AbstractMesh[];\r\n private _quadraticBezierCurve: Nullable;\r\n private _selectionFeature: Nullable;\r\n private _snapToPositions: Vector3[];\r\n private _snappedToPoint: boolean = false;\r\n private _teleportationRingMaterial?: StandardMaterial;\r\n private _blockedRayColor: Color4;\r\n private _cachedColor4White = new Color4(1, 1, 1, 1);\r\n private _tmpRay = new Ray(new Vector3(), new Vector3());\r\n private _tmpVector = new Vector3();\r\n private _tmpQuaternion = new Quaternion();\r\n\r\n /**\r\n * Skip the next teleportation. This can be controlled by the user to prevent the user from teleportation\r\n * to sections that are not yet \"unlocked\", but should still show the teleportation mesh.\r\n */\r\n public skipNextTeleportation = false;\r\n\r\n /**\r\n * The module's name\r\n */\r\n public static readonly Name = WebXRFeatureName.TELEPORTATION;\r\n /**\r\n * The (Babylon) version of this module.\r\n * This is an integer representing the implementation version.\r\n * This number does not correspond to the webxr specs version\r\n */\r\n public static readonly Version = 1;\r\n\r\n /**\r\n * Is movement backwards enabled\r\n */\r\n public backwardsMovementEnabled = true;\r\n /**\r\n * Distance to travel when moving backwards\r\n */\r\n public backwardsTeleportationDistance: number = 0.7;\r\n /**\r\n * The distance from the user to the inspection point in the direction of the controller\r\n * A higher number will allow the user to move further\r\n * defaults to 5 (meters, in xr units)\r\n */\r\n public parabolicCheckRadius: number = 5;\r\n /**\r\n * Should the module support parabolic ray on top of direct ray\r\n * If enabled, the user will be able to point \"at the sky\" and move according to predefined radius distance\r\n * Very helpful when moving between floors / different heights\r\n */\r\n public parabolicRayEnabled: boolean = true;\r\n\r\n /**\r\n * The second type of ray - straight line.\r\n * Should it be enabled or should the parabolic line be the only one.\r\n */\r\n public straightRayEnabled: boolean = true;\r\n /**\r\n * How much rotation should be applied when rotating right and left\r\n */\r\n public rotationAngle: number = Math.PI / 8;\r\n\r\n /**\r\n * This observable will notify when the target mesh position was updated.\r\n * The picking info it provides contains the point to which the target mesh will move ()\r\n */\r\n public onTargetMeshPositionUpdatedObservable: Observable = new Observable();\r\n\r\n /**\r\n * Is teleportation enabled. Can be used to allow rotation only.\r\n */\r\n public teleportationEnabled: boolean = true;\r\n\r\n private _rotationEnabled: boolean = true;\r\n\r\n /**\r\n * Is rotation enabled when moving forward?\r\n * Disabling this feature will prevent the user from deciding the direction when teleporting\r\n */\r\n public get rotationEnabled(): boolean {\r\n return this._rotationEnabled;\r\n }\r\n\r\n /**\r\n * Sets whether rotation is enabled or not\r\n * @param enabled is rotation enabled when teleportation is shown\r\n */\r\n public set rotationEnabled(enabled: boolean) {\r\n this._rotationEnabled = enabled;\r\n\r\n if (this._options.teleportationTargetMesh) {\r\n const children = this._options.teleportationTargetMesh.getChildMeshes(false, (node) => node.name === \"rotationCone\");\r\n if (children[0]) {\r\n children[0].setEnabled(enabled);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Exposes the currently set teleportation target mesh.\r\n */\r\n public get teleportationTargetMesh(): Nullable {\r\n return this._options.teleportationTargetMesh || null;\r\n }\r\n\r\n /**\r\n * constructs a new teleportation system\r\n * @param _xrSessionManager an instance of WebXRSessionManager\r\n * @param _options configuration object for this feature\r\n */\r\n constructor(_xrSessionManager: WebXRSessionManager, private _options: IWebXRTeleportationOptions) {\r\n super(_xrSessionManager);\r\n // create default mesh if not provided\r\n if (!this._options.teleportationTargetMesh) {\r\n this._createDefaultTargetMesh();\r\n }\r\n\r\n this._floorMeshes = this._options.floorMeshes || [];\r\n this._snapToPositions = this._options.snapPositions || [];\r\n this._blockedRayColor = this._options.blockedRayColor || new Color4(1, 0, 0, 0.75);\r\n\r\n this._setTargetMeshVisibility(false);\r\n }\r\n\r\n /**\r\n * Get the snapPointsOnly flag\r\n */\r\n public get snapPointsOnly(): boolean {\r\n return !!this._options.snapPointsOnly;\r\n }\r\n\r\n /**\r\n * Sets the snapPointsOnly flag\r\n * @param snapToPoints should teleportation be exclusively to snap points\r\n */\r\n public set snapPointsOnly(snapToPoints: boolean) {\r\n this._options.snapPointsOnly = snapToPoints;\r\n }\r\n\r\n /**\r\n * Add a new mesh to the floor meshes array\r\n * @param mesh the mesh to use as floor mesh\r\n */\r\n public addFloorMesh(mesh: AbstractMesh) {\r\n this._floorMeshes.push(mesh);\r\n }\r\n\r\n /**\r\n * Add a mesh to the list of meshes blocking the teleportation ray\r\n * @param mesh The mesh to add to the teleportation-blocking meshes\r\n */\r\n public addBlockerMesh(mesh: AbstractMesh) {\r\n this._options.pickBlockerMeshes = this._options.pickBlockerMeshes || [];\r\n this._options.pickBlockerMeshes.push(mesh);\r\n }\r\n\r\n /**\r\n * Add a new snap-to point to fix teleportation to this position\r\n * @param newSnapPoint The new Snap-To point\r\n */\r\n public addSnapPoint(newSnapPoint: Vector3) {\r\n this._snapToPositions.push(newSnapPoint);\r\n }\r\n\r\n public attach(): boolean {\r\n if (!super.attach()) {\r\n return false;\r\n }\r\n\r\n // Safety reset\r\n this._currentTeleportationControllerId = \"\";\r\n\r\n this._options.xrInput.controllers.forEach(this._attachController);\r\n this._addNewAttachObserver(this._options.xrInput.onControllerAddedObservable, this._attachController);\r\n this._addNewAttachObserver(this._options.xrInput.onControllerRemovedObservable, (controller) => {\r\n // REMOVE the controller\r\n this._detachController(controller.uniqueId);\r\n });\r\n\r\n return true;\r\n }\r\n\r\n public detach(): boolean {\r\n if (!super.detach()) {\r\n return false;\r\n }\r\n\r\n Object.keys(this._controllers).forEach((controllerId) => {\r\n this._detachController(controllerId);\r\n });\r\n\r\n this._setTargetMeshVisibility(false);\r\n this._currentTeleportationControllerId = \"\";\r\n this._controllers = {};\r\n\r\n return true;\r\n }\r\n\r\n public dispose(): void {\r\n super.dispose();\r\n this._options.teleportationTargetMesh && this._options.teleportationTargetMesh.dispose(false, true);\r\n }\r\n\r\n /**\r\n * Remove a mesh from the floor meshes array\r\n * @param mesh the mesh to remove\r\n */\r\n public removeFloorMesh(mesh: AbstractMesh) {\r\n const index = this._floorMeshes.indexOf(mesh);\r\n if (index !== -1) {\r\n this._floorMeshes.splice(index, 1);\r\n }\r\n }\r\n\r\n /**\r\n * Remove a mesh from the blocker meshes array\r\n * @param mesh the mesh to remove\r\n */\r\n public removeBlockerMesh(mesh: AbstractMesh) {\r\n this._options.pickBlockerMeshes = this._options.pickBlockerMeshes || [];\r\n const index = this._options.pickBlockerMeshes.indexOf(mesh);\r\n if (index !== -1) {\r\n this._options.pickBlockerMeshes.splice(index, 1);\r\n }\r\n }\r\n\r\n /**\r\n * Remove a mesh from the floor meshes array using its name\r\n * @param name the mesh name to remove\r\n */\r\n public removeFloorMeshByName(name: string) {\r\n const mesh = this._xrSessionManager.scene.getMeshByName(name);\r\n if (mesh) {\r\n this.removeFloorMesh(mesh);\r\n }\r\n }\r\n\r\n /**\r\n * This function will iterate through the array, searching for this point or equal to it. It will then remove it from the snap-to array\r\n * @param snapPointToRemove the point (or a clone of it) to be removed from the array\r\n * @returns was the point found and removed or not\r\n */\r\n public removeSnapPoint(snapPointToRemove: Vector3): boolean {\r\n // check if the object is in the array\r\n let index = this._snapToPositions.indexOf(snapPointToRemove);\r\n // if not found as an object, compare to the points\r\n if (index === -1) {\r\n for (let i = 0; i < this._snapToPositions.length; ++i) {\r\n // equals? index is i, break the loop\r\n if (this._snapToPositions[i].equals(snapPointToRemove)) {\r\n index = i;\r\n break;\r\n }\r\n }\r\n }\r\n // index is not -1? remove the object\r\n if (index !== -1) {\r\n this._snapToPositions.splice(index, 1);\r\n return true;\r\n }\r\n return false;\r\n }\r\n\r\n /**\r\n * This function sets a selection feature that will be disabled when\r\n * the forward ray is shown and will be reattached when hidden.\r\n * This is used to remove the selection rays when moving.\r\n * @param selectionFeature the feature to disable when forward movement is enabled\r\n */\r\n public setSelectionFeature(selectionFeature: Nullable) {\r\n this._selectionFeature = selectionFeature;\r\n }\r\n\r\n protected _onXRFrame(_xrFrame: XRFrame) {\r\n const frame = this._xrSessionManager.currentFrame;\r\n const scene = this._xrSessionManager.scene;\r\n if (!this.attach || !frame) {\r\n return;\r\n }\r\n\r\n // render target if needed\r\n const targetMesh = this._options.teleportationTargetMesh;\r\n if (this._currentTeleportationControllerId) {\r\n if (!targetMesh) {\r\n return;\r\n }\r\n targetMesh.rotationQuaternion = targetMesh.rotationQuaternion || new Quaternion();\r\n const controllerData = this._controllers[this._currentTeleportationControllerId];\r\n if (controllerData && controllerData.teleportationState.forward) {\r\n // set the rotation\r\n Quaternion.RotationYawPitchRollToRef(\r\n controllerData.teleportationState.currentRotation + controllerData.teleportationState.baseRotation,\r\n 0,\r\n 0,\r\n targetMesh.rotationQuaternion\r\n );\r\n // set the ray and position\r\n\r\n let hitPossible = false;\r\n controllerData.xrController.getWorldPointerRayToRef(this._tmpRay);\r\n if (this.straightRayEnabled) {\r\n // first check if direct ray possible\r\n // pick grounds that are LOWER only. upper will use parabolic path\r\n const pick = scene.pickWithRay(this._tmpRay, (o) => {\r\n // check for mesh-blockers\r\n if (this._options.pickBlockerMeshes && this._options.pickBlockerMeshes.indexOf(o) !== -1) {\r\n return true;\r\n }\r\n const index = this._floorMeshes.indexOf(o);\r\n if (index === -1) {\r\n return false;\r\n }\r\n return this._floorMeshes[index].absolutePosition.y < this._options.xrInput.xrCamera.globalPosition.y;\r\n });\r\n if (pick && pick.pickedMesh && this._options.pickBlockerMeshes && this._options.pickBlockerMeshes.indexOf(pick.pickedMesh) !== -1) {\r\n controllerData.teleportationState.blocked = true;\r\n this._setTargetMeshVisibility(false);\r\n this._showParabolicPath(pick);\r\n return;\r\n } else if (pick && pick.pickedPoint) {\r\n controllerData.teleportationState.blocked = false;\r\n hitPossible = true;\r\n this._setTargetMeshPosition(pick);\r\n this._setTargetMeshVisibility(true);\r\n this._showParabolicPath(pick);\r\n }\r\n }\r\n // straight ray is still the main ray, but disabling the straight line will force parabolic line.\r\n if (this.parabolicRayEnabled && !hitPossible) {\r\n // radius compensation according to pointer rotation around X\r\n const xRotation = controllerData.xrController.pointer.rotationQuaternion!.toEulerAngles().x;\r\n const compensation = 1 + (Math.PI / 2 - Math.abs(xRotation));\r\n // check parabolic ray\r\n const radius = this.parabolicCheckRadius * compensation;\r\n this._tmpRay.origin.addToRef(this._tmpRay.direction.scale(radius * 2), this._tmpVector);\r\n this._tmpVector.y = this._tmpRay.origin.y;\r\n this._tmpRay.origin.addInPlace(this._tmpRay.direction.scale(radius));\r\n this._tmpVector.subtractToRef(this._tmpRay.origin, this._tmpRay.direction);\r\n this._tmpRay.direction.normalize();\r\n\r\n const pick = scene.pickWithRay(this._tmpRay, (o) => {\r\n // check for mesh-blockers\r\n if (this._options.pickBlockerMeshes && this._options.pickBlockerMeshes.indexOf(o) !== -1) {\r\n return true;\r\n }\r\n return this._floorMeshes.indexOf(o) !== -1;\r\n });\r\n if (pick && pick.pickedMesh && this._options.pickBlockerMeshes && this._options.pickBlockerMeshes.indexOf(pick.pickedMesh) !== -1) {\r\n controllerData.teleportationState.blocked = true;\r\n this._setTargetMeshVisibility(false);\r\n this._showParabolicPath(pick);\r\n return;\r\n } else if (pick && pick.pickedPoint) {\r\n controllerData.teleportationState.blocked = false;\r\n hitPossible = true;\r\n this._setTargetMeshPosition(pick);\r\n this._setTargetMeshVisibility(true);\r\n this._showParabolicPath(pick);\r\n }\r\n }\r\n\r\n // if needed, set visible:\r\n this._setTargetMeshVisibility(hitPossible);\r\n } else {\r\n this._setTargetMeshVisibility(false);\r\n }\r\n } else {\r\n this._disposeBezierCurve();\r\n this._setTargetMeshVisibility(false);\r\n }\r\n }\r\n\r\n private _attachController = (xrController: WebXRInputSource) => {\r\n if (this._controllers[xrController.uniqueId] || (this._options.forceHandedness && xrController.inputSource.handedness !== this._options.forceHandedness)) {\r\n // already attached\r\n return;\r\n }\r\n this._controllers[xrController.uniqueId] = {\r\n xrController,\r\n teleportationState: {\r\n forward: false,\r\n backwards: false,\r\n rotating: false,\r\n currentRotation: 0,\r\n baseRotation: 0,\r\n blocked: false,\r\n },\r\n };\r\n const controllerData = this._controllers[xrController.uniqueId];\r\n // motion controller only available to gamepad-enabled input sources.\r\n if (controllerData.xrController.inputSource.targetRayMode === \"tracked-pointer\" && controllerData.xrController.inputSource.gamepad) {\r\n // motion controller support\r\n const initMotionController = () => {\r\n if (xrController.motionController) {\r\n const movementController =\r\n xrController.motionController.getComponentOfType(WebXRControllerComponent.THUMBSTICK_TYPE) ||\r\n xrController.motionController.getComponentOfType(WebXRControllerComponent.TOUCHPAD_TYPE);\r\n if (!movementController || this._options.useMainComponentOnly) {\r\n // use trigger to move on long press\r\n const mainComponent = xrController.motionController.getMainComponent();\r\n if (!mainComponent) {\r\n return;\r\n }\r\n controllerData.teleportationComponent = mainComponent;\r\n controllerData.onButtonChangedObserver = mainComponent.onButtonStateChangedObservable.add(() => {\r\n if (!this.teleportationEnabled) {\r\n return;\r\n }\r\n // did \"pressed\" changed?\r\n if (mainComponent.changes.pressed) {\r\n if (mainComponent.changes.pressed.current) {\r\n // simulate \"forward\" thumbstick push\r\n controllerData.teleportationState.forward = true;\r\n this._currentTeleportationControllerId = controllerData.xrController.uniqueId;\r\n controllerData.teleportationState.baseRotation = this._options.xrInput.xrCamera.rotationQuaternion.toEulerAngles().y;\r\n controllerData.teleportationState.currentRotation = 0;\r\n const timeToSelect = this._options.timeToTeleport || 3000;\r\n setAndStartTimer({\r\n timeout: timeToSelect,\r\n contextObservable: this._xrSessionManager.onXRFrameObservable,\r\n breakCondition: () => !mainComponent.pressed,\r\n onEnded: () => {\r\n if (this._currentTeleportationControllerId === controllerData.xrController.uniqueId && controllerData.teleportationState.forward) {\r\n this._teleportForward(xrController.uniqueId);\r\n }\r\n },\r\n });\r\n } else {\r\n controllerData.teleportationState.forward = false;\r\n this._currentTeleportationControllerId = \"\";\r\n }\r\n }\r\n });\r\n } else {\r\n controllerData.teleportationComponent = movementController;\r\n // use thumbstick (or touchpad if thumbstick not available)\r\n controllerData.onAxisChangedObserver = movementController.onAxisValueChangedObservable.add((axesData) => {\r\n if (axesData.y <= 0.7 && controllerData.teleportationState.backwards) {\r\n controllerData.teleportationState.backwards = false;\r\n }\r\n if (axesData.y > 0.7 && !controllerData.teleportationState.forward && this.backwardsMovementEnabled && !this.snapPointsOnly) {\r\n // teleport backwards\r\n\r\n // General gist: Go Back N units, cast a ray towards the floor. If collided, move.\r\n if (!controllerData.teleportationState.backwards) {\r\n controllerData.teleportationState.backwards = true;\r\n // teleport backwards ONCE\r\n this._tmpQuaternion.copyFrom(this._options.xrInput.xrCamera.rotationQuaternion!);\r\n this._tmpQuaternion.toEulerAnglesToRef(this._tmpVector);\r\n // get only the y rotation\r\n this._tmpVector.x = 0;\r\n this._tmpVector.z = 0;\r\n // get the quaternion\r\n Quaternion.FromEulerVectorToRef(this._tmpVector, this._tmpQuaternion);\r\n this._tmpVector.set(0, 0, this.backwardsTeleportationDistance * (this._xrSessionManager.scene.useRightHandedSystem ? 1.0 : -1.0));\r\n this._tmpVector.rotateByQuaternionToRef(this._tmpQuaternion, this._tmpVector);\r\n this._tmpVector.addInPlace(this._options.xrInput.xrCamera.position);\r\n this._tmpRay.origin.copyFrom(this._tmpVector);\r\n // This will prevent the user from \"falling\" to a lower platform!\r\n // TODO - should this be a flag? 'allow falling to lower platforms'?\r\n this._tmpRay.length = this._options.xrInput.xrCamera.realWorldHeight + 0.1;\r\n // Right handed system had here \"1\" instead of -1. This is unneeded.\r\n this._tmpRay.direction.set(0, -1, 0);\r\n const pick = this._xrSessionManager.scene.pickWithRay(this._tmpRay, (o) => {\r\n return this._floorMeshes.indexOf(o) !== -1;\r\n });\r\n\r\n // pick must exist, but stay safe\r\n if (pick && pick.pickedPoint) {\r\n // Teleport the users feet to where they targeted. Ignore the Y axis.\r\n // If the \"falling to lower platforms\" feature is implemented the Y axis should be set here as well\r\n this._options.xrInput.xrCamera.position.x = pick.pickedPoint.x;\r\n this._options.xrInput.xrCamera.position.z = pick.pickedPoint.z;\r\n }\r\n }\r\n }\r\n if (axesData.y < -0.7 && !this._currentTeleportationControllerId && !controllerData.teleportationState.rotating && this.teleportationEnabled) {\r\n controllerData.teleportationState.forward = true;\r\n this._currentTeleportationControllerId = controllerData.xrController.uniqueId;\r\n controllerData.teleportationState.baseRotation = this._options.xrInput.xrCamera.rotationQuaternion.toEulerAngles().y;\r\n }\r\n if (axesData.x) {\r\n if (!controllerData.teleportationState.forward) {\r\n if (!controllerData.teleportationState.rotating && Math.abs(axesData.x) > 0.7) {\r\n // rotate in the right direction positive is right\r\n controllerData.teleportationState.rotating = true;\r\n const rotation = this.rotationAngle * (axesData.x > 0 ? 1 : -1) * (this._xrSessionManager.scene.useRightHandedSystem ? -1 : 1);\r\n Quaternion.FromEulerAngles(0, rotation, 0).multiplyToRef(\r\n this._options.xrInput.xrCamera.rotationQuaternion,\r\n this._options.xrInput.xrCamera.rotationQuaternion\r\n );\r\n }\r\n } else {\r\n if (this._currentTeleportationControllerId === controllerData.xrController.uniqueId) {\r\n // set the rotation of the forward movement\r\n if (this.rotationEnabled) {\r\n setTimeout(() => {\r\n controllerData.teleportationState.currentRotation = Math.atan2(\r\n axesData.x,\r\n axesData.y * (this._xrSessionManager.scene.useRightHandedSystem ? 1 : -1)\r\n );\r\n });\r\n } else {\r\n controllerData.teleportationState.currentRotation = 0;\r\n }\r\n }\r\n }\r\n } else {\r\n controllerData.teleportationState.rotating = false;\r\n }\r\n\r\n if (axesData.x === 0 && axesData.y === 0) {\r\n if (controllerData.teleportationState.blocked) {\r\n controllerData.teleportationState.blocked = false;\r\n this._setTargetMeshVisibility(false);\r\n }\r\n if (controllerData.teleportationState.forward) {\r\n this._teleportForward(xrController.uniqueId);\r\n }\r\n }\r\n });\r\n }\r\n }\r\n };\r\n if (xrController.motionController) {\r\n initMotionController();\r\n } else {\r\n xrController.onMotionControllerInitObservable.addOnce(() => {\r\n initMotionController();\r\n });\r\n }\r\n } else {\r\n this._xrSessionManager.scene.onPointerObservable.add((pointerInfo) => {\r\n if (pointerInfo.type === PointerEventTypes.POINTERDOWN) {\r\n controllerData.teleportationState.forward = true;\r\n this._currentTeleportationControllerId = controllerData.xrController.uniqueId;\r\n controllerData.teleportationState.baseRotation = this._options.xrInput.xrCamera.rotationQuaternion.toEulerAngles().y;\r\n controllerData.teleportationState.currentRotation = 0;\r\n const timeToSelect = this._options.timeToTeleport || 3000;\r\n setAndStartTimer({\r\n timeout: timeToSelect,\r\n contextObservable: this._xrSessionManager.onXRFrameObservable,\r\n onEnded: () => {\r\n if (this._currentTeleportationControllerId === controllerData.xrController.uniqueId && controllerData.teleportationState.forward) {\r\n this._teleportForward(xrController.uniqueId);\r\n }\r\n },\r\n });\r\n } else if (pointerInfo.type === PointerEventTypes.POINTERUP) {\r\n controllerData.teleportationState.forward = false;\r\n this._currentTeleportationControllerId = \"\";\r\n }\r\n });\r\n }\r\n };\r\n\r\n private _createDefaultTargetMesh() {\r\n // set defaults\r\n this._options.defaultTargetMeshOptions = this._options.defaultTargetMeshOptions || {};\r\n const sceneToRenderTo = this._options.useUtilityLayer\r\n ? this._options.customUtilityLayerScene || UtilityLayerRenderer.DefaultUtilityLayer.utilityLayerScene\r\n : this._xrSessionManager.scene;\r\n const teleportationTarget = CreateGround(\"teleportationTarget\", { width: 2, height: 2, subdivisions: 2 }, sceneToRenderTo);\r\n teleportationTarget.isPickable = false;\r\n\r\n if (this._options.defaultTargetMeshOptions.teleportationCircleMaterial) {\r\n teleportationTarget.material = this._options.defaultTargetMeshOptions.teleportationCircleMaterial;\r\n } else {\r\n const length = 512;\r\n const dynamicTexture = new DynamicTexture(\"teleportationPlaneDynamicTexture\", length, sceneToRenderTo, true);\r\n dynamicTexture.hasAlpha = true;\r\n const context = dynamicTexture.getContext();\r\n const centerX = length / 2;\r\n const centerY = length / 2;\r\n const radius = 200;\r\n context.beginPath();\r\n context.arc(centerX, centerY, radius, 0, 2 * Math.PI, false);\r\n context.fillStyle = this._options.defaultTargetMeshOptions.teleportationFillColor || \"#444444\";\r\n context.fill();\r\n context.lineWidth = 10;\r\n context.strokeStyle = this._options.defaultTargetMeshOptions.teleportationBorderColor || \"#FFFFFF\";\r\n context.stroke();\r\n context.closePath();\r\n dynamicTexture.update();\r\n const teleportationCircleMaterial = new StandardMaterial(\"teleportationPlaneMaterial\", sceneToRenderTo);\r\n teleportationCircleMaterial.diffuseTexture = dynamicTexture;\r\n teleportationTarget.material = teleportationCircleMaterial;\r\n }\r\n\r\n const torus = CreateTorus(\r\n \"torusTeleportation\",\r\n {\r\n diameter: 0.75,\r\n thickness: 0.1,\r\n tessellation: 20,\r\n },\r\n sceneToRenderTo\r\n );\r\n torus.isPickable = false;\r\n torus.parent = teleportationTarget;\r\n if (!this._options.defaultTargetMeshOptions.disableAnimation) {\r\n const animationInnerCircle = new Animation(\"animationInnerCircle\", \"position.y\", 30, Animation.ANIMATIONTYPE_FLOAT, Animation.ANIMATIONLOOPMODE_CYCLE);\r\n const keys = [];\r\n keys.push({\r\n frame: 0,\r\n value: 0,\r\n });\r\n keys.push({\r\n frame: 30,\r\n value: 0.4,\r\n });\r\n keys.push({\r\n frame: 60,\r\n value: 0,\r\n });\r\n animationInnerCircle.setKeys(keys);\r\n const easingFunction = new SineEase();\r\n easingFunction.setEasingMode(EasingFunction.EASINGMODE_EASEINOUT);\r\n animationInnerCircle.setEasingFunction(easingFunction);\r\n torus.animations = [];\r\n torus.animations.push(animationInnerCircle);\r\n sceneToRenderTo.beginAnimation(torus, 0, 60, true);\r\n }\r\n\r\n const cone = CreateCylinder(\"rotationCone\", { diameterTop: 0, tessellation: 4 }, sceneToRenderTo);\r\n cone.isPickable = false;\r\n cone.scaling.set(0.5, 0.12, 0.2);\r\n\r\n cone.rotate(Axis.X, Math.PI / 2);\r\n\r\n cone.position.z = 0.6;\r\n cone.parent = torus;\r\n\r\n if (this._options.defaultTargetMeshOptions.torusArrowMaterial) {\r\n torus.material = this._options.defaultTargetMeshOptions.torusArrowMaterial;\r\n cone.material = this._options.defaultTargetMeshOptions.torusArrowMaterial;\r\n } else {\r\n const torusConeMaterial = new StandardMaterial(\"torusConsMat\", sceneToRenderTo);\r\n torusConeMaterial.disableLighting = !!this._options.defaultTargetMeshOptions.disableLighting;\r\n if (torusConeMaterial.disableLighting) {\r\n torusConeMaterial.emissiveColor = new Color3(0.3, 0.3, 1.0);\r\n } else {\r\n torusConeMaterial.diffuseColor = new Color3(0.3, 0.3, 1.0);\r\n }\r\n torusConeMaterial.alpha = 0.9;\r\n torus.material = torusConeMaterial;\r\n cone.material = torusConeMaterial;\r\n this._teleportationRingMaterial = torusConeMaterial;\r\n }\r\n\r\n if (this._options.renderingGroupId !== undefined) {\r\n teleportationTarget.renderingGroupId = this._options.renderingGroupId;\r\n torus.renderingGroupId = this._options.renderingGroupId;\r\n cone.renderingGroupId = this._options.renderingGroupId;\r\n }\r\n\r\n this._options.teleportationTargetMesh = teleportationTarget;\r\n // hide the teleportation target mesh right after creating it.\r\n this._setTargetMeshVisibility(false);\r\n }\r\n\r\n private _detachController(xrControllerUniqueId: string) {\r\n const controllerData = this._controllers[xrControllerUniqueId];\r\n if (!controllerData) {\r\n return;\r\n }\r\n if (controllerData.teleportationComponent) {\r\n if (controllerData.onAxisChangedObserver) {\r\n controllerData.teleportationComponent.onAxisValueChangedObservable.remove(controllerData.onAxisChangedObserver);\r\n }\r\n if (controllerData.onButtonChangedObserver) {\r\n controllerData.teleportationComponent.onButtonStateChangedObservable.remove(controllerData.onButtonChangedObserver);\r\n }\r\n }\r\n // remove from the map\r\n delete this._controllers[xrControllerUniqueId];\r\n }\r\n\r\n private _findClosestSnapPointWithRadius(realPosition: Vector3, radius: number = this._options.snapToPositionRadius || 0.8) {\r\n let closestPoint: Nullable = null;\r\n let closestDistance = Number.MAX_VALUE;\r\n if (this._snapToPositions.length) {\r\n const radiusSquared = radius * radius;\r\n this._snapToPositions.forEach((position) => {\r\n const dist = Vector3.DistanceSquared(position, realPosition);\r\n if (dist <= radiusSquared && dist < closestDistance) {\r\n closestDistance = dist;\r\n closestPoint = position;\r\n }\r\n });\r\n }\r\n return closestPoint;\r\n }\r\n\r\n private _setTargetMeshPosition(pickInfo: PickingInfo) {\r\n const newPosition = pickInfo.pickedPoint;\r\n if (!this._options.teleportationTargetMesh || !newPosition) {\r\n return;\r\n }\r\n const snapPosition = this._findClosestSnapPointWithRadius(newPosition);\r\n this._snappedToPoint = !!snapPosition;\r\n if (this.snapPointsOnly && !this._snappedToPoint && this._teleportationRingMaterial) {\r\n this._teleportationRingMaterial.diffuseColor.set(1.0, 0.3, 0.3);\r\n } else if (this.snapPointsOnly && this._snappedToPoint && this._teleportationRingMaterial) {\r\n this._teleportationRingMaterial.diffuseColor.set(0.3, 0.3, 1.0);\r\n }\r\n this._options.teleportationTargetMesh.position.copyFrom(snapPosition || newPosition);\r\n this._options.teleportationTargetMesh.position.y += 0.01;\r\n this.onTargetMeshPositionUpdatedObservable.notifyObservers(pickInfo);\r\n }\r\n\r\n private _setTargetMeshVisibility(visible: boolean, force?: boolean) {\r\n if (!this._options.teleportationTargetMesh) {\r\n return;\r\n }\r\n if (this._options.teleportationTargetMesh.isVisible === visible && !force) {\r\n return;\r\n }\r\n this._options.teleportationTargetMesh.isVisible = visible;\r\n this._options.teleportationTargetMesh.getChildren(undefined, false).forEach((m) => {\r\n (m).isVisible = visible;\r\n });\r\n\r\n if (!visible) {\r\n if (this._quadraticBezierCurve) {\r\n this._quadraticBezierCurve.dispose();\r\n this._quadraticBezierCurve = null;\r\n }\r\n if (this._selectionFeature) {\r\n this._selectionFeature.attach();\r\n }\r\n } else {\r\n if (this._selectionFeature) {\r\n this._selectionFeature.detach();\r\n }\r\n }\r\n }\r\n\r\n private _disposeBezierCurve() {\r\n if (this._quadraticBezierCurve) {\r\n this._quadraticBezierCurve.dispose();\r\n this._quadraticBezierCurve = null;\r\n }\r\n }\r\n\r\n private _showParabolicPath(pickInfo: PickingInfo) {\r\n if (!pickInfo.pickedPoint || !this._currentTeleportationControllerId) {\r\n return;\r\n }\r\n\r\n const sceneToRenderTo = this._options.useUtilityLayer\r\n ? this._options.customUtilityLayerScene || UtilityLayerRenderer.DefaultUtilityLayer.utilityLayerScene\r\n : this._xrSessionManager.scene;\r\n\r\n const controllerData = this._controllers[this._currentTeleportationControllerId];\r\n\r\n const quadraticBezierVectors = Curve3.CreateQuadraticBezier(controllerData.xrController.pointer.absolutePosition, pickInfo.ray!.origin, pickInfo.pickedPoint, 25);\r\n const color = controllerData.teleportationState.blocked ? this._blockedRayColor : undefined;\r\n const colorsArray = new Array(26).fill(color || this._cachedColor4White);\r\n if (!this._options.generateRayPathMesh) {\r\n this._quadraticBezierCurve = CreateLines(\r\n \"teleportation path line\",\r\n { points: quadraticBezierVectors.getPoints(), instance: this._quadraticBezierCurve as LinesMesh, updatable: true, colors: colorsArray },\r\n sceneToRenderTo\r\n );\r\n } else {\r\n this._quadraticBezierCurve = this._options.generateRayPathMesh(quadraticBezierVectors.getPoints(), pickInfo);\r\n }\r\n this._quadraticBezierCurve.isPickable = false;\r\n if (this._options.renderingGroupId !== undefined) {\r\n this._quadraticBezierCurve.renderingGroupId = this._options.renderingGroupId;\r\n }\r\n }\r\n\r\n private _teleportForward(controllerId: string) {\r\n const controllerData = this._controllers[controllerId];\r\n if (!controllerData || !controllerData.teleportationState.forward || !this.teleportationEnabled) {\r\n return;\r\n }\r\n controllerData.teleportationState.forward = false;\r\n this._currentTeleportationControllerId = \"\";\r\n if (this.snapPointsOnly && !this._snappedToPoint) {\r\n return;\r\n }\r\n\r\n if (this.skipNextTeleportation) {\r\n this.skipNextTeleportation = false;\r\n return;\r\n }\r\n // do the movement forward here\r\n if (this._options.teleportationTargetMesh && this._options.teleportationTargetMesh.isVisible) {\r\n const height = this._options.xrInput.xrCamera.realWorldHeight;\r\n this._options.xrInput.xrCamera.onBeforeCameraTeleport.notifyObservers(this._options.xrInput.xrCamera.position);\r\n this._options.xrInput.xrCamera.position.copyFrom(this._options.teleportationTargetMesh.position);\r\n this._options.xrInput.xrCamera.position.y += height;\r\n Quaternion.FromEulerAngles(0, controllerData.teleportationState.currentRotation - (this._xrSessionManager.scene.useRightHandedSystem ? Math.PI : 0), 0).multiplyToRef(\r\n this._options.xrInput.xrCamera.rotationQuaternion,\r\n this._options.xrInput.xrCamera.rotationQuaternion\r\n );\r\n this._options.xrInput.xrCamera.onAfterCameraTeleport.notifyObservers(this._options.xrInput.xrCamera.position);\r\n }\r\n }\r\n}\r\n\r\nWebXRFeaturesManager.AddWebXRFeature(\r\n WebXRMotionControllerTeleportation.Name,\r\n (xrSessionManager, options) => {\r\n return () => new WebXRMotionControllerTeleportation(xrSessionManager, options);\r\n },\r\n WebXRMotionControllerTeleportation.Version,\r\n true\r\n);\r\n","import { WebXRExperienceHelper } from \"./webXRExperienceHelper\";\r\nimport type { Scene } from \"../scene\";\r\nimport type { IWebXRInputOptions } from \"./webXRInput\";\r\nimport { WebXRInput } from \"./webXRInput\";\r\nimport type { IWebXRControllerPointerSelectionOptions } from \"./features/WebXRControllerPointerSelection\";\r\nimport { WebXRControllerPointerSelection } from \"./features/WebXRControllerPointerSelection\";\r\nimport type { IWebXRNearInteractionOptions } from \"./features/WebXRNearInteraction\";\r\nimport { WebXRNearInteraction } from \"./features/WebXRNearInteraction\";\r\nimport type { WebXRRenderTarget } from \"./webXRTypes\";\r\nimport type { WebXREnterExitUIOptions } from \"./webXREnterExitUI\";\r\nimport { WebXREnterExitUI } from \"./webXREnterExitUI\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { WebXRManagedOutputCanvasOptions } from \"./webXRManagedOutputCanvas\";\r\nimport type { IWebXRTeleportationOptions } from \"./features/WebXRControllerTeleportation\";\r\nimport { WebXRMotionControllerTeleportation } from \"./features/WebXRControllerTeleportation\";\r\nimport { Logger } from \"../Misc/logger\";\r\n\r\n/**\r\n * Options for the default xr helper\r\n */\r\nexport class WebXRDefaultExperienceOptions {\r\n /**\r\n * Enable or disable default UI to enter XR\r\n */\r\n public disableDefaultUI?: boolean;\r\n /**\r\n * Should pointer selection not initialize.\r\n * Note that disabling pointer selection also disables teleportation.\r\n * Defaults to false.\r\n */\r\n public disablePointerSelection?: boolean;\r\n /**\r\n * Should teleportation not initialize. Defaults to false.\r\n */\r\n public disableTeleportation?: boolean;\r\n /**\r\n * Should nearInteraction not initialize. Defaults to false.\r\n */\r\n public disableNearInteraction?: boolean;\r\n /**\r\n * Floor meshes that will be used for teleport\r\n */\r\n public floorMeshes?: Array;\r\n /**\r\n * If set to true, the first frame will not be used to reset position\r\n * The first frame is mainly used when copying transformation from the old camera\r\n * Mainly used in AR\r\n */\r\n public ignoreNativeCameraTransformation?: boolean;\r\n /**\r\n * Optional configuration for the XR input object\r\n */\r\n public inputOptions?: Partial;\r\n /**\r\n * optional configuration for pointer selection\r\n */\r\n public pointerSelectionOptions?: Partial;\r\n /**\r\n * optional configuration for near interaction\r\n */\r\n public nearInteractionOptions?: Partial;\r\n /**\r\n * optional configuration for teleportation\r\n */\r\n public teleportationOptions?: Partial;\r\n /**\r\n * optional configuration for the output canvas\r\n */\r\n public outputCanvasOptions?: WebXRManagedOutputCanvasOptions;\r\n /**\r\n * optional UI options. This can be used among other to change session mode and reference space type\r\n */\r\n public uiOptions?: Partial;\r\n /**\r\n * When loading teleportation and pointer select, use stable versions instead of latest.\r\n */\r\n public useStablePlugins?: boolean;\r\n\r\n /**\r\n * An optional rendering group id that will be set globally for teleportation, pointer selection and default controller meshes\r\n */\r\n public renderingGroupId?: number;\r\n\r\n /**\r\n * A list of optional features to init the session with\r\n * If set to true, all features we support will be added\r\n */\r\n public optionalFeatures?: boolean | string[];\r\n}\r\n\r\n/**\r\n * Default experience which provides a similar setup to the previous webVRExperience\r\n */\r\nexport class WebXRDefaultExperience {\r\n /**\r\n * Base experience\r\n */\r\n public baseExperience: WebXRExperienceHelper;\r\n /**\r\n * Enables ui for entering/exiting xr\r\n */\r\n public enterExitUI: WebXREnterExitUI;\r\n /**\r\n * Input experience extension\r\n */\r\n public input: WebXRInput;\r\n /**\r\n * Enables laser pointer and selection\r\n */\r\n public pointerSelection: WebXRControllerPointerSelection;\r\n /**\r\n * Default target xr should render to\r\n */\r\n public renderTarget: WebXRRenderTarget;\r\n /**\r\n * Enables teleportation\r\n */\r\n public teleportation: WebXRMotionControllerTeleportation;\r\n\r\n /**\r\n * Enables near interaction for hands/controllers\r\n */\r\n public nearInteraction: WebXRNearInteraction;\r\n\r\n private constructor() {}\r\n\r\n /**\r\n * Creates the default xr experience\r\n * @param scene scene\r\n * @param options options for basic configuration\r\n * @returns resulting WebXRDefaultExperience\r\n */\r\n public static CreateAsync(scene: Scene, options: WebXRDefaultExperienceOptions = {}) {\r\n const result = new WebXRDefaultExperience();\r\n scene.onDisposeObservable.addOnce(() => {\r\n result.dispose();\r\n });\r\n // init the UI right after construction\r\n if (!options.disableDefaultUI) {\r\n const uiOptions: WebXREnterExitUIOptions = {\r\n renderTarget: result.renderTarget,\r\n ...(options.uiOptions || {}),\r\n };\r\n if (options.optionalFeatures) {\r\n if (typeof options.optionalFeatures === \"boolean\") {\r\n uiOptions.optionalFeatures = [\"hit-test\", \"anchors\", \"plane-detection\", \"hand-tracking\"];\r\n } else {\r\n uiOptions.optionalFeatures = options.optionalFeatures;\r\n }\r\n }\r\n result.enterExitUI = new WebXREnterExitUI(scene, uiOptions);\r\n }\r\n\r\n // Create base experience\r\n return WebXRExperienceHelper.CreateAsync(scene)\r\n .then((xrHelper) => {\r\n result.baseExperience = xrHelper;\r\n\r\n if (options.ignoreNativeCameraTransformation) {\r\n result.baseExperience.camera.compensateOnFirstFrame = false;\r\n }\r\n\r\n // Add controller support\r\n result.input = new WebXRInput(xrHelper.sessionManager, xrHelper.camera, {\r\n controllerOptions: {\r\n renderingGroupId: options.renderingGroupId,\r\n },\r\n ...(options.inputOptions || {}),\r\n });\r\n\r\n if (!options.disablePointerSelection) {\r\n // Add default pointer selection\r\n const pointerSelectionOptions = {\r\n ...options.pointerSelectionOptions,\r\n xrInput: result.input,\r\n renderingGroupId: options.renderingGroupId,\r\n };\r\n\r\n result.pointerSelection = (\r\n result.baseExperience.featuresManager.enableFeature(\r\n WebXRControllerPointerSelection.Name,\r\n options.useStablePlugins ? \"stable\" : \"latest\",\r\n pointerSelectionOptions\r\n )\r\n );\r\n\r\n if (!options.disableTeleportation) {\r\n // Add default teleportation, including rotation\r\n result.teleportation = result.baseExperience.featuresManager.enableFeature(\r\n WebXRMotionControllerTeleportation.Name,\r\n options.useStablePlugins ? \"stable\" : \"latest\",\r\n {\r\n floorMeshes: options.floorMeshes,\r\n xrInput: result.input,\r\n renderingGroupId: options.renderingGroupId,\r\n ...options.teleportationOptions,\r\n }\r\n );\r\n result.teleportation.setSelectionFeature(result.pointerSelection);\r\n }\r\n }\r\n\r\n if (!options.disableNearInteraction) {\r\n // Add default pointer selection\r\n result.nearInteraction = result.baseExperience.featuresManager.enableFeature(\r\n WebXRNearInteraction.Name,\r\n options.useStablePlugins ? \"stable\" : \"latest\",\r\n {\r\n xrInput: result.input,\r\n farInteractionFeature: result.pointerSelection,\r\n renderingGroupId: options.renderingGroupId,\r\n useUtilityLayer: true,\r\n enableNearInteractionOnAllControllers: true,\r\n ...options.nearInteractionOptions,\r\n }\r\n );\r\n }\r\n\r\n // Create the WebXR output target\r\n result.renderTarget = result.baseExperience.sessionManager.getWebXRRenderTarget(options.outputCanvasOptions);\r\n\r\n if (!options.disableDefaultUI) {\r\n // Create ui for entering/exiting xr\r\n return result.enterExitUI.setHelperAsync(result.baseExperience, result.renderTarget);\r\n } else {\r\n return;\r\n }\r\n })\r\n .then(() => {\r\n return result;\r\n })\r\n .catch((error) => {\r\n Logger.Error(\"Error initializing XR\");\r\n Logger.Error(error);\r\n return result;\r\n });\r\n }\r\n\r\n /**\r\n * Disposes of the experience helper\r\n */\r\n public dispose() {\r\n if (this.baseExperience) {\r\n this.baseExperience.dispose();\r\n }\r\n if (this.input) {\r\n this.input.dispose();\r\n }\r\n if (this.enterExitUI) {\r\n this.enterExitUI.dispose();\r\n }\r\n if (this.renderTarget) {\r\n this.renderTarget.dispose();\r\n }\r\n }\r\n}\r\n","import { Logger } from \"../Misc/logger\";\r\nimport type { Nullable } from \"../types\";\r\nimport { Scene } from \"../scene\";\r\nimport { Vector3 } from \"../Maths/math.vector\";\r\nimport type { Mesh } from \"../Meshes/mesh\";\r\nimport type { BaseTexture } from \"../Materials/Textures/baseTexture\";\r\nimport { Texture } from \"../Materials/Textures/texture\";\r\nimport { StandardMaterial } from \"../Materials/standardMaterial\";\r\nimport { PBRMaterial } from \"../Materials/PBR/pbrMaterial\";\r\nimport { HemisphericLight } from \"../Lights/hemisphericLight\";\r\nimport type { IEnvironmentHelperOptions } from \"./environmentHelper\";\r\nimport { EnvironmentHelper } from \"./environmentHelper\";\r\nimport { FreeCamera } from \"../Cameras/freeCamera\";\r\nimport { ArcRotateCamera } from \"../Cameras/arcRotateCamera\";\r\nimport type { TargetCamera } from \"../Cameras/targetCamera\";\r\nimport type { VRExperienceHelperOptions } from \"../Cameras/VR/vrExperienceHelper\";\r\nimport { VRExperienceHelper } from \"../Cameras/VR/vrExperienceHelper\";\r\n\r\nimport \"../Materials/Textures/Loaders/ddsTextureLoader\";\r\nimport \"../Materials/Textures/Loaders/envTextureLoader\";\r\nimport \"../Materials/Textures/Loaders/ktxTextureLoader\";\r\nimport { CreateBox } from \"../Meshes/Builders/boxBuilder\";\r\nimport type { WebXRDefaultExperienceOptions } from \"../XR/webXRDefaultExperience\";\r\nimport { WebXRDefaultExperience } from \"../XR/webXRDefaultExperience\";\r\n\r\n/** @internal */\r\n// eslint-disable-next-line no-var\r\nexport var _forceSceneHelpersToBundle = true;\r\n\r\ndeclare module \"../scene\" {\r\n export interface Scene {\r\n /**\r\n * Creates a default light for the scene.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/fastBuildWorld#create-default-light\r\n * @param replace has the default false, when true replaces the existing lights in the scene with a hemispheric light\r\n */\r\n createDefaultLight(replace?: boolean): void;\r\n\r\n /**\r\n * Creates a default camera for the scene.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/fastBuildWorld#create-default-camera\r\n * @param createArcRotateCamera has the default false which creates a free camera, when true creates an arc rotate camera\r\n * @param replace has default false, when true replaces the active camera in the scene\r\n * @param attachCameraControls has default false, when true attaches camera controls to the canvas.\r\n */\r\n createDefaultCamera(createArcRotateCamera?: boolean, replace?: boolean, attachCameraControls?: boolean): void;\r\n\r\n /**\r\n * Creates a default camera and a default light.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/fastBuildWorld#create-default-camera-or-light\r\n * @param createArcRotateCamera has the default false which creates a free camera, when true creates an arc rotate camera\r\n * @param replace has the default false, when true replaces the active camera/light in the scene\r\n * @param attachCameraControls has the default false, when true attaches camera controls to the canvas.\r\n */\r\n createDefaultCameraOrLight(createArcRotateCamera?: boolean, replace?: boolean, attachCameraControls?: boolean): void;\r\n\r\n /**\r\n * Creates a new sky box\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/fastBuildWorld#create-default-skybox\r\n * @param environmentTexture defines the texture to use as environment texture\r\n * @param pbr has default false which requires the StandardMaterial to be used, when true PBRMaterial must be used\r\n * @param scale defines the overall scale of the skybox\r\n * @param blur is only available when pbr is true, default is 0, no blur, maximum value is 1\r\n * @param setGlobalEnvTexture has default true indicating that scene.environmentTexture must match the current skybox texture\r\n * @returns a new mesh holding the sky box\r\n */\r\n createDefaultSkybox(environmentTexture?: BaseTexture, pbr?: boolean, scale?: number, blur?: number, setGlobalEnvTexture?: boolean): Nullable;\r\n\r\n /**\r\n * Creates a new environment\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/fastBuildWorld#create-default-environment\r\n * @param options defines the options you can use to configure the environment\r\n * @returns the new EnvironmentHelper\r\n */\r\n createDefaultEnvironment(options?: Partial): Nullable;\r\n\r\n /**\r\n * Creates a new VREXperienceHelper\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/cameras/webVRHelper\r\n * @param webVROptions defines the options used to create the new VREXperienceHelper\r\n * @deprecated Please use createDefaultXRExperienceAsync instead\r\n * @returns a new VREXperienceHelper\r\n */\r\n createDefaultVRExperience(webVROptions?: VRExperienceHelperOptions): VRExperienceHelper;\r\n\r\n /**\r\n * Creates a new WebXRDefaultExperience\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/webXR/introToWebXR\r\n * @param options experience options\r\n * @returns a promise for a new WebXRDefaultExperience\r\n */\r\n createDefaultXRExperienceAsync(options?: WebXRDefaultExperienceOptions): Promise;\r\n }\r\n}\r\n\r\nScene.prototype.createDefaultLight = function (replace = false): void {\r\n // Dispose existing light in replace mode.\r\n if (replace) {\r\n if (this.lights) {\r\n for (let i = 0; i < this.lights.length; i++) {\r\n this.lights[i].dispose();\r\n }\r\n }\r\n }\r\n\r\n // Light\r\n if (this.lights.length === 0) {\r\n new HemisphericLight(\"default light\", Vector3.Up(), this);\r\n }\r\n};\r\n\r\nScene.prototype.createDefaultCamera = function (createArcRotateCamera = false, replace = false, attachCameraControls = false): void {\r\n // Dispose existing camera in replace mode.\r\n if (replace) {\r\n if (this.activeCamera) {\r\n this.activeCamera.dispose();\r\n this.activeCamera = null;\r\n }\r\n }\r\n\r\n // Camera\r\n if (!this.activeCamera) {\r\n const worldExtends = this.getWorldExtends((mesh) => mesh.isVisible && mesh.isEnabled());\r\n const worldSize = worldExtends.max.subtract(worldExtends.min);\r\n const worldCenter = worldExtends.min.add(worldSize.scale(0.5));\r\n\r\n let camera: TargetCamera;\r\n let radius = worldSize.length() * 1.5;\r\n // empty scene scenario!\r\n if (!isFinite(radius)) {\r\n radius = 1;\r\n worldCenter.copyFromFloats(0, 0, 0);\r\n }\r\n if (createArcRotateCamera) {\r\n const arcRotateCamera = new ArcRotateCamera(\"default camera\", -(Math.PI / 2), Math.PI / 2, radius, worldCenter, this);\r\n arcRotateCamera.lowerRadiusLimit = radius * 0.01;\r\n arcRotateCamera.wheelPrecision = 100 / radius;\r\n camera = arcRotateCamera;\r\n } else {\r\n const freeCamera = new FreeCamera(\"default camera\", new Vector3(worldCenter.x, worldCenter.y, -radius), this);\r\n freeCamera.setTarget(worldCenter);\r\n camera = freeCamera;\r\n }\r\n camera.minZ = radius * 0.01;\r\n camera.maxZ = radius * 1000;\r\n camera.speed = radius * 0.2;\r\n this.activeCamera = camera;\r\n\r\n if (attachCameraControls) {\r\n camera.attachControl();\r\n }\r\n }\r\n};\r\n\r\nScene.prototype.createDefaultCameraOrLight = function (createArcRotateCamera = false, replace = false, attachCameraControls = false): void {\r\n this.createDefaultLight(replace);\r\n this.createDefaultCamera(createArcRotateCamera, replace, attachCameraControls);\r\n};\r\n\r\nScene.prototype.createDefaultSkybox = function (environmentTexture?: BaseTexture, pbr = false, scale = 1000, blur = 0, setGlobalEnvTexture = true): Nullable {\r\n if (!environmentTexture) {\r\n Logger.Warn(\"Can not create default skybox without environment texture.\");\r\n return null;\r\n }\r\n\r\n if (setGlobalEnvTexture) {\r\n if (environmentTexture) {\r\n this.environmentTexture = environmentTexture;\r\n }\r\n }\r\n\r\n // Skybox\r\n const hdrSkybox = CreateBox(\"hdrSkyBox\", { size: scale }, this);\r\n if (pbr) {\r\n const hdrSkyboxMaterial = new PBRMaterial(\"skyBox\", this);\r\n hdrSkyboxMaterial.backFaceCulling = false;\r\n hdrSkyboxMaterial.reflectionTexture = environmentTexture.clone();\r\n if (hdrSkyboxMaterial.reflectionTexture) {\r\n hdrSkyboxMaterial.reflectionTexture.coordinatesMode = Texture.SKYBOX_MODE;\r\n }\r\n hdrSkyboxMaterial.microSurface = 1.0 - blur;\r\n hdrSkyboxMaterial.disableLighting = true;\r\n hdrSkyboxMaterial.twoSidedLighting = true;\r\n hdrSkybox.material = hdrSkyboxMaterial;\r\n } else {\r\n const skyboxMaterial = new StandardMaterial(\"skyBox\", this);\r\n skyboxMaterial.backFaceCulling = false;\r\n skyboxMaterial.reflectionTexture = environmentTexture.clone();\r\n if (skyboxMaterial.reflectionTexture) {\r\n skyboxMaterial.reflectionTexture.coordinatesMode = Texture.SKYBOX_MODE;\r\n }\r\n skyboxMaterial.disableLighting = true;\r\n hdrSkybox.material = skyboxMaterial;\r\n }\r\n hdrSkybox.isPickable = false;\r\n hdrSkybox.infiniteDistance = true;\r\n hdrSkybox.ignoreCameraMaxZ = true;\r\n return hdrSkybox;\r\n};\r\n\r\nScene.prototype.createDefaultEnvironment = function (options: Partial): Nullable {\r\n if (EnvironmentHelper) {\r\n return new EnvironmentHelper(options, this);\r\n }\r\n return null;\r\n};\r\n\r\nScene.prototype.createDefaultVRExperience = function (webVROptions: VRExperienceHelperOptions = {}): VRExperienceHelper {\r\n return new VRExperienceHelper(this, webVROptions);\r\n};\r\n\r\nScene.prototype.createDefaultXRExperienceAsync = function (options: WebXRDefaultExperienceOptions = {}): Promise {\r\n return WebXRDefaultExperience.CreateAsync(this, options).then((helper) => {\r\n return helper;\r\n });\r\n};\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/helperFunctions\";\nimport \"./ShadersInclude/clipPlaneFragmentDeclaration\";\nimport \"./ShadersInclude/clipPlaneFragment\";\n\nconst name = \"glowMapGenerationPixelShader\";\nconst shader = `#if defined(DIFFUSE_ISLINEAR) || defined(EMISSIVE_ISLINEAR)\n#include\n#endif\n#ifdef DIFFUSE\nvarying vec2 vUVDiffuse;uniform sampler2D diffuseSampler;\n#endif\n#ifdef OPACITY\nvarying vec2 vUVOpacity;uniform sampler2D opacitySampler;uniform float opacityIntensity;\n#endif\n#ifdef EMISSIVE\nvarying vec2 vUVEmissive;uniform sampler2D emissiveSampler;\n#endif\n#ifdef VERTEXALPHA\nvarying vec4 vColor;\n#endif\nuniform vec4 glowColor;uniform float glowIntensity;\n#include\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{\n#include\nvec4 finalColor=glowColor;\n#ifdef DIFFUSE\nvec4 albedoTexture=texture2D(diffuseSampler,vUVDiffuse);\n#ifdef DIFFUSE_ISLINEAR\nalbedoTexture=toGammaSpace(albedoTexture);\n#endif\n#ifdef GLOW\nfinalColor.a*=albedoTexture.a;\n#endif\n#ifdef HIGHLIGHT\nfinalColor.a=albedoTexture.a;\n#endif\n#endif\n#ifdef OPACITY\nvec4 opacityMap=texture2D(opacitySampler,vUVOpacity);\n#ifdef OPACITYRGB\nfinalColor.a*=getLuminance(opacityMap.rgb);\n#else\nfinalColor.a*=opacityMap.a;\n#endif\nfinalColor.a*=opacityIntensity;\n#endif\n#ifdef VERTEXALPHA\nfinalColor.a*=vColor.a;\n#endif\n#ifdef ALPHATEST\nif (finalColor.a\n#include\n#include\n#include[0..maxSimultaneousMorphTargets]\n#include\n#include\nuniform mat4 viewProjection;varying vec4 vPosition;\n#ifdef UV1\nattribute vec2 uv;\n#endif\n#ifdef UV2\nattribute vec2 uv2;\n#endif\n#ifdef DIFFUSE\nvarying vec2 vUVDiffuse;uniform mat4 diffuseMatrix;\n#endif\n#ifdef OPACITY\nvarying vec2 vUVOpacity;uniform mat4 opacityMatrix;\n#endif\n#ifdef EMISSIVE\nvarying vec2 vUVEmissive;uniform mat4 emissiveMatrix;\n#endif\n#ifdef VERTEXALPHA\nattribute vec4 color;varying vec4 vColor;\n#endif\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void)\n{vec3 positionUpdated=position;\n#ifdef UV1\nvec2 uvUpdated=uv;\n#endif\n#include\n#include[0..maxSimultaneousMorphTargets]\n#include\n#include\n#include\nvec4 worldPos=finalWorld*vec4(positionUpdated,1.0);\n#ifdef CUBEMAP\nvPosition=worldPos;gl_Position=viewProjection*finalWorld*vec4(position,1.0);\n#else\nvPosition=viewProjection*worldPos;gl_Position=vPosition;\n#endif\n#ifdef DIFFUSE\n#ifdef DIFFUSEUV1\nvUVDiffuse=vec2(diffuseMatrix*vec4(uvUpdated,1.0,0.0));\n#endif\n#ifdef DIFFUSEUV2\nvUVDiffuse=vec2(diffuseMatrix*vec4(uv2,1.0,0.0));\n#endif\n#endif\n#ifdef OPACITY\n#ifdef OPACITYUV1\nvUVOpacity=vec2(opacityMatrix*vec4(uvUpdated,1.0,0.0));\n#endif\n#ifdef OPACITYUV2\nvUVOpacity=vec2(opacityMatrix*vec4(uv2,1.0,0.0));\n#endif\n#endif\n#ifdef EMISSIVE\n#ifdef EMISSIVEUV1\nvUVEmissive=vec2(emissiveMatrix*vec4(uvUpdated,1.0,0.0));\n#endif\n#ifdef EMISSIVEUV2\nvUVEmissive=vec2(emissiveMatrix*vec4(uv2,1.0,0.0));\n#endif\n#endif\n#ifdef VERTEXALPHA\nvColor=color;\n#endif\n#include\n}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const glowMapGenerationVertexShader = { name, shader };\n","import { serialize, serializeAsColor4, serializeAsCameraReference } from \"../Misc/decorators\";\r\nimport { Tools } from \"../Misc/tools\";\r\nimport type { SmartArray } from \"../Misc/smartArray\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\nimport type { Scene } from \"../scene\";\r\nimport type { ISize } from \"../Maths/math.size\";\r\nimport { Color4 } from \"../Maths/math.color\";\r\nimport { Engine } from \"../Engines/engine\";\r\nimport { EngineStore } from \"../Engines/engineStore\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport type { SubMesh } from \"../Meshes/subMesh\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { Mesh } from \"../Meshes/mesh\";\r\nimport type { PostProcess } from \"../PostProcesses/postProcess\";\r\nimport type { BaseTexture } from \"../Materials/Textures/baseTexture\";\r\nimport { Texture } from \"../Materials/Textures/texture\";\r\nimport { RenderTargetTexture } from \"../Materials/Textures/renderTargetTexture\";\r\nimport type { Effect } from \"../Materials/effect\";\r\nimport { Material } from \"../Materials/material\";\r\nimport { MaterialHelper } from \"../Materials/materialHelper\";\r\nimport { Constants } from \"../Engines/constants\";\r\n\r\nimport \"../Shaders/glowMapGeneration.fragment\";\r\nimport \"../Shaders/glowMapGeneration.vertex\";\r\nimport { _WarnImport } from \"../Misc/devTools\";\r\nimport type { DataBuffer } from \"../Buffers/dataBuffer\";\r\nimport { EffectFallbacks } from \"../Materials/effectFallbacks\";\r\nimport { DrawWrapper } from \"../Materials/drawWrapper\";\r\nimport { addClipPlaneUniforms, bindClipPlane, prepareStringDefinesForClipPlanes } from \"../Materials/clipPlaneMaterialHelper\";\r\n\r\n/**\r\n * Effect layer options. This helps customizing the behaviour\r\n * of the effect layer.\r\n */\r\nexport interface IEffectLayerOptions {\r\n /**\r\n * Multiplication factor apply to the canvas size to compute the render target size\r\n * used to generated the objects (the smaller the faster). Default: 0.5\r\n */\r\n mainTextureRatio: number;\r\n\r\n /**\r\n * Enforces a fixed size texture to ensure effect stability across devices. Default: undefined\r\n */\r\n mainTextureFixedSize?: number;\r\n\r\n /**\r\n * Alpha blending mode used to apply the blur. Default depends of the implementation. Default: ALPHA_COMBINE\r\n */\r\n alphaBlendingMode: number;\r\n\r\n /**\r\n * The camera attached to the layer. Default: null\r\n */\r\n camera: Nullable;\r\n\r\n /**\r\n * The rendering group to draw the layer in. Default: -1\r\n */\r\n renderingGroupId: number;\r\n\r\n /**\r\n * The type of the main texture. Default: TEXTURETYPE_UNSIGNED_INT\r\n */\r\n mainTextureType: number;\r\n\r\n /**\r\n * Whether or not to generate a stencil buffer. Default: false\r\n */\r\n generateStencilBuffer: boolean;\r\n}\r\n\r\n/**\r\n * The effect layer Helps adding post process effect blended with the main pass.\r\n *\r\n * This can be for instance use to generate glow or highlight effects on the scene.\r\n *\r\n * The effect layer class can not be used directly and is intented to inherited from to be\r\n * customized per effects.\r\n */\r\nexport abstract class EffectLayer {\r\n private _vertexBuffers: { [key: string]: Nullable } = {};\r\n private _indexBuffer: Nullable;\r\n private _effectLayerOptions: IEffectLayerOptions;\r\n private _mergeDrawWrapper: DrawWrapper[];\r\n\r\n protected _scene: Scene;\r\n protected _engine: Engine;\r\n protected _maxSize: number = 0;\r\n protected _mainTextureDesiredSize: ISize = { width: 0, height: 0 };\r\n protected _mainTexture: RenderTargetTexture;\r\n protected _shouldRender = true;\r\n protected _postProcesses: PostProcess[] = [];\r\n protected _textures: BaseTexture[] = [];\r\n protected _emissiveTextureAndColor: { texture: Nullable; color: Color4 } = { texture: null, color: new Color4() };\r\n protected _effectIntensity: { [meshUniqueId: number]: number } = {};\r\n\r\n /**\r\n * The name of the layer\r\n */\r\n @serialize()\r\n public name: string;\r\n\r\n /**\r\n * The clear color of the texture used to generate the glow map.\r\n */\r\n @serializeAsColor4()\r\n public neutralColor: Color4 = new Color4();\r\n\r\n /**\r\n * Specifies whether the highlight layer is enabled or not.\r\n */\r\n @serialize()\r\n public isEnabled: boolean = true;\r\n\r\n /**\r\n * Gets the camera attached to the layer.\r\n */\r\n @serializeAsCameraReference()\r\n public get camera(): Nullable {\r\n return this._effectLayerOptions.camera;\r\n }\r\n\r\n /**\r\n * Gets the rendering group id the layer should render in.\r\n */\r\n @serialize()\r\n public get renderingGroupId(): number {\r\n return this._effectLayerOptions.renderingGroupId;\r\n }\r\n public set renderingGroupId(renderingGroupId: number) {\r\n this._effectLayerOptions.renderingGroupId = renderingGroupId;\r\n }\r\n\r\n /**\r\n * Specifies if the bounding boxes should be rendered normally or if they should undergo the effect of the layer\r\n */\r\n @serialize()\r\n public disableBoundingBoxesFromEffectLayer = false;\r\n\r\n /**\r\n * An event triggered when the effect layer has been disposed.\r\n */\r\n public onDisposeObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when the effect layer is about rendering the main texture with the glowy parts.\r\n */\r\n public onBeforeRenderMainTextureObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when the generated texture is being merged in the scene.\r\n */\r\n public onBeforeComposeObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when the mesh is rendered into the effect render target.\r\n */\r\n public onBeforeRenderMeshToEffect = new Observable();\r\n\r\n /**\r\n * An event triggered after the mesh has been rendered into the effect render target.\r\n */\r\n public onAfterRenderMeshToEffect = new Observable();\r\n\r\n /**\r\n * An event triggered when the generated texture has been merged in the scene.\r\n */\r\n public onAfterComposeObservable = new Observable();\r\n\r\n /**\r\n * An event triggered when the effect layer changes its size.\r\n */\r\n public onSizeChangedObservable = new Observable();\r\n\r\n /**\r\n * Gets the main texture where the effect is rendered\r\n */\r\n public get mainTexture() {\r\n return this._mainTexture;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _SceneComponentInitialization: (scene: Scene) => void = (_) => {\r\n throw _WarnImport(\"EffectLayerSceneComponent\");\r\n };\r\n\r\n private _materialForRendering: { [id: string]: [AbstractMesh, Material] } = {};\r\n\r\n /**\r\n * Sets a specific material to be used to render a mesh/a list of meshes in the layer\r\n * @param mesh mesh or array of meshes\r\n * @param material material to use by the layer when rendering the mesh(es). If undefined is passed, the specific material created by the layer will be used.\r\n */\r\n public setMaterialForRendering(mesh: AbstractMesh | AbstractMesh[], material?: Material): void {\r\n this._mainTexture.setMaterialForRendering(mesh, material);\r\n if (Array.isArray(mesh)) {\r\n for (let i = 0; i < mesh.length; ++i) {\r\n const currentMesh = mesh[i];\r\n if (!material) {\r\n delete this._materialForRendering[currentMesh.uniqueId];\r\n } else {\r\n this._materialForRendering[currentMesh.uniqueId] = [currentMesh, material];\r\n }\r\n }\r\n } else {\r\n if (!material) {\r\n delete this._materialForRendering[mesh.uniqueId];\r\n } else {\r\n this._materialForRendering[mesh.uniqueId] = [mesh, material];\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Gets the intensity of the effect for a specific mesh.\r\n * @param mesh The mesh to get the effect intensity for\r\n * @returns The intensity of the effect for the mesh\r\n */\r\n public getEffectIntensity(mesh: AbstractMesh) {\r\n return this._effectIntensity[mesh.uniqueId] ?? 1;\r\n }\r\n\r\n /**\r\n * Sets the intensity of the effect for a specific mesh.\r\n * @param mesh The mesh to set the effect intensity for\r\n * @param intensity The intensity of the effect for the mesh\r\n */\r\n public setEffectIntensity(mesh: AbstractMesh, intensity: number): void {\r\n this._effectIntensity[mesh.uniqueId] = intensity;\r\n }\r\n\r\n /**\r\n * Instantiates a new effect Layer and references it in the scene.\r\n * @param name The name of the layer\r\n * @param scene The scene to use the layer in\r\n */\r\n constructor(\r\n /** The Friendly of the effect in the scene */\r\n name: string,\r\n scene?: Scene\r\n ) {\r\n this.name = name;\r\n\r\n this._scene = scene || EngineStore.LastCreatedScene;\r\n EffectLayer._SceneComponentInitialization(this._scene);\r\n\r\n this._engine = this._scene.getEngine();\r\n this._maxSize = this._engine.getCaps().maxTextureSize;\r\n this._scene.effectLayers.push(this);\r\n\r\n this._mergeDrawWrapper = [];\r\n\r\n // Generate Buffers\r\n this._generateIndexBuffer();\r\n this._generateVertexBuffer();\r\n }\r\n\r\n /**\r\n * Get the effect name of the layer.\r\n * @returns The effect name\r\n */\r\n public abstract getEffectName(): string;\r\n\r\n /**\r\n * Checks for the readiness of the element composing the layer.\r\n * @param subMesh the mesh to check for\r\n * @param useInstances specify whether or not to use instances to render the mesh\r\n * @returns true if ready otherwise, false\r\n */\r\n public abstract isReady(subMesh: SubMesh, useInstances: boolean): boolean;\r\n\r\n /**\r\n * Returns whether or not the layer needs stencil enabled during the mesh rendering.\r\n * @returns true if the effect requires stencil during the main canvas render pass.\r\n */\r\n public abstract needStencil(): boolean;\r\n\r\n /**\r\n * Create the merge effect. This is the shader use to blit the information back\r\n * to the main canvas at the end of the scene rendering.\r\n * @returns The effect containing the shader used to merge the effect on the main canvas\r\n */\r\n protected abstract _createMergeEffect(): Effect;\r\n\r\n /**\r\n * Creates the render target textures and post processes used in the effect layer.\r\n */\r\n protected abstract _createTextureAndPostProcesses(): void;\r\n\r\n /**\r\n * Implementation specific of rendering the generating effect on the main canvas.\r\n * @param effect The effect used to render through\r\n * @param renderNum Index of the _internalRender call (0 for the first time _internalRender is called, 1 for the second time, etc. _internalRender is called the number of times returned by _numInternalDraws())\r\n */\r\n protected abstract _internalRender(effect: Effect, renderIndex: number): void;\r\n\r\n /**\r\n * Sets the required values for both the emissive texture and and the main color.\r\n */\r\n protected abstract _setEmissiveTextureAndColor(mesh: Mesh, subMesh: SubMesh, material: Material): void;\r\n\r\n /**\r\n * Free any resources and references associated to a mesh.\r\n * Internal use\r\n * @param mesh The mesh to free.\r\n */\r\n public abstract _disposeMesh(mesh: Mesh): void;\r\n\r\n /**\r\n * Serializes this layer (Glow or Highlight for example)\r\n * @returns a serialized layer object\r\n */\r\n public abstract serialize?(): any;\r\n\r\n /**\r\n * Number of times _internalRender will be called. Some effect layers need to render the mesh several times, so they should override this method with the number of times the mesh should be rendered\r\n * @returns Number of times a mesh must be rendered in the layer\r\n */\r\n protected _numInternalDraws(): number {\r\n return 1;\r\n }\r\n\r\n /**\r\n * Initializes the effect layer with the required options.\r\n * @param options Sets of none mandatory options to use with the layer (see IEffectLayerOptions for more information)\r\n */\r\n protected _init(options: Partial): void {\r\n // Adapt options\r\n this._effectLayerOptions = {\r\n mainTextureRatio: 0.5,\r\n alphaBlendingMode: Constants.ALPHA_COMBINE,\r\n camera: null,\r\n renderingGroupId: -1,\r\n mainTextureType: Constants.TEXTURETYPE_UNSIGNED_INT,\r\n generateStencilBuffer: false,\r\n ...options,\r\n };\r\n\r\n this._setMainTextureSize();\r\n this._createMainTexture();\r\n this._createTextureAndPostProcesses();\r\n }\r\n\r\n /**\r\n * Generates the index buffer of the full screen quad blending to the main canvas.\r\n */\r\n private _generateIndexBuffer(): void {\r\n // Indices\r\n const indices = [];\r\n indices.push(0);\r\n indices.push(1);\r\n indices.push(2);\r\n\r\n indices.push(0);\r\n indices.push(2);\r\n indices.push(3);\r\n\r\n this._indexBuffer = this._engine.createIndexBuffer(indices);\r\n }\r\n\r\n /**\r\n * Generates the vertex buffer of the full screen quad blending to the main canvas.\r\n */\r\n private _generateVertexBuffer(): void {\r\n // VBO\r\n const vertices = [];\r\n vertices.push(1, 1);\r\n vertices.push(-1, 1);\r\n vertices.push(-1, -1);\r\n vertices.push(1, -1);\r\n\r\n const vertexBuffer = new VertexBuffer(this._engine, vertices, VertexBuffer.PositionKind, false, false, 2);\r\n this._vertexBuffers[VertexBuffer.PositionKind] = vertexBuffer;\r\n }\r\n\r\n /**\r\n * Sets the main texture desired size which is the closest power of two\r\n * of the engine canvas size.\r\n */\r\n private _setMainTextureSize(): void {\r\n if (this._effectLayerOptions.mainTextureFixedSize) {\r\n this._mainTextureDesiredSize.width = this._effectLayerOptions.mainTextureFixedSize;\r\n this._mainTextureDesiredSize.height = this._effectLayerOptions.mainTextureFixedSize;\r\n } else {\r\n this._mainTextureDesiredSize.width = this._engine.getRenderWidth() * this._effectLayerOptions.mainTextureRatio;\r\n this._mainTextureDesiredSize.height = this._engine.getRenderHeight() * this._effectLayerOptions.mainTextureRatio;\r\n\r\n this._mainTextureDesiredSize.width = this._engine.needPOTTextures\r\n ? Engine.GetExponentOfTwo(this._mainTextureDesiredSize.width, this._maxSize)\r\n : this._mainTextureDesiredSize.width;\r\n this._mainTextureDesiredSize.height = this._engine.needPOTTextures\r\n ? Engine.GetExponentOfTwo(this._mainTextureDesiredSize.height, this._maxSize)\r\n : this._mainTextureDesiredSize.height;\r\n }\r\n\r\n this._mainTextureDesiredSize.width = Math.floor(this._mainTextureDesiredSize.width);\r\n this._mainTextureDesiredSize.height = Math.floor(this._mainTextureDesiredSize.height);\r\n }\r\n\r\n /**\r\n * Creates the main texture for the effect layer.\r\n */\r\n protected _createMainTexture(): void {\r\n this._mainTexture = new RenderTargetTexture(\r\n \"EffectLayerMainRTT\",\r\n {\r\n width: this._mainTextureDesiredSize.width,\r\n height: this._mainTextureDesiredSize.height,\r\n },\r\n this._scene,\r\n false,\r\n true,\r\n this._effectLayerOptions.mainTextureType,\r\n false,\r\n Texture.TRILINEAR_SAMPLINGMODE,\r\n true,\r\n this._effectLayerOptions.generateStencilBuffer\r\n );\r\n this._mainTexture.activeCamera = this._effectLayerOptions.camera;\r\n this._mainTexture.wrapU = Texture.CLAMP_ADDRESSMODE;\r\n this._mainTexture.wrapV = Texture.CLAMP_ADDRESSMODE;\r\n this._mainTexture.anisotropicFilteringLevel = 1;\r\n this._mainTexture.updateSamplingMode(Texture.BILINEAR_SAMPLINGMODE);\r\n this._mainTexture.renderParticles = false;\r\n this._mainTexture.renderList = null;\r\n this._mainTexture.ignoreCameraViewport = true;\r\n\r\n for (const id in this._materialForRendering) {\r\n const [mesh, material] = this._materialForRendering[id];\r\n this._mainTexture.setMaterialForRendering(mesh, material);\r\n }\r\n\r\n this._mainTexture.customIsReadyFunction = (mesh: AbstractMesh, refreshRate: number, preWarm?: boolean) => {\r\n if ((preWarm || refreshRate === 0) && mesh.subMeshes) {\r\n for (let i = 0; i < mesh.subMeshes.length; ++i) {\r\n const subMesh = mesh.subMeshes[i];\r\n const material = subMesh.getMaterial();\r\n const renderingMesh = subMesh.getRenderingMesh();\r\n\r\n if (!material) {\r\n continue;\r\n }\r\n\r\n const batch = renderingMesh._getInstancesRenderList(subMesh._id, !!subMesh.getReplacementMesh());\r\n const hardwareInstancedRendering = batch.hardwareInstancedRendering[subMesh._id] || renderingMesh.hasThinInstances;\r\n\r\n this._setEmissiveTextureAndColor(renderingMesh, subMesh, material);\r\n\r\n if (!this._isReady(subMesh, hardwareInstancedRendering, this._emissiveTextureAndColor.texture)) {\r\n return false;\r\n }\r\n }\r\n }\r\n\r\n return true;\r\n };\r\n\r\n // Custom render function\r\n this._mainTexture.customRenderFunction = (\r\n opaqueSubMeshes: SmartArray,\r\n alphaTestSubMeshes: SmartArray,\r\n transparentSubMeshes: SmartArray,\r\n depthOnlySubMeshes: SmartArray\r\n ): void => {\r\n this.onBeforeRenderMainTextureObservable.notifyObservers(this);\r\n\r\n let index: number;\r\n\r\n const engine = this._scene.getEngine();\r\n\r\n if (depthOnlySubMeshes.length) {\r\n engine.setColorWrite(false);\r\n for (index = 0; index < depthOnlySubMeshes.length; index++) {\r\n this._renderSubMesh(depthOnlySubMeshes.data[index]);\r\n }\r\n engine.setColorWrite(true);\r\n }\r\n\r\n for (index = 0; index < opaqueSubMeshes.length; index++) {\r\n this._renderSubMesh(opaqueSubMeshes.data[index]);\r\n }\r\n\r\n for (index = 0; index < alphaTestSubMeshes.length; index++) {\r\n this._renderSubMesh(alphaTestSubMeshes.data[index]);\r\n }\r\n\r\n const previousAlphaMode = engine.getAlphaMode();\r\n\r\n for (index = 0; index < transparentSubMeshes.length; index++) {\r\n this._renderSubMesh(transparentSubMeshes.data[index], true);\r\n }\r\n\r\n engine.setAlphaMode(previousAlphaMode);\r\n };\r\n\r\n this._mainTexture.onClearObservable.add((engine: Engine) => {\r\n engine.clear(this.neutralColor, true, true, true);\r\n });\r\n\r\n // Prevent package size in es6 (getBoundingBoxRenderer might not be present)\r\n if (this._scene.getBoundingBoxRenderer) {\r\n const boundingBoxRendererEnabled = this._scene.getBoundingBoxRenderer().enabled;\r\n\r\n this._mainTexture.onBeforeBindObservable.add(() => {\r\n this._scene.getBoundingBoxRenderer().enabled = !this.disableBoundingBoxesFromEffectLayer && boundingBoxRendererEnabled;\r\n });\r\n\r\n this._mainTexture.onAfterUnbindObservable.add(() => {\r\n this._scene.getBoundingBoxRenderer().enabled = boundingBoxRendererEnabled;\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Adds specific effects defines.\r\n * @param defines The defines to add specifics to.\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n protected _addCustomEffectDefines(defines: string[]): void {\r\n // Nothing to add by default.\r\n }\r\n\r\n /**\r\n * Checks for the readiness of the element composing the layer.\r\n * @param subMesh the mesh to check for\r\n * @param useInstances specify whether or not to use instances to render the mesh\r\n * @param emissiveTexture the associated emissive texture used to generate the glow\r\n * @returns true if ready otherwise, false\r\n */\r\n protected _isReady(subMesh: SubMesh, useInstances: boolean, emissiveTexture: Nullable): boolean {\r\n const engine = this._scene.getEngine();\r\n const mesh = subMesh.getMesh();\r\n\r\n const renderingMaterial = mesh._internalAbstractMeshDataInfo._materialForRenderPass?.[engine.currentRenderPassId];\r\n\r\n if (renderingMaterial) {\r\n return renderingMaterial.isReadyForSubMesh(mesh, subMesh, useInstances);\r\n }\r\n\r\n const material = subMesh.getMaterial();\r\n\r\n if (!material) {\r\n return false;\r\n }\r\n\r\n if (this._useMeshMaterial(subMesh.getRenderingMesh())) {\r\n return material.isReadyForSubMesh(subMesh.getMesh(), subMesh, useInstances);\r\n }\r\n\r\n const defines: string[] = [];\r\n\r\n const attribs = [VertexBuffer.PositionKind];\r\n\r\n let uv1 = false;\r\n let uv2 = false;\r\n\r\n // Diffuse\r\n if (material) {\r\n const needAlphaTest = material.needAlphaTesting();\r\n\r\n const diffuseTexture = material.getAlphaTestTexture();\r\n const needAlphaBlendFromDiffuse =\r\n diffuseTexture && diffuseTexture.hasAlpha && ((material as any).useAlphaFromDiffuseTexture || (material as any)._useAlphaFromAlbedoTexture);\r\n\r\n if (diffuseTexture && (needAlphaTest || needAlphaBlendFromDiffuse)) {\r\n defines.push(\"#define DIFFUSE\");\r\n if (mesh.isVerticesDataPresent(VertexBuffer.UV2Kind) && diffuseTexture.coordinatesIndex === 1) {\r\n defines.push(\"#define DIFFUSEUV2\");\r\n uv2 = true;\r\n } else if (mesh.isVerticesDataPresent(VertexBuffer.UVKind)) {\r\n defines.push(\"#define DIFFUSEUV1\");\r\n uv1 = true;\r\n }\r\n\r\n if (needAlphaTest) {\r\n defines.push(\"#define ALPHATEST\");\r\n defines.push(\"#define ALPHATESTVALUE 0.4\");\r\n }\r\n if (!diffuseTexture.gammaSpace) {\r\n defines.push(\"#define DIFFUSE_ISLINEAR\");\r\n }\r\n }\r\n\r\n const opacityTexture = (material as any).opacityTexture;\r\n if (opacityTexture) {\r\n defines.push(\"#define OPACITY\");\r\n if (mesh.isVerticesDataPresent(VertexBuffer.UV2Kind) && opacityTexture.coordinatesIndex === 1) {\r\n defines.push(\"#define OPACITYUV2\");\r\n uv2 = true;\r\n } else if (mesh.isVerticesDataPresent(VertexBuffer.UVKind)) {\r\n defines.push(\"#define OPACITYUV1\");\r\n uv1 = true;\r\n }\r\n }\r\n }\r\n\r\n // Emissive\r\n if (emissiveTexture) {\r\n defines.push(\"#define EMISSIVE\");\r\n if (mesh.isVerticesDataPresent(VertexBuffer.UV2Kind) && emissiveTexture.coordinatesIndex === 1) {\r\n defines.push(\"#define EMISSIVEUV2\");\r\n uv2 = true;\r\n } else if (mesh.isVerticesDataPresent(VertexBuffer.UVKind)) {\r\n defines.push(\"#define EMISSIVEUV1\");\r\n uv1 = true;\r\n }\r\n if (!emissiveTexture.gammaSpace) {\r\n defines.push(\"#define EMISSIVE_ISLINEAR\");\r\n }\r\n }\r\n\r\n // Vertex\r\n if (mesh.useVertexColors && mesh.isVerticesDataPresent(VertexBuffer.ColorKind) && mesh.hasVertexAlpha && material.transparencyMode !== Material.MATERIAL_OPAQUE) {\r\n attribs.push(VertexBuffer.ColorKind);\r\n defines.push(\"#define VERTEXALPHA\");\r\n }\r\n\r\n if (uv1) {\r\n attribs.push(VertexBuffer.UVKind);\r\n defines.push(\"#define UV1\");\r\n }\r\n if (uv2) {\r\n attribs.push(VertexBuffer.UV2Kind);\r\n defines.push(\"#define UV2\");\r\n }\r\n\r\n // Bones\r\n const fallbacks = new EffectFallbacks();\r\n if (mesh.useBones && mesh.computeBonesUsingShaders) {\r\n attribs.push(VertexBuffer.MatricesIndicesKind);\r\n attribs.push(VertexBuffer.MatricesWeightsKind);\r\n if (mesh.numBoneInfluencers > 4) {\r\n attribs.push(VertexBuffer.MatricesIndicesExtraKind);\r\n attribs.push(VertexBuffer.MatricesWeightsExtraKind);\r\n }\r\n\r\n defines.push(\"#define NUM_BONE_INFLUENCERS \" + mesh.numBoneInfluencers);\r\n\r\n const skeleton = mesh.skeleton;\r\n if (skeleton && skeleton.isUsingTextureForMatrices) {\r\n defines.push(\"#define BONETEXTURE\");\r\n } else {\r\n defines.push(\"#define BonesPerMesh \" + (skeleton ? skeleton.bones.length + 1 : 0));\r\n }\r\n\r\n if (mesh.numBoneInfluencers > 0) {\r\n fallbacks.addCPUSkinningFallback(0, mesh);\r\n }\r\n } else {\r\n defines.push(\"#define NUM_BONE_INFLUENCERS 0\");\r\n }\r\n\r\n // Morph targets\r\n const manager = (mesh).morphTargetManager;\r\n let morphInfluencers = 0;\r\n if (manager) {\r\n if (manager.numInfluencers > 0) {\r\n defines.push(\"#define MORPHTARGETS\");\r\n morphInfluencers = manager.numInfluencers;\r\n defines.push(\"#define NUM_MORPH_INFLUENCERS \" + morphInfluencers);\r\n if (manager.isUsingTextureForTargets) {\r\n defines.push(\"#define MORPHTARGETS_TEXTURE\");\r\n }\r\n MaterialHelper.PrepareAttributesForMorphTargetsInfluencers(attribs, mesh, morphInfluencers);\r\n }\r\n }\r\n\r\n // Instances\r\n if (useInstances) {\r\n defines.push(\"#define INSTANCES\");\r\n MaterialHelper.PushAttributesForInstances(attribs);\r\n if (subMesh.getRenderingMesh().hasThinInstances) {\r\n defines.push(\"#define THIN_INSTANCES\");\r\n }\r\n }\r\n\r\n // ClipPlanes\r\n prepareStringDefinesForClipPlanes(material, this._scene, defines);\r\n\r\n this._addCustomEffectDefines(defines);\r\n\r\n // Get correct effect\r\n const drawWrapper = subMesh._getDrawWrapper(undefined, true)!;\r\n const cachedDefines = drawWrapper.defines as string;\r\n const join = defines.join(\"\\n\");\r\n if (cachedDefines !== join) {\r\n const uniforms = [\r\n \"world\",\r\n \"mBones\",\r\n \"viewProjection\",\r\n \"glowColor\",\r\n \"morphTargetInfluences\",\r\n \"boneTextureWidth\",\r\n \"diffuseMatrix\",\r\n \"emissiveMatrix\",\r\n \"opacityMatrix\",\r\n \"opacityIntensity\",\r\n \"morphTargetTextureInfo\",\r\n \"morphTargetTextureIndices\",\r\n \"glowIntensity\",\r\n ];\r\n\r\n addClipPlaneUniforms(uniforms);\r\n\r\n drawWrapper.setEffect(\r\n this._engine.createEffect(\r\n \"glowMapGeneration\",\r\n attribs,\r\n uniforms,\r\n [\"diffuseSampler\", \"emissiveSampler\", \"opacitySampler\", \"boneSampler\", \"morphTargets\"],\r\n join,\r\n fallbacks,\r\n undefined,\r\n undefined,\r\n { maxSimultaneousMorphTargets: morphInfluencers }\r\n ),\r\n join\r\n );\r\n }\r\n\r\n return drawWrapper.effect!.isReady();\r\n }\r\n\r\n /**\r\n * Renders the glowing part of the scene by blending the blurred glowing meshes on top of the rendered scene.\r\n */\r\n public render(): void {\r\n for (let i = 0; i < this._postProcesses.length; i++) {\r\n if (!this._postProcesses[i].isReady()) {\r\n return;\r\n }\r\n }\r\n\r\n const engine = this._scene.getEngine();\r\n const numDraws = this._numInternalDraws();\r\n\r\n // Check\r\n let isReady = true;\r\n for (let i = 0; i < numDraws; ++i) {\r\n let currentEffect = this._mergeDrawWrapper[i];\r\n if (!currentEffect) {\r\n currentEffect = this._mergeDrawWrapper[i] = new DrawWrapper(this._engine);\r\n currentEffect.setEffect(this._createMergeEffect());\r\n }\r\n isReady = isReady && currentEffect.effect!.isReady();\r\n }\r\n\r\n if (!isReady) {\r\n return;\r\n }\r\n\r\n this.onBeforeComposeObservable.notifyObservers(this);\r\n\r\n const previousAlphaMode = engine.getAlphaMode();\r\n\r\n for (let i = 0; i < numDraws; ++i) {\r\n const currentEffect = this._mergeDrawWrapper[i];\r\n\r\n // Render\r\n engine.enableEffect(currentEffect);\r\n engine.setState(false);\r\n\r\n // VBOs\r\n engine.bindBuffers(this._vertexBuffers, this._indexBuffer, currentEffect.effect!);\r\n\r\n // Go Blend.\r\n engine.setAlphaMode(this._effectLayerOptions.alphaBlendingMode);\r\n\r\n // Blends the map on the main canvas.\r\n this._internalRender(currentEffect.effect!, i);\r\n }\r\n\r\n // Restore Alpha\r\n engine.setAlphaMode(previousAlphaMode);\r\n\r\n this.onAfterComposeObservable.notifyObservers(this);\r\n\r\n // Handle size changes.\r\n const size = this._mainTexture.getSize();\r\n this._setMainTextureSize();\r\n if (\r\n (size.width !== this._mainTextureDesiredSize.width || size.height !== this._mainTextureDesiredSize.height) &&\r\n this._mainTextureDesiredSize.width !== 0 &&\r\n this._mainTextureDesiredSize.height !== 0\r\n ) {\r\n // Recreate RTT and post processes on size change.\r\n this.onSizeChangedObservable.notifyObservers(this);\r\n this._disposeTextureAndPostProcesses();\r\n this._createMainTexture();\r\n this._createTextureAndPostProcesses();\r\n }\r\n }\r\n\r\n /**\r\n * Determine if a given mesh will be used in the current effect.\r\n * @param mesh mesh to test\r\n * @returns true if the mesh will be used\r\n */\r\n public hasMesh(mesh: AbstractMesh): boolean {\r\n if (this.renderingGroupId === -1 || mesh.renderingGroupId === this.renderingGroupId) {\r\n return true;\r\n }\r\n return false;\r\n }\r\n\r\n /**\r\n * Returns true if the layer contains information to display, otherwise false.\r\n * @returns true if the glow layer should be rendered\r\n */\r\n public shouldRender(): boolean {\r\n return this.isEnabled && this._shouldRender;\r\n }\r\n\r\n /**\r\n * Returns true if the mesh should render, otherwise false.\r\n * @param mesh The mesh to render\r\n * @returns true if it should render otherwise false\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n protected _shouldRenderMesh(mesh: AbstractMesh): boolean {\r\n return true;\r\n }\r\n\r\n /**\r\n * Returns true if the mesh can be rendered, otherwise false.\r\n * @param mesh The mesh to render\r\n * @param material The material used on the mesh\r\n * @returns true if it can be rendered otherwise false\r\n */\r\n protected _canRenderMesh(mesh: AbstractMesh, material: Material): boolean {\r\n return !material.needAlphaBlendingForMesh(mesh);\r\n }\r\n\r\n /**\r\n * Returns true if the mesh should render, otherwise false.\r\n * @returns true if it should render otherwise false\r\n */\r\n protected _shouldRenderEmissiveTextureForMesh(): boolean {\r\n return true;\r\n }\r\n\r\n /**\r\n * Renders the submesh passed in parameter to the generation map.\r\n * @param subMesh\r\n * @param enableAlphaMode\r\n */\r\n protected _renderSubMesh(subMesh: SubMesh, enableAlphaMode: boolean = false): void {\r\n if (!this.shouldRender()) {\r\n return;\r\n }\r\n\r\n const material = subMesh.getMaterial();\r\n const ownerMesh = subMesh.getMesh();\r\n const replacementMesh = subMesh.getReplacementMesh();\r\n const renderingMesh = subMesh.getRenderingMesh();\r\n const effectiveMesh = subMesh.getEffectiveMesh();\r\n const scene = this._scene;\r\n const engine = scene.getEngine();\r\n\r\n effectiveMesh._internalAbstractMeshDataInfo._isActiveIntermediate = false;\r\n\r\n if (!material) {\r\n return;\r\n }\r\n\r\n // Do not block in blend mode.\r\n if (!this._canRenderMesh(renderingMesh, material)) {\r\n return;\r\n }\r\n\r\n // Culling\r\n let sideOrientation = renderingMesh.overrideMaterialSideOrientation ?? material.sideOrientation;\r\n const mainDeterminant = effectiveMesh._getWorldMatrixDeterminant();\r\n if (mainDeterminant < 0) {\r\n sideOrientation = sideOrientation === Material.ClockWiseSideOrientation ? Material.CounterClockWiseSideOrientation : Material.ClockWiseSideOrientation;\r\n }\r\n\r\n const reverse = sideOrientation === Material.ClockWiseSideOrientation;\r\n engine.setState(material.backFaceCulling, material.zOffset, undefined, reverse, material.cullBackFaces, undefined, material.zOffsetUnits);\r\n\r\n // Managing instances\r\n const batch = renderingMesh._getInstancesRenderList(subMesh._id, !!replacementMesh);\r\n if (batch.mustReturn) {\r\n return;\r\n }\r\n\r\n // Early Exit per mesh\r\n if (!this._shouldRenderMesh(renderingMesh)) {\r\n return;\r\n }\r\n\r\n const hardwareInstancedRendering = batch.hardwareInstancedRendering[subMesh._id] || renderingMesh.hasThinInstances;\r\n\r\n this._setEmissiveTextureAndColor(renderingMesh, subMesh, material);\r\n\r\n this.onBeforeRenderMeshToEffect.notifyObservers(ownerMesh);\r\n\r\n if (this._useMeshMaterial(renderingMesh)) {\r\n renderingMesh.render(subMesh, enableAlphaMode, replacementMesh || undefined);\r\n } else if (this._isReady(subMesh, hardwareInstancedRendering, this._emissiveTextureAndColor.texture)) {\r\n const renderingMaterial = effectiveMesh._internalAbstractMeshDataInfo._materialForRenderPass?.[engine.currentRenderPassId];\r\n\r\n let drawWrapper = subMesh._getDrawWrapper();\r\n if (!drawWrapper && renderingMaterial) {\r\n drawWrapper = renderingMaterial._getDrawWrapper();\r\n }\r\n\r\n if (!drawWrapper) {\r\n return;\r\n }\r\n\r\n const effect = drawWrapper.effect!;\r\n\r\n engine.enableEffect(drawWrapper);\r\n if (!hardwareInstancedRendering) {\r\n renderingMesh._bind(subMesh, effect, material.fillMode);\r\n }\r\n\r\n if (!renderingMaterial) {\r\n effect.setMatrix(\"viewProjection\", scene.getTransformMatrix());\r\n effect.setMatrix(\"world\", effectiveMesh.getWorldMatrix());\r\n effect.setFloat4(\r\n \"glowColor\",\r\n this._emissiveTextureAndColor.color.r,\r\n this._emissiveTextureAndColor.color.g,\r\n this._emissiveTextureAndColor.color.b,\r\n this._emissiveTextureAndColor.color.a\r\n );\r\n } else {\r\n renderingMaterial.bindForSubMesh(effectiveMesh.getWorldMatrix(), effectiveMesh as Mesh, subMesh);\r\n }\r\n\r\n if (!renderingMaterial) {\r\n const needAlphaTest = material.needAlphaTesting();\r\n\r\n const diffuseTexture = material.getAlphaTestTexture();\r\n const needAlphaBlendFromDiffuse =\r\n diffuseTexture && diffuseTexture.hasAlpha && ((material as any).useAlphaFromDiffuseTexture || (material as any)._useAlphaFromAlbedoTexture);\r\n\r\n if (diffuseTexture && (needAlphaTest || needAlphaBlendFromDiffuse)) {\r\n effect.setTexture(\"diffuseSampler\", diffuseTexture);\r\n const textureMatrix = diffuseTexture.getTextureMatrix();\r\n\r\n if (textureMatrix) {\r\n effect.setMatrix(\"diffuseMatrix\", textureMatrix);\r\n }\r\n }\r\n\r\n const opacityTexture = (material as any).opacityTexture;\r\n if (opacityTexture) {\r\n effect.setTexture(\"opacitySampler\", opacityTexture);\r\n effect.setFloat(\"opacityIntensity\", opacityTexture.level);\r\n const textureMatrix = opacityTexture.getTextureMatrix();\r\n if (textureMatrix) {\r\n effect.setMatrix(\"opacityMatrix\", textureMatrix);\r\n }\r\n }\r\n\r\n // Glow emissive only\r\n if (this._emissiveTextureAndColor.texture) {\r\n effect.setTexture(\"emissiveSampler\", this._emissiveTextureAndColor.texture);\r\n effect.setMatrix(\"emissiveMatrix\", this._emissiveTextureAndColor.texture.getTextureMatrix());\r\n }\r\n\r\n // Bones\r\n if (renderingMesh.useBones && renderingMesh.computeBonesUsingShaders && renderingMesh.skeleton) {\r\n const skeleton = renderingMesh.skeleton;\r\n\r\n if (skeleton.isUsingTextureForMatrices) {\r\n const boneTexture = skeleton.getTransformMatrixTexture(renderingMesh);\r\n if (!boneTexture) {\r\n return;\r\n }\r\n\r\n effect.setTexture(\"boneSampler\", boneTexture);\r\n effect.setFloat(\"boneTextureWidth\", 4.0 * (skeleton.bones.length + 1));\r\n } else {\r\n effect.setMatrices(\"mBones\", skeleton.getTransformMatrices(renderingMesh));\r\n }\r\n }\r\n\r\n // Morph targets\r\n MaterialHelper.BindMorphTargetParameters(renderingMesh, effect);\r\n if (renderingMesh.morphTargetManager && renderingMesh.morphTargetManager.isUsingTextureForTargets) {\r\n renderingMesh.morphTargetManager._bind(effect);\r\n }\r\n\r\n // Alpha mode\r\n if (enableAlphaMode) {\r\n engine.setAlphaMode(material.alphaMode);\r\n }\r\n\r\n // Intensity of effect\r\n effect.setFloat(\"glowIntensity\", this.getEffectIntensity(renderingMesh));\r\n\r\n // Clip planes\r\n bindClipPlane(effect, material, scene);\r\n }\r\n\r\n // Draw\r\n renderingMesh._processRendering(effectiveMesh, subMesh, effect, material.fillMode, batch, hardwareInstancedRendering, (isInstance, world) =>\r\n effect.setMatrix(\"world\", world)\r\n );\r\n } else {\r\n // Need to reset refresh rate of the main map\r\n this._mainTexture.resetRefreshCounter();\r\n }\r\n\r\n this.onAfterRenderMeshToEffect.notifyObservers(ownerMesh);\r\n }\r\n\r\n /**\r\n * Defines whether the current material of the mesh should be use to render the effect.\r\n * @param mesh defines the current mesh to render\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n protected _useMeshMaterial(mesh: AbstractMesh): boolean {\r\n return false;\r\n }\r\n\r\n /**\r\n * Rebuild the required buffers.\r\n * @internal Internal use only.\r\n */\r\n public _rebuild(): void {\r\n const vb = this._vertexBuffers[VertexBuffer.PositionKind];\r\n\r\n if (vb) {\r\n vb._rebuild();\r\n }\r\n\r\n this._generateIndexBuffer();\r\n }\r\n\r\n /**\r\n * Dispose only the render target textures and post process.\r\n */\r\n private _disposeTextureAndPostProcesses(): void {\r\n this._mainTexture.dispose();\r\n\r\n for (let i = 0; i < this._postProcesses.length; i++) {\r\n if (this._postProcesses[i]) {\r\n this._postProcesses[i].dispose();\r\n }\r\n }\r\n this._postProcesses = [];\r\n\r\n for (let i = 0; i < this._textures.length; i++) {\r\n if (this._textures[i]) {\r\n this._textures[i].dispose();\r\n }\r\n }\r\n this._textures = [];\r\n }\r\n\r\n /**\r\n * Dispose the highlight layer and free resources.\r\n */\r\n public dispose(): void {\r\n const vertexBuffer = this._vertexBuffers[VertexBuffer.PositionKind];\r\n if (vertexBuffer) {\r\n vertexBuffer.dispose();\r\n this._vertexBuffers[VertexBuffer.PositionKind] = null;\r\n }\r\n\r\n if (this._indexBuffer) {\r\n this._scene.getEngine()._releaseBuffer(this._indexBuffer);\r\n this._indexBuffer = null;\r\n }\r\n\r\n for (const drawWrapper of this._mergeDrawWrapper) {\r\n drawWrapper.dispose();\r\n }\r\n this._mergeDrawWrapper = [];\r\n\r\n // Clean textures and post processes\r\n this._disposeTextureAndPostProcesses();\r\n\r\n // Remove from scene\r\n const index = this._scene.effectLayers.indexOf(this, 0);\r\n if (index > -1) {\r\n this._scene.effectLayers.splice(index, 1);\r\n }\r\n\r\n // Callback\r\n this.onDisposeObservable.notifyObservers(this);\r\n\r\n this.onDisposeObservable.clear();\r\n this.onBeforeRenderMainTextureObservable.clear();\r\n this.onBeforeComposeObservable.clear();\r\n this.onBeforeRenderMeshToEffect.clear();\r\n this.onAfterRenderMeshToEffect.clear();\r\n this.onAfterComposeObservable.clear();\r\n this.onSizeChangedObservable.clear();\r\n }\r\n\r\n /**\r\n * Gets the class name of the effect layer\r\n * @returns the string with the class name of the effect layer\r\n */\r\n public getClassName(): string {\r\n return \"EffectLayer\";\r\n }\r\n\r\n /**\r\n * Creates an effect layer from parsed effect layer data\r\n * @param parsedEffectLayer defines effect layer data\r\n * @param scene defines the current scene\r\n * @param rootUrl defines the root URL containing the effect layer information\r\n * @returns a parsed effect Layer\r\n */\r\n public static Parse(parsedEffectLayer: any, scene: Scene, rootUrl: string): EffectLayer {\r\n const effectLayerType = Tools.Instantiate(parsedEffectLayer.customType);\r\n\r\n return effectLayerType.Parse(parsedEffectLayer, scene, rootUrl);\r\n }\r\n}\r\n","import { Camera } from \"../Cameras/camera\";\r\nimport type { Scene } from \"../scene\";\r\nimport type { Engine } from \"../Engines/engine\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { RenderTargetTexture } from \"../Materials/Textures/renderTargetTexture\";\r\nimport type { ISceneSerializableComponent } from \"../sceneComponent\";\r\nimport { SceneComponentConstants } from \"../sceneComponent\";\r\nimport { EffectLayer } from \"./effectLayer\";\r\nimport { AbstractScene } from \"../abstractScene\";\r\nimport type { AssetContainer } from \"../assetContainer\";\r\nimport { EngineStore } from \"../Engines/engineStore\";\r\n// Adds the parser to the scene parsers.\r\nAbstractScene.AddParser(SceneComponentConstants.NAME_EFFECTLAYER, (parsedData: any, scene: Scene, container: AssetContainer, rootUrl: string) => {\r\n if (parsedData.effectLayers) {\r\n if (!container.effectLayers) {\r\n container.effectLayers = new Array();\r\n }\r\n\r\n for (let index = 0; index < parsedData.effectLayers.length; index++) {\r\n const effectLayer = EffectLayer.Parse(parsedData.effectLayers[index], scene, rootUrl);\r\n container.effectLayers.push(effectLayer);\r\n }\r\n }\r\n});\r\n\r\ndeclare module \"../abstractScene\" {\r\n export interface AbstractScene {\r\n /**\r\n * The list of effect layers (highlights/glow) added to the scene\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/highlightLayer\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/glowLayer\r\n */\r\n effectLayers: Array;\r\n\r\n /**\r\n * Removes the given effect layer from this scene.\r\n * @param toRemove defines the effect layer to remove\r\n * @returns the index of the removed effect layer\r\n */\r\n removeEffectLayer(toRemove: EffectLayer): number;\r\n\r\n /**\r\n * Adds the given effect layer to this scene\r\n * @param newEffectLayer defines the effect layer to add\r\n */\r\n addEffectLayer(newEffectLayer: EffectLayer): void;\r\n }\r\n}\r\n\r\nAbstractScene.prototype.removeEffectLayer = function (toRemove: EffectLayer): number {\r\n const index = this.effectLayers.indexOf(toRemove);\r\n if (index !== -1) {\r\n this.effectLayers.splice(index, 1);\r\n }\r\n\r\n return index;\r\n};\r\n\r\nAbstractScene.prototype.addEffectLayer = function (newEffectLayer: EffectLayer): void {\r\n this.effectLayers.push(newEffectLayer);\r\n};\r\n\r\n/**\r\n * Defines the layer scene component responsible to manage any effect layers\r\n * in a given scene.\r\n */\r\nexport class EffectLayerSceneComponent implements ISceneSerializableComponent {\r\n /**\r\n * The component name helpful to identify the component in the list of scene components.\r\n */\r\n public readonly name = SceneComponentConstants.NAME_EFFECTLAYER;\r\n\r\n /**\r\n * The scene the component belongs to.\r\n */\r\n public scene: Scene;\r\n\r\n private _engine: Engine;\r\n private _renderEffects = false;\r\n private _needStencil = false;\r\n private _previousStencilState = false;\r\n\r\n /**\r\n * Creates a new instance of the component for the given scene\r\n * @param scene Defines the scene to register the component in\r\n */\r\n constructor(scene?: Scene) {\r\n this.scene = scene || EngineStore.LastCreatedScene;\r\n if (!this.scene) {\r\n return;\r\n }\r\n this._engine = this.scene.getEngine();\r\n this.scene.effectLayers = new Array();\r\n }\r\n\r\n /**\r\n * Registers the component in a given scene\r\n */\r\n public register(): void {\r\n this.scene._isReadyForMeshStage.registerStep(SceneComponentConstants.STEP_ISREADYFORMESH_EFFECTLAYER, this, this._isReadyForMesh);\r\n\r\n this.scene._cameraDrawRenderTargetStage.registerStep(SceneComponentConstants.STEP_CAMERADRAWRENDERTARGET_EFFECTLAYER, this, this._renderMainTexture);\r\n\r\n this.scene._beforeCameraDrawStage.registerStep(SceneComponentConstants.STEP_BEFORECAMERADRAW_EFFECTLAYER, this, this._setStencil);\r\n\r\n this.scene._afterRenderingGroupDrawStage.registerStep(SceneComponentConstants.STEP_AFTERRENDERINGGROUPDRAW_EFFECTLAYER_DRAW, this, this._drawRenderingGroup);\r\n\r\n this.scene._afterCameraDrawStage.registerStep(SceneComponentConstants.STEP_AFTERCAMERADRAW_EFFECTLAYER, this, this._setStencilBack);\r\n this.scene._afterCameraDrawStage.registerStep(SceneComponentConstants.STEP_AFTERCAMERADRAW_EFFECTLAYER_DRAW, this, this._drawCamera);\r\n }\r\n\r\n /**\r\n * Rebuilds the elements related to this component in case of\r\n * context lost for instance.\r\n */\r\n public rebuild(): void {\r\n const layers = this.scene.effectLayers;\r\n for (const effectLayer of layers) {\r\n effectLayer._rebuild();\r\n }\r\n }\r\n\r\n /**\r\n * Serializes the component data to the specified json object\r\n * @param serializationObject The object to serialize to\r\n */\r\n public serialize(serializationObject: any): void {\r\n // Effect layers\r\n serializationObject.effectLayers = [];\r\n\r\n const layers = this.scene.effectLayers;\r\n for (const effectLayer of layers) {\r\n if (effectLayer.serialize) {\r\n serializationObject.effectLayers.push(effectLayer.serialize());\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Adds all the elements from the container to the scene\r\n * @param container the container holding the elements\r\n */\r\n public addFromContainer(container: AbstractScene): void {\r\n if (!container.effectLayers) {\r\n return;\r\n }\r\n container.effectLayers.forEach((o) => {\r\n this.scene.addEffectLayer(o);\r\n });\r\n }\r\n\r\n /**\r\n * Removes all the elements in the container from the scene\r\n * @param container contains the elements to remove\r\n * @param dispose if the removed element should be disposed (default: false)\r\n */\r\n public removeFromContainer(container: AbstractScene, dispose?: boolean): void {\r\n if (!container.effectLayers) {\r\n return;\r\n }\r\n container.effectLayers.forEach((o) => {\r\n this.scene.removeEffectLayer(o);\r\n if (dispose) {\r\n o.dispose();\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Disposes the component and the associated resources.\r\n */\r\n public dispose(): void {\r\n const layers = this.scene.effectLayers;\r\n while (layers.length) {\r\n layers[0].dispose();\r\n }\r\n }\r\n\r\n private _isReadyForMesh(mesh: AbstractMesh, hardwareInstancedRendering: boolean): boolean {\r\n const currentRenderPassId = this._engine.currentRenderPassId;\r\n const layers = this.scene.effectLayers;\r\n for (const layer of layers) {\r\n if (!layer.hasMesh(mesh)) {\r\n continue;\r\n }\r\n\r\n const renderTarget = (layer)._mainTexture;\r\n this._engine.currentRenderPassId = renderTarget.renderPassId;\r\n\r\n for (const subMesh of mesh.subMeshes) {\r\n if (!layer.isReady(subMesh, hardwareInstancedRendering)) {\r\n this._engine.currentRenderPassId = currentRenderPassId;\r\n return false;\r\n }\r\n }\r\n }\r\n this._engine.currentRenderPassId = currentRenderPassId;\r\n return true;\r\n }\r\n\r\n private _renderMainTexture(camera: Camera): boolean {\r\n this._renderEffects = false;\r\n this._needStencil = false;\r\n\r\n let needRebind = false;\r\n\r\n const layers = this.scene.effectLayers;\r\n if (layers && layers.length > 0) {\r\n this._previousStencilState = this._engine.getStencilBuffer();\r\n for (const effectLayer of layers) {\r\n if (\r\n effectLayer.shouldRender() &&\r\n (!effectLayer.camera ||\r\n (effectLayer.camera.cameraRigMode === Camera.RIG_MODE_NONE && camera === effectLayer.camera) ||\r\n (effectLayer.camera.cameraRigMode !== Camera.RIG_MODE_NONE && effectLayer.camera._rigCameras.indexOf(camera) > -1))\r\n ) {\r\n this._renderEffects = true;\r\n this._needStencil = this._needStencil || effectLayer.needStencil();\r\n\r\n const renderTarget = (effectLayer)._mainTexture;\r\n if (renderTarget._shouldRender()) {\r\n this.scene.incrementRenderId();\r\n renderTarget.render(false, false);\r\n needRebind = true;\r\n }\r\n }\r\n }\r\n\r\n this.scene.incrementRenderId();\r\n }\r\n\r\n return needRebind;\r\n }\r\n\r\n private _setStencil() {\r\n // Activate effect Layer stencil\r\n if (this._needStencil) {\r\n this._engine.setStencilBuffer(true);\r\n }\r\n }\r\n\r\n private _setStencilBack() {\r\n // Restore effect Layer stencil\r\n if (this._needStencil) {\r\n this._engine.setStencilBuffer(this._previousStencilState);\r\n }\r\n }\r\n\r\n private _draw(renderingGroupId: number): void {\r\n if (this._renderEffects) {\r\n this._engine.setDepthBuffer(false);\r\n\r\n const layers = this.scene.effectLayers;\r\n for (let i = 0; i < layers.length; i++) {\r\n const effectLayer = layers[i];\r\n if (effectLayer.renderingGroupId === renderingGroupId) {\r\n if (effectLayer.shouldRender()) {\r\n effectLayer.render();\r\n }\r\n }\r\n }\r\n this._engine.setDepthBuffer(true);\r\n }\r\n }\r\n\r\n private _drawCamera(): void {\r\n if (this._renderEffects) {\r\n this._draw(-1);\r\n }\r\n }\r\n private _drawRenderingGroup(index: number): void {\r\n if (!this.scene._isInIntermediateRendering() && this._renderEffects) {\r\n this._draw(index);\r\n }\r\n }\r\n}\r\n\r\nEffectLayer._SceneComponentInitialization = (scene: Scene) => {\r\n let component = scene._getComponent(SceneComponentConstants.NAME_EFFECTLAYER) as EffectLayerSceneComponent;\r\n if (!component) {\r\n component = new EffectLayerSceneComponent(scene);\r\n scene._addComponent(component);\r\n }\r\n};\r\n","import { serialize, serializeAsVector3 } from \"../Misc/decorators\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Matrix, Vector3 } from \"../Maths/math.vector\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport { Light } from \"./light\";\r\nimport { Axis } from \"../Maths/math.axis\";\r\n/**\r\n * Interface describing all the common properties and methods a shadow light needs to implement.\r\n * This helps both the shadow generator and materials to generate the corresponding shadow maps\r\n * as well as binding the different shadow properties to the effects.\r\n */\r\nexport interface IShadowLight extends Light {\r\n /**\r\n * The light id in the scene (used in scene.getLightById for instance)\r\n */\r\n id: string;\r\n /**\r\n * The position the shadow will be casted from.\r\n */\r\n position: Vector3;\r\n /**\r\n * In 2d mode (needCube being false), the direction used to cast the shadow.\r\n */\r\n direction: Vector3;\r\n /**\r\n * The transformed position. Position of the light in world space taking parenting in account.\r\n */\r\n transformedPosition: Vector3;\r\n /**\r\n * The transformed direction. Direction of the light in world space taking parenting in account.\r\n */\r\n transformedDirection: Vector3;\r\n /**\r\n * The friendly name of the light in the scene.\r\n */\r\n name: string;\r\n /**\r\n * Defines the shadow projection clipping minimum z value.\r\n */\r\n shadowMinZ: number;\r\n /**\r\n * Defines the shadow projection clipping maximum z value.\r\n */\r\n shadowMaxZ: number;\r\n\r\n /**\r\n * Computes the transformed information (transformedPosition and transformedDirection in World space) of the current light\r\n * @returns true if the information has been computed, false if it does not need to (no parenting)\r\n */\r\n computeTransformedInformation(): boolean;\r\n\r\n /**\r\n * Gets the scene the light belongs to.\r\n * @returns The scene\r\n */\r\n getScene(): Scene;\r\n\r\n /**\r\n * Callback defining a custom Projection Matrix Builder.\r\n * This can be used to override the default projection matrix computation.\r\n */\r\n customProjectionMatrixBuilder: (viewMatrix: Matrix, renderList: Array, result: Matrix) => void;\r\n\r\n /**\r\n * Sets the shadow projection matrix in parameter to the generated projection matrix.\r\n * @param matrix The matrix to updated with the projection information\r\n * @param viewMatrix The transform matrix of the light\r\n * @param renderList The list of mesh to render in the map\r\n * @returns The current light\r\n */\r\n setShadowProjectionMatrix(matrix: Matrix, viewMatrix: Matrix, renderList: Array): IShadowLight;\r\n\r\n /**\r\n * Gets the current depth scale used in ESM.\r\n * @returns The scale\r\n */\r\n getDepthScale(): number;\r\n\r\n /**\r\n * Returns whether or not the shadow generation require a cube texture or a 2d texture.\r\n * @returns true if a cube texture needs to be use\r\n */\r\n needCube(): boolean;\r\n\r\n /**\r\n * Detects if the projection matrix requires to be recomputed this frame.\r\n * @returns true if it requires to be recomputed otherwise, false.\r\n */\r\n needProjectionMatrixCompute(): boolean;\r\n\r\n /**\r\n * Forces the shadow generator to recompute the projection matrix even if position and direction did not changed.\r\n */\r\n forceProjectionMatrixCompute(): void;\r\n\r\n /**\r\n * Get the direction to use to render the shadow map. In case of cube texture, the face index can be passed.\r\n * @param faceIndex The index of the face we are computed the direction to generate shadow\r\n * @returns The set direction in 2d mode otherwise the direction to the cubemap face if needCube() is true\r\n */\r\n getShadowDirection(faceIndex?: number): Vector3;\r\n\r\n /**\r\n * Gets the minZ used for shadow according to both the scene and the light.\r\n * @param activeCamera The camera we are returning the min for\r\n * @returns the depth min z\r\n */\r\n getDepthMinZ(activeCamera: Camera): number;\r\n\r\n /**\r\n * Gets the maxZ used for shadow according to both the scene and the light.\r\n * @param activeCamera The camera we are returning the max for\r\n * @returns the depth max z\r\n */\r\n getDepthMaxZ(activeCamera: Camera): number;\r\n}\r\n\r\n/**\r\n * Base implementation IShadowLight\r\n * It groups all the common behaviour in order to reduce duplication and better follow the DRY pattern.\r\n */\r\nexport abstract class ShadowLight extends Light implements IShadowLight {\r\n protected abstract _setDefaultShadowProjectionMatrix(matrix: Matrix, viewMatrix: Matrix, renderList: Array): void;\r\n\r\n protected _position: Vector3;\r\n protected _setPosition(value: Vector3) {\r\n this._position = value;\r\n }\r\n /**\r\n * Sets the position the shadow will be casted from. Also use as the light position for both\r\n * point and spot lights.\r\n */\r\n @serializeAsVector3()\r\n public get position(): Vector3 {\r\n return this._position;\r\n }\r\n /**\r\n * Sets the position the shadow will be casted from. Also use as the light position for both\r\n * point and spot lights.\r\n */\r\n public set position(value: Vector3) {\r\n this._setPosition(value);\r\n }\r\n\r\n protected _direction: Vector3;\r\n protected _setDirection(value: Vector3) {\r\n this._direction = value;\r\n }\r\n /**\r\n * In 2d mode (needCube being false), gets the direction used to cast the shadow.\r\n * Also use as the light direction on spot and directional lights.\r\n */\r\n @serializeAsVector3()\r\n public get direction(): Vector3 {\r\n return this._direction;\r\n }\r\n /**\r\n * In 2d mode (needCube being false), sets the direction used to cast the shadow.\r\n * Also use as the light direction on spot and directional lights.\r\n */\r\n public set direction(value: Vector3) {\r\n this._setDirection(value);\r\n }\r\n\r\n protected _shadowMinZ: number;\r\n /**\r\n * Gets the shadow projection clipping minimum z value.\r\n */\r\n @serialize()\r\n public get shadowMinZ(): number {\r\n return this._shadowMinZ;\r\n }\r\n /**\r\n * Sets the shadow projection clipping minimum z value.\r\n */\r\n public set shadowMinZ(value: number) {\r\n this._shadowMinZ = value;\r\n this.forceProjectionMatrixCompute();\r\n }\r\n\r\n protected _shadowMaxZ: number;\r\n /**\r\n * Sets the shadow projection clipping maximum z value.\r\n */\r\n @serialize()\r\n public get shadowMaxZ(): number {\r\n return this._shadowMaxZ;\r\n }\r\n /**\r\n * Gets the shadow projection clipping maximum z value.\r\n */\r\n public set shadowMaxZ(value: number) {\r\n this._shadowMaxZ = value;\r\n this.forceProjectionMatrixCompute();\r\n }\r\n\r\n /**\r\n * Callback defining a custom Projection Matrix Builder.\r\n * This can be used to override the default projection matrix computation.\r\n */\r\n public customProjectionMatrixBuilder: (viewMatrix: Matrix, renderList: Array, result: Matrix) => void;\r\n\r\n /**\r\n * The transformed position. Position of the light in world space taking parenting in account.\r\n */\r\n public transformedPosition: Vector3;\r\n\r\n /**\r\n * The transformed direction. Direction of the light in world space taking parenting in account.\r\n */\r\n public transformedDirection: Vector3;\r\n\r\n private _needProjectionMatrixCompute: boolean = true;\r\n\r\n /**\r\n * Computes the transformed information (transformedPosition and transformedDirection in World space) of the current light\r\n * @returns true if the information has been computed, false if it does not need to (no parenting)\r\n */\r\n public computeTransformedInformation(): boolean {\r\n if (this.parent && this.parent.getWorldMatrix) {\r\n if (!this.transformedPosition) {\r\n this.transformedPosition = Vector3.Zero();\r\n }\r\n Vector3.TransformCoordinatesToRef(this.position, this.parent.getWorldMatrix(), this.transformedPosition);\r\n\r\n // In case the direction is present.\r\n if (this.direction) {\r\n if (!this.transformedDirection) {\r\n this.transformedDirection = Vector3.Zero();\r\n }\r\n Vector3.TransformNormalToRef(this.direction, this.parent.getWorldMatrix(), this.transformedDirection);\r\n }\r\n return true;\r\n }\r\n return false;\r\n }\r\n\r\n /**\r\n * Return the depth scale used for the shadow map.\r\n * @returns the depth scale.\r\n */\r\n public getDepthScale(): number {\r\n return 50.0;\r\n }\r\n\r\n /**\r\n * Get the direction to use to render the shadow map. In case of cube texture, the face index can be passed.\r\n * @param faceIndex The index of the face we are computed the direction to generate shadow\r\n * @returns The set direction in 2d mode otherwise the direction to the cubemap face if needCube() is true\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public getShadowDirection(faceIndex?: number): Vector3 {\r\n return this.transformedDirection ? this.transformedDirection : this.direction;\r\n }\r\n\r\n /**\r\n * Returns the ShadowLight absolute position in the World.\r\n * @returns the position vector in world space\r\n */\r\n public getAbsolutePosition(): Vector3 {\r\n return this.transformedPosition ? this.transformedPosition : this.position;\r\n }\r\n\r\n /**\r\n * Sets the ShadowLight direction toward the passed target.\r\n * @param target The point to target in local space\r\n * @returns the updated ShadowLight direction\r\n */\r\n public setDirectionToTarget(target: Vector3): Vector3 {\r\n this.direction = Vector3.Normalize(target.subtract(this.position));\r\n return this.direction;\r\n }\r\n\r\n /**\r\n * Returns the light rotation in euler definition.\r\n * @returns the x y z rotation in local space.\r\n */\r\n public getRotation(): Vector3 {\r\n this.direction.normalize();\r\n const xaxis = Vector3.Cross(this.direction, Axis.Y);\r\n const yaxis = Vector3.Cross(xaxis, this.direction);\r\n return Vector3.RotationFromAxis(xaxis, yaxis, this.direction);\r\n }\r\n\r\n /**\r\n * Returns whether or not the shadow generation require a cube texture or a 2d texture.\r\n * @returns true if a cube texture needs to be use\r\n */\r\n public needCube(): boolean {\r\n return false;\r\n }\r\n\r\n /**\r\n * Detects if the projection matrix requires to be recomputed this frame.\r\n * @returns true if it requires to be recomputed otherwise, false.\r\n */\r\n public needProjectionMatrixCompute(): boolean {\r\n return this._needProjectionMatrixCompute;\r\n }\r\n\r\n /**\r\n * Forces the shadow generator to recompute the projection matrix even if position and direction did not changed.\r\n */\r\n public forceProjectionMatrixCompute(): void {\r\n this._needProjectionMatrixCompute = true;\r\n }\r\n\r\n /** @internal */\r\n public _initCache() {\r\n super._initCache();\r\n\r\n this._cache.position = Vector3.Zero();\r\n }\r\n\r\n /** @internal */\r\n public _isSynchronized(): boolean {\r\n if (!this._cache.position.equals(this.position)) {\r\n return false;\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Computes the world matrix of the node\r\n * @param force defines if the cache version should be invalidated forcing the world matrix to be created from scratch\r\n * @returns the world matrix\r\n */\r\n public computeWorldMatrix(force?: boolean): Matrix {\r\n if (!force && this.isSynchronized()) {\r\n this._currentRenderId = this.getScene().getRenderId();\r\n return this._worldMatrix;\r\n }\r\n\r\n this._updateCache();\r\n this._cache.position.copyFrom(this.position);\r\n\r\n if (!this._worldMatrix) {\r\n this._worldMatrix = Matrix.Identity();\r\n }\r\n\r\n Matrix.TranslationToRef(this.position.x, this.position.y, this.position.z, this._worldMatrix);\r\n\r\n if (this.parent && this.parent.getWorldMatrix) {\r\n this._worldMatrix.multiplyToRef(this.parent.getWorldMatrix(), this._worldMatrix);\r\n\r\n this._markSyncedWithParent();\r\n }\r\n\r\n // Cache the determinant\r\n this._worldMatrixDeterminantIsDirty = true;\r\n\r\n return this._worldMatrix;\r\n }\r\n\r\n /**\r\n * Gets the minZ used for shadow according to both the scene and the light.\r\n * @param activeCamera The camera we are returning the min for\r\n * @returns the depth min z\r\n */\r\n public getDepthMinZ(activeCamera: Camera): number {\r\n return this.shadowMinZ !== undefined ? this.shadowMinZ : activeCamera.minZ;\r\n }\r\n\r\n /**\r\n * Gets the maxZ used for shadow according to both the scene and the light.\r\n * @param activeCamera The camera we are returning the max for\r\n * @returns the depth max z\r\n */\r\n public getDepthMaxZ(activeCamera: Camera): number {\r\n return this.shadowMaxZ !== undefined ? this.shadowMaxZ : activeCamera.maxZ;\r\n }\r\n\r\n /**\r\n * Sets the shadow projection matrix in parameter to the generated projection matrix.\r\n * @param matrix The matrix to updated with the projection information\r\n * @param viewMatrix The transform matrix of the light\r\n * @param renderList The list of mesh to render in the map\r\n * @returns The current light\r\n */\r\n public setShadowProjectionMatrix(matrix: Matrix, viewMatrix: Matrix, renderList: Array): IShadowLight {\r\n if (this.customProjectionMatrixBuilder) {\r\n this.customProjectionMatrixBuilder(viewMatrix, renderList, matrix);\r\n } else {\r\n this._setDefaultShadowProjectionMatrix(matrix, viewMatrix, renderList);\r\n }\r\n return this;\r\n }\r\n\r\n /** @internal */\r\n protected _syncParentEnabledState() {\r\n super._syncParentEnabledState();\r\n if (!this.parent || !this.parent.getWorldMatrix) {\r\n (this.transformedPosition as any) = null;\r\n (this.transformedDirection as any) = null;\r\n }\r\n }\r\n}\r\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"bayerDitherFunctions\";\nconst shader = `float bayerDither2(vec2 _P) {return mod(2.0*_P.y+_P.x+1.0,4.0);}\nfloat bayerDither4(vec2 _P) {vec2 P1=mod(_P,2.0); \nvec2 P2=floor(0.5*mod(_P,4.0)); \nreturn 4.0*bayerDither2(P1)+bayerDither2(P2);}\nfloat bayerDither8(vec2 _P) {vec2 P1=mod(_P,2.0); \nvec2 P2=floor(0.5 *mod(_P,4.0)); \nvec2 P4=floor(0.25*mod(_P,8.0)); \nreturn 4.0*(4.0*bayerDither2(P1)+bayerDither2(P2))+bayerDither2(P4);}\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const bayerDitherFunctions = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\nimport \"./packingFunctions\";\nimport \"./bayerDitherFunctions\";\n\nconst name = \"shadowMapFragmentExtraDeclaration\";\nconst shader = `#if SM_FLOAT==0\n#include\n#endif\n#if SM_SOFTTRANSPARENTSHADOW==1\n#include\nuniform float softTransparentShadowSM;\n#endif\nvarying float vDepthMetricSM;\n#if SM_USEDISTANCE==1\nuniform vec3 lightDataSM;varying vec3 vPositionWSM;\n#endif\nuniform vec3 biasAndScaleSM;uniform vec2 depthValuesSM;\n#if defined(SM_DEPTHCLAMP) && SM_DEPTHCLAMP==1\nvarying float zSM;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const shadowMapFragmentExtraDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"shadowMapFragment\";\nconst shader = `float depthSM=vDepthMetricSM;\n#if defined(SM_DEPTHCLAMP) && SM_DEPTHCLAMP==1\n#if SM_USEDISTANCE==1\ndepthSM=(length(vPositionWSM-lightDataSM)+depthValuesSM.x)/depthValuesSM.y+biasAndScaleSM.x;\n#else\n#ifdef USE_REVERSE_DEPTHBUFFER\ndepthSM=(-zSM+depthValuesSM.x)/depthValuesSM.y+biasAndScaleSM.x;\n#else\ndepthSM=(zSM+depthValuesSM.x)/depthValuesSM.y+biasAndScaleSM.x;\n#endif\n#endif\n#ifdef USE_REVERSE_DEPTHBUFFER\ngl_FragDepth=clamp(1.0-depthSM,0.0,1.0);\n#else\ngl_FragDepth=clamp(depthSM,0.0,1.0); \n#endif\n#elif SM_USEDISTANCE==1\ndepthSM=(length(vPositionWSM-lightDataSM)+depthValuesSM.x)/depthValuesSM.y+biasAndScaleSM.x;\n#endif\n#if SM_ESM==1\ndepthSM=clamp(exp(-min(87.,biasAndScaleSM.z*depthSM)),0.,1.);\n#endif\n#if SM_FLOAT==1\ngl_FragColor=vec4(depthSM,1.0,1.0,1.0);\n#else\ngl_FragColor=pack(depthSM);\n#endif\nreturn;`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const shadowMapFragment = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/shadowMapFragmentExtraDeclaration\";\nimport \"./ShadersInclude/clipPlaneFragmentDeclaration\";\nimport \"./ShadersInclude/clipPlaneFragment\";\nimport \"./ShadersInclude/shadowMapFragment\";\n\nconst name = \"shadowMapPixelShader\";\nconst shader = `#include\n#ifdef ALPHATEXTURE\nvarying vec2 vUV;uniform sampler2D diffuseSampler;\n#endif\n#include\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{\n#include\n#ifdef ALPHATEXTURE\nfloat alphaFromAlphaTexture=texture2D(diffuseSampler,vUV).a;\n#ifdef ALPHATESTVALUE\nif (alphaFromAlphaTexture=softTransparentShadowSM*alphaFromAlphaTexture) discard;\n#else\nif ((bayerDither8(floor(mod(gl_FragCoord.xy,8.0))))/64.0>=softTransparentShadowSM) discard;\n#endif\n#endif\n#include\n}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const shadowMapPixelShader = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"sceneVertexDeclaration\";\nconst shader = `uniform mat4 viewProjection;\n#ifdef MULTIVIEW\nuniform mat4 viewProjectionR;\n#endif\nuniform mat4 view;uniform mat4 projection;uniform vec4 vEyePosition;\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const sceneVertexDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"meshVertexDeclaration\";\nconst shader = `uniform mat4 world;uniform float visibility;\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const meshVertexDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\nimport \"./sceneVertexDeclaration\";\nimport \"./meshVertexDeclaration\";\n\nconst name = \"shadowMapVertexDeclaration\";\nconst shader = `#include\n#include\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const shadowMapVertexDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\nimport \"./sceneUboDeclaration\";\nimport \"./meshUboDeclaration\";\n\nconst name = \"shadowMapUboDeclaration\";\nconst shader = `layout(std140,column_major) uniform;\n#include\n#include\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const shadowMapUboDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"shadowMapVertexExtraDeclaration\";\nconst shader = `#if SM_NORMALBIAS==1\nuniform vec3 lightDataSM;\n#endif\nuniform vec3 biasAndScaleSM;uniform vec2 depthValuesSM;varying float vDepthMetricSM;\n#if SM_USEDISTANCE==1\nvarying vec3 vPositionWSM;\n#endif\n#if defined(SM_DEPTHCLAMP) && SM_DEPTHCLAMP==1\nvarying float zSM;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const shadowMapVertexExtraDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"shadowMapVertexNormalBias\";\nconst shader = `#if SM_NORMALBIAS==1\n#if SM_DIRECTIONINLIGHTDATA==1\nvec3 worldLightDirSM=normalize(-lightDataSM.xyz);\n#else\nvec3 directionToLightSM=lightDataSM.xyz-worldPos.xyz;vec3 worldLightDirSM=normalize(directionToLightSM);\n#endif\nfloat ndlSM=dot(vNormalW,worldLightDirSM);float sinNLSM=sqrt(1.0-ndlSM*ndlSM);float normalBiasSM=biasAndScaleSM.y*sinNLSM;worldPos.xyz-=vNormalW*normalBiasSM;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const shadowMapVertexNormalBias = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"shadowMapVertexMetric\";\nconst shader = `#if SM_USEDISTANCE==1\nvPositionWSM=worldPos.xyz;\n#endif\n#if SM_DEPTHTEXTURE==1\n#ifdef IS_NDC_HALF_ZRANGE\n#define BIASFACTOR 0.5\n#else\n#define BIASFACTOR 1.0\n#endif\n#ifdef USE_REVERSE_DEPTHBUFFER\ngl_Position.z-=biasAndScaleSM.x*gl_Position.w*BIASFACTOR;\n#else\ngl_Position.z+=biasAndScaleSM.x*gl_Position.w*BIASFACTOR;\n#endif\n#endif\n#if defined(SM_DEPTHCLAMP) && SM_DEPTHCLAMP==1\nzSM=gl_Position.z;gl_Position.z=0.0;\n#elif SM_USEDISTANCE==0\n#ifdef USE_REVERSE_DEPTHBUFFER\nvDepthMetricSM=(-gl_Position.z+depthValuesSM.x)/depthValuesSM.y+biasAndScaleSM.x;\n#else\nvDepthMetricSM=(gl_Position.z+depthValuesSM.x)/depthValuesSM.y+biasAndScaleSM.x;\n#endif\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const shadowMapVertexMetric = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/bonesDeclaration\";\nimport \"./ShadersInclude/bakedVertexAnimationDeclaration\";\nimport \"./ShadersInclude/morphTargetsVertexGlobalDeclaration\";\nimport \"./ShadersInclude/morphTargetsVertexDeclaration\";\nimport \"./ShadersInclude/helperFunctions\";\nimport \"./ShadersInclude/shadowMapVertexDeclaration\";\nimport \"./ShadersInclude/shadowMapUboDeclaration\";\nimport \"./ShadersInclude/shadowMapVertexExtraDeclaration\";\nimport \"./ShadersInclude/clipPlaneVertexDeclaration\";\nimport \"./ShadersInclude/morphTargetsVertexGlobal\";\nimport \"./ShadersInclude/morphTargetsVertex\";\nimport \"./ShadersInclude/instancesVertex\";\nimport \"./ShadersInclude/bonesVertex\";\nimport \"./ShadersInclude/bakedVertexAnimation\";\nimport \"./ShadersInclude/shadowMapVertexNormalBias\";\nimport \"./ShadersInclude/shadowMapVertexMetric\";\nimport \"./ShadersInclude/clipPlaneVertex\";\n\nconst name = \"shadowMapVertexShader\";\nconst shader = `attribute vec3 position;\n#ifdef NORMAL\nattribute vec3 normal;\n#endif\n#include\n#include\n#include\n#include[0..maxSimultaneousMorphTargets]\n#ifdef INSTANCES\nattribute vec4 world0;attribute vec4 world1;attribute vec4 world2;attribute vec4 world3;\n#endif\n#include\n#include<__decl__shadowMapVertex>\n#ifdef ALPHATEXTURE\nvarying vec2 vUV;uniform mat4 diffuseMatrix;\n#ifdef UV1\nattribute vec2 uv;\n#endif\n#ifdef UV2\nattribute vec2 uv2;\n#endif\n#endif\n#include\n#include\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void)\n{vec3 positionUpdated=position;\n#ifdef UV1\nvec2 uvUpdated=uv;\n#endif\n#ifdef NORMAL\nvec3 normalUpdated=normal;\n#endif\n#include\n#include[0..maxSimultaneousMorphTargets]\n#include\n#include\n#include\nvec4 worldPos=finalWorld*vec4(positionUpdated,1.0);\n#ifdef NORMAL\nmat3 normWorldSM=mat3(finalWorld);\n#if defined(INSTANCES) && defined(THIN_INSTANCES)\nvec3 vNormalW=normalUpdated/vec3(dot(normWorldSM[0],normWorldSM[0]),dot(normWorldSM[1],normWorldSM[1]),dot(normWorldSM[2],normWorldSM[2]));vNormalW=normalize(normWorldSM*vNormalW);\n#else\n#ifdef NONUNIFORMSCALING\nnormWorldSM=transposeMat3(inverseMat3(normWorldSM));\n#endif\nvec3 vNormalW=normalize(normWorldSM*normalUpdated);\n#endif\n#endif\n#include\ngl_Position=viewProjection*worldPos;\n#include\n#ifdef ALPHATEXTURE\n#ifdef UV1\nvUV=vec2(diffuseMatrix*vec4(uvUpdated,1.0,0.0));\n#endif\n#ifdef UV2\nvUV=vec2(diffuseMatrix*vec4(uv2,1.0,0.0));\n#endif\n#endif\n#include\n}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const shadowMapVertexShader = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\n\nconst name = \"depthBoxBlurPixelShader\";\nconst shader = `varying vec2 vUV;uniform sampler2D textureSampler;uniform vec2 screenSize;\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{vec4 colorDepth=vec4(0.0);for (int x=-OFFSET; x<=OFFSET; x++)\nfor (int y=-OFFSET; y<=OFFSET; y++)\ncolorDepth+=texture2D(textureSampler,vUV+vec2(x,y)/screenSize);gl_FragColor=(colorDepth/float((OFFSET*2+1)*(OFFSET*2+1)));}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const depthBoxBlurPixelShader = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"shadowMapFragmentSoftTransparentShadow\";\nconst shader = `#if SM_SOFTTRANSPARENTSHADOW==1\nif ((bayerDither8(floor(mod(gl_FragCoord.xy,8.0))))/64.0>=softTransparentShadowSM*alpha) discard;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const shadowMapFragmentSoftTransparentShadow = { name, shader };\n","import type { SmartArray } from \"../../Misc/smartArray\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { Matrix, Vector3, Vector2 } from \"../../Maths/math.vector\";\r\nimport { Color4 } from \"../../Maths/math.color\";\r\nimport { VertexBuffer } from \"../../Buffers/buffer\";\r\nimport type { SubMesh } from \"../../Meshes/subMesh\";\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport type { Mesh } from \"../../Meshes/mesh\";\r\n\r\nimport type { IShadowLight } from \"../../Lights/shadowLight\";\r\nimport { Light } from \"../../Lights/light\";\r\nimport type { MaterialDefines } from \"../../Materials/materialDefines\";\r\nimport { MaterialHelper } from \"../../Materials/materialHelper\";\r\nimport type { Effect, IEffectCreationOptions } from \"../../Materials/effect\";\r\nimport { Texture } from \"../../Materials/Textures/texture\";\r\nimport { RenderTargetTexture } from \"../../Materials/Textures/renderTargetTexture\";\r\n\r\nimport { PostProcess } from \"../../PostProcesses/postProcess\";\r\nimport { BlurPostProcess } from \"../../PostProcesses/blurPostProcess\";\r\nimport { Constants } from \"../../Engines/constants\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport { _WarnImport } from \"../../Misc/devTools\";\r\nimport { EffectFallbacks } from \"../../Materials/effectFallbacks\";\r\nimport { RenderingManager } from \"../../Rendering/renderingManager\";\r\nimport { DrawWrapper } from \"../../Materials/drawWrapper\";\r\nimport type { UniformBuffer } from \"../../Materials/uniformBuffer\";\r\nimport type { Camera } from \"../../Cameras/camera\";\r\n\r\nimport \"../../Shaders/shadowMap.fragment\";\r\nimport \"../../Shaders/shadowMap.vertex\";\r\nimport \"../../Shaders/depthBoxBlur.fragment\";\r\nimport \"../../Shaders/ShadersInclude/shadowMapFragmentSoftTransparentShadow\";\r\nimport { addClipPlaneUniforms, bindClipPlane, prepareStringDefinesForClipPlanes } from \"../../Materials/clipPlaneMaterialHelper\";\r\nimport type { BaseTexture } from \"../../Materials/Textures/baseTexture\";\r\n\r\n/**\r\n * Defines the options associated with the creation of a custom shader for a shadow generator.\r\n */\r\nexport interface ICustomShaderOptions {\r\n /**\r\n * Gets or sets the custom shader name to use\r\n */\r\n shaderName: string;\r\n\r\n /**\r\n * The list of attribute names used in the shader\r\n */\r\n attributes?: string[];\r\n\r\n /**\r\n * The list of uniform names used in the shader\r\n */\r\n uniforms?: string[];\r\n\r\n /**\r\n * The list of sampler names used in the shader\r\n */\r\n samplers?: string[];\r\n\r\n /**\r\n * The list of defines used in the shader\r\n */\r\n defines?: string[];\r\n}\r\n\r\n/**\r\n * Interface to implement to create a shadow generator compatible with BJS.\r\n */\r\nexport interface IShadowGenerator {\r\n /** Gets or set the id of the shadow generator. It will be the one from the light if not defined */\r\n id: string;\r\n /**\r\n * Gets the main RTT containing the shadow map (usually storing depth from the light point of view).\r\n * @returns The render target texture if present otherwise, null\r\n */\r\n getShadowMap(): Nullable;\r\n\r\n /**\r\n * Determine whether the shadow generator is ready or not (mainly all effects and related post processes needs to be ready).\r\n * @param subMesh The submesh we want to render in the shadow map\r\n * @param useInstances Defines whether will draw in the map using instances\r\n * @param isTransparent Indicates that isReady is called for a transparent subMesh\r\n * @returns true if ready otherwise, false\r\n */\r\n isReady(subMesh: SubMesh, useInstances: boolean, isTransparent: boolean): boolean;\r\n\r\n /**\r\n * Prepare all the defines in a material relying on a shadow map at the specified light index.\r\n * @param defines Defines of the material we want to update\r\n * @param lightIndex Index of the light in the enabled light list of the material\r\n */\r\n prepareDefines(defines: MaterialDefines, lightIndex: number): void;\r\n /**\r\n * Binds the shadow related information inside of an effect (information like near, far, darkness...\r\n * defined in the generator but impacting the effect).\r\n * It implies the uniforms available on the materials are the standard BJS ones.\r\n * @param lightIndex Index of the light in the enabled light list of the material owning the effect\r\n * @param effect The effect we are binding the information for\r\n */\r\n bindShadowLight(lightIndex: string, effect: Effect): void;\r\n /**\r\n * Gets the transformation matrix used to project the meshes into the map from the light point of view.\r\n * (eq to shadow projection matrix * light transform matrix)\r\n * @returns The transform matrix used to create the shadow map\r\n */\r\n getTransformMatrix(): Matrix;\r\n\r\n /**\r\n * Recreates the shadow map dependencies like RTT and post processes. This can be used during the switch between\r\n * Cube and 2D textures for instance.\r\n */\r\n recreateShadowMap(): void;\r\n\r\n /**\r\n * Forces all the attached effect to compile to enable rendering only once ready vs. lazily compiling effects.\r\n * @param onCompiled Callback triggered at the and of the effects compilation\r\n * @param options Sets of optional options forcing the compilation with different modes\r\n */\r\n forceCompilation(onCompiled?: (generator: IShadowGenerator) => void, options?: Partial<{ useInstances: boolean }>): void;\r\n\r\n /**\r\n * Forces all the attached effect to compile to enable rendering only once ready vs. lazily compiling effects.\r\n * @param options Sets of optional options forcing the compilation with different modes\r\n * @returns A promise that resolves when the compilation completes\r\n */\r\n forceCompilationAsync(options?: Partial<{ useInstances: boolean }>): Promise;\r\n\r\n /**\r\n * Serializes the shadow generator setup to a json object.\r\n * @returns The serialized JSON object\r\n */\r\n serialize(): any;\r\n\r\n /**\r\n * Disposes the Shadow map and related Textures and effects.\r\n */\r\n dispose(): void;\r\n}\r\n\r\n/**\r\n * Default implementation IShadowGenerator.\r\n * This is the main object responsible of generating shadows in the framework.\r\n * Documentation: https://doc.babylonjs.com/features/featuresDeepDive/lights/shadows\r\n */\r\nexport class ShadowGenerator implements IShadowGenerator {\r\n /**\r\n * Name of the shadow generator class\r\n */\r\n public static CLASSNAME = \"ShadowGenerator\";\r\n\r\n /**\r\n * Shadow generator mode None: no filtering applied.\r\n */\r\n public static readonly FILTER_NONE = 0;\r\n /**\r\n * Shadow generator mode ESM: Exponential Shadow Mapping.\r\n * (http://developer.download.nvidia.com/presentations/2008/GDC/GDC08_SoftShadowMapping.pdf)\r\n */\r\n public static readonly FILTER_EXPONENTIALSHADOWMAP = 1;\r\n /**\r\n * Shadow generator mode Poisson Sampling: Percentage Closer Filtering.\r\n * (Multiple Tap around evenly distributed around the pixel are used to evaluate the shadow strength)\r\n */\r\n public static readonly FILTER_POISSONSAMPLING = 2;\r\n /**\r\n * Shadow generator mode ESM: Blurred Exponential Shadow Mapping.\r\n * (http://developer.download.nvidia.com/presentations/2008/GDC/GDC08_SoftShadowMapping.pdf)\r\n */\r\n public static readonly FILTER_BLUREXPONENTIALSHADOWMAP = 3;\r\n /**\r\n * Shadow generator mode ESM: Exponential Shadow Mapping using the inverse of the exponential preventing\r\n * edge artifacts on steep falloff.\r\n * (http://developer.download.nvidia.com/presentations/2008/GDC/GDC08_SoftShadowMapping.pdf)\r\n */\r\n public static readonly FILTER_CLOSEEXPONENTIALSHADOWMAP = 4;\r\n /**\r\n * Shadow generator mode ESM: Blurred Exponential Shadow Mapping using the inverse of the exponential preventing\r\n * edge artifacts on steep falloff.\r\n * (http://developer.download.nvidia.com/presentations/2008/GDC/GDC08_SoftShadowMapping.pdf)\r\n */\r\n public static readonly FILTER_BLURCLOSEEXPONENTIALSHADOWMAP = 5;\r\n /**\r\n * Shadow generator mode PCF: Percentage Closer Filtering\r\n * benefits from Webgl 2 shadow samplers. Fallback to Poisson Sampling in Webgl 1\r\n * (https://developer.nvidia.com/gpugems/GPUGems/gpugems_ch11.html)\r\n */\r\n public static readonly FILTER_PCF = 6;\r\n /**\r\n * Shadow generator mode PCSS: Percentage Closering Soft Shadow.\r\n * benefits from Webgl 2 shadow samplers. Fallback to Poisson Sampling in Webgl 1\r\n * Contact Hardening\r\n */\r\n public static readonly FILTER_PCSS = 7;\r\n\r\n /**\r\n * Reserved for PCF and PCSS\r\n * Highest Quality.\r\n *\r\n * Execute PCF on a 5*5 kernel improving a lot the shadow aliasing artifacts.\r\n *\r\n * Execute PCSS with 32 taps blocker search and 64 taps PCF.\r\n */\r\n public static readonly QUALITY_HIGH = 0;\r\n /**\r\n * Reserved for PCF and PCSS\r\n * Good tradeoff for quality/perf cross devices\r\n *\r\n * Execute PCF on a 3*3 kernel.\r\n *\r\n * Execute PCSS with 16 taps blocker search and 32 taps PCF.\r\n */\r\n public static readonly QUALITY_MEDIUM = 1;\r\n /**\r\n * Reserved for PCF and PCSS\r\n * The lowest quality but the fastest.\r\n *\r\n * Execute PCF on a 1*1 kernel.\r\n *\r\n * Execute PCSS with 16 taps blocker search and 16 taps PCF.\r\n */\r\n public static readonly QUALITY_LOW = 2;\r\n\r\n /**\r\n * Defines the default alpha cutoff value used for transparent alpha tested materials.\r\n */\r\n public static DEFAULT_ALPHA_CUTOFF = 0.5;\r\n\r\n /** Gets or set the id of the shadow generator. It will be the one from the light if not defined */\r\n public id: string;\r\n\r\n /** Gets or sets the custom shader name to use */\r\n public customShaderOptions: ICustomShaderOptions;\r\n\r\n /** Gets or sets a custom function to allow/disallow rendering a sub mesh in the shadow map */\r\n public customAllowRendering: (subMesh: SubMesh) => boolean;\r\n\r\n /**\r\n * Observable triggered before the shadow is rendered. Can be used to update internal effect state\r\n */\r\n public onBeforeShadowMapRenderObservable = new Observable();\r\n\r\n /**\r\n * Observable triggered after the shadow is rendered. Can be used to restore internal effect state\r\n */\r\n public onAfterShadowMapRenderObservable = new Observable();\r\n\r\n /**\r\n * Observable triggered before a mesh is rendered in the shadow map.\r\n * Can be used to update internal effect state (that you can get from the onBeforeShadowMapRenderObservable)\r\n */\r\n public onBeforeShadowMapRenderMeshObservable = new Observable();\r\n\r\n /**\r\n * Observable triggered after a mesh is rendered in the shadow map.\r\n * Can be used to update internal effect state (that you can get from the onAfterShadowMapRenderObservable)\r\n */\r\n public onAfterShadowMapRenderMeshObservable = new Observable();\r\n\r\n protected _bias = 0.00005;\r\n /**\r\n * Gets the bias: offset applied on the depth preventing acnea (in light direction).\r\n */\r\n public get bias(): number {\r\n return this._bias;\r\n }\r\n /**\r\n * Sets the bias: offset applied on the depth preventing acnea (in light direction).\r\n */\r\n public set bias(bias: number) {\r\n this._bias = bias;\r\n }\r\n\r\n protected _normalBias = 0;\r\n /**\r\n * Gets the normalBias: offset applied on the depth preventing acnea (along side the normal direction and proportional to the light/normal angle).\r\n */\r\n public get normalBias(): number {\r\n return this._normalBias;\r\n }\r\n /**\r\n * Sets the normalBias: offset applied on the depth preventing acnea (along side the normal direction and proportional to the light/normal angle).\r\n */\r\n public set normalBias(normalBias: number) {\r\n this._normalBias = normalBias;\r\n }\r\n\r\n protected _blurBoxOffset = 1;\r\n /**\r\n * Gets the blur box offset: offset applied during the blur pass.\r\n * Only useful if useKernelBlur = false\r\n */\r\n public get blurBoxOffset(): number {\r\n return this._blurBoxOffset;\r\n }\r\n /**\r\n * Sets the blur box offset: offset applied during the blur pass.\r\n * Only useful if useKernelBlur = false\r\n */\r\n public set blurBoxOffset(value: number) {\r\n if (this._blurBoxOffset === value) {\r\n return;\r\n }\r\n\r\n this._blurBoxOffset = value;\r\n this._disposeBlurPostProcesses();\r\n }\r\n\r\n protected _blurScale = 2;\r\n /**\r\n * Gets the blur scale: scale of the blurred texture compared to the main shadow map.\r\n * 2 means half of the size.\r\n */\r\n public get blurScale(): number {\r\n return this._blurScale;\r\n }\r\n /**\r\n * Sets the blur scale: scale of the blurred texture compared to the main shadow map.\r\n * 2 means half of the size.\r\n */\r\n public set blurScale(value: number) {\r\n if (this._blurScale === value) {\r\n return;\r\n }\r\n\r\n this._blurScale = value;\r\n this._disposeBlurPostProcesses();\r\n }\r\n\r\n protected _blurKernel = 1;\r\n /**\r\n * Gets the blur kernel: kernel size of the blur pass.\r\n * Only useful if useKernelBlur = true\r\n */\r\n public get blurKernel(): number {\r\n return this._blurKernel;\r\n }\r\n /**\r\n * Sets the blur kernel: kernel size of the blur pass.\r\n * Only useful if useKernelBlur = true\r\n */\r\n public set blurKernel(value: number) {\r\n if (this._blurKernel === value) {\r\n return;\r\n }\r\n\r\n this._blurKernel = value;\r\n this._disposeBlurPostProcesses();\r\n }\r\n\r\n protected _useKernelBlur = false;\r\n /**\r\n * Gets whether the blur pass is a kernel blur (if true) or box blur.\r\n * Only useful in filtered mode (useBlurExponentialShadowMap...)\r\n */\r\n public get useKernelBlur(): boolean {\r\n return this._useKernelBlur;\r\n }\r\n /**\r\n * Sets whether the blur pass is a kernel blur (if true) or box blur.\r\n * Only useful in filtered mode (useBlurExponentialShadowMap...)\r\n */\r\n public set useKernelBlur(value: boolean) {\r\n if (this._useKernelBlur === value) {\r\n return;\r\n }\r\n\r\n this._useKernelBlur = value;\r\n this._disposeBlurPostProcesses();\r\n }\r\n\r\n protected _depthScale: number;\r\n /**\r\n * Gets the depth scale used in ESM mode.\r\n */\r\n public get depthScale(): number {\r\n return this._depthScale !== undefined ? this._depthScale : this._light.getDepthScale();\r\n }\r\n /**\r\n * Sets the depth scale used in ESM mode.\r\n * This can override the scale stored on the light.\r\n */\r\n public set depthScale(value: number) {\r\n this._depthScale = value;\r\n }\r\n\r\n protected _validateFilter(filter: number): number {\r\n return filter;\r\n }\r\n\r\n protected _filter = ShadowGenerator.FILTER_NONE;\r\n /**\r\n * Gets the current mode of the shadow generator (normal, PCF, ESM...).\r\n * The returned value is a number equal to one of the available mode defined in ShadowMap.FILTER_x like _FILTER_NONE\r\n */\r\n public get filter(): number {\r\n return this._filter;\r\n }\r\n /**\r\n * Sets the current mode of the shadow generator (normal, PCF, ESM...).\r\n * The returned value is a number equal to one of the available mode defined in ShadowMap.FILTER_x like _FILTER_NONE\r\n */\r\n public set filter(value: number) {\r\n value = this._validateFilter(value);\r\n\r\n // Blurring the cubemap is going to be too expensive. Reverting to unblurred version\r\n if (this._light.needCube()) {\r\n if (value === ShadowGenerator.FILTER_BLUREXPONENTIALSHADOWMAP) {\r\n this.useExponentialShadowMap = true;\r\n return;\r\n } else if (value === ShadowGenerator.FILTER_BLURCLOSEEXPONENTIALSHADOWMAP) {\r\n this.useCloseExponentialShadowMap = true;\r\n return;\r\n }\r\n // PCF on cubemap would also be expensive\r\n else if (value === ShadowGenerator.FILTER_PCF || value === ShadowGenerator.FILTER_PCSS) {\r\n this.usePoissonSampling = true;\r\n return;\r\n }\r\n }\r\n\r\n // Weblg1 fallback for PCF.\r\n if (value === ShadowGenerator.FILTER_PCF || value === ShadowGenerator.FILTER_PCSS) {\r\n if (!this._scene.getEngine()._features.supportShadowSamplers) {\r\n this.usePoissonSampling = true;\r\n return;\r\n }\r\n }\r\n\r\n if (this._filter === value) {\r\n return;\r\n }\r\n\r\n this._filter = value;\r\n this._disposeBlurPostProcesses();\r\n this._applyFilterValues();\r\n this._light._markMeshesAsLightDirty();\r\n }\r\n\r\n /**\r\n * Gets if the current filter is set to Poisson Sampling.\r\n */\r\n public get usePoissonSampling(): boolean {\r\n return this.filter === ShadowGenerator.FILTER_POISSONSAMPLING;\r\n }\r\n /**\r\n * Sets the current filter to Poisson Sampling.\r\n */\r\n public set usePoissonSampling(value: boolean) {\r\n const filter = this._validateFilter(ShadowGenerator.FILTER_POISSONSAMPLING);\r\n\r\n if (!value && this.filter !== ShadowGenerator.FILTER_POISSONSAMPLING) {\r\n return;\r\n }\r\n\r\n this.filter = value ? filter : ShadowGenerator.FILTER_NONE;\r\n }\r\n\r\n /**\r\n * Gets if the current filter is set to ESM.\r\n */\r\n public get useExponentialShadowMap(): boolean {\r\n return this.filter === ShadowGenerator.FILTER_EXPONENTIALSHADOWMAP;\r\n }\r\n /**\r\n * Sets the current filter is to ESM.\r\n */\r\n public set useExponentialShadowMap(value: boolean) {\r\n const filter = this._validateFilter(ShadowGenerator.FILTER_EXPONENTIALSHADOWMAP);\r\n\r\n if (!value && this.filter !== ShadowGenerator.FILTER_EXPONENTIALSHADOWMAP) {\r\n return;\r\n }\r\n this.filter = value ? filter : ShadowGenerator.FILTER_NONE;\r\n }\r\n\r\n /**\r\n * Gets if the current filter is set to filtered ESM.\r\n */\r\n public get useBlurExponentialShadowMap(): boolean {\r\n return this.filter === ShadowGenerator.FILTER_BLUREXPONENTIALSHADOWMAP;\r\n }\r\n /**\r\n * Gets if the current filter is set to filtered ESM.\r\n */\r\n public set useBlurExponentialShadowMap(value: boolean) {\r\n const filter = this._validateFilter(ShadowGenerator.FILTER_BLUREXPONENTIALSHADOWMAP);\r\n\r\n if (!value && this.filter !== ShadowGenerator.FILTER_BLUREXPONENTIALSHADOWMAP) {\r\n return;\r\n }\r\n this.filter = value ? filter : ShadowGenerator.FILTER_NONE;\r\n }\r\n\r\n /**\r\n * Gets if the current filter is set to \"close ESM\" (using the inverse of the\r\n * exponential to prevent steep falloff artifacts).\r\n */\r\n public get useCloseExponentialShadowMap(): boolean {\r\n return this.filter === ShadowGenerator.FILTER_CLOSEEXPONENTIALSHADOWMAP;\r\n }\r\n /**\r\n * Sets the current filter to \"close ESM\" (using the inverse of the\r\n * exponential to prevent steep falloff artifacts).\r\n */\r\n public set useCloseExponentialShadowMap(value: boolean) {\r\n const filter = this._validateFilter(ShadowGenerator.FILTER_CLOSEEXPONENTIALSHADOWMAP);\r\n\r\n if (!value && this.filter !== ShadowGenerator.FILTER_CLOSEEXPONENTIALSHADOWMAP) {\r\n return;\r\n }\r\n this.filter = value ? filter : ShadowGenerator.FILTER_NONE;\r\n }\r\n\r\n /**\r\n * Gets if the current filter is set to filtered \"close ESM\" (using the inverse of the\r\n * exponential to prevent steep falloff artifacts).\r\n */\r\n public get useBlurCloseExponentialShadowMap(): boolean {\r\n return this.filter === ShadowGenerator.FILTER_BLURCLOSEEXPONENTIALSHADOWMAP;\r\n }\r\n /**\r\n * Sets the current filter to filtered \"close ESM\" (using the inverse of the\r\n * exponential to prevent steep falloff artifacts).\r\n */\r\n public set useBlurCloseExponentialShadowMap(value: boolean) {\r\n const filter = this._validateFilter(ShadowGenerator.FILTER_BLURCLOSEEXPONENTIALSHADOWMAP);\r\n\r\n if (!value && this.filter !== ShadowGenerator.FILTER_BLURCLOSEEXPONENTIALSHADOWMAP) {\r\n return;\r\n }\r\n this.filter = value ? filter : ShadowGenerator.FILTER_NONE;\r\n }\r\n\r\n /**\r\n * Gets if the current filter is set to \"PCF\" (percentage closer filtering).\r\n */\r\n public get usePercentageCloserFiltering(): boolean {\r\n return this.filter === ShadowGenerator.FILTER_PCF;\r\n }\r\n /**\r\n * Sets the current filter to \"PCF\" (percentage closer filtering).\r\n */\r\n public set usePercentageCloserFiltering(value: boolean) {\r\n const filter = this._validateFilter(ShadowGenerator.FILTER_PCF);\r\n\r\n if (!value && this.filter !== ShadowGenerator.FILTER_PCF) {\r\n return;\r\n }\r\n this.filter = value ? filter : ShadowGenerator.FILTER_NONE;\r\n }\r\n\r\n protected _filteringQuality = ShadowGenerator.QUALITY_HIGH;\r\n /**\r\n * Gets the PCF or PCSS Quality.\r\n * Only valid if usePercentageCloserFiltering or usePercentageCloserFiltering is true.\r\n */\r\n public get filteringQuality(): number {\r\n return this._filteringQuality;\r\n }\r\n /**\r\n * Sets the PCF or PCSS Quality.\r\n * Only valid if usePercentageCloserFiltering or usePercentageCloserFiltering is true.\r\n */\r\n public set filteringQuality(filteringQuality: number) {\r\n if (this._filteringQuality === filteringQuality) {\r\n return;\r\n }\r\n\r\n this._filteringQuality = filteringQuality;\r\n\r\n this._disposeBlurPostProcesses();\r\n this._applyFilterValues();\r\n this._light._markMeshesAsLightDirty();\r\n }\r\n\r\n /**\r\n * Gets if the current filter is set to \"PCSS\" (contact hardening).\r\n */\r\n public get useContactHardeningShadow(): boolean {\r\n return this.filter === ShadowGenerator.FILTER_PCSS;\r\n }\r\n /**\r\n * Sets the current filter to \"PCSS\" (contact hardening).\r\n */\r\n public set useContactHardeningShadow(value: boolean) {\r\n const filter = this._validateFilter(ShadowGenerator.FILTER_PCSS);\r\n\r\n if (!value && this.filter !== ShadowGenerator.FILTER_PCSS) {\r\n return;\r\n }\r\n this.filter = value ? filter : ShadowGenerator.FILTER_NONE;\r\n }\r\n\r\n protected _contactHardeningLightSizeUVRatio = 0.1;\r\n /**\r\n * Gets the Light Size (in shadow map uv unit) used in PCSS to determine the blocker search area and the penumbra size.\r\n * Using a ratio helps keeping shape stability independently of the map size.\r\n *\r\n * It does not account for the light projection as it was having too much\r\n * instability during the light setup or during light position changes.\r\n *\r\n * Only valid if useContactHardeningShadow is true.\r\n */\r\n public get contactHardeningLightSizeUVRatio(): number {\r\n return this._contactHardeningLightSizeUVRatio;\r\n }\r\n /**\r\n * Sets the Light Size (in shadow map uv unit) used in PCSS to determine the blocker search area and the penumbra size.\r\n * Using a ratio helps keeping shape stability independently of the map size.\r\n *\r\n * It does not account for the light projection as it was having too much\r\n * instability during the light setup or during light position changes.\r\n *\r\n * Only valid if useContactHardeningShadow is true.\r\n */\r\n public set contactHardeningLightSizeUVRatio(contactHardeningLightSizeUVRatio: number) {\r\n this._contactHardeningLightSizeUVRatio = contactHardeningLightSizeUVRatio;\r\n }\r\n\r\n protected _darkness = 0;\r\n\r\n /** Gets or sets the actual darkness of a shadow */\r\n public get darkness() {\r\n return this._darkness;\r\n }\r\n\r\n public set darkness(value: number) {\r\n this.setDarkness(value);\r\n }\r\n\r\n /**\r\n * Returns the darkness value (float). This can only decrease the actual darkness of a shadow.\r\n * 0 means strongest and 1 would means no shadow.\r\n * @returns the darkness.\r\n */\r\n public getDarkness(): number {\r\n return this._darkness;\r\n }\r\n /**\r\n * Sets the darkness value (float). This can only decrease the actual darkness of a shadow.\r\n * @param darkness The darkness value 0 means strongest and 1 would means no shadow.\r\n * @returns the shadow generator allowing fluent coding.\r\n */\r\n public setDarkness(darkness: number): ShadowGenerator {\r\n if (darkness >= 1.0) {\r\n this._darkness = 1.0;\r\n } else if (darkness <= 0.0) {\r\n this._darkness = 0.0;\r\n } else {\r\n this._darkness = darkness;\r\n }\r\n return this;\r\n }\r\n\r\n protected _transparencyShadow = false;\r\n\r\n /** Gets or sets the ability to have transparent shadow */\r\n public get transparencyShadow() {\r\n return this._transparencyShadow;\r\n }\r\n\r\n public set transparencyShadow(value: boolean) {\r\n this.setTransparencyShadow(value);\r\n }\r\n\r\n /**\r\n * Sets the ability to have transparent shadow (boolean).\r\n * @param transparent True if transparent else False\r\n * @returns the shadow generator allowing fluent coding\r\n */\r\n public setTransparencyShadow(transparent: boolean): ShadowGenerator {\r\n this._transparencyShadow = transparent;\r\n return this;\r\n }\r\n\r\n /**\r\n * Enables or disables shadows with varying strength based on the transparency\r\n * When it is enabled, the strength of the shadow is taken equal to mesh.visibility\r\n * If you enabled an alpha texture on your material, the alpha value red from the texture is also combined to compute the strength:\r\n * mesh.visibility * alphaTexture.a\r\n * The texture used is the diffuse by default, but it can be set to the opacity by setting useOpacityTextureForTransparentShadow\r\n * Note that by definition transparencyShadow must be set to true for enableSoftTransparentShadow to work!\r\n */\r\n public enableSoftTransparentShadow: boolean = false;\r\n\r\n /**\r\n * If this is true, use the opacity texture's alpha channel for transparent shadows instead of the diffuse one\r\n */\r\n public useOpacityTextureForTransparentShadow: boolean = false;\r\n\r\n protected _shadowMap: Nullable;\r\n protected _shadowMap2: Nullable;\r\n\r\n /**\r\n * Gets the main RTT containing the shadow map (usually storing depth from the light point of view).\r\n * @returns The render target texture if present otherwise, null\r\n */\r\n public getShadowMap(): Nullable {\r\n return this._shadowMap;\r\n }\r\n\r\n /**\r\n * Gets the RTT used during rendering (can be a blurred version of the shadow map or the shadow map itself).\r\n * @returns The render target texture if the shadow map is present otherwise, null\r\n */\r\n public getShadowMapForRendering(): Nullable {\r\n if (this._shadowMap2) {\r\n return this._shadowMap2;\r\n }\r\n\r\n return this._shadowMap;\r\n }\r\n\r\n /**\r\n * Gets the class name of that object\r\n * @returns \"ShadowGenerator\"\r\n */\r\n public getClassName(): string {\r\n return ShadowGenerator.CLASSNAME;\r\n }\r\n\r\n /**\r\n * Helper function to add a mesh and its descendants to the list of shadow casters.\r\n * @param mesh Mesh to add\r\n * @param includeDescendants boolean indicating if the descendants should be added. Default to true\r\n * @returns the Shadow Generator itself\r\n */\r\n public addShadowCaster(mesh: AbstractMesh, includeDescendants = true): ShadowGenerator {\r\n if (!this._shadowMap) {\r\n return this;\r\n }\r\n\r\n if (!this._shadowMap.renderList) {\r\n this._shadowMap.renderList = [];\r\n }\r\n\r\n if (this._shadowMap.renderList.indexOf(mesh) === -1) {\r\n this._shadowMap.renderList.push(mesh);\r\n }\r\n\r\n if (includeDescendants) {\r\n for (const childMesh of mesh.getChildMeshes()) {\r\n if (this._shadowMap.renderList.indexOf(childMesh) === -1) {\r\n this._shadowMap.renderList.push(childMesh);\r\n }\r\n }\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Helper function to remove a mesh and its descendants from the list of shadow casters\r\n * @param mesh Mesh to remove\r\n * @param includeDescendants boolean indicating if the descendants should be removed. Default to true\r\n * @returns the Shadow Generator itself\r\n */\r\n public removeShadowCaster(mesh: AbstractMesh, includeDescendants = true): ShadowGenerator {\r\n if (!this._shadowMap || !this._shadowMap.renderList) {\r\n return this;\r\n }\r\n\r\n const index = this._shadowMap.renderList.indexOf(mesh);\r\n\r\n if (index !== -1) {\r\n this._shadowMap.renderList.splice(index, 1);\r\n }\r\n\r\n if (includeDescendants) {\r\n for (const child of mesh.getChildren()) {\r\n this.removeShadowCaster(child);\r\n }\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Controls the extent to which the shadows fade out at the edge of the frustum\r\n */\r\n public frustumEdgeFalloff = 0;\r\n\r\n protected _light: IShadowLight;\r\n /**\r\n * Returns the associated light object.\r\n * @returns the light generating the shadow\r\n */\r\n public getLight(): IShadowLight {\r\n return this._light;\r\n }\r\n\r\n /**\r\n * If true the shadow map is generated by rendering the back face of the mesh instead of the front face.\r\n * This can help with self-shadowing as the geometry making up the back of objects is slightly offset.\r\n * It might on the other hand introduce peter panning.\r\n */\r\n public forceBackFacesOnly = false;\r\n\r\n protected _camera: Nullable;\r\n\r\n protected _getCamera() {\r\n return this._camera ?? this._scene.activeCamera;\r\n }\r\n\r\n protected _scene: Scene;\r\n protected _useRedTextureType: boolean;\r\n protected _lightDirection = Vector3.Zero();\r\n\r\n protected _viewMatrix = Matrix.Zero();\r\n protected _projectionMatrix = Matrix.Zero();\r\n protected _transformMatrix = Matrix.Zero();\r\n protected _cachedPosition: Vector3 = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\r\n protected _cachedDirection: Vector3 = new Vector3(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\r\n protected _cachedDefines: string;\r\n protected _currentRenderId: number;\r\n protected _boxBlurPostprocess: Nullable;\r\n protected _kernelBlurXPostprocess: Nullable;\r\n protected _kernelBlurYPostprocess: Nullable;\r\n protected _blurPostProcesses: PostProcess[];\r\n protected _mapSize: number;\r\n protected _currentFaceIndex = 0;\r\n protected _currentFaceIndexCache = 0;\r\n protected _textureType: number;\r\n protected _defaultTextureMatrix = Matrix.Identity();\r\n protected _storedUniqueId: Nullable;\r\n protected _useUBO: boolean;\r\n protected _sceneUBOs: UniformBuffer[];\r\n protected _currentSceneUBO: UniformBuffer;\r\n protected _opacityTexture: Nullable;\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _SceneComponentInitialization: (scene: Scene) => void = (_) => {\r\n throw _WarnImport(\"ShadowGeneratorSceneComponent\");\r\n };\r\n\r\n /**\r\n * Gets or sets the size of the texture what stores the shadows\r\n */\r\n public get mapSize(): number {\r\n return this._mapSize;\r\n }\r\n\r\n public set mapSize(size: number) {\r\n this._mapSize = size;\r\n this._light._markMeshesAsLightDirty();\r\n this.recreateShadowMap();\r\n }\r\n\r\n /**\r\n * Creates a ShadowGenerator object.\r\n * A ShadowGenerator is the required tool to use the shadows.\r\n * Each light casting shadows needs to use its own ShadowGenerator.\r\n * Documentation : https://doc.babylonjs.com/features/featuresDeepDive/lights/shadows\r\n * @param mapSize The size of the texture what stores the shadows. Example : 1024.\r\n * @param light The light object generating the shadows.\r\n * @param usefullFloatFirst By default the generator will try to use half float textures but if you need precision (for self shadowing for instance), you can use this option to enforce full float texture.\r\n * @param camera Camera associated with this shadow generator (default: null). If null, takes the scene active camera at the time we need to access it\r\n * @param useRedTextureType Forces the generator to use a Red instead of a RGBA type for the shadow map texture format (default: false)\r\n */\r\n constructor(mapSize: number, light: IShadowLight, usefullFloatFirst?: boolean, camera?: Nullable, useRedTextureType?: boolean) {\r\n this._mapSize = mapSize;\r\n this._light = light;\r\n this._scene = light.getScene();\r\n this._camera = camera ?? null;\r\n this._useRedTextureType = !!useRedTextureType;\r\n\r\n let shadowGenerators = light._shadowGenerators;\r\n if (!shadowGenerators) {\r\n shadowGenerators = light._shadowGenerators = new Map();\r\n }\r\n shadowGenerators.set(this._camera, this);\r\n this.id = light.id;\r\n this._useUBO = this._scene.getEngine().supportsUniformBuffers;\r\n\r\n if (this._useUBO) {\r\n this._sceneUBOs = [];\r\n this._sceneUBOs.push(this._scene.createSceneUniformBuffer(`Scene for Shadow Generator (light \"${this._light.name}\")`));\r\n }\r\n\r\n ShadowGenerator._SceneComponentInitialization(this._scene);\r\n\r\n // Texture type fallback from float to int if not supported.\r\n const caps = this._scene.getEngine().getCaps();\r\n\r\n if (!usefullFloatFirst) {\r\n if (caps.textureHalfFloatRender && caps.textureHalfFloatLinearFiltering) {\r\n this._textureType = Constants.TEXTURETYPE_HALF_FLOAT;\r\n } else if (caps.textureFloatRender && caps.textureFloatLinearFiltering) {\r\n this._textureType = Constants.TEXTURETYPE_FLOAT;\r\n } else {\r\n this._textureType = Constants.TEXTURETYPE_UNSIGNED_INT;\r\n }\r\n } else {\r\n if (caps.textureFloatRender && caps.textureFloatLinearFiltering) {\r\n this._textureType = Constants.TEXTURETYPE_FLOAT;\r\n } else if (caps.textureHalfFloatRender && caps.textureHalfFloatLinearFiltering) {\r\n this._textureType = Constants.TEXTURETYPE_HALF_FLOAT;\r\n } else {\r\n this._textureType = Constants.TEXTURETYPE_UNSIGNED_INT;\r\n }\r\n }\r\n\r\n this._initializeGenerator();\r\n this._applyFilterValues();\r\n }\r\n\r\n protected _initializeGenerator(): void {\r\n this._light._markMeshesAsLightDirty();\r\n this._initializeShadowMap();\r\n }\r\n\r\n protected _createTargetRenderTexture(): void {\r\n const engine = this._scene.getEngine();\r\n if (engine._features.supportDepthStencilTexture) {\r\n this._shadowMap = new RenderTargetTexture(\r\n this._light.name + \"_shadowMap\",\r\n this._mapSize,\r\n this._scene,\r\n false,\r\n true,\r\n this._textureType,\r\n this._light.needCube(),\r\n undefined,\r\n false,\r\n false,\r\n undefined,\r\n this._useRedTextureType ? Constants.TEXTUREFORMAT_RED : Constants.TEXTUREFORMAT_RGBA\r\n );\r\n this._shadowMap.createDepthStencilTexture(engine.useReverseDepthBuffer ? Constants.GREATER : Constants.LESS, true);\r\n } else {\r\n this._shadowMap = new RenderTargetTexture(this._light.name + \"_shadowMap\", this._mapSize, this._scene, false, true, this._textureType, this._light.needCube());\r\n }\r\n this._shadowMap.noPrePassRenderer = true;\r\n }\r\n\r\n protected _initializeShadowMap(): void {\r\n this._createTargetRenderTexture();\r\n\r\n if (this._shadowMap === null) {\r\n return;\r\n }\r\n\r\n this._shadowMap.wrapU = Texture.CLAMP_ADDRESSMODE;\r\n this._shadowMap.wrapV = Texture.CLAMP_ADDRESSMODE;\r\n this._shadowMap.anisotropicFilteringLevel = 1;\r\n this._shadowMap.updateSamplingMode(Texture.BILINEAR_SAMPLINGMODE);\r\n this._shadowMap.renderParticles = false;\r\n this._shadowMap.ignoreCameraViewport = true;\r\n if (this._storedUniqueId) {\r\n this._shadowMap.uniqueId = this._storedUniqueId;\r\n }\r\n\r\n // Custom render function.\r\n this._shadowMap.customRenderFunction = this._renderForShadowMap.bind(this);\r\n\r\n // Force the mesh is ready function to true as we are double checking it\r\n // in the custom render function. Also it prevents side effects and useless\r\n // shader variations in DEPTHPREPASS mode.\r\n this._shadowMap.customIsReadyFunction = () => {\r\n return true;\r\n };\r\n\r\n const engine = this._scene.getEngine();\r\n\r\n this._shadowMap.onBeforeBindObservable.add(() => {\r\n this._currentSceneUBO = this._scene.getSceneUniformBuffer();\r\n engine._debugPushGroup?.(`shadow map generation for pass id ${engine.currentRenderPassId}`, 1);\r\n });\r\n\r\n // Record Face Index before render.\r\n this._shadowMap.onBeforeRenderObservable.add((faceIndex: number) => {\r\n if (this._sceneUBOs) {\r\n this._scene.setSceneUniformBuffer(this._sceneUBOs[0]);\r\n }\r\n this._currentFaceIndex = faceIndex;\r\n if (this._filter === ShadowGenerator.FILTER_PCF) {\r\n engine.setColorWrite(false);\r\n }\r\n this.getTransformMatrix(); // generate the view/projection matrix\r\n this._scene.setTransformMatrix(this._viewMatrix, this._projectionMatrix);\r\n if (this._useUBO) {\r\n this._scene.getSceneUniformBuffer().unbindEffect();\r\n this._scene.finalizeSceneUbo();\r\n }\r\n });\r\n\r\n // Blur if required after render.\r\n this._shadowMap.onAfterUnbindObservable.add(() => {\r\n if (this._sceneUBOs) {\r\n this._scene.setSceneUniformBuffer(this._currentSceneUBO);\r\n }\r\n this._scene.updateTransformMatrix(); // restore the view/projection matrices of the active camera\r\n\r\n if (this._filter === ShadowGenerator.FILTER_PCF) {\r\n engine.setColorWrite(true);\r\n }\r\n if (!this.useBlurExponentialShadowMap && !this.useBlurCloseExponentialShadowMap) {\r\n engine._debugPopGroup?.(1);\r\n return;\r\n }\r\n const shadowMap = this.getShadowMapForRendering();\r\n\r\n if (shadowMap) {\r\n this._scene.postProcessManager.directRender(this._blurPostProcesses, shadowMap.renderTarget, true);\r\n engine.unBindFramebuffer(shadowMap.renderTarget!, true);\r\n engine._debugPopGroup?.(1);\r\n }\r\n });\r\n\r\n // Clear according to the chosen filter.\r\n const clearZero = new Color4(0, 0, 0, 0);\r\n const clearOne = new Color4(1.0, 1.0, 1.0, 1.0);\r\n this._shadowMap.onClearObservable.add((engine) => {\r\n if (this._filter === ShadowGenerator.FILTER_PCF) {\r\n engine.clear(clearOne, false, true, false);\r\n } else if (this.useExponentialShadowMap || this.useBlurExponentialShadowMap) {\r\n engine.clear(clearZero, true, true, false);\r\n } else {\r\n engine.clear(clearOne, true, true, false);\r\n }\r\n });\r\n\r\n // Recreate on resize.\r\n this._shadowMap.onResizeObservable.add((rtt) => {\r\n this._storedUniqueId = this._shadowMap!.uniqueId;\r\n this._mapSize = rtt.getRenderSize();\r\n this._light._markMeshesAsLightDirty();\r\n this.recreateShadowMap();\r\n });\r\n\r\n // Ensures rendering groupids do not erase the depth buffer\r\n // or we would lose the shadows information.\r\n for (let i = RenderingManager.MIN_RENDERINGGROUPS; i < RenderingManager.MAX_RENDERINGGROUPS; i++) {\r\n this._shadowMap.setRenderingAutoClearDepthStencil(i, false);\r\n }\r\n }\r\n\r\n protected _initializeBlurRTTAndPostProcesses(): void {\r\n const engine = this._scene.getEngine();\r\n const targetSize = this._mapSize / this.blurScale;\r\n\r\n if (!this.useKernelBlur || this.blurScale !== 1.0) {\r\n this._shadowMap2 = new RenderTargetTexture(this._light.name + \"_shadowMap2\", targetSize, this._scene, false, true, this._textureType, undefined, undefined, false);\r\n this._shadowMap2.wrapU = Texture.CLAMP_ADDRESSMODE;\r\n this._shadowMap2.wrapV = Texture.CLAMP_ADDRESSMODE;\r\n this._shadowMap2.updateSamplingMode(Texture.BILINEAR_SAMPLINGMODE);\r\n }\r\n\r\n if (this.useKernelBlur) {\r\n this._kernelBlurXPostprocess = new BlurPostProcess(\r\n this._light.name + \"KernelBlurX\",\r\n new Vector2(1, 0),\r\n this.blurKernel,\r\n 1.0,\r\n null,\r\n Texture.BILINEAR_SAMPLINGMODE,\r\n engine,\r\n false,\r\n this._textureType\r\n );\r\n this._kernelBlurXPostprocess.width = targetSize;\r\n this._kernelBlurXPostprocess.height = targetSize;\r\n this._kernelBlurXPostprocess.externalTextureSamplerBinding = true;\r\n this._kernelBlurXPostprocess.onApplyObservable.add((effect) => {\r\n effect.setTexture(\"textureSampler\", this._shadowMap);\r\n });\r\n\r\n this._kernelBlurYPostprocess = new BlurPostProcess(\r\n this._light.name + \"KernelBlurY\",\r\n new Vector2(0, 1),\r\n this.blurKernel,\r\n 1.0,\r\n null,\r\n Texture.BILINEAR_SAMPLINGMODE,\r\n engine,\r\n false,\r\n this._textureType\r\n );\r\n\r\n this._kernelBlurXPostprocess.autoClear = false;\r\n this._kernelBlurYPostprocess.autoClear = false;\r\n\r\n if (this._textureType === Constants.TEXTURETYPE_UNSIGNED_INT) {\r\n (this._kernelBlurXPostprocess).packedFloat = true;\r\n (this._kernelBlurYPostprocess).packedFloat = true;\r\n }\r\n\r\n this._blurPostProcesses = [this._kernelBlurXPostprocess, this._kernelBlurYPostprocess];\r\n } else {\r\n this._boxBlurPostprocess = new PostProcess(\r\n this._light.name + \"DepthBoxBlur\",\r\n \"depthBoxBlur\",\r\n [\"screenSize\", \"boxOffset\"],\r\n [],\r\n 1.0,\r\n null,\r\n Texture.BILINEAR_SAMPLINGMODE,\r\n engine,\r\n false,\r\n \"#define OFFSET \" + this._blurBoxOffset,\r\n this._textureType\r\n );\r\n this._boxBlurPostprocess.externalTextureSamplerBinding = true;\r\n this._boxBlurPostprocess.onApplyObservable.add((effect) => {\r\n effect.setFloat2(\"screenSize\", targetSize, targetSize);\r\n effect.setTexture(\"textureSampler\", this._shadowMap);\r\n });\r\n\r\n this._boxBlurPostprocess.autoClear = false;\r\n\r\n this._blurPostProcesses = [this._boxBlurPostprocess];\r\n }\r\n }\r\n\r\n protected _renderForShadowMap(\r\n opaqueSubMeshes: SmartArray,\r\n alphaTestSubMeshes: SmartArray,\r\n transparentSubMeshes: SmartArray,\r\n depthOnlySubMeshes: SmartArray\r\n ): void {\r\n let index: number;\r\n\r\n if (depthOnlySubMeshes.length) {\r\n for (index = 0; index < depthOnlySubMeshes.length; index++) {\r\n this._renderSubMeshForShadowMap(depthOnlySubMeshes.data[index]);\r\n }\r\n }\r\n\r\n for (index = 0; index < opaqueSubMeshes.length; index++) {\r\n this._renderSubMeshForShadowMap(opaqueSubMeshes.data[index]);\r\n }\r\n\r\n for (index = 0; index < alphaTestSubMeshes.length; index++) {\r\n this._renderSubMeshForShadowMap(alphaTestSubMeshes.data[index]);\r\n }\r\n\r\n if (this._transparencyShadow) {\r\n for (index = 0; index < transparentSubMeshes.length; index++) {\r\n this._renderSubMeshForShadowMap(transparentSubMeshes.data[index], true);\r\n }\r\n } else {\r\n for (index = 0; index < transparentSubMeshes.length; index++) {\r\n transparentSubMeshes.data[index].getEffectiveMesh()._internalAbstractMeshDataInfo._isActiveIntermediate = false;\r\n }\r\n }\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n protected _bindCustomEffectForRenderSubMeshForShadowMap(subMesh: SubMesh, effect: Effect, mesh: AbstractMesh): void {\r\n effect.setMatrix(\"viewProjection\", this.getTransformMatrix());\r\n }\r\n\r\n protected _renderSubMeshForShadowMap(subMesh: SubMesh, isTransparent: boolean = false): void {\r\n const renderingMesh = subMesh.getRenderingMesh();\r\n const effectiveMesh = subMesh.getEffectiveMesh();\r\n const scene = this._scene;\r\n const engine = scene.getEngine();\r\n const material = subMesh.getMaterial();\r\n\r\n effectiveMesh._internalAbstractMeshDataInfo._isActiveIntermediate = false;\r\n\r\n if (!material || subMesh.verticesCount === 0 || subMesh._renderId === scene.getRenderId()) {\r\n return;\r\n }\r\n\r\n // Culling\r\n const detNeg = effectiveMesh._getWorldMatrixDeterminant() < 0;\r\n let sideOrientation = renderingMesh.overrideMaterialSideOrientation ?? material.sideOrientation;\r\n if (detNeg) {\r\n sideOrientation =\r\n sideOrientation === Constants.MATERIAL_ClockWiseSideOrientation ? Constants.MATERIAL_CounterClockWiseSideOrientation : Constants.MATERIAL_ClockWiseSideOrientation;\r\n }\r\n const reverseSideOrientation = sideOrientation === Constants.MATERIAL_ClockWiseSideOrientation;\r\n\r\n engine.setState(material.backFaceCulling, undefined, undefined, reverseSideOrientation, material.cullBackFaces);\r\n\r\n // Managing instances\r\n const batch = renderingMesh._getInstancesRenderList(subMesh._id, !!subMesh.getReplacementMesh());\r\n if (batch.mustReturn) {\r\n return;\r\n }\r\n\r\n const hardwareInstancedRendering =\r\n engine.getCaps().instancedArrays &&\r\n ((batch.visibleInstances[subMesh._id] !== null && batch.visibleInstances[subMesh._id] !== undefined) || renderingMesh.hasThinInstances);\r\n\r\n if (this.customAllowRendering && !this.customAllowRendering(subMesh)) {\r\n return;\r\n }\r\n\r\n if (this.isReady(subMesh, hardwareInstancedRendering, isTransparent)) {\r\n subMesh._renderId = scene.getRenderId();\r\n\r\n const shadowDepthWrapper = material.shadowDepthWrapper;\r\n\r\n const drawWrapper = shadowDepthWrapper?.getEffect(subMesh, this, engine.currentRenderPassId) ?? subMesh._getDrawWrapper()!;\r\n const effect = DrawWrapper.GetEffect(drawWrapper)!;\r\n\r\n engine.enableEffect(drawWrapper);\r\n\r\n if (!hardwareInstancedRendering) {\r\n renderingMesh._bind(subMesh, effect, material.fillMode);\r\n }\r\n\r\n this.getTransformMatrix(); // make sure _cachedDirection et _cachedPosition are up to date\r\n\r\n effect.setFloat3(\"biasAndScaleSM\", this.bias, this.normalBias, this.depthScale);\r\n\r\n if (this.getLight().getTypeID() === Light.LIGHTTYPEID_DIRECTIONALLIGHT) {\r\n effect.setVector3(\"lightDataSM\", this._cachedDirection);\r\n } else {\r\n effect.setVector3(\"lightDataSM\", this._cachedPosition);\r\n }\r\n\r\n const camera = this._getCamera();\r\n if (camera) {\r\n effect.setFloat2(\"depthValuesSM\", this.getLight().getDepthMinZ(camera), this.getLight().getDepthMinZ(camera) + this.getLight().getDepthMaxZ(camera));\r\n }\r\n\r\n if (isTransparent && this.enableSoftTransparentShadow) {\r\n effect.setFloat(\"softTransparentShadowSM\", effectiveMesh.visibility * material.alpha);\r\n }\r\n\r\n if (shadowDepthWrapper) {\r\n subMesh._setMainDrawWrapperOverride(drawWrapper);\r\n if (shadowDepthWrapper.standalone) {\r\n shadowDepthWrapper.baseMaterial.bindForSubMesh(effectiveMesh.getWorldMatrix(), renderingMesh, subMesh);\r\n } else {\r\n material.bindForSubMesh(effectiveMesh.getWorldMatrix(), renderingMesh, subMesh);\r\n }\r\n subMesh._setMainDrawWrapperOverride(null);\r\n } else {\r\n // Alpha test\r\n if (this._opacityTexture) {\r\n effect.setTexture(\"diffuseSampler\", this._opacityTexture);\r\n effect.setMatrix(\"diffuseMatrix\", this._opacityTexture.getTextureMatrix() || this._defaultTextureMatrix);\r\n }\r\n\r\n // Bones\r\n if (renderingMesh.useBones && renderingMesh.computeBonesUsingShaders && renderingMesh.skeleton) {\r\n const skeleton = renderingMesh.skeleton;\r\n\r\n if (skeleton.isUsingTextureForMatrices) {\r\n const boneTexture = skeleton.getTransformMatrixTexture(renderingMesh);\r\n\r\n if (!boneTexture) {\r\n return;\r\n }\r\n\r\n effect.setTexture(\"boneSampler\", boneTexture);\r\n effect.setFloat(\"boneTextureWidth\", 4.0 * (skeleton.bones.length + 1));\r\n } else {\r\n effect.setMatrices(\"mBones\", skeleton.getTransformMatrices(renderingMesh));\r\n }\r\n }\r\n\r\n // Morph targets\r\n MaterialHelper.BindMorphTargetParameters(renderingMesh, effect);\r\n\r\n if (renderingMesh.morphTargetManager && renderingMesh.morphTargetManager.isUsingTextureForTargets) {\r\n renderingMesh.morphTargetManager._bind(effect);\r\n }\r\n\r\n // Clip planes\r\n bindClipPlane(effect, material, scene);\r\n }\r\n\r\n if (!this._useUBO && !shadowDepthWrapper) {\r\n this._bindCustomEffectForRenderSubMeshForShadowMap(subMesh, effect, effectiveMesh);\r\n }\r\n\r\n MaterialHelper.BindSceneUniformBuffer(effect, this._scene.getSceneUniformBuffer());\r\n this._scene.getSceneUniformBuffer().bindUniformBuffer();\r\n\r\n const world = effectiveMesh.getWorldMatrix();\r\n\r\n // In the non hardware instanced mode, the Mesh ubo update is done by the callback passed to renderingMesh._processRendering (see below)\r\n if (hardwareInstancedRendering) {\r\n effectiveMesh.getMeshUniformBuffer().bindToEffect(effect, \"Mesh\");\r\n effectiveMesh.transferToEffect(world);\r\n }\r\n\r\n if (this.forceBackFacesOnly) {\r\n engine.setState(true, 0, false, true, material.cullBackFaces);\r\n }\r\n\r\n // Observables\r\n this.onBeforeShadowMapRenderMeshObservable.notifyObservers(renderingMesh);\r\n this.onBeforeShadowMapRenderObservable.notifyObservers(effect);\r\n\r\n // Draw\r\n renderingMesh._processRendering(effectiveMesh, subMesh, effect, material.fillMode, batch, hardwareInstancedRendering, (isInstance, worldOverride) => {\r\n if (effectiveMesh !== renderingMesh && !isInstance) {\r\n renderingMesh.getMeshUniformBuffer().bindToEffect(effect, \"Mesh\");\r\n renderingMesh.transferToEffect(worldOverride);\r\n } else {\r\n effectiveMesh.getMeshUniformBuffer().bindToEffect(effect, \"Mesh\");\r\n effectiveMesh.transferToEffect(isInstance ? worldOverride : world);\r\n }\r\n });\r\n\r\n if (this.forceBackFacesOnly) {\r\n engine.setState(true, 0, false, false, material.cullBackFaces);\r\n }\r\n\r\n // Observables\r\n this.onAfterShadowMapRenderObservable.notifyObservers(effect);\r\n this.onAfterShadowMapRenderMeshObservable.notifyObservers(renderingMesh);\r\n } else {\r\n // Need to reset refresh rate of the shadowMap\r\n if (this._shadowMap) {\r\n this._shadowMap.resetRefreshCounter();\r\n }\r\n }\r\n }\r\n\r\n protected _applyFilterValues(): void {\r\n if (!this._shadowMap) {\r\n return;\r\n }\r\n\r\n if (this.filter === ShadowGenerator.FILTER_NONE || this.filter === ShadowGenerator.FILTER_PCSS) {\r\n this._shadowMap.updateSamplingMode(Texture.NEAREST_SAMPLINGMODE);\r\n } else {\r\n this._shadowMap.updateSamplingMode(Texture.BILINEAR_SAMPLINGMODE);\r\n }\r\n }\r\n\r\n /**\r\n * Forces all the attached effect to compile to enable rendering only once ready vs. lazily compiling effects.\r\n * @param onCompiled Callback triggered at the and of the effects compilation\r\n * @param options Sets of optional options forcing the compilation with different modes\r\n */\r\n public forceCompilation(onCompiled?: (generator: IShadowGenerator) => void, options?: Partial<{ useInstances: boolean }>): void {\r\n const localOptions = {\r\n useInstances: false,\r\n ...options,\r\n };\r\n\r\n const shadowMap = this.getShadowMap();\r\n if (!shadowMap) {\r\n if (onCompiled) {\r\n onCompiled(this);\r\n }\r\n return;\r\n }\r\n\r\n const renderList = shadowMap.renderList;\r\n if (!renderList) {\r\n if (onCompiled) {\r\n onCompiled(this);\r\n }\r\n return;\r\n }\r\n\r\n const subMeshes = new Array();\r\n for (const mesh of renderList) {\r\n subMeshes.push(...mesh.subMeshes);\r\n }\r\n if (subMeshes.length === 0) {\r\n if (onCompiled) {\r\n onCompiled(this);\r\n }\r\n return;\r\n }\r\n\r\n let currentIndex = 0;\r\n\r\n const checkReady = () => {\r\n if (!this._scene || !this._scene.getEngine()) {\r\n return;\r\n }\r\n\r\n while (\r\n this.isReady(\r\n subMeshes[currentIndex],\r\n localOptions.useInstances,\r\n subMeshes[currentIndex].getMaterial()?.needAlphaBlendingForMesh(subMeshes[currentIndex].getMesh()) ?? false\r\n )\r\n ) {\r\n currentIndex++;\r\n if (currentIndex >= subMeshes.length) {\r\n if (onCompiled) {\r\n onCompiled(this);\r\n }\r\n return;\r\n }\r\n }\r\n setTimeout(checkReady, 16);\r\n };\r\n\r\n checkReady();\r\n }\r\n\r\n /**\r\n * Forces all the attached effect to compile to enable rendering only once ready vs. lazily compiling effects.\r\n * @param options Sets of optional options forcing the compilation with different modes\r\n * @returns A promise that resolves when the compilation completes\r\n */\r\n public forceCompilationAsync(options?: Partial<{ useInstances: boolean }>): Promise {\r\n return new Promise((resolve) => {\r\n this.forceCompilation(() => {\r\n resolve();\r\n }, options);\r\n });\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n protected _isReadyCustomDefines(defines: any, subMesh: SubMesh, useInstances: boolean): void {}\r\n\r\n private _prepareShadowDefines(subMesh: SubMesh, useInstances: boolean, defines: string[], isTransparent: boolean): string[] {\r\n defines.push(\"#define SM_LIGHTTYPE_\" + this._light.getClassName().toUpperCase());\r\n\r\n defines.push(\"#define SM_FLOAT \" + (this._textureType !== Constants.TEXTURETYPE_UNSIGNED_INT ? \"1\" : \"0\"));\r\n\r\n defines.push(\"#define SM_ESM \" + (this.useExponentialShadowMap || this.useBlurExponentialShadowMap ? \"1\" : \"0\"));\r\n\r\n defines.push(\"#define SM_DEPTHTEXTURE \" + (this.usePercentageCloserFiltering || this.useContactHardeningShadow ? \"1\" : \"0\"));\r\n\r\n const mesh = subMesh.getMesh();\r\n\r\n // Normal bias.\r\n defines.push(\"#define SM_NORMALBIAS \" + (this.normalBias && mesh.isVerticesDataPresent(VertexBuffer.NormalKind) ? \"1\" : \"0\"));\r\n defines.push(\"#define SM_DIRECTIONINLIGHTDATA \" + (this.getLight().getTypeID() === Light.LIGHTTYPEID_DIRECTIONALLIGHT ? \"1\" : \"0\"));\r\n\r\n // Point light\r\n defines.push(\"#define SM_USEDISTANCE \" + (this._light.needCube() ? \"1\" : \"0\"));\r\n\r\n // Soft transparent shadows\r\n defines.push(\"#define SM_SOFTTRANSPARENTSHADOW \" + (this.enableSoftTransparentShadow && isTransparent ? \"1\" : \"0\"));\r\n\r\n this._isReadyCustomDefines(defines, subMesh, useInstances);\r\n\r\n return defines;\r\n }\r\n\r\n /**\r\n * Determine whether the shadow generator is ready or not (mainly all effects and related post processes needs to be ready).\r\n * @param subMesh The submesh we want to render in the shadow map\r\n * @param useInstances Defines whether will draw in the map using instances\r\n * @param isTransparent Indicates that isReady is called for a transparent subMesh\r\n * @returns true if ready otherwise, false\r\n */\r\n public isReady(subMesh: SubMesh, useInstances: boolean, isTransparent: boolean): boolean {\r\n const material = subMesh.getMaterial(),\r\n shadowDepthWrapper = material?.shadowDepthWrapper;\r\n\r\n this._opacityTexture = null;\r\n\r\n if (!material) {\r\n return false;\r\n }\r\n\r\n const defines: string[] = [];\r\n\r\n this._prepareShadowDefines(subMesh, useInstances, defines, isTransparent);\r\n\r\n if (shadowDepthWrapper) {\r\n if (!shadowDepthWrapper.isReadyForSubMesh(subMesh, defines, this, useInstances, this._scene.getEngine().currentRenderPassId)) {\r\n return false;\r\n }\r\n } else {\r\n const subMeshEffect = subMesh._getDrawWrapper(undefined, true)!;\r\n\r\n let effect = subMeshEffect.effect!;\r\n let cachedDefines = subMeshEffect.defines;\r\n\r\n const attribs = [VertexBuffer.PositionKind];\r\n\r\n const mesh = subMesh.getMesh();\r\n\r\n // Normal bias.\r\n if (this.normalBias && mesh.isVerticesDataPresent(VertexBuffer.NormalKind)) {\r\n attribs.push(VertexBuffer.NormalKind);\r\n defines.push(\"#define NORMAL\");\r\n if (mesh.nonUniformScaling) {\r\n defines.push(\"#define NONUNIFORMSCALING\");\r\n }\r\n }\r\n\r\n // Alpha test\r\n const needAlphaTesting = material.needAlphaTesting();\r\n\r\n if (needAlphaTesting || material.needAlphaBlending()) {\r\n if (this.useOpacityTextureForTransparentShadow) {\r\n this._opacityTexture = (material as any).opacityTexture;\r\n } else {\r\n this._opacityTexture = material.getAlphaTestTexture();\r\n }\r\n if (this._opacityTexture) {\r\n if (!this._opacityTexture.isReady()) {\r\n return false;\r\n }\r\n\r\n const alphaCutOff = (material as any).alphaCutOff ?? ShadowGenerator.DEFAULT_ALPHA_CUTOFF;\r\n\r\n defines.push(\"#define ALPHATEXTURE\");\r\n if (needAlphaTesting) {\r\n defines.push(`#define ALPHATESTVALUE ${alphaCutOff}${alphaCutOff % 1 === 0 ? \".\" : \"\"}`);\r\n }\r\n if (mesh.isVerticesDataPresent(VertexBuffer.UVKind)) {\r\n attribs.push(VertexBuffer.UVKind);\r\n defines.push(\"#define UV1\");\r\n }\r\n if (mesh.isVerticesDataPresent(VertexBuffer.UV2Kind)) {\r\n if (this._opacityTexture.coordinatesIndex === 1) {\r\n attribs.push(VertexBuffer.UV2Kind);\r\n defines.push(\"#define UV2\");\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Bones\r\n const fallbacks = new EffectFallbacks();\r\n if (mesh.useBones && mesh.computeBonesUsingShaders && mesh.skeleton) {\r\n attribs.push(VertexBuffer.MatricesIndicesKind);\r\n attribs.push(VertexBuffer.MatricesWeightsKind);\r\n if (mesh.numBoneInfluencers > 4) {\r\n attribs.push(VertexBuffer.MatricesIndicesExtraKind);\r\n attribs.push(VertexBuffer.MatricesWeightsExtraKind);\r\n }\r\n const skeleton = mesh.skeleton;\r\n defines.push(\"#define NUM_BONE_INFLUENCERS \" + mesh.numBoneInfluencers);\r\n if (mesh.numBoneInfluencers > 0) {\r\n fallbacks.addCPUSkinningFallback(0, mesh);\r\n }\r\n\r\n if (skeleton.isUsingTextureForMatrices) {\r\n defines.push(\"#define BONETEXTURE\");\r\n } else {\r\n defines.push(\"#define BonesPerMesh \" + (skeleton.bones.length + 1));\r\n }\r\n } else {\r\n defines.push(\"#define NUM_BONE_INFLUENCERS 0\");\r\n }\r\n\r\n // Morph targets\r\n const manager = (mesh).morphTargetManager;\r\n let morphInfluencers = 0;\r\n if (manager) {\r\n if (manager.numInfluencers > 0) {\r\n defines.push(\"#define MORPHTARGETS\");\r\n morphInfluencers = manager.numInfluencers;\r\n defines.push(\"#define NUM_MORPH_INFLUENCERS \" + morphInfluencers);\r\n if (manager.isUsingTextureForTargets) {\r\n defines.push(\"#define MORPHTARGETS_TEXTURE\");\r\n }\r\n MaterialHelper.PrepareAttributesForMorphTargetsInfluencers(attribs, mesh, morphInfluencers);\r\n }\r\n }\r\n\r\n // ClipPlanes\r\n prepareStringDefinesForClipPlanes(material, this._scene, defines);\r\n\r\n // Instances\r\n if (useInstances) {\r\n defines.push(\"#define INSTANCES\");\r\n MaterialHelper.PushAttributesForInstances(attribs);\r\n if (subMesh.getRenderingMesh().hasThinInstances) {\r\n defines.push(\"#define THIN_INSTANCES\");\r\n }\r\n }\r\n\r\n if (this.customShaderOptions) {\r\n if (this.customShaderOptions.defines) {\r\n for (const define of this.customShaderOptions.defines) {\r\n if (defines.indexOf(define) === -1) {\r\n defines.push(define);\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Get correct effect\r\n const join = defines.join(\"\\n\");\r\n if (cachedDefines !== join) {\r\n cachedDefines = join;\r\n\r\n let shaderName = \"shadowMap\";\r\n const uniforms = [\r\n \"world\",\r\n \"mBones\",\r\n \"viewProjection\",\r\n \"diffuseMatrix\",\r\n \"lightDataSM\",\r\n \"depthValuesSM\",\r\n \"biasAndScaleSM\",\r\n \"morphTargetInfluences\",\r\n \"boneTextureWidth\",\r\n \"softTransparentShadowSM\",\r\n \"morphTargetTextureInfo\",\r\n \"morphTargetTextureIndices\",\r\n ];\r\n const samplers = [\"diffuseSampler\", \"boneSampler\", \"morphTargets\"];\r\n const uniformBuffers = [\"Scene\", \"Mesh\"];\r\n\r\n addClipPlaneUniforms(uniforms);\r\n\r\n // Custom shader?\r\n if (this.customShaderOptions) {\r\n shaderName = this.customShaderOptions.shaderName;\r\n\r\n if (this.customShaderOptions.attributes) {\r\n for (const attrib of this.customShaderOptions.attributes) {\r\n if (attribs.indexOf(attrib) === -1) {\r\n attribs.push(attrib);\r\n }\r\n }\r\n }\r\n\r\n if (this.customShaderOptions.uniforms) {\r\n for (const uniform of this.customShaderOptions.uniforms) {\r\n if (uniforms.indexOf(uniform) === -1) {\r\n uniforms.push(uniform);\r\n }\r\n }\r\n }\r\n\r\n if (this.customShaderOptions.samplers) {\r\n for (const sampler of this.customShaderOptions.samplers) {\r\n if (samplers.indexOf(sampler) === -1) {\r\n samplers.push(sampler);\r\n }\r\n }\r\n }\r\n }\r\n\r\n const engine = this._scene.getEngine();\r\n\r\n effect = engine.createEffect(\r\n shaderName,\r\n {\r\n attributes: attribs,\r\n uniformsNames: uniforms,\r\n uniformBuffersNames: uniformBuffers,\r\n samplers: samplers,\r\n defines: join,\r\n fallbacks: fallbacks,\r\n onCompiled: null,\r\n onError: null,\r\n indexParameters: { maxSimultaneousMorphTargets: morphInfluencers },\r\n },\r\n engine\r\n );\r\n\r\n subMeshEffect.setEffect(effect, cachedDefines);\r\n }\r\n\r\n if (!effect.isReady()) {\r\n return false;\r\n }\r\n }\r\n\r\n if (this.useBlurExponentialShadowMap || this.useBlurCloseExponentialShadowMap) {\r\n if (!this._blurPostProcesses || !this._blurPostProcesses.length) {\r\n this._initializeBlurRTTAndPostProcesses();\r\n }\r\n }\r\n\r\n if (this._kernelBlurXPostprocess && !this._kernelBlurXPostprocess.isReady()) {\r\n return false;\r\n }\r\n if (this._kernelBlurYPostprocess && !this._kernelBlurYPostprocess.isReady()) {\r\n return false;\r\n }\r\n if (this._boxBlurPostprocess && !this._boxBlurPostprocess.isReady()) {\r\n return false;\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Prepare all the defines in a material relying on a shadow map at the specified light index.\r\n * @param defines Defines of the material we want to update\r\n * @param lightIndex Index of the light in the enabled light list of the material\r\n */\r\n public prepareDefines(defines: any, lightIndex: number): void {\r\n const scene = this._scene;\r\n const light = this._light;\r\n\r\n if (!scene.shadowsEnabled || !light.shadowEnabled) {\r\n return;\r\n }\r\n\r\n defines[\"SHADOW\" + lightIndex] = true;\r\n\r\n if (this.useContactHardeningShadow) {\r\n defines[\"SHADOWPCSS\" + lightIndex] = true;\r\n if (this._filteringQuality === ShadowGenerator.QUALITY_LOW) {\r\n defines[\"SHADOWLOWQUALITY\" + lightIndex] = true;\r\n } else if (this._filteringQuality === ShadowGenerator.QUALITY_MEDIUM) {\r\n defines[\"SHADOWMEDIUMQUALITY\" + lightIndex] = true;\r\n }\r\n // else default to high.\r\n } else if (this.usePercentageCloserFiltering) {\r\n defines[\"SHADOWPCF\" + lightIndex] = true;\r\n if (this._filteringQuality === ShadowGenerator.QUALITY_LOW) {\r\n defines[\"SHADOWLOWQUALITY\" + lightIndex] = true;\r\n } else if (this._filteringQuality === ShadowGenerator.QUALITY_MEDIUM) {\r\n defines[\"SHADOWMEDIUMQUALITY\" + lightIndex] = true;\r\n }\r\n // else default to high.\r\n } else if (this.usePoissonSampling) {\r\n defines[\"SHADOWPOISSON\" + lightIndex] = true;\r\n } else if (this.useExponentialShadowMap || this.useBlurExponentialShadowMap) {\r\n defines[\"SHADOWESM\" + lightIndex] = true;\r\n } else if (this.useCloseExponentialShadowMap || this.useBlurCloseExponentialShadowMap) {\r\n defines[\"SHADOWCLOSEESM\" + lightIndex] = true;\r\n }\r\n\r\n if (light.needCube()) {\r\n defines[\"SHADOWCUBE\" + lightIndex] = true;\r\n }\r\n }\r\n\r\n /**\r\n * Binds the shadow related information inside of an effect (information like near, far, darkness...\r\n * defined in the generator but impacting the effect).\r\n * @param lightIndex Index of the light in the enabled light list of the material owning the effect\r\n * @param effect The effect we are binding the information for\r\n */\r\n public bindShadowLight(lightIndex: string, effect: Effect): void {\r\n const light = this._light;\r\n const scene = this._scene;\r\n\r\n if (!scene.shadowsEnabled || !light.shadowEnabled) {\r\n return;\r\n }\r\n\r\n const camera = this._getCamera();\r\n if (!camera) {\r\n return;\r\n }\r\n\r\n const shadowMap = this.getShadowMap();\r\n\r\n if (!shadowMap) {\r\n return;\r\n }\r\n\r\n if (!light.needCube()) {\r\n effect.setMatrix(\"lightMatrix\" + lightIndex, this.getTransformMatrix());\r\n }\r\n\r\n // Only PCF uses depth stencil texture.\r\n if (this._filter === ShadowGenerator.FILTER_PCF) {\r\n effect.setDepthStencilTexture(\"shadowSampler\" + lightIndex, this.getShadowMapForRendering());\r\n light._uniformBuffer.updateFloat4(\"shadowsInfo\", this.getDarkness(), shadowMap.getSize().width, 1 / shadowMap.getSize().width, this.frustumEdgeFalloff, lightIndex);\r\n } else if (this._filter === ShadowGenerator.FILTER_PCSS) {\r\n effect.setDepthStencilTexture(\"shadowSampler\" + lightIndex, this.getShadowMapForRendering());\r\n effect.setTexture(\"depthSampler\" + lightIndex, this.getShadowMapForRendering());\r\n light._uniformBuffer.updateFloat4(\r\n \"shadowsInfo\",\r\n this.getDarkness(),\r\n 1 / shadowMap.getSize().width,\r\n this._contactHardeningLightSizeUVRatio * shadowMap.getSize().width,\r\n this.frustumEdgeFalloff,\r\n lightIndex\r\n );\r\n } else {\r\n effect.setTexture(\"shadowSampler\" + lightIndex, this.getShadowMapForRendering());\r\n light._uniformBuffer.updateFloat4(\"shadowsInfo\", this.getDarkness(), this.blurScale / shadowMap.getSize().width, this.depthScale, this.frustumEdgeFalloff, lightIndex);\r\n }\r\n\r\n light._uniformBuffer.updateFloat2(\r\n \"depthValues\",\r\n this.getLight().getDepthMinZ(camera),\r\n this.getLight().getDepthMinZ(camera) + this.getLight().getDepthMaxZ(camera),\r\n lightIndex\r\n );\r\n }\r\n\r\n /**\r\n * Gets the transformation matrix used to project the meshes into the map from the light point of view.\r\n * (eq to shadow projection matrix * light transform matrix)\r\n * @returns The transform matrix used to create the shadow map\r\n */\r\n public getTransformMatrix(): Matrix {\r\n const scene = this._scene;\r\n if (this._currentRenderId === scene.getRenderId() && this._currentFaceIndexCache === this._currentFaceIndex) {\r\n return this._transformMatrix;\r\n }\r\n\r\n this._currentRenderId = scene.getRenderId();\r\n this._currentFaceIndexCache = this._currentFaceIndex;\r\n\r\n let lightPosition = this._light.position;\r\n if (this._light.computeTransformedInformation()) {\r\n lightPosition = this._light.transformedPosition;\r\n }\r\n\r\n Vector3.NormalizeToRef(this._light.getShadowDirection(this._currentFaceIndex), this._lightDirection);\r\n if (Math.abs(Vector3.Dot(this._lightDirection, Vector3.Up())) === 1.0) {\r\n this._lightDirection.z = 0.0000000000001; // Required to avoid perfectly perpendicular light\r\n }\r\n\r\n if (\r\n this._light.needProjectionMatrixCompute() ||\r\n !this._cachedPosition ||\r\n !this._cachedDirection ||\r\n !lightPosition.equals(this._cachedPosition) ||\r\n !this._lightDirection.equals(this._cachedDirection)\r\n ) {\r\n this._cachedPosition.copyFrom(lightPosition);\r\n this._cachedDirection.copyFrom(this._lightDirection);\r\n\r\n Matrix.LookAtLHToRef(lightPosition, lightPosition.add(this._lightDirection), Vector3.Up(), this._viewMatrix);\r\n\r\n const shadowMap = this.getShadowMap();\r\n\r\n if (shadowMap) {\r\n const renderList = shadowMap.renderList;\r\n\r\n if (renderList) {\r\n this._light.setShadowProjectionMatrix(this._projectionMatrix, this._viewMatrix, renderList);\r\n }\r\n }\r\n\r\n this._viewMatrix.multiplyToRef(this._projectionMatrix, this._transformMatrix);\r\n }\r\n\r\n return this._transformMatrix;\r\n }\r\n\r\n /**\r\n * Recreates the shadow map dependencies like RTT and post processes. This can be used during the switch between\r\n * Cube and 2D textures for instance.\r\n */\r\n public recreateShadowMap(): void {\r\n const shadowMap = this._shadowMap;\r\n if (!shadowMap) {\r\n return;\r\n }\r\n\r\n // Track render list.\r\n const renderList = shadowMap.renderList;\r\n // Clean up existing data.\r\n this._disposeRTTandPostProcesses();\r\n // Reinitializes.\r\n this._initializeGenerator();\r\n // Reaffect the filter to ensure a correct fallback if necessary.\r\n this.filter = this._filter;\r\n // Reaffect the filter.\r\n this._applyFilterValues();\r\n // Reaffect Render List.\r\n if (renderList) {\r\n // Note: don't do this._shadowMap!.renderList = renderList;\r\n // The renderList hooked array is accessing the old RenderTargetTexture (see RenderTargetTexture._hookArray), which is disposed at this point (by the call to _disposeRTTandPostProcesses)\r\n if (!this._shadowMap!.renderList) {\r\n this._shadowMap!.renderList = [];\r\n }\r\n for (const mesh of renderList) {\r\n this._shadowMap!.renderList.push(mesh);\r\n }\r\n } else {\r\n this._shadowMap!.renderList = null;\r\n }\r\n }\r\n\r\n protected _disposeBlurPostProcesses(): void {\r\n if (this._shadowMap2) {\r\n this._shadowMap2.dispose();\r\n this._shadowMap2 = null;\r\n }\r\n\r\n if (this._boxBlurPostprocess) {\r\n this._boxBlurPostprocess.dispose();\r\n this._boxBlurPostprocess = null;\r\n }\r\n\r\n if (this._kernelBlurXPostprocess) {\r\n this._kernelBlurXPostprocess.dispose();\r\n this._kernelBlurXPostprocess = null;\r\n }\r\n\r\n if (this._kernelBlurYPostprocess) {\r\n this._kernelBlurYPostprocess.dispose();\r\n this._kernelBlurYPostprocess = null;\r\n }\r\n\r\n this._blurPostProcesses = [];\r\n }\r\n\r\n protected _disposeRTTandPostProcesses(): void {\r\n if (this._shadowMap) {\r\n this._shadowMap.dispose();\r\n this._shadowMap = null;\r\n }\r\n\r\n this._disposeBlurPostProcesses();\r\n }\r\n\r\n protected _disposeSceneUBOs(): void {\r\n if (this._sceneUBOs) {\r\n for (const ubo of this._sceneUBOs) {\r\n ubo.dispose();\r\n }\r\n this._sceneUBOs = [];\r\n }\r\n }\r\n\r\n /**\r\n * Disposes the ShadowGenerator.\r\n * Returns nothing.\r\n */\r\n public dispose(): void {\r\n this._disposeRTTandPostProcesses();\r\n\r\n this._disposeSceneUBOs();\r\n\r\n if (this._light) {\r\n if (this._light._shadowGenerators) {\r\n const iterator = this._light._shadowGenerators.entries();\r\n for (let entry = iterator.next(); entry.done !== true; entry = iterator.next()) {\r\n const [camera, shadowGenerator] = entry.value;\r\n if (shadowGenerator === this) {\r\n this._light._shadowGenerators.delete(camera);\r\n }\r\n }\r\n if (this._light._shadowGenerators.size === 0) {\r\n this._light._shadowGenerators = null;\r\n }\r\n }\r\n this._light._markMeshesAsLightDirty();\r\n }\r\n\r\n this.onBeforeShadowMapRenderMeshObservable.clear();\r\n this.onBeforeShadowMapRenderObservable.clear();\r\n this.onAfterShadowMapRenderMeshObservable.clear();\r\n this.onAfterShadowMapRenderObservable.clear();\r\n }\r\n\r\n /**\r\n * Serializes the shadow generator setup to a json object.\r\n * @returns The serialized JSON object\r\n */\r\n public serialize(): any {\r\n const serializationObject: any = {};\r\n const shadowMap = this.getShadowMap();\r\n\r\n if (!shadowMap) {\r\n return serializationObject;\r\n }\r\n\r\n serializationObject.className = this.getClassName();\r\n serializationObject.lightId = this._light.id;\r\n serializationObject.cameraId = this._camera?.id;\r\n serializationObject.id = this.id;\r\n serializationObject.mapSize = shadowMap.getRenderSize();\r\n serializationObject.forceBackFacesOnly = this.forceBackFacesOnly;\r\n serializationObject.darkness = this.getDarkness();\r\n serializationObject.transparencyShadow = this._transparencyShadow;\r\n serializationObject.frustumEdgeFalloff = this.frustumEdgeFalloff;\r\n serializationObject.bias = this.bias;\r\n serializationObject.normalBias = this.normalBias;\r\n serializationObject.usePercentageCloserFiltering = this.usePercentageCloserFiltering;\r\n serializationObject.useContactHardeningShadow = this.useContactHardeningShadow;\r\n serializationObject.contactHardeningLightSizeUVRatio = this.contactHardeningLightSizeUVRatio;\r\n serializationObject.filteringQuality = this.filteringQuality;\r\n serializationObject.useExponentialShadowMap = this.useExponentialShadowMap;\r\n serializationObject.useBlurExponentialShadowMap = this.useBlurExponentialShadowMap;\r\n serializationObject.useCloseExponentialShadowMap = this.useBlurExponentialShadowMap;\r\n serializationObject.useBlurCloseExponentialShadowMap = this.useBlurExponentialShadowMap;\r\n serializationObject.usePoissonSampling = this.usePoissonSampling;\r\n serializationObject.depthScale = this.depthScale;\r\n serializationObject.blurBoxOffset = this.blurBoxOffset;\r\n serializationObject.blurKernel = this.blurKernel;\r\n serializationObject.blurScale = this.blurScale;\r\n serializationObject.useKernelBlur = this.useKernelBlur;\r\n\r\n serializationObject.renderList = [];\r\n if (shadowMap.renderList) {\r\n for (let meshIndex = 0; meshIndex < shadowMap.renderList.length; meshIndex++) {\r\n const mesh = shadowMap.renderList[meshIndex];\r\n\r\n serializationObject.renderList.push(mesh.id);\r\n }\r\n }\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Parses a serialized ShadowGenerator and returns a new ShadowGenerator.\r\n * @param parsedShadowGenerator The JSON object to parse\r\n * @param scene The scene to create the shadow map for\r\n * @param constr A function that builds a shadow generator or undefined to create an instance of the default shadow generator\r\n * @returns The parsed shadow generator\r\n */\r\n public static Parse(parsedShadowGenerator: any, scene: Scene, constr?: (mapSize: number, light: IShadowLight, camera: Nullable) => ShadowGenerator): ShadowGenerator {\r\n const light = scene.getLightById(parsedShadowGenerator.lightId);\r\n const camera: Nullable = parsedShadowGenerator.cameraId !== undefined ? scene.getCameraById(parsedShadowGenerator.cameraId) : null;\r\n const shadowGenerator = constr ? constr(parsedShadowGenerator.mapSize, light, camera) : new ShadowGenerator(parsedShadowGenerator.mapSize, light, undefined, camera);\r\n const shadowMap = shadowGenerator.getShadowMap();\r\n\r\n for (let meshIndex = 0; meshIndex < parsedShadowGenerator.renderList.length; meshIndex++) {\r\n const meshes = scene.getMeshesById(parsedShadowGenerator.renderList[meshIndex]);\r\n meshes.forEach(function (mesh) {\r\n if (!shadowMap) {\r\n return;\r\n }\r\n if (!shadowMap.renderList) {\r\n shadowMap.renderList = [];\r\n }\r\n shadowMap.renderList.push(mesh);\r\n });\r\n }\r\n\r\n if (parsedShadowGenerator.id !== undefined) {\r\n shadowGenerator.id = parsedShadowGenerator.id;\r\n }\r\n\r\n shadowGenerator.forceBackFacesOnly = !!parsedShadowGenerator.forceBackFacesOnly;\r\n\r\n if (parsedShadowGenerator.darkness !== undefined) {\r\n shadowGenerator.setDarkness(parsedShadowGenerator.darkness);\r\n }\r\n\r\n if (parsedShadowGenerator.transparencyShadow) {\r\n shadowGenerator.setTransparencyShadow(true);\r\n }\r\n\r\n if (parsedShadowGenerator.frustumEdgeFalloff !== undefined) {\r\n shadowGenerator.frustumEdgeFalloff = parsedShadowGenerator.frustumEdgeFalloff;\r\n }\r\n\r\n if (parsedShadowGenerator.bias !== undefined) {\r\n shadowGenerator.bias = parsedShadowGenerator.bias;\r\n }\r\n\r\n if (parsedShadowGenerator.normalBias !== undefined) {\r\n shadowGenerator.normalBias = parsedShadowGenerator.normalBias;\r\n }\r\n\r\n if (parsedShadowGenerator.usePercentageCloserFiltering) {\r\n shadowGenerator.usePercentageCloserFiltering = true;\r\n } else if (parsedShadowGenerator.useContactHardeningShadow) {\r\n shadowGenerator.useContactHardeningShadow = true;\r\n } else if (parsedShadowGenerator.usePoissonSampling) {\r\n shadowGenerator.usePoissonSampling = true;\r\n } else if (parsedShadowGenerator.useExponentialShadowMap) {\r\n shadowGenerator.useExponentialShadowMap = true;\r\n } else if (parsedShadowGenerator.useBlurExponentialShadowMap) {\r\n shadowGenerator.useBlurExponentialShadowMap = true;\r\n } else if (parsedShadowGenerator.useCloseExponentialShadowMap) {\r\n shadowGenerator.useCloseExponentialShadowMap = true;\r\n } else if (parsedShadowGenerator.useBlurCloseExponentialShadowMap) {\r\n shadowGenerator.useBlurCloseExponentialShadowMap = true;\r\n }\r\n // Backward compat\r\n else if (parsedShadowGenerator.useVarianceShadowMap) {\r\n shadowGenerator.useExponentialShadowMap = true;\r\n } else if (parsedShadowGenerator.useBlurVarianceShadowMap) {\r\n shadowGenerator.useBlurExponentialShadowMap = true;\r\n }\r\n\r\n if (parsedShadowGenerator.contactHardeningLightSizeUVRatio !== undefined) {\r\n shadowGenerator.contactHardeningLightSizeUVRatio = parsedShadowGenerator.contactHardeningLightSizeUVRatio;\r\n }\r\n\r\n if (parsedShadowGenerator.filteringQuality !== undefined) {\r\n shadowGenerator.filteringQuality = parsedShadowGenerator.filteringQuality;\r\n }\r\n\r\n if (parsedShadowGenerator.depthScale) {\r\n shadowGenerator.depthScale = parsedShadowGenerator.depthScale;\r\n }\r\n\r\n if (parsedShadowGenerator.blurScale) {\r\n shadowGenerator.blurScale = parsedShadowGenerator.blurScale;\r\n }\r\n\r\n if (parsedShadowGenerator.blurBoxOffset) {\r\n shadowGenerator.blurBoxOffset = parsedShadowGenerator.blurBoxOffset;\r\n }\r\n\r\n if (parsedShadowGenerator.useKernelBlur) {\r\n shadowGenerator.useKernelBlur = parsedShadowGenerator.useKernelBlur;\r\n }\r\n\r\n if (parsedShadowGenerator.blurKernel) {\r\n shadowGenerator.blurKernel = parsedShadowGenerator.blurKernel;\r\n }\r\n\r\n return shadowGenerator;\r\n }\r\n}\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/clipPlaneFragmentDeclaration\";\nimport \"./ShadersInclude/packingFunctions\";\nimport \"./ShadersInclude/clipPlaneFragment\";\n\nconst name = \"depthPixelShader\";\nconst shader = `#ifdef ALPHATEST\nvarying vec2 vUV;uniform sampler2D diffuseSampler;\n#endif\n#include\nvarying float vDepthMetric;\n#ifdef PACKED\n#include\n#endif\n#ifdef STORE_CAMERASPACE_Z\nvarying vec4 vViewPos;\n#endif\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{\n#include\n#ifdef ALPHATEST\nif (texture2D(diffuseSampler,vUV).a<0.4)\ndiscard;\n#endif\n#ifdef STORE_CAMERASPACE_Z\n#ifdef PACKED\ngl_FragColor=pack(vViewPos.z);\n#else\ngl_FragColor=vec4(vViewPos.z,0.0,0.0,1.0);\n#endif\n#else\n#ifdef NONLINEARDEPTH\n#ifdef PACKED\ngl_FragColor=pack(gl_FragCoord.z);\n#else\ngl_FragColor=vec4(gl_FragCoord.z,0.0,0.0,0.0);\n#endif\n#else\n#ifdef PACKED\ngl_FragColor=pack(vDepthMetric);\n#else\ngl_FragColor=vec4(vDepthMetric,0.0,0.0,1.0);\n#endif\n#endif\n#endif\n}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const depthPixelShader = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/bonesDeclaration\";\nimport \"./ShadersInclude/bakedVertexAnimationDeclaration\";\nimport \"./ShadersInclude/morphTargetsVertexGlobalDeclaration\";\nimport \"./ShadersInclude/morphTargetsVertexDeclaration\";\nimport \"./ShadersInclude/clipPlaneVertexDeclaration\";\nimport \"./ShadersInclude/instancesDeclaration\";\nimport \"./ShadersInclude/morphTargetsVertexGlobal\";\nimport \"./ShadersInclude/morphTargetsVertex\";\nimport \"./ShadersInclude/instancesVertex\";\nimport \"./ShadersInclude/bonesVertex\";\nimport \"./ShadersInclude/bakedVertexAnimation\";\nimport \"./ShadersInclude/clipPlaneVertex\";\n\nconst name = \"depthVertexShader\";\nconst shader = `attribute vec3 position;\n#include\n#include\n#include\n#include[0..maxSimultaneousMorphTargets]\n#include\n#include\nuniform mat4 viewProjection;uniform vec2 depthValues;\n#if defined(ALPHATEST) || defined(NEED_UV)\nvarying vec2 vUV;uniform mat4 diffuseMatrix;\n#ifdef UV1\nattribute vec2 uv;\n#endif\n#ifdef UV2\nattribute vec2 uv2;\n#endif\n#endif\n#ifdef STORE_CAMERASPACE_Z\nuniform mat4 view;varying vec4 vViewPos;\n#endif\nvarying float vDepthMetric;\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void)\n{vec3 positionUpdated=position;\n#ifdef UV1\nvec2 uvUpdated=uv;\n#endif\n#include\n#include[0..maxSimultaneousMorphTargets]\n#include\n#include\n#include\nvec4 worldPos=finalWorld*vec4(positionUpdated,1.0);\n#include\ngl_Position=viewProjection*worldPos;\n#ifdef STORE_CAMERASPACE_Z\nvViewPos=view*worldPos;\n#else\n#ifdef USE_REVERSE_DEPTHBUFFER\nvDepthMetric=((-gl_Position.z+depthValues.x)/(depthValues.y));\n#else\nvDepthMetric=((gl_Position.z+depthValues.x)/(depthValues.y));\n#endif\n#endif\n#if defined(ALPHATEST) || defined(BASIC_RENDER)\n#ifdef UV1\nvUV=vec2(diffuseMatrix*vec4(uvUpdated,1.0,0.0));\n#endif\n#ifdef UV2\nvUV=vec2(diffuseMatrix*vec4(uv2,1.0,0.0));\n#endif\n#endif\n}\n`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const depthVertexShader = { name, shader };\n","import type { Nullable } from \"../types\";\r\nimport { Color4 } from \"../Maths/math.color\";\r\nimport type { Mesh } from \"../Meshes/mesh\";\r\nimport type { SubMesh } from \"../Meshes/subMesh\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport type { SmartArray } from \"../Misc/smartArray\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Texture } from \"../Materials/Textures/texture\";\r\nimport { RenderTargetTexture } from \"../Materials/Textures/renderTargetTexture\";\r\nimport { MaterialHelper } from \"../Materials/materialHelper\";\r\nimport { Camera } from \"../Cameras/camera\";\r\nimport { Constants } from \"../Engines/constants\";\r\n\r\nimport \"../Shaders/depth.fragment\";\r\nimport \"../Shaders/depth.vertex\";\r\nimport { _WarnImport } from \"../Misc/devTools\";\r\nimport { addClipPlaneUniforms, bindClipPlane, prepareStringDefinesForClipPlanes } from \"../Materials/clipPlaneMaterialHelper\";\r\n\r\nimport type { Material } from \"../Materials/material\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\n\r\n/**\r\n * This represents a depth renderer in Babylon.\r\n * A depth renderer will render to it's depth map every frame which can be displayed or used in post processing\r\n */\r\nexport class DepthRenderer {\r\n private _scene: Scene;\r\n private _depthMap: RenderTargetTexture;\r\n private readonly _storeNonLinearDepth: boolean;\r\n private readonly _storeCameraSpaceZ: boolean;\r\n\r\n /** Color used to clear the depth texture. Default: (1,0,0,1) */\r\n public clearColor: Color4;\r\n\r\n /** Get if the depth renderer is using packed depth or not */\r\n public readonly isPacked: boolean;\r\n\r\n private _camera: Nullable;\r\n\r\n /** Enable or disable the depth renderer. When disabled, the depth texture is not updated */\r\n public enabled = true;\r\n\r\n /** Force writing the transparent objects into the depth map */\r\n public forceDepthWriteTransparentMeshes = false;\r\n\r\n /**\r\n * Specifies that the depth renderer will only be used within\r\n * the camera it is created for.\r\n * This can help forcing its rendering during the camera processing.\r\n */\r\n public useOnlyInActiveCamera: boolean = false;\r\n\r\n /** If true, reverse the culling of materials before writing to the depth texture.\r\n * So, basically, when \"true\", back facing instead of front facing faces are rasterized into the texture\r\n */\r\n public reverseCulling = false;\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _SceneComponentInitialization: (scene: Scene) => void = (_) => {\r\n throw _WarnImport(\"DepthRendererSceneComponent\");\r\n };\r\n\r\n /**\r\n * Sets a specific material to be used to render a mesh/a list of meshes by the depth renderer\r\n * @param mesh mesh or array of meshes\r\n * @param material material to use by the depth render when rendering the mesh(es). If undefined is passed, the specific material created by the depth renderer will be used.\r\n */\r\n public setMaterialForRendering(mesh: AbstractMesh | AbstractMesh[], material?: Material): void {\r\n this._depthMap.setMaterialForRendering(mesh, material);\r\n }\r\n\r\n /**\r\n * Instantiates a depth renderer\r\n * @param scene The scene the renderer belongs to\r\n * @param type The texture type of the depth map (default: Engine.TEXTURETYPE_FLOAT)\r\n * @param camera The camera to be used to render the depth map (default: scene's active camera)\r\n * @param storeNonLinearDepth Defines whether the depth is stored linearly like in Babylon Shadows or directly like glFragCoord.z\r\n * @param samplingMode The sampling mode to be used with the render target (Linear, Nearest...) (default: TRILINEAR_SAMPLINGMODE)\r\n * @param storeCameraSpaceZ Defines whether the depth stored is the Z coordinate in camera space. If true, storeNonLinearDepth has no effect. (Default: false)\r\n * @param name Name of the render target (default: DepthRenderer)\r\n */\r\n constructor(\r\n scene: Scene,\r\n type: number = Constants.TEXTURETYPE_FLOAT,\r\n camera: Nullable = null,\r\n storeNonLinearDepth = false,\r\n samplingMode = Texture.TRILINEAR_SAMPLINGMODE,\r\n storeCameraSpaceZ = false,\r\n name?: string\r\n ) {\r\n this._scene = scene;\r\n this._storeNonLinearDepth = storeNonLinearDepth;\r\n this._storeCameraSpaceZ = storeCameraSpaceZ;\r\n this.isPacked = type === Constants.TEXTURETYPE_UNSIGNED_BYTE;\r\n if (this.isPacked) {\r\n this.clearColor = new Color4(1.0, 1.0, 1.0, 1.0);\r\n } else {\r\n this.clearColor = new Color4(storeCameraSpaceZ ? 1e8 : 1.0, 0.0, 0.0, 1.0);\r\n }\r\n\r\n DepthRenderer._SceneComponentInitialization(this._scene);\r\n\r\n const engine = scene.getEngine();\r\n\r\n this._camera = camera;\r\n\r\n if (samplingMode !== Texture.NEAREST_SAMPLINGMODE) {\r\n if (type === Constants.TEXTURETYPE_FLOAT && !engine._caps.textureFloatLinearFiltering) {\r\n samplingMode = Texture.NEAREST_SAMPLINGMODE;\r\n }\r\n if (type === Constants.TEXTURETYPE_HALF_FLOAT && !engine._caps.textureHalfFloatLinearFiltering) {\r\n samplingMode = Texture.NEAREST_SAMPLINGMODE;\r\n }\r\n }\r\n\r\n // Render target\r\n const format = this.isPacked || !engine._features.supportExtendedTextureFormats ? Constants.TEXTUREFORMAT_RGBA : Constants.TEXTUREFORMAT_R;\r\n this._depthMap = new RenderTargetTexture(\r\n name ?? \"DepthRenderer\",\r\n { width: engine.getRenderWidth(), height: engine.getRenderHeight() },\r\n this._scene,\r\n false,\r\n true,\r\n type,\r\n false,\r\n samplingMode,\r\n undefined,\r\n undefined,\r\n undefined,\r\n format\r\n );\r\n this._depthMap.wrapU = Texture.CLAMP_ADDRESSMODE;\r\n this._depthMap.wrapV = Texture.CLAMP_ADDRESSMODE;\r\n this._depthMap.refreshRate = 1;\r\n this._depthMap.renderParticles = false;\r\n this._depthMap.renderList = null;\r\n this._depthMap.noPrePassRenderer = true;\r\n\r\n // Camera to get depth map from to support multiple concurrent cameras\r\n this._depthMap.activeCamera = this._camera;\r\n this._depthMap.ignoreCameraViewport = true;\r\n this._depthMap.useCameraPostProcesses = false;\r\n\r\n // set default depth value to 1.0 (far away)\r\n this._depthMap.onClearObservable.add((engine) => {\r\n engine.clear(this.clearColor, true, true, true);\r\n });\r\n\r\n this._depthMap.onBeforeBindObservable.add(() => {\r\n engine._debugPushGroup?.(\"depth renderer\", 1);\r\n });\r\n\r\n this._depthMap.onAfterUnbindObservable.add(() => {\r\n engine._debugPopGroup?.(1);\r\n });\r\n\r\n this._depthMap.customIsReadyFunction = (mesh: AbstractMesh, refreshRate: number, preWarm?: boolean) => {\r\n if ((preWarm || refreshRate === 0) && mesh.subMeshes) {\r\n for (let i = 0; i < mesh.subMeshes.length; ++i) {\r\n const subMesh = mesh.subMeshes[i];\r\n const renderingMesh = subMesh.getRenderingMesh();\r\n\r\n const batch = renderingMesh._getInstancesRenderList(subMesh._id, !!subMesh.getReplacementMesh());\r\n const hardwareInstancedRendering =\r\n engine.getCaps().instancedArrays &&\r\n ((batch.visibleInstances[subMesh._id] !== null && batch.visibleInstances[subMesh._id] !== undefined) || renderingMesh.hasThinInstances);\r\n\r\n if (!this.isReady(subMesh, hardwareInstancedRendering)) {\r\n return false;\r\n }\r\n }\r\n }\r\n\r\n return true;\r\n };\r\n\r\n // Custom render function\r\n const renderSubMesh = (subMesh: SubMesh): void => {\r\n const renderingMesh = subMesh.getRenderingMesh();\r\n const effectiveMesh = subMesh.getEffectiveMesh();\r\n const scene = this._scene;\r\n const engine = scene.getEngine();\r\n const material = subMesh.getMaterial();\r\n\r\n effectiveMesh._internalAbstractMeshDataInfo._isActiveIntermediate = false;\r\n\r\n if (!material || effectiveMesh.infiniteDistance || material.disableDepthWrite || subMesh.verticesCount === 0 || subMesh._renderId === scene.getRenderId()) {\r\n return;\r\n }\r\n\r\n // Culling\r\n const detNeg = effectiveMesh._getWorldMatrixDeterminant() < 0;\r\n let sideOrientation = renderingMesh.overrideMaterialSideOrientation ?? material.sideOrientation;\r\n if (detNeg) {\r\n sideOrientation =\r\n sideOrientation === Constants.MATERIAL_ClockWiseSideOrientation\r\n ? Constants.MATERIAL_CounterClockWiseSideOrientation\r\n : Constants.MATERIAL_ClockWiseSideOrientation;\r\n }\r\n const reverseSideOrientation = sideOrientation === Constants.MATERIAL_ClockWiseSideOrientation;\r\n\r\n engine.setState(material.backFaceCulling, 0, false, reverseSideOrientation, this.reverseCulling ? !material.cullBackFaces : material.cullBackFaces);\r\n\r\n // Managing instances\r\n const batch = renderingMesh._getInstancesRenderList(subMesh._id, !!subMesh.getReplacementMesh());\r\n\r\n if (batch.mustReturn) {\r\n return;\r\n }\r\n\r\n const hardwareInstancedRendering =\r\n engine.getCaps().instancedArrays &&\r\n ((batch.visibleInstances[subMesh._id] !== null && batch.visibleInstances[subMesh._id] !== undefined) || renderingMesh.hasThinInstances);\r\n\r\n const camera = this._camera || scene.activeCamera;\r\n if (this.isReady(subMesh, hardwareInstancedRendering) && camera) {\r\n subMesh._renderId = scene.getRenderId();\r\n\r\n const renderingMaterial = effectiveMesh._internalAbstractMeshDataInfo._materialForRenderPass?.[engine.currentRenderPassId];\r\n\r\n let drawWrapper = subMesh._getDrawWrapper();\r\n if (!drawWrapper && renderingMaterial) {\r\n drawWrapper = renderingMaterial._getDrawWrapper();\r\n }\r\n const cameraIsOrtho = camera.mode === Camera.ORTHOGRAPHIC_CAMERA;\r\n\r\n if (!drawWrapper) {\r\n return;\r\n }\r\n\r\n const effect = drawWrapper.effect!;\r\n\r\n engine.enableEffect(drawWrapper);\r\n\r\n if (!hardwareInstancedRendering) {\r\n renderingMesh._bind(subMesh, effect, material.fillMode);\r\n }\r\n\r\n if (!renderingMaterial) {\r\n effect.setMatrix(\"viewProjection\", scene.getTransformMatrix());\r\n effect.setMatrix(\"world\", effectiveMesh.getWorldMatrix());\r\n if (this._storeCameraSpaceZ) {\r\n effect.setMatrix(\"view\", scene.getViewMatrix());\r\n }\r\n } else {\r\n renderingMaterial.bindForSubMesh(effectiveMesh.getWorldMatrix(), effectiveMesh as Mesh, subMesh);\r\n }\r\n\r\n let minZ: number, maxZ: number;\r\n\r\n if (cameraIsOrtho) {\r\n minZ = !engine.useReverseDepthBuffer && engine.isNDCHalfZRange ? 0 : 1;\r\n maxZ = engine.useReverseDepthBuffer && engine.isNDCHalfZRange ? 0 : 1;\r\n } else {\r\n minZ = engine.useReverseDepthBuffer && engine.isNDCHalfZRange ? camera.minZ : engine.isNDCHalfZRange ? 0 : camera.minZ;\r\n maxZ = engine.useReverseDepthBuffer && engine.isNDCHalfZRange ? 0 : camera.maxZ;\r\n }\r\n\r\n effect.setFloat2(\"depthValues\", minZ, minZ + maxZ);\r\n\r\n if (!renderingMaterial) {\r\n // Alpha test\r\n if (material.needAlphaTesting()) {\r\n const alphaTexture = material.getAlphaTestTexture();\r\n\r\n if (alphaTexture) {\r\n effect.setTexture(\"diffuseSampler\", alphaTexture);\r\n effect.setMatrix(\"diffuseMatrix\", alphaTexture.getTextureMatrix());\r\n }\r\n }\r\n\r\n // Bones\r\n if (renderingMesh.useBones && renderingMesh.computeBonesUsingShaders && renderingMesh.skeleton) {\r\n const skeleton = renderingMesh.skeleton;\r\n\r\n if (skeleton.isUsingTextureForMatrices) {\r\n const boneTexture = skeleton.getTransformMatrixTexture(renderingMesh);\r\n if (!boneTexture) {\r\n return;\r\n }\r\n\r\n effect.setTexture(\"boneSampler\", boneTexture);\r\n effect.setFloat(\"boneTextureWidth\", 4.0 * (skeleton.bones.length + 1));\r\n } else {\r\n effect.setMatrices(\"mBones\", skeleton.getTransformMatrices(renderingMesh));\r\n }\r\n }\r\n\r\n // Clip planes\r\n bindClipPlane(effect, material, scene);\r\n\r\n // Morph targets\r\n MaterialHelper.BindMorphTargetParameters(renderingMesh, effect);\r\n if (renderingMesh.morphTargetManager && renderingMesh.morphTargetManager.isUsingTextureForTargets) {\r\n renderingMesh.morphTargetManager._bind(effect);\r\n }\r\n }\r\n\r\n // Draw\r\n renderingMesh._processRendering(effectiveMesh, subMesh, effect, material.fillMode, batch, hardwareInstancedRendering, (isInstance, world) =>\r\n effect.setMatrix(\"world\", world)\r\n );\r\n }\r\n };\r\n\r\n this._depthMap.customRenderFunction = (\r\n opaqueSubMeshes: SmartArray,\r\n alphaTestSubMeshes: SmartArray,\r\n transparentSubMeshes: SmartArray,\r\n depthOnlySubMeshes: SmartArray\r\n ): void => {\r\n let index;\r\n\r\n if (depthOnlySubMeshes.length) {\r\n for (index = 0; index < depthOnlySubMeshes.length; index++) {\r\n renderSubMesh(depthOnlySubMeshes.data[index]);\r\n }\r\n }\r\n\r\n for (index = 0; index < opaqueSubMeshes.length; index++) {\r\n renderSubMesh(opaqueSubMeshes.data[index]);\r\n }\r\n\r\n for (index = 0; index < alphaTestSubMeshes.length; index++) {\r\n renderSubMesh(alphaTestSubMeshes.data[index]);\r\n }\r\n\r\n if (this.forceDepthWriteTransparentMeshes) {\r\n for (index = 0; index < transparentSubMeshes.length; index++) {\r\n renderSubMesh(transparentSubMeshes.data[index]);\r\n }\r\n } else {\r\n for (index = 0; index < transparentSubMeshes.length; index++) {\r\n transparentSubMeshes.data[index].getEffectiveMesh()._internalAbstractMeshDataInfo._isActiveIntermediate = false;\r\n }\r\n }\r\n };\r\n }\r\n\r\n /**\r\n * Creates the depth rendering effect and checks if the effect is ready.\r\n * @param subMesh The submesh to be used to render the depth map of\r\n * @param useInstances If multiple world instances should be used\r\n * @returns if the depth renderer is ready to render the depth map\r\n */\r\n public isReady(subMesh: SubMesh, useInstances: boolean): boolean {\r\n const engine = this._scene.getEngine();\r\n const mesh = subMesh.getMesh();\r\n const scene = mesh.getScene();\r\n\r\n const renderingMaterial = mesh._internalAbstractMeshDataInfo._materialForRenderPass?.[engine.currentRenderPassId];\r\n\r\n if (renderingMaterial) {\r\n return renderingMaterial.isReadyForSubMesh(mesh, subMesh, useInstances);\r\n }\r\n\r\n const material = subMesh.getMaterial();\r\n if (!material || material.disableDepthWrite) {\r\n return false;\r\n }\r\n\r\n const defines = [];\r\n\r\n const attribs = [VertexBuffer.PositionKind];\r\n\r\n // Alpha test\r\n if (material && material.needAlphaTesting() && material.getAlphaTestTexture()) {\r\n defines.push(\"#define ALPHATEST\");\r\n if (mesh.isVerticesDataPresent(VertexBuffer.UVKind)) {\r\n attribs.push(VertexBuffer.UVKind);\r\n defines.push(\"#define UV1\");\r\n }\r\n if (mesh.isVerticesDataPresent(VertexBuffer.UV2Kind)) {\r\n attribs.push(VertexBuffer.UV2Kind);\r\n defines.push(\"#define UV2\");\r\n }\r\n }\r\n\r\n // Bones\r\n if (mesh.useBones && mesh.computeBonesUsingShaders) {\r\n attribs.push(VertexBuffer.MatricesIndicesKind);\r\n attribs.push(VertexBuffer.MatricesWeightsKind);\r\n if (mesh.numBoneInfluencers > 4) {\r\n attribs.push(VertexBuffer.MatricesIndicesExtraKind);\r\n attribs.push(VertexBuffer.MatricesWeightsExtraKind);\r\n }\r\n defines.push(\"#define NUM_BONE_INFLUENCERS \" + mesh.numBoneInfluencers);\r\n defines.push(\"#define BonesPerMesh \" + (mesh.skeleton ? mesh.skeleton.bones.length + 1 : 0));\r\n\r\n const skeleton = subMesh.getRenderingMesh().skeleton;\r\n\r\n if (skeleton?.isUsingTextureForMatrices) {\r\n defines.push(\"#define BONETEXTURE\");\r\n }\r\n } else {\r\n defines.push(\"#define NUM_BONE_INFLUENCERS 0\");\r\n }\r\n\r\n // Morph targets\r\n const morphTargetManager = (mesh as Mesh).morphTargetManager;\r\n let numMorphInfluencers = 0;\r\n if (morphTargetManager) {\r\n if (morphTargetManager.numInfluencers > 0) {\r\n numMorphInfluencers = morphTargetManager.numInfluencers;\r\n\r\n defines.push(\"#define MORPHTARGETS\");\r\n defines.push(\"#define NUM_MORPH_INFLUENCERS \" + numMorphInfluencers);\r\n\r\n if (morphTargetManager.isUsingTextureForTargets) {\r\n defines.push(\"#define MORPHTARGETS_TEXTURE\");\r\n }\r\n\r\n MaterialHelper.PrepareAttributesForMorphTargetsInfluencers(attribs, mesh, numMorphInfluencers);\r\n }\r\n }\r\n\r\n // Instances\r\n if (useInstances) {\r\n defines.push(\"#define INSTANCES\");\r\n MaterialHelper.PushAttributesForInstances(attribs);\r\n if (subMesh.getRenderingMesh().hasThinInstances) {\r\n defines.push(\"#define THIN_INSTANCES\");\r\n }\r\n }\r\n\r\n // None linear depth\r\n if (this._storeNonLinearDepth) {\r\n defines.push(\"#define NONLINEARDEPTH\");\r\n }\r\n\r\n // Store camera space Z coordinate instead of NDC Z\r\n if (this._storeCameraSpaceZ) {\r\n defines.push(\"#define STORE_CAMERASPACE_Z\");\r\n }\r\n\r\n // Float Mode\r\n if (this.isPacked) {\r\n defines.push(\"#define PACKED\");\r\n }\r\n\r\n // Clip planes\r\n prepareStringDefinesForClipPlanes(material, scene, defines);\r\n\r\n // Get correct effect\r\n const drawWrapper = subMesh._getDrawWrapper(undefined, true)!;\r\n const cachedDefines = drawWrapper.defines;\r\n const join = defines.join(\"\\n\");\r\n if (cachedDefines !== join) {\r\n const uniforms = [\r\n \"world\",\r\n \"mBones\",\r\n \"boneTextureWidth\",\r\n \"viewProjection\",\r\n \"view\",\r\n \"diffuseMatrix\",\r\n \"depthValues\",\r\n \"morphTargetInfluences\",\r\n \"morphTargetTextureInfo\",\r\n \"morphTargetTextureIndices\",\r\n ];\r\n addClipPlaneUniforms(uniforms);\r\n\r\n drawWrapper.setEffect(\r\n engine.createEffect(\"depth\", attribs, uniforms, [\"diffuseSampler\", \"morphTargets\", \"boneSampler\"], join, undefined, undefined, undefined, {\r\n maxSimultaneousMorphTargets: numMorphInfluencers,\r\n }),\r\n join\r\n );\r\n }\r\n\r\n return drawWrapper.effect!.isReady();\r\n }\r\n\r\n /**\r\n * Gets the texture which the depth map will be written to.\r\n * @returns The depth map texture\r\n */\r\n public getDepthMap(): RenderTargetTexture {\r\n return this._depthMap;\r\n }\r\n\r\n /**\r\n * Disposes of the depth renderer.\r\n */\r\n public dispose(): void {\r\n const keysToDelete = [];\r\n for (const key in this._scene._depthRenderer) {\r\n const depthRenderer = this._scene._depthRenderer[key];\r\n if (depthRenderer === this) {\r\n keysToDelete.push(key);\r\n }\r\n }\r\n\r\n if (keysToDelete.length > 0) {\r\n this._depthMap.dispose();\r\n\r\n for (const key of keysToDelete) {\r\n delete this._scene._depthRenderer[key];\r\n }\r\n }\r\n }\r\n}\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\n\nconst name = \"minmaxReduxPixelShader\";\nconst shader = `varying vec2 vUV;uniform sampler2D textureSampler;\n#if defined(INITIAL)\nuniform sampler2D sourceTexture;uniform vec2 texSize;void main(void)\n{ivec2 coord=ivec2(vUV*(texSize-1.0));float f1=texelFetch(sourceTexture,coord,0).r;float f2=texelFetch(sourceTexture,coord+ivec2(1,0),0).r;float f3=texelFetch(sourceTexture,coord+ivec2(1,1),0).r;float f4=texelFetch(sourceTexture,coord+ivec2(0,1),0).r;float minz=min(min(min(f1,f2),f3),f4);\n#ifdef DEPTH_REDUX\nfloat maxz=max(max(max(sign(1.0-f1)*f1,sign(1.0-f2)*f2),sign(1.0-f3)*f3),sign(1.0-f4)*f4);\n#else\nfloat maxz=max(max(max(f1,f2),f3),f4);\n#endif\nglFragColor=vec4(minz,maxz,0.,0.);}\n#elif defined(MAIN)\nuniform vec2 texSize;void main(void)\n{ivec2 coord=ivec2(vUV*(texSize-1.0));vec2 f1=texelFetch(textureSampler,coord,0).rg;vec2 f2=texelFetch(textureSampler,coord+ivec2(1,0),0).rg;vec2 f3=texelFetch(textureSampler,coord+ivec2(1,1),0).rg;vec2 f4=texelFetch(textureSampler,coord+ivec2(0,1),0).rg;float minz=min(min(min(f1.x,f2.x),f3.x),f4.x);float maxz=max(max(max(f1.y,f2.y),f3.y),f4.y);glFragColor=vec4(minz,maxz,0.,0.);}\n#elif defined(ONEBEFORELAST)\nuniform ivec2 texSize;void main(void)\n{ivec2 coord=ivec2(vUV*vec2(texSize-1));vec2 f1=texelFetch(textureSampler,coord % texSize,0).rg;vec2 f2=texelFetch(textureSampler,(coord+ivec2(1,0)) % texSize,0).rg;vec2 f3=texelFetch(textureSampler,(coord+ivec2(1,1)) % texSize,0).rg;vec2 f4=texelFetch(textureSampler,(coord+ivec2(0,1)) % texSize,0).rg;float minz=min(f1.x,f2.x);float maxz=max(f1.y,f2.y);glFragColor=vec4(minz,maxz,0.,0.);}\n#elif defined(LAST)\nvoid main(void)\n{glFragColor=vec4(0.);if (true) { \ndiscard;}}\n#endif\n`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const minmaxReduxPixelShader = { name, shader };\n","import type { Nullable } from \"../types\";\r\nimport type { RenderTargetTexture } from \"../Materials/Textures/renderTargetTexture\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport { DepthRenderer } from \"../Rendering/depthRenderer\";\r\n\r\nimport { MinMaxReducer } from \"./minMaxReducer\";\r\n\r\n/**\r\n * This class is a small wrapper around the MinMaxReducer class to compute the min/max values of a depth texture\r\n */\r\nexport class DepthReducer extends MinMaxReducer {\r\n private _depthRenderer: Nullable;\r\n private _depthRendererId: string;\r\n\r\n /**\r\n * Gets the depth renderer used for the computation.\r\n * Note that the result is null if you provide your own renderer when calling setDepthRenderer.\r\n */\r\n public get depthRenderer(): Nullable {\r\n return this._depthRenderer;\r\n }\r\n\r\n /**\r\n * Creates a depth reducer\r\n * @param camera The camera used to render the depth texture\r\n */\r\n constructor(camera: Camera) {\r\n super(camera);\r\n }\r\n\r\n /**\r\n * Sets the depth renderer to use to generate the depth map\r\n * @param depthRenderer The depth renderer to use. If not provided, a new one will be created automatically\r\n * @param type The texture type of the depth map (default: TEXTURETYPE_HALF_FLOAT)\r\n * @param forceFullscreenViewport Forces the post processes used for the reduction to be applied without taking into account viewport (defaults to true)\r\n */\r\n public setDepthRenderer(depthRenderer: Nullable = null, type: number = Constants.TEXTURETYPE_HALF_FLOAT, forceFullscreenViewport = true): void {\r\n const scene = this._camera.getScene();\r\n\r\n if (this._depthRenderer) {\r\n delete scene._depthRenderer[this._depthRendererId];\r\n\r\n this._depthRenderer.dispose();\r\n this._depthRenderer = null;\r\n }\r\n\r\n if (depthRenderer === null) {\r\n if (!scene._depthRenderer) {\r\n scene._depthRenderer = {};\r\n }\r\n\r\n depthRenderer = this._depthRenderer = new DepthRenderer(scene, type, this._camera, false, Constants.TEXTURE_NEAREST_SAMPLINGMODE);\r\n depthRenderer.enabled = false;\r\n\r\n this._depthRendererId = \"minmax\" + this._camera.id;\r\n scene._depthRenderer[this._depthRendererId] = depthRenderer;\r\n }\r\n\r\n super.setSourceTexture(depthRenderer.getDepthMap(), true, type, forceFullscreenViewport);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public setSourceTexture(sourceTexture: RenderTargetTexture, depthRedux: boolean, type: number = Constants.TEXTURETYPE_HALF_FLOAT, forceFullscreenViewport = true): void {\r\n super.setSourceTexture(sourceTexture, depthRedux, type, forceFullscreenViewport);\r\n }\r\n\r\n /**\r\n * Activates the reduction computation.\r\n * When activated, the observers registered in onAfterReductionPerformed are\r\n * called after the computation is performed\r\n */\r\n public activate(): void {\r\n if (this._depthRenderer) {\r\n this._depthRenderer.enabled = true;\r\n }\r\n\r\n super.activate();\r\n }\r\n\r\n /**\r\n * Deactivates the reduction computation.\r\n */\r\n public deactivate(): void {\r\n super.deactivate();\r\n\r\n if (this._depthRenderer) {\r\n this._depthRenderer.enabled = false;\r\n }\r\n }\r\n\r\n /**\r\n * Disposes the depth reducer\r\n * @param disposeAll true to dispose all the resources. You should always call this function with true as the parameter (or without any parameter as it is the default one). This flag is meant to be used internally.\r\n */\r\n public dispose(disposeAll = true): void {\r\n super.dispose(disposeAll);\r\n\r\n if (this._depthRenderer && disposeAll) {\r\n const scene = this._depthRenderer.getDepthMap().getScene();\r\n if (scene) {\r\n delete scene._depthRenderer[this._depthRendererId];\r\n }\r\n\r\n this._depthRenderer.dispose();\r\n this._depthRenderer = null;\r\n }\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\nimport type { RenderTargetTexture } from \"../Materials/Textures/renderTargetTexture\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport type { Observer } from \"./observable\";\r\nimport { Observable } from \"./observable\";\r\nimport type { Effect } from \"../Materials/effect\";\r\nimport { PostProcess } from \"../PostProcesses/postProcess\";\r\nimport { PostProcessManager } from \"../PostProcesses/postProcessManager\";\r\n\r\nimport type { ThinEngine } from \"../Engines/thinEngine\";\r\n\r\nimport \"../Shaders/minmaxRedux.fragment\";\r\n\r\n/**\r\n * This class computes a min/max reduction from a texture: it means it computes the minimum\r\n * and maximum values from all values of the texture.\r\n * It is performed on the GPU for better performances, thanks to a succession of post processes.\r\n * The source values are read from the red channel of the texture.\r\n */\r\nexport class MinMaxReducer {\r\n /**\r\n * Observable triggered when the computation has been performed\r\n */\r\n public onAfterReductionPerformed = new Observable<{ min: number; max: number }>();\r\n\r\n protected _camera: Camera;\r\n protected _sourceTexture: Nullable;\r\n protected _reductionSteps: Nullable>;\r\n protected _postProcessManager: PostProcessManager;\r\n protected _onAfterUnbindObserver: Nullable>;\r\n protected _forceFullscreenViewport = true;\r\n protected _onContextRestoredObserver: Nullable>;\r\n\r\n /**\r\n * Creates a min/max reducer\r\n * @param camera The camera to use for the post processes\r\n */\r\n constructor(camera: Camera) {\r\n this._camera = camera;\r\n this._postProcessManager = new PostProcessManager(camera.getScene());\r\n\r\n this._onContextRestoredObserver = camera.getEngine().onContextRestoredObservable.add(() => {\r\n this._postProcessManager._rebuild();\r\n });\r\n }\r\n\r\n /**\r\n * Gets the texture used to read the values from.\r\n */\r\n public get sourceTexture(): Nullable {\r\n return this._sourceTexture;\r\n }\r\n\r\n /**\r\n * Sets the source texture to read the values from.\r\n * One must indicate if the texture is a depth texture or not through the depthRedux parameter\r\n * because in such textures '1' value must not be taken into account to compute the maximum\r\n * as this value is used to clear the texture.\r\n * Note that the computation is not activated by calling this function, you must call activate() for that!\r\n * @param sourceTexture The texture to read the values from. The values should be in the red channel.\r\n * @param depthRedux Indicates if the texture is a depth texture or not\r\n * @param type The type of the textures created for the reduction (defaults to TEXTURETYPE_HALF_FLOAT)\r\n * @param forceFullscreenViewport Forces the post processes used for the reduction to be applied without taking into account viewport (defaults to true)\r\n */\r\n public setSourceTexture(sourceTexture: RenderTargetTexture, depthRedux: boolean, type: number = Constants.TEXTURETYPE_HALF_FLOAT, forceFullscreenViewport = true): void {\r\n if (sourceTexture === this._sourceTexture) {\r\n return;\r\n }\r\n\r\n this.dispose(false);\r\n\r\n this._sourceTexture = sourceTexture;\r\n this._reductionSteps = [];\r\n this._forceFullscreenViewport = forceFullscreenViewport;\r\n\r\n const scene = this._camera.getScene();\r\n\r\n // create the first step\r\n const reductionInitial = new PostProcess(\r\n \"Initial reduction phase\",\r\n \"minmaxRedux\", // shader\r\n [\"texSize\"],\r\n [\"sourceTexture\"], // textures\r\n 1.0, // options\r\n null, // camera\r\n Constants.TEXTURE_NEAREST_NEAREST, // sampling\r\n scene.getEngine(), // engine\r\n false, // reusable\r\n \"#define INITIAL\" + (depthRedux ? \"\\n#define DEPTH_REDUX\" : \"\"), // defines\r\n type,\r\n undefined,\r\n undefined,\r\n undefined,\r\n Constants.TEXTUREFORMAT_RG\r\n );\r\n\r\n reductionInitial.autoClear = false;\r\n reductionInitial.forceFullscreenViewport = forceFullscreenViewport;\r\n\r\n let w = this._sourceTexture.getRenderWidth(),\r\n h = this._sourceTexture.getRenderHeight();\r\n\r\n reductionInitial.onApply = ((w: number, h: number) => {\r\n return (effect: Effect) => {\r\n effect.setTexture(\"sourceTexture\", this._sourceTexture);\r\n effect.setFloat2(\"texSize\", w, h);\r\n };\r\n })(w, h);\r\n\r\n this._reductionSteps.push(reductionInitial);\r\n\r\n let index = 1;\r\n\r\n // create the additional steps\r\n while (w > 1 || h > 1) {\r\n w = Math.max(Math.round(w / 2), 1);\r\n h = Math.max(Math.round(h / 2), 1);\r\n\r\n const reduction = new PostProcess(\r\n \"Reduction phase \" + index,\r\n \"minmaxRedux\", // shader\r\n [\"texSize\"],\r\n null,\r\n { width: w, height: h }, // options\r\n null, // camera\r\n Constants.TEXTURE_NEAREST_NEAREST, // sampling\r\n scene.getEngine(), // engine\r\n false, // reusable\r\n \"#define \" + (w == 1 && h == 1 ? \"LAST\" : w == 1 || h == 1 ? \"ONEBEFORELAST\" : \"MAIN\"), // defines\r\n type,\r\n undefined,\r\n undefined,\r\n undefined,\r\n Constants.TEXTUREFORMAT_RG\r\n );\r\n\r\n reduction.autoClear = false;\r\n reduction.forceFullscreenViewport = forceFullscreenViewport;\r\n\r\n reduction.onApply = ((w: number, h: number) => {\r\n return (effect: Effect) => {\r\n if (w == 1 || h == 1) {\r\n effect.setInt2(\"texSize\", w, h);\r\n } else {\r\n effect.setFloat2(\"texSize\", w, h);\r\n }\r\n };\r\n })(w, h);\r\n\r\n this._reductionSteps.push(reduction);\r\n\r\n index++;\r\n\r\n if (w == 1 && h == 1) {\r\n const func = (w: number, h: number, reduction: PostProcess) => {\r\n const buffer = new Float32Array(4 * w * h),\r\n minmax = { min: 0, max: 0 };\r\n return () => {\r\n scene.getEngine()._readTexturePixels(reduction.inputTexture.texture!, w, h, -1, 0, buffer, false);\r\n minmax.min = buffer[0];\r\n minmax.max = buffer[1];\r\n this.onAfterReductionPerformed.notifyObservers(minmax);\r\n };\r\n };\r\n reduction.onAfterRenderObservable.add(func(w, h, reduction));\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Defines the refresh rate of the computation.\r\n * Use 0 to compute just once, 1 to compute on every frame, 2 to compute every two frames and so on...\r\n */\r\n public get refreshRate(): number {\r\n return this._sourceTexture ? this._sourceTexture.refreshRate : -1;\r\n }\r\n\r\n public set refreshRate(value: number) {\r\n if (this._sourceTexture) {\r\n this._sourceTexture.refreshRate = value;\r\n }\r\n }\r\n\r\n protected _activated = false;\r\n\r\n /**\r\n * Gets the activation status of the reducer\r\n */\r\n public get activated(): boolean {\r\n return this._activated;\r\n }\r\n\r\n /**\r\n * Activates the reduction computation.\r\n * When activated, the observers registered in onAfterReductionPerformed are\r\n * called after the computation is performed\r\n */\r\n public activate(): void {\r\n if (this._onAfterUnbindObserver || !this._sourceTexture) {\r\n return;\r\n }\r\n\r\n this._onAfterUnbindObserver = this._sourceTexture.onAfterUnbindObservable.add(() => {\r\n const engine = this._camera.getScene().getEngine();\r\n engine._debugPushGroup?.(`min max reduction`, 1);\r\n this._reductionSteps![0].activate(this._camera);\r\n this._postProcessManager.directRender(this._reductionSteps!, this._reductionSteps![0].inputTexture, this._forceFullscreenViewport);\r\n engine.unBindFramebuffer(this._reductionSteps![0].inputTexture, false);\r\n engine._debugPopGroup?.(1);\r\n });\r\n\r\n this._activated = true;\r\n }\r\n\r\n /**\r\n * Deactivates the reduction computation.\r\n */\r\n public deactivate(): void {\r\n if (!this._onAfterUnbindObserver || !this._sourceTexture) {\r\n return;\r\n }\r\n\r\n this._sourceTexture.onAfterUnbindObservable.remove(this._onAfterUnbindObserver);\r\n this._onAfterUnbindObserver = null;\r\n this._activated = false;\r\n }\r\n\r\n /**\r\n * Disposes the min/max reducer\r\n * @param disposeAll true to dispose all the resources. You should always call this function with true as the parameter (or without any parameter as it is the default one). This flag is meant to be used internally.\r\n */\r\n public dispose(disposeAll = true): void {\r\n if (disposeAll) {\r\n this.onAfterReductionPerformed.clear();\r\n\r\n if (this._onContextRestoredObserver) {\r\n this._camera.getEngine().onContextRestoredObservable.remove(this._onContextRestoredObserver);\r\n this._onContextRestoredObserver = null;\r\n }\r\n }\r\n\r\n this.deactivate();\r\n\r\n if (this._reductionSteps) {\r\n for (let i = 0; i < this._reductionSteps.length; ++i) {\r\n this._reductionSteps[i].dispose();\r\n }\r\n this._reductionSteps = null;\r\n }\r\n\r\n if (this._postProcessManager && disposeAll) {\r\n this._postProcessManager.dispose();\r\n }\r\n\r\n this._sourceTexture = null;\r\n }\r\n}\r\n","import type { Nullable } from \"../../types\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { Matrix, Vector3 } from \"../../Maths/math.vector\";\r\nimport type { SubMesh } from \"../../Meshes/subMesh\";\r\n\r\nimport type { IShadowLight } from \"../../Lights/shadowLight\";\r\nimport type { Effect } from \"../../Materials/effect\";\r\nimport { RenderTargetTexture } from \"../../Materials/Textures/renderTargetTexture\";\r\n\r\nimport { Constants } from \"../../Engines/constants\";\r\n\r\nimport \"../../Shaders/shadowMap.fragment\";\r\nimport \"../../Shaders/shadowMap.vertex\";\r\nimport \"../../Shaders/depthBoxBlur.fragment\";\r\nimport type { Observer } from \"../../Misc/observable\";\r\nimport { _WarnImport } from \"../../Misc/devTools\";\r\nimport { ShadowGenerator } from \"./shadowGenerator\";\r\nimport type { DirectionalLight } from \"../directionalLight\";\r\n\r\nimport { BoundingInfo } from \"../../Culling/boundingInfo\";\r\nimport type { DepthRenderer } from \"../../Rendering/depthRenderer\";\r\nimport { DepthReducer } from \"../../Misc/depthReducer\";\r\n\r\nimport { Logger } from \"../../Misc/logger\";\r\nimport { EngineStore } from \"../../Engines/engineStore\";\r\nimport type { Camera } from \"../../Cameras/camera\";\r\n\r\ninterface ICascade {\r\n prevBreakDistance: number;\r\n breakDistance: number;\r\n}\r\n\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nconst UpDir = Vector3.Up();\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nconst ZeroVec = Vector3.Zero();\r\n\r\nconst tmpv1 = new Vector3(),\r\n tmpv2 = new Vector3(),\r\n tmpMatrix = new Matrix();\r\n\r\n/**\r\n * A CSM implementation allowing casting shadows on large scenes.\r\n * Documentation : https://doc.babylonjs.com/babylon101/cascadedShadows\r\n * Based on: https://github.com/TheRealMJP/Shadows and https://johanmedestrom.wordpress.com/2016/03/18/opengl-cascaded-shadow-maps/\r\n */\r\nexport class CascadedShadowGenerator extends ShadowGenerator {\r\n private static readonly _FrustumCornersNDCSpace = [\r\n new Vector3(-1.0, +1.0, -1.0),\r\n new Vector3(+1.0, +1.0, -1.0),\r\n new Vector3(+1.0, -1.0, -1.0),\r\n new Vector3(-1.0, -1.0, -1.0),\r\n new Vector3(-1.0, +1.0, +1.0),\r\n new Vector3(+1.0, +1.0, +1.0),\r\n new Vector3(+1.0, -1.0, +1.0),\r\n new Vector3(-1.0, -1.0, +1.0),\r\n ];\r\n\r\n /**\r\n * Name of the CSM class\r\n */\r\n public static CLASSNAME = \"CascadedShadowGenerator\";\r\n\r\n /**\r\n * Defines the default number of cascades used by the CSM.\r\n */\r\n public static readonly DEFAULT_CASCADES_COUNT = 4;\r\n /**\r\n * Defines the minimum number of cascades used by the CSM.\r\n */\r\n public static MIN_CASCADES_COUNT = 2;\r\n /**\r\n * Defines the maximum number of cascades used by the CSM.\r\n */\r\n public static MAX_CASCADES_COUNT = 4;\r\n\r\n protected _validateFilter(filter: number): number {\r\n if (filter === ShadowGenerator.FILTER_NONE || filter === ShadowGenerator.FILTER_PCF || filter === ShadowGenerator.FILTER_PCSS) {\r\n return filter;\r\n }\r\n\r\n console.error('Unsupported filter \"' + filter + '\"!');\r\n\r\n return ShadowGenerator.FILTER_NONE;\r\n }\r\n\r\n /**\r\n * Gets or sets the actual darkness of the soft shadows while using PCSS filtering (value between 0. and 1.)\r\n */\r\n public penumbraDarkness: number;\r\n\r\n private _numCascades: number;\r\n\r\n /**\r\n * Gets or set the number of cascades used by the CSM.\r\n */\r\n public get numCascades(): number {\r\n return this._numCascades;\r\n }\r\n\r\n public set numCascades(value: number) {\r\n value = Math.min(Math.max(value, CascadedShadowGenerator.MIN_CASCADES_COUNT), CascadedShadowGenerator.MAX_CASCADES_COUNT);\r\n if (value === this._numCascades) {\r\n return;\r\n }\r\n\r\n this._numCascades = value;\r\n this.recreateShadowMap();\r\n this._recreateSceneUBOs();\r\n }\r\n\r\n /**\r\n * Sets this to true if you want that the edges of the shadows don't \"swimm\" / \"shimmer\" when rotating the camera.\r\n * The trade off is that you lose some precision in the shadow rendering when enabling this setting.\r\n */\r\n public stabilizeCascades: boolean;\r\n\r\n private _freezeShadowCastersBoundingInfo: boolean;\r\n private _freezeShadowCastersBoundingInfoObservable: Nullable>;\r\n\r\n /**\r\n * Enables or disables the shadow casters bounding info computation.\r\n * If your shadow casters don't move, you can disable this feature.\r\n * If it is enabled, the bounding box computation is done every frame.\r\n */\r\n public get freezeShadowCastersBoundingInfo(): boolean {\r\n return this._freezeShadowCastersBoundingInfo;\r\n }\r\n\r\n public set freezeShadowCastersBoundingInfo(freeze: boolean) {\r\n if (this._freezeShadowCastersBoundingInfoObservable && freeze) {\r\n this._scene.onBeforeRenderObservable.remove(this._freezeShadowCastersBoundingInfoObservable);\r\n this._freezeShadowCastersBoundingInfoObservable = null;\r\n }\r\n\r\n if (!this._freezeShadowCastersBoundingInfoObservable && !freeze) {\r\n this._freezeShadowCastersBoundingInfoObservable = this._scene.onBeforeRenderObservable.add(this._computeShadowCastersBoundingInfo.bind(this));\r\n }\r\n\r\n this._freezeShadowCastersBoundingInfo = freeze;\r\n\r\n if (freeze) {\r\n this._computeShadowCastersBoundingInfo();\r\n }\r\n }\r\n\r\n private _scbiMin: Vector3;\r\n private _scbiMax: Vector3;\r\n\r\n protected _computeShadowCastersBoundingInfo(): void {\r\n this._scbiMin.copyFromFloats(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\r\n this._scbiMax.copyFromFloats(Number.MIN_VALUE, Number.MIN_VALUE, Number.MIN_VALUE);\r\n\r\n if (this._shadowMap && this._shadowMap.renderList) {\r\n const renderList = this._shadowMap.renderList;\r\n for (let meshIndex = 0; meshIndex < renderList.length; meshIndex++) {\r\n const mesh = renderList[meshIndex];\r\n\r\n if (!mesh) {\r\n continue;\r\n }\r\n\r\n const boundingInfo = mesh.getBoundingInfo(),\r\n boundingBox = boundingInfo.boundingBox;\r\n\r\n this._scbiMin.minimizeInPlace(boundingBox.minimumWorld);\r\n this._scbiMax.maximizeInPlace(boundingBox.maximumWorld);\r\n }\r\n\r\n const meshes = this._scene.meshes;\r\n for (let meshIndex = 0; meshIndex < meshes.length; meshIndex++) {\r\n const mesh = meshes[meshIndex];\r\n\r\n if (!mesh || !mesh.isVisible || !mesh.isEnabled || !mesh.receiveShadows) {\r\n continue;\r\n }\r\n\r\n const boundingInfo = mesh.getBoundingInfo(),\r\n boundingBox = boundingInfo.boundingBox;\r\n\r\n this._scbiMin.minimizeInPlace(boundingBox.minimumWorld);\r\n this._scbiMax.maximizeInPlace(boundingBox.maximumWorld);\r\n }\r\n }\r\n\r\n this._shadowCastersBoundingInfo.reConstruct(this._scbiMin, this._scbiMax);\r\n }\r\n\r\n protected _shadowCastersBoundingInfo: BoundingInfo;\r\n\r\n /**\r\n * Gets or sets the shadow casters bounding info.\r\n * If you provide your own shadow casters bounding info, first enable freezeShadowCastersBoundingInfo\r\n * so that the system won't overwrite the bounds you provide\r\n */\r\n public get shadowCastersBoundingInfo(): BoundingInfo {\r\n return this._shadowCastersBoundingInfo;\r\n }\r\n\r\n public set shadowCastersBoundingInfo(boundingInfo: BoundingInfo) {\r\n this._shadowCastersBoundingInfo = boundingInfo;\r\n }\r\n\r\n protected _breaksAreDirty: boolean;\r\n\r\n protected _minDistance: number;\r\n protected _maxDistance: number;\r\n\r\n /**\r\n * Sets the minimal and maximal distances to use when computing the cascade breaks.\r\n *\r\n * The values of min / max are typically the depth zmin and zmax values of your scene, for a given frame.\r\n * If you don't know these values, simply leave them to their defaults and don't call this function.\r\n * @param min minimal distance for the breaks (default to 0.)\r\n * @param max maximal distance for the breaks (default to 1.)\r\n */\r\n public setMinMaxDistance(min: number, max: number): void {\r\n if (this._minDistance === min && this._maxDistance === max) {\r\n return;\r\n }\r\n\r\n if (min > max) {\r\n min = 0;\r\n max = 1;\r\n }\r\n\r\n if (min < 0) {\r\n min = 0;\r\n }\r\n\r\n if (max > 1) {\r\n max = 1;\r\n }\r\n\r\n this._minDistance = min;\r\n this._maxDistance = max;\r\n this._breaksAreDirty = true;\r\n }\r\n\r\n /** Gets the minimal distance used in the cascade break computation */\r\n public get minDistance(): number {\r\n return this._minDistance;\r\n }\r\n\r\n /** Gets the maximal distance used in the cascade break computation */\r\n public get maxDistance(): number {\r\n return this._maxDistance;\r\n }\r\n\r\n /**\r\n * Gets the class name of that object\r\n * @returns \"CascadedShadowGenerator\"\r\n */\r\n public getClassName(): string {\r\n return CascadedShadowGenerator.CLASSNAME;\r\n }\r\n\r\n private _cascadeMinExtents: Array;\r\n private _cascadeMaxExtents: Array;\r\n\r\n /**\r\n * Gets a cascade minimum extents\r\n * @param cascadeIndex index of the cascade\r\n * @returns the minimum cascade extents\r\n */\r\n public getCascadeMinExtents(cascadeIndex: number): Nullable {\r\n return cascadeIndex >= 0 && cascadeIndex < this._numCascades ? this._cascadeMinExtents[cascadeIndex] : null;\r\n }\r\n\r\n /**\r\n * Gets a cascade maximum extents\r\n * @param cascadeIndex index of the cascade\r\n * @returns the maximum cascade extents\r\n */\r\n public getCascadeMaxExtents(cascadeIndex: number): Nullable {\r\n return cascadeIndex >= 0 && cascadeIndex < this._numCascades ? this._cascadeMaxExtents[cascadeIndex] : null;\r\n }\r\n\r\n private _cascades: Array;\r\n private _currentLayer: number;\r\n private _viewSpaceFrustumsZ: Array;\r\n private _viewMatrices: Array;\r\n private _projectionMatrices: Array;\r\n private _transformMatrices: Array;\r\n private _transformMatricesAsArray: Float32Array;\r\n private _frustumLengths: Array;\r\n private _lightSizeUVCorrection: Array;\r\n private _depthCorrection: Array;\r\n private _frustumCornersWorldSpace: Array>;\r\n private _frustumCenter: Array;\r\n private _shadowCameraPos: Array;\r\n\r\n private _shadowMaxZ: number;\r\n /**\r\n * Gets the shadow max z distance. It's the limit beyond which shadows are not displayed.\r\n * It defaults to camera.maxZ\r\n */\r\n public get shadowMaxZ(): number {\r\n if (!this._getCamera()) {\r\n return 0;\r\n }\r\n return this._shadowMaxZ;\r\n }\r\n /**\r\n * Sets the shadow max z distance.\r\n */\r\n public set shadowMaxZ(value: number) {\r\n const camera = this._getCamera();\r\n if (!camera) {\r\n this._shadowMaxZ = value;\r\n return;\r\n }\r\n if (this._shadowMaxZ === value || value < camera.minZ || value > camera.maxZ) {\r\n return;\r\n }\r\n this._shadowMaxZ = value;\r\n this._light._markMeshesAsLightDirty();\r\n this._breaksAreDirty = true;\r\n }\r\n\r\n protected _debug: boolean;\r\n\r\n /**\r\n * Gets or sets the debug flag.\r\n * When enabled, the cascades are materialized by different colors on the screen.\r\n */\r\n public get debug(): boolean {\r\n return this._debug;\r\n }\r\n\r\n public set debug(dbg: boolean) {\r\n this._debug = dbg;\r\n this._light._markMeshesAsLightDirty();\r\n }\r\n\r\n private _depthClamp: boolean;\r\n\r\n /**\r\n * Gets or sets the depth clamping value.\r\n *\r\n * When enabled, it improves the shadow quality because the near z plane of the light frustum don't need to be adjusted\r\n * to account for the shadow casters far away.\r\n *\r\n * Note that this property is incompatible with PCSS filtering, so it won't be used in that case.\r\n */\r\n public get depthClamp(): boolean {\r\n return this._depthClamp;\r\n }\r\n\r\n public set depthClamp(value: boolean) {\r\n this._depthClamp = value;\r\n }\r\n\r\n private _cascadeBlendPercentage: number;\r\n\r\n /**\r\n * Gets or sets the percentage of blending between two cascades (value between 0. and 1.).\r\n * It defaults to 0.1 (10% blending).\r\n */\r\n public get cascadeBlendPercentage(): number {\r\n return this._cascadeBlendPercentage;\r\n }\r\n\r\n public set cascadeBlendPercentage(value: number) {\r\n this._cascadeBlendPercentage = value;\r\n this._light._markMeshesAsLightDirty();\r\n }\r\n\r\n private _lambda: number;\r\n\r\n /**\r\n * Gets or set the lambda parameter.\r\n * This parameter is used to split the camera frustum and create the cascades.\r\n * It's a value between 0. and 1.: If 0, the split is a uniform split of the frustum, if 1 it is a logarithmic split.\r\n * For all values in-between, it's a linear combination of the uniform and logarithm split algorithm.\r\n */\r\n public get lambda(): number {\r\n return this._lambda;\r\n }\r\n\r\n public set lambda(value: number) {\r\n const lambda = Math.min(Math.max(value, 0), 1);\r\n if (this._lambda == lambda) {\r\n return;\r\n }\r\n this._lambda = lambda;\r\n this._breaksAreDirty = true;\r\n }\r\n\r\n /**\r\n * Gets the view matrix corresponding to a given cascade\r\n * @param cascadeNum cascade to retrieve the view matrix from\r\n * @returns the cascade view matrix\r\n */\r\n public getCascadeViewMatrix(cascadeNum: number): Nullable {\r\n return cascadeNum >= 0 && cascadeNum < this._numCascades ? this._viewMatrices[cascadeNum] : null;\r\n }\r\n\r\n /**\r\n * Gets the projection matrix corresponding to a given cascade\r\n * @param cascadeNum cascade to retrieve the projection matrix from\r\n * @returns the cascade projection matrix\r\n */\r\n public getCascadeProjectionMatrix(cascadeNum: number): Nullable {\r\n return cascadeNum >= 0 && cascadeNum < this._numCascades ? this._projectionMatrices[cascadeNum] : null;\r\n }\r\n\r\n /**\r\n * Gets the transformation matrix corresponding to a given cascade\r\n * @param cascadeNum cascade to retrieve the transformation matrix from\r\n * @returns the cascade transformation matrix\r\n */\r\n public getCascadeTransformMatrix(cascadeNum: number): Nullable {\r\n return cascadeNum >= 0 && cascadeNum < this._numCascades ? this._transformMatrices[cascadeNum] : null;\r\n }\r\n\r\n private _depthRenderer: Nullable;\r\n /**\r\n * Sets the depth renderer to use when autoCalcDepthBounds is enabled.\r\n *\r\n * Note that if no depth renderer is set, a new one will be automatically created internally when necessary.\r\n *\r\n * You should call this function if you already have a depth renderer enabled in your scene, to avoid\r\n * doing multiple depth rendering each frame. If you provide your own depth renderer, make sure it stores linear depth!\r\n * @param depthRenderer The depth renderer to use when autoCalcDepthBounds is enabled. If you pass null or don't call this function at all, a depth renderer will be automatically created\r\n */\r\n public setDepthRenderer(depthRenderer: Nullable): void {\r\n this._depthRenderer = depthRenderer;\r\n\r\n if (this._depthReducer) {\r\n this._depthReducer.setDepthRenderer(this._depthRenderer);\r\n }\r\n }\r\n\r\n private _depthReducer: Nullable;\r\n private _autoCalcDepthBounds: boolean;\r\n\r\n /**\r\n * Gets or sets the autoCalcDepthBounds property.\r\n *\r\n * When enabled, a depth rendering pass is first performed (with an internally created depth renderer or with the one\r\n * you provide by calling setDepthRenderer). Then, a min/max reducing is applied on the depth map to compute the\r\n * minimal and maximal depth of the map and those values are used as inputs for the setMinMaxDistance() function.\r\n * It can greatly enhance the shadow quality, at the expense of more GPU works.\r\n * When using this option, you should increase the value of the lambda parameter, and even set it to 1 for best results.\r\n */\r\n public get autoCalcDepthBounds(): boolean {\r\n return this._autoCalcDepthBounds;\r\n }\r\n\r\n public set autoCalcDepthBounds(value: boolean) {\r\n const camera = this._getCamera();\r\n\r\n if (!camera) {\r\n return;\r\n }\r\n\r\n this._autoCalcDepthBounds = value;\r\n\r\n if (!value) {\r\n if (this._depthReducer) {\r\n this._depthReducer.deactivate();\r\n }\r\n this.setMinMaxDistance(0, 1);\r\n return;\r\n }\r\n\r\n if (!this._depthReducer) {\r\n this._depthReducer = new DepthReducer(camera);\r\n this._depthReducer.onAfterReductionPerformed.add((minmax: { min: number; max: number }) => {\r\n let min = minmax.min,\r\n max = minmax.max;\r\n if (min >= max) {\r\n min = 0;\r\n max = 1;\r\n }\r\n if (min != this._minDistance || max != this._maxDistance) {\r\n this.setMinMaxDistance(min, max);\r\n }\r\n });\r\n this._depthReducer.setDepthRenderer(this._depthRenderer);\r\n }\r\n\r\n this._depthReducer.activate();\r\n }\r\n\r\n /**\r\n * Defines the refresh rate of the min/max computation used when autoCalcDepthBounds is set to true\r\n * Use 0 to compute just once, 1 to compute on every frame, 2 to compute every two frames and so on...\r\n * Note that if you provided your own depth renderer through a call to setDepthRenderer, you are responsible\r\n * for setting the refresh rate on the renderer yourself!\r\n */\r\n public get autoCalcDepthBoundsRefreshRate(): number {\r\n return this._depthReducer?.depthRenderer?.getDepthMap().refreshRate ?? -1;\r\n }\r\n\r\n public set autoCalcDepthBoundsRefreshRate(value: number) {\r\n if (this._depthReducer?.depthRenderer) {\r\n this._depthReducer.depthRenderer.getDepthMap().refreshRate = value;\r\n }\r\n }\r\n\r\n /**\r\n * Create the cascade breaks according to the lambda, shadowMaxZ and min/max distance properties, as well as the camera near and far planes.\r\n * This function is automatically called when updating lambda, shadowMaxZ and min/max distances, however you should call it yourself if\r\n * you change the camera near/far planes!\r\n */\r\n public splitFrustum(): void {\r\n this._breaksAreDirty = true;\r\n }\r\n\r\n private _splitFrustum(): void {\r\n const camera = this._getCamera();\r\n if (!camera) {\r\n return;\r\n }\r\n\r\n const near = camera.minZ,\r\n far = camera.maxZ || this._shadowMaxZ, // account for infinite far plane (ie. maxZ = 0)\r\n cameraRange = far - near,\r\n minDistance = this._minDistance,\r\n maxDistance = this._shadowMaxZ < far && this._shadowMaxZ >= near ? Math.min((this._shadowMaxZ - near) / (far - near), this._maxDistance) : this._maxDistance;\r\n\r\n const minZ = near + minDistance * cameraRange,\r\n maxZ = near + maxDistance * cameraRange;\r\n\r\n const range = maxZ - minZ,\r\n ratio = maxZ / minZ;\r\n\r\n for (let cascadeIndex = 0; cascadeIndex < this._cascades.length; ++cascadeIndex) {\r\n const p = (cascadeIndex + 1) / this._numCascades,\r\n log = minZ * ratio ** p,\r\n uniform = minZ + range * p;\r\n\r\n const d = this._lambda * (log - uniform) + uniform;\r\n\r\n this._cascades[cascadeIndex].prevBreakDistance = cascadeIndex === 0 ? minDistance : this._cascades[cascadeIndex - 1].breakDistance;\r\n this._cascades[cascadeIndex].breakDistance = (d - near) / cameraRange;\r\n\r\n this._viewSpaceFrustumsZ[cascadeIndex] = d;\r\n this._frustumLengths[cascadeIndex] = (this._cascades[cascadeIndex].breakDistance - this._cascades[cascadeIndex].prevBreakDistance) * cameraRange;\r\n }\r\n\r\n this._breaksAreDirty = false;\r\n }\r\n\r\n private _computeMatrices(): void {\r\n const scene = this._scene;\r\n\r\n const camera = this._getCamera();\r\n if (!camera) {\r\n return;\r\n }\r\n\r\n Vector3.NormalizeToRef(this._light.getShadowDirection(0), this._lightDirection);\r\n if (Math.abs(Vector3.Dot(this._lightDirection, Vector3.Up())) === 1.0) {\r\n this._lightDirection.z = 0.0000000000001; // Required to avoid perfectly perpendicular light\r\n }\r\n\r\n this._cachedDirection.copyFrom(this._lightDirection);\r\n\r\n const useReverseDepthBuffer = scene.getEngine().useReverseDepthBuffer;\r\n\r\n for (let cascadeIndex = 0; cascadeIndex < this._numCascades; ++cascadeIndex) {\r\n this._computeFrustumInWorldSpace(cascadeIndex);\r\n this._computeCascadeFrustum(cascadeIndex);\r\n\r\n this._cascadeMaxExtents[cascadeIndex].subtractToRef(this._cascadeMinExtents[cascadeIndex], tmpv1); // tmpv1 = cascadeExtents\r\n\r\n // Get position of the shadow camera\r\n this._frustumCenter[cascadeIndex].addToRef(this._lightDirection.scale(this._cascadeMinExtents[cascadeIndex].z), this._shadowCameraPos[cascadeIndex]);\r\n\r\n // Come up with a new orthographic camera for the shadow caster\r\n Matrix.LookAtLHToRef(this._shadowCameraPos[cascadeIndex], this._frustumCenter[cascadeIndex], UpDir, this._viewMatrices[cascadeIndex]);\r\n\r\n let minZ = 0,\r\n maxZ = tmpv1.z;\r\n\r\n // Try to tighten minZ and maxZ based on the bounding box of the shadow casters\r\n const boundingInfo = this._shadowCastersBoundingInfo;\r\n\r\n boundingInfo.update(this._viewMatrices[cascadeIndex]);\r\n\r\n maxZ = Math.min(maxZ, boundingInfo.boundingBox.maximumWorld.z);\r\n\r\n if (!this._depthClamp || this.filter === ShadowGenerator.FILTER_PCSS) {\r\n // If we don't use depth clamping, we must set minZ so that all shadow casters are in the light frustum\r\n minZ = Math.min(minZ, boundingInfo.boundingBox.minimumWorld.z);\r\n } else {\r\n // If using depth clamping, we can adjust minZ to reduce the [minZ, maxZ] range (and get some additional precision in the shadow map)\r\n minZ = Math.max(minZ, boundingInfo.boundingBox.minimumWorld.z);\r\n }\r\n\r\n Matrix.OrthoOffCenterLHToRef(\r\n this._cascadeMinExtents[cascadeIndex].x,\r\n this._cascadeMaxExtents[cascadeIndex].x,\r\n this._cascadeMinExtents[cascadeIndex].y,\r\n this._cascadeMaxExtents[cascadeIndex].y,\r\n useReverseDepthBuffer ? maxZ : minZ,\r\n useReverseDepthBuffer ? minZ : maxZ,\r\n this._projectionMatrices[cascadeIndex],\r\n scene.getEngine().isNDCHalfZRange\r\n );\r\n\r\n this._cascadeMinExtents[cascadeIndex].z = minZ;\r\n this._cascadeMaxExtents[cascadeIndex].z = maxZ;\r\n\r\n this._viewMatrices[cascadeIndex].multiplyToRef(this._projectionMatrices[cascadeIndex], this._transformMatrices[cascadeIndex]);\r\n\r\n // Create the rounding matrix, by projecting the world-space origin and determining\r\n // the fractional offset in texel space\r\n Vector3.TransformCoordinatesToRef(ZeroVec, this._transformMatrices[cascadeIndex], tmpv1); // tmpv1 = shadowOrigin\r\n tmpv1.scaleInPlace(this._mapSize / 2);\r\n\r\n tmpv2.copyFromFloats(Math.round(tmpv1.x), Math.round(tmpv1.y), Math.round(tmpv1.z)); // tmpv2 = roundedOrigin\r\n tmpv2.subtractInPlace(tmpv1).scaleInPlace(2 / this._mapSize); // tmpv2 = roundOffset\r\n\r\n Matrix.TranslationToRef(tmpv2.x, tmpv2.y, 0.0, tmpMatrix);\r\n\r\n this._projectionMatrices[cascadeIndex].multiplyToRef(tmpMatrix, this._projectionMatrices[cascadeIndex]);\r\n this._viewMatrices[cascadeIndex].multiplyToRef(this._projectionMatrices[cascadeIndex], this._transformMatrices[cascadeIndex]);\r\n\r\n this._transformMatrices[cascadeIndex].copyToArray(this._transformMatricesAsArray, cascadeIndex * 16);\r\n }\r\n }\r\n\r\n // Get the 8 points of the view frustum in world space\r\n private _computeFrustumInWorldSpace(cascadeIndex: number): void {\r\n const camera = this._getCamera();\r\n if (!camera) {\r\n return;\r\n }\r\n\r\n const prevSplitDist = this._cascades[cascadeIndex].prevBreakDistance,\r\n splitDist = this._cascades[cascadeIndex].breakDistance;\r\n\r\n const isNDCHalfZRange = this._scene.getEngine().isNDCHalfZRange;\r\n\r\n camera.getViewMatrix(); // make sure the transformation matrix we get when calling 'getTransformationMatrix()' is calculated with an up to date view matrix\r\n\r\n const cameraInfiniteFarPlane = camera.maxZ === 0;\r\n const saveCameraMaxZ = camera.maxZ;\r\n\r\n if (cameraInfiniteFarPlane) {\r\n camera.maxZ = this._shadowMaxZ;\r\n camera.getProjectionMatrix(true);\r\n }\r\n\r\n const invViewProj = Matrix.Invert(camera.getTransformationMatrix());\r\n\r\n if (cameraInfiniteFarPlane) {\r\n camera.maxZ = saveCameraMaxZ;\r\n camera.getProjectionMatrix(true);\r\n }\r\n\r\n const cornerIndexOffset = this._scene.getEngine().useReverseDepthBuffer ? 4 : 0;\r\n for (let cornerIndex = 0; cornerIndex < CascadedShadowGenerator._FrustumCornersNDCSpace.length; ++cornerIndex) {\r\n tmpv1.copyFrom(CascadedShadowGenerator._FrustumCornersNDCSpace[(cornerIndex + cornerIndexOffset) % CascadedShadowGenerator._FrustumCornersNDCSpace.length]);\r\n if (isNDCHalfZRange && tmpv1.z === -1) {\r\n tmpv1.z = 0;\r\n }\r\n Vector3.TransformCoordinatesToRef(tmpv1, invViewProj, this._frustumCornersWorldSpace[cascadeIndex][cornerIndex]);\r\n }\r\n\r\n // Get the corners of the current cascade slice of the view frustum\r\n for (let cornerIndex = 0; cornerIndex < CascadedShadowGenerator._FrustumCornersNDCSpace.length / 2; ++cornerIndex) {\r\n tmpv1.copyFrom(this._frustumCornersWorldSpace[cascadeIndex][cornerIndex + 4]).subtractInPlace(this._frustumCornersWorldSpace[cascadeIndex][cornerIndex]);\r\n tmpv2.copyFrom(tmpv1).scaleInPlace(prevSplitDist); // near corner ray\r\n tmpv1.scaleInPlace(splitDist); // far corner ray\r\n\r\n tmpv1.addInPlace(this._frustumCornersWorldSpace[cascadeIndex][cornerIndex]);\r\n\r\n this._frustumCornersWorldSpace[cascadeIndex][cornerIndex + 4].copyFrom(tmpv1);\r\n this._frustumCornersWorldSpace[cascadeIndex][cornerIndex].addInPlace(tmpv2);\r\n }\r\n }\r\n\r\n private _computeCascadeFrustum(cascadeIndex: number): void {\r\n this._cascadeMinExtents[cascadeIndex].copyFromFloats(Number.MAX_VALUE, Number.MAX_VALUE, Number.MAX_VALUE);\r\n this._cascadeMaxExtents[cascadeIndex].copyFromFloats(Number.MIN_VALUE, Number.MIN_VALUE, Number.MIN_VALUE);\r\n this._frustumCenter[cascadeIndex].copyFromFloats(0, 0, 0);\r\n\r\n const camera = this._getCamera();\r\n\r\n if (!camera) {\r\n return;\r\n }\r\n\r\n // Calculate the centroid of the view frustum slice\r\n for (let cornerIndex = 0; cornerIndex < this._frustumCornersWorldSpace[cascadeIndex].length; ++cornerIndex) {\r\n this._frustumCenter[cascadeIndex].addInPlace(this._frustumCornersWorldSpace[cascadeIndex][cornerIndex]);\r\n }\r\n\r\n this._frustumCenter[cascadeIndex].scaleInPlace(1 / this._frustumCornersWorldSpace[cascadeIndex].length);\r\n\r\n if (this.stabilizeCascades) {\r\n // Calculate the radius of a bounding sphere surrounding the frustum corners\r\n let sphereRadius = 0;\r\n for (let cornerIndex = 0; cornerIndex < this._frustumCornersWorldSpace[cascadeIndex].length; ++cornerIndex) {\r\n const dist = this._frustumCornersWorldSpace[cascadeIndex][cornerIndex].subtractToRef(this._frustumCenter[cascadeIndex], tmpv1).length();\r\n sphereRadius = Math.max(sphereRadius, dist);\r\n }\r\n\r\n sphereRadius = Math.ceil(sphereRadius * 16) / 16;\r\n\r\n this._cascadeMaxExtents[cascadeIndex].copyFromFloats(sphereRadius, sphereRadius, sphereRadius);\r\n this._cascadeMinExtents[cascadeIndex].copyFromFloats(-sphereRadius, -sphereRadius, -sphereRadius);\r\n } else {\r\n // Create a temporary view matrix for the light\r\n const lightCameraPos = this._frustumCenter[cascadeIndex];\r\n\r\n this._frustumCenter[cascadeIndex].addToRef(this._lightDirection, tmpv1); // tmpv1 = look at\r\n\r\n Matrix.LookAtLHToRef(lightCameraPos, tmpv1, UpDir, tmpMatrix); // matrix = lightView\r\n\r\n // Calculate an AABB around the frustum corners\r\n for (let cornerIndex = 0; cornerIndex < this._frustumCornersWorldSpace[cascadeIndex].length; ++cornerIndex) {\r\n Vector3.TransformCoordinatesToRef(this._frustumCornersWorldSpace[cascadeIndex][cornerIndex], tmpMatrix, tmpv1);\r\n\r\n this._cascadeMinExtents[cascadeIndex].minimizeInPlace(tmpv1);\r\n this._cascadeMaxExtents[cascadeIndex].maximizeInPlace(tmpv1);\r\n }\r\n }\r\n }\r\n\r\n protected _recreateSceneUBOs(): void {\r\n this._disposeSceneUBOs();\r\n if (this._sceneUBOs) {\r\n for (let i = 0; i < this._numCascades; ++i) {\r\n this._sceneUBOs.push(this._scene.createSceneUniformBuffer(`Scene for CSM Shadow Generator (light \"${this._light.name}\" cascade #${i})`));\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Support test.\r\n */\r\n public static get IsSupported(): boolean {\r\n const engine = EngineStore.LastCreatedEngine;\r\n if (!engine) {\r\n return false;\r\n }\r\n return engine._features.supportCSM;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _SceneComponentInitialization: (scene: Scene) => void = (_) => {\r\n throw _WarnImport(\"ShadowGeneratorSceneComponent\");\r\n };\r\n\r\n /**\r\n * Creates a Cascaded Shadow Generator object.\r\n * A ShadowGenerator is the required tool to use the shadows.\r\n * Each directional light casting shadows needs to use its own ShadowGenerator.\r\n * Documentation : https://doc.babylonjs.com/babylon101/cascadedShadows\r\n * @param mapSize The size of the texture what stores the shadows. Example : 1024.\r\n * @param light The directional light object generating the shadows.\r\n * @param usefulFloatFirst By default the generator will try to use half float textures but if you need precision (for self shadowing for instance), you can use this option to enforce full float texture.\r\n * @param camera Camera associated with this shadow generator (default: null). If null, takes the scene active camera at the time we need to access it\r\n * @param useRedTextureType Forces the generator to use a Red instead of a RGBA type for the shadow map texture format (default: true)\r\n */\r\n constructor(mapSize: number, light: DirectionalLight, usefulFloatFirst?: boolean, camera?: Nullable, useRedTextureType = true) {\r\n if (!CascadedShadowGenerator.IsSupported) {\r\n Logger.Error(\"CascadedShadowMap is not supported by the current engine.\");\r\n return;\r\n }\r\n\r\n super(mapSize, light, usefulFloatFirst, camera, useRedTextureType);\r\n\r\n this.usePercentageCloserFiltering = true;\r\n }\r\n\r\n protected _initializeGenerator(): void {\r\n this.penumbraDarkness = this.penumbraDarkness ?? 1.0;\r\n this._numCascades = this._numCascades ?? CascadedShadowGenerator.DEFAULT_CASCADES_COUNT;\r\n this.stabilizeCascades = this.stabilizeCascades ?? false;\r\n this._freezeShadowCastersBoundingInfoObservable = this._freezeShadowCastersBoundingInfoObservable ?? null;\r\n this.freezeShadowCastersBoundingInfo = this.freezeShadowCastersBoundingInfo ?? false;\r\n this._scbiMin = this._scbiMin ?? new Vector3(0, 0, 0);\r\n this._scbiMax = this._scbiMax ?? new Vector3(0, 0, 0);\r\n this._shadowCastersBoundingInfo = this._shadowCastersBoundingInfo ?? new BoundingInfo(new Vector3(0, 0, 0), new Vector3(0, 0, 0));\r\n this._breaksAreDirty = this._breaksAreDirty ?? true;\r\n this._minDistance = this._minDistance ?? 0;\r\n this._maxDistance = this._maxDistance ?? 1;\r\n this._currentLayer = this._currentLayer ?? 0;\r\n this._shadowMaxZ = this._shadowMaxZ ?? this._getCamera()?.maxZ ?? 10000;\r\n this._debug = this._debug ?? false;\r\n this._depthClamp = this._depthClamp ?? true;\r\n this._cascadeBlendPercentage = this._cascadeBlendPercentage ?? 0.1;\r\n this._lambda = this._lambda ?? 0.5;\r\n this._autoCalcDepthBounds = this._autoCalcDepthBounds ?? false;\r\n\r\n this._recreateSceneUBOs();\r\n\r\n super._initializeGenerator();\r\n }\r\n\r\n protected _createTargetRenderTexture(): void {\r\n const engine = this._scene.getEngine();\r\n const size = { width: this._mapSize, height: this._mapSize, layers: this.numCascades };\r\n this._shadowMap = new RenderTargetTexture(\r\n this._light.name + \"_CSMShadowMap\",\r\n size,\r\n this._scene,\r\n false,\r\n true,\r\n this._textureType,\r\n false,\r\n undefined,\r\n false,\r\n false,\r\n undefined,\r\n this._useRedTextureType ? Constants.TEXTUREFORMAT_RED : Constants.TEXTUREFORMAT_RGBA\r\n );\r\n this._shadowMap.createDepthStencilTexture(engine.useReverseDepthBuffer ? Constants.GREATER : Constants.LESS, true);\r\n this._shadowMap.noPrePassRenderer = true;\r\n }\r\n\r\n protected _initializeShadowMap(): void {\r\n super._initializeShadowMap();\r\n\r\n if (this._shadowMap === null) {\r\n return;\r\n }\r\n\r\n this._transformMatricesAsArray = new Float32Array(this._numCascades * 16);\r\n this._viewSpaceFrustumsZ = new Array(this._numCascades);\r\n this._frustumLengths = new Array(this._numCascades);\r\n this._lightSizeUVCorrection = new Array(this._numCascades * 2);\r\n this._depthCorrection = new Array(this._numCascades);\r\n\r\n this._cascades = [];\r\n this._viewMatrices = [];\r\n this._projectionMatrices = [];\r\n this._transformMatrices = [];\r\n this._cascadeMinExtents = [];\r\n this._cascadeMaxExtents = [];\r\n this._frustumCenter = [];\r\n this._shadowCameraPos = [];\r\n this._frustumCornersWorldSpace = [];\r\n\r\n for (let cascadeIndex = 0; cascadeIndex < this._numCascades; ++cascadeIndex) {\r\n this._cascades[cascadeIndex] = {\r\n prevBreakDistance: 0,\r\n breakDistance: 0,\r\n };\r\n\r\n this._viewMatrices[cascadeIndex] = Matrix.Zero();\r\n this._projectionMatrices[cascadeIndex] = Matrix.Zero();\r\n this._transformMatrices[cascadeIndex] = Matrix.Zero();\r\n this._cascadeMinExtents[cascadeIndex] = new Vector3();\r\n this._cascadeMaxExtents[cascadeIndex] = new Vector3();\r\n this._frustumCenter[cascadeIndex] = new Vector3();\r\n this._shadowCameraPos[cascadeIndex] = new Vector3();\r\n this._frustumCornersWorldSpace[cascadeIndex] = new Array(CascadedShadowGenerator._FrustumCornersNDCSpace.length);\r\n\r\n for (let i = 0; i < CascadedShadowGenerator._FrustumCornersNDCSpace.length; ++i) {\r\n this._frustumCornersWorldSpace[cascadeIndex][i] = new Vector3();\r\n }\r\n }\r\n\r\n const engine = this._scene.getEngine();\r\n\r\n this._shadowMap.onBeforeBindObservable.clear();\r\n this._shadowMap.onBeforeRenderObservable.clear();\r\n\r\n this._shadowMap.onBeforeRenderObservable.add((layer: number) => {\r\n if (this._sceneUBOs) {\r\n this._scene.setSceneUniformBuffer(this._sceneUBOs[layer]);\r\n }\r\n this._currentLayer = layer;\r\n if (this._filter === ShadowGenerator.FILTER_PCF) {\r\n engine.setColorWrite(false);\r\n }\r\n this._scene.setTransformMatrix(this.getCascadeViewMatrix(layer)!, this.getCascadeProjectionMatrix(layer)!);\r\n if (this._useUBO) {\r\n this._scene.getSceneUniformBuffer().unbindEffect();\r\n this._scene.finalizeSceneUbo();\r\n }\r\n });\r\n\r\n this._shadowMap.onBeforeBindObservable.add(() => {\r\n this._currentSceneUBO = this._scene.getSceneUniformBuffer();\r\n engine._debugPushGroup?.(`cascaded shadow map generation for pass id ${engine.currentRenderPassId}`, 1);\r\n if (this._breaksAreDirty) {\r\n this._splitFrustum();\r\n }\r\n this._computeMatrices();\r\n });\r\n\r\n this._splitFrustum();\r\n }\r\n\r\n protected _bindCustomEffectForRenderSubMeshForShadowMap(subMesh: SubMesh, effect: Effect): void {\r\n effect.setMatrix(\"viewProjection\", this.getCascadeTransformMatrix(this._currentLayer)!);\r\n }\r\n\r\n protected _isReadyCustomDefines(defines: any): void {\r\n defines.push(\"#define SM_DEPTHCLAMP \" + (this._depthClamp && this._filter !== ShadowGenerator.FILTER_PCSS ? \"1\" : \"0\"));\r\n }\r\n\r\n /**\r\n * Prepare all the defines in a material relying on a shadow map at the specified light index.\r\n * @param defines Defines of the material we want to update\r\n * @param lightIndex Index of the light in the enabled light list of the material\r\n */\r\n public prepareDefines(defines: any, lightIndex: number): void {\r\n super.prepareDefines(defines, lightIndex);\r\n\r\n const scene = this._scene;\r\n const light = this._light;\r\n\r\n if (!scene.shadowsEnabled || !light.shadowEnabled) {\r\n return;\r\n }\r\n\r\n defines[\"SHADOWCSM\" + lightIndex] = true;\r\n defines[\"SHADOWCSMDEBUG\" + lightIndex] = this.debug;\r\n defines[\"SHADOWCSMNUM_CASCADES\" + lightIndex] = this.numCascades;\r\n defines[\"SHADOWCSM_RIGHTHANDED\" + lightIndex] = scene.useRightHandedSystem;\r\n\r\n const camera = this._getCamera();\r\n\r\n if (camera && this._shadowMaxZ <= (camera.maxZ || this._shadowMaxZ)) {\r\n defines[\"SHADOWCSMUSESHADOWMAXZ\" + lightIndex] = true;\r\n }\r\n\r\n if (this.cascadeBlendPercentage === 0) {\r\n defines[\"SHADOWCSMNOBLEND\" + lightIndex] = true;\r\n }\r\n }\r\n\r\n /**\r\n * Binds the shadow related information inside of an effect (information like near, far, darkness...\r\n * defined in the generator but impacting the effect).\r\n * @param lightIndex Index of the light in the enabled light list of the material owning the effect\r\n * @param effect The effect we are binfing the information for\r\n */\r\n public bindShadowLight(lightIndex: string, effect: Effect): void {\r\n const light = this._light;\r\n const scene = this._scene;\r\n\r\n if (!scene.shadowsEnabled || !light.shadowEnabled) {\r\n return;\r\n }\r\n\r\n const camera = this._getCamera();\r\n if (!camera) {\r\n return;\r\n }\r\n\r\n const shadowMap = this.getShadowMap();\r\n if (!shadowMap) {\r\n return;\r\n }\r\n\r\n const width = shadowMap.getSize().width;\r\n\r\n effect.setMatrices(\"lightMatrix\" + lightIndex, this._transformMatricesAsArray);\r\n effect.setArray(\"viewFrustumZ\" + lightIndex, this._viewSpaceFrustumsZ);\r\n effect.setFloat(\"cascadeBlendFactor\" + lightIndex, this.cascadeBlendPercentage === 0 ? 10000 : 1 / this.cascadeBlendPercentage);\r\n effect.setArray(\"frustumLengths\" + lightIndex, this._frustumLengths);\r\n\r\n // Only PCF uses depth stencil texture.\r\n if (this._filter === ShadowGenerator.FILTER_PCF) {\r\n effect.setDepthStencilTexture(\"shadowSampler\" + lightIndex, shadowMap);\r\n light._uniformBuffer.updateFloat4(\"shadowsInfo\", this.getDarkness(), width, 1 / width, this.frustumEdgeFalloff, lightIndex);\r\n } else if (this._filter === ShadowGenerator.FILTER_PCSS) {\r\n for (let cascadeIndex = 0; cascadeIndex < this._numCascades; ++cascadeIndex) {\r\n this._lightSizeUVCorrection[cascadeIndex * 2 + 0] =\r\n cascadeIndex === 0\r\n ? 1\r\n : (this._cascadeMaxExtents[0].x - this._cascadeMinExtents[0].x) / (this._cascadeMaxExtents[cascadeIndex].x - this._cascadeMinExtents[cascadeIndex].x); // x correction\r\n this._lightSizeUVCorrection[cascadeIndex * 2 + 1] =\r\n cascadeIndex === 0\r\n ? 1\r\n : (this._cascadeMaxExtents[0].y - this._cascadeMinExtents[0].y) / (this._cascadeMaxExtents[cascadeIndex].y - this._cascadeMinExtents[cascadeIndex].y); // y correction\r\n this._depthCorrection[cascadeIndex] =\r\n cascadeIndex === 0\r\n ? 1\r\n : (this._cascadeMaxExtents[cascadeIndex].z - this._cascadeMinExtents[cascadeIndex].z) / (this._cascadeMaxExtents[0].z - this._cascadeMinExtents[0].z);\r\n }\r\n effect.setDepthStencilTexture(\"shadowSampler\" + lightIndex, shadowMap);\r\n effect.setTexture(\"depthSampler\" + lightIndex, shadowMap);\r\n effect.setArray2(\"lightSizeUVCorrection\" + lightIndex, this._lightSizeUVCorrection);\r\n effect.setArray(\"depthCorrection\" + lightIndex, this._depthCorrection);\r\n effect.setFloat(\"penumbraDarkness\" + lightIndex, this.penumbraDarkness);\r\n light._uniformBuffer.updateFloat4(\"shadowsInfo\", this.getDarkness(), 1 / width, this._contactHardeningLightSizeUVRatio * width, this.frustumEdgeFalloff, lightIndex);\r\n } else {\r\n effect.setTexture(\"shadowSampler\" + lightIndex, shadowMap);\r\n light._uniformBuffer.updateFloat4(\"shadowsInfo\", this.getDarkness(), width, 1 / width, this.frustumEdgeFalloff, lightIndex);\r\n }\r\n\r\n light._uniformBuffer.updateFloat2(\r\n \"depthValues\",\r\n this.getLight().getDepthMinZ(camera),\r\n this.getLight().getDepthMinZ(camera) + this.getLight().getDepthMaxZ(camera),\r\n lightIndex\r\n );\r\n }\r\n\r\n /**\r\n * Gets the transformation matrix of the first cascade used to project the meshes into the map from the light point of view.\r\n * (eq to view projection * shadow projection matrices)\r\n * @returns The transform matrix used to create the shadow map\r\n */\r\n public getTransformMatrix(): Matrix {\r\n return this.getCascadeTransformMatrix(0)!;\r\n }\r\n\r\n /**\r\n * Disposes the ShadowGenerator.\r\n * Returns nothing.\r\n */\r\n public dispose(): void {\r\n super.dispose();\r\n\r\n if (this._freezeShadowCastersBoundingInfoObservable) {\r\n this._scene.onBeforeRenderObservable.remove(this._freezeShadowCastersBoundingInfoObservable);\r\n this._freezeShadowCastersBoundingInfoObservable = null;\r\n }\r\n\r\n if (this._depthReducer) {\r\n this._depthReducer.dispose();\r\n this._depthReducer = null;\r\n }\r\n }\r\n\r\n /**\r\n * Serializes the shadow generator setup to a json object.\r\n * @returns The serialized JSON object\r\n */\r\n public serialize(): any {\r\n const serializationObject: any = super.serialize();\r\n const shadowMap = this.getShadowMap();\r\n\r\n if (!shadowMap) {\r\n return serializationObject;\r\n }\r\n\r\n serializationObject.numCascades = this._numCascades;\r\n serializationObject.debug = this._debug;\r\n serializationObject.stabilizeCascades = this.stabilizeCascades;\r\n serializationObject.lambda = this._lambda;\r\n serializationObject.cascadeBlendPercentage = this.cascadeBlendPercentage;\r\n serializationObject.depthClamp = this._depthClamp;\r\n serializationObject.autoCalcDepthBounds = this.autoCalcDepthBounds;\r\n serializationObject.shadowMaxZ = this._shadowMaxZ;\r\n serializationObject.penumbraDarkness = this.penumbraDarkness;\r\n\r\n serializationObject.freezeShadowCastersBoundingInfo = this._freezeShadowCastersBoundingInfo;\r\n serializationObject.minDistance = this.minDistance;\r\n serializationObject.maxDistance = this.maxDistance;\r\n\r\n serializationObject.renderList = [];\r\n if (shadowMap.renderList) {\r\n for (let meshIndex = 0; meshIndex < shadowMap.renderList.length; meshIndex++) {\r\n const mesh = shadowMap.renderList[meshIndex];\r\n\r\n serializationObject.renderList.push(mesh.id);\r\n }\r\n }\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Parses a serialized ShadowGenerator and returns a new ShadowGenerator.\r\n * @param parsedShadowGenerator The JSON object to parse\r\n * @param scene The scene to create the shadow map for\r\n * @returns The parsed shadow generator\r\n */\r\n public static Parse(parsedShadowGenerator: any, scene: Scene): ShadowGenerator {\r\n const shadowGenerator = ShadowGenerator.Parse(\r\n parsedShadowGenerator,\r\n scene,\r\n (mapSize: number, light: IShadowLight, camera: Nullable) => new CascadedShadowGenerator(mapSize, light, undefined, camera)\r\n ) as CascadedShadowGenerator;\r\n\r\n if (parsedShadowGenerator.numCascades !== undefined) {\r\n shadowGenerator.numCascades = parsedShadowGenerator.numCascades;\r\n }\r\n\r\n if (parsedShadowGenerator.debug !== undefined) {\r\n shadowGenerator.debug = parsedShadowGenerator.debug;\r\n }\r\n\r\n if (parsedShadowGenerator.stabilizeCascades !== undefined) {\r\n shadowGenerator.stabilizeCascades = parsedShadowGenerator.stabilizeCascades;\r\n }\r\n\r\n if (parsedShadowGenerator.lambda !== undefined) {\r\n shadowGenerator.lambda = parsedShadowGenerator.lambda;\r\n }\r\n\r\n if (parsedShadowGenerator.cascadeBlendPercentage !== undefined) {\r\n shadowGenerator.cascadeBlendPercentage = parsedShadowGenerator.cascadeBlendPercentage;\r\n }\r\n\r\n if (parsedShadowGenerator.depthClamp !== undefined) {\r\n shadowGenerator.depthClamp = parsedShadowGenerator.depthClamp;\r\n }\r\n\r\n if (parsedShadowGenerator.autoCalcDepthBounds !== undefined) {\r\n shadowGenerator.autoCalcDepthBounds = parsedShadowGenerator.autoCalcDepthBounds;\r\n }\r\n\r\n if (parsedShadowGenerator.shadowMaxZ !== undefined) {\r\n shadowGenerator.shadowMaxZ = parsedShadowGenerator.shadowMaxZ;\r\n }\r\n\r\n if (parsedShadowGenerator.penumbraDarkness !== undefined) {\r\n shadowGenerator.penumbraDarkness = parsedShadowGenerator.penumbraDarkness;\r\n }\r\n\r\n if (parsedShadowGenerator.freezeShadowCastersBoundingInfo !== undefined) {\r\n shadowGenerator.freezeShadowCastersBoundingInfo = parsedShadowGenerator.freezeShadowCastersBoundingInfo;\r\n }\r\n\r\n if (parsedShadowGenerator.minDistance !== undefined && parsedShadowGenerator.maxDistance !== undefined) {\r\n shadowGenerator.setMinMaxDistance(parsedShadowGenerator.minDistance, parsedShadowGenerator.maxDistance);\r\n }\r\n\r\n return shadowGenerator;\r\n }\r\n}\r\n","import type { SmartArrayNoDuplicate } from \"../../Misc/smartArray\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { RenderTargetTexture } from \"../../Materials/Textures/renderTargetTexture\";\r\nimport { ShadowGenerator } from \"./shadowGenerator\";\r\nimport { CascadedShadowGenerator } from \"./cascadedShadowGenerator\";\r\nimport type { ISceneSerializableComponent } from \"../../sceneComponent\";\r\nimport { SceneComponentConstants } from \"../../sceneComponent\";\r\nimport { AbstractScene } from \"../../abstractScene\";\r\n// Adds the parser to the scene parsers.\r\nAbstractScene.AddParser(SceneComponentConstants.NAME_SHADOWGENERATOR, (parsedData: any, scene: Scene) => {\r\n // Shadows\r\n if (parsedData.shadowGenerators !== undefined && parsedData.shadowGenerators !== null) {\r\n for (let index = 0, cache = parsedData.shadowGenerators.length; index < cache; index++) {\r\n const parsedShadowGenerator = parsedData.shadowGenerators[index];\r\n if (parsedShadowGenerator.className === CascadedShadowGenerator.CLASSNAME) {\r\n CascadedShadowGenerator.Parse(parsedShadowGenerator, scene);\r\n } else {\r\n ShadowGenerator.Parse(parsedShadowGenerator, scene);\r\n }\r\n // SG would be available on their associated lights\r\n }\r\n }\r\n});\r\n\r\n/**\r\n * Defines the shadow generator component responsible to manage any shadow generators\r\n * in a given scene.\r\n */\r\nexport class ShadowGeneratorSceneComponent implements ISceneSerializableComponent {\r\n /**\r\n * The component name helpful to identify the component in the list of scene components.\r\n */\r\n public readonly name = SceneComponentConstants.NAME_SHADOWGENERATOR;\r\n\r\n /**\r\n * The scene the component belongs to.\r\n */\r\n public scene: Scene;\r\n\r\n /**\r\n * Creates a new instance of the component for the given scene\r\n * @param scene Defines the scene to register the component in\r\n */\r\n constructor(scene: Scene) {\r\n this.scene = scene;\r\n }\r\n\r\n /**\r\n * Registers the component in a given scene\r\n */\r\n public register(): void {\r\n this.scene._gatherRenderTargetsStage.registerStep(SceneComponentConstants.STEP_GATHERRENDERTARGETS_SHADOWGENERATOR, this, this._gatherRenderTargets);\r\n }\r\n\r\n /**\r\n * Rebuilds the elements related to this component in case of\r\n * context lost for instance.\r\n */\r\n public rebuild(): void {\r\n // Nothing To Do Here.\r\n }\r\n\r\n /**\r\n * Serializes the component data to the specified json object\r\n * @param serializationObject The object to serialize to\r\n */\r\n public serialize(serializationObject: any): void {\r\n // Shadows\r\n serializationObject.shadowGenerators = [];\r\n const lights = this.scene.lights;\r\n for (const light of lights) {\r\n const shadowGenerators = light.getShadowGenerators();\r\n if (shadowGenerators) {\r\n const iterator = shadowGenerators.values();\r\n for (let key = iterator.next(); key.done !== true; key = iterator.next()) {\r\n const shadowGenerator = key.value;\r\n serializationObject.shadowGenerators.push(shadowGenerator.serialize());\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Adds all the elements from the container to the scene\r\n * @param container the container holding the elements\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public addFromContainer(container: AbstractScene): void {\r\n // Nothing To Do Here. (directly attached to a light)\r\n }\r\n\r\n /**\r\n * Removes all the elements in the container from the scene\r\n * @param container contains the elements to remove\r\n * @param dispose if the removed element should be disposed (default: false)\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public removeFromContainer(container: AbstractScene, dispose?: boolean): void {\r\n // Nothing To Do Here. (directly attached to a light)\r\n }\r\n\r\n /**\r\n * Rebuilds the elements related to this component in case of\r\n * context lost for instance.\r\n */\r\n public dispose(): void {\r\n // Nothing To Do Here.\r\n }\r\n\r\n private _gatherRenderTargets(renderTargets: SmartArrayNoDuplicate): void {\r\n // Shadows\r\n const scene = this.scene;\r\n if (this.scene.shadowsEnabled) {\r\n for (let lightIndex = 0; lightIndex < scene.lights.length; lightIndex++) {\r\n const light = scene.lights[lightIndex];\r\n const shadowGenerators = light.getShadowGenerators();\r\n\r\n if (light.isEnabled() && light.shadowEnabled && shadowGenerators) {\r\n const iterator = shadowGenerators.values();\r\n for (let key = iterator.next(); key.done !== true; key = iterator.next()) {\r\n const shadowGenerator = key.value;\r\n const shadowMap = shadowGenerator.getShadowMap();\r\n if (scene.textures.indexOf(shadowMap) !== -1) {\r\n renderTargets.push(shadowMap);\r\n }\r\n }\r\n }\r\n }\r\n }\r\n }\r\n}\r\n\r\nShadowGenerator._SceneComponentInitialization = (scene: Scene) => {\r\n let component = scene._getComponent(SceneComponentConstants.NAME_SHADOWGENERATOR);\r\n if (!component) {\r\n component = new ShadowGeneratorSceneComponent(scene);\r\n scene._addComponent(component);\r\n }\r\n};\r\n","import { serialize } from \"../Misc/decorators\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Matrix, Vector3 } from \"../Maths/math.vector\";\r\nimport { Node } from \"../node\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport { Light } from \"./light\";\r\nimport { ShadowLight } from \"./shadowLight\";\r\nimport type { Effect } from \"../Materials/effect\";\r\nNode.AddNodeConstructor(\"Light_Type_1\", (name, scene) => {\r\n return () => new DirectionalLight(name, Vector3.Zero(), scene);\r\n});\r\n\r\n/**\r\n * A directional light is defined by a direction (what a surprise!).\r\n * The light is emitted from everywhere in the specified direction, and has an infinite range.\r\n * An example of a directional light is when a distance planet is lit by the apparently parallel lines of light from its sun. Light in a downward direction will light the top of an object.\r\n * Documentation: https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction\r\n */\r\nexport class DirectionalLight extends ShadowLight {\r\n private _shadowFrustumSize = 0;\r\n /**\r\n * Fix frustum size for the shadow generation. This is disabled if the value is 0.\r\n */\r\n @serialize()\r\n public get shadowFrustumSize(): number {\r\n return this._shadowFrustumSize;\r\n }\r\n /**\r\n * Specifies a fix frustum size for the shadow generation.\r\n */\r\n public set shadowFrustumSize(value: number) {\r\n this._shadowFrustumSize = value;\r\n this.forceProjectionMatrixCompute();\r\n }\r\n\r\n private _shadowOrthoScale = 0.1;\r\n /**\r\n * Gets the shadow projection scale against the optimal computed one.\r\n * 0.1 by default which means that the projection window is increase by 10% from the optimal size.\r\n * This does not impact in fixed frustum size (shadowFrustumSize being set)\r\n */\r\n @serialize()\r\n public get shadowOrthoScale(): number {\r\n return this._shadowOrthoScale;\r\n }\r\n /**\r\n * Sets the shadow projection scale against the optimal computed one.\r\n * 0.1 by default which means that the projection window is increase by 10% from the optimal size.\r\n * This does not impact in fixed frustum size (shadowFrustumSize being set)\r\n */\r\n public set shadowOrthoScale(value: number) {\r\n this._shadowOrthoScale = value;\r\n this.forceProjectionMatrixCompute();\r\n }\r\n\r\n /**\r\n * Automatically compute the projection matrix to best fit (including all the casters)\r\n * on each frame.\r\n */\r\n @serialize()\r\n public autoUpdateExtends = true;\r\n\r\n /**\r\n * Automatically compute the shadowMinZ and shadowMaxZ for the projection matrix to best fit (including all the casters)\r\n * on each frame. autoUpdateExtends must be set to true for this to work\r\n */\r\n @serialize()\r\n public autoCalcShadowZBounds = false;\r\n\r\n // Cache\r\n @serialize(\"orthoLeft\")\r\n private _orthoLeft = Number.MAX_VALUE;\r\n @serialize(\"orthoRight\")\r\n private _orthoRight = Number.MIN_VALUE;\r\n @serialize(\"orthoTop\")\r\n private _orthoTop = Number.MIN_VALUE;\r\n @serialize(\"orthoBottom\")\r\n private _orthoBottom = Number.MAX_VALUE;\r\n\r\n /**\r\n * Gets or sets the orthoLeft property used to build the light frustum\r\n */\r\n public get orthoLeft(): number {\r\n return this._orthoLeft;\r\n }\r\n\r\n public set orthoLeft(left: number) {\r\n this._orthoLeft = left;\r\n }\r\n\r\n /**\r\n * Gets or sets the orthoRight property used to build the light frustum\r\n */\r\n public get orthoRight(): number {\r\n return this._orthoRight;\r\n }\r\n\r\n public set orthoRight(right: number) {\r\n this._orthoRight = right;\r\n }\r\n\r\n /**\r\n * Gets or sets the orthoTop property used to build the light frustum\r\n */\r\n public get orthoTop(): number {\r\n return this._orthoTop;\r\n }\r\n\r\n public set orthoTop(top: number) {\r\n this._orthoTop = top;\r\n }\r\n\r\n /**\r\n * Gets or sets the orthoBottom property used to build the light frustum\r\n */\r\n public get orthoBottom(): number {\r\n return this._orthoBottom;\r\n }\r\n\r\n public set orthoBottom(bottom: number) {\r\n this._orthoBottom = bottom;\r\n }\r\n\r\n /**\r\n * Creates a DirectionalLight object in the scene, oriented towards the passed direction (Vector3).\r\n * The directional light is emitted from everywhere in the given direction.\r\n * It can cast shadows.\r\n * Documentation : https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction\r\n * @param name The friendly name of the light\r\n * @param direction The direction of the light\r\n * @param scene The scene the light belongs to\r\n */\r\n constructor(name: string, direction: Vector3, scene?: Scene) {\r\n super(name, scene);\r\n this.position = direction.scale(-1.0);\r\n this.direction = direction;\r\n }\r\n\r\n /**\r\n * Returns the string \"DirectionalLight\".\r\n * @returns The class name\r\n */\r\n public getClassName(): string {\r\n return \"DirectionalLight\";\r\n }\r\n\r\n /**\r\n * Returns the integer 1.\r\n * @returns The light Type id as a constant defines in Light.LIGHTTYPEID_x\r\n */\r\n public getTypeID(): number {\r\n return Light.LIGHTTYPEID_DIRECTIONALLIGHT;\r\n }\r\n\r\n /**\r\n * Sets the passed matrix \"matrix\" as projection matrix for the shadows cast by the light according to the passed view matrix.\r\n * Returns the DirectionalLight Shadow projection matrix.\r\n * @param matrix\r\n * @param viewMatrix\r\n * @param renderList\r\n */\r\n protected _setDefaultShadowProjectionMatrix(matrix: Matrix, viewMatrix: Matrix, renderList: Array): void {\r\n if (this.shadowFrustumSize > 0) {\r\n this._setDefaultFixedFrustumShadowProjectionMatrix(matrix);\r\n } else {\r\n this._setDefaultAutoExtendShadowProjectionMatrix(matrix, viewMatrix, renderList);\r\n }\r\n }\r\n\r\n /**\r\n * Sets the passed matrix \"matrix\" as fixed frustum projection matrix for the shadows cast by the light according to the passed view matrix.\r\n * Returns the DirectionalLight Shadow projection matrix.\r\n * @param matrix\r\n */\r\n protected _setDefaultFixedFrustumShadowProjectionMatrix(matrix: Matrix): void {\r\n const activeCamera = this.getScene().activeCamera;\r\n\r\n if (!activeCamera) {\r\n return;\r\n }\r\n\r\n Matrix.OrthoLHToRef(\r\n this.shadowFrustumSize,\r\n this.shadowFrustumSize,\r\n this.shadowMinZ !== undefined ? this.shadowMinZ : activeCamera.minZ,\r\n this.shadowMaxZ !== undefined ? this.shadowMaxZ : activeCamera.maxZ,\r\n matrix,\r\n this.getScene().getEngine().isNDCHalfZRange\r\n );\r\n }\r\n\r\n /**\r\n * Sets the passed matrix \"matrix\" as auto extend projection matrix for the shadows cast by the light according to the passed view matrix.\r\n * Returns the DirectionalLight Shadow projection matrix.\r\n * @param matrix\r\n * @param viewMatrix\r\n * @param renderList\r\n */\r\n protected _setDefaultAutoExtendShadowProjectionMatrix(matrix: Matrix, viewMatrix: Matrix, renderList: Array): void {\r\n const activeCamera = this.getScene().activeCamera;\r\n\r\n if (!activeCamera) {\r\n return;\r\n }\r\n\r\n // Check extends\r\n if (this.autoUpdateExtends || this._orthoLeft === Number.MAX_VALUE) {\r\n const tempVector3 = Vector3.Zero();\r\n\r\n this._orthoLeft = Number.MAX_VALUE;\r\n this._orthoRight = Number.MIN_VALUE;\r\n this._orthoTop = Number.MIN_VALUE;\r\n this._orthoBottom = Number.MAX_VALUE;\r\n\r\n let shadowMinZ = Number.MAX_VALUE;\r\n let shadowMaxZ = Number.MIN_VALUE;\r\n\r\n for (let meshIndex = 0; meshIndex < renderList.length; meshIndex++) {\r\n const mesh = renderList[meshIndex];\r\n\r\n if (!mesh) {\r\n continue;\r\n }\r\n\r\n const boundingInfo = mesh.getBoundingInfo();\r\n const boundingBox = boundingInfo.boundingBox;\r\n\r\n for (let index = 0; index < boundingBox.vectorsWorld.length; index++) {\r\n Vector3.TransformCoordinatesToRef(boundingBox.vectorsWorld[index], viewMatrix, tempVector3);\r\n\r\n if (tempVector3.x < this._orthoLeft) {\r\n this._orthoLeft = tempVector3.x;\r\n }\r\n if (tempVector3.y < this._orthoBottom) {\r\n this._orthoBottom = tempVector3.y;\r\n }\r\n\r\n if (tempVector3.x > this._orthoRight) {\r\n this._orthoRight = tempVector3.x;\r\n }\r\n if (tempVector3.y > this._orthoTop) {\r\n this._orthoTop = tempVector3.y;\r\n }\r\n if (this.autoCalcShadowZBounds) {\r\n if (tempVector3.z < shadowMinZ) {\r\n shadowMinZ = tempVector3.z;\r\n }\r\n if (tempVector3.z > shadowMaxZ) {\r\n shadowMaxZ = tempVector3.z;\r\n }\r\n }\r\n }\r\n }\r\n\r\n if (this.autoCalcShadowZBounds) {\r\n this._shadowMinZ = shadowMinZ;\r\n this._shadowMaxZ = shadowMaxZ;\r\n }\r\n }\r\n\r\n const xOffset = this._orthoRight - this._orthoLeft;\r\n const yOffset = this._orthoTop - this._orthoBottom;\r\n\r\n const minZ = this.shadowMinZ !== undefined ? this.shadowMinZ : activeCamera.minZ;\r\n const maxZ = this.shadowMaxZ !== undefined ? this.shadowMaxZ : activeCamera.maxZ;\r\n\r\n const useReverseDepthBuffer = this.getScene().getEngine().useReverseDepthBuffer;\r\n\r\n Matrix.OrthoOffCenterLHToRef(\r\n this._orthoLeft - xOffset * this.shadowOrthoScale,\r\n this._orthoRight + xOffset * this.shadowOrthoScale,\r\n this._orthoBottom - yOffset * this.shadowOrthoScale,\r\n this._orthoTop + yOffset * this.shadowOrthoScale,\r\n useReverseDepthBuffer ? maxZ : minZ,\r\n useReverseDepthBuffer ? minZ : maxZ,\r\n matrix,\r\n this.getScene().getEngine().isNDCHalfZRange\r\n );\r\n }\r\n\r\n protected _buildUniformLayout(): void {\r\n this._uniformBuffer.addUniform(\"vLightData\", 4);\r\n this._uniformBuffer.addUniform(\"vLightDiffuse\", 4);\r\n this._uniformBuffer.addUniform(\"vLightSpecular\", 4);\r\n this._uniformBuffer.addUniform(\"shadowsInfo\", 3);\r\n this._uniformBuffer.addUniform(\"depthValues\", 2);\r\n this._uniformBuffer.create();\r\n }\r\n\r\n /**\r\n * Sets the passed Effect object with the DirectionalLight transformed position (or position if not parented) and the passed name.\r\n * @param effect The effect to update\r\n * @param lightIndex The index of the light in the effect to update\r\n * @returns The directional light\r\n */\r\n public transferToEffect(effect: Effect, lightIndex: string): DirectionalLight {\r\n if (this.computeTransformedInformation()) {\r\n this._uniformBuffer.updateFloat4(\"vLightData\", this.transformedDirection.x, this.transformedDirection.y, this.transformedDirection.z, 1, lightIndex);\r\n return this;\r\n }\r\n this._uniformBuffer.updateFloat4(\"vLightData\", this.direction.x, this.direction.y, this.direction.z, 1, lightIndex);\r\n return this;\r\n }\r\n\r\n public transferToNodeMaterialEffect(effect: Effect, lightDataUniformName: string): Light {\r\n if (this.computeTransformedInformation()) {\r\n effect.setFloat3(lightDataUniformName, this.transformedDirection.x, this.transformedDirection.y, this.transformedDirection.z);\r\n return this;\r\n }\r\n\r\n effect.setFloat3(lightDataUniformName, this.direction.x, this.direction.y, this.direction.z);\r\n return this;\r\n }\r\n\r\n /**\r\n * Gets the minZ used for shadow according to both the scene and the light.\r\n *\r\n * Values are fixed on directional lights as it relies on an ortho projection hence the need to convert being\r\n * -1 and 1 to 0 and 1 doing (depth + min) / (min + max) -> (depth + 1) / (1 + 1) -> (depth * 0.5) + 0.5.\r\n * (when not using reverse depth buffer / NDC half Z range)\r\n * @param activeCamera The camera we are returning the min for\r\n * @returns the depth min z\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public getDepthMinZ(activeCamera: Camera): number {\r\n const engine = this._scene.getEngine();\r\n return !engine.useReverseDepthBuffer && engine.isNDCHalfZRange ? 0 : 1;\r\n }\r\n\r\n /**\r\n * Gets the maxZ used for shadow according to both the scene and the light.\r\n *\r\n * Values are fixed on directional lights as it relies on an ortho projection hence the need to convert being\r\n * -1 and 1 to 0 and 1 doing (depth + min) / (min + max) -> (depth + 1) / (1 + 1) -> (depth * 0.5) + 0.5.\r\n * (when not using reverse depth buffer / NDC half Z range)\r\n * @param activeCamera The camera we are returning the max for\r\n * @returns the depth max z\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public getDepthMaxZ(activeCamera: Camera): number {\r\n const engine = this._scene.getEngine();\r\n return engine.useReverseDepthBuffer && engine.isNDCHalfZRange ? 0 : 1;\r\n }\r\n\r\n /**\r\n * Prepares the list of defines specific to the light type.\r\n * @param defines the list of defines\r\n * @param lightIndex defines the index of the light for the effect\r\n */\r\n public prepareLightSpecificDefines(defines: any, lightIndex: number): void {\r\n defines[\"DIRLIGHT\" + lightIndex] = true;\r\n }\r\n}\r\n","import { serialize } from \"../Misc/decorators\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Matrix, Vector3 } from \"../Maths/math.vector\";\r\nimport { Node } from \"../node\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport { Light } from \"./light\";\r\nimport { ShadowLight } from \"./shadowLight\";\r\nimport type { Effect } from \"../Materials/effect\";\r\n\r\nNode.AddNodeConstructor(\"Light_Type_0\", (name, scene) => {\r\n return () => new PointLight(name, Vector3.Zero(), scene);\r\n});\r\n\r\n/**\r\n * A point light is a light defined by an unique point in world space.\r\n * The light is emitted in every direction from this point.\r\n * A good example of a point light is a standard light bulb.\r\n * Documentation: https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction\r\n */\r\nexport class PointLight extends ShadowLight {\r\n private _shadowAngle = Math.PI / 2;\r\n /**\r\n * Getter: In case of direction provided, the shadow will not use a cube texture but simulate a spot shadow as a fallback\r\n * This specifies what angle the shadow will use to be created.\r\n *\r\n * It default to 90 degrees to work nicely with the cube texture generation for point lights shadow maps.\r\n */\r\n @serialize()\r\n public get shadowAngle(): number {\r\n return this._shadowAngle;\r\n }\r\n /**\r\n * Setter: In case of direction provided, the shadow will not use a cube texture but simulate a spot shadow as a fallback\r\n * This specifies what angle the shadow will use to be created.\r\n *\r\n * It default to 90 degrees to work nicely with the cube texture generation for point lights shadow maps.\r\n */\r\n public set shadowAngle(value: number) {\r\n this._shadowAngle = value;\r\n this.forceProjectionMatrixCompute();\r\n }\r\n\r\n /**\r\n * Gets the direction if it has been set.\r\n * In case of direction provided, the shadow will not use a cube texture but simulate a spot shadow as a fallback\r\n */\r\n public get direction(): Vector3 {\r\n return this._direction;\r\n }\r\n\r\n /**\r\n * In case of direction provided, the shadow will not use a cube texture but simulate a spot shadow as a fallback\r\n */\r\n public set direction(value: Vector3) {\r\n const previousNeedCube = this.needCube();\r\n this._direction = value;\r\n if (this.needCube() !== previousNeedCube && this._shadowGenerators) {\r\n const iterator = this._shadowGenerators.values();\r\n for (let key = iterator.next(); key.done !== true; key = iterator.next()) {\r\n const shadowGenerator = key.value;\r\n shadowGenerator.recreateShadowMap();\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Creates a PointLight object from the passed name and position (Vector3) and adds it in the scene.\r\n * A PointLight emits the light in every direction.\r\n * It can cast shadows.\r\n * If the scene camera is already defined and you want to set your PointLight at the camera position, just set it :\r\n * ```javascript\r\n * var pointLight = new PointLight(\"pl\", camera.position, scene);\r\n * ```\r\n * Documentation : https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction\r\n * @param name The light friendly name\r\n * @param position The position of the point light in the scene\r\n * @param scene The scene the lights belongs to\r\n */\r\n constructor(name: string, position: Vector3, scene?: Scene) {\r\n super(name, scene);\r\n this.position = position;\r\n }\r\n\r\n /**\r\n * Returns the string \"PointLight\"\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"PointLight\";\r\n }\r\n\r\n /**\r\n * Returns the integer 0.\r\n * @returns The light Type id as a constant defines in Light.LIGHTTYPEID_x\r\n */\r\n public getTypeID(): number {\r\n return Light.LIGHTTYPEID_POINTLIGHT;\r\n }\r\n\r\n /**\r\n * Specifies whether or not the shadowmap should be a cube texture.\r\n * @returns true if the shadowmap needs to be a cube texture.\r\n */\r\n public needCube(): boolean {\r\n return !this.direction;\r\n }\r\n\r\n /**\r\n * Returns a new Vector3 aligned with the PointLight cube system according to the passed cube face index (integer).\r\n * @param faceIndex The index of the face we are computed the direction to generate shadow\r\n * @returns The set direction in 2d mode otherwise the direction to the cubemap face if needCube() is true\r\n */\r\n public getShadowDirection(faceIndex?: number): Vector3 {\r\n if (this.direction) {\r\n return super.getShadowDirection(faceIndex);\r\n } else {\r\n switch (faceIndex) {\r\n case 0:\r\n return new Vector3(1.0, 0.0, 0.0);\r\n case 1:\r\n return new Vector3(-1.0, 0.0, 0.0);\r\n case 2:\r\n return new Vector3(0.0, -1.0, 0.0);\r\n case 3:\r\n return new Vector3(0.0, 1.0, 0.0);\r\n case 4:\r\n return new Vector3(0.0, 0.0, 1.0);\r\n case 5:\r\n return new Vector3(0.0, 0.0, -1.0);\r\n }\r\n }\r\n\r\n return Vector3.Zero();\r\n }\r\n\r\n /**\r\n * Sets the passed matrix \"matrix\" as a left-handed perspective projection matrix with the following settings :\r\n * - fov = PI / 2\r\n * - aspect ratio : 1.0\r\n * - z-near and far equal to the active camera minZ and maxZ.\r\n * Returns the PointLight.\r\n * @param matrix\r\n * @param viewMatrix\r\n * @param renderList\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n protected _setDefaultShadowProjectionMatrix(matrix: Matrix, viewMatrix: Matrix, renderList: Array): void {\r\n const activeCamera = this.getScene().activeCamera;\r\n\r\n if (!activeCamera) {\r\n return;\r\n }\r\n\r\n const minZ = this.shadowMinZ !== undefined ? this.shadowMinZ : activeCamera.minZ;\r\n const maxZ = this.shadowMaxZ !== undefined ? this.shadowMaxZ : activeCamera.maxZ;\r\n\r\n const useReverseDepthBuffer = this.getScene().getEngine().useReverseDepthBuffer;\r\n\r\n Matrix.PerspectiveFovLHToRef(\r\n this.shadowAngle,\r\n 1.0,\r\n useReverseDepthBuffer ? maxZ : minZ,\r\n useReverseDepthBuffer ? minZ : maxZ,\r\n matrix,\r\n true,\r\n this._scene.getEngine().isNDCHalfZRange,\r\n undefined,\r\n useReverseDepthBuffer\r\n );\r\n }\r\n\r\n protected _buildUniformLayout(): void {\r\n this._uniformBuffer.addUniform(\"vLightData\", 4);\r\n this._uniformBuffer.addUniform(\"vLightDiffuse\", 4);\r\n this._uniformBuffer.addUniform(\"vLightSpecular\", 4);\r\n this._uniformBuffer.addUniform(\"vLightFalloff\", 4);\r\n this._uniformBuffer.addUniform(\"shadowsInfo\", 3);\r\n this._uniformBuffer.addUniform(\"depthValues\", 2);\r\n this._uniformBuffer.create();\r\n }\r\n\r\n /**\r\n * Sets the passed Effect \"effect\" with the PointLight transformed position (or position, if none) and passed name (string).\r\n * @param effect The effect to update\r\n * @param lightIndex The index of the light in the effect to update\r\n * @returns The point light\r\n */\r\n public transferToEffect(effect: Effect, lightIndex: string): PointLight {\r\n if (this.computeTransformedInformation()) {\r\n this._uniformBuffer.updateFloat4(\"vLightData\", this.transformedPosition.x, this.transformedPosition.y, this.transformedPosition.z, 0.0, lightIndex);\r\n } else {\r\n this._uniformBuffer.updateFloat4(\"vLightData\", this.position.x, this.position.y, this.position.z, 0, lightIndex);\r\n }\r\n\r\n this._uniformBuffer.updateFloat4(\"vLightFalloff\", this.range, this._inverseSquaredRange, 0, 0, lightIndex);\r\n return this;\r\n }\r\n\r\n public transferToNodeMaterialEffect(effect: Effect, lightDataUniformName: string) {\r\n if (this.computeTransformedInformation()) {\r\n effect.setFloat3(lightDataUniformName, this.transformedPosition.x, this.transformedPosition.y, this.transformedPosition.z);\r\n } else {\r\n effect.setFloat3(lightDataUniformName, this.position.x, this.position.y, this.position.z);\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Prepares the list of defines specific to the light type.\r\n * @param defines the list of defines\r\n * @param lightIndex defines the index of the light for the effect\r\n */\r\n public prepareLightSpecificDefines(defines: any, lightIndex: number): void {\r\n defines[\"POINTLIGHT\" + lightIndex] = true;\r\n }\r\n}\r\n","import { serialize, serializeAsTexture } from \"../Misc/decorators\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Matrix, Vector3 } from \"../Maths/math.vector\";\r\nimport { Node } from \"../node\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { Effect } from \"../Materials/effect\";\r\nimport type { BaseTexture } from \"../Materials/Textures/baseTexture\";\r\nimport { Light } from \"./light\";\r\nimport { ShadowLight } from \"./shadowLight\";\r\nimport { Texture } from \"../Materials/Textures/texture\";\r\nimport type { ProceduralTexture } from \"../Materials/Textures/Procedurals/proceduralTexture\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\n\r\nNode.AddNodeConstructor(\"Light_Type_2\", (name, scene) => {\r\n return () => new SpotLight(name, Vector3.Zero(), Vector3.Zero(), 0, 0, scene);\r\n});\r\n\r\n/**\r\n * A spot light is defined by a position, a direction, an angle, and an exponent.\r\n * These values define a cone of light starting from the position, emitting toward the direction.\r\n * The angle, in radians, defines the size (field of illumination) of the spotlight's conical beam,\r\n * and the exponent defines the speed of the decay of the light with distance (reach).\r\n * Documentation: https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction\r\n */\r\nexport class SpotLight extends ShadowLight {\r\n /*\r\n upVector , rightVector and direction will form the coordinate system for this spot light.\r\n These three vectors will be used as projection matrix when doing texture projection.\r\n\r\n Also we have the following rules always holds:\r\n direction cross up = right\r\n right cross direction = up\r\n up cross right = forward\r\n\r\n light_near and light_far will control the range of the texture projection. If a plane is\r\n out of the range in spot light space, there is no texture projection.\r\n */\r\n\r\n private _angle: number;\r\n private _innerAngle: number = 0;\r\n private _cosHalfAngle: number;\r\n\r\n private _lightAngleScale: number;\r\n private _lightAngleOffset: number;\r\n\r\n /**\r\n * Gets the cone angle of the spot light in Radians.\r\n */\r\n @serialize()\r\n public get angle(): number {\r\n return this._angle;\r\n }\r\n /**\r\n * Sets the cone angle of the spot light in Radians.\r\n */\r\n public set angle(value: number) {\r\n this._angle = value;\r\n this._cosHalfAngle = Math.cos(value * 0.5);\r\n this._projectionTextureProjectionLightDirty = true;\r\n this.forceProjectionMatrixCompute();\r\n this._computeAngleValues();\r\n }\r\n\r\n /**\r\n * Only used in gltf falloff mode, this defines the angle where\r\n * the directional falloff will start before cutting at angle which could be seen\r\n * as outer angle.\r\n */\r\n @serialize()\r\n public get innerAngle(): number {\r\n return this._innerAngle;\r\n }\r\n /**\r\n * Only used in gltf falloff mode, this defines the angle where\r\n * the directional falloff will start before cutting at angle which could be seen\r\n * as outer angle.\r\n */\r\n public set innerAngle(value: number) {\r\n this._innerAngle = value;\r\n this._computeAngleValues();\r\n }\r\n\r\n private _shadowAngleScale: number;\r\n /**\r\n * Allows scaling the angle of the light for shadow generation only.\r\n */\r\n @serialize()\r\n public get shadowAngleScale(): number {\r\n return this._shadowAngleScale;\r\n }\r\n /**\r\n * Allows scaling the angle of the light for shadow generation only.\r\n */\r\n public set shadowAngleScale(value: number) {\r\n this._shadowAngleScale = value;\r\n this.forceProjectionMatrixCompute();\r\n }\r\n\r\n /**\r\n * The light decay speed with the distance from the emission spot.\r\n */\r\n @serialize()\r\n public exponent: number;\r\n\r\n private _projectionTextureMatrix = Matrix.Zero();\r\n /**\r\n * Allows reading the projection texture\r\n */\r\n public get projectionTextureMatrix(): Matrix {\r\n return this._projectionTextureMatrix;\r\n }\r\n\r\n protected _projectionTextureLightNear: number = 1e-6;\r\n /**\r\n * Gets the near clip of the Spotlight for texture projection.\r\n */\r\n @serialize()\r\n public get projectionTextureLightNear(): number {\r\n return this._projectionTextureLightNear;\r\n }\r\n /**\r\n * Sets the near clip of the Spotlight for texture projection.\r\n */\r\n public set projectionTextureLightNear(value: number) {\r\n this._projectionTextureLightNear = value;\r\n this._projectionTextureProjectionLightDirty = true;\r\n }\r\n\r\n protected _projectionTextureLightFar: number = 1000.0;\r\n /**\r\n * Gets the far clip of the Spotlight for texture projection.\r\n */\r\n @serialize()\r\n public get projectionTextureLightFar(): number {\r\n return this._projectionTextureLightFar;\r\n }\r\n /**\r\n * Sets the far clip of the Spotlight for texture projection.\r\n */\r\n public set projectionTextureLightFar(value: number) {\r\n this._projectionTextureLightFar = value;\r\n this._projectionTextureProjectionLightDirty = true;\r\n }\r\n\r\n protected _projectionTextureUpDirection: Vector3 = Vector3.Up();\r\n /**\r\n * Gets the Up vector of the Spotlight for texture projection.\r\n */\r\n @serialize()\r\n public get projectionTextureUpDirection(): Vector3 {\r\n return this._projectionTextureUpDirection;\r\n }\r\n /**\r\n * Sets the Up vector of the Spotlight for texture projection.\r\n */\r\n public set projectionTextureUpDirection(value: Vector3) {\r\n this._projectionTextureUpDirection = value;\r\n this._projectionTextureProjectionLightDirty = true;\r\n }\r\n\r\n @serializeAsTexture(\"projectedLightTexture\")\r\n private _projectionTexture: Nullable;\r\n\r\n /**\r\n * Gets the projection texture of the light.\r\n */\r\n public get projectionTexture(): Nullable {\r\n return this._projectionTexture;\r\n }\r\n /**\r\n * Sets the projection texture of the light.\r\n */\r\n public set projectionTexture(value: Nullable) {\r\n if (this._projectionTexture === value) {\r\n return;\r\n }\r\n this._projectionTexture = value;\r\n this._projectionTextureDirty = true;\r\n if (this._projectionTexture && !this._projectionTexture.isReady()) {\r\n if (SpotLight._IsProceduralTexture(this._projectionTexture)) {\r\n this._projectionTexture.getEffect().executeWhenCompiled(() => {\r\n this._markMeshesAsLightDirty();\r\n });\r\n } else if (SpotLight._IsTexture(this._projectionTexture)) {\r\n this._projectionTexture.onLoadObservable.addOnce(() => {\r\n this._markMeshesAsLightDirty();\r\n });\r\n }\r\n }\r\n }\r\n\r\n private static _IsProceduralTexture(texture: BaseTexture): texture is ProceduralTexture {\r\n return (texture as ProceduralTexture).onGeneratedObservable !== undefined;\r\n }\r\n\r\n private static _IsTexture(texture: BaseTexture): texture is Texture {\r\n return (texture as Texture).onLoadObservable !== undefined;\r\n }\r\n\r\n private _projectionTextureViewLightDirty = true;\r\n private _projectionTextureProjectionLightDirty = true;\r\n private _projectionTextureDirty = true;\r\n private _projectionTextureViewTargetVector = Vector3.Zero();\r\n private _projectionTextureViewLightMatrix = Matrix.Zero();\r\n\r\n private _projectionTextureProjectionLightMatrix = Matrix.Zero();\r\n /**\r\n * Gets or sets the light projection matrix as used by the projection texture\r\n */\r\n public get projectionTextureProjectionLightMatrix(): Matrix {\r\n return this._projectionTextureProjectionLightMatrix;\r\n }\r\n\r\n public set projectionTextureProjectionLightMatrix(projection: Matrix) {\r\n this._projectionTextureProjectionLightMatrix = projection;\r\n this._projectionTextureProjectionLightDirty = false;\r\n this._projectionTextureDirty = true;\r\n }\r\n\r\n private _projectionTextureScalingMatrix = Matrix.FromValues(0.5, 0.0, 0.0, 0.0, 0.0, 0.5, 0.0, 0.0, 0.0, 0.0, 0.5, 0.0, 0.5, 0.5, 0.5, 1.0);\r\n\r\n /**\r\n * Creates a SpotLight object in the scene. A spot light is a simply light oriented cone.\r\n * It can cast shadows.\r\n * Documentation : https://doc.babylonjs.com/features/featuresDeepDive/lights/lights_introduction\r\n * @param name The light friendly name\r\n * @param position The position of the spot light in the scene\r\n * @param direction The direction of the light in the scene\r\n * @param angle The cone angle of the light in Radians\r\n * @param exponent The light decay speed with the distance from the emission spot\r\n * @param scene The scene the lights belongs to\r\n */\r\n constructor(name: string, position: Vector3, direction: Vector3, angle: number, exponent: number, scene?: Scene) {\r\n super(name, scene);\r\n\r\n this.position = position;\r\n this.direction = direction;\r\n this.angle = angle;\r\n this.exponent = exponent;\r\n }\r\n\r\n /**\r\n * Returns the string \"SpotLight\".\r\n * @returns the class name\r\n */\r\n public getClassName(): string {\r\n return \"SpotLight\";\r\n }\r\n\r\n /**\r\n * Returns the integer 2.\r\n * @returns The light Type id as a constant defines in Light.LIGHTTYPEID_x\r\n */\r\n public getTypeID(): number {\r\n return Light.LIGHTTYPEID_SPOTLIGHT;\r\n }\r\n\r\n /**\r\n * Overrides the direction setter to recompute the projection texture view light Matrix.\r\n * @param value\r\n */\r\n protected _setDirection(value: Vector3) {\r\n super._setDirection(value);\r\n this._projectionTextureViewLightDirty = true;\r\n }\r\n\r\n /**\r\n * Overrides the position setter to recompute the projection texture view light Matrix.\r\n * @param value\r\n */\r\n protected _setPosition(value: Vector3) {\r\n super._setPosition(value);\r\n this._projectionTextureViewLightDirty = true;\r\n }\r\n\r\n /**\r\n * Sets the passed matrix \"matrix\" as perspective projection matrix for the shadows and the passed view matrix with the fov equal to the SpotLight angle and and aspect ratio of 1.0.\r\n * Returns the SpotLight.\r\n * @param matrix\r\n * @param viewMatrix\r\n * @param renderList\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n protected _setDefaultShadowProjectionMatrix(matrix: Matrix, viewMatrix: Matrix, renderList: Array): void {\r\n const activeCamera = this.getScene().activeCamera;\r\n\r\n if (!activeCamera) {\r\n return;\r\n }\r\n\r\n this._shadowAngleScale = this._shadowAngleScale || 1;\r\n const angle = this._shadowAngleScale * this._angle;\r\n\r\n const minZ = this.shadowMinZ !== undefined ? this.shadowMinZ : activeCamera.minZ;\r\n const maxZ = this.shadowMaxZ !== undefined ? this.shadowMaxZ : activeCamera.maxZ;\r\n\r\n const useReverseDepthBuffer = this.getScene().getEngine().useReverseDepthBuffer;\r\n\r\n Matrix.PerspectiveFovLHToRef(\r\n angle,\r\n 1.0,\r\n useReverseDepthBuffer ? maxZ : minZ,\r\n useReverseDepthBuffer ? minZ : maxZ,\r\n matrix,\r\n true,\r\n this._scene.getEngine().isNDCHalfZRange,\r\n undefined,\r\n useReverseDepthBuffer\r\n );\r\n }\r\n\r\n protected _computeProjectionTextureViewLightMatrix(): void {\r\n this._projectionTextureViewLightDirty = false;\r\n this._projectionTextureDirty = true;\r\n\r\n this.getAbsolutePosition().addToRef(this.direction, this._projectionTextureViewTargetVector);\r\n Matrix.LookAtLHToRef(this.getAbsolutePosition(), this._projectionTextureViewTargetVector, this._projectionTextureUpDirection, this._projectionTextureViewLightMatrix);\r\n }\r\n\r\n protected _computeProjectionTextureProjectionLightMatrix(): void {\r\n this._projectionTextureProjectionLightDirty = false;\r\n this._projectionTextureDirty = true;\r\n\r\n const lightFar = this.projectionTextureLightFar;\r\n const lightNear = this.projectionTextureLightNear;\r\n\r\n const P = lightFar / (lightFar - lightNear);\r\n const Q = -P * lightNear;\r\n const S = 1.0 / Math.tan(this._angle / 2.0);\r\n const A = 1.0;\r\n\r\n Matrix.FromValuesToRef(S / A, 0.0, 0.0, 0.0, 0.0, S, 0.0, 0.0, 0.0, 0.0, P, 1.0, 0.0, 0.0, Q, 0.0, this._projectionTextureProjectionLightMatrix);\r\n }\r\n\r\n /**\r\n * Main function for light texture projection matrix computing.\r\n */\r\n protected _computeProjectionTextureMatrix(): void {\r\n this._projectionTextureDirty = false;\r\n\r\n this._projectionTextureViewLightMatrix.multiplyToRef(this._projectionTextureProjectionLightMatrix, this._projectionTextureMatrix);\r\n if (this._projectionTexture instanceof Texture) {\r\n const u = this._projectionTexture.uScale / 2.0;\r\n const v = this._projectionTexture.vScale / 2.0;\r\n Matrix.FromValuesToRef(u, 0.0, 0.0, 0.0, 0.0, v, 0.0, 0.0, 0.0, 0.0, 0.5, 0.0, 0.5, 0.5, 0.5, 1.0, this._projectionTextureScalingMatrix);\r\n }\r\n this._projectionTextureMatrix.multiplyToRef(this._projectionTextureScalingMatrix, this._projectionTextureMatrix);\r\n }\r\n\r\n protected _buildUniformLayout(): void {\r\n this._uniformBuffer.addUniform(\"vLightData\", 4);\r\n this._uniformBuffer.addUniform(\"vLightDiffuse\", 4);\r\n this._uniformBuffer.addUniform(\"vLightSpecular\", 4);\r\n this._uniformBuffer.addUniform(\"vLightDirection\", 3);\r\n this._uniformBuffer.addUniform(\"vLightFalloff\", 4);\r\n this._uniformBuffer.addUniform(\"shadowsInfo\", 3);\r\n this._uniformBuffer.addUniform(\"depthValues\", 2);\r\n this._uniformBuffer.create();\r\n }\r\n\r\n private _computeAngleValues(): void {\r\n this._lightAngleScale = 1.0 / Math.max(0.001, Math.cos(this._innerAngle * 0.5) - this._cosHalfAngle);\r\n this._lightAngleOffset = -this._cosHalfAngle * this._lightAngleScale;\r\n }\r\n\r\n /**\r\n * Sets the passed Effect \"effect\" with the Light textures.\r\n * @param effect The effect to update\r\n * @param lightIndex The index of the light in the effect to update\r\n * @returns The light\r\n */\r\n public transferTexturesToEffect(effect: Effect, lightIndex: string): Light {\r\n if (this.projectionTexture && this.projectionTexture.isReady()) {\r\n if (this._projectionTextureViewLightDirty) {\r\n this._computeProjectionTextureViewLightMatrix();\r\n }\r\n if (this._projectionTextureProjectionLightDirty) {\r\n this._computeProjectionTextureProjectionLightMatrix();\r\n }\r\n if (this._projectionTextureDirty) {\r\n this._computeProjectionTextureMatrix();\r\n }\r\n effect.setMatrix(\"textureProjectionMatrix\" + lightIndex, this._projectionTextureMatrix);\r\n effect.setTexture(\"projectionLightSampler\" + lightIndex, this.projectionTexture);\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets the passed Effect object with the SpotLight transformed position (or position if not parented) and normalized direction.\r\n * @param effect The effect to update\r\n * @param lightIndex The index of the light in the effect to update\r\n * @returns The spot light\r\n */\r\n public transferToEffect(effect: Effect, lightIndex: string): SpotLight {\r\n let normalizeDirection;\r\n\r\n if (this.computeTransformedInformation()) {\r\n this._uniformBuffer.updateFloat4(\"vLightData\", this.transformedPosition.x, this.transformedPosition.y, this.transformedPosition.z, this.exponent, lightIndex);\r\n\r\n normalizeDirection = Vector3.Normalize(this.transformedDirection);\r\n } else {\r\n this._uniformBuffer.updateFloat4(\"vLightData\", this.position.x, this.position.y, this.position.z, this.exponent, lightIndex);\r\n\r\n normalizeDirection = Vector3.Normalize(this.direction);\r\n }\r\n\r\n this._uniformBuffer.updateFloat4(\"vLightDirection\", normalizeDirection.x, normalizeDirection.y, normalizeDirection.z, this._cosHalfAngle, lightIndex);\r\n\r\n this._uniformBuffer.updateFloat4(\"vLightFalloff\", this.range, this._inverseSquaredRange, this._lightAngleScale, this._lightAngleOffset, lightIndex);\r\n return this;\r\n }\r\n\r\n public transferToNodeMaterialEffect(effect: Effect, lightDataUniformName: string) {\r\n let normalizeDirection;\r\n\r\n if (this.computeTransformedInformation()) {\r\n normalizeDirection = Vector3.Normalize(this.transformedDirection);\r\n } else {\r\n normalizeDirection = Vector3.Normalize(this.direction);\r\n }\r\n\r\n if (this.getScene().useRightHandedSystem) {\r\n effect.setFloat3(lightDataUniformName, -normalizeDirection.x, -normalizeDirection.y, -normalizeDirection.z);\r\n } else {\r\n effect.setFloat3(lightDataUniformName, normalizeDirection.x, normalizeDirection.y, normalizeDirection.z);\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Disposes the light and the associated resources.\r\n */\r\n public dispose(): void {\r\n super.dispose();\r\n if (this._projectionTexture) {\r\n this._projectionTexture.dispose();\r\n }\r\n }\r\n\r\n /**\r\n * Gets the minZ used for shadow according to both the scene and the light.\r\n * @param activeCamera The camera we are returning the min for\r\n * @returns the depth min z\r\n */\r\n public getDepthMinZ(activeCamera: Camera): number {\r\n const engine = this._scene.getEngine();\r\n const minZ = this.shadowMinZ !== undefined ? this.shadowMinZ : activeCamera.minZ;\r\n\r\n return engine.useReverseDepthBuffer && engine.isNDCHalfZRange ? minZ : this._scene.getEngine().isNDCHalfZRange ? 0 : minZ;\r\n }\r\n\r\n /**\r\n * Gets the maxZ used for shadow according to both the scene and the light.\r\n * @param activeCamera The camera we are returning the max for\r\n * @returns the depth max z\r\n */\r\n public getDepthMaxZ(activeCamera: Camera): number {\r\n const engine = this._scene.getEngine();\r\n const maxZ = this.shadowMaxZ !== undefined ? this.shadowMaxZ : activeCamera.maxZ;\r\n\r\n return engine.useReverseDepthBuffer && engine.isNDCHalfZRange ? 0 : maxZ;\r\n }\r\n\r\n /**\r\n * Prepares the list of defines specific to the light type.\r\n * @param defines the list of defines\r\n * @param lightIndex defines the index of the light for the effect\r\n */\r\n public prepareLightSpecificDefines(defines: any, lightIndex: number): void {\r\n defines[\"SPOTLIGHT\" + lightIndex] = true;\r\n defines[\"PROJECTEDLIGHTTEXTURE\" + lightIndex] = this.projectionTexture && this.projectionTexture.isReady() ? true : false;\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/no-unused-vars */\r\nimport type { Scene, IDisposable } from \"../scene\";\r\nimport { EngineStore } from \"../Engines/engineStore\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport { Mesh } from \"../Meshes/mesh\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { Observer } from \"./observable\";\r\nimport { Observable } from \"./observable\";\r\n\r\n/**\r\n * Defines the root class used to create scene optimization to use with SceneOptimizer\r\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\r\n */\r\nexport class SceneOptimization {\r\n /**\r\n * Gets a string describing the action executed by the current optimization\r\n * @returns description string\r\n */\r\n public getDescription(): string {\r\n return \"\";\r\n }\r\n\r\n /**\r\n * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization\r\n * @param scene defines the current scene where to apply this optimization\r\n * @param optimizer defines the current optimizer\r\n * @returns true if everything that can be done was applied\r\n */\r\n public apply(scene: Scene, optimizer: SceneOptimizer): boolean {\r\n return true;\r\n }\r\n\r\n /**\r\n * Creates the SceneOptimization object\r\n * @param priority defines the priority of this optimization (0 by default which means first in the list)\r\n */\r\n constructor(\r\n /**\r\n * Defines the priority of this optimization (0 by default which means first in the list)\r\n */\r\n public priority: number = 0\r\n ) {}\r\n}\r\n\r\n/**\r\n * Defines an optimization used to reduce the size of render target textures\r\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\r\n */\r\nexport class TextureOptimization extends SceneOptimization {\r\n /**\r\n * Gets a string describing the action executed by the current optimization\r\n * @returns description string\r\n */\r\n public getDescription(): string {\r\n return \"Reducing render target texture size to \" + this.maximumSize;\r\n }\r\n\r\n /**\r\n * Creates the TextureOptimization object\r\n * @param priority defines the priority of this optimization (0 by default which means first in the list)\r\n * @param maximumSize defines the maximum sized allowed for textures (1024 is the default value). If a texture is bigger, it will be scaled down using a factor defined by the step parameter\r\n * @param step defines the factor (0.5 by default) used to scale down textures bigger than maximum sized allowed.\r\n */\r\n constructor(\r\n /**\r\n * Defines the priority of this optimization (0 by default which means first in the list)\r\n */\r\n public priority: number = 0,\r\n /**\r\n * Defines the maximum sized allowed for textures (1024 is the default value). If a texture is bigger, it will be scaled down using a factor defined by the step parameter\r\n */\r\n public maximumSize: number = 1024,\r\n /**\r\n * Defines the factor (0.5 by default) used to scale down textures bigger than maximum sized allowed.\r\n */\r\n public step = 0.5\r\n ) {\r\n super(priority);\r\n }\r\n\r\n /**\r\n * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization\r\n * @param scene defines the current scene where to apply this optimization\r\n * @param optimizer defines the current optimizer\r\n * @returns true if everything that can be done was applied\r\n */\r\n public apply(scene: Scene, optimizer: SceneOptimizer): boolean {\r\n let allDone = true;\r\n for (let index = 0; index < scene.textures.length; index++) {\r\n const texture = scene.textures[index];\r\n\r\n if (!texture.canRescale || (texture).getContext) {\r\n continue;\r\n }\r\n\r\n const currentSize = texture.getSize();\r\n const maxDimension = Math.max(currentSize.width, currentSize.height);\r\n\r\n if (maxDimension > this.maximumSize) {\r\n texture.scale(this.step);\r\n allDone = false;\r\n }\r\n }\r\n\r\n return allDone;\r\n }\r\n}\r\n\r\n/**\r\n * Defines an optimization used to increase or decrease the rendering resolution\r\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\r\n */\r\nexport class HardwareScalingOptimization extends SceneOptimization {\r\n private _currentScale = -1;\r\n private _directionOffset = 1;\r\n\r\n /**\r\n * Gets a string describing the action executed by the current optimization\r\n * @returns description string\r\n */\r\n public getDescription(): string {\r\n return \"Setting hardware scaling level to \" + this._currentScale;\r\n }\r\n\r\n /**\r\n * Creates the HardwareScalingOptimization object\r\n * @param priority defines the priority of this optimization (0 by default which means first in the list)\r\n * @param maximumScale defines the maximum scale to use (2 by default)\r\n * @param step defines the step to use between two passes (0.5 by default)\r\n */\r\n constructor(\r\n /**\r\n * Defines the priority of this optimization (0 by default which means first in the list)\r\n */\r\n public priority: number = 0,\r\n /**\r\n * Defines the maximum scale to use (2 by default)\r\n */\r\n public maximumScale: number = 2,\r\n /**\r\n * Defines the step to use between two passes (0.5 by default)\r\n */\r\n public step: number = 0.25\r\n ) {\r\n super(priority);\r\n }\r\n\r\n /**\r\n * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization\r\n * @param scene defines the current scene where to apply this optimization\r\n * @param optimizer defines the current optimizer\r\n * @returns true if everything that can be done was applied\r\n */\r\n public apply(scene: Scene, optimizer: SceneOptimizer): boolean {\r\n if (this._currentScale === -1) {\r\n this._currentScale = scene.getEngine().getHardwareScalingLevel();\r\n if (this._currentScale > this.maximumScale) {\r\n this._directionOffset = -1;\r\n }\r\n }\r\n\r\n this._currentScale += this._directionOffset * this.step;\r\n\r\n scene.getEngine().setHardwareScalingLevel(this._currentScale);\r\n\r\n return this._directionOffset === 1 ? this._currentScale >= this.maximumScale : this._currentScale <= this.maximumScale;\r\n }\r\n}\r\n\r\n/**\r\n * Defines an optimization used to remove shadows\r\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\r\n */\r\nexport class ShadowsOptimization extends SceneOptimization {\r\n /**\r\n * Gets a string describing the action executed by the current optimization\r\n * @returns description string\r\n */\r\n public getDescription(): string {\r\n return \"Turning shadows on/off\";\r\n }\r\n\r\n /**\r\n * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization\r\n * @param scene defines the current scene where to apply this optimization\r\n * @param optimizer defines the current optimizer\r\n * @returns true if everything that can be done was applied\r\n */\r\n public apply(scene: Scene, optimizer: SceneOptimizer): boolean {\r\n scene.shadowsEnabled = optimizer.isInImprovementMode;\r\n return true;\r\n }\r\n}\r\n\r\n/**\r\n * Defines an optimization used to turn post-processes off\r\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\r\n */\r\nexport class PostProcessesOptimization extends SceneOptimization {\r\n /**\r\n * Gets a string describing the action executed by the current optimization\r\n * @returns description string\r\n */\r\n public getDescription(): string {\r\n return \"Turning post-processes on/off\";\r\n }\r\n\r\n /**\r\n * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization\r\n * @param scene defines the current scene where to apply this optimization\r\n * @param optimizer defines the current optimizer\r\n * @returns true if everything that can be done was applied\r\n */\r\n public apply(scene: Scene, optimizer: SceneOptimizer): boolean {\r\n scene.postProcessesEnabled = optimizer.isInImprovementMode;\r\n return true;\r\n }\r\n}\r\n\r\n/**\r\n * Defines an optimization used to turn lens flares off\r\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\r\n */\r\nexport class LensFlaresOptimization extends SceneOptimization {\r\n /**\r\n * Gets a string describing the action executed by the current optimization\r\n * @returns description string\r\n */\r\n public getDescription(): string {\r\n return \"Turning lens flares on/off\";\r\n }\r\n\r\n /**\r\n * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization\r\n * @param scene defines the current scene where to apply this optimization\r\n * @param optimizer defines the current optimizer\r\n * @returns true if everything that can be done was applied\r\n */\r\n public apply(scene: Scene, optimizer: SceneOptimizer): boolean {\r\n scene.lensFlaresEnabled = optimizer.isInImprovementMode;\r\n return true;\r\n }\r\n}\r\n\r\n/**\r\n * Defines an optimization based on user defined callback.\r\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\r\n */\r\nexport class CustomOptimization extends SceneOptimization {\r\n /**\r\n * Callback called to apply the custom optimization.\r\n */\r\n public onApply: (scene: Scene, optimizer: SceneOptimizer) => boolean;\r\n\r\n /**\r\n * Callback called to get custom description\r\n */\r\n public onGetDescription: () => string;\r\n\r\n /**\r\n * Gets a string describing the action executed by the current optimization\r\n * @returns description string\r\n */\r\n public getDescription(): string {\r\n if (this.onGetDescription) {\r\n return this.onGetDescription();\r\n }\r\n\r\n return \"Running user defined callback\";\r\n }\r\n\r\n /**\r\n * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization\r\n * @param scene defines the current scene where to apply this optimization\r\n * @param optimizer defines the current optimizer\r\n * @returns true if everything that can be done was applied\r\n */\r\n public apply(scene: Scene, optimizer: SceneOptimizer): boolean {\r\n if (this.onApply) {\r\n return this.onApply(scene, optimizer);\r\n }\r\n return true;\r\n }\r\n}\r\n\r\n/**\r\n * Defines an optimization used to turn particles off\r\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\r\n */\r\nexport class ParticlesOptimization extends SceneOptimization {\r\n /**\r\n * Gets a string describing the action executed by the current optimization\r\n * @returns description string\r\n */\r\n public getDescription(): string {\r\n return \"Turning particles on/off\";\r\n }\r\n\r\n /**\r\n * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization\r\n * @param scene defines the current scene where to apply this optimization\r\n * @param optimizer defines the current optimizer\r\n * @returns true if everything that can be done was applied\r\n */\r\n public apply(scene: Scene, optimizer: SceneOptimizer): boolean {\r\n scene.particlesEnabled = optimizer.isInImprovementMode;\r\n return true;\r\n }\r\n}\r\n\r\n/**\r\n * Defines an optimization used to turn render targets off\r\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\r\n */\r\nexport class RenderTargetsOptimization extends SceneOptimization {\r\n /**\r\n * Gets a string describing the action executed by the current optimization\r\n * @returns description string\r\n */\r\n public getDescription(): string {\r\n return \"Turning render targets off\";\r\n }\r\n\r\n /**\r\n * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization\r\n * @param scene defines the current scene where to apply this optimization\r\n * @param optimizer defines the current optimizer\r\n * @returns true if everything that can be done was applied\r\n */\r\n public apply(scene: Scene, optimizer: SceneOptimizer): boolean {\r\n scene.renderTargetsEnabled = optimizer.isInImprovementMode;\r\n return true;\r\n }\r\n}\r\n\r\n/**\r\n * Defines an optimization used to merge meshes with compatible materials\r\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\r\n */\r\nexport class MergeMeshesOptimization extends SceneOptimization {\r\n private static _UpdateSelectionTree = false;\r\n\r\n /**\r\n * Gets or sets a boolean which defines if optimization octree has to be updated\r\n */\r\n public static get UpdateSelectionTree(): boolean {\r\n return MergeMeshesOptimization._UpdateSelectionTree;\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean which defines if optimization octree has to be updated\r\n */\r\n public static set UpdateSelectionTree(value: boolean) {\r\n MergeMeshesOptimization._UpdateSelectionTree = value;\r\n }\r\n\r\n /**\r\n * Gets a string describing the action executed by the current optimization\r\n * @returns description string\r\n */\r\n public getDescription(): string {\r\n return \"Merging similar meshes together\";\r\n }\r\n\r\n private _canBeMerged = (abstractMesh: AbstractMesh): boolean => {\r\n if (!(abstractMesh instanceof Mesh)) {\r\n return false;\r\n }\r\n\r\n const mesh = abstractMesh;\r\n\r\n if (mesh.isDisposed()) {\r\n return false;\r\n }\r\n\r\n if (!mesh.isVisible || !mesh.isEnabled()) {\r\n return false;\r\n }\r\n\r\n if (mesh.instances.length > 0) {\r\n return false;\r\n }\r\n\r\n if (mesh.skeleton || mesh.hasLODLevels) {\r\n return false;\r\n }\r\n\r\n if (mesh.getTotalVertices() === 0) {\r\n return false;\r\n }\r\n\r\n return true;\r\n };\r\n\r\n /**\r\n * This function will be called by the SceneOptimizer when its priority is reached in order to apply the change required by the current optimization\r\n * @param scene defines the current scene where to apply this optimization\r\n * @param optimizer defines the current optimizer\r\n * @param updateSelectionTree defines that the selection octree has to be updated (false by default)\r\n * @returns true if everything that can be done was applied\r\n */\r\n public apply(scene: Scene, optimizer: SceneOptimizer, updateSelectionTree?: boolean): boolean {\r\n const globalPool = scene.meshes.slice(0);\r\n let globalLength = globalPool.length;\r\n\r\n for (let index = 0; index < globalLength; index++) {\r\n const currentPool = new Array();\r\n const current = globalPool[index];\r\n\r\n // Checks\r\n if (!this._canBeMerged(current)) {\r\n continue;\r\n }\r\n\r\n currentPool.push(current);\r\n\r\n // Find compatible meshes\r\n for (let subIndex = index + 1; subIndex < globalLength; subIndex++) {\r\n const otherMesh = globalPool[subIndex];\r\n\r\n if (!this._canBeMerged(otherMesh)) {\r\n continue;\r\n }\r\n\r\n if (otherMesh.material !== current.material) {\r\n continue;\r\n }\r\n\r\n if (otherMesh.checkCollisions !== current.checkCollisions) {\r\n continue;\r\n }\r\n\r\n currentPool.push(otherMesh);\r\n globalLength--;\r\n\r\n globalPool.splice(subIndex, 1);\r\n\r\n subIndex--;\r\n }\r\n\r\n if (currentPool.length < 2) {\r\n continue;\r\n }\r\n\r\n // Merge meshes\r\n Mesh.MergeMeshes(currentPool, undefined, true);\r\n }\r\n\r\n // Call the octree system optimization if it is defined.\r\n const sceneAsAny = scene as any;\r\n if (sceneAsAny.createOrUpdateSelectionOctree) {\r\n if (updateSelectionTree != undefined) {\r\n if (updateSelectionTree) {\r\n sceneAsAny.createOrUpdateSelectionOctree();\r\n }\r\n } else if (MergeMeshesOptimization.UpdateSelectionTree) {\r\n sceneAsAny.createOrUpdateSelectionOctree();\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n}\r\n\r\n/**\r\n * Defines a list of options used by SceneOptimizer\r\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\r\n */\r\nexport class SceneOptimizerOptions {\r\n /**\r\n * Gets the list of optimizations to apply\r\n */\r\n public optimizations = new Array();\r\n\r\n /**\r\n * Creates a new list of options used by SceneOptimizer\r\n * @param targetFrameRate defines the target frame rate to reach (60 by default)\r\n * @param trackerDuration defines the interval between two checks (2000ms by default)\r\n */\r\n constructor(\r\n /**\r\n * Defines the target frame rate to reach (60 by default)\r\n */\r\n public targetFrameRate: number = 60,\r\n /**\r\n * Defines the interval between two checks (2000ms by default)\r\n */\r\n public trackerDuration: number = 2000\r\n ) {}\r\n\r\n /**\r\n * Add a new optimization\r\n * @param optimization defines the SceneOptimization to add to the list of active optimizations\r\n * @returns the current SceneOptimizerOptions\r\n */\r\n public addOptimization(optimization: SceneOptimization): SceneOptimizerOptions {\r\n this.optimizations.push(optimization);\r\n return this;\r\n }\r\n\r\n /**\r\n * Add a new custom optimization\r\n * @param onApply defines the callback called to apply the custom optimization (true if everything that can be done was applied)\r\n * @param onGetDescription defines the callback called to get the description attached with the optimization.\r\n * @param priority defines the priority of this optimization (0 by default which means first in the list)\r\n * @returns the current SceneOptimizerOptions\r\n */\r\n public addCustomOptimization(onApply: (scene: Scene, optimizer: SceneOptimizer) => boolean, onGetDescription: () => string, priority: number = 0): SceneOptimizerOptions {\r\n const optimization = new CustomOptimization(priority);\r\n optimization.onApply = onApply;\r\n optimization.onGetDescription = onGetDescription;\r\n\r\n this.optimizations.push(optimization);\r\n return this;\r\n }\r\n\r\n /**\r\n * Creates a list of pre-defined optimizations aimed to reduce the visual impact on the scene\r\n * @param targetFrameRate defines the target frame rate (60 by default)\r\n * @returns a SceneOptimizerOptions object\r\n */\r\n public static LowDegradationAllowed(targetFrameRate?: number): SceneOptimizerOptions {\r\n const result = new SceneOptimizerOptions(targetFrameRate);\r\n\r\n let priority = 0;\r\n result.addOptimization(new MergeMeshesOptimization(priority));\r\n result.addOptimization(new ShadowsOptimization(priority));\r\n result.addOptimization(new LensFlaresOptimization(priority));\r\n\r\n // Next priority\r\n priority++;\r\n result.addOptimization(new PostProcessesOptimization(priority));\r\n result.addOptimization(new ParticlesOptimization(priority));\r\n\r\n // Next priority\r\n priority++;\r\n result.addOptimization(new TextureOptimization(priority, 1024));\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a list of pre-defined optimizations aimed to have a moderate impact on the scene visual\r\n * @param targetFrameRate defines the target frame rate (60 by default)\r\n * @returns a SceneOptimizerOptions object\r\n */\r\n public static ModerateDegradationAllowed(targetFrameRate?: number): SceneOptimizerOptions {\r\n const result = new SceneOptimizerOptions(targetFrameRate);\r\n\r\n let priority = 0;\r\n result.addOptimization(new MergeMeshesOptimization(priority));\r\n result.addOptimization(new ShadowsOptimization(priority));\r\n result.addOptimization(new LensFlaresOptimization(priority));\r\n\r\n // Next priority\r\n priority++;\r\n result.addOptimization(new PostProcessesOptimization(priority));\r\n result.addOptimization(new ParticlesOptimization(priority));\r\n\r\n // Next priority\r\n priority++;\r\n result.addOptimization(new TextureOptimization(priority, 512));\r\n\r\n // Next priority\r\n priority++;\r\n result.addOptimization(new RenderTargetsOptimization(priority));\r\n\r\n // Next priority\r\n priority++;\r\n result.addOptimization(new HardwareScalingOptimization(priority, 2));\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a list of pre-defined optimizations aimed to have a big impact on the scene visual\r\n * @param targetFrameRate defines the target frame rate (60 by default)\r\n * @returns a SceneOptimizerOptions object\r\n */\r\n public static HighDegradationAllowed(targetFrameRate?: number): SceneOptimizerOptions {\r\n const result = new SceneOptimizerOptions(targetFrameRate);\r\n\r\n let priority = 0;\r\n result.addOptimization(new MergeMeshesOptimization(priority));\r\n result.addOptimization(new ShadowsOptimization(priority));\r\n result.addOptimization(new LensFlaresOptimization(priority));\r\n\r\n // Next priority\r\n priority++;\r\n result.addOptimization(new PostProcessesOptimization(priority));\r\n result.addOptimization(new ParticlesOptimization(priority));\r\n\r\n // Next priority\r\n priority++;\r\n result.addOptimization(new TextureOptimization(priority, 256));\r\n\r\n // Next priority\r\n priority++;\r\n result.addOptimization(new RenderTargetsOptimization(priority));\r\n\r\n // Next priority\r\n priority++;\r\n result.addOptimization(new HardwareScalingOptimization(priority, 4));\r\n\r\n return result;\r\n }\r\n}\r\n\r\n/**\r\n * Class used to run optimizations in order to reach a target frame rate\r\n * @description More details at https://doc.babylonjs.com/features/featuresDeepDive/scene/sceneOptimizer\r\n */\r\nexport class SceneOptimizer implements IDisposable {\r\n private _isRunning = false;\r\n private _options: SceneOptimizerOptions;\r\n private _scene: Scene;\r\n private _currentPriorityLevel = 0;\r\n private _targetFrameRate = 60;\r\n private _trackerDuration = 2000;\r\n private _currentFrameRate = 0;\r\n private _sceneDisposeObserver: Nullable>;\r\n private _improvementMode = false;\r\n\r\n /**\r\n * Defines an observable called when the optimizer reaches the target frame rate\r\n */\r\n public onSuccessObservable = new Observable();\r\n /**\r\n * Defines an observable called when the optimizer enables an optimization\r\n */\r\n public onNewOptimizationAppliedObservable = new Observable();\r\n /**\r\n * Defines an observable called when the optimizer is not able to reach the target frame rate\r\n */\r\n public onFailureObservable = new Observable();\r\n\r\n /**\r\n * Gets or sets a boolean indicating if the optimizer is in improvement mode\r\n */\r\n public get isInImprovementMode(): boolean {\r\n return this._improvementMode;\r\n }\r\n\r\n public set isInImprovementMode(value: boolean) {\r\n this._improvementMode = value;\r\n }\r\n\r\n /**\r\n * Gets the current priority level (0 at start)\r\n */\r\n public get currentPriorityLevel(): number {\r\n return this._currentPriorityLevel;\r\n }\r\n\r\n /**\r\n * Gets the current frame rate checked by the SceneOptimizer\r\n */\r\n public get currentFrameRate(): number {\r\n return this._currentFrameRate;\r\n }\r\n\r\n /**\r\n * Gets or sets the current target frame rate (60 by default)\r\n */\r\n public get targetFrameRate(): number {\r\n return this._targetFrameRate;\r\n }\r\n\r\n /**\r\n * Gets or sets the current target frame rate (60 by default)\r\n */\r\n public set targetFrameRate(value: number) {\r\n this._targetFrameRate = value;\r\n }\r\n\r\n /**\r\n * Gets or sets the current interval between two checks (every 2000ms by default)\r\n */\r\n public get trackerDuration(): number {\r\n return this._trackerDuration;\r\n }\r\n\r\n /**\r\n * Gets or sets the current interval between two checks (every 2000ms by default)\r\n */\r\n public set trackerDuration(value: number) {\r\n this._trackerDuration = value;\r\n }\r\n\r\n /**\r\n * Gets the list of active optimizations\r\n */\r\n public get optimizations(): SceneOptimization[] {\r\n return this._options.optimizations;\r\n }\r\n\r\n /**\r\n * Creates a new SceneOptimizer\r\n * @param scene defines the scene to work on\r\n * @param options defines the options to use with the SceneOptimizer\r\n * @param autoGeneratePriorities defines if priorities must be generated and not read from SceneOptimization property (true by default)\r\n * @param improvementMode defines if the scene optimizer must run the maximum optimization while staying over a target frame instead of trying to reach the target framerate (false by default)\r\n */\r\n public constructor(scene: Scene, options?: SceneOptimizerOptions, autoGeneratePriorities = true, improvementMode = false) {\r\n if (!options) {\r\n this._options = new SceneOptimizerOptions();\r\n } else {\r\n this._options = options;\r\n }\r\n\r\n if (this._options.targetFrameRate) {\r\n this._targetFrameRate = this._options.targetFrameRate;\r\n }\r\n\r\n if (this._options.trackerDuration) {\r\n this._trackerDuration = this._options.trackerDuration;\r\n }\r\n\r\n if (autoGeneratePriorities) {\r\n let priority = 0;\r\n for (const optim of this._options.optimizations) {\r\n optim.priority = priority++;\r\n }\r\n }\r\n\r\n this._improvementMode = improvementMode;\r\n this._scene = scene || EngineStore.LastCreatedScene;\r\n this._sceneDisposeObserver = this._scene.onDisposeObservable.add(() => {\r\n this._sceneDisposeObserver = null;\r\n this.dispose();\r\n });\r\n }\r\n\r\n /**\r\n * Stops the current optimizer\r\n */\r\n public stop() {\r\n this._isRunning = false;\r\n }\r\n\r\n /**\r\n * Reset the optimizer to initial step (current priority level = 0)\r\n */\r\n public reset() {\r\n this._currentPriorityLevel = 0;\r\n }\r\n\r\n /**\r\n * Start the optimizer. By default it will try to reach a specific framerate\r\n * but if the optimizer is set with improvementMode === true then it will run all optimization while frame rate is above the target frame rate\r\n */\r\n public start() {\r\n if (this._isRunning) {\r\n return;\r\n }\r\n\r\n this._isRunning = true;\r\n\r\n // Let's wait for the scene to be ready before running our check\r\n this._scene.executeWhenReady(() => {\r\n setTimeout(() => {\r\n this._checkCurrentState();\r\n }, this._trackerDuration);\r\n });\r\n }\r\n\r\n private _checkCurrentState() {\r\n if (!this._isRunning) {\r\n return;\r\n }\r\n\r\n const scene = this._scene;\r\n const options = this._options;\r\n\r\n this._currentFrameRate = Math.round(scene.getEngine().getFps());\r\n\r\n if ((this._improvementMode && this._currentFrameRate <= this._targetFrameRate) || (!this._improvementMode && this._currentFrameRate >= this._targetFrameRate)) {\r\n this._isRunning = false;\r\n this.onSuccessObservable.notifyObservers(this);\r\n return;\r\n }\r\n\r\n // Apply current level of optimizations\r\n let allDone = true;\r\n let noOptimizationApplied = true;\r\n for (let index = 0; index < options.optimizations.length; index++) {\r\n const optimization = options.optimizations[index];\r\n\r\n if (optimization.priority === this._currentPriorityLevel) {\r\n noOptimizationApplied = false;\r\n allDone = allDone && optimization.apply(scene, this);\r\n this.onNewOptimizationAppliedObservable.notifyObservers(optimization);\r\n }\r\n }\r\n\r\n // If no optimization was applied, this is a failure :(\r\n if (noOptimizationApplied) {\r\n this._isRunning = false;\r\n this.onFailureObservable.notifyObservers(this);\r\n\r\n return;\r\n }\r\n\r\n // If all optimizations were done, move to next level\r\n if (allDone) {\r\n this._currentPriorityLevel++;\r\n }\r\n\r\n // Let's the system running for a specific amount of time before checking FPS\r\n scene.executeWhenReady(() => {\r\n setTimeout(() => {\r\n this._checkCurrentState();\r\n }, this._trackerDuration);\r\n });\r\n }\r\n\r\n /**\r\n * Release all resources\r\n */\r\n public dispose(): void {\r\n this.stop();\r\n this.onSuccessObservable.clear();\r\n this.onFailureObservable.clear();\r\n this.onNewOptimizationAppliedObservable.clear();\r\n if (this._sceneDisposeObserver) {\r\n this._scene.onDisposeObservable.remove(this._sceneDisposeObserver);\r\n }\r\n }\r\n\r\n /**\r\n * Helper function to create a SceneOptimizer with one single line of code\r\n * @param scene defines the scene to work on\r\n * @param options defines the options to use with the SceneOptimizer\r\n * @param onSuccess defines a callback to call on success\r\n * @param onFailure defines a callback to call on failure\r\n * @returns the new SceneOptimizer object\r\n */\r\n public static OptimizeAsync(scene: Scene, options?: SceneOptimizerOptions, onSuccess?: () => void, onFailure?: () => void): SceneOptimizer {\r\n const optimizer = new SceneOptimizer(scene, options || SceneOptimizerOptions.ModerateDegradationAllowed(), false);\r\n\r\n if (onSuccess) {\r\n optimizer.onSuccessObservable.add(() => {\r\n onSuccess();\r\n });\r\n }\r\n\r\n if (onFailure) {\r\n optimizer.onFailureObservable.add(() => {\r\n onFailure();\r\n });\r\n }\r\n\r\n optimizer.start();\r\n\r\n return optimizer;\r\n }\r\n}\r\n","import type { Geometry } from \"../Meshes/geometry\";\r\nimport { Mesh } from \"../Meshes/mesh\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport { MultiMaterial } from \"../Materials/multiMaterial\";\r\nimport type { Material } from \"../Materials/material\";\r\nimport type { Scene } from \"../scene\";\r\nimport type { Light } from \"../Lights/light\";\r\nimport { SerializationHelper } from \"./decorators\";\r\nimport { Texture } from \"../Materials/Textures/texture\";\r\nimport type { CubeTexture } from \"../Materials/Textures/cubeTexture\";\r\nimport type { Node } from \"../node\";\r\nimport type { TransformNode } from \"../Meshes/transformNode\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\n\r\nlet serializedGeometries: Geometry[] = [];\r\nconst SerializeGeometry = (geometry: Geometry, serializationGeometries: any): any => {\r\n if (geometry.doNotSerialize) {\r\n return;\r\n }\r\n\r\n serializationGeometries.vertexData.push(geometry.serializeVerticeData());\r\n\r\n (serializedGeometries)[geometry.id] = true;\r\n};\r\n\r\nconst SerializeMesh = (mesh: Mesh, serializationScene: any): any => {\r\n const serializationObject: any = {};\r\n\r\n // Geometry\r\n const geometry = mesh._geometry;\r\n if (geometry) {\r\n if (!mesh.getScene().getGeometryById(geometry.id)) {\r\n // Geometry was in the memory but not added to the scene, nevertheless it's better to serialize to be able to reload the mesh with its geometry\r\n SerializeGeometry(geometry, serializationScene.geometries);\r\n }\r\n }\r\n\r\n // Custom\r\n if (mesh.serialize) {\r\n mesh.serialize(serializationObject);\r\n }\r\n\r\n return serializationObject;\r\n};\r\n\r\nconst FinalizeSingleNode = (node: Node, serializationObject: any) => {\r\n if ((node as Mesh)._isMesh) {\r\n const mesh = node as Mesh;\r\n //only works if the mesh is already loaded\r\n if (mesh.delayLoadState === Constants.DELAYLOADSTATE_LOADED || mesh.delayLoadState === Constants.DELAYLOADSTATE_NONE) {\r\n const serializeMaterial = (material: Material) => {\r\n serializationObject.materials = serializationObject.materials || [];\r\n if (mesh.material && !serializationObject.materials.some((mat: Material) => mat.id === (mesh.material).id)) {\r\n serializationObject.materials.push(material.serialize());\r\n }\r\n };\r\n\r\n //serialize material\r\n if (mesh.material && !mesh.material.doNotSerialize) {\r\n if (mesh.material instanceof MultiMaterial) {\r\n serializationObject.multiMaterials = serializationObject.multiMaterials || [];\r\n if (!serializationObject.multiMaterials.some((mat: Material) => mat.id === (mesh.material).id)) {\r\n serializationObject.multiMaterials.push(mesh.material.serialize());\r\n for (const submaterial of mesh.material.subMaterials) {\r\n if (submaterial) {\r\n serializeMaterial(submaterial);\r\n }\r\n }\r\n }\r\n } else {\r\n serializeMaterial(mesh.material);\r\n }\r\n } else if (!mesh.material) {\r\n serializeMaterial(mesh.getScene().defaultMaterial);\r\n }\r\n\r\n //serialize geometry\r\n const geometry = mesh._geometry;\r\n if (geometry) {\r\n if (!serializationObject.geometries) {\r\n serializationObject.geometries = {};\r\n\r\n serializationObject.geometries.boxes = [];\r\n serializationObject.geometries.spheres = [];\r\n serializationObject.geometries.cylinders = [];\r\n serializationObject.geometries.toruses = [];\r\n serializationObject.geometries.grounds = [];\r\n serializationObject.geometries.planes = [];\r\n serializationObject.geometries.torusKnots = [];\r\n serializationObject.geometries.vertexData = [];\r\n }\r\n\r\n SerializeGeometry(geometry, serializationObject.geometries);\r\n }\r\n // Skeletons\r\n if (mesh.skeleton && !mesh.skeleton.doNotSerialize) {\r\n serializationObject.skeletons = serializationObject.skeletons || [];\r\n serializationObject.skeletons.push(mesh.skeleton.serialize());\r\n }\r\n\r\n //serialize the actual mesh\r\n serializationObject.meshes = serializationObject.meshes || [];\r\n serializationObject.meshes.push(SerializeMesh(mesh, serializationObject));\r\n }\r\n } else if (node.getClassName() === \"TransformNode\") {\r\n const transformNode = node as TransformNode;\r\n serializationObject.transformNodes.push(transformNode.serialize());\r\n } else if (node.getClassName().indexOf(\"Camera\") !== -1) {\r\n const camera = node as Camera;\r\n serializationObject.cameras.push(camera.serialize());\r\n } else if (node.getClassName().indexOf(\"Light\") !== -1) {\r\n const light = node as Light;\r\n serializationObject.lights.push(light.serialize());\r\n }\r\n};\r\n\r\n/**\r\n * Class used to serialize a scene into a string\r\n */\r\nexport class SceneSerializer {\r\n /**\r\n * Clear cache used by a previous serialization\r\n */\r\n public static ClearCache(): void {\r\n serializedGeometries = [];\r\n }\r\n\r\n /**\r\n * Serialize a scene into a JSON compatible object\r\n * Note that if the current engine does not support synchronous texture reading (like WebGPU), you should use SerializeAsync instead\r\n * as else you may not retrieve the proper base64 encoded texture data (when using the Texture.ForceSerializeBuffers flag)\r\n * @param scene defines the scene to serialize\r\n * @returns a JSON compatible object\r\n */\r\n public static Serialize(scene: Scene): any {\r\n return SceneSerializer._Serialize(scene);\r\n }\r\n\r\n private static _Serialize(scene: Scene, checkSyncReadSupported = true): any {\r\n const serializationObject: any = {};\r\n\r\n if (checkSyncReadSupported && !scene.getEngine()._features.supportSyncTextureRead && Texture.ForceSerializeBuffers) {\r\n console.warn(\"The serialization object may not contain the proper base64 encoded texture data! You should use the SerializeAsync method instead.\");\r\n }\r\n\r\n SceneSerializer.ClearCache();\r\n\r\n // Scene\r\n serializationObject.useDelayedTextureLoading = scene.useDelayedTextureLoading;\r\n serializationObject.autoClear = scene.autoClear;\r\n serializationObject.clearColor = scene.clearColor.asArray();\r\n serializationObject.ambientColor = scene.ambientColor.asArray();\r\n serializationObject.gravity = scene.gravity.asArray();\r\n serializationObject.collisionsEnabled = scene.collisionsEnabled;\r\n serializationObject.useRightHandedSystem = scene.useRightHandedSystem;\r\n\r\n // Fog\r\n if (scene.fogMode && scene.fogMode !== 0) {\r\n serializationObject.fogMode = scene.fogMode;\r\n serializationObject.fogColor = scene.fogColor.asArray();\r\n serializationObject.fogStart = scene.fogStart;\r\n serializationObject.fogEnd = scene.fogEnd;\r\n serializationObject.fogDensity = scene.fogDensity;\r\n }\r\n\r\n //Physics\r\n if (scene.isPhysicsEnabled && scene.isPhysicsEnabled()) {\r\n const physicEngine = scene.getPhysicsEngine();\r\n\r\n if (physicEngine) {\r\n serializationObject.physicsEnabled = true;\r\n serializationObject.physicsGravity = physicEngine.gravity.asArray();\r\n serializationObject.physicsEngine = physicEngine.getPhysicsPluginName();\r\n }\r\n }\r\n\r\n // Metadata\r\n if (scene.metadata) {\r\n serializationObject.metadata = scene.metadata;\r\n }\r\n\r\n // Morph targets\r\n serializationObject.morphTargetManagers = [];\r\n for (const abstractMesh of scene.meshes) {\r\n const manager = (abstractMesh).morphTargetManager;\r\n\r\n if (manager) {\r\n serializationObject.morphTargetManagers.push(manager.serialize());\r\n }\r\n }\r\n\r\n // Lights\r\n serializationObject.lights = [];\r\n let index: number;\r\n let light: Light;\r\n for (index = 0; index < scene.lights.length; index++) {\r\n light = scene.lights[index];\r\n\r\n if (!light.doNotSerialize) {\r\n serializationObject.lights.push(light.serialize());\r\n }\r\n }\r\n\r\n // Cameras\r\n serializationObject.cameras = [];\r\n for (index = 0; index < scene.cameras.length; index++) {\r\n const camera = scene.cameras[index];\r\n\r\n if (!camera.doNotSerialize) {\r\n serializationObject.cameras.push(camera.serialize());\r\n }\r\n }\r\n\r\n if (scene.activeCamera) {\r\n serializationObject.activeCameraID = scene.activeCamera.id;\r\n }\r\n\r\n // Animations\r\n SerializationHelper.AppendSerializedAnimations(scene, serializationObject);\r\n\r\n // Animation Groups\r\n if (scene.animationGroups && scene.animationGroups.length > 0) {\r\n serializationObject.animationGroups = [];\r\n for (let animationGroupIndex = 0; animationGroupIndex < scene.animationGroups.length; animationGroupIndex++) {\r\n const animationGroup = scene.animationGroups[animationGroupIndex];\r\n\r\n serializationObject.animationGroups.push(animationGroup.serialize());\r\n }\r\n }\r\n\r\n // Reflection probes\r\n if (scene.reflectionProbes && scene.reflectionProbes.length > 0) {\r\n serializationObject.reflectionProbes = [];\r\n\r\n for (index = 0; index < scene.reflectionProbes.length; index++) {\r\n const reflectionProbe = scene.reflectionProbes[index];\r\n serializationObject.reflectionProbes.push(reflectionProbe.serialize());\r\n }\r\n }\r\n\r\n // Materials\r\n serializationObject.materials = [];\r\n serializationObject.multiMaterials = [];\r\n let material: Material;\r\n for (index = 0; index < scene.materials.length; index++) {\r\n material = scene.materials[index];\r\n if (!material.doNotSerialize) {\r\n serializationObject.materials.push(material.serialize());\r\n }\r\n }\r\n\r\n // MultiMaterials\r\n serializationObject.multiMaterials = [];\r\n for (index = 0; index < scene.multiMaterials.length; index++) {\r\n const multiMaterial = scene.multiMaterials[index];\r\n serializationObject.multiMaterials.push(multiMaterial.serialize());\r\n }\r\n\r\n // Environment texture\r\n if (scene.environmentTexture) {\r\n if ((scene.environmentTexture as CubeTexture)._files) {\r\n serializationObject.environmentTexture = scene.environmentTexture.serialize();\r\n } else {\r\n serializationObject.environmentTexture = scene.environmentTexture.name;\r\n serializationObject.environmentTextureRotationY = (scene.environmentTexture as CubeTexture).rotationY;\r\n }\r\n }\r\n\r\n // Environment Intensity\r\n serializationObject.environmentIntensity = scene.environmentIntensity;\r\n\r\n // Skeletons\r\n serializationObject.skeletons = [];\r\n for (index = 0; index < scene.skeletons.length; index++) {\r\n const skeleton = scene.skeletons[index];\r\n if (!skeleton.doNotSerialize) {\r\n serializationObject.skeletons.push(skeleton.serialize());\r\n }\r\n }\r\n\r\n // Transform nodes\r\n serializationObject.transformNodes = [];\r\n for (index = 0; index < scene.transformNodes.length; index++) {\r\n if (!scene.transformNodes[index].doNotSerialize) {\r\n serializationObject.transformNodes.push(scene.transformNodes[index].serialize());\r\n }\r\n }\r\n\r\n // Geometries\r\n serializationObject.geometries = {};\r\n\r\n serializationObject.geometries.boxes = [];\r\n serializationObject.geometries.spheres = [];\r\n serializationObject.geometries.cylinders = [];\r\n serializationObject.geometries.toruses = [];\r\n serializationObject.geometries.grounds = [];\r\n serializationObject.geometries.planes = [];\r\n serializationObject.geometries.torusKnots = [];\r\n serializationObject.geometries.vertexData = [];\r\n\r\n serializedGeometries = [];\r\n const geometries = scene.getGeometries();\r\n for (index = 0; index < geometries.length; index++) {\r\n const geometry = geometries[index];\r\n\r\n if (geometry.isReady()) {\r\n SerializeGeometry(geometry, serializationObject.geometries);\r\n }\r\n }\r\n\r\n // Meshes\r\n serializationObject.meshes = [];\r\n for (index = 0; index < scene.meshes.length; index++) {\r\n const abstractMesh = scene.meshes[index];\r\n\r\n if (abstractMesh instanceof Mesh) {\r\n const mesh = abstractMesh;\r\n if (!mesh.doNotSerialize) {\r\n if (mesh.delayLoadState === Constants.DELAYLOADSTATE_LOADED || mesh.delayLoadState === Constants.DELAYLOADSTATE_NONE) {\r\n serializationObject.meshes.push(SerializeMesh(mesh, serializationObject));\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Particles Systems\r\n serializationObject.particleSystems = [];\r\n for (index = 0; index < scene.particleSystems.length; index++) {\r\n serializationObject.particleSystems.push(scene.particleSystems[index].serialize(false));\r\n }\r\n\r\n // Post processes\r\n serializationObject.postProcesses = [];\r\n for (index = 0; index < scene.postProcesses.length; index++) {\r\n serializationObject.postProcesses.push(scene.postProcesses[index].serialize());\r\n }\r\n\r\n // Action Manager\r\n if (scene.actionManager) {\r\n serializationObject.actions = scene.actionManager.serialize(\"scene\");\r\n }\r\n\r\n // Components\r\n for (const component of scene._serializableComponents) {\r\n component.serialize(serializationObject);\r\n }\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Serialize a scene into a JSON compatible object\r\n * @param scene defines the scene to serialize\r\n * @returns a JSON promise compatible object\r\n */\r\n public static SerializeAsync(scene: Scene): Promise {\r\n const serializationObject = SceneSerializer._Serialize(scene, false);\r\n\r\n const promises: Array> = [];\r\n\r\n this._CollectPromises(serializationObject, promises);\r\n\r\n return Promise.all(promises).then(() => serializationObject);\r\n }\r\n\r\n private static _CollectPromises(obj: any, promises: Array>): void {\r\n if (Array.isArray(obj)) {\r\n for (let i = 0; i < obj.length; ++i) {\r\n const o = obj[i];\r\n if (o instanceof Promise) {\r\n promises.push(o.then((res: any) => (obj[i] = res)));\r\n } else if (o instanceof Object || Array.isArray(o)) {\r\n this._CollectPromises(o, promises);\r\n }\r\n }\r\n } else if (obj instanceof Object) {\r\n for (const name in obj) {\r\n if (Object.prototype.hasOwnProperty.call(obj, name)) {\r\n const o = obj[name];\r\n if (o instanceof Promise) {\r\n promises.push(o.then((res: any) => (obj[name] = res)));\r\n } else if (o instanceof Object || Array.isArray(o)) {\r\n this._CollectPromises(o, promises);\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Serialize a mesh into a JSON compatible object\r\n * @param toSerialize defines the mesh to serialize\r\n * @param withParents defines if parents must be serialized as well\r\n * @param withChildren defines if children must be serialized as well\r\n * @returns a JSON compatible object\r\n */\r\n public static SerializeMesh(toSerialize: any /* Mesh || Mesh[] */, withParents: boolean = false, withChildren: boolean = false): any {\r\n const serializationObject: any = {};\r\n serializationObject.meshes = [];\r\n serializationObject.transformNodes = [];\r\n serializationObject.cameras = [];\r\n serializationObject.lights = [];\r\n\r\n SceneSerializer.ClearCache();\r\n\r\n toSerialize = toSerialize instanceof Array ? toSerialize : [toSerialize];\r\n\r\n if (withParents || withChildren) {\r\n //deliberate for loop! not for each, appended should be processed as well.\r\n for (let i = 0; i < toSerialize.length; ++i) {\r\n if (withChildren) {\r\n toSerialize[i].getDescendants().forEach((node: Node) => {\r\n if (toSerialize.indexOf(node) < 0 && !node.doNotSerialize) {\r\n toSerialize.push(node);\r\n }\r\n });\r\n }\r\n //make sure the array doesn't contain the object already\r\n if (withParents && toSerialize[i].parent && toSerialize.indexOf(toSerialize[i].parent) < 0 && !toSerialize[i].parent.doNotSerialize) {\r\n toSerialize.push(toSerialize[i].parent);\r\n }\r\n }\r\n }\r\n\r\n toSerialize.forEach((mesh: Node) => {\r\n FinalizeSingleNode(mesh, serializationObject);\r\n });\r\n\r\n return serializationObject;\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\nimport { Scene } from \"../scene\";\r\nimport type { SmartArrayNoDuplicate } from \"../Misc/smartArray\";\r\nimport { DepthRenderer } from \"./depthRenderer\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport type { ISceneComponent } from \"../sceneComponent\";\r\nimport { SceneComponentConstants } from \"../sceneComponent\";\r\nimport type { RenderTargetTexture } from \"../Materials/Textures/renderTargetTexture\";\r\n\r\ndeclare module \"../scene\" {\r\n export interface Scene {\r\n /** @internal (Backing field) */\r\n _depthRenderer: { [id: string]: DepthRenderer };\r\n\r\n /**\r\n * Creates a depth renderer a given camera which contains a depth map which can be used for post processing.\r\n * @param camera The camera to create the depth renderer on (default: scene's active camera)\r\n * @param storeNonLinearDepth Defines whether the depth is stored linearly like in Babylon Shadows or directly like glFragCoord.z\r\n * @param force32bitsFloat Forces 32 bits float when supported (else 16 bits float is prioritized over 32 bits float if supported)\r\n * @param samplingMode The sampling mode to be used with the render target (Linear, Nearest...)\r\n * @param storeCameraSpaceZ Defines whether the depth stored is the Z coordinate in camera space. If true, storeNonLinearDepth has no effect. (Default: false)\r\n * @returns the created depth renderer\r\n */\r\n enableDepthRenderer(\r\n camera?: Nullable,\r\n storeNonLinearDepth?: boolean,\r\n force32bitsFloat?: boolean,\r\n samplingMode?: number,\r\n storeCameraSpaceZ?: boolean\r\n ): DepthRenderer;\r\n\r\n /**\r\n * Disables a depth renderer for a given camera\r\n * @param camera The camera to disable the depth renderer on (default: scene's active camera)\r\n */\r\n disableDepthRenderer(camera?: Nullable): void;\r\n }\r\n}\r\n\r\nScene.prototype.enableDepthRenderer = function (\r\n camera?: Nullable,\r\n storeNonLinearDepth = false,\r\n force32bitsFloat: boolean = false,\r\n samplingMode = Constants.TEXTURE_TRILINEAR_SAMPLINGMODE,\r\n storeCameraSpaceZ: boolean = false\r\n): DepthRenderer {\r\n camera = camera || this.activeCamera;\r\n if (!camera) {\r\n throw \"No camera available to enable depth renderer\";\r\n }\r\n if (!this._depthRenderer) {\r\n this._depthRenderer = {};\r\n }\r\n if (!this._depthRenderer[camera.id]) {\r\n const supportFullfloat = !!this.getEngine().getCaps().textureFloatRender;\r\n let textureType = 0;\r\n if (this.getEngine().getCaps().textureHalfFloatRender && (!force32bitsFloat || !supportFullfloat)) {\r\n textureType = Constants.TEXTURETYPE_HALF_FLOAT;\r\n } else if (supportFullfloat) {\r\n textureType = Constants.TEXTURETYPE_FLOAT;\r\n } else {\r\n textureType = Constants.TEXTURETYPE_UNSIGNED_BYTE;\r\n }\r\n this._depthRenderer[camera.id] = new DepthRenderer(this, textureType, camera, storeNonLinearDepth, samplingMode, storeCameraSpaceZ);\r\n }\r\n\r\n return this._depthRenderer[camera.id];\r\n};\r\n\r\nScene.prototype.disableDepthRenderer = function (camera?: Nullable): void {\r\n camera = camera || this.activeCamera;\r\n if (!camera || !this._depthRenderer || !this._depthRenderer[camera.id]) {\r\n return;\r\n }\r\n\r\n this._depthRenderer[camera.id].dispose();\r\n};\r\n\r\n/**\r\n * Defines the Depth Renderer scene component responsible to manage a depth buffer useful\r\n * in several rendering techniques.\r\n */\r\nexport class DepthRendererSceneComponent implements ISceneComponent {\r\n /**\r\n * The component name helpful to identify the component in the list of scene components.\r\n */\r\n public readonly name = SceneComponentConstants.NAME_DEPTHRENDERER;\r\n\r\n /**\r\n * The scene the component belongs to.\r\n */\r\n public scene: Scene;\r\n\r\n /**\r\n * Creates a new instance of the component for the given scene\r\n * @param scene Defines the scene to register the component in\r\n */\r\n constructor(scene: Scene) {\r\n this.scene = scene;\r\n }\r\n\r\n /**\r\n * Registers the component in a given scene\r\n */\r\n public register(): void {\r\n this.scene._gatherRenderTargetsStage.registerStep(SceneComponentConstants.STEP_GATHERRENDERTARGETS_DEPTHRENDERER, this, this._gatherRenderTargets);\r\n this.scene._gatherActiveCameraRenderTargetsStage.registerStep(\r\n SceneComponentConstants.STEP_GATHERACTIVECAMERARENDERTARGETS_DEPTHRENDERER,\r\n this,\r\n this._gatherActiveCameraRenderTargets\r\n );\r\n }\r\n\r\n /**\r\n * Rebuilds the elements related to this component in case of\r\n * context lost for instance.\r\n */\r\n public rebuild(): void {\r\n // Nothing to do for this component\r\n }\r\n\r\n /**\r\n * Disposes the component and the associated resources\r\n */\r\n public dispose(): void {\r\n for (const key in this.scene._depthRenderer) {\r\n this.scene._depthRenderer[key].dispose();\r\n }\r\n }\r\n\r\n private _gatherRenderTargets(renderTargets: SmartArrayNoDuplicate): void {\r\n if (this.scene._depthRenderer) {\r\n for (const key in this.scene._depthRenderer) {\r\n const depthRenderer = this.scene._depthRenderer[key];\r\n if (depthRenderer.enabled && !depthRenderer.useOnlyInActiveCamera) {\r\n renderTargets.push(depthRenderer.getDepthMap());\r\n }\r\n }\r\n }\r\n }\r\n\r\n private _gatherActiveCameraRenderTargets(renderTargets: SmartArrayNoDuplicate): void {\r\n if (this.scene._depthRenderer) {\r\n for (const key in this.scene._depthRenderer) {\r\n const depthRenderer = this.scene._depthRenderer[key];\r\n if (depthRenderer.enabled && depthRenderer.useOnlyInActiveCamera && this.scene.activeCamera!.id === key) {\r\n renderTargets.push(depthRenderer.getDepthMap());\r\n }\r\n }\r\n }\r\n }\r\n}\r\n\r\nDepthRenderer._SceneComponentInitialization = (scene: Scene) => {\r\n // Register the G Buffer component to the scene.\r\n let component = scene._getComponent(SceneComponentConstants.NAME_DEPTHRENDERER) as DepthRendererSceneComponent;\r\n if (!component) {\r\n component = new DepthRendererSceneComponent(scene);\r\n scene._addComponent(component);\r\n }\r\n};\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/clipPlaneFragmentDeclaration\";\nimport \"./ShadersInclude/clipPlaneFragment\";\n\nconst name = \"linePixelShader\";\nconst shader = `#include\nuniform vec4 color;\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_FRAGMENT_MAIN_BEGIN\n#include\ngl_FragColor=color;\n#define CUSTOM_FRAGMENT_MAIN_END\n}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const linePixelShader = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/instancesDeclaration\";\nimport \"./ShadersInclude/clipPlaneVertexDeclaration\";\nimport \"./ShadersInclude/instancesVertex\";\nimport \"./ShadersInclude/clipPlaneVertex\";\n\nconst name = \"lineVertexShader\";\nconst shader = `#include\n#include\nattribute vec3 position;attribute vec4 normal;uniform mat4 viewProjection;uniform float width;uniform float aspectRatio;\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\n#include\nmat4 worldViewProjection=viewProjection*finalWorld;vec4 viewPosition=worldViewProjection*vec4(position,1.0);vec4 viewPositionNext=worldViewProjection*vec4(normal.xyz,1.0);vec2 currentScreen=viewPosition.xy/viewPosition.w;vec2 nextScreen=viewPositionNext.xy/viewPositionNext.w;currentScreen.x*=aspectRatio;nextScreen.x*=aspectRatio;vec2 dir=normalize(nextScreen-currentScreen);vec2 normalDir=vec2(-dir.y,dir.x);normalDir*=width/2.0;normalDir.x/=aspectRatio;vec4 offset=vec4(normalDir*normal.w,0.0,0.0);gl_Position=viewPosition+offset;\n#if defined(CLIPPLANE) || defined(CLIPPLANE2) || defined(CLIPPLANE3) || defined(CLIPPLANE4) || defined(CLIPPLANE5) || defined(CLIPPLANE6)\nvec4 worldPos=finalWorld*vec4(position,1.0);\n#include\n#endif\n#define CUSTOM_VERTEX_MAIN_END\n}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const lineVertexShader = { name, shader };\n","import type { Immutable, Nullable } from \"../types\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { Mesh } from \"../Meshes/mesh\";\r\nimport { LinesMesh, InstancedLinesMesh } from \"../Meshes/linesMesh\";\r\nimport type { Matrix } from \"../Maths/math.vector\";\r\nimport { Vector3, TmpVectors } from \"../Maths/math.vector\";\r\nimport type { IDisposable, Scene } from \"../scene\";\r\nimport type { Observer } from \"../Misc/observable\";\r\nimport type { Effect } from \"../Materials/effect\";\r\nimport { Material } from \"../Materials/material\";\r\nimport { ShaderMaterial } from \"../Materials/shaderMaterial\";\r\nimport { Camera } from \"../Cameras/camera\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport type { Node } from \"../node\";\r\n\r\nimport \"../Shaders/line.fragment\";\r\nimport \"../Shaders/line.vertex\";\r\nimport type { DataBuffer } from \"../Buffers/dataBuffer\";\r\nimport { SmartArray } from \"../Misc/smartArray\";\r\nimport { DrawWrapper } from \"../Materials/drawWrapper\";\r\n\r\ndeclare module \"../scene\" {\r\n export interface Scene {\r\n /** @internal */\r\n _edgeRenderLineShader: Nullable;\r\n }\r\n}\r\n\r\ndeclare module \"../Meshes/abstractMesh\" {\r\n export interface AbstractMesh {\r\n /**\r\n * Gets the edgesRenderer associated with the mesh\r\n */\r\n edgesRenderer: Nullable;\r\n }\r\n}\r\nAbstractMesh.prototype.disableEdgesRendering = function (): AbstractMesh {\r\n if (this._edgesRenderer) {\r\n this._edgesRenderer.dispose();\r\n this._edgesRenderer = null;\r\n }\r\n return this;\r\n};\r\n\r\nAbstractMesh.prototype.enableEdgesRendering = function (epsilon = 0.95, checkVerticesInsteadOfIndices = false, options?: IEdgesRendererOptions): AbstractMesh {\r\n this.disableEdgesRendering();\r\n this._edgesRenderer = new EdgesRenderer(this, epsilon, checkVerticesInsteadOfIndices, true, options);\r\n return this;\r\n};\r\n\r\nObject.defineProperty(AbstractMesh.prototype, \"edgesRenderer\", {\r\n get: function (this: AbstractMesh) {\r\n return this._edgesRenderer;\r\n },\r\n enumerable: true,\r\n configurable: true,\r\n});\r\n\r\ndeclare module \"../Meshes/linesMesh\" {\r\n export interface LinesMesh {\r\n /**\r\n * Enables the edge rendering mode on the mesh.\r\n * This mode makes the mesh edges visible\r\n * @param epsilon defines the maximal distance between two angles to detect a face\r\n * @param checkVerticesInsteadOfIndices indicates that we should check vertex list directly instead of faces\r\n * @returns the currentAbstractMesh\r\n * @see https://www.babylonjs-playground.com/#19O9TU#0\r\n */\r\n enableEdgesRendering(epsilon?: number, checkVerticesInsteadOfIndices?: boolean): AbstractMesh;\r\n }\r\n}\r\nLinesMesh.prototype.enableEdgesRendering = function (epsilon = 0.95, checkVerticesInsteadOfIndices = false): AbstractMesh {\r\n this.disableEdgesRendering();\r\n this._edgesRenderer = new LineEdgesRenderer(this, epsilon, checkVerticesInsteadOfIndices);\r\n return this;\r\n};\r\n\r\ndeclare module \"../Meshes/linesMesh\" {\r\n export interface InstancedLinesMesh {\r\n /**\r\n * Enables the edge rendering mode on the mesh.\r\n * This mode makes the mesh edges visible\r\n * @param epsilon defines the maximal distance between two angles to detect a face\r\n * @param checkVerticesInsteadOfIndices indicates that we should check vertex list directly instead of faces\r\n * @returns the current InstancedLinesMesh\r\n * @see https://www.babylonjs-playground.com/#19O9TU#0\r\n */\r\n enableEdgesRendering(epsilon?: number, checkVerticesInsteadOfIndices?: boolean): InstancedLinesMesh;\r\n }\r\n}\r\n\r\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\r\nInstancedLinesMesh.prototype.enableEdgesRendering = function (epsilon = 0.95, checkVerticesInsteadOfIndices = false): InstancedLinesMesh {\r\n LinesMesh.prototype.enableEdgesRendering.apply(this, arguments);\r\n return this;\r\n};\r\n\r\n/**\r\n * FaceAdjacencies Helper class to generate edges\r\n */\r\nclass FaceAdjacencies {\r\n public edges = new Array();\r\n public p0: Vector3;\r\n public p1: Vector3;\r\n public p2: Vector3;\r\n public edgesConnectedCount = 0;\r\n}\r\n\r\n/**\r\n * Defines the minimum contract an Edges renderer should follow.\r\n */\r\nexport interface IEdgesRenderer extends IDisposable {\r\n /**\r\n * Gets or sets a boolean indicating if the edgesRenderer is active\r\n */\r\n isEnabled: boolean;\r\n\r\n /**\r\n * Renders the edges of the attached mesh,\r\n */\r\n render(): void;\r\n\r\n /**\r\n * Checks whether or not the edges renderer is ready to render.\r\n * @returns true if ready, otherwise false.\r\n */\r\n isReady(): boolean;\r\n\r\n /**\r\n * List of instances to render in case the source mesh has instances\r\n */\r\n customInstances: SmartArray;\r\n}\r\n\r\n/**\r\n * Defines the additional options of the edges renderer\r\n */\r\nexport interface IEdgesRendererOptions {\r\n /**\r\n * Gets or sets a boolean indicating that the alternate edge finder algorithm must be used\r\n * If not defined, the default value is true\r\n */\r\n useAlternateEdgeFinder?: boolean;\r\n\r\n /**\r\n * Gets or sets a boolean indicating that the vertex merger fast processing must be used.\r\n * If not defined, the default value is true.\r\n * You should normally leave it undefined (or set it to true), except if you see some artifacts in the edges rendering (can happen with complex geometries)\r\n * This option is used only if useAlternateEdgeFinder = true\r\n */\r\n useFastVertexMerger?: boolean;\r\n\r\n /**\r\n * During edges processing, the vertices are merged if they are close enough: epsilonVertexMerge is the limit within which vertices are considered to be equal.\r\n * The default value is 1e-6\r\n * This option is used only if useAlternateEdgeFinder = true\r\n */\r\n epsilonVertexMerge?: number;\r\n\r\n /**\r\n * Gets or sets a boolean indicating that tessellation should be applied before finding the edges. You may need to activate this option if your geometry is a bit\r\n * unusual, like having a vertex of a triangle in-between two vertices of an edge of another triangle. It happens often when using CSG to construct meshes.\r\n * This option is used only if useAlternateEdgeFinder = true\r\n */\r\n applyTessellation?: boolean;\r\n\r\n /**\r\n * The limit under which 3 vertices are considered to be aligned. 3 vertices PQR are considered aligned if distance(PQ) + distance(QR) - distance(PR) < epsilonVertexAligned\r\n * The default value is 1e-6\r\n * This option is used only if useAlternateEdgeFinder = true\r\n */\r\n epsilonVertexAligned?: number;\r\n\r\n /**\r\n * Gets or sets a boolean indicating that degenerated triangles should not be processed.\r\n * Degenerated triangles are triangles that have 2 or 3 vertices with the same coordinates\r\n */\r\n removeDegeneratedTriangles?: boolean;\r\n}\r\n\r\n/**\r\n * This class is used to generate edges of the mesh that could then easily be rendered in a scene.\r\n */\r\nexport class EdgesRenderer implements IEdgesRenderer {\r\n /**\r\n * Define the size of the edges with an orthographic camera\r\n */\r\n public edgesWidthScalerForOrthographic = 1000.0;\r\n\r\n /**\r\n * Define the size of the edges with a perspective camera\r\n */\r\n public edgesWidthScalerForPerspective = 50.0;\r\n\r\n protected _source: AbstractMesh;\r\n protected _linesPositions = new Array();\r\n protected _linesNormals = new Array();\r\n protected _linesIndices = new Array();\r\n protected _epsilon: number;\r\n protected _indicesCount: number;\r\n protected _drawWrapper?: DrawWrapper;\r\n\r\n protected _lineShader: ShaderMaterial;\r\n protected _ib: DataBuffer;\r\n protected _buffers: { [key: string]: Nullable } = {};\r\n protected _buffersForInstances: { [key: string]: Nullable } = {};\r\n protected _checkVerticesInsteadOfIndices = false;\r\n protected _options: Nullable;\r\n\r\n private _meshRebuildObserver: Nullable>;\r\n private _meshDisposeObserver: Nullable>;\r\n\r\n /** Gets or sets a boolean indicating if the edgesRenderer is active */\r\n public isEnabled = true;\r\n\r\n /** Gets the vertices generated by the edge renderer */\r\n public get linesPositions(): Immutable> {\r\n return this._linesPositions;\r\n }\r\n\r\n /** Gets the normals generated by the edge renderer */\r\n public get linesNormals(): Immutable> {\r\n return this._linesNormals;\r\n }\r\n\r\n /** Gets the indices generated by the edge renderer */\r\n public get linesIndices(): Immutable> {\r\n return this._linesIndices;\r\n }\r\n\r\n /**\r\n * Gets or sets the shader used to draw the lines\r\n */\r\n public get lineShader(): ShaderMaterial {\r\n return this._lineShader;\r\n }\r\n\r\n public set lineShader(shader: ShaderMaterial) {\r\n this._lineShader = shader;\r\n }\r\n\r\n /**\r\n * List of instances to render in case the source mesh has instances\r\n */\r\n public customInstances = new SmartArray(32);\r\n\r\n private static _GetShader(scene: Scene): ShaderMaterial {\r\n if (!scene._edgeRenderLineShader) {\r\n const shader = new ShaderMaterial(\r\n \"lineShader\",\r\n scene,\r\n \"line\",\r\n {\r\n attributes: [\"position\", \"normal\"],\r\n uniforms: [\"world\", \"viewProjection\", \"color\", \"width\", \"aspectRatio\"],\r\n },\r\n false\r\n );\r\n\r\n shader.disableDepthWrite = true;\r\n shader.backFaceCulling = false;\r\n shader.checkReadyOnEveryCall = scene.getEngine().isWebGPU;\r\n\r\n scene._edgeRenderLineShader = shader;\r\n }\r\n\r\n return scene._edgeRenderLineShader;\r\n }\r\n\r\n /**\r\n * Creates an instance of the EdgesRenderer. It is primarily use to display edges of a mesh.\r\n * Beware when you use this class with complex objects as the adjacencies computation can be really long\r\n * @param source Mesh used to create edges\r\n * @param epsilon sum of angles in adjacency to check for edge\r\n * @param checkVerticesInsteadOfIndices bases the edges detection on vertices vs indices. Note that this parameter is not used if options.useAlternateEdgeFinder = true\r\n * @param generateEdgesLines - should generate Lines or only prepare resources.\r\n * @param options The options to apply when generating the edges\r\n */\r\n constructor(source: AbstractMesh, epsilon = 0.95, checkVerticesInsteadOfIndices = false, generateEdgesLines = true, options?: IEdgesRendererOptions) {\r\n this._source = source;\r\n this._checkVerticesInsteadOfIndices = checkVerticesInsteadOfIndices;\r\n this._options = options ?? null;\r\n\r\n this._epsilon = epsilon;\r\n if (this._source.getScene().getEngine().isWebGPU) {\r\n this._drawWrapper = new DrawWrapper(source.getEngine());\r\n }\r\n\r\n this._prepareRessources();\r\n if (generateEdgesLines) {\r\n if (options?.useAlternateEdgeFinder ?? true) {\r\n this._generateEdgesLinesAlternate();\r\n } else {\r\n this._generateEdgesLines();\r\n }\r\n }\r\n\r\n this._meshRebuildObserver = this._source.onRebuildObservable.add(() => {\r\n this._rebuild();\r\n });\r\n\r\n this._meshDisposeObserver = this._source.onDisposeObservable.add(() => {\r\n this.dispose();\r\n });\r\n }\r\n\r\n protected _prepareRessources(): void {\r\n if (this._lineShader) {\r\n return;\r\n }\r\n\r\n this._lineShader = EdgesRenderer._GetShader(this._source.getScene());\r\n }\r\n\r\n /** @internal */\r\n public _rebuild(): void {\r\n let buffer = this._buffers[VertexBuffer.PositionKind];\r\n if (buffer) {\r\n buffer._rebuild();\r\n }\r\n\r\n buffer = this._buffers[VertexBuffer.NormalKind];\r\n if (buffer) {\r\n buffer._rebuild();\r\n }\r\n\r\n const scene = this._source.getScene();\r\n const engine = scene.getEngine();\r\n this._ib = engine.createIndexBuffer(this._linesIndices);\r\n }\r\n\r\n /**\r\n * Releases the required resources for the edges renderer\r\n */\r\n public dispose(): void {\r\n this._source.onRebuildObservable.remove(this._meshRebuildObserver);\r\n this._source.onDisposeObservable.remove(this._meshDisposeObserver);\r\n\r\n let buffer = this._buffers[VertexBuffer.PositionKind];\r\n if (buffer) {\r\n buffer.dispose();\r\n this._buffers[VertexBuffer.PositionKind] = null;\r\n }\r\n buffer = this._buffers[VertexBuffer.NormalKind];\r\n if (buffer) {\r\n buffer.dispose();\r\n this._buffers[VertexBuffer.NormalKind] = null;\r\n }\r\n\r\n if (this._ib) {\r\n this._source.getScene().getEngine()._releaseBuffer(this._ib);\r\n }\r\n this._lineShader.dispose();\r\n\r\n this._drawWrapper?.dispose();\r\n }\r\n\r\n protected _processEdgeForAdjacencies(pa: number, pb: number, p0: number, p1: number, p2: number): number {\r\n if ((pa === p0 && pb === p1) || (pa === p1 && pb === p0)) {\r\n return 0;\r\n }\r\n\r\n if ((pa === p1 && pb === p2) || (pa === p2 && pb === p1)) {\r\n return 1;\r\n }\r\n\r\n if ((pa === p2 && pb === p0) || (pa === p0 && pb === p2)) {\r\n return 2;\r\n }\r\n\r\n return -1;\r\n }\r\n\r\n protected _processEdgeForAdjacenciesWithVertices(pa: Vector3, pb: Vector3, p0: Vector3, p1: Vector3, p2: Vector3): number {\r\n const eps = 1e-10;\r\n if ((pa.equalsWithEpsilon(p0, eps) && pb.equalsWithEpsilon(p1, eps)) || (pa.equalsWithEpsilon(p1, eps) && pb.equalsWithEpsilon(p0, eps))) {\r\n return 0;\r\n }\r\n\r\n if ((pa.equalsWithEpsilon(p1, eps) && pb.equalsWithEpsilon(p2, eps)) || (pa.equalsWithEpsilon(p2, eps) && pb.equalsWithEpsilon(p1, eps))) {\r\n return 1;\r\n }\r\n\r\n if ((pa.equalsWithEpsilon(p2, eps) && pb.equalsWithEpsilon(p0, eps)) || (pa.equalsWithEpsilon(p0, eps) && pb.equalsWithEpsilon(p2, eps))) {\r\n return 2;\r\n }\r\n\r\n return -1;\r\n }\r\n\r\n /**\r\n * Checks if the pair of p0 and p1 is en edge\r\n * @param faceIndex\r\n * @param edge\r\n * @param faceNormals\r\n * @param p0\r\n * @param p1\r\n * @private\r\n */\r\n protected _checkEdge(faceIndex: number, edge: number, faceNormals: Array, p0: Vector3, p1: Vector3): void {\r\n let needToCreateLine;\r\n\r\n if (edge === undefined) {\r\n needToCreateLine = true;\r\n } else {\r\n const dotProduct = Vector3.Dot(faceNormals[faceIndex], faceNormals[edge]);\r\n\r\n needToCreateLine = dotProduct < this._epsilon;\r\n }\r\n\r\n if (needToCreateLine) {\r\n this.createLine(p0, p1, this._linesPositions.length / 3);\r\n }\r\n }\r\n\r\n /**\r\n * push line into the position, normal and index buffer\r\n * @param p0\r\n * @param p1\r\n * @param offset\r\n * @protected\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n protected createLine(p0: Vector3, p1: Vector3, offset: number) {\r\n // Positions\r\n this._linesPositions.push(p0.x, p0.y, p0.z, p0.x, p0.y, p0.z, p1.x, p1.y, p1.z, p1.x, p1.y, p1.z);\r\n\r\n // Normals\r\n this._linesNormals.push(p1.x, p1.y, p1.z, -1, p1.x, p1.y, p1.z, 1, p0.x, p0.y, p0.z, -1, p0.x, p0.y, p0.z, 1);\r\n\r\n // Indices\r\n this._linesIndices.push(offset, offset + 1, offset + 2, offset, offset + 2, offset + 3);\r\n }\r\n\r\n /**\r\n * See https://playground.babylonjs.com/#R3JR6V#1 for a visual display of the algorithm\r\n * @param edgePoints\r\n * @param indexTriangle\r\n * @param indices\r\n * @param remapVertexIndices\r\n */\r\n private _tessellateTriangle(edgePoints: Array>, indexTriangle: number, indices: Array, remapVertexIndices: Array): void {\r\n const makePointList = (edgePoints: Array<[number, number]>, pointIndices: Array, firstIndex: number) => {\r\n if (firstIndex >= 0) {\r\n pointIndices.push(firstIndex);\r\n }\r\n\r\n for (let i = 0; i < edgePoints.length; ++i) {\r\n pointIndices.push(edgePoints[i][0]);\r\n }\r\n };\r\n\r\n let startEdge = 0;\r\n\r\n if (edgePoints[1].length >= edgePoints[0].length && edgePoints[1].length >= edgePoints[2].length) {\r\n startEdge = 1;\r\n } else if (edgePoints[2].length >= edgePoints[0].length && edgePoints[2].length >= edgePoints[1].length) {\r\n startEdge = 2;\r\n }\r\n\r\n for (let e = 0; e < 3; ++e) {\r\n if (e === startEdge) {\r\n edgePoints[e].sort((a, b) => (a[1] < b[1] ? -1 : a[1] > b[1] ? 1 : 0));\r\n } else {\r\n edgePoints[e].sort((a, b) => (a[1] > b[1] ? -1 : a[1] < b[1] ? 1 : 0));\r\n }\r\n }\r\n\r\n const mainPointIndices: Array = [],\r\n otherPointIndices: Array = [];\r\n\r\n makePointList(edgePoints[startEdge], mainPointIndices, -1);\r\n\r\n const numMainPoints = mainPointIndices.length;\r\n\r\n for (let i = startEdge + 2; i >= startEdge + 1; --i) {\r\n makePointList(edgePoints[i % 3], otherPointIndices, i !== startEdge + 2 ? remapVertexIndices[indices[indexTriangle + ((i + 1) % 3)]] : -1);\r\n }\r\n\r\n const numOtherPoints = otherPointIndices.length;\r\n\r\n const idxMain = 0;\r\n const idxOther = 0;\r\n\r\n indices.push(remapVertexIndices[indices[indexTriangle + startEdge]], mainPointIndices[0], otherPointIndices[0]);\r\n indices.push(remapVertexIndices[indices[indexTriangle + ((startEdge + 1) % 3)]], otherPointIndices[numOtherPoints - 1], mainPointIndices[numMainPoints - 1]);\r\n\r\n const bucketIsMain = numMainPoints <= numOtherPoints;\r\n\r\n const bucketStep = bucketIsMain ? numMainPoints : numOtherPoints;\r\n const bucketLimit = bucketIsMain ? numOtherPoints : numMainPoints;\r\n const bucketIdxLimit = bucketIsMain ? numMainPoints - 1 : numOtherPoints - 1;\r\n const winding = bucketIsMain ? 0 : 1;\r\n\r\n let numTris = numMainPoints + numOtherPoints - 2;\r\n\r\n let bucketIdx = bucketIsMain ? idxMain : idxOther;\r\n let nbucketIdx = bucketIsMain ? idxOther : idxMain;\r\n const bucketPoints = bucketIsMain ? mainPointIndices : otherPointIndices;\r\n const nbucketPoints = bucketIsMain ? otherPointIndices : mainPointIndices;\r\n\r\n let bucket = 0;\r\n\r\n while (numTris-- > 0) {\r\n if (winding) {\r\n indices.push(bucketPoints[bucketIdx], nbucketPoints[nbucketIdx]);\r\n } else {\r\n indices.push(nbucketPoints[nbucketIdx], bucketPoints[bucketIdx]);\r\n }\r\n\r\n bucket += bucketStep;\r\n\r\n let lastIdx;\r\n\r\n if (bucket >= bucketLimit && bucketIdx < bucketIdxLimit) {\r\n lastIdx = bucketPoints[++bucketIdx];\r\n bucket -= bucketLimit;\r\n } else {\r\n lastIdx = nbucketPoints[++nbucketIdx];\r\n }\r\n\r\n indices.push(lastIdx);\r\n }\r\n\r\n indices[indexTriangle + 0] = indices[indices.length - 3];\r\n indices[indexTriangle + 1] = indices[indices.length - 2];\r\n indices[indexTriangle + 2] = indices[indices.length - 1];\r\n\r\n indices.length = indices.length - 3;\r\n }\r\n\r\n private _generateEdgesLinesAlternate(): void {\r\n const positions = this._source.getVerticesData(VertexBuffer.PositionKind);\r\n let indices = this._source.getIndices();\r\n\r\n if (!indices || !positions) {\r\n return;\r\n }\r\n\r\n if (!Array.isArray(indices)) {\r\n indices = Array.from(indices);\r\n }\r\n\r\n /**\r\n * Find all vertices that are at the same location (with an epsilon) and remapp them on the same vertex\r\n */\r\n const useFastVertexMerger = this._options?.useFastVertexMerger ?? true;\r\n const epsVertexMerge = useFastVertexMerger ? Math.round(-Math.log(this._options?.epsilonVertexMerge ?? 1e-6) / Math.log(10)) : this._options?.epsilonVertexMerge ?? 1e-6;\r\n const remapVertexIndices: Array = [];\r\n const uniquePositions: Array = []; // list of unique index of vertices - needed for tessellation\r\n\r\n if (useFastVertexMerger) {\r\n const mapVertices: { [key: string]: number } = {};\r\n for (let v1 = 0; v1 < positions.length; v1 += 3) {\r\n const x1 = positions[v1 + 0],\r\n y1 = positions[v1 + 1],\r\n z1 = positions[v1 + 2];\r\n\r\n const key = x1.toFixed(epsVertexMerge) + \"|\" + y1.toFixed(epsVertexMerge) + \"|\" + z1.toFixed(epsVertexMerge);\r\n\r\n if (mapVertices[key] !== undefined) {\r\n remapVertexIndices.push(mapVertices[key]);\r\n } else {\r\n const idx = v1 / 3;\r\n mapVertices[key] = idx;\r\n remapVertexIndices.push(idx);\r\n uniquePositions.push(idx);\r\n }\r\n }\r\n } else {\r\n for (let v1 = 0; v1 < positions.length; v1 += 3) {\r\n const x1 = positions[v1 + 0],\r\n y1 = positions[v1 + 1],\r\n z1 = positions[v1 + 2];\r\n let found = false;\r\n for (let v2 = 0; v2 < v1 && !found; v2 += 3) {\r\n const x2 = positions[v2 + 0],\r\n y2 = positions[v2 + 1],\r\n z2 = positions[v2 + 2];\r\n\r\n if (Math.abs(x1 - x2) < epsVertexMerge && Math.abs(y1 - y2) < epsVertexMerge && Math.abs(z1 - z2) < epsVertexMerge) {\r\n remapVertexIndices.push(v2 / 3);\r\n found = true;\r\n break;\r\n }\r\n }\r\n\r\n if (!found) {\r\n remapVertexIndices.push(v1 / 3);\r\n uniquePositions.push(v1 / 3);\r\n }\r\n }\r\n }\r\n\r\n if (this._options?.applyTessellation) {\r\n /**\r\n * Tessellate triangles if necessary:\r\n *\r\n * A\r\n * +\r\n * |\\\r\n * | \\\r\n * | \\\r\n * E + \\\r\n * /| \\\r\n * / | \\\r\n * / | \\\r\n * +---+-------+ B\r\n * D C\r\n *\r\n * For the edges to be rendered correctly, the ABC triangle has to be split into ABE and BCE, else AC is considered to be an edge, whereas only AE should be.\r\n *\r\n * The tessellation process looks for the vertices like E that are in-between two other vertices making of an edge and create new triangles as necessary\r\n */\r\n\r\n // First step: collect the triangles to tessellate\r\n const epsVertexAligned = this._options?.epsilonVertexAligned ?? 1e-6;\r\n const mustTesselate: Array<{ index: number; edgesPoints: Array> }> = []; // liste of triangles that must be tessellated\r\n\r\n for (let index = 0; index < indices.length; index += 3) {\r\n // loop over all triangles\r\n let triangleToTessellate: { index: number; edgesPoints: Array> } | undefined;\r\n\r\n for (let i = 0; i < 3; ++i) {\r\n // loop over the 3 edges of the triangle\r\n const p0Index = remapVertexIndices[indices[index + i]];\r\n const p1Index = remapVertexIndices[indices[index + ((i + 1) % 3)]];\r\n const p2Index = remapVertexIndices[indices[index + ((i + 2) % 3)]];\r\n\r\n if (p0Index === p1Index) {\r\n continue;\r\n } // degenerated triangle - don't process\r\n\r\n const p0x = positions[p0Index * 3 + 0],\r\n p0y = positions[p0Index * 3 + 1],\r\n p0z = positions[p0Index * 3 + 2];\r\n const p1x = positions[p1Index * 3 + 0],\r\n p1y = positions[p1Index * 3 + 1],\r\n p1z = positions[p1Index * 3 + 2];\r\n\r\n const p0p1 = Math.sqrt((p1x - p0x) * (p1x - p0x) + (p1y - p0y) * (p1y - p0y) + (p1z - p0z) * (p1z - p0z));\r\n\r\n for (let v = 0; v < uniquePositions.length - 1; v++) {\r\n // loop over all (unique) vertices and look for the ones that would be in-between p0 and p1\r\n const vIndex = uniquePositions[v];\r\n\r\n if (vIndex === p0Index || vIndex === p1Index || vIndex === p2Index) {\r\n continue;\r\n } // don't handle the vertex if it is a vertex of the current triangle\r\n\r\n const x = positions[vIndex * 3 + 0],\r\n y = positions[vIndex * 3 + 1],\r\n z = positions[vIndex * 3 + 2];\r\n\r\n const p0p = Math.sqrt((x - p0x) * (x - p0x) + (y - p0y) * (y - p0y) + (z - p0z) * (z - p0z));\r\n const pp1 = Math.sqrt((x - p1x) * (x - p1x) + (y - p1y) * (y - p1y) + (z - p1z) * (z - p1z));\r\n\r\n if (Math.abs(p0p + pp1 - p0p1) < epsVertexAligned) {\r\n // vertices are aligned and p in-between p0 and p1 if distance(p0, p) + distance (p, p1) ~ distance(p0, p1)\r\n if (!triangleToTessellate) {\r\n triangleToTessellate = {\r\n index: index,\r\n edgesPoints: [[], [], []],\r\n };\r\n mustTesselate.push(triangleToTessellate);\r\n }\r\n triangleToTessellate.edgesPoints[i].push([vIndex, p0p]);\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Second step: tesselate the triangles\r\n for (let t = 0; t < mustTesselate.length; ++t) {\r\n const triangle = mustTesselate[t];\r\n\r\n this._tessellateTriangle(triangle.edgesPoints, triangle.index, indices, remapVertexIndices);\r\n }\r\n\r\n mustTesselate.length = 0;\r\n }\r\n\r\n /**\r\n * Collect the edges to render\r\n */\r\n const edges: { [key: string]: { normal: Vector3; done: boolean; index: number; i: number } } = {};\r\n\r\n for (let index = 0; index < indices.length; index += 3) {\r\n let faceNormal;\r\n for (let i = 0; i < 3; ++i) {\r\n let p0Index = remapVertexIndices[indices[index + i]];\r\n let p1Index = remapVertexIndices[indices[index + ((i + 1) % 3)]];\r\n const p2Index = remapVertexIndices[indices[index + ((i + 2) % 3)]];\r\n\r\n if (p0Index === p1Index || ((p0Index === p2Index || p1Index === p2Index) && this._options?.removeDegeneratedTriangles)) {\r\n continue;\r\n }\r\n\r\n TmpVectors.Vector3[0].copyFromFloats(positions[p0Index * 3 + 0], positions[p0Index * 3 + 1], positions[p0Index * 3 + 2]);\r\n TmpVectors.Vector3[1].copyFromFloats(positions[p1Index * 3 + 0], positions[p1Index * 3 + 1], positions[p1Index * 3 + 2]);\r\n TmpVectors.Vector3[2].copyFromFloats(positions[p2Index * 3 + 0], positions[p2Index * 3 + 1], positions[p2Index * 3 + 2]);\r\n\r\n if (!faceNormal) {\r\n TmpVectors.Vector3[1].subtractToRef(TmpVectors.Vector3[0], TmpVectors.Vector3[3]);\r\n TmpVectors.Vector3[2].subtractToRef(TmpVectors.Vector3[1], TmpVectors.Vector3[4]);\r\n faceNormal = Vector3.Cross(TmpVectors.Vector3[3], TmpVectors.Vector3[4]);\r\n faceNormal.normalize();\r\n }\r\n\r\n if (p0Index > p1Index) {\r\n const tmp = p0Index;\r\n p0Index = p1Index;\r\n p1Index = tmp;\r\n }\r\n\r\n const key = p0Index + \"_\" + p1Index;\r\n const ei = edges[key];\r\n\r\n if (ei) {\r\n if (!ei.done) {\r\n const dotProduct = Vector3.Dot(faceNormal, ei.normal);\r\n\r\n if (dotProduct < this._epsilon) {\r\n this.createLine(TmpVectors.Vector3[0], TmpVectors.Vector3[1], this._linesPositions.length / 3);\r\n }\r\n\r\n ei.done = true;\r\n }\r\n } else {\r\n edges[key] = { normal: faceNormal, done: false, index: index, i: i };\r\n }\r\n }\r\n }\r\n\r\n for (const key in edges) {\r\n const ei = edges[key];\r\n if (!ei.done) {\r\n // Orphaned edge - we must display it\r\n const p0Index = remapVertexIndices[indices[ei.index + ei.i]];\r\n const p1Index = remapVertexIndices[indices[ei.index + ((ei.i + 1) % 3)]];\r\n\r\n TmpVectors.Vector3[0].copyFromFloats(positions[p0Index * 3 + 0], positions[p0Index * 3 + 1], positions[p0Index * 3 + 2]);\r\n TmpVectors.Vector3[1].copyFromFloats(positions[p1Index * 3 + 0], positions[p1Index * 3 + 1], positions[p1Index * 3 + 2]);\r\n\r\n this.createLine(TmpVectors.Vector3[0], TmpVectors.Vector3[1], this._linesPositions.length / 3);\r\n }\r\n }\r\n\r\n /**\r\n * Merge into a single mesh\r\n */\r\n const engine = this._source.getScene().getEngine();\r\n\r\n this._buffers[VertexBuffer.PositionKind] = new VertexBuffer(engine, this._linesPositions, VertexBuffer.PositionKind, false);\r\n this._buffers[VertexBuffer.NormalKind] = new VertexBuffer(engine, this._linesNormals, VertexBuffer.NormalKind, false, false, 4);\r\n\r\n this._buffersForInstances[VertexBuffer.PositionKind] = this._buffers[VertexBuffer.PositionKind];\r\n this._buffersForInstances[VertexBuffer.NormalKind] = this._buffers[VertexBuffer.NormalKind];\r\n\r\n this._ib = engine.createIndexBuffer(this._linesIndices);\r\n\r\n this._indicesCount = this._linesIndices.length;\r\n }\r\n\r\n /**\r\n * Generates lines edges from adjacencjes\r\n * @private\r\n */\r\n _generateEdgesLines(): void {\r\n const positions = this._source.getVerticesData(VertexBuffer.PositionKind);\r\n const indices = this._source.getIndices();\r\n\r\n if (!indices || !positions) {\r\n return;\r\n }\r\n\r\n // First let's find adjacencies\r\n const adjacencies = new Array();\r\n const faceNormals = new Array();\r\n let index: number;\r\n let faceAdjacencies: FaceAdjacencies;\r\n\r\n // Prepare faces\r\n for (index = 0; index < indices.length; index += 3) {\r\n faceAdjacencies = new FaceAdjacencies();\r\n const p0Index = indices[index];\r\n const p1Index = indices[index + 1];\r\n const p2Index = indices[index + 2];\r\n\r\n faceAdjacencies.p0 = new Vector3(positions[p0Index * 3], positions[p0Index * 3 + 1], positions[p0Index * 3 + 2]);\r\n faceAdjacencies.p1 = new Vector3(positions[p1Index * 3], positions[p1Index * 3 + 1], positions[p1Index * 3 + 2]);\r\n faceAdjacencies.p2 = new Vector3(positions[p2Index * 3], positions[p2Index * 3 + 1], positions[p2Index * 3 + 2]);\r\n const faceNormal = Vector3.Cross(faceAdjacencies.p1.subtract(faceAdjacencies.p0), faceAdjacencies.p2.subtract(faceAdjacencies.p1));\r\n\r\n faceNormal.normalize();\r\n\r\n faceNormals.push(faceNormal);\r\n adjacencies.push(faceAdjacencies);\r\n }\r\n\r\n // Scan\r\n for (index = 0; index < adjacencies.length; index++) {\r\n faceAdjacencies = adjacencies[index];\r\n\r\n for (let otherIndex = index + 1; otherIndex < adjacencies.length; otherIndex++) {\r\n const otherFaceAdjacencies = adjacencies[otherIndex];\r\n\r\n if (faceAdjacencies.edgesConnectedCount === 3) {\r\n // Full\r\n break;\r\n }\r\n\r\n if (otherFaceAdjacencies.edgesConnectedCount === 3) {\r\n // Full\r\n continue;\r\n }\r\n\r\n const otherP0 = indices[otherIndex * 3];\r\n const otherP1 = indices[otherIndex * 3 + 1];\r\n const otherP2 = indices[otherIndex * 3 + 2];\r\n\r\n for (let edgeIndex = 0; edgeIndex < 3; edgeIndex++) {\r\n let otherEdgeIndex: number = 0;\r\n\r\n if (faceAdjacencies.edges[edgeIndex] !== undefined) {\r\n continue;\r\n }\r\n\r\n switch (edgeIndex) {\r\n case 0:\r\n if (this._checkVerticesInsteadOfIndices) {\r\n otherEdgeIndex = this._processEdgeForAdjacenciesWithVertices(\r\n faceAdjacencies.p0,\r\n faceAdjacencies.p1,\r\n otherFaceAdjacencies.p0,\r\n otherFaceAdjacencies.p1,\r\n otherFaceAdjacencies.p2\r\n );\r\n } else {\r\n otherEdgeIndex = this._processEdgeForAdjacencies(indices[index * 3], indices[index * 3 + 1], otherP0, otherP1, otherP2);\r\n }\r\n break;\r\n case 1:\r\n if (this._checkVerticesInsteadOfIndices) {\r\n otherEdgeIndex = this._processEdgeForAdjacenciesWithVertices(\r\n faceAdjacencies.p1,\r\n faceAdjacencies.p2,\r\n otherFaceAdjacencies.p0,\r\n otherFaceAdjacencies.p1,\r\n otherFaceAdjacencies.p2\r\n );\r\n } else {\r\n otherEdgeIndex = this._processEdgeForAdjacencies(indices[index * 3 + 1], indices[index * 3 + 2], otherP0, otherP1, otherP2);\r\n }\r\n break;\r\n case 2:\r\n if (this._checkVerticesInsteadOfIndices) {\r\n otherEdgeIndex = this._processEdgeForAdjacenciesWithVertices(\r\n faceAdjacencies.p2,\r\n faceAdjacencies.p0,\r\n otherFaceAdjacencies.p0,\r\n otherFaceAdjacencies.p1,\r\n otherFaceAdjacencies.p2\r\n );\r\n } else {\r\n otherEdgeIndex = this._processEdgeForAdjacencies(indices[index * 3 + 2], indices[index * 3], otherP0, otherP1, otherP2);\r\n }\r\n break;\r\n }\r\n\r\n if (otherEdgeIndex === -1) {\r\n continue;\r\n }\r\n\r\n faceAdjacencies.edges[edgeIndex] = otherIndex;\r\n otherFaceAdjacencies.edges[otherEdgeIndex] = index;\r\n\r\n faceAdjacencies.edgesConnectedCount++;\r\n otherFaceAdjacencies.edgesConnectedCount++;\r\n\r\n if (faceAdjacencies.edgesConnectedCount === 3) {\r\n break;\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Create lines\r\n for (index = 0; index < adjacencies.length; index++) {\r\n // We need a line when a face has no adjacency on a specific edge or if all the adjacencies has an angle greater than epsilon\r\n const current = adjacencies[index];\r\n\r\n this._checkEdge(index, current.edges[0], faceNormals, current.p0, current.p1);\r\n this._checkEdge(index, current.edges[1], faceNormals, current.p1, current.p2);\r\n this._checkEdge(index, current.edges[2], faceNormals, current.p2, current.p0);\r\n }\r\n\r\n // Merge into a single mesh\r\n const engine = this._source.getScene().getEngine();\r\n\r\n this._buffers[VertexBuffer.PositionKind] = new VertexBuffer(engine, this._linesPositions, VertexBuffer.PositionKind, false);\r\n this._buffers[VertexBuffer.NormalKind] = new VertexBuffer(engine, this._linesNormals, VertexBuffer.NormalKind, false, false, 4);\r\n\r\n this._buffersForInstances[VertexBuffer.PositionKind] = this._buffers[VertexBuffer.PositionKind];\r\n this._buffersForInstances[VertexBuffer.NormalKind] = this._buffers[VertexBuffer.NormalKind];\r\n\r\n this._ib = engine.createIndexBuffer(this._linesIndices);\r\n\r\n this._indicesCount = this._linesIndices.length;\r\n }\r\n\r\n /**\r\n * Checks whether or not the edges renderer is ready to render.\r\n * @returns true if ready, otherwise false.\r\n */\r\n public isReady(): boolean {\r\n return this._lineShader.isReady(this._source, (this._source.hasInstances && this.customInstances.length > 0) || this._source.hasThinInstances);\r\n }\r\n\r\n /**\r\n * Renders the edges of the attached mesh,\r\n */\r\n public render(): void {\r\n const scene = this._source.getScene();\r\n\r\n const currentDrawWrapper = this._lineShader._getDrawWrapper();\r\n if (this._drawWrapper) {\r\n this._lineShader._setDrawWrapper(this._drawWrapper);\r\n }\r\n\r\n if (!this.isReady() || !scene.activeCamera) {\r\n this._lineShader._setDrawWrapper(currentDrawWrapper);\r\n return;\r\n }\r\n\r\n const hasInstances = this._source.hasInstances && this.customInstances.length > 0;\r\n const useBuffersWithInstances = hasInstances || this._source.hasThinInstances;\r\n\r\n let instanceCount = 0;\r\n\r\n if (useBuffersWithInstances) {\r\n this._buffersForInstances[\"world0\"] = (this._source as Mesh).getVertexBuffer(\"world0\");\r\n this._buffersForInstances[\"world1\"] = (this._source as Mesh).getVertexBuffer(\"world1\");\r\n this._buffersForInstances[\"world2\"] = (this._source as Mesh).getVertexBuffer(\"world2\");\r\n this._buffersForInstances[\"world3\"] = (this._source as Mesh).getVertexBuffer(\"world3\");\r\n\r\n if (hasInstances) {\r\n const instanceStorage = (this._source as Mesh)._instanceDataStorage;\r\n\r\n instanceCount = this.customInstances.length;\r\n\r\n if (!instanceStorage.instancesData) {\r\n if (!this._source.getScene()._activeMeshesFrozen) {\r\n this.customInstances.reset();\r\n }\r\n return;\r\n }\r\n\r\n if (!instanceStorage.isFrozen) {\r\n let offset = 0;\r\n\r\n for (let i = 0; i < instanceCount; ++i) {\r\n this.customInstances.data[i].copyToArray(instanceStorage.instancesData, offset);\r\n offset += 16;\r\n }\r\n\r\n instanceStorage.instancesBuffer!.updateDirectly(instanceStorage.instancesData, 0, instanceCount);\r\n }\r\n } else {\r\n instanceCount = (this._source as Mesh).thinInstanceCount;\r\n }\r\n }\r\n\r\n const engine = scene.getEngine();\r\n this._lineShader._preBind();\r\n\r\n if (this._source.edgesColor.a !== 1) {\r\n engine.setAlphaMode(Constants.ALPHA_COMBINE);\r\n } else {\r\n engine.setAlphaMode(Constants.ALPHA_DISABLE);\r\n }\r\n\r\n // VBOs\r\n engine.bindBuffers(useBuffersWithInstances ? this._buffersForInstances : this._buffers, this._ib, this._lineShader.getEffect());\r\n\r\n scene.resetCachedMaterial();\r\n this._lineShader.setColor4(\"color\", this._source.edgesColor);\r\n\r\n if (scene.activeCamera.mode === Camera.ORTHOGRAPHIC_CAMERA) {\r\n this._lineShader.setFloat(\"width\", this._source.edgesWidth / this.edgesWidthScalerForOrthographic);\r\n } else {\r\n this._lineShader.setFloat(\"width\", this._source.edgesWidth / this.edgesWidthScalerForPerspective);\r\n }\r\n\r\n this._lineShader.setFloat(\"aspectRatio\", engine.getAspectRatio(scene.activeCamera));\r\n this._lineShader.bind(this._source.getWorldMatrix());\r\n\r\n // Draw order\r\n engine.drawElementsType(Material.TriangleFillMode, 0, this._indicesCount, instanceCount);\r\n this._lineShader.unbind();\r\n\r\n if (useBuffersWithInstances) {\r\n engine.unbindInstanceAttributes();\r\n }\r\n\r\n if (!this._source.getScene()._activeMeshesFrozen) {\r\n this.customInstances.reset();\r\n }\r\n\r\n this._lineShader._setDrawWrapper(currentDrawWrapper);\r\n }\r\n}\r\n\r\n/**\r\n * LineEdgesRenderer for LineMeshes to remove unnecessary triangulation\r\n */\r\nexport class LineEdgesRenderer extends EdgesRenderer {\r\n /**\r\n * This constructor turns off auto generating edges line in Edges Renderer to make it here.\r\n * @param source LineMesh used to generate edges\r\n * @param epsilon not important (specified angle for edge detection)\r\n * @param checkVerticesInsteadOfIndices not important for LineMesh\r\n */\r\n constructor(source: AbstractMesh, epsilon = 0.95, checkVerticesInsteadOfIndices = false) {\r\n super(source, epsilon, checkVerticesInsteadOfIndices, false);\r\n this._generateEdgesLines();\r\n }\r\n\r\n /**\r\n * Generate edges for each line in LinesMesh. Every Line should be rendered as edge.\r\n */\r\n _generateEdgesLines(): void {\r\n const positions = this._source.getVerticesData(VertexBuffer.PositionKind);\r\n const indices = this._source.getIndices();\r\n\r\n if (!indices || !positions) {\r\n return;\r\n }\r\n\r\n const p0 = TmpVectors.Vector3[0];\r\n const p1 = TmpVectors.Vector3[1];\r\n const len = indices.length - 1;\r\n for (let i = 0, offset = 0; i < len; i += 2, offset += 4) {\r\n Vector3.FromArrayToRef(positions, 3 * indices[i], p0);\r\n Vector3.FromArrayToRef(positions, 3 * indices[i + 1], p1);\r\n this.createLine(p0, p1, offset);\r\n }\r\n\r\n // Merge into a single mesh\r\n const engine = this._source.getScene().getEngine();\r\n\r\n this._buffers[VertexBuffer.PositionKind] = new VertexBuffer(engine, this._linesPositions, VertexBuffer.PositionKind, false);\r\n this._buffers[VertexBuffer.NormalKind] = new VertexBuffer(engine, this._linesNormals, VertexBuffer.NormalKind, false, false, 4);\r\n\r\n this._ib = engine.createIndexBuffer(this._linesIndices);\r\n\r\n this._indicesCount = this._linesIndices.length;\r\n }\r\n}\r\n","import { InternalTexture, InternalTextureSource } from \"../../Materials/Textures/internalTexture\";\r\nimport type { IMultiRenderTargetOptions } from \"../../Materials/Textures/multiRenderTarget\";\r\nimport { Logger } from \"../../Misc/logger\";\r\nimport type { Nullable } from \"../../types\";\r\nimport { Constants } from \"../constants\";\r\nimport { ThinEngine } from \"../thinEngine\";\r\nimport type { RenderTargetWrapper } from \"../renderTargetWrapper\";\r\nimport type { WebGLRenderTargetWrapper } from \"../WebGL/webGLRenderTargetWrapper\";\r\nimport type { WebGLHardwareTexture } from \"../WebGL/webGLHardwareTexture\";\r\nimport type { TextureSize } from \"../../Materials/Textures/textureCreationOptions\";\r\n\r\ndeclare module \"../../Engines/thinEngine\" {\r\n export interface ThinEngine {\r\n /**\r\n * Unbind a list of render target textures from the webGL context\r\n * This is used only when drawBuffer extension or webGL2 are active\r\n * @param rtWrapper defines the render target wrapper to unbind\r\n * @param disableGenerateMipMaps defines a boolean indicating that mipmaps must not be generated\r\n * @param onBeforeUnbind defines a function which will be called before the effective unbind\r\n */\r\n unBindMultiColorAttachmentFramebuffer(rtWrapper: RenderTargetWrapper, disableGenerateMipMaps: boolean, onBeforeUnbind?: () => void): void;\r\n\r\n /**\r\n * Create a multi render target texture\r\n * @see https://doc.babylonjs.com/setup/support/webGL2#multiple-render-target\r\n * @param size defines the size of the texture\r\n * @param options defines the creation options\r\n * @param initializeBuffers if set to true, the engine will make an initializing call of drawBuffers\r\n * @returns a new render target wrapper ready to render textures\r\n */\r\n createMultipleRenderTarget(size: TextureSize, options: IMultiRenderTargetOptions, initializeBuffers?: boolean): RenderTargetWrapper;\r\n\r\n /**\r\n * Update the sample count for a given multiple render target texture\r\n * @see https://doc.babylonjs.com/setup/support/webGL2#multisample-render-targets\r\n * @param rtWrapper defines the render target wrapper to update\r\n * @param samples defines the sample count to set\r\n * @param initializeBuffers if set to true, the engine will make an initializing call of drawBuffers\r\n * @returns the effective sample count (could be 0 if multisample render targets are not supported)\r\n */\r\n updateMultipleRenderTargetTextureSampleCount(rtWrapper: Nullable, samples: number, initializeBuffers?: boolean): number;\r\n\r\n /**\r\n * Select a subsets of attachments to draw to.\r\n * @param attachments gl attachments\r\n */\r\n bindAttachments(attachments: number[]): void;\r\n\r\n /**\r\n * Creates a layout object to draw/clear on specific textures in a MRT\r\n * @param textureStatus textureStatus[i] indicates if the i-th is active\r\n * @returns A layout to be fed to the engine, calling `bindAttachments`.\r\n */\r\n buildTextureLayout(textureStatus: boolean[]): number[];\r\n\r\n /**\r\n * Restores the webgl state to only draw on the main color attachment\r\n * when the frame buffer associated is the canvas frame buffer\r\n */\r\n restoreSingleAttachment(): void;\r\n\r\n /**\r\n * Restores the webgl state to only draw on the main color attachment\r\n * when the frame buffer associated is not the canvas frame buffer\r\n */\r\n restoreSingleAttachmentForRenderTarget(): void;\r\n }\r\n}\r\n\r\nThinEngine.prototype.restoreSingleAttachment = function (): void {\r\n const gl = this._gl;\r\n\r\n this.bindAttachments([gl.BACK]);\r\n};\r\n\r\nThinEngine.prototype.restoreSingleAttachmentForRenderTarget = function (): void {\r\n const gl = this._gl;\r\n\r\n this.bindAttachments([gl.COLOR_ATTACHMENT0]);\r\n};\r\n\r\nThinEngine.prototype.buildTextureLayout = function (textureStatus: boolean[]): number[] {\r\n const gl = this._gl;\r\n\r\n const result = [];\r\n\r\n for (let i = 0; i < textureStatus.length; i++) {\r\n if (textureStatus[i]) {\r\n result.push((gl)[\"COLOR_ATTACHMENT\" + i]);\r\n } else {\r\n result.push(gl.NONE);\r\n }\r\n }\r\n\r\n return result;\r\n};\r\n\r\nThinEngine.prototype.bindAttachments = function (attachments: number[]): void {\r\n const gl = this._gl;\r\n\r\n gl.drawBuffers(attachments);\r\n};\r\n\r\nThinEngine.prototype.unBindMultiColorAttachmentFramebuffer = function (\r\n rtWrapper: WebGLRenderTargetWrapper,\r\n disableGenerateMipMaps: boolean = false,\r\n onBeforeUnbind?: () => void\r\n): void {\r\n this._currentRenderTarget = null;\r\n\r\n // If MSAA, we need to bitblt back to main texture\r\n const gl = this._gl;\r\n\r\n const attachments = rtWrapper._attachments!;\r\n const count = attachments.length;\r\n\r\n if (rtWrapper._MSAAFramebuffer) {\r\n gl.bindFramebuffer(gl.READ_FRAMEBUFFER, rtWrapper._MSAAFramebuffer);\r\n gl.bindFramebuffer(gl.DRAW_FRAMEBUFFER, rtWrapper._framebuffer);\r\n\r\n for (let i = 0; i < count; i++) {\r\n const texture = rtWrapper.textures![i];\r\n\r\n for (let j = 0; j < count; j++) {\r\n attachments[j] = gl.NONE;\r\n }\r\n\r\n attachments[i] = (gl)[this.webGLVersion > 1 ? \"COLOR_ATTACHMENT\" + i : \"COLOR_ATTACHMENT\" + i + \"_WEBGL\"];\r\n gl.readBuffer(attachments[i]);\r\n gl.drawBuffers(attachments);\r\n gl.blitFramebuffer(0, 0, texture.width, texture.height, 0, 0, texture.width, texture.height, gl.COLOR_BUFFER_BIT, gl.NEAREST);\r\n }\r\n\r\n for (let i = 0; i < count; i++) {\r\n attachments[i] = (gl)[this.webGLVersion > 1 ? \"COLOR_ATTACHMENT\" + i : \"COLOR_ATTACHMENT\" + i + \"_WEBGL\"];\r\n }\r\n\r\n gl.drawBuffers(attachments);\r\n }\r\n\r\n for (let i = 0; i < count; i++) {\r\n const texture = rtWrapper.textures![i];\r\n if (texture?.generateMipMaps && !disableGenerateMipMaps && !texture.isCube) {\r\n this._bindTextureDirectly(gl.TEXTURE_2D, texture, true);\r\n gl.generateMipmap(gl.TEXTURE_2D);\r\n this._bindTextureDirectly(gl.TEXTURE_2D, null);\r\n }\r\n }\r\n\r\n if (onBeforeUnbind) {\r\n if (rtWrapper._MSAAFramebuffer) {\r\n // Bind the correct framebuffer\r\n this._bindUnboundFramebuffer(rtWrapper._framebuffer);\r\n }\r\n onBeforeUnbind();\r\n }\r\n\r\n this._bindUnboundFramebuffer(null);\r\n};\r\n\r\nThinEngine.prototype.createMultipleRenderTarget = function (size: TextureSize, options: IMultiRenderTargetOptions, initializeBuffers: boolean = true): RenderTargetWrapper {\r\n let generateMipMaps = false;\r\n let generateDepthBuffer = true;\r\n let generateStencilBuffer = false;\r\n let generateDepthTexture = false;\r\n let depthTextureFormat = Constants.TEXTUREFORMAT_DEPTH16;\r\n let textureCount = 1;\r\n\r\n const defaultType = Constants.TEXTURETYPE_UNSIGNED_INT;\r\n const defaultSamplingMode = Constants.TEXTURE_TRILINEAR_SAMPLINGMODE;\r\n const defaultUseSRGBBuffer = false;\r\n const defaultFormat = Constants.TEXTUREFORMAT_RGBA;\r\n const defaultTarget = Constants.TEXTURE_2D;\r\n\r\n let types = new Array();\r\n let samplingModes = new Array();\r\n let useSRGBBuffers = new Array();\r\n let formats = new Array();\r\n let targets = new Array();\r\n let faceIndex = new Array();\r\n let layerIndex = new Array();\r\n let layers = new Array();\r\n\r\n const rtWrapper = this._createHardwareRenderTargetWrapper(true, false, size) as WebGLRenderTargetWrapper;\r\n\r\n if (options !== undefined) {\r\n generateMipMaps = options.generateMipMaps === undefined ? false : options.generateMipMaps;\r\n generateDepthBuffer = options.generateDepthBuffer === undefined ? true : options.generateDepthBuffer;\r\n generateStencilBuffer = options.generateStencilBuffer === undefined ? false : options.generateStencilBuffer;\r\n generateDepthTexture = options.generateDepthTexture === undefined ? false : options.generateDepthTexture;\r\n textureCount = options.textureCount || 1;\r\n\r\n if (options.types) {\r\n types = options.types;\r\n }\r\n if (options.samplingModes) {\r\n samplingModes = options.samplingModes;\r\n }\r\n if (options.useSRGBBuffers) {\r\n useSRGBBuffers = options.useSRGBBuffers;\r\n }\r\n if (options.formats) {\r\n formats = options.formats;\r\n }\r\n if (options.targetTypes) {\r\n targets = options.targetTypes;\r\n }\r\n if (options.faceIndex) {\r\n faceIndex = options.faceIndex;\r\n }\r\n if (options.layerIndex) {\r\n layerIndex = options.layerIndex;\r\n }\r\n if (options.layerCounts) {\r\n layers = options.layerCounts;\r\n }\r\n if (\r\n this.webGLVersion > 1 &&\r\n (options.depthTextureFormat === Constants.TEXTUREFORMAT_DEPTH24_STENCIL8 ||\r\n options.depthTextureFormat === Constants.TEXTUREFORMAT_DEPTH24UNORM_STENCIL8 ||\r\n options.depthTextureFormat === Constants.TEXTUREFORMAT_DEPTH24 ||\r\n options.depthTextureFormat === Constants.TEXTUREFORMAT_DEPTH32_FLOAT ||\r\n options.depthTextureFormat === Constants.TEXTUREFORMAT_DEPTH32FLOAT_STENCIL8)\r\n ) {\r\n depthTextureFormat = options.depthTextureFormat;\r\n }\r\n }\r\n const gl = this._gl;\r\n // Create the framebuffer\r\n const framebuffer = gl.createFramebuffer();\r\n this._bindUnboundFramebuffer(framebuffer);\r\n\r\n const width = (<{ width: number; height: number }>size).width || size;\r\n const height = (<{ width: number; height: number }>size).height || size;\r\n\r\n const textures: InternalTexture[] = [];\r\n const attachments: number[] = [];\r\n\r\n const useStencilTexture =\r\n this.webGLVersion > 1 &&\r\n generateDepthTexture &&\r\n (options.depthTextureFormat === Constants.TEXTUREFORMAT_DEPTH24_STENCIL8 ||\r\n options.depthTextureFormat === Constants.TEXTUREFORMAT_DEPTH24UNORM_STENCIL8 ||\r\n options.depthTextureFormat === Constants.TEXTUREFORMAT_DEPTH32FLOAT_STENCIL8);\r\n const depthStencilBuffer = this._setupFramebufferDepthAttachments(!useStencilTexture && generateStencilBuffer, !generateDepthTexture && generateDepthBuffer, width, height);\r\n\r\n rtWrapper._framebuffer = framebuffer;\r\n rtWrapper._depthStencilBuffer = depthStencilBuffer;\r\n rtWrapper._generateDepthBuffer = !generateDepthTexture && generateDepthBuffer;\r\n rtWrapper._generateStencilBuffer = !useStencilTexture && generateStencilBuffer;\r\n rtWrapper._attachments = attachments;\r\n\r\n for (let i = 0; i < textureCount; i++) {\r\n let samplingMode = samplingModes[i] || defaultSamplingMode;\r\n let type = types[i] || defaultType;\r\n let useSRGBBuffer = useSRGBBuffers[i] || defaultUseSRGBBuffer;\r\n const format = formats[i] || defaultFormat;\r\n\r\n const target = targets[i] || defaultTarget;\r\n const layerCount = layers[i] ?? 1;\r\n\r\n if (type === Constants.TEXTURETYPE_FLOAT && !this._caps.textureFloatLinearFiltering) {\r\n // if floating point linear (gl.FLOAT) then force to NEAREST_SAMPLINGMODE\r\n samplingMode = Constants.TEXTURE_NEAREST_SAMPLINGMODE;\r\n } else if (type === Constants.TEXTURETYPE_HALF_FLOAT && !this._caps.textureHalfFloatLinearFiltering) {\r\n // if floating point linear (HALF_FLOAT) then force to NEAREST_SAMPLINGMODE\r\n samplingMode = Constants.TEXTURE_NEAREST_SAMPLINGMODE;\r\n }\r\n\r\n const filters = this._getSamplingParameters(samplingMode, generateMipMaps);\r\n if (type === Constants.TEXTURETYPE_FLOAT && !this._caps.textureFloat) {\r\n type = Constants.TEXTURETYPE_UNSIGNED_INT;\r\n Logger.Warn(\"Float textures are not supported. Render target forced to TEXTURETYPE_UNSIGNED_BYTE type\");\r\n }\r\n\r\n useSRGBBuffer = useSRGBBuffer && this._caps.supportSRGBBuffers && (this.webGLVersion > 1 || this.isWebGPU);\r\n\r\n const isWebGL2 = this.webGLVersion > 1;\r\n const attachment = (gl)[isWebGL2 ? \"COLOR_ATTACHMENT\" + i : \"COLOR_ATTACHMENT\" + i + \"_WEBGL\"];\r\n\r\n attachments.push(attachment);\r\n\r\n if (target === -1) {\r\n continue;\r\n }\r\n\r\n const texture = new InternalTexture(this, InternalTextureSource.MultiRenderTarget);\r\n textures[i] = texture;\r\n\r\n gl.activeTexture((gl)[\"TEXTURE\" + i]);\r\n gl.bindTexture(target, texture._hardwareTexture!.underlyingResource);\r\n\r\n gl.texParameteri(target, gl.TEXTURE_MAG_FILTER, filters.mag);\r\n gl.texParameteri(target, gl.TEXTURE_MIN_FILTER, filters.min);\r\n gl.texParameteri(target, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\r\n gl.texParameteri(target, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\r\n\r\n const internalSizedFormat = this._getRGBABufferInternalSizedFormat(type, format, useSRGBBuffer);\r\n const internalFormat = this._getInternalFormat(format);\r\n const webGLTextureType = this._getWebGLTextureType(type);\r\n\r\n if (isWebGL2 && (target === Constants.TEXTURE_2D_ARRAY || target === Constants.TEXTURE_3D)) {\r\n if (target === Constants.TEXTURE_2D_ARRAY) {\r\n texture.is2DArray = true;\r\n } else {\r\n texture.is3D = true;\r\n }\r\n\r\n texture.baseDepth = texture.depth = layerCount;\r\n\r\n gl.texImage3D(target, 0, internalSizedFormat, width, height, layerCount, 0, internalFormat, webGLTextureType, null);\r\n } else if (target === Constants.TEXTURE_CUBE_MAP) {\r\n // We have to generate all faces to complete the framebuffer\r\n for (let i = 0; i < 6; i++) {\r\n gl.texImage2D(gl.TEXTURE_CUBE_MAP_POSITIVE_X + i, 0, internalSizedFormat, width, height, 0, internalFormat, webGLTextureType, null);\r\n }\r\n texture.isCube = true;\r\n } else {\r\n gl.texImage2D(gl.TEXTURE_2D, 0, internalSizedFormat, width, height, 0, internalFormat, webGLTextureType, null);\r\n }\r\n\r\n if (generateMipMaps) {\r\n gl.generateMipmap(target);\r\n }\r\n\r\n // Unbind\r\n this._bindTextureDirectly(target, null);\r\n\r\n texture.baseWidth = width;\r\n texture.baseHeight = height;\r\n texture.width = width;\r\n texture.height = height;\r\n texture.isReady = true;\r\n texture.samples = 1;\r\n texture.generateMipMaps = generateMipMaps;\r\n texture.samplingMode = samplingMode;\r\n texture.type = type;\r\n texture._useSRGBBuffer = useSRGBBuffer;\r\n texture.format = format;\r\n\r\n this._internalTexturesCache.push(texture);\r\n }\r\n\r\n if (generateDepthTexture && this._caps.depthTextureExtension) {\r\n // Depth texture\r\n const depthTexture = new InternalTexture(this, InternalTextureSource.Depth);\r\n\r\n let depthTextureType = Constants.TEXTURETYPE_UNSIGNED_SHORT;\r\n let glDepthTextureInternalFormat: GLenum = gl.DEPTH_COMPONENT16;\r\n let glDepthTextureFormat: GLenum = gl.DEPTH_COMPONENT;\r\n let glDepthTextureType: GLenum = gl.UNSIGNED_SHORT;\r\n let glDepthTextureAttachment: GLenum = gl.DEPTH_ATTACHMENT;\r\n if (this.webGLVersion < 2) {\r\n glDepthTextureInternalFormat = gl.DEPTH_COMPONENT;\r\n } else {\r\n if (depthTextureFormat === Constants.TEXTUREFORMAT_DEPTH32_FLOAT) {\r\n depthTextureType = Constants.TEXTURETYPE_FLOAT;\r\n glDepthTextureType = gl.FLOAT;\r\n glDepthTextureInternalFormat = gl.DEPTH_COMPONENT32F;\r\n } else if (depthTextureFormat === Constants.TEXTUREFORMAT_DEPTH32FLOAT_STENCIL8) {\r\n depthTextureType = Constants.TEXTURETYPE_UNSIGNED_INT;\r\n glDepthTextureType = gl.FLOAT_32_UNSIGNED_INT_24_8_REV;\r\n glDepthTextureInternalFormat = gl.DEPTH32F_STENCIL8;\r\n glDepthTextureFormat = gl.DEPTH_STENCIL;\r\n glDepthTextureAttachment = gl.DEPTH_STENCIL_ATTACHMENT;\r\n } else if (depthTextureFormat === Constants.TEXTUREFORMAT_DEPTH24) {\r\n depthTextureType = Constants.TEXTURETYPE_UNSIGNED_INT;\r\n glDepthTextureType = gl.UNSIGNED_INT;\r\n glDepthTextureInternalFormat = gl.DEPTH_COMPONENT24;\r\n glDepthTextureAttachment = gl.DEPTH_ATTACHMENT;\r\n } else if (depthTextureFormat === Constants.TEXTUREFORMAT_DEPTH24_STENCIL8 || depthTextureFormat === Constants.TEXTUREFORMAT_DEPTH24UNORM_STENCIL8) {\r\n depthTextureType = Constants.TEXTURETYPE_UNSIGNED_INT_24_8;\r\n glDepthTextureType = gl.UNSIGNED_INT_24_8;\r\n glDepthTextureInternalFormat = gl.DEPTH24_STENCIL8;\r\n glDepthTextureFormat = gl.DEPTH_STENCIL;\r\n glDepthTextureAttachment = gl.DEPTH_STENCIL_ATTACHMENT;\r\n }\r\n }\r\n\r\n gl.activeTexture(gl.TEXTURE0);\r\n gl.bindTexture(gl.TEXTURE_2D, depthTexture._hardwareTexture!.underlyingResource);\r\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST);\r\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST);\r\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\r\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\r\n gl.texImage2D(gl.TEXTURE_2D, 0, glDepthTextureInternalFormat, width, height, 0, glDepthTextureFormat, glDepthTextureType, null);\r\n\r\n gl.framebufferTexture2D(gl.FRAMEBUFFER, glDepthTextureAttachment, gl.TEXTURE_2D, depthTexture._hardwareTexture!.underlyingResource, 0);\r\n\r\n depthTexture.baseWidth = width;\r\n depthTexture.baseHeight = height;\r\n depthTexture.width = width;\r\n depthTexture.height = height;\r\n depthTexture.isReady = true;\r\n depthTexture.samples = 1;\r\n depthTexture.generateMipMaps = generateMipMaps;\r\n depthTexture.samplingMode = Constants.TEXTURE_NEAREST_SAMPLINGMODE;\r\n depthTexture.format = depthTextureFormat;\r\n depthTexture.type = depthTextureType;\r\n\r\n textures[textureCount] = depthTexture;\r\n this._internalTexturesCache.push(depthTexture);\r\n }\r\n rtWrapper.setTextures(textures);\r\n if (initializeBuffers) {\r\n gl.drawBuffers(attachments);\r\n }\r\n\r\n this._bindUnboundFramebuffer(null);\r\n\r\n rtWrapper.setLayerAndFaceIndices(layerIndex, faceIndex);\r\n\r\n this.resetTextureCache();\r\n\r\n return rtWrapper;\r\n};\r\n\r\nThinEngine.prototype.updateMultipleRenderTargetTextureSampleCount = function (\r\n rtWrapper: Nullable,\r\n samples: number,\r\n initializeBuffers: boolean = true\r\n): number {\r\n if (this.webGLVersion < 2 || !rtWrapper || !rtWrapper.texture) {\r\n return 1;\r\n }\r\n\r\n if (rtWrapper.samples === samples) {\r\n return samples;\r\n }\r\n\r\n const count = rtWrapper._attachments!.length;\r\n\r\n if (count === 0) {\r\n return 1;\r\n }\r\n\r\n const gl = this._gl;\r\n\r\n samples = Math.min(samples, this.getCaps().maxMSAASamples);\r\n\r\n // Dispose previous render buffers\r\n const useDepthStencil = !!rtWrapper._depthStencilBuffer;\r\n if (useDepthStencil) {\r\n gl.deleteRenderbuffer(rtWrapper._depthStencilBuffer);\r\n rtWrapper._depthStencilBuffer = null;\r\n }\r\n\r\n if (rtWrapper._MSAAFramebuffer) {\r\n gl.deleteFramebuffer(rtWrapper._MSAAFramebuffer);\r\n rtWrapper._MSAAFramebuffer = null;\r\n }\r\n\r\n if (samples > 1 && typeof gl.renderbufferStorageMultisample === \"function\") {\r\n const framebuffer = gl.createFramebuffer();\r\n\r\n if (!framebuffer) {\r\n throw new Error(\"Unable to create multi sampled framebuffer\");\r\n }\r\n\r\n rtWrapper._MSAAFramebuffer = framebuffer;\r\n this._bindUnboundFramebuffer(framebuffer);\r\n\r\n const attachments = [];\r\n\r\n for (let i = 0; i < count; i++) {\r\n const texture = rtWrapper.textures![i];\r\n const hardwareTexture = texture._hardwareTexture as WebGLHardwareTexture;\r\n\r\n hardwareTexture.releaseMSAARenderBuffers();\r\n }\r\n\r\n for (let i = 0; i < count; i++) {\r\n const texture = rtWrapper.textures![i];\r\n const hardwareTexture = texture._hardwareTexture as WebGLHardwareTexture;\r\n const attachment = (gl)[this.webGLVersion > 1 ? \"COLOR_ATTACHMENT\" + i : \"COLOR_ATTACHMENT\" + i + \"_WEBGL\"];\r\n\r\n const colorRenderbuffer = this._createRenderBuffer(\r\n texture.width,\r\n texture.height,\r\n samples,\r\n -1 /* not used */,\r\n this._getRGBAMultiSampleBufferFormat(texture.type, texture.format),\r\n attachment\r\n );\r\n\r\n if (!colorRenderbuffer) {\r\n throw new Error(\"Unable to create multi sampled framebuffer\");\r\n }\r\n\r\n hardwareTexture.addMSAARenderBuffer(colorRenderbuffer);\r\n texture.samples = samples;\r\n\r\n attachments.push(attachment);\r\n }\r\n if (initializeBuffers) {\r\n gl.drawBuffers(attachments);\r\n }\r\n } else {\r\n this._bindUnboundFramebuffer(rtWrapper._framebuffer);\r\n }\r\n\r\n if (useDepthStencil) {\r\n rtWrapper._depthStencilBuffer = this._setupFramebufferDepthAttachments(\r\n rtWrapper._generateStencilBuffer,\r\n rtWrapper._generateDepthBuffer,\r\n rtWrapper.texture.width,\r\n rtWrapper.texture.height,\r\n samples\r\n );\r\n }\r\n\r\n this._bindUnboundFramebuffer(null);\r\n\r\n return samples;\r\n};\r\n","import type { Scene } from \"../../scene\";\r\nimport type { Engine } from \"../../Engines/engine\";\r\nimport { Texture } from \"../../Materials/Textures/texture\";\r\nimport { RenderTargetTexture } from \"../../Materials/Textures/renderTargetTexture\";\r\nimport { Constants } from \"../../Engines/constants\";\r\n\r\nimport \"../../Engines/Extensions/engine.multiRender\";\r\nimport type { InternalTexture } from \"./internalTexture\";\r\n\r\n/**\r\n * Creation options of the multi render target texture.\r\n */\r\nexport interface IMultiRenderTargetOptions {\r\n /**\r\n * Define if the texture needs to create mip maps after render.\r\n */\r\n generateMipMaps?: boolean;\r\n /**\r\n * Define the types of all the draw buffers we want to create\r\n */\r\n types?: number[];\r\n /**\r\n * Define the sampling modes of all the draw buffers we want to create\r\n */\r\n samplingModes?: number[];\r\n /**\r\n * Define if sRGB format should be used for each of the draw buffers we want to create\r\n */\r\n useSRGBBuffers?: boolean[];\r\n /**\r\n * Define if a depth buffer is required\r\n */\r\n generateDepthBuffer?: boolean;\r\n /**\r\n * Define if a stencil buffer is required\r\n */\r\n generateStencilBuffer?: boolean;\r\n /**\r\n * Define if a depth texture is required instead of a depth buffer\r\n */\r\n generateDepthTexture?: boolean;\r\n /**\r\n * Define the internal format of the buffer in the RTT (RED, RG, RGB, RGBA (default), ALPHA...) of all the draw buffers we want to create\r\n */\r\n formats?: number[];\r\n /**\r\n * Define depth texture format to use\r\n */\r\n depthTextureFormat?: number;\r\n /**\r\n * Define the number of desired draw buffers\r\n */\r\n textureCount?: number;\r\n /**\r\n * Define if aspect ratio should be adapted to the texture or stay the scene one\r\n */\r\n doNotChangeAspectRatio?: boolean;\r\n /**\r\n * Define the default type of the buffers we are creating\r\n */\r\n defaultType?: number;\r\n /**\r\n * Define the default type of the buffers we are creating\r\n */\r\n drawOnlyOnFirstAttachmentByDefault?: boolean;\r\n /**\r\n * Define the type of texture at each attahment index (of Constants.TEXTURE_2D, .TEXTURE_2D_ARRAY, .TEXTURE_CUBE_MAP, .TEXTURE_CUBE_MAP_ARRAY, .TEXTURE_3D).\r\n * You can also use the -1 value to indicate that no texture should be created but that you will assign a texture to that attachment index later.\r\n * Can be useful when you want to attach several layers of the same 2DArrayTexture / 3DTexture or several faces of the same CubeMapTexture: Use the setInternalTexture\r\n * method for that purpose, after the MultiRenderTarget has been created.\r\n */\r\n targetTypes?: number[];\r\n /**\r\n * Define the face index of each texture in the textures array (if applicable, given the corresponding targetType) at creation time (for Constants.TEXTURE_CUBE_MAP and .TEXTURE_CUBE_MAP_ARRAY).\r\n * Can be changed at any time by calling setLayerAndFaceIndices or setLayerAndFaceIndex\r\n */\r\n faceIndex?: number[];\r\n /**\r\n * Define the layer index of each texture in the textures array (if applicable, given the corresponding targetType) at creation time (for Constants.TEXTURE_3D, .TEXTURE_2D_ARRAY, and .TEXTURE_CUBE_MAP_ARRAY).\r\n * Can be changed at any time by calling setLayerAndFaceIndices or setLayerAndFaceIndex\r\n */\r\n layerIndex?: number[];\r\n /**\r\n * Define the number of layer of each texture in the textures array (if applicable, given the corresponding targetType) (for Constants.TEXTURE_3D, .TEXTURE_2D_ARRAY, and .TEXTURE_CUBE_MAP_ARRAY)\r\n */\r\n layerCounts?: number[];\r\n}\r\n\r\n/**\r\n * A multi render target, like a render target provides the ability to render to a texture.\r\n * Unlike the render target, it can render to several draw buffers in one draw.\r\n * This is specially interesting in deferred rendering or for any effects requiring more than\r\n * just one color from a single pass.\r\n */\r\nexport class MultiRenderTarget extends RenderTargetTexture {\r\n private _textures: Texture[];\r\n private _multiRenderTargetOptions: IMultiRenderTargetOptions;\r\n private _count: number;\r\n private _drawOnlyOnFirstAttachmentByDefault: boolean;\r\n private _textureNames?: string[];\r\n\r\n /**\r\n * Get if draw buffers are currently supported by the used hardware and browser.\r\n */\r\n public get isSupported(): boolean {\r\n return this._engine?.getCaps().drawBuffersExtension ?? false;\r\n }\r\n\r\n /**\r\n * Get the list of textures generated by the multi render target.\r\n */\r\n public get textures(): Texture[] {\r\n return this._textures;\r\n }\r\n\r\n /**\r\n * Gets the number of textures in this MRT. This number can be different from `_textures.length` in case a depth texture is generated.\r\n */\r\n public get count(): number {\r\n return this._count;\r\n }\r\n\r\n /**\r\n * Get the depth texture generated by the multi render target if options.generateDepthTexture has been set\r\n */\r\n public get depthTexture(): Texture {\r\n return this._textures[this._textures.length - 1];\r\n }\r\n\r\n /**\r\n * Set the wrapping mode on U of all the textures we are rendering to.\r\n * Can be any of the Texture. (CLAMP_ADDRESSMODE, MIRROR_ADDRESSMODE or WRAP_ADDRESSMODE)\r\n */\r\n public set wrapU(wrap: number) {\r\n if (this._textures) {\r\n for (let i = 0; i < this._textures.length; i++) {\r\n this._textures[i].wrapU = wrap;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Set the wrapping mode on V of all the textures we are rendering to.\r\n * Can be any of the Texture. (CLAMP_ADDRESSMODE, MIRROR_ADDRESSMODE or WRAP_ADDRESSMODE)\r\n */\r\n public set wrapV(wrap: number) {\r\n if (this._textures) {\r\n for (let i = 0; i < this._textures.length; i++) {\r\n this._textures[i].wrapV = wrap;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Instantiate a new multi render target texture.\r\n * A multi render target, like a render target provides the ability to render to a texture.\r\n * Unlike the render target, it can render to several draw buffers in one draw.\r\n * This is specially interesting in deferred rendering or for any effects requiring more than\r\n * just one color from a single pass.\r\n * @param name Define the name of the texture\r\n * @param size Define the size of the buffers to render to\r\n * @param count Define the number of target we are rendering into\r\n * @param scene Define the scene the texture belongs to\r\n * @param options Define the options used to create the multi render target\r\n * @param textureNames Define the names to set to the textures (if count > 0 - optional)\r\n */\r\n constructor(name: string, size: any, count: number, scene?: Scene, options?: IMultiRenderTargetOptions, textureNames?: string[]) {\r\n const generateMipMaps = options && options.generateMipMaps ? options.generateMipMaps : false;\r\n const generateDepthTexture = options && options.generateDepthTexture ? options.generateDepthTexture : false;\r\n const depthTextureFormat = options && options.depthTextureFormat ? options.depthTextureFormat : Constants.TEXTUREFORMAT_DEPTH16;\r\n const doNotChangeAspectRatio = !options || options.doNotChangeAspectRatio === undefined ? true : options.doNotChangeAspectRatio;\r\n const drawOnlyOnFirstAttachmentByDefault = options && options.drawOnlyOnFirstAttachmentByDefault ? options.drawOnlyOnFirstAttachmentByDefault : false;\r\n super(name, size, scene, generateMipMaps, doNotChangeAspectRatio, undefined, undefined, undefined, undefined, undefined, undefined, undefined, true);\r\n\r\n if (!this.isSupported) {\r\n this.dispose();\r\n return;\r\n }\r\n\r\n this._textureNames = textureNames;\r\n\r\n const types: number[] = [];\r\n const samplingModes: number[] = [];\r\n const useSRGBBuffers: boolean[] = [];\r\n const formats: number[] = [];\r\n const targetTypes: number[] = [];\r\n const faceIndex: number[] = [];\r\n const layerIndex: number[] = [];\r\n const layerCounts: number[] = [];\r\n this._initTypes(count, types, samplingModes, useSRGBBuffers, formats, targetTypes, faceIndex, layerIndex, layerCounts, options);\r\n\r\n const generateDepthBuffer = !options || options.generateDepthBuffer === undefined ? true : options.generateDepthBuffer;\r\n const generateStencilBuffer = !options || options.generateStencilBuffer === undefined ? false : options.generateStencilBuffer;\r\n\r\n this._multiRenderTargetOptions = {\r\n samplingModes: samplingModes,\r\n generateMipMaps: generateMipMaps,\r\n generateDepthBuffer: generateDepthBuffer,\r\n generateStencilBuffer: generateStencilBuffer,\r\n generateDepthTexture: generateDepthTexture,\r\n depthTextureFormat: depthTextureFormat,\r\n types: types,\r\n textureCount: count,\r\n useSRGBBuffers: useSRGBBuffers,\r\n formats: formats,\r\n targetTypes: targetTypes,\r\n faceIndex: faceIndex,\r\n layerIndex: layerIndex,\r\n layerCounts: layerCounts,\r\n };\r\n\r\n this._count = count;\r\n this._drawOnlyOnFirstAttachmentByDefault = drawOnlyOnFirstAttachmentByDefault;\r\n\r\n if (count > 0) {\r\n this._createInternalTextures();\r\n this._createTextures(textureNames);\r\n }\r\n }\r\n\r\n private _initTypes(\r\n count: number,\r\n types: number[],\r\n samplingModes: number[],\r\n useSRGBBuffers: boolean[],\r\n formats: number[],\r\n targets: number[],\r\n faceIndex: number[],\r\n layerIndex: number[],\r\n layerCounts: number[],\r\n options?: IMultiRenderTargetOptions\r\n ) {\r\n for (let i = 0; i < count; i++) {\r\n if (options && options.types && options.types[i] !== undefined) {\r\n types.push(options.types[i]);\r\n } else {\r\n types.push(options && options.defaultType ? options.defaultType : Constants.TEXTURETYPE_UNSIGNED_INT);\r\n }\r\n\r\n if (options && options.samplingModes && options.samplingModes[i] !== undefined) {\r\n samplingModes.push(options.samplingModes[i]);\r\n } else {\r\n samplingModes.push(Texture.BILINEAR_SAMPLINGMODE);\r\n }\r\n\r\n if (options && options.useSRGBBuffers && options.useSRGBBuffers[i] !== undefined) {\r\n useSRGBBuffers.push(options.useSRGBBuffers[i]);\r\n } else {\r\n useSRGBBuffers.push(false);\r\n }\r\n\r\n if (options && options.formats && options.formats[i] !== undefined) {\r\n formats.push(options.formats[i]);\r\n } else {\r\n formats.push(Constants.TEXTUREFORMAT_RGBA);\r\n }\r\n\r\n if (options && options.targetTypes && options.targetTypes[i] !== undefined) {\r\n targets.push(options.targetTypes[i]);\r\n } else {\r\n targets.push(Constants.TEXTURE_2D);\r\n }\r\n\r\n if (options && options.faceIndex && options.faceIndex[i] !== undefined) {\r\n faceIndex.push(options.faceIndex[i]);\r\n } else {\r\n faceIndex.push(0);\r\n }\r\n\r\n if (options && options.layerIndex && options.layerIndex[i] !== undefined) {\r\n layerIndex.push(options.layerIndex[i]);\r\n } else {\r\n layerIndex.push(0);\r\n }\r\n\r\n if (options && options.layerCounts && options.layerCounts[i] !== undefined) {\r\n layerCounts.push(options.layerCounts[i]);\r\n } else {\r\n layerCounts.push(1);\r\n }\r\n }\r\n }\r\n\r\n private _createInternaTextureIndexMapping() {\r\n const mapMainInternalTexture2Index: { [key: number]: number } = {};\r\n const mapInternalTexture2MainIndex: number[] = [];\r\n\r\n if (!this._renderTarget) {\r\n return mapInternalTexture2MainIndex;\r\n }\r\n\r\n const internalTextures = this._renderTarget!.textures!;\r\n for (let i = 0; i < internalTextures.length; i++) {\r\n const texture = internalTextures[i];\r\n if (!texture) {\r\n continue;\r\n }\r\n const mainIndex = mapMainInternalTexture2Index[texture.uniqueId];\r\n if (mainIndex !== undefined) {\r\n mapInternalTexture2MainIndex[i] = mainIndex;\r\n } else {\r\n mapMainInternalTexture2Index[texture.uniqueId] = i;\r\n }\r\n }\r\n\r\n return mapInternalTexture2MainIndex;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _rebuild(forceFullRebuild: boolean = false, textureNames?: string[]): void {\r\n if (this._count < 1) {\r\n return;\r\n }\r\n\r\n const mapInternalTexture2MainIndex = this._createInternaTextureIndexMapping();\r\n\r\n this.releaseInternalTextures();\r\n this._createInternalTextures();\r\n\r\n if (forceFullRebuild) {\r\n this._releaseTextures();\r\n this._createTextures(textureNames);\r\n }\r\n\r\n const internalTextures = this._renderTarget!.textures!;\r\n for (let i = 0; i < internalTextures.length; i++) {\r\n const texture = this._textures[i];\r\n if (mapInternalTexture2MainIndex[i] !== undefined) {\r\n this._renderTarget!.setTexture(internalTextures[mapInternalTexture2MainIndex[i]], i);\r\n }\r\n texture._texture = internalTextures[i];\r\n if (texture._texture) {\r\n texture._noMipmap = !texture._texture.useMipMaps;\r\n texture._useSRGBBuffer = texture._texture._useSRGBBuffer;\r\n }\r\n }\r\n\r\n if (this.samples !== 1) {\r\n this._renderTarget!.setSamples(this.samples, !this._drawOnlyOnFirstAttachmentByDefault, true);\r\n }\r\n }\r\n\r\n private _createInternalTextures(): void {\r\n this._renderTarget = this._getEngine()!.createMultipleRenderTarget(this._size, this._multiRenderTargetOptions, !this._drawOnlyOnFirstAttachmentByDefault);\r\n this._texture = this._renderTarget.texture;\r\n }\r\n\r\n private _releaseTextures(): void {\r\n if (this._textures) {\r\n for (let i = 0; i < this._textures.length; i++) {\r\n this._textures[i]._texture = null; // internal textures are released by a call to releaseInternalTextures()\r\n this._textures[i].dispose();\r\n }\r\n }\r\n }\r\n\r\n private _createTextures(textureNames?: string[]): void {\r\n const internalTextures = this._renderTarget!.textures!;\r\n this._textures = [];\r\n for (let i = 0; i < internalTextures.length; i++) {\r\n const texture = new Texture(null, this.getScene());\r\n if (textureNames?.[i]) {\r\n texture.name = textureNames[i];\r\n }\r\n texture._texture = internalTextures[i];\r\n if (texture._texture) {\r\n texture._noMipmap = !texture._texture.useMipMaps;\r\n texture._useSRGBBuffer = texture._texture._useSRGBBuffer;\r\n }\r\n this._textures.push(texture);\r\n }\r\n }\r\n\r\n /**\r\n * Replaces an internal texture within the MRT. Useful to share textures between MultiRenderTarget.\r\n * @param texture The new texture to set in the MRT\r\n * @param index The index of the texture to replace\r\n * @param disposePrevious Set to true if the previous internal texture should be disposed\r\n */\r\n public setInternalTexture(texture: InternalTexture, index: number, disposePrevious: boolean = true) {\r\n if (!this.renderTarget) {\r\n return;\r\n }\r\n\r\n if (index === 0) {\r\n this._texture = texture;\r\n }\r\n\r\n this.renderTarget.setTexture(texture, index, disposePrevious);\r\n\r\n if (!this.textures[index]) {\r\n this.textures[index] = new Texture(null, this.getScene());\r\n this.textures[index].name = this._textureNames?.[index] ?? this.textures[index].name;\r\n }\r\n this.textures[index]._texture = texture;\r\n this.textures[index]._noMipmap = !texture.useMipMaps;\r\n this.textures[index]._useSRGBBuffer = texture._useSRGBBuffer;\r\n\r\n this._count = this.renderTarget.textures ? this.renderTarget.textures.length : 0;\r\n\r\n if (this._multiRenderTargetOptions.types) {\r\n this._multiRenderTargetOptions.types[index] = texture.type;\r\n }\r\n if (this._multiRenderTargetOptions.samplingModes) {\r\n this._multiRenderTargetOptions.samplingModes[index] = texture.samplingMode;\r\n }\r\n if (this._multiRenderTargetOptions.useSRGBBuffers) {\r\n this._multiRenderTargetOptions.useSRGBBuffers[index] = texture._useSRGBBuffer;\r\n }\r\n if (this._multiRenderTargetOptions.targetTypes && this._multiRenderTargetOptions.targetTypes[index] !== -1) {\r\n let target: number = 0;\r\n if (texture.is2DArray) {\r\n target = Constants.TEXTURE_2D_ARRAY;\r\n } else if (texture.isCube) {\r\n target = Constants.TEXTURE_CUBE_MAP;\r\n } /*else if (texture.isCubeArray) {\r\n target = Constants.TEXTURE_CUBE_MAP_ARRAY;\r\n }*/ else if (texture.is3D) {\r\n target = Constants.TEXTURE_3D;\r\n } else {\r\n target = Constants.TEXTURE_2D;\r\n }\r\n this._multiRenderTargetOptions.targetTypes[index] = target;\r\n }\r\n }\r\n\r\n /**\r\n * Changes an attached texture's face index or layer.\r\n * @param index The index of the texture to modify the attachment of\r\n * @param layerIndex The layer index of the texture to be attached to the framebuffer\r\n * @param faceIndex The face index of the texture to be attached to the framebuffer\r\n */\r\n public setLayerAndFaceIndex(index: number, layerIndex: number = -1, faceIndex: number = -1) {\r\n if (!this.textures[index] || !this.renderTarget) {\r\n return;\r\n }\r\n\r\n if (this._multiRenderTargetOptions.layerIndex) {\r\n this._multiRenderTargetOptions.layerIndex[index] = layerIndex;\r\n }\r\n if (this._multiRenderTargetOptions.faceIndex) {\r\n this._multiRenderTargetOptions.faceIndex[index] = faceIndex;\r\n }\r\n\r\n this.renderTarget.setLayerAndFaceIndex(index, layerIndex, faceIndex);\r\n }\r\n\r\n /**\r\n * Changes every attached texture's face index or layer.\r\n * @param layerIndices The layer indices of the texture to be attached to the framebuffer\r\n * @param faceIndices The face indices of the texture to be attached to the framebuffer\r\n */\r\n public setLayerAndFaceIndices(layerIndices: number[], faceIndices: number[]) {\r\n if (!this.renderTarget) {\r\n return;\r\n }\r\n\r\n this._multiRenderTargetOptions.layerIndex = layerIndices;\r\n this._multiRenderTargetOptions.faceIndex = faceIndices;\r\n\r\n this.renderTarget.setLayerAndFaceIndices(layerIndices, faceIndices);\r\n }\r\n\r\n /**\r\n * Define the number of samples used if MSAA is enabled.\r\n */\r\n public get samples(): number {\r\n return this._samples;\r\n }\r\n\r\n public set samples(value: number) {\r\n if (this._renderTarget) {\r\n this._samples = this._renderTarget.setSamples(value);\r\n } else {\r\n // In case samples are set with 0 textures created, we must save the desired samples value\r\n this._samples = value;\r\n }\r\n }\r\n\r\n /**\r\n * Resize all the textures in the multi render target.\r\n * Be careful as it will recreate all the data in the new texture.\r\n * @param size Define the new size\r\n */\r\n public resize(size: any) {\r\n this._processSizeParameter(size, false);\r\n this._rebuild(undefined, this._textureNames);\r\n }\r\n\r\n /**\r\n * Changes the number of render targets in this MRT\r\n * Be careful as it will recreate all the data in the new texture.\r\n * @param count new texture count\r\n * @param options Specifies texture types and sampling modes for new textures\r\n * @param textureNames Specifies the names of the textures (optional)\r\n */\r\n public updateCount(count: number, options?: IMultiRenderTargetOptions, textureNames?: string[]) {\r\n this._multiRenderTargetOptions.textureCount = count;\r\n this._count = count;\r\n\r\n const types: number[] = [];\r\n const samplingModes: number[] = [];\r\n const useSRGBBuffers: boolean[] = [];\r\n const formats: number[] = [];\r\n const targetTypes: number[] = [];\r\n const faceIndex: number[] = [];\r\n const layerIndex: number[] = [];\r\n const layerCounts: number[] = [];\r\n\r\n this._textureNames = textureNames;\r\n\r\n this._initTypes(count, types, samplingModes, useSRGBBuffers, formats, targetTypes, faceIndex, layerIndex, layerCounts, options);\r\n this._multiRenderTargetOptions.types = types;\r\n this._multiRenderTargetOptions.samplingModes = samplingModes;\r\n this._multiRenderTargetOptions.useSRGBBuffers = useSRGBBuffers;\r\n this._multiRenderTargetOptions.formats = formats;\r\n this._multiRenderTargetOptions.targetTypes = targetTypes;\r\n this._multiRenderTargetOptions.faceIndex = faceIndex;\r\n this._multiRenderTargetOptions.layerIndex = layerIndex;\r\n this._multiRenderTargetOptions.layerCounts = layerCounts;\r\n\r\n this._rebuild(true, textureNames);\r\n }\r\n\r\n protected _unbindFrameBuffer(engine: Engine, faceIndex: number): void {\r\n if (this._renderTarget) {\r\n engine.unBindMultiColorAttachmentFramebuffer(this._renderTarget, this.isCube, () => {\r\n this.onAfterRenderObservable.notifyObservers(faceIndex);\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Dispose the render targets and their associated resources\r\n * @param doNotDisposeInternalTextures\r\n */\r\n public dispose(doNotDisposeInternalTextures = false): void {\r\n this._releaseTextures();\r\n if (!doNotDisposeInternalTextures) {\r\n this.releaseInternalTextures();\r\n } else {\r\n // Prevent internal texture dispose in super.dispose\r\n this._texture = null;\r\n }\r\n super.dispose();\r\n }\r\n\r\n /**\r\n * Release all the underlying texture used as draw buffers.\r\n */\r\n public releaseInternalTextures(): void {\r\n const internalTextures = this._renderTarget?.textures;\r\n\r\n if (!internalTextures) {\r\n return;\r\n }\r\n\r\n for (let i = internalTextures.length - 1; i >= 0; i--) {\r\n this._textures[i]._texture = null;\r\n }\r\n\r\n this._renderTarget?.dispose();\r\n this._renderTarget = null;\r\n }\r\n}\r\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"mrtFragmentDeclaration\";\nconst shader = `#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE)\nlayout(location=0) out vec4 glFragData[{X}];\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const mrtFragmentDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/clipPlaneFragmentDeclaration\";\nimport \"./ShadersInclude/mrtFragmentDeclaration\";\nimport \"./ShadersInclude/bumpFragmentMainFunctions\";\nimport \"./ShadersInclude/bumpFragmentFunctions\";\nimport \"./ShadersInclude/helperFunctions\";\nimport \"./ShadersInclude/clipPlaneFragment\";\nimport \"./ShadersInclude/bumpFragment\";\n\nconst name = \"geometryPixelShader\";\nconst shader = `#extension GL_EXT_draw_buffers : require\n#if defined(BUMP) || !defined(NORMAL)\n#extension GL_OES_standard_derivatives : enable\n#endif\nprecision highp float;\n#ifdef BUMP\nvarying mat4 vWorldView;varying vec3 vNormalW;\n#else\nvarying vec3 vNormalV;\n#endif\nvarying vec4 vViewPos;\n#if defined(POSITION) || defined(BUMP)\nvarying vec3 vPositionW;\n#endif\n#ifdef VELOCITY\nvarying vec4 vCurrentPosition;varying vec4 vPreviousPosition;\n#endif\n#ifdef NEED_UV\nvarying vec2 vUV;\n#endif\n#ifdef BUMP\nuniform vec3 vBumpInfos;uniform vec2 vTangentSpaceParams;\n#endif\n#if defined(REFLECTIVITY)\n#if defined(ORMTEXTURE) || defined(SPECULARGLOSSINESSTEXTURE) || defined(REFLECTIVITYTEXTURE)\nuniform sampler2D reflectivitySampler;varying vec2 vReflectivityUV;\n#endif\n#ifdef ALBEDOTEXTURE\nvarying vec2 vAlbedoUV;uniform sampler2D albedoSampler;\n#endif\n#ifdef REFLECTIVITYCOLOR\nuniform vec3 reflectivityColor;\n#endif\n#ifdef ALBEDOCOLOR\nuniform vec3 albedoColor;\n#endif\n#ifdef METALLIC\nuniform float metallic;\n#endif\n#if defined(ROUGHNESS) || defined(GLOSSINESS)\nuniform float glossiness;\n#endif\n#endif\n#if defined(ALPHATEST) && defined(NEED_UV)\nuniform sampler2D diffuseSampler;\n#endif\n#include\n#include[RENDER_TARGET_COUNT]\n#include\n#include\n#include\nvoid main() {\n#include\n#ifdef ALPHATEST\nif (texture2D(diffuseSampler,vUV).a<0.4)\ndiscard;\n#endif\nvec3 normalOutput;\n#ifdef BUMP\nvec3 normalW=normalize(vNormalW);\n#include\nnormalOutput=normalize(vec3(vWorldView*vec4(normalW,0.0)));\n#else\nnormalOutput=normalize(vNormalV);\n#endif\n#ifdef PREPASS\n#ifdef PREPASS_DEPTH\ngl_FragData[DEPTH_INDEX]=vec4(vViewPos.z/vViewPos.w,0.0,0.0,1.0);\n#endif\n#ifdef PREPASS_NORMAL\ngl_FragData[NORMAL_INDEX]=vec4(normalOutput,1.0);\n#endif\n#else\ngl_FragData[0]=vec4(vViewPos.z/vViewPos.w,0.0,0.0,1.0);gl_FragData[1]=vec4(normalOutput,1.0);\n#endif\n#ifdef POSITION\ngl_FragData[POSITION_INDEX]=vec4(vPositionW,1.0);\n#endif\n#ifdef VELOCITY\nvec2 a=(vCurrentPosition.xy/vCurrentPosition.w)*0.5+0.5;vec2 b=(vPreviousPosition.xy/vPreviousPosition.w)*0.5+0.5;vec2 velocity=abs(a-b);velocity=vec2(pow(velocity.x,1.0/3.0),pow(velocity.y,1.0/3.0))*sign(a-b)*0.5+0.5;gl_FragData[VELOCITY_INDEX]=vec4(velocity,0.0,1.0);\n#endif\n#ifdef REFLECTIVITY\nvec4 reflectivity=vec4(0.0,0.0,0.0,1.0);\n#ifdef METALLICWORKFLOW\nfloat metal=1.0;float roughness=1.0;\n#ifdef ORMTEXTURE\nmetal*=texture2D(reflectivitySampler,vReflectivityUV).b;roughness*=texture2D(reflectivitySampler,vReflectivityUV).g;\n#endif\n#ifdef METALLIC\nmetal*=metallic;\n#endif\n#ifdef ROUGHNESS\nroughness*=(1.0-glossiness); \n#endif\nreflectivity.a-=roughness;vec3 color=vec3(1.0);\n#ifdef ALBEDOTEXTURE\ncolor=texture2D(albedoSampler,vAlbedoUV).rgb;\n#ifdef GAMMAALBEDO\ncolor=toLinearSpace(color);\n#endif\n#endif\n#ifdef ALBEDOCOLOR\ncolor*=albedoColor.xyz;\n#endif\nreflectivity.rgb=mix(vec3(0.04),color,metal);\n#else\n#if defined(SPECULARGLOSSINESSTEXTURE) || defined(REFLECTIVITYTEXTURE)\nreflectivity=texture2D(reflectivitySampler,vReflectivityUV);\n#ifdef GAMMAREFLECTIVITYTEXTURE\nreflectivity.rgb=toLinearSpace(reflectivity.rgb);\n#endif\n#else \n#ifdef REFLECTIVITYCOLOR\nreflectivity.rgb=toLinearSpace(reflectivityColor.xyz);reflectivity.a=1.0;\n#endif\n#endif\n#ifdef GLOSSINESSS\nreflectivity.a*=glossiness; \n#endif\n#endif\ngl_FragData[REFLECTIVITY_INDEX]=reflectivity;\n#endif\n}\n`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const geometryPixelShader = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"geometryVertexDeclaration\";\nconst shader = `uniform mat4 viewProjection;uniform mat4 view;`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const geometryVertexDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\nimport \"./sceneUboDeclaration\";\n\nconst name = \"geometryUboDeclaration\";\nconst shader = `#include\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const geometryUboDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/bonesDeclaration\";\nimport \"./ShadersInclude/bakedVertexAnimationDeclaration\";\nimport \"./ShadersInclude/morphTargetsVertexGlobalDeclaration\";\nimport \"./ShadersInclude/morphTargetsVertexDeclaration\";\nimport \"./ShadersInclude/instancesDeclaration\";\nimport \"./ShadersInclude/geometryVertexDeclaration\";\nimport \"./ShadersInclude/geometryUboDeclaration\";\nimport \"./ShadersInclude/clipPlaneVertexDeclaration\";\nimport \"./ShadersInclude/morphTargetsVertexGlobal\";\nimport \"./ShadersInclude/morphTargetsVertex\";\nimport \"./ShadersInclude/instancesVertex\";\nimport \"./ShadersInclude/bonesVertex\";\nimport \"./ShadersInclude/bakedVertexAnimation\";\nimport \"./ShadersInclude/clipPlaneVertex\";\nimport \"./ShadersInclude/bumpVertex\";\n\nconst name = \"geometryVertexShader\";\nconst shader = `precision highp float;\n#include\n#include\n#include\n#include[0..maxSimultaneousMorphTargets]\n#include\n#include<__decl__geometryVertex>\n#include\nattribute vec3 position;attribute vec3 normal;\n#ifdef NEED_UV\nvarying vec2 vUV;\n#ifdef ALPHATEST\nuniform mat4 diffuseMatrix;\n#endif\n#ifdef BUMP\nuniform mat4 bumpMatrix;varying vec2 vBumpUV;\n#endif\n#ifdef REFLECTIVITY\nuniform mat4 reflectivityMatrix;uniform mat4 albedoMatrix;varying vec2 vReflectivityUV;varying vec2 vAlbedoUV;\n#endif\n#ifdef UV1\nattribute vec2 uv;\n#endif\n#ifdef UV2\nattribute vec2 uv2;\n#endif\n#endif\n#ifdef BUMP\nvarying mat4 vWorldView;\n#endif\n#ifdef BUMP\nvarying vec3 vNormalW;\n#else\nvarying vec3 vNormalV;\n#endif\nvarying vec4 vViewPos;\n#if defined(POSITION) || defined(BUMP)\nvarying vec3 vPositionW;\n#endif\n#ifdef VELOCITY\nuniform mat4 previousViewProjection;varying vec4 vCurrentPosition;varying vec4 vPreviousPosition;\n#endif\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void)\n{vec3 positionUpdated=position;vec3 normalUpdated=normal;\n#ifdef UV1\nvec2 uvUpdated=uv;\n#endif\n#include\n#include[0..maxSimultaneousMorphTargets]\n#include\n#if defined(VELOCITY) && !defined(BONES_VELOCITY_ENABLED)\nvCurrentPosition=viewProjection*finalWorld*vec4(positionUpdated,1.0);vPreviousPosition=previousViewProjection*finalPreviousWorld*vec4(positionUpdated,1.0);\n#endif\n#include\n#include\nvec4 worldPos=vec4(finalWorld*vec4(positionUpdated,1.0));\n#ifdef BUMP\nvWorldView=view*finalWorld;vNormalW=normalUpdated;\n#else\nvNormalV=normalize(vec3((view*finalWorld)*vec4(normalUpdated,0.0)));\n#endif\nvViewPos=view*worldPos;\n#if defined(VELOCITY) && defined(BONES_VELOCITY_ENABLED)\nvCurrentPosition=viewProjection*finalWorld*vec4(positionUpdated,1.0);\n#if NUM_BONE_INFLUENCERS>0\nmat4 previousInfluence;previousInfluence=mPreviousBones[int(matricesIndices[0])]*matricesWeights[0];\n#if NUM_BONE_INFLUENCERS>1\npreviousInfluence+=mPreviousBones[int(matricesIndices[1])]*matricesWeights[1];\n#endif\n#if NUM_BONE_INFLUENCERS>2\npreviousInfluence+=mPreviousBones[int(matricesIndices[2])]*matricesWeights[2];\n#endif\n#if NUM_BONE_INFLUENCERS>3\npreviousInfluence+=mPreviousBones[int(matricesIndices[3])]*matricesWeights[3];\n#endif\n#if NUM_BONE_INFLUENCERS>4\npreviousInfluence+=mPreviousBones[int(matricesIndicesExtra[0])]*matricesWeightsExtra[0];\n#endif\n#if NUM_BONE_INFLUENCERS>5\npreviousInfluence+=mPreviousBones[int(matricesIndicesExtra[1])]*matricesWeightsExtra[1];\n#endif\n#if NUM_BONE_INFLUENCERS>6\npreviousInfluence+=mPreviousBones[int(matricesIndicesExtra[2])]*matricesWeightsExtra[2];\n#endif\n#if NUM_BONE_INFLUENCERS>7\npreviousInfluence+=mPreviousBones[int(matricesIndicesExtra[3])]*matricesWeightsExtra[3];\n#endif\nvPreviousPosition=previousViewProjection*finalPreviousWorld*previousInfluence*vec4(positionUpdated,1.0);\n#else\nvPreviousPosition=previousViewProjection*finalPreviousWorld*vec4(positionUpdated,1.0);\n#endif\n#endif\n#if defined(POSITION) || defined(BUMP)\nvPositionW=worldPos.xyz/worldPos.w;\n#endif\ngl_Position=viewProjection*finalWorld*vec4(positionUpdated,1.0);\n#include\n#ifdef NEED_UV\n#ifdef UV1\n#if defined(ALPHATEST) && defined(ALPHATEST_UV1)\nvUV=vec2(diffuseMatrix*vec4(uvUpdated,1.0,0.0));\n#else\nvUV=uv;\n#endif\n#ifdef BUMP_UV1\nvBumpUV=vec2(bumpMatrix*vec4(uvUpdated,1.0,0.0));\n#endif\n#ifdef REFLECTIVITY_UV1\nvReflectivityUV=vec2(reflectivityMatrix*vec4(uvUpdated,1.0,0.0));\n#endif\n#ifdef ALBEDO_UV1\nvAlbedoUV=vec2(albedoMatrix*vec4(uvUpdated,1.0,0.0));\n#endif\n#endif\n#ifdef UV2\n#if defined(ALPHATEST) && defined(ALPHATEST_UV2)\nvUV=vec2(diffuseMatrix*vec4(uv2,1.0,0.0));\n#else\nvUV=uv2;\n#endif\n#ifdef BUMP_UV2\nvBumpUV=vec2(bumpMatrix*vec4(uv2,1.0,0.0));\n#endif\n#ifdef REFLECTIVITY_UV2\nvReflectivityUV=vec2(reflectivityMatrix*vec4(uv2,1.0,0.0));\n#endif\n#ifdef ALBEDO_UV2\nvAlbedoUV=vec2(albedoMatrix*vec4(uv2,1.0,0.0));\n#endif\n#endif\n#endif\n#include\n}\n`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const geometryVertexShader = { name, shader };\n","import { Matrix } from \"../Maths/math.vector\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport type { SubMesh } from \"../Meshes/subMesh\";\r\nimport type { Mesh } from \"../Meshes/mesh\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport type { SmartArray } from \"../Misc/smartArray\";\r\nimport { Texture } from \"../Materials/Textures/texture\";\r\nimport type { InternalTexture } from \"../Materials/Textures/internalTexture\";\r\nimport { MultiRenderTarget } from \"../Materials/Textures/multiRenderTarget\";\r\nimport type { PrePassRenderer } from \"../Rendering/prePassRenderer\";\r\nimport { MaterialHelper } from \"../Materials/materialHelper\";\r\nimport type { Scene } from \"../scene\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport { Color4 } from \"../Maths/math.color\";\r\nimport { _WarnImport } from \"../Misc/devTools\";\r\nimport type { Observer } from \"../Misc/observable\";\r\nimport type { Engine } from \"../Engines/engine\";\r\nimport type { Nullable } from \"../types\";\r\nimport { Material } from \"../Materials/material\";\r\n\r\nimport \"../Shaders/geometry.fragment\";\r\nimport \"../Shaders/geometry.vertex\";\r\nimport { MaterialFlags } from \"../Materials/materialFlags\";\r\nimport { addClipPlaneUniforms, bindClipPlane, prepareStringDefinesForClipPlanes } from \"../Materials/clipPlaneMaterialHelper\";\r\n\r\n/** @internal */\r\ninterface ISavedTransformationMatrix {\r\n world: Matrix;\r\n viewProjection: Matrix;\r\n}\r\n\r\n/** list the uniforms used by the geometry renderer */\r\nconst uniforms = [\r\n \"world\",\r\n \"mBones\",\r\n \"viewProjection\",\r\n \"diffuseMatrix\",\r\n \"view\",\r\n \"previousWorld\",\r\n \"previousViewProjection\",\r\n \"mPreviousBones\",\r\n \"bumpMatrix\",\r\n \"reflectivityMatrix\",\r\n \"albedoMatrix\",\r\n \"reflectivityColor\",\r\n \"albedoColor\",\r\n \"metallic\",\r\n \"glossiness\",\r\n \"vTangentSpaceParams\",\r\n \"vBumpInfos\",\r\n \"morphTargetInfluences\",\r\n \"morphTargetTextureInfo\",\r\n \"morphTargetTextureIndices\",\r\n];\r\naddClipPlaneUniforms(uniforms);\r\n\r\n/**\r\n * This renderer is helpful to fill one of the render target with a geometry buffer.\r\n */\r\nexport class GeometryBufferRenderer {\r\n /**\r\n * Constant used to retrieve the depth texture index in the G-Buffer textures array\r\n * using getIndex(GeometryBufferRenderer.DEPTH_TEXTURE_INDEX)\r\n */\r\n public static readonly DEPTH_TEXTURE_TYPE = 0;\r\n /**\r\n * Constant used to retrieve the normal texture index in the G-Buffer textures array\r\n * using getIndex(GeometryBufferRenderer.NORMAL_TEXTURE_INDEX)\r\n */\r\n public static readonly NORMAL_TEXTURE_TYPE = 1;\r\n /**\r\n * Constant used to retrieve the position texture index in the G-Buffer textures array\r\n * using getIndex(GeometryBufferRenderer.POSITION_TEXTURE_INDEX)\r\n */\r\n public static readonly POSITION_TEXTURE_TYPE = 2;\r\n /**\r\n * Constant used to retrieve the velocity texture index in the G-Buffer textures array\r\n * using getIndex(GeometryBufferRenderer.VELOCITY_TEXTURE_INDEX)\r\n */\r\n public static readonly VELOCITY_TEXTURE_TYPE = 3;\r\n /**\r\n * Constant used to retrieve the reflectivity texture index in the G-Buffer textures array\r\n * using the getIndex(GeometryBufferRenderer.REFLECTIVITY_TEXTURE_TYPE)\r\n */\r\n public static readonly REFLECTIVITY_TEXTURE_TYPE = 4;\r\n\r\n /**\r\n * Dictionary used to store the previous transformation matrices of each rendered mesh\r\n * in order to compute objects velocities when enableVelocity is set to \"true\"\r\n * @internal\r\n */\r\n public _previousTransformationMatrices: { [index: number]: ISavedTransformationMatrix } = {};\r\n /**\r\n * Dictionary used to store the previous bones transformation matrices of each rendered mesh\r\n * in order to compute objects velocities when enableVelocity is set to \"true\"\r\n * @internal\r\n */\r\n public _previousBonesTransformationMatrices: { [index: number]: Float32Array } = {};\r\n /**\r\n * Array used to store the ignored skinned meshes while computing velocity map (typically used by the motion blur post-process).\r\n * Avoids computing bones velocities and computes only mesh's velocity itself (position, rotation, scaling).\r\n */\r\n public excludedSkinnedMeshesFromVelocity: AbstractMesh[] = [];\r\n\r\n /** Gets or sets a boolean indicating if transparent meshes should be rendered */\r\n public renderTransparentMeshes = true;\r\n\r\n private _scene: Scene;\r\n private _resizeObserver: Nullable> = null;\r\n private _multiRenderTarget: MultiRenderTarget;\r\n private _ratio: number;\r\n private _enablePosition: boolean = false;\r\n private _enableVelocity: boolean = false;\r\n private _enableReflectivity: boolean = false;\r\n private _depthFormat: number;\r\n private _clearColor = new Color4(0, 0, 0, 0);\r\n private _clearDepthColor = new Color4(1e8, 0, 0, 1); // \"infinity\" value - depth in the depth texture is view.z, not a 0..1 value!\r\n\r\n private _positionIndex: number = -1;\r\n private _velocityIndex: number = -1;\r\n private _reflectivityIndex: number = -1;\r\n private _depthIndex: number = -1;\r\n private _normalIndex: number = -1;\r\n\r\n private _linkedWithPrePass: boolean = false;\r\n private _prePassRenderer: PrePassRenderer;\r\n private _attachmentsFromPrePass: number[];\r\n private _useUbo: boolean;\r\n\r\n protected _cachedDefines: string;\r\n\r\n /**\r\n * @internal\r\n * Sets up internal structures to share outputs with PrePassRenderer\r\n * This method should only be called by the PrePassRenderer itself\r\n */\r\n public _linkPrePassRenderer(prePassRenderer: PrePassRenderer) {\r\n this._linkedWithPrePass = true;\r\n this._prePassRenderer = prePassRenderer;\r\n\r\n if (this._multiRenderTarget) {\r\n // prevents clearing of the RT since it's done by prepass\r\n this._multiRenderTarget.onClearObservable.clear();\r\n this._multiRenderTarget.onClearObservable.add(() => {\r\n // pass\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n * Separates internal structures from PrePassRenderer so the geometry buffer can now operate by itself.\r\n * This method should only be called by the PrePassRenderer itself\r\n */\r\n public _unlinkPrePassRenderer() {\r\n this._linkedWithPrePass = false;\r\n this._createRenderTargets();\r\n }\r\n\r\n /**\r\n * @internal\r\n * Resets the geometry buffer layout\r\n */\r\n public _resetLayout() {\r\n this._enablePosition = false;\r\n this._enableReflectivity = false;\r\n this._enableVelocity = false;\r\n this._attachmentsFromPrePass = [];\r\n }\r\n\r\n /**\r\n * @internal\r\n * Replaces a texture in the geometry buffer renderer\r\n * Useful when linking textures of the prepass renderer\r\n */\r\n public _forceTextureType(geometryBufferType: number, index: number) {\r\n if (geometryBufferType === GeometryBufferRenderer.POSITION_TEXTURE_TYPE) {\r\n this._positionIndex = index;\r\n this._enablePosition = true;\r\n } else if (geometryBufferType === GeometryBufferRenderer.VELOCITY_TEXTURE_TYPE) {\r\n this._velocityIndex = index;\r\n this._enableVelocity = true;\r\n } else if (geometryBufferType === GeometryBufferRenderer.REFLECTIVITY_TEXTURE_TYPE) {\r\n this._reflectivityIndex = index;\r\n this._enableReflectivity = true;\r\n } else if (geometryBufferType === GeometryBufferRenderer.DEPTH_TEXTURE_TYPE) {\r\n this._depthIndex = index;\r\n } else if (geometryBufferType === GeometryBufferRenderer.NORMAL_TEXTURE_TYPE) {\r\n this._normalIndex = index;\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n * Sets texture attachments\r\n * Useful when linking textures of the prepass renderer\r\n */\r\n public _setAttachments(attachments: number[]) {\r\n this._attachmentsFromPrePass = attachments;\r\n }\r\n\r\n /**\r\n * @internal\r\n * Replaces the first texture which is hard coded as a depth texture in the geometry buffer\r\n * Useful when linking textures of the prepass renderer\r\n */\r\n public _linkInternalTexture(internalTexture: InternalTexture) {\r\n this._multiRenderTarget.setInternalTexture(internalTexture, 0, false);\r\n }\r\n\r\n /**\r\n * Gets the render list (meshes to be rendered) used in the G buffer.\r\n */\r\n public get renderList() {\r\n return this._multiRenderTarget.renderList;\r\n }\r\n\r\n /**\r\n * Set the render list (meshes to be rendered) used in the G buffer.\r\n */\r\n public set renderList(meshes: Nullable) {\r\n this._multiRenderTarget.renderList = meshes;\r\n }\r\n\r\n /**\r\n * Gets whether or not G buffer are supported by the running hardware.\r\n * This requires draw buffer supports\r\n */\r\n public get isSupported(): boolean {\r\n return this._multiRenderTarget.isSupported;\r\n }\r\n\r\n /**\r\n * Returns the index of the given texture type in the G-Buffer textures array\r\n * @param textureType The texture type constant. For example GeometryBufferRenderer.POSITION_TEXTURE_INDEX\r\n * @returns the index of the given texture type in the G-Buffer textures array\r\n */\r\n public getTextureIndex(textureType: number): number {\r\n switch (textureType) {\r\n case GeometryBufferRenderer.POSITION_TEXTURE_TYPE:\r\n return this._positionIndex;\r\n case GeometryBufferRenderer.VELOCITY_TEXTURE_TYPE:\r\n return this._velocityIndex;\r\n case GeometryBufferRenderer.REFLECTIVITY_TEXTURE_TYPE:\r\n return this._reflectivityIndex;\r\n default:\r\n return -1;\r\n }\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if objects positions are enabled for the G buffer.\r\n */\r\n public get enablePosition(): boolean {\r\n return this._enablePosition;\r\n }\r\n\r\n /**\r\n * Sets whether or not objects positions are enabled for the G buffer.\r\n */\r\n public set enablePosition(enable: boolean) {\r\n this._enablePosition = enable;\r\n\r\n // PrePass handles index and texture links\r\n if (!this._linkedWithPrePass) {\r\n this.dispose();\r\n this._createRenderTargets();\r\n }\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if objects velocities are enabled for the G buffer.\r\n */\r\n public get enableVelocity(): boolean {\r\n return this._enableVelocity;\r\n }\r\n\r\n /**\r\n * Sets whether or not objects velocities are enabled for the G buffer.\r\n */\r\n public set enableVelocity(enable: boolean) {\r\n this._enableVelocity = enable;\r\n\r\n if (!enable) {\r\n this._previousTransformationMatrices = {};\r\n }\r\n\r\n if (!this._linkedWithPrePass) {\r\n this.dispose();\r\n this._createRenderTargets();\r\n }\r\n\r\n this._scene.needsPreviousWorldMatrices = enable;\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating if objects reflectivity are enabled in the G buffer.\r\n */\r\n public get enableReflectivity(): boolean {\r\n return this._enableReflectivity;\r\n }\r\n\r\n /**\r\n * Sets whether or not objects reflectivity are enabled for the G buffer.\r\n * For Metallic-Roughness workflow with ORM texture, we assume that ORM texture is defined according to the default layout:\r\n * pbr.useRoughnessFromMetallicTextureAlpha = false;\r\n * pbr.useRoughnessFromMetallicTextureGreen = true;\r\n * pbr.useMetallnessFromMetallicTextureBlue = true;\r\n */\r\n public set enableReflectivity(enable: boolean) {\r\n this._enableReflectivity = enable;\r\n\r\n if (!this._linkedWithPrePass) {\r\n this.dispose();\r\n this._createRenderTargets();\r\n }\r\n }\r\n\r\n /**\r\n * If set to true (default: false), the depth texture will be cleared with the depth value corresponding to the far plane (1 in normal mode, 0 in reverse depth buffer mode)\r\n * If set to false, the depth texture is always cleared with 0.\r\n */\r\n public useSpecificClearForDepthTexture = false;\r\n\r\n /**\r\n * Gets the scene associated with the buffer.\r\n */\r\n public get scene(): Scene {\r\n return this._scene;\r\n }\r\n\r\n /**\r\n * Gets the ratio used by the buffer during its creation.\r\n * How big is the buffer related to the main canvas.\r\n */\r\n public get ratio(): number {\r\n return this._ratio;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _SceneComponentInitialization: (scene: Scene) => void = (_) => {\r\n throw _WarnImport(\"GeometryBufferRendererSceneComponent\");\r\n };\r\n\r\n /**\r\n * Creates a new G Buffer for the scene\r\n * @param scene The scene the buffer belongs to\r\n * @param ratio How big is the buffer related to the main canvas (default: 1)\r\n * @param depthFormat Format of the depth texture (default: Constants.TEXTUREFORMAT_DEPTH16)\r\n */\r\n constructor(scene: Scene, ratio: number = 1, depthFormat = Constants.TEXTUREFORMAT_DEPTH16) {\r\n this._scene = scene;\r\n this._ratio = ratio;\r\n this._useUbo = scene.getEngine().supportsUniformBuffers;\r\n this._depthFormat = depthFormat;\r\n\r\n GeometryBufferRenderer._SceneComponentInitialization(this._scene);\r\n\r\n // Render target\r\n this._createRenderTargets();\r\n }\r\n\r\n /**\r\n * Checks whether everything is ready to render a submesh to the G buffer.\r\n * @param subMesh the submesh to check readiness for\r\n * @param useInstances is the mesh drawn using instance or not\r\n * @returns true if ready otherwise false\r\n */\r\n public isReady(subMesh: SubMesh, useInstances: boolean): boolean {\r\n const material = subMesh.getMaterial();\r\n\r\n if (material && material.disableDepthWrite) {\r\n return false;\r\n }\r\n\r\n const defines = [];\r\n const attribs = [VertexBuffer.PositionKind, VertexBuffer.NormalKind];\r\n const mesh = subMesh.getMesh();\r\n\r\n // Alpha test\r\n if (material) {\r\n let needUv = false;\r\n if (material.needAlphaTesting() && material.getAlphaTestTexture()) {\r\n defines.push(\"#define ALPHATEST\");\r\n defines.push(`#define ALPHATEST_UV${material.getAlphaTestTexture().coordinatesIndex + 1}`);\r\n needUv = true;\r\n }\r\n\r\n if (material.bumpTexture && MaterialFlags.BumpTextureEnabled) {\r\n defines.push(\"#define BUMP\");\r\n defines.push(`#define BUMP_UV${material.bumpTexture.coordinatesIndex + 1}`);\r\n needUv = true;\r\n }\r\n\r\n if (this._enableReflectivity) {\r\n let metallicWorkflow = false;\r\n // for PBR materials: cf. https://doc.babylonjs.com/features/featuresDeepDive/materials/using/masterPBR\r\n if (material.getClassName() === \"PBRMetallicRoughnessMaterial\") {\r\n // if it is a PBR material in MetallicRoughness Mode:\r\n if (material.metallicRoughnessTexture !== null) {\r\n defines.push(\"#define ORMTEXTURE\");\r\n defines.push(`#define REFLECTIVITY_UV${material.metallicRoughnessTexture.coordinatesIndex + 1}`);\r\n defines.push(\"#define METALLICWORKFLOW\");\r\n needUv = true;\r\n metallicWorkflow = true;\r\n }\r\n if (material.metallic !== null) {\r\n defines.push(\"#define METALLIC\");\r\n defines.push(\"#define METALLICWORKFLOW\");\r\n metallicWorkflow = true;\r\n }\r\n if (material.roughness !== null) {\r\n defines.push(\"#define ROUGHNESS\");\r\n defines.push(\"#define METALLICWORKFLOW\");\r\n metallicWorkflow = true;\r\n }\r\n if (metallicWorkflow) {\r\n if (material.baseTexture !== null) {\r\n defines.push(\"#define ALBEDOTEXTURE\");\r\n defines.push(`#define ALBEDO_UV${material.baseTexture.coordinatesIndex + 1}`);\r\n if (material.baseTexture.gammaSpace) {\r\n defines.push(\"#define GAMMAALBEDO\");\r\n }\r\n needUv = true;\r\n }\r\n if (material.baseColor !== null) {\r\n defines.push(\"#define ALBEDOCOLOR\");\r\n }\r\n }\r\n } else if (material.getClassName() === \"PBRSpecularGlossinessMaterial\") {\r\n // if it is a PBR material in Specular/Glossiness Mode:\r\n if (material.specularGlossinessTexture !== null) {\r\n defines.push(\"#define SPECULARGLOSSINESSTEXTURE\");\r\n defines.push(`#define REFLECTIVITY_UV${material.specularGlossinessTexture.coordinatesIndex + 1}`);\r\n needUv = true;\r\n if (material.specularGlossinessTexture.gammaSpace) {\r\n defines.push(\"#define GAMMAREFLECTIVITYTEXTURE\");\r\n }\r\n } else {\r\n if (material.specularColor !== null) {\r\n defines.push(\"#define REFLECTIVITYCOLOR\");\r\n }\r\n }\r\n if (material.glossiness !== null) {\r\n defines.push(\"#define GLOSSINESS\");\r\n }\r\n } else if (material.getClassName() === \"PBRMaterial\") {\r\n // if it is the bigger PBRMaterial\r\n if (material.metallicTexture !== null) {\r\n defines.push(\"#define ORMTEXTURE\");\r\n defines.push(`#define REFLECTIVITY_UV${material.metallicTexture.coordinatesIndex + 1}`);\r\n defines.push(\"#define METALLICWORKFLOW\");\r\n needUv = true;\r\n metallicWorkflow = true;\r\n }\r\n if (material.metallic !== null) {\r\n defines.push(\"#define METALLIC\");\r\n defines.push(\"#define METALLICWORKFLOW\");\r\n metallicWorkflow = true;\r\n }\r\n\r\n if (material.roughness !== null) {\r\n defines.push(\"#define ROUGHNESS\");\r\n defines.push(\"#define METALLICWORKFLOW\");\r\n metallicWorkflow = true;\r\n }\r\n\r\n if (metallicWorkflow) {\r\n if (material.albedoTexture !== null) {\r\n defines.push(\"#define ALBEDOTEXTURE\");\r\n defines.push(`#define ALBEDO_UV${material.albedoTexture.coordinatesIndex + 1}`);\r\n if (material.albedoTexture.gammaSpace) {\r\n defines.push(\"#define GAMMAALBEDO\");\r\n }\r\n needUv = true;\r\n }\r\n if (material.albedoColor !== null) {\r\n defines.push(\"#define ALBEDOCOLOR\");\r\n }\r\n } else {\r\n // SpecularGlossiness Model\r\n if (material.reflectivityTexture !== null) {\r\n defines.push(\"#define SPECULARGLOSSINESSTEXTURE\");\r\n defines.push(`#define REFLECTIVITY_UV${material.reflectivityTexture.coordinatesIndex + 1}`);\r\n if (material.reflectivityTexture.gammaSpace) {\r\n defines.push(\"#define GAMMAREFLECTIVITYTEXTURE\");\r\n }\r\n needUv = true;\r\n } else if (material.reflectivityColor !== null) {\r\n defines.push(\"#define REFLECTIVITYCOLOR\");\r\n }\r\n if (material.microSurface !== null) {\r\n defines.push(\"#define GLOSSINESS\");\r\n }\r\n }\r\n } else if (material.getClassName() === \"StandardMaterial\") {\r\n // if StandardMaterial:\r\n if (material.specularTexture !== null) {\r\n defines.push(\"#define REFLECTIVITYTEXTURE\");\r\n defines.push(`#define REFLECTIVITY_UV${material.specularTexture.coordinatesIndex + 1}`);\r\n if (material.specularTexture.gammaSpace) {\r\n defines.push(\"#define GAMMAREFLECTIVITYTEXTURE\");\r\n }\r\n needUv = true;\r\n }\r\n if (material.specularColor !== null) {\r\n defines.push(\"#define REFLECTIVITYCOLOR\");\r\n }\r\n }\r\n }\r\n\r\n if (needUv) {\r\n defines.push(\"#define NEED_UV\");\r\n if (mesh.isVerticesDataPresent(VertexBuffer.UVKind)) {\r\n attribs.push(VertexBuffer.UVKind);\r\n defines.push(\"#define UV1\");\r\n }\r\n if (mesh.isVerticesDataPresent(VertexBuffer.UV2Kind)) {\r\n attribs.push(VertexBuffer.UV2Kind);\r\n defines.push(\"#define UV2\");\r\n }\r\n }\r\n }\r\n\r\n // PrePass\r\n if (this._linkedWithPrePass) {\r\n defines.push(\"#define PREPASS\");\r\n if (this._depthIndex !== -1) {\r\n defines.push(\"#define DEPTH_INDEX \" + this._depthIndex);\r\n defines.push(\"#define PREPASS_DEPTH\");\r\n }\r\n if (this._normalIndex !== -1) {\r\n defines.push(\"#define NORMAL_INDEX \" + this._normalIndex);\r\n defines.push(\"#define PREPASS_NORMAL\");\r\n }\r\n }\r\n\r\n // Buffers\r\n if (this._enablePosition) {\r\n defines.push(\"#define POSITION\");\r\n defines.push(\"#define POSITION_INDEX \" + this._positionIndex);\r\n }\r\n\r\n if (this._enableVelocity) {\r\n defines.push(\"#define VELOCITY\");\r\n defines.push(\"#define VELOCITY_INDEX \" + this._velocityIndex);\r\n if (this.excludedSkinnedMeshesFromVelocity.indexOf(mesh) === -1) {\r\n defines.push(\"#define BONES_VELOCITY_ENABLED\");\r\n }\r\n }\r\n\r\n if (this._enableReflectivity) {\r\n defines.push(\"#define REFLECTIVITY\");\r\n defines.push(\"#define REFLECTIVITY_INDEX \" + this._reflectivityIndex);\r\n }\r\n\r\n // Bones\r\n if (mesh.useBones && mesh.computeBonesUsingShaders) {\r\n attribs.push(VertexBuffer.MatricesIndicesKind);\r\n attribs.push(VertexBuffer.MatricesWeightsKind);\r\n if (mesh.numBoneInfluencers > 4) {\r\n attribs.push(VertexBuffer.MatricesIndicesExtraKind);\r\n attribs.push(VertexBuffer.MatricesWeightsExtraKind);\r\n }\r\n defines.push(\"#define NUM_BONE_INFLUENCERS \" + mesh.numBoneInfluencers);\r\n defines.push(\"#define BonesPerMesh \" + (mesh.skeleton ? mesh.skeleton.bones.length + 1 : 0));\r\n } else {\r\n defines.push(\"#define NUM_BONE_INFLUENCERS 0\");\r\n }\r\n\r\n // Morph targets\r\n const morphTargetManager = (mesh as Mesh).morphTargetManager;\r\n let numMorphInfluencers = 0;\r\n if (morphTargetManager) {\r\n if (morphTargetManager.numInfluencers > 0) {\r\n numMorphInfluencers = morphTargetManager.numInfluencers;\r\n\r\n defines.push(\"#define MORPHTARGETS\");\r\n defines.push(\"#define NUM_MORPH_INFLUENCERS \" + numMorphInfluencers);\r\n if (morphTargetManager.isUsingTextureForTargets) {\r\n defines.push(\"#define MORPHTARGETS_TEXTURE\");\r\n }\r\n MaterialHelper.PrepareAttributesForMorphTargetsInfluencers(attribs, mesh, numMorphInfluencers);\r\n }\r\n }\r\n\r\n // Instances\r\n if (useInstances) {\r\n defines.push(\"#define INSTANCES\");\r\n MaterialHelper.PushAttributesForInstances(attribs, this._enableVelocity);\r\n if (subMesh.getRenderingMesh().hasThinInstances) {\r\n defines.push(\"#define THIN_INSTANCES\");\r\n }\r\n }\r\n\r\n // Setup textures count\r\n if (this._linkedWithPrePass) {\r\n defines.push(\"#define RENDER_TARGET_COUNT \" + this._attachmentsFromPrePass.length);\r\n } else {\r\n defines.push(\"#define RENDER_TARGET_COUNT \" + this._multiRenderTarget.textures.length);\r\n }\r\n\r\n prepareStringDefinesForClipPlanes(material, this._scene, defines);\r\n\r\n // Get correct effect\r\n const engine = this._scene.getEngine();\r\n const drawWrapper = subMesh._getDrawWrapper(undefined, true)!;\r\n const cachedDefines = drawWrapper.defines;\r\n const join = defines.join(\"\\n\");\r\n if (cachedDefines !== join) {\r\n drawWrapper.setEffect(\r\n engine.createEffect(\r\n \"geometry\",\r\n {\r\n attributes: attribs,\r\n uniformsNames: uniforms,\r\n samplers: [\"diffuseSampler\", \"bumpSampler\", \"reflectivitySampler\", \"albedoSampler\", \"morphTargets\"],\r\n defines: join,\r\n onCompiled: null,\r\n fallbacks: null,\r\n onError: null,\r\n uniformBuffersNames: [\"Scene\"],\r\n indexParameters: { buffersCount: this._multiRenderTarget.textures.length - 1, maxSimultaneousMorphTargets: numMorphInfluencers },\r\n },\r\n engine\r\n ),\r\n join\r\n );\r\n }\r\n\r\n return drawWrapper.effect!.isReady();\r\n }\r\n\r\n /**\r\n * Gets the current underlying G Buffer.\r\n * @returns the buffer\r\n */\r\n public getGBuffer(): MultiRenderTarget {\r\n return this._multiRenderTarget;\r\n }\r\n\r\n /**\r\n * Gets the number of samples used to render the buffer (anti aliasing).\r\n */\r\n public get samples(): number {\r\n return this._multiRenderTarget.samples;\r\n }\r\n\r\n /**\r\n * Sets the number of samples used to render the buffer (anti aliasing).\r\n */\r\n public set samples(value: number) {\r\n this._multiRenderTarget.samples = value;\r\n }\r\n\r\n /**\r\n * Disposes the renderer and frees up associated resources.\r\n */\r\n public dispose(): void {\r\n if (this._resizeObserver) {\r\n const engine = this._scene.getEngine();\r\n engine.onResizeObservable.remove(this._resizeObserver);\r\n this._resizeObserver = null;\r\n }\r\n this.getGBuffer().dispose();\r\n }\r\n\r\n private _assignRenderTargetIndices(): [number, string[]] {\r\n const textureNames: string[] = [];\r\n let count = 2;\r\n\r\n textureNames.push(\"gBuffer_Depth\", \"gBuffer_Normal\");\r\n\r\n if (this._enablePosition) {\r\n this._positionIndex = count;\r\n count++;\r\n textureNames.push(\"gBuffer_Position\");\r\n }\r\n\r\n if (this._enableVelocity) {\r\n this._velocityIndex = count;\r\n count++;\r\n textureNames.push(\"gBuffer_Velocity\");\r\n }\r\n\r\n if (this._enableReflectivity) {\r\n this._reflectivityIndex = count;\r\n count++;\r\n textureNames.push(\"gBuffer_Reflectivity\");\r\n }\r\n\r\n return [count, textureNames];\r\n }\r\n\r\n protected _createRenderTargets(): void {\r\n const engine = this._scene.getEngine();\r\n const [count, textureNames] = this._assignRenderTargetIndices();\r\n\r\n let type = Constants.TEXTURETYPE_UNSIGNED_BYTE;\r\n if (engine._caps.textureFloat && engine._caps.textureFloatLinearFiltering) {\r\n type = Constants.TEXTURETYPE_FLOAT;\r\n } else if (engine._caps.textureHalfFloat && engine._caps.textureHalfFloatLinearFiltering) {\r\n type = Constants.TEXTURETYPE_HALF_FLOAT;\r\n }\r\n\r\n this._multiRenderTarget = new MultiRenderTarget(\r\n \"gBuffer\",\r\n { width: engine.getRenderWidth() * this._ratio, height: engine.getRenderHeight() * this._ratio },\r\n count,\r\n this._scene,\r\n { generateMipMaps: false, generateDepthTexture: true, defaultType: type, depthTextureFormat: this._depthFormat },\r\n textureNames.concat(\"gBuffer_DepthBuffer\")\r\n );\r\n if (!this.isSupported) {\r\n return;\r\n }\r\n this._multiRenderTarget.wrapU = Texture.CLAMP_ADDRESSMODE;\r\n this._multiRenderTarget.wrapV = Texture.CLAMP_ADDRESSMODE;\r\n this._multiRenderTarget.refreshRate = 1;\r\n this._multiRenderTarget.renderParticles = false;\r\n this._multiRenderTarget.renderList = null;\r\n\r\n // Depth is always the first texture in the geometry buffer renderer!\r\n const layoutAttachmentsAll = [true];\r\n const layoutAttachmentsAllButDepth = [false];\r\n const layoutAttachmentsDepthOnly = [true];\r\n\r\n for (let i = 1; i < count; ++i) {\r\n layoutAttachmentsAll.push(true);\r\n layoutAttachmentsDepthOnly.push(false);\r\n layoutAttachmentsAllButDepth.push(true);\r\n }\r\n\r\n const attachmentsAll = engine.buildTextureLayout(layoutAttachmentsAll);\r\n const attachmentsAllButDepth = engine.buildTextureLayout(layoutAttachmentsAllButDepth);\r\n const attachmentsDepthOnly = engine.buildTextureLayout(layoutAttachmentsDepthOnly);\r\n\r\n this._multiRenderTarget.onClearObservable.add((engine) => {\r\n engine.bindAttachments(this.useSpecificClearForDepthTexture ? attachmentsAllButDepth : attachmentsAll);\r\n engine.clear(this._clearColor, true, true, true);\r\n if (this.useSpecificClearForDepthTexture) {\r\n engine.bindAttachments(attachmentsDepthOnly);\r\n engine.clear(this._clearDepthColor, true, true, true);\r\n }\r\n engine.bindAttachments(attachmentsAll);\r\n });\r\n\r\n this._resizeObserver = engine.onResizeObservable.add(() => {\r\n if (this._multiRenderTarget) {\r\n this._multiRenderTarget.resize({ width: engine.getRenderWidth() * this._ratio, height: engine.getRenderHeight() * this._ratio });\r\n }\r\n });\r\n\r\n // Custom render function\r\n const renderSubMesh = (subMesh: SubMesh): void => {\r\n const renderingMesh = subMesh.getRenderingMesh();\r\n const effectiveMesh = subMesh.getEffectiveMesh();\r\n const scene = this._scene;\r\n const engine = scene.getEngine();\r\n const material = subMesh.getMaterial();\r\n\r\n if (!material) {\r\n return;\r\n }\r\n\r\n effectiveMesh._internalAbstractMeshDataInfo._isActiveIntermediate = false;\r\n\r\n // Velocity\r\n if (this._enableVelocity && !this._previousTransformationMatrices[effectiveMesh.uniqueId]) {\r\n this._previousTransformationMatrices[effectiveMesh.uniqueId] = {\r\n world: Matrix.Identity(),\r\n viewProjection: scene.getTransformMatrix(),\r\n };\r\n\r\n if (renderingMesh.skeleton) {\r\n const bonesTransformations = renderingMesh.skeleton.getTransformMatrices(renderingMesh);\r\n this._previousBonesTransformationMatrices[renderingMesh.uniqueId] = this._copyBonesTransformationMatrices(\r\n bonesTransformations,\r\n new Float32Array(bonesTransformations.length)\r\n );\r\n }\r\n }\r\n\r\n // Managing instances\r\n const batch = renderingMesh._getInstancesRenderList(subMesh._id, !!subMesh.getReplacementMesh());\r\n\r\n if (batch.mustReturn) {\r\n return;\r\n }\r\n\r\n const hardwareInstancedRendering = engine.getCaps().instancedArrays && (batch.visibleInstances[subMesh._id] !== null || renderingMesh.hasThinInstances);\r\n const world = effectiveMesh.getWorldMatrix();\r\n\r\n if (this.isReady(subMesh, hardwareInstancedRendering)) {\r\n const drawWrapper = subMesh._getDrawWrapper();\r\n\r\n if (!drawWrapper) {\r\n return;\r\n }\r\n\r\n const effect = drawWrapper.effect!;\r\n\r\n engine.enableEffect(drawWrapper);\r\n if (!hardwareInstancedRendering) {\r\n renderingMesh._bind(subMesh, effect, material.fillMode);\r\n }\r\n\r\n if (!this._useUbo) {\r\n effect.setMatrix(\"viewProjection\", scene.getTransformMatrix());\r\n effect.setMatrix(\"view\", scene.getViewMatrix());\r\n } else {\r\n MaterialHelper.BindSceneUniformBuffer(effect, this._scene.getSceneUniformBuffer());\r\n this._scene.finalizeSceneUbo();\r\n }\r\n\r\n let sideOrientation: Nullable;\r\n const instanceDataStorage = (renderingMesh as Mesh)._instanceDataStorage;\r\n\r\n if (!instanceDataStorage.isFrozen && (material.backFaceCulling || renderingMesh.overrideMaterialSideOrientation !== null)) {\r\n const mainDeterminant = effectiveMesh._getWorldMatrixDeterminant();\r\n sideOrientation = renderingMesh.overrideMaterialSideOrientation;\r\n if (sideOrientation === null) {\r\n sideOrientation = material.sideOrientation;\r\n }\r\n if (mainDeterminant < 0) {\r\n sideOrientation = sideOrientation === Material.ClockWiseSideOrientation ? Material.CounterClockWiseSideOrientation : Material.ClockWiseSideOrientation;\r\n }\r\n } else {\r\n sideOrientation = instanceDataStorage.sideOrientation;\r\n }\r\n\r\n material._preBind(drawWrapper, sideOrientation);\r\n\r\n // Alpha test\r\n if (material.needAlphaTesting()) {\r\n const alphaTexture = material.getAlphaTestTexture();\r\n if (alphaTexture) {\r\n effect.setTexture(\"diffuseSampler\", alphaTexture);\r\n effect.setMatrix(\"diffuseMatrix\", alphaTexture.getTextureMatrix());\r\n }\r\n }\r\n\r\n // Bump\r\n if (material.bumpTexture && scene.getEngine().getCaps().standardDerivatives && MaterialFlags.BumpTextureEnabled) {\r\n effect.setFloat3(\"vBumpInfos\", material.bumpTexture.coordinatesIndex, 1.0 / material.bumpTexture.level, material.parallaxScaleBias);\r\n effect.setMatrix(\"bumpMatrix\", material.bumpTexture.getTextureMatrix());\r\n effect.setTexture(\"bumpSampler\", material.bumpTexture);\r\n effect.setFloat2(\"vTangentSpaceParams\", material.invertNormalMapX ? -1.0 : 1.0, material.invertNormalMapY ? -1.0 : 1.0);\r\n }\r\n\r\n // Reflectivity\r\n if (this._enableReflectivity) {\r\n // for PBR materials: cf. https://doc.babylonjs.com/features/featuresDeepDive/materials/using/masterPBR\r\n if (material.getClassName() === \"PBRMetallicRoughnessMaterial\") {\r\n // if it is a PBR material in MetallicRoughness Mode:\r\n if (material.metallicRoughnessTexture !== null) {\r\n effect.setTexture(\"reflectivitySampler\", material.metallicRoughnessTexture);\r\n effect.setMatrix(\"reflectivityMatrix\", material.metallicRoughnessTexture.getTextureMatrix());\r\n }\r\n if (material.metallic !== null) {\r\n effect.setFloat(\"metallic\", material.metallic);\r\n }\r\n if (material.roughness !== null) {\r\n effect.setFloat(\"glossiness\", 1.0 - material.roughness);\r\n }\r\n if (material.baseTexture !== null) {\r\n effect.setTexture(\"albedoSampler\", material.baseTexture);\r\n effect.setMatrix(\"albedoMatrix\", material.baseTexture.getTextureMatrix());\r\n }\r\n if (material.baseColor !== null) {\r\n effect.setColor3(\"albedoColor\", material.baseColor);\r\n }\r\n } else if (material.getClassName() === \"PBRSpecularGlossinessMaterial\") {\r\n // if it is a PBR material in Specular/Glossiness Mode:\r\n if (material.specularGlossinessTexture !== null) {\r\n effect.setTexture(\"reflectivitySampler\", material.specularGlossinessTexture);\r\n effect.setMatrix(\"reflectivityMatrix\", material.specularGlossinessTexture.getTextureMatrix());\r\n } else {\r\n if (material.specularColor !== null) {\r\n effect.setColor3(\"reflectivityColor\", material.specularColor);\r\n }\r\n }\r\n if (material.glossiness !== null) {\r\n effect.setFloat(\"glossiness\", material.glossiness);\r\n }\r\n } else if (material.getClassName() === \"PBRMaterial\") {\r\n // if it is the bigger PBRMaterial\r\n if (material.metallicTexture !== null) {\r\n effect.setTexture(\"reflectivitySampler\", material.metallicTexture);\r\n effect.setMatrix(\"reflectivityMatrix\", material.metallicTexture.getTextureMatrix());\r\n }\r\n if (material.metallic !== null) {\r\n effect.setFloat(\"metallic\", material.metallic);\r\n }\r\n\r\n if (material.roughness !== null) {\r\n effect.setFloat(\"glossiness\", 1.0 - material.roughness);\r\n }\r\n\r\n if (material.roughness !== null || material.metallic !== null || material.metallicTexture !== null) {\r\n // MetallicRoughness Model\r\n if (material.albedoTexture !== null) {\r\n effect.setTexture(\"albedoSampler\", material.albedoTexture);\r\n effect.setMatrix(\"albedoMatrix\", material.albedoTexture.getTextureMatrix());\r\n }\r\n if (material.albedoColor !== null) {\r\n effect.setColor3(\"albedoColor\", material.albedoColor);\r\n }\r\n } else {\r\n // SpecularGlossiness Model\r\n if (material.reflectivityTexture !== null) {\r\n effect.setTexture(\"reflectivitySampler\", material.reflectivityTexture);\r\n effect.setMatrix(\"reflectivityMatrix\", material.reflectivityTexture.getTextureMatrix());\r\n } else if (material.reflectivityColor !== null) {\r\n effect.setColor3(\"reflectivityColor\", material.reflectivityColor);\r\n }\r\n if (material.microSurface !== null) {\r\n effect.setFloat(\"glossiness\", material.microSurface);\r\n }\r\n }\r\n } else if (material.getClassName() === \"StandardMaterial\") {\r\n // if StandardMaterial:\r\n if (material.specularTexture !== null) {\r\n effect.setTexture(\"reflectivitySampler\", material.specularTexture);\r\n effect.setMatrix(\"reflectivityMatrix\", material.specularTexture.getTextureMatrix());\r\n }\r\n if (material.specularColor !== null) {\r\n effect.setColor3(\"reflectivityColor\", material.specularColor);\r\n }\r\n }\r\n }\r\n\r\n // Clip plane\r\n bindClipPlane(effect, material, this._scene);\r\n\r\n // Bones\r\n if (renderingMesh.useBones && renderingMesh.computeBonesUsingShaders && renderingMesh.skeleton) {\r\n effect.setMatrices(\"mBones\", renderingMesh.skeleton.getTransformMatrices(renderingMesh));\r\n if (this._enableVelocity) {\r\n effect.setMatrices(\"mPreviousBones\", this._previousBonesTransformationMatrices[renderingMesh.uniqueId]);\r\n }\r\n }\r\n\r\n // Morph targets\r\n MaterialHelper.BindMorphTargetParameters(renderingMesh, effect);\r\n if (renderingMesh.morphTargetManager && renderingMesh.morphTargetManager.isUsingTextureForTargets) {\r\n renderingMesh.morphTargetManager._bind(effect);\r\n }\r\n\r\n // Velocity\r\n if (this._enableVelocity) {\r\n effect.setMatrix(\"previousWorld\", this._previousTransformationMatrices[effectiveMesh.uniqueId].world);\r\n effect.setMatrix(\"previousViewProjection\", this._previousTransformationMatrices[effectiveMesh.uniqueId].viewProjection);\r\n }\r\n\r\n if (hardwareInstancedRendering && renderingMesh.hasThinInstances) {\r\n effect.setMatrix(\"world\", world);\r\n }\r\n\r\n // Draw\r\n renderingMesh._processRendering(effectiveMesh, subMesh, effect, material.fillMode, batch, hardwareInstancedRendering, (isInstance, w) => {\r\n if (!isInstance) {\r\n effect.setMatrix(\"world\", w);\r\n }\r\n });\r\n }\r\n\r\n // Velocity\r\n if (this._enableVelocity) {\r\n this._previousTransformationMatrices[effectiveMesh.uniqueId].world = world.clone();\r\n this._previousTransformationMatrices[effectiveMesh.uniqueId].viewProjection = this._scene.getTransformMatrix().clone();\r\n if (renderingMesh.skeleton) {\r\n this._copyBonesTransformationMatrices(\r\n renderingMesh.skeleton.getTransformMatrices(renderingMesh),\r\n this._previousBonesTransformationMatrices[effectiveMesh.uniqueId]\r\n );\r\n }\r\n }\r\n };\r\n\r\n this._multiRenderTarget.customIsReadyFunction = (mesh: AbstractMesh, refreshRate: number, preWarm?: boolean) => {\r\n if ((preWarm || refreshRate === 0) && mesh.subMeshes) {\r\n for (let i = 0; i < mesh.subMeshes.length; ++i) {\r\n const subMesh = mesh.subMeshes[i];\r\n const material = subMesh.getMaterial();\r\n const renderingMesh = subMesh.getRenderingMesh();\r\n\r\n if (!material) {\r\n continue;\r\n }\r\n\r\n const batch = renderingMesh._getInstancesRenderList(subMesh._id, !!subMesh.getReplacementMesh());\r\n const hardwareInstancedRendering = engine.getCaps().instancedArrays && (batch.visibleInstances[subMesh._id] !== null || renderingMesh.hasThinInstances);\r\n\r\n if (!this.isReady(subMesh, hardwareInstancedRendering)) {\r\n return false;\r\n }\r\n }\r\n }\r\n\r\n return true;\r\n };\r\n\r\n this._multiRenderTarget.customRenderFunction = (\r\n opaqueSubMeshes: SmartArray,\r\n alphaTestSubMeshes: SmartArray,\r\n transparentSubMeshes: SmartArray,\r\n depthOnlySubMeshes: SmartArray\r\n ): void => {\r\n let index;\r\n\r\n if (this._linkedWithPrePass) {\r\n if (!this._prePassRenderer.enabled) {\r\n return;\r\n }\r\n this._scene.getEngine().bindAttachments(this._attachmentsFromPrePass);\r\n }\r\n\r\n if (depthOnlySubMeshes.length) {\r\n engine.setColorWrite(false);\r\n for (index = 0; index < depthOnlySubMeshes.length; index++) {\r\n renderSubMesh(depthOnlySubMeshes.data[index]);\r\n }\r\n engine.setColorWrite(true);\r\n }\r\n\r\n for (index = 0; index < opaqueSubMeshes.length; index++) {\r\n renderSubMesh(opaqueSubMeshes.data[index]);\r\n }\r\n\r\n engine.setDepthWrite(false);\r\n for (index = 0; index < alphaTestSubMeshes.length; index++) {\r\n renderSubMesh(alphaTestSubMeshes.data[index]);\r\n }\r\n\r\n if (this.renderTransparentMeshes) {\r\n for (index = 0; index < transparentSubMeshes.length; index++) {\r\n renderSubMesh(transparentSubMeshes.data[index]);\r\n }\r\n }\r\n engine.setDepthWrite(true);\r\n };\r\n }\r\n\r\n // Copies the bones transformation matrices into the target array and returns the target's reference\r\n private _copyBonesTransformationMatrices(source: Float32Array, target: Float32Array): Float32Array {\r\n for (let i = 0; i < source.length; i++) {\r\n target[i] = source[i];\r\n }\r\n\r\n return target;\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\nimport { Scene } from \"../scene\";\r\nimport type { ISceneComponent } from \"../sceneComponent\";\r\nimport { SceneComponentConstants } from \"../sceneComponent\";\r\nimport type { SmartArrayNoDuplicate } from \"../Misc/smartArray\";\r\nimport type { RenderTargetTexture } from \"../Materials/Textures/renderTargetTexture\";\r\nimport { GeometryBufferRenderer } from \"./geometryBufferRenderer\";\r\nimport { Constants } from \"../Engines/constants\";\r\n\r\ndeclare module \"../scene\" {\r\n export interface Scene {\r\n /** @internal (Backing field) */\r\n _geometryBufferRenderer: Nullable;\r\n\r\n /**\r\n * Gets or Sets the current geometry buffer associated to the scene.\r\n */\r\n geometryBufferRenderer: Nullable;\r\n\r\n /**\r\n * Enables a GeometryBufferRender and associates it with the scene\r\n * @param ratio defines the scaling ratio to apply to the renderer (1 by default which means same resolution)\r\n * @param depthFormat Format of the depth texture (default: Constants.TEXTUREFORMAT_DEPTH16)\r\n * @returns the GeometryBufferRenderer\r\n */\r\n enableGeometryBufferRenderer(ratio?: number, depthFormat?: number): Nullable;\r\n\r\n /**\r\n * Disables the GeometryBufferRender associated with the scene\r\n */\r\n disableGeometryBufferRenderer(): void;\r\n }\r\n}\r\n\r\nObject.defineProperty(Scene.prototype, \"geometryBufferRenderer\", {\r\n get: function (this: Scene) {\r\n return this._geometryBufferRenderer;\r\n },\r\n set: function (this: Scene, value: Nullable) {\r\n if (value && value.isSupported) {\r\n this._geometryBufferRenderer = value;\r\n }\r\n },\r\n enumerable: true,\r\n configurable: true,\r\n});\r\n\r\nScene.prototype.enableGeometryBufferRenderer = function (ratio: number = 1, depthFormat = Constants.TEXTUREFORMAT_DEPTH16): Nullable {\r\n if (this._geometryBufferRenderer) {\r\n return this._geometryBufferRenderer;\r\n }\r\n\r\n this._geometryBufferRenderer = new GeometryBufferRenderer(this, ratio, depthFormat);\r\n if (!this._geometryBufferRenderer.isSupported) {\r\n this._geometryBufferRenderer = null;\r\n }\r\n\r\n return this._geometryBufferRenderer;\r\n};\r\n\r\nScene.prototype.disableGeometryBufferRenderer = function (): void {\r\n if (!this._geometryBufferRenderer) {\r\n return;\r\n }\r\n\r\n this._geometryBufferRenderer.dispose();\r\n this._geometryBufferRenderer = null;\r\n};\r\n\r\n/**\r\n * Defines the Geometry Buffer scene component responsible to manage a G-Buffer useful\r\n * in several rendering techniques.\r\n */\r\nexport class GeometryBufferRendererSceneComponent implements ISceneComponent {\r\n /**\r\n * The component name helpful to identify the component in the list of scene components.\r\n */\r\n public readonly name = SceneComponentConstants.NAME_GEOMETRYBUFFERRENDERER;\r\n\r\n /**\r\n * The scene the component belongs to.\r\n */\r\n public scene: Scene;\r\n\r\n /**\r\n * Creates a new instance of the component for the given scene\r\n * @param scene Defines the scene to register the component in\r\n */\r\n constructor(scene: Scene) {\r\n this.scene = scene;\r\n }\r\n\r\n /**\r\n * Registers the component in a given scene\r\n */\r\n public register(): void {\r\n this.scene._gatherRenderTargetsStage.registerStep(SceneComponentConstants.STEP_GATHERRENDERTARGETS_GEOMETRYBUFFERRENDERER, this, this._gatherRenderTargets);\r\n }\r\n\r\n /**\r\n * Rebuilds the elements related to this component in case of\r\n * context lost for instance.\r\n */\r\n public rebuild(): void {\r\n // Nothing to do for this component\r\n }\r\n\r\n /**\r\n * Disposes the component and the associated resources\r\n */\r\n public dispose(): void {\r\n // Nothing to do for this component\r\n }\r\n\r\n private _gatherRenderTargets(renderTargets: SmartArrayNoDuplicate): void {\r\n if (this.scene._geometryBufferRenderer) {\r\n renderTargets.push(this.scene._geometryBufferRenderer.getGBuffer());\r\n }\r\n }\r\n}\r\n\r\nGeometryBufferRenderer._SceneComponentInitialization = (scene: Scene) => {\r\n // Register the G Buffer component to the scene.\r\n let component = scene._getComponent(SceneComponentConstants.NAME_GEOMETRYBUFFERRENDERER) as GeometryBufferRendererSceneComponent;\r\n if (!component) {\r\n component = new GeometryBufferRendererSceneComponent(scene);\r\n scene._addComponent(component);\r\n }\r\n};\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/clipPlaneFragmentDeclaration\";\nimport \"./ShadersInclude/logDepthDeclaration\";\nimport \"./ShadersInclude/clipPlaneFragment\";\nimport \"./ShadersInclude/logDepthFragment\";\n\nconst name = \"outlinePixelShader\";\nconst shader = `#ifdef LOGARITHMICDEPTH\n#extension GL_EXT_frag_depth : enable\n#endif\nuniform vec4 color;\n#ifdef ALPHATEST\nvarying vec2 vUV;uniform sampler2D diffuseSampler;\n#endif\n#include\n#include\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_FRAGMENT_MAIN_BEGIN\n#include\n#ifdef ALPHATEST\nif (texture2D(diffuseSampler,vUV).a<0.4)\ndiscard;\n#endif\n#include\ngl_FragColor=color;\n#define CUSTOM_FRAGMENT_MAIN_END\n}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const outlinePixelShader = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/bonesDeclaration\";\nimport \"./ShadersInclude/bakedVertexAnimationDeclaration\";\nimport \"./ShadersInclude/morphTargetsVertexGlobalDeclaration\";\nimport \"./ShadersInclude/morphTargetsVertexDeclaration\";\nimport \"./ShadersInclude/clipPlaneVertexDeclaration\";\nimport \"./ShadersInclude/instancesDeclaration\";\nimport \"./ShadersInclude/logDepthDeclaration\";\nimport \"./ShadersInclude/morphTargetsVertexGlobal\";\nimport \"./ShadersInclude/morphTargetsVertex\";\nimport \"./ShadersInclude/instancesVertex\";\nimport \"./ShadersInclude/bonesVertex\";\nimport \"./ShadersInclude/bakedVertexAnimation\";\nimport \"./ShadersInclude/clipPlaneVertex\";\nimport \"./ShadersInclude/logDepthVertex\";\n\nconst name = \"outlineVertexShader\";\nconst shader = `attribute vec3 position;attribute vec3 normal;\n#include\n#include\n#include\n#include[0..maxSimultaneousMorphTargets]\n#include\nuniform float offset;\n#include\nuniform mat4 viewProjection;\n#ifdef ALPHATEST\nvarying vec2 vUV;uniform mat4 diffuseMatrix;\n#ifdef UV1\nattribute vec2 uv;\n#endif\n#ifdef UV2\nattribute vec2 uv2;\n#endif\n#endif\n#include\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void)\n{vec3 positionUpdated=position;vec3 normalUpdated=normal;\n#ifdef UV1\nvec2 uvUpdated=uv;\n#endif\n#include\n#include[0..maxSimultaneousMorphTargets]\nvec3 offsetPosition=positionUpdated+(normalUpdated*offset);\n#include\n#include\n#include\nvec4 worldPos=finalWorld*vec4(offsetPosition,1.0);gl_Position=viewProjection*worldPos;\n#ifdef ALPHATEST\n#ifdef UV1\nvUV=vec2(diffuseMatrix*vec4(uvUpdated,1.0,0.0));\n#endif\n#ifdef UV2\nvUV=vec2(diffuseMatrix*vec4(uv2,1.0,0.0));\n#endif\n#endif\n#include\n#include\n}\n`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const outlineVertexShader = { name, shader };\n","import { VertexBuffer } from \"../Buffers/buffer\";\r\nimport type { SubMesh } from \"../Meshes/subMesh\";\r\nimport type { _InstancesBatch } from \"../Meshes/mesh\";\r\nimport { Mesh } from \"../Meshes/mesh\";\r\nimport { Scene } from \"../scene\";\r\nimport type { Engine } from \"../Engines/engine\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport type { ISceneComponent } from \"../sceneComponent\";\r\nimport { SceneComponentConstants } from \"../sceneComponent\";\r\nimport { DrawWrapper } from \"../Materials/drawWrapper\";\r\nimport { MaterialHelper } from \"../Materials/materialHelper\";\r\n\r\nimport \"../Shaders/outline.fragment\";\r\nimport \"../Shaders/outline.vertex\";\r\nimport { addClipPlaneUniforms, bindClipPlane, prepareStringDefinesForClipPlanes } from \"core/Materials/clipPlaneMaterialHelper\";\r\n\r\ndeclare module \"../scene\" {\r\n export interface Scene {\r\n /** @internal */\r\n _outlineRenderer: OutlineRenderer;\r\n\r\n /**\r\n * Gets the outline renderer associated with the scene\r\n * @returns a OutlineRenderer\r\n */\r\n getOutlineRenderer(): OutlineRenderer;\r\n }\r\n}\r\n\r\n/**\r\n * Gets the outline renderer associated with the scene\r\n * @returns a OutlineRenderer\r\n */\r\nScene.prototype.getOutlineRenderer = function (): OutlineRenderer {\r\n if (!this._outlineRenderer) {\r\n this._outlineRenderer = new OutlineRenderer(this);\r\n }\r\n return this._outlineRenderer;\r\n};\r\n\r\ndeclare module \"../Meshes/abstractMesh\" {\r\n export interface AbstractMesh {\r\n /** @internal (Backing field) */\r\n _renderOutline: boolean;\r\n /**\r\n * Gets or sets a boolean indicating if the outline must be rendered as well\r\n * @see https://www.babylonjs-playground.com/#10WJ5S#3\r\n */\r\n renderOutline: boolean;\r\n\r\n /** @internal (Backing field) */\r\n _renderOverlay: boolean;\r\n /**\r\n * Gets or sets a boolean indicating if the overlay must be rendered as well\r\n * @see https://www.babylonjs-playground.com/#10WJ5S#2\r\n */\r\n renderOverlay: boolean;\r\n }\r\n}\r\n\r\nObject.defineProperty(Mesh.prototype, \"renderOutline\", {\r\n get: function (this: Mesh) {\r\n return this._renderOutline;\r\n },\r\n set: function (this: Mesh, value: boolean) {\r\n if (value) {\r\n // Lazy Load the component.\r\n this.getScene().getOutlineRenderer();\r\n }\r\n this._renderOutline = value;\r\n },\r\n enumerable: true,\r\n configurable: true,\r\n});\r\n\r\nObject.defineProperty(Mesh.prototype, \"renderOverlay\", {\r\n get: function (this: Mesh) {\r\n return this._renderOverlay;\r\n },\r\n set: function (this: Mesh, value: boolean) {\r\n if (value) {\r\n // Lazy Load the component.\r\n this.getScene().getOutlineRenderer();\r\n }\r\n this._renderOverlay = value;\r\n },\r\n enumerable: true,\r\n configurable: true,\r\n});\r\n\r\n/**\r\n * This class is responsible to draw the outline/overlay of meshes.\r\n * It should not be used directly but through the available method on mesh.\r\n */\r\nexport class OutlineRenderer implements ISceneComponent {\r\n /**\r\n * Stencil value used to avoid outline being seen within the mesh when the mesh is transparent\r\n */\r\n private static _StencilReference = 0x04;\r\n /**\r\n * The name of the component. Each component must have a unique name.\r\n */\r\n public name = SceneComponentConstants.NAME_OUTLINERENDERER;\r\n\r\n /**\r\n * The scene the component belongs to.\r\n */\r\n public scene: Scene;\r\n\r\n /**\r\n * Defines a zOffset default Factor to prevent zFighting between the overlay and the mesh.\r\n */\r\n public zOffset = 1;\r\n\r\n /**\r\n * Defines a zOffset default Unit to prevent zFighting between the overlay and the mesh.\r\n */\r\n public zOffsetUnits = 4; // 4 to account for projection a bit by default\r\n\r\n private _engine: Engine;\r\n private _savedDepthWrite: boolean;\r\n private _passIdForDrawWrapper: number[];\r\n\r\n /**\r\n * Instantiates a new outline renderer. (There could be only one per scene).\r\n * @param scene Defines the scene it belongs to\r\n */\r\n constructor(scene: Scene) {\r\n this.scene = scene;\r\n this._engine = scene.getEngine();\r\n this.scene._addComponent(this);\r\n this._passIdForDrawWrapper = [];\r\n for (let i = 0; i < 4; ++i) {\r\n this._passIdForDrawWrapper[i] = this._engine.createRenderPassId(`Outline Renderer (${i})`);\r\n }\r\n }\r\n\r\n /**\r\n * Register the component to one instance of a scene.\r\n */\r\n public register(): void {\r\n this.scene._beforeRenderingMeshStage.registerStep(SceneComponentConstants.STEP_BEFORERENDERINGMESH_OUTLINE, this, this._beforeRenderingMesh);\r\n this.scene._afterRenderingMeshStage.registerStep(SceneComponentConstants.STEP_AFTERRENDERINGMESH_OUTLINE, this, this._afterRenderingMesh);\r\n }\r\n\r\n /**\r\n * Rebuilds the elements related to this component in case of\r\n * context lost for instance.\r\n */\r\n public rebuild(): void {\r\n // Nothing to do here.\r\n }\r\n\r\n /**\r\n * Disposes the component and the associated resources.\r\n */\r\n public dispose(): void {\r\n for (let i = 0; i < this._passIdForDrawWrapper.length; ++i) {\r\n this._engine.releaseRenderPassId(this._passIdForDrawWrapper[i]);\r\n }\r\n }\r\n\r\n /**\r\n * Renders the outline in the canvas.\r\n * @param subMesh Defines the sumesh to render\r\n * @param batch Defines the batch of meshes in case of instances\r\n * @param useOverlay Defines if the rendering is for the overlay or the outline\r\n * @param renderPassId Render pass id to use to render the mesh\r\n */\r\n public render(subMesh: SubMesh, batch: _InstancesBatch, useOverlay: boolean = false, renderPassId?: number): void {\r\n renderPassId = renderPassId ?? this._passIdForDrawWrapper[0];\r\n\r\n const scene = this.scene;\r\n const engine = scene.getEngine();\r\n\r\n const hardwareInstancedRendering =\r\n engine.getCaps().instancedArrays &&\r\n ((batch.visibleInstances[subMesh._id] !== null && batch.visibleInstances[subMesh._id] !== undefined) || subMesh.getRenderingMesh().hasThinInstances);\r\n\r\n if (!this.isReady(subMesh, hardwareInstancedRendering, renderPassId)) {\r\n return;\r\n }\r\n\r\n const ownerMesh = subMesh.getMesh();\r\n const replacementMesh = ownerMesh._internalAbstractMeshDataInfo._actAsRegularMesh ? ownerMesh : null;\r\n const renderingMesh = subMesh.getRenderingMesh();\r\n const effectiveMesh = replacementMesh ? replacementMesh : renderingMesh;\r\n const material = subMesh.getMaterial();\r\n\r\n if (!material || !scene.activeCamera) {\r\n return;\r\n }\r\n\r\n const drawWrapper = subMesh._getDrawWrapper(renderPassId)!;\r\n const effect = DrawWrapper.GetEffect(drawWrapper)!;\r\n\r\n engine.enableEffect(drawWrapper);\r\n\r\n // Logarithmic depth\r\n if ((material).useLogarithmicDepth) {\r\n effect.setFloat(\"logarithmicDepthConstant\", 2.0 / (Math.log(scene.activeCamera.maxZ + 1.0) / Math.LN2));\r\n }\r\n\r\n effect.setFloat(\"offset\", useOverlay ? 0 : renderingMesh.outlineWidth);\r\n effect.setColor4(\"color\", useOverlay ? renderingMesh.overlayColor : renderingMesh.outlineColor, useOverlay ? renderingMesh.overlayAlpha : material.alpha);\r\n effect.setMatrix(\"viewProjection\", scene.getTransformMatrix());\r\n effect.setMatrix(\"world\", effectiveMesh.getWorldMatrix());\r\n\r\n // Bones\r\n if (renderingMesh.useBones && renderingMesh.computeBonesUsingShaders && renderingMesh.skeleton) {\r\n effect.setMatrices(\"mBones\", renderingMesh.skeleton.getTransformMatrices(renderingMesh));\r\n }\r\n\r\n if (renderingMesh.morphTargetManager && renderingMesh.morphTargetManager.isUsingTextureForTargets) {\r\n renderingMesh.morphTargetManager._bind(effect);\r\n }\r\n\r\n // Morph targets\r\n MaterialHelper.BindMorphTargetParameters(renderingMesh, effect);\r\n\r\n if (!hardwareInstancedRendering) {\r\n renderingMesh._bind(subMesh, effect, material.fillMode);\r\n }\r\n\r\n // Alpha test\r\n if (material && material.needAlphaTesting()) {\r\n const alphaTexture = material.getAlphaTestTexture();\r\n if (alphaTexture) {\r\n effect.setTexture(\"diffuseSampler\", alphaTexture);\r\n effect.setMatrix(\"diffuseMatrix\", alphaTexture.getTextureMatrix());\r\n }\r\n }\r\n\r\n // Clip plane\r\n bindClipPlane(effect, material, scene);\r\n\r\n engine.setZOffset(-this.zOffset);\r\n engine.setZOffsetUnits(-this.zOffsetUnits);\r\n\r\n renderingMesh._processRendering(effectiveMesh, subMesh, effect, material.fillMode, batch, hardwareInstancedRendering, (isInstance, world) => {\r\n effect.setMatrix(\"world\", world);\r\n });\r\n\r\n engine.setZOffset(0);\r\n engine.setZOffsetUnits(0);\r\n }\r\n\r\n /**\r\n * Returns whether or not the outline renderer is ready for a given submesh.\r\n * All the dependencies e.g. submeshes, texture, effect... mus be ready\r\n * @param subMesh Defines the submesh to check readiness for\r\n * @param useInstances Defines whether wee are trying to render instances or not\r\n * @param renderPassId Render pass id to use to render the mesh\r\n * @returns true if ready otherwise false\r\n */\r\n public isReady(subMesh: SubMesh, useInstances: boolean, renderPassId?: number): boolean {\r\n renderPassId = renderPassId ?? this._passIdForDrawWrapper[0];\r\n\r\n const defines = [];\r\n const attribs = [VertexBuffer.PositionKind, VertexBuffer.NormalKind];\r\n\r\n const mesh = subMesh.getMesh();\r\n const material = subMesh.getMaterial();\r\n\r\n if (!material) {\r\n return false;\r\n }\r\n\r\n const scene = mesh.getScene();\r\n\r\n // Alpha test\r\n if (material.needAlphaTesting()) {\r\n defines.push(\"#define ALPHATEST\");\r\n if (mesh.isVerticesDataPresent(VertexBuffer.UVKind)) {\r\n attribs.push(VertexBuffer.UVKind);\r\n defines.push(\"#define UV1\");\r\n }\r\n if (mesh.isVerticesDataPresent(VertexBuffer.UV2Kind)) {\r\n attribs.push(VertexBuffer.UV2Kind);\r\n defines.push(\"#define UV2\");\r\n }\r\n }\r\n //Logarithmic depth\r\n if ((material).useLogarithmicDepth) {\r\n defines.push(\"#define LOGARITHMICDEPTH\");\r\n }\r\n // Clip planes\r\n prepareStringDefinesForClipPlanes(material, scene, defines);\r\n\r\n // Bones\r\n if (mesh.useBones && mesh.computeBonesUsingShaders) {\r\n attribs.push(VertexBuffer.MatricesIndicesKind);\r\n attribs.push(VertexBuffer.MatricesWeightsKind);\r\n if (mesh.numBoneInfluencers > 4) {\r\n attribs.push(VertexBuffer.MatricesIndicesExtraKind);\r\n attribs.push(VertexBuffer.MatricesWeightsExtraKind);\r\n }\r\n defines.push(\"#define NUM_BONE_INFLUENCERS \" + mesh.numBoneInfluencers);\r\n defines.push(\"#define BonesPerMesh \" + (mesh.skeleton ? mesh.skeleton.bones.length + 1 : 0));\r\n } else {\r\n defines.push(\"#define NUM_BONE_INFLUENCERS 0\");\r\n }\r\n\r\n // Morph targets\r\n const morphTargetManager = (mesh as Mesh).morphTargetManager;\r\n let numMorphInfluencers = 0;\r\n if (morphTargetManager) {\r\n if (morphTargetManager.numInfluencers > 0) {\r\n numMorphInfluencers = morphTargetManager.numInfluencers;\r\n\r\n defines.push(\"#define MORPHTARGETS\");\r\n defines.push(\"#define NUM_MORPH_INFLUENCERS \" + numMorphInfluencers);\r\n\r\n if (morphTargetManager.isUsingTextureForTargets) {\r\n defines.push(\"#define MORPHTARGETS_TEXTURE\");\r\n }\r\n\r\n MaterialHelper.PrepareAttributesForMorphTargetsInfluencers(attribs, mesh, numMorphInfluencers);\r\n }\r\n }\r\n\r\n // Instances\r\n if (useInstances) {\r\n defines.push(\"#define INSTANCES\");\r\n MaterialHelper.PushAttributesForInstances(attribs);\r\n if (subMesh.getRenderingMesh().hasThinInstances) {\r\n defines.push(\"#define THIN_INSTANCES\");\r\n }\r\n }\r\n\r\n // Get correct effect\r\n const drawWrapper = subMesh._getDrawWrapper(renderPassId, true)!;\r\n const cachedDefines = drawWrapper.defines;\r\n const join = defines.join(\"\\n\");\r\n\r\n if (cachedDefines !== join) {\r\n const uniforms = [\r\n \"world\",\r\n \"mBones\",\r\n \"viewProjection\",\r\n \"diffuseMatrix\",\r\n \"offset\",\r\n \"color\",\r\n \"logarithmicDepthConstant\",\r\n \"morphTargetInfluences\",\r\n \"morphTargetTextureInfo\",\r\n \"morphTargetTextureIndices\",\r\n ];\r\n addClipPlaneUniforms(uniforms);\r\n\r\n drawWrapper.setEffect(\r\n this.scene.getEngine().createEffect(\"outline\", attribs, uniforms, [\"diffuseSampler\", \"morphTargets\"], join, undefined, undefined, undefined, {\r\n maxSimultaneousMorphTargets: numMorphInfluencers,\r\n }),\r\n join\r\n );\r\n }\r\n\r\n return drawWrapper.effect!.isReady();\r\n }\r\n\r\n private _beforeRenderingMesh(mesh: Mesh, subMesh: SubMesh, batch: _InstancesBatch): void {\r\n // Outline - step 1\r\n this._savedDepthWrite = this._engine.getDepthWrite();\r\n if (mesh.renderOutline) {\r\n const material = subMesh.getMaterial();\r\n if (material && material.needAlphaBlendingForMesh(mesh)) {\r\n this._engine.cacheStencilState();\r\n // Draw only to stencil buffer for the original mesh\r\n // The resulting stencil buffer will be used so the outline is not visible inside the mesh when the mesh is transparent\r\n this._engine.setDepthWrite(false);\r\n this._engine.setColorWrite(false);\r\n this._engine.setStencilBuffer(true);\r\n this._engine.setStencilOperationPass(Constants.REPLACE);\r\n this._engine.setStencilFunction(Constants.ALWAYS);\r\n this._engine.setStencilMask(OutlineRenderer._StencilReference);\r\n this._engine.setStencilFunctionReference(OutlineRenderer._StencilReference);\r\n this._engine.stencilStateComposer.useStencilGlobalOnly = true;\r\n this.render(subMesh, batch, /* This sets offset to 0 */ true, this._passIdForDrawWrapper[1]);\r\n\r\n this._engine.setColorWrite(true);\r\n this._engine.setStencilFunction(Constants.NOTEQUAL);\r\n }\r\n\r\n // Draw the outline using the above stencil if needed to avoid drawing within the mesh\r\n this._engine.setDepthWrite(false);\r\n this.render(subMesh, batch, false, this._passIdForDrawWrapper[0]);\r\n this._engine.setDepthWrite(this._savedDepthWrite);\r\n\r\n if (material && material.needAlphaBlendingForMesh(mesh)) {\r\n this._engine.stencilStateComposer.useStencilGlobalOnly = false;\r\n this._engine.restoreStencilState();\r\n }\r\n }\r\n }\r\n\r\n private _afterRenderingMesh(mesh: Mesh, subMesh: SubMesh, batch: _InstancesBatch): void {\r\n // Overlay\r\n if (mesh.renderOverlay) {\r\n const currentMode = this._engine.getAlphaMode();\r\n const alphaBlendState = this._engine.alphaState.alphaBlend;\r\n this._engine.setAlphaMode(Constants.ALPHA_COMBINE);\r\n this.render(subMesh, batch, true, this._passIdForDrawWrapper[3]);\r\n this._engine.setAlphaMode(currentMode);\r\n this._engine.setDepthWrite(this._savedDepthWrite);\r\n this._engine.alphaState.alphaBlend = alphaBlendState;\r\n }\r\n\r\n // Outline - step 2\r\n if (mesh.renderOutline && this._savedDepthWrite) {\r\n this._engine.setDepthWrite(true);\r\n this._engine.setColorWrite(false);\r\n this.render(subMesh, batch, false, this._passIdForDrawWrapper[2]);\r\n this._engine.setColorWrite(true);\r\n }\r\n }\r\n}\r\n","import type { IMultiRenderTargetOptions } from \"./multiRenderTarget\";\r\nimport { MultiRenderTarget } from \"./multiRenderTarget\";\r\nimport type { Engine } from \"../../Engines/engine\";\r\nimport type { RenderTargetTexture } from \"./renderTargetTexture\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { PostProcess } from \"../../PostProcesses/postProcess\";\r\nimport { ImageProcessingPostProcess } from \"../../PostProcesses/imageProcessingPostProcess\";\r\nimport type { Nullable } from \"../../types\";\r\n\r\n/**\r\n * A multi render target designed to render the prepass.\r\n * Prepass is a scene component used to render information in multiple textures\r\n * alongside with the scene materials rendering.\r\n * Note : This is an internal class, and you should NOT need to instanciate this.\r\n * Only the `PrePassRenderer` should instanciate this class.\r\n * It is more likely that you need a regular `MultiRenderTarget`\r\n * @internal\r\n */\r\nexport class PrePassRenderTarget extends MultiRenderTarget {\r\n /**\r\n * @internal\r\n */\r\n public _beforeCompositionPostProcesses: PostProcess[] = [];\r\n /**\r\n * Image processing post process for composition\r\n */\r\n public imageProcessingPostProcess: ImageProcessingPostProcess;\r\n\r\n /**\r\n * @internal\r\n */\r\n public _engine: Engine;\r\n\r\n /**\r\n * @internal\r\n */\r\n public _scene: Scene;\r\n\r\n /**\r\n * @internal\r\n */\r\n public _outputPostProcess: Nullable;\r\n\r\n /**\r\n * @internal\r\n */\r\n public _internalTextureDirty = false;\r\n\r\n /**\r\n * Is this render target enabled for prepass rendering\r\n */\r\n public enabled: boolean = false;\r\n\r\n /**\r\n * Render target associated with this prePassRenderTarget\r\n * If this is `null`, it means this prePassRenderTarget is associated with the scene\r\n */\r\n public renderTargetTexture: Nullable = null;\r\n\r\n public constructor(name: string, renderTargetTexture: Nullable, size: any, count: number, scene?: Scene, options?: IMultiRenderTargetOptions | undefined) {\r\n super(name, size, count, scene, options);\r\n\r\n this.renderTargetTexture = renderTargetTexture;\r\n }\r\n\r\n /**\r\n * Creates a composition effect for this RT\r\n * @internal\r\n */\r\n public _createCompositionEffect() {\r\n this.imageProcessingPostProcess = new ImageProcessingPostProcess(\"prePassComposition\", 1, null, undefined, this._engine);\r\n this.imageProcessingPostProcess._updateParameters();\r\n }\r\n\r\n /**\r\n * Checks that the size of this RT is still adapted to the desired render size.\r\n * @internal\r\n */\r\n public _checkSize() {\r\n const requiredWidth = this._engine.getRenderWidth(true);\r\n const requiredHeight = this._engine.getRenderHeight(true);\r\n\r\n const width = this.getRenderWidth();\r\n const height = this.getRenderHeight();\r\n\r\n if (width !== requiredWidth || height !== requiredHeight) {\r\n this.resize({ width: requiredWidth, height: requiredHeight });\r\n\r\n this._internalTextureDirty = true;\r\n }\r\n }\r\n\r\n /**\r\n * Changes the number of render targets in this MRT\r\n * Be careful as it will recreate all the data in the new texture.\r\n * @param count new texture count\r\n * @param options Specifies texture types and sampling modes for new textures\r\n * @param textureNames Specifies the names of the textures (optional)\r\n */\r\n public updateCount(count: number, options?: IMultiRenderTargetOptions, textureNames?: string[]) {\r\n super.updateCount(count, options, textureNames);\r\n this._internalTextureDirty = true;\r\n }\r\n\r\n /**\r\n * Resets the post processes chains applied to this RT.\r\n * @internal\r\n */\r\n public _resetPostProcessChain() {\r\n this._beforeCompositionPostProcesses.length = 0;\r\n }\r\n\r\n /**\r\n * Diposes this render target\r\n */\r\n public dispose() {\r\n const scene = this._scene;\r\n\r\n super.dispose();\r\n\r\n if (scene && scene.prePassRenderer) {\r\n const index = scene.prePassRenderer.renderTargets.indexOf(this);\r\n\r\n if (index !== -1) {\r\n scene.prePassRenderer.renderTargets.splice(index, 1);\r\n }\r\n }\r\n\r\n if (this.imageProcessingPostProcess) {\r\n this.imageProcessingPostProcess.dispose();\r\n }\r\n\r\n if (this.renderTargetTexture) {\r\n this.renderTargetTexture._prePassRenderTarget = null;\r\n }\r\n\r\n if (this._outputPostProcess) {\r\n this._outputPostProcess.autoClear = true;\r\n this._outputPostProcess.restoreDefaultInputTexture();\r\n }\r\n }\r\n}\r\n","import { PrePassRenderTarget } from \"../Materials/Textures/prePassRenderTarget\";\r\nimport type { Scene } from \"../scene\";\r\nimport type { Engine } from \"../Engines/engine\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport type { PostProcess } from \"../PostProcesses/postProcess\";\r\nimport type { Effect } from \"../Materials/effect\";\r\nimport { _WarnImport } from \"../Misc/devTools\";\r\nimport { Color4 } from \"../Maths/math.color\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\nimport { Material } from \"../Materials/material\";\r\nimport type { SubMesh } from \"../Meshes/subMesh\";\r\nimport type { PrePassEffectConfiguration } from \"./prePassEffectConfiguration\";\r\nimport type { RenderTargetTexture } from \"../Materials/Textures/renderTargetTexture\";\r\nimport { GeometryBufferRenderer } from \"../Rendering/geometryBufferRenderer\";\r\n\r\n/**\r\n * Renders a pre pass of the scene\r\n * This means every mesh in the scene will be rendered to a render target texture\r\n * And then this texture will be composited to the rendering canvas with post processes\r\n * It is necessary for effects like subsurface scattering or deferred shading\r\n */\r\nexport class PrePassRenderer {\r\n /**\r\n * @internal\r\n */\r\n public static _SceneComponentInitialization: (scene: Scene) => void = (_) => {\r\n throw _WarnImport(\"PrePassRendererSceneComponent\");\r\n };\r\n\r\n /**\r\n * To save performance, we can excluded skinned meshes from the prepass\r\n */\r\n public excludedSkinnedMesh: AbstractMesh[] = [];\r\n\r\n /**\r\n * Force material to be excluded from the prepass\r\n * Can be useful when `useGeometryBufferFallback` is set to `true`\r\n * and you don't want a material to show in the effect.\r\n */\r\n public excludedMaterials: Material[] = [];\r\n\r\n private _scene: Scene;\r\n private _engine: Engine;\r\n\r\n /**\r\n * Number of textures in the multi render target texture where the scene is directly rendered\r\n */\r\n public mrtCount: number = 0;\r\n\r\n private _mrtTypes: number[] = [];\r\n private _mrtFormats: number[] = [];\r\n private _mrtLayout: number[] = [];\r\n private _mrtNames: string[] = [];\r\n private _textureIndices: number[] = [];\r\n\r\n private _multiRenderAttachments: number[];\r\n private _defaultAttachments: number[];\r\n private _clearAttachments: number[];\r\n private _clearDepthAttachments: number[];\r\n\r\n /**\r\n * Returns the index of a texture in the multi render target texture array.\r\n * @param type Texture type\r\n * @returns The index\r\n */\r\n public getIndex(type: number): number {\r\n return this._textureIndices[type];\r\n }\r\n\r\n /**\r\n * How many samples are used for MSAA of the scene render target\r\n */\r\n public get samples() {\r\n return this.defaultRT.samples;\r\n }\r\n\r\n public set samples(n: number) {\r\n this.defaultRT.samples = n;\r\n }\r\n\r\n private _useSpecificClearForDepthTexture = false;\r\n\r\n /**\r\n * If set to true (default: false), the depth texture will be cleared with the depth value corresponding to the far plane (1 in normal mode, 0 in reverse depth buffer mode)\r\n * If set to false, the depth texture is always cleared with 0.\r\n */\r\n public get useSpecificClearForDepthTexture() {\r\n return this._useSpecificClearForDepthTexture;\r\n }\r\n\r\n public set useSpecificClearForDepthTexture(value: boolean) {\r\n if (this._useSpecificClearForDepthTexture === value) {\r\n return;\r\n }\r\n\r\n this._useSpecificClearForDepthTexture = value;\r\n this._isDirty = true;\r\n }\r\n\r\n /**\r\n * Describes the types and formats of the textures used by the pre-pass renderer\r\n */\r\n public static TextureFormats = [\r\n {\r\n purpose: Constants.PREPASS_IRRADIANCE_TEXTURE_TYPE,\r\n type: Constants.TEXTURETYPE_HALF_FLOAT,\r\n format: Constants.TEXTUREFORMAT_RGBA,\r\n name: \"prePass_Irradiance\",\r\n },\r\n {\r\n purpose: Constants.PREPASS_POSITION_TEXTURE_TYPE,\r\n type: Constants.TEXTURETYPE_HALF_FLOAT,\r\n format: Constants.TEXTUREFORMAT_RGBA,\r\n name: \"prePass_Position\",\r\n },\r\n {\r\n purpose: Constants.PREPASS_VELOCITY_TEXTURE_TYPE,\r\n type: Constants.TEXTURETYPE_UNSIGNED_INT,\r\n format: Constants.TEXTUREFORMAT_RGBA,\r\n name: \"prePass_Velocity\",\r\n },\r\n {\r\n purpose: Constants.PREPASS_REFLECTIVITY_TEXTURE_TYPE,\r\n type: Constants.TEXTURETYPE_UNSIGNED_INT,\r\n format: Constants.TEXTUREFORMAT_RGBA,\r\n name: \"prePass_Reflectivity\",\r\n },\r\n {\r\n purpose: Constants.PREPASS_COLOR_TEXTURE_TYPE,\r\n type: Constants.TEXTURETYPE_HALF_FLOAT,\r\n format: Constants.TEXTUREFORMAT_RGBA,\r\n name: \"prePass_Color\",\r\n },\r\n {\r\n purpose: Constants.PREPASS_DEPTH_TEXTURE_TYPE,\r\n type: Constants.TEXTURETYPE_FLOAT,\r\n format: Constants.TEXTUREFORMAT_R,\r\n name: \"prePass_Depth\",\r\n },\r\n {\r\n purpose: Constants.PREPASS_NORMAL_TEXTURE_TYPE,\r\n type: Constants.TEXTURETYPE_HALF_FLOAT,\r\n format: Constants.TEXTUREFORMAT_RGBA,\r\n name: \"prePass_Normal\",\r\n },\r\n {\r\n purpose: Constants.PREPASS_ALBEDO_SQRT_TEXTURE_TYPE,\r\n type: Constants.TEXTURETYPE_UNSIGNED_INT,\r\n format: Constants.TEXTUREFORMAT_RGBA,\r\n name: \"prePass_Albedo\",\r\n },\r\n ];\r\n\r\n private _isDirty: boolean = true;\r\n\r\n /**\r\n * The render target where the scene is directly rendered\r\n */\r\n public defaultRT: PrePassRenderTarget;\r\n\r\n /**\r\n * Configuration for prepass effects\r\n */\r\n private _effectConfigurations: PrePassEffectConfiguration[] = [];\r\n\r\n /**\r\n * @returns the prepass render target for the rendering pass.\r\n * If we are currently rendering a render target, it returns the PrePassRenderTarget\r\n * associated with that render target. Otherwise, it returns the scene default PrePassRenderTarget\r\n */\r\n public getRenderTarget(): PrePassRenderTarget {\r\n return this._currentTarget;\r\n }\r\n\r\n /**\r\n * @internal\r\n * Managed by the scene component\r\n * @param prePassRenderTarget\r\n */\r\n public _setRenderTarget(prePassRenderTarget: Nullable): void {\r\n if (prePassRenderTarget) {\r\n this._currentTarget = prePassRenderTarget;\r\n } else {\r\n this._currentTarget = this.defaultRT;\r\n this._engine.currentRenderPassId = this._scene.activeCamera?.renderPassId ?? this._currentTarget.renderPassId;\r\n }\r\n }\r\n\r\n /**\r\n * Returns true if the currently rendered prePassRenderTarget is the one\r\n * associated with the scene.\r\n */\r\n public get currentRTisSceneRT(): boolean {\r\n return this._currentTarget === this.defaultRT;\r\n }\r\n\r\n private _geometryBuffer: Nullable;\r\n\r\n /**\r\n * Prevents the PrePassRenderer from using the GeometryBufferRenderer as a fallback\r\n */\r\n public doNotUseGeometryRendererFallback = true;\r\n\r\n private _refreshGeometryBufferRendererLink() {\r\n if (!this.doNotUseGeometryRendererFallback) {\r\n this._geometryBuffer = this._scene.enableGeometryBufferRenderer();\r\n\r\n if (!this._geometryBuffer) {\r\n // Not supported\r\n this.doNotUseGeometryRendererFallback = true;\r\n return;\r\n }\r\n\r\n this._geometryBuffer._linkPrePassRenderer(this);\r\n } else {\r\n if (this._geometryBuffer) {\r\n this._geometryBuffer._unlinkPrePassRenderer();\r\n }\r\n this._geometryBuffer = null;\r\n this._scene.disableGeometryBufferRenderer();\r\n }\r\n }\r\n\r\n private _currentTarget: PrePassRenderTarget;\r\n\r\n /**\r\n * All the render targets generated by prepass\r\n */\r\n public renderTargets: PrePassRenderTarget[] = [];\r\n\r\n private readonly _clearColor = new Color4(0, 0, 0, 0);\r\n private readonly _clearDepthColor = new Color4(1e8, 0, 0, 1); // \"infinity\" value - depth in the depth texture is view.z, not a 0..1 value!\r\n\r\n private _enabled: boolean = false;\r\n\r\n private _needsCompositionForThisPass = false;\r\n private _postProcessesSourceForThisPass: Nullable[];\r\n\r\n /**\r\n * Indicates if the prepass is enabled\r\n */\r\n public get enabled() {\r\n return this._enabled;\r\n }\r\n\r\n /**\r\n * Set to true to disable gamma transform in PrePass.\r\n * Can be useful in case you already proceed to gamma transform on a material level\r\n * and your post processes don't need to be in linear color space.\r\n */\r\n public disableGammaTransform = false;\r\n\r\n /**\r\n * Instantiates a prepass renderer\r\n * @param scene The scene\r\n */\r\n constructor(scene: Scene) {\r\n this._scene = scene;\r\n this._engine = scene.getEngine();\r\n\r\n let type = Constants.TEXTURETYPE_UNSIGNED_BYTE;\r\n if (this._engine._caps.textureFloat && this._engine._caps.textureFloatLinearFiltering) {\r\n type = Constants.TEXTURETYPE_FLOAT;\r\n } else if (this._engine._caps.textureHalfFloat && this._engine._caps.textureHalfFloatLinearFiltering) {\r\n type = Constants.TEXTURETYPE_HALF_FLOAT;\r\n }\r\n\r\n if (type !== Constants.TEXTURETYPE_FLOAT) {\r\n for (let i = 0; i < PrePassRenderer.TextureFormats.length; ++i) {\r\n if (PrePassRenderer.TextureFormats[i].type === Constants.TEXTURETYPE_FLOAT) {\r\n PrePassRenderer.TextureFormats[Constants.PREPASS_DEPTH_TEXTURE_TYPE].type = type;\r\n }\r\n }\r\n }\r\n\r\n PrePassRenderer._SceneComponentInitialization(this._scene);\r\n this.defaultRT = this._createRenderTarget(\"sceneprePassRT\", null);\r\n this._currentTarget = this.defaultRT;\r\n }\r\n\r\n /**\r\n * Creates a new PrePassRenderTarget\r\n * This should be the only way to instantiate a `PrePassRenderTarget`\r\n * @param name Name of the `PrePassRenderTarget`\r\n * @param renderTargetTexture RenderTarget the `PrePassRenderTarget` will be attached to.\r\n * Can be `null` if the created `PrePassRenderTarget` is attached to the scene (default framebuffer).\r\n * @internal\r\n */\r\n public _createRenderTarget(name: string, renderTargetTexture: Nullable): PrePassRenderTarget {\r\n const rt = new PrePassRenderTarget(name, renderTargetTexture, { width: this._engine.getRenderWidth(), height: this._engine.getRenderHeight() }, 0, this._scene, {\r\n generateMipMaps: false,\r\n generateStencilBuffer: this._engine.isStencilEnable,\r\n defaultType: Constants.TEXTURETYPE_UNSIGNED_INT,\r\n types: [],\r\n drawOnlyOnFirstAttachmentByDefault: true,\r\n });\r\n\r\n this.renderTargets.push(rt);\r\n\r\n if (this._enabled) {\r\n // The pre-pass renderer is already enabled, so make sure we create the render target with the correct number of textures\r\n this._update();\r\n }\r\n\r\n return rt;\r\n }\r\n\r\n /**\r\n * Indicates if rendering a prepass is supported\r\n */\r\n public get isSupported() {\r\n return this._scene.getEngine().getCaps().drawBuffersExtension;\r\n }\r\n\r\n /**\r\n * Sets the proper output textures to draw in the engine.\r\n * @param effect The effect that is drawn. It can be or not be compatible with drawing to several output textures.\r\n * @param subMesh Submesh on which the effect is applied\r\n */\r\n public bindAttachmentsForEffect(effect: Effect, subMesh: SubMesh) {\r\n const material = subMesh.getMaterial();\r\n const isPrePassCapable = material && material.isPrePassCapable;\r\n const excluded = material && this.excludedMaterials.indexOf(material) !== -1;\r\n\r\n if (this.enabled && this._currentTarget.enabled) {\r\n if (effect._multiTarget && isPrePassCapable && !excluded) {\r\n this._engine.bindAttachments(this._multiRenderAttachments);\r\n } else {\r\n if (this._engine._currentRenderTarget) {\r\n this._engine.bindAttachments(this._defaultAttachments);\r\n } else {\r\n this._engine.restoreSingleAttachment();\r\n }\r\n\r\n if (this._geometryBuffer && this.currentRTisSceneRT && !excluded) {\r\n this._geometryBuffer.renderList!.push(subMesh.getRenderingMesh());\r\n }\r\n }\r\n }\r\n }\r\n\r\n private _reinitializeAttachments() {\r\n const multiRenderLayout = [];\r\n const clearLayout = [false];\r\n const clearDepthLayout = [false];\r\n const defaultLayout = [true];\r\n\r\n for (let i = 0; i < this.mrtCount; i++) {\r\n multiRenderLayout.push(true);\r\n\r\n if (i > 0) {\r\n if (this._useSpecificClearForDepthTexture && this._mrtLayout[i] === Constants.PREPASS_DEPTH_TEXTURE_TYPE) {\r\n clearLayout.push(false);\r\n clearDepthLayout.push(true);\r\n } else {\r\n clearLayout.push(true);\r\n clearDepthLayout.push(false);\r\n }\r\n defaultLayout.push(false);\r\n }\r\n }\r\n\r\n this._multiRenderAttachments = this._engine.buildTextureLayout(multiRenderLayout);\r\n this._clearAttachments = this._engine.buildTextureLayout(clearLayout);\r\n this._clearDepthAttachments = this._engine.buildTextureLayout(clearDepthLayout);\r\n this._defaultAttachments = this._engine.buildTextureLayout(defaultLayout);\r\n }\r\n\r\n private _resetLayout() {\r\n for (let i = 0; i < PrePassRenderer.TextureFormats.length; i++) {\r\n this._textureIndices[PrePassRenderer.TextureFormats[i].purpose] = -1;\r\n }\r\n\r\n this._textureIndices[Constants.PREPASS_COLOR_TEXTURE_TYPE] = 0;\r\n this._mrtLayout = [Constants.PREPASS_COLOR_TEXTURE_TYPE];\r\n this._mrtTypes = [PrePassRenderer.TextureFormats[Constants.PREPASS_COLOR_TEXTURE_TYPE].type];\r\n this._mrtFormats = [PrePassRenderer.TextureFormats[Constants.PREPASS_COLOR_TEXTURE_TYPE].format];\r\n this._mrtNames = [PrePassRenderer.TextureFormats[Constants.PREPASS_COLOR_TEXTURE_TYPE].name];\r\n this.mrtCount = 1;\r\n }\r\n\r\n private _updateGeometryBufferLayout() {\r\n this._refreshGeometryBufferRendererLink();\r\n\r\n if (this._geometryBuffer) {\r\n this._geometryBuffer._resetLayout();\r\n\r\n const texturesActivated = [];\r\n\r\n for (let i = 0; i < this._mrtLayout.length; i++) {\r\n texturesActivated.push(false);\r\n }\r\n\r\n this._geometryBuffer._linkInternalTexture(this.defaultRT.getInternalTexture()!);\r\n\r\n const matches = [\r\n {\r\n prePassConstant: Constants.PREPASS_DEPTH_TEXTURE_TYPE,\r\n geometryBufferConstant: GeometryBufferRenderer.DEPTH_TEXTURE_TYPE,\r\n },\r\n {\r\n prePassConstant: Constants.PREPASS_NORMAL_TEXTURE_TYPE,\r\n geometryBufferConstant: GeometryBufferRenderer.NORMAL_TEXTURE_TYPE,\r\n },\r\n {\r\n prePassConstant: Constants.PREPASS_POSITION_TEXTURE_TYPE,\r\n geometryBufferConstant: GeometryBufferRenderer.POSITION_TEXTURE_TYPE,\r\n },\r\n {\r\n prePassConstant: Constants.PREPASS_REFLECTIVITY_TEXTURE_TYPE,\r\n geometryBufferConstant: GeometryBufferRenderer.REFLECTIVITY_TEXTURE_TYPE,\r\n },\r\n {\r\n prePassConstant: Constants.PREPASS_VELOCITY_TEXTURE_TYPE,\r\n geometryBufferConstant: GeometryBufferRenderer.VELOCITY_TEXTURE_TYPE,\r\n },\r\n ];\r\n\r\n // replace textures in the geometryBuffer RT\r\n for (let i = 0; i < matches.length; i++) {\r\n const index = this._mrtLayout.indexOf(matches[i].prePassConstant);\r\n if (index !== -1) {\r\n this._geometryBuffer._forceTextureType(matches[i].geometryBufferConstant, index);\r\n texturesActivated[index] = true;\r\n }\r\n }\r\n\r\n this._geometryBuffer._setAttachments(this._engine.buildTextureLayout(texturesActivated));\r\n }\r\n }\r\n\r\n /**\r\n * Restores attachments for single texture draw.\r\n */\r\n public restoreAttachments() {\r\n if (this.enabled && this._currentTarget.enabled && this._defaultAttachments) {\r\n if (this._engine._currentRenderTarget) {\r\n this._engine.bindAttachments(this._defaultAttachments);\r\n } else {\r\n this._engine.restoreSingleAttachment();\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public _beforeDraw(camera?: Camera, faceIndex?: number, layer?: number) {\r\n // const previousEnabled = this._enabled && this._currentTarget.enabled;\r\n\r\n if (this._isDirty) {\r\n this._update();\r\n }\r\n\r\n if (!this._enabled || !this._currentTarget.enabled) {\r\n return;\r\n }\r\n\r\n if (this._geometryBuffer) {\r\n this._geometryBuffer.renderList = [];\r\n }\r\n\r\n this._setupOutputForThisPass(this._currentTarget, camera);\r\n }\r\n\r\n private _prepareFrame(prePassRenderTarget: PrePassRenderTarget, faceIndex?: number, layer?: number) {\r\n if (prePassRenderTarget.renderTargetTexture) {\r\n prePassRenderTarget.renderTargetTexture._prepareFrame(this._scene, faceIndex, layer, prePassRenderTarget.renderTargetTexture.useCameraPostProcesses);\r\n } else if (this._postProcessesSourceForThisPass.length) {\r\n this._scene.postProcessManager._prepareFrame();\r\n } else {\r\n this._engine.restoreDefaultFramebuffer();\r\n }\r\n }\r\n\r\n /**\r\n * Sets an intermediary texture between prepass and postprocesses. This texture\r\n * will be used as input for post processes\r\n * @param rt\r\n * @returns true if there are postprocesses that will use this texture,\r\n * false if there is no postprocesses - and the function has no effect\r\n */\r\n public setCustomOutput(rt: RenderTargetTexture) {\r\n const firstPP = this._postProcessesSourceForThisPass[0];\r\n if (!firstPP) {\r\n return false;\r\n }\r\n\r\n firstPP.inputTexture = rt.renderTarget!;\r\n\r\n return true;\r\n }\r\n\r\n private _renderPostProcesses(prePassRenderTarget: PrePassRenderTarget, faceIndex?: number) {\r\n const firstPP = this._postProcessesSourceForThisPass[0];\r\n const outputTexture = firstPP ? firstPP.inputTexture : prePassRenderTarget.renderTargetTexture ? prePassRenderTarget.renderTargetTexture.renderTarget : null;\r\n\r\n // Build post process chain for this prepass post draw\r\n let postProcessChain = this._currentTarget._beforeCompositionPostProcesses;\r\n\r\n if (this._needsCompositionForThisPass) {\r\n postProcessChain = postProcessChain.concat([this._currentTarget.imageProcessingPostProcess]);\r\n }\r\n\r\n // Activates and renders the chain\r\n if (postProcessChain.length) {\r\n this._scene.postProcessManager._prepareFrame(this._currentTarget.renderTarget?.texture, postProcessChain);\r\n this._scene.postProcessManager.directRender(postProcessChain, outputTexture, false, faceIndex);\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _afterDraw(faceIndex?: number, layer?: number) {\r\n if (this._enabled && this._currentTarget.enabled) {\r\n this._prepareFrame(this._currentTarget, faceIndex, layer);\r\n this._renderPostProcesses(this._currentTarget, faceIndex);\r\n }\r\n }\r\n\r\n /**\r\n * Clears the current prepass render target (in the sense of settings pixels to the scene clear color value)\r\n * @internal\r\n */\r\n public _clear() {\r\n if (this._enabled && this._currentTarget.enabled) {\r\n this._bindFrameBuffer(this._currentTarget);\r\n\r\n // Clearing other attachment with 0 on all other attachments\r\n this._engine.bindAttachments(this._clearAttachments);\r\n this._engine.clear(this._clearColor, true, false, false);\r\n if (this._useSpecificClearForDepthTexture) {\r\n this._engine.bindAttachments(this._clearDepthAttachments);\r\n this._engine.clear(this._clearDepthColor, true, false, false);\r\n }\r\n // Regular clear color with the scene clear color of the 1st attachment\r\n this._engine.bindAttachments(this._defaultAttachments);\r\n }\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n private _bindFrameBuffer(prePassRenderTarget: PrePassRenderTarget) {\r\n if (this._enabled && this._currentTarget.enabled) {\r\n this._currentTarget._checkSize();\r\n const internalTexture = this._currentTarget.renderTarget;\r\n if (internalTexture) {\r\n this._engine.bindFramebuffer(internalTexture);\r\n }\r\n }\r\n }\r\n\r\n private _setEnabled(enabled: boolean) {\r\n this._enabled = enabled;\r\n }\r\n\r\n private _setRenderTargetEnabled(prePassRenderTarget: PrePassRenderTarget, enabled: boolean) {\r\n prePassRenderTarget.enabled = enabled;\r\n if (!enabled) {\r\n this._unlinkInternalTexture(prePassRenderTarget);\r\n }\r\n }\r\n\r\n /**\r\n * Adds an effect configuration to the prepass render target.\r\n * If an effect has already been added, it won't add it twice and will return the configuration\r\n * already present.\r\n * @param cfg the effect configuration\r\n * @returns the effect configuration now used by the prepass\r\n */\r\n public addEffectConfiguration(cfg: PrePassEffectConfiguration): PrePassEffectConfiguration {\r\n // Do not add twice\r\n for (let i = 0; i < this._effectConfigurations.length; i++) {\r\n if (this._effectConfigurations[i].name === cfg.name) {\r\n return this._effectConfigurations[i];\r\n }\r\n }\r\n\r\n this._effectConfigurations.push(cfg);\r\n return cfg;\r\n }\r\n\r\n private _enable() {\r\n const previousMrtCount = this.mrtCount;\r\n\r\n for (let i = 0; i < this._effectConfigurations.length; i++) {\r\n if (this._effectConfigurations[i].enabled) {\r\n this._enableTextures(this._effectConfigurations[i].texturesRequired);\r\n }\r\n }\r\n\r\n for (let i = 0; i < this.renderTargets.length; i++) {\r\n if (this.mrtCount !== previousMrtCount || this.renderTargets[i].count !== this.mrtCount) {\r\n this.renderTargets[i].updateCount(this.mrtCount, { types: this._mrtTypes, formats: this._mrtFormats }, this._mrtNames.concat(\"prePass_DepthBuffer\"));\r\n }\r\n\r\n this.renderTargets[i]._resetPostProcessChain();\r\n\r\n for (let j = 0; j < this._effectConfigurations.length; j++) {\r\n if (this._effectConfigurations[j].enabled) {\r\n // TODO : subsurface scattering has 1 scene-wide effect configuration\r\n // solution : do not stock postProcess on effectConfiguration, but in the prepassRenderTarget (hashmap configuration => postProcess)\r\n // And call createPostProcess whenever the post process does not exist in the RT\r\n if (!this._effectConfigurations[j].postProcess && this._effectConfigurations[j].createPostProcess) {\r\n this._effectConfigurations[j].createPostProcess!();\r\n }\r\n\r\n if (this._effectConfigurations[j].postProcess) {\r\n this.renderTargets[i]._beforeCompositionPostProcesses.push(this._effectConfigurations[j].postProcess!);\r\n }\r\n }\r\n }\r\n }\r\n\r\n this._reinitializeAttachments();\r\n this._setEnabled(true);\r\n this._updateGeometryBufferLayout();\r\n }\r\n\r\n private _disable() {\r\n this._setEnabled(false);\r\n\r\n for (let i = 0; i < this.renderTargets.length; i++) {\r\n this._setRenderTargetEnabled(this.renderTargets[i], false);\r\n }\r\n\r\n this._resetLayout();\r\n\r\n for (let i = 0; i < this._effectConfigurations.length; i++) {\r\n this._effectConfigurations[i].enabled = false;\r\n }\r\n }\r\n\r\n private _getPostProcessesSource(prePassRenderTarget: PrePassRenderTarget, camera?: Camera): Nullable[] {\r\n if (camera) {\r\n return camera._postProcesses;\r\n } else if (prePassRenderTarget.renderTargetTexture) {\r\n if (prePassRenderTarget.renderTargetTexture.useCameraPostProcesses) {\r\n const camera = prePassRenderTarget.renderTargetTexture.activeCamera ? prePassRenderTarget.renderTargetTexture.activeCamera : this._scene.activeCamera;\r\n return camera ? camera._postProcesses : [];\r\n } else if (prePassRenderTarget.renderTargetTexture.postProcesses) {\r\n return prePassRenderTarget.renderTargetTexture.postProcesses;\r\n } else {\r\n return [];\r\n }\r\n } else {\r\n return this._scene.activeCamera ? this._scene.activeCamera._postProcesses : [];\r\n }\r\n }\r\n\r\n private _setupOutputForThisPass(prePassRenderTarget: PrePassRenderTarget, camera?: Camera) {\r\n // Order is : draw ===> prePassRenderTarget._postProcesses ==> ipp ==> camera._postProcesses\r\n const secondaryCamera = camera && this._scene.activeCameras && !!this._scene.activeCameras.length && this._scene.activeCameras.indexOf(camera) !== 0;\r\n this._postProcessesSourceForThisPass = this._getPostProcessesSource(prePassRenderTarget, camera);\r\n this._postProcessesSourceForThisPass = this._postProcessesSourceForThisPass.filter((pp) => {\r\n return pp != null;\r\n });\r\n this._scene.autoClear = true;\r\n\r\n const cameraHasImageProcessing = this._hasImageProcessing(this._postProcessesSourceForThisPass);\r\n this._needsCompositionForThisPass = !cameraHasImageProcessing && !this.disableGammaTransform && this._needsImageProcessing() && !secondaryCamera;\r\n\r\n const firstCameraPP = this._getFirstPostProcess(this._postProcessesSourceForThisPass);\r\n const firstPrePassPP = prePassRenderTarget._beforeCompositionPostProcesses && prePassRenderTarget._beforeCompositionPostProcesses[0];\r\n let firstPP = null;\r\n\r\n // Setting the scene-wide post process configuration\r\n this._scene.imageProcessingConfiguration.applyByPostProcess = this._needsCompositionForThisPass || cameraHasImageProcessing;\r\n\r\n // Create composition effect if needed\r\n if (this._needsCompositionForThisPass && !prePassRenderTarget.imageProcessingPostProcess) {\r\n prePassRenderTarget._createCompositionEffect();\r\n }\r\n\r\n // Setting the prePassRenderTarget as input texture of the first PP\r\n if (firstPrePassPP) {\r\n firstPP = firstPrePassPP;\r\n } else if (this._needsCompositionForThisPass) {\r\n firstPP = prePassRenderTarget.imageProcessingPostProcess;\r\n } else if (firstCameraPP) {\r\n firstPP = firstCameraPP;\r\n }\r\n\r\n this._bindFrameBuffer(prePassRenderTarget);\r\n this._linkInternalTexture(prePassRenderTarget, firstPP);\r\n }\r\n\r\n private _linkInternalTexture(prePassRenderTarget: PrePassRenderTarget, postProcess: Nullable) {\r\n if (postProcess) {\r\n postProcess.autoClear = false;\r\n postProcess.inputTexture = prePassRenderTarget.renderTarget!;\r\n }\r\n\r\n if (prePassRenderTarget._outputPostProcess !== postProcess) {\r\n if (prePassRenderTarget._outputPostProcess) {\r\n this._unlinkInternalTexture(prePassRenderTarget);\r\n }\r\n prePassRenderTarget._outputPostProcess = postProcess;\r\n }\r\n\r\n if (prePassRenderTarget._internalTextureDirty) {\r\n this._updateGeometryBufferLayout();\r\n prePassRenderTarget._internalTextureDirty = false;\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _unlinkInternalTexture(prePassRenderTarget: PrePassRenderTarget) {\r\n if (prePassRenderTarget._outputPostProcess) {\r\n prePassRenderTarget._outputPostProcess.autoClear = true;\r\n prePassRenderTarget._outputPostProcess.restoreDefaultInputTexture();\r\n prePassRenderTarget._outputPostProcess = null;\r\n }\r\n }\r\n\r\n private _needsImageProcessing(): boolean {\r\n for (let i = 0; i < this._effectConfigurations.length; i++) {\r\n if (this._effectConfigurations[i].enabled && this._effectConfigurations[i].needsImageProcessing) {\r\n return true;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n private _hasImageProcessing(postProcesses: Nullable[]): boolean {\r\n let isIPPAlreadyPresent = false;\r\n if (postProcesses) {\r\n for (let i = 0; i < postProcesses.length; i++) {\r\n if (postProcesses[i]?.getClassName() === \"ImageProcessingPostProcess\") {\r\n isIPPAlreadyPresent = true;\r\n break;\r\n }\r\n }\r\n }\r\n\r\n return isIPPAlreadyPresent;\r\n }\r\n\r\n /**\r\n * Internal, gets the first post proces.\r\n * @param postProcesses\r\n * @returns the first post process to be run on this camera.\r\n */\r\n private _getFirstPostProcess(postProcesses: Nullable[]): Nullable {\r\n for (let ppIndex = 0; ppIndex < postProcesses.length; ppIndex++) {\r\n if (postProcesses[ppIndex] !== null) {\r\n return postProcesses[ppIndex];\r\n }\r\n }\r\n return null;\r\n }\r\n\r\n /**\r\n * Marks the prepass renderer as dirty, triggering a check if the prepass is necessary for the next rendering.\r\n */\r\n public markAsDirty() {\r\n this._isDirty = true;\r\n }\r\n\r\n /**\r\n * Enables a texture on the MultiRenderTarget for prepass\r\n * @param types\r\n */\r\n private _enableTextures(types: number[]) {\r\n // For velocity : enable storage of previous matrices for instances\r\n this._scene.needsPreviousWorldMatrices = false;\r\n\r\n for (let i = 0; i < types.length; i++) {\r\n const type = types[i];\r\n\r\n if (this._textureIndices[type] === -1) {\r\n this._textureIndices[type] = this._mrtLayout.length;\r\n this._mrtLayout.push(type);\r\n\r\n this._mrtTypes.push(PrePassRenderer.TextureFormats[type].type);\r\n this._mrtFormats.push(PrePassRenderer.TextureFormats[type].format);\r\n this._mrtNames.push(PrePassRenderer.TextureFormats[type].name);\r\n this.mrtCount++;\r\n }\r\n\r\n if (type === Constants.PREPASS_VELOCITY_TEXTURE_TYPE) {\r\n this._scene.needsPreviousWorldMatrices = true;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Makes sure that the prepass renderer is up to date if it has been dirtified.\r\n */\r\n public update() {\r\n if (this._isDirty) {\r\n this._update();\r\n }\r\n }\r\n\r\n private _update() {\r\n this._disable();\r\n let enablePrePass = false;\r\n this._scene.imageProcessingConfiguration.applyByPostProcess = false;\r\n\r\n if (this._scene._depthPeelingRenderer && this._scene.useOrderIndependentTransparency) {\r\n this._scene._depthPeelingRenderer.setPrePassRenderer(this);\r\n enablePrePass = true;\r\n }\r\n\r\n for (let i = 0; i < this._scene.materials.length; i++) {\r\n if (this._scene.materials[i].setPrePassRenderer(this)) {\r\n enablePrePass = true;\r\n }\r\n }\r\n\r\n if (enablePrePass) {\r\n this._setRenderTargetEnabled(this.defaultRT, true);\r\n }\r\n\r\n let postProcesses;\r\n\r\n for (let i = 0; i < this.renderTargets.length; i++) {\r\n if (this.renderTargets[i].renderTargetTexture) {\r\n postProcesses = this._getPostProcessesSource(this.renderTargets[i]);\r\n } else {\r\n const camera = this._scene.activeCamera;\r\n if (!camera) {\r\n continue;\r\n }\r\n\r\n postProcesses = camera._postProcesses;\r\n }\r\n\r\n if (!postProcesses) {\r\n continue;\r\n }\r\n\r\n postProcesses = >postProcesses.filter((pp) => {\r\n return pp != null;\r\n });\r\n\r\n if (postProcesses) {\r\n for (let j = 0; j < postProcesses.length; j++) {\r\n if (postProcesses[j].setPrePassRenderer(this)) {\r\n this._setRenderTargetEnabled(this.renderTargets[i], true);\r\n enablePrePass = true;\r\n }\r\n }\r\n\r\n if (this._hasImageProcessing(postProcesses)) {\r\n this._scene.imageProcessingConfiguration.applyByPostProcess = true;\r\n }\r\n }\r\n }\r\n\r\n this._markAllMaterialsAsPrePassDirty();\r\n this._isDirty = false;\r\n\r\n if (enablePrePass) {\r\n this._enable();\r\n }\r\n }\r\n\r\n private _markAllMaterialsAsPrePassDirty() {\r\n const materials = this._scene.materials;\r\n\r\n for (let i = 0; i < materials.length; i++) {\r\n materials[i].markAsDirty(Material.PrePassDirtyFlag);\r\n }\r\n }\r\n\r\n /**\r\n * Disposes the prepass renderer.\r\n */\r\n public dispose() {\r\n for (let i = this.renderTargets.length - 1; i >= 0; i--) {\r\n this.renderTargets[i].dispose();\r\n }\r\n\r\n for (let i = 0; i < this._effectConfigurations.length; i++) {\r\n if (this._effectConfigurations[i].dispose) {\r\n this._effectConfigurations[i].dispose!();\r\n }\r\n }\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\nimport { Scene } from \"../scene\";\r\nimport type { ISceneComponent } from \"../sceneComponent\";\r\nimport { SceneComponentConstants } from \"../sceneComponent\";\r\nimport { PrePassRenderer } from \"./prePassRenderer\";\r\nimport { Logger } from \"../Misc/logger\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { SubMesh } from \"../Meshes/subMesh\";\r\nimport type { _InstancesBatch } from \"../Meshes/mesh\";\r\nimport type { Effect } from \"../Materials/effect\";\r\nimport type { Camera } from \"../Cameras/camera\";\r\nimport type { RenderTargetTexture } from \"../Materials/Textures/renderTargetTexture\";\r\nimport type { PrePassRenderTarget } from \"../Materials/Textures/prePassRenderTarget\";\r\n\r\ndeclare module \"../abstractScene\" {\r\n export interface AbstractScene {\r\n /** @internal (Backing field) */\r\n _prePassRenderer: Nullable;\r\n\r\n /**\r\n * Gets or Sets the current prepass renderer associated to the scene.\r\n */\r\n prePassRenderer: Nullable;\r\n\r\n /**\r\n * Enables the prepass and associates it with the scene\r\n * @returns the PrePassRenderer\r\n */\r\n enablePrePassRenderer(): Nullable;\r\n\r\n /**\r\n * Disables the prepass associated with the scene\r\n */\r\n disablePrePassRenderer(): void;\r\n }\r\n}\r\n\r\ndeclare module \"../Materials/Textures/renderTargetTexture\" {\r\n export interface RenderTargetTexture {\r\n /**\r\n * Gets or sets a boolean indicating that the prepass renderer should not be used with this render target\r\n */\r\n noPrePassRenderer: boolean;\r\n /** @internal */\r\n _prePassRenderTarget: Nullable;\r\n }\r\n}\r\n\r\nObject.defineProperty(Scene.prototype, \"prePassRenderer\", {\r\n get: function (this: Scene) {\r\n return this._prePassRenderer;\r\n },\r\n set: function (this: Scene, value: Nullable) {\r\n if (value && value.isSupported) {\r\n this._prePassRenderer = value;\r\n }\r\n },\r\n enumerable: true,\r\n configurable: true,\r\n});\r\n\r\nScene.prototype.enablePrePassRenderer = function (): Nullable {\r\n if (this._prePassRenderer) {\r\n return this._prePassRenderer;\r\n }\r\n\r\n this._prePassRenderer = new PrePassRenderer(this);\r\n\r\n if (!this._prePassRenderer.isSupported) {\r\n this._prePassRenderer = null;\r\n Logger.Error(\"PrePassRenderer needs WebGL 2 support.\\n\" + \"Maybe you tried to use the following features that need the PrePassRenderer :\\n\" + \" + Subsurface Scattering\");\r\n }\r\n\r\n return this._prePassRenderer;\r\n};\r\n\r\nScene.prototype.disablePrePassRenderer = function (): void {\r\n if (!this._prePassRenderer) {\r\n return;\r\n }\r\n\r\n this._prePassRenderer.dispose();\r\n this._prePassRenderer = null;\r\n};\r\n\r\n/**\r\n * Defines the Geometry Buffer scene component responsible to manage a G-Buffer useful\r\n * in several rendering techniques.\r\n */\r\nexport class PrePassRendererSceneComponent implements ISceneComponent {\r\n /**\r\n * The component name helpful to identify the component in the list of scene components.\r\n */\r\n public readonly name = SceneComponentConstants.NAME_PREPASSRENDERER;\r\n\r\n /**\r\n * The scene the component belongs to.\r\n */\r\n public scene: Scene;\r\n\r\n /**\r\n * Creates a new instance of the component for the given scene\r\n * @param scene Defines the scene to register the component in\r\n */\r\n constructor(scene: Scene) {\r\n this.scene = scene;\r\n }\r\n\r\n /**\r\n * Registers the component in a given scene\r\n */\r\n public register(): void {\r\n this.scene._beforeCameraDrawStage.registerStep(SceneComponentConstants.STEP_BEFORECAMERADRAW_PREPASS, this, this._beforeCameraDraw);\r\n this.scene._afterCameraDrawStage.registerStep(SceneComponentConstants.STEP_AFTERCAMERADRAW_PREPASS, this, this._afterCameraDraw);\r\n this.scene._beforeRenderTargetDrawStage.registerStep(SceneComponentConstants.STEP_BEFORERENDERTARGETDRAW_PREPASS, this, this._beforeRenderTargetDraw);\r\n this.scene._afterRenderTargetDrawStage.registerStep(SceneComponentConstants.STEP_AFTERCAMERADRAW_PREPASS, this, this._afterRenderTargetDraw);\r\n\r\n this.scene._beforeClearStage.registerStep(SceneComponentConstants.STEP_BEFORECLEAR_PREPASS, this, this._beforeClearStage);\r\n this.scene._beforeRenderTargetClearStage.registerStep(SceneComponentConstants.STEP_BEFORERENDERTARGETCLEAR_PREPASS, this, this._beforeRenderTargetClearStage);\r\n\r\n this.scene._beforeRenderingMeshStage.registerStep(SceneComponentConstants.STEP_BEFORERENDERINGMESH_PREPASS, this, this._beforeRenderingMeshStage);\r\n this.scene._afterRenderingMeshStage.registerStep(SceneComponentConstants.STEP_AFTERRENDERINGMESH_PREPASS, this, this._afterRenderingMeshStage);\r\n }\r\n\r\n private _beforeRenderTargetDraw(renderTarget: RenderTargetTexture, faceIndex?: number, layer?: number) {\r\n if (this.scene.prePassRenderer && !renderTarget.noPrePassRenderer) {\r\n this.scene.prePassRenderer._setRenderTarget(renderTarget._prePassRenderTarget);\r\n this.scene.prePassRenderer._beforeDraw(undefined, faceIndex, layer);\r\n }\r\n }\r\n\r\n private _afterRenderTargetDraw(renderTarget: RenderTargetTexture, faceIndex?: number, layer?: number) {\r\n if (this.scene.prePassRenderer && !renderTarget.noPrePassRenderer) {\r\n this.scene.prePassRenderer._afterDraw(faceIndex, layer);\r\n }\r\n }\r\n\r\n private _beforeRenderTargetClearStage(renderTarget: RenderTargetTexture) {\r\n if (this.scene.prePassRenderer && !renderTarget.noPrePassRenderer) {\r\n if (!renderTarget._prePassRenderTarget) {\r\n renderTarget._prePassRenderTarget = this.scene.prePassRenderer._createRenderTarget(renderTarget.name + \"_prePassRTT\", renderTarget);\r\n }\r\n this.scene.prePassRenderer._setRenderTarget(renderTarget._prePassRenderTarget);\r\n this.scene.prePassRenderer._clear();\r\n }\r\n }\r\n\r\n private _beforeCameraDraw(camera: Camera) {\r\n if (this.scene.prePassRenderer) {\r\n this.scene.prePassRenderer._setRenderTarget(null);\r\n this.scene.prePassRenderer._beforeDraw(camera);\r\n }\r\n }\r\n\r\n private _afterCameraDraw() {\r\n if (this.scene.prePassRenderer) {\r\n this.scene.prePassRenderer._afterDraw();\r\n }\r\n }\r\n\r\n private _beforeClearStage() {\r\n if (this.scene.prePassRenderer) {\r\n this.scene.prePassRenderer._setRenderTarget(null);\r\n this.scene.prePassRenderer._clear();\r\n }\r\n }\r\n\r\n private _beforeRenderingMeshStage(mesh: AbstractMesh, subMesh: SubMesh, batch: _InstancesBatch, effect: Nullable) {\r\n if (!effect) {\r\n return;\r\n }\r\n\r\n // Render to MRT\r\n const scene = mesh.getScene();\r\n if (scene.prePassRenderer) {\r\n scene.prePassRenderer.bindAttachmentsForEffect(effect, subMesh);\r\n }\r\n }\r\n\r\n private _afterRenderingMeshStage(mesh: AbstractMesh) {\r\n const scene = mesh.getScene();\r\n\r\n if (scene.prePassRenderer) {\r\n scene.prePassRenderer.restoreAttachments();\r\n }\r\n }\r\n\r\n /**\r\n * Rebuilds the elements related to this component in case of\r\n * context lost for instance.\r\n */\r\n public rebuild(): void {\r\n // Release textures first\r\n this.scene.disablePrePassRenderer();\r\n\r\n // Re-enable\r\n this.scene.enablePrePassRenderer();\r\n }\r\n\r\n /**\r\n * Disposes the component and the associated resources\r\n */\r\n public dispose(): void {\r\n this.scene.disablePrePassRenderer();\r\n }\r\n}\r\n\r\nPrePassRenderer._SceneComponentInitialization = (scene: Scene) => {\r\n // Register the G Buffer component to the scene.\r\n let component = scene._getComponent(SceneComponentConstants.NAME_PREPASSRENDERER) as PrePassRendererSceneComponent;\r\n if (!component) {\r\n component = new PrePassRendererSceneComponent(scene);\r\n scene._addComponent(component);\r\n }\r\n};\r\n","import { eParentChangeBehavior, KbViewerSettings } from '@models/interfaces/settings';\r\n\r\nconst settingDefaults: KbViewerSettings = {\r\n parentChangeBehavior: eParentChangeBehavior.PRESERVE_POSITION,\r\n};\r\ntype KbViewerSettingsOptions = keyof KbViewerSettings;\r\nconst settingsKey = 'KBVIEWER_SETTINGS';\r\n\r\nconst savedSettings = localStorage.getItem(settingsKey);\r\nconst settings: KbViewerSettings = savedSettings\r\n ? { ...settingDefaults, ...JSON.parse(savedSettings) }\r\n : { ...settingDefaults };\r\n\r\nexport class SettingsManagerService {\r\n static get(name: keyof KbViewerSettings): KbViewerSettings[T] {\r\n return settings[name];\r\n }\r\n\r\n static set(name: keyof KbViewerSettings, value: KbViewerSettings[T]): void {\r\n settings[name] = value;\r\n localStorage.setItem(settingsKey, JSON.stringify(settings));\r\n }\r\n}\r\n","import type { Nullable, FloatArray } from \"../../types\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { Vector3, Vector2, Vector4 } from \"../../Maths/math.vector\";\r\nimport { TmpVectors } from \"../../Maths/math.vector\";\r\nimport type { Color4 } from \"../../Maths/math.color\";\r\nimport { Mesh, _CreationDataStorage } from \"../mesh\";\r\nimport { VertexBuffer } from \"../../Buffers/buffer\";\r\nimport { VertexData } from \"../mesh.vertexData\";\r\nimport { CompatibilityOptions } from \"../../Compat/compatibilityOptions\";\r\n\r\n/**\r\n * Creates the VertexData for a Ribbon\r\n * @param options an object used to set the following optional parameters for the ribbon, required but can be empty\r\n * * pathArray array of paths, each of which an array of successive Vector3\r\n * * closeArray creates a seam between the first and the last paths of the pathArray, optional, default false\r\n * * closePath creates a seam between the first and the last points of each path of the path array, optional, default false\r\n * * offset a positive integer, only used when pathArray contains a single path (offset = 10 means the point 1 is joined to the point 11), default rounded half size of the pathArray length\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * * invertUV swaps in the U and V coordinates when applying a texture, optional, default false\r\n * * uvs a linear array, of length 2 * number of vertices, of custom UV values, optional\r\n * * colors a linear array, of length 4 * number of vertices, of custom color values, optional\r\n * @param options.pathArray\r\n * @param options.closeArray\r\n * @param options.closePath\r\n * @param options.offset\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param options.invertUV\r\n * @param options.uvs\r\n * @param options.colors\r\n * @returns the VertexData of the ribbon\r\n */\r\nexport function CreateRibbonVertexData(options: {\r\n pathArray: Vector3[][];\r\n closeArray?: boolean;\r\n closePath?: boolean;\r\n offset?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n invertUV?: boolean;\r\n uvs?: Vector2[];\r\n colors?: Color4[];\r\n}): VertexData {\r\n let pathArray: Vector3[][] = options.pathArray;\r\n const closeArray: boolean = options.closeArray || false;\r\n const closePath: boolean = options.closePath || false;\r\n const invertUV: boolean = options.invertUV || false;\r\n const defaultOffset: number = Math.floor(pathArray[0].length / 2);\r\n let offset: number = options.offset || defaultOffset;\r\n offset = offset > defaultOffset ? defaultOffset : Math.floor(offset); // offset max allowed : defaultOffset\r\n const sideOrientation: number = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\r\n const customUV = options.uvs;\r\n const customColors = options.colors;\r\n\r\n const positions: number[] = [];\r\n const indices: number[] = [];\r\n const normals: number[] = [];\r\n const uvs: number[] = [];\r\n\r\n const us: number[][] = []; // us[path_id] = [uDist1, uDist2, uDist3 ... ] distances between points on path path_id\r\n const vs: number[][] = []; // vs[i] = [vDist1, vDist2, vDist3, ... ] distances between points i of consecutive paths from pathArray\r\n const uTotalDistance: number[] = []; // uTotalDistance[p] : total distance of path p\r\n const vTotalDistance: number[] = []; // vTotalDistance[i] : total distance between points i of first and last path from pathArray\r\n let minlg: number; // minimal length among all paths from pathArray\r\n const lg: number[] = []; // array of path lengths : nb of vertex per path\r\n const idx: number[] = []; // array of path indexes : index of each path (first vertex) in the total vertex number\r\n let p: number; // path iterator\r\n let i: number; // point iterator\r\n let j: number; // point iterator\r\n\r\n // if single path in pathArray\r\n if (pathArray.length < 2) {\r\n const ar1: Vector3[] = [];\r\n const ar2: Vector3[] = [];\r\n for (i = 0; i < pathArray[0].length - offset; i++) {\r\n ar1.push(pathArray[0][i]);\r\n ar2.push(pathArray[0][i + offset]);\r\n }\r\n pathArray = [ar1, ar2];\r\n }\r\n\r\n // positions and horizontal distances (u)\r\n let idc: number = 0;\r\n const closePathCorr: number = closePath ? 1 : 0; // the final index will be +1 if closePath\r\n let path: Vector3[];\r\n let l: number;\r\n minlg = pathArray[0].length;\r\n let vectlg: number;\r\n let dist: number;\r\n for (p = 0; p < pathArray.length; p++) {\r\n uTotalDistance[p] = 0;\r\n us[p] = [0];\r\n path = pathArray[p];\r\n l = path.length;\r\n minlg = minlg < l ? minlg : l;\r\n\r\n j = 0;\r\n while (j < l) {\r\n positions.push(path[j].x, path[j].y, path[j].z);\r\n if (j > 0) {\r\n vectlg = path[j].subtract(path[j - 1]).length();\r\n dist = vectlg + uTotalDistance[p];\r\n us[p].push(dist);\r\n uTotalDistance[p] = dist;\r\n }\r\n j++;\r\n }\r\n\r\n if (closePath) {\r\n // an extra hidden vertex is added in the \"positions\" array\r\n j--;\r\n positions.push(path[0].x, path[0].y, path[0].z);\r\n vectlg = path[j].subtract(path[0]).length();\r\n dist = vectlg + uTotalDistance[p];\r\n us[p].push(dist);\r\n uTotalDistance[p] = dist;\r\n }\r\n\r\n lg[p] = l + closePathCorr;\r\n idx[p] = idc;\r\n idc += l + closePathCorr;\r\n }\r\n\r\n // vertical distances (v)\r\n let path1: Vector3[];\r\n let path2: Vector3[];\r\n let vertex1: Nullable = null;\r\n let vertex2: Nullable = null;\r\n for (i = 0; i < minlg + closePathCorr; i++) {\r\n vTotalDistance[i] = 0;\r\n vs[i] = [0];\r\n for (p = 0; p < pathArray.length - 1; p++) {\r\n path1 = pathArray[p];\r\n path2 = pathArray[p + 1];\r\n if (i === minlg) {\r\n // closePath\r\n vertex1 = path1[0];\r\n vertex2 = path2[0];\r\n } else {\r\n vertex1 = path1[i];\r\n vertex2 = path2[i];\r\n }\r\n vectlg = vertex2.subtract(vertex1).length();\r\n dist = vectlg + vTotalDistance[i];\r\n vs[i].push(dist);\r\n vTotalDistance[i] = dist;\r\n }\r\n\r\n if (closeArray && vertex2 && vertex1) {\r\n path1 = pathArray[p];\r\n path2 = pathArray[0];\r\n if (i === minlg) {\r\n // closePath\r\n vertex2 = path2[0];\r\n }\r\n vectlg = vertex2.subtract(vertex1).length();\r\n dist = vectlg + vTotalDistance[i];\r\n vTotalDistance[i] = dist;\r\n }\r\n }\r\n\r\n // uvs\r\n let u: number;\r\n let v: number;\r\n if (customUV) {\r\n for (p = 0; p < customUV.length; p++) {\r\n uvs.push(customUV[p].x, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - customUV[p].y : customUV[p].y);\r\n }\r\n } else {\r\n for (p = 0; p < pathArray.length; p++) {\r\n for (i = 0; i < minlg + closePathCorr; i++) {\r\n u = uTotalDistance[p] != 0.0 ? us[p][i] / uTotalDistance[p] : 0.0;\r\n v = vTotalDistance[i] != 0.0 ? vs[i][p] / vTotalDistance[i] : 0.0;\r\n if (invertUV) {\r\n uvs.push(v, u);\r\n } else {\r\n uvs.push(u, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - v : v);\r\n }\r\n }\r\n }\r\n }\r\n\r\n // indices\r\n p = 0; // path index\r\n let pi: number = 0; // positions array index\r\n let l1: number = lg[p] - 1; // path1 length\r\n let l2: number = lg[p + 1] - 1; // path2 length\r\n let min: number = l1 < l2 ? l1 : l2; // current path stop index\r\n let shft: number = idx[1] - idx[0]; // shift\r\n const path1nb: number = closeArray ? lg.length : lg.length - 1; // number of path1 to iterate\ton\r\n\r\n while (pi <= min && p < path1nb) {\r\n // stay under min and don't go over next to last path\r\n // draw two triangles between path1 (p1) and path2 (p2) : (p1.pi, p2.pi, p1.pi+1) and (p2.pi+1, p1.pi+1, p2.pi) clockwise\r\n\r\n indices.push(pi, pi + shft, pi + 1);\r\n indices.push(pi + shft + 1, pi + 1, pi + shft);\r\n pi += 1;\r\n if (pi === min) {\r\n // if end of one of two consecutive paths reached, go to next existing path\r\n p++;\r\n if (p === lg.length - 1) {\r\n // last path of pathArray reached <=> closeArray == true\r\n shft = idx[0] - idx[p];\r\n l1 = lg[p] - 1;\r\n l2 = lg[0] - 1;\r\n } else {\r\n shft = idx[p + 1] - idx[p];\r\n l1 = lg[p] - 1;\r\n l2 = lg[p + 1] - 1;\r\n }\r\n pi = idx[p];\r\n min = l1 < l2 ? l1 + pi : l2 + pi;\r\n }\r\n }\r\n\r\n // normals\r\n VertexData.ComputeNormals(positions, indices, normals);\r\n\r\n if (closePath) {\r\n // update both the first and last vertex normals to their average value\r\n let indexFirst: number = 0;\r\n let indexLast: number = 0;\r\n for (p = 0; p < pathArray.length; p++) {\r\n indexFirst = idx[p] * 3;\r\n if (p + 1 < pathArray.length) {\r\n indexLast = (idx[p + 1] - 1) * 3;\r\n } else {\r\n indexLast = normals.length - 3;\r\n }\r\n normals[indexFirst] = (normals[indexFirst] + normals[indexLast]) * 0.5;\r\n normals[indexFirst + 1] = (normals[indexFirst + 1] + normals[indexLast + 1]) * 0.5;\r\n normals[indexFirst + 2] = (normals[indexFirst + 2] + normals[indexLast + 2]) * 0.5;\r\n normals[indexLast] = normals[indexFirst];\r\n normals[indexLast + 1] = normals[indexFirst + 1];\r\n normals[indexLast + 2] = normals[indexFirst + 2];\r\n }\r\n }\r\n\r\n // sides\r\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\r\n\r\n // Colors\r\n let colors: Nullable = null;\r\n if (customColors) {\r\n colors = new Float32Array(customColors.length * 4);\r\n for (let c = 0; c < customColors.length; c++) {\r\n colors[c * 4] = customColors[c].r;\r\n colors[c * 4 + 1] = customColors[c].g;\r\n colors[c * 4 + 2] = customColors[c].b;\r\n colors[c * 4 + 3] = customColors[c].a;\r\n }\r\n }\r\n\r\n // Result\r\n const vertexData = new VertexData();\r\n const positions32 = new Float32Array(positions);\r\n const normals32 = new Float32Array(normals);\r\n const uvs32 = new Float32Array(uvs);\r\n\r\n vertexData.indices = indices;\r\n vertexData.positions = positions32;\r\n vertexData.normals = normals32;\r\n vertexData.uvs = uvs32;\r\n if (colors) {\r\n vertexData.set(colors, VertexBuffer.ColorKind);\r\n }\r\n\r\n if (closePath) {\r\n (vertexData)._idx = idx;\r\n }\r\n\r\n return vertexData;\r\n}\r\n\r\n/**\r\n * Creates a ribbon mesh. The ribbon is a parametric shape. It has no predefined shape. Its final shape will depend on the input parameters\r\n * * The parameter `pathArray` is a required array of paths, what are each an array of successive Vector3. The pathArray parameter depicts the ribbon geometry\r\n * * The parameter `closeArray` (boolean, default false) creates a seam between the first and the last paths of the path array\r\n * * The parameter `closePath` (boolean, default false) creates a seam between the first and the last points of each path of the path array\r\n * * The parameter `offset` (positive integer, default : rounded half size of the pathArray length), is taken in account only if the `pathArray` is containing a single path\r\n * * It's the offset to join the points from the same path. Ex : offset = 10 means the point 1 is joined to the point 11\r\n * * The optional parameter `instance` is an instance of an existing Ribbon object to be updated with the passed `pathArray` parameter : https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#ribbon\r\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\r\n * * The optional parameter `invertUV` (boolean, default false) swaps in the geometry the U and V coordinates to apply a texture\r\n * * The parameter `uvs` is an optional flat array of `Vector2` to update/set each ribbon vertex with its own custom UV values instead of the computed ones\r\n * * The parameters `colors` is an optional flat array of `Color4` to set/update each ribbon vertex with its own custom color values\r\n * * Note that if you use the parameters `uvs` or `colors`, the passed arrays must be populated with the right number of elements, it is to say the number of ribbon vertices. Remember that if you set `closePath` to `true`, there's one extra vertex per path in the geometry\r\n * * Moreover, you can use the parameter `color` with `instance` (to update the ribbon), only if you previously used it at creation time\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param options.pathArray\r\n * @param options.closeArray\r\n * @param options.closePath\r\n * @param options.offset\r\n * @param options.updatable\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param options.instance\r\n * @param options.invertUV\r\n * @param options.uvs\r\n * @param options.colors\r\n * @param scene defines the hosting scene\r\n * @returns the ribbon mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param/ribbon_extra\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param\r\n */\r\nexport function CreateRibbon(\r\n name: string,\r\n options: {\r\n pathArray: Vector3[][];\r\n closeArray?: boolean;\r\n closePath?: boolean;\r\n offset?: number;\r\n updatable?: boolean;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n instance?: Mesh;\r\n invertUV?: boolean;\r\n uvs?: Vector2[];\r\n colors?: Color4[];\r\n },\r\n scene: Nullable = null\r\n): Mesh {\r\n const pathArray = options.pathArray;\r\n const closeArray = options.closeArray;\r\n const closePath = options.closePath;\r\n const sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n const instance = options.instance;\r\n const updatable = options.updatable;\r\n\r\n if (instance) {\r\n // existing ribbon instance update\r\n // positionFunction : ribbon case\r\n // only pathArray and sideOrientation parameters are taken into account for positions update\r\n const minimum = TmpVectors.Vector3[0].setAll(Number.MAX_VALUE);\r\n const maximum = TmpVectors.Vector3[1].setAll(-Number.MAX_VALUE);\r\n const positionFunction = (positions: FloatArray) => {\r\n let minlg = pathArray[0].length;\r\n const mesh = instance;\r\n let i = 0;\r\n const ns = mesh._originalBuilderSideOrientation === Mesh.DOUBLESIDE ? 2 : 1;\r\n for (let si = 1; si <= ns; ++si) {\r\n for (let p = 0; p < pathArray.length; ++p) {\r\n const path = pathArray[p];\r\n const l = path.length;\r\n minlg = minlg < l ? minlg : l;\r\n for (let j = 0; j < minlg; ++j) {\r\n const pathPoint = path[j];\r\n positions[i] = pathPoint.x;\r\n positions[i + 1] = pathPoint.y;\r\n positions[i + 2] = pathPoint.z;\r\n minimum.minimizeInPlaceFromFloats(pathPoint.x, pathPoint.y, pathPoint.z);\r\n maximum.maximizeInPlaceFromFloats(pathPoint.x, pathPoint.y, pathPoint.z);\r\n i += 3;\r\n }\r\n if (mesh._creationDataStorage && mesh._creationDataStorage.closePath) {\r\n const pathPoint = path[0];\r\n positions[i] = pathPoint.x;\r\n positions[i + 1] = pathPoint.y;\r\n positions[i + 2] = pathPoint.z;\r\n i += 3;\r\n }\r\n }\r\n }\r\n };\r\n const positions = instance.getVerticesData(VertexBuffer.PositionKind);\r\n positionFunction(positions);\r\n if (instance.hasBoundingInfo) {\r\n instance.getBoundingInfo().reConstruct(minimum, maximum, instance._worldMatrix);\r\n } else {\r\n instance.buildBoundingInfo(minimum, maximum, instance._worldMatrix);\r\n }\r\n instance.updateVerticesData(VertexBuffer.PositionKind, positions, false, false);\r\n if (options.colors) {\r\n const colors = instance.getVerticesData(VertexBuffer.ColorKind);\r\n for (let c = 0, colorIndex = 0; c < options.colors.length; c++, colorIndex += 4) {\r\n const color = options.colors[c];\r\n colors[colorIndex] = color.r;\r\n colors[colorIndex + 1] = color.g;\r\n colors[colorIndex + 2] = color.b;\r\n colors[colorIndex + 3] = color.a;\r\n }\r\n instance.updateVerticesData(VertexBuffer.ColorKind, colors, false, false);\r\n }\r\n if (options.uvs) {\r\n const uvs = instance.getVerticesData(VertexBuffer.UVKind);\r\n for (let i = 0; i < options.uvs.length; i++) {\r\n uvs[i * 2] = options.uvs[i].x;\r\n uvs[i * 2 + 1] = CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - options.uvs[i].y : options.uvs[i].y;\r\n }\r\n instance.updateVerticesData(VertexBuffer.UVKind, uvs, false, false);\r\n }\r\n if (!instance.areNormalsFrozen || instance.isFacetDataEnabled) {\r\n const indices = instance.getIndices();\r\n const normals = instance.getVerticesData(VertexBuffer.NormalKind);\r\n const params = instance.isFacetDataEnabled ? instance.getFacetDataParameters() : null;\r\n VertexData.ComputeNormals(positions, indices, normals, params);\r\n\r\n if (instance._creationDataStorage && instance._creationDataStorage.closePath) {\r\n let indexFirst: number = 0;\r\n let indexLast: number = 0;\r\n for (let p = 0; p < pathArray.length; p++) {\r\n indexFirst = instance._creationDataStorage!.idx[p] * 3;\r\n if (p + 1 < pathArray.length) {\r\n indexLast = (instance._creationDataStorage!.idx[p + 1] - 1) * 3;\r\n } else {\r\n indexLast = normals.length - 3;\r\n }\r\n normals[indexFirst] = (normals[indexFirst] + normals[indexLast]) * 0.5;\r\n normals[indexFirst + 1] = (normals[indexFirst + 1] + normals[indexLast + 1]) * 0.5;\r\n normals[indexFirst + 2] = (normals[indexFirst + 2] + normals[indexLast + 2]) * 0.5;\r\n normals[indexLast] = normals[indexFirst];\r\n normals[indexLast + 1] = normals[indexFirst + 1];\r\n normals[indexLast + 2] = normals[indexFirst + 2];\r\n }\r\n }\r\n if (!instance.areNormalsFrozen) {\r\n instance.updateVerticesData(VertexBuffer.NormalKind, normals, false, false);\r\n }\r\n }\r\n\r\n return instance;\r\n } else {\r\n // new ribbon creation\r\n\r\n const ribbon = new Mesh(name, scene);\r\n ribbon._originalBuilderSideOrientation = sideOrientation;\r\n ribbon._creationDataStorage = new _CreationDataStorage();\r\n\r\n const vertexData = CreateRibbonVertexData(options);\r\n if (closePath) {\r\n ribbon._creationDataStorage.idx = (vertexData)._idx;\r\n }\r\n ribbon._creationDataStorage.closePath = closePath;\r\n ribbon._creationDataStorage.closeArray = closeArray;\r\n\r\n vertexData.applyToMesh(ribbon, updatable);\r\n\r\n return ribbon;\r\n }\r\n}\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated use CreateRibbon directly\r\n */\r\nexport const RibbonBuilder = {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreateRibbon,\r\n};\r\n\r\nVertexData.CreateRibbon = CreateRibbonVertexData;\r\n\r\nMesh.CreateRibbon = (\r\n name: string,\r\n pathArray: Vector3[][],\r\n closeArray: boolean = false,\r\n closePath: boolean,\r\n offset: number,\r\n scene?: Scene,\r\n updatable: boolean = false,\r\n sideOrientation?: number,\r\n instance?: Mesh\r\n) => {\r\n return CreateRibbon(\r\n name,\r\n {\r\n pathArray: pathArray,\r\n closeArray: closeArray,\r\n closePath: closePath,\r\n offset: offset,\r\n updatable: updatable,\r\n sideOrientation: sideOrientation,\r\n instance: instance,\r\n },\r\n scene\r\n );\r\n};\r\n","import type { Nullable } from \"../../types\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { Vector4 } from \"../../Maths/math.vector\";\r\nimport { Mesh } from \"../mesh\";\r\nimport { VertexData } from \"../mesh.vertexData\";\r\nimport { CompatibilityOptions } from \"../../Compat/compatibilityOptions\";\r\n\r\n/**\r\n * Creates the VertexData of the Disc or regular Polygon\r\n * @param options an object used to set the following optional parameters for the disc, required but can be empty\r\n * * radius the radius of the disc, optional default 0.5\r\n * * tessellation the number of polygon sides, optional, default 64\r\n * * arc a number from 0 to 1, to create an unclosed polygon based on the fraction of the circumference given by the arc value, optional, default 1\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.radius\r\n * @param options.tessellation\r\n * @param options.arc\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @returns the VertexData of the box\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport function CreateDiscVertexData(options: {\r\n radius?: number;\r\n tessellation?: number;\r\n arc?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n}): VertexData {\r\n const positions = new Array();\r\n const indices = new Array();\r\n const normals = new Array();\r\n const uvs = new Array();\r\n\r\n const radius = options.radius || 0.5;\r\n const tessellation = options.tessellation || 64;\r\n const arc: number = options.arc && (options.arc <= 0 || options.arc > 1) ? 1.0 : options.arc || 1.0;\r\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\r\n\r\n // positions and uvs\r\n positions.push(0, 0, 0); // disc center first\r\n uvs.push(0.5, 0.5);\r\n\r\n const theta = Math.PI * 2 * arc;\r\n const step = arc === 1 ? theta / tessellation : theta / (tessellation - 1);\r\n let a = 0;\r\n for (let t = 0; t < tessellation; t++) {\r\n const x = Math.cos(a);\r\n const y = Math.sin(a);\r\n const u = (x + 1) / 2;\r\n const v = (1 - y) / 2;\r\n positions.push(radius * x, radius * y, 0);\r\n uvs.push(u, CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - v : v);\r\n a += step;\r\n }\r\n if (arc === 1) {\r\n positions.push(positions[3], positions[4], positions[5]); // close the circle\r\n uvs.push(uvs[2], CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - uvs[3] : uvs[3]);\r\n }\r\n\r\n //indices\r\n const vertexNb = positions.length / 3;\r\n for (let i = 1; i < vertexNb - 1; i++) {\r\n indices.push(i + 1, 0, i);\r\n }\r\n\r\n // result\r\n VertexData.ComputeNormals(positions, indices, normals);\r\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\r\n\r\n const vertexData = new VertexData();\r\n\r\n vertexData.indices = indices;\r\n vertexData.positions = positions;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n\r\n return vertexData;\r\n}\r\n\r\n/**\r\n * Creates a plane polygonal mesh. By default, this is a disc\r\n * * The parameter `radius` sets the radius size (float) of the polygon (default 0.5)\r\n * * The parameter `tessellation` sets the number of polygon sides (positive integer, default 64). So a tessellation valued to 3 will build a triangle, to 4 a square, etc\r\n * * You can create an unclosed polygon with the parameter `arc` (positive float, default 1), valued between 0 and 1, what is the ratio of the circumference : 2 x PI x ratio\r\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param options.radius\r\n * @param options.tessellation\r\n * @param options.arc\r\n * @param options.updatable\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param scene defines the hosting scene\r\n * @returns the plane polygonal mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#disc-or-regular-polygon\r\n */\r\nexport function CreateDisc(\r\n name: string,\r\n options: { radius?: number; tessellation?: number; arc?: number; updatable?: boolean; sideOrientation?: number; frontUVs?: Vector4; backUVs?: Vector4 } = {},\r\n scene: Nullable = null\r\n): Mesh {\r\n const disc = new Mesh(name, scene);\r\n\r\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n disc._originalBuilderSideOrientation = options.sideOrientation;\r\n\r\n const vertexData = CreateDiscVertexData(options);\r\n\r\n vertexData.applyToMesh(disc, options.updatable);\r\n\r\n return disc;\r\n}\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated please use CreateDisc directly\r\n */\r\nexport const DiscBuilder = {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreateDisc,\r\n};\r\n\r\nVertexData.CreateDisc = CreateDiscVertexData;\r\n\r\nMesh.CreateDisc = (name: string, radius: number, tessellation: number, scene: Nullable = null, updatable?: boolean, sideOrientation?: number): Mesh => {\r\n const options = {\r\n radius,\r\n tessellation,\r\n sideOrientation,\r\n updatable,\r\n };\r\n\r\n return CreateDisc(name, options, scene);\r\n};\r\n","import type { Nullable } from \"../../types\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { Vector4 } from \"../../Maths/math.vector\";\r\nimport { Mesh } from \"../mesh\";\r\nimport { VertexData } from \"../mesh.vertexData\";\r\n\r\n/**\r\n * Creates the VertexData for a tiled plane\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set/tiled_plane\r\n * @param options an object used to set the following optional parameters for the tiled plane, required but can be empty\r\n * * pattern a limited pattern arrangement depending on the number\r\n * * size of the box\r\n * * width of the box, overwrites size\r\n * * height of the box, overwrites size\r\n * * tileSize sets the width, height and depth of the tile to the value of size, optional default 1\r\n * * tileWidth sets the width (x direction) of the tile, overwrites the width set by size, optional, default size\r\n * * tileHeight sets the height (y direction) of the tile, overwrites the height set by size, optional, default size\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * alignHorizontal places whole tiles aligned to the center, left or right of a row\r\n * * alignVertical places whole tiles aligned to the center, left or right of a column\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.pattern\r\n * @param options.tileSize\r\n * @param options.tileWidth\r\n * @param options.tileHeight\r\n * @param options.size\r\n * @param options.width\r\n * @param options.height\r\n * @param options.alignHorizontal\r\n * @param options.alignVertical\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @returns the VertexData of the tiled plane\r\n */\r\nexport function CreateTiledPlaneVertexData(options: {\r\n pattern?: number;\r\n tileSize?: number;\r\n tileWidth?: number;\r\n tileHeight?: number;\r\n size?: number;\r\n width?: number;\r\n height?: number;\r\n alignHorizontal?: number;\r\n alignVertical?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n}): VertexData {\r\n const flipTile = options.pattern || Mesh.NO_FLIP;\r\n const tileWidth = options.tileWidth || options.tileSize || 1;\r\n const tileHeight = options.tileHeight || options.tileSize || 1;\r\n const alignH = options.alignHorizontal || 0;\r\n const alignV = options.alignVertical || 0;\r\n\r\n const width = options.width || options.size || 1;\r\n const tilesX = Math.floor(width / tileWidth);\r\n let offsetX = width - tilesX * tileWidth;\r\n\r\n const height = options.height || options.size || 1;\r\n const tilesY = Math.floor(height / tileHeight);\r\n let offsetY = height - tilesY * tileHeight;\r\n\r\n const halfWidth = (tileWidth * tilesX) / 2;\r\n const halfHeight = (tileHeight * tilesY) / 2;\r\n\r\n let adjustX = 0;\r\n let adjustY = 0;\r\n let startX = 0;\r\n let startY = 0;\r\n let endX = 0;\r\n let endY = 0;\r\n\r\n //Part Tiles\r\n if (offsetX > 0 || offsetY > 0) {\r\n startX = -halfWidth;\r\n startY = -halfHeight;\r\n endX = halfWidth;\r\n endY = halfHeight;\r\n\r\n switch (alignH) {\r\n case Mesh.CENTER:\r\n offsetX /= 2;\r\n startX -= offsetX;\r\n endX += offsetX;\r\n break;\r\n case Mesh.LEFT:\r\n endX += offsetX;\r\n adjustX = -offsetX / 2;\r\n break;\r\n case Mesh.RIGHT:\r\n startX -= offsetX;\r\n adjustX = offsetX / 2;\r\n break;\r\n }\r\n\r\n switch (alignV) {\r\n case Mesh.CENTER:\r\n offsetY /= 2;\r\n startY -= offsetY;\r\n endY += offsetY;\r\n break;\r\n case Mesh.BOTTOM:\r\n endY += offsetY;\r\n adjustY = -offsetY / 2;\r\n break;\r\n case Mesh.TOP:\r\n startY -= offsetY;\r\n adjustY = offsetY / 2;\r\n break;\r\n }\r\n }\r\n\r\n const positions = [];\r\n const normals = [];\r\n const uvBase = [];\r\n uvBase[0] = [0, 0, 1, 0, 1, 1, 0, 1];\r\n uvBase[1] = [0, 0, 1, 0, 1, 1, 0, 1];\r\n if (flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.ROTATE_ROW) {\r\n uvBase[1] = [1, 1, 0, 1, 0, 0, 1, 0];\r\n }\r\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.FLIP_ROW) {\r\n uvBase[1] = [1, 0, 0, 0, 0, 1, 1, 1];\r\n }\r\n if (flipTile === Mesh.FLIP_N_ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\r\n uvBase[1] = [0, 1, 1, 1, 1, 0, 0, 0];\r\n }\r\n let uvs: Array = [];\r\n const colors = [];\r\n const indices = [];\r\n let index = 0;\r\n for (let y = 0; y < tilesY; y++) {\r\n for (let x = 0; x < tilesX; x++) {\r\n positions.push(-halfWidth + x * tileWidth + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\r\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\r\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\r\n positions.push(-halfWidth + x * tileWidth + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\r\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\r\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_TILE) {\r\n uvs = uvs.concat(uvBase[((x % 2) + (y % 2)) % 2]);\r\n } else if (flipTile === Mesh.FLIP_ROW || flipTile === Mesh.ROTATE_ROW || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\r\n uvs = uvs.concat(uvBase[y % 2]);\r\n } else {\r\n uvs = uvs.concat(uvBase[0]);\r\n }\r\n colors.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1);\r\n normals.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1);\r\n index += 4;\r\n }\r\n }\r\n\r\n //Part Tiles\r\n if (offsetX > 0 || offsetY > 0) {\r\n const partialBottomRow: boolean = offsetY > 0 && (alignV === Mesh.CENTER || alignV === Mesh.TOP);\r\n const partialTopRow: boolean = offsetY > 0 && (alignV === Mesh.CENTER || alignV === Mesh.BOTTOM);\r\n const partialLeftCol: boolean = offsetX > 0 && (alignH === Mesh.CENTER || alignH === Mesh.RIGHT);\r\n const partialRightCol: boolean = offsetX > 0 && (alignH === Mesh.CENTER || alignH === Mesh.LEFT);\r\n let uvPart: Array = [];\r\n let a, b, c, d: number;\r\n\r\n //corners\r\n if (partialBottomRow && partialLeftCol) {\r\n //bottom left corner\r\n positions.push(startX + adjustX, startY + adjustY, 0);\r\n positions.push(-halfWidth + adjustX, startY + adjustY, 0);\r\n positions.push(-halfWidth + adjustX, startY + offsetY + adjustY, 0);\r\n positions.push(startX + adjustX, startY + offsetY + adjustY, 0);\r\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\r\n index += 4;\r\n a = 1 - offsetX / tileWidth;\r\n b = 1 - offsetY / tileHeight;\r\n c = 1;\r\n d = 1;\r\n uvPart = [a, b, c, b, c, d, a, d];\r\n if (flipTile === Mesh.ROTATE_ROW) {\r\n uvPart = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\r\n }\r\n if (flipTile === Mesh.FLIP_ROW) {\r\n uvPart = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\r\n }\r\n if (flipTile === Mesh.FLIP_N_ROTATE_ROW) {\r\n uvPart = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\r\n }\r\n uvs = uvs.concat(uvPart);\r\n colors.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1);\r\n normals.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1);\r\n }\r\n\r\n if (partialBottomRow && partialRightCol) {\r\n //bottom right corner\r\n positions.push(halfWidth + adjustX, startY + adjustY, 0);\r\n positions.push(endX + adjustX, startY + adjustY, 0);\r\n positions.push(endX + adjustX, startY + offsetY + adjustY, 0);\r\n positions.push(halfWidth + adjustX, startY + offsetY + adjustY, 0);\r\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\r\n index += 4;\r\n a = 0;\r\n b = 1 - offsetY / tileHeight;\r\n c = offsetX / tileWidth;\r\n d = 1;\r\n uvPart = [a, b, c, b, c, d, a, d];\r\n if (flipTile === Mesh.ROTATE_ROW || (flipTile === Mesh.ROTATE_TILE && tilesX % 2 === 0)) {\r\n uvPart = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\r\n }\r\n if (flipTile === Mesh.FLIP_ROW || (flipTile === Mesh.FLIP_TILE && tilesX % 2 === 0)) {\r\n uvPart = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\r\n }\r\n if (flipTile === Mesh.FLIP_N_ROTATE_ROW || (flipTile === Mesh.FLIP_N_ROTATE_TILE && tilesX % 2 === 0)) {\r\n uvPart = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\r\n }\r\n uvs = uvs.concat(uvPart);\r\n colors.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1);\r\n normals.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1);\r\n }\r\n\r\n if (partialTopRow && partialLeftCol) {\r\n //top left corner\r\n positions.push(startX + adjustX, halfHeight + adjustY, 0);\r\n positions.push(-halfWidth + adjustX, halfHeight + adjustY, 0);\r\n positions.push(-halfWidth + adjustX, endY + adjustY, 0);\r\n positions.push(startX + adjustX, endY + adjustY, 0);\r\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\r\n index += 4;\r\n a = 1 - offsetX / tileWidth;\r\n b = 0;\r\n c = 1;\r\n d = offsetY / tileHeight;\r\n uvPart = [a, b, c, b, c, d, a, d];\r\n if ((flipTile === Mesh.ROTATE_ROW && tilesY % 2 === 1) || (flipTile === Mesh.ROTATE_TILE && tilesY % 1 === 0)) {\r\n uvPart = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\r\n }\r\n if ((flipTile === Mesh.FLIP_ROW && tilesY % 2 === 1) || (flipTile === Mesh.FLIP_TILE && tilesY % 2 === 0)) {\r\n uvPart = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\r\n }\r\n if ((flipTile === Mesh.FLIP_N_ROTATE_ROW && tilesY % 2 === 1) || (flipTile === Mesh.FLIP_N_ROTATE_TILE && tilesY % 2 === 0)) {\r\n uvPart = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\r\n }\r\n uvs = uvs.concat(uvPart);\r\n colors.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1);\r\n normals.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1);\r\n }\r\n\r\n if (partialTopRow && partialRightCol) {\r\n //top right corner\r\n positions.push(halfWidth + adjustX, halfHeight + adjustY, 0);\r\n positions.push(endX + adjustX, halfHeight + adjustY, 0);\r\n positions.push(endX + adjustX, endY + adjustY, 0);\r\n positions.push(halfWidth + adjustX, endY + adjustY, 0);\r\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\r\n index += 4;\r\n a = 0;\r\n b = 0;\r\n c = offsetX / tileWidth;\r\n d = offsetY / tileHeight;\r\n uvPart = [a, b, c, b, c, d, a, d];\r\n if ((flipTile === Mesh.ROTATE_ROW && tilesY % 2 === 1) || (flipTile === Mesh.ROTATE_TILE && (tilesY + tilesX) % 2 === 1)) {\r\n uvPart = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\r\n }\r\n if ((flipTile === Mesh.FLIP_ROW && tilesY % 2 === 1) || (flipTile === Mesh.FLIP_TILE && (tilesY + tilesX) % 2 === 1)) {\r\n uvPart = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\r\n }\r\n if ((flipTile === Mesh.FLIP_N_ROTATE_ROW && tilesY % 2 === 1) || (flipTile === Mesh.FLIP_N_ROTATE_TILE && (tilesY + tilesX) % 2 === 1)) {\r\n uvPart = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\r\n }\r\n uvs = uvs.concat(uvPart);\r\n colors.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1);\r\n normals.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1);\r\n }\r\n\r\n //part rows\r\n if (partialBottomRow) {\r\n const uvBaseBR = [];\r\n a = 0;\r\n b = 1 - offsetY / tileHeight;\r\n c = 1;\r\n d = 1;\r\n uvBaseBR[0] = [a, b, c, b, c, d, a, d];\r\n uvBaseBR[1] = [a, b, c, b, c, d, a, d];\r\n if (flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.ROTATE_ROW) {\r\n uvBaseBR[1] = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\r\n }\r\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.FLIP_ROW) {\r\n uvBaseBR[1] = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\r\n }\r\n if (flipTile === Mesh.FLIP_N_ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\r\n uvBaseBR[1] = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\r\n }\r\n for (let x = 0; x < tilesX; x++) {\r\n positions.push(-halfWidth + x * tileWidth + adjustX, startY + adjustY, 0);\r\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, startY + adjustY, 0);\r\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, startY + offsetY + adjustY, 0);\r\n positions.push(-halfWidth + x * tileWidth + adjustX, startY + offsetY + adjustY, 0);\r\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\r\n index += 4;\r\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_TILE) {\r\n uvs = uvs.concat(uvBaseBR[(x + 1) % 2]);\r\n } else if (flipTile === Mesh.FLIP_ROW || flipTile === Mesh.ROTATE_ROW || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\r\n uvs = uvs.concat(uvBaseBR[1]);\r\n } else {\r\n uvs = uvs.concat(uvBaseBR[0]);\r\n }\r\n colors.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1);\r\n normals.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1);\r\n }\r\n }\r\n\r\n if (partialTopRow) {\r\n const uvBaseTR = [];\r\n a = 0;\r\n b = 0;\r\n c = 1;\r\n d = offsetY / tileHeight;\r\n uvBaseTR[0] = [a, b, c, b, c, d, a, d];\r\n uvBaseTR[1] = [a, b, c, b, c, d, a, d];\r\n if (flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.ROTATE_ROW) {\r\n uvBaseTR[1] = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\r\n }\r\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.FLIP_ROW) {\r\n uvBaseTR[1] = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\r\n }\r\n if (flipTile === Mesh.FLIP_N_ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\r\n uvBaseTR[1] = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\r\n }\r\n for (let x = 0; x < tilesX; x++) {\r\n positions.push(-halfWidth + x * tileWidth + adjustX, endY - offsetY + adjustY, 0);\r\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, endY - offsetY + adjustY, 0);\r\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, endY + adjustY, 0);\r\n positions.push(-halfWidth + x * tileWidth + adjustX, endY + adjustY, 0);\r\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\r\n index += 4;\r\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_TILE) {\r\n uvs = uvs.concat(uvBaseTR[(x + tilesY) % 2]);\r\n } else if (flipTile === Mesh.FLIP_ROW || flipTile === Mesh.ROTATE_ROW || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\r\n uvs = uvs.concat(uvBaseTR[tilesY % 2]);\r\n } else {\r\n uvs = uvs.concat(uvBaseTR[0]);\r\n }\r\n colors.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1);\r\n normals.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1);\r\n }\r\n }\r\n\r\n if (partialLeftCol) {\r\n const uvBaseLC = [];\r\n a = 1 - offsetX / tileWidth;\r\n b = 0;\r\n c = 1;\r\n d = 1;\r\n uvBaseLC[0] = [a, b, c, b, c, d, a, d];\r\n uvBaseLC[1] = [a, b, c, b, c, d, a, d];\r\n if (flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.ROTATE_ROW) {\r\n uvBaseLC[1] = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\r\n }\r\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.FLIP_ROW) {\r\n uvBaseLC[1] = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\r\n }\r\n if (flipTile === Mesh.FLIP_N_ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\r\n uvBaseLC[1] = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\r\n }\r\n for (let y = 0; y < tilesY; y++) {\r\n positions.push(startX + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\r\n positions.push(startX + offsetX + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\r\n positions.push(startX + offsetX + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\r\n positions.push(startX + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\r\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\r\n index += 4;\r\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_TILE) {\r\n uvs = uvs.concat(uvBaseLC[(y + 1) % 2]);\r\n } else if (flipTile === Mesh.FLIP_ROW || flipTile === Mesh.ROTATE_ROW || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\r\n uvs = uvs.concat(uvBaseLC[y % 2]);\r\n } else {\r\n uvs = uvs.concat(uvBaseLC[0]);\r\n }\r\n colors.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1);\r\n normals.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1);\r\n }\r\n }\r\n\r\n if (partialRightCol) {\r\n const uvBaseRC = [];\r\n a = 0;\r\n b = 0;\r\n c = offsetX / tileHeight;\r\n d = 1;\r\n uvBaseRC[0] = [a, b, c, b, c, d, a, d];\r\n uvBaseRC[1] = [a, b, c, b, c, d, a, d];\r\n if (flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.ROTATE_ROW) {\r\n uvBaseRC[1] = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\r\n }\r\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.FLIP_ROW) {\r\n uvBaseRC[1] = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\r\n }\r\n if (flipTile === Mesh.FLIP_N_ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\r\n uvBaseRC[1] = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\r\n }\r\n for (let y = 0; y < tilesY; y++) {\r\n positions.push(endX - offsetX + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\r\n positions.push(endX + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\r\n positions.push(endX + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\r\n positions.push(endX - offsetX + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\r\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\r\n index += 4;\r\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_TILE) {\r\n uvs = uvs.concat(uvBaseRC[(y + tilesX) % 2]);\r\n } else if (flipTile === Mesh.FLIP_ROW || flipTile === Mesh.ROTATE_ROW || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\r\n uvs = uvs.concat(uvBaseRC[y % 2]);\r\n } else {\r\n uvs = uvs.concat(uvBaseRC[0]);\r\n }\r\n colors.push(1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1);\r\n normals.push(0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1);\r\n }\r\n }\r\n }\r\n\r\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\r\n\r\n // sides\r\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\r\n\r\n // Result\r\n const vertexData = new VertexData();\r\n\r\n vertexData.indices = indices;\r\n vertexData.positions = positions;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n\r\n const totalColors = sideOrientation === VertexData.DOUBLESIDE ? colors.concat(colors) : colors;\r\n vertexData.colors = totalColors;\r\n\r\n return vertexData;\r\n}\r\n\r\n/**\r\n * Creates a tiled plane mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set/tiled_plane\r\n * @param name defines the name of the mesh\r\n * @param options an object used to set the following optional parameters for the tiled plane, required but can be empty\r\n * * pattern a limited pattern arrangement depending on the number\r\n * * size of the box\r\n * * width of the box, overwrites size\r\n * * height of the box, overwrites size\r\n * * tileSize sets the width, height and depth of the tile to the value of size, optional default 1\r\n * * tileWidth sets the width (x direction) of the tile, overwrites the width set by size, optional, default size\r\n * * tileHeight sets the height (y direction) of the tile, overwrites the height set by size, optional, default size\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * alignHorizontal places whole tiles aligned to the center, left or right of a row\r\n * * alignVertical places whole tiles aligned to the center, left or right of a column\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.pattern\r\n * @param options.tileSize\r\n * @param options.tileWidth\r\n * @param options.tileHeight\r\n * @param options.size\r\n * @param options.width\r\n * @param options.height\r\n * @param options.alignHorizontal\r\n * @param options.alignVertical\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param options.updatable\r\n * @param scene defines the hosting scene\r\n * @returns the box mesh\r\n */\r\nexport function CreateTiledPlane(\r\n name: string,\r\n options: {\r\n pattern?: number;\r\n tileSize?: number;\r\n tileWidth?: number;\r\n tileHeight?: number;\r\n size?: number;\r\n width?: number;\r\n height?: number;\r\n alignHorizontal?: number;\r\n alignVertical?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n updatable?: boolean;\r\n },\r\n scene: Nullable = null\r\n): Mesh {\r\n const plane = new Mesh(name, scene);\r\n\r\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n plane._originalBuilderSideOrientation = options.sideOrientation;\r\n\r\n const vertexData = CreateTiledPlaneVertexData(options);\r\n\r\n vertexData.applyToMesh(plane, options.updatable);\r\n\r\n return plane;\r\n}\r\n\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated use CreateTiledPlane instead\r\n */\r\nexport const TiledPlaneBuilder = {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreateTiledPlane,\r\n};\r\n\r\nVertexData.CreateTiledPlane = CreateTiledPlaneVertexData;\r\n","import type { Nullable } from \"../../types\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { Matrix, Vector3, Vector4 } from \"../../Maths/math.vector\";\r\nimport { Color4 } from \"../../Maths/math.color\";\r\nimport { Mesh } from \"../mesh\";\r\nimport { VertexData } from \"../mesh.vertexData\";\r\nimport { CreateTiledPlaneVertexData } from \"./tiledPlaneBuilder\";\r\nimport { CompatibilityOptions } from \"../../Compat/compatibilityOptions\";\r\n\r\n/**\r\n * Creates the VertexData for a tiled box\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set/tiled_box\r\n * @param options an object used to set the following optional parameters for the tiled box, required but can be empty\r\n * * pattern sets the rotation or reflection pattern for the tiles,\r\n * * size of the box\r\n * * width of the box, overwrites size\r\n * * height of the box, overwrites size\r\n * * depth of the box, overwrites size\r\n * * tileSize sets the size of a tile\r\n * * tileWidth sets the tile width and overwrites tileSize\r\n * * tileHeight sets the tile width and overwrites tileSize\r\n * * faceUV an array of 6 Vector4 elements used to set different images to each box side\r\n * * faceColors an array of 6 Color3 elements used to set different colors to each box side\r\n * * alignHorizontal places whole tiles aligned to the center, left or right of a row\r\n * * alignVertical places whole tiles aligned to the center, left or right of a column\r\n * @param options.pattern\r\n * @param options.size\r\n * @param options.width\r\n * @param options.height\r\n * @param options.depth\r\n * @param options.tileSize\r\n * @param options.tileWidth\r\n * @param options.tileHeight\r\n * @param options.faceUV\r\n * @param options.faceColors\r\n * @param options.alignHorizontal\r\n * @param options.alignVertical\r\n * @param options.sideOrientation\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * @returns the VertexData of the TiledBox\r\n */\r\nexport function CreateTiledBoxVertexData(options: {\r\n pattern?: number;\r\n size?: number;\r\n width?: number;\r\n height?: number;\r\n depth?: number;\r\n tileSize?: number;\r\n tileWidth?: number;\r\n tileHeight?: number;\r\n faceUV?: Vector4[];\r\n faceColors?: Color4[];\r\n alignHorizontal?: number;\r\n alignVertical?: number;\r\n sideOrientation?: number;\r\n}): VertexData {\r\n const nbFaces = 6;\r\n\r\n const faceUV: Vector4[] = options.faceUV || new Array(6);\r\n const faceColors = options.faceColors;\r\n\r\n const flipTile = options.pattern || Mesh.NO_FLIP;\r\n\r\n const width = options.width || options.size || 1;\r\n const height = options.height || options.size || 1;\r\n const depth = options.depth || options.size || 1;\r\n const tileWidth = options.tileWidth || options.tileSize || 1;\r\n const tileHeight = options.tileHeight || options.tileSize || 1;\r\n const alignH = options.alignHorizontal || 0;\r\n const alignV = options.alignVertical || 0;\r\n\r\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\r\n\r\n // default face colors and UV if undefined\r\n for (let f = 0; f < nbFaces; f++) {\r\n if (faceUV[f] === undefined) {\r\n faceUV[f] = new Vector4(0, 0, 1, 1);\r\n }\r\n if (faceColors && faceColors[f] === undefined) {\r\n faceColors[f] = new Color4(1, 1, 1, 1);\r\n }\r\n }\r\n\r\n const halfWidth = width / 2;\r\n const halfHeight = height / 2;\r\n const halfDepth = depth / 2;\r\n\r\n const faceVertexData: Array = [];\r\n\r\n for (let f = 0; f < 2; f++) {\r\n //front and back\r\n faceVertexData[f] = CreateTiledPlaneVertexData({\r\n pattern: flipTile,\r\n tileWidth: tileWidth,\r\n tileHeight: tileHeight,\r\n width: width,\r\n height: height,\r\n alignVertical: alignV,\r\n alignHorizontal: alignH,\r\n sideOrientation: sideOrientation,\r\n });\r\n }\r\n\r\n for (let f = 2; f < 4; f++) {\r\n //sides\r\n faceVertexData[f] = CreateTiledPlaneVertexData({\r\n pattern: flipTile,\r\n tileWidth: tileWidth,\r\n tileHeight: tileHeight,\r\n width: depth,\r\n height: height,\r\n alignVertical: alignV,\r\n alignHorizontal: alignH,\r\n sideOrientation: sideOrientation,\r\n });\r\n }\r\n\r\n let baseAlignV = alignV;\r\n if (alignV === Mesh.BOTTOM) {\r\n baseAlignV = Mesh.TOP;\r\n } else if (alignV === Mesh.TOP) {\r\n baseAlignV = Mesh.BOTTOM;\r\n }\r\n\r\n for (let f = 4; f < 6; f++) {\r\n //top and bottom\r\n faceVertexData[f] = CreateTiledPlaneVertexData({\r\n pattern: flipTile,\r\n tileWidth: tileWidth,\r\n tileHeight: tileHeight,\r\n width: width,\r\n height: depth,\r\n alignVertical: baseAlignV,\r\n alignHorizontal: alignH,\r\n sideOrientation: sideOrientation,\r\n });\r\n }\r\n\r\n let positions: Array = [];\r\n let normals: Array = [];\r\n let uvs: Array = [];\r\n let indices: Array = [];\r\n const colors: Array = [];\r\n const facePositions: Array> = [];\r\n const faceNormals: Array> = [];\r\n\r\n const newFaceUV: Array> = [];\r\n let lu: number = 0;\r\n\r\n let li: number = 0;\r\n\r\n for (let f = 0; f < nbFaces; f++) {\r\n const len = faceVertexData[f].positions!.length;\r\n facePositions[f] = [];\r\n faceNormals[f] = [];\r\n for (let p = 0; p < len / 3; p++) {\r\n facePositions[f].push(new Vector3(faceVertexData[f].positions![3 * p], faceVertexData[f].positions![3 * p + 1], faceVertexData[f].positions![3 * p + 2]));\r\n faceNormals[f].push(new Vector3(faceVertexData[f].normals![3 * p], faceVertexData[f].normals![3 * p + 1], faceVertexData[f].normals![3 * p + 2]));\r\n }\r\n // uvs\r\n lu = faceVertexData[f].uvs!.length;\r\n newFaceUV[f] = [];\r\n for (let i = 0; i < lu; i += 2) {\r\n newFaceUV[f][i] = faceUV[f].x + (faceUV[f].z - faceUV[f].x) * faceVertexData[f].uvs![i];\r\n newFaceUV[f][i + 1] = faceUV[f].y + (faceUV[f].w - faceUV[f].y) * faceVertexData[f].uvs![i + 1];\r\n\r\n if (CompatibilityOptions.UseOpenGLOrientationForUV) {\r\n newFaceUV[f][i + 1] = 1.0 - newFaceUV[f][i + 1];\r\n }\r\n }\r\n uvs = uvs.concat(newFaceUV[f]);\r\n\r\n indices = indices.concat(>faceVertexData[f].indices!.map((x: number) => x + li));\r\n li += facePositions[f].length;\r\n if (faceColors) {\r\n for (let c = 0; c < 4; c++) {\r\n colors.push(faceColors[f].r, faceColors[f].g, faceColors[f].b, faceColors[f].a);\r\n }\r\n }\r\n }\r\n\r\n const vec0 = new Vector3(0, 0, halfDepth);\r\n const mtrx0 = Matrix.RotationY(Math.PI);\r\n positions = facePositions[0]\r\n .map((entry) => Vector3.TransformNormal(entry, mtrx0).add(vec0))\r\n .map((entry) => [entry.x, entry.y, entry.z])\r\n .reduce((accumulator: Array, currentValue) => accumulator.concat(currentValue), []);\r\n normals = faceNormals[0]\r\n .map((entry) => Vector3.TransformNormal(entry, mtrx0))\r\n .map((entry) => [entry.x, entry.y, entry.z])\r\n .reduce((accumulator: Array, currentValue) => accumulator.concat(currentValue), []);\r\n positions = positions.concat(\r\n facePositions[1]\r\n .map((entry) => entry.subtract(vec0))\r\n .map((entry) => [entry.x, entry.y, entry.z])\r\n .reduce((accumulator: Array, currentValue) => accumulator.concat(currentValue), [])\r\n );\r\n normals = normals.concat(faceNormals[1].map((entry) => [entry.x, entry.y, entry.z]).reduce((accumulator: Array, currentValue) => accumulator.concat(currentValue), []));\r\n\r\n const vec2 = new Vector3(halfWidth, 0, 0);\r\n const mtrx2 = Matrix.RotationY(-Math.PI / 2);\r\n positions = positions.concat(\r\n facePositions[2]\r\n .map((entry) => Vector3.TransformNormal(entry, mtrx2).add(vec2))\r\n .map((entry) => [entry.x, entry.y, entry.z])\r\n .reduce((accumulator: Array, currentValue) => accumulator.concat(currentValue), [])\r\n );\r\n normals = normals.concat(\r\n faceNormals[2]\r\n .map((entry) => Vector3.TransformNormal(entry, mtrx2))\r\n .map((entry) => [entry.x, entry.y, entry.z])\r\n .reduce((accumulator: Array, currentValue) => accumulator.concat(currentValue), [])\r\n );\r\n const mtrx3 = Matrix.RotationY(Math.PI / 2);\r\n positions = positions.concat(\r\n facePositions[3]\r\n .map((entry) => Vector3.TransformNormal(entry, mtrx3).subtract(vec2))\r\n .map((entry) => [entry.x, entry.y, entry.z])\r\n .reduce((accumulator: Array, currentValue) => accumulator.concat(currentValue), [])\r\n );\r\n normals = normals.concat(\r\n faceNormals[3]\r\n .map((entry) => Vector3.TransformNormal(entry, mtrx3))\r\n .map((entry) => [entry.x, entry.y, entry.z])\r\n .reduce((accumulator: Array, currentValue) => accumulator.concat(currentValue), [])\r\n );\r\n\r\n const vec4 = new Vector3(0, halfHeight, 0);\r\n const mtrx4 = Matrix.RotationX(Math.PI / 2);\r\n positions = positions.concat(\r\n facePositions[4]\r\n .map((entry) => Vector3.TransformNormal(entry, mtrx4).add(vec4))\r\n .map((entry) => [entry.x, entry.y, entry.z])\r\n .reduce((accumulator: Array, currentValue) => accumulator.concat(currentValue), [])\r\n );\r\n normals = normals.concat(\r\n faceNormals[4]\r\n .map((entry) => Vector3.TransformNormal(entry, mtrx4))\r\n .map((entry) => [entry.x, entry.y, entry.z])\r\n .reduce((accumulator: Array, currentValue) => accumulator.concat(currentValue), [])\r\n );\r\n const mtrx5 = Matrix.RotationX(-Math.PI / 2);\r\n positions = positions.concat(\r\n facePositions[5]\r\n .map((entry) => Vector3.TransformNormal(entry, mtrx5).subtract(vec4))\r\n .map((entry) => [entry.x, entry.y, entry.z])\r\n .reduce((accumulator: Array, currentValue) => accumulator.concat(currentValue), [])\r\n );\r\n normals = normals.concat(\r\n faceNormals[5]\r\n .map((entry) => Vector3.TransformNormal(entry, mtrx5))\r\n .map((entry) => [entry.x, entry.y, entry.z])\r\n .reduce((accumulator: Array, currentValue) => accumulator.concat(currentValue), [])\r\n );\r\n\r\n // sides\r\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs);\r\n\r\n // Result\r\n const vertexData = new VertexData();\r\n\r\n vertexData.indices = indices;\r\n vertexData.positions = positions;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n\r\n if (faceColors) {\r\n const totalColors = sideOrientation === VertexData.DOUBLESIDE ? colors.concat(colors) : colors;\r\n vertexData.colors = totalColors;\r\n }\r\n\r\n return vertexData;\r\n}\r\n\r\n/**\r\n * Creates a tiled box mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set/tiled_box\r\n * @param name defines the name of the mesh\r\n * @param options an object used to set the following optional parameters for the tiled box, required but can be empty\r\n * * pattern sets the rotation or reflection pattern for the tiles,\r\n * * size of the box\r\n * * width of the box, overwrites size\r\n * * height of the box, overwrites size\r\n * * depth of the box, overwrites size\r\n * * tileSize sets the size of a tile\r\n * * tileWidth sets the tile width and overwrites tileSize\r\n * * tileHeight sets the tile width and overwrites tileSize\r\n * * faceUV an array of 6 Vector4 elements used to set different images to each box side\r\n * * faceColors an array of 6 Color3 elements used to set different colors to each box side\r\n * * alignHorizontal places whole tiles aligned to the center, left or right of a row\r\n * * alignVertical places whole tiles aligned to the center, left or right of a column\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * @param options.pattern\r\n * @param options.width\r\n * @param options.height\r\n * @param options.depth\r\n * @param options.tileSize\r\n * @param options.tileWidth\r\n * @param options.tileHeight\r\n * @param options.alignHorizontal\r\n * @param options.alignVertical\r\n * @param options.faceUV\r\n * @param options.faceColors\r\n * @param options.sideOrientation\r\n * @param options.updatable\r\n * @param scene defines the hosting scene\r\n * @returns the box mesh\r\n */\r\nexport function CreateTiledBox(\r\n name: string,\r\n options: {\r\n pattern?: number;\r\n width?: number;\r\n height?: number;\r\n depth?: number;\r\n tileSize?: number;\r\n tileWidth?: number;\r\n tileHeight?: number;\r\n alignHorizontal?: number;\r\n alignVertical?: number;\r\n faceUV?: Vector4[];\r\n faceColors?: Color4[];\r\n sideOrientation?: number;\r\n updatable?: boolean;\r\n },\r\n scene: Nullable = null\r\n): Mesh {\r\n const box = new Mesh(name, scene);\r\n\r\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n box._originalBuilderSideOrientation = options.sideOrientation;\r\n\r\n const vertexData = CreateTiledBoxVertexData(options);\r\n\r\n vertexData.applyToMesh(box, options.updatable);\r\n\r\n return box;\r\n}\r\n\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated use CreateTiledBox instead\r\n */\r\nexport const TiledBoxBuilder = {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreateTiledBox,\r\n};\r\n\r\nVertexData.CreateTiledBox = CreateTiledBoxVertexData;\r\n","import type { Vector4 } from \"../../Maths/math.vector\";\r\nimport { Vector3 } from \"../../Maths/math.vector\";\r\nimport { Mesh } from \"../mesh\";\r\nimport { VertexData } from \"../mesh.vertexData\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { CompatibilityOptions } from \"../../Compat/compatibilityOptions\";\r\n\r\n// based on http://code.google.com/p/away3d/source/browse/trunk/fp10/Away3D/src/away3d/primitives/TorusKnot.as?spec=svn2473&r=2473\r\n/**\r\n * Creates the VertexData for a TorusKnot\r\n * @param options an object used to set the following optional parameters for the TorusKnot, required but can be empty\r\n * * radius the radius of the torus knot, optional, default 2\r\n * * tube the thickness of the tube, optional, default 0.5\r\n * * radialSegments the number of sides on each tube segments, optional, default 32\r\n * * tubularSegments the number of tubes to decompose the knot into, optional, default 32\r\n * * p the number of windings around the z axis, optional, default 2\r\n * * q the number of windings around the x axis, optional, default 3\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.radius\r\n * @param options.tube\r\n * @param options.radialSegments\r\n * @param options.tubularSegments\r\n * @param options.p\r\n * @param options.q\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @returns the VertexData of the Torus Knot\r\n */\r\nexport function CreateTorusKnotVertexData(options: {\r\n radius?: number;\r\n tube?: number;\r\n radialSegments?: number;\r\n tubularSegments?: number;\r\n p?: number;\r\n q?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n}): VertexData {\r\n const indices = new Array();\r\n const positions = new Array();\r\n const normals = new Array();\r\n const uvs = new Array();\r\n\r\n const radius = options.radius || 2;\r\n const tube = options.tube || 0.5;\r\n const radialSegments = options.radialSegments || 32;\r\n const tubularSegments = options.tubularSegments || 32;\r\n const p = options.p || 2;\r\n const q = options.q || 3;\r\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\r\n\r\n // Helper\r\n const getPos = (angle: number) => {\r\n const cu = Math.cos(angle);\r\n const su = Math.sin(angle);\r\n const quOverP = (q / p) * angle;\r\n const cs = Math.cos(quOverP);\r\n\r\n const tx = radius * (2 + cs) * 0.5 * cu;\r\n const ty = radius * (2 + cs) * su * 0.5;\r\n const tz = radius * Math.sin(quOverP) * 0.5;\r\n\r\n return new Vector3(tx, ty, tz);\r\n };\r\n\r\n // Vertices\r\n let i: number;\r\n let j: number;\r\n for (i = 0; i <= radialSegments; i++) {\r\n const modI = i % radialSegments;\r\n const u = (modI / radialSegments) * 2 * p * Math.PI;\r\n const p1 = getPos(u);\r\n const p2 = getPos(u + 0.01);\r\n const tang = p2.subtract(p1);\r\n let n = p2.add(p1);\r\n\r\n const bitan = Vector3.Cross(tang, n);\r\n n = Vector3.Cross(bitan, tang);\r\n\r\n bitan.normalize();\r\n n.normalize();\r\n\r\n for (j = 0; j < tubularSegments; j++) {\r\n const modJ = j % tubularSegments;\r\n const v = (modJ / tubularSegments) * 2 * Math.PI;\r\n const cx = -tube * Math.cos(v);\r\n const cy = tube * Math.sin(v);\r\n\r\n positions.push(p1.x + cx * n.x + cy * bitan.x);\r\n positions.push(p1.y + cx * n.y + cy * bitan.y);\r\n positions.push(p1.z + cx * n.z + cy * bitan.z);\r\n\r\n uvs.push(i / radialSegments);\r\n uvs.push(CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - j / tubularSegments : j / tubularSegments);\r\n }\r\n }\r\n\r\n for (i = 0; i < radialSegments; i++) {\r\n for (j = 0; j < tubularSegments; j++) {\r\n const jNext = (j + 1) % tubularSegments;\r\n const a = i * tubularSegments + j;\r\n const b = (i + 1) * tubularSegments + j;\r\n const c = (i + 1) * tubularSegments + jNext;\r\n const d = i * tubularSegments + jNext;\r\n\r\n indices.push(d);\r\n indices.push(b);\r\n indices.push(a);\r\n indices.push(d);\r\n indices.push(c);\r\n indices.push(b);\r\n }\r\n }\r\n\r\n // Normals\r\n VertexData.ComputeNormals(positions, indices, normals);\r\n\r\n // Sides\r\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\r\n\r\n // Result\r\n const vertexData = new VertexData();\r\n\r\n vertexData.indices = indices;\r\n vertexData.positions = positions;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n\r\n return vertexData;\r\n}\r\n\r\n/**\r\n * Creates a torus knot mesh\r\n * * The parameter `radius` sets the global radius size (float) of the torus knot (default 2)\r\n * * The parameter `radialSegments` sets the number of sides on each tube segments (positive integer, default 32)\r\n * * The parameter `tubularSegments` sets the number of tubes to decompose the knot into (positive integer, default 32)\r\n * * The parameters `p` and `q` are the number of windings on each axis (positive integers, default 2 and 3)\r\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param options.radius\r\n * @param options.tube\r\n * @param options.radialSegments\r\n * @param options.tubularSegments\r\n * @param options.p\r\n * @param options.q\r\n * @param options.updatable\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param scene defines the hosting scene\r\n * @returns the torus knot mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#torus-knot\r\n */\r\nexport function CreateTorusKnot(\r\n name: string,\r\n options: {\r\n radius?: number;\r\n tube?: number;\r\n radialSegments?: number;\r\n tubularSegments?: number;\r\n p?: number;\r\n q?: number;\r\n updatable?: boolean;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n } = {},\r\n scene?: Scene\r\n): Mesh {\r\n const torusKnot = new Mesh(name, scene);\r\n\r\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n torusKnot._originalBuilderSideOrientation = options.sideOrientation;\r\n\r\n const vertexData = CreateTorusKnotVertexData(options);\r\n\r\n vertexData.applyToMesh(torusKnot, options.updatable);\r\n\r\n return torusKnot;\r\n}\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated use CreateTorusKnot instead\r\n */\r\nexport const TorusKnotBuilder = {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreateTorusKnot,\r\n};\r\n\r\nVertexData.CreateTorusKnot = CreateTorusKnotVertexData;\r\n\r\nMesh.CreateTorusKnot = (\r\n name: string,\r\n radius: number,\r\n tube: number,\r\n radialSegments: number,\r\n tubularSegments: number,\r\n p: number,\r\n q: number,\r\n scene?: Scene,\r\n updatable?: boolean,\r\n sideOrientation?: number\r\n): Mesh => {\r\n const options = {\r\n radius,\r\n tube,\r\n radialSegments,\r\n tubularSegments,\r\n p,\r\n q,\r\n sideOrientation,\r\n updatable,\r\n };\r\n\r\n return CreateTorusKnot(name, options, scene);\r\n};\r\n","import { Logger } from \"../Misc/logger\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Vector3, Vector2 } from \"../Maths/math.vector\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport { Mesh } from \"../Meshes/mesh\";\r\nimport { VertexData } from \"../Meshes/mesh.vertexData\";\r\nimport type { Nullable } from \"../types\";\r\nimport { Path2 } from \"../Maths/math.path\";\r\nimport { Epsilon } from \"../Maths/math.constants\";\r\nimport { EngineStore } from \"../Engines/engineStore\";\r\n\r\ndeclare let earcut: any;\r\n/**\r\n * Vector2 wth index property\r\n */\r\nclass IndexedVector2 extends Vector2 {\r\n constructor(\r\n original: Vector2,\r\n /** Index of the vector2 */\r\n public index: number\r\n ) {\r\n super(original.x, original.y);\r\n }\r\n}\r\n\r\n/**\r\n * Defines points to create a polygon\r\n */\r\nclass PolygonPoints {\r\n elements = new Array();\r\n\r\n add(originalPoints: Array): Array {\r\n const result = new Array();\r\n originalPoints.forEach((point) => {\r\n const newPoint = new IndexedVector2(point, this.elements.length);\r\n result.push(newPoint);\r\n this.elements.push(newPoint);\r\n });\r\n\r\n return result;\r\n }\r\n\r\n computeBounds(): { min: Vector2; max: Vector2; width: number; height: number } {\r\n const lmin = new Vector2(this.elements[0].x, this.elements[0].y);\r\n const lmax = new Vector2(this.elements[0].x, this.elements[0].y);\r\n\r\n this.elements.forEach((point) => {\r\n // x\r\n if (point.x < lmin.x) {\r\n lmin.x = point.x;\r\n } else if (point.x > lmax.x) {\r\n lmax.x = point.x;\r\n }\r\n\r\n // y\r\n if (point.y < lmin.y) {\r\n lmin.y = point.y;\r\n } else if (point.y > lmax.y) {\r\n lmax.y = point.y;\r\n }\r\n });\r\n\r\n return {\r\n min: lmin,\r\n max: lmax,\r\n width: lmax.x - lmin.x,\r\n height: lmax.y - lmin.y,\r\n };\r\n }\r\n}\r\n\r\n/**\r\n * Polygon\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#non-regular-polygon\r\n */\r\nexport class Polygon {\r\n /**\r\n * Creates a rectangle\r\n * @param xmin bottom X coord\r\n * @param ymin bottom Y coord\r\n * @param xmax top X coord\r\n * @param ymax top Y coord\r\n * @returns points that make the resulting rectangle\r\n */\r\n static Rectangle(xmin: number, ymin: number, xmax: number, ymax: number): Vector2[] {\r\n return [new Vector2(xmin, ymin), new Vector2(xmax, ymin), new Vector2(xmax, ymax), new Vector2(xmin, ymax)];\r\n }\r\n\r\n /**\r\n * Creates a circle\r\n * @param radius radius of circle\r\n * @param cx scale in x\r\n * @param cy scale in y\r\n * @param numberOfSides number of sides that make up the circle\r\n * @returns points that make the resulting circle\r\n */\r\n static Circle(radius: number, cx: number = 0, cy: number = 0, numberOfSides: number = 32): Vector2[] {\r\n const result = new Array();\r\n\r\n let angle = 0;\r\n const increment = (Math.PI * 2) / numberOfSides;\r\n\r\n for (let i = 0; i < numberOfSides; i++) {\r\n result.push(new Vector2(cx + Math.cos(angle) * radius, cy + Math.sin(angle) * radius));\r\n angle -= increment;\r\n }\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a polygon from input string\r\n * @param input Input polygon data\r\n * @returns the parsed points\r\n */\r\n static Parse(input: string): Vector2[] {\r\n const floats = input\r\n .split(/[^-+eE.\\d]+/)\r\n .map(parseFloat)\r\n .filter((val) => !isNaN(val));\r\n let i: number;\r\n const result = [];\r\n for (i = 0; i < (floats.length & 0x7ffffffe); i += 2) {\r\n result.push(new Vector2(floats[i], floats[i + 1]));\r\n }\r\n return result;\r\n }\r\n\r\n /**\r\n * Starts building a polygon from x and y coordinates\r\n * @param x x coordinate\r\n * @param y y coordinate\r\n * @returns the started path2\r\n */\r\n static StartingAt(x: number, y: number): Path2 {\r\n return Path2.StartingAt(x, y);\r\n }\r\n}\r\n\r\n/**\r\n * Builds a polygon\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param/polyMeshBuilder\r\n */\r\nexport class PolygonMeshBuilder {\r\n private _points = new PolygonPoints();\r\n private _outlinepoints = new PolygonPoints();\r\n private _holes = new Array();\r\n\r\n private _name: string;\r\n private _scene: Nullable;\r\n\r\n private _epoints: number[] = new Array();\r\n private _eholes: number[] = new Array();\r\n\r\n private _addToepoint(points: Vector2[]) {\r\n for (const p of points) {\r\n this._epoints.push(p.x, p.y);\r\n }\r\n }\r\n\r\n /**\r\n * Babylon reference to the earcut plugin.\r\n */\r\n public bjsEarcut: any;\r\n\r\n /**\r\n * Creates a PolygonMeshBuilder\r\n * @param name name of the builder\r\n * @param contours Path of the polygon\r\n * @param scene scene to add to when creating the mesh\r\n * @param earcutInjection can be used to inject your own earcut reference\r\n */\r\n constructor(name: string, contours: Path2 | Vector2[] | any, scene?: Scene, earcutInjection = earcut) {\r\n this.bjsEarcut = earcutInjection;\r\n this._name = name;\r\n this._scene = scene || EngineStore.LastCreatedScene;\r\n\r\n let points: Vector2[];\r\n if (contours instanceof Path2) {\r\n points = (contours).getPoints();\r\n } else {\r\n points = contours;\r\n }\r\n\r\n this._addToepoint(points);\r\n\r\n this._points.add(points);\r\n this._outlinepoints.add(points);\r\n\r\n if (typeof this.bjsEarcut === \"undefined\") {\r\n Logger.Warn(\"Earcut was not found, the polygon will not be built.\");\r\n }\r\n }\r\n\r\n /**\r\n * Adds a hole within the polygon\r\n * @param hole Array of points defining the hole\r\n * @returns this\r\n */\r\n addHole(hole: Vector2[]): PolygonMeshBuilder {\r\n this._points.add(hole);\r\n const holepoints = new PolygonPoints();\r\n holepoints.add(hole);\r\n this._holes.push(holepoints);\r\n\r\n this._eholes.push(this._epoints.length / 2);\r\n this._addToepoint(hole);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Creates the polygon\r\n * @param updatable If the mesh should be updatable\r\n * @param depth The depth of the mesh created\r\n * @param smoothingThreshold Dot product threshold for smoothed normals\r\n * @returns the created mesh\r\n */\r\n build(updatable: boolean = false, depth: number = 0, smoothingThreshold: number = 2): Mesh {\r\n const result = new Mesh(this._name, this._scene);\r\n\r\n const vertexData = this.buildVertexData(depth, smoothingThreshold);\r\n\r\n result.setVerticesData(VertexBuffer.PositionKind, vertexData.positions, updatable);\r\n result.setVerticesData(VertexBuffer.NormalKind, vertexData.normals, updatable);\r\n result.setVerticesData(VertexBuffer.UVKind, vertexData.uvs, updatable);\r\n result.setIndices(vertexData.indices);\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates the polygon\r\n * @param depth The depth of the mesh created\r\n * @param smoothingThreshold Dot product threshold for smoothed normals\r\n * @returns the created VertexData\r\n */\r\n buildVertexData(depth: number = 0, smoothingThreshold: number = 2): VertexData {\r\n const result = new VertexData();\r\n\r\n const normals = new Array();\r\n const positions = new Array();\r\n const uvs = new Array();\r\n\r\n const bounds = this._points.computeBounds();\r\n this._points.elements.forEach((p) => {\r\n normals.push(0, 1.0, 0);\r\n positions.push(p.x, 0, p.y);\r\n uvs.push((p.x - bounds.min.x) / bounds.width, (p.y - bounds.min.y) / bounds.height);\r\n });\r\n\r\n const indices = new Array();\r\n\r\n const res = this.bjsEarcut(this._epoints, this._eholes, 2);\r\n\r\n for (let i = 0; i < res.length; i++) {\r\n indices.push(res[i]);\r\n }\r\n\r\n if (depth > 0) {\r\n const positionscount = positions.length / 3; //get the current pointcount\r\n\r\n this._points.elements.forEach((p) => {\r\n //add the elements at the depth\r\n normals.push(0, -1.0, 0);\r\n positions.push(p.x, -depth, p.y);\r\n uvs.push(1 - (p.x - bounds.min.x) / bounds.width, 1 - (p.y - bounds.min.y) / bounds.height);\r\n });\r\n\r\n const totalCount = indices.length;\r\n for (let i = 0; i < totalCount; i += 3) {\r\n const i0 = indices[i + 0];\r\n const i1 = indices[i + 1];\r\n const i2 = indices[i + 2];\r\n\r\n indices.push(i2 + positionscount);\r\n indices.push(i1 + positionscount);\r\n indices.push(i0 + positionscount);\r\n }\r\n\r\n //Add the sides\r\n this._addSide(positions, normals, uvs, indices, bounds, this._outlinepoints, depth, false, smoothingThreshold);\r\n\r\n this._holes.forEach((hole) => {\r\n this._addSide(positions, normals, uvs, indices, bounds, hole, depth, true, smoothingThreshold);\r\n });\r\n }\r\n\r\n result.indices = indices;\r\n result.positions = positions;\r\n result.normals = normals;\r\n result.uvs = uvs;\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Adds a side to the polygon\r\n * @param positions points that make the polygon\r\n * @param normals normals of the polygon\r\n * @param uvs uvs of the polygon\r\n * @param indices indices of the polygon\r\n * @param bounds bounds of the polygon\r\n * @param points points of the polygon\r\n * @param depth depth of the polygon\r\n * @param flip flip of the polygon\r\n * @param smoothingThreshold\r\n */\r\n private _addSide(positions: any[], normals: any[], uvs: any[], indices: any[], bounds: any, points: PolygonPoints, depth: number, flip: boolean, smoothingThreshold: number) {\r\n let startIndex: number = positions.length / 3;\r\n let ulength: number = 0;\r\n for (let i: number = 0; i < points.elements.length; i++) {\r\n const p: IndexedVector2 = points.elements[i];\r\n const p1: IndexedVector2 = points.elements[(i + 1) % points.elements.length];\r\n\r\n positions.push(p.x, 0, p.y);\r\n positions.push(p.x, -depth, p.y);\r\n positions.push(p1.x, 0, p1.y);\r\n positions.push(p1.x, -depth, p1.y);\r\n\r\n const p0: IndexedVector2 = points.elements[(i + points.elements.length - 1) % points.elements.length];\r\n const p2: IndexedVector2 = points.elements[(i + 2) % points.elements.length];\r\n\r\n let vc = new Vector3(-(p1.y - p.y), 0, p1.x - p.x);\r\n let vp = new Vector3(-(p.y - p0.y), 0, p.x - p0.x);\r\n let vn = new Vector3(-(p2.y - p1.y), 0, p2.x - p1.x);\r\n\r\n if (!flip) {\r\n vc = vc.scale(-1);\r\n vp = vp.scale(-1);\r\n vn = vn.scale(-1);\r\n }\r\n\r\n const vc_norm = vc.normalizeToNew();\r\n let vp_norm = vp.normalizeToNew();\r\n let vn_norm = vn.normalizeToNew();\r\n\r\n const dotp = Vector3.Dot(vp_norm, vc_norm);\r\n if (dotp > smoothingThreshold) {\r\n if (dotp < Epsilon - 1) {\r\n vp_norm = new Vector3(p.x, 0, p.y).subtract(new Vector3(p1.x, 0, p1.y)).normalize();\r\n } else {\r\n // cheap average weighed by side length\r\n vp_norm = vp.add(vc).normalize();\r\n }\r\n } else {\r\n vp_norm = vc_norm;\r\n }\r\n\r\n const dotn = Vector3.Dot(vn, vc);\r\n if (dotn > smoothingThreshold) {\r\n if (dotn < Epsilon - 1) {\r\n // back to back\r\n vn_norm = new Vector3(p1.x, 0, p1.y).subtract(new Vector3(p.x, 0, p.y)).normalize();\r\n } else {\r\n // cheap average weighed by side length\r\n vn_norm = vn.add(vc).normalize();\r\n }\r\n } else {\r\n vn_norm = vc_norm;\r\n }\r\n\r\n uvs.push(ulength / bounds.width, 0);\r\n uvs.push(ulength / bounds.width, 1);\r\n ulength += vc.length();\r\n uvs.push(ulength / bounds.width, 0);\r\n uvs.push(ulength / bounds.width, 1);\r\n\r\n normals.push(vp_norm.x, vp_norm.y, vp_norm.z);\r\n normals.push(vp_norm.x, vp_norm.y, vp_norm.z);\r\n normals.push(vn_norm.x, vn_norm.y, vn_norm.z);\r\n normals.push(vn_norm.x, vn_norm.y, vn_norm.z);\r\n\r\n if (!flip) {\r\n indices.push(startIndex);\r\n indices.push(startIndex + 1);\r\n indices.push(startIndex + 2);\r\n\r\n indices.push(startIndex + 1);\r\n indices.push(startIndex + 3);\r\n indices.push(startIndex + 2);\r\n } else {\r\n indices.push(startIndex);\r\n indices.push(startIndex + 2);\r\n indices.push(startIndex + 1);\r\n\r\n indices.push(startIndex + 1);\r\n indices.push(startIndex + 2);\r\n indices.push(startIndex + 3);\r\n }\r\n startIndex += 4;\r\n }\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport type { Scene } from \"../../scene\";\r\nimport type { Vector3 } from \"../../Maths/math.vector\";\r\nimport { Vector2, Vector4 } from \"../../Maths/math.vector\";\r\nimport { Color4 } from \"../../Maths/math.color\";\r\nimport { Mesh } from \"../mesh\";\r\nimport { VertexData } from \"../mesh.vertexData\";\r\nimport { PolygonMeshBuilder } from \"../polygonMesh\";\r\nimport type { FloatArray, IndicesArray, Nullable } from \"../../types\";\r\nimport { VertexBuffer } from \"../../Buffers/buffer\";\r\nimport { EngineStore } from \"../../Engines/engineStore\";\r\nimport { CompatibilityOptions } from \"../../Compat/compatibilityOptions\";\r\n\r\ndeclare let earcut: any;\r\n\r\n/**\r\n * Creates the VertexData for an irregular Polygon in the XoZ plane using a mesh built by polygonTriangulation.build()\r\n * All parameters are provided by CreatePolygon as needed\r\n * @param polygon a mesh built from polygonTriangulation.build()\r\n * @param sideOrientation takes the values Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * @param fUV an array of Vector4 elements used to set different images to the top, rings and bottom respectively\r\n * @param fColors an array of Color3 elements used to set different colors to the top, rings and bottom respectively\r\n * @param frontUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * @param backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param wrp a boolean, default false, when true and fUVs used texture is wrapped around all sides, when false texture is applied side\r\n * @returns the VertexData of the Polygon\r\n */\r\nexport function CreatePolygonVertexData(polygon: Mesh, sideOrientation: number, fUV?: Vector4[], fColors?: Color4[], frontUVs?: Vector4, backUVs?: Vector4, wrp?: boolean) {\r\n const faceUV: Vector4[] = fUV || new Array(3);\r\n const faceColors = fColors;\r\n const colors = [];\r\n const wrap: boolean = wrp || false;\r\n\r\n // default face colors and UV if undefined\r\n for (let f = 0; f < 3; f++) {\r\n if (faceUV[f] === undefined) {\r\n faceUV[f] = new Vector4(0, 0, 1, 1);\r\n }\r\n if (faceColors && faceColors[f] === undefined) {\r\n faceColors[f] = new Color4(1, 1, 1, 1);\r\n }\r\n }\r\n\r\n const positions = polygon.getVerticesData(VertexBuffer.PositionKind);\r\n const normals = polygon.getVerticesData(VertexBuffer.NormalKind);\r\n const uvs = polygon.getVerticesData(VertexBuffer.UVKind);\r\n const indices = polygon.getIndices();\r\n const startIndex = positions.length / 9;\r\n let disp = 0;\r\n let distX = 0;\r\n let distZ = 0;\r\n let dist = 0;\r\n let totalLen = 0;\r\n const cumulate = [0];\r\n if (wrap) {\r\n for (let idx = startIndex; idx < positions.length / 3; idx += 4) {\r\n distX = positions[3 * (idx + 2)] - positions[3 * idx];\r\n distZ = positions[3 * (idx + 2) + 2] - positions[3 * idx + 2];\r\n dist = Math.sqrt(distX * distX + distZ * distZ);\r\n totalLen += dist;\r\n cumulate.push(totalLen);\r\n }\r\n }\r\n // set face colours and textures\r\n let idx: number = 0;\r\n let face: number = 0;\r\n for (let index = 0; index < normals.length; index += 3) {\r\n //Edge Face no. 1\r\n if (Math.abs(normals[index + 1]) < 0.001) {\r\n face = 1;\r\n }\r\n //Top Face no. 0\r\n if (Math.abs(normals[index + 1] - 1) < 0.001) {\r\n face = 0;\r\n }\r\n //Bottom Face no. 2\r\n if (Math.abs(normals[index + 1] + 1) < 0.001) {\r\n face = 2;\r\n }\r\n idx = index / 3;\r\n if (face === 1) {\r\n disp = idx - startIndex;\r\n if (disp % 4 < 1.5) {\r\n if (wrap) {\r\n uvs[2 * idx] = faceUV[face].x + ((faceUV[face].z - faceUV[face].x) * cumulate[Math.floor(disp / 4)]) / totalLen;\r\n } else {\r\n uvs[2 * idx] = faceUV[face].x;\r\n }\r\n } else {\r\n if (wrap) {\r\n uvs[2 * idx] = faceUV[face].x + ((faceUV[face].z - faceUV[face].x) * cumulate[Math.floor(disp / 4) + 1]) / totalLen;\r\n } else {\r\n uvs[2 * idx] = faceUV[face].z;\r\n }\r\n }\r\n if (disp % 2 === 0) {\r\n uvs[2 * idx + 1] = CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - faceUV[face].w : faceUV[face].w;\r\n } else {\r\n uvs[2 * idx + 1] = CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - faceUV[face].y : faceUV[face].y;\r\n }\r\n } else {\r\n uvs[2 * idx] = (1 - uvs[2 * idx]) * faceUV[face].x + uvs[2 * idx] * faceUV[face].z;\r\n uvs[2 * idx + 1] = (1 - uvs[2 * idx + 1]) * faceUV[face].y + uvs[2 * idx + 1] * faceUV[face].w;\r\n\r\n if (CompatibilityOptions.UseOpenGLOrientationForUV) {\r\n uvs[2 * idx + 1] = 1.0 - uvs[2 * idx + 1];\r\n }\r\n }\r\n if (faceColors) {\r\n colors.push(faceColors[face].r, faceColors[face].g, faceColors[face].b, faceColors[face].a);\r\n }\r\n }\r\n\r\n // sides\r\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, frontUVs, backUVs);\r\n\r\n // Result\r\n const vertexData = new VertexData();\r\n vertexData.indices = indices;\r\n vertexData.positions = positions;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n\r\n if (faceColors) {\r\n const totalColors = sideOrientation === VertexData.DOUBLESIDE ? colors.concat(colors) : colors;\r\n vertexData.colors = totalColors;\r\n }\r\n\r\n return vertexData;\r\n}\r\n\r\n/**\r\n * Creates a polygon mesh\r\n * The polygon's shape will depend on the input parameters and is constructed parallel to a ground mesh\r\n * * The parameter `shape` is a required array of successive Vector3 representing the corners of the polygon in th XoZ plane, that is y = 0 for all vectors\r\n * * You can set the mesh side orientation with the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4)\r\n * * Remember you can only change the shape positions, not their number when updating a polygon\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param options.shape\r\n * @param options.holes\r\n * @param options.depth\r\n * @param options.smoothingThreshold\r\n * @param options.faceUV\r\n * @param options.faceColors\r\n * @param options.updatable\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param options.wrap\r\n * @param scene defines the hosting scene\r\n * @param earcutInjection can be used to inject your own earcut reference\r\n * @returns the polygon mesh\r\n */\r\nexport function CreatePolygon(\r\n name: string,\r\n options: {\r\n shape: Vector3[];\r\n holes?: Vector3[][];\r\n depth?: number;\r\n smoothingThreshold?: number;\r\n faceUV?: Vector4[];\r\n faceColors?: Color4[];\r\n updatable?: boolean;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n wrap?: boolean;\r\n },\r\n scene: Nullable = null,\r\n earcutInjection = earcut\r\n): Mesh {\r\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n const shape = options.shape;\r\n const holes = options.holes || [];\r\n const depth = options.depth || 0;\r\n const smoothingThreshold = options.smoothingThreshold || 2;\r\n const contours: Array = [];\r\n let hole: Array = [];\r\n\r\n for (let i = 0; i < shape.length; i++) {\r\n contours[i] = new Vector2(shape[i].x, shape[i].z);\r\n }\r\n const epsilon = 0.00000001;\r\n if (contours[0].equalsWithEpsilon(contours[contours.length - 1], epsilon)) {\r\n contours.pop();\r\n }\r\n\r\n const polygonTriangulation = new PolygonMeshBuilder(name, contours, scene || EngineStore.LastCreatedScene!, earcutInjection);\r\n for (let hNb = 0; hNb < holes.length; hNb++) {\r\n hole = [];\r\n for (let hPoint = 0; hPoint < holes[hNb].length; hPoint++) {\r\n hole.push(new Vector2(holes[hNb][hPoint].x, holes[hNb][hPoint].z));\r\n }\r\n polygonTriangulation.addHole(hole);\r\n }\r\n //updatability is set during applyToMesh; setting to true in triangulation build produces errors\r\n const polygon = polygonTriangulation.build(false, depth, smoothingThreshold);\r\n polygon._originalBuilderSideOrientation = options.sideOrientation;\r\n const vertexData = CreatePolygonVertexData(polygon, options.sideOrientation, options.faceUV, options.faceColors, options.frontUVs, options.backUVs, options.wrap);\r\n vertexData.applyToMesh(polygon, options.updatable);\r\n\r\n return polygon;\r\n}\r\n\r\n/**\r\n * Creates an extruded polygon mesh, with depth in the Y direction.\r\n * * You can set different colors and different images to the top, bottom and extruded side by using the parameters `faceColors` (an array of 3 Color3 elements) and `faceUV` (an array of 3 Vector4 elements)\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/texturePerBoxFace\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param options.shape\r\n * @param options.holes\r\n * @param options.depth\r\n * @param options.faceUV\r\n * @param options.faceColors\r\n * @param options.updatable\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param options.wrap\r\n * @param scene defines the hosting scene\r\n * @param earcutInjection can be used to inject your own earcut reference\r\n * @returns the polygon mesh\r\n */\r\nexport function ExtrudePolygon(\r\n name: string,\r\n options: {\r\n shape: Vector3[];\r\n holes?: Vector3[][];\r\n depth?: number;\r\n faceUV?: Vector4[];\r\n faceColors?: Color4[];\r\n updatable?: boolean;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n wrap?: boolean;\r\n },\r\n scene: Nullable = null,\r\n earcutInjection = earcut\r\n): Mesh {\r\n return CreatePolygon(name, options, scene, earcutInjection);\r\n}\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated use the functions directly from the module\r\n */\r\nexport const PolygonBuilder = {\r\n ExtrudePolygon,\r\n CreatePolygon,\r\n};\r\n\r\nVertexData.CreatePolygon = CreatePolygonVertexData;\r\nMesh.CreatePolygon = (name: string, shape: Vector3[], scene: Scene, holes?: Vector3[][], updatable?: boolean, sideOrientation?: number, earcutInjection = earcut): Mesh => {\r\n const options = {\r\n shape: shape,\r\n holes: holes,\r\n updatable: updatable,\r\n sideOrientation: sideOrientation,\r\n };\r\n return CreatePolygon(name, options, scene, earcutInjection);\r\n};\r\n\r\nMesh.ExtrudePolygon = (\r\n name: string,\r\n shape: Vector3[],\r\n depth: number,\r\n scene: Scene,\r\n holes?: Vector3[][],\r\n updatable?: boolean,\r\n sideOrientation?: number,\r\n earcutInjection = earcut\r\n): Mesh => {\r\n const options = {\r\n shape: shape,\r\n holes: holes,\r\n depth: depth,\r\n updatable: updatable,\r\n sideOrientation: sideOrientation,\r\n };\r\n return ExtrudePolygon(name, options, scene, earcutInjection);\r\n};\r\n","import type { Nullable } from \"../../types\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { Vector4 } from \"../../Maths/math.vector\";\r\nimport { Vector3, TmpVectors, Matrix } from \"../../Maths/math.vector\";\r\nimport { Mesh } from \"../mesh\";\r\nimport { CreateRibbon } from \"./ribbonBuilder\";\r\nimport { Path3D } from \"../../Maths/math.path\";\r\n\r\n/**\r\n * Creates an extruded shape mesh. The extrusion is a parametric shape. It has no predefined shape. Its final shape will depend on the input parameters.\r\n * * The parameter `shape` is a required array of successive Vector3. This array depicts the shape to be extruded in its local space : the shape must be designed in the xOy plane and will be extruded along the Z axis.\r\n * * The parameter `path` is a required array of successive Vector3. This is the axis curve the shape is extruded along.\r\n * * The parameter `rotation` (float, default 0 radians) is the angle value to rotate the shape each step (each path point), from the former step (so rotation added each step) along the curve.\r\n * * The parameter `scale` (float, default 1) is the value to scale the shape.\r\n * * The parameter `closeShape` (boolean, default false) closes the shape when true, since v5.0.0.\r\n * * The parameter `closePath` (boolean, default false) closes the path when true and no caps, since v5.0.0.\r\n * * The parameter `cap` sets the way the extruded shape is capped. Possible values : BABYLON.Mesh.NO_CAP (default), BABYLON.Mesh.CAP_START, BABYLON.Mesh.CAP_END, BABYLON.Mesh.CAP_ALL\r\n * * The optional parameter `instance` is an instance of an existing ExtrudedShape object to be updated with the passed `shape`, `path`, `scale` or `rotation` parameters : https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#extruded-shape\r\n * * Remember you can only change the shape or path point positions, not their number when updating an extruded shape.\r\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\r\n * * The optional parameter `invertUV` (boolean, default false) swaps in the geometry the U and V coordinates to apply a texture.\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\r\n * * The optional parameter `firstNormal` (Vector3) defines the direction of the first normal of the supplied path. Consider using this for any path that is straight, and particular for paths in the xy plane.\r\n * * The optional `adjustFrame` (boolean, default false) will cause the internally generated Path3D tangents, normals, and binormals to be adjusted so that a) they are always well-defined, and b) they do not reverse from one path point to the next. This prevents the extruded shape from being flipped and/or rotated with resulting mesh self-intersections. This is primarily useful for straight paths that can reverse direction.\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param options.shape\r\n * @param options.path\r\n * @param options.scale\r\n * @param options.rotation\r\n * @param options.closeShape\r\n * @param options.closePath\r\n * @param options.cap\r\n * @param options.updatable\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param options.instance\r\n * @param options.invertUV\r\n * @param options.firstNormal\r\n * @param options.adjustFrame\r\n * @param scene defines the hosting scene\r\n * @returns the extruded shape mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#extruded-shapes\r\n */\r\nexport function ExtrudeShape(\r\n name: string,\r\n options: {\r\n shape: Vector3[];\r\n path: Vector3[];\r\n scale?: number;\r\n rotation?: number;\r\n closeShape?: boolean;\r\n closePath?: boolean;\r\n cap?: number;\r\n updatable?: boolean;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n instance?: Mesh;\r\n invertUV?: boolean;\r\n firstNormal?: Vector3;\r\n adjustFrame?: boolean;\r\n },\r\n scene: Nullable = null\r\n): Mesh {\r\n const path = options.path;\r\n const shape = options.shape;\r\n const scale = options.scale || 1;\r\n const rotation = options.rotation || 0;\r\n const cap = options.cap === 0 ? 0 : options.cap || Mesh.NO_CAP;\r\n const updatable = options.updatable;\r\n const sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n const instance = options.instance || null;\r\n const invertUV = options.invertUV || false;\r\n const closeShape = options.closeShape || false;\r\n const closePath = options.closePath || false;\r\n\r\n return _ExtrudeShapeGeneric(\r\n name,\r\n shape,\r\n path,\r\n scale,\r\n rotation,\r\n null,\r\n null,\r\n closePath,\r\n closeShape,\r\n cap,\r\n false,\r\n scene,\r\n updatable ? true : false,\r\n sideOrientation,\r\n instance,\r\n invertUV,\r\n options.frontUVs || null,\r\n options.backUVs || null,\r\n options.firstNormal || null,\r\n options.adjustFrame ? true : false\r\n );\r\n}\r\n\r\n/**\r\n * Creates an custom extruded shape mesh.\r\n * The custom extrusion is a parametric shape. It has no predefined shape. Its final shape will depend on the input parameters.\r\n * * The parameter `shape` is a required array of successive Vector3. This array depicts the shape to be extruded in its local space : the shape must be designed in the xOy plane and will be extruded along the Z axis.\r\n * * The parameter `path` is a required array of successive Vector3. This is the axis curve the shape is extruded along.\r\n * * The parameter `rotationFunction` (JS function) is a custom Javascript function called on each path point. This function is passed the position i of the point in the path and the distance of this point from the beginning of the path\r\n * * It must returns a float value that will be the rotation in radians applied to the shape on each path point.\r\n * * The parameter `scaleFunction` (JS function) is a custom Javascript function called on each path point. This function is passed the position i of the point in the path and the distance of this point from the beginning of the path\r\n * * It must returns a float value that will be the scale value applied to the shape on each path point\r\n * * The parameter `closeShape` (boolean, default false) closes the shape when true, since v5.0.0.\r\n * * The parameter `closePath` (boolean, default false) closes the path when true and no caps, since v5.0.0.\r\n * * The parameter `ribbonClosePath` (boolean, default false) forces the extrusion underlying ribbon to close all the paths in its `pathArray` - depreciated in favor of closeShape\r\n * * The parameter `ribbonCloseArray` (boolean, default false) forces the extrusion underlying ribbon to close its `pathArray` - depreciated in favor of closePath\r\n * * The parameter `cap` sets the way the extruded shape is capped. Possible values : BABYLON.Mesh.NO_CAP (default), BABYLON.Mesh.CAP_START, BABYLON.Mesh.CAP_END, BABYLON.Mesh.CAP_ALL\r\n * * The optional parameter `instance` is an instance of an existing ExtrudedShape object to be updated with the passed `shape`, `path`, `scale` or `rotation` parameters : https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#extruded-shape\r\n * * Remember you can only change the shape or path point positions, not their number when updating an extruded shape\r\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\r\n * * The optional parameter `invertUV` (boolean, default false) swaps in the geometry the U and V coordinates to apply a texture\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\r\n * * The optional parameter `firstNormal` (Vector3) defines the direction of the first normal of the supplied path. It should be supplied when the path is in the xy plane, and particularly if these sections are straight, because the underlying Path3D object will pick a normal in the xy plane that causes the extrusion to be collapsed into the plane. This should be used for any path that is straight.\r\n * * The optional `adjustFrame` (boolean, default false) will cause the internally generated Path3D tangents, normals, and binormals to be adjusted so that a) they are always well-defined, and b) they do not reverse from one path point to the next. This prevents the extruded shape from being flipped and/or rotated with resulting mesh self-intersections. This is primarily useful for straight paths that can reverse direction.\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param options.shape\r\n * @param options.path\r\n * @param options.scaleFunction\r\n * @param options.rotationFunction\r\n * @param options.ribbonCloseArray\r\n * @param options.ribbonClosePath\r\n * @param options.closeShape\r\n * @param options.closePath\r\n * @param options.cap\r\n * @param options.updatable\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param options.instance\r\n * @param options.invertUV\r\n * @param options.firstNormal\r\n * @param options.adjustFrame\r\n * @param scene defines the hosting scene\r\n * @returns the custom extruded shape mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#custom-extruded-shapes\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#extruded-shapes\r\n */\r\nexport function ExtrudeShapeCustom(\r\n name: string,\r\n options: {\r\n shape: Vector3[];\r\n path: Vector3[];\r\n scaleFunction?: Nullable<{ (i: number, distance: number): number }>;\r\n rotationFunction?: Nullable<{ (i: number, distance: number): number }>;\r\n ribbonCloseArray?: boolean;\r\n ribbonClosePath?: boolean;\r\n closeShape?: boolean;\r\n closePath?: boolean;\r\n cap?: number;\r\n updatable?: boolean;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n instance?: Mesh;\r\n invertUV?: boolean;\r\n firstNormal?: Vector3;\r\n adjustFrame?: boolean;\r\n },\r\n scene: Nullable = null\r\n): Mesh {\r\n const path = options.path;\r\n const shape = options.shape;\r\n const scaleFunction =\r\n options.scaleFunction ||\r\n (() => {\r\n return 1;\r\n });\r\n const rotationFunction =\r\n options.rotationFunction ||\r\n (() => {\r\n return 0;\r\n });\r\n const ribbonCloseArray = options.closePath || options.ribbonCloseArray || false;\r\n const ribbonClosePath = options.closeShape || options.ribbonClosePath || false;\r\n const cap = options.cap === 0 ? 0 : options.cap || Mesh.NO_CAP;\r\n const updatable = options.updatable;\r\n const firstNormal = options.firstNormal || null;\r\n const adjustFrame = options.adjustFrame || false;\r\n const sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n const instance = options.instance;\r\n const invertUV = options.invertUV || false;\r\n return _ExtrudeShapeGeneric(\r\n name,\r\n shape,\r\n path,\r\n null,\r\n null,\r\n scaleFunction,\r\n rotationFunction,\r\n ribbonCloseArray,\r\n ribbonClosePath,\r\n cap,\r\n true,\r\n scene,\r\n updatable ? true : false,\r\n sideOrientation,\r\n instance || null,\r\n invertUV,\r\n options.frontUVs || null,\r\n options.backUVs || null,\r\n firstNormal,\r\n adjustFrame\r\n );\r\n}\r\n\r\nfunction _ExtrudeShapeGeneric(\r\n name: string,\r\n shape: Vector3[],\r\n curve: Vector3[],\r\n scale: Nullable,\r\n rotation: Nullable,\r\n scaleFunction: Nullable<{ (i: number, distance: number): number }>,\r\n rotateFunction: Nullable<{ (i: number, distance: number): number }>,\r\n rbCA: boolean,\r\n rbCP: boolean,\r\n cap: number,\r\n custom: boolean,\r\n scene: Nullable,\r\n updtbl: boolean,\r\n side: number,\r\n instance: Nullable,\r\n invertUV: boolean,\r\n frontUVs: Nullable,\r\n backUVs: Nullable,\r\n firstNormal: Nullable,\r\n adjustFrame: boolean\r\n): Mesh {\r\n // extrusion geometry\r\n const extrusionPathArray = (\r\n shape: Vector3[],\r\n curve: Vector3[],\r\n path3D: Path3D,\r\n shapePaths: Vector3[][],\r\n scale: Nullable,\r\n rotation: Nullable,\r\n scaleFunction: Nullable<{ (i: number, distance: number): number }>,\r\n rotateFunction: Nullable<{ (i: number, distance: number): number }>,\r\n cap: number,\r\n custom: boolean,\r\n adjustFrame: boolean\r\n ) => {\r\n const tangents = path3D.getTangents();\r\n const normals = path3D.getNormals();\r\n const binormals = path3D.getBinormals();\r\n const distances = path3D.getDistances();\r\n if (adjustFrame) {\r\n /* fix tangents,normals, binormals */\r\n for (let i = 0; i < tangents.length; i++) {\r\n if (tangents[i].x == 0 && tangents[i].y == 0 && tangents[i].z == 0) {\r\n tangents[i].copyFrom(tangents[i - 1]);\r\n }\r\n if (normals[i].x == 0 && normals[i].y == 0 && normals[i].z == 0) {\r\n normals[i].copyFrom(normals[i - 1]);\r\n }\r\n if (binormals[i].x == 0 && binormals[i].y == 0 && binormals[i].z == 0) {\r\n binormals[i].copyFrom(binormals[i - 1]);\r\n }\r\n if (i > 0) {\r\n let v = tangents[i - 1];\r\n if (Vector3.Dot(v, tangents[i]) < 0) {\r\n tangents[i].scaleInPlace(-1);\r\n }\r\n v = normals[i - 1];\r\n if (Vector3.Dot(v, normals[i]) < 0) {\r\n normals[i].scaleInPlace(-1);\r\n }\r\n v = binormals[i - 1];\r\n if (Vector3.Dot(v, binormals[i]) < 0) {\r\n binormals[i].scaleInPlace(-1);\r\n }\r\n }\r\n }\r\n }\r\n let angle = 0;\r\n const returnScale = () => {\r\n return scale !== null ? scale : 1;\r\n };\r\n const returnRotation = () => {\r\n return rotation !== null ? rotation : 0;\r\n };\r\n const rotate: { (i: number, distance: number): number } = custom && rotateFunction ? rotateFunction : returnRotation;\r\n const scl: { (i: number, distance: number): number } = custom && scaleFunction ? scaleFunction : returnScale;\r\n let index = cap === Mesh.NO_CAP || cap === Mesh.CAP_END ? 0 : 2;\r\n const rotationMatrix: Matrix = TmpVectors.Matrix[0];\r\n\r\n for (let i = 0; i < curve.length; i++) {\r\n const shapePath = new Array();\r\n const angleStep = rotate(i, distances[i]);\r\n const scaleRatio = scl(i, distances[i]);\r\n Matrix.RotationAxisToRef(tangents[i], angle, rotationMatrix);\r\n for (let p = 0; p < shape.length; p++) {\r\n const planed = tangents[i].scale(shape[p].z).add(normals[i].scale(shape[p].x)).add(binormals[i].scale(shape[p].y));\r\n const rotated = Vector3.Zero();\r\n Vector3.TransformCoordinatesToRef(planed, rotationMatrix, rotated);\r\n rotated.scaleInPlace(scaleRatio).addInPlace(curve[i]);\r\n shapePath[p] = rotated;\r\n }\r\n shapePaths[index] = shapePath;\r\n angle += angleStep;\r\n index++;\r\n }\r\n // cap\r\n const capPath = (shapePath: Vector3[]) => {\r\n const pointCap = Array();\r\n const barycenter = Vector3.Zero();\r\n let i: number;\r\n for (i = 0; i < shapePath.length; i++) {\r\n barycenter.addInPlace(shapePath[i]);\r\n }\r\n barycenter.scaleInPlace(1.0 / shapePath.length);\r\n for (i = 0; i < shapePath.length; i++) {\r\n pointCap.push(barycenter);\r\n }\r\n return pointCap;\r\n };\r\n switch (cap) {\r\n case Mesh.NO_CAP:\r\n break;\r\n case Mesh.CAP_START:\r\n shapePaths[0] = capPath(shapePaths[2]);\r\n shapePaths[1] = shapePaths[2];\r\n break;\r\n case Mesh.CAP_END:\r\n shapePaths[index] = shapePaths[index - 1];\r\n shapePaths[index + 1] = capPath(shapePaths[index - 1]);\r\n break;\r\n case Mesh.CAP_ALL:\r\n shapePaths[0] = capPath(shapePaths[2]);\r\n shapePaths[1] = shapePaths[2];\r\n shapePaths[index] = shapePaths[index - 1];\r\n shapePaths[index + 1] = capPath(shapePaths[index - 1]);\r\n break;\r\n default:\r\n break;\r\n }\r\n return shapePaths;\r\n };\r\n let path3D;\r\n let pathArray;\r\n if (instance) {\r\n // instance update\r\n const storage = instance._creationDataStorage!;\r\n path3D = firstNormal ? storage.path3D.update(curve, firstNormal) : storage.path3D.update(curve);\r\n pathArray = extrusionPathArray(shape, curve, storage.path3D, storage.pathArray, scale, rotation, scaleFunction, rotateFunction, storage.cap, custom, adjustFrame);\r\n instance = CreateRibbon(\"\", { pathArray, closeArray: false, closePath: false, offset: 0, updatable: false, sideOrientation: 0, instance }, scene || undefined);\r\n\r\n return instance;\r\n }\r\n // extruded shape creation\r\n path3D = firstNormal ? new Path3D(curve, firstNormal) : new Path3D(curve);\r\n const newShapePaths = new Array>();\r\n cap = cap < 0 || cap > 3 ? 0 : cap;\r\n pathArray = extrusionPathArray(shape, curve, path3D, newShapePaths, scale, rotation, scaleFunction, rotateFunction, cap, custom, adjustFrame);\r\n const extrudedGeneric = CreateRibbon(\r\n name,\r\n {\r\n pathArray: pathArray,\r\n closeArray: rbCA,\r\n closePath: rbCP,\r\n updatable: updtbl,\r\n sideOrientation: side,\r\n invertUV: invertUV,\r\n frontUVs: frontUVs || undefined,\r\n backUVs: backUVs || undefined,\r\n },\r\n scene\r\n );\r\n extrudedGeneric._creationDataStorage!.pathArray = pathArray;\r\n extrudedGeneric._creationDataStorage!.path3D = path3D;\r\n extrudedGeneric._creationDataStorage!.cap = cap;\r\n\r\n return extrudedGeneric;\r\n}\r\n\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated please use the functions directly from the module\r\n */\r\nexport const ShapeBuilder = {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n ExtrudeShape,\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n ExtrudeShapeCustom,\r\n};\r\n\r\nMesh.ExtrudeShape = (\r\n name: string,\r\n shape: Vector3[],\r\n path: Vector3[],\r\n scale: number,\r\n rotation: number,\r\n cap: number,\r\n scene: Nullable = null,\r\n updatable?: boolean,\r\n sideOrientation?: number,\r\n instance?: Mesh\r\n): Mesh => {\r\n const options = {\r\n shape: shape,\r\n path: path,\r\n scale: scale,\r\n rotation: rotation,\r\n cap: cap === 0 ? 0 : cap || Mesh.NO_CAP,\r\n sideOrientation: sideOrientation,\r\n instance: instance,\r\n updatable: updatable,\r\n };\r\n\r\n return ExtrudeShape(name, options, scene);\r\n};\r\n\r\nMesh.ExtrudeShapeCustom = (\r\n name: string,\r\n shape: Vector3[],\r\n path: Vector3[],\r\n scaleFunction: Nullable<{ (i: number, distance: number): number }>,\r\n rotationFunction: Nullable<{ (i: number, distance: number): number }>,\r\n ribbonCloseArray: boolean,\r\n ribbonClosePath: boolean,\r\n cap: number,\r\n scene: Scene,\r\n updatable?: boolean,\r\n sideOrientation?: number,\r\n instance?: Mesh\r\n): Mesh => {\r\n const options = {\r\n shape: shape,\r\n path: path,\r\n scaleFunction: scaleFunction,\r\n rotationFunction: rotationFunction,\r\n ribbonCloseArray: ribbonCloseArray,\r\n ribbonClosePath: ribbonClosePath,\r\n cap: cap === 0 ? 0 : cap || Mesh.NO_CAP,\r\n sideOrientation: sideOrientation,\r\n instance: instance,\r\n updatable: updatable,\r\n };\r\n\r\n return ExtrudeShapeCustom(name, options, scene);\r\n};\r\n","import type { Scene } from \"../../scene\";\r\nimport type { Vector4 } from \"../../Maths/math.vector\";\r\nimport { Vector3 } from \"../../Maths/math.vector\";\r\nimport { Mesh } from \"../mesh\";\r\nimport { CreateRibbon } from \"./ribbonBuilder\";\r\nimport type { Nullable } from \"../../types\";\r\n\r\n/**\r\n * Creates lathe mesh.\r\n * The lathe is a shape with a symmetry axis : a 2D model shape is rotated around this axis to design the lathe\r\n * * The parameter `shape` is a required array of successive Vector3. This array depicts the shape to be rotated in its local space : the shape must be designed in the xOy plane and will be rotated around the Y axis. It's usually a 2D shape, so the Vector3 z coordinates are often set to zero\r\n * * The parameter `radius` (positive float, default 1) is the radius value of the lathe\r\n * * The parameter `tessellation` (positive integer, default 64) is the side number of the lathe\r\n * * The parameter `clip` (positive integer, default 0) is the number of sides to not create without effecting the general shape of the sides\r\n * * The parameter `arc` (positive float, default 1) is the ratio of the lathe. 0.5 builds for instance half a lathe, so an opened shape\r\n * * The parameter `closed` (boolean, default true) opens/closes the lathe circumference. This should be set to false when used with the parameter \"arc\"\r\n * * The parameter `cap` sets the way the extruded shape is capped. Possible values : BABYLON.Mesh.NO_CAP (default), BABYLON.Mesh.CAP_START, BABYLON.Mesh.CAP_END, BABYLON.Mesh.CAP_ALL\r\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\r\n * * The optional parameter `invertUV` (boolean, default false) swaps in the geometry the U and V coordinates to apply a texture\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param options.shape\r\n * @param options.radius\r\n * @param options.tessellation\r\n * @param options.clip\r\n * @param options.arc\r\n * @param options.closed\r\n * @param options.updatable\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param options.cap\r\n * @param options.invertUV\r\n * @param scene defines the hosting scene\r\n * @returns the lathe mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param#lathe\r\n */\r\nexport function CreateLathe(\r\n name: string,\r\n options: {\r\n shape: Vector3[];\r\n radius?: number;\r\n tessellation?: number;\r\n clip?: number;\r\n arc?: number;\r\n closed?: boolean;\r\n updatable?: boolean;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n cap?: number;\r\n invertUV?: boolean;\r\n },\r\n scene: Nullable = null\r\n): Mesh {\r\n const arc: number = options.arc ? (options.arc <= 0 || options.arc > 1 ? 1.0 : options.arc) : 1.0;\r\n const closed: boolean = options.closed === undefined ? true : options.closed;\r\n const shape = options.shape;\r\n const radius = options.radius || 1;\r\n const tessellation = options.tessellation || 64;\r\n const clip = options.clip || 0;\r\n const updatable = options.updatable;\r\n const sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n const cap = options.cap || Mesh.NO_CAP;\r\n const pi2 = Math.PI * 2;\r\n const paths = new Array();\r\n const invertUV = options.invertUV || false;\r\n\r\n let i = 0;\r\n let p = 0;\r\n const step = (pi2 / tessellation) * arc;\r\n let rotated;\r\n let path: Array;\r\n for (i = 0; i <= tessellation - clip; i++) {\r\n path = [];\r\n if (cap == Mesh.CAP_START || cap == Mesh.CAP_ALL) {\r\n path.push(new Vector3(0, shape[0].y, 0));\r\n path.push(new Vector3(Math.cos(i * step) * shape[0].x * radius, shape[0].y, Math.sin(i * step) * shape[0].x * radius));\r\n }\r\n for (p = 0; p < shape.length; p++) {\r\n rotated = new Vector3(Math.cos(i * step) * shape[p].x * radius, shape[p].y, Math.sin(i * step) * shape[p].x * radius);\r\n path.push(rotated);\r\n }\r\n if (cap == Mesh.CAP_END || cap == Mesh.CAP_ALL) {\r\n path.push(new Vector3(Math.cos(i * step) * shape[shape.length - 1].x * radius, shape[shape.length - 1].y, Math.sin(i * step) * shape[shape.length - 1].x * radius));\r\n path.push(new Vector3(0, shape[shape.length - 1].y, 0));\r\n }\r\n paths.push(path);\r\n }\r\n\r\n // lathe ribbon\r\n const lathe = CreateRibbon(\r\n name,\r\n { pathArray: paths, closeArray: closed, sideOrientation: sideOrientation, updatable: updatable, invertUV: invertUV, frontUVs: options.frontUVs, backUVs: options.backUVs },\r\n scene\r\n );\r\n return lathe;\r\n}\r\n\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated use the function direction from the module\r\n */\r\nexport const LatheBuilder = {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreateLathe,\r\n};\r\n\r\nMesh.CreateLathe = (name: string, shape: Vector3[], radius: number, tessellation: number, scene: Scene, updatable?: boolean, sideOrientation?: number): Mesh => {\r\n const options = {\r\n shape: shape,\r\n radius: radius,\r\n tessellation: tessellation,\r\n sideOrientation: sideOrientation,\r\n updatable: updatable,\r\n };\r\n\r\n return CreateLathe(name, options, scene);\r\n};\r\n","import type { Nullable } from \"../../types\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { Vector4 } from \"../../Maths/math.vector\";\r\nimport { Vector3, TmpVectors, Matrix } from \"../../Maths/math.vector\";\r\nimport { Mesh } from \"../mesh\";\r\nimport { CreateRibbon } from \"./ribbonBuilder\";\r\nimport { Path3D } from \"../../Maths/math.path\";\r\n\r\n/**\r\n * Creates a tube mesh.\r\n * The tube is a parametric shape. It has no predefined shape. Its final shape will depend on the input parameters\r\n * * The parameter `path` is a required array of successive Vector3. It is the curve used as the axis of the tube\r\n * * The parameter `radius` (positive float, default 1) sets the tube radius size\r\n * * The parameter `tessellation` (positive float, default 64) is the number of sides on the tubular surface\r\n * * The parameter `radiusFunction` (javascript function, default null) is a vanilla javascript function. If it is not null, it overrides the parameter `radius`\r\n * * This function is called on each point of the tube path and is passed the index `i` of the i-th point and the distance of this point from the first point of the path. It must return a radius value (positive float)\r\n * * The parameter `arc` (positive float, maximum 1, default 1) is the ratio to apply to the tube circumference : 2 x PI x arc\r\n * * The parameter `cap` sets the way the extruded shape is capped. Possible values : BABYLON.Mesh.NO_CAP (default), BABYLON.Mesh.CAP_START, BABYLON.Mesh.CAP_END, BABYLON.Mesh.CAP_ALL\r\n * * The optional parameter `instance` is an instance of an existing Tube object to be updated with the passed `pathArray` parameter. The `path`Array HAS to have the SAME number of points as the previous one: https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#tube\r\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\r\n * * The optional parameter `invertUV` (boolean, default false) swaps in the geometry the U and V coordinates to apply a texture\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created. The NUMBER of points CAN'T CHANGE, only their positions.\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param options.path\r\n * @param options.radius\r\n * @param options.tessellation\r\n * @param options.radiusFunction\r\n * @param options.cap\r\n * @param options.arc\r\n * @param options.updatable\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param options.instance\r\n * @param options.invertUV\r\n * @param scene defines the hosting scene\r\n * @returns the tube mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#tube\r\n */\r\nexport function CreateTube(\r\n name: string,\r\n options: {\r\n path: Vector3[];\r\n radius?: number;\r\n tessellation?: number;\r\n radiusFunction?: { (i: number, distance: number): number };\r\n cap?: number;\r\n arc?: number;\r\n updatable?: boolean;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n instance?: Mesh;\r\n invertUV?: boolean;\r\n },\r\n scene: Nullable = null\r\n): Mesh {\r\n const path = options.path;\r\n let instance = options.instance;\r\n let radius = 1.0;\r\n\r\n if (options.radius !== undefined) {\r\n radius = options.radius;\r\n } else if (instance) {\r\n radius = instance._creationDataStorage!.radius;\r\n }\r\n\r\n const tessellation = options.tessellation || 64 | 0;\r\n const radiusFunction = options.radiusFunction || null;\r\n let cap = options.cap || Mesh.NO_CAP;\r\n const invertUV = options.invertUV || false;\r\n const updatable = options.updatable;\r\n const sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n options.arc = options.arc && (options.arc <= 0.0 || options.arc > 1.0) ? 1.0 : options.arc || 1.0;\r\n\r\n // tube geometry\r\n const tubePathArray = (\r\n path: Vector3[],\r\n path3D: Path3D,\r\n circlePaths: Vector3[][],\r\n radius: number,\r\n tessellation: number,\r\n radiusFunction: Nullable<{ (i: number, distance: number): number }>,\r\n cap: number,\r\n arc: number\r\n ) => {\r\n const tangents = path3D.getTangents();\r\n const normals = path3D.getNormals();\r\n const distances = path3D.getDistances();\r\n const pi2 = Math.PI * 2;\r\n const step = (pi2 / tessellation) * arc;\r\n const returnRadius: { (i: number, distance: number): number } = () => radius;\r\n const radiusFunctionFinal: { (i: number, distance: number): number } = radiusFunction || returnRadius;\r\n\r\n let circlePath: Vector3[];\r\n let rad: number;\r\n let normal: Vector3;\r\n let rotated: Vector3;\r\n const rotationMatrix: Matrix = TmpVectors.Matrix[0];\r\n let index = cap === Mesh.NO_CAP || cap === Mesh.CAP_END ? 0 : 2;\r\n for (let i = 0; i < path.length; i++) {\r\n rad = radiusFunctionFinal(i, distances[i]); // current radius\r\n circlePath = Array(); // current circle array\r\n normal = normals[i]; // current normal\r\n for (let t = 0; t < tessellation; t++) {\r\n Matrix.RotationAxisToRef(tangents[i], step * t, rotationMatrix);\r\n rotated = circlePath[t] ? circlePath[t] : Vector3.Zero();\r\n Vector3.TransformCoordinatesToRef(normal, rotationMatrix, rotated);\r\n rotated.scaleInPlace(rad).addInPlace(path[i]);\r\n circlePath[t] = rotated;\r\n }\r\n circlePaths[index] = circlePath;\r\n index++;\r\n }\r\n // cap\r\n const capPath = (nbPoints: number, pathIndex: number): Array => {\r\n const pointCap = Array();\r\n for (let i = 0; i < nbPoints; i++) {\r\n pointCap.push(path[pathIndex]);\r\n }\r\n return pointCap;\r\n };\r\n switch (cap) {\r\n case Mesh.NO_CAP:\r\n break;\r\n case Mesh.CAP_START:\r\n circlePaths[0] = capPath(tessellation, 0);\r\n circlePaths[1] = circlePaths[2].slice(0);\r\n break;\r\n case Mesh.CAP_END:\r\n circlePaths[index] = circlePaths[index - 1].slice(0);\r\n circlePaths[index + 1] = capPath(tessellation, path.length - 1);\r\n break;\r\n case Mesh.CAP_ALL:\r\n circlePaths[0] = capPath(tessellation, 0);\r\n circlePaths[1] = circlePaths[2].slice(0);\r\n circlePaths[index] = circlePaths[index - 1].slice(0);\r\n circlePaths[index + 1] = capPath(tessellation, path.length - 1);\r\n break;\r\n default:\r\n break;\r\n }\r\n return circlePaths;\r\n };\r\n\r\n let path3D;\r\n let pathArray;\r\n if (instance) {\r\n // tube update\r\n const storage = instance._creationDataStorage!;\r\n const arc = options.arc || storage.arc;\r\n path3D = storage.path3D.update(path);\r\n pathArray = tubePathArray(path, path3D, storage.pathArray, radius, storage.tessellation, radiusFunction, storage.cap, arc);\r\n instance = CreateRibbon(\"\", { pathArray: pathArray, instance: instance });\r\n // Update mode, no need to recreate the storage.\r\n storage.path3D = path3D;\r\n storage.pathArray = pathArray;\r\n storage.arc = arc;\r\n storage.radius = radius;\r\n\r\n return instance;\r\n }\r\n\r\n // tube creation\r\n path3D = new Path3D(path);\r\n const newPathArray = new Array>();\r\n cap = cap < 0 || cap > 3 ? 0 : cap;\r\n pathArray = tubePathArray(path, path3D, newPathArray, radius, tessellation, radiusFunction, cap, options.arc);\r\n const tube = CreateRibbon(\r\n name,\r\n {\r\n pathArray: pathArray,\r\n closePath: true,\r\n closeArray: false,\r\n updatable: updatable,\r\n sideOrientation: sideOrientation,\r\n invertUV: invertUV,\r\n frontUVs: options.frontUVs,\r\n backUVs: options.backUVs,\r\n },\r\n scene\r\n );\r\n tube._creationDataStorage!.pathArray = pathArray;\r\n tube._creationDataStorage!.path3D = path3D;\r\n tube._creationDataStorage!.tessellation = tessellation;\r\n tube._creationDataStorage!.cap = cap;\r\n tube._creationDataStorage!.arc = options.arc;\r\n tube._creationDataStorage!.radius = radius;\r\n\r\n return tube;\r\n}\r\n\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated use CreateTube directly\r\n */\r\nexport const TubeBuilder = {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreateTube,\r\n};\r\n\r\nMesh.CreateTube = (\r\n name: string,\r\n path: Vector3[],\r\n radius: number,\r\n tessellation: number,\r\n radiusFunction: { (i: number, distance: number): number },\r\n cap: number,\r\n scene: Scene,\r\n updatable?: boolean,\r\n sideOrientation?: number,\r\n instance?: Mesh\r\n): Mesh => {\r\n const options = {\r\n path: path,\r\n radius: radius,\r\n tessellation: tessellation,\r\n radiusFunction: radiusFunction,\r\n arc: 1,\r\n cap: cap,\r\n updatable: updatable,\r\n sideOrientation: sideOrientation,\r\n instance: instance,\r\n };\r\n return CreateTube(name, options, scene);\r\n};\r\n","import type { Scene } from \"../../scene\";\r\nimport { Vector4 } from \"../../Maths/math.vector\";\r\nimport { Color4 } from \"../../Maths/math.color\";\r\nimport { Mesh } from \"../mesh\";\r\nimport { VertexData } from \"../mesh.vertexData\";\r\nimport type { Nullable } from \"../../types\";\r\nimport { CompatibilityOptions } from \"../../Compat/compatibilityOptions\";\r\n\r\n// inspired from // http://stemkoski.github.io/Three.js/Polyhedra.html\r\n/**\r\n * Creates the VertexData for a Polyhedron\r\n * @param options an object used to set the following optional parameters for the polyhedron, required but can be empty\r\n * * type provided types are:\r\n * * 0 : Tetrahedron, 1 : Octahedron, 2 : Dodecahedron, 3 : Icosahedron, 4 : Rhombicuboctahedron, 5 : Triangular Prism, 6 : Pentagonal Prism, 7 : Hexagonal Prism, 8 : Square Pyramid (J1)\r\n * * 9 : Pentagonal Pyramid (J2), 10 : Triangular Dipyramid (J12), 11 : Pentagonal Dipyramid (J13), 12 : Elongated Square Dipyramid (J15), 13 : Elongated Pentagonal Dipyramid (J16), 14 : Elongated Pentagonal Cupola (J20)\r\n * * size the size of the IcoSphere, optional default 1\r\n * * sizeX allows stretching in the x direction, optional, default size\r\n * * sizeY allows stretching in the y direction, optional, default size\r\n * * sizeZ allows stretching in the z direction, optional, default size\r\n * * custom a number that overwrites the type to create from an extended set of polyhedron from https://www.babylonjs-playground.com/#21QRSK#15 with minimised editor\r\n * * faceUV an array of Vector4 elements used to set different images to the top, rings and bottom respectively\r\n * * faceColors an array of Color3 elements used to set different colors to the top, rings and bottom respectively\r\n * * flat when true creates a flat shaded mesh, optional, default true\r\n * * subdivisions increasing the subdivisions increases the number of faces, optional, default 4\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.type\r\n * @param options.size\r\n * @param options.sizeX\r\n * @param options.sizeY\r\n * @param options.sizeZ\r\n * @param options.custom\r\n * @param options.faceUV\r\n * @param options.faceColors\r\n * @param options.flat\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @returns the VertexData of the Polyhedron\r\n */\r\nexport function CreatePolyhedronVertexData(options: {\r\n type?: number;\r\n size?: number;\r\n sizeX?: number;\r\n sizeY?: number;\r\n sizeZ?: number;\r\n custom?: any;\r\n faceUV?: Vector4[];\r\n faceColors?: Color4[];\r\n flat?: boolean;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n}): VertexData {\r\n // provided polyhedron types :\r\n // 0 : Tetrahedron, 1 : Octahedron, 2 : Dodecahedron, 3 : Icosahedron, 4 : Rhombicuboctahedron, 5 : Triangular Prism, 6 : Pentagonal Prism, 7 : Hexagonal Prism, 8 : Square Pyramid (J1)\r\n // 9 : Pentagonal Pyramid (J2), 10 : Triangular Dipyramid (J12), 11 : Pentagonal Dipyramid (J13), 12 : Elongated Square Dipyramid (J15), 13 : Elongated Pentagonal Dipyramid (J16), 14 : Elongated Pentagonal Cupola (J20)\r\n const polyhedra: { vertex: number[][]; face: number[][] }[] = [];\r\n polyhedra[0] = {\r\n vertex: [\r\n [0, 0, 1.732051],\r\n [1.632993, 0, -0.5773503],\r\n [-0.8164966, 1.414214, -0.5773503],\r\n [-0.8164966, -1.414214, -0.5773503],\r\n ],\r\n face: [\r\n [0, 1, 2],\r\n [0, 2, 3],\r\n [0, 3, 1],\r\n [1, 3, 2],\r\n ],\r\n };\r\n polyhedra[1] = {\r\n vertex: [\r\n [0, 0, 1.414214],\r\n [1.414214, 0, 0],\r\n [0, 1.414214, 0],\r\n [-1.414214, 0, 0],\r\n [0, -1.414214, 0],\r\n [0, 0, -1.414214],\r\n ],\r\n face: [\r\n [0, 1, 2],\r\n [0, 2, 3],\r\n [0, 3, 4],\r\n [0, 4, 1],\r\n [1, 4, 5],\r\n [1, 5, 2],\r\n [2, 5, 3],\r\n [3, 5, 4],\r\n ],\r\n };\r\n polyhedra[2] = {\r\n vertex: [\r\n [0, 0, 1.070466],\r\n [0.7136442, 0, 0.7978784],\r\n [-0.3568221, 0.618034, 0.7978784],\r\n [-0.3568221, -0.618034, 0.7978784],\r\n [0.7978784, 0.618034, 0.3568221],\r\n [0.7978784, -0.618034, 0.3568221],\r\n [-0.9341724, 0.381966, 0.3568221],\r\n [0.1362939, 1, 0.3568221],\r\n [0.1362939, -1, 0.3568221],\r\n [-0.9341724, -0.381966, 0.3568221],\r\n [0.9341724, 0.381966, -0.3568221],\r\n [0.9341724, -0.381966, -0.3568221],\r\n [-0.7978784, 0.618034, -0.3568221],\r\n [-0.1362939, 1, -0.3568221],\r\n [-0.1362939, -1, -0.3568221],\r\n [-0.7978784, -0.618034, -0.3568221],\r\n [0.3568221, 0.618034, -0.7978784],\r\n [0.3568221, -0.618034, -0.7978784],\r\n [-0.7136442, 0, -0.7978784],\r\n [0, 0, -1.070466],\r\n ],\r\n face: [\r\n [0, 1, 4, 7, 2],\r\n [0, 2, 6, 9, 3],\r\n [0, 3, 8, 5, 1],\r\n [1, 5, 11, 10, 4],\r\n [2, 7, 13, 12, 6],\r\n [3, 9, 15, 14, 8],\r\n [4, 10, 16, 13, 7],\r\n [5, 8, 14, 17, 11],\r\n [6, 12, 18, 15, 9],\r\n [10, 11, 17, 19, 16],\r\n [12, 13, 16, 19, 18],\r\n [14, 15, 18, 19, 17],\r\n ],\r\n };\r\n polyhedra[3] = {\r\n vertex: [\r\n [0, 0, 1.175571],\r\n [1.051462, 0, 0.5257311],\r\n [0.3249197, 1, 0.5257311],\r\n [-0.8506508, 0.618034, 0.5257311],\r\n [-0.8506508, -0.618034, 0.5257311],\r\n [0.3249197, -1, 0.5257311],\r\n [0.8506508, 0.618034, -0.5257311],\r\n [0.8506508, -0.618034, -0.5257311],\r\n [-0.3249197, 1, -0.5257311],\r\n [-1.051462, 0, -0.5257311],\r\n [-0.3249197, -1, -0.5257311],\r\n [0, 0, -1.175571],\r\n ],\r\n face: [\r\n [0, 1, 2],\r\n [0, 2, 3],\r\n [0, 3, 4],\r\n [0, 4, 5],\r\n [0, 5, 1],\r\n [1, 5, 7],\r\n [1, 7, 6],\r\n [1, 6, 2],\r\n [2, 6, 8],\r\n [2, 8, 3],\r\n [3, 8, 9],\r\n [3, 9, 4],\r\n [4, 9, 10],\r\n [4, 10, 5],\r\n [5, 10, 7],\r\n [6, 7, 11],\r\n [6, 11, 8],\r\n [7, 10, 11],\r\n [8, 11, 9],\r\n [9, 11, 10],\r\n ],\r\n };\r\n polyhedra[4] = {\r\n vertex: [\r\n [0, 0, 1.070722],\r\n [0.7148135, 0, 0.7971752],\r\n [-0.104682, 0.7071068, 0.7971752],\r\n [-0.6841528, 0.2071068, 0.7971752],\r\n [-0.104682, -0.7071068, 0.7971752],\r\n [0.6101315, 0.7071068, 0.5236279],\r\n [1.04156, 0.2071068, 0.1367736],\r\n [0.6101315, -0.7071068, 0.5236279],\r\n [-0.3574067, 1, 0.1367736],\r\n [-0.7888348, -0.5, 0.5236279],\r\n [-0.9368776, 0.5, 0.1367736],\r\n [-0.3574067, -1, 0.1367736],\r\n [0.3574067, 1, -0.1367736],\r\n [0.9368776, -0.5, -0.1367736],\r\n [0.7888348, 0.5, -0.5236279],\r\n [0.3574067, -1, -0.1367736],\r\n [-0.6101315, 0.7071068, -0.5236279],\r\n [-1.04156, -0.2071068, -0.1367736],\r\n [-0.6101315, -0.7071068, -0.5236279],\r\n [0.104682, 0.7071068, -0.7971752],\r\n [0.6841528, -0.2071068, -0.7971752],\r\n [0.104682, -0.7071068, -0.7971752],\r\n [-0.7148135, 0, -0.7971752],\r\n [0, 0, -1.070722],\r\n ],\r\n face: [\r\n [0, 2, 3],\r\n [1, 6, 5],\r\n [4, 9, 11],\r\n [7, 15, 13],\r\n [8, 16, 10],\r\n [12, 14, 19],\r\n [17, 22, 18],\r\n [20, 21, 23],\r\n [0, 1, 5, 2],\r\n [0, 3, 9, 4],\r\n [0, 4, 7, 1],\r\n [1, 7, 13, 6],\r\n [2, 5, 12, 8],\r\n [2, 8, 10, 3],\r\n [3, 10, 17, 9],\r\n [4, 11, 15, 7],\r\n [5, 6, 14, 12],\r\n [6, 13, 20, 14],\r\n [8, 12, 19, 16],\r\n [9, 17, 18, 11],\r\n [10, 16, 22, 17],\r\n [11, 18, 21, 15],\r\n [13, 15, 21, 20],\r\n [14, 20, 23, 19],\r\n [16, 19, 23, 22],\r\n [18, 22, 23, 21],\r\n ],\r\n };\r\n polyhedra[5] = {\r\n vertex: [\r\n [0, 0, 1.322876],\r\n [1.309307, 0, 0.1889822],\r\n [-0.9819805, 0.8660254, 0.1889822],\r\n [0.1636634, -1.299038, 0.1889822],\r\n [0.3273268, 0.8660254, -0.9449112],\r\n [-0.8183171, -0.4330127, -0.9449112],\r\n ],\r\n face: [\r\n [0, 3, 1],\r\n [2, 4, 5],\r\n [0, 1, 4, 2],\r\n [0, 2, 5, 3],\r\n [1, 3, 5, 4],\r\n ],\r\n };\r\n polyhedra[6] = {\r\n vertex: [\r\n [0, 0, 1.159953],\r\n [1.013464, 0, 0.5642542],\r\n [-0.3501431, 0.9510565, 0.5642542],\r\n [-0.7715208, -0.6571639, 0.5642542],\r\n [0.6633206, 0.9510565, -0.03144481],\r\n [0.8682979, -0.6571639, -0.3996071],\r\n [-1.121664, 0.2938926, -0.03144481],\r\n [-0.2348831, -1.063314, -0.3996071],\r\n [0.5181548, 0.2938926, -0.9953061],\r\n [-0.5850262, -0.112257, -0.9953061],\r\n ],\r\n face: [\r\n [0, 1, 4, 2],\r\n [0, 2, 6, 3],\r\n [1, 5, 8, 4],\r\n [3, 6, 9, 7],\r\n [5, 7, 9, 8],\r\n [0, 3, 7, 5, 1],\r\n [2, 4, 8, 9, 6],\r\n ],\r\n };\r\n polyhedra[7] = {\r\n vertex: [\r\n [0, 0, 1.118034],\r\n [0.8944272, 0, 0.6708204],\r\n [-0.2236068, 0.8660254, 0.6708204],\r\n [-0.7826238, -0.4330127, 0.6708204],\r\n [0.6708204, 0.8660254, 0.2236068],\r\n [1.006231, -0.4330127, -0.2236068],\r\n [-1.006231, 0.4330127, 0.2236068],\r\n [-0.6708204, -0.8660254, -0.2236068],\r\n [0.7826238, 0.4330127, -0.6708204],\r\n [0.2236068, -0.8660254, -0.6708204],\r\n [-0.8944272, 0, -0.6708204],\r\n [0, 0, -1.118034],\r\n ],\r\n face: [\r\n [0, 1, 4, 2],\r\n [0, 2, 6, 3],\r\n [1, 5, 8, 4],\r\n [3, 6, 10, 7],\r\n [5, 9, 11, 8],\r\n [7, 10, 11, 9],\r\n [0, 3, 7, 9, 5, 1],\r\n [2, 4, 8, 11, 10, 6],\r\n ],\r\n };\r\n polyhedra[8] = {\r\n vertex: [\r\n [-0.729665, 0.670121, 0.319155],\r\n [-0.655235, -0.29213, -0.754096],\r\n [-0.093922, -0.607123, 0.537818],\r\n [0.702196, 0.595691, 0.485187],\r\n [0.776626, -0.36656, -0.588064],\r\n ],\r\n face: [\r\n [1, 4, 2],\r\n [0, 1, 2],\r\n [3, 0, 2],\r\n [4, 3, 2],\r\n [4, 1, 0, 3],\r\n ],\r\n };\r\n polyhedra[9] = {\r\n vertex: [\r\n [-0.868849, -0.100041, 0.61257],\r\n [-0.329458, 0.976099, 0.28078],\r\n [-0.26629, -0.013796, -0.477654],\r\n [-0.13392, -1.034115, 0.229829],\r\n [0.738834, 0.707117, -0.307018],\r\n [0.859683, -0.535264, -0.338508],\r\n ],\r\n face: [\r\n [3, 0, 2],\r\n [5, 3, 2],\r\n [4, 5, 2],\r\n [1, 4, 2],\r\n [0, 1, 2],\r\n [0, 3, 5, 4, 1],\r\n ],\r\n };\r\n polyhedra[10] = {\r\n vertex: [\r\n [-0.610389, 0.243975, 0.531213],\r\n [-0.187812, -0.48795, -0.664016],\r\n [-0.187812, 0.9759, -0.664016],\r\n [0.187812, -0.9759, 0.664016],\r\n [0.798201, 0.243975, 0.132803],\r\n ],\r\n face: [\r\n [1, 3, 0],\r\n [3, 4, 0],\r\n [3, 1, 4],\r\n [0, 2, 1],\r\n [0, 4, 2],\r\n [2, 4, 1],\r\n ],\r\n };\r\n polyhedra[11] = {\r\n vertex: [\r\n [-1.028778, 0.392027, -0.048786],\r\n [-0.640503, -0.646161, 0.621837],\r\n [-0.125162, -0.395663, -0.540059],\r\n [0.004683, 0.888447, -0.651988],\r\n [0.125161, 0.395663, 0.540059],\r\n [0.632925, -0.791376, 0.433102],\r\n [1.031672, 0.157063, -0.354165],\r\n ],\r\n face: [\r\n [3, 2, 0],\r\n [2, 1, 0],\r\n [2, 5, 1],\r\n [0, 4, 3],\r\n [0, 1, 4],\r\n [4, 1, 5],\r\n [2, 3, 6],\r\n [3, 4, 6],\r\n [5, 2, 6],\r\n [4, 5, 6],\r\n ],\r\n };\r\n polyhedra[12] = {\r\n vertex: [\r\n [-0.669867, 0.334933, -0.529576],\r\n [-0.669867, 0.334933, 0.529577],\r\n [-0.4043, 1.212901, 0],\r\n [-0.334933, -0.669867, -0.529576],\r\n [-0.334933, -0.669867, 0.529577],\r\n [0.334933, 0.669867, -0.529576],\r\n [0.334933, 0.669867, 0.529577],\r\n [0.4043, -1.212901, 0],\r\n [0.669867, -0.334933, -0.529576],\r\n [0.669867, -0.334933, 0.529577],\r\n ],\r\n face: [\r\n [8, 9, 7],\r\n [6, 5, 2],\r\n [3, 8, 7],\r\n [5, 0, 2],\r\n [4, 3, 7],\r\n [0, 1, 2],\r\n [9, 4, 7],\r\n [1, 6, 2],\r\n [9, 8, 5, 6],\r\n [8, 3, 0, 5],\r\n [3, 4, 1, 0],\r\n [4, 9, 6, 1],\r\n ],\r\n };\r\n polyhedra[13] = {\r\n vertex: [\r\n [-0.931836, 0.219976, -0.264632],\r\n [-0.636706, 0.318353, 0.692816],\r\n [-0.613483, -0.735083, -0.264632],\r\n [-0.326545, 0.979634, 0],\r\n [-0.318353, -0.636706, 0.692816],\r\n [-0.159176, 0.477529, -0.856368],\r\n [0.159176, -0.477529, -0.856368],\r\n [0.318353, 0.636706, 0.692816],\r\n [0.326545, -0.979634, 0],\r\n [0.613482, 0.735082, -0.264632],\r\n [0.636706, -0.318353, 0.692816],\r\n [0.931835, -0.219977, -0.264632],\r\n ],\r\n face: [\r\n [11, 10, 8],\r\n [7, 9, 3],\r\n [6, 11, 8],\r\n [9, 5, 3],\r\n [2, 6, 8],\r\n [5, 0, 3],\r\n [4, 2, 8],\r\n [0, 1, 3],\r\n [10, 4, 8],\r\n [1, 7, 3],\r\n [10, 11, 9, 7],\r\n [11, 6, 5, 9],\r\n [6, 2, 0, 5],\r\n [2, 4, 1, 0],\r\n [4, 10, 7, 1],\r\n ],\r\n };\r\n polyhedra[14] = {\r\n vertex: [\r\n [-0.93465, 0.300459, -0.271185],\r\n [-0.838689, -0.260219, -0.516017],\r\n [-0.711319, 0.717591, 0.128359],\r\n [-0.710334, -0.156922, 0.080946],\r\n [-0.599799, 0.556003, -0.725148],\r\n [-0.503838, -0.004675, -0.969981],\r\n [-0.487004, 0.26021, 0.48049],\r\n [-0.460089, -0.750282, -0.512622],\r\n [-0.376468, 0.973135, -0.325605],\r\n [-0.331735, -0.646985, 0.084342],\r\n [-0.254001, 0.831847, 0.530001],\r\n [-0.125239, -0.494738, -0.966586],\r\n [0.029622, 0.027949, 0.730817],\r\n [0.056536, -0.982543, -0.262295],\r\n [0.08085, 1.087391, 0.076037],\r\n [0.125583, -0.532729, 0.485984],\r\n [0.262625, 0.599586, 0.780328],\r\n [0.391387, -0.726999, -0.716259],\r\n [0.513854, -0.868287, 0.139347],\r\n [0.597475, 0.85513, 0.326364],\r\n [0.641224, 0.109523, 0.783723],\r\n [0.737185, -0.451155, 0.538891],\r\n [0.848705, -0.612742, -0.314616],\r\n [0.976075, 0.365067, 0.32976],\r\n [1.072036, -0.19561, 0.084927],\r\n ],\r\n face: [\r\n [15, 18, 21],\r\n [12, 20, 16],\r\n [6, 10, 2],\r\n [3, 0, 1],\r\n [9, 7, 13],\r\n [2, 8, 4, 0],\r\n [0, 4, 5, 1],\r\n [1, 5, 11, 7],\r\n [7, 11, 17, 13],\r\n [13, 17, 22, 18],\r\n [18, 22, 24, 21],\r\n [21, 24, 23, 20],\r\n [20, 23, 19, 16],\r\n [16, 19, 14, 10],\r\n [10, 14, 8, 2],\r\n [15, 9, 13, 18],\r\n [12, 15, 21, 20],\r\n [6, 12, 16, 10],\r\n [3, 6, 2, 0],\r\n [9, 3, 1, 7],\r\n [9, 15, 12, 6, 3],\r\n [22, 17, 11, 5, 4, 8, 14, 19, 23, 24],\r\n ],\r\n };\r\n\r\n const type: number = options.type && (options.type < 0 || options.type >= polyhedra.length) ? 0 : options.type || 0;\r\n const size = options.size;\r\n const sizeX: number = options.sizeX || size || 1;\r\n const sizeY: number = options.sizeY || size || 1;\r\n const sizeZ: number = options.sizeZ || size || 1;\r\n const data: { vertex: number[][]; face: number[][]; name?: string; category?: string } = options.custom || polyhedra[type];\r\n const nbfaces = data.face.length;\r\n const faceUV = options.faceUV || new Array(nbfaces);\r\n const faceColors = options.faceColors;\r\n const flat = options.flat === undefined ? true : options.flat;\r\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\r\n\r\n const positions = new Array();\r\n const indices = new Array();\r\n const normals = new Array();\r\n const uvs = new Array();\r\n const colors = new Array();\r\n let index = 0;\r\n let faceIdx = 0; // face cursor in the array \"indexes\"\r\n const indexes = new Array();\r\n let i = 0;\r\n let f = 0;\r\n let u: number, v: number, ang: number, x: number, y: number, tmp: number;\r\n\r\n // default face colors and UV if undefined\r\n if (flat) {\r\n for (f = 0; f < nbfaces; f++) {\r\n if (faceColors && faceColors[f] === undefined) {\r\n faceColors[f] = new Color4(1, 1, 1, 1);\r\n }\r\n if (faceUV && faceUV[f] === undefined) {\r\n faceUV[f] = new Vector4(0, 0, 1, 1);\r\n }\r\n }\r\n }\r\n\r\n if (!flat) {\r\n for (i = 0; i < data.vertex.length; i++) {\r\n positions.push(data.vertex[i][0] * sizeX, data.vertex[i][1] * sizeY, data.vertex[i][2] * sizeZ);\r\n uvs.push(0, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 : 0);\r\n }\r\n for (f = 0; f < nbfaces; f++) {\r\n for (i = 0; i < data.face[f].length - 2; i++) {\r\n indices.push(data.face[f][0], data.face[f][i + 2], data.face[f][i + 1]);\r\n }\r\n }\r\n } else {\r\n for (f = 0; f < nbfaces; f++) {\r\n const fl = data.face[f].length; // number of vertices of the current face\r\n ang = (2 * Math.PI) / fl;\r\n x = 0.5 * Math.tan(ang / 2);\r\n y = 0.5;\r\n\r\n // positions, uvs, colors\r\n for (i = 0; i < fl; i++) {\r\n // positions\r\n positions.push(data.vertex[data.face[f][i]][0] * sizeX, data.vertex[data.face[f][i]][1] * sizeY, data.vertex[data.face[f][i]][2] * sizeZ);\r\n indexes.push(index);\r\n index++;\r\n // uvs\r\n u = faceUV[f].x + (faceUV[f].z - faceUV[f].x) * (0.5 + x);\r\n v = faceUV[f].y + (faceUV[f].w - faceUV[f].y) * (y - 0.5);\r\n uvs.push(u, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - v : v);\r\n tmp = x * Math.cos(ang) - y * Math.sin(ang);\r\n y = x * Math.sin(ang) + y * Math.cos(ang);\r\n x = tmp;\r\n // colors\r\n if (faceColors) {\r\n colors.push(faceColors[f].r, faceColors[f].g, faceColors[f].b, faceColors[f].a);\r\n }\r\n }\r\n\r\n // indices from indexes\r\n for (i = 0; i < fl - 2; i++) {\r\n indices.push(indexes[0 + faceIdx], indexes[i + 2 + faceIdx], indexes[i + 1 + faceIdx]);\r\n }\r\n faceIdx += fl;\r\n }\r\n }\r\n\r\n VertexData.ComputeNormals(positions, indices, normals);\r\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\r\n\r\n const vertexData = new VertexData();\r\n vertexData.positions = positions;\r\n vertexData.indices = indices;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n if (faceColors && flat) {\r\n vertexData.colors = colors;\r\n }\r\n return vertexData;\r\n}\r\n\r\n/**\r\n * Creates a polyhedron mesh\r\n * * The parameter `type` (positive integer, max 14, default 0) sets the polyhedron type to build among the 15 embbeded types. Please refer to the type sheet in the tutorial to choose the wanted type\r\n * * The parameter `size` (positive float, default 1) sets the polygon size\r\n * * You can overwrite the `size` on each dimension bu using the parameters `sizeX`, `sizeY` or `sizeZ` (positive floats, default to `size` value)\r\n * * You can build other polyhedron types than the 15 embbeded ones by setting the parameter `custom` (`polyhedronObject`, default null). If you set the parameter `custom`, this overrides the parameter `type`\r\n * * A `polyhedronObject` is a formatted javascript object. You'll find a full file with pre-set polyhedra here : https://github.com/BabylonJS/Extensions/tree/master/Polyhedron\r\n * * You can set the color and the UV of each side of the polyhedron with the parameters `faceColors` (Color4, default `(1, 1, 1, 1)`) and faceUV (Vector4, default `(0, 0, 1, 1)`)\r\n * * To understand how to set `faceUV` or `faceColors`, please read this by considering the right number of faces of your polyhedron, instead of only 6 for the box : https://doc.babylonjs.com/features/featuresDeepDive/materials/using/texturePerBoxFace\r\n * * The parameter `flat` (boolean, default true). If set to false, it gives the polyhedron a single global face, so less vertices and shared normals. In this case, `faceColors` and `faceUV` are ignored\r\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param options.type\r\n * @param options.size\r\n * @param options.sizeX\r\n * @param options.sizeY\r\n * @param options.sizeZ\r\n * @param options.custom\r\n * @param options.faceUV\r\n * @param options.faceColors\r\n * @param options.flat\r\n * @param options.updatable\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param scene defines the hosting scene\r\n * @returns the polyhedron mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/polyhedra\r\n */\r\nexport function CreatePolyhedron(\r\n name: string,\r\n options: {\r\n type?: number;\r\n size?: number;\r\n sizeX?: number;\r\n sizeY?: number;\r\n sizeZ?: number;\r\n custom?: any;\r\n faceUV?: Vector4[];\r\n faceColors?: Color4[];\r\n flat?: boolean;\r\n updatable?: boolean;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n } = {},\r\n scene: Nullable = null\r\n): Mesh {\r\n const polyhedron = new Mesh(name, scene);\r\n\r\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n polyhedron._originalBuilderSideOrientation = options.sideOrientation;\r\n\r\n const vertexData = CreatePolyhedronVertexData(options);\r\n\r\n vertexData.applyToMesh(polyhedron, options.updatable);\r\n\r\n return polyhedron;\r\n}\r\n\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated use the function directly from the module\r\n */\r\nexport const PolyhedronBuilder = {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreatePolyhedron,\r\n};\r\n\r\nVertexData.CreatePolyhedron = CreatePolyhedronVertexData;\r\n\r\nMesh.CreatePolyhedron = (\r\n name: string,\r\n options: {\r\n type?: number;\r\n size?: number;\r\n sizeX?: number;\r\n sizeY?: number;\r\n sizeZ?: number;\r\n custom?: any;\r\n faceUV?: Vector4[];\r\n faceColors?: Color4[];\r\n updatable?: boolean;\r\n sideOrientation?: number;\r\n },\r\n scene: Scene\r\n): Mesh => {\r\n return CreatePolyhedron(name, options, scene);\r\n};\r\n","import type { Scene } from \"../../scene\";\r\nimport type { Vector4 } from \"../../Maths/math.vector\";\r\nimport { Vector3, Vector2 } from \"../../Maths/math.vector\";\r\nimport { Mesh } from \"../mesh\";\r\nimport { VertexData } from \"../mesh.vertexData\";\r\nimport type { Nullable } from \"../../types\";\r\nimport { CompatibilityOptions } from \"../../Compat/compatibilityOptions\";\r\n\r\n/**\r\n * Creates the VertexData of the IcoSphere\r\n * @param options an object used to set the following optional parameters for the IcoSphere, required but can be empty\r\n * * radius the radius of the IcoSphere, optional default 1\r\n * * radiusX allows stretching in the x direction, optional, default radius\r\n * * radiusY allows stretching in the y direction, optional, default radius\r\n * * radiusZ allows stretching in the z direction, optional, default radius\r\n * * flat when true creates a flat shaded mesh, optional, default true\r\n * * subdivisions increasing the subdivisions increases the number of faces, optional, default 4\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.radius\r\n * @param options.radiusX\r\n * @param options.radiusY\r\n * @param options.radiusZ\r\n * @param options.flat\r\n * @param options.subdivisions\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @returns the VertexData of the IcoSphere\r\n */\r\nexport function CreateIcoSphereVertexData(options: {\r\n radius?: number;\r\n radiusX?: number;\r\n radiusY?: number;\r\n radiusZ?: number;\r\n flat?: boolean;\r\n subdivisions?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n}): VertexData {\r\n const sideOrientation = options.sideOrientation || VertexData.DEFAULTSIDE;\r\n const radius = options.radius || 1;\r\n const flat = options.flat === undefined ? true : options.flat;\r\n const subdivisions = (options.subdivisions || 4) | 0;\r\n const radiusX = options.radiusX || radius;\r\n const radiusY = options.radiusY || radius;\r\n const radiusZ = options.radiusZ || radius;\r\n\r\n const t = (1 + Math.sqrt(5)) / 2;\r\n\r\n // 12 vertex x,y,z\r\n const icoVertices = [\r\n -1,\r\n t,\r\n -0,\r\n 1,\r\n t,\r\n 0,\r\n -1,\r\n -t,\r\n 0,\r\n 1,\r\n -t,\r\n 0, // v0-3\r\n 0,\r\n -1,\r\n -t,\r\n 0,\r\n 1,\r\n -t,\r\n 0,\r\n -1,\r\n t,\r\n 0,\r\n 1,\r\n t, // v4-7\r\n t,\r\n 0,\r\n 1,\r\n t,\r\n 0,\r\n -1,\r\n -t,\r\n 0,\r\n 1,\r\n -t,\r\n 0,\r\n -1, // v8-11\r\n ];\r\n\r\n // index of 3 vertex makes a face of icopshere\r\n const ico_indices = [\r\n 0, 11, 5, 0, 5, 1, 0, 1, 7, 0, 7, 10, 12, 22, 23, 1, 5, 20, 5, 11, 4, 23, 22, 13, 22, 18, 6, 7, 1, 8, 14, 21, 4, 14, 4, 2, 16, 13, 6, 15, 6, 19, 3, 8, 9, 4, 21, 5, 13, 17,\r\n 23, 6, 13, 22, 19, 6, 18, 9, 8, 1,\r\n ];\r\n // vertex for uv have aliased position, not for UV\r\n const vertices_unalias_id = [\r\n 0,\r\n 1,\r\n 2,\r\n 3,\r\n 4,\r\n 5,\r\n 6,\r\n 7,\r\n 8,\r\n 9,\r\n 10,\r\n 11,\r\n // vertex alias\r\n 0, // 12: 0 + 12\r\n 2, // 13: 2 + 11\r\n 3, // 14: 3 + 11\r\n 3, // 15: 3 + 12\r\n 3, // 16: 3 + 13\r\n 4, // 17: 4 + 13\r\n 7, // 18: 7 + 11\r\n 8, // 19: 8 + 11\r\n 9, // 20: 9 + 11\r\n 9, // 21: 9 + 12\r\n 10, // 22: A + 12\r\n 11, // 23: B + 12\r\n ];\r\n\r\n // uv as integer step (not pixels !)\r\n const ico_vertexuv = [\r\n 5,\r\n 1,\r\n 3,\r\n 1,\r\n 6,\r\n 4,\r\n 0,\r\n 0, // v0-3\r\n 5,\r\n 3,\r\n 4,\r\n 2,\r\n 2,\r\n 2,\r\n 4,\r\n 0, // v4-7\r\n 2,\r\n 0,\r\n 1,\r\n 1,\r\n 6,\r\n 0,\r\n 6,\r\n 2, // v8-11\r\n // vertex alias (for same vertex on different faces)\r\n 0,\r\n 4, // 12: 0 + 12\r\n 3,\r\n 3, // 13: 2 + 11\r\n 4,\r\n 4, // 14: 3 + 11\r\n 3,\r\n 1, // 15: 3 + 12\r\n 4,\r\n 2, // 16: 3 + 13\r\n 4,\r\n 4, // 17: 4 + 13\r\n 0,\r\n 2, // 18: 7 + 11\r\n 1,\r\n 1, // 19: 8 + 11\r\n 2,\r\n 2, // 20: 9 + 11\r\n 3,\r\n 3, // 21: 9 + 12\r\n 1,\r\n 3, // 22: A + 12\r\n 2,\r\n 4, // 23: B + 12\r\n ];\r\n\r\n // Vertices[0, 1, ...9, A, B] : position on UV plane\r\n // '+' indicate duplicate position to be fixed (3,9:0,2,3,4,7,8,A,B)\r\n // First island of uv mapping\r\n // v = 4h 3+ 2\r\n // v = 3h 9+ 4\r\n // v = 2h 9+ 5 B\r\n // v = 1h 9 1 0\r\n // v = 0h 3 8 7 A\r\n // u = 0 1 2 3 4 5 6 *a\r\n\r\n // Second island of uv mapping\r\n // v = 4h 0+ B+ 4+\r\n // v = 3h A+ 2+\r\n // v = 2h 7+ 6 3+\r\n // v = 1h 8+ 3+\r\n // v = 0h\r\n // u = 0 1 2 3 4 5 6 *a\r\n\r\n // Face layout on texture UV mapping\r\n // ============\r\n // \\ 4 /\\ 16 / ======\r\n // \\ / \\ / /\\ 11 /\r\n // \\/ 7 \\/ / \\ /\r\n // ======= / 10 \\/\r\n // /\\ 17 /\\ =======\r\n // / \\ / \\ \\ 15 /\\\r\n // / 8 \\/ 12 \\ \\ / \\\r\n // ============ \\/ 6 \\\r\n // \\ 18 /\\ ============\r\n // \\ / \\ \\ 5 /\\ 0 /\r\n // \\/ 13 \\ \\ / \\ /\r\n // ======= \\/ 1 \\/\r\n // =============\r\n // /\\ 19 /\\ 2 /\\\r\n // / \\ / \\ / \\\r\n // / 14 \\/ 9 \\/ 3 \\\r\n // ===================\r\n\r\n // uv step is u:1 or 0.5, v:cos(30)=sqrt(3)/2, ratio approx is 84/97\r\n const ustep = 138 / 1024;\r\n const vstep = 239 / 1024;\r\n const uoffset = 60 / 1024;\r\n const voffset = 26 / 1024;\r\n // Second island should have margin, not to touch the first island\r\n // avoid any borderline artefact in pixel rounding\r\n const island_u_offset = -40 / 1024;\r\n const island_v_offset = +20 / 1024;\r\n // face is either island 0 or 1 :\r\n // second island is for faces : [4, 7, 8, 12, 13, 16, 17, 18]\r\n const island = [\r\n 0,\r\n 0,\r\n 0,\r\n 0,\r\n 1, // 0 - 4\r\n 0,\r\n 0,\r\n 1,\r\n 1,\r\n 0, // 5 - 9\r\n 0,\r\n 0,\r\n 1,\r\n 1,\r\n 0, // 10 - 14\r\n 0,\r\n 1,\r\n 1,\r\n 1,\r\n 0, // 15 - 19\r\n ];\r\n\r\n const indices = new Array();\r\n const positions = new Array();\r\n const normals = new Array();\r\n const uvs = new Array();\r\n\r\n let current_indice = 0;\r\n // prepare array of 3 vector (empty) (to be worked in place, shared for each face)\r\n const face_vertex_pos = new Array(3);\r\n const face_vertex_uv = new Array(3);\r\n let v012;\r\n for (v012 = 0; v012 < 3; v012++) {\r\n face_vertex_pos[v012] = Vector3.Zero();\r\n face_vertex_uv[v012] = Vector2.Zero();\r\n }\r\n // create all with normals\r\n for (let face = 0; face < 20; face++) {\r\n // 3 vertex per face\r\n for (v012 = 0; v012 < 3; v012++) {\r\n // look up vertex 0,1,2 to its index in 0 to 11 (or 23 including alias)\r\n const v_id = ico_indices[3 * face + v012];\r\n // vertex have 3D position (x,y,z)\r\n face_vertex_pos[v012].copyFromFloats(\r\n icoVertices[3 * vertices_unalias_id[v_id]],\r\n icoVertices[3 * vertices_unalias_id[v_id] + 1],\r\n icoVertices[3 * vertices_unalias_id[v_id] + 2]\r\n );\r\n // Normalize to get normal\r\n face_vertex_pos[v012].normalize();\r\n\r\n // uv Coordinates from vertex ID\r\n face_vertex_uv[v012].copyFromFloats(\r\n ico_vertexuv[2 * v_id] * ustep + uoffset + island[face] * island_u_offset,\r\n ico_vertexuv[2 * v_id + 1] * vstep + voffset + island[face] * island_v_offset\r\n );\r\n }\r\n\r\n // Subdivide the face (interpolate pos, norm, uv)\r\n // - pos is linear interpolation, then projected to sphere (converge polyhedron to sphere)\r\n // - norm is linear interpolation of vertex corner normal\r\n // (to be checked if better to re-calc from face vertex, or if approximation is OK ??? )\r\n // - uv is linear interpolation\r\n //\r\n // Topology is as below for sub-divide by 2\r\n // vertex shown as v0,v1,v2\r\n // interp index is i1 to progress in range [v0,v1[\r\n // interp index is i2 to progress in range [v0,v2[\r\n // face index as (i1,i2) for /\\ : (i1,i2),(i1+1,i2),(i1,i2+1)\r\n // and (i1,i2)' for \\/ : (i1+1,i2),(i1+1,i2+1),(i1,i2+1)\r\n //\r\n //\r\n // i2 v2\r\n // ^ ^\r\n // / / \\\r\n // / / \\\r\n // / / \\\r\n // / / (0,1) \\\r\n // / #---------\\\r\n // / / \\ (0,0)'/ \\\r\n // / / \\ / \\\r\n // / / \\ / \\\r\n // / / (0,0) \\ / (1,0) \\\r\n // / #---------#---------\\\r\n // v0 v1\r\n //\r\n // --------------------> i1\r\n //\r\n // interp of (i1,i2):\r\n // along i2 : x0=lerp(v0,v2, i2/S) <---> x1=lerp(v1,v2, i2/S)\r\n // along i1 : lerp(x0,x1, i1/(S-i2))\r\n //\r\n // centroid of triangle is needed to get help normal computation\r\n // (c1,c2) are used for centroid location\r\n\r\n const interp_vertex = (i1: number, i2: number, c1: number, c2: number) => {\r\n // vertex is interpolated from\r\n // - face_vertex_pos[0..2]\r\n // - face_vertex_uv[0..2]\r\n const pos_x0 = Vector3.Lerp(face_vertex_pos[0], face_vertex_pos[2], i2 / subdivisions);\r\n const pos_x1 = Vector3.Lerp(face_vertex_pos[1], face_vertex_pos[2], i2 / subdivisions);\r\n const pos_interp = subdivisions === i2 ? face_vertex_pos[2] : Vector3.Lerp(pos_x0, pos_x1, i1 / (subdivisions - i2));\r\n pos_interp.normalize();\r\n\r\n let vertex_normal;\r\n if (flat) {\r\n // in flat mode, recalculate normal as face centroid normal\r\n const centroid_x0 = Vector3.Lerp(face_vertex_pos[0], face_vertex_pos[2], c2 / subdivisions);\r\n const centroid_x1 = Vector3.Lerp(face_vertex_pos[1], face_vertex_pos[2], c2 / subdivisions);\r\n vertex_normal = Vector3.Lerp(centroid_x0, centroid_x1, c1 / (subdivisions - c2));\r\n } else {\r\n // in smooth mode, recalculate normal from each single vertex position\r\n vertex_normal = new Vector3(pos_interp.x, pos_interp.y, pos_interp.z);\r\n }\r\n // Vertex normal need correction due to X,Y,Z radius scaling\r\n vertex_normal.x /= radiusX;\r\n vertex_normal.y /= radiusY;\r\n vertex_normal.z /= radiusZ;\r\n vertex_normal.normalize();\r\n\r\n const uv_x0 = Vector2.Lerp(face_vertex_uv[0], face_vertex_uv[2], i2 / subdivisions);\r\n const uv_x1 = Vector2.Lerp(face_vertex_uv[1], face_vertex_uv[2], i2 / subdivisions);\r\n const uv_interp = subdivisions === i2 ? face_vertex_uv[2] : Vector2.Lerp(uv_x0, uv_x1, i1 / (subdivisions - i2));\r\n positions.push(pos_interp.x * radiusX, pos_interp.y * radiusY, pos_interp.z * radiusZ);\r\n normals.push(vertex_normal.x, vertex_normal.y, vertex_normal.z);\r\n uvs.push(uv_interp.x, CompatibilityOptions.UseOpenGLOrientationForUV ? 1.0 - uv_interp.y : uv_interp.y);\r\n // push each vertex has member of a face\r\n // Same vertex can belong to multiple face, it is pushed multiple time (duplicate vertex are present)\r\n indices.push(current_indice);\r\n current_indice++;\r\n };\r\n\r\n for (let i2 = 0; i2 < subdivisions; i2++) {\r\n for (let i1 = 0; i1 + i2 < subdivisions; i1++) {\r\n // face : (i1,i2) for /\\ :\r\n // interp for : (i1,i2),(i1+1,i2),(i1,i2+1)\r\n interp_vertex(i1, i2, i1 + 1.0 / 3, i2 + 1.0 / 3);\r\n interp_vertex(i1 + 1, i2, i1 + 1.0 / 3, i2 + 1.0 / 3);\r\n interp_vertex(i1, i2 + 1, i1 + 1.0 / 3, i2 + 1.0 / 3);\r\n if (i1 + i2 + 1 < subdivisions) {\r\n // face : (i1,i2)' for \\/ :\r\n // interp for (i1+1,i2),(i1+1,i2+1),(i1,i2+1)\r\n interp_vertex(i1 + 1, i2, i1 + 2.0 / 3, i2 + 2.0 / 3);\r\n interp_vertex(i1 + 1, i2 + 1, i1 + 2.0 / 3, i2 + 2.0 / 3);\r\n interp_vertex(i1, i2 + 1, i1 + 2.0 / 3, i2 + 2.0 / 3);\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Sides\r\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\r\n\r\n // Result\r\n const vertexData = new VertexData();\r\n vertexData.indices = indices;\r\n vertexData.positions = positions;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n return vertexData;\r\n}\r\n\r\n/**\r\n * Creates a sphere based upon an icosahedron with 20 triangular faces which can be subdivided\r\n * * The parameter `radius` sets the radius size (float) of the icosphere (default 1)\r\n * * You can set some different icosphere dimensions, for instance to build an ellipsoid, by using the parameters `radiusX`, `radiusY` and `radiusZ` (all by default have the same value of `radius`)\r\n * * The parameter `subdivisions` sets the number of subdivisions (positive integer, default 4). The more subdivisions, the more faces on the icosphere whatever its size\r\n * * The parameter `flat` (boolean, default true) gives each side its own normals. Set it to false to get a smooth continuous light reflection on the surface\r\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param options.radius\r\n * @param options.radiusX\r\n * @param options.radiusY\r\n * @param options.radiusZ\r\n * @param options.flat\r\n * @param options.subdivisions\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param options.updatable\r\n * @param scene defines the hosting scene\r\n * @returns the icosahedron mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/polyhedra#icosphere\r\n */\r\nexport function CreateIcoSphere(\r\n name: string,\r\n options: {\r\n radius?: number;\r\n radiusX?: number;\r\n radiusY?: number;\r\n radiusZ?: number;\r\n flat?: boolean;\r\n subdivisions?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n updatable?: boolean;\r\n } = {},\r\n scene: Nullable = null\r\n): Mesh {\r\n const sphere = new Mesh(name, scene);\r\n\r\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n sphere._originalBuilderSideOrientation = options.sideOrientation;\r\n\r\n const vertexData = CreateIcoSphereVertexData(options);\r\n\r\n vertexData.applyToMesh(sphere, options.updatable);\r\n\r\n return sphere;\r\n}\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated use the function directly from the module\r\n */\r\nexport const IcoSphereBuilder = {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreateIcoSphere,\r\n};\r\n\r\nVertexData.CreateIcoSphere = CreateIcoSphereVertexData;\r\n\r\nMesh.CreateIcoSphere = (name: string, options: { radius?: number; flat?: boolean; subdivisions?: number; sideOrientation?: number; updatable?: boolean }, scene: Scene): Mesh => {\r\n return CreateIcoSphere(name, options, scene);\r\n};\r\n","import type { Nullable, IndicesArray, FloatArray } from \"../../types\";\r\nimport { Vector3, Matrix, Vector2, TmpVectors } from \"../../Maths/math.vector\";\r\nimport { Scalar } from \"../../Maths/math.scalar\";\r\nimport { Mesh } from \"../mesh\";\r\nimport { VertexBuffer } from \"../../Buffers/buffer\";\r\nimport { VertexData } from \"../mesh.vertexData\";\r\nimport type { AbstractMesh } from \"../abstractMesh\";\r\nimport type { Camera } from \"../../Cameras/camera\";\r\nimport { CompatibilityOptions } from \"../../Compat/compatibilityOptions\";\r\n\r\nconst xpAxis = new Vector3(1, 0, 0);\r\nconst xnAxis = new Vector3(-1, 0, 0);\r\nconst ypAxis = new Vector3(0, 1, 0);\r\nconst ynAxis = new Vector3(0, -1, 0);\r\nconst zpAxis = new Vector3(0, 0, 1);\r\nconst znAxis = new Vector3(0, 0, -1);\r\n\r\n/** @internal */\r\nclass DecalVertex {\r\n constructor(\r\n public position: Vector3 = Vector3.Zero(),\r\n public normal: Vector3 = Vector3.Up(),\r\n public uv: Vector2 = Vector2.Zero(),\r\n public vertexIdx: number = 0,\r\n public vertexIdxForBones: number = 0,\r\n public localPositionOverride: Nullable = null,\r\n public localNormalOverride: Nullable = null,\r\n public matrixIndicesOverride: Nullable = null,\r\n public matrixWeightsOverride: Nullable = null\r\n ) {}\r\n public clone(): DecalVertex {\r\n return new DecalVertex(\r\n this.position.clone(),\r\n this.normal.clone(),\r\n this.uv.clone(),\r\n this.vertexIdx,\r\n this.vertexIdxForBones,\r\n this.localPositionOverride?.slice(),\r\n this.localNormalOverride?.slice(),\r\n this.matrixIndicesOverride?.slice(),\r\n this.matrixWeightsOverride?.slice()\r\n );\r\n }\r\n}\r\n\r\n/**\r\n * Creates a decal mesh.\r\n * A decal is a mesh usually applied as a model onto the surface of another mesh. So don't forget the parameter `sourceMesh` depicting the decal\r\n * * The parameter `position` (Vector3, default `(0, 0, 0)`) sets the position of the decal in World coordinates\r\n * * The parameter `normal` (Vector3, default `Vector3.Up`) sets the normal of the mesh where the decal is applied onto in World coordinates\r\n * * The parameter `size` (Vector3, default `(1, 1, 1)`) sets the decal scaling\r\n * * The parameter `angle` (float in radian, default 0) sets the angle to rotate the decal\r\n * * The parameter `captureUVS` defines if we need to capture the uvs or compute them\r\n * * The parameter `cullBackFaces` defines if the back faces should be removed from the decal mesh\r\n * * The parameter `localMode` defines that the computations should be done with the local mesh coordinates instead of the world space coordinates.\r\n * * Use this mode if you want the decal to be parented to the sourceMesh and move/rotate with it.\r\n * Note: Meshes with morph targets are not supported!\r\n * @param name defines the name of the mesh\r\n * @param sourceMesh defines the mesh where the decal must be applied\r\n * @param options defines the options used to create the mesh\r\n * @param options.position\r\n * @param options.normal\r\n * @param options.size\r\n * @param options.angle\r\n * @param options.captureUVS\r\n * @param options.cullBackFaces\r\n * @param options.localMode\r\n * @returns the decal mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/decals\r\n */\r\nexport function CreateDecal(\r\n name: string,\r\n sourceMesh: AbstractMesh,\r\n options: { position?: Vector3; normal?: Vector3; size?: Vector3; angle?: number; captureUVS?: boolean; cullBackFaces?: boolean; localMode?: boolean }\r\n): Mesh {\r\n const hasSkeleton = !!sourceMesh.skeleton;\r\n const useLocalComputation = options.localMode || hasSkeleton;\r\n const meshHasOverridenMaterial = (sourceMesh as Mesh).overrideMaterialSideOrientation !== null && (sourceMesh as Mesh).overrideMaterialSideOrientation !== undefined;\r\n\r\n const indices = sourceMesh.getIndices();\r\n const positions = hasSkeleton ? sourceMesh.getPositionData(true, true) : sourceMesh.getVerticesData(VertexBuffer.PositionKind);\r\n const normals = hasSkeleton ? sourceMesh.getNormalsData(true, true) : sourceMesh.getVerticesData(VertexBuffer.NormalKind);\r\n const localPositions = useLocalComputation ? (hasSkeleton ? sourceMesh.getVerticesData(VertexBuffer.PositionKind) : positions) : null;\r\n const localNormals = useLocalComputation ? (hasSkeleton ? sourceMesh.getVerticesData(VertexBuffer.NormalKind) : normals) : null;\r\n const uvs = sourceMesh.getVerticesData(VertexBuffer.UVKind);\r\n const matIndices = hasSkeleton ? sourceMesh.getVerticesData(VertexBuffer.MatricesIndicesKind) : null;\r\n const matWeights = hasSkeleton ? sourceMesh.getVerticesData(VertexBuffer.MatricesWeightsKind) : null;\r\n const matIndicesExtra = hasSkeleton ? sourceMesh.getVerticesData(VertexBuffer.MatricesIndicesExtraKind) : null;\r\n const matWeightsExtra = hasSkeleton ? sourceMesh.getVerticesData(VertexBuffer.MatricesWeightsExtraKind) : null;\r\n\r\n const position = options.position || Vector3.Zero();\r\n let normal = options.normal || Vector3.Up();\r\n const size = options.size || Vector3.One();\r\n const angle = options.angle || 0;\r\n\r\n // Getting correct rotation\r\n if (!normal) {\r\n const target = new Vector3(0, 0, 1);\r\n const camera = sourceMesh.getScene().activeCamera;\r\n const cameraWorldTarget = Vector3.TransformCoordinates(target, camera.getWorldMatrix());\r\n\r\n normal = camera.globalPosition.subtract(cameraWorldTarget);\r\n }\r\n\r\n const yaw = -Math.atan2(normal.z, normal.x) - Math.PI / 2;\r\n const len = Math.sqrt(normal.x * normal.x + normal.z * normal.z);\r\n const pitch = Math.atan2(normal.y, len);\r\n\r\n const vertexData = new VertexData();\r\n vertexData.indices = [];\r\n vertexData.positions = [];\r\n vertexData.normals = [];\r\n vertexData.uvs = [];\r\n vertexData.matricesIndices = hasSkeleton ? [] : null;\r\n vertexData.matricesWeights = hasSkeleton ? [] : null;\r\n vertexData.matricesIndicesExtra = matIndicesExtra ? [] : null;\r\n vertexData.matricesWeightsExtra = matWeightsExtra ? [] : null;\r\n\r\n let currentVertexDataIndex = 0;\r\n\r\n const extractDecalVector3 = (indexId: number, transformMatrix: Matrix): DecalVertex => {\r\n const result = new DecalVertex();\r\n if (!indices || !positions || !normals) {\r\n return result;\r\n }\r\n\r\n const vertexId = indices[indexId];\r\n\r\n result.vertexIdx = vertexId * 3;\r\n result.vertexIdxForBones = vertexId * 4;\r\n\r\n // Send vector to decal local world\r\n result.position = new Vector3(positions[vertexId * 3], positions[vertexId * 3 + 1], positions[vertexId * 3 + 2]);\r\n Vector3.TransformCoordinatesToRef(result.position, transformMatrix, result.position);\r\n\r\n // Get normal\r\n result.normal = new Vector3(normals[vertexId * 3], normals[vertexId * 3 + 1], normals[vertexId * 3 + 2]);\r\n Vector3.TransformNormalToRef(result.normal, transformMatrix, result.normal);\r\n\r\n if (options.captureUVS && uvs) {\r\n const v = uvs[vertexId * 2 + 1];\r\n result.uv = new Vector2(uvs[vertexId * 2], CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - v : v);\r\n }\r\n\r\n return result;\r\n };\r\n\r\n const emptyArray = [0, 0, 0, 0];\r\n\r\n // Inspired by https://github.com/mrdoob/three.js/blob/eee231960882f6f3b6113405f524956145148146/examples/js/geometries/DecalGeometry.js\r\n const clip = (vertices: DecalVertex[], axis: Vector3): Nullable => {\r\n if (vertices.length === 0) {\r\n return vertices;\r\n }\r\n\r\n const clipSize = 0.5 * Math.abs(Vector3.Dot(size, axis));\r\n\r\n const indexOf = (arr: FloatArray | number[], val: number, start: number, num: number) => {\r\n for (let i = 0; i < num; ++i) {\r\n if (arr[start + i] === val) {\r\n return start + i;\r\n }\r\n }\r\n return -1;\r\n };\r\n\r\n const clipVertices = (v0: DecalVertex, v1: DecalVertex): DecalVertex => {\r\n const clipFactor = Vector3.GetClipFactor(v0.position, v1.position, axis, clipSize);\r\n\r\n let indices = emptyArray;\r\n let weights = emptyArray;\r\n\r\n if (matIndices && matWeights) {\r\n const mat0Index = v0.matrixIndicesOverride ? 0 : v0.vertexIdxForBones;\r\n const v0Indices = v0.matrixIndicesOverride ?? matIndices;\r\n const v0Weights = v0.matrixWeightsOverride ?? matWeights;\r\n\r\n const mat1Index = v1.matrixIndicesOverride ? 0 : v1.vertexIdxForBones;\r\n const v1Indices = v1.matrixIndicesOverride ?? matIndices;\r\n const v1Weights = v1.matrixWeightsOverride ?? matWeights;\r\n\r\n indices = [0, 0, 0, 0];\r\n weights = [0, 0, 0, 0];\r\n\r\n let index = 0;\r\n for (let i = 0; i < 4; ++i) {\r\n if (v0Weights[mat0Index + i] > 0) {\r\n const idx = indexOf(v1Indices, v0Indices[mat0Index + i], mat1Index, 4);\r\n indices[index] = v0Indices[mat0Index + i];\r\n weights[index] = Scalar.Lerp(v0Weights[mat0Index + i], idx >= 0 ? v1Weights[idx] : 0, clipFactor);\r\n index++;\r\n }\r\n }\r\n\r\n for (let i = 0; i < 4 && index < 4; ++i) {\r\n const ind = v1Indices[mat1Index + i];\r\n if (indexOf(v0Indices, ind, mat0Index, 4) !== -1) continue;\r\n\r\n indices[index] = ind;\r\n weights[index] = Scalar.Lerp(0, v1Weights[mat1Index + i], clipFactor);\r\n index++;\r\n }\r\n\r\n const sumw = weights[0] + weights[1] + weights[2] + weights[3];\r\n\r\n weights[0] /= sumw;\r\n weights[1] /= sumw;\r\n weights[2] /= sumw;\r\n weights[3] /= sumw;\r\n }\r\n\r\n const v0LocalPositionX = v0.localPositionOverride ? v0.localPositionOverride[0] : localPositions?.[v0.vertexIdx] ?? 0;\r\n const v0LocalPositionY = v0.localPositionOverride ? v0.localPositionOverride[1] : localPositions?.[v0.vertexIdx + 1] ?? 0;\r\n const v0LocalPositionZ = v0.localPositionOverride ? v0.localPositionOverride[2] : localPositions?.[v0.vertexIdx + 2] ?? 0;\r\n\r\n const v1LocalPositionX = v1.localPositionOverride ? v1.localPositionOverride[0] : localPositions?.[v1.vertexIdx] ?? 0;\r\n const v1LocalPositionY = v1.localPositionOverride ? v1.localPositionOverride[1] : localPositions?.[v1.vertexIdx + 1] ?? 0;\r\n const v1LocalPositionZ = v1.localPositionOverride ? v1.localPositionOverride[2] : localPositions?.[v1.vertexIdx + 2] ?? 0;\r\n\r\n const v0LocalNormalX = v0.localNormalOverride ? v0.localNormalOverride[0] : localNormals?.[v0.vertexIdx] ?? 0;\r\n const v0LocalNormalY = v0.localNormalOverride ? v0.localNormalOverride[1] : localNormals?.[v0.vertexIdx + 1] ?? 0;\r\n const v0LocalNormalZ = v0.localNormalOverride ? v0.localNormalOverride[2] : localNormals?.[v0.vertexIdx + 2] ?? 0;\r\n\r\n const v1LocalNormalX = v1.localNormalOverride ? v1.localNormalOverride[0] : localNormals?.[v1.vertexIdx] ?? 0;\r\n const v1LocalNormalY = v1.localNormalOverride ? v1.localNormalOverride[1] : localNormals?.[v1.vertexIdx + 1] ?? 0;\r\n const v1LocalNormalZ = v1.localNormalOverride ? v1.localNormalOverride[2] : localNormals?.[v1.vertexIdx + 2] ?? 0;\r\n\r\n const interpNormalX = v0LocalNormalX + (v1LocalNormalX - v0LocalNormalX) * clipFactor;\r\n const interpNormalY = v0LocalNormalY + (v1LocalNormalY - v0LocalNormalY) * clipFactor;\r\n const interpNormalZ = v0LocalNormalZ + (v1LocalNormalZ - v0LocalNormalZ) * clipFactor;\r\n\r\n const norm = Math.sqrt(interpNormalX * interpNormalX + interpNormalY * interpNormalY + interpNormalZ * interpNormalZ);\r\n\r\n return new DecalVertex(\r\n Vector3.Lerp(v0.position, v1.position, clipFactor),\r\n Vector3.Lerp(v0.normal, v1.normal, clipFactor).normalize(),\r\n Vector2.Lerp(v0.uv, v1.uv, clipFactor),\r\n -1,\r\n -1,\r\n localPositions\r\n ? [\r\n v0LocalPositionX + (v1LocalPositionX - v0LocalPositionX) * clipFactor,\r\n v0LocalPositionY + (v1LocalPositionY - v0LocalPositionY) * clipFactor,\r\n v0LocalPositionZ + (v1LocalPositionZ - v0LocalPositionZ) * clipFactor,\r\n ]\r\n : null,\r\n localNormals ? [interpNormalX / norm, interpNormalY / norm, interpNormalZ / norm] : null,\r\n indices,\r\n weights\r\n );\r\n };\r\n\r\n let clipResult: Nullable = null;\r\n\r\n if (vertices.length > 3) {\r\n clipResult = new Array();\r\n }\r\n\r\n for (let index = 0; index < vertices.length; index += 3) {\r\n let total = 0;\r\n let nV1: Nullable = null;\r\n let nV2: Nullable = null;\r\n let nV3: Nullable = null;\r\n let nV4: Nullable = null;\r\n\r\n const d1 = Vector3.Dot(vertices[index].position, axis) - clipSize;\r\n const d2 = Vector3.Dot(vertices[index + 1].position, axis) - clipSize;\r\n const d3 = Vector3.Dot(vertices[index + 2].position, axis) - clipSize;\r\n\r\n const v1Out = d1 > 0;\r\n const v2Out = d2 > 0;\r\n const v3Out = d3 > 0;\r\n\r\n total = (v1Out ? 1 : 0) + (v2Out ? 1 : 0) + (v3Out ? 1 : 0);\r\n\r\n switch (total) {\r\n case 0:\r\n if (vertices.length > 3) {\r\n clipResult!.push(vertices[index]);\r\n clipResult!.push(vertices[index + 1]);\r\n clipResult!.push(vertices[index + 2]);\r\n } else {\r\n clipResult = vertices;\r\n }\r\n break;\r\n case 1:\r\n clipResult = clipResult ?? new Array();\r\n if (v1Out) {\r\n nV1 = vertices[index + 1];\r\n nV2 = vertices[index + 2];\r\n nV3 = clipVertices(vertices[index], nV1);\r\n nV4 = clipVertices(vertices[index], nV2);\r\n }\r\n\r\n if (v2Out) {\r\n nV1 = vertices[index];\r\n nV2 = vertices[index + 2];\r\n nV3 = clipVertices(vertices[index + 1], nV1);\r\n nV4 = clipVertices(vertices[index + 1], nV2);\r\n\r\n clipResult.push(nV3);\r\n clipResult.push(nV2.clone());\r\n clipResult.push(nV1.clone());\r\n\r\n clipResult.push(nV2.clone());\r\n clipResult.push(nV3.clone());\r\n clipResult.push(nV4);\r\n break;\r\n }\r\n if (v3Out) {\r\n nV1 = vertices[index];\r\n nV2 = vertices[index + 1];\r\n nV3 = clipVertices(vertices[index + 2], nV1);\r\n nV4 = clipVertices(vertices[index + 2], nV2);\r\n }\r\n\r\n if (nV1 && nV2 && nV3 && nV4) {\r\n clipResult.push(nV1.clone());\r\n clipResult.push(nV2.clone());\r\n clipResult.push(nV3);\r\n\r\n clipResult.push(nV4);\r\n clipResult.push(nV3.clone());\r\n clipResult.push(nV2.clone());\r\n }\r\n break;\r\n case 2:\r\n clipResult = clipResult ?? new Array();\r\n if (!v1Out) {\r\n nV1 = vertices[index].clone();\r\n nV2 = clipVertices(nV1, vertices[index + 1]);\r\n nV3 = clipVertices(nV1, vertices[index + 2]);\r\n clipResult.push(nV1);\r\n clipResult.push(nV2);\r\n clipResult.push(nV3);\r\n }\r\n if (!v2Out) {\r\n nV1 = vertices[index + 1].clone();\r\n nV2 = clipVertices(nV1, vertices[index + 2]);\r\n nV3 = clipVertices(nV1, vertices[index]);\r\n clipResult.push(nV1);\r\n clipResult.push(nV2);\r\n clipResult.push(nV3);\r\n }\r\n if (!v3Out) {\r\n nV1 = vertices[index + 2].clone();\r\n nV2 = clipVertices(nV1, vertices[index]);\r\n nV3 = clipVertices(nV1, vertices[index + 1]);\r\n clipResult.push(nV1);\r\n clipResult.push(nV2);\r\n clipResult.push(nV3);\r\n }\r\n break;\r\n case 3:\r\n break;\r\n }\r\n }\r\n\r\n return clipResult;\r\n };\r\n\r\n const sourceMeshAsMesh = sourceMesh instanceof Mesh ? sourceMesh : null;\r\n const matrixData = sourceMeshAsMesh?._thinInstanceDataStorage.matrixData;\r\n\r\n const numMatrices = sourceMeshAsMesh?.thinInstanceCount || 1;\r\n const thinInstanceMatrix = TmpVectors.Matrix[0];\r\n\r\n thinInstanceMatrix.copyFrom(Matrix.IdentityReadOnly);\r\n\r\n for (let m = 0; m < numMatrices; ++m) {\r\n if (sourceMeshAsMesh?.hasThinInstances && matrixData) {\r\n const ofst = m * 16;\r\n\r\n thinInstanceMatrix.setRowFromFloats(0, matrixData[ofst + 0], matrixData[ofst + 1], matrixData[ofst + 2], matrixData[ofst + 3]);\r\n thinInstanceMatrix.setRowFromFloats(1, matrixData[ofst + 4], matrixData[ofst + 5], matrixData[ofst + 6], matrixData[ofst + 7]);\r\n thinInstanceMatrix.setRowFromFloats(2, matrixData[ofst + 8], matrixData[ofst + 9], matrixData[ofst + 10], matrixData[ofst + 11]);\r\n thinInstanceMatrix.setRowFromFloats(3, matrixData[ofst + 12], matrixData[ofst + 13], matrixData[ofst + 14], matrixData[ofst + 15]);\r\n }\r\n\r\n // Matrix\r\n const decalWorldMatrix = Matrix.RotationYawPitchRoll(yaw, pitch, angle).multiply(Matrix.Translation(position.x, position.y, position.z));\r\n const inverseDecalWorldMatrix = Matrix.Invert(decalWorldMatrix);\r\n const meshWorldMatrix = sourceMesh.getWorldMatrix();\r\n const transformMatrix = thinInstanceMatrix.multiply(meshWorldMatrix).multiply(inverseDecalWorldMatrix);\r\n\r\n const oneFaceVertices = new Array(3);\r\n\r\n for (let index = 0; index < indices.length; index += 3) {\r\n let faceVertices: Nullable = oneFaceVertices;\r\n\r\n faceVertices[0] = extractDecalVector3(index, transformMatrix);\r\n if (meshHasOverridenMaterial && useLocalComputation) {\r\n faceVertices[1] = extractDecalVector3(index + 2, transformMatrix);\r\n faceVertices[2] = extractDecalVector3(index + 1, transformMatrix);\r\n } else {\r\n faceVertices[1] = extractDecalVector3(index + 1, transformMatrix);\r\n faceVertices[2] = extractDecalVector3(index + 2, transformMatrix);\r\n }\r\n\r\n if (options.cullBackFaces) {\r\n // If all the normals of the vertices of the face are pointing away from the view direction we discard the face.\r\n // As computations are done in the decal coordinate space, the viewDirection is (0,0,1), so when dot(vertexNormal, -viewDirection) <= 0 the vertex is culled\r\n if (-faceVertices[0].normal.z <= 0 && -faceVertices[1].normal.z <= 0 && -faceVertices[2].normal.z <= 0) {\r\n continue;\r\n }\r\n }\r\n\r\n // Clip\r\n faceVertices = clip(faceVertices, xpAxis);\r\n if (!faceVertices) continue;\r\n faceVertices = clip(faceVertices, xnAxis);\r\n if (!faceVertices) continue;\r\n faceVertices = clip(faceVertices, ypAxis);\r\n if (!faceVertices) continue;\r\n faceVertices = clip(faceVertices, ynAxis);\r\n if (!faceVertices) continue;\r\n faceVertices = clip(faceVertices, zpAxis);\r\n if (!faceVertices) continue;\r\n faceVertices = clip(faceVertices, znAxis);\r\n if (!faceVertices) continue;\r\n\r\n // Add UVs and get back to world\r\n for (let vIndex = 0; vIndex < faceVertices.length; vIndex++) {\r\n const vertex = faceVertices[vIndex];\r\n\r\n //TODO check for Int32Array | Uint32Array | Uint16Array\r\n (vertexData.indices).push(currentVertexDataIndex);\r\n if (useLocalComputation) {\r\n if (vertex.localPositionOverride) {\r\n vertexData.positions[currentVertexDataIndex * 3] = vertex.localPositionOverride[0];\r\n vertexData.positions[currentVertexDataIndex * 3 + 1] = vertex.localPositionOverride[1];\r\n vertexData.positions[currentVertexDataIndex * 3 + 2] = vertex.localPositionOverride[2];\r\n } else if (localPositions) {\r\n vertexData.positions[currentVertexDataIndex * 3] = localPositions[vertex.vertexIdx];\r\n vertexData.positions[currentVertexDataIndex * 3 + 1] = localPositions[vertex.vertexIdx + 1];\r\n vertexData.positions[currentVertexDataIndex * 3 + 2] = localPositions[vertex.vertexIdx + 2];\r\n }\r\n if (vertex.localNormalOverride) {\r\n vertexData.normals[currentVertexDataIndex * 3] = vertex.localNormalOverride[0];\r\n vertexData.normals[currentVertexDataIndex * 3 + 1] = vertex.localNormalOverride[1];\r\n vertexData.normals[currentVertexDataIndex * 3 + 2] = vertex.localNormalOverride[2];\r\n } else if (localNormals) {\r\n vertexData.normals[currentVertexDataIndex * 3] = localNormals[vertex.vertexIdx];\r\n vertexData.normals[currentVertexDataIndex * 3 + 1] = localNormals[vertex.vertexIdx + 1];\r\n vertexData.normals[currentVertexDataIndex * 3 + 2] = localNormals[vertex.vertexIdx + 2];\r\n }\r\n } else {\r\n vertex.position.toArray(vertexData.positions, currentVertexDataIndex * 3);\r\n vertex.normal.toArray(vertexData.normals, currentVertexDataIndex * 3);\r\n }\r\n if (vertexData.matricesIndices && vertexData.matricesWeights) {\r\n if (vertex.matrixIndicesOverride) {\r\n vertexData.matricesIndices[currentVertexDataIndex * 4] = vertex.matrixIndicesOverride[0];\r\n vertexData.matricesIndices[currentVertexDataIndex * 4 + 1] = vertex.matrixIndicesOverride[1];\r\n vertexData.matricesIndices[currentVertexDataIndex * 4 + 2] = vertex.matrixIndicesOverride[2];\r\n vertexData.matricesIndices[currentVertexDataIndex * 4 + 3] = vertex.matrixIndicesOverride[3];\r\n } else {\r\n if (matIndices) {\r\n vertexData.matricesIndices[currentVertexDataIndex * 4] = matIndices[vertex.vertexIdxForBones];\r\n vertexData.matricesIndices[currentVertexDataIndex * 4 + 1] = matIndices[vertex.vertexIdxForBones + 1];\r\n vertexData.matricesIndices[currentVertexDataIndex * 4 + 2] = matIndices[vertex.vertexIdxForBones + 2];\r\n vertexData.matricesIndices[currentVertexDataIndex * 4 + 3] = matIndices[vertex.vertexIdxForBones + 3];\r\n }\r\n if (matIndicesExtra && vertexData.matricesIndicesExtra) {\r\n vertexData.matricesIndicesExtra[currentVertexDataIndex * 4] = matIndicesExtra[vertex.vertexIdxForBones];\r\n vertexData.matricesIndicesExtra[currentVertexDataIndex * 4 + 1] = matIndicesExtra[vertex.vertexIdxForBones + 1];\r\n vertexData.matricesIndicesExtra[currentVertexDataIndex * 4 + 2] = matIndicesExtra[vertex.vertexIdxForBones + 2];\r\n vertexData.matricesIndicesExtra[currentVertexDataIndex * 4 + 3] = matIndicesExtra[vertex.vertexIdxForBones + 3];\r\n }\r\n }\r\n if (vertex.matrixWeightsOverride) {\r\n vertexData.matricesWeights[currentVertexDataIndex * 4] = vertex.matrixWeightsOverride[0];\r\n vertexData.matricesWeights[currentVertexDataIndex * 4 + 1] = vertex.matrixWeightsOverride[1];\r\n vertexData.matricesWeights[currentVertexDataIndex * 4 + 2] = vertex.matrixWeightsOverride[2];\r\n vertexData.matricesWeights[currentVertexDataIndex * 4 + 3] = vertex.matrixWeightsOverride[3];\r\n } else {\r\n if (matWeights) {\r\n vertexData.matricesWeights[currentVertexDataIndex * 4] = matWeights[vertex.vertexIdxForBones];\r\n vertexData.matricesWeights[currentVertexDataIndex * 4 + 1] = matWeights[vertex.vertexIdxForBones + 1];\r\n vertexData.matricesWeights[currentVertexDataIndex * 4 + 2] = matWeights[vertex.vertexIdxForBones + 2];\r\n vertexData.matricesWeights[currentVertexDataIndex * 4 + 3] = matWeights[vertex.vertexIdxForBones + 3];\r\n }\r\n if (matWeightsExtra && vertexData.matricesWeightsExtra) {\r\n vertexData.matricesWeightsExtra[currentVertexDataIndex * 4] = matWeightsExtra[vertex.vertexIdxForBones];\r\n vertexData.matricesWeightsExtra[currentVertexDataIndex * 4 + 1] = matWeightsExtra[vertex.vertexIdxForBones + 1];\r\n vertexData.matricesWeightsExtra[currentVertexDataIndex * 4 + 2] = matWeightsExtra[vertex.vertexIdxForBones + 2];\r\n vertexData.matricesWeightsExtra[currentVertexDataIndex * 4 + 3] = matWeightsExtra[vertex.vertexIdxForBones + 3];\r\n }\r\n }\r\n }\r\n\r\n if (!options.captureUVS) {\r\n (vertexData.uvs).push(0.5 + vertex.position.x / size.x);\r\n const v = 0.5 + vertex.position.y / size.y;\r\n (vertexData.uvs).push(CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - v : v);\r\n } else {\r\n vertex.uv.toArray(vertexData.uvs, currentVertexDataIndex * 2);\r\n }\r\n currentVertexDataIndex++;\r\n }\r\n }\r\n }\r\n\r\n // Avoid the \"Setting vertex data kind 'XXX' with an empty array\" warning when calling vertexData.applyToMesh\r\n if (vertexData.indices.length === 0) vertexData.indices = null;\r\n if (vertexData.positions.length === 0) vertexData.positions = null;\r\n if (vertexData.normals.length === 0) vertexData.normals = null;\r\n if (vertexData.uvs.length === 0) vertexData.uvs = null;\r\n if (vertexData.matricesIndices?.length === 0) vertexData.matricesIndices = null;\r\n if (vertexData.matricesWeights?.length === 0) vertexData.matricesWeights = null;\r\n if (vertexData.matricesIndicesExtra?.length === 0) vertexData.matricesIndicesExtra = null;\r\n if (vertexData.matricesWeightsExtra?.length === 0) vertexData.matricesWeightsExtra = null;\r\n\r\n // Return mesh\r\n const decal = new Mesh(name, sourceMesh.getScene());\r\n vertexData.applyToMesh(decal);\r\n\r\n if (useLocalComputation) {\r\n decal.skeleton = sourceMesh.skeleton;\r\n decal.parent = sourceMesh;\r\n } else {\r\n decal.position = position.clone();\r\n decal.rotation = new Vector3(pitch, yaw, angle);\r\n }\r\n\r\n decal.computeWorldMatrix(true);\r\n decal.refreshBoundingInfo(true, true);\r\n\r\n return decal;\r\n}\r\n\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated use the function directly from the module\r\n */\r\nexport const DecalBuilder = {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreateDecal,\r\n};\r\n\r\nMesh.CreateDecal = (name: string, sourceMesh: AbstractMesh, position: Vector3, normal: Vector3, size: Vector3, angle: number): Mesh => {\r\n const options = {\r\n position,\r\n normal,\r\n size,\r\n angle,\r\n };\r\n\r\n return CreateDecal(name, sourceMesh, options);\r\n};\r\n","import { VertexData } from \"../mesh.vertexData\";\r\nimport { Vector2, Vector3, Matrix } from \"../../Maths/math.vector\";\r\nimport { Mesh } from \"../mesh\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { CompatibilityOptions } from \"../../Compat/compatibilityOptions\";\r\n/**\r\n * Scripts based off of https://github.com/maximeq/three-js-capsule-geometry/blob/master/src/CapsuleBufferGeometry.js\r\n * @param options the constructors options used to shape the mesh.\r\n * @returns the capsule VertexData\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set/capsule\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport function CreateCapsuleVertexData(\r\n options: ICreateCapsuleOptions = {\r\n subdivisions: 2,\r\n tessellation: 16,\r\n height: 1,\r\n radius: 0.25,\r\n capSubdivisions: 6,\r\n }\r\n): VertexData {\r\n const subdivisions = Math.max(options.subdivisions ? options.subdivisions : 2, 1) | 0;\r\n const tessellation = Math.max(options.tessellation ? options.tessellation : 16, 3) | 0;\r\n const height = Math.max(options.height ? options.height : 1, 0);\r\n const radius = Math.max(options.radius ? options.radius : 0.25, 0);\r\n const capDetail = Math.max(options.capSubdivisions ? options.capSubdivisions : 6, 1) | 0;\r\n\r\n const radialSegments = tessellation;\r\n const heightSegments = subdivisions;\r\n\r\n const radiusTop = Math.max(options.radiusTop ? options.radiusTop : radius, 0);\r\n const radiusBottom = Math.max(options.radiusBottom ? options.radiusBottom : radius, 0);\r\n\r\n const heightMinusCaps = height - (radiusTop + radiusBottom);\r\n\r\n const thetaStart = 0.0;\r\n const thetaLength = 2.0 * Math.PI;\r\n\r\n const capsTopSegments = Math.max(options.topCapSubdivisions ? options.topCapSubdivisions : capDetail, 1);\r\n const capsBottomSegments = Math.max(options.bottomCapSubdivisions ? options.bottomCapSubdivisions : capDetail, 1);\r\n\r\n const alpha = Math.acos((radiusBottom - radiusTop) / height);\r\n\r\n let indices = [];\r\n const vertices = [];\r\n const normals = [];\r\n const uvs = [];\r\n\r\n let index = 0;\r\n const indexArray = [],\r\n halfHeight = heightMinusCaps * 0.5;\r\n const pi2 = Math.PI * 0.5;\r\n\r\n let x, y;\r\n const normal = Vector3.Zero();\r\n const vertex = Vector3.Zero();\r\n\r\n const cosAlpha = Math.cos(alpha);\r\n const sinAlpha = Math.sin(alpha);\r\n\r\n const coneLength = new Vector2(radiusTop * sinAlpha, halfHeight + radiusTop * cosAlpha)\r\n .subtract(new Vector2(radiusBottom * sinAlpha, -halfHeight + radiusBottom * cosAlpha))\r\n .length();\r\n\r\n // Total length for v texture coord\r\n const vl = radiusTop * alpha + coneLength + radiusBottom * (pi2 - alpha);\r\n\r\n let v = 0;\r\n for (y = 0; y <= capsTopSegments; y++) {\r\n const indexRow = [];\r\n\r\n const a = pi2 - alpha * (y / capsTopSegments);\r\n\r\n v += (radiusTop * alpha) / capsTopSegments;\r\n\r\n const cosA = Math.cos(a);\r\n const sinA = Math.sin(a);\r\n\r\n // calculate the radius of the current row\r\n const _radius = cosA * radiusTop;\r\n\r\n for (x = 0; x <= radialSegments; x++) {\r\n const u = x / radialSegments;\r\n const theta = u * thetaLength + thetaStart;\r\n const sinTheta = Math.sin(theta);\r\n const cosTheta = Math.cos(theta);\r\n // vertex\r\n vertex.x = _radius * sinTheta;\r\n vertex.y = halfHeight + sinA * radiusTop;\r\n vertex.z = _radius * cosTheta;\r\n vertices.push(vertex.x, vertex.y, vertex.z);\r\n // normal\r\n normal.set(cosA * sinTheta, sinA, cosA * cosTheta);\r\n normals.push(normal.x, normal.y, normal.z);\r\n // uv\r\n uvs.push(u, CompatibilityOptions.UseOpenGLOrientationForUV ? v / vl : 1 - v / vl);\r\n // save index of vertex in respective row\r\n indexRow.push(index);\r\n // increase index\r\n index++;\r\n }\r\n // now save vertices of the row in our index array\r\n indexArray.push(indexRow);\r\n }\r\n\r\n const coneHeight = height - radiusTop - radiusBottom + cosAlpha * radiusTop - cosAlpha * radiusBottom;\r\n const slope = (sinAlpha * (radiusBottom - radiusTop)) / coneHeight;\r\n\r\n for (y = 1; y <= heightSegments; y++) {\r\n const indexRow = [];\r\n v += coneLength / heightSegments;\r\n // calculate the radius of the current row\r\n const _radius = sinAlpha * ((y * (radiusBottom - radiusTop)) / heightSegments + radiusTop);\r\n for (x = 0; x <= radialSegments; x++) {\r\n const u = x / radialSegments;\r\n const theta = u * thetaLength + thetaStart;\r\n const sinTheta = Math.sin(theta);\r\n const cosTheta = Math.cos(theta);\r\n // vertex\r\n vertex.x = _radius * sinTheta;\r\n vertex.y = halfHeight + cosAlpha * radiusTop - (y * coneHeight) / heightSegments;\r\n vertex.z = _radius * cosTheta;\r\n vertices.push(vertex.x, vertex.y, vertex.z);\r\n // normal\r\n normal.set(sinTheta, slope, cosTheta).normalize();\r\n normals.push(normal.x, normal.y, normal.z);\r\n // uv\r\n uvs.push(u, CompatibilityOptions.UseOpenGLOrientationForUV ? v / vl : 1 - v / vl);\r\n // save index of vertex in respective row\r\n indexRow.push(index);\r\n // increase index\r\n index++;\r\n }\r\n // now save vertices of the row in our index array\r\n indexArray.push(indexRow);\r\n }\r\n\r\n for (y = 1; y <= capsBottomSegments; y++) {\r\n const indexRow = [];\r\n const a = pi2 - alpha - (Math.PI - alpha) * (y / capsBottomSegments);\r\n v += (radiusBottom * alpha) / capsBottomSegments;\r\n const cosA = Math.cos(a);\r\n const sinA = Math.sin(a);\r\n // calculate the radius of the current row\r\n const _radius = cosA * radiusBottom;\r\n for (x = 0; x <= radialSegments; x++) {\r\n const u = x / radialSegments;\r\n const theta = u * thetaLength + thetaStart;\r\n const sinTheta = Math.sin(theta);\r\n const cosTheta = Math.cos(theta);\r\n // vertex\r\n vertex.x = _radius * sinTheta;\r\n vertex.y = -halfHeight + sinA * radiusBottom;\r\n vertex.z = _radius * cosTheta;\r\n vertices.push(vertex.x, vertex.y, vertex.z);\r\n // normal\r\n normal.set(cosA * sinTheta, sinA, cosA * cosTheta);\r\n normals.push(normal.x, normal.y, normal.z);\r\n // uv\r\n uvs.push(u, CompatibilityOptions.UseOpenGLOrientationForUV ? v / vl : 1 - v / vl);\r\n // save index of vertex in respective row\r\n indexRow.push(index);\r\n // increase index\r\n index++;\r\n }\r\n // now save vertices of the row in our index array\r\n indexArray.push(indexRow);\r\n }\r\n // generate indices\r\n for (x = 0; x < radialSegments; x++) {\r\n for (y = 0; y < capsTopSegments + heightSegments + capsBottomSegments; y++) {\r\n // we use the index array to access the correct indices\r\n const i1 = indexArray[y][x];\r\n const i2 = indexArray[y + 1][x];\r\n const i3 = indexArray[y + 1][x + 1];\r\n const i4 = indexArray[y][x + 1];\r\n // face one\r\n indices.push(i1);\r\n indices.push(i2);\r\n indices.push(i4);\r\n // face two\r\n indices.push(i2);\r\n indices.push(i3);\r\n indices.push(i4);\r\n }\r\n }\r\n\r\n indices = indices.reverse();\r\n\r\n if (options.orientation && !options.orientation.equals(Vector3.Up())) {\r\n const m = new Matrix();\r\n options.orientation\r\n .clone()\r\n .scale(Math.PI * 0.5)\r\n .cross(Vector3.Up())\r\n .toQuaternion()\r\n .toRotationMatrix(m);\r\n const v = Vector3.Zero();\r\n for (let i = 0; i < vertices.length; i += 3) {\r\n v.set(vertices[i], vertices[i + 1], vertices[i + 2]);\r\n Vector3.TransformCoordinatesToRef(v.clone(), m, v);\r\n vertices[i] = v.x;\r\n vertices[i + 1] = v.y;\r\n vertices[i + 2] = v.z;\r\n }\r\n }\r\n\r\n const vDat = new VertexData();\r\n vDat.positions = vertices;\r\n vDat.normals = normals;\r\n vDat.uvs = uvs;\r\n vDat.indices = indices;\r\n\r\n return vDat;\r\n}\r\n\r\n/**\r\n * The options Interface for creating a Capsule Mesh\r\n */\r\nexport interface ICreateCapsuleOptions {\r\n /** The Orientation of the capsule. Default : Vector3.Up() */\r\n orientation?: Vector3;\r\n\r\n /** Number of sub segments on the tube section of the capsule running parallel to orientation. */\r\n subdivisions?: number;\r\n\r\n /** Number of cylindrical segments on the capsule. */\r\n tessellation?: number;\r\n\r\n /** Height or Length of the capsule. */\r\n height?: number;\r\n\r\n /** Radius of the capsule. */\r\n radius?: number;\r\n\r\n /** Number of sub segments on the cap sections of the capsule running parallel to orientation. */\r\n capSubdivisions?: number;\r\n\r\n /** Overwrite for the top radius. */\r\n radiusTop?: number;\r\n\r\n /** Overwrite for the bottom radius. */\r\n radiusBottom?: number;\r\n\r\n /** Overwrite for the top capSubdivisions. */\r\n topCapSubdivisions?: number;\r\n\r\n /** Overwrite for the bottom capSubdivisions. */\r\n bottomCapSubdivisions?: number;\r\n\r\n /** Internal geometry is supposed to change once created. */\r\n updatable?: boolean;\r\n}\r\n\r\n/**\r\n * Creates a capsule or a pill mesh\r\n * @param name defines the name of the mesh\r\n * @param options The constructors options.\r\n * @param scene The scene the mesh is scoped to.\r\n * @returns Capsule Mesh\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport function CreateCapsule(\r\n name: string,\r\n options: ICreateCapsuleOptions = {\r\n orientation: Vector3.Up(),\r\n subdivisions: 2,\r\n tessellation: 16,\r\n height: 1,\r\n radius: 0.25,\r\n capSubdivisions: 6,\r\n updatable: false,\r\n },\r\n scene: Nullable = null\r\n): Mesh {\r\n const capsule = new Mesh(name, scene);\r\n const vertexData = CreateCapsuleVertexData(options);\r\n vertexData.applyToMesh(capsule, options.updatable);\r\n return capsule;\r\n}\r\n\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated please use CreateCapsule directly\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport const CapsuleBuilder = {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreateCapsule,\r\n};\r\n\r\n/**\r\n * Creates a capsule or a pill mesh\r\n * @param name defines the name of the mesh.\r\n * @param options the constructors options used to shape the mesh.\r\n * @param scene defines the scene the mesh is scoped to.\r\n * @returns the capsule mesh\r\n * @see https://doc.babylonjs.com/how_to/capsule_shape\r\n */\r\nMesh.CreateCapsule = (name: string, options: ICreateCapsuleOptions, scene?: Nullable): Mesh => {\r\n return CreateCapsule(name, options, scene);\r\n};\r\n\r\nVertexData.CreateCapsule = CreateCapsuleVertexData;\r\n","import { Logger } from \"../Misc/logger\";\r\nimport { Vector3 } from \"./math.vector\";\r\n\r\n/**\r\n * Class representing an isovector a vector containing 2 INTEGER coordinates\r\n * x axis is horizontal\r\n * y axis is 60 deg counter clockwise from positive y axis\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class _IsoVector {\r\n /**\r\n * Creates a new isovector from the given x and y coordinates\r\n * @param x defines the first coordinate, must be an integer\r\n * @param y defines the second coordinate, must be an integer\r\n */\r\n constructor(\r\n /** defines the first coordinate */\r\n public x: number = 0,\r\n /** defines the second coordinate */\r\n public y: number = 0\r\n ) {\r\n if (x !== Math.floor(x)) {\r\n x === Math.floor(x);\r\n Logger.Warn(\"x is not an integer, floor(x) used\");\r\n }\r\n if (y !== Math.floor(y)) {\r\n y === Math.floor(y);\r\n Logger.Warn(\"y is not an integer, floor(y) used\");\r\n }\r\n }\r\n\r\n // Operators\r\n\r\n /**\r\n * Gets a new IsoVector copied from the IsoVector\r\n * @returns a new IsoVector\r\n */\r\n public clone(): _IsoVector {\r\n return new _IsoVector(this.x, this.y);\r\n }\r\n\r\n /**\r\n * Rotates one IsoVector 60 degrees counter clockwise about another\r\n * Please note that this is an in place operation\r\n * @param other an IsoVector a center of rotation\r\n * @returns the rotated IsoVector\r\n */\r\n public rotate60About(other: _IsoVector) {\r\n //other IsoVector\r\n const x: number = this.x;\r\n this.x = other.x + other.y - this.y;\r\n this.y = x + this.y - other.x;\r\n return this;\r\n }\r\n\r\n /**\r\n * Rotates one IsoVector 60 degrees clockwise about another\r\n * Please note that this is an in place operation\r\n * @param other an IsoVector as center of rotation\r\n * @returns the rotated IsoVector\r\n */\r\n public rotateNeg60About(other: _IsoVector) {\r\n const x = this.x;\r\n this.x = x + this.y - other.y;\r\n this.y = other.x + other.y - x;\r\n return this;\r\n }\r\n\r\n /**\r\n * For an equilateral triangle OAB with O at isovector (0, 0) and A at isovector (m, n)\r\n * Rotates one IsoVector 120 degrees counter clockwise about the center of the triangle\r\n * Please note that this is an in place operation\r\n * @param m integer a measure a Primary triangle of order (m, n) m > n\r\n * @param n >= 0 integer a measure for a Primary triangle of order (m, n)\r\n * @returns the rotated IsoVector\r\n */\r\n public rotate120(m: number, n: number) {\r\n //m, n integers\r\n if (m !== Math.floor(m)) {\r\n m === Math.floor(m);\r\n Logger.Warn(\"m not an integer only floor(m) used\");\r\n }\r\n if (n !== Math.floor(n)) {\r\n n === Math.floor(n);\r\n Logger.Warn(\"n not an integer only floor(n) used\");\r\n }\r\n const x = this.x;\r\n this.x = m - x - this.y;\r\n this.y = n + x;\r\n return this;\r\n }\r\n\r\n /**\r\n * For an equilateral triangle OAB with O at isovector (0, 0) and A at isovector (m, n)\r\n * Rotates one IsoVector 120 degrees clockwise about the center of the triangle\r\n * Please note that this is an in place operation\r\n * @param m integer a measure a Primary triangle of order (m, n) m > n\r\n * @param n >= 0 integer a measure for a Primary triangle of order (m, n)\r\n * @returns the rotated IsoVector\r\n */\r\n public rotateNeg120(m: number, n: number) {\r\n //m, n integers\r\n if (m !== Math.floor(m)) {\r\n m === Math.floor(m);\r\n Logger.Warn(\"m is not an integer, floor(m) used\");\r\n }\r\n if (n !== Math.floor(n)) {\r\n n === Math.floor(n);\r\n Logger.Warn(\"n is not an integer, floor(n) used\");\r\n }\r\n const x = this.x;\r\n this.x = this.y - n;\r\n this.y = m + n - x - this.y;\r\n return this;\r\n }\r\n\r\n /**\r\n * Transforms an IsoVector to one in Cartesian 3D space based on an isovector\r\n * @param origin an IsoVector\r\n * @param isoGridSize\r\n * @returns Point as a Vector3\r\n */\r\n public toCartesianOrigin(origin: _IsoVector, isoGridSize: number) {\r\n const point = Vector3.Zero();\r\n point.x = origin.x + 2 * this.x * isoGridSize + this.y * isoGridSize;\r\n point.y = origin.y + Math.sqrt(3) * this.y * isoGridSize;\r\n return point;\r\n }\r\n\r\n // Statics\r\n\r\n /**\r\n * Gets a new IsoVector(0, 0)\r\n * @returns a new IsoVector\r\n */\r\n public static Zero(): _IsoVector {\r\n return new _IsoVector(0, 0);\r\n }\r\n}\r\n","import { Vector3, TmpVectors } from \"../Maths/math.vector\";\r\nimport { Scalar } from \"../Maths/math.scalar\";\r\nimport { PHI } from \"../Maths/math.constants\";\r\nimport { _IsoVector } from \"../Maths/math.isovector\";\r\n\r\n/**\r\n * Class representing data for one face OAB of an equilateral icosahedron\r\n * When O is the isovector (0, 0), A is isovector (m, n)\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class _PrimaryIsoTriangle {\r\n //properties\r\n public m: number;\r\n public n: number;\r\n public cartesian: Vector3[] = [];\r\n public vertices: _IsoVector[] = [];\r\n public max: number[] = [];\r\n public min: number[] = [];\r\n public vecToidx: { [key: string]: number };\r\n public vertByDist: { [key: string]: number[] };\r\n public closestTo: number[][] = [];\r\n\r\n public innerFacets: string[][] = [];\r\n public isoVecsABOB: _IsoVector[][] = [];\r\n public isoVecsOBOA: _IsoVector[][] = [];\r\n public isoVecsBAOA: _IsoVector[][] = [];\r\n public vertexTypes: number[][] = [];\r\n\r\n public coau: number;\r\n public cobu: number;\r\n public coav: number;\r\n public cobv: number;\r\n\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public IDATA: PolyhedronData = new PolyhedronData(\r\n \"icosahedron\",\r\n \"Regular\",\r\n [\r\n [0, PHI, -1],\r\n [-PHI, 1, 0],\r\n [-1, 0, -PHI],\r\n [1, 0, -PHI],\r\n [PHI, 1, 0],\r\n [0, PHI, 1],\r\n [-1, 0, PHI],\r\n [-PHI, -1, 0],\r\n [0, -PHI, -1],\r\n [PHI, -1, 0],\r\n [1, 0, PHI],\r\n [0, -PHI, 1],\r\n ],\r\n [\r\n [0, 2, 1],\r\n [0, 3, 2],\r\n [0, 4, 3],\r\n [0, 5, 4],\r\n [0, 1, 5],\r\n [7, 6, 1],\r\n [8, 7, 2],\r\n [9, 8, 3],\r\n [10, 9, 4],\r\n [6, 10, 5],\r\n [2, 7, 1],\r\n [3, 8, 2],\r\n [4, 9, 3],\r\n [5, 10, 4],\r\n [1, 6, 5],\r\n [11, 6, 7],\r\n [11, 7, 8],\r\n [11, 8, 9],\r\n [11, 9, 10],\r\n [11, 10, 6],\r\n ]\r\n );\r\n\r\n /**\r\n * Creates the PrimaryIsoTriangle Triangle OAB\r\n * @param m an integer\r\n * @param n an integer\r\n */\r\n\r\n //operators\r\n public setIndices() {\r\n let indexCount = 12; // 12 vertices already assigned\r\n const vecToidx: { [key: string]: number } = {}; //maps iso-vectors to indexCount;\r\n const m = this.m;\r\n const n = this.n;\r\n let g = m; // hcf of m, n when n != 0\r\n let m1 = 1;\r\n let n1 = 0;\r\n if (n !== 0) {\r\n g = Scalar.HCF(m, n);\r\n }\r\n m1 = m / g;\r\n n1 = n / g;\r\n\r\n let fr: number | string; //face to the right of current face\r\n let rot: number | string; //rotation about which vertex for fr\r\n let O: number;\r\n let A: number;\r\n let B: number;\r\n const oVec: _IsoVector = _IsoVector.Zero();\r\n const aVec = new _IsoVector(m, n);\r\n const bVec = new _IsoVector(-n, m + n);\r\n const oaVec: _IsoVector = _IsoVector.Zero();\r\n const abVec: _IsoVector = _IsoVector.Zero();\r\n const obVec: _IsoVector = _IsoVector.Zero();\r\n let verts: number[] = [];\r\n let idx: string;\r\n let idxR: string;\r\n let isoId: string;\r\n let isoIdR: string;\r\n\r\n const closestTo: number[][] = [];\r\n const vDist = this.vertByDist;\r\n\r\n const matchIdx = (f: number, fr: number, isoId: string, isoIdR: string) => {\r\n idx = f + \"|\" + isoId;\r\n idxR = fr + \"|\" + isoIdR;\r\n if (!(idx in vecToidx || idxR in vecToidx)) {\r\n vecToidx[idx] = indexCount;\r\n vecToidx[idxR] = indexCount;\r\n indexCount++;\r\n } else if (idx in vecToidx && !(idxR in vecToidx)) {\r\n vecToidx[idxR] = vecToidx[idx];\r\n } else if (idxR in vecToidx && !(idx in vecToidx)) {\r\n vecToidx[idx] = vecToidx[idxR];\r\n }\r\n if (vDist[isoId][0] > 2) {\r\n closestTo[vecToidx[idx]] = [-vDist[isoId][0], vDist[isoId][1], vecToidx[idx]];\r\n } else {\r\n closestTo[vecToidx[idx]] = [verts[vDist[isoId][0]], vDist[isoId][1], vecToidx[idx]];\r\n }\r\n };\r\n\r\n this.IDATA.edgematch = [\r\n [1, \"B\"],\r\n [2, \"B\"],\r\n [3, \"B\"],\r\n [4, \"B\"],\r\n [0, \"B\"],\r\n [10, \"O\", 14, \"A\"],\r\n [11, \"O\", 10, \"A\"],\r\n [12, \"O\", 11, \"A\"],\r\n [13, \"O\", 12, \"A\"],\r\n [14, \"O\", 13, \"A\"],\r\n [0, \"O\"],\r\n [1, \"O\"],\r\n [2, \"O\"],\r\n [3, \"O\"],\r\n [4, \"O\"],\r\n [19, \"B\", 5, \"A\"],\r\n [15, \"B\", 6, \"A\"],\r\n [16, \"B\", 7, \"A\"],\r\n [17, \"B\", 8, \"A\"],\r\n [18, \"B\", 9, \"A\"],\r\n ];\r\n\r\n /***edges AB to OB***** rotation about B*/\r\n for (let f = 0; f < 20; f++) {\r\n //f current face\r\n\r\n verts = this.IDATA.face[f];\r\n O = verts[2];\r\n A = verts[1];\r\n B = verts[0];\r\n\r\n isoId = oVec.x + \"|\" + oVec.y;\r\n idx = f + \"|\" + isoId;\r\n if (!(idx in vecToidx)) {\r\n vecToidx[idx] = O;\r\n closestTo[O] = [verts[vDist[isoId][0]], vDist[isoId][1]];\r\n }\r\n\r\n isoId = aVec.x + \"|\" + aVec.y;\r\n idx = f + \"|\" + isoId;\r\n if (!(idx in vecToidx)) {\r\n vecToidx[idx] = A;\r\n closestTo[A] = [verts[vDist[isoId][0]], vDist[isoId][1]];\r\n }\r\n\r\n isoId = bVec.x + \"|\" + bVec.y;\r\n idx = f + \"|\" + isoId;\r\n if (!(idx in vecToidx)) {\r\n vecToidx[idx] = B;\r\n closestTo[B] = [verts[vDist[isoId][0]], vDist[isoId][1]];\r\n }\r\n\r\n //for edge vertices\r\n fr = this.IDATA.edgematch[f][0];\r\n rot = this.IDATA.edgematch[f][1];\r\n if (rot === \"B\") {\r\n for (let i = 1; i < g; i++) {\r\n abVec.x = m - i * (m1 + n1);\r\n abVec.y = n + i * m1;\r\n obVec.x = -i * n1;\r\n obVec.y = i * (m1 + n1);\r\n isoId = abVec.x + \"|\" + abVec.y;\r\n isoIdR = obVec.x + \"|\" + obVec.y;\r\n matchIdx(f, fr, isoId, isoIdR);\r\n }\r\n }\r\n\r\n if (rot === \"O\") {\r\n for (let i = 1; i < g; i++) {\r\n obVec.x = -i * n1;\r\n obVec.y = i * (m1 + n1);\r\n oaVec.x = i * m1;\r\n oaVec.y = i * n1;\r\n isoId = obVec.x + \"|\" + obVec.y;\r\n isoIdR = oaVec.x + \"|\" + oaVec.y;\r\n matchIdx(f, fr, isoId, isoIdR);\r\n }\r\n }\r\n\r\n fr = this.IDATA.edgematch[f][2];\r\n rot = this.IDATA.edgematch[f][3];\r\n if (rot && rot === \"A\") {\r\n for (let i = 1; i < g; i++) {\r\n oaVec.x = i * m1;\r\n oaVec.y = i * n1;\r\n abVec.x = m - (g - i) * (m1 + n1); //reversed for BA\r\n abVec.y = n + (g - i) * m1; //reversed for BA\r\n isoId = oaVec.x + \"|\" + oaVec.y;\r\n isoIdR = abVec.x + \"|\" + abVec.y;\r\n matchIdx(f, fr, isoId, isoIdR);\r\n }\r\n }\r\n\r\n for (let i = 0; i < this.vertices.length; i++) {\r\n isoId = this.vertices[i].x + \"|\" + this.vertices[i].y;\r\n idx = f + \"|\" + isoId;\r\n if (!(idx in vecToidx)) {\r\n vecToidx[idx] = indexCount++;\r\n if (vDist[isoId][0] > 2) {\r\n closestTo[vecToidx[idx]] = [-vDist[isoId][0], vDist[isoId][1], vecToidx[idx]];\r\n } else {\r\n closestTo[vecToidx[idx]] = [verts[vDist[isoId][0]], vDist[isoId][1], vecToidx[idx]];\r\n }\r\n }\r\n }\r\n }\r\n\r\n this.closestTo = closestTo;\r\n this.vecToidx = vecToidx;\r\n }\r\n\r\n public calcCoeffs() {\r\n const m = this.m;\r\n const n = this.n;\r\n const thirdR3 = Math.sqrt(3) / 3;\r\n\r\n const LSQD = m * m + n * n + m * n;\r\n\r\n this.coau = (m + n) / LSQD;\r\n this.cobu = -n / LSQD;\r\n this.coav = (-thirdR3 * (m - n)) / LSQD;\r\n this.cobv = (thirdR3 * (2 * m + n)) / LSQD;\r\n }\r\n\r\n public createInnerFacets() {\r\n const m = this.m;\r\n const n = this.n;\r\n for (let y = 0; y < n + m + 1; y++) {\r\n for (let x = this.min[y]; x < this.max[y] + 1; x++) {\r\n if (x < this.max[y] && x < this.max[y + 1] + 1) {\r\n this.innerFacets.push([\"|\" + x + \"|\" + y, \"|\" + x + \"|\" + (y + 1), \"|\" + (x + 1) + \"|\" + y]);\r\n }\r\n if (y > 0 && x < this.max[y - 1] && x + 1 < this.max[y] + 1) {\r\n this.innerFacets.push([\"|\" + x + \"|\" + y, \"|\" + (x + 1) + \"|\" + y, \"|\" + (x + 1) + \"|\" + (y - 1)]);\r\n }\r\n }\r\n }\r\n }\r\n\r\n public edgeVecsABOB() {\r\n const m = this.m;\r\n const n = this.n;\r\n\r\n const B = new _IsoVector(-n, m + n);\r\n\r\n for (let y = 1; y < m + n; y++) {\r\n const point = new _IsoVector(this.min[y], y);\r\n const prev = new _IsoVector(this.min[y - 1], y - 1);\r\n const next = new _IsoVector(this.min[y + 1], y + 1);\r\n const pointR = point.clone();\r\n const prevR = prev.clone();\r\n const nextR = next.clone();\r\n\r\n pointR.rotate60About(B);\r\n prevR.rotate60About(B);\r\n nextR.rotate60About(B);\r\n\r\n const maxPoint = new _IsoVector(this.max[pointR.y], pointR.y);\r\n const maxPrev = new _IsoVector(this.max[pointR.y - 1], pointR.y - 1);\r\n const maxLeftPrev = new _IsoVector(this.max[pointR.y - 1] - 1, pointR.y - 1);\r\n\r\n if (pointR.x !== maxPoint.x || pointR.y !== maxPoint.y) {\r\n if (pointR.x !== maxPrev.x) {\r\n // type2\r\n //up\r\n this.vertexTypes.push([1, 0, 0]);\r\n this.isoVecsABOB.push([point, maxPrev, maxLeftPrev]);\r\n //down\r\n this.vertexTypes.push([1, 0, 0]);\r\n this.isoVecsABOB.push([point, maxLeftPrev, maxPoint]);\r\n } else if (pointR.y === nextR.y) {\r\n // type1\r\n //up\r\n this.vertexTypes.push([1, 1, 0]);\r\n this.isoVecsABOB.push([point, prev, maxPrev]);\r\n //down\r\n this.vertexTypes.push([1, 0, 1]);\r\n this.isoVecsABOB.push([point, maxPrev, next]);\r\n } else {\r\n // type 0\r\n //up\r\n this.vertexTypes.push([1, 1, 0]);\r\n this.isoVecsABOB.push([point, prev, maxPrev]);\r\n //down\r\n this.vertexTypes.push([1, 0, 0]);\r\n this.isoVecsABOB.push([point, maxPrev, maxPoint]);\r\n }\r\n }\r\n }\r\n }\r\n\r\n public mapABOBtoOBOA() {\r\n const point = new _IsoVector(0, 0);\r\n for (let i = 0; i < this.isoVecsABOB.length; i++) {\r\n const temp = [];\r\n for (let j = 0; j < 3; j++) {\r\n point.x = this.isoVecsABOB[i][j].x;\r\n point.y = this.isoVecsABOB[i][j].y;\r\n if (this.vertexTypes[i][j] === 0) {\r\n point.rotateNeg120(this.m, this.n);\r\n }\r\n temp.push(point.clone());\r\n }\r\n this.isoVecsOBOA.push(temp);\r\n }\r\n }\r\n\r\n public mapABOBtoBAOA() {\r\n const point = new _IsoVector(0, 0);\r\n for (let i = 0; i < this.isoVecsABOB.length; i++) {\r\n const temp = [];\r\n for (let j = 0; j < 3; j++) {\r\n point.x = this.isoVecsABOB[i][j].x;\r\n point.y = this.isoVecsABOB[i][j].y;\r\n if (this.vertexTypes[i][j] === 1) {\r\n point.rotate120(this.m, this.n);\r\n }\r\n temp.push(point.clone());\r\n }\r\n this.isoVecsBAOA.push(temp);\r\n }\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public MapToFace(faceNb: number, geodesicData: PolyhedronData) {\r\n const F = this.IDATA.face[faceNb];\r\n const oidx = F[2];\r\n const aidx = F[1];\r\n const bidx = F[0];\r\n\r\n const O = Vector3.FromArray(this.IDATA.vertex[oidx]);\r\n const A = Vector3.FromArray(this.IDATA.vertex[aidx]);\r\n const B = Vector3.FromArray(this.IDATA.vertex[bidx]);\r\n\r\n const OA = A.subtract(O);\r\n const OB = B.subtract(O);\r\n\r\n const x: Vector3 = OA.scale(this.coau).add(OB.scale(this.cobu));\r\n const y: Vector3 = OA.scale(this.coav).add(OB.scale(this.cobv));\r\n\r\n const mapped = [];\r\n\r\n let idx: string;\r\n let tempVec: Vector3 = TmpVectors.Vector3[0];\r\n for (let i = 0; i < this.cartesian.length; i++) {\r\n tempVec = x.scale(this.cartesian[i].x).add(y.scale(this.cartesian[i].y)).add(O);\r\n mapped[i] = [tempVec.x, tempVec.y, tempVec.z];\r\n idx = faceNb + \"|\" + this.vertices[i].x + \"|\" + this.vertices[i].y;\r\n geodesicData.vertex[this.vecToidx[idx]] = [tempVec.x, tempVec.y, tempVec.z];\r\n }\r\n }\r\n\r\n //statics\r\n /**Creates a primary triangle\r\n * @internal\r\n */\r\n\r\n public build(m: number, n: number) {\r\n const vertices = new Array<_IsoVector>();\r\n\r\n const O: _IsoVector = _IsoVector.Zero();\r\n const A: _IsoVector = new _IsoVector(m, n);\r\n const B: _IsoVector = new _IsoVector(-n, m + n);\r\n vertices.push(O, A, B);\r\n\r\n //max internal isoceles triangle vertices\r\n for (let y = n; y < m + 1; y++) {\r\n for (let x = 0; x < m + 1 - y; x++) {\r\n vertices.push(new _IsoVector(x, y));\r\n }\r\n }\r\n\r\n //shared vertices along edges when needed\r\n if (n > 0) {\r\n const g = Scalar.HCF(m, n);\r\n const m1 = m / g;\r\n const n1 = n / g;\r\n\r\n for (let i = 1; i < g; i++) {\r\n vertices.push(new _IsoVector(i * m1, i * n1)); //OA\r\n vertices.push(new _IsoVector(-i * n1, i * (m1 + n1))); //OB\r\n vertices.push(new _IsoVector(m - i * (m1 + n1), n + i * m1)); // AB\r\n }\r\n\r\n //lower rows vertices and their rotations\r\n const ratio = m / n;\r\n for (let y = 1; y < n; y++) {\r\n for (let x = 0; x < y * ratio; x++) {\r\n vertices.push(new _IsoVector(x, y));\r\n vertices.push(new _IsoVector(x, y).rotate120(m, n));\r\n vertices.push(new _IsoVector(x, y).rotateNeg120(m, n));\r\n }\r\n }\r\n }\r\n //order vertices by x and then y\r\n vertices.sort((a, b) => {\r\n return a.x - b.x;\r\n });\r\n\r\n vertices.sort((a, b) => {\r\n return a.y - b.y;\r\n });\r\n\r\n const min = new Array(m + n + 1);\r\n const max = new Array(m + n + 1);\r\n for (let i = 0; i < min.length; i++) {\r\n min[i] = Infinity;\r\n max[i] = -Infinity;\r\n }\r\n\r\n let y: number = 0;\r\n let x: number = 0;\r\n\r\n const len: number = vertices.length;\r\n for (let i = 0; i < len; i++) {\r\n x = vertices[i].x;\r\n y = vertices[i].y;\r\n min[y] = Math.min(x, min[y]);\r\n max[y] = Math.max(x, max[y]);\r\n }\r\n\r\n //calculates the distance of a vertex from a given primary vertex\r\n const distFrom = (vert: _IsoVector, primVert: string) => {\r\n const v = vert.clone();\r\n if (primVert === \"A\") {\r\n v.rotateNeg120(m, n);\r\n }\r\n if (primVert === \"B\") {\r\n v.rotate120(m, n);\r\n }\r\n if (v.x < 0) {\r\n return v.y;\r\n }\r\n return v.x + v.y;\r\n };\r\n\r\n const cartesian: Vector3[] = [];\r\n const distFromO: number[] = [];\r\n const distFromA: number[] = [];\r\n const distFromB: number[] = [];\r\n const vertByDist: { [key: string]: number[] } = {};\r\n const vertData: number[][] = [];\r\n let closest: number = -1;\r\n let dist: number = -1;\r\n for (let i = 0; i < len; i++) {\r\n cartesian[i] = vertices[i].toCartesianOrigin(new _IsoVector(0, 0), 0.5);\r\n distFromO[i] = distFrom(vertices[i], \"O\");\r\n distFromA[i] = distFrom(vertices[i], \"A\");\r\n distFromB[i] = distFrom(vertices[i], \"B\");\r\n\r\n if (distFromO[i] === distFromA[i] && distFromA[i] === distFromB[i]) {\r\n closest = 3;\r\n dist = distFromO[i];\r\n } else if (distFromO[i] === distFromA[i]) {\r\n closest = 4;\r\n dist = distFromO[i];\r\n } else if (distFromA[i] === distFromB[i]) {\r\n closest = 5;\r\n dist = distFromA[i];\r\n } else if (distFromB[i] === distFromO[i]) {\r\n closest = 6;\r\n dist = distFromO[i];\r\n }\r\n if (distFromO[i] < distFromA[i] && distFromO[i] < distFromB[i]) {\r\n closest = 2;\r\n dist = distFromO[i];\r\n }\r\n if (distFromA[i] < distFromO[i] && distFromA[i] < distFromB[i]) {\r\n closest = 1;\r\n dist = distFromA[i];\r\n }\r\n if (distFromB[i] < distFromA[i] && distFromB[i] < distFromO[i]) {\r\n closest = 0;\r\n dist = distFromB[i];\r\n }\r\n vertData.push([closest, dist, vertices[i].x, vertices[i].y]);\r\n }\r\n\r\n vertData.sort((a, b) => {\r\n return a[2] - b[2];\r\n });\r\n vertData.sort((a, b) => {\r\n return a[3] - b[3];\r\n });\r\n vertData.sort((a, b) => {\r\n return a[1] - b[1];\r\n });\r\n vertData.sort((a, b) => {\r\n return a[0] - b[0];\r\n });\r\n\r\n for (let v = 0; v < vertData.length; v++) {\r\n vertByDist[vertData[v][2] + \"|\" + vertData[v][3]] = [vertData[v][0], vertData[v][1], v];\r\n }\r\n\r\n this.m = m;\r\n this.n = n;\r\n this.vertices = vertices;\r\n this.vertByDist = vertByDist;\r\n this.cartesian = cartesian;\r\n this.min = min;\r\n this.max = max;\r\n\r\n return this;\r\n }\r\n}\r\n\r\n/** Builds Polyhedron Data\r\n * @internal\r\n */\r\n\r\nexport class PolyhedronData {\r\n public edgematch: (number | string)[][];\r\n\r\n constructor(public name: string, public category: string, public vertex: number[][], public face: number[][]) {}\r\n}\r\n\r\n/**\r\n * This class Extends the PolyhedronData Class to provide measures for a Geodesic Polyhedron\r\n */\r\nexport class GeodesicData extends PolyhedronData {\r\n /**\r\n * @internal\r\n */\r\n public edgematch: (number | string)[][];\r\n /**\r\n * @internal\r\n */\r\n public adjacentFaces: number[][];\r\n /**\r\n * @internal\r\n */\r\n public sharedNodes: number;\r\n /**\r\n * @internal\r\n */\r\n public poleNodes: number;\r\n /**\r\n * @internal\r\n */\r\n public innerToData(face: number, primTri: _PrimaryIsoTriangle) {\r\n for (let i = 0; i < primTri.innerFacets.length; i++) {\r\n this.face.push(primTri.innerFacets[i].map((el) => primTri.vecToidx[face + el]));\r\n }\r\n }\r\n /**\r\n * @internal\r\n */\r\n public mapABOBtoDATA(faceNb: number, primTri: _PrimaryIsoTriangle) {\r\n const fr = primTri.IDATA.edgematch[faceNb][0];\r\n for (let i = 0; i < primTri.isoVecsABOB.length; i++) {\r\n const temp = [];\r\n for (let j = 0; j < 3; j++) {\r\n if (primTri.vertexTypes[i][j] === 0) {\r\n temp.push(faceNb + \"|\" + primTri.isoVecsABOB[i][j].x + \"|\" + primTri.isoVecsABOB[i][j].y);\r\n } else {\r\n temp.push(fr + \"|\" + primTri.isoVecsABOB[i][j].x + \"|\" + primTri.isoVecsABOB[i][j].y);\r\n }\r\n }\r\n this.face.push([primTri.vecToidx[temp[0]], primTri.vecToidx[temp[1]], primTri.vecToidx[temp[2]]]);\r\n }\r\n }\r\n /**\r\n * @internal\r\n */\r\n public mapOBOAtoDATA(faceNb: number, primTri: _PrimaryIsoTriangle) {\r\n const fr = primTri.IDATA.edgematch[faceNb][0];\r\n for (let i = 0; i < primTri.isoVecsOBOA.length; i++) {\r\n const temp = [];\r\n for (let j = 0; j < 3; j++) {\r\n if (primTri.vertexTypes[i][j] === 1) {\r\n temp.push(faceNb + \"|\" + primTri.isoVecsOBOA[i][j].x + \"|\" + primTri.isoVecsOBOA[i][j].y);\r\n } else {\r\n temp.push(fr + \"|\" + primTri.isoVecsOBOA[i][j].x + \"|\" + primTri.isoVecsOBOA[i][j].y);\r\n }\r\n }\r\n this.face.push([primTri.vecToidx[temp[0]], primTri.vecToidx[temp[1]], primTri.vecToidx[temp[2]]]);\r\n }\r\n }\r\n /**\r\n * @internal\r\n */\r\n public mapBAOAtoDATA(faceNb: number, primTri: _PrimaryIsoTriangle) {\r\n const fr = primTri.IDATA.edgematch[faceNb][2];\r\n for (let i = 0; i < primTri.isoVecsBAOA.length; i++) {\r\n const temp = [];\r\n for (let j = 0; j < 3; j++) {\r\n if (primTri.vertexTypes[i][j] === 1) {\r\n temp.push(faceNb + \"|\" + primTri.isoVecsBAOA[i][j].x + \"|\" + primTri.isoVecsBAOA[i][j].y);\r\n } else {\r\n temp.push(fr + \"|\" + primTri.isoVecsBAOA[i][j].x + \"|\" + primTri.isoVecsBAOA[i][j].y);\r\n }\r\n }\r\n this.face.push([primTri.vecToidx[temp[0]], primTri.vecToidx[temp[1]], primTri.vecToidx[temp[2]]]);\r\n }\r\n }\r\n /**\r\n * @internal\r\n */\r\n public orderData(primTri: _PrimaryIsoTriangle) {\r\n const nearTo: number[][][] = [];\r\n for (let i = 0; i < 13; i++) {\r\n nearTo[i] = [];\r\n }\r\n const close: number[][] = primTri.closestTo;\r\n for (let i = 0; i < close.length; i++) {\r\n if (close[i][0] > -1) {\r\n if (close[i][1] > 0) {\r\n nearTo[close[i][0]].push([i, close[i][1]]);\r\n }\r\n } else {\r\n nearTo[12].push([i, close[i][0]]);\r\n }\r\n }\r\n\r\n const near: number[] = [];\r\n for (let i = 0; i < 12; i++) {\r\n near[i] = i;\r\n }\r\n let nearIndex = 12;\r\n for (let i = 0; i < 12; i++) {\r\n nearTo[i].sort((a: number[], b: number[]) => {\r\n return a[1] - b[1];\r\n });\r\n for (let j = 0; j < nearTo[i].length; j++) {\r\n near[nearTo[i][j][0]] = nearIndex++;\r\n }\r\n }\r\n\r\n for (let j = 0; j < nearTo[12].length; j++) {\r\n near[nearTo[12][j][0]] = nearIndex++;\r\n }\r\n\r\n for (let i = 0; i < this.vertex.length; i++) {\r\n this.vertex[i].push(near[i]);\r\n }\r\n\r\n this.vertex.sort((a, b) => {\r\n return a[3] - b[3];\r\n });\r\n\r\n for (let i = 0; i < this.vertex.length; i++) {\r\n this.vertex[i].pop();\r\n }\r\n\r\n for (let i = 0; i < this.face.length; i++) {\r\n for (let j = 0; j < this.face[i].length; j++) {\r\n this.face[i][j] = near[this.face[i][j]];\r\n }\r\n }\r\n\r\n this.sharedNodes = nearTo[12].length;\r\n this.poleNodes = this.vertex.length - this.sharedNodes;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public setOrder(m: number, faces: number[]) {\r\n const adjVerts: number[] = [];\r\n const dualFaces: number[] = [];\r\n let face: number = faces.pop();\r\n dualFaces.push(face);\r\n let index = this.face[face].indexOf(m);\r\n index = (index + 2) % 3;\r\n let v = this.face[face][index];\r\n adjVerts.push(v);\r\n let f = 0;\r\n while (faces.length > 0) {\r\n face = faces[f];\r\n if (this.face[face].indexOf(v) > -1) {\r\n // v is a vertex of face f\r\n index = (this.face[face].indexOf(v) + 1) % 3;\r\n v = this.face[face][index];\r\n adjVerts.push(v);\r\n dualFaces.push(face);\r\n faces.splice(f, 1);\r\n f = 0;\r\n } else {\r\n f++;\r\n }\r\n }\r\n this.adjacentFaces.push(adjVerts);\r\n return dualFaces;\r\n }\r\n /**\r\n * @internal\r\n */\r\n public toGoldbergPolyhedronData(): PolyhedronData {\r\n const goldbergPolyhedronData: PolyhedronData = new PolyhedronData(\"GeoDual\", \"Goldberg\", [], []);\r\n goldbergPolyhedronData.name = \"GD dual\";\r\n const verticesNb: number = this.vertex.length;\r\n const map = new Array(verticesNb);\r\n for (let v = 0; v < verticesNb; v++) {\r\n map[v] = [];\r\n }\r\n for (let f = 0; f < this.face.length; f++) {\r\n for (let i = 0; i < 3; i++) {\r\n map[this.face[f][i]].push(f);\r\n }\r\n }\r\n let cx = 0;\r\n let cy = 0;\r\n let cz = 0;\r\n let face = [];\r\n let vertex = [];\r\n this.adjacentFaces = [];\r\n for (let m = 0; m < map.length; m++) {\r\n goldbergPolyhedronData.face[m] = this.setOrder(m, map[m].concat([]));\r\n map[m].forEach((el: number) => {\r\n cx = 0;\r\n cy = 0;\r\n cz = 0;\r\n face = this.face[el];\r\n for (let i = 0; i < 3; i++) {\r\n vertex = this.vertex[face[i]];\r\n cx += vertex[0];\r\n cy += vertex[1];\r\n cz += vertex[2];\r\n }\r\n goldbergPolyhedronData.vertex[el] = [cx / 3, cy / 3, cz / 3];\r\n });\r\n }\r\n return goldbergPolyhedronData;\r\n }\r\n\r\n //statics\r\n /**Builds the data for a Geodesic Polyhedron from a primary triangle\r\n * @param primTri the primary triangle\r\n * @internal\r\n */\r\n\r\n public static BuildGeodesicData(primTri: _PrimaryIsoTriangle) {\r\n const geodesicData = new GeodesicData(\r\n \"Geodesic-m-n\",\r\n \"Geodesic\",\r\n [\r\n [0, PHI, -1],\r\n [-PHI, 1, 0],\r\n [-1, 0, -PHI],\r\n [1, 0, -PHI],\r\n [PHI, 1, 0],\r\n [0, PHI, 1],\r\n [-1, 0, PHI],\r\n [-PHI, -1, 0],\r\n [0, -PHI, -1],\r\n [PHI, -1, 0],\r\n [1, 0, PHI],\r\n [0, -PHI, 1],\r\n ],\r\n []\r\n );\r\n\r\n primTri.setIndices();\r\n primTri.calcCoeffs();\r\n primTri.createInnerFacets();\r\n primTri.edgeVecsABOB();\r\n primTri.mapABOBtoOBOA();\r\n primTri.mapABOBtoBAOA();\r\n\r\n for (let f = 0; f < primTri.IDATA.face.length; f++) {\r\n primTri.MapToFace(f, geodesicData);\r\n geodesicData.innerToData(f, primTri);\r\n if (primTri.IDATA.edgematch[f][1] === \"B\") {\r\n geodesicData.mapABOBtoDATA(f, primTri);\r\n }\r\n if (primTri.IDATA.edgematch[f][1] === \"O\") {\r\n geodesicData.mapOBOAtoDATA(f, primTri);\r\n }\r\n if (primTri.IDATA.edgematch[f][3] === \"A\") {\r\n geodesicData.mapBAOAtoDATA(f, primTri);\r\n }\r\n }\r\n\r\n geodesicData.orderData(primTri);\r\n const radius = 1;\r\n geodesicData.vertex = geodesicData.vertex.map(function (el) {\r\n const a = el[0];\r\n const b = el[1];\r\n const c = el[2];\r\n const d = Math.sqrt(a * a + b * b + c * c);\r\n el[0] *= radius / d;\r\n el[1] *= radius / d;\r\n el[2] *= radius / d;\r\n return el;\r\n });\r\n\r\n return geodesicData;\r\n }\r\n}\r\n","import type { Scene } from \"../scene\";\r\nimport type { Vector2 } from \"../Maths/math.vector\";\r\nimport { Vector3 } from \"../Maths/math.vector\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport { Mesh } from \"../Meshes/mesh\";\r\nimport { Color4 } from \"../Maths/math.color\";\r\nimport { Logger } from \"../Misc/logger\";\r\nimport type { FloatArray } from \"../types\";\r\n\r\nMesh._GoldbergMeshParser = (parsedMesh: any, scene: Scene): GoldbergMesh => {\r\n return GoldbergMesh.Parse(parsedMesh, scene);\r\n};\r\n\r\n/**\r\n * Defines the set of goldberg data used to create the polygon\r\n */\r\nexport type GoldbergData = {\r\n /**\r\n * The list of Goldberg faces colors\r\n */\r\n faceColors: Color4[];\r\n /**\r\n * The list of Goldberg faces centers\r\n */\r\n faceCenters: Vector3[];\r\n /**\r\n * The list of Goldberg faces Z axis\r\n */\r\n faceZaxis: Vector3[];\r\n /**\r\n * The list of Goldberg faces Y axis\r\n */\r\n faceXaxis: Vector3[];\r\n /**\r\n * The list of Goldberg faces X axis\r\n */\r\n faceYaxis: Vector3[];\r\n /**\r\n * Defines the number of shared faces\r\n */\r\n nbSharedFaces: number;\r\n /**\r\n * Defines the number of unshared faces\r\n */\r\n nbUnsharedFaces: number;\r\n /**\r\n * Defines the total number of goldberg faces\r\n */\r\n nbFaces: number;\r\n /**\r\n * Defines the number of goldberg faces at the pole\r\n */\r\n nbFacesAtPole: number;\r\n /**\r\n * Defines the number of adjacent faces per goldberg faces\r\n */\r\n adjacentFaces: number[][];\r\n};\r\n\r\n/**\r\n * Mesh for a Goldberg Polyhedron which is made from 12 pentagonal and the rest hexagonal faces\r\n * @see https://en.wikipedia.org/wiki/Goldberg_polyhedron\r\n */\r\nexport class GoldbergMesh extends Mesh {\r\n /**\r\n * Defines the specific Goldberg data used in this mesh construction.\r\n */\r\n public goldbergData: GoldbergData = {\r\n faceColors: [],\r\n faceCenters: [],\r\n faceZaxis: [],\r\n faceXaxis: [],\r\n faceYaxis: [],\r\n nbSharedFaces: 0,\r\n nbUnsharedFaces: 0,\r\n nbFaces: 0,\r\n nbFacesAtPole: 0,\r\n adjacentFaces: [],\r\n };\r\n\r\n /**\r\n * Gets the related Goldberg face from pole infos\r\n * @param poleOrShared Defines the pole index or the shared face index if the fromPole parameter is passed in\r\n * @param fromPole Defines an optional pole index to find the related info from\r\n * @returns the goldberg face number\r\n */\r\n public relatedGoldbergFace(poleOrShared: number, fromPole?: number): number {\r\n if (fromPole === void 0) {\r\n if (poleOrShared > this.goldbergData.nbUnsharedFaces - 1) {\r\n Logger.Warn(\"Maximum number of unshared faces used\");\r\n poleOrShared = this.goldbergData.nbUnsharedFaces - 1;\r\n }\r\n return this.goldbergData.nbUnsharedFaces + poleOrShared;\r\n }\r\n if (poleOrShared > 11) {\r\n Logger.Warn(\"Last pole used\");\r\n poleOrShared = 11;\r\n }\r\n if (fromPole > this.goldbergData.nbFacesAtPole - 1) {\r\n Logger.Warn(\"Maximum number of faces at a pole used\");\r\n fromPole = this.goldbergData.nbFacesAtPole - 1;\r\n }\r\n\r\n return 12 + poleOrShared * this.goldbergData.nbFacesAtPole + fromPole;\r\n }\r\n\r\n private _changeGoldbergFaceColors(colorRange: (number | Color4)[][]): number[] {\r\n for (let i = 0; i < colorRange.length; i++) {\r\n const min: number = colorRange[i][0];\r\n const max: number = colorRange[i][1];\r\n const col: Color4 = colorRange[i][2];\r\n for (let f = min; f < max + 1; f++) {\r\n this.goldbergData.faceColors[f] = col;\r\n }\r\n }\r\n\r\n const newCols: number[] = [];\r\n for (let f = 0; f < 12; f++) {\r\n for (let i = 0; i < 5; i++) {\r\n newCols.push(this.goldbergData.faceColors[f].r, this.goldbergData.faceColors[f].g, this.goldbergData.faceColors[f].b, this.goldbergData.faceColors[f].a);\r\n }\r\n }\r\n for (let f = 12; f < this.goldbergData.faceColors.length; f++) {\r\n for (let i = 0; i < 6; i++) {\r\n newCols.push(this.goldbergData.faceColors[f].r, this.goldbergData.faceColors[f].g, this.goldbergData.faceColors[f].b, this.goldbergData.faceColors[f].a);\r\n }\r\n }\r\n return newCols;\r\n }\r\n\r\n /**\r\n * Set new goldberg face colors\r\n * @param colorRange the new color to apply to the mesh\r\n */\r\n public setGoldbergFaceColors(colorRange: (number | Color4)[][]) {\r\n const newCols = this._changeGoldbergFaceColors(colorRange);\r\n this.setVerticesData(VertexBuffer.ColorKind, newCols);\r\n }\r\n\r\n /**\r\n * Updates new goldberg face colors\r\n * @param colorRange the new color to apply to the mesh\r\n */\r\n public updateGoldbergFaceColors(colorRange: (number | Color4)[][]) {\r\n const newCols = this._changeGoldbergFaceColors(colorRange);\r\n this.updateVerticesData(VertexBuffer.ColorKind, newCols);\r\n }\r\n\r\n private _changeGoldbergFaceUVs(uvRange: (number | Vector2)[][]): FloatArray {\r\n const uvs: FloatArray = this.getVerticesData(VertexBuffer.UVKind)!!;\r\n for (let i = 0; i < uvRange.length; i++) {\r\n const min: number = uvRange[i][0];\r\n const max: number = uvRange[i][1];\r\n const center: Vector2 = uvRange[i][2];\r\n const radius: number = uvRange[i][3];\r\n const angle: number = uvRange[i][4];\r\n const points5: number[] = [];\r\n const points6: number[] = [];\r\n let u: number;\r\n let v: number;\r\n for (let p = 0; p < 5; p++) {\r\n u = center.x + radius * Math.cos(angle + (p * Math.PI) / 2.5);\r\n v = center.y + radius * Math.sin(angle + (p * Math.PI) / 2.5);\r\n if (u < 0) {\r\n u = 0;\r\n }\r\n if (u > 1) {\r\n u = 1;\r\n }\r\n points5.push(u, v);\r\n }\r\n for (let p = 0; p < 6; p++) {\r\n u = center.x + radius * Math.cos(angle + (p * Math.PI) / 3);\r\n v = center.y + radius * Math.sin(angle + (p * Math.PI) / 3);\r\n if (u < 0) {\r\n u = 0;\r\n }\r\n if (u > 1) {\r\n u = 1;\r\n }\r\n points6.push(u, v);\r\n }\r\n for (let f = min; f < Math.min(12, max + 1); f++) {\r\n for (let p = 0; p < 5; p++) {\r\n uvs[10 * f + 2 * p] = points5[2 * p];\r\n uvs[10 * f + 2 * p + 1] = points5[2 * p + 1];\r\n }\r\n }\r\n for (let f = Math.max(12, min); f < max + 1; f++) {\r\n for (let p = 0; p < 6; p++) {\r\n //120 + 12 * (f - 12) = 12 * f - 24\r\n uvs[12 * f - 24 + 2 * p] = points6[2 * p];\r\n uvs[12 * f - 23 + 2 * p] = points6[2 * p + 1];\r\n }\r\n }\r\n }\r\n return uvs;\r\n }\r\n\r\n /**\r\n * set new goldberg face UVs\r\n * @param uvRange the new UVs to apply to the mesh\r\n */\r\n public setGoldbergFaceUVs(uvRange: (number | Vector2)[][]) {\r\n const newUVs: FloatArray = this._changeGoldbergFaceUVs(uvRange);\r\n this.setVerticesData(VertexBuffer.UVKind, newUVs);\r\n }\r\n\r\n /**\r\n * Updates new goldberg face UVs\r\n * @param uvRange the new UVs to apply to the mesh\r\n */\r\n public updateGoldbergFaceUVs(uvRange: (number | Vector2)[][]) {\r\n const newUVs = this._changeGoldbergFaceUVs(uvRange);\r\n this.updateVerticesData(VertexBuffer.UVKind, newUVs);\r\n }\r\n\r\n /**\r\n * Places a mesh on a particular face of the goldberg polygon\r\n * @param mesh Defines the mesh to position\r\n * @param face Defines the face to position onto\r\n * @param position Defines the position relative to the face we are positioning the mesh onto\r\n */\r\n public placeOnGoldbergFaceAt(mesh: Mesh, face: number, position: Vector3) {\r\n const orientation = Vector3.RotationFromAxis(this.goldbergData.faceXaxis[face], this.goldbergData.faceYaxis[face], this.goldbergData.faceZaxis[face]);\r\n mesh.rotation = orientation;\r\n mesh.position = this.goldbergData.faceCenters[face]\r\n .add(this.goldbergData.faceXaxis[face].scale(position.x))\r\n .add(this.goldbergData.faceYaxis[face].scale(position.y))\r\n .add(this.goldbergData.faceZaxis[face].scale(position.z));\r\n }\r\n\r\n /**\r\n * Serialize current mesh\r\n * @param serializationObject defines the object which will receive the serialization data\r\n */\r\n public serialize(serializationObject: any): void {\r\n super.serialize(serializationObject);\r\n serializationObject.type = \"GoldbergMesh\";\r\n\r\n const goldbergData: any = {};\r\n goldbergData.adjacentFaces = this.goldbergData.adjacentFaces;\r\n goldbergData.nbSharedFaces = this.goldbergData.nbSharedFaces;\r\n goldbergData.nbUnsharedFaces = this.goldbergData.nbUnsharedFaces;\r\n goldbergData.nbFaces = this.goldbergData.nbFaces;\r\n goldbergData.nbFacesAtPole = this.goldbergData.nbFacesAtPole;\r\n\r\n if (this.goldbergData.faceColors) {\r\n goldbergData.faceColors = [];\r\n for (const color of this.goldbergData.faceColors) {\r\n goldbergData.faceColors.push(color.asArray());\r\n }\r\n }\r\n if (this.goldbergData.faceCenters) {\r\n goldbergData.faceCenters = [];\r\n for (const vector of this.goldbergData.faceCenters) {\r\n goldbergData.faceCenters.push(vector.asArray());\r\n }\r\n }\r\n if (this.goldbergData.faceZaxis) {\r\n goldbergData.faceZaxis = [];\r\n for (const vector of this.goldbergData.faceZaxis) {\r\n goldbergData.faceZaxis.push(vector.asArray());\r\n }\r\n }\r\n if (this.goldbergData.faceYaxis) {\r\n goldbergData.faceYaxis = [];\r\n for (const vector of this.goldbergData.faceYaxis) {\r\n goldbergData.faceYaxis.push(vector.asArray());\r\n }\r\n }\r\n if (this.goldbergData.faceXaxis) {\r\n goldbergData.faceXaxis = [];\r\n for (const vector of this.goldbergData.faceXaxis) {\r\n goldbergData.faceXaxis.push(vector.asArray());\r\n }\r\n }\r\n\r\n serializationObject.goldbergData = goldbergData;\r\n }\r\n\r\n /**\r\n * Parses a serialized goldberg mesh\r\n * @param parsedMesh the serialized mesh\r\n * @param scene the scene to create the goldberg mesh in\r\n * @returns the created goldberg mesh\r\n */\r\n public static Parse(parsedMesh: any, scene: Scene): GoldbergMesh {\r\n const goldbergData = parsedMesh.goldbergData;\r\n goldbergData.faceColors = goldbergData.faceColors.map((el: number[]) => Color4.FromArray(el));\r\n goldbergData.faceCenters = goldbergData.faceCenters.map((el: number[]) => Vector3.FromArray(el));\r\n goldbergData.faceZaxis = goldbergData.faceZaxis.map((el: number[]) => Vector3.FromArray(el));\r\n goldbergData.faceXaxis = goldbergData.faceXaxis.map((el: number[]) => Vector3.FromArray(el));\r\n goldbergData.faceYaxis = goldbergData.faceYaxis.map((el: number[]) => Vector3.FromArray(el));\r\n\r\n const goldberg = new GoldbergMesh(parsedMesh.name, scene);\r\n goldberg.goldbergData = goldbergData;\r\n\r\n return goldberg;\r\n }\r\n}\r\n","import { Path2 } from \"../../Maths/math.path\";\r\nimport { Vector3 } from \"../../Maths/math.vector\";\r\nimport type { Scene } from \"../../scene\";\r\nimport type { Nullable } from \"../../types\";\r\nimport { Mesh } from \"../mesh\";\r\nimport { ExtrudePolygon } from \"./polygonBuilder\";\r\n\r\ndeclare let earcut: any;\r\n\r\n/**\r\n * Parser inspired by https://github.com/mrdoob/three.js/blob/master/examples/jsm/loaders/FontLoader.js\r\n */\r\n\r\n// Interfaces\r\n\r\n/**\r\n * Represents glyph data generated by http://gero3.github.io/facetype.js/\r\n */\r\nexport interface IGlyphData {\r\n /** Commands used to draw (line, move, curve, etc..) */\r\n o: string;\r\n\r\n /** Width */\r\n ha: number;\r\n}\r\n\r\n/**\r\n * Represents font data generated by http://gero3.github.io/facetype.js/\r\n */\r\nexport interface IFontData {\r\n /**\r\n * Font resolution\r\n */\r\n resolution: number;\r\n /** Underline tickness */\r\n underlineThickness: number;\r\n /** Bounding box */\r\n boundingBox: {\r\n yMax: number;\r\n yMin: number;\r\n };\r\n /** List of supported glyphs */\r\n glyphs: { [key: string]: IGlyphData };\r\n}\r\n\r\n// Shape functions\r\nclass ShapePath {\r\n private _paths: Path2[] = [];\r\n private _tempPaths: Path2[] = [];\r\n private _holes: Path2[] = [];\r\n private _currentPath: Path2;\r\n private _resolution: number;\r\n\r\n /** Create the ShapePath used to support glyphs */\r\n constructor(resolution: number) {\r\n this._resolution = resolution;\r\n }\r\n\r\n /** Move the virtual cursor to a coordinate */\r\n moveTo(x: number, y: number) {\r\n this._currentPath = new Path2(x, y);\r\n this._tempPaths.push(this._currentPath);\r\n }\r\n\r\n /** Draw a line from the virtual cursor to a given coordinate */\r\n lineTo(x: number, y: number) {\r\n this._currentPath.addLineTo(x, y);\r\n }\r\n\r\n /** Create a quadratic curve from the virtual cursor to a given coordinate */\r\n quadraticCurveTo(cpx: number, cpy: number, x: number, y: number) {\r\n this._currentPath.addQuadraticCurveTo(cpx, cpy, x, y, this._resolution);\r\n }\r\n\r\n /** Create a bezier curve from the virtual cursor to a given coordinate */\r\n bezierCurveTo(cpx1: number, cpy1: number, cpx2: number, cpy2: number, x: number, y: number) {\r\n this._currentPath.addBezierCurveTo(cpx1, cpy1, cpx2, cpy2, x, y, this._resolution);\r\n }\r\n\r\n /** Extract holes based on CW / CCW */\r\n extractHoles() {\r\n for (const path of this._tempPaths) {\r\n if (path.area() > 0) {\r\n this._holes.push(path);\r\n } else {\r\n this._paths.push(path);\r\n }\r\n }\r\n\r\n if (!this._paths.length && this._holes.length) {\r\n const temp = this._holes;\r\n this._holes = this._paths;\r\n this._paths = temp;\r\n }\r\n\r\n this._tempPaths.length = 0;\r\n }\r\n\r\n /** Gets the list of paths */\r\n get paths() {\r\n return this._paths;\r\n }\r\n\r\n /** Gets the list of holes */\r\n get holes() {\r\n return this._holes;\r\n }\r\n}\r\n\r\n// Utility functions\r\nfunction CreateShapePath(\r\n char: string,\r\n scale: number,\r\n offsetX: number,\r\n offsetY: number,\r\n resolution: number,\r\n fontData: IFontData\r\n): Nullable<{\r\n offsetX: number;\r\n shapePath: ShapePath;\r\n}> {\r\n const glyph = fontData.glyphs[char] || fontData.glyphs[\"?\"];\r\n\r\n if (!glyph) {\r\n // return if there is no glyph data\r\n return null;\r\n }\r\n\r\n const shapePath = new ShapePath(resolution);\r\n\r\n if (glyph.o) {\r\n const outline = glyph.o.split(\" \");\r\n\r\n for (let i = 0, l = outline.length; i < l; ) {\r\n const action = outline[i++];\r\n\r\n switch (action) {\r\n case \"m\": {\r\n // moveTo\r\n const x = parseInt(outline[i++]) * scale + offsetX;\r\n const y = parseInt(outline[i++]) * scale + offsetY;\r\n\r\n shapePath.moveTo(x, y);\r\n break;\r\n }\r\n case \"l\": {\r\n // lineTo\r\n const x = parseInt(outline[i++]) * scale + offsetX;\r\n const y = parseInt(outline[i++]) * scale + offsetY;\r\n\r\n shapePath.lineTo(x, y);\r\n break;\r\n }\r\n case \"q\": {\r\n // quadraticCurveTo\r\n const cpx = parseInt(outline[i++]) * scale + offsetX;\r\n const cpy = parseInt(outline[i++]) * scale + offsetY;\r\n const cpx1 = parseInt(outline[i++]) * scale + offsetX;\r\n const cpy1 = parseInt(outline[i++]) * scale + offsetY;\r\n\r\n shapePath.quadraticCurveTo(cpx1, cpy1, cpx, cpy);\r\n break;\r\n }\r\n case \"b\": {\r\n // bezierCurveTo\r\n const cpx = parseInt(outline[i++]) * scale + offsetX;\r\n const cpy = parseInt(outline[i++]) * scale + offsetY;\r\n const cpx1 = parseInt(outline[i++]) * scale + offsetX;\r\n const cpy1 = parseInt(outline[i++]) * scale + offsetY;\r\n const cpx2 = parseInt(outline[i++]) * scale + offsetX;\r\n const cpy2 = parseInt(outline[i++]) * scale + offsetY;\r\n\r\n shapePath.bezierCurveTo(cpx1, cpy1, cpx2, cpy2, cpx, cpy);\r\n break;\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Extract holes (based on clockwise data)\r\n shapePath.extractHoles();\r\n\r\n return { offsetX: glyph.ha * scale, shapePath: shapePath };\r\n}\r\n\r\n/**\r\n * Creates shape paths from a text and font\r\n * @param text the text\r\n * @param size size of the font\r\n * @param resolution resolution of the font\r\n * @param fontData defines the font data (can be generated with http://gero3.github.io/facetype.js/)\r\n * @returns array of ShapePath objects\r\n */\r\nexport function CreateTextShapePaths(text: string, size: number, resolution: number, fontData: IFontData) {\r\n const chars = Array.from(text);\r\n const scale = size / fontData.resolution;\r\n const line_height = (fontData.boundingBox.yMax - fontData.boundingBox.yMin + fontData.underlineThickness) * scale;\r\n\r\n const shapePaths: ShapePath[] = [];\r\n\r\n let offsetX = 0,\r\n offsetY = 0;\r\n\r\n for (let i = 0; i < chars.length; i++) {\r\n const char = chars[i];\r\n\r\n if (char === \"\\n\") {\r\n offsetX = 0;\r\n offsetY -= line_height;\r\n } else {\r\n const ret = CreateShapePath(char, scale, offsetX, offsetY, resolution, fontData);\r\n\r\n if (ret) {\r\n offsetX += ret.offsetX;\r\n shapePaths.push(ret.shapePath);\r\n }\r\n }\r\n }\r\n\r\n return shapePaths;\r\n}\r\n\r\n/**\r\n * Create a text mesh\r\n * @param name defines the name of the mesh\r\n * @param text defines the text to use to build the mesh\r\n * @param fontData defines the font data (can be generated with http://gero3.github.io/facetype.js/)\r\n * @param options defines options used to create the mesh\r\n * @param scene defines the hosting scene\r\n * @param earcutInjection can be used to inject your own earcut reference\r\n * @returns a new Mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set/text\r\n */\r\nexport function CreateText(\r\n name: string,\r\n text: string,\r\n fontData: IFontData,\r\n options: {\r\n size?: number;\r\n resolution?: number;\r\n depth?: number;\r\n sideOrientation?: number;\r\n } = {\r\n size: 50,\r\n resolution: 8,\r\n depth: 1.0,\r\n },\r\n scene: Nullable = null,\r\n earcutInjection = earcut\r\n): Nullable {\r\n // First we need to generate the paths\r\n const shapePaths = CreateTextShapePaths(text, options.size || 50, options.resolution || 8, fontData);\r\n\r\n // And extrude them\r\n const meshes: Mesh[] = [];\r\n for (const shapePath of shapePaths) {\r\n if (!shapePath.paths.length) {\r\n continue;\r\n }\r\n\r\n const holes = shapePath.holes.slice(); // Copy it as we will update the copy\r\n for (const path of shapePath.paths) {\r\n const holeVectors: Vector3[][] = [];\r\n const shapeVectors: Vector3[] = [];\r\n const points = path.getPoints();\r\n for (const point of points) {\r\n shapeVectors.push(new Vector3(point.x, 0, point.y)); // ExtrudePolygon expects data on the xz plane\r\n }\r\n\r\n // Holes\r\n const localHolesCopy = holes.slice();\r\n for (const hole of localHolesCopy) {\r\n const points = hole.getPoints();\r\n\r\n let found = false;\r\n for (const point of points) {\r\n if (path.isPointInside(point)) {\r\n found = true;\r\n break;\r\n }\r\n }\r\n\r\n if (!found) {\r\n continue;\r\n }\r\n\r\n const holePoints: Vector3[] = [];\r\n for (const point of points) {\r\n holePoints.push(new Vector3(point.x, 0, point.y)); // ExtrudePolygon expects data on the xz plane\r\n }\r\n holeVectors.push(holePoints);\r\n\r\n // Remove the hole as it was already used\r\n holes.splice(holes.indexOf(hole), 1);\r\n }\r\n\r\n // There is at least a hole but it was unaffected\r\n if (!holeVectors.length && holes.length) {\r\n for (const hole of holes) {\r\n const points = hole.getPoints();\r\n const holePoints: Vector3[] = [];\r\n for (const point of points) {\r\n holePoints.push(new Vector3(point.x, 0, point.y)); // ExtrudePolygon expects data on the xz plane\r\n }\r\n holeVectors.push(holePoints);\r\n }\r\n }\r\n\r\n // Extrusion!\r\n const mesh = ExtrudePolygon(\r\n name,\r\n {\r\n shape: shapeVectors,\r\n holes: holeVectors.length ? holeVectors : undefined,\r\n depth: options.depth || 1.0,\r\n sideOrientation: Mesh._GetDefaultSideOrientation(options.sideOrientation || Mesh.DOUBLESIDE),\r\n },\r\n scene,\r\n earcutInjection\r\n );\r\n meshes.push(mesh);\r\n }\r\n }\r\n\r\n // Then we can merge everyone into one single mesh\r\n const newMesh = Mesh.MergeMeshes(meshes, true, true);\r\n\r\n if (newMesh) {\r\n // Move pivot to center\r\n const bbox = newMesh?.getBoundingInfo();\r\n newMesh.position.x = -bbox?.boundingBox.extendSizeWorld._x;\r\n newMesh.position.y = -bbox?.boundingBox.extendSizeWorld._y;\r\n newMesh.position.z = -bbox?.boundingBox.extendSizeWorld._z;\r\n newMesh.name = name;\r\n\r\n newMesh.rotation.x = -Math.PI / 2;\r\n\r\n newMesh.bakeCurrentTransformIntoVertices();\r\n }\r\n\r\n return newMesh;\r\n}\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport { CreateRibbon } from \"./Builders/ribbonBuilder\";\r\nimport { CreateDisc } from \"./Builders/discBuilder\";\r\nimport { CreateBox } from \"./Builders/boxBuilder\";\r\nimport { CreateTiledBox } from \"./Builders/tiledBoxBuilder\";\r\nimport { CreateSphere } from \"./Builders/sphereBuilder\";\r\nimport { CreateCylinder } from \"./Builders/cylinderBuilder\";\r\nimport { CreateTorus } from \"./Builders/torusBuilder\";\r\nimport { CreateTorusKnot } from \"./Builders/torusKnotBuilder\";\r\nimport { CreateDashedLines, CreateLineSystem, CreateLines } from \"./Builders/linesBuilder\";\r\nimport { CreatePolygon, ExtrudePolygon } from \"./Builders/polygonBuilder\";\r\nimport { ExtrudeShape, ExtrudeShapeCustom } from \"./Builders/shapeBuilder\";\r\nimport { CreateLathe } from \"./Builders/latheBuilder\";\r\nimport { CreatePlane } from \"./Builders/planeBuilder\";\r\nimport { CreateTiledPlane } from \"./Builders/tiledPlaneBuilder\";\r\nimport { CreateGround, CreateGroundFromHeightMap, CreateTiledGround } from \"./Builders/groundBuilder\";\r\nimport { CreateTube } from \"./Builders/tubeBuilder\";\r\nimport { CreatePolyhedron } from \"./Builders/polyhedronBuilder\";\r\nimport { CreateIcoSphere } from \"./Builders/icoSphereBuilder\";\r\nimport { CreateDecal } from \"./Builders/decalBuilder\";\r\nimport { CreateCapsule } from \"./Builders/capsuleBuilder\";\r\nimport { CreateGeodesic } from \"./Builders/geodesicBuilder\";\r\nimport { CreateGoldberg } from \"./Builders/goldbergBuilder\";\r\nimport { CreateText } from \"./Builders/textBuilder\";\r\n\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n */\r\nexport const MeshBuilder = {\r\n CreateBox,\r\n CreateTiledBox,\r\n CreateSphere,\r\n CreateDisc,\r\n CreateIcoSphere,\r\n CreateRibbon,\r\n CreateCylinder,\r\n CreateTorus,\r\n CreateTorusKnot,\r\n CreateLineSystem,\r\n CreateLines,\r\n CreateDashedLines,\r\n ExtrudeShape,\r\n ExtrudeShapeCustom,\r\n CreateLathe,\r\n CreateTiledPlane,\r\n CreatePlane,\r\n CreateGround,\r\n CreateTiledGround,\r\n CreateGroundFromHeightMap,\r\n CreatePolygon,\r\n ExtrudePolygon,\r\n CreateTube,\r\n CreatePolyhedron,\r\n CreateGeodesic,\r\n CreateGoldberg,\r\n CreateDecal,\r\n CreateCapsule,\r\n CreateText,\r\n};\r\n","import type { Scene } from \"../../scene\";\r\nimport type { Vector4 } from \"../../Maths/math.vector\";\r\nimport type { Color4 } from \"../../Maths/math.color\";\r\nimport type { Mesh } from \"../../Meshes/mesh\";\r\nimport { CreatePolyhedron } from \"./polyhedronBuilder\";\r\nimport type { Nullable } from \"../../types\";\r\nimport { Logger } from \"../../Misc/logger\";\r\nimport { _PrimaryIsoTriangle, GeodesicData } from \"../geodesicMesh\";\r\n\r\n/**\r\n * Creates the Mesh for a Geodesic Polyhedron\r\n * @see https://en.wikipedia.org/wiki/Geodesic_polyhedron\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/polyhedra/geodesic_poly\r\n * @param name defines the name of the mesh\r\n * @param options an object used to set the following optional parameters for the polyhedron, required but can be empty\r\n * * m number of horizontal steps along an isogrid\r\n * * n number of angled steps along an isogrid\r\n * * size the size of the Geodesic, optional default 1\r\n * * sizeX allows stretching in the x direction, optional, default size\r\n * * sizeY allows stretching in the y direction, optional, default size\r\n * * sizeZ allows stretching in the z direction, optional, default size\r\n * * faceUV an array of Vector4 elements used to set different images to the top, rings and bottom respectively\r\n * * faceColors an array of Color3 elements used to set different colors to the top, rings and bottom respectively\r\n * * flat when true creates a flat shaded mesh, optional, default true\r\n * * subdivisions increasing the subdivisions increases the number of faces, optional, default 4\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * @param options.n\r\n * @param options.size\r\n * @param options.sizeX\r\n * @param options.sizeY\r\n * @param options.sizeZ\r\n * @param options.faceUV\r\n * @param options.faceColors\r\n * @param options.flat\r\n * @param options.updatable\r\n * @param options.sideOrientation\r\n * @param options.frontUVs\r\n * @param options.backUVs\r\n * @param options.m\r\n * @param scene defines the hosting scene\r\n * @returns Geodesic mesh\r\n */\r\nexport function CreateGeodesic(\r\n name: string,\r\n options: {\r\n m?: number;\r\n n?: number;\r\n size?: number;\r\n sizeX?: number;\r\n sizeY?: number;\r\n sizeZ?: number;\r\n faceUV?: Vector4[];\r\n faceColors?: Color4[];\r\n flat?: boolean;\r\n updatable?: boolean;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n },\r\n scene: Nullable = null\r\n): Mesh {\r\n let m: number = options.m || 1;\r\n if (m !== Math.floor(m)) {\r\n m === Math.floor(m);\r\n Logger.Warn(\"m not an integer only floor(m) used\");\r\n }\r\n let n: number = options.n || 0;\r\n if (n !== Math.floor(n)) {\r\n n === Math.floor(n);\r\n Logger.Warn(\"n not an integer only floor(n) used\");\r\n }\r\n if (n > m) {\r\n const temp = n;\r\n n = m;\r\n m = temp;\r\n Logger.Warn(\"n > m therefore m and n swapped\");\r\n }\r\n const primTri: _PrimaryIsoTriangle = new _PrimaryIsoTriangle();\r\n primTri.build(m, n);\r\n const geodesicData = GeodesicData.BuildGeodesicData(primTri);\r\n\r\n const geoOptions: object = {\r\n custom: geodesicData,\r\n size: options.size,\r\n sizeX: options.sizeX,\r\n sizeY: options.sizeY,\r\n sizeZ: options.sizeZ,\r\n faceUV: options.faceUV,\r\n faceColors: options.faceColors,\r\n flat: options.flat,\r\n updatable: options.updatable,\r\n sideOrientation: options.sideOrientation,\r\n frontUVs: options.frontUVs,\r\n backUVs: options.backUVs,\r\n };\r\n const geodesic = CreatePolyhedron(name, geoOptions, scene);\r\n\r\n return geodesic;\r\n}\r\n","import type { Scene } from \"../../scene\";\r\nimport { Vector3 } from \"../../Maths/math.vector\";\r\nimport { Color4 } from \"../../Maths/math.color\";\r\nimport { Mesh } from \"../../Meshes/mesh\";\r\nimport { VertexData } from \"../mesh.vertexData\";\r\nimport type { Nullable } from \"../../types\";\r\nimport { Logger } from \"../../Misc/logger\";\r\nimport type { PolyhedronData } from \"../geodesicMesh\";\r\nimport { _PrimaryIsoTriangle, GeodesicData } from \"../geodesicMesh\";\r\nimport { GoldbergMesh } from \"../goldbergMesh\";\r\nimport { CompatibilityOptions } from \"../../Compat/compatibilityOptions\";\r\n\r\n/**\r\n * Defines the set of data required to create goldberg vertex data.\r\n */\r\nexport type GoldbergVertexDataOption = {\r\n /**\r\n * the size of the Goldberg, optional default 1\r\n */\r\n size?: number;\r\n /**\r\n * allows stretching in the x direction, optional, default size\r\n */\r\n sizeX?: number;\r\n /**\r\n * allows stretching in the y direction, optional, default size\r\n */\r\n sizeY?: number;\r\n /**\r\n * allows stretching in the z direction, optional, default size\r\n */\r\n sizeZ?: number;\r\n /**\r\n * optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n */\r\n sideOrientation?: number;\r\n};\r\n\r\n/**\r\n * Defines the set of data required to create a goldberg mesh.\r\n */\r\nexport type GoldbergCreationOption = {\r\n /**\r\n * number of horizontal steps along an isogrid\r\n */\r\n m?: number;\r\n /**\r\n * number of angled steps along an isogrid\r\n */\r\n n?: number;\r\n /**\r\n * defines if the mesh must be flagged as updatable\r\n */\r\n updatable?: boolean;\r\n} & GoldbergVertexDataOption;\r\n\r\n/**\r\n * Creates the Mesh for a Goldberg Polyhedron\r\n * @param options an object used to set the following optional parameters for the polyhedron, required but can be empty\r\n * @param goldbergData polyhedronData defining the Goldberg polyhedron\r\n * @returns GoldbergSphere mesh\r\n */\r\nexport function CreateGoldbergVertexData(options: GoldbergVertexDataOption, goldbergData: PolyhedronData): VertexData {\r\n const size = options.size;\r\n const sizeX: number = options.sizeX || size || 1;\r\n const sizeY: number = options.sizeY || size || 1;\r\n const sizeZ: number = options.sizeZ || size || 1;\r\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\r\n\r\n const positions = new Array();\r\n const indices = new Array();\r\n const normals = new Array();\r\n const uvs = new Array();\r\n\r\n let minX = Infinity;\r\n let maxX = -Infinity;\r\n let minY = Infinity;\r\n let maxY = -Infinity;\r\n\r\n for (let v = 0; v < goldbergData.vertex.length; v++) {\r\n minX = Math.min(minX, goldbergData.vertex[v][0] * sizeX);\r\n maxX = Math.max(maxX, goldbergData.vertex[v][0] * sizeX);\r\n minY = Math.min(minY, goldbergData.vertex[v][1] * sizeY);\r\n maxY = Math.max(maxY, goldbergData.vertex[v][1] * sizeY);\r\n }\r\n\r\n let index: number = 0;\r\n for (let f = 0; f < goldbergData.face.length; f++) {\r\n const verts = goldbergData.face[f];\r\n const a = Vector3.FromArray(goldbergData.vertex[verts[0]]);\r\n const b = Vector3.FromArray(goldbergData.vertex[verts[2]]);\r\n const c = Vector3.FromArray(goldbergData.vertex[verts[1]]);\r\n const ba = b.subtract(a);\r\n const ca = c.subtract(a);\r\n const norm = Vector3.Cross(ca, ba).normalize();\r\n for (let v = 0; v < verts.length; v++) {\r\n normals.push(norm.x, norm.y, norm.z);\r\n const pdata = goldbergData.vertex[verts[v]];\r\n positions.push(pdata[0] * sizeX, pdata[1] * sizeY, pdata[2] * sizeZ);\r\n const vCoord = (pdata[1] * sizeY - minY) / (maxY - minY);\r\n uvs.push((pdata[0] * sizeX - minX) / (maxX - minX), CompatibilityOptions.UseOpenGLOrientationForUV ? 1 - vCoord : vCoord);\r\n }\r\n for (let v = 0; v < verts.length - 2; v++) {\r\n indices.push(index, index + v + 2, index + v + 1);\r\n }\r\n index += verts.length;\r\n }\r\n\r\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs);\r\n\r\n const vertexData = new VertexData();\r\n vertexData.positions = positions;\r\n vertexData.indices = indices;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n return vertexData;\r\n}\r\n\r\n/**\r\n * Creates the Mesh for a Goldberg Polyhedron which is made from 12 pentagonal and the rest hexagonal faces\r\n * @see https://en.wikipedia.org/wiki/Goldberg_polyhedron\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/polyhedra/goldberg_poly\r\n * @param name defines the name of the mesh\r\n * @param options an object used to set the following optional parameters for the polyhedron, required but can be empty\r\n * @param scene defines the hosting scene\r\n * @returns Goldberg mesh\r\n */\r\nexport function CreateGoldberg(name: string, options: GoldbergCreationOption, scene: Nullable = null): GoldbergMesh {\r\n const size = options.size;\r\n const sizeX: number = options.sizeX || size || 1;\r\n const sizeY: number = options.sizeY || size || 1;\r\n const sizeZ: number = options.sizeZ || size || 1;\r\n let m: number = options.m || 1;\r\n if (m !== Math.floor(m)) {\r\n m === Math.floor(m);\r\n Logger.Warn(\"m not an integer only floor(m) used\");\r\n }\r\n let n: number = options.n || 0;\r\n if (n !== Math.floor(n)) {\r\n n === Math.floor(n);\r\n Logger.Warn(\"n not an integer only floor(n) used\");\r\n }\r\n if (n > m) {\r\n const temp = n;\r\n n = m;\r\n m = temp;\r\n Logger.Warn(\"n > m therefore m and n swapped\");\r\n }\r\n const primTri: _PrimaryIsoTriangle = new _PrimaryIsoTriangle();\r\n primTri.build(m, n);\r\n const geodesicData = GeodesicData.BuildGeodesicData(primTri);\r\n const goldbergData = geodesicData.toGoldbergPolyhedronData();\r\n\r\n const goldberg = new GoldbergMesh(name, scene);\r\n\r\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n goldberg._originalBuilderSideOrientation = options.sideOrientation;\r\n\r\n const vertexData = CreateGoldbergVertexData(options, goldbergData);\r\n\r\n vertexData.applyToMesh(goldberg, options.updatable);\r\n\r\n goldberg.goldbergData.nbSharedFaces = geodesicData.sharedNodes;\r\n goldberg.goldbergData.nbUnsharedFaces = geodesicData.poleNodes;\r\n goldberg.goldbergData.adjacentFaces = geodesicData.adjacentFaces;\r\n goldberg.goldbergData.nbFaces = goldberg.goldbergData.nbSharedFaces + goldberg.goldbergData.nbUnsharedFaces;\r\n goldberg.goldbergData.nbFacesAtPole = (goldberg.goldbergData.nbUnsharedFaces - 12) / 12;\r\n for (let f = 0; f < geodesicData.vertex.length; f++) {\r\n goldberg.goldbergData.faceCenters.push(Vector3.FromArray(geodesicData.vertex[f]));\r\n goldberg.goldbergData.faceCenters[f].x *= sizeX;\r\n goldberg.goldbergData.faceCenters[f].y *= sizeY;\r\n goldberg.goldbergData.faceCenters[f].z *= sizeZ;\r\n goldberg.goldbergData.faceColors.push(new Color4(1, 1, 1, 1));\r\n }\r\n\r\n for (let f = 0; f < goldbergData.face.length; f++) {\r\n const verts = goldbergData.face[f];\r\n const a = Vector3.FromArray(goldbergData.vertex[verts[0]]);\r\n const b = Vector3.FromArray(goldbergData.vertex[verts[2]]);\r\n const c = Vector3.FromArray(goldbergData.vertex[verts[1]]);\r\n const ba = b.subtract(a);\r\n const ca = c.subtract(a);\r\n const norm = Vector3.Cross(ca, ba).normalize();\r\n const z = Vector3.Cross(ca, norm).normalize();\r\n goldberg.goldbergData.faceXaxis.push(ca.normalize());\r\n goldberg.goldbergData.faceYaxis.push(norm);\r\n goldberg.goldbergData.faceZaxis.push(z);\r\n }\r\n\r\n return goldberg;\r\n}\r\n","import { Camera } from '@babylonjs/core/Cameras/camera';\r\nimport { DynamicTexture } from '@babylonjs/core/Materials/Textures/dynamicTexture';\r\nimport { StandardMaterial } from '@babylonjs/core/Materials/standardMaterial';\r\nimport { Color3, Color4, Vector3 } from '@babylonjs/core/Maths';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { MeshBuilder } from '@babylonjs/core/Meshes/meshBuilder';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { eAxis } from '@models/classes/enums';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { SceneNode } from '@models/classes/scene-node';\r\nimport { Highlighter } from '@view/modules/highlighter';\r\nimport { getCameraPositionWithOffset } from './math-helper';\r\n\r\nexport interface ITextLabelStyle {\r\n font?: string;\r\n}\r\nexport interface ITextLabel {\r\n id?: string;\r\n text: string;\r\n position: KbVector;\r\n style?: ITextLabelStyle;\r\n color: Color4;\r\n}\r\n\r\nexport class GuiHelper {\r\n private textLabelMap = new Map();\r\n private gridSettings: {\r\n gridMesh?: TransformNode;\r\n gridAxesMesh?: TransformNode;\r\n } = {};\r\n\r\n constructor(private scene: Scene, private highlighter: Highlighter) {\r\n const cameraPosition = new Vector3();\r\n scene.registerBeforeRender(() => {\r\n if (scene.activeCamera && this.textLabelMap.size > 0) {\r\n this.textLabelMap.forEach(mesh => {\r\n const camera = scene.activeCamera!;\r\n cameraPosition.copyFrom(getCameraPositionWithOffset(camera));\r\n mesh.lookAt(cameraPosition);\r\n if (camera.mode !== Camera.ORTHOGRAPHIC_CAMERA) {\r\n const size = cameraPosition.subtractInPlace(mesh.position).length() * 0.05;\r\n mesh.scaling = mesh.scaling.copyFromFloats(size, size, size);\r\n }\r\n });\r\n }\r\n });\r\n }\r\n\r\n enableGrid(sceneNode: SceneNode, skipOrigin = false) {\r\n if (this.gridSettings.gridMesh) {\r\n return;\r\n }\r\n const bgColor = sceneNode?.backgroundColor;\r\n let bgColor3: Color3;\r\n if (!bgColor) {\r\n bgColor3 = new Color3(this.scene.clearColor.r, this.scene.clearColor.g, this.scene.clearColor.b);\r\n } else {\r\n bgColor3 = bgColor.a === 0 ? Color3.Black() : new Color3(bgColor.r, bgColor.g, bgColor.b);\r\n }\r\n const gridStepSize = sceneNode.gridStepSize || 1;\r\n const gridSize = sceneNode.gridSize || 10;\r\n const gridNormal = sceneNode.gridNormal;\r\n const gridMajorInterval = sceneNode.gridMajorInterval || 0;\r\n\r\n const strongLineColor = Color4.FromColor3(bgColor3.toLuminance() > 0.5 ? Color3.Black() : Color3.White());\r\n const lightLineColor = strongLineColor.scale(0.4).add(Color4.FromColor3(bgColor3.scale(0.6)));\r\n\r\n const points = this.calculateGridPoints(gridStepSize, gridSize, gridNormal);\r\n const pointColors: Color4[][] = [];\r\n const linesPerSide = points.length / 2;\r\n for (let s = 0; s < 2; s++) {\r\n for (let i = 0; i < linesPerSide; i++) {\r\n if (gridMajorInterval === 0) {\r\n pointColors.push([lightLineColor, lightLineColor]);\r\n } else if (Math.ceil(i / 2) % gridMajorInterval === 0) {\r\n pointColors.push([strongLineColor, strongLineColor]);\r\n } else {\r\n pointColors.push([lightLineColor, lightLineColor]);\r\n }\r\n }\r\n }\r\n\r\n if (skipOrigin) {\r\n points.splice(linesPerSide, 1);\r\n points.splice(0, 1);\r\n pointColors.splice(linesPerSide, 1);\r\n pointColors.splice(0, 1);\r\n }\r\n\r\n const gridMesh = MeshBuilder.CreateLineSystem(\r\n 'grid',\r\n {\r\n lines: points,\r\n colors: pointColors,\r\n updatable: false,\r\n },\r\n this.scene\r\n );\r\n\r\n // gridMesh.color = lineColor.scale(0.4).add(bgColor3.scale(0.6));\r\n gridMesh.isPickable = false;\r\n\r\n this.gridSettings.gridMesh = gridMesh;\r\n }\r\n\r\n disableGrid() {\r\n if (this.gridSettings.gridMesh) {\r\n this.gridSettings.gridMesh.dispose();\r\n delete this.gridSettings.gridMesh;\r\n }\r\n }\r\n\r\n enableGridAxes(sceneNode: SceneNode) {\r\n if (this.gridSettings.gridAxesMesh) {\r\n return;\r\n }\r\n // const bgColor = sceneNode.backgroundColor;\r\n // let bgColor3: Color3;\r\n // if (!bgColor) {\r\n // bgColor3 = new Color3(this.scene.clearColor.r, this.scene.clearColor.g, this.scene.clearColor.b);\r\n // } else {\r\n // bgColor3 = bgColor.a === 0 ? Color3.Black() : new Color3(bgColor.r, bgColor.g, bgColor.b);\r\n // }\r\n\r\n // const lineColor = bgColor3.toLuminance() > 0.5 ? Color3.Black() : Color3.White();\r\n const gridStepSize = sceneNode.gridStepSize || 1;\r\n const gridSize = (sceneNode.gridSize || 10) * gridStepSize;\r\n\r\n const colors = [\r\n Color4.FromColor3(Color3.Red()),\r\n Color4.FromColor3(Color3.Green()),\r\n Color4.FromColor3(Color3.Blue()),\r\n ];\r\n const pointsAxes = [\r\n [new Vector3(-gridSize, 0, 0), new Vector3(gridSize, 0, 0)],\r\n [new Vector3(0, -gridSize, 0), new Vector3(0, gridSize, 0)],\r\n [new Vector3(0, 0, -gridSize), new Vector3(0, 0, gridSize)],\r\n ];\r\n\r\n const pointColors = [\r\n [colors[0], colors[0]],\r\n [colors[1], colors[1]],\r\n [colors[2], colors[2]],\r\n ];\r\n\r\n const gridMeshAxes = MeshBuilder.CreateLineSystem(\r\n 'gridAxes',\r\n {\r\n lines: pointsAxes,\r\n updatable: false,\r\n colors: pointColors,\r\n },\r\n this.scene\r\n );\r\n\r\n gridMeshAxes.isPickable = false;\r\n\r\n this.gridSettings.gridAxesMesh = gridMeshAxes;\r\n\r\n this.createTextLabels([\r\n { text: 'x', position: new KbVector(gridSize, 0.5, 0), color: colors[0] },\r\n { text: 'y', position: new KbVector(0.5, gridSize, 0), color: colors[1] },\r\n { text: 'z', position: new KbVector(0, 0.5, gridSize), color: colors[2] },\r\n ]);\r\n }\r\n\r\n disableGridAxes() {\r\n if (this.gridSettings.gridAxesMesh) {\r\n this.gridSettings.gridAxesMesh.dispose();\r\n delete this.gridSettings.gridAxesMesh;\r\n }\r\n }\r\n\r\n private createTextLabels(items: ITextLabel[], style: ITextLabelStyle = {}) {\r\n if (!style.font) {\r\n style.font = 'bold 32px sans-serif';\r\n }\r\n\r\n items.forEach(label => {\r\n const font = label.style && label.style.font ? label.style.font : style.font!;\r\n\r\n if (!label.id) {\r\n label.id = label.text;\r\n }\r\n\r\n if (this.textLabelMap.has(label.id)) {\r\n const labelMesh = this.textLabelMap.get(label.id)!;\r\n labelMesh.position = label.position.toVec3();\r\n } else {\r\n // Set height for plane\r\n const fontSize = parseInt(font.replace(/\\D+(\\d+)px.*/, '$1')) || 32;\r\n const planeHeight = fontSize / 32;\r\n const textureHeight = 2 * fontSize;\r\n const ratio = planeHeight / textureHeight;\r\n\r\n // Use a temporay canvas to calculate the length of the text\r\n const temp = document.createElement('canvas');\r\n temp.width = 64;\r\n temp.height = 64;\r\n const tmpctx = temp.getContext('2d');\r\n if (tmpctx) {\r\n tmpctx.font = font;\r\n const textureWidth = tmpctx.measureText(label.text).width + 8;\r\n temp.width = 0;\r\n temp.height = 0;\r\n\r\n // Create dynamic texture and write the text\r\n const dynamicId = `axisLabel${label.id}`;\r\n const planeWidth = textureWidth * ratio;\r\n const dynamicTexture = new DynamicTexture(\r\n dynamicId + '-labelTexture',\r\n { width: textureWidth, height: textureHeight },\r\n this.scene,\r\n false\r\n );\r\n const mat = new StandardMaterial(dynamicId + '-labelMaterial', this.scene);\r\n mat.emissiveTexture = dynamicTexture;\r\n mat.diffuseTexture = dynamicTexture;\r\n mat.sideOrientation = Mesh.BACKSIDE;\r\n dynamicTexture.hasAlpha = true;\r\n dynamicTexture.drawText(\r\n label.text,\r\n null,\r\n null,\r\n font,\r\n label.color.toHexString(),\r\n 'rgba(0,0,0,0)',\r\n true\r\n );\r\n\r\n // Create plane and set dynamic texture as material\r\n const plane = MeshBuilder.CreatePlane(\r\n dynamicId,\r\n { width: planeWidth, height: planeHeight },\r\n this.scene\r\n );\r\n plane.material = mat;\r\n plane.position = label.position.toVec3();\r\n plane.isPickable = false;\r\n\r\n this.textLabelMap.set(label.id, plane);\r\n }\r\n }\r\n });\r\n }\r\n\r\n private calculateGridPoints(stepSize: number, gridCount: number, normal: eAxis, gridFullSize?: number) {\r\n const points: Vector3[][] = [];\r\n const wV1 = new Vector3(),\r\n wV2 = new Vector3(),\r\n vStep = new Vector3();\r\n\r\n if (gridFullSize === undefined) {\r\n gridFullSize = gridCount;\r\n }\r\n\r\n let uVec: Vector3;\r\n let vVec: Vector3;\r\n switch (normal) {\r\n case eAxis.x:\r\n uVec = new Vector3(0, stepSize, 0);\r\n vVec = new Vector3(0, 0, stepSize);\r\n break;\r\n case eAxis.y:\r\n uVec = new Vector3(stepSize, 0, 0);\r\n vVec = new Vector3(0, 0, stepSize);\r\n break;\r\n case eAxis.z:\r\n uVec = new Vector3(stepSize, 0, 0);\r\n vVec = new Vector3(0, stepSize, 0);\r\n break;\r\n }\r\n\r\n wV1.setAll(0).addInPlace(vVec).scaleInPlace(-1).scaleInPlace(gridFullSize);\r\n wV2.setAll(0).scaleInPlace(-1).addInPlace(vVec).scaleInPlace(gridFullSize);\r\n for (let i = 0; i < gridCount * 2 + 1; i++) {\r\n const d = (2 * (i % 2) - 1) * Math.ceil(i / 2);\r\n vStep.copyFrom(uVec).scaleInPlace(d);\r\n points.push([wV1.add(vStep), wV2.add(vStep)]);\r\n }\r\n wV1.setAll(0).addInPlace(uVec).scaleInPlace(-1).scaleInPlace(gridFullSize);\r\n wV2.setAll(0).scaleInPlace(-1).addInPlace(uVec).scaleInPlace(gridFullSize);\r\n for (let i = 0; i < gridCount * 2 + 1; i++) {\r\n const d = (2 * (i % 2) - 1) * Math.ceil(i / 2);\r\n\r\n vStep.copyFrom(vVec).scaleInPlace(d);\r\n points.push([wV1.add(vStep), wV2.add(vStep)]);\r\n }\r\n\r\n return points;\r\n }\r\n}\r\n","import { Camera } from '@babylonjs/core/Cameras/camera';\r\nimport { Matrix, Vector3 } from '@babylonjs/core/Maths';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { WebXRCamera } from '@babylonjs/core/XR/webXRCamera';\r\n\r\nexport interface IScreenPosition {\r\n x: number;\r\n y: number;\r\n}\r\n\r\nexport interface IScreenBoundingBox extends IScreenPosition {\r\n top: number;\r\n left: number;\r\n width: number;\r\n height: number;\r\n}\r\n\r\n/**\r\n * Generic utilities for the babylon scene\r\n */\r\n\r\n/** returns the x,y pixel position of the node on the canvas */\r\nexport function nodeToScreen(bNode: TransformNode, camera: Camera) {\r\n camera = getCamera(camera);\r\n const nodeWorldPos = bNode.getAbsolutePosition(); // Vector3.TransformCoordinates(bNode.position, bNode.getWorldMatrix());\r\n return toScreen(nodeWorldPos, camera);\r\n}\r\n\r\n/** converts screen units */\r\nexport function toPixels(viewportUnits: IScreenPosition, camera: Camera) {\r\n camera = getCamera(camera);\r\n const canvasRect = camera.getEngine().getRenderingCanvasClientRect() as ClientRect;\r\n return {\r\n x: viewportUnits.x * canvasRect.width,\r\n y: viewportUnits.y * canvasRect.height,\r\n } as IScreenPosition;\r\n}\r\n\r\n/** returns the screen position as normalized device units a ratio from 0 to 1 */\r\nexport function toNDC(worldPos: Vector3, camera: Camera) {\r\n camera = getCamera(camera);\r\n const pos = Vector3.Project(worldPos, Matrix.IdentityReadOnly, camera.getTransformationMatrix(), camera.viewport);\r\n return {\r\n x: pos.x,\r\n y: pos.y,\r\n } as IScreenPosition;\r\n}\r\n\r\nexport function toScreen(worldPos: Vector3, camera: Camera) {\r\n camera = getCamera(camera);\r\n const scene = camera.getScene();\r\n const engine = scene.getEngine();\r\n const pos = Vector3.Project(\r\n worldPos,\r\n Matrix.IdentityReadOnly,\r\n scene.getTransformMatrix(),\r\n camera.viewport.toGlobal(engine.getRenderWidth(), engine.getRenderHeight())\r\n );\r\n return {\r\n x: pos.x,\r\n y: pos.y,\r\n } as IScreenPosition;\r\n}\r\n\r\nexport function getScreenBoundingBox(\r\n node: TransformNode,\r\n camera: Camera,\r\n includeDescendants = true\r\n): IScreenBoundingBox {\r\n camera = getCamera(camera);\r\n const bv = getHierarchyBoundingVectors(node, includeDescendants);\r\n return boundingVectorsToScreen(bv, camera);\r\n}\r\n\r\n// export function getBoundingVectors(bNode: AbstractMesh | TransformNode | undefined, scene: Scene, includeDescendants = true) {\r\n// let bv: { min: Vector3, max: Vector3 } | undefined;\r\n// if (!bNode) {\r\n// bv = scene.getWorldExtends();\r\n// } else {\r\n// bv = bNode.getHierarchyBoundingVectors(true);\r\n// if (bv.min.x == Number.MAX_VALUE && bv.min.y == Number.MAX_VALUE && bv.min.z == Number.MAX_VALUE && bv.max.x == -Number.MAX_VALUE && bv.max.y == -Number.MAX_VALUE && bv.max.z == -Number.MAX_VALUE) {\r\n// bv = {\r\n// min: bNode.absolutePosition.clone(),\r\n// max: bNode.absolutePosition.clone()\r\n// }\r\n// }\r\n// }\r\n\r\n// return bv;\r\n// }\r\n\r\n/**\r\n * Babylon's getHierarchyBoundingVectors has a bug that surfaces when called on a transformNode that has no real meshes in it's child hierarchy.\r\n * Use this wrapper instead\r\n * @param bNode\r\n * @param includeDescendants\r\n */\r\nexport function getHierarchyBoundingVectors(bNode: AbstractMesh | TransformNode, includeDescendants = true) {\r\n let bv = bNode.getHierarchyBoundingVectors(includeDescendants);\r\n //to handle a babylon bug that returns maxvalues when the node is an empty transformnode\r\n if (\r\n bv.min.x == Number.MAX_VALUE &&\r\n bv.min.y == Number.MAX_VALUE &&\r\n bv.min.z == Number.MAX_VALUE &&\r\n bv.max.x == -Number.MAX_VALUE &&\r\n bv.max.y == -Number.MAX_VALUE &&\r\n bv.max.z == -Number.MAX_VALUE\r\n ) {\r\n bv = {\r\n min: bNode.absolutePosition.clone(),\r\n max: bNode.absolutePosition.clone(),\r\n };\r\n }\r\n\r\n return bv;\r\n}\r\n\r\nexport function boundingVectorsToScreen(\r\n boundingVectors: { min: Vector3; max: Vector3 },\r\n camera: Camera\r\n): IScreenBoundingBox {\r\n camera = getCamera(camera);\r\n const bv = boundingVectors;\r\n const xs = new Array();\r\n const ys = new Array();\r\n\r\n for (let i = 0; i < 8; i++) {\r\n const x = bv[Math.floor(i / 4) ? 'max' : 'min'].x;\r\n const y = bv[Math.floor(i / 2) % 2 ? 'max' : 'min'].y;\r\n const z = bv[i % 2 ? 'max' : 'min'].z;\r\n const v = new Vector3(x, y, z);\r\n const sp = toNDC(v, camera);\r\n xs.push(sp.x);\r\n ys.push(sp.y);\r\n }\r\n\r\n const minx = Math.min(...xs);\r\n const maxx = Math.max(...xs);\r\n const miny = Math.min(...ys);\r\n const maxy = Math.max(...ys);\r\n\r\n const width = maxx - minx;\r\n const height = maxy - miny;\r\n const x = minx + width / 2;\r\n const y = miny + height / 2;\r\n const engine = camera.getEngine();\r\n const canvas = engine.getRenderingCanvas()!;\r\n const canvasWidth = canvas.width * engine.getHardwareScalingLevel();\r\n const canvasHeight = canvas.height * engine.getHardwareScalingLevel();\r\n\r\n return {\r\n x: x * canvasWidth,\r\n y: y * canvasHeight,\r\n top: miny * canvasHeight,\r\n left: minx * canvasWidth,\r\n width: width * canvasWidth,\r\n height: height * canvasHeight,\r\n } as IScreenBoundingBox;\r\n}\r\n\r\nexport function updateMatrixOfAncestorChain(ancestor: TransformNode, child: TransformNode) {\r\n while (child != ancestor && child instanceof TransformNode) {\r\n child.computeWorldMatrix(true);\r\n child = child.parent as TransformNode;\r\n }\r\n if (child) {\r\n child.computeWorldMatrix(true);\r\n }\r\n}\r\n\r\nfunction getCamera(camera: Camera) {\r\n if (camera instanceof WebXRCamera) {\r\n return camera.rigCameras[0];\r\n }\r\n return camera;\r\n}\r\n","import { Subject } from 'rxjs';\r\nimport { UUID } from '../models/classes/utilities';\r\n\r\nexport enum eWorkerTasks {\r\n usdConvertGeometry = 'usdConvertGeometry',\r\n usdPackage = 'usdPackageStart',\r\n}\r\n\r\nexport enum eKB3DMessageType {\r\n taskStart = 'taskStart',\r\n taskError = 'taskError',\r\n taskProgress = 'taskProgress',\r\n taskComplete = 'taskComplete',\r\n}\r\n\r\nexport interface IKB3DMessage {\r\n type: eKB3DMessageType;\r\n payload: T;\r\n}\r\n\r\nexport interface IKB3DTaskMessage extends IKB3DMessage {\r\n task: eWorkerTasks;\r\n taskId: string;\r\n}\r\n\r\nconst THREAD_COUNT = 2;\r\n\r\nexport class Kb3dWorker {\r\n workerPool: Worker[] = [];\r\n toggle = 0;\r\n requestsInFlight = new Map>();\r\n\r\n public static workerPath = '';\r\n\r\n constructor() {\r\n if (!Kb3dWorker.workerPath) {\r\n throw Error('Worker Script path not set');\r\n }\r\n const workerPath = Kb3dWorker.workerPath;\r\n\r\n const content = `importScripts( \"${workerPath}\" );`;\r\n const workerBlobUrl = URL.createObjectURL(new Blob([content], { type: 'text/javascript' }));\r\n\r\n const workerReturnMessage = (event: MessageEvent) => {\r\n if ('task' in event.data) {\r\n const data = event.data;\r\n const sub = this.requestsInFlight.get(data.taskId);\r\n\r\n if (sub) {\r\n // We have to remove the request from the inflight map first because the sub.next can trigger the dispose\r\n // But the sub.complete has to be after the sub.next\r\n if (data.type === eKB3DMessageType.taskComplete) {\r\n this.requestsInFlight.delete(data.taskId);\r\n }\r\n\r\n if (data.type === eKB3DMessageType.taskError) {\r\n sub.error(data.payload);\r\n } else {\r\n sub.next(data);\r\n }\r\n\r\n if (data.type === eKB3DMessageType.taskComplete) {\r\n sub.complete();\r\n }\r\n }\r\n }\r\n };\r\n\r\n for (let i = 0; i < THREAD_COUNT; i++) {\r\n const newWorker = new Worker(workerBlobUrl);\r\n newWorker.onmessage = workerReturnMessage;\r\n this.workerPool.push(newWorker);\r\n }\r\n URL.revokeObjectURL(workerBlobUrl);\r\n }\r\n\r\n startTask(task: eWorkerTasks, taskPayload: S, transfer?: ArrayBuffer[]) {\r\n const messagePayload: IKB3DTaskMessage = {\r\n type: eKB3DMessageType.taskStart,\r\n task,\r\n taskId: UUID.generate(),\r\n payload: taskPayload,\r\n };\r\n this.workerPool[this.toggle].postMessage(messagePayload, transfer ? { transfer } : undefined);\r\n\r\n const sub = new Subject>();\r\n this.requestsInFlight.set(messagePayload.taskId, sub);\r\n this.toggle = (this.toggle + 1) % THREAD_COUNT;\r\n\r\n return sub;\r\n }\r\n\r\n dispose() {\r\n this.workerPool.forEach(worker => {\r\n worker.terminate();\r\n });\r\n\r\n this.requestsInFlight.forEach(sub => {\r\n sub.error('Worker Terminated');\r\n sub.complete();\r\n });\r\n this.requestsInFlight.clear();\r\n }\r\n}\r\n","/*\r\n * Polyfill for createImageBitmap\r\n * https://developer.mozilla.org/en-US/docs/Web/API/WindowOrWorkerGlobalScope/createImageBitmap\r\n *\r\n * Supports CanvasImageSource (img, video, canvas) sources, Blobs, and ImageData.\r\n *\r\n * From:\r\n * - https://dev.to/nektro/createimagebitmap-polyfill-for-safari-and-edge-228\r\n * - https://gist.github.com/MonsieurV/fb640c29084c171b4444184858a91bc7\r\n * Updated by:\r\n * - Yoan Tournade \r\n * - diachedelic, https://gist.github.com/diachedelic\r\n * - Paul Ellis, https://pseudosavant.com\r\n */\r\n\r\n(function createImageBitmapIIFE(global: any) {\r\n function isCanvasImageSource(el: any) {\r\n const validElements = ['img', 'video', 'canvas'];\r\n\r\n return el && el.tagName && validElements.includes(el.tagName.toLowerCase());\r\n }\r\n\r\n function idealSize(currentValue: number, newValue: number, numerator: number, denominator: number) {\r\n if (typeof newValue === 'number') return newValue;\r\n if (typeof numerator !== 'number' || typeof denominator !== 'number') return currentValue;\r\n\r\n return (numerator / denominator) * currentValue;\r\n }\r\n\r\n if (!('createImageBitmap' in global)) {\r\n global.createImageBitmap = async function polyfillCreateImageBitmap(data: any, opts: any): Promise {\r\n return new Promise((resolve, reject) => {\r\n opts = opts || {};\r\n\r\n let dataURL;\r\n const canvas = document.createElement('canvas');\r\n\r\n try {\r\n const ctx = canvas.getContext('2d')!;\r\n\r\n if (data instanceof Blob) {\r\n dataURL = URL.createObjectURL(data);\r\n } else if (isCanvasImageSource(data)) {\r\n const width = data.naturalWidth || data.videoWidth || data.clientWidth || data.width;\r\n const height = data.naturalHeight || data.videoHeight || data.clientHeight || data.height;\r\n canvas.width = idealSize(width, opts.resizeWidth, opts.resizeHeight, height);\r\n canvas.height = idealSize(height, opts.resizeHeight, opts.resizeWidth, width);\r\n\r\n ctx.drawImage(data, 0, 0, canvas.width, canvas.height);\r\n\r\n dataURL = canvas.toDataURL();\r\n } else if (data instanceof ImageData) {\r\n canvas.width = idealSize(data.width, opts.resizeWidth, opts.resizeHeight, data.height);\r\n canvas.height = idealSize(data.height, opts.resizeHeight, opts.resizeWidth, data.width);\r\n\r\n ctx.putImageData(data, 0, 0);\r\n\r\n dataURL = canvas.toDataURL();\r\n } else {\r\n reject('createImageBitmap does not handle the provided image source type');\r\n }\r\n\r\n const img = new Image();\r\n img.onerror = reject;\r\n img.onload = () => resolve(img);\r\n img.src = dataURL as string;\r\n } finally {\r\n // avoid memory leaks on iOS Safari, see https://stackoverflow.com/a/52586606\r\n canvas.width = 0;\r\n canvas.height = 0;\r\n }\r\n });\r\n };\r\n }\r\n})(window);\r\n","var version = '18.5.0';\n\n/**\n * Tween.js - Licensed under the MIT license\n * https://github.com/tweenjs/tween.js\n * ----------------------------------------------\n *\n * See https://github.com/tweenjs/tween.js/graphs/contributors for the full list of contributors.\n * Thank you all, you're awesome!\n */\n\n\nvar _Group = function () {\n\tthis._tweens = {};\n\tthis._tweensAddedDuringUpdate = {};\n};\n\n_Group.prototype = {\n\tgetAll: function () {\n\n\t\treturn Object.keys(this._tweens).map(function (tweenId) {\n\t\t\treturn this._tweens[tweenId];\n\t\t}.bind(this));\n\n\t},\n\n\tremoveAll: function () {\n\n\t\tthis._tweens = {};\n\n\t},\n\n\tadd: function (tween) {\n\n\t\tthis._tweens[tween.getId()] = tween;\n\t\tthis._tweensAddedDuringUpdate[tween.getId()] = tween;\n\n\t},\n\n\tremove: function (tween) {\n\n\t\tdelete this._tweens[tween.getId()];\n\t\tdelete this._tweensAddedDuringUpdate[tween.getId()];\n\n\t},\n\n\tupdate: function (time, preserve) {\n\n\t\tvar tweenIds = Object.keys(this._tweens);\n\n\t\tif (tweenIds.length === 0) {\n\t\t\treturn false;\n\t\t}\n\n\t\ttime = time !== undefined ? time : TWEEN.now();\n\n\t\t// Tweens are updated in \"batches\". If you add a new tween during an\n\t\t// update, then the new tween will be updated in the next batch.\n\t\t// If you remove a tween during an update, it may or may not be updated.\n\t\t// However, if the removed tween was added during the current batch,\n\t\t// then it will not be updated.\n\t\twhile (tweenIds.length > 0) {\n\t\t\tthis._tweensAddedDuringUpdate = {};\n\n\t\t\tfor (var i = 0; i < tweenIds.length; i++) {\n\n\t\t\t\tvar tween = this._tweens[tweenIds[i]];\n\n\t\t\t\tif (tween && tween.update(time) === false) {\n\t\t\t\t\ttween._isPlaying = false;\n\n\t\t\t\t\tif (!preserve) {\n\t\t\t\t\t\tdelete this._tweens[tweenIds[i]];\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\ttweenIds = Object.keys(this._tweensAddedDuringUpdate);\n\t\t}\n\n\t\treturn true;\n\n\t}\n};\n\nvar TWEEN = new _Group();\n\nTWEEN.Group = _Group;\nTWEEN._nextId = 0;\nTWEEN.nextId = function () {\n\treturn TWEEN._nextId++;\n};\n\n\n// Include a performance.now polyfill.\n// In node.js, use process.hrtime.\nif (typeof (self) === 'undefined' && typeof (process) !== 'undefined' && process.hrtime) {\n\tTWEEN.now = function () {\n\t\tvar time = process.hrtime();\n\n\t\t// Convert [seconds, nanoseconds] to milliseconds.\n\t\treturn time[0] * 1000 + time[1] / 1000000;\n\t};\n}\n// In a browser, use self.performance.now if it is available.\nelse if (typeof (self) !== 'undefined' &&\n self.performance !== undefined &&\n\t\t self.performance.now !== undefined) {\n\t// This must be bound, because directly assigning this function\n\t// leads to an invocation exception in Chrome.\n\tTWEEN.now = self.performance.now.bind(self.performance);\n}\n// Use Date.now if it is available.\nelse if (Date.now !== undefined) {\n\tTWEEN.now = Date.now;\n}\n// Otherwise, use 'new Date().getTime()'.\nelse {\n\tTWEEN.now = function () {\n\t\treturn new Date().getTime();\n\t};\n}\n\n\nTWEEN.Tween = function (object, group) {\n\tthis._isPaused = false;\n\tthis._pauseStart = null;\n\tthis._object = object;\n\tthis._valuesStart = {};\n\tthis._valuesEnd = {};\n\tthis._valuesStartRepeat = {};\n\tthis._duration = 1000;\n\tthis._repeat = 0;\n\tthis._repeatDelayTime = undefined;\n\tthis._yoyo = false;\n\tthis._isPlaying = false;\n\tthis._reversed = false;\n\tthis._delayTime = 0;\n\tthis._startTime = null;\n\tthis._easingFunction = TWEEN.Easing.Linear.None;\n\tthis._interpolationFunction = TWEEN.Interpolation.Linear;\n\tthis._chainedTweens = [];\n\tthis._onStartCallback = null;\n\tthis._onStartCallbackFired = false;\n\tthis._onUpdateCallback = null;\n\tthis._onRepeatCallback = null;\n\tthis._onCompleteCallback = null;\n\tthis._onStopCallback = null;\n\tthis._group = group || TWEEN;\n\tthis._id = TWEEN.nextId();\n\n};\n\nTWEEN.Tween.prototype = {\n\tgetId: function () {\n\t\treturn this._id;\n\t},\n\n\tisPlaying: function () {\n\t\treturn this._isPlaying;\n\t},\n\n\tisPaused: function () {\n\t\treturn this._isPaused;\n\t},\n\n\tto: function (properties, duration) {\n\n\t\tthis._valuesEnd = Object.create(properties);\n\n\t\tif (duration !== undefined) {\n\t\t\tthis._duration = duration;\n\t\t}\n\n\t\treturn this;\n\n\t},\n\n\tduration: function duration(d) {\n\t\tthis._duration = d;\n\t\treturn this;\n\t},\n\n\tstart: function (time) {\n\n\t\tthis._group.add(this);\n\n\t\tthis._isPlaying = true;\n\n\t\tthis._isPaused = false;\n\n\t\tthis._onStartCallbackFired = false;\n\n\t\tthis._startTime = time !== undefined ? typeof time === 'string' ? TWEEN.now() + parseFloat(time) : time : TWEEN.now();\n\t\tthis._startTime += this._delayTime;\n\n\t\tfor (var property in this._valuesEnd) {\n\n\t\t\t// Check if an Array was provided as property value\n\t\t\tif (this._valuesEnd[property] instanceof Array) {\n\n\t\t\t\tif (this._valuesEnd[property].length === 0) {\n\t\t\t\t\tcontinue;\n\t\t\t\t}\n\n\t\t\t\t// Create a local copy of the Array with the start value at the front\n\t\t\t\tthis._valuesEnd[property] = [this._object[property]].concat(this._valuesEnd[property]);\n\n\t\t\t}\n\n\t\t\t// If `to()` specifies a property that doesn't exist in the source object,\n\t\t\t// we should not set that property in the object\n\t\t\tif (this._object[property] === undefined) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\t// Save the starting value, but only once.\n\t\t\tif (typeof(this._valuesStart[property]) === 'undefined') {\n\t\t\t\tthis._valuesStart[property] = this._object[property];\n\t\t\t}\n\n\t\t\tif ((this._valuesStart[property] instanceof Array) === false) {\n\t\t\t\tthis._valuesStart[property] *= 1.0; // Ensures we're using numbers, not strings\n\t\t\t}\n\n\t\t\tthis._valuesStartRepeat[property] = this._valuesStart[property] || 0;\n\n\t\t}\n\n\t\treturn this;\n\n\t},\n\n\tstop: function () {\n\n\t\tif (!this._isPlaying) {\n\t\t\treturn this;\n\t\t}\n\n\t\tthis._group.remove(this);\n\n\t\tthis._isPlaying = false;\n\n\t\tthis._isPaused = false;\n\n\t\tif (this._onStopCallback !== null) {\n\t\t\tthis._onStopCallback(this._object);\n\t\t}\n\n\t\tthis.stopChainedTweens();\n\t\treturn this;\n\n\t},\n\n\tend: function () {\n\n\t\tthis.update(Infinity);\n\t\treturn this;\n\n\t},\n\n\tpause: function(time) {\n\n\t\tif (this._isPaused || !this._isPlaying) {\n\t\t\treturn this;\n\t\t}\n\n\t\tthis._isPaused = true;\n\n\t\tthis._pauseStart = time === undefined ? TWEEN.now() : time;\n\n\t\tthis._group.remove(this);\n\n\t\treturn this;\n\n\t},\n\n\tresume: function(time) {\n\n\t\tif (!this._isPaused || !this._isPlaying) {\n\t\t\treturn this;\n\t\t}\n\n\t\tthis._isPaused = false;\n\n\t\tthis._startTime += (time === undefined ? TWEEN.now() : time)\n\t\t\t- this._pauseStart;\n\n\t\tthis._pauseStart = 0;\n\n\t\tthis._group.add(this);\n\n\t\treturn this;\n\n\t},\n\n\tstopChainedTweens: function () {\n\n\t\tfor (var i = 0, numChainedTweens = this._chainedTweens.length; i < numChainedTweens; i++) {\n\t\t\tthis._chainedTweens[i].stop();\n\t\t}\n\n\t},\n\n\tgroup: function (group) {\n\t\tthis._group = group;\n\t\treturn this;\n\t},\n\n\tdelay: function (amount) {\n\n\t\tthis._delayTime = amount;\n\t\treturn this;\n\n\t},\n\n\trepeat: function (times) {\n\n\t\tthis._repeat = times;\n\t\treturn this;\n\n\t},\n\n\trepeatDelay: function (amount) {\n\n\t\tthis._repeatDelayTime = amount;\n\t\treturn this;\n\n\t},\n\n\tyoyo: function (yoyo) {\n\n\t\tthis._yoyo = yoyo;\n\t\treturn this;\n\n\t},\n\n\teasing: function (easingFunction) {\n\n\t\tthis._easingFunction = easingFunction;\n\t\treturn this;\n\n\t},\n\n\tinterpolation: function (interpolationFunction) {\n\n\t\tthis._interpolationFunction = interpolationFunction;\n\t\treturn this;\n\n\t},\n\n\tchain: function () {\n\n\t\tthis._chainedTweens = arguments;\n\t\treturn this;\n\n\t},\n\n\tonStart: function (callback) {\n\n\t\tthis._onStartCallback = callback;\n\t\treturn this;\n\n\t},\n\n\tonUpdate: function (callback) {\n\n\t\tthis._onUpdateCallback = callback;\n\t\treturn this;\n\n\t},\n\n\tonRepeat: function onRepeat(callback) {\n\n\t\tthis._onRepeatCallback = callback;\n\t\treturn this;\n\n\t},\n\n\tonComplete: function (callback) {\n\n\t\tthis._onCompleteCallback = callback;\n\t\treturn this;\n\n\t},\n\n\tonStop: function (callback) {\n\n\t\tthis._onStopCallback = callback;\n\t\treturn this;\n\n\t},\n\n\tupdate: function (time) {\n\n\t\tvar property;\n\t\tvar elapsed;\n\t\tvar value;\n\n\t\tif (time < this._startTime) {\n\t\t\treturn true;\n\t\t}\n\n\t\tif (this._onStartCallbackFired === false) {\n\n\t\t\tif (this._onStartCallback !== null) {\n\t\t\t\tthis._onStartCallback(this._object);\n\t\t\t}\n\n\t\t\tthis._onStartCallbackFired = true;\n\t\t}\n\n\t\telapsed = (time - this._startTime) / this._duration;\n\t\telapsed = (this._duration === 0 || elapsed > 1) ? 1 : elapsed;\n\n\t\tvalue = this._easingFunction(elapsed);\n\n\t\tfor (property in this._valuesEnd) {\n\n\t\t\t// Don't update properties that do not exist in the source object\n\t\t\tif (this._valuesStart[property] === undefined) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tvar start = this._valuesStart[property] || 0;\n\t\t\tvar end = this._valuesEnd[property];\n\n\t\t\tif (end instanceof Array) {\n\n\t\t\t\tthis._object[property] = this._interpolationFunction(end, value);\n\n\t\t\t} else {\n\n\t\t\t\t// Parses relative end values with start as base (e.g.: +10, -3)\n\t\t\t\tif (typeof (end) === 'string') {\n\n\t\t\t\t\tif (end.charAt(0) === '+' || end.charAt(0) === '-') {\n\t\t\t\t\t\tend = start + parseFloat(end);\n\t\t\t\t\t} else {\n\t\t\t\t\t\tend = parseFloat(end);\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// Protect against non numeric properties.\n\t\t\t\tif (typeof (end) === 'number') {\n\t\t\t\t\tthis._object[property] = start + (end - start) * value;\n\t\t\t\t}\n\n\t\t\t}\n\n\t\t}\n\n\t\tif (this._onUpdateCallback !== null) {\n\t\t\tthis._onUpdateCallback(this._object, elapsed);\n\t\t}\n\n\t\tif (elapsed === 1) {\n\n\t\t\tif (this._repeat > 0) {\n\n\t\t\t\tif (isFinite(this._repeat)) {\n\t\t\t\t\tthis._repeat--;\n\t\t\t\t}\n\n\t\t\t\t// Reassign starting values, restart by making startTime = now\n\t\t\t\tfor (property in this._valuesStartRepeat) {\n\n\t\t\t\t\tif (typeof (this._valuesEnd[property]) === 'string') {\n\t\t\t\t\t\tthis._valuesStartRepeat[property] = this._valuesStartRepeat[property] + parseFloat(this._valuesEnd[property]);\n\t\t\t\t\t}\n\n\t\t\t\t\tif (this._yoyo) {\n\t\t\t\t\t\tvar tmp = this._valuesStartRepeat[property];\n\n\t\t\t\t\t\tthis._valuesStartRepeat[property] = this._valuesEnd[property];\n\t\t\t\t\t\tthis._valuesEnd[property] = tmp;\n\t\t\t\t\t}\n\n\t\t\t\t\tthis._valuesStart[property] = this._valuesStartRepeat[property];\n\n\t\t\t\t}\n\n\t\t\t\tif (this._yoyo) {\n\t\t\t\t\tthis._reversed = !this._reversed;\n\t\t\t\t}\n\n\t\t\t\tif (this._repeatDelayTime !== undefined) {\n\t\t\t\t\tthis._startTime = time + this._repeatDelayTime;\n\t\t\t\t} else {\n\t\t\t\t\tthis._startTime = time + this._delayTime;\n\t\t\t\t}\n\n\t\t\t\tif (this._onRepeatCallback !== null) {\n\t\t\t\t\tthis._onRepeatCallback(this._object);\n\t\t\t\t}\n\n\t\t\t\treturn true;\n\n\t\t\t} else {\n\n\t\t\t\tif (this._onCompleteCallback !== null) {\n\n\t\t\t\t\tthis._onCompleteCallback(this._object);\n\t\t\t\t}\n\n\t\t\t\tfor (var i = 0, numChainedTweens = this._chainedTweens.length; i < numChainedTweens; i++) {\n\t\t\t\t\t// Make the chained tweens start exactly at the time they should,\n\t\t\t\t\t// even if the `update()` method was called way past the duration of the tween\n\t\t\t\t\tthis._chainedTweens[i].start(this._startTime + this._duration);\n\t\t\t\t}\n\n\t\t\t\treturn false;\n\n\t\t\t}\n\n\t\t}\n\n\t\treturn true;\n\n\t}\n};\n\n\nTWEEN.Easing = {\n\n\tLinear: {\n\n\t\tNone: function (k) {\n\n\t\t\treturn k;\n\n\t\t}\n\n\t},\n\n\tQuadratic: {\n\n\t\tIn: function (k) {\n\n\t\t\treturn k * k;\n\n\t\t},\n\n\t\tOut: function (k) {\n\n\t\t\treturn k * (2 - k);\n\n\t\t},\n\n\t\tInOut: function (k) {\n\n\t\t\tif ((k *= 2) < 1) {\n\t\t\t\treturn 0.5 * k * k;\n\t\t\t}\n\n\t\t\treturn - 0.5 * (--k * (k - 2) - 1);\n\n\t\t}\n\n\t},\n\n\tCubic: {\n\n\t\tIn: function (k) {\n\n\t\t\treturn k * k * k;\n\n\t\t},\n\n\t\tOut: function (k) {\n\n\t\t\treturn --k * k * k + 1;\n\n\t\t},\n\n\t\tInOut: function (k) {\n\n\t\t\tif ((k *= 2) < 1) {\n\t\t\t\treturn 0.5 * k * k * k;\n\t\t\t}\n\n\t\t\treturn 0.5 * ((k -= 2) * k * k + 2);\n\n\t\t}\n\n\t},\n\n\tQuartic: {\n\n\t\tIn: function (k) {\n\n\t\t\treturn k * k * k * k;\n\n\t\t},\n\n\t\tOut: function (k) {\n\n\t\t\treturn 1 - (--k * k * k * k);\n\n\t\t},\n\n\t\tInOut: function (k) {\n\n\t\t\tif ((k *= 2) < 1) {\n\t\t\t\treturn 0.5 * k * k * k * k;\n\t\t\t}\n\n\t\t\treturn - 0.5 * ((k -= 2) * k * k * k - 2);\n\n\t\t}\n\n\t},\n\n\tQuintic: {\n\n\t\tIn: function (k) {\n\n\t\t\treturn k * k * k * k * k;\n\n\t\t},\n\n\t\tOut: function (k) {\n\n\t\t\treturn --k * k * k * k * k + 1;\n\n\t\t},\n\n\t\tInOut: function (k) {\n\n\t\t\tif ((k *= 2) < 1) {\n\t\t\t\treturn 0.5 * k * k * k * k * k;\n\t\t\t}\n\n\t\t\treturn 0.5 * ((k -= 2) * k * k * k * k + 2);\n\n\t\t}\n\n\t},\n\n\tSinusoidal: {\n\n\t\tIn: function (k) {\n\n\t\t\treturn 1 - Math.cos(k * Math.PI / 2);\n\n\t\t},\n\n\t\tOut: function (k) {\n\n\t\t\treturn Math.sin(k * Math.PI / 2);\n\n\t\t},\n\n\t\tInOut: function (k) {\n\n\t\t\treturn 0.5 * (1 - Math.cos(Math.PI * k));\n\n\t\t}\n\n\t},\n\n\tExponential: {\n\n\t\tIn: function (k) {\n\n\t\t\treturn k === 0 ? 0 : Math.pow(1024, k - 1);\n\n\t\t},\n\n\t\tOut: function (k) {\n\n\t\t\treturn k === 1 ? 1 : 1 - Math.pow(2, - 10 * k);\n\n\t\t},\n\n\t\tInOut: function (k) {\n\n\t\t\tif (k === 0) {\n\t\t\t\treturn 0;\n\t\t\t}\n\n\t\t\tif (k === 1) {\n\t\t\t\treturn 1;\n\t\t\t}\n\n\t\t\tif ((k *= 2) < 1) {\n\t\t\t\treturn 0.5 * Math.pow(1024, k - 1);\n\t\t\t}\n\n\t\t\treturn 0.5 * (- Math.pow(2, - 10 * (k - 1)) + 2);\n\n\t\t}\n\n\t},\n\n\tCircular: {\n\n\t\tIn: function (k) {\n\n\t\t\treturn 1 - Math.sqrt(1 - k * k);\n\n\t\t},\n\n\t\tOut: function (k) {\n\n\t\t\treturn Math.sqrt(1 - (--k * k));\n\n\t\t},\n\n\t\tInOut: function (k) {\n\n\t\t\tif ((k *= 2) < 1) {\n\t\t\t\treturn - 0.5 * (Math.sqrt(1 - k * k) - 1);\n\t\t\t}\n\n\t\t\treturn 0.5 * (Math.sqrt(1 - (k -= 2) * k) + 1);\n\n\t\t}\n\n\t},\n\n\tElastic: {\n\n\t\tIn: function (k) {\n\n\t\t\tif (k === 0) {\n\t\t\t\treturn 0;\n\t\t\t}\n\n\t\t\tif (k === 1) {\n\t\t\t\treturn 1;\n\t\t\t}\n\n\t\t\treturn -Math.pow(2, 10 * (k - 1)) * Math.sin((k - 1.1) * 5 * Math.PI);\n\n\t\t},\n\n\t\tOut: function (k) {\n\n\t\t\tif (k === 0) {\n\t\t\t\treturn 0;\n\t\t\t}\n\n\t\t\tif (k === 1) {\n\t\t\t\treturn 1;\n\t\t\t}\n\n\t\t\treturn Math.pow(2, -10 * k) * Math.sin((k - 0.1) * 5 * Math.PI) + 1;\n\n\t\t},\n\n\t\tInOut: function (k) {\n\n\t\t\tif (k === 0) {\n\t\t\t\treturn 0;\n\t\t\t}\n\n\t\t\tif (k === 1) {\n\t\t\t\treturn 1;\n\t\t\t}\n\n\t\t\tk *= 2;\n\n\t\t\tif (k < 1) {\n\t\t\t\treturn -0.5 * Math.pow(2, 10 * (k - 1)) * Math.sin((k - 1.1) * 5 * Math.PI);\n\t\t\t}\n\n\t\t\treturn 0.5 * Math.pow(2, -10 * (k - 1)) * Math.sin((k - 1.1) * 5 * Math.PI) + 1;\n\n\t\t}\n\n\t},\n\n\tBack: {\n\n\t\tIn: function (k) {\n\n\t\t\tvar s = 1.70158;\n\n\t\t\treturn k * k * ((s + 1) * k - s);\n\n\t\t},\n\n\t\tOut: function (k) {\n\n\t\t\tvar s = 1.70158;\n\n\t\t\treturn --k * k * ((s + 1) * k + s) + 1;\n\n\t\t},\n\n\t\tInOut: function (k) {\n\n\t\t\tvar s = 1.70158 * 1.525;\n\n\t\t\tif ((k *= 2) < 1) {\n\t\t\t\treturn 0.5 * (k * k * ((s + 1) * k - s));\n\t\t\t}\n\n\t\t\treturn 0.5 * ((k -= 2) * k * ((s + 1) * k + s) + 2);\n\n\t\t}\n\n\t},\n\n\tBounce: {\n\n\t\tIn: function (k) {\n\n\t\t\treturn 1 - TWEEN.Easing.Bounce.Out(1 - k);\n\n\t\t},\n\n\t\tOut: function (k) {\n\n\t\t\tif (k < (1 / 2.75)) {\n\t\t\t\treturn 7.5625 * k * k;\n\t\t\t} else if (k < (2 / 2.75)) {\n\t\t\t\treturn 7.5625 * (k -= (1.5 / 2.75)) * k + 0.75;\n\t\t\t} else if (k < (2.5 / 2.75)) {\n\t\t\t\treturn 7.5625 * (k -= (2.25 / 2.75)) * k + 0.9375;\n\t\t\t} else {\n\t\t\t\treturn 7.5625 * (k -= (2.625 / 2.75)) * k + 0.984375;\n\t\t\t}\n\n\t\t},\n\n\t\tInOut: function (k) {\n\n\t\t\tif (k < 0.5) {\n\t\t\t\treturn TWEEN.Easing.Bounce.In(k * 2) * 0.5;\n\t\t\t}\n\n\t\t\treturn TWEEN.Easing.Bounce.Out(k * 2 - 1) * 0.5 + 0.5;\n\n\t\t}\n\n\t}\n\n};\n\nTWEEN.Interpolation = {\n\n\tLinear: function (v, k) {\n\n\t\tvar m = v.length - 1;\n\t\tvar f = m * k;\n\t\tvar i = Math.floor(f);\n\t\tvar fn = TWEEN.Interpolation.Utils.Linear;\n\n\t\tif (k < 0) {\n\t\t\treturn fn(v[0], v[1], f);\n\t\t}\n\n\t\tif (k > 1) {\n\t\t\treturn fn(v[m], v[m - 1], m - f);\n\t\t}\n\n\t\treturn fn(v[i], v[i + 1 > m ? m : i + 1], f - i);\n\n\t},\n\n\tBezier: function (v, k) {\n\n\t\tvar b = 0;\n\t\tvar n = v.length - 1;\n\t\tvar pw = Math.pow;\n\t\tvar bn = TWEEN.Interpolation.Utils.Bernstein;\n\n\t\tfor (var i = 0; i <= n; i++) {\n\t\t\tb += pw(1 - k, n - i) * pw(k, i) * v[i] * bn(n, i);\n\t\t}\n\n\t\treturn b;\n\n\t},\n\n\tCatmullRom: function (v, k) {\n\n\t\tvar m = v.length - 1;\n\t\tvar f = m * k;\n\t\tvar i = Math.floor(f);\n\t\tvar fn = TWEEN.Interpolation.Utils.CatmullRom;\n\n\t\tif (v[0] === v[m]) {\n\n\t\t\tif (k < 0) {\n\t\t\t\ti = Math.floor(f = m * (1 + k));\n\t\t\t}\n\n\t\t\treturn fn(v[(i - 1 + m) % m], v[i], v[(i + 1) % m], v[(i + 2) % m], f - i);\n\n\t\t} else {\n\n\t\t\tif (k < 0) {\n\t\t\t\treturn v[0] - (fn(v[0], v[0], v[1], v[1], -f) - v[0]);\n\t\t\t}\n\n\t\t\tif (k > 1) {\n\t\t\t\treturn v[m] - (fn(v[m], v[m], v[m - 1], v[m - 1], f - m) - v[m]);\n\t\t\t}\n\n\t\t\treturn fn(v[i ? i - 1 : 0], v[i], v[m < i + 1 ? m : i + 1], v[m < i + 2 ? m : i + 2], f - i);\n\n\t\t}\n\n\t},\n\n\tUtils: {\n\n\t\tLinear: function (p0, p1, t) {\n\n\t\t\treturn (p1 - p0) * t + p0;\n\n\t\t},\n\n\t\tBernstein: function (n, i) {\n\n\t\t\tvar fc = TWEEN.Interpolation.Utils.Factorial;\n\n\t\t\treturn fc(n) / fc(i) / fc(n - i);\n\n\t\t},\n\n\t\tFactorial: (function () {\n\n\t\t\tvar a = [1];\n\n\t\t\treturn function (n) {\n\n\t\t\t\tvar s = 1;\n\n\t\t\t\tif (a[n]) {\n\t\t\t\t\treturn a[n];\n\t\t\t\t}\n\n\t\t\t\tfor (var i = n; i > 1; i--) {\n\t\t\t\t\ts *= i;\n\t\t\t\t}\n\n\t\t\t\ta[n] = s;\n\t\t\t\treturn s;\n\n\t\t\t};\n\n\t\t})(),\n\n\t\tCatmullRom: function (p0, p1, p2, p3, t) {\n\n\t\t\tvar v0 = (p2 - p0) * 0.5;\n\t\t\tvar v1 = (p3 - p1) * 0.5;\n\t\t\tvar t2 = t * t;\n\t\t\tvar t3 = t * t2;\n\n\t\t\treturn (2 * p1 - 2 * p2 + v0 + v1) * t3 + (- 3 * p1 + 3 * p2 - 2 * v0 - v1) * t2 + v0 * t + p1;\n\n\t\t}\n\n\t}\n\n};\nTWEEN.version = version;\n\nexport default TWEEN;\n","import { Scene } from '@babylonjs/core/scene';\r\nimport { eEasing } from '@models/classes/enums';\r\nimport { Kb3dObject } from '@models/classes/kb3d-object';\r\nimport { KbColor } from '@models/classes/primitives/kb-color';\r\nimport { KbQuaternion } from '@models/classes/primitives/kb-quaternion';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { Token } from '@models/classes/token';\r\nimport TWEEN from '@tweenjs/tween.js';\r\nimport { KbViewer } from '../kb-viewer';\r\n\r\nexport interface IAnimation {\r\n name?: string;\r\n /** the node to animate */\r\n node: Kb3dObject;\r\n /** the property of the node that will be animated */\r\n prop: string;\r\n /** the $type of the property value */\r\n propType?: string;\r\n /** the value to animate to */\r\n toValue: any;\r\n easing?: eEasing;\r\n /** the length of the animation */\r\n duration?: number;\r\n /** the number of iterations to run the animations. -1 for infinite */\r\n iterations?: number;\r\n /** the number of ms to wait before starting the animation */\r\n delay?: number;\r\n}\r\n\r\nexport interface IActiveAnimation {\r\n animation: IAnimation;\r\n tween: TWEEN.Tween;\r\n}\r\n\r\nexport class AnimationModule {\r\n constructor(private kbViewer: KbViewer, private scene: Scene) {}\r\n\r\n protected activeAnimations = new Set();\r\n\r\n public get animationsAreRunning() {\r\n return this.activeAnimations.size > 0;\r\n }\r\n\r\n public queue(a: IAnimation): Promise {\r\n if (!a.node) throw 'node required';\r\n if (a.toValue == null) throw 'value required';\r\n\r\n a = this.applyDefaults(a);\r\n\r\n return new Promise(resolve => {\r\n const origValue = (a.node as any)[a.prop];\r\n const origValueType = origValue?.$type;\r\n const val = this.valueIn(origValue);\r\n const toVal = this.valueIn(a.toValue);\r\n\r\n const tween = new TWEEN.Tween(val)\r\n .to(toVal, a.duration!)\r\n .delay(a.delay!)\r\n .repeat(a.iterations! < 0 ? Infinity : a.iterations! - 1)\r\n .easing(this.getTweenEasingFn(a.easing!))\r\n .onUpdate(() => {\r\n (a.node as any)[a.prop] = this.valueOut(val, origValueType);\r\n })\r\n .onComplete(() => {\r\n this.activeAnimations.delete(activeAnim);\r\n resolve();\r\n })\r\n .start();\r\n\r\n const activeAnim = { animation: a, tween };\r\n this.activeAnimations.add(activeAnim);\r\n this.kbViewer.animationWarningsNeedsUpdate.next(this.activeAnimations.size);\r\n });\r\n }\r\n\r\n public stop(name: string) {\r\n if (name != null) {\r\n const anims = [...this.activeAnimations.values()];\r\n for (const t of anims.filter(a => name.isEqual(a.animation.name))) {\r\n t.tween.stop();\r\n this.activeAnimations.delete(t);\r\n }\r\n }\r\n }\r\n\r\n public stopAll() {\r\n for (const a of this.activeAnimations) {\r\n a.tween.stop();\r\n }\r\n this.activeAnimations.clear();\r\n }\r\n\r\n /** should be called by the render loop */\r\n public update() {\r\n TWEEN.update(); //update all animations\r\n }\r\n\r\n protected applyDefaults(a: IAnimation) {\r\n return {\r\n duration: 1000,\r\n easing: eEasing.linear,\r\n iterations: 1,\r\n delay: 0,\r\n\r\n ...a,\r\n } as IAnimation;\r\n }\r\n\r\n protected valueIn(val: any) {\r\n if (val instanceof KbVector) {\r\n return val.toObject();\r\n } else if (val instanceof KbColor) {\r\n return val.toObject();\r\n } else if (val instanceof KbQuaternion) {\r\n return val.toObject();\r\n } else {\r\n return { value: val };\r\n }\r\n }\r\n\r\n protected valueOut(val: any, $type: string) {\r\n if (!$type) {\r\n return val.value;\r\n } else {\r\n if ($type == Token.KbVector.key) {\r\n return KbVector.From(val);\r\n } else if ($type == Token.KbColor.key) {\r\n return KbColor.From(val);\r\n } else if ($type == Token.KbQuaternion.key) {\r\n return KbQuaternion.From(val);\r\n } else {\r\n throw 'unrecognized type';\r\n }\r\n }\r\n }\r\n\r\n protected getTweenEasingFn(e: eEasing): (k: number) => number {\r\n const split = e.splitWords().map(s => s.capitalize());\r\n const inOut = split.length == 3 ? split[1] + split[2] : split.length == 2 ? split[1] : 'None';\r\n return (TWEEN.Easing as any)[split[0]][inOut];\r\n }\r\n}\r\n","import { WebXRFeatureName, WebXRFeaturesManager } from \"../webXRFeaturesManager\";\r\nimport type { WebXRSessionManager } from \"../webXRSessionManager\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport { Matrix, Vector3, Quaternion } from \"../../Maths/math.vector\";\r\nimport type { TransformNode } from \"../../Meshes/transformNode\";\r\nimport { WebXRAbstractFeature } from \"./WebXRAbstractFeature\";\r\nimport type { IWebXRHitResult } from \"./WebXRHitTest\";\r\nimport { Tools } from \"../../Misc/tools\";\r\n\r\n/**\r\n * Configuration options of the anchor system\r\n */\r\nexport interface IWebXRAnchorSystemOptions {\r\n /**\r\n * a node that will be used to convert local to world coordinates\r\n */\r\n worldParentNode?: TransformNode;\r\n\r\n /**\r\n * If set to true a reference of the created anchors will be kept until the next session starts\r\n * If not defined, anchors will be removed from the array when the feature is detached or the session ended.\r\n */\r\n doNotRemoveAnchorsOnSessionEnded?: boolean;\r\n}\r\n\r\n/**\r\n * A babylon container for an XR Anchor\r\n */\r\nexport interface IWebXRAnchor {\r\n /**\r\n * A babylon-assigned ID for this anchor\r\n */\r\n id: number;\r\n /**\r\n * Transformation matrix to apply to an object attached to this anchor\r\n */\r\n transformationMatrix: Matrix;\r\n /**\r\n * The native anchor object\r\n */\r\n xrAnchor: XRAnchor;\r\n\r\n /**\r\n * if defined, this object will be constantly updated by the anchor's position and rotation\r\n */\r\n attachedNode?: TransformNode;\r\n\r\n /**\r\n * Remove this anchor from the scene\r\n */\r\n remove(): void;\r\n}\r\n\r\n/**\r\n * An internal interface for a future (promise based) anchor\r\n */\r\ninterface IWebXRFutureAnchor {\r\n /**\r\n * The native anchor\r\n */\r\n nativeAnchor?: XRAnchor;\r\n /**\r\n * Was this request submitted to the xr frame?\r\n */\r\n submitted: boolean;\r\n /**\r\n * Was this promise resolved already?\r\n */\r\n resolved: boolean;\r\n /**\r\n * A resolve function\r\n */\r\n resolve: (xrAnchor: IWebXRAnchor) => void;\r\n /**\r\n * A reject function\r\n */\r\n reject: (msg?: string) => void;\r\n /**\r\n * The XR Transformation of the future anchor\r\n */\r\n xrTransformation: XRRigidTransform;\r\n}\r\n\r\nlet anchorIdProvider = 0;\r\n\r\n/**\r\n * An implementation of the anchor system for WebXR.\r\n * For further information see https://github.com/immersive-web/anchors/\r\n */\r\nexport class WebXRAnchorSystem extends WebXRAbstractFeature {\r\n private _lastFrameDetected: XRAnchorSet = new Set();\r\n\r\n private _trackedAnchors: Array = [];\r\n\r\n private _referenceSpaceForFrameAnchors: XRReferenceSpace;\r\n\r\n private _futureAnchors: IWebXRFutureAnchor[] = [];\r\n\r\n /**\r\n * The module's name\r\n */\r\n public static readonly Name = WebXRFeatureName.ANCHOR_SYSTEM;\r\n /**\r\n * The (Babylon) version of this module.\r\n * This is an integer representing the implementation version.\r\n * This number does not correspond to the WebXR specs version\r\n */\r\n public static readonly Version = 1;\r\n\r\n /**\r\n * Observers registered here will be executed when a new anchor was added to the session\r\n */\r\n public onAnchorAddedObservable: Observable = new Observable();\r\n /**\r\n * Observers registered here will be executed when an anchor was removed from the session\r\n */\r\n public onAnchorRemovedObservable: Observable = new Observable();\r\n /**\r\n * Observers registered here will be executed when an existing anchor updates\r\n * This can execute N times every frame\r\n */\r\n public onAnchorUpdatedObservable: Observable = new Observable();\r\n\r\n /**\r\n * Set the reference space to use for anchor creation, when not using a hit test.\r\n * Will default to the session's reference space if not defined\r\n */\r\n public set referenceSpaceForFrameAnchors(referenceSpace: XRReferenceSpace) {\r\n this._referenceSpaceForFrameAnchors = referenceSpace;\r\n }\r\n\r\n /**\r\n * constructs a new anchor system\r\n * @param _xrSessionManager an instance of WebXRSessionManager\r\n * @param _options configuration object for this feature\r\n */\r\n constructor(_xrSessionManager: WebXRSessionManager, private _options: IWebXRAnchorSystemOptions = {}) {\r\n super(_xrSessionManager);\r\n this.xrNativeFeatureName = \"anchors\";\r\n }\r\n\r\n private _tmpVector = new Vector3();\r\n private _tmpQuaternion = new Quaternion();\r\n\r\n private _populateTmpTransformation(position: Vector3, rotationQuaternion: Quaternion) {\r\n this._tmpVector.copyFrom(position);\r\n this._tmpQuaternion.copyFrom(rotationQuaternion);\r\n if (!this._xrSessionManager.scene.useRightHandedSystem) {\r\n this._tmpVector.z *= -1;\r\n this._tmpQuaternion.z *= -1;\r\n this._tmpQuaternion.w *= -1;\r\n }\r\n return {\r\n position: this._tmpVector,\r\n rotationQuaternion: this._tmpQuaternion,\r\n };\r\n }\r\n\r\n /**\r\n * Create a new anchor point using a hit test result at a specific point in the scene\r\n * An anchor is tracked only after it is added to the trackerAnchors in xrFrame. The promise returned here does not yet guaranty that.\r\n * Use onAnchorAddedObservable to get newly added anchors if you require tracking guaranty.\r\n *\r\n * @param hitTestResult The hit test result to use for this anchor creation\r\n * @param position an optional position offset for this anchor\r\n * @param rotationQuaternion an optional rotation offset for this anchor\r\n * @returns A promise that fulfills when babylon has created the corresponding WebXRAnchor object and tracking has begun\r\n */\r\n public async addAnchorPointUsingHitTestResultAsync(\r\n hitTestResult: IWebXRHitResult,\r\n position: Vector3 = new Vector3(),\r\n rotationQuaternion: Quaternion = new Quaternion()\r\n ): Promise {\r\n // convert to XR space (right handed) if needed\r\n this._populateTmpTransformation(position, rotationQuaternion);\r\n // the matrix that we'll use\r\n const m = new XRRigidTransform(\r\n { x: this._tmpVector.x, y: this._tmpVector.y, z: this._tmpVector.z },\r\n { x: this._tmpQuaternion.x, y: this._tmpQuaternion.y, z: this._tmpQuaternion.z, w: this._tmpQuaternion.w }\r\n );\r\n if (!hitTestResult.xrHitResult.createAnchor) {\r\n this.detach();\r\n throw new Error(\"Anchors not enabled in this environment/browser\");\r\n } else {\r\n try {\r\n const nativeAnchor = await hitTestResult.xrHitResult.createAnchor(m);\r\n return new Promise((resolve, reject) => {\r\n this._futureAnchors.push({\r\n nativeAnchor,\r\n resolved: false,\r\n submitted: true,\r\n xrTransformation: m,\r\n resolve,\r\n reject,\r\n });\r\n });\r\n } catch (error) {\r\n throw new Error(error);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Add a new anchor at a specific position and rotation\r\n * This function will add a new anchor per default in the next available frame. Unless forced, the createAnchor function\r\n * will be called in the next xrFrame loop to make sure that the anchor can be created correctly.\r\n * An anchor is tracked only after it is added to the trackerAnchors in xrFrame. The promise returned here does not yet guaranty that.\r\n * Use onAnchorAddedObservable to get newly added anchors if you require tracking guaranty.\r\n *\r\n * @param position the position in which to add an anchor\r\n * @param rotationQuaternion an optional rotation for the anchor transformation\r\n * @param forceCreateInCurrentFrame force the creation of this anchor in the current frame. Must be called inside xrFrame loop!\r\n * @returns A promise that fulfills when babylon has created the corresponding WebXRAnchor object and tracking has begun\r\n */\r\n public async addAnchorAtPositionAndRotationAsync(\r\n position: Vector3,\r\n rotationQuaternion: Quaternion = new Quaternion(),\r\n forceCreateInCurrentFrame = false\r\n ): Promise {\r\n // convert to XR space (right handed) if needed\r\n this._populateTmpTransformation(position, rotationQuaternion);\r\n // the matrix that we'll use\r\n const xrTransformation = new XRRigidTransform(\r\n { x: this._tmpVector.x, y: this._tmpVector.y, z: this._tmpVector.z },\r\n { x: this._tmpQuaternion.x, y: this._tmpQuaternion.y, z: this._tmpQuaternion.z, w: this._tmpQuaternion.w }\r\n );\r\n const xrAnchor =\r\n forceCreateInCurrentFrame && this.attached && this._xrSessionManager.currentFrame\r\n ? await this._createAnchorAtTransformation(xrTransformation, this._xrSessionManager.currentFrame)\r\n : undefined;\r\n // add the transformation to the future anchors list\r\n return new Promise((resolve, reject) => {\r\n this._futureAnchors.push({\r\n nativeAnchor: xrAnchor,\r\n resolved: false,\r\n submitted: false,\r\n xrTransformation,\r\n resolve,\r\n reject,\r\n });\r\n });\r\n }\r\n\r\n /**\r\n * Get the list of anchors currently being tracked by the system\r\n */\r\n public get anchors(): IWebXRAnchor[] {\r\n return this._trackedAnchors;\r\n }\r\n\r\n /**\r\n * detach this feature.\r\n * Will usually be called by the features manager\r\n *\r\n * @returns true if successful.\r\n */\r\n public detach(): boolean {\r\n if (!super.detach()) {\r\n return false;\r\n }\r\n\r\n if (!this._options.doNotRemoveAnchorsOnSessionEnded) {\r\n while (this._trackedAnchors.length) {\r\n const toRemove = this._trackedAnchors.pop();\r\n if (toRemove) {\r\n try {\r\n // try to natively remove it as well\r\n toRemove.remove();\r\n } catch (e) {\r\n // no-op\r\n }\r\n // as the xr frame loop is removed, we need to notify manually\r\n this.onAnchorRemovedObservable.notifyObservers(toRemove);\r\n }\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Dispose this feature and all of the resources attached\r\n */\r\n public dispose(): void {\r\n this._futureAnchors.length = 0;\r\n super.dispose();\r\n this.onAnchorAddedObservable.clear();\r\n this.onAnchorRemovedObservable.clear();\r\n this.onAnchorUpdatedObservable.clear();\r\n }\r\n\r\n protected _onXRFrame(frame: XRFrame) {\r\n if (!this.attached || !frame) {\r\n return;\r\n }\r\n\r\n const trackedAnchors = frame.trackedAnchors;\r\n if (trackedAnchors) {\r\n const toRemove = this._trackedAnchors\r\n .filter((anchor) => !trackedAnchors.has(anchor.xrAnchor))\r\n .map((anchor) => {\r\n const index = this._trackedAnchors.indexOf(anchor);\r\n return index;\r\n });\r\n let idxTracker = 0;\r\n toRemove.forEach((index) => {\r\n const anchor = this._trackedAnchors.splice(index - idxTracker, 1)[0];\r\n this.onAnchorRemovedObservable.notifyObservers(anchor);\r\n idxTracker++;\r\n });\r\n // now check for new ones\r\n trackedAnchors.forEach((xrAnchor) => {\r\n if (!this._lastFrameDetected.has(xrAnchor)) {\r\n const newAnchor: Partial = {\r\n id: anchorIdProvider++,\r\n xrAnchor: xrAnchor,\r\n remove: () => xrAnchor.delete(),\r\n };\r\n const anchor = this._updateAnchorWithXRFrame(xrAnchor, newAnchor, frame);\r\n this._trackedAnchors.push(anchor);\r\n this.onAnchorAddedObservable.notifyObservers(anchor);\r\n // search for the future anchor promise that matches this\r\n const results = this._futureAnchors.filter((futureAnchor) => futureAnchor.nativeAnchor === xrAnchor);\r\n const result = results[0];\r\n if (result) {\r\n result.resolve(anchor);\r\n result.resolved = true;\r\n }\r\n } else {\r\n const index = this._findIndexInAnchorArray(xrAnchor);\r\n const anchor = this._trackedAnchors[index];\r\n try {\r\n // anchors update every frame\r\n this._updateAnchorWithXRFrame(xrAnchor, anchor, frame);\r\n if (anchor.attachedNode) {\r\n anchor.attachedNode.rotationQuaternion = anchor.attachedNode.rotationQuaternion || new Quaternion();\r\n anchor.transformationMatrix.decompose(anchor.attachedNode.scaling, anchor.attachedNode.rotationQuaternion, anchor.attachedNode.position);\r\n }\r\n this.onAnchorUpdatedObservable.notifyObservers(anchor);\r\n } catch (e) {\r\n Tools.Warn(`Anchor could not be updated`);\r\n }\r\n }\r\n });\r\n this._lastFrameDetected = trackedAnchors;\r\n }\r\n\r\n // process future anchors\r\n this._futureAnchors.forEach((futureAnchor) => {\r\n if (!futureAnchor.resolved && !futureAnchor.submitted) {\r\n this._createAnchorAtTransformation(futureAnchor.xrTransformation, frame).then(\r\n (nativeAnchor) => {\r\n futureAnchor.nativeAnchor = nativeAnchor;\r\n },\r\n (error) => {\r\n futureAnchor.resolved = true;\r\n futureAnchor.reject(error);\r\n }\r\n );\r\n futureAnchor.submitted = true;\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * avoiding using Array.find for global support.\r\n * @param xrAnchor the plane to find in the array\r\n */\r\n private _findIndexInAnchorArray(xrAnchor: XRAnchor) {\r\n for (let i = 0; i < this._trackedAnchors.length; ++i) {\r\n if (this._trackedAnchors[i].xrAnchor === xrAnchor) {\r\n return i;\r\n }\r\n }\r\n return -1;\r\n }\r\n\r\n private _updateAnchorWithXRFrame(xrAnchor: XRAnchor, anchor: Partial, xrFrame: XRFrame): IWebXRAnchor {\r\n // matrix\r\n const pose = xrFrame.getPose(xrAnchor.anchorSpace, this._xrSessionManager.referenceSpace);\r\n if (pose) {\r\n const mat = anchor.transformationMatrix || new Matrix();\r\n Matrix.FromArrayToRef(pose.transform.matrix, 0, mat);\r\n if (!this._xrSessionManager.scene.useRightHandedSystem) {\r\n mat.toggleModelMatrixHandInPlace();\r\n }\r\n anchor.transformationMatrix = mat;\r\n if (!this._options.worldParentNode) {\r\n // Logger.Warn(\"Please provide a world parent node to apply world transformation\");\r\n } else {\r\n mat.multiplyToRef(this._options.worldParentNode.getWorldMatrix(), mat);\r\n }\r\n }\r\n\r\n return anchor;\r\n }\r\n\r\n private async _createAnchorAtTransformation(xrTransformation: XRRigidTransform, xrFrame: XRFrame) {\r\n if (xrFrame.createAnchor) {\r\n try {\r\n return xrFrame.createAnchor(xrTransformation, this._referenceSpaceForFrameAnchors ?? this._xrSessionManager.referenceSpace);\r\n } catch (error) {\r\n throw new Error(error);\r\n }\r\n } else {\r\n this.detach();\r\n throw new Error(\"Anchors are not enabled in your browser\");\r\n }\r\n }\r\n}\r\n\r\n// register the plugin\r\nWebXRFeaturesManager.AddWebXRFeature(\r\n WebXRAnchorSystem.Name,\r\n (xrSessionManager, options) => {\r\n return () => new WebXRAnchorSystem(xrSessionManager, options);\r\n },\r\n WebXRAnchorSystem.Version\r\n);\r\n","import { WebXRFeaturesManager, WebXRFeatureName } from \"../webXRFeaturesManager\";\r\nimport type { WebXRSessionManager } from \"../webXRSessionManager\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport { Vector3, Matrix, Quaternion } from \"../../Maths/math.vector\";\r\nimport { WebXRAbstractFeature } from \"./WebXRAbstractFeature\";\r\nimport type { IWebXRLegacyHitTestOptions, IWebXRLegacyHitResult, IWebXRHitTestFeature } from \"./WebXRHitTestLegacy\";\r\nimport { Tools } from \"../../Misc/tools\";\r\nimport type { Nullable } from \"../../types\";\r\n\r\n/**\r\n * Options used for hit testing (version 2)\r\n */\r\nexport interface IWebXRHitTestOptions extends IWebXRLegacyHitTestOptions {\r\n /**\r\n * Do not create a permanent hit test. Will usually be used when only\r\n * transient inputs are needed.\r\n */\r\n disablePermanentHitTest?: boolean;\r\n /**\r\n * Enable transient (for example touch-based) hit test inspections\r\n */\r\n enableTransientHitTest?: boolean;\r\n /**\r\n * Override the default transient hit test profile (generic-touchscreen).\r\n */\r\n transientHitTestProfile?: string;\r\n /**\r\n * Offset ray for the permanent hit test\r\n */\r\n offsetRay?: Vector3;\r\n /**\r\n * Offset ray for the transient hit test\r\n */\r\n transientOffsetRay?: Vector3;\r\n /**\r\n * Instead of using viewer space for hit tests, use the reference space defined in the session manager\r\n */\r\n useReferenceSpace?: boolean;\r\n\r\n /**\r\n * Override the default entity type(s) of the hit-test result\r\n */\r\n entityTypes?: XRHitTestTrackableType[];\r\n}\r\n\r\n/**\r\n * Interface defining the babylon result of hit-test\r\n */\r\nexport interface IWebXRHitResult extends IWebXRLegacyHitResult {\r\n /**\r\n * The input source that generated this hit test (if transient)\r\n */\r\n inputSource?: XRInputSource;\r\n /**\r\n * Is this a transient hit test\r\n */\r\n isTransient?: boolean;\r\n /**\r\n * Position of the hit test result\r\n */\r\n position: Vector3;\r\n /**\r\n * Rotation of the hit test result\r\n */\r\n rotationQuaternion: Quaternion;\r\n\r\n /**\r\n * The native hit test result\r\n */\r\n xrHitResult: XRHitTestResult;\r\n}\r\n\r\n/**\r\n * The currently-working hit-test module.\r\n * Hit test (or Ray-casting) is used to interact with the real world.\r\n * For further information read here - https://github.com/immersive-web/hit-test\r\n *\r\n * Tested on chrome (mobile) 80.\r\n */\r\nexport class WebXRHitTest extends WebXRAbstractFeature implements IWebXRHitTestFeature {\r\n private _tmpMat: Matrix = new Matrix();\r\n private _tmpPos: Vector3 = new Vector3();\r\n private _tmpQuat: Quaternion = new Quaternion();\r\n private _transientXrHitTestSource: Nullable;\r\n // in XR space z-forward is negative\r\n private _xrHitTestSource: Nullable;\r\n private _initHitTestSource = (referenceSpace: XRReferenceSpace) => {\r\n if (!referenceSpace) {\r\n return;\r\n }\r\n const offsetRay = new XRRay(this.options.offsetRay || {});\r\n const hitTestOptions: XRHitTestOptionsInit = {\r\n space: this.options.useReferenceSpace ? referenceSpace : this._xrSessionManager.viewerReferenceSpace,\r\n offsetRay: offsetRay,\r\n };\r\n if (this.options.entityTypes) {\r\n hitTestOptions.entityTypes = this.options.entityTypes;\r\n }\r\n if (!hitTestOptions.space) {\r\n Tools.Warn(\"waiting for viewer reference space to initialize\");\r\n return;\r\n }\r\n this._xrSessionManager.session.requestHitTestSource!(hitTestOptions).then((hitTestSource) => {\r\n if (this._xrHitTestSource) {\r\n this._xrHitTestSource.cancel();\r\n }\r\n this._xrHitTestSource = hitTestSource;\r\n });\r\n };\r\n\r\n /**\r\n * The module's name\r\n */\r\n public static readonly Name = WebXRFeatureName.HIT_TEST;\r\n /**\r\n * The (Babylon) version of this module.\r\n * This is an integer representing the implementation version.\r\n * This number does not correspond to the WebXR specs version\r\n */\r\n public static readonly Version = 2;\r\n\r\n /**\r\n * When set to true, each hit test will have its own position/rotation objects\r\n * When set to false, position and rotation objects will be reused for each hit test. It is expected that\r\n * the developers will clone them or copy them as they see fit.\r\n */\r\n public autoCloneTransformation: boolean = false;\r\n /**\r\n * Triggered when new babylon (transformed) hit test results are available\r\n * Note - this will be called when results come back from the device. It can be an empty array!!\r\n */\r\n public onHitTestResultObservable: Observable = new Observable();\r\n /**\r\n * Use this to temporarily pause hit test checks.\r\n */\r\n public paused: boolean = false;\r\n\r\n /**\r\n * Creates a new instance of the hit test feature\r\n * @param _xrSessionManager an instance of WebXRSessionManager\r\n * @param options options to use when constructing this feature\r\n */\r\n constructor(\r\n _xrSessionManager: WebXRSessionManager,\r\n /**\r\n * options to use when constructing this feature\r\n */\r\n public readonly options: IWebXRHitTestOptions = {}\r\n ) {\r\n super(_xrSessionManager);\r\n this.xrNativeFeatureName = \"hit-test\";\r\n Tools.Warn(\"Hit test is an experimental and unstable feature.\");\r\n }\r\n\r\n /**\r\n * attach this feature\r\n * Will usually be called by the features manager\r\n *\r\n * @returns true if successful.\r\n */\r\n public attach(): boolean {\r\n if (!super.attach()) {\r\n return false;\r\n }\r\n\r\n // Feature enabled, but not available\r\n if (!this._xrSessionManager.session.requestHitTestSource) {\r\n return false;\r\n }\r\n\r\n if (!this.options.disablePermanentHitTest) {\r\n if (this._xrSessionManager.referenceSpace) {\r\n this._initHitTestSource(this._xrSessionManager.referenceSpace);\r\n }\r\n this._xrSessionManager.onXRReferenceSpaceChanged.add(this._initHitTestSource);\r\n }\r\n if (this.options.enableTransientHitTest) {\r\n const offsetRay = new XRRay(this.options.transientOffsetRay || {});\r\n this._xrSessionManager.session.requestHitTestSourceForTransientInput!({\r\n profile: this.options.transientHitTestProfile || \"generic-touchscreen\",\r\n offsetRay,\r\n entityTypes: this.options.entityTypes,\r\n }).then((hitSource) => {\r\n this._transientXrHitTestSource = hitSource;\r\n });\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * detach this feature.\r\n * Will usually be called by the features manager\r\n *\r\n * @returns true if successful.\r\n */\r\n public detach(): boolean {\r\n if (!super.detach()) {\r\n return false;\r\n }\r\n if (this._xrHitTestSource) {\r\n this._xrHitTestSource.cancel();\r\n this._xrHitTestSource = null;\r\n }\r\n this._xrSessionManager.onXRReferenceSpaceChanged.removeCallback(this._initHitTestSource);\r\n if (this._transientXrHitTestSource) {\r\n this._transientXrHitTestSource.cancel();\r\n this._transientXrHitTestSource = null;\r\n }\r\n return true;\r\n }\r\n\r\n /**\r\n * Dispose this feature and all of the resources attached\r\n */\r\n public dispose(): void {\r\n super.dispose();\r\n this.onHitTestResultObservable.clear();\r\n }\r\n\r\n protected _onXRFrame(frame: XRFrame) {\r\n // make sure we do nothing if (async) not attached\r\n if (!this.attached || this.paused) {\r\n return;\r\n }\r\n\r\n if (this._xrHitTestSource) {\r\n const results = frame.getHitTestResults(this._xrHitTestSource);\r\n this._processWebXRHitTestResult(results);\r\n }\r\n if (this._transientXrHitTestSource) {\r\n const hitTestResultsPerInputSource = frame.getHitTestResultsForTransientInput(this._transientXrHitTestSource);\r\n\r\n hitTestResultsPerInputSource.forEach((resultsPerInputSource) => {\r\n this._processWebXRHitTestResult(resultsPerInputSource.results, resultsPerInputSource.inputSource);\r\n });\r\n }\r\n }\r\n\r\n private _processWebXRHitTestResult(hitTestResults: readonly XRHitTestResult[], inputSource?: XRInputSource) {\r\n const results: IWebXRHitResult[] = [];\r\n hitTestResults.forEach((hitTestResult) => {\r\n const pose = hitTestResult.getPose(this._xrSessionManager.referenceSpace);\r\n if (!pose) {\r\n return;\r\n }\r\n const pos = pose.transform.position;\r\n const quat = pose.transform.orientation;\r\n this._tmpPos.set(pos.x, pos.y, pos.z);\r\n this._tmpQuat.set(quat.x, quat.y, quat.z, quat.w);\r\n Matrix.FromFloat32ArrayToRefScaled(pose.transform.matrix, 0, 1, this._tmpMat);\r\n if (!this._xrSessionManager.scene.useRightHandedSystem) {\r\n this._tmpPos.z *= -1;\r\n this._tmpQuat.z *= -1;\r\n this._tmpQuat.w *= -1;\r\n this._tmpMat.toggleModelMatrixHandInPlace();\r\n }\r\n\r\n const result: IWebXRHitResult = {\r\n position: this.autoCloneTransformation ? this._tmpPos.clone() : this._tmpPos,\r\n rotationQuaternion: this.autoCloneTransformation ? this._tmpQuat.clone() : this._tmpQuat,\r\n transformationMatrix: this.autoCloneTransformation ? this._tmpMat.clone() : this._tmpMat,\r\n inputSource: inputSource,\r\n isTransient: !!inputSource,\r\n xrHitResult: hitTestResult,\r\n };\r\n results.push(result);\r\n });\r\n\r\n this.onHitTestResultObservable.notifyObservers(results);\r\n }\r\n}\r\n\r\n//register the plugin versions\r\nWebXRFeaturesManager.AddWebXRFeature(\r\n WebXRHitTest.Name,\r\n (xrSessionManager, options) => {\r\n return () => new WebXRHitTest(xrSessionManager, options);\r\n },\r\n WebXRHitTest.Version,\r\n false\r\n);\r\n","import { WebXRFeaturesManager, WebXRFeatureName } from \"../webXRFeaturesManager\";\r\nimport type { TransformNode } from \"../../Meshes/transformNode\";\r\nimport type { WebXRSessionManager } from \"../webXRSessionManager\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport { Vector3, Matrix } from \"../../Maths/math.vector\";\r\nimport { WebXRAbstractFeature } from \"./WebXRAbstractFeature\";\r\n\r\ndeclare const XRPlane: XRPlane;\r\n\r\n/**\r\n * Options used in the plane detector module\r\n */\r\nexport interface IWebXRPlaneDetectorOptions {\r\n /**\r\n * The node to use to transform the local results to world coordinates\r\n */\r\n worldParentNode?: TransformNode;\r\n /**\r\n * If set to true a reference of the created planes will be kept until the next session starts\r\n * If not defined, planes will be removed from the array when the feature is detached or the session ended.\r\n */\r\n doNotRemovePlanesOnSessionEnded?: boolean;\r\n /**\r\n * Preferred detector configuration, not all preferred options will be supported by all platforms.\r\n */\r\n preferredDetectorOptions?: XRGeometryDetectorOptions;\r\n}\r\n\r\n/**\r\n * A babylon interface for a WebXR plane.\r\n * A Plane is actually a polygon, built from N points in space\r\n *\r\n * Supported in chrome 79, not supported in canary 81 ATM\r\n */\r\nexport interface IWebXRPlane {\r\n /**\r\n * a babylon-assigned ID for this polygon\r\n */\r\n id: number;\r\n /**\r\n * an array of vector3 points in babylon space. right/left hand system is taken into account.\r\n */\r\n polygonDefinition: Array;\r\n /**\r\n * A transformation matrix to apply on the mesh that will be built using the polygonDefinition\r\n * Local vs. World are decided if worldParentNode was provided or not in the options when constructing the module\r\n */\r\n transformationMatrix: Matrix;\r\n /**\r\n * the native xr-plane object\r\n */\r\n xrPlane: XRPlane;\r\n}\r\n\r\nlet planeIdProvider = 0;\r\n\r\n/**\r\n * The plane detector is used to detect planes in the real world when in AR\r\n * For more information see https://github.com/immersive-web/real-world-geometry/\r\n */\r\nexport class WebXRPlaneDetector extends WebXRAbstractFeature {\r\n private _detectedPlanes: Array = [];\r\n private _enabled: boolean = false;\r\n private _lastFrameDetected: XRPlaneSet = new Set();\r\n\r\n /**\r\n * The module's name\r\n */\r\n public static readonly Name = WebXRFeatureName.PLANE_DETECTION;\r\n /**\r\n * The (Babylon) version of this module.\r\n * This is an integer representing the implementation version.\r\n * This number does not correspond to the WebXR specs version\r\n */\r\n public static readonly Version = 1;\r\n\r\n /**\r\n * Observers registered here will be executed when a new plane was added to the session\r\n */\r\n public onPlaneAddedObservable: Observable = new Observable();\r\n /**\r\n * Observers registered here will be executed when a plane is no longer detected in the session\r\n */\r\n public onPlaneRemovedObservable: Observable = new Observable();\r\n /**\r\n * Observers registered here will be executed when an existing plane updates (for example - expanded)\r\n * This can execute N times every frame\r\n */\r\n public onPlaneUpdatedObservable: Observable = new Observable();\r\n\r\n /**\r\n * construct a new Plane Detector\r\n * @param _xrSessionManager an instance of xr Session manager\r\n * @param _options configuration to use when constructing this feature\r\n */\r\n constructor(_xrSessionManager: WebXRSessionManager, private _options: IWebXRPlaneDetectorOptions = {}) {\r\n super(_xrSessionManager);\r\n this.xrNativeFeatureName = \"plane-detection\";\r\n if (this._xrSessionManager.session) {\r\n this._init();\r\n } else {\r\n this._xrSessionManager.onXRSessionInit.addOnce(() => {\r\n this._init();\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * detach this feature.\r\n * Will usually be called by the features manager\r\n *\r\n * @returns true if successful.\r\n */\r\n public detach(): boolean {\r\n if (!super.detach()) {\r\n return false;\r\n }\r\n\r\n if (!this._options.doNotRemovePlanesOnSessionEnded) {\r\n while (this._detectedPlanes.length) {\r\n const toRemove = this._detectedPlanes.pop();\r\n if (toRemove) {\r\n this.onPlaneRemovedObservable.notifyObservers(toRemove);\r\n }\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Dispose this feature and all of the resources attached\r\n */\r\n public dispose(): void {\r\n super.dispose();\r\n this.onPlaneAddedObservable.clear();\r\n this.onPlaneRemovedObservable.clear();\r\n this.onPlaneUpdatedObservable.clear();\r\n }\r\n\r\n /**\r\n * Check if the needed objects are defined.\r\n * This does not mean that the feature is enabled, but that the objects needed are well defined.\r\n */\r\n public isCompatible(): boolean {\r\n return typeof XRPlane !== \"undefined\";\r\n }\r\n\r\n protected _onXRFrame(frame: XRFrame) {\r\n if (!this.attached || !this._enabled || !frame) {\r\n return;\r\n }\r\n\r\n const detectedPlanes = frame.detectedPlanes || frame.worldInformation?.detectedPlanes;\r\n if (detectedPlanes) {\r\n // remove all planes that are not currently detected in the frame\r\n for (let planeIdx = 0; planeIdx < this._detectedPlanes.length; planeIdx++) {\r\n const plane = this._detectedPlanes[planeIdx];\r\n if (!detectedPlanes.has(plane.xrPlane)) {\r\n this._detectedPlanes.splice(planeIdx--, 1);\r\n this.onPlaneRemovedObservable.notifyObservers(plane);\r\n }\r\n }\r\n\r\n // now check for new ones\r\n detectedPlanes.forEach((xrPlane) => {\r\n if (!this._lastFrameDetected.has(xrPlane)) {\r\n const newPlane: Partial = {\r\n id: planeIdProvider++,\r\n xrPlane: xrPlane,\r\n polygonDefinition: [],\r\n };\r\n const plane = this._updatePlaneWithXRPlane(xrPlane, newPlane, frame);\r\n this._detectedPlanes.push(plane);\r\n this.onPlaneAddedObservable.notifyObservers(plane);\r\n } else {\r\n // updated?\r\n if (xrPlane.lastChangedTime === this._xrSessionManager.currentTimestamp) {\r\n const index = this._findIndexInPlaneArray(xrPlane);\r\n const plane = this._detectedPlanes[index];\r\n this._updatePlaneWithXRPlane(xrPlane, plane, frame);\r\n this.onPlaneUpdatedObservable.notifyObservers(plane);\r\n }\r\n }\r\n });\r\n this._lastFrameDetected = detectedPlanes;\r\n }\r\n }\r\n\r\n private _init() {\r\n const internalInit = () => {\r\n this._enabled = true;\r\n if (this._detectedPlanes.length) {\r\n this._detectedPlanes.length = 0;\r\n }\r\n };\r\n\r\n // Only supported by BabylonNative\r\n if (!!this._xrSessionManager.isNative && !!this._options.preferredDetectorOptions && !!this._xrSessionManager.session.trySetPreferredPlaneDetectorOptions) {\r\n this._xrSessionManager.session.trySetPreferredPlaneDetectorOptions(this._options.preferredDetectorOptions);\r\n }\r\n\r\n if (!this._xrSessionManager.session.updateWorldTrackingState) {\r\n internalInit();\r\n return;\r\n }\r\n this._xrSessionManager.session.updateWorldTrackingState({ planeDetectionState: { enabled: true } });\r\n internalInit();\r\n }\r\n\r\n private _updatePlaneWithXRPlane(xrPlane: XRPlane, plane: Partial, xrFrame: XRFrame): IWebXRPlane {\r\n plane.polygonDefinition = xrPlane.polygon.map((xrPoint) => {\r\n const rightHandedSystem = this._xrSessionManager.scene.useRightHandedSystem ? 1 : -1;\r\n return new Vector3(xrPoint.x, xrPoint.y, xrPoint.z * rightHandedSystem);\r\n });\r\n // matrix\r\n const pose = xrFrame.getPose(xrPlane.planeSpace, this._xrSessionManager.referenceSpace);\r\n if (pose) {\r\n const mat = plane.transformationMatrix || new Matrix();\r\n Matrix.FromArrayToRef(pose.transform.matrix, 0, mat);\r\n if (!this._xrSessionManager.scene.useRightHandedSystem) {\r\n mat.toggleModelMatrixHandInPlace();\r\n }\r\n plane.transformationMatrix = mat;\r\n if (this._options.worldParentNode) {\r\n mat.multiplyToRef(this._options.worldParentNode.getWorldMatrix(), mat);\r\n }\r\n }\r\n\r\n return plane;\r\n }\r\n\r\n /**\r\n * avoiding using Array.find for global support.\r\n * @param xrPlane the plane to find in the array\r\n */\r\n private _findIndexInPlaneArray(xrPlane: XRPlane) {\r\n for (let i = 0; i < this._detectedPlanes.length; ++i) {\r\n if (this._detectedPlanes[i].xrPlane === xrPlane) {\r\n return i;\r\n }\r\n }\r\n return -1;\r\n }\r\n}\r\n\r\n//register the plugin\r\nWebXRFeaturesManager.AddWebXRFeature(\r\n WebXRPlaneDetector.Name,\r\n (xrSessionManager, options) => {\r\n return () => new WebXRPlaneDetector(xrSessionManager, options);\r\n },\r\n WebXRPlaneDetector.Version\r\n);\r\n","import type { IAnimatable } from \"../Animations/animatable.interface\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport type { Nullable, FloatArray } from \"../types\";\r\nimport type { Scene } from \"../scene\";\r\nimport { EngineStore } from \"../Engines/engineStore\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport type { AnimationPropertiesOverride } from \"../Animations/animationPropertiesOverride\";\r\nimport { serialize, SerializationHelper } from \"../Misc/decorators\";\r\nimport { GetClass } from \"../Misc/typeStore\";\r\n\r\nimport type { Animation } from \"../Animations/animation\";\r\n\r\n/**\r\n * Defines a target to use with MorphTargetManager\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/morphTargets\r\n */\r\nexport class MorphTarget implements IAnimatable {\r\n /**\r\n * Gets or sets the list of animations\r\n */\r\n public animations = new Array();\r\n\r\n private _scene: Nullable;\r\n private _positions: Nullable = null;\r\n private _normals: Nullable = null;\r\n private _tangents: Nullable = null;\r\n private _uvs: Nullable = null;\r\n private _influence: number;\r\n private _uniqueId = 0;\r\n\r\n /**\r\n * Observable raised when the influence changes\r\n */\r\n public onInfluenceChanged = new Observable();\r\n\r\n /** @internal */\r\n public _onDataLayoutChanged = new Observable();\r\n\r\n /**\r\n * Gets or sets the influence of this target (ie. its weight in the overall morphing)\r\n */\r\n public get influence(): number {\r\n return this._influence;\r\n }\r\n\r\n public set influence(influence: number) {\r\n if (this._influence === influence) {\r\n return;\r\n }\r\n\r\n const previous = this._influence;\r\n this._influence = influence;\r\n\r\n if (this.onInfluenceChanged.hasObservers()) {\r\n this.onInfluenceChanged.notifyObservers(previous === 0 || influence === 0);\r\n }\r\n }\r\n\r\n /**\r\n * Gets or sets the id of the morph Target\r\n */\r\n @serialize()\r\n public id: string;\r\n\r\n private _animationPropertiesOverride: Nullable = null;\r\n\r\n /**\r\n * Gets or sets the animation properties override\r\n */\r\n public get animationPropertiesOverride(): Nullable {\r\n if (!this._animationPropertiesOverride && this._scene) {\r\n return this._scene.animationPropertiesOverride;\r\n }\r\n return this._animationPropertiesOverride;\r\n }\r\n\r\n public set animationPropertiesOverride(value: Nullable) {\r\n this._animationPropertiesOverride = value;\r\n }\r\n\r\n /**\r\n * Creates a new MorphTarget\r\n * @param name defines the name of the target\r\n * @param influence defines the influence to use\r\n * @param scene defines the scene the morphtarget belongs to\r\n */\r\n public constructor(\r\n /** defines the name of the target */\r\n public name: string,\r\n influence = 0,\r\n scene: Nullable = null\r\n ) {\r\n this._scene = scene || EngineStore.LastCreatedScene;\r\n this.influence = influence;\r\n\r\n if (this._scene) {\r\n this._uniqueId = this._scene.getUniqueId();\r\n }\r\n }\r\n\r\n /**\r\n * Gets the unique ID of this manager\r\n */\r\n public get uniqueId(): number {\r\n return this._uniqueId;\r\n }\r\n\r\n /**\r\n * Gets a boolean defining if the target contains position data\r\n */\r\n public get hasPositions(): boolean {\r\n return !!this._positions;\r\n }\r\n\r\n /**\r\n * Gets a boolean defining if the target contains normal data\r\n */\r\n public get hasNormals(): boolean {\r\n return !!this._normals;\r\n }\r\n\r\n /**\r\n * Gets a boolean defining if the target contains tangent data\r\n */\r\n public get hasTangents(): boolean {\r\n return !!this._tangents;\r\n }\r\n\r\n /**\r\n * Gets a boolean defining if the target contains texture coordinates data\r\n */\r\n public get hasUVs(): boolean {\r\n return !!this._uvs;\r\n }\r\n\r\n /**\r\n * Affects position data to this target\r\n * @param data defines the position data to use\r\n */\r\n public setPositions(data: Nullable) {\r\n const hadPositions = this.hasPositions;\r\n\r\n this._positions = data;\r\n\r\n if (hadPositions !== this.hasPositions) {\r\n this._onDataLayoutChanged.notifyObservers(undefined);\r\n }\r\n }\r\n\r\n /**\r\n * Gets the position data stored in this target\r\n * @returns a FloatArray containing the position data (or null if not present)\r\n */\r\n public getPositions(): Nullable {\r\n return this._positions;\r\n }\r\n\r\n /**\r\n * Affects normal data to this target\r\n * @param data defines the normal data to use\r\n */\r\n public setNormals(data: Nullable) {\r\n const hadNormals = this.hasNormals;\r\n\r\n this._normals = data;\r\n\r\n if (hadNormals !== this.hasNormals) {\r\n this._onDataLayoutChanged.notifyObservers(undefined);\r\n }\r\n }\r\n\r\n /**\r\n * Gets the normal data stored in this target\r\n * @returns a FloatArray containing the normal data (or null if not present)\r\n */\r\n public getNormals(): Nullable {\r\n return this._normals;\r\n }\r\n\r\n /**\r\n * Affects tangent data to this target\r\n * @param data defines the tangent data to use\r\n */\r\n public setTangents(data: Nullable) {\r\n const hadTangents = this.hasTangents;\r\n\r\n this._tangents = data;\r\n\r\n if (hadTangents !== this.hasTangents) {\r\n this._onDataLayoutChanged.notifyObservers(undefined);\r\n }\r\n }\r\n\r\n /**\r\n * Gets the tangent data stored in this target\r\n * @returns a FloatArray containing the tangent data (or null if not present)\r\n */\r\n public getTangents(): Nullable {\r\n return this._tangents;\r\n }\r\n\r\n /**\r\n * Affects texture coordinates data to this target\r\n * @param data defines the texture coordinates data to use\r\n */\r\n public setUVs(data: Nullable) {\r\n const hadUVs = this.hasUVs;\r\n\r\n this._uvs = data;\r\n\r\n if (hadUVs !== this.hasUVs) {\r\n this._onDataLayoutChanged.notifyObservers(undefined);\r\n }\r\n }\r\n\r\n /**\r\n * Gets the texture coordinates data stored in this target\r\n * @returns a FloatArray containing the texture coordinates data (or null if not present)\r\n */\r\n public getUVs(): Nullable {\r\n return this._uvs;\r\n }\r\n\r\n /**\r\n * Clone the current target\r\n * @returns a new MorphTarget\r\n */\r\n public clone(): MorphTarget {\r\n const newOne = SerializationHelper.Clone(() => new MorphTarget(this.name, this.influence, this._scene), this);\r\n\r\n newOne._positions = this._positions;\r\n newOne._normals = this._normals;\r\n newOne._tangents = this._tangents;\r\n newOne._uvs = this._uvs;\r\n\r\n return newOne;\r\n }\r\n\r\n /**\r\n * Serializes the current target into a Serialization object\r\n * @returns the serialized object\r\n */\r\n public serialize(): any {\r\n const serializationObject: any = {};\r\n\r\n serializationObject.name = this.name;\r\n serializationObject.influence = this.influence;\r\n\r\n serializationObject.positions = Array.prototype.slice.call(this.getPositions());\r\n if (this.id != null) {\r\n serializationObject.id = this.id;\r\n }\r\n if (this.hasNormals) {\r\n serializationObject.normals = Array.prototype.slice.call(this.getNormals());\r\n }\r\n if (this.hasTangents) {\r\n serializationObject.tangents = Array.prototype.slice.call(this.getTangents());\r\n }\r\n if (this.hasUVs) {\r\n serializationObject.uvs = Array.prototype.slice.call(this.getUVs());\r\n }\r\n\r\n // Animations\r\n SerializationHelper.AppendSerializedAnimations(this, serializationObject);\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Returns the string \"MorphTarget\"\r\n * @returns \"MorphTarget\"\r\n */\r\n public getClassName(): string {\r\n return \"MorphTarget\";\r\n }\r\n\r\n // Statics\r\n\r\n /**\r\n * Creates a new target from serialized data\r\n * @param serializationObject defines the serialized data to use\r\n * @param scene defines the hosting scene\r\n * @returns a new MorphTarget\r\n */\r\n public static Parse(serializationObject: any, scene?: Scene): MorphTarget {\r\n const result = new MorphTarget(serializationObject.name, serializationObject.influence);\r\n\r\n result.setPositions(serializationObject.positions);\r\n\r\n if (serializationObject.id != null) {\r\n result.id = serializationObject.id;\r\n }\r\n if (serializationObject.normals) {\r\n result.setNormals(serializationObject.normals);\r\n }\r\n if (serializationObject.tangents) {\r\n result.setTangents(serializationObject.tangents);\r\n }\r\n if (serializationObject.uvs) {\r\n result.setUVs(serializationObject.uvs);\r\n }\r\n\r\n // Animations\r\n if (serializationObject.animations) {\r\n for (let animationIndex = 0; animationIndex < serializationObject.animations.length; animationIndex++) {\r\n const parsedAnimation = serializationObject.animations[animationIndex];\r\n const internalClass = GetClass(\"BABYLON.Animation\");\r\n if (internalClass) {\r\n result.animations.push(internalClass.Parse(parsedAnimation));\r\n }\r\n }\r\n\r\n if (serializationObject.autoAnimate && scene) {\r\n scene.beginAnimation(\r\n result,\r\n serializationObject.autoAnimateFrom,\r\n serializationObject.autoAnimateTo,\r\n serializationObject.autoAnimateLoop,\r\n serializationObject.autoAnimateSpeed || 1.0\r\n );\r\n }\r\n }\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Creates a MorphTarget from mesh data\r\n * @param mesh defines the source mesh\r\n * @param name defines the name to use for the new target\r\n * @param influence defines the influence to attach to the target\r\n * @returns a new MorphTarget\r\n */\r\n public static FromMesh(mesh: AbstractMesh, name?: string, influence?: number): MorphTarget {\r\n if (!name) {\r\n name = mesh.name;\r\n }\r\n\r\n const result = new MorphTarget(name, influence, mesh.getScene());\r\n\r\n result.setPositions(mesh.getVerticesData(VertexBuffer.PositionKind));\r\n\r\n if (mesh.isVerticesDataPresent(VertexBuffer.NormalKind)) {\r\n result.setNormals(mesh.getVerticesData(VertexBuffer.NormalKind));\r\n }\r\n if (mesh.isVerticesDataPresent(VertexBuffer.TangentKind)) {\r\n result.setTangents(mesh.getVerticesData(VertexBuffer.TangentKind));\r\n }\r\n if (mesh.isVerticesDataPresent(VertexBuffer.UVKind)) {\r\n result.setUVs(mesh.getVerticesData(VertexBuffer.UVKind));\r\n }\r\n\r\n return result;\r\n }\r\n}\r\n","import type { IBufferView, AccessorComponentType, IAccessor } from \"babylonjs-gltf2interface\";\r\nimport { AccessorType } from \"babylonjs-gltf2interface\";\r\n\r\nimport type { FloatArray, Nullable } from \"core/types\";\r\nimport type { Vector4 } from \"core/Maths/math.vector\";\r\nimport { Vector3 } from \"core/Maths/math.vector\";\r\n\r\n/**\r\n * @internal\r\n */\r\nexport class _GLTFUtilities {\r\n /**\r\n * Creates a buffer view based on the supplied arguments\r\n * @param bufferIndex index value of the specified buffer\r\n * @param byteOffset byte offset value\r\n * @param byteLength byte length of the bufferView\r\n * @param byteStride byte distance between conequential elements\r\n * @param name name of the buffer view\r\n * @returns bufferView for glTF\r\n */\r\n public static _CreateBufferView(bufferIndex: number, byteOffset: number, byteLength: number, byteStride?: number, name?: string): IBufferView {\r\n const bufferview: IBufferView = { buffer: bufferIndex, byteLength: byteLength };\r\n if (byteOffset) {\r\n bufferview.byteOffset = byteOffset;\r\n }\r\n if (name) {\r\n bufferview.name = name;\r\n }\r\n if (byteStride) {\r\n bufferview.byteStride = byteStride;\r\n }\r\n\r\n return bufferview;\r\n }\r\n\r\n /**\r\n * Creates an accessor based on the supplied arguments\r\n * @param bufferviewIndex The index of the bufferview referenced by this accessor\r\n * @param name The name of the accessor\r\n * @param type The type of the accessor\r\n * @param componentType The datatype of components in the attribute\r\n * @param count The number of attributes referenced by this accessor\r\n * @param byteOffset The offset relative to the start of the bufferView in bytes\r\n * @param min Minimum value of each component in this attribute\r\n * @param max Maximum value of each component in this attribute\r\n * @returns accessor for glTF\r\n */\r\n public static _CreateAccessor(\r\n bufferviewIndex: number,\r\n name: string,\r\n type: AccessorType,\r\n componentType: AccessorComponentType,\r\n count: number,\r\n byteOffset: Nullable,\r\n min: Nullable,\r\n max: Nullable\r\n ): IAccessor {\r\n const accessor: IAccessor = { name: name, bufferView: bufferviewIndex, componentType: componentType, count: count, type: type };\r\n\r\n if (min != null) {\r\n accessor.min = min;\r\n }\r\n if (max != null) {\r\n accessor.max = max;\r\n }\r\n if (byteOffset != null) {\r\n accessor.byteOffset = byteOffset;\r\n }\r\n\r\n return accessor;\r\n }\r\n\r\n /**\r\n * Calculates the minimum and maximum values of an array of position floats\r\n * @param positions Positions array of a mesh\r\n * @param vertexStart Starting vertex offset to calculate min and max values\r\n * @param vertexCount Number of vertices to check for min and max values\r\n * @returns min number array and max number array\r\n */\r\n public static _CalculateMinMaxPositions(positions: FloatArray, vertexStart: number, vertexCount: number): { min: number[]; max: number[] } {\r\n const min = [Infinity, Infinity, Infinity];\r\n const max = [-Infinity, -Infinity, -Infinity];\r\n const positionStrideSize = 3;\r\n let indexOffset: number;\r\n let position: Vector3;\r\n let vector: number[];\r\n\r\n if (vertexCount) {\r\n for (let i = vertexStart, length = vertexStart + vertexCount; i < length; ++i) {\r\n indexOffset = positionStrideSize * i;\r\n\r\n position = Vector3.FromArray(positions, indexOffset);\r\n vector = position.asArray();\r\n\r\n for (let j = 0; j < positionStrideSize; ++j) {\r\n const num = vector[j];\r\n if (num < min[j]) {\r\n min[j] = num;\r\n }\r\n if (num > max[j]) {\r\n max[j] = num;\r\n }\r\n ++indexOffset;\r\n }\r\n }\r\n }\r\n return { min, max };\r\n }\r\n\r\n public static _NormalizeTangentFromRef(tangent: Vector4) {\r\n const length = Math.sqrt(tangent.x * tangent.x + tangent.y * tangent.y + tangent.z * tangent.z);\r\n if (length > 0) {\r\n tangent.x /= length;\r\n tangent.y /= length;\r\n tangent.z /= length;\r\n }\r\n }\r\n\r\n public static _GetDataAccessorElementCount(accessorType: AccessorType) {\r\n switch (accessorType) {\r\n case AccessorType.MAT2:\r\n return 4;\r\n case AccessorType.MAT3:\r\n return 9;\r\n case AccessorType.MAT4:\r\n return 16;\r\n case AccessorType.SCALAR:\r\n return 1;\r\n case AccessorType.VEC2:\r\n return 2;\r\n case AccessorType.VEC3:\r\n return 3;\r\n case AccessorType.VEC4:\r\n return 4;\r\n }\r\n }\r\n}\r\n","import type { IAnimation, INode, IBufferView, IAccessor, IAnimationSampler, IAnimationChannel } from \"babylonjs-gltf2interface\";\r\nimport { AnimationSamplerInterpolation, AnimationChannelTargetPath, AccessorType, AccessorComponentType } from \"babylonjs-gltf2interface\";\r\nimport type { Node } from \"core/node\";\r\nimport type { Nullable } from \"core/types\";\r\nimport { Vector3, Quaternion } from \"core/Maths/math.vector\";\r\nimport { Tools } from \"core/Misc/tools\";\r\nimport { Animation } from \"core/Animations/animation\";\r\nimport { TransformNode } from \"core/Meshes/transformNode\";\r\nimport type { Scene } from \"core/scene\";\r\nimport { MorphTarget } from \"core/Morph/morphTarget\";\r\nimport { Mesh } from \"core/Meshes/mesh\";\r\n\r\nimport type { _BinaryWriter } from \"./glTFExporter\";\r\nimport { _GLTFUtilities } from \"./glTFUtilities\";\r\nimport type { IAnimationKey } from \"core/Animations/animationKey\";\r\nimport { AnimationKeyInterpolation } from \"core/Animations/animationKey\";\r\n\r\nimport { Camera } from \"core/Cameras/camera\";\r\nimport { Light } from \"core/Lights/light\";\r\n\r\n/**\r\n * @internal\r\n * Interface to store animation data.\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport interface _IAnimationData {\r\n /**\r\n * Keyframe data.\r\n */\r\n inputs: number[];\r\n /**\r\n * Value data.\r\n */\r\n outputs: number[][];\r\n /**\r\n * Animation interpolation data.\r\n */\r\n samplerInterpolation: AnimationSamplerInterpolation;\r\n /**\r\n * Minimum keyframe value.\r\n */\r\n inputsMin: number;\r\n /**\r\n * Maximum keyframe value.\r\n */\r\n inputsMax: number;\r\n}\r\n\r\n/**\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport interface _IAnimationInfo {\r\n /**\r\n * The target channel for the animation\r\n */\r\n animationChannelTargetPath: AnimationChannelTargetPath;\r\n /**\r\n * The glTF accessor type for the data.\r\n */\r\n dataAccessorType: AccessorType.VEC3 | AccessorType.VEC4 | AccessorType.SCALAR;\r\n /**\r\n * Specifies if quaternions should be used.\r\n */\r\n useQuaternion: boolean;\r\n}\r\n\r\n/**\r\n * @internal\r\n * Enum for handling in tangent and out tangent.\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nenum _TangentType {\r\n /**\r\n * Specifies that input tangents are used.\r\n */\r\n INTANGENT,\r\n /**\r\n * Specifies that output tangents are used.\r\n */\r\n OUTTANGENT,\r\n}\r\n\r\n/**\r\n * @internal\r\n * Utility class for generating glTF animation data from BabylonJS.\r\n */\r\nexport class _GLTFAnimation {\r\n /**\r\n * Determine if a node is transformable - ie has properties it should be part of animation of transformation.\r\n * @param babylonNode the node to test\r\n * @returns true if can be animated, false otherwise. False if the parameter is null or undefined.\r\n */\r\n private static _IsTransformable(babylonNode: Node): boolean {\r\n return babylonNode && (babylonNode instanceof TransformNode || babylonNode instanceof Camera || babylonNode instanceof Light);\r\n }\r\n\r\n /**\r\n * @ignore\r\n *\r\n * Creates glTF channel animation from BabylonJS animation.\r\n * @param babylonTransformNode - BabylonJS mesh.\r\n * @param animation - animation.\r\n * @param animationChannelTargetPath - The target animation channel.\r\n * @param useQuaternion - Specifies if quaternions are used.\r\n * @returns nullable IAnimationData\r\n */\r\n public static _CreateNodeAnimation(\r\n babylonTransformNode: Node,\r\n animation: Animation,\r\n animationChannelTargetPath: AnimationChannelTargetPath,\r\n useQuaternion: boolean,\r\n animationSampleRate: number\r\n ): Nullable<_IAnimationData> {\r\n if (this._IsTransformable(babylonTransformNode)) {\r\n const inputs: number[] = [];\r\n const outputs: number[][] = [];\r\n const keyFrames = animation.getKeys();\r\n const minMaxKeyFrames = _GLTFAnimation._CalculateMinMaxKeyFrames(keyFrames);\r\n const interpolationOrBake = _GLTFAnimation._DeduceInterpolation(keyFrames, animationChannelTargetPath, useQuaternion);\r\n\r\n const interpolation = interpolationOrBake.interpolationType;\r\n const shouldBakeAnimation = interpolationOrBake.shouldBakeAnimation;\r\n\r\n if (shouldBakeAnimation) {\r\n _GLTFAnimation._CreateBakedAnimation(\r\n babylonTransformNode,\r\n animation,\r\n animationChannelTargetPath,\r\n minMaxKeyFrames.min,\r\n minMaxKeyFrames.max,\r\n animation.framePerSecond,\r\n animationSampleRate,\r\n inputs,\r\n outputs,\r\n minMaxKeyFrames,\r\n useQuaternion\r\n );\r\n } else {\r\n if (interpolation === AnimationSamplerInterpolation.LINEAR || interpolation === AnimationSamplerInterpolation.STEP) {\r\n _GLTFAnimation._CreateLinearOrStepAnimation(babylonTransformNode, animation, animationChannelTargetPath, inputs, outputs, useQuaternion);\r\n } else if (interpolation === AnimationSamplerInterpolation.CUBICSPLINE) {\r\n _GLTFAnimation._CreateCubicSplineAnimation(babylonTransformNode, animation, animationChannelTargetPath, inputs, outputs, useQuaternion);\r\n } else {\r\n _GLTFAnimation._CreateBakedAnimation(\r\n babylonTransformNode,\r\n animation,\r\n animationChannelTargetPath,\r\n minMaxKeyFrames.min,\r\n minMaxKeyFrames.max,\r\n animation.framePerSecond,\r\n animationSampleRate,\r\n inputs,\r\n outputs,\r\n minMaxKeyFrames,\r\n useQuaternion\r\n );\r\n }\r\n }\r\n\r\n if (inputs.length && outputs.length) {\r\n const result: _IAnimationData = {\r\n inputs: inputs,\r\n outputs: outputs,\r\n samplerInterpolation: interpolation,\r\n inputsMin: shouldBakeAnimation ? minMaxKeyFrames.min : Tools.FloatRound(minMaxKeyFrames.min / animation.framePerSecond),\r\n inputsMax: shouldBakeAnimation ? minMaxKeyFrames.max : Tools.FloatRound(minMaxKeyFrames.max / animation.framePerSecond),\r\n };\r\n\r\n return result;\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n private static _DeduceAnimationInfo(animation: Animation): Nullable<_IAnimationInfo> {\r\n let animationChannelTargetPath: Nullable = null;\r\n let dataAccessorType = AccessorType.VEC3;\r\n let useQuaternion: boolean = false;\r\n const property = animation.targetProperty.split(\".\");\r\n switch (property[0]) {\r\n case \"scaling\": {\r\n animationChannelTargetPath = AnimationChannelTargetPath.SCALE;\r\n break;\r\n }\r\n case \"position\": {\r\n animationChannelTargetPath = AnimationChannelTargetPath.TRANSLATION;\r\n break;\r\n }\r\n case \"rotation\": {\r\n dataAccessorType = AccessorType.VEC4;\r\n animationChannelTargetPath = AnimationChannelTargetPath.ROTATION;\r\n break;\r\n }\r\n case \"rotationQuaternion\": {\r\n dataAccessorType = AccessorType.VEC4;\r\n useQuaternion = true;\r\n animationChannelTargetPath = AnimationChannelTargetPath.ROTATION;\r\n break;\r\n }\r\n case \"influence\": {\r\n dataAccessorType = AccessorType.SCALAR;\r\n animationChannelTargetPath = AnimationChannelTargetPath.WEIGHTS;\r\n break;\r\n }\r\n default: {\r\n Tools.Error(`Unsupported animatable property ${property[0]}`);\r\n }\r\n }\r\n if (animationChannelTargetPath) {\r\n return { animationChannelTargetPath: animationChannelTargetPath, dataAccessorType: dataAccessorType, useQuaternion: useQuaternion };\r\n } else {\r\n Tools.Error(\"animation channel target path and data accessor type could be deduced\");\r\n }\r\n return null;\r\n }\r\n\r\n /**\r\n * @ignore\r\n * Create node animations from the transform node animations\r\n * @param babylonNode\r\n * @param runtimeGLTFAnimation\r\n * @param idleGLTFAnimations\r\n * @param nodeMap\r\n * @param nodes\r\n * @param binaryWriter\r\n * @param bufferViews\r\n * @param accessors\r\n * @param animationSampleRate\r\n */\r\n public static _CreateNodeAnimationFromNodeAnimations(\r\n babylonNode: Node,\r\n runtimeGLTFAnimation: IAnimation,\r\n idleGLTFAnimations: IAnimation[],\r\n nodeMap: { [key: number]: number },\r\n nodes: INode[],\r\n binaryWriter: _BinaryWriter,\r\n bufferViews: IBufferView[],\r\n accessors: IAccessor[],\r\n animationSampleRate: number,\r\n shouldExportAnimation?: (animation: Animation) => boolean\r\n ) {\r\n let glTFAnimation: IAnimation;\r\n if (_GLTFAnimation._IsTransformable(babylonNode)) {\r\n if (babylonNode.animations) {\r\n for (const animation of babylonNode.animations) {\r\n if (shouldExportAnimation && !shouldExportAnimation(animation)) {\r\n continue;\r\n }\r\n const animationInfo = _GLTFAnimation._DeduceAnimationInfo(animation);\r\n if (animationInfo) {\r\n glTFAnimation = {\r\n name: animation.name,\r\n samplers: [],\r\n channels: [],\r\n };\r\n _GLTFAnimation._AddAnimation(\r\n `${animation.name}`,\r\n animation.hasRunningRuntimeAnimations ? runtimeGLTFAnimation : glTFAnimation,\r\n babylonNode,\r\n animation,\r\n animationInfo.dataAccessorType,\r\n animationInfo.animationChannelTargetPath,\r\n nodeMap,\r\n binaryWriter,\r\n bufferViews,\r\n accessors,\r\n animationInfo.useQuaternion,\r\n animationSampleRate\r\n );\r\n if (glTFAnimation.samplers.length && glTFAnimation.channels.length) {\r\n idleGLTFAnimations.push(glTFAnimation);\r\n }\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * @ignore\r\n * Create individual morph animations from the mesh's morph target animation tracks\r\n * @param babylonNode\r\n * @param runtimeGLTFAnimation\r\n * @param idleGLTFAnimations\r\n * @param nodeMap\r\n * @param nodes\r\n * @param binaryWriter\r\n * @param bufferViews\r\n * @param accessors\r\n * @param animationSampleRate\r\n */\r\n public static _CreateMorphTargetAnimationFromMorphTargetAnimations(\r\n babylonNode: Node,\r\n runtimeGLTFAnimation: IAnimation,\r\n idleGLTFAnimations: IAnimation[],\r\n nodeMap: { [key: number]: number },\r\n nodes: INode[],\r\n binaryWriter: _BinaryWriter,\r\n bufferViews: IBufferView[],\r\n accessors: IAccessor[],\r\n animationSampleRate: number,\r\n shouldExportAnimation?: (animation: Animation) => boolean\r\n ) {\r\n let glTFAnimation: IAnimation;\r\n if (babylonNode instanceof Mesh) {\r\n const morphTargetManager = babylonNode.morphTargetManager;\r\n if (morphTargetManager) {\r\n for (let i = 0; i < morphTargetManager.numTargets; ++i) {\r\n const morphTarget = morphTargetManager.getTarget(i);\r\n for (const animation of morphTarget.animations) {\r\n if (shouldExportAnimation && !shouldExportAnimation(animation)) {\r\n continue;\r\n }\r\n const combinedAnimation = new Animation(\r\n `${animation.name}`,\r\n \"influence\",\r\n animation.framePerSecond,\r\n animation.dataType,\r\n animation.loopMode,\r\n animation.enableBlending\r\n );\r\n const combinedAnimationKeys: IAnimationKey[] = [];\r\n const animationKeys = animation.getKeys();\r\n\r\n for (let j = 0; j < animationKeys.length; ++j) {\r\n const animationKey = animationKeys[j];\r\n for (let k = 0; k < morphTargetManager.numTargets; ++k) {\r\n if (k == i) {\r\n combinedAnimationKeys.push(animationKey);\r\n } else {\r\n combinedAnimationKeys.push({ frame: animationKey.frame, value: 0 });\r\n }\r\n }\r\n }\r\n combinedAnimation.setKeys(combinedAnimationKeys);\r\n const animationInfo = _GLTFAnimation._DeduceAnimationInfo(combinedAnimation);\r\n if (animationInfo) {\r\n glTFAnimation = {\r\n name: combinedAnimation.name,\r\n samplers: [],\r\n channels: [],\r\n };\r\n _GLTFAnimation._AddAnimation(\r\n animation.name,\r\n animation.hasRunningRuntimeAnimations ? runtimeGLTFAnimation : glTFAnimation,\r\n babylonNode,\r\n combinedAnimation,\r\n animationInfo.dataAccessorType,\r\n animationInfo.animationChannelTargetPath,\r\n nodeMap,\r\n binaryWriter,\r\n bufferViews,\r\n accessors,\r\n animationInfo.useQuaternion,\r\n animationSampleRate,\r\n morphTargetManager.numTargets\r\n );\r\n if (glTFAnimation.samplers.length && glTFAnimation.channels.length) {\r\n idleGLTFAnimations.push(glTFAnimation);\r\n }\r\n }\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * @ignore\r\n * Create node and morph animations from the animation groups\r\n * @param babylonScene\r\n * @param glTFAnimations\r\n * @param nodeMap\r\n * @param nodes\r\n * @param binaryWriter\r\n * @param bufferViews\r\n * @param accessors\r\n * @param animationSampleRate\r\n */\r\n public static _CreateNodeAndMorphAnimationFromAnimationGroups(\r\n babylonScene: Scene,\r\n glTFAnimations: IAnimation[],\r\n nodeMap: { [key: number]: number },\r\n binaryWriter: _BinaryWriter,\r\n bufferViews: IBufferView[],\r\n accessors: IAccessor[],\r\n animationSampleRate: number,\r\n shouldExportAnimation?: (animation: Animation) => boolean\r\n ) {\r\n let glTFAnimation: IAnimation;\r\n if (babylonScene.animationGroups) {\r\n const animationGroups = babylonScene.animationGroups;\r\n for (const animationGroup of animationGroups) {\r\n const morphAnimations: Map> = new Map();\r\n const sampleAnimations: Map = new Map();\r\n const morphAnimationMeshes: Set = new Set();\r\n const animationGroupFrameDiff = animationGroup.to - animationGroup.from;\r\n glTFAnimation = {\r\n name: animationGroup.name,\r\n channels: [],\r\n samplers: [],\r\n };\r\n for (let i = 0; i < animationGroup.targetedAnimations.length; ++i) {\r\n const targetAnimation = animationGroup.targetedAnimations[i];\r\n const target = targetAnimation.target;\r\n const animation = targetAnimation.animation;\r\n if (shouldExportAnimation && !shouldExportAnimation(animation)) {\r\n continue;\r\n }\r\n if (this._IsTransformable(target) || (target.length === 1 && this._IsTransformable(target[0]))) {\r\n const animationInfo = _GLTFAnimation._DeduceAnimationInfo(targetAnimation.animation);\r\n if (animationInfo) {\r\n const babylonTransformNode = this._IsTransformable(target) ? target : this._IsTransformable(target[0]) ? target[0] : null;\r\n if (babylonTransformNode) {\r\n _GLTFAnimation._AddAnimation(\r\n `${animation.name}`,\r\n glTFAnimation,\r\n babylonTransformNode,\r\n animation,\r\n animationInfo.dataAccessorType,\r\n animationInfo.animationChannelTargetPath,\r\n nodeMap,\r\n binaryWriter,\r\n bufferViews,\r\n accessors,\r\n animationInfo.useQuaternion,\r\n animationSampleRate\r\n );\r\n }\r\n }\r\n } else if (target instanceof MorphTarget || (target.length === 1 && target[0] instanceof MorphTarget)) {\r\n const animationInfo = _GLTFAnimation._DeduceAnimationInfo(targetAnimation.animation);\r\n if (animationInfo) {\r\n const babylonMorphTarget = target instanceof MorphTarget ? (target as MorphTarget) : (target[0] as MorphTarget);\r\n if (babylonMorphTarget) {\r\n const babylonMorphTargetManager = babylonScene.morphTargetManagers.find((morphTargetManager) => {\r\n for (let j = 0; j < morphTargetManager.numTargets; ++j) {\r\n if (morphTargetManager.getTarget(j) === babylonMorphTarget) {\r\n return true;\r\n }\r\n }\r\n return false;\r\n });\r\n if (babylonMorphTargetManager) {\r\n const babylonMesh = babylonScene.meshes.find((mesh) => {\r\n return (mesh as Mesh).morphTargetManager === babylonMorphTargetManager;\r\n }) as Mesh;\r\n if (babylonMesh) {\r\n if (!morphAnimations.has(babylonMesh)) {\r\n morphAnimations.set(babylonMesh, new Map());\r\n }\r\n morphAnimations.get(babylonMesh)?.set(babylonMorphTarget, animation);\r\n morphAnimationMeshes.add(babylonMesh);\r\n sampleAnimations.set(babylonMesh, animation);\r\n }\r\n }\r\n }\r\n }\r\n } else {\r\n // this is the place for the KHR_animation_pointer.\r\n }\r\n }\r\n morphAnimationMeshes.forEach((mesh) => {\r\n const morphTargetManager = mesh.morphTargetManager!;\r\n let combinedAnimationGroup: Nullable = null;\r\n const animationKeys: IAnimationKey[] = [];\r\n const sampleAnimation = sampleAnimations.get(mesh)!;\r\n const sampleAnimationKeys = sampleAnimation.getKeys();\r\n const numAnimationKeys = sampleAnimationKeys.length;\r\n /*\r\n Due to how glTF expects morph target animation data to be formatted, we need to rearrange the individual morph target animation tracks,\r\n such that we have a single animation, where a given keyframe input value has successive output values for each morph target belonging to the manager.\r\n See: https://github.com/KhronosGroup/glTF/tree/master/specification/2.0#animations\r\n\r\n We do this via constructing a new Animation track, and interleaving the frames of each morph target animation track in the current Animation Group\r\n We reuse the Babylon Animation data structure for ease of handling export of cubic spline animation keys, and to reuse the\r\n existing _GLTFAnimation.AddAnimation codepath with minimal modification, however the constructed Babylon Animation is NOT intended for use in-engine.\r\n */\r\n for (let i = 0; i < numAnimationKeys; ++i) {\r\n for (let j = 0; j < morphTargetManager.numTargets; ++j) {\r\n const morphTarget = morphTargetManager.getTarget(j);\r\n const animationsByMorphTarget = morphAnimations.get(mesh);\r\n if (animationsByMorphTarget) {\r\n const morphTargetAnimation = animationsByMorphTarget.get(morphTarget);\r\n if (morphTargetAnimation) {\r\n if (!combinedAnimationGroup) {\r\n combinedAnimationGroup = new Animation(\r\n `${animationGroup.name}_${mesh.name}_MorphWeightAnimation`,\r\n \"influence\",\r\n morphTargetAnimation.framePerSecond,\r\n Animation.ANIMATIONTYPE_FLOAT,\r\n morphTargetAnimation.loopMode,\r\n morphTargetAnimation.enableBlending\r\n );\r\n }\r\n animationKeys.push(morphTargetAnimation.getKeys()[i]);\r\n } else {\r\n animationKeys.push({\r\n frame: animationGroup.from + (animationGroupFrameDiff / numAnimationKeys) * i,\r\n value: morphTarget.influence,\r\n inTangent: sampleAnimationKeys[0].inTangent ? 0 : undefined,\r\n outTangent: sampleAnimationKeys[0].outTangent ? 0 : undefined,\r\n });\r\n }\r\n }\r\n }\r\n }\r\n combinedAnimationGroup!.setKeys(animationKeys);\r\n const animationInfo = _GLTFAnimation._DeduceAnimationInfo(combinedAnimationGroup!);\r\n if (animationInfo) {\r\n _GLTFAnimation._AddAnimation(\r\n `${animationGroup.name}_${mesh.name}_MorphWeightAnimation`,\r\n glTFAnimation,\r\n mesh,\r\n combinedAnimationGroup!,\r\n animationInfo.dataAccessorType,\r\n animationInfo.animationChannelTargetPath,\r\n nodeMap,\r\n binaryWriter,\r\n bufferViews,\r\n accessors,\r\n animationInfo.useQuaternion,\r\n animationSampleRate,\r\n morphTargetManager?.numTargets\r\n );\r\n }\r\n });\r\n if (glTFAnimation.channels.length && glTFAnimation.samplers.length) {\r\n glTFAnimations.push(glTFAnimation);\r\n }\r\n }\r\n }\r\n }\r\n\r\n private static _AddAnimation(\r\n name: string,\r\n glTFAnimation: IAnimation,\r\n babylonTransformNode: Node,\r\n animation: Animation,\r\n dataAccessorType: AccessorType,\r\n animationChannelTargetPath: AnimationChannelTargetPath,\r\n nodeMap: { [key: number]: number },\r\n binaryWriter: _BinaryWriter,\r\n bufferViews: IBufferView[],\r\n accessors: IAccessor[],\r\n useQuaternion: boolean,\r\n animationSampleRate: number,\r\n morphAnimationChannels?: number\r\n ) {\r\n const animationData = _GLTFAnimation._CreateNodeAnimation(babylonTransformNode, animation, animationChannelTargetPath, useQuaternion, animationSampleRate);\r\n let bufferView: IBufferView;\r\n let accessor: IAccessor;\r\n let keyframeAccessorIndex: number;\r\n let dataAccessorIndex: number;\r\n let outputLength: number;\r\n let animationSampler: IAnimationSampler;\r\n let animationChannel: IAnimationChannel;\r\n\r\n if (animationData) {\r\n /*\r\n * Now that we have the glTF converted morph target animation data,\r\n * we can remove redundant input data so that we have n input frames,\r\n * and morphAnimationChannels * n output frames\r\n */\r\n if (morphAnimationChannels) {\r\n let index = 0;\r\n let currentInput: number = 0;\r\n const newInputs: number[] = [];\r\n while (animationData.inputs.length > 0) {\r\n currentInput = animationData.inputs.shift()!;\r\n if (index % morphAnimationChannels == 0) {\r\n newInputs.push(currentInput);\r\n }\r\n index++;\r\n }\r\n animationData.inputs = newInputs;\r\n }\r\n\r\n const nodeIndex = nodeMap[babylonTransformNode.uniqueId];\r\n\r\n // Creates buffer view and accessor for key frames.\r\n let byteLength = animationData.inputs.length * 4;\r\n bufferView = _GLTFUtilities._CreateBufferView(0, binaryWriter.getByteOffset(), byteLength, undefined, `${name} keyframe data view`);\r\n bufferViews.push(bufferView);\r\n animationData.inputs.forEach(function (input) {\r\n binaryWriter.setFloat32(input);\r\n });\r\n\r\n accessor = _GLTFUtilities._CreateAccessor(\r\n bufferViews.length - 1,\r\n `${name} keyframes`,\r\n AccessorType.SCALAR,\r\n AccessorComponentType.FLOAT,\r\n animationData.inputs.length,\r\n null,\r\n [animationData.inputsMin],\r\n [animationData.inputsMax]\r\n );\r\n accessors.push(accessor);\r\n keyframeAccessorIndex = accessors.length - 1;\r\n\r\n // create bufferview and accessor for keyed values.\r\n outputLength = animationData.outputs.length;\r\n byteLength = _GLTFUtilities._GetDataAccessorElementCount(dataAccessorType) * 4 * animationData.outputs.length;\r\n\r\n // check for in and out tangents\r\n bufferView = _GLTFUtilities._CreateBufferView(0, binaryWriter.getByteOffset(), byteLength, undefined, `${name} data view`);\r\n bufferViews.push(bufferView);\r\n\r\n animationData.outputs.forEach(function (output) {\r\n output.forEach(function (entry) {\r\n binaryWriter.setFloat32(entry);\r\n });\r\n });\r\n\r\n accessor = _GLTFUtilities._CreateAccessor(bufferViews.length - 1, `${name} data`, dataAccessorType, AccessorComponentType.FLOAT, outputLength, null, null, null);\r\n accessors.push(accessor);\r\n dataAccessorIndex = accessors.length - 1;\r\n\r\n // create sampler\r\n animationSampler = {\r\n interpolation: animationData.samplerInterpolation,\r\n input: keyframeAccessorIndex,\r\n output: dataAccessorIndex,\r\n };\r\n glTFAnimation.samplers.push(animationSampler);\r\n\r\n // create channel\r\n animationChannel = {\r\n sampler: glTFAnimation.samplers.length - 1,\r\n target: {\r\n node: nodeIndex,\r\n path: animationChannelTargetPath,\r\n },\r\n };\r\n glTFAnimation.channels.push(animationChannel);\r\n }\r\n }\r\n\r\n /**\r\n * Create a baked animation\r\n * @param babylonTransformNode BabylonJS mesh\r\n * @param animation BabylonJS animation corresponding to the BabylonJS mesh\r\n * @param animationChannelTargetPath animation target channel\r\n * @param minFrame minimum animation frame\r\n * @param maxFrame maximum animation frame\r\n * @param fps frames per second of the animation\r\n * @param sampleRate\r\n * @param inputs input key frames of the animation\r\n * @param outputs output key frame data of the animation\r\n * @param minMaxFrames\r\n * @param minMaxFrames.min\r\n * @param minMaxFrames.max\r\n * @param useQuaternion specifies if quaternions should be used\r\n */\r\n private static _CreateBakedAnimation(\r\n babylonTransformNode: Node,\r\n animation: Animation,\r\n animationChannelTargetPath: AnimationChannelTargetPath,\r\n minFrame: number,\r\n maxFrame: number,\r\n fps: number,\r\n sampleRate: number,\r\n inputs: number[],\r\n outputs: number[][],\r\n minMaxFrames: { min: number; max: number },\r\n useQuaternion: boolean\r\n ) {\r\n let value: number | Vector3 | Quaternion;\r\n const quaternionCache: Quaternion = Quaternion.Identity();\r\n let previousTime: Nullable = null;\r\n let time: number;\r\n let maxUsedFrame: Nullable = null;\r\n let currKeyFrame: Nullable = null;\r\n let nextKeyFrame: Nullable = null;\r\n let prevKeyFrame: Nullable = null;\r\n let endFrame: Nullable = null;\r\n minMaxFrames.min = Tools.FloatRound(minFrame / fps);\r\n\r\n const keyFrames = animation.getKeys();\r\n\r\n for (let i = 0, length = keyFrames.length; i < length; ++i) {\r\n endFrame = null;\r\n currKeyFrame = keyFrames[i];\r\n\r\n if (i + 1 < length) {\r\n nextKeyFrame = keyFrames[i + 1];\r\n if ((currKeyFrame.value.equals && currKeyFrame.value.equals(nextKeyFrame.value)) || currKeyFrame.value === nextKeyFrame.value) {\r\n if (i === 0) {\r\n // set the first frame to itself\r\n endFrame = currKeyFrame.frame;\r\n } else {\r\n continue;\r\n }\r\n } else {\r\n endFrame = nextKeyFrame.frame;\r\n }\r\n } else {\r\n // at the last key frame\r\n prevKeyFrame = keyFrames[i - 1];\r\n if ((currKeyFrame.value.equals && currKeyFrame.value.equals(prevKeyFrame.value)) || currKeyFrame.value === prevKeyFrame.value) {\r\n continue;\r\n } else {\r\n endFrame = maxFrame;\r\n }\r\n }\r\n if (endFrame) {\r\n for (let f = currKeyFrame.frame; f <= endFrame; f += sampleRate) {\r\n time = Tools.FloatRound(f / fps);\r\n if (time === previousTime) {\r\n continue;\r\n }\r\n previousTime = time;\r\n maxUsedFrame = time;\r\n const state = {\r\n key: 0,\r\n repeatCount: 0,\r\n loopMode: animation.loopMode,\r\n };\r\n value = animation._interpolate(f, state);\r\n\r\n _GLTFAnimation._SetInterpolatedValue(babylonTransformNode, value, time, animation, animationChannelTargetPath, quaternionCache, inputs, outputs, useQuaternion);\r\n }\r\n }\r\n }\r\n if (maxUsedFrame) {\r\n minMaxFrames.max = maxUsedFrame;\r\n }\r\n }\r\n\r\n private static _ConvertFactorToVector3OrQuaternion(\r\n factor: number,\r\n babylonTransformNode: Node,\r\n animation: Animation,\r\n animationChannelTargetPath: AnimationChannelTargetPath,\r\n useQuaternion: boolean\r\n ): Vector3 | Quaternion {\r\n const basePositionRotationOrScale = _GLTFAnimation._GetBasePositionRotationOrScale(babylonTransformNode, animationChannelTargetPath, useQuaternion);\r\n // handles single component x, y, z or w component animation by using a base property and animating over a component.\r\n const property = animation.targetProperty.split(\".\");\r\n const componentName = property ? property[1] : \"\"; // x, y, z, or w component\r\n const value = useQuaternion ? Quaternion.FromArray(basePositionRotationOrScale).normalize() : Vector3.FromArray(basePositionRotationOrScale);\r\n\r\n switch (componentName) {\r\n case \"x\":\r\n case \"y\":\r\n case \"z\": {\r\n value[componentName] = factor;\r\n break;\r\n }\r\n case \"w\": {\r\n (value as Quaternion).w = factor;\r\n break;\r\n }\r\n default: {\r\n Tools.Error(`glTFAnimation: Unsupported component name \"${componentName}\"!`);\r\n }\r\n }\r\n\r\n return value;\r\n }\r\n\r\n private static _SetInterpolatedValue(\r\n babylonTransformNode: Node,\r\n value: number | Vector3 | Quaternion,\r\n time: number,\r\n animation: Animation,\r\n animationChannelTargetPath: AnimationChannelTargetPath,\r\n quaternionCache: Quaternion,\r\n inputs: number[],\r\n outputs: number[][],\r\n useQuaternion: boolean\r\n ) {\r\n let cacheValue: Vector3 | Quaternion | number;\r\n inputs.push(time);\r\n\r\n if (animationChannelTargetPath === AnimationChannelTargetPath.WEIGHTS) {\r\n outputs.push([value as number]);\r\n return;\r\n }\r\n\r\n if (animation.dataType === Animation.ANIMATIONTYPE_FLOAT) {\r\n value = this._ConvertFactorToVector3OrQuaternion(value as number, babylonTransformNode, animation, animationChannelTargetPath, useQuaternion);\r\n }\r\n\r\n if (animationChannelTargetPath === AnimationChannelTargetPath.ROTATION) {\r\n if (useQuaternion) {\r\n quaternionCache = value as Quaternion;\r\n } else {\r\n cacheValue = value as Vector3;\r\n Quaternion.RotationYawPitchRollToRef(cacheValue.y, cacheValue.x, cacheValue.z, quaternionCache);\r\n }\r\n outputs.push(quaternionCache.asArray());\r\n } else {\r\n // scaling and position animation\r\n cacheValue = value as Vector3;\r\n outputs.push(cacheValue.asArray());\r\n }\r\n }\r\n\r\n /**\r\n * Creates linear animation from the animation key frames\r\n * @param babylonTransformNode BabylonJS mesh\r\n * @param animation BabylonJS animation\r\n * @param animationChannelTargetPath The target animation channel\r\n * @param inputs Array to store the key frame times\r\n * @param outputs Array to store the key frame data\r\n * @param useQuaternion Specifies if quaternions are used in the animation\r\n */\r\n private static _CreateLinearOrStepAnimation(\r\n babylonTransformNode: Node,\r\n animation: Animation,\r\n animationChannelTargetPath: AnimationChannelTargetPath,\r\n inputs: number[],\r\n outputs: number[][],\r\n useQuaternion: boolean\r\n ) {\r\n for (const keyFrame of animation.getKeys()) {\r\n inputs.push(keyFrame.frame / animation.framePerSecond); // keyframes in seconds.\r\n _GLTFAnimation._AddKeyframeValue(keyFrame, animation, outputs, animationChannelTargetPath, babylonTransformNode, useQuaternion);\r\n }\r\n }\r\n\r\n /**\r\n * Creates cubic spline animation from the animation key frames\r\n * @param babylonTransformNode BabylonJS mesh\r\n * @param animation BabylonJS animation\r\n * @param animationChannelTargetPath The target animation channel\r\n * @param frameDelta The difference between the last and first frame of the animation\r\n * @param inputs Array to store the key frame times\r\n * @param outputs Array to store the key frame data\r\n * @param useQuaternion Specifies if quaternions are used in the animation\r\n */\r\n private static _CreateCubicSplineAnimation(\r\n babylonTransformNode: Node,\r\n animation: Animation,\r\n animationChannelTargetPath: AnimationChannelTargetPath,\r\n inputs: number[],\r\n outputs: number[][],\r\n useQuaternion: boolean\r\n ) {\r\n animation.getKeys().forEach(function (keyFrame) {\r\n inputs.push(keyFrame.frame / animation.framePerSecond); // keyframes in seconds.\r\n _GLTFAnimation._AddSplineTangent(_TangentType.INTANGENT, outputs, animationChannelTargetPath, AnimationSamplerInterpolation.CUBICSPLINE, keyFrame, useQuaternion);\r\n _GLTFAnimation._AddKeyframeValue(keyFrame, animation, outputs, animationChannelTargetPath, babylonTransformNode, useQuaternion);\r\n\r\n _GLTFAnimation._AddSplineTangent(_TangentType.OUTTANGENT, outputs, animationChannelTargetPath, AnimationSamplerInterpolation.CUBICSPLINE, keyFrame, useQuaternion);\r\n });\r\n }\r\n\r\n private static _GetBasePositionRotationOrScale(babylonTransformNode: Node, animationChannelTargetPath: AnimationChannelTargetPath, useQuaternion: boolean) {\r\n let basePositionRotationOrScale: number[];\r\n if (animationChannelTargetPath === AnimationChannelTargetPath.ROTATION) {\r\n if (useQuaternion) {\r\n const q = (babylonTransformNode as TransformNode).rotationQuaternion;\r\n basePositionRotationOrScale = (q ?? Quaternion.Identity()).asArray();\r\n } else {\r\n const r: Vector3 = (babylonTransformNode as TransformNode).rotation;\r\n basePositionRotationOrScale = (r ?? Vector3.Zero()).asArray();\r\n }\r\n } else if (animationChannelTargetPath === AnimationChannelTargetPath.TRANSLATION) {\r\n const p: Vector3 = (babylonTransformNode as TransformNode).position;\r\n basePositionRotationOrScale = (p ?? Vector3.Zero()).asArray();\r\n } else {\r\n // scale\r\n const s: Vector3 = (babylonTransformNode as TransformNode).scaling;\r\n basePositionRotationOrScale = (s ?? Vector3.One()).asArray();\r\n }\r\n return basePositionRotationOrScale;\r\n }\r\n\r\n /**\r\n * Adds a key frame value\r\n * @param keyFrame\r\n * @param animation\r\n * @param outputs\r\n * @param animationChannelTargetPath\r\n * @param babylonTransformNode\r\n * @param useQuaternion\r\n */\r\n private static _AddKeyframeValue(\r\n keyFrame: IAnimationKey,\r\n animation: Animation,\r\n outputs: number[][],\r\n animationChannelTargetPath: AnimationChannelTargetPath,\r\n babylonTransformNode: Node,\r\n useQuaternion: boolean\r\n ) {\r\n let newPositionRotationOrScale: Nullable;\r\n const animationType = animation.dataType;\r\n if (animationType === Animation.ANIMATIONTYPE_VECTOR3) {\r\n let value = keyFrame.value.asArray();\r\n if (animationChannelTargetPath === AnimationChannelTargetPath.ROTATION) {\r\n const array = Vector3.FromArray(value);\r\n const rotationQuaternion = Quaternion.RotationYawPitchRoll(array.y, array.x, array.z);\r\n value = rotationQuaternion.asArray();\r\n }\r\n outputs.push(value); // scale vector.\r\n } else if (animationType === Animation.ANIMATIONTYPE_FLOAT) {\r\n if (animationChannelTargetPath === AnimationChannelTargetPath.WEIGHTS) {\r\n outputs.push([keyFrame.value]);\r\n } else {\r\n // handles single component x, y, z or w component animation by using a base property and animating over a component.\r\n newPositionRotationOrScale = this._ConvertFactorToVector3OrQuaternion(\r\n keyFrame.value as number,\r\n babylonTransformNode,\r\n animation,\r\n animationChannelTargetPath,\r\n useQuaternion\r\n );\r\n if (newPositionRotationOrScale) {\r\n if (animationChannelTargetPath === AnimationChannelTargetPath.ROTATION) {\r\n const posRotScale = useQuaternion\r\n ? (newPositionRotationOrScale as Quaternion)\r\n : Quaternion.RotationYawPitchRoll(newPositionRotationOrScale.y, newPositionRotationOrScale.x, newPositionRotationOrScale.z).normalize();\r\n outputs.push(posRotScale.asArray());\r\n }\r\n outputs.push(newPositionRotationOrScale.asArray());\r\n }\r\n }\r\n } else if (animationType === Animation.ANIMATIONTYPE_QUATERNION) {\r\n outputs.push((keyFrame.value as Quaternion).normalize().asArray());\r\n } else {\r\n Tools.Error(\"glTFAnimation: Unsupported key frame values for animation!\");\r\n }\r\n }\r\n\r\n /**\r\n * Determine the interpolation based on the key frames\r\n * @param keyFrames\r\n * @param animationChannelTargetPath\r\n * @param useQuaternion\r\n */\r\n private static _DeduceInterpolation(\r\n keyFrames: IAnimationKey[],\r\n animationChannelTargetPath: AnimationChannelTargetPath,\r\n useQuaternion: boolean\r\n ): { interpolationType: AnimationSamplerInterpolation; shouldBakeAnimation: boolean } {\r\n let interpolationType: AnimationSamplerInterpolation | undefined;\r\n let shouldBakeAnimation = false;\r\n let key: IAnimationKey;\r\n\r\n if (animationChannelTargetPath === AnimationChannelTargetPath.ROTATION && !useQuaternion) {\r\n return { interpolationType: AnimationSamplerInterpolation.LINEAR, shouldBakeAnimation: true };\r\n }\r\n\r\n for (let i = 0, length = keyFrames.length; i < length; ++i) {\r\n key = keyFrames[i];\r\n if (key.inTangent || key.outTangent) {\r\n if (interpolationType) {\r\n if (interpolationType !== AnimationSamplerInterpolation.CUBICSPLINE) {\r\n interpolationType = AnimationSamplerInterpolation.LINEAR;\r\n shouldBakeAnimation = true;\r\n break;\r\n }\r\n } else {\r\n interpolationType = AnimationSamplerInterpolation.CUBICSPLINE;\r\n }\r\n } else {\r\n if (interpolationType) {\r\n if (\r\n interpolationType === AnimationSamplerInterpolation.CUBICSPLINE ||\r\n (key.interpolation && key.interpolation === AnimationKeyInterpolation.STEP && interpolationType !== AnimationSamplerInterpolation.STEP)\r\n ) {\r\n interpolationType = AnimationSamplerInterpolation.LINEAR;\r\n shouldBakeAnimation = true;\r\n break;\r\n }\r\n } else {\r\n if (key.interpolation && key.interpolation === AnimationKeyInterpolation.STEP) {\r\n interpolationType = AnimationSamplerInterpolation.STEP;\r\n } else {\r\n interpolationType = AnimationSamplerInterpolation.LINEAR;\r\n }\r\n }\r\n }\r\n }\r\n if (!interpolationType) {\r\n interpolationType = AnimationSamplerInterpolation.LINEAR;\r\n }\r\n\r\n return { interpolationType: interpolationType, shouldBakeAnimation: shouldBakeAnimation };\r\n }\r\n\r\n /**\r\n * Adds an input tangent or output tangent to the output data\r\n * If an input tangent or output tangent is missing, it uses the zero vector or zero quaternion\r\n * @param babylonTransformNode\r\n * @param tangentType Specifies which type of tangent to handle (inTangent or outTangent)\r\n * @param outputs The animation data by keyframe\r\n * @param animationChannelTargetPath The target animation channel\r\n * @param interpolation The interpolation type\r\n * @param keyFrame The key frame with the animation data\r\n * @param frameDelta Time difference between two frames used to scale the tangent by the frame delta\r\n * @param useQuaternion Specifies if quaternions are used\r\n */\r\n private static _AddSplineTangent(\r\n tangentType: _TangentType,\r\n outputs: number[][],\r\n animationChannelTargetPath: AnimationChannelTargetPath,\r\n interpolation: AnimationSamplerInterpolation,\r\n keyFrame: IAnimationKey,\r\n useQuaternion: boolean\r\n ) {\r\n let tangent: number[];\r\n const tangentValue: Vector3 | Quaternion | number = tangentType === _TangentType.INTANGENT ? keyFrame.inTangent : keyFrame.outTangent;\r\n if (interpolation === AnimationSamplerInterpolation.CUBICSPLINE) {\r\n if (animationChannelTargetPath === AnimationChannelTargetPath.ROTATION) {\r\n if (tangentValue) {\r\n if (useQuaternion) {\r\n tangent = (tangentValue as Quaternion).asArray();\r\n } else {\r\n const array = tangentValue as Vector3;\r\n tangent = Quaternion.RotationYawPitchRoll(array.y, array.x, array.z).asArray();\r\n }\r\n } else {\r\n tangent = [0, 0, 0, 0];\r\n }\r\n } else if (animationChannelTargetPath === AnimationChannelTargetPath.WEIGHTS) {\r\n if (tangentValue) {\r\n tangent = [tangentValue as number];\r\n } else {\r\n tangent = [0];\r\n }\r\n } else {\r\n if (tangentValue) {\r\n tangent = (tangentValue as Vector3).asArray();\r\n } else {\r\n tangent = [0, 0, 0];\r\n }\r\n }\r\n\r\n outputs.push(tangent);\r\n }\r\n }\r\n\r\n /**\r\n * Get the minimum and maximum key frames' frame values\r\n * @param keyFrames animation key frames\r\n * @returns the minimum and maximum key frame value\r\n */\r\n private static _CalculateMinMaxKeyFrames(keyFrames: IAnimationKey[]): { min: number; max: number } {\r\n let min: number = Infinity;\r\n let max: number = -Infinity;\r\n keyFrames.forEach(function (keyFrame) {\r\n min = Math.min(min, keyFrame.frame);\r\n max = Math.max(max, keyFrame.frame);\r\n });\r\n\r\n return { min: min, max: max };\r\n }\r\n}\r\n","import { ImageMimeType } from \"babylonjs-gltf2interface\";\r\n\r\n/**\r\n * Class for holding and downloading glTF file data\r\n */\r\nexport class GLTFData {\r\n /**\r\n * Object which contains the file name as the key and its data as the value\r\n */\r\n glTFFiles: { [fileName: string]: string | Blob };\r\n\r\n /**\r\n * Initializes the glTF file object\r\n */\r\n public constructor() {\r\n this.glTFFiles = {};\r\n }\r\n\r\n /**\r\n * Downloads the glTF data as files based on their names and data\r\n */\r\n public downloadFiles(): void {\r\n /**\r\n * Checks for a matching suffix at the end of a string (for ES5 and lower)\r\n * @param str Source string\r\n * @param suffix Suffix to search for in the source string\r\n * @returns Boolean indicating whether the suffix was found (true) or not (false)\r\n */\r\n function endsWith(str: string, suffix: string): boolean {\r\n return str.indexOf(suffix, str.length - suffix.length) !== -1;\r\n }\r\n\r\n for (const key in this.glTFFiles) {\r\n const link = document.createElement(\"a\");\r\n document.body.appendChild(link);\r\n link.setAttribute(\"type\", \"hidden\");\r\n link.download = key;\r\n const blob = this.glTFFiles[key];\r\n let mimeType;\r\n\r\n if (endsWith(key, \".glb\")) {\r\n mimeType = { type: \"model/gltf-binary\" };\r\n } else if (endsWith(key, \".bin\")) {\r\n mimeType = { type: \"application/octet-stream\" };\r\n } else if (endsWith(key, \".gltf\")) {\r\n mimeType = { type: \"model/gltf+json\" };\r\n } else if (endsWith(key, \".jpeg\") || endsWith(key, \".jpg\")) {\r\n mimeType = { type: ImageMimeType.JPEG };\r\n } else if (endsWith(key, \".png\")) {\r\n mimeType = { type: ImageMimeType.PNG };\r\n }\r\n\r\n link.href = window.URL.createObjectURL(new Blob([blob], mimeType));\r\n link.click();\r\n }\r\n }\r\n}\r\n","import type { Nullable } from \"../../types\";\r\nimport { InternalTexture, InternalTextureSource } from \"../../Materials/Textures/internalTexture\";\r\nimport { Logger } from \"../../Misc/logger\";\r\nimport { Tools } from \"../../Misc/tools\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { Constants } from \"../constants\";\r\nimport { ThinEngine } from \"../thinEngine\";\r\nimport type { IWebRequest } from \"../../Misc/interfaces/iWebRequest\";\r\n\r\ndeclare module \"../../Engines/thinEngine\" {\r\n export interface ThinEngine {\r\n /**\r\n * Creates a raw texture\r\n * @param data defines the data to store in the texture\r\n * @param width defines the width of the texture\r\n * @param height defines the height of the texture\r\n * @param format defines the format of the data\r\n * @param generateMipMaps defines if the engine should generate the mip levels\r\n * @param invertY defines if data must be stored with Y axis inverted\r\n * @param samplingMode defines the required sampling mode (Texture.NEAREST_SAMPLINGMODE by default)\r\n * @param compression defines the compression used (null by default)\r\n * @param type defines the type fo the data (Engine.TEXTURETYPE_UNSIGNED_INT by default)\r\n * @param creationFlags specific flags to use when creating the texture (Constants.TEXTURE_CREATIONFLAG_STORAGE for storage textures, for eg)\r\n * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU).\r\n * @returns the raw texture inside an InternalTexture\r\n */\r\n createRawTexture(\r\n data: Nullable,\r\n width: number,\r\n height: number,\r\n format: number,\r\n generateMipMaps: boolean,\r\n invertY: boolean,\r\n samplingMode: number,\r\n compression: Nullable,\r\n type: number,\r\n creationFlags?: number,\r\n useSRGBBuffer?: boolean\r\n ): InternalTexture;\r\n\r\n /**\r\n * Update a raw texture\r\n * @param texture defines the texture to update\r\n * @param data defines the data to store in the texture\r\n * @param format defines the format of the data\r\n * @param invertY defines if data must be stored with Y axis inverted\r\n */\r\n updateRawTexture(texture: Nullable, data: Nullable, format: number, invertY: boolean): void;\r\n\r\n /**\r\n * Update a raw texture\r\n * @param texture defines the texture to update\r\n * @param data defines the data to store in the texture\r\n * @param format defines the format of the data\r\n * @param invertY defines if data must be stored with Y axis inverted\r\n * @param compression defines the compression used (null by default)\r\n * @param type defines the type fo the data (Engine.TEXTURETYPE_UNSIGNED_INT by default)\r\n * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU).\r\n */\r\n updateRawTexture(\r\n texture: Nullable,\r\n data: Nullable,\r\n format: number,\r\n invertY: boolean,\r\n compression: Nullable,\r\n type: number,\r\n useSRGBBuffer: boolean\r\n ): void;\r\n\r\n /**\r\n * Creates a new raw cube texture\r\n * @param data defines the array of data to use to create each face\r\n * @param size defines the size of the textures\r\n * @param format defines the format of the data\r\n * @param type defines the type of the data (like Engine.TEXTURETYPE_UNSIGNED_INT)\r\n * @param generateMipMaps defines if the engine should generate the mip levels\r\n * @param invertY defines if data must be stored with Y axis inverted\r\n * @param samplingMode defines the required sampling mode (like Texture.NEAREST_SAMPLINGMODE)\r\n * @param compression defines the compression used (null by default)\r\n * @returns the cube texture as an InternalTexture\r\n */\r\n createRawCubeTexture(\r\n data: Nullable,\r\n size: number,\r\n format: number,\r\n type: number,\r\n generateMipMaps: boolean,\r\n invertY: boolean,\r\n samplingMode: number,\r\n compression: Nullable\r\n ): InternalTexture;\r\n\r\n /**\r\n * Update a raw cube texture\r\n * @param texture defines the texture to update\r\n * @param data defines the data to store\r\n * @param format defines the data format\r\n * @param type defines the type fo the data (Engine.TEXTURETYPE_UNSIGNED_INT by default)\r\n * @param invertY defines if data must be stored with Y axis inverted\r\n */\r\n updateRawCubeTexture(texture: InternalTexture, data: ArrayBufferView[], format: number, type: number, invertY: boolean): void;\r\n\r\n /**\r\n * Update a raw cube texture\r\n * @param texture defines the texture to update\r\n * @param data defines the data to store\r\n * @param format defines the data format\r\n * @param type defines the type fo the data (Engine.TEXTURETYPE_UNSIGNED_INT by default)\r\n * @param invertY defines if data must be stored with Y axis inverted\r\n * @param compression defines the compression used (null by default)\r\n */\r\n updateRawCubeTexture(texture: InternalTexture, data: ArrayBufferView[], format: number, type: number, invertY: boolean, compression: Nullable): void;\r\n\r\n /**\r\n * Update a raw cube texture\r\n * @param texture defines the texture to update\r\n * @param data defines the data to store\r\n * @param format defines the data format\r\n * @param type defines the type fo the data (Engine.TEXTURETYPE_UNSIGNED_INT by default)\r\n * @param invertY defines if data must be stored with Y axis inverted\r\n * @param compression defines the compression used (null by default)\r\n * @param level defines which level of the texture to update\r\n */\r\n updateRawCubeTexture(texture: InternalTexture, data: ArrayBufferView[], format: number, type: number, invertY: boolean, compression: Nullable, level: number): void;\r\n\r\n /**\r\n * Creates a new raw cube texture from a specified url\r\n * @param url defines the url where the data is located\r\n * @param scene defines the current scene\r\n * @param size defines the size of the textures\r\n * @param format defines the format of the data\r\n * @param type defines the type fo the data (like Engine.TEXTURETYPE_UNSIGNED_INT)\r\n * @param noMipmap defines if the engine should avoid generating the mip levels\r\n * @param callback defines a callback used to extract texture data from loaded data\r\n * @param mipmapGenerator defines to provide an optional tool to generate mip levels\r\n * @param onLoad defines a callback called when texture is loaded\r\n * @param onError defines a callback called if there is an error\r\n * @returns the cube texture as an InternalTexture\r\n */\r\n createRawCubeTextureFromUrl(\r\n url: string,\r\n scene: Nullable,\r\n size: number,\r\n format: number,\r\n type: number,\r\n noMipmap: boolean,\r\n callback: (ArrayBuffer: ArrayBuffer) => Nullable,\r\n mipmapGenerator: Nullable<(faces: ArrayBufferView[]) => ArrayBufferView[][]>,\r\n onLoad: Nullable<() => void>,\r\n onError: Nullable<(message?: string, exception?: any) => void>\r\n ): InternalTexture;\r\n\r\n /**\r\n * Creates a new raw cube texture from a specified url\r\n * @param url defines the url where the data is located\r\n * @param scene defines the current scene\r\n * @param size defines the size of the textures\r\n * @param format defines the format of the data\r\n * @param type defines the type fo the data (like Engine.TEXTURETYPE_UNSIGNED_INT)\r\n * @param noMipmap defines if the engine should avoid generating the mip levels\r\n * @param callback defines a callback used to extract texture data from loaded data\r\n * @param mipmapGenerator defines to provide an optional tool to generate mip levels\r\n * @param onLoad defines a callback called when texture is loaded\r\n * @param onError defines a callback called if there is an error\r\n * @param samplingMode defines the required sampling mode (like Texture.NEAREST_SAMPLINGMODE)\r\n * @param invertY defines if data must be stored with Y axis inverted\r\n * @returns the cube texture as an InternalTexture\r\n */\r\n createRawCubeTextureFromUrl(\r\n url: string,\r\n scene: Nullable,\r\n size: number,\r\n format: number,\r\n type: number,\r\n noMipmap: boolean,\r\n callback: (ArrayBuffer: ArrayBuffer) => Nullable,\r\n mipmapGenerator: Nullable<(faces: ArrayBufferView[]) => ArrayBufferView[][]>,\r\n onLoad: Nullable<() => void>,\r\n onError: Nullable<(message?: string, exception?: any) => void>,\r\n samplingMode: number,\r\n invertY: boolean\r\n ): InternalTexture;\r\n\r\n /**\r\n * Creates a new raw 3D texture\r\n * @param data defines the data used to create the texture\r\n * @param width defines the width of the texture\r\n * @param height defines the height of the texture\r\n * @param depth defines the depth of the texture\r\n * @param format defines the format of the texture\r\n * @param generateMipMaps defines if the engine must generate mip levels\r\n * @param invertY defines if data must be stored with Y axis inverted\r\n * @param samplingMode defines the required sampling mode (like Texture.NEAREST_SAMPLINGMODE)\r\n * @param compression defines the compressed used (can be null)\r\n * @param textureType defines the compressed used (can be null)\r\n * @param creationFlags specific flags to use when creating the texture (Constants.TEXTURE_CREATIONFLAG_STORAGE for storage textures, for eg)\r\n * @returns a new raw 3D texture (stored in an InternalTexture)\r\n */\r\n createRawTexture3D(\r\n data: Nullable,\r\n width: number,\r\n height: number,\r\n depth: number,\r\n format: number,\r\n generateMipMaps: boolean,\r\n invertY: boolean,\r\n samplingMode: number,\r\n compression: Nullable,\r\n textureType: number,\r\n creationFlags?: number\r\n ): InternalTexture;\r\n\r\n /**\r\n * Update a raw 3D texture\r\n * @param texture defines the texture to update\r\n * @param data defines the data to store\r\n * @param format defines the data format\r\n * @param invertY defines if data must be stored with Y axis inverted\r\n */\r\n updateRawTexture3D(texture: InternalTexture, data: Nullable, format: number, invertY: boolean): void;\r\n\r\n /**\r\n * Update a raw 3D texture\r\n * @param texture defines the texture to update\r\n * @param data defines the data to store\r\n * @param format defines the data format\r\n * @param invertY defines if data must be stored with Y axis inverted\r\n * @param compression defines the used compression (can be null)\r\n * @param textureType defines the texture Type (Engine.TEXTURETYPE_UNSIGNED_INT, Engine.TEXTURETYPE_FLOAT...)\r\n */\r\n updateRawTexture3D(texture: InternalTexture, data: Nullable, format: number, invertY: boolean, compression: Nullable, textureType: number): void;\r\n\r\n /**\r\n * Creates a new raw 2D array texture\r\n * @param data defines the data used to create the texture\r\n * @param width defines the width of the texture\r\n * @param height defines the height of the texture\r\n * @param depth defines the number of layers of the texture\r\n * @param format defines the format of the texture\r\n * @param generateMipMaps defines if the engine must generate mip levels\r\n * @param invertY defines if data must be stored with Y axis inverted\r\n * @param samplingMode defines the required sampling mode (like Texture.NEAREST_SAMPLINGMODE)\r\n * @param compression defines the compressed used (can be null)\r\n * @param textureType defines the compressed used (can be null)\r\n * @param creationFlags specific flags to use when creating the texture (Constants.TEXTURE_CREATIONFLAG_STORAGE for storage textures, for eg)\r\n * @returns a new raw 2D array texture (stored in an InternalTexture)\r\n */\r\n createRawTexture2DArray(\r\n data: Nullable,\r\n width: number,\r\n height: number,\r\n depth: number,\r\n format: number,\r\n generateMipMaps: boolean,\r\n invertY: boolean,\r\n samplingMode: number,\r\n compression: Nullable,\r\n textureType: number,\r\n creationFlags?: number\r\n ): InternalTexture;\r\n\r\n /**\r\n * Update a raw 2D array texture\r\n * @param texture defines the texture to update\r\n * @param data defines the data to store\r\n * @param format defines the data format\r\n * @param invertY defines if data must be stored with Y axis inverted\r\n */\r\n updateRawTexture2DArray(texture: InternalTexture, data: Nullable, format: number, invertY: boolean): void;\r\n\r\n /**\r\n * Update a raw 2D array texture\r\n * @param texture defines the texture to update\r\n * @param data defines the data to store\r\n * @param format defines the data format\r\n * @param invertY defines if data must be stored with Y axis inverted\r\n * @param compression defines the used compression (can be null)\r\n * @param textureType defines the texture Type (Engine.TEXTURETYPE_UNSIGNED_INT, Engine.TEXTURETYPE_FLOAT...)\r\n */\r\n updateRawTexture2DArray(\r\n texture: InternalTexture,\r\n data: Nullable,\r\n format: number,\r\n invertY: boolean,\r\n compression: Nullable,\r\n textureType: number\r\n ): void;\r\n }\r\n}\r\n\r\nThinEngine.prototype.updateRawTexture = function (\r\n texture: Nullable,\r\n data: Nullable,\r\n format: number,\r\n invertY: boolean,\r\n compression: Nullable = null,\r\n type: number = Constants.TEXTURETYPE_UNSIGNED_INT,\r\n useSRGBBuffer: boolean = false\r\n): void {\r\n if (!texture) {\r\n return;\r\n }\r\n // Babylon's internalSizedFomat but gl's texImage2D internalFormat\r\n const internalSizedFomat = this._getRGBABufferInternalSizedFormat(type, format, useSRGBBuffer);\r\n\r\n // Babylon's internalFormat but gl's texImage2D format\r\n const internalFormat = this._getInternalFormat(format);\r\n const textureType = this._getWebGLTextureType(type);\r\n this._bindTextureDirectly(this._gl.TEXTURE_2D, texture, true);\r\n this._unpackFlipY(invertY === undefined ? true : invertY ? true : false);\r\n\r\n if (!this._doNotHandleContextLost) {\r\n texture._bufferView = data;\r\n texture.format = format;\r\n texture.type = type;\r\n texture.invertY = invertY;\r\n texture._compression = compression;\r\n }\r\n\r\n if (texture.width % 4 !== 0) {\r\n this._gl.pixelStorei(this._gl.UNPACK_ALIGNMENT, 1);\r\n }\r\n\r\n if (compression && data) {\r\n this._gl.compressedTexImage2D(this._gl.TEXTURE_2D, 0, (this.getCaps().s3tc)[compression], texture.width, texture.height, 0, data);\r\n } else {\r\n this._gl.texImage2D(this._gl.TEXTURE_2D, 0, internalSizedFomat, texture.width, texture.height, 0, internalFormat, textureType, data);\r\n }\r\n\r\n if (texture.generateMipMaps) {\r\n this._gl.generateMipmap(this._gl.TEXTURE_2D);\r\n }\r\n this._bindTextureDirectly(this._gl.TEXTURE_2D, null);\r\n // this.resetTextureCache();\r\n texture.isReady = true;\r\n};\r\n\r\nThinEngine.prototype.createRawTexture = function (\r\n data: Nullable,\r\n width: number,\r\n height: number,\r\n format: number,\r\n generateMipMaps: boolean,\r\n invertY: boolean,\r\n samplingMode: number,\r\n compression: Nullable = null,\r\n type: number = Constants.TEXTURETYPE_UNSIGNED_INT,\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n creationFlags = 0,\r\n useSRGBBuffer = false\r\n): InternalTexture {\r\n const texture = new InternalTexture(this, InternalTextureSource.Raw);\r\n texture.baseWidth = width;\r\n texture.baseHeight = height;\r\n texture.width = width;\r\n texture.height = height;\r\n texture.format = format;\r\n texture.generateMipMaps = generateMipMaps;\r\n texture.samplingMode = samplingMode;\r\n texture.invertY = invertY;\r\n texture._compression = compression;\r\n texture.type = type;\r\n texture._useSRGBBuffer = this._getUseSRGBBuffer(useSRGBBuffer, !generateMipMaps);\r\n\r\n if (!this._doNotHandleContextLost) {\r\n texture._bufferView = data;\r\n }\r\n\r\n this.updateRawTexture(texture, data, format, invertY, compression, type, texture._useSRGBBuffer);\r\n this._bindTextureDirectly(this._gl.TEXTURE_2D, texture, true);\r\n\r\n // Filters\r\n const filters = this._getSamplingParameters(samplingMode, generateMipMaps);\r\n\r\n this._gl.texParameteri(this._gl.TEXTURE_2D, this._gl.TEXTURE_MAG_FILTER, filters.mag);\r\n this._gl.texParameteri(this._gl.TEXTURE_2D, this._gl.TEXTURE_MIN_FILTER, filters.min);\r\n\r\n if (generateMipMaps) {\r\n this._gl.generateMipmap(this._gl.TEXTURE_2D);\r\n }\r\n\r\n this._bindTextureDirectly(this._gl.TEXTURE_2D, null);\r\n\r\n this._internalTexturesCache.push(texture);\r\n\r\n return texture;\r\n};\r\n\r\nThinEngine.prototype.createRawCubeTexture = function (\r\n data: Nullable,\r\n size: number,\r\n format: number,\r\n type: number,\r\n generateMipMaps: boolean,\r\n invertY: boolean,\r\n samplingMode: number,\r\n compression: Nullable = null\r\n): InternalTexture {\r\n const gl = this._gl;\r\n const texture = new InternalTexture(this, InternalTextureSource.CubeRaw);\r\n texture.isCube = true;\r\n texture.format = format;\r\n texture.type = type;\r\n if (!this._doNotHandleContextLost) {\r\n texture._bufferViewArray = data;\r\n }\r\n\r\n const textureType = this._getWebGLTextureType(type);\r\n let internalFormat = this._getInternalFormat(format);\r\n\r\n if (internalFormat === gl.RGB) {\r\n internalFormat = gl.RGBA;\r\n }\r\n\r\n // Mipmap generation needs a sized internal format that is both color-renderable and texture-filterable\r\n if (textureType === gl.FLOAT && !this._caps.textureFloatLinearFiltering) {\r\n generateMipMaps = false;\r\n samplingMode = Constants.TEXTURE_NEAREST_SAMPLINGMODE;\r\n Logger.Warn(\"Float texture filtering is not supported. Mipmap generation and sampling mode are forced to false and TEXTURE_NEAREST_SAMPLINGMODE, respectively.\");\r\n } else if (textureType === this._gl.HALF_FLOAT_OES && !this._caps.textureHalfFloatLinearFiltering) {\r\n generateMipMaps = false;\r\n samplingMode = Constants.TEXTURE_NEAREST_SAMPLINGMODE;\r\n Logger.Warn(\"Half float texture filtering is not supported. Mipmap generation and sampling mode are forced to false and TEXTURE_NEAREST_SAMPLINGMODE, respectively.\");\r\n } else if (textureType === gl.FLOAT && !this._caps.textureFloatRender) {\r\n generateMipMaps = false;\r\n Logger.Warn(\"Render to float textures is not supported. Mipmap generation forced to false.\");\r\n } else if (textureType === gl.HALF_FLOAT && !this._caps.colorBufferFloat) {\r\n generateMipMaps = false;\r\n Logger.Warn(\"Render to half float textures is not supported. Mipmap generation forced to false.\");\r\n }\r\n\r\n const width = size;\r\n const height = width;\r\n\r\n texture.width = width;\r\n texture.height = height;\r\n texture.invertY = invertY;\r\n texture._compression = compression;\r\n\r\n // Double check on POT to generate Mips.\r\n const isPot = !this.needPOTTextures || (Tools.IsExponentOfTwo(texture.width) && Tools.IsExponentOfTwo(texture.height));\r\n if (!isPot) {\r\n generateMipMaps = false;\r\n }\r\n\r\n // Upload data if needed. The texture won't be ready until then.\r\n if (data) {\r\n this.updateRawCubeTexture(texture, data, format, type, invertY, compression);\r\n } else {\r\n const internalSizedFomat = this._getRGBABufferInternalSizedFormat(type);\r\n const level = 0;\r\n\r\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, texture, true);\r\n\r\n for (let faceIndex = 0; faceIndex < 6; faceIndex++) {\r\n if (compression) {\r\n gl.compressedTexImage2D(\r\n gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex,\r\n level,\r\n (this.getCaps().s3tc)[compression],\r\n texture.width,\r\n texture.height,\r\n 0,\r\n undefined as any\r\n );\r\n } else {\r\n gl.texImage2D(gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex, level, internalSizedFomat, texture.width, texture.height, 0, internalFormat, textureType, null);\r\n }\r\n }\r\n\r\n this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, null);\r\n }\r\n\r\n this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, texture, true);\r\n\r\n // Filters\r\n if (data && generateMipMaps) {\r\n this._gl.generateMipmap(this._gl.TEXTURE_CUBE_MAP);\r\n }\r\n\r\n const filters = this._getSamplingParameters(samplingMode, generateMipMaps);\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_MAG_FILTER, filters.mag);\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_MIN_FILTER, filters.min);\r\n\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\r\n gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\r\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, null);\r\n\r\n texture.generateMipMaps = generateMipMaps;\r\n texture.samplingMode = samplingMode;\r\n texture.isReady = true;\r\n\r\n return texture;\r\n};\r\n\r\nThinEngine.prototype.updateRawCubeTexture = function (\r\n texture: InternalTexture,\r\n data: ArrayBufferView[],\r\n format: number,\r\n type: number,\r\n invertY: boolean,\r\n compression: Nullable = null,\r\n level: number = 0\r\n): void {\r\n texture._bufferViewArray = data;\r\n texture.format = format;\r\n texture.type = type;\r\n texture.invertY = invertY;\r\n texture._compression = compression;\r\n\r\n const gl = this._gl;\r\n const textureType = this._getWebGLTextureType(type);\r\n let internalFormat = this._getInternalFormat(format);\r\n const internalSizedFomat = this._getRGBABufferInternalSizedFormat(type);\r\n\r\n let needConversion = false;\r\n if (internalFormat === gl.RGB) {\r\n internalFormat = gl.RGBA;\r\n needConversion = true;\r\n }\r\n\r\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, texture, true);\r\n this._unpackFlipY(invertY === undefined ? true : invertY ? true : false);\r\n\r\n if (texture.width % 4 !== 0) {\r\n gl.pixelStorei(gl.UNPACK_ALIGNMENT, 1);\r\n }\r\n\r\n // Data are known to be in +X +Y +Z -X -Y -Z\r\n for (let faceIndex = 0; faceIndex < 6; faceIndex++) {\r\n let faceData = data[faceIndex];\r\n\r\n if (compression) {\r\n gl.compressedTexImage2D(\r\n gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex,\r\n level,\r\n (this.getCaps().s3tc)[compression],\r\n texture.width,\r\n texture.height,\r\n 0,\r\n faceData\r\n );\r\n } else {\r\n if (needConversion) {\r\n faceData = _convertRGBtoRGBATextureData(faceData, texture.width, texture.height, type);\r\n }\r\n gl.texImage2D(gl.TEXTURE_CUBE_MAP_POSITIVE_X + faceIndex, level, internalSizedFomat, texture.width, texture.height, 0, internalFormat, textureType, faceData);\r\n }\r\n }\r\n\r\n const isPot = !this.needPOTTextures || (Tools.IsExponentOfTwo(texture.width) && Tools.IsExponentOfTwo(texture.height));\r\n if (isPot && texture.generateMipMaps && level === 0) {\r\n this._gl.generateMipmap(this._gl.TEXTURE_CUBE_MAP);\r\n }\r\n this._bindTextureDirectly(this._gl.TEXTURE_CUBE_MAP, null);\r\n\r\n // this.resetTextureCache();\r\n texture.isReady = true;\r\n};\r\n\r\nThinEngine.prototype.createRawCubeTextureFromUrl = function (\r\n url: string,\r\n scene: Nullable,\r\n size: number,\r\n format: number,\r\n type: number,\r\n noMipmap: boolean,\r\n callback: (ArrayBuffer: ArrayBuffer) => Nullable,\r\n mipmapGenerator: Nullable<(faces: ArrayBufferView[]) => ArrayBufferView[][]>,\r\n onLoad: Nullable<() => void> = null,\r\n onError: Nullable<(message?: string, exception?: any) => void> = null,\r\n samplingMode: number = Constants.TEXTURE_TRILINEAR_SAMPLINGMODE,\r\n invertY: boolean = false\r\n): InternalTexture {\r\n const gl = this._gl;\r\n const texture = this.createRawCubeTexture(null, size, format, type, !noMipmap, invertY, samplingMode, null);\r\n scene?.addPendingData(texture);\r\n texture.url = url;\r\n texture.isReady = false;\r\n this._internalTexturesCache.push(texture);\r\n\r\n const onerror = (request?: IWebRequest, exception?: any) => {\r\n scene?.removePendingData(texture);\r\n if (onError && request) {\r\n onError(request.status + \" \" + request.statusText, exception);\r\n }\r\n };\r\n\r\n const internalCallback = (data: any) => {\r\n const width = texture.width;\r\n const faceDataArrays = callback(data);\r\n\r\n if (!faceDataArrays) {\r\n return;\r\n }\r\n\r\n if (mipmapGenerator) {\r\n const textureType = this._getWebGLTextureType(type);\r\n let internalFormat = this._getInternalFormat(format);\r\n const internalSizedFomat = this._getRGBABufferInternalSizedFormat(type);\r\n\r\n let needConversion = false;\r\n if (internalFormat === gl.RGB) {\r\n internalFormat = gl.RGBA;\r\n needConversion = true;\r\n }\r\n\r\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, texture, true);\r\n this._unpackFlipY(false);\r\n\r\n const mipData = mipmapGenerator(faceDataArrays);\r\n for (let level = 0; level < mipData.length; level++) {\r\n const mipSize = width >> level;\r\n\r\n for (let faceIndex = 0; faceIndex < 6; faceIndex++) {\r\n let mipFaceData = mipData[level][faceIndex];\r\n if (needConversion) {\r\n mipFaceData = _convertRGBtoRGBATextureData(mipFaceData, mipSize, mipSize, type);\r\n }\r\n gl.texImage2D(faceIndex, level, internalSizedFomat, mipSize, mipSize, 0, internalFormat, textureType, mipFaceData);\r\n }\r\n }\r\n\r\n this._bindTextureDirectly(gl.TEXTURE_CUBE_MAP, null);\r\n } else {\r\n this.updateRawCubeTexture(texture, faceDataArrays, format, type, invertY);\r\n }\r\n\r\n texture.isReady = true;\r\n // this.resetTextureCache();\r\n scene?.removePendingData(texture);\r\n\r\n texture.onLoadedObservable.notifyObservers(texture);\r\n texture.onLoadedObservable.clear();\r\n\r\n if (onLoad) {\r\n onLoad();\r\n }\r\n };\r\n\r\n this._loadFile(\r\n url,\r\n (data) => {\r\n internalCallback(data);\r\n },\r\n undefined,\r\n scene?.offlineProvider,\r\n true,\r\n onerror\r\n );\r\n\r\n return texture;\r\n};\r\n\r\n/**\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nfunction _convertRGBtoRGBATextureData(rgbData: any, width: number, height: number, textureType: number): ArrayBufferView {\r\n // Create new RGBA data container.\r\n let rgbaData: any;\r\n let val1 = 1;\r\n if (textureType === Constants.TEXTURETYPE_FLOAT) {\r\n rgbaData = new Float32Array(width * height * 4);\r\n } else if (textureType === Constants.TEXTURETYPE_HALF_FLOAT) {\r\n rgbaData = new Uint16Array(width * height * 4);\r\n val1 = 15360; // 15360 is the encoding of 1 in half float\r\n } else if (textureType === Constants.TEXTURETYPE_UNSIGNED_INTEGER) {\r\n rgbaData = new Uint32Array(width * height * 4);\r\n } else {\r\n rgbaData = new Uint8Array(width * height * 4);\r\n }\r\n\r\n // Convert each pixel.\r\n for (let x = 0; x < width; x++) {\r\n for (let y = 0; y < height; y++) {\r\n const index = (y * width + x) * 3;\r\n const newIndex = (y * width + x) * 4;\r\n\r\n // Map Old Value to new value.\r\n rgbaData[newIndex + 0] = rgbData[index + 0];\r\n rgbaData[newIndex + 1] = rgbData[index + 1];\r\n rgbaData[newIndex + 2] = rgbData[index + 2];\r\n\r\n // Add fully opaque alpha channel.\r\n rgbaData[newIndex + 3] = val1;\r\n }\r\n }\r\n\r\n return rgbaData;\r\n}\r\n\r\n/**\r\n * Create a function for createRawTexture3D/createRawTexture2DArray\r\n * @param is3D true for TEXTURE_3D and false for TEXTURE_2D_ARRAY\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nfunction _makeCreateRawTextureFunction(is3D: boolean) {\r\n return function (\r\n this: ThinEngine,\r\n data: Nullable,\r\n width: number,\r\n height: number,\r\n depth: number,\r\n format: number,\r\n generateMipMaps: boolean,\r\n invertY: boolean,\r\n samplingMode: number,\r\n compression: Nullable = null,\r\n textureType: number = Constants.TEXTURETYPE_UNSIGNED_INT\r\n ): InternalTexture {\r\n const target = is3D ? this._gl.TEXTURE_3D : this._gl.TEXTURE_2D_ARRAY;\r\n const source = is3D ? InternalTextureSource.Raw3D : InternalTextureSource.Raw2DArray;\r\n const texture = new InternalTexture(this, source);\r\n texture.baseWidth = width;\r\n texture.baseHeight = height;\r\n texture.baseDepth = depth;\r\n texture.width = width;\r\n texture.height = height;\r\n texture.depth = depth;\r\n texture.format = format;\r\n texture.type = textureType;\r\n texture.generateMipMaps = generateMipMaps;\r\n texture.samplingMode = samplingMode;\r\n if (is3D) {\r\n texture.is3D = true;\r\n } else {\r\n texture.is2DArray = true;\r\n }\r\n\r\n if (!this._doNotHandleContextLost) {\r\n texture._bufferView = data;\r\n }\r\n\r\n if (is3D) {\r\n this.updateRawTexture3D(texture, data, format, invertY, compression, textureType);\r\n } else {\r\n this.updateRawTexture2DArray(texture, data, format, invertY, compression, textureType);\r\n }\r\n this._bindTextureDirectly(target, texture, true);\r\n\r\n // Filters\r\n const filters = this._getSamplingParameters(samplingMode, generateMipMaps);\r\n\r\n this._gl.texParameteri(target, this._gl.TEXTURE_MAG_FILTER, filters.mag);\r\n this._gl.texParameteri(target, this._gl.TEXTURE_MIN_FILTER, filters.min);\r\n\r\n if (generateMipMaps) {\r\n this._gl.generateMipmap(target);\r\n }\r\n\r\n this._bindTextureDirectly(target, null);\r\n\r\n this._internalTexturesCache.push(texture);\r\n\r\n return texture;\r\n };\r\n}\r\n\r\nThinEngine.prototype.createRawTexture2DArray = _makeCreateRawTextureFunction(false);\r\nThinEngine.prototype.createRawTexture3D = _makeCreateRawTextureFunction(true);\r\n\r\n/**\r\n * Create a function for updateRawTexture3D/updateRawTexture2DArray\r\n * @param is3D true for TEXTURE_3D and false for TEXTURE_2D_ARRAY\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nfunction _makeUpdateRawTextureFunction(is3D: boolean) {\r\n return function (\r\n this: ThinEngine,\r\n texture: InternalTexture,\r\n data: Nullable,\r\n format: number,\r\n invertY: boolean,\r\n compression: Nullable = null,\r\n textureType: number = Constants.TEXTURETYPE_UNSIGNED_INT\r\n ): void {\r\n const target = is3D ? this._gl.TEXTURE_3D : this._gl.TEXTURE_2D_ARRAY;\r\n const internalType = this._getWebGLTextureType(textureType);\r\n const internalFormat = this._getInternalFormat(format);\r\n const internalSizedFomat = this._getRGBABufferInternalSizedFormat(textureType, format);\r\n\r\n this._bindTextureDirectly(target, texture, true);\r\n this._unpackFlipY(invertY === undefined ? true : invertY ? true : false);\r\n\r\n if (!this._doNotHandleContextLost) {\r\n texture._bufferView = data;\r\n texture.format = format;\r\n texture.invertY = invertY;\r\n texture._compression = compression;\r\n }\r\n\r\n if (texture.width % 4 !== 0) {\r\n this._gl.pixelStorei(this._gl.UNPACK_ALIGNMENT, 1);\r\n }\r\n\r\n if (compression && data) {\r\n this._gl.compressedTexImage3D(target, 0, (this.getCaps().s3tc)[compression], texture.width, texture.height, texture.depth, 0, data);\r\n } else {\r\n this._gl.texImage3D(target, 0, internalSizedFomat, texture.width, texture.height, texture.depth, 0, internalFormat, internalType, data);\r\n }\r\n\r\n if (texture.generateMipMaps) {\r\n this._gl.generateMipmap(target);\r\n }\r\n this._bindTextureDirectly(target, null);\r\n // this.resetTextureCache();\r\n texture.isReady = true;\r\n };\r\n}\r\n\r\nThinEngine.prototype.updateRawTexture2DArray = _makeUpdateRawTextureFunction(false);\r\nThinEngine.prototype.updateRawTexture3D = _makeUpdateRawTextureFunction(true);\r\n","import { Texture } from \"./texture\";\r\nimport { Constants } from \"../../Engines/constants\";\r\nimport \"../../Engines/Extensions/engine.rawTexture\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { ThinEngine } from \"../../Engines/thinEngine\";\r\n\r\nimport type { Scene } from \"../../scene\";\r\n\r\n/**\r\n * Raw texture can help creating a texture directly from an array of data.\r\n * This can be super useful if you either get the data from an uncompressed source or\r\n * if you wish to create your texture pixel by pixel.\r\n */\r\nexport class RawTexture extends Texture {\r\n /**\r\n * Instantiates a new RawTexture.\r\n * Raw texture can help creating a texture directly from an array of data.\r\n * This can be super useful if you either get the data from an uncompressed source or\r\n * if you wish to create your texture pixel by pixel.\r\n * @param data define the array of data to use to create the texture (null to create an empty texture)\r\n * @param width define the width of the texture\r\n * @param height define the height of the texture\r\n * @param format define the format of the data (RGB, RGBA... Engine.TEXTUREFORMAT_xxx)\r\n * @param sceneOrEngine defines the scene or engine the texture will belong to\r\n * @param generateMipMaps define whether mip maps should be generated or not\r\n * @param invertY define if the data should be flipped on Y when uploaded to the GPU\r\n * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE)\r\n * @param type define the format of the data (int, float... Engine.TEXTURETYPE_xxx)\r\n * @param creationFlags specific flags to use when creating the texture (Constants.TEXTURE_CREATIONFLAG_STORAGE for storage textures, for eg)\r\n * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU).\r\n */\r\n constructor(\r\n data: Nullable,\r\n width: number,\r\n height: number,\r\n /**\r\n * Define the format of the data (RGB, RGBA... Engine.TEXTUREFORMAT_xxx)\r\n */\r\n public format: number,\r\n sceneOrEngine: Nullable,\r\n generateMipMaps: boolean = true,\r\n invertY: boolean = false,\r\n samplingMode: number = Constants.TEXTURE_TRILINEAR_SAMPLINGMODE,\r\n type: number = Constants.TEXTURETYPE_UNSIGNED_INT,\r\n creationFlags?: number,\r\n useSRGBBuffer?: boolean\r\n ) {\r\n super(null, sceneOrEngine, !generateMipMaps, invertY, undefined, undefined, undefined, undefined, undefined, undefined, undefined, undefined, creationFlags);\r\n\r\n if (!this._engine) {\r\n return;\r\n }\r\n\r\n if (!this._engine._caps.textureFloatLinearFiltering && type === Constants.TEXTURETYPE_FLOAT) {\r\n samplingMode = Constants.TEXTURE_NEAREST_SAMPLINGMODE;\r\n }\r\n if (!this._engine._caps.textureHalfFloatLinearFiltering && type === Constants.TEXTURETYPE_HALF_FLOAT) {\r\n samplingMode = Constants.TEXTURE_NEAREST_SAMPLINGMODE;\r\n }\r\n\r\n this._texture = this._engine.createRawTexture(data, width, height, format, generateMipMaps, invertY, samplingMode, null, type, creationFlags ?? 0, useSRGBBuffer ?? false);\r\n\r\n this.wrapU = Texture.CLAMP_ADDRESSMODE;\r\n this.wrapV = Texture.CLAMP_ADDRESSMODE;\r\n }\r\n\r\n /**\r\n * Updates the texture underlying data.\r\n * @param data Define the new data of the texture\r\n */\r\n public update(data: ArrayBufferView): void {\r\n this._getEngine()!.updateRawTexture(this._texture, data, this._texture!.format, this._texture!.invertY, null, this._texture!.type, this._texture!._useSRGBBuffer);\r\n }\r\n\r\n /**\r\n * Creates a luminance texture from some data.\r\n * @param data Define the texture data\r\n * @param width Define the width of the texture\r\n * @param height Define the height of the texture\r\n * @param sceneOrEngine defines the scene or engine the texture will belong to\r\n * @param generateMipMaps Define whether or not to create mip maps for the texture\r\n * @param invertY define if the data should be flipped on Y when uploaded to the GPU\r\n * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE)\r\n * @returns the luminance texture\r\n */\r\n public static CreateLuminanceTexture(\r\n data: Nullable,\r\n width: number,\r\n height: number,\r\n sceneOrEngine: Nullable,\r\n generateMipMaps: boolean = true,\r\n invertY: boolean = false,\r\n samplingMode: number = Constants.TEXTURE_TRILINEAR_SAMPLINGMODE\r\n ): RawTexture {\r\n return new RawTexture(data, width, height, Constants.TEXTUREFORMAT_LUMINANCE, sceneOrEngine, generateMipMaps, invertY, samplingMode);\r\n }\r\n\r\n /**\r\n * Creates a luminance alpha texture from some data.\r\n * @param data Define the texture data\r\n * @param width Define the width of the texture\r\n * @param height Define the height of the texture\r\n * @param sceneOrEngine defines the scene or engine the texture will belong to\r\n * @param generateMipMaps Define whether or not to create mip maps for the texture\r\n * @param invertY define if the data should be flipped on Y when uploaded to the GPU\r\n * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE)\r\n * @returns the luminance alpha texture\r\n */\r\n public static CreateLuminanceAlphaTexture(\r\n data: Nullable,\r\n width: number,\r\n height: number,\r\n sceneOrEngine: Nullable,\r\n generateMipMaps: boolean = true,\r\n invertY: boolean = false,\r\n samplingMode: number = Constants.TEXTURE_TRILINEAR_SAMPLINGMODE\r\n ): RawTexture {\r\n return new RawTexture(data, width, height, Constants.TEXTUREFORMAT_LUMINANCE_ALPHA, sceneOrEngine, generateMipMaps, invertY, samplingMode);\r\n }\r\n\r\n /**\r\n * Creates an alpha texture from some data.\r\n * @param data Define the texture data\r\n * @param width Define the width of the texture\r\n * @param height Define the height of the texture\r\n * @param sceneOrEngine defines the scene or engine the texture will belong to\r\n * @param generateMipMaps Define whether or not to create mip maps for the texture\r\n * @param invertY define if the data should be flipped on Y when uploaded to the GPU\r\n * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE)\r\n * @returns the alpha texture\r\n */\r\n public static CreateAlphaTexture(\r\n data: Nullable,\r\n width: number,\r\n height: number,\r\n sceneOrEngine: Nullable,\r\n generateMipMaps: boolean = true,\r\n invertY: boolean = false,\r\n samplingMode: number = Constants.TEXTURE_TRILINEAR_SAMPLINGMODE\r\n ): RawTexture {\r\n return new RawTexture(data, width, height, Constants.TEXTUREFORMAT_ALPHA, sceneOrEngine, generateMipMaps, invertY, samplingMode);\r\n }\r\n\r\n /**\r\n * Creates a RGB texture from some data.\r\n * @param data Define the texture data\r\n * @param width Define the width of the texture\r\n * @param height Define the height of the texture\r\n * @param sceneOrEngine defines the scene or engine the texture will belong to\r\n * @param generateMipMaps Define whether or not to create mip maps for the texture\r\n * @param invertY define if the data should be flipped on Y when uploaded to the GPU\r\n * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE)\r\n * @param type define the format of the data (int, float... Engine.TEXTURETYPE_xxx)\r\n * @param creationFlags specific flags to use when creating the texture (Constants.TEXTURE_CREATIONFLAG_STORAGE for storage textures, for eg)\r\n * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU).\r\n * @returns the RGB alpha texture\r\n */\r\n public static CreateRGBTexture(\r\n data: Nullable,\r\n width: number,\r\n height: number,\r\n sceneOrEngine: Nullable,\r\n generateMipMaps: boolean = true,\r\n invertY: boolean = false,\r\n samplingMode: number = Constants.TEXTURE_TRILINEAR_SAMPLINGMODE,\r\n type: number = Constants.TEXTURETYPE_UNSIGNED_INT,\r\n creationFlags: number = 0,\r\n useSRGBBuffer: boolean = false\r\n ): RawTexture {\r\n return new RawTexture(data, width, height, Constants.TEXTUREFORMAT_RGB, sceneOrEngine, generateMipMaps, invertY, samplingMode, type, creationFlags, useSRGBBuffer);\r\n }\r\n\r\n /**\r\n * Creates a RGBA texture from some data.\r\n * @param data Define the texture data\r\n * @param width Define the width of the texture\r\n * @param height Define the height of the texture\r\n * @param sceneOrEngine defines the scene or engine the texture will belong to\r\n * @param generateMipMaps Define whether or not to create mip maps for the texture\r\n * @param invertY define if the data should be flipped on Y when uploaded to the GPU\r\n * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE)\r\n * @param type define the format of the data (int, float... Engine.TEXTURETYPE_xxx)\r\n * @param creationFlags specific flags to use when creating the texture (Constants.TEXTURE_CREATIONFLAG_STORAGE for storage textures, for eg)\r\n * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU).\r\n * @returns the RGBA texture\r\n */\r\n public static CreateRGBATexture(\r\n data: Nullable,\r\n width: number,\r\n height: number,\r\n sceneOrEngine: Nullable,\r\n generateMipMaps: boolean = true,\r\n invertY: boolean = false,\r\n samplingMode: number = Constants.TEXTURE_TRILINEAR_SAMPLINGMODE,\r\n type: number = Constants.TEXTURETYPE_UNSIGNED_INT,\r\n creationFlags: number = 0,\r\n useSRGBBuffer: boolean = false\r\n ): RawTexture {\r\n return new RawTexture(data, width, height, Constants.TEXTUREFORMAT_RGBA, sceneOrEngine, generateMipMaps, invertY, samplingMode, type, creationFlags, useSRGBBuffer);\r\n }\r\n\r\n /**\r\n * Creates a RGBA storage texture from some data.\r\n * @param data Define the texture data\r\n * @param width Define the width of the texture\r\n * @param height Define the height of the texture\r\n * @param sceneOrEngine defines the scene or engine the texture will belong to\r\n * @param generateMipMaps Define whether or not to create mip maps for the texture\r\n * @param invertY define if the data should be flipped on Y when uploaded to the GPU\r\n * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE)\r\n * @param type define the format of the data (int, float... Engine.TEXTURETYPE_xxx)\r\n * @param useSRGBBuffer defines if the texture must be loaded in a sRGB GPU buffer (if supported by the GPU).\r\n * @returns the RGBA texture\r\n */\r\n public static CreateRGBAStorageTexture(\r\n data: Nullable,\r\n width: number,\r\n height: number,\r\n sceneOrEngine: Nullable,\r\n generateMipMaps: boolean = true,\r\n invertY: boolean = false,\r\n samplingMode: number = Constants.TEXTURE_TRILINEAR_SAMPLINGMODE,\r\n type: number = Constants.TEXTURETYPE_UNSIGNED_INT,\r\n useSRGBBuffer: boolean = false\r\n ): RawTexture {\r\n return new RawTexture(\r\n data,\r\n width,\r\n height,\r\n Constants.TEXTUREFORMAT_RGBA,\r\n sceneOrEngine,\r\n generateMipMaps,\r\n invertY,\r\n samplingMode,\r\n type,\r\n Constants.TEXTURE_CREATIONFLAG_STORAGE,\r\n useSRGBBuffer\r\n );\r\n }\r\n\r\n /**\r\n * Creates a R texture from some data.\r\n * @param data Define the texture data\r\n * @param width Define the width of the texture\r\n * @param height Define the height of the texture\r\n * @param sceneOrEngine defines the scene or engine the texture will belong to\r\n * @param generateMipMaps Define whether or not to create mip maps for the texture\r\n * @param invertY define if the data should be flipped on Y when uploaded to the GPU\r\n * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE)\r\n * @param type define the format of the data (int, float... Engine.TEXTURETYPE_xxx)\r\n * @returns the R texture\r\n */\r\n public static CreateRTexture(\r\n data: Nullable,\r\n width: number,\r\n height: number,\r\n sceneOrEngine: Nullable,\r\n generateMipMaps: boolean = true,\r\n invertY: boolean = false,\r\n samplingMode: number = Texture.TRILINEAR_SAMPLINGMODE,\r\n type: number = Constants.TEXTURETYPE_FLOAT\r\n ): RawTexture {\r\n return new RawTexture(data, width, height, Constants.TEXTUREFORMAT_R, sceneOrEngine, generateMipMaps, invertY, samplingMode, type);\r\n }\r\n\r\n /**\r\n * Creates a R storage texture from some data.\r\n * @param data Define the texture data\r\n * @param width Define the width of the texture\r\n * @param height Define the height of the texture\r\n * @param sceneOrEngine defines the scene or engine the texture will belong to\r\n * @param generateMipMaps Define whether or not to create mip maps for the texture\r\n * @param invertY define if the data should be flipped on Y when uploaded to the GPU\r\n * @param samplingMode define the texture sampling mode (Texture.xxx_SAMPLINGMODE)\r\n * @param type define the format of the data (int, float... Engine.TEXTURETYPE_xxx)\r\n * @returns the R texture\r\n */\r\n public static CreateRStorageTexture(\r\n data: Nullable,\r\n width: number,\r\n height: number,\r\n sceneOrEngine: Nullable,\r\n generateMipMaps: boolean = true,\r\n invertY: boolean = false,\r\n samplingMode: number = Texture.TRILINEAR_SAMPLINGMODE,\r\n type: number = Constants.TEXTURETYPE_FLOAT\r\n ): RawTexture {\r\n return new RawTexture(data, width, height, Constants.TEXTUREFORMAT_R, sceneOrEngine, generateMipMaps, invertY, samplingMode, type, Constants.TEXTURE_CREATIONFLAG_STORAGE);\r\n }\r\n}\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\n/** Defines the cross module used constants to avoid circular dependencies */\r\nexport class Constants {\r\n /** Defines that alpha blending is disabled */\r\n public static readonly ALPHA_DISABLE = 0;\r\n /** Defines that alpha blending is SRC ALPHA * SRC + DEST */\r\n public static readonly ALPHA_ADD = 1;\r\n /** Defines that alpha blending is SRC ALPHA * SRC + (1 - SRC ALPHA) * DEST */\r\n public static readonly ALPHA_COMBINE = 2;\r\n /** Defines that alpha blending is DEST - SRC * DEST */\r\n public static readonly ALPHA_SUBTRACT = 3;\r\n /** Defines that alpha blending is SRC * DEST */\r\n public static readonly ALPHA_MULTIPLY = 4;\r\n /** Defines that alpha blending is SRC ALPHA * SRC + (1 - SRC) * DEST */\r\n public static readonly ALPHA_MAXIMIZED = 5;\r\n /** Defines that alpha blending is SRC + DEST */\r\n public static readonly ALPHA_ONEONE = 6;\r\n /** Defines that alpha blending is SRC + (1 - SRC ALPHA) * DEST */\r\n public static readonly ALPHA_PREMULTIPLIED = 7;\r\n /**\r\n * Defines that alpha blending is SRC + (1 - SRC ALPHA) * DEST\r\n * Alpha will be set to (1 - SRC ALPHA) * DEST ALPHA\r\n */\r\n public static readonly ALPHA_PREMULTIPLIED_PORTERDUFF = 8;\r\n /** Defines that alpha blending is CST * SRC + (1 - CST) * DEST */\r\n public static readonly ALPHA_INTERPOLATE = 9;\r\n /**\r\n * Defines that alpha blending is SRC + (1 - SRC) * DEST\r\n * Alpha will be set to SRC ALPHA + (1 - SRC ALPHA) * DEST ALPHA\r\n */\r\n public static readonly ALPHA_SCREENMODE = 10;\r\n /**\r\n * Defines that alpha blending is SRC + DST\r\n * Alpha will be set to SRC ALPHA + DST ALPHA\r\n */\r\n public static readonly ALPHA_ONEONE_ONEONE = 11;\r\n /**\r\n * Defines that alpha blending is SRC * DST ALPHA + DST\r\n * Alpha will be set to 0\r\n */\r\n public static readonly ALPHA_ALPHATOCOLOR = 12;\r\n /**\r\n * Defines that alpha blending is SRC * (1 - DST) + DST * (1 - SRC)\r\n */\r\n public static readonly ALPHA_REVERSEONEMINUS = 13;\r\n /**\r\n * Defines that alpha blending is SRC + DST * (1 - SRC ALPHA)\r\n * Alpha will be set to SRC ALPHA + DST ALPHA * (1 - SRC ALPHA)\r\n */\r\n public static readonly ALPHA_SRC_DSTONEMINUSSRCALPHA = 14;\r\n /**\r\n * Defines that alpha blending is SRC + DST\r\n * Alpha will be set to SRC ALPHA\r\n */\r\n public static readonly ALPHA_ONEONE_ONEZERO = 15;\r\n /**\r\n * Defines that alpha blending is SRC * (1 - DST) + DST * (1 - SRC)\r\n * Alpha will be set to DST ALPHA\r\n */\r\n public static readonly ALPHA_EXCLUSION = 16;\r\n /**\r\n * Defines that alpha blending is SRC * SRC ALPHA + DST * (1 - SRC ALPHA)\r\n * Alpha will be set to SRC ALPHA + (1 - SRC ALPHA) * DST ALPHA\r\n */\r\n public static readonly ALPHA_LAYER_ACCUMULATE = 17;\r\n\r\n /** Defines that alpha blending equation a SUM */\r\n public static readonly ALPHA_EQUATION_ADD = 0;\r\n /** Defines that alpha blending equation a SUBSTRACTION */\r\n public static readonly ALPHA_EQUATION_SUBSTRACT = 1;\r\n /** Defines that alpha blending equation a REVERSE SUBSTRACTION */\r\n public static readonly ALPHA_EQUATION_REVERSE_SUBTRACT = 2;\r\n /** Defines that alpha blending equation a MAX operation */\r\n public static readonly ALPHA_EQUATION_MAX = 3;\r\n /** Defines that alpha blending equation a MIN operation */\r\n public static readonly ALPHA_EQUATION_MIN = 4;\r\n /**\r\n * Defines that alpha blending equation a DARKEN operation:\r\n * It takes the min of the src and sums the alpha channels.\r\n */\r\n public static readonly ALPHA_EQUATION_DARKEN = 5;\r\n\r\n /** Defines that the resource is not delayed*/\r\n public static readonly DELAYLOADSTATE_NONE = 0;\r\n /** Defines that the resource was successfully delay loaded */\r\n public static readonly DELAYLOADSTATE_LOADED = 1;\r\n /** Defines that the resource is currently delay loading */\r\n public static readonly DELAYLOADSTATE_LOADING = 2;\r\n /** Defines that the resource is delayed and has not started loading */\r\n public static readonly DELAYLOADSTATE_NOTLOADED = 4;\r\n\r\n // Depth or Stencil test Constants.\r\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will never pass. i.e. Nothing will be drawn */\r\n public static readonly NEVER = 0x0200;\r\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will always pass. i.e. Pixels will be drawn in the order they are drawn */\r\n public static readonly ALWAYS = 0x0207;\r\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is less than the stored value */\r\n public static readonly LESS = 0x0201;\r\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is equals to the stored value */\r\n public static readonly EQUAL = 0x0202;\r\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is less than or equal to the stored value */\r\n public static readonly LEQUAL = 0x0203;\r\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is greater than the stored value */\r\n public static readonly GREATER = 0x0204;\r\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is greater than or equal to the stored value */\r\n public static readonly GEQUAL = 0x0206;\r\n /** Passed to depthFunction or stencilFunction to specify depth or stencil tests will pass if the new depth value is not equal to the stored value */\r\n public static readonly NOTEQUAL = 0x0205;\r\n\r\n // Stencil Actions Constants.\r\n /** Passed to stencilOperation to specify that stencil value must be kept */\r\n public static readonly KEEP = 0x1e00;\r\n /** Passed to stencilOperation to specify that stencil value must be zero */\r\n public static readonly ZERO = 0x0000;\r\n /** Passed to stencilOperation to specify that stencil value must be replaced */\r\n public static readonly REPLACE = 0x1e01;\r\n /** Passed to stencilOperation to specify that stencil value must be incremented */\r\n public static readonly INCR = 0x1e02;\r\n /** Passed to stencilOperation to specify that stencil value must be decremented */\r\n public static readonly DECR = 0x1e03;\r\n /** Passed to stencilOperation to specify that stencil value must be inverted */\r\n public static readonly INVERT = 0x150a;\r\n /** Passed to stencilOperation to specify that stencil value must be incremented with wrapping */\r\n public static readonly INCR_WRAP = 0x8507;\r\n /** Passed to stencilOperation to specify that stencil value must be decremented with wrapping */\r\n public static readonly DECR_WRAP = 0x8508;\r\n\r\n /** Texture is not repeating outside of 0..1 UVs */\r\n public static readonly TEXTURE_CLAMP_ADDRESSMODE = 0;\r\n /** Texture is repeating outside of 0..1 UVs */\r\n public static readonly TEXTURE_WRAP_ADDRESSMODE = 1;\r\n /** Texture is repeating and mirrored */\r\n public static readonly TEXTURE_MIRROR_ADDRESSMODE = 2;\r\n\r\n /** Flag to create a storage texture */\r\n public static readonly TEXTURE_CREATIONFLAG_STORAGE = 1;\r\n\r\n /** ALPHA */\r\n public static readonly TEXTUREFORMAT_ALPHA = 0;\r\n /** LUMINANCE */\r\n public static readonly TEXTUREFORMAT_LUMINANCE = 1;\r\n /** LUMINANCE_ALPHA */\r\n public static readonly TEXTUREFORMAT_LUMINANCE_ALPHA = 2;\r\n /** RGB */\r\n public static readonly TEXTUREFORMAT_RGB = 4;\r\n /** RGBA */\r\n public static readonly TEXTUREFORMAT_RGBA = 5;\r\n /** RED */\r\n public static readonly TEXTUREFORMAT_RED = 6;\r\n /** RED (2nd reference) */\r\n public static readonly TEXTUREFORMAT_R = 6;\r\n /** RG */\r\n public static readonly TEXTUREFORMAT_RG = 7;\r\n /** RED_INTEGER */\r\n public static readonly TEXTUREFORMAT_RED_INTEGER = 8;\r\n /** RED_INTEGER (2nd reference) */\r\n public static readonly TEXTUREFORMAT_R_INTEGER = 8;\r\n /** RG_INTEGER */\r\n public static readonly TEXTUREFORMAT_RG_INTEGER = 9;\r\n /** RGB_INTEGER */\r\n public static readonly TEXTUREFORMAT_RGB_INTEGER = 10;\r\n /** RGBA_INTEGER */\r\n public static readonly TEXTUREFORMAT_RGBA_INTEGER = 11;\r\n /** BGRA */\r\n public static readonly TEXTUREFORMAT_BGRA = 12;\r\n\r\n /** Depth 24 bits + Stencil 8 bits */\r\n public static readonly TEXTUREFORMAT_DEPTH24_STENCIL8 = 13;\r\n /** Depth 32 bits float */\r\n public static readonly TEXTUREFORMAT_DEPTH32_FLOAT = 14;\r\n /** Depth 16 bits */\r\n public static readonly TEXTUREFORMAT_DEPTH16 = 15;\r\n /** Depth 24 bits */\r\n public static readonly TEXTUREFORMAT_DEPTH24 = 16;\r\n /** Depth 24 bits unorm + Stencil 8 bits */\r\n public static readonly TEXTUREFORMAT_DEPTH24UNORM_STENCIL8 = 17;\r\n /** Depth 32 bits float + Stencil 8 bits */\r\n public static readonly TEXTUREFORMAT_DEPTH32FLOAT_STENCIL8 = 18;\r\n /** Stencil 8 bits */\r\n public static readonly TEXTUREFORMAT_STENCIL8 = 19;\r\n\r\n /** Compressed BC7 */\r\n public static readonly TEXTUREFORMAT_COMPRESSED_RGBA_BPTC_UNORM = 36492;\r\n /** Compressed BC7 (SRGB) */\r\n public static readonly TEXTUREFORMAT_COMPRESSED_SRGB_ALPHA_BPTC_UNORM = 36493;\r\n /** Compressed BC6 unsigned float */\r\n public static readonly TEXTUREFORMAT_COMPRESSED_RGB_BPTC_UNSIGNED_FLOAT = 36495;\r\n /** Compressed BC6 signed float */\r\n public static readonly TEXTUREFORMAT_COMPRESSED_RGB_BPTC_SIGNED_FLOAT = 36494;\r\n /** Compressed BC3 */\r\n public static readonly TEXTUREFORMAT_COMPRESSED_RGBA_S3TC_DXT5 = 33779;\r\n /** Compressed BC3 (SRGB) */\r\n public static readonly TEXTUREFORMAT_COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT = 35919;\r\n /** Compressed BC2 */\r\n public static readonly TEXTUREFORMAT_COMPRESSED_RGBA_S3TC_DXT3 = 33778;\r\n /** Compressed BC2 (SRGB) */\r\n public static readonly TEXTUREFORMAT_COMPRESSED_SRGB_ALPHA_S3TC_DXT3_EXT = 35918;\r\n /** Compressed BC1 (RGBA) */\r\n public static readonly TEXTUREFORMAT_COMPRESSED_RGBA_S3TC_DXT1 = 33777;\r\n /** Compressed BC1 (RGB) */\r\n public static readonly TEXTUREFORMAT_COMPRESSED_RGB_S3TC_DXT1 = 33776;\r\n /** Compressed BC1 (SRGB+A) */\r\n public static readonly TEXTUREFORMAT_COMPRESSED_SRGB_ALPHA_S3TC_DXT1_EXT = 35917;\r\n /** Compressed BC1 (SRGB) */\r\n public static readonly TEXTUREFORMAT_COMPRESSED_SRGB_S3TC_DXT1_EXT = 35916;\r\n /** Compressed ASTC 4x4 */\r\n public static readonly TEXTUREFORMAT_COMPRESSED_RGBA_ASTC_4x4 = 37808;\r\n /** Compressed ASTC 4x4 (SRGB) */\r\n public static readonly TEXTUREFORMAT_COMPRESSED_SRGB8_ALPHA8_ASTC_4x4_KHR = 37840;\r\n /** Compressed ETC1 (RGB) */\r\n public static readonly TEXTUREFORMAT_COMPRESSED_RGB_ETC1_WEBGL = 36196;\r\n /** Compressed ETC2 (RGB) */\r\n public static readonly TEXTUREFORMAT_COMPRESSED_RGB8_ETC2 = 37492;\r\n /** Compressed ETC2 (SRGB) */\r\n public static readonly TEXTUREFORMAT_COMPRESSED_SRGB8_ETC2 = 37493;\r\n /** Compressed ETC2 (RGB+A1) */\r\n public static readonly TEXTUREFORMAT_COMPRESSED_RGB8_PUNCHTHROUGH_ALPHA1_ETC2 = 37494;\r\n /** Compressed ETC2 (SRGB+A1)*/\r\n public static readonly TEXTUREFORMAT_COMPRESSED_SRGB8_PUNCHTHROUGH_ALPHA1_ETC2 = 37495;\r\n /** Compressed ETC2 (RGB+A) */\r\n public static readonly TEXTUREFORMAT_COMPRESSED_RGBA8_ETC2_EAC = 37496;\r\n /** Compressed ETC2 (SRGB+1) */\r\n public static readonly TEXTUREFORMAT_COMPRESSED_SRGB8_ALPHA8_ETC2_EAC = 37497;\r\n\r\n /** UNSIGNED_BYTE */\r\n public static readonly TEXTURETYPE_UNSIGNED_BYTE = 0;\r\n /** UNSIGNED_BYTE (2nd reference) */\r\n public static readonly TEXTURETYPE_UNSIGNED_INT = 0;\r\n /** FLOAT */\r\n public static readonly TEXTURETYPE_FLOAT = 1;\r\n /** HALF_FLOAT */\r\n public static readonly TEXTURETYPE_HALF_FLOAT = 2;\r\n /** BYTE */\r\n public static readonly TEXTURETYPE_BYTE = 3;\r\n /** SHORT */\r\n public static readonly TEXTURETYPE_SHORT = 4;\r\n /** UNSIGNED_SHORT */\r\n public static readonly TEXTURETYPE_UNSIGNED_SHORT = 5;\r\n /** INT */\r\n public static readonly TEXTURETYPE_INT = 6;\r\n /** UNSIGNED_INT */\r\n public static readonly TEXTURETYPE_UNSIGNED_INTEGER = 7;\r\n /** UNSIGNED_SHORT_4_4_4_4 */\r\n public static readonly TEXTURETYPE_UNSIGNED_SHORT_4_4_4_4 = 8;\r\n /** UNSIGNED_SHORT_5_5_5_1 */\r\n public static readonly TEXTURETYPE_UNSIGNED_SHORT_5_5_5_1 = 9;\r\n /** UNSIGNED_SHORT_5_6_5 */\r\n public static readonly TEXTURETYPE_UNSIGNED_SHORT_5_6_5 = 10;\r\n /** UNSIGNED_INT_2_10_10_10_REV */\r\n public static readonly TEXTURETYPE_UNSIGNED_INT_2_10_10_10_REV = 11;\r\n /** UNSIGNED_INT_24_8 */\r\n public static readonly TEXTURETYPE_UNSIGNED_INT_24_8 = 12;\r\n /** UNSIGNED_INT_10F_11F_11F_REV */\r\n public static readonly TEXTURETYPE_UNSIGNED_INT_10F_11F_11F_REV = 13;\r\n /** UNSIGNED_INT_5_9_9_9_REV */\r\n public static readonly TEXTURETYPE_UNSIGNED_INT_5_9_9_9_REV = 14;\r\n /** FLOAT_32_UNSIGNED_INT_24_8_REV */\r\n public static readonly TEXTURETYPE_FLOAT_32_UNSIGNED_INT_24_8_REV = 15;\r\n /** UNDEFINED */\r\n public static readonly TEXTURETYPE_UNDEFINED = 16;\r\n\r\n /** 2D Texture target*/\r\n public static readonly TEXTURE_2D = 3553;\r\n /** 2D Array Texture target */\r\n public static readonly TEXTURE_2D_ARRAY = 35866;\r\n /** Cube Map Texture target */\r\n public static readonly TEXTURE_CUBE_MAP = 34067;\r\n /** Cube Map Array Texture target */\r\n public static readonly TEXTURE_CUBE_MAP_ARRAY = 0xdeadbeef;\r\n /** 3D Texture target */\r\n public static readonly TEXTURE_3D = 32879;\r\n\r\n /** nearest is mag = nearest and min = nearest and no mip */\r\n public static readonly TEXTURE_NEAREST_SAMPLINGMODE = 1;\r\n /** mag = nearest and min = nearest and mip = none */\r\n public static readonly TEXTURE_NEAREST_NEAREST = 1;\r\n\r\n /** Bilinear is mag = linear and min = linear and no mip */\r\n public static readonly TEXTURE_BILINEAR_SAMPLINGMODE = 2;\r\n /** mag = linear and min = linear and mip = none */\r\n public static readonly TEXTURE_LINEAR_LINEAR = 2;\r\n\r\n /** Trilinear is mag = linear and min = linear and mip = linear */\r\n public static readonly TEXTURE_TRILINEAR_SAMPLINGMODE = 3;\r\n /** Trilinear is mag = linear and min = linear and mip = linear */\r\n public static readonly TEXTURE_LINEAR_LINEAR_MIPLINEAR = 3;\r\n\r\n /** mag = nearest and min = nearest and mip = nearest */\r\n public static readonly TEXTURE_NEAREST_NEAREST_MIPNEAREST = 4;\r\n /** mag = nearest and min = linear and mip = nearest */\r\n public static readonly TEXTURE_NEAREST_LINEAR_MIPNEAREST = 5;\r\n /** mag = nearest and min = linear and mip = linear */\r\n public static readonly TEXTURE_NEAREST_LINEAR_MIPLINEAR = 6;\r\n /** mag = nearest and min = linear and mip = none */\r\n public static readonly TEXTURE_NEAREST_LINEAR = 7;\r\n /** nearest is mag = nearest and min = nearest and mip = linear */\r\n public static readonly TEXTURE_NEAREST_NEAREST_MIPLINEAR = 8;\r\n /** mag = linear and min = nearest and mip = nearest */\r\n public static readonly TEXTURE_LINEAR_NEAREST_MIPNEAREST = 9;\r\n /** mag = linear and min = nearest and mip = linear */\r\n public static readonly TEXTURE_LINEAR_NEAREST_MIPLINEAR = 10;\r\n /** Bilinear is mag = linear and min = linear and mip = nearest */\r\n public static readonly TEXTURE_LINEAR_LINEAR_MIPNEAREST = 11;\r\n /** mag = linear and min = nearest and mip = none */\r\n public static readonly TEXTURE_LINEAR_NEAREST = 12;\r\n\r\n /** Explicit coordinates mode */\r\n public static readonly TEXTURE_EXPLICIT_MODE = 0;\r\n /** Spherical coordinates mode */\r\n public static readonly TEXTURE_SPHERICAL_MODE = 1;\r\n /** Planar coordinates mode */\r\n public static readonly TEXTURE_PLANAR_MODE = 2;\r\n /** Cubic coordinates mode */\r\n public static readonly TEXTURE_CUBIC_MODE = 3;\r\n /** Projection coordinates mode */\r\n public static readonly TEXTURE_PROJECTION_MODE = 4;\r\n /** Skybox coordinates mode */\r\n public static readonly TEXTURE_SKYBOX_MODE = 5;\r\n /** Inverse Cubic coordinates mode */\r\n public static readonly TEXTURE_INVCUBIC_MODE = 6;\r\n /** Equirectangular coordinates mode */\r\n public static readonly TEXTURE_EQUIRECTANGULAR_MODE = 7;\r\n /** Equirectangular Fixed coordinates mode */\r\n public static readonly TEXTURE_FIXED_EQUIRECTANGULAR_MODE = 8;\r\n /** Equirectangular Fixed Mirrored coordinates mode */\r\n public static readonly TEXTURE_FIXED_EQUIRECTANGULAR_MIRRORED_MODE = 9;\r\n\r\n /** Offline (baking) quality for texture filtering */\r\n public static readonly TEXTURE_FILTERING_QUALITY_OFFLINE = 4096;\r\n\r\n /** High quality for texture filtering */\r\n public static readonly TEXTURE_FILTERING_QUALITY_HIGH = 64;\r\n\r\n /** Medium quality for texture filtering */\r\n public static readonly TEXTURE_FILTERING_QUALITY_MEDIUM = 16;\r\n\r\n /** Low quality for texture filtering */\r\n public static readonly TEXTURE_FILTERING_QUALITY_LOW = 8;\r\n\r\n // Texture rescaling mode\r\n /** Defines that texture rescaling will use a floor to find the closer power of 2 size */\r\n public static readonly SCALEMODE_FLOOR = 1;\r\n /** Defines that texture rescaling will look for the nearest power of 2 size */\r\n public static readonly SCALEMODE_NEAREST = 2;\r\n /** Defines that texture rescaling will use a ceil to find the closer power of 2 size */\r\n public static readonly SCALEMODE_CEILING = 3;\r\n\r\n /**\r\n * The dirty texture flag value\r\n */\r\n public static readonly MATERIAL_TextureDirtyFlag = 1;\r\n /**\r\n * The dirty light flag value\r\n */\r\n public static readonly MATERIAL_LightDirtyFlag = 2;\r\n /**\r\n * The dirty fresnel flag value\r\n */\r\n public static readonly MATERIAL_FresnelDirtyFlag = 4;\r\n /**\r\n * The dirty attribute flag value\r\n */\r\n public static readonly MATERIAL_AttributesDirtyFlag = 8;\r\n /**\r\n * The dirty misc flag value\r\n */\r\n public static readonly MATERIAL_MiscDirtyFlag = 16;\r\n /**\r\n * The dirty prepass flag value\r\n */\r\n public static readonly MATERIAL_PrePassDirtyFlag = 32;\r\n /**\r\n * The all dirty flag value\r\n */\r\n public static readonly MATERIAL_AllDirtyFlag = 63;\r\n\r\n /**\r\n * Returns the triangle fill mode\r\n */\r\n public static readonly MATERIAL_TriangleFillMode = 0;\r\n /**\r\n * Returns the wireframe mode\r\n */\r\n public static readonly MATERIAL_WireFrameFillMode = 1;\r\n /**\r\n * Returns the point fill mode\r\n */\r\n public static readonly MATERIAL_PointFillMode = 2;\r\n /**\r\n * Returns the point list draw mode\r\n */\r\n public static readonly MATERIAL_PointListDrawMode = 3;\r\n /**\r\n * Returns the line list draw mode\r\n */\r\n public static readonly MATERIAL_LineListDrawMode = 4;\r\n /**\r\n * Returns the line loop draw mode\r\n */\r\n public static readonly MATERIAL_LineLoopDrawMode = 5;\r\n /**\r\n * Returns the line strip draw mode\r\n */\r\n public static readonly MATERIAL_LineStripDrawMode = 6;\r\n\r\n /**\r\n * Returns the triangle strip draw mode\r\n */\r\n public static readonly MATERIAL_TriangleStripDrawMode = 7;\r\n /**\r\n * Returns the triangle fan draw mode\r\n */\r\n public static readonly MATERIAL_TriangleFanDrawMode = 8;\r\n\r\n /**\r\n * Stores the clock-wise side orientation\r\n */\r\n public static readonly MATERIAL_ClockWiseSideOrientation = 0;\r\n /**\r\n * Stores the counter clock-wise side orientation\r\n */\r\n public static readonly MATERIAL_CounterClockWiseSideOrientation = 1;\r\n\r\n /**\r\n * Nothing\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\r\n */\r\n public static readonly ACTION_NothingTrigger = 0;\r\n /**\r\n * On pick\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\r\n */\r\n public static readonly ACTION_OnPickTrigger = 1;\r\n /**\r\n * On left pick\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\r\n */\r\n public static readonly ACTION_OnLeftPickTrigger = 2;\r\n /**\r\n * On right pick\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\r\n */\r\n public static readonly ACTION_OnRightPickTrigger = 3;\r\n /**\r\n * On center pick\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\r\n */\r\n public static readonly ACTION_OnCenterPickTrigger = 4;\r\n /**\r\n * On pick down\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\r\n */\r\n public static readonly ACTION_OnPickDownTrigger = 5;\r\n /**\r\n * On double pick\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\r\n */\r\n public static readonly ACTION_OnDoublePickTrigger = 6;\r\n /**\r\n * On pick up\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\r\n */\r\n public static readonly ACTION_OnPickUpTrigger = 7;\r\n /**\r\n * On pick out.\r\n * This trigger will only be raised if you also declared a OnPickDown\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\r\n */\r\n public static readonly ACTION_OnPickOutTrigger = 16;\r\n /**\r\n * On long press\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\r\n */\r\n public static readonly ACTION_OnLongPressTrigger = 8;\r\n /**\r\n * On pointer over\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\r\n */\r\n public static readonly ACTION_OnPointerOverTrigger = 9;\r\n /**\r\n * On pointer out\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\r\n */\r\n public static readonly ACTION_OnPointerOutTrigger = 10;\r\n /**\r\n * On every frame\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\r\n */\r\n public static readonly ACTION_OnEveryFrameTrigger = 11;\r\n /**\r\n * On intersection enter\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\r\n */\r\n public static readonly ACTION_OnIntersectionEnterTrigger = 12;\r\n /**\r\n * On intersection exit\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\r\n */\r\n public static readonly ACTION_OnIntersectionExitTrigger = 13;\r\n /**\r\n * On key down\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\r\n */\r\n public static readonly ACTION_OnKeyDownTrigger = 14;\r\n /**\r\n * On key up\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/events/actions#triggers\r\n */\r\n public static readonly ACTION_OnKeyUpTrigger = 15;\r\n\r\n /**\r\n * Billboard mode will only apply to Y axis\r\n */\r\n public static readonly PARTICLES_BILLBOARDMODE_Y = 2;\r\n /**\r\n * Billboard mode will apply to all axes\r\n */\r\n public static readonly PARTICLES_BILLBOARDMODE_ALL = 7;\r\n /**\r\n * Special billboard mode where the particle will be biilboard to the camera but rotated to align with direction\r\n */\r\n public static readonly PARTICLES_BILLBOARDMODE_STRETCHED = 8;\r\n /**\r\n * Special billboard mode where the particle will be billboard to the camera but only around the axis of the direction of particle emission\r\n */\r\n public static readonly PARTICLES_BILLBOARDMODE_STRETCHED_LOCAL = 9;\r\n\r\n /** Default culling strategy : this is an exclusion test and it's the more accurate.\r\n * Test order :\r\n * Is the bounding sphere outside the frustum ?\r\n * If not, are the bounding box vertices outside the frustum ?\r\n * It not, then the cullable object is in the frustum.\r\n */\r\n public static readonly MESHES_CULLINGSTRATEGY_STANDARD = 0;\r\n /** Culling strategy : Bounding Sphere Only.\r\n * This is an exclusion test. It's faster than the standard strategy because the bounding box is not tested.\r\n * It's also less accurate than the standard because some not visible objects can still be selected.\r\n * Test : is the bounding sphere outside the frustum ?\r\n * If not, then the cullable object is in the frustum.\r\n */\r\n public static readonly MESHES_CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY = 1;\r\n /** Culling strategy : Optimistic Inclusion.\r\n * This in an inclusion test first, then the standard exclusion test.\r\n * This can be faster when a cullable object is expected to be almost always in the camera frustum.\r\n * This could also be a little slower than the standard test when the tested object center is not the frustum but one of its bounding box vertex is still inside.\r\n * Anyway, it's as accurate as the standard strategy.\r\n * Test :\r\n * Is the cullable object bounding sphere center in the frustum ?\r\n * If not, apply the default culling strategy.\r\n */\r\n public static readonly MESHES_CULLINGSTRATEGY_OPTIMISTIC_INCLUSION = 2;\r\n /** Culling strategy : Optimistic Inclusion then Bounding Sphere Only.\r\n * This in an inclusion test first, then the bounding sphere only exclusion test.\r\n * This can be the fastest test when a cullable object is expected to be almost always in the camera frustum.\r\n * This could also be a little slower than the BoundingSphereOnly strategy when the tested object center is not in the frustum but its bounding sphere still intersects it.\r\n * It's less accurate than the standard strategy and as accurate as the BoundingSphereOnly strategy.\r\n * Test :\r\n * Is the cullable object bounding sphere center in the frustum ?\r\n * If not, apply the Bounding Sphere Only strategy. No Bounding Box is tested here.\r\n */\r\n public static readonly MESHES_CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY = 3;\r\n\r\n /**\r\n * No logging while loading\r\n */\r\n public static readonly SCENELOADER_NO_LOGGING = 0;\r\n /**\r\n * Minimal logging while loading\r\n */\r\n public static readonly SCENELOADER_MINIMAL_LOGGING = 1;\r\n /**\r\n * Summary logging while loading\r\n */\r\n public static readonly SCENELOADER_SUMMARY_LOGGING = 2;\r\n /**\r\n * Detailed logging while loading\r\n */\r\n public static readonly SCENELOADER_DETAILED_LOGGING = 3;\r\n\r\n /**\r\n * Constant used to retrieve the irradiance texture index in the textures array in the prepass\r\n * using getIndex(Constants.PREPASS_IRRADIANCE_TEXTURE_TYPE)\r\n */\r\n public static readonly PREPASS_IRRADIANCE_TEXTURE_TYPE = 0;\r\n /**\r\n * Constant used to retrieve the position texture index in the textures array in the prepass\r\n * using getIndex(Constants.PREPASS_POSITION_TEXTURE_INDEX)\r\n */\r\n public static readonly PREPASS_POSITION_TEXTURE_TYPE = 1;\r\n /**\r\n * Constant used to retrieve the velocity texture index in the textures array in the prepass\r\n * using getIndex(Constants.PREPASS_VELOCITY_TEXTURE_INDEX)\r\n */\r\n public static readonly PREPASS_VELOCITY_TEXTURE_TYPE = 2;\r\n /**\r\n * Constant used to retrieve the reflectivity texture index in the textures array in the prepass\r\n * using the getIndex(Constants.PREPASS_REFLECTIVITY_TEXTURE_TYPE)\r\n */\r\n public static readonly PREPASS_REFLECTIVITY_TEXTURE_TYPE = 3;\r\n /**\r\n * Constant used to retrieve the lit color texture index in the textures array in the prepass\r\n * using the getIndex(Constants.PREPASS_COLOR_TEXTURE_TYPE)\r\n */\r\n public static readonly PREPASS_COLOR_TEXTURE_TYPE = 4;\r\n /**\r\n * Constant used to retrieve depth index in the textures array in the prepass\r\n * using the getIndex(Constants.PREPASS_DEPTH_TEXTURE_TYPE)\r\n */\r\n public static readonly PREPASS_DEPTH_TEXTURE_TYPE = 5;\r\n /**\r\n * Constant used to retrieve normal index in the textures array in the prepass\r\n * using the getIndex(Constants.PREPASS_NORMAL_TEXTURE_TYPE)\r\n */\r\n public static readonly PREPASS_NORMAL_TEXTURE_TYPE = 6;\r\n /**\r\n * Constant used to retrieve albedo index in the textures array in the prepass\r\n * using the getIndex(Constants.PREPASS_ALBEDO_SQRT_TEXTURE_TYPE)\r\n */\r\n public static readonly PREPASS_ALBEDO_SQRT_TEXTURE_TYPE = 7;\r\n\r\n /** Flag to create a readable buffer (the buffer can be the source of a copy) */\r\n public static readonly BUFFER_CREATIONFLAG_READ = 1;\r\n /** Flag to create a writable buffer (the buffer can be the destination of a copy) */\r\n public static readonly BUFFER_CREATIONFLAG_WRITE = 2;\r\n /** Flag to create a readable and writable buffer */\r\n public static readonly BUFFER_CREATIONFLAG_READWRITE = 3;\r\n /** Flag to create a buffer suitable to be used as a uniform buffer */\r\n public static readonly BUFFER_CREATIONFLAG_UNIFORM = 4;\r\n /** Flag to create a buffer suitable to be used as a vertex buffer */\r\n public static readonly BUFFER_CREATIONFLAG_VERTEX = 8;\r\n /** Flag to create a buffer suitable to be used as an index buffer */\r\n public static readonly BUFFER_CREATIONFLAG_INDEX = 16;\r\n /** Flag to create a buffer suitable to be used as a storage buffer */\r\n public static readonly BUFFER_CREATIONFLAG_STORAGE = 32;\r\n\r\n /**\r\n * Prefixes used by the engine for sub mesh draw wrappers\r\n */\r\n\r\n /** @internal */\r\n public static readonly RENDERPASS_MAIN = 0;\r\n\r\n /**\r\n * Constant used as key code for Alt key\r\n */\r\n public static readonly INPUT_ALT_KEY = 18;\r\n\r\n /**\r\n * Constant used as key code for Ctrl key\r\n */\r\n public static readonly INPUT_CTRL_KEY = 17;\r\n\r\n /**\r\n * Constant used as key code for Meta key (Left Win, Left Cmd)\r\n */\r\n public static readonly INPUT_META_KEY1 = 91;\r\n\r\n /**\r\n * Constant used as key code for Meta key (Right Win)\r\n */\r\n public static readonly INPUT_META_KEY2 = 92;\r\n\r\n /**\r\n * Constant used as key code for Meta key (Right Win, Right Cmd)\r\n */\r\n public static readonly INPUT_META_KEY3 = 93;\r\n\r\n /**\r\n * Constant used as key code for Shift key\r\n */\r\n public static readonly INPUT_SHIFT_KEY = 16;\r\n\r\n /** Standard snapshot rendering. In this mode, some form of dynamic behavior is possible (for eg, uniform buffers are still updated) */\r\n public static readonly SNAPSHOTRENDERING_STANDARD = 0;\r\n\r\n /** Fast snapshot rendering. In this mode, everything is static and only some limited form of dynamic behaviour is possible */\r\n public static readonly SNAPSHOTRENDERING_FAST = 1;\r\n\r\n /**\r\n * This is the default projection mode used by the cameras.\r\n * It helps recreating a feeling of perspective and better appreciate depth.\r\n * This is the best way to simulate real life cameras.\r\n */\r\n public static readonly PERSPECTIVE_CAMERA = 0;\r\n /**\r\n * This helps creating camera with an orthographic mode.\r\n * Orthographic is commonly used in engineering as a means to produce object specifications that communicate dimensions unambiguously, each line of 1 unit length (cm, meter..whatever) will appear to have the same length everywhere on the drawing. This allows the drafter to dimension only a subset of lines and let the reader know that other lines of that length on the drawing are also that length in reality. Every parallel line in the drawing is also parallel in the object.\r\n */\r\n public static readonly ORTHOGRAPHIC_CAMERA = 1;\r\n\r\n /**\r\n * This is the default FOV mode for perspective cameras.\r\n * This setting aligns the upper and lower bounds of the viewport to the upper and lower bounds of the camera frustum.\r\n */\r\n public static readonly FOVMODE_VERTICAL_FIXED = 0;\r\n /**\r\n * This setting aligns the left and right bounds of the viewport to the left and right bounds of the camera frustum.\r\n */\r\n public static readonly FOVMODE_HORIZONTAL_FIXED = 1;\r\n\r\n /**\r\n * This specifies there is no need for a camera rig.\r\n * Basically only one eye is rendered corresponding to the camera.\r\n */\r\n public static readonly RIG_MODE_NONE = 0;\r\n /**\r\n * Simulates a camera Rig with one blue eye and one red eye.\r\n * This can be use with 3d blue and red glasses.\r\n */\r\n public static readonly RIG_MODE_STEREOSCOPIC_ANAGLYPH = 10;\r\n /**\r\n * Defines that both eyes of the camera will be rendered side by side with a parallel target.\r\n */\r\n public static readonly RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_PARALLEL = 11;\r\n /**\r\n * Defines that both eyes of the camera will be rendered side by side with a none parallel target.\r\n */\r\n public static readonly RIG_MODE_STEREOSCOPIC_SIDEBYSIDE_CROSSEYED = 12;\r\n /**\r\n * Defines that both eyes of the camera will be rendered over under each other.\r\n */\r\n public static readonly RIG_MODE_STEREOSCOPIC_OVERUNDER = 13;\r\n /**\r\n * Defines that both eyes of the camera will be rendered on successive lines interlaced for passive 3d monitors.\r\n */\r\n public static readonly RIG_MODE_STEREOSCOPIC_INTERLACED = 14;\r\n /**\r\n * Defines that both eyes of the camera should be renderered in a VR mode (carbox).\r\n */\r\n public static readonly RIG_MODE_VR = 20;\r\n /**\r\n * Defines that both eyes of the camera should be renderered in a VR mode (webVR).\r\n */\r\n public static readonly RIG_MODE_WEBVR = 21;\r\n /**\r\n * Custom rig mode allowing rig cameras to be populated manually with any number of cameras\r\n */\r\n public static readonly RIG_MODE_CUSTOM = 22;\r\n\r\n /**\r\n * Maximum number of uv sets supported\r\n */\r\n public static readonly MAX_SUPPORTED_UV_SETS = 6;\r\n\r\n /**\r\n * GL constants\r\n */\r\n /** Alpha blend equation: ADD */\r\n public static readonly GL_ALPHA_EQUATION_ADD = 0x8006;\r\n /** Alpha equation: MIN */\r\n public static readonly GL_ALPHA_EQUATION_MIN = 0x8007;\r\n /** Alpha equation: MAX */\r\n public static readonly GL_ALPHA_EQUATION_MAX = 0x8008;\r\n /** Alpha equation: SUBTRACT */\r\n public static readonly GL_ALPHA_EQUATION_SUBTRACT = 0x800a;\r\n /** Alpha equation: REVERSE_SUBTRACT */\r\n public static readonly GL_ALPHA_EQUATION_REVERSE_SUBTRACT = 0x800b;\r\n\r\n /** Alpha blend function: SRC */\r\n public static readonly GL_ALPHA_FUNCTION_SRC = 0x0300;\r\n /** Alpha blend function: ONE_MINUS_SRC */\r\n public static readonly GL_ALPHA_FUNCTION_ONE_MINUS_SRC_COLOR = 0x0301;\r\n /** Alpha blend function: SRC_ALPHA */\r\n public static readonly GL_ALPHA_FUNCTION_SRC_ALPHA = 0x0302;\r\n /** Alpha blend function: ONE_MINUS_SRC_ALPHA */\r\n public static readonly GL_ALPHA_FUNCTION_ONE_MINUS_SRC_ALPHA = 0x0303;\r\n /** Alpha blend function: DST_ALPHA */\r\n public static readonly GL_ALPHA_FUNCTION_DST_ALPHA = 0x0304;\r\n /** Alpha blend function: ONE_MINUS_DST_ALPHA */\r\n public static readonly GL_ALPHA_FUNCTION_ONE_MINUS_DST_ALPHA = 0x0305;\r\n /** Alpha blend function: ONE_MINUS_DST */\r\n public static readonly GL_ALPHA_FUNCTION_DST_COLOR = 0x0306;\r\n /** Alpha blend function: ONE_MINUS_DST */\r\n public static readonly GL_ALPHA_FUNCTION_ONE_MINUS_DST_COLOR = 0x0307;\r\n /** Alpha blend function: SRC_ALPHA_SATURATED */\r\n public static readonly GL_ALPHA_FUNCTION_SRC_ALPHA_SATURATED = 0x0308;\r\n /** Alpha blend function: CONSTANT */\r\n public static readonly GL_ALPHA_FUNCTION_CONSTANT_COLOR = 0x8001;\r\n /** Alpha blend function: ONE_MINUS_CONSTANT */\r\n public static readonly GL_ALPHA_FUNCTION_ONE_MINUS_CONSTANT_COLOR = 0x8002;\r\n /** Alpha blend function: CONSTANT_ALPHA */\r\n public static readonly GL_ALPHA_FUNCTION_CONSTANT_ALPHA = 0x8003;\r\n /** Alpha blend function: ONE_MINUS_CONSTANT_ALPHA */\r\n public static readonly GL_ALPHA_FUNCTION_ONE_MINUS_CONSTANT_ALPHA = 0x8004;\r\n\r\n /** URL to the snippet server. Points to the public snippet server by default */\r\n public static SnippetUrl = \"https://snippet.babylonjs.com\";\r\n}\r\n","import type { ITextureInfo, IMaterial, IMaterialPbrMetallicRoughness, IMaterialOcclusionTextureInfo, ISampler, IMaterialExtension } from \"babylonjs-gltf2interface\";\r\nimport { ImageMimeType, MaterialAlphaMode, TextureMagFilter, TextureMinFilter, TextureWrapMode } from \"babylonjs-gltf2interface\";\r\n\r\nimport type { Nullable } from \"core/types\";\r\nimport { Vector2 } from \"core/Maths/math.vector\";\r\nimport { Color3 } from \"core/Maths/math.color\";\r\nimport { Scalar } from \"core/Maths/math.scalar\";\r\nimport { Tools } from \"core/Misc/tools\";\r\nimport { TextureTools } from \"core/Misc/textureTools\";\r\nimport type { BaseTexture } from \"core/Materials/Textures/baseTexture\";\r\nimport { Texture } from \"core/Materials/Textures/texture\";\r\nimport { RawTexture } from \"core/Materials/Textures/rawTexture\";\r\n\r\nimport type { Scene } from \"core/scene\";\r\n\r\nimport type { _Exporter } from \"./glTFExporter\";\r\nimport { Constants } from \"core/Engines/constants\";\r\nimport { DumpTools } from \"core/Misc/dumpTools\";\r\n\r\nimport type { Material } from \"core/Materials/material\";\r\nimport type { StandardMaterial } from \"core/Materials/standardMaterial\";\r\nimport type { PBRBaseMaterial } from \"core/Materials/PBR/pbrBaseMaterial\";\r\nimport type { PBRMaterial } from \"core/Materials/PBR/pbrMaterial\";\r\n\r\n/**\r\n * Interface for storing specular glossiness factors\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\ninterface _IPBRSpecularGlossiness {\r\n /**\r\n * Represents the linear diffuse factors of the material\r\n */\r\n diffuseColor: Color3;\r\n /**\r\n * Represents the linear specular factors of the material\r\n */\r\n specularColor: Color3;\r\n /**\r\n * Represents the smoothness of the material\r\n */\r\n glossiness: number;\r\n}\r\n\r\n/**\r\n * Interface for storing metallic roughness factors\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\ninterface _IPBRMetallicRoughness {\r\n /**\r\n * Represents the albedo color of the material\r\n */\r\n baseColor: Color3;\r\n /**\r\n * Represents the metalness of the material\r\n */\r\n metallic: Nullable;\r\n /**\r\n * Represents the roughness of the material\r\n */\r\n roughness: Nullable;\r\n /**\r\n * The metallic roughness texture data\r\n */\r\n metallicRoughnessTextureData?: Nullable;\r\n /**\r\n * The base color texture data\r\n */\r\n baseColorTextureData?: Nullable;\r\n}\r\n\r\nfunction getFileExtensionFromMimeType(mimeType: ImageMimeType): string {\r\n switch (mimeType) {\r\n case ImageMimeType.JPEG:\r\n return \".jpg\";\r\n case ImageMimeType.PNG:\r\n return \".png\";\r\n case ImageMimeType.WEBP:\r\n return \".webp\";\r\n }\r\n}\r\n\r\n/**\r\n * Utility methods for working with glTF material conversion properties. This class should only be used internally\r\n * @internal\r\n */\r\nexport class _GLTFMaterialExporter {\r\n /**\r\n * Represents the dielectric specular values for R, G and B\r\n */\r\n private static readonly _DielectricSpecular: Color3 = new Color3(0.04, 0.04, 0.04);\r\n\r\n /**\r\n * Allows the maximum specular power to be defined for material calculations\r\n */\r\n private static readonly _MaxSpecularPower = 1024;\r\n\r\n /**\r\n * Mapping to store textures\r\n */\r\n private _textureMap: { [textureId: string]: ITextureInfo } = {};\r\n\r\n // Mapping of internal textures to images to avoid exporting duplicate images.\r\n private _internalTextureToImage: { [uniqueId: number]: { [mimeType: string]: Promise } } = {};\r\n\r\n /**\r\n * Numeric tolerance value\r\n */\r\n private static readonly _Epsilon = 1e-6;\r\n\r\n /**\r\n * Reference to the glTF Exporter\r\n */\r\n private _exporter: _Exporter;\r\n\r\n constructor(exporter: _Exporter) {\r\n this._textureMap = {};\r\n this._exporter = exporter;\r\n }\r\n\r\n /**\r\n * Specifies if two colors are approximately equal in value\r\n * @param color1 first color to compare to\r\n * @param color2 second color to compare to\r\n * @param epsilon threshold value\r\n */\r\n private static _FuzzyEquals(color1: Color3, color2: Color3, epsilon: number): boolean {\r\n return Scalar.WithinEpsilon(color1.r, color2.r, epsilon) && Scalar.WithinEpsilon(color1.g, color2.g, epsilon) && Scalar.WithinEpsilon(color1.b, color2.b, epsilon);\r\n }\r\n\r\n /**\r\n * Gets the materials from a Babylon scene and converts them to glTF materials\r\n * @param exportMaterials\r\n * @param mimeType texture mime type\r\n * @param hasTextureCoords specifies if texture coordinates are present on the material\r\n */\r\n public _convertMaterialsToGLTFAsync(exportMaterials: Set, mimeType: ImageMimeType, hasTextureCoords: boolean) {\r\n const promises: Promise[] = [];\r\n exportMaterials.forEach((material) => {\r\n if (material.getClassName() === \"StandardMaterial\") {\r\n promises.push(this._convertStandardMaterialAsync(material as StandardMaterial, mimeType, hasTextureCoords));\r\n } else if (material.getClassName().indexOf(\"PBR\") !== -1) {\r\n promises.push(this._convertPBRMaterialAsync(material as PBRMaterial, mimeType, hasTextureCoords));\r\n } else {\r\n Tools.Warn(`Unsupported material type: ${material.name}`);\r\n }\r\n });\r\n\r\n return Promise.all(promises).then(() => {\r\n /* do nothing */\r\n });\r\n }\r\n\r\n /**\r\n * Makes a copy of the glTF material without the texture parameters\r\n * @param originalMaterial original glTF material\r\n * @returns glTF material without texture parameters\r\n */\r\n public _stripTexturesFromMaterial(originalMaterial: IMaterial): IMaterial {\r\n const newMaterial: IMaterial = {};\r\n if (originalMaterial) {\r\n newMaterial.name = originalMaterial.name;\r\n newMaterial.doubleSided = originalMaterial.doubleSided;\r\n newMaterial.alphaMode = originalMaterial.alphaMode;\r\n newMaterial.alphaCutoff = originalMaterial.alphaCutoff;\r\n newMaterial.emissiveFactor = originalMaterial.emissiveFactor;\r\n const originalPBRMetallicRoughness = originalMaterial.pbrMetallicRoughness;\r\n if (originalPBRMetallicRoughness) {\r\n newMaterial.pbrMetallicRoughness = {};\r\n newMaterial.pbrMetallicRoughness.baseColorFactor = originalPBRMetallicRoughness.baseColorFactor;\r\n newMaterial.pbrMetallicRoughness.metallicFactor = originalPBRMetallicRoughness.metallicFactor;\r\n newMaterial.pbrMetallicRoughness.roughnessFactor = originalPBRMetallicRoughness.roughnessFactor;\r\n }\r\n }\r\n return newMaterial;\r\n }\r\n\r\n /**\r\n * Specifies if the material has any texture parameters present\r\n * @param material glTF Material\r\n * @returns boolean specifying if texture parameters are present\r\n */\r\n public _hasTexturesPresent(material: IMaterial): boolean {\r\n if (material.emissiveTexture || material.normalTexture || material.occlusionTexture) {\r\n return true;\r\n }\r\n const pbrMat = material.pbrMetallicRoughness;\r\n if (pbrMat) {\r\n if (pbrMat.baseColorTexture || pbrMat.metallicRoughnessTexture) {\r\n return true;\r\n }\r\n }\r\n\r\n if (material.extensions) {\r\n for (const extension in material.extensions) {\r\n const extensionObject = material.extensions[extension];\r\n if (extensionObject as IMaterialExtension) {\r\n return extensionObject.hasTextures?.();\r\n }\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n public _getTextureInfo(babylonTexture: Nullable): Nullable {\r\n if (babylonTexture) {\r\n const textureUid = babylonTexture.uid;\r\n if (textureUid in this._textureMap) {\r\n return this._textureMap[textureUid];\r\n }\r\n }\r\n return null;\r\n }\r\n\r\n /**\r\n * Converts a Babylon StandardMaterial to a glTF Metallic Roughness Material\r\n * @param babylonStandardMaterial\r\n * @returns glTF Metallic Roughness Material representation\r\n */\r\n public _convertToGLTFPBRMetallicRoughness(babylonStandardMaterial: StandardMaterial): IMaterialPbrMetallicRoughness {\r\n // Defines a cubic bezier curve where x is specular power and y is roughness\r\n const P0 = new Vector2(0, 1);\r\n const P1 = new Vector2(0, 0.1);\r\n const P2 = new Vector2(0, 0.1);\r\n const P3 = new Vector2(1300, 0.1);\r\n\r\n /**\r\n * Given the control points, solve for x based on a given t for a cubic bezier curve\r\n * @param t a value between 0 and 1\r\n * @param p0 first control point\r\n * @param p1 second control point\r\n * @param p2 third control point\r\n * @param p3 fourth control point\r\n * @returns number result of cubic bezier curve at the specified t\r\n */\r\n function cubicBezierCurve(t: number, p0: number, p1: number, p2: number, p3: number): number {\r\n return (1 - t) * (1 - t) * (1 - t) * p0 + 3 * (1 - t) * (1 - t) * t * p1 + 3 * (1 - t) * t * t * p2 + t * t * t * p3;\r\n }\r\n\r\n /**\r\n * Evaluates a specified specular power value to determine the appropriate roughness value,\r\n * based on a pre-defined cubic bezier curve with specular on the abscissa axis (x-axis)\r\n * and roughness on the ordinant axis (y-axis)\r\n * @param specularPower specular power of standard material\r\n * @returns Number representing the roughness value\r\n */\r\n function solveForRoughness(specularPower: number): number {\r\n // Given P0.x = 0, P1.x = 0, P2.x = 0\r\n // x = t * t * t * P3.x\r\n // t = (x / P3.x)^(1/3)\r\n const t = Math.pow(specularPower / P3.x, 0.333333);\r\n return cubicBezierCurve(t, P0.y, P1.y, P2.y, P3.y);\r\n }\r\n\r\n const diffuse = babylonStandardMaterial.diffuseColor.toLinearSpace(babylonStandardMaterial.getScene().getEngine().useExactSrgbConversions).scale(0.5);\r\n const opacity = babylonStandardMaterial.alpha;\r\n const specularPower = Scalar.Clamp(babylonStandardMaterial.specularPower, 0, _GLTFMaterialExporter._MaxSpecularPower);\r\n\r\n const roughness = solveForRoughness(specularPower);\r\n\r\n const glTFPbrMetallicRoughness: IMaterialPbrMetallicRoughness = {\r\n baseColorFactor: [diffuse.r, diffuse.g, diffuse.b, opacity],\r\n metallicFactor: 0,\r\n roughnessFactor: roughness,\r\n };\r\n\r\n return glTFPbrMetallicRoughness;\r\n }\r\n\r\n /**\r\n * Computes the metallic factor\r\n * @param diffuse diffused value\r\n * @param specular specular value\r\n * @param oneMinusSpecularStrength one minus the specular strength\r\n * @returns metallic value\r\n */\r\n public static _SolveMetallic(diffuse: number, specular: number, oneMinusSpecularStrength: number): number {\r\n if (specular < this._DielectricSpecular.r) {\r\n this._DielectricSpecular;\r\n return 0;\r\n }\r\n\r\n const a = this._DielectricSpecular.r;\r\n const b = (diffuse * oneMinusSpecularStrength) / (1.0 - this._DielectricSpecular.r) + specular - 2.0 * this._DielectricSpecular.r;\r\n const c = this._DielectricSpecular.r - specular;\r\n const D = b * b - 4.0 * a * c;\r\n return Scalar.Clamp((-b + Math.sqrt(D)) / (2.0 * a), 0, 1);\r\n }\r\n\r\n /**\r\n * Sets the glTF alpha mode to a glTF material from the Babylon Material\r\n * @param glTFMaterial glTF material\r\n * @param babylonMaterial Babylon material\r\n */\r\n private static _SetAlphaMode(glTFMaterial: IMaterial, babylonMaterial: Material & { alphaCutOff: number }): void {\r\n if (babylonMaterial.needAlphaBlending()) {\r\n glTFMaterial.alphaMode = MaterialAlphaMode.BLEND;\r\n } else if (babylonMaterial.needAlphaTesting()) {\r\n glTFMaterial.alphaMode = MaterialAlphaMode.MASK;\r\n glTFMaterial.alphaCutoff = babylonMaterial.alphaCutOff;\r\n }\r\n }\r\n\r\n /**\r\n * Converts a Babylon Standard Material to a glTF Material\r\n * @param babylonStandardMaterial BJS Standard Material\r\n * @param mimeType mime type to use for the textures\r\n * @param hasTextureCoords specifies if texture coordinates are present on the submesh to determine if textures should be applied\r\n */\r\n public _convertStandardMaterialAsync(babylonStandardMaterial: StandardMaterial, mimeType: ImageMimeType, hasTextureCoords: boolean): Promise {\r\n const materialMap = this._exporter._materialMap;\r\n const materials = this._exporter._materials;\r\n const promises = [];\r\n const pbrMetallicRoughness = this._convertToGLTFPBRMetallicRoughness(babylonStandardMaterial);\r\n\r\n const material: IMaterial = { name: babylonStandardMaterial.name };\r\n if (babylonStandardMaterial.backFaceCulling != null && !babylonStandardMaterial.backFaceCulling) {\r\n if (!babylonStandardMaterial.twoSidedLighting) {\r\n Tools.Warn(babylonStandardMaterial.name + \": Back-face culling disabled and two-sided lighting disabled is not supported in glTF.\");\r\n }\r\n material.doubleSided = true;\r\n }\r\n if (hasTextureCoords) {\r\n if (babylonStandardMaterial.diffuseTexture) {\r\n promises.push(\r\n this._exportTextureAsync(babylonStandardMaterial.diffuseTexture, mimeType).then((textureInfo) => {\r\n if (textureInfo) {\r\n pbrMetallicRoughness.baseColorTexture = textureInfo;\r\n }\r\n })\r\n );\r\n }\r\n const bumpTexture = babylonStandardMaterial.bumpTexture;\r\n if (bumpTexture) {\r\n promises.push(\r\n this._exportTextureAsync(bumpTexture, mimeType).then((textureInfo) => {\r\n if (textureInfo) {\r\n material.normalTexture = textureInfo;\r\n if (bumpTexture.level !== 1) {\r\n material.normalTexture.scale = bumpTexture.level;\r\n }\r\n }\r\n })\r\n );\r\n }\r\n if (babylonStandardMaterial.emissiveTexture) {\r\n material.emissiveFactor = [1.0, 1.0, 1.0];\r\n\r\n promises.push(\r\n this._exportTextureAsync(babylonStandardMaterial.emissiveTexture, mimeType).then((textureInfo) => {\r\n if (textureInfo) {\r\n material.emissiveTexture = textureInfo;\r\n }\r\n })\r\n );\r\n }\r\n if (babylonStandardMaterial.ambientTexture) {\r\n promises.push(\r\n this._exportTextureAsync(babylonStandardMaterial.ambientTexture, mimeType).then((textureInfo) => {\r\n if (textureInfo) {\r\n const occlusionTexture: IMaterialOcclusionTextureInfo = {\r\n index: textureInfo.index,\r\n };\r\n material.occlusionTexture = occlusionTexture;\r\n }\r\n })\r\n );\r\n }\r\n }\r\n\r\n if (babylonStandardMaterial.alpha < 1.0 || babylonStandardMaterial.opacityTexture) {\r\n if (babylonStandardMaterial.alphaMode === Constants.ALPHA_COMBINE) {\r\n material.alphaMode = MaterialAlphaMode.BLEND;\r\n } else {\r\n Tools.Warn(babylonStandardMaterial.name + \": glTF 2.0 does not support alpha mode: \" + babylonStandardMaterial.alphaMode.toString());\r\n }\r\n }\r\n if (babylonStandardMaterial.emissiveColor && !_GLTFMaterialExporter._FuzzyEquals(babylonStandardMaterial.emissiveColor, Color3.Black(), _GLTFMaterialExporter._Epsilon)) {\r\n material.emissiveFactor = babylonStandardMaterial.emissiveColor.asArray();\r\n }\r\n\r\n material.pbrMetallicRoughness = pbrMetallicRoughness;\r\n _GLTFMaterialExporter._SetAlphaMode(material, babylonStandardMaterial);\r\n\r\n materials.push(material);\r\n materialMap[babylonStandardMaterial.uniqueId] = materials.length - 1;\r\n\r\n return this._finishMaterial(promises, material, babylonStandardMaterial, mimeType);\r\n }\r\n\r\n private _finishMaterial(promises: Promise[], glTFMaterial: IMaterial, babylonMaterial: Material, mimeType: ImageMimeType) {\r\n return Promise.all(promises).then(() => {\r\n const textures = this._exporter._extensionsPostExportMaterialAdditionalTextures(\"exportMaterial\", glTFMaterial, babylonMaterial);\r\n let tasks: Nullable>[]> = null;\r\n\r\n for (const texture of textures) {\r\n if (!tasks) {\r\n tasks = [];\r\n }\r\n tasks.push(this._exportTextureAsync(texture, mimeType));\r\n }\r\n\r\n if (!tasks) {\r\n tasks = [Promise.resolve(null)];\r\n }\r\n\r\n return Promise.all(tasks).then(() => {\r\n const extensionWork = this._exporter._extensionsPostExportMaterialAsync(\"exportMaterial\", glTFMaterial, babylonMaterial);\r\n if (!extensionWork) {\r\n return glTFMaterial;\r\n }\r\n return extensionWork.then(() => glTFMaterial);\r\n });\r\n });\r\n }\r\n\r\n /**\r\n * Converts an image typed array buffer to a base64 image\r\n * @param buffer typed array buffer\r\n * @param width width of the image\r\n * @param height height of the image\r\n * @param mimeType mimetype of the image\r\n * @returns base64 image string\r\n */\r\n private async _getImageDataAsync(buffer: Uint8Array | Float32Array, width: number, height: number, mimeType: ImageMimeType): Promise {\r\n const textureType = Constants.TEXTURETYPE_UNSIGNED_INT;\r\n\r\n const hostingScene = this._exporter._babylonScene;\r\n const engine = hostingScene.getEngine();\r\n\r\n // Create a temporary texture with the texture buffer data\r\n const tempTexture = engine.createRawTexture(buffer, width, height, Constants.TEXTUREFORMAT_RGBA, false, true, Texture.NEAREST_SAMPLINGMODE, null, textureType);\r\n\r\n await TextureTools.ApplyPostProcess(\"pass\", tempTexture, hostingScene, textureType, Constants.TEXTURE_NEAREST_SAMPLINGMODE, Constants.TEXTUREFORMAT_RGBA);\r\n\r\n const data = await engine._readTexturePixels(tempTexture, width, height);\r\n\r\n return (await DumpTools.DumpDataAsync(width, height, data, mimeType, undefined, true, true)) as ArrayBuffer;\r\n }\r\n\r\n /**\r\n * Generates a white texture based on the specified width and height\r\n * @param width width of the texture in pixels\r\n * @param height height of the texture in pixels\r\n * @param scene babylonjs scene\r\n * @returns white texture\r\n */\r\n private _createWhiteTexture(width: number, height: number, scene: Scene): Texture {\r\n const data = new Uint8Array(width * height * 4);\r\n\r\n for (let i = 0; i < data.length; i = i + 4) {\r\n data[i] = data[i + 1] = data[i + 2] = data[i + 3] = 0xff;\r\n }\r\n\r\n const rawTexture = RawTexture.CreateRGBATexture(data, width, height, scene);\r\n\r\n return rawTexture;\r\n }\r\n\r\n /**\r\n * Resizes the two source textures to the same dimensions. If a texture is null, a default white texture is generated. If both textures are null, returns null\r\n * @param texture1 first texture to resize\r\n * @param texture2 second texture to resize\r\n * @param scene babylonjs scene\r\n * @returns resized textures or null\r\n */\r\n private _resizeTexturesToSameDimensions(texture1: Nullable, texture2: Nullable, scene: Scene): { texture1: BaseTexture; texture2: BaseTexture } {\r\n const texture1Size = texture1 ? texture1.getSize() : { width: 0, height: 0 };\r\n const texture2Size = texture2 ? texture2.getSize() : { width: 0, height: 0 };\r\n let resizedTexture1: BaseTexture;\r\n let resizedTexture2: BaseTexture;\r\n\r\n if (texture1Size.width < texture2Size.width) {\r\n if (texture1 && texture1 instanceof Texture) {\r\n resizedTexture1 = TextureTools.CreateResizedCopy(texture1, texture2Size.width, texture2Size.height, true);\r\n } else {\r\n resizedTexture1 = this._createWhiteTexture(texture2Size.width, texture2Size.height, scene);\r\n }\r\n resizedTexture2 = texture2!;\r\n } else if (texture1Size.width > texture2Size.width) {\r\n if (texture2 && texture2 instanceof Texture) {\r\n resizedTexture2 = TextureTools.CreateResizedCopy(texture2, texture1Size.width, texture1Size.height, true);\r\n } else {\r\n resizedTexture2 = this._createWhiteTexture(texture1Size.width, texture1Size.height, scene);\r\n }\r\n resizedTexture1 = texture1!;\r\n } else {\r\n resizedTexture1 = texture1!;\r\n resizedTexture2 = texture2!;\r\n }\r\n\r\n return {\r\n texture1: resizedTexture1!,\r\n texture2: resizedTexture2!,\r\n };\r\n }\r\n\r\n /**\r\n * Converts an array of pixels to a Float32Array\r\n * Throws an error if the pixel format is not supported\r\n * @param pixels - array buffer containing pixel values\r\n * @returns Float32 of pixels\r\n */\r\n private _convertPixelArrayToFloat32(pixels: ArrayBufferView): Float32Array {\r\n if (pixels instanceof Uint8Array) {\r\n const length = pixels.length;\r\n const buffer = new Float32Array(pixels.length);\r\n for (let i = 0; i < length; ++i) {\r\n buffer[i] = pixels[i] / 255;\r\n }\r\n return buffer;\r\n } else if (pixels instanceof Float32Array) {\r\n return pixels;\r\n } else {\r\n throw new Error(\"Unsupported pixel format!\");\r\n }\r\n }\r\n\r\n /**\r\n * Convert Specular Glossiness Textures to Metallic Roughness\r\n * See link below for info on the material conversions from PBR Metallic/Roughness and Specular/Glossiness\r\n * @link https://github.com/KhronosGroup/glTF/blob/main/extensions/2.0/Archived/KHR_materials_pbrSpecularGlossiness/examples/convert-between-workflows-bjs/js/babylon.pbrUtilities.js\r\n * @param diffuseTexture texture used to store diffuse information\r\n * @param specularGlossinessTexture texture used to store specular and glossiness information\r\n * @param factors specular glossiness material factors\r\n * @param mimeType the mime type to use for the texture\r\n * @returns pbr metallic roughness interface or null\r\n */\r\n private async _convertSpecularGlossinessTexturesToMetallicRoughnessAsync(\r\n diffuseTexture: Nullable,\r\n specularGlossinessTexture: Nullable,\r\n factors: _IPBRSpecularGlossiness,\r\n mimeType: ImageMimeType\r\n ): Promise<_IPBRMetallicRoughness> {\r\n const promises = new Array>();\r\n if (!(diffuseTexture || specularGlossinessTexture)) {\r\n return Promise.reject(\"_ConvertSpecularGlosinessTexturesToMetallicRoughness: diffuse and specular glossiness textures are not defined!\");\r\n }\r\n\r\n const scene: Nullable = diffuseTexture ? diffuseTexture.getScene() : specularGlossinessTexture ? specularGlossinessTexture.getScene() : null;\r\n if (scene) {\r\n const resizedTextures = this._resizeTexturesToSameDimensions(diffuseTexture, specularGlossinessTexture, scene);\r\n\r\n const diffuseSize = resizedTextures.texture1?.getSize();\r\n\r\n let diffuseBuffer: Float32Array;\r\n let specularGlossinessBuffer: Float32Array;\r\n\r\n const width = diffuseSize.width;\r\n const height = diffuseSize.height;\r\n\r\n const diffusePixels = await resizedTextures.texture1.readPixels();\r\n const specularPixels = await resizedTextures.texture2.readPixels();\r\n\r\n if (diffusePixels) {\r\n diffuseBuffer = this._convertPixelArrayToFloat32(diffusePixels);\r\n } else {\r\n return Promise.reject(\"Failed to retrieve pixels from diffuse texture!\");\r\n }\r\n if (specularPixels) {\r\n specularGlossinessBuffer = this._convertPixelArrayToFloat32(specularPixels);\r\n } else {\r\n return Promise.reject(\"Failed to retrieve pixels from specular glossiness texture!\");\r\n }\r\n\r\n const byteLength = specularGlossinessBuffer.byteLength;\r\n\r\n const metallicRoughnessBuffer = new Uint8Array(byteLength);\r\n const baseColorBuffer = new Uint8Array(byteLength);\r\n\r\n const strideSize = 4;\r\n const maxBaseColor = Color3.Black();\r\n let maxMetallic = 0;\r\n let maxRoughness = 0;\r\n\r\n for (let h = 0; h < height; ++h) {\r\n for (let w = 0; w < width; ++w) {\r\n const offset = (width * h + w) * strideSize;\r\n\r\n const diffuseColor = new Color3(diffuseBuffer[offset], diffuseBuffer[offset + 1], diffuseBuffer[offset + 2])\r\n .toLinearSpace(scene.getEngine().useExactSrgbConversions)\r\n .multiply(factors.diffuseColor);\r\n const specularColor = new Color3(specularGlossinessBuffer[offset], specularGlossinessBuffer[offset + 1], specularGlossinessBuffer[offset + 2])\r\n .toLinearSpace(scene.getEngine().useExactSrgbConversions)\r\n .multiply(factors.specularColor);\r\n const glossiness = specularGlossinessBuffer[offset + 3] * factors.glossiness;\r\n\r\n const specularGlossiness: _IPBRSpecularGlossiness = {\r\n diffuseColor: diffuseColor,\r\n specularColor: specularColor,\r\n glossiness: glossiness,\r\n };\r\n\r\n const metallicRoughness = this._convertSpecularGlossinessToMetallicRoughness(specularGlossiness);\r\n maxBaseColor.r = Math.max(maxBaseColor.r, metallicRoughness.baseColor.r);\r\n maxBaseColor.g = Math.max(maxBaseColor.g, metallicRoughness.baseColor.g);\r\n maxBaseColor.b = Math.max(maxBaseColor.b, metallicRoughness.baseColor.b);\r\n maxMetallic = Math.max(maxMetallic, metallicRoughness.metallic!);\r\n maxRoughness = Math.max(maxRoughness, metallicRoughness.roughness!);\r\n\r\n baseColorBuffer[offset] = metallicRoughness.baseColor.r * 255;\r\n baseColorBuffer[offset + 1] = metallicRoughness.baseColor.g * 255;\r\n baseColorBuffer[offset + 2] = metallicRoughness.baseColor.b * 255;\r\n baseColorBuffer[offset + 3] = resizedTextures.texture1.hasAlpha ? diffuseBuffer[offset + 3] * 255 : 255;\r\n\r\n metallicRoughnessBuffer[offset] = 0;\r\n metallicRoughnessBuffer[offset + 1] = metallicRoughness.roughness! * 255;\r\n metallicRoughnessBuffer[offset + 2] = metallicRoughness.metallic! * 255;\r\n metallicRoughnessBuffer[offset + 3] = 255;\r\n }\r\n }\r\n\r\n // Retrieves the metallic roughness factors from the maximum texture values.\r\n const metallicRoughnessFactors: _IPBRMetallicRoughness = {\r\n baseColor: maxBaseColor,\r\n metallic: maxMetallic,\r\n roughness: maxRoughness,\r\n };\r\n\r\n let writeOutMetallicRoughnessTexture = false;\r\n let writeOutBaseColorTexture = false;\r\n\r\n for (let h = 0; h < height; ++h) {\r\n for (let w = 0; w < width; ++w) {\r\n const destinationOffset = (width * h + w) * strideSize;\r\n\r\n baseColorBuffer[destinationOffset] /= metallicRoughnessFactors.baseColor.r > _GLTFMaterialExporter._Epsilon ? metallicRoughnessFactors.baseColor.r : 1;\r\n baseColorBuffer[destinationOffset + 1] /= metallicRoughnessFactors.baseColor.g > _GLTFMaterialExporter._Epsilon ? metallicRoughnessFactors.baseColor.g : 1;\r\n baseColorBuffer[destinationOffset + 2] /= metallicRoughnessFactors.baseColor.b > _GLTFMaterialExporter._Epsilon ? metallicRoughnessFactors.baseColor.b : 1;\r\n\r\n const linearBaseColorPixel = Color3.FromInts(\r\n baseColorBuffer[destinationOffset],\r\n baseColorBuffer[destinationOffset + 1],\r\n baseColorBuffer[destinationOffset + 2]\r\n );\r\n const sRGBBaseColorPixel = linearBaseColorPixel.toGammaSpace(scene.getEngine().useExactSrgbConversions);\r\n baseColorBuffer[destinationOffset] = sRGBBaseColorPixel.r * 255;\r\n baseColorBuffer[destinationOffset + 1] = sRGBBaseColorPixel.g * 255;\r\n baseColorBuffer[destinationOffset + 2] = sRGBBaseColorPixel.b * 255;\r\n\r\n if (!_GLTFMaterialExporter._FuzzyEquals(sRGBBaseColorPixel, Color3.White(), _GLTFMaterialExporter._Epsilon)) {\r\n writeOutBaseColorTexture = true;\r\n }\r\n\r\n metallicRoughnessBuffer[destinationOffset + 1] /=\r\n metallicRoughnessFactors.roughness! > _GLTFMaterialExporter._Epsilon ? metallicRoughnessFactors.roughness! : 1;\r\n metallicRoughnessBuffer[destinationOffset + 2] /= metallicRoughnessFactors.metallic! > _GLTFMaterialExporter._Epsilon ? metallicRoughnessFactors.metallic! : 1;\r\n\r\n const metallicRoughnessPixel = Color3.FromInts(255, metallicRoughnessBuffer[destinationOffset + 1], metallicRoughnessBuffer[destinationOffset + 2]);\r\n\r\n if (!_GLTFMaterialExporter._FuzzyEquals(metallicRoughnessPixel, Color3.White(), _GLTFMaterialExporter._Epsilon)) {\r\n writeOutMetallicRoughnessTexture = true;\r\n }\r\n }\r\n }\r\n\r\n if (writeOutMetallicRoughnessTexture) {\r\n promises.push(\r\n this._getImageDataAsync(metallicRoughnessBuffer, width, height, mimeType).then((data) => {\r\n metallicRoughnessFactors.metallicRoughnessTextureData = data;\r\n })\r\n );\r\n }\r\n if (writeOutBaseColorTexture) {\r\n promises.push(\r\n this._getImageDataAsync(baseColorBuffer, width, height, mimeType).then((data) => {\r\n metallicRoughnessFactors.baseColorTextureData = data;\r\n })\r\n );\r\n }\r\n\r\n return Promise.all(promises).then(() => {\r\n return metallicRoughnessFactors;\r\n });\r\n } else {\r\n return Promise.reject(\"_ConvertSpecularGlossinessTexturesToMetallicRoughness: Scene from textures is missing!\");\r\n }\r\n }\r\n\r\n /**\r\n * Converts specular glossiness material properties to metallic roughness\r\n * @param specularGlossiness interface with specular glossiness material properties\r\n * @returns interface with metallic roughness material properties\r\n */\r\n private _convertSpecularGlossinessToMetallicRoughness(specularGlossiness: _IPBRSpecularGlossiness): _IPBRMetallicRoughness {\r\n const diffusePerceivedBrightness = this._getPerceivedBrightness(specularGlossiness.diffuseColor);\r\n const specularPerceivedBrightness = this._getPerceivedBrightness(specularGlossiness.specularColor);\r\n const oneMinusSpecularStrength = 1 - this._getMaxComponent(specularGlossiness.specularColor);\r\n const metallic = _GLTFMaterialExporter._SolveMetallic(diffusePerceivedBrightness, specularPerceivedBrightness, oneMinusSpecularStrength);\r\n const baseColorFromDiffuse = specularGlossiness.diffuseColor.scale(\r\n oneMinusSpecularStrength / (1.0 - _GLTFMaterialExporter._DielectricSpecular.r) / Math.max(1 - metallic, _GLTFMaterialExporter._Epsilon)\r\n );\r\n const baseColorFromSpecular = specularGlossiness.specularColor\r\n .subtract(_GLTFMaterialExporter._DielectricSpecular.scale(1 - metallic))\r\n .scale(1 / Math.max(metallic, _GLTFMaterialExporter._Epsilon));\r\n let baseColor = Color3.Lerp(baseColorFromDiffuse, baseColorFromSpecular, metallic * metallic);\r\n baseColor = baseColor.clampToRef(0, 1, baseColor);\r\n\r\n const metallicRoughness: _IPBRMetallicRoughness = {\r\n baseColor: baseColor,\r\n metallic: metallic,\r\n roughness: 1 - specularGlossiness.glossiness,\r\n };\r\n\r\n return metallicRoughness;\r\n }\r\n\r\n /**\r\n * Calculates the surface reflectance, independent of lighting conditions\r\n * @param color Color source to calculate brightness from\r\n * @returns number representing the perceived brightness, or zero if color is undefined\r\n */\r\n private _getPerceivedBrightness(color: Color3): number {\r\n if (color) {\r\n return Math.sqrt(0.299 * color.r * color.r + 0.587 * color.g * color.g + 0.114 * color.b * color.b);\r\n }\r\n return 0;\r\n }\r\n\r\n /**\r\n * Returns the maximum color component value\r\n * @param color\r\n * @returns maximum color component value, or zero if color is null or undefined\r\n */\r\n private _getMaxComponent(color: Color3): number {\r\n if (color) {\r\n return Math.max(color.r, Math.max(color.g, color.b));\r\n }\r\n return 0;\r\n }\r\n\r\n /**\r\n * Convert a PBRMaterial (Metallic/Roughness) to Metallic Roughness factors\r\n * @param babylonPBRMaterial BJS PBR Metallic Roughness Material\r\n * @param mimeType mime type to use for the textures\r\n * @param glTFPbrMetallicRoughness glTF PBR Metallic Roughness interface\r\n * @param hasTextureCoords specifies if texture coordinates are present on the submesh to determine if textures should be applied\r\n * @returns glTF PBR Metallic Roughness factors\r\n */\r\n private _convertMetalRoughFactorsToMetallicRoughnessAsync(\r\n babylonPBRMaterial: PBRBaseMaterial,\r\n mimeType: ImageMimeType,\r\n glTFPbrMetallicRoughness: IMaterialPbrMetallicRoughness,\r\n hasTextureCoords: boolean\r\n ): Promise<_IPBRMetallicRoughness> {\r\n const promises = [];\r\n const baseColor = babylonPBRMaterial._albedoColor;\r\n const metallic = babylonPBRMaterial._metallic;\r\n const roughness = babylonPBRMaterial._roughness;\r\n const metallicRoughness: _IPBRMetallicRoughness = {\r\n baseColor: baseColor,\r\n metallic: metallic,\r\n roughness: roughness,\r\n };\r\n\r\n if (hasTextureCoords) {\r\n const albedoTexture = babylonPBRMaterial._albedoTexture;\r\n if (albedoTexture) {\r\n promises.push(\r\n this._exportTextureAsync(babylonPBRMaterial._albedoTexture!, mimeType).then((glTFTexture) => {\r\n if (glTFTexture) {\r\n glTFPbrMetallicRoughness.baseColorTexture = glTFTexture;\r\n }\r\n })\r\n );\r\n }\r\n const metallicTexture = babylonPBRMaterial._metallicTexture;\r\n if (metallicTexture) {\r\n promises.push(\r\n this._exportTextureAsync(metallicTexture, mimeType).then((glTFTexture) => {\r\n if (glTFTexture) {\r\n glTFPbrMetallicRoughness.metallicRoughnessTexture = glTFTexture;\r\n }\r\n })\r\n );\r\n }\r\n }\r\n return Promise.all(promises).then(() => {\r\n return metallicRoughness;\r\n });\r\n }\r\n\r\n private _getTextureSampler(texture: Nullable): ISampler {\r\n const sampler: ISampler = {};\r\n if (!texture || !(texture instanceof Texture)) {\r\n return sampler;\r\n }\r\n\r\n const wrapS = this._getGLTFTextureWrapMode(texture.wrapU);\r\n if (wrapS !== TextureWrapMode.REPEAT) {\r\n sampler.wrapS = wrapS;\r\n }\r\n\r\n const wrapT = this._getGLTFTextureWrapMode(texture.wrapV);\r\n if (wrapT !== TextureWrapMode.REPEAT) {\r\n sampler.wrapT = wrapT;\r\n }\r\n\r\n switch (texture.samplingMode) {\r\n case Texture.LINEAR_LINEAR: {\r\n sampler.magFilter = TextureMagFilter.LINEAR;\r\n sampler.minFilter = TextureMinFilter.LINEAR;\r\n break;\r\n }\r\n case Texture.LINEAR_NEAREST: {\r\n sampler.magFilter = TextureMagFilter.LINEAR;\r\n sampler.minFilter = TextureMinFilter.NEAREST;\r\n break;\r\n }\r\n case Texture.NEAREST_LINEAR: {\r\n sampler.magFilter = TextureMagFilter.NEAREST;\r\n sampler.minFilter = TextureMinFilter.LINEAR;\r\n break;\r\n }\r\n case Texture.NEAREST_LINEAR_MIPLINEAR: {\r\n sampler.magFilter = TextureMagFilter.NEAREST;\r\n sampler.minFilter = TextureMinFilter.LINEAR_MIPMAP_LINEAR;\r\n break;\r\n }\r\n case Texture.NEAREST_NEAREST: {\r\n sampler.magFilter = TextureMagFilter.NEAREST;\r\n sampler.minFilter = TextureMinFilter.NEAREST;\r\n break;\r\n }\r\n case Texture.NEAREST_LINEAR_MIPNEAREST: {\r\n sampler.magFilter = TextureMagFilter.NEAREST;\r\n sampler.minFilter = TextureMinFilter.LINEAR_MIPMAP_NEAREST;\r\n break;\r\n }\r\n case Texture.LINEAR_NEAREST_MIPNEAREST: {\r\n sampler.magFilter = TextureMagFilter.LINEAR;\r\n sampler.minFilter = TextureMinFilter.NEAREST_MIPMAP_NEAREST;\r\n break;\r\n }\r\n case Texture.LINEAR_NEAREST_MIPLINEAR: {\r\n sampler.magFilter = TextureMagFilter.LINEAR;\r\n sampler.minFilter = TextureMinFilter.NEAREST_MIPMAP_LINEAR;\r\n break;\r\n }\r\n case Texture.NEAREST_NEAREST_MIPLINEAR: {\r\n sampler.magFilter = TextureMagFilter.NEAREST;\r\n sampler.minFilter = TextureMinFilter.NEAREST_MIPMAP_LINEAR;\r\n break;\r\n }\r\n case Texture.LINEAR_LINEAR_MIPLINEAR: {\r\n sampler.magFilter = TextureMagFilter.LINEAR;\r\n sampler.minFilter = TextureMinFilter.LINEAR_MIPMAP_LINEAR;\r\n break;\r\n }\r\n case Texture.LINEAR_LINEAR_MIPNEAREST: {\r\n sampler.magFilter = TextureMagFilter.LINEAR;\r\n sampler.minFilter = TextureMinFilter.LINEAR_MIPMAP_NEAREST;\r\n break;\r\n }\r\n case Texture.NEAREST_NEAREST_MIPNEAREST: {\r\n sampler.magFilter = TextureMagFilter.NEAREST;\r\n sampler.minFilter = TextureMinFilter.NEAREST_MIPMAP_NEAREST;\r\n break;\r\n }\r\n }\r\n\r\n return sampler;\r\n }\r\n\r\n private _getGLTFTextureWrapMode(wrapMode: number): TextureWrapMode {\r\n switch (wrapMode) {\r\n case Texture.WRAP_ADDRESSMODE: {\r\n return TextureWrapMode.REPEAT;\r\n }\r\n case Texture.CLAMP_ADDRESSMODE: {\r\n return TextureWrapMode.CLAMP_TO_EDGE;\r\n }\r\n case Texture.MIRROR_ADDRESSMODE: {\r\n return TextureWrapMode.MIRRORED_REPEAT;\r\n }\r\n default: {\r\n Tools.Error(`Unsupported Texture Wrap Mode ${wrapMode}!`);\r\n return TextureWrapMode.REPEAT;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Convert a PBRMaterial (Specular/Glossiness) to Metallic Roughness factors\r\n * @param babylonPBRMaterial BJS PBR Metallic Roughness Material\r\n * @param mimeType mime type to use for the textures\r\n * @param glTFPbrMetallicRoughness glTF PBR Metallic Roughness interface\r\n * @param hasTextureCoords specifies if texture coordinates are present on the submesh to determine if textures should be applied\r\n * @returns glTF PBR Metallic Roughness factors\r\n */\r\n private _convertSpecGlossFactorsToMetallicRoughnessAsync(\r\n babylonPBRMaterial: PBRBaseMaterial,\r\n mimeType: ImageMimeType,\r\n pbrMetallicRoughness: IMaterialPbrMetallicRoughness,\r\n hasTextureCoords: boolean\r\n ): Promise<_IPBRMetallicRoughness> {\r\n return Promise.resolve().then(() => {\r\n const specGloss: _IPBRSpecularGlossiness = {\r\n diffuseColor: babylonPBRMaterial._albedoColor,\r\n specularColor: babylonPBRMaterial._reflectivityColor,\r\n glossiness: babylonPBRMaterial._microSurface,\r\n };\r\n const albedoTexture = babylonPBRMaterial._albedoTexture;\r\n const reflectivityTexture = babylonPBRMaterial._reflectivityTexture;\r\n const useMicrosurfaceFromReflectivityMapAlpha = babylonPBRMaterial._useMicroSurfaceFromReflectivityMapAlpha;\r\n if (reflectivityTexture && !useMicrosurfaceFromReflectivityMapAlpha) {\r\n return Promise.reject(\"_ConvertPBRMaterial: Glossiness values not included in the reflectivity texture are currently not supported\");\r\n }\r\n if ((albedoTexture || reflectivityTexture) && hasTextureCoords) {\r\n const samplerIndex = this._exportTextureSampler(albedoTexture || reflectivityTexture);\r\n return this._convertSpecularGlossinessTexturesToMetallicRoughnessAsync(albedoTexture, reflectivityTexture, specGloss, mimeType).then((metallicRoughnessFactors) => {\r\n const textures = this._exporter._textures;\r\n if (metallicRoughnessFactors.baseColorTextureData) {\r\n const imageIndex = this._exportImage(`baseColor${textures.length}`, mimeType, metallicRoughnessFactors.baseColorTextureData);\r\n pbrMetallicRoughness.baseColorTexture = this._exportTextureInfo(imageIndex, samplerIndex, albedoTexture?.coordinatesIndex);\r\n }\r\n if (metallicRoughnessFactors.metallicRoughnessTextureData) {\r\n const imageIndex = this._exportImage(`metallicRoughness${textures.length}`, mimeType, metallicRoughnessFactors.metallicRoughnessTextureData);\r\n pbrMetallicRoughness.metallicRoughnessTexture = this._exportTextureInfo(imageIndex, samplerIndex, reflectivityTexture?.coordinatesIndex);\r\n }\r\n\r\n return metallicRoughnessFactors;\r\n });\r\n } else {\r\n return this._convertSpecularGlossinessToMetallicRoughness(specGloss);\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Converts a Babylon PBR Base Material to a glTF Material\r\n * @param babylonPBRMaterial BJS PBR Base Material\r\n * @param mimeType mime type to use for the textures\r\n * @param hasTextureCoords specifies if texture coordinates are present on the submesh to determine if textures should be applied\r\n */\r\n public _convertPBRMaterialAsync(babylonPBRMaterial: PBRBaseMaterial, mimeType: ImageMimeType, hasTextureCoords: boolean): Promise {\r\n const glTFPbrMetallicRoughness: IMaterialPbrMetallicRoughness = {};\r\n const glTFMaterial: IMaterial = {\r\n name: babylonPBRMaterial.name,\r\n };\r\n const useMetallicRoughness = babylonPBRMaterial.isMetallicWorkflow();\r\n\r\n if (useMetallicRoughness) {\r\n const albedoColor = babylonPBRMaterial._albedoColor;\r\n const alpha = babylonPBRMaterial.alpha;\r\n if (albedoColor) {\r\n glTFPbrMetallicRoughness.baseColorFactor = [albedoColor.r, albedoColor.g, albedoColor.b, alpha];\r\n }\r\n return this._convertMetalRoughFactorsToMetallicRoughnessAsync(babylonPBRMaterial, mimeType, glTFPbrMetallicRoughness, hasTextureCoords).then((metallicRoughness) => {\r\n return this._setMetallicRoughnessPbrMaterial(metallicRoughness, babylonPBRMaterial, glTFMaterial, glTFPbrMetallicRoughness, mimeType, hasTextureCoords);\r\n });\r\n } else {\r\n return this._convertSpecGlossFactorsToMetallicRoughnessAsync(babylonPBRMaterial, mimeType, glTFPbrMetallicRoughness, hasTextureCoords).then((metallicRoughness) => {\r\n return this._setMetallicRoughnessPbrMaterial(metallicRoughness, babylonPBRMaterial, glTFMaterial, glTFPbrMetallicRoughness, mimeType, hasTextureCoords);\r\n });\r\n }\r\n }\r\n\r\n private _setMetallicRoughnessPbrMaterial(\r\n metallicRoughness: Nullable<_IPBRMetallicRoughness>,\r\n babylonPBRMaterial: PBRBaseMaterial,\r\n glTFMaterial: IMaterial,\r\n glTFPbrMetallicRoughness: IMaterialPbrMetallicRoughness,\r\n mimeType: ImageMimeType,\r\n hasTextureCoords: boolean\r\n ): Promise {\r\n const materialMap = this._exporter._materialMap;\r\n const materials = this._exporter._materials;\r\n const promises = [];\r\n if (metallicRoughness) {\r\n _GLTFMaterialExporter._SetAlphaMode(glTFMaterial, babylonPBRMaterial as PBRMaterial);\r\n if (\r\n !(\r\n _GLTFMaterialExporter._FuzzyEquals(metallicRoughness.baseColor, Color3.White(), _GLTFMaterialExporter._Epsilon) &&\r\n babylonPBRMaterial.alpha >= _GLTFMaterialExporter._Epsilon\r\n )\r\n ) {\r\n glTFPbrMetallicRoughness.baseColorFactor = [metallicRoughness.baseColor.r, metallicRoughness.baseColor.g, metallicRoughness.baseColor.b, babylonPBRMaterial.alpha];\r\n }\r\n\r\n if (metallicRoughness.metallic != null && metallicRoughness.metallic !== 1) {\r\n glTFPbrMetallicRoughness.metallicFactor = metallicRoughness.metallic;\r\n }\r\n if (metallicRoughness.roughness != null && metallicRoughness.roughness !== 1) {\r\n glTFPbrMetallicRoughness.roughnessFactor = metallicRoughness.roughness;\r\n }\r\n\r\n if (babylonPBRMaterial.backFaceCulling != null && !babylonPBRMaterial.backFaceCulling) {\r\n if (!babylonPBRMaterial._twoSidedLighting) {\r\n Tools.Warn(babylonPBRMaterial.name + \": Back-face culling disabled and two-sided lighting disabled is not supported in glTF.\");\r\n }\r\n glTFMaterial.doubleSided = true;\r\n }\r\n\r\n if (hasTextureCoords) {\r\n const bumpTexture = babylonPBRMaterial._bumpTexture;\r\n if (bumpTexture) {\r\n const promise = this._exportTextureAsync(bumpTexture, mimeType).then((glTFTexture) => {\r\n if (glTFTexture) {\r\n glTFMaterial.normalTexture = glTFTexture;\r\n if (bumpTexture.level !== 1) {\r\n glTFMaterial.normalTexture.scale = bumpTexture.level;\r\n }\r\n }\r\n });\r\n promises.push(promise);\r\n }\r\n const ambientTexture = babylonPBRMaterial._ambientTexture;\r\n if (ambientTexture) {\r\n const promise = this._exportTextureAsync(ambientTexture, mimeType).then((glTFTexture) => {\r\n if (glTFTexture) {\r\n const occlusionTexture: IMaterialOcclusionTextureInfo = {\r\n index: glTFTexture.index,\r\n texCoord: glTFTexture.texCoord,\r\n extensions: glTFTexture.extensions,\r\n };\r\n\r\n glTFMaterial.occlusionTexture = occlusionTexture;\r\n const ambientTextureStrength = babylonPBRMaterial._ambientTextureStrength;\r\n if (ambientTextureStrength) {\r\n occlusionTexture.strength = ambientTextureStrength;\r\n }\r\n }\r\n });\r\n promises.push(promise);\r\n }\r\n const emissiveTexture = babylonPBRMaterial._emissiveTexture;\r\n if (emissiveTexture) {\r\n const promise = this._exportTextureAsync(emissiveTexture, mimeType).then((glTFTexture) => {\r\n if (glTFTexture) {\r\n glTFMaterial.emissiveTexture = glTFTexture;\r\n }\r\n });\r\n promises.push(promise);\r\n }\r\n }\r\n const emissiveColor = babylonPBRMaterial._emissiveColor;\r\n if (!_GLTFMaterialExporter._FuzzyEquals(emissiveColor, Color3.Black(), _GLTFMaterialExporter._Epsilon)) {\r\n glTFMaterial.emissiveFactor = emissiveColor.asArray();\r\n }\r\n\r\n glTFMaterial.pbrMetallicRoughness = glTFPbrMetallicRoughness;\r\n materials.push(glTFMaterial);\r\n materialMap[babylonPBRMaterial.uniqueId] = materials.length - 1;\r\n }\r\n\r\n return this._finishMaterial(promises, glTFMaterial, babylonPBRMaterial, mimeType);\r\n }\r\n\r\n private _getPixelsFromTexture(babylonTexture: BaseTexture): Promise> {\r\n const pixels =\r\n babylonTexture.textureType === Constants.TEXTURETYPE_UNSIGNED_INT\r\n ? (babylonTexture.readPixels() as Promise)\r\n : (babylonTexture.readPixels() as Promise);\r\n return pixels;\r\n }\r\n\r\n /**\r\n * Extracts a texture from a Babylon texture into file data and glTF data\r\n * @param babylonTexture Babylon texture to extract\r\n * @param mimeType Mime Type of the babylonTexture\r\n * @returns glTF texture info, or null if the texture format is not supported\r\n */\r\n public _exportTextureAsync(babylonTexture: BaseTexture, mimeType: ImageMimeType): Promise> {\r\n const extensionPromise = this._exporter._extensionsPreExportTextureAsync(\"exporter\", babylonTexture as Texture, mimeType);\r\n if (!extensionPromise) {\r\n return this._exportTextureInfoAsync(babylonTexture, mimeType);\r\n }\r\n\r\n return extensionPromise.then((texture) => {\r\n if (!texture) {\r\n return this._exportTextureInfoAsync(babylonTexture, mimeType);\r\n }\r\n return this._exportTextureInfoAsync(texture, mimeType);\r\n });\r\n }\r\n\r\n public async _exportTextureInfoAsync(babylonTexture: BaseTexture, mimeType: ImageMimeType): Promise> {\r\n const textureUid = babylonTexture.uid;\r\n if (!(textureUid in this._textureMap)) {\r\n const pixels = await this._getPixelsFromTexture(babylonTexture);\r\n if (!pixels) {\r\n return null;\r\n }\r\n\r\n const samplerIndex = this._exportTextureSampler(babylonTexture);\r\n\r\n // Preserve texture mime type if defined\r\n const textureMimeType = (babylonTexture as Texture).mimeType;\r\n if (textureMimeType) {\r\n switch (textureMimeType) {\r\n case \"image/jpeg\":\r\n case \"image/png\":\r\n case \"image/webp\":\r\n mimeType = textureMimeType as ImageMimeType;\r\n break;\r\n default:\r\n Tools.Warn(`Unsupported media type: ${textureMimeType}`);\r\n break;\r\n }\r\n }\r\n\r\n const internalTextureToImage = this._internalTextureToImage;\r\n const internalTextureUniqueId = babylonTexture.getInternalTexture()!.uniqueId;\r\n internalTextureToImage[internalTextureUniqueId] ||= {};\r\n let imageIndexPromise = internalTextureToImage[internalTextureUniqueId][mimeType];\r\n if (imageIndexPromise === undefined) {\r\n const size = babylonTexture.getSize();\r\n imageIndexPromise = (async () => {\r\n const data = await this._getImageDataAsync(pixels, size.width, size.height, mimeType);\r\n return this._exportImage(babylonTexture.name, mimeType, data);\r\n })();\r\n internalTextureToImage[internalTextureUniqueId][mimeType] = imageIndexPromise;\r\n }\r\n\r\n const textureInfo = this._exportTextureInfo(await imageIndexPromise, samplerIndex, babylonTexture.coordinatesIndex);\r\n this._textureMap[textureUid] = textureInfo;\r\n this._exporter._extensionsPostExportTextures(\"exporter\", this._textureMap[textureUid], babylonTexture);\r\n }\r\n\r\n return this._textureMap[textureUid];\r\n }\r\n\r\n private _exportImage(name: string, mimeType: ImageMimeType, data: ArrayBuffer): number {\r\n const imageData = this._exporter._imageData;\r\n\r\n const baseName = name.replace(/\\.\\/|\\/|\\.\\\\|\\\\/g, \"_\");\r\n const extension = getFileExtensionFromMimeType(mimeType);\r\n let fileName = baseName + extension;\r\n if (fileName in imageData) {\r\n fileName = `${baseName}_${Tools.RandomId()}${extension}`;\r\n }\r\n\r\n imageData[fileName] = {\r\n data: data,\r\n mimeType: mimeType,\r\n };\r\n\r\n const images = this._exporter._images;\r\n images.push({\r\n name: name,\r\n uri: fileName,\r\n });\r\n\r\n return images.length - 1;\r\n }\r\n\r\n private _exportTextureInfo(imageIndex: number, samplerIndex: number, coordinatesIndex?: number): ITextureInfo {\r\n const textures = this._exporter._textures;\r\n let textureIndex = textures.findIndex((t) => t.sampler == samplerIndex && t.source === imageIndex);\r\n if (textureIndex === -1) {\r\n textureIndex = textures.length;\r\n textures.push({\r\n source: imageIndex,\r\n sampler: samplerIndex,\r\n });\r\n }\r\n\r\n const textureInfo: ITextureInfo = { index: textureIndex };\r\n if (coordinatesIndex) {\r\n textureInfo.texCoord = coordinatesIndex;\r\n }\r\n return textureInfo;\r\n }\r\n\r\n private _exportTextureSampler(texture: Nullable): number {\r\n const sampler = this._getTextureSampler(texture);\r\n\r\n // if a pre-existing sampler with identical parameters exists, then reuse the previous sampler\r\n const samplers = this._exporter._samplers;\r\n const samplerIndex = samplers.findIndex(\r\n (s) => s.minFilter === sampler.minFilter && s.magFilter === sampler.magFilter && s.wrapS === sampler.wrapS && s.wrapT === sampler.wrapT\r\n );\r\n if (samplerIndex !== -1) {\r\n return samplerIndex;\r\n }\r\n\r\n samplers.push(sampler);\r\n return samplers.length - 1;\r\n }\r\n}\r\n","import type {\r\n IBufferView,\r\n IAccessor,\r\n INode,\r\n IScene,\r\n IMesh,\r\n IMaterial,\r\n ITexture,\r\n IImage,\r\n ISampler,\r\n IAnimation,\r\n IMeshPrimitive,\r\n IBuffer,\r\n IGLTF,\r\n ITextureInfo,\r\n ISkin,\r\n ICamera,\r\n} from \"babylonjs-gltf2interface\";\r\nimport { AccessorType, ImageMimeType, MeshPrimitiveMode, AccessorComponentType, CameraType } from \"babylonjs-gltf2interface\";\r\n\r\nimport type { FloatArray, IndicesArray, Nullable } from \"core/types\";\r\nimport { Matrix, TmpVectors } from \"core/Maths/math.vector\";\r\nimport { Vector2, Vector3, Vector4, Quaternion } from \"core/Maths/math.vector\";\r\nimport { Color3, Color4 } from \"core/Maths/math.color\";\r\nimport { Tools } from \"core/Misc/tools\";\r\nimport { VertexBuffer } from \"core/Buffers/buffer\";\r\nimport type { Node } from \"core/node\";\r\nimport { TransformNode } from \"core/Meshes/transformNode\";\r\nimport type { AbstractMesh } from \"core/Meshes/abstractMesh\";\r\nimport type { SubMesh } from \"core/Meshes/subMesh\";\r\nimport { Mesh } from \"core/Meshes/mesh\";\r\nimport type { MorphTarget } from \"core/Morph/morphTarget\";\r\nimport { LinesMesh } from \"core/Meshes/linesMesh\";\r\nimport { InstancedMesh } from \"core/Meshes/instancedMesh\";\r\nimport type { Bone } from \"core/Bones/bone\";\r\nimport type { BaseTexture } from \"core/Materials/Textures/baseTexture\";\r\nimport type { Texture } from \"core/Materials/Textures/texture\";\r\nimport { Material } from \"core/Materials/material\";\r\nimport { Engine } from \"core/Engines/engine\";\r\nimport type { Scene } from \"core/scene\";\r\n\r\nimport type { IGLTFExporterExtensionV2 } from \"./glTFExporterExtension\";\r\nimport { _GLTFMaterialExporter } from \"./glTFMaterialExporter\";\r\nimport type { IExportOptions } from \"./glTFSerializer\";\r\nimport { _GLTFUtilities } from \"./glTFUtilities\";\r\nimport { GLTFData } from \"./glTFData\";\r\nimport { _GLTFAnimation } from \"./glTFAnimation\";\r\nimport { Camera } from \"core/Cameras/camera\";\r\nimport { EngineStore } from \"core/Engines/engineStore\";\r\nimport { MultiMaterial } from \"core/Materials/multiMaterial\";\r\n\r\n// Matrix that converts handedness on the X-axis.\r\nconst convertHandednessMatrix = Matrix.Compose(new Vector3(-1, 1, 1), Quaternion.Identity(), Vector3.Zero());\r\n\r\nfunction isNoopNode(node: Node, useRightHandedSystem: boolean): boolean {\r\n if (!(node instanceof TransformNode)) {\r\n return false;\r\n }\r\n\r\n // Transform\r\n if (useRightHandedSystem) {\r\n const matrix = node.getWorldMatrix();\r\n if (!matrix.isIdentity()) {\r\n return false;\r\n }\r\n } else {\r\n const matrix = node.getWorldMatrix().multiplyToRef(convertHandednessMatrix, TmpVectors.Matrix[0]);\r\n if (!matrix.isIdentity()) {\r\n return false;\r\n }\r\n }\r\n\r\n // Geometry\r\n if ((node instanceof Mesh && node.geometry) || (node instanceof InstancedMesh && node.sourceMesh.geometry)) {\r\n return false;\r\n }\r\n\r\n return true;\r\n}\r\n\r\nfunction convertNodeHandedness(node: INode): void {\r\n const translation = Vector3.FromArrayToRef(node.translation || [0, 0, 0], 0, TmpVectors.Vector3[0]);\r\n const rotation = Quaternion.FromArrayToRef(node.rotation || [0, 0, 0, 1], 0, TmpVectors.Quaternion[0]);\r\n const scale = Vector3.FromArrayToRef(node.scale || [1, 1, 1], 0, TmpVectors.Vector3[1]);\r\n const matrix = Matrix.ComposeToRef(scale, rotation, translation, TmpVectors.Matrix[0]).multiplyToRef(convertHandednessMatrix, TmpVectors.Matrix[0]);\r\n\r\n matrix.decompose(scale, rotation, translation);\r\n\r\n if (translation.equalsToFloats(0, 0, 0)) {\r\n delete node.translation;\r\n } else {\r\n node.translation = translation.asArray();\r\n }\r\n\r\n if (Quaternion.IsIdentity(rotation)) {\r\n delete node.rotation;\r\n } else {\r\n node.rotation = rotation.asArray();\r\n }\r\n\r\n if (scale.equalsToFloats(1, 1, 1)) {\r\n delete node.scale;\r\n } else {\r\n node.scale = scale.asArray();\r\n }\r\n}\r\n\r\n/**\r\n * Utility interface for storing vertex attribute data\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\ninterface _IVertexAttributeData {\r\n /**\r\n * Specifies the Babylon Vertex Buffer Type (Position, Normal, Color, etc.)\r\n */\r\n kind: string;\r\n\r\n /**\r\n * Specifies the glTF Accessor Type (VEC2, VEC3, etc.)\r\n */\r\n accessorType: AccessorType;\r\n\r\n /**\r\n * Specifies the glTF Accessor Component Type (BYTE, UNSIGNED_BYTE, FLOAT, SHORT, INT, etc..)\r\n */\r\n accessorComponentType: AccessorComponentType;\r\n\r\n /**\r\n * Specifies the BufferView index for the vertex attribute data\r\n */\r\n bufferViewIndex?: number;\r\n\r\n byteStride?: number;\r\n}\r\n/**\r\n * Converts Babylon Scene into glTF 2.0.\r\n * @internal\r\n */\r\nexport class _Exporter {\r\n /**\r\n * Stores the glTF to export\r\n */\r\n public _glTF: IGLTF;\r\n /**\r\n * Stores all generated buffer views, which represents views into the main glTF buffer data\r\n */\r\n public _bufferViews: IBufferView[];\r\n /**\r\n * Stores all the generated accessors, which is used for accessing the data within the buffer views in glTF\r\n */\r\n public _accessors: IAccessor[];\r\n /**\r\n * Stores all the generated nodes, which contains transform and/or mesh information per node\r\n */\r\n public _nodes: INode[];\r\n /**\r\n * Stores all the generated glTF scenes, which stores multiple node hierarchies\r\n */\r\n private _scenes: IScene[];\r\n /**\r\n * Stores all the generated glTF cameras\r\n */\r\n private _cameras: ICamera[];\r\n /**\r\n * Stores all the generated mesh information, each containing a set of primitives to render in glTF\r\n */\r\n private _meshes: IMesh[];\r\n /**\r\n * Stores all the generated material information, which represents the appearance of each primitive\r\n */\r\n public _materials: IMaterial[];\r\n\r\n public _materialMap: { [materialID: number]: number };\r\n /**\r\n * Stores all the generated texture information, which is referenced by glTF materials\r\n */\r\n public _textures: ITexture[];\r\n /**\r\n * Stores all the generated image information, which is referenced by glTF textures\r\n */\r\n public _images: IImage[];\r\n\r\n /**\r\n * Stores all the texture samplers\r\n */\r\n public _samplers: ISampler[];\r\n /**\r\n * Stores all the generated glTF skins\r\n */\r\n public _skins: ISkin[];\r\n /**\r\n * Stores all the generated animation samplers, which is referenced by glTF animations\r\n */\r\n /**\r\n * Stores the animations for glTF models\r\n */\r\n private _animations: IAnimation[];\r\n /**\r\n * Stores the total amount of bytes stored in the glTF buffer\r\n */\r\n private _totalByteLength: number;\r\n /**\r\n * Stores a reference to the Babylon scene containing the source geometry and material information\r\n */\r\n public _babylonScene: Scene;\r\n /**\r\n * Stores a map of the image data, where the key is the file name and the value\r\n * is the image data\r\n */\r\n public _imageData: { [fileName: string]: { data: ArrayBuffer; mimeType: ImageMimeType } };\r\n\r\n private _orderedImageData: Array<{ data: ArrayBuffer; mimeType: ImageMimeType }>;\r\n\r\n /**\r\n * Stores a map of the unique id of a node to its index in the node array\r\n */\r\n private _nodeMap: { [key: number]: number };\r\n\r\n /**\r\n * Baked animation sample rate\r\n */\r\n private _animationSampleRate: number;\r\n\r\n private _options: IExportOptions;\r\n\r\n private _localEngine: Engine;\r\n\r\n public _glTFMaterialExporter: _GLTFMaterialExporter;\r\n\r\n private _extensions: { [name: string]: IGLTFExporterExtensionV2 } = {};\r\n\r\n private static _ExtensionNames = new Array();\r\n private static _ExtensionFactories: { [name: string]: (exporter: _Exporter) => IGLTFExporterExtensionV2 } = {};\r\n\r\n private _applyExtension(\r\n node: Nullable,\r\n extensions: IGLTFExporterExtensionV2[],\r\n index: number,\r\n actionAsync: (extension: IGLTFExporterExtensionV2, node: Nullable) => Promise> | undefined\r\n ): Promise> {\r\n if (index >= extensions.length) {\r\n return Promise.resolve(node);\r\n }\r\n\r\n const currentPromise = actionAsync(extensions[index], node);\r\n\r\n if (!currentPromise) {\r\n return this._applyExtension(node, extensions, index + 1, actionAsync);\r\n }\r\n\r\n return currentPromise.then((newNode) => this._applyExtension(newNode, extensions, index + 1, actionAsync));\r\n }\r\n\r\n private _applyExtensions(\r\n node: Nullable,\r\n actionAsync: (extension: IGLTFExporterExtensionV2, node: Nullable) => Promise> | undefined\r\n ): Promise> {\r\n const extensions: IGLTFExporterExtensionV2[] = [];\r\n for (const name of _Exporter._ExtensionNames) {\r\n extensions.push(this._extensions[name]);\r\n }\r\n\r\n return this._applyExtension(node, extensions, 0, actionAsync);\r\n }\r\n\r\n public _extensionsPreExportTextureAsync(context: string, babylonTexture: Nullable, mimeType: ImageMimeType): Promise> {\r\n return this._applyExtensions(babylonTexture, (extension, node) => extension.preExportTextureAsync && extension.preExportTextureAsync(context, node, mimeType));\r\n }\r\n\r\n public _extensionsPostExportMeshPrimitiveAsync(\r\n context: string,\r\n meshPrimitive: IMeshPrimitive,\r\n babylonSubMesh: SubMesh,\r\n binaryWriter: _BinaryWriter\r\n ): Promise> {\r\n return this._applyExtensions(\r\n meshPrimitive,\r\n (extension, node) => extension.postExportMeshPrimitiveAsync && extension.postExportMeshPrimitiveAsync(context, node, babylonSubMesh, binaryWriter)\r\n );\r\n }\r\n\r\n public _extensionsPostExportNodeAsync(\r\n context: string,\r\n node: Nullable,\r\n babylonNode: Node,\r\n nodeMap: { [key: number]: number },\r\n binaryWriter: _BinaryWriter\r\n ): Promise> {\r\n return this._applyExtensions(node, (extension, node) => extension.postExportNodeAsync && extension.postExportNodeAsync(context, node, babylonNode, nodeMap, binaryWriter));\r\n }\r\n\r\n public _extensionsPostExportMaterialAsync(context: string, material: Nullable, babylonMaterial: Material): Promise> {\r\n return this._applyExtensions(material, (extension, node) => extension.postExportMaterialAsync && extension.postExportMaterialAsync(context, node, babylonMaterial));\r\n }\r\n\r\n public _extensionsPostExportMaterialAdditionalTextures(context: string, material: IMaterial, babylonMaterial: Material): BaseTexture[] {\r\n const output: BaseTexture[] = [];\r\n\r\n for (const name of _Exporter._ExtensionNames) {\r\n const extension = this._extensions[name];\r\n\r\n if (extension.postExportMaterialAdditionalTextures) {\r\n output.push(...extension.postExportMaterialAdditionalTextures(context, material, babylonMaterial));\r\n }\r\n }\r\n\r\n return output;\r\n }\r\n\r\n public _extensionsPostExportTextures(context: string, textureInfo: ITextureInfo, babylonTexture: BaseTexture): void {\r\n for (const name of _Exporter._ExtensionNames) {\r\n const extension = this._extensions[name];\r\n\r\n if (extension.postExportTexture) {\r\n extension.postExportTexture(context, textureInfo, babylonTexture);\r\n }\r\n }\r\n }\r\n\r\n private _forEachExtensions(action: (extension: IGLTFExporterExtensionV2) => void): void {\r\n for (const name of _Exporter._ExtensionNames) {\r\n const extension = this._extensions[name];\r\n if (extension.enabled) {\r\n action(extension);\r\n }\r\n }\r\n }\r\n\r\n private _extensionsOnExporting(): void {\r\n this._forEachExtensions((extension) => {\r\n if (extension.wasUsed) {\r\n if (this._glTF.extensionsUsed == null) {\r\n this._glTF.extensionsUsed = [];\r\n }\r\n\r\n if (this._glTF.extensionsUsed.indexOf(extension.name) === -1) {\r\n this._glTF.extensionsUsed.push(extension.name);\r\n }\r\n\r\n if (extension.required) {\r\n if (this._glTF.extensionsRequired == null) {\r\n this._glTF.extensionsRequired = [];\r\n }\r\n if (this._glTF.extensionsRequired.indexOf(extension.name) === -1) {\r\n this._glTF.extensionsRequired.push(extension.name);\r\n }\r\n }\r\n\r\n if (this._glTF.extensions == null) {\r\n this._glTF.extensions = {};\r\n }\r\n\r\n if (extension.onExporting) {\r\n extension.onExporting();\r\n }\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Load glTF serializer extensions\r\n */\r\n private _loadExtensions(): void {\r\n for (const name of _Exporter._ExtensionNames) {\r\n const extension = _Exporter._ExtensionFactories[name](this);\r\n this._extensions[name] = extension;\r\n }\r\n }\r\n\r\n /**\r\n * Creates a glTF Exporter instance, which can accept optional exporter options\r\n * @param babylonScene Babylon scene object\r\n * @param options Options to modify the behavior of the exporter\r\n */\r\n public constructor(babylonScene?: Nullable, options?: IExportOptions) {\r\n this._glTF = {\r\n asset: { generator: `Babylon.js v${Engine.Version}`, version: \"2.0\" },\r\n };\r\n babylonScene = babylonScene || EngineStore.LastCreatedScene;\r\n if (!babylonScene) {\r\n return;\r\n }\r\n this._babylonScene = babylonScene;\r\n this._bufferViews = [];\r\n this._accessors = [];\r\n this._meshes = [];\r\n this._scenes = [];\r\n this._cameras = [];\r\n this._nodes = [];\r\n this._images = [];\r\n this._materials = [];\r\n this._materialMap = [];\r\n this._textures = [];\r\n this._samplers = [];\r\n this._skins = [];\r\n this._animations = [];\r\n this._imageData = {};\r\n this._orderedImageData = [];\r\n this._options = options || {};\r\n this._animationSampleRate = this._options.animationSampleRate || 1 / 60;\r\n\r\n this._glTFMaterialExporter = new _GLTFMaterialExporter(this);\r\n this._loadExtensions();\r\n }\r\n\r\n public dispose() {\r\n for (const extensionKey in this._extensions) {\r\n const extension = this._extensions[extensionKey];\r\n\r\n extension.dispose();\r\n }\r\n }\r\n\r\n public get options() {\r\n return this._options;\r\n }\r\n\r\n /**\r\n * Registers a glTF exporter extension\r\n * @param name Name of the extension to export\r\n * @param factory The factory function that creates the exporter extension\r\n */\r\n public static RegisterExtension(name: string, factory: (exporter: _Exporter) => IGLTFExporterExtensionV2): void {\r\n if (_Exporter.UnregisterExtension(name)) {\r\n Tools.Warn(`Extension with the name ${name} already exists`);\r\n }\r\n\r\n _Exporter._ExtensionFactories[name] = factory;\r\n _Exporter._ExtensionNames.push(name);\r\n }\r\n\r\n /**\r\n * Un-registers an exporter extension\r\n * @param name The name fo the exporter extension\r\n * @returns A boolean indicating whether the extension has been un-registered\r\n */\r\n public static UnregisterExtension(name: string): boolean {\r\n if (!_Exporter._ExtensionFactories[name]) {\r\n return false;\r\n }\r\n delete _Exporter._ExtensionFactories[name];\r\n\r\n const index = _Exporter._ExtensionNames.indexOf(name);\r\n if (index !== -1) {\r\n _Exporter._ExtensionNames.splice(index, 1);\r\n }\r\n\r\n return true;\r\n }\r\n\r\n private _reorderIndicesBasedOnPrimitiveMode(submesh: SubMesh, primitiveMode: number, babylonIndices: IndicesArray, byteOffset: number, binaryWriter: _BinaryWriter) {\r\n switch (primitiveMode) {\r\n case Material.TriangleFillMode: {\r\n if (!byteOffset) {\r\n byteOffset = 0;\r\n }\r\n for (let i = submesh.indexStart, length = submesh.indexStart + submesh.indexCount; i < length; i = i + 3) {\r\n const index = byteOffset + i * 4;\r\n // swap the second and third indices\r\n const secondIndex = binaryWriter.getUInt32(index + 4);\r\n const thirdIndex = binaryWriter.getUInt32(index + 8);\r\n binaryWriter.setUInt32(thirdIndex, index + 4);\r\n binaryWriter.setUInt32(secondIndex, index + 8);\r\n }\r\n break;\r\n }\r\n case Material.TriangleFanDrawMode: {\r\n for (let i = submesh.indexStart + submesh.indexCount - 1, start = submesh.indexStart; i >= start; --i) {\r\n binaryWriter.setUInt32(babylonIndices[i], byteOffset);\r\n byteOffset += 4;\r\n }\r\n break;\r\n }\r\n case Material.TriangleStripDrawMode: {\r\n if (submesh.indexCount >= 3) {\r\n binaryWriter.setUInt32(babylonIndices[submesh.indexStart + 2], byteOffset + 4);\r\n binaryWriter.setUInt32(babylonIndices[submesh.indexStart + 1], byteOffset + 8);\r\n }\r\n break;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Reorders the vertex attribute data based on the primitive mode. This is necessary when indices are not available and the winding order is\r\n * clock-wise during export to glTF\r\n * @param submesh BabylonJS submesh\r\n * @param primitiveMode Primitive mode of the mesh\r\n * @param vertexBufferKind The type of vertex attribute\r\n * @param meshAttributeArray The vertex attribute data\r\n * @param byteOffset The offset to the binary data\r\n * @param binaryWriter The binary data for the glTF file\r\n */\r\n private _reorderVertexAttributeDataBasedOnPrimitiveMode(\r\n submesh: SubMesh,\r\n primitiveMode: number,\r\n vertexBufferKind: string,\r\n meshAttributeArray: FloatArray,\r\n byteOffset: number,\r\n binaryWriter: _BinaryWriter\r\n ): void {\r\n switch (primitiveMode) {\r\n case Material.TriangleFillMode: {\r\n this._reorderTriangleFillMode(submesh, vertexBufferKind, meshAttributeArray, byteOffset, binaryWriter);\r\n break;\r\n }\r\n case Material.TriangleStripDrawMode: {\r\n this._reorderTriangleStripDrawMode(submesh, vertexBufferKind, meshAttributeArray, byteOffset, binaryWriter);\r\n break;\r\n }\r\n case Material.TriangleFanDrawMode: {\r\n this._reorderTriangleFanMode(submesh, vertexBufferKind, meshAttributeArray, byteOffset, binaryWriter);\r\n break;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Reorders the vertex attributes in the correct triangle mode order . This is necessary when indices are not available and the winding order is\r\n * clock-wise during export to glTF\r\n * @param submesh BabylonJS submesh\r\n * @param vertexBufferKind The type of vertex attribute\r\n * @param meshAttributeArray The vertex attribute data\r\n * @param byteOffset The offset to the binary data\r\n * @param binaryWriter The binary data for the glTF file\r\n */\r\n private _reorderTriangleFillMode(submesh: SubMesh, vertexBufferKind: string, meshAttributeArray: FloatArray, byteOffset: number, binaryWriter: _BinaryWriter) {\r\n const vertexBuffer = this._getVertexBufferFromMesh(vertexBufferKind, submesh.getMesh() as Mesh);\r\n if (vertexBuffer) {\r\n const stride = vertexBuffer.byteStride / VertexBuffer.GetTypeByteLength(vertexBuffer.type);\r\n if (submesh.verticesCount % 3 !== 0) {\r\n Tools.Error(\"The submesh vertices for the triangle fill mode is not divisible by 3!\");\r\n } else {\r\n const vertexData: Vector2[] | Vector3[] | Vector4[] = [];\r\n let index = 0;\r\n switch (vertexBufferKind) {\r\n case VertexBuffer.PositionKind:\r\n case VertexBuffer.NormalKind: {\r\n for (let x = submesh.verticesStart; x < submesh.verticesStart + submesh.verticesCount; x = x + 3) {\r\n index = x * stride;\r\n (vertexData as Vector3[]).push(Vector3.FromArray(meshAttributeArray, index));\r\n (vertexData as Vector3[]).push(Vector3.FromArray(meshAttributeArray, index + 2 * stride));\r\n (vertexData as Vector3[]).push(Vector3.FromArray(meshAttributeArray, index + stride));\r\n }\r\n break;\r\n }\r\n case VertexBuffer.TangentKind: {\r\n for (let x = submesh.verticesStart; x < submesh.verticesStart + submesh.verticesCount; x = x + 3) {\r\n index = x * stride;\r\n (vertexData as Vector4[]).push(Vector4.FromArray(meshAttributeArray, index));\r\n (vertexData as Vector4[]).push(Vector4.FromArray(meshAttributeArray, index + 2 * stride));\r\n (vertexData as Vector4[]).push(Vector4.FromArray(meshAttributeArray, index + stride));\r\n }\r\n break;\r\n }\r\n case VertexBuffer.ColorKind: {\r\n const size = vertexBuffer.getSize();\r\n for (let x = submesh.verticesStart; x < submesh.verticesStart + submesh.verticesCount; x = x + size) {\r\n index = x * stride;\r\n if (size === 4) {\r\n (vertexData as Vector4[]).push(Vector4.FromArray(meshAttributeArray, index));\r\n (vertexData as Vector4[]).push(Vector4.FromArray(meshAttributeArray, index + 2 * stride));\r\n (vertexData as Vector4[]).push(Vector4.FromArray(meshAttributeArray, index + stride));\r\n } else {\r\n (vertexData as Vector3[]).push(Vector3.FromArray(meshAttributeArray, index));\r\n (vertexData as Vector3[]).push(Vector3.FromArray(meshAttributeArray, index + 2 * stride));\r\n (vertexData as Vector3[]).push(Vector3.FromArray(meshAttributeArray, index + stride));\r\n }\r\n }\r\n break;\r\n }\r\n case VertexBuffer.UVKind:\r\n case VertexBuffer.UV2Kind: {\r\n for (let x = submesh.verticesStart; x < submesh.verticesStart + submesh.verticesCount; x = x + 3) {\r\n index = x * stride;\r\n (vertexData as Vector2[]).push(Vector2.FromArray(meshAttributeArray, index));\r\n (vertexData as Vector2[]).push(Vector2.FromArray(meshAttributeArray, index + 2 * stride));\r\n (vertexData as Vector2[]).push(Vector2.FromArray(meshAttributeArray, index + stride));\r\n }\r\n break;\r\n }\r\n default: {\r\n Tools.Error(`Unsupported Vertex Buffer type: ${vertexBufferKind}`);\r\n }\r\n }\r\n this._writeVertexAttributeData(vertexData, byteOffset, vertexBufferKind, binaryWriter);\r\n }\r\n } else {\r\n Tools.Warn(`reorderTriangleFillMode: Vertex Buffer Kind ${vertexBufferKind} not present!`);\r\n }\r\n }\r\n\r\n /**\r\n * Reorders the vertex attributes in the correct triangle strip order. This is necessary when indices are not available and the winding order is\r\n * clock-wise during export to glTF\r\n * @param submesh BabylonJS submesh\r\n * @param vertexBufferKind The type of vertex attribute\r\n * @param meshAttributeArray The vertex attribute data\r\n * @param byteOffset The offset to the binary data\r\n * @param binaryWriter The binary data for the glTF file\r\n */\r\n private _reorderTriangleStripDrawMode(submesh: SubMesh, vertexBufferKind: string, meshAttributeArray: FloatArray, byteOffset: number, binaryWriter: _BinaryWriter) {\r\n const vertexBuffer = this._getVertexBufferFromMesh(vertexBufferKind, submesh.getMesh() as Mesh);\r\n if (vertexBuffer) {\r\n const stride = vertexBuffer.byteStride / VertexBuffer.GetTypeByteLength(vertexBuffer.type);\r\n\r\n const vertexData: Vector2[] | Vector3[] | Vector4[] = [];\r\n let index = 0;\r\n switch (vertexBufferKind) {\r\n case VertexBuffer.PositionKind:\r\n case VertexBuffer.NormalKind: {\r\n index = submesh.verticesStart;\r\n (vertexData as Vector3[]).push(Vector3.FromArray(meshAttributeArray, index + 2 * stride));\r\n (vertexData as Vector3[]).push(Vector3.FromArray(meshAttributeArray, index + stride));\r\n break;\r\n }\r\n case VertexBuffer.TangentKind: {\r\n for (let x = submesh.verticesStart + submesh.verticesCount - 1; x >= submesh.verticesStart; --x) {\r\n index = x * stride;\r\n (vertexData as Vector4[]).push(Vector4.FromArray(meshAttributeArray, index));\r\n }\r\n break;\r\n }\r\n case VertexBuffer.ColorKind: {\r\n for (let x = submesh.verticesStart + submesh.verticesCount - 1; x >= submesh.verticesStart; --x) {\r\n index = x * stride;\r\n vertexBuffer.getSize() === 4\r\n ? (vertexData as Vector4[]).push(Vector4.FromArray(meshAttributeArray, index))\r\n : (vertexData as Vector3[]).push(Vector3.FromArray(meshAttributeArray, index));\r\n }\r\n break;\r\n }\r\n case VertexBuffer.UVKind:\r\n case VertexBuffer.UV2Kind: {\r\n for (let x = submesh.verticesStart + submesh.verticesCount - 1; x >= submesh.verticesStart; --x) {\r\n index = x * stride;\r\n (vertexData as Vector2[]).push(Vector2.FromArray(meshAttributeArray, index));\r\n }\r\n break;\r\n }\r\n default: {\r\n Tools.Error(`Unsupported Vertex Buffer type: ${vertexBufferKind}`);\r\n }\r\n }\r\n this._writeVertexAttributeData(vertexData, byteOffset + 12, vertexBufferKind, binaryWriter);\r\n } else {\r\n Tools.Warn(`reorderTriangleStripDrawMode: Vertex buffer kind ${vertexBufferKind} not present!`);\r\n }\r\n }\r\n\r\n /**\r\n * Reorders the vertex attributes in the correct triangle fan order. This is necessary when indices are not available and the winding order is\r\n * clock-wise during export to glTF\r\n * @param submesh BabylonJS submesh\r\n * @param vertexBufferKind The type of vertex attribute\r\n * @param meshAttributeArray The vertex attribute data\r\n * @param byteOffset The offset to the binary data\r\n * @param binaryWriter The binary data for the glTF file\r\n */\r\n private _reorderTriangleFanMode(submesh: SubMesh, vertexBufferKind: string, meshAttributeArray: FloatArray, byteOffset: number, binaryWriter: _BinaryWriter) {\r\n const vertexBuffer = this._getVertexBufferFromMesh(vertexBufferKind, submesh.getMesh() as Mesh);\r\n if (vertexBuffer) {\r\n const stride = vertexBuffer.byteStride / VertexBuffer.GetTypeByteLength(vertexBuffer.type);\r\n\r\n const vertexData: Vector2[] | Vector3[] | Vector4[] = [];\r\n let index = 0;\r\n switch (vertexBufferKind) {\r\n case VertexBuffer.PositionKind:\r\n case VertexBuffer.NormalKind: {\r\n for (let x = submesh.verticesStart + submesh.verticesCount - 1; x >= submesh.verticesStart; --x) {\r\n index = x * stride;\r\n (vertexData as Vector3[]).push(Vector3.FromArray(meshAttributeArray, index));\r\n }\r\n break;\r\n }\r\n case VertexBuffer.TangentKind: {\r\n for (let x = submesh.verticesStart + submesh.verticesCount - 1; x >= submesh.verticesStart; --x) {\r\n index = x * stride;\r\n (vertexData as Vector4[]).push(Vector4.FromArray(meshAttributeArray, index));\r\n }\r\n break;\r\n }\r\n case VertexBuffer.ColorKind: {\r\n for (let x = submesh.verticesStart + submesh.verticesCount - 1; x >= submesh.verticesStart; --x) {\r\n index = x * stride;\r\n (vertexData as Vector4[]).push(Vector4.FromArray(meshAttributeArray, index));\r\n vertexBuffer.getSize() === 4\r\n ? (vertexData as Vector4[]).push(Vector4.FromArray(meshAttributeArray, index))\r\n : (vertexData as Vector3[]).push(Vector3.FromArray(meshAttributeArray, index));\r\n }\r\n break;\r\n }\r\n case VertexBuffer.UVKind:\r\n case VertexBuffer.UV2Kind: {\r\n for (let x = submesh.verticesStart + submesh.verticesCount - 1; x >= submesh.verticesStart; --x) {\r\n index = x * stride;\r\n (vertexData as Vector2[]).push(Vector2.FromArray(meshAttributeArray, index));\r\n }\r\n break;\r\n }\r\n default: {\r\n Tools.Error(`Unsupported Vertex Buffer type: ${vertexBufferKind}`);\r\n }\r\n }\r\n this._writeVertexAttributeData(vertexData, byteOffset, vertexBufferKind, binaryWriter);\r\n } else {\r\n Tools.Warn(`reorderTriangleFanMode: Vertex buffer kind ${vertexBufferKind} not present!`);\r\n }\r\n }\r\n\r\n /**\r\n * Writes the vertex attribute data to binary\r\n * @param vertices The vertices to write to the binary writer\r\n * @param byteOffset The offset into the binary writer to overwrite binary data\r\n * @param vertexAttributeKind The vertex attribute type\r\n * @param binaryWriter The writer containing the binary data\r\n */\r\n private _writeVertexAttributeData(vertices: Vector2[] | Vector3[] | Vector4[], byteOffset: number, vertexAttributeKind: string, binaryWriter: _BinaryWriter) {\r\n for (const vertex of vertices) {\r\n if (vertexAttributeKind === VertexBuffer.NormalKind) {\r\n vertex.normalize();\r\n } else if (vertexAttributeKind === VertexBuffer.TangentKind && vertex instanceof Vector4) {\r\n _GLTFUtilities._NormalizeTangentFromRef(vertex);\r\n }\r\n\r\n for (const component of vertex.asArray()) {\r\n binaryWriter.setFloat32(component, byteOffset);\r\n byteOffset += 4;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Writes mesh attribute data to a data buffer\r\n * Returns the bytelength of the data\r\n * @param vertexBufferKind Indicates what kind of vertex data is being passed in\r\n * @param attributeComponentKind\r\n * @param meshAttributeArray Array containing the attribute data\r\n * @param stride Specifies the space between data\r\n * @param binaryWriter The buffer to write the binary data to\r\n * @param babylonTransformNode\r\n */\r\n public _writeAttributeData(\r\n vertexBufferKind: string,\r\n attributeComponentKind: AccessorComponentType,\r\n meshAttributeArray: FloatArray,\r\n stride: number,\r\n binaryWriter: _BinaryWriter,\r\n babylonTransformNode: TransformNode\r\n ) {\r\n let vertexAttributes: number[][] = [];\r\n let index: number;\r\n\r\n switch (vertexBufferKind) {\r\n case VertexBuffer.PositionKind: {\r\n for (let k = 0, length = meshAttributeArray.length / stride; k < length; ++k) {\r\n index = k * stride;\r\n const vertexData = Vector3.FromArray(meshAttributeArray, index);\r\n vertexAttributes.push(vertexData.asArray());\r\n }\r\n break;\r\n }\r\n case VertexBuffer.NormalKind: {\r\n for (let k = 0, length = meshAttributeArray.length / stride; k < length; ++k) {\r\n index = k * stride;\r\n const vertexData = Vector3.FromArray(meshAttributeArray, index);\r\n vertexAttributes.push(vertexData.normalize().asArray());\r\n }\r\n break;\r\n }\r\n case VertexBuffer.TangentKind: {\r\n for (let k = 0, length = meshAttributeArray.length / stride; k < length; ++k) {\r\n index = k * stride;\r\n const vertexData = Vector4.FromArray(meshAttributeArray, index);\r\n _GLTFUtilities._NormalizeTangentFromRef(vertexData);\r\n vertexAttributes.push(vertexData.asArray());\r\n }\r\n break;\r\n }\r\n case VertexBuffer.ColorKind: {\r\n const meshMaterial = (babylonTransformNode as Mesh).material;\r\n const convertToLinear = meshMaterial ? meshMaterial.getClassName() === \"StandardMaterial\" : true;\r\n const vertexData: Color3 | Color4 = stride === 3 ? new Color3() : new Color4();\r\n const useExactSrgbConversions = this._babylonScene.getEngine().useExactSrgbConversions;\r\n for (let k = 0, length = meshAttributeArray.length / stride; k < length; ++k) {\r\n index = k * stride;\r\n if (stride === 3) {\r\n Color3.FromArrayToRef(meshAttributeArray, index, vertexData as Color3);\r\n if (convertToLinear) {\r\n (vertexData as Color3).toLinearSpaceToRef(vertexData as Color3, useExactSrgbConversions);\r\n }\r\n } else {\r\n Color4.FromArrayToRef(meshAttributeArray, index, vertexData as Color4);\r\n if (convertToLinear) {\r\n (vertexData as Color4).toLinearSpaceToRef(vertexData as Color4, useExactSrgbConversions);\r\n }\r\n }\r\n vertexAttributes.push(vertexData.asArray());\r\n }\r\n break;\r\n }\r\n case VertexBuffer.UVKind:\r\n case VertexBuffer.UV2Kind: {\r\n for (let k = 0, length = meshAttributeArray.length / stride; k < length; ++k) {\r\n index = k * stride;\r\n const vertexData = Vector2.FromArray(meshAttributeArray, index);\r\n vertexAttributes.push(vertexData.asArray());\r\n }\r\n break;\r\n }\r\n case VertexBuffer.MatricesIndicesKind:\r\n case VertexBuffer.MatricesIndicesExtraKind: {\r\n for (let k = 0, length = meshAttributeArray.length / stride; k < length; ++k) {\r\n index = k * stride;\r\n const vertexData = Vector4.FromArray(meshAttributeArray, index);\r\n vertexAttributes.push(vertexData.asArray());\r\n }\r\n break;\r\n }\r\n case VertexBuffer.MatricesWeightsKind:\r\n case VertexBuffer.MatricesWeightsExtraKind: {\r\n for (let k = 0, length = meshAttributeArray.length / stride; k < length; ++k) {\r\n index = k * stride;\r\n const vertexData = Vector4.FromArray(meshAttributeArray, index);\r\n vertexAttributes.push(vertexData.asArray());\r\n }\r\n break;\r\n }\r\n default: {\r\n Tools.Warn(\"Unsupported Vertex Buffer Type: \" + vertexBufferKind);\r\n vertexAttributes = [];\r\n }\r\n }\r\n\r\n let writeBinaryFunc;\r\n switch (attributeComponentKind) {\r\n case AccessorComponentType.UNSIGNED_BYTE: {\r\n writeBinaryFunc = binaryWriter.setUInt8.bind(binaryWriter);\r\n break;\r\n }\r\n case AccessorComponentType.UNSIGNED_SHORT: {\r\n writeBinaryFunc = binaryWriter.setUInt16.bind(binaryWriter);\r\n break;\r\n }\r\n case AccessorComponentType.UNSIGNED_INT: {\r\n writeBinaryFunc = binaryWriter.setUInt32.bind(binaryWriter);\r\n break;\r\n }\r\n case AccessorComponentType.FLOAT: {\r\n writeBinaryFunc = binaryWriter.setFloat32.bind(binaryWriter);\r\n break;\r\n }\r\n default: {\r\n Tools.Warn(\"Unsupported Attribute Component kind: \" + attributeComponentKind);\r\n return;\r\n }\r\n }\r\n\r\n for (const vertexAttribute of vertexAttributes) {\r\n for (const component of vertexAttribute) {\r\n writeBinaryFunc(component);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Writes mesh attribute data to a data buffer\r\n * Returns the bytelength of the data\r\n * @param vertexBufferKind Indicates what kind of vertex data is being passed in\r\n * @param attributeComponentKind\r\n * @param meshPrimitive\r\n * @param morphTarget\r\n * @param meshAttributeArray Array containing the attribute data\r\n * @param morphTargetAttributeArray\r\n * @param stride Specifies the space between data\r\n * @param binaryWriter The buffer to write the binary data to\r\n * @param minMax\r\n */\r\n public writeMorphTargetAttributeData(\r\n vertexBufferKind: string,\r\n attributeComponentKind: AccessorComponentType,\r\n meshPrimitive: SubMesh,\r\n meshAttributeArray: FloatArray,\r\n morphTargetAttributeArray: FloatArray,\r\n stride: number,\r\n binaryWriter: _BinaryWriter,\r\n minMax?: any\r\n ) {\r\n let vertexAttributes: number[][] = [];\r\n let index: number;\r\n let difference: Vector3 = new Vector3();\r\n let difference4: Vector4 = new Vector4(0, 0, 0, 0);\r\n\r\n switch (vertexBufferKind) {\r\n case VertexBuffer.PositionKind: {\r\n for (let k = meshPrimitive.verticesStart; k < meshPrimitive.verticesCount; ++k) {\r\n index = meshPrimitive.indexStart + k * stride;\r\n const vertexData = Vector3.FromArray(meshAttributeArray, index);\r\n const morphData = Vector3.FromArray(morphTargetAttributeArray, index);\r\n difference = morphData.subtractToRef(vertexData, difference);\r\n if (minMax) {\r\n minMax.min.copyFromFloats(Math.min(difference.x, minMax.min.x), Math.min(difference.y, minMax.min.y), Math.min(difference.z, minMax.min.z));\r\n minMax.max.copyFromFloats(Math.max(difference.x, minMax.max.x), Math.max(difference.y, minMax.max.y), Math.max(difference.z, minMax.max.z));\r\n }\r\n vertexAttributes.push(difference.asArray());\r\n }\r\n break;\r\n }\r\n case VertexBuffer.NormalKind: {\r\n for (let k = meshPrimitive.verticesStart; k < meshPrimitive.verticesCount; ++k) {\r\n index = meshPrimitive.indexStart + k * stride;\r\n const vertexData = Vector3.FromArray(meshAttributeArray, index).normalize();\r\n const morphData = Vector3.FromArray(morphTargetAttributeArray, index).normalize();\r\n difference = morphData.subtractToRef(vertexData, difference);\r\n vertexAttributes.push(difference.asArray());\r\n }\r\n break;\r\n }\r\n case VertexBuffer.TangentKind: {\r\n for (let k = meshPrimitive.verticesStart; k < meshPrimitive.verticesCount; ++k) {\r\n index = meshPrimitive.indexStart + k * (stride + 1);\r\n const vertexData = Vector4.FromArray(meshAttributeArray, index);\r\n _GLTFUtilities._NormalizeTangentFromRef(vertexData);\r\n const morphData = Vector4.FromArray(morphTargetAttributeArray, index);\r\n _GLTFUtilities._NormalizeTangentFromRef(morphData);\r\n difference4 = morphData.subtractToRef(vertexData, difference4);\r\n vertexAttributes.push([difference4.x, difference4.y, difference4.z]);\r\n }\r\n break;\r\n }\r\n default: {\r\n Tools.Warn(\"Unsupported Vertex Buffer Type: \" + vertexBufferKind);\r\n vertexAttributes = [];\r\n }\r\n }\r\n\r\n let writeBinaryFunc;\r\n switch (attributeComponentKind) {\r\n case AccessorComponentType.UNSIGNED_BYTE: {\r\n writeBinaryFunc = binaryWriter.setUInt8.bind(binaryWriter);\r\n break;\r\n }\r\n case AccessorComponentType.UNSIGNED_SHORT: {\r\n writeBinaryFunc = binaryWriter.setUInt16.bind(binaryWriter);\r\n break;\r\n }\r\n case AccessorComponentType.UNSIGNED_INT: {\r\n writeBinaryFunc = binaryWriter.setUInt32.bind(binaryWriter);\r\n break;\r\n }\r\n case AccessorComponentType.FLOAT: {\r\n writeBinaryFunc = binaryWriter.setFloat32.bind(binaryWriter);\r\n break;\r\n }\r\n default: {\r\n Tools.Warn(\"Unsupported Attribute Component kind: \" + attributeComponentKind);\r\n return;\r\n }\r\n }\r\n\r\n for (const vertexAttribute of vertexAttributes) {\r\n for (const component of vertexAttribute) {\r\n writeBinaryFunc(component);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Generates glTF json data\r\n * @param shouldUseGlb Indicates whether the json should be written for a glb file\r\n * @param glTFPrefix Text to use when prefixing a glTF file\r\n * @param prettyPrint Indicates whether the json file should be pretty printed (true) or not (false)\r\n * @returns json data as string\r\n */\r\n private _generateJSON(shouldUseGlb: boolean, glTFPrefix?: string, prettyPrint?: boolean): string {\r\n const buffer: IBuffer = { byteLength: this._totalByteLength };\r\n let imageName: string;\r\n let imageData: { data: ArrayBuffer; mimeType: ImageMimeType };\r\n let bufferView: IBufferView;\r\n let byteOffset: number = this._totalByteLength;\r\n\r\n if (buffer.byteLength) {\r\n this._glTF.buffers = [buffer];\r\n }\r\n if (this._nodes && this._nodes.length) {\r\n this._glTF.nodes = this._nodes;\r\n }\r\n if (this._meshes && this._meshes.length) {\r\n this._glTF.meshes = this._meshes;\r\n }\r\n if (this._scenes && this._scenes.length) {\r\n this._glTF.scenes = this._scenes;\r\n this._glTF.scene = 0;\r\n }\r\n if (this._cameras && this._cameras.length) {\r\n this._glTF.cameras = this._cameras;\r\n }\r\n if (this._bufferViews && this._bufferViews.length) {\r\n this._glTF.bufferViews = this._bufferViews;\r\n }\r\n if (this._accessors && this._accessors.length) {\r\n this._glTF.accessors = this._accessors;\r\n }\r\n if (this._animations && this._animations.length) {\r\n this._glTF.animations = this._animations;\r\n }\r\n if (this._materials && this._materials.length) {\r\n this._glTF.materials = this._materials;\r\n }\r\n if (this._textures && this._textures.length) {\r\n this._glTF.textures = this._textures;\r\n }\r\n if (this._samplers && this._samplers.length) {\r\n this._glTF.samplers = this._samplers;\r\n }\r\n if (this._skins && this._skins.length) {\r\n this._glTF.skins = this._skins;\r\n }\r\n if (this._images && this._images.length) {\r\n if (!shouldUseGlb) {\r\n this._glTF.images = this._images;\r\n } else {\r\n this._glTF.images = [];\r\n\r\n this._images.forEach((image) => {\r\n if (image.uri) {\r\n imageData = this._imageData[image.uri];\r\n this._orderedImageData.push(imageData);\r\n imageName = image.uri.split(\".\")[0] + \" image\";\r\n bufferView = _GLTFUtilities._CreateBufferView(0, byteOffset, imageData.data.byteLength, undefined, imageName);\r\n byteOffset += imageData.data.byteLength;\r\n this._bufferViews.push(bufferView);\r\n image.bufferView = this._bufferViews.length - 1;\r\n image.name = imageName;\r\n image.mimeType = imageData.mimeType;\r\n image.uri = undefined;\r\n if (!this._glTF.images) {\r\n this._glTF.images = [];\r\n }\r\n this._glTF.images.push(image);\r\n }\r\n });\r\n // Replace uri with bufferview and mime type for glb\r\n buffer.byteLength = byteOffset;\r\n }\r\n }\r\n\r\n if (!shouldUseGlb) {\r\n buffer.uri = glTFPrefix + \".bin\";\r\n }\r\n\r\n const jsonText = prettyPrint ? JSON.stringify(this._glTF, null, 2) : JSON.stringify(this._glTF);\r\n\r\n return jsonText;\r\n }\r\n\r\n /**\r\n * Generates data for .gltf and .bin files based on the glTF prefix string\r\n * @param glTFPrefix Text to use when prefixing a glTF file\r\n * @param dispose Dispose the exporter\r\n * @returns GLTFData with glTF file data\r\n */\r\n public _generateGLTFAsync(glTFPrefix: string, dispose = true): Promise {\r\n return this._generateBinaryAsync().then((binaryBuffer) => {\r\n this._extensionsOnExporting();\r\n const jsonText = this._generateJSON(false, glTFPrefix, true);\r\n const bin = new Blob([binaryBuffer], { type: \"application/octet-stream\" });\r\n\r\n const glTFFileName = glTFPrefix + \".gltf\";\r\n const glTFBinFile = glTFPrefix + \".bin\";\r\n\r\n const container = new GLTFData();\r\n\r\n container.glTFFiles[glTFFileName] = jsonText;\r\n container.glTFFiles[glTFBinFile] = bin;\r\n\r\n if (this._imageData) {\r\n for (const image in this._imageData) {\r\n container.glTFFiles[image] = new Blob([this._imageData[image].data], { type: this._imageData[image].mimeType });\r\n }\r\n }\r\n\r\n if (dispose) {\r\n this.dispose();\r\n }\r\n\r\n return container;\r\n });\r\n }\r\n\r\n /**\r\n * Creates a binary buffer for glTF\r\n * @returns array buffer for binary data\r\n */\r\n private _generateBinaryAsync(): Promise {\r\n const binaryWriter = new _BinaryWriter(4);\r\n return this._createSceneAsync(binaryWriter).then(() => {\r\n if (this._localEngine) {\r\n this._localEngine.dispose();\r\n }\r\n return binaryWriter.getArrayBuffer();\r\n });\r\n }\r\n\r\n /**\r\n * Pads the number to a multiple of 4\r\n * @param num number to pad\r\n * @returns padded number\r\n */\r\n private _getPadding(num: number): number {\r\n const remainder = num % 4;\r\n const padding = remainder === 0 ? remainder : 4 - remainder;\r\n\r\n return padding;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _generateGLBAsync(glTFPrefix: string, dispose = true): Promise {\r\n return this._generateBinaryAsync().then((binaryBuffer) => {\r\n this._extensionsOnExporting();\r\n const jsonText = this._generateJSON(true);\r\n const glbFileName = glTFPrefix + \".glb\";\r\n const headerLength = 12;\r\n const chunkLengthPrefix = 8;\r\n let jsonLength = jsonText.length;\r\n let encodedJsonText;\r\n let imageByteLength = 0;\r\n // make use of TextEncoder when available\r\n if (typeof TextEncoder !== \"undefined\") {\r\n const encoder = new TextEncoder();\r\n encodedJsonText = encoder.encode(jsonText);\r\n jsonLength = encodedJsonText.length;\r\n }\r\n for (let i = 0; i < this._orderedImageData.length; ++i) {\r\n imageByteLength += this._orderedImageData[i].data.byteLength;\r\n }\r\n const jsonPadding = this._getPadding(jsonLength);\r\n const binPadding = this._getPadding(binaryBuffer.byteLength);\r\n const imagePadding = this._getPadding(imageByteLength);\r\n\r\n const byteLength = headerLength + 2 * chunkLengthPrefix + jsonLength + jsonPadding + binaryBuffer.byteLength + binPadding + imageByteLength + imagePadding;\r\n\r\n //header\r\n const headerBuffer = new ArrayBuffer(headerLength);\r\n const headerBufferView = new DataView(headerBuffer);\r\n headerBufferView.setUint32(0, 0x46546c67, true); //glTF\r\n headerBufferView.setUint32(4, 2, true); // version\r\n headerBufferView.setUint32(8, byteLength, true); // total bytes in file\r\n\r\n //json chunk\r\n const jsonChunkBuffer = new ArrayBuffer(chunkLengthPrefix + jsonLength + jsonPadding);\r\n const jsonChunkBufferView = new DataView(jsonChunkBuffer);\r\n jsonChunkBufferView.setUint32(0, jsonLength + jsonPadding, true);\r\n jsonChunkBufferView.setUint32(4, 0x4e4f534a, true);\r\n\r\n //json chunk bytes\r\n const jsonData = new Uint8Array(jsonChunkBuffer, chunkLengthPrefix);\r\n // if TextEncoder was available, we can simply copy the encoded array\r\n if (encodedJsonText) {\r\n jsonData.set(encodedJsonText);\r\n } else {\r\n const blankCharCode = \"_\".charCodeAt(0);\r\n for (let i = 0; i < jsonLength; ++i) {\r\n const charCode = jsonText.charCodeAt(i);\r\n // if the character doesn't fit into a single UTF-16 code unit, just put a blank character\r\n if (charCode != jsonText.codePointAt(i)) {\r\n jsonData[i] = blankCharCode;\r\n } else {\r\n jsonData[i] = charCode;\r\n }\r\n }\r\n }\r\n\r\n //json padding\r\n const jsonPaddingView = new Uint8Array(jsonChunkBuffer, chunkLengthPrefix + jsonLength);\r\n for (let i = 0; i < jsonPadding; ++i) {\r\n jsonPaddingView[i] = 0x20;\r\n }\r\n\r\n //binary chunk\r\n const binaryChunkBuffer = new ArrayBuffer(chunkLengthPrefix);\r\n const binaryChunkBufferView = new DataView(binaryChunkBuffer);\r\n binaryChunkBufferView.setUint32(0, binaryBuffer.byteLength + imageByteLength + imagePadding, true);\r\n binaryChunkBufferView.setUint32(4, 0x004e4942, true);\r\n\r\n // binary padding\r\n const binPaddingBuffer = new ArrayBuffer(binPadding);\r\n const binPaddingView = new Uint8Array(binPaddingBuffer);\r\n for (let i = 0; i < binPadding; ++i) {\r\n binPaddingView[i] = 0;\r\n }\r\n\r\n const imagePaddingBuffer = new ArrayBuffer(imagePadding);\r\n const imagePaddingView = new Uint8Array(imagePaddingBuffer);\r\n for (let i = 0; i < imagePadding; ++i) {\r\n imagePaddingView[i] = 0;\r\n }\r\n\r\n const glbData = [headerBuffer, jsonChunkBuffer, binaryChunkBuffer, binaryBuffer];\r\n\r\n // binary data\r\n for (let i = 0; i < this._orderedImageData.length; ++i) {\r\n glbData.push(this._orderedImageData[i].data);\r\n }\r\n\r\n glbData.push(binPaddingBuffer);\r\n\r\n glbData.push(imagePaddingBuffer);\r\n\r\n const glbFile = new Blob(glbData, { type: \"application/octet-stream\" });\r\n\r\n const container = new GLTFData();\r\n container.glTFFiles[glbFileName] = glbFile;\r\n\r\n if (this._localEngine != null) {\r\n this._localEngine.dispose();\r\n }\r\n\r\n if (dispose) {\r\n this.dispose();\r\n }\r\n\r\n return container;\r\n });\r\n }\r\n\r\n /**\r\n * Sets the TRS for each node\r\n * @param node glTF Node for storing the transformation data\r\n * @param babylonTransformNode Babylon mesh used as the source for the transformation data\r\n */\r\n private _setNodeTransformation(node: INode, babylonTransformNode: TransformNode): void {\r\n if (!babylonTransformNode.getPivotPoint().equalsToFloats(0, 0, 0)) {\r\n Tools.Warn(\"Pivot points are not supported in the glTF serializer\");\r\n }\r\n if (!babylonTransformNode.position.equalsToFloats(0, 0, 0)) {\r\n node.translation = babylonTransformNode.position.asArray();\r\n }\r\n\r\n if (!babylonTransformNode.scaling.equalsToFloats(1, 1, 1)) {\r\n node.scale = babylonTransformNode.scaling.asArray();\r\n }\r\n\r\n const rotationQuaternion = Quaternion.RotationYawPitchRoll(babylonTransformNode.rotation.y, babylonTransformNode.rotation.x, babylonTransformNode.rotation.z);\r\n if (babylonTransformNode.rotationQuaternion) {\r\n rotationQuaternion.multiplyInPlace(babylonTransformNode.rotationQuaternion);\r\n }\r\n if (!Quaternion.IsIdentity(rotationQuaternion)) {\r\n node.rotation = rotationQuaternion.normalize().asArray();\r\n }\r\n }\r\n\r\n private _setCameraTransformation(node: INode, babylonCamera: Camera): void {\r\n if (!babylonCamera.position.equalsToFloats(0, 0, 0)) {\r\n node.translation = babylonCamera.position.asArray();\r\n }\r\n\r\n const rotationQuaternion = (babylonCamera).rotationQuaternion; // we target the local transformation if one.\r\n\r\n if (rotationQuaternion && !Quaternion.IsIdentity(rotationQuaternion)) {\r\n node.rotation = rotationQuaternion.normalize().asArray();\r\n }\r\n }\r\n\r\n private _getVertexBufferFromMesh(attributeKind: string, bufferMesh: Mesh): Nullable {\r\n if (bufferMesh.isVerticesDataPresent(attributeKind, true)) {\r\n const vertexBuffer = bufferMesh.getVertexBuffer(attributeKind, true);\r\n if (vertexBuffer) {\r\n return vertexBuffer;\r\n }\r\n }\r\n return null;\r\n }\r\n\r\n /**\r\n * Creates a bufferview based on the vertices type for the Babylon mesh\r\n * @param kind Indicates the type of vertices data\r\n * @param attributeComponentKind Indicates the numerical type used to store the data\r\n * @param babylonTransformNode The Babylon mesh to get the vertices data from\r\n * @param binaryWriter The buffer to write the bufferview data to\r\n * @param byteStride\r\n */\r\n private _createBufferViewKind(\r\n kind: string,\r\n attributeComponentKind: AccessorComponentType,\r\n babylonTransformNode: TransformNode,\r\n binaryWriter: _BinaryWriter,\r\n byteStride: number\r\n ) {\r\n const bufferMesh =\r\n babylonTransformNode instanceof Mesh\r\n ? (babylonTransformNode as Mesh)\r\n : babylonTransformNode instanceof InstancedMesh\r\n ? (babylonTransformNode as InstancedMesh).sourceMesh\r\n : null;\r\n\r\n if (bufferMesh) {\r\n const vertexBuffer = bufferMesh.getVertexBuffer(kind, true);\r\n const vertexData = bufferMesh.getVerticesData(kind, undefined, undefined, true);\r\n\r\n if (vertexBuffer && vertexData) {\r\n const typeByteLength = VertexBuffer.GetTypeByteLength(attributeComponentKind);\r\n const byteLength = vertexData.length * typeByteLength;\r\n const bufferView = _GLTFUtilities._CreateBufferView(0, binaryWriter.getByteOffset(), byteLength, byteStride, kind + \" - \" + bufferMesh.name);\r\n this._bufferViews.push(bufferView);\r\n\r\n this._writeAttributeData(kind, attributeComponentKind, vertexData, byteStride / typeByteLength, binaryWriter, babylonTransformNode);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Creates a bufferview based on the vertices type for the Babylon mesh\r\n * @param babylonSubMesh The Babylon submesh that the morph target is applied to\r\n * @param meshPrimitive\r\n * @param babylonMorphTarget the morph target to be exported\r\n * @param binaryWriter The buffer to write the bufferview data to\r\n */\r\n private _setMorphTargetAttributes(babylonSubMesh: SubMesh, meshPrimitive: IMeshPrimitive, babylonMorphTarget: MorphTarget, binaryWriter: _BinaryWriter) {\r\n if (babylonMorphTarget) {\r\n if (!meshPrimitive.targets) {\r\n meshPrimitive.targets = [];\r\n }\r\n const target: { [attribute: string]: number } = {};\r\n const mesh = babylonSubMesh.getMesh() as Mesh;\r\n if (babylonMorphTarget.hasNormals) {\r\n const vertexNormals = mesh.getVerticesData(VertexBuffer.NormalKind, undefined, undefined, true)!;\r\n const morphNormals = babylonMorphTarget.getNormals()!;\r\n const count = babylonSubMesh.verticesCount;\r\n const byteStride = 12; // 3 x 4 byte floats\r\n const byteLength = count * byteStride;\r\n const bufferView = _GLTFUtilities._CreateBufferView(0, binaryWriter.getByteOffset(), byteLength, byteStride, babylonMorphTarget.name + \"_NORMAL\");\r\n this._bufferViews.push(bufferView);\r\n\r\n const bufferViewIndex = this._bufferViews.length - 1;\r\n const accessor = _GLTFUtilities._CreateAccessor(\r\n bufferViewIndex,\r\n babylonMorphTarget.name + \" - \" + \"NORMAL\",\r\n AccessorType.VEC3,\r\n AccessorComponentType.FLOAT,\r\n count,\r\n 0,\r\n null,\r\n null\r\n );\r\n this._accessors.push(accessor);\r\n target.NORMAL = this._accessors.length - 1;\r\n\r\n this.writeMorphTargetAttributeData(VertexBuffer.NormalKind, AccessorComponentType.FLOAT, babylonSubMesh, vertexNormals, morphNormals, byteStride / 4, binaryWriter);\r\n }\r\n if (babylonMorphTarget.hasPositions) {\r\n const vertexPositions = mesh.getVerticesData(VertexBuffer.PositionKind, undefined, undefined, true)!;\r\n const morphPositions = babylonMorphTarget.getPositions()!;\r\n const count = babylonSubMesh.verticesCount;\r\n const byteStride = 12; // 3 x 4 byte floats\r\n const byteLength = count * byteStride;\r\n const bufferView = _GLTFUtilities._CreateBufferView(0, binaryWriter.getByteOffset(), byteLength, byteStride, babylonMorphTarget.name + \"_POSITION\");\r\n this._bufferViews.push(bufferView);\r\n\r\n const bufferViewIndex = this._bufferViews.length - 1;\r\n const minMax = { min: new Vector3(Infinity, Infinity, Infinity), max: new Vector3(-Infinity, -Infinity, -Infinity) };\r\n const accessor = _GLTFUtilities._CreateAccessor(\r\n bufferViewIndex,\r\n babylonMorphTarget.name + \" - \" + \"POSITION\",\r\n AccessorType.VEC3,\r\n AccessorComponentType.FLOAT,\r\n count,\r\n 0,\r\n null,\r\n null\r\n );\r\n this._accessors.push(accessor);\r\n target.POSITION = this._accessors.length - 1;\r\n\r\n this.writeMorphTargetAttributeData(\r\n VertexBuffer.PositionKind,\r\n AccessorComponentType.FLOAT,\r\n babylonSubMesh,\r\n vertexPositions,\r\n morphPositions,\r\n byteStride / 4,\r\n binaryWriter,\r\n minMax\r\n );\r\n accessor.min = minMax.min!.asArray();\r\n accessor.max = minMax.max!.asArray();\r\n }\r\n if (babylonMorphTarget.hasTangents) {\r\n const vertexTangents = mesh.getVerticesData(VertexBuffer.TangentKind, undefined, undefined, true)!;\r\n const morphTangents = babylonMorphTarget.getTangents()!;\r\n const count = babylonSubMesh.verticesCount;\r\n const byteStride = 12; // 3 x 4 byte floats\r\n const byteLength = count * byteStride;\r\n const bufferView = _GLTFUtilities._CreateBufferView(0, binaryWriter.getByteOffset(), byteLength, byteStride, babylonMorphTarget.name + \"_NORMAL\");\r\n this._bufferViews.push(bufferView);\r\n\r\n const bufferViewIndex = this._bufferViews.length - 1;\r\n const accessor = _GLTFUtilities._CreateAccessor(\r\n bufferViewIndex,\r\n babylonMorphTarget.name + \" - \" + \"TANGENT\",\r\n AccessorType.VEC3,\r\n AccessorComponentType.FLOAT,\r\n count,\r\n 0,\r\n null,\r\n null\r\n );\r\n this._accessors.push(accessor);\r\n target.TANGENT = this._accessors.length - 1;\r\n\r\n this.writeMorphTargetAttributeData(\r\n VertexBuffer.TangentKind,\r\n AccessorComponentType.FLOAT,\r\n babylonSubMesh,\r\n vertexTangents,\r\n morphTangents,\r\n byteStride / 4,\r\n binaryWriter\r\n );\r\n }\r\n meshPrimitive.targets.push(target);\r\n }\r\n }\r\n\r\n /**\r\n * The primitive mode of the Babylon mesh\r\n * @param babylonMesh The BabylonJS mesh\r\n */\r\n private _getMeshPrimitiveMode(babylonMesh: AbstractMesh): number {\r\n if (babylonMesh instanceof LinesMesh) {\r\n return Material.LineListDrawMode;\r\n }\r\n if (babylonMesh instanceof InstancedMesh || babylonMesh instanceof Mesh) {\r\n const baseMesh = babylonMesh instanceof Mesh ? babylonMesh : babylonMesh.sourceMesh;\r\n if (typeof baseMesh.overrideRenderingFillMode === \"number\") {\r\n return baseMesh.overrideRenderingFillMode;\r\n }\r\n }\r\n return babylonMesh.material ? babylonMesh.material.fillMode : Material.TriangleFillMode;\r\n }\r\n\r\n /**\r\n * Sets the primitive mode of the glTF mesh primitive\r\n * @param meshPrimitive glTF mesh primitive\r\n * @param primitiveMode The primitive mode\r\n */\r\n private _setPrimitiveMode(meshPrimitive: IMeshPrimitive, primitiveMode: number) {\r\n switch (primitiveMode) {\r\n case Material.TriangleFillMode: {\r\n // glTF defaults to using Triangle Mode\r\n break;\r\n }\r\n case Material.TriangleStripDrawMode: {\r\n meshPrimitive.mode = MeshPrimitiveMode.TRIANGLE_STRIP;\r\n break;\r\n }\r\n case Material.TriangleFanDrawMode: {\r\n meshPrimitive.mode = MeshPrimitiveMode.TRIANGLE_FAN;\r\n break;\r\n }\r\n case Material.PointListDrawMode: {\r\n meshPrimitive.mode = MeshPrimitiveMode.POINTS;\r\n break;\r\n }\r\n case Material.PointFillMode: {\r\n meshPrimitive.mode = MeshPrimitiveMode.POINTS;\r\n break;\r\n }\r\n case Material.LineLoopDrawMode: {\r\n meshPrimitive.mode = MeshPrimitiveMode.LINE_LOOP;\r\n break;\r\n }\r\n case Material.LineListDrawMode: {\r\n meshPrimitive.mode = MeshPrimitiveMode.LINES;\r\n break;\r\n }\r\n case Material.LineStripDrawMode: {\r\n meshPrimitive.mode = MeshPrimitiveMode.LINE_STRIP;\r\n break;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Sets the vertex attribute accessor based of the glTF mesh primitive\r\n * @param meshPrimitive glTF mesh primitive\r\n * @param attributeKind vertex attribute\r\n * @returns boolean specifying if uv coordinates are present\r\n */\r\n private _setAttributeKind(meshPrimitive: IMeshPrimitive, attributeKind: string): void {\r\n switch (attributeKind) {\r\n case VertexBuffer.PositionKind: {\r\n meshPrimitive.attributes.POSITION = this._accessors.length - 1;\r\n break;\r\n }\r\n case VertexBuffer.NormalKind: {\r\n meshPrimitive.attributes.NORMAL = this._accessors.length - 1;\r\n break;\r\n }\r\n case VertexBuffer.ColorKind: {\r\n meshPrimitive.attributes.COLOR_0 = this._accessors.length - 1;\r\n break;\r\n }\r\n case VertexBuffer.TangentKind: {\r\n meshPrimitive.attributes.TANGENT = this._accessors.length - 1;\r\n break;\r\n }\r\n case VertexBuffer.UVKind: {\r\n meshPrimitive.attributes.TEXCOORD_0 = this._accessors.length - 1;\r\n break;\r\n }\r\n case VertexBuffer.UV2Kind: {\r\n meshPrimitive.attributes.TEXCOORD_1 = this._accessors.length - 1;\r\n break;\r\n }\r\n case VertexBuffer.MatricesIndicesKind: {\r\n meshPrimitive.attributes.JOINTS_0 = this._accessors.length - 1;\r\n break;\r\n }\r\n case VertexBuffer.MatricesIndicesExtraKind: {\r\n meshPrimitive.attributes.JOINTS_1 = this._accessors.length - 1;\r\n break;\r\n }\r\n case VertexBuffer.MatricesWeightsKind: {\r\n meshPrimitive.attributes.WEIGHTS_0 = this._accessors.length - 1;\r\n break;\r\n }\r\n case VertexBuffer.MatricesWeightsExtraKind: {\r\n meshPrimitive.attributes.WEIGHTS_1 = this._accessors.length - 1;\r\n break;\r\n }\r\n default: {\r\n Tools.Warn(\"Unsupported Vertex Buffer Type: \" + attributeKind);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Sets data for the primitive attributes of each submesh\r\n * @param mesh glTF Mesh object to store the primitive attribute information\r\n * @param babylonTransformNode Babylon mesh to get the primitive attribute data from\r\n * @param binaryWriter Buffer to write the attribute data to\r\n */\r\n private _setPrimitiveAttributesAsync(mesh: IMesh, babylonTransformNode: TransformNode, binaryWriter: _BinaryWriter): Promise {\r\n const promises: Promise[] = [];\r\n let bufferMesh: Nullable = null;\r\n let bufferView: IBufferView;\r\n let minMax: { min: Nullable; max: Nullable };\r\n\r\n if (babylonTransformNode instanceof Mesh) {\r\n bufferMesh = babylonTransformNode as Mesh;\r\n } else if (babylonTransformNode instanceof InstancedMesh) {\r\n bufferMesh = (babylonTransformNode as InstancedMesh).sourceMesh;\r\n }\r\n const attributeData: _IVertexAttributeData[] = [\r\n { kind: VertexBuffer.PositionKind, accessorType: AccessorType.VEC3, accessorComponentType: AccessorComponentType.FLOAT, byteStride: 12 },\r\n { kind: VertexBuffer.NormalKind, accessorType: AccessorType.VEC3, accessorComponentType: AccessorComponentType.FLOAT, byteStride: 12 },\r\n { kind: VertexBuffer.ColorKind, accessorType: AccessorType.VEC4, accessorComponentType: AccessorComponentType.FLOAT, byteStride: 16 },\r\n { kind: VertexBuffer.TangentKind, accessorType: AccessorType.VEC4, accessorComponentType: AccessorComponentType.FLOAT, byteStride: 16 },\r\n { kind: VertexBuffer.UVKind, accessorType: AccessorType.VEC2, accessorComponentType: AccessorComponentType.FLOAT, byteStride: 8 },\r\n { kind: VertexBuffer.UV2Kind, accessorType: AccessorType.VEC2, accessorComponentType: AccessorComponentType.FLOAT, byteStride: 8 },\r\n { kind: VertexBuffer.MatricesIndicesKind, accessorType: AccessorType.VEC4, accessorComponentType: AccessorComponentType.UNSIGNED_SHORT, byteStride: 8 },\r\n { kind: VertexBuffer.MatricesIndicesExtraKind, accessorType: AccessorType.VEC4, accessorComponentType: AccessorComponentType.UNSIGNED_SHORT, byteStride: 8 },\r\n { kind: VertexBuffer.MatricesWeightsKind, accessorType: AccessorType.VEC4, accessorComponentType: AccessorComponentType.FLOAT, byteStride: 16 },\r\n { kind: VertexBuffer.MatricesWeightsExtraKind, accessorType: AccessorType.VEC4, accessorComponentType: AccessorComponentType.FLOAT, byteStride: 16 },\r\n ];\r\n\r\n if (bufferMesh) {\r\n let indexBufferViewIndex: Nullable = null;\r\n const primitiveMode = this._getMeshPrimitiveMode(bufferMesh);\r\n const vertexAttributeBufferViews: { [attributeKind: string]: number } = {};\r\n const morphTargetManager = bufferMesh.morphTargetManager;\r\n\r\n // For each BabylonMesh, create bufferviews for each 'kind'\r\n for (const attribute of attributeData) {\r\n const attributeKind = attribute.kind;\r\n const attributeComponentKind = attribute.accessorComponentType;\r\n if (bufferMesh.isVerticesDataPresent(attributeKind, true)) {\r\n const vertexBuffer = this._getVertexBufferFromMesh(attributeKind, bufferMesh);\r\n attribute.byteStride = vertexBuffer\r\n ? vertexBuffer.getSize() * VertexBuffer.GetTypeByteLength(attribute.accessorComponentType)\r\n : VertexBuffer.DeduceStride(attributeKind) * 4;\r\n if (attribute.byteStride === 12) {\r\n attribute.accessorType = AccessorType.VEC3;\r\n }\r\n\r\n this._createBufferViewKind(attributeKind, attributeComponentKind, babylonTransformNode, binaryWriter, attribute.byteStride);\r\n attribute.bufferViewIndex = this._bufferViews.length - 1;\r\n vertexAttributeBufferViews[attributeKind] = attribute.bufferViewIndex;\r\n }\r\n }\r\n\r\n if (bufferMesh.getTotalIndices()) {\r\n const indices = bufferMesh.getIndices();\r\n if (indices) {\r\n const byteLength = indices.length * 4;\r\n bufferView = _GLTFUtilities._CreateBufferView(0, binaryWriter.getByteOffset(), byteLength, undefined, \"Indices - \" + bufferMesh.name);\r\n this._bufferViews.push(bufferView);\r\n indexBufferViewIndex = this._bufferViews.length - 1;\r\n\r\n for (let k = 0, length = indices.length; k < length; ++k) {\r\n binaryWriter.setUInt32(indices[k]);\r\n }\r\n }\r\n }\r\n\r\n if (bufferMesh.subMeshes) {\r\n // go through all mesh primitives (submeshes)\r\n for (const submesh of bufferMesh.subMeshes) {\r\n let babylonMaterial = submesh.getMaterial() || bufferMesh.getScene().defaultMaterial;\r\n\r\n let materialIndex: Nullable = null;\r\n if (babylonMaterial) {\r\n if (bufferMesh instanceof LinesMesh) {\r\n // get the color from the lines mesh and set it in the material\r\n const material: IMaterial = {\r\n name: bufferMesh.name + \" material\",\r\n };\r\n if (!bufferMesh.color.equals(Color3.White()) || bufferMesh.alpha < 1) {\r\n material.pbrMetallicRoughness = {\r\n baseColorFactor: bufferMesh.color.asArray().concat([bufferMesh.alpha]),\r\n };\r\n }\r\n this._materials.push(material);\r\n materialIndex = this._materials.length - 1;\r\n } else if (babylonMaterial instanceof MultiMaterial) {\r\n const subMaterial = babylonMaterial.subMaterials[submesh.materialIndex];\r\n if (subMaterial) {\r\n babylonMaterial = subMaterial;\r\n materialIndex = this._materialMap[babylonMaterial.uniqueId];\r\n }\r\n } else {\r\n materialIndex = this._materialMap[babylonMaterial.uniqueId];\r\n }\r\n }\r\n\r\n const glTFMaterial: Nullable = materialIndex != null ? this._materials[materialIndex] : null;\r\n\r\n const meshPrimitive: IMeshPrimitive = { attributes: {} };\r\n this._setPrimitiveMode(meshPrimitive, primitiveMode);\r\n\r\n for (const attribute of attributeData) {\r\n const attributeKind = attribute.kind;\r\n if ((attributeKind === VertexBuffer.UVKind || attributeKind === VertexBuffer.UV2Kind) && !this._options.exportUnusedUVs) {\r\n if (!glTFMaterial || !this._glTFMaterialExporter._hasTexturesPresent(glTFMaterial)) {\r\n continue;\r\n }\r\n }\r\n const vertexData = bufferMesh.getVerticesData(attributeKind, undefined, undefined, true);\r\n if (vertexData) {\r\n const vertexBuffer = this._getVertexBufferFromMesh(attributeKind, bufferMesh);\r\n if (vertexBuffer) {\r\n const stride = vertexBuffer.getSize();\r\n const bufferViewIndex = attribute.bufferViewIndex;\r\n if (bufferViewIndex != undefined) {\r\n // check to see if bufferviewindex has a numeric value assigned.\r\n minMax = { min: null, max: null };\r\n if (attributeKind == VertexBuffer.PositionKind) {\r\n minMax = _GLTFUtilities._CalculateMinMaxPositions(vertexData, 0, vertexData.length / stride);\r\n }\r\n const accessor = _GLTFUtilities._CreateAccessor(\r\n bufferViewIndex,\r\n attributeKind + \" - \" + babylonTransformNode.name,\r\n attribute.accessorType,\r\n attribute.accessorComponentType,\r\n vertexData.length / stride,\r\n 0,\r\n minMax.min,\r\n minMax.max\r\n );\r\n this._accessors.push(accessor);\r\n this._setAttributeKind(meshPrimitive, attributeKind);\r\n }\r\n }\r\n }\r\n }\r\n\r\n if (indexBufferViewIndex) {\r\n // Create accessor\r\n const accessor = _GLTFUtilities._CreateAccessor(\r\n indexBufferViewIndex,\r\n \"indices - \" + babylonTransformNode.name,\r\n AccessorType.SCALAR,\r\n AccessorComponentType.UNSIGNED_INT,\r\n submesh.indexCount,\r\n submesh.indexStart * 4,\r\n null,\r\n null\r\n );\r\n this._accessors.push(accessor);\r\n meshPrimitive.indices = this._accessors.length - 1;\r\n }\r\n\r\n if (materialIndex != null && Object.keys(meshPrimitive.attributes).length > 0) {\r\n const sideOrientation = bufferMesh.overrideMaterialSideOrientation !== null ? bufferMesh.overrideMaterialSideOrientation : babylonMaterial.sideOrientation;\r\n\r\n if (sideOrientation === (this._babylonScene.useRightHandedSystem ? Material.ClockWiseSideOrientation : Material.CounterClockWiseSideOrientation)) {\r\n let byteOffset = indexBufferViewIndex != null ? this._bufferViews[indexBufferViewIndex].byteOffset : null;\r\n if (byteOffset == null) {\r\n byteOffset = 0;\r\n }\r\n let babylonIndices: Nullable = null;\r\n if (indexBufferViewIndex != null) {\r\n babylonIndices = bufferMesh.getIndices();\r\n }\r\n if (babylonIndices) {\r\n this._reorderIndicesBasedOnPrimitiveMode(submesh, primitiveMode, babylonIndices, byteOffset, binaryWriter);\r\n } else {\r\n for (const attribute of attributeData) {\r\n const vertexData = bufferMesh.getVerticesData(attribute.kind, undefined, undefined, true);\r\n if (vertexData) {\r\n const byteOffset = this._bufferViews[vertexAttributeBufferViews[attribute.kind]].byteOffset || 0;\r\n this._reorderVertexAttributeDataBasedOnPrimitiveMode(submesh, primitiveMode, attribute.kind, vertexData, byteOffset, binaryWriter);\r\n }\r\n }\r\n }\r\n }\r\n\r\n meshPrimitive.material = materialIndex;\r\n }\r\n if (morphTargetManager) {\r\n let target;\r\n for (let i = 0; i < morphTargetManager.numTargets; ++i) {\r\n target = morphTargetManager.getTarget(i);\r\n this._setMorphTargetAttributes(submesh, meshPrimitive, target, binaryWriter);\r\n }\r\n }\r\n\r\n mesh.primitives.push(meshPrimitive);\r\n\r\n this._extensionsPostExportMeshPrimitiveAsync(\"postExport\", meshPrimitive, submesh, binaryWriter);\r\n promises.push();\r\n }\r\n }\r\n }\r\n return Promise.all(promises).then(() => {\r\n /* do nothing */\r\n });\r\n }\r\n\r\n /**\r\n * Creates a glTF scene based on the array of meshes\r\n * Returns the the total byte offset\r\n * @param babylonScene Babylon scene to get the mesh data from\r\n * @param binaryWriter Buffer to write binary data to\r\n */\r\n private _createSceneAsync(binaryWriter: _BinaryWriter): Promise {\r\n const scene: IScene = { nodes: [] };\r\n let glTFNodeIndex: number;\r\n let glTFNode: INode;\r\n let directDescendents: Node[];\r\n const nodes: Node[] = [...this._babylonScene.transformNodes, ...this._babylonScene.meshes, ...this._babylonScene.lights, ...this._babylonScene.cameras];\r\n const removedRootNodes = new Set();\r\n\r\n // Scene metadata\r\n if (this._babylonScene.metadata) {\r\n if (this._options.metadataSelector) {\r\n scene.extras = this._options.metadataSelector(this._babylonScene.metadata);\r\n } else if (this._babylonScene.metadata.gltf) {\r\n scene.extras = this._babylonScene.metadata.gltf.extras;\r\n }\r\n }\r\n\r\n // Remove no-op root nodes\r\n if ((this._options.removeNoopRootNodes ?? true) && !this._options.includeCoordinateSystemConversionNodes) {\r\n for (const rootNode of this._babylonScene.rootNodes) {\r\n if (isNoopNode(rootNode, this._babylonScene.useRightHandedSystem)) {\r\n removedRootNodes.add(rootNode);\r\n\r\n // Exclude the node from list of nodes to export\r\n nodes.splice(nodes.indexOf(rootNode), 1);\r\n }\r\n }\r\n }\r\n\r\n // Export babylon cameras to glTFCamera\r\n const cameraMap = new Map();\r\n this._babylonScene.cameras.forEach((camera) => {\r\n if (!this._options.shouldExportNode || this._options.shouldExportNode(camera)) {\r\n const glTFCamera: ICamera = {\r\n type: camera.mode === Camera.PERSPECTIVE_CAMERA ? CameraType.PERSPECTIVE : CameraType.ORTHOGRAPHIC,\r\n };\r\n\r\n if (camera.name) {\r\n glTFCamera.name = camera.name;\r\n }\r\n\r\n if (glTFCamera.type === CameraType.PERSPECTIVE) {\r\n glTFCamera.perspective = {\r\n aspectRatio: camera.getEngine().getAspectRatio(camera),\r\n yfov: camera.fovMode === Camera.FOVMODE_VERTICAL_FIXED ? camera.fov : camera.fov * camera.getEngine().getAspectRatio(camera),\r\n znear: camera.minZ,\r\n zfar: camera.maxZ,\r\n };\r\n } else if (glTFCamera.type === CameraType.ORTHOGRAPHIC) {\r\n const halfWidth = camera.orthoLeft && camera.orthoRight ? 0.5 * (camera.orthoRight - camera.orthoLeft) : camera.getEngine().getRenderWidth() * 0.5;\r\n const halfHeight = camera.orthoBottom && camera.orthoTop ? 0.5 * (camera.orthoTop - camera.orthoBottom) : camera.getEngine().getRenderHeight() * 0.5;\r\n glTFCamera.orthographic = {\r\n xmag: halfWidth,\r\n ymag: halfHeight,\r\n znear: camera.minZ,\r\n zfar: camera.maxZ,\r\n };\r\n }\r\n\r\n cameraMap.set(camera, this._cameras.length);\r\n this._cameras.push(glTFCamera);\r\n }\r\n });\r\n\r\n const [exportNodes, exportMaterials] = this._getExportNodes(nodes);\r\n return this._glTFMaterialExporter._convertMaterialsToGLTFAsync(exportMaterials, ImageMimeType.PNG, true).then(() => {\r\n return this._createNodeMapAndAnimationsAsync(exportNodes, binaryWriter).then((nodeMap) => {\r\n return this._createSkinsAsync(nodeMap, binaryWriter).then((skinMap) => {\r\n this._nodeMap = nodeMap;\r\n\r\n this._totalByteLength = binaryWriter.getByteOffset();\r\n if (this._totalByteLength == undefined) {\r\n throw new Error(\"undefined byte length!\");\r\n }\r\n\r\n // Build Hierarchy with the node map.\r\n for (const babylonNode of nodes) {\r\n glTFNodeIndex = this._nodeMap[babylonNode.uniqueId];\r\n if (glTFNodeIndex !== undefined) {\r\n glTFNode = this._nodes[glTFNodeIndex];\r\n\r\n if (babylonNode.metadata) {\r\n if (this._options.metadataSelector) {\r\n glTFNode.extras = this._options.metadataSelector(babylonNode.metadata);\r\n } else if (babylonNode.metadata.gltf) {\r\n glTFNode.extras = babylonNode.metadata.gltf.extras;\r\n }\r\n }\r\n\r\n if (babylonNode instanceof Camera) {\r\n glTFNode.camera = cameraMap.get(babylonNode);\r\n }\r\n\r\n if (this._options.shouldExportNode && !this._options.shouldExportNode(babylonNode)) {\r\n Tools.Log(\"Omitting \" + babylonNode.name + \" from scene.\");\r\n } else {\r\n if (!babylonNode.parent && !this._babylonScene.useRightHandedSystem) {\r\n convertNodeHandedness(glTFNode);\r\n }\r\n\r\n if (!babylonNode.parent || removedRootNodes.has(babylonNode.parent)) {\r\n scene.nodes.push(glTFNodeIndex);\r\n }\r\n }\r\n\r\n if (babylonNode instanceof Mesh) {\r\n if (babylonNode.skeleton) {\r\n glTFNode.skin = skinMap[babylonNode.skeleton.uniqueId];\r\n }\r\n }\r\n\r\n directDescendents = babylonNode.getDescendants(true);\r\n if (!glTFNode.children && directDescendents && directDescendents.length) {\r\n const children: number[] = [];\r\n for (const descendent of directDescendents) {\r\n if (this._nodeMap[descendent.uniqueId] != null) {\r\n children.push(this._nodeMap[descendent.uniqueId]);\r\n }\r\n }\r\n if (children.length) {\r\n glTFNode.children = children;\r\n }\r\n }\r\n }\r\n }\r\n if (scene.nodes.length) {\r\n this._scenes.push(scene);\r\n }\r\n });\r\n });\r\n });\r\n }\r\n\r\n /**\r\n * Getting the nodes and materials that would be exported.\r\n * @param nodes Babylon transform nodes\r\n * @returns Array of nodes which would be exported.\r\n * @returns Set of materials which would be exported.\r\n */\r\n private _getExportNodes(nodes: Node[]): [Node[], Set] {\r\n const exportNodes: Node[] = [];\r\n const exportMaterials: Set = new Set();\r\n\r\n for (const babylonNode of nodes) {\r\n if (!this._options.shouldExportNode || this._options.shouldExportNode(babylonNode)) {\r\n exportNodes.push(babylonNode);\r\n\r\n const babylonMesh = babylonNode as AbstractMesh;\r\n if (babylonMesh.subMeshes && babylonMesh.subMeshes.length > 0) {\r\n const material = babylonMesh.material || babylonMesh.getScene().defaultMaterial;\r\n if (material instanceof MultiMaterial) {\r\n for (const subMaterial of material.subMaterials) {\r\n if (subMaterial) {\r\n exportMaterials.add(subMaterial);\r\n }\r\n }\r\n } else {\r\n exportMaterials.add(material);\r\n }\r\n }\r\n } else {\r\n `Excluding node ${babylonNode.name}`;\r\n }\r\n }\r\n\r\n return [exportNodes, exportMaterials];\r\n }\r\n\r\n /**\r\n * Creates a mapping of Node unique id to node index and handles animations\r\n * @param nodes Babylon transform nodes\r\n * @param binaryWriter Buffer to write binary data to\r\n * @returns Node mapping of unique id to index\r\n */\r\n private _createNodeMapAndAnimationsAsync(nodes: Node[], binaryWriter: _BinaryWriter): Promise<{ [key: number]: number }> {\r\n let promiseChain = Promise.resolve();\r\n const nodeMap: { [key: number]: number } = {};\r\n let nodeIndex: number;\r\n const runtimeGLTFAnimation: IAnimation = {\r\n name: \"runtime animations\",\r\n channels: [],\r\n samplers: [],\r\n };\r\n const idleGLTFAnimations: IAnimation[] = [];\r\n\r\n for (const babylonNode of nodes) {\r\n promiseChain = promiseChain.then(() => {\r\n return this._createNodeAsync(babylonNode, binaryWriter).then((node) => {\r\n const promise = this._extensionsPostExportNodeAsync(\"createNodeAsync\", node, babylonNode, nodeMap, binaryWriter);\r\n if (promise == null) {\r\n Tools.Warn(`Not exporting node ${babylonNode.name}`);\r\n return Promise.resolve();\r\n } else {\r\n return promise.then((node) => {\r\n if (!node) {\r\n return;\r\n }\r\n this._nodes.push(node);\r\n nodeIndex = this._nodes.length - 1;\r\n nodeMap[babylonNode.uniqueId] = nodeIndex;\r\n\r\n if (!this._babylonScene.animationGroups.length) {\r\n _GLTFAnimation._CreateMorphTargetAnimationFromMorphTargetAnimations(\r\n babylonNode,\r\n runtimeGLTFAnimation,\r\n idleGLTFAnimations,\r\n nodeMap,\r\n this._nodes,\r\n binaryWriter,\r\n this._bufferViews,\r\n this._accessors,\r\n this._animationSampleRate,\r\n this._options.shouldExportAnimation\r\n );\r\n if (babylonNode.animations.length) {\r\n _GLTFAnimation._CreateNodeAnimationFromNodeAnimations(\r\n babylonNode,\r\n runtimeGLTFAnimation,\r\n idleGLTFAnimations,\r\n nodeMap,\r\n this._nodes,\r\n binaryWriter,\r\n this._bufferViews,\r\n this._accessors,\r\n this._animationSampleRate,\r\n this._options.shouldExportAnimation\r\n );\r\n }\r\n }\r\n });\r\n }\r\n });\r\n });\r\n }\r\n\r\n return promiseChain.then(() => {\r\n if (runtimeGLTFAnimation.channels.length && runtimeGLTFAnimation.samplers.length) {\r\n this._animations.push(runtimeGLTFAnimation);\r\n }\r\n idleGLTFAnimations.forEach((idleGLTFAnimation) => {\r\n if (idleGLTFAnimation.channels.length && idleGLTFAnimation.samplers.length) {\r\n this._animations.push(idleGLTFAnimation);\r\n }\r\n });\r\n\r\n if (this._babylonScene.animationGroups.length) {\r\n _GLTFAnimation._CreateNodeAndMorphAnimationFromAnimationGroups(\r\n this._babylonScene,\r\n this._animations,\r\n nodeMap,\r\n binaryWriter,\r\n this._bufferViews,\r\n this._accessors,\r\n this._animationSampleRate,\r\n this._options.shouldExportAnimation\r\n );\r\n }\r\n\r\n return nodeMap;\r\n });\r\n }\r\n\r\n /**\r\n * Creates a glTF node from a Babylon mesh\r\n * @param babylonNode Source Babylon mesh\r\n * @param binaryWriter Buffer for storing geometry data\r\n * @returns glTF node\r\n */\r\n private _createNodeAsync(babylonNode: Node, binaryWriter: _BinaryWriter): Promise {\r\n return Promise.resolve().then(() => {\r\n // create node to hold translation/rotation/scale and the mesh\r\n const node: INode = {};\r\n // create mesh\r\n const mesh: IMesh = { primitives: [] };\r\n\r\n if (babylonNode.name) {\r\n node.name = babylonNode.name;\r\n }\r\n\r\n if (babylonNode instanceof TransformNode) {\r\n // Set transformation\r\n this._setNodeTransformation(node, babylonNode);\r\n if (babylonNode instanceof Mesh) {\r\n const morphTargetManager = babylonNode.morphTargetManager;\r\n if (morphTargetManager && morphTargetManager.numTargets > 0) {\r\n mesh.weights = [];\r\n for (let i = 0; i < morphTargetManager.numTargets; ++i) {\r\n mesh.weights.push(morphTargetManager.getTarget(i).influence);\r\n }\r\n }\r\n }\r\n return this._setPrimitiveAttributesAsync(mesh, babylonNode, binaryWriter).then(() => {\r\n if (mesh.primitives.length) {\r\n this._meshes.push(mesh);\r\n node.mesh = this._meshes.length - 1;\r\n }\r\n return node;\r\n });\r\n } else if (babylonNode instanceof Camera) {\r\n this._setCameraTransformation(node, babylonNode);\r\n return node;\r\n } else {\r\n return node;\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Creates a glTF skin from a Babylon skeleton\r\n * @param babylonScene Babylon Scene\r\n * @param nodeMap Babylon transform nodes\r\n * @param binaryWriter Buffer to write binary data to\r\n * @returns Node mapping of unique id to index\r\n */\r\n private _createSkinsAsync(nodeMap: { [key: number]: number }, binaryWriter: _BinaryWriter): Promise<{ [key: number]: number }> {\r\n const promiseChain = Promise.resolve();\r\n const skinMap: { [key: number]: number } = {};\r\n for (const skeleton of this._babylonScene.skeletons) {\r\n if (skeleton.bones.length <= 0) {\r\n continue;\r\n }\r\n // create skin\r\n const skin: ISkin = { joints: [] };\r\n const inverseBindMatrices: Matrix[] = [];\r\n\r\n const boneIndexMap: { [index: number]: Bone } = {};\r\n let maxBoneIndex = -1;\r\n for (let i = 0; i < skeleton.bones.length; ++i) {\r\n const bone = skeleton.bones[i];\r\n const boneIndex = bone.getIndex() ?? i;\r\n if (boneIndex !== -1) {\r\n boneIndexMap[boneIndex] = bone;\r\n if (boneIndex > maxBoneIndex) {\r\n maxBoneIndex = boneIndex;\r\n }\r\n }\r\n }\r\n\r\n for (let boneIndex = 0; boneIndex <= maxBoneIndex; ++boneIndex) {\r\n const bone = boneIndexMap[boneIndex];\r\n inverseBindMatrices.push(bone.getInvertedAbsoluteTransform());\r\n\r\n const transformNode = bone.getTransformNode();\r\n if (transformNode && nodeMap[transformNode.uniqueId] !== null && nodeMap[transformNode.uniqueId] !== undefined) {\r\n skin.joints.push(nodeMap[transformNode.uniqueId]);\r\n } else {\r\n Tools.Warn(\"Exporting a bone without a linked transform node is currently unsupported\");\r\n }\r\n }\r\n\r\n if (skin.joints.length > 0) {\r\n // create buffer view for inverse bind matrices\r\n const byteStride = 64; // 4 x 4 matrix of 32 bit float\r\n const byteLength = inverseBindMatrices.length * byteStride;\r\n const bufferViewOffset = binaryWriter.getByteOffset();\r\n const bufferView = _GLTFUtilities._CreateBufferView(0, bufferViewOffset, byteLength, undefined, \"InverseBindMatrices\" + \" - \" + skeleton.name);\r\n this._bufferViews.push(bufferView);\r\n const bufferViewIndex = this._bufferViews.length - 1;\r\n const bindMatrixAccessor = _GLTFUtilities._CreateAccessor(\r\n bufferViewIndex,\r\n \"InverseBindMatrices\" + \" - \" + skeleton.name,\r\n AccessorType.MAT4,\r\n AccessorComponentType.FLOAT,\r\n inverseBindMatrices.length,\r\n null,\r\n null,\r\n null\r\n );\r\n const inverseBindAccessorIndex = this._accessors.push(bindMatrixAccessor) - 1;\r\n skin.inverseBindMatrices = inverseBindAccessorIndex;\r\n this._skins.push(skin);\r\n skinMap[skeleton.uniqueId] = this._skins.length - 1;\r\n\r\n inverseBindMatrices.forEach((mat) => {\r\n mat.m.forEach((cell: number) => {\r\n binaryWriter.setFloat32(cell);\r\n });\r\n });\r\n }\r\n }\r\n return promiseChain.then(() => {\r\n return skinMap;\r\n });\r\n }\r\n}\r\n\r\n/**\r\n * @internal\r\n *\r\n * Stores glTF binary data. If the array buffer byte length is exceeded, it doubles in size dynamically\r\n */\r\nexport class _BinaryWriter {\r\n /**\r\n * Array buffer which stores all binary data\r\n */\r\n private _arrayBuffer: ArrayBuffer;\r\n /**\r\n * View of the array buffer\r\n */\r\n private _dataView: DataView;\r\n /**\r\n * byte offset of data in array buffer\r\n */\r\n private _byteOffset: number;\r\n /**\r\n * Initialize binary writer with an initial byte length\r\n * @param byteLength Initial byte length of the array buffer\r\n */\r\n constructor(byteLength: number) {\r\n this._arrayBuffer = new ArrayBuffer(byteLength);\r\n this._dataView = new DataView(this._arrayBuffer);\r\n this._byteOffset = 0;\r\n }\r\n /**\r\n * Resize the array buffer to the specified byte length\r\n * @param byteLength\r\n */\r\n private _resizeBuffer(byteLength: number): ArrayBuffer {\r\n const newBuffer = new ArrayBuffer(byteLength);\r\n const copyOldBufferSize = Math.min(this._arrayBuffer.byteLength, byteLength);\r\n const oldUint8Array = new Uint8Array(this._arrayBuffer, 0, copyOldBufferSize);\r\n const newUint8Array = new Uint8Array(newBuffer);\r\n newUint8Array.set(oldUint8Array, 0);\r\n this._arrayBuffer = newBuffer;\r\n this._dataView = new DataView(this._arrayBuffer);\r\n\r\n return newBuffer;\r\n }\r\n /**\r\n * Get an array buffer with the length of the byte offset\r\n * @returns ArrayBuffer resized to the byte offset\r\n */\r\n public getArrayBuffer(): ArrayBuffer {\r\n return this._resizeBuffer(this.getByteOffset());\r\n }\r\n /**\r\n * Get the byte offset of the array buffer\r\n * @returns byte offset\r\n */\r\n public getByteOffset(): number {\r\n if (this._byteOffset == undefined) {\r\n throw new Error(\"Byte offset is undefined!\");\r\n }\r\n return this._byteOffset;\r\n }\r\n /**\r\n * Stores an UInt8 in the array buffer\r\n * @param entry\r\n * @param byteOffset If defined, specifies where to set the value as an offset.\r\n */\r\n public setUInt8(entry: number, byteOffset?: number) {\r\n if (byteOffset != null) {\r\n if (byteOffset < this._byteOffset) {\r\n this._dataView.setUint8(byteOffset, entry);\r\n } else {\r\n Tools.Error(\"BinaryWriter: byteoffset is greater than the current binary buffer length!\");\r\n }\r\n } else {\r\n if (this._byteOffset + 1 > this._arrayBuffer.byteLength) {\r\n this._resizeBuffer(this._arrayBuffer.byteLength * 2);\r\n }\r\n this._dataView.setUint8(this._byteOffset, entry);\r\n this._byteOffset += 1;\r\n }\r\n }\r\n\r\n /**\r\n * Stores an UInt16 in the array buffer\r\n * @param entry\r\n * @param byteOffset If defined, specifies where to set the value as an offset.\r\n */\r\n public setUInt16(entry: number, byteOffset?: number) {\r\n if (byteOffset != null) {\r\n if (byteOffset < this._byteOffset) {\r\n this._dataView.setUint16(byteOffset, entry, true);\r\n } else {\r\n Tools.Error(\"BinaryWriter: byteoffset is greater than the current binary buffer length!\");\r\n }\r\n } else {\r\n if (this._byteOffset + 2 > this._arrayBuffer.byteLength) {\r\n this._resizeBuffer(this._arrayBuffer.byteLength * 2);\r\n }\r\n this._dataView.setUint16(this._byteOffset, entry, true);\r\n this._byteOffset += 2;\r\n }\r\n }\r\n\r\n /**\r\n * Gets an UInt32 in the array buffer\r\n * @param byteOffset If defined, specifies where to set the value as an offset.\r\n */\r\n public getUInt32(byteOffset: number): number {\r\n if (byteOffset < this._byteOffset) {\r\n return this._dataView.getUint32(byteOffset, true);\r\n } else {\r\n Tools.Error(\"BinaryWriter: byteoffset is greater than the current binary buffer length!\");\r\n throw new Error(\"BinaryWriter: byteoffset is greater than the current binary buffer length!\");\r\n }\r\n }\r\n\r\n public getVector3Float32FromRef(vector3: Vector3, byteOffset: number): void {\r\n if (byteOffset + 8 > this._byteOffset) {\r\n Tools.Error(`BinaryWriter: byteoffset is greater than the current binary buffer length!`);\r\n } else {\r\n vector3.x = this._dataView.getFloat32(byteOffset, true);\r\n vector3.y = this._dataView.getFloat32(byteOffset + 4, true);\r\n vector3.z = this._dataView.getFloat32(byteOffset + 8, true);\r\n }\r\n }\r\n\r\n public setVector3Float32FromRef(vector3: Vector3, byteOffset: number): void {\r\n if (byteOffset + 8 > this._byteOffset) {\r\n Tools.Error(`BinaryWriter: byteoffset is greater than the current binary buffer length!`);\r\n } else {\r\n this._dataView.setFloat32(byteOffset, vector3.x, true);\r\n this._dataView.setFloat32(byteOffset + 4, vector3.y, true);\r\n this._dataView.setFloat32(byteOffset + 8, vector3.z, true);\r\n }\r\n }\r\n\r\n public getVector4Float32FromRef(vector4: Vector4, byteOffset: number): void {\r\n if (byteOffset + 12 > this._byteOffset) {\r\n Tools.Error(`BinaryWriter: byteoffset is greater than the current binary buffer length!`);\r\n } else {\r\n vector4.x = this._dataView.getFloat32(byteOffset, true);\r\n vector4.y = this._dataView.getFloat32(byteOffset + 4, true);\r\n vector4.z = this._dataView.getFloat32(byteOffset + 8, true);\r\n vector4.w = this._dataView.getFloat32(byteOffset + 12, true);\r\n }\r\n }\r\n\r\n public setVector4Float32FromRef(vector4: Vector4, byteOffset: number): void {\r\n if (byteOffset + 12 > this._byteOffset) {\r\n Tools.Error(`BinaryWriter: byteoffset is greater than the current binary buffer length!`);\r\n } else {\r\n this._dataView.setFloat32(byteOffset, vector4.x, true);\r\n this._dataView.setFloat32(byteOffset + 4, vector4.y, true);\r\n this._dataView.setFloat32(byteOffset + 8, vector4.z, true);\r\n this._dataView.setFloat32(byteOffset + 12, vector4.w, true);\r\n }\r\n }\r\n /**\r\n * Stores a Float32 in the array buffer\r\n * @param entry\r\n * @param byteOffset\r\n */\r\n public setFloat32(entry: number, byteOffset?: number) {\r\n if (isNaN(entry)) {\r\n Tools.Error(\"Invalid data being written!\");\r\n }\r\n if (byteOffset != null) {\r\n if (byteOffset < this._byteOffset) {\r\n this._dataView.setFloat32(byteOffset, entry, true);\r\n } else {\r\n Tools.Error(\"BinaryWriter: byteoffset is greater than the current binary length!\");\r\n }\r\n }\r\n if (this._byteOffset + 4 > this._arrayBuffer.byteLength) {\r\n this._resizeBuffer(this._arrayBuffer.byteLength * 2);\r\n }\r\n this._dataView.setFloat32(this._byteOffset, entry, true);\r\n this._byteOffset += 4;\r\n }\r\n /**\r\n * Stores an UInt32 in the array buffer\r\n * @param entry\r\n * @param byteOffset If defined, specifies where to set the value as an offset.\r\n */\r\n public setUInt32(entry: number, byteOffset?: number) {\r\n if (byteOffset != null) {\r\n if (byteOffset < this._byteOffset) {\r\n this._dataView.setUint32(byteOffset, entry, true);\r\n } else {\r\n Tools.Error(\"BinaryWriter: byteoffset is greater than the current binary buffer length!\");\r\n }\r\n } else {\r\n if (this._byteOffset + 4 > this._arrayBuffer.byteLength) {\r\n this._resizeBuffer(this._arrayBuffer.byteLength * 2);\r\n }\r\n this._dataView.setUint32(this._byteOffset, entry, true);\r\n this._byteOffset += 4;\r\n }\r\n }\r\n /**\r\n * Stores an Int16 in the array buffer\r\n * @param entry\r\n * @param byteOffset If defined, specifies where to set the value as an offset.\r\n */\r\n public setInt16(entry: number, byteOffset?: number) {\r\n if (byteOffset != null) {\r\n if (byteOffset < this._byteOffset) {\r\n this._dataView.setInt16(byteOffset, entry, true);\r\n } else {\r\n Tools.Error(\"BinaryWriter: byteoffset is greater than the current binary buffer length!\");\r\n }\r\n } else {\r\n if (this._byteOffset + 2 > this._arrayBuffer.byteLength) {\r\n this._resizeBuffer(this._arrayBuffer.byteLength * 2);\r\n }\r\n this._dataView.setInt16(this._byteOffset, entry, true);\r\n this._byteOffset += 2;\r\n }\r\n }\r\n /**\r\n * Stores a byte in the array buffer\r\n * @param entry\r\n * @param byteOffset If defined, specifies where to set the value as an offset.\r\n */\r\n public setByte(entry: number, byteOffset?: number) {\r\n if (byteOffset != null) {\r\n if (byteOffset < this._byteOffset) {\r\n this._dataView.setInt8(byteOffset, entry);\r\n } else {\r\n Tools.Error(\"BinaryWriter: byteoffset is greater than the current binary buffer length!\");\r\n }\r\n } else {\r\n if (this._byteOffset + 1 > this._arrayBuffer.byteLength) {\r\n this._resizeBuffer(this._arrayBuffer.byteLength * 2);\r\n }\r\n this._dataView.setInt8(this._byteOffset, entry);\r\n this._byteOffset++;\r\n }\r\n }\r\n}\r\n","import type { Node } from \"core/node\";\r\nimport type { Scene } from \"core/scene\";\r\nimport type { Animation } from \"core/Animations/animation\";\r\nimport type { GLTFData } from \"./glTFData\";\r\nimport { _Exporter } from \"./glTFExporter\";\r\n\r\n/**\r\n * Holds a collection of exporter options and parameters\r\n */\r\nexport interface IExportOptions {\r\n /**\r\n * Function which indicates whether a babylon node should be exported or not\r\n * @param node source Babylon node. It is used to check whether it should be exported to glTF or not\r\n * @returns boolean, which indicates whether the node should be exported (true) or not (false)\r\n */\r\n shouldExportNode?(node: Node): boolean;\r\n\r\n /**\r\n * Function which indicates whether an animation on the scene should be exported or not\r\n * @param animation source animation\r\n * @returns boolean, which indicates whether the animation should be exported (true) or not (false)\r\n */\r\n shouldExportAnimation?(animation: Animation): boolean;\r\n\r\n /**\r\n * Function used to extract the part of node's metadata that will be exported into glTF node extras\r\n * @param metadata source metadata to read from\r\n * @returns the data to store to glTF node extras\r\n */\r\n metadataSelector?(metadata: any): any;\r\n\r\n /**\r\n * The sample rate to bake animation curves. Defaults to 1 / 60.\r\n */\r\n animationSampleRate?: number;\r\n\r\n /**\r\n * Begin serialization without waiting for the scene to be ready. Defaults to false.\r\n */\r\n exportWithoutWaitingForScene?: boolean;\r\n\r\n /**\r\n * Indicates if unused vertex uv attributes should be included in export. Defaults to false.\r\n */\r\n exportUnusedUVs?: boolean;\r\n\r\n /**\r\n * Remove no-op root nodes when possible. Defaults to true.\r\n */\r\n removeNoopRootNodes?: boolean;\r\n\r\n /**\r\n * Indicates if coordinate system swapping root nodes should be included in export. Defaults to false.\r\n * @deprecated Please use removeNoopRootNodes instead\r\n */\r\n includeCoordinateSystemConversionNodes?: boolean;\r\n}\r\n\r\n/**\r\n * Class for generating glTF data from a Babylon scene.\r\n */\r\nexport class GLTF2Export {\r\n /**\r\n * Exports the geometry of the scene to .gltf file format asynchronously\r\n * @param scene Babylon scene with scene hierarchy information\r\n * @param filePrefix File prefix to use when generating the glTF file\r\n * @param options Exporter options\r\n * @returns Returns an object with a .gltf file and associates texture names\r\n * as keys and their data and paths as values\r\n */\r\n public static GLTFAsync(scene: Scene, filePrefix: string, options?: IExportOptions): Promise {\r\n return scene.whenReadyAsync().then(() => {\r\n const glTFPrefix = filePrefix.replace(/\\.[^/.]+$/, \"\");\r\n const gltfGenerator = new _Exporter(scene, options);\r\n return gltfGenerator._generateGLTFAsync(glTFPrefix);\r\n });\r\n }\r\n\r\n private static _PreExportAsync(scene: Scene, options?: IExportOptions): Promise {\r\n return Promise.resolve().then(() => {\r\n if (options && options.exportWithoutWaitingForScene) {\r\n return Promise.resolve();\r\n } else {\r\n return scene.whenReadyAsync();\r\n }\r\n });\r\n }\r\n\r\n private static _PostExportAsync(scene: Scene, glTFData: GLTFData, options?: IExportOptions): Promise {\r\n return Promise.resolve().then(() => {\r\n if (options && options.exportWithoutWaitingForScene) {\r\n return glTFData;\r\n } else {\r\n return glTFData;\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Exports the geometry of the scene to .glb file format asychronously\r\n * @param scene Babylon scene with scene hierarchy information\r\n * @param filePrefix File prefix to use when generating glb file\r\n * @param options Exporter options\r\n * @returns Returns an object with a .glb filename as key and data as value\r\n */\r\n public static GLBAsync(scene: Scene, filePrefix: string, options?: IExportOptions): Promise {\r\n return this._PreExportAsync(scene, options).then(() => {\r\n const glTFPrefix = filePrefix.replace(/\\.[^/.]+$/, \"\");\r\n const gltfGenerator = new _Exporter(scene, options);\r\n return gltfGenerator._generateGLBAsync(glTFPrefix).then((glTFData) => {\r\n return this._PostExportAsync(scene, glTFData, options);\r\n });\r\n });\r\n }\r\n}\r\n","import type { ITextureInfo, IKHRTextureTransform } from \"babylonjs-gltf2interface\";\r\nimport { Tools } from \"core/Misc/tools\";\r\nimport type { Texture } from \"core/Materials/Textures/texture\";\r\nimport type { Nullable } from \"core/types\";\r\nimport type { IGLTFExporterExtensionV2 } from \"../glTFExporterExtension\";\r\nimport { _Exporter } from \"../glTFExporter\";\r\n\r\nconst NAME = \"KHR_texture_transform\";\r\n\r\n/**\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class KHR_texture_transform implements IGLTFExporterExtensionV2 {\r\n /** Name of this extension */\r\n public readonly name = NAME;\r\n\r\n /** Defines whether this extension is enabled */\r\n public enabled = true;\r\n\r\n /** Defines whether this extension is required */\r\n public required = false;\r\n\r\n /** Reference to the glTF exporter */\r\n private _wasUsed = false;\r\n\r\n constructor() {}\r\n\r\n public dispose() {}\r\n\r\n /** @internal */\r\n public get wasUsed() {\r\n return this._wasUsed;\r\n }\r\n\r\n public postExportTexture?(context: string, textureInfo: ITextureInfo, babylonTexture: Texture): void {\r\n const canUseExtension =\r\n babylonTexture &&\r\n ((babylonTexture.uAng === 0 && babylonTexture.wAng === 0 && babylonTexture.vAng === 0) ||\r\n (babylonTexture.uRotationCenter === 0 && babylonTexture.vRotationCenter === 0));\r\n\r\n if (canUseExtension) {\r\n const textureTransform: IKHRTextureTransform = {};\r\n let transformIsRequired = false;\r\n\r\n if (babylonTexture.uOffset !== 0 || babylonTexture.vOffset !== 0) {\r\n textureTransform.offset = [babylonTexture.uOffset, babylonTexture.vOffset];\r\n transformIsRequired = true;\r\n }\r\n\r\n if (babylonTexture.uScale !== 1 || babylonTexture.vScale !== 1) {\r\n textureTransform.scale = [babylonTexture.uScale, babylonTexture.vScale];\r\n transformIsRequired = true;\r\n }\r\n\r\n if (babylonTexture.wAng !== 0) {\r\n textureTransform.rotation = -babylonTexture.wAng;\r\n transformIsRequired = true;\r\n }\r\n\r\n if (babylonTexture.coordinatesIndex !== 0) {\r\n textureTransform.texCoord = babylonTexture.coordinatesIndex;\r\n transformIsRequired = true;\r\n }\r\n\r\n if (!transformIsRequired) {\r\n return;\r\n }\r\n\r\n this._wasUsed = true;\r\n if (!textureInfo.extensions) {\r\n textureInfo.extensions = {};\r\n }\r\n textureInfo.extensions[NAME] = textureTransform;\r\n }\r\n }\r\n\r\n public preExportTextureAsync(context: string, babylonTexture: Texture): Promise> {\r\n return new Promise((resolve, reject) => {\r\n const scene = babylonTexture.getScene();\r\n if (!scene) {\r\n reject(`${context}: \"scene\" is not defined for Babylon texture ${babylonTexture.name}!`);\r\n return;\r\n }\r\n\r\n /*\r\n * The KHR_texture_transform schema only supports w rotation around the origin.\r\n * See https://github.com/KhronosGroup/glTF/tree/main/extensions/2.0/Khronos/KHR_texture_transform#gltf-schema-updates.\r\n */\r\n if (babylonTexture.uAng !== 0 || babylonTexture.vAng !== 0) {\r\n Tools.Warn(`${context}: Texture ${babylonTexture.name} with rotation in the u or v axis is not supported in glTF.`);\r\n resolve(null);\r\n } else if (babylonTexture.wAng !== 0 && (babylonTexture.uRotationCenter !== 0 || babylonTexture.vRotationCenter !== 0)) {\r\n Tools.Warn(`${context}: Texture ${babylonTexture.name} with rotation not centered at the origin cannot be exported with ${NAME}`);\r\n resolve(null);\r\n } else {\r\n resolve(babylonTexture);\r\n }\r\n });\r\n }\r\n}\r\n\r\n_Exporter.RegisterExtension(NAME, () => new KHR_texture_transform());\r\n","import type { SpotLight } from \"core/Lights/spotLight\";\r\nimport type { Nullable } from \"core/types\";\r\nimport { Vector3, Quaternion, TmpVectors, Matrix } from \"core/Maths/math.vector\";\r\nimport { Color3 } from \"core/Maths/math.color\";\r\nimport { Light } from \"core/Lights/light\";\r\nimport type { Node } from \"core/node\";\r\nimport { ShadowLight } from \"core/Lights/shadowLight\";\r\nimport type { INode, IKHRLightsPunctual_LightReference, IKHRLightsPunctual_Light, IKHRLightsPunctual } from \"babylonjs-gltf2interface\";\r\nimport { KHRLightsPunctual_LightType } from \"babylonjs-gltf2interface\";\r\nimport type { IGLTFExporterExtensionV2 } from \"../glTFExporterExtension\";\r\nimport { _Exporter } from \"../glTFExporter\";\r\nimport { Logger } from \"core/Misc/logger\";\r\nimport { _GLTFUtilities } from \"../glTFUtilities\";\r\n\r\nconst NAME = \"KHR_lights_punctual\";\r\n\r\n/**\r\n * [Specification](https://github.com/KhronosGroup/glTF/blob/master/extensions/2.0/Khronos/KHR_lights_punctual/README.md)\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class KHR_lights_punctual implements IGLTFExporterExtensionV2 {\r\n /** The name of this extension. */\r\n public readonly name = NAME;\r\n\r\n /** Defines whether this extension is enabled. */\r\n public enabled = true;\r\n\r\n /** Defines whether this extension is required */\r\n public required = false;\r\n\r\n /** Reference to the glTF exporter */\r\n private _exporter: _Exporter;\r\n\r\n private _lights: IKHRLightsPunctual;\r\n\r\n /**\r\n * @internal\r\n */\r\n constructor(exporter: _Exporter) {\r\n this._exporter = exporter;\r\n }\r\n\r\n /** @internal */\r\n public dispose() {\r\n (this._lights as any) = null;\r\n }\r\n\r\n /** @internal */\r\n public get wasUsed() {\r\n return !!this._lights;\r\n }\r\n\r\n /** @internal */\r\n public onExporting(): void {\r\n this._exporter!._glTF.extensions![NAME] = this._lights;\r\n }\r\n /**\r\n * Define this method to modify the default behavior when exporting a node\r\n * @param context The context when exporting the node\r\n * @param node glTF node\r\n * @param babylonNode BabylonJS node\r\n * @param nodeMap Node mapping of unique id to glTF node index\r\n * @returns nullable INode promise\r\n */\r\n public postExportNodeAsync(context: string, node: Nullable, babylonNode: Node, nodeMap: { [key: number]: number }): Promise> {\r\n return new Promise((resolve) => {\r\n if (node && babylonNode instanceof ShadowLight) {\r\n let light: IKHRLightsPunctual_Light;\r\n\r\n const lightType =\r\n babylonNode.getTypeID() == Light.LIGHTTYPEID_POINTLIGHT\r\n ? KHRLightsPunctual_LightType.POINT\r\n : babylonNode.getTypeID() == Light.LIGHTTYPEID_DIRECTIONALLIGHT\r\n ? KHRLightsPunctual_LightType.DIRECTIONAL\r\n : babylonNode.getTypeID() == Light.LIGHTTYPEID_SPOTLIGHT\r\n ? KHRLightsPunctual_LightType.SPOT\r\n : null;\r\n if (lightType == null) {\r\n Logger.Warn(`${context}: Light ${babylonNode.name} is not supported in ${NAME}`);\r\n } else {\r\n if (!babylonNode.position.equalsToFloats(0, 0, 0)) {\r\n node.translation = babylonNode.position.asArray();\r\n }\r\n if (lightType !== KHRLightsPunctual_LightType.POINT) {\r\n const localAxis = babylonNode.direction;\r\n const yaw = -Math.atan2(localAxis.z, localAxis.x) + Math.PI / 2;\r\n const len = Math.sqrt(localAxis.x * localAxis.x + localAxis.z * localAxis.z);\r\n const pitch = -Math.atan2(localAxis.y, len);\r\n const lightRotationQuaternion = Quaternion.RotationYawPitchRoll(yaw + Math.PI, pitch, 0);\r\n if (!Quaternion.IsIdentity(lightRotationQuaternion)) {\r\n node.rotation = lightRotationQuaternion.asArray();\r\n }\r\n }\r\n\r\n if (babylonNode.falloffType !== Light.FALLOFF_GLTF) {\r\n Logger.Warn(`${context}: Light falloff for ${babylonNode.name} does not match the ${NAME} specification!`);\r\n }\r\n light = {\r\n type: lightType,\r\n };\r\n if (!babylonNode.diffuse.equals(Color3.White())) {\r\n light.color = babylonNode.diffuse.asArray();\r\n }\r\n if (babylonNode.intensity !== 1.0) {\r\n light.intensity = babylonNode.intensity;\r\n }\r\n if (babylonNode.range !== Number.MAX_VALUE) {\r\n light.range = babylonNode.range;\r\n }\r\n\r\n if (lightType === KHRLightsPunctual_LightType.SPOT) {\r\n const babylonSpotLight = babylonNode as SpotLight;\r\n if (babylonSpotLight.angle !== Math.PI / 2.0) {\r\n if (light.spot == null) {\r\n light.spot = {};\r\n }\r\n light.spot.outerConeAngle = babylonSpotLight.angle / 2.0;\r\n }\r\n if (babylonSpotLight.innerAngle !== 0) {\r\n if (light.spot == null) {\r\n light.spot = {};\r\n }\r\n light.spot.innerConeAngle = babylonSpotLight.innerAngle / 2.0;\r\n }\r\n }\r\n\r\n this._lights ||= {\r\n lights: [],\r\n };\r\n\r\n this._lights.lights.push(light);\r\n\r\n const lightReference: IKHRLightsPunctual_LightReference = {\r\n light: this._lights.lights.length - 1,\r\n };\r\n\r\n // Avoid duplicating the Light's parent node if possible.\r\n const parentBabylonNode = babylonNode.parent;\r\n if (parentBabylonNode && parentBabylonNode.getChildren().length == 1) {\r\n const parentNode = this._exporter._nodes[nodeMap[parentBabylonNode.uniqueId]];\r\n if (parentNode) {\r\n const parentTranslation = Vector3.FromArrayToRef(parentNode.translation || [0, 0, 0], 0, TmpVectors.Vector3[0]);\r\n const parentRotation = Quaternion.FromArrayToRef(parentNode.rotation || [0, 0, 0, 1], 0, TmpVectors.Quaternion[0]);\r\n const parentScale = Vector3.FromArrayToRef(parentNode.scale || [1, 1, 1], 0, TmpVectors.Vector3[1]);\r\n const parentMatrix = Matrix.ComposeToRef(parentScale, parentRotation, parentTranslation, TmpVectors.Matrix[0]);\r\n\r\n const translation = Vector3.FromArrayToRef(node.translation || [0, 0, 0], 0, TmpVectors.Vector3[2]);\r\n const rotation = Quaternion.FromArrayToRef(node.rotation || [0, 0, 0, 1], 0, TmpVectors.Quaternion[1]);\r\n const matrix = Matrix.ComposeToRef(Vector3.OneReadOnly, rotation, translation, TmpVectors.Matrix[1]);\r\n\r\n parentMatrix.multiplyToRef(matrix, matrix);\r\n matrix.decompose(parentScale, parentRotation, parentTranslation);\r\n\r\n if (parentTranslation.equalsToFloats(0, 0, 0)) {\r\n delete parentNode.translation;\r\n } else {\r\n parentNode.translation = parentTranslation.asArray();\r\n }\r\n\r\n if (Quaternion.IsIdentity(parentRotation)) {\r\n delete parentNode.rotation;\r\n } else {\r\n parentNode.rotation = parentRotation.asArray();\r\n }\r\n\r\n if (parentScale.equalsToFloats(1, 1, 1)) {\r\n delete parentNode.scale;\r\n } else {\r\n parentNode.scale = parentScale.asArray();\r\n }\r\n\r\n parentNode.extensions ||= {};\r\n parentNode.extensions[NAME] = lightReference;\r\n\r\n // Do not export the original node\r\n resolve(null);\r\n return;\r\n }\r\n }\r\n\r\n node.extensions ||= {};\r\n node.extensions[NAME] = lightReference;\r\n }\r\n }\r\n resolve(node);\r\n });\r\n }\r\n}\r\n\r\n_Exporter.RegisterExtension(NAME, (exporter) => new KHR_lights_punctual(exporter));\r\n","import type { IMaterial, IKHRMaterialsClearcoat } from \"babylonjs-gltf2interface\";\r\nimport type { IGLTFExporterExtensionV2 } from \"../glTFExporterExtension\";\r\nimport { _Exporter } from \"../glTFExporter\";\r\nimport type { Material } from \"core/Materials/material\";\r\nimport { PBRBaseMaterial } from \"core/Materials/PBR/pbrBaseMaterial\";\r\nimport type { BaseTexture } from \"core/Materials/Textures/baseTexture\";\r\n\r\nimport { Tools } from \"core/Misc/tools\";\r\n\r\nconst NAME = \"KHR_materials_clearcoat\";\r\n\r\n/**\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class KHR_materials_clearcoat implements IGLTFExporterExtensionV2 {\r\n /** Name of this extension */\r\n public readonly name = NAME;\r\n\r\n /** Defines whether this extension is enabled */\r\n public enabled = true;\r\n\r\n /** Defines whether this extension is required */\r\n public required = false;\r\n\r\n private _exporter: _Exporter;\r\n\r\n private _wasUsed = false;\r\n\r\n constructor(exporter: _Exporter) {\r\n this._exporter = exporter;\r\n }\r\n\r\n public dispose() {}\r\n\r\n /** @internal */\r\n public get wasUsed() {\r\n return this._wasUsed;\r\n }\r\n\r\n public postExportMaterialAdditionalTextures?(context: string, node: IMaterial, babylonMaterial: Material): BaseTexture[] {\r\n const additionalTextures: BaseTexture[] = [];\r\n if (babylonMaterial instanceof PBRBaseMaterial) {\r\n if (babylonMaterial.clearCoat.isEnabled) {\r\n if (babylonMaterial.clearCoat.texture) {\r\n additionalTextures.push(babylonMaterial.clearCoat.texture);\r\n }\r\n if (!babylonMaterial.clearCoat.useRoughnessFromMainTexture && babylonMaterial.clearCoat.textureRoughness) {\r\n additionalTextures.push(babylonMaterial.clearCoat.textureRoughness);\r\n }\r\n if (babylonMaterial.clearCoat.bumpTexture) {\r\n additionalTextures.push(babylonMaterial.clearCoat.bumpTexture);\r\n }\r\n return additionalTextures;\r\n }\r\n }\r\n\r\n return [];\r\n }\r\n\r\n public postExportMaterialAsync?(context: string, node: IMaterial, babylonMaterial: Material): Promise {\r\n return new Promise((resolve) => {\r\n if (babylonMaterial instanceof PBRBaseMaterial) {\r\n if (!babylonMaterial.clearCoat.isEnabled) {\r\n resolve(node);\r\n return;\r\n }\r\n\r\n this._wasUsed = true;\r\n\r\n node.extensions = node.extensions || {};\r\n\r\n const clearCoatTextureInfo = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.clearCoat.texture);\r\n let clearCoatTextureRoughnessInfo;\r\n if (babylonMaterial.clearCoat.useRoughnessFromMainTexture) {\r\n clearCoatTextureRoughnessInfo = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.clearCoat.texture);\r\n } else {\r\n clearCoatTextureRoughnessInfo = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.clearCoat.textureRoughness);\r\n }\r\n\r\n if (babylonMaterial.clearCoat.isTintEnabled) {\r\n Tools.Warn(`Clear Color tint is not supported for glTF export. Ignoring for: ${babylonMaterial.name}`);\r\n }\r\n\r\n if (babylonMaterial.clearCoat.remapF0OnInterfaceChange) {\r\n Tools.Warn(`Clear Color F0 remapping is not supported for glTF export. Ignoring for: ${babylonMaterial.name}`);\r\n }\r\n\r\n const clearCoatNormalTextureInfo = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.clearCoat.bumpTexture);\r\n\r\n const clearCoatInfo: IKHRMaterialsClearcoat = {\r\n clearcoatFactor: babylonMaterial.clearCoat.intensity,\r\n clearcoatTexture: clearCoatTextureInfo ?? undefined,\r\n clearcoatRoughnessFactor: babylonMaterial.clearCoat.roughness,\r\n clearcoatRoughnessTexture: clearCoatTextureRoughnessInfo ?? undefined,\r\n clearcoatNormalTexture: clearCoatNormalTextureInfo ?? undefined,\r\n hasTextures: () => {\r\n return clearCoatInfo.clearcoatTexture !== null || clearCoatInfo.clearcoatRoughnessTexture !== null || clearCoatInfo.clearcoatRoughnessTexture !== null;\r\n },\r\n };\r\n\r\n node.extensions[NAME] = clearCoatInfo;\r\n }\r\n resolve(node);\r\n });\r\n }\r\n}\r\n\r\n_Exporter.RegisterExtension(NAME, (exporter) => new KHR_materials_clearcoat(exporter));\r\n","import type { IMaterial, IKHRMaterialsIridescence } from \"babylonjs-gltf2interface\";\r\nimport type { IGLTFExporterExtensionV2 } from \"../glTFExporterExtension\";\r\nimport { _Exporter } from \"../glTFExporter\";\r\nimport type { Material } from \"core/Materials/material\";\r\nimport { PBRBaseMaterial } from \"core/Materials/PBR/pbrBaseMaterial\";\r\nimport type { BaseTexture } from \"core/Materials/Textures/baseTexture\";\r\n\r\nconst NAME = \"KHR_materials_iridescence\";\r\n\r\n/**\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class KHR_materials_iridescence implements IGLTFExporterExtensionV2 {\r\n /** Name of this extension */\r\n public readonly name = NAME;\r\n\r\n /** Defines whether this extension is enabled */\r\n public enabled = true;\r\n\r\n /** Defines whether this extension is required */\r\n public required = false;\r\n\r\n private _exporter: _Exporter;\r\n\r\n private _wasUsed = false;\r\n\r\n constructor(exporter: _Exporter) {\r\n this._exporter = exporter;\r\n }\r\n\r\n public dispose() {}\r\n\r\n /** @internal */\r\n public get wasUsed() {\r\n return this._wasUsed;\r\n }\r\n\r\n public postExportMaterialAdditionalTextures?(context: string, node: IMaterial, babylonMaterial: Material): BaseTexture[] {\r\n const additionalTextures: BaseTexture[] = [];\r\n if (babylonMaterial instanceof PBRBaseMaterial) {\r\n if (babylonMaterial.iridescence.isEnabled) {\r\n if (babylonMaterial.iridescence.texture) {\r\n additionalTextures.push(babylonMaterial.iridescence.texture);\r\n }\r\n if (babylonMaterial.iridescence.thicknessTexture && babylonMaterial.iridescence.thicknessTexture !== babylonMaterial.iridescence.texture) {\r\n additionalTextures.push(babylonMaterial.iridescence.thicknessTexture);\r\n }\r\n return additionalTextures;\r\n }\r\n }\r\n\r\n return [];\r\n }\r\n\r\n public postExportMaterialAsync?(context: string, node: IMaterial, babylonMaterial: Material): Promise {\r\n return new Promise((resolve) => {\r\n if (babylonMaterial instanceof PBRBaseMaterial) {\r\n if (!babylonMaterial.iridescence.isEnabled) {\r\n resolve(node);\r\n return;\r\n }\r\n\r\n this._wasUsed = true;\r\n\r\n node.extensions = node.extensions || {};\r\n\r\n const iridescenceTextureInfo = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.iridescence.texture);\r\n const iridescenceThicknessTextureInfo = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.iridescence.thicknessTexture);\r\n\r\n const iridescenceInfo: IKHRMaterialsIridescence = {\r\n iridescenceFactor: babylonMaterial.iridescence.intensity,\r\n iridescenceIor: babylonMaterial.iridescence.indexOfRefraction,\r\n iridescenceThicknessMinimum: babylonMaterial.iridescence.minimumThickness,\r\n iridescenceThicknessMaximum: babylonMaterial.iridescence.maximumThickness,\r\n\r\n iridescenceTexture: iridescenceTextureInfo ?? undefined,\r\n iridescenceThicknessTexture: iridescenceThicknessTextureInfo ?? undefined,\r\n hasTextures: () => {\r\n return iridescenceInfo.iridescenceTexture !== null || iridescenceInfo.iridescenceThicknessTexture !== null;\r\n },\r\n };\r\n\r\n node.extensions[NAME] = iridescenceInfo;\r\n }\r\n resolve(node);\r\n });\r\n }\r\n}\r\n\r\n_Exporter.RegisterExtension(NAME, (exporter) => new KHR_materials_iridescence(exporter));\r\n","import type { IMaterial, IKHRMaterialsAnisotropy } from \"babylonjs-gltf2interface\";\r\nimport type { IGLTFExporterExtensionV2 } from \"../glTFExporterExtension\";\r\nimport { _Exporter } from \"../glTFExporter\";\r\nimport type { Material } from \"core/Materials/material\";\r\nimport { PBRBaseMaterial } from \"core/Materials/PBR/pbrBaseMaterial\";\r\nimport type { BaseTexture } from \"core/Materials/Textures/baseTexture\";\r\n\r\nconst NAME = \"KHR_materials_anisotropy\";\r\n\r\n/**\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class KHR_materials_anisotropy implements IGLTFExporterExtensionV2 {\r\n /** Name of this extension */\r\n public readonly name = NAME;\r\n\r\n /** Defines whether this extension is enabled */\r\n public enabled = true;\r\n\r\n /** Defines whether this extension is required */\r\n public required = false;\r\n\r\n private _exporter: _Exporter;\r\n\r\n private _wasUsed = false;\r\n\r\n constructor(exporter: _Exporter) {\r\n this._exporter = exporter;\r\n }\r\n\r\n public dispose() {}\r\n\r\n /** @internal */\r\n public get wasUsed() {\r\n return this._wasUsed;\r\n }\r\n\r\n public postExportMaterialAdditionalTextures?(context: string, node: IMaterial, babylonMaterial: Material): BaseTexture[] {\r\n const additionalTextures: BaseTexture[] = [];\r\n if (babylonMaterial instanceof PBRBaseMaterial) {\r\n if (babylonMaterial.anisotropy.isEnabled && !babylonMaterial.anisotropy.legacy) {\r\n if (babylonMaterial.anisotropy.texture) {\r\n additionalTextures.push(babylonMaterial.anisotropy.texture);\r\n }\r\n return additionalTextures;\r\n }\r\n }\r\n\r\n return [];\r\n }\r\n\r\n public postExportMaterialAsync?(context: string, node: IMaterial, babylonMaterial: Material): Promise {\r\n return new Promise((resolve) => {\r\n if (babylonMaterial instanceof PBRBaseMaterial) {\r\n if (!babylonMaterial.anisotropy.isEnabled || babylonMaterial.anisotropy.legacy) {\r\n resolve(node);\r\n return;\r\n }\r\n\r\n this._wasUsed = true;\r\n\r\n node.extensions = node.extensions || {};\r\n\r\n const anisotropyTextureInfo = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.anisotropy.texture);\r\n\r\n const anisotropyInfo: IKHRMaterialsAnisotropy = {\r\n anisotropyStrength: babylonMaterial.anisotropy.intensity,\r\n anisotropyRotation: babylonMaterial.anisotropy.angle,\r\n anisotropyTexture: anisotropyTextureInfo ?? undefined,\r\n hasTextures: () => {\r\n return anisotropyInfo.anisotropyTexture !== null;\r\n },\r\n };\r\n\r\n node.extensions[NAME] = anisotropyInfo;\r\n }\r\n resolve(node);\r\n });\r\n }\r\n}\r\n\r\n_Exporter.RegisterExtension(NAME, (exporter) => new KHR_materials_anisotropy(exporter));\r\n","import type { IMaterial, IKHRMaterialsSheen } from \"babylonjs-gltf2interface\";\r\nimport type { IGLTFExporterExtensionV2 } from \"../glTFExporterExtension\";\r\nimport { _Exporter } from \"../glTFExporter\";\r\nimport type { Material } from \"core/Materials/material\";\r\nimport { PBRMaterial } from \"core/Materials/PBR/pbrMaterial\";\r\nimport type { BaseTexture } from \"core/Materials/Textures/baseTexture\";\r\n\r\nconst NAME = \"KHR_materials_sheen\";\r\n\r\n/**\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class KHR_materials_sheen implements IGLTFExporterExtensionV2 {\r\n /** Name of this extension */\r\n public readonly name = NAME;\r\n\r\n /** Defines whether this extension is enabled */\r\n public enabled = true;\r\n\r\n /** Defines whether this extension is required */\r\n public required = false;\r\n\r\n private _wasUsed = false;\r\n\r\n private _exporter: _Exporter;\r\n\r\n constructor(exporter: _Exporter) {\r\n this._exporter = exporter;\r\n }\r\n\r\n public dispose() {}\r\n\r\n /** @internal */\r\n public get wasUsed() {\r\n return this._wasUsed;\r\n }\r\n\r\n public postExportMaterialAdditionalTextures(context: string, node: IMaterial, babylonMaterial: Material): BaseTexture[] {\r\n if (babylonMaterial instanceof PBRMaterial) {\r\n if (babylonMaterial.sheen.isEnabled && babylonMaterial.sheen.texture) {\r\n return [babylonMaterial.sheen.texture];\r\n }\r\n }\r\n\r\n return [];\r\n }\r\n\r\n public postExportMaterialAsync(context: string, node: IMaterial, babylonMaterial: Material): Promise {\r\n return new Promise((resolve) => {\r\n if (babylonMaterial instanceof PBRMaterial) {\r\n if (!babylonMaterial.sheen.isEnabled) {\r\n resolve(node);\r\n return;\r\n }\r\n\r\n this._wasUsed = true;\r\n\r\n if (node.extensions == null) {\r\n node.extensions = {};\r\n }\r\n const sheenInfo: IKHRMaterialsSheen = {\r\n sheenColorFactor: babylonMaterial.sheen.color.asArray(),\r\n sheenRoughnessFactor: babylonMaterial.sheen.roughness ?? 0,\r\n hasTextures: () => {\r\n return sheenInfo.sheenColorTexture !== null || sheenInfo.sheenRoughnessTexture !== null;\r\n },\r\n };\r\n\r\n if (babylonMaterial.sheen.texture) {\r\n sheenInfo.sheenColorTexture = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.sheen.texture) ?? undefined;\r\n }\r\n\r\n if (babylonMaterial.sheen.textureRoughness && !babylonMaterial.sheen.useRoughnessFromMainTexture) {\r\n sheenInfo.sheenRoughnessTexture = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.sheen.textureRoughness) ?? undefined;\r\n } else if (babylonMaterial.sheen.texture && babylonMaterial.sheen.useRoughnessFromMainTexture) {\r\n sheenInfo.sheenRoughnessTexture = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.sheen.texture) ?? undefined;\r\n }\r\n\r\n node.extensions[NAME] = sheenInfo;\r\n }\r\n resolve(node);\r\n });\r\n }\r\n}\r\n\r\n_Exporter.RegisterExtension(NAME, (exporter) => new KHR_materials_sheen(exporter));\r\n","import type { IMaterial } from \"babylonjs-gltf2interface\";\r\nimport type { IGLTFExporterExtensionV2 } from \"../glTFExporterExtension\";\r\nimport { _Exporter } from \"../glTFExporter\";\r\nimport type { Material } from \"core/Materials/material\";\r\nimport { PBRMaterial } from \"core/Materials/PBR/pbrMaterial\";\r\nimport { StandardMaterial } from \"core/Materials/standardMaterial\";\r\n\r\nconst NAME = \"KHR_materials_unlit\";\r\n\r\n/**\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class KHR_materials_unlit implements IGLTFExporterExtensionV2 {\r\n /** Name of this extension */\r\n public readonly name = NAME;\r\n\r\n /** Defines whether this extension is enabled */\r\n public enabled = true;\r\n\r\n /** Defines whether this extension is required */\r\n public required = false;\r\n\r\n private _wasUsed = false;\r\n\r\n constructor() {}\r\n\r\n /** @internal */\r\n public get wasUsed() {\r\n return this._wasUsed;\r\n }\r\n\r\n public dispose() {}\r\n\r\n public postExportMaterialAsync?(context: string, node: IMaterial, babylonMaterial: Material): Promise {\r\n return new Promise((resolve) => {\r\n let unlitMaterial = false;\r\n\r\n if (babylonMaterial instanceof PBRMaterial) {\r\n unlitMaterial = babylonMaterial.unlit;\r\n } else if (babylonMaterial instanceof StandardMaterial) {\r\n unlitMaterial = babylonMaterial.disableLighting;\r\n }\r\n\r\n if (unlitMaterial) {\r\n this._wasUsed = true;\r\n\r\n if (node.extensions == null) {\r\n node.extensions = {};\r\n }\r\n\r\n node.extensions[NAME] = {};\r\n }\r\n\r\n resolve(node);\r\n });\r\n }\r\n}\r\n\r\n_Exporter.RegisterExtension(NAME, () => new KHR_materials_unlit());\r\n","import type { IMaterial, IKHRMaterialsIor } from \"babylonjs-gltf2interface\";\r\nimport type { IGLTFExporterExtensionV2 } from \"../glTFExporterExtension\";\r\nimport { _Exporter } from \"../glTFExporter\";\r\nimport type { Material } from \"core/Materials/material\";\r\nimport { PBRMaterial } from \"core/Materials/PBR/pbrMaterial\";\r\n\r\nconst NAME = \"KHR_materials_ior\";\r\n\r\n/**\r\n * [Specification](https://github.com/KhronosGroup/glTF/blob/main/extensions/2.0/Khronos/KHR_materials_ior/README.md)\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class KHR_materials_ior implements IGLTFExporterExtensionV2 {\r\n /** Name of this extension */\r\n public readonly name = NAME;\r\n\r\n /** Defines whether this extension is enabled */\r\n public enabled = true;\r\n\r\n /** Defines whether this extension is required */\r\n public required = false;\r\n\r\n private _wasUsed = false;\r\n\r\n constructor() {}\r\n\r\n public dispose() {}\r\n\r\n /** @internal */\r\n public get wasUsed() {\r\n return this._wasUsed;\r\n }\r\n\r\n private _isExtensionEnabled(mat: PBRMaterial): boolean {\r\n // This extension must not be used on a material that also uses KHR_materials_unlit\r\n if (mat.unlit) {\r\n return false;\r\n }\r\n return mat.indexOfRefraction != undefined && mat.indexOfRefraction != 1.5; // 1.5 is normative default value.\r\n }\r\n\r\n public postExportMaterialAsync?(context: string, node: IMaterial, babylonMaterial: Material): Promise {\r\n return new Promise((resolve) => {\r\n if (babylonMaterial instanceof PBRMaterial && this._isExtensionEnabled(babylonMaterial)) {\r\n this._wasUsed = true;\r\n\r\n const iorInfo: IKHRMaterialsIor = {\r\n ior: babylonMaterial.indexOfRefraction,\r\n };\r\n node.extensions = node.extensions || {};\r\n node.extensions[NAME] = iorInfo;\r\n }\r\n resolve(node);\r\n });\r\n }\r\n}\r\n\r\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n_Exporter.RegisterExtension(NAME, (exporter) => new KHR_materials_ior());\r\n","import type { IMaterial, IKHRMaterialsSpecular } from \"babylonjs-gltf2interface\";\r\nimport type { IGLTFExporterExtensionV2 } from \"../glTFExporterExtension\";\r\nimport { _Exporter } from \"../glTFExporter\";\r\nimport type { Material } from \"core/Materials/material\";\r\nimport { PBRMaterial } from \"core/Materials/PBR/pbrMaterial\";\r\nimport type { BaseTexture } from \"core/Materials/Textures/baseTexture\";\r\n\r\nconst NAME = \"KHR_materials_specular\";\r\n\r\n/**\r\n * [Specification](https://github.com/KhronosGroup/glTF/blob/main/extensions/2.0/Khronos/KHR_materials_specular/README.md)\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class KHR_materials_specular implements IGLTFExporterExtensionV2 {\r\n /** Name of this extension */\r\n public readonly name = NAME;\r\n\r\n /** Defines whether this extension is enabled */\r\n public enabled = true;\r\n\r\n /** Defines whether this extension is required */\r\n public required = false;\r\n\r\n private _exporter: _Exporter;\r\n\r\n private _wasUsed = false;\r\n\r\n constructor(exporter: _Exporter) {\r\n this._exporter = exporter;\r\n }\r\n\r\n public dispose() {}\r\n\r\n /** @internal */\r\n public get wasUsed() {\r\n return this._wasUsed;\r\n }\r\n\r\n public postExportMaterialAdditionalTextures?(context: string, node: IMaterial, babylonMaterial: Material): BaseTexture[] {\r\n const additionalTextures: BaseTexture[] = [];\r\n\r\n if (babylonMaterial instanceof PBRMaterial) {\r\n if (this._isExtensionEnabled(babylonMaterial)) {\r\n if (babylonMaterial.metallicReflectanceTexture) {\r\n additionalTextures.push(babylonMaterial.metallicReflectanceTexture);\r\n }\r\n if (babylonMaterial.reflectanceTexture) {\r\n additionalTextures.push(babylonMaterial.reflectanceTexture);\r\n }\r\n return additionalTextures;\r\n }\r\n }\r\n\r\n return additionalTextures;\r\n }\r\n\r\n private _isExtensionEnabled(mat: PBRMaterial): boolean {\r\n // This extension must not be used on a material that also uses KHR_materials_unlit\r\n if (mat.unlit) {\r\n return false;\r\n }\r\n return (\r\n (mat.metallicF0Factor != undefined && mat.metallicF0Factor != 1.0) ||\r\n (mat.metallicReflectanceColor != undefined && !mat.metallicReflectanceColor.equalsFloats(1.0, 1.0, 1.0)) ||\r\n this._hasTexturesExtension(mat)\r\n );\r\n }\r\n\r\n private _hasTexturesExtension(mat: PBRMaterial): boolean {\r\n return mat.metallicReflectanceTexture != null || mat.reflectanceTexture != null;\r\n }\r\n\r\n public postExportMaterialAsync?(context: string, node: IMaterial, babylonMaterial: Material): Promise {\r\n return new Promise((resolve) => {\r\n if (babylonMaterial instanceof PBRMaterial && this._isExtensionEnabled(babylonMaterial)) {\r\n this._wasUsed = true;\r\n\r\n node.extensions = node.extensions || {};\r\n\r\n const metallicReflectanceTexture = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.metallicReflectanceTexture) ?? undefined;\r\n const reflectanceTexture = this._exporter._glTFMaterialExporter._getTextureInfo(babylonMaterial.reflectanceTexture) ?? undefined;\r\n const metallicF0Factor = babylonMaterial.metallicF0Factor == 1.0 ? undefined : babylonMaterial.metallicF0Factor;\r\n const metallicReflectanceColor = babylonMaterial.metallicReflectanceColor.equalsFloats(1.0, 1.0, 1.0)\r\n ? undefined\r\n : babylonMaterial.metallicReflectanceColor.asArray();\r\n\r\n const specularInfo: IKHRMaterialsSpecular = {\r\n specularFactor: metallicF0Factor,\r\n specularTexture: metallicReflectanceTexture,\r\n specularColorFactor: metallicReflectanceColor,\r\n specularColorTexture: reflectanceTexture,\r\n hasTextures: () => {\r\n return this._hasTexturesExtension(babylonMaterial);\r\n },\r\n };\r\n node.extensions[NAME] = specularInfo;\r\n }\r\n resolve(node);\r\n });\r\n }\r\n}\r\n\r\n_Exporter.RegisterExtension(NAME, (exporter) => new KHR_materials_specular(exporter));\r\n","import type { IMaterial, IKHRMaterialsVolume } from \"babylonjs-gltf2interface\";\r\nimport type { IGLTFExporterExtensionV2 } from \"../glTFExporterExtension\";\r\nimport { _Exporter } from \"../glTFExporter\";\r\nimport type { Material } from \"core/Materials/material\";\r\nimport { PBRMaterial } from \"core/Materials/PBR/pbrMaterial\";\r\nimport type { BaseTexture } from \"core/Materials/Textures/baseTexture\";\r\nimport { Color3 } from \"core/Maths/math.color\";\r\n\r\nconst NAME = \"KHR_materials_volume\";\r\n\r\n/**\r\n * [Specification](https://github.com/KhronosGroup/glTF/blob/main/extensions/2.0/Khronos/KHR_materials_volume/README.md)\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class KHR_materials_volume implements IGLTFExporterExtensionV2 {\r\n /** Name of this extension */\r\n public readonly name = NAME;\r\n\r\n /** Defines whether this extension is enabled */\r\n public enabled = true;\r\n\r\n /** Defines whether this extension is required */\r\n public required = false;\r\n\r\n private _exporter: _Exporter;\r\n\r\n private _wasUsed = false;\r\n\r\n constructor(exporter: _Exporter) {\r\n this._exporter = exporter;\r\n }\r\n\r\n public dispose() {}\r\n\r\n /** @internal */\r\n public get wasUsed() {\r\n return this._wasUsed;\r\n }\r\n\r\n public postExportMaterialAdditionalTextures?(context: string, node: IMaterial, babylonMaterial: Material): BaseTexture[] {\r\n const additionalTextures: BaseTexture[] = [];\r\n\r\n if (babylonMaterial instanceof PBRMaterial) {\r\n if (this._isExtensionEnabled(babylonMaterial)) {\r\n if (babylonMaterial.subSurface.thicknessTexture) {\r\n additionalTextures.push(babylonMaterial.subSurface.thicknessTexture);\r\n }\r\n return additionalTextures;\r\n }\r\n }\r\n\r\n return additionalTextures;\r\n }\r\n\r\n private _isExtensionEnabled(mat: PBRMaterial): boolean {\r\n // This extension must not be used on a material that also uses KHR_materials_unlit\r\n if (mat.unlit) {\r\n return false;\r\n }\r\n const subs = mat.subSurface;\r\n // this extension requires either the KHR_materials_transmission or KHR_materials_translucency extensions.\r\n if (!subs.isRefractionEnabled && !subs.isTranslucencyEnabled) {\r\n return false;\r\n }\r\n return (\r\n (subs.maximumThickness != undefined && subs.maximumThickness != 0) ||\r\n (subs.tintColorAtDistance != undefined && subs.tintColorAtDistance != Number.POSITIVE_INFINITY) ||\r\n (subs.tintColor != undefined && subs.tintColor != Color3.White()) ||\r\n this._hasTexturesExtension(mat)\r\n );\r\n }\r\n\r\n private _hasTexturesExtension(mat: PBRMaterial): boolean {\r\n return mat.subSurface.thicknessTexture != null;\r\n }\r\n\r\n public postExportMaterialAsync?(context: string, node: IMaterial, babylonMaterial: Material): Promise {\r\n return new Promise((resolve) => {\r\n if (babylonMaterial instanceof PBRMaterial && this._isExtensionEnabled(babylonMaterial)) {\r\n this._wasUsed = true;\r\n\r\n const subs = babylonMaterial.subSurface;\r\n const thicknessFactor = subs.maximumThickness == 0 ? undefined : subs.maximumThickness;\r\n const thicknessTexture = this._exporter._glTFMaterialExporter._getTextureInfo(subs.thicknessTexture) ?? undefined;\r\n const attenuationDistance = subs.tintColorAtDistance == Number.POSITIVE_INFINITY ? undefined : subs.tintColorAtDistance;\r\n const attenuationColor = subs.tintColor.equalsFloats(1.0, 1.0, 1.0) ? undefined : subs.tintColor.asArray();\r\n\r\n const volumeInfo: IKHRMaterialsVolume = {\r\n thicknessFactor: thicknessFactor,\r\n thicknessTexture: thicknessTexture,\r\n attenuationDistance: attenuationDistance,\r\n attenuationColor: attenuationColor,\r\n hasTextures: () => {\r\n return this._hasTexturesExtension(babylonMaterial);\r\n },\r\n };\r\n node.extensions = node.extensions || {};\r\n node.extensions[NAME] = volumeInfo;\r\n }\r\n resolve(node);\r\n });\r\n }\r\n}\r\n\r\n_Exporter.RegisterExtension(NAME, (exporter) => new KHR_materials_volume(exporter));\r\n","import type { IMaterial, IKHRMaterialsTransmission } from \"babylonjs-gltf2interface\";\r\nimport type { IGLTFExporterExtensionV2 } from \"../glTFExporterExtension\";\r\nimport { _Exporter } from \"../glTFExporter\";\r\nimport type { Material } from \"core/Materials/material\";\r\nimport { PBRMaterial } from \"core/Materials/PBR/pbrMaterial\";\r\nimport type { BaseTexture } from \"core/Materials/Textures/baseTexture\";\r\n\r\nconst NAME = \"KHR_materials_transmission\";\r\n\r\n/**\r\n * [Specification](https://github.com/KhronosGroup/glTF/blob/main/extensions/2.0/Khronos/KHR_materials_transmission/README.md)\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class KHR_materials_transmission implements IGLTFExporterExtensionV2 {\r\n /** Name of this extension */\r\n public readonly name = NAME;\r\n\r\n /** Defines whether this extension is enabled */\r\n public enabled = true;\r\n\r\n /** Defines whether this extension is required */\r\n public required = false;\r\n\r\n private _exporter: _Exporter;\r\n\r\n private _wasUsed = false;\r\n\r\n constructor(exporter: _Exporter) {\r\n this._exporter = exporter;\r\n }\r\n\r\n public dispose() {}\r\n\r\n /** @internal */\r\n public get wasUsed() {\r\n return this._wasUsed;\r\n }\r\n\r\n public postExportMaterialAdditionalTextures?(context: string, node: IMaterial, babylonMaterial: Material): BaseTexture[] {\r\n const additionalTextures: BaseTexture[] = [];\r\n\r\n if (babylonMaterial instanceof PBRMaterial) {\r\n if (this._isExtensionEnabled(babylonMaterial)) {\r\n if (babylonMaterial.subSurface.thicknessTexture) {\r\n additionalTextures.push(babylonMaterial.subSurface.thicknessTexture);\r\n }\r\n return additionalTextures;\r\n }\r\n }\r\n\r\n return additionalTextures;\r\n }\r\n\r\n private _isExtensionEnabled(mat: PBRMaterial): boolean {\r\n // This extension must not be used on a material that also uses KHR_materials_unlit\r\n if (mat.unlit) {\r\n return false;\r\n }\r\n const subs = mat.subSurface;\r\n return (subs.isRefractionEnabled && subs.refractionIntensity != undefined && subs.refractionIntensity != 0) || this._hasTexturesExtension(mat);\r\n }\r\n\r\n private _hasTexturesExtension(mat: PBRMaterial): boolean {\r\n return mat.subSurface.refractionIntensityTexture != null;\r\n }\r\n\r\n public postExportMaterialAsync?(context: string, node: IMaterial, babylonMaterial: Material): Promise {\r\n return new Promise((resolve) => {\r\n if (babylonMaterial instanceof PBRMaterial && this._isExtensionEnabled(babylonMaterial)) {\r\n this._wasUsed = true;\r\n\r\n const subs = babylonMaterial.subSurface;\r\n const transmissionFactor = subs.refractionIntensity === 0 ? undefined : subs.refractionIntensity;\r\n\r\n const transmissionTexture = this._exporter._glTFMaterialExporter._getTextureInfo(subs.refractionIntensityTexture) ?? undefined;\r\n\r\n const volumeInfo: IKHRMaterialsTransmission = {\r\n transmissionFactor: transmissionFactor,\r\n transmissionTexture: transmissionTexture,\r\n hasTextures: () => {\r\n return this._hasTexturesExtension(babylonMaterial);\r\n },\r\n };\r\n node.extensions = node.extensions || {};\r\n node.extensions[NAME] = volumeInfo;\r\n }\r\n resolve(node);\r\n });\r\n }\r\n}\r\n\r\n_Exporter.RegisterExtension(NAME, (exporter) => new KHR_materials_transmission(exporter));\r\n","import type { Nullable, DeepImmutableObject } from \"../types\";\r\nimport { Mesh } from \"../Meshes/mesh\";\r\nimport { VertexBuffer, Buffer } from \"../Buffers/buffer\";\r\nimport { Matrix, Vector3, TmpVectors } from \"../Maths/math.vector\";\r\nimport { Logger } from \"../Misc/logger\";\r\nimport { BoundingInfo } from \"core/Culling/boundingInfo\";\r\n\r\ndeclare module \"./mesh\" {\r\n export interface Mesh {\r\n /**\r\n * Gets or sets a boolean defining if we want picking to pick thin instances as well\r\n */\r\n thinInstanceEnablePicking: boolean;\r\n /**\r\n * Creates a new thin instance\r\n * @param matrix the matrix or array of matrices (position, rotation, scale) of the thin instance(s) to create\r\n * @param refresh true to refresh the underlying gpu buffer (default: true). If you do multiple calls to this method in a row, set refresh to true only for the last call to save performance\r\n * @returns the thin instance index number. If you pass an array of matrices, other instance indexes are index+1, index+2, etc\r\n */\r\n thinInstanceAdd(matrix: DeepImmutableObject | Array>, refresh?: boolean): number;\r\n\r\n /**\r\n * Adds the transformation (matrix) of the current mesh as a thin instance\r\n * @param refresh true to refresh the underlying gpu buffer (default: true). If you do multiple calls to this method in a row, set refresh to true only for the last call to save performance\r\n * @returns the thin instance index number\r\n */\r\n thinInstanceAddSelf(refresh?: boolean): number;\r\n\r\n /**\r\n * Registers a custom attribute to be used with thin instances\r\n * @param kind name of the attribute\r\n * @param stride size in floats of the attribute\r\n */\r\n thinInstanceRegisterAttribute(kind: string, stride: number): void;\r\n\r\n /**\r\n * Sets the matrix of a thin instance\r\n * @param index index of the thin instance\r\n * @param matrix matrix to set\r\n * @param refresh true to refresh the underlying gpu buffer (default: true). If you do multiple calls to this method in a row, set refresh to true only for the last call to save performance\r\n */\r\n thinInstanceSetMatrixAt(index: number, matrix: DeepImmutableObject, refresh?: boolean): void;\r\n\r\n /**\r\n * Sets the value of a custom attribute for a thin instance\r\n * @param kind name of the attribute\r\n * @param index index of the thin instance\r\n * @param value value to set\r\n * @param refresh true to refresh the underlying gpu buffer (default: true). If you do multiple calls to this method in a row, set refresh to true only for the last call to save performance\r\n */\r\n thinInstanceSetAttributeAt(kind: string, index: number, value: Array, refresh?: boolean): void;\r\n\r\n /**\r\n * Gets / sets the number of thin instances to display. Note that you can't set a number higher than what the underlying buffer can handle.\r\n */\r\n thinInstanceCount: number;\r\n\r\n /**\r\n * Sets a buffer to be used with thin instances. This method is a faster way to setup multiple instances than calling thinInstanceAdd repeatedly\r\n * @param kind name of the attribute. Use \"matrix\" to setup the buffer of matrices\r\n * @param buffer buffer to set\r\n * @param stride size in floats of each value of the buffer\r\n * @param staticBuffer indicates that the buffer is static, so that you won't change it after it is set (better performances - false by default)\r\n */\r\n thinInstanceSetBuffer(kind: string, buffer: Nullable, stride?: number, staticBuffer?: boolean): void;\r\n\r\n /**\r\n * Gets the list of world matrices\r\n * @returns an array containing all the world matrices from the thin instances\r\n */\r\n thinInstanceGetWorldMatrices(): Matrix[];\r\n\r\n /**\r\n * Synchronize the gpu buffers with a thin instance buffer. Call this method if you update later on the buffers passed to thinInstanceSetBuffer\r\n * @param kind name of the attribute to update. Use \"matrix\" to update the buffer of matrices\r\n */\r\n thinInstanceBufferUpdated(kind: string): void;\r\n\r\n /**\r\n * Applies a partial update to a buffer directly on the GPU\r\n * Note that the buffer located on the CPU is NOT updated! It's up to you to update it (or not) with the same data you pass to this method\r\n * @param kind name of the attribute to update. Use \"matrix\" to update the buffer of matrices\r\n * @param data the data to set in the GPU buffer\r\n * @param offset the offset in the GPU buffer where to update the data\r\n */\r\n thinInstancePartialBufferUpdate(kind: string, data: Float32Array, offset: number): void;\r\n\r\n /**\r\n * Refreshes the bounding info, taking into account all the thin instances defined\r\n * @param forceRefreshParentInfo true to force recomputing the mesh bounding info and use it to compute the aggregated bounding info\r\n * @param applySkeleton defines whether to apply the skeleton before computing the bounding info\r\n * @param applyMorph defines whether to apply the morph target before computing the bounding info\r\n */\r\n thinInstanceRefreshBoundingInfo(forceRefreshParentInfo?: boolean, applySkeleton?: boolean, applyMorph?: boolean): void;\r\n\r\n /** @internal */\r\n _thinInstanceInitializeUserStorage(): void;\r\n\r\n /** @internal */\r\n _thinInstanceUpdateBufferSize(kind: string, numInstances?: number): void;\r\n\r\n /** @internal */\r\n _thinInstanceCreateMatrixBuffer(kind: string, buffer: Nullable, staticBuffer: boolean): Buffer;\r\n\r\n /** @internal */\r\n _userThinInstanceBuffersStorage: {\r\n data: { [key: string]: Float32Array };\r\n sizes: { [key: string]: number };\r\n vertexBuffers: { [key: string]: Nullable };\r\n strides: { [key: string]: number };\r\n };\r\n }\r\n}\r\n\r\nMesh.prototype.thinInstanceAdd = function (matrix: DeepImmutableObject | Array>, refresh: boolean = true): number {\r\n if (!this.getScene().getEngine().getCaps().instancedArrays) {\r\n Logger.Error(\"Thin Instances are not supported on this device as Instanced Array extension not supported\");\r\n return -1;\r\n }\r\n\r\n this._thinInstanceUpdateBufferSize(\"matrix\", Array.isArray(matrix) ? matrix.length : 1);\r\n\r\n const index = this._thinInstanceDataStorage.instancesCount;\r\n\r\n if (Array.isArray(matrix)) {\r\n for (let i = 0; i < matrix.length; ++i) {\r\n this.thinInstanceSetMatrixAt(this._thinInstanceDataStorage.instancesCount++, matrix[i], i === matrix.length - 1 && refresh);\r\n }\r\n } else {\r\n this.thinInstanceSetMatrixAt(this._thinInstanceDataStorage.instancesCount++, matrix, refresh);\r\n }\r\n\r\n return index;\r\n};\r\n\r\nMesh.prototype.thinInstanceAddSelf = function (refresh: boolean = true): number {\r\n return this.thinInstanceAdd(Matrix.IdentityReadOnly, refresh);\r\n};\r\n\r\nMesh.prototype.thinInstanceRegisterAttribute = function (kind: string, stride: number): void {\r\n // preserve backward compatibility\r\n if (kind === VertexBuffer.ColorKind) {\r\n kind = VertexBuffer.ColorInstanceKind;\r\n }\r\n\r\n this.removeVerticesData(kind);\r\n\r\n this._thinInstanceInitializeUserStorage();\r\n\r\n this._userThinInstanceBuffersStorage.strides[kind] = stride;\r\n this._userThinInstanceBuffersStorage.sizes[kind] = stride * Math.max(32, this._thinInstanceDataStorage.instancesCount); // Initial size\r\n this._userThinInstanceBuffersStorage.data[kind] = new Float32Array(this._userThinInstanceBuffersStorage.sizes[kind]);\r\n this._userThinInstanceBuffersStorage.vertexBuffers[kind] = new VertexBuffer(this.getEngine(), this._userThinInstanceBuffersStorage.data[kind], kind, true, false, stride, true);\r\n\r\n this.setVerticesBuffer(this._userThinInstanceBuffersStorage.vertexBuffers[kind]!);\r\n};\r\n\r\nMesh.prototype.thinInstanceSetMatrixAt = function (index: number, matrix: DeepImmutableObject, refresh: boolean = true): boolean {\r\n if (!this._thinInstanceDataStorage.matrixData || index >= this._thinInstanceDataStorage.instancesCount) {\r\n return false;\r\n }\r\n\r\n const matrixData = this._thinInstanceDataStorage.matrixData;\r\n\r\n matrix.copyToArray(matrixData, index * 16);\r\n\r\n if (this._thinInstanceDataStorage.worldMatrices) {\r\n this._thinInstanceDataStorage.worldMatrices[index] = matrix as Matrix;\r\n }\r\n\r\n if (refresh) {\r\n this.thinInstanceBufferUpdated(\"matrix\");\r\n\r\n if (!this.doNotSyncBoundingInfo) {\r\n this.thinInstanceRefreshBoundingInfo(false);\r\n }\r\n }\r\n\r\n return true;\r\n};\r\n\r\nMesh.prototype.thinInstanceSetAttributeAt = function (kind: string, index: number, value: Array, refresh: boolean = true): boolean {\r\n // preserve backward compatibility\r\n if (kind === VertexBuffer.ColorKind) {\r\n kind = VertexBuffer.ColorInstanceKind;\r\n }\r\n\r\n if (!this._userThinInstanceBuffersStorage || !this._userThinInstanceBuffersStorage.data[kind] || index >= this._thinInstanceDataStorage.instancesCount) {\r\n return false;\r\n }\r\n\r\n this._thinInstanceUpdateBufferSize(kind, 0); // make sur the buffer for the kind attribute is big enough\r\n\r\n this._userThinInstanceBuffersStorage.data[kind].set(value, index * this._userThinInstanceBuffersStorage.strides[kind]);\r\n\r\n if (refresh) {\r\n this.thinInstanceBufferUpdated(kind);\r\n }\r\n\r\n return true;\r\n};\r\n\r\nObject.defineProperty(Mesh.prototype, \"thinInstanceCount\", {\r\n get: function (this: Mesh) {\r\n return this._thinInstanceDataStorage.instancesCount;\r\n },\r\n set: function (this: Mesh, value: number) {\r\n const matrixData = this._thinInstanceDataStorage.matrixData ?? this.source?._thinInstanceDataStorage.matrixData;\r\n const numMaxInstances = matrixData ? matrixData.length / 16 : 0;\r\n\r\n if (value <= numMaxInstances) {\r\n this._thinInstanceDataStorage.instancesCount = value;\r\n }\r\n },\r\n enumerable: true,\r\n configurable: true,\r\n});\r\n\r\nMesh.prototype._thinInstanceCreateMatrixBuffer = function (kind: string, buffer: Float32Array, staticBuffer: boolean = false): Buffer {\r\n // preserve backward compatibility\r\n if (kind === VertexBuffer.ColorKind) {\r\n kind = VertexBuffer.ColorInstanceKind;\r\n }\r\n\r\n const matrixBuffer = new Buffer(this.getEngine(), buffer, !staticBuffer, 16, false, true);\r\n\r\n for (let i = 0; i < 4; i++) {\r\n this.setVerticesBuffer(matrixBuffer.createVertexBuffer(kind + i, i * 4, 4));\r\n }\r\n\r\n return matrixBuffer;\r\n};\r\n\r\nMesh.prototype.thinInstanceSetBuffer = function (kind: string, buffer: Nullable, stride: number = 0, staticBuffer: boolean = false): void {\r\n stride = stride || 16;\r\n\r\n if (kind === \"matrix\") {\r\n this._thinInstanceDataStorage.matrixBuffer?.dispose();\r\n this._thinInstanceDataStorage.matrixBuffer = null;\r\n this._thinInstanceDataStorage.matrixBufferSize = buffer ? buffer.length : 32 * stride;\r\n this._thinInstanceDataStorage.matrixData = buffer;\r\n this._thinInstanceDataStorage.worldMatrices = null;\r\n\r\n if (buffer !== null) {\r\n this._thinInstanceDataStorage.instancesCount = buffer.length / stride;\r\n this._thinInstanceDataStorage.matrixBuffer = this._thinInstanceCreateMatrixBuffer(\"world\", buffer, staticBuffer);\r\n\r\n if (!this.doNotSyncBoundingInfo) {\r\n this.thinInstanceRefreshBoundingInfo(false);\r\n }\r\n } else {\r\n this._thinInstanceDataStorage.instancesCount = 0;\r\n if (!this.doNotSyncBoundingInfo) {\r\n // mesh has no more thin instances, so need to recompute the bounding box because it's the regular mesh that will now be displayed\r\n this.refreshBoundingInfo();\r\n }\r\n }\r\n } else if (kind === \"previousMatrix\") {\r\n this._thinInstanceDataStorage.previousMatrixBuffer?.dispose();\r\n this._thinInstanceDataStorage.previousMatrixBuffer = null;\r\n this._thinInstanceDataStorage.previousMatrixData = buffer;\r\n if (buffer !== null) {\r\n this._thinInstanceDataStorage.previousMatrixBuffer = this._thinInstanceCreateMatrixBuffer(\"previousWorld\", buffer, staticBuffer);\r\n }\r\n } else {\r\n // color for instanced mesh is ColorInstanceKind and not ColorKind because of native that needs to do the differenciation\r\n // hot switching kind here to preserve backward compatibility\r\n if (kind === VertexBuffer.ColorKind) {\r\n kind = VertexBuffer.ColorInstanceKind;\r\n }\r\n\r\n if (buffer === null) {\r\n if (this._userThinInstanceBuffersStorage?.data[kind]) {\r\n this.removeVerticesData(kind);\r\n delete this._userThinInstanceBuffersStorage.data[kind];\r\n delete this._userThinInstanceBuffersStorage.strides[kind];\r\n delete this._userThinInstanceBuffersStorage.sizes[kind];\r\n delete this._userThinInstanceBuffersStorage.vertexBuffers[kind];\r\n }\r\n } else {\r\n this._thinInstanceInitializeUserStorage();\r\n\r\n this._userThinInstanceBuffersStorage.data[kind] = buffer;\r\n this._userThinInstanceBuffersStorage.strides[kind] = stride;\r\n this._userThinInstanceBuffersStorage.sizes[kind] = buffer.length;\r\n this._userThinInstanceBuffersStorage.vertexBuffers[kind] = new VertexBuffer(this.getEngine(), buffer, kind, !staticBuffer, false, stride, true);\r\n\r\n this.setVerticesBuffer(this._userThinInstanceBuffersStorage.vertexBuffers[kind]!);\r\n }\r\n }\r\n};\r\n\r\nMesh.prototype.thinInstanceBufferUpdated = function (kind: string): void {\r\n if (kind === \"matrix\") {\r\n this._thinInstanceDataStorage.matrixBuffer?.updateDirectly(this._thinInstanceDataStorage.matrixData!, 0, this._thinInstanceDataStorage.instancesCount);\r\n } else if (kind === \"previousMatrix\") {\r\n this._thinInstanceDataStorage.previousMatrixBuffer?.updateDirectly(this._thinInstanceDataStorage.previousMatrixData!, 0, this._thinInstanceDataStorage.instancesCount);\r\n } else {\r\n // preserve backward compatibility\r\n if (kind === VertexBuffer.ColorKind) {\r\n kind = VertexBuffer.ColorInstanceKind;\r\n }\r\n\r\n if (this._userThinInstanceBuffersStorage?.vertexBuffers[kind]) {\r\n this._userThinInstanceBuffersStorage.vertexBuffers[kind]!.updateDirectly(this._userThinInstanceBuffersStorage.data[kind], 0);\r\n }\r\n }\r\n};\r\n\r\nMesh.prototype.thinInstancePartialBufferUpdate = function (kind: string, data: Float32Array, offset: number): void {\r\n if (kind === \"matrix\") {\r\n if (this._thinInstanceDataStorage.matrixBuffer) {\r\n this._thinInstanceDataStorage.matrixBuffer.updateDirectly(data, offset);\r\n }\r\n } else {\r\n // preserve backward compatibility\r\n if (kind === VertexBuffer.ColorKind) {\r\n kind = VertexBuffer.ColorInstanceKind;\r\n }\r\n\r\n if (this._userThinInstanceBuffersStorage?.vertexBuffers[kind]) {\r\n this._userThinInstanceBuffersStorage.vertexBuffers[kind]!.updateDirectly(data, offset);\r\n }\r\n }\r\n};\r\n\r\nMesh.prototype.thinInstanceGetWorldMatrices = function (): Matrix[] {\r\n if (!this._thinInstanceDataStorage.matrixData || !this._thinInstanceDataStorage.matrixBuffer) {\r\n return [];\r\n }\r\n const matrixData = this._thinInstanceDataStorage.matrixData;\r\n\r\n if (!this._thinInstanceDataStorage.worldMatrices) {\r\n this._thinInstanceDataStorage.worldMatrices = new Array();\r\n\r\n for (let i = 0; i < this._thinInstanceDataStorage.instancesCount; ++i) {\r\n this._thinInstanceDataStorage.worldMatrices[i] = Matrix.FromArray(matrixData, i * 16);\r\n }\r\n }\r\n\r\n return this._thinInstanceDataStorage.worldMatrices;\r\n};\r\n\r\nMesh.prototype.thinInstanceRefreshBoundingInfo = function (forceRefreshParentInfo: boolean = false, applySkeleton: boolean = false, applyMorph: boolean = false) {\r\n if (!this._thinInstanceDataStorage.matrixData || !this._thinInstanceDataStorage.matrixBuffer) {\r\n return;\r\n }\r\n\r\n const vectors = this._thinInstanceDataStorage.boundingVectors;\r\n\r\n if (forceRefreshParentInfo || !this.rawBoundingInfo) {\r\n vectors.length = 0;\r\n this.refreshBoundingInfo(applySkeleton, applyMorph);\r\n const boundingInfo = this.getBoundingInfo();\r\n this.rawBoundingInfo = new BoundingInfo(boundingInfo.minimum, boundingInfo.maximum);\r\n }\r\n\r\n const boundingInfo = this.getBoundingInfo();\r\n const matrixData = this._thinInstanceDataStorage.matrixData;\r\n\r\n if (vectors.length === 0) {\r\n for (let v = 0; v < boundingInfo.boundingBox.vectors.length; ++v) {\r\n vectors.push(boundingInfo.boundingBox.vectors[v].clone());\r\n }\r\n }\r\n\r\n TmpVectors.Vector3[0].setAll(Number.POSITIVE_INFINITY); // min\r\n TmpVectors.Vector3[1].setAll(Number.NEGATIVE_INFINITY); // max\r\n\r\n for (let i = 0; i < this._thinInstanceDataStorage.instancesCount; ++i) {\r\n Matrix.FromArrayToRef(matrixData, i * 16, TmpVectors.Matrix[0]);\r\n\r\n for (let v = 0; v < vectors.length; ++v) {\r\n Vector3.TransformCoordinatesToRef(vectors[v], TmpVectors.Matrix[0], TmpVectors.Vector3[2]);\r\n TmpVectors.Vector3[0].minimizeInPlace(TmpVectors.Vector3[2]);\r\n TmpVectors.Vector3[1].maximizeInPlace(TmpVectors.Vector3[2]);\r\n }\r\n }\r\n\r\n boundingInfo.reConstruct(TmpVectors.Vector3[0], TmpVectors.Vector3[1]);\r\n\r\n this._updateBoundingInfo();\r\n};\r\n\r\nMesh.prototype._thinInstanceUpdateBufferSize = function (kind: string, numInstances: number = 1) {\r\n // preserve backward compatibility\r\n if (kind === VertexBuffer.ColorKind) {\r\n kind = VertexBuffer.ColorInstanceKind;\r\n }\r\n\r\n const kindIsMatrix = kind === \"matrix\";\r\n\r\n if (!kindIsMatrix && (!this._userThinInstanceBuffersStorage || !this._userThinInstanceBuffersStorage.strides[kind])) {\r\n return;\r\n }\r\n\r\n const stride = kindIsMatrix ? 16 : this._userThinInstanceBuffersStorage.strides[kind];\r\n const currentSize = kindIsMatrix ? this._thinInstanceDataStorage.matrixBufferSize : this._userThinInstanceBuffersStorage.sizes[kind];\r\n let data = kindIsMatrix ? this._thinInstanceDataStorage.matrixData : this._userThinInstanceBuffersStorage.data[kind];\r\n\r\n const bufferSize = (this._thinInstanceDataStorage.instancesCount + numInstances) * stride;\r\n\r\n let newSize = currentSize;\r\n\r\n while (newSize < bufferSize) {\r\n newSize *= 2;\r\n }\r\n\r\n if (!data || currentSize != newSize) {\r\n if (!data) {\r\n data = new Float32Array(newSize);\r\n } else {\r\n const newData = new Float32Array(newSize);\r\n newData.set(data, 0);\r\n data = newData;\r\n }\r\n\r\n if (kindIsMatrix) {\r\n this._thinInstanceDataStorage.matrixBuffer?.dispose();\r\n this._thinInstanceDataStorage.matrixBuffer = this._thinInstanceCreateMatrixBuffer(\"world\", data, false);\r\n this._thinInstanceDataStorage.matrixData = data;\r\n this._thinInstanceDataStorage.matrixBufferSize = newSize;\r\n if (this._scene.needsPreviousWorldMatrices && !this._thinInstanceDataStorage.previousMatrixData) {\r\n this._thinInstanceDataStorage.previousMatrixBuffer?.dispose();\r\n this._thinInstanceDataStorage.previousMatrixBuffer = this._thinInstanceCreateMatrixBuffer(\"previousWorld\", data, false);\r\n }\r\n } else {\r\n this._userThinInstanceBuffersStorage.vertexBuffers[kind]?.dispose();\r\n\r\n this._userThinInstanceBuffersStorage.data[kind] = data;\r\n this._userThinInstanceBuffersStorage.sizes[kind] = newSize;\r\n this._userThinInstanceBuffersStorage.vertexBuffers[kind] = new VertexBuffer(this.getEngine(), data, kind, true, false, stride, true);\r\n\r\n this.setVerticesBuffer(this._userThinInstanceBuffersStorage.vertexBuffers[kind]!);\r\n }\r\n }\r\n};\r\n\r\nMesh.prototype._thinInstanceInitializeUserStorage = function () {\r\n if (!this._userThinInstanceBuffersStorage) {\r\n this._userThinInstanceBuffersStorage = {\r\n data: {},\r\n sizes: {},\r\n vertexBuffers: {},\r\n strides: {},\r\n };\r\n }\r\n};\r\n\r\nMesh.prototype._disposeThinInstanceSpecificData = function () {\r\n if (this._thinInstanceDataStorage?.matrixBuffer) {\r\n this._thinInstanceDataStorage.matrixBuffer.dispose();\r\n this._thinInstanceDataStorage.matrixBuffer = null;\r\n }\r\n};\r\n","import type { IBufferView, IAccessor, INode, IEXTMeshGpuInstancing } from \"babylonjs-gltf2interface\";\r\nimport { AccessorType, AccessorComponentType } from \"babylonjs-gltf2interface\";\r\nimport type { IGLTFExporterExtensionV2 } from \"../glTFExporterExtension\";\r\nimport type { _BinaryWriter } from \"../glTFExporter\";\r\nimport { _Exporter } from \"../glTFExporter\";\r\nimport type { Nullable } from \"core/types\";\r\nimport type { Node } from \"core/node\";\r\nimport { Mesh } from \"core/Meshes/mesh\";\r\nimport \"core/Meshes/thinInstanceMesh\";\r\nimport { TmpVectors, Quaternion, Vector3 } from \"core/Maths/math.vector\";\r\nimport { VertexBuffer } from \"core/Buffers/buffer\";\r\n\r\nconst NAME = \"EXT_mesh_gpu_instancing\";\r\n\r\n/**\r\n * [Specification](https://github.com/KhronosGroup/glTF/blob/main/extensions/2.0/Vendor/EXT_mesh_gpu_instancing/README.md)\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class EXT_mesh_gpu_instancing implements IGLTFExporterExtensionV2 {\r\n /** Name of this extension */\r\n public readonly name = NAME;\r\n\r\n /** Defines whether this extension is enabled */\r\n public enabled = true;\r\n\r\n /** Defines whether this extension is required */\r\n public required = false;\r\n\r\n private _exporter: _Exporter;\r\n\r\n private _wasUsed = false;\r\n\r\n constructor(exporter: _Exporter) {\r\n this._exporter = exporter;\r\n }\r\n\r\n public dispose() {}\r\n\r\n /** @internal */\r\n public get wasUsed() {\r\n return this._wasUsed;\r\n }\r\n\r\n public postExportNodeAsync(\r\n context: string,\r\n node: Nullable,\r\n babylonNode: Node,\r\n nodeMap: { [key: number]: number },\r\n binaryWriter: _BinaryWriter\r\n ): Promise> {\r\n return new Promise((resolve) => {\r\n if (node && babylonNode instanceof Mesh) {\r\n if (babylonNode.hasThinInstances && binaryWriter) {\r\n this._wasUsed = true;\r\n\r\n const noTranslation = Vector3.Zero();\r\n const noRotation = Quaternion.Identity();\r\n const noScale = Vector3.One();\r\n\r\n // retreive all the instance world matrix\r\n const matrix = babylonNode.thinInstanceGetWorldMatrices();\r\n\r\n const iwt = TmpVectors.Vector3[2];\r\n const iwr = TmpVectors.Quaternion[1];\r\n const iws = TmpVectors.Vector3[3];\r\n\r\n let hasAnyInstanceWorldTranslation = false;\r\n let hasAnyInstanceWorldRotation = false;\r\n let hasAnyInstanceWorldScale = false;\r\n\r\n // prepare temp buffers\r\n const translationBuffer = new Float32Array(babylonNode.thinInstanceCount * 3);\r\n const rotationBuffer = new Float32Array(babylonNode.thinInstanceCount * 4);\r\n const scaleBuffer = new Float32Array(babylonNode.thinInstanceCount * 3);\r\n\r\n let i = 0;\r\n for (const m of matrix) {\r\n m.decompose(iws, iwr, iwt);\r\n\r\n // fill the temp buffer\r\n translationBuffer.set(iwt.asArray(), i * 3);\r\n rotationBuffer.set(iwr.normalize().asArray(), i * 4); // ensure the quaternion is normalized\r\n scaleBuffer.set(iws.asArray(), i * 3);\r\n\r\n // this is where we decide if there is any transformation\r\n hasAnyInstanceWorldTranslation = hasAnyInstanceWorldTranslation || !iwt.equalsWithEpsilon(noTranslation);\r\n hasAnyInstanceWorldRotation = hasAnyInstanceWorldRotation || !iwr.equalsWithEpsilon(noRotation);\r\n hasAnyInstanceWorldScale = hasAnyInstanceWorldScale || !iws.equalsWithEpsilon(noScale);\r\n\r\n i++;\r\n }\r\n\r\n const extension: IEXTMeshGpuInstancing = {\r\n attributes: {},\r\n };\r\n\r\n // do we need to write TRANSLATION ?\r\n if (hasAnyInstanceWorldTranslation) {\r\n extension.attributes[\"TRANSLATION\"] = this._buildAccessor(\r\n translationBuffer,\r\n AccessorType.VEC3,\r\n babylonNode.thinInstanceCount,\r\n binaryWriter,\r\n AccessorComponentType.FLOAT\r\n );\r\n }\r\n // do we need to write ROTATION ?\r\n if (hasAnyInstanceWorldRotation) {\r\n const componentType = AccessorComponentType.FLOAT; // we decided to stay on FLOAT for now see https://github.com/BabylonJS/Babylon.js/pull/12495\r\n extension.attributes[\"ROTATION\"] = this._buildAccessor(rotationBuffer, AccessorType.VEC4, babylonNode.thinInstanceCount, binaryWriter, componentType);\r\n }\r\n // do we need to write SCALE ?\r\n if (hasAnyInstanceWorldScale) {\r\n extension.attributes[\"SCALE\"] = this._buildAccessor(\r\n scaleBuffer,\r\n AccessorType.VEC3,\r\n babylonNode.thinInstanceCount,\r\n binaryWriter,\r\n AccessorComponentType.FLOAT\r\n );\r\n }\r\n\r\n /* eslint-enable @typescript-eslint/naming-convention*/\r\n node.extensions = node.extensions || {};\r\n node.extensions[NAME] = extension;\r\n }\r\n }\r\n resolve(node);\r\n });\r\n }\r\n\r\n private _buildAccessor(buffer: Float32Array, type: AccessorType, count: number, binaryWriter: _BinaryWriter, componentType: AccessorComponentType): number {\r\n // write the buffer\r\n const bufferOffset = binaryWriter.getByteOffset();\r\n switch (componentType) {\r\n case AccessorComponentType.FLOAT: {\r\n for (let i = 0; i != buffer.length; i++) {\r\n binaryWriter.setFloat32(buffer[i]);\r\n }\r\n break;\r\n }\r\n case AccessorComponentType.BYTE: {\r\n for (let i = 0; i != buffer.length; i++) {\r\n binaryWriter.setByte(buffer[i] * 127);\r\n }\r\n break;\r\n }\r\n case AccessorComponentType.SHORT: {\r\n for (let i = 0; i != buffer.length; i++) {\r\n binaryWriter.setInt16(buffer[i] * 32767);\r\n }\r\n\r\n break;\r\n }\r\n }\r\n // build the buffer view\r\n const bv: IBufferView = { buffer: 0, byteOffset: bufferOffset, byteLength: buffer.length * VertexBuffer.GetTypeByteLength(componentType) };\r\n const bufferViewIndex = this._exporter._bufferViews.length;\r\n this._exporter._bufferViews.push(bv);\r\n\r\n // finally build the accessor\r\n const accessorIndex = this._exporter._accessors.length;\r\n const accessor: IAccessor = {\r\n bufferView: bufferViewIndex,\r\n componentType: componentType,\r\n count: count,\r\n type: type,\r\n normalized: componentType == AccessorComponentType.BYTE || componentType == AccessorComponentType.SHORT,\r\n };\r\n this._exporter._accessors.push(accessor);\r\n return accessorIndex;\r\n }\r\n}\r\n\r\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n_Exporter.RegisterExtension(NAME, (exporter) => new EXT_mesh_gpu_instancing(exporter));\r\n","import type { IGLTFExporterExtensionV2 } from \"../glTFExporterExtension\";\r\nimport { _Exporter } from \"../glTFExporter\";\r\nimport type { Material } from \"core/Materials/material\";\r\nimport { PBRMaterial } from \"core/Materials/PBR/pbrMaterial\";\r\nimport type { IMaterial, IKHRMaterialsEmissiveStrength } from \"babylonjs-gltf2interface\";\r\n\r\nconst NAME = \"KHR_materials_emissive_strength\";\r\n\r\n/**\r\n * [Specification](https://github.com/KhronosGroup/glTF/blob/main/extensions/2.0/Khronos/KHR_materials_emissive_strength/README.md)\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class KHR_materials_emissive_strength implements IGLTFExporterExtensionV2 {\r\n /** Name of this extension */\r\n public readonly name = NAME;\r\n\r\n /** Defines whether this extension is enabled */\r\n public enabled = true;\r\n\r\n /** Defines whether this extension is required */\r\n public required = false;\r\n\r\n private _wasUsed = false;\r\n\r\n public dispose() {}\r\n\r\n /** @interal */\r\n public get wasUsed() {\r\n return this._wasUsed;\r\n }\r\n\r\n public postExportMaterialAsync(context: string, node: IMaterial, babylonMaterial: Material): Promise {\r\n return new Promise((resolve) => {\r\n if (!(babylonMaterial instanceof PBRMaterial)) {\r\n return resolve(node);\r\n }\r\n\r\n const emissiveColor = babylonMaterial.emissiveColor.asArray();\r\n const tempEmissiveStrength = Math.max(...emissiveColor);\r\n\r\n if (tempEmissiveStrength > 1) {\r\n this._wasUsed = true;\r\n\r\n node.extensions ||= {};\r\n\r\n const emissiveStrengthInfo: IKHRMaterialsEmissiveStrength = {\r\n emissiveStrength: tempEmissiveStrength,\r\n };\r\n\r\n // Normalize each value of the emissive factor to have a max value of 1\r\n const newEmissiveFactor = babylonMaterial.emissiveColor.scale(1 / emissiveStrengthInfo.emissiveStrength);\r\n\r\n node.emissiveFactor = newEmissiveFactor.asArray();\r\n node.extensions[NAME] = emissiveStrengthInfo;\r\n }\r\n\r\n return resolve(node);\r\n });\r\n }\r\n}\r\n\r\n_Exporter.RegisterExtension(NAME, (exporter) => new KHR_materials_emissive_strength());\r\n","import { Node } from '@babylonjs/core/node';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { GLTF2Export, GLTFData, IExportOptions } from '@babylonjs/serializers/glTF';\r\nimport { Connector, ViewpointNode } from '@models/classes';\r\nimport { getByDynamicId } from '@models/classes/kb3d-manager';\r\n\r\nexport class GLTFKb3dExporter {\r\n constructor(private scene: Scene) {}\r\n\r\n /** generates the current scene into a glb file and returns an object url for it */\r\n public generateGlb(filename = 'scene'): Promise {\r\n return GLTF2Export.GLBAsync(this.scene, filename, this._glbExportOptions).then(glbData => {\r\n return this.createGlbBlob(glbData);\r\n });\r\n }\r\n\r\n public exportGlb(filename = 'scene') {\r\n return GLTF2Export.GLBAsync(this.scene, filename, this._glbExportOptions).then(glbData => {\r\n const dispose = this.downloadBlob(`${filename}.glb`, this.createGlbBlob(glbData));\r\n setTimeout(() => dispose());\r\n });\r\n }\r\n\r\n public exportGltf(filename = 'scene') {\r\n return GLTF2Export.GLTFAsync(this.scene, filename, this._glbExportOptions).then(glbData => {\r\n const disposeFuncs: Array<() => void> = [];\r\n for (const key in glbData.glTFFiles) {\r\n const blob = glbData.glTFFiles[key];\r\n let mimeType;\r\n if (endsWith(key, '.glb')) {\r\n mimeType = { type: 'model/gltf-binary' };\r\n } else if (endsWith(key, '.bin')) {\r\n mimeType = { type: 'application/octet-stream' };\r\n } else if (endsWith(key, '.gltf')) {\r\n mimeType = { type: 'model/gltf+json' };\r\n } else if (endsWith(key, '.jpeg') || endsWith(key, '.jpg')) {\r\n mimeType = { type: 'image/jpeg' /* ImageMimeType.JPEG */ };\r\n } else if (endsWith(key, '.png')) {\r\n mimeType = { type: 'image/png' /* ImageMimeType.PNG */ };\r\n }\r\n\r\n disposeFuncs.push(this.downloadBlob(key, new Blob([blob], mimeType)));\r\n }\r\n setTimeout(() => disposeFuncs.forEach(dispose => dispose()));\r\n });\r\n\r\n function endsWith(str: string, suffix: string) {\r\n return str.indexOf(suffix, str.length - suffix.length) !== -1;\r\n }\r\n }\r\n\r\n private _glbExportOptions: IExportOptions = {\r\n shouldExportNode: n => this.shouldExportNode(n),\r\n };\r\n\r\n protected shouldExportNode(node: Node) {\r\n const kb3dNode = getByDynamicId(node.id);\r\n if (kb3dNode instanceof Connector) {\r\n return false;\r\n } else if (kb3dNode instanceof ViewpointNode) {\r\n return false;\r\n }\r\n return kb3dNode != null;\r\n }\r\n\r\n protected createGlbBlob(glbData: GLTFData) {\r\n //we're expecting one file since it's a GLB\r\n const key = Object.keys(glbData.glTFFiles)[0];\r\n const blob = glbData.glTFFiles[key];\r\n const mimeType = { type: 'model/gltf-binary' };\r\n return new Blob([blob], mimeType);\r\n }\r\n\r\n protected downloadBlob(fileName: string, blob: Blob) {\r\n const link = document.createElement('a');\r\n document.body.appendChild(link);\r\n link.setAttribute('type', 'hidden');\r\n link.download = fileName;\r\n link.href = window.URL.createObjectURL(blob);\r\n link.click();\r\n return dispose;\r\n\r\n function dispose() {\r\n window.URL.revokeObjectURL(link.href);\r\n link.remove();\r\n }\r\n }\r\n}\r\n","import type { Nullable } from \"../../types\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { Matrix } from \"../../Maths/math.vector\";\r\nimport type { InternalTexture } from \"../../Materials/Textures/internalTexture\";\r\nimport { BaseTexture } from \"../../Materials/Textures/baseTexture\";\r\nimport { Constants } from \"../../Engines/constants\";\r\nimport { RegisterClass } from \"../../Misc/typeStore\";\r\nimport type { ThinEngine } from \"../../Engines/thinEngine\";\r\n\r\n// Ensures Raw texture are included\r\nimport \"../../Engines/Extensions/engine.rawTexture\";\r\n\r\n/**\r\n * This represents a color grading texture. This acts as a lookup table LUT, useful during post process\r\n * It can help converting any input color in a desired output one. This can then be used to create effects\r\n * from sepia, black and white to sixties or futuristic rendering...\r\n *\r\n * The only supported format is currently 3dl.\r\n * More information on LUT: https://en.wikipedia.org/wiki/3D_lookup_table\r\n */\r\nexport class ColorGradingTexture extends BaseTexture {\r\n /**\r\n * The texture URL.\r\n */\r\n public url: string;\r\n\r\n /**\r\n * Empty line regex stored for GC.\r\n */\r\n private static _NoneEmptyLineRegex = /\\S+/;\r\n\r\n private _textureMatrix: Matrix;\r\n private _onLoad: Nullable<() => void>;\r\n\r\n /**\r\n * Instantiates a ColorGradingTexture from the following parameters.\r\n *\r\n * @param url The location of the color grading data (currently only supporting 3dl)\r\n * @param sceneOrEngine The scene or engine the texture will be used in\r\n * @param onLoad defines a callback triggered when the texture has been loaded\r\n */\r\n constructor(url: string, sceneOrEngine: Scene | ThinEngine, onLoad: Nullable<() => void> = null) {\r\n super(sceneOrEngine);\r\n\r\n if (!url) {\r\n return;\r\n }\r\n\r\n this._textureMatrix = Matrix.Identity();\r\n this.name = url;\r\n this.url = url;\r\n this._onLoad = onLoad;\r\n\r\n this._texture = this._getFromCache(url, true);\r\n\r\n if (!this._texture) {\r\n const scene = this.getScene();\r\n if (scene) {\r\n if (!scene.useDelayedTextureLoading) {\r\n this._loadTexture();\r\n } else {\r\n this.delayLoadState = Constants.DELAYLOADSTATE_NOTLOADED;\r\n }\r\n } else {\r\n this._loadTexture();\r\n }\r\n } else {\r\n this._triggerOnLoad();\r\n }\r\n }\r\n\r\n /**\r\n * Fires the onload event from the constructor if requested.\r\n */\r\n private _triggerOnLoad(): void {\r\n if (this._onLoad) {\r\n this._onLoad();\r\n }\r\n }\r\n\r\n /**\r\n * Returns the texture matrix used in most of the material.\r\n * This is not used in color grading but keep for troubleshooting purpose (easily swap diffuse by colorgrading to look in).\r\n */\r\n public getTextureMatrix(): Matrix {\r\n return this._textureMatrix;\r\n }\r\n\r\n /**\r\n * Occurs when the file being loaded is a .3dl LUT file.\r\n */\r\n private _load3dlTexture() {\r\n const engine = this._getEngine()!;\r\n let texture: InternalTexture;\r\n if (!engine._features.support3DTextures) {\r\n texture = engine.createRawTexture(\r\n null,\r\n 1,\r\n 1,\r\n Constants.TEXTUREFORMAT_RGBA,\r\n false,\r\n false,\r\n Constants.TEXTURE_BILINEAR_SAMPLINGMODE,\r\n null,\r\n Constants.TEXTURETYPE_UNSIGNED_INT\r\n );\r\n } else {\r\n texture = engine.createRawTexture3D(\r\n null,\r\n 1,\r\n 1,\r\n 1,\r\n Constants.TEXTUREFORMAT_RGBA,\r\n false,\r\n false,\r\n Constants.TEXTURE_BILINEAR_SAMPLINGMODE,\r\n null,\r\n Constants.TEXTURETYPE_UNSIGNED_INT\r\n );\r\n }\r\n\r\n this._texture = texture;\r\n this._texture.isReady = false;\r\n\r\n this.isCube = false;\r\n this.is3D = engine._features.support3DTextures;\r\n this.wrapU = Constants.TEXTURE_CLAMP_ADDRESSMODE;\r\n this.wrapV = Constants.TEXTURE_CLAMP_ADDRESSMODE;\r\n this.wrapR = Constants.TEXTURE_CLAMP_ADDRESSMODE;\r\n this.anisotropicFilteringLevel = 1;\r\n\r\n const callback = (text: string | ArrayBuffer) => {\r\n if (typeof text !== \"string\") {\r\n return;\r\n }\r\n\r\n let data: Nullable = null;\r\n let tempData: Nullable = null;\r\n\r\n let line: string;\r\n const lines = text.split(\"\\n\");\r\n let size = 0,\r\n pixelIndexW = 0,\r\n pixelIndexH = 0,\r\n pixelIndexSlice = 0;\r\n let maxColor = 0;\r\n\r\n for (let i = 0; i < lines.length; i++) {\r\n line = lines[i];\r\n\r\n if (!ColorGradingTexture._NoneEmptyLineRegex.test(line)) {\r\n continue;\r\n }\r\n\r\n if (line.indexOf(\"#\") === 0) {\r\n continue;\r\n }\r\n\r\n const words = line.split(\" \");\r\n if (size === 0) {\r\n // Number of space + one\r\n size = words.length;\r\n data = new Uint8Array(size * size * size * 4); // volume texture of side size and rgb 8\r\n tempData = new Float32Array(size * size * size * 4);\r\n continue;\r\n }\r\n\r\n if (size != 0) {\r\n const r = Math.max(parseInt(words[0]), 0);\r\n const g = Math.max(parseInt(words[1]), 0);\r\n const b = Math.max(parseInt(words[2]), 0);\r\n\r\n maxColor = Math.max(r, maxColor);\r\n maxColor = Math.max(g, maxColor);\r\n maxColor = Math.max(b, maxColor);\r\n\r\n const pixelStorageIndex = (pixelIndexW + pixelIndexSlice * size + pixelIndexH * size * size) * 4;\r\n\r\n if (tempData) {\r\n tempData[pixelStorageIndex + 0] = r;\r\n tempData[pixelStorageIndex + 1] = g;\r\n tempData[pixelStorageIndex + 2] = b;\r\n }\r\n\r\n // Keep for reference in case of back compat problems.\r\n // pixelIndexSlice++;\r\n // if (pixelIndexSlice % size == 0) {\r\n // pixelIndexH++;\r\n // pixelIndexSlice = 0;\r\n // if (pixelIndexH % size == 0) {\r\n // pixelIndexW++;\r\n // pixelIndexH = 0;\r\n // }\r\n // }\r\n\r\n pixelIndexH++;\r\n if (pixelIndexH % size == 0) {\r\n pixelIndexSlice++;\r\n pixelIndexH = 0;\r\n if (pixelIndexSlice % size == 0) {\r\n pixelIndexW++;\r\n pixelIndexSlice = 0;\r\n }\r\n }\r\n }\r\n }\r\n\r\n if (tempData && data) {\r\n for (let i = 0; i < tempData.length; i++) {\r\n if (i > 0 && (i + 1) % 4 === 0) {\r\n data[i] = 255;\r\n } else {\r\n const value = tempData[i];\r\n data[i] = (value / maxColor) * 255;\r\n }\r\n }\r\n }\r\n\r\n if (texture.is3D) {\r\n texture.updateSize(size, size, size);\r\n engine.updateRawTexture3D(texture, data, Constants.TEXTUREFORMAT_RGBA, false);\r\n } else {\r\n texture.updateSize(size * size, size);\r\n engine.updateRawTexture(texture, data, Constants.TEXTUREFORMAT_RGBA, false);\r\n }\r\n\r\n texture.isReady = true;\r\n this._triggerOnLoad();\r\n };\r\n\r\n const scene = this.getScene();\r\n if (scene) {\r\n scene._loadFile(this.url, callback);\r\n } else {\r\n engine._loadFile(this.url, callback);\r\n }\r\n\r\n return this._texture;\r\n }\r\n\r\n /**\r\n * Starts the loading process of the texture.\r\n */\r\n private _loadTexture() {\r\n if (this.url && this.url.toLocaleLowerCase().indexOf(\".3dl\") == this.url.length - 4) {\r\n this._load3dlTexture();\r\n }\r\n }\r\n\r\n /**\r\n * Clones the color grading texture.\r\n */\r\n public clone(): ColorGradingTexture {\r\n const newTexture = new ColorGradingTexture(this.url, this.getScene() || this._getEngine()!);\r\n\r\n // Base texture\r\n newTexture.level = this.level;\r\n\r\n return newTexture;\r\n }\r\n\r\n /**\r\n * Called during delayed load for textures.\r\n */\r\n public delayLoad(): void {\r\n if (this.delayLoadState !== Constants.DELAYLOADSTATE_NOTLOADED) {\r\n return;\r\n }\r\n\r\n this.delayLoadState = Constants.DELAYLOADSTATE_LOADED;\r\n this._texture = this._getFromCache(this.url, true);\r\n\r\n if (!this._texture) {\r\n this._loadTexture();\r\n }\r\n }\r\n\r\n /**\r\n * Parses a color grading texture serialized by Babylon.\r\n * @param parsedTexture The texture information being parsedTexture\r\n * @param scene The scene to load the texture in\r\n * @returns A color grading texture\r\n */\r\n public static Parse(parsedTexture: any, scene: Scene): Nullable {\r\n let texture = null;\r\n if (parsedTexture.name && !parsedTexture.isRenderTarget) {\r\n texture = new ColorGradingTexture(parsedTexture.name, scene);\r\n texture.name = parsedTexture.name;\r\n texture.level = parsedTexture.level;\r\n }\r\n return texture;\r\n }\r\n\r\n /**\r\n * Serializes the LUT texture to json format.\r\n */\r\n public serialize(): any {\r\n if (!this.name) {\r\n return null;\r\n }\r\n\r\n const serializationObject: any = {};\r\n serializationObject.name = this.name;\r\n serializationObject.level = this.level;\r\n serializationObject.customType = \"BABYLON.ColorGradingTexture\";\r\n\r\n return serializationObject;\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.ColorGradingTexture\", ColorGradingTexture);\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport type { Nullable } from \"../../types\";\r\nimport { Vector3 } from \"../../Maths/math.vector\";\r\nimport { Constants } from \"../../Engines/constants\";\r\n\r\n/**\r\n * CubeMap information grouping all the data for each faces as well as the cubemap size.\r\n */\r\nexport interface CubeMapInfo {\r\n /**\r\n * The pixel array for the front face.\r\n * This is stored in format, left to right, up to down format.\r\n */\r\n front: Nullable;\r\n\r\n /**\r\n * The pixel array for the back face.\r\n * This is stored in format, left to right, up to down format.\r\n */\r\n back: Nullable;\r\n\r\n /**\r\n * The pixel array for the left face.\r\n * This is stored in format, left to right, up to down format.\r\n */\r\n left: Nullable;\r\n\r\n /**\r\n * The pixel array for the right face.\r\n * This is stored in format, left to right, up to down format.\r\n */\r\n right: Nullable;\r\n\r\n /**\r\n * The pixel array for the up face.\r\n * This is stored in format, left to right, up to down format.\r\n */\r\n up: Nullable;\r\n\r\n /**\r\n * The pixel array for the down face.\r\n * This is stored in format, left to right, up to down format.\r\n */\r\n down: Nullable;\r\n\r\n /**\r\n * The size of the cubemap stored.\r\n *\r\n * Each faces will be size * size pixels.\r\n */\r\n size: number;\r\n\r\n /**\r\n * The format of the texture.\r\n *\r\n * RGBA, RGB.\r\n */\r\n format: number;\r\n\r\n /**\r\n * The type of the texture data.\r\n *\r\n * UNSIGNED_INT, FLOAT.\r\n */\r\n type: number;\r\n\r\n /**\r\n * Specifies whether the texture is in gamma space.\r\n */\r\n gammaSpace: boolean;\r\n}\r\n\r\n/**\r\n * Helper class useful to convert panorama picture to their cubemap representation in 6 faces.\r\n */\r\nexport class PanoramaToCubeMapTools {\r\n private static FACE_LEFT = [new Vector3(-1.0, -1.0, -1.0), new Vector3(1.0, -1.0, -1.0), new Vector3(-1.0, 1.0, -1.0), new Vector3(1.0, 1.0, -1.0)];\r\n private static FACE_RIGHT = [new Vector3(1.0, -1.0, 1.0), new Vector3(-1.0, -1.0, 1.0), new Vector3(1.0, 1.0, 1.0), new Vector3(-1.0, 1.0, 1.0)];\r\n private static FACE_FRONT = [new Vector3(1.0, -1.0, -1.0), new Vector3(1.0, -1.0, 1.0), new Vector3(1.0, 1.0, -1.0), new Vector3(1.0, 1.0, 1.0)];\r\n private static FACE_BACK = [new Vector3(-1.0, -1.0, 1.0), new Vector3(-1.0, -1.0, -1.0), new Vector3(-1.0, 1.0, 1.0), new Vector3(-1.0, 1.0, -1.0)];\r\n private static FACE_DOWN = [new Vector3(1.0, 1.0, -1.0), new Vector3(1.0, 1.0, 1.0), new Vector3(-1.0, 1.0, -1.0), new Vector3(-1.0, 1.0, 1.0)];\r\n private static FACE_UP = [new Vector3(-1.0, -1.0, -1.0), new Vector3(-1.0, -1.0, 1.0), new Vector3(1.0, -1.0, -1.0), new Vector3(1.0, -1.0, 1.0)];\r\n\r\n /**\r\n * Converts a panorama stored in RGB right to left up to down format into a cubemap (6 faces).\r\n *\r\n * @param float32Array The source data.\r\n * @param inputWidth The width of the input panorama.\r\n * @param inputHeight The height of the input panorama.\r\n * @param size The willing size of the generated cubemap (each faces will be size * size pixels)\r\n * @returns The cubemap data\r\n */\r\n public static ConvertPanoramaToCubemap(float32Array: Float32Array, inputWidth: number, inputHeight: number, size: number, supersample = false): CubeMapInfo {\r\n if (!float32Array) {\r\n throw \"ConvertPanoramaToCubemap: input cannot be null\";\r\n }\r\n\r\n if (float32Array.length != inputWidth * inputHeight * 3) {\r\n throw \"ConvertPanoramaToCubemap: input size is wrong\";\r\n }\r\n\r\n const textureFront = this.CreateCubemapTexture(size, this.FACE_FRONT, float32Array, inputWidth, inputHeight, supersample);\r\n const textureBack = this.CreateCubemapTexture(size, this.FACE_BACK, float32Array, inputWidth, inputHeight, supersample);\r\n const textureLeft = this.CreateCubemapTexture(size, this.FACE_LEFT, float32Array, inputWidth, inputHeight, supersample);\r\n const textureRight = this.CreateCubemapTexture(size, this.FACE_RIGHT, float32Array, inputWidth, inputHeight, supersample);\r\n const textureUp = this.CreateCubemapTexture(size, this.FACE_UP, float32Array, inputWidth, inputHeight, supersample);\r\n const textureDown = this.CreateCubemapTexture(size, this.FACE_DOWN, float32Array, inputWidth, inputHeight, supersample);\r\n\r\n return {\r\n front: textureFront,\r\n back: textureBack,\r\n left: textureLeft,\r\n right: textureRight,\r\n up: textureUp,\r\n down: textureDown,\r\n size: size,\r\n type: Constants.TEXTURETYPE_FLOAT,\r\n format: Constants.TEXTUREFORMAT_RGB,\r\n gammaSpace: false,\r\n };\r\n }\r\n\r\n private static CreateCubemapTexture(texSize: number, faceData: Vector3[], float32Array: Float32Array, inputWidth: number, inputHeight: number, supersample = false) {\r\n const buffer = new ArrayBuffer(texSize * texSize * 4 * 3);\r\n const textureArray = new Float32Array(buffer);\r\n\r\n // If supersampling, determine number of samples needed when source texture width is divided for 4 cube faces\r\n const samples = supersample ? Math.max(1, Math.round(inputWidth / 4 / texSize)) : 1;\r\n const sampleFactor = 1 / samples;\r\n const sampleFactorSqr = sampleFactor * sampleFactor;\r\n\r\n const rotDX1 = faceData[1].subtract(faceData[0]).scale(sampleFactor / texSize);\r\n const rotDX2 = faceData[3].subtract(faceData[2]).scale(sampleFactor / texSize);\r\n\r\n const dy = 1 / texSize;\r\n let fy = 0;\r\n\r\n for (let y = 0; y < texSize; y++) {\r\n for (let sy = 0; sy < samples; sy++) {\r\n let xv1 = faceData[0];\r\n let xv2 = faceData[2];\r\n\r\n for (let x = 0; x < texSize; x++) {\r\n for (let sx = 0; sx < samples; sx++) {\r\n const v = xv2.subtract(xv1).scale(fy).add(xv1);\r\n v.normalize();\r\n\r\n const color = this.CalcProjectionSpherical(v, float32Array, inputWidth, inputHeight);\r\n\r\n // 3 channels per pixels\r\n textureArray[y * texSize * 3 + x * 3 + 0] += color.r * sampleFactorSqr;\r\n textureArray[y * texSize * 3 + x * 3 + 1] += color.g * sampleFactorSqr;\r\n textureArray[y * texSize * 3 + x * 3 + 2] += color.b * sampleFactorSqr;\r\n\r\n xv1 = xv1.add(rotDX1);\r\n xv2 = xv2.add(rotDX2);\r\n }\r\n }\r\n\r\n fy += dy * sampleFactor;\r\n }\r\n }\r\n\r\n return textureArray;\r\n }\r\n\r\n private static CalcProjectionSpherical(vDir: Vector3, float32Array: Float32Array, inputWidth: number, inputHeight: number): any {\r\n let theta = Math.atan2(vDir.z, vDir.x);\r\n const phi = Math.acos(vDir.y);\r\n\r\n while (theta < -Math.PI) {\r\n theta += 2 * Math.PI;\r\n }\r\n while (theta > Math.PI) {\r\n theta -= 2 * Math.PI;\r\n }\r\n\r\n let dx = theta / Math.PI;\r\n const dy = phi / Math.PI;\r\n\r\n // recenter.\r\n dx = dx * 0.5 + 0.5;\r\n\r\n let px = Math.round(dx * inputWidth);\r\n if (px < 0) {\r\n px = 0;\r\n } else if (px >= inputWidth) {\r\n px = inputWidth - 1;\r\n }\r\n\r\n let py = Math.round(dy * inputHeight);\r\n if (py < 0) {\r\n py = 0;\r\n } else if (py >= inputHeight) {\r\n py = inputHeight - 1;\r\n }\r\n\r\n const inputY = inputHeight - py - 1;\r\n const r = float32Array[inputY * inputWidth * 3 + px * 3 + 0];\r\n const g = float32Array[inputY * inputWidth * 3 + px * 3 + 1];\r\n const b = float32Array[inputY * inputWidth * 3 + px * 3 + 2];\r\n\r\n return {\r\n r: r,\r\n g: g,\r\n b: b,\r\n };\r\n }\r\n}\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\n\nconst name = \"hdrFilteringVertexShader\";\nconst shader = `attribute vec2 position;varying vec3 direction;uniform vec3 up;uniform vec3 right;uniform vec3 front;\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\nmat3 view=mat3(up,right,front);direction=view*vec3(position,1.0);gl_Position=vec4(position,0.0,1.0);\n#define CUSTOM_VERTEX_MAIN_END\n}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const hdrFilteringVertexShader = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/helperFunctions\";\nimport \"./ShadersInclude/importanceSampling\";\nimport \"./ShadersInclude/pbrBRDFFunctions\";\nimport \"./ShadersInclude/hdrFilteringFunctions\";\n\nconst name = \"hdrFilteringPixelShader\";\nconst shader = `#include\n#include\n#include\n#include\nuniform float alphaG;uniform samplerCube inputTexture;uniform vec2 vFilteringInfo;uniform float hdrScale;varying vec3 direction;void main() {vec3 color=radiance(alphaG,inputTexture,direction,vFilteringInfo);gl_FragColor=vec4(color*hdrScale,1.0);}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const hdrFilteringPixelShader = { name, shader };\n","import { Vector3 } from \"../../../Maths/math\";\r\nimport { Scalar } from \"../../../Maths/math.scalar\";\r\nimport type { BaseTexture } from \"../baseTexture\";\r\nimport type { ThinEngine } from \"../../../Engines/thinEngine\";\r\nimport type { Effect } from \"../../../Materials/effect\";\r\nimport { Constants } from \"../../../Engines/constants\";\r\nimport { EffectWrapper, EffectRenderer } from \"../../../Materials/effectRenderer\";\r\nimport type { Nullable } from \"../../../types\";\r\nimport type { RenderTargetWrapper } from \"../../../Engines/renderTargetWrapper\";\r\n\r\nimport \"../../../Shaders/hdrFiltering.vertex\";\r\nimport \"../../../Shaders/hdrFiltering.fragment\";\r\nimport { Logger } from \"../../../Misc/logger\";\r\n\r\n/**\r\n * Options for texture filtering\r\n */\r\ninterface IHDRFilteringOptions {\r\n /**\r\n * Scales pixel intensity for the input HDR map.\r\n */\r\n hdrScale?: number;\r\n\r\n /**\r\n * Quality of the filter. Should be `Constants.TEXTURE_FILTERING_QUALITY_OFFLINE` for prefiltering\r\n */\r\n quality?: number;\r\n}\r\n\r\n/**\r\n * Filters HDR maps to get correct renderings of PBR reflections\r\n */\r\nexport class HDRFiltering {\r\n private _engine: ThinEngine;\r\n private _effectRenderer: EffectRenderer;\r\n private _effectWrapper: EffectWrapper;\r\n\r\n private _lodGenerationOffset: number = 0;\r\n private _lodGenerationScale: number = 0.8;\r\n\r\n /**\r\n * Quality switch for prefiltering. Should be set to `Constants.TEXTURE_FILTERING_QUALITY_OFFLINE` unless\r\n * you care about baking speed.\r\n */\r\n public quality: number = Constants.TEXTURE_FILTERING_QUALITY_OFFLINE;\r\n\r\n /**\r\n * Scales pixel intensity for the input HDR map.\r\n */\r\n public hdrScale: number = 1;\r\n\r\n /**\r\n * Instantiates HDR filter for reflection maps\r\n *\r\n * @param engine Thin engine\r\n * @param options Options\r\n */\r\n constructor(engine: ThinEngine, options: IHDRFilteringOptions = {}) {\r\n // pass\r\n this._engine = engine;\r\n this.hdrScale = options.hdrScale || this.hdrScale;\r\n this.quality = options.quality || this.quality;\r\n }\r\n\r\n private _createRenderTarget(size: number): RenderTargetWrapper {\r\n let textureType = Constants.TEXTURETYPE_UNSIGNED_BYTE;\r\n if (this._engine.getCaps().textureHalfFloatRender) {\r\n textureType = Constants.TEXTURETYPE_HALF_FLOAT;\r\n } else if (this._engine.getCaps().textureFloatRender) {\r\n textureType = Constants.TEXTURETYPE_FLOAT;\r\n }\r\n\r\n const rtWrapper = this._engine.createRenderTargetCubeTexture(size, {\r\n format: Constants.TEXTUREFORMAT_RGBA,\r\n type: textureType,\r\n createMipMaps: true,\r\n generateMipMaps: false,\r\n generateDepthBuffer: false,\r\n generateStencilBuffer: false,\r\n samplingMode: Constants.TEXTURE_NEAREST_SAMPLINGMODE,\r\n });\r\n this._engine.updateTextureWrappingMode(rtWrapper.texture!, Constants.TEXTURE_CLAMP_ADDRESSMODE, Constants.TEXTURE_CLAMP_ADDRESSMODE, Constants.TEXTURE_CLAMP_ADDRESSMODE);\r\n\r\n this._engine.updateTextureSamplingMode(Constants.TEXTURE_TRILINEAR_SAMPLINGMODE, rtWrapper.texture!, true);\r\n\r\n return rtWrapper;\r\n }\r\n\r\n private _prefilterInternal(texture: BaseTexture): BaseTexture {\r\n const width = texture.getSize().width;\r\n const mipmapsCount = Scalar.ILog2(width) + 1;\r\n\r\n const effect = this._effectWrapper.effect;\r\n const outputTexture = this._createRenderTarget(width);\r\n this._effectRenderer.saveStates();\r\n this._effectRenderer.setViewport();\r\n\r\n const intTexture = texture.getInternalTexture();\r\n if (intTexture) {\r\n // Just in case generate fresh clean mips.\r\n this._engine.updateTextureSamplingMode(Constants.TEXTURE_TRILINEAR_SAMPLINGMODE, intTexture, true);\r\n }\r\n\r\n this._effectRenderer.applyEffectWrapper(this._effectWrapper);\r\n\r\n const directions = [\r\n [new Vector3(0, 0, -1), new Vector3(0, -1, 0), new Vector3(1, 0, 0)], // PositiveX\r\n [new Vector3(0, 0, 1), new Vector3(0, -1, 0), new Vector3(-1, 0, 0)], // NegativeX\r\n [new Vector3(1, 0, 0), new Vector3(0, 0, 1), new Vector3(0, 1, 0)], // PositiveY\r\n [new Vector3(1, 0, 0), new Vector3(0, 0, -1), new Vector3(0, -1, 0)], // NegativeY\r\n [new Vector3(1, 0, 0), new Vector3(0, -1, 0), new Vector3(0, 0, 1)], // PositiveZ\r\n [new Vector3(-1, 0, 0), new Vector3(0, -1, 0), new Vector3(0, 0, -1)], // NegativeZ\r\n ];\r\n\r\n effect.setFloat(\"hdrScale\", this.hdrScale);\r\n effect.setFloat2(\"vFilteringInfo\", texture.getSize().width, mipmapsCount);\r\n effect.setTexture(\"inputTexture\", texture);\r\n\r\n for (let face = 0; face < 6; face++) {\r\n effect.setVector3(\"up\", directions[face][0]);\r\n effect.setVector3(\"right\", directions[face][1]);\r\n effect.setVector3(\"front\", directions[face][2]);\r\n\r\n for (let lod = 0; lod < mipmapsCount; lod++) {\r\n this._engine.bindFramebuffer(outputTexture, face, undefined, undefined, true, lod);\r\n this._effectRenderer.applyEffectWrapper(this._effectWrapper);\r\n\r\n let alpha = Math.pow(2, (lod - this._lodGenerationOffset) / this._lodGenerationScale) / width;\r\n if (lod === 0) {\r\n alpha = 0;\r\n }\r\n\r\n effect.setFloat(\"alphaG\", alpha);\r\n\r\n this._effectRenderer.draw();\r\n }\r\n }\r\n\r\n // Cleanup\r\n this._effectRenderer.restoreStates();\r\n this._engine.restoreDefaultFramebuffer();\r\n this._engine._releaseTexture(texture._texture!);\r\n\r\n // Internal Swap\r\n const type = outputTexture.texture!.type;\r\n const format = outputTexture.texture!.format;\r\n\r\n outputTexture._swapAndDie(texture._texture!);\r\n\r\n texture._texture!.type = type;\r\n texture._texture!.format = format;\r\n\r\n // New settings\r\n texture.gammaSpace = false;\r\n texture.lodGenerationOffset = this._lodGenerationOffset;\r\n texture.lodGenerationScale = this._lodGenerationScale;\r\n texture._prefiltered = true;\r\n\r\n return texture;\r\n }\r\n\r\n private _createEffect(texture: BaseTexture, onCompiled?: Nullable<(effect: Effect) => void>): EffectWrapper {\r\n const defines = [];\r\n if (texture.gammaSpace) {\r\n defines.push(\"#define GAMMA_INPUT\");\r\n }\r\n\r\n defines.push(\"#define NUM_SAMPLES \" + this.quality + \"u\"); // unsigned int\r\n\r\n const effectWrapper = new EffectWrapper({\r\n engine: this._engine,\r\n name: \"hdrFiltering\",\r\n vertexShader: \"hdrFiltering\",\r\n fragmentShader: \"hdrFiltering\",\r\n samplerNames: [\"inputTexture\"],\r\n uniformNames: [\"vSampleDirections\", \"vWeights\", \"up\", \"right\", \"front\", \"vFilteringInfo\", \"hdrScale\", \"alphaG\"],\r\n useShaderStore: true,\r\n defines,\r\n onCompiled: onCompiled,\r\n });\r\n\r\n return effectWrapper;\r\n }\r\n\r\n /**\r\n * Get a value indicating if the filter is ready to be used\r\n * @param texture Texture to filter\r\n * @returns true if the filter is ready\r\n */\r\n public isReady(texture: BaseTexture) {\r\n return texture.isReady() && this._effectWrapper.effect.isReady();\r\n }\r\n\r\n /**\r\n * Prefilters a cube texture to have mipmap levels representing roughness values.\r\n * Prefiltering will be invoked at the end of next rendering pass.\r\n * This has to be done once the map is loaded, and has not been prefiltered by a third party software.\r\n * See http://blog.selfshadow.com/publications/s2013-shading-course/karis/s2013_pbs_epic_notes_v2.pdf for more information\r\n * @param texture Texture to filter\r\n * @param onFinished Callback when filtering is done\r\n * @returns Promise called when prefiltering is done\r\n */\r\n public prefilter(texture: BaseTexture, onFinished: Nullable<() => void> = null): Promise {\r\n if (!this._engine._features.allowTexturePrefiltering) {\r\n Logger.Warn(\"HDR prefiltering is not available in WebGL 1., you can use real time filtering instead.\");\r\n return Promise.reject(\"HDR prefiltering is not available in WebGL 1., you can use real time filtering instead.\");\r\n }\r\n\r\n return new Promise((resolve) => {\r\n this._effectRenderer = new EffectRenderer(this._engine);\r\n this._effectWrapper = this._createEffect(texture);\r\n this._effectWrapper.effect.executeWhenCompiled(() => {\r\n this._prefilterInternal(texture);\r\n this._effectRenderer.dispose();\r\n this._effectWrapper.dispose();\r\n resolve();\r\n if (onFinished) {\r\n onFinished();\r\n }\r\n });\r\n });\r\n }\r\n}\r\n","import type { CubeMapInfo } from \"./panoramaToCubemap\";\r\nimport { PanoramaToCubeMapTools } from \"./panoramaToCubemap\";\r\n\r\n/**\r\n * Header information of HDR texture files.\r\n */\r\nexport interface HDRInfo {\r\n /**\r\n * The height of the texture in pixels.\r\n */\r\n height: number;\r\n\r\n /**\r\n * The width of the texture in pixels.\r\n */\r\n width: number;\r\n\r\n /**\r\n * The index of the beginning of the data in the binary file.\r\n */\r\n dataPosition: number;\r\n}\r\n\r\n/**\r\n * This groups tools to convert HDR texture to native colors array.\r\n */\r\nexport class HDRTools {\r\n private static _Ldexp(mantissa: number, exponent: number): number {\r\n if (exponent > 1023) {\r\n return mantissa * Math.pow(2, 1023) * Math.pow(2, exponent - 1023);\r\n }\r\n\r\n if (exponent < -1074) {\r\n return mantissa * Math.pow(2, -1074) * Math.pow(2, exponent + 1074);\r\n }\r\n\r\n return mantissa * Math.pow(2, exponent);\r\n }\r\n\r\n private static _Rgbe2float(float32array: Float32Array, red: number, green: number, blue: number, exponent: number, index: number) {\r\n if (exponent > 0) {\r\n /*nonzero pixel*/\r\n exponent = this._Ldexp(1.0, exponent - (128 + 8));\r\n\r\n float32array[index + 0] = red * exponent;\r\n float32array[index + 1] = green * exponent;\r\n float32array[index + 2] = blue * exponent;\r\n } else {\r\n float32array[index + 0] = 0;\r\n float32array[index + 1] = 0;\r\n float32array[index + 2] = 0;\r\n }\r\n }\r\n\r\n private static _ReadStringLine(uint8array: Uint8Array, startIndex: number): string {\r\n let line = \"\";\r\n let character = \"\";\r\n\r\n for (let i = startIndex; i < uint8array.length - startIndex; i++) {\r\n character = String.fromCharCode(uint8array[i]);\r\n\r\n if (character == \"\\n\") {\r\n break;\r\n }\r\n\r\n line += character;\r\n }\r\n\r\n return line;\r\n }\r\n\r\n /**\r\n * Reads header information from an RGBE texture stored in a native array.\r\n * More information on this format are available here:\r\n * https://en.wikipedia.org/wiki/RGBE_image_format\r\n *\r\n * @param uint8array The binary file stored in native array.\r\n * @returns The header information.\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public static RGBE_ReadHeader(uint8array: Uint8Array): HDRInfo {\r\n let height: number = 0;\r\n let width: number = 0;\r\n\r\n let line = this._ReadStringLine(uint8array, 0);\r\n if (line[0] != \"#\" || line[1] != \"?\") {\r\n throw \"Bad HDR Format.\";\r\n }\r\n\r\n let endOfHeader = false;\r\n let findFormat = false;\r\n let lineIndex: number = 0;\r\n\r\n do {\r\n lineIndex += line.length + 1;\r\n line = this._ReadStringLine(uint8array, lineIndex);\r\n\r\n if (line == \"FORMAT=32-bit_rle_rgbe\") {\r\n findFormat = true;\r\n } else if (line.length == 0) {\r\n endOfHeader = true;\r\n }\r\n } while (!endOfHeader);\r\n\r\n if (!findFormat) {\r\n throw \"HDR Bad header format, unsupported FORMAT\";\r\n }\r\n\r\n lineIndex += line.length + 1;\r\n line = this._ReadStringLine(uint8array, lineIndex);\r\n\r\n const sizeRegexp = /^-Y (.*) \\+X (.*)$/g;\r\n const match = sizeRegexp.exec(line);\r\n\r\n // TODO. Support +Y and -X if needed.\r\n if (!match || match.length < 3) {\r\n throw \"HDR Bad header format, no size\";\r\n }\r\n width = parseInt(match[2]);\r\n height = parseInt(match[1]);\r\n\r\n if (width < 8 || width > 0x7fff) {\r\n throw \"HDR Bad header format, unsupported size\";\r\n }\r\n\r\n lineIndex += line.length + 1;\r\n\r\n return {\r\n height: height,\r\n width: width,\r\n dataPosition: lineIndex,\r\n };\r\n }\r\n\r\n /**\r\n * Returns the cubemap information (each faces texture data) extracted from an RGBE texture.\r\n * This RGBE texture needs to store the information as a panorama.\r\n *\r\n * More information on this format are available here:\r\n * https://en.wikipedia.org/wiki/RGBE_image_format\r\n *\r\n * @param buffer The binary file stored in an array buffer.\r\n * @param size The expected size of the extracted cubemap.\r\n * @returns The Cube Map information.\r\n */\r\n public static GetCubeMapTextureData(buffer: ArrayBuffer, size: number, supersample = false): CubeMapInfo {\r\n const uint8array = new Uint8Array(buffer);\r\n const hdrInfo = this.RGBE_ReadHeader(uint8array);\r\n const data = this.RGBE_ReadPixels(uint8array, hdrInfo);\r\n\r\n const cubeMapData = PanoramaToCubeMapTools.ConvertPanoramaToCubemap(data, hdrInfo.width, hdrInfo.height, size, supersample);\r\n\r\n return cubeMapData;\r\n }\r\n\r\n /**\r\n * Returns the pixels data extracted from an RGBE texture.\r\n * This pixels will be stored left to right up to down in the R G B order in one array.\r\n *\r\n * More information on this format are available here:\r\n * https://en.wikipedia.org/wiki/RGBE_image_format\r\n *\r\n * @param uint8array The binary file stored in an array buffer.\r\n * @param hdrInfo The header information of the file.\r\n * @returns The pixels data in RGB right to left up to down order.\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n public static RGBE_ReadPixels(uint8array: Uint8Array, hdrInfo: HDRInfo): Float32Array {\r\n return this._RGBEReadPixelsRLE(uint8array, hdrInfo);\r\n }\r\n\r\n private static _RGBEReadPixelsRLE(uint8array: Uint8Array, hdrInfo: HDRInfo): Float32Array {\r\n let num_scanlines = hdrInfo.height;\r\n const scanline_width = hdrInfo.width;\r\n\r\n let a: number, b: number, c: number, d: number, count: number;\r\n let dataIndex = hdrInfo.dataPosition;\r\n let index = 0,\r\n endIndex = 0,\r\n i = 0;\r\n\r\n const scanLineArrayBuffer = new ArrayBuffer(scanline_width * 4); // four channel R G B E\r\n const scanLineArray = new Uint8Array(scanLineArrayBuffer);\r\n\r\n // 3 channels of 4 bytes per pixel in float.\r\n const resultBuffer = new ArrayBuffer(hdrInfo.width * hdrInfo.height * 4 * 3);\r\n const resultArray = new Float32Array(resultBuffer);\r\n\r\n // read in each successive scanline\r\n while (num_scanlines > 0) {\r\n a = uint8array[dataIndex++];\r\n b = uint8array[dataIndex++];\r\n c = uint8array[dataIndex++];\r\n d = uint8array[dataIndex++];\r\n\r\n if (a != 2 || b != 2 || c & 0x80 || hdrInfo.width < 8 || hdrInfo.width > 32767) {\r\n return this._RGBEReadPixelsNOTRLE(uint8array, hdrInfo);\r\n }\r\n\r\n if (((c << 8) | d) != scanline_width) {\r\n throw \"HDR Bad header format, wrong scan line width\";\r\n }\r\n\r\n index = 0;\r\n\r\n // read each of the four channels for the scanline into the buffer\r\n for (i = 0; i < 4; i++) {\r\n endIndex = (i + 1) * scanline_width;\r\n\r\n while (index < endIndex) {\r\n a = uint8array[dataIndex++];\r\n b = uint8array[dataIndex++];\r\n\r\n if (a > 128) {\r\n // a run of the same value\r\n count = a - 128;\r\n if (count == 0 || count > endIndex - index) {\r\n throw \"HDR Bad Format, bad scanline data (run)\";\r\n }\r\n\r\n while (count-- > 0) {\r\n scanLineArray[index++] = b;\r\n }\r\n } else {\r\n // a non-run\r\n count = a;\r\n if (count == 0 || count > endIndex - index) {\r\n throw \"HDR Bad Format, bad scanline data (non-run)\";\r\n }\r\n\r\n scanLineArray[index++] = b;\r\n if (--count > 0) {\r\n for (let j = 0; j < count; j++) {\r\n scanLineArray[index++] = uint8array[dataIndex++];\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n // now convert data from buffer into floats\r\n for (i = 0; i < scanline_width; i++) {\r\n a = scanLineArray[i];\r\n b = scanLineArray[i + scanline_width];\r\n c = scanLineArray[i + 2 * scanline_width];\r\n d = scanLineArray[i + 3 * scanline_width];\r\n\r\n this._Rgbe2float(resultArray, a, b, c, d, (hdrInfo.height - num_scanlines) * scanline_width * 3 + i * 3);\r\n }\r\n\r\n num_scanlines--;\r\n }\r\n\r\n return resultArray;\r\n }\r\n\r\n private static _RGBEReadPixelsNOTRLE(uint8array: Uint8Array, hdrInfo: HDRInfo): Float32Array {\r\n // this file is not run length encoded\r\n // read values sequentially\r\n\r\n let num_scanlines = hdrInfo.height;\r\n const scanline_width = hdrInfo.width;\r\n\r\n let a: number, b: number, c: number, d: number, i: number;\r\n let dataIndex = hdrInfo.dataPosition;\r\n\r\n // 3 channels of 4 bytes per pixel in float.\r\n const resultBuffer = new ArrayBuffer(hdrInfo.width * hdrInfo.height * 4 * 3);\r\n const resultArray = new Float32Array(resultBuffer);\r\n\r\n // read in each successive scanline\r\n while (num_scanlines > 0) {\r\n for (i = 0; i < hdrInfo.width; i++) {\r\n a = uint8array[dataIndex++];\r\n b = uint8array[dataIndex++];\r\n c = uint8array[dataIndex++];\r\n d = uint8array[dataIndex++];\r\n\r\n this._Rgbe2float(resultArray, a, b, c, d, (hdrInfo.height - num_scanlines) * scanline_width * 3 + i * 3);\r\n }\r\n\r\n num_scanlines--;\r\n }\r\n\r\n return resultArray;\r\n }\r\n}\r\n","import type { Nullable } from \"../../types\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { Matrix, Vector3 } from \"../../Maths/math.vector\";\r\nimport { BaseTexture } from \"../../Materials/Textures/baseTexture\";\r\nimport { Texture } from \"../../Materials/Textures/texture\";\r\nimport { Constants } from \"../../Engines/constants\";\r\nimport { HDRTools } from \"../../Misc/HighDynamicRange/hdr\";\r\nimport { CubeMapToSphericalPolynomialTools } from \"../../Misc/HighDynamicRange/cubemapToSphericalPolynomial\";\r\nimport { RegisterClass } from \"../../Misc/typeStore\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport { Tools } from \"../../Misc/tools\";\r\nimport { ToGammaSpace } from \"../../Maths/math.constants\";\r\nimport type { ThinEngine } from \"../../Engines/thinEngine\";\r\nimport { HDRFiltering } from \"../../Materials/Textures/Filtering/hdrFiltering\";\r\nimport { ToHalfFloat } from \"../../Misc/textureTools\";\r\nimport \"../../Engines/Extensions/engine.rawTexture\";\r\nimport \"../../Materials/Textures/baseTexture.polynomial\";\r\n\r\n/**\r\n * This represents a texture coming from an HDR input.\r\n *\r\n * The only supported format is currently panorama picture stored in RGBE format.\r\n * Example of such files can be found on Poly Haven: https://polyhaven.com/hdris\r\n */\r\nexport class HDRCubeTexture extends BaseTexture {\r\n private static _FacesMapping = [\"right\", \"left\", \"up\", \"down\", \"front\", \"back\"];\r\n\r\n private _generateHarmonics = true;\r\n private _noMipmap: boolean;\r\n private _prefilterOnLoad: boolean;\r\n private _textureMatrix: Matrix;\r\n private _size: number;\r\n private _supersample: boolean;\r\n private _onLoad: () => void;\r\n private _onError: Nullable<() => void> = null;\r\n\r\n /**\r\n * The texture URL.\r\n */\r\n public url: string;\r\n\r\n protected _isBlocking: boolean = true;\r\n /**\r\n * Sets whether or not the texture is blocking during loading.\r\n */\r\n public set isBlocking(value: boolean) {\r\n this._isBlocking = value;\r\n }\r\n /**\r\n * Gets whether or not the texture is blocking during loading.\r\n */\r\n public get isBlocking(): boolean {\r\n return this._isBlocking;\r\n }\r\n\r\n protected _rotationY: number = 0;\r\n /**\r\n * Sets texture matrix rotation angle around Y axis in radians.\r\n */\r\n public set rotationY(value: number) {\r\n this._rotationY = value;\r\n this.setReflectionTextureMatrix(Matrix.RotationY(this._rotationY));\r\n }\r\n /**\r\n * Gets texture matrix rotation angle around Y axis radians.\r\n */\r\n public get rotationY(): number {\r\n return this._rotationY;\r\n }\r\n\r\n /**\r\n * Gets or sets the center of the bounding box associated with the cube texture\r\n * It must define where the camera used to render the texture was set\r\n */\r\n public boundingBoxPosition = Vector3.Zero();\r\n\r\n private _boundingBoxSize: Vector3;\r\n\r\n /**\r\n * Gets or sets the size of the bounding box associated with the cube texture\r\n * When defined, the cubemap will switch to local mode\r\n * @see https://community.arm.com/graphics/b/blog/posts/reflections-based-on-local-cubemaps-in-unity\r\n * @example https://www.babylonjs-playground.com/#RNASML\r\n */\r\n public set boundingBoxSize(value: Vector3) {\r\n if (this._boundingBoxSize && this._boundingBoxSize.equals(value)) {\r\n return;\r\n }\r\n this._boundingBoxSize = value;\r\n const scene = this.getScene();\r\n if (scene) {\r\n scene.markAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag);\r\n }\r\n }\r\n public get boundingBoxSize(): Vector3 {\r\n return this._boundingBoxSize;\r\n }\r\n\r\n /**\r\n * Observable triggered once the texture has been loaded.\r\n */\r\n public onLoadObservable: Observable = new Observable();\r\n\r\n /**\r\n * Instantiates an HDRTexture from the following parameters.\r\n *\r\n * @param url The location of the HDR raw data (Panorama stored in RGBE format)\r\n * @param sceneOrEngine The scene or engine the texture will be used in\r\n * @param size The cubemap desired size (the more it increases the longer the generation will be)\r\n * @param noMipmap Forces to not generate the mipmap if true\r\n * @param generateHarmonics Specifies whether you want to extract the polynomial harmonics during the generation process\r\n * @param gammaSpace Specifies if the texture will be use in gamma or linear space (the PBR material requires those texture in linear space, but the standard material would require them in Gamma space)\r\n * @param prefilterOnLoad Prefilters HDR texture to allow use of this texture as a PBR reflection texture.\r\n * @param onLoad\r\n * @param onError\r\n */\r\n constructor(\r\n url: string,\r\n sceneOrEngine: Scene | ThinEngine,\r\n size: number,\r\n noMipmap = false,\r\n generateHarmonics = true,\r\n gammaSpace = false,\r\n prefilterOnLoad = false,\r\n onLoad: Nullable<() => void> = null,\r\n onError: Nullable<(message?: string, exception?: any) => void> = null,\r\n supersample = false\r\n ) {\r\n super(sceneOrEngine);\r\n\r\n if (!url) {\r\n return;\r\n }\r\n\r\n this._coordinatesMode = Texture.CUBIC_MODE;\r\n this.name = url;\r\n this.url = url;\r\n this.hasAlpha = false;\r\n this.isCube = true;\r\n this._textureMatrix = Matrix.Identity();\r\n this._prefilterOnLoad = prefilterOnLoad;\r\n this._onLoad = () => {\r\n this.onLoadObservable.notifyObservers(this);\r\n if (onLoad) {\r\n onLoad();\r\n }\r\n };\r\n\r\n this._onError = onError;\r\n this.gammaSpace = gammaSpace;\r\n\r\n this._noMipmap = noMipmap;\r\n this._size = size;\r\n this._supersample = supersample;\r\n this._generateHarmonics = generateHarmonics;\r\n\r\n this._texture = this._getFromCache(url, this._noMipmap, undefined, undefined, undefined, this.isCube);\r\n\r\n if (!this._texture) {\r\n if (!this.getScene()?.useDelayedTextureLoading) {\r\n this._loadTexture();\r\n } else {\r\n this.delayLoadState = Constants.DELAYLOADSTATE_NOTLOADED;\r\n }\r\n } else {\r\n if (this._texture.isReady) {\r\n Tools.SetImmediate(() => this._onLoad());\r\n } else {\r\n this._texture.onLoadedObservable.add(this._onLoad);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Get the current class name of the texture useful for serialization or dynamic coding.\r\n * @returns \"HDRCubeTexture\"\r\n */\r\n public getClassName(): string {\r\n return \"HDRCubeTexture\";\r\n }\r\n\r\n /**\r\n * Occurs when the file is raw .hdr file.\r\n */\r\n private _loadTexture() {\r\n const engine = this._getEngine()!;\r\n const caps = engine.getCaps();\r\n\r\n let textureType = Constants.TEXTURETYPE_UNSIGNED_BYTE;\r\n if (caps.textureFloat && caps.textureFloatLinearFiltering) {\r\n textureType = Constants.TEXTURETYPE_FLOAT;\r\n } else if (caps.textureHalfFloat && caps.textureHalfFloatLinearFiltering) {\r\n textureType = Constants.TEXTURETYPE_HALF_FLOAT;\r\n }\r\n\r\n const callback = (buffer: ArrayBuffer): Nullable => {\r\n this.lodGenerationOffset = 0.0;\r\n this.lodGenerationScale = 0.8;\r\n\r\n // Extract the raw linear data.\r\n const data = HDRTools.GetCubeMapTextureData(buffer, this._size, this._supersample);\r\n\r\n // Generate harmonics if needed.\r\n if (this._generateHarmonics) {\r\n const sphericalPolynomial = CubeMapToSphericalPolynomialTools.ConvertCubeMapToSphericalPolynomial(data);\r\n this.sphericalPolynomial = sphericalPolynomial;\r\n }\r\n\r\n const results = [];\r\n\r\n let byteArray: Nullable = null;\r\n let shortArray: Nullable = null;\r\n\r\n // Push each faces.\r\n for (let j = 0; j < 6; j++) {\r\n // Create fallback array\r\n if (textureType === Constants.TEXTURETYPE_HALF_FLOAT) {\r\n shortArray = new Uint16Array(this._size * this._size * 3);\r\n } else if (textureType === Constants.TEXTURETYPE_UNSIGNED_BYTE) {\r\n // 3 channels of 1 bytes per pixel in bytes.\r\n byteArray = new Uint8Array(this._size * this._size * 3);\r\n }\r\n\r\n const dataFace = (data)[HDRCubeTexture._FacesMapping[j]];\r\n\r\n // If special cases.\r\n if (this.gammaSpace || shortArray || byteArray) {\r\n for (let i = 0; i < this._size * this._size; i++) {\r\n // Put in gamma space if requested.\r\n if (this.gammaSpace) {\r\n dataFace[i * 3 + 0] = Math.pow(dataFace[i * 3 + 0], ToGammaSpace);\r\n dataFace[i * 3 + 1] = Math.pow(dataFace[i * 3 + 1], ToGammaSpace);\r\n dataFace[i * 3 + 2] = Math.pow(dataFace[i * 3 + 2], ToGammaSpace);\r\n }\r\n\r\n // Convert to half float texture for fallback.\r\n if (shortArray) {\r\n shortArray[i * 3 + 0] = ToHalfFloat(dataFace[i * 3 + 0]);\r\n shortArray[i * 3 + 1] = ToHalfFloat(dataFace[i * 3 + 1]);\r\n shortArray[i * 3 + 2] = ToHalfFloat(dataFace[i * 3 + 2]);\r\n }\r\n\r\n // Convert to int texture for fallback.\r\n if (byteArray) {\r\n let r = Math.max(dataFace[i * 3 + 0] * 255, 0);\r\n let g = Math.max(dataFace[i * 3 + 1] * 255, 0);\r\n let b = Math.max(dataFace[i * 3 + 2] * 255, 0);\r\n\r\n // May use luminance instead if the result is not accurate.\r\n const max = Math.max(Math.max(r, g), b);\r\n if (max > 255) {\r\n const scale = 255 / max;\r\n r *= scale;\r\n g *= scale;\r\n b *= scale;\r\n }\r\n\r\n byteArray[i * 3 + 0] = r;\r\n byteArray[i * 3 + 1] = g;\r\n byteArray[i * 3 + 2] = b;\r\n }\r\n }\r\n }\r\n\r\n if (shortArray) {\r\n results.push(shortArray);\r\n } else if (byteArray) {\r\n results.push(byteArray);\r\n } else {\r\n results.push(dataFace);\r\n }\r\n }\r\n\r\n return results;\r\n };\r\n\r\n if (engine._features.allowTexturePrefiltering && this._prefilterOnLoad) {\r\n const previousOnLoad = this._onLoad;\r\n const hdrFiltering = new HDRFiltering(engine);\r\n this._onLoad = () => {\r\n hdrFiltering.prefilter(this, previousOnLoad);\r\n };\r\n }\r\n\r\n this._texture = engine.createRawCubeTextureFromUrl(\r\n this.url,\r\n this.getScene(),\r\n this._size,\r\n Constants.TEXTUREFORMAT_RGB,\r\n textureType,\r\n this._noMipmap,\r\n callback,\r\n null,\r\n this._onLoad,\r\n this._onError\r\n );\r\n }\r\n\r\n public clone(): HDRCubeTexture {\r\n const newTexture = new HDRCubeTexture(this.url, this.getScene() || this._getEngine()!, this._size, this._noMipmap, this._generateHarmonics, this.gammaSpace);\r\n\r\n // Base texture\r\n newTexture.level = this.level;\r\n newTexture.wrapU = this.wrapU;\r\n newTexture.wrapV = this.wrapV;\r\n newTexture.coordinatesIndex = this.coordinatesIndex;\r\n newTexture.coordinatesMode = this.coordinatesMode;\r\n\r\n return newTexture;\r\n }\r\n\r\n // Methods\r\n public delayLoad(): void {\r\n if (this.delayLoadState !== Constants.DELAYLOADSTATE_NOTLOADED) {\r\n return;\r\n }\r\n\r\n this.delayLoadState = Constants.DELAYLOADSTATE_LOADED;\r\n this._texture = this._getFromCache(this.url, this._noMipmap);\r\n\r\n if (!this._texture) {\r\n this._loadTexture();\r\n }\r\n }\r\n\r\n /**\r\n * Get the texture reflection matrix used to rotate/transform the reflection.\r\n * @returns the reflection matrix\r\n */\r\n public getReflectionTextureMatrix(): Matrix {\r\n return this._textureMatrix;\r\n }\r\n\r\n /**\r\n * Set the texture reflection matrix used to rotate/transform the reflection.\r\n * @param value Define the reflection matrix to set\r\n */\r\n public setReflectionTextureMatrix(value: Matrix): void {\r\n this._textureMatrix = value;\r\n\r\n if (value.updateFlag === this._textureMatrix.updateFlag) {\r\n return;\r\n }\r\n\r\n if (value.isIdentity() !== this._textureMatrix.isIdentity()) {\r\n this.getScene()?.markAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag, (mat) => mat.getActiveTextures().indexOf(this) !== -1);\r\n }\r\n }\r\n\r\n /**\r\n * Dispose the texture and release its associated resources.\r\n */\r\n public dispose(): void {\r\n this.onLoadObservable.clear();\r\n super.dispose();\r\n }\r\n\r\n /**\r\n * Parses a JSON representation of an HDR Texture in order to create the texture\r\n * @param parsedTexture Define the JSON representation\r\n * @param scene Define the scene the texture should be created in\r\n * @param rootUrl Define the root url in case we need to load relative dependencies\r\n * @returns the newly created texture after parsing\r\n */\r\n public static Parse(parsedTexture: any, scene: Scene, rootUrl: string): Nullable {\r\n let texture = null;\r\n if (parsedTexture.name && !parsedTexture.isRenderTarget) {\r\n texture = new HDRCubeTexture(\r\n rootUrl + parsedTexture.name,\r\n scene,\r\n parsedTexture.size,\r\n parsedTexture.noMipmap,\r\n parsedTexture.generateHarmonics,\r\n parsedTexture.useInGammaSpace\r\n );\r\n texture.name = parsedTexture.name;\r\n texture.hasAlpha = parsedTexture.hasAlpha;\r\n texture.level = parsedTexture.level;\r\n texture.coordinatesMode = parsedTexture.coordinatesMode;\r\n texture.isBlocking = parsedTexture.isBlocking;\r\n }\r\n if (texture) {\r\n if (parsedTexture.boundingBoxPosition) {\r\n (texture).boundingBoxPosition = Vector3.FromArray(parsedTexture.boundingBoxPosition);\r\n }\r\n if (parsedTexture.boundingBoxSize) {\r\n (texture).boundingBoxSize = Vector3.FromArray(parsedTexture.boundingBoxSize);\r\n }\r\n if (parsedTexture.rotationY) {\r\n (texture).rotationY = parsedTexture.rotationY;\r\n }\r\n }\r\n return texture;\r\n }\r\n\r\n public serialize(): any {\r\n if (!this.name) {\r\n return null;\r\n }\r\n\r\n const serializationObject: any = {};\r\n serializationObject.name = this.name;\r\n serializationObject.hasAlpha = this.hasAlpha;\r\n serializationObject.isCube = true;\r\n serializationObject.level = this.level;\r\n serializationObject.size = this._size;\r\n serializationObject.coordinatesMode = this.coordinatesMode;\r\n serializationObject.useInGammaSpace = this.gammaSpace;\r\n serializationObject.generateHarmonics = this._generateHarmonics;\r\n serializationObject.customType = \"BABYLON.HDRCubeTexture\";\r\n serializationObject.noMipmap = this._noMipmap;\r\n serializationObject.isBlocking = this._isBlocking;\r\n serializationObject.rotationY = this._rotationY;\r\n\r\n return serializationObject;\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.HDRCubeTexture\", HDRCubeTexture);\r\n","import { ThinEngine } from \"../../Engines/thinEngine\";\r\nimport type { InternalTexture } from \"../../Materials/Textures/internalTexture\";\r\nimport type { Nullable } from \"../../types\";\r\nimport { Constants } from \"../constants\";\r\nimport type { ExternalTexture } from \"../../Materials/Textures/externalTexture\";\r\n\r\ndeclare module \"../../Engines/thinEngine\" {\r\n export interface ThinEngine {\r\n /**\r\n * Update a video texture\r\n * @param texture defines the texture to update\r\n * @param video defines the video element to use\r\n * @param invertY defines if data must be stored with Y axis inverted\r\n */\r\n updateVideoTexture(texture: Nullable, video: HTMLVideoElement | Nullable, invertY: boolean): void;\r\n }\r\n}\r\n\r\nThinEngine.prototype.updateVideoTexture = function (texture: Nullable, video: HTMLVideoElement, invertY: boolean): void {\r\n if (!texture || texture._isDisabled) {\r\n return;\r\n }\r\n\r\n const glformat = this._getInternalFormat(texture.format);\r\n const internalFormat = this._getRGBABufferInternalSizedFormat(Constants.TEXTURETYPE_UNSIGNED_BYTE, texture.format);\r\n\r\n const wasPreviouslyBound = this._bindTextureDirectly(this._gl.TEXTURE_2D, texture, true);\r\n this._unpackFlipY(!invertY); // Video are upside down by default\r\n\r\n try {\r\n // Testing video texture support\r\n if (this._videoTextureSupported === undefined) {\r\n // clear old errors just in case.\r\n this._gl.getError();\r\n\r\n this._gl.texImage2D(this._gl.TEXTURE_2D, 0, internalFormat, glformat, this._gl.UNSIGNED_BYTE, video);\r\n\r\n if (this._gl.getError() !== 0) {\r\n this._videoTextureSupported = false;\r\n } else {\r\n this._videoTextureSupported = true;\r\n }\r\n }\r\n\r\n // Copy video through the current working canvas if video texture is not supported\r\n if (!this._videoTextureSupported) {\r\n if (!texture._workingCanvas) {\r\n texture._workingCanvas = this.createCanvas(texture.width, texture.height);\r\n const context = texture._workingCanvas.getContext(\"2d\");\r\n\r\n if (!context) {\r\n throw new Error(\"Unable to get 2d context\");\r\n }\r\n\r\n texture._workingContext = context;\r\n texture._workingCanvas.width = texture.width;\r\n texture._workingCanvas.height = texture.height;\r\n }\r\n\r\n texture._workingContext!.clearRect(0, 0, texture.width, texture.height);\r\n texture._workingContext!.drawImage(video, 0, 0, video.videoWidth, video.videoHeight, 0, 0, texture.width, texture.height);\r\n\r\n this._gl.texImage2D(this._gl.TEXTURE_2D, 0, internalFormat, glformat, this._gl.UNSIGNED_BYTE, texture._workingCanvas as TexImageSource);\r\n } else {\r\n this._gl.texImage2D(this._gl.TEXTURE_2D, 0, internalFormat, glformat, this._gl.UNSIGNED_BYTE, video);\r\n }\r\n\r\n if (texture.generateMipMaps) {\r\n this._gl.generateMipmap(this._gl.TEXTURE_2D);\r\n }\r\n\r\n if (!wasPreviouslyBound) {\r\n this._bindTextureDirectly(this._gl.TEXTURE_2D, null);\r\n }\r\n // this.resetTextureCache();\r\n texture.isReady = true;\r\n } catch (ex) {\r\n // Something unexpected\r\n // Let's disable the texture\r\n texture._isDisabled = true;\r\n }\r\n};\r\n","import { ThinEngine } from \"../../Engines/thinEngine\";\r\nimport type { ExternalTexture } from \"../../Materials/Textures/externalTexture\";\r\nimport type { Nullable } from \"../../types\";\r\n\r\ndeclare module \"../../Engines/thinEngine\" {\r\n export interface ThinEngine {\r\n /**\r\n * Creates an external texture\r\n * @param video video element\r\n * @returns the external texture, or null if external textures are not supported by the engine\r\n */\r\n createExternalTexture(video: HTMLVideoElement): Nullable;\r\n\r\n /**\r\n * Sets an internal texture to the according uniform.\r\n * @param name The name of the uniform in the effect\r\n * @param texture The texture to apply\r\n */\r\n setExternalTexture(name: string, texture: Nullable): void;\r\n }\r\n}\r\n\r\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\r\nThinEngine.prototype.createExternalTexture = function (video: HTMLVideoElement): Nullable {\r\n return null;\r\n};\r\n\r\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\r\nThinEngine.prototype.setExternalTexture = function (name: string, texture: Nullable): void {\r\n throw new Error(\"setExternalTexture: This engine does not support external textures!\");\r\n};\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport type { InternalTexture } from \"../Materials/Textures/internalTexture\";\r\nimport { Logger } from \"../Misc/logger\";\r\n\r\n//private static _TYPE_NO_DATA = 0;\r\nconst _TYPE_INDEXED = 1;\r\nconst _TYPE_RGB = 2;\r\nconst _TYPE_GREY = 3;\r\nconst _TYPE_RLE_INDEXED = 9;\r\nconst _TYPE_RLE_RGB = 10;\r\nconst _TYPE_RLE_GREY = 11;\r\nconst _ORIGIN_MASK = 0x30;\r\nconst _ORIGIN_SHIFT = 0x04;\r\nconst _ORIGIN_BL = 0x00;\r\nconst _ORIGIN_BR = 0x01;\r\nconst _ORIGIN_UL = 0x02;\r\nconst _ORIGIN_UR = 0x03;\r\n\r\n/**\r\n * Gets the header of a TGA file\r\n * @param data defines the TGA data\r\n * @returns the header\r\n */\r\nexport function GetTGAHeader(data: Uint8Array): any {\r\n let offset = 0;\r\n\r\n const header = {\r\n id_length: data[offset++],\r\n colormap_type: data[offset++],\r\n image_type: data[offset++],\r\n colormap_index: data[offset++] | (data[offset++] << 8),\r\n colormap_length: data[offset++] | (data[offset++] << 8),\r\n colormap_size: data[offset++],\r\n origin: [data[offset++] | (data[offset++] << 8), data[offset++] | (data[offset++] << 8)],\r\n width: data[offset++] | (data[offset++] << 8),\r\n height: data[offset++] | (data[offset++] << 8),\r\n pixel_size: data[offset++],\r\n flags: data[offset++],\r\n };\r\n\r\n return header;\r\n}\r\n\r\n/**\r\n * Uploads TGA content to a Babylon Texture\r\n * @internal\r\n */\r\nexport function UploadContent(texture: InternalTexture, data: Uint8Array): void {\r\n // Not enough data to contain header ?\r\n if (data.length < 19) {\r\n Logger.Error(\"Unable to load TGA file - Not enough data to contain header\");\r\n return;\r\n }\r\n\r\n // Read Header\r\n let offset = 18;\r\n const header = GetTGAHeader(data);\r\n\r\n // Assume it's a valid Targa file.\r\n if (header.id_length + offset > data.length) {\r\n Logger.Error(\"Unable to load TGA file - Not enough data\");\r\n return;\r\n }\r\n\r\n // Skip not needed data\r\n offset += header.id_length;\r\n\r\n let use_rle = false;\r\n let use_pal = false;\r\n let use_grey = false;\r\n\r\n // Get some informations.\r\n switch (header.image_type) {\r\n case _TYPE_RLE_INDEXED:\r\n use_rle = true;\r\n // eslint-disable-next-line no-fallthrough\r\n case _TYPE_INDEXED:\r\n use_pal = true;\r\n break;\r\n\r\n case _TYPE_RLE_RGB:\r\n use_rle = true;\r\n // eslint-disable-next-line no-fallthrough\r\n case _TYPE_RGB:\r\n // use_rgb = true;\r\n break;\r\n\r\n case _TYPE_RLE_GREY:\r\n use_rle = true;\r\n // eslint-disable-next-line no-fallthrough\r\n case _TYPE_GREY:\r\n use_grey = true;\r\n break;\r\n }\r\n\r\n let pixel_data;\r\n\r\n // var numAlphaBits = header.flags & 0xf;\r\n const pixel_size = header.pixel_size >> 3;\r\n const pixel_total = header.width * header.height * pixel_size;\r\n\r\n // Read palettes\r\n let palettes;\r\n\r\n if (use_pal) {\r\n palettes = data.subarray(offset, (offset += header.colormap_length * (header.colormap_size >> 3)));\r\n }\r\n\r\n // Read LRE\r\n if (use_rle) {\r\n pixel_data = new Uint8Array(pixel_total);\r\n\r\n let c, count, i;\r\n let localOffset = 0;\r\n const pixels = new Uint8Array(pixel_size);\r\n\r\n while (offset < pixel_total && localOffset < pixel_total) {\r\n c = data[offset++];\r\n count = (c & 0x7f) + 1;\r\n\r\n // RLE pixels\r\n if (c & 0x80) {\r\n // Bind pixel tmp array\r\n for (i = 0; i < pixel_size; ++i) {\r\n pixels[i] = data[offset++];\r\n }\r\n\r\n // Copy pixel array\r\n for (i = 0; i < count; ++i) {\r\n pixel_data.set(pixels, localOffset + i * pixel_size);\r\n }\r\n\r\n localOffset += pixel_size * count;\r\n }\r\n // Raw pixels\r\n else {\r\n count *= pixel_size;\r\n for (i = 0; i < count; ++i) {\r\n pixel_data[localOffset + i] = data[offset++];\r\n }\r\n localOffset += count;\r\n }\r\n }\r\n }\r\n // RAW Pixels\r\n else {\r\n pixel_data = data.subarray(offset, (offset += use_pal ? header.width * header.height : pixel_total));\r\n }\r\n\r\n // Load to texture\r\n let x_start, y_start, x_step, y_step, y_end, x_end;\r\n\r\n switch ((header.flags & _ORIGIN_MASK) >> _ORIGIN_SHIFT) {\r\n default:\r\n case _ORIGIN_UL:\r\n x_start = 0;\r\n x_step = 1;\r\n x_end = header.width;\r\n y_start = 0;\r\n y_step = 1;\r\n y_end = header.height;\r\n break;\r\n\r\n case _ORIGIN_BL:\r\n x_start = 0;\r\n x_step = 1;\r\n x_end = header.width;\r\n y_start = header.height - 1;\r\n y_step = -1;\r\n y_end = -1;\r\n break;\r\n\r\n case _ORIGIN_UR:\r\n x_start = header.width - 1;\r\n x_step = -1;\r\n x_end = -1;\r\n y_start = 0;\r\n y_step = 1;\r\n y_end = header.height;\r\n break;\r\n\r\n case _ORIGIN_BR:\r\n x_start = header.width - 1;\r\n x_step = -1;\r\n x_end = -1;\r\n y_start = header.height - 1;\r\n y_step = -1;\r\n y_end = -1;\r\n break;\r\n }\r\n\r\n // Load the specify method\r\n const func = \"_getImageData\" + (use_grey ? \"Grey\" : \"\") + header.pixel_size + \"bits\";\r\n const imageData = (TGATools)[func](header, palettes, pixel_data, y_start, y_step, y_end, x_start, x_step, x_end);\r\n\r\n const engine = texture.getEngine();\r\n engine._uploadDataToTextureDirectly(texture, imageData);\r\n}\r\n\r\n/**\r\n * @internal\r\n */\r\nfunction _getImageData8bits(\r\n header: any,\r\n palettes: Uint8Array,\r\n pixel_data: Uint8Array,\r\n y_start: number,\r\n y_step: number,\r\n y_end: number,\r\n x_start: number,\r\n x_step: number,\r\n x_end: number\r\n): Uint8Array {\r\n const image = pixel_data,\r\n colormap = palettes;\r\n const width = header.width,\r\n height = header.height;\r\n let color,\r\n i = 0,\r\n x,\r\n y;\r\n\r\n const imageData = new Uint8Array(width * height * 4);\r\n\r\n for (y = y_start; y !== y_end; y += y_step) {\r\n for (x = x_start; x !== x_end; x += x_step, i++) {\r\n color = image[i];\r\n imageData[(x + width * y) * 4 + 3] = 255;\r\n imageData[(x + width * y) * 4 + 2] = colormap[color * 3 + 0];\r\n imageData[(x + width * y) * 4 + 1] = colormap[color * 3 + 1];\r\n imageData[(x + width * y) * 4 + 0] = colormap[color * 3 + 2];\r\n }\r\n }\r\n\r\n return imageData;\r\n}\r\n\r\n/**\r\n * @internal\r\n */\r\nfunction _getImageData16bits(\r\n header: any,\r\n palettes: Uint8Array,\r\n pixel_data: Uint8Array,\r\n y_start: number,\r\n y_step: number,\r\n y_end: number,\r\n x_start: number,\r\n x_step: number,\r\n x_end: number\r\n): Uint8Array {\r\n const image = pixel_data;\r\n const width = header.width,\r\n height = header.height;\r\n let color,\r\n i = 0,\r\n x,\r\n y;\r\n\r\n const imageData = new Uint8Array(width * height * 4);\r\n\r\n for (y = y_start; y !== y_end; y += y_step) {\r\n for (x = x_start; x !== x_end; x += x_step, i += 2) {\r\n color = image[i + 0] + (image[i + 1] << 8); // Inversed ?\r\n const r = ((((color & 0x7c00) >> 10) * 255) / 0x1f) | 0;\r\n const g = ((((color & 0x03e0) >> 5) * 255) / 0x1f) | 0;\r\n const b = (((color & 0x001f) * 255) / 0x1f) | 0;\r\n\r\n imageData[(x + width * y) * 4 + 0] = r;\r\n imageData[(x + width * y) * 4 + 1] = g;\r\n imageData[(x + width * y) * 4 + 2] = b;\r\n imageData[(x + width * y) * 4 + 3] = color & 0x8000 ? 0 : 255;\r\n }\r\n }\r\n\r\n return imageData;\r\n}\r\n\r\n/**\r\n * @internal\r\n */\r\nfunction _getImageData24bits(\r\n header: any,\r\n palettes: Uint8Array,\r\n pixel_data: Uint8Array,\r\n y_start: number,\r\n y_step: number,\r\n y_end: number,\r\n x_start: number,\r\n x_step: number,\r\n x_end: number\r\n): Uint8Array {\r\n const image = pixel_data;\r\n const width = header.width,\r\n height = header.height;\r\n let i = 0,\r\n x,\r\n y;\r\n\r\n const imageData = new Uint8Array(width * height * 4);\r\n\r\n for (y = y_start; y !== y_end; y += y_step) {\r\n for (x = x_start; x !== x_end; x += x_step, i += 3) {\r\n imageData[(x + width * y) * 4 + 3] = 255;\r\n imageData[(x + width * y) * 4 + 2] = image[i + 0];\r\n imageData[(x + width * y) * 4 + 1] = image[i + 1];\r\n imageData[(x + width * y) * 4 + 0] = image[i + 2];\r\n }\r\n }\r\n\r\n return imageData;\r\n}\r\n\r\n/**\r\n * @internal\r\n */\r\nfunction _getImageData32bits(\r\n header: any,\r\n palettes: Uint8Array,\r\n pixel_data: Uint8Array,\r\n y_start: number,\r\n y_step: number,\r\n y_end: number,\r\n x_start: number,\r\n x_step: number,\r\n x_end: number\r\n): Uint8Array {\r\n const image = pixel_data;\r\n const width = header.width,\r\n height = header.height;\r\n let i = 0,\r\n x,\r\n y;\r\n\r\n const imageData = new Uint8Array(width * height * 4);\r\n\r\n for (y = y_start; y !== y_end; y += y_step) {\r\n for (x = x_start; x !== x_end; x += x_step, i += 4) {\r\n imageData[(x + width * y) * 4 + 2] = image[i + 0];\r\n imageData[(x + width * y) * 4 + 1] = image[i + 1];\r\n imageData[(x + width * y) * 4 + 0] = image[i + 2];\r\n imageData[(x + width * y) * 4 + 3] = image[i + 3];\r\n }\r\n }\r\n\r\n return imageData;\r\n}\r\n\r\n/**\r\n * @internal\r\n */\r\nfunction _getImageDataGrey8bits(\r\n header: any,\r\n palettes: Uint8Array,\r\n pixel_data: Uint8Array,\r\n y_start: number,\r\n y_step: number,\r\n y_end: number,\r\n x_start: number,\r\n x_step: number,\r\n x_end: number\r\n): Uint8Array {\r\n const image = pixel_data;\r\n const width = header.width,\r\n height = header.height;\r\n let color,\r\n i = 0,\r\n x,\r\n y;\r\n\r\n const imageData = new Uint8Array(width * height * 4);\r\n\r\n for (y = y_start; y !== y_end; y += y_step) {\r\n for (x = x_start; x !== x_end; x += x_step, i++) {\r\n color = image[i];\r\n imageData[(x + width * y) * 4 + 0] = color;\r\n imageData[(x + width * y) * 4 + 1] = color;\r\n imageData[(x + width * y) * 4 + 2] = color;\r\n imageData[(x + width * y) * 4 + 3] = 255;\r\n }\r\n }\r\n\r\n return imageData;\r\n}\r\n\r\n/**\r\n * @internal\r\n */\r\nfunction _getImageDataGrey16bits(\r\n header: any,\r\n palettes: Uint8Array,\r\n pixel_data: Uint8Array,\r\n y_start: number,\r\n y_step: number,\r\n y_end: number,\r\n x_start: number,\r\n x_step: number,\r\n x_end: number\r\n): Uint8Array {\r\n const image = pixel_data;\r\n const width = header.width,\r\n height = header.height;\r\n let i = 0,\r\n x,\r\n y;\r\n\r\n const imageData = new Uint8Array(width * height * 4);\r\n\r\n for (y = y_start; y !== y_end; y += y_step) {\r\n for (x = x_start; x !== x_end; x += x_step, i += 2) {\r\n imageData[(x + width * y) * 4 + 0] = image[i + 0];\r\n imageData[(x + width * y) * 4 + 1] = image[i + 0];\r\n imageData[(x + width * y) * 4 + 2] = image[i + 0];\r\n imageData[(x + width * y) * 4 + 3] = image[i + 1];\r\n }\r\n }\r\n\r\n return imageData;\r\n}\r\n\r\n/**\r\n * Based on jsTGALoader - Javascript loader for TGA file\r\n * By Vincent Thibault\r\n * @see http://blog.robrowser.com/javascript-tga-loader.html\r\n */\r\nexport const TGATools = {\r\n /**\r\n * Gets the header of a TGA file\r\n * @param data defines the TGA data\r\n * @returns the header\r\n */\r\n GetTGAHeader,\r\n\r\n /**\r\n * Uploads TGA content to a Babylon Texture\r\n * @internal\r\n */\r\n UploadContent,\r\n\r\n /** @internal */\r\n _getImageData8bits,\r\n\r\n /** @internal */\r\n _getImageData16bits,\r\n /** @internal */\r\n _getImageData24bits,\r\n\r\n /** @internal */\r\n _getImageData32bits,\r\n\r\n /** @internal */\r\n _getImageDataGrey8bits,\r\n /** @internal */\r\n _getImageDataGrey16bits,\r\n};\r\n","import { GetTGAHeader, UploadContent } from \"../../../Misc/tga\";\r\nimport { Engine } from \"../../../Engines/engine\";\r\nimport type { InternalTexture } from \"../../../Materials/Textures/internalTexture\";\r\nimport type { IInternalTextureLoader } from \"../../../Materials/Textures/internalTextureLoader\";\r\n\r\n/**\r\n * Implementation of the TGA Texture Loader.\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class _TGATextureLoader implements IInternalTextureLoader {\r\n /**\r\n * Defines whether the loader supports cascade loading the different faces.\r\n */\r\n public readonly supportCascades = false;\r\n\r\n /**\r\n * This returns if the loader support the current file information.\r\n * @param extension defines the file extension of the file being loaded\r\n * @returns true if the loader can load the specified file\r\n */\r\n public canLoad(extension: string): boolean {\r\n return extension.endsWith(\".tga\");\r\n }\r\n\r\n /**\r\n * Uploads the cube texture data to the WebGL texture. It has already been bound.\r\n */\r\n public loadCubeData(): void {\r\n throw \".env not supported in Cube.\";\r\n }\r\n\r\n /**\r\n * Uploads the 2D texture data to the WebGL texture. It has already been bound once in the callback.\r\n * @param data contains the texture data\r\n * @param texture defines the BabylonJS internal texture\r\n * @param callback defines the method to call once ready to upload\r\n */\r\n public loadData(\r\n data: ArrayBufferView,\r\n texture: InternalTexture,\r\n callback: (width: number, height: number, loadMipmap: boolean, isCompressed: boolean, done: () => void) => void\r\n ): void {\r\n const bytes = new Uint8Array(data.buffer, data.byteOffset, data.byteLength);\r\n\r\n const header = GetTGAHeader(bytes);\r\n callback(header.width, header.height, texture.generateMipMaps, false, () => {\r\n UploadContent(texture, bytes);\r\n });\r\n }\r\n}\r\n\r\n// Register the loader.\r\nEngine._TextureLoaders.push(new _TGATextureLoader());\r\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport type { Nullable } from \"../types\";\r\nimport { Tools } from \"./tools\";\r\nimport { Texture } from \"../Materials/Textures/texture\";\r\nimport { InternalTexture, InternalTextureSource } from \"../Materials/Textures/internalTexture\";\r\nimport { Scalar } from \"../Maths/math.scalar\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport type { Engine } from \"../Engines/engine\";\r\n\r\n/**\r\n * Info about the .basis files\r\n */\r\nclass BasisFileInfo {\r\n /**\r\n * If the file has alpha\r\n */\r\n public hasAlpha: boolean;\r\n /**\r\n * Info about each image of the basis file\r\n */\r\n public images: Array<{ levels: Array<{ width: number; height: number; transcodedPixels: ArrayBufferView }> }>;\r\n}\r\n\r\n/**\r\n * Result of transcoding a basis file\r\n */\r\nclass TranscodeResult {\r\n /**\r\n * Info about the .basis file\r\n */\r\n public fileInfo: BasisFileInfo;\r\n /**\r\n * Format to use when loading the file\r\n */\r\n public format: number;\r\n}\r\n\r\n/**\r\n * Configuration options for the Basis transcoder\r\n */\r\nexport class BasisTranscodeConfiguration {\r\n /**\r\n * Supported compression formats used to determine the supported output format of the transcoder\r\n */\r\n supportedCompressionFormats?: {\r\n /**\r\n * etc1 compression format\r\n */\r\n etc1?: boolean;\r\n /**\r\n * s3tc compression format\r\n */\r\n s3tc?: boolean;\r\n /**\r\n * pvrtc compression format\r\n */\r\n pvrtc?: boolean;\r\n /**\r\n * etc2 compression format\r\n */\r\n etc2?: boolean;\r\n /**\r\n * astc compression format\r\n */\r\n astc?: boolean;\r\n /**\r\n * bc7 compression format\r\n */\r\n bc7?: boolean;\r\n };\r\n /**\r\n * If mipmap levels should be loaded for transcoded images (Default: true)\r\n */\r\n loadMipmapLevels?: boolean;\r\n /**\r\n * Index of a single image to load (Default: all images)\r\n */\r\n loadSingleImage?: number;\r\n}\r\n\r\n/**\r\n * @internal\r\n * Enum of basis transcoder formats\r\n */\r\nenum BASIS_FORMATS {\r\n cTFETC1 = 0,\r\n cTFETC2 = 1,\r\n cTFBC1 = 2,\r\n cTFBC3 = 3,\r\n cTFBC4 = 4,\r\n cTFBC5 = 5,\r\n cTFBC7 = 6,\r\n cTFPVRTC1_4_RGB = 8,\r\n cTFPVRTC1_4_RGBA = 9,\r\n cTFASTC_4x4 = 10,\r\n cTFATC_RGB = 11,\r\n cTFATC_RGBA_INTERPOLATED_ALPHA = 12,\r\n cTFRGBA32 = 13,\r\n cTFRGB565 = 14,\r\n cTFBGR565 = 15,\r\n cTFRGBA4444 = 16,\r\n cTFFXT1_RGB = 17,\r\n cTFPVRTC2_4_RGB = 18,\r\n cTFPVRTC2_4_RGBA = 19,\r\n cTFETC2_EAC_R11 = 20,\r\n cTFETC2_EAC_RG11 = 21,\r\n}\r\n\r\n/**\r\n * Used to load .Basis files\r\n * See https://github.com/BinomialLLC/basis_universal/tree/master/webgl\r\n */\r\nexport const BasisToolsOptions = {\r\n /**\r\n * URL to use when loading the basis transcoder\r\n */\r\n JSModuleURL: \"https://cdn.babylonjs.com/basisTranscoder/1/basis_transcoder.js\",\r\n /**\r\n * URL to use when loading the wasm module for the transcoder\r\n */\r\n WasmModuleURL: \"https://cdn.babylonjs.com/basisTranscoder/1/basis_transcoder.wasm\",\r\n};\r\n\r\n/**\r\n * Get the internal format to be passed to texImage2D corresponding to the .basis format value\r\n * @param basisFormat format chosen from GetSupportedTranscodeFormat\r\n * @param engine\r\n * @returns internal format corresponding to the Basis format\r\n */\r\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\r\nexport const GetInternalFormatFromBasisFormat = (basisFormat: number, engine: Engine) => {\r\n let format;\r\n switch (basisFormat) {\r\n case BASIS_FORMATS.cTFETC1:\r\n format = Constants.TEXTUREFORMAT_COMPRESSED_RGB_ETC1_WEBGL;\r\n break;\r\n case BASIS_FORMATS.cTFBC1:\r\n format = Constants.TEXTUREFORMAT_COMPRESSED_RGB_S3TC_DXT1;\r\n break;\r\n case BASIS_FORMATS.cTFBC4:\r\n format = Constants.TEXTUREFORMAT_COMPRESSED_RGBA_S3TC_DXT5;\r\n break;\r\n case BASIS_FORMATS.cTFASTC_4x4:\r\n format = Constants.TEXTUREFORMAT_COMPRESSED_RGBA_ASTC_4x4;\r\n break;\r\n case BASIS_FORMATS.cTFETC2:\r\n format = Constants.TEXTUREFORMAT_COMPRESSED_RGBA8_ETC2_EAC;\r\n break;\r\n case BASIS_FORMATS.cTFBC7:\r\n format = Constants.TEXTUREFORMAT_COMPRESSED_RGBA_BPTC_UNORM;\r\n break;\r\n }\r\n\r\n if (format === undefined) {\r\n throw \"The chosen Basis transcoder format is not currently supported\";\r\n }\r\n\r\n return format;\r\n};\r\n\r\nlet _WorkerPromise: Nullable> = null;\r\nlet _Worker: Nullable = null;\r\nlet _actionId = 0;\r\nconst _IgnoreSupportedFormats = false;\r\nconst _CreateWorkerAsync = () => {\r\n if (!_WorkerPromise) {\r\n _WorkerPromise = new Promise((res, reject) => {\r\n if (_Worker) {\r\n res(_Worker);\r\n } else {\r\n Tools.LoadFileAsync(BasisToolsOptions.WasmModuleURL)\r\n .then((wasmBinary) => {\r\n if (typeof URL !== \"function\") {\r\n return reject(\"Basis transcoder requires an environment with a URL constructor\");\r\n }\r\n const workerBlobUrl = URL.createObjectURL(new Blob([`(${workerFunc})()`], { type: \"application/javascript\" }));\r\n _Worker = new Worker(workerBlobUrl);\r\n\r\n const initHandler = (msg: any) => {\r\n if (msg.data.action === \"init\") {\r\n _Worker!.removeEventListener(\"message\", initHandler);\r\n res(_Worker!);\r\n } else if (msg.data.action === \"error\") {\r\n reject(msg.data.error || \"error initializing worker\");\r\n }\r\n };\r\n _Worker.addEventListener(\"message\", initHandler);\r\n _Worker.postMessage({ action: \"init\", url: BasisToolsOptions.JSModuleURL, wasmBinary: wasmBinary });\r\n })\r\n .catch(reject);\r\n }\r\n });\r\n }\r\n return _WorkerPromise;\r\n};\r\n\r\n/**\r\n * Transcodes a loaded image file to compressed pixel data\r\n * @param data image data to transcode\r\n * @param config configuration options for the transcoding\r\n * @returns a promise resulting in the transcoded image\r\n */\r\nexport const TranscodeAsync = (data: ArrayBuffer | ArrayBufferView, config: BasisTranscodeConfiguration): Promise => {\r\n const dataView = data instanceof ArrayBuffer ? new Uint8Array(data) : data;\r\n\r\n return new Promise((res, rej) => {\r\n _CreateWorkerAsync().then(\r\n () => {\r\n const actionId = _actionId++;\r\n const messageHandler = (msg: any) => {\r\n if (msg.data.action === \"transcode\" && msg.data.id === actionId) {\r\n _Worker!.removeEventListener(\"message\", messageHandler);\r\n if (!msg.data.success) {\r\n rej(\"Transcode is not supported on this device\");\r\n } else {\r\n res(msg.data);\r\n }\r\n }\r\n };\r\n _Worker!.addEventListener(\"message\", messageHandler);\r\n\r\n const dataViewCopy = new Uint8Array(dataView.byteLength);\r\n dataViewCopy.set(new Uint8Array(dataView.buffer, dataView.byteOffset, dataView.byteLength));\r\n _Worker!.postMessage({ action: \"transcode\", id: actionId, imageData: dataViewCopy, config: config, ignoreSupportedFormats: _IgnoreSupportedFormats }, [\r\n dataViewCopy.buffer,\r\n ]);\r\n },\r\n (error) => {\r\n rej(error);\r\n }\r\n );\r\n });\r\n};\r\n\r\n/**\r\n * Binds a texture according to its underlying target.\r\n * @param texture texture to bind\r\n * @param engine the engine to bind the texture in\r\n */\r\nconst BindTexture = (texture: InternalTexture, engine: Engine): void => {\r\n let target: GLenum = engine._gl?.TEXTURE_2D;\r\n if (texture.isCube) {\r\n target = engine._gl?.TEXTURE_CUBE_MAP;\r\n }\r\n\r\n engine._bindTextureDirectly(target, texture, true);\r\n};\r\n\r\n/**\r\n * Loads a texture from the transcode result\r\n * @param texture texture load to\r\n * @param transcodeResult the result of transcoding the basis file to load from\r\n */\r\nexport const LoadTextureFromTranscodeResult = (texture: InternalTexture, transcodeResult: TranscodeResult) => {\r\n const engine = texture.getEngine() as Engine;\r\n for (let i = 0; i < transcodeResult.fileInfo.images.length; i++) {\r\n const rootImage = transcodeResult.fileInfo.images[i].levels[0];\r\n texture._invertVScale = texture.invertY;\r\n if (transcodeResult.format === -1 || transcodeResult.format === BASIS_FORMATS.cTFRGB565) {\r\n // No compatable compressed format found, fallback to RGB\r\n texture.type = Constants.TEXTURETYPE_UNSIGNED_SHORT_5_6_5;\r\n texture.format = Constants.TEXTUREFORMAT_RGB;\r\n\r\n if (engine._features.basisNeedsPOT && (Scalar.Log2(rootImage.width) % 1 !== 0 || Scalar.Log2(rootImage.height) % 1 !== 0)) {\r\n // Create non power of two texture\r\n const source = new InternalTexture(engine, InternalTextureSource.Temp);\r\n\r\n texture._invertVScale = texture.invertY;\r\n source.type = Constants.TEXTURETYPE_UNSIGNED_SHORT_5_6_5;\r\n source.format = Constants.TEXTUREFORMAT_RGB;\r\n // Fallback requires aligned width/height\r\n source.width = (rootImage.width + 3) & ~3;\r\n source.height = (rootImage.height + 3) & ~3;\r\n BindTexture(source, engine);\r\n engine._uploadDataToTextureDirectly(source, new Uint16Array(rootImage.transcodedPixels.buffer), i, 0, Constants.TEXTUREFORMAT_RGB, true);\r\n\r\n // Resize to power of two\r\n engine._rescaleTexture(source, texture, engine.scenes[0], engine._getInternalFormat(Constants.TEXTUREFORMAT_RGB), () => {\r\n engine._releaseTexture(source);\r\n BindTexture(texture, engine);\r\n });\r\n } else {\r\n // Fallback is already inverted\r\n texture._invertVScale = !texture.invertY;\r\n\r\n // Upload directly\r\n texture.width = (rootImage.width + 3) & ~3;\r\n texture.height = (rootImage.height + 3) & ~3;\r\n texture.samplingMode = Constants.TEXTURE_LINEAR_LINEAR;\r\n BindTexture(texture, engine);\r\n engine._uploadDataToTextureDirectly(texture, new Uint16Array(rootImage.transcodedPixels.buffer), i, 0, Constants.TEXTUREFORMAT_RGB, true);\r\n }\r\n } else {\r\n texture.width = rootImage.width;\r\n texture.height = rootImage.height;\r\n texture.generateMipMaps = transcodeResult.fileInfo.images[i].levels.length > 1;\r\n\r\n const format = BasisTools.GetInternalFormatFromBasisFormat(transcodeResult.format!, engine);\r\n texture.format = format;\r\n\r\n BindTexture(texture, engine);\r\n\r\n // Upload all mip levels in the file\r\n transcodeResult.fileInfo.images[i].levels.forEach((level: any, index: number) => {\r\n engine._uploadCompressedDataToTextureDirectly(texture, format, level.width, level.height, level.transcodedPixels, i, index);\r\n });\r\n\r\n if (engine._features.basisNeedsPOT && (Scalar.Log2(texture.width) % 1 !== 0 || Scalar.Log2(texture.height) % 1 !== 0)) {\r\n Tools.Warn(\r\n \"Loaded .basis texture width and height are not a power of two. Texture wrapping will be set to Texture.CLAMP_ADDRESSMODE as other modes are not supported with non power of two dimensions in webGL 1.\"\r\n );\r\n texture._cachedWrapU = Texture.CLAMP_ADDRESSMODE;\r\n texture._cachedWrapV = Texture.CLAMP_ADDRESSMODE;\r\n }\r\n }\r\n }\r\n};\r\n\r\n/**\r\n * Used to load .Basis files\r\n * See https://github.com/BinomialLLC/basis_universal/tree/master/webgl\r\n */\r\nexport const BasisTools = {\r\n /**\r\n * URL to use when loading the basis transcoder\r\n */\r\n JSModuleURL: BasisToolsOptions.JSModuleURL,\r\n /**\r\n * URL to use when loading the wasm module for the transcoder\r\n */\r\n WasmModuleURL: BasisToolsOptions.WasmModuleURL,\r\n\r\n /**\r\n * Get the internal format to be passed to texImage2D corresponding to the .basis format value\r\n * @param basisFormat format chosen from GetSupportedTranscodeFormat\r\n * @returns internal format corresponding to the Basis format\r\n */\r\n GetInternalFormatFromBasisFormat,\r\n\r\n /**\r\n * Transcodes a loaded image file to compressed pixel data\r\n * @param data image data to transcode\r\n * @param config configuration options for the transcoding\r\n * @returns a promise resulting in the transcoded image\r\n */\r\n TranscodeAsync,\r\n\r\n /**\r\n * Loads a texture from the transcode result\r\n * @param texture texture load to\r\n * @param transcodeResult the result of transcoding the basis file to load from\r\n */\r\n LoadTextureFromTranscodeResult,\r\n};\r\n\r\n// WorkerGlobalScope\r\ndeclare function importScripts(...urls: string[]): void;\r\ndeclare function postMessage(message: any, transfer?: any[]): void;\r\ndeclare let BASIS: any;\r\nfunction workerFunc(): void {\r\n const _BASIS_FORMAT = {\r\n cTFETC1: 0,\r\n cTFETC2: 1,\r\n cTFBC1: 2,\r\n cTFBC3: 3,\r\n cTFBC4: 4,\r\n cTFBC5: 5,\r\n cTFBC7: 6,\r\n cTFPVRTC1_4_RGB: 8,\r\n cTFPVRTC1_4_RGBA: 9,\r\n cTFASTC_4x4: 10,\r\n cTFATC_RGB: 11,\r\n cTFATC_RGBA_INTERPOLATED_ALPHA: 12,\r\n cTFRGBA32: 13,\r\n cTFRGB565: 14,\r\n cTFBGR565: 15,\r\n cTFRGBA4444: 16,\r\n cTFFXT1_RGB: 17,\r\n cTFPVRTC2_4_RGB: 18,\r\n cTFPVRTC2_4_RGBA: 19,\r\n cTFETC2_EAC_R11: 20,\r\n cTFETC2_EAC_RG11: 21,\r\n };\r\n let transcoderModulePromise: Nullable> = null;\r\n onmessage = (event) => {\r\n if (event.data.action === \"init\") {\r\n // Load the transcoder if it hasn't been yet\r\n if (!transcoderModulePromise) {\r\n // make sure we loaded the script correctly\r\n try {\r\n importScripts(event.data.url);\r\n } catch (e) {\r\n postMessage({ action: \"error\", error: e });\r\n }\r\n transcoderModulePromise = BASIS({\r\n // Override wasm binary\r\n wasmBinary: event.data.wasmBinary,\r\n });\r\n }\r\n if (transcoderModulePromise !== null) {\r\n transcoderModulePromise.then((m) => {\r\n BASIS = m;\r\n m.initializeBasis();\r\n postMessage({ action: \"init\" });\r\n });\r\n }\r\n } else if (event.data.action === \"transcode\") {\r\n // Transcode the basis image and return the resulting pixels\r\n const config: BasisTranscodeConfiguration = event.data.config;\r\n const imgData = event.data.imageData;\r\n const loadedFile = new BASIS.BasisFile(imgData);\r\n const fileInfo = GetFileInfo(loadedFile);\r\n let format = event.data.ignoreSupportedFormats ? null : GetSupportedTranscodeFormat(event.data.config, fileInfo);\r\n\r\n let needsConversion = false;\r\n if (format === null) {\r\n needsConversion = true;\r\n format = fileInfo.hasAlpha ? _BASIS_FORMAT.cTFBC3 : _BASIS_FORMAT.cTFBC1;\r\n }\r\n\r\n // Begin transcode\r\n let success = true;\r\n if (!loadedFile.startTranscoding()) {\r\n success = false;\r\n }\r\n\r\n const buffers: Array = [];\r\n for (let imageIndex = 0; imageIndex < fileInfo.images.length; imageIndex++) {\r\n if (!success) {\r\n break;\r\n }\r\n const image = fileInfo.images[imageIndex];\r\n if (config.loadSingleImage === undefined || config.loadSingleImage === imageIndex) {\r\n let mipCount = image.levels.length;\r\n if (config.loadMipmapLevels === false) {\r\n mipCount = 1;\r\n }\r\n for (let levelIndex = 0; levelIndex < mipCount; levelIndex++) {\r\n const levelInfo = image.levels[levelIndex];\r\n\r\n const pixels = TranscodeLevel(loadedFile, imageIndex, levelIndex, format!, needsConversion);\r\n if (!pixels) {\r\n success = false;\r\n break;\r\n }\r\n levelInfo.transcodedPixels = pixels;\r\n buffers.push(levelInfo.transcodedPixels.buffer);\r\n }\r\n }\r\n }\r\n // Close file\r\n loadedFile.close();\r\n loadedFile.delete();\r\n\r\n if (needsConversion) {\r\n format = -1;\r\n }\r\n if (!success) {\r\n postMessage({ action: \"transcode\", success: success, id: event.data.id });\r\n } else {\r\n postMessage({ action: \"transcode\", success: success, id: event.data.id, fileInfo: fileInfo, format: format }, buffers);\r\n }\r\n }\r\n };\r\n\r\n /**\r\n * Detects the supported transcode format for the file\r\n * @param config transcode config\r\n * @param fileInfo info about the file\r\n * @returns the chosed format or null if none are supported\r\n */\r\n function GetSupportedTranscodeFormat(config: BasisTranscodeConfiguration, fileInfo: BasisFileInfo): Nullable {\r\n let format = null;\r\n if (config.supportedCompressionFormats) {\r\n if (config.supportedCompressionFormats.astc) {\r\n format = _BASIS_FORMAT.cTFASTC_4x4;\r\n } else if (config.supportedCompressionFormats.bc7) {\r\n format = _BASIS_FORMAT.cTFBC7;\r\n } else if (config.supportedCompressionFormats.s3tc) {\r\n format = fileInfo.hasAlpha ? _BASIS_FORMAT.cTFBC3 : _BASIS_FORMAT.cTFBC1;\r\n } else if (config.supportedCompressionFormats.pvrtc) {\r\n format = fileInfo.hasAlpha ? _BASIS_FORMAT.cTFPVRTC1_4_RGBA : _BASIS_FORMAT.cTFPVRTC1_4_RGB;\r\n } else if (config.supportedCompressionFormats.etc2) {\r\n format = _BASIS_FORMAT.cTFETC2;\r\n } else if (config.supportedCompressionFormats.etc1) {\r\n format = _BASIS_FORMAT.cTFETC1;\r\n } else {\r\n format = _BASIS_FORMAT.cTFRGB565;\r\n }\r\n }\r\n return format;\r\n }\r\n\r\n /**\r\n * Retrieves information about the basis file eg. dimensions\r\n * @param basisFile the basis file to get the info from\r\n * @returns information about the basis file\r\n */\r\n function GetFileInfo(basisFile: any): BasisFileInfo {\r\n const hasAlpha = basisFile.getHasAlpha();\r\n const imageCount = basisFile.getNumImages();\r\n const images = [];\r\n for (let i = 0; i < imageCount; i++) {\r\n const imageInfo = {\r\n levels: [] as Array,\r\n };\r\n const levelCount = basisFile.getNumLevels(i);\r\n for (let level = 0; level < levelCount; level++) {\r\n const levelInfo = {\r\n width: basisFile.getImageWidth(i, level),\r\n height: basisFile.getImageHeight(i, level),\r\n };\r\n imageInfo.levels.push(levelInfo);\r\n }\r\n images.push(imageInfo);\r\n }\r\n const info = { hasAlpha, images };\r\n return info;\r\n }\r\n\r\n function TranscodeLevel(loadedFile: any, imageIndex: number, levelIndex: number, format: number, convertToRgb565: boolean): Nullable {\r\n const dstSize = loadedFile.getImageTranscodedSizeInBytes(imageIndex, levelIndex, format);\r\n let dst: Uint8Array | Uint16Array = new Uint8Array(dstSize);\r\n if (!loadedFile.transcodeImage(dst, imageIndex, levelIndex, format, 1, 0)) {\r\n return null;\r\n }\r\n // If no supported format is found, load as dxt and convert to rgb565\r\n if (convertToRgb565) {\r\n const alignedWidth = (loadedFile.getImageWidth(imageIndex, levelIndex) + 3) & ~3;\r\n const alignedHeight = (loadedFile.getImageHeight(imageIndex, levelIndex) + 3) & ~3;\r\n dst = ConvertDxtToRgb565(dst, 0, alignedWidth, alignedHeight);\r\n }\r\n return dst;\r\n }\r\n\r\n /**\r\n * From https://github.com/BinomialLLC/basis_universal/blob/master/webgl/texture/dxt-to-rgb565.js\r\n * An unoptimized version of dxtToRgb565. Also, the floating\r\n * point math used to compute the colors actually results in\r\n * slightly different colors compared to hardware DXT decoders.\r\n * @param src dxt src pixels\r\n * @param srcByteOffset offset for the start of src\r\n * @param width aligned width of the image\r\n * @param height aligned height of the image\r\n * @returns the converted pixels\r\n */\r\n function ConvertDxtToRgb565(src: Uint8Array, srcByteOffset: number, width: number, height: number): Uint16Array {\r\n const c = new Uint16Array(4);\r\n const dst = new Uint16Array(width * height);\r\n\r\n const blockWidth = width / 4;\r\n const blockHeight = height / 4;\r\n for (let blockY = 0; blockY < blockHeight; blockY++) {\r\n for (let blockX = 0; blockX < blockWidth; blockX++) {\r\n const i = srcByteOffset + 8 * (blockY * blockWidth + blockX);\r\n c[0] = src[i] | (src[i + 1] << 8);\r\n c[1] = src[i + 2] | (src[i + 3] << 8);\r\n c[2] =\r\n ((2 * (c[0] & 0x1f) + 1 * (c[1] & 0x1f)) / 3) |\r\n (((2 * (c[0] & 0x7e0) + 1 * (c[1] & 0x7e0)) / 3) & 0x7e0) |\r\n (((2 * (c[0] & 0xf800) + 1 * (c[1] & 0xf800)) / 3) & 0xf800);\r\n c[3] =\r\n ((2 * (c[1] & 0x1f) + 1 * (c[0] & 0x1f)) / 3) |\r\n (((2 * (c[1] & 0x7e0) + 1 * (c[0] & 0x7e0)) / 3) & 0x7e0) |\r\n (((2 * (c[1] & 0xf800) + 1 * (c[0] & 0xf800)) / 3) & 0xf800);\r\n for (let row = 0; row < 4; row++) {\r\n const m = src[i + 4 + row];\r\n let dstI = (blockY * 4 + row) * width + blockX * 4;\r\n dst[dstI++] = c[m & 0x3];\r\n dst[dstI++] = c[(m >> 2) & 0x3];\r\n dst[dstI++] = c[(m >> 4) & 0x3];\r\n dst[dstI++] = c[(m >> 6) & 0x3];\r\n }\r\n }\r\n }\r\n return dst;\r\n }\r\n}\r\n\r\nObject.defineProperty(BasisTools, \"JSModuleURL\", {\r\n get: function (this: null) {\r\n return BasisToolsOptions.JSModuleURL;\r\n },\r\n set: function (this: null, value: string) {\r\n BasisToolsOptions.JSModuleURL = value;\r\n },\r\n});\r\n\r\nObject.defineProperty(BasisTools, \"WasmModuleURL\", {\r\n get: function (this: null) {\r\n return BasisToolsOptions.WasmModuleURL;\r\n },\r\n set: function (this: null, value: string) {\r\n BasisToolsOptions.WasmModuleURL = value;\r\n },\r\n});\r\n","import { HDRTools } from \"../../../Misc/HighDynamicRange/hdr\";\r\nimport { Engine } from \"../../../Engines/engine\";\r\nimport type { InternalTexture } from \"../../../Materials/Textures/internalTexture\";\r\nimport type { IInternalTextureLoader } from \"../../../Materials/Textures/internalTextureLoader\";\r\nimport { Constants } from \"../../../Engines/constants\";\r\n\r\n/**\r\n * Implementation of the HDR Texture Loader.\r\n * @internal\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class _HDRTextureLoader implements IInternalTextureLoader {\r\n /**\r\n * Defines whether the loader supports cascade loading the different faces.\r\n */\r\n public readonly supportCascades = false;\r\n\r\n /**\r\n * This returns if the loader support the current file information.\r\n * @param extension defines the file extension of the file being loaded\r\n * @returns true if the loader can load the specified file\r\n */\r\n public canLoad(extension: string): boolean {\r\n return extension.endsWith(\".hdr\");\r\n }\r\n\r\n /**\r\n * Uploads the cube texture data to the WebGL texture. It has already been bound.\r\n */\r\n public loadCubeData(): void {\r\n throw \".env not supported in Cube.\";\r\n }\r\n\r\n /**\r\n * Uploads the 2D texture data to the WebGL texture. It has already been bound once in the callback.\r\n * @param data contains the texture data\r\n * @param texture defines the BabylonJS internal texture\r\n * @param callback defines the method to call once ready to upload\r\n */\r\n public loadData(\r\n data: ArrayBufferView,\r\n texture: InternalTexture,\r\n callback: (width: number, height: number, loadMipmap: boolean, isCompressed: boolean, done: () => void) => void\r\n ): void {\r\n const uint8array = new Uint8Array(data.buffer, data.byteOffset, data.byteLength);\r\n const hdrInfo = HDRTools.RGBE_ReadHeader(uint8array);\r\n const pixelsDataRGB32 = HDRTools.RGBE_ReadPixels(uint8array, hdrInfo);\r\n\r\n const pixels = hdrInfo.width * hdrInfo.height;\r\n const pixelsDataRGBA32 = new Float32Array(pixels * 4);\r\n for (let i = 0; i < pixels; i += 1) {\r\n pixelsDataRGBA32[i * 4] = pixelsDataRGB32[i * 3];\r\n pixelsDataRGBA32[i * 4 + 1] = pixelsDataRGB32[i * 3 + 1];\r\n pixelsDataRGBA32[i * 4 + 2] = pixelsDataRGB32[i * 3 + 2];\r\n pixelsDataRGBA32[i * 4 + 3] = 1;\r\n }\r\n\r\n callback(hdrInfo.width, hdrInfo.height, texture.generateMipMaps, false, () => {\r\n const engine = texture.getEngine();\r\n texture.type = Constants.TEXTURETYPE_FLOAT;\r\n texture.format = Constants.TEXTUREFORMAT_RGBA;\r\n texture._gammaSpace = false;\r\n engine._uploadDataToTextureDirectly(texture, pixelsDataRGBA32);\r\n });\r\n }\r\n}\r\n\r\n// Register the loader.\r\nEngine._TextureLoaders.push(new _HDRTextureLoader());\r\n","import type { Nullable } from \"../../../types\";\r\nimport { Engine } from \"../../../Engines/engine\";\r\nimport type { InternalTexture } from \"../../../Materials/Textures/internalTexture\";\r\nimport type { IInternalTextureLoader } from \"../../../Materials/Textures/internalTextureLoader\";\r\nimport { LoadTextureFromTranscodeResult, TranscodeAsync } from \"../../../Misc/basis\";\r\nimport { Tools } from \"../../../Misc/tools\";\r\n\r\n/**\r\n * Loader for .basis file format\r\n */\r\n// eslint-disable-next-line @typescript-eslint/naming-convention\r\nexport class _BasisTextureLoader implements IInternalTextureLoader {\r\n /**\r\n * Defines whether the loader supports cascade loading the different faces.\r\n */\r\n public readonly supportCascades = false;\r\n\r\n /**\r\n * This returns if the loader support the current file information.\r\n * @param extension defines the file extension of the file being loaded\r\n * @returns true if the loader can load the specified file\r\n */\r\n public canLoad(extension: string): boolean {\r\n return extension.endsWith(\".basis\");\r\n }\r\n\r\n /**\r\n * Uploads the cube texture data to the WebGL texture. It has already been bound.\r\n * @param data contains the texture data\r\n * @param texture defines the BabylonJS internal texture\r\n * @param createPolynomials will be true if polynomials have been requested\r\n * @param onLoad defines the callback to trigger once the texture is ready\r\n * @param onError defines the callback to trigger in case of error\r\n */\r\n public loadCubeData(\r\n data: ArrayBufferView | ArrayBufferView[],\r\n texture: InternalTexture,\r\n createPolynomials: boolean,\r\n onLoad: Nullable<(data?: any) => void>,\r\n onError: Nullable<(message?: string, exception?: any) => void>\r\n ): void {\r\n if (Array.isArray(data)) {\r\n return;\r\n }\r\n const caps = texture.getEngine().getCaps();\r\n const transcodeConfig = {\r\n supportedCompressionFormats: {\r\n etc1: caps.etc1 ? true : false,\r\n s3tc: caps.s3tc ? true : false,\r\n pvrtc: caps.pvrtc ? true : false,\r\n etc2: caps.etc2 ? true : false,\r\n astc: caps.astc ? true : false,\r\n bc7: caps.bptc ? true : false,\r\n },\r\n };\r\n TranscodeAsync(data, transcodeConfig)\r\n .then((result) => {\r\n const hasMipmap = result.fileInfo.images[0].levels.length > 1 && texture.generateMipMaps;\r\n LoadTextureFromTranscodeResult(texture, result);\r\n (texture.getEngine() as Engine)._setCubeMapTextureParams(texture, hasMipmap);\r\n texture.isReady = true;\r\n texture.onLoadedObservable.notifyObservers(texture);\r\n texture.onLoadedObservable.clear();\r\n if (onLoad) {\r\n onLoad();\r\n }\r\n })\r\n .catch((err) => {\r\n const errorMessage = \"Failed to transcode Basis file, transcoding may not be supported on this device\";\r\n Tools.Warn(errorMessage);\r\n texture.isReady = true;\r\n if (onError) {\r\n onError(err);\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Uploads the 2D texture data to the WebGL texture. It has already been bound once in the callback.\r\n * @param data contains the texture data\r\n * @param texture defines the BabylonJS internal texture\r\n * @param callback defines the method to call once ready to upload\r\n */\r\n public loadData(\r\n data: ArrayBufferView,\r\n texture: InternalTexture,\r\n callback: (width: number, height: number, loadMipmap: boolean, isCompressed: boolean, done: () => void, failedLoading?: boolean) => void\r\n ): void {\r\n const caps = texture.getEngine().getCaps();\r\n const transcodeConfig = {\r\n supportedCompressionFormats: {\r\n etc1: caps.etc1 ? true : false,\r\n s3tc: caps.s3tc ? true : false,\r\n pvrtc: caps.pvrtc ? true : false,\r\n etc2: caps.etc2 ? true : false,\r\n astc: caps.astc ? true : false,\r\n bc7: caps.bptc ? true : false,\r\n },\r\n };\r\n TranscodeAsync(data, transcodeConfig)\r\n .then((result) => {\r\n const rootImage = result.fileInfo.images[0].levels[0];\r\n const hasMipmap = result.fileInfo.images[0].levels.length > 1 && texture.generateMipMaps;\r\n callback(rootImage.width, rootImage.height, hasMipmap, result.format !== -1, () => {\r\n LoadTextureFromTranscodeResult(texture, result);\r\n });\r\n })\r\n .catch((err) => {\r\n Tools.Warn(\"Failed to transcode Basis file, transcoding may not be supported on this device\");\r\n Tools.Warn(`Failed to transcode Basis file: ${err}`);\r\n callback(0, 0, false, false, () => {}, true);\r\n });\r\n }\r\n}\r\n\r\n// Register the loader.\r\nEngine._TextureLoaders.push(new _BasisTextureLoader());\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\n\nconst name = \"noisePixelShader\";\nconst shader = `uniform float brightness;uniform float persistence;uniform float timeScale;varying vec2 vUV;vec2 hash22(vec2 p)\n{p=p*mat2(127.1,311.7,269.5,183.3);p=-1.0+2.0*fract(sin(p)*43758.5453123);return sin(p*6.283+timeScale);}\nfloat interpolationNoise(vec2 p)\n{vec2 pi=floor(p);vec2 pf=p-pi;vec2 w=pf*pf*(3.-2.*pf);float f00=dot(hash22(pi+vec2(.0,.0)),pf-vec2(.0,.0));float f01=dot(hash22(pi+vec2(.0,1.)),pf-vec2(.0,1.));float f10=dot(hash22(pi+vec2(1.0,0.)),pf-vec2(1.0,0.));float f11=dot(hash22(pi+vec2(1.0,1.)),pf-vec2(1.0,1.));float xm1=mix(f00,f10,w.x);float xm2=mix(f01,f11,w.x);float ym=mix(xm1,xm2,w.y); \nreturn ym;}\nfloat perlinNoise2D(float x,float y)\n{float sum=0.0;float frequency=0.0;float amplitude=0.0;for(int i=0; i = EngineStore.LastCreatedScene, fallbackTexture?: Texture, generateMipMaps?: boolean) {\r\n super(name, size, \"noise\", scene, fallbackTexture, generateMipMaps);\r\n this.autoClear = false;\r\n this._updateShaderUniforms();\r\n }\r\n\r\n private _updateShaderUniforms() {\r\n const scene = this.getScene();\r\n\r\n if (!scene) {\r\n return;\r\n }\r\n\r\n this.time += scene.getAnimationRatio() * this.animationSpeedFactor * 0.01;\r\n\r\n this.setFloat(\"brightness\", this.brightness);\r\n this.setFloat(\"persistence\", this.persistence);\r\n this.setFloat(\"timeScale\", this.time);\r\n }\r\n\r\n protected _getDefines(): string {\r\n return \"#define OCTAVES \" + (this.octaves | 0);\r\n }\r\n\r\n /**\r\n * Generate the current state of the procedural texture\r\n * @param useCameraPostProcess\r\n */\r\n public render(useCameraPostProcess?: boolean) {\r\n this._updateShaderUniforms();\r\n super.render(useCameraPostProcess);\r\n }\r\n\r\n /**\r\n * Serializes this noise procedural texture\r\n * @returns a serialized noise procedural texture object\r\n */\r\n public serialize(): any {\r\n const serializationObject: any = {};\r\n serializationObject.customType = \"BABYLON.NoiseProceduralTexture\";\r\n\r\n serializationObject.brightness = this.brightness;\r\n serializationObject.octaves = this.octaves;\r\n serializationObject.persistence = this.persistence;\r\n serializationObject.animationSpeedFactor = this.animationSpeedFactor;\r\n serializationObject.size = this.getSize().width;\r\n serializationObject.generateMipMaps = this._generateMipMaps;\r\n serializationObject.time = this.time;\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Clone the texture.\r\n * @returns the cloned texture\r\n */\r\n public clone(): NoiseProceduralTexture {\r\n const textureSize = this.getSize();\r\n const newTexture = new NoiseProceduralTexture(\r\n this.name,\r\n textureSize.width,\r\n this.getScene(),\r\n this._fallbackTexture ? this._fallbackTexture : undefined,\r\n this._generateMipMaps\r\n );\r\n\r\n // Base texture\r\n newTexture.hasAlpha = this.hasAlpha;\r\n newTexture.level = this.level;\r\n\r\n // RenderTarget Texture\r\n newTexture.coordinatesMode = this.coordinatesMode;\r\n\r\n // Noise Specifics\r\n newTexture.brightness = this.brightness;\r\n newTexture.octaves = this.octaves;\r\n newTexture.persistence = this.persistence;\r\n newTexture.animationSpeedFactor = this.animationSpeedFactor;\r\n newTexture.time = this.time;\r\n\r\n return newTexture;\r\n }\r\n\r\n /**\r\n * Creates a NoiseProceduralTexture from parsed noise procedural texture data\r\n * @param parsedTexture defines parsed texture data\r\n * @param scene defines the current scene\r\n * @returns a parsed NoiseProceduralTexture\r\n */\r\n public static Parse(parsedTexture: any, scene: Scene): NoiseProceduralTexture {\r\n const texture = new NoiseProceduralTexture(parsedTexture.name, parsedTexture.size, scene, undefined, parsedTexture.generateMipMaps);\r\n\r\n texture.brightness = parsedTexture.brightness;\r\n texture.octaves = parsedTexture.octaves;\r\n texture.persistence = parsedTexture.persistence;\r\n texture.animationSpeedFactor = parsedTexture.animationSpeedFactor;\r\n texture.time = parsedTexture.time ?? 0;\r\n\r\n return texture;\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.NoiseProceduralTexture\", NoiseProceduralTexture);\r\n","import type { Scene } from \"../../scene\";\r\nimport { Plane } from \"../../Maths/math.plane\";\r\nimport { RenderTargetTexture } from \"../../Materials/Textures/renderTargetTexture\";\r\n/**\r\n * Creates a refraction texture used by refraction channel of the standard material.\r\n * It is like a mirror but to see through a material.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#refractiontexture\r\n */\r\nexport class RefractionTexture extends RenderTargetTexture {\r\n /**\r\n * Define the reflection plane we want to use. The refractionPlane is usually set to the constructed refractor.\r\n * It is possible to directly set the refractionPlane by directly using a Plane(a, b, c, d) where a, b and c give the plane normal vector (a, b, c) and d is a scalar displacement from the refractionPlane to the origin. However in all but the very simplest of situations it is more straight forward to set it to the refractor as stated in the doc.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#refraction\r\n */\r\n public refractionPlane = new Plane(0, 1, 0, 1);\r\n\r\n /**\r\n * Define how deep under the surface we should see.\r\n */\r\n public depth = 2.0;\r\n\r\n /**\r\n * Creates a refraction texture used by refraction channel of the standard material.\r\n * It is like a mirror but to see through a material.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/materials/using/reflectionTexture#refraction\r\n * @param name Define the texture name\r\n * @param size Define the size of the underlying texture\r\n * @param scene Define the scene the refraction belongs to\r\n * @param generateMipMaps Define if we need to generate mips level for the refraction\r\n */\r\n constructor(name: string, size: number, scene?: Scene, generateMipMaps?: boolean) {\r\n super(name, size, scene, generateMipMaps, true);\r\n\r\n this.onBeforeRenderObservable.add(() => {\r\n this.getScene()!.clipPlane = this.refractionPlane;\r\n });\r\n\r\n this.onAfterRenderObservable.add(() => {\r\n this.getScene()!.clipPlane = null;\r\n });\r\n }\r\n\r\n /**\r\n * Clone the refraction texture.\r\n * @returns the cloned texture\r\n */\r\n public clone(): RefractionTexture {\r\n const scene = this.getScene();\r\n\r\n if (!scene) {\r\n return this;\r\n }\r\n\r\n const textureSize = this.getSize();\r\n const newTexture = new RefractionTexture(this.name, textureSize.width, scene, this._generateMipMaps);\r\n\r\n // Base texture\r\n newTexture.hasAlpha = this.hasAlpha;\r\n newTexture.level = this.level;\r\n\r\n // Refraction Texture\r\n newTexture.refractionPlane = this.refractionPlane.clone();\r\n if (this.renderList) {\r\n newTexture.renderList = this.renderList.slice(0);\r\n }\r\n newTexture.depth = this.depth;\r\n\r\n return newTexture;\r\n }\r\n\r\n /**\r\n * Serialize the texture to a JSON representation you could use in Parse later on\r\n * @returns the serialized JSON representation\r\n */\r\n public serialize(): any {\r\n if (!this.name) {\r\n return null;\r\n }\r\n\r\n const serializationObject = super.serialize();\r\n\r\n serializationObject.mirrorPlane = this.refractionPlane.asArray();\r\n serializationObject.depth = this.depth;\r\n\r\n return serializationObject;\r\n }\r\n}\r\n","// DEFLATE is a complex format; to read this code, you should probably check the RFC first:\n// https://tools.ietf.org/html/rfc1951\n// You may also wish to take a look at the guide I made about this program:\n// https://gist.github.com/101arrowz/253f31eb5abc3d9275ab943003ffecad\n// Some of the following code is similar to that of UZIP.js:\n// https://github.com/photopea/UZIP.js\n// However, the vast majority of the codebase has diverged from UZIP.js to increase performance and reduce bundle size.\n// Sometimes 0 will appear where -1 would be more appropriate. This is because using a uint\n// is better for memory in most engines (I *think*).\nvar ch2 = {};\nvar wk = (function (c, id, msg, transfer, cb) {\n var w = new Worker(ch2[id] || (ch2[id] = URL.createObjectURL(new Blob([\n c + ';addEventListener(\"error\",function(e){e=e.error;postMessage({$e$:[e.message,e.code,e.stack]})})'\n ], { type: 'text/javascript' }))));\n w.onmessage = function (e) {\n var d = e.data, ed = d.$e$;\n if (ed) {\n var err = new Error(ed[0]);\n err['code'] = ed[1];\n err.stack = ed[2];\n cb(err, null);\n }\n else\n cb(null, d);\n };\n w.postMessage(msg, transfer);\n return w;\n});\n\n// aliases for shorter compressed code (most minifers don't do this)\nvar u8 = Uint8Array, u16 = Uint16Array, u32 = Uint32Array;\n// fixed length extra bits\nvar fleb = new u8([0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 2, 2, 2, 2, 3, 3, 3, 3, 4, 4, 4, 4, 5, 5, 5, 5, 0, /* unused */ 0, 0, /* impossible */ 0]);\n// fixed distance extra bits\n// see fleb note\nvar fdeb = new u8([0, 0, 0, 0, 1, 1, 2, 2, 3, 3, 4, 4, 5, 5, 6, 6, 7, 7, 8, 8, 9, 9, 10, 10, 11, 11, 12, 12, 13, 13, /* unused */ 0, 0]);\n// code length index map\nvar clim = new u8([16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15]);\n// get base, reverse index map from extra bits\nvar freb = function (eb, start) {\n var b = new u16(31);\n for (var i = 0; i < 31; ++i) {\n b[i] = start += 1 << eb[i - 1];\n }\n // numbers here are at max 18 bits\n var r = new u32(b[30]);\n for (var i = 1; i < 30; ++i) {\n for (var j = b[i]; j < b[i + 1]; ++j) {\n r[j] = ((j - b[i]) << 5) | i;\n }\n }\n return [b, r];\n};\nvar _a = freb(fleb, 2), fl = _a[0], revfl = _a[1];\n// we can ignore the fact that the other numbers are wrong; they never happen anyway\nfl[28] = 258, revfl[258] = 28;\nvar _b = freb(fdeb, 0), fd = _b[0], revfd = _b[1];\n// map of value to reverse (assuming 16 bits)\nvar rev = new u16(32768);\nfor (var i = 0; i < 32768; ++i) {\n // reverse table algorithm from SO\n var x = ((i & 0xAAAA) >>> 1) | ((i & 0x5555) << 1);\n x = ((x & 0xCCCC) >>> 2) | ((x & 0x3333) << 2);\n x = ((x & 0xF0F0) >>> 4) | ((x & 0x0F0F) << 4);\n rev[i] = (((x & 0xFF00) >>> 8) | ((x & 0x00FF) << 8)) >>> 1;\n}\n// create huffman tree from u8 \"map\": index -> code length for code index\n// mb (max bits) must be at most 15\n// TODO: optimize/split up?\nvar hMap = (function (cd, mb, r) {\n var s = cd.length;\n // index\n var i = 0;\n // u16 \"map\": index -> # of codes with bit length = index\n var l = new u16(mb);\n // length of cd must be 288 (total # of codes)\n for (; i < s; ++i) {\n if (cd[i])\n ++l[cd[i] - 1];\n }\n // u16 \"map\": index -> minimum code for bit length = index\n var le = new u16(mb);\n for (i = 0; i < mb; ++i) {\n le[i] = (le[i - 1] + l[i - 1]) << 1;\n }\n var co;\n if (r) {\n // u16 \"map\": index -> number of actual bits, symbol for code\n co = new u16(1 << mb);\n // bits to remove for reverser\n var rvb = 15 - mb;\n for (i = 0; i < s; ++i) {\n // ignore 0 lengths\n if (cd[i]) {\n // num encoding both symbol and bits read\n var sv = (i << 4) | cd[i];\n // free bits\n var r_1 = mb - cd[i];\n // start value\n var v = le[cd[i] - 1]++ << r_1;\n // m is end value\n for (var m = v | ((1 << r_1) - 1); v <= m; ++v) {\n // every 16 bit value starting with the code yields the same result\n co[rev[v] >>> rvb] = sv;\n }\n }\n }\n }\n else {\n co = new u16(s);\n for (i = 0; i < s; ++i) {\n if (cd[i]) {\n co[i] = rev[le[cd[i] - 1]++] >>> (15 - cd[i]);\n }\n }\n }\n return co;\n});\n// fixed length tree\nvar flt = new u8(288);\nfor (var i = 0; i < 144; ++i)\n flt[i] = 8;\nfor (var i = 144; i < 256; ++i)\n flt[i] = 9;\nfor (var i = 256; i < 280; ++i)\n flt[i] = 7;\nfor (var i = 280; i < 288; ++i)\n flt[i] = 8;\n// fixed distance tree\nvar fdt = new u8(32);\nfor (var i = 0; i < 32; ++i)\n fdt[i] = 5;\n// fixed length map\nvar flm = /*#__PURE__*/ hMap(flt, 9, 0), flrm = /*#__PURE__*/ hMap(flt, 9, 1);\n// fixed distance map\nvar fdm = /*#__PURE__*/ hMap(fdt, 5, 0), fdrm = /*#__PURE__*/ hMap(fdt, 5, 1);\n// find max of array\nvar max = function (a) {\n var m = a[0];\n for (var i = 1; i < a.length; ++i) {\n if (a[i] > m)\n m = a[i];\n }\n return m;\n};\n// read d, starting at bit p and mask with m\nvar bits = function (d, p, m) {\n var o = (p / 8) | 0;\n return ((d[o] | (d[o + 1] << 8)) >> (p & 7)) & m;\n};\n// read d, starting at bit p continuing for at least 16 bits\nvar bits16 = function (d, p) {\n var o = (p / 8) | 0;\n return ((d[o] | (d[o + 1] << 8) | (d[o + 2] << 16)) >> (p & 7));\n};\n// get end of byte\nvar shft = function (p) { return ((p + 7) / 8) | 0; };\n// typed array slice - allows garbage collector to free original reference,\n// while being more compatible than .slice\nvar slc = function (v, s, e) {\n if (s == null || s < 0)\n s = 0;\n if (e == null || e > v.length)\n e = v.length;\n // can't use .constructor in case user-supplied\n var n = new (v instanceof u16 ? u16 : v instanceof u32 ? u32 : u8)(e - s);\n n.set(v.subarray(s, e));\n return n;\n};\n/**\n * Codes for errors generated within this library\n */\nexport var FlateErrorCode = {\n UnexpectedEOF: 0,\n InvalidBlockType: 1,\n InvalidLengthLiteral: 2,\n InvalidDistance: 3,\n StreamFinished: 4,\n NoStreamHandler: 5,\n InvalidHeader: 6,\n NoCallback: 7,\n InvalidUTF8: 8,\n ExtraFieldTooLong: 9,\n InvalidDate: 10,\n FilenameTooLong: 11,\n StreamFinishing: 12,\n InvalidZipData: 13,\n UnknownCompressionMethod: 14\n};\n// error codes\nvar ec = [\n 'unexpected EOF',\n 'invalid block type',\n 'invalid length/literal',\n 'invalid distance',\n 'stream finished',\n 'no stream handler',\n ,\n 'no callback',\n 'invalid UTF-8 data',\n 'extra field too long',\n 'date not in range 1980-2099',\n 'filename too long',\n 'stream finishing',\n 'invalid zip data'\n // determined by unknown compression method\n];\n;\nvar err = function (ind, msg, nt) {\n var e = new Error(msg || ec[ind]);\n e.code = ind;\n if (Error.captureStackTrace)\n Error.captureStackTrace(e, err);\n if (!nt)\n throw e;\n return e;\n};\n// expands raw DEFLATE data\nvar inflt = function (dat, buf, st) {\n // source length\n var sl = dat.length;\n if (!sl || (st && st.f && !st.l))\n return buf || new u8(0);\n // have to estimate size\n var noBuf = !buf || st;\n // no state\n var noSt = !st || st.i;\n if (!st)\n st = {};\n // Assumes roughly 33% compression ratio average\n if (!buf)\n buf = new u8(sl * 3);\n // ensure buffer can fit at least l elements\n var cbuf = function (l) {\n var bl = buf.length;\n // need to increase size to fit\n if (l > bl) {\n // Double or set to necessary, whichever is greater\n var nbuf = new u8(Math.max(bl * 2, l));\n nbuf.set(buf);\n buf = nbuf;\n }\n };\n // last chunk bitpos bytes\n var final = st.f || 0, pos = st.p || 0, bt = st.b || 0, lm = st.l, dm = st.d, lbt = st.m, dbt = st.n;\n // total bits\n var tbts = sl * 8;\n do {\n if (!lm) {\n // BFINAL - this is only 1 when last chunk is next\n final = bits(dat, pos, 1);\n // type: 0 = no compression, 1 = fixed huffman, 2 = dynamic huffman\n var type = bits(dat, pos + 1, 3);\n pos += 3;\n if (!type) {\n // go to end of byte boundary\n var s = shft(pos) + 4, l = dat[s - 4] | (dat[s - 3] << 8), t = s + l;\n if (t > sl) {\n if (noSt)\n err(0);\n break;\n }\n // ensure size\n if (noBuf)\n cbuf(bt + l);\n // Copy over uncompressed data\n buf.set(dat.subarray(s, t), bt);\n // Get new bitpos, update byte count\n st.b = bt += l, st.p = pos = t * 8, st.f = final;\n continue;\n }\n else if (type == 1)\n lm = flrm, dm = fdrm, lbt = 9, dbt = 5;\n else if (type == 2) {\n // literal lengths\n var hLit = bits(dat, pos, 31) + 257, hcLen = bits(dat, pos + 10, 15) + 4;\n var tl = hLit + bits(dat, pos + 5, 31) + 1;\n pos += 14;\n // length+distance tree\n var ldt = new u8(tl);\n // code length tree\n var clt = new u8(19);\n for (var i = 0; i < hcLen; ++i) {\n // use index map to get real code\n clt[clim[i]] = bits(dat, pos + i * 3, 7);\n }\n pos += hcLen * 3;\n // code lengths bits\n var clb = max(clt), clbmsk = (1 << clb) - 1;\n // code lengths map\n var clm = hMap(clt, clb, 1);\n for (var i = 0; i < tl;) {\n var r = clm[bits(dat, pos, clbmsk)];\n // bits read\n pos += r & 15;\n // symbol\n var s = r >>> 4;\n // code length to copy\n if (s < 16) {\n ldt[i++] = s;\n }\n else {\n // copy count\n var c = 0, n = 0;\n if (s == 16)\n n = 3 + bits(dat, pos, 3), pos += 2, c = ldt[i - 1];\n else if (s == 17)\n n = 3 + bits(dat, pos, 7), pos += 3;\n else if (s == 18)\n n = 11 + bits(dat, pos, 127), pos += 7;\n while (n--)\n ldt[i++] = c;\n }\n }\n // length tree distance tree\n var lt = ldt.subarray(0, hLit), dt = ldt.subarray(hLit);\n // max length bits\n lbt = max(lt);\n // max dist bits\n dbt = max(dt);\n lm = hMap(lt, lbt, 1);\n dm = hMap(dt, dbt, 1);\n }\n else\n err(1);\n if (pos > tbts) {\n if (noSt)\n err(0);\n break;\n }\n }\n // Make sure the buffer can hold this + the largest possible addition\n // Maximum chunk size (practically, theoretically infinite) is 2^17;\n if (noBuf)\n cbuf(bt + 131072);\n var lms = (1 << lbt) - 1, dms = (1 << dbt) - 1;\n var lpos = pos;\n for (;; lpos = pos) {\n // bits read, code\n var c = lm[bits16(dat, pos) & lms], sym = c >>> 4;\n pos += c & 15;\n if (pos > tbts) {\n if (noSt)\n err(0);\n break;\n }\n if (!c)\n err(2);\n if (sym < 256)\n buf[bt++] = sym;\n else if (sym == 256) {\n lpos = pos, lm = null;\n break;\n }\n else {\n var add = sym - 254;\n // no extra bits needed if less\n if (sym > 264) {\n // index\n var i = sym - 257, b = fleb[i];\n add = bits(dat, pos, (1 << b) - 1) + fl[i];\n pos += b;\n }\n // dist\n var d = dm[bits16(dat, pos) & dms], dsym = d >>> 4;\n if (!d)\n err(3);\n pos += d & 15;\n var dt = fd[dsym];\n if (dsym > 3) {\n var b = fdeb[dsym];\n dt += bits16(dat, pos) & ((1 << b) - 1), pos += b;\n }\n if (pos > tbts) {\n if (noSt)\n err(0);\n break;\n }\n if (noBuf)\n cbuf(bt + 131072);\n var end = bt + add;\n for (; bt < end; bt += 4) {\n buf[bt] = buf[bt - dt];\n buf[bt + 1] = buf[bt + 1 - dt];\n buf[bt + 2] = buf[bt + 2 - dt];\n buf[bt + 3] = buf[bt + 3 - dt];\n }\n bt = end;\n }\n }\n st.l = lm, st.p = lpos, st.b = bt, st.f = final;\n if (lm)\n final = 1, st.m = lbt, st.d = dm, st.n = dbt;\n } while (!final);\n return bt == buf.length ? buf : slc(buf, 0, bt);\n};\n// starting at p, write the minimum number of bits that can hold v to d\nvar wbits = function (d, p, v) {\n v <<= p & 7;\n var o = (p / 8) | 0;\n d[o] |= v;\n d[o + 1] |= v >>> 8;\n};\n// starting at p, write the minimum number of bits (>8) that can hold v to d\nvar wbits16 = function (d, p, v) {\n v <<= p & 7;\n var o = (p / 8) | 0;\n d[o] |= v;\n d[o + 1] |= v >>> 8;\n d[o + 2] |= v >>> 16;\n};\n// creates code lengths from a frequency table\nvar hTree = function (d, mb) {\n // Need extra info to make a tree\n var t = [];\n for (var i = 0; i < d.length; ++i) {\n if (d[i])\n t.push({ s: i, f: d[i] });\n }\n var s = t.length;\n var t2 = t.slice();\n if (!s)\n return [et, 0];\n if (s == 1) {\n var v = new u8(t[0].s + 1);\n v[t[0].s] = 1;\n return [v, 1];\n }\n t.sort(function (a, b) { return a.f - b.f; });\n // after i2 reaches last ind, will be stopped\n // freq must be greater than largest possible number of symbols\n t.push({ s: -1, f: 25001 });\n var l = t[0], r = t[1], i0 = 0, i1 = 1, i2 = 2;\n t[0] = { s: -1, f: l.f + r.f, l: l, r: r };\n // efficient algorithm from UZIP.js\n // i0 is lookbehind, i2 is lookahead - after processing two low-freq\n // symbols that combined have high freq, will start processing i2 (high-freq,\n // non-composite) symbols instead\n // see https://reddit.com/r/photopea/comments/ikekht/uzipjs_questions/\n while (i1 != s - 1) {\n l = t[t[i0].f < t[i2].f ? i0++ : i2++];\n r = t[i0 != i1 && t[i0].f < t[i2].f ? i0++ : i2++];\n t[i1++] = { s: -1, f: l.f + r.f, l: l, r: r };\n }\n var maxSym = t2[0].s;\n for (var i = 1; i < s; ++i) {\n if (t2[i].s > maxSym)\n maxSym = t2[i].s;\n }\n // code lengths\n var tr = new u16(maxSym + 1);\n // max bits in tree\n var mbt = ln(t[i1 - 1], tr, 0);\n if (mbt > mb) {\n // more algorithms from UZIP.js\n // TODO: find out how this code works (debt)\n // ind debt\n var i = 0, dt = 0;\n // left cost\n var lft = mbt - mb, cst = 1 << lft;\n t2.sort(function (a, b) { return tr[b.s] - tr[a.s] || a.f - b.f; });\n for (; i < s; ++i) {\n var i2_1 = t2[i].s;\n if (tr[i2_1] > mb) {\n dt += cst - (1 << (mbt - tr[i2_1]));\n tr[i2_1] = mb;\n }\n else\n break;\n }\n dt >>>= lft;\n while (dt > 0) {\n var i2_2 = t2[i].s;\n if (tr[i2_2] < mb)\n dt -= 1 << (mb - tr[i2_2]++ - 1);\n else\n ++i;\n }\n for (; i >= 0 && dt; --i) {\n var i2_3 = t2[i].s;\n if (tr[i2_3] == mb) {\n --tr[i2_3];\n ++dt;\n }\n }\n mbt = mb;\n }\n return [new u8(tr), mbt];\n};\n// get the max length and assign length codes\nvar ln = function (n, l, d) {\n return n.s == -1\n ? Math.max(ln(n.l, l, d + 1), ln(n.r, l, d + 1))\n : (l[n.s] = d);\n};\n// length codes generation\nvar lc = function (c) {\n var s = c.length;\n // Note that the semicolon was intentional\n while (s && !c[--s])\n ;\n var cl = new u16(++s);\n // ind num streak\n var cli = 0, cln = c[0], cls = 1;\n var w = function (v) { cl[cli++] = v; };\n for (var i = 1; i <= s; ++i) {\n if (c[i] == cln && i != s)\n ++cls;\n else {\n if (!cln && cls > 2) {\n for (; cls > 138; cls -= 138)\n w(32754);\n if (cls > 2) {\n w(cls > 10 ? ((cls - 11) << 5) | 28690 : ((cls - 3) << 5) | 12305);\n cls = 0;\n }\n }\n else if (cls > 3) {\n w(cln), --cls;\n for (; cls > 6; cls -= 6)\n w(8304);\n if (cls > 2)\n w(((cls - 3) << 5) | 8208), cls = 0;\n }\n while (cls--)\n w(cln);\n cls = 1;\n cln = c[i];\n }\n }\n return [cl.subarray(0, cli), s];\n};\n// calculate the length of output from tree, code lengths\nvar clen = function (cf, cl) {\n var l = 0;\n for (var i = 0; i < cl.length; ++i)\n l += cf[i] * cl[i];\n return l;\n};\n// writes a fixed block\n// returns the new bit pos\nvar wfblk = function (out, pos, dat) {\n // no need to write 00 as type: TypedArray defaults to 0\n var s = dat.length;\n var o = shft(pos + 2);\n out[o] = s & 255;\n out[o + 1] = s >>> 8;\n out[o + 2] = out[o] ^ 255;\n out[o + 3] = out[o + 1] ^ 255;\n for (var i = 0; i < s; ++i)\n out[o + i + 4] = dat[i];\n return (o + 4 + s) * 8;\n};\n// writes a block\nvar wblk = function (dat, out, final, syms, lf, df, eb, li, bs, bl, p) {\n wbits(out, p++, final);\n ++lf[256];\n var _a = hTree(lf, 15), dlt = _a[0], mlb = _a[1];\n var _b = hTree(df, 15), ddt = _b[0], mdb = _b[1];\n var _c = lc(dlt), lclt = _c[0], nlc = _c[1];\n var _d = lc(ddt), lcdt = _d[0], ndc = _d[1];\n var lcfreq = new u16(19);\n for (var i = 0; i < lclt.length; ++i)\n lcfreq[lclt[i] & 31]++;\n for (var i = 0; i < lcdt.length; ++i)\n lcfreq[lcdt[i] & 31]++;\n var _e = hTree(lcfreq, 7), lct = _e[0], mlcb = _e[1];\n var nlcc = 19;\n for (; nlcc > 4 && !lct[clim[nlcc - 1]]; --nlcc)\n ;\n var flen = (bl + 5) << 3;\n var ftlen = clen(lf, flt) + clen(df, fdt) + eb;\n var dtlen = clen(lf, dlt) + clen(df, ddt) + eb + 14 + 3 * nlcc + clen(lcfreq, lct) + (2 * lcfreq[16] + 3 * lcfreq[17] + 7 * lcfreq[18]);\n if (flen <= ftlen && flen <= dtlen)\n return wfblk(out, p, dat.subarray(bs, bs + bl));\n var lm, ll, dm, dl;\n wbits(out, p, 1 + (dtlen < ftlen)), p += 2;\n if (dtlen < ftlen) {\n lm = hMap(dlt, mlb, 0), ll = dlt, dm = hMap(ddt, mdb, 0), dl = ddt;\n var llm = hMap(lct, mlcb, 0);\n wbits(out, p, nlc - 257);\n wbits(out, p + 5, ndc - 1);\n wbits(out, p + 10, nlcc - 4);\n p += 14;\n for (var i = 0; i < nlcc; ++i)\n wbits(out, p + 3 * i, lct[clim[i]]);\n p += 3 * nlcc;\n var lcts = [lclt, lcdt];\n for (var it = 0; it < 2; ++it) {\n var clct = lcts[it];\n for (var i = 0; i < clct.length; ++i) {\n var len = clct[i] & 31;\n wbits(out, p, llm[len]), p += lct[len];\n if (len > 15)\n wbits(out, p, (clct[i] >>> 5) & 127), p += clct[i] >>> 12;\n }\n }\n }\n else {\n lm = flm, ll = flt, dm = fdm, dl = fdt;\n }\n for (var i = 0; i < li; ++i) {\n if (syms[i] > 255) {\n var len = (syms[i] >>> 18) & 31;\n wbits16(out, p, lm[len + 257]), p += ll[len + 257];\n if (len > 7)\n wbits(out, p, (syms[i] >>> 23) & 31), p += fleb[len];\n var dst = syms[i] & 31;\n wbits16(out, p, dm[dst]), p += dl[dst];\n if (dst > 3)\n wbits16(out, p, (syms[i] >>> 5) & 8191), p += fdeb[dst];\n }\n else {\n wbits16(out, p, lm[syms[i]]), p += ll[syms[i]];\n }\n }\n wbits16(out, p, lm[256]);\n return p + ll[256];\n};\n// deflate options (nice << 13) | chain\nvar deo = /*#__PURE__*/ new u32([65540, 131080, 131088, 131104, 262176, 1048704, 1048832, 2114560, 2117632]);\n// empty\nvar et = /*#__PURE__*/ new u8(0);\n// compresses data into a raw DEFLATE buffer\nvar dflt = function (dat, lvl, plvl, pre, post, lst) {\n var s = dat.length;\n var o = new u8(pre + s + 5 * (1 + Math.ceil(s / 7000)) + post);\n // writing to this writes to the output buffer\n var w = o.subarray(pre, o.length - post);\n var pos = 0;\n if (!lvl || s < 8) {\n for (var i = 0; i <= s; i += 65535) {\n // end\n var e = i + 65535;\n if (e >= s) {\n // write final block\n w[pos >> 3] = lst;\n }\n pos = wfblk(w, pos + 1, dat.subarray(i, e));\n }\n }\n else {\n var opt = deo[lvl - 1];\n var n = opt >>> 13, c = opt & 8191;\n var msk_1 = (1 << plvl) - 1;\n // prev 2-byte val map curr 2-byte val map\n var prev = new u16(32768), head = new u16(msk_1 + 1);\n var bs1_1 = Math.ceil(plvl / 3), bs2_1 = 2 * bs1_1;\n var hsh = function (i) { return (dat[i] ^ (dat[i + 1] << bs1_1) ^ (dat[i + 2] << bs2_1)) & msk_1; };\n // 24576 is an arbitrary number of maximum symbols per block\n // 424 buffer for last block\n var syms = new u32(25000);\n // length/literal freq distance freq\n var lf = new u16(288), df = new u16(32);\n // l/lcnt exbits index l/lind waitdx bitpos\n var lc_1 = 0, eb = 0, i = 0, li = 0, wi = 0, bs = 0;\n for (; i < s; ++i) {\n // hash value\n // deopt when i > s - 3 - at end, deopt acceptable\n var hv = hsh(i);\n // index mod 32768 previous index mod\n var imod = i & 32767, pimod = head[hv];\n prev[imod] = pimod;\n head[hv] = imod;\n // We always should modify head and prev, but only add symbols if\n // this data is not yet processed (\"wait\" for wait index)\n if (wi <= i) {\n // bytes remaining\n var rem = s - i;\n if ((lc_1 > 7000 || li > 24576) && rem > 423) {\n pos = wblk(dat, w, 0, syms, lf, df, eb, li, bs, i - bs, pos);\n li = lc_1 = eb = 0, bs = i;\n for (var j = 0; j < 286; ++j)\n lf[j] = 0;\n for (var j = 0; j < 30; ++j)\n df[j] = 0;\n }\n // len dist chain\n var l = 2, d = 0, ch_1 = c, dif = (imod - pimod) & 32767;\n if (rem > 2 && hv == hsh(i - dif)) {\n var maxn = Math.min(n, rem) - 1;\n var maxd = Math.min(32767, i);\n // max possible length\n // not capped at dif because decompressors implement \"rolling\" index population\n var ml = Math.min(258, rem);\n while (dif <= maxd && --ch_1 && imod != pimod) {\n if (dat[i + l] == dat[i + l - dif]) {\n var nl = 0;\n for (; nl < ml && dat[i + nl] == dat[i + nl - dif]; ++nl)\n ;\n if (nl > l) {\n l = nl, d = dif;\n // break out early when we reach \"nice\" (we are satisfied enough)\n if (nl > maxn)\n break;\n // now, find the rarest 2-byte sequence within this\n // length of literals and search for that instead.\n // Much faster than just using the start\n var mmd = Math.min(dif, nl - 2);\n var md = 0;\n for (var j = 0; j < mmd; ++j) {\n var ti = (i - dif + j + 32768) & 32767;\n var pti = prev[ti];\n var cd = (ti - pti + 32768) & 32767;\n if (cd > md)\n md = cd, pimod = ti;\n }\n }\n }\n // check the previous match\n imod = pimod, pimod = prev[imod];\n dif += (imod - pimod + 32768) & 32767;\n }\n }\n // d will be nonzero only when a match was found\n if (d) {\n // store both dist and len data in one Uint32\n // Make sure this is recognized as a len/dist with 28th bit (2^28)\n syms[li++] = 268435456 | (revfl[l] << 18) | revfd[d];\n var lin = revfl[l] & 31, din = revfd[d] & 31;\n eb += fleb[lin] + fdeb[din];\n ++lf[257 + lin];\n ++df[din];\n wi = i + l;\n ++lc_1;\n }\n else {\n syms[li++] = dat[i];\n ++lf[dat[i]];\n }\n }\n }\n pos = wblk(dat, w, lst, syms, lf, df, eb, li, bs, i - bs, pos);\n // this is the easiest way to avoid needing to maintain state\n if (!lst && pos & 7)\n pos = wfblk(w, pos + 1, et);\n }\n return slc(o, 0, pre + shft(pos) + post);\n};\n// CRC32 table\nvar crct = /*#__PURE__*/ (function () {\n var t = new Int32Array(256);\n for (var i = 0; i < 256; ++i) {\n var c = i, k = 9;\n while (--k)\n c = ((c & 1) && -306674912) ^ (c >>> 1);\n t[i] = c;\n }\n return t;\n})();\n// CRC32\nvar crc = function () {\n var c = -1;\n return {\n p: function (d) {\n // closures have awful performance\n var cr = c;\n for (var i = 0; i < d.length; ++i)\n cr = crct[(cr & 255) ^ d[i]] ^ (cr >>> 8);\n c = cr;\n },\n d: function () { return ~c; }\n };\n};\n// Alder32\nvar adler = function () {\n var a = 1, b = 0;\n return {\n p: function (d) {\n // closures have awful performance\n var n = a, m = b;\n var l = d.length | 0;\n for (var i = 0; i != l;) {\n var e = Math.min(i + 2655, l);\n for (; i < e; ++i)\n m += n += d[i];\n n = (n & 65535) + 15 * (n >> 16), m = (m & 65535) + 15 * (m >> 16);\n }\n a = n, b = m;\n },\n d: function () {\n a %= 65521, b %= 65521;\n return (a & 255) << 24 | (a >>> 8) << 16 | (b & 255) << 8 | (b >>> 8);\n }\n };\n};\n;\n// deflate with opts\nvar dopt = function (dat, opt, pre, post, st) {\n return dflt(dat, opt.level == null ? 6 : opt.level, opt.mem == null ? Math.ceil(Math.max(8, Math.min(13, Math.log(dat.length))) * 1.5) : (12 + opt.mem), pre, post, !st);\n};\n// Walmart object spread\nvar mrg = function (a, b) {\n var o = {};\n for (var k in a)\n o[k] = a[k];\n for (var k in b)\n o[k] = b[k];\n return o;\n};\n// worker clone\n// This is possibly the craziest part of the entire codebase, despite how simple it may seem.\n// The only parameter to this function is a closure that returns an array of variables outside of the function scope.\n// We're going to try to figure out the variable names used in the closure as strings because that is crucial for workerization.\n// We will return an object mapping of true variable name to value (basically, the current scope as a JS object).\n// The reason we can't just use the original variable names is minifiers mangling the toplevel scope.\n// This took me three weeks to figure out how to do.\nvar wcln = function (fn, fnStr, td) {\n var dt = fn();\n var st = fn.toString();\n var ks = st.slice(st.indexOf('[') + 1, st.lastIndexOf(']')).replace(/ /g, '').split(',');\n for (var i = 0; i < dt.length; ++i) {\n var v = dt[i], k = ks[i];\n if (typeof v == 'function') {\n fnStr += ';' + k + '=';\n var st_1 = v.toString();\n if (v.prototype) {\n // for global objects\n if (st_1.indexOf('[native code]') != -1) {\n var spInd = st_1.indexOf(' ', 8) + 1;\n fnStr += st_1.slice(spInd, st_1.indexOf('(', spInd));\n }\n else {\n fnStr += st_1;\n for (var t in v.prototype)\n fnStr += ';' + k + '.prototype.' + t + '=' + v.prototype[t].toString();\n }\n }\n else\n fnStr += st_1;\n }\n else\n td[k] = v;\n }\n return [fnStr, td];\n};\nvar ch = [];\n// clone bufs\nvar cbfs = function (v) {\n var tl = [];\n for (var k in v) {\n if (v[k] instanceof u8 || v[k] instanceof u16 || v[k] instanceof u32)\n tl.push((v[k] = new v[k].constructor(v[k])).buffer);\n }\n return tl;\n};\n// use a worker to execute code\nvar wrkr = function (fns, init, id, cb) {\n var _a;\n if (!ch[id]) {\n var fnStr = '', td_1 = {}, m = fns.length - 1;\n for (var i = 0; i < m; ++i)\n _a = wcln(fns[i], fnStr, td_1), fnStr = _a[0], td_1 = _a[1];\n ch[id] = wcln(fns[m], fnStr, td_1);\n }\n var td = mrg({}, ch[id][1]);\n return wk(ch[id][0] + ';onmessage=function(e){for(var k in e.data)self[k]=e.data[k];onmessage=' + init.toString() + '}', id, td, cbfs(td), cb);\n};\n// base async inflate fn\nvar bInflt = function () { return [u8, u16, u32, fleb, fdeb, clim, fl, fd, flrm, fdrm, rev, ec, hMap, max, bits, bits16, shft, slc, err, inflt, inflateSync, pbf, gu8]; };\nvar bDflt = function () { return [u8, u16, u32, fleb, fdeb, clim, revfl, revfd, flm, flt, fdm, fdt, rev, deo, et, hMap, wbits, wbits16, hTree, ln, lc, clen, wfblk, wblk, shft, slc, dflt, dopt, deflateSync, pbf]; };\n// gzip extra\nvar gze = function () { return [gzh, gzhl, wbytes, crc, crct]; };\n// gunzip extra\nvar guze = function () { return [gzs, gzl]; };\n// zlib extra\nvar zle = function () { return [zlh, wbytes, adler]; };\n// unzlib extra\nvar zule = function () { return [zlv]; };\n// post buf\nvar pbf = function (msg) { return postMessage(msg, [msg.buffer]); };\n// get u8\nvar gu8 = function (o) { return o && o.size && new u8(o.size); };\n// async helper\nvar cbify = function (dat, opts, fns, init, id, cb) {\n var w = wrkr(fns, init, id, function (err, dat) {\n w.terminate();\n cb(err, dat);\n });\n w.postMessage([dat, opts], opts.consume ? [dat.buffer] : []);\n return function () { w.terminate(); };\n};\n// auto stream\nvar astrm = function (strm) {\n strm.ondata = function (dat, final) { return postMessage([dat, final], [dat.buffer]); };\n return function (ev) { return strm.push(ev.data[0], ev.data[1]); };\n};\n// async stream attach\nvar astrmify = function (fns, strm, opts, init, id) {\n var t;\n var w = wrkr(fns, init, id, function (err, dat) {\n if (err)\n w.terminate(), strm.ondata.call(strm, err);\n else {\n if (dat[1])\n w.terminate();\n strm.ondata.call(strm, err, dat[0], dat[1]);\n }\n });\n w.postMessage(opts);\n strm.push = function (d, f) {\n if (!strm.ondata)\n err(5);\n if (t)\n strm.ondata(err(4, 0, 1), null, !!f);\n w.postMessage([d, t = f], [d.buffer]);\n };\n strm.terminate = function () { w.terminate(); };\n};\n// read 2 bytes\nvar b2 = function (d, b) { return d[b] | (d[b + 1] << 8); };\n// read 4 bytes\nvar b4 = function (d, b) { return (d[b] | (d[b + 1] << 8) | (d[b + 2] << 16) | (d[b + 3] << 24)) >>> 0; };\nvar b8 = function (d, b) { return b4(d, b) + (b4(d, b + 4) * 4294967296); };\n// write bytes\nvar wbytes = function (d, b, v) {\n for (; v; ++b)\n d[b] = v, v >>>= 8;\n};\n// gzip header\nvar gzh = function (c, o) {\n var fn = o.filename;\n c[0] = 31, c[1] = 139, c[2] = 8, c[8] = o.level < 2 ? 4 : o.level == 9 ? 2 : 0, c[9] = 3; // assume Unix\n if (o.mtime != 0)\n wbytes(c, 4, Math.floor(new Date(o.mtime || Date.now()) / 1000));\n if (fn) {\n c[3] = 8;\n for (var i = 0; i <= fn.length; ++i)\n c[i + 10] = fn.charCodeAt(i);\n }\n};\n// gzip footer: -8 to -4 = CRC, -4 to -0 is length\n// gzip start\nvar gzs = function (d) {\n if (d[0] != 31 || d[1] != 139 || d[2] != 8)\n err(6, 'invalid gzip data');\n var flg = d[3];\n var st = 10;\n if (flg & 4)\n st += d[10] | (d[11] << 8) + 2;\n for (var zs = (flg >> 3 & 1) + (flg >> 4 & 1); zs > 0; zs -= !d[st++])\n ;\n return st + (flg & 2);\n};\n// gzip length\nvar gzl = function (d) {\n var l = d.length;\n return ((d[l - 4] | d[l - 3] << 8 | d[l - 2] << 16) | (d[l - 1] << 24)) >>> 0;\n};\n// gzip header length\nvar gzhl = function (o) { return 10 + ((o.filename && (o.filename.length + 1)) || 0); };\n// zlib header\nvar zlh = function (c, o) {\n var lv = o.level, fl = lv == 0 ? 0 : lv < 6 ? 1 : lv == 9 ? 3 : 2;\n c[0] = 120, c[1] = (fl << 6) | (fl ? (32 - 2 * fl) : 1);\n};\n// zlib valid\nvar zlv = function (d) {\n if ((d[0] & 15) != 8 || (d[0] >>> 4) > 7 || ((d[0] << 8 | d[1]) % 31))\n err(6, 'invalid zlib data');\n if (d[1] & 32)\n err(6, 'invalid zlib data: preset dictionaries not supported');\n};\nfunction AsyncCmpStrm(opts, cb) {\n if (!cb && typeof opts == 'function')\n cb = opts, opts = {};\n this.ondata = cb;\n return opts;\n}\n// zlib footer: -4 to -0 is Adler32\n/**\n * Streaming DEFLATE compression\n */\nvar Deflate = /*#__PURE__*/ (function () {\n function Deflate(opts, cb) {\n if (!cb && typeof opts == 'function')\n cb = opts, opts = {};\n this.ondata = cb;\n this.o = opts || {};\n }\n Deflate.prototype.p = function (c, f) {\n this.ondata(dopt(c, this.o, 0, 0, !f), f);\n };\n /**\n * Pushes a chunk to be deflated\n * @param chunk The chunk to push\n * @param final Whether this is the last chunk\n */\n Deflate.prototype.push = function (chunk, final) {\n if (!this.ondata)\n err(5);\n if (this.d)\n err(4);\n this.d = final;\n this.p(chunk, final || false);\n };\n return Deflate;\n}());\nexport { Deflate };\n/**\n * Asynchronous streaming DEFLATE compression\n */\nvar AsyncDeflate = /*#__PURE__*/ (function () {\n function AsyncDeflate(opts, cb) {\n astrmify([\n bDflt,\n function () { return [astrm, Deflate]; }\n ], this, AsyncCmpStrm.call(this, opts, cb), function (ev) {\n var strm = new Deflate(ev.data);\n onmessage = astrm(strm);\n }, 6);\n }\n return AsyncDeflate;\n}());\nexport { AsyncDeflate };\nexport function deflate(data, opts, cb) {\n if (!cb)\n cb = opts, opts = {};\n if (typeof cb != 'function')\n err(7);\n return cbify(data, opts, [\n bDflt,\n ], function (ev) { return pbf(deflateSync(ev.data[0], ev.data[1])); }, 0, cb);\n}\n/**\n * Compresses data with DEFLATE without any wrapper\n * @param data The data to compress\n * @param opts The compression options\n * @returns The deflated version of the data\n */\nexport function deflateSync(data, opts) {\n return dopt(data, opts || {}, 0, 0);\n}\n/**\n * Streaming DEFLATE decompression\n */\nvar Inflate = /*#__PURE__*/ (function () {\n /**\n * Creates an inflation stream\n * @param cb The callback to call whenever data is inflated\n */\n function Inflate(cb) {\n this.s = {};\n this.p = new u8(0);\n this.ondata = cb;\n }\n Inflate.prototype.e = function (c) {\n if (!this.ondata)\n err(5);\n if (this.d)\n err(4);\n var l = this.p.length;\n var n = new u8(l + c.length);\n n.set(this.p), n.set(c, l), this.p = n;\n };\n Inflate.prototype.c = function (final) {\n this.d = this.s.i = final || false;\n var bts = this.s.b;\n var dt = inflt(this.p, this.o, this.s);\n this.ondata(slc(dt, bts, this.s.b), this.d);\n this.o = slc(dt, this.s.b - 32768), this.s.b = this.o.length;\n this.p = slc(this.p, (this.s.p / 8) | 0), this.s.p &= 7;\n };\n /**\n * Pushes a chunk to be inflated\n * @param chunk The chunk to push\n * @param final Whether this is the final chunk\n */\n Inflate.prototype.push = function (chunk, final) {\n this.e(chunk), this.c(final);\n };\n return Inflate;\n}());\nexport { Inflate };\n/**\n * Asynchronous streaming DEFLATE decompression\n */\nvar AsyncInflate = /*#__PURE__*/ (function () {\n /**\n * Creates an asynchronous inflation stream\n * @param cb The callback to call whenever data is deflated\n */\n function AsyncInflate(cb) {\n this.ondata = cb;\n astrmify([\n bInflt,\n function () { return [astrm, Inflate]; }\n ], this, 0, function () {\n var strm = new Inflate();\n onmessage = astrm(strm);\n }, 7);\n }\n return AsyncInflate;\n}());\nexport { AsyncInflate };\nexport function inflate(data, opts, cb) {\n if (!cb)\n cb = opts, opts = {};\n if (typeof cb != 'function')\n err(7);\n return cbify(data, opts, [\n bInflt\n ], function (ev) { return pbf(inflateSync(ev.data[0], gu8(ev.data[1]))); }, 1, cb);\n}\n/**\n * Expands DEFLATE data with no wrapper\n * @param data The data to decompress\n * @param out Where to write the data. Saves memory if you know the decompressed size and provide an output buffer of that length.\n * @returns The decompressed version of the data\n */\nexport function inflateSync(data, out) {\n return inflt(data, out);\n}\n// before you yell at me for not just using extends, my reason is that TS inheritance is hard to workerize.\n/**\n * Streaming GZIP compression\n */\nvar Gzip = /*#__PURE__*/ (function () {\n function Gzip(opts, cb) {\n this.c = crc();\n this.l = 0;\n this.v = 1;\n Deflate.call(this, opts, cb);\n }\n /**\n * Pushes a chunk to be GZIPped\n * @param chunk The chunk to push\n * @param final Whether this is the last chunk\n */\n Gzip.prototype.push = function (chunk, final) {\n Deflate.prototype.push.call(this, chunk, final);\n };\n Gzip.prototype.p = function (c, f) {\n this.c.p(c);\n this.l += c.length;\n var raw = dopt(c, this.o, this.v && gzhl(this.o), f && 8, !f);\n if (this.v)\n gzh(raw, this.o), this.v = 0;\n if (f)\n wbytes(raw, raw.length - 8, this.c.d()), wbytes(raw, raw.length - 4, this.l);\n this.ondata(raw, f);\n };\n return Gzip;\n}());\nexport { Gzip };\n/**\n * Asynchronous streaming GZIP compression\n */\nvar AsyncGzip = /*#__PURE__*/ (function () {\n function AsyncGzip(opts, cb) {\n astrmify([\n bDflt,\n gze,\n function () { return [astrm, Deflate, Gzip]; }\n ], this, AsyncCmpStrm.call(this, opts, cb), function (ev) {\n var strm = new Gzip(ev.data);\n onmessage = astrm(strm);\n }, 8);\n }\n return AsyncGzip;\n}());\nexport { AsyncGzip };\nexport function gzip(data, opts, cb) {\n if (!cb)\n cb = opts, opts = {};\n if (typeof cb != 'function')\n err(7);\n return cbify(data, opts, [\n bDflt,\n gze,\n function () { return [gzipSync]; }\n ], function (ev) { return pbf(gzipSync(ev.data[0], ev.data[1])); }, 2, cb);\n}\n/**\n * Compresses data with GZIP\n * @param data The data to compress\n * @param opts The compression options\n * @returns The gzipped version of the data\n */\nexport function gzipSync(data, opts) {\n if (!opts)\n opts = {};\n var c = crc(), l = data.length;\n c.p(data);\n var d = dopt(data, opts, gzhl(opts), 8), s = d.length;\n return gzh(d, opts), wbytes(d, s - 8, c.d()), wbytes(d, s - 4, l), d;\n}\n/**\n * Streaming GZIP decompression\n */\nvar Gunzip = /*#__PURE__*/ (function () {\n /**\n * Creates a GUNZIP stream\n * @param cb The callback to call whenever data is inflated\n */\n function Gunzip(cb) {\n this.v = 1;\n Inflate.call(this, cb);\n }\n /**\n * Pushes a chunk to be GUNZIPped\n * @param chunk The chunk to push\n * @param final Whether this is the last chunk\n */\n Gunzip.prototype.push = function (chunk, final) {\n Inflate.prototype.e.call(this, chunk);\n if (this.v) {\n var s = this.p.length > 3 ? gzs(this.p) : 4;\n if (s >= this.p.length && !final)\n return;\n this.p = this.p.subarray(s), this.v = 0;\n }\n if (final) {\n if (this.p.length < 8)\n err(6, 'invalid gzip data');\n this.p = this.p.subarray(0, -8);\n }\n // necessary to prevent TS from using the closure value\n // This allows for workerization to function correctly\n Inflate.prototype.c.call(this, final);\n };\n return Gunzip;\n}());\nexport { Gunzip };\n/**\n * Asynchronous streaming GZIP decompression\n */\nvar AsyncGunzip = /*#__PURE__*/ (function () {\n /**\n * Creates an asynchronous GUNZIP stream\n * @param cb The callback to call whenever data is deflated\n */\n function AsyncGunzip(cb) {\n this.ondata = cb;\n astrmify([\n bInflt,\n guze,\n function () { return [astrm, Inflate, Gunzip]; }\n ], this, 0, function () {\n var strm = new Gunzip();\n onmessage = astrm(strm);\n }, 9);\n }\n return AsyncGunzip;\n}());\nexport { AsyncGunzip };\nexport function gunzip(data, opts, cb) {\n if (!cb)\n cb = opts, opts = {};\n if (typeof cb != 'function')\n err(7);\n return cbify(data, opts, [\n bInflt,\n guze,\n function () { return [gunzipSync]; }\n ], function (ev) { return pbf(gunzipSync(ev.data[0])); }, 3, cb);\n}\n/**\n * Expands GZIP data\n * @param data The data to decompress\n * @param out Where to write the data. GZIP already encodes the output size, so providing this doesn't save memory.\n * @returns The decompressed version of the data\n */\nexport function gunzipSync(data, out) {\n return inflt(data.subarray(gzs(data), -8), out || new u8(gzl(data)));\n}\n/**\n * Streaming Zlib compression\n */\nvar Zlib = /*#__PURE__*/ (function () {\n function Zlib(opts, cb) {\n this.c = adler();\n this.v = 1;\n Deflate.call(this, opts, cb);\n }\n /**\n * Pushes a chunk to be zlibbed\n * @param chunk The chunk to push\n * @param final Whether this is the last chunk\n */\n Zlib.prototype.push = function (chunk, final) {\n Deflate.prototype.push.call(this, chunk, final);\n };\n Zlib.prototype.p = function (c, f) {\n this.c.p(c);\n var raw = dopt(c, this.o, this.v && 2, f && 4, !f);\n if (this.v)\n zlh(raw, this.o), this.v = 0;\n if (f)\n wbytes(raw, raw.length - 4, this.c.d());\n this.ondata(raw, f);\n };\n return Zlib;\n}());\nexport { Zlib };\n/**\n * Asynchronous streaming Zlib compression\n */\nvar AsyncZlib = /*#__PURE__*/ (function () {\n function AsyncZlib(opts, cb) {\n astrmify([\n bDflt,\n zle,\n function () { return [astrm, Deflate, Zlib]; }\n ], this, AsyncCmpStrm.call(this, opts, cb), function (ev) {\n var strm = new Zlib(ev.data);\n onmessage = astrm(strm);\n }, 10);\n }\n return AsyncZlib;\n}());\nexport { AsyncZlib };\nexport function zlib(data, opts, cb) {\n if (!cb)\n cb = opts, opts = {};\n if (typeof cb != 'function')\n err(7);\n return cbify(data, opts, [\n bDflt,\n zle,\n function () { return [zlibSync]; }\n ], function (ev) { return pbf(zlibSync(ev.data[0], ev.data[1])); }, 4, cb);\n}\n/**\n * Compress data with Zlib\n * @param data The data to compress\n * @param opts The compression options\n * @returns The zlib-compressed version of the data\n */\nexport function zlibSync(data, opts) {\n if (!opts)\n opts = {};\n var a = adler();\n a.p(data);\n var d = dopt(data, opts, 2, 4);\n return zlh(d, opts), wbytes(d, d.length - 4, a.d()), d;\n}\n/**\n * Streaming Zlib decompression\n */\nvar Unzlib = /*#__PURE__*/ (function () {\n /**\n * Creates a Zlib decompression stream\n * @param cb The callback to call whenever data is inflated\n */\n function Unzlib(cb) {\n this.v = 1;\n Inflate.call(this, cb);\n }\n /**\n * Pushes a chunk to be unzlibbed\n * @param chunk The chunk to push\n * @param final Whether this is the last chunk\n */\n Unzlib.prototype.push = function (chunk, final) {\n Inflate.prototype.e.call(this, chunk);\n if (this.v) {\n if (this.p.length < 2 && !final)\n return;\n this.p = this.p.subarray(2), this.v = 0;\n }\n if (final) {\n if (this.p.length < 4)\n err(6, 'invalid zlib data');\n this.p = this.p.subarray(0, -4);\n }\n // necessary to prevent TS from using the closure value\n // This allows for workerization to function correctly\n Inflate.prototype.c.call(this, final);\n };\n return Unzlib;\n}());\nexport { Unzlib };\n/**\n * Asynchronous streaming Zlib decompression\n */\nvar AsyncUnzlib = /*#__PURE__*/ (function () {\n /**\n * Creates an asynchronous Zlib decompression stream\n * @param cb The callback to call whenever data is deflated\n */\n function AsyncUnzlib(cb) {\n this.ondata = cb;\n astrmify([\n bInflt,\n zule,\n function () { return [astrm, Inflate, Unzlib]; }\n ], this, 0, function () {\n var strm = new Unzlib();\n onmessage = astrm(strm);\n }, 11);\n }\n return AsyncUnzlib;\n}());\nexport { AsyncUnzlib };\nexport function unzlib(data, opts, cb) {\n if (!cb)\n cb = opts, opts = {};\n if (typeof cb != 'function')\n err(7);\n return cbify(data, opts, [\n bInflt,\n zule,\n function () { return [unzlibSync]; }\n ], function (ev) { return pbf(unzlibSync(ev.data[0], gu8(ev.data[1]))); }, 5, cb);\n}\n/**\n * Expands Zlib data\n * @param data The data to decompress\n * @param out Where to write the data. Saves memory if you know the decompressed size and provide an output buffer of that length.\n * @returns The decompressed version of the data\n */\nexport function unzlibSync(data, out) {\n return inflt((zlv(data), data.subarray(2, -4)), out);\n}\n// Default algorithm for compression (used because having a known output size allows faster decompression)\nexport { gzip as compress, AsyncGzip as AsyncCompress };\n// Default algorithm for compression (used because having a known output size allows faster decompression)\nexport { gzipSync as compressSync, Gzip as Compress };\n/**\n * Streaming GZIP, Zlib, or raw DEFLATE decompression\n */\nvar Decompress = /*#__PURE__*/ (function () {\n /**\n * Creates a decompression stream\n * @param cb The callback to call whenever data is decompressed\n */\n function Decompress(cb) {\n this.G = Gunzip;\n this.I = Inflate;\n this.Z = Unzlib;\n this.ondata = cb;\n }\n /**\n * Pushes a chunk to be decompressed\n * @param chunk The chunk to push\n * @param final Whether this is the last chunk\n */\n Decompress.prototype.push = function (chunk, final) {\n if (!this.ondata)\n err(5);\n if (!this.s) {\n if (this.p && this.p.length) {\n var n = new u8(this.p.length + chunk.length);\n n.set(this.p), n.set(chunk, this.p.length);\n }\n else\n this.p = chunk;\n if (this.p.length > 2) {\n var _this_1 = this;\n var cb = function () { _this_1.ondata.apply(_this_1, arguments); };\n this.s = (this.p[0] == 31 && this.p[1] == 139 && this.p[2] == 8)\n ? new this.G(cb)\n : ((this.p[0] & 15) != 8 || (this.p[0] >> 4) > 7 || ((this.p[0] << 8 | this.p[1]) % 31))\n ? new this.I(cb)\n : new this.Z(cb);\n this.s.push(this.p, final);\n this.p = null;\n }\n }\n else\n this.s.push(chunk, final);\n };\n return Decompress;\n}());\nexport { Decompress };\n/**\n * Asynchronous streaming GZIP, Zlib, or raw DEFLATE decompression\n */\nvar AsyncDecompress = /*#__PURE__*/ (function () {\n /**\n * Creates an asynchronous decompression stream\n * @param cb The callback to call whenever data is decompressed\n */\n function AsyncDecompress(cb) {\n this.G = AsyncGunzip;\n this.I = AsyncInflate;\n this.Z = AsyncUnzlib;\n this.ondata = cb;\n }\n /**\n * Pushes a chunk to be decompressed\n * @param chunk The chunk to push\n * @param final Whether this is the last chunk\n */\n AsyncDecompress.prototype.push = function (chunk, final) {\n Decompress.prototype.push.call(this, chunk, final);\n };\n return AsyncDecompress;\n}());\nexport { AsyncDecompress };\nexport function decompress(data, opts, cb) {\n if (!cb)\n cb = opts, opts = {};\n if (typeof cb != 'function')\n err(7);\n return (data[0] == 31 && data[1] == 139 && data[2] == 8)\n ? gunzip(data, opts, cb)\n : ((data[0] & 15) != 8 || (data[0] >> 4) > 7 || ((data[0] << 8 | data[1]) % 31))\n ? inflate(data, opts, cb)\n : unzlib(data, opts, cb);\n}\n/**\n * Expands compressed GZIP, Zlib, or raw DEFLATE data, automatically detecting the format\n * @param data The data to decompress\n * @param out Where to write the data. Saves memory if you know the decompressed size and provide an output buffer of that length.\n * @returns The decompressed version of the data\n */\nexport function decompressSync(data, out) {\n return (data[0] == 31 && data[1] == 139 && data[2] == 8)\n ? gunzipSync(data, out)\n : ((data[0] & 15) != 8 || (data[0] >> 4) > 7 || ((data[0] << 8 | data[1]) % 31))\n ? inflateSync(data, out)\n : unzlibSync(data, out);\n}\n// flatten a directory structure\nvar fltn = function (d, p, t, o) {\n for (var k in d) {\n var val = d[k], n = p + k;\n if (val instanceof u8)\n t[n] = [val, o];\n else if (Array.isArray(val))\n t[n] = [val[0], mrg(o, val[1])];\n else\n fltn(val, n + '/', t, o);\n }\n};\n// text encoder\nvar te = typeof TextEncoder != 'undefined' && /*#__PURE__*/ new TextEncoder();\n// text decoder\nvar td = typeof TextDecoder != 'undefined' && /*#__PURE__*/ new TextDecoder();\n// text decoder stream\nvar tds = 0;\ntry {\n td.decode(et, { stream: true });\n tds = 1;\n}\ncatch (e) { }\n// decode UTF8\nvar dutf8 = function (d) {\n for (var r = '', i = 0;;) {\n var c = d[i++];\n var eb = (c > 127) + (c > 223) + (c > 239);\n if (i + eb > d.length)\n return [r, slc(d, i - 1)];\n if (!eb)\n r += String.fromCharCode(c);\n else if (eb == 3) {\n c = ((c & 15) << 18 | (d[i++] & 63) << 12 | (d[i++] & 63) << 6 | (d[i++] & 63)) - 65536,\n r += String.fromCharCode(55296 | (c >> 10), 56320 | (c & 1023));\n }\n else if (eb & 1)\n r += String.fromCharCode((c & 31) << 6 | (d[i++] & 63));\n else\n r += String.fromCharCode((c & 15) << 12 | (d[i++] & 63) << 6 | (d[i++] & 63));\n }\n};\n/**\n * Streaming UTF-8 decoding\n */\nvar DecodeUTF8 = /*#__PURE__*/ (function () {\n /**\n * Creates a UTF-8 decoding stream\n * @param cb The callback to call whenever data is decoded\n */\n function DecodeUTF8(cb) {\n this.ondata = cb;\n if (tds)\n this.t = new TextDecoder();\n else\n this.p = et;\n }\n /**\n * Pushes a chunk to be decoded from UTF-8 binary\n * @param chunk The chunk to push\n * @param final Whether this is the last chunk\n */\n DecodeUTF8.prototype.push = function (chunk, final) {\n if (!this.ondata)\n err(5);\n final = !!final;\n if (this.t) {\n this.ondata(this.t.decode(chunk, { stream: true }), final);\n if (final) {\n if (this.t.decode().length)\n err(8);\n this.t = null;\n }\n return;\n }\n if (!this.p)\n err(4);\n var dat = new u8(this.p.length + chunk.length);\n dat.set(this.p);\n dat.set(chunk, this.p.length);\n var _a = dutf8(dat), ch = _a[0], np = _a[1];\n if (final) {\n if (np.length)\n err(8);\n this.p = null;\n }\n else\n this.p = np;\n this.ondata(ch, final);\n };\n return DecodeUTF8;\n}());\nexport { DecodeUTF8 };\n/**\n * Streaming UTF-8 encoding\n */\nvar EncodeUTF8 = /*#__PURE__*/ (function () {\n /**\n * Creates a UTF-8 decoding stream\n * @param cb The callback to call whenever data is encoded\n */\n function EncodeUTF8(cb) {\n this.ondata = cb;\n }\n /**\n * Pushes a chunk to be encoded to UTF-8\n * @param chunk The string data to push\n * @param final Whether this is the last chunk\n */\n EncodeUTF8.prototype.push = function (chunk, final) {\n if (!this.ondata)\n err(5);\n if (this.d)\n err(4);\n this.ondata(strToU8(chunk), this.d = final || false);\n };\n return EncodeUTF8;\n}());\nexport { EncodeUTF8 };\n/**\n * Converts a string into a Uint8Array for use with compression/decompression methods\n * @param str The string to encode\n * @param latin1 Whether or not to interpret the data as Latin-1. This should\n * not need to be true unless decoding a binary string.\n * @returns The string encoded in UTF-8/Latin-1 binary\n */\nexport function strToU8(str, latin1) {\n if (latin1) {\n var ar_1 = new u8(str.length);\n for (var i = 0; i < str.length; ++i)\n ar_1[i] = str.charCodeAt(i);\n return ar_1;\n }\n if (te)\n return te.encode(str);\n var l = str.length;\n var ar = new u8(str.length + (str.length >> 1));\n var ai = 0;\n var w = function (v) { ar[ai++] = v; };\n for (var i = 0; i < l; ++i) {\n if (ai + 5 > ar.length) {\n var n = new u8(ai + 8 + ((l - i) << 1));\n n.set(ar);\n ar = n;\n }\n var c = str.charCodeAt(i);\n if (c < 128 || latin1)\n w(c);\n else if (c < 2048)\n w(192 | (c >> 6)), w(128 | (c & 63));\n else if (c > 55295 && c < 57344)\n c = 65536 + (c & 1023 << 10) | (str.charCodeAt(++i) & 1023),\n w(240 | (c >> 18)), w(128 | ((c >> 12) & 63)), w(128 | ((c >> 6) & 63)), w(128 | (c & 63));\n else\n w(224 | (c >> 12)), w(128 | ((c >> 6) & 63)), w(128 | (c & 63));\n }\n return slc(ar, 0, ai);\n}\n/**\n * Converts a Uint8Array to a string\n * @param dat The data to decode to string\n * @param latin1 Whether or not to interpret the data as Latin-1. This should\n * not need to be true unless encoding to binary string.\n * @returns The original UTF-8/Latin-1 string\n */\nexport function strFromU8(dat, latin1) {\n if (latin1) {\n var r = '';\n for (var i = 0; i < dat.length; i += 16384)\n r += String.fromCharCode.apply(null, dat.subarray(i, i + 16384));\n return r;\n }\n else if (td)\n return td.decode(dat);\n else {\n var _a = dutf8(dat), out = _a[0], ext = _a[1];\n if (ext.length)\n err(8);\n return out;\n }\n}\n;\n// deflate bit flag\nvar dbf = function (l) { return l == 1 ? 3 : l < 6 ? 2 : l == 9 ? 1 : 0; };\n// skip local zip header\nvar slzh = function (d, b) { return b + 30 + b2(d, b + 26) + b2(d, b + 28); };\n// read zip header\nvar zh = function (d, b, z) {\n var fnl = b2(d, b + 28), fn = strFromU8(d.subarray(b + 46, b + 46 + fnl), !(b2(d, b + 8) & 2048)), es = b + 46 + fnl, bs = b4(d, b + 20);\n var _a = z && bs == 4294967295 ? z64e(d, es) : [bs, b4(d, b + 24), b4(d, b + 42)], sc = _a[0], su = _a[1], off = _a[2];\n return [b2(d, b + 10), sc, su, fn, es + b2(d, b + 30) + b2(d, b + 32), off];\n};\n// read zip64 extra field\nvar z64e = function (d, b) {\n for (; b2(d, b) != 1; b += 4 + b2(d, b + 2))\n ;\n return [b8(d, b + 12), b8(d, b + 4), b8(d, b + 20)];\n};\n// extra field length\nvar exfl = function (ex) {\n var le = 0;\n if (ex) {\n for (var k in ex) {\n var l = ex[k].length;\n if (l > 65535)\n err(9);\n le += l + 4;\n }\n }\n return le;\n};\n// write zip header\nvar wzh = function (d, b, f, fn, u, c, ce, co) {\n var fl = fn.length, ex = f.extra, col = co && co.length;\n var exl = exfl(ex);\n wbytes(d, b, ce != null ? 0x2014B50 : 0x4034B50), b += 4;\n if (ce != null)\n d[b++] = 20, d[b++] = f.os;\n d[b] = 20, b += 2; // spec compliance? what's that?\n d[b++] = (f.flag << 1) | (c == null && 8), d[b++] = u && 8;\n d[b++] = f.compression & 255, d[b++] = f.compression >> 8;\n var dt = new Date(f.mtime == null ? Date.now() : f.mtime), y = dt.getFullYear() - 1980;\n if (y < 0 || y > 119)\n err(10);\n wbytes(d, b, (y << 25) | ((dt.getMonth() + 1) << 21) | (dt.getDate() << 16) | (dt.getHours() << 11) | (dt.getMinutes() << 5) | (dt.getSeconds() >>> 1)), b += 4;\n if (c != null) {\n wbytes(d, b, f.crc);\n wbytes(d, b + 4, c);\n wbytes(d, b + 8, f.size);\n }\n wbytes(d, b + 12, fl);\n wbytes(d, b + 14, exl), b += 16;\n if (ce != null) {\n wbytes(d, b, col);\n wbytes(d, b + 6, f.attrs);\n wbytes(d, b + 10, ce), b += 14;\n }\n d.set(fn, b);\n b += fl;\n if (exl) {\n for (var k in ex) {\n var exf = ex[k], l = exf.length;\n wbytes(d, b, +k);\n wbytes(d, b + 2, l);\n d.set(exf, b + 4), b += 4 + l;\n }\n }\n if (col)\n d.set(co, b), b += col;\n return b;\n};\n// write zip footer (end of central directory)\nvar wzf = function (o, b, c, d, e) {\n wbytes(o, b, 0x6054B50); // skip disk\n wbytes(o, b + 8, c);\n wbytes(o, b + 10, c);\n wbytes(o, b + 12, d);\n wbytes(o, b + 16, e);\n};\n/**\n * A pass-through stream to keep data uncompressed in a ZIP archive.\n */\nvar ZipPassThrough = /*#__PURE__*/ (function () {\n /**\n * Creates a pass-through stream that can be added to ZIP archives\n * @param filename The filename to associate with this data stream\n */\n function ZipPassThrough(filename) {\n this.filename = filename;\n this.c = crc();\n this.size = 0;\n this.compression = 0;\n }\n /**\n * Processes a chunk and pushes to the output stream. You can override this\n * method in a subclass for custom behavior, but by default this passes\n * the data through. You must call this.ondata(err, chunk, final) at some\n * point in this method.\n * @param chunk The chunk to process\n * @param final Whether this is the last chunk\n */\n ZipPassThrough.prototype.process = function (chunk, final) {\n this.ondata(null, chunk, final);\n };\n /**\n * Pushes a chunk to be added. If you are subclassing this with a custom\n * compression algorithm, note that you must push data from the source\n * file only, pre-compression.\n * @param chunk The chunk to push\n * @param final Whether this is the last chunk\n */\n ZipPassThrough.prototype.push = function (chunk, final) {\n if (!this.ondata)\n err(5);\n this.c.p(chunk);\n this.size += chunk.length;\n if (final)\n this.crc = this.c.d();\n this.process(chunk, final || false);\n };\n return ZipPassThrough;\n}());\nexport { ZipPassThrough };\n// I don't extend because TypeScript extension adds 1kB of runtime bloat\n/**\n * Streaming DEFLATE compression for ZIP archives. Prefer using AsyncZipDeflate\n * for better performance\n */\nvar ZipDeflate = /*#__PURE__*/ (function () {\n /**\n * Creates a DEFLATE stream that can be added to ZIP archives\n * @param filename The filename to associate with this data stream\n * @param opts The compression options\n */\n function ZipDeflate(filename, opts) {\n var _this_1 = this;\n if (!opts)\n opts = {};\n ZipPassThrough.call(this, filename);\n this.d = new Deflate(opts, function (dat, final) {\n _this_1.ondata(null, dat, final);\n });\n this.compression = 8;\n this.flag = dbf(opts.level);\n }\n ZipDeflate.prototype.process = function (chunk, final) {\n try {\n this.d.push(chunk, final);\n }\n catch (e) {\n this.ondata(e, null, final);\n }\n };\n /**\n * Pushes a chunk to be deflated\n * @param chunk The chunk to push\n * @param final Whether this is the last chunk\n */\n ZipDeflate.prototype.push = function (chunk, final) {\n ZipPassThrough.prototype.push.call(this, chunk, final);\n };\n return ZipDeflate;\n}());\nexport { ZipDeflate };\n/**\n * Asynchronous streaming DEFLATE compression for ZIP archives\n */\nvar AsyncZipDeflate = /*#__PURE__*/ (function () {\n /**\n * Creates a DEFLATE stream that can be added to ZIP archives\n * @param filename The filename to associate with this data stream\n * @param opts The compression options\n */\n function AsyncZipDeflate(filename, opts) {\n var _this_1 = this;\n if (!opts)\n opts = {};\n ZipPassThrough.call(this, filename);\n this.d = new AsyncDeflate(opts, function (err, dat, final) {\n _this_1.ondata(err, dat, final);\n });\n this.compression = 8;\n this.flag = dbf(opts.level);\n this.terminate = this.d.terminate;\n }\n AsyncZipDeflate.prototype.process = function (chunk, final) {\n this.d.push(chunk, final);\n };\n /**\n * Pushes a chunk to be deflated\n * @param chunk The chunk to push\n * @param final Whether this is the last chunk\n */\n AsyncZipDeflate.prototype.push = function (chunk, final) {\n ZipPassThrough.prototype.push.call(this, chunk, final);\n };\n return AsyncZipDeflate;\n}());\nexport { AsyncZipDeflate };\n// TODO: Better tree shaking\n/**\n * A zippable archive to which files can incrementally be added\n */\nvar Zip = /*#__PURE__*/ (function () {\n /**\n * Creates an empty ZIP archive to which files can be added\n * @param cb The callback to call whenever data for the generated ZIP archive\n * is available\n */\n function Zip(cb) {\n this.ondata = cb;\n this.u = [];\n this.d = 1;\n }\n /**\n * Adds a file to the ZIP archive\n * @param file The file stream to add\n */\n Zip.prototype.add = function (file) {\n var _this_1 = this;\n if (!this.ondata)\n err(5);\n // finishing or finished\n if (this.d & 2)\n this.ondata(err(4 + (this.d & 1) * 8, 0, 1), null, false);\n else {\n var f = strToU8(file.filename), fl_1 = f.length;\n var com = file.comment, o = com && strToU8(com);\n var u = fl_1 != file.filename.length || (o && (com.length != o.length));\n var hl_1 = fl_1 + exfl(file.extra) + 30;\n if (fl_1 > 65535)\n this.ondata(err(11, 0, 1), null, false);\n var header = new u8(hl_1);\n wzh(header, 0, file, f, u);\n var chks_1 = [header];\n var pAll_1 = function () {\n for (var _i = 0, chks_2 = chks_1; _i < chks_2.length; _i++) {\n var chk = chks_2[_i];\n _this_1.ondata(null, chk, false);\n }\n chks_1 = [];\n };\n var tr_1 = this.d;\n this.d = 0;\n var ind_1 = this.u.length;\n var uf_1 = mrg(file, {\n f: f,\n u: u,\n o: o,\n t: function () {\n if (file.terminate)\n file.terminate();\n },\n r: function () {\n pAll_1();\n if (tr_1) {\n var nxt = _this_1.u[ind_1 + 1];\n if (nxt)\n nxt.r();\n else\n _this_1.d = 1;\n }\n tr_1 = 1;\n }\n });\n var cl_1 = 0;\n file.ondata = function (err, dat, final) {\n if (err) {\n _this_1.ondata(err, dat, final);\n _this_1.terminate();\n }\n else {\n cl_1 += dat.length;\n chks_1.push(dat);\n if (final) {\n var dd = new u8(16);\n wbytes(dd, 0, 0x8074B50);\n wbytes(dd, 4, file.crc);\n wbytes(dd, 8, cl_1);\n wbytes(dd, 12, file.size);\n chks_1.push(dd);\n uf_1.c = cl_1, uf_1.b = hl_1 + cl_1 + 16, uf_1.crc = file.crc, uf_1.size = file.size;\n if (tr_1)\n uf_1.r();\n tr_1 = 1;\n }\n else if (tr_1)\n pAll_1();\n }\n };\n this.u.push(uf_1);\n }\n };\n /**\n * Ends the process of adding files and prepares to emit the final chunks.\n * This *must* be called after adding all desired files for the resulting\n * ZIP file to work properly.\n */\n Zip.prototype.end = function () {\n var _this_1 = this;\n if (this.d & 2) {\n this.ondata(err(4 + (this.d & 1) * 8, 0, 1), null, true);\n return;\n }\n if (this.d)\n this.e();\n else\n this.u.push({\n r: function () {\n if (!(_this_1.d & 1))\n return;\n _this_1.u.splice(-1, 1);\n _this_1.e();\n },\n t: function () { }\n });\n this.d = 3;\n };\n Zip.prototype.e = function () {\n var bt = 0, l = 0, tl = 0;\n for (var _i = 0, _a = this.u; _i < _a.length; _i++) {\n var f = _a[_i];\n tl += 46 + f.f.length + exfl(f.extra) + (f.o ? f.o.length : 0);\n }\n var out = new u8(tl + 22);\n for (var _b = 0, _c = this.u; _b < _c.length; _b++) {\n var f = _c[_b];\n wzh(out, bt, f, f.f, f.u, f.c, l, f.o);\n bt += 46 + f.f.length + exfl(f.extra) + (f.o ? f.o.length : 0), l += f.b;\n }\n wzf(out, bt, this.u.length, tl, l);\n this.ondata(null, out, true);\n this.d = 2;\n };\n /**\n * A method to terminate any internal workers used by the stream. Subsequent\n * calls to add() will fail.\n */\n Zip.prototype.terminate = function () {\n for (var _i = 0, _a = this.u; _i < _a.length; _i++) {\n var f = _a[_i];\n f.t();\n }\n this.d = 2;\n };\n return Zip;\n}());\nexport { Zip };\nexport function zip(data, opts, cb) {\n if (!cb)\n cb = opts, opts = {};\n if (typeof cb != 'function')\n err(7);\n var r = {};\n fltn(data, '', r, opts);\n var k = Object.keys(r);\n var lft = k.length, o = 0, tot = 0;\n var slft = lft, files = new Array(lft);\n var term = [];\n var tAll = function () {\n for (var i = 0; i < term.length; ++i)\n term[i]();\n };\n var cbd = function (a, b) {\n mt(function () { cb(a, b); });\n };\n mt(function () { cbd = cb; });\n var cbf = function () {\n var out = new u8(tot + 22), oe = o, cdl = tot - o;\n tot = 0;\n for (var i = 0; i < slft; ++i) {\n var f = files[i];\n try {\n var l = f.c.length;\n wzh(out, tot, f, f.f, f.u, l);\n var badd = 30 + f.f.length + exfl(f.extra);\n var loc = tot + badd;\n out.set(f.c, loc);\n wzh(out, o, f, f.f, f.u, l, tot, f.m), o += 16 + badd + (f.m ? f.m.length : 0), tot = loc + l;\n }\n catch (e) {\n return cbd(e, null);\n }\n }\n wzf(out, o, files.length, cdl, oe);\n cbd(null, out);\n };\n if (!lft)\n cbf();\n var _loop_1 = function (i) {\n var fn = k[i];\n var _a = r[fn], file = _a[0], p = _a[1];\n var c = crc(), size = file.length;\n c.p(file);\n var f = strToU8(fn), s = f.length;\n var com = p.comment, m = com && strToU8(com), ms = m && m.length;\n var exl = exfl(p.extra);\n var compression = p.level == 0 ? 0 : 8;\n var cbl = function (e, d) {\n if (e) {\n tAll();\n cbd(e, null);\n }\n else {\n var l = d.length;\n files[i] = mrg(p, {\n size: size,\n crc: c.d(),\n c: d,\n f: f,\n m: m,\n u: s != fn.length || (m && (com.length != ms)),\n compression: compression\n });\n o += 30 + s + exl + l;\n tot += 76 + 2 * (s + exl) + (ms || 0) + l;\n if (!--lft)\n cbf();\n }\n };\n if (s > 65535)\n cbl(err(11, 0, 1), null);\n if (!compression)\n cbl(null, file);\n else if (size < 160000) {\n try {\n cbl(null, deflateSync(file, p));\n }\n catch (e) {\n cbl(e, null);\n }\n }\n else\n term.push(deflate(file, p, cbl));\n };\n // Cannot use lft because it can decrease\n for (var i = 0; i < slft; ++i) {\n _loop_1(i);\n }\n return tAll;\n}\n/**\n * Synchronously creates a ZIP file. Prefer using `zip` for better performance\n * with more than one file.\n * @param data The directory structure for the ZIP archive\n * @param opts The main options, merged with per-file options\n * @returns The generated ZIP archive\n */\nexport function zipSync(data, opts) {\n if (!opts)\n opts = {};\n var r = {};\n var files = [];\n fltn(data, '', r, opts);\n var o = 0;\n var tot = 0;\n for (var fn in r) {\n var _a = r[fn], file = _a[0], p = _a[1];\n var compression = p.level == 0 ? 0 : 8;\n var f = strToU8(fn), s = f.length;\n var com = p.comment, m = com && strToU8(com), ms = m && m.length;\n var exl = exfl(p.extra);\n if (s > 65535)\n err(11);\n var d = compression ? deflateSync(file, p) : file, l = d.length;\n var c = crc();\n c.p(file);\n files.push(mrg(p, {\n size: file.length,\n crc: c.d(),\n c: d,\n f: f,\n m: m,\n u: s != fn.length || (m && (com.length != ms)),\n o: o,\n compression: compression\n }));\n o += 30 + s + exl + l;\n tot += 76 + 2 * (s + exl) + (ms || 0) + l;\n }\n var out = new u8(tot + 22), oe = o, cdl = tot - o;\n for (var i = 0; i < files.length; ++i) {\n var f = files[i];\n wzh(out, f.o, f, f.f, f.u, f.c.length);\n var badd = 30 + f.f.length + exfl(f.extra);\n out.set(f.c, f.o + badd);\n wzh(out, o, f, f.f, f.u, f.c.length, f.o, f.m), o += 16 + badd + (f.m ? f.m.length : 0);\n }\n wzf(out, o, files.length, cdl, oe);\n return out;\n}\n/**\n * Streaming pass-through decompression for ZIP archives\n */\nvar UnzipPassThrough = /*#__PURE__*/ (function () {\n function UnzipPassThrough() {\n }\n UnzipPassThrough.prototype.push = function (data, final) {\n this.ondata(null, data, final);\n };\n UnzipPassThrough.compression = 0;\n return UnzipPassThrough;\n}());\nexport { UnzipPassThrough };\n/**\n * Streaming DEFLATE decompression for ZIP archives. Prefer AsyncZipInflate for\n * better performance.\n */\nvar UnzipInflate = /*#__PURE__*/ (function () {\n /**\n * Creates a DEFLATE decompression that can be used in ZIP archives\n */\n function UnzipInflate() {\n var _this_1 = this;\n this.i = new Inflate(function (dat, final) {\n _this_1.ondata(null, dat, final);\n });\n }\n UnzipInflate.prototype.push = function (data, final) {\n try {\n this.i.push(data, final);\n }\n catch (e) {\n this.ondata(e, null, final);\n }\n };\n UnzipInflate.compression = 8;\n return UnzipInflate;\n}());\nexport { UnzipInflate };\n/**\n * Asynchronous streaming DEFLATE decompression for ZIP archives\n */\nvar AsyncUnzipInflate = /*#__PURE__*/ (function () {\n /**\n * Creates a DEFLATE decompression that can be used in ZIP archives\n */\n function AsyncUnzipInflate(_, sz) {\n var _this_1 = this;\n if (sz < 320000) {\n this.i = new Inflate(function (dat, final) {\n _this_1.ondata(null, dat, final);\n });\n }\n else {\n this.i = new AsyncInflate(function (err, dat, final) {\n _this_1.ondata(err, dat, final);\n });\n this.terminate = this.i.terminate;\n }\n }\n AsyncUnzipInflate.prototype.push = function (data, final) {\n if (this.i.terminate)\n data = slc(data, 0);\n this.i.push(data, final);\n };\n AsyncUnzipInflate.compression = 8;\n return AsyncUnzipInflate;\n}());\nexport { AsyncUnzipInflate };\n/**\n * A ZIP archive decompression stream that emits files as they are discovered\n */\nvar Unzip = /*#__PURE__*/ (function () {\n /**\n * Creates a ZIP decompression stream\n * @param cb The callback to call whenever a file in the ZIP archive is found\n */\n function Unzip(cb) {\n this.onfile = cb;\n this.k = [];\n this.o = {\n 0: UnzipPassThrough\n };\n this.p = et;\n }\n /**\n * Pushes a chunk to be unzipped\n * @param chunk The chunk to push\n * @param final Whether this is the last chunk\n */\n Unzip.prototype.push = function (chunk, final) {\n var _this_1 = this;\n if (!this.onfile)\n err(5);\n if (!this.p)\n err(4);\n if (this.c > 0) {\n var len = Math.min(this.c, chunk.length);\n var toAdd = chunk.subarray(0, len);\n this.c -= len;\n if (this.d)\n this.d.push(toAdd, !this.c);\n else\n this.k[0].push(toAdd);\n chunk = chunk.subarray(len);\n if (chunk.length)\n return this.push(chunk, final);\n }\n else {\n var f = 0, i = 0, is = void 0, buf = void 0;\n if (!this.p.length)\n buf = chunk;\n else if (!chunk.length)\n buf = this.p;\n else {\n buf = new u8(this.p.length + chunk.length);\n buf.set(this.p), buf.set(chunk, this.p.length);\n }\n var l = buf.length, oc = this.c, add = oc && this.d;\n var _loop_2 = function () {\n var _a;\n var sig = b4(buf, i);\n if (sig == 0x4034B50) {\n f = 1, is = i;\n this_1.d = null;\n this_1.c = 0;\n var bf = b2(buf, i + 6), cmp_1 = b2(buf, i + 8), u = bf & 2048, dd = bf & 8, fnl = b2(buf, i + 26), es = b2(buf, i + 28);\n if (l > i + 30 + fnl + es) {\n var chks_3 = [];\n this_1.k.unshift(chks_3);\n f = 2;\n var sc_1 = b4(buf, i + 18), su_1 = b4(buf, i + 22);\n var fn_1 = strFromU8(buf.subarray(i + 30, i += 30 + fnl), !u);\n if (sc_1 == 4294967295) {\n _a = dd ? [-2] : z64e(buf, i), sc_1 = _a[0], su_1 = _a[1];\n }\n else if (dd)\n sc_1 = -1;\n i += es;\n this_1.c = sc_1;\n var d_1;\n var file_1 = {\n name: fn_1,\n compression: cmp_1,\n start: function () {\n if (!file_1.ondata)\n err(5);\n if (!sc_1)\n file_1.ondata(null, et, true);\n else {\n var ctr = _this_1.o[cmp_1];\n if (!ctr)\n file_1.ondata(err(14, 'unknown compression type ' + cmp_1, 1), null, false);\n d_1 = sc_1 < 0 ? new ctr(fn_1) : new ctr(fn_1, sc_1, su_1);\n d_1.ondata = function (err, dat, final) { file_1.ondata(err, dat, final); };\n for (var _i = 0, chks_4 = chks_3; _i < chks_4.length; _i++) {\n var dat = chks_4[_i];\n d_1.push(dat, false);\n }\n if (_this_1.k[0] == chks_3 && _this_1.c)\n _this_1.d = d_1;\n else\n d_1.push(et, true);\n }\n },\n terminate: function () {\n if (d_1 && d_1.terminate)\n d_1.terminate();\n }\n };\n if (sc_1 >= 0)\n file_1.size = sc_1, file_1.originalSize = su_1;\n this_1.onfile(file_1);\n }\n return \"break\";\n }\n else if (oc) {\n if (sig == 0x8074B50) {\n is = i += 12 + (oc == -2 && 8), f = 3, this_1.c = 0;\n return \"break\";\n }\n else if (sig == 0x2014B50) {\n is = i -= 4, f = 3, this_1.c = 0;\n return \"break\";\n }\n }\n };\n var this_1 = this;\n for (; i < l - 4; ++i) {\n var state_1 = _loop_2();\n if (state_1 === \"break\")\n break;\n }\n this.p = et;\n if (oc < 0) {\n var dat = f ? buf.subarray(0, is - 12 - (oc == -2 && 8) - (b4(buf, is - 16) == 0x8074B50 && 4)) : buf.subarray(0, i);\n if (add)\n add.push(dat, !!f);\n else\n this.k[+(f == 2)].push(dat);\n }\n if (f & 2)\n return this.push(buf.subarray(i), final);\n this.p = buf.subarray(i);\n }\n if (final) {\n if (this.c)\n err(13);\n this.p = null;\n }\n };\n /**\n * Registers a decoder with the stream, allowing for files compressed with\n * the compression type provided to be expanded correctly\n * @param decoder The decoder constructor\n */\n Unzip.prototype.register = function (decoder) {\n this.o[decoder.compression] = decoder;\n };\n return Unzip;\n}());\nexport { Unzip };\nvar mt = typeof queueMicrotask == 'function' ? queueMicrotask : typeof setTimeout == 'function' ? setTimeout : function (fn) { fn(); };\nexport function unzip(data, opts, cb) {\n if (!cb)\n cb = opts, opts = {};\n if (typeof cb != 'function')\n err(7);\n var term = [];\n var tAll = function () {\n for (var i = 0; i < term.length; ++i)\n term[i]();\n };\n var files = {};\n var cbd = function (a, b) {\n mt(function () { cb(a, b); });\n };\n mt(function () { cbd = cb; });\n var e = data.length - 22;\n for (; b4(data, e) != 0x6054B50; --e) {\n if (!e || data.length - e > 65558) {\n cbd(err(13, 0, 1), null);\n return tAll;\n }\n }\n ;\n var lft = b2(data, e + 8);\n if (lft) {\n var c = lft;\n var o = b4(data, e + 16);\n var z = o == 4294967295;\n if (z) {\n e = b4(data, e - 12);\n if (b4(data, e) != 0x6064B50) {\n cbd(err(13, 0, 1), null);\n return tAll;\n }\n c = lft = b4(data, e + 32);\n o = b4(data, e + 48);\n }\n var fltr = opts && opts.filter;\n var _loop_3 = function (i) {\n var _a = zh(data, o, z), c_1 = _a[0], sc = _a[1], su = _a[2], fn = _a[3], no = _a[4], off = _a[5], b = slzh(data, off);\n o = no;\n var cbl = function (e, d) {\n if (e) {\n tAll();\n cbd(e, null);\n }\n else {\n if (d)\n files[fn] = d;\n if (!--lft)\n cbd(null, files);\n }\n };\n if (!fltr || fltr({\n name: fn,\n size: sc,\n originalSize: su,\n compression: c_1\n })) {\n if (!c_1)\n cbl(null, slc(data, b, b + sc));\n else if (c_1 == 8) {\n var infl = data.subarray(b, b + sc);\n if (sc < 320000) {\n try {\n cbl(null, inflateSync(infl, new u8(su)));\n }\n catch (e) {\n cbl(e, null);\n }\n }\n else\n term.push(inflate(infl, { size: su }, cbl));\n }\n else\n cbl(err(14, 'unknown compression type ' + c_1, 1), null);\n }\n else\n cbl(null, null);\n };\n for (var i = 0; i < c; ++i) {\n _loop_3(i);\n }\n }\n else\n cbd(null, {});\n return tAll;\n}\n/**\n * Synchronously decompresses a ZIP archive. Prefer using `unzip` for better\n * performance with more than one file.\n * @param data The raw compressed ZIP file\n * @param opts The ZIP extraction options\n * @returns The decompressed files\n */\nexport function unzipSync(data, opts) {\n var files = {};\n var e = data.length - 22;\n for (; b4(data, e) != 0x6054B50; --e) {\n if (!e || data.length - e > 65558)\n err(13);\n }\n ;\n var c = b2(data, e + 8);\n if (!c)\n return {};\n var o = b4(data, e + 16);\n var z = o == 4294967295;\n if (z) {\n e = b4(data, e - 12);\n if (b4(data, e) != 0x6064B50)\n err(13);\n c = b4(data, e + 32);\n o = b4(data, e + 48);\n }\n var fltr = opts && opts.filter;\n for (var i = 0; i < c; ++i) {\n var _a = zh(data, o, z), c_2 = _a[0], sc = _a[1], su = _a[2], fn = _a[3], no = _a[4], off = _a[5], b = slzh(data, off);\n o = no;\n if (!fltr || fltr({\n name: fn,\n size: sc,\n originalSize: su,\n compression: c_2\n })) {\n if (!c_2)\n files[fn] = slc(data, b, b + sc);\n else if (c_2 == 8)\n files[fn] = inflateSync(data.subarray(b, b + sc), new u8(su));\n else\n err(14, 'unknown compression type ' + c_2);\n }\n }\n return files;\n}\n","import { PBRMaterial } from '@babylonjs/core/Materials/PBR/pbrMaterial';\r\nimport { Matrix } from '@babylonjs/core/Maths/math.vector';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { VertexBuffer } from '@babylonjs/core/Meshes/buffer';\r\nimport { Geometry } from '@babylonjs/core/Meshes/geometry';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { DeepImmutable, FloatArray, IndicesArray, Nullable } from '@babylonjs/core/types';\r\nimport { AsyncZippable, strToU8, zip } from 'fflate';\r\nimport { SceneNode } from '../../../models/classes/scene-node';\r\n/** Be careful when adding dependencies in this file. Importing and using a constructor function or a barrel file will balloon the size of the worker js bundle */\r\n\r\nexport type TypeIntArray = Exclude;\r\nexport type TypedArray = TypeIntArray | Float32Array | Float64Array;\r\n\r\nexport interface GeometryTransferrable {\r\n payload: GeometryConversionPayload;\r\n transfer: ArrayBuffer[];\r\n}\r\n\r\nexport interface GeometryConversionPayload {\r\n options: USDOpts;\r\n id: string;\r\n indices: TypeIntArray;\r\n vertexBuffers: {\r\n [index: string]: Float32Array;\r\n };\r\n}\r\n\r\nexport interface USDOpts {\r\n unitsPerMeter: number;\r\n xrAllowScaling: boolean;\r\n}\r\n\r\nexport interface ConversionResult {\r\n file: Uint8Array;\r\n}\r\n\r\nexport class USDZConverter {\r\n parse(scene: Scene, sceneNode: SceneNode, useWorker = false): Promise {\r\n return Promise.reject();\r\n }\r\n\r\n convertGeometry(payload: GeometryConversionPayload) {\r\n const meshObject = this.buildGeometryFromBuffers(payload);\r\n return buildUSDAFile(payload.options, meshObject);\r\n }\r\n\r\n zipUSDZFiles(files: AsyncZippable) {\r\n return new Promise((resolve, reject) => {\r\n zip(\r\n files,\r\n {\r\n level: 0,\r\n },\r\n (err, res) => {\r\n if (err) {\r\n reject(err);\r\n } else {\r\n const blob = new Blob([res.buffer], { type: 'model/vnd.usdz+zip' });\r\n resolve(blob);\r\n }\r\n }\r\n );\r\n });\r\n }\r\n\r\n protected buildGeometryFromBuffers(payload: GeometryConversionPayload) {\r\n const positions = payload.vertexBuffers[VertexBuffer.PositionKind];\r\n if (!positions) {\r\n throw 'No indices in geometry';\r\n }\r\n const vertexCount = Math.floor(positions.length / 3);\r\n\r\n return buildGeometryFromData(\r\n payload.indices,\r\n vertexCount,\r\n positions,\r\n payload.vertexBuffers[VertexBuffer.NormalKind],\r\n payload.vertexBuffers[VertexBuffer.UVKind]\r\n );\r\n }\r\n\r\n protected buildGeometryFromObject(geometry: Geometry) {\r\n const indices = geometry.getIndices();\r\n const vertexCount = geometry.getTotalVertices();\r\n const positions = geometry.getVerticesData(VertexBuffer.PositionKind);\r\n if (!indices) {\r\n throw 'No indices in geometry';\r\n }\r\n if (!positions) {\r\n throw 'No indices in geometry';\r\n }\r\n return buildGeometryFromData(\r\n indices,\r\n vertexCount,\r\n positions,\r\n geometry.getVerticesData(VertexBuffer.NormalKind),\r\n geometry.getVerticesData(VertexBuffer.UVKind)\r\n );\r\n }\r\n}\r\n\r\nexport const PRECISION = 7;\r\n\r\nexport function buildHeader(opts: USDOpts) {\r\n return `#usda 1.0\r\n(\r\n customLayerData = {\r\n string creator = \"KB3D USDZConverter\"\r\n }\r\n metersPerUnit = ${1 / opts.unitsPerMeter}\r\n upAxis = \"Y\"\r\n)\r\n`;\r\n}\r\n\r\nexport function buildUSDAFile(opts: USDOpts, dataToInsert: string) {\r\n let output = buildHeader(opts);\r\n output += dataToInsert;\r\n return strToU8(output);\r\n}\r\n\r\n// Xform\r\nexport function buildXform(object: AbstractMesh, geometry: Geometry, material: PBRMaterial, appendFileName = '') {\r\n const name = 'Object__' + escapeGUID(object.id) + appendFileName;\r\n const worldMatrix = object.computeWorldMatrix(true);\r\n const transform = buildMatrix(worldMatrix);\r\n\r\n if (worldMatrix.determinant() < 0) {\r\n console.warn('USDZConverter: USDZ does not support negative scales', object);\r\n }\r\n\r\n return `def Xform \"${name}\" (\r\n prepend references = @./geometries/Geometry__${escapeGUID(geometry.id)}${appendFileName}.usd@
\r\n)\r\n{\r\n matrix4d xformOp:transform = ${transform}\r\n uniform token[] xformOpOrder = [\"xformOp:transform\"]\r\n rel material:binding = \r\n}\r\n`;\r\n}\r\n\r\nfunction buildMatrix(matrix: Matrix) {\r\n const array = matrix.toArray();\r\n return `( ${buildMatrixRow(array, 0)}, ${buildMatrixRow(array, 4)}, ${buildMatrixRow(array, 8)}, ${buildMatrixRow(\r\n array,\r\n 12\r\n )} )`;\r\n}\r\n\r\nfunction buildMatrixRow(array: DeepImmutable, offset: number) {\r\n return `(${array[offset + 0]}, ${array[offset + 1]}, ${array[offset + 2]}, ${array[offset + 3]})`;\r\n} // Mesh\r\n\r\nfunction buildGeometryFromData(\r\n indices: IndicesArray,\r\n verticesCount: number,\r\n positions: number[] | Float32Array,\r\n normals: Nullable,\r\n uvs: Nullable\r\n) {\r\n const vertexCounts = Array(Math.floor(indices.length / 3))\r\n .fill(3)\r\n .join(', ');\r\n\r\n const name = 'Geometry';\r\n\r\n return `\r\ndef \"Geometry\"\r\n{\r\n def Mesh \"${name}\"\r\n {\r\n int[] faceVertexCounts = [${vertexCounts}]\r\n int[] faceVertexIndices = [${stringifyIntArray(indices)}]\r\n normal3f[] normals = [${buildVector3Array(normals, verticesCount)}] (\r\n interpolation = \"vertex\"\r\n )\r\n point3f[] points = [${buildVector3Array(positions, verticesCount)}]\r\n float2[] primvars:st = [${buildVector2Array(uvs, verticesCount)}] (\r\n interpolation = \"vertex\"\r\n )\r\n uniform token subdivisionScheme = \"none\"\r\n uniform token orientation = \"leftHanded\"\r\n }\r\n}\r\n`;\r\n}\r\n\r\nfunction buildVector3Array(data: Nullable, count: number) {\r\n let res = '';\r\n if (!data) {\r\n console.warn('USDZConverter: Data missing.');\r\n for (let i = 0; i < count; i++) {\r\n res += '(0, 0, 0), ';\r\n }\r\n return res;\r\n }\r\n\r\n return stringifyFloatArray(data, count, 3);\r\n}\r\n\r\nfunction buildVector2Array(data: Nullable, count: number) {\r\n let res = '';\r\n if (!data) {\r\n console.warn('USDZConverter: Data missing.');\r\n for (let i = 0; i < count; i++) {\r\n res += '(0, 0), ';\r\n }\r\n return res;\r\n }\r\n\r\n return stringifyFloatArray(data, count, 2);\r\n}\r\n\r\nfunction stringifyIntArray(data: IndicesArray) {\r\n const count = data.length;\r\n let res = '';\r\n for (let i = 0; i < count; i++) {\r\n res += `${data[i]}, `;\r\n }\r\n\r\n return res.slice(0, -2);\r\n}\r\n\r\nfunction stringifyFloatArray(data: FloatArray, count: number, perStruc: number) {\r\n let res = '';\r\n if (perStruc === 2) {\r\n for (let i = 0; i < count; i++) {\r\n res += `(${data[i * 2].toPrecision(PRECISION)}, ${data[i * 2 + 1].toPrecision(PRECISION)}), `;\r\n }\r\n } else if (perStruc === 3) {\r\n for (let i = 0; i < count; i++) {\r\n res += `(${data[i * 3].toPrecision(PRECISION)}, ${data[i * 3 + 1].toPrecision(PRECISION)}, ${data[\r\n i * 3 + 2\r\n ].toPrecision(PRECISION)}), `;\r\n }\r\n } else {\r\n throw Error('Not supported');\r\n }\r\n return res.slice(0, -2);\r\n}\r\n\r\nexport function escapeGUID(guid: string) {\r\n return guid.replace(/[-\\s]+/g, '_');\r\n}\r\n","import { Material } from '@babylonjs/core/Materials/material';\r\nimport { PBRMaterial } from '@babylonjs/core/Materials/PBR/pbrMaterial';\r\nimport { BaseTexture, DynamicTexture, Texture } from '@babylonjs/core/Materials/Textures';\r\nimport { Color3, Matrix, Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { Geometry } from '@babylonjs/core/Meshes/geometry';\r\nimport { LinesMesh } from '@babylonjs/core/Meshes/linesMesh';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { VertexData } from '@babylonjs/core/Meshes/mesh.vertexData';\r\nimport { Node } from '@babylonjs/core/node';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { Nullable } from '@babylonjs/core/types';\r\nimport { getBinClampedSize } from '@common-util/texture-util';\r\nimport { AsyncZippable, AsyncZippableFile, strToU8 } from 'fflate';\r\nimport { map, take } from 'rxjs/operators';\r\nimport { SceneNode } from '../../../models/classes/scene-node';\r\nimport { eWorkerTasks, Kb3dWorker } from '../../kb-worker';\r\nimport {\r\n buildHeader,\r\n buildUSDAFile,\r\n buildXform,\r\n ConversionResult,\r\n escapeGUID,\r\n GeometryConversionPayload,\r\n GeometryTransferrable,\r\n PRECISION,\r\n TypedArray,\r\n USDOpts,\r\n USDZConverter\r\n} from './usdz-converter';\r\n\r\nconst canvasResize = document.createElement('canvas');\r\n\r\nexport class USDZBabylonConverter extends USDZConverter {\r\n parse(scene: Scene, sceneNode: SceneNode, useWorker = false) {\r\n const toCleanup: Array = [];\r\n const files: Record = {};\r\n const modelFileName = 'model.usda'; // model file should be first in USDZ archive so we init it here\r\n\r\n files[modelFileName] = new Uint8Array(0);\r\n let output = buildHeader(sceneNode);\r\n const materials: Record = {};\r\n const textures: Record = {};\r\n\r\n const conversionWorker = useWorker ? new Kb3dWorker() : null;\r\n const asyncGenerators: Array> = [];\r\n\r\n scene.meshes.forEach(mesh => {\r\n if (\r\n mesh instanceof Mesh &&\r\n mesh.isEnabled() &&\r\n mesh.visibility > 0 &&\r\n mesh.geometry &&\r\n !(mesh instanceof LinesMesh) &&\r\n // for filtering out skyboxes\r\n !mesh.infiniteDistance\r\n ) {\r\n let subMeshes: Array>;\r\n let geometries: Array>;\r\n let meshMaterials: Array>;\r\n const baseGeometry = mesh.geometry;\r\n\r\n if (mesh.subMeshes && mesh.subMeshes.length > 1) {\r\n const splitGeometryByMaterialId = new Map<\r\n Material,\r\n { indices: number[]; positions: number[]; normals: number[]; uvs: number[] }\r\n >();\r\n\r\n const baseVertexData = VertexData.ExtractFromGeometry(baseGeometry);\r\n mesh.subMeshes.forEach((subMesh, index) => {\r\n const subMeshMaterial: Nullable = subMesh.getMaterial();\r\n if (subMeshMaterial) {\r\n let vertexData = splitGeometryByMaterialId.get(subMeshMaterial);\r\n if (!vertexData) {\r\n vertexData = { indices: [], positions: [], normals: [], uvs: [] };\r\n splitGeometryByMaterialId.set(subMeshMaterial, vertexData);\r\n }\r\n // Add polygons from this submesh to the mesh for this material\r\n let newIdx = vertexData.indices.length;\r\n const endIndex = subMesh.indexStart + subMesh.indexCount;\r\n for (let i = subMesh.indexStart; i < endIndex; i++) {\r\n if (baseVertexData.indices && baseVertexData.indices[i] !== undefined) {\r\n const oldIdx = baseVertexData.indices[i];\r\n\r\n vertexData.indices.push(newIdx);\r\n const i2 = oldIdx * 2;\r\n const i3 = oldIdx * 3;\r\n if (\r\n baseVertexData.positions &&\r\n typeof baseVertexData.positions[i3] !== 'undefined'\r\n ) {\r\n vertexData.positions.push(\r\n baseVertexData.positions[i3],\r\n baseVertexData.positions[i3 + 1],\r\n baseVertexData.positions[i3 + 2]\r\n );\r\n }\r\n if (baseVertexData.normals && typeof baseVertexData.normals[i3] !== 'undefined') {\r\n vertexData.normals.push(\r\n baseVertexData.normals[i3],\r\n baseVertexData.normals[i3 + 1],\r\n baseVertexData.normals[i3 + 2]\r\n );\r\n }\r\n if (baseVertexData.uvs && typeof baseVertexData.uvs[i2] !== 'undefined') {\r\n vertexData.uvs.push(baseVertexData.uvs[i2], baseVertexData.uvs[i2 + 1]);\r\n }\r\n\r\n newIdx++;\r\n }\r\n }\r\n }\r\n\r\n return null;\r\n });\r\n\r\n subMeshes = [];\r\n geometries = [];\r\n meshMaterials = [];\r\n splitGeometryByMaterialId.forEach((vertexData, material) => {\r\n if (material instanceof PBRMaterial) {\r\n const newMesh = mesh.clone();\r\n newMesh.id = mesh.id + '__' + material.id;\r\n newMesh.material = material;\r\n const newVertexData = new VertexData();\r\n newVertexData.indices = vertexData.indices;\r\n newVertexData.positions = vertexData.positions;\r\n newVertexData.normals = vertexData.normals;\r\n newVertexData.uvs = vertexData.uvs;\r\n const newGeometry = Geometry.CreateGeometryForMesh(newMesh);\r\n newGeometry.id = `sub-${newMesh.id}`;\r\n newVertexData.applyToGeometry(newGeometry);\r\n subMeshes.push(newMesh);\r\n geometries.push(newGeometry);\r\n meshMaterials.push(material);\r\n toCleanup.push(newGeometry, newMesh);\r\n } else {\r\n console.warn(`USDZExporter: No PBR material on submesh`);\r\n }\r\n });\r\n } else {\r\n subMeshes = [mesh];\r\n geometries = [baseGeometry];\r\n if (mesh.material instanceof PBRMaterial) {\r\n meshMaterials = [mesh.material];\r\n } else {\r\n meshMaterials = [];\r\n console.warn('USDZExporter: No PBR material on mesh');\r\n }\r\n }\r\n\r\n for (let i = 0; i < geometries.length; i++) {\r\n const subMesh = subMeshes[i];\r\n const geometry = geometries[i];\r\n const meshMaterial = meshMaterials[i];\r\n if (!subMesh || !geometry || !meshMaterial) {\r\n continue;\r\n }\r\n\r\n // const appendFileName = `__${i}`;\r\n const geometryFileName = 'geometries/Geometry__' + escapeGUID(geometry.id) + '.usd';\r\n\r\n if (!(geometryFileName in files)) {\r\n if (conversionWorker) {\r\n const geometryData = buildGeometryTransferrable(\r\n { unitsPerMeter: sceneNode.unitsPerMeter, xrAllowScaling: sceneNode.xrAllowScaling },\r\n geometry\r\n );\r\n if (geometryData) {\r\n asyncGenerators.push(\r\n conversionWorker\r\n .startTask(\r\n eWorkerTasks.usdConvertGeometry,\r\n geometryData.payload,\r\n geometryData.transfer\r\n )\r\n .pipe(\r\n take(1),\r\n map(res => {\r\n files[geometryFileName] = res.payload.file;\r\n })\r\n )\r\n .toPromise()\r\n );\r\n }\r\n } else {\r\n const meshObject = this.buildGeometryFromObject(geometry);\r\n files[geometryFileName] = buildUSDAFile(sceneNode, meshObject);\r\n }\r\n }\r\n\r\n if (!(meshMaterial.id in meshMaterial)) {\r\n materials[meshMaterial.id] = meshMaterial;\r\n }\r\n\r\n output += buildXform(subMesh, geometry, meshMaterial);\r\n }\r\n }\r\n });\r\n\r\n output += buildMaterials(materials, textures);\r\n files[modelFileName] = strToU8(output);\r\n const textureGenerators: Array<() => Promise> = [];\r\n for (const id in textures) {\r\n const texture = textures[id];\r\n const color = id.match(/[0-9A-F]{6}$/);\r\n const color3 = color && color.length === 1 ? Color3.FromHexString(`#${color[0]}`) : undefined;\r\n const isRGBA = texture.hasAlpha;\r\n textureGenerators.push(() =>\r\n processImageData(texture, isRGBA ? 'image/png' : 'image/jpeg', color3)\r\n .then(blob => {\r\n if (blob) {\r\n return blob.arrayBuffer();\r\n }\r\n })\r\n .then(buffer => {\r\n if (buffer) {\r\n files[`textures/Texture__${id}.${isRGBA ? 'png' : 'jpg'}`] = new Uint8Array(buffer);\r\n }\r\n })\r\n );\r\n } // 64 byte alignment\r\n // https://github.com/101arrowz/fflate/issues/39#issuecomment-777263109\r\n\r\n return Promise.all([processSequentially(asyncGenerators), processSequentially(textureGenerators)])\r\n .then(() => {\r\n let offset = 0;\r\n\r\n for (const filename in files) {\r\n const file = files[filename] as Uint8Array;\r\n const headerSize = 34 + filename.length;\r\n offset += headerSize;\r\n const offsetMod64 = offset & 63;\r\n\r\n if (offsetMod64 !== 4) {\r\n const padLength = 64 - offsetMod64;\r\n const padding = new Uint8Array(padLength);\r\n files[filename] = [\r\n file,\r\n {\r\n extra: {\r\n 12345: padding,\r\n },\r\n },\r\n ];\r\n }\r\n\r\n offset = file.length;\r\n }\r\n\r\n if (conversionWorker) {\r\n const transfer: ArrayBuffer[] = [];\r\n for (const fileName in files) {\r\n const file = files[fileName];\r\n if (file instanceof Uint8Array) {\r\n transfer.push(file.buffer);\r\n } else {\r\n transfer.push(file[0].buffer);\r\n }\r\n }\r\n\r\n return conversionWorker\r\n .startTask(eWorkerTasks.usdPackage, files, transfer)\r\n .pipe(\r\n take(1),\r\n map(res => {\r\n canvasResize.width = 0;\r\n canvasResize.height = 0;\r\n conversionWorker.dispose();\r\n return res.payload;\r\n })\r\n )\r\n .toPromise();\r\n } else {\r\n return this.zipUSDZFiles(files);\r\n }\r\n })\r\n .then(res => {\r\n toCleanup.forEach(item => item.dispose());\r\n return res;\r\n });\r\n }\r\n}\r\n\r\nasync function processSequentially(cbs: Array | (() => Promise)>) {\r\n const result: T[] = [];\r\n for (const cb of cbs) {\r\n if (cb instanceof Promise) {\r\n result.push(await cb);\r\n } else {\r\n result.push(await cb());\r\n }\r\n }\r\n\r\n return result;\r\n}\r\n\r\nexport function processImageData(texture: Texture, mimeType: string, color?: Color3): Promise {\r\n if (texture instanceof DynamicTexture) {\r\n const context = texture.getContext();\r\n\r\n const targetTexureSize = getBinClampedSize(context.canvas.width, context.canvas.height);\r\n canvasResize.width = targetTexureSize;\r\n canvasResize.height = targetTexureSize;\r\n\r\n const resizeContext = canvasResize.getContext('2d')!;\r\n const canvas = context.canvas;\r\n\r\n if (!(canvas instanceof HTMLCanvasElement)) {\r\n throw Error('texture does not have a canvas');\r\n }\r\n\r\n resizeContext.drawImage(canvas, 0, 0, targetTexureSize, targetTexureSize);\r\n\r\n if (color) {\r\n const imagedata = resizeContext.getImageData(0, 0, targetTexureSize, targetTexureSize);\r\n const data = imagedata.data;\r\n for (let i = 0; i < data.length; i += 4) {\r\n data[i + 0] = data[i + 0] * color.r;\r\n data[i + 1] = data[i + 1] * color.g;\r\n data[i + 2] = data[i + 2] * color.b;\r\n }\r\n resizeContext.putImageData(imagedata, 0, 0);\r\n }\r\n\r\n return canvasToBlob(canvasResize, mimeType);\r\n } else {\r\n const size = texture.getSize();\r\n\r\n const dataPromise = texture.readPixels() as Promise;\r\n if (!dataPromise) {\r\n return Promise.resolve(undefined);\r\n }\r\n\r\n return dataPromise.then(data => {\r\n if (data instanceof Float32Array || data instanceof Float64Array) {\r\n for (let i = 0; i < data.length; i += 4) {\r\n data[i + 0] = data[i + 0] * 255;\r\n data[i + 1] = data[i + 1] * 255;\r\n data[i + 2] = data[i + 2] * 255;\r\n data[i + 3] = data[i + 3] * 255;\r\n }\r\n }\r\n\r\n let targetTexureSize = getBinClampedSize(size.width, size.height);\r\n if (targetTexureSize > 1024) {\r\n targetTexureSize = 1024;\r\n }\r\n\r\n canvasResize.width = targetTexureSize;\r\n canvasResize.height = targetTexureSize;\r\n\r\n const pxLength = targetTexureSize * targetTexureSize;\r\n const resizedData = new Uint8ClampedArray(pxLength * 4);\r\n if (targetTexureSize !== size.width || targetTexureSize !== size.height) {\r\n const widthRatio = size.width / targetTexureSize;\r\n const heightRatio = size.height / targetTexureSize;\r\n // in the interest of speed, just downscale with nearest neighbor algorithm instead of averaging pixels.\r\n for (let i = 0; i < pxLength; i++) {\r\n const x = i % targetTexureSize;\r\n // Y must be flipped for USDZ output\r\n const y = Math.floor(targetTexureSize - i / targetTexureSize);\r\n const targetIndex = (Math.floor(x * widthRatio) + Math.floor(y * heightRatio) * size.width) * 4;\r\n const destIndex = i * 4;\r\n resizedData[destIndex + 0] = data[targetIndex + 0];\r\n resizedData[destIndex + 1] = data[targetIndex + 1];\r\n resizedData[destIndex + 2] = data[targetIndex + 2];\r\n resizedData[destIndex + 3] = data[targetIndex + 3];\r\n }\r\n } else {\r\n const lastLineOffset = (targetTexureSize - 1) * targetTexureSize;\r\n for (let y = 0; y < targetTexureSize / 2; y++) {\r\n const yOffset = y * targetTexureSize;\r\n const yOffsetTarget = lastLineOffset - yOffset;\r\n for (let x = 0; x < targetTexureSize; x++) {\r\n const offset = (yOffset + x) * 4;\r\n const offsetTarget = (yOffsetTarget + x) * 4;\r\n resizedData[offset] = data[offsetTarget];\r\n resizedData[offset + 1] = data[offsetTarget + 1];\r\n resizedData[offset + 2] = data[offsetTarget + 2];\r\n resizedData[offset + 3] = data[offsetTarget + 3];\r\n resizedData[offsetTarget] = data[offset];\r\n resizedData[offsetTarget + 1] = data[offset + 1];\r\n resizedData[offsetTarget + 2] = data[offset + 2];\r\n resizedData[offsetTarget + 3] = data[offset + 3];\r\n }\r\n }\r\n }\r\n\r\n if (color) {\r\n for (let i = 0; i < resizedData.length; i += 4) {\r\n resizedData[i + 0] = resizedData[i + 0] * color.r;\r\n resizedData[i + 1] = resizedData[i + 1] * color.g;\r\n resizedData[i + 2] = resizedData[i + 2] * color.b;\r\n }\r\n }\r\n\r\n const imageData = new ImageData(resizedData, targetTexureSize, targetTexureSize);\r\n\r\n const resizeContext = canvasResize.getContext('2d')!;\r\n resizeContext.putImageData(imageData, 0, 0);\r\n\r\n return canvasToBlob(canvasResize, mimeType);\r\n });\r\n }\r\n}\r\n\r\nfunction canvasToBlob(canvasResize: HTMLCanvasElement, mimeType: string) {\r\n return new Promise((resolve, reject) => {\r\n canvasResize.toBlob(\r\n result => {\r\n if (result) {\r\n resolve(result);\r\n }\r\n reject('Could not generate texture');\r\n },\r\n mimeType,\r\n 1\r\n );\r\n });\r\n}\r\n\r\nfunction buildMaterials(materials: Record, textures: Record) {\r\n const array = [];\r\n\r\n for (const uuid in materials) {\r\n const material = materials[uuid];\r\n array.push(buildMaterial(material, textures));\r\n }\r\n\r\n return `def \"Materials\"\r\n{\r\n${array.join('')}\r\n}\r\n`;\r\n}\r\n\r\nfunction buildMaterial(material: PBRMaterial, textures: Record) {\r\n // https://graphics.pixar.com/usd/docs/UsdPreviewSurface-Proposal.html\r\n const pad = ' ';\r\n const inputs: string[] = [];\r\n const samplers: string[] = [];\r\n const materialId = escapeGUID(material.id);\r\n\r\n const newV = new Vector3();\r\n const tMatrix = new Matrix();\r\n const t2Matrix = new Matrix();\r\n const t3Matrix = new Matrix();\r\n\r\n function buildTexture(texture: BaseTexture, mapType: string, color?: Color3) {\r\n if (texture instanceof Texture) {\r\n const textureName = escapeGUID(texture.name);\r\n const parsedTextureId = textureName + (color ? '__' + color.toHexString().substring(1) : '');\r\n textures[parsedTextureId] = texture;\r\n const angleDeg = (texture.wAng * 180) / Math.PI;\r\n\r\n // USD doesn't seem to have a good way to apply a matrix to a texture. But we can use the rotation and scale straight,\r\n // and grab the translation off of the babylon texture matrix. The matrix includes the rotation pivot related translation\r\n // which is nasty to calculate without a matrix and USD doesn't support UV Pivot\r\n const textureMatrix = texture.getTextureMatrix().m;\r\n Matrix.RotationYawPitchRollToRef(texture.vAng, texture.uAng, texture.wAng, tMatrix);\r\n Matrix.ScalingToRef(texture.uScale, texture.vScale, 1, t2Matrix);\r\n tMatrix.multiplyToRef(t2Matrix, t3Matrix);\r\n t3Matrix.invert();\r\n\r\n Vector3.TransformCoordinatesFromFloatsToRef(textureMatrix[8], textureMatrix[9], 0, t3Matrix, newV);\r\n\r\n return `\r\n def Shader \"Transform2d_${mapType}\" (\r\n sdrMetadata = {\r\n string role = \"math\"\r\n }\r\n )\r\n {\r\n uniform token info:id = \"UsdTransform2d\"\r\n float2 inputs:in.connect = \r\n float2 inputs:scale = (${texture.uScale}, ${texture.vScale})\r\n float2 inputs:translation = (${newV.x}, ${newV.y})\r\n float inputs:rotation = ${angleDeg.toPrecision(PRECISION)}\r\n float2 outputs:result\r\n }\r\n def Shader \"Texture__${textureName}_${mapType}\"\r\n {\r\n uniform token info:id = \"UsdUVTexture\"\r\n asset inputs:file = @./textures/Texture__${parsedTextureId}.${texture.hasAlpha ? 'png' : 'jpg'}@\r\n float2 inputs:st.connect = \r\n token inputs:wrapS = \"repeat\"\r\n token inputs:wrapT = \"repeat\"\r\n float outputs:r\r\n float outputs:g\r\n float outputs:b\r\n ${texture.hasAlpha ? 'float outputs:a' : ''}\r\n float3 outputs:rgb\r\n }`;\r\n }\r\n\r\n return '';\r\n }\r\n\r\n if (material.albedoTexture) {\r\n inputs.push(\r\n `${pad}color3f inputs:diffuseColor.connect = `\r\n );\r\n samplers.push(buildTexture(material.albedoTexture, 'diffuse', material.albedoColor));\r\n } else {\r\n inputs.push(`${pad}color3f inputs:diffuseColor = ${buildColor(material.albedoColor)}`);\r\n }\r\n\r\n if (material.emissiveTexture) {\r\n inputs.push(\r\n `${pad}color3f inputs:emissiveColor.connect = `\r\n );\r\n samplers.push(buildTexture(material.emissiveTexture, 'emissive'));\r\n } else if (material.emissiveColor.toLuminance() > 0) {\r\n inputs.push(`${pad}color3f inputs:emissiveColor = ${buildColor(material.emissiveColor)}`);\r\n }\r\n\r\n if (material.bumpTexture) {\r\n inputs.push(\r\n `${pad}normal3f inputs:normal.connect = `\r\n );\r\n samplers.push(buildTexture(material.bumpTexture, 'normal'));\r\n }\r\n\r\n if (material.ambientTexture !== null) {\r\n inputs.push(\r\n `${pad}float inputs:occlusion.connect = `\r\n );\r\n samplers.push(buildTexture(material.ambientTexture, 'occlusion'));\r\n }\r\n\r\n if (material.microSurfaceTexture !== null && material.roughness === 1) {\r\n inputs.push(\r\n `${pad}float inputs:roughness.connect = `\r\n );\r\n samplers.push(buildTexture(material.microSurfaceTexture, 'roughness'));\r\n } else {\r\n inputs.push(`${pad}float inputs:roughness = ${material.roughness}`);\r\n }\r\n\r\n if (material.metallicTexture !== null && material.metallic === 1) {\r\n inputs.push(\r\n `${pad}float inputs:metallic.connect = `\r\n );\r\n samplers.push(buildTexture(material.metallicTexture, 'metallic'));\r\n } else {\r\n inputs.push(`${pad}float inputs:metallic = ${material.metallic}`);\r\n }\r\n\r\n if (material.opacityTexture !== null) {\r\n inputs.push(\r\n `${pad}float inputs:opacity.connect = `\r\n );\r\n inputs.push(`${pad}float inputs:opacityThreshold = 0.0001`);\r\n samplers.push(buildTexture(material.opacityTexture, 'opacity'));\r\n } else if (material.albedoTexture && material.albedoTexture.hasAlpha) {\r\n inputs.push(\r\n `${pad}float inputs:opacity.connect = `\r\n );\r\n inputs.push(`${pad}float inputs:opacityThreshold = 0.0001`);\r\n } else {\r\n inputs.push(`${pad}float inputs:opacity = ${material.alpha}`);\r\n }\r\n\r\n if (material.clearCoat && material.clearCoat.isEnabled) {\r\n inputs.push(`${pad}float inputs:clearcoat = ${material.clearCoat.intensity}`);\r\n inputs.push(`${pad}float inputs:clearcoatRoughness = ${material.clearCoat.roughness}`);\r\n // TODO More properties?\r\n }\r\n inputs.push(`${pad}float inputs:ior = ${material.indexOfRefraction}`);\r\n // TODO More properties?\r\n\r\n return `\r\n def Material \"Material__${materialId}\"\r\n {\r\n def Shader \"PreviewSurface\"\r\n {\r\n uniform token info:id = \"UsdPreviewSurface\"\r\n${inputs.join('\\n')}\r\n int inputs:useSpecularWorkflow = 0\r\n token outputs:surface\r\n }\r\n token outputs:surface.connect = \r\n token inputs:frame:stPrimvarName = \"st\"\r\n def Shader \"uvReader_st\"\r\n {\r\n uniform token info:id = \"UsdPrimvarReader_float2\"\r\n token inputs:varname.connect = \r\n float2 inputs:fallback = (0.0, 0.0)\r\n float2 outputs:result\r\n }\r\n${samplers.join('\\n')}\r\n }\r\n`;\r\n}\r\n\r\nfunction buildGeometryTransferrable(options: USDOpts, geometry: Geometry): GeometryTransferrable | undefined {\r\n const transfer: ArrayBuffer[] = [];\r\n const geometryBuffers = geometry.getVertexBuffers();\r\n const indices = geometry.getIndices();\r\n if (geometryBuffers && indices) {\r\n const vertexBuffers = Object.keys(geometryBuffers).reduce((res, type) => {\r\n const data = geometryBuffers[type].getData()!;\r\n\r\n if ('buffer' in data) {\r\n res[type] = data as Float32Array;\r\n transfer.push(res[type].slice().buffer);\r\n } else if (data instanceof ArrayBuffer || 'length' in data) {\r\n res[type] = new Float32Array(data);\r\n transfer.push(res[type].buffer);\r\n } else {\r\n console.warn('Unsupported geometry');\r\n }\r\n\r\n return res;\r\n }, {} as { [index: string]: Float32Array });\r\n\r\n const indexBuffer = Array.isArray(indices) ? new Uint32Array(indices) : indices;\r\n\r\n transfer.push(indexBuffer.slice().buffer);\r\n\r\n const result: GeometryTransferrable = {\r\n payload: {\r\n options,\r\n id: geometry.id,\r\n indices: indexBuffer,\r\n vertexBuffers,\r\n },\r\n transfer,\r\n };\r\n\r\n return result;\r\n }\r\n}\r\n\r\nfunction buildColor(color: Color3) {\r\n return `(${color.r}, ${color.g}, ${color.b})`;\r\n}\r\n","import { Matrix, Quaternion, Vector3 } from '@babylonjs/core/Maths/math.vector';\r\nimport { WebXRSessionManager } from '@babylonjs/core/XR/webXRSessionManager';\r\n\r\nexport interface IXrHitResult {\r\n position: Vector3;\r\n rotation: Quaternion;\r\n matrix: Matrix;\r\n xrResult: XRHitTestResult;\r\n}\r\n\r\nexport class XrHitTester {\r\n constructor(public sessionMgr: WebXRSessionManager) {}\r\n\r\n /** holds the hit test source which is filled after initialize() has been called */\r\n public source: XRHitTestSource | undefined;\r\n /** temp variables for saving memory while processing hit results */\r\n private _tmpPos = new Vector3();\r\n private _tmpMat = new Matrix();\r\n private _tmpQuat = new Quaternion();\r\n\r\n public initialize(offsetRay?: XRRay) {\r\n const session = this.sessionMgr.session;\r\n\r\n //setup initial hit testing source\r\n if (session) {\r\n return session.requestHitTestSource!({\r\n space: this.sessionMgr.viewerReferenceSpace, //we use the viewer ref space for the source\r\n offsetRay: offsetRay,\r\n }).then(hitSource => {\r\n return (this.source = hitSource);\r\n });\r\n }\r\n }\r\n\r\n public getHitResult(frame: XRFrame): IXrHitResult | undefined {\r\n if (!this.source) return;\r\n\r\n const hits = frame.getHitTestResults(this.source);\r\n if (!hits.length) return;\r\n\r\n const hit = hits[0]; //only concentrate on the first hit... might want to reconsider this later\r\n return XrHitTester.FromXRHitTestResult(hit, this.sessionMgr.referenceSpace);\r\n\r\n // // Check that the y-coordinate of the normal is large enough that the normal\r\n // // is pointing up.\r\n // return hitMatrix hitMatrix.elements[5] > 0.75 ?\r\n // hitPosition.setFromMatrixPosition(hitMatrix) :\r\n // null;\r\n }\r\n\r\n public static FromXRHitTestResult(hit: XRHitTestResult, refSpace: XRSpace) {\r\n const pose = hit.getPose(refSpace);\r\n if (pose == null) return;\r\n\r\n const xrpos = pose.transform.position;\r\n const xrquat = pose.transform.orientation;\r\n const mat = new Matrix();\r\n Matrix.FromFloat32ArrayToRefScaled(pose.transform.matrix, 0, 1, mat);\r\n\r\n return {\r\n position: new Vector3(xrpos.x, xrpos.y, xrpos.z),\r\n rotation: new Quaternion(xrquat.x, xrquat.y, xrquat.z, xrquat.w),\r\n matrix: mat,\r\n xrResult: hit,\r\n } as IXrHitResult;\r\n }\r\n\r\n public dispose() {\r\n if (this.source) {\r\n this.source.cancel();\r\n this.source = undefined;\r\n }\r\n }\r\n}\r\n","import { DirectionalLight, ShadowLight } from '@babylonjs/core/Lights';\r\nimport { ShadowGenerator } from '@babylonjs/core/Lights/Shadows/shadowGenerator';\r\nimport { Matrix, Vector3 } from '@babylonjs/core/Maths/math.vector';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { eShadowFilter, eShadowResolution } from '@models/classes/enums';\r\nimport { getByDynamicId } from '@models/classes/kb3d-manager';\r\nimport { SpaceNode } from '@models/classes/meshes/space-node';\r\nimport { BoundingBoxCache, BoundingBoxInfo } from '@view/modules/bounding-box-cache';\r\nimport { Subscription } from 'rxjs';\r\n\r\n/**\r\n * Manages shadow generators and their association with lights and meshes that participate\r\n */\r\nexport class ShadowManager {\r\n constructor(private _bbCache: BoundingBoxCache) {\r\n this._bbCache.rootBoundingBoxChanged.subscribe(bbInfo => {\r\n //invalidate our projection matrix cache\r\n this.projectionMatrixCache.clear();\r\n });\r\n }\r\n\r\n private generatorMap = new Map();\r\n /** the meshes that are casting shadows. The mesh renderer adds and removes them from this list as their boolean flag is changed */\r\n private casters = new Set();\r\n /** the meshes that are receiving shadows. The mesh renderer adds and removes them from this list as their boolean flag is changed */\r\n private receivers = new Set();\r\n /**\r\n * Stores the calculated shadow projection matrix for directional lights. This cache is invalidated when\r\n * the root scene bounding box has changed or meshes change their casting / receiving flags.\r\n *\r\n * By default, babylon calculates the projection matrix of the shadows on every frame.\r\n * To be more efficient, we cache the projection matrix until something that warrants updating it happens.\r\n */\r\n private projectionMatrixCache = new Map();\r\n\r\n private bbSubscription: Subscription;\r\n /**\r\n * Add a shadow generator for the given light. Should only be called when the light.generatesShadows is set to true\r\n * @param light\r\n */\r\n public add(light: ShadowLight, shadowResolution: eShadowResolution) {\r\n let sg = this.get(light);\r\n if (!sg) {\r\n sg = new ShadowGenerator(ShadowManager.convertShadowResultion(shadowResolution), light);\r\n this.generatorMap.set(light, sg);\r\n this.addAllCastersToGenerator(sg);\r\n if (light instanceof DirectionalLight) {\r\n /**\r\n * Directional lights are tricky, as their light shadow fulstrum needs to be the optimal size for the shadows to look good.\r\n * The built-in calculation of the projection matrix does not account for meshes that are receiving shadows. So we override\r\n * the calculation here. It also gives us a way to cache and increase performance.\r\n */\r\n light.autoUpdateExtends = false;\r\n light.autoCalcShadowZBounds = false;\r\n light.customProjectionMatrixBuilder = (\r\n viewMatrix: Matrix,\r\n renderList: AbstractMesh[],\r\n matrix: Matrix\r\n ) => {\r\n return this.customProjectionMatrixBuilder(light, viewMatrix, renderList, matrix);\r\n };\r\n }\r\n }\r\n return sg;\r\n }\r\n\r\n /**\r\n * Get the shadow generator for a specific light (if it has one)\r\n * @param light\r\n */\r\n public get(light: ShadowLight) {\r\n return this.generatorMap.get(light);\r\n }\r\n\r\n /** Remove the shadow generator for the given light */\r\n public remove(light: ShadowLight) {\r\n const sg = this.get(light);\r\n (light as any).customProjectionMatrixBuilder = undefined;\r\n if (sg) {\r\n this.generatorMap.delete(light);\r\n sg.dispose();\r\n }\r\n }\r\n\r\n /**\r\n * Adds a mesh as a shadow caster. Should only be called by the mesh renderer when the meshNode.castShadows flag is changed to true\r\n */\r\n public addCaster(mesh: AbstractMesh) {\r\n if (!this.casters.has(mesh)) {\r\n this.casters.add(mesh);\r\n for (const sg of this.generatorMap.values()) {\r\n sg.addShadowCaster(mesh);\r\n }\r\n this.projectionMatrixCache.clear();\r\n }\r\n }\r\n\r\n /**\r\n * Removes a mesh as a shadow caster. Should only be called by the mesh renderer when the meshNode.castShadows flag is changed to false\r\n */\r\n public removeCaster(mesh: AbstractMesh) {\r\n if (this.casters.has(mesh)) {\r\n this.casters.delete(mesh);\r\n for (const sg of this.generatorMap.values()) {\r\n sg.removeShadowCaster(mesh);\r\n }\r\n this.projectionMatrixCache.clear();\r\n }\r\n }\r\n\r\n /**\r\n * Adds/removes the mesh as a shadow caster\r\n * @param mesh\r\n * @param value\r\n */\r\n public toggleCaster(mesh: Mesh, value: boolean) {\r\n if (value) {\r\n this.addCaster(mesh);\r\n } else {\r\n this.removeCaster(mesh);\r\n }\r\n }\r\n\r\n protected addAllCastersToGenerator(sg: ShadowGenerator) {\r\n for (const c of this.casters.values()) {\r\n sg.addShadowCaster(c);\r\n }\r\n }\r\n\r\n public addReceiver(mesh: AbstractMesh) {\r\n if (!this.receivers.has(mesh)) {\r\n this.receivers.add(mesh);\r\n this.projectionMatrixCache.clear();\r\n }\r\n }\r\n\r\n public removeReceiver(mesh: AbstractMesh) {\r\n if (this.receivers.has(mesh)) {\r\n this.receivers.delete(mesh);\r\n this.projectionMatrixCache.clear();\r\n }\r\n }\r\n\r\n public toggleReceiver(mesh: AbstractMesh, value: boolean) {\r\n if (value) {\r\n this.addReceiver(mesh);\r\n } else {\r\n this.removeReceiver(mesh);\r\n }\r\n }\r\n\r\n public dispose() {\r\n this.projectionMatrixCache.clear();\r\n for (const light of this.generatorMap.keys()) {\r\n this.remove(light);\r\n }\r\n }\r\n\r\n public clearProjectMatrixCache() {\r\n this.projectionMatrixCache.clear();\r\n }\r\n /**\r\n * Directional lights are tricky, as their light shadow fulstrum needs to be the optimal size for the shadows to look good.\r\n * The built-in calculation of the projection matrix does not account for meshes that are receiving shadows. So we override\r\n * the calculation here. It also gives us a way to cache and increase performance.\r\n *\r\n * This is based on the built-in DirectionalLight.prototype._setDefaultAutoExtendShadowProjectionMatrix\r\n * but with these additions:\r\n * * caching of the projection matrix\r\n * * inclusion of shadow receivers in the box calculation so shadows are not chopped off with PCSS\r\n * * uses the shadowOrthoScale of the light to increase the minZ/maxZ range to avoid z-fighting on planes at one of the extents of the box\r\n */\r\n protected customProjectionMatrixBuilder(\r\n light: DirectionalLight,\r\n viewMatrix: Matrix,\r\n renderList: AbstractMesh[],\r\n matrix: Matrix\r\n ) {\r\n if (this.projectionMatrixCache.has(light)) {\r\n matrix.copyFrom(this.projectionMatrixCache.get(light)!);\r\n } else {\r\n const allMeshes = new Set([...this.casters, ...this.receivers]);\r\n\r\n const shadowOrthoScale = light.shadowOrthoScale;\r\n let orthoLeft = Number.MAX_VALUE;\r\n let orthoRight = Number.MIN_VALUE;\r\n let orthoTop = Number.MIN_VALUE;\r\n let orthoBottom = Number.MAX_VALUE;\r\n let shadowMinZ = Number.MAX_VALUE;\r\n let shadowMaxZ = Number.MIN_VALUE;\r\n const tempVector3 = Vector3.Zero();\r\n\r\n //loop through each caster/reciever to build up the fulstrum box\r\n for (const m of allMeshes) {\r\n if (m.isEnabled(true)) {\r\n const node = getByDynamicId(m.id);\r\n let bbInfo: BoundingBoxInfo | undefined;\r\n if (node) {\r\n bbInfo = this._bbCache.get(node);\r\n } else {\r\n // //not a kb3d node, but could be a shadow plan in XR\r\n // let binfo = m.getBoundingInfo();\r\n // if (binfo){\r\n // bbInfo = toBoundingBoxInfo(binfo);\r\n // }\r\n }\r\n if (bbInfo) {\r\n const boundingBox = bbInfo.boundingInfo.boundingBox;\r\n for (let index = 0; index < boundingBox.vectorsWorld.length; index++) {\r\n Vector3.TransformCoordinatesToRef(boundingBox.vectorsWorld[index], viewMatrix, tempVector3);\r\n if (tempVector3.x < orthoLeft) {\r\n orthoLeft = tempVector3.x;\r\n }\r\n if (tempVector3.y < orthoBottom) {\r\n orthoBottom = tempVector3.y;\r\n }\r\n if (tempVector3.x > orthoRight) {\r\n orthoRight = tempVector3.x;\r\n }\r\n if (tempVector3.y > orthoTop) {\r\n orthoTop = tempVector3.y;\r\n }\r\n\r\n if (tempVector3.z < shadowMinZ) {\r\n shadowMinZ = tempVector3.z;\r\n }\r\n if (tempVector3.z > shadowMaxZ) {\r\n shadowMaxZ = tempVector3.z;\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n const xOffset = orthoRight - orthoLeft;\r\n const yOffset = orthoTop - orthoBottom;\r\n\r\n /** for exponential blur, it works way better when the max z is alot larger than the scene to get soft shadows */\r\n const sg = this.get(light)!;\r\n if (sg.filter == ShadowGenerator.FILTER_BLUREXPONENTIALSHADOWMAP) {\r\n shadowMaxZ = shadowMaxZ * 50;\r\n }\r\n const zOffset = shadowMaxZ - shadowMinZ;\r\n\r\n Matrix.OrthoOffCenterLHToRef(\r\n orthoLeft - xOffset * shadowOrthoScale,\r\n orthoRight + xOffset * shadowOrthoScale,\r\n orthoBottom - yOffset * shadowOrthoScale,\r\n orthoTop + yOffset * shadowOrthoScale,\r\n /**\r\n * Babylon does not account for the orthoScale when calculating minZ/maxZ, which leads to shadow z fighting on planes receiving shadows.\r\n * So here we change this behavior to offset by the scale just like left/right/top/bottom\r\n */\r\n shadowMinZ - zOffset * shadowOrthoScale,\r\n shadowMaxZ + zOffset * shadowOrthoScale,\r\n matrix\r\n );\r\n\r\n this.projectionMatrixCache.set(light, matrix);\r\n // console.count(\"shadow manager processed\");\r\n }\r\n }\r\n\r\n public static convertShadowResultion(r: eShadowResolution) {\r\n switch (r) {\r\n case eShadowResolution.low:\r\n return 512;\r\n case eShadowResolution.medium:\r\n return 1024;\r\n case eShadowResolution.high:\r\n return 2048;\r\n }\r\n }\r\n\r\n public static convertShadowFilter(f: eShadowFilter) {\r\n switch (f) {\r\n case eShadowFilter.none:\r\n return ShadowGenerator.FILTER_NONE;\r\n case eShadowFilter.simple:\r\n return ShadowGenerator.FILTER_PCF;\r\n case eShadowFilter.advanced:\r\n return ShadowGenerator.FILTER_PCSS;\r\n }\r\n }\r\n}\r\n","import { DirectionalLight, Light, ShadowGenerator } from '@babylonjs/core/Lights';\r\nimport { Vector3 } from '@babylonjs/core/Maths';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { eShadowResolution } from '@models/classes/enums';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { ShadowManager } from '@view/render/shadow-manager';\r\n\r\nexport class XrLight extends DirectionalLight {\r\n constructor(public v: KbViewer, public scene: Scene) {\r\n super('xrLight', new Vector3(0, -3, 0), scene);\r\n\r\n //turn all existing lights off and save their setting\r\n for (const l of scene.lights) {\r\n if (l !== this) {\r\n this._originalLightEnabledMap.set(l, l.isEnabled(false));\r\n l.setEnabled(false);\r\n }\r\n }\r\n\r\n this.position = this.direction.negate();\r\n this.intensity = 0.5;\r\n\r\n this._shadowManager = new ShadowManager(this.v.bbCache);\r\n this._sg = this._shadowManager.add(this, eShadowResolution.low);\r\n this._sg.setDarkness(0.3);\r\n this._sg.filter = ShadowGenerator.FILTER_BLUREXPONENTIALSHADOWMAP;\r\n // this._sg.blurBoxOffset = 10;\r\n // this._sg.blurScale = 2;\r\n\r\n this._sg.useKernelBlur = true;\r\n this._sg.blurKernel = 64; //64;\r\n this._sg.blurScale = 2;\r\n\r\n /**\r\n * Add all meshes from the scene to the shadow caster.\r\n * This includes the shadow plane so that extents are auto calculated to include it\r\n */\r\n const rootNode = this.v.rendererManager.sceneRenderer.get3d(this.v.sceneNode)!;\r\n for (const m of rootNode.getChildMeshes(false)) {\r\n if (m.name == 'xrShadowPlane') {\r\n this._shadowManager.addReceiver(m);\r\n } else {\r\n this._shadowManager.addCaster(m);\r\n }\r\n // console.log(`adding mesh '${m.name}' to shadow generator`);\r\n }\r\n }\r\n\r\n private _shadowManager: ShadowManager;\r\n private _sg: ShadowGenerator;\r\n private _originalLightEnabledMap = new Map();\r\n\r\n public setDirection(x: number, y: number, z: number) {\r\n this.direction.set(x, y, z);\r\n this.position = this.direction.negate();\r\n this._shadowManager.clearProjectMatrixCache();\r\n }\r\n\r\n public dispose() {\r\n for (const [l, e] of this._originalLightEnabledMap) {\r\n l.setEnabled(e);\r\n }\r\n this._originalLightEnabledMap.clear();\r\n if (this._shadowManager) this._shadowManager.dispose();\r\n (this._shadowManager as any) = undefined;\r\n (this._sg as any) = undefined;\r\n\r\n super.dispose();\r\n }\r\n}\r\n","import { Animation } from '@babylonjs/core/Animations/animation';\r\nimport { EasingFunction, SineEase } from '@babylonjs/core/Animations/easing';\r\nimport { StandardMaterial } from '@babylonjs/core/Materials/standardMaterial';\r\nimport { Color3 } from '@babylonjs/core/Maths/math.color';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { VertexData } from '@babylonjs/core/Meshes/mesh.vertexData';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { Subscription } from 'rxjs';\r\nimport { BoundingBoxInfo } from '../bounding-box-cache';\r\n\r\nconst SEGMENTS = 12;\r\nconst RADIUS = 0.05;\r\nconst DELTA_PHI = Math.PI / (2 * SEGMENTS);\r\nconst LINE_WIDTH = 0.02;\r\nconst FRAME_RATE = 1 / 0.3;\r\n\r\nexport class XrMarker extends Mesh {\r\n constructor(public v: KbViewer, public scene: Scene, public naturalScale: number) {\r\n super('xrMarker', scene);\r\n\r\n if (!XrMarker.MarkerMaterial) {\r\n const m = (XrMarker.MarkerMaterial = new StandardMaterial('xrMarkerMaterial', this.scene));\r\n m.emissiveColor = Color3.White();\r\n m.disableLighting = true;\r\n m.backFaceCulling = false;\r\n m.alpha = 0.5;\r\n }\r\n this.material = XrMarker.MarkerMaterial;\r\n\r\n this.resize();\r\n }\r\n\r\n private subscription: Subscription = new Subscription();\r\n static MarkerMaterial: StandardMaterial;\r\n private _lineWidth: number;\r\n private _radius: number;\r\n /**\r\n * Whether the marker and model have been placed yet.\r\n * The marker is put inside the scene root node so it transforms with\r\n * the model once placed. So our calculations change based on that\r\n **/\r\n private _placed: boolean;\r\n\r\n public place(rootNode: TransformNode) {\r\n this.setParent(rootNode);\r\n this.rotation.set(0, 0, 0);\r\n this._placed = true;\r\n\r\n this.resize();\r\n this.hide();\r\n\r\n this.subscription.add(\r\n this.v.bbCache.rootBoundingBoxChanged.subscribe(bb => {\r\n this.resizeToBoundingBox(this.v.bbCache.getWorld(this.v.sceneNode)!);\r\n })\r\n );\r\n }\r\n\r\n public show() {\r\n this.animateVisibility(1, EasingFunction.EASINGMODE_EASEIN);\r\n }\r\n\r\n public hide() {\r\n this.animateVisibility(0, EasingFunction.EASINGMODE_EASEOUT);\r\n }\r\n\r\n private animateVisibility(to: number, easingFunction: number) {\r\n const a = new Animation('hide', 'visibility', FRAME_RATE, Animation.ANIMATIONTYPE_FLOAT);\r\n a.setKeys([\r\n {\r\n frame: 0,\r\n value: this.visibility,\r\n },\r\n {\r\n frame: 1,\r\n value: to,\r\n },\r\n ]);\r\n const easing = new SineEase();\r\n easing.setEasingMode(easingFunction);\r\n a.setEasingFunction(easing);\r\n this.scene.stopAnimation(this);\r\n this.scene.beginDirectAnimation(this, [a], 0, 1, false);\r\n }\r\n\r\n private resize() {\r\n const bb = this.v.bbCache.get(this.v.sceneNode)!;\r\n if (bb) {\r\n this.resizeToBoundingBox(bb);\r\n }\r\n }\r\n\r\n private resizeToBoundingBox(bb: BoundingBoxInfo) {\r\n let deltaZ: number;\r\n let deltaX: number;\r\n\r\n if (!this._placed) {\r\n deltaX = (bb.max.x - bb.min.x) * this.naturalScale;\r\n deltaZ = (bb.max.z - bb.min.z) * this.naturalScale;\r\n this._lineWidth = LINE_WIDTH;\r\n this._radius = RADIUS;\r\n } else {\r\n deltaX = bb.max.x - bb.min.x;\r\n deltaZ = bb.max.z - bb.min.z;\r\n this._lineWidth = LINE_WIDTH / this.naturalScale;\r\n this._radius = RADIUS / this.naturalScale;\r\n }\r\n\r\n const vertexData = this.getVertexData(deltaX, deltaZ);\r\n vertexData.applyToMesh(this, true);\r\n\r\n if (this._placed) {\r\n //it's in the root node now, so center to the root node local bb\r\n this.position.set(bb.center.x, 0, bb.center.z);\r\n }\r\n }\r\n\r\n protected getVertexData(xSize: number, zSize: number) {\r\n const vertices: number[] = [];\r\n const indices: number[] = [];\r\n const normals: number[] = [];\r\n\r\n const x = xSize / 2;\r\n const z = zSize / 2;\r\n this.addCorner(vertices, x, z);\r\n this.addCorner(vertices, -x, z);\r\n this.addCorner(vertices, -x, -z);\r\n this.addCorner(vertices, x, -z);\r\n\r\n const numVertices = vertices.length / 3;\r\n for (let i = 0; i < numVertices - 2; i += 2) {\r\n indices.push(i, i + 1, i + 3, i, i + 3, i + 2);\r\n }\r\n const i = numVertices - 2;\r\n indices.push(i, i + 1, 1, i, 1, 0);\r\n\r\n VertexData.ComputeNormals(vertices, indices, normals);\r\n const vdata = new VertexData();\r\n vdata.indices = indices;\r\n vdata.positions = vertices;\r\n vdata.normals = normals;\r\n\r\n return vdata;\r\n }\r\n\r\n protected addCorner(vertices: number[], cornerX: number, cornerZ: number) {\r\n let phi = cornerX > 0 ? (cornerZ > 0 ? 0 : -Math.PI / 2) : cornerZ > 0 ? Math.PI / 2 : Math.PI;\r\n for (let i = 0; i <= SEGMENTS; ++i) {\r\n vertices.push(\r\n cornerX + (this._radius - this._lineWidth) * Math.cos(phi),\r\n 0,\r\n cornerZ + (this._radius - this._lineWidth) * Math.sin(phi),\r\n cornerX + this._radius * Math.cos(phi),\r\n 0,\r\n cornerZ + this._radius * Math.sin(phi)\r\n );\r\n phi += DELTA_PHI;\r\n }\r\n }\r\n\r\n public dispose() {\r\n this.subscription.unsubscribe();\r\n super.dispose();\r\n }\r\n}\r\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"imageProcessingCompatibility\";\nconst shader = `#ifdef IMAGEPROCESSINGPOSTPROCESS\ngl_FragColor.rgb=pow(gl_FragColor.rgb,vec3(2.2));\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const imageProcessingCompatibility = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"core/Engines/shaderStore\";\nimport \"core/Shaders/ShadersInclude/helperFunctions\";\nimport \"core/Shaders/ShadersInclude/lightFragmentDeclaration\";\nimport \"core/Shaders/ShadersInclude/lightUboDeclaration\";\nimport \"core/Shaders/ShadersInclude/lightsFragmentFunctions\";\nimport \"core/Shaders/ShadersInclude/shadowsFragmentFunctions\";\nimport \"core/Shaders/ShadersInclude/clipPlaneFragmentDeclaration\";\nimport \"core/Shaders/ShadersInclude/fogFragmentDeclaration\";\nimport \"core/Shaders/ShadersInclude/clipPlaneFragment\";\nimport \"core/Shaders/ShadersInclude/lightFragment\";\nimport \"core/Shaders/ShadersInclude/fogFragment\";\nimport \"core/Shaders/ShadersInclude/imageProcessingCompatibility\";\n\nconst name = \"shadowOnlyPixelShader\";\nconst shader = `precision highp float;uniform vec4 vEyePosition;uniform float alpha;uniform vec3 shadowColor;varying vec3 vPositionW;\n#ifdef NORMAL\nvarying vec3 vNormalW;\n#endif\n#include\n#include<__decl__lightFragment>[0..maxSimultaneousLights]\n#include\n#include\n#include\n#include\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_FRAGMENT_MAIN_BEGIN\n#include\nvec3 viewDirectionW=normalize(vEyePosition.xyz-vPositionW);\n#ifdef NORMAL\nvec3 normalW=normalize(vNormalW);\n#else\nvec3 normalW=vec3(1.0,1.0,1.0);\n#endif\nvec3 diffuseBase=vec3(0.,0.,0.);lightingInfo info;float shadow=1.;float glossiness=0.;\n#include[0..1]\nvec4 color=vec4(shadowColor,(1.0-clamp(shadow,0.,1.))*alpha);\n#include\ngl_FragColor=color;\n#include\n#define CUSTOM_FRAGMENT_MAIN_END\n}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const shadowOnlyPixelShader = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"core/Engines/shaderStore\";\nimport \"core/Shaders/ShadersInclude/bonesDeclaration\";\nimport \"core/Shaders/ShadersInclude/bakedVertexAnimationDeclaration\";\nimport \"core/Shaders/ShadersInclude/instancesDeclaration\";\nimport \"core/Shaders/ShadersInclude/clipPlaneVertexDeclaration\";\nimport \"core/Shaders/ShadersInclude/fogVertexDeclaration\";\nimport \"core/Shaders/ShadersInclude/lightFragmentDeclaration\";\nimport \"core/Shaders/ShadersInclude/lightUboDeclaration\";\nimport \"core/Shaders/ShadersInclude/instancesVertex\";\nimport \"core/Shaders/ShadersInclude/bonesVertex\";\nimport \"core/Shaders/ShadersInclude/bakedVertexAnimation\";\nimport \"core/Shaders/ShadersInclude/clipPlaneVertex\";\nimport \"core/Shaders/ShadersInclude/fogVertex\";\nimport \"core/Shaders/ShadersInclude/shadowsVertex\";\n\nconst name = \"shadowOnlyVertexShader\";\nconst shader = `precision highp float;attribute vec3 position;\n#ifdef NORMAL\nattribute vec3 normal;\n#endif\n#include\n#include\n#include\nuniform mat4 view;uniform mat4 viewProjection;\n#ifdef POINTSIZE\nuniform float pointSize;\n#endif\nvarying vec3 vPositionW;\n#ifdef NORMAL\nvarying vec3 vNormalW;\n#endif\n#ifdef VERTEXCOLOR\nvarying vec4 vColor;\n#endif\n#include\n#include\n#include<__decl__lightFragment>[0..maxSimultaneousLights]\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\n#include\n#include\n#include\nvec4 worldPos=finalWorld*vec4(position,1.0);gl_Position=viewProjection*worldPos;vPositionW=vec3(worldPos);\n#ifdef NORMAL\nvNormalW=normalize(vec3(finalWorld*vec4(normal,0.0)));\n#endif\n#include\n#include\n#include[0..maxSimultaneousLights]\n#if defined(POINTSIZE) && !defined(WEBGPU)\ngl_PointSize=pointSize;\n#endif\n#define CUSTOM_VERTEX_MAIN_END\n}\n`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const shadowOnlyVertexShader = { name, shader };\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport type { Nullable } from \"core/types\";\r\nimport { SerializationHelper } from \"core/Misc/decorators\";\r\nimport type { Matrix } from \"core/Maths/math.vector\";\r\nimport { Color3 } from \"core/Maths/math.color\";\r\nimport type { BaseTexture } from \"core/Materials/Textures/baseTexture\";\r\nimport type { IShadowLight } from \"core/Lights/shadowLight\";\r\nimport type { IEffectCreationOptions } from \"core/Materials/effect\";\r\nimport { MaterialDefines } from \"core/Materials/materialDefines\";\r\nimport { MaterialHelper } from \"core/Materials/materialHelper\";\r\nimport { PushMaterial } from \"core/Materials/pushMaterial\";\r\nimport { VertexBuffer } from \"core/Buffers/buffer\";\r\nimport type { AbstractMesh } from \"core/Meshes/abstractMesh\";\r\nimport type { SubMesh } from \"core/Meshes/subMesh\";\r\nimport type { Mesh } from \"core/Meshes/mesh\";\r\nimport { Scene } from \"core/scene\";\r\nimport { RegisterClass } from \"core/Misc/typeStore\";\r\n\r\nimport \"./shadowOnly.fragment\";\r\nimport \"./shadowOnly.vertex\";\r\nimport { EffectFallbacks } from \"core/Materials/effectFallbacks\";\r\nimport type { CascadedShadowGenerator } from \"core/Lights/Shadows/cascadedShadowGenerator\";\r\nimport { addClipPlaneUniforms, bindClipPlane } from \"core/Materials/clipPlaneMaterialHelper\";\r\n\r\nclass ShadowOnlyMaterialDefines extends MaterialDefines {\r\n public CLIPPLANE = false;\r\n public CLIPPLANE2 = false;\r\n public CLIPPLANE3 = false;\r\n public CLIPPLANE4 = false;\r\n public CLIPPLANE5 = false;\r\n public CLIPPLANE6 = false;\r\n public POINTSIZE = false;\r\n public FOG = false;\r\n public NORMAL = false;\r\n public NUM_BONE_INFLUENCERS = 0;\r\n public BonesPerMesh = 0;\r\n public INSTANCES = false;\r\n public IMAGEPROCESSINGPOSTPROCESS = false;\r\n public SKIPFINALCOLORCLAMP = false;\r\n\r\n constructor() {\r\n super();\r\n this.rebuild();\r\n }\r\n}\r\n\r\nexport class ShadowOnlyMaterial extends PushMaterial {\r\n private _activeLight: IShadowLight;\r\n private _needAlphaBlending = true;\r\n\r\n constructor(name: string, scene?: Scene) {\r\n super(name, scene);\r\n }\r\n\r\n public shadowColor = Color3.Black();\r\n\r\n public needAlphaBlending(): boolean {\r\n return this._needAlphaBlending;\r\n }\r\n\r\n public needAlphaTesting(): boolean {\r\n return false;\r\n }\r\n\r\n public getAlphaTestTexture(): Nullable {\r\n return null;\r\n }\r\n\r\n public get activeLight(): IShadowLight {\r\n return this._activeLight;\r\n }\r\n\r\n public set activeLight(light: IShadowLight) {\r\n this._activeLight = light;\r\n }\r\n\r\n private _getFirstShadowLightForMesh(mesh: AbstractMesh): Nullable {\r\n for (const light of mesh.lightSources) {\r\n if (light.shadowEnabled) {\r\n return light as IShadowLight;\r\n }\r\n }\r\n return null;\r\n }\r\n\r\n // Methods\r\n public isReadyForSubMesh(mesh: AbstractMesh, subMesh: SubMesh, useInstances?: boolean): boolean {\r\n if (this.isFrozen) {\r\n if (subMesh.effect && subMesh.effect._wasPreviouslyReady && subMesh.effect._wasPreviouslyUsingInstances === useInstances) {\r\n return true;\r\n }\r\n }\r\n\r\n if (!subMesh.materialDefines) {\r\n subMesh.materialDefines = new ShadowOnlyMaterialDefines();\r\n }\r\n\r\n const defines = subMesh.materialDefines;\r\n const scene = this.getScene();\r\n\r\n if (this._isReadyForSubMesh(subMesh)) {\r\n return true;\r\n }\r\n\r\n const engine = scene.getEngine();\r\n\r\n // Ensure that active light is the first shadow light\r\n if (this._activeLight) {\r\n for (const light of mesh.lightSources) {\r\n if (light.shadowEnabled) {\r\n if (this._activeLight === light) {\r\n break; // We are good\r\n }\r\n\r\n const lightPosition = mesh.lightSources.indexOf(this._activeLight);\r\n\r\n if (lightPosition !== -1) {\r\n mesh.lightSources.splice(lightPosition, 1);\r\n mesh.lightSources.splice(0, 0, this._activeLight);\r\n }\r\n break;\r\n }\r\n }\r\n }\r\n\r\n MaterialHelper.PrepareDefinesForFrameBoundValues(scene, engine, this, defines, useInstances ? true : false);\r\n\r\n MaterialHelper.PrepareDefinesForMisc(mesh, scene, false, this.pointsCloud, this.fogEnabled, this._shouldTurnAlphaTestOn(mesh), defines);\r\n\r\n defines._needNormals = MaterialHelper.PrepareDefinesForLights(scene, mesh, defines, false, 1);\r\n\r\n const shadowGenerator = this._getFirstShadowLightForMesh(mesh)?.getShadowGenerator();\r\n\r\n this._needAlphaBlending = true;\r\n\r\n if (shadowGenerator && (shadowGenerator as any).getClassName && (shadowGenerator as any).getClassName() === \"CascadedShadowGenerator\") {\r\n const csg = shadowGenerator as CascadedShadowGenerator;\r\n\r\n this._needAlphaBlending = !csg.autoCalcDepthBounds;\r\n }\r\n\r\n // Attribs\r\n MaterialHelper.PrepareDefinesForAttributes(mesh, defines, false, true);\r\n\r\n // Get correct effect\r\n if (defines.isDirty) {\r\n defines.markAsProcessed();\r\n\r\n scene.resetCachedMaterial();\r\n\r\n // Fallbacks\r\n const fallbacks = new EffectFallbacks();\r\n if (defines.FOG) {\r\n fallbacks.addFallback(1, \"FOG\");\r\n }\r\n\r\n MaterialHelper.HandleFallbacksForShadows(defines, fallbacks, 1);\r\n\r\n if (defines.NUM_BONE_INFLUENCERS > 0) {\r\n fallbacks.addCPUSkinningFallback(0, mesh);\r\n }\r\n\r\n defines.IMAGEPROCESSINGPOSTPROCESS = scene.imageProcessingConfiguration.applyByPostProcess;\r\n\r\n //Attributes\r\n const attribs = [VertexBuffer.PositionKind];\r\n\r\n if (defines.NORMAL) {\r\n attribs.push(VertexBuffer.NormalKind);\r\n }\r\n\r\n MaterialHelper.PrepareAttributesForBones(attribs, mesh, defines, fallbacks);\r\n MaterialHelper.PrepareAttributesForInstances(attribs, defines);\r\n\r\n const shaderName = \"shadowOnly\";\r\n const join = defines.toString();\r\n const uniforms = [\"world\", \"view\", \"viewProjection\", \"vEyePosition\", \"vLightsType\", \"vFogInfos\", \"vFogColor\", \"pointSize\", \"alpha\", \"shadowColor\", \"mBones\"];\r\n const samplers = new Array();\r\n\r\n const uniformBuffers = new Array();\r\n\r\n addClipPlaneUniforms(uniforms);\r\n MaterialHelper.PrepareUniformsAndSamplersList({\r\n uniformsNames: uniforms,\r\n uniformBuffersNames: uniformBuffers,\r\n samplers: samplers,\r\n defines: defines,\r\n maxSimultaneousLights: 1,\r\n });\r\n\r\n subMesh.setEffect(\r\n scene.getEngine().createEffect(\r\n shaderName,\r\n {\r\n attributes: attribs,\r\n uniformsNames: uniforms,\r\n uniformBuffersNames: uniformBuffers,\r\n samplers: samplers,\r\n defines: join,\r\n fallbacks: fallbacks,\r\n onCompiled: this.onCompiled,\r\n onError: this.onError,\r\n indexParameters: { maxSimultaneousLights: 1 },\r\n },\r\n engine\r\n ),\r\n defines,\r\n this._materialContext\r\n );\r\n }\r\n if (!subMesh.effect || !subMesh.effect.isReady()) {\r\n return false;\r\n }\r\n\r\n defines._renderId = scene.getRenderId();\r\n subMesh.effect._wasPreviouslyReady = true;\r\n subMesh.effect._wasPreviouslyUsingInstances = !!useInstances;\r\n\r\n return true;\r\n }\r\n\r\n public bindForSubMesh(world: Matrix, mesh: Mesh, subMesh: SubMesh): void {\r\n const scene = this.getScene();\r\n\r\n const defines = subMesh.materialDefines;\r\n if (!defines) {\r\n return;\r\n }\r\n\r\n const effect = subMesh.effect;\r\n if (!effect) {\r\n return;\r\n }\r\n this._activeEffect = effect;\r\n\r\n // Matrices\r\n this.bindOnlyWorldMatrix(world);\r\n this._activeEffect.setMatrix(\"viewProjection\", scene.getTransformMatrix());\r\n\r\n // Bones\r\n MaterialHelper.BindBonesParameters(mesh, this._activeEffect);\r\n\r\n if (this._mustRebind(scene, effect)) {\r\n // Clip plane\r\n bindClipPlane(effect, this, scene);\r\n\r\n // Point size\r\n if (this.pointsCloud) {\r\n this._activeEffect.setFloat(\"pointSize\", this.pointSize);\r\n }\r\n\r\n this._activeEffect.setFloat(\"alpha\", this.alpha);\r\n this._activeEffect.setColor3(\"shadowColor\", this.shadowColor);\r\n\r\n scene.bindEyePosition(effect);\r\n }\r\n\r\n // Lights\r\n if (scene.lightsEnabled) {\r\n MaterialHelper.BindLights(scene, mesh, this._activeEffect, defines, 1);\r\n\r\n const light = this._getFirstShadowLightForMesh(mesh);\r\n\r\n if (light) {\r\n // Make sure the uniforms for this light will be rebound for other materials using this light when rendering the current frame.\r\n // Indeed, there is an optimization in Light that binds the light uniforms only once per frame for a given light (if using ubo).\r\n // Doing this way assumes that all uses of this light are the same, meaning all parameters passed to Light._bindLlight\r\n // are the same, notably useSpecular. However, isReadyForSubMesh (see above) is passing false for this parameter, which may not be\r\n // the value the other materials may pass.\r\n light._renderId = -1;\r\n }\r\n }\r\n\r\n // View\r\n if ((scene.fogEnabled && mesh.applyFog && scene.fogMode !== Scene.FOGMODE_NONE) || defines[\"SHADOWCSM0\"]) {\r\n this._activeEffect.setMatrix(\"view\", scene.getViewMatrix());\r\n }\r\n\r\n // Fog\r\n MaterialHelper.BindFogParameters(scene, mesh, this._activeEffect);\r\n\r\n this._afterBind(mesh, this._activeEffect);\r\n }\r\n\r\n public clone(name: string): ShadowOnlyMaterial {\r\n return SerializationHelper.Clone(() => new ShadowOnlyMaterial(name, this.getScene()), this);\r\n }\r\n\r\n public serialize(): any {\r\n const serializationObject = super.serialize();\r\n serializationObject.customType = \"BABYLON.ShadowOnlyMaterial\";\r\n return serializationObject;\r\n }\r\n\r\n public getClassName(): string {\r\n return \"ShadowOnlyMaterial\";\r\n }\r\n\r\n // Statics\r\n public static Parse(source: any, scene: Scene, rootUrl: string): ShadowOnlyMaterial {\r\n return SerializationHelper.Parse(() => new ShadowOnlyMaterial(source.name, scene), source, scene, rootUrl);\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.ShadowOnlyMaterial\", ShadowOnlyMaterial);\r\n","import { Matrix, Quaternion } from '@babylonjs/core/Maths';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { VertexData } from '@babylonjs/core/Meshes/mesh.vertexData';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { ShadowOnlyMaterial } from '@babylonjs/Materials/shadowOnly/shadowOnlyMaterial';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { Subscription } from 'rxjs';\r\nimport { BoundingBoxInfo } from '../bounding-box-cache';\r\n\r\n/** padding in size to make enough space for shadow blur */\r\nconst SIZE_RATIO = 4;\r\n\r\nexport class XrShadowPlane extends Mesh {\r\n constructor(public v: KbViewer, public scene: Scene) {\r\n super('xrShadowPlane', scene);\r\n\r\n this.receiveShadows = true;\r\n this.isPickable = false;\r\n this.shadowMaterial = new ShadowOnlyMaterial('xrShadowMaterial', scene);\r\n this.shadowMaterial.alpha = 0.4;\r\n\r\n this.material = this.shadowMaterial;\r\n // this.material = v.rendererManager.material.defaultMaterial;\r\n\r\n this.setEnabled(false);\r\n }\r\n\r\n private subscription: Subscription = new Subscription();\r\n private shadowMaterial: ShadowOnlyMaterial;\r\n\r\n public place(rootNode: TransformNode) {\r\n this.setParent(rootNode);\r\n\r\n const bbCache = this.v.bbCache;\r\n const bb = bbCache.get(this.v.sceneNode)!;\r\n if (bb) {\r\n this.resize(bb);\r\n this.setEnabled(true);\r\n\r\n this.subscription.add(\r\n bbCache.rootBoundingBoxChanged.subscribe(newBb => {\r\n this.resize(bb);\r\n })\r\n );\r\n }\r\n }\r\n\r\n protected resize(bb: BoundingBoxInfo) {\r\n const width = (bb.max.x - bb.min.x) * SIZE_RATIO;\r\n const height = (bb.max.z - bb.min.z) * SIZE_RATIO;\r\n\r\n const vertexData = VertexData.CreatePlane({\r\n width,\r\n height,\r\n sideOrientation: 0,\r\n });\r\n //bake in a 90 degree rotation so it's flat on the ground\r\n const rotMatrix = new Matrix();\r\n Quaternion.FromEulerAngles(Math.PI / 2, 0, 0).toRotationMatrix(rotMatrix);\r\n vertexData.transform(rotMatrix);\r\n vertexData.applyToMesh(this, true);\r\n\r\n this.position.set(bb.center.x, 0, bb.center.z);\r\n }\r\n\r\n public dispose() {\r\n this.subscription.unsubscribe();\r\n super.dispose();\r\n if (this.shadowMaterial) this.shadowMaterial.dispose();\r\n }\r\n}\r\n","import { WebXRSessionManager } from '@babylonjs/core/XR/webXRSessionManager';\r\nimport { IXrHitResult, XrHitTester } from './xr-hit-tester';\r\n\r\nexport interface IXrTransientHitResult {\r\n inputSource: XRInputSource;\r\n results: IXrHitResult[];\r\n}\r\n\r\nexport class XrTransientHitTester {\r\n constructor(public sessionMgr: WebXRSessionManager) {}\r\n\r\n /** holds the hit test source which is filled after initialize() has been called */\r\n public source: XRTransientInputHitTestSource | undefined;\r\n\r\n public initialize(offsetRay?: XRRay) {\r\n const session = this.sessionMgr.session;\r\n\r\n return session.requestHitTestSourceForTransientInput!({\r\n profile: 'generic-touchscreen',\r\n offsetRay: offsetRay,\r\n }).then(hitSource => {\r\n return (this.source = hitSource);\r\n });\r\n }\r\n\r\n public getFingers(frame: XRFrame): IXrTransientHitResult[] {\r\n if (!this.source) return [];\r\n\r\n const fingers = frame.getHitTestResultsForTransientInput(this.source);\r\n if (!fingers.length) return [];\r\n\r\n const convertedFingers = new Array();\r\n for (const f of fingers) {\r\n const r = {\r\n inputSource: f.inputSource,\r\n results: [],\r\n } as IXrTransientHitResult;\r\n convertedFingers.push(r);\r\n\r\n for (const hit of f.results) {\r\n const convertedHit = XrHitTester.FromXRHitTestResult(hit, this.sessionMgr.referenceSpace);\r\n if (convertedHit) r.results.push(convertedHit);\r\n }\r\n }\r\n\r\n return convertedFingers;\r\n }\r\n\r\n public dispose() {\r\n if (this.source) {\r\n this.source.cancel();\r\n this.source = undefined;\r\n }\r\n }\r\n}\r\n","import { Camera } from '@babylonjs/core/Cameras/camera';\r\nimport { PointerEventTypes } from '@babylonjs/core/Events/pointerEvents';\r\nimport { Quaternion, Vector2, Vector3 } from '@babylonjs/core/Maths';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { WebXRCamera } from '@babylonjs/core/XR/webXRCamera';\r\nimport { WebXRSessionManager } from '@babylonjs/core/XR/webXRSessionManager';\r\nimport { WebXRRenderTarget } from '@babylonjs/core/XR/webXRTypes';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { eBackground } from '@models/classes/enums';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { Subject } from 'rxjs';\r\nimport { XrHitTester } from './xr-hit-tester';\r\nimport { XrLight } from './xr-light';\r\nimport { XrMarker } from './xr-marker';\r\nimport { XrShadowPlane } from './xr-shadow-plane';\r\nimport { IXrTransientHitResult, XrTransientHitTester } from './xr-transient-hit-tester';\r\n\r\n/**\r\n * Stub for future lighting estimation types\r\n */\r\ndeclare global {\r\n interface XRLightEstimate {\r\n readonly sphericalHarmonicsCoefficients: Float32Array;\r\n readonly primaryLightDirection: DOMPointReadOnly;\r\n readonly primaryLightIntensity: DOMPointReadOnly;\r\n }\r\n interface XRLightProbe {\r\n readonly probeSpace: XRSpace;\r\n }\r\n interface XRSession {\r\n requestLightProbe(): Promise;\r\n }\r\n interface XRFrame {\r\n getLightEstimate(lightProbe: XRLightProbe): XRLightEstimate;\r\n }\r\n}\r\n// interface XRSystem extends EventTarget {\r\n// requestSession(mode: XRSessionMode, options?: XRSessionInit): Promise;\r\n// isSessionSupported(mode: XRSessionMode): Promise;\r\n// }\r\n\r\n// declare global {\r\n// interface Navigator {\r\n// xr?: XRSystem;\r\n// }\r\n// }\r\n\r\nenum eXrState {\r\n searchingForHit,\r\n initialPlacement,\r\n placed,\r\n}\r\n\r\nconst ROTATION_RATE = 1.5;\r\nconst SCALE_SNAP = 1.2;\r\n\r\nexport class Xr {\r\n constructor(public v: KbViewer, public scene: Scene) {}\r\n\r\n public onSessionEnd = new Subject();\r\n private _sessionMgr: WebXRSessionManager;\r\n private _session: XRSession;\r\n private _renderTarget: WebXRRenderTarget;\r\n private _state = eXrState.searchingForHit;\r\n private _originalSceneCamera: Camera | null;\r\n private _originalBackground: eBackground;\r\n private _xrCamera: WebXRCamera;\r\n private _refSpace: XRReferenceSpace;\r\n private _initialHitTester: XrHitTester | undefined;\r\n private _transientHitTester: XrTransientHitTester | undefined;\r\n private _marker: XrMarker;\r\n private _light: XrLight;\r\n private _shadow: XrShadowPlane;\r\n private _rootNode: TransformNode;\r\n private _viewerPose: XRViewerPose | undefined;\r\n private _lastFrame: XRFrame | undefined;\r\n private _isTranslating: boolean;\r\n private _isRotating: boolean;\r\n private _isScaling: boolean;\r\n private _goalScale = 0;\r\n private _goalPosition: Vector3;\r\n private _goalYaw = 0;\r\n private _lastScalar = 0;\r\n private _dragNodeStartPos: Vector3 | undefined;\r\n private _selectStartScreenPos: Vector2 | undefined;\r\n private _fingerStartPos: Vector3 | undefined;\r\n private _dragHitTester: XrTransientHitTester | undefined;\r\n private _scaleSnapHigh: number = SCALE_SNAP;\r\n private _scaleSnapLow: number = 1 / SCALE_SNAP;\r\n private _naturalScale = 1;\r\n private _lightProbe: XRLightProbe | undefined;\r\n private _wasJustPlaced = false;\r\n private sessionEndingPromise?: Promise;\r\n\r\n public startSession(mode: XRSessionMode, refSpaceType: XRReferenceSpaceType, overlayElem?: HTMLElement) {\r\n const xrInitOptions = {\r\n optionalFeatures: ['light-estimation'],\r\n requiredFeatures: ['hit-test'],\r\n } as XRSessionInit;\r\n\r\n if (overlayElem) {\r\n overlayElem.addEventListener('beforexrselect', ev => {\r\n if (ev.target != overlayElem) {\r\n ev.preventDefault();\r\n }\r\n });\r\n xrInitOptions.optionalFeatures?.push('dom-overlay');\r\n (xrInitOptions as any).domOverlay = {\r\n root: overlayElem,\r\n };\r\n }\r\n if (refSpaceType !== 'viewer' && refSpaceType !== 'local') {\r\n xrInitOptions.optionalFeatures?.push(refSpaceType);\r\n }\r\n\r\n this._naturalScale = 1 / this.v.sceneNode.unitsPerMeter;\r\n this._sessionMgr = new WebXRSessionManager(this.scene);\r\n this._sessionMgr.onXRSessionEnded.addOnce(() => this.sessionEnded());\r\n\r\n this._xrCamera = new WebXRCamera('', this.scene, this._sessionMgr);\r\n\r\n return this._sessionMgr\r\n .initializeAsync()\r\n .then(() => {\r\n this._renderTarget = this._sessionMgr.getWebXRRenderTarget();\r\n return this._sessionMgr.initializeSessionAsync(mode, xrInitOptions).then(() => {\r\n this._session = this._sessionMgr.session;\r\n });\r\n })\r\n .then(() => {\r\n return this._sessionMgr.setReferenceSpaceTypeAsync(refSpaceType).then(refSpace => {\r\n this._refSpace = refSpace;\r\n });\r\n })\r\n .then(() => {\r\n return this._renderTarget.initializeXRLayerAsync(this._sessionMgr.session);\r\n })\r\n .then(() => {\r\n return this._sessionMgr.updateRenderState({\r\n depthFar: this._xrCamera!.maxZ,\r\n depthNear: this._xrCamera!.minZ,\r\n baseLayer: this._renderTarget.xrLayer!,\r\n });\r\n })\r\n .then(() => {\r\n this._rootNode = this.v.rendererManager.sceneRenderer.get3d(this.v.sceneNode)!; //get the root transform node for our scene... this is what we will move to position the scene in AR\r\n this._rootNode.setEnabled(false);\r\n\r\n //subscribe to render loop\r\n this._sessionMgr.onXRFrameObservable.add((frame: XRFrame) => this.onXRFrame(frame));\r\n this._sessionMgr.runXRRenderLoop(); //start the render loop\r\n\r\n this._originalSceneCamera = this.scene.activeCamera;\r\n this.scene.activeCamera = this._xrCamera!; //this line is important\r\n this.scene.autoClear = false;\r\n this._xrCamera!.compensateOnFirstFrame = false;\r\n\r\n this._originalBackground = this.v.sceneNode.background;\r\n this.v.sceneNode.background = eBackground.transparent;\r\n\r\n //add xr input handlers\r\n this._session.addEventListener('selectstart', this.onSelectStart);\r\n this._session.addEventListener('selectend', this.onSelectEnd);\r\n\r\n this._initialHitTester = new XrHitTester(this._sessionMgr);\r\n return this._initialHitTester.initialize();\r\n });\r\n }\r\n\r\n public endSession() {\r\n if (this._sessionMgr) {\r\n this.sessionEndingPromise = this._sessionMgr.exitXRAsync();\r\n return this.sessionEndingPromise;\r\n } else {\r\n return Promise.resolve();\r\n }\r\n }\r\n\r\n protected sessionEnded() {\r\n if (this.sessionEndingPromise) {\r\n this.sessionEndingPromise.then(() => {\r\n this.sessionEndingPromise = undefined;\r\n this.sessionEnded();\r\n });\r\n } else {\r\n this._session.removeEventListener('selectstart', this.onSelectStart);\r\n this._session.removeEventListener('selectend', this.onSelectEnd);\r\n\r\n if (this._xrCamera) this._xrCamera.dispose();\r\n if (this._sessionMgr) this._sessionMgr.dispose();\r\n if (this._renderTarget) this._renderTarget.dispose();\r\n if (this._initialHitTester) this._initialHitTester.dispose();\r\n if (this._marker) this._marker.dispose();\r\n if (this._light) this._light.dispose();\r\n if (this._shadow) this._shadow.dispose();\r\n\r\n if (this._originalSceneCamera) {\r\n this.scene.activeCamera = this._originalSceneCamera;\r\n }\r\n this.scene.autoClear = true;\r\n this.v.sceneNode.background = this._originalBackground || eBackground.transparent;\r\n this._rootNode.scaling = Vector3.One();\r\n this._rootNode.rotation = Vector3.Zero();\r\n this._rootNode.position = Vector3.Zero();\r\n this._rootNode.setEnabled(true);\r\n this.onSessionEnd.next();\r\n }\r\n }\r\n\r\n protected onXRFrame(frame: XRFrame) {\r\n this._lastFrame = frame;\r\n //console.log(`user height: ${this._xrCamera.realWorldHeight}`);\r\n this._viewerPose = frame.getViewerPose(this._sessionMgr.referenceSpace);\r\n\r\n this.placeInitialMarker(frame);\r\n\r\n this.processInput(frame);\r\n this.setSceneTransform();\r\n this.runLightingEstimation(frame);\r\n }\r\n\r\n private _lightingEstimationCount = 0;\r\n protected runLightingEstimation(frame: XRFrame) {\r\n if (this._lightProbe) {\r\n this._lightingEstimationCount++;\r\n if (this._lightingEstimationCount == 30) {\r\n this._lightingEstimationCount = 0;\r\n const estimate = frame.getLightEstimate(this._lightProbe);\r\n // const probePose = frame.getPose(this._lightProbe.probeSpace, this._refSpace);\r\n // console.log({probePose});\r\n // console.log({ lightEstimate: estimate });\r\n if (estimate && this._light) {\r\n const direction = new Vector3(\r\n estimate.primaryLightDirection.x,\r\n estimate.primaryLightDirection.y,\r\n estimate.primaryLightDirection.z\r\n );\r\n direction.negateInPlace();\r\n this._light.setDirection(direction.x, direction.y, direction.z);\r\n\r\n const intensity = Math.max(\r\n 1.0,\r\n Math.max(\r\n estimate.primaryLightIntensity.x,\r\n Math.max(estimate.primaryLightIntensity.y, estimate.primaryLightIntensity.z)\r\n )\r\n );\r\n\r\n this._light.specular.set(\r\n estimate.primaryLightIntensity.x / intensity,\r\n estimate.primaryLightIntensity.y / intensity,\r\n estimate.primaryLightIntensity.z / intensity\r\n );\r\n this._light.diffuse.set(\r\n estimate.primaryLightIntensity.x / intensity,\r\n estimate.primaryLightIntensity.y / intensity,\r\n estimate.primaryLightIntensity.z / intensity\r\n );\r\n\r\n this._light.intensity = intensity;\r\n }\r\n }\r\n }\r\n }\r\n\r\n protected placeInitialMarker(frame: XRFrame) {\r\n if (this._state == eXrState.searchingForHit || this._state == eXrState.initialPlacement) {\r\n const hit = this._initialHitTester!.getHitResult(frame);\r\n if (!hit) return;\r\n\r\n if (this._state == eXrState.searchingForHit) {\r\n if (!this._marker) {\r\n this._marker = new XrMarker(this.v, this.scene, this._naturalScale);\r\n this._marker.setEnabled(true);\r\n }\r\n this._state = eXrState.initialPlacement; //we got our first hit so we change state\r\n }\r\n\r\n if (this._state == eXrState.initialPlacement) {\r\n const quat = new Quaternion();\r\n hit.matrix.decompose(this._marker.scaling, quat, this._marker.position);\r\n this._marker.rotation = quat.toEulerAngles();\r\n // console.log(JSON.stringify({ markerPosition: this._marker.position }));\r\n }\r\n }\r\n }\r\n\r\n protected onSelectStart = (event: XRInputSourceEvent) => {\r\n if (this._state == eXrState.initialPlacement) {\r\n this.placeModel();\r\n this._state = eXrState.placed;\r\n\r\n this._initialHitTester!.dispose();\r\n this._initialHitTester = undefined;\r\n\r\n this._transientHitTester = new XrTransientHitTester(this._sessionMgr);\r\n this._transientHitTester.initialize();\r\n this._wasJustPlaced = true;\r\n } else if (this._state == eXrState.placed) {\r\n const fingers = this._transientHitTester?.getFingers(this._lastFrame!);\r\n\r\n if (fingers?.length === 1) {\r\n /** for one finger we do a hit test on kb3d to see if they pressed the model */\r\n this._selectStartScreenPos = this.screenNdcToPixels(event.inputSource.gamepad!.axes); //get the screen coordinates of the finger press\r\n const pickInfo = this.pick(this._selectStartScreenPos);\r\n if (pickInfo) {\r\n // console.log({ pickInfo });\r\n this._isTranslating = true;\r\n this._dragNodeStartPos = this._rootNode.position.clone();\r\n // this._dragNodeStartScreenPos = toScreen(this._rootNode.absolutePosition, this.getRigCamera());\r\n // this._fingerStartScreenPos = this.screenNdcToPixels(fingers[0].inputSource.gamepad!.axes);\r\n const dragHitTester = new XrTransientHitTester(this._sessionMgr);\r\n\r\n const offset = this._dragNodeStartPos.subtract(pickInfo.pickedPoint.toVec3());\r\n // console.log({nodePos: this._dragNodeStartPos, pickedPoint: pickInfo.pickedPoint});\r\n const offsetRay = new XRRay(offset);\r\n\r\n dragHitTester.initialize(offsetRay).then(() => {\r\n this._dragHitTester = dragHitTester;\r\n });\r\n } else {\r\n this._isRotating = true;\r\n this._goalYaw = this._rootNode.rotation.y;\r\n this._lastScalar = event.inputSource.gamepad!.axes[0];\r\n }\r\n\r\n //simulate pointer down observable\r\n const args = this.v.interaction.getKb3dPointerArgs({\r\n type: PointerEventTypes.POINTERDOWN,\r\n event: new PointerEvent('pointerdown', { pointerId: event.inputSource.gamepad!.id } as any),\r\n pickInfo: pickInfo!,\r\n });\r\n this.v.interaction.handlePointerDown(args);\r\n this._marker.show();\r\n } else if (fingers?.length === 2 && this.v.sceneNode.xrAllowScaling) {\r\n this._isScaling = true;\r\n this._lastScalar = this.getFingerSeparation(fingers) / this._rootNode.scaling.x;\r\n this._marker.show();\r\n }\r\n }\r\n };\r\n\r\n protected onSelectEnd = (event: XRInputSourceEvent) => {\r\n this._isTranslating = false;\r\n this._isRotating = false;\r\n this._isScaling = false;\r\n if (this._dragHitTester) {\r\n this._dragHitTester.dispose();\r\n }\r\n this._dragHitTester = undefined;\r\n this._fingerStartPos = undefined;\r\n\r\n //simulate pointer up observable\r\n const fingers = this._transientHitTester?.getFingers(this._lastFrame!);\r\n const screenCoord = this.screenNdcToPixels(event.inputSource.gamepad!.axes); //get the screen coordinates of the finger press\r\n const pickInfo = this.pick(screenCoord);\r\n const args = this.v.interaction.getKb3dPointerArgs({\r\n type: PointerEventTypes.POINTERUP,\r\n event: new PointerEvent('pointerup', { pointerId: event.inputSource.gamepad!.id } as any),\r\n pickInfo: pickInfo!,\r\n });\r\n this.v.interaction.handlePointerUp(args);\r\n\r\n //simulate the pointer tap observable if the input hasn't moved much\r\n if (this._state == eXrState.placed) {\r\n if (fingers?.length === 1 && !this._wasJustPlaced) {\r\n const deltaScreen = Vector2.Distance(this._selectStartScreenPos || screenCoord, screenCoord);\r\n if (deltaScreen < 5) {\r\n const args = this.v.interaction.getKb3dPointerArgs({\r\n type: PointerEventTypes.POINTERTAP,\r\n event: new PointerEvent('pointertap', { pointerId: event.inputSource.gamepad!.id } as any),\r\n pickInfo: pickInfo!,\r\n });\r\n this.v.interaction.handleClick(args);\r\n }\r\n }\r\n\r\n this._marker.hide();\r\n this._wasJustPlaced = false;\r\n }\r\n };\r\n\r\n protected processInput(frame: XRFrame) {\r\n if (this._state == eXrState.placed && (this._isTranslating || this._isRotating || this._isScaling)) {\r\n const fingers = this._transientHitTester!.getFingers(frame);\r\n\r\n if (this._isScaling) {\r\n if (fingers.length < 2) {\r\n //the user removed their 2nd finger, so the scale is cancelled\r\n this._isScaling = false;\r\n } else {\r\n const separation = this.getFingerSeparation(fingers);\r\n const scale = separation / this._lastScalar;\r\n this._goalScale =\r\n scale < this._scaleSnapHigh && scale > this._scaleSnapLow ? this._naturalScale : scale;\r\n }\r\n } else if (fingers.length === 2 && this.v.sceneNode.xrAllowScaling) {\r\n // happens when translating or rotating and another finger was added. We switch to scaling in this instance\r\n this._isTranslating = false;\r\n this._isRotating = false;\r\n this._isScaling = true;\r\n const scale = this._rootNode.scaling.x;\r\n this._lastScalar = this.getFingerSeparation(fingers) / scale;\r\n } else if (this._isRotating) {\r\n const inputSource = fingers[0].inputSource;\r\n const dragX = inputSource.gamepad!.axes[0];\r\n this._goalYaw += (dragX - this._lastScalar) * ROTATION_RATE;\r\n this._lastScalar = dragX;\r\n } else if (this._isTranslating) {\r\n if (fingers.length > 0 && this._dragHitTester) {\r\n const dragFingers = this._dragHitTester.getFingers(frame);\r\n if (dragFingers.length && dragFingers[0].results.length) {\r\n const dragFinger = dragFingers[0];\r\n\r\n const dragFingerResult = dragFinger.results[0];\r\n if (!this._fingerStartPos) this._fingerStartPos = dragFingerResult.position;\r\n\r\n const deltaPos = dragFingerResult.position.subtract(this._fingerStartPos);\r\n this._goalPosition = this._dragNodeStartPos!.add(deltaPos);\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n protected placeModel() {\r\n this._goalPosition = this._marker.position.clone();\r\n this._goalYaw = this._marker.rotation.y;\r\n this._goalScale = this._naturalScale;\r\n this._scaleSnapHigh = this._naturalScale * SCALE_SNAP;\r\n this._scaleSnapLow = this._naturalScale / SCALE_SNAP;\r\n\r\n //we now add the marker to the root node so it can scale and move with the model\r\n this._marker.place(this._rootNode);\r\n\r\n if (this.v.sceneNode.xrAddShadow) {\r\n this._shadow = new XrShadowPlane(this.v, this.scene);\r\n this._shadow.place(this._rootNode);\r\n\r\n this._light = new XrLight(this.v, this.scene);\r\n\r\n if (this.v.sceneNode.xrLightingEstimation && this._session.requestLightProbe) {\r\n this._session.requestLightProbe().then(lp => (this._lightProbe = lp));\r\n }\r\n }\r\n\r\n this._rootNode.setEnabled(true);\r\n\r\n //this._marker.setEnabled(false);\r\n this._state = eXrState.placed;\r\n }\r\n\r\n /** does a pick of the kb3d scene (NOT a webxr hit test). Accepts x/y as screen pixel coordinates */\r\n protected pick(screenCoord: Vector2) {\r\n /**\r\n * The WebXRCamera in babylon appears to be for reference only, and it's position/rotation\r\n * are updated on each frame to be the same as the viewer pose from XR.\r\n *\r\n * However, the forward ray from the WebXRCamera is not what we would expect, it's pointing\r\n * in exactly the opposite direction. However, inside the WebXRCamera are rig cameras (only 1\r\n * in the case of AR) who are responsible for actually doing the babylon rendering. The rig\r\n * camera direction is what we would expect. For now we are using the rig camera to do\r\n * the picking, but we should keep an eye on this part as new babylon/webxr releases come out,\r\n * as something seems amiss.\r\n */\r\n const hardwareScale = this.scene.getEngine().getHardwareScalingLevel();\r\n /** adjust by hardware scale as webxr does not support it, but the pick routine uses it */\r\n const pickInfo = this.v.interaction.pick(\r\n screenCoord.x * hardwareScale,\r\n screenCoord.y * hardwareScale,\r\n this.getRigCamera()\r\n );\r\n\r\n return pickInfo;\r\n }\r\n\r\n protected getFingerSeparation(fingers: IXrTransientHitResult[]): number {\r\n const firstFinger = fingers[0].inputSource.gamepad!.axes;\r\n const secondFinger = fingers[1].inputSource.gamepad!.axes;\r\n const deltaX = secondFinger[0] - firstFinger[0];\r\n const deltaY = secondFinger[1] - firstFinger[1];\r\n return Math.sqrt(deltaX * deltaX + deltaY * deltaY);\r\n }\r\n\r\n /** set's the transforms of the scene to the goal values */\r\n protected setSceneTransform() {\r\n if (this._state == eXrState.placed) {\r\n let invalidateRootBoundingBox = false;\r\n if (this._rootNode.scaling.x != this._goalScale) {\r\n this._rootNode.scaling = new Vector3(this._goalScale, this._goalScale, this._goalScale);\r\n invalidateRootBoundingBox = true;\r\n }\r\n if (this._rootNode.rotation.y != this._goalYaw) {\r\n this._rootNode.rotation.y = this._goalYaw; // = Quaternion.FromEulerAngles(0, this._goalYaw, 0);\r\n // this._marker.rotation.y = this._goalYaw;\r\n invalidateRootBoundingBox = true;\r\n }\r\n\r\n if (\r\n this._goalPosition &&\r\n !this._goalPosition.equalsToFloats(\r\n this._rootNode.position.x,\r\n this._rootNode.position.y,\r\n this._rootNode.position.z\r\n )\r\n ) {\r\n this._rootNode.position.set(this._goalPosition.x, this._goalPosition.y, this._goalPosition.z);\r\n //this._marker.position.set(this._goalPosition.x, this._goalPosition.y, this._goalPosition.z);\r\n invalidateRootBoundingBox = true;\r\n }\r\n\r\n if (invalidateRootBoundingBox) {\r\n this.v.bbCache.invalidate(this.v.sceneNode);\r\n }\r\n }\r\n }\r\n\r\n protected getRigCamera() {\r\n const rigCamera = this._xrCamera.rigCameras[0];\r\n return rigCamera;\r\n }\r\n\r\n protected screenNdcToPixels(axes: readonly number[]) {\r\n const canvas = this.scene.getEngine().getRenderingCanvas() as HTMLCanvasElement;\r\n //need to convert NDC units to pixels\r\n const x = (axes[0] + 1) * (canvas.width / 2);\r\n const y = (axes[1] + 1) * (canvas.height / 2);\r\n\r\n return new Vector2(x, y);\r\n }\r\n\r\n protected screenPixelsToNdc(pos: { x: number; y: number }) {\r\n const canvas = this.scene.getEngine().getRenderingCanvas() as HTMLCanvasElement;\r\n return {\r\n x: pos.x / (canvas.width / 2) - 1,\r\n y: pos.y / (canvas.height / 2) - 2,\r\n };\r\n }\r\n}\r\n","import '@babylonjs/core/XR/features/WebXRAnchorSystem';\r\nimport '@babylonjs/core/XR/features/WebXRHitTest';\r\nimport '@babylonjs/core/XR/features/WebXRPlaneDetector';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { Subject } from 'rxjs';\r\nimport { first } from 'rxjs/operators';\r\nimport { GLTFKb3dExporter } from './exporter-gltf';\r\nimport { USDZBabylonConverter } from './usdz/usdz-babylon-converter';\r\nimport { Xr } from './xr/xr';\r\n\r\nconst IS_IOS =\r\n /iPad|iPhone|iPod/.test(navigator.platform) || (navigator.platform === 'MacIntel' && navigator.maxTouchPoints > 1); //https://stackoverflow.com/a/58065241\r\n\r\nconst IS_ANDROID = /android/i.test(navigator.userAgent);\r\n\r\nexport class AR {\r\n currentObjectUrl?: string;\r\n isDebug = false;\r\n\r\n constructor(private v: KbViewer, private scene: Scene) {\r\n this.isDebug = !!v.options.debugMode;\r\n this.gltfExporter = new GLTFKb3dExporter(scene);\r\n }\r\n\r\n public xrActive = false;\r\n public xrLoading = false;\r\n public onSessionEnd = new Subject();\r\n\r\n private _xr: Xr;\r\n private gltfExporter: GLTFKb3dExporter;\r\n\r\n public toggle() {\r\n if (this.xrActive) {\r\n return this.end();\r\n } else {\r\n return this.start();\r\n }\r\n }\r\n\r\n public start(overlayElem?: HTMLElement, onEnterFunc?: () => PromiseLike) {\r\n // Have to use a callback because android should run this after entering XR, but ios must run it before entering XR\r\n if (!onEnterFunc) {\r\n onEnterFunc = () => Promise.resolve();\r\n }\r\n const _onEnterFunc = onEnterFunc;\r\n if (IS_ANDROID) {\r\n return this.startWebXR(overlayElem).then(() => _onEnterFunc());\r\n } else if (IS_IOS || this.isDebug) {\r\n const usdz = new USDZBabylonConverter();\r\n if (this.v.sceneNode) {\r\n const self = this;\r\n\r\n return this.v\r\n .whenAssetsLoaded()\r\n .then(() => _onEnterFunc())\r\n .then(() => this.v.forceAndWaitForNextRender())\r\n .then(() => usdz.parse(this.scene, this.v.sceneNode, true))\r\n .then(res => {\r\n const a = document.createElement('a');\r\n\r\n if (a.relList.supports('ar') || this.isDebug) {\r\n if (self.currentObjectUrl) {\r\n window.URL.revokeObjectURL(self.currentObjectUrl);\r\n self.currentObjectUrl = undefined;\r\n }\r\n\r\n self.currentObjectUrl = window.URL.createObjectURL(res);\r\n\r\n const img = document.createElement('img');\r\n img.src = '/images/ar_ios.png';\r\n a.appendChild(img);\r\n document.body.appendChild(a);\r\n a.href = self.currentObjectUrl;\r\n a.download = 'test.usdz';\r\n a.rel = 'ar';\r\n\r\n // Fire XR Exit right away on iOS since the USDZ is already generated and static now\r\n // So, whatever scene changes they want when the user exits AR should already be in the browser\r\n this.xrActive = false;\r\n this.xrLoading = false;\r\n this.onSessionEnd.next();\r\n\r\n setTimeout(() => {\r\n a.click();\r\n\r\n setTimeout(() => {\r\n document.body.removeChild(a);\r\n }, 0);\r\n }, 50);\r\n\r\n return undefined;\r\n }\r\n\r\n return Promise.reject('Your device does not support AR.');\r\n });\r\n } else {\r\n return Promise.reject('Scene not initialized');\r\n }\r\n }\r\n\r\n return Promise.reject('Your device does not support AR.');\r\n }\r\n\r\n public end() {\r\n if (IS_ANDROID) {\r\n return this.endWebXr();\r\n } else if (IS_IOS) {\r\n if (this.currentObjectUrl) {\r\n URL.revokeObjectURL(this.currentObjectUrl);\r\n }\r\n //TODO usdz\r\n }\r\n\r\n return Promise.resolve();\r\n }\r\n\r\n public startWebXR(overlayElem?: HTMLElement) {\r\n this._xr = new Xr(this.v, this.scene);\r\n this._xr.onSessionEnd.pipe(first()).subscribe(() => {\r\n this.xrActive = false;\r\n this.xrLoading = false;\r\n this.onSessionEnd.next();\r\n });\r\n this.xrLoading = true;\r\n return this._xr.startSession('immersive-ar', 'local-floor', overlayElem).then(() => {\r\n this.xrActive = true;\r\n this.xrLoading = false;\r\n });\r\n }\r\n\r\n protected endWebXr() {\r\n if (this._xr) {\r\n return this._xr.endSession();\r\n } else {\r\n return Promise.resolve();\r\n }\r\n }\r\n /**\r\n * Takes a URL to a USDZ file and sets the appropriate fields so that Safari\r\n * iOS can intent to their AR Quick Look.\r\n */\r\n public startIosQuickLook(usdzSrc: string, canScale: boolean) {\r\n const anchor = document.createElement('a');\r\n anchor.setAttribute('rel', 'ar');\r\n anchor.appendChild(document.createElement('img'));\r\n\r\n const modelUrl = new URL(usdzSrc, self.location.toString());\r\n if (!canScale) {\r\n modelUrl.hash = 'allowsContentScaling=0';\r\n }\r\n anchor.setAttribute('href', modelUrl.toString());\r\n anchor.click();\r\n }\r\n\r\n /**\r\n * Android SceneViewer is a native app. You build an intent link to tell\r\n * the browser to open scene viewer and show a GLB. It does not accept blob urls,\r\n * so the link must go to a file on the server: https://github.com/google/model-viewer/issues/617 *\r\n */\r\n // public startAndroidSceneViewer(glbSrc: string) {\r\n // this.gltfExporter.generateGlb().then(blob => {\r\n // let objectUrl = window.URL.createObjectURL(blob);\r\n // window.URL.revokeObjectURL(objectUrl);\r\n // //TODO: need to send GLB to server, as sceneViewer won't accept blob urls from the browser\r\n // const a = document.createElement('a');\r\n // const fallbackUrl = encodeURIComponent(self.location.toString());\r\n // /**temp hack */ objectUrl =\r\n // 'https://storage.googleapis.com/ar-answers-in-search-models/static/Tiger/model.glb';\r\n // const href = `intent://arvr.google.com/scene-viewer/1.0?file=${objectUrl}&mode=ar_preferred#Intent;scheme=https;package=com.google.android.googlequicksearchbox;action=android.intent.action.VIEW;S.browser_fallback_url=${fallbackUrl};end;`;\r\n // a.setAttribute('href', href);\r\n // a.click();\r\n // });\r\n // }\r\n}\r\n","import { Observable } from '../Observable';\nimport { Operator } from '../Operator';\nimport { Subscriber } from '../Subscriber';\nimport { EmptyError } from '../util/EmptyError';\nimport { OperatorFunction } from '../../internal/types';\nimport { filter } from './filter';\nimport { take } from './take';\nimport { defaultIfEmpty } from './defaultIfEmpty';\nimport { throwIfEmpty } from './throwIfEmpty';\nimport { identity } from '../util/identity';\n\n/* tslint:disable:max-line-length */\nexport function first(\n predicate?: null,\n defaultValue?: D\n): OperatorFunction;\nexport function first(\n predicate: (value: T, index: number, source: Observable) => value is S,\n defaultValue?: S\n): OperatorFunction;\nexport function first(\n predicate: (value: T, index: number, source: Observable) => boolean,\n defaultValue?: D\n): OperatorFunction;\n/* tslint:enable:max-line-length */\n\n/**\n * Emits only the first value (or the first value that meets some condition)\n * emitted by the source Observable.\n *\n * Emits only the first value. Or emits only the first\n * value that passes some test.\n *\n * ![](first.png)\n *\n * If called with no arguments, `first` emits the first value of the source\n * Observable, then completes. If called with a `predicate` function, `first`\n * emits the first value of the source that matches the specified condition. It\n * may also take a deprecated `resultSelector` function to produce the output\n * value from the input value, and a `defaultValue` to emit in case the source\n * completes before it is able to emit a valid value. Throws an error if\n * `defaultValue` was not provided and a matching element is not found.\n *\n * ## Examples\n * Emit only the first click that happens on the DOM\n * ```ts\n * import { fromEvent } from 'rxjs';\n * import { first } from 'rxjs/operators';\n *\n * const clicks = fromEvent(document, 'click');\n * const result = clicks.pipe(first());\n * result.subscribe(x => console.log(x));\n * ```\n *\n * Emits the first click that happens on a DIV\n * ```ts\n * import { fromEvent } from 'rxjs';\n * import { first } from 'rxjs/operators';\n *\n * const clicks = fromEvent(document, 'click');\n * const result = clicks.pipe(first(ev => ev.target.tagName === 'DIV'));\n * result.subscribe(x => console.log(x));\n * ```\n *\n * @see {@link filter}\n * @see {@link find}\n * @see {@link take}\n *\n * @throws {EmptyError} Delivers an EmptyError to the Observer's `error`\n * callback if the Observable completes before any `next` notification was sent.\n *\n * @param {function(value: T, index: number, source: Observable): boolean} [predicate]\n * An optional function called with each item to test for condition matching.\n * @param {R} [defaultValue] The default value emitted in case no valid value\n * was found on the source.\n * @return {Observable} An Observable of the first item that matches the\n * condition.\n * @method first\n * @owner Observable\n */\nexport function first(\n predicate?: ((value: T, index: number, source: Observable) => boolean) | null,\n defaultValue?: D\n): OperatorFunction {\n const hasDefaultValue = arguments.length >= 2;\n return (source: Observable) => source.pipe(\n predicate ? filter((v, i) => predicate(v, i, source)) : identity,\n take(1),\n hasDefaultValue ? defaultIfEmpty(defaultValue) : throwIfEmpty(() => new EmptyError()),\n );\n}\n","import { BoundingInfo } from '@babylonjs/core/Culling/boundingInfo';\r\nimport { Matrix, Vector3 } from '@babylonjs/core/Maths';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { getByDynamicId } from '@models/classes/kb3d-manager';\r\nimport { MeshNode } from '@models/classes/meshes/mesh-node';\r\nimport { SpaceNode } from '@models/classes/meshes/space-node';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { BehaviorSubject, Observable, Subject } from 'rxjs';\r\nimport { mergeMap } from 'rxjs/operators';\r\nimport { KbViewer } from '../kb-viewer';\r\nimport { Kb3dBoundingBoxChangeHandler } from '@models/classes/handlers';\r\n\r\nexport interface BoundingBoxInfo {\r\n boundingInfo: BoundingInfo;\r\n min: KbVector;\r\n max: KbVector;\r\n center: KbVector;\r\n}\r\n\r\nexport function toBoundingBoxInfo(binfo: BoundingInfo) {\r\n return {\r\n boundingInfo: binfo,\r\n min: KbVector.FromVec3(binfo.minimum),\r\n max: KbVector.FromVec3(binfo.maximum),\r\n center: KbVector.FromVec3(binfo.boundingBox.center),\r\n } as BoundingBoxInfo;\r\n}\r\n\r\ninterface IVectorLike {\r\n x?: number;\r\n y?: number;\r\n z?: number;\r\n}\r\n\r\nconst MAX_DIGESTS = 30;\r\n\r\nexport class BoundingBoxCache {\r\n constructor(private v: KbViewer) {}\r\n\r\n public boundingBoxInvalidated = new Subject();\r\n public rootBoundingBoxChanged = new Subject();\r\n private _digestCount = 0;\r\n private _db = new Map();\r\n private _watchedNodes = new Map>();\r\n private _changedNodes = new Set(); //stored dynamic id's of changed nodes\r\n\r\n public watchForChanges(node: SpaceNode): Observable {\r\n let subject = this._watchedNodes.get(node._dynamicId);\r\n\r\n if (!subject) {\r\n subject = new BehaviorSubject(undefined);\r\n this._watchedNodes.set(node._dynamicId, subject);\r\n }\r\n const _subject = subject;\r\n\r\n // This is using an internal implementation of rxjs, hence the any. But this is the only way we can add teardown logic.\r\n const observable = new Observable(s => {\r\n s.next();\r\n return () => {\r\n setTimeout(() => {\r\n if (_subject.observers.length === 0) {\r\n _subject.complete();\r\n this._watchedNodes.delete(node._dynamicId);\r\n }\r\n });\r\n };\r\n });\r\n\r\n return observable.pipe(mergeMap(() => _subject));\r\n }\r\n\r\n public refreshAndGet(node: SpaceNode) {\r\n this._db.delete(node._dynamicId);\r\n return this.get(node);\r\n }\r\n\r\n public get(node?: SpaceNode) {\r\n if (node && node.visible) {\r\n let bb = this._db.get(node._dynamicId);\r\n if (!bb) {\r\n let bbInfo: BoundingInfo | undefined;\r\n const bnode = this.v.rendererManager.mesh.get3d(node);\r\n if (node instanceof MeshNode) {\r\n if (bnode) {\r\n bbInfo = (bnode as AbstractMesh).getBoundingInfo();\r\n }\r\n } else {\r\n const childNodes = node.getSpaceChildren();\r\n const min: IVectorLike = {};\r\n const max: IVectorLike = {};\r\n let empty = true;\r\n\r\n for (const c of childNodes) {\r\n const cbb = this.get(c);\r\n // If cbb is undefined, that means the child node is not visible or doesn't exist\r\n if (cbb) {\r\n empty = false;\r\n const parentMin = cbb.min.toVec3();\r\n const parentMax = cbb.max.toVec3();\r\n const childBNode = this.v.rendererManager.mesh.get3d(c);\r\n if (childBNode) {\r\n const mat = Matrix.Compose(\r\n childBNode.scaling,\r\n childBNode.rotationQuaternion!,\r\n childBNode.position\r\n );\r\n Vector3.TransformCoordinatesToRef(parentMin, mat, parentMin);\r\n Vector3.TransformCoordinatesToRef(parentMax, mat, parentMax);\r\n }\r\n\r\n if (min.x == null) min.x = parentMin.x;\r\n if (min.y == null) min.y = parentMin.y;\r\n if (min.z == null) min.z = parentMin.z;\r\n if (max.x == null) max.x = parentMax.x;\r\n if (max.y == null) max.y = parentMax.y;\r\n if (max.z == null) max.z = parentMax.z;\r\n\r\n min.x = Math.min(min.x, parentMin.x, parentMax.x);\r\n min.y = Math.min(min.y, parentMin.y, parentMax.y);\r\n min.z = Math.min(min.z, parentMin.z, parentMax.z);\r\n max.x = Math.max(max.x, parentMin.x, parentMax.x);\r\n max.y = Math.max(max.y, parentMin.y, parentMax.y);\r\n max.z = Math.max(max.z, parentMin.z, parentMax.z);\r\n }\r\n }\r\n\r\n if (empty) {\r\n return undefined; //don't consider empty groups\r\n }\r\n\r\n bbInfo = new BoundingInfo(\r\n new Vector3(min.x || 0, min.y || 0, min.z || 0),\r\n new Vector3(max.x || 0, max.y || 0, max.z || 0)\r\n );\r\n }\r\n\r\n if (bbInfo) {\r\n bb = toBoundingBoxInfo(bbInfo);\r\n this._db.set(node._dynamicId, bb);\r\n }\r\n }\r\n return bb;\r\n } else {\r\n return undefined;\r\n }\r\n }\r\n\r\n public invalidate(node: SpaceNode | undefined) {\r\n if (node) {\r\n //when a node is invalidated, all of it's ancestors need to be invalidated\r\n let n = node;\r\n while (n && n instanceof SpaceNode) {\r\n this._changedNodes.add(n._dynamicId);\r\n n = n._parent as SpaceNode;\r\n }\r\n }\r\n }\r\n\r\n public update() {\r\n let boundingBoxesStable = true as null | boolean;\r\n\r\n if (this._changedNodes.size === 0) {\r\n return;\r\n }\r\n\r\n if(!this.v.userIsInteracting()) {\r\n this._digestCount++;\r\n }\r\n\r\n const watchersToUpdateAndClear = new Map();\r\n\r\n for (const dynamicId of this._changedNodes) {\r\n const n = getByDynamicId(dynamicId);\r\n if (!n) {\r\n continue;\r\n }\r\n const externalWatchers = n._manager?.boundingboxChangeHandlers.getByKey(n);\r\n const hasRootWatchers =\r\n this.v.sceneNode._dynamicId === n._dynamicId && this.rootBoundingBoxChanged.observers.length > 0;\r\n const hasWatchers =\r\n this._watchedNodes.has(n._dynamicId) || (externalWatchers !== undefined && externalWatchers.length > 0);\r\n\r\n const oldBoundingBox = this._db.get(n._dynamicId);\r\n this._db.delete(n._dynamicId);\r\n this.boundingBoxInvalidated.next(n);\r\n\r\n if (!hasWatchers && !hasRootWatchers) {\r\n continue;\r\n }\r\n\r\n const bbInfo = this.get(n);\r\n const boundingBoxChanged = !boundingBoxesMatch(bbInfo, oldBoundingBox);\r\n const bbWatchersSubject = this._watchedNodes.get(n._dynamicId);\r\n\r\n if (boundingBoxChanged) {\r\n boundingBoxesStable = false;\r\n }\r\n\r\n if (bbWatchersSubject) {\r\n bbWatchersSubject.next(bbInfo);\r\n }\r\n if (externalWatchers && bbInfo) {\r\n for (const w of externalWatchers) {\r\n const once = w.mode === 'once';\r\n // Always call the handler if the mode is 'once', because it will be expecting an update regardless of whether it changed or not.\r\n // For persistent watchers, only call it if the bb actually changed.\r\n if (boundingBoxChanged && !once) {\r\n w.handler(bbInfo);\r\n }\r\n if (once) {\r\n watchersToUpdateAndClear.set(n, w);\r\n }\r\n }\r\n }\r\n\r\n if (hasRootWatchers && bbInfo && boundingBoxChanged) {\r\n this.rootBoundingBoxChanged.next(bbInfo);\r\n }\r\n }\r\n\r\n if (boundingBoxesStable && this._digestCount > 3) {\r\n this._changedNodes.clear();\r\n for (const [node, watcher] of watchersToUpdateAndClear.entries()) {\r\n const bbInfo = this.get(node);\r\n if (bbInfo) {\r\n watcher.handler(bbInfo);\r\n }\r\n node._manager?.boundingboxChangeHandlers.deleteByKeyValue(node, watcher);\r\n }\r\n watchersToUpdateAndClear.clear();\r\n this._digestCount = 0;\r\n } else if (!boundingBoxesStable && this._digestCount > MAX_DIGESTS) {\r\n console.error('Bounding box digest cycle exceeded, but the meshes are not consistent');\r\n this._changedNodes.clear();\r\n watchersToUpdateAndClear.clear();\r\n this._digestCount = 0;\r\n }\r\n }\r\n\r\n /**\r\n * Gets the world bounding box for the specified node.\r\n * Be aware that this call is not cached, so use accordingly for performance reasons\r\n * @param node\r\n */\r\n public getWorld(node?: SpaceNode) {\r\n if (node && node.visible) {\r\n const bb = this.get(node)!;\r\n if (!bb) {\r\n return null;\r\n }\r\n const bnode = this.v.rendererManager.mesh.get3d(node);\r\n const bbInfo = new BoundingInfo(\r\n new Vector3(bb.min.x || 0, bb.min.y || 0, bb.min.z || 0),\r\n new Vector3(bb.max.x || 0, bb.max.y || 0, bb.max.z || 0),\r\n bnode?.getWorldMatrix()\r\n );\r\n return {\r\n boundingInfo: bbInfo,\r\n min: KbVector.FromVec3(bbInfo.boundingBox.minimumWorld),\r\n max: KbVector.FromVec3(bbInfo.boundingBox.maximumWorld),\r\n center: KbVector.FromVec3(bbInfo.boundingBox.centerWorld),\r\n } as BoundingBoxInfo;\r\n }\r\n return null;\r\n }\r\n\r\n public dispose() {\r\n this._db.clear();\r\n this._watchedNodes.forEach(subject => {\r\n subject.complete();\r\n });\r\n this._watchedNodes.clear();\r\n }\r\n}\r\n\r\nfunction boundingBoxesMatch(a: BoundingBoxInfo | undefined, b: BoundingBoxInfo | undefined) {\r\n if (!a && !b) {\r\n return true;\r\n }\r\n if (!a || !b) {\r\n return false;\r\n }\r\n\r\n return a.center.equal(b.center) && a.min.equal(b.min) && a.max.equal(b.max);\r\n}\r\n","import { Color3, Color4, Quaternion, Vector3 } from '@babylonjs/core/Maths';\r\nimport { KbColor } from '@models/classes/primitives/kb-color';\r\nimport { KbQuaternion } from '@models/classes/primitives/kb-quaternion';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\n\r\n/** KbVector extension methods - we have them here instead of in the class so that the class isn't referencing babylon */\r\ndeclare module '@models/classes/primitives/kb-vector' {\r\n interface KbVector {\r\n toVec3(ref?: Vector3): Vector3;\r\n }\r\n namespace KbVector {\r\n let FromVec3: (vector: Vector3) => KbVector;\r\n }\r\n}\r\nObject.defineProperty(KbVector.prototype, 'toVec3', {\r\n enumerable: false,\r\n writable: true,\r\n value: function (ref?: Vector3) {\r\n // add ref option\r\n if (!ref) {\r\n ref = new Vector3();\r\n }\r\n return this ? ref.copyFromFloats(this.x, this.y, this.z) : ref.setAll(0);\r\n },\r\n});\r\nObject.defineProperty(KbVector, 'FromVec3', {\r\n enumerable: false,\r\n writable: true,\r\n value: function (vector: Vector3) {\r\n return new KbVector(vector.x, vector.y, vector.z);\r\n },\r\n});\r\n\r\n/** KbQuaternion extension methods */\r\ndeclare module '@models/classes/primitives/kb-quaternion' {\r\n interface KbQuaternion {\r\n toQuat(): Quaternion;\r\n }\r\n namespace KbQuaternion {\r\n let FromQuat: (quaternion: Quaternion) => KbQuaternion;\r\n }\r\n}\r\nObject.defineProperty(KbQuaternion.prototype, 'toQuat', {\r\n enumerable: false,\r\n writable: true,\r\n value: function () {\r\n return this ? new Quaternion(this.x, this.y, this.z, this.w) : new Quaternion(0, 0, 0, 1);\r\n },\r\n});\r\nObject.defineProperty(KbQuaternion, 'FromQuat', {\r\n enumerable: false,\r\n writable: true,\r\n value: function (q: Quaternion) {\r\n return new KbQuaternion(q.x, q.y, q.z, q.w);\r\n },\r\n});\r\n\r\n/** KbColor extension methods */\r\ndeclare module '@models/classes/primitives/kb-color' {\r\n interface KbColor {\r\n toColor3(): Color3;\r\n toColor4(): Color4;\r\n }\r\n}\r\nObject.defineProperty(KbColor.prototype, 'toColor3', {\r\n enumerable: false,\r\n writable: true,\r\n value: function () {\r\n return this ? new Color3(this.r, this.g, this.b) : Color3.Black();\r\n },\r\n});\r\nObject.defineProperty(KbColor.prototype, 'toColor4', {\r\n enumerable: false,\r\n writable: true,\r\n value: function () {\r\n return this ? new Color4(this.r, this.g, this.b, this.a) : new Color4(0, 0, 0, 0);\r\n },\r\n});\r\n","import { Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { BaseConnector, isSketchControlPoint } from '@models/classes';\r\nimport { IKbVectorLike, KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { IBoundingBox } from '@models/classes/primitives/primitive-utilities';\r\n\r\nexport class BoundingBoxRelativeHelper {\r\n static convertRelativeToAbsolute(c: BaseConnector, bb: IBoundingBox, position?: IKbVectorLike) {\r\n if (!position) {\r\n position = c.position;\r\n }\r\n if (isSketchControlPoint(c)) {\r\n return new Vector3(\r\n ((position.x + 1) / 2) * bb.max.x + bb.min.x,\r\n ((position.y + 1) / 2) * bb.max.y + bb.min.y,\r\n ((position.z + 1) / 2) * bb.max.z + bb.min.z\r\n );\r\n } else {\r\n return new Vector3(\r\n bb.center.x + (position.x * Math.abs(bb.max.x - bb.min.x)) / 2,\r\n bb.center.y + (position.y * Math.abs(bb.max.y - bb.min.y)) / 2,\r\n bb.center.z + (position.z * Math.abs(bb.max.z - bb.min.z)) / 2\r\n );\r\n }\r\n }\r\n\r\n static convertAbsoluteToRelative(c: BaseConnector, bb: IBoundingBox, position?: IKbVectorLike) {\r\n if (!position) {\r\n position = c.position;\r\n }\r\n if (isSketchControlPoint(c)) {\r\n return new Vector3(\r\n ((position.x - bb.min.x) / bb.max.x) * 2 - 1,\r\n ((position.y - bb.min.y) / bb.max.y) * 2 - 1,\r\n ((position.z - bb.min.z) / bb.max.z) * 2 - 1\r\n );\r\n } else {\r\n return new Vector3(\r\n ((position.x - bb.center.x) * 2) / Math.abs(bb.max.x - bb.min.x),\r\n ((position.y - bb.center.y) * 2) / Math.abs(bb.max.y - bb.min.y),\r\n ((position.z - bb.center.z) * 2) / Math.abs(bb.max.z - bb.min.z)\r\n );\r\n }\r\n }\r\n\r\n static convertRelativeDirectionToAbsolute(c: BaseConnector, direction: IKbVectorLike, bb: IBoundingBox) {\r\n if (isSketchControlPoint(c)) {\r\n return new Vector3(\r\n direction.x * (bb.min.x - bb.max.x),\r\n direction.y * (bb.min.y - bb.max.y),\r\n direction.z * (bb.min.z - bb.max.z)\r\n );\r\n } else {\r\n return new Vector3(direction.x, direction.y, direction.z);\r\n }\r\n }\r\n static convertAbsoluteDirectionToRelative(c: BaseConnector, direction: IKbVectorLike, bb: IBoundingBox) {\r\n if (isSketchControlPoint(c)) {\r\n return new Vector3(\r\n direction.x / (bb.min.x - bb.max.x),\r\n direction.y / (bb.min.y - bb.max.y),\r\n direction.z / (bb.min.z - bb.max.z)\r\n );\r\n } else {\r\n return new Vector3(direction.x, direction.y, direction.z);\r\n }\r\n }\r\n}\r\n\r\nexport class BoundingBoxRelativeHelperKbVector {\r\n static convertRelativeToAbsolute(c: BaseConnector, bb: IBoundingBox, position?: IKbVectorLike) {\r\n return KbVector.FromVec3(BoundingBoxRelativeHelper.convertRelativeToAbsolute(c, bb, position));\r\n }\r\n\r\n static convertAbsoluteToRelative(c: BaseConnector, bb: IBoundingBox, position?: IKbVectorLike) {\r\n return KbVector.FromVec3(BoundingBoxRelativeHelper.convertAbsoluteToRelative(c, bb, position));\r\n }\r\n\r\n static convertRelativeDirectionToAbsolute(c: BaseConnector, direction: IKbVectorLike, bb: IBoundingBox) {\r\n return KbVector.FromVec3(BoundingBoxRelativeHelper.convertRelativeDirectionToAbsolute(c, direction, bb));\r\n }\r\n static convertAbsoluteDirectionToRelative(c: BaseConnector, direction: IKbVectorLike, bb: IBoundingBox) {\r\n return KbVector.FromVec3(BoundingBoxRelativeHelper.convertAbsoluteDirectionToRelative(c, direction, bb));\r\n }\r\n}\r\n","export enum eSideOrientationBabylonMap {\r\n frontSide = 0,\r\n backSide = 1,\r\n doubleSide = 2,\r\n}\r\nexport enum eUvMapAxisBabylonMap {\r\n x = 0,\r\n y = 1,\r\n z = 2,\r\n}\r\nexport enum eUvModeBabylonMap {\r\n planar = 0,\r\n box = 1,\r\n cylindrical = 2,\r\n spherical = 3,\r\n}\r\nexport enum eAlphaModeBabylonMap {\r\n disable = 0,\r\n add = 1,\r\n combine = 2,\r\n subtract = 3,\r\n multiply = 4,\r\n}\r\n","import { DirectionalLight, HemisphericLight, Light, SpotLight } from '@babylonjs/core/Lights';\r\nimport { StandardMaterial } from '@babylonjs/core/Materials/standardMaterial';\r\nimport { Color3 } from '@babylonjs/core/Maths/math';\r\nimport { CylinderBuilder } from '@babylonjs/core/Meshes/Builders/cylinderBuilder';\r\nimport { GroundBuilder } from '@babylonjs/core/Meshes/Builders/groundBuilder';\r\nimport { SphereBuilder } from '@babylonjs/core/Meshes/Builders/sphereBuilder';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { Scene } from '@babylonjs/core/scene';\r\n\r\nexport function createLightVisualization(light: Light, scene: Scene, color?: Color3) {\r\n let lightPositionGizmo: Mesh;\r\n\r\n if (!color) {\r\n color = Color3.Yellow();\r\n }\r\n\r\n if (light instanceof HemisphericLight) {\r\n lightPositionGizmo = SphereBuilder.CreateSphere(\r\n 'lightPositionMarker',\r\n { diameter: 0.75, slice: 0.5, segments: 6 },\r\n scene\r\n );\r\n const coloredMaterial = new StandardMaterial('', scene);\r\n coloredMaterial.wireframe = true;\r\n coloredMaterial.emissiveColor = color;\r\n lightPositionGizmo.material = coloredMaterial;\r\n } else if (light instanceof SpotLight) {\r\n lightPositionGizmo = CylinderBuilder.CreateCylinder(\r\n 'lightPositionCone',\r\n { diameterTop: 0.5, height: 0.5, diameterBottom: 0, tessellation: 12 },\r\n scene\r\n );\r\n lightPositionGizmo.position.y = 0.25;\r\n lightPositionGizmo.bakeCurrentTransformIntoVertices();\r\n\r\n const coloredMaterial = new StandardMaterial('', scene);\r\n coloredMaterial.wireframe = true;\r\n coloredMaterial.emissiveColor = color;\r\n lightPositionGizmo.material = coloredMaterial;\r\n } else if (light instanceof DirectionalLight) {\r\n const gizmoOffset = 1;\r\n lightPositionGizmo = GroundBuilder.CreateGround(\r\n 'lightPositionPlane',\r\n {\r\n height: 1.5,\r\n width: 1.5,\r\n subdivisions: 1,\r\n },\r\n scene\r\n );\r\n const cone = CylinderBuilder.CreateCylinder(\r\n 'cylinder',\r\n { diameterTop: 0, height: 0.075, diameterBottom: 0.0375, tessellation: 96 },\r\n scene\r\n );\r\n const line = CylinderBuilder.CreateCylinder(\r\n 'cylinder',\r\n { diameterTop: 0.005, height: 0.5, diameterBottom: 0.005, tessellation: 96 },\r\n scene\r\n );\r\n\r\n const coloredMaterial = new StandardMaterial('', scene);\r\n coloredMaterial.wireframe = true;\r\n coloredMaterial.emissiveColor = color;\r\n\r\n lightPositionGizmo.material = coloredMaterial;\r\n cone.material = coloredMaterial;\r\n line.material = coloredMaterial;\r\n //cone.rotation.x = Math.PI / 2;\r\n lightPositionGizmo.position.y = -gizmoOffset;\r\n lightPositionGizmo.bakeCurrentTransformIntoVertices();\r\n\r\n cone.parent = lightPositionGizmo;\r\n line.parent = lightPositionGizmo;\r\n cone.position.y = 0.275 - gizmoOffset;\r\n line.position.y = -gizmoOffset;\r\n //line.rotation.x = Math.PI / 2;\r\n } else {\r\n // PointLight\r\n lightPositionGizmo = SphereBuilder.CreateSphere('lightPositionMarker', { diameter: 0.1 }, scene);\r\n const coloredMaterial = new StandardMaterial('', scene);\r\n coloredMaterial.emissiveColor = color;\r\n lightPositionGizmo.material = coloredMaterial;\r\n }\r\n\r\n lightPositionGizmo.isPickable = false;\r\n lightPositionGizmo.setEnabled(true);\r\n return lightPositionGizmo;\r\n}\r\n","export class Quad {\r\n constructor(public capacity = 4) {}\r\n\r\n public points = new Array();\r\n}\r\n\r\nexport interface IPoint {\r\n x: number;\r\n y: number;\r\n}\r\n\r\nexport interface IBoundary {\r\n xmin: number;\r\n xmax: number;\r\n ymin: number;\r\n ymax: number;\r\n}\r\n\r\nexport class Boundary implements IBoundary {\r\n constructor(public xmin: number, public xmax: number, public ymin: number, public ymax: number) {}\r\n\r\n boundaryContains(point: IPoint) {\r\n return this.xmin <= point.x && this.xmax >= point.x && this.ymin <= point.y && this.ymax >= point.y;\r\n }\r\n\r\n intersects(b: Boundary) {\r\n if (this.xmin >= b.xmax || b.xmin >= this.xmax || this.ymin >= b.ymax || b.ymin >= this.ymax) {\r\n return false;\r\n }\r\n\r\n return true;\r\n }\r\n\r\n get halfPoint() {\r\n return {\r\n x: this.xmin + this.width / 2,\r\n y: this.ymin + this.height / 2,\r\n } as IPoint;\r\n }\r\n\r\n get width() {\r\n return this.xmax - this.xmin;\r\n }\r\n get height() {\r\n return this.ymax - this.ymin;\r\n }\r\n}\r\n\r\nexport interface IQuadTreeOptions extends IBoundary {\r\n items: T[];\r\n xSelector: (item: T) => number;\r\n ySelector: (item: T) => number;\r\n capacity: number;\r\n}\r\n\r\ninterface IProximityItem {\r\n item: T;\r\n proximity: number;\r\n}\r\n\r\nexport class QuadTree extends Boundary {\r\n constructor(public o: IQuadTreeOptions) {\r\n super(o.xmin, o.xmax, o.ymin, o.ymax);\r\n\r\n this.xSelector = o.xSelector;\r\n this.ySelector = o.ySelector;\r\n this.capacity = o.capacity;\r\n\r\n if (o.items) {\r\n for (const item of o.items) this.add(item);\r\n }\r\n }\r\n\r\n public items = new Array();\r\n private size = 0;\r\n private duplicatesMap: Record = {};\r\n public xSelector: (item: T) => number;\r\n public ySelector: (item: T) => number;\r\n public capacity: number;\r\n\r\n public nw: QuadTree;\r\n public ne: QuadTree;\r\n public sw: QuadTree;\r\n public se: QuadTree;\r\n\r\n private _divided = false;\r\n public get divided() {\r\n return this._divided;\r\n }\r\n\r\n public get quads() {\r\n const arr = Array>();\r\n if (this.nw) arr.push(this.nw);\r\n if (this.ne) arr.push(this.ne);\r\n if (this.sw) arr.push(this.sw);\r\n if (this.se) arr.push(this.se);\r\n return arr;\r\n }\r\n\r\n public find(point: IPoint, radius: number): Array> {\r\n const b = new Boundary(point.x - radius, point.x + radius, point.y - radius, point.y + radius);\r\n const results = this.findByBoundary(b);\r\n\r\n //sort by proximity\r\n results.sortBy(r => r.proximity);\r\n return results;\r\n }\r\n\r\n public findByBoundary(boundary: Boundary): Array> {\r\n const results = new Array>();\r\n\r\n if (!this.intersects(boundary)) {\r\n return results;\r\n }\r\n\r\n if (this.divided) {\r\n for (const q of this.quads) {\r\n results.push(...q.findByBoundary(boundary));\r\n }\r\n } else {\r\n //check items at this level\r\n for (const i of this.items) {\r\n if (boundary.boundaryContains(this.itemPoint(i))) {\r\n results.push(this.getWithProximity(i, boundary.halfPoint));\r\n }\r\n }\r\n }\r\n\r\n return results;\r\n }\r\n\r\n protected getWithProximity(item: T, searchPoint: IPoint): IProximityItem {\r\n const itemPoint = this.itemPoint(item);\r\n return {\r\n item: item,\r\n proximity: Math.sqrt(Math.pow(searchPoint.x - itemPoint.x, 2) + Math.pow(searchPoint.y - itemPoint.y, 2)),\r\n };\r\n }\r\n\r\n protected positionKey(point: { x: number; y: number }) {\r\n return point.x + ',' + point.y;\r\n }\r\n\r\n public add(item: T): boolean {\r\n const point = this.itemPoint(item);\r\n const key = this.positionKey(point);\r\n if (!this.boundaryContains(point)) {\r\n return false;\r\n }\r\n // If there is another point in this quad with the **exact** same position, add it to this quad as well.\r\n // This may result in overfilling the quad but there's no other way to handle exact duplicate points\r\n if (this.duplicatesMap[key]) {\r\n this.items.push(item);\r\n return true;\r\n }\r\n\r\n //if there is space for it, and no sub-divisions, add it to this quad\r\n if (this.size < this.o.capacity && !this.divided) {\r\n this.items.push(item);\r\n this.duplicatesMap[key] = item;\r\n this.size++;\r\n return true;\r\n }\r\n\r\n if (!this.divided) {\r\n this.subdivide();\r\n }\r\n\r\n for (const q of this.quads) {\r\n if (q.add(item)) return true;\r\n }\r\n\r\n throw Error('unexpected error creating quadtree');\r\n }\r\n\r\n protected subdivide() {\r\n // No need to check y, if x is infinitesimally small then y is as well\r\n if (this.xmax - this.xmin <= Number.EPSILON * 2) {\r\n throw Error('unexpected error creating quadtree. Not enough granularity');\r\n }\r\n\r\n this.nw = this.createDivision({\r\n items: this.items,\r\n xmin: this.xmin,\r\n xmax: this.halfPoint.x,\r\n ymin: this.ymin,\r\n ymax: this.halfPoint.y,\r\n });\r\n this.ne = this.createDivision({\r\n items: this.items,\r\n xmin: this.halfPoint.x + Number.EPSILON,\r\n xmax: this.xmax,\r\n ymin: this.ymin,\r\n ymax: this.halfPoint.y,\r\n });\r\n this.sw = this.createDivision({\r\n items: this.items,\r\n xmin: this.xmin,\r\n xmax: this.halfPoint.x,\r\n ymin: this.halfPoint.y + Number.EPSILON,\r\n ymax: this.ymax,\r\n });\r\n this.se = this.createDivision({\r\n items: this.items,\r\n xmin: this.halfPoint.x + Number.EPSILON,\r\n xmax: this.xmax,\r\n ymin: this.halfPoint.y + Number.EPSILON,\r\n ymax: this.ymax,\r\n });\r\n\r\n this.items.clear();\r\n this.size = 0;\r\n this._divided = true;\r\n }\r\n\r\n protected itemPoint(item: T): IPoint {\r\n return {\r\n x: this.xSelector(item),\r\n y: this.ySelector(item),\r\n };\r\n }\r\n\r\n protected createDivision(options: Partial>) {\r\n return new QuadTree({\r\n ...this.o,\r\n ...options,\r\n });\r\n }\r\n}\r\n","// @ts-nocheck third party library\r\n/* eslint-disable */\r\n\r\nexport function earcut(data: number[] | Float32Array, holeIndices: number[], dim?: number): number[] {\r\n dim = dim || 2;\r\n\r\n var hasHoles = holeIndices && holeIndices.length,\r\n outerLen = hasHoles ? holeIndices[0] * dim : data.length,\r\n outerNode = linkedList(data, 0, outerLen, dim, true),\r\n triangles = [];\r\n\r\n if (!outerNode || outerNode.next === outerNode.prev) return triangles;\r\n\r\n var minX, minY, maxX, maxY, x, y, invSize;\r\n\r\n if (hasHoles) outerNode = eliminateHoles(data, holeIndices, outerNode, dim);\r\n\r\n // if the shape is not too simple, we'll use z-order curve hash later; calculate polygon bbox\r\n if (data.length > 80 * dim) {\r\n minX = maxX = data[0];\r\n minY = maxY = data[1];\r\n\r\n for (var i = dim; i < outerLen; i += dim) {\r\n x = data[i];\r\n y = data[i + 1];\r\n if (x < minX) minX = x;\r\n if (y < minY) minY = y;\r\n if (x > maxX) maxX = x;\r\n if (y > maxY) maxY = y;\r\n }\r\n\r\n // minX, minY and invSize are later used to transform coords into integers for z-order calculation\r\n invSize = Math.max(maxX - minX, maxY - minY);\r\n invSize = invSize !== 0 ? 32767 / invSize : 0;\r\n }\r\n\r\n earcutLinked(outerNode, triangles, dim, minX, minY, invSize, 0);\r\n\r\n return triangles;\r\n}\r\n\r\n// create a circular doubly linked list from polygon points in the specified winding order\r\nfunction linkedList(data, start, end, dim, clockwise) {\r\n var i, last;\r\n\r\n if (clockwise === signedArea(data, start, end, dim) > 0) {\r\n for (i = start; i < end; i += dim) last = insertNode(i, data[i], data[i + 1], last);\r\n } else {\r\n for (i = end - dim; i >= start; i -= dim) last = insertNode(i, data[i], data[i + 1], last);\r\n }\r\n\r\n if (last && equals(last, last.next)) {\r\n removeNode(last);\r\n last = last.next;\r\n }\r\n\r\n return last;\r\n}\r\n\r\n// eliminate colinear or duplicate points\r\nfunction filterPoints(start, end) {\r\n if (!start) return start;\r\n if (!end) end = start;\r\n\r\n var p = start,\r\n again;\r\n do {\r\n again = false;\r\n\r\n if (!p.steiner && (equals(p, p.next) || area(p.prev, p, p.next) === 0)) {\r\n removeNode(p);\r\n p = end = p.prev;\r\n if (p === p.next) break;\r\n again = true;\r\n } else {\r\n p = p.next;\r\n }\r\n } while (again || p !== end);\r\n\r\n return end;\r\n}\r\n\r\n// main ear slicing loop which triangulates a polygon (given as a linked list)\r\nfunction earcutLinked(ear, triangles, dim, minX, minY, invSize, pass) {\r\n if (!ear) return;\r\n\r\n // interlink polygon nodes in z-order\r\n if (!pass && invSize) indexCurve(ear, minX, minY, invSize);\r\n\r\n var stop = ear,\r\n prev,\r\n next;\r\n\r\n // iterate through ears, slicing them one by one\r\n while (ear.prev !== ear.next) {\r\n prev = ear.prev;\r\n next = ear.next;\r\n\r\n if (invSize ? isEarHashed(ear, minX, minY, invSize) : isEar(ear)) {\r\n // cut off the triangle\r\n triangles.push((prev.i / dim) | 0);\r\n triangles.push((ear.i / dim) | 0);\r\n triangles.push((next.i / dim) | 0);\r\n\r\n removeNode(ear);\r\n\r\n // skipping the next vertex leads to less sliver triangles\r\n ear = next.next;\r\n stop = next.next;\r\n\r\n continue;\r\n }\r\n\r\n ear = next;\r\n\r\n // if we looped through the whole remaining polygon and can't find any more ears\r\n if (ear === stop) {\r\n // try filtering points and slicing again\r\n if (!pass) {\r\n earcutLinked(filterPoints(ear), triangles, dim, minX, minY, invSize, 1);\r\n\r\n // if this didn't work, try curing all small self-intersections locally\r\n } else if (pass === 1) {\r\n ear = cureLocalIntersections(filterPoints(ear), triangles, dim);\r\n earcutLinked(ear, triangles, dim, minX, minY, invSize, 2);\r\n\r\n // as a last resort, try splitting the remaining polygon into two\r\n } else if (pass === 2) {\r\n splitEarcut(ear, triangles, dim, minX, minY, invSize);\r\n }\r\n\r\n break;\r\n }\r\n }\r\n}\r\n\r\n// check whether a polygon node forms a valid ear with adjacent nodes\r\nfunction isEar(ear) {\r\n var a = ear.prev,\r\n b = ear,\r\n c = ear.next;\r\n\r\n if (area(a, b, c) >= 0) return false; // reflex, can't be an ear\r\n\r\n // now make sure we don't have other points inside the potential ear\r\n var ax = a.x,\r\n bx = b.x,\r\n cx = c.x,\r\n ay = a.y,\r\n by = b.y,\r\n cy = c.y;\r\n\r\n // triangle bbox; min & max are calculated like this for speed\r\n var x0 = ax < bx ? (ax < cx ? ax : cx) : bx < cx ? bx : cx,\r\n y0 = ay < by ? (ay < cy ? ay : cy) : by < cy ? by : cy,\r\n x1 = ax > bx ? (ax > cx ? ax : cx) : bx > cx ? bx : cx,\r\n y1 = ay > by ? (ay > cy ? ay : cy) : by > cy ? by : cy;\r\n\r\n var p = c.next;\r\n while (p !== a) {\r\n if (\r\n p.x >= x0 &&\r\n p.x <= x1 &&\r\n p.y >= y0 &&\r\n p.y <= y1 &&\r\n pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) &&\r\n area(p.prev, p, p.next) >= 0\r\n )\r\n return false;\r\n p = p.next;\r\n }\r\n\r\n return true;\r\n}\r\n\r\nfunction isEarHashed(ear, minX, minY, invSize) {\r\n var a = ear.prev,\r\n b = ear,\r\n c = ear.next;\r\n\r\n if (area(a, b, c) >= 0) return false; // reflex, can't be an ear\r\n\r\n var ax = a.x,\r\n bx = b.x,\r\n cx = c.x,\r\n ay = a.y,\r\n by = b.y,\r\n cy = c.y;\r\n\r\n // triangle bbox; min & max are calculated like this for speed\r\n var x0 = ax < bx ? (ax < cx ? ax : cx) : bx < cx ? bx : cx,\r\n y0 = ay < by ? (ay < cy ? ay : cy) : by < cy ? by : cy,\r\n x1 = ax > bx ? (ax > cx ? ax : cx) : bx > cx ? bx : cx,\r\n y1 = ay > by ? (ay > cy ? ay : cy) : by > cy ? by : cy;\r\n\r\n // z-order range for the current triangle bbox;\r\n var minZ = zOrder(x0, y0, minX, minY, invSize),\r\n maxZ = zOrder(x1, y1, minX, minY, invSize);\r\n\r\n var p = ear.prevZ,\r\n n = ear.nextZ;\r\n\r\n // look for points inside the triangle in both directions\r\n while (p && p.z >= minZ && n && n.z <= maxZ) {\r\n if (\r\n p.x >= x0 &&\r\n p.x <= x1 &&\r\n p.y >= y0 &&\r\n p.y <= y1 &&\r\n p !== a &&\r\n p !== c &&\r\n pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) &&\r\n area(p.prev, p, p.next) >= 0\r\n )\r\n return false;\r\n p = p.prevZ;\r\n\r\n if (\r\n n.x >= x0 &&\r\n n.x <= x1 &&\r\n n.y >= y0 &&\r\n n.y <= y1 &&\r\n n !== a &&\r\n n !== c &&\r\n pointInTriangle(ax, ay, bx, by, cx, cy, n.x, n.y) &&\r\n area(n.prev, n, n.next) >= 0\r\n )\r\n return false;\r\n n = n.nextZ;\r\n }\r\n\r\n // look for remaining points in decreasing z-order\r\n while (p && p.z >= minZ) {\r\n if (\r\n p.x >= x0 &&\r\n p.x <= x1 &&\r\n p.y >= y0 &&\r\n p.y <= y1 &&\r\n p !== a &&\r\n p !== c &&\r\n pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) &&\r\n area(p.prev, p, p.next) >= 0\r\n )\r\n return false;\r\n p = p.prevZ;\r\n }\r\n\r\n // look for remaining points in increasing z-order\r\n while (n && n.z <= maxZ) {\r\n if (\r\n n.x >= x0 &&\r\n n.x <= x1 &&\r\n n.y >= y0 &&\r\n n.y <= y1 &&\r\n n !== a &&\r\n n !== c &&\r\n pointInTriangle(ax, ay, bx, by, cx, cy, n.x, n.y) &&\r\n area(n.prev, n, n.next) >= 0\r\n )\r\n return false;\r\n n = n.nextZ;\r\n }\r\n\r\n return true;\r\n}\r\n\r\n// go through all polygon nodes and cure small local self-intersections\r\nfunction cureLocalIntersections(start, triangles, dim) {\r\n var p = start;\r\n do {\r\n var a = p.prev,\r\n b = p.next.next;\r\n\r\n if (!equals(a, b) && intersects(a, p, p.next, b) && locallyInside(a, b) && locallyInside(b, a)) {\r\n triangles.push((a.i / dim) | 0);\r\n triangles.push((p.i / dim) | 0);\r\n triangles.push((b.i / dim) | 0);\r\n\r\n // remove two nodes involved\r\n removeNode(p);\r\n removeNode(p.next);\r\n\r\n p = start = b;\r\n }\r\n p = p.next;\r\n } while (p !== start);\r\n\r\n return filterPoints(p);\r\n}\r\n\r\n// try splitting polygon into two and triangulate them independently\r\nfunction splitEarcut(start, triangles, dim, minX, minY, invSize) {\r\n // look for a valid diagonal that divides the polygon into two\r\n var a = start;\r\n do {\r\n var b = a.next.next;\r\n while (b !== a.prev) {\r\n if (a.i !== b.i && isValidDiagonal(a, b)) {\r\n // split the polygon in two by the diagonal\r\n var c = splitPolygon(a, b);\r\n\r\n // filter colinear points around the cuts\r\n a = filterPoints(a, a.next);\r\n c = filterPoints(c, c.next);\r\n\r\n // run earcut on each half\r\n earcutLinked(a, triangles, dim, minX, minY, invSize, 0);\r\n earcutLinked(c, triangles, dim, minX, minY, invSize, 0);\r\n return;\r\n }\r\n b = b.next;\r\n }\r\n a = a.next;\r\n } while (a !== start);\r\n}\r\n\r\n// link every hole into the outer loop, producing a single-ring polygon without holes\r\nfunction eliminateHoles(data, holeIndices, outerNode, dim) {\r\n var queue = [],\r\n i,\r\n len,\r\n start,\r\n end,\r\n list;\r\n\r\n for (i = 0, len = holeIndices.length; i < len; i++) {\r\n start = holeIndices[i] * dim;\r\n end = i < len - 1 ? holeIndices[i + 1] * dim : data.length;\r\n list = linkedList(data, start, end, dim, false);\r\n if (list === list.next) list.steiner = true;\r\n queue.push(getLeftmost(list));\r\n }\r\n\r\n queue.sort(compareX);\r\n\r\n // process holes from left to right\r\n for (i = 0; i < queue.length; i++) {\r\n outerNode = eliminateHole(queue[i], outerNode);\r\n }\r\n\r\n return outerNode;\r\n}\r\n\r\nfunction compareX(a, b) {\r\n return a.x - b.x;\r\n}\r\n\r\n// find a bridge between vertices that connects hole with an outer ring and and link it\r\nfunction eliminateHole(hole, outerNode) {\r\n var bridge = findHoleBridge(hole, outerNode);\r\n if (!bridge) {\r\n return outerNode;\r\n }\r\n\r\n var bridgeReverse = splitPolygon(bridge, hole);\r\n\r\n // filter collinear points around the cuts\r\n filterPoints(bridgeReverse, bridgeReverse.next);\r\n return filterPoints(bridge, bridge.next);\r\n}\r\n\r\n// David Eberly's algorithm for finding a bridge between hole and outer polygon\r\nfunction findHoleBridge(hole, outerNode) {\r\n var p = outerNode,\r\n hx = hole.x,\r\n hy = hole.y,\r\n qx = -Infinity,\r\n m;\r\n\r\n // find a segment intersected by a ray from the hole's leftmost point to the left;\r\n // segment's endpoint with lesser x will be potential connection point\r\n do {\r\n if (hy <= p.y && hy >= p.next.y && p.next.y !== p.y) {\r\n var x = p.x + ((hy - p.y) * (p.next.x - p.x)) / (p.next.y - p.y);\r\n if (x <= hx && x > qx) {\r\n qx = x;\r\n m = p.x < p.next.x ? p : p.next;\r\n if (x === hx) return m; // hole touches outer segment; pick leftmost endpoint\r\n }\r\n }\r\n p = p.next;\r\n } while (p !== outerNode);\r\n\r\n if (!m) return null;\r\n\r\n // look for points inside the triangle of hole point, segment intersection and endpoint;\r\n // if there are no points found, we have a valid connection;\r\n // otherwise choose the point of the minimum angle with the ray as connection point\r\n\r\n var stop = m,\r\n mx = m.x,\r\n my = m.y,\r\n tanMin = Infinity,\r\n tan;\r\n\r\n p = m;\r\n\r\n do {\r\n if (\r\n hx >= p.x &&\r\n p.x >= mx &&\r\n hx !== p.x &&\r\n pointInTriangle(hy < my ? hx : qx, hy, mx, my, hy < my ? qx : hx, hy, p.x, p.y)\r\n ) {\r\n tan = Math.abs(hy - p.y) / (hx - p.x); // tangential\r\n\r\n if (\r\n locallyInside(p, hole) &&\r\n (tan < tanMin || (tan === tanMin && (p.x > m.x || (p.x === m.x && sectorContainsSector(m, p)))))\r\n ) {\r\n m = p;\r\n tanMin = tan;\r\n }\r\n }\r\n\r\n p = p.next;\r\n } while (p !== stop);\r\n\r\n return m;\r\n}\r\n\r\n// whether sector in vertex m contains sector in vertex p in the same coordinates\r\nfunction sectorContainsSector(m, p) {\r\n return area(m.prev, m, p.prev) < 0 && area(p.next, m, m.next) < 0;\r\n}\r\n\r\n// interlink polygon nodes in z-order\r\nfunction indexCurve(start, minX, minY, invSize) {\r\n var p = start;\r\n do {\r\n if (p.z === 0) p.z = zOrder(p.x, p.y, minX, minY, invSize);\r\n p.prevZ = p.prev;\r\n p.nextZ = p.next;\r\n p = p.next;\r\n } while (p !== start);\r\n\r\n p.prevZ.nextZ = null;\r\n p.prevZ = null;\r\n\r\n sortLinked(p);\r\n}\r\n\r\n// Simon Tatham's linked list merge sort algorithm\r\n// http://www.chiark.greenend.org.uk/~sgtatham/algorithms/listsort.html\r\nfunction sortLinked(list) {\r\n var i,\r\n p,\r\n q,\r\n e,\r\n tail,\r\n numMerges,\r\n pSize,\r\n qSize,\r\n inSize = 1;\r\n\r\n do {\r\n p = list;\r\n list = null;\r\n tail = null;\r\n numMerges = 0;\r\n\r\n while (p) {\r\n numMerges++;\r\n q = p;\r\n pSize = 0;\r\n for (i = 0; i < inSize; i++) {\r\n pSize++;\r\n q = q.nextZ;\r\n if (!q) break;\r\n }\r\n qSize = inSize;\r\n\r\n while (pSize > 0 || (qSize > 0 && q)) {\r\n if (pSize !== 0 && (qSize === 0 || !q || p.z <= q.z)) {\r\n e = p;\r\n p = p.nextZ;\r\n pSize--;\r\n } else {\r\n e = q;\r\n q = q.nextZ;\r\n qSize--;\r\n }\r\n\r\n if (tail) tail.nextZ = e;\r\n else list = e;\r\n\r\n e.prevZ = tail;\r\n tail = e;\r\n }\r\n\r\n p = q;\r\n }\r\n\r\n tail.nextZ = null;\r\n inSize *= 2;\r\n } while (numMerges > 1);\r\n\r\n return list;\r\n}\r\n\r\n// z-order of a point given coords and inverse of the longer side of data bbox\r\nfunction zOrder(x, y, minX, minY, invSize) {\r\n // coords are transformed into non-negative 15-bit integer range\r\n x = ((x - minX) * invSize) | 0;\r\n y = ((y - minY) * invSize) | 0;\r\n\r\n x = (x | (x << 8)) & 0x00ff00ff;\r\n x = (x | (x << 4)) & 0x0f0f0f0f;\r\n x = (x | (x << 2)) & 0x33333333;\r\n x = (x | (x << 1)) & 0x55555555;\r\n\r\n y = (y | (y << 8)) & 0x00ff00ff;\r\n y = (y | (y << 4)) & 0x0f0f0f0f;\r\n y = (y | (y << 2)) & 0x33333333;\r\n y = (y | (y << 1)) & 0x55555555;\r\n\r\n return x | (y << 1);\r\n}\r\n\r\n// find the leftmost node of a polygon ring\r\nfunction getLeftmost(start) {\r\n var p = start,\r\n leftmost = start;\r\n do {\r\n if (p.x < leftmost.x || (p.x === leftmost.x && p.y < leftmost.y)) leftmost = p;\r\n p = p.next;\r\n } while (p !== start);\r\n\r\n return leftmost;\r\n}\r\n\r\n// check if a point lies within a convex triangle\r\nfunction pointInTriangle(ax, ay, bx, by, cx, cy, px, py) {\r\n return (\r\n (cx - px) * (ay - py) >= (ax - px) * (cy - py) &&\r\n (ax - px) * (by - py) >= (bx - px) * (ay - py) &&\r\n (bx - px) * (cy - py) >= (cx - px) * (by - py)\r\n );\r\n}\r\n\r\n// check if a diagonal between two polygon nodes is valid (lies in polygon interior)\r\nfunction isValidDiagonal(a, b) {\r\n return (\r\n a.next.i !== b.i &&\r\n a.prev.i !== b.i &&\r\n !intersectsPolygon(a, b) && // dones't intersect other edges\r\n ((locallyInside(a, b) &&\r\n locallyInside(b, a) &&\r\n middleInside(a, b) && // locally visible\r\n (area(a.prev, a, b.prev) || area(a, b.prev, b))) || // does not create opposite-facing sectors\r\n (equals(a, b) && area(a.prev, a, a.next) > 0 && area(b.prev, b, b.next) > 0))\r\n ); // special zero-length case\r\n}\r\n\r\n// signed area of a triangle\r\nfunction area(p, q, r) {\r\n return (q.y - p.y) * (r.x - q.x) - (q.x - p.x) * (r.y - q.y);\r\n}\r\n\r\n// check if two points are equal\r\nfunction equals(p1, p2) {\r\n return p1.x === p2.x && p1.y === p2.y;\r\n}\r\n\r\n// check if two segments intersect\r\nfunction intersects(p1, q1, p2, q2) {\r\n var o1 = sign(area(p1, q1, p2));\r\n var o2 = sign(area(p1, q1, q2));\r\n var o3 = sign(area(p2, q2, p1));\r\n var o4 = sign(area(p2, q2, q1));\r\n\r\n if (o1 !== o2 && o3 !== o4) return true; // general case\r\n\r\n if (o1 === 0 && onSegment(p1, p2, q1)) return true; // p1, q1 and p2 are collinear and p2 lies on p1q1\r\n if (o2 === 0 && onSegment(p1, q2, q1)) return true; // p1, q1 and q2 are collinear and q2 lies on p1q1\r\n if (o3 === 0 && onSegment(p2, p1, q2)) return true; // p2, q2 and p1 are collinear and p1 lies on p2q2\r\n if (o4 === 0 && onSegment(p2, q1, q2)) return true; // p2, q2 and q1 are collinear and q1 lies on p2q2\r\n\r\n return false;\r\n}\r\n\r\n// for collinear points p, q, r, check if point q lies on segment pr\r\nfunction onSegment(p, q, r) {\r\n return (\r\n q.x <= Math.max(p.x, r.x) && q.x >= Math.min(p.x, r.x) && q.y <= Math.max(p.y, r.y) && q.y >= Math.min(p.y, r.y)\r\n );\r\n}\r\n\r\nfunction sign(num) {\r\n return num > 0 ? 1 : num < 0 ? -1 : 0;\r\n}\r\n\r\n// check if a polygon diagonal intersects any polygon segments\r\nfunction intersectsPolygon(a, b) {\r\n var p = a;\r\n do {\r\n if (p.i !== a.i && p.next.i !== a.i && p.i !== b.i && p.next.i !== b.i && intersects(p, p.next, a, b))\r\n return true;\r\n p = p.next;\r\n } while (p !== a);\r\n\r\n return false;\r\n}\r\n\r\n// check if a polygon diagonal is locally inside the polygon\r\nfunction locallyInside(a, b) {\r\n return area(a.prev, a, a.next) < 0\r\n ? area(a, b, a.next) >= 0 && area(a, a.prev, b) >= 0\r\n : area(a, b, a.prev) < 0 || area(a, a.next, b) < 0;\r\n}\r\n\r\n// check if the middle point of a polygon diagonal is inside the polygon\r\nfunction middleInside(a, b) {\r\n var p = a,\r\n inside = false,\r\n px = (a.x + b.x) / 2,\r\n py = (a.y + b.y) / 2;\r\n do {\r\n if (\r\n p.y > py !== p.next.y > py &&\r\n p.next.y !== p.y &&\r\n px < ((p.next.x - p.x) * (py - p.y)) / (p.next.y - p.y) + p.x\r\n )\r\n inside = !inside;\r\n p = p.next;\r\n } while (p !== a);\r\n\r\n return inside;\r\n}\r\n\r\n// link two polygon vertices with a bridge; if the vertices belong to the same ring, it splits polygon into two;\r\n// if one belongs to the outer ring and another to a hole, it merges it into a single ring\r\nfunction splitPolygon(a, b) {\r\n var a2 = new Node(a.i, a.x, a.y),\r\n b2 = new Node(b.i, b.x, b.y),\r\n an = a.next,\r\n bp = b.prev;\r\n\r\n a.next = b;\r\n b.prev = a;\r\n\r\n a2.next = an;\r\n an.prev = a2;\r\n\r\n b2.next = a2;\r\n a2.prev = b2;\r\n\r\n bp.next = b2;\r\n b2.prev = bp;\r\n\r\n return b2;\r\n}\r\n\r\n// create a node and optionally link it with previous one (in a circular doubly linked list)\r\nfunction insertNode(i, x, y, last) {\r\n var p = new Node(i, x, y);\r\n\r\n if (!last) {\r\n p.prev = p;\r\n p.next = p;\r\n } else {\r\n p.next = last.next;\r\n p.prev = last;\r\n last.next.prev = p;\r\n last.next = p;\r\n }\r\n return p;\r\n}\r\n\r\nfunction removeNode(p) {\r\n p.next.prev = p.prev;\r\n p.prev.next = p.next;\r\n\r\n if (p.prevZ) p.prevZ.nextZ = p.nextZ;\r\n if (p.nextZ) p.nextZ.prevZ = p.prevZ;\r\n}\r\n\r\nfunction Node(i, x, y) {\r\n // vertex index in coordinates array\r\n this.i = i;\r\n\r\n // vertex coordinates\r\n this.x = x;\r\n this.y = y;\r\n\r\n // previous and next vertex nodes in a polygon ring\r\n this.prev = null;\r\n this.next = null;\r\n\r\n // z-order curve value\r\n this.z = 0;\r\n\r\n // previous and next nodes in z-order\r\n this.prevZ = null;\r\n this.nextZ = null;\r\n\r\n // indicates whether this is a steiner point\r\n this.steiner = false;\r\n}\r\n\r\n// return a percentage difference between the polygon area and its triangulation area;\r\n// used to verify correctness of triangulation\r\nearcut.deviation = function (data, holeIndices, dim, triangles) {\r\n var hasHoles = holeIndices && holeIndices.length;\r\n var outerLen = hasHoles ? holeIndices[0] * dim : data.length;\r\n\r\n var polygonArea = Math.abs(signedArea(data, 0, outerLen, dim));\r\n if (hasHoles) {\r\n for (var i = 0, len = holeIndices.length; i < len; i++) {\r\n var start = holeIndices[i] * dim;\r\n var end = i < len - 1 ? holeIndices[i + 1] * dim : data.length;\r\n polygonArea -= Math.abs(signedArea(data, start, end, dim));\r\n }\r\n }\r\n\r\n var trianglesArea = 0;\r\n for (i = 0; i < triangles.length; i += 3) {\r\n var a = triangles[i] * dim;\r\n var b = triangles[i + 1] * dim;\r\n var c = triangles[i + 2] * dim;\r\n trianglesArea += Math.abs(\r\n (data[a] - data[c]) * (data[b + 1] - data[a + 1]) - (data[a] - data[b]) * (data[c + 1] - data[a + 1])\r\n );\r\n }\r\n\r\n return polygonArea === 0 && trianglesArea === 0 ? 0 : Math.abs((trianglesArea - polygonArea) / polygonArea);\r\n};\r\n\r\nfunction signedArea(data, start, end, dim) {\r\n var sum = 0;\r\n for (var i = start, j = end - dim; i < end; i += dim) {\r\n sum += (data[j] - data[i]) * (data[i + 1] + data[j + 1]);\r\n j = i;\r\n }\r\n return sum;\r\n}\r\n\r\n// turn a polygon in a multi-dimensional array form (e.g. as in GeoJSON) into a form Earcut accepts\r\nearcut.flatten = function (data) {\r\n var dim = data[0][0].length,\r\n result = { vertices: [], holes: [], dimensions: dim },\r\n holeIndex = 0;\r\n\r\n for (var i = 0; i < data.length; i++) {\r\n for (var j = 0; j < data[i].length; j++) {\r\n for (var d = 0; d < dim; d++) result.vertices.push(data[i][j][d]);\r\n }\r\n if (i > 0) {\r\n holeIndex += data[i - 1].length;\r\n result.holes.push(holeIndex);\r\n }\r\n }\r\n return result;\r\n};\r\n","import { Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { VertexBuffer } from '@babylonjs/core/Meshes/buffer';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { VertexData } from '@babylonjs/core/Meshes/mesh.vertexData';\r\nimport { SubMesh } from '@babylonjs/core/Meshes/subMesh';\r\nimport { IndicesArray, Nullable } from '@babylonjs/core/types';\r\nimport { KbGeometry } from '@models/classes';\r\nimport { SubmeshData } from '@models/classes/features/submesh-feature';\r\n\r\nexport class SubmeshVertexData extends VertexData {\r\n /** Array of linked index segments, i.e. 0,1,1,2,2,3. A break in the link corresponds to a new path. */\r\n paths?: Nullable;\r\n /** Array of pairs of numbers, the first is the hole identifier, and the second is the index at which the hole starts in the path array */\r\n holes?: Nullable;\r\n submeshData?: Nullable;\r\n\r\n constructor() {\r\n super();\r\n this.indices = null;\r\n this.positions = null;\r\n this.normals = null;\r\n this.tangents = null;\r\n this.submeshData = null;\r\n }\r\n\r\n /**\r\n * @param kbGeometry the kbGeometry to copy the vertex data from.\r\n * Note that this copipes the vertex data by reference, so mutating the data WILL modify the kbGeometry.\r\n */\r\n static FromGeometry(kbGeometry: KbGeometry) {\r\n const result = new SubmeshVertexData();\r\n result.indices = kbGeometry.indices || null;\r\n result.positions = kbGeometry.positions || null;\r\n result.normals = kbGeometry.normals || null;\r\n result.tangents = kbGeometry.tangents || null;\r\n result.uvs = kbGeometry.uvs || null;\r\n result.submeshData = [\r\n {\r\n materialIndex: 0,\r\n verticesStart: 0,\r\n verticesCount: kbGeometry.positions ? kbGeometry.positions.length / 3 : 0,\r\n indexStart: 0,\r\n indexCount: kbGeometry.indices ? kbGeometry.indices.length : 0,\r\n },\r\n ];\r\n\r\n return result;\r\n }\r\n\r\n static ExtractFromMesh(mesh: Mesh, copyWhenShared?: boolean | undefined, forceCopy?: boolean | undefined) {\r\n const result = new SubmeshVertexData();\r\n const vertexData = super.ExtractFromMesh(mesh, copyWhenShared, forceCopy);\r\n\r\n SubmeshVertexData.CopyToRef(vertexData, result);\r\n if (mesh.subMeshes) {\r\n result.submeshData = extractSubmeshData(mesh.subMeshes);\r\n }\r\n if (mesh.metadata && mesh.metadata['paths']) {\r\n result.paths = copyWhenShared ? mesh.metadata['paths'].slice() : mesh.metadata['paths'];\r\n }\r\n if (mesh.metadata && mesh.metadata['holes']) {\r\n result.holes = copyWhenShared ? mesh.metadata['holes'].slice() : mesh.metadata['holes'];\r\n }\r\n\r\n return result;\r\n }\r\n\r\n static CopyToRef(source: Partial, target: SubmeshVertexData) {\r\n target.positions = source.positions || null;\r\n if ('paths' in source) {\r\n target.paths = source.paths || null;\r\n }\r\n if ('holes' in source) {\r\n target.holes = source.holes || null;\r\n }\r\n target.normals = source.normals || null;\r\n target.tangents = source.tangents || null;\r\n target.uvs = source.uvs || null;\r\n target.uvs2 = source.uvs2 || null;\r\n target.uvs3 = source.uvs3 || null;\r\n target.uvs4 = source.uvs4 || null;\r\n target.uvs5 = source.uvs5 || null;\r\n target.uvs6 = source.uvs6 || null;\r\n target.colors = source.colors || null;\r\n target.matricesIndices = source.matricesIndices || null;\r\n target.matricesWeights = source.matricesWeights || null;\r\n target.matricesIndicesExtra = source.matricesIndicesExtra || null;\r\n target.matricesWeightsExtra = source.matricesWeightsExtra || null;\r\n target.indices = source.indices || null;\r\n target.submeshData = (source as SubmeshVertexData).submeshData;\r\n }\r\n\r\n static ExtendToRef(source: Partial, target: SubmeshVertexData) {\r\n if (source.positions !== undefined) target.positions = source.positions;\r\n if ('paths' in source && source.paths !== undefined) target.paths = source.paths;\r\n if ('holes' in source && source.holes !== undefined) target.holes = source.holes;\r\n if (source.normals !== undefined) target.normals = source.normals;\r\n if (source.tangents !== undefined) target.tangents = source.tangents;\r\n if (source.uvs !== undefined) target.uvs = source.uvs;\r\n if (source.uvs2 !== undefined) target.uvs2 = source.uvs2;\r\n if (source.uvs3 !== undefined) target.uvs3 = source.uvs3;\r\n if (source.uvs4 !== undefined) target.uvs4 = source.uvs4;\r\n if (source.uvs5 !== undefined) target.uvs5 = source.uvs5;\r\n if (source.uvs6 !== undefined) target.uvs6 = source.uvs6;\r\n if (source.colors !== undefined) target.colors = source.colors;\r\n if (source.matricesIndices !== undefined) target.matricesIndices = source.matricesIndices;\r\n if (source.matricesWeights !== undefined) target.matricesWeights = source.matricesWeights;\r\n if (source.matricesIndicesExtra !== undefined) target.matricesIndicesExtra = source.matricesIndicesExtra;\r\n if (source.matricesWeightsExtra !== undefined) target.matricesWeightsExtra = source.matricesWeightsExtra;\r\n if (source.indices !== undefined) target.indices = source.indices;\r\n if ((source as SubmeshVertexData).submeshData !== undefined) {\r\n target.submeshData = (source as SubmeshVertexData).submeshData;\r\n }\r\n }\r\n\r\n deepClone() {\r\n const result = new SubmeshVertexData();\r\n result.indices = this.indices ? this.indices.slice() : null;\r\n result.positions = this.positions ? this.positions.slice() : null;\r\n result.paths = this.paths ? this.paths.slice() : null;\r\n result.holes = this.holes ? this.holes.slice() : null;\r\n result.normals = this.normals ? this.normals.slice() : null;\r\n result.tangents = this.tangents ? this.tangents.slice() : null;\r\n result.uvs = this.uvs ? this.uvs.slice() : null;\r\n result.submeshData = this.submeshData ? JSON.parse(JSON.stringify(this.submeshData)) : undefined;\r\n\r\n return result;\r\n }\r\n\r\n shallowClone() {\r\n const result = new SubmeshVertexData();\r\n SubmeshVertexData.CopyToRef(this, result);\r\n return result;\r\n }\r\n\r\n copyFrom(source: Partial) {\r\n SubmeshVertexData.CopyToRef(source, this);\r\n }\r\n\r\n merge(\r\n others: SubmeshVertexData | SubmeshVertexData[],\r\n use32BitsIndices?: boolean,\r\n forceCloneIndices?: boolean,\r\n mergeMaterialIds?: boolean,\r\n enableCompletion?: boolean\r\n ): SubmeshVertexData {\r\n if (!Array.isArray(others)) {\r\n others = [others];\r\n }\r\n\r\n let pathIdxOffset = this.paths ? this.paths.length : 0;\r\n let segmentIdxOffset = 0;\r\n let vertexIdxOffset = this.positions ? this.positions.length / 3 : 0;\r\n const newPaths: number[] = [];\r\n const newHoles: number[] = [];\r\n\r\n if (this.paths) {\r\n for (let i = 2; i < this.paths.length; i += 2) {\r\n if (this.paths[i] !== this.paths[i - 1]) {\r\n segmentIdxOffset++;\r\n }\r\n }\r\n if (this.holes) {\r\n segmentIdxOffset -= this.holes.length / 2;\r\n }\r\n segmentIdxOffset++;\r\n }\r\n\r\n others.map(other => {\r\n if (other.holes) {\r\n for (let i = 0; i < other.holes.length; i += 2) {\r\n newHoles.push(other.holes[i] + segmentIdxOffset, other.holes[i + 1] + pathIdxOffset);\r\n }\r\n segmentIdxOffset -= other.holes.length / 2;\r\n }\r\n if (other.paths) {\r\n for (let i = 0; i < other.paths.length; i++) {\r\n newPaths.push(other.paths[i] + vertexIdxOffset);\r\n }\r\n for (let i = 2; i < other.paths.length; i += 2) {\r\n if (other.paths[i] !== other.paths[i - 1]) {\r\n segmentIdxOffset++;\r\n }\r\n }\r\n segmentIdxOffset++;\r\n }\r\n\r\n vertexIdxOffset += other.positions ? other.positions.length / 3 : 0;\r\n pathIdxOffset += other.paths ? other.paths.length : 0;\r\n });\r\n\r\n super.merge(others, use32BitsIndices, true, mergeMaterialIds, enableCompletion);\r\n\r\n if (newHoles.length) {\r\n if (!this.holes) {\r\n this.holes = new Int32Array();\r\n }\r\n if ('push' in this.holes) {\r\n for (let i = 0; i < newHoles.length; i++) {\r\n this.holes.push(newHoles[i]);\r\n }\r\n } else {\r\n const newArr: Int32Array = new Int32Array(this.holes.length + newHoles.length);\r\n newArr.set(this.holes);\r\n newArr.set(newHoles, this.holes.length);\r\n this.holes = newArr;\r\n }\r\n }\r\n if (newPaths.length) {\r\n if (!this.paths) {\r\n this.paths = new Int32Array();\r\n }\r\n if ('push' in this.paths) {\r\n for (let i = 0; i < newPaths.length; i++) {\r\n this.paths.push(newPaths[i]);\r\n }\r\n } else {\r\n const newArr: Int32Array = new Int32Array(this.paths.length + newPaths.length);\r\n newArr.set(this.paths);\r\n newArr.set(newPaths, this.paths.length);\r\n this.paths = newArr;\r\n }\r\n }\r\n return this;\r\n }\r\n\r\n applyToMesh(mesh: Mesh, updatable = true, writeSubmeshes = true) {\r\n if (!this.submeshData && writeSubmeshes && mesh.subMeshes && mesh.subMeshes.length > 1) {\r\n this.submeshData = extractSubmeshData(mesh.subMeshes);\r\n }\r\n const result = super.applyToMesh(mesh, updatable);\r\n if (this.paths) {\r\n if (!mesh.metadata) {\r\n mesh.metadata = {};\r\n }\r\n mesh.metadata['paths'] = this.paths.slice();\r\n }\r\n if (this.holes) {\r\n if (!mesh.metadata) {\r\n mesh.metadata = {};\r\n }\r\n mesh.metadata['holes'] = this.holes.slice();\r\n }\r\n if (mesh.geometry) {\r\n // Applying geometry doesn't remove vertex types if they are unset so remove them manually\r\n if (!this.normals) {\r\n mesh.geometry.removeVerticesData(VertexBuffer.NormalKind);\r\n }\r\n if (!this.uvs) {\r\n mesh.geometry.removeVerticesData(VertexBuffer.UVKind);\r\n }\r\n }\r\n\r\n if (writeSubmeshes) {\r\n if (this.submeshData) {\r\n mesh.subMeshes = [];\r\n this.submeshData.forEach(submesh => {\r\n // SubMeshes add themselves to the mesh\r\n new SubMesh(\r\n submesh.materialIndex,\r\n submesh.verticesStart,\r\n submesh.verticesCount,\r\n submesh.indexStart,\r\n submesh.indexCount,\r\n mesh\r\n );\r\n });\r\n }\r\n } else {\r\n this.submeshData = extractSubmeshData(mesh.subMeshes);\r\n }\r\n\r\n return result;\r\n }\r\n}\r\n\r\nfunction extractSubmeshData(subMeshes: SubMesh[]): SubmeshData[] {\r\n return subMeshes.map(submesh => {\r\n return {\r\n materialIndex: submesh.materialIndex,\r\n verticesStart: submesh.verticesStart,\r\n verticesCount: submesh.verticesCount,\r\n indexStart: submesh.indexStart,\r\n indexCount: submesh.indexCount,\r\n };\r\n });\r\n}\r\n\r\n/**\r\n * @param vertexData\r\n * @returns null if geometry is valid, string if invalid\r\n */\r\nexport function verifyGeometry(vertexData: SubmeshVertexData): null | 'vertices' | 'submeshes' {\r\n let outputError: null | 'vertices' | 'submeshes' = null;\r\n if (vertexData.indices && vertexData.positions && vertexData.indices.length > 0) {\r\n const indicesCount = vertexData.indices.length;\r\n const vertexCount = vertexData.positions.length / 3;\r\n\r\n if (vertexData.submeshData) {\r\n vertexData.submeshData.forEach(submesh => {\r\n if (\r\n submesh.indexStart + submesh.indexCount > indicesCount ||\r\n submesh.verticesStart + submesh.verticesCount > vertexCount\r\n ) {\r\n outputError = 'submeshes';\r\n }\r\n });\r\n }\r\n\r\n if (!outputError) {\r\n const c = vertexData.indices.length;\r\n let maxVertex = -1;\r\n for (let i = 0; i < c; i++) {\r\n maxVertex = Math.max(maxVertex, vertexData.indices[i]);\r\n }\r\n if (maxVertex > vertexData.positions.length / 3) {\r\n outputError = 'vertices';\r\n }\r\n }\r\n }\r\n\r\n return outputError;\r\n}\r\n\r\nexport function getPolygonNormal(_points: Vector3[] | number[] | Float32Array) {\r\n const normal = Vector3.Zero();\r\n if (!_points) {\r\n return normal;\r\n }\r\n\r\n if (typeof _points[0] === 'number') {\r\n const points = _points as number[] | Float32Array;\r\n if (points.length < 9) {\r\n return normal;\r\n }\r\n\r\n const p1 = Vector3.Zero();\r\n const p2 = Vector3.Zero();\r\n\r\n let i = 0;\r\n while (p1.length() === 0 && points.length > i + 5) {\r\n p1.set(points[i + 3], points[i + 4], points[i + 5]);\r\n p1.subtractFromFloatsToRef(points[i], points[i + 1], points[i + 2], p1);\r\n i += 3;\r\n }\r\n\r\n p2.set(points[i], points[i + 1], points[i + 2]);\r\n while (normal.length() === 0 && points.length > i + 3) {\r\n p2.set(points[i + 3], points[i + 4], points[i + 5]);\r\n p2.subtractFromFloatsToRef(points[i], points[i + 1], points[i + 2], p2);\r\n Vector3.CrossToRef(p1, p2, normal);\r\n i += 3;\r\n }\r\n } else {\r\n const points = _points as Vector3[];\r\n if (points.length < 3) {\r\n return normal;\r\n }\r\n const p1 = Vector3.Zero();\r\n const p2 = Vector3.Zero();\r\n\r\n let i = 0;\r\n while (p1.length() === 0 && points.length > i + 1) {\r\n p1.copyFrom(points[i + 1]);\r\n p1.subtractInPlace(points[i]);\r\n i++;\r\n }\r\n\r\n while (normal.length() === 0 && points.length > i + 1) {\r\n p2.copyFrom(points[i + 1]);\r\n p2.subtractInPlace(points[i]);\r\n Vector3.CrossToRef(p1, p2, normal);\r\n i++;\r\n }\r\n }\r\n\r\n return normal.normalize();\r\n}\r\n","import { Matrix, Quaternion, Vector3 } from '@babylonjs/core/Maths/math.vector';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { BoundingBox, Font } from 'opentype.js';\r\nimport { earcut } from './earcut';\r\nimport { SubmeshVertexData } from './geometry-helper';\r\n\r\nconst MAX_BEZIER_STEPS = 10;\r\nconst BEZIER_STEP_SIZE = 20.0;\r\nconst GLYPH_COORDS_SCALE = 0.0005;\r\nconst IDENTITY_Q = Quaternion.Identity();\r\n\r\ninterface Point {\r\n x: number;\r\n y: number;\r\n}\r\n\r\ninterface GlyphDef {\r\n index: number;\r\n advanceWidth: number;\r\n boundingBox: BoundingBox;\r\n vertexData?: SubmeshVertexData;\r\n}\r\n\r\n// class for converting path commands into point data\r\nclass TextMeshPolygon {\r\n points: Point[] = [];\r\n children: TextMeshPolygon[] = [];\r\n area = 0.0;\r\n\r\n distance(p1: Point, p2: Point) {\r\n const dx = p1.x - p2.x,\r\n dy = p1.y - p2.y;\r\n return Math.sqrt(dx * dx + dy * dy);\r\n }\r\n\r\n lerp(p1: Point, p2: Point, t: number) {\r\n return { x: (1 - t) * p1.x + t * p2.x, y: (1 - t) * p1.y + t * p2.y };\r\n }\r\n\r\n cross(p1: Point, p2: Point) {\r\n return p1.x * p2.y - p1.y * p2.x;\r\n }\r\n\r\n moveTo(p: Point) {\r\n this.points.push(p);\r\n }\r\n\r\n lineTo(p: Point) {\r\n this.points.push(p);\r\n }\r\n\r\n close() {\r\n let cur = this.points[this.points.length - 1];\r\n\r\n this.points.forEach(next => {\r\n this.area += 0.5 * this.cross(cur, next);\r\n cur = next;\r\n });\r\n }\r\n\r\n conicTo(p: Point, p1: Point) {\r\n const p0 = this.points[this.points.length - 1];\r\n const dist = this.distance(p0, p1) + this.distance(p1, p);\r\n const steps = Math.max(2, Math.min(MAX_BEZIER_STEPS, dist / BEZIER_STEP_SIZE));\r\n\r\n for (let i = 1; i <= steps; ++i) {\r\n const t = i / steps;\r\n this.points.push(this.lerp(this.lerp(p0, p1, t), this.lerp(p1, p, t), t));\r\n }\r\n }\r\n\r\n cubicTo(p: Point, p1: Point, p2: Point) {\r\n const p0 = this.points[this.points.length - 1];\r\n const dist = this.distance(p0, p1) + this.distance(p1, p2) + this.distance(p2, p);\r\n const steps = Math.max(2, Math.min(MAX_BEZIER_STEPS, dist / BEZIER_STEP_SIZE));\r\n\r\n for (let i = 1; i <= steps; ++i) {\r\n const t = i / steps;\r\n const a = this.lerp(this.lerp(p0, p1, t), this.lerp(p1, p2, t), t);\r\n const b = this.lerp(this.lerp(p1, p2, t), this.lerp(p2, p, t), t);\r\n this.points.push(this.lerp(a, b, t));\r\n }\r\n }\r\n\r\n inside(p: Point) {\r\n const epsilon = 1e-6;\r\n let count = 0,\r\n cur = this.points[this.points.length - 1];\r\n\r\n this.points.forEach(next => {\r\n const p0 = cur.y < next.y ? cur : next;\r\n const p1 = cur.y < next.y ? next : cur;\r\n\r\n if (p0.y < p.y + epsilon && p1.y > p.y + epsilon) {\r\n if ((p1.x - p0.x) * (p.y - p0.y) > (p.x - p0.x) * (p1.y - p0.y)) {\r\n count++;\r\n }\r\n }\r\n\r\n cur = next;\r\n });\r\n return count % 2 !== 0;\r\n }\r\n}\r\n\r\nexport class TextMeshFont {\r\n glyphsParent: Mesh;\r\n glyphs: Record = {};\r\n\r\n constructor(name: string, public font: Font, private scene: Scene) {\r\n this.font = font;\r\n this.glyphsParent = new Mesh(name, scene);\r\n }\r\n\r\n createGlyph(ch: string) {\r\n if (this.glyphs[ch]) {\r\n return this.glyphs[ch];\r\n }\r\n\r\n const glyph = this.font.charToGlyph(ch);\r\n\r\n if (!glyph || !glyph.advanceWidth) {\r\n return undefined;\r\n }\r\n this.glyphs[ch] = {\r\n index: glyph.index,\r\n advanceWidth: glyph.advanceWidth,\r\n boundingBox: glyph.getBoundingBox(),\r\n };\r\n const glyphDef = this.glyphs[ch];\r\n\r\n if (!glyph.path || !glyph.path.commands || !glyph.path.commands.length) {\r\n return glyphDef;\r\n }\r\n const polys: TextMeshPolygon[] = [];\r\n glyph.path.commands.forEach(command => {\r\n const { type, x, y, x1, y1, x2, y2 } = command as unknown as Record & { type: string };\r\n switch (type) {\r\n case 'M':\r\n polys.push(new TextMeshPolygon());\r\n polys[polys.length - 1].moveTo({ x, y });\r\n break;\r\n case 'L':\r\n polys[polys.length - 1].moveTo({ x, y });\r\n break;\r\n case 'C':\r\n polys[polys.length - 1].cubicTo({ x, y }, { x: x1, y: y1 }, { x: x2, y: y2 });\r\n break;\r\n case 'Q':\r\n polys[polys.length - 1].conicTo({ x, y }, { x: x1, y: y1 });\r\n break;\r\n case 'Z':\r\n polys[polys.length - 1].close();\r\n break;\r\n }\r\n });\r\n\r\n // sort contours by descending area\r\n polys.sort((a, b) => Math.abs(b.area) - Math.abs(a.area));\r\n\r\n // classify contours to find holes and their 'parents'\r\n const root = [];\r\n\r\n for (let i = 0; i < polys.length; ++i) {\r\n let parent: TextMeshPolygon | null = null;\r\n for (let j = i - 1; j >= 0; --j) {\r\n // a contour is a hole if it is inside its parent and has different winding\r\n if (polys[j].inside(polys[i].points[0]) && polys[i].area * polys[j].area < 0) {\r\n parent = polys[j];\r\n break;\r\n }\r\n }\r\n if (parent) {\r\n parent.children.push(polys[i]);\r\n } else {\r\n root.push(polys[i]);\r\n }\r\n }\r\n\r\n const totalPoints = polys.reduce((sum, p) => sum + p.points.length, 0);\r\n const vertexData = new Float32Array(totalPoints * 2);\r\n let vertexCount = 0;\r\n let pathSegmentCount = 0;\r\n const indices: number[] = [];\r\n const paths: number[] = [];\r\n const holePaths: number[] = [];\r\n\r\n function process(poly: TextMeshPolygon) {\r\n // construct input for earcut\r\n const coords: number[] = [];\r\n const holes: number[] = [];\r\n const n = poly.points.length - 1;\r\n for (let i = 0; i < n; ++i) {\r\n const { x, y } = poly.points[n - i];\r\n if (i > 0) {\r\n const idx = coords.length / 2;\r\n paths.push(vertexCount + idx - 1, vertexCount + idx);\r\n }\r\n coords.push(x, y);\r\n }\r\n const parentSegmentIdx = pathSegmentCount;\r\n poly.children.forEach((child, i) => {\r\n // children's children are new, separate shapes\r\n child.children.forEach(process);\r\n\r\n // idx 0 of a hole is the segment index of the segment the hole is inside. each separate poly is a separate segment\r\n holePaths.push(parentSegmentIdx, paths.length);\r\n holes.push(coords.length / 2);\r\n const n = child.points.length - 1;\r\n for (let i = 0; i < n; ++i) {\r\n const { x, y } = child.points[i];\r\n if (i > 0) {\r\n const idx = coords.length / 2;\r\n paths.push(idx - 1, idx);\r\n }\r\n coords.push(x, y);\r\n }\r\n pathSegmentCount++;\r\n });\r\n\r\n // add vertex data\r\n vertexData.set(coords, vertexCount * 2);\r\n // add index data\r\n earcut(coords, holes).forEach(i => {\r\n indices.push(i + vertexCount);\r\n });\r\n vertexCount += coords.length / 2;\r\n pathSegmentCount++;\r\n }\r\n\r\n root.forEach(process);\r\n\r\n const meshdata = new SubmeshVertexData();\r\n const vertices: number[] = [];\r\n const normals: number[] = [];\r\n\r\n for (let i = 0; i < vertexCount; i++) {\r\n vertices.push(vertexData[i * 2] * GLYPH_COORDS_SCALE);\r\n vertices.push(vertexData[i * 2 + 1] * GLYPH_COORDS_SCALE);\r\n vertices.push(0);\r\n\r\n normals.push(0);\r\n normals.push(0);\r\n normals.push(1);\r\n }\r\n\r\n const flipIndices = [];\r\n for (let i = 0; i < indices.length; i += 3) {\r\n flipIndices.push(indices[i], indices[i + 2], indices[i + 1]);\r\n }\r\n\r\n meshdata.positions = vertices;\r\n meshdata.indices = flipIndices;\r\n meshdata.normals = normals;\r\n meshdata.paths = paths;\r\n meshdata.holes = holePaths;\r\n\r\n glyphDef.vertexData = meshdata;\r\n\r\n return glyphDef;\r\n }\r\n}\r\n\r\nexport class TextMeshVertexData {\r\n instances: Record = {};\r\n\r\n constructor(private textMeshFont: TextMeshFont) {}\r\n\r\n getText(text: string, fontSize = 1, charSpacing = 1, radius = 0) {\r\n radius = radius / fontSize;\r\n const scaleMatrix = Matrix.Compose(new Vector3(fontSize, fontSize, 1), IDENTITY_Q, Vector3.ZeroReadOnly);\r\n const vertexData = new SubmeshVertexData();\r\n const pos = Vector3.Zero();\r\n const transformMatrix = Matrix.Identity();\r\n\r\n const generatedVertices: Array<{ data: SubmeshVertexData; transform: Matrix; center: number }> = [];\r\n\r\n for (let i = 0; i < text.length; i++) {\r\n const ch1 = text[i];\r\n\r\n if (ch1 === '\\n') {\r\n pos.x = 0;\r\n pos.y -= 1.1;\r\n continue;\r\n }\r\n transformMatrix.setTranslation(pos);\r\n\r\n const ch2 = text[i + 1];\r\n const g = this.textMeshFont.createGlyph(ch1);\r\n\r\n if (!g) {\r\n continue;\r\n }\r\n let advance = g.advanceWidth;\r\n\r\n if (g.vertexData) {\r\n generatedVertices.push({\r\n data: g.vertexData.deepClone(),\r\n transform: transformMatrix.clone(),\r\n center: (advance / 2 + g.boundingBox.x1) * GLYPH_COORDS_SCALE,\r\n });\r\n }\r\n\r\n if (advance) {\r\n if (ch2 && this.textMeshFont.glyphs[ch2]) {\r\n const kern = this.textMeshFont.font.getKerningValue(g.index, this.textMeshFont.glyphs[ch2].index);\r\n\r\n if (kern) {\r\n advance += kern;\r\n }\r\n }\r\n\r\n pos.x += (advance - g.boundingBox.x2 + g.boundingBox.x2 * charSpacing) * GLYPH_COORDS_SCALE;\r\n }\r\n }\r\n\r\n if (generatedVertices.length > 0) {\r\n const centerTransform = Matrix.Compose(Vector3.OneReadOnly, IDENTITY_Q, pos.scale(-0.5));\r\n\r\n for (const generatedVertexData of generatedVertices) {\r\n const transform = generatedVertexData.transform;\r\n transform.multiplyToRef(centerTransform, transform);\r\n\r\n // Add curve to text\r\n if (radius !== 0) {\r\n const baseOffset = transform.getTranslation();\r\n const angle = (baseOffset.x + generatedVertexData.center) / radius;\r\n const pivotMatrix = Matrix.Compose(\r\n Vector3.OneReadOnly,\r\n IDENTITY_Q,\r\n new Vector3(generatedVertexData.center, -radius, 0)\r\n );\r\n const rotMatrix = Matrix.Compose(\r\n Vector3.OneReadOnly,\r\n Quaternion.RotationAxis(new Vector3(0, 0, -1), angle),\r\n Vector3.ZeroReadOnly\r\n );\r\n const curveTransform = pivotMatrix.clone().invert();\r\n curveTransform.multiplyToRef(rotMatrix, curveTransform);\r\n curveTransform.multiplyToRef(pivotMatrix, curveTransform);\r\n curveTransform.setTranslation(\r\n curveTransform.getTranslation().addInPlaceFromFloats(-generatedVertexData.center, 0, 0)\r\n );\r\n\r\n transform.copyFrom(curveTransform);\r\n }\r\n\r\n transform.multiplyToRef(scaleMatrix, transform);\r\n generatedVertexData.data.transform(transform);\r\n }\r\n\r\n vertexData.copyFrom(generatedVertices[0].data);\r\n\r\n if (generatedVertices.length > 1) {\r\n vertexData.merge(generatedVertices.slice(1).map(generatedVertexData => generatedVertexData.data));\r\n }\r\n }\r\n\r\n // const calcW = this.textMeshFont.font.getAdvanceWidth(text) * GLYPH_COORDS_SCALE;\r\n\r\n // if (vertexData.positions) {\r\n // const max = Vector3.Zero();\r\n // for (let i = 0; i < vertexData.positions.length; i += 3) {\r\n // const p = vertexData.positions;\r\n // max.x = Math.max(max.x, p[i]);\r\n // max.y = Math.max(max.y, p[i + 1]);\r\n // max.z = Math.max(max.z, p[i + 2]);\r\n // }\r\n // const centerTransform = Matrix.Compose(Vector3.OneReadOnly, Quaternion.Identity(), max.scale(-0.5));\r\n // vertexData.transform(centerTransform);\r\n // }\r\n\r\n return vertexData;\r\n }\r\n}\r\n","import { Engine } from \"../Engines/engine\";\r\nimport { RawTexture } from \"./Textures/rawTexture\";\r\nimport { MaterialPluginBase } from \"./materialPluginBase\";\r\nimport type { Scene } from \"../scene\";\r\nimport type { UniformBuffer } from \"./uniformBuffer\";\r\nimport { Vector2, TmpVectors } from \"../Maths/math.vector\";\r\nimport { Color3 } from \"../Maths/math.color\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { Material } from \"./material\";\r\nimport { MaterialDefines } from \"./materialDefines\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { BaseTexture } from \"./Textures/baseTexture\";\r\nimport { RegisterClass } from \"../Misc/typeStore\";\r\n\r\n/**\r\n * Material types for GreasedLine\r\n * {@link https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param/greased_line#materialtype}\r\n */\r\nexport enum GreasedLineMeshMaterialType {\r\n /**\r\n * StandardMaterial\r\n */\r\n MATERIAL_TYPE_STANDARD = 0,\r\n /**\r\n * PBR Material\r\n */\r\n MATERIAL_TYPE_PBR = 1,\r\n}\r\n\r\n/**\r\n * Color blending mode of the @see GreasedLineMaterial and the base material\r\n * {@link https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param/greased_line#colormode}\r\n */\r\nexport enum GreasedLineMeshColorMode {\r\n /**\r\n * Color blending mode SET\r\n */\r\n COLOR_MODE_SET = 0,\r\n /**\r\n * Color blending mode ADD\r\n */\r\n COLOR_MODE_ADD = 1,\r\n /**\r\n * Color blending mode ADD\r\n */\r\n COLOR_MODE_MULTIPLY = 2,\r\n}\r\n\r\n/**\r\n * Color distribution type of the @see colors.\r\n * {@link https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param/greased_line#colordistributiontype}\r\n *\r\n */\r\nexport enum GreasedLineMeshColorDistributionType {\r\n /**\r\n * Colors distributed between segments of the line\r\n */\r\n COLOR_DISTRIBUTION_TYPE_SEGMENT = 0,\r\n /**\r\n * Colors distributed along the line ingoring the segments\r\n */\r\n COLOR_DISTRIBUTION_TYPE_LINE = 1,\r\n}\r\n\r\n/**\r\n * Options for GreasedLineMaterial\r\n */\r\nexport interface GreasedLineMaterialOptions {\r\n /**\r\n * Line width. If sizeAttenuation os false scene units will be used for width.\r\n * Defaults to 0.1 if @see sizeAttenuation is false, or to 1 if it's true.\r\n */\r\n width?: number;\r\n /**\r\n * If false then width units = scene units. If true then line will width be reduced.\r\n * Defaults to false.\r\n */\r\n sizeAttenuation?: boolean;\r\n /**\r\n * Type of the material to use to render the line.\r\n * Defaults to StandardMaterial.\r\n */\r\n materialType?: GreasedLineMeshMaterialType;\r\n /**\r\n * Color of the line. Applies to all line segments.\r\n * Defaults to White.\r\n */\r\n color?: Color3;\r\n /**\r\n * Color mode of the line. Applies to all line segments.\r\n * The pixel color from the material shader will be modified with the value of @see color using the colorMode.\r\n * Defaults to @see GreasedLineMeshColorMode.SET\r\n */\r\n colorMode?: GreasedLineMeshColorMode;\r\n /**\r\n * Colors of the line segments.\r\n * Defaults to empty.\r\n */\r\n colors?: Color3[];\r\n /**\r\n * If true, @see colors are used, otherwise they're ignored.\r\n * Defaults to false.\r\n */\r\n useColors?: boolean;\r\n /**\r\n * Sampling type of the colors texture\r\n * Defaults to NEAREST_NEAREST.\r\n */\r\n colorsSampling?: number;\r\n /**\r\n * The method used to distribute the colors along the line.\r\n * You can use segment distribution when each segment will use on color from the color table.\r\n * Or you can use line distribution when the colors are distributed evenly along the line ignoring the segments.\r\n */\r\n colorDistributionType?: GreasedLineMeshColorDistributionType;\r\n /**\r\n * If true, dashing is used.\r\n * Defaults to false.\r\n */\r\n useDash?: boolean;\r\n /**\r\n * @see GreasedLinePluginMaterial.setDashCount\r\n * Number of dashes in the line.\r\n * Defaults to 1.\r\n */\r\n dashCount?: number;\r\n /**\r\n * Offset of the dashes along the line. 0 to 1.\r\n * Defaults to 0.\r\n * @see GreasedLinePluginMaterial.setDashOffset\r\n */\r\n dashOffset?: number;\r\n /**\r\n * Length of the dash. 0 to 1. 0.5 means half empty, half drawn.\r\n * Defaults to 0.5.\r\n * @see GreasedLinePluginMaterial.setDashRatio\r\n */\r\n dashRatio?: number;\r\n /**\r\n * Sets the line length visibility.\r\n * 0 - 0% of the line will be visible.\r\n * 1 - 100% of the line will be visible.\r\n * @see GreasedLinePluginMaterial.setVisibility\r\n */\r\n visibility?: number;\r\n /**\r\n * Defaults to engine.getRenderWidth() and engine.getRenderHeight()\r\n * Rendering resolution\r\n */\r\n resolution?: Vector2;\r\n}\r\n\r\n/**\r\n * @internal\r\n */\r\nexport class MaterialGreasedLineDefines extends MaterialDefines {\r\n /**\r\n * The material has a color option specified\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n GREASED_LINE_HAS_COLOR = false;\r\n /**\r\n * The material's size attenuation optiom\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n GREASED_LINE_SIZE_ATTENUATION = false;\r\n /**\r\n * The type of color distribution is set to line this value equals to true.\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n GREASED_LINE_COLOR_DISTRIBUTION_TYPE_LINE = false;\r\n /**\r\n * True if scene is in right handed coordinate system.\r\n */\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n GREASED_LNE_RIGHT_HANDED_COORDINATE_SYSTEM = false;\r\n}\r\n\r\n/**\r\n * GreasedLinePluginMaterial for GreasedLineMesh\r\n */\r\nexport class GreasedLinePluginMaterial extends MaterialPluginBase {\r\n /**\r\n * Plugin name\r\n */\r\n public static readonly GREASED_LINE_MATERIAL_NAME = \"GreasedLinePluginMaterial\";\r\n\r\n /**\r\n * Default line color for newly created lines\r\n */\r\n public static DEFAULT_COLOR = Color3.White();\r\n /**\r\n * Default line width when sizeAttenuation is true\r\n */\r\n public static DEFAULT_WIDTH_ATTENUATED = 1;\r\n /**\r\n * Defaule line width\r\n */\r\n public static DEFAULT_WIDTH = 0.1;\r\n\r\n private static _EmptyColorsTexture: BaseTexture;\r\n\r\n /**\r\n * Whether to use the colors option to colorize the line\r\n */\r\n public useColors: boolean;\r\n\r\n /**\r\n * Normalized value of how much of the line will be visible\r\n * 0 - 0% of the line will be visible\r\n * 1 - 100% of the line will be visible\r\n */\r\n public visibility: number;\r\n\r\n /**\r\n * Dash offset\r\n */\r\n public dashOffset: number;\r\n\r\n /**\r\n * Length of the dash. 0 to 1. 0.5 means half empty, half drawn.\r\n */\r\n public dashRatio: number;\r\n\r\n /**\r\n * Line base width. At each point the line width is calculated by widths[pointIndex] * width\r\n */\r\n public width: number;\r\n\r\n /**\r\n * The type of sampling of the colors texture. The values are the same when using with textures.\r\n */\r\n public colorsSampling: number;\r\n\r\n /**\r\n * Turns on/off dash mode\r\n */\r\n public useDash: boolean;\r\n\r\n /**\r\n * The mixing mode of the color paramater. Default value is GreasedLineMeshColorMode.SET\r\n * @see GreasedLineMeshColorMode\r\n */\r\n public colorMode: GreasedLineMeshColorMode;\r\n\r\n private _dashCount: number;\r\n private _dashArray: number;\r\n private _color: Nullable;\r\n private _colors: Nullable;\r\n private _colorsDistributionType: GreasedLineMeshColorDistributionType;\r\n private _resolution: Vector2;\r\n private _aspect: number;\r\n private _sizeAttenuation: boolean;\r\n\r\n private _colorsTexture?: RawTexture;\r\n\r\n private _engine: Engine;\r\n\r\n constructor(material: Material, private _scene: Scene, options?: GreasedLineMaterialOptions) {\r\n options = options || {\r\n color: GreasedLinePluginMaterial.DEFAULT_COLOR,\r\n };\r\n\r\n const defines = new MaterialGreasedLineDefines();\r\n defines.GREASED_LINE_HAS_COLOR = !!options.color;\r\n defines.GREASED_LINE_SIZE_ATTENUATION = options.sizeAttenuation ?? false;\r\n defines.GREASED_LINE_COLOR_DISTRIBUTION_TYPE_LINE = options.colorDistributionType === GreasedLineMeshColorDistributionType.COLOR_DISTRIBUTION_TYPE_LINE;\r\n defines.GREASED_LNE_RIGHT_HANDED_COORDINATE_SYSTEM = _scene.useRightHandedSystem;\r\n super(material, GreasedLinePluginMaterial.GREASED_LINE_MATERIAL_NAME, 200, defines);\r\n\r\n this._scene = this._scene ?? material.getScene();\r\n this._engine = this._scene.getEngine();\r\n\r\n this.visibility = options.visibility ?? 1;\r\n this.useDash = options.useDash ?? false;\r\n this.dashRatio = options.dashRatio ?? 0.5;\r\n this.dashOffset = options.dashOffset ?? 0;\r\n this.width = options.width ? options.width : options.sizeAttenuation ? GreasedLinePluginMaterial.DEFAULT_WIDTH_ATTENUATED : GreasedLinePluginMaterial.DEFAULT_WIDTH;\r\n this._sizeAttenuation = options.sizeAttenuation ?? false;\r\n this.colorMode = options.colorMode ?? GreasedLineMeshColorMode.COLOR_MODE_SET;\r\n this._color = options.color ?? null;\r\n this.useColors = options.useColors ?? false;\r\n this._colorsDistributionType = options.colorDistributionType ?? GreasedLineMeshColorDistributionType.COLOR_DISTRIBUTION_TYPE_SEGMENT;\r\n this.colorsSampling = options.colorsSampling ?? RawTexture.NEAREST_NEAREST;\r\n this._colors = options.colors ?? null;\r\n\r\n this.dashCount = options.dashCount ?? 1; // calculate the _dashArray value, call the setter\r\n this.resolution = options.resolution ?? new Vector2(this._engine.getRenderWidth(), this._engine.getRenderHeight()); // calculate aspect call the setter\r\n\r\n if (this._colors) {\r\n this._createColorsTexture(`${material.name}-colors-texture`, this._colors);\r\n } else {\r\n this._color = this._color ?? GreasedLinePluginMaterial.DEFAULT_COLOR;\r\n GreasedLinePluginMaterial._PrepareEmptyColorsTexture(_scene);\r\n }\r\n\r\n this._enable(true); // always enabled\r\n }\r\n\r\n /**\r\n * Get the shader attributes\r\n * @param attributes array which will be filled with the attributes\r\n */\r\n getAttributes(attributes: string[]) {\r\n attributes.push(\"grl_offsets\");\r\n attributes.push(\"grl_previousAndSide\");\r\n attributes.push(\"grl_nextAndCounters\");\r\n attributes.push(\"grl_widths\");\r\n attributes.push(\"grl_colorPointers\");\r\n }\r\n\r\n /**\r\n * Get the shader samplers\r\n * @param samplers\r\n */\r\n getSamplers(samplers: string[]) {\r\n samplers.push(\"grl_colors\");\r\n }\r\n\r\n /**\r\n * Get the shader textures\r\n * @param activeTextures\r\n */\r\n public getActiveTextures(activeTextures: BaseTexture[]): void {\r\n if (this._colorsTexture) {\r\n activeTextures.push(this._colorsTexture);\r\n }\r\n }\r\n\r\n /**\r\n * Get the shader uniforms\r\n * @returns uniforms\r\n */\r\n getUniforms() {\r\n const ubo = [\r\n { name: \"grl_projection\", size: 16, type: \"mat4\" },\r\n { name: \"grl_singleColor\", size: 3, type: \"vec3\" },\r\n { name: \"grl_aspect_resolution_lineWidth\", size: 4, type: \"vec4\" },\r\n { name: \"grl_dashOptions\", size: 4, type: \"vec4\" },\r\n { name: \"grl_colorMode_visibility_colorsWidth_useColors\", size: 4, type: \"vec4\" },\r\n ];\r\n\r\n return {\r\n ubo,\r\n vertex: `\r\n uniform vec4 grl_aspect_resolution_lineWidth;\r\n uniform mat4 grl_projection;\r\n `,\r\n fragment: `\r\n uniform vec4 grl_dashOptions;\r\n uniform vec4 grl_colorMode_visibility_colorsWidth_useColors;\r\n uniform vec3 grl_singleColor;\r\n `,\r\n };\r\n }\r\n\r\n // only getter, it doesn't make sense to use this plugin on a mesh other than GreasedLineMesh\r\n // and it doesn't make sense to disable it on the mesh\r\n get isEnabled() {\r\n return true;\r\n }\r\n\r\n /**\r\n * Bind the uniform buffer\r\n * @param uniformBuffer\r\n */\r\n bindForSubMesh(uniformBuffer: UniformBuffer) {\r\n const activeCamera = this._scene.activeCamera;\r\n\r\n if (activeCamera) {\r\n const projection = activeCamera.getProjectionMatrix();\r\n uniformBuffer.updateMatrix(\"grl_projection\", projection);\r\n } else {\r\n throw Error(\"GreasedLinePluginMaterial requires an active camera.\");\r\n }\r\n\r\n const resolutionLineWidth = TmpVectors.Vector4[0];\r\n resolutionLineWidth.x = this._aspect;\r\n resolutionLineWidth.y = this._resolution.x;\r\n resolutionLineWidth.z = this._resolution.y;\r\n resolutionLineWidth.w = this.width;\r\n uniformBuffer.updateVector4(\"grl_aspect_resolution_lineWidth\", resolutionLineWidth);\r\n\r\n const dashOptions = TmpVectors.Vector4[0];\r\n dashOptions.x = GreasedLinePluginMaterial._BooleanToNumber(this.useDash);\r\n dashOptions.y = this._dashArray;\r\n dashOptions.z = this.dashOffset;\r\n dashOptions.w = this.dashRatio;\r\n uniformBuffer.updateVector4(\"grl_dashOptions\", dashOptions);\r\n\r\n const colorModeVisibilityColorsWidthUseColors = TmpVectors.Vector4[1];\r\n colorModeVisibilityColorsWidthUseColors.x = this.colorMode;\r\n colorModeVisibilityColorsWidthUseColors.y = this.visibility;\r\n colorModeVisibilityColorsWidthUseColors.z = this._colorsTexture ? this._colorsTexture.getSize().width : 0;\r\n colorModeVisibilityColorsWidthUseColors.w = GreasedLinePluginMaterial._BooleanToNumber(this.useColors);\r\n uniformBuffer.updateVector4(\"grl_colorMode_visibility_colorsWidth_useColors\", colorModeVisibilityColorsWidthUseColors);\r\n\r\n if (this._color) {\r\n uniformBuffer.updateColor3(\"grl_singleColor\", this._color);\r\n }\r\n\r\n uniformBuffer.setTexture(\"grl_colors\", this._colorsTexture ?? GreasedLinePluginMaterial._EmptyColorsTexture);\r\n }\r\n\r\n /**\r\n * Prepare the defines\r\n * @param defines\r\n * @param _scene\r\n * @param _mesh\r\n */\r\n prepareDefines(defines: MaterialGreasedLineDefines, _scene: Scene, _mesh: AbstractMesh) {\r\n defines.GREASED_LINE_HAS_COLOR = !!this._color;\r\n defines.GREASED_LINE_SIZE_ATTENUATION = this._sizeAttenuation ?? false;\r\n defines.GREASED_LINE_COLOR_DISTRIBUTION_TYPE_LINE = this._colorsDistributionType === GreasedLineMeshColorDistributionType.COLOR_DISTRIBUTION_TYPE_LINE;\r\n defines.GREASED_LNE_RIGHT_HANDED_COORDINATE_SYSTEM = _scene.useRightHandedSystem;\r\n }\r\n\r\n /**\r\n * Get the class name\r\n * @returns class name\r\n */\r\n getClassName() {\r\n return GreasedLinePluginMaterial.GREASED_LINE_MATERIAL_NAME;\r\n }\r\n\r\n /**\r\n * Get shader code\r\n * @param shaderType vertex/fragment\r\n * @returns shader code\r\n */\r\n getCustomCode(shaderType: string): Nullable<{ [pointName: string]: string }> {\r\n if (shaderType === \"vertex\") {\r\n return {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CUSTOM_VERTEX_DEFINITIONS: `\r\n attribute vec4 grl_previousAndSide;\r\n attribute vec4 grl_nextAndCounters;\r\n attribute float grl_widths;\r\n attribute vec3 grl_offsets;\r\n attribute float grl_colorPointers;\r\n\r\n varying float grlCounters;\r\n varying float grlColorPointer;\r\n\r\n vec2 grlFix( vec4 i, float aspect ) {\r\n vec2 res = i.xy / i.w;\r\n res.x *= aspect;\r\n return res;\r\n }\r\n `,\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CUSTOM_VERTEX_UPDATE_POSITION: `\r\n vec3 grlPositionOffset = grl_offsets;\r\n positionUpdated += grlPositionOffset;\r\n `,\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CUSTOM_VERTEX_MAIN_END: `\r\n\r\n float grlAspect = grl_aspect_resolution_lineWidth.x;\r\n float grlBaseWidth = grl_aspect_resolution_lineWidth.w;\r\n\r\n grlColorPointer = grl_colorPointers;\r\n\r\n vec3 grlPrevious = grl_previousAndSide.xyz;\r\n float grlSide = grl_previousAndSide.w;\r\n\r\n vec3 grlNext = grl_nextAndCounters.xyz;\r\n grlCounters = grl_nextAndCounters.w;\r\n\r\n\r\n mat4 grlMatrix = viewProjection * world;\r\n vec4 grlFinalPosition = grlMatrix * vec4( positionUpdated , 1.0 );\r\n vec4 grlPrevPos = grlMatrix * vec4( grlPrevious + grlPositionOffset, 1.0 );\r\n vec4 grlNextPos = grlMatrix * vec4( grlNext + grlPositionOffset, 1.0 );\r\n\r\n vec2 grlCurrentP = grlFix( grlFinalPosition, grlAspect );\r\n vec2 grlPrevP = grlFix( grlPrevPos, grlAspect );\r\n vec2 grlNextP = grlFix( grlNextPos, grlAspect );\r\n\r\n float grlWidth = grlBaseWidth * grl_widths;\r\n\r\n vec2 grlDir;\r\n if( grlNextP == grlCurrentP ) grlDir = normalize( grlCurrentP - grlPrevP );\r\n else if( grlPrevP == grlCurrentP ) grlDir = normalize( grlNextP - grlCurrentP );\r\n else {\r\n vec2 grlDir1 = normalize( grlCurrentP - grlPrevP );\r\n vec2 grlDir2 = normalize( grlNextP - grlCurrentP );\r\n grlDir = normalize( grlDir1 + grlDir2 );\r\n }\r\n vec4 grlNormal = vec4( -grlDir.y, grlDir.x, 0., 1. );\r\n #ifdef GREASED_LNE_RIGHT_HANDED_COORDINATE_SYSTEM\r\n grlNormal.xy *= -.5 * grlWidth;\r\n #else\r\n grlNormal.xy *= .5 * grlWidth;\r\n #endif\r\n grlNormal *= grl_projection;\r\n #ifdef GREASED_LINE_SIZE_ATTENUATION\r\n grlNormal.xy *= grlFinalPosition.w;\r\n grlNormal.xy /= ( vec4( grl_aspect_resolution_lineWidth.yz, 0., 1. ) * grl_projection ).xy;\r\n #endif\r\n grlFinalPosition.xy += grlNormal.xy * grlSide;\r\n gl_Position = grlFinalPosition;\r\n\r\n vPositionW = vec3(grlFinalPosition);\r\n\r\n `,\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n \"!gl_Position\\\\=viewProjection\\\\*worldPos;\": \"//\", // remove\r\n };\r\n }\r\n\r\n if (shaderType === \"fragment\") {\r\n return {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CUSTOM_FRAGMENT_DEFINITIONS: `\r\n varying float grlCounters;\r\n varying float grlColorPointer;\r\n uniform sampler2D grl_colors;\r\n `,\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CUSTOM_FRAGMENT_MAIN_END: `\r\n float grlColorMode = grl_colorMode_visibility_colorsWidth_useColors.x;\r\n float grlVisibility = grl_colorMode_visibility_colorsWidth_useColors.y;\r\n float grlColorsWidth = grl_colorMode_visibility_colorsWidth_useColors.z;\r\n float grlUseColors = grl_colorMode_visibility_colorsWidth_useColors.w;\r\n\r\n float grlUseDash = grl_dashOptions.x;\r\n float grlDashArray = grl_dashOptions.y;\r\n float grlDashOffset = grl_dashOptions.z;\r\n float grlDashRatio = grl_dashOptions.w;\r\n\r\n gl_FragColor.a *= step(grlCounters, grlVisibility);\r\n if( gl_FragColor.a == 0. ) discard;\r\n\r\n if(grlUseDash == 1.){\r\n gl_FragColor.a *= ceil(mod(grlCounters + grlDashOffset, grlDashArray) - (grlDashArray * grlDashRatio));\r\n if (gl_FragColor.a == 0.) discard;\r\n }\r\n\r\n #ifdef GREASED_LINE_HAS_COLOR\r\n if (grlColorMode == ${GreasedLineMeshColorMode.COLOR_MODE_SET}.) {\r\n gl_FragColor.rgb = grl_singleColor;\r\n } else if (grlColorMode == ${GreasedLineMeshColorMode.COLOR_MODE_ADD}.) {\r\n gl_FragColor.rgb += grl_singleColor;\r\n } else if (grlColorMode == ${GreasedLineMeshColorMode.COLOR_MODE_MULTIPLY}.) {\r\n gl_FragColor.rgb *= grl_singleColor;\r\n }\r\n #else\r\n if (grlUseColors == 1.) {\r\n #ifdef GREASED_LINE_COLOR_DISTRIBUTION_TYPE_LINE\r\n vec4 grlColor = texture2D(grl_colors, vec2(grlCounters, 0.), 0.);\r\n #else\r\n vec4 grlColor = texture2D(grl_colors, vec2(grlColorPointer/grlColorsWidth, 0.), 0.);\r\n #endif\r\n if (grlColorMode == ${GreasedLineMeshColorMode.COLOR_MODE_SET}.) {\r\n gl_FragColor = grlColor;\r\n } else if (grlColorMode == ${GreasedLineMeshColorMode.COLOR_MODE_ADD}.) {\r\n gl_FragColor += grlColor;\r\n } else if (grlColorMode == ${GreasedLineMeshColorMode.COLOR_MODE_MULTIPLY}.) {\r\n gl_FragColor *= grlColor;\r\n }\r\n }\r\n #endif\r\n `,\r\n };\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Converts boolean to number.\r\n * @param bool\r\n * @returns 1 if true, 0 if false.\r\n */\r\n private static _BooleanToNumber(bool?: boolean) {\r\n return bool ? 1 : 0;\r\n }\r\n\r\n /**\r\n * Converts an array of Color3 to Uint8Array\r\n * @param colors Arrray of Color3\r\n * @returns Uin8Array of colors [r, g, b, a, r, g, b, a, ...]\r\n */\r\n private static _Color3toRGBAUint8(colors: Color3[]) {\r\n const colorTable: Uint8Array = new Uint8Array(colors.length * 4);\r\n for (let i = 0, j = 0; i < colors.length; i++) {\r\n colorTable[j++] = colors[i].r * 255;\r\n colorTable[j++] = colors[i].g * 255;\r\n colorTable[j++] = colors[i].b * 255;\r\n colorTable[j++] = 255;\r\n }\r\n\r\n return colorTable;\r\n }\r\n\r\n /**\r\n * Creates a RawTexture from an RGBA color array and sets it on the plugin material instance.\r\n * @param name name of the texture\r\n * @param colors Uint8Array of colors\r\n */\r\n private _createColorsTexture(name: string, colors: Color3[]) {\r\n const colorsArray = GreasedLinePluginMaterial._Color3toRGBAUint8(colors);\r\n this._colorsTexture = new RawTexture(colorsArray, colors.length, 1, Engine.TEXTUREFORMAT_RGBA, this._scene, false, true, this.colorsSampling);\r\n this._colorsTexture.name = name;\r\n }\r\n\r\n /**\r\n * Disposes the plugin material.\r\n */\r\n public dispose(): void {\r\n this._colorsTexture?.dispose();\r\n super.dispose();\r\n }\r\n\r\n /**\r\n * Returns the colors used to colorize the line\r\n */\r\n get colors() {\r\n return this._colors;\r\n }\r\n\r\n /**\r\n * Sets the colors used to colorize the line\r\n */\r\n set colors(value: Nullable) {\r\n this.setColors(value);\r\n }\r\n\r\n /**\r\n * Creates or updates the colors texture\r\n * @param colors color table RGBA\r\n * @param lazy if lazy, the colors are not updated\r\n * @param forceUpdate force creation of a new texture\r\n * @returns\r\n */\r\n public setColors(colors: Nullable, lazy = false, forceUpdate = false): void {\r\n const origColorsCount = this._colors?.length ?? 0;\r\n\r\n this._colors = colors;\r\n\r\n if (colors === null || colors.length === 0) {\r\n this._colorsTexture?.dispose();\r\n return;\r\n }\r\n\r\n if (lazy && !forceUpdate) {\r\n return;\r\n }\r\n\r\n if (this._colorsTexture && origColorsCount === colors.length && !forceUpdate) {\r\n const colorArray = GreasedLinePluginMaterial._Color3toRGBAUint8(colors);\r\n this._colorsTexture.update(colorArray);\r\n } else {\r\n this._colorsTexture?.dispose();\r\n this._createColorsTexture(`${this._material.name}-colors-texture`, colors);\r\n }\r\n }\r\n\r\n /**\r\n * Updates the material. Use when material created in lazy mode.\r\n */\r\n public updateLazy() {\r\n if (this._colors) {\r\n this.setColors(this._colors, false, true);\r\n }\r\n }\r\n\r\n /**\r\n * Gets the number of dashes in the line\r\n */\r\n get dashCount() {\r\n return this._dashCount;\r\n }\r\n /**\r\n * Sets the number of dashes in the line\r\n * @param value dash\r\n */\r\n set dashCount(value: number) {\r\n this._dashCount = value;\r\n this._dashArray = 1 / value;\r\n }\r\n\r\n /**\r\n * False means 1 unit in width = 1 unit on scene, true means 1 unit in width is reduced on the screen to make better looking lines\r\n */\r\n get sizeAttenuation() {\r\n return this._sizeAttenuation;\r\n }\r\n\r\n /**\r\n * Turn on/off attenuation of the width option and widths array.\r\n * @param value false means 1 unit in width = 1 unit on scene, true means 1 unit in width is reduced on the screen to make better looking lines\r\n */\r\n set sizeAttenuation(value: boolean) {\r\n this._sizeAttenuation = value;\r\n this.markAllDefinesAsDirty();\r\n }\r\n\r\n /**\r\n * Gets the color of the line\r\n */\r\n get color() {\r\n return this.color;\r\n }\r\n\r\n /**\r\n * Sets the color of the line\r\n * @param value Color3 or null to clear the color. You need to clear the color if you use colors and useColors = true\r\n */\r\n set color(value: Nullable) {\r\n this.setColor(value);\r\n }\r\n\r\n /**\r\n * Sets the color of the line. If set the whole line will be mixed with this color according to the colorMode option.\r\n * @param value color\r\n */\r\n public setColor(value: Nullable, doNotMarkDirty = false) {\r\n if ((this._color === null && value !== null) || (this._color !== null && value === null)) {\r\n this._color = value;\r\n !doNotMarkDirty && this.markAllDefinesAsDirty();\r\n } else {\r\n this._color = value;\r\n }\r\n }\r\n\r\n /**\r\n * Gets the color distributiopn type\r\n */\r\n get colorsDistributionType() {\r\n return this._colorsDistributionType;\r\n }\r\n\r\n /**\r\n * Sets the color distribution type\r\n * @see GreasedLineMeshColorDistributionType\r\n * @param value color distribution type\r\n */\r\n set colorsDistributionType(value: GreasedLineMeshColorDistributionType) {\r\n this._colorsDistributionType = value;\r\n this.markAllDefinesAsDirty();\r\n }\r\n\r\n /**\r\n * Gets the resolution\r\n */\r\n get resolution() {\r\n return this._resolution;\r\n }\r\n\r\n /**\r\n * Sets the resolution\r\n * @param value resolution of the screen for GreasedLine\r\n */\r\n set resolution(value: Vector2) {\r\n this._aspect = value.x / value.y;\r\n this._resolution = value;\r\n }\r\n\r\n /**\r\n * Serializes this plugin material\r\n * @returns serializationObjec\r\n */\r\n public serialize(): any {\r\n const serializationObject = super.serialize();\r\n\r\n const greasedLineMaterialOptions: GreasedLineMaterialOptions = {\r\n colorDistributionType: this._colorsDistributionType,\r\n colorsSampling: this.colorsSampling,\r\n colorMode: this.colorMode,\r\n dashCount: this._dashCount,\r\n dashOffset: this.dashOffset,\r\n dashRatio: this.dashRatio,\r\n resolution: this._resolution,\r\n sizeAttenuation: this._sizeAttenuation,\r\n useColors: this.useColors,\r\n useDash: this.useDash,\r\n visibility: this.visibility,\r\n width: this.width,\r\n };\r\n\r\n this._colors && (greasedLineMaterialOptions.colors = this._colors);\r\n this._color && (greasedLineMaterialOptions.color = this._color);\r\n\r\n serializationObject.greasedLineMaterialOptions = greasedLineMaterialOptions;\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Parses a serialized objects\r\n * @param source serialized object\r\n * @param scene scene\r\n * @param rootUrl root url for textures\r\n */\r\n public parse(source: any, scene: Scene, rootUrl: string): void {\r\n super.parse(source, scene, rootUrl);\r\n const greasedLineMaterialOptions = source.greasedLineMaterialOptions;\r\n\r\n this._colorsTexture?.dispose();\r\n\r\n if (greasedLineMaterialOptions.colors) {\r\n this._createColorsTexture(`${this._material.name}-colors-texture`, greasedLineMaterialOptions.colors);\r\n } else {\r\n GreasedLinePluginMaterial._PrepareEmptyColorsTexture(scene);\r\n }\r\n\r\n greasedLineMaterialOptions.color && this.setColor(greasedLineMaterialOptions.color, true);\r\n greasedLineMaterialOptions.colorDistributionType && (this.colorsDistributionType = greasedLineMaterialOptions.colorDistributionType);\r\n greasedLineMaterialOptions.colorsSampling && (this.colorsSampling = greasedLineMaterialOptions.colorsSampling);\r\n greasedLineMaterialOptions.colorMode && (this.colorMode = greasedLineMaterialOptions.colorMode);\r\n greasedLineMaterialOptions.useColors && (this.useColors = greasedLineMaterialOptions.useColors);\r\n greasedLineMaterialOptions.visibility && (this.visibility = greasedLineMaterialOptions.visibility);\r\n greasedLineMaterialOptions.useDash && (this.useDash = greasedLineMaterialOptions.useDash);\r\n greasedLineMaterialOptions.dashCount && (this.dashCount = greasedLineMaterialOptions.dashCount);\r\n greasedLineMaterialOptions.dashRatio && (this.dashRatio = greasedLineMaterialOptions.dashRatio);\r\n greasedLineMaterialOptions.dashOffset && (this.dashOffset = greasedLineMaterialOptions.dashOffset);\r\n greasedLineMaterialOptions.width && (this.width = greasedLineMaterialOptions.width);\r\n greasedLineMaterialOptions.sizeAttenuation && (this.sizeAttenuation = greasedLineMaterialOptions.sizeAttenuation);\r\n greasedLineMaterialOptions.resolution && (this.resolution = greasedLineMaterialOptions.resolution);\r\n\r\n this.markAllDefinesAsDirty();\r\n }\r\n\r\n /**\r\n * A minimum size texture for the colors sampler2D when there is no colors texture defined yet.\r\n * For fast switching using the useColors property without the need to use defines.\r\n * @param scene Scene\r\n */\r\n private static _PrepareEmptyColorsTexture(scene: Scene) {\r\n if (!this._EmptyColorsTexture) {\r\n const colorsArray = new Uint8Array(4);\r\n this._EmptyColorsTexture = new RawTexture(colorsArray, 1, 1, Engine.TEXTUREFORMAT_RGBA, scene, false, false, RawTexture.NEAREST_NEAREST);\r\n this._EmptyColorsTexture.name = \"grlEmptyColorsTexture\";\r\n }\r\n }\r\n}\r\n\r\nRegisterClass(`BABYLON.${GreasedLinePluginMaterial.GREASED_LINE_MATERIAL_NAME}`, GreasedLinePluginMaterial);\r\n","import type { GreasedLineMaterialOptions } from \"../../Materials/greasedLinePluginMaterial\";\r\nimport { GreasedLineMeshMaterialType, GreasedLinePluginMaterial } from \"../../Materials/greasedLinePluginMaterial\";\r\nimport { StandardMaterial } from \"./../../Materials/standardMaterial\";\r\nimport { PBRMaterial } from \"../../Materials/PBR/pbrMaterial\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { GreasedLineMeshOptions } from \"../greasedLineMesh\";\r\nimport { GreasedLineMesh } from \"../greasedLineMesh\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { EngineStore } from \"../../Engines/engineStore\";\r\nimport type { Color3 } from \"core/Maths/math.color\";\r\n\r\n/**\r\n * How are the colors distributed along the color table\r\n * {@link https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param/greased_line#colors-and-colordistribution}\r\n */\r\nexport enum GreasedLineMeshColorDistribution {\r\n /**\r\n * Do no modify the color table\r\n */\r\n COLOR_DISTRIBUTION_NONE = 0,\r\n /**\r\n * Repeat the colors until the color table is full\r\n */\r\n COLOR_DISTRIBUTION_REPEAT = 1,\r\n /**\r\n * Distribute the colors evenly through the color table\r\n */\r\n COLOR_DISTRIBUTION_EVEN = 2,\r\n /**\r\n * Put the colors to start of the color table a fill the rest with the default color\r\n */\r\n COLOR_DISTRIBUTION_START = 3,\r\n /**\r\n * Put the colors to the end of the color table and fill the rest with the default color\r\n */\r\n COLOR_DISTRIBUTION_END = 4,\r\n /**\r\n * Put the colors to start and to the end of the color table and fill the gap between with the default color\r\n */\r\n COLOR_DISTRIBUTION_START_END = 5,\r\n}\r\n\r\n/**\r\n * How are the widths distributed along the width table\r\n * {@link https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param/greased_line#widths-and-widthdistribution}\r\n */\r\nexport enum GreasedLineMeshWidthDistribution {\r\n /**\r\n * Do no modify the width table\r\n */\r\n WIDTH_DISTRIBUTION_NONE = 0,\r\n /**\r\n * Repeat the widths until the width table is full\r\n */\r\n WIDTH_DISTRIBUTION_REPEAT = 1,\r\n /**\r\n * Distribute the widths evenly through the width table\r\n */\r\n WIDTH_DISTRIBUTION_EVEN = 2,\r\n /**\r\n * Put the widths to start of the width table a fill the rest with the default width\r\n */\r\n WIDTH_DISTRIBUTION_START = 3,\r\n /**\r\n * Put the widths to the end of the width table and fill the rest with the default width\r\n */\r\n WIDTH_DISTRIBUTION_END = 4,\r\n /**\r\n * Put the widths to start and to the end of the width table and fill the gap between with the default width\r\n */\r\n WIDTH_DISTRIBUTION_START_END = 5,\r\n}\r\n\r\n/**\r\n * Material options for GreasedLineBuilder\r\n */\r\nexport interface GreasedLineMaterialBuilderOptions extends GreasedLineMaterialOptions {\r\n /**\r\n * If set to true a new material will be created and a new material plugin will be attached\r\n * to the material. The material will be set on the mesh. If the instance option is specified in the mesh options,\r\n * no material will be created/assigned. Defaults to true.\r\n */\r\n createAndAssignMaterial?: boolean;\r\n /**\r\n * Distribution of the colors if the color table contains fewer entries than needed. Defaults to GreasedLineMeshColorDistribution.COLOR_DISTRIBUTION_START\r\n * @see CompleteGreasedLineColorTable\r\n */\r\n colorDistribution?: GreasedLineMeshColorDistribution;\r\n}\r\n\r\n/**\r\n * Line mesh options for GreasedLineBuilder\r\n */\r\nexport interface GreasedLineMeshBuilderOptions extends GreasedLineMeshOptions {\r\n /**\r\n * Distribution of the widths if the width table contains fewer entries than needed. Defaults to GreasedLineMeshWidthDistribution.WIDTH_DISTRIBUTION_START\r\n * @see CompleteGreasedLineWidthTable\r\n */\r\n widthDistribution?: GreasedLineMeshWidthDistribution;\r\n}\r\n\r\n/**\r\n * Builder class for create GreasedLineMeshes\r\n */\r\n\r\n/**\r\n * Creates a new @see GreasedLinePluginMaterial\r\n * @param name name of the material\r\n * @param options material options @see GreasedLineMaterialOptions\r\n * @param scene scene or null to use the last scene\r\n * @returns StandardMaterial or PBRMaterial with the @see GreasedLinePluginMaterial attached to it\r\n */\r\nexport function CreateGreasedLineMaterial(name: string, options: GreasedLineMaterialOptions, scene: Nullable) {\r\n scene = (scene ?? EngineStore.LastCreatedScene);\r\n\r\n const material = options.materialType === GreasedLineMeshMaterialType.MATERIAL_TYPE_PBR ? new PBRMaterial(name, scene) : new StandardMaterial(name, scene);\r\n new GreasedLinePluginMaterial(material, scene, options);\r\n\r\n return material;\r\n}\r\n/**\r\n * Creates a GreasedLine mesh\r\n * @param name name of the mesh\r\n * @param options options for the mesh\r\n * @param materialOptions material options for the mesh\r\n * @param scene scene where the mesh will be created\r\n * @returns instance of GreasedLineMesh\r\n */\r\nexport function CreateGreasedLine(name: string, options: GreasedLineMeshBuilderOptions, materialOptions?: Nullable, scene?: Nullable) {\r\n scene = (scene ?? EngineStore.LastCreatedScene);\r\n\r\n let instance;\r\n const allPoints = GreasedLineMesh.ConvertPoints(options.points);\r\n\r\n let length = 0;\r\n if (Array.isArray(allPoints[0])) {\r\n allPoints.forEach((points) => {\r\n length += points.length / 3;\r\n });\r\n }\r\n\r\n options.widthDistribution = options.widthDistribution ?? GreasedLineMeshWidthDistribution.WIDTH_DISTRIBUTION_START;\r\n\r\n materialOptions = materialOptions ?? {\r\n color: GreasedLinePluginMaterial.DEFAULT_COLOR,\r\n };\r\n materialOptions.createAndAssignMaterial = materialOptions.createAndAssignMaterial ?? true;\r\n materialOptions.colorDistribution = materialOptions?.colorDistribution ?? GreasedLineMeshColorDistribution.COLOR_DISTRIBUTION_START;\r\n\r\n const widths = CompleteGreasedLineWidthTable(length, options.widths ?? [], options.widthDistribution);\r\n\r\n const colors = materialOptions?.colors\r\n ? CompleteGreasedLineColorTable(length, materialOptions.colors, materialOptions.colorDistribution, materialOptions.color ?? GreasedLinePluginMaterial.DEFAULT_COLOR)\r\n : undefined;\r\n\r\n // create new mesh if instance is not defined\r\n if (!options.instance) {\r\n const initialGreasedLineOptions: GreasedLineMeshOptions = {\r\n points: allPoints,\r\n updatable: options.updatable,\r\n widths,\r\n lazy: options.lazy,\r\n };\r\n\r\n instance = new GreasedLineMesh(name, scene, initialGreasedLineOptions);\r\n\r\n if (materialOptions) {\r\n const initialMaterialOptions: GreasedLineMaterialOptions = {\r\n materialType: materialOptions.materialType,\r\n dashCount: materialOptions.dashCount,\r\n dashOffset: materialOptions.dashOffset,\r\n dashRatio: materialOptions.dashRatio,\r\n resolution: materialOptions.resolution,\r\n sizeAttenuation: materialOptions.sizeAttenuation,\r\n useColors: materialOptions.useColors,\r\n useDash: materialOptions.useDash,\r\n visibility: materialOptions.visibility,\r\n width: materialOptions.width,\r\n color: materialOptions.color,\r\n colorMode: materialOptions.colorMode,\r\n colorsSampling: materialOptions.colorsSampling,\r\n colorDistributionType: materialOptions.colorDistributionType,\r\n colors,\r\n };\r\n\r\n if (materialOptions.createAndAssignMaterial) {\r\n const material = materialOptions.materialType === GreasedLineMeshMaterialType.MATERIAL_TYPE_PBR ? new PBRMaterial(name, scene) : new StandardMaterial(name, scene);\r\n new GreasedLinePluginMaterial(material, scene, initialMaterialOptions);\r\n instance.material = material;\r\n }\r\n }\r\n } else {\r\n // update the data on the mesh instance\r\n instance = options.instance;\r\n const currentWidths = instance.widths;\r\n\r\n if (currentWidths) {\r\n const newWidths = currentWidths.slice();\r\n for (const w of widths) {\r\n newWidths.push(w);\r\n }\r\n instance.widths = newWidths;\r\n } else {\r\n instance.widths = widths;\r\n }\r\n instance.addPoints(allPoints);\r\n }\r\n\r\n // add colors\r\n // it will merge if any colors already on the instance\r\n if (colors && options.instance) {\r\n if (options.instance.material instanceof StandardMaterial || instance.material instanceof PBRMaterial) {\r\n if (options.instance.greasedLineMaterial) {\r\n const currentColors = options.instance.greasedLineMaterial.colors;\r\n if (currentColors) {\r\n const newColors = currentColors.concat(colors);\r\n options.instance.greasedLineMaterial.setColors(newColors, instance.isLazy());\r\n }\r\n }\r\n }\r\n }\r\n\r\n return instance;\r\n}\r\n\r\n/**\r\n * Completes the width table/fills the missing entries. It means it creates a width entry for every point of the line mesh.\r\n * You can provide more points the widths when creating the mesh. This function will fill the empty entries.\r\n * The algorithm used to fill the empty entries can be\r\n * GreasedLineMeshWidthDistribution.REPEAT - the width table will be repeatedly copied to the empty values [wL, wU] = [wL, wU, wL, wU, wL, wU, wL, wU, ...]\r\n * GreasedLineMeshWidthDistribution.EVEN - the width table will be evenly copied to the empty values [wL, wU] = [wL, wL, wL, wL, wU, wU, wU, wU]\r\n * GreasedLineMeshWidthDistribution.START - the width table will be copied at the start of the empty values\r\n * and rest will be filled width the default width upper and default width lower values [wU, wL] = [wL, wU, dwL, dwU, dwL, dwU, dwL, dwU]\r\n * GreasedLineMeshWidthDistribution.END - the width table will be copied at the end of the empty values\r\n * and rest will be filled width the default values [wL, wU] = [wL, wU, dwL, dwU, dwL, dwU, wL, wU]\r\n * @param pointCount number of points of the line mesh\r\n * @param widths array of widths [widhtLower, widthUpper, widthLower, widthUpper ...]. Two widths (lower/upper) per point.\r\n * @param widthsDistribution how to distribute widths if the widths array has fewer entries than pointCount\r\n * @param defaultWidthUpper the default value which will be used to fill empty width entries - upper width\r\n * @param defaultWidthLower the default value which will be used to fill empty width entries - lower width\r\n * @returns completed width table.\r\n */\r\nexport function CompleteGreasedLineWidthTable(\r\n pointCount: number,\r\n widths: number[],\r\n widthsDistribution: GreasedLineMeshWidthDistribution,\r\n defaultWidthUpper = 1,\r\n defaultWidthLower = 1\r\n): number[] {\r\n const missingCount = pointCount - widths.length / 2;\r\n\r\n const widthsData: number[] = [];\r\n if (missingCount < 0) {\r\n return widths.slice(0, pointCount * 2);\r\n }\r\n\r\n // is the width table shorter than the point table?\r\n if (missingCount > 0) {\r\n // it is, fill in the missing elements\r\n if (widthsDistribution === GreasedLineMeshWidthDistribution.WIDTH_DISTRIBUTION_START_END) {\r\n const halfCount = Math.floor(widths.length / 2);\r\n\r\n // start sector\r\n for (let i = 0, j = 0; i < halfCount - 1; i++) {\r\n widthsData.push(widths[j++]);\r\n widthsData.push(widths[j++]);\r\n }\r\n\r\n // middle sector\r\n const widthL = widths[halfCount / 2];\r\n const widthU = widths[halfCount / 2 + 1];\r\n for (let i = 0; i < missingCount; i++) {\r\n widthsData.push(widthU);\r\n widthsData.push(widthL);\r\n }\r\n\r\n // end sector\r\n for (let i = halfCount; i < widths.length; i += 2) {\r\n widthsData.push(widths[i]);\r\n widthsData.push(widths[i + 1]);\r\n }\r\n } else if (widthsDistribution === GreasedLineMeshWidthDistribution.WIDTH_DISTRIBUTION_START) {\r\n // start sector\r\n for (let i = 0; i < widths.length; i += 2) {\r\n widthsData.push(widths[i]);\r\n widthsData.push(widths[i + 1]);\r\n }\r\n\r\n // end sector\r\n for (let i = 0; i < missingCount; i++) {\r\n widthsData.push(defaultWidthUpper);\r\n widthsData.push(defaultWidthLower);\r\n }\r\n } else if (widthsDistribution === GreasedLineMeshWidthDistribution.WIDTH_DISTRIBUTION_END) {\r\n // start sector\r\n for (let i = 0; i < missingCount; i++) {\r\n widthsData.push(defaultWidthUpper);\r\n widthsData.push(defaultWidthLower);\r\n }\r\n\r\n // end sector\r\n for (let i = 0; i < widths.length; i += 2) {\r\n widthsData.push(widths[i]);\r\n widthsData.push(widths[i + 1]);\r\n }\r\n } else if (widthsDistribution === GreasedLineMeshWidthDistribution.WIDTH_DISTRIBUTION_REPEAT) {\r\n let i = 0;\r\n for (let x = 0; x < pointCount; x++) {\r\n widthsData.push(widths[i++]);\r\n widthsData.push(widths[i++]);\r\n\r\n if (i === widths.length) {\r\n i = 0;\r\n }\r\n }\r\n } else if (widthsDistribution === GreasedLineMeshWidthDistribution.WIDTH_DISTRIBUTION_EVEN) {\r\n let j = 0;\r\n const widthsectorLength = widths.length / ((pointCount - 1) * 2);\r\n for (let x = 0; x < pointCount; x++) {\r\n const i = Math.floor(j);\r\n\r\n widthsData.push(widths[i]);\r\n widthsData.push(widths[i + 1]);\r\n\r\n j += widthsectorLength;\r\n }\r\n }\r\n } else {\r\n for (let i = 0; i < widths.length; i++) {\r\n widthsData.push(widths[i]);\r\n }\r\n }\r\n\r\n return widthsData;\r\n}\r\n\r\n/**\r\n * Completes the color table/fill the missing color entries. It means it creates a color entry for every point of the line mesh.\r\n * You can provide more points the colors when creating the mesh. This function will fill the empty entries.\r\n * The algorithm used to fill the empty entries can be\r\n * GreasedLineMesColorhDistribution.REPEAT - the color table will be repeatedly copied to the empty values [c1, c2] = [c1, c2, c1, c2, c1, c2, c1, c2]\r\n * GreasedLineMesColorhDistribution.EVEN - the color table will be evenly copied to the empty values [c1, c2] = [c1, c1, c1, c1, c2, c2, c2, c2]\r\n * GreasedLineMesColorhDistribution.START - the color table will be copied at the start of the empty values\r\n * and rest will be filled color the default color value [c1, c2] = [c1, c2, dc, dc, dc, dc, dc, dc]\r\n * GreasedLineMesColorhDistribution.START_END - the color table will be copied at the start and the end of the empty values\r\n * and rest will be filled color the default color value [c1, c2] = [c1, c2, dc, dc, dc, dc, c1, c2]\r\n * @param pointCount number of points of the line mesh\r\n * @param colors array of Color3 for the color table\r\n * @param colorDistribution how to distribute colors if the colors array has fewer entries than pointCount\r\n * @param defaultColor default color to be used to fill empty entries in the color table\r\n * @returns completed array of Color3s\r\n */\r\nexport function CompleteGreasedLineColorTable(pointCount: number, colors: Color3[], colorDistribution: GreasedLineMeshColorDistribution, defaultColor: Color3): Color3[] {\r\n const missingCount = pointCount - colors.length;\r\n if (missingCount < 0) {\r\n return colors.slice(0, pointCount);\r\n }\r\n\r\n const colorsData: Color3[] = [];\r\n // is the color table shorter than the point table?\r\n if (missingCount > 0) {\r\n // it is, fill in the missing elements\r\n if (colorDistribution === GreasedLineMeshColorDistribution.COLOR_DISTRIBUTION_START_END) {\r\n const halfCount = Math.floor(colors.length / 2);\r\n\r\n // start sector\r\n for (let i = 0; i < halfCount; i++) {\r\n colorsData.push(colors[i]);\r\n }\r\n\r\n // middle sector\r\n for (let i = 0; i < missingCount - 1; i++) {\r\n colorsData.push(defaultColor);\r\n }\r\n\r\n // end sector\r\n for (let i = halfCount; i < colors.length; i++) {\r\n colorsData.push(colors[i]);\r\n }\r\n } else if (colorDistribution === GreasedLineMeshColorDistribution.COLOR_DISTRIBUTION_START) {\r\n // start sector\r\n for (let i = 0; i < colors.length; i++) {\r\n colorsData.push(colors[i]);\r\n }\r\n\r\n // end sector\r\n for (let i = 0; i < missingCount; i++) {\r\n colorsData.push(defaultColor);\r\n }\r\n } else if (colorDistribution === GreasedLineMeshColorDistribution.COLOR_DISTRIBUTION_END) {\r\n // start sector\r\n for (let i = 0; i < missingCount - 1; i++) {\r\n colorsData.push(defaultColor);\r\n }\r\n\r\n // end sector\r\n for (let i = 0; i < colors.length; i++) {\r\n colorsData.push(colors[i]);\r\n }\r\n } else if (colorDistribution === GreasedLineMeshColorDistribution.COLOR_DISTRIBUTION_REPEAT) {\r\n let i = 0;\r\n for (let x = 0; x < pointCount; x++) {\r\n colorsData.push(colors[i]);\r\n\r\n i++;\r\n\r\n if (i === colors.length) {\r\n i = 0;\r\n }\r\n }\r\n } else if (colorDistribution === GreasedLineMeshColorDistribution.COLOR_DISTRIBUTION_EVEN) {\r\n let j = 0;\r\n const colorSectorLength = colors.length / (pointCount - 1);\r\n for (let x = 0; x < pointCount - 1; x++) {\r\n const i = Math.floor(j);\r\n\r\n colorsData.push(colors[i]);\r\n\r\n j += colorSectorLength;\r\n }\r\n } else if (colorDistribution === GreasedLineMeshColorDistribution.COLOR_DISTRIBUTION_NONE) {\r\n for (let i = 0; i < colors.length; i++) {\r\n colorsData.push(colors[i]);\r\n }\r\n }\r\n } else {\r\n for (let i = 0; i < pointCount; i++) {\r\n colorsData.push(colors[i]);\r\n }\r\n }\r\n\r\n return colorsData;\r\n}\r\n``;\r\n","import { Mesh } from \"../mesh\";\r\nimport type { Scene } from \"../../scene\";\r\nimport { CreateSphere } from \"../Builders/sphereBuilder\";\r\nimport { CreateDisc } from \"./discBuilder\";\r\n\r\n/**\r\n * Creates a hemisphere mesh\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param options.segments\r\n * @param options.diameter\r\n * @param options.sideOrientation\r\n * @param scene defines the hosting scene\r\n * @returns the hemisphere mesh\r\n */\r\nexport function CreateHemisphere(name: string, options: { segments?: number; diameter?: number; sideOrientation?: number } = {}, scene?: Scene): Mesh {\r\n if (!options.diameter) {\r\n options.diameter = 1;\r\n }\r\n if (!options.segments) {\r\n options.segments = 16;\r\n }\r\n\r\n const halfSphere = CreateSphere(\"\", { slice: 0.5, diameter: options.diameter, segments: options.segments }, scene);\r\n const disc = CreateDisc(\"\", { radius: options.diameter / 2, tessellation: options.segments * 3 + (4 - options.segments) }, scene);\r\n disc.rotation.x = -Math.PI / 2;\r\n disc.parent = halfSphere;\r\n\r\n const merged = Mesh.MergeMeshes([disc, halfSphere], true);\r\n merged.name = name;\r\n\r\n return merged;\r\n}\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated use the function directly from the module\r\n */\r\nexport const HemisphereBuilder = {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreateHemisphere,\r\n};\r\n\r\n/**\r\n * Creates a hemispheric light\r\n * @param name\r\n * @param segments\r\n * @param diameter\r\n * @param scene\r\n */\r\nMesh.CreateHemisphere = (name: string, segments: number, diameter: number, scene?: Scene): Mesh => {\r\n const options = {\r\n segments: segments,\r\n diameter: diameter,\r\n };\r\n\r\n return CreateHemisphere(name, options, scene);\r\n};\r\n","import { Curve3 } from \"../Maths/math.path\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport { TmpVectors, Vector3 } from \"../Maths/math.vector\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { IFontData } from \"../Meshes/Builders/textBuilder\";\r\nimport { CreateTextShapePaths } from \"../Meshes/Builders/textBuilder\";\r\nimport type { FloatArray, IndicesArray } from \"../types\";\r\n\r\n/**\r\n * Tool functions for GreasedLine\r\n */\r\nexport class GreasedLineTools {\r\n /**\r\n * Omit zero length lines predicate for the MeshesToLines function\r\n * @param p1 point1 position of the face\r\n * @param p2 point2 position of the face\r\n * @param p3 point3 position of the face\r\n * @returns\r\n */\r\n public static OmitZeroLengthPredicate(p1: Vector3, p2: Vector3, p3: Vector3) {\r\n return p1.lengthSquared() === 0 && p2.lengthSquared() === 0 && p3.lengthSquared() === 0;\r\n }\r\n /**\r\n * Gets mesh triangles as line positions\r\n * @param meshes array of meshes\r\n * @param omitZeroLengthLines do not generate a line when the distance if the vertices in the triangle equals to zero\r\n * @returns array of arrays of points\r\n */\r\n public static MeshesToLines(\r\n meshes: AbstractMesh[],\r\n predicate?: (\r\n p1: Vector3,\r\n p2: Vector3,\r\n p3: Vector3,\r\n indiceIndex: number,\r\n vertexIndex: number,\r\n mesh: AbstractMesh,\r\n meshIndex: number,\r\n vertices: FloatArray,\r\n indices: IndicesArray\r\n ) => Vector3[]\r\n ) {\r\n const points: Vector3[][] = [];\r\n\r\n meshes.forEach((m, meshIndex) => {\r\n const vertices = m.getVerticesData(VertexBuffer.PositionKind);\r\n const indices = m.getIndices();\r\n if (vertices && indices) {\r\n for (let i = 0, ii = 0; i < indices.length; i++) {\r\n const vi1 = indices[ii++] * 3;\r\n const vi2 = indices[ii++] * 3;\r\n const vi3 = indices[ii++] * 3;\r\n\r\n const p1 = new Vector3(vertices[vi1], vertices[vi1 + 1], vertices[vi1 + 2]);\r\n const p2 = new Vector3(vertices[vi2], vertices[vi2 + 1], vertices[vi2 + 2]);\r\n const p3 = new Vector3(vertices[vi3], vertices[vi3 + 1], vertices[vi3 + 2]);\r\n\r\n if (predicate) {\r\n const pointsFromPredicate = predicate(p1, p2, p3, i, vi1, m, meshIndex, vertices, indices);\r\n pointsFromPredicate && points.push(pointsFromPredicate);\r\n } else {\r\n points.push([p1, p2, p3, p1]);\r\n }\r\n }\r\n }\r\n });\r\n\r\n return points;\r\n }\r\n\r\n /**\r\n * Converts number coordinates to Vector3s\r\n * @param points number array of x, y, z, x, y z, ... coordinates\r\n * @returns Vector3 array\r\n */\r\n public static ToVector3Array(points: number[]) {\r\n const array: Vector3[] = [];\r\n for (let i = 0; i < points.length; i += 3) {\r\n array.push(new Vector3(points[i], points[i + 1], points[i + 2]));\r\n }\r\n return array;\r\n }\r\n\r\n /**\r\n * Gets a number array from a Vector3 array.\r\n * You can you for example to convert your Vector3[] offsets to the required number[] for the offsets option.\r\n * @param points Vector3 array\r\n * @returns an array of x, y, z coordinates as numbers [x, y, z, x, y, z, x, y, z, ....]\r\n */\r\n public static ToNumberArray(points: Vector3[]) {\r\n return points.flatMap((v) => [v.x, v.y, v.z]);\r\n }\r\n\r\n /**\r\n * Calculates the sum of points of every line and the number of points in each line.\r\n * This function is useful when you are drawing multiple lines in one mesh and you want\r\n * to know the counts. For example for creating an offsets table.\r\n * @param points point array\r\n * @returns points count info\r\n */\r\n public static GetPointsCountInfo(points: number[][]): { total: number; counts: number[] } {\r\n const counts = new Array(points.length);\r\n let total = 0;\r\n for (let n = points.length; n--; ) {\r\n counts[n] = points[n].length / 3;\r\n total += counts[n];\r\n }\r\n return { total, counts };\r\n }\r\n\r\n /**\r\n * Gets the length of the line counting all it's segments length\r\n * @param data array of line points\r\n * @returns length of the line\r\n */\r\n public static GetLineLength(data: Vector3[] | number[]): number {\r\n if (data.length === 0) {\r\n return 0;\r\n }\r\n\r\n let points: Vector3[];\r\n if (typeof data[0] === \"number\") {\r\n points = GreasedLineTools.ToVector3Array(data);\r\n } else {\r\n points = data;\r\n }\r\n\r\n const tmp = TmpVectors.Vector3[0];\r\n let length = 0;\r\n for (let index = 0; index < points.length - 1; index++) {\r\n const point1 = points[index];\r\n const point2 = points[index + 1];\r\n length += point2.subtractToRef(point1, tmp).length();\r\n }\r\n return length;\r\n }\r\n\r\n /**\r\n * Divides a segment into smaller segments.\r\n * A segment is a part of the line between it's two points.\r\n * @param point1 first point of the line\r\n * @param point2 second point of the line\r\n * @param segmentCount number of segments we want to have in the divided line\r\n * @returns\r\n */\r\n public static SegmentizeSegmentByCount(point1: Vector3, point2: Vector3, segmentCount: number): Vector3[] {\r\n const dividedLinePoints: Vector3[] = [];\r\n const diff = point2.subtract(point1);\r\n const divisor = TmpVectors.Vector3[0];\r\n divisor.setAll(segmentCount);\r\n const segmentVector = TmpVectors.Vector3[1];\r\n diff.divideToRef(divisor, segmentVector);\r\n\r\n let nextPoint = point1.clone();\r\n dividedLinePoints.push(nextPoint);\r\n for (let index = 0; index < segmentCount; index++) {\r\n nextPoint = nextPoint.clone();\r\n dividedLinePoints.push(nextPoint.addInPlace(segmentVector));\r\n }\r\n\r\n return dividedLinePoints;\r\n }\r\n\r\n /**\r\n * Divides a line into segments.\r\n * A segment is a part of the line between it's two points.\r\n * @param what line points\r\n * @param segmentLength length of each segment of the resulting line (distance between two line points)\r\n * @returns line point\r\n */\r\n public static SegmentizeLineBySegmentLength(what: Vector3[] | number[] | { point1: Vector3; point2: Vector3; length: number }[], segmentLength: number): Vector3[] {\r\n const subLines = what[0] instanceof Vector3 ? GreasedLineTools.GetLineSegments(what as Vector3[]) : (what as { point1: Vector3; point2: Vector3; length: number }[]);\r\n const points: Vector3[] = [];\r\n subLines.forEach((s) => {\r\n if (s.length > segmentLength) {\r\n const segments = GreasedLineTools.SegmentizeSegmentByCount(s.point1, s.point2, Math.ceil(s.length / segmentLength));\r\n segments.forEach((seg) => {\r\n points.push(seg);\r\n });\r\n } else {\r\n points.push(s.point1);\r\n points.push(s.point2);\r\n }\r\n });\r\n return points;\r\n }\r\n\r\n /**\r\n * Divides a line into segments.\r\n * A segment is a part of the line between it's two points.\r\n * @param what line points\r\n * @param segmentCount number of segments\r\n * @returns line point\r\n */\r\n public static SegmentizeLineBySegmentCount(what: Vector3[], segmentCount: number): Vector3[] {\r\n const segmentLength = GreasedLineTools.GetLineLength(what) / segmentCount;\r\n return GreasedLineTools.SegmentizeLineBySegmentLength(what, segmentLength);\r\n }\r\n /**\r\n * Gets line segments.\r\n * A segment is a part of the line between it's two points.\r\n * @param points line points\r\n * @returns segments information of the line segment including starting point, ending point and the distance between them\r\n */\r\n public static GetLineSegments(points: Vector3[]): { point1: Vector3; point2: Vector3; length: number }[] {\r\n const segments = [];\r\n for (let index = 0; index < points.length - 1; index++) {\r\n const point1 = points[index];\r\n const point2 = points[index + 1];\r\n const length = point2.subtract(point1).length();\r\n segments.push({ point1, point2, length });\r\n }\r\n\r\n return segments;\r\n }\r\n\r\n /**\r\n * Gets the minimum and the maximum length of a line segment in the line.\r\n * A segment is a part of the line between it's two points.\r\n * @param points line points\r\n * @returns\r\n */\r\n public static GetMinMaxSegmentLength(points: Vector3[]): { min: number; max: number } {\r\n const subLines = GreasedLineTools.GetLineSegments(points);\r\n const sorted = subLines.sort((s) => s.length);\r\n return {\r\n min: sorted[0].length,\r\n max: sorted[sorted.length - 1].length,\r\n };\r\n }\r\n\r\n /**\r\n * Finds the last visible position in world space of the line according to the visibility parameter\r\n * @param lineSegments segments of the line\r\n * @param lineLength total length of the line\r\n * @param visbility normalized value of visibility\r\n * @returns world space coordinate of the last visible piece of the line\r\n */\r\n public static GetPositionOnLineByVisibility(lineSegments: { point1: Vector3; point2: Vector3; length: number }[], lineLength: number, visbility: number, localSpace = false) {\r\n const lengthVisibilityRatio = lineLength * visbility;\r\n let sumSegmentLengths = 0;\r\n let segmentIndex = 0;\r\n\r\n const lineSegmentsLength = lineSegments.length;\r\n for (let i = 0; i < lineSegmentsLength; i++) {\r\n if (lengthVisibilityRatio <= sumSegmentLengths + lineSegments[i].length) {\r\n segmentIndex = i;\r\n break;\r\n }\r\n sumSegmentLengths += lineSegments[i].length;\r\n }\r\n\r\n const s = (lengthVisibilityRatio - sumSegmentLengths) / lineSegments[segmentIndex].length;\r\n\r\n lineSegments[segmentIndex].point2.subtractToRef(lineSegments[segmentIndex].point1, TmpVectors.Vector3[0]);\r\n TmpVectors.Vector3[1] = TmpVectors.Vector3[0].multiplyByFloats(s, s, s);\r\n if (!localSpace) {\r\n TmpVectors.Vector3[1].addInPlace(lineSegments[segmentIndex].point1);\r\n }\r\n\r\n return TmpVectors.Vector3[1].clone();\r\n }\r\n\r\n /**\r\n * Creates lines in a shape of circle/arc.\r\n * A segment is a part of the line between it's two points.\r\n * @param radiusX radiusX of the circle\r\n * @param segments number of segments in the circle\r\n * @param z z coordinate of the points. Defaults to 0.\r\n * @param radiusY radiusY of the circle - you can draw an oval if using different values\r\n * @param segmentAngle angle offset of the segments. Defaults to Math.PI * 2 / segments. Change this value to draw a part of the circle.\r\n * @returns line points\r\n */\r\n public static GetCircleLinePoints(radiusX: number, segments: number, z = 0, radiusY = radiusX, segmentAngle = (Math.PI * 2) / segments) {\r\n const points: Vector3[] = [];\r\n for (let i = 0; i <= segments; i++) {\r\n points.push(new Vector3(Math.cos(i * segmentAngle) * radiusX, Math.sin(i * segmentAngle) * radiusY, z));\r\n }\r\n return points;\r\n }\r\n\r\n /**\r\n * Gets line points in a shape of a bezier curve\r\n * @param p0 bezier point0\r\n * @param p1 bezier point1\r\n * @param p2 bezier point2\r\n * @param segments number of segments in the curve\r\n * @returns\r\n */\r\n public static GetBezierLinePoints(p0: Vector3, p1: Vector3, p2: Vector3, segments: number) {\r\n return Curve3.CreateQuadraticBezier(p0, p1, p2, segments)\r\n .getPoints()\r\n .flatMap((v) => [v.x, v.y, v.z]);\r\n }\r\n\r\n /**\r\n *\r\n * @param position position of the arrow cap (mainly you want to create a triangle, set widthUp and widthDown to the same value and omit widthStartUp and widthStartDown)\r\n * @param direction direction which the arrow points to\r\n * @param length length (size) of the arrow cap itself\r\n * @param widthUp the arrow width above the line\r\n * @param widthDown the arrow width belove the line\r\n * @param widthStartUp the arrow width at the start of the arrow above the line. In most scenarios this is 0.\r\n * @param widthStartDown the arrow width at the start of the arrow below the line. In most scenarios this is 0.\r\n * @returns\r\n */\r\n public static GetArrowCap(position: Vector3, direction: Vector3, length: number, widthUp: number, widthDown: number, widthStartUp = 0, widthStartDown = 0) {\r\n const points = [position.clone(), position.add(direction.multiplyByFloats(length, length, length))];\r\n const widths = [widthUp, widthDown, widthStartUp, widthStartDown];\r\n\r\n return {\r\n points,\r\n widths,\r\n };\r\n }\r\n\r\n /**\r\n * Gets 3D positions of points from a text and font\r\n * @param text Text\r\n * @param size Size of the font\r\n * @param resolution Resolution of the font\r\n * @param fontData defines the font data (can be generated with http://gero3.github.io/facetype.js/)\r\n * @param z z coordinate\r\n * @param includeInner include the inner parts of the font in the result. Default true. If false, only the outlines will be returned.\r\n * @returns number[][] of 3D positions\r\n */\r\n public static GetPointsFromText(text: string, size: number, resolution: number, fontData: IFontData, z = 0, includeInner = true) {\r\n const allPoints = [];\r\n const shapePaths = CreateTextShapePaths(text, size, resolution, fontData);\r\n\r\n for (const sp of shapePaths) {\r\n for (const p of sp.paths) {\r\n const points = [];\r\n const points2d = p.getPoints();\r\n for (const p2d of points2d) {\r\n points.push(p2d.x, p2d.y, z);\r\n }\r\n allPoints.push(points);\r\n }\r\n\r\n if (includeInner) {\r\n for (const h of sp.holes) {\r\n const holes = [];\r\n const points2d = h.getPoints();\r\n for (const p2d of points2d) {\r\n holes.push(p2d.x, p2d.y, z);\r\n }\r\n allPoints.push(holes);\r\n }\r\n }\r\n }\r\n\r\n return allPoints;\r\n }\r\n}\r\n","import type { Scene } from \"../scene\";\r\nimport type { Matrix } from \"../Maths/math.vector\";\r\nimport { Vector3 } from \"../Maths/math.vector\";\r\nimport { GreasedLinePluginMaterial } from \"../Materials/greasedLinePluginMaterial\";\r\nimport { Mesh } from \"./mesh\";\r\nimport type { Ray, TrianglePickingPredicate } from \"../Culling/ray\";\r\nimport { Buffer, VertexBuffer } from \"../Buffers/buffer\";\r\nimport { VertexData } from \"./mesh.vertexData\";\r\nimport { PickingInfo } from \"../Collisions/pickingInfo\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { Node } from \"../node\";\r\nimport { DeepCopier } from \"../Misc/deepCopier\";\r\nimport { GreasedLineTools } from \"../Misc/greasedLineTools\";\r\n\r\nexport type GreasedLinePoints = Vector3[] | Vector3[][] | Float32Array | Float32Array[] | number[][] | number[];\r\n\r\n/**\r\n * Options for creating a GreasedLineMesh\r\n */\r\nexport interface GreasedLineMeshOptions {\r\n /**\r\n * Points of the line.\r\n */\r\n points: GreasedLinePoints;\r\n /**\r\n * Each line segmment (from point to point) can have it's width multiplier. Final width = widths[segmentIdx] * width.\r\n * Defaults to empty array.\r\n */\r\n widths?: number[];\r\n /**\r\n * If instance is specified, lines are added to the specified instance.\r\n * Defaults to undefined.\r\n */\r\n instance?: GreasedLineMesh;\r\n /**\r\n * You can manually set the color pointers so you can control which segment/part\r\n * will use which color from the colors material option\r\n */\r\n colorPointers?: number[];\r\n /**\r\n * UVs for the mesh\r\n */\r\n uvs?: number[];\r\n /**\r\n * If true, offsets and widths are updatable.\r\n * Defaults to false.\r\n */\r\n updatable?: boolean;\r\n /**\r\n * Use when @see instance is specified.\r\n * If true, the line will be rendered only after calling instance.updateLazy(). If false, line will be rerendered after every call to @see CreateGreasedLine\r\n * Defaults to false.\r\n */\r\n lazy?: boolean;\r\n}\r\n\r\nMesh._GreasedLineMeshParser = (parsedMesh: any, scene: Scene): Mesh => {\r\n return GreasedLineMesh.Parse(parsedMesh, scene);\r\n};\r\n\r\n/**\r\n * GreasedLine\r\n */\r\nexport class GreasedLineMesh extends Mesh {\r\n private _vertexPositions: number[];\r\n private _previousAndSide: number[];\r\n private _nextAndCounters: number[];\r\n\r\n private _indices: number[];\r\n private _uvs: number[];\r\n private _points: number[][];\r\n private _offsets: number[];\r\n private _colorPointers: number[];\r\n private _widths: number[];\r\n\r\n private _offsetsBuffer?: Buffer;\r\n private _widthsBuffer?: Buffer;\r\n private _colorPointersBuffer?: Buffer;\r\n\r\n private _lazy = false;\r\n private _updatable = false;\r\n\r\n private static _V_START = new Vector3();\r\n private static _V_END = new Vector3();\r\n private static _V_OFFSET_START = new Vector3();\r\n private static _V_OFFSET_END = new Vector3();\r\n\r\n /**\r\n * Treshold used to pick the mesh\r\n */\r\n public intersectionThreshold = 0.1;\r\n\r\n constructor(public readonly name: string, scene: Scene, private _options: GreasedLineMeshOptions) {\r\n super(name, scene, null, null, false, false);\r\n\r\n this._lazy = _options.lazy ?? false;\r\n this._updatable = _options.updatable ?? false;\r\n\r\n this._vertexPositions = [];\r\n this._indices = [];\r\n this._uvs = [];\r\n this._points = [];\r\n this._colorPointers = _options.colorPointers ?? [];\r\n this._widths = _options.widths ?? new Array(_options.points.length).fill(1);\r\n\r\n this._previousAndSide = [];\r\n this._nextAndCounters = [];\r\n\r\n if (_options.points) {\r\n this.addPoints(GreasedLineMesh.ConvertPoints(_options.points));\r\n }\r\n }\r\n\r\n /**\r\n * \"GreasedLineMesh\"\r\n * @returns \"GreasedLineMesh\"\r\n */\r\n public getClassName(): string {\r\n return \"GreasedLineMesh\";\r\n }\r\n\r\n /**\r\n * Converts GreasedLinePoints to number[][]\r\n * @param points GreasedLinePoints\r\n * @returns number[][] with x, y, z coordinates of the points, like [[x, y, z, x, y, z, ...], [x, y, z, ...]]\r\n */\r\n public static ConvertPoints(points: GreasedLinePoints): number[][] {\r\n if (points.length && Array.isArray(points) && typeof points[0] === \"number\") {\r\n return [points];\r\n } else if (points.length && Array.isArray(points[0]) && typeof points[0][0] === \"number\") {\r\n return points;\r\n } else if (points.length && !Array.isArray(points[0]) && points[0] instanceof Vector3) {\r\n const positions: number[] = [];\r\n for (let j = 0; j < points.length; j++) {\r\n const p = points[j] as Vector3;\r\n positions.push(p.x, p.y, p.z);\r\n }\r\n return [positions];\r\n } else if (points.length > 0 && Array.isArray(points[0]) && points[0].length > 0 && points[0][0] instanceof Vector3) {\r\n const positions: number[][] = [];\r\n const vectorPoints = points as Vector3[][];\r\n vectorPoints.forEach((p) => {\r\n positions.push(p.flatMap((p2) => [p2.x, p2.y, p2.z]));\r\n });\r\n return positions;\r\n } else if (points instanceof Float32Array) {\r\n return [Array.from(points)];\r\n } else if (points.length && points[0] instanceof Float32Array) {\r\n const positions: number[][] = [];\r\n points.forEach((p) => {\r\n positions.push(Array.from(p as Float32Array));\r\n });\r\n return positions;\r\n }\r\n\r\n return [];\r\n }\r\n\r\n /**\r\n * Updated a lazy line. Rerenders the line and updates boundinfo as well.\r\n */\r\n public updateLazy() {\r\n this.setPoints(this._points);\r\n if (!this._options.colorPointers) {\r\n this._updateColorPointers();\r\n }\r\n this._createVertexBuffers();\r\n this.refreshBoundingInfo();\r\n\r\n this.greasedLineMaterial?.updateLazy();\r\n }\r\n\r\n /**\r\n * Dispose the line and it's resources\r\n */\r\n public dispose() {\r\n super.dispose();\r\n }\r\n\r\n /**\r\n *\r\n * @returns true if the mesh was created in lazy mode\r\n */\r\n public isLazy(): boolean {\r\n return this._lazy;\r\n }\r\n\r\n /**\r\n * Return the the points offsets\r\n */\r\n get offsets() {\r\n return this._offsets;\r\n }\r\n\r\n /**\r\n * Sets point offests\r\n * @param offsets offset table [x,y,z, x,y,z, ....]\r\n */\r\n set offsets(offsets: number[]) {\r\n this._offsets = offsets;\r\n if (!this._offsetsBuffer) {\r\n this._createOffsetsBuffer(offsets);\r\n } else {\r\n this._offsetsBuffer && this._offsetsBuffer.update(offsets);\r\n }\r\n }\r\n\r\n /**\r\n * Gets widths at each line point like [widthLower, widthUpper, widthLower, widthUpper, ...]\r\n */\r\n get widths() {\r\n return this._widths;\r\n }\r\n\r\n /**\r\n * Sets widths at each line point\r\n * @param widths width table [widthLower, widthUpper, widthLower, widthUpper ...]\r\n */\r\n set widths(widths: number[]) {\r\n this._widths = widths;\r\n if (!this._lazy) {\r\n this._widthsBuffer && this._widthsBuffer.update(widths);\r\n }\r\n }\r\n\r\n /**\r\n * Gets the color pointer. Each vertex need a color pointer. These color pointers points to the colors in the color table @see colors\r\n */\r\n get colorPointers() {\r\n return this._colorPointers;\r\n }\r\n\r\n /**\r\n * Sets the color pointer\r\n * @param colorPointers array of color pointer in the colors array. One pointer for every vertex is needed.\r\n */\r\n set colorPointers(colorPointers: number[]) {\r\n this._colorPointers = colorPointers;\r\n if (!this._lazy) {\r\n this._colorPointersBuffer && this._colorPointersBuffer.update(colorPointers);\r\n }\r\n }\r\n\r\n /**\r\n * Gets the pluginMaterial associated with line\r\n */\r\n get greasedLineMaterial() {\r\n return this.material?.pluginManager?.getPlugin(GreasedLinePluginMaterial.GREASED_LINE_MATERIAL_NAME);\r\n }\r\n\r\n /**\r\n * Return copy the points.\r\n */\r\n get points() {\r\n const pointsCopy: number[][] = [];\r\n DeepCopier.DeepCopy(this._points, pointsCopy);\r\n return pointsCopy;\r\n }\r\n\r\n /**\r\n * Adds new points to the line. It doesn't rerenders the line if in lazy mode.\r\n * @param points points table\r\n */\r\n public addPoints(points: number[][]) {\r\n for (const p of points) {\r\n this._points.push(p);\r\n }\r\n\r\n if (!this._lazy) {\r\n this.setPoints(this._points);\r\n }\r\n }\r\n\r\n private _updateColorPointers() {\r\n let colorPointer = 0;\r\n this._colorPointers = [];\r\n this._points.forEach((p) => {\r\n for (let jj = 0; jj < p.length; jj += 3) {\r\n this._colorPointers.push(colorPointer);\r\n this._colorPointers.push(colorPointer++);\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Sets line points and rerenders the line.\r\n * @param points points table\r\n */\r\n public setPoints(points: number[][]) {\r\n this._points = points;\r\n this._options.points = points;\r\n\r\n this._initGreasedLine();\r\n\r\n let indiceOffset = 0;\r\n\r\n points.forEach((p) => {\r\n const counters: number[] = [];\r\n const positions: number[] = [];\r\n const indices: number[] = [];\r\n\r\n const totalLength = GreasedLineTools.GetLineLength(p);\r\n for (let j = 0, jj = 0; jj < p.length; j++, jj += 3) {\r\n const partialLine = p.slice(0, jj + 3);\r\n const partialLineLength = GreasedLineTools.GetLineLength(partialLine);\r\n const c = partialLineLength / totalLength;\r\n\r\n positions.push(p[jj], p[jj + 1], p[jj + 2]);\r\n positions.push(p[jj], p[jj + 1], p[jj + 2]);\r\n counters.push(c);\r\n counters.push(c);\r\n\r\n if (jj < p.length - 3) {\r\n const n = j * 2 + indiceOffset;\r\n indices.push(n, n + 1, n + 2);\r\n indices.push(n + 2, n + 1, n + 3);\r\n }\r\n }\r\n\r\n indiceOffset += (p.length / 3) * 2;\r\n\r\n const previous: number[] = [];\r\n const next: number[] = [];\r\n const side: number[] = [];\r\n let uvs: number[] = [];\r\n\r\n this._preprocess(positions, previous, next, side, uvs);\r\n\r\n for (const vp of positions) {\r\n this._vertexPositions.push(vp);\r\n }\r\n\r\n for (const i of indices) {\r\n this._indices.push(i);\r\n }\r\n\r\n for (let i = 0; i < side.length; i++) {\r\n this._previousAndSide.push(previous[i * 3], previous[i * 3 + 1], previous[i * 3 + 2], side[i]);\r\n this._nextAndCounters.push(next[i * 3], next[i * 3 + 1], next[i * 3 + 2], counters[i]);\r\n }\r\n\r\n uvs = this._options.uvs ?? uvs;\r\n for (const uv of uvs) {\r\n this._uvs.push(uv);\r\n }\r\n });\r\n\r\n if (!this._lazy) {\r\n if (!this._options.colorPointers) {\r\n this._updateColorPointers();\r\n }\r\n this._createVertexBuffers();\r\n this.refreshBoundingInfo();\r\n }\r\n }\r\n\r\n private _createLineOptions() {\r\n const lineOptions: GreasedLineMeshOptions = {\r\n points: this._points,\r\n colorPointers: this._colorPointers,\r\n lazy: this._lazy,\r\n updatable: this._updatable,\r\n uvs: this._uvs,\r\n widths: this._widths,\r\n };\r\n return lineOptions;\r\n }\r\n\r\n /**\r\n * Clones the GreasedLineMesh.\r\n * @param name new line name\r\n * @param newParent new parent node\r\n * @returns cloned line\r\n */\r\n public clone(name: string = `${this.name}-cloned`, newParent?: Nullable) {\r\n const lineOptions = this._createLineOptions();\r\n const deepCopiedLineOptions = {};\r\n DeepCopier.DeepCopy(lineOptions, deepCopiedLineOptions, [\"instance\"]);\r\n\r\n const cloned = new GreasedLineMesh(name, this._scene, deepCopiedLineOptions);\r\n if (newParent) {\r\n cloned.parent = newParent;\r\n }\r\n\r\n cloned.material = this.material;\r\n\r\n return cloned;\r\n }\r\n\r\n /**\r\n * Serializes this GreasedLineMesh\r\n * @param serializationObject object to write serialization to\r\n */\r\n public serialize(serializationObject: any): void {\r\n super.serialize(serializationObject);\r\n serializationObject.type = this.getClassName();\r\n\r\n serializationObject.lineOptions = this._createLineOptions();\r\n }\r\n\r\n /**\r\n * Parses a serialized GreasedLineMesh\r\n * @param parsedMesh the serialized GreasedLineMesh\r\n * @param scene the scene to create the GreasedLineMesh in\r\n * @returns the created GreasedLineMesh\r\n */\r\n public static Parse(parsedMesh: any, scene: Scene): Mesh {\r\n const lineOptions = parsedMesh.lineOptions;\r\n const name = parsedMesh.name;\r\n const result = new GreasedLineMesh(name, scene, lineOptions);\r\n return result;\r\n }\r\n\r\n /**\r\n * Checks whether a ray is intersecting this GreasedLineMesh\r\n * @param ray ray to check the intersection of this mesh with\r\n * @param fastCheck not supported\r\n * @param trianglePredicate not supported\r\n * @param onlyBoundingInfo defines a boolean indicating if picking should only happen using bounding info (false by default)\r\n * @param worldToUse not supported\r\n * @param skipBoundingInfo a boolean indicating if we should skip the bounding info check\r\n * @returns the picking info\r\n */\r\n public intersects(\r\n ray: Ray,\r\n fastCheck?: boolean,\r\n trianglePredicate?: TrianglePickingPredicate,\r\n onlyBoundingInfo = false,\r\n worldToUse?: Matrix,\r\n skipBoundingInfo = false\r\n ): PickingInfo {\r\n const pickingInfo = new PickingInfo();\r\n const intersections = this.findAllIntersections(ray, fastCheck, trianglePredicate, onlyBoundingInfo, worldToUse, skipBoundingInfo, true);\r\n if (intersections?.length === 1) {\r\n const intersection = intersections[0];\r\n pickingInfo.hit = true;\r\n pickingInfo.distance = intersection.distance;\r\n pickingInfo.ray = ray;\r\n pickingInfo.pickedMesh = this;\r\n pickingInfo.pickedPoint = intersection.point;\r\n }\r\n return pickingInfo;\r\n }\r\n\r\n /**\r\n * Gets all intersections of a ray and the line\r\n * @param ray Ray to check the intersection of this mesh with\r\n * @param _fastCheck not supported\r\n * @param _trianglePredicate not supported\r\n * @param onlyBoundingInfo defines a boolean indicating if picking should only happen using bounding info (false by default)\r\n * @param _worldToUse not supported\r\n * @param skipBoundingInfo a boolean indicating if we should skip the bounding info check\r\n * @param firstOnly If true, the first and only intersection is immediatelly returned if found\r\n * @returns intersection(s)\r\n */\r\n public findAllIntersections(\r\n ray: Ray,\r\n _fastCheck?: boolean,\r\n _trianglePredicate?: TrianglePickingPredicate,\r\n onlyBoundingInfo = false,\r\n _worldToUse?: Matrix,\r\n skipBoundingInfo = false,\r\n firstOnly = false\r\n ): { distance: number; point: Vector3 }[] | undefined {\r\n if (onlyBoundingInfo && !skipBoundingInfo && ray.intersectsSphere(this._boundingSphere, this.intersectionThreshold) === false) {\r\n return;\r\n }\r\n\r\n const indices = this.getIndices();\r\n const positions = this.getVerticesData(VertexBuffer.PositionKind);\r\n const widths = this._widths;\r\n\r\n const lineWidth = this.greasedLineMaterial?.width ?? 1;\r\n\r\n const intersects = [];\r\n if (indices && positions && widths) {\r\n let i = 0,\r\n l = 0;\r\n for (i = 0, l = indices.length - 1; i < l; i += 3) {\r\n const a = indices[i];\r\n const b = indices[i + 1];\r\n\r\n GreasedLineMesh._V_START.fromArray(positions, a * 3);\r\n GreasedLineMesh._V_END.fromArray(positions, b * 3);\r\n\r\n if (this._offsets) {\r\n GreasedLineMesh._V_OFFSET_START.fromArray(this._offsets, a * 3);\r\n GreasedLineMesh._V_OFFSET_END.fromArray(this._offsets, b * 3);\r\n GreasedLineMesh._V_START.addInPlace(GreasedLineMesh._V_OFFSET_START);\r\n GreasedLineMesh._V_END.addInPlace(GreasedLineMesh._V_OFFSET_END);\r\n }\r\n\r\n const iFloored = Math.floor(i / 3);\r\n const width = widths[iFloored] !== undefined ? widths[iFloored] : 1;\r\n const precision = (this.intersectionThreshold * (lineWidth * width)) / 2;\r\n\r\n const distance = ray.intersectionSegment(GreasedLineMesh._V_START, GreasedLineMesh._V_END, precision);\r\n if (distance !== -1) {\r\n intersects.push({\r\n distance: distance,\r\n point: ray.direction.normalize().multiplyByFloats(distance, distance, distance).add(ray.origin),\r\n });\r\n if (firstOnly) {\r\n return intersects;\r\n }\r\n }\r\n }\r\n i = l;\r\n }\r\n\r\n return intersects;\r\n }\r\n\r\n private _initGreasedLine() {\r\n this._vertexPositions = [];\r\n this._previousAndSide = [];\r\n this._nextAndCounters = [];\r\n this._indices = [];\r\n this._uvs = [];\r\n }\r\n\r\n private get _boundingSphere() {\r\n return this.getBoundingInfo().boundingSphere;\r\n }\r\n\r\n private static _CompareV3(positionIdx1: number, positionIdx2: number, positions: number[]) {\r\n const arrayIdx1 = positionIdx1 * 6;\r\n const arrayIdx2 = positionIdx2 * 6;\r\n return positions[arrayIdx1] === positions[arrayIdx2] && positions[arrayIdx1 + 1] === positions[arrayIdx2 + 1] && positions[arrayIdx1 + 2] === positions[arrayIdx2 + 2];\r\n }\r\n\r\n private static _CopyV3(positionIdx: number, positions: number[]) {\r\n const arrayIdx = positionIdx * 6;\r\n return [positions[arrayIdx], positions[arrayIdx + 1], positions[arrayIdx + 2]];\r\n }\r\n\r\n private _preprocess(positions: number[], previous: number[], next: number[], side: number[], uvs: number[]) {\r\n const l = positions.length / 6;\r\n\r\n let v: number[] = [];\r\n\r\n if (GreasedLineMesh._CompareV3(0, l - 1, positions)) {\r\n v = GreasedLineMesh._CopyV3(l - 2, positions);\r\n } else {\r\n v = GreasedLineMesh._CopyV3(0, positions);\r\n }\r\n previous.push(v[0], v[1], v[2]);\r\n previous.push(v[0], v[1], v[2]);\r\n\r\n for (let j = 0; j < l; j++) {\r\n side.push(1);\r\n side.push(-1);\r\n\r\n // uvs\r\n if (!this._options.uvs) {\r\n uvs.push(j / (l - 1), 0);\r\n uvs.push(j / (l - 1), 1);\r\n }\r\n\r\n if (j < l - 1) {\r\n v = GreasedLineMesh._CopyV3(j, positions);\r\n previous.push(v[0], v[1], v[2]);\r\n previous.push(v[0], v[1], v[2]);\r\n }\r\n if (j > 0) {\r\n v = GreasedLineMesh._CopyV3(j, positions);\r\n next.push(v[0], v[1], v[2]);\r\n next.push(v[0], v[1], v[2]);\r\n }\r\n }\r\n\r\n if (GreasedLineMesh._CompareV3(l - 1, 0, positions)) {\r\n v = GreasedLineMesh._CopyV3(1, positions);\r\n } else {\r\n v = GreasedLineMesh._CopyV3(l - 1, positions);\r\n }\r\n next.push(v[0], v[1], v[2]);\r\n next.push(v[0], v[1], v[2]);\r\n\r\n return {\r\n previous,\r\n next,\r\n uvs,\r\n side,\r\n };\r\n }\r\n\r\n private _createVertexBuffers() {\r\n const vertexData = new VertexData();\r\n vertexData.positions = this._vertexPositions;\r\n vertexData.indices = this._indices;\r\n vertexData.uvs = this._uvs;\r\n vertexData.applyToMesh(this, this._options.updatable);\r\n\r\n const engine = this._scene.getEngine();\r\n\r\n const previousAndSideBuffer = new Buffer(engine, this._previousAndSide, false, 4);\r\n this.setVerticesBuffer(previousAndSideBuffer.createVertexBuffer(\"grl_previousAndSide\", 0, 4));\r\n\r\n const nextAndCountersBuffer = new Buffer(engine, this._nextAndCounters, false, 4);\r\n this.setVerticesBuffer(nextAndCountersBuffer.createVertexBuffer(\"grl_nextAndCounters\", 0, 4));\r\n\r\n const widthBuffer = new Buffer(engine, this._widths, this._updatable, 1);\r\n this.setVerticesBuffer(widthBuffer.createVertexBuffer(\"grl_widths\", 0, 1));\r\n this._widthsBuffer = widthBuffer;\r\n\r\n const colorPointersBuffer = new Buffer(engine, this._colorPointers, this._updatable, 1);\r\n this.setVerticesBuffer(colorPointersBuffer.createVertexBuffer(\"grl_colorPointers\", 0, 1));\r\n this._colorPointersBuffer = colorPointersBuffer;\r\n }\r\n\r\n private _createOffsetsBuffer(offsets: number[]) {\r\n const engine = this._scene.getEngine();\r\n\r\n const offsetBuffer = new Buffer(engine, offsets, this._updatable, 3);\r\n this.setVerticesBuffer(offsetBuffer.createVertexBuffer(\"grl_offsets\", 0, 3));\r\n this._offsetsBuffer = offsetBuffer;\r\n }\r\n}\r\n","import { StandardMaterial } from '@babylonjs/core/Materials/standardMaterial';\r\nimport { Color3, Matrix, Quaternion, Vector2, Vector3 } from '@babylonjs/core/Maths';\r\nimport { CreateBox, CreateCylinder, CreatePlane, CreateSphere } from '@babylonjs/core/Meshes/Builders';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { eUvMode } from '@models/classes/enums';\r\nimport { UvMapFeature } from '@models/classes/features/uv-map-feature';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { SubmeshVertexData } from './geometry-helper';\r\nimport { applyMatrix4, computeNormal, rotationToVector } from './math-helper';\r\n\r\nconst pWorking = new Vector3();\r\nconst p1 = new Vector3();\r\nconst p2 = new Vector3();\r\nconst p3 = new Vector3();\r\nconst uv1 = new Vector2();\r\nconst uv2 = new Vector2();\r\nconst uv3 = new Vector2();\r\nconst workingUv = new Vector2();\r\nconst workingUv2 = new Vector2();\r\nconst working1 = new Vector3();\r\n\r\nexport class UVMapHelper {\r\n generateUVMapInPlace(\r\n vertexData: Partial,\r\n feature: UvMapFeature,\r\n indexList?: number[]\r\n ): Partial {\r\n const newControlPointMap: Record = {};\r\n const axis = feature.projectionAxis.toVec3();\r\n const rotation = rotationToVector(axis, new Vector3(0, 1, 0));\r\n rotation.multiplyInPlace(Quaternion.RotationAxis(new Vector3(0, 1, 0), (feature.angle / 180) * Math.PI));\r\n // const matrixRotation = new Matrix();\r\n // Matrix.FromQuaternionToRef(rotation, matrixRotation);\r\n const transformMatrix = Matrix.Compose(\r\n feature.scale ? feature.scale.toVec3() : Vector3.One(),\r\n rotation,\r\n Vector3.Zero()\r\n );\r\n\r\n const transformMatrixInv = Matrix.Invert(transformMatrix);\r\n\r\n if (!vertexData.positions) {\r\n throw Error('No vertex positions on mesh');\r\n }\r\n if (!vertexData.indices) {\r\n throw Error('No vertex indices on mesh');\r\n }\r\n const geometryPositions = vertexData.positions;\r\n const geometryIndices = vertexData.indices;\r\n\r\n const controlPointCount = geometryPositions.length / 3;\r\n\r\n const newUvs =\r\n vertexData.uvs && vertexData.uvs.length === controlPointCount * 2\r\n ? new Float32Array(vertexData.uvs)\r\n : new Float32Array(controlPointCount * 2);\r\n const appendVertexData = {\r\n positions: [] as number[],\r\n normals: [] as number[],\r\n uvs: [] as number[],\r\n };\r\n\r\n const nbFaces = indexList ? indexList.length / 3 : geometryIndices.length / 3;\r\n const touchedControlPoints = new Int8Array(controlPointCount);\r\n\r\n if (indexList) {\r\n indexList.forEach(index => {\r\n touchedControlPoints[geometryIndices[index]] = 1;\r\n });\r\n }\r\n const outputUvs = [uv1, uv2, uv3];\r\n\r\n for (let i = 0; i < nbFaces; i++) {\r\n /** here we generate the new UVs */\r\n let i1: number;\r\n let i2: number;\r\n let i3: number;\r\n if (indexList) {\r\n i1 = indexList[i * 3];\r\n i2 = indexList[i * 3 + 1];\r\n i3 = indexList[i * 3 + 2];\r\n } else {\r\n i1 = i * 3;\r\n i2 = i * 3 + 1;\r\n i3 = i * 3 + 2;\r\n }\r\n const v1 = geometryIndices[i1];\r\n const v2 = geometryIndices[i2];\r\n const v3 = geometryIndices[i3];\r\n\r\n if (feature.mode === eUvMode.useExisting) {\r\n if (vertexData.uvs) {\r\n const angle = (feature.angle / 180) * Math.PI;\r\n const vertices = [v1, v2, v3];\r\n\r\n for (let i = 0; i < 3; i++) {\r\n const uvIdx = vertices[i] * 2;\r\n\r\n const u = vertexData.uvs[uvIdx] - 0.5;\r\n const v = vertexData.uvs[uvIdx + 1] - 0.5;\r\n\r\n let newU = u * Math.cos(angle) - v * Math.sin(angle) + 0.5;\r\n let newV = u * Math.sin(angle) + v * Math.cos(angle) + 0.5;\r\n newU = newU * feature.uTile + feature.uOffset;\r\n newV = newV * feature.vTile + feature.vOffset;\r\n\r\n outputUvs[i].copyFromFloats(newU, newV);\r\n }\r\n }\r\n } else {\r\n const start = v1 * 3;\r\n p1.copyFromFloats(geometryPositions[start], geometryPositions[start + 1], geometryPositions[start + 2]);\r\n\r\n const start2 = v2 * 3;\r\n p2.copyFromFloats(\r\n geometryPositions[start2],\r\n geometryPositions[start2 + 1],\r\n geometryPositions[start2 + 2]\r\n );\r\n\r\n const start3 = v3 * 3;\r\n p3.copyFromFloats(\r\n geometryPositions[start3],\r\n geometryPositions[start3 + 1],\r\n geometryPositions[start3 + 2]\r\n );\r\n const normal = computeNormal([p1, p2, p3]);\r\n\r\n applyMatrix4(normal, transformMatrixInv);\r\n\r\n if (feature.center) {\r\n const center = feature.center.toVec3();\r\n p1.subtractInPlace(center);\r\n p2.subtractInPlace(center);\r\n p3.subtractInPlace(center);\r\n }\r\n this.updateFaceUVMapToRef(normal, p1, feature, transformMatrixInv, uv1);\r\n this.updateFaceUVMapToRef(normal, p2, feature, transformMatrixInv, uv2);\r\n this.updateFaceUVMapToRef(normal, p3, feature, transformMatrixInv, uv3);\r\n\r\n /**\r\n * Find the UV at the midpoint position between each edge of the polygon.\r\n * This should closely match the midpoints of the generated vertex UVs.If it does not, we must wrap the UVs at that edge to avoid a UV seam.\r\n * The tolerance is an arbitrary number to avoid artifacts. If wrapping is occuring when it shouldn't, increase this number.\r\n * If wrapping is not occurring when it should, reduce it.\r\n * As you try to wrap a uv on polygons that are not arranged in a good way to accept them (i.e wrap a sphere onto a 8-vertex cube), the error will become significant.\r\n */\r\n const tolerance = 0.05;\r\n working1.copyFrom(p1);\r\n this.updateFaceUVMapToRef(\r\n normal,\r\n working1.addInPlace(p2).scaleInPlace(0.5),\r\n feature,\r\n transformMatrixInv,\r\n workingUv\r\n );\r\n workingUv2.copyFrom(uv1);\r\n workingUv2.addInPlace(uv2).scaleInPlace(0.5);\r\n workingUv.subtractInPlace(workingUv2);\r\n if (Math.abs(workingUv.x % 1) > tolerance) {\r\n if (uv1.x > uv2.x) uv1.x -= 1;\r\n else uv2.x -= 1;\r\n }\r\n if (Math.abs(workingUv.y % 1) > tolerance) {\r\n if (uv1.y > uv2.y) uv1.y -= 1;\r\n else uv2.y -= 1;\r\n }\r\n\r\n if (uv2.x >= 0 && uv2.y >= 0) {\r\n working1.copyFrom(p2);\r\n this.updateFaceUVMapToRef(\r\n normal,\r\n working1.addInPlace(p3).scaleInPlace(0.5),\r\n feature,\r\n transformMatrixInv,\r\n workingUv\r\n );\r\n workingUv2.copyFrom(uv2);\r\n workingUv2.addInPlace(uv3).scaleInPlace(0.5);\r\n workingUv.subtractInPlace(workingUv2);\r\n if (Math.abs(workingUv.x % 1) > tolerance) {\r\n if (uv2.x > uv3.x) uv2.x -= 1;\r\n else uv3.x -= 1;\r\n }\r\n if (Math.abs(workingUv.y % 1) > tolerance) {\r\n if (uv2.y > uv3.y) uv2.y -= 1;\r\n else uv3.y -= 1;\r\n }\r\n }\r\n\r\n if (uv1.x >= 0 && uv1.y >= 0 && uv3.x >= 0 && uv3.y >= 0) {\r\n working1.copyFrom(p1);\r\n this.updateFaceUVMapToRef(\r\n normal,\r\n working1.addInPlace(p3).scaleInPlace(0.5),\r\n feature,\r\n transformMatrixInv,\r\n workingUv\r\n );\r\n workingUv2.copyFrom(uv1);\r\n workingUv2.addInPlace(uv3).scaleInPlace(0.5);\r\n workingUv.subtractInPlace(workingUv2);\r\n if (Math.abs(workingUv.x % 1) > tolerance) {\r\n if (uv1.x > uv3.x) uv1.x -= 1;\r\n else uv3.x -= 1;\r\n }\r\n if (Math.abs(workingUv.y % 1) > tolerance) {\r\n if (uv1.y > uv3.y) uv1.y -= 1;\r\n else uv3.y -= 1;\r\n }\r\n }\r\n }\r\n /** Done generating the new UVs, stored in uv1, uv2 and uv3 for this polygon */\r\n\r\n if (touchedControlPoints[v1]) {\r\n expandControlPoints(i1, v1, uv1);\r\n } else {\r\n newUvs[v1 * 2] = uv1.x;\r\n newUvs[v1 * 2 + 1] = uv1.y;\r\n }\r\n\r\n if (touchedControlPoints[v2]) {\r\n expandControlPoints(i2, v2, uv2);\r\n } else {\r\n newUvs[v2 * 2] = uv2.x;\r\n newUvs[v2 * 2 + 1] = uv2.y;\r\n }\r\n\r\n if (touchedControlPoints[v3]) {\r\n expandControlPoints(i3, v3, uv3);\r\n } else {\r\n newUvs[v3 * 2] = uv3.x;\r\n newUvs[v3 * 2 + 1] = uv3.y;\r\n }\r\n\r\n touchedControlPoints[v1] = 1;\r\n touchedControlPoints[v2] = 1;\r\n touchedControlPoints[v3] = 1;\r\n }\r\n\r\n if (appendVertexData.positions.length > 0) {\r\n const newPositions = new Float32Array(vertexData.positions.length + appendVertexData.positions.length);\r\n newPositions.set(vertexData.positions);\r\n newPositions.set(appendVertexData.positions, vertexData.positions.length);\r\n vertexData.positions = newPositions;\r\n }\r\n if (vertexData.normals && appendVertexData.normals.length > 0) {\r\n const newNormals = new Float32Array(vertexData.normals.length + appendVertexData.normals.length);\r\n newNormals.set(vertexData.normals);\r\n newNormals.set(appendVertexData.normals, vertexData.normals.length);\r\n vertexData.normals = newNormals;\r\n }\r\n if (appendVertexData.uvs.length > 0) {\r\n const newUvsTyped = new Float32Array(newUvs.length + appendVertexData.uvs.length);\r\n newUvsTyped.set(newUvs);\r\n newUvsTyped.set(appendVertexData.uvs, newUvs.length);\r\n vertexData.uvs = newUvsTyped;\r\n } else {\r\n vertexData.uvs = new Float32Array(newUvs);\r\n }\r\n\r\n return vertexData;\r\n\r\n function expandControlPoints(index: number, vertexIndex: number, newUv: Vector2) {\r\n if (newUvs[vertexIndex * 2] !== newUv.x || newUvs[vertexIndex * 2 + 1] !== newUv.y) {\r\n if (!vertexData.positions || !vertexData.indices) {\r\n throw Error('Invalid geometry: positions or indices are null');\r\n }\r\n const indices = vertexData.indices;\r\n const positions = vertexData.positions;\r\n const normals = vertexData.normals;\r\n\r\n const key =\r\n positions[vertexIndex * 3].toPrecision(6) +\r\n ',' +\r\n positions[vertexIndex * 3 + 1].toPrecision(6) +\r\n ',' +\r\n positions[vertexIndex * 3 + 2].toPrecision(6) +\r\n '_' +\r\n (normals\r\n ? normals[vertexIndex * 3].toPrecision(6) +\r\n ',' +\r\n normals[vertexIndex * 3 + 1].toPrecision(6) +\r\n ',' +\r\n normals[vertexIndex * 3 + 2].toPrecision(6) +\r\n '_'\r\n : '') +\r\n newUv.x.toPrecision(6) +\r\n ',' +\r\n newUv.y.toPrecision(6);\r\n\r\n if (newControlPointMap[key]) {\r\n indices[index] = newControlPointMap[key];\r\n } else {\r\n const newVertexIndex = (positions.length + appendVertexData.positions.length) / 3;\r\n indices[index] = newVertexIndex;\r\n\r\n appendVertexData.positions.push(...positions.slice(vertexIndex * 3, vertexIndex * 3 + 3));\r\n if (vertexData.normals) {\r\n appendVertexData.normals.push(\r\n ...vertexData.normals.slice(vertexIndex * 3, vertexIndex * 3 + 3)\r\n );\r\n }\r\n appendVertexData.uvs.push(newUv.x, newUv.y);\r\n\r\n if (vertexData.submeshData) {\r\n for (let i = 0; i < vertexData.submeshData.length; i++) {\r\n const submeshData = vertexData.submeshData[i];\r\n if (\r\n submeshData.verticesStart <= vertexIndex &&\r\n submeshData.verticesStart + submeshData.verticesCount > vertexIndex\r\n ) {\r\n submeshData.verticesCount += 1;\r\n } else if (submeshData.verticesStart > vertexIndex) {\r\n submeshData.verticesStart += 1;\r\n }\r\n }\r\n }\r\n\r\n newControlPointMap[key] = newVertexIndex;\r\n }\r\n }\r\n }\r\n }\r\n\r\n computeAutoFitSize(mesh: AbstractMesh): KbVector {\r\n const boundingBox = mesh.getBoundingInfo().boundingBox;\r\n\r\n //const center = boundingBox.center;\r\n\r\n const max = boundingBox.maximum;\r\n const min = boundingBox.minimum;\r\n\r\n return new KbVector(max.x - min.x, max.y - min.y, max.z - min.z);\r\n }\r\n\r\n private updateFaceUVMapToRef(\r\n faceNormal: Vector3,\r\n position: Vector3,\r\n feat: UvMapFeature,\r\n transformMatrixInv: Matrix,\r\n uv: Vector2\r\n ): Vector2 {\r\n pWorking.copyFrom(position);\r\n applyMatrix4(pWorking, transformMatrixInv);\r\n\r\n switch (feat.mode) {\r\n case eUvMode.planar:\r\n this.planarUVMap(pWorking, uv);\r\n break;\r\n case eUvMode.box:\r\n this.boxUVMap(pWorking, faceNormal, uv);\r\n break;\r\n case eUvMode.cylindrical:\r\n this.cylindricalUVMap(pWorking, uv);\r\n break;\r\n case eUvMode.spherical:\r\n this.sphericalUVMap(pWorking, uv);\r\n break;\r\n case eUvMode.useExisting:\r\n //don't do anything so that existing UVs of mesh are being used instead\r\n break;\r\n default:\r\n uv.set(0, 0);\r\n break;\r\n }\r\n if (feat.uTile && feat.uTile !== 1) {\r\n uv.x *= feat.uTile;\r\n }\r\n if (feat.vTile && feat.vTile !== 1) {\r\n uv.y *= feat.vTile;\r\n }\r\n return uv;\r\n }\r\n\r\n private planarUVMap(vertex: Vector3, uv: Vector2): Vector2 {\r\n let u = vertex.x;\r\n let v = -vertex.z;\r\n\r\n u = u + 0.5;\r\n v = v + 0.5;\r\n uv.set(u, v);\r\n return uv;\r\n }\r\n\r\n private cylindricalUVMap(vertex: Vector3, uv: Vector2): Vector2 {\r\n let lon;\r\n if (vertex.x === 0 && vertex.z === 0) {\r\n lon = 0;\r\n } else {\r\n lon = Math.atan2(vertex.x, vertex.z);\r\n if (lon < 0) {\r\n lon += Math.PI * 2;\r\n }\r\n }\r\n const t = vertex.y + 0.5;\r\n lon = lon / (Math.PI * 2);\r\n uv.set(lon, t);\r\n return uv;\r\n }\r\n\r\n private boxUVMap(vertex: Vector3, normal: Vector3, uv: Vector2): Vector2 {\r\n let s = 0,\r\n t = 0;\r\n const x = vertex.x,\r\n y = vertex.y,\r\n z = vertex.z;\r\n const nx = Math.abs(normal.x),\r\n ny = Math.abs(normal.y),\r\n nz = Math.abs(normal.z);\r\n if (nx >= ny && nx >= nz) {\r\n s = -z + 0.5;\r\n t = y + 0.5;\r\n }\r\n if (ny >= nx && ny >= nz) {\r\n s = x + 0.5;\r\n t = -z + 0.5;\r\n }\r\n if (nz >= nx && nz >= ny) {\r\n s = x + 0.5;\r\n t = y + 0.5;\r\n }\r\n // not sure what this is for but it screws up the mapping and flips the uv's halfway through\r\n // if (normal.x < 0) {\r\n // s = -s + 1\r\n // }\r\n // if (normal.y < 0) {\r\n // s = -s + 1\r\n // }\r\n // if (normal.z < 0) {\r\n // s = -s + 1\r\n // }\r\n uv.set(s, t);\r\n return uv;\r\n }\r\n\r\n private sphericalUVMap(vertex: Vector3, uv: Vector2): Vector2 {\r\n const newVertex = vertex.clone();\r\n const lonlat = xyzToLongitudeLatitude(newVertex.x, -newVertex.y, newVertex.z);\r\n const lon = lonlat.lon / (Math.PI * 2);\r\n const lat = 0.5 - lonlat.lat / Math.PI;\r\n uv.set(lon, lat);\r\n return uv;\r\n\r\n function xyzToLongitudeLatitude(x: number, y: number, z: number) {\r\n let lo: number, la: number;\r\n const EPSILON = 1e-6;\r\n if (Math.abs(x) < EPSILON && Math.abs(z) < EPSILON) {\r\n lo = 0;\r\n if (Math.abs(y) < EPSILON) {\r\n la = 0;\r\n } else {\r\n la = y < 0 ? -(Math.PI / 2) : Math.PI / 2;\r\n }\r\n } else {\r\n lo = Math.atan2(x, z);\r\n if (lo < 0) {\r\n lo += Math.PI * 2;\r\n }\r\n const h = Math.sqrt(x * x + z * z);\r\n la = Math.atan2(y, h);\r\n }\r\n return {\r\n lon: lo,\r\n lat: la,\r\n };\r\n }\r\n }\r\n}\r\n\r\nexport function createUvMapVisualization(uvMap: UvMapFeature, scene: Scene, color?: Color3) {\r\n let uvMapVisualization: Mesh;\r\n\r\n if (!color) {\r\n color = Color3.Yellow();\r\n }\r\n const coloredMaterial = new StandardMaterial('', scene);\r\n coloredMaterial.wireframe = true;\r\n coloredMaterial.emissiveColor = color;\r\n\r\n if (uvMap.mode === eUvMode.spherical) {\r\n uvMapVisualization = CreateSphere('UvMapSphere', { segments: 6 }, scene);\r\n } else if (uvMap.mode === eUvMode.cylindrical) {\r\n uvMapVisualization = CreateCylinder('UvMapCylinder', { tessellation: 10 }, scene);\r\n } else if (uvMap.mode === eUvMode.box) {\r\n uvMapVisualization = CreateBox('UvBox', {}, scene);\r\n } else if (uvMap.mode === eUvMode.planar) {\r\n uvMapVisualization = CreatePlane('mirrorPlane', {}, scene);\r\n uvMapVisualization.rotation = new Vector3(Math.PI / 2, 0, 0);\r\n uvMapVisualization.bakeCurrentTransformIntoVertices();\r\n } else {\r\n uvMapVisualization = CreatePlane('mirrorPlane', {}, scene);\r\n uvMapVisualization.bakeCurrentTransformIntoVertices();\r\n uvMapVisualization.setEnabled(false);\r\n }\r\n uvMapVisualization.material = coloredMaterial;\r\n\r\n return uvMapVisualization;\r\n}\r\n","import { IVertexGroup } from '@models/classes/kb3d-models';\r\n\r\nexport function removeVerticesFromAll(\r\n vertexGroups: Array>,\r\n startIndex: number,\r\n endIndex: number\r\n): void {\r\n for (let i = 0; i < vertexGroups.length; i++) {\r\n const vertexGroup = vertexGroups[i];\r\n if (vertexGroupsIntersect(vertexGroup, startIndex, endIndex)) {\r\n const result = excludeVertices(vertexGroup, startIndex, endIndex);\r\n vertexGroups.splice(i, 1, ...result);\r\n }\r\n }\r\n}\r\n\r\nexport function mergeVerticesIntoAll>(vertexGroups: T[], newVertex: T): void {\r\n vertexGroups.push(newVertex);\r\n vertexGroups.sort((a, b) => a.startIndex! - b.startIndex!);\r\n\r\n for (let i = 0; i < vertexGroups.length; i++) {\r\n const vertexGroup = vertexGroups[i];\r\n const nextVertexGroup = vertexGroups[i + 1];\r\n\r\n if (nextVertexGroup && nextVertexGroup.startIndex < vertexGroup.endIndex) {\r\n const mergedVertexGroup = { ...vertexGroup, endIndex: nextVertexGroup.endIndex };\r\n vertexGroups.splice(i, 2, mergedVertexGroup);\r\n i--;\r\n }\r\n }\r\n}\r\n\r\nexport function excludeVertices>(\r\n vertexGroup: T,\r\n startIndex: number,\r\n endIndex: number\r\n): T[] {\r\n if (vertexGroupsIntersect(vertexGroup, startIndex, endIndex)) {\r\n const result = [] as T[];\r\n\r\n // Maintain the selections of either side of the current vertex\r\n if (startIndex > vertexGroup.startIndex) {\r\n result.push({\r\n ...vertexGroup,\r\n endIndex: startIndex,\r\n });\r\n }\r\n if (endIndex < vertexGroup.endIndex) {\r\n result.push({\r\n ...vertexGroup,\r\n startIndex: endIndex,\r\n });\r\n }\r\n\r\n return result;\r\n } else {\r\n return [vertexGroup];\r\n }\r\n}\r\n\r\nexport function mergeVertices>(vertexGroup: T, newVertex: T): T[] {\r\n const result = [] as T[];\r\n const newVertexGroup = { ...vertexGroup };\r\n const startIndex = newVertex.startIndex;\r\n const endIndex = newVertex.endIndex;\r\n\r\n if (vertexGroupsIntersect(vertexGroup, startIndex, endIndex)) {\r\n // Maintain the selections of either side of the current vertex\r\n if (startIndex < vertexGroup.startIndex) {\r\n newVertexGroup.startIndex = startIndex;\r\n }\r\n if (endIndex > vertexGroup.endIndex) {\r\n newVertexGroup.endIndex = endIndex;\r\n }\r\n result.push(newVertexGroup);\r\n } else {\r\n if (startIndex > vertexGroup.startIndex) {\r\n result.push(newVertexGroup, newVertex);\r\n } else {\r\n result.push(newVertex, newVertexGroup);\r\n }\r\n }\r\n\r\n return result;\r\n}\r\n\r\nfunction vertexGroupsIntersect(vertexGroup: Required, startIndex: number, endIndex: number) {\r\n return startIndex <= vertexGroup.endIndex && endIndex >= vertexGroup.startIndex;\r\n}\r\n","import { StandardMaterial } from '@babylonjs/core/Materials/standardMaterial';\r\nimport { Color3, Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { MeshBuilder } from '@babylonjs/core/Meshes/meshBuilder';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { eViewpointTargetMode, eViewpointType } from '@models/classes/enums';\r\nimport { ViewpointNode } from '@models/classes/viewpoints';\r\n\r\nexport function createViewpointVisualization(viewpoint: ViewpointNode, scene: Scene, color?: Color3) {\r\n if (!color) {\r\n color = Color3.Green();\r\n }\r\n let targetPositionMarker: Mesh | undefined;\r\n\r\n const viewpointPositionGizmo = MeshBuilder.CreateCylinder(\r\n 'cameraGizmoLens',\r\n { diameterTop: 0.3, height: 0.2, diameterBottom: 0.2, tessellation: 8 },\r\n scene\r\n );\r\n const coloredMaterial = new StandardMaterial('', scene);\r\n coloredMaterial.wireframe = true;\r\n coloredMaterial.emissiveColor = color;\r\n viewpointPositionGizmo.material = coloredMaterial;\r\n\r\n const lightPositionGizmoBox = MeshBuilder.CreateBox('cameraGizmoBody', { size: 0.4 }, scene);\r\n lightPositionGizmoBox.setParent(viewpointPositionGizmo);\r\n lightPositionGizmoBox.position = new Vector3(0, -0.3, 0);\r\n lightPositionGizmoBox.material = coloredMaterial;\r\n\r\n viewpointPositionGizmo.isPickable = false;\r\n viewpointPositionGizmo.setEnabled(true);\r\n\r\n if (viewpoint.type === eViewpointType.orbit && viewpoint.targetMode === eViewpointTargetMode.point) {\r\n targetPositionMarker = MeshBuilder.CreateSphere('viewpointTargetMarker', { diameter: 0.15 }, scene);\r\n const coloredMaterial = new StandardMaterial('', scene);\r\n coloredMaterial.emissiveColor = color;\r\n targetPositionMarker.material = coloredMaterial;\r\n targetPositionMarker.isPickable = false;\r\n targetPositionMarker.setEnabled(true);\r\n }\r\n\r\n return { viewpointPositionGizmo, targetPositionMarker };\r\n}\r\n","import { BaseTexture } from '@babylonjs/core/Materials/Textures/baseTexture';\r\nimport { Material } from '@babylonjs/core/Materials/material';\r\nimport { Node } from '@babylonjs/core/node';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { Kb3dObject, Token, instanceOf } from '@models/classes';\r\nimport { ITrackedClass } from '@models/interfaces/tracked-class';\r\nimport { SyncService } from '@models/services';\r\nimport { KbViewer } from '@view/kb-viewer';\r\n\r\n/** The maxiumum amount of allowed time to spend in a single render cycle, to avoid locking the browser */\r\nexport const MAX_RENDER_TIME = 20000;\r\n\r\nexport abstract class Renderer {\r\n constructor(protected viewer: KbViewer, protected scene: Scene) {}\r\n\r\n public babylonMap = new Map();\r\n\r\n /**\r\n * The function to update the renderer scene for the given renderer type\r\n * @param node The node to update\r\n * @returns nothing\r\n */\r\n protected abstract update3dInternal(node: ITrackedClass): void;\r\n\r\n /**\r\n * Token that defines the type or base type of model that this\r\n * renderer processes\r\n */\r\n public abstract modelToken: Token;\r\n\r\n /**\r\n * Determines whether the given node should be handled by this renderer. Can be\r\n * overriden for special behavior\r\n */\r\n public handlesNode(node: Kb3dObject): boolean {\r\n return instanceOf(node, this.modelToken);\r\n }\r\n\r\n public update3d(node: ITrackedClass) {\r\n this.update3dInternal(node);\r\n if (this.viewer.options.validator) {\r\n this.viewer.options.validator.validateNode(node);\r\n }\r\n }\r\n\r\n public delete3d(node: ITrackedClass) {\r\n const o = this.babylonMap.get(node._dynamicId);\r\n if (o) {\r\n if (o instanceof Material) {\r\n o.dispose(true, true);\r\n } else {\r\n o.dispose();\r\n }\r\n }\r\n this.babylonMap.delete(node._dynamicId);\r\n\r\n SyncService.nodeDeletionProcessed(node);\r\n }\r\n\r\n public get3d(node: ITrackedClass | T): T3D | undefined {\r\n return this.babylonMap.get(node._dynamicId);\r\n }\r\n\r\n public set3d(node: ITrackedClass | T, babylonObject: T3D) {\r\n this.babylonMap.set(node._dynamicId, babylonObject);\r\n }\r\n\r\n public getChanges(node: ITrackedClass) {\r\n const isNew = this.get3d(node) == null;\r\n const changes = new Map(node.changes);\r\n if (isNew) {\r\n for (const prop in node) {\r\n changes.set(prop as Extract, node[prop as keyof T]);\r\n }\r\n }\r\n return changes;\r\n }\r\n\r\n public dispose() {\r\n this.babylonMap.clear();\r\n }\r\n}\r\n","import { PBRMaterial } from '@babylonjs/core/Materials/PBR/pbrMaterial';\r\nimport { DynamicTexture } from '@babylonjs/core/Materials/Textures/dynamicTexture';\r\nimport { Color3, Color4, Plane, Quaternion, Vector3, Vector4 } from '@babylonjs/core/Maths';\r\nimport { PlaneBuilder } from '@babylonjs/core/Meshes/Builders/planeBuilder';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { LinesMesh } from '@babylonjs/core/Meshes/linesMesh';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { MeshBuilder } from '@babylonjs/core/Meshes/meshBuilder';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { getBinClampedSize } from '@common-util/texture-util';\r\nimport { AnnotationNode } from '@models/classes/annotations';\r\nimport { eAnnotationLineStyle } from '@models/classes/enums';\r\nimport { Connector } from '@models/classes/features/connector';\r\nimport { Kb3dObject } from '@models/classes/kb3d-object';\r\nimport { SceneNode } from '@models/classes/scene-node';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass } from '@models/interfaces';\r\nimport { getCameraPositionWithOffset, rotationToVector, roundToPlaces, transformVectorWithQuat } from '@view/helpers';\r\nimport { Subscription } from 'rxjs';\r\nimport { KbViewer } from '../kb-viewer';\r\nimport { Renderer } from './renderer';\r\nimport { RendererManager } from './renderer-manager';\r\n\r\nconst TEXT_RESOLUTION = 128;\r\nexport interface ILineMeshOptions {\r\n points: Vector3[];\r\n updatable?: boolean;\r\n instance?: LinesMesh;\r\n colors?: Color4[];\r\n dashSize?: number;\r\n gapSize?: number;\r\n dashNb?: number;\r\n}\r\nexport interface AnnotationParts {\r\n line?: ILineMeshOptions;\r\n textMesh: AbstractMesh;\r\n textMaterial: PBRMaterial;\r\n textContainer: TransformNode;\r\n rootMesh: TransformNode;\r\n billboard: boolean;\r\n}\r\n\r\nexport interface AnnotationInfo {\r\n meshInfo?: AnnotationParts;\r\n boundingInfo?: {\r\n targetNode: Kb3dObject;\r\n subscription: Subscription;\r\n };\r\n positionNeedsUpdate: boolean;\r\n textNeedsUpdate: boolean;\r\n lineNeedsUpdate: boolean;\r\n}\r\n\r\nexport class AnnotationRenderer extends Renderer {\r\n utilityLayer: UtilityLayerRenderer;\r\n modelToken = Token.AnnotationNode;\r\n annotationStore = new Map();\r\n billboards = new Map();\r\n\r\n constructor(protected viewer: KbViewer, protected scene: Scene, protected rendererManager: RendererManager) {\r\n super(viewer, scene);\r\n this.utilityLayer = viewer.utilityLayer;\r\n scene.registerBeforeRender(() => {\r\n if (scene.activeCamera && this.billboards.size > 0) {\r\n this.billboards.forEach(meshInfo => {\r\n const camera = scene.activeCamera!;\r\n meshInfo.rootMesh.lookAt(getCameraPositionWithOffset(camera), 0, Math.PI / 2);\r\n if (meshInfo.rootMesh.parent) {\r\n if (!meshInfo.rootMesh.rotationQuaternion) {\r\n meshInfo.rootMesh.rotationQuaternion = Quaternion.Identity();\r\n }\r\n const parentRotation = meshInfo.rootMesh.parent.computeWorldMatrix().getRotationMatrix();\r\n parentRotation.invert();\r\n const parentRotationQuat = Quaternion.FromRotationMatrix(parentRotation);\r\n meshInfo.rootMesh.rotationQuaternion = parentRotationQuat.multiplyInPlace(\r\n meshInfo.rootMesh.rotationQuaternion\r\n );\r\n }\r\n });\r\n }\r\n });\r\n }\r\n\r\n update3dInternal(node: ITrackedClass) {\r\n let annotationInfo = this.annotationStore.get(node._dynamicId);\r\n if (!annotationInfo) {\r\n annotationInfo = {\r\n positionNeedsUpdate: false,\r\n textNeedsUpdate: false,\r\n lineNeedsUpdate: false,\r\n };\r\n this.annotationStore.set(node._dynamicId, annotationInfo);\r\n }\r\n\r\n if (annotationInfo.meshInfo && node.changes.has('keepOnTop')) {\r\n this.delete3d(node, true);\r\n annotationInfo.meshInfo = undefined;\r\n annotationInfo.positionNeedsUpdate = true;\r\n annotationInfo.textNeedsUpdate = true;\r\n annotationInfo.lineNeedsUpdate = true;\r\n }\r\n const scene = node.keepOnTop ? this.utilityLayer.utilityLayerScene : this.scene;\r\n\r\n if (\r\n (!annotationInfo.meshInfo && node.enabled && node.targetId) ||\r\n (node.changes.hasAny('isPickable') && node.enabled && node.targetId)\r\n ) {\r\n if (annotationInfo.meshInfo) {\r\n annotationInfo.meshInfo = undefined;\r\n }\r\n annotationInfo.meshInfo = createAnnotationTextModels(node, scene);\r\n\r\n // Put the annotation on the parent scene so the scene visibility affects the node\r\n if (node._parentScene) {\r\n const parentSceneMesh = this.viewer.getRendererNode(node._parentScene);\r\n annotationInfo.meshInfo.rootMesh.setParent(parentSceneMesh);\r\n }\r\n\r\n this.set3d(node, annotationInfo.meshInfo.rootMesh);\r\n }\r\n const meshInfo = annotationInfo.meshInfo;\r\n\r\n // Set position of annotation\r\n if (node.changes.hasAny('targetId', 'offset', 'billboard', 'horizontalBoxAlign', 'verticalBoxAlign')) {\r\n annotationInfo.positionNeedsUpdate = true;\r\n }\r\n if (node.changes.hasAny('color', 'lineStyle')) {\r\n annotationInfo.lineNeedsUpdate = true;\r\n }\r\n if (\r\n node.changes.hasAny(\r\n 'fontFamily',\r\n 'fontSize',\r\n 'lineSpacing',\r\n 'label',\r\n 'color',\r\n 'preloadFontStylesheet',\r\n 'horizontalTextAlign'\r\n )\r\n ) {\r\n annotationInfo.textNeedsUpdate = true;\r\n }\r\n\r\n if (meshInfo) {\r\n if (node.changes.hasAny('enabled', 'billboard', 'keepOnTop')) {\r\n if (node.enabled && node.billboard) {\r\n this.billboards.set(node._dynamicId, meshInfo);\r\n } else {\r\n this.billboards.delete(node._dynamicId);\r\n }\r\n }\r\n\r\n let needsFont = false;\r\n if (node.preloadFontStylesheet && node.fontFamily) {\r\n const fontService = node._manager!.getService(Token.FontService);\r\n\r\n if (!fontService.readyFonts.has(node.fontFamily)) {\r\n needsFont = true;\r\n fontService.loadFont(node.preloadFontStylesheet, [node.fontFamily]).then(resultFonts => {\r\n if (resultFonts.length > 0) {\r\n node.setPropertyIsModified('fontFamily', true);\r\n }\r\n });\r\n }\r\n }\r\n\r\n if (node.enabled && node.targetId && !needsFont) {\r\n const color = node.color.toColor4();\r\n\r\n if (annotationInfo.positionNeedsUpdate) {\r\n let targetNode: Kb3dObject | undefined;\r\n if (node.targetId && node._manager) {\r\n const manager = node._manager;\r\n targetNode = manager.getById(node.targetId);\r\n }\r\n\r\n if (\r\n annotationInfo.boundingInfo &&\r\n annotationInfo.boundingInfo.targetNode._dynamicId !== targetNode?._dynamicId\r\n ) {\r\n annotationInfo.boundingInfo.subscription.unsubscribe();\r\n }\r\n if (targetNode) {\r\n if (targetNode instanceof Connector) {\r\n // Only supporting connectors for now\r\n if (targetNode._parent) {\r\n annotationInfo.boundingInfo = {\r\n targetNode: targetNode,\r\n subscription: this.viewer.bbCache\r\n .watchForChanges(targetNode._parent)\r\n .subscribe(() => {\r\n node.setPropertyIsModified('targetId', true);\r\n }),\r\n };\r\n }\r\n }\r\n\r\n const transform = this.updatePositionFromAnnotation(node, targetNode, meshInfo);\r\n\r\n if (transform) {\r\n meshInfo.rootMesh.position = transform.position;\r\n // The annotation plane mesh position doesn't update in certain situations when the parent transform is moved,\r\n // so we have to manually update the world matrix\r\n meshInfo.rootMesh.getChildMeshes().forEach(mesh => mesh.computeWorldMatrix(true));\r\n meshInfo.textContainer.rotationQuaternion = transform.textRotation;\r\n if (!node.billboard) {\r\n meshInfo.rootMesh.rotationQuaternion = transform.rotation;\r\n }\r\n\r\n if (transform && transform.offset && transform.lineOffset && node.offset.length() > 0) {\r\n const basePoint = transform.position.subtract(transform.offset);\r\n meshInfo.line = {\r\n points: [basePoint, basePoint.add(transform.lineOffset)],\r\n updatable: true,\r\n colors: [color, color],\r\n instance: meshInfo.line?.instance,\r\n };\r\n\r\n // If the line is not getting rebuilt, then just update the line position\r\n if (!annotationInfo.lineNeedsUpdate) {\r\n meshInfo.line.instance = drawLine(node, meshInfo.line, 1, scene);\r\n }\r\n } else {\r\n if (meshInfo.line?.instance) {\r\n meshInfo.line.instance.dispose();\r\n }\r\n meshInfo.line = undefined;\r\n }\r\n }\r\n\r\n annotationInfo.positionNeedsUpdate = false;\r\n }\r\n }\r\n\r\n if (annotationInfo.lineNeedsUpdate) {\r\n if (meshInfo.line) {\r\n if (meshInfo.line.instance) {\r\n meshInfo.line.instance.dispose();\r\n meshInfo.line.instance = undefined;\r\n }\r\n\r\n meshInfo.line.colors = [];\r\n for (let i = 0; i < meshInfo.line.points.length; i++) {\r\n meshInfo.line.colors.push(color);\r\n }\r\n meshInfo.line.instance = drawLine(node, meshInfo.line, 1, scene);\r\n if (meshInfo.line.instance && node._parentScene) {\r\n const parentSceneMesh = this.viewer.getRendererNode(node._parentScene);\r\n meshInfo.line.instance.parent = parentSceneMesh;\r\n }\r\n\r\n annotationInfo.lineNeedsUpdate = false;\r\n }\r\n }\r\n\r\n if (annotationInfo.textNeedsUpdate) {\r\n updateAnnotationText(\r\n node,\r\n meshInfo.textMesh,\r\n meshInfo.textMaterial,\r\n this.scene.getEngine().webGLVersion < 2\r\n );\r\n annotationInfo.textNeedsUpdate = false;\r\n }\r\n }\r\n\r\n const enabled = node.enabled && !!node.targetId;\r\n\r\n if (meshInfo.rootMesh.isEnabled() !== enabled) {\r\n meshInfo.rootMesh.setEnabled(enabled);\r\n meshInfo.line?.instance?.setEnabled(enabled);\r\n }\r\n }\r\n }\r\n\r\n dispose() {\r\n this.billboards.clear();\r\n this.annotationStore.clear();\r\n super.dispose();\r\n }\r\n\r\n delete3d(node: ITrackedClass, keepInStore = false) {\r\n const annotationModels = this.annotationStore.get(node._dynamicId);\r\n if (annotationModels && annotationModels.meshInfo) {\r\n annotationModels.meshInfo.rootMesh.dispose();\r\n if (annotationModels.meshInfo.line?.instance) {\r\n annotationModels.meshInfo.line.instance.dispose();\r\n }\r\n }\r\n if (!keepInStore) {\r\n this.annotationStore.delete(node._dynamicId);\r\n this.billboards.delete(node._dynamicId);\r\n }\r\n super.delete3d(node);\r\n }\r\n\r\n private updatePositionFromAnnotation(\r\n node: ITrackedClass,\r\n targetNode: Kb3dObject,\r\n meshInfo?: AnnotationParts\r\n ) {\r\n let newPosition: Vector3 | undefined;\r\n let newDirection: Vector3 | undefined;\r\n let newAngle: number | undefined;\r\n\r\n // Only supporting connectors for now, but could support direct mesh targets in the future\r\n if (targetNode instanceof Connector) {\r\n newPosition = getPositionFromConnnector(targetNode, this.rendererManager, node._parentScene);\r\n newDirection = targetNode.direction.toVec3();\r\n newAngle = targetNode.angle;\r\n }\r\n if (newPosition && newDirection && newAngle !== undefined) {\r\n const newOffset = node.offset.toVec3();\r\n if (meshInfo) {\r\n const newPivot = new Vector3(0, 0, 0);\r\n if (node.horizontalBoxAlign == 'left') {\r\n meshInfo.rootMesh.getPivotPoint();\r\n newOffset.x += meshInfo.textMesh.scaling.x / 2;\r\n newPivot.x -= meshInfo.textMesh.scaling.x / 2;\r\n } else if (node.horizontalBoxAlign == 'right') {\r\n newOffset.x -= meshInfo.textMesh.scaling.x / 2;\r\n newPivot.x += meshInfo.textMesh.scaling.x / 2;\r\n }\r\n if (node.verticalBoxAlign == 'top') {\r\n newOffset.z += meshInfo.textMesh.scaling.z / 2;\r\n newPivot.z -= meshInfo.textMesh.scaling.z / 2;\r\n } else if (node.verticalBoxAlign == 'bottom') {\r\n newOffset.z -= meshInfo.textMesh.scaling.z / 2;\r\n newPivot.z += meshInfo.textMesh.scaling.z / 2;\r\n }\r\n meshInfo.rootMesh.setPivotPoint(newPivot);\r\n }\r\n\r\n const transform = calculateTransform(newPosition, newDirection, newAngle, newOffset, node.offset.toVec3());\r\n return transform;\r\n }\r\n }\r\n}\r\n\r\nexport function createAnnotationTextModels(node: AnnotationNode, scene: Scene, isPickable = false) {\r\n const containerMesh = new TransformNode(node._dynamicId, scene);\r\n\r\n //Create plane\r\n const plane = PlaneBuilder.CreatePlane(\r\n `${node._dynamicId}_plane`,\r\n {\r\n width: 1,\r\n height: 1,\r\n frontUVs: new Vector4(1, 0, 0, 1),\r\n backUVs: new Vector4(0, 0, 1, 1),\r\n sideOrientation: Mesh.DOUBLESIDE,\r\n sourcePlane: new Plane(0, 1, 0, 0),\r\n },\r\n scene\r\n );\r\n const planeContainer = new TransformNode(`${node._dynamicId}_planeContainer`);\r\n\r\n const dynamicTexture = new DynamicTexture(\r\n `${node._dynamicId}_planeTexture`,\r\n { width: TEXT_RESOLUTION, height: TEXT_RESOLUTION },\r\n scene,\r\n true\r\n );\r\n dynamicTexture.isBlocking = false;\r\n dynamicTexture.hasAlpha = true;\r\n\r\n //create material\r\n const mat = new PBRMaterial(`${node._dynamicId}_mat`, scene);\r\n if (node.keepOnTop) {\r\n mat.emissiveColor = Color3.White();\r\n }\r\n mat.albedoTexture = dynamicTexture;\r\n mat.useAlphaFromAlbedoTexture = true;\r\n mat.backFaceCulling = true;\r\n mat.zOffset = -1;\r\n mat.albedoColor = Color3.White();\r\n mat.roughness = 1;\r\n mat.metallic = 0;\r\n\r\n //apply material\r\n plane.material = mat;\r\n plane.parent = planeContainer;\r\n planeContainer.parent = containerMesh;\r\n plane.isPickable = isPickable || node.isPickable;\r\n\r\n const annotationModels = {\r\n textMesh: plane,\r\n textContainer: planeContainer,\r\n textMaterial: mat,\r\n rootMesh: containerMesh,\r\n billboard: node.billboard,\r\n positionNeedsUpdate: false,\r\n textNeedsUpdate: false,\r\n };\r\n\r\n return annotationModels;\r\n}\r\n\r\nexport function updateAnnotationText(\r\n node: AnnotationNode,\r\n mesh: AbstractMesh,\r\n textMaterial: PBRMaterial,\r\n webgl1Context: boolean,\r\n defaultValue = ' ',\r\n values: any = {}\r\n) {\r\n const mat = textMaterial;\r\n const dynamicTexture = mat.albedoTexture as DynamicTexture;\r\n const fontFamily = node.fontFamily || 'Arial';\r\n const labelSize = node.fontSize;\r\n const labelText = evalHandlebarString(node.label || defaultValue, values);\r\n const textGap = node.lineSpacing;\r\n const textColor = node.color.toColor3().toLinearSpace();\r\n if (node.keepOnTop) {\r\n mat.emissiveColor = textColor;\r\n }\r\n\r\n // Determine the space needed for the text\r\n const ctx = dynamicTexture.getContext();\r\n ctx.font = TEXT_RESOLUTION + 'px ' + fontFamily;\r\n const labelStats = getMultiLineStats(labelText, ctx as CanvasRenderingContext2D);\r\n const textWidth = labelStats.maximumWidth;\r\n const textHeight = labelStats.maximumHeight;\r\n const labelArray = labelStats.labelArray;\r\n const lineWidths = labelStats.lineWidths;\r\n labelArray.reverse();\r\n lineWidths.reverse();\r\n const totalCanvasHeight = (textHeight + textGap) * labelArray.length - textGap;\r\n ctx.clearRect(0 - textWidth / 2, 0, textWidth / 2, totalCanvasHeight);\r\n //ctx.clearRect(0, 0, textWidth, totalCanvasHeight);\r\n let heightRatioChange = 1;\r\n if (labelArray.length > 1) {\r\n heightRatioChange = totalCanvasHeight / textHeight;\r\n }\r\n mesh.scaling = new Vector3((textWidth / textHeight) * labelSize, labelSize * heightRatioChange, 1);\r\n\r\n let textBaseline: CanvasTextBaseline = 'bottom';\r\n if (labelStats.maximumDescent > 0) {\r\n textBaseline = 'alphabetic';\r\n }\r\n let drawFont: string;\r\n if (webgl1Context) {\r\n let textureSize = getBinClampedSize(textWidth, totalCanvasHeight, 'ceil');\r\n if (textureSize > 2048) {\r\n textureSize = 2048;\r\n }\r\n dynamicTexture.scaleTo(textureSize, textureSize);\r\n ctx.scale(textureSize / textWidth, textureSize / totalCanvasHeight);\r\n drawFont = TEXT_RESOLUTION + 'px ' + fontFamily;\r\n } else {\r\n dynamicTexture.scaleTo(textWidth, totalCanvasHeight);\r\n drawFont = TEXT_RESOLUTION + 'px ' + fontFamily;\r\n }\r\n ctx.font = drawFont;\r\n if (ctx instanceof CanvasRenderingContext2D) {\r\n ctx.textBaseline = textBaseline;\r\n }\r\n ctx.fillStyle = textColor.toHexString();\r\n\r\n let drawHeight = totalCanvasHeight;\r\n for (let i = 0; i < labelArray.length; i++) {\r\n const lineWidth = lineWidths[i];\r\n //ctx.fillStyle = 'rgb(132,244,235, 0.3)';\r\n //ctx.fillRect(0, 0, textWidth, (textHeight + textGap) * labelArray.length - textGap);\r\n //rightAllign\r\n if (node.horizontalTextAlign === 'right') {\r\n ctx.fillText(labelArray[i], textWidth - lineWidth, drawHeight - labelStats.maximumDescent);\r\n }\r\n //leftAllign\r\n else if (node.horizontalTextAlign === 'left') {\r\n ctx.fillText(labelArray[i], 0, drawHeight - labelStats.maximumDescent);\r\n } else {\r\n //centered:\r\n ctx.fillText(labelArray[i], textWidth / 2 - lineWidth / 2, drawHeight - labelStats.maximumDescent);\r\n }\r\n drawHeight = drawHeight - textHeight - textGap;\r\n }\r\n // reset the context\r\n ctx.setTransform(1, 0, 0, 1, 0, 0);\r\n dynamicTexture.update(true);\r\n}\r\n\r\nexport function getPositionFromConnnector(\r\n targetNode: Connector,\r\n rendererManager: RendererManager,\r\n relativeToScene: SceneNode | undefined\r\n) {\r\n const targetMesh = rendererManager.connector.get3d(targetNode);\r\n if (targetMesh) {\r\n targetMesh.computeWorldMatrix(true);\r\n const result = targetMesh.absolutePosition.clone();\r\n\r\n if (relativeToScene) {\r\n const sceneMesh = rendererManager.viewer.getRendererNode(relativeToScene);\r\n if (sceneMesh) {\r\n // console.log('pre', newPosition, newDirection, newAngle);\r\n const transform = sceneMesh.getWorldMatrix().clone();\r\n transform.invert();\r\n return Vector3.TransformCoordinates(result, transform);\r\n }\r\n }\r\n return result;\r\n }\r\n return Vector3.Zero();\r\n}\r\n\r\nexport function calculateTransform(\r\n position: Vector3,\r\n direction: Vector3,\r\n angle: number,\r\n offset?: Vector3,\r\n lineOffset?: Vector3\r\n) {\r\n const newPosition = position.clone();\r\n const newRotation = rotationToVector(direction);\r\n const textRotation = Quaternion.RotationAxis(Vector3.Up(), (angle - 90) * (Math.PI / 180));\r\n let offsetWithTransform: Vector3 | undefined;\r\n let lineOffsetWithTransform: Vector3 | undefined;\r\n\r\n if (offset) {\r\n offsetWithTransform = transformVectorWithQuat(offset, newRotation);\r\n newPosition.addInPlace(offsetWithTransform);\r\n }\r\n\r\n if (lineOffset) {\r\n lineOffsetWithTransform = transformVectorWithQuat(lineOffset, newRotation);\r\n }\r\n\r\n return {\r\n position: newPosition,\r\n rotation: newRotation,\r\n textRotation: textRotation,\r\n offset: offsetWithTransform,\r\n lineOffset: lineOffsetWithTransform,\r\n };\r\n}\r\n\r\nexport function drawLine(\r\n node: AnnotationNode,\r\n lineInfo: AnnotationParts['line'],\r\n lineLength: number,\r\n scene: Scene,\r\n distance?: number\r\n) {\r\n if (lineInfo && node.lineStyle !== eAnnotationLineStyle.none) {\r\n let linesMesh: LinesMesh;\r\n if (node.lineStyle === eAnnotationLineStyle.solid) {\r\n linesMesh = MeshBuilder.CreateLines(`${node._dynamicId}_line`, lineInfo, scene);\r\n } else {\r\n lineInfo.dashNb = lineLength * 30;\r\n if (node.lineStyle === eAnnotationLineStyle.dashed) {\r\n lineInfo.gapSize = 4;\r\n lineInfo.dashSize = 6;\r\n if (distance) {\r\n // lineInfo.gapSize *= distance;\r\n // lineInfo.dashSize *= distance;\r\n lineInfo.dashNb = (lineInfo.dashNb / distance) * 1.5;\r\n }\r\n } else {\r\n lineInfo.gapSize = 6;\r\n lineInfo.dashSize = 1;\r\n }\r\n linesMesh = MeshBuilder.CreateDashedLines(`${node._dynamicId}_line`, lineInfo, scene);\r\n linesMesh.color = node.color.toColor3();\r\n }\r\n lineInfo.instance = linesMesh;\r\n lineInfo.instance.isPickable = false;\r\n\r\n return linesMesh;\r\n }\r\n}\r\n\r\nfunction evalHandlebarString(label: string, values: any) {\r\n if (!label.match(/\\{.*\\}/)) {\r\n return label;\r\n }\r\n const array = label.matchAll(/\\{([\\d\\w+\\-*/]+)\\}/g);\r\n let exp = array.next();\r\n while (!exp.done) {\r\n let mathExp = exp.value[1];\r\n for (const key in values) {\r\n mathExp = mathExp.replace(key, values[key]);\r\n }\r\n if (mathExp.match(/\\d/)) {\r\n // Strip all invalid math expression characters\r\n mathExp = mathExp.replace(/[^0-9.+*/-]/g, '');\r\n const numberVal = mathEval(mathExp);\r\n label = label.replace(exp.value[0], roundToPlaces(numberVal, 0.01).toString(10));\r\n } else {\r\n label = label.replace(exp.value[0], mathExp);\r\n }\r\n exp = array.next();\r\n }\r\n\r\n return label;\r\n}\r\n\r\nfunction getMultiLineStats(label: string, ctx: CanvasRenderingContext2D) {\r\n // 402 is a florin and 96 is a backtick, generally the tallest and highest characters.\r\n const fullStringTextSize = ctx.measureText(label + String.fromCharCode(402, 71, 95, 96, 103));\r\n\r\n const substrings: string[] = label.split('\\\\n').map(label => label.trim());\r\n const lineWidths: number[] = [];\r\n let maximumWidth = 0;\r\n let maximumHeight = 0;\r\n let maximumDescent = 0;\r\n substrings.forEach(string => {\r\n if (string.length > maximumWidth) {\r\n maximumWidth = string.length;\r\n }\r\n const textSize = ctx.measureText(string);\r\n\r\n if (textSize.actualBoundingBoxLeft !== undefined && textSize.actualBoundingBoxRight !== undefined) {\r\n const textWidth = Math.abs(textSize.actualBoundingBoxLeft) + Math.abs(textSize.actualBoundingBoxRight);\r\n if (textWidth > maximumWidth) {\r\n maximumWidth = textWidth;\r\n }\r\n lineWidths.push(textWidth);\r\n } else {\r\n if (textSize.width > maximumWidth) {\r\n maximumWidth = textSize.width;\r\n }\r\n lineWidths.push(textSize.width);\r\n }\r\n });\r\n\r\n // The render puppetteer browser does not have fontBoundingBoxAscent or Descent.\r\n // That would be a better way to measure text for consistency. When upgrading puppeteer see if we can use fontBoundingBoxAscent\r\n if (\r\n fullStringTextSize.actualBoundingBoxAscent !== undefined &&\r\n fullStringTextSize.actualBoundingBoxDescent !== undefined\r\n ) {\r\n maximumHeight =\r\n Math.abs(fullStringTextSize.actualBoundingBoxAscent) +\r\n Math.abs(fullStringTextSize.actualBoundingBoxDescent);\r\n maximumDescent = fullStringTextSize.actualBoundingBoxDescent;\r\n }\r\n\r\n if (maximumHeight === 0) {\r\n maximumHeight = TEXT_RESOLUTION;\r\n }\r\n\r\n const result = {\r\n labelArray: substrings,\r\n lineWidths,\r\n maximumHeight,\r\n maximumWidth,\r\n maximumDescent,\r\n };\r\n return result;\r\n}\r\n\r\nlet loopCount = 0;\r\nfunction mathEval(exp: string) {\r\n loopCount++;\r\n if (loopCount > 1000) {\r\n console.warn('escaped infinite loop in math evaluator');\r\n return NaN;\r\n }\r\n if (!exp.match(/\\D/)) {\r\n return parseFloat(exp);\r\n }\r\n\r\n let result: number;\r\n if (exp.match(/\\+/)) {\r\n const parts = exp.split('+').reverse();\r\n result = mathEval(parts.pop()!);\r\n while (parts.length > 0) {\r\n result += mathEval(parts.pop()!);\r\n }\r\n } else if (exp.match(/-/)) {\r\n const parts = exp.split('-').reverse();\r\n result = mathEval(parts.pop()!);\r\n while (parts.length > 0) {\r\n result -= mathEval(parts.pop()!);\r\n }\r\n } else if (exp.match(/\\*/)) {\r\n const parts = exp.split('*').reverse();\r\n result = mathEval(parts.pop()!);\r\n while (parts.length > 0) {\r\n result *= mathEval(parts.pop()!);\r\n }\r\n } else if (exp.match(/\\//)) {\r\n const parts = exp.split('/').reverse();\r\n result = mathEval(parts.pop()!);\r\n while (parts.length > 0) {\r\n result /= mathEval(parts.pop()!);\r\n }\r\n } else {\r\n result = parseFloat(exp);\r\n }\r\n\r\n return result;\r\n}\r\n","import { Color4, Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { LinesMesh } from '@babylonjs/core/Meshes/linesMesh';\r\nimport { MeshBuilder } from '@babylonjs/core/Meshes/meshBuilder';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport {\r\n DimensionNode,\r\n Kb3dObject,\r\n eAnnotationLineStyle,\r\n eDimensionAxis,\r\n eDimensionLineCapStyle,\r\n} from '@models/classes';\r\nimport { AnnotationNode } from '@models/classes/annotations/annotation-node';\r\nimport { Connector } from '@models/classes/features/connector';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass } from '@models/interfaces/tracked-class';\r\nimport { getCameraPositionWithOffset, rotationToVector, transformVectorWithQuat } from '@view/helpers/math-helper';\r\nimport { Subscription, merge } from 'rxjs';\r\nimport { AnnotationInfo } from '.';\r\nimport { KbViewer } from '../kb-viewer';\r\nimport {\r\n AnnotationParts,\r\n ILineMeshOptions,\r\n calculateTransform,\r\n createAnnotationTextModels,\r\n drawLine,\r\n getPositionFromConnnector,\r\n updateAnnotationText,\r\n} from './annotation';\r\nimport { Renderer } from './renderer';\r\nimport { RendererManager } from './renderer-manager';\r\n\r\nexport interface ICapMeshOptions {\r\n lines: Vector3[][];\r\n updatable?: boolean;\r\n colors?: Color4[][];\r\n instance?: LinesMesh;\r\n}\r\n\r\ninterface DimensionParts extends AnnotationParts {\r\n line: ILineMeshOptions;\r\n lineCap: ICapMeshOptions;\r\n distance: number;\r\n}\r\n\r\nexport interface DimensionInfo extends AnnotationInfo {\r\n meshInfo?: DimensionParts;\r\n canCreate: boolean;\r\n boundingInfo?: {\r\n targetNode: Kb3dObject;\r\n targetNode2: Kb3dObject;\r\n subscription: Subscription;\r\n };\r\n}\r\n\r\n/**\r\n * Move line position calculation to private function so it can be called from bbSubscription\r\n * Move text font family creation stuff to common function and call it from here with midpoint of line\r\n * Store the line mesh in annotationStore, just recreate it from the private function when something changes\r\n */\r\n\r\nexport class DimensionRenderer extends Renderer {\r\n utilityLayer: UtilityLayerRenderer;\r\n modelToken = Token.DimensionNode;\r\n\r\n watchPosition = new Map<\r\n string,\r\n {\r\n node: ITrackedClass;\r\n containerMesh: TransformNode;\r\n targetNode: Kb3dObject;\r\n }\r\n >();\r\n dimensionStore = new Map();\r\n billboards = new Map();\r\n\r\n constructor(protected viewer: KbViewer, protected scene: Scene, protected rendererManager: RendererManager) {\r\n super(viewer, scene);\r\n this.utilityLayer = viewer.utilityLayer;\r\n scene.registerBeforeRender(() => {\r\n if (scene.activeCamera && this.dimensionStore.size > 0) {\r\n this.billboards.forEach(meshInfo => {\r\n const camera = scene.activeCamera!;\r\n meshInfo.rootMesh.lookAt(getCameraPositionWithOffset(camera), 0, Math.PI / 2);\r\n });\r\n }\r\n });\r\n }\r\n\r\n update3dInternal(node: ITrackedClass) {\r\n let dimensionInfo = this.dimensionStore.get(node._dynamicId);\r\n if (!dimensionInfo) {\r\n dimensionInfo = {\r\n positionNeedsUpdate: false,\r\n textNeedsUpdate: false,\r\n lineNeedsUpdate: false,\r\n canCreate: false,\r\n };\r\n this.dimensionStore.set(node._dynamicId, dimensionInfo);\r\n }\r\n\r\n const rootMesh = dimensionInfo.meshInfo?.rootMesh;\r\n if (rootMesh && node.scaleWithCamera) {\r\n if (!node.scaleWithCamera) {\r\n this.viewer.camera.clearMaintainedNodeScreenSize(rootMesh);\r\n } else {\r\n this.viewer.camera.maintainNodeScreenSize(rootMesh, (dist: number, mode: number) => {\r\n if (dimensionInfo && dimensionInfo.meshInfo && node.lineStyle === eAnnotationLineStyle.dashed) {\r\n this.updateLineFromCamera(node, dimensionInfo.meshInfo, scene, dist);\r\n }\r\n });\r\n }\r\n }\r\n\r\n if (dimensionInfo.meshInfo && node.changes.has('keepOnTop')) {\r\n this.delete3d(node, true);\r\n dimensionInfo.meshInfo = undefined;\r\n dimensionInfo.positionNeedsUpdate = true;\r\n dimensionInfo.textNeedsUpdate = true;\r\n dimensionInfo.lineNeedsUpdate = true;\r\n }\r\n const scene = node.keepOnTop ? this.utilityLayer.utilityLayerScene : this.scene;\r\n\r\n if (!dimensionInfo.meshInfo && node.enabled && node.targetId) {\r\n dimensionInfo.meshInfo = {\r\n ...createAnnotationTextModels(node, scene, this.viewer.options.editorMode),\r\n line: {\r\n points: [],\r\n colors: [],\r\n updatable: true,\r\n },\r\n lineCap: {\r\n lines: [],\r\n colors: [],\r\n updatable: true,\r\n },\r\n distance: 0,\r\n };\r\n\r\n // Put the dimensions on the parent scene so the scene visibility affects the node\r\n if (node._parentScene) {\r\n const parentSceneMesh = this.viewer.getRendererNode(node._parentScene);\r\n dimensionInfo.meshInfo.rootMesh.setParent(parentSceneMesh);\r\n }\r\n\r\n this.set3d(node, dimensionInfo.meshInfo.rootMesh);\r\n }\r\n const meshInfo = dimensionInfo.meshInfo;\r\n // Set position of annotation\r\n if (\r\n node.changes.hasAny(\r\n 'enabled',\r\n 'targetId',\r\n 'target2Id',\r\n 'offset',\r\n 'billboard',\r\n 'labelOffset',\r\n 'axis',\r\n 'dimensionGap'\r\n )\r\n ) {\r\n dimensionInfo.positionNeedsUpdate = true;\r\n }\r\n\r\n if (node.changes.hasAny('color', 'lineStyle', 'lineCapStyle')) {\r\n dimensionInfo.lineNeedsUpdate = true;\r\n }\r\n\r\n if (\r\n node.changes.hasAny(\r\n 'fontFamily',\r\n 'fontSize',\r\n 'label',\r\n 'color',\r\n 'targetId',\r\n 'target2Id',\r\n 'dimensionUnitPrecision'\r\n )\r\n ) {\r\n dimensionInfo.textNeedsUpdate = true;\r\n }\r\n\r\n if (meshInfo) {\r\n if (node.changes.hasAny('enabled', 'billboard', 'keepOnTop')) {\r\n if (node.enabled && node.billboard) {\r\n this.billboards.set(node._dynamicId, meshInfo);\r\n } else {\r\n this.billboards.delete(node._dynamicId);\r\n }\r\n }\r\n\r\n if (node.enabled) {\r\n if (dimensionInfo.positionNeedsUpdate || dimensionInfo.lineNeedsUpdate) {\r\n dimensionInfo.canCreate = this.updateDimensionControlPoints(node, dimensionInfo);\r\n\r\n // If the line is not getting rebuilt, then just update the line position\r\n if (dimensionInfo.canCreate && !dimensionInfo.lineNeedsUpdate) {\r\n meshInfo.line.instance = drawLine(node, meshInfo.line, 1, scene);\r\n meshInfo.lineCap.instance = MeshBuilder.CreateLineSystem(\r\n `${node._dynamicId}_lineCap`,\r\n meshInfo.lineCap,\r\n scene\r\n );\r\n }\r\n\r\n dimensionInfo.positionNeedsUpdate = false;\r\n }\r\n\r\n if (dimensionInfo.canCreate) {\r\n if (dimensionInfo.lineNeedsUpdate) {\r\n if (meshInfo.line.instance) {\r\n meshInfo.line.instance.dispose();\r\n delete meshInfo.line.instance;\r\n }\r\n if (meshInfo.lineCap.instance) {\r\n meshInfo.lineCap.instance.dispose();\r\n delete meshInfo.lineCap.instance;\r\n }\r\n\r\n this.updateDimensionLineColor(node, meshInfo.line, meshInfo.lineCap);\r\n\r\n meshInfo.line.instance = drawLine(node, meshInfo.line, meshInfo.distance, scene);\r\n\r\n if (node.lineCapStyle !== eDimensionLineCapStyle.none) {\r\n const lineCapsMesh = MeshBuilder.CreateLineSystem(\r\n `${node._dynamicId}_lineCap`,\r\n meshInfo.lineCap,\r\n scene\r\n );\r\n if (!meshInfo.lineCap.instance) {\r\n meshInfo.lineCap.instance = lineCapsMesh;\r\n }\r\n }\r\n\r\n // Put the lines on the parent scene so they get it's visibility properties\r\n if (node._parentScene) {\r\n const parentSceneMesh = this.viewer.getRendererNode(node._parentScene);\r\n if (meshInfo.line.instance) {\r\n meshInfo.line.instance.parent = parentSceneMesh;\r\n }\r\n if (meshInfo.lineCap.instance) {\r\n meshInfo.lineCap.instance.parent = parentSceneMesh;\r\n }\r\n }\r\n\r\n dimensionInfo.lineNeedsUpdate = false;\r\n }\r\n\r\n if (dimensionInfo.textNeedsUpdate) {\r\n let distanceDefaultVal: string;\r\n if (\r\n node._parentScene &&\r\n node._parentScene.dimensionUnitScalar &&\r\n node.dimensionUnitPrecision !== undefined\r\n ) {\r\n let distance = node._parentScene.dimensionUnitScalar * meshInfo.distance;\r\n const precisionMultiplier = Math.pow(10, Math.abs(node.dimensionUnitPrecision));\r\n distance =\r\n Math.round((distance + Number.EPSILON) * precisionMultiplier) / precisionMultiplier;\r\n distanceDefaultVal = distance + (node._parentScene.dimensionUnitSymbol || '');\r\n } else {\r\n let distance = meshInfo.distance;\r\n const precisionMultiplier = Math.pow(10, Math.abs(node.dimensionUnitPrecision));\r\n distance =\r\n Math.round((distance + Number.EPSILON) * precisionMultiplier) / precisionMultiplier;\r\n distanceDefaultVal = distance.toString(10);\r\n }\r\n updateAnnotationText(\r\n node,\r\n meshInfo.textMesh,\r\n meshInfo.textMaterial,\r\n this.scene.getEngine().webGLVersion < 2,\r\n distanceDefaultVal,\r\n { v: meshInfo.distance }\r\n );\r\n\r\n dimensionInfo.textNeedsUpdate = false;\r\n }\r\n }\r\n }\r\n\r\n if (meshInfo.rootMesh.isEnabled() !== node.enabled) {\r\n meshInfo.rootMesh.setEnabled(node.enabled);\r\n meshInfo.line.instance?.setEnabled(node.enabled);\r\n meshInfo.lineCap.instance?.setEnabled(node.enabled);\r\n }\r\n }\r\n }\r\n\r\n delete3d(node: ITrackedClass, keepInStore = false) {\r\n const dimensionModels = this.dimensionStore.get(node._dynamicId);\r\n if (dimensionModels?.meshInfo) {\r\n this.deleteLine(dimensionModels.meshInfo);\r\n }\r\n if (!keepInStore) {\r\n this.dimensionStore.delete(node._dynamicId);\r\n this.billboards.delete(node._dynamicId);\r\n }\r\n super.delete3d(node);\r\n }\r\n\r\n dispose() {\r\n this.billboards.clear();\r\n this.dimensionStore.clear();\r\n super.dispose();\r\n }\r\n\r\n private deleteLine(dimensionModels: DimensionParts) {\r\n if (dimensionModels.line.instance) {\r\n dimensionModels.line.instance.dispose();\r\n delete dimensionModels.line.instance;\r\n }\r\n if (dimensionModels.lineCap.instance) {\r\n dimensionModels.lineCap.instance.dispose();\r\n delete dimensionModels.lineCap.instance;\r\n }\r\n }\r\n\r\n private updateLine(\r\n dimensionInfo: DimensionInfo,\r\n node: ITrackedClass,\r\n meshInfo: DimensionParts,\r\n scene: Scene\r\n ) {\r\n dimensionInfo.canCreate = this.updateDimensionControlPoints(node, dimensionInfo);\r\n\r\n // If the line is not getting rebuilt, then just update the line position\r\n if (dimensionInfo.canCreate && !dimensionInfo.lineNeedsUpdate) {\r\n meshInfo.line.instance = drawLine(node, meshInfo.line, 1, scene);\r\n meshInfo.lineCap.instance = MeshBuilder.CreateLineSystem(\r\n `${node._dynamicId}_lineCap`,\r\n meshInfo.lineCap,\r\n scene\r\n );\r\n }\r\n\r\n dimensionInfo.positionNeedsUpdate = false;\r\n if (dimensionInfo.canCreate) {\r\n if (dimensionInfo.lineNeedsUpdate) {\r\n if (meshInfo.line.instance) {\r\n meshInfo.line.instance.dispose();\r\n delete meshInfo.line.instance;\r\n }\r\n if (meshInfo.lineCap.instance) {\r\n meshInfo.lineCap.instance.dispose();\r\n delete meshInfo.lineCap.instance;\r\n }\r\n\r\n this.updateDimensionLineColor(node, meshInfo.line, meshInfo.lineCap);\r\n\r\n meshInfo.line.instance = drawLine(node, meshInfo.line, meshInfo.distance, scene);\r\n\r\n if (node.lineCapStyle !== eDimensionLineCapStyle.none) {\r\n const lineCapsMesh = MeshBuilder.CreateLineSystem(\r\n `${node._dynamicId}_lineCap`,\r\n meshInfo.lineCap,\r\n scene\r\n );\r\n if (!meshInfo.lineCap.instance) {\r\n meshInfo.lineCap.instance = lineCapsMesh;\r\n }\r\n }\r\n\r\n // Put the lines on the parent scene so they get it's visibility properties\r\n if (node._parentScene) {\r\n const parentSceneMesh = this.viewer.getRendererNode(node._parentScene);\r\n if (meshInfo.line.instance) {\r\n meshInfo.line.instance.parent = parentSceneMesh;\r\n }\r\n if (meshInfo.lineCap.instance) {\r\n meshInfo.lineCap.instance.parent = parentSceneMesh;\r\n }\r\n }\r\n\r\n dimensionInfo.lineNeedsUpdate = false;\r\n }\r\n\r\n if (dimensionInfo.textNeedsUpdate) {\r\n let distanceDefaultVal: string;\r\n if (\r\n node._parentScene &&\r\n node._parentScene.dimensionUnitScalar &&\r\n node.dimensionUnitPrecision !== undefined\r\n ) {\r\n let distance = node._parentScene.dimensionUnitScalar * meshInfo.distance;\r\n const precisionMultiplier = Math.pow(10, Math.abs(node.dimensionUnitPrecision));\r\n distance = Math.round((distance + Number.EPSILON) * precisionMultiplier) / precisionMultiplier;\r\n distanceDefaultVal = distance + (node._parentScene.dimensionUnitSymbol || '');\r\n } else {\r\n let distance = meshInfo.distance;\r\n const precisionMultiplier = Math.pow(10, Math.abs(node.dimensionUnitPrecision));\r\n distance = Math.round((distance + Number.EPSILON) * precisionMultiplier) / precisionMultiplier;\r\n distanceDefaultVal = distance.toString(10);\r\n }\r\n updateAnnotationText(\r\n node,\r\n meshInfo.textMesh,\r\n meshInfo.textMaterial,\r\n this.scene.getEngine().webGLVersion < 2,\r\n distanceDefaultVal,\r\n { v: meshInfo.distance }\r\n );\r\n\r\n dimensionInfo.textNeedsUpdate = false;\r\n }\r\n }\r\n }\r\n\r\n private updateLineFromCamera(\r\n node: ITrackedClass,\r\n meshInfo: DimensionParts,\r\n scene: Scene,\r\n distance: number\r\n ) {\r\n if (meshInfo.line.instance) {\r\n meshInfo.line.instance.dispose();\r\n delete meshInfo.line.instance;\r\n }\r\n\r\n this.updateDimensionLineColor(node, meshInfo.line, meshInfo.lineCap);\r\n\r\n meshInfo.line.instance = drawLine(node, meshInfo.line, 1, scene, distance); //CreateDashedLines('lines', meshInfo.line);\r\n if (node._parentScene) {\r\n const parentSceneMesh = this.viewer.getRendererNode(node._parentScene);\r\n if (meshInfo.line.instance) {\r\n meshInfo.line.instance.parent = parentSceneMesh;\r\n }\r\n if (meshInfo.lineCap.instance) {\r\n meshInfo.lineCap.instance.parent = parentSceneMesh;\r\n }\r\n }\r\n }\r\n\r\n private updateDimensionControlPoints(node: ITrackedClass, dimensionInfo: DimensionInfo) {\r\n if (!dimensionInfo.meshInfo) {\r\n return false;\r\n }\r\n const dimensionModel = dimensionInfo.meshInfo;\r\n\r\n const lineOptions = dimensionModel.line;\r\n const annotationMesh = dimensionModel.rootMesh;\r\n let target1: Kb3dObject | undefined;\r\n let target2: Kb3dObject | undefined;\r\n\r\n if (node.targetId && node.target2Id && node._manager) {\r\n const manager = node._manager;\r\n target1 = manager.getById(node.targetId);\r\n target2 = manager.getById(node.target2Id);\r\n }\r\n\r\n if (\r\n dimensionInfo.boundingInfo &&\r\n (dimensionInfo.boundingInfo.targetNode._dynamicId !== target1?._dynamicId ||\r\n dimensionInfo.boundingInfo.targetNode2._dynamicId !== target2?._dynamicId)\r\n ) {\r\n dimensionInfo.boundingInfo.subscription.unsubscribe();\r\n dimensionInfo.boundingInfo = undefined;\r\n }\r\n\r\n if (target1 && target2) {\r\n let position1: Vector3 | undefined;\r\n let position2: Vector3 | undefined;\r\n let direction: Vector3 | undefined;\r\n let angleAdjustment = 0;\r\n\r\n // Only supporting connectors for now, but could support direct mesh targets in the future\r\n if (target1 instanceof Connector && target2 instanceof Connector) {\r\n position1 = getPositionFromConnnector(target1, this.rendererManager, node._parentScene);\r\n position2 = getPositionFromConnnector(target2, this.rendererManager, node._parentScene);\r\n direction = target1.direction.toVec3().normalize();\r\n angleAdjustment = target1.angle;\r\n\r\n if (target1._parent && target2._parent) {\r\n dimensionInfo.boundingInfo = {\r\n targetNode: target1,\r\n targetNode2: target2,\r\n subscription: merge(\r\n this.viewer.bbCache.watchForChanges(target1._parent),\r\n this.viewer.bbCache.watchForChanges(target2._parent)\r\n ).subscribe(() => {\r\n node.setPropertyIsModified('targetId', true);\r\n node.setPropertyIsModified('target2Id', true);\r\n }),\r\n };\r\n }\r\n }\r\n\r\n if (position1 && position2 && direction) {\r\n /** Calculate the dimension line */\r\n const offset = node.offset.toVec3();\r\n const startConnector = calculateTransform(position1, direction, angleAdjustment, offset);\r\n const endConnector = calculateTransform(position2, direction, angleAdjustment, offset);\r\n const dimensionLine = endConnector.position.subtract(startConnector.position);\r\n const startPostion = startConnector.position;\r\n // Depending on the dimension axis, just use the\r\n let endPosition: Vector3;\r\n if (node.axis === eDimensionAxis.free) {\r\n endPosition = endConnector.position;\r\n } else {\r\n if (node.axis === eDimensionAxis.x) {\r\n endPosition = startPostion.add(dimensionLine.multiply(new Vector3(1, 0, 0)));\r\n } else if (node.axis === eDimensionAxis.y) {\r\n endPosition = startPostion.add(dimensionLine.multiply(new Vector3(0, 1, 0)));\r\n } else {\r\n endPosition = startPostion.add(dimensionLine.multiply(new Vector3(0, 0, 1)));\r\n }\r\n }\r\n const rectifiedDimensionLine = endPosition.subtract(startPostion);\r\n const lineDirectionCross = Vector3.Cross(rectifiedDimensionLine, direction).normalize();\r\n\r\n lineOptions.points = [startPostion, endPosition];\r\n dimensionModel.distance = rectifiedDimensionLine.length();\r\n annotationMesh.position = startPostion.add(endPosition).scaleInPlace(0.5);\r\n annotationMesh.position.addInPlace(\r\n transformVectorWithQuat(node.labelOffset.toVec3(), startConnector.rotation)\r\n );\r\n dimensionModel.textContainer.rotationQuaternion = startConnector.textRotation;\r\n\r\n if (!node.billboard) {\r\n annotationMesh.rotationQuaternion = rotationToVector(lineDirectionCross);\r\n }\r\n\r\n /** Calculate the line caps */\r\n const capSize = direction.scale(node.fontSize / 3);\r\n\r\n const lineToStartConnector = direction.scale(\r\n Vector3.Dot(transformVectorWithQuat(offset, startConnector.rotation), direction) *\r\n (1 - node.dimensionGap)\r\n );\r\n const endOffset =\r\n node.axis === eDimensionAxis.free\r\n ? transformVectorWithQuat(offset, startConnector.rotation)\r\n : endPosition.subtract(position2);\r\n const lineToEndConnector = direction.scale(Vector3.Dot(endOffset, direction) * (1 - node.dimensionGap));\r\n\r\n dimensionModel.lineCap.lines = [];\r\n if (node.lineCapStyle !== eDimensionLineCapStyle.none) {\r\n dimensionModel.lineCap.lines = [\r\n [startPostion.subtract(lineToStartConnector), startPostion, startPostion.add(capSize)],\r\n [endPosition.subtract(lineToEndConnector), endPosition, endPosition.add(capSize)],\r\n ];\r\n }\r\n if (node.lineCapStyle === eDimensionLineCapStyle.arrow) {\r\n const capArrowOffset = rectifiedDimensionLine.normalizeToNew().scaleInPlace(capSize.length());\r\n dimensionModel.lineCap.lines.push(\r\n [\r\n startPostion.subtract(capSize).addInPlace(capArrowOffset),\r\n startPostion,\r\n startPostion.add(capSize).addInPlace(capArrowOffset),\r\n ],\r\n [\r\n endPosition.subtract(capSize).subtractInPlace(capArrowOffset),\r\n endPosition,\r\n endPosition.add(capSize).subtractInPlace(capArrowOffset),\r\n ]\r\n );\r\n }\r\n\r\n return true;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n private updateDimensionLineColor(\r\n node: ITrackedClass,\r\n lineOptions: ILineMeshOptions,\r\n lineCapOptions: ICapMeshOptions\r\n ) {\r\n const color = node.color.toColor4();\r\n lineOptions.colors = [];\r\n for (let i = 0; i < lineOptions.points.length; i++) {\r\n lineOptions.colors.push(color);\r\n }\r\n\r\n lineCapOptions.colors = [];\r\n for (let i = 0; i < 4; i++) {\r\n const lineColor: Color4[] = [];\r\n lineCapOptions.colors.push(lineColor);\r\n for (let j = 0; j < 3; j++) {\r\n lineColor.push(color);\r\n }\r\n }\r\n }\r\n}\r\n\r\nexport function getDimensionLinePoints(node: DimensionNode, rendererManager: RendererManager) {\r\n if (node.targetId && node.target2Id && node._manager) {\r\n const manager = node._manager;\r\n const target1 = manager.getById(node.targetId);\r\n const target2 = manager.getById(node.target2Id);\r\n let position1: Vector3 | undefined;\r\n let position2: Vector3 | undefined;\r\n let direction: Vector3 | undefined;\r\n if (target1 instanceof Connector && target2 instanceof Connector) {\r\n position1 = getPositionFromConnnector(target1, rendererManager, node._parentScene);\r\n position2 = getPositionFromConnnector(target2, rendererManager, node._parentScene);\r\n direction = target1.direction.toVec3();\r\n }\r\n\r\n if (position1 && position2 && direction) {\r\n const lineStart = calculateTransform(position1, direction, 0, node.offset.toVec3());\r\n const lineEnd = calculateTransform(position2, direction, 0, node.offset.toVec3());\r\n return [lineStart.position, lineEnd.position];\r\n }\r\n }\r\n\r\n return undefined;\r\n}\r\n","import { Material } from '@babylonjs/core/Materials/material';\r\nimport { MultiMaterial } from '@babylonjs/core/Materials/multiMaterial';\r\nimport { StandardMaterial } from '@babylonjs/core/Materials/standardMaterial';\r\nimport { Axis, Color3, Matrix, Quaternion, Vector3 } from '@babylonjs/core/Maths';\r\nimport { CreateCylinderVertexData } from '@babylonjs/core/Meshes/Builders/cylinderBuilder';\r\nimport { CreateSphereVertexData } from '@babylonjs/core/Meshes/Builders/sphereBuilder';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { LinesMesh } from '@babylonjs/core/Meshes/linesMesh';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { SubMesh } from '@babylonjs/core/Meshes/subMesh';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { BaseConnector, Connector, IKbVectorLike, Kb3dObject, KbVector, SketchNode, instanceOf } from '@models/classes';\r\nimport { eAxis, eConnectorPositionMode } from '@models/classes/enums';\r\nimport { SpaceNode } from '@models/classes/meshes/space-node';\r\nimport { KbColor } from '@models/classes/primitives/kb-color';\r\nimport { IBoundingBox } from '@models/classes/primitives/primitive-utilities';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass, isTrackedClass } from '@models/interfaces/tracked-class';\r\nimport { BoundingBoxRelativeHelper } from '@view/helpers/bounding-box-relative-helper';\r\nimport { getRotationBtwVectors, toRadians } from '@view/helpers/math-helper';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { Subscription } from 'rxjs';\r\nimport { Renderer } from '../renderer';\r\nimport { RendererManager } from '../renderer-manager';\r\n\r\nexport interface ConnectorParts {\r\n mesh: AbstractMesh;\r\n handleContainer?: TransformNode;\r\n handleBar?: LinesMesh;\r\n directionHandle1?: AbstractMesh;\r\n directionHandle2?: AbstractMesh;\r\n}\r\n\r\nexport const CONNECTOR_DIAM = 0.1;\r\n\r\nexport interface IConnectorLayer {\r\n template: Mesh | null;\r\n map: Map;\r\n scene: Scene;\r\n withArrows: boolean;\r\n visibility: number;\r\n pickable: boolean;\r\n parts: Map;\r\n}\r\n\r\nexport class ConnectorRenderer extends Renderer {\r\n protected layers = new Map<'real' | 'utility', IConnectorLayer>();\r\n protected subscriptionMap = new Map();\r\n\r\n protected readonly controlPointScaleRatio: number = 0.24;\r\n\r\n constructor(viewer: KbViewer, scene: Scene, protected rendererManager: RendererManager) {\r\n super(viewer, scene);\r\n\r\n this.layers.set('real', {\r\n template: null,\r\n map: this.babylonMap,\r\n scene,\r\n withArrows: !!viewer.options.editorMode,\r\n visibility: 1.0,\r\n pickable: !!viewer.options.editorMode,\r\n parts: new Map(),\r\n });\r\n this.layers.set('utility', {\r\n template: null,\r\n map: new Map(),\r\n scene: viewer.utilityLayer.utilityLayerScene,\r\n withArrows: !!viewer.options.editorMode,\r\n visibility: 0.3,\r\n pickable: false, // so utility layer doesn't steal the clicks\r\n parts: new Map(),\r\n });\r\n }\r\n modelToken = Token.Connector;\r\n protected connectorMaterial: Material;\r\n protected selectedMaterial: Material;\r\n\r\n public handlesNode(node: Kb3dObject): boolean {\r\n return instanceOf(node, Token.Connector);\r\n }\r\n\r\n update3dInternal(c: ITrackedClass, forceScreenScale = false) {\r\n /**\r\n * enabled, but not visible\r\n * enabled and visible\r\n * not enabled\r\n */\r\n const mesh = this.getMesh(c, this.layers.get('real')!);\r\n\r\n const realLayer = this.layers.get('real')!;\r\n const utilityLayer = this.layers.get('utility')!;\r\n const utilityMesh = this.getMesh(c, this.layers.get('utility')!);\r\n const realParts = realLayer.parts.get(c._dynamicId);\r\n const utilityParts = utilityLayer.parts.get(c._dynamicId);\r\n\r\n if (this.viewer.options.editorMode) {\r\n this.viewer.camera.maintainNodeScreenSize(mesh, dist => {\r\n const scaledSize = dist * this.controlPointScaleRatio;\r\n realParts?.mesh.scaling.set(scaledSize, scaledSize, scaledSize);\r\n realParts?.handleContainer?.scaling.set(1 / scaledSize, 1 / scaledSize, 1 / scaledSize);\r\n realParts?.directionHandle1?.scaling.set(scaledSize, scaledSize, scaledSize);\r\n realParts?.directionHandle2?.scaling.set(scaledSize, scaledSize, scaledSize);\r\n utilityParts?.mesh.scaling.set(scaledSize, scaledSize, scaledSize);\r\n utilityParts?.handleContainer?.scaling.set(1 / scaledSize, 1 / scaledSize, 1 / scaledSize);\r\n utilityParts?.directionHandle1?.scaling.set(scaledSize, scaledSize, scaledSize);\r\n utilityParts?.directionHandle2?.scaling.set(scaledSize, scaledSize, scaledSize);\r\n });\r\n }\r\n\r\n const changes = this.getChanges(c) as Map, any>;\r\n if (changes.has('_isPickable')) {\r\n mesh.isPickable = c._isPickable;\r\n }\r\n\r\n if (changes.has('positionZeroToOne') || changes.has('_sketchControlPoints')) {\r\n if (c.positionMode === eConnectorPositionMode.moveAlongSketch) {\r\n if (c instanceof Connector) {\r\n //this.updatePositionMoveAlongSketch(c);\r\n this.updatePosition(c);\r\n this.updateDirection(c);\r\n }\r\n }\r\n }\r\n\r\n if (changes.has('position') || changes.has('positionMode')) {\r\n const bb = c._parent instanceof SpaceNode ? this.viewer.bbCache.get(c._parent) : undefined;\r\n this.updatePosition(c, bb);\r\n /**\r\n * subscribe to bounding box changes of the parent to know that\r\n * we need to invalidate the position of the connector if applicable\r\n */\r\n if (changes.has('positionMode')) {\r\n if (c.positionMode == eConnectorPositionMode.boundingBox) {\r\n this.addSubscription(c);\r\n } else {\r\n this.removeSubscription(c);\r\n }\r\n }\r\n }\r\n\r\n if (changes.hasAny('direction', 'angle')) {\r\n this.updateDirection(c);\r\n this.updatePosition(c);\r\n }\r\n\r\n if (changes.hasAny('enabled', '_visible')) {\r\n if (c.enabled && c._visible) {\r\n mesh.setEnabled(true);\r\n utilityMesh.setEnabled(true);\r\n } else {\r\n mesh.setEnabled(false);\r\n utilityMesh.setEnabled(false);\r\n }\r\n }\r\n\r\n if (changes.has('_selected')) {\r\n if (c._selected) {\r\n mesh.material = this.getSelectedMaterial(this.layers.get('real')!.withArrows);\r\n } else {\r\n mesh.material = this.getConnectorMaterial(this.layers.get('real')!.withArrows);\r\n }\r\n utilityMesh.material = mesh.material;\r\n }\r\n\r\n return { mesh, utilityMesh };\r\n }\r\n\r\n protected addSubscription(c: ITrackedClass) {\r\n if (c._parent instanceof SpaceNode) {\r\n const mesh = c._parent;\r\n\r\n if (!this.subscriptionMap.has(c)) {\r\n this.subscriptionMap.set(\r\n c,\r\n this.viewer.bbCache.watchForChanges(mesh).subscribe(bbInfo => {\r\n c.setPropertyIsModified('position', true); // Trigger changes on mates and everything using that connector\r\n this.updatePosition(c, bbInfo);\r\n this.updateScale(c);\r\n })\r\n );\r\n }\r\n }\r\n }\r\n\r\n protected removeSubscription(c: T) {\r\n const s = this.subscriptionMap.get(c);\r\n if (s) {\r\n s.unsubscribe();\r\n this.subscriptionMap.delete(c);\r\n }\r\n }\r\n\r\n public updatePosition(c: T, bb?: IBoundingBox) {\r\n const mesh = this.getMesh(c, this.layers.get('real')!);\r\n const utilityMesh = this.getMesh(c, this.layers.get('utility')!);\r\n if (c.positionMode == eConnectorPositionMode.moveAlongSketch) {\r\n if (c instanceof Connector) {\r\n const newPosition = this.updatePositionMoveAlongSketch(c);\r\n if (newPosition) {\r\n mesh.position = utilityMesh.position = newPosition;\r\n }\r\n }\r\n } else if (c.positionMode == eConnectorPositionMode.absolute) {\r\n mesh.position = utilityMesh.position = c.position.toVec3();\r\n } else if (c.positionMode == eConnectorPositionMode.boundingBox) {\r\n if (!bb) {\r\n bb = c._parent instanceof SpaceNode ? this.viewer.bbCache.get(c._parent) : undefined;\r\n }\r\n if (bb) {\r\n mesh.position = BoundingBoxRelativeHelper.convertRelativeToAbsolute(c, bb);\r\n } else {\r\n mesh.position = new Vector3(c.position.x, c.position.y, c.position.z);\r\n }\r\n utilityMesh.position = mesh.position;\r\n }\r\n if (isTrackedClass(c)) {\r\n this.rendererManager.referenceProcessor.triggerChangesOnReferencers(c);\r\n }\r\n }\r\n\r\n public updateDirection(c: T) {\r\n const mesh = this.getMesh(c, this.layers.get('real')!);\r\n const utilityMesh = this.getMesh(c, this.layers.get('utility')!);\r\n if (c.positionMode === eConnectorPositionMode.moveAlongSketch) {\r\n if (c instanceof Connector) {\r\n let newDirection = this.updateDirectionMoveAlongSketch(c);\r\n if (newDirection) {\r\n newDirection = newDirection.normalize();\r\n\r\n if (c._parent instanceof SketchNode) {\r\n const normal = c._parent.normal;\r\n const normalVector = new Vector3(1, 0, 0);\r\n if (normal === eAxis.y) {\r\n normalVector.set(0, 1, 0);\r\n } else if (normal === eAxis.z) {\r\n normalVector.set(0, 0, 1);\r\n }\r\n if (mesh.parent instanceof TransformNode) {\r\n const connectorDirection = newDirection;\r\n const angleRot = Quaternion.RotationAxis(connectorDirection, toRadians(c.angle));\r\n const angleRotMatrix = new Matrix();\r\n angleRot.toRotationMatrix(angleRotMatrix);\r\n const parentDirection = Vector3.TransformCoordinates(normalVector, angleRotMatrix);\r\n const crossProduct = Vector3.Cross(connectorDirection, parentDirection);\r\n // prettier-ignore\r\n const matrix = Matrix.FromArray([connectorDirection.x, connectorDirection.y, connectorDirection.z, 0,\r\n parentDirection.x, parentDirection.y, parentDirection.z, 0,\r\n crossProduct.x, crossProduct.y, crossProduct.z, 0,\r\n 0, 0, 0, 1,\r\n ]);\r\n const newQuaternion = Quaternion.FromRotationMatrix(matrix);\r\n mesh.rotationQuaternion = utilityMesh.rotationQuaternion = newQuaternion;\r\n }\r\n }\r\n }\r\n }\r\n } else {\r\n /**\r\n * treat the positive x axis as the direction vector,\r\n * and the angle around that axis defined by the y axis.\r\n *\r\n * See Dragger.connect for a more in depth explanation\r\n */\r\n const deltaBtwXaxisAndDirection = getRotationBtwVectors(Axis.X, c.direction.toVec3().normalize());\r\n const angleRot = Quaternion.RotationAxis(Axis.X, toRadians(c.angle));\r\n mesh.rotationQuaternion = utilityMesh.rotationQuaternion = deltaBtwXaxisAndDirection.multiply(angleRot);\r\n }\r\n this.updateScale(c);\r\n }\r\n\r\n /**\r\n * connector meshes are children of their parent mesh, meaning they will scale\r\n * with any of their ancestor meshes or groups.\r\n * To counteract this we inverse their world scale.\r\n * WARNING! this technique does not work well when an ancestor has a non-uniform scale\r\n * @param c\r\n */\r\n public updateScale(c: T) {\r\n if (this.layers.get('real')!.map.get(c._dynamicId)) {\r\n //this could get called at any time, make sure the mesh already exists\r\n const mesh = this.getMesh(c, this.layers.get('real')!);\r\n const utilityMesh = this.getMesh(c, this.layers.get('utility')!);\r\n\r\n mesh.scaling = Vector3.One();\r\n mesh.computeWorldMatrix(true);\r\n\r\n if (mesh.parent instanceof TransformNode) {\r\n const worldScale = mesh.parent.absoluteScaling;\r\n\r\n const numerator = (this.viewer.sceneNode.connectorSize || CONNECTOR_DIAM) / CONNECTOR_DIAM;\r\n const transformMatrix = Matrix.Scaling(\r\n numerator / worldScale.x,\r\n numerator / worldScale.y,\r\n numerator / worldScale.z\r\n );\r\n mesh.setPreTransformMatrix(transformMatrix);\r\n utilityMesh.setPreTransformMatrix(transformMatrix);\r\n\r\n if (isTrackedClass(c)) {\r\n this.rendererManager.referenceProcessor.triggerChangesOnReferencers(c);\r\n }\r\n }\r\n }\r\n }\r\n\r\n protected updatePositionMoveAlongSketch(connector: Connector) {\r\n const node = connector._parent;\r\n const result = new Vector3(0, 0, 0);\r\n if (node instanceof SketchNode) {\r\n const sketchLength = node.getSketchLength();\r\n const posZeroToOne = connector.positionZeroToOne;\r\n let trackedLength = 0;\r\n let targetNotFound = true;\r\n node.paths.forEach(path => {\r\n let lastPosition: IKbVectorLike | null = null;\r\n let lastLength = 0;\r\n path._points.forEach(point => {\r\n if (!lastPosition) {\r\n lastPosition = point;\r\n } else {\r\n const currentPos = point;\r\n const distance = Math.sqrt(\r\n Math.pow(lastPosition.x - currentPos.x, 2) +\r\n Math.pow(lastPosition.y - currentPos.y, 2) +\r\n Math.pow(lastPosition.z - currentPos.z, 2)\r\n );\r\n lastLength = trackedLength;\r\n trackedLength += distance;\r\n const distanceBetweenPoints = trackedLength - lastLength;\r\n const distanceFromP1toTarget = posZeroToOne * sketchLength - lastLength;\r\n const percentageBetween2Points = distanceFromP1toTarget / distanceBetweenPoints;\r\n const vectorBetween2Points = new KbVector(\r\n currentPos.x - lastPosition.x,\r\n currentPos.y - lastPosition.y,\r\n currentPos.z - lastPosition.z\r\n );\r\n if (trackedLength > posZeroToOne * sketchLength && targetNotFound) {\r\n const calculatedPosition = vectorBetween2Points\r\n .scale(percentageBetween2Points)\r\n .add(lastPosition);\r\n result.set(calculatedPosition.x, calculatedPosition.y, calculatedPosition.z);\r\n targetNotFound = false;\r\n }\r\n lastPosition = currentPos;\r\n }\r\n });\r\n if (posZeroToOne === 1) {\r\n const path = node.paths[node.paths.length - 1];\r\n const controlPoint = path._points[path._points.length - 1];\r\n result.set(controlPoint.x, controlPoint.y, controlPoint.z);\r\n }\r\n });\r\n }\r\n return result;\r\n }\r\n\r\n protected updateDirectionMoveAlongSketch(connector: Connector) {\r\n const node = connector._parent;\r\n const result = new Vector3(0, 0, 0);\r\n if (node instanceof SketchNode) {\r\n const sketchLength = node.getSketchLength();\r\n const posZeroToOne = connector.positionZeroToOne;\r\n let trackedLength = 0;\r\n\r\n let targetNotFound = true;\r\n let lastPathVector: KbVector | null = null;\r\n let nextPathVector: KbVector | null = null;\r\n node.paths.forEach(path => {\r\n let lastPosition: IKbVectorLike | null = null;\r\n let lastLength = 0;\r\n path._points.forEach(point => {\r\n if (!lastPosition) {\r\n lastPosition = point;\r\n } else {\r\n const currentPos = point;\r\n if (lastPosition) {\r\n const index = path._points.indexOf(lastPosition) + 1;\r\n const index2 = index + 1;\r\n if (index !== path._points.length) {\r\n const nextPosition = path._points[index];\r\n if (index2 !== path._points.length) {\r\n const nextNextPosition = path._points[index2];\r\n nextPathVector = new KbVector(\r\n nextNextPosition.x - nextPosition.x,\r\n nextNextPosition.y - nextPosition.y,\r\n nextNextPosition.z - nextPosition.z\r\n ).normalize();\r\n }\r\n }\r\n }\r\n const distance = Math.sqrt(\r\n Math.pow(lastPosition.x - currentPos.x, 2) +\r\n Math.pow(lastPosition.y - currentPos.y, 2) +\r\n Math.pow(lastPosition.z - currentPos.z, 2)\r\n );\r\n lastLength = trackedLength;\r\n trackedLength += distance;\r\n const distanceBetweenPoints = trackedLength - lastLength;\r\n const distanceFromP1toTarget = posZeroToOne * sketchLength - lastLength;\r\n const percentageBetween2Points = distanceFromP1toTarget / distanceBetweenPoints;\r\n const vectorBetween2Points = new KbVector(\r\n currentPos.x - lastPosition.x,\r\n currentPos.y - lastPosition.y,\r\n currentPos.z - lastPosition.z\r\n );\r\n if (trackedLength > posZeroToOne * sketchLength && targetNotFound) {\r\n let calculatedDirection = null;\r\n if (!connector.interpolateDirection) {\r\n calculatedDirection = vectorBetween2Points.normalize();\r\n } else {\r\n if (!lastPathVector) {\r\n if (nextPathVector) {\r\n calculatedDirection = vectorBetween2Points\r\n .normalize()\r\n .scale(1 - percentageBetween2Points)\r\n .add(\r\n nextPathVector\r\n .normalize()\r\n .add(vectorBetween2Points.normalize())\r\n .normalize()\r\n .scale(percentageBetween2Points)\r\n )\r\n .normalize();\r\n } else {\r\n calculatedDirection = vectorBetween2Points.normalize();\r\n }\r\n } else if (!nextPathVector) {\r\n calculatedDirection = lastPathVector\r\n .scale(1 - percentageBetween2Points)\r\n .add(vectorBetween2Points.scale(1 + percentageBetween2Points))\r\n .scale(1 / 2)\r\n .normalize();\r\n } else {\r\n if (percentageBetween2Points <= 0.5) {\r\n calculatedDirection = lastPathVector\r\n .normalize()\r\n .add(vectorBetween2Points.normalize())\r\n .normalize()\r\n .scale(1 - percentageBetween2Points * 2)\r\n .add(vectorBetween2Points.scale(percentageBetween2Points * 2))\r\n .normalize();\r\n } else {\r\n calculatedDirection = vectorBetween2Points\r\n .normalize()\r\n .scale(1 - (percentageBetween2Points - 0.5) * 2)\r\n .add(\r\n nextPathVector\r\n .normalize()\r\n .add(vectorBetween2Points.normalize())\r\n .normalize()\r\n .scale((percentageBetween2Points - 0.5) * 2)\r\n )\r\n .normalize();\r\n }\r\n }\r\n }\r\n targetNotFound = false;\r\n result.set(calculatedDirection.x, calculatedDirection.y, calculatedDirection.z);\r\n }\r\n lastPathVector = vectorBetween2Points;\r\n lastPosition = currentPos;\r\n }\r\n });\r\n });\r\n if (posZeroToOne === 1) {\r\n if (posZeroToOne === 1) {\r\n const path = node.paths[node.paths.length - 1];\r\n const currentPos = path._points[path._points.length - 1];\r\n const lastPosition = path._points[path._points.length - 2];\r\n const vectorBetween2Points = new KbVector(\r\n currentPos.x - lastPosition.x,\r\n currentPos.y - lastPosition.y,\r\n currentPos.z - lastPosition.z\r\n ).normalize();\r\n result.set(vectorBetween2Points.x, vectorBetween2Points.y, vectorBetween2Points.z);\r\n }\r\n }\r\n }\r\n return result;\r\n }\r\n\r\n protected getMesh(c: T, layer: IConnectorLayer) {\r\n let mesh = layer.map.get(c._dynamicId);\r\n if (!mesh) {\r\n if (!layer.template) {\r\n layer.template = this.createMeshTemplate(layer);\r\n }\r\n\r\n mesh = layer.template.clone(c._dynamicId); // layer.template.createInstance(c._dynamicId);\r\n mesh.id = c._dynamicId;\r\n\r\n const parentNode = c._parent;\r\n\r\n if (parentNode && parentNode instanceof SpaceNode) {\r\n const parent3d = this.viewer.getRendererNode(parentNode);\r\n if (parent3d) {\r\n mesh.setParent(parent3d);\r\n } else {\r\n console.warn(`mesh not found for ${parentNode.name}`);\r\n }\r\n }\r\n\r\n mesh.isPickable = layer.pickable;\r\n mesh.position = Vector3.Zero();\r\n mesh.rotationQuaternion = Quaternion.Identity();\r\n layer.map.set(c._dynamicId, mesh);\r\n layer.parts.set(c._dynamicId, {\r\n mesh,\r\n });\r\n }\r\n return mesh;\r\n }\r\n\r\n protected createMeshTemplate(layer: IConnectorLayer) {\r\n const ball = CreateSphereVertexData({\r\n segments: 8,\r\n diameter: CONNECTOR_DIAM,\r\n });\r\n\r\n if (layer.withArrows) {\r\n const arrowHead = CreateCylinderVertexData({\r\n diameterTop: 0,\r\n height: CONNECTOR_DIAM / 2,\r\n diameterBottom: CONNECTOR_DIAM / 4,\r\n tessellation: 6,\r\n });\r\n arrowHead.transform(\r\n Matrix.Compose(\r\n Vector3.One(),\r\n Quaternion.FromEulerAngles(Math.PI / 2, 0, 0),\r\n new Vector3(0, 0, CONNECTOR_DIAM * 1.75)\r\n )\r\n );\r\n\r\n const arrowShaft = CreateCylinderVertexData({\r\n diameterTop: CONNECTOR_DIAM / 10,\r\n height: CONNECTOR_DIAM,\r\n diameterBottom: CONNECTOR_DIAM / 10,\r\n tessellation: 6,\r\n });\r\n\r\n arrowShaft.transform(\r\n Matrix.Compose(\r\n Vector3.One(),\r\n Quaternion.FromEulerAngles(Math.PI / 2, 0, 0),\r\n new Vector3(0, 0, CONNECTOR_DIAM)\r\n )\r\n );\r\n\r\n ball.merge(arrowHead).merge(arrowShaft);\r\n }\r\n\r\n const mesh = new Mesh('', layer.scene);\r\n ball.applyToMesh(mesh, false); //updateable false\r\n\r\n if (layer.withArrows) {\r\n //let ballHalf1 = new SubMesh(0, 0, 231, 0, 600, mesh);\r\n new SubMesh(1, 0, 231, 0, 600, mesh); // ball half 2\r\n }\r\n\r\n mesh.rotation.y = Math.PI / 2;\r\n mesh.bakeCurrentTransformIntoVertices();\r\n mesh.position = Vector3.Zero();\r\n mesh.material = this.getConnectorMaterial(layer.withArrows);\r\n mesh.visibility = layer.visibility;\r\n mesh.setEnabled(false);\r\n return mesh;\r\n }\r\n\r\n public delete3d(c: ITrackedClass) {\r\n const utilityLayer = this.layers.get('utility')!;\r\n const utilityMesh = utilityLayer.map.get(c._dynamicId);\r\n if (utilityMesh) {\r\n utilityMesh.dispose();\r\n utilityLayer.map.delete(c._dynamicId);\r\n }\r\n\r\n this.removeSubscription(c);\r\n super.delete3d(c);\r\n }\r\n\r\n protected getConnectorMaterial(twoTone: boolean) {\r\n if (!this.connectorMaterial) {\r\n if (twoTone) {\r\n const m1 = new StandardMaterial('connectorMaterial0', this.scene);\r\n m1.emissiveColor = KbColor.KbmaxOrange().toColor3();\r\n m1.disableLighting = true;\r\n\r\n const m2 = new StandardMaterial('connectorMaterial1', this.scene);\r\n m2.emissiveColor = new Color3(0, 101 / 255, 1); //new Color3(226/255, 101/255, 29/255);// KbColor.KbmaxBlue().toColor3();\r\n m2.disableLighting = true;\r\n\r\n const m = new MultiMaterial('connectorMaterial', this.scene);\r\n m.subMaterials.push(m1, m2);\r\n this.connectorMaterial = m;\r\n } else {\r\n const m = new StandardMaterial('connectorMaterial', this.scene);\r\n m.emissiveColor = this.viewer.highlightColor.toColor3();\r\n m.disableLighting = true;\r\n this.connectorMaterial = m;\r\n }\r\n }\r\n return this.connectorMaterial;\r\n }\r\n\r\n protected getSelectedMaterial(withArrows: boolean) {\r\n if (!this.selectedMaterial) {\r\n this.selectedMaterial = this.getConnectorMaterial(withArrows);\r\n if (withArrows) {\r\n const m1 = new StandardMaterial('connectorSelectedMaterial0', this.scene);\r\n m1.emissiveColor = new Color3(254 / 255, 233 / 255, 7 / 255);\r\n m1.disableLighting = true;\r\n\r\n const m2 = new StandardMaterial('connectorSelectedMaterial1', this.scene);\r\n m2.emissiveColor = new Color3(2 / 255, 251 / 255, 248 / 255);\r\n m2.disableLighting = true;\r\n\r\n const m = new MultiMaterial('connectorSelectedMaterial', this.scene);\r\n m.subMaterials.push(m1, m2);\r\n this.selectedMaterial = m;\r\n } else {\r\n const m = new StandardMaterial('connectorSelectedMaterial', this.scene);\r\n m.emissiveColor = this.viewer.options.highlightColor!.toColor3();\r\n m.disableLighting = true;\r\n this.selectedMaterial = m;\r\n }\r\n }\r\n return this.selectedMaterial;\r\n }\r\n\r\n public dispose() {\r\n super.dispose();\r\n\r\n for (const [, layer] of this.layers) {\r\n if (layer.template) {\r\n layer.template.dispose();\r\n }\r\n layer.map.clear();\r\n }\r\n if (this.connectorMaterial) {\r\n this.connectorMaterial.dispose();\r\n }\r\n }\r\n}\r\n","import { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { eMeshChangeTypes, Feature } from '@models/classes/features/feature';\r\nimport { ITrackedClass } from '@models/interfaces/tracked-class';\r\nimport { SyncService } from '@models/services/sync-service';\r\nimport { SubmeshVertexData } from '@view/helpers/geometry-helper';\r\nimport { Renderer } from '../renderer';\r\n\r\nexport abstract class FeatureRenderer extends Renderer {\r\n vertexDataCache = new Map>();\r\n\r\n public updateMeshGeometryCache(\r\n feature: T,\r\n vertexData: SubmeshVertexData,\r\n vertexDataChanges: Partial\r\n ): Partial {\r\n const self = this;\r\n const newVertexData: Partial = {};\r\n\r\n // Grab the properties in the mesh vertex that are needed for the feature\r\n // Only swap out the properties in the old vertex data with the stored copy if they have NOT been affected by another vertex\r\n if (feature.affectedBy) {\r\n const cachedData = this.vertexDataCache.get(feature._dynamicId);\r\n feature.affectedBy.forEach(affectedBy => {\r\n if (affectedBy === eMeshChangeTypes.VertexIndices) {\r\n if (vertexDataChanges.indices === null) {\r\n newVertexData.indices = null;\r\n clearVertexDataCache('indices');\r\n } else if (vertexDataChanges.indices) {\r\n newVertexData.indices = vertexDataChanges.indices.slice();\r\n updateVertexDataCache('indices', vertexDataChanges.indices);\r\n } else if (cachedData) {\r\n newVertexData.indices = cachedData.indices?.slice() as any;\r\n } else {\r\n newVertexData.indices = vertexData.indices!.slice();\r\n updateVertexDataCache('indices', vertexData.indices);\r\n }\r\n\r\n if (vertexDataChanges.paths === null) {\r\n newVertexData.paths = null;\r\n clearVertexDataCache('paths');\r\n } else if (vertexDataChanges.paths) {\r\n newVertexData.paths = vertexDataChanges.paths.slice();\r\n updateVertexDataCache('paths', vertexDataChanges.paths);\r\n } else if (cachedData) {\r\n newVertexData.paths = cachedData.paths?.slice() as any;\r\n } else {\r\n if (vertexData.paths) {\r\n newVertexData.paths = vertexData.paths.slice();\r\n updateVertexDataCache('paths', vertexData.paths);\r\n } else {\r\n newVertexData.paths = null;\r\n }\r\n }\r\n\r\n if (vertexDataChanges.holes === null) {\r\n newVertexData.holes = null;\r\n clearVertexDataCache('holes');\r\n } else if (vertexDataChanges.holes) {\r\n newVertexData.holes = vertexDataChanges.holes.slice();\r\n updateVertexDataCache('holes', vertexDataChanges.holes);\r\n } else if (cachedData) {\r\n newVertexData.holes = cachedData.holes?.slice() as any;\r\n } else {\r\n if (vertexData.holes) {\r\n newVertexData.holes = vertexData.holes.slice();\r\n updateVertexDataCache('holes', vertexData.holes);\r\n } else {\r\n newVertexData.holes = null;\r\n }\r\n }\r\n }\r\n if (affectedBy === eMeshChangeTypes.VertexPositions) {\r\n if (vertexDataChanges.positions === null) {\r\n newVertexData.positions = null;\r\n clearVertexDataCache('positions');\r\n } else if (vertexDataChanges.positions) {\r\n newVertexData.positions = vertexDataChanges.positions.slice();\r\n updateVertexDataCache('positions', vertexDataChanges.positions);\r\n } else if (cachedData) {\r\n newVertexData.positions = cachedData.positions?.slice() as any; // compiler complains about the slice generic type for some reason\r\n } else {\r\n if (!vertexData.positions) {\r\n throw Error('Invalid geometry: positions is null');\r\n }\r\n newVertexData.positions = vertexData.positions.slice();\r\n updateVertexDataCache('positions', vertexData.positions);\r\n }\r\n }\r\n if (affectedBy === eMeshChangeTypes.VertexNormals) {\r\n if (vertexDataChanges.normals === null) {\r\n newVertexData.normals = null;\r\n clearVertexDataCache('normals');\r\n } else if (vertexDataChanges.normals) {\r\n newVertexData.normals = vertexDataChanges.normals.slice();\r\n updateVertexDataCache('normals', vertexDataChanges.normals);\r\n } else if (cachedData) {\r\n newVertexData.normals = cachedData.normals?.slice() as any;\r\n } else {\r\n if (vertexData.normals) {\r\n newVertexData.normals = vertexData.normals.slice();\r\n updateVertexDataCache('normals', vertexData.normals);\r\n } else {\r\n newVertexData.normals = null;\r\n }\r\n }\r\n }\r\n if (affectedBy === eMeshChangeTypes.VertexUvs) {\r\n if (vertexDataChanges.uvs === null) {\r\n newVertexData.uvs = null;\r\n clearVertexDataCache('uvs');\r\n } else if (vertexDataChanges.uvs) {\r\n newVertexData.uvs = vertexDataChanges.uvs.slice();\r\n updateVertexDataCache('uvs', vertexDataChanges.uvs);\r\n } else if (cachedData) {\r\n newVertexData.uvs = cachedData.uvs?.slice() as any;\r\n } else {\r\n if (vertexData.uvs) {\r\n newVertexData.uvs = vertexData.uvs.slice();\r\n updateVertexDataCache('uvs', vertexData.uvs);\r\n } else {\r\n newVertexData.uvs = null;\r\n }\r\n }\r\n }\r\n if (affectedBy === eMeshChangeTypes.Submeshes) {\r\n if (vertexDataChanges.submeshData === null) {\r\n newVertexData.submeshData = null;\r\n clearVertexDataCache('submeshData');\r\n } else if (vertexDataChanges.submeshData) {\r\n newVertexData.submeshData = JSON.parse(JSON.stringify(vertexDataChanges.submeshData));\r\n updateVertexDataCache('submeshData', JSON.parse(JSON.stringify(vertexDataChanges.submeshData)));\r\n } else if (cachedData) {\r\n const cachedSubmeshes = cachedData.submeshData;\r\n newVertexData.submeshData = cachedSubmeshes\r\n ? JSON.parse(JSON.stringify(cachedSubmeshes))\r\n : undefined;\r\n } else {\r\n if (vertexData.submeshData) {\r\n newVertexData.submeshData = JSON.parse(JSON.stringify(vertexData.submeshData));\r\n updateVertexDataCache('submeshData', JSON.parse(JSON.stringify(vertexData.submeshData)));\r\n } else {\r\n newVertexData.submeshData = null;\r\n }\r\n }\r\n }\r\n });\r\n }\r\n\r\n function clearVertexDataCache(prop: keyof SubmeshVertexData) {\r\n const vertexDataCache = self.vertexDataCache.get(feature._dynamicId);\r\n if (vertexDataCache) {\r\n delete vertexDataCache[prop];\r\n }\r\n }\r\n\r\n function updateVertexDataCache(prop: keyof SubmeshVertexData, value: any) {\r\n if (!value) {\r\n throw Error('invalid geometry does not have ' + prop);\r\n }\r\n\r\n let vertexDataCache = self.vertexDataCache.get(feature._dynamicId);\r\n if (!vertexDataCache) {\r\n vertexDataCache = {};\r\n self.vertexDataCache.set(feature._dynamicId, vertexDataCache);\r\n }\r\n\r\n if (typeof value.slice !== 'function') {\r\n throw Error('fixme: how do I clone this property?');\r\n }\r\n vertexDataCache[prop] = value.slice(); // Need to make sure everything is cloned when inserted into the cache\r\n }\r\n\r\n return newVertexData;\r\n }\r\n\r\n public updateMeshGeometry(\r\n feature: T,\r\n vertexData: SubmeshVertexData,\r\n vertexDataChanges: Partial\r\n ): Partial {\r\n return this.updateMeshGeometryCache(feature, vertexData, vertexDataChanges);\r\n }\r\n\r\n public delete3d(node: ITrackedClass, preserveCache?: boolean) {\r\n if (!preserveCache) {\r\n this.vertexDataCache.delete(node._dynamicId);\r\n }\r\n const t3d = this.get3d(node);\r\n if (t3d) {\r\n super.delete3d(node);\r\n } else {\r\n SyncService.nodeDeletionProcessed(node);\r\n }\r\n }\r\n\r\n public clearCache(features?: Feature[]) {\r\n if (features) {\r\n features.forEach(feature => {\r\n this.vertexDataCache.delete(feature._dynamicId);\r\n });\r\n } else {\r\n this.vertexDataCache.clear();\r\n }\r\n }\r\n\r\n public dispose() {\r\n super.dispose();\r\n this.clearCache();\r\n }\r\n}\r\n","import { Quaternion, Vector3 } from '@babylonjs/core/Maths/math.vector';\r\nimport { eFeatureCenter } from '@models/classes/enums';\r\nimport { MoveVerticesFeature } from '@models/classes/features/move-vertices-feature';\r\nimport { Token } from '@models/classes/token';\r\nimport { SubmeshVertexData } from '@view/helpers/geometry-helper';\r\nimport { FeatureRenderer } from './feature';\r\n\r\nexport class MoveVerticesRenderer extends FeatureRenderer {\r\n modelToken = Token.MoveVerticesFeature;\r\n updateMeshGeometry(\r\n feature: MoveVerticesFeature,\r\n vertexData: SubmeshVertexData,\r\n vertexDataChanges: Partial\r\n ) {\r\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\r\n const parentMesh = feature._parent;\r\n\r\n if (parentMesh && newVertexData.positions && newVertexData.positions.length > 0) {\r\n const newMeshVertexPositions = newVertexData.positions;\r\n if (feature.vertexIndices && feature.vertexIndices.length > 0) {\r\n let pivot: Vector3 | undefined;\r\n let rotationQuaternion: Quaternion | undefined;\r\n let scaleVector: Vector3 | undefined;\r\n\r\n if (feature.rotationAngle !== 0 && !feature.rotationAxis.equal({ x: 0, y: 0, z: 0 })) {\r\n rotationQuaternion = Quaternion.RotationAxis(\r\n feature.rotationAxis.toVec3(),\r\n (feature.rotationAngle / 180) * Math.PI\r\n );\r\n }\r\n if (!feature.scale.equal({ x: 1, y: 1, z: 1 })) {\r\n scaleVector = feature.scale.toVec3();\r\n }\r\n\r\n // Calculate vertex center if needed\r\n if (rotationQuaternion || scaleVector) {\r\n if (feature.center === eFeatureCenter.meshPivotPoint) {\r\n pivot = parentMesh.pivot.toVec3();\r\n } else {\r\n // Calculate vertices barycenter\r\n pivot = Vector3.Zero();\r\n const currentMeshVertexPositions = newVertexData.positions;\r\n if (currentMeshVertexPositions) {\r\n feature.vertexIndices.forEach(idx => {\r\n pivot!.x += currentMeshVertexPositions[idx * 3];\r\n pivot!.y += currentMeshVertexPositions[idx * 3 + 1];\r\n pivot!.z += currentMeshVertexPositions[idx * 3 + 2];\r\n });\r\n\r\n pivot.scaleInPlace(1 / feature.vertexIndices.length);\r\n }\r\n }\r\n }\r\n\r\n const vertexPosition = new Vector3();\r\n feature.vertexIndices.forEach(index => {\r\n newMeshVertexPositions[index * 3] += feature.translation.x;\r\n newMeshVertexPositions[index * 3 + 1] += feature.translation.y;\r\n newMeshVertexPositions[index * 3 + 2] += feature.translation.z;\r\n\r\n if (pivot && (rotationQuaternion || scaleVector)) {\r\n vertexPosition.copyFromFloats(\r\n newMeshVertexPositions[index * 3] - pivot.x,\r\n newMeshVertexPositions[index * 3 + 1] - pivot.y,\r\n newMeshVertexPositions[index * 3 + 2] - pivot.z\r\n );\r\n\r\n if (rotationQuaternion) {\r\n vertexPosition.rotateByQuaternionToRef(rotationQuaternion, vertexPosition);\r\n }\r\n\r\n if (scaleVector) {\r\n vertexPosition.multiplyInPlace(scaleVector);\r\n }\r\n\r\n vertexPosition.addInPlace(pivot);\r\n\r\n newMeshVertexPositions[index * 3] = vertexPosition.x;\r\n newMeshVertexPositions[index * 3 + 1] = vertexPosition.y;\r\n newMeshVertexPositions[index * 3 + 2] = vertexPosition.z;\r\n }\r\n });\r\n }\r\n }\r\n\r\n return newVertexData;\r\n }\r\n\r\n update3dInternal() {}\r\n}\r\n","import { Vector3 } from '@babylonjs/core/Maths/math.vector';\r\nimport { SubmeshData } from '@models/classes';\r\nimport { NormalSmoothingFeature } from '@models/classes/features/normal-smoothing-feature';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass } from '@models/interfaces/tracked-class';\r\nimport { addArrays } from '@view/helpers';\r\nimport { SubmeshVertexData } from '@view/helpers/geometry-helper';\r\nimport { FeatureRenderer } from './feature';\r\n\r\nexport class NormalSmoothingRenderer extends FeatureRenderer {\r\n modelToken = Token.NormalSmoothingFeature;\r\n vertexNeighbors = new Map();\r\n planeNormals = new Map();\r\n vectorA = new Vector3();\r\n vectorB = new Vector3();\r\n vectorC = new Vector3();\r\n normalVector = new Vector3();\r\n totalSelectedIndices = 0;\r\n updateMeshGeometry(\r\n feature: ITrackedClass,\r\n vertexData: SubmeshVertexData,\r\n vertexDataChanges: Partial\r\n ) {\r\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\r\n let vertexGroups = [];\r\n\r\n //if no faces are selected, select all faces to smooth entire mesh.\r\n if (feature.vertexGroups.length === 0 && newVertexData.indices) {\r\n for (let i = 0; i < newVertexData.indices.length; i += 6) {\r\n vertexGroups.push({ startIndex: i, endIndex: i + 6 });\r\n }\r\n } else {\r\n vertexGroups = feature.vertexGroups.sort((groupA, groupB) => groupA.startIndex - groupB.startIndex);\r\n }\r\n\r\n if (newVertexData.indices) {\r\n const planeNormal = new Vector3();\r\n let normals: number[];\r\n let positions: number[];\r\n let uvs: number[];\r\n let indices: number[];\r\n const newPositions: number[] = [];\r\n const newNormals: number[] = [];\r\n const newUvs: number[] = [];\r\n const submeshDataArray: SubmeshData[] = [];\r\n if (newVertexData.normals) {\r\n normals = newVertexData.normals as number[];\r\n } else if (vertexData.normals) {\r\n normals = Array.from(vertexData.normals) as number[];\r\n } else {\r\n normals = [];\r\n }\r\n\r\n if (newVertexData.positions) {\r\n positions = newVertexData.positions as number[];\r\n } else if (vertexData.positions) {\r\n positions = Array.from(vertexData.positions) as number[];\r\n } else {\r\n positions = [];\r\n }\r\n\r\n if (newVertexData.uvs) {\r\n uvs = newVertexData.uvs as number[];\r\n } else if (vertexData.uvs) {\r\n uvs = Array.from(vertexData.uvs) as number[];\r\n } else {\r\n uvs = [];\r\n }\r\n\r\n if (newVertexData.indices) {\r\n indices = newVertexData.indices as number[];\r\n } else if (vertexData.indices) {\r\n indices = Array.from(vertexData.indices) as number[];\r\n } else {\r\n indices = [];\r\n }\r\n\r\n this.vertexNeighbors.clear();\r\n this.planeNormals.clear();\r\n this.totalSelectedIndices = 0;\r\n vertexGroups.forEach(vertexGroup => {\r\n this.totalSelectedIndices += vertexGroup.endIndex - vertexGroup.startIndex;\r\n });\r\n\r\n if (feature.smooth) {\r\n //looping through the vertex groups and finding all the neighboring faces\r\n vertexGroups.forEach(vertexGroup => {\r\n const planeCoordinates = [];\r\n const planePositions = [];\r\n for (let i = vertexGroup.startIndex; i < vertexGroup.endIndex; i++) {\r\n if (positions && indices) {\r\n let key = '';\r\n for (let j = 0; j < 3; j++) {\r\n const vertexIndex = indices[i];\r\n const coordinate = positions[vertexIndex * 3 + j];\r\n planePositions.push(coordinate);\r\n if (j == 2) {\r\n planeCoordinates.push([...planePositions]);\r\n planePositions.clear();\r\n }\r\n key += String(coordinate);\r\n }\r\n\r\n if (planeCoordinates.length == 3) {\r\n this.calculateNormalToRef(planeCoordinates, planeNormal);\r\n this.planeNormals.set(Math.floor(i / 3), planeNormal.asArray());\r\n planeCoordinates.clear();\r\n }\r\n\r\n let neighbor = this.vertexNeighbors.get(key);\r\n if (neighbor) {\r\n neighbor.push(i);\r\n this.vertexNeighbors.set(key, neighbor);\r\n } else {\r\n neighbor = [];\r\n neighbor.push(i);\r\n this.vertexNeighbors.set(key, neighbor);\r\n }\r\n }\r\n }\r\n });\r\n\r\n this.vertexNeighbors.forEach(position => {\r\n const newNormal = this.calculateSmoothedUnitVector(position);\r\n\r\n position.forEach(index => {\r\n if (normals && indices) {\r\n for (let i = 0; i < 3; i++) {\r\n normals[indices[index] * 3 + i] = newNormal[i];\r\n }\r\n }\r\n });\r\n });\r\n } else {\r\n //looping through the selected faces to get the face normals\r\n vertexGroups.forEach(vertexGroup => {\r\n const planeCoordinates = [];\r\n const planePositions = [];\r\n for (let i = vertexGroup.startIndex; i < vertexGroup.endIndex; i++) {\r\n if (positions && indices) {\r\n for (let j = 0; j < 3; j++) {\r\n const vertexIndex = indices[i];\r\n const coordinate = positions[vertexIndex * 3 + j];\r\n planePositions.push(coordinate);\r\n if (j == 2) {\r\n planeCoordinates.push([...planePositions]);\r\n planePositions.clear();\r\n }\r\n }\r\n\r\n if (planeCoordinates.length == 3) {\r\n this.calculateNormalToRef(planeCoordinates, planeNormal);\r\n const planeNormalArray = this.calculateUnitVector(planeNormal.asArray());\r\n this.planeNormals.set(i, planeNormalArray);\r\n this.planeNormals.set(i - 1, planeNormalArray);\r\n this.planeNormals.set(i - 2, planeNormalArray);\r\n planeCoordinates.clear();\r\n }\r\n }\r\n }\r\n });\r\n\r\n //looping through it again to change/add the normals\r\n let totalVertices = 0;\r\n if (positions) {\r\n totalVertices = positions.length / 3;\r\n }\r\n const visitedVertices: boolean[] = [];\r\n vertexGroups.forEach(vertexGroup => {\r\n for (let i = vertexGroup.startIndex; i < vertexGroup.endIndex; i++) {\r\n const vertex = indices[i];\r\n const planeNormal = this.planeNormals.get(i);\r\n const currentNormal: number[] = [];\r\n const currentPosition: number[] = [];\r\n const currentUv: number[] = [];\r\n\r\n for (let i = 0; i < 3; i++) {\r\n currentNormal.push(normals[vertex * 3 + i]);\r\n currentPosition.push(positions[vertex * 3 + i]);\r\n }\r\n for (let i = 0; i < 2; i++) {\r\n currentUv.push(uvs[vertex * 2 + i]);\r\n }\r\n\r\n //vertex is already visited, therefor we add a new one.\r\n //else: replace the existing vertex\r\n if (visitedVertices[vertex]) {\r\n for (let i = 0; i < 3; i++) {\r\n if (planeNormal) {\r\n newNormals.push(planeNormal[i]);\r\n newPositions.push(currentPosition[i]);\r\n }\r\n }\r\n for (let i = 0; i < 2; i++) {\r\n newUvs.push(currentUv[i]);\r\n }\r\n\r\n indices[i] = totalVertices;\r\n totalVertices += 1;\r\n\r\n //handling submesh data\r\n if (vertexData.submeshData) {\r\n for (let i = 0; i < vertexData.submeshData.length; i++) {\r\n const submeshData = vertexData.submeshData[i];\r\n if (\r\n submeshData.verticesStart <= vertex &&\r\n submeshData.verticesStart + submeshData.verticesCount > vertex\r\n ) {\r\n submeshData.verticesCount += 1;\r\n } else if (submeshData.verticesStart > vertex) {\r\n submeshData.verticesStart += 1;\r\n }\r\n submeshData.indexCount = (positions.length + newPositions.length) / 3;\r\n submeshDataArray[i] = submeshData;\r\n }\r\n }\r\n } else {\r\n visitedVertices[vertex] = true;\r\n if (planeNormal) {\r\n for (let i = 0; i < 3; i++) {\r\n normals[vertex * 3 + i] = planeNormal[i];\r\n }\r\n }\r\n }\r\n }\r\n });\r\n const resultPositions = new Float32Array(positions.length + newPositions.length);\r\n resultPositions.set(positions);\r\n resultPositions.set(newPositions, positions.length);\r\n const resultNormals = new Float32Array(normals.length + newNormals.length);\r\n resultNormals.set(normals);\r\n resultNormals.set(newNormals, normals.length);\r\n const resultUvs = new Float32Array(uvs.length + newUvs.length);\r\n resultUvs.set(uvs);\r\n resultUvs.set(newUvs, uvs.length);\r\n newVertexData.positions = resultPositions;\r\n newVertexData.normals = resultNormals;\r\n newVertexData.uvs = resultUvs;\r\n newVertexData.indices = indices;\r\n newVertexData.submeshData = submeshDataArray;\r\n }\r\n }\r\n this.viewer.updateNormalSmoothingWarnings(feature, this.totalSelectedIndices / 3, false);\r\n return newVertexData;\r\n }\r\n\r\n update3dInternal() {}\r\n\r\n public delete3d(node: ITrackedClass, preserveCache?: boolean | undefined): void {\r\n this.viewer.updateNormalSmoothingWarnings(node, 0, true);\r\n super.delete3d(node);\r\n }\r\n //Formula for normal of a plane: (vectorA-VectorB) x (VectorB-VectorC)\r\n calculateNormalToRef(coordinates: number[][], normalVector: Vector3) {\r\n Vector3.FromArrayToRef(coordinates[0], 0, this.vectorA);\r\n Vector3.FromArrayToRef(coordinates[1], 0, this.vectorB);\r\n Vector3.FromArrayToRef(coordinates[2], 0, this.vectorC);\r\n this.vectorA.subtractInPlace(this.vectorB);\r\n this.vectorB.subtractInPlace(this.vectorC);\r\n return Vector3.CrossToRef(this.vectorB, this.vectorA, normalVector);\r\n }\r\n\r\n calculateSmoothedUnitVector(vertices: number[]) {\r\n let result: number[] = [];\r\n vertices.forEach(vertex => {\r\n const normalVector = this.planeNormals.get(Math.floor(vertex / 3));\r\n if (normalVector) {\r\n result = addArrays(result, normalVector);\r\n }\r\n });\r\n return this.calculateUnitVector(result);\r\n }\r\n\r\n calculateUnitVector(vector: number[]) {\r\n let divider = 0;\r\n const result: number[] = [];\r\n vector.forEach(element => {\r\n divider += element * element;\r\n });\r\n divider = Math.sqrt(divider);\r\n vector.forEach(element => {\r\n result.push(element / divider);\r\n });\r\n return result;\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\nimport type { Matrix } from \"../Maths/math.vector\";\r\nimport { Vector3, TmpVectors, Quaternion, Vector4, Vector2 } from \"../Maths/math.vector\";\r\nimport { Color4 } from \"../Maths/math.color\";\r\nimport type { Mesh } from \"../Meshes/mesh\";\r\nimport { BoundingInfo } from \"../Culling/boundingInfo\";\r\nimport { BoundingSphere } from \"../Culling/boundingSphere\";\r\nimport type { SolidParticleSystem } from \"./solidParticleSystem\";\r\nimport { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { Plane } from \"../Maths/math.plane\";\r\nimport type { Material } from \"../Materials/material\";\r\n/**\r\n * Represents one particle of a solid particle system.\r\n */\r\nexport class SolidParticle {\r\n /**\r\n * particle global index\r\n */\r\n public idx: number = 0;\r\n /**\r\n * particle identifier\r\n */\r\n public id: number = 0;\r\n /**\r\n * The color of the particle\r\n */\r\n public color: Nullable = new Color4(1.0, 1.0, 1.0, 1.0);\r\n /**\r\n * The world space position of the particle.\r\n */\r\n public position: Vector3 = Vector3.Zero();\r\n /**\r\n * The world space rotation of the particle. (Not use if rotationQuaternion is set)\r\n */\r\n public rotation: Vector3 = Vector3.Zero();\r\n /**\r\n * The world space rotation quaternion of the particle.\r\n */\r\n public rotationQuaternion: Nullable;\r\n /**\r\n * The scaling of the particle.\r\n */\r\n public scaling: Vector3 = Vector3.One();\r\n /**\r\n * The uvs of the particle.\r\n */\r\n public uvs: Vector4 = new Vector4(0.0, 0.0, 1.0, 1.0);\r\n /**\r\n * The current speed of the particle.\r\n */\r\n public velocity: Vector3 = Vector3.Zero();\r\n /**\r\n * The pivot point in the particle local space.\r\n */\r\n public pivot: Vector3 = Vector3.Zero();\r\n /**\r\n * Must the particle be translated from its pivot point in its local space ?\r\n * In this case, the pivot point is set at the origin of the particle local space and the particle is translated.\r\n * Default : false\r\n */\r\n public translateFromPivot: boolean = false;\r\n /**\r\n * Is the particle active or not ?\r\n */\r\n public alive: boolean = true;\r\n /**\r\n * Is the particle visible or not ?\r\n */\r\n public isVisible: boolean = true;\r\n /**\r\n * Index of this particle in the global \"positions\" array (Internal use)\r\n * @internal\r\n */\r\n public _pos: number = 0;\r\n /**\r\n * @internal Index of this particle in the global \"indices\" array (Internal use)\r\n */\r\n public _ind: number = 0;\r\n /**\r\n * @internal ModelShape of this particle (Internal use)\r\n */\r\n public _model: ModelShape;\r\n /**\r\n * ModelShape id of this particle\r\n */\r\n public shapeId: number = 0;\r\n /**\r\n * Index of the particle in its shape id\r\n */\r\n public idxInShape: number = 0;\r\n /**\r\n * @internal Reference to the shape model BoundingInfo object (Internal use)\r\n */\r\n public _modelBoundingInfo: BoundingInfo;\r\n private _boundingInfo: BoundingInfo;\r\n /**\r\n * @internal Reference to the SPS what the particle belongs to (Internal use)\r\n */\r\n public _sps: SolidParticleSystem;\r\n /**\r\n * @internal Still set as invisible in order to skip useless computations (Internal use)\r\n */\r\n public _stillInvisible: boolean = false;\r\n /**\r\n * @internal Last computed particle rotation matrix\r\n */\r\n public _rotationMatrix: number[] = [1.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 1.0];\r\n /**\r\n * Parent particle Id, if any.\r\n * Default null.\r\n */\r\n public parentId: Nullable = null;\r\n /**\r\n * The particle material identifier (integer) when MultiMaterials are enabled in the SPS.\r\n */\r\n public materialIndex: Nullable = null;\r\n /**\r\n * Custom object or properties.\r\n */\r\n public props: Nullable = null;\r\n /**\r\n * The culling strategy to use to check whether the solid particle must be culled or not when using isInFrustum().\r\n * The possible values are :\r\n * - AbstractMesh.CULLINGSTRATEGY_STANDARD\r\n * - AbstractMesh.CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY\r\n * - AbstractMesh.CULLINGSTRATEGY_OPTIMISTIC_INCLUSION\r\n * - AbstractMesh.CULLINGSTRATEGY_OPTIMISTIC_INCLUSION_THEN_BSPHERE_ONLY\r\n * The default value for solid particles is AbstractMesh.CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY\r\n * Please read each static variable documentation in the class AbstractMesh to get details about the culling process.\r\n * */\r\n public cullingStrategy = AbstractMesh.CULLINGSTRATEGY_BOUNDINGSPHERE_ONLY;\r\n\r\n /**\r\n * @internal Internal global position in the SPS.\r\n */\r\n public _globalPosition: Vector3 = Vector3.Zero();\r\n\r\n /**\r\n * Particle BoundingInfo object\r\n * @returns a BoundingInfo\r\n */\r\n public getBoundingInfo(): BoundingInfo {\r\n return this._boundingInfo;\r\n }\r\n\r\n /**\r\n * Returns true if there is already a bounding info\r\n */\r\n public get hasBoundingInfo(): boolean {\r\n return this._boundingInfo !== null;\r\n }\r\n\r\n /**\r\n * Creates a Solid Particle object.\r\n * Don't create particles manually, use instead the Solid Particle System internal tools like _addParticle()\r\n * @param particleIndex (integer) is the particle index in the Solid Particle System pool.\r\n * @param particleId (integer) is the particle identifier. Unless some particles are removed from the SPS, it's the same value than the particle idx.\r\n * @param positionIndex (integer) is the starting index of the particle vertices in the SPS \"positions\" array.\r\n * @param indiceIndex (integer) is the starting index of the particle indices in the SPS \"indices\" array.\r\n * @param model (ModelShape) is a reference to the model shape on what the particle is designed.\r\n * @param shapeId (integer) is the model shape identifier in the SPS.\r\n * @param idxInShape (integer) is the index of the particle in the current model (ex: the 10th box of addShape(box, 30))\r\n * @param sps defines the sps it is associated to\r\n * @param modelBoundingInfo is the reference to the model BoundingInfo used for intersection computations.\r\n * @param materialIndex is the particle material identifier (integer) when the MultiMaterials are enabled in the SPS.\r\n */\r\n constructor(\r\n particleIndex: number,\r\n particleId: number,\r\n positionIndex: number,\r\n indiceIndex: number,\r\n model: Nullable,\r\n shapeId: number,\r\n idxInShape: number,\r\n sps: SolidParticleSystem,\r\n modelBoundingInfo: Nullable = null,\r\n materialIndex: Nullable = null\r\n ) {\r\n this.idx = particleIndex;\r\n this.id = particleId;\r\n this._pos = positionIndex;\r\n this._ind = indiceIndex;\r\n this._model = model;\r\n this.shapeId = shapeId;\r\n this.idxInShape = idxInShape;\r\n this._sps = sps;\r\n if (modelBoundingInfo) {\r\n this._modelBoundingInfo = modelBoundingInfo;\r\n this._boundingInfo = new BoundingInfo(modelBoundingInfo.minimum, modelBoundingInfo.maximum);\r\n }\r\n if (materialIndex !== null) {\r\n this.materialIndex = materialIndex;\r\n }\r\n }\r\n /**\r\n * Copies the particle property values into the existing target : position, rotation, scaling, uvs, colors, pivot, parent, visibility, alive\r\n * @param target the particle target\r\n * @returns the current particle\r\n */\r\n public copyToRef(target: SolidParticle): SolidParticle {\r\n target.position.copyFrom(this.position);\r\n target.rotation.copyFrom(this.rotation);\r\n if (this.rotationQuaternion) {\r\n if (target.rotationQuaternion) {\r\n target.rotationQuaternion!.copyFrom(this.rotationQuaternion!);\r\n } else {\r\n target.rotationQuaternion = this.rotationQuaternion.clone();\r\n }\r\n }\r\n target.scaling.copyFrom(this.scaling);\r\n if (this.color) {\r\n if (target.color) {\r\n target.color!.copyFrom(this.color!);\r\n } else {\r\n target.color = this.color.clone();\r\n }\r\n }\r\n target.uvs.copyFrom(this.uvs);\r\n target.velocity.copyFrom(this.velocity);\r\n target.pivot.copyFrom(this.pivot);\r\n target.translateFromPivot = this.translateFromPivot;\r\n target.alive = this.alive;\r\n target.isVisible = this.isVisible;\r\n target.parentId = this.parentId;\r\n target.cullingStrategy = this.cullingStrategy;\r\n if (this.materialIndex !== null) {\r\n target.materialIndex = this.materialIndex;\r\n }\r\n return this;\r\n }\r\n /**\r\n * Legacy support, changed scale to scaling\r\n */\r\n public get scale(): Vector3 {\r\n return this.scaling;\r\n }\r\n\r\n /**\r\n * Legacy support, changed scale to scaling\r\n */\r\n public set scale(scale: Vector3) {\r\n this.scaling = scale;\r\n }\r\n\r\n /**\r\n * Legacy support, changed quaternion to rotationQuaternion\r\n */\r\n public get quaternion(): Nullable {\r\n return this.rotationQuaternion;\r\n }\r\n\r\n /**\r\n * Legacy support, changed quaternion to rotationQuaternion\r\n */\r\n public set quaternion(q: Nullable) {\r\n this.rotationQuaternion = q;\r\n }\r\n\r\n /**\r\n * Returns a boolean. True if the particle intersects another particle or another mesh, else false.\r\n * The intersection is computed on the particle bounding sphere and Axis Aligned Bounding Box (AABB)\r\n * @param target is the object (solid particle or mesh) what the intersection is computed against.\r\n * @returns true if it intersects\r\n */\r\n public intersectsMesh(target: Mesh | SolidParticle): boolean {\r\n if (!this._boundingInfo || !target.hasBoundingInfo) {\r\n return false;\r\n }\r\n if (this._sps._bSphereOnly) {\r\n return BoundingSphere.Intersects(this._boundingInfo.boundingSphere, target.getBoundingInfo().boundingSphere);\r\n }\r\n return this._boundingInfo.intersects(target.getBoundingInfo(), false);\r\n }\r\n\r\n /**\r\n * Returns `true` if the solid particle is within the frustum defined by the passed array of planes.\r\n * A particle is in the frustum if its bounding box intersects the frustum\r\n * @param frustumPlanes defines the frustum to test\r\n * @returns true if the particle is in the frustum planes\r\n */\r\n public isInFrustum(frustumPlanes: Plane[]): boolean {\r\n return this._boundingInfo !== null && this._boundingInfo.isInFrustum(frustumPlanes, this.cullingStrategy);\r\n }\r\n\r\n /**\r\n * get the rotation matrix of the particle\r\n * @internal\r\n */\r\n public getRotationMatrix(m: Matrix) {\r\n let quaternion: Quaternion;\r\n if (this.rotationQuaternion) {\r\n quaternion = this.rotationQuaternion;\r\n } else {\r\n quaternion = TmpVectors.Quaternion[0];\r\n const rotation = this.rotation;\r\n Quaternion.RotationYawPitchRollToRef(rotation.y, rotation.x, rotation.z, quaternion);\r\n }\r\n\r\n quaternion.toRotationMatrix(m);\r\n }\r\n}\r\n\r\n/**\r\n * Represents the shape of the model used by one particle of a solid particle system.\r\n * SPS internal tool, don't use it manually.\r\n */\r\nexport class ModelShape {\r\n /**\r\n * Get or set the shapeId\r\n * @deprecated Please use shapeId instead\r\n */\r\n public get shapeID(): number {\r\n return this.shapeId;\r\n }\r\n public set shapeID(shapeID: number) {\r\n this.shapeId = shapeID;\r\n }\r\n /**\r\n * The shape id\r\n * @internal\r\n */\r\n public shapeId: number;\r\n /**\r\n * flat array of model positions (internal use)\r\n * @internal\r\n */\r\n public _shape: Vector3[];\r\n /**\r\n * flat array of model UVs (internal use)\r\n * @internal\r\n */\r\n public _shapeUV: number[];\r\n /**\r\n * color array of the model\r\n * @internal\r\n */\r\n public _shapeColors: number[];\r\n /**\r\n * indices array of the model\r\n * @internal\r\n */\r\n public _indices: number[];\r\n /**\r\n * normals array of the model\r\n * @internal\r\n */\r\n public _normals: number[];\r\n /**\r\n * length of the shape in the model indices array (internal use)\r\n * @internal\r\n */\r\n public _indicesLength: number = 0;\r\n /**\r\n * Custom position function (internal use)\r\n * @internal\r\n */\r\n public _positionFunction: Nullable<(particle: SolidParticle, i: number, s: number) => void>;\r\n /**\r\n * Custom vertex function (internal use)\r\n * @internal\r\n */\r\n public _vertexFunction: Nullable<(particle: SolidParticle, vertex: Vector3, i: number) => void>;\r\n /**\r\n * Model material (internal use)\r\n * @internal\r\n */\r\n public _material: Nullable;\r\n\r\n /**\r\n * Creates a ModelShape object. This is an internal simplified reference to a mesh used as for a model to replicate particles from by the SPS.\r\n * SPS internal tool, don't use it manually.\r\n * @internal\r\n */\r\n constructor(\r\n id: number,\r\n shape: Vector3[],\r\n indices: number[],\r\n normals: number[],\r\n colors: number[],\r\n shapeUV: number[],\r\n posFunction: Nullable<(particle: SolidParticle, i: number, s: number) => void>,\r\n vtxFunction: Nullable<(particle: SolidParticle, vertex: Vector3, i: number) => void>,\r\n material: Nullable\r\n ) {\r\n this.shapeId = id;\r\n this._shape = shape;\r\n this._indices = indices;\r\n this._indicesLength = indices.length;\r\n this._shapeUV = shapeUV;\r\n this._shapeColors = colors;\r\n this._normals = normals;\r\n this._positionFunction = posFunction;\r\n this._vertexFunction = vtxFunction;\r\n this._material = material;\r\n }\r\n}\r\n\r\n/**\r\n * Represents a Depth Sorted Particle in the solid particle system.\r\n * @internal\r\n */\r\nexport class DepthSortedParticle {\r\n /**\r\n * Particle index\r\n */\r\n public idx: number = 0;\r\n /**\r\n * Index of the particle in the \"indices\" array\r\n */\r\n public ind: number = 0;\r\n /**\r\n * Length of the particle shape in the \"indices\" array\r\n */\r\n public indicesLength: number = 0;\r\n /**\r\n * Squared distance from the particle to the camera\r\n */\r\n public sqDistance: number = 0.0;\r\n /**\r\n * Material index when used with MultiMaterials\r\n */\r\n public materialIndex: number = 0;\r\n\r\n /**\r\n * Creates a new sorted particle\r\n * @param idx\r\n * @param ind\r\n * @param indLength\r\n * @param materialIndex\r\n */\r\n constructor(idx: number, ind: number, indLength: number, materialIndex: number) {\r\n this.idx = idx;\r\n this.ind = ind;\r\n this.indicesLength = indLength;\r\n this.materialIndex = materialIndex;\r\n }\r\n}\r\n\r\n/**\r\n * Represents a solid particle vertex\r\n */\r\nexport class SolidParticleVertex {\r\n /**\r\n * Vertex position\r\n */\r\n public position: Vector3;\r\n /**\r\n * Vertex color\r\n */\r\n public color: Color4;\r\n /**\r\n * Vertex UV\r\n */\r\n public uv: Vector2;\r\n /**\r\n * Creates a new solid particle vertex\r\n */\r\n constructor() {\r\n this.position = Vector3.Zero();\r\n this.color = new Color4(1.0, 1.0, 1.0, 1.0);\r\n this.uv = Vector2.Zero();\r\n }\r\n // Getters and Setters for back-compatibility\r\n /** Vertex x coordinate */\r\n public get x(): number {\r\n return this.position.x;\r\n }\r\n public set x(val: number) {\r\n this.position.x = val;\r\n }\r\n /** Vertex y coordinate */\r\n public get y(): number {\r\n return this.position.y;\r\n }\r\n public set y(val: number) {\r\n this.position.y = val;\r\n }\r\n /** Vertex z coordinate */\r\n public get z(): number {\r\n return this.position.z;\r\n }\r\n public set z(val: number) {\r\n this.position.z = val;\r\n }\r\n}\r\n","import type { Nullable, IndicesArray, FloatArray } from \"../types\";\r\nimport { Vector3, Matrix, TmpVectors, Quaternion } from \"../Maths/math.vector\";\r\nimport { Color4 } from \"../Maths/math.color\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport { VertexData } from \"../Meshes/mesh.vertexData\";\r\nimport { Mesh } from \"../Meshes/mesh\";\r\nimport { CreateDisc } from \"../Meshes/Builders/discBuilder\";\r\nimport { EngineStore } from \"../Engines/engineStore\";\r\nimport type { Scene, IDisposable } from \"../scene\";\r\nimport { DepthSortedParticle, SolidParticle, ModelShape, SolidParticleVertex } from \"./solidParticle\";\r\nimport type { TargetCamera } from \"../Cameras/targetCamera\";\r\nimport { BoundingInfo } from \"../Culling/boundingInfo\";\r\nimport { Axis } from \"../Maths/math.axis\";\r\nimport { SubMesh } from \"../Meshes/subMesh\";\r\nimport type { Material } from \"../Materials/material\";\r\nimport { StandardMaterial } from \"../Materials/standardMaterial\";\r\nimport { MultiMaterial } from \"../Materials/multiMaterial\";\r\nimport type { PickingInfo } from \"../Collisions/pickingInfo\";\r\n\r\n/**\r\n * The SPS is a single updatable mesh. The solid particles are simply separate parts or faces of this big mesh.\r\n *As it is just a mesh, the SPS has all the same properties than any other BJS mesh : not more, not less. It can be scaled, rotated, translated, enlighted, textured, moved, etc.\r\n\r\n * The SPS is also a particle system. It provides some methods to manage the particles.\r\n * However it is behavior agnostic. This means it has no emitter, no particle physics, no particle recycler. You have to implement your own behavior.\r\n *\r\n * Full documentation here : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_intro\r\n */\r\nexport class SolidParticleSystem implements IDisposable {\r\n /**\r\n * The SPS array of Solid Particle objects. Just access each particle as with any classic array.\r\n * Example : var p = SPS.particles[i];\r\n */\r\n public particles: SolidParticle[] = new Array();\r\n /**\r\n * The SPS total number of particles. Read only. Use SPS.counter instead if you need to set your own value.\r\n */\r\n public nbParticles: number = 0;\r\n /**\r\n * If the particles must ever face the camera (default false). Useful for planar particles.\r\n */\r\n public billboard: boolean = false;\r\n /**\r\n * Recompute normals when adding a shape\r\n */\r\n public recomputeNormals: boolean = false;\r\n /**\r\n * This a counter ofr your own usage. It's not set by any SPS functions.\r\n */\r\n public counter: number = 0;\r\n /**\r\n * The SPS name. This name is also given to the underlying mesh.\r\n */\r\n public name: string;\r\n /**\r\n * The SPS mesh. It's a standard BJS Mesh, so all the methods from the Mesh class are available.\r\n */\r\n public mesh: Mesh;\r\n /**\r\n * This empty object is intended to store some SPS specific or temporary values in order to lower the Garbage Collector activity.\r\n * Please read : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/optimize_sps#limit-garbage-collection\r\n */\r\n public vars: any = {};\r\n /**\r\n * This array is populated when the SPS is set as 'pickable'.\r\n * Each key of this array is a `faceId` value that you can get from a pickResult object.\r\n * Each element of this array is an object `{idx: int, faceId: int}`.\r\n * `idx` is the picked particle index in the `SPS.particles` array\r\n * `faceId` is the picked face index counted within this particle.\r\n * This array is the first element of the pickedBySubMesh array : sps.pickBySubMesh[0].\r\n * It's not pertinent to use it when using a SPS with the support for MultiMaterial enabled.\r\n * Use the method SPS.pickedParticle(pickingInfo) instead.\r\n * Please read : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/picking_sps\r\n */\r\n public pickedParticles: { idx: number; faceId: number }[];\r\n /**\r\n * This array is populated when the SPS is set as 'pickable'\r\n * Each key of this array is a submesh index.\r\n * Each element of this array is a second array defined like this :\r\n * Each key of this second array is a `faceId` value that you can get from a pickResult object.\r\n * Each element of this second array is an object `{idx: int, faceId: int}`.\r\n * `idx` is the picked particle index in the `SPS.particles` array\r\n * `faceId` is the picked face index counted within this particle.\r\n * It's better to use the method SPS.pickedParticle(pickingInfo) rather than using directly this array.\r\n * Please read : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/picking_sps\r\n */\r\n public pickedBySubMesh: { idx: number; faceId: number }[][];\r\n /**\r\n * This array is populated when `enableDepthSort` is set to true.\r\n * Each element of this array is an instance of the class DepthSortedParticle.\r\n */\r\n public depthSortedParticles: DepthSortedParticle[];\r\n\r\n /**\r\n * If the particle intersection must be computed only with the bounding sphere (no bounding box computation, so faster). (Internal use only)\r\n * @internal\r\n */\r\n public _bSphereOnly: boolean = false;\r\n /**\r\n * A number to multiply the bounding sphere radius by in order to reduce it for instance. (Internal use only)\r\n * @internal\r\n */\r\n public _bSphereRadiusFactor: number = 1.0;\r\n\r\n protected _scene: Scene;\r\n protected _positions: number[] = new Array();\r\n protected _indices: number[] = new Array();\r\n protected _normals: number[] = new Array();\r\n protected _colors: number[] = new Array();\r\n protected _uvs: number[] = new Array();\r\n protected _indices32: IndicesArray; // used as depth sorted array if depth sort enabled, else used as typed indices\r\n protected _positions32: Float32Array; // updated positions for the VBO\r\n protected _normals32: Float32Array; // updated normals for the VBO\r\n protected _fixedNormal32: Float32Array; // initial normal references\r\n protected _colors32: Float32Array;\r\n protected _uvs32: Float32Array;\r\n protected _index: number = 0; // indices index\r\n protected _updatable: boolean = true;\r\n protected _pickable: boolean = false;\r\n protected _isVisibilityBoxLocked = false;\r\n protected _alwaysVisible: boolean = false;\r\n protected _depthSort: boolean = false;\r\n protected _expandable: boolean = false;\r\n protected _shapeCounter: number = 0;\r\n protected _copy: SolidParticle = new SolidParticle(0, 0, 0, 0, null, 0, 0, this);\r\n protected _color: Color4 = new Color4(0, 0, 0, 0);\r\n protected _computeParticleColor: boolean = true;\r\n protected _computeParticleTexture: boolean = true;\r\n protected _computeParticleRotation: boolean = true;\r\n protected _computeParticleVertex: boolean = false;\r\n protected _computeBoundingBox: boolean = false;\r\n protected _autoFixFaceOrientation: boolean = false;\r\n protected _depthSortParticles: boolean = true;\r\n protected _camera: TargetCamera;\r\n protected _mustUnrotateFixedNormals = false;\r\n protected _particlesIntersect: boolean = false;\r\n protected _needs32Bits: boolean = false;\r\n protected _isNotBuilt: boolean = true;\r\n protected _lastParticleId: number = 0;\r\n protected _idxOfId: number[] = []; // array : key = particle.id / value = particle.idx\r\n protected _multimaterialEnabled: boolean = false;\r\n protected _useModelMaterial: boolean = false;\r\n protected _indicesByMaterial: number[];\r\n protected _materialIndexes: number[];\r\n protected _depthSortFunction = (p1: DepthSortedParticle, p2: DepthSortedParticle) => p2.sqDistance - p1.sqDistance;\r\n protected _materialSortFunction = (p1: DepthSortedParticle, p2: DepthSortedParticle) => p1.materialIndex - p2.materialIndex;\r\n protected _materials: Material[];\r\n protected _multimaterial: MultiMaterial;\r\n protected _materialIndexesById: any;\r\n protected _defaultMaterial: Material;\r\n protected _autoUpdateSubMeshes: boolean = false;\r\n protected _tmpVertex: SolidParticleVertex;\r\n protected _recomputeInvisibles: boolean = false;\r\n\r\n /**\r\n * Creates a SPS (Solid Particle System) object.\r\n * @param name (String) is the SPS name, this will be the underlying mesh name.\r\n * @param scene (Scene) is the scene in which the SPS is added.\r\n * @param options defines the options of the sps e.g.\r\n * * updatable (optional boolean, default true) : if the SPS must be updatable or immutable.\r\n * * isPickable (optional boolean, default false) : if the solid particles must be pickable.\r\n * * enableDepthSort (optional boolean, default false) : if the solid particles must be sorted in the geometry according to their distance to the camera.\r\n * * useModelMaterial (optional boolean, default false) : if the model materials must be used to create the SPS multimaterial. This enables the multimaterial supports of the SPS.\r\n * * enableMultiMaterial (optional boolean, default false) : if the solid particles can be given different materials.\r\n * * expandable (optional boolean, default false) : if particles can still be added after the initial SPS mesh creation.\r\n * * particleIntersection (optional boolean, default false) : if the solid particle intersections must be computed.\r\n * * boundingSphereOnly (optional boolean, default false) : if the particle intersection must be computed only with the bounding sphere (no bounding box computation, so faster).\r\n * * bSphereRadiusFactor (optional float, default 1.0) : a number to multiply the bounding sphere radius by in order to reduce it for instance.\r\n * * computeBoundingBox (optional boolean, default false): if the bounding box of the entire SPS will be computed (for occlusion detection, for example). If it is false, the bounding box will be the bounding box of the first particle.\r\n * * autoFixFaceOrientation (optional boolean, default false): if the particle face orientations will be flipped for transformations that change orientation (scale (-1, 1, 1), for example)\r\n * @param options.updatable\r\n * @param options.isPickable\r\n * @param options.enableDepthSort\r\n * @param options.particleIntersection\r\n * @param options.boundingSphereOnly\r\n * @param options.bSphereRadiusFactor\r\n * @param options.expandable\r\n * @param options.useModelMaterial\r\n * @param options.enableMultiMaterial\r\n * @param options.computeBoundingBox\r\n * @param options.autoFixFaceOrientation\r\n * @example bSphereRadiusFactor = 1.0 / Math.sqrt(3.0) => the bounding sphere exactly matches a spherical mesh.\r\n */\r\n constructor(\r\n name: string,\r\n scene: Scene,\r\n options?: {\r\n updatable?: boolean;\r\n isPickable?: boolean;\r\n enableDepthSort?: boolean;\r\n particleIntersection?: boolean;\r\n boundingSphereOnly?: boolean;\r\n bSphereRadiusFactor?: number;\r\n expandable?: boolean;\r\n useModelMaterial?: boolean;\r\n enableMultiMaterial?: boolean;\r\n computeBoundingBox?: boolean;\r\n autoFixFaceOrientation?: boolean;\r\n }\r\n ) {\r\n this.name = name;\r\n this._scene = scene || EngineStore.LastCreatedScene;\r\n this._camera = scene.activeCamera;\r\n this._pickable = options ? options.isPickable : false;\r\n this._depthSort = options ? options.enableDepthSort : false;\r\n this._multimaterialEnabled = options ? options.enableMultiMaterial : false;\r\n this._useModelMaterial = options ? options.useModelMaterial : false;\r\n this._multimaterialEnabled = this._useModelMaterial ? true : this._multimaterialEnabled;\r\n this._expandable = options ? options.expandable : false;\r\n this._particlesIntersect = options ? options.particleIntersection : false;\r\n this._bSphereOnly = options ? options.boundingSphereOnly : false;\r\n this._bSphereRadiusFactor = options && options.bSphereRadiusFactor ? options.bSphereRadiusFactor : 1.0;\r\n this._computeBoundingBox = options?.computeBoundingBox ? options.computeBoundingBox : false;\r\n this._autoFixFaceOrientation = options?.autoFixFaceOrientation ? options.autoFixFaceOrientation : false;\r\n if (options && options.updatable !== undefined) {\r\n this._updatable = options.updatable;\r\n } else {\r\n this._updatable = true;\r\n }\r\n if (this._pickable) {\r\n this.pickedBySubMesh = [[]];\r\n this.pickedParticles = this.pickedBySubMesh[0];\r\n }\r\n if (this._depthSort || this._multimaterialEnabled) {\r\n this.depthSortedParticles = [];\r\n }\r\n if (this._multimaterialEnabled) {\r\n this._multimaterial = new MultiMaterial(this.name + \"MultiMaterial\", this._scene);\r\n this._materials = [];\r\n this._materialIndexesById = {};\r\n }\r\n this._tmpVertex = new SolidParticleVertex();\r\n }\r\n\r\n /**\r\n * Builds the SPS underlying mesh. Returns a standard Mesh.\r\n * If no model shape was added to the SPS, the returned mesh is just a single triangular plane.\r\n * @returns the created mesh\r\n */\r\n public buildMesh(): Mesh {\r\n if (!this._isNotBuilt && this.mesh) {\r\n return this.mesh;\r\n }\r\n if (this.nbParticles === 0 && !this.mesh) {\r\n const triangle = CreateDisc(\"\", { radius: 1, tessellation: 3 }, this._scene);\r\n this.addShape(triangle, 1);\r\n triangle.dispose();\r\n }\r\n this._indices32 = this._needs32Bits ? new Uint32Array(this._indices) : new Uint16Array(this._indices);\r\n this._positions32 = new Float32Array(this._positions);\r\n this._uvs32 = new Float32Array(this._uvs);\r\n this._colors32 = new Float32Array(this._colors);\r\n\r\n if (!this.mesh) {\r\n // in case it's already expanded\r\n const mesh = new Mesh(this.name, this._scene);\r\n this.mesh = mesh;\r\n }\r\n if (!this._updatable && this._multimaterialEnabled) {\r\n this._sortParticlesByMaterial(); // this may reorder the indices32\r\n }\r\n if (this.recomputeNormals) {\r\n VertexData.ComputeNormals(this._positions32, this._indices32, this._normals);\r\n }\r\n\r\n this._normals32 = new Float32Array(this._normals);\r\n this._fixedNormal32 = new Float32Array(this._normals);\r\n if (this._mustUnrotateFixedNormals) {\r\n // the particles could be created already rotated in the mesh with a positionFunction\r\n this._unrotateFixedNormals();\r\n }\r\n const vertexData = new VertexData();\r\n vertexData.indices = this._depthSort ? this._indices : this._indices32;\r\n vertexData.set(this._positions32, VertexBuffer.PositionKind);\r\n vertexData.set(this._normals32, VertexBuffer.NormalKind);\r\n\r\n if (this._uvs32.length > 0) {\r\n vertexData.set(this._uvs32, VertexBuffer.UVKind);\r\n }\r\n if (this._colors32.length > 0) {\r\n vertexData.set(this._colors32, VertexBuffer.ColorKind);\r\n }\r\n\r\n vertexData.applyToMesh(this.mesh, this._updatable);\r\n this.mesh.isPickable = this._pickable;\r\n\r\n if (this._pickable) {\r\n let faceId = 0;\r\n for (let p = 0; p < this.nbParticles; p++) {\r\n const part = this.particles[p];\r\n const lind = part._model._indicesLength;\r\n for (let i = 0; i < lind; i++) {\r\n const f = i % 3;\r\n if (f == 0) {\r\n const pickedData = { idx: part.idx, faceId: faceId };\r\n this.pickedParticles[faceId] = pickedData;\r\n faceId++;\r\n }\r\n }\r\n }\r\n }\r\n\r\n if (this._multimaterialEnabled) {\r\n this.setMultiMaterial(this._materials);\r\n }\r\n\r\n if (!this._expandable) {\r\n // free memory\r\n if (!this._depthSort && !this._multimaterialEnabled && !this._autoFixFaceOrientation) {\r\n (this._indices) = null;\r\n }\r\n (this._positions) = null;\r\n (this._normals) = null;\r\n (this._uvs) = null;\r\n (this._colors) = null;\r\n\r\n if (!this._updatable) {\r\n this.particles.length = 0;\r\n }\r\n }\r\n this._isNotBuilt = false;\r\n this.recomputeNormals = false;\r\n this._recomputeInvisibles = true;\r\n return this.mesh;\r\n }\r\n\r\n /**\r\n * Digests the mesh and generates as many solid particles in the system as wanted. Returns the SPS.\r\n * These particles will have the same geometry than the mesh parts and will be positioned at the same localisation than the mesh original places.\r\n * Thus the particles generated from `digest()` have their property `position` set yet.\r\n * @param mesh ( Mesh ) is the mesh to be digested\r\n * @param options {facetNb} (optional integer, default 1) is the number of mesh facets per particle, this parameter is overridden by the parameter `number` if any\r\n * {delta} (optional integer, default 0) is the random extra number of facets per particle , each particle will have between `facetNb` and `facetNb + delta` facets\r\n * {number} (optional positive integer) is the wanted number of particles : each particle is built with `mesh_total_facets / number` facets\r\n * {storage} (optional existing array) is an array where the particles will be stored for a further use instead of being inserted in the SPS.\r\n * @param options.facetNb\r\n * @param options.number\r\n * @param options.delta\r\n * @param options.storage\r\n * @returns the current SPS\r\n */\r\n public digest(mesh: Mesh, options?: { facetNb?: number; number?: number; delta?: number; storage?: [] }): SolidParticleSystem {\r\n let size: number = (options && options.facetNb) || 1;\r\n let number: number = (options && options.number) || 0;\r\n let delta: number = (options && options.delta) || 0;\r\n const meshPos = mesh.getVerticesData(VertexBuffer.PositionKind);\r\n const meshInd = mesh.getIndices();\r\n const meshUV = mesh.getVerticesData(VertexBuffer.UVKind);\r\n const meshCol = mesh.getVerticesData(VertexBuffer.ColorKind);\r\n const meshNor = mesh.getVerticesData(VertexBuffer.NormalKind);\r\n const storage = options && options.storage ? options.storage : null;\r\n\r\n let f: number = 0; // facet counter\r\n const totalFacets: number = meshInd.length / 3; // a facet is a triangle, so 3 indices\r\n // compute size from number\r\n if (number) {\r\n number = number > totalFacets ? totalFacets : number;\r\n size = Math.round(totalFacets / number);\r\n delta = 0;\r\n } else {\r\n size = size > totalFacets ? totalFacets : size;\r\n }\r\n\r\n const facetPos: number[] = []; // submesh positions\r\n const facetNor: number[] = [];\r\n const facetInd: number[] = []; // submesh indices\r\n const facetUV: number[] = []; // submesh UV\r\n const facetCol: number[] = []; // submesh colors\r\n const barycenter: Vector3 = Vector3.Zero();\r\n const sizeO: number = size;\r\n\r\n while (f < totalFacets) {\r\n size = sizeO + Math.floor((1 + delta) * Math.random());\r\n if (f > totalFacets - size) {\r\n size = totalFacets - f;\r\n }\r\n // reset temp arrays\r\n facetPos.length = 0;\r\n facetNor.length = 0;\r\n facetInd.length = 0;\r\n facetUV.length = 0;\r\n facetCol.length = 0;\r\n\r\n // iterate over \"size\" facets\r\n let fi: number = 0;\r\n for (let j = f * 3; j < (f + size) * 3; j++) {\r\n facetInd.push(fi);\r\n const i: number = meshInd[j];\r\n const i3: number = i * 3;\r\n facetPos.push(meshPos[i3], meshPos[i3 + 1], meshPos[i3 + 2]);\r\n facetNor.push(meshNor[i3], meshNor[i3 + 1], meshNor[i3 + 2]);\r\n if (meshUV) {\r\n const i2: number = i * 2;\r\n facetUV.push(meshUV[i2], meshUV[i2 + 1]);\r\n }\r\n if (meshCol) {\r\n const i4: number = i * 4;\r\n facetCol.push(meshCol[i4], meshCol[i4 + 1], meshCol[i4 + 2], meshCol[i4 + 3]);\r\n }\r\n fi++;\r\n }\r\n\r\n // create a model shape for each single particle\r\n let idx: number = this.nbParticles;\r\n const shape: Vector3[] = this._posToShape(facetPos);\r\n const shapeUV: number[] = this._uvsToShapeUV(facetUV);\r\n const shapeInd = facetInd.slice();\r\n const shapeCol = facetCol.slice();\r\n const shapeNor = facetNor.slice();\r\n\r\n // compute the barycenter of the shape\r\n barycenter.copyFromFloats(0, 0, 0);\r\n let v: number;\r\n for (v = 0; v < shape.length; v++) {\r\n barycenter.addInPlace(shape[v]);\r\n }\r\n barycenter.scaleInPlace(1 / shape.length);\r\n\r\n // shift the shape from its barycenter to the origin\r\n // and compute the BBox required for intersection.\r\n const minimum: Vector3 = new Vector3(Infinity, Infinity, Infinity);\r\n const maximum: Vector3 = new Vector3(-Infinity, -Infinity, -Infinity);\r\n for (v = 0; v < shape.length; v++) {\r\n shape[v].subtractInPlace(barycenter);\r\n minimum.minimizeInPlaceFromFloats(shape[v].x, shape[v].y, shape[v].z);\r\n maximum.maximizeInPlaceFromFloats(shape[v].x, shape[v].y, shape[v].z);\r\n }\r\n let bInfo;\r\n if (this._particlesIntersect) {\r\n bInfo = new BoundingInfo(minimum, maximum);\r\n }\r\n let material = null;\r\n if (this._useModelMaterial) {\r\n material = mesh.material ? mesh.material : this._setDefaultMaterial();\r\n }\r\n const modelShape = new ModelShape(this._shapeCounter, shape, shapeInd, shapeNor, shapeCol, shapeUV, null, null, material);\r\n\r\n // add the particle in the SPS\r\n const currentPos = this._positions.length;\r\n const currentInd = this._indices.length;\r\n this._meshBuilder(\r\n this._index,\r\n currentInd,\r\n shape,\r\n this._positions,\r\n shapeInd,\r\n this._indices,\r\n facetUV,\r\n this._uvs,\r\n shapeCol,\r\n this._colors,\r\n shapeNor,\r\n this._normals,\r\n idx,\r\n 0,\r\n null,\r\n modelShape\r\n );\r\n this._addParticle(idx, this._lastParticleId, currentPos, currentInd, modelShape, this._shapeCounter, 0, bInfo, storage);\r\n // initialize the particle position\r\n this.particles[this.nbParticles].position.addInPlace(barycenter);\r\n\r\n if (!storage) {\r\n this._index += shape.length;\r\n idx++;\r\n this.nbParticles++;\r\n this._lastParticleId++;\r\n }\r\n this._shapeCounter++;\r\n f += size;\r\n }\r\n this._isNotBuilt = true; // buildMesh() is now expected for setParticles() to work\r\n return this;\r\n }\r\n\r\n /**\r\n * Unrotate the fixed normals in case the mesh was built with pre-rotated particles, ex : use of positionFunction in addShape()\r\n * @internal\r\n */\r\n protected _unrotateFixedNormals() {\r\n let index = 0;\r\n let idx = 0;\r\n const tmpNormal = TmpVectors.Vector3[0];\r\n const quaternion = TmpVectors.Quaternion[0];\r\n const invertedRotMatrix = TmpVectors.Matrix[0];\r\n for (let p = 0; p < this.particles.length; p++) {\r\n const particle = this.particles[p];\r\n const shape = particle._model._shape;\r\n\r\n // computing the inverse of the rotation matrix from the quaternion\r\n // is equivalent to computing the matrix of the inverse quaternion, i.e of the conjugate quaternion\r\n if (particle.rotationQuaternion) {\r\n particle.rotationQuaternion.conjugateToRef(quaternion);\r\n } else {\r\n const rotation = particle.rotation;\r\n Quaternion.RotationYawPitchRollToRef(rotation.y, rotation.x, rotation.z, quaternion);\r\n quaternion.conjugateInPlace();\r\n }\r\n quaternion.toRotationMatrix(invertedRotMatrix);\r\n\r\n for (let pt = 0; pt < shape.length; pt++) {\r\n idx = index + pt * 3;\r\n Vector3.TransformNormalFromFloatsToRef(this._normals32[idx], this._normals32[idx + 1], this._normals32[idx + 2], invertedRotMatrix, tmpNormal);\r\n tmpNormal.toArray(this._fixedNormal32, idx);\r\n }\r\n index = idx + 3;\r\n }\r\n }\r\n\r\n /**\r\n * Resets the temporary working copy particle\r\n * @internal\r\n */\r\n protected _resetCopy() {\r\n const copy = this._copy;\r\n copy.position.setAll(0);\r\n copy.rotation.setAll(0);\r\n copy.rotationQuaternion = null;\r\n copy.scaling.setAll(1);\r\n copy.uvs.copyFromFloats(0.0, 0.0, 1.0, 1.0);\r\n copy.color = null;\r\n copy.translateFromPivot = false;\r\n copy.shapeId = 0;\r\n copy.materialIndex = null;\r\n }\r\n\r\n /**\r\n * Inserts the shape model geometry in the global SPS mesh by updating the positions, indices, normals, colors, uvs arrays\r\n * @param p the current index in the positions array to be updated\r\n * @param ind the current index in the indices array\r\n * @param shape a Vector3 array, the shape geometry\r\n * @param positions the positions array to be updated\r\n * @param meshInd the shape indices array\r\n * @param indices the indices array to be updated\r\n * @param meshUV the shape uv array\r\n * @param uvs the uv array to be updated\r\n * @param meshCol the shape color array\r\n * @param colors the color array to be updated\r\n * @param meshNor the shape normals array\r\n * @param normals the normals array to be updated\r\n * @param idx the particle index\r\n * @param idxInShape the particle index in its shape\r\n * @param options the addShape() method passed options\r\n * @param model\r\n * @model the particle model\r\n * @internal\r\n */\r\n protected _meshBuilder(\r\n p: number,\r\n ind: number,\r\n shape: Vector3[],\r\n positions: number[],\r\n meshInd: IndicesArray,\r\n indices: number[],\r\n meshUV: number[] | Float32Array,\r\n uvs: number[],\r\n meshCol: number[] | Float32Array,\r\n colors: number[],\r\n meshNor: number[] | Float32Array,\r\n normals: number[],\r\n idx: number,\r\n idxInShape: number,\r\n options: any,\r\n model: ModelShape\r\n ): SolidParticle {\r\n let i;\r\n let u = 0;\r\n let c = 0;\r\n let n = 0;\r\n\r\n this._resetCopy();\r\n const copy = this._copy;\r\n const storeApart = options && options.storage ? true : false;\r\n copy.idx = idx;\r\n copy.idxInShape = idxInShape;\r\n copy.shapeId = model.shapeId;\r\n if (this._useModelMaterial) {\r\n const materialId = model._material!.uniqueId;\r\n const materialIndexesById = this._materialIndexesById;\r\n if (!Object.prototype.hasOwnProperty.call(materialIndexesById, materialId)) {\r\n materialIndexesById[materialId] = this._materials.length;\r\n this._materials.push(model._material!);\r\n }\r\n const matIdx = materialIndexesById[materialId];\r\n copy.materialIndex = matIdx;\r\n }\r\n\r\n if (options && options.positionFunction) {\r\n // call to custom positionFunction\r\n options.positionFunction(copy, idx, idxInShape);\r\n this._mustUnrotateFixedNormals = true;\r\n }\r\n\r\n // in case the particle geometry must NOT be inserted in the SPS mesh geometry\r\n if (storeApart) {\r\n return copy;\r\n }\r\n\r\n const rotMatrix = TmpVectors.Matrix[0];\r\n const tmpVertex = this._tmpVertex;\r\n const tmpVector = tmpVertex.position;\r\n const tmpColor = tmpVertex.color;\r\n const tmpUV = tmpVertex.uv;\r\n const tmpRotated = TmpVectors.Vector3[1];\r\n const pivotBackTranslation = TmpVectors.Vector3[2];\r\n const scaledPivot = TmpVectors.Vector3[3];\r\n Matrix.IdentityToRef(rotMatrix);\r\n copy.getRotationMatrix(rotMatrix);\r\n\r\n copy.pivot.multiplyToRef(copy.scaling, scaledPivot);\r\n\r\n if (copy.translateFromPivot) {\r\n pivotBackTranslation.setAll(0.0);\r\n } else {\r\n pivotBackTranslation.copyFrom(scaledPivot);\r\n }\r\n\r\n const someVertexFunction = options && options.vertexFunction;\r\n for (i = 0; i < shape.length; i++) {\r\n tmpVector.copyFrom(shape[i]);\r\n if (copy.color) {\r\n tmpColor.copyFrom(copy.color);\r\n }\r\n if (meshUV) {\r\n tmpUV.copyFromFloats(meshUV[u], meshUV[u + 1]);\r\n }\r\n if (someVertexFunction) {\r\n options.vertexFunction(copy, tmpVertex, i);\r\n }\r\n\r\n tmpVector.multiplyInPlace(copy.scaling).subtractInPlace(scaledPivot);\r\n Vector3.TransformCoordinatesToRef(tmpVector, rotMatrix, tmpRotated);\r\n tmpRotated.addInPlace(pivotBackTranslation).addInPlace(copy.position);\r\n positions.push(tmpRotated.x, tmpRotated.y, tmpRotated.z);\r\n\r\n if (meshUV) {\r\n const copyUvs = copy.uvs;\r\n uvs.push((copyUvs.z - copyUvs.x) * tmpUV.x + copyUvs.x, (copyUvs.w - copyUvs.y) * tmpUV.y + copyUvs.y);\r\n u += 2;\r\n }\r\n\r\n if (copy.color) {\r\n this._color.copyFrom(tmpColor);\r\n } else {\r\n const color = this._color;\r\n if (meshCol && meshCol[c] !== undefined) {\r\n color.r = meshCol[c];\r\n color.g = meshCol[c + 1];\r\n color.b = meshCol[c + 2];\r\n color.a = meshCol[c + 3];\r\n } else {\r\n color.r = 1.0;\r\n color.g = 1.0;\r\n color.b = 1.0;\r\n color.a = 1.0;\r\n }\r\n }\r\n colors.push(this._color.r, this._color.g, this._color.b, this._color.a);\r\n c += 4;\r\n\r\n if (!this.recomputeNormals && meshNor) {\r\n Vector3.TransformNormalFromFloatsToRef(meshNor[n], meshNor[n + 1], meshNor[n + 2], rotMatrix, tmpVector);\r\n normals.push(tmpVector.x, tmpVector.y, tmpVector.z);\r\n n += 3;\r\n }\r\n }\r\n\r\n for (i = 0; i < meshInd.length; i++) {\r\n const current_ind = p + meshInd[i];\r\n indices.push(current_ind);\r\n if (current_ind > 65535) {\r\n this._needs32Bits = true;\r\n }\r\n }\r\n\r\n if (this._depthSort || this._multimaterialEnabled) {\r\n const matIndex = copy.materialIndex !== null ? copy.materialIndex : 0;\r\n this.depthSortedParticles.push(new DepthSortedParticle(idx, ind, meshInd.length, matIndex));\r\n }\r\n\r\n return copy;\r\n }\r\n\r\n /**\r\n * Returns a shape Vector3 array from positions float array\r\n * @param positions float array\r\n * @returns a vector3 array\r\n * @internal\r\n */\r\n protected _posToShape(positions: number[] | Float32Array): Vector3[] {\r\n const shape = [];\r\n for (let i = 0; i < positions.length; i += 3) {\r\n shape.push(Vector3.FromArray(positions, i));\r\n }\r\n return shape;\r\n }\r\n\r\n /**\r\n * Returns a shapeUV array from a float uvs (array deep copy)\r\n * @param uvs as a float array\r\n * @returns a shapeUV array\r\n * @internal\r\n */\r\n protected _uvsToShapeUV(uvs: number[] | Float32Array): number[] {\r\n const shapeUV = [];\r\n if (uvs) {\r\n for (let i = 0; i < uvs.length; i++) {\r\n shapeUV.push(uvs[i]);\r\n }\r\n }\r\n return shapeUV;\r\n }\r\n\r\n /**\r\n * Adds a new particle object in the particles array\r\n * @param idx particle index in particles array\r\n * @param id particle id\r\n * @param idxpos positionIndex : the starting index of the particle vertices in the SPS \"positions\" array\r\n * @param idxind indiceIndex : he starting index of the particle indices in the SPS \"indices\" array\r\n * @param model particle ModelShape object\r\n * @param shapeId model shape identifier\r\n * @param idxInShape index of the particle in the current model\r\n * @param bInfo model bounding info object\r\n * @param storage target storage array, if any\r\n * @internal\r\n */\r\n protected _addParticle(\r\n idx: number,\r\n id: number,\r\n idxpos: number,\r\n idxind: number,\r\n model: ModelShape,\r\n shapeId: number,\r\n idxInShape: number,\r\n bInfo: Nullable = null,\r\n storage: Nullable<[]> = null\r\n ): SolidParticle {\r\n const sp = new SolidParticle(idx, id, idxpos, idxind, model, shapeId, idxInShape, this, bInfo);\r\n const target = storage ? storage : this.particles;\r\n target.push(sp);\r\n return sp;\r\n }\r\n\r\n /**\r\n * Adds some particles to the SPS from the model shape. Returns the shape id.\r\n * Please read the doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/immutable_sps\r\n * @param mesh is any Mesh object that will be used as a model for the solid particles.\r\n * @param nb (positive integer) the number of particles to be created from this model\r\n * @param options {positionFunction} is an optional javascript function to called for each particle on SPS creation.\r\n * {vertexFunction} is an optional javascript function to called for each vertex of each particle on SPS creation\r\n * {storage} (optional existing array) is an array where the particles will be stored for a further use instead of being inserted in the SPS.\r\n * @param options.positionFunction\r\n * @param options.vertexFunction\r\n * @param options.storage\r\n * @returns the number of shapes in the system\r\n */\r\n public addShape(mesh: Mesh, nb: number, options?: { positionFunction?: any; vertexFunction?: any; storage?: [] }): number {\r\n const meshPos = mesh.getVerticesData(VertexBuffer.PositionKind);\r\n const meshInd = mesh.getIndices();\r\n const meshUV = mesh.getVerticesData(VertexBuffer.UVKind);\r\n const meshCol = mesh.getVerticesData(VertexBuffer.ColorKind);\r\n const meshNor = mesh.getVerticesData(VertexBuffer.NormalKind);\r\n this.recomputeNormals = meshNor ? false : true;\r\n const indices = Array.from(meshInd);\r\n const shapeNormals = Array.from(meshNor);\r\n const shapeColors = meshCol ? Array.from(meshCol) : [];\r\n const storage = options && options.storage ? options.storage : null;\r\n let bbInfo: Nullable = null;\r\n if (this._particlesIntersect) {\r\n bbInfo = mesh.getBoundingInfo();\r\n }\r\n\r\n const shape = this._posToShape(meshPos);\r\n const shapeUV = this._uvsToShapeUV(meshUV);\r\n\r\n const posfunc = options ? options.positionFunction : null;\r\n const vtxfunc = options ? options.vertexFunction : null;\r\n let material = null;\r\n if (this._useModelMaterial) {\r\n material = mesh.material ? mesh.material : this._setDefaultMaterial();\r\n }\r\n const modelShape = new ModelShape(this._shapeCounter, shape, indices, shapeNormals, shapeColors, shapeUV, posfunc, vtxfunc, material);\r\n\r\n // particles\r\n for (let i = 0; i < nb; i++) {\r\n this._insertNewParticle(this.nbParticles, i, modelShape, shape, meshInd, meshUV, meshCol, meshNor, bbInfo, storage, options);\r\n }\r\n this._shapeCounter++;\r\n this._isNotBuilt = true; // buildMesh() call is now expected for setParticles() to work\r\n return this._shapeCounter - 1;\r\n }\r\n\r\n /**\r\n * Rebuilds a particle back to its just built status : if needed, recomputes the custom positions and vertices\r\n * @internal\r\n */\r\n protected _rebuildParticle(particle: SolidParticle, reset: boolean = false): void {\r\n this._resetCopy();\r\n const copy = this._copy;\r\n if (particle._model._positionFunction) {\r\n // recall to stored custom positionFunction\r\n particle._model._positionFunction(copy, particle.idx, particle.idxInShape);\r\n }\r\n\r\n const rotMatrix = TmpVectors.Matrix[0];\r\n const tmpVertex = TmpVectors.Vector3[0];\r\n const tmpRotated = TmpVectors.Vector3[1];\r\n const pivotBackTranslation = TmpVectors.Vector3[2];\r\n const scaledPivot = TmpVectors.Vector3[3];\r\n\r\n copy.getRotationMatrix(rotMatrix);\r\n\r\n particle.pivot.multiplyToRef(particle.scaling, scaledPivot);\r\n\r\n if (copy.translateFromPivot) {\r\n pivotBackTranslation.copyFromFloats(0.0, 0.0, 0.0);\r\n } else {\r\n pivotBackTranslation.copyFrom(scaledPivot);\r\n }\r\n\r\n const shape = particle._model._shape;\r\n\r\n for (let pt = 0; pt < shape.length; pt++) {\r\n tmpVertex.copyFrom(shape[pt]);\r\n if (particle._model._vertexFunction) {\r\n particle._model._vertexFunction(copy, tmpVertex, pt); // recall to stored vertexFunction\r\n }\r\n\r\n tmpVertex.multiplyInPlace(copy.scaling).subtractInPlace(scaledPivot);\r\n Vector3.TransformCoordinatesToRef(tmpVertex, rotMatrix, tmpRotated);\r\n tmpRotated\r\n .addInPlace(pivotBackTranslation)\r\n .addInPlace(copy.position)\r\n .toArray(this._positions32, particle._pos + pt * 3);\r\n }\r\n if (reset) {\r\n particle.position.setAll(0.0);\r\n particle.rotation.setAll(0.0);\r\n particle.rotationQuaternion = null;\r\n particle.scaling.setAll(1.0);\r\n particle.uvs.setAll(0.0);\r\n particle.pivot.setAll(0.0);\r\n particle.translateFromPivot = false;\r\n particle.parentId = null;\r\n }\r\n }\r\n\r\n /**\r\n * Rebuilds the whole mesh and updates the VBO : custom positions and vertices are recomputed if needed.\r\n * @param reset boolean, default false : if the particles must be reset at position and rotation zero, scaling 1, color white, initial UVs and not parented.\r\n * @returns the SPS.\r\n */\r\n public rebuildMesh(reset: boolean = false): SolidParticleSystem {\r\n for (let p = 0; p < this.particles.length; p++) {\r\n this._rebuildParticle(this.particles[p], reset);\r\n }\r\n this.mesh.updateVerticesData(VertexBuffer.PositionKind, this._positions32, false, false);\r\n return this;\r\n }\r\n\r\n /** Removes the particles from the start-th to the end-th included from an expandable SPS (required).\r\n * Returns an array with the removed particles.\r\n * If the number of particles to remove is lower than zero or greater than the global remaining particle number, then an empty array is returned.\r\n * The SPS can't be empty so at least one particle needs to remain in place.\r\n * Under the hood, the VertexData array, so the VBO buffer, is recreated each call.\r\n * @param start index of the first particle to remove\r\n * @param end index of the last particle to remove (included)\r\n * @returns an array populated with the removed particles\r\n */\r\n public removeParticles(start: number, end: number): SolidParticle[] {\r\n const nb = end - start + 1;\r\n if (!this._expandable || nb <= 0 || nb >= this.nbParticles || !this._updatable) {\r\n return [];\r\n }\r\n const particles = this.particles;\r\n const currentNb = this.nbParticles;\r\n if (end < currentNb - 1) {\r\n // update the particle indexes in the positions array in case they're remaining particles after the last removed\r\n const firstRemaining = end + 1;\r\n const shiftPos = particles[firstRemaining]._pos - particles[start]._pos;\r\n const shifInd = particles[firstRemaining]._ind - particles[start]._ind;\r\n for (let i = firstRemaining; i < currentNb; i++) {\r\n const part = particles[i];\r\n part._pos -= shiftPos;\r\n part._ind -= shifInd;\r\n }\r\n }\r\n const removed = particles.splice(start, nb);\r\n this._positions.length = 0;\r\n this._indices.length = 0;\r\n this._colors.length = 0;\r\n this._uvs.length = 0;\r\n this._normals.length = 0;\r\n this._index = 0;\r\n this._idxOfId.length = 0;\r\n if (this._depthSort || this._multimaterialEnabled) {\r\n this.depthSortedParticles = [];\r\n }\r\n let ind = 0;\r\n const particlesLength = particles.length;\r\n for (let p = 0; p < particlesLength; p++) {\r\n const particle = particles[p];\r\n const model = particle._model;\r\n const shape = model._shape;\r\n const modelIndices = model._indices;\r\n const modelNormals = model._normals;\r\n const modelColors = model._shapeColors;\r\n const modelUVs = model._shapeUV;\r\n particle.idx = p;\r\n this._idxOfId[particle.id] = p;\r\n this._meshBuilder(\r\n this._index,\r\n ind,\r\n shape,\r\n this._positions,\r\n modelIndices,\r\n this._indices,\r\n modelUVs,\r\n this._uvs,\r\n modelColors,\r\n this._colors,\r\n modelNormals,\r\n this._normals,\r\n particle.idx,\r\n particle.idxInShape,\r\n null,\r\n model\r\n );\r\n this._index += shape.length;\r\n ind += modelIndices.length;\r\n }\r\n this.nbParticles -= nb;\r\n this._isNotBuilt = true; // buildMesh() call is now expected for setParticles() to work\r\n return removed;\r\n }\r\n\r\n /**\r\n * Inserts some pre-created particles in the solid particle system so that they can be managed by setParticles().\r\n * @param solidParticleArray an array populated with Solid Particles objects\r\n * @returns the SPS\r\n */\r\n public insertParticlesFromArray(solidParticleArray: SolidParticle[]): SolidParticleSystem {\r\n if (!this._expandable) {\r\n return this;\r\n }\r\n let idxInShape = 0;\r\n let currentShapeId = solidParticleArray[0].shapeId;\r\n const nb = solidParticleArray.length;\r\n for (let i = 0; i < nb; i++) {\r\n const sp = solidParticleArray[i];\r\n const model = sp._model;\r\n const shape = model._shape;\r\n const meshInd = model._indices;\r\n const meshUV = model._shapeUV;\r\n const meshCol = model._shapeColors;\r\n const meshNor = model._normals;\r\n const noNor = meshNor ? false : true;\r\n this.recomputeNormals = noNor || this.recomputeNormals;\r\n const bbInfo = sp.getBoundingInfo();\r\n const newPart = this._insertNewParticle(this.nbParticles, idxInShape, model, shape, meshInd, meshUV, meshCol, meshNor, bbInfo, null, null);\r\n sp.copyToRef(newPart!);\r\n idxInShape++;\r\n if (currentShapeId != sp.shapeId) {\r\n currentShapeId = sp.shapeId;\r\n idxInShape = 0;\r\n }\r\n }\r\n this._isNotBuilt = true; // buildMesh() call is now expected for setParticles() to work\r\n return this;\r\n }\r\n\r\n /**\r\n * Creates a new particle and modifies the SPS mesh geometry :\r\n * - calls _meshBuilder() to increase the SPS mesh geometry step by step\r\n * - calls _addParticle() to populate the particle array\r\n * factorized code from addShape() and insertParticlesFromArray()\r\n * @param idx particle index in the particles array\r\n * @param i particle index in its shape\r\n * @param modelShape particle ModelShape object\r\n * @param shape shape vertex array\r\n * @param meshInd shape indices array\r\n * @param meshUV shape uv array\r\n * @param meshCol shape color array\r\n * @param meshNor shape normals array\r\n * @param bbInfo shape bounding info\r\n * @param storage target particle storage\r\n * @param options\r\n * @options addShape() passed options\r\n * @internal\r\n */\r\n protected _insertNewParticle(\r\n idx: number,\r\n i: number,\r\n modelShape: ModelShape,\r\n shape: Vector3[],\r\n meshInd: IndicesArray,\r\n meshUV: number[] | Float32Array,\r\n meshCol: number[] | Float32Array,\r\n meshNor: number[] | Float32Array,\r\n bbInfo: Nullable,\r\n storage: Nullable<[]>,\r\n options: any\r\n ): Nullable {\r\n const currentPos = this._positions.length;\r\n const currentInd = this._indices.length;\r\n const currentCopy = this._meshBuilder(\r\n this._index,\r\n currentInd,\r\n shape,\r\n this._positions,\r\n meshInd,\r\n this._indices,\r\n meshUV,\r\n this._uvs,\r\n meshCol,\r\n this._colors,\r\n meshNor,\r\n this._normals,\r\n idx,\r\n i,\r\n options,\r\n modelShape\r\n );\r\n let sp: Nullable = null;\r\n if (this._updatable) {\r\n sp = this._addParticle(this.nbParticles, this._lastParticleId, currentPos, currentInd, modelShape, this._shapeCounter, i, bbInfo, storage);\r\n sp.position.copyFrom(currentCopy.position);\r\n sp.rotation.copyFrom(currentCopy.rotation);\r\n if (currentCopy.rotationQuaternion) {\r\n if (sp.rotationQuaternion) {\r\n sp.rotationQuaternion.copyFrom(currentCopy.rotationQuaternion);\r\n } else {\r\n sp.rotationQuaternion = currentCopy.rotationQuaternion.clone();\r\n }\r\n }\r\n if (currentCopy.color) {\r\n if (sp.color) {\r\n sp.color.copyFrom(currentCopy.color);\r\n } else {\r\n sp.color = currentCopy.color.clone();\r\n }\r\n }\r\n sp.scaling.copyFrom(currentCopy.scaling);\r\n sp.uvs.copyFrom(currentCopy.uvs);\r\n if (currentCopy.materialIndex !== null) {\r\n sp.materialIndex = currentCopy.materialIndex;\r\n }\r\n if (this.expandable) {\r\n this._idxOfId[sp.id] = sp.idx;\r\n }\r\n }\r\n if (!storage) {\r\n this._index += shape.length;\r\n this.nbParticles++;\r\n this._lastParticleId++;\r\n }\r\n return sp;\r\n }\r\n\r\n /**\r\n * Sets all the particles : this method actually really updates the mesh according to the particle positions, rotations, colors, textures, etc.\r\n * This method calls `updateParticle()` for each particle of the SPS.\r\n * For an animated SPS, it is usually called within the render loop.\r\n * This methods does nothing if called on a non updatable or not yet built SPS. Example : buildMesh() not called after having added or removed particles from an expandable SPS.\r\n * @param start The particle index in the particle array where to start to compute the particle property values _(default 0)_\r\n * @param end The particle index in the particle array where to stop to compute the particle property values _(default nbParticle - 1)_\r\n * @param update If the mesh must be finally updated on this call after all the particle computations _(default true)_\r\n * @returns the SPS.\r\n */\r\n public setParticles(start: number = 0, end: number = this.nbParticles - 1, update: boolean = true): SolidParticleSystem {\r\n if (!this._updatable || this._isNotBuilt) {\r\n return this;\r\n }\r\n\r\n // custom beforeUpdate\r\n this.beforeUpdateParticles(start, end, update);\r\n\r\n const rotMatrix = TmpVectors.Matrix[0];\r\n const invertedMatrix = TmpVectors.Matrix[1];\r\n const mesh = this.mesh;\r\n const colors32 = this._colors32;\r\n const positions32 = this._positions32;\r\n const normals32 = this._normals32;\r\n const uvs32 = this._uvs32;\r\n const indices32 = this._indices32;\r\n const indices = this._indices;\r\n const fixedNormal32 = this._fixedNormal32;\r\n const depthSortParticles = this._depthSort && this._depthSortParticles;\r\n\r\n const tempVectors = TmpVectors.Vector3;\r\n const camAxisX = tempVectors[5].copyFromFloats(1.0, 0.0, 0.0);\r\n const camAxisY = tempVectors[6].copyFromFloats(0.0, 1.0, 0.0);\r\n const camAxisZ = tempVectors[7].copyFromFloats(0.0, 0.0, 1.0);\r\n const minimum = tempVectors[8].setAll(Number.MAX_VALUE);\r\n const maximum = tempVectors[9].setAll(-Number.MAX_VALUE);\r\n const camInvertedPosition = tempVectors[10].setAll(0);\r\n\r\n const tmpVertex = this._tmpVertex;\r\n const tmpVector = tmpVertex.position;\r\n const tmpColor = tmpVertex.color;\r\n const tmpUV = tmpVertex.uv;\r\n\r\n // cases when the World Matrix is to be computed first\r\n if (this.billboard || this._depthSort) {\r\n this.mesh.computeWorldMatrix(true);\r\n this.mesh._worldMatrix.invertToRef(invertedMatrix);\r\n }\r\n // if the particles will always face the camera\r\n if (this.billboard) {\r\n // compute the camera position and un-rotate it by the current mesh rotation\r\n const tmpVector0 = tempVectors[0];\r\n this._camera.getDirectionToRef(Axis.Z, tmpVector0);\r\n Vector3.TransformNormalToRef(tmpVector0, invertedMatrix, camAxisZ);\r\n camAxisZ.normalize();\r\n // same for camera up vector extracted from the cam view matrix\r\n const view = this._camera.getViewMatrix(true);\r\n Vector3.TransformNormalFromFloatsToRef(view.m[1], view.m[5], view.m[9], invertedMatrix, camAxisY);\r\n Vector3.CrossToRef(camAxisY, camAxisZ, camAxisX);\r\n camAxisY.normalize();\r\n camAxisX.normalize();\r\n }\r\n\r\n // if depthSort, compute the camera global position in the mesh local system\r\n if (this._depthSort) {\r\n Vector3.TransformCoordinatesToRef(this._camera.globalPosition, invertedMatrix, camInvertedPosition); // then un-rotate the camera\r\n }\r\n\r\n Matrix.IdentityToRef(rotMatrix);\r\n let idx = 0; // current position index in the global array positions32\r\n let index = 0; // position start index in the global array positions32 of the current particle\r\n let colidx = 0; // current color index in the global array colors32\r\n let colorIndex = 0; // color start index in the global array colors32 of the current particle\r\n let uvidx = 0; // current uv index in the global array uvs32\r\n let uvIndex = 0; // uv start index in the global array uvs32 of the current particle\r\n let pt = 0; // current index in the particle model shape\r\n\r\n if (this.mesh.isFacetDataEnabled) {\r\n this._computeBoundingBox = true;\r\n }\r\n\r\n end = end >= this.nbParticles ? this.nbParticles - 1 : end;\r\n if (this._computeBoundingBox) {\r\n if (start != 0 || end != this.nbParticles - 1) {\r\n // only some particles are updated, then use the current existing BBox basis. Note : it can only increase.\r\n const boundingInfo = this.mesh.getBoundingInfo();\r\n if (boundingInfo) {\r\n minimum.copyFrom(boundingInfo.minimum);\r\n maximum.copyFrom(boundingInfo.maximum);\r\n }\r\n }\r\n }\r\n\r\n // particle loop\r\n index = this.particles[start]._pos;\r\n const vpos = (index / 3) | 0;\r\n colorIndex = vpos * 4;\r\n uvIndex = vpos * 2;\r\n\r\n for (let p = start; p <= end; p++) {\r\n const particle = this.particles[p];\r\n\r\n // call to custom user function to update the particle properties\r\n this.updateParticle(particle);\r\n\r\n const shape = particle._model._shape;\r\n const shapeUV = particle._model._shapeUV;\r\n const particleRotationMatrix = particle._rotationMatrix;\r\n const particlePosition = particle.position;\r\n const particleRotation = particle.rotation;\r\n const particleScaling = particle.scaling;\r\n const particleGlobalPosition = particle._globalPosition;\r\n\r\n // camera-particle distance for depth sorting\r\n if (depthSortParticles) {\r\n const dsp = this.depthSortedParticles[p];\r\n dsp.idx = particle.idx;\r\n dsp.ind = particle._ind;\r\n dsp.indicesLength = particle._model._indicesLength;\r\n dsp.sqDistance = Vector3.DistanceSquared(particle.position, camInvertedPosition);\r\n }\r\n\r\n // skip the computations for inactive or already invisible particles\r\n if (!particle.alive || (particle._stillInvisible && !particle.isVisible && !this._recomputeInvisibles)) {\r\n // increment indexes for the next particle\r\n pt = shape.length;\r\n index += pt * 3;\r\n colorIndex += pt * 4;\r\n uvIndex += pt * 2;\r\n continue;\r\n }\r\n\r\n if (particle.isVisible) {\r\n particle._stillInvisible = false; // un-mark permanent invisibility\r\n\r\n const scaledPivot = tempVectors[12];\r\n particle.pivot.multiplyToRef(particleScaling, scaledPivot);\r\n\r\n // particle rotation matrix\r\n if (this.billboard) {\r\n particleRotation.x = 0.0;\r\n particleRotation.y = 0.0;\r\n }\r\n if (this._computeParticleRotation || this.billboard) {\r\n particle.getRotationMatrix(rotMatrix);\r\n }\r\n\r\n const particleHasParent = particle.parentId !== null;\r\n if (particleHasParent) {\r\n const parent = this.getParticleById(particle.parentId!);\r\n if (parent) {\r\n const parentRotationMatrix = parent._rotationMatrix;\r\n const parentGlobalPosition = parent._globalPosition;\r\n\r\n const rotatedY = particlePosition.x * parentRotationMatrix[1] + particlePosition.y * parentRotationMatrix[4] + particlePosition.z * parentRotationMatrix[7];\r\n const rotatedX = particlePosition.x * parentRotationMatrix[0] + particlePosition.y * parentRotationMatrix[3] + particlePosition.z * parentRotationMatrix[6];\r\n const rotatedZ = particlePosition.x * parentRotationMatrix[2] + particlePosition.y * parentRotationMatrix[5] + particlePosition.z * parentRotationMatrix[8];\r\n\r\n particleGlobalPosition.x = parentGlobalPosition.x + rotatedX;\r\n particleGlobalPosition.y = parentGlobalPosition.y + rotatedY;\r\n particleGlobalPosition.z = parentGlobalPosition.z + rotatedZ;\r\n\r\n if (this._computeParticleRotation || this.billboard) {\r\n const rotMatrixValues = rotMatrix.m;\r\n particleRotationMatrix[0] =\r\n rotMatrixValues[0] * parentRotationMatrix[0] + rotMatrixValues[1] * parentRotationMatrix[3] + rotMatrixValues[2] * parentRotationMatrix[6];\r\n particleRotationMatrix[1] =\r\n rotMatrixValues[0] * parentRotationMatrix[1] + rotMatrixValues[1] * parentRotationMatrix[4] + rotMatrixValues[2] * parentRotationMatrix[7];\r\n particleRotationMatrix[2] =\r\n rotMatrixValues[0] * parentRotationMatrix[2] + rotMatrixValues[1] * parentRotationMatrix[5] + rotMatrixValues[2] * parentRotationMatrix[8];\r\n particleRotationMatrix[3] =\r\n rotMatrixValues[4] * parentRotationMatrix[0] + rotMatrixValues[5] * parentRotationMatrix[3] + rotMatrixValues[6] * parentRotationMatrix[6];\r\n particleRotationMatrix[4] =\r\n rotMatrixValues[4] * parentRotationMatrix[1] + rotMatrixValues[5] * parentRotationMatrix[4] + rotMatrixValues[6] * parentRotationMatrix[7];\r\n particleRotationMatrix[5] =\r\n rotMatrixValues[4] * parentRotationMatrix[2] + rotMatrixValues[5] * parentRotationMatrix[5] + rotMatrixValues[6] * parentRotationMatrix[8];\r\n particleRotationMatrix[6] =\r\n rotMatrixValues[8] * parentRotationMatrix[0] + rotMatrixValues[9] * parentRotationMatrix[3] + rotMatrixValues[10] * parentRotationMatrix[6];\r\n particleRotationMatrix[7] =\r\n rotMatrixValues[8] * parentRotationMatrix[1] + rotMatrixValues[9] * parentRotationMatrix[4] + rotMatrixValues[10] * parentRotationMatrix[7];\r\n particleRotationMatrix[8] =\r\n rotMatrixValues[8] * parentRotationMatrix[2] + rotMatrixValues[9] * parentRotationMatrix[5] + rotMatrixValues[10] * parentRotationMatrix[8];\r\n }\r\n } else {\r\n // in case the parent were removed at some moment\r\n particle.parentId = null;\r\n }\r\n } else {\r\n particleGlobalPosition.x = particlePosition.x;\r\n particleGlobalPosition.y = particlePosition.y;\r\n particleGlobalPosition.z = particlePosition.z;\r\n\r\n if (this._computeParticleRotation || this.billboard) {\r\n const rotMatrixValues = rotMatrix.m;\r\n particleRotationMatrix[0] = rotMatrixValues[0];\r\n particleRotationMatrix[1] = rotMatrixValues[1];\r\n particleRotationMatrix[2] = rotMatrixValues[2];\r\n particleRotationMatrix[3] = rotMatrixValues[4];\r\n particleRotationMatrix[4] = rotMatrixValues[5];\r\n particleRotationMatrix[5] = rotMatrixValues[6];\r\n particleRotationMatrix[6] = rotMatrixValues[8];\r\n particleRotationMatrix[7] = rotMatrixValues[9];\r\n particleRotationMatrix[8] = rotMatrixValues[10];\r\n }\r\n }\r\n\r\n const pivotBackTranslation = tempVectors[11];\r\n if (particle.translateFromPivot) {\r\n pivotBackTranslation.setAll(0.0);\r\n } else {\r\n pivotBackTranslation.copyFrom(scaledPivot);\r\n }\r\n\r\n // particle vertex loop\r\n for (pt = 0; pt < shape.length; pt++) {\r\n idx = index + pt * 3;\r\n colidx = colorIndex + pt * 4;\r\n uvidx = uvIndex + pt * 2;\r\n const iu = 2 * pt;\r\n const iv = iu + 1;\r\n\r\n tmpVector.copyFrom(shape[pt]);\r\n if (this._computeParticleColor && particle.color) {\r\n tmpColor.copyFrom(particle.color);\r\n }\r\n if (this._computeParticleTexture) {\r\n tmpUV.copyFromFloats(shapeUV[iu], shapeUV[iv]);\r\n }\r\n if (this._computeParticleVertex) {\r\n this.updateParticleVertex(particle, tmpVertex, pt);\r\n }\r\n\r\n // positions\r\n const vertexX = tmpVector.x * particleScaling.x - scaledPivot.x;\r\n const vertexY = tmpVector.y * particleScaling.y - scaledPivot.y;\r\n const vertexZ = tmpVector.z * particleScaling.z - scaledPivot.z;\r\n\r\n let rotatedX = vertexX * particleRotationMatrix[0] + vertexY * particleRotationMatrix[3] + vertexZ * particleRotationMatrix[6];\r\n let rotatedY = vertexX * particleRotationMatrix[1] + vertexY * particleRotationMatrix[4] + vertexZ * particleRotationMatrix[7];\r\n let rotatedZ = vertexX * particleRotationMatrix[2] + vertexY * particleRotationMatrix[5] + vertexZ * particleRotationMatrix[8];\r\n\r\n rotatedX += pivotBackTranslation.x;\r\n rotatedY += pivotBackTranslation.y;\r\n rotatedZ += pivotBackTranslation.z;\r\n\r\n const px = (positions32[idx] = particleGlobalPosition.x + camAxisX.x * rotatedX + camAxisY.x * rotatedY + camAxisZ.x * rotatedZ);\r\n const py = (positions32[idx + 1] = particleGlobalPosition.y + camAxisX.y * rotatedX + camAxisY.y * rotatedY + camAxisZ.y * rotatedZ);\r\n const pz = (positions32[idx + 2] = particleGlobalPosition.z + camAxisX.z * rotatedX + camAxisY.z * rotatedY + camAxisZ.z * rotatedZ);\r\n\r\n if (this._computeBoundingBox) {\r\n minimum.minimizeInPlaceFromFloats(px, py, pz);\r\n maximum.maximizeInPlaceFromFloats(px, py, pz);\r\n }\r\n\r\n // normals : if the particles can't be morphed then just rotate the normals, what is much more faster than ComputeNormals()\r\n if (!this._computeParticleVertex) {\r\n const normalx = fixedNormal32[idx];\r\n const normaly = fixedNormal32[idx + 1];\r\n const normalz = fixedNormal32[idx + 2];\r\n\r\n const rotatedx = normalx * particleRotationMatrix[0] + normaly * particleRotationMatrix[3] + normalz * particleRotationMatrix[6];\r\n const rotatedy = normalx * particleRotationMatrix[1] + normaly * particleRotationMatrix[4] + normalz * particleRotationMatrix[7];\r\n const rotatedz = normalx * particleRotationMatrix[2] + normaly * particleRotationMatrix[5] + normalz * particleRotationMatrix[8];\r\n\r\n normals32[idx] = camAxisX.x * rotatedx + camAxisY.x * rotatedy + camAxisZ.x * rotatedz;\r\n normals32[idx + 1] = camAxisX.y * rotatedx + camAxisY.y * rotatedy + camAxisZ.y * rotatedz;\r\n normals32[idx + 2] = camAxisX.z * rotatedx + camAxisY.z * rotatedy + camAxisZ.z * rotatedz;\r\n }\r\n\r\n if (this._computeParticleColor && particle.color) {\r\n const colors32 = this._colors32;\r\n colors32[colidx] = tmpColor.r;\r\n colors32[colidx + 1] = tmpColor.g;\r\n colors32[colidx + 2] = tmpColor.b;\r\n colors32[colidx + 3] = tmpColor.a;\r\n }\r\n\r\n if (this._computeParticleTexture) {\r\n const uvs = particle.uvs;\r\n uvs32[uvidx] = tmpUV.x * (uvs.z - uvs.x) + uvs.x;\r\n uvs32[uvidx + 1] = tmpUV.y * (uvs.w - uvs.y) + uvs.y;\r\n }\r\n }\r\n }\r\n // particle just set invisible : scaled to zero and positioned at the origin\r\n else {\r\n particle._stillInvisible = true; // mark the particle as invisible\r\n for (pt = 0; pt < shape.length; pt++) {\r\n idx = index + pt * 3;\r\n colidx = colorIndex + pt * 4;\r\n uvidx = uvIndex + pt * 2;\r\n\r\n positions32[idx] = positions32[idx + 1] = positions32[idx + 2] = 0;\r\n normals32[idx] = normals32[idx + 1] = normals32[idx + 2] = 0;\r\n if (this._computeParticleColor && particle.color) {\r\n const color = particle.color;\r\n colors32[colidx] = color.r;\r\n colors32[colidx + 1] = color.g;\r\n colors32[colidx + 2] = color.b;\r\n colors32[colidx + 3] = color.a;\r\n }\r\n if (this._computeParticleTexture) {\r\n const uvs = particle.uvs;\r\n uvs32[uvidx] = shapeUV[pt * 2] * (uvs.z - uvs.x) + uvs.x;\r\n uvs32[uvidx + 1] = shapeUV[pt * 2 + 1] * (uvs.w - uvs.y) + uvs.y;\r\n }\r\n }\r\n }\r\n\r\n // if the particle intersections must be computed : update the bbInfo\r\n if (this._particlesIntersect) {\r\n const bInfo = particle.getBoundingInfo();\r\n const bBox = bInfo.boundingBox;\r\n const bSphere = bInfo.boundingSphere;\r\n const modelBoundingInfo = particle._modelBoundingInfo;\r\n if (!this._bSphereOnly) {\r\n // place, scale and rotate the particle bbox within the SPS local system, then update it\r\n const modelBoundingInfoVectors = modelBoundingInfo.boundingBox.vectors;\r\n\r\n const tempMin = tempVectors[1];\r\n const tempMax = tempVectors[2];\r\n tempMin.setAll(Number.MAX_VALUE);\r\n tempMax.setAll(-Number.MAX_VALUE);\r\n for (let b = 0; b < 8; b++) {\r\n const scaledX = modelBoundingInfoVectors[b].x * particleScaling.x;\r\n const scaledY = modelBoundingInfoVectors[b].y * particleScaling.y;\r\n const scaledZ = modelBoundingInfoVectors[b].z * particleScaling.z;\r\n const rotatedX = scaledX * particleRotationMatrix[0] + scaledY * particleRotationMatrix[3] + scaledZ * particleRotationMatrix[6];\r\n const rotatedY = scaledX * particleRotationMatrix[1] + scaledY * particleRotationMatrix[4] + scaledZ * particleRotationMatrix[7];\r\n const rotatedZ = scaledX * particleRotationMatrix[2] + scaledY * particleRotationMatrix[5] + scaledZ * particleRotationMatrix[8];\r\n const x = particlePosition.x + camAxisX.x * rotatedX + camAxisY.x * rotatedY + camAxisZ.x * rotatedZ;\r\n const y = particlePosition.y + camAxisX.y * rotatedX + camAxisY.y * rotatedY + camAxisZ.y * rotatedZ;\r\n const z = particlePosition.z + camAxisX.z * rotatedX + camAxisY.z * rotatedY + camAxisZ.z * rotatedZ;\r\n tempMin.minimizeInPlaceFromFloats(x, y, z);\r\n tempMax.maximizeInPlaceFromFloats(x, y, z);\r\n }\r\n\r\n bBox.reConstruct(tempMin, tempMax, mesh._worldMatrix);\r\n }\r\n\r\n // place and scale the particle bouding sphere in the SPS local system, then update it\r\n const minBbox = modelBoundingInfo.minimum.multiplyToRef(particleScaling, tempVectors[1]);\r\n const maxBbox = modelBoundingInfo.maximum.multiplyToRef(particleScaling, tempVectors[2]);\r\n\r\n const bSphereCenter = maxBbox.addToRef(minBbox, tempVectors[3]).scaleInPlace(0.5).addInPlace(particleGlobalPosition);\r\n const halfDiag = maxBbox.subtractToRef(minBbox, tempVectors[4]).scaleInPlace(0.5 * this._bSphereRadiusFactor);\r\n const bSphereMinBbox = bSphereCenter.subtractToRef(halfDiag, tempVectors[1]);\r\n const bSphereMaxBbox = bSphereCenter.addToRef(halfDiag, tempVectors[2]);\r\n bSphere.reConstruct(bSphereMinBbox, bSphereMaxBbox, mesh._worldMatrix);\r\n }\r\n\r\n // increment indexes for the next particle\r\n index = idx + 3;\r\n colorIndex = colidx + 4;\r\n uvIndex = uvidx + 2;\r\n }\r\n\r\n // if the VBO must be updated\r\n if (update) {\r\n if (this._computeParticleColor) {\r\n const vb = mesh.getVertexBuffer(VertexBuffer.ColorKind);\r\n if (vb && !mesh.isPickable) {\r\n vb.updateDirectly(colors32, 0);\r\n } else {\r\n mesh.updateVerticesData(VertexBuffer.ColorKind, colors32, false, false);\r\n }\r\n }\r\n if (this._computeParticleTexture) {\r\n const vb = mesh.getVertexBuffer(VertexBuffer.UVKind);\r\n if (vb && !mesh.isPickable) {\r\n vb.updateDirectly(uvs32, 0);\r\n } else {\r\n mesh.updateVerticesData(VertexBuffer.UVKind, uvs32, false, false);\r\n }\r\n }\r\n const vbp = mesh.getVertexBuffer(VertexBuffer.PositionKind);\r\n if (vbp && !mesh.isPickable) {\r\n vbp.updateDirectly(positions32, 0);\r\n } else {\r\n mesh.updateVerticesData(VertexBuffer.PositionKind, positions32, false, false);\r\n }\r\n if (!mesh.areNormalsFrozen || mesh.isFacetDataEnabled) {\r\n if (this._computeParticleVertex || mesh.isFacetDataEnabled) {\r\n // recompute the normals only if the particles can be morphed, update then also the normal reference array _fixedNormal32[]\r\n const params = mesh.isFacetDataEnabled ? mesh.getFacetDataParameters() : null;\r\n VertexData.ComputeNormals(positions32, indices32, normals32, params);\r\n for (let i = 0; i < normals32.length; i++) {\r\n fixedNormal32[i] = normals32[i];\r\n }\r\n }\r\n if (!mesh.areNormalsFrozen) {\r\n const vb = mesh.getVertexBuffer(VertexBuffer.NormalKind);\r\n if (vb && !mesh.isPickable) {\r\n vb.updateDirectly(normals32, 0);\r\n } else {\r\n mesh.updateVerticesData(VertexBuffer.NormalKind, normals32, false, false);\r\n }\r\n }\r\n }\r\n if (depthSortParticles) {\r\n const depthSortedParticles = this.depthSortedParticles;\r\n depthSortedParticles.sort(this._depthSortFunction);\r\n const dspl = depthSortedParticles.length;\r\n let sid = 0;\r\n let faceId = 0;\r\n for (let sorted = 0; sorted < dspl; sorted++) {\r\n const sortedParticle = depthSortedParticles[sorted];\r\n const lind = sortedParticle.indicesLength;\r\n const sind = sortedParticle.ind;\r\n for (let i = 0; i < lind; i++) {\r\n indices32[sid] = indices[sind + i];\r\n sid++;\r\n if (this._pickable) {\r\n const f = i % 3;\r\n if (f == 0) {\r\n const pickedData = this.pickedParticles[faceId];\r\n pickedData.idx = sortedParticle.idx;\r\n pickedData.faceId = faceId;\r\n faceId++;\r\n }\r\n }\r\n }\r\n }\r\n }\r\n if (this._autoFixFaceOrientation) {\r\n let particleInd = 0;\r\n\r\n for (let particleIdx = 0; particleIdx < this.particles.length; particleIdx++) {\r\n const particle = depthSortParticles ? this.particles[this.depthSortedParticles[particleIdx].idx] : this.particles[particleIdx];\r\n const flipFaces = particle.scale.x * particle.scale.y * particle.scale.z < 0;\r\n\r\n if (flipFaces) {\r\n for (let faceInd = 0; faceInd < particle._model._indicesLength; faceInd += 3) {\r\n const tmp = indices[particle._ind + faceInd];\r\n indices32[particleInd + faceInd] = indices[particle._ind + faceInd + 1];\r\n indices32[particleInd + faceInd + 1] = tmp;\r\n }\r\n }\r\n\r\n particleInd += particle._model._indicesLength;\r\n }\r\n }\r\n\r\n if (depthSortParticles || this._autoFixFaceOrientation) {\r\n mesh.updateIndices(indices32);\r\n }\r\n }\r\n if (this._computeBoundingBox) {\r\n if (mesh.hasBoundingInfo) {\r\n mesh.getBoundingInfo().reConstruct(minimum, maximum, mesh._worldMatrix);\r\n } else {\r\n mesh.buildBoundingInfo(minimum, maximum, mesh._worldMatrix);\r\n }\r\n }\r\n if (this._autoUpdateSubMeshes) {\r\n this.computeSubMeshes();\r\n }\r\n this._recomputeInvisibles = false;\r\n this.afterUpdateParticles(start, end, update);\r\n return this;\r\n }\r\n\r\n /**\r\n * Disposes the SPS.\r\n */\r\n public dispose(): void {\r\n this.mesh.dispose();\r\n this.vars = null;\r\n // drop references to internal big arrays for the GC\r\n (this._positions) = null;\r\n (this._indices) = null;\r\n (this._normals) = null;\r\n (this._uvs) = null;\r\n (this._colors) = null;\r\n (this._indices32) = null;\r\n (this._positions32) = null;\r\n (this._normals32) = null;\r\n (this._fixedNormal32) = null;\r\n (this._uvs32) = null;\r\n (this._colors32) = null;\r\n (this.pickedParticles) = null;\r\n (this.pickedBySubMesh) = null;\r\n (this._materials) = null;\r\n (this._materialIndexes) = null;\r\n (this._indicesByMaterial) = null;\r\n (this._idxOfId) = null;\r\n }\r\n /** Returns an object {idx: number faceId: number} for the picked particle from the passed pickingInfo object.\r\n * idx is the particle index in the SPS\r\n * faceId is the picked face index counted within this particle.\r\n * Returns null if the pickInfo can't identify a picked particle.\r\n * @param pickingInfo (PickingInfo object)\r\n * @returns {idx: number, faceId: number} or null\r\n */\r\n public pickedParticle(pickingInfo: PickingInfo): Nullable<{ idx: number; faceId: number }> {\r\n if (pickingInfo.hit) {\r\n const subMesh = pickingInfo.subMeshId;\r\n const faceId = pickingInfo.faceId - this.mesh.subMeshes[subMesh].indexStart / 3;\r\n const picked = this.pickedBySubMesh;\r\n if (picked[subMesh] && picked[subMesh][faceId]) {\r\n return picked[subMesh][faceId];\r\n }\r\n }\r\n return null;\r\n }\r\n\r\n /**\r\n * Returns a SolidParticle object from its identifier : particle.id\r\n * @param id (integer) the particle Id\r\n * @returns the searched particle or null if not found in the SPS.\r\n */\r\n public getParticleById(id: number): Nullable {\r\n const p = this.particles[id];\r\n if (p && p.id == id) {\r\n return p;\r\n }\r\n const particles = this.particles;\r\n const idx = this._idxOfId[id];\r\n if (idx !== undefined) {\r\n return particles[idx];\r\n }\r\n let i = 0;\r\n const nb = this.nbParticles;\r\n while (i < nb) {\r\n const particle = particles[i];\r\n if (particle.id == id) {\r\n return particle;\r\n }\r\n i++;\r\n }\r\n return null;\r\n }\r\n\r\n /**\r\n * Returns a new array populated with the particles having the passed shapeId.\r\n * @param shapeId (integer) the shape identifier\r\n * @returns a new solid particle array\r\n */\r\n public getParticlesByShapeId(shapeId: number): SolidParticle[] {\r\n const ref: SolidParticle[] = [];\r\n this.getParticlesByShapeIdToRef(shapeId, ref);\r\n return ref;\r\n }\r\n\r\n /**\r\n * Populates the passed array \"ref\" with the particles having the passed shapeId.\r\n * @param shapeId the shape identifier\r\n * @returns the SPS\r\n * @param ref\r\n */\r\n public getParticlesByShapeIdToRef(shapeId: number, ref: SolidParticle[]): SolidParticleSystem {\r\n ref.length = 0;\r\n for (let i = 0; i < this.nbParticles; i++) {\r\n const p = this.particles[i];\r\n if (p.shapeId == shapeId) {\r\n ref.push(p);\r\n }\r\n }\r\n return this;\r\n }\r\n /**\r\n * Computes the required SubMeshes according the materials assigned to the particles.\r\n * @returns the solid particle system.\r\n * Does nothing if called before the SPS mesh is built.\r\n */\r\n public computeSubMeshes(): SolidParticleSystem {\r\n if (!this.mesh || !this._multimaterialEnabled) {\r\n return this;\r\n }\r\n const depthSortedParticles = this.depthSortedParticles;\r\n if (this.particles.length > 0) {\r\n for (let p = 0; p < this.particles.length; p++) {\r\n const part = this.particles[p];\r\n if (!part.materialIndex) {\r\n part.materialIndex = 0;\r\n }\r\n const sortedPart = depthSortedParticles[p];\r\n sortedPart.materialIndex = part.materialIndex;\r\n sortedPart.ind = part._ind;\r\n sortedPart.indicesLength = part._model._indicesLength;\r\n sortedPart.idx = part.idx;\r\n }\r\n }\r\n this._sortParticlesByMaterial();\r\n const indicesByMaterial = this._indicesByMaterial;\r\n const materialIndexes = this._materialIndexes;\r\n const mesh = this.mesh;\r\n mesh.subMeshes = [];\r\n const vcount = mesh.getTotalVertices();\r\n for (let m = 0; m < materialIndexes.length; m++) {\r\n const start = indicesByMaterial[m];\r\n const count = indicesByMaterial[m + 1] - start;\r\n const matIndex = materialIndexes[m];\r\n new SubMesh(matIndex, 0, vcount, start, count, mesh);\r\n }\r\n return this;\r\n }\r\n /**\r\n * Sorts the solid particles by material when MultiMaterial is enabled.\r\n * Updates the indices32 array.\r\n * Updates the indicesByMaterial array.\r\n * Updates the mesh indices array.\r\n * @returns the SPS\r\n * @internal\r\n */\r\n protected _sortParticlesByMaterial(): SolidParticleSystem {\r\n const indicesByMaterial = [0];\r\n this._indicesByMaterial = indicesByMaterial;\r\n const materialIndexes: number[] = [];\r\n this._materialIndexes = materialIndexes;\r\n const depthSortedParticles = this.depthSortedParticles;\r\n depthSortedParticles.sort(this._materialSortFunction);\r\n const length = depthSortedParticles.length;\r\n const indices32 = this._indices32;\r\n const indices = this._indices;\r\n\r\n let subMeshIndex = 0;\r\n let subMeshFaceId = 0;\r\n let sid = 0;\r\n let lastMatIndex = depthSortedParticles[0].materialIndex;\r\n materialIndexes.push(lastMatIndex);\r\n if (this._pickable) {\r\n this.pickedBySubMesh = [[]];\r\n this.pickedParticles = this.pickedBySubMesh[0];\r\n }\r\n for (let sorted = 0; sorted < length; sorted++) {\r\n const sortedPart = depthSortedParticles[sorted];\r\n const lind = sortedPart.indicesLength;\r\n const sind = sortedPart.ind;\r\n if (sortedPart.materialIndex !== lastMatIndex) {\r\n lastMatIndex = sortedPart.materialIndex;\r\n indicesByMaterial.push(sid);\r\n materialIndexes.push(lastMatIndex);\r\n if (this._pickable) {\r\n subMeshIndex++;\r\n this.pickedBySubMesh[subMeshIndex] = [];\r\n subMeshFaceId = 0;\r\n }\r\n }\r\n let faceId = 0;\r\n for (let i = 0; i < lind; i++) {\r\n indices32[sid] = indices[sind + i];\r\n if (this._pickable) {\r\n const f = i % 3;\r\n if (f == 0) {\r\n const pickedData = this.pickedBySubMesh[subMeshIndex][subMeshFaceId];\r\n if (pickedData) {\r\n pickedData.idx = sortedPart.idx;\r\n pickedData.faceId = faceId;\r\n } else {\r\n this.pickedBySubMesh[subMeshIndex][subMeshFaceId] = { idx: sortedPart.idx, faceId: faceId };\r\n }\r\n subMeshFaceId++;\r\n faceId++;\r\n }\r\n }\r\n sid++;\r\n }\r\n }\r\n\r\n indicesByMaterial.push(indices32.length); // add the last number to ease the indices start/count values for subMeshes creation\r\n if (this._updatable) {\r\n this.mesh.updateIndices(indices32);\r\n }\r\n return this;\r\n }\r\n /**\r\n * Sets the material indexes by id materialIndexesById[id] = materialIndex\r\n * @internal\r\n */\r\n protected _setMaterialIndexesById() {\r\n this._materialIndexesById = {};\r\n for (let i = 0; i < this._materials.length; i++) {\r\n const id = this._materials[i].uniqueId;\r\n this._materialIndexesById[id] = i;\r\n }\r\n }\r\n /**\r\n * Returns an array with unique values of Materials from the passed array\r\n * @param array the material array to be checked and filtered\r\n * @internal\r\n */\r\n protected _filterUniqueMaterialId(array: Material[]): Material[] {\r\n const filtered = array.filter(function (value, index, self) {\r\n return self.indexOf(value) === index;\r\n });\r\n return filtered;\r\n }\r\n /**\r\n * Sets a new Standard Material as _defaultMaterial if not already set.\r\n * @internal\r\n */\r\n protected _setDefaultMaterial(): Material {\r\n if (!this._defaultMaterial) {\r\n this._defaultMaterial = new StandardMaterial(this.name + \"DefaultMaterial\", this._scene);\r\n }\r\n return this._defaultMaterial;\r\n }\r\n /**\r\n * Visibility helper : Recomputes the visible size according to the mesh bounding box\r\n * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_visibility\r\n * @returns the SPS.\r\n */\r\n public refreshVisibleSize(): SolidParticleSystem {\r\n if (!this._isVisibilityBoxLocked) {\r\n this.mesh.refreshBoundingInfo();\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Visibility helper : Sets the size of a visibility box, this sets the underlying mesh bounding box.\r\n * @param size the size (float) of the visibility box\r\n * note : this doesn't lock the SPS mesh bounding box.\r\n * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_visibility\r\n */\r\n public setVisibilityBox(size: number): void {\r\n const vis = size / 2;\r\n this.mesh.buildBoundingInfo(new Vector3(-vis, -vis, -vis), new Vector3(vis, vis, vis));\r\n }\r\n\r\n /**\r\n * Gets whether the SPS as always visible or not\r\n * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_visibility\r\n */\r\n public get isAlwaysVisible(): boolean {\r\n return this._alwaysVisible;\r\n }\r\n\r\n /**\r\n * Sets the SPS as always visible or not\r\n * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_visibility\r\n */\r\n public set isAlwaysVisible(val: boolean) {\r\n this._alwaysVisible = val;\r\n this.mesh.alwaysSelectAsActiveMesh = val;\r\n }\r\n\r\n /**\r\n * Sets the SPS visibility box as locked or not. This enables/disables the underlying mesh bounding box updates.\r\n * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_visibility\r\n */\r\n public set isVisibilityBoxLocked(val: boolean) {\r\n this._isVisibilityBoxLocked = val;\r\n\r\n const boundingInfo = this.mesh.getBoundingInfo();\r\n\r\n boundingInfo.isLocked = val;\r\n }\r\n\r\n /**\r\n * Gets if the SPS visibility box as locked or not. This enables/disables the underlying mesh bounding box updates.\r\n * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_visibility\r\n */\r\n public get isVisibilityBoxLocked(): boolean {\r\n return this._isVisibilityBoxLocked;\r\n }\r\n\r\n /**\r\n * Tells to `setParticles()` to compute the particle rotations or not.\r\n * Default value : true. The SPS is faster when it's set to false.\r\n * Note : the particle rotations aren't stored values, so setting `computeParticleRotation` to false will prevents the particle to rotate.\r\n */\r\n public set computeParticleRotation(val: boolean) {\r\n this._computeParticleRotation = val;\r\n }\r\n\r\n /**\r\n * Tells to `setParticles()` to compute the particle colors or not.\r\n * Default value : true. The SPS is faster when it's set to false.\r\n * Note : the particle colors are stored values, so setting `computeParticleColor` to false will keep yet the last colors set.\r\n */\r\n public set computeParticleColor(val: boolean) {\r\n this._computeParticleColor = val;\r\n }\r\n\r\n public set computeParticleTexture(val: boolean) {\r\n this._computeParticleTexture = val;\r\n }\r\n /**\r\n * Tells to `setParticles()` to call the vertex function for each vertex of each particle, or not.\r\n * Default value : false. The SPS is faster when it's set to false.\r\n * Note : the particle custom vertex positions aren't stored values.\r\n */\r\n public set computeParticleVertex(val: boolean) {\r\n this._computeParticleVertex = val;\r\n }\r\n /**\r\n * Tells to `setParticles()` to compute or not the mesh bounding box when computing the particle positions.\r\n */\r\n public set computeBoundingBox(val: boolean) {\r\n this._computeBoundingBox = val;\r\n }\r\n /**\r\n * Tells to `setParticles()` to sort or not the distance between each particle and the camera.\r\n * Skipped when `enableDepthSort` is set to `false` (default) at construction time.\r\n * Default : `true`\r\n */\r\n public set depthSortParticles(val: boolean) {\r\n this._depthSortParticles = val;\r\n }\r\n\r\n /**\r\n * Gets if `setParticles()` computes the particle rotations or not.\r\n * Default value : true. The SPS is faster when it's set to false.\r\n * Note : the particle rotations aren't stored values, so setting `computeParticleRotation` to false will prevents the particle to rotate.\r\n */\r\n public get computeParticleRotation(): boolean {\r\n return this._computeParticleRotation;\r\n }\r\n\r\n /**\r\n * Gets if `setParticles()` computes the particle colors or not.\r\n * Default value : true. The SPS is faster when it's set to false.\r\n * Note : the particle colors are stored values, so setting `computeParticleColor` to false will keep yet the last colors set.\r\n */\r\n public get computeParticleColor(): boolean {\r\n return this._computeParticleColor;\r\n }\r\n\r\n /**\r\n * Gets if `setParticles()` computes the particle textures or not.\r\n * Default value : true. The SPS is faster when it's set to false.\r\n * Note : the particle textures are stored values, so setting `computeParticleTexture` to false will keep yet the last colors set.\r\n */\r\n public get computeParticleTexture(): boolean {\r\n return this._computeParticleTexture;\r\n }\r\n\r\n /**\r\n * Gets if `setParticles()` calls the vertex function for each vertex of each particle, or not.\r\n * Default value : false. The SPS is faster when it's set to false.\r\n * Note : the particle custom vertex positions aren't stored values.\r\n */\r\n public get computeParticleVertex(): boolean {\r\n return this._computeParticleVertex;\r\n }\r\n\r\n /**\r\n * Gets if `setParticles()` computes or not the mesh bounding box when computing the particle positions.\r\n */\r\n public get computeBoundingBox(): boolean {\r\n return this._computeBoundingBox;\r\n }\r\n\r\n /**\r\n * Gets if `setParticles()` sorts or not the distance between each particle and the camera.\r\n * Skipped when `enableDepthSort` is set to `false` (default) at construction time.\r\n * Default : `true`\r\n */\r\n public get depthSortParticles(): boolean {\r\n return this._depthSortParticles;\r\n }\r\n\r\n /**\r\n * Gets if the SPS is created as expandable at construction time.\r\n * Default : `false`\r\n */\r\n public get expandable(): boolean {\r\n return this._expandable;\r\n }\r\n /**\r\n * Gets if the SPS supports the Multi Materials\r\n */\r\n public get multimaterialEnabled(): boolean {\r\n return this._multimaterialEnabled;\r\n }\r\n /**\r\n * Gets if the SPS uses the model materials for its own multimaterial.\r\n */\r\n public get useModelMaterial(): boolean {\r\n return this._useModelMaterial;\r\n }\r\n /**\r\n * The SPS used material array.\r\n */\r\n public get materials(): Material[] {\r\n return this._materials;\r\n }\r\n /**\r\n * Sets the SPS MultiMaterial from the passed materials.\r\n * Note : the passed array is internally copied and not used then by reference.\r\n * @param materials an array of material objects. This array indexes are the materialIndex values of the particles.\r\n */\r\n public setMultiMaterial(materials: Material[]) {\r\n this._materials = this._filterUniqueMaterialId(materials);\r\n this._setMaterialIndexesById();\r\n if (this._multimaterial) {\r\n this._multimaterial.dispose();\r\n }\r\n this._multimaterial = new MultiMaterial(this.name + \"MultiMaterial\", this._scene);\r\n for (let m = 0; m < this._materials.length; m++) {\r\n this._multimaterial.subMaterials.push(this._materials[m]);\r\n }\r\n this.computeSubMeshes();\r\n this.mesh.material = this._multimaterial;\r\n }\r\n /**\r\n * The SPS computed multimaterial object\r\n */\r\n public get multimaterial(): MultiMaterial {\r\n return this._multimaterial;\r\n }\r\n public set multimaterial(mm) {\r\n this._multimaterial = mm;\r\n }\r\n /**\r\n * If the subMeshes must be updated on the next call to setParticles()\r\n */\r\n public get autoUpdateSubMeshes(): boolean {\r\n return this._autoUpdateSubMeshes;\r\n }\r\n public set autoUpdateSubMeshes(val: boolean) {\r\n this._autoUpdateSubMeshes = val;\r\n }\r\n // =======================================================================\r\n // Particle behavior logic\r\n // these following methods may be overwritten by the user to fit his needs\r\n\r\n /**\r\n * This function does nothing. It may be overwritten to set all the particle first values.\r\n * The SPS doesn't call this function, you may have to call it by your own.\r\n * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/manage_sps_particles\r\n */\r\n public initParticles(): void {}\r\n\r\n /**\r\n * This function does nothing. It may be overwritten to recycle a particle.\r\n * The SPS doesn't call this function, you may have to call it by your own.\r\n * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/manage_sps_particles\r\n * @param particle The particle to recycle\r\n * @returns the recycled particle\r\n */\r\n public recycleParticle(particle: SolidParticle): SolidParticle {\r\n return particle;\r\n }\r\n\r\n /**\r\n * Updates a particle : this function should be overwritten by the user.\r\n * It is called on each particle by `setParticles()`. This is the place to code each particle behavior.\r\n * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/manage_sps_particles\r\n * @example : just set a particle position or velocity and recycle conditions\r\n * @param particle The particle to update\r\n * @returns the updated particle\r\n */\r\n public updateParticle(particle: SolidParticle): SolidParticle {\r\n return particle;\r\n }\r\n\r\n /**\r\n * Updates a vertex of a particle : it can be overwritten by the user.\r\n * This will be called on each vertex particle by `setParticles()` if `computeParticleVertex` is set to true only.\r\n * @param particle the current particle\r\n * @param vertex the current vertex of the current particle : a SolidParticleVertex object\r\n * @param pt the index of the current vertex in the particle shape\r\n * doc : https://doc.babylonjs.com/features/featuresDeepDive/particles/solid_particle_system/sps_vertices\r\n * @example : just set a vertex particle position or color\r\n * @returns the sps\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public updateParticleVertex(particle: SolidParticle, vertex: SolidParticleVertex, pt: number): SolidParticleSystem {\r\n return this;\r\n }\r\n\r\n /**\r\n * This will be called before any other treatment by `setParticles()` and will be passed three parameters.\r\n * This does nothing and may be overwritten by the user.\r\n * @param start the particle index in the particle array where to stop to iterate, same than the value passed to setParticle()\r\n * @param stop the particle index in the particle array where to stop to iterate, same than the value passed to setParticle()\r\n * @param update the boolean update value actually passed to setParticles()\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public beforeUpdateParticles(start?: number, stop?: number, update?: boolean): void {}\r\n /**\r\n * This will be called by `setParticles()` after all the other treatments and just before the actual mesh update.\r\n * This will be passed three parameters.\r\n * This does nothing and may be overwritten by the user.\r\n * @param start the particle index in the particle array where to stop to iterate, same than the value passed to setParticle()\r\n * @param stop the particle index in the particle array where to stop to iterate, same than the value passed to setParticle()\r\n * @param update the boolean update value actually passed to setParticles()\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public afterUpdateParticles(start?: number, stop?: number, update?: boolean): void {}\r\n}\r\n","import { VertexBuffer } from '@babylonjs/core/Meshes/buffer';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\n\r\n/** This will mutate the mesh geometry. Only use this on temp/intermediate mesh data */\r\nexport function ensureNormalsExist(mesh: Mesh) {\r\n if (!mesh.getVerticesData(VertexBuffer.NormalKind)) {\r\n mesh.createNormals(true);\r\n mesh.convertToFlatShadedMesh();\r\n }\r\n}\r\n","import { Quaternion, Vector3 } from '@babylonjs/core/Maths';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { VertexData } from '@babylonjs/core/Meshes/mesh.vertexData';\r\nimport { SolidParticle } from '@babylonjs/core/Particles/solidParticle';\r\nimport { SolidParticleSystem } from '@babylonjs/core/Particles/solidParticleSystem';\r\nimport { IKbVectorLike, KbVector } from '@models/classes';\r\nimport { PatternFeature } from '@models/classes/features/pattern-feature';\r\nimport { MeshNode } from '@models/classes/meshes/mesh-node';\r\nimport { SketchNode } from '@models/classes/meshes/sketch-node';\r\nimport { ITrackedClass } from '@models/interfaces/tracked-class';\r\nimport { SubmeshVertexData } from '@view/helpers/geometry-helper';\r\nimport { FeatureRenderer } from './feature';\r\nimport { ensureNormalsExist } from './utils';\r\n\r\ntype ParticleUpdaterFunction = (\r\n result: SolidParticle,\r\n iterator: IterableIterator,\r\n index?: number\r\n) => void;\r\ninterface IPatternDef {\r\n sps: SolidParticleSystem;\r\n size: number;\r\n resultMesh?: Mesh;\r\n sourceNode: MeshNode;\r\n featureUpdaters: Map;\r\n needsUpdate: boolean; // Whether we need to reposition the particles on this pattern\r\n needsRebuild: boolean; // Whether we need to rebuild the pattern mesh to create less or more geometry polygons\r\n needsSubmeshRebuild: boolean; // Whether we need to rebuild the submesh data on the pattern mesh\r\n shape: Mesh;\r\n}\r\nconst patternStore = new Map();\r\nconst identityRotation = Quaternion.Identity();\r\n\r\nexport abstract class PatternRenderer extends FeatureRenderer {\r\n update3dInternal() {}\r\n\r\n updateMeshGeometry(\r\n feature: PatternFeature,\r\n vertexData: SubmeshVertexData,\r\n vertexDataChanges: Partial,\r\n setParticlePositionFunc?: (idx: number, particle: SolidParticle) => void\r\n ) {\r\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\r\n\r\n const meshNode = feature._parent!;\r\n const mesh = this.viewer.getRendererNode(meshNode);\r\n\r\n if (mesh) {\r\n /** See line 156 */\r\n // let _pattern = patternStore.get(meshNode._dynamicId);\r\n let _pattern = null;\r\n\r\n // If there are no enabled pattern features before this one, update the pattern shape as some geometry may have changed\r\n if (!_pattern) {\r\n const meshUpdated = new Mesh('patternTempGeometry');\r\n meshUpdated.setEnabled(false);\r\n const tempVertexData = new SubmeshVertexData();\r\n tempVertexData.copyFrom(newVertexData);\r\n tempVertexData.applyToMesh(meshUpdated, false);\r\n\r\n ensureNormalsExist(meshUpdated);\r\n\r\n const sps = new SolidParticleSystem(`${mesh.id}_${feature._dynamicId}`, this.scene, {\r\n enableDepthSort: true,\r\n enableMultiMaterial: true,\r\n expandable: true,\r\n });\r\n\r\n if (feature) {\r\n if (feature.meshId) {\r\n const sketchNode = this.viewer.getNode(feature.meshId);\r\n if (sketchNode instanceof SketchNode) {\r\n let distance = feature.distance;\r\n if (distance <= 0) {\r\n distance = 1;\r\n }\r\n const positionArray = this.getPositionArray(\r\n sketchNode,\r\n feature.distance,\r\n feature.addEndMesh\r\n );\r\n sps.addShape(meshUpdated, positionArray.length);\r\n sps.counter = positionArray.length;\r\n feature.count = positionArray.length;\r\n const sketchPosition = sketchNode.position.toVec3();\r\n sps.initParticles = () => {\r\n for (let p = 0; p < sps.counter; p++) {\r\n const particle = sps.particles[p];\r\n particle.position = positionArray[p].add(sketchPosition);\r\n }\r\n };\r\n }\r\n } else {\r\n sps.addShape(meshUpdated, feature.count + 1);\r\n sps.initParticles = () => {\r\n for (let p = 0; p < sps.nbParticles; p++) {\r\n const particle = sps.particles[p];\r\n if (particle.rotationQuaternion) {\r\n particle.rotationQuaternion!.copyFrom(identityRotation);\r\n } else {\r\n particle.rotationQuaternion = Quaternion.Identity();\r\n }\r\n particle.position.setAll(0);\r\n }\r\n };\r\n }\r\n }\r\n\r\n const newPattern: IPatternDef = {\r\n sps,\r\n size: 1,\r\n featureUpdaters: new Map(),\r\n sourceNode: meshNode,\r\n needsUpdate: true,\r\n needsRebuild: true,\r\n needsSubmeshRebuild: true,\r\n shape: meshUpdated,\r\n };\r\n sps.updateParticle = particle => {\r\n // Process the last update function as it will call the prior ones recursively\r\n const updates = newPattern.featureUpdaters.values();\r\n const update: ParticleUpdaterFunction = updates.next().value;\r\n update(particle, updates);\r\n // Do we need this?\r\n // particle.scaling.copyFrom(mesh.scaling);\r\n return particle;\r\n };\r\n patternStore.set(meshNode._dynamicId, newPattern);\r\n _pattern = newPattern;\r\n }\r\n const pattern = _pattern;\r\n pattern.needsUpdate = true;\r\n\r\n pattern.featureUpdaters.set(feature._dynamicId, (particle, iterator, index) => {\r\n if (feature.enabled) {\r\n const particleIndex = particle.idx;\r\n if (particleIndex >= pattern.size) {\r\n particle.isVisible = false;\r\n return;\r\n }\r\n if (!particle.isVisible) {\r\n particle.isVisible = true;\r\n }\r\n const featureItemIndex = index !== undefined ? index : particleIndex;\r\n\r\n const i = featureItemIndex % feature.count;\r\n if (i > 0 && setParticlePositionFunc) {\r\n setParticlePositionFunc(i, particle);\r\n }\r\n\r\n const nextUpdateFunction: ParticleUpdaterFunction = iterator.next().value;\r\n if (nextUpdateFunction) {\r\n nextUpdateFunction(particle, iterator, Math.floor(featureItemIndex / feature.count));\r\n }\r\n }\r\n });\r\n\r\n pattern.size = Array.from(pattern.featureUpdaters.keys()).reduce((result, featureId) => {\r\n const feature = meshNode.features.find(feature => feature._dynamicId === featureId) as PatternFeature;\r\n return result * feature!.count;\r\n }, 1);\r\n\r\n if (\r\n !pattern.needsRebuild &&\r\n (vertexDataChanges.indices || vertexDataChanges.positions || vertexDataChanges.uvs)\r\n ) {\r\n /**\r\n * If the shape of the mesh changed, then we have to remove and re-add every particle to recompute the geometry.\r\n * Potential optimization here is to only recompute vertex positions if the vertex count is the same,\r\n * and only do the following expensive operation if the vertex count (indices?) changes.\r\n */\r\n const currentSize = pattern.sps.nbParticles;\r\n pattern.sps.addShape(pattern.shape, Math.max(pattern.size, pattern.sps.nbParticles));\r\n pattern.sps.removeParticles(0, currentSize - 1);\r\n pattern.needsRebuild = true;\r\n } else if (pattern.sps.nbParticles < pattern.size) {\r\n // If we don't have enough particles to complete the pattern, expand the SPS as needed\r\n pattern.sps.addShape(pattern.shape, pattern.size - pattern.sps.nbParticles);\r\n pattern.needsRebuild = true;\r\n }\r\n\r\n if (vertexDataChanges.submeshData) {\r\n pattern.needsSubmeshRebuild = true;\r\n }\r\n\r\n /**\r\n * We are removing the pattern feature combination system because it was causing a lot of issues around the cache, with enabling/disabling and reordering features.\r\n * In the future we will just add multi-direction properties into the pattern feature instead.\r\n */\r\n // const nextFeature = meshNode.features[thisIndex + 1];\r\n // if (!nextFeature || !(nextFeature instanceof PatternFeature)) {\r\n this.processParticleSystem(pattern, newVertexData); // mutates newVertexData\r\n patternStore.delete(meshNode._dynamicId); // Delete the pattern since we are done processing it\r\n // }\r\n }\r\n\r\n return newVertexData;\r\n }\r\n\r\n processParticleSystem(pattern: IPatternDef, resultVertexData: Partial) {\r\n if (!pattern.needsUpdate) {\r\n return;\r\n }\r\n\r\n // Shrink the particle system if there are way too many particles (more than 50% too many)\r\n // We don't want to shrink unless necessary beca use there's a performance hit to rebuilding the mesh\r\n if (pattern.sps.nbParticles - pattern.size > pattern.size / 2) {\r\n pattern.sps.removeParticles(pattern.size, pattern.sps.nbParticles - 1);\r\n pattern.needsRebuild = true;\r\n }\r\n\r\n pattern.sps.initParticles();\r\n\r\n if (!pattern.resultMesh || pattern.needsRebuild) {\r\n pattern.resultMesh = pattern.sps.buildMesh();\r\n pattern.needsRebuild = false;\r\n pattern.needsSubmeshRebuild = true;\r\n pattern.resultMesh.setEnabled(false);\r\n }\r\n\r\n pattern.sps.computeBoundingBox = true;\r\n pattern.sps.setParticles();\r\n\r\n const totalIndicesSourceCount = resultVertexData.indices!.length;\r\n const totalVerticesSourceCount = resultVertexData.positions!.length / 3;\r\n\r\n const newVertexData = VertexData.ExtractFromMesh(pattern.resultMesh);\r\n resultVertexData.indices = newVertexData.indices;\r\n resultVertexData.positions = newVertexData.positions;\r\n if (newVertexData.normals) {\r\n resultVertexData.normals = newVertexData.normals;\r\n }\r\n if (resultVertexData.uvs && newVertexData.uvs) {\r\n // Don't add UVs if UVs never existed\r\n resultVertexData.uvs = newVertexData.uvs;\r\n }\r\n\r\n // If the source mesh has submeshes, then we need to update them\r\n if (resultVertexData.submeshData) {\r\n const submeshesToClone = resultVertexData.submeshData.slice(1);\r\n\r\n const totalIndicesResultCount = resultVertexData.indices!.length;\r\n const totalVerticesResultCount = resultVertexData.positions!.length / 3;\r\n // resultMeshCsg.dispose();\r\n const baseMaterialIndex = resultVertexData.submeshData[0].materialIndex;\r\n\r\n resultVertexData.submeshData = [];\r\n resultVertexData.submeshData.push({\r\n materialIndex: baseMaterialIndex,\r\n verticesStart: 0,\r\n verticesCount: totalVerticesResultCount,\r\n indexStart: 0,\r\n indexCount: totalIndicesResultCount,\r\n });\r\n\r\n const spsPatternSize = totalVerticesResultCount / totalVerticesSourceCount;\r\n\r\n if (submeshesToClone.length > 0) {\r\n submeshesToClone.forEach(subMesh => {\r\n for (let i = 0; i < spsPatternSize; i++) {\r\n resultVertexData.submeshData!.push({\r\n materialIndex: subMesh.materialIndex,\r\n verticesStart: subMesh.verticesStart + i * totalVerticesSourceCount,\r\n verticesCount: subMesh.verticesCount,\r\n indexStart: subMesh.indexStart + i * totalIndicesSourceCount,\r\n indexCount: subMesh.indexCount,\r\n });\r\n }\r\n });\r\n }\r\n }\r\n }\r\n\r\n getPositionArray(node: SketchNode, distance: number, addEndMesh: boolean) {\r\n const result: Vector3[] = [];\r\n const sketchLength = node.getSketchLength();\r\n const numberOfMeshes = sketchLength / distance;\r\n const percentagePoints: number[] = [];\r\n for (let i = 0; i < numberOfMeshes; i++) {\r\n percentagePoints.push((1 / numberOfMeshes) * i);\r\n }\r\n\r\n percentagePoints.forEach(point => {\r\n result.push(this.getIntervalPosition(node, point));\r\n });\r\n\r\n if (addEndMesh) {\r\n const lastPath = node.paths[node.paths.length - 1];\r\n const lastPoint = lastPath._points[lastPath._points.length - 1];\r\n const lastPointVec3 = new Vector3(lastPoint.x, lastPoint.y, lastPoint.z);\r\n result.push(lastPointVec3);\r\n }\r\n\r\n return result;\r\n }\r\n\r\n protected getIntervalPosition(node: SketchNode, posZeroToOne: number) {\r\n const result = new Vector3(0, 0, 0);\r\n const sketchLength = node.getSketchLength();\r\n let trackedLength = 0;\r\n let targetNotFound = true;\r\n node.paths.forEach(path => {\r\n let lastPosition: IKbVectorLike | null = null;\r\n let lastLength = 0;\r\n path._points.forEach(point => {\r\n if (!lastPosition) {\r\n lastPosition = point;\r\n } else {\r\n const currentPos = point;\r\n const distance = Math.sqrt(\r\n Math.pow(lastPosition.x - currentPos.x, 2) +\r\n Math.pow(lastPosition.y - currentPos.y, 2) +\r\n Math.pow(lastPosition.z - currentPos.z, 2)\r\n );\r\n lastLength = trackedLength;\r\n trackedLength += distance;\r\n const distanceBetweenPoints = trackedLength - lastLength;\r\n const distanceFromP1toTarget = posZeroToOne * sketchLength - lastLength;\r\n const percentageBetween2Points = distanceFromP1toTarget / distanceBetweenPoints;\r\n const vectorBetween2Points = new KbVector(\r\n currentPos.x - lastPosition.x,\r\n currentPos.y - lastPosition.y,\r\n currentPos.z - lastPosition.z\r\n );\r\n if (trackedLength > posZeroToOne * sketchLength && targetNotFound) {\r\n const calculatedPosition = vectorBetween2Points\r\n .scale(percentageBetween2Points)\r\n .add(lastPosition);\r\n result.set(calculatedPosition.x, calculatedPosition.y, calculatedPosition.z);\r\n targetNotFound = false;\r\n }\r\n lastPosition = currentPos;\r\n }\r\n });\r\n if (posZeroToOne === 1) {\r\n const path = node.paths[node.paths.length - 1];\r\n const controlPoint = path._points[path._points.length - 1];\r\n result.set(controlPoint.x, controlPoint.y, controlPoint.z);\r\n }\r\n });\r\n\r\n return result;\r\n }\r\n\r\n delete3d(feature: ITrackedClass, preserveCache?: boolean) {\r\n patternStore.forEach((pattern, meshNode) => {\r\n if (pattern.featureUpdaters.has(feature._dynamicId)) {\r\n pattern.featureUpdaters.delete(feature._dynamicId);\r\n }\r\n if (pattern.featureUpdaters.size === 0) {\r\n patternStore.delete(meshNode);\r\n }\r\n });\r\n\r\n super.delete3d(feature, preserveCache);\r\n }\r\n\r\n dispose() {\r\n super.dispose();\r\n patternStore.clear();\r\n }\r\n}\r\n","import { MultiMaterial } from '@babylonjs/core/Materials/multiMaterial';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { eUvMode } from '@models/classes';\r\nimport { SubmeshData, SubmeshFeature } from '@models/classes/features/submesh-feature';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass } from '@models/interfaces/tracked-class';\r\nimport { SubmeshVertexData } from '@view/helpers/geometry-helper';\r\nimport { UVMapHelper } from '@view/helpers/uvmap-helper';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { RendererManager } from '../renderer-manager';\r\nimport { FeatureRenderer } from './feature';\r\n\r\nconst uvHelper = new UVMapHelper();\r\nconst multiMaterialUseCount = new Map();\r\nconst featureMappings = new Map<\r\n string,\r\n { multiMaterial?: MultiMaterial; multiMaterialIndex?: number; initialized?: boolean }\r\n>();\r\n\r\nexport class SubmeshRenderer extends FeatureRenderer {\r\n constructor(viewer: KbViewer, scene: Scene, private rendererManager: RendererManager) {\r\n super(viewer, scene);\r\n }\r\n modelToken = Token.SubmeshFeature;\r\n updateMeshGeometry(\r\n feature: ITrackedClass,\r\n vertexData: SubmeshVertexData,\r\n vertexDataChanges: Partial\r\n ) {\r\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\r\n const submeshFeatureMapping = featureMappings.get(feature._dynamicId);\r\n\r\n if (!submeshFeatureMapping) {\r\n console.warn(`INFO: Submesh mapping not found for ${feature.name}`);\r\n return newVertexData;\r\n }\r\n\r\n if (!newVertexData.submeshData) {\r\n newVertexData.submeshData = [];\r\n }\r\n\r\n if (submeshFeatureMapping.multiMaterialIndex) {\r\n const verticesCount = (vertexDataChanges.positions || newVertexData.positions!).length / 3;\r\n const indicesCount = (vertexDataChanges.indices || newVertexData.indices!).length;\r\n\r\n for (let i = 0; i < feature.vertexGroups.length; i++) {\r\n const vertexGroup = feature.vertexGroups[i];\r\n\r\n const newSubmesh: SubmeshData = {\r\n materialIndex: submeshFeatureMapping.multiMaterialIndex,\r\n verticesStart: 0,\r\n verticesCount: verticesCount,\r\n indexStart: vertexGroup.startIndex,\r\n indexCount: Math.min(vertexGroup.endIndex, indicesCount) - vertexGroup.startIndex,\r\n };\r\n\r\n const existingIndex = newVertexData.submeshData.findIndex(\r\n existingSubmesh =>\r\n existingSubmesh.materialIndex === newSubmesh.materialIndex &&\r\n existingSubmesh.verticesStart === newSubmesh.verticesStart &&\r\n existingSubmesh.verticesCount === newSubmesh.verticesCount &&\r\n existingSubmesh.indexStart === newSubmesh.indexStart &&\r\n existingSubmesh.indexCount === newSubmesh.indexCount\r\n );\r\n\r\n if (existingIndex >= 0) {\r\n newVertexData.submeshData.splice(existingIndex, 1);\r\n }\r\n\r\n newVertexData.submeshData.push(newSubmesh);\r\n }\r\n }\r\n\r\n // Calculate UV maps if that is enabled on this feature\r\n if (feature.uvMapping) {\r\n const uvMapVertexes: number[] = [];\r\n\r\n for (let i = 0; i < feature.vertexGroups.length; i++) {\r\n const vertexGroup = feature.vertexGroups[i];\r\n\r\n for (let i = vertexGroup.startIndex; i < vertexGroup.endIndex; i++) {\r\n uvMapVertexes.push(i);\r\n }\r\n }\r\n\r\n if (uvMapVertexes.length > 0) {\r\n let existingUvs: number[] = [];\r\n if (newVertexData.uvs) {\r\n existingUvs = [...newVertexData.uvs];\r\n }\r\n\r\n uvHelper.generateUVMapInPlace(newVertexData, feature, uvMapVertexes);\r\n\r\n // for projection mode \"use existing\", only apply the new uvs to the selected faces of the submesh\r\n if (feature.mode === eUvMode.useExisting) {\r\n feature.vertexGroups.forEach(vertexGroup => {\r\n for (let i = vertexGroup.startIndex; i < vertexGroup.endIndex; i++) {\r\n if (newVertexData.indices && newVertexData.uvs) {\r\n existingUvs[newVertexData.indices[i] * 2] =\r\n newVertexData.uvs[newVertexData.indices[i] * 2];\r\n existingUvs[newVertexData.indices[i] * 2 + 1] =\r\n newVertexData.uvs[newVertexData.indices[i] * 2 + 1];\r\n }\r\n }\r\n });\r\n newVertexData.uvs = existingUvs;\r\n }\r\n }\r\n }\r\n\r\n return newVertexData;\r\n }\r\n\r\n update3dInternal(feature: ITrackedClass) {\r\n let _submeshFeatureMapping = featureMappings.get(feature._dynamicId);\r\n\r\n if (feature.changes.has('materialId') || !_submeshFeatureMapping) {\r\n const meshNode = feature._parent!;\r\n const mesh = this.viewer.getRendererNode(meshNode) as Mesh;\r\n\r\n if (!mesh) {\r\n return;\r\n }\r\n\r\n if (!_submeshFeatureMapping) {\r\n _submeshFeatureMapping = {};\r\n featureMappings.set(feature._dynamicId, _submeshFeatureMapping);\r\n }\r\n\r\n const submeshFeatureMapping = _submeshFeatureMapping;\r\n\r\n if (feature.materialId) {\r\n if (!(mesh.material instanceof MultiMaterial)) {\r\n const multiMaterial = new MultiMaterial(meshNode._dynamicId, this.scene);\r\n if (mesh.material) {\r\n multiMaterial.subMaterials[0] = mesh.material;\r\n } else if (!mesh.material) {\r\n // If a submesh multimaterial has a null material at index 0, it causes shadows to not work\r\n multiMaterial.subMaterials[0] = this.rendererManager.material.defaultMaterial;\r\n }\r\n submeshFeatureMapping.multiMaterial = multiMaterial;\r\n multiMaterialUseCount.set(multiMaterial, 0);\r\n mesh.material = multiMaterial;\r\n } else {\r\n submeshFeatureMapping.multiMaterial = mesh.material;\r\n }\r\n\r\n this.rendererManager.mesh.fetchMaterialNode(meshNode, feature.materialId).then(\r\n materialNode => {\r\n // The material may have changed on the mesh, so we have to re-fetch the reference just in case.\r\n if (!(mesh.material instanceof MultiMaterial)) {\r\n throw Error('Another script has removed the multimaterial on a mesh');\r\n }\r\n const multiMaterial = mesh.material;\r\n if (multiMaterial !== submeshFeatureMapping.multiMaterial) {\r\n console.warn(\r\n `Submesh feature ${feature.name} lost reference to a multimaterial. Possible memory leak?`\r\n );\r\n submeshFeatureMapping.multiMaterial = multiMaterial;\r\n }\r\n\r\n // If the material is already part of the submaterials, do nothing\r\n const existingSubmaterialIndex = multiMaterial.subMaterials.findIndex(\r\n material => material?.id === materialNode._dynamicId\r\n );\r\n if (submeshFeatureMapping?.multiMaterialIndex === existingSubmaterialIndex) {\r\n return;\r\n } else if (existingSubmaterialIndex > -1) {\r\n submeshFeatureMapping.multiMaterialIndex = existingSubmaterialIndex;\r\n\r\n // Triggers change on this feature so the feature stack refreshes with the material index on this submesh\r\n feature.setPropertyIsModified('vertexGroups', true);\r\n } else {\r\n // init the material if the node is showing\r\n if (meshNode.visible) {\r\n materialNode.init();\r\n }\r\n\r\n const material = this.viewer.getRendererNode(materialNode);\r\n if (!material) {\r\n throw Error('Material could not be found in renderer. Has the material been created?');\r\n }\r\n\r\n submeshFeatureMapping.multiMaterialIndex = multiMaterial.subMaterials.length;\r\n multiMaterial.subMaterials.push(material);\r\n multiMaterialUseCount.set(\r\n multiMaterial,\r\n (multiMaterialUseCount.get(multiMaterial) || 0) + 1\r\n );\r\n\r\n // Triggers change on this feature so the feature stack refreshes with the material index on this submesh\r\n feature.setPropertyIsModified('vertexGroups', true);\r\n }\r\n },\r\n err => {\r\n feature.materialId = '';\r\n }\r\n );\r\n }\r\n }\r\n }\r\n\r\n public delete3d(feature: ITrackedClass, preserveCache?: boolean): void {\r\n if (!preserveCache) {\r\n const submeshFeatureMapping = featureMappings.get(feature._dynamicId);\r\n if (submeshFeatureMapping && submeshFeatureMapping.multiMaterial) {\r\n const multiMaterial = submeshFeatureMapping.multiMaterial;\r\n const count = multiMaterialUseCount.get(multiMaterial);\r\n\r\n if (count !== undefined) {\r\n if (count <= 1) {\r\n const meshNode = feature._parent!;\r\n const mesh = this.viewer.getRendererNode(meshNode) as Mesh;\r\n if (mesh) {\r\n mesh.material = multiMaterial.getSubMaterial(0);\r\n }\r\n multiMaterialUseCount.delete(multiMaterial);\r\n multiMaterial.dispose(true, true, false);\r\n } else {\r\n multiMaterialUseCount.set(multiMaterial, count - 1);\r\n }\r\n }\r\n }\r\n featureMappings.delete(feature._dynamicId);\r\n }\r\n super.delete3d(feature, preserveCache);\r\n }\r\n}\r\n","\r\nimport { Color3, Quaternion, Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { UvMapFeature } from '@models/classes/features/uv-map-feature';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass } from '@models/interfaces';\r\nimport { rotationToVector } from '@view/helpers';\r\nimport { SubmeshVertexData } from '@view/helpers/geometry-helper';\r\nimport { createUvMapVisualization, UVMapHelper } from '@view/helpers/uvmap-helper';\r\nimport { FeatureRenderer } from './feature';\r\n\r\nconst uvHelper = new UVMapHelper();\r\n\r\nexport class UvMapRenderer extends FeatureRenderer {\r\n modelToken = Token.UvMapFeature;\r\n babylonUvPreviewsMap = new Map();\r\n previewLayer: UtilityLayerRenderer = this.viewer.utilityLayer;\r\n\r\n updateMeshGeometry(\r\n feature: UvMapFeature,\r\n vertexData: SubmeshVertexData,\r\n vertexDataChanges: Partial\r\n ) {\r\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\r\n\r\n uvHelper.generateUVMapInPlace(newVertexData, feature);\r\n\r\n return newVertexData;\r\n }\r\n\r\n protected updatePreviewMesh(feature: UvMapFeature) {\r\n let visualizationMesh = this.babylonUvPreviewsMap.get(feature.id);\r\n const scene = this.previewLayer.utilityLayerScene;\r\n\r\n if (!visualizationMesh) {\r\n visualizationMesh = createUvMapVisualization(feature, scene, Color3.Yellow());\r\n visualizationMesh.isPickable = false;\r\n this.babylonUvPreviewsMap.set(feature.id, visualizationMesh);\r\n } else {\r\n this.deletePreview3d(feature);\r\n visualizationMesh = createUvMapVisualization(feature, scene, Color3.Yellow());\r\n visualizationMesh.isPickable = false;\r\n this.babylonUvPreviewsMap.set(feature.id, visualizationMesh);\r\n }\r\n\r\n let meshPosition = visualizationMesh.getAbsolutePosition();\r\n if (feature._parent) {\r\n const featureParent = this.viewer.getRendererNode(feature._parent);\r\n meshPosition = featureParent.getAbsolutePosition();\r\n }\r\n meshPosition.add(feature.center.toVec3());\r\n meshPosition.addInPlace(feature.center.toVec3());\r\n visualizationMesh.position.set(meshPosition.x, meshPosition.y, meshPosition.z);\r\n //visualizationMesh.project;\r\n\r\n const newNormal = new Vector3();\r\n\r\n newNormal.set(feature.projectionAxis.x, feature.projectionAxis.y, feature.projectionAxis.z);\r\n\r\n visualizationMesh.rotationQuaternion = Quaternion.RotationAxis(newNormal, (feature.angle / 180) * Math.PI);\r\n visualizationMesh.rotationQuaternion = visualizationMesh.rotationQuaternion.multiply(\r\n rotationToVector(newNormal)\r\n );\r\n visualizationMesh.scaling.set(feature.scale.x, feature.scale.y, feature.scale.z);\r\n }\r\n\r\n update3dInternal(feature: ITrackedClass) {\r\n const visualizationMesh = this.babylonUvPreviewsMap.get(feature.id);\r\n\r\n visualizationMesh?.setEnabled(feature._previewMaterialVisible);\r\n\r\n // change visibility here\r\n if (feature._previewMaterialVisible) {\r\n this.updatePreviewMesh(feature);\r\n }\r\n }\r\n\r\n delete3d(feature: UvMapFeature) {\r\n this.deletePreview3d(feature);\r\n super.delete3d(feature as ITrackedClass);\r\n }\r\n\r\n deletePreview3d(feature: UvMapFeature) {\r\n const previewMesh = this.babylonUvPreviewsMap.get(feature.id);\r\n if (previewMesh) {\r\n previewMesh.dispose();\r\n }\r\n this.babylonUvPreviewsMap.delete(feature.id);\r\n }\r\n}\r\n","import {\r\n DirectionalLight,\r\n HemisphericLight,\r\n Light,\r\n PointLight,\r\n ShadowGenerator,\r\n ShadowLight,\r\n SpotLight,\r\n} from '@babylonjs/core/Lights';\r\nimport { Color3, Quaternion, Vector3 } from '@babylonjs/core/Maths';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { DirectionalLightNode, HemisphericLightNode, PointLightNode, SpotLightNode, Token } from '@models/classes';\r\nimport { LightNode } from '@models/classes/lights/light-node';\r\nimport { ShadowLightNode } from '@models/classes/lights/shadow-light-node';\r\nimport { ITrackedClass } from '@models/interfaces/tracked-class';\r\nimport { createLightVisualization } from '@view/helpers/light-helper';\r\nimport { rotationToVector, toRadians } from '@view/helpers/math-helper';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { Renderer } from './renderer';\r\nimport { ShadowManager } from './shadow-manager';\r\n\r\ntype AnyLightNode = PointLightNode & DirectionalLightNode & HemisphericLightNode & SpotLightNode;\r\n\r\nexport class LightRenderer extends Renderer {\r\n babylonLightPreviewsMap = new Map();\r\n gizmoLayer: UtilityLayerRenderer;\r\n\r\n modelToken = Token.LightNode;\r\n\r\n set showPreviews(show: boolean) {\r\n this.showAllLightPreviews = show;\r\n this.babylonMap.forEach(light => {\r\n this.updatePreviewMesh(light);\r\n });\r\n }\r\n\r\n protected showAllLightPreviews = false;\r\n\r\n constructor(viewer: KbViewer, scene: Scene, private shadowManager: ShadowManager) {\r\n super(viewer, scene);\r\n this.gizmoLayer = this.viewer.utilityLayer;\r\n }\r\n\r\n update3dInternal(node: ITrackedClass) {\r\n if (!node._parent && node._parentScene?.isNested()) {\r\n //don't render global lights in nested scenes\r\n return;\r\n }\r\n\r\n let light = this.get3d(node);\r\n const changes = this.getChanges(node) as Map;\r\n\r\n if (!light) {\r\n if (node instanceof PointLightNode) {\r\n light = new PointLight(node._dynamicId, node.position.toVec3(), this.scene);\r\n } else if (node instanceof HemisphericLightNode) {\r\n light = new HemisphericLight(node._dynamicId, node.direction.toVec3(), this.scene);\r\n } else if (node instanceof SpotLightNode) {\r\n light = new SpotLight(\r\n node._dynamicId,\r\n node.position.toVec3(),\r\n node.direction.toVec3(),\r\n (node.coneAngle * Math.PI) / 180,\r\n 2,\r\n this.scene\r\n );\r\n (light as SpotLight).exponent = 2;\r\n } else if (node instanceof DirectionalLightNode) {\r\n light = new DirectionalLight(node._dynamicId, node.direction.toVec3(), this.scene);\r\n }\r\n\r\n this.set3d(node, light as Light);\r\n }\r\n\r\n const pLight = light as Light;\r\n\r\n if (changes.has('visible')) {\r\n // Can be falsy so need to check the map\r\n pLight.setEnabled(node.visible);\r\n }\r\n if (changes.has('intensity')) {\r\n // Can be falsy so need to check the map\r\n pLight.intensity = node.intensity;\r\n }\r\n if (changes.has('color')) {\r\n pLight.specular = node.color.toColor3();\r\n pLight.diffuse = node.color.toColor3();\r\n }\r\n if (changes.has('_parent')) {\r\n if (node._parent) {\r\n pLight.parent = this.viewer.getRendererNode(node._parent);\r\n } else {\r\n pLight.parent = null;\r\n }\r\n }\r\n\r\n let lightRotation: Quaternion;\r\n if (node.useEuler) {\r\n lightRotation = Quaternion.FromEulerAngles(\r\n toRadians(node.eulerRotation.x),\r\n toRadians(node.eulerRotation.y),\r\n toRadians(node.eulerRotation.z)\r\n );\r\n } else {\r\n lightRotation = Quaternion.RotationAxis(node.rotationAxis.toVec3(), toRadians(node.rotationAngle));\r\n }\r\n const rotationChanged = changes.hasAny('useEuler', 'eulerRotation', 'rotationAxis', 'rotationAngle');\r\n\r\n if (node instanceof PointLightNode) {\r\n const pLight = light as PointLight;\r\n if (changes.has('position')) {\r\n pLight.position = node.position.toVec3();\r\n }\r\n } else if (node instanceof HemisphericLightNode) {\r\n const pLight = light as HemisphericLight;\r\n if (changes.has('direction') || rotationChanged) {\r\n pLight.direction = this.rotateDirection(node.direction.toVec3(), lightRotation);\r\n }\r\n if (changes.has('groundColor')) {\r\n pLight.groundColor = node.groundColor.toColor3();\r\n }\r\n } else if (node instanceof SpotLightNode) {\r\n const pLight = light as SpotLight;\r\n if (changes.has('position')) {\r\n pLight.position = node.position.toVec3();\r\n }\r\n if (changes.has('direction') || rotationChanged) {\r\n pLight.direction = this.rotateDirection(node.direction.toVec3(), lightRotation);\r\n }\r\n if (changes.has('coneAngle')) {\r\n pLight.angle = (node.coneAngle * Math.PI) / 180;\r\n }\r\n } else if (node instanceof DirectionalLightNode) {\r\n const pLight = light as DirectionalLight;\r\n // pLight.position = new Vector3(0, 5, 0);\r\n if (changes.has('direction') || rotationChanged) {\r\n pLight.direction = this.rotateDirection(node.direction.toVec3(), lightRotation);\r\n //for directional lights, we hardcode the position to be negative direction for shadows\r\n pLight.position = pLight.direction.clone().negate();\r\n }\r\n }\r\n\r\n if (node instanceof ShadowLightNode) {\r\n const shadowLightNode = node as ITrackedClass;\r\n const shadowLight = light as ShadowLight;\r\n\r\n let sg = this.shadowManager.get(shadowLight);\r\n let justCreated = false;\r\n if (changes.has('generatesShadows')) {\r\n if (shadowLightNode.generatesShadows) {\r\n sg = this.shadowManager.add(shadowLight, shadowLightNode.shadowResolution);\r\n shadowLight.shadowEnabled = true;\r\n justCreated = true;\r\n changes.set('shadowFilter', shadowLightNode.shadowFilter);\r\n changes.set('shadowQuality', shadowLightNode.shadowQuality);\r\n } else {\r\n if (sg) {\r\n shadowLight.shadowEnabled = false;\r\n this.shadowManager.remove(shadowLight);\r\n sg = undefined;\r\n }\r\n }\r\n }\r\n\r\n //if the resolution is changed, and the shadow generator wasn't just created, we need to recreate it\r\n if (changes.has('shadowResolution') && sg && !justCreated) {\r\n this.shadowManager.remove(shadowLight);\r\n sg = this.shadowManager.add(shadowLight, shadowLightNode.shadowResolution);\r\n changes.set('shadowFilter', shadowLightNode.shadowFilter);\r\n changes.set('shadowQuality', shadowLightNode.shadowQuality);\r\n changes.set('processTransparency', shadowLightNode.processTransparency);\r\n }\r\n\r\n if (sg) {\r\n if (changes.has('shadowFilter')) {\r\n sg.filter = ShadowManager.convertShadowFilter(shadowLightNode.shadowFilter);\r\n }\r\n\r\n if (changes.has('processTransparency')) {\r\n sg.transparencyShadow = shadowLightNode.processTransparency;\r\n sg.enableSoftTransparentShadow = shadowLightNode.processTransparency;\r\n }\r\n\r\n if (changes.has('shadowQuality')) {\r\n sg.filteringQuality = (ShadowGenerator as any)[\r\n `QUALITY_${shadowLightNode.shadowQuality.toUpperCase()}`\r\n ];\r\n }\r\n\r\n //need to update projection matrix if direction has been changed\r\n if (changes.hasAny('direction', 'position')) {\r\n this.shadowManager.clearProjectMatrixCache();\r\n }\r\n }\r\n }\r\n\r\n if (changes.has('_selected')) {\r\n if (!pLight.metadata) {\r\n pLight.metadata = {};\r\n }\r\n pLight.metadata.selected = node._selected;\r\n }\r\n this.viewer.updateLightWarnings();\r\n this.updatePreviewMesh(pLight);\r\n }\r\n\r\n delete3d(node: ITrackedClass) {\r\n if (this.babylonLightPreviewsMap) {\r\n const lightPositionGizmo = this.babylonLightPreviewsMap.get(node._dynamicId);\r\n if (!lightPositionGizmo) {\r\n throw Error('Light gizmo not found for light ' + node.id);\r\n }\r\n lightPositionGizmo.dispose();\r\n this.babylonLightPreviewsMap!.delete(node._dynamicId);\r\n }\r\n\r\n this.shadowManager.remove(this.get3d(node) as ShadowLight); //remove also disposes the shadow generator if it exists\r\n\r\n super.delete3d(node);\r\n this.viewer.updateLightWarnings();\r\n }\r\n\r\n protected updatePreviewMesh(light: Light) {\r\n let lightPositionGizmo = this.babylonLightPreviewsMap.get(light.id);\r\n const scene = this.gizmoLayer.utilityLayerScene;\r\n\r\n if (!lightPositionGizmo) {\r\n lightPositionGizmo = createLightVisualization(light, scene, Color3.Yellow());\r\n lightPositionGizmo.isPickable = false;\r\n this.babylonLightPreviewsMap.set(light.id, lightPositionGizmo);\r\n }\r\n\r\n if (this.showAllLightPreviews) {\r\n lightPositionGizmo.setEnabled(light.metadata.selected || light.isEnabled());\r\n } else {\r\n lightPositionGizmo.setEnabled(light.metadata.selected);\r\n }\r\n\r\n if (light instanceof HemisphericLight) {\r\n lightPositionGizmo.rotationQuaternion = rotationToVector(light.direction);\r\n } else if (light instanceof SpotLight) {\r\n lightPositionGizmo.position = this.getLightPosition(light);\r\n lightPositionGizmo.rotationQuaternion = rotationToVector(light.direction);\r\n lightPositionGizmo.scaling.set(\r\n Math.sin(light.angle / 2),\r\n Math.cos(light.angle / 2),\r\n Math.sin(light.angle / 2)\r\n );\r\n } else if (light instanceof DirectionalLight) {\r\n lightPositionGizmo.rotationQuaternion = rotationToVector(light.direction);\r\n } else if (light instanceof PointLight) {\r\n lightPositionGizmo.position = this.getLightPosition(light);\r\n }\r\n }\r\n\r\n protected rotateDirection(direction: Vector3, rotation: Quaternion) {\r\n direction.rotateByQuaternionToRef(rotation, direction);\r\n return direction;\r\n }\r\n\r\n protected getLightPosition(light: SpotLight | PointLight) {\r\n return light.parent\r\n ? Vector3.TransformCoordinates(light.position, light.parent.computeWorldMatrix(true).clone())\r\n : light.position;\r\n }\r\n}\r\n","import { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { MateNode } from '@models/classes/mates/mate-node';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass } from '@models/interfaces';\r\nimport { SyncService } from '@models/services/sync-service';\r\nimport { Renderer } from './renderer';\r\n\r\nexport class MateRenderer extends Renderer {\r\n modelToken = Token.MateNode;\r\n\r\n update3dInternal(node: ITrackedClass) {}\r\n delete3d(node: ITrackedClass) {\r\n SyncService.nodeDeletionProcessed(node);\r\n }\r\n}\r\n","import { serializeAsMeshReference, serializeAsVector3, SerializationHelper } from \"../Misc/decorators\";\r\nimport { RenderTargetTexture } from \"../Materials/Textures/renderTargetTexture\";\r\nimport { Matrix, Vector3 } from \"../Maths/math.vector\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { Nullable } from \"../types\";\r\nimport { AbstractScene } from \"../abstractScene\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport type { UniformBuffer } from \"../Materials/uniformBuffer\";\r\n\r\ndeclare module \"../abstractScene\" {\r\n export interface AbstractScene {\r\n /**\r\n * The list of reflection probes added to the scene\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/reflectionProbes\r\n */\r\n reflectionProbes: Array;\r\n\r\n /**\r\n * Removes the given reflection probe from this scene.\r\n * @param toRemove The reflection probe to remove\r\n * @returns The index of the removed reflection probe\r\n */\r\n removeReflectionProbe(toRemove: ReflectionProbe): number;\r\n\r\n /**\r\n * Adds the given reflection probe to this scene.\r\n * @param newReflectionProbe The reflection probe to add\r\n */\r\n addReflectionProbe(newReflectionProbe: ReflectionProbe): void;\r\n }\r\n}\r\n\r\nAbstractScene.prototype.removeReflectionProbe = function (toRemove: ReflectionProbe): number {\r\n if (!this.reflectionProbes) {\r\n return -1;\r\n }\r\n\r\n const index = this.reflectionProbes.indexOf(toRemove);\r\n if (index !== -1) {\r\n this.reflectionProbes.splice(index, 1);\r\n }\r\n\r\n return index;\r\n};\r\n\r\nAbstractScene.prototype.addReflectionProbe = function (newReflectionProbe: ReflectionProbe): void {\r\n if (!this.reflectionProbes) {\r\n this.reflectionProbes = [];\r\n }\r\n\r\n this.reflectionProbes.push(newReflectionProbe);\r\n};\r\n\r\n/**\r\n * Class used to generate realtime reflection / refraction cube textures\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/environment/reflectionProbes\r\n */\r\nexport class ReflectionProbe {\r\n private _scene: Scene;\r\n private _renderTargetTexture: RenderTargetTexture;\r\n private _projectionMatrix: Matrix;\r\n private _viewMatrix = Matrix.Identity();\r\n private _target = Vector3.Zero();\r\n private _add = Vector3.Zero();\r\n @serializeAsMeshReference()\r\n private _attachedMesh: Nullable;\r\n\r\n private _invertYAxis = false;\r\n private _sceneUBOs: UniformBuffer[];\r\n private _currentSceneUBO: UniformBuffer;\r\n\r\n /** Gets or sets probe position (center of the cube map) */\r\n @serializeAsVector3()\r\n public position = Vector3.Zero();\r\n\r\n /**\r\n * Gets or sets an object used to store user defined information for the reflection probe.\r\n */\r\n public metadata: any = null;\r\n\r\n /** @internal */\r\n public _parentContainer: Nullable = null;\r\n\r\n /**\r\n * Creates a new reflection probe\r\n * @param name defines the name of the probe\r\n * @param size defines the texture resolution (for each face)\r\n * @param scene defines the hosting scene\r\n * @param generateMipMaps defines if mip maps should be generated automatically (true by default)\r\n * @param useFloat defines if HDR data (float data) should be used to store colors (false by default)\r\n * @param linearSpace defines if the probe should be generated in linear space or not (false by default)\r\n */\r\n constructor(\r\n /** defines the name of the probe */\r\n public name: string,\r\n size: number,\r\n scene: Scene,\r\n generateMipMaps = true,\r\n useFloat = false,\r\n linearSpace = false\r\n ) {\r\n this._scene = scene;\r\n\r\n if (scene.getEngine().supportsUniformBuffers) {\r\n this._sceneUBOs = [];\r\n for (let i = 0; i < 6; ++i) {\r\n this._sceneUBOs.push(scene.createSceneUniformBuffer(`Scene for Reflection Probe (name \"${name}\") face #${i}`));\r\n }\r\n }\r\n\r\n // Create the scene field if not exist.\r\n if (!this._scene.reflectionProbes) {\r\n this._scene.reflectionProbes = new Array();\r\n }\r\n this._scene.reflectionProbes.push(this);\r\n\r\n let textureType = Constants.TEXTURETYPE_UNSIGNED_BYTE;\r\n if (useFloat) {\r\n const caps = this._scene.getEngine().getCaps();\r\n if (caps.textureHalfFloatRender) {\r\n textureType = Constants.TEXTURETYPE_HALF_FLOAT;\r\n } else if (caps.textureFloatRender) {\r\n textureType = Constants.TEXTURETYPE_FLOAT;\r\n }\r\n }\r\n this._renderTargetTexture = new RenderTargetTexture(name, size, scene, generateMipMaps, true, textureType, true);\r\n this._renderTargetTexture.gammaSpace = !linearSpace;\r\n this._renderTargetTexture.invertZ = scene.useRightHandedSystem;\r\n\r\n const useReverseDepthBuffer = scene.getEngine().useReverseDepthBuffer;\r\n\r\n this._renderTargetTexture.onBeforeRenderObservable.add((faceIndex: number) => {\r\n if (this._sceneUBOs) {\r\n scene.setSceneUniformBuffer(this._sceneUBOs[faceIndex]);\r\n scene.getSceneUniformBuffer().unbindEffect();\r\n }\r\n switch (faceIndex) {\r\n case 0:\r\n this._add.copyFromFloats(1, 0, 0);\r\n break;\r\n case 1:\r\n this._add.copyFromFloats(-1, 0, 0);\r\n break;\r\n case 2:\r\n this._add.copyFromFloats(0, this._invertYAxis ? 1 : -1, 0);\r\n break;\r\n case 3:\r\n this._add.copyFromFloats(0, this._invertYAxis ? -1 : 1, 0);\r\n break;\r\n case 4:\r\n this._add.copyFromFloats(0, 0, scene.useRightHandedSystem ? -1 : 1);\r\n break;\r\n case 5:\r\n this._add.copyFromFloats(0, 0, scene.useRightHandedSystem ? 1 : -1);\r\n break;\r\n }\r\n\r\n if (this._attachedMesh) {\r\n this.position.copyFrom(this._attachedMesh.getAbsolutePosition());\r\n }\r\n\r\n this.position.addToRef(this._add, this._target);\r\n\r\n const lookAtFunction = scene.useRightHandedSystem ? Matrix.LookAtRHToRef : Matrix.LookAtLHToRef;\r\n const perspectiveFunction = scene.useRightHandedSystem ? Matrix.PerspectiveFovRH : Matrix.PerspectiveFovLH;\r\n\r\n lookAtFunction(this.position, this._target, Vector3.Up(), this._viewMatrix);\r\n\r\n if (scene.activeCamera) {\r\n this._projectionMatrix = perspectiveFunction(\r\n Math.PI / 2,\r\n 1,\r\n useReverseDepthBuffer ? scene.activeCamera.maxZ : scene.activeCamera.minZ,\r\n useReverseDepthBuffer ? scene.activeCamera.minZ : scene.activeCamera.maxZ,\r\n this._scene.getEngine().isNDCHalfZRange\r\n );\r\n scene.setTransformMatrix(this._viewMatrix, this._projectionMatrix);\r\n if (scene.activeCamera.isRigCamera && !this._renderTargetTexture.activeCamera) {\r\n this._renderTargetTexture.activeCamera = scene.activeCamera.rigParent || null;\r\n }\r\n }\r\n scene._forcedViewPosition = this.position;\r\n });\r\n\r\n let currentApplyByPostProcess: boolean;\r\n\r\n this._renderTargetTexture.onBeforeBindObservable.add(() => {\r\n this._currentSceneUBO = scene.getSceneUniformBuffer();\r\n scene.getEngine()._debugPushGroup?.(`reflection probe generation for ${name}`, 1);\r\n currentApplyByPostProcess = this._scene.imageProcessingConfiguration.applyByPostProcess;\r\n if (linearSpace) {\r\n scene.imageProcessingConfiguration.applyByPostProcess = true;\r\n }\r\n });\r\n\r\n this._renderTargetTexture.onAfterUnbindObservable.add(() => {\r\n scene.imageProcessingConfiguration.applyByPostProcess = currentApplyByPostProcess;\r\n scene._forcedViewPosition = null;\r\n if (this._sceneUBOs) {\r\n scene.setSceneUniformBuffer(this._currentSceneUBO);\r\n }\r\n scene.updateTransformMatrix(true);\r\n scene.getEngine()._debugPopGroup?.(1);\r\n });\r\n }\r\n\r\n /** Gets or sets the number of samples to use for multi-sampling (0 by default). Required WebGL2 */\r\n public get samples(): number {\r\n return this._renderTargetTexture.samples;\r\n }\r\n\r\n public set samples(value: number) {\r\n this._renderTargetTexture.samples = value;\r\n }\r\n\r\n /** Gets or sets the refresh rate to use (on every frame by default) */\r\n public get refreshRate(): number {\r\n return this._renderTargetTexture.refreshRate;\r\n }\r\n\r\n public set refreshRate(value: number) {\r\n this._renderTargetTexture.refreshRate = value;\r\n }\r\n\r\n /**\r\n * Gets the hosting scene\r\n * @returns a Scene\r\n */\r\n public getScene(): Scene {\r\n return this._scene;\r\n }\r\n\r\n /** Gets the internal CubeTexture used to render to */\r\n public get cubeTexture(): RenderTargetTexture {\r\n return this._renderTargetTexture;\r\n }\r\n\r\n /** Gets the list of meshes to render */\r\n public get renderList(): Nullable {\r\n return this._renderTargetTexture.renderList;\r\n }\r\n\r\n /**\r\n * Attach the probe to a specific mesh (Rendering will be done from attached mesh's position)\r\n * @param mesh defines the mesh to attach to\r\n */\r\n public attachToMesh(mesh: Nullable): void {\r\n this._attachedMesh = mesh;\r\n }\r\n\r\n /**\r\n * Specifies whether or not the stencil and depth buffer are cleared between two rendering groups\r\n * @param renderingGroupId The rendering group id corresponding to its index\r\n * @param autoClearDepthStencil Automatically clears depth and stencil between groups if true.\r\n */\r\n public setRenderingAutoClearDepthStencil(renderingGroupId: number, autoClearDepthStencil: boolean): void {\r\n this._renderTargetTexture.setRenderingAutoClearDepthStencil(renderingGroupId, autoClearDepthStencil);\r\n }\r\n\r\n /**\r\n * Clean all associated resources\r\n */\r\n public dispose() {\r\n const index = this._scene.reflectionProbes.indexOf(this);\r\n\r\n if (index !== -1) {\r\n // Remove from the scene if found\r\n this._scene.reflectionProbes.splice(index, 1);\r\n }\r\n\r\n if (this._parentContainer) {\r\n const index = this._parentContainer.reflectionProbes.indexOf(this);\r\n if (index > -1) {\r\n this._parentContainer.reflectionProbes.splice(index, 1);\r\n }\r\n this._parentContainer = null;\r\n }\r\n\r\n if (this._renderTargetTexture) {\r\n this._renderTargetTexture.dispose();\r\n (this._renderTargetTexture) = null;\r\n }\r\n\r\n if (this._sceneUBOs) {\r\n for (const ubo of this._sceneUBOs) {\r\n ubo.dispose();\r\n }\r\n this._sceneUBOs = [];\r\n }\r\n }\r\n\r\n /**\r\n * Converts the reflection probe information to a readable string for debug purpose.\r\n * @param fullDetails Supports for multiple levels of logging within scene loading\r\n * @returns the human readable reflection probe info\r\n */\r\n public toString(fullDetails?: boolean): string {\r\n let ret = \"Name: \" + this.name;\r\n\r\n if (fullDetails) {\r\n ret += \", position: \" + this.position.toString();\r\n\r\n if (this._attachedMesh) {\r\n ret += \", attached mesh: \" + this._attachedMesh.name;\r\n }\r\n }\r\n\r\n return ret;\r\n }\r\n\r\n /**\r\n * Get the class name of the refection probe.\r\n * @returns \"ReflectionProbe\"\r\n */\r\n public getClassName(): string {\r\n return \"ReflectionProbe\";\r\n }\r\n\r\n /**\r\n * Serialize the reflection probe to a JSON representation we can easily use in the respective Parse function.\r\n * @returns The JSON representation of the texture\r\n */\r\n public serialize(): any {\r\n const serializationObject = SerializationHelper.Serialize(this, this._renderTargetTexture.serialize());\r\n serializationObject.isReflectionProbe = true;\r\n serializationObject.metadata = this.metadata;\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Parse the JSON representation of a reflection probe in order to recreate the reflection probe in the given scene.\r\n * @param parsedReflectionProbe Define the JSON representation of the reflection probe\r\n * @param scene Define the scene the parsed reflection probe should be instantiated in\r\n * @param rootUrl Define the root url of the parsing sequence in the case of relative dependencies\r\n * @returns The parsed reflection probe if successful\r\n */\r\n public static Parse(parsedReflectionProbe: any, scene: Scene, rootUrl: string): Nullable {\r\n let reflectionProbe: Nullable = null;\r\n if (scene.reflectionProbes) {\r\n for (let index = 0; index < scene.reflectionProbes.length; index++) {\r\n const rp = scene.reflectionProbes[index];\r\n if (rp.name === parsedReflectionProbe.name) {\r\n reflectionProbe = rp;\r\n break;\r\n }\r\n }\r\n }\r\n\r\n reflectionProbe = SerializationHelper.Parse(\r\n () => reflectionProbe || new ReflectionProbe(parsedReflectionProbe.name, parsedReflectionProbe.renderTargetSize, scene, parsedReflectionProbe._generateMipMaps),\r\n parsedReflectionProbe,\r\n scene,\r\n rootUrl\r\n );\r\n reflectionProbe.cubeTexture._waitingRenderList = parsedReflectionProbe.renderList;\r\n\r\n if (parsedReflectionProbe._attachedMesh) {\r\n reflectionProbe.attachToMesh(scene.getMeshById(parsedReflectionProbe._attachedMesh));\r\n }\r\n\r\n if (parsedReflectionProbe.metadata) {\r\n reflectionProbe.metadata = parsedReflectionProbe.metadata;\r\n }\r\n\r\n return reflectionProbe;\r\n }\r\n}\r\n","import { Constants } from '@babylonjs/core/Engines/constants';\r\nimport { Engine } from '@babylonjs/core/Engines/engine';\r\nimport { PBRBaseSimpleMaterial } from '@babylonjs/core/Materials/PBR/pbrBaseSimpleMaterial';\r\nimport { PBRMaterial } from '@babylonjs/core/Materials/PBR/pbrMaterial';\r\nimport { BaseTexture } from '@babylonjs/core/Materials/Textures/baseTexture';\r\nimport { MirrorTexture } from '@babylonjs/core/Materials/Textures/mirrorTexture';\r\nimport { RefractionTexture } from '@babylonjs/core/Materials/Textures/refractionTexture';\r\nimport { Material } from '@babylonjs/core/Materials/material';\r\nimport { Color3, Plane, Vector2, Vector3 } from '@babylonjs/core/Maths';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { ReflectionProbe } from '@babylonjs/core/Probes/reflectionProbe';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { Connector, SpaceNode } from '@models/classes';\r\nimport { instanceOf } from '@models/classes/decorators/model-utils';\r\nimport { eMaterialWorkflow, eReflectionType, eRenderMode, eTextureChannel } from '@models/classes/enums';\r\nimport { Kb3dManager } from '@models/classes/kb3d-manager';\r\nimport { AdvancedMaterialNode } from '@models/classes/materials/advanced-material-node';\r\nimport { MaterialNode } from '@models/classes/materials/material-node';\r\nimport { MeshNode } from '@models/classes/meshes/mesh-node';\r\nimport { SceneNode } from '@models/classes/scene-node';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass, isTrackedClass } from '@models/interfaces/tracked-class';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { Observable } from 'rxjs';\r\nimport { map, shareReplay } from 'rxjs/operators';\r\nimport { Renderer } from './renderer';\r\nimport { TextureStackRenderer } from './texture-stack';\r\n\r\n// Performance Code from Babylon 5.0. Remove once upgrade to Babylon 5.0 is merged\r\nObject.defineProperty(BaseTexture.prototype, 'hasAlpha', {\r\n get: function () {\r\n return this._hasAlpha;\r\n },\r\n set: function (value) {\r\n const self = this;\r\n if (this._hasAlpha === value) {\r\n return;\r\n }\r\n this._hasAlpha = value;\r\n if (this._scene) {\r\n this._scene.markAllMaterialsAsDirty(Constants.MATERIAL_TextureDirtyFlag, function (mat: Material) {\r\n return mat.hasTexture(self);\r\n });\r\n }\r\n },\r\n enumerable: false,\r\n configurable: true,\r\n});\r\n\r\nexport class MaterialRenderer extends Renderer {\r\n private _globalMaterialCache = new Map>();\r\n private _globalMaterialManager: Kb3dManager;\r\n private _defaultMaterial: PBRMaterial;\r\n private orphanedMeshes = new Map();\r\n\r\n //keyed by the connector id of which the reflection probe is attached to\r\n private reflectionProbeMap = new Map();\r\n public reflectedMeshes = new Map();\r\n //key is the ConnectorId of the reflection probe, String[] is the ids of the meshes using the material;\r\n public meshesUsingReflectionMaterial = new Map();\r\n\r\n get defaultMaterial() {\r\n return this._defaultMaterial;\r\n }\r\n\r\n constructor(viewer: KbViewer, scene: Scene, private textureRenderer: TextureStackRenderer) {\r\n super(viewer, scene);\r\n\r\n this._defaultMaterial = new PBRMaterial('default material', scene);\r\n this._defaultMaterial.albedoColor = new Color3(0.5, 0.5, 0.5);\r\n this._defaultMaterial.alphaMode = Engine.ALPHA_COMBINE;\r\n this._defaultMaterial.metallic = 0;\r\n this._defaultMaterial.roughness = 1;\r\n }\r\n modelToken = Token.MaterialNode;\r\n\r\n _renderMode: eRenderMode;\r\n get renderMode() {\r\n return this._renderMode;\r\n }\r\n set renderMode(mode: eRenderMode) {\r\n this._renderMode = mode;\r\n this.babylonMap.forEach(material => {\r\n this.setRenderModeOnMaterial(material, mode);\r\n });\r\n this.setRenderModeOnMaterial(this._defaultMaterial, mode);\r\n }\r\n\r\n getAllDefinedMaterials() {\r\n return Array.from(this.babylonMap.values());\r\n }\r\n\r\n getGlobalMaterial(dynamicId: string) {\r\n if (!this._globalMaterialManager) {\r\n return undefined;\r\n } else {\r\n return this._globalMaterialManager.getById(dynamicId);\r\n }\r\n }\r\n\r\n delete3d(node: ITrackedClass) {\r\n this.textureRenderer.materialDeleted(node);\r\n const material = this.get3d(node);\r\n if (material instanceof PBRMaterial) {\r\n const orphanedMeshes = material.getBindedMeshes();\r\n this.orphanedMeshes.set(node.id, orphanedMeshes);\r\n }\r\n this.viewer.materialWarningsNeedsUpdate.next();\r\n return super.delete3d(node);\r\n }\r\n\r\n loadGlobalMaterial(materialId: string, managerServices: Map) {\r\n if (!this._globalMaterialManager) {\r\n this._globalMaterialManager = new Kb3dManager({ services: managerServices });\r\n }\r\n const matService = this._globalMaterialManager.getService(Token.MaterialService);\r\n\r\n if (this._globalMaterialCache.has(materialId)) {\r\n return this._globalMaterialCache.get(materialId) as Observable;\r\n } else {\r\n const obs = matService\r\n .getMaterialById(materialId)\r\n .pipe(\r\n map(imaterial => {\r\n if (imaterial) {\r\n let matNode = this._globalMaterialManager.getById(materialId);\r\n const newMatNode = this._globalMaterialManager.deserialize(imaterial.data);\r\n // If the material has already been added to the scene, just update it.\r\n if (matNode) {\r\n this._globalMaterialManager.assign(matNode, newMatNode);\r\n // We have to re-initialize the material to reload the texture\r\n if (matNode._initialized) {\r\n matNode._initialized = false;\r\n matNode.init();\r\n }\r\n } else {\r\n matNode = newMatNode;\r\n matNode._dynamicId = matNode.id; // Each global material is unique\r\n this._globalMaterialManager.register(matNode);\r\n\r\n //now create the material in babylon by rendering it\r\n this.update3d(matNode as ITrackedClass);\r\n }\r\n\r\n return matNode;\r\n }\r\n return undefined;\r\n })\r\n )\r\n .pipe(shareReplay(1));\r\n\r\n this._globalMaterialCache.set(materialId, obs);\r\n return obs;\r\n }\r\n }\r\n\r\n update3dInternal(node: ITrackedClass) {\r\n let material = this.get3d(node);\r\n let _changes = node.changes;\r\n\r\n // since the texture are nested in the material we have to refresh the texture in the material individually\r\n if (instanceOf(node, Token.AdvancedMaterialNode)) {\r\n const _node = node as ITrackedClass;\r\n let changes = _changes as typeof _node.changes;\r\n\r\n let pbrAdvMaterial = material as PBRMaterial | undefined;\r\n if (!pbrAdvMaterial) {\r\n material = pbrAdvMaterial = new PBRMaterial(node._dynamicId, this.scene);\r\n material.name = node.name;\r\n material.id = node._dynamicId;\r\n\r\n const meshesToBind = this.orphanedMeshes.get(node.id);\r\n if (meshesToBind) {\r\n const _pbrAdvMaterial = pbrAdvMaterial;\r\n meshesToBind.forEach(mesh => {\r\n mesh.material = _pbrAdvMaterial;\r\n });\r\n }\r\n\r\n changes = _changes = new Map(Object.entries(node)) as any;\r\n changes.set('textures', node.textures);\r\n\r\n /** hardcode the alphaMode because the material always overrides the engine setting and we need proper alpha for dragging effects */\r\n material.alphaMode = Engine.ALPHA_COMBINE;\r\n material.useParallax = false;\r\n // Logarithmic depth is a good concept but is buggy. Maybe try again later.\r\n // material.useLogarithmicDepth = true;\r\n material.useLightmapAsShadowmap = true;\r\n material.enableSpecularAntiAliasing = true;\r\n material.forceDepthWrite = true;\r\n pbrAdvMaterial.transparencyMode = Material.MATERIAL_ALPHABLEND;\r\n\r\n material.useMetallnessFromMetallicTextureBlue = false;\r\n material.useMicroSurfaceFromReflectivityMapAlpha = false;\r\n material.useRoughnessFromMetallicTextureGreen = false;\r\n\r\n this.setRenderModeOnMaterial(material, this.renderMode);\r\n this.set3d(node, pbrAdvMaterial);\r\n }\r\n\r\n if (changes.has('probePositionTargetId')) {\r\n this.updateReflectionProbes();\r\n }\r\n\r\n if (changes.hasAny('probePositionTargetId', 'subsurfaceReflectionEnabled', 'reflectionType')) {\r\n //if (node.reflectionType === eReflectionType.probe) {\r\n if (!node.probePositionTargetId || !node.subsurfaceReflectionEnabled) {\r\n if (pbrAdvMaterial) {\r\n pbrAdvMaterial.reflectionTexture = null;\r\n }\r\n } else {\r\n if (node.reflectionType === eReflectionType.probe) {\r\n if (node.probePositionTargetId && pbrAdvMaterial) {\r\n let reflectionProbe = this.reflectionProbeMap.get(node.probePositionTargetId);\r\n if (!reflectionProbe || reflectionProbe instanceof MirrorTexture) {\r\n reflectionProbe = new ReflectionProbe(node.probePositionTargetId, 512, this.scene);\r\n this.reflectionProbeMap.set(node.probePositionTargetId, reflectionProbe);\r\n const exceptions = this.meshesUsingReflectionMaterial.get(node.probePositionTargetId);\r\n this.reflectedMeshes.forEach(mesh => {\r\n if (reflectionProbe && reflectionProbe.renderList) {\r\n if (!exceptions?.contains(mesh.id)) {\r\n reflectionProbe.renderList?.push(mesh);\r\n }\r\n }\r\n });\r\n const sceneBackground = this.scene.meshes.find(obj => {\r\n return obj.id === 'hdrSkyBox';\r\n });\r\n if (reflectionProbe && reflectionProbe.renderList && sceneBackground) {\r\n reflectionProbe.renderList.push(sceneBackground);\r\n }\r\n }\r\n const connector = this.viewer.getNode(node.probePositionTargetId);\r\n if (connector instanceof Connector) {\r\n const connectorMesh = this.viewer.getRendererNode(connector);\r\n reflectionProbe.attachToMesh(connectorMesh as AbstractMesh);\r\n }\r\n if (pbrAdvMaterial) {\r\n reflectionProbe.cubeTexture.invertZ = false;\r\n pbrAdvMaterial.reflectionTexture = reflectionProbe.cubeTexture;\r\n }\r\n }\r\n } else {\r\n if (node.probePositionTargetId && pbrAdvMaterial) {\r\n let reflectionProbe = this.reflectionProbeMap.get(node.probePositionTargetId);\r\n let noPlane = false;\r\n if (reflectionProbe && reflectionProbe instanceof MirrorTexture) {\r\n if (!reflectionProbe.mirrorPlane) {\r\n noPlane = true;\r\n }\r\n }\r\n if (!reflectionProbe || reflectionProbe instanceof MirrorTexture || noPlane) {\r\n reflectionProbe = new ReflectionProbe(node.probePositionTargetId, 512, this.scene);\r\n this.reflectionProbeMap.set(node.probePositionTargetId, reflectionProbe);\r\n const exceptions = this.meshesUsingReflectionMaterial.get(node.probePositionTargetId);\r\n this.reflectedMeshes.forEach(mesh => {\r\n if (reflectionProbe && reflectionProbe.renderList) {\r\n if (!exceptions?.contains(mesh.id)) {\r\n reflectionProbe.renderList?.push(mesh);\r\n }\r\n }\r\n });\r\n const sceneBackground = this.scene.meshes.find(obj => {\r\n return obj.id === 'hdrSkyBox';\r\n });\r\n if (reflectionProbe && reflectionProbe.renderList && sceneBackground) {\r\n reflectionProbe.renderList.push(sceneBackground);\r\n }\r\n }\r\n const mirrorTexture = new MirrorTexture('mirrorTexture', 1024, this.scene);\r\n const connector = this.viewer.getNode(node.probePositionTargetId);\r\n let position = new Vector3(0, 0, 0);\r\n if (\r\n connector &&\r\n connector instanceof Connector &&\r\n connector._parent &&\r\n connector._parent instanceof SpaceNode\r\n ) {\r\n const connectorMesh = this.viewer.getRendererNode(connector._parent);\r\n\r\n if (connectorMesh) {\r\n const normal = Vector3.TransformCoordinates(\r\n connector.direction.toVec3().scale(-1),\r\n connectorMesh.getWorldMatrix().getRotationMatrix()\r\n );\r\n position = connectorMesh.absolutePosition;\r\n const reflector = Plane.FromPositionAndNormal(position, normal);\r\n mirrorTexture.mirrorPlane = reflector;\r\n mirrorTexture.renderList = reflectionProbe.renderList;\r\n }\r\n }\r\n if (pbrAdvMaterial) {\r\n mirrorTexture.invertZ = false;\r\n mirrorTexture.adaptiveBlurKernel = node.roughness * 100;\r\n pbrAdvMaterial.reflectionTexture = mirrorTexture;\r\n }\r\n }\r\n }\r\n }\r\n }\r\n if (changes.has('hasTransparency')) {\r\n pbrAdvMaterial.useAlphaFromAlbedoTexture = node.hasTransparency;\r\n this.textureRenderer.setTransparency(node, node.hasTransparency);\r\n }\r\n\r\n if (changes.has('directIntensity')) pbrAdvMaterial.directIntensity = node.directIntensity;\r\n if (changes.has('emissiveIntensity')) pbrAdvMaterial.emissiveIntensity = node.emissiveIntensity;\r\n if (changes.has('environmentIntensity')) pbrAdvMaterial.environmentIntensity = node.environmentIntensity;\r\n if (changes.has('specularIntensity')) pbrAdvMaterial.specularIntensity = node.specularIntensity;\r\n if (changes.has('ambientTextureStrength')) {\r\n pbrAdvMaterial.ambientTextureStrength = node.ambientTextureStrength || 1;\r\n }\r\n\r\n if (changes.has('ambientColor')) {\r\n pbrAdvMaterial.ambientColor = new Color3(\r\n node.ambientColor.r,\r\n node.ambientColor.g,\r\n node.ambientColor.b\r\n ).toLinearSpace();\r\n }\r\n if (changes.has('albedoColor')) {\r\n pbrAdvMaterial.albedoColor = new Color3(\r\n node.albedoColor.r,\r\n node.albedoColor.g,\r\n node.albedoColor.b\r\n ).toLinearSpace();\r\n }\r\n if (changes.has('specularColor')) {\r\n pbrAdvMaterial.reflectivityColor = new Color3(\r\n node.specularColor.r,\r\n node.specularColor.g,\r\n node.specularColor.b\r\n ).toLinearSpace();\r\n }\r\n if (changes.has('reflectionColor')) {\r\n pbrAdvMaterial.reflectionColor = new Color3(\r\n node.reflectionColor.r,\r\n node.reflectionColor.g,\r\n node.reflectionColor.b\r\n ).toLinearSpace();\r\n }\r\n if (changes.has('microSurface')) pbrAdvMaterial.microSurface = node.microSurface;\r\n\r\n if (changes.has('useSpecularOverAlpha')) pbrAdvMaterial.useSpecularOverAlpha = node.useSpecularOverAlpha;\r\n if (changes.has('useRadianceOverAlpha')) pbrAdvMaterial.useRadianceOverAlpha = node.useRadianceOverAlpha;\r\n\r\n if (changes.has('textures') || changes.has('doubleSided') || changes.has('hasTransparency')) {\r\n // Only enable the material transparency handling if necessary\r\n if (\r\n (node.hasTransparency && node.doubleSided) ||\r\n node.textures.find(texture => texture.channel === eTextureChannel.opacity)\r\n ) {\r\n // These properties are necessary to ensure that partial transparency self-blending works correctly. There is a performace impact.\r\n pbrAdvMaterial.separateCullingPass = true;\r\n pbrAdvMaterial.transparencyMode = Material.MATERIAL_ALPHABLEND;\r\n\r\n // Refraction test\r\n // let renderTarget = new RefractionTexture(\"rt\", 1024, this.scene, true);\r\n // renderTarget.renderListPredicate = (mesh) => {\r\n // return mesh.name === 'hdrSkyBox'; // mesh.material === pbrAdvMaterial && mesh.isEnabled();\r\n // }\r\n // renderTarget.lodGenerationScale = 0.5;\r\n // renderTarget.depth = 1;\r\n // pbrAdvMaterial.refractionTexture = renderTarget;\r\n // renderTarget.refractionPlane = new Plane(0, 0, 0, 0);\r\n // pbrAdvMaterial.subSurface.useAlbedoToTintRefraction = true\r\n // pbrAdvMaterial.subSurface.indexOfRefraction = 1.5;\r\n // pbrAdvMaterial.subSurface.isTranslucencyEnabled = true;\r\n // pbrAdvMaterial.subSurface.linkRefractionWithTransparency = true;\r\n } else {\r\n pbrAdvMaterial.separateCullingPass = false;\r\n pbrAdvMaterial.transparencyMode = null;\r\n }\r\n }\r\n\r\n if (changes.has('anisotropicIntensity')) pbrAdvMaterial.anisotropy.intensity = node.anisotropicIntensity;\r\n if (changes.has('anisotropicDirection')) {\r\n pbrAdvMaterial.anisotropy.direction = new Vector2(\r\n node.anisotropicDirection.x,\r\n node.anisotropicDirection.y\r\n );\r\n }\r\n if (changes.has('anisotropyEnabled')) {\r\n pbrAdvMaterial.anisotropy.isEnabled = node.anisotropyEnabled;\r\n if (node.anisotropyEnabled) {\r\n const tangentTexture = node.textures.find(\r\n texture => texture.channel === eTextureChannel.anisotropyTangent\r\n );\r\n if (tangentTexture && isTrackedClass(tangentTexture)) {\r\n tangentTexture.isModified = true;\r\n }\r\n } else {\r\n pbrAdvMaterial.anisotropy.texture = null;\r\n }\r\n }\r\n\r\n if (changes.has('clearcoatIntensity')) pbrAdvMaterial.clearCoat.intensity = node.clearcoatIntensity;\r\n if (changes.has('clearcoatTintEnabled')) pbrAdvMaterial.clearCoat.isTintEnabled = node.clearcoatTintEnabled;\r\n if (changes.has('clearcoatTintIntensity')) {\r\n pbrAdvMaterial.clearCoat.tintThickness = node.clearcoatTintIntensity;\r\n }\r\n if (changes.has('clearcoatTintColor')) {\r\n pbrAdvMaterial.clearCoat.tintColor = node.clearcoatTintColor.toColor3();\r\n }\r\n if (changes.has('clearcoatEnabled')) {\r\n pbrAdvMaterial.clearCoat.isEnabled = node.clearcoatEnabled;\r\n if (node.clearcoatEnabled) {\r\n const clearcoatNormalTexture = node.textures.find(\r\n texture => texture.channel === eTextureChannel.clearCoatNormal\r\n );\r\n if (clearcoatNormalTexture && isTrackedClass(clearcoatNormalTexture)) {\r\n clearcoatNormalTexture.isModified = true;\r\n }\r\n } else {\r\n pbrAdvMaterial.clearCoat.bumpTexture = null;\r\n }\r\n }\r\n\r\n if (changes.has('subsurfaceThickness')) {\r\n pbrAdvMaterial.subSurface.minimumThickness = node.subsurfaceThickness;\r\n }\r\n if (changes.has('subsurfaceTintColor')) {\r\n pbrAdvMaterial.subSurface.tintColor = node.subsurfaceTintColor.toColor3();\r\n }\r\n if (changes.has('subsurfaceUseAlbedoForTintColor')) {\r\n pbrAdvMaterial.subSurface.useAlbedoToTintRefraction = node.subsurfaceUseAlbedoForTintColor;\r\n }\r\n if (changes.has('subsurfaceRefractionEnabled')) {\r\n pbrAdvMaterial.subSurface.isRefractionEnabled = node.subsurfaceRefractionEnabled;\r\n }\r\n if (changes.has('subsurfaceRefractionIntensity')) {\r\n pbrAdvMaterial.subSurface.refractionIntensity = node.subsurfaceRefractionIntensity;\r\n }\r\n if (changes.has('subsurfaceRefractionIndex')) {\r\n pbrAdvMaterial.subSurface.indexOfRefraction = node.subsurfaceRefractionIndex;\r\n }\r\n if (changes.has('subsurfaceRefractionLinkTransparency')) {\r\n pbrAdvMaterial.subSurface.linkRefractionWithTransparency = node.subsurfaceRefractionLinkTransparency;\r\n }\r\n if (changes.has('subsurfaceTranslucencyEnabled')) {\r\n pbrAdvMaterial.subSurface.isTranslucencyEnabled = node.subsurfaceTranslucencyEnabled;\r\n }\r\n if (changes.has('subsurfaceTranslucencyIntensity')) {\r\n pbrAdvMaterial.subSurface.translucencyIntensity = node.subsurfaceTranslucencyIntensity;\r\n }\r\n if (\r\n changes.hasAny('subsurfaceRefractionEnabled', 'subsurfaceTranslucencyEnabled', 'probePositionTargetId')\r\n ) {\r\n if (!node.probePositionTargetId || !node.subsurfaceRefractionEnabled) {\r\n pbrAdvMaterial.refractionTexture = null;\r\n } else {\r\n let refractionProbe = this.reflectionProbeMap.get(node.probePositionTargetId);\r\n if (!refractionProbe) {\r\n refractionProbe = new ReflectionProbe(node.probePositionTargetId, 512, this.scene);\r\n\r\n const connector = this.viewer.getNode(node.probePositionTargetId);\r\n if (connector instanceof Connector) {\r\n const connectorMesh = this.viewer.getRendererNode(connector);\r\n refractionProbe.attachToMesh(connectorMesh as AbstractMesh);\r\n this.reflectionProbeMap.set(node.probePositionTargetId, refractionProbe);\r\n }\r\n const exceptions = this.meshesUsingReflectionMaterial.get(node.probePositionTargetId);\r\n this.reflectedMeshes.forEach(mesh => {\r\n if (refractionProbe && refractionProbe.renderList) {\r\n if (!exceptions?.contains(mesh.id)) {\r\n refractionProbe.renderList?.push(mesh);\r\n }\r\n }\r\n });\r\n const sceneBackground = this.scene.meshes.find(obj => {\r\n return obj.id === 'hdrSkyBox';\r\n });\r\n if (refractionProbe && refractionProbe.renderList && sceneBackground) {\r\n refractionProbe.renderList.push(sceneBackground);\r\n }\r\n }\r\n const refractionTexture = new RefractionTexture('refractionTexture', 1024, this.scene);\r\n const connector = this.viewer.getNode(node.probePositionTargetId);\r\n let position = new Vector3(0, 0, 0);\r\n if (\r\n connector &&\r\n connector instanceof Connector &&\r\n connector._parent &&\r\n connector._parent instanceof SpaceNode\r\n ) {\r\n const connectorMesh = this.viewer.getRendererNode(connector._parent);\r\n\r\n if (connectorMesh) {\r\n const normal = Vector3.TransformCoordinates(\r\n connector.direction.toVec3(),\r\n connectorMesh.getWorldMatrix().getRotationMatrix()\r\n );\r\n position = connectorMesh.absolutePosition;\r\n const reflector = Plane.FromPositionAndNormal(position, normal);\r\n refractionTexture.refractionPlane = reflector;\r\n refractionTexture.renderList = refractionProbe.renderList;\r\n refractionTexture.depth = node.subsurfaceRefractionIndex;\r\n }\r\n }\r\n\r\n if (pbrAdvMaterial) {\r\n refractionTexture.invertZ = false;\r\n pbrAdvMaterial.refractionTexture = refractionTexture;\r\n }\r\n }\r\n if (node.subsurfaceRefractionEnabled || node.subsurfaceTranslucencyEnabled) {\r\n if (!pbrAdvMaterial.subSurface.thicknessTexture) {\r\n const subsurfaceThicknessTexture = node.textures.find(\r\n texture => texture.channel === eTextureChannel.subsurfaceThickness\r\n );\r\n if (subsurfaceThicknessTexture && isTrackedClass(subsurfaceThicknessTexture)) {\r\n subsurfaceThicknessTexture.isModified = true;\r\n }\r\n }\r\n } else {\r\n pbrAdvMaterial.subSurface.thicknessTexture = null;\r\n }\r\n }\r\n if (changes.has('workflowMode')) {\r\n if (node.workflowMode === eMaterialWorkflow.metallicRoughness) {\r\n pbrAdvMaterial.metallic = node.metallic;\r\n pbrAdvMaterial.roughness = node.roughness;\r\n } else {\r\n /* Remove the metallic texture if workflow is not metallic-roughness, otherwise metallic-roughness gets used */\r\n (pbrAdvMaterial.metallicTexture as any) = undefined;\r\n pbrAdvMaterial.metallic = null;\r\n pbrAdvMaterial.roughness = null;\r\n }\r\n if (pbrAdvMaterial.reflectionTexture && pbrAdvMaterial.reflectionTexture instanceof MirrorTexture) {\r\n if (node.roughness !== 0) {\r\n pbrAdvMaterial.reflectionTexture.adaptiveBlurKernel = node.roughness * 100;\r\n } else {\r\n //When setting the adaptiveBlurKernel to 0 we get very odd behavior.\r\n pbrAdvMaterial.reflectionTexture.adaptiveBlurKernel = 0.000001;\r\n }\r\n }\r\n } else if (node.workflowMode === eMaterialWorkflow.metallicRoughness) {\r\n if (changes.has('metallic')) pbrAdvMaterial.metallic = node.metallic;\r\n if (changes.has('roughness')) {\r\n pbrAdvMaterial.roughness = node.roughness;\r\n if (pbrAdvMaterial.reflectionTexture && pbrAdvMaterial.reflectionTexture instanceof MirrorTexture) {\r\n if (node.roughness !== 0) {\r\n pbrAdvMaterial.reflectionTexture.adaptiveBlurKernel = node.roughness * 100;\r\n } else {\r\n //When setting the adaptiveBlurKernel to 0 we get very odd behavior.\r\n pbrAdvMaterial.reflectionTexture.adaptiveBlurKernel = 0.000001;\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n if (material) {\r\n const changes = _changes;\r\n if (changes.has('alpha')) material.alpha = node.alpha;\r\n if (changes.has('zOffset')) material.zOffset = node.zOffset || 0;\r\n if (changes.has('doubleSided')) {\r\n material.backFaceCulling = !node.doubleSided;\r\n (material as PBRMaterial).twoSidedLighting = !!node.doubleSided;\r\n }\r\n if (changes.has('emissiveColor')) {\r\n if (node.emissiveColor) {\r\n material.emissiveColor = new Color3(\r\n node.emissiveColor.r,\r\n node.emissiveColor.g,\r\n node.emissiveColor.b\r\n );\r\n } else {\r\n material.emissiveColor = Color3.Black();\r\n }\r\n }\r\n\r\n // Wireframe mode for debugging geometry\r\n // if(true || node.viewMode === eMaterialViewMode.wireframe) {\r\n // material.wireframe = true;\r\n // material.emissiveColor = Color3.Red();\r\n // }\r\n }\r\n this.viewer.materialWarningsNeedsUpdate.next();\r\n }\r\n\r\n protected setRenderModeOnMaterial(material: Material, mode: eRenderMode) {\r\n if (mode === eRenderMode.wireframe) {\r\n material.pointsCloud = false;\r\n material.wireframe = true;\r\n } else if (mode === eRenderMode.hiddenEdges) {\r\n material.wireframe = false;\r\n material.pointsCloud = true;\r\n } else {\r\n material.pointsCloud = false;\r\n material.wireframe = false;\r\n }\r\n }\r\n\r\n protected getNumberOfMaterialRef(sceneNode: SceneNode, materialNode: MaterialNode): number {\r\n const allMeshes = sceneNode.filterByType(...Token.MeshTokens);\r\n let nrefs = 0;\r\n for (const m of allMeshes) {\r\n if (m.materialId == materialNode._dynamicId) nrefs++;\r\n }\r\n return nrefs;\r\n }\r\n\r\n public updateReflectionProbes() {\r\n this.meshesUsingReflectionMaterial.clear();\r\n this.viewer.sceneNode._manager?.getAllNodes().forEach(mesh => {\r\n if (mesh instanceof MeshNode) {\r\n if (mesh.materialId) {\r\n const material = mesh._manager?.getById(mesh.materialId);\r\n if (material && material instanceof MaterialNode && material.probePositionTargetId) {\r\n let exceptionList = this.meshesUsingReflectionMaterial.get(material.probePositionTargetId);\r\n if (!exceptionList) {\r\n exceptionList = [mesh._dynamicId];\r\n } else {\r\n exceptionList.push(mesh._dynamicId);\r\n }\r\n this.meshesUsingReflectionMaterial.set(material.probePositionTargetId, exceptionList);\r\n }\r\n }\r\n }\r\n });\r\n\r\n const sceneBackground = this.scene.meshes.filter(obj => {\r\n return obj.id === 'hdrSkyBox';\r\n });\r\n\r\n for (const keyValuePair of this.reflectionProbeMap) {\r\n const reflectionProbe = keyValuePair[1];\r\n const connectorId = keyValuePair[0];\r\n reflectionProbe.renderList?.clear();\r\n const exceptions = this.meshesUsingReflectionMaterial.get(connectorId);\r\n this.reflectedMeshes.forEach(mesh => {\r\n if (!exceptions?.contains(mesh.id)) {\r\n reflectionProbe.renderList?.push(mesh);\r\n }\r\n });\r\n if (reflectionProbe && reflectionProbe.renderList) {\r\n reflectionProbe.renderList.push(sceneBackground[0]);\r\n }\r\n }\r\n }\r\n}\r\n","import { Vector3 } from '@babylonjs/core/Maths/math.vector';\r\nimport { eAxis } from '@models/classes/enums';\r\nimport { SketchControlPoint } from '@models/classes/features/sketch-control-point';\r\nimport { Kb3dManager } from '@models/classes/kb3d-manager';\r\nimport { SketchNode } from '@models/classes/meshes/sketch-node';\r\nimport { SketchPath } from '@models/classes/meshes/sketch-node-path';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { Token } from '@models/classes/token';\r\nimport { appendOrIncrementNumber } from '@models/classes/utilities';\r\nimport { earcut } from '@view/helpers/earcut';\r\nimport { SubmeshVertexData, getPolygonNormal } from '@view/helpers/geometry-helper';\r\n\r\nexport function parseSketchGeometry(vertexData: Partial | null) {\r\n const segments: Vector3[][] = [];\r\n const holes = new Map();\r\n\r\n if (vertexData && vertexData.positions && vertexData.paths) {\r\n const paths = vertexData.paths;\r\n const holeIndices = new Map();\r\n if (vertexData.holes) {\r\n for (let i = 0; i < vertexData.holes.length; i += 2) {\r\n holeIndices.set(vertexData.holes[i + 1], vertexData.holes[i]);\r\n }\r\n }\r\n\r\n let i = 0;\r\n let currentHole: null | number = null;\r\n while (i < paths.length) {\r\n const positions = vertexData.positions;\r\n const segmentPoints: Vector3[] = [];\r\n currentHole = holeIndices.has(i) ? holeIndices.get(i)! : null;\r\n\r\n for (; i < paths.length; i += 2) {\r\n segmentPoints.push(Vector3.FromArray(positions, paths[i] * 3));\r\n if (paths[i + 2] !== paths[i + 1]) {\r\n segmentPoints.push(Vector3.FromArray(positions, paths[i + 1] * 3));\r\n i += 2;\r\n break;\r\n }\r\n }\r\n\r\n if (segmentPoints.length > 1) {\r\n if (currentHole !== null) {\r\n let currentHolePoints = holes.get(currentHole);\r\n if (!currentHolePoints) {\r\n currentHolePoints = [];\r\n holes.set(currentHole, currentHolePoints);\r\n }\r\n currentHolePoints.push(segmentPoints);\r\n } else {\r\n segments.push(segmentPoints);\r\n }\r\n }\r\n }\r\n }\r\n\r\n return { segments, holes };\r\n}\r\n\r\nexport function createSketchPointForEditor(manager: Kb3dManager, path: SketchPath, insertIndex?: number) {\r\n if (!path._parent) {\r\n throw new Error('Cannot create a sketch point for a path that has no parent sketch');\r\n }\r\n const sketch = path._parent;\r\n\r\n const c: SketchControlPoint = manager.create(Token.SketchControlPoint);\r\n c.name = 'Control Point';\r\n while (path.controlPoints.find(ec => ec.name === c.name)) {\r\n c.name = appendOrIncrementNumber(c.name);\r\n }\r\n\r\n let previousControlPoint: SketchControlPoint | undefined;\r\n if (!insertIndex) {\r\n // If insert index is zero or undefined, determine the previous control point and use its position for the new control point\r\n if (path.controlPoints.length > 0 && insertIndex === undefined) {\r\n previousControlPoint = path.controlPoints[path.controlPoints.length - 1];\r\n } else {\r\n const currentPathIndex = sketch.paths.findIndex(p => p._dynamicId === path._dynamicId);\r\n if (currentPathIndex > 0) {\r\n const previousPath = sketch.paths[currentPathIndex - 1];\r\n if (previousPath.controlPoints.length > 0) {\r\n previousControlPoint = previousPath.controlPoints[previousPath.controlPoints.length - 1];\r\n }\r\n }\r\n }\r\n } else {\r\n previousControlPoint = path.controlPoints[insertIndex];\r\n }\r\n\r\n const sketchNormal = getSketchNormalVector3(sketch.normal);\r\n // determine the axis making up the sketch plane so we can make sure the direction vectors are on the sketch plane\r\n let axis1: Vector3;\r\n if (sketchNormal.equalsWithEpsilon(Vector3.Forward()) || sketchNormal.equalsWithEpsilon(Vector3.Backward())) {\r\n axis1 = Vector3.Up();\r\n } else {\r\n axis1 = Vector3.Cross(sketchNormal, Vector3.Forward());\r\n }\r\n const axis2 = Vector3.Cross(sketchNormal, axis1);\r\n\r\n if (previousControlPoint) {\r\n // if therre is a previous control point, project its direction onto the sketch plane\r\n c.position = previousControlPoint.position.clone();\r\n const rawDirection = previousControlPoint.direction.toVec3();\r\n const projectedDirection = axis1\r\n .scale(Vector3.Dot(axis1, rawDirection))\r\n .add(axis2.scale(Vector3.Dot(axis2, rawDirection)));\r\n c.direction = KbVector.FromVec3(projectedDirection);\r\n } else {\r\n // if this is the first control point, just pick a direction vector on the sketch plane\r\n c.direction = KbVector.FromVec3(axis1);\r\n }\r\n c.direction2 = c.direction.scale(-1);\r\n\r\n if (insertIndex === undefined) {\r\n path.controlPoints.push(c);\r\n } else {\r\n path.controlPoints.splice(insertIndex, 0, c);\r\n }\r\n for (const eC of path.controlPoints) {\r\n eC._expanded = false;\r\n }\r\n\r\n c._visible = true;\r\n c._expanded = true;\r\n\r\n return c;\r\n}\r\n\r\nexport function getSketchNormalVector3(normal: KbVector | eAxis) {\r\n let planeNormal: Vector3;\r\n switch (normal) {\r\n case eAxis.x:\r\n planeNormal = Vector3.Right();\r\n break;\r\n case eAxis.y:\r\n planeNormal = Vector3.Up();\r\n break;\r\n case eAxis.z:\r\n planeNormal = Vector3.Forward();\r\n break;\r\n default:\r\n planeNormal = normal.toVec3();\r\n }\r\n planeNormal.normalize();\r\n return planeNormal;\r\n}\r\n\r\nexport function projectOnNodeAxis(node: SketchNode, x: number, y: number) {\r\n switch (node.normal) {\r\n case eAxis.y:\r\n return new KbVector(x, 0, -y);\r\n case eAxis.z:\r\n return new KbVector(-x, y, 0);\r\n default:\r\n return new KbVector(0, y, x);\r\n }\r\n}\r\n\r\nexport function axisToVector3(axis: eAxis) {\r\n switch (axis) {\r\n case eAxis.y:\r\n return Vector3.Up();\r\n case eAxis.z:\r\n return Vector3.Forward();\r\n default:\r\n return Vector3.Right();\r\n }\r\n}\r\n\r\n/**\r\n * returns the vertices in the order of positionVectors in sequence, followed by holePositions in sequence!\r\n * Indices will have to be recalculated if the vertex data is replaced with the results of this function.\r\n */\r\nexport function fillSketchPolygon(\r\n positionVectors: Vector3[],\r\n flipNormals: boolean,\r\n sketchNativeNormal: Vector3,\r\n holePositions?: Vector3[][]\r\n) {\r\n const sketchNormal = getPolygonNormal(positionVectors);\r\n if (flipNormals) {\r\n sketchNormal.scaleInPlace(-1);\r\n }\r\n // const geometryNormal = sketchNormal.scale(-1);\r\n const segmentPositions: number[] = [];\r\n let segmentIndices: number[] = [];\r\n\r\n for (let i = 0; i < positionVectors.length; i++) {\r\n const point = positionVectors[i];\r\n segmentPositions.push(point.x, point.y, point.z);\r\n }\r\n\r\n const holes: number[] = [];\r\n if (holePositions) {\r\n for (let i = 0; i < holePositions.length; i++) {\r\n const hole = holePositions[i];\r\n holes.push(segmentPositions.length / 3);\r\n for (let i = 0; i < hole.length; i++) {\r\n const point = hole[i];\r\n segmentPositions.push(point.x, point.y, point.z);\r\n }\r\n }\r\n }\r\n\r\n if (Math.abs(Vector3.Dot(sketchNormal, Vector3.UpReadOnly)) > 0.95) {\r\n const flatPositions = new Float32Array((segmentPositions.length * 2) / 3);\r\n const xAxis = Vector3.Cross(sketchNormal, new Vector3(0, 0, 1));\r\n const yAxis = Vector3.Cross(sketchNormal, xAxis);\r\n\r\n const working = new Vector3();\r\n for (let i = 0; i < segmentPositions.length; i += 3) {\r\n working.set(segmentPositions[i], segmentPositions[i + 1], segmentPositions[i + 2]);\r\n flatPositions.set([Vector3.Dot(working, xAxis), Vector3.Dot(working, yAxis)], (i * 2) / 3);\r\n }\r\n const faceNormal = calculateFaceNormalFromFloats(flatPositions, 2);\r\n if (Vector3.Dot(faceNormal, Vector3.RightHandedForwardReadOnly) < 0) {\r\n for (let i = 0; i < flatPositions.length; i += 2) {\r\n const tmp = flatPositions[i];\r\n flatPositions[i] = flatPositions[i + 1];\r\n flatPositions[i + 1] = tmp;\r\n }\r\n // geometryNormal.scaleInPlace(-1);\r\n }\r\n segmentIndices = earcut(flatPositions, holes, 2);\r\n } else {\r\n segmentIndices = earcut(segmentPositions, holes, 3);\r\n\r\n // Flip the geometry if needed so that it faces the same direction as the faces because earcut isn't aware of the 3d space\r\n const faceNormal = calculateFaceNormal(positionVectors);\r\n // If the rotation handedness is opposite the primary sketch vector, then flip the sketch path\r\n if (Vector3.Dot(faceNormal, sketchNativeNormal) < 0) {\r\n for (let i = 0; i < segmentIndices.length; i += 3) {\r\n const tmp = segmentIndices[i];\r\n segmentIndices[i] = segmentIndices[i + 2];\r\n segmentIndices[i + 2] = tmp;\r\n }\r\n // geometryNormal.scaleInPlace(-1);\r\n }\r\n }\r\n\r\n const segmentNormals: number[] = [];\r\n const geometryNormal = calculateFaceNormal(positionVectors);\r\n if (flipNormals) {\r\n geometryNormal.scaleInPlace(-1);\r\n }\r\n let totalVectorCount = positionVectors.length;\r\n if (holePositions) {\r\n totalVectorCount += holePositions.flat().length;\r\n }\r\n for (let i = 0; i < totalVectorCount; i++) {\r\n segmentNormals.push(geometryNormal.x, geometryNormal.y, geometryNormal.z);\r\n }\r\n const segmentUvs: number[] = [];\r\n for (let i = 0; i < totalVectorCount; i++) {\r\n segmentUvs.push(0, 0);\r\n }\r\n\r\n return {\r\n normals: segmentNormals,\r\n positions: segmentPositions,\r\n indices: segmentIndices,\r\n uvs: segmentUvs,\r\n };\r\n}\r\n\r\nexport function calculateFaceNormal(positionVectors: Vector3[]) {\r\n const v1 = Vector3.Zero();\r\n const v2 = Vector3.Zero();\r\n const barycenter = Vector3.Zero();\r\n for (let i = 0; i < positionVectors.length; i++) {\r\n barycenter.addInPlace(positionVectors[i]);\r\n }\r\n barycenter.scaleInPlace(1 / positionVectors.length);\r\n\r\n const pathHandedness = Vector3.Zero();\r\n if (positionVectors.length > 1) {\r\n for (let iP = 0; iP < positionVectors.length - 1; iP++) {\r\n const point1 = positionVectors[iP];\r\n v1.copyFromFloats(point1.x, point1.y, point1.z);\r\n v1.subtractInPlace(barycenter);\r\n const point2 = positionVectors[iP + 1];\r\n v2.copyFromFloats(point2.x, point2.y, point2.z);\r\n v2.subtractInPlace(barycenter);\r\n pathHandedness.addInPlace(Vector3.Cross(v1, v2));\r\n }\r\n return pathHandedness.normalize();\r\n }\r\n return Vector3.Zero();\r\n}\r\n\r\nexport function calculateFaceNormalFromFloats(points: number[] | Float32Array, dim: number) {\r\n const v1 = Vector3.Zero();\r\n const v2 = Vector3.Zero();\r\n const barycenter = Vector3.Zero();\r\n for (let i = 0; i < points.length; i += dim) {\r\n barycenter.addInPlaceFromFloats(points[i], points[i + 1], dim === 3 ? points[i + 2] : 0);\r\n }\r\n barycenter.scaleInPlace(3 / points.length);\r\n\r\n const pathHandedness = Vector3.Zero();\r\n if (points.length > 3) {\r\n for (let i = 0; i < points.length - dim; i += dim) {\r\n v1.copyFromFloats(points[i], points[i + 1], dim === 3 ? points[i + 2] : 0);\r\n v1.subtractInPlace(barycenter);\r\n v2.copyFromFloats(points[i + dim], points[i + dim + 1], dim === 3 ? points[i + dim + 2] : 0);\r\n v2.subtractInPlace(barycenter);\r\n pathHandedness.addInPlace(Vector3.Cross(v1, v2));\r\n }\r\n return pathHandedness.normalize();\r\n }\r\n return Vector3.Zero();\r\n}\r\n","/*! https://mths.be/codepointat v0.2.0 by @mathias */\nif (!String.prototype.codePointAt) {\n\t(function() {\n\t\t'use strict'; // needed to support `apply`/`call` with `undefined`/`null`\n\t\tvar defineProperty = (function() {\n\t\t\t// IE 8 only supports `Object.defineProperty` on DOM elements\n\t\t\ttry {\n\t\t\t\tvar object = {};\n\t\t\t\tvar $defineProperty = Object.defineProperty;\n\t\t\t\tvar result = $defineProperty(object, object, object) && $defineProperty;\n\t\t\t} catch(error) {}\n\t\t\treturn result;\n\t\t}());\n\t\tvar codePointAt = function(position) {\n\t\t\tif (this == null) {\n\t\t\t\tthrow TypeError();\n\t\t\t}\n\t\t\tvar string = String(this);\n\t\t\tvar size = string.length;\n\t\t\t// `ToInteger`\n\t\t\tvar index = position ? Number(position) : 0;\n\t\t\tif (index != index) { // better `isNaN`\n\t\t\t\tindex = 0;\n\t\t\t}\n\t\t\t// Account for out-of-bounds indices:\n\t\t\tif (index < 0 || index >= size) {\n\t\t\t\treturn undefined;\n\t\t\t}\n\t\t\t// Get the first code unit\n\t\t\tvar first = string.charCodeAt(index);\n\t\t\tvar second;\n\t\t\tif ( // check if it’s the start of a surrogate pair\n\t\t\t\tfirst >= 0xD800 && first <= 0xDBFF && // high surrogate\n\t\t\t\tsize > index + 1 // there is a next code unit\n\t\t\t) {\n\t\t\t\tsecond = string.charCodeAt(index + 1);\n\t\t\t\tif (second >= 0xDC00 && second <= 0xDFFF) { // low surrogate\n\t\t\t\t\t// https://mathiasbynens.be/notes/javascript-encoding#surrogate-formulae\n\t\t\t\t\treturn (first - 0xD800) * 0x400 + second - 0xDC00 + 0x10000;\n\t\t\t\t}\n\t\t\t}\n\t\t\treturn first;\n\t\t};\n\t\tif (defineProperty) {\n\t\t\tdefineProperty(String.prototype, 'codePointAt', {\n\t\t\t\t'value': codePointAt,\n\t\t\t\t'configurable': true,\n\t\t\t\t'writable': true\n\t\t\t});\n\t\t} else {\n\t\t\tString.prototype.codePointAt = codePointAt;\n\t\t}\n\t}());\n}\n","var TINF_OK = 0;\nvar TINF_DATA_ERROR = -3;\n\nfunction Tree() {\n this.table = new Uint16Array(16); /* table of code length counts */\n this.trans = new Uint16Array(288); /* code -> symbol translation table */\n}\n\nfunction Data(source, dest) {\n this.source = source;\n this.sourceIndex = 0;\n this.tag = 0;\n this.bitcount = 0;\n \n this.dest = dest;\n this.destLen = 0;\n \n this.ltree = new Tree(); /* dynamic length/symbol tree */\n this.dtree = new Tree(); /* dynamic distance tree */\n}\n\n/* --------------------------------------------------- *\n * -- uninitialized global data (static structures) -- *\n * --------------------------------------------------- */\n\nvar sltree = new Tree();\nvar sdtree = new Tree();\n\n/* extra bits and base tables for length codes */\nvar length_bits = new Uint8Array(30);\nvar length_base = new Uint16Array(30);\n\n/* extra bits and base tables for distance codes */\nvar dist_bits = new Uint8Array(30);\nvar dist_base = new Uint16Array(30);\n\n/* special ordering of code length codes */\nvar clcidx = new Uint8Array([\n 16, 17, 18, 0, 8, 7, 9, 6,\n 10, 5, 11, 4, 12, 3, 13, 2,\n 14, 1, 15\n]);\n\n/* used by tinf_decode_trees, avoids allocations every call */\nvar code_tree = new Tree();\nvar lengths = new Uint8Array(288 + 32);\n\n/* ----------------------- *\n * -- utility functions -- *\n * ----------------------- */\n\n/* build extra bits and base tables */\nfunction tinf_build_bits_base(bits, base, delta, first) {\n var i, sum;\n\n /* build bits table */\n for (i = 0; i < delta; ++i) bits[i] = 0;\n for (i = 0; i < 30 - delta; ++i) bits[i + delta] = i / delta | 0;\n\n /* build base table */\n for (sum = first, i = 0; i < 30; ++i) {\n base[i] = sum;\n sum += 1 << bits[i];\n }\n}\n\n/* build the fixed huffman trees */\nfunction tinf_build_fixed_trees(lt, dt) {\n var i;\n\n /* build fixed length tree */\n for (i = 0; i < 7; ++i) lt.table[i] = 0;\n\n lt.table[7] = 24;\n lt.table[8] = 152;\n lt.table[9] = 112;\n\n for (i = 0; i < 24; ++i) lt.trans[i] = 256 + i;\n for (i = 0; i < 144; ++i) lt.trans[24 + i] = i;\n for (i = 0; i < 8; ++i) lt.trans[24 + 144 + i] = 280 + i;\n for (i = 0; i < 112; ++i) lt.trans[24 + 144 + 8 + i] = 144 + i;\n\n /* build fixed distance tree */\n for (i = 0; i < 5; ++i) dt.table[i] = 0;\n\n dt.table[5] = 32;\n\n for (i = 0; i < 32; ++i) dt.trans[i] = i;\n}\n\n/* given an array of code lengths, build a tree */\nvar offs = new Uint16Array(16);\n\nfunction tinf_build_tree(t, lengths, off, num) {\n var i, sum;\n\n /* clear code length count table */\n for (i = 0; i < 16; ++i) t.table[i] = 0;\n\n /* scan symbol lengths, and sum code length counts */\n for (i = 0; i < num; ++i) t.table[lengths[off + i]]++;\n\n t.table[0] = 0;\n\n /* compute offset table for distribution sort */\n for (sum = 0, i = 0; i < 16; ++i) {\n offs[i] = sum;\n sum += t.table[i];\n }\n\n /* create code->symbol translation table (symbols sorted by code) */\n for (i = 0; i < num; ++i) {\n if (lengths[off + i]) t.trans[offs[lengths[off + i]]++] = i;\n }\n}\n\n/* ---------------------- *\n * -- decode functions -- *\n * ---------------------- */\n\n/* get one bit from source stream */\nfunction tinf_getbit(d) {\n /* check if tag is empty */\n if (!d.bitcount--) {\n /* load next tag */\n d.tag = d.source[d.sourceIndex++];\n d.bitcount = 7;\n }\n\n /* shift bit out of tag */\n var bit = d.tag & 1;\n d.tag >>>= 1;\n\n return bit;\n}\n\n/* read a num bit value from a stream and add base */\nfunction tinf_read_bits(d, num, base) {\n if (!num)\n return base;\n\n while (d.bitcount < 24) {\n d.tag |= d.source[d.sourceIndex++] << d.bitcount;\n d.bitcount += 8;\n }\n\n var val = d.tag & (0xffff >>> (16 - num));\n d.tag >>>= num;\n d.bitcount -= num;\n return val + base;\n}\n\n/* given a data stream and a tree, decode a symbol */\nfunction tinf_decode_symbol(d, t) {\n while (d.bitcount < 24) {\n d.tag |= d.source[d.sourceIndex++] << d.bitcount;\n d.bitcount += 8;\n }\n \n var sum = 0, cur = 0, len = 0;\n var tag = d.tag;\n\n /* get more bits while code value is above sum */\n do {\n cur = 2 * cur + (tag & 1);\n tag >>>= 1;\n ++len;\n\n sum += t.table[len];\n cur -= t.table[len];\n } while (cur >= 0);\n \n d.tag = tag;\n d.bitcount -= len;\n\n return t.trans[sum + cur];\n}\n\n/* given a data stream, decode dynamic trees from it */\nfunction tinf_decode_trees(d, lt, dt) {\n var hlit, hdist, hclen;\n var i, num, length;\n\n /* get 5 bits HLIT (257-286) */\n hlit = tinf_read_bits(d, 5, 257);\n\n /* get 5 bits HDIST (1-32) */\n hdist = tinf_read_bits(d, 5, 1);\n\n /* get 4 bits HCLEN (4-19) */\n hclen = tinf_read_bits(d, 4, 4);\n\n for (i = 0; i < 19; ++i) lengths[i] = 0;\n\n /* read code lengths for code length alphabet */\n for (i = 0; i < hclen; ++i) {\n /* get 3 bits code length (0-7) */\n var clen = tinf_read_bits(d, 3, 0);\n lengths[clcidx[i]] = clen;\n }\n\n /* build code length tree */\n tinf_build_tree(code_tree, lengths, 0, 19);\n\n /* decode code lengths for the dynamic trees */\n for (num = 0; num < hlit + hdist;) {\n var sym = tinf_decode_symbol(d, code_tree);\n\n switch (sym) {\n case 16:\n /* copy previous code length 3-6 times (read 2 bits) */\n var prev = lengths[num - 1];\n for (length = tinf_read_bits(d, 2, 3); length; --length) {\n lengths[num++] = prev;\n }\n break;\n case 17:\n /* repeat code length 0 for 3-10 times (read 3 bits) */\n for (length = tinf_read_bits(d, 3, 3); length; --length) {\n lengths[num++] = 0;\n }\n break;\n case 18:\n /* repeat code length 0 for 11-138 times (read 7 bits) */\n for (length = tinf_read_bits(d, 7, 11); length; --length) {\n lengths[num++] = 0;\n }\n break;\n default:\n /* values 0-15 represent the actual code lengths */\n lengths[num++] = sym;\n break;\n }\n }\n\n /* build dynamic trees */\n tinf_build_tree(lt, lengths, 0, hlit);\n tinf_build_tree(dt, lengths, hlit, hdist);\n}\n\n/* ----------------------------- *\n * -- block inflate functions -- *\n * ----------------------------- */\n\n/* given a stream and two trees, inflate a block of data */\nfunction tinf_inflate_block_data(d, lt, dt) {\n while (1) {\n var sym = tinf_decode_symbol(d, lt);\n\n /* check for end of block */\n if (sym === 256) {\n return TINF_OK;\n }\n\n if (sym < 256) {\n d.dest[d.destLen++] = sym;\n } else {\n var length, dist, offs;\n var i;\n\n sym -= 257;\n\n /* possibly get more bits from length code */\n length = tinf_read_bits(d, length_bits[sym], length_base[sym]);\n\n dist = tinf_decode_symbol(d, dt);\n\n /* possibly get more bits from distance code */\n offs = d.destLen - tinf_read_bits(d, dist_bits[dist], dist_base[dist]);\n\n /* copy match */\n for (i = offs; i < offs + length; ++i) {\n d.dest[d.destLen++] = d.dest[i];\n }\n }\n }\n}\n\n/* inflate an uncompressed block of data */\nfunction tinf_inflate_uncompressed_block(d) {\n var length, invlength;\n var i;\n \n /* unread from bitbuffer */\n while (d.bitcount > 8) {\n d.sourceIndex--;\n d.bitcount -= 8;\n }\n\n /* get length */\n length = d.source[d.sourceIndex + 1];\n length = 256 * length + d.source[d.sourceIndex];\n\n /* get one's complement of length */\n invlength = d.source[d.sourceIndex + 3];\n invlength = 256 * invlength + d.source[d.sourceIndex + 2];\n\n /* check length */\n if (length !== (~invlength & 0x0000ffff))\n return TINF_DATA_ERROR;\n\n d.sourceIndex += 4;\n\n /* copy block */\n for (i = length; i; --i)\n d.dest[d.destLen++] = d.source[d.sourceIndex++];\n\n /* make sure we start next block on a byte boundary */\n d.bitcount = 0;\n\n return TINF_OK;\n}\n\n/* inflate stream from source to dest */\nfunction tinf_uncompress(source, dest) {\n var d = new Data(source, dest);\n var bfinal, btype, res;\n\n do {\n /* read final block flag */\n bfinal = tinf_getbit(d);\n\n /* read block type (2 bits) */\n btype = tinf_read_bits(d, 2, 0);\n\n /* decompress block */\n switch (btype) {\n case 0:\n /* decompress uncompressed block */\n res = tinf_inflate_uncompressed_block(d);\n break;\n case 1:\n /* decompress block with fixed huffman trees */\n res = tinf_inflate_block_data(d, sltree, sdtree);\n break;\n case 2:\n /* decompress block with dynamic huffman trees */\n tinf_decode_trees(d, d.ltree, d.dtree);\n res = tinf_inflate_block_data(d, d.ltree, d.dtree);\n break;\n default:\n res = TINF_DATA_ERROR;\n }\n\n if (res !== TINF_OK)\n throw new Error('Data error');\n\n } while (!bfinal);\n\n if (d.destLen < d.dest.length) {\n if (typeof d.dest.slice === 'function')\n return d.dest.slice(0, d.destLen);\n else\n return d.dest.subarray(0, d.destLen);\n }\n \n return d.dest;\n}\n\n/* -------------------- *\n * -- initialization -- *\n * -------------------- */\n\n/* build fixed huffman trees */\ntinf_build_fixed_trees(sltree, sdtree);\n\n/* build extra bits and base tables */\ntinf_build_bits_base(length_bits, length_base, 4, 3);\ntinf_build_bits_base(dist_bits, dist_base, 2, 1);\n\n/* fix a special case */\nlength_bits[28] = 0;\nlength_base[28] = 258;\n\nmodule.exports = tinf_uncompress;\n","// The Bounding Box object\n\nfunction derive(v0, v1, v2, v3, t) {\n return Math.pow(1 - t, 3) * v0 +\n 3 * Math.pow(1 - t, 2) * t * v1 +\n 3 * (1 - t) * Math.pow(t, 2) * v2 +\n Math.pow(t, 3) * v3;\n}\n/**\n * A bounding box is an enclosing box that describes the smallest measure within which all the points lie.\n * It is used to calculate the bounding box of a glyph or text path.\n *\n * On initialization, x1/y1/x2/y2 will be NaN. Check if the bounding box is empty using `isEmpty()`.\n *\n * @exports opentype.BoundingBox\n * @class\n * @constructor\n */\nfunction BoundingBox() {\n this.x1 = Number.NaN;\n this.y1 = Number.NaN;\n this.x2 = Number.NaN;\n this.y2 = Number.NaN;\n}\n\n/**\n * Returns true if the bounding box is empty, that is, no points have been added to the box yet.\n */\nBoundingBox.prototype.isEmpty = function() {\n return isNaN(this.x1) || isNaN(this.y1) || isNaN(this.x2) || isNaN(this.y2);\n};\n\n/**\n * Add the point to the bounding box.\n * The x1/y1/x2/y2 coordinates of the bounding box will now encompass the given point.\n * @param {number} x - The X coordinate of the point.\n * @param {number} y - The Y coordinate of the point.\n */\nBoundingBox.prototype.addPoint = function(x, y) {\n if (typeof x === 'number') {\n if (isNaN(this.x1) || isNaN(this.x2)) {\n this.x1 = x;\n this.x2 = x;\n }\n if (x < this.x1) {\n this.x1 = x;\n }\n if (x > this.x2) {\n this.x2 = x;\n }\n }\n if (typeof y === 'number') {\n if (isNaN(this.y1) || isNaN(this.y2)) {\n this.y1 = y;\n this.y2 = y;\n }\n if (y < this.y1) {\n this.y1 = y;\n }\n if (y > this.y2) {\n this.y2 = y;\n }\n }\n};\n\n/**\n * Add a X coordinate to the bounding box.\n * This extends the bounding box to include the X coordinate.\n * This function is used internally inside of addBezier.\n * @param {number} x - The X coordinate of the point.\n */\nBoundingBox.prototype.addX = function(x) {\n this.addPoint(x, null);\n};\n\n/**\n * Add a Y coordinate to the bounding box.\n * This extends the bounding box to include the Y coordinate.\n * This function is used internally inside of addBezier.\n * @param {number} y - The Y coordinate of the point.\n */\nBoundingBox.prototype.addY = function(y) {\n this.addPoint(null, y);\n};\n\n/**\n * Add a Bézier curve to the bounding box.\n * This extends the bounding box to include the entire Bézier.\n * @param {number} x0 - The starting X coordinate.\n * @param {number} y0 - The starting Y coordinate.\n * @param {number} x1 - The X coordinate of the first control point.\n * @param {number} y1 - The Y coordinate of the first control point.\n * @param {number} x2 - The X coordinate of the second control point.\n * @param {number} y2 - The Y coordinate of the second control point.\n * @param {number} x - The ending X coordinate.\n * @param {number} y - The ending Y coordinate.\n */\nBoundingBox.prototype.addBezier = function(x0, y0, x1, y1, x2, y2, x, y) {\n // This code is based on http://nishiohirokazu.blogspot.com/2009/06/how-to-calculate-bezier-curves-bounding.html\n // and https://github.com/icons8/svg-path-bounding-box\n\n const p0 = [x0, y0];\n const p1 = [x1, y1];\n const p2 = [x2, y2];\n const p3 = [x, y];\n\n this.addPoint(x0, y0);\n this.addPoint(x, y);\n\n for (let i = 0; i <= 1; i++) {\n const b = 6 * p0[i] - 12 * p1[i] + 6 * p2[i];\n const a = -3 * p0[i] + 9 * p1[i] - 9 * p2[i] + 3 * p3[i];\n const c = 3 * p1[i] - 3 * p0[i];\n\n if (a === 0) {\n if (b === 0) continue;\n const t = -c / b;\n if (0 < t && t < 1) {\n if (i === 0) this.addX(derive(p0[i], p1[i], p2[i], p3[i], t));\n if (i === 1) this.addY(derive(p0[i], p1[i], p2[i], p3[i], t));\n }\n continue;\n }\n\n const b2ac = Math.pow(b, 2) - 4 * c * a;\n if (b2ac < 0) continue;\n const t1 = (-b + Math.sqrt(b2ac)) / (2 * a);\n if (0 < t1 && t1 < 1) {\n if (i === 0) this.addX(derive(p0[i], p1[i], p2[i], p3[i], t1));\n if (i === 1) this.addY(derive(p0[i], p1[i], p2[i], p3[i], t1));\n }\n const t2 = (-b - Math.sqrt(b2ac)) / (2 * a);\n if (0 < t2 && t2 < 1) {\n if (i === 0) this.addX(derive(p0[i], p1[i], p2[i], p3[i], t2));\n if (i === 1) this.addY(derive(p0[i], p1[i], p2[i], p3[i], t2));\n }\n }\n};\n\n/**\n * Add a quadratic curve to the bounding box.\n * This extends the bounding box to include the entire quadratic curve.\n * @param {number} x0 - The starting X coordinate.\n * @param {number} y0 - The starting Y coordinate.\n * @param {number} x1 - The X coordinate of the control point.\n * @param {number} y1 - The Y coordinate of the control point.\n * @param {number} x - The ending X coordinate.\n * @param {number} y - The ending Y coordinate.\n */\nBoundingBox.prototype.addQuad = function(x0, y0, x1, y1, x, y) {\n const cp1x = x0 + 2 / 3 * (x1 - x0);\n const cp1y = y0 + 2 / 3 * (y1 - y0);\n const cp2x = cp1x + 1 / 3 * (x - x0);\n const cp2y = cp1y + 1 / 3 * (y - y0);\n this.addBezier(x0, y0, cp1x, cp1y, cp2x, cp2y, x, y);\n};\n\nexport default BoundingBox;\n","// Geometric objects\n\nimport BoundingBox from './bbox';\n\n/**\n * A bézier path containing a set of path commands similar to a SVG path.\n * Paths can be drawn on a context using `draw`.\n * @exports opentype.Path\n * @class\n * @constructor\n */\nfunction Path() {\n this.commands = [];\n this.fill = 'black';\n this.stroke = null;\n this.strokeWidth = 1;\n}\n\n/**\n * @param {number} x\n * @param {number} y\n */\nPath.prototype.moveTo = function(x, y) {\n this.commands.push({\n type: 'M',\n x: x,\n y: y\n });\n};\n\n/**\n * @param {number} x\n * @param {number} y\n */\nPath.prototype.lineTo = function(x, y) {\n this.commands.push({\n type: 'L',\n x: x,\n y: y\n });\n};\n\n/**\n * Draws cubic curve\n * @function\n * curveTo\n * @memberof opentype.Path.prototype\n * @param {number} x1 - x of control 1\n * @param {number} y1 - y of control 1\n * @param {number} x2 - x of control 2\n * @param {number} y2 - y of control 2\n * @param {number} x - x of path point\n * @param {number} y - y of path point\n */\n\n/**\n * Draws cubic curve\n * @function\n * bezierCurveTo\n * @memberof opentype.Path.prototype\n * @param {number} x1 - x of control 1\n * @param {number} y1 - y of control 1\n * @param {number} x2 - x of control 2\n * @param {number} y2 - y of control 2\n * @param {number} x - x of path point\n * @param {number} y - y of path point\n * @see curveTo\n */\nPath.prototype.curveTo = Path.prototype.bezierCurveTo = function(x1, y1, x2, y2, x, y) {\n this.commands.push({\n type: 'C',\n x1: x1,\n y1: y1,\n x2: x2,\n y2: y2,\n x: x,\n y: y\n });\n};\n\n/**\n * Draws quadratic curve\n * @function\n * quadraticCurveTo\n * @memberof opentype.Path.prototype\n * @param {number} x1 - x of control\n * @param {number} y1 - y of control\n * @param {number} x - x of path point\n * @param {number} y - y of path point\n */\n\n/**\n * Draws quadratic curve\n * @function\n * quadTo\n * @memberof opentype.Path.prototype\n * @param {number} x1 - x of control\n * @param {number} y1 - y of control\n * @param {number} x - x of path point\n * @param {number} y - y of path point\n */\nPath.prototype.quadTo = Path.prototype.quadraticCurveTo = function(x1, y1, x, y) {\n this.commands.push({\n type: 'Q',\n x1: x1,\n y1: y1,\n x: x,\n y: y\n });\n};\n\n/**\n * Closes the path\n * @function closePath\n * @memberof opentype.Path.prototype\n */\n\n/**\n * Close the path\n * @function close\n * @memberof opentype.Path.prototype\n */\nPath.prototype.close = Path.prototype.closePath = function() {\n this.commands.push({\n type: 'Z'\n });\n};\n\n/**\n * Add the given path or list of commands to the commands of this path.\n * @param {Array} pathOrCommands - another opentype.Path, an opentype.BoundingBox, or an array of commands.\n */\nPath.prototype.extend = function(pathOrCommands) {\n if (pathOrCommands.commands) {\n pathOrCommands = pathOrCommands.commands;\n } else if (pathOrCommands instanceof BoundingBox) {\n const box = pathOrCommands;\n this.moveTo(box.x1, box.y1);\n this.lineTo(box.x2, box.y1);\n this.lineTo(box.x2, box.y2);\n this.lineTo(box.x1, box.y2);\n this.close();\n return;\n }\n\n Array.prototype.push.apply(this.commands, pathOrCommands);\n};\n\n/**\n * Calculate the bounding box of the path.\n * @returns {opentype.BoundingBox}\n */\nPath.prototype.getBoundingBox = function() {\n const box = new BoundingBox();\n\n let startX = 0;\n let startY = 0;\n let prevX = 0;\n let prevY = 0;\n for (let i = 0; i < this.commands.length; i++) {\n const cmd = this.commands[i];\n switch (cmd.type) {\n case 'M':\n box.addPoint(cmd.x, cmd.y);\n startX = prevX = cmd.x;\n startY = prevY = cmd.y;\n break;\n case 'L':\n box.addPoint(cmd.x, cmd.y);\n prevX = cmd.x;\n prevY = cmd.y;\n break;\n case 'Q':\n box.addQuad(prevX, prevY, cmd.x1, cmd.y1, cmd.x, cmd.y);\n prevX = cmd.x;\n prevY = cmd.y;\n break;\n case 'C':\n box.addBezier(prevX, prevY, cmd.x1, cmd.y1, cmd.x2, cmd.y2, cmd.x, cmd.y);\n prevX = cmd.x;\n prevY = cmd.y;\n break;\n case 'Z':\n prevX = startX;\n prevY = startY;\n break;\n default:\n throw new Error('Unexpected path command ' + cmd.type);\n }\n }\n if (box.isEmpty()) {\n box.addPoint(0, 0);\n }\n return box;\n};\n\n/**\n * Draw the path to a 2D context.\n * @param {CanvasRenderingContext2D} ctx - A 2D drawing context.\n */\nPath.prototype.draw = function(ctx) {\n ctx.beginPath();\n for (let i = 0; i < this.commands.length; i += 1) {\n const cmd = this.commands[i];\n if (cmd.type === 'M') {\n ctx.moveTo(cmd.x, cmd.y);\n } else if (cmd.type === 'L') {\n ctx.lineTo(cmd.x, cmd.y);\n } else if (cmd.type === 'C') {\n ctx.bezierCurveTo(cmd.x1, cmd.y1, cmd.x2, cmd.y2, cmd.x, cmd.y);\n } else if (cmd.type === 'Q') {\n ctx.quadraticCurveTo(cmd.x1, cmd.y1, cmd.x, cmd.y);\n } else if (cmd.type === 'Z') {\n ctx.closePath();\n }\n }\n\n if (this.fill) {\n ctx.fillStyle = this.fill;\n ctx.fill();\n }\n\n if (this.stroke) {\n ctx.strokeStyle = this.stroke;\n ctx.lineWidth = this.strokeWidth;\n ctx.stroke();\n }\n};\n\n/**\n * Convert the Path to a string of path data instructions\n * See http://www.w3.org/TR/SVG/paths.html#PathData\n * @param {number} [decimalPlaces=2] - The amount of decimal places for floating-point values\n * @return {string}\n */\nPath.prototype.toPathData = function(decimalPlaces) {\n decimalPlaces = decimalPlaces !== undefined ? decimalPlaces : 2;\n\n function floatToString(v) {\n if (Math.round(v) === v) {\n return '' + Math.round(v);\n } else {\n return v.toFixed(decimalPlaces);\n }\n }\n\n function packValues() {\n let s = '';\n for (let i = 0; i < arguments.length; i += 1) {\n const v = arguments[i];\n if (v >= 0 && i > 0) {\n s += ' ';\n }\n\n s += floatToString(v);\n }\n\n return s;\n }\n\n let d = '';\n for (let i = 0; i < this.commands.length; i += 1) {\n const cmd = this.commands[i];\n if (cmd.type === 'M') {\n d += 'M' + packValues(cmd.x, cmd.y);\n } else if (cmd.type === 'L') {\n d += 'L' + packValues(cmd.x, cmd.y);\n } else if (cmd.type === 'C') {\n d += 'C' + packValues(cmd.x1, cmd.y1, cmd.x2, cmd.y2, cmd.x, cmd.y);\n } else if (cmd.type === 'Q') {\n d += 'Q' + packValues(cmd.x1, cmd.y1, cmd.x, cmd.y);\n } else if (cmd.type === 'Z') {\n d += 'Z';\n }\n }\n\n return d;\n};\n\n/**\n * Convert the path to an SVG element, as a string.\n * @param {number} [decimalPlaces=2] - The amount of decimal places for floating-point values\n * @return {string}\n */\nPath.prototype.toSVG = function(decimalPlaces) {\n let svg = '= 0 && v <= 255, 'Byte value should be between 0 and 255.');\n return [v];\n};\n/**\n * @constant\n * @type {number}\n */\nsizeOf.BYTE = constant(1);\n\n/**\n * Convert a 8-bit signed integer to a list of 1 byte.\n * @param {string}\n * @returns {Array}\n */\nencode.CHAR = function(v) {\n return [v.charCodeAt(0)];\n};\n\n/**\n * @constant\n * @type {number}\n */\nsizeOf.CHAR = constant(1);\n\n/**\n * Convert an ASCII string to a list of bytes.\n * @param {string}\n * @returns {Array}\n */\nencode.CHARARRAY = function(v) {\n if (typeof v === 'undefined') {\n v = '';\n console.warn('Undefined CHARARRAY encountered and treated as an empty string. This is probably caused by a missing glyph name.');\n }\n const b = [];\n for (let i = 0; i < v.length; i += 1) {\n b[i] = v.charCodeAt(i);\n }\n\n return b;\n};\n\n/**\n * @param {Array}\n * @returns {number}\n */\nsizeOf.CHARARRAY = function(v) {\n if (typeof v === 'undefined') {\n return 0;\n }\n return v.length;\n};\n\n/**\n * Convert a 16-bit unsigned integer to a list of 2 bytes.\n * @param {number}\n * @returns {Array}\n */\nencode.USHORT = function(v) {\n return [(v >> 8) & 0xFF, v & 0xFF];\n};\n\n/**\n * @constant\n * @type {number}\n */\nsizeOf.USHORT = constant(2);\n\n/**\n * Convert a 16-bit signed integer to a list of 2 bytes.\n * @param {number}\n * @returns {Array}\n */\nencode.SHORT = function(v) {\n // Two's complement\n if (v >= LIMIT16) {\n v = -(2 * LIMIT16 - v);\n }\n\n return [(v >> 8) & 0xFF, v & 0xFF];\n};\n\n/**\n * @constant\n * @type {number}\n */\nsizeOf.SHORT = constant(2);\n\n/**\n * Convert a 24-bit unsigned integer to a list of 3 bytes.\n * @param {number}\n * @returns {Array}\n */\nencode.UINT24 = function(v) {\n return [(v >> 16) & 0xFF, (v >> 8) & 0xFF, v & 0xFF];\n};\n\n/**\n * @constant\n * @type {number}\n */\nsizeOf.UINT24 = constant(3);\n\n/**\n * Convert a 32-bit unsigned integer to a list of 4 bytes.\n * @param {number}\n * @returns {Array}\n */\nencode.ULONG = function(v) {\n return [(v >> 24) & 0xFF, (v >> 16) & 0xFF, (v >> 8) & 0xFF, v & 0xFF];\n};\n\n/**\n * @constant\n * @type {number}\n */\nsizeOf.ULONG = constant(4);\n\n/**\n * Convert a 32-bit unsigned integer to a list of 4 bytes.\n * @param {number}\n * @returns {Array}\n */\nencode.LONG = function(v) {\n // Two's complement\n if (v >= LIMIT32) {\n v = -(2 * LIMIT32 - v);\n }\n\n return [(v >> 24) & 0xFF, (v >> 16) & 0xFF, (v >> 8) & 0xFF, v & 0xFF];\n};\n\n/**\n * @constant\n * @type {number}\n */\nsizeOf.LONG = constant(4);\n\nencode.FIXED = encode.ULONG;\nsizeOf.FIXED = sizeOf.ULONG;\n\nencode.FWORD = encode.SHORT;\nsizeOf.FWORD = sizeOf.SHORT;\n\nencode.UFWORD = encode.USHORT;\nsizeOf.UFWORD = sizeOf.USHORT;\n\n/**\n * Convert a 32-bit Apple Mac timestamp integer to a list of 8 bytes, 64-bit timestamp.\n * @param {number}\n * @returns {Array}\n */\nencode.LONGDATETIME = function(v) {\n return [0, 0, 0, 0, (v >> 24) & 0xFF, (v >> 16) & 0xFF, (v >> 8) & 0xFF, v & 0xFF];\n};\n\n/**\n * @constant\n * @type {number}\n */\nsizeOf.LONGDATETIME = constant(8);\n\n/**\n * Convert a 4-char tag to a list of 4 bytes.\n * @param {string}\n * @returns {Array}\n */\nencode.TAG = function(v) {\n check.argument(v.length === 4, 'Tag should be exactly 4 ASCII characters.');\n return [v.charCodeAt(0),\n v.charCodeAt(1),\n v.charCodeAt(2),\n v.charCodeAt(3)];\n};\n\n/**\n * @constant\n * @type {number}\n */\nsizeOf.TAG = constant(4);\n\n// CFF data types ///////////////////////////////////////////////////////////\n\nencode.Card8 = encode.BYTE;\nsizeOf.Card8 = sizeOf.BYTE;\n\nencode.Card16 = encode.USHORT;\nsizeOf.Card16 = sizeOf.USHORT;\n\nencode.OffSize = encode.BYTE;\nsizeOf.OffSize = sizeOf.BYTE;\n\nencode.SID = encode.USHORT;\nsizeOf.SID = sizeOf.USHORT;\n\n// Convert a numeric operand or charstring number to a variable-size list of bytes.\n/**\n * Convert a numeric operand or charstring number to a variable-size list of bytes.\n * @param {number}\n * @returns {Array}\n */\nencode.NUMBER = function(v) {\n if (v >= -107 && v <= 107) {\n return [v + 139];\n } else if (v >= 108 && v <= 1131) {\n v = v - 108;\n return [(v >> 8) + 247, v & 0xFF];\n } else if (v >= -1131 && v <= -108) {\n v = -v - 108;\n return [(v >> 8) + 251, v & 0xFF];\n } else if (v >= -32768 && v <= 32767) {\n return encode.NUMBER16(v);\n } else {\n return encode.NUMBER32(v);\n }\n};\n\n/**\n * @param {number}\n * @returns {number}\n */\nsizeOf.NUMBER = function(v) {\n return encode.NUMBER(v).length;\n};\n\n/**\n * Convert a signed number between -32768 and +32767 to a three-byte value.\n * This ensures we always use three bytes, but is not the most compact format.\n * @param {number}\n * @returns {Array}\n */\nencode.NUMBER16 = function(v) {\n return [28, (v >> 8) & 0xFF, v & 0xFF];\n};\n\n/**\n * @constant\n * @type {number}\n */\nsizeOf.NUMBER16 = constant(3);\n\n/**\n * Convert a signed number between -(2^31) and +(2^31-1) to a five-byte value.\n * This is useful if you want to be sure you always use four bytes,\n * at the expense of wasting a few bytes for smaller numbers.\n * @param {number}\n * @returns {Array}\n */\nencode.NUMBER32 = function(v) {\n return [29, (v >> 24) & 0xFF, (v >> 16) & 0xFF, (v >> 8) & 0xFF, v & 0xFF];\n};\n\n/**\n * @constant\n * @type {number}\n */\nsizeOf.NUMBER32 = constant(5);\n\n/**\n * @param {number}\n * @returns {Array}\n */\nencode.REAL = function(v) {\n let value = v.toString();\n\n // Some numbers use an epsilon to encode the value. (e.g. JavaScript will store 0.0000001 as 1e-7)\n // This code converts it back to a number without the epsilon.\n const m = /\\.(\\d*?)(?:9{5,20}|0{5,20})\\d{0,2}(?:e(.+)|$)/.exec(value);\n if (m) {\n const epsilon = parseFloat('1e' + ((m[2] ? +m[2] : 0) + m[1].length));\n value = (Math.round(v * epsilon) / epsilon).toString();\n }\n\n let nibbles = '';\n for (let i = 0, ii = value.length; i < ii; i += 1) {\n const c = value[i];\n if (c === 'e') {\n nibbles += value[++i] === '-' ? 'c' : 'b';\n } else if (c === '.') {\n nibbles += 'a';\n } else if (c === '-') {\n nibbles += 'e';\n } else {\n nibbles += c;\n }\n }\n\n nibbles += (nibbles.length & 1) ? 'f' : 'ff';\n const out = [30];\n for (let i = 0, ii = nibbles.length; i < ii; i += 2) {\n out.push(parseInt(nibbles.substr(i, 2), 16));\n }\n\n return out;\n};\n\n/**\n * @param {number}\n * @returns {number}\n */\nsizeOf.REAL = function(v) {\n return encode.REAL(v).length;\n};\n\nencode.NAME = encode.CHARARRAY;\nsizeOf.NAME = sizeOf.CHARARRAY;\n\nencode.STRING = encode.CHARARRAY;\nsizeOf.STRING = sizeOf.CHARARRAY;\n\n/**\n * @param {DataView} data\n * @param {number} offset\n * @param {number} numBytes\n * @returns {string}\n */\ndecode.UTF8 = function(data, offset, numBytes) {\n const codePoints = [];\n const numChars = numBytes;\n for (let j = 0; j < numChars; j++, offset += 1) {\n codePoints[j] = data.getUint8(offset);\n }\n\n return String.fromCharCode.apply(null, codePoints);\n};\n\n/**\n * @param {DataView} data\n * @param {number} offset\n * @param {number} numBytes\n * @returns {string}\n */\ndecode.UTF16 = function(data, offset, numBytes) {\n const codePoints = [];\n const numChars = numBytes / 2;\n for (let j = 0; j < numChars; j++, offset += 2) {\n codePoints[j] = data.getUint16(offset);\n }\n\n return String.fromCharCode.apply(null, codePoints);\n};\n\n/**\n * Convert a JavaScript string to UTF16-BE.\n * @param {string}\n * @returns {Array}\n */\nencode.UTF16 = function(v) {\n const b = [];\n for (let i = 0; i < v.length; i += 1) {\n const codepoint = v.charCodeAt(i);\n b[b.length] = (codepoint >> 8) & 0xFF;\n b[b.length] = codepoint & 0xFF;\n }\n\n return b;\n};\n\n/**\n * @param {string}\n * @returns {number}\n */\nsizeOf.UTF16 = function(v) {\n return v.length * 2;\n};\n\n// Data for converting old eight-bit Macintosh encodings to Unicode.\n// This representation is optimized for decoding; encoding is slower\n// and needs more memory. The assumption is that all opentype.js users\n// want to open fonts, but saving a font will be comparatively rare\n// so it can be more expensive. Keyed by IANA character set name.\n//\n// Python script for generating these strings:\n//\n// s = u''.join([chr(c).decode('mac_greek') for c in range(128, 256)])\n// print(s.encode('utf-8'))\n/**\n * @private\n */\nconst eightBitMacEncodings = {\n 'x-mac-croatian': // Python: 'mac_croatian'\n 'ÄÅÇÉÑÖÜáàâäãåçéèêëíìîïñóòôöõúùûü†°¢£§•¶ß®Š™´¨≠ŽØ∞±≤≥∆µ∂∑∏š∫ªºΩžø' +\n '¿¡¬√ƒ≈Ć«Č… ÀÃÕŒœĐ—“”‘’÷◊©⁄€‹›Æ»–·‚„‰ÂćÁčÈÍÎÏÌÓÔđÒÚÛÙıˆ˜¯πË˚¸Êæˇ',\n 'x-mac-cyrillic': // Python: 'mac_cyrillic'\n 'АБВГДЕЖЗИЙКЛМНОПРСТУФХЦЧШЩЪЫЬЭЮЯ†°Ґ£§•¶І®©™Ђђ≠Ѓѓ∞±≤≥іµґЈЄєЇїЉљЊњ' +\n 'јЅ¬√ƒ≈∆«»… ЋћЌќѕ–—“”‘’÷„ЎўЏџ№Ёёяабвгдежзийклмнопрстуфхцчшщъыьэю',\n 'x-mac-gaelic': // http://unicode.org/Public/MAPPINGS/VENDORS/APPLE/GAELIC.TXT\n 'ÄÅÇÉÑÖÜáàâäãåçéèêëíìîïñóòôöõúùûü†°¢£§•¶ß®©™´¨≠ÆØḂ±≤≥ḃĊċḊḋḞḟĠġṀæø' +\n 'ṁṖṗɼƒſṠ«»… ÀÃÕŒœ–—“”‘’ṡẛÿŸṪ€‹›Ŷŷṫ·Ỳỳ⁊ÂÊÁËÈÍÎÏÌÓÔ♣ÒÚÛÙıÝýŴŵẄẅẀẁẂẃ',\n 'x-mac-greek': // Python: 'mac_greek'\n 'Ĺ²É³ÖÜ΅àâä΄¨çéèê룙î‰ôö¦€ùûü†ΓΔΘΛΞΠß®©ΣΪ§≠°·Α±≤≥¥ΒΕΖΗΙΚΜΦΫΨΩ' +\n 'άΝ¬ΟΡ≈Τ«»… ΥΧΆΈœ–―“”‘’÷ΉΊΌΎέήίόΏύαβψδεφγηιξκλμνοπώρστθωςχυζϊϋΐΰ\\u00AD',\n 'x-mac-icelandic': // Python: 'mac_iceland'\n 'ÄÅÇÉÑÖÜáàâäãåçéèêëíìîïñóòôöõúùûüÝ°¢£§•¶ß®©™´¨≠ÆØ∞±≤≥¥µ∂∑∏π∫ªºΩæø' +\n '¿¡¬√ƒ≈∆«»… ÀÃÕŒœ–—“”‘’÷◊ÿŸ⁄€ÐðÞþý·‚„‰ÂÊÁËÈÍÎÏÌÓÔÒÚÛÙıˆ˜¯˘˙˚¸˝˛ˇ',\n 'x-mac-inuit': // http://unicode.org/Public/MAPPINGS/VENDORS/APPLE/INUIT.TXT\n 'ᐃᐄᐅᐆᐊᐋᐱᐲᐳᐴᐸᐹᑉᑎᑏᑐᑑᑕᑖᑦᑭᑮᑯᑰᑲᑳᒃᒋᒌᒍᒎᒐᒑ°ᒡᒥᒦ•¶ᒧ®©™ᒨᒪᒫᒻᓂᓃᓄᓅᓇᓈᓐᓯᓰᓱᓲᓴᓵᔅᓕᓖᓗ' +\n 'ᓘᓚᓛᓪᔨᔩᔪᔫᔭ… ᔮᔾᕕᕖᕗ–—“”‘’ᕘᕙᕚᕝᕆᕇᕈᕉᕋᕌᕐᕿᖀᖁᖂᖃᖄᖅᖏᖐᖑᖒᖓᖔᖕᙱᙲᙳᙴᙵᙶᖖᖠᖡᖢᖣᖤᖥᖦᕼŁł',\n 'x-mac-ce': // Python: 'mac_latin2'\n 'ÄĀāÉĄÖÜáąČäčĆć鏟ĎíďĒēĖóėôöõúĚěü†°Ę£§•¶ß®©™ę¨≠ģĮįĪ≤≥īĶ∂∑łĻļĽľĹĺŅ' +\n 'ņѬ√ńŇ∆«»… ňŐÕőŌ–—“”‘’÷◊ōŔŕŘ‹›řŖŗŠ‚„šŚśÁŤťÍŽžŪÓÔūŮÚůŰűŲųÝýķŻŁżĢˇ',\n macintosh: // Python: 'mac_roman'\n 'ÄÅÇÉÑÖÜáàâäãåçéèêëíìîïñóòôöõúùûü†°¢£§•¶ß®©™´¨≠ÆØ∞±≤≥¥µ∂∑∏π∫ªºΩæø' +\n '¿¡¬√ƒ≈∆«»… ÀÃÕŒœ–—“”‘’÷◊ÿŸ⁄€‹›fifl‡·‚„‰ÂÊÁËÈÍÎÏÌÓÔÒÚÛÙıˆ˜¯˘˙˚¸˝˛ˇ',\n 'x-mac-romanian': // Python: 'mac_romanian'\n 'ÄÅÇÉÑÖÜáàâäãåçéèêëíìîïñóòôöõúùûü†°¢£§•¶ß®©™´¨≠ĂȘ∞±≤≥¥µ∂∑∏π∫ªºΩăș' +\n '¿¡¬√ƒ≈∆«»… ÀÃÕŒœ–—“”‘’÷◊ÿŸ⁄€‹›Țț‡·‚„‰ÂÊÁËÈÍÎÏÌÓÔÒÚÛÙıˆ˜¯˘˙˚¸˝˛ˇ',\n 'x-mac-turkish': // Python: 'mac_turkish'\n 'ÄÅÇÉÑÖÜáàâäãåçéèêëíìîïñóòôöõúùûü†°¢£§•¶ß®©™´¨≠ÆØ∞±≤≥¥µ∂∑∏π∫ªºΩæø' +\n '¿¡¬√ƒ≈∆«»… ÀÃÕŒœ–—“”‘’÷◊ÿŸĞğİıŞş‡·‚„‰ÂÊÁËÈÍÎÏÌÓÔÒÚÛÙˆ˜¯˘˙˚¸˝˛ˇ'\n};\n\n/**\n * Decodes an old-style Macintosh string. Returns either a Unicode JavaScript\n * string, or 'undefined' if the encoding is unsupported. For example, we do\n * not support Chinese, Japanese or Korean because these would need large\n * mapping tables.\n * @param {DataView} dataView\n * @param {number} offset\n * @param {number} dataLength\n * @param {string} encoding\n * @returns {string}\n */\ndecode.MACSTRING = function(dataView, offset, dataLength, encoding) {\n const table = eightBitMacEncodings[encoding];\n if (table === undefined) {\n return undefined;\n }\n\n let result = '';\n for (let i = 0; i < dataLength; i++) {\n const c = dataView.getUint8(offset + i);\n // In all eight-bit Mac encodings, the characters 0x00..0x7F are\n // mapped to U+0000..U+007F; we only need to look up the others.\n if (c <= 0x7F) {\n result += String.fromCharCode(c);\n } else {\n result += table[c & 0x7F];\n }\n }\n\n return result;\n};\n\n// Helper function for encode.MACSTRING. Returns a dictionary for mapping\n// Unicode character codes to their 8-bit MacOS equivalent. This table\n// is not exactly a super cheap data structure, but we do not care because\n// encoding Macintosh strings is only rarely needed in typical applications.\nconst macEncodingTableCache = typeof WeakMap === 'function' && new WeakMap();\nlet macEncodingCacheKeys;\nconst getMacEncodingTable = function (encoding) {\n // Since we use encoding as a cache key for WeakMap, it has to be\n // a String object and not a literal. And at least on NodeJS 2.10.1,\n // WeakMap requires that the same String instance is passed for cache hits.\n if (!macEncodingCacheKeys) {\n macEncodingCacheKeys = {};\n for (let e in eightBitMacEncodings) {\n /*jshint -W053 */ // Suppress \"Do not use String as a constructor.\"\n macEncodingCacheKeys[e] = new String(e);\n }\n }\n\n const cacheKey = macEncodingCacheKeys[encoding];\n if (cacheKey === undefined) {\n return undefined;\n }\n\n // We can't do \"if (cache.has(key)) {return cache.get(key)}\" here:\n // since garbage collection may run at any time, it could also kick in\n // between the calls to cache.has() and cache.get(). In that case,\n // we would return 'undefined' even though we do support the encoding.\n if (macEncodingTableCache) {\n const cachedTable = macEncodingTableCache.get(cacheKey);\n if (cachedTable !== undefined) {\n return cachedTable;\n }\n }\n\n const decodingTable = eightBitMacEncodings[encoding];\n if (decodingTable === undefined) {\n return undefined;\n }\n\n const encodingTable = {};\n for (let i = 0; i < decodingTable.length; i++) {\n encodingTable[decodingTable.charCodeAt(i)] = i + 0x80;\n }\n\n if (macEncodingTableCache) {\n macEncodingTableCache.set(cacheKey, encodingTable);\n }\n\n return encodingTable;\n};\n\n/**\n * Encodes an old-style Macintosh string. Returns a byte array upon success.\n * If the requested encoding is unsupported, or if the input string contains\n * a character that cannot be expressed in the encoding, the function returns\n * 'undefined'.\n * @param {string} str\n * @param {string} encoding\n * @returns {Array}\n */\nencode.MACSTRING = function(str, encoding) {\n const table = getMacEncodingTable(encoding);\n if (table === undefined) {\n return undefined;\n }\n\n const result = [];\n for (let i = 0; i < str.length; i++) {\n let c = str.charCodeAt(i);\n\n // In all eight-bit Mac encodings, the characters 0x00..0x7F are\n // mapped to U+0000..U+007F; we only need to look up the others.\n if (c >= 0x80) {\n c = table[c];\n if (c === undefined) {\n // str contains a Unicode character that cannot be encoded\n // in the requested encoding.\n return undefined;\n }\n }\n result[i] = c;\n // result.push(c);\n }\n\n return result;\n};\n\n/**\n * @param {string} str\n * @param {string} encoding\n * @returns {number}\n */\nsizeOf.MACSTRING = function(str, encoding) {\n const b = encode.MACSTRING(str, encoding);\n if (b !== undefined) {\n return b.length;\n } else {\n return 0;\n }\n};\n\n// Helper for encode.VARDELTAS\nfunction isByteEncodable(value) {\n return value >= -128 && value <= 127;\n}\n\n// Helper for encode.VARDELTAS\nfunction encodeVarDeltaRunAsZeroes(deltas, pos, result) {\n let runLength = 0;\n const numDeltas = deltas.length;\n while (pos < numDeltas && runLength < 64 && deltas[pos] === 0) {\n ++pos;\n ++runLength;\n }\n result.push(0x80 | (runLength - 1));\n return pos;\n}\n\n// Helper for encode.VARDELTAS\nfunction encodeVarDeltaRunAsBytes(deltas, offset, result) {\n let runLength = 0;\n const numDeltas = deltas.length;\n let pos = offset;\n while (pos < numDeltas && runLength < 64) {\n const value = deltas[pos];\n if (!isByteEncodable(value)) {\n break;\n }\n\n // Within a byte-encoded run of deltas, a single zero is best\n // stored literally as 0x00 value. However, if we have two or\n // more zeroes in a sequence, it is better to start a new run.\n // Fore example, the sequence of deltas [15, 15, 0, 15, 15]\n // becomes 6 bytes (04 0F 0F 00 0F 0F) when storing the zero\n // within the current run, but 7 bytes (01 0F 0F 80 01 0F 0F)\n // when starting a new run.\n if (value === 0 && pos + 1 < numDeltas && deltas[pos + 1] === 0) {\n break;\n }\n\n ++pos;\n ++runLength;\n }\n result.push(runLength - 1);\n for (let i = offset; i < pos; ++i) {\n result.push((deltas[i] + 256) & 0xff);\n }\n return pos;\n}\n\n// Helper for encode.VARDELTAS\nfunction encodeVarDeltaRunAsWords(deltas, offset, result) {\n let runLength = 0;\n const numDeltas = deltas.length;\n let pos = offset;\n while (pos < numDeltas && runLength < 64) {\n const value = deltas[pos];\n\n // Within a word-encoded run of deltas, it is easiest to start\n // a new run (with a different encoding) whenever we encounter\n // a zero value. For example, the sequence [0x6666, 0, 0x7777]\n // needs 7 bytes when storing the zero inside the current run\n // (42 66 66 00 00 77 77), and equally 7 bytes when starting a\n // new run (40 66 66 80 40 77 77).\n if (value === 0) {\n break;\n }\n\n // Within a word-encoded run of deltas, a single value in the\n // range (-128..127) should be encoded within the current run\n // because it is more compact. For example, the sequence\n // [0x6666, 2, 0x7777] becomes 7 bytes when storing the value\n // literally (42 66 66 00 02 77 77), but 8 bytes when starting\n // a new run (40 66 66 00 02 40 77 77).\n if (isByteEncodable(value) && pos + 1 < numDeltas && isByteEncodable(deltas[pos + 1])) {\n break;\n }\n\n ++pos;\n ++runLength;\n }\n result.push(0x40 | (runLength - 1));\n for (let i = offset; i < pos; ++i) {\n const val = deltas[i];\n result.push(((val + 0x10000) >> 8) & 0xff, (val + 0x100) & 0xff);\n }\n return pos;\n}\n\n/**\n * Encode a list of variation adjustment deltas.\n *\n * Variation adjustment deltas are used in ‘gvar’ and ‘cvar’ tables.\n * They indicate how points (in ‘gvar’) or values (in ‘cvar’) get adjusted\n * when generating instances of variation fonts.\n *\n * @see https://www.microsoft.com/typography/otspec/gvar.htm\n * @see https://developer.apple.com/fonts/TrueType-Reference-Manual/RM06/Chap6gvar.html\n * @param {Array}\n * @return {Array}\n */\nencode.VARDELTAS = function(deltas) {\n let pos = 0;\n const result = [];\n while (pos < deltas.length) {\n const value = deltas[pos];\n if (value === 0) {\n pos = encodeVarDeltaRunAsZeroes(deltas, pos, result);\n } else if (value >= -128 && value <= 127) {\n pos = encodeVarDeltaRunAsBytes(deltas, pos, result);\n } else {\n pos = encodeVarDeltaRunAsWords(deltas, pos, result);\n }\n }\n return result;\n};\n\n// Convert a list of values to a CFF INDEX structure.\n// The values should be objects containing name / type / value.\n/**\n * @param {Array} l\n * @returns {Array}\n */\nencode.INDEX = function(l) {\n //var offset, offsets, offsetEncoder, encodedOffsets, encodedOffset, data,\n // i, v;\n // Because we have to know which data type to use to encode the offsets,\n // we have to go through the values twice: once to encode the data and\n // calculate the offsets, then again to encode the offsets using the fitting data type.\n let offset = 1; // First offset is always 1.\n const offsets = [offset];\n const data = [];\n for (let i = 0; i < l.length; i += 1) {\n const v = encode.OBJECT(l[i]);\n Array.prototype.push.apply(data, v);\n offset += v.length;\n offsets.push(offset);\n }\n\n if (data.length === 0) {\n return [0, 0];\n }\n\n const encodedOffsets = [];\n const offSize = (1 + Math.floor(Math.log(offset) / Math.log(2)) / 8) | 0;\n const offsetEncoder = [undefined, encode.BYTE, encode.USHORT, encode.UINT24, encode.ULONG][offSize];\n for (let i = 0; i < offsets.length; i += 1) {\n const encodedOffset = offsetEncoder(offsets[i]);\n Array.prototype.push.apply(encodedOffsets, encodedOffset);\n }\n\n return Array.prototype.concat(encode.Card16(l.length),\n encode.OffSize(offSize),\n encodedOffsets,\n data);\n};\n\n/**\n * @param {Array}\n * @returns {number}\n */\nsizeOf.INDEX = function(v) {\n return encode.INDEX(v).length;\n};\n\n/**\n * Convert an object to a CFF DICT structure.\n * The keys should be numeric.\n * The values should be objects containing name / type / value.\n * @param {Object} m\n * @returns {Array}\n */\nencode.DICT = function(m) {\n let d = [];\n const keys = Object.keys(m);\n const length = keys.length;\n\n for (let i = 0; i < length; i += 1) {\n // Object.keys() return string keys, but our keys are always numeric.\n const k = parseInt(keys[i], 0);\n const v = m[k];\n // Value comes before the key.\n d = d.concat(encode.OPERAND(v.value, v.type));\n d = d.concat(encode.OPERATOR(k));\n }\n\n return d;\n};\n\n/**\n * @param {Object}\n * @returns {number}\n */\nsizeOf.DICT = function(m) {\n return encode.DICT(m).length;\n};\n\n/**\n * @param {number}\n * @returns {Array}\n */\nencode.OPERATOR = function(v) {\n if (v < 1200) {\n return [v];\n } else {\n return [12, v - 1200];\n }\n};\n\n/**\n * @param {Array} v\n * @param {string}\n * @returns {Array}\n */\nencode.OPERAND = function(v, type) {\n let d = [];\n if (Array.isArray(type)) {\n for (let i = 0; i < type.length; i += 1) {\n check.argument(v.length === type.length, 'Not enough arguments given for type' + type);\n d = d.concat(encode.OPERAND(v[i], type[i]));\n }\n } else {\n if (type === 'SID') {\n d = d.concat(encode.NUMBER(v));\n } else if (type === 'offset') {\n // We make it easy for ourselves and always encode offsets as\n // 4 bytes. This makes offset calculation for the top dict easier.\n d = d.concat(encode.NUMBER32(v));\n } else if (type === 'number') {\n d = d.concat(encode.NUMBER(v));\n } else if (type === 'real') {\n d = d.concat(encode.REAL(v));\n } else {\n throw new Error('Unknown operand type ' + type);\n // FIXME Add support for booleans\n }\n }\n\n return d;\n};\n\nencode.OP = encode.BYTE;\nsizeOf.OP = sizeOf.BYTE;\n\n// memoize charstring encoding using WeakMap if available\nconst wmm = typeof WeakMap === 'function' && new WeakMap();\n\n/**\n * Convert a list of CharString operations to bytes.\n * @param {Array}\n * @returns {Array}\n */\nencode.CHARSTRING = function(ops) {\n // See encode.MACSTRING for why we don't do \"if (wmm && wmm.has(ops))\".\n if (wmm) {\n const cachedValue = wmm.get(ops);\n if (cachedValue !== undefined) {\n return cachedValue;\n }\n }\n\n let d = [];\n const length = ops.length;\n\n for (let i = 0; i < length; i += 1) {\n const op = ops[i];\n d = d.concat(encode[op.type](op.value));\n }\n\n if (wmm) {\n wmm.set(ops, d);\n }\n\n return d;\n};\n\n/**\n * @param {Array}\n * @returns {number}\n */\nsizeOf.CHARSTRING = function(ops) {\n return encode.CHARSTRING(ops).length;\n};\n\n// Utility functions ////////////////////////////////////////////////////////\n\n/**\n * Convert an object containing name / type / value to bytes.\n * @param {Object}\n * @returns {Array}\n */\nencode.OBJECT = function(v) {\n const encodingFunction = encode[v.type];\n check.argument(encodingFunction !== undefined, 'No encoding function for type ' + v.type);\n return encodingFunction(v.value);\n};\n\n/**\n * @param {Object}\n * @returns {number}\n */\nsizeOf.OBJECT = function(v) {\n const sizeOfFunction = sizeOf[v.type];\n check.argument(sizeOfFunction !== undefined, 'No sizeOf function for type ' + v.type);\n return sizeOfFunction(v.value);\n};\n\n/**\n * Convert a table object to bytes.\n * A table contains a list of fields containing the metadata (name, type and default value).\n * The table itself has the field values set as attributes.\n * @param {opentype.Table}\n * @returns {Array}\n */\nencode.TABLE = function(table) {\n let d = [];\n const length = table.fields.length;\n const subtables = [];\n const subtableOffsets = [];\n\n for (let i = 0; i < length; i += 1) {\n const field = table.fields[i];\n const encodingFunction = encode[field.type];\n check.argument(encodingFunction !== undefined, 'No encoding function for field type ' + field.type + ' (' + field.name + ')');\n let value = table[field.name];\n if (value === undefined) {\n value = field.value;\n }\n\n const bytes = encodingFunction(value);\n\n if (field.type === 'TABLE') {\n subtableOffsets.push(d.length);\n d = d.concat([0, 0]);\n subtables.push(bytes);\n } else {\n d = d.concat(bytes);\n }\n }\n\n for (let i = 0; i < subtables.length; i += 1) {\n const o = subtableOffsets[i];\n const offset = d.length;\n check.argument(offset < 65536, 'Table ' + table.tableName + ' too big.');\n d[o] = offset >> 8;\n d[o + 1] = offset & 0xff;\n d = d.concat(subtables[i]);\n }\n\n return d;\n};\n\n/**\n * @param {opentype.Table}\n * @returns {number}\n */\nsizeOf.TABLE = function(table) {\n let numBytes = 0;\n const length = table.fields.length;\n\n for (let i = 0; i < length; i += 1) {\n const field = table.fields[i];\n const sizeOfFunction = sizeOf[field.type];\n check.argument(sizeOfFunction !== undefined, 'No sizeOf function for field type ' + field.type + ' (' + field.name + ')');\n let value = table[field.name];\n if (value === undefined) {\n value = field.value;\n }\n\n numBytes += sizeOfFunction(value);\n\n // Subtables take 2 more bytes for offsets.\n if (field.type === 'TABLE') {\n numBytes += 2;\n }\n }\n\n return numBytes;\n};\n\nencode.RECORD = encode.TABLE;\nsizeOf.RECORD = sizeOf.TABLE;\n\n// Merge in a list of bytes.\nencode.LITERAL = function(v) {\n return v;\n};\n\nsizeOf.LITERAL = function(v) {\n return v.length;\n};\n\nexport { decode, encode, sizeOf };\n","// Table metadata\n\nimport check from './check';\nimport { encode, sizeOf } from './types';\n\n/**\n * @exports opentype.Table\n * @class\n * @param {string} tableName\n * @param {Array} fields\n * @param {Object} options\n * @constructor\n */\nfunction Table(tableName, fields, options) {\n // For coverage tables with coverage format 2, we do not want to add the coverage data directly to the table object,\n // as this will result in wrong encoding order of the coverage data on serialization to bytes.\n // The fallback of using the field values directly when not present on the table is handled in types.encode.TABLE() already.\n if (fields.length && (fields[0].name !== 'coverageFormat' || fields[0].value === 1)) {\n for (let i = 0; i < fields.length; i += 1) {\n const field = fields[i];\n this[field.name] = field.value;\n }\n }\n\n this.tableName = tableName;\n this.fields = fields;\n if (options) {\n const optionKeys = Object.keys(options);\n for (let i = 0; i < optionKeys.length; i += 1) {\n const k = optionKeys[i];\n const v = options[k];\n if (this[k] !== undefined) {\n this[k] = v;\n }\n }\n }\n}\n\n/**\n * Encodes the table and returns an array of bytes\n * @return {Array}\n */\nTable.prototype.encode = function() {\n return encode.TABLE(this);\n};\n\n/**\n * Get the size of the table.\n * @return {number}\n */\nTable.prototype.sizeOf = function() {\n return sizeOf.TABLE(this);\n};\n\n/**\n * @private\n */\nfunction ushortList(itemName, list, count) {\n if (count === undefined) {\n count = list.length;\n }\n const fields = new Array(list.length + 1);\n fields[0] = {name: itemName + 'Count', type: 'USHORT', value: count};\n for (let i = 0; i < list.length; i++) {\n fields[i + 1] = {name: itemName + i, type: 'USHORT', value: list[i]};\n }\n return fields;\n}\n\n/**\n * @private\n */\nfunction tableList(itemName, records, itemCallback) {\n const count = records.length;\n const fields = new Array(count + 1);\n fields[0] = {name: itemName + 'Count', type: 'USHORT', value: count};\n for (let i = 0; i < count; i++) {\n fields[i + 1] = {name: itemName + i, type: 'TABLE', value: itemCallback(records[i], i)};\n }\n return fields;\n}\n\n/**\n * @private\n */\nfunction recordList(itemName, records, itemCallback) {\n const count = records.length;\n let fields = [];\n fields[0] = {name: itemName + 'Count', type: 'USHORT', value: count};\n for (let i = 0; i < count; i++) {\n fields = fields.concat(itemCallback(records[i], i));\n }\n return fields;\n}\n\n// Common Layout Tables\n\n/**\n * @exports opentype.Coverage\n * @class\n * @param {opentype.Table}\n * @constructor\n * @extends opentype.Table\n */\nfunction Coverage(coverageTable) {\n if (coverageTable.format === 1) {\n Table.call(this, 'coverageTable',\n [{name: 'coverageFormat', type: 'USHORT', value: 1}]\n .concat(ushortList('glyph', coverageTable.glyphs))\n );\n } else if (coverageTable.format === 2) {\n Table.call(this, 'coverageTable',\n [{name: 'coverageFormat', type: 'USHORT', value: 2}]\n .concat(recordList('rangeRecord', coverageTable.ranges, function(RangeRecord) {\n return [\n {name: 'startGlyphID', type: 'USHORT', value: RangeRecord.start},\n {name: 'endGlyphID', type: 'USHORT', value: RangeRecord.end},\n {name: 'startCoverageIndex', type: 'USHORT', value: RangeRecord.index},\n ];\n }))\n );\n } else {\n check.assert(false, 'Coverage format must be 1 or 2.');\n }\n}\nCoverage.prototype = Object.create(Table.prototype);\nCoverage.prototype.constructor = Coverage;\n\nfunction ScriptList(scriptListTable) {\n Table.call(this, 'scriptListTable',\n recordList('scriptRecord', scriptListTable, function(scriptRecord, i) {\n const script = scriptRecord.script;\n let defaultLangSys = script.defaultLangSys;\n check.assert(!!defaultLangSys, 'Unable to write GSUB: script ' + scriptRecord.tag + ' has no default language system.');\n return [\n {name: 'scriptTag' + i, type: 'TAG', value: scriptRecord.tag},\n {name: 'script' + i, type: 'TABLE', value: new Table('scriptTable', [\n {name: 'defaultLangSys', type: 'TABLE', value: new Table('defaultLangSys', [\n {name: 'lookupOrder', type: 'USHORT', value: 0},\n {name: 'reqFeatureIndex', type: 'USHORT', value: defaultLangSys.reqFeatureIndex}]\n .concat(ushortList('featureIndex', defaultLangSys.featureIndexes)))}\n ].concat(recordList('langSys', script.langSysRecords, function(langSysRecord, i) {\n const langSys = langSysRecord.langSys;\n return [\n {name: 'langSysTag' + i, type: 'TAG', value: langSysRecord.tag},\n {name: 'langSys' + i, type: 'TABLE', value: new Table('langSys', [\n {name: 'lookupOrder', type: 'USHORT', value: 0},\n {name: 'reqFeatureIndex', type: 'USHORT', value: langSys.reqFeatureIndex}\n ].concat(ushortList('featureIndex', langSys.featureIndexes)))}\n ];\n })))}\n ];\n })\n );\n}\nScriptList.prototype = Object.create(Table.prototype);\nScriptList.prototype.constructor = ScriptList;\n\n/**\n * @exports opentype.FeatureList\n * @class\n * @param {opentype.Table}\n * @constructor\n * @extends opentype.Table\n */\nfunction FeatureList(featureListTable) {\n Table.call(this, 'featureListTable',\n recordList('featureRecord', featureListTable, function(featureRecord, i) {\n const feature = featureRecord.feature;\n return [\n {name: 'featureTag' + i, type: 'TAG', value: featureRecord.tag},\n {name: 'feature' + i, type: 'TABLE', value: new Table('featureTable', [\n {name: 'featureParams', type: 'USHORT', value: feature.featureParams},\n ].concat(ushortList('lookupListIndex', feature.lookupListIndexes)))}\n ];\n })\n );\n}\nFeatureList.prototype = Object.create(Table.prototype);\nFeatureList.prototype.constructor = FeatureList;\n\n/**\n * @exports opentype.LookupList\n * @class\n * @param {opentype.Table}\n * @param {Object}\n * @constructor\n * @extends opentype.Table\n */\nfunction LookupList(lookupListTable, subtableMakers) {\n Table.call(this, 'lookupListTable', tableList('lookup', lookupListTable, function(lookupTable) {\n let subtableCallback = subtableMakers[lookupTable.lookupType];\n check.assert(!!subtableCallback, 'Unable to write GSUB lookup type ' + lookupTable.lookupType + ' tables.');\n return new Table('lookupTable', [\n {name: 'lookupType', type: 'USHORT', value: lookupTable.lookupType},\n {name: 'lookupFlag', type: 'USHORT', value: lookupTable.lookupFlag}\n ].concat(tableList('subtable', lookupTable.subtables, subtableCallback)));\n }));\n}\nLookupList.prototype = Object.create(Table.prototype);\nLookupList.prototype.constructor = LookupList;\n\n// Record = same as Table, but inlined (a Table has an offset and its data is further in the stream)\n// Don't use offsets inside Records (probable bug), only in Tables.\nexport default {\n Table,\n Record: Table,\n Coverage,\n ScriptList,\n FeatureList,\n LookupList,\n ushortList,\n tableList,\n recordList,\n};\n","// Parsing utility functions\n\nimport check from './check';\n\n// Retrieve an unsigned byte from the DataView.\nfunction getByte(dataView, offset) {\n return dataView.getUint8(offset);\n}\n\n// Retrieve an unsigned 16-bit short from the DataView.\n// The value is stored in big endian.\nfunction getUShort(dataView, offset) {\n return dataView.getUint16(offset, false);\n}\n\n// Retrieve a signed 16-bit short from the DataView.\n// The value is stored in big endian.\nfunction getShort(dataView, offset) {\n return dataView.getInt16(offset, false);\n}\n\n// Retrieve an unsigned 32-bit long from the DataView.\n// The value is stored in big endian.\nfunction getULong(dataView, offset) {\n return dataView.getUint32(offset, false);\n}\n\n// Retrieve a 32-bit signed fixed-point number (16.16) from the DataView.\n// The value is stored in big endian.\nfunction getFixed(dataView, offset) {\n const decimal = dataView.getInt16(offset, false);\n const fraction = dataView.getUint16(offset + 2, false);\n return decimal + fraction / 65535;\n}\n\n// Retrieve a 4-character tag from the DataView.\n// Tags are used to identify tables.\nfunction getTag(dataView, offset) {\n let tag = '';\n for (let i = offset; i < offset + 4; i += 1) {\n tag += String.fromCharCode(dataView.getInt8(i));\n }\n\n return tag;\n}\n\n// Retrieve an offset from the DataView.\n// Offsets are 1 to 4 bytes in length, depending on the offSize argument.\nfunction getOffset(dataView, offset, offSize) {\n let v = 0;\n for (let i = 0; i < offSize; i += 1) {\n v <<= 8;\n v += dataView.getUint8(offset + i);\n }\n\n return v;\n}\n\n// Retrieve a number of bytes from start offset to the end offset from the DataView.\nfunction getBytes(dataView, startOffset, endOffset) {\n const bytes = [];\n for (let i = startOffset; i < endOffset; i += 1) {\n bytes.push(dataView.getUint8(i));\n }\n\n return bytes;\n}\n\n// Convert the list of bytes to a string.\nfunction bytesToString(bytes) {\n let s = '';\n for (let i = 0; i < bytes.length; i += 1) {\n s += String.fromCharCode(bytes[i]);\n }\n\n return s;\n}\n\nconst typeOffsets = {\n byte: 1,\n uShort: 2,\n short: 2,\n uLong: 4,\n fixed: 4,\n longDateTime: 8,\n tag: 4\n};\n\n// A stateful parser that changes the offset whenever a value is retrieved.\n// The data is a DataView.\nfunction Parser(data, offset) {\n this.data = data;\n this.offset = offset;\n this.relativeOffset = 0;\n}\n\nParser.prototype.parseByte = function() {\n const v = this.data.getUint8(this.offset + this.relativeOffset);\n this.relativeOffset += 1;\n return v;\n};\n\nParser.prototype.parseChar = function() {\n const v = this.data.getInt8(this.offset + this.relativeOffset);\n this.relativeOffset += 1;\n return v;\n};\n\nParser.prototype.parseCard8 = Parser.prototype.parseByte;\n\nParser.prototype.parseUShort = function() {\n const v = this.data.getUint16(this.offset + this.relativeOffset);\n this.relativeOffset += 2;\n return v;\n};\n\nParser.prototype.parseCard16 = Parser.prototype.parseUShort;\nParser.prototype.parseSID = Parser.prototype.parseUShort;\nParser.prototype.parseOffset16 = Parser.prototype.parseUShort;\n\nParser.prototype.parseShort = function() {\n const v = this.data.getInt16(this.offset + this.relativeOffset);\n this.relativeOffset += 2;\n return v;\n};\n\nParser.prototype.parseF2Dot14 = function() {\n const v = this.data.getInt16(this.offset + this.relativeOffset) / 16384;\n this.relativeOffset += 2;\n return v;\n};\n\nParser.prototype.parseULong = function() {\n const v = getULong(this.data, this.offset + this.relativeOffset);\n this.relativeOffset += 4;\n return v;\n};\n\nParser.prototype.parseOffset32 = Parser.prototype.parseULong;\n\nParser.prototype.parseFixed = function() {\n const v = getFixed(this.data, this.offset + this.relativeOffset);\n this.relativeOffset += 4;\n return v;\n};\n\nParser.prototype.parseString = function(length) {\n const dataView = this.data;\n const offset = this.offset + this.relativeOffset;\n let string = '';\n this.relativeOffset += length;\n for (let i = 0; i < length; i++) {\n string += String.fromCharCode(dataView.getUint8(offset + i));\n }\n\n return string;\n};\n\nParser.prototype.parseTag = function() {\n return this.parseString(4);\n};\n\n// LONGDATETIME is a 64-bit integer.\n// JavaScript and unix timestamps traditionally use 32 bits, so we\n// only take the last 32 bits.\n// + Since until 2038 those bits will be filled by zeros we can ignore them.\nParser.prototype.parseLongDateTime = function() {\n let v = getULong(this.data, this.offset + this.relativeOffset + 4);\n // Subtract seconds between 01/01/1904 and 01/01/1970\n // to convert Apple Mac timestamp to Standard Unix timestamp\n v -= 2082844800;\n this.relativeOffset += 8;\n return v;\n};\n\nParser.prototype.parseVersion = function(minorBase) {\n const major = getUShort(this.data, this.offset + this.relativeOffset);\n\n // How to interpret the minor version is very vague in the spec. 0x5000 is 5, 0x1000 is 1\n // Default returns the correct number if minor = 0xN000 where N is 0-9\n // Set minorBase to 1 for tables that use minor = N where N is 0-9\n const minor = getUShort(this.data, this.offset + this.relativeOffset + 2);\n this.relativeOffset += 4;\n if (minorBase === undefined) minorBase = 0x1000;\n return major + minor / minorBase / 10;\n};\n\nParser.prototype.skip = function(type, amount) {\n if (amount === undefined) {\n amount = 1;\n }\n\n this.relativeOffset += typeOffsets[type] * amount;\n};\n\n///// Parsing lists and records ///////////////////////////////\n\n// Parse a list of 32 bit unsigned integers.\nParser.prototype.parseULongList = function(count) {\n if (count === undefined) { count = this.parseULong(); }\n const offsets = new Array(count);\n const dataView = this.data;\n let offset = this.offset + this.relativeOffset;\n for (let i = 0; i < count; i++) {\n offsets[i] = dataView.getUint32(offset);\n offset += 4;\n }\n\n this.relativeOffset += count * 4;\n return offsets;\n};\n\n// Parse a list of 16 bit unsigned integers. The length of the list can be read on the stream\n// or provided as an argument.\nParser.prototype.parseOffset16List =\nParser.prototype.parseUShortList = function(count) {\n if (count === undefined) { count = this.parseUShort(); }\n const offsets = new Array(count);\n const dataView = this.data;\n let offset = this.offset + this.relativeOffset;\n for (let i = 0; i < count; i++) {\n offsets[i] = dataView.getUint16(offset);\n offset += 2;\n }\n\n this.relativeOffset += count * 2;\n return offsets;\n};\n\n// Parses a list of 16 bit signed integers.\nParser.prototype.parseShortList = function(count) {\n const list = new Array(count);\n const dataView = this.data;\n let offset = this.offset + this.relativeOffset;\n for (let i = 0; i < count; i++) {\n list[i] = dataView.getInt16(offset);\n offset += 2;\n }\n\n this.relativeOffset += count * 2;\n return list;\n};\n\n// Parses a list of bytes.\nParser.prototype.parseByteList = function(count) {\n const list = new Array(count);\n const dataView = this.data;\n let offset = this.offset + this.relativeOffset;\n for (let i = 0; i < count; i++) {\n list[i] = dataView.getUint8(offset++);\n }\n\n this.relativeOffset += count;\n return list;\n};\n\n/**\n * Parse a list of items.\n * Record count is optional, if omitted it is read from the stream.\n * itemCallback is one of the Parser methods.\n */\nParser.prototype.parseList = function(count, itemCallback) {\n if (!itemCallback) {\n itemCallback = count;\n count = this.parseUShort();\n }\n const list = new Array(count);\n for (let i = 0; i < count; i++) {\n list[i] = itemCallback.call(this);\n }\n return list;\n};\n\nParser.prototype.parseList32 = function(count, itemCallback) {\n if (!itemCallback) {\n itemCallback = count;\n count = this.parseULong();\n }\n const list = new Array(count);\n for (let i = 0; i < count; i++) {\n list[i] = itemCallback.call(this);\n }\n return list;\n};\n\n/**\n * Parse a list of records.\n * Record count is optional, if omitted it is read from the stream.\n * Example of recordDescription: { sequenceIndex: Parser.uShort, lookupListIndex: Parser.uShort }\n */\nParser.prototype.parseRecordList = function(count, recordDescription) {\n // If the count argument is absent, read it in the stream.\n if (!recordDescription) {\n recordDescription = count;\n count = this.parseUShort();\n }\n const records = new Array(count);\n const fields = Object.keys(recordDescription);\n for (let i = 0; i < count; i++) {\n const rec = {};\n for (let j = 0; j < fields.length; j++) {\n const fieldName = fields[j];\n const fieldType = recordDescription[fieldName];\n rec[fieldName] = fieldType.call(this);\n }\n records[i] = rec;\n }\n return records;\n};\n\nParser.prototype.parseRecordList32 = function(count, recordDescription) {\n // If the count argument is absent, read it in the stream.\n if (!recordDescription) {\n recordDescription = count;\n count = this.parseULong();\n }\n const records = new Array(count);\n const fields = Object.keys(recordDescription);\n for (let i = 0; i < count; i++) {\n const rec = {};\n for (let j = 0; j < fields.length; j++) {\n const fieldName = fields[j];\n const fieldType = recordDescription[fieldName];\n rec[fieldName] = fieldType.call(this);\n }\n records[i] = rec;\n }\n return records;\n};\n\n// Parse a data structure into an object\n// Example of description: { sequenceIndex: Parser.uShort, lookupListIndex: Parser.uShort }\nParser.prototype.parseStruct = function(description) {\n if (typeof description === 'function') {\n return description.call(this);\n } else {\n const fields = Object.keys(description);\n const struct = {};\n for (let j = 0; j < fields.length; j++) {\n const fieldName = fields[j];\n const fieldType = description[fieldName];\n struct[fieldName] = fieldType.call(this);\n }\n return struct;\n }\n};\n\n/**\n * Parse a GPOS valueRecord\n * https://docs.microsoft.com/en-us/typography/opentype/spec/gpos#value-record\n * valueFormat is optional, if omitted it is read from the stream.\n */\nParser.prototype.parseValueRecord = function(valueFormat) {\n if (valueFormat === undefined) {\n valueFormat = this.parseUShort();\n }\n if (valueFormat === 0) {\n // valueFormat2 in kerning pairs is most often 0\n // in this case return undefined instead of an empty object, to save space\n return;\n }\n const valueRecord = {};\n\n if (valueFormat & 0x0001) { valueRecord.xPlacement = this.parseShort(); }\n if (valueFormat & 0x0002) { valueRecord.yPlacement = this.parseShort(); }\n if (valueFormat & 0x0004) { valueRecord.xAdvance = this.parseShort(); }\n if (valueFormat & 0x0008) { valueRecord.yAdvance = this.parseShort(); }\n\n // Device table (non-variable font) / VariationIndex table (variable font) not supported\n // https://docs.microsoft.com/fr-fr/typography/opentype/spec/chapter2#devVarIdxTbls\n if (valueFormat & 0x0010) { valueRecord.xPlaDevice = undefined; this.parseShort(); }\n if (valueFormat & 0x0020) { valueRecord.yPlaDevice = undefined; this.parseShort(); }\n if (valueFormat & 0x0040) { valueRecord.xAdvDevice = undefined; this.parseShort(); }\n if (valueFormat & 0x0080) { valueRecord.yAdvDevice = undefined; this.parseShort(); }\n\n return valueRecord;\n};\n\n/**\n * Parse a list of GPOS valueRecords\n * https://docs.microsoft.com/en-us/typography/opentype/spec/gpos#value-record\n * valueFormat and valueCount are read from the stream.\n */\nParser.prototype.parseValueRecordList = function() {\n const valueFormat = this.parseUShort();\n const valueCount = this.parseUShort();\n const values = new Array(valueCount);\n for (let i = 0; i < valueCount; i++) {\n values[i] = this.parseValueRecord(valueFormat);\n }\n return values;\n};\n\nParser.prototype.parsePointer = function(description) {\n const structOffset = this.parseOffset16();\n if (structOffset > 0) {\n // NULL offset => return undefined\n return new Parser(this.data, this.offset + structOffset).parseStruct(description);\n }\n return undefined;\n};\n\nParser.prototype.parsePointer32 = function(description) {\n const structOffset = this.parseOffset32();\n if (structOffset > 0) {\n // NULL offset => return undefined\n return new Parser(this.data, this.offset + structOffset).parseStruct(description);\n }\n return undefined;\n};\n\n/**\n * Parse a list of offsets to lists of 16-bit integers,\n * or a list of offsets to lists of offsets to any kind of items.\n * If itemCallback is not provided, a list of list of UShort is assumed.\n * If provided, itemCallback is called on each item and must parse the item.\n * See examples in tables/gsub.js\n */\nParser.prototype.parseListOfLists = function(itemCallback) {\n const offsets = this.parseOffset16List();\n const count = offsets.length;\n const relativeOffset = this.relativeOffset;\n const list = new Array(count);\n for (let i = 0; i < count; i++) {\n const start = offsets[i];\n if (start === 0) {\n // NULL offset\n // Add i as owned property to list. Convenient with assert.\n list[i] = undefined;\n continue;\n }\n this.relativeOffset = start;\n if (itemCallback) {\n const subOffsets = this.parseOffset16List();\n const subList = new Array(subOffsets.length);\n for (let j = 0; j < subOffsets.length; j++) {\n this.relativeOffset = start + subOffsets[j];\n subList[j] = itemCallback.call(this);\n }\n list[i] = subList;\n } else {\n list[i] = this.parseUShortList();\n }\n }\n this.relativeOffset = relativeOffset;\n return list;\n};\n\n///// Complex tables parsing //////////////////////////////////\n\n// Parse a coverage table in a GSUB, GPOS or GDEF table.\n// https://www.microsoft.com/typography/OTSPEC/chapter2.htm\n// parser.offset must point to the start of the table containing the coverage.\nParser.prototype.parseCoverage = function() {\n const startOffset = this.offset + this.relativeOffset;\n const format = this.parseUShort();\n const count = this.parseUShort();\n if (format === 1) {\n return {\n format: 1,\n glyphs: this.parseUShortList(count)\n };\n } else if (format === 2) {\n const ranges = new Array(count);\n for (let i = 0; i < count; i++) {\n ranges[i] = {\n start: this.parseUShort(),\n end: this.parseUShort(),\n index: this.parseUShort()\n };\n }\n return {\n format: 2,\n ranges: ranges\n };\n }\n throw new Error('0x' + startOffset.toString(16) + ': Coverage format must be 1 or 2.');\n};\n\n// Parse a Class Definition Table in a GSUB, GPOS or GDEF table.\n// https://www.microsoft.com/typography/OTSPEC/chapter2.htm\nParser.prototype.parseClassDef = function() {\n const startOffset = this.offset + this.relativeOffset;\n const format = this.parseUShort();\n if (format === 1) {\n return {\n format: 1,\n startGlyph: this.parseUShort(),\n classes: this.parseUShortList()\n };\n } else if (format === 2) {\n return {\n format: 2,\n ranges: this.parseRecordList({\n start: Parser.uShort,\n end: Parser.uShort,\n classId: Parser.uShort\n })\n };\n }\n throw new Error('0x' + startOffset.toString(16) + ': ClassDef format must be 1 or 2.');\n};\n\n///// Static methods ///////////////////////////////////\n// These convenience methods can be used as callbacks and should be called with \"this\" context set to a Parser instance.\n\nParser.list = function(count, itemCallback) {\n return function() {\n return this.parseList(count, itemCallback);\n };\n};\n\nParser.list32 = function(count, itemCallback) {\n return function() {\n return this.parseList32(count, itemCallback);\n };\n};\n\nParser.recordList = function(count, recordDescription) {\n return function() {\n return this.parseRecordList(count, recordDescription);\n };\n};\n\nParser.recordList32 = function(count, recordDescription) {\n return function() {\n return this.parseRecordList32(count, recordDescription);\n };\n};\n\nParser.pointer = function(description) {\n return function() {\n return this.parsePointer(description);\n };\n};\n\nParser.pointer32 = function(description) {\n return function() {\n return this.parsePointer32(description);\n };\n};\n\nParser.tag = Parser.prototype.parseTag;\nParser.byte = Parser.prototype.parseByte;\nParser.uShort = Parser.offset16 = Parser.prototype.parseUShort;\nParser.uShortList = Parser.prototype.parseUShortList;\nParser.uLong = Parser.offset32 = Parser.prototype.parseULong;\nParser.uLongList = Parser.prototype.parseULongList;\nParser.struct = Parser.prototype.parseStruct;\nParser.coverage = Parser.prototype.parseCoverage;\nParser.classDef = Parser.prototype.parseClassDef;\n\n///// Script, Feature, Lookup lists ///////////////////////////////////////////////\n// https://www.microsoft.com/typography/OTSPEC/chapter2.htm\n\nconst langSysTable = {\n reserved: Parser.uShort,\n reqFeatureIndex: Parser.uShort,\n featureIndexes: Parser.uShortList\n};\n\nParser.prototype.parseScriptList = function() {\n return this.parsePointer(Parser.recordList({\n tag: Parser.tag,\n script: Parser.pointer({\n defaultLangSys: Parser.pointer(langSysTable),\n langSysRecords: Parser.recordList({\n tag: Parser.tag,\n langSys: Parser.pointer(langSysTable)\n })\n })\n })) || [];\n};\n\nParser.prototype.parseFeatureList = function() {\n return this.parsePointer(Parser.recordList({\n tag: Parser.tag,\n feature: Parser.pointer({\n featureParams: Parser.offset16,\n lookupListIndexes: Parser.uShortList\n })\n })) || [];\n};\n\nParser.prototype.parseLookupList = function(lookupTableParsers) {\n return this.parsePointer(Parser.list(Parser.pointer(function() {\n const lookupType = this.parseUShort();\n check.argument(1 <= lookupType && lookupType <= 9, 'GPOS/GSUB lookup type ' + lookupType + ' unknown.');\n const lookupFlag = this.parseUShort();\n const useMarkFilteringSet = lookupFlag & 0x10;\n return {\n lookupType: lookupType,\n lookupFlag: lookupFlag,\n subtables: this.parseList(Parser.pointer(lookupTableParsers[lookupType])),\n markFilteringSet: useMarkFilteringSet ? this.parseUShort() : undefined\n };\n }))) || [];\n};\n\nParser.prototype.parseFeatureVariationsList = function() {\n return this.parsePointer32(function() {\n const majorVersion = this.parseUShort();\n const minorVersion = this.parseUShort();\n check.argument(majorVersion === 1 && minorVersion < 1, 'GPOS/GSUB feature variations table unknown.');\n const featureVariations = this.parseRecordList32({\n conditionSetOffset: Parser.offset32,\n featureTableSubstitutionOffset: Parser.offset32\n });\n return featureVariations;\n }) || [];\n};\n\nexport default {\n getByte,\n getCard8: getByte,\n getUShort,\n getCard16: getUShort,\n getShort,\n getULong,\n getFixed,\n getTag,\n getOffset,\n getBytes,\n bytesToString,\n Parser,\n};\n\nexport { Parser };\n","// The `cmap` table stores the mappings from characters to glyphs.\n// https://www.microsoft.com/typography/OTSPEC/cmap.htm\n\nimport check from '../check';\nimport parse from '../parse';\nimport table from '../table';\n\nfunction parseCmapTableFormat12(cmap, p) {\n //Skip reserved.\n p.parseUShort();\n\n // Length in bytes of the sub-tables.\n cmap.length = p.parseULong();\n cmap.language = p.parseULong();\n\n let groupCount;\n cmap.groupCount = groupCount = p.parseULong();\n cmap.glyphIndexMap = {};\n\n for (let i = 0; i < groupCount; i += 1) {\n const startCharCode = p.parseULong();\n const endCharCode = p.parseULong();\n let startGlyphId = p.parseULong();\n\n for (let c = startCharCode; c <= endCharCode; c += 1) {\n cmap.glyphIndexMap[c] = startGlyphId;\n startGlyphId++;\n }\n }\n}\n\nfunction parseCmapTableFormat4(cmap, p, data, start, offset) {\n // Length in bytes of the sub-tables.\n cmap.length = p.parseUShort();\n cmap.language = p.parseUShort();\n\n // segCount is stored x 2.\n let segCount;\n cmap.segCount = segCount = p.parseUShort() >> 1;\n\n // Skip searchRange, entrySelector, rangeShift.\n p.skip('uShort', 3);\n\n // The \"unrolled\" mapping from character codes to glyph indices.\n cmap.glyphIndexMap = {};\n const endCountParser = new parse.Parser(data, start + offset + 14);\n const startCountParser = new parse.Parser(data, start + offset + 16 + segCount * 2);\n const idDeltaParser = new parse.Parser(data, start + offset + 16 + segCount * 4);\n const idRangeOffsetParser = new parse.Parser(data, start + offset + 16 + segCount * 6);\n let glyphIndexOffset = start + offset + 16 + segCount * 8;\n for (let i = 0; i < segCount - 1; i += 1) {\n let glyphIndex;\n const endCount = endCountParser.parseUShort();\n const startCount = startCountParser.parseUShort();\n const idDelta = idDeltaParser.parseShort();\n const idRangeOffset = idRangeOffsetParser.parseUShort();\n for (let c = startCount; c <= endCount; c += 1) {\n if (idRangeOffset !== 0) {\n // The idRangeOffset is relative to the current position in the idRangeOffset array.\n // Take the current offset in the idRangeOffset array.\n glyphIndexOffset = (idRangeOffsetParser.offset + idRangeOffsetParser.relativeOffset - 2);\n\n // Add the value of the idRangeOffset, which will move us into the glyphIndex array.\n glyphIndexOffset += idRangeOffset;\n\n // Then add the character index of the current segment, multiplied by 2 for USHORTs.\n glyphIndexOffset += (c - startCount) * 2;\n glyphIndex = parse.getUShort(data, glyphIndexOffset);\n if (glyphIndex !== 0) {\n glyphIndex = (glyphIndex + idDelta) & 0xFFFF;\n }\n } else {\n glyphIndex = (c + idDelta) & 0xFFFF;\n }\n\n cmap.glyphIndexMap[c] = glyphIndex;\n }\n }\n}\n\n// Parse the `cmap` table. This table stores the mappings from characters to glyphs.\n// There are many available formats, but we only support the Windows format 4 and 12.\n// This function returns a `CmapEncoding` object or null if no supported format could be found.\nfunction parseCmapTable(data, start) {\n const cmap = {};\n cmap.version = parse.getUShort(data, start);\n check.argument(cmap.version === 0, 'cmap table version should be 0.');\n\n // The cmap table can contain many sub-tables, each with their own format.\n // We're only interested in a \"platform 0\" (Unicode format) and \"platform 3\" (Windows format) table.\n cmap.numTables = parse.getUShort(data, start + 2);\n let offset = -1;\n for (let i = cmap.numTables - 1; i >= 0; i -= 1) {\n const platformId = parse.getUShort(data, start + 4 + (i * 8));\n const encodingId = parse.getUShort(data, start + 4 + (i * 8) + 2);\n if ((platformId === 3 && (encodingId === 0 || encodingId === 1 || encodingId === 10)) ||\n (platformId === 0 && (encodingId === 0 || encodingId === 1 || encodingId === 2 || encodingId === 3 || encodingId === 4))) {\n offset = parse.getULong(data, start + 4 + (i * 8) + 4);\n break;\n }\n }\n\n if (offset === -1) {\n // There is no cmap table in the font that we support.\n throw new Error('No valid cmap sub-tables found.');\n }\n\n const p = new parse.Parser(data, start + offset);\n cmap.format = p.parseUShort();\n\n if (cmap.format === 12) {\n parseCmapTableFormat12(cmap, p);\n } else if (cmap.format === 4) {\n parseCmapTableFormat4(cmap, p, data, start, offset);\n } else {\n throw new Error('Only format 4 and 12 cmap tables are supported (found format ' + cmap.format + ').');\n }\n\n return cmap;\n}\n\nfunction addSegment(t, code, glyphIndex) {\n t.segments.push({\n end: code,\n start: code,\n delta: -(code - glyphIndex),\n offset: 0,\n glyphIndex: glyphIndex\n });\n}\n\nfunction addTerminatorSegment(t) {\n t.segments.push({\n end: 0xFFFF,\n start: 0xFFFF,\n delta: 1,\n offset: 0\n });\n}\n\n// Make cmap table, format 4 by default, 12 if needed only\nfunction makeCmapTable(glyphs) {\n // Plan 0 is the base Unicode Plan but emojis, for example are on another plan, and needs cmap 12 format (with 32bit)\n let isPlan0Only = true;\n let i;\n\n // Check if we need to add cmap format 12 or if format 4 only is fine\n for (i = glyphs.length - 1; i > 0; i -= 1) {\n const g = glyphs.get(i);\n if (g.unicode > 65535) {\n console.log('Adding CMAP format 12 (needed!)');\n isPlan0Only = false;\n break;\n }\n }\n\n let cmapTable = [\n {name: 'version', type: 'USHORT', value: 0},\n {name: 'numTables', type: 'USHORT', value: isPlan0Only ? 1 : 2},\n\n // CMAP 4 header\n {name: 'platformID', type: 'USHORT', value: 3},\n {name: 'encodingID', type: 'USHORT', value: 1},\n {name: 'offset', type: 'ULONG', value: isPlan0Only ? 12 : (12 + 8)}\n ];\n\n if (!isPlan0Only)\n cmapTable = cmapTable.concat([\n // CMAP 12 header\n {name: 'cmap12PlatformID', type: 'USHORT', value: 3}, // We encode only for PlatformID = 3 (Windows) because it is supported everywhere\n {name: 'cmap12EncodingID', type: 'USHORT', value: 10},\n {name: 'cmap12Offset', type: 'ULONG', value: 0}\n ]);\n\n cmapTable = cmapTable.concat([\n // CMAP 4 Subtable\n {name: 'format', type: 'USHORT', value: 4},\n {name: 'cmap4Length', type: 'USHORT', value: 0},\n {name: 'language', type: 'USHORT', value: 0},\n {name: 'segCountX2', type: 'USHORT', value: 0},\n {name: 'searchRange', type: 'USHORT', value: 0},\n {name: 'entrySelector', type: 'USHORT', value: 0},\n {name: 'rangeShift', type: 'USHORT', value: 0}\n ]);\n\n const t = new table.Table('cmap', cmapTable);\n\n t.segments = [];\n for (i = 0; i < glyphs.length; i += 1) {\n const glyph = glyphs.get(i);\n for (let j = 0; j < glyph.unicodes.length; j += 1) {\n addSegment(t, glyph.unicodes[j], i);\n }\n\n t.segments = t.segments.sort(function (a, b) {\n return a.start - b.start;\n });\n }\n\n addTerminatorSegment(t);\n\n const segCount = t.segments.length;\n let segCountToRemove = 0;\n\n // CMAP 4\n // Set up parallel segment arrays.\n let endCounts = [];\n let startCounts = [];\n let idDeltas = [];\n let idRangeOffsets = [];\n let glyphIds = [];\n\n // CMAP 12\n let cmap12Groups = [];\n\n // Reminder this loop is not following the specification at 100%\n // The specification -> find suites of characters and make a group\n // Here we're doing one group for each letter\n // Doing as the spec can save 8 times (or more) space\n for (i = 0; i < segCount; i += 1) {\n const segment = t.segments[i];\n\n // CMAP 4\n if (segment.end <= 65535 && segment.start <= 65535) {\n endCounts = endCounts.concat({name: 'end_' + i, type: 'USHORT', value: segment.end});\n startCounts = startCounts.concat({name: 'start_' + i, type: 'USHORT', value: segment.start});\n idDeltas = idDeltas.concat({name: 'idDelta_' + i, type: 'SHORT', value: segment.delta});\n idRangeOffsets = idRangeOffsets.concat({name: 'idRangeOffset_' + i, type: 'USHORT', value: segment.offset});\n if (segment.glyphId !== undefined) {\n glyphIds = glyphIds.concat({name: 'glyph_' + i, type: 'USHORT', value: segment.glyphId});\n }\n } else {\n // Skip Unicode > 65535 (16bit unsigned max) for CMAP 4, will be added in CMAP 12\n segCountToRemove += 1;\n }\n\n // CMAP 12\n // Skip Terminator Segment\n if (!isPlan0Only && segment.glyphIndex !== undefined) {\n cmap12Groups = cmap12Groups.concat({name: 'cmap12Start_' + i, type: 'ULONG', value: segment.start});\n cmap12Groups = cmap12Groups.concat({name: 'cmap12End_' + i, type: 'ULONG', value: segment.end});\n cmap12Groups = cmap12Groups.concat({name: 'cmap12Glyph_' + i, type: 'ULONG', value: segment.glyphIndex});\n }\n }\n\n // CMAP 4 Subtable\n t.segCountX2 = (segCount - segCountToRemove) * 2;\n t.searchRange = Math.pow(2, Math.floor(Math.log((segCount - segCountToRemove)) / Math.log(2))) * 2;\n t.entrySelector = Math.log(t.searchRange / 2) / Math.log(2);\n t.rangeShift = t.segCountX2 - t.searchRange;\n\n t.fields = t.fields.concat(endCounts);\n t.fields.push({name: 'reservedPad', type: 'USHORT', value: 0});\n t.fields = t.fields.concat(startCounts);\n t.fields = t.fields.concat(idDeltas);\n t.fields = t.fields.concat(idRangeOffsets);\n t.fields = t.fields.concat(glyphIds);\n\n t.cmap4Length = 14 + // Subtable header\n endCounts.length * 2 +\n 2 + // reservedPad\n startCounts.length * 2 +\n idDeltas.length * 2 +\n idRangeOffsets.length * 2 +\n glyphIds.length * 2;\n\n if (!isPlan0Only) {\n // CMAP 12 Subtable\n const cmap12Length = 16 + // Subtable header\n cmap12Groups.length * 4;\n\n t.cmap12Offset = 12 + (2 * 2) + 4 + t.cmap4Length;\n t.fields = t.fields.concat([\n {name: 'cmap12Format', type: 'USHORT', value: 12},\n {name: 'cmap12Reserved', type: 'USHORT', value: 0},\n {name: 'cmap12Length', type: 'ULONG', value: cmap12Length},\n {name: 'cmap12Language', type: 'ULONG', value: 0},\n {name: 'cmap12nGroups', type: 'ULONG', value: cmap12Groups.length / 3}\n ]);\n\n t.fields = t.fields.concat(cmap12Groups);\n }\n\n return t;\n}\n\nexport default { parse: parseCmapTable, make: makeCmapTable };\n","// Glyph encoding\n\nconst cffStandardStrings = [\n '.notdef', 'space', 'exclam', 'quotedbl', 'numbersign', 'dollar', 'percent', 'ampersand', 'quoteright',\n 'parenleft', 'parenright', 'asterisk', 'plus', 'comma', 'hyphen', 'period', 'slash', 'zero', 'one', 'two',\n 'three', 'four', 'five', 'six', 'seven', 'eight', 'nine', 'colon', 'semicolon', 'less', 'equal', 'greater',\n 'question', 'at', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S',\n 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', 'bracketleft', 'backslash', 'bracketright', 'asciicircum', 'underscore',\n 'quoteleft', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't',\n 'u', 'v', 'w', 'x', 'y', 'z', 'braceleft', 'bar', 'braceright', 'asciitilde', 'exclamdown', 'cent', 'sterling',\n 'fraction', 'yen', 'florin', 'section', 'currency', 'quotesingle', 'quotedblleft', 'guillemotleft',\n 'guilsinglleft', 'guilsinglright', 'fi', 'fl', 'endash', 'dagger', 'daggerdbl', 'periodcentered', 'paragraph',\n 'bullet', 'quotesinglbase', 'quotedblbase', 'quotedblright', 'guillemotright', 'ellipsis', 'perthousand',\n 'questiondown', 'grave', 'acute', 'circumflex', 'tilde', 'macron', 'breve', 'dotaccent', 'dieresis', 'ring',\n 'cedilla', 'hungarumlaut', 'ogonek', 'caron', 'emdash', 'AE', 'ordfeminine', 'Lslash', 'Oslash', 'OE',\n 'ordmasculine', 'ae', 'dotlessi', 'lslash', 'oslash', 'oe', 'germandbls', 'onesuperior', 'logicalnot', 'mu',\n 'trademark', 'Eth', 'onehalf', 'plusminus', 'Thorn', 'onequarter', 'divide', 'brokenbar', 'degree', 'thorn',\n 'threequarters', 'twosuperior', 'registered', 'minus', 'eth', 'multiply', 'threesuperior', 'copyright',\n 'Aacute', 'Acircumflex', 'Adieresis', 'Agrave', 'Aring', 'Atilde', 'Ccedilla', 'Eacute', 'Ecircumflex',\n 'Edieresis', 'Egrave', 'Iacute', 'Icircumflex', 'Idieresis', 'Igrave', 'Ntilde', 'Oacute', 'Ocircumflex',\n 'Odieresis', 'Ograve', 'Otilde', 'Scaron', 'Uacute', 'Ucircumflex', 'Udieresis', 'Ugrave', 'Yacute',\n 'Ydieresis', 'Zcaron', 'aacute', 'acircumflex', 'adieresis', 'agrave', 'aring', 'atilde', 'ccedilla', 'eacute',\n 'ecircumflex', 'edieresis', 'egrave', 'iacute', 'icircumflex', 'idieresis', 'igrave', 'ntilde', 'oacute',\n 'ocircumflex', 'odieresis', 'ograve', 'otilde', 'scaron', 'uacute', 'ucircumflex', 'udieresis', 'ugrave',\n 'yacute', 'ydieresis', 'zcaron', 'exclamsmall', 'Hungarumlautsmall', 'dollaroldstyle', 'dollarsuperior',\n 'ampersandsmall', 'Acutesmall', 'parenleftsuperior', 'parenrightsuperior', '266 ff', 'onedotenleader',\n 'zerooldstyle', 'oneoldstyle', 'twooldstyle', 'threeoldstyle', 'fouroldstyle', 'fiveoldstyle', 'sixoldstyle',\n 'sevenoldstyle', 'eightoldstyle', 'nineoldstyle', 'commasuperior', 'threequartersemdash', 'periodsuperior',\n 'questionsmall', 'asuperior', 'bsuperior', 'centsuperior', 'dsuperior', 'esuperior', 'isuperior', 'lsuperior',\n 'msuperior', 'nsuperior', 'osuperior', 'rsuperior', 'ssuperior', 'tsuperior', 'ff', 'ffi', 'ffl',\n 'parenleftinferior', 'parenrightinferior', 'Circumflexsmall', 'hyphensuperior', 'Gravesmall', 'Asmall',\n 'Bsmall', 'Csmall', 'Dsmall', 'Esmall', 'Fsmall', 'Gsmall', 'Hsmall', 'Ismall', 'Jsmall', 'Ksmall', 'Lsmall',\n 'Msmall', 'Nsmall', 'Osmall', 'Psmall', 'Qsmall', 'Rsmall', 'Ssmall', 'Tsmall', 'Usmall', 'Vsmall', 'Wsmall',\n 'Xsmall', 'Ysmall', 'Zsmall', 'colonmonetary', 'onefitted', 'rupiah', 'Tildesmall', 'exclamdownsmall',\n 'centoldstyle', 'Lslashsmall', 'Scaronsmall', 'Zcaronsmall', 'Dieresissmall', 'Brevesmall', 'Caronsmall',\n 'Dotaccentsmall', 'Macronsmall', 'figuredash', 'hypheninferior', 'Ogoneksmall', 'Ringsmall', 'Cedillasmall',\n 'questiondownsmall', 'oneeighth', 'threeeighths', 'fiveeighths', 'seveneighths', 'onethird', 'twothirds',\n 'zerosuperior', 'foursuperior', 'fivesuperior', 'sixsuperior', 'sevensuperior', 'eightsuperior', 'ninesuperior',\n 'zeroinferior', 'oneinferior', 'twoinferior', 'threeinferior', 'fourinferior', 'fiveinferior', 'sixinferior',\n 'seveninferior', 'eightinferior', 'nineinferior', 'centinferior', 'dollarinferior', 'periodinferior',\n 'commainferior', 'Agravesmall', 'Aacutesmall', 'Acircumflexsmall', 'Atildesmall', 'Adieresissmall',\n 'Aringsmall', 'AEsmall', 'Ccedillasmall', 'Egravesmall', 'Eacutesmall', 'Ecircumflexsmall', 'Edieresissmall',\n 'Igravesmall', 'Iacutesmall', 'Icircumflexsmall', 'Idieresissmall', 'Ethsmall', 'Ntildesmall', 'Ogravesmall',\n 'Oacutesmall', 'Ocircumflexsmall', 'Otildesmall', 'Odieresissmall', 'OEsmall', 'Oslashsmall', 'Ugravesmall',\n 'Uacutesmall', 'Ucircumflexsmall', 'Udieresissmall', 'Yacutesmall', 'Thornsmall', 'Ydieresissmall', '001.000',\n '001.001', '001.002', '001.003', 'Black', 'Bold', 'Book', 'Light', 'Medium', 'Regular', 'Roman', 'Semibold'];\n\nconst cffStandardEncoding = [\n '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '',\n '', '', '', '', 'space', 'exclam', 'quotedbl', 'numbersign', 'dollar', 'percent', 'ampersand', 'quoteright',\n 'parenleft', 'parenright', 'asterisk', 'plus', 'comma', 'hyphen', 'period', 'slash', 'zero', 'one', 'two',\n 'three', 'four', 'five', 'six', 'seven', 'eight', 'nine', 'colon', 'semicolon', 'less', 'equal', 'greater',\n 'question', 'at', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S',\n 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', 'bracketleft', 'backslash', 'bracketright', 'asciicircum', 'underscore',\n 'quoteleft', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't',\n 'u', 'v', 'w', 'x', 'y', 'z', 'braceleft', 'bar', 'braceright', 'asciitilde', '', '', '', '', '', '', '', '',\n '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '',\n 'exclamdown', 'cent', 'sterling', 'fraction', 'yen', 'florin', 'section', 'currency', 'quotesingle',\n 'quotedblleft', 'guillemotleft', 'guilsinglleft', 'guilsinglright', 'fi', 'fl', '', 'endash', 'dagger',\n 'daggerdbl', 'periodcentered', '', 'paragraph', 'bullet', 'quotesinglbase', 'quotedblbase', 'quotedblright',\n 'guillemotright', 'ellipsis', 'perthousand', '', 'questiondown', '', 'grave', 'acute', 'circumflex', 'tilde',\n 'macron', 'breve', 'dotaccent', 'dieresis', '', 'ring', 'cedilla', '', 'hungarumlaut', 'ogonek', 'caron',\n 'emdash', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', 'AE', '', 'ordfeminine', '', '', '',\n '', 'Lslash', 'Oslash', 'OE', 'ordmasculine', '', '', '', '', '', 'ae', '', '', '', 'dotlessi', '', '',\n 'lslash', 'oslash', 'oe', 'germandbls'];\n\nconst cffExpertEncoding = [\n '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '',\n '', '', '', '', 'space', 'exclamsmall', 'Hungarumlautsmall', '', 'dollaroldstyle', 'dollarsuperior',\n 'ampersandsmall', 'Acutesmall', 'parenleftsuperior', 'parenrightsuperior', 'twodotenleader', 'onedotenleader',\n 'comma', 'hyphen', 'period', 'fraction', 'zerooldstyle', 'oneoldstyle', 'twooldstyle', 'threeoldstyle',\n 'fouroldstyle', 'fiveoldstyle', 'sixoldstyle', 'sevenoldstyle', 'eightoldstyle', 'nineoldstyle', 'colon',\n 'semicolon', 'commasuperior', 'threequartersemdash', 'periodsuperior', 'questionsmall', '', 'asuperior',\n 'bsuperior', 'centsuperior', 'dsuperior', 'esuperior', '', '', 'isuperior', '', '', 'lsuperior', 'msuperior',\n 'nsuperior', 'osuperior', '', '', 'rsuperior', 'ssuperior', 'tsuperior', '', 'ff', 'fi', 'fl', 'ffi', 'ffl',\n 'parenleftinferior', '', 'parenrightinferior', 'Circumflexsmall', 'hyphensuperior', 'Gravesmall', 'Asmall',\n 'Bsmall', 'Csmall', 'Dsmall', 'Esmall', 'Fsmall', 'Gsmall', 'Hsmall', 'Ismall', 'Jsmall', 'Ksmall', 'Lsmall',\n 'Msmall', 'Nsmall', 'Osmall', 'Psmall', 'Qsmall', 'Rsmall', 'Ssmall', 'Tsmall', 'Usmall', 'Vsmall', 'Wsmall',\n 'Xsmall', 'Ysmall', 'Zsmall', 'colonmonetary', 'onefitted', 'rupiah', 'Tildesmall', '', '', '', '', '', '', '',\n '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '',\n 'exclamdownsmall', 'centoldstyle', 'Lslashsmall', '', '', 'Scaronsmall', 'Zcaronsmall', 'Dieresissmall',\n 'Brevesmall', 'Caronsmall', '', 'Dotaccentsmall', '', '', 'Macronsmall', '', '', 'figuredash', 'hypheninferior',\n '', '', 'Ogoneksmall', 'Ringsmall', 'Cedillasmall', '', '', '', 'onequarter', 'onehalf', 'threequarters',\n 'questiondownsmall', 'oneeighth', 'threeeighths', 'fiveeighths', 'seveneighths', 'onethird', 'twothirds', '',\n '', 'zerosuperior', 'onesuperior', 'twosuperior', 'threesuperior', 'foursuperior', 'fivesuperior',\n 'sixsuperior', 'sevensuperior', 'eightsuperior', 'ninesuperior', 'zeroinferior', 'oneinferior', 'twoinferior',\n 'threeinferior', 'fourinferior', 'fiveinferior', 'sixinferior', 'seveninferior', 'eightinferior',\n 'nineinferior', 'centinferior', 'dollarinferior', 'periodinferior', 'commainferior', 'Agravesmall',\n 'Aacutesmall', 'Acircumflexsmall', 'Atildesmall', 'Adieresissmall', 'Aringsmall', 'AEsmall', 'Ccedillasmall',\n 'Egravesmall', 'Eacutesmall', 'Ecircumflexsmall', 'Edieresissmall', 'Igravesmall', 'Iacutesmall',\n 'Icircumflexsmall', 'Idieresissmall', 'Ethsmall', 'Ntildesmall', 'Ogravesmall', 'Oacutesmall',\n 'Ocircumflexsmall', 'Otildesmall', 'Odieresissmall', 'OEsmall', 'Oslashsmall', 'Ugravesmall', 'Uacutesmall',\n 'Ucircumflexsmall', 'Udieresissmall', 'Yacutesmall', 'Thornsmall', 'Ydieresissmall'];\n\nconst standardNames = [\n '.notdef', '.null', 'nonmarkingreturn', 'space', 'exclam', 'quotedbl', 'numbersign', 'dollar', 'percent',\n 'ampersand', 'quotesingle', 'parenleft', 'parenright', 'asterisk', 'plus', 'comma', 'hyphen', 'period', 'slash',\n 'zero', 'one', 'two', 'three', 'four', 'five', 'six', 'seven', 'eight', 'nine', 'colon', 'semicolon', 'less',\n 'equal', 'greater', 'question', 'at', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O',\n 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', 'bracketleft', 'backslash', 'bracketright',\n 'asciicircum', 'underscore', 'grave', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o',\n 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', 'braceleft', 'bar', 'braceright', 'asciitilde',\n 'Adieresis', 'Aring', 'Ccedilla', 'Eacute', 'Ntilde', 'Odieresis', 'Udieresis', 'aacute', 'agrave',\n 'acircumflex', 'adieresis', 'atilde', 'aring', 'ccedilla', 'eacute', 'egrave', 'ecircumflex', 'edieresis',\n 'iacute', 'igrave', 'icircumflex', 'idieresis', 'ntilde', 'oacute', 'ograve', 'ocircumflex', 'odieresis',\n 'otilde', 'uacute', 'ugrave', 'ucircumflex', 'udieresis', 'dagger', 'degree', 'cent', 'sterling', 'section',\n 'bullet', 'paragraph', 'germandbls', 'registered', 'copyright', 'trademark', 'acute', 'dieresis', 'notequal',\n 'AE', 'Oslash', 'infinity', 'plusminus', 'lessequal', 'greaterequal', 'yen', 'mu', 'partialdiff', 'summation',\n 'product', 'pi', 'integral', 'ordfeminine', 'ordmasculine', 'Omega', 'ae', 'oslash', 'questiondown',\n 'exclamdown', 'logicalnot', 'radical', 'florin', 'approxequal', 'Delta', 'guillemotleft', 'guillemotright',\n 'ellipsis', 'nonbreakingspace', 'Agrave', 'Atilde', 'Otilde', 'OE', 'oe', 'endash', 'emdash', 'quotedblleft',\n 'quotedblright', 'quoteleft', 'quoteright', 'divide', 'lozenge', 'ydieresis', 'Ydieresis', 'fraction',\n 'currency', 'guilsinglleft', 'guilsinglright', 'fi', 'fl', 'daggerdbl', 'periodcentered', 'quotesinglbase',\n 'quotedblbase', 'perthousand', 'Acircumflex', 'Ecircumflex', 'Aacute', 'Edieresis', 'Egrave', 'Iacute',\n 'Icircumflex', 'Idieresis', 'Igrave', 'Oacute', 'Ocircumflex', 'apple', 'Ograve', 'Uacute', 'Ucircumflex',\n 'Ugrave', 'dotlessi', 'circumflex', 'tilde', 'macron', 'breve', 'dotaccent', 'ring', 'cedilla', 'hungarumlaut',\n 'ogonek', 'caron', 'Lslash', 'lslash', 'Scaron', 'scaron', 'Zcaron', 'zcaron', 'brokenbar', 'Eth', 'eth',\n 'Yacute', 'yacute', 'Thorn', 'thorn', 'minus', 'multiply', 'onesuperior', 'twosuperior', 'threesuperior',\n 'onehalf', 'onequarter', 'threequarters', 'franc', 'Gbreve', 'gbreve', 'Idotaccent', 'Scedilla', 'scedilla',\n 'Cacute', 'cacute', 'Ccaron', 'ccaron', 'dcroat'];\n\n/**\n * This is the encoding used for fonts created from scratch.\n * It loops through all glyphs and finds the appropriate unicode value.\n * Since it's linear time, other encodings will be faster.\n * @exports opentype.DefaultEncoding\n * @class\n * @constructor\n * @param {opentype.Font}\n */\nfunction DefaultEncoding(font) {\n this.font = font;\n}\n\nDefaultEncoding.prototype.charToGlyphIndex = function(c) {\n const code = c.codePointAt(0);\n const glyphs = this.font.glyphs;\n if (glyphs) {\n for (let i = 0; i < glyphs.length; i += 1) {\n const glyph = glyphs.get(i);\n for (let j = 0; j < glyph.unicodes.length; j += 1) {\n if (glyph.unicodes[j] === code) {\n return i;\n }\n }\n }\n }\n return null;\n};\n\n/**\n * @exports opentype.CmapEncoding\n * @class\n * @constructor\n * @param {Object} cmap - a object with the cmap encoded data\n */\nfunction CmapEncoding(cmap) {\n this.cmap = cmap;\n}\n\n/**\n * @param {string} c - the character\n * @return {number} The glyph index.\n */\nCmapEncoding.prototype.charToGlyphIndex = function(c) {\n return this.cmap.glyphIndexMap[c.codePointAt(0)] || 0;\n};\n\n/**\n * @exports opentype.CffEncoding\n * @class\n * @constructor\n * @param {string} encoding - The encoding\n * @param {Array} charset - The character set.\n */\nfunction CffEncoding(encoding, charset) {\n this.encoding = encoding;\n this.charset = charset;\n}\n\n/**\n * @param {string} s - The character\n * @return {number} The index.\n */\nCffEncoding.prototype.charToGlyphIndex = function(s) {\n const code = s.codePointAt(0);\n const charName = this.encoding[code];\n return this.charset.indexOf(charName);\n};\n\n/**\n * @exports opentype.GlyphNames\n * @class\n * @constructor\n * @param {Object} post\n */\nfunction GlyphNames(post) {\n switch (post.version) {\n case 1:\n this.names = standardNames.slice();\n break;\n case 2:\n this.names = new Array(post.numberOfGlyphs);\n for (let i = 0; i < post.numberOfGlyphs; i++) {\n if (post.glyphNameIndex[i] < standardNames.length) {\n this.names[i] = standardNames[post.glyphNameIndex[i]];\n } else {\n this.names[i] = post.names[post.glyphNameIndex[i] - standardNames.length];\n }\n }\n\n break;\n case 2.5:\n this.names = new Array(post.numberOfGlyphs);\n for (let i = 0; i < post.numberOfGlyphs; i++) {\n this.names[i] = standardNames[i + post.glyphNameIndex[i]];\n }\n\n break;\n case 3:\n this.names = [];\n break;\n default:\n this.names = [];\n break;\n }\n}\n\n/**\n * Gets the index of a glyph by name.\n * @param {string} name - The glyph name\n * @return {number} The index\n */\nGlyphNames.prototype.nameToGlyphIndex = function(name) {\n return this.names.indexOf(name);\n};\n\n/**\n * @param {number} gid\n * @return {string}\n */\nGlyphNames.prototype.glyphIndexToName = function(gid) {\n return this.names[gid];\n};\n\nfunction addGlyphNamesAll(font) {\n let glyph;\n const glyphIndexMap = font.tables.cmap.glyphIndexMap;\n const charCodes = Object.keys(glyphIndexMap);\n\n for (let i = 0; i < charCodes.length; i += 1) {\n const c = charCodes[i];\n const glyphIndex = glyphIndexMap[c];\n glyph = font.glyphs.get(glyphIndex);\n glyph.addUnicode(parseInt(c));\n }\n\n for (let i = 0; i < font.glyphs.length; i += 1) {\n glyph = font.glyphs.get(i);\n if (font.cffEncoding) {\n if (font.isCIDFont) {\n glyph.name = 'gid' + i;\n } else {\n glyph.name = font.cffEncoding.charset[i];\n }\n } else if (font.glyphNames.names) {\n glyph.name = font.glyphNames.glyphIndexToName(i);\n }\n }\n}\n\nfunction addGlyphNamesToUnicodeMap(font) {\n font._IndexToUnicodeMap = {};\n\n const glyphIndexMap = font.tables.cmap.glyphIndexMap;\n const charCodes = Object.keys(glyphIndexMap);\n\n for (let i = 0; i < charCodes.length; i += 1) {\n const c = charCodes[i];\n let glyphIndex = glyphIndexMap[c];\n if (font._IndexToUnicodeMap[glyphIndex] === undefined) {\n font._IndexToUnicodeMap[glyphIndex] = {\n unicodes: [parseInt(c)]\n };\n } else {\n font._IndexToUnicodeMap[glyphIndex].unicodes.push(parseInt(c));\n }\n }\n}\n\n/**\n * @alias opentype.addGlyphNames\n * @param {opentype.Font}\n * @param {Object}\n */\nfunction addGlyphNames(font, opt) {\n if (opt.lowMemory) {\n addGlyphNamesToUnicodeMap(font);\n } else {\n addGlyphNamesAll(font);\n }\n}\n\nexport {\n cffStandardStrings,\n cffStandardEncoding,\n cffExpertEncoding,\n standardNames,\n DefaultEncoding,\n CmapEncoding,\n CffEncoding,\n GlyphNames,\n addGlyphNames\n};\n","// Drawing utility functions.\n\n// Draw a line on the given context from point `x1,y1` to point `x2,y2`.\nfunction line(ctx, x1, y1, x2, y2) {\n ctx.beginPath();\n ctx.moveTo(x1, y1);\n ctx.lineTo(x2, y2);\n ctx.stroke();\n}\n\nexport default { line };\n","// The Glyph object\n\nimport check from './check';\nimport draw from './draw';\nimport Path from './path';\n// import glyf from './tables/glyf' Can't be imported here, because it's a circular dependency\n\nfunction getPathDefinition(glyph, path) {\n let _path = path || new Path();\n return {\n configurable: true,\n\n get: function() {\n if (typeof _path === 'function') {\n _path = _path();\n }\n\n return _path;\n },\n\n set: function(p) {\n _path = p;\n }\n };\n}\n/**\n * @typedef GlyphOptions\n * @type Object\n * @property {string} [name] - The glyph name\n * @property {number} [unicode]\n * @property {Array} [unicodes]\n * @property {number} [xMin]\n * @property {number} [yMin]\n * @property {number} [xMax]\n * @property {number} [yMax]\n * @property {number} [advanceWidth]\n */\n\n// A Glyph is an individual mark that often corresponds to a character.\n// Some glyphs, such as ligatures, are a combination of many characters.\n// Glyphs are the basic building blocks of a font.\n//\n// The `Glyph` class contains utility methods for drawing the path and its points.\n/**\n * @exports opentype.Glyph\n * @class\n * @param {GlyphOptions}\n * @constructor\n */\nfunction Glyph(options) {\n // By putting all the code on a prototype function (which is only declared once)\n // we reduce the memory requirements for larger fonts by some 2%\n this.bindConstructorValues(options);\n}\n\n/**\n * @param {GlyphOptions}\n */\nGlyph.prototype.bindConstructorValues = function(options) {\n this.index = options.index || 0;\n\n // These three values cannot be deferred for memory optimization:\n this.name = options.name || null;\n this.unicode = options.unicode || undefined;\n this.unicodes = options.unicodes || options.unicode !== undefined ? [options.unicode] : [];\n\n // But by binding these values only when necessary, we reduce can\n // the memory requirements by almost 3% for larger fonts.\n if ('xMin' in options) {\n this.xMin = options.xMin;\n }\n\n if ('yMin' in options) {\n this.yMin = options.yMin;\n }\n\n if ('xMax' in options) {\n this.xMax = options.xMax;\n }\n\n if ('yMax' in options) {\n this.yMax = options.yMax;\n }\n\n if ('advanceWidth' in options) {\n this.advanceWidth = options.advanceWidth;\n }\n\n // The path for a glyph is the most memory intensive, and is bound as a value\n // with a getter/setter to ensure we actually do path parsing only once the\n // path is actually needed by anything.\n Object.defineProperty(this, 'path', getPathDefinition(this, options.path));\n};\n\n/**\n * @param {number}\n */\nGlyph.prototype.addUnicode = function(unicode) {\n if (this.unicodes.length === 0) {\n this.unicode = unicode;\n }\n\n this.unicodes.push(unicode);\n};\n\n/**\n * Calculate the minimum bounding box for this glyph.\n * @return {opentype.BoundingBox}\n */\nGlyph.prototype.getBoundingBox = function() {\n return this.path.getBoundingBox();\n};\n\n/**\n * Convert the glyph to a Path we can draw on a drawing context.\n * @param {number} [x=0] - Horizontal position of the beginning of the text.\n * @param {number} [y=0] - Vertical position of the *baseline* of the text.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n * @param {Object=} options - xScale, yScale to stretch the glyph.\n * @param {opentype.Font} if hinting is to be used, the font\n * @return {opentype.Path}\n */\nGlyph.prototype.getPath = function(x, y, fontSize, options, font) {\n x = x !== undefined ? x : 0;\n y = y !== undefined ? y : 0;\n fontSize = fontSize !== undefined ? fontSize : 72;\n let commands;\n let hPoints;\n if (!options) options = { };\n let xScale = options.xScale;\n let yScale = options.yScale;\n\n if (options.hinting && font && font.hinting) {\n // in case of hinting, the hinting engine takes care\n // of scaling the points (not the path) before hinting.\n hPoints = this.path && font.hinting.exec(this, fontSize);\n // in case the hinting engine failed hPoints is undefined\n // and thus reverts to plain rending\n }\n\n if (hPoints) {\n // Call font.hinting.getCommands instead of `glyf.getPath(hPoints).commands` to avoid a circular dependency\n commands = font.hinting.getCommands(hPoints);\n x = Math.round(x);\n y = Math.round(y);\n // TODO in case of hinting xyScaling is not yet supported\n xScale = yScale = 1;\n } else {\n commands = this.path.commands;\n const scale = 1 / (this.path.unitsPerEm || 1000) * fontSize;\n if (xScale === undefined) xScale = scale;\n if (yScale === undefined) yScale = scale;\n }\n\n const p = new Path();\n for (let i = 0; i < commands.length; i += 1) {\n const cmd = commands[i];\n if (cmd.type === 'M') {\n p.moveTo(x + (cmd.x * xScale), y + (-cmd.y * yScale));\n } else if (cmd.type === 'L') {\n p.lineTo(x + (cmd.x * xScale), y + (-cmd.y * yScale));\n } else if (cmd.type === 'Q') {\n p.quadraticCurveTo(x + (cmd.x1 * xScale), y + (-cmd.y1 * yScale),\n x + (cmd.x * xScale), y + (-cmd.y * yScale));\n } else if (cmd.type === 'C') {\n p.curveTo(x + (cmd.x1 * xScale), y + (-cmd.y1 * yScale),\n x + (cmd.x2 * xScale), y + (-cmd.y2 * yScale),\n x + (cmd.x * xScale), y + (-cmd.y * yScale));\n } else if (cmd.type === 'Z') {\n p.closePath();\n }\n }\n\n return p;\n};\n\n/**\n * Split the glyph into contours.\n * This function is here for backwards compatibility, and to\n * provide raw access to the TrueType glyph outlines.\n * @return {Array}\n */\nGlyph.prototype.getContours = function() {\n if (this.points === undefined) {\n return [];\n }\n\n const contours = [];\n let currentContour = [];\n for (let i = 0; i < this.points.length; i += 1) {\n const pt = this.points[i];\n currentContour.push(pt);\n if (pt.lastPointOfContour) {\n contours.push(currentContour);\n currentContour = [];\n }\n }\n\n check.argument(currentContour.length === 0, 'There are still points left in the current contour.');\n return contours;\n};\n\n/**\n * Calculate the xMin/yMin/xMax/yMax/lsb/rsb for a Glyph.\n * @return {Object}\n */\nGlyph.prototype.getMetrics = function() {\n const commands = this.path.commands;\n const xCoords = [];\n const yCoords = [];\n for (let i = 0; i < commands.length; i += 1) {\n const cmd = commands[i];\n if (cmd.type !== 'Z') {\n xCoords.push(cmd.x);\n yCoords.push(cmd.y);\n }\n\n if (cmd.type === 'Q' || cmd.type === 'C') {\n xCoords.push(cmd.x1);\n yCoords.push(cmd.y1);\n }\n\n if (cmd.type === 'C') {\n xCoords.push(cmd.x2);\n yCoords.push(cmd.y2);\n }\n }\n\n const metrics = {\n xMin: Math.min.apply(null, xCoords),\n yMin: Math.min.apply(null, yCoords),\n xMax: Math.max.apply(null, xCoords),\n yMax: Math.max.apply(null, yCoords),\n leftSideBearing: this.leftSideBearing\n };\n\n if (!isFinite(metrics.xMin)) {\n metrics.xMin = 0;\n }\n\n if (!isFinite(metrics.xMax)) {\n metrics.xMax = this.advanceWidth;\n }\n\n if (!isFinite(metrics.yMin)) {\n metrics.yMin = 0;\n }\n\n if (!isFinite(metrics.yMax)) {\n metrics.yMax = 0;\n }\n\n metrics.rightSideBearing = this.advanceWidth - metrics.leftSideBearing - (metrics.xMax - metrics.xMin);\n return metrics;\n};\n\n/**\n * Draw the glyph on the given context.\n * @param {CanvasRenderingContext2D} ctx - A 2D drawing context, like Canvas.\n * @param {number} [x=0] - Horizontal position of the beginning of the text.\n * @param {number} [y=0] - Vertical position of the *baseline* of the text.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n * @param {Object=} options - xScale, yScale to stretch the glyph.\n */\nGlyph.prototype.draw = function(ctx, x, y, fontSize, options) {\n this.getPath(x, y, fontSize, options).draw(ctx);\n};\n\n/**\n * Draw the points of the glyph.\n * On-curve points will be drawn in blue, off-curve points will be drawn in red.\n * @param {CanvasRenderingContext2D} ctx - A 2D drawing context, like Canvas.\n * @param {number} [x=0] - Horizontal position of the beginning of the text.\n * @param {number} [y=0] - Vertical position of the *baseline* of the text.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n */\nGlyph.prototype.drawPoints = function(ctx, x, y, fontSize) {\n function drawCircles(l, x, y, scale) {\n ctx.beginPath();\n for (let j = 0; j < l.length; j += 1) {\n ctx.moveTo(x + (l[j].x * scale), y + (l[j].y * scale));\n ctx.arc(x + (l[j].x * scale), y + (l[j].y * scale), 2, 0, Math.PI * 2, false);\n }\n\n ctx.closePath();\n ctx.fill();\n }\n\n x = x !== undefined ? x : 0;\n y = y !== undefined ? y : 0;\n fontSize = fontSize !== undefined ? fontSize : 24;\n const scale = 1 / this.path.unitsPerEm * fontSize;\n\n const blueCircles = [];\n const redCircles = [];\n const path = this.path;\n for (let i = 0; i < path.commands.length; i += 1) {\n const cmd = path.commands[i];\n if (cmd.x !== undefined) {\n blueCircles.push({x: cmd.x, y: -cmd.y});\n }\n\n if (cmd.x1 !== undefined) {\n redCircles.push({x: cmd.x1, y: -cmd.y1});\n }\n\n if (cmd.x2 !== undefined) {\n redCircles.push({x: cmd.x2, y: -cmd.y2});\n }\n }\n\n ctx.fillStyle = 'blue';\n drawCircles(blueCircles, x, y, scale);\n ctx.fillStyle = 'red';\n drawCircles(redCircles, x, y, scale);\n};\n\n/**\n * Draw lines indicating important font measurements.\n * Black lines indicate the origin of the coordinate system (point 0,0).\n * Blue lines indicate the glyph bounding box.\n * Green line indicates the advance width of the glyph.\n * @param {CanvasRenderingContext2D} ctx - A 2D drawing context, like Canvas.\n * @param {number} [x=0] - Horizontal position of the beginning of the text.\n * @param {number} [y=0] - Vertical position of the *baseline* of the text.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n */\nGlyph.prototype.drawMetrics = function(ctx, x, y, fontSize) {\n let scale;\n x = x !== undefined ? x : 0;\n y = y !== undefined ? y : 0;\n fontSize = fontSize !== undefined ? fontSize : 24;\n scale = 1 / this.path.unitsPerEm * fontSize;\n ctx.lineWidth = 1;\n\n // Draw the origin\n ctx.strokeStyle = 'black';\n draw.line(ctx, x, -10000, x, 10000);\n draw.line(ctx, -10000, y, 10000, y);\n\n // This code is here due to memory optimization: by not using\n // defaults in the constructor, we save a notable amount of memory.\n const xMin = this.xMin || 0;\n let yMin = this.yMin || 0;\n const xMax = this.xMax || 0;\n let yMax = this.yMax || 0;\n const advanceWidth = this.advanceWidth || 0;\n\n // Draw the glyph box\n ctx.strokeStyle = 'blue';\n draw.line(ctx, x + (xMin * scale), -10000, x + (xMin * scale), 10000);\n draw.line(ctx, x + (xMax * scale), -10000, x + (xMax * scale), 10000);\n draw.line(ctx, -10000, y + (-yMin * scale), 10000, y + (-yMin * scale));\n draw.line(ctx, -10000, y + (-yMax * scale), 10000, y + (-yMax * scale));\n\n // Draw the advance width\n ctx.strokeStyle = 'green';\n draw.line(ctx, x + (advanceWidth * scale), -10000, x + (advanceWidth * scale), 10000);\n};\n\nexport default Glyph;\n","// The GlyphSet object\n\nimport Glyph from './glyph';\n\n// Define a property on the glyph that depends on the path being loaded.\nfunction defineDependentProperty(glyph, externalName, internalName) {\n Object.defineProperty(glyph, externalName, {\n get: function() {\n // Request the path property to make sure the path is loaded.\n glyph.path; // jshint ignore:line\n return glyph[internalName];\n },\n set: function(newValue) {\n glyph[internalName] = newValue;\n },\n enumerable: true,\n configurable: true\n });\n}\n\n/**\n * A GlyphSet represents all glyphs available in the font, but modelled using\n * a deferred glyph loader, for retrieving glyphs only once they are absolutely\n * necessary, to keep the memory footprint down.\n * @exports opentype.GlyphSet\n * @class\n * @param {opentype.Font}\n * @param {Array}\n */\nfunction GlyphSet(font, glyphs) {\n this.font = font;\n this.glyphs = {};\n if (Array.isArray(glyphs)) {\n for (let i = 0; i < glyphs.length; i++) {\n const glyph = glyphs[i];\n glyph.path.unitsPerEm = font.unitsPerEm;\n this.glyphs[i] = glyph;\n }\n }\n\n this.length = (glyphs && glyphs.length) || 0;\n}\n\n/**\n * @param {number} index\n * @return {opentype.Glyph}\n */\nGlyphSet.prototype.get = function(index) {\n // this.glyphs[index] is 'undefined' when low memory mode is on. glyph is pushed on request only.\n if (this.glyphs[index] === undefined) {\n this.font._push(index);\n if (typeof this.glyphs[index] === 'function') {\n this.glyphs[index] = this.glyphs[index]();\n }\n\n let glyph = this.glyphs[index];\n let unicodeObj = this.font._IndexToUnicodeMap[index];\n\n if (unicodeObj) {\n for (let j = 0; j < unicodeObj.unicodes.length; j++)\n glyph.addUnicode(unicodeObj.unicodes[j]);\n }\n\n if (this.font.cffEncoding) {\n if (this.font.isCIDFont) {\n glyph.name = 'gid' + index;\n } else {\n glyph.name = this.font.cffEncoding.charset[index];\n }\n } else if (this.font.glyphNames.names) {\n glyph.name = this.font.glyphNames.glyphIndexToName(index);\n }\n\n this.glyphs[index].advanceWidth = this.font._hmtxTableData[index].advanceWidth;\n this.glyphs[index].leftSideBearing = this.font._hmtxTableData[index].leftSideBearing;\n } else {\n if (typeof this.glyphs[index] === 'function') {\n this.glyphs[index] = this.glyphs[index]();\n }\n }\n\n return this.glyphs[index];\n};\n\n/**\n * @param {number} index\n * @param {Object}\n */\nGlyphSet.prototype.push = function(index, loader) {\n this.glyphs[index] = loader;\n this.length++;\n};\n\n/**\n * @alias opentype.glyphLoader\n * @param {opentype.Font} font\n * @param {number} index\n * @return {opentype.Glyph}\n */\nfunction glyphLoader(font, index) {\n return new Glyph({index: index, font: font});\n}\n\n/**\n * Generate a stub glyph that can be filled with all metadata *except*\n * the \"points\" and \"path\" properties, which must be loaded only once\n * the glyph's path is actually requested for text shaping.\n * @alias opentype.ttfGlyphLoader\n * @param {opentype.Font} font\n * @param {number} index\n * @param {Function} parseGlyph\n * @param {Object} data\n * @param {number} position\n * @param {Function} buildPath\n * @return {opentype.Glyph}\n */\nfunction ttfGlyphLoader(font, index, parseGlyph, data, position, buildPath) {\n return function() {\n const glyph = new Glyph({index: index, font: font});\n\n glyph.path = function() {\n parseGlyph(glyph, data, position);\n const path = buildPath(font.glyphs, glyph);\n path.unitsPerEm = font.unitsPerEm;\n return path;\n };\n\n defineDependentProperty(glyph, 'xMin', '_xMin');\n defineDependentProperty(glyph, 'xMax', '_xMax');\n defineDependentProperty(glyph, 'yMin', '_yMin');\n defineDependentProperty(glyph, 'yMax', '_yMax');\n\n return glyph;\n };\n}\n/**\n * @alias opentype.cffGlyphLoader\n * @param {opentype.Font} font\n * @param {number} index\n * @param {Function} parseCFFCharstring\n * @param {string} charstring\n * @return {opentype.Glyph}\n */\nfunction cffGlyphLoader(font, index, parseCFFCharstring, charstring) {\n return function() {\n const glyph = new Glyph({index: index, font: font});\n\n glyph.path = function() {\n const path = parseCFFCharstring(font, glyph, charstring);\n path.unitsPerEm = font.unitsPerEm;\n return path;\n };\n\n return glyph;\n };\n}\n\nexport default { GlyphSet, glyphLoader, ttfGlyphLoader, cffGlyphLoader };\n","// The `CFF` table contains the glyph outlines in PostScript format.\n// https://www.microsoft.com/typography/OTSPEC/cff.htm\n// http://download.microsoft.com/download/8/0/1/801a191c-029d-4af3-9642-555f6fe514ee/cff.pdf\n// http://download.microsoft.com/download/8/0/1/801a191c-029d-4af3-9642-555f6fe514ee/type2.pdf\n\nimport { CffEncoding, cffStandardEncoding, cffExpertEncoding, cffStandardStrings } from '../encoding';\nimport glyphset from '../glyphset';\nimport parse from '../parse';\nimport Path from '../path';\nimport table from '../table';\n\n// Custom equals function that can also check lists.\nfunction equals(a, b) {\n if (a === b) {\n return true;\n } else if (Array.isArray(a) && Array.isArray(b)) {\n if (a.length !== b.length) {\n return false;\n }\n\n for (let i = 0; i < a.length; i += 1) {\n if (!equals(a[i], b[i])) {\n return false;\n }\n }\n\n return true;\n } else {\n return false;\n }\n}\n\n// Subroutines are encoded using the negative half of the number space.\n// See type 2 chapter 4.7 \"Subroutine operators\".\nfunction calcCFFSubroutineBias(subrs) {\n let bias;\n if (subrs.length < 1240) {\n bias = 107;\n } else if (subrs.length < 33900) {\n bias = 1131;\n } else {\n bias = 32768;\n }\n\n return bias;\n}\n\n// Parse a `CFF` INDEX array.\n// An index array consists of a list of offsets, then a list of objects at those offsets.\nfunction parseCFFIndex(data, start, conversionFn) {\n const offsets = [];\n const objects = [];\n const count = parse.getCard16(data, start);\n let objectOffset;\n let endOffset;\n if (count !== 0) {\n const offsetSize = parse.getByte(data, start + 2);\n objectOffset = start + ((count + 1) * offsetSize) + 2;\n let pos = start + 3;\n for (let i = 0; i < count + 1; i += 1) {\n offsets.push(parse.getOffset(data, pos, offsetSize));\n pos += offsetSize;\n }\n\n // The total size of the index array is 4 header bytes + the value of the last offset.\n endOffset = objectOffset + offsets[count];\n } else {\n endOffset = start + 2;\n }\n\n for (let i = 0; i < offsets.length - 1; i += 1) {\n let value = parse.getBytes(data, objectOffset + offsets[i], objectOffset + offsets[i + 1]);\n if (conversionFn) {\n value = conversionFn(value);\n }\n\n objects.push(value);\n }\n\n return {objects: objects, startOffset: start, endOffset: endOffset};\n}\n\nfunction parseCFFIndexLowMemory(data, start) {\n const offsets = [];\n const count = parse.getCard16(data, start);\n let objectOffset;\n let endOffset;\n if (count !== 0) {\n const offsetSize = parse.getByte(data, start + 2);\n objectOffset = start + ((count + 1) * offsetSize) + 2;\n let pos = start + 3;\n for (let i = 0; i < count + 1; i += 1) {\n offsets.push(parse.getOffset(data, pos, offsetSize));\n pos += offsetSize;\n }\n\n // The total size of the index array is 4 header bytes + the value of the last offset.\n endOffset = objectOffset + offsets[count];\n } else {\n endOffset = start + 2;\n }\n\n return {offsets: offsets, startOffset: start, endOffset: endOffset};\n}\nfunction getCffIndexObject(i, offsets, data, start, conversionFn) {\n const count = parse.getCard16(data, start);\n let objectOffset = 0;\n if (count !== 0) {\n const offsetSize = parse.getByte(data, start + 2);\n objectOffset = start + ((count + 1) * offsetSize) + 2;\n }\n\n let value = parse.getBytes(data, objectOffset + offsets[i], objectOffset + offsets[i + 1]);\n if (conversionFn) {\n value = conversionFn(value);\n }\n return value;\n}\n\n// Parse a `CFF` DICT real value.\nfunction parseFloatOperand(parser) {\n let s = '';\n const eof = 15;\n const lookup = ['0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '.', 'E', 'E-', null, '-'];\n while (true) {\n const b = parser.parseByte();\n const n1 = b >> 4;\n const n2 = b & 15;\n\n if (n1 === eof) {\n break;\n }\n\n s += lookup[n1];\n\n if (n2 === eof) {\n break;\n }\n\n s += lookup[n2];\n }\n\n return parseFloat(s);\n}\n\n// Parse a `CFF` DICT operand.\nfunction parseOperand(parser, b0) {\n let b1;\n let b2;\n let b3;\n let b4;\n if (b0 === 28) {\n b1 = parser.parseByte();\n b2 = parser.parseByte();\n return b1 << 8 | b2;\n }\n\n if (b0 === 29) {\n b1 = parser.parseByte();\n b2 = parser.parseByte();\n b3 = parser.parseByte();\n b4 = parser.parseByte();\n return b1 << 24 | b2 << 16 | b3 << 8 | b4;\n }\n\n if (b0 === 30) {\n return parseFloatOperand(parser);\n }\n\n if (b0 >= 32 && b0 <= 246) {\n return b0 - 139;\n }\n\n if (b0 >= 247 && b0 <= 250) {\n b1 = parser.parseByte();\n return (b0 - 247) * 256 + b1 + 108;\n }\n\n if (b0 >= 251 && b0 <= 254) {\n b1 = parser.parseByte();\n return -(b0 - 251) * 256 - b1 - 108;\n }\n\n throw new Error('Invalid b0 ' + b0);\n}\n\n// Convert the entries returned by `parseDict` to a proper dictionary.\n// If a value is a list of one, it is unpacked.\nfunction entriesToObject(entries) {\n const o = {};\n for (let i = 0; i < entries.length; i += 1) {\n const key = entries[i][0];\n const values = entries[i][1];\n let value;\n if (values.length === 1) {\n value = values[0];\n } else {\n value = values;\n }\n\n if (o.hasOwnProperty(key) && !isNaN(o[key])) {\n throw new Error('Object ' + o + ' already has key ' + key);\n }\n\n o[key] = value;\n }\n\n return o;\n}\n\n// Parse a `CFF` DICT object.\n// A dictionary contains key-value pairs in a compact tokenized format.\nfunction parseCFFDict(data, start, size) {\n start = start !== undefined ? start : 0;\n const parser = new parse.Parser(data, start);\n const entries = [];\n let operands = [];\n size = size !== undefined ? size : data.length;\n\n while (parser.relativeOffset < size) {\n let op = parser.parseByte();\n\n // The first byte for each dict item distinguishes between operator (key) and operand (value).\n // Values <= 21 are operators.\n if (op <= 21) {\n // Two-byte operators have an initial escape byte of 12.\n if (op === 12) {\n op = 1200 + parser.parseByte();\n }\n\n entries.push([op, operands]);\n operands = [];\n } else {\n // Since the operands (values) come before the operators (keys), we store all operands in a list\n // until we encounter an operator.\n operands.push(parseOperand(parser, op));\n }\n }\n\n return entriesToObject(entries);\n}\n\n// Given a String Index (SID), return the value of the string.\n// Strings below index 392 are standard CFF strings and are not encoded in the font.\nfunction getCFFString(strings, index) {\n if (index <= 390) {\n index = cffStandardStrings[index];\n } else {\n index = strings[index - 391];\n }\n\n return index;\n}\n\n// Interpret a dictionary and return a new dictionary with readable keys and values for missing entries.\n// This function takes `meta` which is a list of objects containing `operand`, `name` and `default`.\nfunction interpretDict(dict, meta, strings) {\n const newDict = {};\n let value;\n\n // Because we also want to include missing values, we start out from the meta list\n // and lookup values in the dict.\n for (let i = 0; i < meta.length; i += 1) {\n const m = meta[i];\n\n if (Array.isArray(m.type)) {\n const values = [];\n values.length = m.type.length;\n for (let j = 0; j < m.type.length; j++) {\n value = dict[m.op] !== undefined ? dict[m.op][j] : undefined;\n if (value === undefined) {\n value = m.value !== undefined && m.value[j] !== undefined ? m.value[j] : null;\n }\n if (m.type[j] === 'SID') {\n value = getCFFString(strings, value);\n }\n values[j] = value;\n }\n newDict[m.name] = values;\n } else {\n value = dict[m.op];\n if (value === undefined) {\n value = m.value !== undefined ? m.value : null;\n }\n\n if (m.type === 'SID') {\n value = getCFFString(strings, value);\n }\n newDict[m.name] = value;\n }\n }\n\n return newDict;\n}\n\n// Parse the CFF header.\nfunction parseCFFHeader(data, start) {\n const header = {};\n header.formatMajor = parse.getCard8(data, start);\n header.formatMinor = parse.getCard8(data, start + 1);\n header.size = parse.getCard8(data, start + 2);\n header.offsetSize = parse.getCard8(data, start + 3);\n header.startOffset = start;\n header.endOffset = start + 4;\n return header;\n}\n\nconst TOP_DICT_META = [\n {name: 'version', op: 0, type: 'SID'},\n {name: 'notice', op: 1, type: 'SID'},\n {name: 'copyright', op: 1200, type: 'SID'},\n {name: 'fullName', op: 2, type: 'SID'},\n {name: 'familyName', op: 3, type: 'SID'},\n {name: 'weight', op: 4, type: 'SID'},\n {name: 'isFixedPitch', op: 1201, type: 'number', value: 0},\n {name: 'italicAngle', op: 1202, type: 'number', value: 0},\n {name: 'underlinePosition', op: 1203, type: 'number', value: -100},\n {name: 'underlineThickness', op: 1204, type: 'number', value: 50},\n {name: 'paintType', op: 1205, type: 'number', value: 0},\n {name: 'charstringType', op: 1206, type: 'number', value: 2},\n {\n name: 'fontMatrix',\n op: 1207,\n type: ['real', 'real', 'real', 'real', 'real', 'real'],\n value: [0.001, 0, 0, 0.001, 0, 0]\n },\n {name: 'uniqueId', op: 13, type: 'number'},\n {name: 'fontBBox', op: 5, type: ['number', 'number', 'number', 'number'], value: [0, 0, 0, 0]},\n {name: 'strokeWidth', op: 1208, type: 'number', value: 0},\n {name: 'xuid', op: 14, type: [], value: null},\n {name: 'charset', op: 15, type: 'offset', value: 0},\n {name: 'encoding', op: 16, type: 'offset', value: 0},\n {name: 'charStrings', op: 17, type: 'offset', value: 0},\n {name: 'private', op: 18, type: ['number', 'offset'], value: [0, 0]},\n {name: 'ros', op: 1230, type: ['SID', 'SID', 'number']},\n {name: 'cidFontVersion', op: 1231, type: 'number', value: 0},\n {name: 'cidFontRevision', op: 1232, type: 'number', value: 0},\n {name: 'cidFontType', op: 1233, type: 'number', value: 0},\n {name: 'cidCount', op: 1234, type: 'number', value: 8720},\n {name: 'uidBase', op: 1235, type: 'number'},\n {name: 'fdArray', op: 1236, type: 'offset'},\n {name: 'fdSelect', op: 1237, type: 'offset'},\n {name: 'fontName', op: 1238, type: 'SID'}\n];\n\nconst PRIVATE_DICT_META = [\n {name: 'subrs', op: 19, type: 'offset', value: 0},\n {name: 'defaultWidthX', op: 20, type: 'number', value: 0},\n {name: 'nominalWidthX', op: 21, type: 'number', value: 0}\n];\n\n// Parse the CFF top dictionary. A CFF table can contain multiple fonts, each with their own top dictionary.\n// The top dictionary contains the essential metadata for the font, together with the private dictionary.\nfunction parseCFFTopDict(data, strings) {\n const dict = parseCFFDict(data, 0, data.byteLength);\n return interpretDict(dict, TOP_DICT_META, strings);\n}\n\n// Parse the CFF private dictionary. We don't fully parse out all the values, only the ones we need.\nfunction parseCFFPrivateDict(data, start, size, strings) {\n const dict = parseCFFDict(data, start, size);\n return interpretDict(dict, PRIVATE_DICT_META, strings);\n}\n\n// Returns a list of \"Top DICT\"s found using an INDEX list.\n// Used to read both the usual high-level Top DICTs and also the FDArray\n// discovered inside CID-keyed fonts. When a Top DICT has a reference to\n// a Private DICT that is read and saved into the Top DICT.\n//\n// In addition to the expected/optional values as outlined in TOP_DICT_META\n// the following values might be saved into the Top DICT.\n//\n// _subrs [] array of local CFF subroutines from Private DICT\n// _subrsBias bias value computed from number of subroutines\n// (see calcCFFSubroutineBias() and parseCFFCharstring())\n// _defaultWidthX default widths for CFF characters\n// _nominalWidthX bias added to width embedded within glyph description\n//\n// _privateDict saved copy of parsed Private DICT from Top DICT\nfunction gatherCFFTopDicts(data, start, cffIndex, strings) {\n const topDictArray = [];\n for (let iTopDict = 0; iTopDict < cffIndex.length; iTopDict += 1) {\n const topDictData = new DataView(new Uint8Array(cffIndex[iTopDict]).buffer);\n const topDict = parseCFFTopDict(topDictData, strings);\n topDict._subrs = [];\n topDict._subrsBias = 0;\n topDict._defaultWidthX = 0;\n topDict._nominalWidthX = 0;\n const privateSize = topDict.private[0];\n const privateOffset = topDict.private[1];\n if (privateSize !== 0 && privateOffset !== 0) {\n const privateDict = parseCFFPrivateDict(data, privateOffset + start, privateSize, strings);\n topDict._defaultWidthX = privateDict.defaultWidthX;\n topDict._nominalWidthX = privateDict.nominalWidthX;\n if (privateDict.subrs !== 0) {\n const subrOffset = privateOffset + privateDict.subrs;\n const subrIndex = parseCFFIndex(data, subrOffset + start);\n topDict._subrs = subrIndex.objects;\n topDict._subrsBias = calcCFFSubroutineBias(topDict._subrs);\n }\n topDict._privateDict = privateDict;\n }\n topDictArray.push(topDict);\n }\n return topDictArray;\n}\n\n// Parse the CFF charset table, which contains internal names for all the glyphs.\n// This function will return a list of glyph names.\n// See Adobe TN #5176 chapter 13, \"Charsets\".\nfunction parseCFFCharset(data, start, nGlyphs, strings) {\n let sid;\n let count;\n const parser = new parse.Parser(data, start);\n\n // The .notdef glyph is not included, so subtract 1.\n nGlyphs -= 1;\n const charset = ['.notdef'];\n\n const format = parser.parseCard8();\n if (format === 0) {\n for (let i = 0; i < nGlyphs; i += 1) {\n sid = parser.parseSID();\n charset.push(getCFFString(strings, sid));\n }\n } else if (format === 1) {\n while (charset.length <= nGlyphs) {\n sid = parser.parseSID();\n count = parser.parseCard8();\n for (let i = 0; i <= count; i += 1) {\n charset.push(getCFFString(strings, sid));\n sid += 1;\n }\n }\n } else if (format === 2) {\n while (charset.length <= nGlyphs) {\n sid = parser.parseSID();\n count = parser.parseCard16();\n for (let i = 0; i <= count; i += 1) {\n charset.push(getCFFString(strings, sid));\n sid += 1;\n }\n }\n } else {\n throw new Error('Unknown charset format ' + format);\n }\n\n return charset;\n}\n\n// Parse the CFF encoding data. Only one encoding can be specified per font.\n// See Adobe TN #5176 chapter 12, \"Encodings\".\nfunction parseCFFEncoding(data, start, charset) {\n let code;\n const enc = {};\n const parser = new parse.Parser(data, start);\n const format = parser.parseCard8();\n if (format === 0) {\n const nCodes = parser.parseCard8();\n for (let i = 0; i < nCodes; i += 1) {\n code = parser.parseCard8();\n enc[code] = i;\n }\n } else if (format === 1) {\n const nRanges = parser.parseCard8();\n code = 1;\n for (let i = 0; i < nRanges; i += 1) {\n const first = parser.parseCard8();\n const nLeft = parser.parseCard8();\n for (let j = first; j <= first + nLeft; j += 1) {\n enc[j] = code;\n code += 1;\n }\n }\n } else {\n throw new Error('Unknown encoding format ' + format);\n }\n\n return new CffEncoding(enc, charset);\n}\n\n// Take in charstring code and return a Glyph object.\n// The encoding is described in the Type 2 Charstring Format\n// https://www.microsoft.com/typography/OTSPEC/charstr2.htm\nfunction parseCFFCharstring(font, glyph, code) {\n let c1x;\n let c1y;\n let c2x;\n let c2y;\n const p = new Path();\n const stack = [];\n let nStems = 0;\n let haveWidth = false;\n let open = false;\n let x = 0;\n let y = 0;\n let subrs;\n let subrsBias;\n let defaultWidthX;\n let nominalWidthX;\n if (font.isCIDFont) {\n const fdIndex = font.tables.cff.topDict._fdSelect[glyph.index];\n const fdDict = font.tables.cff.topDict._fdArray[fdIndex];\n subrs = fdDict._subrs;\n subrsBias = fdDict._subrsBias;\n defaultWidthX = fdDict._defaultWidthX;\n nominalWidthX = fdDict._nominalWidthX;\n } else {\n subrs = font.tables.cff.topDict._subrs;\n subrsBias = font.tables.cff.topDict._subrsBias;\n defaultWidthX = font.tables.cff.topDict._defaultWidthX;\n nominalWidthX = font.tables.cff.topDict._nominalWidthX;\n }\n let width = defaultWidthX;\n\n function newContour(x, y) {\n if (open) {\n p.closePath();\n }\n\n p.moveTo(x, y);\n open = true;\n }\n\n function parseStems() {\n let hasWidthArg;\n\n // The number of stem operators on the stack is always even.\n // If the value is uneven, that means a width is specified.\n hasWidthArg = stack.length % 2 !== 0;\n if (hasWidthArg && !haveWidth) {\n width = stack.shift() + nominalWidthX;\n }\n\n nStems += stack.length >> 1;\n stack.length = 0;\n haveWidth = true;\n }\n\n function parse(code) {\n let b1;\n let b2;\n let b3;\n let b4;\n let codeIndex;\n let subrCode;\n let jpx;\n let jpy;\n let c3x;\n let c3y;\n let c4x;\n let c4y;\n\n let i = 0;\n while (i < code.length) {\n let v = code[i];\n i += 1;\n switch (v) {\n case 1: // hstem\n parseStems();\n break;\n case 3: // vstem\n parseStems();\n break;\n case 4: // vmoveto\n if (stack.length > 1 && !haveWidth) {\n width = stack.shift() + nominalWidthX;\n haveWidth = true;\n }\n\n y += stack.pop();\n newContour(x, y);\n break;\n case 5: // rlineto\n while (stack.length > 0) {\n x += stack.shift();\n y += stack.shift();\n p.lineTo(x, y);\n }\n\n break;\n case 6: // hlineto\n while (stack.length > 0) {\n x += stack.shift();\n p.lineTo(x, y);\n if (stack.length === 0) {\n break;\n }\n\n y += stack.shift();\n p.lineTo(x, y);\n }\n\n break;\n case 7: // vlineto\n while (stack.length > 0) {\n y += stack.shift();\n p.lineTo(x, y);\n if (stack.length === 0) {\n break;\n }\n\n x += stack.shift();\n p.lineTo(x, y);\n }\n\n break;\n case 8: // rrcurveto\n while (stack.length > 0) {\n c1x = x + stack.shift();\n c1y = y + stack.shift();\n c2x = c1x + stack.shift();\n c2y = c1y + stack.shift();\n x = c2x + stack.shift();\n y = c2y + stack.shift();\n p.curveTo(c1x, c1y, c2x, c2y, x, y);\n }\n\n break;\n case 10: // callsubr\n codeIndex = stack.pop() + subrsBias;\n subrCode = subrs[codeIndex];\n if (subrCode) {\n parse(subrCode);\n }\n\n break;\n case 11: // return\n return;\n case 12: // flex operators\n v = code[i];\n i += 1;\n switch (v) {\n case 35: // flex\n // |- dx1 dy1 dx2 dy2 dx3 dy3 dx4 dy4 dx5 dy5 dx6 dy6 fd flex (12 35) |-\n c1x = x + stack.shift(); // dx1\n c1y = y + stack.shift(); // dy1\n c2x = c1x + stack.shift(); // dx2\n c2y = c1y + stack.shift(); // dy2\n jpx = c2x + stack.shift(); // dx3\n jpy = c2y + stack.shift(); // dy3\n c3x = jpx + stack.shift(); // dx4\n c3y = jpy + stack.shift(); // dy4\n c4x = c3x + stack.shift(); // dx5\n c4y = c3y + stack.shift(); // dy5\n x = c4x + stack.shift(); // dx6\n y = c4y + stack.shift(); // dy6\n stack.shift(); // flex depth\n p.curveTo(c1x, c1y, c2x, c2y, jpx, jpy);\n p.curveTo(c3x, c3y, c4x, c4y, x, y);\n break;\n case 34: // hflex\n // |- dx1 dx2 dy2 dx3 dx4 dx5 dx6 hflex (12 34) |-\n c1x = x + stack.shift(); // dx1\n c1y = y; // dy1\n c2x = c1x + stack.shift(); // dx2\n c2y = c1y + stack.shift(); // dy2\n jpx = c2x + stack.shift(); // dx3\n jpy = c2y; // dy3\n c3x = jpx + stack.shift(); // dx4\n c3y = c2y; // dy4\n c4x = c3x + stack.shift(); // dx5\n c4y = y; // dy5\n x = c4x + stack.shift(); // dx6\n p.curveTo(c1x, c1y, c2x, c2y, jpx, jpy);\n p.curveTo(c3x, c3y, c4x, c4y, x, y);\n break;\n case 36: // hflex1\n // |- dx1 dy1 dx2 dy2 dx3 dx4 dx5 dy5 dx6 hflex1 (12 36) |-\n c1x = x + stack.shift(); // dx1\n c1y = y + stack.shift(); // dy1\n c2x = c1x + stack.shift(); // dx2\n c2y = c1y + stack.shift(); // dy2\n jpx = c2x + stack.shift(); // dx3\n jpy = c2y; // dy3\n c3x = jpx + stack.shift(); // dx4\n c3y = c2y; // dy4\n c4x = c3x + stack.shift(); // dx5\n c4y = c3y + stack.shift(); // dy5\n x = c4x + stack.shift(); // dx6\n p.curveTo(c1x, c1y, c2x, c2y, jpx, jpy);\n p.curveTo(c3x, c3y, c4x, c4y, x, y);\n break;\n case 37: // flex1\n // |- dx1 dy1 dx2 dy2 dx3 dy3 dx4 dy4 dx5 dy5 d6 flex1 (12 37) |-\n c1x = x + stack.shift(); // dx1\n c1y = y + stack.shift(); // dy1\n c2x = c1x + stack.shift(); // dx2\n c2y = c1y + stack.shift(); // dy2\n jpx = c2x + stack.shift(); // dx3\n jpy = c2y + stack.shift(); // dy3\n c3x = jpx + stack.shift(); // dx4\n c3y = jpy + stack.shift(); // dy4\n c4x = c3x + stack.shift(); // dx5\n c4y = c3y + stack.shift(); // dy5\n if (Math.abs(c4x - x) > Math.abs(c4y - y)) {\n x = c4x + stack.shift();\n } else {\n y = c4y + stack.shift();\n }\n\n p.curveTo(c1x, c1y, c2x, c2y, jpx, jpy);\n p.curveTo(c3x, c3y, c4x, c4y, x, y);\n break;\n default:\n console.log('Glyph ' + glyph.index + ': unknown operator ' + 1200 + v);\n stack.length = 0;\n }\n break;\n case 14: // endchar\n if (stack.length > 0 && !haveWidth) {\n width = stack.shift() + nominalWidthX;\n haveWidth = true;\n }\n\n if (open) {\n p.closePath();\n open = false;\n }\n\n break;\n case 18: // hstemhm\n parseStems();\n break;\n case 19: // hintmask\n case 20: // cntrmask\n parseStems();\n i += (nStems + 7) >> 3;\n break;\n case 21: // rmoveto\n if (stack.length > 2 && !haveWidth) {\n width = stack.shift() + nominalWidthX;\n haveWidth = true;\n }\n\n y += stack.pop();\n x += stack.pop();\n newContour(x, y);\n break;\n case 22: // hmoveto\n if (stack.length > 1 && !haveWidth) {\n width = stack.shift() + nominalWidthX;\n haveWidth = true;\n }\n\n x += stack.pop();\n newContour(x, y);\n break;\n case 23: // vstemhm\n parseStems();\n break;\n case 24: // rcurveline\n while (stack.length > 2) {\n c1x = x + stack.shift();\n c1y = y + stack.shift();\n c2x = c1x + stack.shift();\n c2y = c1y + stack.shift();\n x = c2x + stack.shift();\n y = c2y + stack.shift();\n p.curveTo(c1x, c1y, c2x, c2y, x, y);\n }\n\n x += stack.shift();\n y += stack.shift();\n p.lineTo(x, y);\n break;\n case 25: // rlinecurve\n while (stack.length > 6) {\n x += stack.shift();\n y += stack.shift();\n p.lineTo(x, y);\n }\n\n c1x = x + stack.shift();\n c1y = y + stack.shift();\n c2x = c1x + stack.shift();\n c2y = c1y + stack.shift();\n x = c2x + stack.shift();\n y = c2y + stack.shift();\n p.curveTo(c1x, c1y, c2x, c2y, x, y);\n break;\n case 26: // vvcurveto\n if (stack.length % 2) {\n x += stack.shift();\n }\n\n while (stack.length > 0) {\n c1x = x;\n c1y = y + stack.shift();\n c2x = c1x + stack.shift();\n c2y = c1y + stack.shift();\n x = c2x;\n y = c2y + stack.shift();\n p.curveTo(c1x, c1y, c2x, c2y, x, y);\n }\n\n break;\n case 27: // hhcurveto\n if (stack.length % 2) {\n y += stack.shift();\n }\n\n while (stack.length > 0) {\n c1x = x + stack.shift();\n c1y = y;\n c2x = c1x + stack.shift();\n c2y = c1y + stack.shift();\n x = c2x + stack.shift();\n y = c2y;\n p.curveTo(c1x, c1y, c2x, c2y, x, y);\n }\n\n break;\n case 28: // shortint\n b1 = code[i];\n b2 = code[i + 1];\n stack.push(((b1 << 24) | (b2 << 16)) >> 16);\n i += 2;\n break;\n case 29: // callgsubr\n codeIndex = stack.pop() + font.gsubrsBias;\n subrCode = font.gsubrs[codeIndex];\n if (subrCode) {\n parse(subrCode);\n }\n\n break;\n case 30: // vhcurveto\n while (stack.length > 0) {\n c1x = x;\n c1y = y + stack.shift();\n c2x = c1x + stack.shift();\n c2y = c1y + stack.shift();\n x = c2x + stack.shift();\n y = c2y + (stack.length === 1 ? stack.shift() : 0);\n p.curveTo(c1x, c1y, c2x, c2y, x, y);\n if (stack.length === 0) {\n break;\n }\n\n c1x = x + stack.shift();\n c1y = y;\n c2x = c1x + stack.shift();\n c2y = c1y + stack.shift();\n y = c2y + stack.shift();\n x = c2x + (stack.length === 1 ? stack.shift() : 0);\n p.curveTo(c1x, c1y, c2x, c2y, x, y);\n }\n\n break;\n case 31: // hvcurveto\n while (stack.length > 0) {\n c1x = x + stack.shift();\n c1y = y;\n c2x = c1x + stack.shift();\n c2y = c1y + stack.shift();\n y = c2y + stack.shift();\n x = c2x + (stack.length === 1 ? stack.shift() : 0);\n p.curveTo(c1x, c1y, c2x, c2y, x, y);\n if (stack.length === 0) {\n break;\n }\n\n c1x = x;\n c1y = y + stack.shift();\n c2x = c1x + stack.shift();\n c2y = c1y + stack.shift();\n x = c2x + stack.shift();\n y = c2y + (stack.length === 1 ? stack.shift() : 0);\n p.curveTo(c1x, c1y, c2x, c2y, x, y);\n }\n\n break;\n default:\n if (v < 32) {\n console.log('Glyph ' + glyph.index + ': unknown operator ' + v);\n } else if (v < 247) {\n stack.push(v - 139);\n } else if (v < 251) {\n b1 = code[i];\n i += 1;\n stack.push((v - 247) * 256 + b1 + 108);\n } else if (v < 255) {\n b1 = code[i];\n i += 1;\n stack.push(-(v - 251) * 256 - b1 - 108);\n } else {\n b1 = code[i];\n b2 = code[i + 1];\n b3 = code[i + 2];\n b4 = code[i + 3];\n i += 4;\n stack.push(((b1 << 24) | (b2 << 16) | (b3 << 8) | b4) / 65536);\n }\n }\n }\n }\n\n parse(code);\n\n glyph.advanceWidth = width;\n return p;\n}\n\nfunction parseCFFFDSelect(data, start, nGlyphs, fdArrayCount) {\n const fdSelect = [];\n let fdIndex;\n const parser = new parse.Parser(data, start);\n const format = parser.parseCard8();\n if (format === 0) {\n // Simple list of nGlyphs elements\n for (let iGid = 0; iGid < nGlyphs; iGid++) {\n fdIndex = parser.parseCard8();\n if (fdIndex >= fdArrayCount) {\n throw new Error('CFF table CID Font FDSelect has bad FD index value ' + fdIndex + ' (FD count ' + fdArrayCount + ')');\n }\n fdSelect.push(fdIndex);\n }\n } else if (format === 3) {\n // Ranges\n const nRanges = parser.parseCard16();\n let first = parser.parseCard16();\n if (first !== 0) {\n throw new Error('CFF Table CID Font FDSelect format 3 range has bad initial GID ' + first);\n }\n let next;\n for (let iRange = 0; iRange < nRanges; iRange++) {\n fdIndex = parser.parseCard8();\n next = parser.parseCard16();\n if (fdIndex >= fdArrayCount) {\n throw new Error('CFF table CID Font FDSelect has bad FD index value ' + fdIndex + ' (FD count ' + fdArrayCount + ')');\n }\n if (next > nGlyphs) {\n throw new Error('CFF Table CID Font FDSelect format 3 range has bad GID ' + next);\n }\n for (; first < next; first++) {\n fdSelect.push(fdIndex);\n }\n first = next;\n }\n if (next !== nGlyphs) {\n throw new Error('CFF Table CID Font FDSelect format 3 range has bad final GID ' + next);\n }\n } else {\n throw new Error('CFF Table CID Font FDSelect table has unsupported format ' + format);\n }\n return fdSelect;\n}\n\n// Parse the `CFF` table, which contains the glyph outlines in PostScript format.\nfunction parseCFFTable(data, start, font, opt) {\n font.tables.cff = {};\n const header = parseCFFHeader(data, start);\n const nameIndex = parseCFFIndex(data, header.endOffset, parse.bytesToString);\n const topDictIndex = parseCFFIndex(data, nameIndex.endOffset);\n const stringIndex = parseCFFIndex(data, topDictIndex.endOffset, parse.bytesToString);\n const globalSubrIndex = parseCFFIndex(data, stringIndex.endOffset);\n font.gsubrs = globalSubrIndex.objects;\n font.gsubrsBias = calcCFFSubroutineBias(font.gsubrs);\n\n const topDictArray = gatherCFFTopDicts(data, start, topDictIndex.objects, stringIndex.objects);\n if (topDictArray.length !== 1) {\n throw new Error('CFF table has too many fonts in \\'FontSet\\' - count of fonts NameIndex.length = ' + topDictArray.length);\n }\n\n const topDict = topDictArray[0];\n font.tables.cff.topDict = topDict;\n\n if (topDict._privateDict) {\n font.defaultWidthX = topDict._privateDict.defaultWidthX;\n font.nominalWidthX = topDict._privateDict.nominalWidthX;\n }\n\n if (topDict.ros[0] !== undefined && topDict.ros[1] !== undefined) {\n font.isCIDFont = true;\n }\n\n if (font.isCIDFont) {\n let fdArrayOffset = topDict.fdArray;\n let fdSelectOffset = topDict.fdSelect;\n if (fdArrayOffset === 0 || fdSelectOffset === 0) {\n throw new Error('Font is marked as a CID font, but FDArray and/or FDSelect information is missing');\n }\n fdArrayOffset += start;\n const fdArrayIndex = parseCFFIndex(data, fdArrayOffset);\n const fdArray = gatherCFFTopDicts(data, start, fdArrayIndex.objects, stringIndex.objects);\n topDict._fdArray = fdArray;\n fdSelectOffset += start;\n topDict._fdSelect = parseCFFFDSelect(data, fdSelectOffset, font.numGlyphs, fdArray.length);\n }\n\n const privateDictOffset = start + topDict.private[1];\n const privateDict = parseCFFPrivateDict(data, privateDictOffset, topDict.private[0], stringIndex.objects);\n font.defaultWidthX = privateDict.defaultWidthX;\n font.nominalWidthX = privateDict.nominalWidthX;\n\n if (privateDict.subrs !== 0) {\n const subrOffset = privateDictOffset + privateDict.subrs;\n const subrIndex = parseCFFIndex(data, subrOffset);\n font.subrs = subrIndex.objects;\n font.subrsBias = calcCFFSubroutineBias(font.subrs);\n } else {\n font.subrs = [];\n font.subrsBias = 0;\n }\n\n // Offsets in the top dict are relative to the beginning of the CFF data, so add the CFF start offset.\n let charStringsIndex;\n if (opt.lowMemory) {\n charStringsIndex = parseCFFIndexLowMemory(data, start + topDict.charStrings);\n font.nGlyphs = charStringsIndex.offsets.length;\n } else {\n charStringsIndex = parseCFFIndex(data, start + topDict.charStrings);\n font.nGlyphs = charStringsIndex.objects.length;\n }\n\n const charset = parseCFFCharset(data, start + topDict.charset, font.nGlyphs, stringIndex.objects);\n if (topDict.encoding === 0) {\n // Standard encoding\n font.cffEncoding = new CffEncoding(cffStandardEncoding, charset);\n } else if (topDict.encoding === 1) {\n // Expert encoding\n font.cffEncoding = new CffEncoding(cffExpertEncoding, charset);\n } else {\n font.cffEncoding = parseCFFEncoding(data, start + topDict.encoding, charset);\n }\n\n // Prefer the CMAP encoding to the CFF encoding.\n font.encoding = font.encoding || font.cffEncoding;\n\n font.glyphs = new glyphset.GlyphSet(font);\n if (opt.lowMemory) {\n font._push = function(i) {\n const charString = getCffIndexObject(i, charStringsIndex.offsets, data, start + topDict.charStrings);\n font.glyphs.push(i, glyphset.cffGlyphLoader(font, i, parseCFFCharstring, charString));\n };\n } else {\n for (let i = 0; i < font.nGlyphs; i += 1) {\n const charString = charStringsIndex.objects[i];\n font.glyphs.push(i, glyphset.cffGlyphLoader(font, i, parseCFFCharstring, charString));\n }\n }\n}\n\n// Convert a string to a String ID (SID).\n// The list of strings is modified in place.\nfunction encodeString(s, strings) {\n let sid;\n\n // Is the string in the CFF standard strings?\n let i = cffStandardStrings.indexOf(s);\n if (i >= 0) {\n sid = i;\n }\n\n // Is the string already in the string index?\n i = strings.indexOf(s);\n if (i >= 0) {\n sid = i + cffStandardStrings.length;\n } else {\n sid = cffStandardStrings.length + strings.length;\n strings.push(s);\n }\n\n return sid;\n}\n\nfunction makeHeader() {\n return new table.Record('Header', [\n {name: 'major', type: 'Card8', value: 1},\n {name: 'minor', type: 'Card8', value: 0},\n {name: 'hdrSize', type: 'Card8', value: 4},\n {name: 'major', type: 'Card8', value: 1}\n ]);\n}\n\nfunction makeNameIndex(fontNames) {\n const t = new table.Record('Name INDEX', [\n {name: 'names', type: 'INDEX', value: []}\n ]);\n t.names = [];\n for (let i = 0; i < fontNames.length; i += 1) {\n t.names.push({name: 'name_' + i, type: 'NAME', value: fontNames[i]});\n }\n\n return t;\n}\n\n// Given a dictionary's metadata, create a DICT structure.\nfunction makeDict(meta, attrs, strings) {\n const m = {};\n for (let i = 0; i < meta.length; i += 1) {\n const entry = meta[i];\n let value = attrs[entry.name];\n if (value !== undefined && !equals(value, entry.value)) {\n if (entry.type === 'SID') {\n value = encodeString(value, strings);\n }\n\n m[entry.op] = {name: entry.name, type: entry.type, value: value};\n }\n }\n\n return m;\n}\n\n// The Top DICT houses the global font attributes.\nfunction makeTopDict(attrs, strings) {\n const t = new table.Record('Top DICT', [\n {name: 'dict', type: 'DICT', value: {}}\n ]);\n t.dict = makeDict(TOP_DICT_META, attrs, strings);\n return t;\n}\n\nfunction makeTopDictIndex(topDict) {\n const t = new table.Record('Top DICT INDEX', [\n {name: 'topDicts', type: 'INDEX', value: []}\n ]);\n t.topDicts = [{name: 'topDict_0', type: 'TABLE', value: topDict}];\n return t;\n}\n\nfunction makeStringIndex(strings) {\n const t = new table.Record('String INDEX', [\n {name: 'strings', type: 'INDEX', value: []}\n ]);\n t.strings = [];\n for (let i = 0; i < strings.length; i += 1) {\n t.strings.push({name: 'string_' + i, type: 'STRING', value: strings[i]});\n }\n\n return t;\n}\n\nfunction makeGlobalSubrIndex() {\n // Currently we don't use subroutines.\n return new table.Record('Global Subr INDEX', [\n {name: 'subrs', type: 'INDEX', value: []}\n ]);\n}\n\nfunction makeCharsets(glyphNames, strings) {\n const t = new table.Record('Charsets', [\n {name: 'format', type: 'Card8', value: 0}\n ]);\n for (let i = 0; i < glyphNames.length; i += 1) {\n const glyphName = glyphNames[i];\n const glyphSID = encodeString(glyphName, strings);\n t.fields.push({name: 'glyph_' + i, type: 'SID', value: glyphSID});\n }\n\n return t;\n}\n\nfunction glyphToOps(glyph) {\n const ops = [];\n const path = glyph.path;\n ops.push({name: 'width', type: 'NUMBER', value: glyph.advanceWidth});\n let x = 0;\n let y = 0;\n for (let i = 0; i < path.commands.length; i += 1) {\n let dx;\n let dy;\n let cmd = path.commands[i];\n if (cmd.type === 'Q') {\n // CFF only supports bézier curves, so convert the quad to a bézier.\n const _13 = 1 / 3;\n const _23 = 2 / 3;\n\n // We're going to create a new command so we don't change the original path.\n // Since all coordinates are relative, we round() them ASAP to avoid propagating errors.\n cmd = {\n type: 'C',\n x: cmd.x,\n y: cmd.y,\n x1: Math.round(_13 * x + _23 * cmd.x1),\n y1: Math.round(_13 * y + _23 * cmd.y1),\n x2: Math.round(_13 * cmd.x + _23 * cmd.x1),\n y2: Math.round(_13 * cmd.y + _23 * cmd.y1)\n };\n }\n\n if (cmd.type === 'M') {\n dx = Math.round(cmd.x - x);\n dy = Math.round(cmd.y - y);\n ops.push({name: 'dx', type: 'NUMBER', value: dx});\n ops.push({name: 'dy', type: 'NUMBER', value: dy});\n ops.push({name: 'rmoveto', type: 'OP', value: 21});\n x = Math.round(cmd.x);\n y = Math.round(cmd.y);\n } else if (cmd.type === 'L') {\n dx = Math.round(cmd.x - x);\n dy = Math.round(cmd.y - y);\n ops.push({name: 'dx', type: 'NUMBER', value: dx});\n ops.push({name: 'dy', type: 'NUMBER', value: dy});\n ops.push({name: 'rlineto', type: 'OP', value: 5});\n x = Math.round(cmd.x);\n y = Math.round(cmd.y);\n } else if (cmd.type === 'C') {\n const dx1 = Math.round(cmd.x1 - x);\n const dy1 = Math.round(cmd.y1 - y);\n const dx2 = Math.round(cmd.x2 - cmd.x1);\n const dy2 = Math.round(cmd.y2 - cmd.y1);\n dx = Math.round(cmd.x - cmd.x2);\n dy = Math.round(cmd.y - cmd.y2);\n ops.push({name: 'dx1', type: 'NUMBER', value: dx1});\n ops.push({name: 'dy1', type: 'NUMBER', value: dy1});\n ops.push({name: 'dx2', type: 'NUMBER', value: dx2});\n ops.push({name: 'dy2', type: 'NUMBER', value: dy2});\n ops.push({name: 'dx', type: 'NUMBER', value: dx});\n ops.push({name: 'dy', type: 'NUMBER', value: dy});\n ops.push({name: 'rrcurveto', type: 'OP', value: 8});\n x = Math.round(cmd.x);\n y = Math.round(cmd.y);\n }\n\n // Contours are closed automatically.\n }\n\n ops.push({name: 'endchar', type: 'OP', value: 14});\n return ops;\n}\n\nfunction makeCharStringsIndex(glyphs) {\n const t = new table.Record('CharStrings INDEX', [\n {name: 'charStrings', type: 'INDEX', value: []}\n ]);\n\n for (let i = 0; i < glyphs.length; i += 1) {\n const glyph = glyphs.get(i);\n const ops = glyphToOps(glyph);\n t.charStrings.push({name: glyph.name, type: 'CHARSTRING', value: ops});\n }\n\n return t;\n}\n\nfunction makePrivateDict(attrs, strings) {\n const t = new table.Record('Private DICT', [\n {name: 'dict', type: 'DICT', value: {}}\n ]);\n t.dict = makeDict(PRIVATE_DICT_META, attrs, strings);\n return t;\n}\n\nfunction makeCFFTable(glyphs, options) {\n const t = new table.Table('CFF ', [\n {name: 'header', type: 'RECORD'},\n {name: 'nameIndex', type: 'RECORD'},\n {name: 'topDictIndex', type: 'RECORD'},\n {name: 'stringIndex', type: 'RECORD'},\n {name: 'globalSubrIndex', type: 'RECORD'},\n {name: 'charsets', type: 'RECORD'},\n {name: 'charStringsIndex', type: 'RECORD'},\n {name: 'privateDict', type: 'RECORD'}\n ]);\n\n const fontScale = 1 / options.unitsPerEm;\n // We use non-zero values for the offsets so that the DICT encodes them.\n // This is important because the size of the Top DICT plays a role in offset calculation,\n // and the size shouldn't change after we've written correct offsets.\n const attrs = {\n version: options.version,\n fullName: options.fullName,\n familyName: options.familyName,\n weight: options.weightName,\n fontBBox: options.fontBBox || [0, 0, 0, 0],\n fontMatrix: [fontScale, 0, 0, fontScale, 0, 0],\n charset: 999,\n encoding: 0,\n charStrings: 999,\n private: [0, 999]\n };\n\n const privateAttrs = {};\n\n const glyphNames = [];\n let glyph;\n\n // Skip first glyph (.notdef)\n for (let i = 1; i < glyphs.length; i += 1) {\n glyph = glyphs.get(i);\n glyphNames.push(glyph.name);\n }\n\n const strings = [];\n\n t.header = makeHeader();\n t.nameIndex = makeNameIndex([options.postScriptName]);\n let topDict = makeTopDict(attrs, strings);\n t.topDictIndex = makeTopDictIndex(topDict);\n t.globalSubrIndex = makeGlobalSubrIndex();\n t.charsets = makeCharsets(glyphNames, strings);\n t.charStringsIndex = makeCharStringsIndex(glyphs);\n t.privateDict = makePrivateDict(privateAttrs, strings);\n\n // Needs to come at the end, to encode all custom strings used in the font.\n t.stringIndex = makeStringIndex(strings);\n\n const startOffset = t.header.sizeOf() +\n t.nameIndex.sizeOf() +\n t.topDictIndex.sizeOf() +\n t.stringIndex.sizeOf() +\n t.globalSubrIndex.sizeOf();\n attrs.charset = startOffset;\n\n // We use the CFF standard encoding; proper encoding will be handled in cmap.\n attrs.encoding = 0;\n attrs.charStrings = attrs.charset + t.charsets.sizeOf();\n attrs.private[1] = attrs.charStrings + t.charStringsIndex.sizeOf();\n\n // Recreate the Top DICT INDEX with the correct offsets.\n topDict = makeTopDict(attrs, strings);\n t.topDictIndex = makeTopDictIndex(topDict);\n\n return t;\n}\n\nexport default { parse: parseCFFTable, make: makeCFFTable };\n","// The `head` table contains global information about the font.\n// https://www.microsoft.com/typography/OTSPEC/head.htm\n\nimport check from '../check';\nimport parse from '../parse';\nimport table from '../table';\n\n// Parse the header `head` table\nfunction parseHeadTable(data, start) {\n const head = {};\n const p = new parse.Parser(data, start);\n head.version = p.parseVersion();\n head.fontRevision = Math.round(p.parseFixed() * 1000) / 1000;\n head.checkSumAdjustment = p.parseULong();\n head.magicNumber = p.parseULong();\n check.argument(head.magicNumber === 0x5F0F3CF5, 'Font header has wrong magic number.');\n head.flags = p.parseUShort();\n head.unitsPerEm = p.parseUShort();\n head.created = p.parseLongDateTime();\n head.modified = p.parseLongDateTime();\n head.xMin = p.parseShort();\n head.yMin = p.parseShort();\n head.xMax = p.parseShort();\n head.yMax = p.parseShort();\n head.macStyle = p.parseUShort();\n head.lowestRecPPEM = p.parseUShort();\n head.fontDirectionHint = p.parseShort();\n head.indexToLocFormat = p.parseShort();\n head.glyphDataFormat = p.parseShort();\n return head;\n}\n\nfunction makeHeadTable(options) {\n // Apple Mac timestamp epoch is 01/01/1904 not 01/01/1970\n const timestamp = Math.round(new Date().getTime() / 1000) + 2082844800;\n let createdTimestamp = timestamp;\n\n if (options.createdTimestamp) {\n createdTimestamp = options.createdTimestamp + 2082844800;\n }\n\n return new table.Table('head', [\n {name: 'version', type: 'FIXED', value: 0x00010000},\n {name: 'fontRevision', type: 'FIXED', value: 0x00010000},\n {name: 'checkSumAdjustment', type: 'ULONG', value: 0},\n {name: 'magicNumber', type: 'ULONG', value: 0x5F0F3CF5},\n {name: 'flags', type: 'USHORT', value: 0},\n {name: 'unitsPerEm', type: 'USHORT', value: 1000},\n {name: 'created', type: 'LONGDATETIME', value: createdTimestamp},\n {name: 'modified', type: 'LONGDATETIME', value: timestamp},\n {name: 'xMin', type: 'SHORT', value: 0},\n {name: 'yMin', type: 'SHORT', value: 0},\n {name: 'xMax', type: 'SHORT', value: 0},\n {name: 'yMax', type: 'SHORT', value: 0},\n {name: 'macStyle', type: 'USHORT', value: 0},\n {name: 'lowestRecPPEM', type: 'USHORT', value: 0},\n {name: 'fontDirectionHint', type: 'SHORT', value: 2},\n {name: 'indexToLocFormat', type: 'SHORT', value: 0},\n {name: 'glyphDataFormat', type: 'SHORT', value: 0}\n ], options);\n}\n\nexport default { parse: parseHeadTable, make: makeHeadTable };\n","// The `hhea` table contains information for horizontal layout.\n// https://www.microsoft.com/typography/OTSPEC/hhea.htm\n\nimport parse from '../parse';\nimport table from '../table';\n\n// Parse the horizontal header `hhea` table\nfunction parseHheaTable(data, start) {\n const hhea = {};\n const p = new parse.Parser(data, start);\n hhea.version = p.parseVersion();\n hhea.ascender = p.parseShort();\n hhea.descender = p.parseShort();\n hhea.lineGap = p.parseShort();\n hhea.advanceWidthMax = p.parseUShort();\n hhea.minLeftSideBearing = p.parseShort();\n hhea.minRightSideBearing = p.parseShort();\n hhea.xMaxExtent = p.parseShort();\n hhea.caretSlopeRise = p.parseShort();\n hhea.caretSlopeRun = p.parseShort();\n hhea.caretOffset = p.parseShort();\n p.relativeOffset += 8;\n hhea.metricDataFormat = p.parseShort();\n hhea.numberOfHMetrics = p.parseUShort();\n return hhea;\n}\n\nfunction makeHheaTable(options) {\n return new table.Table('hhea', [\n {name: 'version', type: 'FIXED', value: 0x00010000},\n {name: 'ascender', type: 'FWORD', value: 0},\n {name: 'descender', type: 'FWORD', value: 0},\n {name: 'lineGap', type: 'FWORD', value: 0},\n {name: 'advanceWidthMax', type: 'UFWORD', value: 0},\n {name: 'minLeftSideBearing', type: 'FWORD', value: 0},\n {name: 'minRightSideBearing', type: 'FWORD', value: 0},\n {name: 'xMaxExtent', type: 'FWORD', value: 0},\n {name: 'caretSlopeRise', type: 'SHORT', value: 1},\n {name: 'caretSlopeRun', type: 'SHORT', value: 0},\n {name: 'caretOffset', type: 'SHORT', value: 0},\n {name: 'reserved1', type: 'SHORT', value: 0},\n {name: 'reserved2', type: 'SHORT', value: 0},\n {name: 'reserved3', type: 'SHORT', value: 0},\n {name: 'reserved4', type: 'SHORT', value: 0},\n {name: 'metricDataFormat', type: 'SHORT', value: 0},\n {name: 'numberOfHMetrics', type: 'USHORT', value: 0}\n ], options);\n}\n\nexport default { parse: parseHheaTable, make: makeHheaTable };\n","// The `hmtx` table contains the horizontal metrics for all glyphs.\n// https://www.microsoft.com/typography/OTSPEC/hmtx.htm\n\nimport parse from '../parse';\nimport table from '../table';\n\nfunction parseHmtxTableAll(data, start, numMetrics, numGlyphs, glyphs) {\n let advanceWidth;\n let leftSideBearing;\n const p = new parse.Parser(data, start);\n for (let i = 0; i < numGlyphs; i += 1) {\n // If the font is monospaced, only one entry is needed. This last entry applies to all subsequent glyphs.\n if (i < numMetrics) {\n advanceWidth = p.parseUShort();\n leftSideBearing = p.parseShort();\n }\n\n const glyph = glyphs.get(i);\n glyph.advanceWidth = advanceWidth;\n glyph.leftSideBearing = leftSideBearing;\n }\n}\n\nfunction parseHmtxTableOnLowMemory(font, data, start, numMetrics, numGlyphs) {\n font._hmtxTableData = {};\n\n let advanceWidth;\n let leftSideBearing;\n const p = new parse.Parser(data, start);\n for (let i = 0; i < numGlyphs; i += 1) {\n // If the font is monospaced, only one entry is needed. This last entry applies to all subsequent glyphs.\n if (i < numMetrics) {\n advanceWidth = p.parseUShort();\n leftSideBearing = p.parseShort();\n }\n\n font._hmtxTableData[i] = {\n advanceWidth: advanceWidth,\n leftSideBearing: leftSideBearing\n };\n }\n}\n\n// Parse the `hmtx` table, which contains the horizontal metrics for all glyphs.\n// This function augments the glyph array, adding the advanceWidth and leftSideBearing to each glyph.\nfunction parseHmtxTable(font, data, start, numMetrics, numGlyphs, glyphs, opt) {\n if (opt.lowMemory)\n parseHmtxTableOnLowMemory(font, data, start, numMetrics, numGlyphs);\n else\n parseHmtxTableAll(data, start, numMetrics, numGlyphs, glyphs);\n}\n\nfunction makeHmtxTable(glyphs) {\n const t = new table.Table('hmtx', []);\n for (let i = 0; i < glyphs.length; i += 1) {\n const glyph = glyphs.get(i);\n const advanceWidth = glyph.advanceWidth || 0;\n const leftSideBearing = glyph.leftSideBearing || 0;\n t.fields.push({name: 'advanceWidth_' + i, type: 'USHORT', value: advanceWidth});\n t.fields.push({name: 'leftSideBearing_' + i, type: 'SHORT', value: leftSideBearing});\n }\n\n return t;\n}\n\nexport default { parse: parseHmtxTable, make: makeHmtxTable };\n","// The `ltag` table stores IETF BCP-47 language tags. It allows supporting\n// languages for which TrueType does not assign a numeric code.\n// https://developer.apple.com/fonts/TrueType-Reference-Manual/RM06/Chap6ltag.html\n// http://www.w3.org/International/articles/language-tags/\n// http://www.iana.org/assignments/language-subtag-registry/language-subtag-registry\n\nimport check from '../check';\nimport parse from '../parse';\nimport table from '../table';\n\nfunction makeLtagTable(tags) {\n const result = new table.Table('ltag', [\n {name: 'version', type: 'ULONG', value: 1},\n {name: 'flags', type: 'ULONG', value: 0},\n {name: 'numTags', type: 'ULONG', value: tags.length}\n ]);\n\n let stringPool = '';\n const stringPoolOffset = 12 + tags.length * 4;\n for (let i = 0; i < tags.length; ++i) {\n let pos = stringPool.indexOf(tags[i]);\n if (pos < 0) {\n pos = stringPool.length;\n stringPool += tags[i];\n }\n\n result.fields.push({name: 'offset ' + i, type: 'USHORT', value: stringPoolOffset + pos});\n result.fields.push({name: 'length ' + i, type: 'USHORT', value: tags[i].length});\n }\n\n result.fields.push({name: 'stringPool', type: 'CHARARRAY', value: stringPool});\n return result;\n}\n\nfunction parseLtagTable(data, start) {\n const p = new parse.Parser(data, start);\n const tableVersion = p.parseULong();\n check.argument(tableVersion === 1, 'Unsupported ltag table version.');\n // The 'ltag' specification does not define any flags; skip the field.\n p.skip('uLong', 1);\n const numTags = p.parseULong();\n\n const tags = [];\n for (let i = 0; i < numTags; i++) {\n let tag = '';\n const offset = start + p.parseUShort();\n const length = p.parseUShort();\n for (let j = offset; j < offset + length; ++j) {\n tag += String.fromCharCode(data.getInt8(j));\n }\n\n tags.push(tag);\n }\n\n return tags;\n}\n\nexport default { make: makeLtagTable, parse: parseLtagTable };\n","// The `maxp` table establishes the memory requirements for the font.\n// We need it just to get the number of glyphs in the font.\n// https://www.microsoft.com/typography/OTSPEC/maxp.htm\n\nimport parse from '../parse';\nimport table from '../table';\n\n// Parse the maximum profile `maxp` table.\nfunction parseMaxpTable(data, start) {\n const maxp = {};\n const p = new parse.Parser(data, start);\n maxp.version = p.parseVersion();\n maxp.numGlyphs = p.parseUShort();\n if (maxp.version === 1.0) {\n maxp.maxPoints = p.parseUShort();\n maxp.maxContours = p.parseUShort();\n maxp.maxCompositePoints = p.parseUShort();\n maxp.maxCompositeContours = p.parseUShort();\n maxp.maxZones = p.parseUShort();\n maxp.maxTwilightPoints = p.parseUShort();\n maxp.maxStorage = p.parseUShort();\n maxp.maxFunctionDefs = p.parseUShort();\n maxp.maxInstructionDefs = p.parseUShort();\n maxp.maxStackElements = p.parseUShort();\n maxp.maxSizeOfInstructions = p.parseUShort();\n maxp.maxComponentElements = p.parseUShort();\n maxp.maxComponentDepth = p.parseUShort();\n }\n\n return maxp;\n}\n\nfunction makeMaxpTable(numGlyphs) {\n return new table.Table('maxp', [\n {name: 'version', type: 'FIXED', value: 0x00005000},\n {name: 'numGlyphs', type: 'USHORT', value: numGlyphs}\n ]);\n}\n\nexport default { parse: parseMaxpTable, make: makeMaxpTable };\n","// The `name` naming table.\n// https://www.microsoft.com/typography/OTSPEC/name.htm\n\nimport { decode, encode } from '../types';\nimport parse from '../parse';\nimport table from '../table';\n\n// NameIDs for the name table.\nconst nameTableNames = [\n 'copyright', // 0\n 'fontFamily', // 1\n 'fontSubfamily', // 2\n 'uniqueID', // 3\n 'fullName', // 4\n 'version', // 5\n 'postScriptName', // 6\n 'trademark', // 7\n 'manufacturer', // 8\n 'designer', // 9\n 'description', // 10\n 'manufacturerURL', // 11\n 'designerURL', // 12\n 'license', // 13\n 'licenseURL', // 14\n 'reserved', // 15\n 'preferredFamily', // 16\n 'preferredSubfamily', // 17\n 'compatibleFullName', // 18\n 'sampleText', // 19\n 'postScriptFindFontName', // 20\n 'wwsFamily', // 21\n 'wwsSubfamily' // 22\n];\n\nconst macLanguages = {\n 0: 'en',\n 1: 'fr',\n 2: 'de',\n 3: 'it',\n 4: 'nl',\n 5: 'sv',\n 6: 'es',\n 7: 'da',\n 8: 'pt',\n 9: 'no',\n 10: 'he',\n 11: 'ja',\n 12: 'ar',\n 13: 'fi',\n 14: 'el',\n 15: 'is',\n 16: 'mt',\n 17: 'tr',\n 18: 'hr',\n 19: 'zh-Hant',\n 20: 'ur',\n 21: 'hi',\n 22: 'th',\n 23: 'ko',\n 24: 'lt',\n 25: 'pl',\n 26: 'hu',\n 27: 'es',\n 28: 'lv',\n 29: 'se',\n 30: 'fo',\n 31: 'fa',\n 32: 'ru',\n 33: 'zh',\n 34: 'nl-BE',\n 35: 'ga',\n 36: 'sq',\n 37: 'ro',\n 38: 'cz',\n 39: 'sk',\n 40: 'si',\n 41: 'yi',\n 42: 'sr',\n 43: 'mk',\n 44: 'bg',\n 45: 'uk',\n 46: 'be',\n 47: 'uz',\n 48: 'kk',\n 49: 'az-Cyrl',\n 50: 'az-Arab',\n 51: 'hy',\n 52: 'ka',\n 53: 'mo',\n 54: 'ky',\n 55: 'tg',\n 56: 'tk',\n 57: 'mn-CN',\n 58: 'mn',\n 59: 'ps',\n 60: 'ks',\n 61: 'ku',\n 62: 'sd',\n 63: 'bo',\n 64: 'ne',\n 65: 'sa',\n 66: 'mr',\n 67: 'bn',\n 68: 'as',\n 69: 'gu',\n 70: 'pa',\n 71: 'or',\n 72: 'ml',\n 73: 'kn',\n 74: 'ta',\n 75: 'te',\n 76: 'si',\n 77: 'my',\n 78: 'km',\n 79: 'lo',\n 80: 'vi',\n 81: 'id',\n 82: 'tl',\n 83: 'ms',\n 84: 'ms-Arab',\n 85: 'am',\n 86: 'ti',\n 87: 'om',\n 88: 'so',\n 89: 'sw',\n 90: 'rw',\n 91: 'rn',\n 92: 'ny',\n 93: 'mg',\n 94: 'eo',\n 128: 'cy',\n 129: 'eu',\n 130: 'ca',\n 131: 'la',\n 132: 'qu',\n 133: 'gn',\n 134: 'ay',\n 135: 'tt',\n 136: 'ug',\n 137: 'dz',\n 138: 'jv',\n 139: 'su',\n 140: 'gl',\n 141: 'af',\n 142: 'br',\n 143: 'iu',\n 144: 'gd',\n 145: 'gv',\n 146: 'ga',\n 147: 'to',\n 148: 'el-polyton',\n 149: 'kl',\n 150: 'az',\n 151: 'nn'\n};\n\n// MacOS language ID → MacOS script ID\n//\n// Note that the script ID is not sufficient to determine what encoding\n// to use in TrueType files. For some languages, MacOS used a modification\n// of a mainstream script. For example, an Icelandic name would be stored\n// with smRoman in the TrueType naming table, but the actual encoding\n// is a special Icelandic version of the normal Macintosh Roman encoding.\n// As another example, Inuktitut uses an 8-bit encoding for Canadian Aboriginal\n// Syllables but MacOS had run out of available script codes, so this was\n// done as a (pretty radical) \"modification\" of Ethiopic.\n//\n// http://unicode.org/Public/MAPPINGS/VENDORS/APPLE/Readme.txt\nconst macLanguageToScript = {\n 0: 0, // langEnglish → smRoman\n 1: 0, // langFrench → smRoman\n 2: 0, // langGerman → smRoman\n 3: 0, // langItalian → smRoman\n 4: 0, // langDutch → smRoman\n 5: 0, // langSwedish → smRoman\n 6: 0, // langSpanish → smRoman\n 7: 0, // langDanish → smRoman\n 8: 0, // langPortuguese → smRoman\n 9: 0, // langNorwegian → smRoman\n 10: 5, // langHebrew → smHebrew\n 11: 1, // langJapanese → smJapanese\n 12: 4, // langArabic → smArabic\n 13: 0, // langFinnish → smRoman\n 14: 6, // langGreek → smGreek\n 15: 0, // langIcelandic → smRoman (modified)\n 16: 0, // langMaltese → smRoman\n 17: 0, // langTurkish → smRoman (modified)\n 18: 0, // langCroatian → smRoman (modified)\n 19: 2, // langTradChinese → smTradChinese\n 20: 4, // langUrdu → smArabic\n 21: 9, // langHindi → smDevanagari\n 22: 21, // langThai → smThai\n 23: 3, // langKorean → smKorean\n 24: 29, // langLithuanian → smCentralEuroRoman\n 25: 29, // langPolish → smCentralEuroRoman\n 26: 29, // langHungarian → smCentralEuroRoman\n 27: 29, // langEstonian → smCentralEuroRoman\n 28: 29, // langLatvian → smCentralEuroRoman\n 29: 0, // langSami → smRoman\n 30: 0, // langFaroese → smRoman (modified)\n 31: 4, // langFarsi → smArabic (modified)\n 32: 7, // langRussian → smCyrillic\n 33: 25, // langSimpChinese → smSimpChinese\n 34: 0, // langFlemish → smRoman\n 35: 0, // langIrishGaelic → smRoman (modified)\n 36: 0, // langAlbanian → smRoman\n 37: 0, // langRomanian → smRoman (modified)\n 38: 29, // langCzech → smCentralEuroRoman\n 39: 29, // langSlovak → smCentralEuroRoman\n 40: 0, // langSlovenian → smRoman (modified)\n 41: 5, // langYiddish → smHebrew\n 42: 7, // langSerbian → smCyrillic\n 43: 7, // langMacedonian → smCyrillic\n 44: 7, // langBulgarian → smCyrillic\n 45: 7, // langUkrainian → smCyrillic (modified)\n 46: 7, // langByelorussian → smCyrillic\n 47: 7, // langUzbek → smCyrillic\n 48: 7, // langKazakh → smCyrillic\n 49: 7, // langAzerbaijani → smCyrillic\n 50: 4, // langAzerbaijanAr → smArabic\n 51: 24, // langArmenian → smArmenian\n 52: 23, // langGeorgian → smGeorgian\n 53: 7, // langMoldavian → smCyrillic\n 54: 7, // langKirghiz → smCyrillic\n 55: 7, // langTajiki → smCyrillic\n 56: 7, // langTurkmen → smCyrillic\n 57: 27, // langMongolian → smMongolian\n 58: 7, // langMongolianCyr → smCyrillic\n 59: 4, // langPashto → smArabic\n 60: 4, // langKurdish → smArabic\n 61: 4, // langKashmiri → smArabic\n 62: 4, // langSindhi → smArabic\n 63: 26, // langTibetan → smTibetan\n 64: 9, // langNepali → smDevanagari\n 65: 9, // langSanskrit → smDevanagari\n 66: 9, // langMarathi → smDevanagari\n 67: 13, // langBengali → smBengali\n 68: 13, // langAssamese → smBengali\n 69: 11, // langGujarati → smGujarati\n 70: 10, // langPunjabi → smGurmukhi\n 71: 12, // langOriya → smOriya\n 72: 17, // langMalayalam → smMalayalam\n 73: 16, // langKannada → smKannada\n 74: 14, // langTamil → smTamil\n 75: 15, // langTelugu → smTelugu\n 76: 18, // langSinhalese → smSinhalese\n 77: 19, // langBurmese → smBurmese\n 78: 20, // langKhmer → smKhmer\n 79: 22, // langLao → smLao\n 80: 30, // langVietnamese → smVietnamese\n 81: 0, // langIndonesian → smRoman\n 82: 0, // langTagalog → smRoman\n 83: 0, // langMalayRoman → smRoman\n 84: 4, // langMalayArabic → smArabic\n 85: 28, // langAmharic → smEthiopic\n 86: 28, // langTigrinya → smEthiopic\n 87: 28, // langOromo → smEthiopic\n 88: 0, // langSomali → smRoman\n 89: 0, // langSwahili → smRoman\n 90: 0, // langKinyarwanda → smRoman\n 91: 0, // langRundi → smRoman\n 92: 0, // langNyanja → smRoman\n 93: 0, // langMalagasy → smRoman\n 94: 0, // langEsperanto → smRoman\n 128: 0, // langWelsh → smRoman (modified)\n 129: 0, // langBasque → smRoman\n 130: 0, // langCatalan → smRoman\n 131: 0, // langLatin → smRoman\n 132: 0, // langQuechua → smRoman\n 133: 0, // langGuarani → smRoman\n 134: 0, // langAymara → smRoman\n 135: 7, // langTatar → smCyrillic\n 136: 4, // langUighur → smArabic\n 137: 26, // langDzongkha → smTibetan\n 138: 0, // langJavaneseRom → smRoman\n 139: 0, // langSundaneseRom → smRoman\n 140: 0, // langGalician → smRoman\n 141: 0, // langAfrikaans → smRoman\n 142: 0, // langBreton → smRoman (modified)\n 143: 28, // langInuktitut → smEthiopic (modified)\n 144: 0, // langScottishGaelic → smRoman (modified)\n 145: 0, // langManxGaelic → smRoman (modified)\n 146: 0, // langIrishGaelicScript → smRoman (modified)\n 147: 0, // langTongan → smRoman\n 148: 6, // langGreekAncient → smRoman\n 149: 0, // langGreenlandic → smRoman\n 150: 0, // langAzerbaijanRoman → smRoman\n 151: 0 // langNynorsk → smRoman\n};\n\n// While Microsoft indicates a region/country for all its language\n// IDs, we omit the region code if it's equal to the \"most likely\n// region subtag\" according to Unicode CLDR. For scripts, we omit\n// the subtag if it is equal to the Suppress-Script entry in the\n// IANA language subtag registry for IETF BCP 47.\n//\n// For example, Microsoft states that its language code 0x041A is\n// Croatian in Croatia. We transform this to the BCP 47 language code 'hr'\n// and not 'hr-HR' because Croatia is the default country for Croatian,\n// according to Unicode CLDR. As another example, Microsoft states\n// that 0x101A is Croatian (Latin) in Bosnia-Herzegovina. We transform\n// this to 'hr-BA' and not 'hr-Latn-BA' because Latin is the default script\n// for the Croatian language, according to IANA.\n//\n// http://www.unicode.org/cldr/charts/latest/supplemental/likely_subtags.html\n// http://www.iana.org/assignments/language-subtag-registry/language-subtag-registry\nconst windowsLanguages = {\n 0x0436: 'af',\n 0x041C: 'sq',\n 0x0484: 'gsw',\n 0x045E: 'am',\n 0x1401: 'ar-DZ',\n 0x3C01: 'ar-BH',\n 0x0C01: 'ar',\n 0x0801: 'ar-IQ',\n 0x2C01: 'ar-JO',\n 0x3401: 'ar-KW',\n 0x3001: 'ar-LB',\n 0x1001: 'ar-LY',\n 0x1801: 'ary',\n 0x2001: 'ar-OM',\n 0x4001: 'ar-QA',\n 0x0401: 'ar-SA',\n 0x2801: 'ar-SY',\n 0x1C01: 'aeb',\n 0x3801: 'ar-AE',\n 0x2401: 'ar-YE',\n 0x042B: 'hy',\n 0x044D: 'as',\n 0x082C: 'az-Cyrl',\n 0x042C: 'az',\n 0x046D: 'ba',\n 0x042D: 'eu',\n 0x0423: 'be',\n 0x0845: 'bn',\n 0x0445: 'bn-IN',\n 0x201A: 'bs-Cyrl',\n 0x141A: 'bs',\n 0x047E: 'br',\n 0x0402: 'bg',\n 0x0403: 'ca',\n 0x0C04: 'zh-HK',\n 0x1404: 'zh-MO',\n 0x0804: 'zh',\n 0x1004: 'zh-SG',\n 0x0404: 'zh-TW',\n 0x0483: 'co',\n 0x041A: 'hr',\n 0x101A: 'hr-BA',\n 0x0405: 'cs',\n 0x0406: 'da',\n 0x048C: 'prs',\n 0x0465: 'dv',\n 0x0813: 'nl-BE',\n 0x0413: 'nl',\n 0x0C09: 'en-AU',\n 0x2809: 'en-BZ',\n 0x1009: 'en-CA',\n 0x2409: 'en-029',\n 0x4009: 'en-IN',\n 0x1809: 'en-IE',\n 0x2009: 'en-JM',\n 0x4409: 'en-MY',\n 0x1409: 'en-NZ',\n 0x3409: 'en-PH',\n 0x4809: 'en-SG',\n 0x1C09: 'en-ZA',\n 0x2C09: 'en-TT',\n 0x0809: 'en-GB',\n 0x0409: 'en',\n 0x3009: 'en-ZW',\n 0x0425: 'et',\n 0x0438: 'fo',\n 0x0464: 'fil',\n 0x040B: 'fi',\n 0x080C: 'fr-BE',\n 0x0C0C: 'fr-CA',\n 0x040C: 'fr',\n 0x140C: 'fr-LU',\n 0x180C: 'fr-MC',\n 0x100C: 'fr-CH',\n 0x0462: 'fy',\n 0x0456: 'gl',\n 0x0437: 'ka',\n 0x0C07: 'de-AT',\n 0x0407: 'de',\n 0x1407: 'de-LI',\n 0x1007: 'de-LU',\n 0x0807: 'de-CH',\n 0x0408: 'el',\n 0x046F: 'kl',\n 0x0447: 'gu',\n 0x0468: 'ha',\n 0x040D: 'he',\n 0x0439: 'hi',\n 0x040E: 'hu',\n 0x040F: 'is',\n 0x0470: 'ig',\n 0x0421: 'id',\n 0x045D: 'iu',\n 0x085D: 'iu-Latn',\n 0x083C: 'ga',\n 0x0434: 'xh',\n 0x0435: 'zu',\n 0x0410: 'it',\n 0x0810: 'it-CH',\n 0x0411: 'ja',\n 0x044B: 'kn',\n 0x043F: 'kk',\n 0x0453: 'km',\n 0x0486: 'quc',\n 0x0487: 'rw',\n 0x0441: 'sw',\n 0x0457: 'kok',\n 0x0412: 'ko',\n 0x0440: 'ky',\n 0x0454: 'lo',\n 0x0426: 'lv',\n 0x0427: 'lt',\n 0x082E: 'dsb',\n 0x046E: 'lb',\n 0x042F: 'mk',\n 0x083E: 'ms-BN',\n 0x043E: 'ms',\n 0x044C: 'ml',\n 0x043A: 'mt',\n 0x0481: 'mi',\n 0x047A: 'arn',\n 0x044E: 'mr',\n 0x047C: 'moh',\n 0x0450: 'mn',\n 0x0850: 'mn-CN',\n 0x0461: 'ne',\n 0x0414: 'nb',\n 0x0814: 'nn',\n 0x0482: 'oc',\n 0x0448: 'or',\n 0x0463: 'ps',\n 0x0415: 'pl',\n 0x0416: 'pt',\n 0x0816: 'pt-PT',\n 0x0446: 'pa',\n 0x046B: 'qu-BO',\n 0x086B: 'qu-EC',\n 0x0C6B: 'qu',\n 0x0418: 'ro',\n 0x0417: 'rm',\n 0x0419: 'ru',\n 0x243B: 'smn',\n 0x103B: 'smj-NO',\n 0x143B: 'smj',\n 0x0C3B: 'se-FI',\n 0x043B: 'se',\n 0x083B: 'se-SE',\n 0x203B: 'sms',\n 0x183B: 'sma-NO',\n 0x1C3B: 'sms',\n 0x044F: 'sa',\n 0x1C1A: 'sr-Cyrl-BA',\n 0x0C1A: 'sr',\n 0x181A: 'sr-Latn-BA',\n 0x081A: 'sr-Latn',\n 0x046C: 'nso',\n 0x0432: 'tn',\n 0x045B: 'si',\n 0x041B: 'sk',\n 0x0424: 'sl',\n 0x2C0A: 'es-AR',\n 0x400A: 'es-BO',\n 0x340A: 'es-CL',\n 0x240A: 'es-CO',\n 0x140A: 'es-CR',\n 0x1C0A: 'es-DO',\n 0x300A: 'es-EC',\n 0x440A: 'es-SV',\n 0x100A: 'es-GT',\n 0x480A: 'es-HN',\n 0x080A: 'es-MX',\n 0x4C0A: 'es-NI',\n 0x180A: 'es-PA',\n 0x3C0A: 'es-PY',\n 0x280A: 'es-PE',\n 0x500A: 'es-PR',\n\n // Microsoft has defined two different language codes for\n // “Spanish with modern sorting” and “Spanish with traditional\n // sorting”. This makes sense for collation APIs, and it would be\n // possible to express this in BCP 47 language tags via Unicode\n // extensions (eg., es-u-co-trad is Spanish with traditional\n // sorting). However, for storing names in fonts, the distinction\n // does not make sense, so we give “es” in both cases.\n 0x0C0A: 'es',\n 0x040A: 'es',\n\n 0x540A: 'es-US',\n 0x380A: 'es-UY',\n 0x200A: 'es-VE',\n 0x081D: 'sv-FI',\n 0x041D: 'sv',\n 0x045A: 'syr',\n 0x0428: 'tg',\n 0x085F: 'tzm',\n 0x0449: 'ta',\n 0x0444: 'tt',\n 0x044A: 'te',\n 0x041E: 'th',\n 0x0451: 'bo',\n 0x041F: 'tr',\n 0x0442: 'tk',\n 0x0480: 'ug',\n 0x0422: 'uk',\n 0x042E: 'hsb',\n 0x0420: 'ur',\n 0x0843: 'uz-Cyrl',\n 0x0443: 'uz',\n 0x042A: 'vi',\n 0x0452: 'cy',\n 0x0488: 'wo',\n 0x0485: 'sah',\n 0x0478: 'ii',\n 0x046A: 'yo'\n};\n\n// Returns a IETF BCP 47 language code, for example 'zh-Hant'\n// for 'Chinese in the traditional script'.\nfunction getLanguageCode(platformID, languageID, ltag) {\n switch (platformID) {\n case 0: // Unicode\n if (languageID === 0xFFFF) {\n return 'und';\n } else if (ltag) {\n return ltag[languageID];\n }\n\n break;\n\n case 1: // Macintosh\n return macLanguages[languageID];\n\n case 3: // Windows\n return windowsLanguages[languageID];\n }\n\n return undefined;\n}\n\nconst utf16 = 'utf-16';\n\n// MacOS script ID → encoding. This table stores the default case,\n// which can be overridden by macLanguageEncodings.\nconst macScriptEncodings = {\n 0: 'macintosh', // smRoman\n 1: 'x-mac-japanese', // smJapanese\n 2: 'x-mac-chinesetrad', // smTradChinese\n 3: 'x-mac-korean', // smKorean\n 6: 'x-mac-greek', // smGreek\n 7: 'x-mac-cyrillic', // smCyrillic\n 9: 'x-mac-devanagai', // smDevanagari\n 10: 'x-mac-gurmukhi', // smGurmukhi\n 11: 'x-mac-gujarati', // smGujarati\n 12: 'x-mac-oriya', // smOriya\n 13: 'x-mac-bengali', // smBengali\n 14: 'x-mac-tamil', // smTamil\n 15: 'x-mac-telugu', // smTelugu\n 16: 'x-mac-kannada', // smKannada\n 17: 'x-mac-malayalam', // smMalayalam\n 18: 'x-mac-sinhalese', // smSinhalese\n 19: 'x-mac-burmese', // smBurmese\n 20: 'x-mac-khmer', // smKhmer\n 21: 'x-mac-thai', // smThai\n 22: 'x-mac-lao', // smLao\n 23: 'x-mac-georgian', // smGeorgian\n 24: 'x-mac-armenian', // smArmenian\n 25: 'x-mac-chinesesimp', // smSimpChinese\n 26: 'x-mac-tibetan', // smTibetan\n 27: 'x-mac-mongolian', // smMongolian\n 28: 'x-mac-ethiopic', // smEthiopic\n 29: 'x-mac-ce', // smCentralEuroRoman\n 30: 'x-mac-vietnamese', // smVietnamese\n 31: 'x-mac-extarabic' // smExtArabic\n};\n\n// MacOS language ID → encoding. This table stores the exceptional\n// cases, which override macScriptEncodings. For writing MacOS naming\n// tables, we need to emit a MacOS script ID. Therefore, we cannot\n// merge macScriptEncodings into macLanguageEncodings.\n//\n// http://unicode.org/Public/MAPPINGS/VENDORS/APPLE/Readme.txt\nconst macLanguageEncodings = {\n 15: 'x-mac-icelandic', // langIcelandic\n 17: 'x-mac-turkish', // langTurkish\n 18: 'x-mac-croatian', // langCroatian\n 24: 'x-mac-ce', // langLithuanian\n 25: 'x-mac-ce', // langPolish\n 26: 'x-mac-ce', // langHungarian\n 27: 'x-mac-ce', // langEstonian\n 28: 'x-mac-ce', // langLatvian\n 30: 'x-mac-icelandic', // langFaroese\n 37: 'x-mac-romanian', // langRomanian\n 38: 'x-mac-ce', // langCzech\n 39: 'x-mac-ce', // langSlovak\n 40: 'x-mac-ce', // langSlovenian\n 143: 'x-mac-inuit', // langInuktitut\n 146: 'x-mac-gaelic' // langIrishGaelicScript\n};\n\nfunction getEncoding(platformID, encodingID, languageID) {\n switch (platformID) {\n case 0: // Unicode\n return utf16;\n\n case 1: // Apple Macintosh\n return macLanguageEncodings[languageID] || macScriptEncodings[encodingID];\n\n case 3: // Microsoft Windows\n if (encodingID === 1 || encodingID === 10) {\n return utf16;\n }\n\n break;\n }\n\n return undefined;\n}\n\n// Parse the naming `name` table.\n// FIXME: Format 1 additional fields are not supported yet.\n// ltag is the content of the `ltag' table, such as ['en', 'zh-Hans', 'de-CH-1904'].\nfunction parseNameTable(data, start, ltag) {\n const name = {};\n const p = new parse.Parser(data, start);\n const format = p.parseUShort();\n const count = p.parseUShort();\n const stringOffset = p.offset + p.parseUShort();\n for (let i = 0; i < count; i++) {\n const platformID = p.parseUShort();\n const encodingID = p.parseUShort();\n const languageID = p.parseUShort();\n const nameID = p.parseUShort();\n const property = nameTableNames[nameID] || nameID;\n const byteLength = p.parseUShort();\n const offset = p.parseUShort();\n const language = getLanguageCode(platformID, languageID, ltag);\n const encoding = getEncoding(platformID, encodingID, languageID);\n if (encoding !== undefined && language !== undefined) {\n let text;\n if (encoding === utf16) {\n text = decode.UTF16(data, stringOffset + offset, byteLength);\n } else {\n text = decode.MACSTRING(data, stringOffset + offset, byteLength, encoding);\n }\n\n if (text) {\n let translations = name[property];\n if (translations === undefined) {\n translations = name[property] = {};\n }\n\n translations[language] = text;\n }\n }\n }\n\n let langTagCount = 0;\n if (format === 1) {\n // FIXME: Also handle Microsoft's 'name' table 1.\n langTagCount = p.parseUShort();\n }\n\n return name;\n}\n\n// {23: 'foo'} → {'foo': 23}\n// ['bar', 'baz'] → {'bar': 0, 'baz': 1}\nfunction reverseDict(dict) {\n const result = {};\n for (let key in dict) {\n result[dict[key]] = parseInt(key);\n }\n\n return result;\n}\n\nfunction makeNameRecord(platformID, encodingID, languageID, nameID, length, offset) {\n return new table.Record('NameRecord', [\n {name: 'platformID', type: 'USHORT', value: platformID},\n {name: 'encodingID', type: 'USHORT', value: encodingID},\n {name: 'languageID', type: 'USHORT', value: languageID},\n {name: 'nameID', type: 'USHORT', value: nameID},\n {name: 'length', type: 'USHORT', value: length},\n {name: 'offset', type: 'USHORT', value: offset}\n ]);\n}\n\n// Finds the position of needle in haystack, or -1 if not there.\n// Like String.indexOf(), but for arrays.\nfunction findSubArray(needle, haystack) {\n const needleLength = needle.length;\n const limit = haystack.length - needleLength + 1;\n\n loop:\n for (let pos = 0; pos < limit; pos++) {\n for (; pos < limit; pos++) {\n for (let k = 0; k < needleLength; k++) {\n if (haystack[pos + k] !== needle[k]) {\n continue loop;\n }\n }\n\n return pos;\n }\n }\n\n return -1;\n}\n\nfunction addStringToPool(s, pool) {\n let offset = findSubArray(s, pool);\n if (offset < 0) {\n offset = pool.length;\n let i = 0;\n const len = s.length;\n for (; i < len; ++i) {\n pool.push(s[i]);\n }\n\n }\n\n return offset;\n}\n\nfunction makeNameTable(names, ltag) {\n let nameID;\n const nameIDs = [];\n\n const namesWithNumericKeys = {};\n const nameTableIds = reverseDict(nameTableNames);\n for (let key in names) {\n let id = nameTableIds[key];\n if (id === undefined) {\n id = key;\n }\n\n nameID = parseInt(id);\n\n if (isNaN(nameID)) {\n throw new Error('Name table entry \"' + key + '\" does not exist, see nameTableNames for complete list.');\n }\n\n namesWithNumericKeys[nameID] = names[key];\n nameIDs.push(nameID);\n }\n\n const macLanguageIds = reverseDict(macLanguages);\n const windowsLanguageIds = reverseDict(windowsLanguages);\n\n const nameRecords = [];\n const stringPool = [];\n\n for (let i = 0; i < nameIDs.length; i++) {\n nameID = nameIDs[i];\n const translations = namesWithNumericKeys[nameID];\n for (let lang in translations) {\n const text = translations[lang];\n\n // For MacOS, we try to emit the name in the form that was introduced\n // in the initial version of the TrueType spec (in the late 1980s).\n // However, this can fail for various reasons: the requested BCP 47\n // language code might not have an old-style Mac equivalent;\n // we might not have a codec for the needed character encoding;\n // or the name might contain characters that cannot be expressed\n // in the old-style Macintosh encoding. In case of failure, we emit\n // the name in a more modern fashion (Unicode encoding with BCP 47\n // language tags) that is recognized by MacOS 10.5, released in 2009.\n // If fonts were only read by operating systems, we could simply\n // emit all names in the modern form; this would be much easier.\n // However, there are many applications and libraries that read\n // 'name' tables directly, and these will usually only recognize\n // the ancient form (silently skipping the unrecognized names).\n let macPlatform = 1; // Macintosh\n let macLanguage = macLanguageIds[lang];\n let macScript = macLanguageToScript[macLanguage];\n const macEncoding = getEncoding(macPlatform, macScript, macLanguage);\n let macName = encode.MACSTRING(text, macEncoding);\n if (macName === undefined) {\n macPlatform = 0; // Unicode\n macLanguage = ltag.indexOf(lang);\n if (macLanguage < 0) {\n macLanguage = ltag.length;\n ltag.push(lang);\n }\n\n macScript = 4; // Unicode 2.0 and later\n macName = encode.UTF16(text);\n }\n\n const macNameOffset = addStringToPool(macName, stringPool);\n nameRecords.push(makeNameRecord(macPlatform, macScript, macLanguage,\n nameID, macName.length, macNameOffset));\n\n const winLanguage = windowsLanguageIds[lang];\n if (winLanguage !== undefined) {\n const winName = encode.UTF16(text);\n const winNameOffset = addStringToPool(winName, stringPool);\n nameRecords.push(makeNameRecord(3, 1, winLanguage,\n nameID, winName.length, winNameOffset));\n }\n }\n }\n\n nameRecords.sort(function(a, b) {\n return ((a.platformID - b.platformID) ||\n (a.encodingID - b.encodingID) ||\n (a.languageID - b.languageID) ||\n (a.nameID - b.nameID));\n });\n\n const t = new table.Table('name', [\n {name: 'format', type: 'USHORT', value: 0},\n {name: 'count', type: 'USHORT', value: nameRecords.length},\n {name: 'stringOffset', type: 'USHORT', value: 6 + nameRecords.length * 12}\n ]);\n\n for (let r = 0; r < nameRecords.length; r++) {\n t.fields.push({name: 'record_' + r, type: 'RECORD', value: nameRecords[r]});\n }\n\n t.fields.push({name: 'strings', type: 'LITERAL', value: stringPool});\n return t;\n}\n\nexport default { parse: parseNameTable, make: makeNameTable };\n","// The `OS/2` table contains metrics required in OpenType fonts.\n// https://www.microsoft.com/typography/OTSPEC/os2.htm\n\nimport parse from '../parse';\nimport table from '../table';\n\nconst unicodeRanges = [\n {begin: 0x0000, end: 0x007F}, // Basic Latin\n {begin: 0x0080, end: 0x00FF}, // Latin-1 Supplement\n {begin: 0x0100, end: 0x017F}, // Latin Extended-A\n {begin: 0x0180, end: 0x024F}, // Latin Extended-B\n {begin: 0x0250, end: 0x02AF}, // IPA Extensions\n {begin: 0x02B0, end: 0x02FF}, // Spacing Modifier Letters\n {begin: 0x0300, end: 0x036F}, // Combining Diacritical Marks\n {begin: 0x0370, end: 0x03FF}, // Greek and Coptic\n {begin: 0x2C80, end: 0x2CFF}, // Coptic\n {begin: 0x0400, end: 0x04FF}, // Cyrillic\n {begin: 0x0530, end: 0x058F}, // Armenian\n {begin: 0x0590, end: 0x05FF}, // Hebrew\n {begin: 0xA500, end: 0xA63F}, // Vai\n {begin: 0x0600, end: 0x06FF}, // Arabic\n {begin: 0x07C0, end: 0x07FF}, // NKo\n {begin: 0x0900, end: 0x097F}, // Devanagari\n {begin: 0x0980, end: 0x09FF}, // Bengali\n {begin: 0x0A00, end: 0x0A7F}, // Gurmukhi\n {begin: 0x0A80, end: 0x0AFF}, // Gujarati\n {begin: 0x0B00, end: 0x0B7F}, // Oriya\n {begin: 0x0B80, end: 0x0BFF}, // Tamil\n {begin: 0x0C00, end: 0x0C7F}, // Telugu\n {begin: 0x0C80, end: 0x0CFF}, // Kannada\n {begin: 0x0D00, end: 0x0D7F}, // Malayalam\n {begin: 0x0E00, end: 0x0E7F}, // Thai\n {begin: 0x0E80, end: 0x0EFF}, // Lao\n {begin: 0x10A0, end: 0x10FF}, // Georgian\n {begin: 0x1B00, end: 0x1B7F}, // Balinese\n {begin: 0x1100, end: 0x11FF}, // Hangul Jamo\n {begin: 0x1E00, end: 0x1EFF}, // Latin Extended Additional\n {begin: 0x1F00, end: 0x1FFF}, // Greek Extended\n {begin: 0x2000, end: 0x206F}, // General Punctuation\n {begin: 0x2070, end: 0x209F}, // Superscripts And Subscripts\n {begin: 0x20A0, end: 0x20CF}, // Currency Symbol\n {begin: 0x20D0, end: 0x20FF}, // Combining Diacritical Marks For Symbols\n {begin: 0x2100, end: 0x214F}, // Letterlike Symbols\n {begin: 0x2150, end: 0x218F}, // Number Forms\n {begin: 0x2190, end: 0x21FF}, // Arrows\n {begin: 0x2200, end: 0x22FF}, // Mathematical Operators\n {begin: 0x2300, end: 0x23FF}, // Miscellaneous Technical\n {begin: 0x2400, end: 0x243F}, // Control Pictures\n {begin: 0x2440, end: 0x245F}, // Optical Character Recognition\n {begin: 0x2460, end: 0x24FF}, // Enclosed Alphanumerics\n {begin: 0x2500, end: 0x257F}, // Box Drawing\n {begin: 0x2580, end: 0x259F}, // Block Elements\n {begin: 0x25A0, end: 0x25FF}, // Geometric Shapes\n {begin: 0x2600, end: 0x26FF}, // Miscellaneous Symbols\n {begin: 0x2700, end: 0x27BF}, // Dingbats\n {begin: 0x3000, end: 0x303F}, // CJK Symbols And Punctuation\n {begin: 0x3040, end: 0x309F}, // Hiragana\n {begin: 0x30A0, end: 0x30FF}, // Katakana\n {begin: 0x3100, end: 0x312F}, // Bopomofo\n {begin: 0x3130, end: 0x318F}, // Hangul Compatibility Jamo\n {begin: 0xA840, end: 0xA87F}, // Phags-pa\n {begin: 0x3200, end: 0x32FF}, // Enclosed CJK Letters And Months\n {begin: 0x3300, end: 0x33FF}, // CJK Compatibility\n {begin: 0xAC00, end: 0xD7AF}, // Hangul Syllables\n {begin: 0xD800, end: 0xDFFF}, // Non-Plane 0 *\n {begin: 0x10900, end: 0x1091F}, // Phoenicia\n {begin: 0x4E00, end: 0x9FFF}, // CJK Unified Ideographs\n {begin: 0xE000, end: 0xF8FF}, // Private Use Area (plane 0)\n {begin: 0x31C0, end: 0x31EF}, // CJK Strokes\n {begin: 0xFB00, end: 0xFB4F}, // Alphabetic Presentation Forms\n {begin: 0xFB50, end: 0xFDFF}, // Arabic Presentation Forms-A\n {begin: 0xFE20, end: 0xFE2F}, // Combining Half Marks\n {begin: 0xFE10, end: 0xFE1F}, // Vertical Forms\n {begin: 0xFE50, end: 0xFE6F}, // Small Form Variants\n {begin: 0xFE70, end: 0xFEFF}, // Arabic Presentation Forms-B\n {begin: 0xFF00, end: 0xFFEF}, // Halfwidth And Fullwidth Forms\n {begin: 0xFFF0, end: 0xFFFF}, // Specials\n {begin: 0x0F00, end: 0x0FFF}, // Tibetan\n {begin: 0x0700, end: 0x074F}, // Syriac\n {begin: 0x0780, end: 0x07BF}, // Thaana\n {begin: 0x0D80, end: 0x0DFF}, // Sinhala\n {begin: 0x1000, end: 0x109F}, // Myanmar\n {begin: 0x1200, end: 0x137F}, // Ethiopic\n {begin: 0x13A0, end: 0x13FF}, // Cherokee\n {begin: 0x1400, end: 0x167F}, // Unified Canadian Aboriginal Syllabics\n {begin: 0x1680, end: 0x169F}, // Ogham\n {begin: 0x16A0, end: 0x16FF}, // Runic\n {begin: 0x1780, end: 0x17FF}, // Khmer\n {begin: 0x1800, end: 0x18AF}, // Mongolian\n {begin: 0x2800, end: 0x28FF}, // Braille Patterns\n {begin: 0xA000, end: 0xA48F}, // Yi Syllables\n {begin: 0x1700, end: 0x171F}, // Tagalog\n {begin: 0x10300, end: 0x1032F}, // Old Italic\n {begin: 0x10330, end: 0x1034F}, // Gothic\n {begin: 0x10400, end: 0x1044F}, // Deseret\n {begin: 0x1D000, end: 0x1D0FF}, // Byzantine Musical Symbols\n {begin: 0x1D400, end: 0x1D7FF}, // Mathematical Alphanumeric Symbols\n {begin: 0xFF000, end: 0xFFFFD}, // Private Use (plane 15)\n {begin: 0xFE00, end: 0xFE0F}, // Variation Selectors\n {begin: 0xE0000, end: 0xE007F}, // Tags\n {begin: 0x1900, end: 0x194F}, // Limbu\n {begin: 0x1950, end: 0x197F}, // Tai Le\n {begin: 0x1980, end: 0x19DF}, // New Tai Lue\n {begin: 0x1A00, end: 0x1A1F}, // Buginese\n {begin: 0x2C00, end: 0x2C5F}, // Glagolitic\n {begin: 0x2D30, end: 0x2D7F}, // Tifinagh\n {begin: 0x4DC0, end: 0x4DFF}, // Yijing Hexagram Symbols\n {begin: 0xA800, end: 0xA82F}, // Syloti Nagri\n {begin: 0x10000, end: 0x1007F}, // Linear B Syllabary\n {begin: 0x10140, end: 0x1018F}, // Ancient Greek Numbers\n {begin: 0x10380, end: 0x1039F}, // Ugaritic\n {begin: 0x103A0, end: 0x103DF}, // Old Persian\n {begin: 0x10450, end: 0x1047F}, // Shavian\n {begin: 0x10480, end: 0x104AF}, // Osmanya\n {begin: 0x10800, end: 0x1083F}, // Cypriot Syllabary\n {begin: 0x10A00, end: 0x10A5F}, // Kharoshthi\n {begin: 0x1D300, end: 0x1D35F}, // Tai Xuan Jing Symbols\n {begin: 0x12000, end: 0x123FF}, // Cuneiform\n {begin: 0x1D360, end: 0x1D37F}, // Counting Rod Numerals\n {begin: 0x1B80, end: 0x1BBF}, // Sundanese\n {begin: 0x1C00, end: 0x1C4F}, // Lepcha\n {begin: 0x1C50, end: 0x1C7F}, // Ol Chiki\n {begin: 0xA880, end: 0xA8DF}, // Saurashtra\n {begin: 0xA900, end: 0xA92F}, // Kayah Li\n {begin: 0xA930, end: 0xA95F}, // Rejang\n {begin: 0xAA00, end: 0xAA5F}, // Cham\n {begin: 0x10190, end: 0x101CF}, // Ancient Symbols\n {begin: 0x101D0, end: 0x101FF}, // Phaistos Disc\n {begin: 0x102A0, end: 0x102DF}, // Carian\n {begin: 0x1F030, end: 0x1F09F} // Domino Tiles\n];\n\nfunction getUnicodeRange(unicode) {\n for (let i = 0; i < unicodeRanges.length; i += 1) {\n const range = unicodeRanges[i];\n if (unicode >= range.begin && unicode < range.end) {\n return i;\n }\n }\n\n return -1;\n}\n\n// Parse the OS/2 and Windows metrics `OS/2` table\nfunction parseOS2Table(data, start) {\n const os2 = {};\n const p = new parse.Parser(data, start);\n os2.version = p.parseUShort();\n os2.xAvgCharWidth = p.parseShort();\n os2.usWeightClass = p.parseUShort();\n os2.usWidthClass = p.parseUShort();\n os2.fsType = p.parseUShort();\n os2.ySubscriptXSize = p.parseShort();\n os2.ySubscriptYSize = p.parseShort();\n os2.ySubscriptXOffset = p.parseShort();\n os2.ySubscriptYOffset = p.parseShort();\n os2.ySuperscriptXSize = p.parseShort();\n os2.ySuperscriptYSize = p.parseShort();\n os2.ySuperscriptXOffset = p.parseShort();\n os2.ySuperscriptYOffset = p.parseShort();\n os2.yStrikeoutSize = p.parseShort();\n os2.yStrikeoutPosition = p.parseShort();\n os2.sFamilyClass = p.parseShort();\n os2.panose = [];\n for (let i = 0; i < 10; i++) {\n os2.panose[i] = p.parseByte();\n }\n\n os2.ulUnicodeRange1 = p.parseULong();\n os2.ulUnicodeRange2 = p.parseULong();\n os2.ulUnicodeRange3 = p.parseULong();\n os2.ulUnicodeRange4 = p.parseULong();\n os2.achVendID = String.fromCharCode(p.parseByte(), p.parseByte(), p.parseByte(), p.parseByte());\n os2.fsSelection = p.parseUShort();\n os2.usFirstCharIndex = p.parseUShort();\n os2.usLastCharIndex = p.parseUShort();\n os2.sTypoAscender = p.parseShort();\n os2.sTypoDescender = p.parseShort();\n os2.sTypoLineGap = p.parseShort();\n os2.usWinAscent = p.parseUShort();\n os2.usWinDescent = p.parseUShort();\n if (os2.version >= 1) {\n os2.ulCodePageRange1 = p.parseULong();\n os2.ulCodePageRange2 = p.parseULong();\n }\n\n if (os2.version >= 2) {\n os2.sxHeight = p.parseShort();\n os2.sCapHeight = p.parseShort();\n os2.usDefaultChar = p.parseUShort();\n os2.usBreakChar = p.parseUShort();\n os2.usMaxContent = p.parseUShort();\n }\n\n return os2;\n}\n\nfunction makeOS2Table(options) {\n return new table.Table('OS/2', [\n {name: 'version', type: 'USHORT', value: 0x0003},\n {name: 'xAvgCharWidth', type: 'SHORT', value: 0},\n {name: 'usWeightClass', type: 'USHORT', value: 0},\n {name: 'usWidthClass', type: 'USHORT', value: 0},\n {name: 'fsType', type: 'USHORT', value: 0},\n {name: 'ySubscriptXSize', type: 'SHORT', value: 650},\n {name: 'ySubscriptYSize', type: 'SHORT', value: 699},\n {name: 'ySubscriptXOffset', type: 'SHORT', value: 0},\n {name: 'ySubscriptYOffset', type: 'SHORT', value: 140},\n {name: 'ySuperscriptXSize', type: 'SHORT', value: 650},\n {name: 'ySuperscriptYSize', type: 'SHORT', value: 699},\n {name: 'ySuperscriptXOffset', type: 'SHORT', value: 0},\n {name: 'ySuperscriptYOffset', type: 'SHORT', value: 479},\n {name: 'yStrikeoutSize', type: 'SHORT', value: 49},\n {name: 'yStrikeoutPosition', type: 'SHORT', value: 258},\n {name: 'sFamilyClass', type: 'SHORT', value: 0},\n {name: 'bFamilyType', type: 'BYTE', value: 0},\n {name: 'bSerifStyle', type: 'BYTE', value: 0},\n {name: 'bWeight', type: 'BYTE', value: 0},\n {name: 'bProportion', type: 'BYTE', value: 0},\n {name: 'bContrast', type: 'BYTE', value: 0},\n {name: 'bStrokeVariation', type: 'BYTE', value: 0},\n {name: 'bArmStyle', type: 'BYTE', value: 0},\n {name: 'bLetterform', type: 'BYTE', value: 0},\n {name: 'bMidline', type: 'BYTE', value: 0},\n {name: 'bXHeight', type: 'BYTE', value: 0},\n {name: 'ulUnicodeRange1', type: 'ULONG', value: 0},\n {name: 'ulUnicodeRange2', type: 'ULONG', value: 0},\n {name: 'ulUnicodeRange3', type: 'ULONG', value: 0},\n {name: 'ulUnicodeRange4', type: 'ULONG', value: 0},\n {name: 'achVendID', type: 'CHARARRAY', value: 'XXXX'},\n {name: 'fsSelection', type: 'USHORT', value: 0},\n {name: 'usFirstCharIndex', type: 'USHORT', value: 0},\n {name: 'usLastCharIndex', type: 'USHORT', value: 0},\n {name: 'sTypoAscender', type: 'SHORT', value: 0},\n {name: 'sTypoDescender', type: 'SHORT', value: 0},\n {name: 'sTypoLineGap', type: 'SHORT', value: 0},\n {name: 'usWinAscent', type: 'USHORT', value: 0},\n {name: 'usWinDescent', type: 'USHORT', value: 0},\n {name: 'ulCodePageRange1', type: 'ULONG', value: 0},\n {name: 'ulCodePageRange2', type: 'ULONG', value: 0},\n {name: 'sxHeight', type: 'SHORT', value: 0},\n {name: 'sCapHeight', type: 'SHORT', value: 0},\n {name: 'usDefaultChar', type: 'USHORT', value: 0},\n {name: 'usBreakChar', type: 'USHORT', value: 0},\n {name: 'usMaxContext', type: 'USHORT', value: 0}\n ], options);\n}\n\nexport default { parse: parseOS2Table, make: makeOS2Table, unicodeRanges, getUnicodeRange };\n","// The `post` table stores additional PostScript information, such as glyph names.\n// https://www.microsoft.com/typography/OTSPEC/post.htm\n\nimport { standardNames } from '../encoding';\nimport parse from '../parse';\nimport table from '../table';\n\n// Parse the PostScript `post` table\nfunction parsePostTable(data, start) {\n const post = {};\n const p = new parse.Parser(data, start);\n post.version = p.parseVersion();\n post.italicAngle = p.parseFixed();\n post.underlinePosition = p.parseShort();\n post.underlineThickness = p.parseShort();\n post.isFixedPitch = p.parseULong();\n post.minMemType42 = p.parseULong();\n post.maxMemType42 = p.parseULong();\n post.minMemType1 = p.parseULong();\n post.maxMemType1 = p.parseULong();\n switch (post.version) {\n case 1:\n post.names = standardNames.slice();\n break;\n case 2:\n post.numberOfGlyphs = p.parseUShort();\n post.glyphNameIndex = new Array(post.numberOfGlyphs);\n for (let i = 0; i < post.numberOfGlyphs; i++) {\n post.glyphNameIndex[i] = p.parseUShort();\n }\n\n post.names = [];\n for (let i = 0; i < post.numberOfGlyphs; i++) {\n if (post.glyphNameIndex[i] >= standardNames.length) {\n const nameLength = p.parseChar();\n post.names.push(p.parseString(nameLength));\n }\n }\n\n break;\n case 2.5:\n post.numberOfGlyphs = p.parseUShort();\n post.offset = new Array(post.numberOfGlyphs);\n for (let i = 0; i < post.numberOfGlyphs; i++) {\n post.offset[i] = p.parseChar();\n }\n\n break;\n }\n return post;\n}\n\nfunction makePostTable() {\n return new table.Table('post', [\n {name: 'version', type: 'FIXED', value: 0x00030000},\n {name: 'italicAngle', type: 'FIXED', value: 0},\n {name: 'underlinePosition', type: 'FWORD', value: 0},\n {name: 'underlineThickness', type: 'FWORD', value: 0},\n {name: 'isFixedPitch', type: 'ULONG', value: 0},\n {name: 'minMemType42', type: 'ULONG', value: 0},\n {name: 'maxMemType42', type: 'ULONG', value: 0},\n {name: 'minMemType1', type: 'ULONG', value: 0},\n {name: 'maxMemType1', type: 'ULONG', value: 0}\n ]);\n}\n\nexport default { parse: parsePostTable, make: makePostTable };\n","// The `GSUB` table contains ligatures, among other things.\n// https://www.microsoft.com/typography/OTSPEC/gsub.htm\n\nimport check from '../check';\nimport { Parser } from '../parse';\nimport table from '../table';\n\nconst subtableParsers = new Array(9); // subtableParsers[0] is unused\n\n// https://www.microsoft.com/typography/OTSPEC/GSUB.htm#SS\nsubtableParsers[1] = function parseLookup1() {\n const start = this.offset + this.relativeOffset;\n const substFormat = this.parseUShort();\n if (substFormat === 1) {\n return {\n substFormat: 1,\n coverage: this.parsePointer(Parser.coverage),\n deltaGlyphId: this.parseUShort()\n };\n } else if (substFormat === 2) {\n return {\n substFormat: 2,\n coverage: this.parsePointer(Parser.coverage),\n substitute: this.parseOffset16List()\n };\n }\n check.assert(false, '0x' + start.toString(16) + ': lookup type 1 format must be 1 or 2.');\n};\n\n// https://www.microsoft.com/typography/OTSPEC/GSUB.htm#MS\nsubtableParsers[2] = function parseLookup2() {\n const substFormat = this.parseUShort();\n check.argument(substFormat === 1, 'GSUB Multiple Substitution Subtable identifier-format must be 1');\n return {\n substFormat: substFormat,\n coverage: this.parsePointer(Parser.coverage),\n sequences: this.parseListOfLists()\n };\n};\n\n// https://www.microsoft.com/typography/OTSPEC/GSUB.htm#AS\nsubtableParsers[3] = function parseLookup3() {\n const substFormat = this.parseUShort();\n check.argument(substFormat === 1, 'GSUB Alternate Substitution Subtable identifier-format must be 1');\n return {\n substFormat: substFormat,\n coverage: this.parsePointer(Parser.coverage),\n alternateSets: this.parseListOfLists()\n };\n};\n\n// https://www.microsoft.com/typography/OTSPEC/GSUB.htm#LS\nsubtableParsers[4] = function parseLookup4() {\n const substFormat = this.parseUShort();\n check.argument(substFormat === 1, 'GSUB ligature table identifier-format must be 1');\n return {\n substFormat: substFormat,\n coverage: this.parsePointer(Parser.coverage),\n ligatureSets: this.parseListOfLists(function() {\n return {\n ligGlyph: this.parseUShort(),\n components: this.parseUShortList(this.parseUShort() - 1)\n };\n })\n };\n};\n\nconst lookupRecordDesc = {\n sequenceIndex: Parser.uShort,\n lookupListIndex: Parser.uShort\n};\n\n// https://www.microsoft.com/typography/OTSPEC/GSUB.htm#CSF\nsubtableParsers[5] = function parseLookup5() {\n const start = this.offset + this.relativeOffset;\n const substFormat = this.parseUShort();\n\n if (substFormat === 1) {\n return {\n substFormat: substFormat,\n coverage: this.parsePointer(Parser.coverage),\n ruleSets: this.parseListOfLists(function() {\n const glyphCount = this.parseUShort();\n const substCount = this.parseUShort();\n return {\n input: this.parseUShortList(glyphCount - 1),\n lookupRecords: this.parseRecordList(substCount, lookupRecordDesc)\n };\n })\n };\n } else if (substFormat === 2) {\n return {\n substFormat: substFormat,\n coverage: this.parsePointer(Parser.coverage),\n classDef: this.parsePointer(Parser.classDef),\n classSets: this.parseListOfLists(function() {\n const glyphCount = this.parseUShort();\n const substCount = this.parseUShort();\n return {\n classes: this.parseUShortList(glyphCount - 1),\n lookupRecords: this.parseRecordList(substCount, lookupRecordDesc)\n };\n })\n };\n } else if (substFormat === 3) {\n const glyphCount = this.parseUShort();\n const substCount = this.parseUShort();\n return {\n substFormat: substFormat,\n coverages: this.parseList(glyphCount, Parser.pointer(Parser.coverage)),\n lookupRecords: this.parseRecordList(substCount, lookupRecordDesc)\n };\n }\n check.assert(false, '0x' + start.toString(16) + ': lookup type 5 format must be 1, 2 or 3.');\n};\n\n// https://www.microsoft.com/typography/OTSPEC/GSUB.htm#CC\nsubtableParsers[6] = function parseLookup6() {\n const start = this.offset + this.relativeOffset;\n const substFormat = this.parseUShort();\n if (substFormat === 1) {\n return {\n substFormat: 1,\n coverage: this.parsePointer(Parser.coverage),\n chainRuleSets: this.parseListOfLists(function() {\n return {\n backtrack: this.parseUShortList(),\n input: this.parseUShortList(this.parseShort() - 1),\n lookahead: this.parseUShortList(),\n lookupRecords: this.parseRecordList(lookupRecordDesc)\n };\n })\n };\n } else if (substFormat === 2) {\n return {\n substFormat: 2,\n coverage: this.parsePointer(Parser.coverage),\n backtrackClassDef: this.parsePointer(Parser.classDef),\n inputClassDef: this.parsePointer(Parser.classDef),\n lookaheadClassDef: this.parsePointer(Parser.classDef),\n chainClassSet: this.parseListOfLists(function() {\n return {\n backtrack: this.parseUShortList(),\n input: this.parseUShortList(this.parseShort() - 1),\n lookahead: this.parseUShortList(),\n lookupRecords: this.parseRecordList(lookupRecordDesc)\n };\n })\n };\n } else if (substFormat === 3) {\n return {\n substFormat: 3,\n backtrackCoverage: this.parseList(Parser.pointer(Parser.coverage)),\n inputCoverage: this.parseList(Parser.pointer(Parser.coverage)),\n lookaheadCoverage: this.parseList(Parser.pointer(Parser.coverage)),\n lookupRecords: this.parseRecordList(lookupRecordDesc)\n };\n }\n check.assert(false, '0x' + start.toString(16) + ': lookup type 6 format must be 1, 2 or 3.');\n};\n\n// https://www.microsoft.com/typography/OTSPEC/GSUB.htm#ES\nsubtableParsers[7] = function parseLookup7() {\n // Extension Substitution subtable\n const substFormat = this.parseUShort();\n check.argument(substFormat === 1, 'GSUB Extension Substitution subtable identifier-format must be 1');\n const extensionLookupType = this.parseUShort();\n const extensionParser = new Parser(this.data, this.offset + this.parseULong());\n return {\n substFormat: 1,\n lookupType: extensionLookupType,\n extension: subtableParsers[extensionLookupType].call(extensionParser)\n };\n};\n\n// https://www.microsoft.com/typography/OTSPEC/GSUB.htm#RCCS\nsubtableParsers[8] = function parseLookup8() {\n const substFormat = this.parseUShort();\n check.argument(substFormat === 1, 'GSUB Reverse Chaining Contextual Single Substitution Subtable identifier-format must be 1');\n return {\n substFormat: substFormat,\n coverage: this.parsePointer(Parser.coverage),\n backtrackCoverage: this.parseList(Parser.pointer(Parser.coverage)),\n lookaheadCoverage: this.parseList(Parser.pointer(Parser.coverage)),\n substitutes: this.parseUShortList()\n };\n};\n\n// https://www.microsoft.com/typography/OTSPEC/gsub.htm\nfunction parseGsubTable(data, start) {\n start = start || 0;\n const p = new Parser(data, start);\n const tableVersion = p.parseVersion(1);\n check.argument(tableVersion === 1 || tableVersion === 1.1, 'Unsupported GSUB table version.');\n if (tableVersion === 1) {\n return {\n version: tableVersion,\n scripts: p.parseScriptList(),\n features: p.parseFeatureList(),\n lookups: p.parseLookupList(subtableParsers)\n };\n } else {\n return {\n version: tableVersion,\n scripts: p.parseScriptList(),\n features: p.parseFeatureList(),\n lookups: p.parseLookupList(subtableParsers),\n variations: p.parseFeatureVariationsList()\n };\n }\n\n}\n\n// GSUB Writing //////////////////////////////////////////////\nconst subtableMakers = new Array(9);\n\nsubtableMakers[1] = function makeLookup1(subtable) {\n if (subtable.substFormat === 1) {\n return new table.Table('substitutionTable', [\n {name: 'substFormat', type: 'USHORT', value: 1},\n {name: 'coverage', type: 'TABLE', value: new table.Coverage(subtable.coverage)},\n {name: 'deltaGlyphID', type: 'USHORT', value: subtable.deltaGlyphId}\n ]);\n } else {\n return new table.Table('substitutionTable', [\n {name: 'substFormat', type: 'USHORT', value: 2},\n {name: 'coverage', type: 'TABLE', value: new table.Coverage(subtable.coverage)}\n ].concat(table.ushortList('substitute', subtable.substitute)));\n }\n check.fail('Lookup type 1 substFormat must be 1 or 2.');\n};\n\nsubtableMakers[2] = function makeLookup2(subtable) {\n check.assert(subtable.substFormat === 1, 'Lookup type 2 substFormat must be 1.');\n return new table.Table('substitutionTable', [\n {name: 'substFormat', type: 'USHORT', value: 1},\n {name: 'coverage', type: 'TABLE', value: new table.Coverage(subtable.coverage)}\n ].concat(table.tableList('seqSet', subtable.sequences, function(sequenceSet) {\n return new table.Table('sequenceSetTable', table.ushortList('sequence', sequenceSet));\n })));\n};\n\nsubtableMakers[3] = function makeLookup3(subtable) {\n check.assert(subtable.substFormat === 1, 'Lookup type 3 substFormat must be 1.');\n return new table.Table('substitutionTable', [\n {name: 'substFormat', type: 'USHORT', value: 1},\n {name: 'coverage', type: 'TABLE', value: new table.Coverage(subtable.coverage)}\n ].concat(table.tableList('altSet', subtable.alternateSets, function(alternateSet) {\n return new table.Table('alternateSetTable', table.ushortList('alternate', alternateSet));\n })));\n};\n\nsubtableMakers[4] = function makeLookup4(subtable) {\n check.assert(subtable.substFormat === 1, 'Lookup type 4 substFormat must be 1.');\n return new table.Table('substitutionTable', [\n {name: 'substFormat', type: 'USHORT', value: 1},\n {name: 'coverage', type: 'TABLE', value: new table.Coverage(subtable.coverage)}\n ].concat(table.tableList('ligSet', subtable.ligatureSets, function(ligatureSet) {\n return new table.Table('ligatureSetTable', table.tableList('ligature', ligatureSet, function(ligature) {\n return new table.Table('ligatureTable',\n [{name: 'ligGlyph', type: 'USHORT', value: ligature.ligGlyph}]\n .concat(table.ushortList('component', ligature.components, ligature.components.length + 1))\n );\n }));\n })));\n};\n\nsubtableMakers[6] = function makeLookup6(subtable) {\n if (subtable.substFormat === 1) {\n let returnTable = new table.Table('chainContextTable', [\n {name: 'substFormat', type: 'USHORT', value: subtable.substFormat},\n {name: 'coverage', type: 'TABLE', value: new table.Coverage(subtable.coverage)}\n ].concat(table.tableList('chainRuleSet', subtable.chainRuleSets, function(chainRuleSet) {\n return new table.Table('chainRuleSetTable', table.tableList('chainRule', chainRuleSet, function(chainRule) {\n let tableData = table.ushortList('backtrackGlyph', chainRule.backtrack, chainRule.backtrack.length)\n .concat(table.ushortList('inputGlyph', chainRule.input, chainRule.input.length + 1))\n .concat(table.ushortList('lookaheadGlyph', chainRule.lookahead, chainRule.lookahead.length))\n .concat(table.ushortList('substitution', [], chainRule.lookupRecords.length));\n\n chainRule.lookupRecords.forEach((record, i) => {\n tableData = tableData\n .concat({name: 'sequenceIndex' + i, type: 'USHORT', value: record.sequenceIndex})\n .concat({name: 'lookupListIndex' + i, type: 'USHORT', value: record.lookupListIndex});\n });\n return new table.Table('chainRuleTable', tableData);\n }));\n })));\n return returnTable;\n } else if (subtable.substFormat === 2) {\n check.assert(false, 'lookup type 6 format 2 is not yet supported.');\n } else if (subtable.substFormat === 3) {\n let tableData = [\n {name: 'substFormat', type: 'USHORT', value: subtable.substFormat},\n ];\n\n tableData.push({name: 'backtrackGlyphCount', type: 'USHORT', value: subtable.backtrackCoverage.length});\n subtable.backtrackCoverage.forEach((coverage, i) => {\n tableData.push({name: 'backtrackCoverage' + i, type: 'TABLE', value: new table.Coverage(coverage)});\n });\n tableData.push({name: 'inputGlyphCount', type: 'USHORT', value: subtable.inputCoverage.length});\n subtable.inputCoverage.forEach((coverage, i) => {\n tableData.push({name: 'inputCoverage' + i, type: 'TABLE', value: new table.Coverage(coverage)});\n });\n tableData.push({name: 'lookaheadGlyphCount', type: 'USHORT', value: subtable.lookaheadCoverage.length});\n subtable.lookaheadCoverage.forEach((coverage, i) => {\n tableData.push({name: 'lookaheadCoverage' + i, type: 'TABLE', value: new table.Coverage(coverage)});\n });\n\n tableData.push({name: 'substitutionCount', type: 'USHORT', value: subtable.lookupRecords.length});\n subtable.lookupRecords.forEach((record, i) => {\n tableData = tableData\n .concat({name: 'sequenceIndex' + i, type: 'USHORT', value: record.sequenceIndex})\n .concat({name: 'lookupListIndex' + i, type: 'USHORT', value: record.lookupListIndex});\n });\n\n let returnTable = new table.Table('chainContextTable', tableData);\n\n return returnTable;\n }\n\n check.assert(false, 'lookup type 6 format must be 1, 2 or 3.');\n};\n\nfunction makeGsubTable(gsub) {\n return new table.Table('GSUB', [\n {name: 'version', type: 'ULONG', value: 0x10000},\n {name: 'scripts', type: 'TABLE', value: new table.ScriptList(gsub.scripts)},\n {name: 'features', type: 'TABLE', value: new table.FeatureList(gsub.features)},\n {name: 'lookups', type: 'TABLE', value: new table.LookupList(gsub.lookups, subtableMakers)}\n ]);\n}\n\nexport default { parse: parseGsubTable, make: makeGsubTable };\n","// The `GPOS` table contains kerning pairs, among other things.\n// https://www.microsoft.com/typography/OTSPEC/gpos.htm\n\nimport check from '../check';\nimport { decode } from '../types';\nimport parse from '../parse';\nimport table from '../table';\n\n// Parse the metadata `meta` table.\n// https://developer.apple.com/fonts/TrueType-Reference-Manual/RM06/Chap6meta.html\nfunction parseMetaTable(data, start) {\n const p = new parse.Parser(data, start);\n const tableVersion = p.parseULong();\n check.argument(tableVersion === 1, 'Unsupported META table version.');\n p.parseULong(); // flags - currently unused and set to 0\n p.parseULong(); // tableOffset\n const numDataMaps = p.parseULong();\n\n const tags = {};\n for (let i = 0; i < numDataMaps; i++) {\n const tag = p.parseTag();\n const dataOffset = p.parseULong();\n const dataLength = p.parseULong();\n const text = decode.UTF8(data, start + dataOffset, dataLength);\n\n tags[tag] = text;\n }\n return tags;\n}\n\nfunction makeMetaTable(tags) {\n const numTags = Object.keys(tags).length;\n let stringPool = '';\n const stringPoolOffset = 16 + numTags * 12;\n\n const result = new table.Table('meta', [\n {name: 'version', type: 'ULONG', value: 1},\n {name: 'flags', type: 'ULONG', value: 0},\n {name: 'offset', type: 'ULONG', value: stringPoolOffset},\n {name: 'numTags', type: 'ULONG', value: numTags}\n ]);\n\n for (let tag in tags) {\n const pos = stringPool.length;\n stringPool += tags[tag];\n\n result.fields.push({name: 'tag ' + tag, type: 'TAG', value: tag});\n result.fields.push({name: 'offset ' + tag, type: 'ULONG', value: stringPoolOffset + pos});\n result.fields.push({name: 'length ' + tag, type: 'ULONG', value: tags[tag].length});\n }\n\n result.fields.push({name: 'stringPool', type: 'CHARARRAY', value: stringPool});\n\n return result;\n}\n\nexport default { parse: parseMetaTable, make: makeMetaTable };\n","// The `sfnt` wrapper provides organization for the tables in the font.\n// It is the top-level data structure in a font.\n// https://www.microsoft.com/typography/OTSPEC/otff.htm\n// Recommendations for creating OpenType Fonts:\n// http://www.microsoft.com/typography/otspec140/recom.htm\n\nimport check from '../check';\nimport table from '../table';\n\nimport cmap from './cmap';\nimport cff from './cff';\nimport head from './head';\nimport hhea from './hhea';\nimport hmtx from './hmtx';\nimport ltag from './ltag';\nimport maxp from './maxp';\nimport _name from './name';\nimport os2 from './os2';\nimport post from './post';\nimport gsub from './gsub';\nimport meta from './meta';\n\nfunction log2(v) {\n return Math.log(v) / Math.log(2) | 0;\n}\n\nfunction computeCheckSum(bytes) {\n while (bytes.length % 4 !== 0) {\n bytes.push(0);\n }\n\n let sum = 0;\n for (let i = 0; i < bytes.length; i += 4) {\n sum += (bytes[i] << 24) +\n (bytes[i + 1] << 16) +\n (bytes[i + 2] << 8) +\n (bytes[i + 3]);\n }\n\n sum %= Math.pow(2, 32);\n return sum;\n}\n\nfunction makeTableRecord(tag, checkSum, offset, length) {\n return new table.Record('Table Record', [\n {name: 'tag', type: 'TAG', value: tag !== undefined ? tag : ''},\n {name: 'checkSum', type: 'ULONG', value: checkSum !== undefined ? checkSum : 0},\n {name: 'offset', type: 'ULONG', value: offset !== undefined ? offset : 0},\n {name: 'length', type: 'ULONG', value: length !== undefined ? length : 0}\n ]);\n}\n\nfunction makeSfntTable(tables) {\n const sfnt = new table.Table('sfnt', [\n {name: 'version', type: 'TAG', value: 'OTTO'},\n {name: 'numTables', type: 'USHORT', value: 0},\n {name: 'searchRange', type: 'USHORT', value: 0},\n {name: 'entrySelector', type: 'USHORT', value: 0},\n {name: 'rangeShift', type: 'USHORT', value: 0}\n ]);\n sfnt.tables = tables;\n sfnt.numTables = tables.length;\n const highestPowerOf2 = Math.pow(2, log2(sfnt.numTables));\n sfnt.searchRange = 16 * highestPowerOf2;\n sfnt.entrySelector = log2(highestPowerOf2);\n sfnt.rangeShift = sfnt.numTables * 16 - sfnt.searchRange;\n\n const recordFields = [];\n const tableFields = [];\n\n let offset = sfnt.sizeOf() + (makeTableRecord().sizeOf() * sfnt.numTables);\n while (offset % 4 !== 0) {\n offset += 1;\n tableFields.push({name: 'padding', type: 'BYTE', value: 0});\n }\n\n for (let i = 0; i < tables.length; i += 1) {\n const t = tables[i];\n check.argument(t.tableName.length === 4, 'Table name' + t.tableName + ' is invalid.');\n const tableLength = t.sizeOf();\n const tableRecord = makeTableRecord(t.tableName, computeCheckSum(t.encode()), offset, tableLength);\n recordFields.push({name: tableRecord.tag + ' Table Record', type: 'RECORD', value: tableRecord});\n tableFields.push({name: t.tableName + ' table', type: 'RECORD', value: t});\n offset += tableLength;\n check.argument(!isNaN(offset), 'Something went wrong calculating the offset.');\n while (offset % 4 !== 0) {\n offset += 1;\n tableFields.push({name: 'padding', type: 'BYTE', value: 0});\n }\n }\n\n // Table records need to be sorted alphabetically.\n recordFields.sort(function(r1, r2) {\n if (r1.value.tag > r2.value.tag) {\n return 1;\n } else {\n return -1;\n }\n });\n\n sfnt.fields = sfnt.fields.concat(recordFields);\n sfnt.fields = sfnt.fields.concat(tableFields);\n return sfnt;\n}\n\n// Get the metrics for a character. If the string has more than one character\n// this function returns metrics for the first available character.\n// You can provide optional fallback metrics if no characters are available.\nfunction metricsForChar(font, chars, notFoundMetrics) {\n for (let i = 0; i < chars.length; i += 1) {\n const glyphIndex = font.charToGlyphIndex(chars[i]);\n if (glyphIndex > 0) {\n const glyph = font.glyphs.get(glyphIndex);\n return glyph.getMetrics();\n }\n }\n\n return notFoundMetrics;\n}\n\nfunction average(vs) {\n let sum = 0;\n for (let i = 0; i < vs.length; i += 1) {\n sum += vs[i];\n }\n\n return sum / vs.length;\n}\n\n// Convert the font object to a SFNT data structure.\n// This structure contains all the necessary tables and metadata to create a binary OTF file.\nfunction fontToSfntTable(font) {\n const xMins = [];\n const yMins = [];\n const xMaxs = [];\n const yMaxs = [];\n const advanceWidths = [];\n const leftSideBearings = [];\n const rightSideBearings = [];\n let firstCharIndex;\n let lastCharIndex = 0;\n let ulUnicodeRange1 = 0;\n let ulUnicodeRange2 = 0;\n let ulUnicodeRange3 = 0;\n let ulUnicodeRange4 = 0;\n\n for (let i = 0; i < font.glyphs.length; i += 1) {\n const glyph = font.glyphs.get(i);\n const unicode = glyph.unicode | 0;\n\n if (isNaN(glyph.advanceWidth)) {\n throw new Error('Glyph ' + glyph.name + ' (' + i + '): advanceWidth is not a number.');\n }\n\n if (firstCharIndex > unicode || firstCharIndex === undefined) {\n // ignore .notdef char\n if (unicode > 0) {\n firstCharIndex = unicode;\n }\n }\n\n if (lastCharIndex < unicode) {\n lastCharIndex = unicode;\n }\n\n const position = os2.getUnicodeRange(unicode);\n if (position < 32) {\n ulUnicodeRange1 |= 1 << position;\n } else if (position < 64) {\n ulUnicodeRange2 |= 1 << position - 32;\n } else if (position < 96) {\n ulUnicodeRange3 |= 1 << position - 64;\n } else if (position < 123) {\n ulUnicodeRange4 |= 1 << position - 96;\n } else {\n throw new Error('Unicode ranges bits > 123 are reserved for internal usage');\n }\n // Skip non-important characters.\n if (glyph.name === '.notdef') continue;\n const metrics = glyph.getMetrics();\n xMins.push(metrics.xMin);\n yMins.push(metrics.yMin);\n xMaxs.push(metrics.xMax);\n yMaxs.push(metrics.yMax);\n leftSideBearings.push(metrics.leftSideBearing);\n rightSideBearings.push(metrics.rightSideBearing);\n advanceWidths.push(glyph.advanceWidth);\n }\n\n const globals = {\n xMin: Math.min.apply(null, xMins),\n yMin: Math.min.apply(null, yMins),\n xMax: Math.max.apply(null, xMaxs),\n yMax: Math.max.apply(null, yMaxs),\n advanceWidthMax: Math.max.apply(null, advanceWidths),\n advanceWidthAvg: average(advanceWidths),\n minLeftSideBearing: Math.min.apply(null, leftSideBearings),\n maxLeftSideBearing: Math.max.apply(null, leftSideBearings),\n minRightSideBearing: Math.min.apply(null, rightSideBearings)\n };\n globals.ascender = font.ascender;\n globals.descender = font.descender;\n\n const headTable = head.make({\n flags: 3, // 00000011 (baseline for font at y=0; left sidebearing point at x=0)\n unitsPerEm: font.unitsPerEm,\n xMin: globals.xMin,\n yMin: globals.yMin,\n xMax: globals.xMax,\n yMax: globals.yMax,\n lowestRecPPEM: 3,\n createdTimestamp: font.createdTimestamp\n });\n\n const hheaTable = hhea.make({\n ascender: globals.ascender,\n descender: globals.descender,\n advanceWidthMax: globals.advanceWidthMax,\n minLeftSideBearing: globals.minLeftSideBearing,\n minRightSideBearing: globals.minRightSideBearing,\n xMaxExtent: globals.maxLeftSideBearing + (globals.xMax - globals.xMin),\n numberOfHMetrics: font.glyphs.length\n });\n\n const maxpTable = maxp.make(font.glyphs.length);\n\n const os2Table = os2.make(Object.assign({\n xAvgCharWidth: Math.round(globals.advanceWidthAvg),\n usFirstCharIndex: firstCharIndex,\n usLastCharIndex: lastCharIndex,\n ulUnicodeRange1: ulUnicodeRange1,\n ulUnicodeRange2: ulUnicodeRange2,\n ulUnicodeRange3: ulUnicodeRange3,\n ulUnicodeRange4: ulUnicodeRange4,\n // See http://typophile.com/node/13081 for more info on vertical metrics.\n // We get metrics for typical characters (such as \"x\" for xHeight).\n // We provide some fallback characters if characters are unavailable: their\n // ordering was chosen experimentally.\n sTypoAscender: globals.ascender,\n sTypoDescender: globals.descender,\n sTypoLineGap: 0,\n usWinAscent: globals.yMax,\n usWinDescent: Math.abs(globals.yMin),\n ulCodePageRange1: 1, // FIXME: hard-code Latin 1 support for now\n sxHeight: metricsForChar(font, 'xyvw', {yMax: Math.round(globals.ascender / 2)}).yMax,\n sCapHeight: metricsForChar(font, 'HIKLEFJMNTZBDPRAGOQSUVWXY', globals).yMax,\n usDefaultChar: font.hasChar(' ') ? 32 : 0, // Use space as the default character, if available.\n usBreakChar: font.hasChar(' ') ? 32 : 0, // Use space as the break character, if available.\n }, font.tables.os2));\n\n const hmtxTable = hmtx.make(font.glyphs);\n const cmapTable = cmap.make(font.glyphs);\n\n const englishFamilyName = font.getEnglishName('fontFamily');\n const englishStyleName = font.getEnglishName('fontSubfamily');\n const englishFullName = englishFamilyName + ' ' + englishStyleName;\n let postScriptName = font.getEnglishName('postScriptName');\n if (!postScriptName) {\n postScriptName = englishFamilyName.replace(/\\s/g, '') + '-' + englishStyleName;\n }\n\n const names = {};\n for (let n in font.names) {\n names[n] = font.names[n];\n }\n\n if (!names.uniqueID) {\n names.uniqueID = {en: font.getEnglishName('manufacturer') + ':' + englishFullName};\n }\n\n if (!names.postScriptName) {\n names.postScriptName = {en: postScriptName};\n }\n\n if (!names.preferredFamily) {\n names.preferredFamily = font.names.fontFamily;\n }\n\n if (!names.preferredSubfamily) {\n names.preferredSubfamily = font.names.fontSubfamily;\n }\n\n const languageTags = [];\n const nameTable = _name.make(names, languageTags);\n const ltagTable = (languageTags.length > 0 ? ltag.make(languageTags) : undefined);\n\n const postTable = post.make();\n const cffTable = cff.make(font.glyphs, {\n version: font.getEnglishName('version'),\n fullName: englishFullName,\n familyName: englishFamilyName,\n weightName: englishStyleName,\n postScriptName: postScriptName,\n unitsPerEm: font.unitsPerEm,\n fontBBox: [0, globals.yMin, globals.ascender, globals.advanceWidthMax]\n });\n\n const metaTable = (font.metas && Object.keys(font.metas).length > 0) ? meta.make(font.metas) : undefined;\n\n // The order does not matter because makeSfntTable() will sort them.\n const tables = [headTable, hheaTable, maxpTable, os2Table, nameTable, cmapTable, postTable, cffTable, hmtxTable];\n if (ltagTable) {\n tables.push(ltagTable);\n }\n // Optional tables\n if (font.tables.gsub) {\n tables.push(gsub.make(font.tables.gsub));\n }\n if (metaTable) {\n tables.push(metaTable);\n }\n\n const sfntTable = makeSfntTable(tables);\n\n // Compute the font's checkSum and store it in head.checkSumAdjustment.\n const bytes = sfntTable.encode();\n const checkSum = computeCheckSum(bytes);\n const tableFields = sfntTable.fields;\n let checkSumAdjusted = false;\n for (let i = 0; i < tableFields.length; i += 1) {\n if (tableFields[i].name === 'head table') {\n tableFields[i].value.checkSumAdjustment = 0xB1B0AFBA - checkSum;\n checkSumAdjusted = true;\n break;\n }\n }\n\n if (!checkSumAdjusted) {\n throw new Error('Could not find head table with checkSum to adjust.');\n }\n\n return sfntTable;\n}\n\nexport default { make: makeSfntTable, fontToTable: fontToSfntTable, computeCheckSum };\n","// The Layout object is the prototype of Substitution objects, and provides\n// utility methods to manipulate common layout tables (GPOS, GSUB, GDEF...)\n\nimport check from './check';\n\nfunction searchTag(arr, tag) {\n /* jshint bitwise: false */\n let imin = 0;\n let imax = arr.length - 1;\n while (imin <= imax) {\n const imid = (imin + imax) >>> 1;\n const val = arr[imid].tag;\n if (val === tag) {\n return imid;\n } else if (val < tag) {\n imin = imid + 1;\n } else { imax = imid - 1; }\n }\n // Not found: return -1-insertion point\n return -imin - 1;\n}\n\nfunction binSearch(arr, value) {\n /* jshint bitwise: false */\n let imin = 0;\n let imax = arr.length - 1;\n while (imin <= imax) {\n const imid = (imin + imax) >>> 1;\n const val = arr[imid];\n if (val === value) {\n return imid;\n } else if (val < value) {\n imin = imid + 1;\n } else { imax = imid - 1; }\n }\n // Not found: return -1-insertion point\n return -imin - 1;\n}\n\n// binary search in a list of ranges (coverage, class definition)\nfunction searchRange(ranges, value) {\n // jshint bitwise: false\n let range;\n let imin = 0;\n let imax = ranges.length - 1;\n while (imin <= imax) {\n const imid = (imin + imax) >>> 1;\n range = ranges[imid];\n const start = range.start;\n if (start === value) {\n return range;\n } else if (start < value) {\n imin = imid + 1;\n } else { imax = imid - 1; }\n }\n if (imin > 0) {\n range = ranges[imin - 1];\n if (value > range.end) return 0;\n return range;\n }\n}\n\n/**\n * @exports opentype.Layout\n * @class\n */\nfunction Layout(font, tableName) {\n this.font = font;\n this.tableName = tableName;\n}\n\nLayout.prototype = {\n\n /**\n * Binary search an object by \"tag\" property\n * @instance\n * @function searchTag\n * @memberof opentype.Layout\n * @param {Array} arr\n * @param {string} tag\n * @return {number}\n */\n searchTag: searchTag,\n\n /**\n * Binary search in a list of numbers\n * @instance\n * @function binSearch\n * @memberof opentype.Layout\n * @param {Array} arr\n * @param {number} value\n * @return {number}\n */\n binSearch: binSearch,\n\n /**\n * Get or create the Layout table (GSUB, GPOS etc).\n * @param {boolean} create - Whether to create a new one.\n * @return {Object} The GSUB or GPOS table.\n */\n getTable: function(create) {\n let layout = this.font.tables[this.tableName];\n if (!layout && create) {\n layout = this.font.tables[this.tableName] = this.createDefaultTable();\n }\n return layout;\n },\n\n /**\n * Returns all scripts in the substitution table.\n * @instance\n * @return {Array}\n */\n getScriptNames: function() {\n let layout = this.getTable();\n if (!layout) { return []; }\n return layout.scripts.map(function(script) {\n return script.tag;\n });\n },\n\n /**\n * Returns the best bet for a script name.\n * Returns 'DFLT' if it exists.\n * If not, returns 'latn' if it exists.\n * If neither exist, returns undefined.\n */\n getDefaultScriptName: function() {\n let layout = this.getTable();\n if (!layout) { return; }\n let hasLatn = false;\n for (let i = 0; i < layout.scripts.length; i++) {\n const name = layout.scripts[i].tag;\n if (name === 'DFLT') return name;\n if (name === 'latn') hasLatn = true;\n }\n if (hasLatn) return 'latn';\n },\n\n /**\n * Returns all LangSysRecords in the given script.\n * @instance\n * @param {string} [script='DFLT']\n * @param {boolean} create - forces the creation of this script table if it doesn't exist.\n * @return {Object} An object with tag and script properties.\n */\n getScriptTable: function(script, create) {\n const layout = this.getTable(create);\n if (layout) {\n script = script || 'DFLT';\n const scripts = layout.scripts;\n const pos = searchTag(layout.scripts, script);\n if (pos >= 0) {\n return scripts[pos].script;\n } else if (create) {\n const scr = {\n tag: script,\n script: {\n defaultLangSys: {reserved: 0, reqFeatureIndex: 0xffff, featureIndexes: []},\n langSysRecords: []\n }\n };\n scripts.splice(-1 - pos, 0, scr);\n return scr.script;\n }\n }\n },\n\n /**\n * Returns a language system table\n * @instance\n * @param {string} [script='DFLT']\n * @param {string} [language='dlft']\n * @param {boolean} create - forces the creation of this langSysTable if it doesn't exist.\n * @return {Object}\n */\n getLangSysTable: function(script, language, create) {\n const scriptTable = this.getScriptTable(script, create);\n if (scriptTable) {\n if (!language || language === 'dflt' || language === 'DFLT') {\n return scriptTable.defaultLangSys;\n }\n const pos = searchTag(scriptTable.langSysRecords, language);\n if (pos >= 0) {\n return scriptTable.langSysRecords[pos].langSys;\n } else if (create) {\n const langSysRecord = {\n tag: language,\n langSys: {reserved: 0, reqFeatureIndex: 0xffff, featureIndexes: []}\n };\n scriptTable.langSysRecords.splice(-1 - pos, 0, langSysRecord);\n return langSysRecord.langSys;\n }\n }\n },\n\n /**\n * Get a specific feature table.\n * @instance\n * @param {string} [script='DFLT']\n * @param {string} [language='dlft']\n * @param {string} feature - One of the codes listed at https://www.microsoft.com/typography/OTSPEC/featurelist.htm\n * @param {boolean} create - forces the creation of the feature table if it doesn't exist.\n * @return {Object}\n */\n getFeatureTable: function(script, language, feature, create) {\n const langSysTable = this.getLangSysTable(script, language, create);\n if (langSysTable) {\n let featureRecord;\n const featIndexes = langSysTable.featureIndexes;\n const allFeatures = this.font.tables[this.tableName].features;\n // The FeatureIndex array of indices is in arbitrary order,\n // even if allFeatures is sorted alphabetically by feature tag.\n for (let i = 0; i < featIndexes.length; i++) {\n featureRecord = allFeatures[featIndexes[i]];\n if (featureRecord.tag === feature) {\n return featureRecord.feature;\n }\n }\n if (create) {\n const index = allFeatures.length;\n // Automatic ordering of features would require to shift feature indexes in the script list.\n check.assert(index === 0 || feature >= allFeatures[index - 1].tag, 'Features must be added in alphabetical order.');\n featureRecord = {\n tag: feature,\n feature: { params: 0, lookupListIndexes: [] }\n };\n allFeatures.push(featureRecord);\n featIndexes.push(index);\n return featureRecord.feature;\n }\n }\n },\n\n /**\n * Get the lookup tables of a given type for a script/language/feature.\n * @instance\n * @param {string} [script='DFLT']\n * @param {string} [language='dlft']\n * @param {string} feature - 4-letter feature code\n * @param {number} lookupType - 1 to 9\n * @param {boolean} create - forces the creation of the lookup table if it doesn't exist, with no subtables.\n * @return {Object[]}\n */\n getLookupTables: function(script, language, feature, lookupType, create) {\n const featureTable = this.getFeatureTable(script, language, feature, create);\n const tables = [];\n if (featureTable) {\n let lookupTable;\n const lookupListIndexes = featureTable.lookupListIndexes;\n const allLookups = this.font.tables[this.tableName].lookups;\n // lookupListIndexes are in no particular order, so use naive search.\n for (let i = 0; i < lookupListIndexes.length; i++) {\n lookupTable = allLookups[lookupListIndexes[i]];\n if (lookupTable.lookupType === lookupType) {\n tables.push(lookupTable);\n }\n }\n if (tables.length === 0 && create) {\n lookupTable = {\n lookupType: lookupType,\n lookupFlag: 0,\n subtables: [],\n markFilteringSet: undefined\n };\n const index = allLookups.length;\n allLookups.push(lookupTable);\n lookupListIndexes.push(index);\n return [lookupTable];\n }\n }\n return tables;\n },\n\n /**\n * Find a glyph in a class definition table\n * https://docs.microsoft.com/en-us/typography/opentype/spec/chapter2#class-definition-table\n * @param {object} classDefTable - an OpenType Layout class definition table\n * @param {number} glyphIndex - the index of the glyph to find\n * @returns {number} -1 if not found\n */\n getGlyphClass: function(classDefTable, glyphIndex) {\n switch (classDefTable.format) {\n case 1:\n if (classDefTable.startGlyph <= glyphIndex && glyphIndex < classDefTable.startGlyph + classDefTable.classes.length) {\n return classDefTable.classes[glyphIndex - classDefTable.startGlyph];\n }\n return 0;\n case 2:\n const range = searchRange(classDefTable.ranges, glyphIndex);\n return range ? range.classId : 0;\n }\n },\n\n /**\n * Find a glyph in a coverage table\n * https://docs.microsoft.com/en-us/typography/opentype/spec/chapter2#coverage-table\n * @param {object} coverageTable - an OpenType Layout coverage table\n * @param {number} glyphIndex - the index of the glyph to find\n * @returns {number} -1 if not found\n */\n getCoverageIndex: function(coverageTable, glyphIndex) {\n switch (coverageTable.format) {\n case 1:\n const index = binSearch(coverageTable.glyphs, glyphIndex);\n return index >= 0 ? index : -1;\n case 2:\n const range = searchRange(coverageTable.ranges, glyphIndex);\n return range ? range.index + glyphIndex - range.start : -1;\n }\n },\n\n /**\n * Returns the list of glyph indexes of a coverage table.\n * Format 1: the list is stored raw\n * Format 2: compact list as range records.\n * @instance\n * @param {Object} coverageTable\n * @return {Array}\n */\n expandCoverage: function(coverageTable) {\n if (coverageTable.format === 1) {\n return coverageTable.glyphs;\n } else {\n const glyphs = [];\n const ranges = coverageTable.ranges;\n for (let i = 0; i < ranges.length; i++) {\n const range = ranges[i];\n const start = range.start;\n const end = range.end;\n for (let j = start; j <= end; j++) {\n glyphs.push(j);\n }\n }\n return glyphs;\n }\n }\n\n};\n\nexport default Layout;\n","// The Position object provides utility methods to manipulate\n// the GPOS position table.\n\nimport Layout from './layout';\n\n/**\n * @exports opentype.Position\n * @class\n * @extends opentype.Layout\n * @param {opentype.Font}\n * @constructor\n */\nfunction Position(font) {\n Layout.call(this, font, 'gpos');\n}\n\nPosition.prototype = Layout.prototype;\n\n/**\n * Init some data for faster and easier access later.\n */\nPosition.prototype.init = function() {\n const script = this.getDefaultScriptName();\n this.defaultKerningTables = this.getKerningTables(script);\n};\n\n/**\n * Find a glyph pair in a list of lookup tables of type 2 and retrieve the xAdvance kerning value.\n *\n * @param {integer} leftIndex - left glyph index\n * @param {integer} rightIndex - right glyph index\n * @returns {integer}\n */\nPosition.prototype.getKerningValue = function(kerningLookups, leftIndex, rightIndex) {\n for (let i = 0; i < kerningLookups.length; i++) {\n const subtables = kerningLookups[i].subtables;\n for (let j = 0; j < subtables.length; j++) {\n const subtable = subtables[j];\n const covIndex = this.getCoverageIndex(subtable.coverage, leftIndex);\n if (covIndex < 0) continue;\n switch (subtable.posFormat) {\n case 1:\n // Search Pair Adjustment Positioning Format 1\n let pairSet = subtable.pairSets[covIndex];\n for (let k = 0; k < pairSet.length; k++) {\n let pair = pairSet[k];\n if (pair.secondGlyph === rightIndex) {\n return pair.value1 && pair.value1.xAdvance || 0;\n }\n }\n break; // left glyph found, not right glyph - try next subtable\n case 2:\n // Search Pair Adjustment Positioning Format 2\n const class1 = this.getGlyphClass(subtable.classDef1, leftIndex);\n const class2 = this.getGlyphClass(subtable.classDef2, rightIndex);\n const pair = subtable.classRecords[class1][class2];\n return pair.value1 && pair.value1.xAdvance || 0;\n }\n }\n }\n return 0;\n};\n\n/**\n * List all kerning lookup tables.\n *\n * @param {string} [script='DFLT'] - use font.position.getDefaultScriptName() for a better default value\n * @param {string} [language='dflt']\n * @return {object[]} The list of kerning lookup tables (may be empty), or undefined if there is no GPOS table (and we should use the kern table)\n */\nPosition.prototype.getKerningTables = function(script, language) {\n if (this.font.tables.gpos) {\n return this.getLookupTables(script, language, 'kern', 2);\n }\n};\n\nexport default Position;\n","// The Substitution object provides utility methods to manipulate\n// the GSUB substitution table.\n\nimport check from './check';\nimport Layout from './layout';\n\n/**\n * @exports opentype.Substitution\n * @class\n * @extends opentype.Layout\n * @param {opentype.Font}\n * @constructor\n */\nfunction Substitution(font) {\n Layout.call(this, font, 'gsub');\n}\n\n// Check if 2 arrays of primitives are equal.\nfunction arraysEqual(ar1, ar2) {\n const n = ar1.length;\n if (n !== ar2.length) { return false; }\n for (let i = 0; i < n; i++) {\n if (ar1[i] !== ar2[i]) { return false; }\n }\n return true;\n}\n\n// Find the first subtable of a lookup table in a particular format.\nfunction getSubstFormat(lookupTable, format, defaultSubtable) {\n const subtables = lookupTable.subtables;\n for (let i = 0; i < subtables.length; i++) {\n const subtable = subtables[i];\n if (subtable.substFormat === format) {\n return subtable;\n }\n }\n if (defaultSubtable) {\n subtables.push(defaultSubtable);\n return defaultSubtable;\n }\n return undefined;\n}\n\nSubstitution.prototype = Layout.prototype;\n\n/**\n * Create a default GSUB table.\n * @return {Object} gsub - The GSUB table.\n */\nSubstitution.prototype.createDefaultTable = function() {\n // Generate a default empty GSUB table with just a DFLT script and dflt lang sys.\n return {\n version: 1,\n scripts: [{\n tag: 'DFLT',\n script: {\n defaultLangSys: { reserved: 0, reqFeatureIndex: 0xffff, featureIndexes: [] },\n langSysRecords: []\n }\n }],\n features: [],\n lookups: []\n };\n};\n\n/**\n * List all single substitutions (lookup type 1) for a given script, language, and feature.\n * @param {string} [script='DFLT']\n * @param {string} [language='dflt']\n * @param {string} feature - 4-character feature name ('aalt', 'salt', 'ss01'...)\n * @return {Array} substitutions - The list of substitutions.\n */\nSubstitution.prototype.getSingle = function(feature, script, language) {\n const substitutions = [];\n const lookupTables = this.getLookupTables(script, language, feature, 1);\n for (let idx = 0; idx < lookupTables.length; idx++) {\n const subtables = lookupTables[idx].subtables;\n for (let i = 0; i < subtables.length; i++) {\n const subtable = subtables[i];\n const glyphs = this.expandCoverage(subtable.coverage);\n let j;\n if (subtable.substFormat === 1) {\n const delta = subtable.deltaGlyphId;\n for (j = 0; j < glyphs.length; j++) {\n const glyph = glyphs[j];\n substitutions.push({ sub: glyph, by: glyph + delta });\n }\n } else {\n const substitute = subtable.substitute;\n for (j = 0; j < glyphs.length; j++) {\n substitutions.push({ sub: glyphs[j], by: substitute[j] });\n }\n }\n }\n }\n return substitutions;\n};\n\n/**\n * List all multiple substitutions (lookup type 2) for a given script, language, and feature.\n * @param {string} [script='DFLT']\n * @param {string} [language='dflt']\n * @param {string} feature - 4-character feature name ('ccmp', 'stch')\n * @return {Array} substitutions - The list of substitutions.\n */\nSubstitution.prototype.getMultiple = function(feature, script, language) {\n const substitutions = [];\n const lookupTables = this.getLookupTables(script, language, feature, 2);\n for (let idx = 0; idx < lookupTables.length; idx++) {\n const subtables = lookupTables[idx].subtables;\n for (let i = 0; i < subtables.length; i++) {\n const subtable = subtables[i];\n const glyphs = this.expandCoverage(subtable.coverage);\n let j;\n\n for (j = 0; j < glyphs.length; j++) {\n const glyph = glyphs[j];\n const replacements = subtable.sequences[j];\n substitutions.push({ sub: glyph, by: replacements });\n }\n }\n }\n return substitutions;\n};\n\n/**\n * List all alternates (lookup type 3) for a given script, language, and feature.\n * @param {string} [script='DFLT']\n * @param {string} [language='dflt']\n * @param {string} feature - 4-character feature name ('aalt', 'salt'...)\n * @return {Array} alternates - The list of alternates\n */\nSubstitution.prototype.getAlternates = function(feature, script, language) {\n const alternates = [];\n const lookupTables = this.getLookupTables(script, language, feature, 3);\n for (let idx = 0; idx < lookupTables.length; idx++) {\n const subtables = lookupTables[idx].subtables;\n for (let i = 0; i < subtables.length; i++) {\n const subtable = subtables[i];\n const glyphs = this.expandCoverage(subtable.coverage);\n const alternateSets = subtable.alternateSets;\n for (let j = 0; j < glyphs.length; j++) {\n alternates.push({ sub: glyphs[j], by: alternateSets[j] });\n }\n }\n }\n return alternates;\n};\n\n/**\n * List all ligatures (lookup type 4) for a given script, language, and feature.\n * The result is an array of ligature objects like { sub: [ids], by: id }\n * @param {string} feature - 4-letter feature name ('liga', 'rlig', 'dlig'...)\n * @param {string} [script='DFLT']\n * @param {string} [language='dflt']\n * @return {Array} ligatures - The list of ligatures.\n */\nSubstitution.prototype.getLigatures = function(feature, script, language) {\n const ligatures = [];\n const lookupTables = this.getLookupTables(script, language, feature, 4);\n for (let idx = 0; idx < lookupTables.length; idx++) {\n const subtables = lookupTables[idx].subtables;\n for (let i = 0; i < subtables.length; i++) {\n const subtable = subtables[i];\n const glyphs = this.expandCoverage(subtable.coverage);\n const ligatureSets = subtable.ligatureSets;\n for (let j = 0; j < glyphs.length; j++) {\n const startGlyph = glyphs[j];\n const ligSet = ligatureSets[j];\n for (let k = 0; k < ligSet.length; k++) {\n const lig = ligSet[k];\n ligatures.push({\n sub: [startGlyph].concat(lig.components),\n by: lig.ligGlyph\n });\n }\n }\n }\n }\n return ligatures;\n};\n\n/**\n * Add or modify a single substitution (lookup type 1)\n * Format 2, more flexible, is always used.\n * @param {string} feature - 4-letter feature name ('liga', 'rlig', 'dlig'...)\n * @param {Object} substitution - { sub: id, by: id } (format 1 is not supported)\n * @param {string} [script='DFLT']\n * @param {string} [language='dflt']\n */\nSubstitution.prototype.addSingle = function(feature, substitution, script, language) {\n const lookupTable = this.getLookupTables(script, language, feature, 1, true)[0];\n const subtable = getSubstFormat(lookupTable, 2, { // lookup type 1 subtable, format 2, coverage format 1\n substFormat: 2,\n coverage: {format: 1, glyphs: []},\n substitute: []\n });\n check.assert(subtable.coverage.format === 1, 'Single: unable to modify coverage table format ' + subtable.coverage.format);\n const coverageGlyph = substitution.sub;\n let pos = this.binSearch(subtable.coverage.glyphs, coverageGlyph);\n if (pos < 0) {\n pos = -1 - pos;\n subtable.coverage.glyphs.splice(pos, 0, coverageGlyph);\n subtable.substitute.splice(pos, 0, 0);\n }\n subtable.substitute[pos] = substitution.by;\n};\n\n/**\n * Add or modify a multiple substitution (lookup type 2)\n * @param {string} feature - 4-letter feature name ('ccmp', 'stch')\n * @param {Object} substitution - { sub: id, by: [id] } for format 2.\n * @param {string} [script='DFLT']\n * @param {string} [language='dflt']\n */\nSubstitution.prototype.addMultiple = function(feature, substitution, script, language) {\n check.assert(substitution.by instanceof Array && substitution.by.length > 1, 'Multiple: \"by\" must be an array of two or more ids');\n const lookupTable = this.getLookupTables(script, language, feature, 2, true)[0];\n const subtable = getSubstFormat(lookupTable, 1, { // lookup type 2 subtable, format 1, coverage format 1\n substFormat: 1,\n coverage: {format: 1, glyphs: []},\n sequences: []\n });\n check.assert(subtable.coverage.format === 1, 'Multiple: unable to modify coverage table format ' + subtable.coverage.format);\n const coverageGlyph = substitution.sub;\n let pos = this.binSearch(subtable.coverage.glyphs, coverageGlyph);\n if (pos < 0) {\n pos = -1 - pos;\n subtable.coverage.glyphs.splice(pos, 0, coverageGlyph);\n subtable.sequences.splice(pos, 0, 0);\n }\n subtable.sequences[pos] = substitution.by;\n};\n\n/**\n * Add or modify an alternate substitution (lookup type 3)\n * @param {string} feature - 4-letter feature name ('liga', 'rlig', 'dlig'...)\n * @param {Object} substitution - { sub: id, by: [ids] }\n * @param {string} [script='DFLT']\n * @param {string} [language='dflt']\n */\nSubstitution.prototype.addAlternate = function(feature, substitution, script, language) {\n const lookupTable = this.getLookupTables(script, language, feature, 3, true)[0];\n const subtable = getSubstFormat(lookupTable, 1, { // lookup type 3 subtable, format 1, coverage format 1\n substFormat: 1,\n coverage: {format: 1, glyphs: []},\n alternateSets: []\n });\n check.assert(subtable.coverage.format === 1, 'Alternate: unable to modify coverage table format ' + subtable.coverage.format);\n const coverageGlyph = substitution.sub;\n let pos = this.binSearch(subtable.coverage.glyphs, coverageGlyph);\n if (pos < 0) {\n pos = -1 - pos;\n subtable.coverage.glyphs.splice(pos, 0, coverageGlyph);\n subtable.alternateSets.splice(pos, 0, 0);\n }\n subtable.alternateSets[pos] = substitution.by;\n};\n\n/**\n * Add a ligature (lookup type 4)\n * Ligatures with more components must be stored ahead of those with fewer components in order to be found\n * @param {string} feature - 4-letter feature name ('liga', 'rlig', 'dlig'...)\n * @param {Object} ligature - { sub: [ids], by: id }\n * @param {string} [script='DFLT']\n * @param {string} [language='dflt']\n */\nSubstitution.prototype.addLigature = function(feature, ligature, script, language) {\n const lookupTable = this.getLookupTables(script, language, feature, 4, true)[0];\n let subtable = lookupTable.subtables[0];\n if (!subtable) {\n subtable = { // lookup type 4 subtable, format 1, coverage format 1\n substFormat: 1,\n coverage: { format: 1, glyphs: [] },\n ligatureSets: []\n };\n lookupTable.subtables[0] = subtable;\n }\n check.assert(subtable.coverage.format === 1, 'Ligature: unable to modify coverage table format ' + subtable.coverage.format);\n const coverageGlyph = ligature.sub[0];\n const ligComponents = ligature.sub.slice(1);\n const ligatureTable = {\n ligGlyph: ligature.by,\n components: ligComponents\n };\n let pos = this.binSearch(subtable.coverage.glyphs, coverageGlyph);\n if (pos >= 0) {\n // ligatureSet already exists\n const ligatureSet = subtable.ligatureSets[pos];\n for (let i = 0; i < ligatureSet.length; i++) {\n // If ligature already exists, return.\n if (arraysEqual(ligatureSet[i].components, ligComponents)) {\n return;\n }\n }\n // ligature does not exist: add it.\n ligatureSet.push(ligatureTable);\n } else {\n // Create a new ligatureSet and add coverage for the first glyph.\n pos = -1 - pos;\n subtable.coverage.glyphs.splice(pos, 0, coverageGlyph);\n subtable.ligatureSets.splice(pos, 0, [ligatureTable]);\n }\n};\n\n/**\n * List all feature data for a given script and language.\n * @param {string} feature - 4-letter feature name\n * @param {string} [script='DFLT']\n * @param {string} [language='dflt']\n * @return {Array} substitutions - The list of substitutions.\n */\nSubstitution.prototype.getFeature = function(feature, script, language) {\n if (/ss\\d\\d/.test(feature)) {\n // ss01 - ss20\n return this.getSingle(feature, script, language);\n }\n switch (feature) {\n case 'aalt':\n case 'salt':\n return this.getSingle(feature, script, language)\n .concat(this.getAlternates(feature, script, language));\n case 'dlig':\n case 'liga':\n case 'rlig':\n return this.getLigatures(feature, script, language);\n case 'ccmp':\n return this.getMultiple(feature, script, language)\n .concat(this.getLigatures(feature, script, language));\n case 'stch':\n return this.getMultiple(feature, script, language);\n }\n return undefined;\n};\n\n/**\n * Add a substitution to a feature for a given script and language.\n * @param {string} feature - 4-letter feature name\n * @param {Object} sub - the substitution to add (an object like { sub: id or [ids], by: id or [ids] })\n * @param {string} [script='DFLT']\n * @param {string} [language='dflt']\n */\nSubstitution.prototype.add = function(feature, sub, script, language) {\n if (/ss\\d\\d/.test(feature)) {\n // ss01 - ss20\n return this.addSingle(feature, sub, script, language);\n }\n switch (feature) {\n case 'aalt':\n case 'salt':\n if (typeof sub.by === 'number') {\n return this.addSingle(feature, sub, script, language);\n }\n return this.addAlternate(feature, sub, script, language);\n case 'dlig':\n case 'liga':\n case 'rlig':\n return this.addLigature(feature, sub, script, language);\n case 'ccmp':\n if (sub.by instanceof Array) {\n return this.addMultiple(feature, sub, script, language);\n }\n return this.addLigature(feature, sub, script, language);\n }\n return undefined;\n};\n\nexport default Substitution;\n","function isBrowser() {\n return typeof window !== 'undefined';\n}\n\nfunction isNode() {\n return typeof window === 'undefined';\n}\n\nfunction nodeBufferToArrayBuffer(buffer) {\n const ab = new ArrayBuffer(buffer.length);\n const view = new Uint8Array(ab);\n for (let i = 0; i < buffer.length; ++i) {\n view[i] = buffer[i];\n }\n\n return ab;\n}\n\nfunction arrayBufferToNodeBuffer(ab) {\n const buffer = new Buffer(ab.byteLength);\n const view = new Uint8Array(ab);\n for (let i = 0; i < buffer.length; ++i) {\n buffer[i] = view[i];\n }\n\n return buffer;\n}\n\nfunction checkArgument(expression, message) {\n if (!expression) {\n throw message;\n }\n}\n\nexport { isBrowser, isNode, nodeBufferToArrayBuffer, arrayBufferToNodeBuffer, checkArgument };\n","// The `glyf` table describes the glyphs in TrueType outline format.\n// http://www.microsoft.com/typography/otspec/glyf.htm\n\nimport check from '../check';\nimport glyphset from '../glyphset';\nimport parse from '../parse';\nimport Path from '../path';\n\n// Parse the coordinate data for a glyph.\nfunction parseGlyphCoordinate(p, flag, previousValue, shortVectorBitMask, sameBitMask) {\n let v;\n if ((flag & shortVectorBitMask) > 0) {\n // The coordinate is 1 byte long.\n v = p.parseByte();\n // The `same` bit is re-used for short values to signify the sign of the value.\n if ((flag & sameBitMask) === 0) {\n v = -v;\n }\n\n v = previousValue + v;\n } else {\n // The coordinate is 2 bytes long.\n // If the `same` bit is set, the coordinate is the same as the previous coordinate.\n if ((flag & sameBitMask) > 0) {\n v = previousValue;\n } else {\n // Parse the coordinate as a signed 16-bit delta value.\n v = previousValue + p.parseShort();\n }\n }\n\n return v;\n}\n\n// Parse a TrueType glyph.\nfunction parseGlyph(glyph, data, start) {\n const p = new parse.Parser(data, start);\n glyph.numberOfContours = p.parseShort();\n glyph._xMin = p.parseShort();\n glyph._yMin = p.parseShort();\n glyph._xMax = p.parseShort();\n glyph._yMax = p.parseShort();\n let flags;\n let flag;\n\n if (glyph.numberOfContours > 0) {\n // This glyph is not a composite.\n const endPointIndices = glyph.endPointIndices = [];\n for (let i = 0; i < glyph.numberOfContours; i += 1) {\n endPointIndices.push(p.parseUShort());\n }\n\n glyph.instructionLength = p.parseUShort();\n glyph.instructions = [];\n for (let i = 0; i < glyph.instructionLength; i += 1) {\n glyph.instructions.push(p.parseByte());\n }\n\n const numberOfCoordinates = endPointIndices[endPointIndices.length - 1] + 1;\n flags = [];\n for (let i = 0; i < numberOfCoordinates; i += 1) {\n flag = p.parseByte();\n flags.push(flag);\n // If bit 3 is set, we repeat this flag n times, where n is the next byte.\n if ((flag & 8) > 0) {\n const repeatCount = p.parseByte();\n for (let j = 0; j < repeatCount; j += 1) {\n flags.push(flag);\n i += 1;\n }\n }\n }\n\n check.argument(flags.length === numberOfCoordinates, 'Bad flags.');\n\n if (endPointIndices.length > 0) {\n const points = [];\n let point;\n // X/Y coordinates are relative to the previous point, except for the first point which is relative to 0,0.\n if (numberOfCoordinates > 0) {\n for (let i = 0; i < numberOfCoordinates; i += 1) {\n flag = flags[i];\n point = {};\n point.onCurve = !!(flag & 1);\n point.lastPointOfContour = endPointIndices.indexOf(i) >= 0;\n points.push(point);\n }\n\n let px = 0;\n for (let i = 0; i < numberOfCoordinates; i += 1) {\n flag = flags[i];\n point = points[i];\n point.x = parseGlyphCoordinate(p, flag, px, 2, 16);\n px = point.x;\n }\n\n let py = 0;\n for (let i = 0; i < numberOfCoordinates; i += 1) {\n flag = flags[i];\n point = points[i];\n point.y = parseGlyphCoordinate(p, flag, py, 4, 32);\n py = point.y;\n }\n }\n\n glyph.points = points;\n } else {\n glyph.points = [];\n }\n } else if (glyph.numberOfContours === 0) {\n glyph.points = [];\n } else {\n glyph.isComposite = true;\n glyph.points = [];\n glyph.components = [];\n let moreComponents = true;\n while (moreComponents) {\n flags = p.parseUShort();\n const component = {\n glyphIndex: p.parseUShort(),\n xScale: 1,\n scale01: 0,\n scale10: 0,\n yScale: 1,\n dx: 0,\n dy: 0\n };\n if ((flags & 1) > 0) {\n // The arguments are words\n if ((flags & 2) > 0) {\n // values are offset\n component.dx = p.parseShort();\n component.dy = p.parseShort();\n } else {\n // values are matched points\n component.matchedPoints = [p.parseUShort(), p.parseUShort()];\n }\n\n } else {\n // The arguments are bytes\n if ((flags & 2) > 0) {\n // values are offset\n component.dx = p.parseChar();\n component.dy = p.parseChar();\n } else {\n // values are matched points\n component.matchedPoints = [p.parseByte(), p.parseByte()];\n }\n }\n\n if ((flags & 8) > 0) {\n // We have a scale\n component.xScale = component.yScale = p.parseF2Dot14();\n } else if ((flags & 64) > 0) {\n // We have an X / Y scale\n component.xScale = p.parseF2Dot14();\n component.yScale = p.parseF2Dot14();\n } else if ((flags & 128) > 0) {\n // We have a 2x2 transformation\n component.xScale = p.parseF2Dot14();\n component.scale01 = p.parseF2Dot14();\n component.scale10 = p.parseF2Dot14();\n component.yScale = p.parseF2Dot14();\n }\n\n glyph.components.push(component);\n moreComponents = !!(flags & 32);\n }\n if (flags & 0x100) {\n // We have instructions\n glyph.instructionLength = p.parseUShort();\n glyph.instructions = [];\n for (let i = 0; i < glyph.instructionLength; i += 1) {\n glyph.instructions.push(p.parseByte());\n }\n }\n }\n}\n\n// Transform an array of points and return a new array.\nfunction transformPoints(points, transform) {\n const newPoints = [];\n for (let i = 0; i < points.length; i += 1) {\n const pt = points[i];\n const newPt = {\n x: transform.xScale * pt.x + transform.scale01 * pt.y + transform.dx,\n y: transform.scale10 * pt.x + transform.yScale * pt.y + transform.dy,\n onCurve: pt.onCurve,\n lastPointOfContour: pt.lastPointOfContour\n };\n newPoints.push(newPt);\n }\n\n return newPoints;\n}\n\nfunction getContours(points) {\n const contours = [];\n let currentContour = [];\n for (let i = 0; i < points.length; i += 1) {\n const pt = points[i];\n currentContour.push(pt);\n if (pt.lastPointOfContour) {\n contours.push(currentContour);\n currentContour = [];\n }\n }\n\n check.argument(currentContour.length === 0, 'There are still points left in the current contour.');\n return contours;\n}\n\n// Convert the TrueType glyph outline to a Path.\nfunction getPath(points) {\n const p = new Path();\n if (!points) {\n return p;\n }\n\n const contours = getContours(points);\n\n for (let contourIndex = 0; contourIndex < contours.length; ++contourIndex) {\n const contour = contours[contourIndex];\n\n let prev = null;\n let curr = contour[contour.length - 1];\n let next = contour[0];\n\n if (curr.onCurve) {\n p.moveTo(curr.x, curr.y);\n } else {\n if (next.onCurve) {\n p.moveTo(next.x, next.y);\n } else {\n // If both first and last points are off-curve, start at their middle.\n const start = {x: (curr.x + next.x) * 0.5, y: (curr.y + next.y) * 0.5};\n p.moveTo(start.x, start.y);\n }\n }\n\n for (let i = 0; i < contour.length; ++i) {\n prev = curr;\n curr = next;\n next = contour[(i + 1) % contour.length];\n\n if (curr.onCurve) {\n // This is a straight line.\n p.lineTo(curr.x, curr.y);\n } else {\n let prev2 = prev;\n let next2 = next;\n\n if (!prev.onCurve) {\n prev2 = { x: (curr.x + prev.x) * 0.5, y: (curr.y + prev.y) * 0.5 };\n }\n\n if (!next.onCurve) {\n next2 = { x: (curr.x + next.x) * 0.5, y: (curr.y + next.y) * 0.5 };\n }\n\n p.quadraticCurveTo(curr.x, curr.y, next2.x, next2.y);\n }\n }\n\n p.closePath();\n }\n return p;\n}\n\nfunction buildPath(glyphs, glyph) {\n if (glyph.isComposite) {\n for (let j = 0; j < glyph.components.length; j += 1) {\n const component = glyph.components[j];\n const componentGlyph = glyphs.get(component.glyphIndex);\n // Force the ttfGlyphLoader to parse the glyph.\n componentGlyph.getPath();\n if (componentGlyph.points) {\n let transformedPoints;\n if (component.matchedPoints === undefined) {\n // component positioned by offset\n transformedPoints = transformPoints(componentGlyph.points, component);\n } else {\n // component positioned by matched points\n if ((component.matchedPoints[0] > glyph.points.length - 1) ||\n (component.matchedPoints[1] > componentGlyph.points.length - 1)) {\n throw Error('Matched points out of range in ' + glyph.name);\n }\n const firstPt = glyph.points[component.matchedPoints[0]];\n let secondPt = componentGlyph.points[component.matchedPoints[1]];\n const transform = {\n xScale: component.xScale, scale01: component.scale01,\n scale10: component.scale10, yScale: component.yScale,\n dx: 0, dy: 0\n };\n secondPt = transformPoints([secondPt], transform)[0];\n transform.dx = firstPt.x - secondPt.x;\n transform.dy = firstPt.y - secondPt.y;\n transformedPoints = transformPoints(componentGlyph.points, transform);\n }\n glyph.points = glyph.points.concat(transformedPoints);\n }\n }\n }\n\n return getPath(glyph.points);\n}\n\nfunction parseGlyfTableAll(data, start, loca, font) {\n const glyphs = new glyphset.GlyphSet(font);\n\n // The last element of the loca table is invalid.\n for (let i = 0; i < loca.length - 1; i += 1) {\n const offset = loca[i];\n const nextOffset = loca[i + 1];\n if (offset !== nextOffset) {\n glyphs.push(i, glyphset.ttfGlyphLoader(font, i, parseGlyph, data, start + offset, buildPath));\n } else {\n glyphs.push(i, glyphset.glyphLoader(font, i));\n }\n }\n\n return glyphs;\n}\n\nfunction parseGlyfTableOnLowMemory(data, start, loca, font) {\n const glyphs = new glyphset.GlyphSet(font);\n\n font._push = function(i) {\n const offset = loca[i];\n const nextOffset = loca[i + 1];\n if (offset !== nextOffset) {\n glyphs.push(i, glyphset.ttfGlyphLoader(font, i, parseGlyph, data, start + offset, buildPath));\n } else {\n glyphs.push(i, glyphset.glyphLoader(font, i));\n }\n };\n\n return glyphs;\n}\n\n// Parse all the glyphs according to the offsets from the `loca` table.\nfunction parseGlyfTable(data, start, loca, font, opt) {\n if (opt.lowMemory)\n return parseGlyfTableOnLowMemory(data, start, loca, font);\n else\n return parseGlyfTableAll(data, start, loca, font);\n}\n\nexport default { getPath, parse: parseGlyfTable};\n","/* A TrueType font hinting interpreter.\n*\n* (c) 2017 Axel Kittenberger\n*\n* This interpreter has been implemented according to this documentation:\n* https://developer.apple.com/fonts/TrueType-Reference-Manual/RM05/Chap5.html\n*\n* According to the documentation F24DOT6 values are used for pixels.\n* That means calculation is 1/64 pixel accurate and uses integer operations.\n* However, Javascript has floating point operations by default and only\n* those are available. One could make a case to simulate the 1/64 accuracy\n* exactly by truncating after every division operation\n* (for example with << 0) to get pixel exactly results as other TrueType\n* implementations. It may make sense since some fonts are pixel optimized\n* by hand using DELTAP instructions. The current implementation doesn't\n* and rather uses full floating point precision.\n*\n* xScale, yScale and rotation is currently ignored.\n*\n* A few non-trivial instructions are missing as I didn't encounter yet\n* a font that used them to test a possible implementation.\n*\n* Some fonts seem to use undocumented features regarding the twilight zone.\n* Only some of them are implemented as they were encountered.\n*\n* The exports.DEBUG statements are removed on the minified distribution file.\n*/\n'use strict';\n\nimport glyf from './tables/glyf';\n\nlet instructionTable;\nlet exec;\nlet execGlyph;\nlet execComponent;\n\n/*\n* Creates a hinting object.\n*\n* There ought to be exactly one\n* for each truetype font that is used for hinting.\n*/\nfunction Hinting(font) {\n // the font this hinting object is for\n this.font = font;\n\n this.getCommands = function (hPoints) {\n return glyf.getPath(hPoints).commands;\n };\n\n // cached states\n this._fpgmState =\n this._prepState =\n undefined;\n\n // errorState\n // 0 ... all okay\n // 1 ... had an error in a glyf,\n // continue working but stop spamming\n // the console\n // 2 ... error at prep, stop hinting at this ppem\n // 3 ... error at fpeg, stop hinting for this font at all\n this._errorState = 0;\n}\n\n/*\n* Not rounding.\n*/\nfunction roundOff(v) {\n return v;\n}\n\n/*\n* Rounding to grid.\n*/\nfunction roundToGrid(v) {\n //Rounding in TT is supposed to \"symmetrical around zero\"\n return Math.sign(v) * Math.round(Math.abs(v));\n}\n\n/*\n* Rounding to double grid.\n*/\nfunction roundToDoubleGrid(v) {\n return Math.sign(v) * Math.round(Math.abs(v * 2)) / 2;\n}\n\n/*\n* Rounding to half grid.\n*/\nfunction roundToHalfGrid(v) {\n return Math.sign(v) * (Math.round(Math.abs(v) + 0.5) - 0.5);\n}\n\n/*\n* Rounding to up to grid.\n*/\nfunction roundUpToGrid(v) {\n return Math.sign(v) * Math.ceil(Math.abs(v));\n}\n\n/*\n* Rounding to down to grid.\n*/\nfunction roundDownToGrid(v) {\n return Math.sign(v) * Math.floor(Math.abs(v));\n}\n\n/*\n* Super rounding.\n*/\nconst roundSuper = function (v) {\n const period = this.srPeriod;\n let phase = this.srPhase;\n const threshold = this.srThreshold;\n let sign = 1;\n\n if (v < 0) {\n v = -v;\n sign = -1;\n }\n\n v += threshold - phase;\n\n v = Math.trunc(v / period) * period;\n\n v += phase;\n\n // according to http://xgridfit.sourceforge.net/round.html\n if (v < 0) return phase * sign;\n\n return v * sign;\n};\n\n/*\n* Unit vector of x-axis.\n*/\nconst xUnitVector = {\n x: 1,\n\n y: 0,\n\n axis: 'x',\n\n // Gets the projected distance between two points.\n // o1/o2 ... if true, respective original position is used.\n distance: function (p1, p2, o1, o2) {\n return (o1 ? p1.xo : p1.x) - (o2 ? p2.xo : p2.x);\n },\n\n // Moves point p so the moved position has the same relative\n // position to the moved positions of rp1 and rp2 than the\n // original positions had.\n //\n // See APPENDIX on INTERPOLATE at the bottom of this file.\n interpolate: function (p, rp1, rp2, pv) {\n let do1;\n let do2;\n let doa1;\n let doa2;\n let dm1;\n let dm2;\n let dt;\n\n if (!pv || pv === this) {\n do1 = p.xo - rp1.xo;\n do2 = p.xo - rp2.xo;\n dm1 = rp1.x - rp1.xo;\n dm2 = rp2.x - rp2.xo;\n doa1 = Math.abs(do1);\n doa2 = Math.abs(do2);\n dt = doa1 + doa2;\n\n if (dt === 0) {\n p.x = p.xo + (dm1 + dm2) / 2;\n return;\n }\n\n p.x = p.xo + (dm1 * doa2 + dm2 * doa1) / dt;\n return;\n }\n\n do1 = pv.distance(p, rp1, true, true);\n do2 = pv.distance(p, rp2, true, true);\n dm1 = pv.distance(rp1, rp1, false, true);\n dm2 = pv.distance(rp2, rp2, false, true);\n doa1 = Math.abs(do1);\n doa2 = Math.abs(do2);\n dt = doa1 + doa2;\n\n if (dt === 0) {\n xUnitVector.setRelative(p, p, (dm1 + dm2) / 2, pv, true);\n return;\n }\n\n xUnitVector.setRelative(p, p, (dm1 * doa2 + dm2 * doa1) / dt, pv, true);\n },\n\n // Slope of line normal to this\n normalSlope: Number.NEGATIVE_INFINITY,\n\n // Sets the point 'p' relative to point 'rp'\n // by the distance 'd'.\n //\n // See APPENDIX on SETRELATIVE at the bottom of this file.\n //\n // p ... point to set\n // rp ... reference point\n // d ... distance on projection vector\n // pv ... projection vector (undefined = this)\n // org ... if true, uses the original position of rp as reference.\n setRelative: function (p, rp, d, pv, org) {\n if (!pv || pv === this) {\n p.x = (org ? rp.xo : rp.x) + d;\n return;\n }\n\n const rpx = org ? rp.xo : rp.x;\n const rpy = org ? rp.yo : rp.y;\n const rpdx = rpx + d * pv.x;\n const rpdy = rpy + d * pv.y;\n\n p.x = rpdx + (p.y - rpdy) / pv.normalSlope;\n },\n\n // Slope of vector line.\n slope: 0,\n\n // Touches the point p.\n touch: function (p) {\n p.xTouched = true;\n },\n\n // Tests if a point p is touched.\n touched: function (p) {\n return p.xTouched;\n },\n\n // Untouches the point p.\n untouch: function (p) {\n p.xTouched = false;\n }\n};\n\n/*\n* Unit vector of y-axis.\n*/\nconst yUnitVector = {\n x: 0,\n\n y: 1,\n\n axis: 'y',\n\n // Gets the projected distance between two points.\n // o1/o2 ... if true, respective original position is used.\n distance: function (p1, p2, o1, o2) {\n return (o1 ? p1.yo : p1.y) - (o2 ? p2.yo : p2.y);\n },\n\n // Moves point p so the moved position has the same relative\n // position to the moved positions of rp1 and rp2 than the\n // original positions had.\n //\n // See APPENDIX on INTERPOLATE at the bottom of this file.\n interpolate: function (p, rp1, rp2, pv) {\n let do1;\n let do2;\n let doa1;\n let doa2;\n let dm1;\n let dm2;\n let dt;\n\n if (!pv || pv === this) {\n do1 = p.yo - rp1.yo;\n do2 = p.yo - rp2.yo;\n dm1 = rp1.y - rp1.yo;\n dm2 = rp2.y - rp2.yo;\n doa1 = Math.abs(do1);\n doa2 = Math.abs(do2);\n dt = doa1 + doa2;\n\n if (dt === 0) {\n p.y = p.yo + (dm1 + dm2) / 2;\n return;\n }\n\n p.y = p.yo + (dm1 * doa2 + dm2 * doa1) / dt;\n return;\n }\n\n do1 = pv.distance(p, rp1, true, true);\n do2 = pv.distance(p, rp2, true, true);\n dm1 = pv.distance(rp1, rp1, false, true);\n dm2 = pv.distance(rp2, rp2, false, true);\n doa1 = Math.abs(do1);\n doa2 = Math.abs(do2);\n dt = doa1 + doa2;\n\n if (dt === 0) {\n yUnitVector.setRelative(p, p, (dm1 + dm2) / 2, pv, true);\n return;\n }\n\n yUnitVector.setRelative(p, p, (dm1 * doa2 + dm2 * doa1) / dt, pv, true);\n },\n\n // Slope of line normal to this.\n normalSlope: 0,\n\n // Sets the point 'p' relative to point 'rp'\n // by the distance 'd'\n //\n // See APPENDIX on SETRELATIVE at the bottom of this file.\n //\n // p ... point to set\n // rp ... reference point\n // d ... distance on projection vector\n // pv ... projection vector (undefined = this)\n // org ... if true, uses the original position of rp as reference.\n setRelative: function (p, rp, d, pv, org) {\n if (!pv || pv === this) {\n p.y = (org ? rp.yo : rp.y) + d;\n return;\n }\n\n const rpx = org ? rp.xo : rp.x;\n const rpy = org ? rp.yo : rp.y;\n const rpdx = rpx + d * pv.x;\n const rpdy = rpy + d * pv.y;\n\n p.y = rpdy + pv.normalSlope * (p.x - rpdx);\n },\n\n // Slope of vector line.\n slope: Number.POSITIVE_INFINITY,\n\n // Touches the point p.\n touch: function (p) {\n p.yTouched = true;\n },\n\n // Tests if a point p is touched.\n touched: function (p) {\n return p.yTouched;\n },\n\n // Untouches the point p.\n untouch: function (p) {\n p.yTouched = false;\n }\n};\n\nObject.freeze(xUnitVector);\nObject.freeze(yUnitVector);\n\n/*\n* Creates a unit vector that is not x- or y-axis.\n*/\nfunction UnitVector(x, y) {\n this.x = x;\n this.y = y;\n this.axis = undefined;\n this.slope = y / x;\n this.normalSlope = -x / y;\n Object.freeze(this);\n}\n\n/*\n* Gets the projected distance between two points.\n* o1/o2 ... if true, respective original position is used.\n*/\nUnitVector.prototype.distance = function(p1, p2, o1, o2) {\n return (\n this.x * xUnitVector.distance(p1, p2, o1, o2) +\n this.y * yUnitVector.distance(p1, p2, o1, o2)\n );\n};\n\n/*\n* Moves point p so the moved position has the same relative\n* position to the moved positions of rp1 and rp2 than the\n* original positions had.\n*\n* See APPENDIX on INTERPOLATE at the bottom of this file.\n*/\nUnitVector.prototype.interpolate = function(p, rp1, rp2, pv) {\n let dm1;\n let dm2;\n let do1;\n let do2;\n let doa1;\n let doa2;\n let dt;\n\n do1 = pv.distance(p, rp1, true, true);\n do2 = pv.distance(p, rp2, true, true);\n dm1 = pv.distance(rp1, rp1, false, true);\n dm2 = pv.distance(rp2, rp2, false, true);\n doa1 = Math.abs(do1);\n doa2 = Math.abs(do2);\n dt = doa1 + doa2;\n\n if (dt === 0) {\n this.setRelative(p, p, (dm1 + dm2) / 2, pv, true);\n return;\n }\n\n this.setRelative(p, p, (dm1 * doa2 + dm2 * doa1) / dt, pv, true);\n};\n\n/*\n* Sets the point 'p' relative to point 'rp'\n* by the distance 'd'\n*\n* See APPENDIX on SETRELATIVE at the bottom of this file.\n*\n* p ... point to set\n* rp ... reference point\n* d ... distance on projection vector\n* pv ... projection vector (undefined = this)\n* org ... if true, uses the original position of rp as reference.\n*/\nUnitVector.prototype.setRelative = function(p, rp, d, pv, org) {\n pv = pv || this;\n\n const rpx = org ? rp.xo : rp.x;\n const rpy = org ? rp.yo : rp.y;\n const rpdx = rpx + d * pv.x;\n const rpdy = rpy + d * pv.y;\n\n const pvns = pv.normalSlope;\n const fvs = this.slope;\n\n const px = p.x;\n const py = p.y;\n\n p.x = (fvs * px - pvns * rpdx + rpdy - py) / (fvs - pvns);\n p.y = fvs * (p.x - px) + py;\n};\n\n/*\n* Touches the point p.\n*/\nUnitVector.prototype.touch = function(p) {\n p.xTouched = true;\n p.yTouched = true;\n};\n\n/*\n* Returns a unit vector with x/y coordinates.\n*/\nfunction getUnitVector(x, y) {\n const d = Math.sqrt(x * x + y * y);\n\n x /= d;\n y /= d;\n\n if (x === 1 && y === 0) return xUnitVector;\n else if (x === 0 && y === 1) return yUnitVector;\n else return new UnitVector(x, y);\n}\n\n/*\n* Creates a point in the hinting engine.\n*/\nfunction HPoint(\n x,\n y,\n lastPointOfContour,\n onCurve\n) {\n this.x = this.xo = Math.round(x * 64) / 64; // hinted x value and original x-value\n this.y = this.yo = Math.round(y * 64) / 64; // hinted y value and original y-value\n\n this.lastPointOfContour = lastPointOfContour;\n this.onCurve = onCurve;\n this.prevPointOnContour = undefined;\n this.nextPointOnContour = undefined;\n this.xTouched = false;\n this.yTouched = false;\n\n Object.preventExtensions(this);\n}\n\n/*\n* Returns the next touched point on the contour.\n*\n* v ... unit vector to test touch axis.\n*/\nHPoint.prototype.nextTouched = function(v) {\n let p = this.nextPointOnContour;\n\n while (!v.touched(p) && p !== this) p = p.nextPointOnContour;\n\n return p;\n};\n\n/*\n* Returns the previous touched point on the contour\n*\n* v ... unit vector to test touch axis.\n*/\nHPoint.prototype.prevTouched = function(v) {\n let p = this.prevPointOnContour;\n\n while (!v.touched(p) && p !== this) p = p.prevPointOnContour;\n\n return p;\n};\n\n/*\n* The zero point.\n*/\nconst HPZero = Object.freeze(new HPoint(0, 0));\n\n/*\n* The default state of the interpreter.\n*\n* Note: Freezing the defaultState and then deriving from it\n* makes the V8 Javascript engine going awkward,\n* so this is avoided, albeit the defaultState shouldn't\n* ever change.\n*/\nconst defaultState = {\n cvCutIn: 17 / 16, // control value cut in\n deltaBase: 9,\n deltaShift: 0.125,\n loop: 1, // loops some instructions\n minDis: 1, // minimum distance\n autoFlip: true\n};\n\n/*\n* The current state of the interpreter.\n*\n* env ... 'fpgm' or 'prep' or 'glyf'\n* prog ... the program\n*/\nfunction State(env, prog) {\n this.env = env;\n this.stack = [];\n this.prog = prog;\n\n switch (env) {\n case 'glyf' :\n this.zp0 = this.zp1 = this.zp2 = 1;\n this.rp0 = this.rp1 = this.rp2 = 0;\n /* fall through */\n case 'prep' :\n this.fv = this.pv = this.dpv = xUnitVector;\n this.round = roundToGrid;\n }\n}\n\n/*\n* Executes a glyph program.\n*\n* This does the hinting for each glyph.\n*\n* Returns an array of moved points.\n*\n* glyph: the glyph to hint\n* ppem: the size the glyph is rendered for\n*/\nHinting.prototype.exec = function(glyph, ppem) {\n if (typeof ppem !== 'number') {\n throw new Error('Point size is not a number!');\n }\n\n // Received a fatal error, don't do any hinting anymore.\n if (this._errorState > 2) return;\n\n const font = this.font;\n let prepState = this._prepState;\n\n if (!prepState || prepState.ppem !== ppem) {\n let fpgmState = this._fpgmState;\n\n if (!fpgmState) {\n // Executes the fpgm state.\n // This is used by fonts to define functions.\n State.prototype = defaultState;\n\n fpgmState =\n this._fpgmState =\n new State('fpgm', font.tables.fpgm);\n\n fpgmState.funcs = [ ];\n fpgmState.font = font;\n\n if (exports.DEBUG) {\n console.log('---EXEC FPGM---');\n fpgmState.step = -1;\n }\n\n try {\n exec(fpgmState);\n } catch (e) {\n console.log('Hinting error in FPGM:' + e);\n this._errorState = 3;\n return;\n }\n }\n\n // Executes the prep program for this ppem setting.\n // This is used by fonts to set cvt values\n // depending on to be rendered font size.\n\n State.prototype = fpgmState;\n prepState =\n this._prepState =\n new State('prep', font.tables.prep);\n\n prepState.ppem = ppem;\n\n // Creates a copy of the cvt table\n // and scales it to the current ppem setting.\n const oCvt = font.tables.cvt;\n if (oCvt) {\n const cvt = prepState.cvt = new Array(oCvt.length);\n const scale = ppem / font.unitsPerEm;\n for (let c = 0; c < oCvt.length; c++) {\n cvt[c] = oCvt[c] * scale;\n }\n } else {\n prepState.cvt = [];\n }\n\n if (exports.DEBUG) {\n console.log('---EXEC PREP---');\n prepState.step = -1;\n }\n\n try {\n exec(prepState);\n } catch (e) {\n if (this._errorState < 2) {\n console.log('Hinting error in PREP:' + e);\n }\n this._errorState = 2;\n }\n }\n\n if (this._errorState > 1) return;\n\n try {\n return execGlyph(glyph, prepState);\n } catch (e) {\n if (this._errorState < 1) {\n console.log('Hinting error:' + e);\n console.log('Note: further hinting errors are silenced');\n }\n this._errorState = 1;\n return undefined;\n }\n};\n\n/*\n* Executes the hinting program for a glyph.\n*/\nexecGlyph = function(glyph, prepState) {\n // original point positions\n const xScale = prepState.ppem / prepState.font.unitsPerEm;\n const yScale = xScale;\n let components = glyph.components;\n let contours;\n let gZone;\n let state;\n\n State.prototype = prepState;\n if (!components) {\n state = new State('glyf', glyph.instructions);\n if (exports.DEBUG) {\n console.log('---EXEC GLYPH---');\n state.step = -1;\n }\n execComponent(glyph, state, xScale, yScale);\n gZone = state.gZone;\n } else {\n const font = prepState.font;\n gZone = [];\n contours = [];\n for (let i = 0; i < components.length; i++) {\n const c = components[i];\n const cg = font.glyphs.get(c.glyphIndex);\n\n state = new State('glyf', cg.instructions);\n\n if (exports.DEBUG) {\n console.log('---EXEC COMP ' + i + '---');\n state.step = -1;\n }\n\n execComponent(cg, state, xScale, yScale);\n // appends the computed points to the result array\n // post processes the component points\n const dx = Math.round(c.dx * xScale);\n const dy = Math.round(c.dy * yScale);\n const gz = state.gZone;\n const cc = state.contours;\n for (let pi = 0; pi < gz.length; pi++) {\n const p = gz[pi];\n p.xTouched = p.yTouched = false;\n p.xo = p.x = p.x + dx;\n p.yo = p.y = p.y + dy;\n }\n\n const gLen = gZone.length;\n gZone.push.apply(gZone, gz);\n for (let j = 0; j < cc.length; j++) {\n contours.push(cc[j] + gLen);\n }\n }\n\n if (glyph.instructions && !state.inhibitGridFit) {\n // the composite has instructions on its own\n state = new State('glyf', glyph.instructions);\n\n state.gZone = state.z0 = state.z1 = state.z2 = gZone;\n\n state.contours = contours;\n\n // note: HPZero cannot be used here, since\n // the point might be modified\n gZone.push(\n new HPoint(0, 0),\n new HPoint(Math.round(glyph.advanceWidth * xScale), 0)\n );\n\n if (exports.DEBUG) {\n console.log('---EXEC COMPOSITE---');\n state.step = -1;\n }\n\n exec(state);\n\n gZone.length -= 2;\n }\n }\n\n return gZone;\n};\n\n/*\n* Executes the hinting program for a component of a multi-component glyph\n* or of the glyph itself for a non-component glyph.\n*/\nexecComponent = function(glyph, state, xScale, yScale)\n{\n const points = glyph.points || [];\n const pLen = points.length;\n const gZone = state.gZone = state.z0 = state.z1 = state.z2 = [];\n const contours = state.contours = [];\n\n // Scales the original points and\n // makes copies for the hinted points.\n let cp; // current point\n for (let i = 0; i < pLen; i++) {\n cp = points[i];\n\n gZone[i] = new HPoint(\n cp.x * xScale,\n cp.y * yScale,\n cp.lastPointOfContour,\n cp.onCurve\n );\n }\n\n // Chain links the contours.\n let sp; // start point\n let np; // next point\n\n for (let i = 0; i < pLen; i++) {\n cp = gZone[i];\n\n if (!sp) {\n sp = cp;\n contours.push(i);\n }\n\n if (cp.lastPointOfContour) {\n cp.nextPointOnContour = sp;\n sp.prevPointOnContour = cp;\n sp = undefined;\n } else {\n np = gZone[i + 1];\n cp.nextPointOnContour = np;\n np.prevPointOnContour = cp;\n }\n }\n\n if (state.inhibitGridFit) return;\n\n if (exports.DEBUG) {\n console.log('PROCESSING GLYPH', state.stack);\n for (let i = 0; i < pLen; i++) {\n console.log(i, gZone[i].x, gZone[i].y);\n }\n }\n\n gZone.push(\n new HPoint(0, 0),\n new HPoint(Math.round(glyph.advanceWidth * xScale), 0)\n );\n\n exec(state);\n\n // Removes the extra points.\n gZone.length -= 2;\n\n if (exports.DEBUG) {\n console.log('FINISHED GLYPH', state.stack);\n for (let i = 0; i < pLen; i++) {\n console.log(i, gZone[i].x, gZone[i].y);\n }\n }\n};\n\n/*\n* Executes the program loaded in state.\n*/\nexec = function(state) {\n let prog = state.prog;\n\n if (!prog) return;\n\n const pLen = prog.length;\n let ins;\n\n for (state.ip = 0; state.ip < pLen; state.ip++) {\n if (exports.DEBUG) state.step++;\n ins = instructionTable[prog[state.ip]];\n\n if (!ins) {\n throw new Error(\n 'unknown instruction: 0x' +\n Number(prog[state.ip]).toString(16)\n );\n }\n\n ins(state);\n\n // very extensive debugging for each step\n /*\n if (exports.DEBUG) {\n var da;\n if (state.gZone) {\n da = [];\n for (let i = 0; i < state.gZone.length; i++)\n {\n da.push(i + ' ' +\n state.gZone[i].x * 64 + ' ' +\n state.gZone[i].y * 64 + ' ' +\n (state.gZone[i].xTouched ? 'x' : '') +\n (state.gZone[i].yTouched ? 'y' : '')\n );\n }\n console.log('GZ', da);\n }\n\n if (state.tZone) {\n da = [];\n for (let i = 0; i < state.tZone.length; i++) {\n da.push(i + ' ' +\n state.tZone[i].x * 64 + ' ' +\n state.tZone[i].y * 64 + ' ' +\n (state.tZone[i].xTouched ? 'x' : '') +\n (state.tZone[i].yTouched ? 'y' : '')\n );\n }\n console.log('TZ', da);\n }\n\n if (state.stack.length > 10) {\n console.log(\n state.stack.length,\n '...', state.stack.slice(state.stack.length - 10)\n );\n } else {\n console.log(state.stack.length, state.stack);\n }\n }\n */\n }\n};\n\n/*\n* Initializes the twilight zone.\n*\n* This is only done if a SZPx instruction\n* refers to the twilight zone.\n*/\nfunction initTZone(state)\n{\n const tZone = state.tZone = new Array(state.gZone.length);\n\n // no idea if this is actually correct...\n for (let i = 0; i < tZone.length; i++)\n {\n tZone[i] = new HPoint(0, 0);\n }\n}\n\n/*\n* Skips the instruction pointer ahead over an IF/ELSE block.\n* handleElse .. if true breaks on matching ELSE\n*/\nfunction skip(state, handleElse)\n{\n const prog = state.prog;\n let ip = state.ip;\n let nesting = 1;\n let ins;\n\n do {\n ins = prog[++ip];\n if (ins === 0x58) // IF\n nesting++;\n else if (ins === 0x59) // EIF\n nesting--;\n else if (ins === 0x40) // NPUSHB\n ip += prog[ip + 1] + 1;\n else if (ins === 0x41) // NPUSHW\n ip += 2 * prog[ip + 1] + 1;\n else if (ins >= 0xB0 && ins <= 0xB7) // PUSHB\n ip += ins - 0xB0 + 1;\n else if (ins >= 0xB8 && ins <= 0xBF) // PUSHW\n ip += (ins - 0xB8 + 1) * 2;\n else if (handleElse && nesting === 1 && ins === 0x1B) // ELSE\n break;\n } while (nesting > 0);\n\n state.ip = ip;\n}\n\n/*----------------------------------------------------------*\n* And then a lot of instructions... *\n*----------------------------------------------------------*/\n\n// SVTCA[a] Set freedom and projection Vectors To Coordinate Axis\n// 0x00-0x01\nfunction SVTCA(v, state) {\n if (exports.DEBUG) console.log(state.step, 'SVTCA[' + v.axis + ']');\n\n state.fv = state.pv = state.dpv = v;\n}\n\n// SPVTCA[a] Set Projection Vector to Coordinate Axis\n// 0x02-0x03\nfunction SPVTCA(v, state) {\n if (exports.DEBUG) console.log(state.step, 'SPVTCA[' + v.axis + ']');\n\n state.pv = state.dpv = v;\n}\n\n// SFVTCA[a] Set Freedom Vector to Coordinate Axis\n// 0x04-0x05\nfunction SFVTCA(v, state) {\n if (exports.DEBUG) console.log(state.step, 'SFVTCA[' + v.axis + ']');\n\n state.fv = v;\n}\n\n// SPVTL[a] Set Projection Vector To Line\n// 0x06-0x07\nfunction SPVTL(a, state) {\n const stack = state.stack;\n const p2i = stack.pop();\n const p1i = stack.pop();\n const p2 = state.z2[p2i];\n const p1 = state.z1[p1i];\n\n if (exports.DEBUG) console.log('SPVTL[' + a + ']', p2i, p1i);\n\n let dx;\n let dy;\n\n if (!a) {\n dx = p1.x - p2.x;\n dy = p1.y - p2.y;\n } else {\n dx = p2.y - p1.y;\n dy = p1.x - p2.x;\n }\n\n state.pv = state.dpv = getUnitVector(dx, dy);\n}\n\n// SFVTL[a] Set Freedom Vector To Line\n// 0x08-0x09\nfunction SFVTL(a, state) {\n const stack = state.stack;\n const p2i = stack.pop();\n const p1i = stack.pop();\n const p2 = state.z2[p2i];\n const p1 = state.z1[p1i];\n\n if (exports.DEBUG) console.log('SFVTL[' + a + ']', p2i, p1i);\n\n let dx;\n let dy;\n\n if (!a) {\n dx = p1.x - p2.x;\n dy = p1.y - p2.y;\n } else {\n dx = p2.y - p1.y;\n dy = p1.x - p2.x;\n }\n\n state.fv = getUnitVector(dx, dy);\n}\n\n// SPVFS[] Set Projection Vector From Stack\n// 0x0A\nfunction SPVFS(state) {\n const stack = state.stack;\n const y = stack.pop();\n const x = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'SPVFS[]', y, x);\n\n state.pv = state.dpv = getUnitVector(x, y);\n}\n\n// SFVFS[] Set Freedom Vector From Stack\n// 0x0B\nfunction SFVFS(state) {\n const stack = state.stack;\n const y = stack.pop();\n const x = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'SPVFS[]', y, x);\n\n state.fv = getUnitVector(x, y);\n}\n\n// GPV[] Get Projection Vector\n// 0x0C\nfunction GPV(state) {\n const stack = state.stack;\n const pv = state.pv;\n\n if (exports.DEBUG) console.log(state.step, 'GPV[]');\n\n stack.push(pv.x * 0x4000);\n stack.push(pv.y * 0x4000);\n}\n\n// GFV[] Get Freedom Vector\n// 0x0C\nfunction GFV(state) {\n const stack = state.stack;\n const fv = state.fv;\n\n if (exports.DEBUG) console.log(state.step, 'GFV[]');\n\n stack.push(fv.x * 0x4000);\n stack.push(fv.y * 0x4000);\n}\n\n// SFVTPV[] Set Freedom Vector To Projection Vector\n// 0x0E\nfunction SFVTPV(state) {\n state.fv = state.pv;\n\n if (exports.DEBUG) console.log(state.step, 'SFVTPV[]');\n}\n\n// ISECT[] moves point p to the InterSECTion of two lines\n// 0x0F\nfunction ISECT(state)\n{\n const stack = state.stack;\n const pa0i = stack.pop();\n const pa1i = stack.pop();\n const pb0i = stack.pop();\n const pb1i = stack.pop();\n const pi = stack.pop();\n const z0 = state.z0;\n const z1 = state.z1;\n const pa0 = z0[pa0i];\n const pa1 = z0[pa1i];\n const pb0 = z1[pb0i];\n const pb1 = z1[pb1i];\n const p = state.z2[pi];\n\n if (exports.DEBUG) console.log('ISECT[], ', pa0i, pa1i, pb0i, pb1i, pi);\n\n // math from\n // en.wikipedia.org/wiki/Line%E2%80%93line_intersection#Given_two_points_on_each_line\n\n const x1 = pa0.x;\n const y1 = pa0.y;\n const x2 = pa1.x;\n const y2 = pa1.y;\n const x3 = pb0.x;\n const y3 = pb0.y;\n const x4 = pb1.x;\n const y4 = pb1.y;\n\n const div = (x1 - x2) * (y3 - y4) - (y1 - y2) * (x3 - x4);\n const f1 = x1 * y2 - y1 * x2;\n const f2 = x3 * y4 - y3 * x4;\n\n p.x = (f1 * (x3 - x4) - f2 * (x1 - x2)) / div;\n p.y = (f1 * (y3 - y4) - f2 * (y1 - y2)) / div;\n}\n\n// SRP0[] Set Reference Point 0\n// 0x10\nfunction SRP0(state) {\n state.rp0 = state.stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'SRP0[]', state.rp0);\n}\n\n// SRP1[] Set Reference Point 1\n// 0x11\nfunction SRP1(state) {\n state.rp1 = state.stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'SRP1[]', state.rp1);\n}\n\n// SRP1[] Set Reference Point 2\n// 0x12\nfunction SRP2(state) {\n state.rp2 = state.stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'SRP2[]', state.rp2);\n}\n\n// SZP0[] Set Zone Pointer 0\n// 0x13\nfunction SZP0(state) {\n const n = state.stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'SZP0[]', n);\n\n state.zp0 = n;\n\n switch (n) {\n case 0:\n if (!state.tZone) initTZone(state);\n state.z0 = state.tZone;\n break;\n case 1 :\n state.z0 = state.gZone;\n break;\n default :\n throw new Error('Invalid zone pointer');\n }\n}\n\n// SZP1[] Set Zone Pointer 1\n// 0x14\nfunction SZP1(state) {\n const n = state.stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'SZP1[]', n);\n\n state.zp1 = n;\n\n switch (n) {\n case 0:\n if (!state.tZone) initTZone(state);\n state.z1 = state.tZone;\n break;\n case 1 :\n state.z1 = state.gZone;\n break;\n default :\n throw new Error('Invalid zone pointer');\n }\n}\n\n// SZP2[] Set Zone Pointer 2\n// 0x15\nfunction SZP2(state) {\n const n = state.stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'SZP2[]', n);\n\n state.zp2 = n;\n\n switch (n) {\n case 0:\n if (!state.tZone) initTZone(state);\n state.z2 = state.tZone;\n break;\n case 1 :\n state.z2 = state.gZone;\n break;\n default :\n throw new Error('Invalid zone pointer');\n }\n}\n\n// SZPS[] Set Zone PointerS\n// 0x16\nfunction SZPS(state) {\n const n = state.stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'SZPS[]', n);\n\n state.zp0 = state.zp1 = state.zp2 = n;\n\n switch (n) {\n case 0:\n if (!state.tZone) initTZone(state);\n state.z0 = state.z1 = state.z2 = state.tZone;\n break;\n case 1 :\n state.z0 = state.z1 = state.z2 = state.gZone;\n break;\n default :\n throw new Error('Invalid zone pointer');\n }\n}\n\n// SLOOP[] Set LOOP variable\n// 0x17\nfunction SLOOP(state) {\n state.loop = state.stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'SLOOP[]', state.loop);\n}\n\n// RTG[] Round To Grid\n// 0x18\nfunction RTG(state) {\n if (exports.DEBUG) console.log(state.step, 'RTG[]');\n\n state.round = roundToGrid;\n}\n\n// RTHG[] Round To Half Grid\n// 0x19\nfunction RTHG(state) {\n if (exports.DEBUG) console.log(state.step, 'RTHG[]');\n\n state.round = roundToHalfGrid;\n}\n\n// SMD[] Set Minimum Distance\n// 0x1A\nfunction SMD(state) {\n const d = state.stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'SMD[]', d);\n\n state.minDis = d / 0x40;\n}\n\n// ELSE[] ELSE clause\n// 0x1B\nfunction ELSE(state) {\n // This instruction has been reached by executing a then branch\n // so it just skips ahead until matching EIF.\n //\n // In case the IF was negative the IF[] instruction already\n // skipped forward over the ELSE[]\n\n if (exports.DEBUG) console.log(state.step, 'ELSE[]');\n\n skip(state, false);\n}\n\n// JMPR[] JuMP Relative\n// 0x1C\nfunction JMPR(state) {\n const o = state.stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'JMPR[]', o);\n\n // A jump by 1 would do nothing.\n state.ip += o - 1;\n}\n\n// SCVTCI[] Set Control Value Table Cut-In\n// 0x1D\nfunction SCVTCI(state) {\n const n = state.stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'SCVTCI[]', n);\n\n state.cvCutIn = n / 0x40;\n}\n\n// DUP[] DUPlicate top stack element\n// 0x20\nfunction DUP(state) {\n const stack = state.stack;\n\n if (exports.DEBUG) console.log(state.step, 'DUP[]');\n\n stack.push(stack[stack.length - 1]);\n}\n\n// POP[] POP top stack element\n// 0x21\nfunction POP(state) {\n if (exports.DEBUG) console.log(state.step, 'POP[]');\n\n state.stack.pop();\n}\n\n// CLEAR[] CLEAR the stack\n// 0x22\nfunction CLEAR(state) {\n if (exports.DEBUG) console.log(state.step, 'CLEAR[]');\n\n state.stack.length = 0;\n}\n\n// SWAP[] SWAP the top two elements on the stack\n// 0x23\nfunction SWAP(state) {\n const stack = state.stack;\n\n const a = stack.pop();\n const b = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'SWAP[]');\n\n stack.push(a);\n stack.push(b);\n}\n\n// DEPTH[] DEPTH of the stack\n// 0x24\nfunction DEPTH(state) {\n const stack = state.stack;\n\n if (exports.DEBUG) console.log(state.step, 'DEPTH[]');\n\n stack.push(stack.length);\n}\n\n// LOOPCALL[] LOOPCALL function\n// 0x2A\nfunction LOOPCALL(state) {\n const stack = state.stack;\n const fn = stack.pop();\n const c = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'LOOPCALL[]', fn, c);\n\n // saves callers program\n const cip = state.ip;\n const cprog = state.prog;\n\n state.prog = state.funcs[fn];\n\n // executes the function\n for (let i = 0; i < c; i++) {\n exec(state);\n\n if (exports.DEBUG) console.log(\n ++state.step,\n i + 1 < c ? 'next loopcall' : 'done loopcall',\n i\n );\n }\n\n // restores the callers program\n state.ip = cip;\n state.prog = cprog;\n}\n\n// CALL[] CALL function\n// 0x2B\nfunction CALL(state) {\n const fn = state.stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'CALL[]', fn);\n\n // saves callers program\n const cip = state.ip;\n const cprog = state.prog;\n\n state.prog = state.funcs[fn];\n\n // executes the function\n exec(state);\n\n // restores the callers program\n state.ip = cip;\n state.prog = cprog;\n\n if (exports.DEBUG) console.log(++state.step, 'returning from', fn);\n}\n\n// CINDEX[] Copy the INDEXed element to the top of the stack\n// 0x25\nfunction CINDEX(state) {\n const stack = state.stack;\n const k = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'CINDEX[]', k);\n\n // In case of k == 1, it copies the last element after popping\n // thus stack.length - k.\n stack.push(stack[stack.length - k]);\n}\n\n// MINDEX[] Move the INDEXed element to the top of the stack\n// 0x26\nfunction MINDEX(state) {\n const stack = state.stack;\n const k = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'MINDEX[]', k);\n\n stack.push(stack.splice(stack.length - k, 1)[0]);\n}\n\n// FDEF[] Function DEFinition\n// 0x2C\nfunction FDEF(state) {\n if (state.env !== 'fpgm') throw new Error('FDEF not allowed here');\n const stack = state.stack;\n const prog = state.prog;\n let ip = state.ip;\n\n const fn = stack.pop();\n const ipBegin = ip;\n\n if (exports.DEBUG) console.log(state.step, 'FDEF[]', fn);\n\n while (prog[++ip] !== 0x2D);\n\n state.ip = ip;\n state.funcs[fn] = prog.slice(ipBegin + 1, ip);\n}\n\n// MDAP[a] Move Direct Absolute Point\n// 0x2E-0x2F\nfunction MDAP(round, state) {\n const pi = state.stack.pop();\n const p = state.z0[pi];\n const fv = state.fv;\n const pv = state.pv;\n\n if (exports.DEBUG) console.log(state.step, 'MDAP[' + round + ']', pi);\n\n let d = pv.distance(p, HPZero);\n\n if (round) d = state.round(d);\n\n fv.setRelative(p, HPZero, d, pv);\n fv.touch(p);\n\n state.rp0 = state.rp1 = pi;\n}\n\n// IUP[a] Interpolate Untouched Points through the outline\n// 0x30\nfunction IUP(v, state) {\n const z2 = state.z2;\n const pLen = z2.length - 2;\n let cp;\n let pp;\n let np;\n\n if (exports.DEBUG) console.log(state.step, 'IUP[' + v.axis + ']');\n\n for (let i = 0; i < pLen; i++) {\n cp = z2[i]; // current point\n\n // if this point has been touched go on\n if (v.touched(cp)) continue;\n\n pp = cp.prevTouched(v);\n\n // no point on the contour has been touched?\n if (pp === cp) continue;\n\n np = cp.nextTouched(v);\n\n if (pp === np) {\n // only one point on the contour has been touched\n // so simply moves the point like that\n\n v.setRelative(cp, cp, v.distance(pp, pp, false, true), v, true);\n }\n\n v.interpolate(cp, pp, np, v);\n }\n}\n\n// SHP[] SHift Point using reference point\n// 0x32-0x33\nfunction SHP(a, state) {\n const stack = state.stack;\n const rpi = a ? state.rp1 : state.rp2;\n const rp = (a ? state.z0 : state.z1)[rpi];\n const fv = state.fv;\n const pv = state.pv;\n let loop = state.loop;\n const z2 = state.z2;\n\n while (loop--)\n {\n const pi = stack.pop();\n const p = z2[pi];\n\n const d = pv.distance(rp, rp, false, true);\n fv.setRelative(p, p, d, pv);\n fv.touch(p);\n\n if (exports.DEBUG) {\n console.log(\n state.step,\n (state.loop > 1 ?\n 'loop ' + (state.loop - loop) + ': ' :\n ''\n ) +\n 'SHP[' + (a ? 'rp1' : 'rp2') + ']', pi\n );\n }\n }\n\n state.loop = 1;\n}\n\n// SHC[] SHift Contour using reference point\n// 0x36-0x37\nfunction SHC(a, state) {\n const stack = state.stack;\n const rpi = a ? state.rp1 : state.rp2;\n const rp = (a ? state.z0 : state.z1)[rpi];\n const fv = state.fv;\n const pv = state.pv;\n const ci = stack.pop();\n const sp = state.z2[state.contours[ci]];\n let p = sp;\n\n if (exports.DEBUG) console.log(state.step, 'SHC[' + a + ']', ci);\n\n const d = pv.distance(rp, rp, false, true);\n\n do {\n if (p !== rp) fv.setRelative(p, p, d, pv);\n p = p.nextPointOnContour;\n } while (p !== sp);\n}\n\n// SHZ[] SHift Zone using reference point\n// 0x36-0x37\nfunction SHZ(a, state) {\n const stack = state.stack;\n const rpi = a ? state.rp1 : state.rp2;\n const rp = (a ? state.z0 : state.z1)[rpi];\n const fv = state.fv;\n const pv = state.pv;\n\n const e = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'SHZ[' + a + ']', e);\n\n let z;\n switch (e) {\n case 0 : z = state.tZone; break;\n case 1 : z = state.gZone; break;\n default : throw new Error('Invalid zone');\n }\n\n let p;\n const d = pv.distance(rp, rp, false, true);\n const pLen = z.length - 2;\n for (let i = 0; i < pLen; i++)\n {\n p = z[i];\n fv.setRelative(p, p, d, pv);\n //if (p !== rp) fv.setRelative(p, p, d, pv);\n }\n}\n\n// SHPIX[] SHift point by a PIXel amount\n// 0x38\nfunction SHPIX(state) {\n const stack = state.stack;\n let loop = state.loop;\n const fv = state.fv;\n const d = stack.pop() / 0x40;\n const z2 = state.z2;\n\n while (loop--) {\n const pi = stack.pop();\n const p = z2[pi];\n\n if (exports.DEBUG) {\n console.log(\n state.step,\n (state.loop > 1 ? 'loop ' + (state.loop - loop) + ': ' : '') +\n 'SHPIX[]', pi, d\n );\n }\n\n fv.setRelative(p, p, d);\n fv.touch(p);\n }\n\n state.loop = 1;\n}\n\n// IP[] Interpolate Point\n// 0x39\nfunction IP(state) {\n const stack = state.stack;\n const rp1i = state.rp1;\n const rp2i = state.rp2;\n let loop = state.loop;\n const rp1 = state.z0[rp1i];\n const rp2 = state.z1[rp2i];\n const fv = state.fv;\n const pv = state.dpv;\n const z2 = state.z2;\n\n while (loop--) {\n const pi = stack.pop();\n const p = z2[pi];\n\n if (exports.DEBUG) {\n console.log(\n state.step,\n (state.loop > 1 ? 'loop ' + (state.loop - loop) + ': ' : '') +\n 'IP[]', pi, rp1i, '<->', rp2i\n );\n }\n\n fv.interpolate(p, rp1, rp2, pv);\n\n fv.touch(p);\n }\n\n state.loop = 1;\n}\n\n// MSIRP[a] Move Stack Indirect Relative Point\n// 0x3A-0x3B\nfunction MSIRP(a, state) {\n const stack = state.stack;\n const d = stack.pop() / 64;\n const pi = stack.pop();\n const p = state.z1[pi];\n const rp0 = state.z0[state.rp0];\n const fv = state.fv;\n const pv = state.pv;\n\n fv.setRelative(p, rp0, d, pv);\n fv.touch(p);\n\n if (exports.DEBUG) console.log(state.step, 'MSIRP[' + a + ']', d, pi);\n\n state.rp1 = state.rp0;\n state.rp2 = pi;\n if (a) state.rp0 = pi;\n}\n\n// ALIGNRP[] Align to reference point.\n// 0x3C\nfunction ALIGNRP(state) {\n const stack = state.stack;\n const rp0i = state.rp0;\n const rp0 = state.z0[rp0i];\n let loop = state.loop;\n const fv = state.fv;\n const pv = state.pv;\n const z1 = state.z1;\n\n while (loop--) {\n const pi = stack.pop();\n const p = z1[pi];\n\n if (exports.DEBUG) {\n console.log(\n state.step,\n (state.loop > 1 ? 'loop ' + (state.loop - loop) + ': ' : '') +\n 'ALIGNRP[]', pi\n );\n }\n\n fv.setRelative(p, rp0, 0, pv);\n fv.touch(p);\n }\n\n state.loop = 1;\n}\n\n// RTG[] Round To Double Grid\n// 0x3D\nfunction RTDG(state) {\n if (exports.DEBUG) console.log(state.step, 'RTDG[]');\n\n state.round = roundToDoubleGrid;\n}\n\n// MIAP[a] Move Indirect Absolute Point\n// 0x3E-0x3F\nfunction MIAP(round, state) {\n const stack = state.stack;\n const n = stack.pop();\n const pi = stack.pop();\n const p = state.z0[pi];\n const fv = state.fv;\n const pv = state.pv;\n let cv = state.cvt[n];\n\n if (exports.DEBUG) {\n console.log(\n state.step,\n 'MIAP[' + round + ']',\n n, '(', cv, ')', pi\n );\n }\n\n let d = pv.distance(p, HPZero);\n\n if (round) {\n if (Math.abs(d - cv) < state.cvCutIn) d = cv;\n\n d = state.round(d);\n }\n\n fv.setRelative(p, HPZero, d, pv);\n\n if (state.zp0 === 0) {\n p.xo = p.x;\n p.yo = p.y;\n }\n\n fv.touch(p);\n\n state.rp0 = state.rp1 = pi;\n}\n\n// NPUSB[] PUSH N Bytes\n// 0x40\nfunction NPUSHB(state) {\n const prog = state.prog;\n let ip = state.ip;\n const stack = state.stack;\n\n const n = prog[++ip];\n\n if (exports.DEBUG) console.log(state.step, 'NPUSHB[]', n);\n\n for (let i = 0; i < n; i++) stack.push(prog[++ip]);\n\n state.ip = ip;\n}\n\n// NPUSHW[] PUSH N Words\n// 0x41\nfunction NPUSHW(state) {\n let ip = state.ip;\n const prog = state.prog;\n const stack = state.stack;\n const n = prog[++ip];\n\n if (exports.DEBUG) console.log(state.step, 'NPUSHW[]', n);\n\n for (let i = 0; i < n; i++) {\n let w = (prog[++ip] << 8) | prog[++ip];\n if (w & 0x8000) w = -((w ^ 0xffff) + 1);\n stack.push(w);\n }\n\n state.ip = ip;\n}\n\n// WS[] Write Store\n// 0x42\nfunction WS(state) {\n const stack = state.stack;\n let store = state.store;\n\n if (!store) store = state.store = [];\n\n const v = stack.pop();\n const l = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'WS', v, l);\n\n store[l] = v;\n}\n\n// RS[] Read Store\n// 0x43\nfunction RS(state) {\n const stack = state.stack;\n const store = state.store;\n\n const l = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'RS', l);\n\n const v = (store && store[l]) || 0;\n\n stack.push(v);\n}\n\n// WCVTP[] Write Control Value Table in Pixel units\n// 0x44\nfunction WCVTP(state) {\n const stack = state.stack;\n\n const v = stack.pop();\n const l = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'WCVTP', v, l);\n\n state.cvt[l] = v / 0x40;\n}\n\n// RCVT[] Read Control Value Table entry\n// 0x45\nfunction RCVT(state) {\n const stack = state.stack;\n const cvte = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'RCVT', cvte);\n\n stack.push(state.cvt[cvte] * 0x40);\n}\n\n// GC[] Get Coordinate projected onto the projection vector\n// 0x46-0x47\nfunction GC(a, state) {\n const stack = state.stack;\n const pi = stack.pop();\n const p = state.z2[pi];\n\n if (exports.DEBUG) console.log(state.step, 'GC[' + a + ']', pi);\n\n stack.push(state.dpv.distance(p, HPZero, a, false) * 0x40);\n}\n\n// MD[a] Measure Distance\n// 0x49-0x4A\nfunction MD(a, state) {\n const stack = state.stack;\n const pi2 = stack.pop();\n const pi1 = stack.pop();\n const p2 = state.z1[pi2];\n const p1 = state.z0[pi1];\n const d = state.dpv.distance(p1, p2, a, a);\n\n if (exports.DEBUG) console.log(state.step, 'MD[' + a + ']', pi2, pi1, '->', d);\n\n state.stack.push(Math.round(d * 64));\n}\n\n// MPPEM[] Measure Pixels Per EM\n// 0x4B\nfunction MPPEM(state) {\n if (exports.DEBUG) console.log(state.step, 'MPPEM[]');\n state.stack.push(state.ppem);\n}\n\n// FLIPON[] set the auto FLIP Boolean to ON\n// 0x4D\nfunction FLIPON(state) {\n if (exports.DEBUG) console.log(state.step, 'FLIPON[]');\n state.autoFlip = true;\n}\n\n// LT[] Less Than\n// 0x50\nfunction LT(state) {\n const stack = state.stack;\n const e2 = stack.pop();\n const e1 = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'LT[]', e2, e1);\n\n stack.push(e1 < e2 ? 1 : 0);\n}\n\n// LTEQ[] Less Than or EQual\n// 0x53\nfunction LTEQ(state) {\n const stack = state.stack;\n const e2 = stack.pop();\n const e1 = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'LTEQ[]', e2, e1);\n\n stack.push(e1 <= e2 ? 1 : 0);\n}\n\n// GTEQ[] Greater Than\n// 0x52\nfunction GT(state) {\n const stack = state.stack;\n const e2 = stack.pop();\n const e1 = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'GT[]', e2, e1);\n\n stack.push(e1 > e2 ? 1 : 0);\n}\n\n// GTEQ[] Greater Than or EQual\n// 0x53\nfunction GTEQ(state) {\n const stack = state.stack;\n const e2 = stack.pop();\n const e1 = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'GTEQ[]', e2, e1);\n\n stack.push(e1 >= e2 ? 1 : 0);\n}\n\n// EQ[] EQual\n// 0x54\nfunction EQ(state) {\n const stack = state.stack;\n const e2 = stack.pop();\n const e1 = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'EQ[]', e2, e1);\n\n stack.push(e2 === e1 ? 1 : 0);\n}\n\n// NEQ[] Not EQual\n// 0x55\nfunction NEQ(state) {\n const stack = state.stack;\n const e2 = stack.pop();\n const e1 = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'NEQ[]', e2, e1);\n\n stack.push(e2 !== e1 ? 1 : 0);\n}\n\n// ODD[] ODD\n// 0x56\nfunction ODD(state) {\n const stack = state.stack;\n const n = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'ODD[]', n);\n\n stack.push(Math.trunc(n) % 2 ? 1 : 0);\n}\n\n// EVEN[] EVEN\n// 0x57\nfunction EVEN(state) {\n const stack = state.stack;\n const n = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'EVEN[]', n);\n\n stack.push(Math.trunc(n) % 2 ? 0 : 1);\n}\n\n// IF[] IF test\n// 0x58\nfunction IF(state) {\n let test = state.stack.pop();\n let ins;\n\n if (exports.DEBUG) console.log(state.step, 'IF[]', test);\n\n // if test is true it just continues\n // if not the ip is skipped until matching ELSE or EIF\n if (!test) {\n skip(state, true);\n\n if (exports.DEBUG) console.log(state.step, ins === 0x1B ? 'ELSE[]' : 'EIF[]');\n }\n}\n\n// EIF[] End IF\n// 0x59\nfunction EIF(state) {\n // this can be reached normally when\n // executing an else branch.\n // -> just ignore it\n\n if (exports.DEBUG) console.log(state.step, 'EIF[]');\n}\n\n// AND[] logical AND\n// 0x5A\nfunction AND(state) {\n const stack = state.stack;\n const e2 = stack.pop();\n const e1 = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'AND[]', e2, e1);\n\n stack.push(e2 && e1 ? 1 : 0);\n}\n\n// OR[] logical OR\n// 0x5B\nfunction OR(state) {\n const stack = state.stack;\n const e2 = stack.pop();\n const e1 = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'OR[]', e2, e1);\n\n stack.push(e2 || e1 ? 1 : 0);\n}\n\n// NOT[] logical NOT\n// 0x5C\nfunction NOT(state) {\n const stack = state.stack;\n const e = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'NOT[]', e);\n\n stack.push(e ? 0 : 1);\n}\n\n// DELTAP1[] DELTA exception P1\n// DELTAP2[] DELTA exception P2\n// DELTAP3[] DELTA exception P3\n// 0x5D, 0x71, 0x72\nfunction DELTAP123(b, state) {\n const stack = state.stack;\n const n = stack.pop();\n const fv = state.fv;\n const pv = state.pv;\n const ppem = state.ppem;\n const base = state.deltaBase + (b - 1) * 16;\n const ds = state.deltaShift;\n const z0 = state.z0;\n\n if (exports.DEBUG) console.log(state.step, 'DELTAP[' + b + ']', n, stack);\n\n for (let i = 0; i < n; i++) {\n const pi = stack.pop();\n const arg = stack.pop();\n const appem = base + ((arg & 0xF0) >> 4);\n if (appem !== ppem) continue;\n\n let mag = (arg & 0x0F) - 8;\n if (mag >= 0) mag++;\n if (exports.DEBUG) console.log(state.step, 'DELTAPFIX', pi, 'by', mag * ds);\n\n const p = z0[pi];\n fv.setRelative(p, p, mag * ds, pv);\n }\n}\n\n// SDB[] Set Delta Base in the graphics state\n// 0x5E\nfunction SDB(state) {\n const stack = state.stack;\n const n = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'SDB[]', n);\n\n state.deltaBase = n;\n}\n\n// SDS[] Set Delta Shift in the graphics state\n// 0x5F\nfunction SDS(state) {\n const stack = state.stack;\n const n = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'SDS[]', n);\n\n state.deltaShift = Math.pow(0.5, n);\n}\n\n// ADD[] ADD\n// 0x60\nfunction ADD(state) {\n const stack = state.stack;\n const n2 = stack.pop();\n const n1 = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'ADD[]', n2, n1);\n\n stack.push(n1 + n2);\n}\n\n// SUB[] SUB\n// 0x61\nfunction SUB(state) {\n const stack = state.stack;\n const n2 = stack.pop();\n const n1 = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'SUB[]', n2, n1);\n\n stack.push(n1 - n2);\n}\n\n// DIV[] DIV\n// 0x62\nfunction DIV(state) {\n const stack = state.stack;\n const n2 = stack.pop();\n const n1 = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'DIV[]', n2, n1);\n\n stack.push(n1 * 64 / n2);\n}\n\n// MUL[] MUL\n// 0x63\nfunction MUL(state) {\n const stack = state.stack;\n const n2 = stack.pop();\n const n1 = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'MUL[]', n2, n1);\n\n stack.push(n1 * n2 / 64);\n}\n\n// ABS[] ABSolute value\n// 0x64\nfunction ABS(state) {\n const stack = state.stack;\n const n = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'ABS[]', n);\n\n stack.push(Math.abs(n));\n}\n\n// NEG[] NEGate\n// 0x65\nfunction NEG(state) {\n const stack = state.stack;\n let n = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'NEG[]', n);\n\n stack.push(-n);\n}\n\n// FLOOR[] FLOOR\n// 0x66\nfunction FLOOR(state) {\n const stack = state.stack;\n const n = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'FLOOR[]', n);\n\n stack.push(Math.floor(n / 0x40) * 0x40);\n}\n\n// CEILING[] CEILING\n// 0x67\nfunction CEILING(state) {\n const stack = state.stack;\n const n = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'CEILING[]', n);\n\n stack.push(Math.ceil(n / 0x40) * 0x40);\n}\n\n// ROUND[ab] ROUND value\n// 0x68-0x6B\nfunction ROUND(dt, state) {\n const stack = state.stack;\n const n = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'ROUND[]');\n\n stack.push(state.round(n / 0x40) * 0x40);\n}\n\n// WCVTF[] Write Control Value Table in Funits\n// 0x70\nfunction WCVTF(state) {\n const stack = state.stack;\n const v = stack.pop();\n const l = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'WCVTF[]', v, l);\n\n state.cvt[l] = v * state.ppem / state.font.unitsPerEm;\n}\n\n// DELTAC1[] DELTA exception C1\n// DELTAC2[] DELTA exception C2\n// DELTAC3[] DELTA exception C3\n// 0x73, 0x74, 0x75\nfunction DELTAC123(b, state) {\n const stack = state.stack;\n const n = stack.pop();\n const ppem = state.ppem;\n const base = state.deltaBase + (b - 1) * 16;\n const ds = state.deltaShift;\n\n if (exports.DEBUG) console.log(state.step, 'DELTAC[' + b + ']', n, stack);\n\n for (let i = 0; i < n; i++) {\n const c = stack.pop();\n const arg = stack.pop();\n const appem = base + ((arg & 0xF0) >> 4);\n if (appem !== ppem) continue;\n\n let mag = (arg & 0x0F) - 8;\n if (mag >= 0) mag++;\n\n const delta = mag * ds;\n\n if (exports.DEBUG) console.log(state.step, 'DELTACFIX', c, 'by', delta);\n\n state.cvt[c] += delta;\n }\n}\n\n// SROUND[] Super ROUND\n// 0x76\nfunction SROUND(state) {\n let n = state.stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'SROUND[]', n);\n\n state.round = roundSuper;\n\n let period;\n\n switch (n & 0xC0) {\n case 0x00:\n period = 0.5;\n break;\n case 0x40:\n period = 1;\n break;\n case 0x80:\n period = 2;\n break;\n default:\n throw new Error('invalid SROUND value');\n }\n\n state.srPeriod = period;\n\n switch (n & 0x30) {\n case 0x00:\n state.srPhase = 0;\n break;\n case 0x10:\n state.srPhase = 0.25 * period;\n break;\n case 0x20:\n state.srPhase = 0.5 * period;\n break;\n case 0x30:\n state.srPhase = 0.75 * period;\n break;\n default: throw new Error('invalid SROUND value');\n }\n\n n &= 0x0F;\n\n if (n === 0) state.srThreshold = 0;\n else state.srThreshold = (n / 8 - 0.5) * period;\n}\n\n// S45ROUND[] Super ROUND 45 degrees\n// 0x77\nfunction S45ROUND(state) {\n let n = state.stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'S45ROUND[]', n);\n\n state.round = roundSuper;\n\n let period;\n\n switch (n & 0xC0) {\n case 0x00:\n period = Math.sqrt(2) / 2;\n break;\n case 0x40:\n period = Math.sqrt(2);\n break;\n case 0x80:\n period = 2 * Math.sqrt(2);\n break;\n default:\n throw new Error('invalid S45ROUND value');\n }\n\n state.srPeriod = period;\n\n switch (n & 0x30) {\n case 0x00:\n state.srPhase = 0;\n break;\n case 0x10:\n state.srPhase = 0.25 * period;\n break;\n case 0x20:\n state.srPhase = 0.5 * period;\n break;\n case 0x30:\n state.srPhase = 0.75 * period;\n break;\n default:\n throw new Error('invalid S45ROUND value');\n }\n\n n &= 0x0F;\n\n if (n === 0) state.srThreshold = 0;\n else state.srThreshold = (n / 8 - 0.5) * period;\n}\n\n// ROFF[] Round Off\n// 0x7A\nfunction ROFF(state) {\n if (exports.DEBUG) console.log(state.step, 'ROFF[]');\n\n state.round = roundOff;\n}\n\n// RUTG[] Round Up To Grid\n// 0x7C\nfunction RUTG(state) {\n if (exports.DEBUG) console.log(state.step, 'RUTG[]');\n\n state.round = roundUpToGrid;\n}\n\n// RDTG[] Round Down To Grid\n// 0x7D\nfunction RDTG(state) {\n if (exports.DEBUG) console.log(state.step, 'RDTG[]');\n\n state.round = roundDownToGrid;\n}\n\n// SCANCTRL[] SCAN conversion ConTRoL\n// 0x85\nfunction SCANCTRL(state) {\n const n = state.stack.pop();\n\n // ignored by opentype.js\n\n if (exports.DEBUG) console.log(state.step, 'SCANCTRL[]', n);\n}\n\n// SDPVTL[a] Set Dual Projection Vector To Line\n// 0x86-0x87\nfunction SDPVTL(a, state) {\n const stack = state.stack;\n const p2i = stack.pop();\n const p1i = stack.pop();\n const p2 = state.z2[p2i];\n const p1 = state.z1[p1i];\n\n if (exports.DEBUG) console.log(state.step, 'SDPVTL[' + a + ']', p2i, p1i);\n\n let dx;\n let dy;\n\n if (!a) {\n dx = p1.x - p2.x;\n dy = p1.y - p2.y;\n } else {\n dx = p2.y - p1.y;\n dy = p1.x - p2.x;\n }\n\n state.dpv = getUnitVector(dx, dy);\n}\n\n// GETINFO[] GET INFOrmation\n// 0x88\nfunction GETINFO(state) {\n const stack = state.stack;\n const sel = stack.pop();\n let r = 0;\n\n if (exports.DEBUG) console.log(state.step, 'GETINFO[]', sel);\n\n // v35 as in no subpixel hinting\n if (sel & 0x01) r = 35;\n\n // TODO rotation and stretch currently not supported\n // and thus those GETINFO are always 0.\n\n // opentype.js is always gray scaling\n if (sel & 0x20) r |= 0x1000;\n\n stack.push(r);\n}\n\n// ROLL[] ROLL the top three stack elements\n// 0x8A\nfunction ROLL(state) {\n const stack = state.stack;\n const a = stack.pop();\n const b = stack.pop();\n const c = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'ROLL[]');\n\n stack.push(b);\n stack.push(a);\n stack.push(c);\n}\n\n// MAX[] MAXimum of top two stack elements\n// 0x8B\nfunction MAX(state) {\n const stack = state.stack;\n const e2 = stack.pop();\n const e1 = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'MAX[]', e2, e1);\n\n stack.push(Math.max(e1, e2));\n}\n\n// MIN[] MINimum of top two stack elements\n// 0x8C\nfunction MIN(state) {\n const stack = state.stack;\n const e2 = stack.pop();\n const e1 = stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'MIN[]', e2, e1);\n\n stack.push(Math.min(e1, e2));\n}\n\n// SCANTYPE[] SCANTYPE\n// 0x8D\nfunction SCANTYPE(state) {\n const n = state.stack.pop();\n // ignored by opentype.js\n if (exports.DEBUG) console.log(state.step, 'SCANTYPE[]', n);\n}\n\n// INSTCTRL[] INSTCTRL\n// 0x8D\nfunction INSTCTRL(state) {\n const s = state.stack.pop();\n let v = state.stack.pop();\n\n if (exports.DEBUG) console.log(state.step, 'INSTCTRL[]', s, v);\n\n switch (s) {\n case 1 : state.inhibitGridFit = !!v; return;\n case 2 : state.ignoreCvt = !!v; return;\n default: throw new Error('invalid INSTCTRL[] selector');\n }\n}\n\n// PUSHB[abc] PUSH Bytes\n// 0xB0-0xB7\nfunction PUSHB(n, state) {\n const stack = state.stack;\n const prog = state.prog;\n let ip = state.ip;\n\n if (exports.DEBUG) console.log(state.step, 'PUSHB[' + n + ']');\n\n for (let i = 0; i < n; i++) stack.push(prog[++ip]);\n\n state.ip = ip;\n}\n\n// PUSHW[abc] PUSH Words\n// 0xB8-0xBF\nfunction PUSHW(n, state) {\n let ip = state.ip;\n const prog = state.prog;\n const stack = state.stack;\n\n if (exports.DEBUG) console.log(state.ip, 'PUSHW[' + n + ']');\n\n for (let i = 0; i < n; i++) {\n let w = (prog[++ip] << 8) | prog[++ip];\n if (w & 0x8000) w = -((w ^ 0xffff) + 1);\n stack.push(w);\n }\n\n state.ip = ip;\n}\n\n// MDRP[abcde] Move Direct Relative Point\n// 0xD0-0xEF\n// (if indirect is 0)\n//\n// and\n//\n// MIRP[abcde] Move Indirect Relative Point\n// 0xE0-0xFF\n// (if indirect is 1)\n\nfunction MDRP_MIRP(indirect, setRp0, keepD, ro, dt, state) {\n const stack = state.stack;\n const cvte = indirect && stack.pop();\n const pi = stack.pop();\n const rp0i = state.rp0;\n const rp = state.z0[rp0i];\n const p = state.z1[pi];\n\n const md = state.minDis;\n const fv = state.fv;\n const pv = state.dpv;\n let od; // original distance\n let d; // moving distance\n let sign; // sign of distance\n let cv;\n\n d = od = pv.distance(p, rp, true, true);\n sign = d >= 0 ? 1 : -1; // Math.sign would be 0 in case of 0\n\n // TODO consider autoFlip\n d = Math.abs(d);\n\n if (indirect) {\n cv = state.cvt[cvte];\n\n if (ro && Math.abs(d - cv) < state.cvCutIn) d = cv;\n }\n\n if (keepD && d < md) d = md;\n\n if (ro) d = state.round(d);\n\n fv.setRelative(p, rp, sign * d, pv);\n fv.touch(p);\n\n if (exports.DEBUG) {\n console.log(\n state.step,\n (indirect ? 'MIRP[' : 'MDRP[') +\n (setRp0 ? 'M' : 'm') +\n (keepD ? '>' : '_') +\n (ro ? 'R' : '_') +\n (dt === 0 ? 'Gr' : (dt === 1 ? 'Bl' : (dt === 2 ? 'Wh' : ''))) +\n ']',\n indirect ?\n cvte + '(' + state.cvt[cvte] + ',' + cv + ')' :\n '',\n pi,\n '(d =', od, '->', sign * d, ')'\n );\n }\n\n state.rp1 = state.rp0;\n state.rp2 = pi;\n if (setRp0) state.rp0 = pi;\n}\n\n/*\n* The instruction table.\n*/\ninstructionTable = [\n /* 0x00 */ SVTCA.bind(undefined, yUnitVector),\n /* 0x01 */ SVTCA.bind(undefined, xUnitVector),\n /* 0x02 */ SPVTCA.bind(undefined, yUnitVector),\n /* 0x03 */ SPVTCA.bind(undefined, xUnitVector),\n /* 0x04 */ SFVTCA.bind(undefined, yUnitVector),\n /* 0x05 */ SFVTCA.bind(undefined, xUnitVector),\n /* 0x06 */ SPVTL.bind(undefined, 0),\n /* 0x07 */ SPVTL.bind(undefined, 1),\n /* 0x08 */ SFVTL.bind(undefined, 0),\n /* 0x09 */ SFVTL.bind(undefined, 1),\n /* 0x0A */ SPVFS,\n /* 0x0B */ SFVFS,\n /* 0x0C */ GPV,\n /* 0x0D */ GFV,\n /* 0x0E */ SFVTPV,\n /* 0x0F */ ISECT,\n /* 0x10 */ SRP0,\n /* 0x11 */ SRP1,\n /* 0x12 */ SRP2,\n /* 0x13 */ SZP0,\n /* 0x14 */ SZP1,\n /* 0x15 */ SZP2,\n /* 0x16 */ SZPS,\n /* 0x17 */ SLOOP,\n /* 0x18 */ RTG,\n /* 0x19 */ RTHG,\n /* 0x1A */ SMD,\n /* 0x1B */ ELSE,\n /* 0x1C */ JMPR,\n /* 0x1D */ SCVTCI,\n /* 0x1E */ undefined, // TODO SSWCI\n /* 0x1F */ undefined, // TODO SSW\n /* 0x20 */ DUP,\n /* 0x21 */ POP,\n /* 0x22 */ CLEAR,\n /* 0x23 */ SWAP,\n /* 0x24 */ DEPTH,\n /* 0x25 */ CINDEX,\n /* 0x26 */ MINDEX,\n /* 0x27 */ undefined, // TODO ALIGNPTS\n /* 0x28 */ undefined,\n /* 0x29 */ undefined, // TODO UTP\n /* 0x2A */ LOOPCALL,\n /* 0x2B */ CALL,\n /* 0x2C */ FDEF,\n /* 0x2D */ undefined, // ENDF (eaten by FDEF)\n /* 0x2E */ MDAP.bind(undefined, 0),\n /* 0x2F */ MDAP.bind(undefined, 1),\n /* 0x30 */ IUP.bind(undefined, yUnitVector),\n /* 0x31 */ IUP.bind(undefined, xUnitVector),\n /* 0x32 */ SHP.bind(undefined, 0),\n /* 0x33 */ SHP.bind(undefined, 1),\n /* 0x34 */ SHC.bind(undefined, 0),\n /* 0x35 */ SHC.bind(undefined, 1),\n /* 0x36 */ SHZ.bind(undefined, 0),\n /* 0x37 */ SHZ.bind(undefined, 1),\n /* 0x38 */ SHPIX,\n /* 0x39 */ IP,\n /* 0x3A */ MSIRP.bind(undefined, 0),\n /* 0x3B */ MSIRP.bind(undefined, 1),\n /* 0x3C */ ALIGNRP,\n /* 0x3D */ RTDG,\n /* 0x3E */ MIAP.bind(undefined, 0),\n /* 0x3F */ MIAP.bind(undefined, 1),\n /* 0x40 */ NPUSHB,\n /* 0x41 */ NPUSHW,\n /* 0x42 */ WS,\n /* 0x43 */ RS,\n /* 0x44 */ WCVTP,\n /* 0x45 */ RCVT,\n /* 0x46 */ GC.bind(undefined, 0),\n /* 0x47 */ GC.bind(undefined, 1),\n /* 0x48 */ undefined, // TODO SCFS\n /* 0x49 */ MD.bind(undefined, 0),\n /* 0x4A */ MD.bind(undefined, 1),\n /* 0x4B */ MPPEM,\n /* 0x4C */ undefined, // TODO MPS\n /* 0x4D */ FLIPON,\n /* 0x4E */ undefined, // TODO FLIPOFF\n /* 0x4F */ undefined, // TODO DEBUG\n /* 0x50 */ LT,\n /* 0x51 */ LTEQ,\n /* 0x52 */ GT,\n /* 0x53 */ GTEQ,\n /* 0x54 */ EQ,\n /* 0x55 */ NEQ,\n /* 0x56 */ ODD,\n /* 0x57 */ EVEN,\n /* 0x58 */ IF,\n /* 0x59 */ EIF,\n /* 0x5A */ AND,\n /* 0x5B */ OR,\n /* 0x5C */ NOT,\n /* 0x5D */ DELTAP123.bind(undefined, 1),\n /* 0x5E */ SDB,\n /* 0x5F */ SDS,\n /* 0x60 */ ADD,\n /* 0x61 */ SUB,\n /* 0x62 */ DIV,\n /* 0x63 */ MUL,\n /* 0x64 */ ABS,\n /* 0x65 */ NEG,\n /* 0x66 */ FLOOR,\n /* 0x67 */ CEILING,\n /* 0x68 */ ROUND.bind(undefined, 0),\n /* 0x69 */ ROUND.bind(undefined, 1),\n /* 0x6A */ ROUND.bind(undefined, 2),\n /* 0x6B */ ROUND.bind(undefined, 3),\n /* 0x6C */ undefined, // TODO NROUND[ab]\n /* 0x6D */ undefined, // TODO NROUND[ab]\n /* 0x6E */ undefined, // TODO NROUND[ab]\n /* 0x6F */ undefined, // TODO NROUND[ab]\n /* 0x70 */ WCVTF,\n /* 0x71 */ DELTAP123.bind(undefined, 2),\n /* 0x72 */ DELTAP123.bind(undefined, 3),\n /* 0x73 */ DELTAC123.bind(undefined, 1),\n /* 0x74 */ DELTAC123.bind(undefined, 2),\n /* 0x75 */ DELTAC123.bind(undefined, 3),\n /* 0x76 */ SROUND,\n /* 0x77 */ S45ROUND,\n /* 0x78 */ undefined, // TODO JROT[]\n /* 0x79 */ undefined, // TODO JROF[]\n /* 0x7A */ ROFF,\n /* 0x7B */ undefined,\n /* 0x7C */ RUTG,\n /* 0x7D */ RDTG,\n /* 0x7E */ POP, // actually SANGW, supposed to do only a pop though\n /* 0x7F */ POP, // actually AA, supposed to do only a pop though\n /* 0x80 */ undefined, // TODO FLIPPT\n /* 0x81 */ undefined, // TODO FLIPRGON\n /* 0x82 */ undefined, // TODO FLIPRGOFF\n /* 0x83 */ undefined,\n /* 0x84 */ undefined,\n /* 0x85 */ SCANCTRL,\n /* 0x86 */ SDPVTL.bind(undefined, 0),\n /* 0x87 */ SDPVTL.bind(undefined, 1),\n /* 0x88 */ GETINFO,\n /* 0x89 */ undefined, // TODO IDEF\n /* 0x8A */ ROLL,\n /* 0x8B */ MAX,\n /* 0x8C */ MIN,\n /* 0x8D */ SCANTYPE,\n /* 0x8E */ INSTCTRL,\n /* 0x8F */ undefined,\n /* 0x90 */ undefined,\n /* 0x91 */ undefined,\n /* 0x92 */ undefined,\n /* 0x93 */ undefined,\n /* 0x94 */ undefined,\n /* 0x95 */ undefined,\n /* 0x96 */ undefined,\n /* 0x97 */ undefined,\n /* 0x98 */ undefined,\n /* 0x99 */ undefined,\n /* 0x9A */ undefined,\n /* 0x9B */ undefined,\n /* 0x9C */ undefined,\n /* 0x9D */ undefined,\n /* 0x9E */ undefined,\n /* 0x9F */ undefined,\n /* 0xA0 */ undefined,\n /* 0xA1 */ undefined,\n /* 0xA2 */ undefined,\n /* 0xA3 */ undefined,\n /* 0xA4 */ undefined,\n /* 0xA5 */ undefined,\n /* 0xA6 */ undefined,\n /* 0xA7 */ undefined,\n /* 0xA8 */ undefined,\n /* 0xA9 */ undefined,\n /* 0xAA */ undefined,\n /* 0xAB */ undefined,\n /* 0xAC */ undefined,\n /* 0xAD */ undefined,\n /* 0xAE */ undefined,\n /* 0xAF */ undefined,\n /* 0xB0 */ PUSHB.bind(undefined, 1),\n /* 0xB1 */ PUSHB.bind(undefined, 2),\n /* 0xB2 */ PUSHB.bind(undefined, 3),\n /* 0xB3 */ PUSHB.bind(undefined, 4),\n /* 0xB4 */ PUSHB.bind(undefined, 5),\n /* 0xB5 */ PUSHB.bind(undefined, 6),\n /* 0xB6 */ PUSHB.bind(undefined, 7),\n /* 0xB7 */ PUSHB.bind(undefined, 8),\n /* 0xB8 */ PUSHW.bind(undefined, 1),\n /* 0xB9 */ PUSHW.bind(undefined, 2),\n /* 0xBA */ PUSHW.bind(undefined, 3),\n /* 0xBB */ PUSHW.bind(undefined, 4),\n /* 0xBC */ PUSHW.bind(undefined, 5),\n /* 0xBD */ PUSHW.bind(undefined, 6),\n /* 0xBE */ PUSHW.bind(undefined, 7),\n /* 0xBF */ PUSHW.bind(undefined, 8),\n /* 0xC0 */ MDRP_MIRP.bind(undefined, 0, 0, 0, 0, 0),\n /* 0xC1 */ MDRP_MIRP.bind(undefined, 0, 0, 0, 0, 1),\n /* 0xC2 */ MDRP_MIRP.bind(undefined, 0, 0, 0, 0, 2),\n /* 0xC3 */ MDRP_MIRP.bind(undefined, 0, 0, 0, 0, 3),\n /* 0xC4 */ MDRP_MIRP.bind(undefined, 0, 0, 0, 1, 0),\n /* 0xC5 */ MDRP_MIRP.bind(undefined, 0, 0, 0, 1, 1),\n /* 0xC6 */ MDRP_MIRP.bind(undefined, 0, 0, 0, 1, 2),\n /* 0xC7 */ MDRP_MIRP.bind(undefined, 0, 0, 0, 1, 3),\n /* 0xC8 */ MDRP_MIRP.bind(undefined, 0, 0, 1, 0, 0),\n /* 0xC9 */ MDRP_MIRP.bind(undefined, 0, 0, 1, 0, 1),\n /* 0xCA */ MDRP_MIRP.bind(undefined, 0, 0, 1, 0, 2),\n /* 0xCB */ MDRP_MIRP.bind(undefined, 0, 0, 1, 0, 3),\n /* 0xCC */ MDRP_MIRP.bind(undefined, 0, 0, 1, 1, 0),\n /* 0xCD */ MDRP_MIRP.bind(undefined, 0, 0, 1, 1, 1),\n /* 0xCE */ MDRP_MIRP.bind(undefined, 0, 0, 1, 1, 2),\n /* 0xCF */ MDRP_MIRP.bind(undefined, 0, 0, 1, 1, 3),\n /* 0xD0 */ MDRP_MIRP.bind(undefined, 0, 1, 0, 0, 0),\n /* 0xD1 */ MDRP_MIRP.bind(undefined, 0, 1, 0, 0, 1),\n /* 0xD2 */ MDRP_MIRP.bind(undefined, 0, 1, 0, 0, 2),\n /* 0xD3 */ MDRP_MIRP.bind(undefined, 0, 1, 0, 0, 3),\n /* 0xD4 */ MDRP_MIRP.bind(undefined, 0, 1, 0, 1, 0),\n /* 0xD5 */ MDRP_MIRP.bind(undefined, 0, 1, 0, 1, 1),\n /* 0xD6 */ MDRP_MIRP.bind(undefined, 0, 1, 0, 1, 2),\n /* 0xD7 */ MDRP_MIRP.bind(undefined, 0, 1, 0, 1, 3),\n /* 0xD8 */ MDRP_MIRP.bind(undefined, 0, 1, 1, 0, 0),\n /* 0xD9 */ MDRP_MIRP.bind(undefined, 0, 1, 1, 0, 1),\n /* 0xDA */ MDRP_MIRP.bind(undefined, 0, 1, 1, 0, 2),\n /* 0xDB */ MDRP_MIRP.bind(undefined, 0, 1, 1, 0, 3),\n /* 0xDC */ MDRP_MIRP.bind(undefined, 0, 1, 1, 1, 0),\n /* 0xDD */ MDRP_MIRP.bind(undefined, 0, 1, 1, 1, 1),\n /* 0xDE */ MDRP_MIRP.bind(undefined, 0, 1, 1, 1, 2),\n /* 0xDF */ MDRP_MIRP.bind(undefined, 0, 1, 1, 1, 3),\n /* 0xE0 */ MDRP_MIRP.bind(undefined, 1, 0, 0, 0, 0),\n /* 0xE1 */ MDRP_MIRP.bind(undefined, 1, 0, 0, 0, 1),\n /* 0xE2 */ MDRP_MIRP.bind(undefined, 1, 0, 0, 0, 2),\n /* 0xE3 */ MDRP_MIRP.bind(undefined, 1, 0, 0, 0, 3),\n /* 0xE4 */ MDRP_MIRP.bind(undefined, 1, 0, 0, 1, 0),\n /* 0xE5 */ MDRP_MIRP.bind(undefined, 1, 0, 0, 1, 1),\n /* 0xE6 */ MDRP_MIRP.bind(undefined, 1, 0, 0, 1, 2),\n /* 0xE7 */ MDRP_MIRP.bind(undefined, 1, 0, 0, 1, 3),\n /* 0xE8 */ MDRP_MIRP.bind(undefined, 1, 0, 1, 0, 0),\n /* 0xE9 */ MDRP_MIRP.bind(undefined, 1, 0, 1, 0, 1),\n /* 0xEA */ MDRP_MIRP.bind(undefined, 1, 0, 1, 0, 2),\n /* 0xEB */ MDRP_MIRP.bind(undefined, 1, 0, 1, 0, 3),\n /* 0xEC */ MDRP_MIRP.bind(undefined, 1, 0, 1, 1, 0),\n /* 0xED */ MDRP_MIRP.bind(undefined, 1, 0, 1, 1, 1),\n /* 0xEE */ MDRP_MIRP.bind(undefined, 1, 0, 1, 1, 2),\n /* 0xEF */ MDRP_MIRP.bind(undefined, 1, 0, 1, 1, 3),\n /* 0xF0 */ MDRP_MIRP.bind(undefined, 1, 1, 0, 0, 0),\n /* 0xF1 */ MDRP_MIRP.bind(undefined, 1, 1, 0, 0, 1),\n /* 0xF2 */ MDRP_MIRP.bind(undefined, 1, 1, 0, 0, 2),\n /* 0xF3 */ MDRP_MIRP.bind(undefined, 1, 1, 0, 0, 3),\n /* 0xF4 */ MDRP_MIRP.bind(undefined, 1, 1, 0, 1, 0),\n /* 0xF5 */ MDRP_MIRP.bind(undefined, 1, 1, 0, 1, 1),\n /* 0xF6 */ MDRP_MIRP.bind(undefined, 1, 1, 0, 1, 2),\n /* 0xF7 */ MDRP_MIRP.bind(undefined, 1, 1, 0, 1, 3),\n /* 0xF8 */ MDRP_MIRP.bind(undefined, 1, 1, 1, 0, 0),\n /* 0xF9 */ MDRP_MIRP.bind(undefined, 1, 1, 1, 0, 1),\n /* 0xFA */ MDRP_MIRP.bind(undefined, 1, 1, 1, 0, 2),\n /* 0xFB */ MDRP_MIRP.bind(undefined, 1, 1, 1, 0, 3),\n /* 0xFC */ MDRP_MIRP.bind(undefined, 1, 1, 1, 1, 0),\n /* 0xFD */ MDRP_MIRP.bind(undefined, 1, 1, 1, 1, 1),\n /* 0xFE */ MDRP_MIRP.bind(undefined, 1, 1, 1, 1, 2),\n /* 0xFF */ MDRP_MIRP.bind(undefined, 1, 1, 1, 1, 3)\n];\n\nexport default Hinting;\n\n/*****************************\n Mathematical Considerations\n******************************\n\nfv ... refers to freedom vector\npv ... refers to projection vector\nrp ... refers to reference point\np ... refers to to point being operated on\nd ... refers to distance\n\nSETRELATIVE:\n============\n\ncase freedom vector == x-axis:\n------------------------------\n\n (pv)\n .-'\n rpd .-'\n .-*\n d .-'90°'\n .-' '\n .-' '\n *-' ' b\n rp '\n '\n '\n p *----------*-------------- (fv)\n pm\n\n rpdx = rpx + d * pv.x\n rpdy = rpy + d * pv.y\n\n equation of line b\n\n y - rpdy = pvns * (x- rpdx)\n\n y = p.y\n\n x = rpdx + ( p.y - rpdy ) / pvns\n\n\ncase freedom vector == y-axis:\n------------------------------\n\n * pm\n |\\\n | \\\n | \\\n | \\\n | \\\n | \\\n | \\\n | \\\n | \\\n | \\ b\n | \\\n | \\\n | \\ .-' (pv)\n | 90° \\.-'\n | .-'* rpd\n | .-'\n * *-' d\n p rp\n\n rpdx = rpx + d * pv.x\n rpdy = rpy + d * pv.y\n\n equation of line b:\n pvns ... normal slope to pv\n\n y - rpdy = pvns * (x - rpdx)\n\n x = p.x\n\n y = rpdy + pvns * (p.x - rpdx)\n\n\n\ngeneric case:\n-------------\n\n\n .'(fv)\n .'\n .* pm\n .' !\n .' .\n .' !\n .' . b\n .' !\n * .\n p !\n 90° . ... (pv)\n ...-*-'''\n ...---''' rpd\n ...---''' d\n *--'''\n rp\n\n rpdx = rpx + d * pv.x\n rpdy = rpy + d * pv.y\n\n equation of line b:\n pvns... normal slope to pv\n\n y - rpdy = pvns * (x - rpdx)\n\n equation of freedom vector line:\n fvs ... slope of freedom vector (=fy/fx)\n\n y - py = fvs * (x - px)\n\n\n on pm both equations are true for same x/y\n\n y - rpdy = pvns * (x - rpdx)\n\n y - py = fvs * (x - px)\n\n form to y and set equal:\n\n pvns * (x - rpdx) + rpdy = fvs * (x - px) + py\n\n expand:\n\n pvns * x - pvns * rpdx + rpdy = fvs * x - fvs * px + py\n\n switch:\n\n fvs * x - fvs * px + py = pvns * x - pvns * rpdx + rpdy\n\n solve for x:\n\n fvs * x - pvns * x = fvs * px - pvns * rpdx - py + rpdy\n\n\n\n fvs * px - pvns * rpdx + rpdy - py\n x = -----------------------------------\n fvs - pvns\n\n and:\n\n y = fvs * (x - px) + py\n\n\n\nINTERPOLATE:\n============\n\nExamples of point interpolation.\n\nThe weight of the movement of the reference point gets bigger\nthe further the other reference point is away, thus the safest\noption (that is avoiding 0/0 divisions) is to weight the\noriginal distance of the other point by the sum of both distances.\n\nIf the sum of both distances is 0, then move the point by the\narithmetic average of the movement of both reference points.\n\n\n\n\n (+6)\n rp1o *---->*rp1\n . . (+12)\n . . rp2o *---------->* rp2\n . . . .\n . . . .\n . 10 20 . .\n |.........|...................| .\n . . .\n . . (+8) .\n po *------>*p .\n . . .\n . 12 . 24 .\n |...........|.......................|\n 36\n\n\n-------\n\n\n\n (+10)\n rp1o *-------->*rp1\n . . (-10)\n . . rp2 *<---------* rpo2\n . . . .\n . . . .\n . 10 . 30 . .\n |.........|.............................|\n . .\n . (+5) .\n po *--->* p .\n . . .\n . . 20 .\n |....|..............|\n 5 15\n\n\n-------\n\n\n (+10)\n rp1o *-------->*rp1\n . .\n . .\n rp2o *-------->*rp2\n\n\n (+10)\n po *-------->* p\n\n-------\n\n\n (+10)\n rp1o *-------->*rp1\n . .\n . .(+30)\n rp2o *---------------------------->*rp2\n\n\n (+25)\n po *----------------------->* p\n\n\n\nvim: set ts=4 sw=4 expandtab:\n*****/\n","/**\n * Converts a string into a list of tokens.\n */\n\n/**\n * Create a new token\n * @param {string} char a single char\n */\nfunction Token(char) {\n this.char = char;\n this.state = {};\n this.activeState = null;\n}\n\n/**\n * Create a new context range\n * @param {number} startIndex range start index\n * @param {number} endOffset range end index offset\n * @param {string} contextName owner context name\n */\nfunction ContextRange(startIndex, endOffset, contextName) {\n this.contextName = contextName;\n this.startIndex = startIndex;\n this.endOffset = endOffset;\n}\n\n/**\n * Check context start and end\n * @param {string} contextName a unique context name\n * @param {function} checkStart a predicate function the indicates a context's start\n * @param {function} checkEnd a predicate function the indicates a context's end\n */\nfunction ContextChecker(contextName, checkStart, checkEnd) {\n this.contextName = contextName;\n this.openRange = null;\n this.ranges = [];\n this.checkStart = checkStart;\n this.checkEnd = checkEnd;\n}\n\n/**\n * @typedef ContextParams\n * @type Object\n * @property {array} context context items\n * @property {number} currentIndex current item index\n */\n\n/**\n * Create a context params\n * @param {array} context a list of items\n * @param {number} currentIndex current item index\n */\nfunction ContextParams(context, currentIndex) {\n this.context = context;\n this.index = currentIndex;\n this.length = context.length;\n this.current = context[currentIndex];\n this.backtrack = context.slice(0, currentIndex);\n this.lookahead = context.slice(currentIndex + 1);\n}\n\n/**\n * Create an event instance\n * @param {string} eventId event unique id\n */\nfunction Event(eventId) {\n this.eventId = eventId;\n this.subscribers = [];\n}\n\n/**\n * Initialize a core events and auto subscribe required event handlers\n * @param {any} events an object that enlists core events handlers\n */\nfunction initializeCoreEvents(events) {\n const coreEvents = [\n 'start', 'end', 'next', 'newToken', 'contextStart',\n 'contextEnd', 'insertToken', 'removeToken', 'removeRange',\n 'replaceToken', 'replaceRange', 'composeRUD', 'updateContextsRanges'\n ];\n\n coreEvents.forEach(eventId => {\n Object.defineProperty(this.events, eventId, {\n value: new Event(eventId)\n });\n });\n\n if (!!events) {\n coreEvents.forEach(eventId => {\n const event = events[eventId];\n if (typeof event === 'function') {\n this.events[eventId].subscribe(event);\n }\n });\n }\n const requiresContextUpdate = [\n 'insertToken', 'removeToken', 'removeRange',\n 'replaceToken', 'replaceRange', 'composeRUD'\n ];\n requiresContextUpdate.forEach(eventId => {\n this.events[eventId].subscribe(\n this.updateContextsRanges\n );\n });\n}\n\n/**\n * Converts a string into a list of tokens\n * @param {any} events tokenizer core events\n */\nfunction Tokenizer(events) {\n this.tokens = [];\n this.registeredContexts = {};\n this.contextCheckers = [];\n this.events = {};\n this.registeredModifiers = [];\n\n initializeCoreEvents.call(this, events);\n}\n\n/**\n * Sets the state of a token, usually called by a state modifier.\n * @param {string} key state item key\n * @param {any} value state item value\n */\nToken.prototype.setState = function(key, value) {\n this.state[key] = value;\n this.activeState = { key, value: this.state[key] };\n return this.activeState;\n};\n\nToken.prototype.getState = function (stateId) {\n return this.state[stateId] || null;\n};\n\n/**\n * Checks if an index exists in the tokens list.\n * @param {number} index token index\n */\nTokenizer.prototype.inboundIndex = function(index) {\n return index >= 0 && index < this.tokens.length;\n};\n\n/**\n * Compose and apply a list of operations (replace, update, delete)\n * @param {array} RUDs replace, update and delete operations\n * TODO: Perf. Optimization (lengthBefore === lengthAfter ? dispatch once)\n */\nTokenizer.prototype.composeRUD = function (RUDs) {\n const silent = true;\n const state = RUDs.map(RUD => (\n this[RUD[0]].apply(this, RUD.slice(1).concat(silent))\n ));\n const hasFAILObject = obj => (\n typeof obj === 'object' &&\n obj.hasOwnProperty('FAIL')\n );\n if (state.every(hasFAILObject)) {\n return {\n FAIL: `composeRUD: one or more operations hasn't completed successfully`,\n report: state.filter(hasFAILObject)\n };\n }\n this.dispatch('composeRUD', [state.filter(op => !hasFAILObject(op))]);\n};\n\n/**\n * Replace a range of tokens with a list of tokens\n * @param {number} startIndex range start index\n * @param {number} offset range offset\n * @param {token} tokens a list of tokens to replace\n * @param {boolean} silent dispatch events and update context ranges\n */\nTokenizer.prototype.replaceRange = function (startIndex, offset, tokens, silent) {\n offset = offset !== null ? offset : this.tokens.length;\n const isTokenType = tokens.every(token => token instanceof Token);\n if (!isNaN(startIndex) && this.inboundIndex(startIndex) && isTokenType) {\n const replaced = this.tokens.splice.apply(\n this.tokens, [startIndex, offset].concat(tokens)\n );\n if (!silent) this.dispatch('replaceToken', [startIndex, offset, tokens]);\n return [replaced, tokens];\n } else {\n return { FAIL: 'replaceRange: invalid tokens or startIndex.' };\n }\n};\n\n/**\n * Replace a token with another token\n * @param {number} index token index\n * @param {token} token a token to replace\n * @param {boolean} silent dispatch events and update context ranges\n */\nTokenizer.prototype.replaceToken = function (index, token, silent) {\n if (!isNaN(index) && this.inboundIndex(index) && token instanceof Token) {\n const replaced = this.tokens.splice(index, 1, token);\n if (!silent) this.dispatch('replaceToken', [index, token]);\n return [replaced[0], token];\n } else {\n return { FAIL: 'replaceToken: invalid token or index.' };\n }\n};\n\n/**\n * Removes a range of tokens\n * @param {number} startIndex range start index\n * @param {number} offset range offset\n * @param {boolean} silent dispatch events and update context ranges\n */\nTokenizer.prototype.removeRange = function(startIndex, offset, silent) {\n offset = !isNaN(offset) ? offset : this.tokens.length;\n const tokens = this.tokens.splice(startIndex, offset);\n if (!silent) this.dispatch('removeRange', [tokens, startIndex, offset]);\n return tokens;\n};\n\n/**\n * Remove a token at a certain index\n * @param {number} index token index\n * @param {boolean} silent dispatch events and update context ranges\n */\nTokenizer.prototype.removeToken = function(index, silent) {\n if (!isNaN(index) && this.inboundIndex(index)) {\n const token = this.tokens.splice(index, 1);\n if (!silent) this.dispatch('removeToken', [token, index]);\n return token;\n } else {\n return { FAIL: 'removeToken: invalid token index.' };\n }\n};\n\n/**\n * Insert a list of tokens at a certain index\n * @param {array} tokens a list of tokens to insert\n * @param {number} index insert the list of tokens at index\n * @param {boolean} silent dispatch events and update context ranges\n */\nTokenizer.prototype.insertToken = function (tokens, index, silent) {\n const tokenType = tokens.every(\n token => token instanceof Token\n );\n if (tokenType) {\n this.tokens.splice.apply(\n this.tokens, [index, 0].concat(tokens)\n );\n if (!silent) this.dispatch('insertToken', [tokens, index]);\n return tokens;\n } else {\n return { FAIL: 'insertToken: invalid token(s).' };\n }\n};\n\n/**\n * A state modifier that is called on 'newToken' event\n * @param {string} modifierId state modifier id\n * @param {function} condition a predicate function that returns true or false\n * @param {function} modifier a function to update token state\n */\nTokenizer.prototype.registerModifier = function(modifierId, condition, modifier) {\n this.events.newToken.subscribe(function(token, contextParams) {\n const conditionParams = [token, contextParams];\n const canApplyModifier = (\n condition === null ||\n condition.apply(this, conditionParams) === true\n );\n const modifierParams = [token, contextParams];\n if (canApplyModifier) {\n let newStateValue = modifier.apply(this, modifierParams);\n token.setState(modifierId, newStateValue);\n }\n });\n this.registeredModifiers.push(modifierId);\n};\n\n/**\n * Subscribe a handler to an event\n * @param {function} eventHandler an event handler function\n */\nEvent.prototype.subscribe = function (eventHandler) {\n if (typeof eventHandler === 'function') {\n return ((this.subscribers.push(eventHandler)) - 1);\n } else {\n return { FAIL: `invalid '${this.eventId}' event handler`};\n }\n};\n\n/**\n * Unsubscribe an event handler\n * @param {string} subsId subscription id\n */\nEvent.prototype.unsubscribe = function (subsId) {\n this.subscribers.splice(subsId, 1);\n};\n\n/**\n * Sets context params current value index\n * @param {number} index context params current value index\n */\nContextParams.prototype.setCurrentIndex = function(index) {\n this.index = index;\n this.current = this.context[index];\n this.backtrack = this.context.slice(0, index);\n this.lookahead = this.context.slice(index + 1);\n};\n\n/**\n * Get an item at an offset from the current value\n * example (current value is 3):\n * 1 2 [3] 4 5 | items values\n * -2 -1 0 1 2 | offset values\n * @param {number} offset an offset from current value index\n */\nContextParams.prototype.get = function (offset) {\n switch (true) {\n case (offset === 0):\n return this.current;\n case (offset < 0 && Math.abs(offset) <= this.backtrack.length):\n return this.backtrack.slice(offset)[0];\n case (offset > 0 && offset <= this.lookahead.length):\n return this.lookahead[offset - 1];\n default:\n return null;\n }\n};\n\n/**\n * Converts a context range into a string value\n * @param {contextRange} range a context range\n */\nTokenizer.prototype.rangeToText = function (range) {\n if (range instanceof ContextRange) {\n return (\n this.getRangeTokens(range)\n .map(token => token.char).join('')\n );\n }\n};\n\n/**\n * Converts all tokens into a string\n */\nTokenizer.prototype.getText = function () {\n return this.tokens.map(token => token.char).join('');\n};\n\n/**\n * Get a context by name\n * @param {string} contextName context name to get\n */\nTokenizer.prototype.getContext = function (contextName) {\n let context = this.registeredContexts[contextName];\n return !!context ? context : null;\n};\n\n/**\n * Subscribes a new event handler to an event\n * @param {string} eventName event name to subscribe to\n * @param {function} eventHandler a function to be invoked on event\n */\nTokenizer.prototype.on = function(eventName, eventHandler) {\n const event = this.events[eventName];\n if (!!event) {\n return event.subscribe(eventHandler);\n } else {\n return null;\n }\n};\n\n/**\n * Dispatches an event\n * @param {string} eventName event name\n * @param {any} args event handler arguments\n */\nTokenizer.prototype.dispatch = function(eventName, args) {\n const event = this.events[eventName];\n if (event instanceof Event) {\n event.subscribers.forEach(subscriber => {\n subscriber.apply(this, args || []);\n });\n }\n};\n\n/**\n * Register a new context checker\n * @param {string} contextName a unique context name\n * @param {function} contextStartCheck a predicate function that returns true on context start\n * @param {function} contextEndCheck a predicate function that returns true on context end\n * TODO: call tokenize on registration to update context ranges with the new context.\n */\nTokenizer.prototype.registerContextChecker = function(contextName, contextStartCheck, contextEndCheck) {\n if (!!this.getContext(contextName)) return {\n FAIL:\n `context name '${contextName}' is already registered.`\n };\n if (typeof contextStartCheck !== 'function') return {\n FAIL:\n `missing context start check.`\n };\n if (typeof contextEndCheck !== 'function') return {\n FAIL:\n `missing context end check.`\n };\n const contextCheckers = new ContextChecker(\n contextName, contextStartCheck, contextEndCheck\n );\n this.registeredContexts[contextName] = contextCheckers;\n this.contextCheckers.push(contextCheckers);\n return contextCheckers;\n};\n\n/**\n * Gets a context range tokens\n * @param {contextRange} range a context range\n */\nTokenizer.prototype.getRangeTokens = function(range) {\n const endIndex = range.startIndex + range.endOffset;\n return [].concat(\n this.tokens\n .slice(range.startIndex, endIndex)\n );\n};\n\n/**\n * Gets the ranges of a context\n * @param {string} contextName context name\n */\nTokenizer.prototype.getContextRanges = function(contextName) {\n const context = this.getContext(contextName);\n if (!!context) {\n return context.ranges;\n } else {\n return { FAIL: `context checker '${contextName}' is not registered.` };\n }\n};\n\n/**\n * Resets context ranges to run context update\n */\nTokenizer.prototype.resetContextsRanges = function () {\n const registeredContexts = this.registeredContexts;\n for (const contextName in registeredContexts) {\n if (registeredContexts.hasOwnProperty(contextName)) {\n const context = registeredContexts[contextName];\n context.ranges = [];\n }\n }\n};\n\n/**\n * Updates context ranges\n */\nTokenizer.prototype.updateContextsRanges = function () {\n this.resetContextsRanges();\n const chars = this.tokens.map(token => token.char);\n for (let i = 0; i < chars.length; i++) {\n const contextParams = new ContextParams(chars, i);\n this.runContextCheck(contextParams);\n }\n this.dispatch('updateContextsRanges', [this.registeredContexts]);\n};\n\n/**\n * Sets the end offset of an open range\n * @param {number} offset range end offset\n * @param {string} contextName context name\n */\nTokenizer.prototype.setEndOffset = function (offset, contextName) {\n const startIndex = this.getContext(contextName).openRange.startIndex;\n let range = new ContextRange(startIndex, offset, contextName);\n const ranges = this.getContext(contextName).ranges;\n range.rangeId = `${contextName}.${ranges.length}`;\n ranges.push(range);\n this.getContext(contextName).openRange = null;\n return range;\n};\n\n/**\n * Runs a context check on the current context\n * @param {contextParams} contextParams current context params\n */\nTokenizer.prototype.runContextCheck = function(contextParams) {\n const index = contextParams.index;\n this.contextCheckers.forEach(contextChecker => {\n let contextName = contextChecker.contextName;\n let openRange = this.getContext(contextName).openRange;\n if (!openRange && contextChecker.checkStart(contextParams)) {\n openRange = new ContextRange(index, null, contextName);\n this.getContext(contextName).openRange = openRange;\n this.dispatch('contextStart', [contextName, index]);\n }\n if (!!openRange && contextChecker.checkEnd(contextParams)) {\n const offset = (index - openRange.startIndex) + 1;\n const range = this.setEndOffset(offset, contextName);\n this.dispatch('contextEnd', [contextName, range]);\n }\n });\n};\n\n/**\n * Converts a text into a list of tokens\n * @param {string} text a text to tokenize\n */\nTokenizer.prototype.tokenize = function (text) {\n this.tokens = [];\n this.resetContextsRanges();\n let chars = Array.from(text);\n this.dispatch('start');\n for (let i = 0; i < chars.length; i++) {\n const char = chars[i];\n const contextParams = new ContextParams(chars, i);\n this.dispatch('next', [contextParams]);\n this.runContextCheck(contextParams);\n let token = new Token(char);\n this.tokens.push(token);\n this.dispatch('newToken', [token, contextParams]);\n }\n this.dispatch('end', [this.tokens]);\n return this.tokens;\n};\n\nexport default Tokenizer;\nexport { Token, Event, ContextRange, ContextParams };\n","// ╭─┄┄┄────────────────────────┄─────────────────────────────────────────────╮\n// ┊ Character Class Assertions ┊ Checks if a char belongs to a certain class ┊\n// ╰─╾──────────────────────────┄─────────────────────────────────────────────╯\n// jscs:disable maximumLineLength\n/**\n * Check if a char is Arabic\n * @param {string} c a single char\n */\nexport function isArabicChar(c) {\n return /[\\u0600-\\u065F\\u066A-\\u06D2\\u06FA-\\u06FF]/.test(c);\n}\n\n/**\n * Check if a char is an isolated arabic char\n * @param {string} c a single char\n */\nexport function isIsolatedArabicChar(char) {\n return /[\\u0630\\u0690\\u0621\\u0631\\u0661\\u0671\\u0622\\u0632\\u0672\\u0692\\u06C2\\u0623\\u0673\\u0693\\u06C3\\u0624\\u0694\\u06C4\\u0625\\u0675\\u0695\\u06C5\\u06E5\\u0676\\u0696\\u06C6\\u0627\\u0677\\u0697\\u06C7\\u0648\\u0688\\u0698\\u06C8\\u0689\\u0699\\u06C9\\u068A\\u06CA\\u066B\\u068B\\u06CB\\u068C\\u068D\\u06CD\\u06FD\\u068E\\u06EE\\u06FE\\u062F\\u068F\\u06CF\\u06EF]/.test(char);\n}\n\n/**\n * Check if a char is an Arabic Tashkeel char\n * @param {string} c a single char\n */\nexport function isTashkeelArabicChar(char) {\n return /[\\u0600-\\u0605\\u060C-\\u060E\\u0610-\\u061B\\u061E\\u064B-\\u065F\\u0670\\u06D6-\\u06DC\\u06DF-\\u06E4\\u06E7\\u06E8\\u06EA-\\u06ED]/.test(char);\n}\n\n/**\n * Check if a char is Latin\n * @param {string} c a single char\n */\nexport function isLatinChar(c) {\n return /[A-z]/.test(c);\n}\n\n/**\n * Check if a char is whitespace char\n * @param {string} c a single char\n */\nexport function isWhiteSpace(c) {\n return /\\s/.test(c);\n}\n","/**\n * Query a feature by some of it's properties to lookup a glyph substitution.\n */\n\nimport { ContextParams } from '../tokenizer';\nimport { isTashkeelArabicChar } from '../char';\n\n/**\n * Create feature query instance\n * @param {Font} font opentype font instance\n */\nfunction FeatureQuery(font) {\n this.font = font;\n this.features = {};\n}\n\n/**\n * @typedef SubstitutionAction\n * @type Object\n * @property {number} id substitution type\n * @property {string} tag feature tag\n * @property {any} substitution substitution value(s)\n */\n\n/**\n * Create a substitution action instance\n * @param {SubstitutionAction} action\n */\nfunction SubstitutionAction(action) {\n this.id = action.id;\n this.tag = action.tag;\n this.substitution = action.substitution;\n}\n\n/**\n * Lookup a coverage table\n * @param {number} glyphIndex glyph index\n * @param {CoverageTable} coverage coverage table\n */\nfunction lookupCoverage(glyphIndex, coverage) {\n if (!glyphIndex) return -1;\n switch (coverage.format) {\n case 1:\n return coverage.glyphs.indexOf(glyphIndex);\n\n case 2:\n let ranges = coverage.ranges;\n for (let i = 0; i < ranges.length; i++) {\n const range = ranges[i];\n if (glyphIndex >= range.start && glyphIndex <= range.end) {\n let offset = glyphIndex - range.start;\n return range.index + offset;\n }\n }\n break;\n default:\n return -1; // not found\n }\n return -1;\n}\n\n/**\n * Handle a single substitution - format 1\n * @param {ContextParams} contextParams context params to lookup\n */\nfunction singleSubstitutionFormat1(glyphIndex, subtable) {\n let substituteIndex = lookupCoverage(glyphIndex, subtable.coverage);\n if (substituteIndex === -1) return null;\n return glyphIndex + subtable.deltaGlyphId;\n}\n\n/**\n * Handle a single substitution - format 2\n * @param {ContextParams} contextParams context params to lookup\n */\nfunction singleSubstitutionFormat2(glyphIndex, subtable) {\n let substituteIndex = lookupCoverage(glyphIndex, subtable.coverage);\n if (substituteIndex === -1) return null;\n return subtable.substitute[substituteIndex];\n}\n\n/**\n * Lookup a list of coverage tables\n * @param {any} coverageList a list of coverage tables\n * @param {ContextParams} contextParams context params to lookup\n */\nfunction lookupCoverageList(coverageList, contextParams) {\n let lookupList = [];\n for (let i = 0; i < coverageList.length; i++) {\n const coverage = coverageList[i];\n let glyphIndex = contextParams.current;\n glyphIndex = Array.isArray(glyphIndex) ? glyphIndex[0] : glyphIndex;\n const lookupIndex = lookupCoverage(glyphIndex, coverage);\n if (lookupIndex !== -1) {\n lookupList.push(lookupIndex);\n }\n }\n if (lookupList.length !== coverageList.length) return -1;\n return lookupList;\n}\n\n/**\n * Handle chaining context substitution - format 3\n * @param {ContextParams} contextParams context params to lookup\n */\nfunction chainingSubstitutionFormat3(contextParams, subtable) {\n const lookupsCount = (\n subtable.inputCoverage.length +\n subtable.lookaheadCoverage.length +\n subtable.backtrackCoverage.length\n );\n if (contextParams.context.length < lookupsCount) return [];\n // INPUT LOOKUP //\n let inputLookups = lookupCoverageList(\n subtable.inputCoverage, contextParams\n );\n if (inputLookups === -1) return [];\n // LOOKAHEAD LOOKUP //\n const lookaheadOffset = subtable.inputCoverage.length - 1;\n if (contextParams.lookahead.length < subtable.lookaheadCoverage.length) return [];\n let lookaheadContext = contextParams.lookahead.slice(lookaheadOffset);\n while (lookaheadContext.length && isTashkeelArabicChar(lookaheadContext[0].char)) {\n lookaheadContext.shift();\n }\n const lookaheadParams = new ContextParams(lookaheadContext, 0);\n let lookaheadLookups = lookupCoverageList(\n subtable.lookaheadCoverage, lookaheadParams\n );\n // BACKTRACK LOOKUP //\n let backtrackContext = [].concat(contextParams.backtrack);\n backtrackContext.reverse();\n while (backtrackContext.length && isTashkeelArabicChar(backtrackContext[0].char)) {\n backtrackContext.shift();\n }\n if (backtrackContext.length < subtable.backtrackCoverage.length) return [];\n const backtrackParams = new ContextParams(backtrackContext, 0);\n let backtrackLookups = lookupCoverageList(\n subtable.backtrackCoverage, backtrackParams\n );\n const contextRulesMatch = (\n inputLookups.length === subtable.inputCoverage.length &&\n lookaheadLookups.length === subtable.lookaheadCoverage.length &&\n backtrackLookups.length === subtable.backtrackCoverage.length\n );\n let substitutions = [];\n if (contextRulesMatch) {\n for (let i = 0; i < subtable.lookupRecords.length; i++) {\n const lookupRecord = subtable.lookupRecords[i];\n const lookupListIndex = lookupRecord.lookupListIndex;\n const lookupTable = this.getLookupByIndex(lookupListIndex);\n for (let s = 0; s < lookupTable.subtables.length; s++) {\n const subtable = lookupTable.subtables[s];\n const lookup = this.getLookupMethod(lookupTable, subtable);\n const substitutionType = this.getSubstitutionType(lookupTable, subtable);\n if (substitutionType === '12') {\n for (let n = 0; n < inputLookups.length; n++) {\n const glyphIndex = contextParams.get(n);\n const substitution = lookup(glyphIndex);\n if (substitution) substitutions.push(substitution);\n }\n }\n }\n }\n }\n return substitutions;\n}\n\n/**\n * Handle ligature substitution - format 1\n * @param {ContextParams} contextParams context params to lookup\n */\nfunction ligatureSubstitutionFormat1(contextParams, subtable) {\n // COVERAGE LOOKUP //\n let glyphIndex = contextParams.current;\n let ligSetIndex = lookupCoverage(glyphIndex, subtable.coverage);\n if (ligSetIndex === -1) return null;\n // COMPONENTS LOOKUP\n // (!) note, components are ordered in the written direction.\n let ligature;\n let ligatureSet = subtable.ligatureSets[ligSetIndex];\n for (let s = 0; s < ligatureSet.length; s++) {\n ligature = ligatureSet[s];\n for (let l = 0; l < ligature.components.length; l++) {\n const lookaheadItem = contextParams.lookahead[l];\n const component = ligature.components[l];\n if (lookaheadItem !== component) break;\n if (l === ligature.components.length - 1) return ligature;\n }\n }\n return null;\n}\n\n/**\n * Handle decomposition substitution - format 1\n * @param {number} glyphIndex glyph index\n * @param {any} subtable subtable\n */\nfunction decompositionSubstitutionFormat1(glyphIndex, subtable) {\n let substituteIndex = lookupCoverage(glyphIndex, subtable.coverage);\n if (substituteIndex === -1) return null;\n return subtable.sequences[substituteIndex];\n}\n\n/**\n * Get default script features indexes\n */\nFeatureQuery.prototype.getDefaultScriptFeaturesIndexes = function () {\n const scripts = this.font.tables.gsub.scripts;\n for (let s = 0; s < scripts.length; s++) {\n const script = scripts[s];\n if (script.tag === 'DFLT') return (\n script.script.defaultLangSys.featureIndexes\n );\n }\n return [];\n};\n\n/**\n * Get feature indexes of a specific script\n * @param {string} scriptTag script tag\n */\nFeatureQuery.prototype.getScriptFeaturesIndexes = function(scriptTag) {\n const tables = this.font.tables;\n if (!tables.gsub) return [];\n if (!scriptTag) return this.getDefaultScriptFeaturesIndexes();\n const scripts = this.font.tables.gsub.scripts;\n for (let i = 0; i < scripts.length; i++) {\n const script = scripts[i];\n if (script.tag === scriptTag && script.script.defaultLangSys) {\n return script.script.defaultLangSys.featureIndexes;\n } else {\n let langSysRecords = script.langSysRecords;\n if (!!langSysRecords) {\n for (let j = 0; j < langSysRecords.length; j++) {\n const langSysRecord = langSysRecords[j];\n if (langSysRecord.tag === scriptTag) {\n let langSys = langSysRecord.langSys;\n return langSys.featureIndexes;\n }\n }\n }\n }\n }\n return this.getDefaultScriptFeaturesIndexes();\n};\n\n/**\n * Map a feature tag to a gsub feature\n * @param {any} features gsub features\n * @param {string} scriptTag script tag\n */\nFeatureQuery.prototype.mapTagsToFeatures = function (features, scriptTag) {\n let tags = {};\n for (let i = 0; i < features.length; i++) {\n const tag = features[i].tag;\n const feature = features[i].feature;\n tags[tag] = feature;\n }\n this.features[scriptTag].tags = tags;\n};\n\n/**\n * Get features of a specific script\n * @param {string} scriptTag script tag\n */\nFeatureQuery.prototype.getScriptFeatures = function (scriptTag) {\n let features = this.features[scriptTag];\n if (this.features.hasOwnProperty(scriptTag)) return features;\n const featuresIndexes = this.getScriptFeaturesIndexes(scriptTag);\n if (!featuresIndexes) return null;\n const gsub = this.font.tables.gsub;\n features = featuresIndexes.map(index => gsub.features[index]);\n this.features[scriptTag] = features;\n this.mapTagsToFeatures(features, scriptTag);\n return features;\n};\n\n/**\n * Get substitution type\n * @param {any} lookupTable lookup table\n * @param {any} subtable subtable\n */\nFeatureQuery.prototype.getSubstitutionType = function(lookupTable, subtable) {\n const lookupType = lookupTable.lookupType.toString();\n const substFormat = subtable.substFormat.toString();\n return lookupType + substFormat;\n};\n\n/**\n * Get lookup method\n * @param {any} lookupTable lookup table\n * @param {any} subtable subtable\n */\nFeatureQuery.prototype.getLookupMethod = function(lookupTable, subtable) {\n let substitutionType = this.getSubstitutionType(lookupTable, subtable);\n switch (substitutionType) {\n case '11':\n return glyphIndex => singleSubstitutionFormat1.apply(\n this, [glyphIndex, subtable]\n );\n case '12':\n return glyphIndex => singleSubstitutionFormat2.apply(\n this, [glyphIndex, subtable]\n );\n case '63':\n return contextParams => chainingSubstitutionFormat3.apply(\n this, [contextParams, subtable]\n );\n case '41':\n return contextParams => ligatureSubstitutionFormat1.apply(\n this, [contextParams, subtable]\n );\n case '21':\n return glyphIndex => decompositionSubstitutionFormat1.apply(\n this, [glyphIndex, subtable]\n );\n default:\n throw new Error(\n `lookupType: ${lookupTable.lookupType} - ` +\n `substFormat: ${subtable.substFormat} ` +\n `is not yet supported`\n );\n }\n};\n\n/**\n * [ LOOKUP TYPES ]\n * -------------------------------\n * Single 1;\n * Multiple 2;\n * Alternate 3;\n * Ligature 4;\n * Context 5;\n * ChainingContext 6;\n * ExtensionSubstitution 7;\n * ReverseChainingContext 8;\n * -------------------------------\n *\n */\n\n/**\n * @typedef FQuery\n * @type Object\n * @param {string} tag feature tag\n * @param {string} script feature script\n * @param {ContextParams} contextParams context params\n */\n\n/**\n * Lookup a feature using a query parameters\n * @param {FQuery} query feature query\n */\nFeatureQuery.prototype.lookupFeature = function (query) {\n let contextParams = query.contextParams;\n let currentIndex = contextParams.index;\n const feature = this.getFeature({\n tag: query.tag, script: query.script\n });\n if (!feature) return new Error(\n `font '${this.font.names.fullName.en}' ` +\n `doesn't support feature '${query.tag}' ` +\n `for script '${query.script}'.`\n );\n const lookups = this.getFeatureLookups(feature);\n const substitutions = [].concat(contextParams.context);\n for (let l = 0; l < lookups.length; l++) {\n const lookupTable = lookups[l];\n const subtables = this.getLookupSubtables(lookupTable);\n for (let s = 0; s < subtables.length; s++) {\n const subtable = subtables[s];\n const substType = this.getSubstitutionType(lookupTable, subtable);\n const lookup = this.getLookupMethod(lookupTable, subtable);\n let substitution;\n switch (substType) {\n case '11':\n substitution = lookup(contextParams.current);\n if (substitution) {\n substitutions.splice(currentIndex, 1, new SubstitutionAction({\n id: 11, tag: query.tag, substitution\n }));\n }\n break;\n case '12':\n substitution = lookup(contextParams.current);\n if (substitution) {\n substitutions.splice(currentIndex, 1, new SubstitutionAction({\n id: 12, tag: query.tag, substitution\n }));\n }\n break;\n case '63':\n substitution = lookup(contextParams);\n if (Array.isArray(substitution) && substitution.length) {\n substitutions.splice(currentIndex, 1, new SubstitutionAction({\n id: 63, tag: query.tag, substitution\n }));\n }\n break;\n case '41':\n substitution = lookup(contextParams);\n if (substitution) {\n substitutions.splice(currentIndex, 1, new SubstitutionAction({\n id: 41, tag: query.tag, substitution\n }));\n }\n break;\n case '21':\n substitution = lookup(contextParams.current);\n if (substitution) {\n substitutions.splice(currentIndex, 1, new SubstitutionAction({\n id: 21, tag: query.tag, substitution\n }));\n }\n break;\n }\n contextParams = new ContextParams(substitutions, currentIndex);\n if (Array.isArray(substitution) && !substitution.length) continue;\n substitution = null;\n }\n }\n return substitutions.length ? substitutions : null;\n};\n\n/**\n * Checks if a font supports a specific features\n * @param {FQuery} query feature query object\n */\nFeatureQuery.prototype.supports = function (query) {\n if (!query.script) return false;\n this.getScriptFeatures(query.script);\n const supportedScript = this.features.hasOwnProperty(query.script);\n if (!query.tag) return supportedScript;\n const supportedFeature = (\n this.features[query.script].some(feature => feature.tag === query.tag)\n );\n return supportedScript && supportedFeature;\n};\n\n/**\n * Get lookup table subtables\n * @param {any} lookupTable lookup table\n */\nFeatureQuery.prototype.getLookupSubtables = function (lookupTable) {\n return lookupTable.subtables || null;\n};\n\n/**\n * Get lookup table by index\n * @param {number} index lookup table index\n */\nFeatureQuery.prototype.getLookupByIndex = function (index) {\n const lookups = this.font.tables.gsub.lookups;\n return lookups[index] || null;\n};\n\n/**\n * Get lookup tables for a feature\n * @param {string} feature\n */\nFeatureQuery.prototype.getFeatureLookups = function (feature) {\n // TODO: memoize\n return feature.lookupListIndexes.map(this.getLookupByIndex.bind(this));\n};\n\n/**\n * Query a feature by it's properties\n * @param {any} query an object that describes the properties of a query\n */\nFeatureQuery.prototype.getFeature = function getFeature(query) {\n if (!this.font) return { FAIL: `No font was found`};\n if (!this.features.hasOwnProperty(query.script)) {\n this.getScriptFeatures(query.script);\n }\n const scriptFeatures = this.features[query.script];\n if (!scriptFeatures) return (\n { FAIL: `No feature for script ${query.script}`}\n );\n if (!scriptFeatures.tags[query.tag]) return null;\n return this.features[query.script].tags[query.tag];\n};\n\nexport default FeatureQuery;\nexport { FeatureQuery, SubstitutionAction };\n","/**\n * Arabic word context checkers\n */\n\nimport { isArabicChar } from '../../../char';\n\nfunction arabicWordStartCheck(contextParams) {\n const char = contextParams.current;\n const prevChar = contextParams.get(-1);\n return (\n // ? arabic first char\n (prevChar === null && isArabicChar(char)) ||\n // ? arabic char preceded with a non arabic char\n (!isArabicChar(prevChar) && isArabicChar(char))\n );\n}\n\nfunction arabicWordEndCheck(contextParams) {\n const nextChar = contextParams.get(1);\n return (\n // ? last arabic char\n (nextChar === null) ||\n // ? next char is not arabic\n (!isArabicChar(nextChar))\n );\n}\n\nexport default {\n startCheck: arabicWordStartCheck,\n endCheck: arabicWordEndCheck\n};\n","/**\n * Arabic sentence context checkers\n */\n\nimport { isArabicChar, isWhiteSpace, isTashkeelArabicChar } from '../../../char';\n\nfunction arabicSentenceStartCheck(contextParams) {\n const char = contextParams.current;\n const prevChar = contextParams.get(-1);\n return (\n // ? an arabic char preceded with a non arabic char\n (isArabicChar(char) || isTashkeelArabicChar(char)) &&\n !isArabicChar(prevChar)\n );\n}\n\nfunction arabicSentenceEndCheck(contextParams) {\n const nextChar = contextParams.get(1);\n switch (true) {\n case nextChar === null:\n return true;\n case (!isArabicChar(nextChar) && !isTashkeelArabicChar(nextChar)):\n const nextIsWhitespace = isWhiteSpace(nextChar);\n if (!nextIsWhitespace) return true;\n if (nextIsWhitespace) {\n let arabicCharAhead = false;\n arabicCharAhead = (\n contextParams.lookahead.some(\n c => isArabicChar(c) || isTashkeelArabicChar(c)\n )\n );\n if (!arabicCharAhead) return true;\n }\n break;\n default:\n return false;\n }\n}\n\nexport default {\n startCheck: arabicSentenceStartCheck,\n endCheck: arabicSentenceEndCheck\n};\n","import { SubstitutionAction } from './featureQuery';\n\n/**\n * Apply single substitution format 1\n * @param {Array} substitutions substitutions\n * @param {any} tokens a list of tokens\n * @param {number} index token index\n */\nfunction singleSubstitutionFormat1(action, tokens, index) {\n tokens[index].setState(action.tag, action.substitution);\n}\n\n/**\n * Apply single substitution format 2\n * @param {Array} substitutions substitutions\n * @param {any} tokens a list of tokens\n * @param {number} index token index\n */\nfunction singleSubstitutionFormat2(action, tokens, index) {\n tokens[index].setState(action.tag, action.substitution);\n}\n\n/**\n * Apply chaining context substitution format 3\n * @param {Array} substitutions substitutions\n * @param {any} tokens a list of tokens\n * @param {number} index token index\n */\nfunction chainingSubstitutionFormat3(action, tokens, index) {\n action.substitution.forEach((subst, offset) => {\n const token = tokens[index + offset];\n token.setState(action.tag, subst);\n });\n}\n\n/**\n * Apply ligature substitution format 1\n * @param {Array} substitutions substitutions\n * @param {any} tokens a list of tokens\n * @param {number} index token index\n */\nfunction ligatureSubstitutionFormat1(action, tokens, index) {\n let token = tokens[index];\n token.setState(action.tag, action.substitution.ligGlyph);\n const compsCount = action.substitution.components.length;\n for (let i = 0; i < compsCount; i++) {\n token = tokens[index + i + 1];\n token.setState('deleted', true);\n }\n}\n\n/**\n * Supported substitutions\n */\nconst SUBSTITUTIONS = {\n 11: singleSubstitutionFormat1,\n 12: singleSubstitutionFormat2,\n 63: chainingSubstitutionFormat3,\n 41: ligatureSubstitutionFormat1\n};\n\n/**\n * Apply substitutions to a list of tokens\n * @param {Array} substitutions substitutions\n * @param {any} tokens a list of tokens\n * @param {number} index token index\n */\nfunction applySubstitution(action, tokens, index) {\n if (action instanceof SubstitutionAction && SUBSTITUTIONS[action.id]) {\n SUBSTITUTIONS[action.id](action, tokens, index);\n }\n}\n\nexport default applySubstitution;\n","/**\n * Apply Arabic presentation forms to a range of tokens\n */\n\nimport { ContextParams } from '../../tokenizer';\nimport { isIsolatedArabicChar, isTashkeelArabicChar } from '../../char';\nimport { SubstitutionAction } from '../featureQuery';\nimport applySubstitution from '../applySubstitution';\n\n/**\n * Check if a char can be connected to it's preceding char\n * @param {ContextParams} charContextParams context params of a char\n */\nfunction willConnectPrev(charContextParams) {\n let backtrack = [].concat(charContextParams.backtrack);\n for (let i = backtrack.length - 1; i >= 0; i--) {\n const prevChar = backtrack[i];\n const isolated = isIsolatedArabicChar(prevChar);\n const tashkeel = isTashkeelArabicChar(prevChar);\n if (!isolated && !tashkeel) return true;\n if (isolated) return false;\n }\n return false;\n}\n\n/**\n * Check if a char can be connected to it's proceeding char\n * @param {ContextParams} charContextParams context params of a char\n */\nfunction willConnectNext(charContextParams) {\n if (isIsolatedArabicChar(charContextParams.current)) return false;\n for (let i = 0; i < charContextParams.lookahead.length; i++) {\n const nextChar = charContextParams.lookahead[i];\n const tashkeel = isTashkeelArabicChar(nextChar);\n if (!tashkeel) return true;\n }\n return false;\n}\n\n/**\n * Apply arabic presentation forms to a list of tokens\n * @param {ContextRange} range a range of tokens\n */\nfunction arabicPresentationForms(range) {\n const script = 'arab';\n const tags = this.featuresTags[script];\n const tokens = this.tokenizer.getRangeTokens(range);\n if (tokens.length === 1) return;\n let contextParams = new ContextParams(\n tokens.map(token => token.getState('glyphIndex')\n ), 0);\n const charContextParams = new ContextParams(\n tokens.map(token => token.char\n ), 0);\n tokens.forEach((token, index) => {\n if (isTashkeelArabicChar(token.char)) return;\n contextParams.setCurrentIndex(index);\n charContextParams.setCurrentIndex(index);\n let CONNECT = 0; // 2 bits 00 (10: can connect next) (01: can connect prev)\n if (willConnectPrev(charContextParams)) CONNECT |= 1;\n if (willConnectNext(charContextParams)) CONNECT |= 2;\n let tag;\n switch (CONNECT) {\n case 1: (tag = 'fina'); break;\n case 2: (tag = 'init'); break;\n case 3: (tag = 'medi'); break;\n }\n if (tags.indexOf(tag) === -1) return;\n let substitutions = this.query.lookupFeature({\n tag, script, contextParams\n });\n if (substitutions instanceof Error) return console.info(substitutions.message);\n substitutions.forEach((action, index) => {\n if (action instanceof SubstitutionAction) {\n applySubstitution(action, tokens, index);\n contextParams.context[index] = action.substitution;\n }\n });\n });\n}\n\nexport default arabicPresentationForms;\nexport { arabicPresentationForms };\n","/**\n * Apply Arabic required ligatures feature to a range of tokens\n */\n\nimport { ContextParams } from '../../tokenizer';\nimport applySubstitution from '../applySubstitution';\n\n/**\n * Update context params\n * @param {any} tokens a list of tokens\n * @param {number} index current item index\n */\nfunction getContextParams(tokens, index) {\n const context = tokens.map(token => token.activeState.value);\n return new ContextParams(context, index || 0);\n}\n\n/**\n * Apply Arabic required ligatures to a context range\n * @param {ContextRange} range a range of tokens\n */\nfunction arabicRequiredLigatures(range) {\n const script = 'arab';\n let tokens = this.tokenizer.getRangeTokens(range);\n let contextParams = getContextParams(tokens);\n contextParams.context.forEach((glyphIndex, index) => {\n contextParams.setCurrentIndex(index);\n let substitutions = this.query.lookupFeature({\n tag: 'rlig', script, contextParams\n });\n if (substitutions.length) {\n substitutions.forEach(\n action => applySubstitution(action, tokens, index)\n );\n contextParams = getContextParams(tokens);\n }\n });\n}\n\nexport default arabicRequiredLigatures;\nexport { arabicRequiredLigatures };\n","/**\n * Latin word context checkers\n */\n\nimport { isLatinChar } from '../../../char';\n\nfunction latinWordStartCheck(contextParams) {\n const char = contextParams.current;\n const prevChar = contextParams.get(-1);\n return (\n // ? latin first char\n (prevChar === null && isLatinChar(char)) ||\n // ? latin char preceded with a non latin char\n (!isLatinChar(prevChar) && isLatinChar(char))\n );\n}\n\nfunction latinWordEndCheck(contextParams) {\n const nextChar = contextParams.get(1);\n return (\n // ? last latin char\n (nextChar === null) ||\n // ? next char is not latin\n (!isLatinChar(nextChar))\n );\n}\n\nexport default {\n startCheck: latinWordStartCheck,\n endCheck: latinWordEndCheck\n};\n","/**\n * Apply Latin ligature feature to a range of tokens\n */\n\nimport { ContextParams } from '../../tokenizer';\nimport applySubstitution from '../applySubstitution';\n\n/**\n * Update context params\n * @param {any} tokens a list of tokens\n * @param {number} index current item index\n */\nfunction getContextParams(tokens, index) {\n const context = tokens.map(token => token.activeState.value);\n return new ContextParams(context, index || 0);\n}\n\n/**\n * Apply Arabic required ligatures to a context range\n * @param {ContextRange} range a range of tokens\n */\nfunction latinLigature(range) {\n const script = 'latn';\n let tokens = this.tokenizer.getRangeTokens(range);\n let contextParams = getContextParams(tokens);\n contextParams.context.forEach((glyphIndex, index) => {\n contextParams.setCurrentIndex(index);\n let substitutions = this.query.lookupFeature({\n tag: 'liga', script, contextParams\n });\n if (substitutions.length) {\n substitutions.forEach(\n action => applySubstitution(action, tokens, index)\n );\n contextParams = getContextParams(tokens);\n }\n });\n}\n\nexport default latinLigature;\n","/**\n * Infer bidirectional properties for a given text and apply\n * the corresponding layout rules.\n */\n\nimport Tokenizer from './tokenizer';\nimport FeatureQuery from './features/featureQuery';\nimport arabicWordCheck from './features/arab/contextCheck/arabicWord';\nimport arabicSentenceCheck from './features/arab/contextCheck/arabicSentence';\nimport arabicPresentationForms from './features/arab/arabicPresentationForms';\nimport arabicRequiredLigatures from './features/arab/arabicRequiredLigatures';\nimport latinWordCheck from './features/latn/contextCheck/latinWord';\nimport latinLigature from './features/latn/latinLigatures';\n\n/**\n * Create Bidi. features\n * @param {string} baseDir text base direction. value either 'ltr' or 'rtl'\n */\nfunction Bidi(baseDir) {\n this.baseDir = baseDir || 'ltr';\n this.tokenizer = new Tokenizer();\n this.featuresTags = {};\n}\n\n/**\n * Sets Bidi text\n * @param {string} text a text input\n */\nBidi.prototype.setText = function (text) {\n this.text = text;\n};\n\n/**\n * Store essential context checks:\n * arabic word check for applying gsub features\n * arabic sentence check for adjusting arabic layout\n */\nBidi.prototype.contextChecks = ({\n latinWordCheck,\n arabicWordCheck,\n arabicSentenceCheck\n});\n\n/**\n * Register arabic word check\n */\nfunction registerContextChecker(checkId) {\n const check = this.contextChecks[`${checkId}Check`];\n return this.tokenizer.registerContextChecker(\n checkId, check.startCheck, check.endCheck\n );\n}\n\n/**\n * Perform pre tokenization procedure then\n * tokenize text input\n */\nfunction tokenizeText() {\n registerContextChecker.call(this, 'latinWord');\n registerContextChecker.call(this, 'arabicWord');\n registerContextChecker.call(this, 'arabicSentence');\n return this.tokenizer.tokenize(this.text);\n}\n\n/**\n * Reverse arabic sentence layout\n * TODO: check base dir before applying adjustments - priority low\n */\nfunction reverseArabicSentences() {\n const ranges = this.tokenizer.getContextRanges('arabicSentence');\n ranges.forEach(range => {\n let rangeTokens = this.tokenizer.getRangeTokens(range);\n this.tokenizer.replaceRange(\n range.startIndex,\n range.endOffset,\n rangeTokens.reverse()\n );\n });\n}\n\n/**\n * Register supported features tags\n * @param {script} script script tag\n * @param {Array} tags features tags list\n */\nBidi.prototype.registerFeatures = function (script, tags) {\n const supportedTags = tags.filter(\n tag => this.query.supports({script, tag})\n );\n if (!this.featuresTags.hasOwnProperty(script)) {\n this.featuresTags[script] = supportedTags;\n } else {\n this.featuresTags[script] =\n this.featuresTags[script].concat(supportedTags);\n }\n};\n\n/**\n * Apply GSUB features\n * @param {Array} tagsList a list of features tags\n * @param {string} script a script tag\n * @param {Font} font opentype font instance\n */\nBidi.prototype.applyFeatures = function (font, features) {\n if (!font) throw new Error(\n 'No valid font was provided to apply features'\n );\n if (!this.query) this.query = new FeatureQuery(font);\n for (let f = 0; f < features.length; f++) {\n const feature = features[f];\n if (!this.query.supports({script: feature.script})) continue;\n this.registerFeatures(feature.script, feature.tags);\n }\n};\n\n/**\n * Register a state modifier\n * @param {string} modifierId state modifier id\n * @param {function} condition a predicate function that returns true or false\n * @param {function} modifier a modifier function to set token state\n */\nBidi.prototype.registerModifier = function (modifierId, condition, modifier) {\n this.tokenizer.registerModifier(modifierId, condition, modifier);\n};\n\n/**\n * Check if 'glyphIndex' is registered\n */\nfunction checkGlyphIndexStatus() {\n if (this.tokenizer.registeredModifiers.indexOf('glyphIndex') === -1) {\n throw new Error(\n 'glyphIndex modifier is required to apply ' +\n 'arabic presentation features.'\n );\n }\n}\n\n/**\n * Apply arabic presentation forms features\n */\nfunction applyArabicPresentationForms() {\n const script = 'arab';\n if (!this.featuresTags.hasOwnProperty(script)) return;\n checkGlyphIndexStatus.call(this);\n const ranges = this.tokenizer.getContextRanges('arabicWord');\n ranges.forEach(range => {\n arabicPresentationForms.call(this, range);\n });\n}\n\n/**\n * Apply required arabic ligatures\n */\nfunction applyArabicRequireLigatures() {\n const script = 'arab';\n if (!this.featuresTags.hasOwnProperty(script)) return;\n const tags = this.featuresTags[script];\n if (tags.indexOf('rlig') === -1) return;\n checkGlyphIndexStatus.call(this);\n const ranges = this.tokenizer.getContextRanges('arabicWord');\n ranges.forEach(range => {\n arabicRequiredLigatures.call(this, range);\n });\n}\n\n/**\n * Apply required arabic ligatures\n */\nfunction applyLatinLigatures() {\n const script = 'latn';\n if (!this.featuresTags.hasOwnProperty(script)) return;\n const tags = this.featuresTags[script];\n if (tags.indexOf('liga') === -1) return;\n checkGlyphIndexStatus.call(this);\n const ranges = this.tokenizer.getContextRanges('latinWord');\n ranges.forEach(range => {\n latinLigature.call(this, range);\n });\n}\n\n/**\n * Check if a context is registered\n * @param {string} contextId context id\n */\nBidi.prototype.checkContextReady = function (contextId) {\n return !!this.tokenizer.getContext(contextId);\n};\n\n/**\n * Apply features to registered contexts\n */\nBidi.prototype.applyFeaturesToContexts = function () {\n if (this.checkContextReady('arabicWord')) {\n applyArabicPresentationForms.call(this);\n applyArabicRequireLigatures.call(this);\n }\n if (this.checkContextReady('latinWord')) {\n applyLatinLigatures.call(this);\n }\n if (this.checkContextReady('arabicSentence')) {\n reverseArabicSentences.call(this);\n }\n};\n\n/**\n * process text input\n * @param {string} text an input text\n */\nBidi.prototype.processText = function(text) {\n if (!this.text || this.text !== text) {\n this.setText(text);\n tokenizeText.call(this);\n this.applyFeaturesToContexts();\n }\n};\n\n/**\n * Process a string of text to identify and adjust\n * bidirectional text entities.\n * @param {string} text input text\n */\nBidi.prototype.getBidiText = function (text) {\n this.processText(text);\n return this.tokenizer.getText();\n};\n\n/**\n * Get the current state index of each token\n * @param {text} text an input text\n */\nBidi.prototype.getTextGlyphs = function (text) {\n this.processText(text);\n let indexes = [];\n for (let i = 0; i < this.tokenizer.tokens.length; i++) {\n const token = this.tokenizer.tokens[i];\n if (token.state.deleted) continue;\n const index = token.activeState.value;\n indexes.push(Array.isArray(index) ? index[0] : index);\n }\n return indexes;\n};\n\nexport default Bidi;\n","// The Font object\n\nimport Path from './path';\nimport sfnt from './tables/sfnt';\nimport { DefaultEncoding } from './encoding';\nimport glyphset from './glyphset';\nimport Position from './position';\nimport Substitution from './substitution';\nimport { isBrowser, checkArgument, arrayBufferToNodeBuffer } from './util';\nimport HintingTrueType from './hintingtt';\nimport Bidi from './bidi';\n\n/**\n * @typedef FontOptions\n * @type Object\n * @property {Boolean} empty - whether to create a new empty font\n * @property {string} familyName\n * @property {string} styleName\n * @property {string=} fullName\n * @property {string=} postScriptName\n * @property {string=} designer\n * @property {string=} designerURL\n * @property {string=} manufacturer\n * @property {string=} manufacturerURL\n * @property {string=} license\n * @property {string=} licenseURL\n * @property {string=} version\n * @property {string=} description\n * @property {string=} copyright\n * @property {string=} trademark\n * @property {Number} unitsPerEm\n * @property {Number} ascender\n * @property {Number} descender\n * @property {Number} createdTimestamp\n * @property {string=} weightClass\n * @property {string=} widthClass\n * @property {string=} fsSelection\n */\n\n/**\n * A Font represents a loaded OpenType font file.\n * It contains a set of glyphs and methods to draw text on a drawing context,\n * or to get a path representing the text.\n * @exports opentype.Font\n * @class\n * @param {FontOptions}\n * @constructor\n */\nfunction Font(options) {\n options = options || {};\n options.tables = options.tables || {};\n\n if (!options.empty) {\n // Check that we've provided the minimum set of names.\n checkArgument(options.familyName, 'When creating a new Font object, familyName is required.');\n checkArgument(options.styleName, 'When creating a new Font object, styleName is required.');\n checkArgument(options.unitsPerEm, 'When creating a new Font object, unitsPerEm is required.');\n checkArgument(options.ascender, 'When creating a new Font object, ascender is required.');\n checkArgument(options.descender <= 0, 'When creating a new Font object, negative descender value is required.');\n\n // OS X will complain if the names are empty, so we put a single space everywhere by default.\n this.names = {\n fontFamily: {en: options.familyName || ' '},\n fontSubfamily: {en: options.styleName || ' '},\n fullName: {en: options.fullName || options.familyName + ' ' + options.styleName},\n // postScriptName may not contain any whitespace\n postScriptName: {en: options.postScriptName || (options.familyName + options.styleName).replace(/\\s/g, '')},\n designer: {en: options.designer || ' '},\n designerURL: {en: options.designerURL || ' '},\n manufacturer: {en: options.manufacturer || ' '},\n manufacturerURL: {en: options.manufacturerURL || ' '},\n license: {en: options.license || ' '},\n licenseURL: {en: options.licenseURL || ' '},\n version: {en: options.version || 'Version 0.1'},\n description: {en: options.description || ' '},\n copyright: {en: options.copyright || ' '},\n trademark: {en: options.trademark || ' '}\n };\n this.unitsPerEm = options.unitsPerEm || 1000;\n this.ascender = options.ascender;\n this.descender = options.descender;\n this.createdTimestamp = options.createdTimestamp;\n this.tables = Object.assign(options.tables, {\n os2: Object.assign({\n usWeightClass: options.weightClass || this.usWeightClasses.MEDIUM,\n usWidthClass: options.widthClass || this.usWidthClasses.MEDIUM,\n fsSelection: options.fsSelection || this.fsSelectionValues.REGULAR,\n }, options.tables.os2)\n });\n }\n\n this.supported = true; // Deprecated: parseBuffer will throw an error if font is not supported.\n this.glyphs = new glyphset.GlyphSet(this, options.glyphs || []);\n this.encoding = new DefaultEncoding(this);\n this.position = new Position(this);\n this.substitution = new Substitution(this);\n this.tables = this.tables || {};\n\n // needed for low memory mode only.\n this._push = null;\n this._hmtxTableData = {};\n\n Object.defineProperty(this, 'hinting', {\n get: function() {\n if (this._hinting) return this._hinting;\n if (this.outlinesFormat === 'truetype') {\n return (this._hinting = new HintingTrueType(this));\n }\n }\n });\n}\n\n/**\n * Check if the font has a glyph for the given character.\n * @param {string}\n * @return {Boolean}\n */\nFont.prototype.hasChar = function(c) {\n return this.encoding.charToGlyphIndex(c) !== null;\n};\n\n/**\n * Convert the given character to a single glyph index.\n * Note that this function assumes that there is a one-to-one mapping between\n * the given character and a glyph; for complex scripts this might not be the case.\n * @param {string}\n * @return {Number}\n */\nFont.prototype.charToGlyphIndex = function(s) {\n return this.encoding.charToGlyphIndex(s);\n};\n\n/**\n * Convert the given character to a single Glyph object.\n * Note that this function assumes that there is a one-to-one mapping between\n * the given character and a glyph; for complex scripts this might not be the case.\n * @param {string}\n * @return {opentype.Glyph}\n */\nFont.prototype.charToGlyph = function(c) {\n const glyphIndex = this.charToGlyphIndex(c);\n let glyph = this.glyphs.get(glyphIndex);\n if (!glyph) {\n // .notdef\n glyph = this.glyphs.get(0);\n }\n\n return glyph;\n};\n\n/**\n * Update features\n * @param {any} options features options\n */\nFont.prototype.updateFeatures = function (options) {\n // TODO: update all features options not only 'latn'.\n return this.defaultRenderOptions.features.map(feature => {\n if (feature.script === 'latn') {\n return {\n script: 'latn',\n tags: feature.tags.filter(tag => options[tag])\n };\n } else {\n return feature;\n }\n });\n};\n\n/**\n * Convert the given text to a list of Glyph objects.\n * Note that there is no strict one-to-one mapping between characters and\n * glyphs, so the list of returned glyphs can be larger or smaller than the\n * length of the given string.\n * @param {string}\n * @param {GlyphRenderOptions} [options]\n * @return {opentype.Glyph[]}\n */\nFont.prototype.stringToGlyphs = function(s, options) {\n\n const bidi = new Bidi();\n\n // Create and register 'glyphIndex' state modifier\n const charToGlyphIndexMod = token => this.charToGlyphIndex(token.char);\n bidi.registerModifier('glyphIndex', null, charToGlyphIndexMod);\n\n // roll-back to default features\n let features = options ?\n this.updateFeatures(options.features) :\n this.defaultRenderOptions.features;\n\n bidi.applyFeatures(this, features);\n\n const indexes = bidi.getTextGlyphs(s);\n\n let length = indexes.length;\n\n // convert glyph indexes to glyph objects\n const glyphs = new Array(length);\n const notdef = this.glyphs.get(0);\n for (let i = 0; i < length; i += 1) {\n glyphs[i] = this.glyphs.get(indexes[i]) || notdef;\n }\n return glyphs;\n};\n\n/**\n * @param {string}\n * @return {Number}\n */\nFont.prototype.nameToGlyphIndex = function(name) {\n return this.glyphNames.nameToGlyphIndex(name);\n};\n\n/**\n * @param {string}\n * @return {opentype.Glyph}\n */\nFont.prototype.nameToGlyph = function(name) {\n const glyphIndex = this.nameToGlyphIndex(name);\n let glyph = this.glyphs.get(glyphIndex);\n if (!glyph) {\n // .notdef\n glyph = this.glyphs.get(0);\n }\n\n return glyph;\n};\n\n/**\n * @param {Number}\n * @return {String}\n */\nFont.prototype.glyphIndexToName = function(gid) {\n if (!this.glyphNames.glyphIndexToName) {\n return '';\n }\n\n return this.glyphNames.glyphIndexToName(gid);\n};\n\n/**\n * Retrieve the value of the kerning pair between the left glyph (or its index)\n * and the right glyph (or its index). If no kerning pair is found, return 0.\n * The kerning value gets added to the advance width when calculating the spacing\n * between glyphs.\n * For GPOS kerning, this method uses the default script and language, which covers\n * most use cases. To have greater control, use font.position.getKerningValue .\n * @param {opentype.Glyph} leftGlyph\n * @param {opentype.Glyph} rightGlyph\n * @return {Number}\n */\nFont.prototype.getKerningValue = function(leftGlyph, rightGlyph) {\n leftGlyph = leftGlyph.index || leftGlyph;\n rightGlyph = rightGlyph.index || rightGlyph;\n const gposKerning = this.position.defaultKerningTables;\n if (gposKerning) {\n return this.position.getKerningValue(gposKerning, leftGlyph, rightGlyph);\n }\n // \"kern\" table\n return this.kerningPairs[leftGlyph + ',' + rightGlyph] || 0;\n};\n\n/**\n * @typedef GlyphRenderOptions\n * @type Object\n * @property {string} [script] - script used to determine which features to apply. By default, 'DFLT' or 'latn' is used.\n * See https://www.microsoft.com/typography/otspec/scripttags.htm\n * @property {string} [language='dflt'] - language system used to determine which features to apply.\n * See https://www.microsoft.com/typography/developers/opentype/languagetags.aspx\n * @property {boolean} [kerning=true] - whether to include kerning values\n * @property {object} [features] - OpenType Layout feature tags. Used to enable or disable the features of the given script/language system.\n * See https://www.microsoft.com/typography/otspec/featuretags.htm\n */\nFont.prototype.defaultRenderOptions = {\n kerning: true,\n features: [\n /**\n * these 4 features are required to render Arabic text properly\n * and shouldn't be turned off when rendering arabic text.\n */\n { script: 'arab', tags: ['init', 'medi', 'fina', 'rlig'] },\n { script: 'latn', tags: ['liga', 'rlig'] }\n ]\n};\n\n/**\n * Helper function that invokes the given callback for each glyph in the given text.\n * The callback gets `(glyph, x, y, fontSize, options)`.* @param {string} text\n * @param {string} text - The text to apply.\n * @param {number} [x=0] - Horizontal position of the beginning of the text.\n * @param {number} [y=0] - Vertical position of the *baseline* of the text.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n * @param {GlyphRenderOptions=} options\n * @param {Function} callback\n */\nFont.prototype.forEachGlyph = function(text, x, y, fontSize, options, callback) {\n x = x !== undefined ? x : 0;\n y = y !== undefined ? y : 0;\n fontSize = fontSize !== undefined ? fontSize : 72;\n options = Object.assign({}, this.defaultRenderOptions, options);\n const fontScale = 1 / this.unitsPerEm * fontSize;\n const glyphs = this.stringToGlyphs(text, options);\n let kerningLookups;\n if (options.kerning) {\n const script = options.script || this.position.getDefaultScriptName();\n kerningLookups = this.position.getKerningTables(script, options.language);\n }\n for (let i = 0; i < glyphs.length; i += 1) {\n const glyph = glyphs[i];\n callback.call(this, glyph, x, y, fontSize, options);\n if (glyph.advanceWidth) {\n x += glyph.advanceWidth * fontScale;\n }\n\n if (options.kerning && i < glyphs.length - 1) {\n // We should apply position adjustment lookups in a more generic way.\n // Here we only use the xAdvance value.\n const kerningValue = kerningLookups ?\n this.position.getKerningValue(kerningLookups, glyph.index, glyphs[i + 1].index) :\n this.getKerningValue(glyph, glyphs[i + 1]);\n x += kerningValue * fontScale;\n }\n\n if (options.letterSpacing) {\n x += options.letterSpacing * fontSize;\n } else if (options.tracking) {\n x += (options.tracking / 1000) * fontSize;\n }\n }\n return x;\n};\n\n/**\n * Create a Path object that represents the given text.\n * @param {string} text - The text to create.\n * @param {number} [x=0] - Horizontal position of the beginning of the text.\n * @param {number} [y=0] - Vertical position of the *baseline* of the text.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n * @param {GlyphRenderOptions=} options\n * @return {opentype.Path}\n */\nFont.prototype.getPath = function(text, x, y, fontSize, options) {\n const fullPath = new Path();\n this.forEachGlyph(text, x, y, fontSize, options, function(glyph, gX, gY, gFontSize) {\n const glyphPath = glyph.getPath(gX, gY, gFontSize, options, this);\n fullPath.extend(glyphPath);\n });\n return fullPath;\n};\n\n/**\n * Create an array of Path objects that represent the glyphs of a given text.\n * @param {string} text - The text to create.\n * @param {number} [x=0] - Horizontal position of the beginning of the text.\n * @param {number} [y=0] - Vertical position of the *baseline* of the text.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n * @param {GlyphRenderOptions=} options\n * @return {opentype.Path[]}\n */\nFont.prototype.getPaths = function(text, x, y, fontSize, options) {\n const glyphPaths = [];\n this.forEachGlyph(text, x, y, fontSize, options, function(glyph, gX, gY, gFontSize) {\n const glyphPath = glyph.getPath(gX, gY, gFontSize, options, this);\n glyphPaths.push(glyphPath);\n });\n\n return glyphPaths;\n};\n\n/**\n * Returns the advance width of a text.\n *\n * This is something different than Path.getBoundingBox() as for example a\n * suffixed whitespace increases the advanceWidth but not the bounding box\n * or an overhanging letter like a calligraphic 'f' might have a quite larger\n * bounding box than its advance width.\n *\n * This corresponds to canvas2dContext.measureText(text).width\n *\n * @param {string} text - The text to create.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n * @param {GlyphRenderOptions=} options\n * @return advance width\n */\nFont.prototype.getAdvanceWidth = function(text, fontSize, options) {\n return this.forEachGlyph(text, 0, 0, fontSize, options, function() {});\n};\n\n/**\n * Draw the text on the given drawing context.\n * @param {CanvasRenderingContext2D} ctx - A 2D drawing context, like Canvas.\n * @param {string} text - The text to create.\n * @param {number} [x=0] - Horizontal position of the beginning of the text.\n * @param {number} [y=0] - Vertical position of the *baseline* of the text.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n * @param {GlyphRenderOptions=} options\n */\nFont.prototype.draw = function(ctx, text, x, y, fontSize, options) {\n this.getPath(text, x, y, fontSize, options).draw(ctx);\n};\n\n/**\n * Draw the points of all glyphs in the text.\n * On-curve points will be drawn in blue, off-curve points will be drawn in red.\n * @param {CanvasRenderingContext2D} ctx - A 2D drawing context, like Canvas.\n * @param {string} text - The text to create.\n * @param {number} [x=0] - Horizontal position of the beginning of the text.\n * @param {number} [y=0] - Vertical position of the *baseline* of the text.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n * @param {GlyphRenderOptions=} options\n */\nFont.prototype.drawPoints = function(ctx, text, x, y, fontSize, options) {\n this.forEachGlyph(text, x, y, fontSize, options, function(glyph, gX, gY, gFontSize) {\n glyph.drawPoints(ctx, gX, gY, gFontSize);\n });\n};\n\n/**\n * Draw lines indicating important font measurements for all glyphs in the text.\n * Black lines indicate the origin of the coordinate system (point 0,0).\n * Blue lines indicate the glyph bounding box.\n * Green line indicates the advance width of the glyph.\n * @param {CanvasRenderingContext2D} ctx - A 2D drawing context, like Canvas.\n * @param {string} text - The text to create.\n * @param {number} [x=0] - Horizontal position of the beginning of the text.\n * @param {number} [y=0] - Vertical position of the *baseline* of the text.\n * @param {number} [fontSize=72] - Font size in pixels. We scale the glyph units by `1 / unitsPerEm * fontSize`.\n * @param {GlyphRenderOptions=} options\n */\nFont.prototype.drawMetrics = function(ctx, text, x, y, fontSize, options) {\n this.forEachGlyph(text, x, y, fontSize, options, function(glyph, gX, gY, gFontSize) {\n glyph.drawMetrics(ctx, gX, gY, gFontSize);\n });\n};\n\n/**\n * @param {string}\n * @return {string}\n */\nFont.prototype.getEnglishName = function(name) {\n const translations = this.names[name];\n if (translations) {\n return translations.en;\n }\n};\n\n/**\n * Validate\n */\nFont.prototype.validate = function() {\n const warnings = [];\n const _this = this;\n\n function assert(predicate, message) {\n if (!predicate) {\n warnings.push(message);\n }\n }\n\n function assertNamePresent(name) {\n const englishName = _this.getEnglishName(name);\n assert(englishName && englishName.trim().length > 0,\n 'No English ' + name + ' specified.');\n }\n\n // Identification information\n assertNamePresent('fontFamily');\n assertNamePresent('weightName');\n assertNamePresent('manufacturer');\n assertNamePresent('copyright');\n assertNamePresent('version');\n\n // Dimension information\n assert(this.unitsPerEm > 0, 'No unitsPerEm specified.');\n};\n\n/**\n * Convert the font object to a SFNT data structure.\n * This structure contains all the necessary tables and metadata to create a binary OTF file.\n * @return {opentype.Table}\n */\nFont.prototype.toTables = function() {\n return sfnt.fontToTable(this);\n};\n/**\n * @deprecated Font.toBuffer is deprecated. Use Font.toArrayBuffer instead.\n */\nFont.prototype.toBuffer = function() {\n console.warn('Font.toBuffer is deprecated. Use Font.toArrayBuffer instead.');\n return this.toArrayBuffer();\n};\n/**\n * Converts a `opentype.Font` into an `ArrayBuffer`\n * @return {ArrayBuffer}\n */\nFont.prototype.toArrayBuffer = function() {\n const sfntTable = this.toTables();\n const bytes = sfntTable.encode();\n const buffer = new ArrayBuffer(bytes.length);\n const intArray = new Uint8Array(buffer);\n for (let i = 0; i < bytes.length; i++) {\n intArray[i] = bytes[i];\n }\n\n return buffer;\n};\n\n/**\n * Initiate a download of the OpenType font.\n */\nFont.prototype.download = function(fileName) {\n const familyName = this.getEnglishName('fontFamily');\n const styleName = this.getEnglishName('fontSubfamily');\n fileName = fileName || familyName.replace(/\\s/g, '') + '-' + styleName + '.otf';\n const arrayBuffer = this.toArrayBuffer();\n\n if (isBrowser()) {\n window.URL = window.URL || window.webkitURL;\n\n if (window.URL) {\n const dataView = new DataView(arrayBuffer);\n const blob = new Blob([dataView], {type: 'font/opentype'});\n\n let link = document.createElement('a');\n link.href = window.URL.createObjectURL(blob);\n link.download = fileName;\n\n let event = document.createEvent('MouseEvents');\n event.initEvent('click', true, false);\n link.dispatchEvent(event);\n } else {\n console.warn('Font file could not be downloaded. Try using a different browser.');\n }\n } else {\n const fs = require('fs');\n const buffer = arrayBufferToNodeBuffer(arrayBuffer);\n fs.writeFileSync(fileName, buffer);\n }\n};\n/**\n * @private\n */\nFont.prototype.fsSelectionValues = {\n ITALIC: 0x001, //1\n UNDERSCORE: 0x002, //2\n NEGATIVE: 0x004, //4\n OUTLINED: 0x008, //8\n STRIKEOUT: 0x010, //16\n BOLD: 0x020, //32\n REGULAR: 0x040, //64\n USER_TYPO_METRICS: 0x080, //128\n WWS: 0x100, //256\n OBLIQUE: 0x200 //512\n};\n\n/**\n * @private\n */\nFont.prototype.usWidthClasses = {\n ULTRA_CONDENSED: 1,\n EXTRA_CONDENSED: 2,\n CONDENSED: 3,\n SEMI_CONDENSED: 4,\n MEDIUM: 5,\n SEMI_EXPANDED: 6,\n EXPANDED: 7,\n EXTRA_EXPANDED: 8,\n ULTRA_EXPANDED: 9\n};\n\n/**\n * @private\n */\nFont.prototype.usWeightClasses = {\n THIN: 100,\n EXTRA_LIGHT: 200,\n LIGHT: 300,\n NORMAL: 400,\n MEDIUM: 500,\n SEMI_BOLD: 600,\n BOLD: 700,\n EXTRA_BOLD: 800,\n BLACK: 900\n};\n\nexport default Font;\n","// The `fvar` table stores font variation axes and instances.\n// https://developer.apple.com/fonts/TrueType-Reference-Manual/RM06/Chap6fvar.html\n\nimport check from '../check';\nimport parse from '../parse';\nimport table from '../table';\n\nfunction addName(name, names) {\n const nameString = JSON.stringify(name);\n let nameID = 256;\n for (let nameKey in names) {\n let n = parseInt(nameKey);\n if (!n || n < 256) {\n continue;\n }\n\n if (JSON.stringify(names[nameKey]) === nameString) {\n return n;\n }\n\n if (nameID <= n) {\n nameID = n + 1;\n }\n }\n\n names[nameID] = name;\n return nameID;\n}\n\nfunction makeFvarAxis(n, axis, names) {\n const nameID = addName(axis.name, names);\n return [\n {name: 'tag_' + n, type: 'TAG', value: axis.tag},\n {name: 'minValue_' + n, type: 'FIXED', value: axis.minValue << 16},\n {name: 'defaultValue_' + n, type: 'FIXED', value: axis.defaultValue << 16},\n {name: 'maxValue_' + n, type: 'FIXED', value: axis.maxValue << 16},\n {name: 'flags_' + n, type: 'USHORT', value: 0},\n {name: 'nameID_' + n, type: 'USHORT', value: nameID}\n ];\n}\n\nfunction parseFvarAxis(data, start, names) {\n const axis = {};\n const p = new parse.Parser(data, start);\n axis.tag = p.parseTag();\n axis.minValue = p.parseFixed();\n axis.defaultValue = p.parseFixed();\n axis.maxValue = p.parseFixed();\n p.skip('uShort', 1); // reserved for flags; no values defined\n axis.name = names[p.parseUShort()] || {};\n return axis;\n}\n\nfunction makeFvarInstance(n, inst, axes, names) {\n const nameID = addName(inst.name, names);\n const fields = [\n {name: 'nameID_' + n, type: 'USHORT', value: nameID},\n {name: 'flags_' + n, type: 'USHORT', value: 0}\n ];\n\n for (let i = 0; i < axes.length; ++i) {\n const axisTag = axes[i].tag;\n fields.push({\n name: 'axis_' + n + ' ' + axisTag,\n type: 'FIXED',\n value: inst.coordinates[axisTag] << 16\n });\n }\n\n return fields;\n}\n\nfunction parseFvarInstance(data, start, axes, names) {\n const inst = {};\n const p = new parse.Parser(data, start);\n inst.name = names[p.parseUShort()] || {};\n p.skip('uShort', 1); // reserved for flags; no values defined\n\n inst.coordinates = {};\n for (let i = 0; i < axes.length; ++i) {\n inst.coordinates[axes[i].tag] = p.parseFixed();\n }\n\n return inst;\n}\n\nfunction makeFvarTable(fvar, names) {\n const result = new table.Table('fvar', [\n {name: 'version', type: 'ULONG', value: 0x10000},\n {name: 'offsetToData', type: 'USHORT', value: 0},\n {name: 'countSizePairs', type: 'USHORT', value: 2},\n {name: 'axisCount', type: 'USHORT', value: fvar.axes.length},\n {name: 'axisSize', type: 'USHORT', value: 20},\n {name: 'instanceCount', type: 'USHORT', value: fvar.instances.length},\n {name: 'instanceSize', type: 'USHORT', value: 4 + fvar.axes.length * 4}\n ]);\n result.offsetToData = result.sizeOf();\n\n for (let i = 0; i < fvar.axes.length; i++) {\n result.fields = result.fields.concat(makeFvarAxis(i, fvar.axes[i], names));\n }\n\n for (let j = 0; j < fvar.instances.length; j++) {\n result.fields = result.fields.concat(makeFvarInstance(j, fvar.instances[j], fvar.axes, names));\n }\n\n return result;\n}\n\nfunction parseFvarTable(data, start, names) {\n const p = new parse.Parser(data, start);\n const tableVersion = p.parseULong();\n check.argument(tableVersion === 0x00010000, 'Unsupported fvar table version.');\n const offsetToData = p.parseOffset16();\n // Skip countSizePairs.\n p.skip('uShort', 1);\n const axisCount = p.parseUShort();\n const axisSize = p.parseUShort();\n const instanceCount = p.parseUShort();\n const instanceSize = p.parseUShort();\n\n const axes = [];\n for (let i = 0; i < axisCount; i++) {\n axes.push(parseFvarAxis(data, start + offsetToData + i * axisSize, names));\n }\n\n const instances = [];\n const instanceStart = start + offsetToData + axisCount * axisSize;\n for (let j = 0; j < instanceCount; j++) {\n instances.push(parseFvarInstance(data, instanceStart + j * instanceSize, axes, names));\n }\n\n return {axes: axes, instances: instances};\n}\n\nexport default { make: makeFvarTable, parse: parseFvarTable };\n","// The `GDEF` table contains various glyph properties\n// https://docs.microsoft.com/en-us/typography/opentype/spec/gdef\n\nimport check from '../check';\nimport { Parser } from '../parse';\n\nvar attachList = function() {\n return {\n coverage: this.parsePointer(Parser.coverage),\n attachPoints: this.parseList(Parser.pointer(Parser.uShortList))\n };\n};\n\nvar caretValue = function() {\n var format = this.parseUShort();\n check.argument(format === 1 || format === 2 || format === 3,\n 'Unsupported CaretValue table version.');\n if (format === 1) {\n return { coordinate: this.parseShort() };\n } else if (format === 2) {\n return { pointindex: this.parseShort() };\n } else if (format === 3) {\n // Device / Variation Index tables unsupported\n return { coordinate: this.parseShort() };\n }\n};\n\nvar ligGlyph = function() {\n return this.parseList(Parser.pointer(caretValue));\n};\n\nvar ligCaretList = function() {\n return {\n coverage: this.parsePointer(Parser.coverage),\n ligGlyphs: this.parseList(Parser.pointer(ligGlyph))\n };\n};\n\nvar markGlyphSets = function() {\n this.parseUShort(); // Version\n return this.parseList(Parser.pointer(Parser.coverage));\n};\n\nfunction parseGDEFTable(data, start) {\n start = start || 0;\n const p = new Parser(data, start);\n const tableVersion = p.parseVersion(1);\n check.argument(tableVersion === 1 || tableVersion === 1.2 || tableVersion === 1.3,\n 'Unsupported GDEF table version.');\n var gdef = {\n version: tableVersion,\n classDef: p.parsePointer(Parser.classDef),\n attachList: p.parsePointer(attachList),\n ligCaretList: p.parsePointer(ligCaretList),\n markAttachClassDef: p.parsePointer(Parser.classDef)\n };\n if (tableVersion >= 1.2) {\n gdef.markGlyphSets = p.parsePointer(markGlyphSets);\n }\n return gdef;\n}\nexport default { parse: parseGDEFTable };\n","// The `GPOS` table contains kerning pairs, among other things.\n// https://docs.microsoft.com/en-us/typography/opentype/spec/gpos\n\nimport check from '../check';\nimport { Parser } from '../parse';\nimport table from '../table';\n\nconst subtableParsers = new Array(10); // subtableParsers[0] is unused\n\n// https://docs.microsoft.com/en-us/typography/opentype/spec/gpos#lookup-type-1-single-adjustment-positioning-subtable\n// this = Parser instance\nsubtableParsers[1] = function parseLookup1() {\n const start = this.offset + this.relativeOffset;\n const posformat = this.parseUShort();\n if (posformat === 1) {\n return {\n posFormat: 1,\n coverage: this.parsePointer(Parser.coverage),\n value: this.parseValueRecord()\n };\n } else if (posformat === 2) {\n return {\n posFormat: 2,\n coverage: this.parsePointer(Parser.coverage),\n values: this.parseValueRecordList()\n };\n }\n check.assert(false, '0x' + start.toString(16) + ': GPOS lookup type 1 format must be 1 or 2.');\n};\n\n// https://docs.microsoft.com/en-us/typography/opentype/spec/gpos#lookup-type-2-pair-adjustment-positioning-subtable\nsubtableParsers[2] = function parseLookup2() {\n const start = this.offset + this.relativeOffset;\n const posFormat = this.parseUShort();\n check.assert(posFormat === 1 || posFormat === 2, '0x' + start.toString(16) + ': GPOS lookup type 2 format must be 1 or 2.');\n const coverage = this.parsePointer(Parser.coverage);\n const valueFormat1 = this.parseUShort();\n const valueFormat2 = this.parseUShort();\n if (posFormat === 1) {\n // Adjustments for Glyph Pairs\n return {\n posFormat: posFormat,\n coverage: coverage,\n valueFormat1: valueFormat1,\n valueFormat2: valueFormat2,\n pairSets: this.parseList(Parser.pointer(Parser.list(function() {\n return { // pairValueRecord\n secondGlyph: this.parseUShort(),\n value1: this.parseValueRecord(valueFormat1),\n value2: this.parseValueRecord(valueFormat2)\n };\n })))\n };\n } else if (posFormat === 2) {\n const classDef1 = this.parsePointer(Parser.classDef);\n const classDef2 = this.parsePointer(Parser.classDef);\n const class1Count = this.parseUShort();\n const class2Count = this.parseUShort();\n return {\n // Class Pair Adjustment\n posFormat: posFormat,\n coverage: coverage,\n valueFormat1: valueFormat1,\n valueFormat2: valueFormat2,\n classDef1: classDef1,\n classDef2: classDef2,\n class1Count: class1Count,\n class2Count: class2Count,\n classRecords: this.parseList(class1Count, Parser.list(class2Count, function() {\n return {\n value1: this.parseValueRecord(valueFormat1),\n value2: this.parseValueRecord(valueFormat2)\n };\n }))\n };\n }\n};\n\nsubtableParsers[3] = function parseLookup3() { return { error: 'GPOS Lookup 3 not supported' }; };\nsubtableParsers[4] = function parseLookup4() { return { error: 'GPOS Lookup 4 not supported' }; };\nsubtableParsers[5] = function parseLookup5() { return { error: 'GPOS Lookup 5 not supported' }; };\nsubtableParsers[6] = function parseLookup6() { return { error: 'GPOS Lookup 6 not supported' }; };\nsubtableParsers[7] = function parseLookup7() { return { error: 'GPOS Lookup 7 not supported' }; };\nsubtableParsers[8] = function parseLookup8() { return { error: 'GPOS Lookup 8 not supported' }; };\nsubtableParsers[9] = function parseLookup9() { return { error: 'GPOS Lookup 9 not supported' }; };\n\n// https://docs.microsoft.com/en-us/typography/opentype/spec/gpos\nfunction parseGposTable(data, start) {\n start = start || 0;\n const p = new Parser(data, start);\n const tableVersion = p.parseVersion(1);\n check.argument(tableVersion === 1 || tableVersion === 1.1, 'Unsupported GPOS table version ' + tableVersion);\n\n if (tableVersion === 1) {\n return {\n version: tableVersion,\n scripts: p.parseScriptList(),\n features: p.parseFeatureList(),\n lookups: p.parseLookupList(subtableParsers)\n };\n } else {\n return {\n version: tableVersion,\n scripts: p.parseScriptList(),\n features: p.parseFeatureList(),\n lookups: p.parseLookupList(subtableParsers),\n variations: p.parseFeatureVariationsList()\n };\n }\n\n}\n\n// GPOS Writing //////////////////////////////////////////////\n// NOT SUPPORTED\nconst subtableMakers = new Array(10);\n\nfunction makeGposTable(gpos) {\n return new table.Table('GPOS', [\n {name: 'version', type: 'ULONG', value: 0x10000},\n {name: 'scripts', type: 'TABLE', value: new table.ScriptList(gpos.scripts)},\n {name: 'features', type: 'TABLE', value: new table.FeatureList(gpos.features)},\n {name: 'lookups', type: 'TABLE', value: new table.LookupList(gpos.lookups, subtableMakers)}\n ]);\n}\n\nexport default { parse: parseGposTable, make: makeGposTable };\n","// The `kern` table contains kerning pairs.\n// Note that some fonts use the GPOS OpenType layout table to specify kerning.\n// https://www.microsoft.com/typography/OTSPEC/kern.htm\n\nimport check from '../check';\nimport parse from '../parse';\n\nfunction parseWindowsKernTable(p) {\n const pairs = {};\n // Skip nTables.\n p.skip('uShort');\n const subtableVersion = p.parseUShort();\n check.argument(subtableVersion === 0, 'Unsupported kern sub-table version.');\n // Skip subtableLength, subtableCoverage\n p.skip('uShort', 2);\n const nPairs = p.parseUShort();\n // Skip searchRange, entrySelector, rangeShift.\n p.skip('uShort', 3);\n for (let i = 0; i < nPairs; i += 1) {\n const leftIndex = p.parseUShort();\n const rightIndex = p.parseUShort();\n const value = p.parseShort();\n pairs[leftIndex + ',' + rightIndex] = value;\n }\n return pairs;\n}\n\nfunction parseMacKernTable(p) {\n const pairs = {};\n // The Mac kern table stores the version as a fixed (32 bits) but we only loaded the first 16 bits.\n // Skip the rest.\n p.skip('uShort');\n const nTables = p.parseULong();\n //check.argument(nTables === 1, 'Only 1 subtable is supported (got ' + nTables + ').');\n if (nTables > 1) {\n console.warn('Only the first kern subtable is supported.');\n }\n p.skip('uLong');\n const coverage = p.parseUShort();\n const subtableVersion = coverage & 0xFF;\n p.skip('uShort');\n if (subtableVersion === 0) {\n const nPairs = p.parseUShort();\n // Skip searchRange, entrySelector, rangeShift.\n p.skip('uShort', 3);\n for (let i = 0; i < nPairs; i += 1) {\n const leftIndex = p.parseUShort();\n const rightIndex = p.parseUShort();\n const value = p.parseShort();\n pairs[leftIndex + ',' + rightIndex] = value;\n }\n }\n return pairs;\n}\n\n// Parse the `kern` table which contains kerning pairs.\nfunction parseKernTable(data, start) {\n const p = new parse.Parser(data, start);\n const tableVersion = p.parseUShort();\n if (tableVersion === 0) {\n return parseWindowsKernTable(p);\n } else if (tableVersion === 1) {\n return parseMacKernTable(p);\n } else {\n throw new Error('Unsupported kern table version (' + tableVersion + ').');\n }\n}\n\nexport default { parse: parseKernTable };\n","// The `loca` table stores the offsets to the locations of the glyphs in the font.\n// https://www.microsoft.com/typography/OTSPEC/loca.htm\n\nimport parse from '../parse';\n\n// Parse the `loca` table. This table stores the offsets to the locations of the glyphs in the font,\n// relative to the beginning of the glyphData table.\n// The number of glyphs stored in the `loca` table is specified in the `maxp` table (under numGlyphs)\n// The loca table has two versions: a short version where offsets are stored as uShorts, and a long\n// version where offsets are stored as uLongs. The `head` table specifies which version to use\n// (under indexToLocFormat).\nfunction parseLocaTable(data, start, numGlyphs, shortVersion) {\n const p = new parse.Parser(data, start);\n const parseFn = shortVersion ? p.parseUShort : p.parseULong;\n // There is an extra entry after the last index element to compute the length of the last glyph.\n // That's why we use numGlyphs + 1.\n const glyphOffsets = [];\n for (let i = 0; i < numGlyphs + 1; i += 1) {\n let glyphOffset = parseFn.call(p);\n if (shortVersion) {\n // The short table version stores the actual offset divided by 2.\n glyphOffset *= 2;\n }\n\n glyphOffsets.push(glyphOffset);\n }\n\n return glyphOffsets;\n}\n\nexport default { parse: parseLocaTable };\n","// opentype.js\n// https://github.com/opentypejs/opentype.js\n// (c) 2015 Frederik De Bleser\n// opentype.js may be freely distributed under the MIT license.\n\n/* global DataView, Uint8Array, XMLHttpRequest */\n\nimport 'string.prototype.codepointat';\nimport inflate from 'tiny-inflate';\nimport Font from './font';\nimport Glyph from './glyph';\nimport { CmapEncoding, GlyphNames, addGlyphNames } from './encoding';\nimport parse from './parse';\nimport BoundingBox from './bbox';\nimport Path from './path';\nimport { nodeBufferToArrayBuffer } from './util';\nimport cmap from './tables/cmap';\nimport cff from './tables/cff';\nimport fvar from './tables/fvar';\nimport glyf from './tables/glyf';\nimport gdef from './tables/gdef';\nimport gpos from './tables/gpos';\nimport gsub from './tables/gsub';\nimport head from './tables/head';\nimport hhea from './tables/hhea';\nimport hmtx from './tables/hmtx';\nimport kern from './tables/kern';\nimport ltag from './tables/ltag';\nimport loca from './tables/loca';\nimport maxp from './tables/maxp';\nimport _name from './tables/name';\nimport os2 from './tables/os2';\nimport post from './tables/post';\nimport meta from './tables/meta';\n\n/**\n * The opentype library.\n * @namespace opentype\n */\n\n// File loaders /////////////////////////////////////////////////////////\n/**\n * Loads a font from a file. The callback throws an error message as the first parameter if it fails\n * and the font as an ArrayBuffer in the second parameter if it succeeds.\n * @param {string} path - The path of the file\n * @param {Function} callback - The function to call when the font load completes\n */\nfunction loadFromFile(path, callback) {\n const fs = require('fs');\n fs.readFile(path, function(err, buffer) {\n if (err) {\n return callback(err.message);\n }\n\n callback(null, nodeBufferToArrayBuffer(buffer));\n });\n}\n/**\n * Loads a font from a URL. The callback throws an error message as the first parameter if it fails\n * and the font as an ArrayBuffer in the second parameter if it succeeds.\n * @param {string} url - The URL of the font file.\n * @param {Function} callback - The function to call when the font load completes\n */\nfunction loadFromUrl(url, callback) {\n const request = new XMLHttpRequest();\n request.open('get', url, true);\n request.responseType = 'arraybuffer';\n request.onload = function() {\n if (request.response) {\n return callback(null, request.response);\n } else {\n return callback('Font could not be loaded: ' + request.statusText);\n }\n };\n\n request.onerror = function () {\n callback('Font could not be loaded');\n };\n\n request.send();\n}\n\n// Table Directory Entries //////////////////////////////////////////////\n/**\n * Parses OpenType table entries.\n * @param {DataView}\n * @param {Number}\n * @return {Object[]}\n */\nfunction parseOpenTypeTableEntries(data, numTables) {\n const tableEntries = [];\n let p = 12;\n for (let i = 0; i < numTables; i += 1) {\n const tag = parse.getTag(data, p);\n const checksum = parse.getULong(data, p + 4);\n const offset = parse.getULong(data, p + 8);\n const length = parse.getULong(data, p + 12);\n tableEntries.push({tag: tag, checksum: checksum, offset: offset, length: length, compression: false});\n p += 16;\n }\n\n return tableEntries;\n}\n\n/**\n * Parses WOFF table entries.\n * @param {DataView}\n * @param {Number}\n * @return {Object[]}\n */\nfunction parseWOFFTableEntries(data, numTables) {\n const tableEntries = [];\n let p = 44; // offset to the first table directory entry.\n for (let i = 0; i < numTables; i += 1) {\n const tag = parse.getTag(data, p);\n const offset = parse.getULong(data, p + 4);\n const compLength = parse.getULong(data, p + 8);\n const origLength = parse.getULong(data, p + 12);\n let compression;\n if (compLength < origLength) {\n compression = 'WOFF';\n } else {\n compression = false;\n }\n\n tableEntries.push({tag: tag, offset: offset, compression: compression,\n compressedLength: compLength, length: origLength});\n p += 20;\n }\n\n return tableEntries;\n}\n\n/**\n * @typedef TableData\n * @type Object\n * @property {DataView} data - The DataView\n * @property {number} offset - The data offset.\n */\n\n/**\n * @param {DataView}\n * @param {Object}\n * @return {TableData}\n */\nfunction uncompressTable(data, tableEntry) {\n if (tableEntry.compression === 'WOFF') {\n const inBuffer = new Uint8Array(data.buffer, tableEntry.offset + 2, tableEntry.compressedLength - 2);\n const outBuffer = new Uint8Array(tableEntry.length);\n inflate(inBuffer, outBuffer);\n if (outBuffer.byteLength !== tableEntry.length) {\n throw new Error('Decompression error: ' + tableEntry.tag + ' decompressed length doesn\\'t match recorded length');\n }\n\n const view = new DataView(outBuffer.buffer, 0);\n return {data: view, offset: 0};\n } else {\n return {data: data, offset: tableEntry.offset};\n }\n}\n\n// Public API ///////////////////////////////////////////////////////////\n\n/**\n * Parse the OpenType file data (as an ArrayBuffer) and return a Font object.\n * Throws an error if the font could not be parsed.\n * @param {ArrayBuffer}\n * @param {Object} opt - options for parsing\n * @return {opentype.Font}\n */\nfunction parseBuffer(buffer, opt) {\n opt = (opt === undefined || opt === null) ? {} : opt;\n\n let indexToLocFormat;\n let ltagTable;\n\n // Since the constructor can also be called to create new fonts from scratch, we indicate this\n // should be an empty font that we'll fill with our own data.\n const font = new Font({empty: true});\n\n // OpenType fonts use big endian byte ordering.\n // We can't rely on typed array view types, because they operate with the endianness of the host computer.\n // Instead we use DataViews where we can specify endianness.\n const data = new DataView(buffer, 0);\n let numTables;\n let tableEntries = [];\n const signature = parse.getTag(data, 0);\n if (signature === String.fromCharCode(0, 1, 0, 0) || signature === 'true' || signature === 'typ1') {\n font.outlinesFormat = 'truetype';\n numTables = parse.getUShort(data, 4);\n tableEntries = parseOpenTypeTableEntries(data, numTables);\n } else if (signature === 'OTTO') {\n font.outlinesFormat = 'cff';\n numTables = parse.getUShort(data, 4);\n tableEntries = parseOpenTypeTableEntries(data, numTables);\n } else if (signature === 'wOFF') {\n const flavor = parse.getTag(data, 4);\n if (flavor === String.fromCharCode(0, 1, 0, 0)) {\n font.outlinesFormat = 'truetype';\n } else if (flavor === 'OTTO') {\n font.outlinesFormat = 'cff';\n } else {\n throw new Error('Unsupported OpenType flavor ' + signature);\n }\n\n numTables = parse.getUShort(data, 12);\n tableEntries = parseWOFFTableEntries(data, numTables);\n } else {\n throw new Error('Unsupported OpenType signature ' + signature);\n }\n\n let cffTableEntry;\n let fvarTableEntry;\n let glyfTableEntry;\n let gdefTableEntry;\n let gposTableEntry;\n let gsubTableEntry;\n let hmtxTableEntry;\n let kernTableEntry;\n let locaTableEntry;\n let nameTableEntry;\n let metaTableEntry;\n let p;\n\n for (let i = 0; i < numTables; i += 1) {\n const tableEntry = tableEntries[i];\n let table;\n switch (tableEntry.tag) {\n case 'cmap':\n table = uncompressTable(data, tableEntry);\n font.tables.cmap = cmap.parse(table.data, table.offset);\n font.encoding = new CmapEncoding(font.tables.cmap);\n break;\n case 'cvt ' :\n table = uncompressTable(data, tableEntry);\n p = new parse.Parser(table.data, table.offset);\n font.tables.cvt = p.parseShortList(tableEntry.length / 2);\n break;\n case 'fvar':\n fvarTableEntry = tableEntry;\n break;\n case 'fpgm' :\n table = uncompressTable(data, tableEntry);\n p = new parse.Parser(table.data, table.offset);\n font.tables.fpgm = p.parseByteList(tableEntry.length);\n break;\n case 'head':\n table = uncompressTable(data, tableEntry);\n font.tables.head = head.parse(table.data, table.offset);\n font.unitsPerEm = font.tables.head.unitsPerEm;\n indexToLocFormat = font.tables.head.indexToLocFormat;\n break;\n case 'hhea':\n table = uncompressTable(data, tableEntry);\n font.tables.hhea = hhea.parse(table.data, table.offset);\n font.ascender = font.tables.hhea.ascender;\n font.descender = font.tables.hhea.descender;\n font.numberOfHMetrics = font.tables.hhea.numberOfHMetrics;\n break;\n case 'hmtx':\n hmtxTableEntry = tableEntry;\n break;\n case 'ltag':\n table = uncompressTable(data, tableEntry);\n ltagTable = ltag.parse(table.data, table.offset);\n break;\n case 'maxp':\n table = uncompressTable(data, tableEntry);\n font.tables.maxp = maxp.parse(table.data, table.offset);\n font.numGlyphs = font.tables.maxp.numGlyphs;\n break;\n case 'name':\n nameTableEntry = tableEntry;\n break;\n case 'OS/2':\n table = uncompressTable(data, tableEntry);\n font.tables.os2 = os2.parse(table.data, table.offset);\n break;\n case 'post':\n table = uncompressTable(data, tableEntry);\n font.tables.post = post.parse(table.data, table.offset);\n font.glyphNames = new GlyphNames(font.tables.post);\n break;\n case 'prep' :\n table = uncompressTable(data, tableEntry);\n p = new parse.Parser(table.data, table.offset);\n font.tables.prep = p.parseByteList(tableEntry.length);\n break;\n case 'glyf':\n glyfTableEntry = tableEntry;\n break;\n case 'loca':\n locaTableEntry = tableEntry;\n break;\n case 'CFF ':\n cffTableEntry = tableEntry;\n break;\n case 'kern':\n kernTableEntry = tableEntry;\n break;\n case 'GDEF':\n gdefTableEntry = tableEntry;\n break;\n case 'GPOS':\n gposTableEntry = tableEntry;\n break;\n case 'GSUB':\n gsubTableEntry = tableEntry;\n break;\n case 'meta':\n metaTableEntry = tableEntry;\n break;\n }\n }\n\n const nameTable = uncompressTable(data, nameTableEntry);\n font.tables.name = _name.parse(nameTable.data, nameTable.offset, ltagTable);\n font.names = font.tables.name;\n\n if (glyfTableEntry && locaTableEntry) {\n const shortVersion = indexToLocFormat === 0;\n const locaTable = uncompressTable(data, locaTableEntry);\n const locaOffsets = loca.parse(locaTable.data, locaTable.offset, font.numGlyphs, shortVersion);\n const glyfTable = uncompressTable(data, glyfTableEntry);\n font.glyphs = glyf.parse(glyfTable.data, glyfTable.offset, locaOffsets, font, opt);\n } else if (cffTableEntry) {\n const cffTable = uncompressTable(data, cffTableEntry);\n cff.parse(cffTable.data, cffTable.offset, font, opt);\n } else {\n throw new Error('Font doesn\\'t contain TrueType or CFF outlines.');\n }\n\n const hmtxTable = uncompressTable(data, hmtxTableEntry);\n hmtx.parse(font, hmtxTable.data, hmtxTable.offset, font.numberOfHMetrics, font.numGlyphs, font.glyphs, opt);\n addGlyphNames(font, opt);\n\n if (kernTableEntry) {\n const kernTable = uncompressTable(data, kernTableEntry);\n font.kerningPairs = kern.parse(kernTable.data, kernTable.offset);\n } else {\n font.kerningPairs = {};\n }\n\n if (gdefTableEntry) {\n const gdefTable = uncompressTable(data, gdefTableEntry);\n font.tables.gdef = gdef.parse(gdefTable.data, gdefTable.offset);\n }\n\n if (gposTableEntry) {\n const gposTable = uncompressTable(data, gposTableEntry);\n font.tables.gpos = gpos.parse(gposTable.data, gposTable.offset);\n font.position.init();\n }\n\n if (gsubTableEntry) {\n const gsubTable = uncompressTable(data, gsubTableEntry);\n font.tables.gsub = gsub.parse(gsubTable.data, gsubTable.offset);\n }\n\n if (fvarTableEntry) {\n const fvarTable = uncompressTable(data, fvarTableEntry);\n font.tables.fvar = fvar.parse(fvarTable.data, fvarTable.offset, font.names);\n }\n\n if (metaTableEntry) {\n const metaTable = uncompressTable(data, metaTableEntry);\n font.tables.meta = meta.parse(metaTable.data, metaTable.offset);\n font.metas = font.tables.meta;\n }\n\n return font;\n}\n\n/**\n * Asynchronously load the font from a URL or a filesystem. When done, call the callback\n * with two arguments `(err, font)`. The `err` will be null on success,\n * the `font` is a Font object.\n * We use the node.js callback convention so that\n * opentype.js can integrate with frameworks like async.js.\n * @alias opentype.load\n * @param {string} url - The URL of the font to load.\n * @param {Function} callback - The callback.\n */\nfunction load(url, callback, opt) {\n opt = (opt === undefined || opt === null) ? {} : opt;\n const isNode = typeof window === 'undefined';\n const loadFn = isNode && !opt.isUrl ? loadFromFile : loadFromUrl;\n\n return new Promise((resolve, reject) => {\n loadFn(url, function(err, arrayBuffer) {\n if (err) {\n if (callback) {\n return callback(err);\n } else {\n reject(err);\n }\n }\n let font;\n try {\n font = parseBuffer(arrayBuffer, opt);\n } catch (e) {\n if (callback) {\n return callback(e, null);\n } else {\n reject(e);\n }\n }\n if (callback) {\n return callback(null, font);\n } else {\n resolve(font);\n }\n });\n });\n}\n\n/**\n * Synchronously load the font from a URL or file.\n * When done, returns the font object or throws an error.\n * @alias opentype.loadSync\n * @param {string} url - The URL of the font to load.\n * @param {Object} opt - opt.lowMemory\n * @return {opentype.Font}\n */\nfunction loadSync(url, opt) {\n const fs = require('fs');\n const buffer = fs.readFileSync(url);\n return parseBuffer(nodeBufferToArrayBuffer(buffer), opt);\n}\n\nexport {\n Font,\n Glyph,\n Path,\n BoundingBox,\n parse as _parse,\n parseBuffer as parse,\n load,\n loadSync\n};\n","import { Camera } from '@babylonjs/core/Cameras/camera';\r\nimport { BoundingBox } from '@babylonjs/core/Culling/boundingBox';\r\nimport { MultiMaterial } from '@babylonjs/core/Materials/multiMaterial';\r\nimport { StandardMaterial } from '@babylonjs/core/Materials/standardMaterial';\r\nimport { Color3, Color4, Matrix, Quaternion, Vector3 } from '@babylonjs/core/Maths';\r\nimport { CreatePlane } from '@babylonjs/core/Meshes/Builders/planeBuilder';\r\nimport { SphereBuilder as BabylonSphereBuilder } from '@babylonjs/core/Meshes/Builders/sphereBuilder';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { LinesMesh } from '@babylonjs/core/Meshes/linesMesh';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { VertexData } from '@babylonjs/core/Meshes/mesh.vertexData';\r\nimport { MeshBuilder } from '@babylonjs/core/Meshes/meshBuilder';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport '@babylonjs/core/Rendering/outlineRenderer';\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { eAxis, eCylinderCap, eRenderMode } from '@models/classes/enums';\r\nimport { getByDynamicId } from '@models/classes/kb3d-manager';\r\nimport { IUvData } from '@models/classes/kb3d-models';\r\nimport { Kb3dObject } from '@models/classes/kb3d-object';\r\nimport { MaterialNode } from '@models/classes/materials/material-node';\r\nimport {\r\n MeshGroupNode,\r\n SketchNode,\r\n SpaceNode,\r\n TextNode,\r\n isBoxMesh,\r\n isConeMesh,\r\n isCylinderMesh,\r\n isDiscMesh,\r\n isPlaneMesh,\r\n isSketchNode,\r\n isSphereMesh,\r\n isTextNode,\r\n isTorusMesh,\r\n} from '@models/classes/meshes';\r\nimport { CustomMeshNode } from '@models/classes/meshes/custom-mesh-node';\r\nimport { MeshNode } from '@models/classes/meshes/mesh-node';\r\nimport {\r\n isCustomMeshNode,\r\n isMeshGroupNode,\r\n isMeshNode,\r\n isPrimitiveMeshNode,\r\n} from '@models/classes/meshes/mesh-utilities';\r\nimport { PrimitiveMeshNode } from '@models/classes/meshes/primitive-mesh-node';\r\nimport { KbColor, KbGeometry } from '@models/classes/primitives';\r\nimport { KbQuaternion } from '@models/classes/primitives/kb-quaternion';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { SceneNode, isSceneNode } from '@models/classes/scene-node';\r\nimport { Token } from '@models/classes/token';\r\nimport { isDefined } from '@models/classes/utilities';\r\nimport { eParentChangeBehavior } from '@models/interfaces/settings';\r\nimport { ITrackedClass, isTrackedClass } from '@models/interfaces/tracked-class';\r\nimport { SettingsManagerService } from '@models/services/settings-manager-service';\r\nimport { eSideOrientationBabylonMap } from '@view/helpers/enum-maps';\r\nimport { SubmeshVertexData, verifyGeometry } from '@view/helpers/geometry-helper';\r\nimport { toRadians } from '@view/helpers/math-helper';\r\nimport { CylinderBuilder } from '@view/helpers/mesh-builders/cylinder-builder';\r\nimport { DiscBuilder } from '@view/helpers/mesh-builders/disc-builder';\r\nimport { PlaneBuilder } from '@view/helpers/mesh-builders/plane-builder';\r\nimport { SphereBuilder } from '@view/helpers/mesh-builders/sphere-builder';\r\nimport { TorusBuilder } from '@view/helpers/mesh-builders/torus-builder';\r\nimport { TextMeshFont, TextMeshVertexData } from '@view/helpers/text-mesh-helper';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport {\r\n axisToVector3,\r\n calculateFaceNormal,\r\n fillSketchPolygon,\r\n parseSketchGeometry,\r\n} from '@view/modules/sketching/util';\r\nimport { parse as fontParse } from 'opentype.js';\r\nimport { Subject } from 'rxjs';\r\nimport { take, takeUntil } from 'rxjs/operators';\r\nimport { Renderer } from './renderer';\r\nimport { RendererManager } from './renderer-manager';\r\nimport { ShadowManager } from './shadow-manager';\r\n\r\nconst primitiveShapeProperties = new Set([\r\n 'height',\r\n 'width',\r\n 'depth',\r\n 'radius',\r\n 'radiusTop',\r\n 'radiusBottom',\r\n 'thickness',\r\n 'arc',\r\n 'latitudeStart',\r\n 'latitudeSlice',\r\n 'tessellation',\r\n 'subdivisions',\r\n 'capMode',\r\n 'segments',\r\n]);\r\n\r\nconst PIVOT_GIZMO_SCALE = 0.33;\r\n\r\nconst capModeMap: Record = {\r\n [eCylinderCap.both]: Mesh.CAP_ALL,\r\n [eCylinderCap.start]: Mesh.CAP_START,\r\n [eCylinderCap.end]: Mesh.CAP_END,\r\n [eCylinderCap.none]: Mesh.NO_CAP,\r\n};\r\n\r\ntype meshNodeSharedChanges = ITrackedClass['changes'];\r\nconst textMeshFonts = new Map();\r\n\r\n// const lineColor = new Color4(1, 1, 1);\r\n// const selectedLineColor = new Color4(1, 0, 0);\r\n\r\nexport class MeshRenderer extends Renderer {\r\n utilityLayer: UtilityLayerRenderer;\r\n disposed = new Subject();\r\n pivotGizmos = new Map();\r\n\r\n sketchPreviews = new Map();\r\n sketchTracings = new Map();\r\n sketchPlanePreviews = new Map();\r\n\r\n private renderMode: eRenderMode;\r\n\r\n constructor(\r\n viewer: KbViewer,\r\n scene: Scene,\r\n private rendererManager: RendererManager,\r\n private shadowManager: ShadowManager\r\n ) {\r\n super(viewer, scene);\r\n this.utilityLayer = viewer.utilityLayer;\r\n\r\n const tempVector = new Vector3();\r\n viewer.afterRender.pipe(takeUntil(this.disposed)).subscribe(() => {\r\n let dist: number;\r\n const camera = this.utilityLayer.utilityLayerScene.activeCamera!;\r\n\r\n this.pivotGizmos.forEach(gizmo => {\r\n if (camera.mode === Camera.ORTHOGRAPHIC_CAMERA) {\r\n dist = ((camera.orthoRight || 0) - (camera.orthoLeft || 0)) / 2;\r\n } else {\r\n const cameraPosition = camera.globalPosition;\r\n gizmo.position.subtractToRef(cameraPosition, tempVector);\r\n dist = tempVector.length() * PIVOT_GIZMO_SCALE;\r\n }\r\n\r\n gizmo.scaling.set(dist, dist, dist);\r\n });\r\n });\r\n }\r\n\r\n modelToken = Token.SpaceNode;\r\n handlesNode(node: Kb3dObject) {\r\n return isMeshNode(node) || isMeshGroupNode(node) || (isSceneNode(node) && node.isNested());\r\n }\r\n\r\n setRenderMode(mode: eRenderMode, sceneNode: SceneNode) {\r\n this.renderMode = mode;\r\n this.babylonMap.forEach(mesh => {\r\n if (mesh instanceof Mesh) {\r\n this.setRenderModeOnMesh(mesh, sceneNode);\r\n }\r\n });\r\n }\r\n\r\n dispose() {\r\n super.dispose();\r\n this.pivotGizmos.forEach(gizmo => {\r\n gizmo.dispose();\r\n });\r\n this.pivotGizmos.clear();\r\n this.disposed.next();\r\n }\r\n\r\n loadNewGeometry(node: CustomMeshNode) {\r\n node._loading = node.geometryId;\r\n const meshService = node._manager!.getService(Token.MeshService);\r\n meshService!.hydrateGeometry(node).then(() => {\r\n node._loading = null;\r\n this.invalidateBoundingBox(node);\r\n });\r\n }\r\n\r\n parentsVisible(node: SpaceNode) {\r\n while (node._parent && node._parent instanceof SpaceNode) {\r\n if (!node._parent.visible) {\r\n return false;\r\n }\r\n node = node._parent;\r\n }\r\n return true;\r\n }\r\n\r\n update3dInternal(node: ITrackedClass) {\r\n const isNewMesh = !this.get3d(node);\r\n if (isCustomMeshNode(node)) {\r\n if (\r\n node.visible &&\r\n this.parentsVisible(node) &&\r\n node.hasGeometry &&\r\n !node._geometry &&\r\n !node.geometryLoaded &&\r\n !node._loading\r\n ) {\r\n this.loadNewGeometry(node);\r\n }\r\n }\r\n\r\n const mesh = this.createOrUpdateMesh(node);\r\n\r\n const pivotGizmo = this.pivotGizmos.get(node._dynamicId);\r\n if (node._selected && mesh instanceof Mesh && !this.viewer.highlighter.has(node)) {\r\n this.viewer.highlighter.add(node, KbColor.Red());\r\n if (pivotGizmo) {\r\n pivotGizmo.setEnabled(true);\r\n }\r\n } else if (!node._selected && mesh instanceof Mesh && this.viewer.highlighter.has(node)) {\r\n this.viewer.highlighter.remove(node);\r\n if (pivotGizmo) {\r\n pivotGizmo.setEnabled(false);\r\n }\r\n }\r\n\r\n if (node instanceof SketchNode) {\r\n if (node._selected && node.visible) {\r\n const oldPreview = this.sketchPlanePreviews.get(node.id);\r\n if (oldPreview) {\r\n oldPreview.dispose();\r\n }\r\n const mesh = this.createSketchPlaneVisualization(node);\r\n if (node.visible && mesh) {\r\n this.sketchPlanePreviews.set(node.id, mesh);\r\n } else {\r\n mesh?.dispose();\r\n this.sketchPlanePreviews.delete(node.id);\r\n }\r\n } else if (!node._selected || !node.visible) {\r\n const SketchPlancePreviewMesh = this.sketchPlanePreviews.get(node.id);\r\n SketchPlancePreviewMesh?.dispose();\r\n this.sketchPlanePreviews.delete(node.id);\r\n }\r\n }\r\n\r\n if (!mesh) {\r\n throw Error('Could not create mesh');\r\n }\r\n\r\n if (isNewMesh || node.changes.has('_parent')) {\r\n this.updateParentMeshFromNode(node, mesh, isNewMesh);\r\n }\r\n }\r\n\r\n rebuildBaseGeometry(node: SpaceNode, preTransformChanged = false) {\r\n const mesh = this.get3d(node);\r\n if (mesh && mesh instanceof Mesh) {\r\n let meshVertexData: SubmeshVertexData | null = null;\r\n\r\n if (isPrimitiveMeshNode(node)) {\r\n const newMesh = this.createPrimitiveMesh(node);\r\n meshVertexData = SubmeshVertexData.ExtractFromMesh(newMesh);\r\n newMesh.dispose();\r\n\r\n if (isDefined(node.uvOverrides)) {\r\n this.setBabylonUvData(node.uvOverrides, meshVertexData);\r\n }\r\n\r\n if (node instanceof MeshNode && meshVertexData) {\r\n // Always pretransform primitives as they are always rebuilt from scratch\r\n this.preTransformVertices(meshVertexData, node);\r\n }\r\n } else if (isCustomMeshNode(node)) {\r\n // If custom geometry does not have a geometry, don't do anything. Return null.\r\n if (node._geometry && node._geometry.positions) {\r\n meshVertexData = this.getMeshVertexData(node);\r\n }\r\n } else if (isSketchNode(node)) {\r\n meshVertexData = this.buildSketchGeometry(node);\r\n } else if (isTextNode(node) && isTrackedClass(node)) {\r\n meshVertexData = this.buildTextGeometry(node);\r\n } else {\r\n throw Error('geometry needs to be recalculated but mesh is not a valid primitive or custom mesh');\r\n }\r\n\r\n return meshVertexData;\r\n }\r\n\r\n return null;\r\n }\r\n\r\n private buildSketchGeometry(node: SketchNode) {\r\n const result = new SubmeshVertexData();\r\n const positionVectors: Vector3[] = [];\r\n let pathSegmentIndices: number[][] = [];\r\n\r\n // Keep track of the path groups so we can match the hole paths with the groups in the vertex data\r\n // The key is the path group key, and the value is the indices of the generated polygons that are part of that path\r\n // One connected path may have multiple sketch paths because it requires different types of curves\r\n const pathGroupKeys = new Map();\r\n const holePathsReferencingKeys: string[] = [];\r\n\r\n let currentPath: number[] | undefined;\r\n let lastPoint: Vector3 | undefined;\r\n for (let i = 0; i < node.paths.length; i++) {\r\n const path = node.paths[i];\r\n\r\n if (path._points && path._points.length > 0) {\r\n let thisPoint: Vector3 | undefined;\r\n\r\n for (let iP = 0; iP < path._points.length; iP++) {\r\n const { x, y, z } = path._points[iP];\r\n thisPoint = new Vector3(x, y, z);\r\n\r\n // Separate paths that don't have a shared start/end to different logical paths\r\n if (!lastPoint || !thisPoint.equals(lastPoint)) {\r\n if (iP === 0 || !currentPath) {\r\n if (!pathGroupKeys.has(path.pathGroupKey)) {\r\n pathGroupKeys.set(path.pathGroupKey, pathSegmentIndices.length);\r\n }\r\n if (path.holeForPathGroup) {\r\n holePathsReferencingKeys[pathSegmentIndices.length] = path.holeForPathGroup;\r\n }\r\n currentPath = [];\r\n pathSegmentIndices.push(currentPath);\r\n }\r\n\r\n currentPath.push(positionVectors.length);\r\n positionVectors.push(thisPoint);\r\n }\r\n\r\n lastPoint = thisPoint;\r\n }\r\n }\r\n }\r\n\r\n if (positionVectors.length === 0 || pathSegmentIndices.length === 0) {\r\n return null;\r\n }\r\n\r\n const positions: number[] = [];\r\n const indices: number[] = [];\r\n const normals: number[] = [];\r\n const paths: number[] = [];\r\n\r\n if (node.flipPath) {\r\n pathSegmentIndices.reverse();\r\n pathSegmentIndices.forEach(segment => segment.reverse());\r\n }\r\n\r\n const primaryNormal = axisToVector3(node.normal);\r\n // /**\r\n // * We want the sketch to always \"face\" in the positive direction of the axis for consistency\r\n // * Otherwise it's indeterminate which way the sketch will face and makes the sketch hard to manage\r\n // * because it depends on the handed-ness of the points (clockwise or counter-clockwise)\r\n // */\r\n pathSegmentIndices.forEach(segment => {\r\n const segmentPositions: Vector3[] = [];\r\n for (let i = 0; i < segment.length; i++) {\r\n const point = positionVectors[segment[i]];\r\n segmentPositions.push(point);\r\n }\r\n\r\n // However, don't flip non-closed paths because it makes no sense and may be a drive curve\r\n const closedPath =\r\n (segmentPositions.length > 1 &&\r\n segmentPositions[0].equalsWithEpsilon(segmentPositions[segmentPositions.length - 1])) ||\r\n node.closePath;\r\n if (closedPath) {\r\n const faceNormal = calculateFaceNormal(segmentPositions);\r\n // If the rotation handedness is opposite the primary sketch vector, then flip the sketch path\r\n if (Vector3.Dot(faceNormal, primaryNormal) < 0) {\r\n segment.reverse();\r\n }\r\n }\r\n });\r\n\r\n // Parse the holes and match them to their parent polygons\r\n const holesByPathsWithIndices: number[][][] = [];\r\n pathSegmentIndices.forEach((segment, idx) => {\r\n if (!holePathsReferencingKeys[idx]) {\r\n return;\r\n }\r\n const parentPathIndex = pathGroupKeys.get(holePathsReferencingKeys[idx]);\r\n if (parentPathIndex === undefined) {\r\n return;\r\n }\r\n // This segment is a hole\r\n if (!holesByPathsWithIndices[parentPathIndex]) {\r\n holesByPathsWithIndices[parentPathIndex] = [];\r\n }\r\n holesByPathsWithIndices[parentPathIndex].push(segment);\r\n });\r\n\r\n if (node.addFace) {\r\n // We have to reorder the segment indices because the vertex position order gets modified by the face fill\r\n // Due to holes needing be be defined alongside the parent polygons\r\n const reorderedPathSegmentIndices: number[][] = [];\r\n let positionPointer = 0;\r\n pathSegmentIndices.forEach((segment, idx) => {\r\n // Skip holes in the loop since they are processed with their parent polygons\r\n if (holePathsReferencingKeys[idx]) {\r\n return;\r\n }\r\n const segmentPositions: Vector3[] = [];\r\n const segmentIndices: number[] = [];\r\n for (let i = 0; i < segment.length; i++) {\r\n const point = positionVectors[segment[i]];\r\n segmentPositions.push(point);\r\n segmentIndices.push(positionPointer);\r\n positionPointer++;\r\n }\r\n reorderedPathSegmentIndices.push(segmentIndices);\r\n\r\n const holesPositions: Vector3[][] = [];\r\n if (holesByPathsWithIndices[idx]) {\r\n for (let hi = 0; hi < holesByPathsWithIndices[idx].length; hi++) {\r\n const hole = holesByPathsWithIndices[idx][hi];\r\n const holeData: Vector3[] = [];\r\n const holeIndices: number[] = [];\r\n for (let i = 0; i < hole.length; i++) {\r\n const point = positionVectors[hole[i]];\r\n holeData.push(point);\r\n holeIndices.push(positionPointer);\r\n positionPointer++;\r\n }\r\n holesPositions.push(holeData);\r\n reorderedPathSegmentIndices.push(holeIndices);\r\n }\r\n }\r\n\r\n const newMeshVertices = fillSketchPolygon(\r\n segmentPositions,\r\n node.flipPath,\r\n primaryNormal,\r\n holesPositions\r\n );\r\n\r\n // Fix the indices to point at the right vertices\r\n const startIndex = positions.length / 3;\r\n for (let i = 0; i < newMeshVertices.indices.length; i++) {\r\n newMeshVertices.indices[i] += startIndex;\r\n }\r\n\r\n indices.push(...newMeshVertices.indices);\r\n positions.push(...newMeshVertices.positions);\r\n normals.push(...newMeshVertices.normals);\r\n });\r\n\r\n //const normals = new Float32Array(positionVectors.length * 3);\r\n const uvs = new Float32Array(positionVectors.length * 2);\r\n uvs.fill(0);\r\n\r\n result.positions = new Float32Array(positions);\r\n result.indices = new Uint16Array(indices);\r\n result.normals = new Float32Array(normals);\r\n result.uvs = uvs;\r\n pathSegmentIndices = reorderedPathSegmentIndices;\r\n } else {\r\n const positions = new Float32Array(positionVectors.length * 3);\r\n for (let i = 0; i < positionVectors.length; i++) {\r\n const point = positionVectors[i];\r\n positions.set([point.x, point.y, point.z], i * 3);\r\n }\r\n result.positions = positions;\r\n }\r\n\r\n const indexMapToGrouped: number[] = [];\r\n let i = 0;\r\n pathSegmentIndices.forEach((segment, indexOfPath) => {\r\n // Skip holes since they are processed later\r\n if (holePathsReferencingKeys[indexOfPath]) {\r\n return;\r\n }\r\n indexMapToGrouped[indexOfPath] = i;\r\n for (let i = 1; i < segment.length; i++) {\r\n paths.push(segment[i - 1], segment[i]);\r\n }\r\n\r\n if (segment.length > 2 && node.closePath) {\r\n paths.push(segment[segment.length - 1], segment[0]);\r\n }\r\n i++;\r\n });\r\n\r\n const holes: number[] = [];\r\n pathSegmentIndices.forEach((segment, indexOfPath) => {\r\n // Add holes after the regular polys\r\n if (!holePathsReferencingKeys[indexOfPath]) {\r\n return;\r\n }\r\n const parentPathIndex = pathGroupKeys.get(holePathsReferencingKeys[indexOfPath]);\r\n if (parentPathIndex === undefined) {\r\n return;\r\n }\r\n\r\n holes.push(indexMapToGrouped[parentPathIndex], paths.length);\r\n for (let i = 1; i < segment.length; i++) {\r\n paths.push(segment[i - 1], segment[i]);\r\n }\r\n\r\n if (segment.length > 2 && node.closePath) {\r\n paths.push(segment[segment.length - 1], segment[0]);\r\n }\r\n });\r\n\r\n result.paths = new Uint16Array(paths);\r\n result.holes = new Uint16Array(holes);\r\n node._sketchPoints = new Float32Array(result.positions);\r\n let counter = -1;\r\n const startingPoints = [];\r\n //for loop building the starting points array which stores sketchControlPoints at the start of a line segment\r\n for (let i = 0; i < node.paths.length; i++) {\r\n const path = node.paths[i];\r\n let lastPoint: Vector3 | undefined;\r\n\r\n if (path._points && path._points.length > 0) {\r\n let thisPoint: Vector3 | undefined;\r\n\r\n for (let iP = 0; iP < path._points.length; iP++) {\r\n const { x, y, z } = path._points[iP];\r\n thisPoint = new Vector3(x, y, z);\r\n\r\n startingPoints.push(counter);\r\n if (!lastPoint || !thisPoint.equals(lastPoint)) {\r\n if (iP === 0 || !currentPath) {\r\n startingPoints.pop();\r\n }\r\n }\r\n\r\n lastPoint = thisPoint;\r\n counter++;\r\n }\r\n }\r\n }\r\n node._lineSegmentStartingPoints = startingPoints;\r\n node.calculatedBoundingBox = this.calculateSketchBoundingBox(node);\r\n\r\n return result;\r\n }\r\n\r\n private updateSketchPreview(\r\n sketchNode: ITrackedClass,\r\n vertexData: SubmeshVertexData | null,\r\n mesh: AbstractMesh\r\n ) {\r\n const previewMesh = this.sketchPreviews.get(sketchNode._dynamicId);\r\n\r\n const { segments } = parseSketchGeometry(vertexData);\r\n\r\n if (segments.length > 0) {\r\n const baseColor = sketchNode._rootScene?.edgeColor\r\n ? sketchNode._rootScene.edgeColor.toColor4()\r\n : new Color4(1, 1, 1, 1);\r\n const colors: Color4[][] = [];\r\n\r\n segments.forEach(segmentPoints => {\r\n const segmentColors: Color4[] = [];\r\n colors.push(segmentColors);\r\n segmentPoints.forEach(() => {\r\n segmentColors.push(baseColor);\r\n });\r\n });\r\n\r\n const lineColor = colors.length > 0 ? colors : undefined;\r\n\r\n if (\r\n previewMesh &&\r\n !sketchNode.changes.has('addFace') &&\r\n !sketchNode.changes.has('paths') &&\r\n !sketchNode.changes.has('closePath')\r\n ) {\r\n MeshBuilder.CreateLineSystem(\r\n `${sketchNode._dynamicId}_preview`,\r\n {\r\n lines: segments,\r\n instance: previewMesh,\r\n colors: lineColor,\r\n updatable: true,\r\n },\r\n this.scene\r\n );\r\n } else {\r\n if (previewMesh) {\r\n previewMesh.dispose();\r\n }\r\n const newPreviewMesh = MeshBuilder.CreateLineSystem(\r\n `${sketchNode._dynamicId}_preview`,\r\n {\r\n lines: segments,\r\n colors: lineColor,\r\n updatable: true,\r\n },\r\n this.scene\r\n );\r\n this.sketchPreviews.set(sketchNode._dynamicId, newPreviewMesh);\r\n newPreviewMesh.parent = mesh;\r\n newPreviewMesh.enableEdgesRendering();\r\n newPreviewMesh.edgesColor = baseColor;\r\n this.viewer.camera.maintainNodeScreenSize(newPreviewMesh, (dist: number, mode: number) => {\r\n if (mode === Camera.PERSPECTIVE_CAMERA) {\r\n newPreviewMesh.edgesWidth = dist * 0.25;\r\n } else {\r\n newPreviewMesh.edgesWidth = 4;\r\n }\r\n });\r\n }\r\n } else if (previewMesh) {\r\n previewMesh.dispose();\r\n this.sketchPreviews.delete(sketchNode._dynamicId);\r\n }\r\n }\r\n\r\n private updateSketchTracing(\r\n sketchNode: ITrackedClass,\r\n vertexData: SubmeshVertexData | null,\r\n mesh: AbstractMesh\r\n ) {\r\n const previewMesh = this.sketchTracings.get(sketchNode._dynamicId);\r\n const sketchColor = sketchNode.pathColor.toColor4();\r\n if (sketchNode.tracePath) {\r\n if (previewMesh) {\r\n previewMesh.dispose();\r\n }\r\n //create/update drawings\r\n const linePoints: Vector3[] = [];\r\n sketchNode.paths.forEach(path => {\r\n path._points.forEach(point => {\r\n linePoints.push(new Vector3(point.x, point.y, point.z));\r\n });\r\n });\r\n if (sketchNode.closePath) {\r\n const point = sketchNode.paths[0]._points[0];\r\n linePoints.push(new Vector3(point.x, point.y, point.z));\r\n }\r\n const lines = Mesh.CreateLines('lines', linePoints, this.scene, true);\r\n lines.enableEdgesRendering();\r\n lines.edgesWidth = sketchNode.lineThickness;\r\n lines.edgesColor = sketchColor;\r\n this.sketchTracings.set(sketchNode._dynamicId, lines);\r\n } else {\r\n if (previewMesh || (previewMesh && sketchNode.tracePath === false)) {\r\n previewMesh.dispose();\r\n this.sketchTracings.delete(sketchNode._dynamicId);\r\n }\r\n }\r\n }\r\n\r\n private buildTextGeometry(node: ITrackedClass): SubmeshVertexData | null {\r\n const fontService = node._manager!.getService(Token.FontService);\r\n\r\n if (!node.fontFile || !node.text) {\r\n return null;\r\n }\r\n\r\n if (!fontService.readyFonts.has(node.fontFile)) {\r\n fontService.loadFontByFile(node.fontFile).then(resultData => {\r\n const fontBuffer = resultData;\r\n if (fontBuffer) {\r\n node.setPropertyIsModified('fontFile', true);\r\n }\r\n });\r\n return null;\r\n } else {\r\n let vertexDataBuilder = textMeshFonts.get(node.fontFile);\r\n\r\n if (!vertexDataBuilder) {\r\n const fontBuffer = fontService.readyFonts.get(node.fontFile)!;\r\n const openTypeFont = fontParse(fontBuffer);\r\n const font = new TextMeshFont(node.fontFile, openTypeFont, this.scene);\r\n vertexDataBuilder = new TextMeshVertexData(font);\r\n textMeshFonts.set(node.fontFile, vertexDataBuilder);\r\n }\r\n return vertexDataBuilder.getText(node.text, node.fontSize, node.charSpacing, node.curveRadius);\r\n }\r\n }\r\n\r\n getMeshGeometry(node: SpaceNode) {\r\n const mesh = this.get3d(node);\r\n if (mesh && mesh instanceof Mesh && mesh._geometry) {\r\n const geometry = SubmeshVertexData.ExtractFromMesh(mesh);\r\n // // Make sure the geometry is valid and actually has some geometry in it so consumers of this function don't have to deep check.\r\n // if(!geometry.indices || !geometry.positions || geometry.indices.length === 0) {\r\n // return null;\r\n // }\r\n\r\n return geometry;\r\n }\r\n return null;\r\n }\r\n\r\n fetchMaterialNode(meshNode: MeshNode, materialId: string) {\r\n const localMaterial = meshNode._manager?.getById(materialId); //find the material if it's local\r\n\r\n return new Promise((resolve, reject) => {\r\n if (localMaterial) {\r\n //it's a local material\r\n resolve(localMaterial);\r\n } else {\r\n // Not a local material, so it's a global or library material\r\n this.rendererManager.material\r\n .loadGlobalMaterial(materialId, meshNode._manager!.services)\r\n .pipe(take(1))\r\n .subscribe(globalLibMaterial => {\r\n if (!globalLibMaterial) {\r\n console.warn(\r\n `Cannot find global material ${materialId} on node ${meshNode.name} (${meshNode.id})`\r\n );\r\n reject('could not load global material');\r\n } else {\r\n resolve(globalLibMaterial);\r\n }\r\n });\r\n }\r\n });\r\n }\r\n\r\n setRenderModeOnMesh(mesh: Mesh, sceneNode: SceneNode) {\r\n const mode = this.renderMode;\r\n if (mesh.geometry) {\r\n if (mode === eRenderMode.edge || mode === eRenderMode.hiddenEdges) {\r\n mesh.enableEdgesRendering(sceneNode.edgeDisplayTolerance);\r\n mesh.renderOutline = true;\r\n\r\n mesh.edgesWidth = sceneNode.edgeWidth;\r\n mesh.edgesColor = sceneNode.edgeColor.toColor4();\r\n mesh.outlineWidth = sceneNode.edgeWidth / 100;\r\n mesh.outlineColor = sceneNode.edgeColor.toColor3();\r\n\r\n // Another option for the hidden edges renderer but results in render artifacts especially when meshes go on top of each other.\r\n // if (mode === eRenderMode.hiddenEdges) {\r\n // mesh.metadata = { ...mesh.metadata, visibility: mesh.visibility };\r\n // mesh.visibility = 0.001;\r\n // }\r\n } else {\r\n // mesh.visibility = mesh.metadata?.visibility || 1;\r\n mesh.renderOutline = false;\r\n mesh.disableEdgesRendering();\r\n }\r\n }\r\n }\r\n\r\n private getMeshVertexData(node: CustomMeshNode) {\r\n if (!node._geometry) {\r\n throw Error('Cannot get vertex data because mesh has no geometry');\r\n }\r\n\r\n let meshVertexData = SubmeshVertexData.FromGeometry(node._geometry);\r\n meshVertexData = this.preTransformVertices(meshVertexData, node, node._geometry);\r\n\r\n return meshVertexData;\r\n }\r\n\r\n private preTransformVertices(meshVertexData: SubmeshVertexData, node: MeshNode, cachedGeometry?: KbGeometry) {\r\n if (node.preTransform && node.preTransform.length !== 16) {\r\n throw Error('invalid pretransform array ');\r\n }\r\n\r\n // If pretransform is not set on the node, the transform we want is the identity matrix (no change from original)\r\n const transform = node.preTransform ? Matrix.FromArray(node.preTransform) : Matrix.Identity();\r\n // Check if the geometry has a transform applied\r\n const existingTransform = cachedGeometry?._hasTransform;\r\n\r\n // If the geometry has already been transformed, then we have to undo that transform to get the correct transform\r\n // Starting with the invert matrix of the existing transform will cancel out the existing transform and result in the original mesh\r\n if (existingTransform) {\r\n Matrix.FromArray(existingTransform).invert().multiplyToRef(transform, transform);\r\n }\r\n\r\n // If the transform we want (taking the existing transform into account) is the identity matrix, do nothing and return the geometry\r\n if (!transform.isIdentity()) {\r\n if (existingTransform) {\r\n // If the geometry has already been transformed, don't overwrite the transform so we don't repeatedly transform it\r\n meshVertexData = meshVertexData.deepClone();\r\n } else if (cachedGeometry) {\r\n // If the geoemtry has no existing transform, set the hasTransform property since it will get mutated with the new transform.\r\n cachedGeometry._hasTransform = node.preTransform;\r\n }\r\n meshVertexData.transform(transform);\r\n }\r\n\r\n return meshVertexData;\r\n }\r\n\r\n private updateParentMeshFromNode(node: ITrackedClass, targetNode: TransformNode, isNewMesh: boolean) {\r\n const parentNode = node._parent;\r\n if (parentNode && parentNode instanceof SpaceNode) {\r\n const isRootScene = parentNode instanceof SceneNode && !parentNode.isNested();\r\n const parent3d = isRootScene\r\n ? this.rendererManager.sceneRenderer.get3d(parentNode as SceneNode)\r\n : this.get3d(parentNode);\r\n if (parent3d) {\r\n if (targetNode instanceof Mesh) {\r\n targetNode.setParent(parent3d);\r\n } else {\r\n targetNode.parent = parent3d;\r\n }\r\n } else {\r\n console.error(`Could not find mesh by ID ${parentNode.id} in scene!`);\r\n }\r\n } else {\r\n return; // if the node has no parent, it's not part of the scene anymore\r\n }\r\n\r\n if (\r\n isNewMesh ||\r\n SettingsManagerService.get('parentChangeBehavior') === eParentChangeBehavior.PRESERVE_PROPERTY\r\n ) {\r\n this.updateMeshTransform(node, targetNode, null);\r\n } else {\r\n node.position = KbVector.FromVec3(targetNode.position);\r\n\r\n if (targetNode.rotationQuaternion) {\r\n node.useEuler = false;\r\n node.setRotationQuaternion(\r\n new KbQuaternion(\r\n targetNode.rotationQuaternion.x,\r\n targetNode.rotationQuaternion.y,\r\n targetNode.rotationQuaternion.z,\r\n targetNode.rotationQuaternion.w\r\n )\r\n );\r\n } else if (node.eulerRotation) {\r\n node.useEuler = true;\r\n node.eulerRotation = new KbVector(targetNode.rotation.x, targetNode.rotation.y, targetNode.rotation.z);\r\n }\r\n\r\n node.scaling = targetNode.scaling.x;\r\n }\r\n this.invalidateBoundingBox(parentNode); //invalidate parent node in the bounding box cache as now it has a new child\r\n }\r\n\r\n private recursiveLoadChildMeshGeometry(node: SpaceNode) {\r\n const children = node.getChildren();\r\n children.forEach(child => {\r\n if (child instanceof SpaceNode && child.visible) {\r\n if (child instanceof CustomMeshNode && isTrackedClass(child)) {\r\n if (child.hasGeometry && !child._geometry && !child.geometryLoaded && !child._loading) {\r\n this.loadNewGeometry(child);\r\n }\r\n } else {\r\n this.recursiveLoadChildMeshGeometry(child);\r\n }\r\n }\r\n });\r\n }\r\n\r\n private createOrUpdateMesh(\r\n node: ITrackedClass,\r\n otherMesh?: Mesh\r\n ): T {\r\n let mesh: TransformNode | AbstractMesh | undefined;\r\n if (otherMesh) {\r\n mesh = otherMesh;\r\n } else {\r\n mesh = this.get3d(node);\r\n }\r\n const changes = node.changes as meshNodeSharedChanges;\r\n let allChanges = false;\r\n let meshVerticesNeedReset = false;\r\n if (!mesh) {\r\n // New Mesh\r\n\r\n allChanges = true;\r\n if (isMeshNode(node)) changes.set('features', node.features);\r\n\r\n if (isMeshGroupNode(node) || isSceneNode(node)) {\r\n mesh = new TransformNode(node._dynamicId, this.scene);\r\n } else {\r\n const newMesh = (mesh = new Mesh(node._dynamicId, this.scene));\r\n if (isCustomMeshNode(node) && isTrackedClass(node) && node._geometry) {\r\n this.setBabylonVertexData(node, newMesh);\r\n } else {\r\n meshVerticesNeedReset = true;\r\n }\r\n }\r\n mesh.scaling = Vector3.One();\r\n mesh.name = node.name;\r\n mesh.id = node._dynamicId;\r\n\r\n this.set3d(node, mesh);\r\n } else if (isCustomMeshNode(node) && changes.has('geometryId') && node._loading !== node.geometryId) {\r\n // Only load new geometry if the geometry hasn't just changed\r\n this.loadNewGeometry(node);\r\n } else if (isCustomMeshNode(node) && isTrackedClass(node) && node._geometry && node.changes.has('_geometry')) {\r\n // If we're changing the geometry, we always need to (re)apply all mesh properties\r\n // to the mesh as the new geometry won't have the right properties\r\n allChanges = true;\r\n changes.set('features', node.features);\r\n }\r\n\r\n if (\r\n allChanges ||\r\n changes.hasAny(\r\n 'position',\r\n 'useEuler',\r\n 'eulerRotation',\r\n 'rotationAxis',\r\n 'rotationAngle',\r\n 'pivot',\r\n 'scaling',\r\n '_showPivot'\r\n )\r\n ) {\r\n this.updateMeshTransform(node, mesh, allChanges ? null : (changes as ITrackedClass['changes']));\r\n if (isMeshNode(node)) {\r\n this.rendererManager.material.reflectedMeshes.set(node._dynamicId, mesh as AbstractMesh);\r\n }\r\n this.rendererManager.material.updateReflectionProbes();\r\n }\r\n\r\n if (changes.has('scaling')) {\r\n this.updateConnectorScaling(node);\r\n }\r\n\r\n const pivotGizmo = this.pivotGizmos.get(node._dynamicId);\r\n\r\n if (allChanges || changes.has('visible')) {\r\n mesh.setEnabled(!!node.visible);\r\n this.viewer.polygonTracker.next();\r\n this.viewer.warningTracker.next();\r\n\r\n if (pivotGizmo) {\r\n pivotGizmo.setEnabled(!!node.visible);\r\n }\r\n this.invalidateBoundingBox(node);\r\n\r\n // If a group node is made visible and it has a child node that has geometry load it now\r\n if (node.visible && node instanceof MeshGroupNode) {\r\n let parentNode = node._parent;\r\n let allParentsVisible = true;\r\n while (parentNode) {\r\n if (!parentNode.visible) {\r\n allParentsVisible = false;\r\n break;\r\n }\r\n parentNode = parentNode._parent;\r\n }\r\n if (allParentsVisible) {\r\n this.recursiveLoadChildMeshGeometry(node);\r\n }\r\n }\r\n }\r\n\r\n if (mesh instanceof Mesh) {\r\n const meshNode = node as ITrackedClass;\r\n mesh.isBlocker = false;\r\n\r\n if (allChanges || changes.has('receiveShadows')) {\r\n mesh.receiveShadows = !!meshNode.receiveShadows;\r\n this.shadowManager.toggleReceiver(mesh, !!meshNode.receiveShadows);\r\n }\r\n\r\n if (changes.has('isPickable')) {\r\n if (meshNode.isPickable !== undefined) {\r\n mesh.isPickable = meshNode.isPickable;\r\n } else {\r\n mesh.isPickable = true;\r\n }\r\n }\r\n\r\n if (allChanges || changes.has('castShadows')) this.shadowManager.toggleCaster(mesh, !!meshNode.castShadows);\r\n if (allChanges || changes.has('checkCollisions')) mesh.checkCollisions = !!meshNode.checkCollisions;\r\n if (allChanges || changes.has('flipSurfaceOrientation')) {\r\n if (meshNode.flipSurfaceOrientation) {\r\n mesh.overrideMaterialSideOrientation = Mesh.BACKSIDE;\r\n } else {\r\n mesh.overrideMaterialSideOrientation = Mesh.FRONTSIDE;\r\n }\r\n }\r\n\r\n if (this.renderMode !== eRenderMode.hiddenEdges) {\r\n if (allChanges || changes.has('opacity')) {\r\n mesh.visibility = meshNode.opacity == null ? 1 : meshNode.opacity;\r\n }\r\n }\r\n\r\n const warningTracker = {\r\n name: meshNode.name,\r\n isEnabled: meshNode.visible,\r\n castShadows: meshNode.castShadows || false,\r\n receiveShadows: meshNode.receiveShadows || false,\r\n polygonNumber: meshNode._polygonCount || 0,\r\n featureNumber: node.features.length,\r\n };\r\n this.viewer.MeshWarningTracker.set(mesh.id, warningTracker);\r\n // Material\r\n if (allChanges || changes.has('materialId') || changes.has('visible')) {\r\n const _mesh = mesh;\r\n\r\n // If the material is a multimaterial, make sure all the material nodes making it up are initialized\r\n if (meshNode.visible && mesh.material instanceof MultiMaterial) {\r\n mesh.material.subMaterials.forEach(material => {\r\n if (material) {\r\n const localMaterial = getByDynamicId(material.id, false);\r\n\r\n if (localMaterial) {\r\n localMaterial.init();\r\n } else {\r\n // Global material ?\r\n const globalMaterial = this.rendererManager.material.getGlobalMaterial(material.id);\r\n if (globalMaterial) {\r\n globalMaterial.init();\r\n }\r\n }\r\n }\r\n });\r\n }\r\n\r\n if (meshNode.materialId) {\r\n this.fetchMaterialNode(meshNode, meshNode.materialId).then(\r\n materialNode => {\r\n if (meshNode.visible) {\r\n this.setMaterialOnMesh(meshNode, _mesh, materialNode);\r\n }\r\n },\r\n () => {\r\n meshNode.materialId = undefined;\r\n }\r\n );\r\n } else {\r\n if (mesh.material instanceof MultiMaterial) {\r\n mesh.material.subMaterials[0] = this.rendererManager.material.defaultMaterial;\r\n mesh.material.forceCompilation(mesh); // If you don't do this webgl vomits errors because we invalidated the cache\r\n } else {\r\n mesh.material = this.rendererManager.material.defaultMaterial;\r\n }\r\n }\r\n\r\n if (isMeshNode(node)) {\r\n this.rendererManager.material.reflectedMeshes.set(node._dynamicId, mesh);\r\n }\r\n this.rendererManager.material.updateReflectionProbes();\r\n }\r\n\r\n let sketchNode: ITrackedClass | undefined;\r\n let updateSketchDrawing = false;\r\n\r\n // For primitives, recalculate the mesh if the shape properties changed\r\n if (isPrimitiveMeshNode(meshNode)) {\r\n if (allChanges) {\r\n meshVerticesNeedReset = true;\r\n } else {\r\n meshNode.changes.forEach((_, key) => {\r\n if (primitiveShapeProperties.has(key)) {\r\n meshVerticesNeedReset = true;\r\n }\r\n });\r\n }\r\n } else if (isCustomMeshNode(meshNode) && isTrackedClass(meshNode)) {\r\n meshVerticesNeedReset = meshNode.changes.has('_geometry') && !!meshNode._geometry;\r\n } else if (isSketchNode(meshNode) && isTrackedClass(meshNode)) {\r\n sketchNode = meshNode;\r\n meshVerticesNeedReset = sketchNode.changes.hasAny(\r\n 'paths',\r\n 'addFace',\r\n 'flipPath',\r\n 'clonePath',\r\n 'closePath',\r\n '_previewVisible'\r\n );\r\n\r\n updateSketchDrawing = sketchNode.changes.hasAny(\r\n 'paths',\r\n 'addFace',\r\n 'flipPath',\r\n 'clonePath',\r\n 'closePath',\r\n '_previewVisible',\r\n 'lineThickness',\r\n 'pathColor',\r\n 'tracePath'\r\n );\r\n\r\n if (sketchNode.changes.hasAny('boundingXOverride', 'boundingYOverride', 'boundingZOverride')) {\r\n sketchNode._boundingBoxChanges.next(sketchNode.getBoundingBox());\r\n }\r\n } else if (isTextNode(meshNode) && isTrackedClass(meshNode)) {\r\n meshVerticesNeedReset = meshNode.changes.hasAny(\r\n 'text',\r\n 'fontSize',\r\n 'curveRadius',\r\n 'charSpacing',\r\n 'fontFile'\r\n );\r\n }\r\n\r\n let preTransformChanged = false;\r\n if (allChanges || changes.hasAny('features', 'preTransform')) {\r\n meshVerticesNeedReset = true;\r\n preTransformChanged = !(allChanges || changes.has('_geometry')); // If geometry has changed, then the geometry was just loaded\r\n }\r\n\r\n // Features or shape properties have been changed, we have to recalculate original geometry and reprocess features on top\r\n if (allChanges || changes.has('uvOverrides') || meshVerticesNeedReset || updateSketchDrawing) {\r\n const meshVertexData = this.rebuildBaseGeometry(node, preTransformChanged);\r\n if (meshVertexData) {\r\n // This call should also apply the new vertex data.\r\n this.rendererManager.features.resetMeshGeometry(meshNode, null, meshVertexData);\r\n } else {\r\n if (mesh.geometry) {\r\n // If mesh geometry is set, the mesh was previously initialized but now is being generated with no geometry so we should destroy the old geometry\r\n // Failing to do so will cause Babylon geometry issues\r\n mesh.dispose();\r\n this.delete3d(node);\r\n }\r\n }\r\n\r\n if (sketchNode && sketchNode._previewVisible) {\r\n this.updateSketchPreview(sketchNode, meshVertexData, mesh);\r\n }\r\n\r\n if (sketchNode && sketchNode.tracePath && updateSketchDrawing) {\r\n if (sketchNode) {\r\n this.updateSketchTracing(sketchNode, meshVertexData, mesh);\r\n }\r\n }\r\n }\r\n\r\n // If this is a sketch, see if we need to show a preview of it\r\n if (sketchNode && sketchNode.changes.has('_previewVisible')) {\r\n const previewMesh = this.sketchPreviews.get(meshNode._dynamicId);\r\n if (previewMesh) {\r\n previewMesh.setEnabled(sketchNode._previewVisible);\r\n }\r\n }\r\n this.viewer.meshWarningsNeedsUpdate.next();\r\n }\r\n return mesh as unknown as T;\r\n }\r\n\r\n private updateMeshTransform(\r\n node: SpaceNode,\r\n mesh: T,\r\n changes: ITrackedClass['changes'] | null // null means all changes\r\n ): T {\r\n const pivot = node.pivot.toVec3();\r\n const originOffset = pivot.scale(-1);\r\n\r\n const newPosition = node.position.toVec3();\r\n\r\n let newRotation: Quaternion;\r\n if (node.useEuler) {\r\n newRotation = Quaternion.FromEulerAngles(\r\n toRadians(node.eulerRotation.x),\r\n toRadians(node.eulerRotation.y),\r\n toRadians(node.eulerRotation.z)\r\n );\r\n } else {\r\n newRotation = Quaternion.RotationAxis(node.rotationAxis.toVec3(), toRadians(node.rotationAngle));\r\n }\r\n const rotationMatrix = new Matrix();\r\n newRotation.toRotationMatrix(rotationMatrix);\r\n Vector3.TransformCoordinatesToRef(originOffset, rotationMatrix, originOffset);\r\n mesh.rotationQuaternion = newRotation;\r\n\r\n const scalingMatrix = Matrix.Scaling(node.scaling, node.scaling, node.scaling);\r\n Vector3.TransformCoordinatesToRef(originOffset, scalingMatrix, originOffset);\r\n mesh.scaling = new Vector3().setAll(node.scaling);\r\n\r\n newPosition.addInPlace(originOffset).addInPlace(pivot);\r\n\r\n if (!mesh.position.equals(newPosition)) {\r\n mesh.position = newPosition;\r\n }\r\n\r\n this.rendererManager.features.transformChanged(node);\r\n this.invalidateBoundingBox(!changes || changes.has('scaling') ? node : node._parent); //scaling, changes current node, but all others only change the parent\r\n\r\n if (node._showPivot) {\r\n let pivotGizmo = this.pivotGizmos.get(node._dynamicId);\r\n if (!pivotGizmo) {\r\n pivotGizmo = BabylonSphereBuilder.CreateSphere(\r\n 'pivotSphere',\r\n { diameter: 0.05 },\r\n this.utilityLayer.utilityLayerScene\r\n );\r\n\r\n const coloredMaterial = new StandardMaterial('', this.utilityLayer.utilityLayerScene);\r\n coloredMaterial.emissiveColor = Color3.Red();\r\n pivotGizmo.material = coloredMaterial;\r\n this.pivotGizmos.set(node._dynamicId, pivotGizmo);\r\n }\r\n if (!pivotGizmo.isEnabled()) {\r\n pivotGizmo.setEnabled(true);\r\n }\r\n\r\n const worldPivot = pivot.add(node.position.toVec3());\r\n if (mesh.parent) {\r\n const transform = mesh.parent.computeWorldMatrix(true);\r\n Vector3.TransformCoordinatesToRef(worldPivot, transform, worldPivot);\r\n }\r\n\r\n pivotGizmo.position = worldPivot;\r\n } else {\r\n const pivotGizmo = this.pivotGizmos.get(node._dynamicId);\r\n if (pivotGizmo && pivotGizmo.isEnabled()) {\r\n pivotGizmo.setEnabled(false);\r\n }\r\n }\r\n\r\n return mesh;\r\n }\r\n\r\n private createPrimitiveMesh(node: PrimitiveMeshNode): Mesh {\r\n if (!isPrimitiveMeshNode(node)) throw Error('Primitive not set');\r\n\r\n if (isBoxMesh(node)) {\r\n if (node.tessellation <= 1) {\r\n return MeshBuilder.CreateBox(\r\n node._dynamicId,\r\n {\r\n height: node.height,\r\n width: node.width,\r\n depth: node.depth,\r\n sideOrientation: eSideOrientationBabylonMap[node.sideOrientation],\r\n updatable: true,\r\n },\r\n this.scene\r\n );\r\n } else {\r\n return MeshBuilder.CreateTiledBox(\r\n node._dynamicId,\r\n {\r\n height: node.height,\r\n width: node.width,\r\n depth: node.depth,\r\n sideOrientation: eSideOrientationBabylonMap[node.sideOrientation],\r\n updatable: true,\r\n tileSize: 1 / node.tessellation,\r\n },\r\n this.scene\r\n );\r\n }\r\n } else if (isSphereMesh(node)) {\r\n return SphereBuilder.CreateSphere(\r\n node._dynamicId,\r\n {\r\n segments: node.tessellation,\r\n diameter: node.radius * 2,\r\n arc: node.arc / 360,\r\n sliceStart: node.latitudeStart / 180,\r\n slice: node.latitudeSlice / 180,\r\n sideOrientation: eSideOrientationBabylonMap[node.sideOrientation],\r\n updatable: true,\r\n },\r\n this.scene\r\n );\r\n } else if (isCylinderMesh(node)) {\r\n return CylinderBuilder.CreateCylinder(\r\n node._dynamicId,\r\n {\r\n height: node.height,\r\n diameter: node.radius * 2,\r\n tessellation: node.tessellation,\r\n subdivisions: node.subdivisions,\r\n hasRings: node.subdivisions > 1,\r\n cap: capModeMap[node.capMode],\r\n sideOrientation: eSideOrientationBabylonMap[node.sideOrientation],\r\n arc: node.arc / 360,\r\n updatable: true,\r\n },\r\n this.scene\r\n );\r\n } else if (isConeMesh(node)) {\r\n return MeshBuilder.CreateCylinder(\r\n node._dynamicId,\r\n {\r\n height: node.height,\r\n diameterTop: node.radiusTop * 2,\r\n diameterBottom: node.radiusBottom * 2,\r\n tessellation: node.tessellation,\r\n sideOrientation: eSideOrientationBabylonMap[node.sideOrientation],\r\n updatable: true,\r\n },\r\n this.scene\r\n );\r\n } else if (isTorusMesh(node)) {\r\n return TorusBuilder.CreateTorus(\r\n node._dynamicId,\r\n {\r\n diameter: node.radius * 2,\r\n thickness: node.thickness,\r\n tessellation: node.tessellation,\r\n sideOrientation: eSideOrientationBabylonMap[node.sideOrientation],\r\n arc: node.arc / 360,\r\n sliceStart: node.latitudeStart / 360,\r\n slice: node.latitudeSlice / 360,\r\n updatable: true,\r\n },\r\n this.scene\r\n );\r\n } else if (isPlaneMesh(node)) {\r\n const mesh = PlaneBuilder.CreatePlane(\r\n node._dynamicId,\r\n {\r\n width: node.width,\r\n height: node.depth,\r\n tileHeight: node.depth / node.segments,\r\n tileWidth: node.width / node.segments,\r\n sideOrientation: eSideOrientationBabylonMap[node.sideOrientation],\r\n updatable: true,\r\n },\r\n this.scene\r\n );\r\n mesh.rotation.x = Math.PI / 2;\r\n mesh.bakeCurrentTransformIntoVertices();\r\n return mesh;\r\n } else if (isDiscMesh(node)) {\r\n return DiscBuilder.CreateDisc(\r\n node._dynamicId,\r\n {\r\n radius: node.radius,\r\n tessellation: node.tessellation,\r\n sideOrientation: eSideOrientationBabylonMap[node.sideOrientation],\r\n updatable: true,\r\n },\r\n this.scene\r\n );\r\n } else {\r\n throw Error('Mesh primitive not implemented');\r\n }\r\n }\r\n\r\n private setBabylonVertexData(kbMesh: CustomMeshNode, mesh: Mesh, preserveSubmeshes = true) {\r\n if (!kbMesh._geometry) {\r\n return;\r\n }\r\n const vertexData = this.getMeshVertexData(kbMesh);\r\n\r\n const invalid = verifyGeometry(vertexData);\r\n if (invalid) {\r\n console.error(`Invalid or corrupted geometry in mesh ${kbMesh.name}. The ${invalid} are invalid.`);\r\n } else {\r\n vertexData.applyToMesh(mesh, true, preserveSubmeshes);\r\n kbMesh._polygonCount = mesh.getTotalIndices() / 3;\r\n kbMesh._verticesCount = mesh.getTotalVertices();\r\n this.viewer.polygonTracker.next();\r\n this.viewer.warningTracker.next();\r\n }\r\n\r\n this.setRenderModeOnMesh(mesh, kbMesh._rootScene!);\r\n\r\n if (mesh.isFacetDataEnabled) {\r\n mesh.updateFacetData();\r\n }\r\n }\r\n\r\n private setBabylonUvData(uvData: IUvData, vertexData: VertexData) {\r\n if (uvData.uvs) {\r\n vertexData.uvs = uvData.uvs;\r\n }\r\n if (uvData.uv2s) {\r\n vertexData.uvs2 = uvData.uv2s;\r\n }\r\n if (uvData.uv3s) {\r\n vertexData.uvs3 = uvData.uv3s;\r\n }\r\n if (uvData.uv4s) {\r\n vertexData.uvs4 = uvData.uv4s;\r\n }\r\n if (uvData.uv5s) {\r\n vertexData.uvs5 = uvData.uv5s;\r\n }\r\n if (uvData.uv6s) {\r\n vertexData.uvs6 = uvData.uv6s;\r\n }\r\n\r\n return vertexData;\r\n }\r\n\r\n protected setMaterialOnMesh(meshNode: MeshNode, babylonMesh: Mesh, materialNode: MaterialNode) {\r\n if (meshNode.visible) {\r\n materialNode.init();\r\n }\r\n\r\n const material = this.rendererManager.material.get3d(materialNode);\r\n if (!material) {\r\n throw Error('Material could not be found in renderer. Has the material been created?');\r\n }\r\n\r\n // If the mesh already has a material and it's a multimaterial, then update the material on index 0\r\n // Which is reserved for the \"base\" material.\r\n if (babylonMesh.material instanceof MultiMaterial) {\r\n babylonMesh.material.subMaterials[0] = material;\r\n babylonMesh.material.forceCompilation(babylonMesh);\r\n } else if (!babylonMesh.material || babylonMesh.material.id != materialNode._dynamicId) {\r\n babylonMesh.material = material;\r\n }\r\n }\r\n\r\n protected invalidateBoundingBox(node?: SpaceNode) {\r\n if (node) {\r\n this.viewer.bbCache.invalidate(node);\r\n }\r\n }\r\n\r\n public delete3d(node: ITrackedClass) {\r\n if (isSketchNode(node)) {\r\n node._boundingBoxChanges.complete();\r\n }\r\n\r\n const bnode = this.get3d(node);\r\n if (bnode && bnode instanceof Mesh) {\r\n this.viewer.MeshWarningTracker.delete(bnode.id);\r\n this.shadowManager.removeCaster(bnode);\r\n this.shadowManager.removeReceiver(bnode);\r\n if (bnode.material instanceof MultiMaterial) {\r\n bnode.material.dispose(true, true, false); // multimaterials are 1:1 with meshes\r\n }\r\n }\r\n this.rendererManager.material.reflectedMeshes.delete(node._dynamicId);\r\n this.rendererManager.material.updateReflectionProbes();\r\n this.viewer.meshWarningsNeedsUpdate.next();\r\n super.delete3d(node);\r\n this.viewer.polygonTracker.next();\r\n this.viewer.warningTracker.next();\r\n }\r\n\r\n public createSketchPlaneVisualization(sketch: SketchNode) {\r\n const color = Color3.Yellow();\r\n\r\n const boundingBox = sketch.calculatedBoundingBox;\r\n let squareSize = 3;\r\n let calculatedSize = 0;\r\n let boundingBoxCenter = new Vector3(0, 0, 0);\r\n if (boundingBox) {\r\n boundingBoxCenter = boundingBox.center.clone();\r\n calculatedSize = boundingBox.minimum.subtract(boundingBox.maximum).length();\r\n }\r\n if (calculatedSize > squareSize) {\r\n squareSize = calculatedSize * 1;\r\n }\r\n const coloredMaterial = new StandardMaterial('', this.scene);\r\n coloredMaterial.wireframe = true;\r\n coloredMaterial.emissiveColor = color;\r\n const sketchPlaneVisualization = CreatePlane('mirrorPlane', { size: squareSize }, this.scene);\r\n const sketchMesh = this.viewer.getRendererNode(sketch);\r\n if (sketchMesh) {\r\n const translation = sketchMesh.computeWorldMatrix();\r\n const position = Vector3.TransformCoordinates(boundingBoxCenter, translation);\r\n sketchPlaneVisualization.position = new Vector3(position.x, position.y, position.z);\r\n const sketchNormal = new Vector3(0, 0, 0);\r\n if (sketch.normal === eAxis.x) {\r\n sketchNormal.set(1, 0, 0);\r\n } else if (sketch.normal === eAxis.y) {\r\n sketchNormal.set(0, 1, 0);\r\n } else {\r\n sketchNormal.set(0, 0, 1);\r\n }\r\n const rotationVector = Vector3.TransformCoordinates(\r\n sketchNormal,\r\n sketchMesh.computeWorldMatrix().getRotationMatrix()\r\n );\r\n rotationVector.z = rotationVector.z * -1;\r\n const resultQuaternion = new Quaternion();\r\n Quaternion.FromUnitVectorsToRef(rotationVector, new Vector3(0, 0, 1), resultQuaternion);\r\n sketchPlaneVisualization.rotationQuaternion = resultQuaternion;\r\n\r\n //uvMapVisualization.rotation = Vector3.RotationFromAxis()\r\n sketchPlaneVisualization.material = coloredMaterial;\r\n\r\n sketchPlaneVisualization.bakeCurrentTransformIntoVertices();\r\n return sketchPlaneVisualization;\r\n } else {\r\n console.warn(\r\n 'Sketch visualization plane cannot be generated, please check direction and position of sketchpoints.'\r\n );\r\n return null;\r\n }\r\n }\r\n\r\n public updateConnectorScaling(node: SpaceNode) {\r\n if (node instanceof SketchNode) {\r\n node.paths.forEach(path => {\r\n path.controlPoints.forEach(controlPoint => {\r\n this.viewer.rendererManager.controlPointRenderer.updateScale(controlPoint);\r\n });\r\n });\r\n } else {\r\n if (node.connectors) {\r\n node.connectors.forEach(connector => {\r\n this.viewer.rendererManager.connector.updateScale(connector);\r\n });\r\n }\r\n const childNodes = node.getChildren();\r\n if (childNodes) {\r\n childNodes.forEach(child => {\r\n if (child instanceof SpaceNode) {\r\n this.updateConnectorScaling(child);\r\n }\r\n });\r\n }\r\n }\r\n }\r\n\r\n public calculateSketchBoundingBox(node: SketchNode) {\r\n const firstPoint = node.paths[0]._points[0];\r\n const minVector: Vector3 = new Vector3(firstPoint.x, firstPoint.y, firstPoint.z);\r\n const maxVector: Vector3 = new Vector3(firstPoint.x, firstPoint.y, firstPoint.z);\r\n node.paths.forEach(path => {\r\n path._points.forEach(point => {\r\n if (point.x < minVector.x) {\r\n minVector.x = point.x;\r\n }\r\n if (point.y < minVector.y) {\r\n minVector.y = point.y;\r\n }\r\n if (point.z < minVector.z) {\r\n minVector.z = point.z;\r\n }\r\n if (point.x > maxVector.x) {\r\n maxVector.x = point.x;\r\n }\r\n if (point.y > maxVector.y) {\r\n maxVector.y = point.y;\r\n }\r\n if (point.z > maxVector.z) {\r\n maxVector.z = point.z;\r\n }\r\n });\r\n });\r\n return new BoundingBox(minVector, maxVector);\r\n }\r\n}\r\n","import { Matrix, Vector3, Vector4 } from '@babylonjs/core/Maths';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { VertexData } from '@babylonjs/core/Meshes/mesh.vertexData';\r\n\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n */\r\nexport class SphereBuilder {\r\n /**\r\n * Creates a torus mesh\r\n * * The parameter `diameter` sets the diameter size (float) of the torus (default 1)\r\n * * The parameter `thickness` sets the diameter size of the tube of the torus (float, default 0.5)\r\n * * The parameter `tessellation` sets the number of torus sides (postive integer, default 16)\r\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/babylon101/discover_basic_elements#side-orientation\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param scene defines the hosting scene\r\n * @returns the torus mesh\r\n * @see https://doc.babylonjs.com/how_to/set_shapes#torus\r\n */\r\n public static CreateSphere(\r\n name: string,\r\n options: {\r\n segments?: number;\r\n diameter?: number;\r\n diameterX?: number;\r\n diameterY?: number;\r\n diameterZ?: number;\r\n arc?: number;\r\n slice?: number;\r\n sliceStart?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n updatable?: boolean;\r\n },\r\n scene: any\r\n ): Mesh {\r\n const sphere = new Mesh(name, scene);\r\n\r\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n sphere._originalBuilderSideOrientation = options.sideOrientation;\r\n\r\n const vertexData = createSphere(options);\r\n\r\n vertexData.applyToMesh(sphere, options.updatable);\r\n\r\n return sphere;\r\n\r\n function createSphere(options: {\r\n segments?: number;\r\n diameter?: number;\r\n diameterX?: number;\r\n diameterY?: number;\r\n diameterZ?: number;\r\n arc?: number;\r\n slice?: number;\r\n sliceStart?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n }) {\r\n const segments: number = options.segments || 32;\r\n const diameterX: number = options.diameterX || options.diameter || 1;\r\n const diameterY: number = options.diameterY || options.diameter || 1;\r\n const diameterZ: number = options.diameterZ || options.diameter || 1;\r\n const arc: number = options.arc && (options.arc <= 0 || options.arc > 1) ? 1.0 : options.arc || 1.0;\r\n const sliceStart =\r\n options.sliceStart && (options.sliceStart < 0 || options.sliceStart > 1) ? 0 : options.sliceStart || 0;\r\n let slice: number = options.slice && options.slice <= 0 ? 1.0 : options.slice || 1.0;\r\n const sideOrientation =\r\n options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\r\n\r\n if (slice + sliceStart > 1) {\r\n slice = 1 - sliceStart;\r\n }\r\n\r\n const radius = new Vector3(diameterX / 2, diameterY / 2, diameterZ / 2);\r\n\r\n const totalZRotationSteps = 2 + segments;\r\n const totalYRotationSegments = 2 * totalZRotationSteps;\r\n const totalYRotationSteps = totalYRotationSegments + 1;\r\n\r\n const indices = [];\r\n const positions = [];\r\n const normals = [];\r\n const uvs = [];\r\n\r\n for (let zRotationStep = 0; zRotationStep <= totalZRotationSteps; zRotationStep++) {\r\n const normalizedZ = zRotationStep / totalZRotationSteps;\r\n const angleZ = normalizedZ * Math.PI * slice + sliceStart * Math.PI;\r\n\r\n for (let yRotationStep = 0; yRotationStep < totalYRotationSteps; yRotationStep++) {\r\n const normalizedY = yRotationStep / totalYRotationSegments;\r\n\r\n const angleY = normalizedY * Math.PI * 2 * arc;\r\n\r\n const rotationZ = Matrix.RotationZ(-angleZ);\r\n const rotationY = Matrix.RotationY(angleY);\r\n const afterRotZ = Vector3.TransformCoordinates(Vector3.Up(), rotationZ);\r\n const complete = Vector3.TransformCoordinates(afterRotZ, rotationY);\r\n\r\n const vertex = complete.multiply(radius);\r\n const normal = complete.divide(radius).normalize();\r\n\r\n positions.push(vertex.x, vertex.y, vertex.z);\r\n normals.push(normal.x, normal.y, normal.z);\r\n uvs.push(normalizedY, 1 - normalizedZ);\r\n\r\n if (yRotationStep < totalYRotationSegments || arc === 1) {\r\n const nextYStep = (yRotationStep + 1) % totalYRotationSteps;\r\n if (sliceStart === 0 && zRotationStep === 0) {\r\n indices.push(0); // C\r\n indices.push(nextYStep); // B\r\n indices.push(yRotationStep); // A\r\n } else if (zRotationStep > 0) {\r\n indices.push((zRotationStep - 1) * totalYRotationSteps + yRotationStep); // C\r\n indices.push(zRotationStep * totalYRotationSteps + nextYStep); // B\r\n indices.push(zRotationStep * totalYRotationSteps + yRotationStep); // A\r\n\r\n indices.push((zRotationStep - 1) * totalYRotationSteps + yRotationStep); // C\r\n indices.push((zRotationStep - 1) * totalYRotationSteps + nextYStep); // D\r\n indices.push(zRotationStep * totalYRotationSteps + nextYStep); // B\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Sides\r\n VertexData._ComputeSides(\r\n sideOrientation,\r\n positions,\r\n indices,\r\n normals,\r\n uvs,\r\n options.frontUVs,\r\n options.backUVs\r\n );\r\n\r\n // Result\r\n const vertexData = new VertexData();\r\n\r\n vertexData.indices = indices;\r\n vertexData.positions = positions;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n\r\n return vertexData;\r\n }\r\n }\r\n}\r\n","import { Axis, Color4, Vector2, Vector3, Vector4 } from '@babylonjs/core/Maths';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { VertexData } from '@babylonjs/core/Meshes/mesh.vertexData';\r\nimport { Nullable } from '@babylonjs/core/types';\r\n\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n */\r\nexport class CylinderBuilder {\r\n /**\r\n * Creates a torus mesh\r\n * * The parameter `diameter` sets the diameter size (float) of the torus (default 1)\r\n * * The parameter `thickness` sets the diameter size of the tube of the torus (float, default 0.5)\r\n * * The parameter `tessellation` sets the number of torus sides (postive integer, default 16)\r\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/babylon101/Cylinderover_basic_elements#side-orientation\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param scene defines the hosting scene\r\n * @returns the torus mesh\r\n * @see https://doc.babylonjs.com/how_to/set_shapes#torus\r\n */\r\n public static CreateCylinder(\r\n name: string,\r\n options: {\r\n height?: number;\r\n diameterTop?: number;\r\n diameterBottom?: number;\r\n diameter?: number;\r\n tessellation?: number;\r\n subdivisions?: number;\r\n arc?: number;\r\n faceColors?: Color4[];\r\n faceUV?: Vector4[];\r\n hasRings?: boolean;\r\n enclose?: boolean;\r\n cap?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n updatable?: boolean;\r\n },\r\n scene: any\r\n ): Mesh {\r\n const Cylinder = new Mesh(name, scene);\r\n\r\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n Cylinder._originalBuilderSideOrientation = options.sideOrientation;\r\n\r\n const vertexData = createCylinder(options);\r\n\r\n vertexData.applyToMesh(Cylinder, options.updatable);\r\n\r\n return Cylinder;\r\n\r\n function createCylinder(options: {\r\n height?: number;\r\n diameterTop?: number;\r\n diameterBottom?: number;\r\n diameter?: number;\r\n tessellation?: number;\r\n subdivisions?: number;\r\n arc?: number;\r\n faceColors?: Color4[];\r\n faceUV?: Vector4[];\r\n hasRings?: boolean;\r\n enclose?: boolean;\r\n cap?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n }) {\r\n const height: number = options.height || 2;\r\n let diameterTop: number = options.diameterTop === 0 ? 0 : options.diameterTop || options.diameter || 1;\r\n let diameterBottom: number =\r\n options.diameterBottom === 0 ? 0 : options.diameterBottom || options.diameter || 1;\r\n diameterTop = diameterTop || 0.00001; // Prevent broken normals\r\n diameterBottom = diameterBottom || 0.00001; // Prevent broken normals\r\n const tessellation: number = options.tessellation || 24;\r\n const subdivisions: number = options.subdivisions || 1;\r\n const hasRings: boolean = options.hasRings ? true : false;\r\n const enclose: boolean = options.enclose ? true : false;\r\n const cap = options.cap === 0 ? 0 : options.cap || Mesh.CAP_ALL;\r\n const arc: number = options.arc && (options.arc <= 0 || options.arc > 1) ? 1.0 : options.arc || 1.0;\r\n const sideOrientation: number =\r\n options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\r\n const faceUV: Vector4[] = options.faceUV || new Array(3);\r\n const faceColors = options.faceColors;\r\n // default face colors and UV if undefined\r\n const quadNb: number = arc !== 1 && enclose ? 2 : 0;\r\n const ringNb: number = hasRings ? subdivisions : 1;\r\n const surfaceNb: number = 2 + (1 + quadNb) * ringNb;\r\n let f: number;\r\n\r\n for (f = 0; f < surfaceNb; f++) {\r\n if (faceColors && faceColors[f] === undefined) {\r\n faceColors[f] = new Color4(1, 1, 1, 1);\r\n }\r\n }\r\n for (f = 0; f < surfaceNb; f++) {\r\n if (faceUV && faceUV[f] === undefined) {\r\n faceUV[f] = new Vector4(0, 0, 1, 1);\r\n }\r\n }\r\n\r\n const indices = new Array();\r\n const positions = new Array();\r\n const normals = new Array();\r\n const uvs = new Array();\r\n const colors = new Array();\r\n\r\n const angle_step = (Math.PI * 2 * arc) / tessellation;\r\n let angle: number;\r\n let h: number;\r\n let radius: number;\r\n const tan = (diameterBottom - diameterTop) / 2 / height;\r\n const ringVertex: Vector3 = Vector3.Zero();\r\n const ringNormal: Vector3 = Vector3.Zero();\r\n const ringFirstVertex: Vector3 = Vector3.Zero();\r\n const ringFirstNormal: Vector3 = Vector3.Zero();\r\n const quadNormal: Vector3 = Vector3.Zero();\r\n const Y: Vector3 = Axis.Y;\r\n\r\n // positions, normals, uvs\r\n let i: number;\r\n let j: number;\r\n let r: number;\r\n let ringIdx = 1;\r\n let s = 1; // surface index\r\n let cs = 0;\r\n let v = 0;\r\n\r\n for (i = 0; i <= subdivisions; i++) {\r\n h = i / subdivisions;\r\n radius = (h * (diameterTop - diameterBottom) + diameterBottom) / 2;\r\n ringIdx = hasRings && i !== 0 && i !== subdivisions ? 2 : 1;\r\n for (r = 0; r < ringIdx; r++) {\r\n if (hasRings) {\r\n s += r;\r\n }\r\n if (enclose) {\r\n s += 2 * r;\r\n }\r\n\r\n for (j = 0; j <= tessellation; j++) {\r\n angle = j * angle_step;\r\n\r\n // position\r\n ringVertex.x = Math.cos(-angle) * radius;\r\n ringVertex.y = -height / 2 + h * height;\r\n ringVertex.z = Math.sin(-angle) * radius;\r\n\r\n // normal\r\n if (diameterTop === 0 && i === subdivisions) {\r\n // if no top cap, reuse former normals\r\n ringNormal.x = normals[normals.length - (tessellation + 1) * 3];\r\n ringNormal.y = normals[normals.length - (tessellation + 1) * 3 + 1];\r\n ringNormal.z = normals[normals.length - (tessellation + 1) * 3 + 2];\r\n } else {\r\n ringNormal.x = ringVertex.x;\r\n ringNormal.z = ringVertex.z;\r\n ringNormal.y = Math.sqrt(ringNormal.x * ringNormal.x + ringNormal.z * ringNormal.z) * tan;\r\n ringNormal.normalize();\r\n }\r\n\r\n // keep first ring vertex values for enclose\r\n if (j === 0) {\r\n ringFirstVertex.copyFrom(ringVertex);\r\n ringFirstNormal.copyFrom(ringNormal);\r\n }\r\n\r\n positions.push(ringVertex.x, ringVertex.y, ringVertex.z);\r\n normals.push(ringNormal.x, ringNormal.y, ringNormal.z);\r\n if (hasRings) {\r\n v = cs !== s ? faceUV[s].y : faceUV[s].w;\r\n } else {\r\n v = faceUV[s].y + (faceUV[s].w - faceUV[s].y) * h;\r\n }\r\n uvs.push(faceUV[s].x + ((faceUV[s].z - faceUV[s].x) * j) / tessellation, v);\r\n if (faceColors) {\r\n colors.push(faceColors[s].r, faceColors[s].g, faceColors[s].b, faceColors[s].a);\r\n }\r\n }\r\n\r\n // if enclose, add four vertices and their dedicated normals\r\n if (arc !== 1 && enclose) {\r\n positions.push(ringVertex.x, ringVertex.y, ringVertex.z);\r\n positions.push(0, ringVertex.y, 0);\r\n positions.push(0, ringVertex.y, 0);\r\n positions.push(ringFirstVertex.x, ringFirstVertex.y, ringFirstVertex.z);\r\n Vector3.CrossToRef(Y, ringNormal, quadNormal);\r\n quadNormal.normalize();\r\n normals.push(\r\n quadNormal.x,\r\n quadNormal.y,\r\n quadNormal.z,\r\n quadNormal.x,\r\n quadNormal.y,\r\n quadNormal.z\r\n );\r\n Vector3.CrossToRef(ringFirstNormal, Y, quadNormal);\r\n quadNormal.normalize();\r\n normals.push(\r\n quadNormal.x,\r\n quadNormal.y,\r\n quadNormal.z,\r\n quadNormal.x,\r\n quadNormal.y,\r\n quadNormal.z\r\n );\r\n if (hasRings) {\r\n v = cs !== s ? faceUV[s + 1].y : faceUV[s + 1].w;\r\n } else {\r\n v = faceUV[s + 1].y + (faceUV[s + 1].w - faceUV[s + 1].y) * h;\r\n }\r\n uvs.push(faceUV[s + 1].x, v);\r\n uvs.push(faceUV[s + 1].z, v);\r\n if (hasRings) {\r\n v = cs !== s ? faceUV[s + 2].y : faceUV[s + 2].w;\r\n } else {\r\n v = faceUV[s + 2].y + (faceUV[s + 2].w - faceUV[s + 2].y) * h;\r\n }\r\n uvs.push(faceUV[s + 2].x, v);\r\n uvs.push(faceUV[s + 2].z, v);\r\n if (faceColors) {\r\n colors.push(\r\n faceColors[s + 1].r,\r\n faceColors[s + 1].g,\r\n faceColors[s + 1].b,\r\n faceColors[s + 1].a\r\n );\r\n colors.push(\r\n faceColors[s + 1].r,\r\n faceColors[s + 1].g,\r\n faceColors[s + 1].b,\r\n faceColors[s + 1].a\r\n );\r\n colors.push(\r\n faceColors[s + 2].r,\r\n faceColors[s + 2].g,\r\n faceColors[s + 2].b,\r\n faceColors[s + 2].a\r\n );\r\n colors.push(\r\n faceColors[s + 2].r,\r\n faceColors[s + 2].g,\r\n faceColors[s + 2].b,\r\n faceColors[s + 2].a\r\n );\r\n }\r\n }\r\n if (cs !== s) {\r\n cs = s;\r\n }\r\n }\r\n }\r\n\r\n // indices\r\n const e: number = arc !== 1 && enclose ? tessellation + 4 : tessellation; // correction of number of iteration if enclose\r\n let sI: number;\r\n i = 0;\r\n for (sI = 0; sI < subdivisions; sI++) {\r\n let i0 = 0;\r\n let i1 = 0;\r\n let i2 = 0;\r\n let i3 = 0;\r\n for (j = 0; j < tessellation; j++) {\r\n i0 = i * (e + 1) + j; // top left\r\n i1 = (i + 1) * (e + 1) + j; // bottom left\r\n i2 = i * (e + 1) + (j + 1); // top right\r\n i3 = (i + 1) * (e + 1) + (j + 1); // bottom right\r\n indices.push(i0, i1, i2);\r\n indices.push(i3, i2, i1);\r\n }\r\n if (arc !== 1 && enclose) {\r\n // if enclose, add two quads\r\n indices.push(i0 + 2, i1 + 2, i2 + 2);\r\n indices.push(i3 + 2, i2 + 2, i1 + 2);\r\n indices.push(i0 + 4, i1 + 4, i2 + 4);\r\n indices.push(i3 + 4, i2 + 4, i1 + 4);\r\n } else if (arc === 1) {\r\n indices.push(i2, i3, i * (e + 1));\r\n indices.push((i + 1) * (e + 1), i * (e + 1), i3);\r\n }\r\n i = hasRings ? i + 2 : i + 1;\r\n }\r\n\r\n // Caps\r\n const createCylinderCap = (isTop: boolean) => {\r\n const radius = isTop ? diameterTop / 2 : diameterBottom / 2;\r\n if (radius === 0) {\r\n return;\r\n }\r\n\r\n // Cap positions, normals & uvs\r\n let angle;\r\n let circleVector;\r\n let i: number;\r\n const u: Vector4 = isTop ? faceUV[surfaceNb - 1] : faceUV[0];\r\n let c: Nullable = null;\r\n if (faceColors) {\r\n c = isTop ? faceColors[surfaceNb - 1] : faceColors[0];\r\n }\r\n // cap center\r\n const vbase = positions.length / 3;\r\n const offset = isTop ? height / 2 : -height / 2;\r\n const center = new Vector3(0, offset, 0);\r\n positions.push(center.x, center.y, center.z);\r\n normals.push(0, isTop ? 1 : -1, 0);\r\n uvs.push(u.x + (u.z - u.x) * 0.5, u.y + (u.w - u.y) * 0.5);\r\n if (c) {\r\n colors.push(c.r, c.g, c.b, c.a);\r\n }\r\n\r\n const textureScale = new Vector2(0.5, 0.5);\r\n for (i = 0; i <= tessellation; i++) {\r\n angle = i * angle_step;\r\n const cos = Math.cos(-angle);\r\n const sin = Math.sin(-angle);\r\n circleVector = new Vector3(cos * radius, offset, sin * radius);\r\n const textureCoordinate = new Vector2(cos * textureScale.x + 0.5, sin * textureScale.y + 0.5);\r\n positions.push(circleVector.x, circleVector.y, circleVector.z);\r\n normals.push(0, isTop ? 1 : -1, 0);\r\n uvs.push(u.x + (u.z - u.x) * textureCoordinate.x, u.y + (u.w - u.y) * textureCoordinate.y);\r\n if (c) {\r\n colors.push(c.r, c.g, c.b, c.a);\r\n }\r\n }\r\n // Cap indices\r\n for (i = 0; i < tessellation; i++) {\r\n if (!isTop) {\r\n indices.push(vbase);\r\n indices.push(vbase + (i + 1));\r\n indices.push(vbase + (i + 2));\r\n } else {\r\n indices.push(vbase);\r\n indices.push(vbase + (i + 2));\r\n indices.push(vbase + (i + 1));\r\n }\r\n }\r\n // Close the loop\r\n if (arc === 1) {\r\n if (!isTop) {\r\n indices.push(vbase, vbase + tessellation + 1, vbase + 1);\r\n } else {\r\n indices.push(vbase, vbase + 1, vbase + tessellation + 1);\r\n }\r\n }\r\n };\r\n\r\n // add caps to geometry based on cap parameter\r\n if (cap === Mesh.CAP_START || cap === Mesh.CAP_ALL) {\r\n createCylinderCap(false);\r\n }\r\n if (cap === Mesh.CAP_END || cap === Mesh.CAP_ALL) {\r\n createCylinderCap(true);\r\n }\r\n\r\n // Sides\r\n VertexData._ComputeSides(\r\n sideOrientation,\r\n positions,\r\n indices,\r\n normals,\r\n uvs,\r\n options.frontUVs,\r\n options.backUVs\r\n );\r\n\r\n const vertexData = new VertexData();\r\n\r\n vertexData.indices = indices;\r\n vertexData.positions = positions;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n if (faceColors) {\r\n vertexData.colors = colors;\r\n }\r\n\r\n return vertexData;\r\n }\r\n }\r\n}\r\n","import { Matrix, Vector2, Vector3, Vector4 } from '@babylonjs/core/Maths';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { VertexData } from '@babylonjs/core/Meshes/mesh.vertexData';\r\n\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n */\r\nexport class TorusBuilder {\r\n /**\r\n * Creates a torus mesh\r\n * * The parameter `diameter` sets the diameter size (float) of the torus (default 1)\r\n * * The parameter `thickness` sets the diameter size of the tube of the torus (float, default 0.5)\r\n * * The parameter `tessellation` sets the number of torus sides (postive integer, default 16)\r\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/babylon101/discover_basic_elements#side-orientation\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param scene defines the hosting scene\r\n * @returns the torus mesh\r\n * @see https://doc.babylonjs.com/how_to/set_shapes#torus\r\n */\r\n public static CreateTorus(\r\n name: string,\r\n options: {\r\n diameter?: number;\r\n thickness?: number;\r\n tessellation?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n arc?: number;\r\n slice?: number;\r\n updatable?: boolean;\r\n sliceStart?: number;\r\n },\r\n scene: any\r\n ): Mesh {\r\n const torus = new Mesh(name, scene);\r\n\r\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n torus._originalBuilderSideOrientation = options.sideOrientation;\r\n\r\n const vertexData = createTorus(options);\r\n\r\n vertexData.applyToMesh(torus, options.updatable);\r\n\r\n return torus;\r\n\r\n function createTorus(options: {\r\n diameter?: number;\r\n thickness?: number;\r\n tessellation?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n arc?: number;\r\n slice?: number;\r\n sliceStart?: number;\r\n }) {\r\n const indices = [];\r\n const positions = [];\r\n const normals = [];\r\n const uvs = [];\r\n\r\n const diameter = options.diameter || 1;\r\n const thickness = options.thickness || 0.5;\r\n const tessellation = options.tessellation || 12;\r\n const arc = options.arc && (options.arc <= 0 || options.arc > 1) ? 1.0 : options.arc || 1.0;\r\n let slice = options.slice && options.slice <= 0 ? 1.0 : options.slice || 1.0;\r\n const sliceStart =\r\n options.sliceStart && (options.sliceStart < 0 || options.sliceStart > 1) ? 0 : options.sliceStart || 0;\r\n const sideOrientation =\r\n options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\r\n\r\n if (slice + sliceStart > 1) {\r\n slice = 1 - sliceStart;\r\n }\r\n\r\n const outerSegments = Math.ceil(tessellation * arc * Math.PI); // Need more segments for the torus circumference\r\n const innerSegments = Math.ceil(tessellation * slice);\r\n const closedRing = arc === 1; // handle float errors\r\n const closedTube = slice === 1;\r\n const lastOuterRing = outerSegments + 1;\r\n const lastTubeSection = innerSegments + 1;\r\n\r\n for (let i = 0; i < lastOuterRing; i++) {\r\n const u = i / outerSegments;\r\n\r\n const outerAngle = u * (arc * Math.PI * 2) - Math.PI / 2;\r\n\r\n const transform = Matrix.Translation(diameter / 2, 0, 0).multiply(Matrix.RotationY(outerAngle));\r\n\r\n for (let j = 0; j < lastTubeSection; j++) {\r\n const v = 1 - j / innerSegments;\r\n\r\n const innerAngle = (j / innerSegments) * (Math.PI * 2 * slice) + Math.PI * 2 * sliceStart;\r\n const dx = Math.cos(innerAngle);\r\n const dy = Math.sin(innerAngle);\r\n\r\n // Create a vertex.\r\n let normal = new Vector3(dx, dy, 0);\r\n let position = normal.scale(thickness / 2);\r\n const textureCoordinate = new Vector2(u, v);\r\n\r\n position = Vector3.TransformCoordinates(position, transform);\r\n normal = Vector3.TransformNormal(normal, transform);\r\n\r\n positions.push(position.x, position.y, position.z);\r\n normals.push(normal.x, normal.y, normal.z);\r\n uvs.push(textureCoordinate.x, textureCoordinate.y);\r\n\r\n if ((i < outerSegments || closedRing) && (j < innerSegments || closedTube)) {\r\n // And create indices for two triangles.\r\n const nextI = (i + 1) % lastOuterRing;\r\n const nextJ = (j + 1) % lastTubeSection;\r\n\r\n indices.push(i * lastTubeSection + j); // A\r\n indices.push(i * lastTubeSection + nextJ); // B\r\n indices.push(nextI * lastTubeSection + j); // C\r\n\r\n indices.push(i * lastTubeSection + nextJ); // B\r\n indices.push(nextI * lastTubeSection + nextJ); // D\r\n indices.push(nextI * lastTubeSection + j); // C\r\n }\r\n }\r\n }\r\n\r\n // Sides\r\n VertexData._ComputeSides(\r\n sideOrientation,\r\n positions,\r\n indices,\r\n normals,\r\n uvs,\r\n options.frontUVs,\r\n options.backUVs\r\n );\r\n\r\n // Result\r\n const vertexData = new VertexData();\r\n\r\n vertexData.indices = indices;\r\n vertexData.positions = positions;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n\r\n return vertexData;\r\n }\r\n }\r\n}\r\n","import { Vector4 } from '@babylonjs/core/Maths/math.vector';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { VertexData } from '@babylonjs/core/Meshes/mesh.vertexData';\r\n\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n */\r\nexport class PlaneBuilder {\r\n /**\r\n * Creates a torus mesh\r\n * * The parameter `diameter` sets the diameter size (float) of the torus (default 1)\r\n * * The parameter `thickness` sets the diameter size of the tube of the torus (float, default 0.5)\r\n * * The parameter `tessellation` sets the number of torus sides (postive integer, default 16)\r\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/babylon101/discover_basic_elements#side-orientation\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param scene defines the hosting scene\r\n * @returns the torus mesh\r\n * @see https://doc.babylonjs.com/how_to/set_shapes#torus\r\n */\r\n public static CreatePlane(\r\n name: string,\r\n options: {\r\n pattern?: number;\r\n tileSize?: number;\r\n tileWidth?: number;\r\n tileHeight?: number;\r\n size?: number;\r\n width?: number;\r\n height?: number;\r\n alignHorizontal?: number;\r\n alignVertical?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n updatable?: boolean;\r\n },\r\n scene: any\r\n ): Mesh {\r\n const plane = new Mesh(name, scene);\r\n\r\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n\r\n const flipTile = options.pattern || Mesh.NO_FLIP;\r\n const tileWidth = options.tileWidth || options.tileSize || 1;\r\n const tileHeight = options.tileHeight || options.tileSize || 1;\r\n const alignH = options.alignHorizontal || 0;\r\n const alignV = options.alignVertical || 0;\r\n\r\n const width = options.width || options.size || 1;\r\n const tilesX = Math.floor(width / tileWidth);\r\n let offsetX = width - tilesX * tileWidth;\r\n\r\n const height = options.height || options.size || 1;\r\n const tilesY = Math.floor(height / tileHeight);\r\n let offsetY = height - tilesY * tileHeight;\r\n\r\n const halfWidth = (tileWidth * tilesX) / 2;\r\n const halfHeight = (tileHeight * tilesY) / 2;\r\n\r\n const indices = [];\r\n const positions = [];\r\n const normals = new Float32Array(tilesX * tilesY * 12);\r\n\r\n const colorSetArray = [1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1];\r\n const normalSetArray = [0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1];\r\n const colorSet = Float32Array.from(colorSetArray);\r\n const normalSet = Float32Array.from(normalSetArray);\r\n\r\n let adjustX = 0;\r\n let adjustY = 0;\r\n let startX = 0;\r\n let startY = 0;\r\n let endX = 0;\r\n let endY = 0;\r\n\r\n //Part Tiles\r\n if (offsetX > 0 || offsetY > 0) {\r\n startX = -halfWidth;\r\n startY = -halfHeight;\r\n endX = halfWidth;\r\n endY = halfHeight;\r\n\r\n switch (alignH) {\r\n case Mesh.CENTER:\r\n offsetX /= 2;\r\n startX -= offsetX;\r\n endX += offsetX;\r\n break;\r\n case Mesh.LEFT:\r\n endX += offsetX;\r\n adjustX = -offsetX / 2;\r\n break;\r\n case Mesh.RIGHT:\r\n startX -= offsetX;\r\n adjustX = offsetX / 2;\r\n break;\r\n }\r\n\r\n switch (alignV) {\r\n case Mesh.CENTER:\r\n offsetY /= 2;\r\n startY -= offsetY;\r\n endY += offsetY;\r\n break;\r\n case Mesh.BOTTOM:\r\n endY += offsetY;\r\n adjustY = -offsetY / 2;\r\n break;\r\n case Mesh.TOP:\r\n startY -= offsetY;\r\n adjustY = offsetY / 2;\r\n break;\r\n }\r\n }\r\n\r\n const uvBase = [];\r\n uvBase[0] = [0, 0, 1, 0, 1, 1, 0, 1];\r\n uvBase[1] = [0, 0, 1, 0, 1, 1, 0, 1];\r\n if (flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.ROTATE_ROW) {\r\n uvBase[1] = [1, 1, 0, 1, 0, 0, 1, 0];\r\n }\r\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.FLIP_ROW) {\r\n uvBase[1] = [1, 0, 0, 0, 0, 1, 1, 1];\r\n }\r\n if (flipTile === Mesh.FLIP_N_ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\r\n uvBase[1] = [0, 1, 1, 1, 1, 0, 0, 0];\r\n }\r\n\r\n const uvs = new Float32Array(tilesY * tilesX * 8);\r\n const colors = new Float32Array(tilesY * tilesX * 16);\r\n let index = 0;\r\n\r\n for (let y = 0; y < tilesY; y++) {\r\n for (let x = 0; x < tilesX; x++) {\r\n positions.push(-halfWidth + x * tileWidth + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\r\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\r\n positions.push(\r\n -halfWidth + (x + 1) * tileWidth + adjustX,\r\n -halfHeight + (y + 1) * tileHeight + adjustY,\r\n 0\r\n );\r\n positions.push(-halfWidth + x * tileWidth + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\r\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\r\n if (\r\n flipTile === Mesh.FLIP_TILE ||\r\n flipTile === Mesh.ROTATE_TILE ||\r\n flipTile === Mesh.FLIP_N_ROTATE_TILE\r\n ) {\r\n uvs.set(uvBase[((x % 2) + (y % 2)) % 2], index * 2);\r\n } else if (\r\n flipTile === Mesh.FLIP_ROW ||\r\n flipTile === Mesh.ROTATE_ROW ||\r\n flipTile === Mesh.FLIP_N_ROTATE_ROW\r\n ) {\r\n uvs.set(uvBase[y % 2], index * 2);\r\n } else {\r\n uvs.set(uvBase[0], index * 2);\r\n }\r\n\r\n colors.set(colorSet, index * 4);\r\n normals.set(normalSet, index * 3);\r\n index += 4;\r\n }\r\n }\r\n\r\n //Part Tiles\r\n if (offsetX > 0 || offsetY > 0) {\r\n const partialBottomRow: boolean = offsetY > 0 && (alignV === Mesh.CENTER || alignV === Mesh.TOP);\r\n const partialTopRow: boolean = offsetY > 0 && (alignV === Mesh.CENTER || alignV === Mesh.BOTTOM);\r\n const partialLeftCol: boolean = offsetX > 0 && (alignH === Mesh.CENTER || alignH === Mesh.RIGHT);\r\n const partialRightCol: boolean = offsetX > 0 && (alignH === Mesh.CENTER || alignH === Mesh.LEFT);\r\n let uvPart: number[] = [];\r\n let a, b, c, d: number;\r\n\r\n //corners\r\n if (partialBottomRow && partialLeftCol) {\r\n //bottom left corner\r\n positions.push(startX + adjustX, startY + adjustY, 0);\r\n positions.push(-halfWidth + adjustX, startY + adjustY, 0);\r\n positions.push(-halfWidth + adjustX, startY + offsetY + adjustY, 0);\r\n positions.push(startX + adjustX, startY + offsetY + adjustY, 0);\r\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\r\n index += 4;\r\n a = 1 - offsetX / tileWidth;\r\n b = 1 - offsetY / tileHeight;\r\n c = 1;\r\n d = 1;\r\n uvPart = [a, b, c, b, c, d, a, d];\r\n if (flipTile === Mesh.ROTATE_ROW) {\r\n uvPart = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\r\n }\r\n if (flipTile === Mesh.FLIP_ROW) {\r\n uvPart = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\r\n }\r\n if (flipTile === Mesh.FLIP_N_ROTATE_ROW) {\r\n uvPart = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\r\n }\r\n uvs.set(uvPart, index * 2);\r\n uvs.set(uvPart, index);\r\n\r\n colors.set(colorSet, index * 4);\r\n normals.set(normalSet, index * 3);\r\n }\r\n\r\n if (partialBottomRow && partialRightCol) {\r\n //bottom right corner\r\n positions.push(halfWidth + adjustX, startY + adjustY, 0);\r\n positions.push(endX + adjustX, startY + adjustY, 0);\r\n positions.push(endX + adjustX, startY + offsetY + adjustY, 0);\r\n positions.push(halfWidth + adjustX, startY + offsetY + adjustY, 0);\r\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\r\n index += 4;\r\n a = 0;\r\n b = 1 - offsetY / tileHeight;\r\n c = offsetX / tileWidth;\r\n d = 1;\r\n uvPart = [a, b, c, b, c, d, a, d];\r\n if (flipTile === Mesh.ROTATE_ROW || (flipTile === Mesh.ROTATE_TILE && tilesX % 2 === 0)) {\r\n uvPart = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\r\n }\r\n if (flipTile === Mesh.FLIP_ROW || (flipTile === Mesh.FLIP_TILE && tilesX % 2 === 0)) {\r\n uvPart = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\r\n }\r\n if (flipTile === Mesh.FLIP_N_ROTATE_ROW || (flipTile === Mesh.FLIP_N_ROTATE_TILE && tilesX % 2 === 0)) {\r\n uvPart = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\r\n }\r\n uvs.set(uvPart, index * 2);\r\n\r\n colors.set(colorSet, index * 4);\r\n normals.set(normalSet, index * 3);\r\n }\r\n\r\n if (partialTopRow && partialLeftCol) {\r\n //top left corner\r\n positions.push(startX + adjustX, halfHeight + adjustY, 0);\r\n positions.push(-halfWidth + adjustX, halfHeight + adjustY, 0);\r\n positions.push(-halfWidth + adjustX, endY + adjustY, 0);\r\n positions.push(startX + adjustX, endY + adjustY, 0);\r\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\r\n index += 4;\r\n a = 1 - offsetX / tileWidth;\r\n b = 0;\r\n c = 1;\r\n d = offsetY / tileHeight;\r\n uvPart = [a, b, c, b, c, d, a, d];\r\n if (\r\n (flipTile === Mesh.ROTATE_ROW && tilesY % 2 === 1) ||\r\n (flipTile === Mesh.ROTATE_TILE && tilesY % 1 === 0)\r\n ) {\r\n uvPart = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\r\n }\r\n if (\r\n (flipTile === Mesh.FLIP_ROW && tilesY % 2 === 1) ||\r\n (flipTile === Mesh.FLIP_TILE && tilesY % 2 === 0)\r\n ) {\r\n uvPart = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\r\n }\r\n if (\r\n (flipTile === Mesh.FLIP_N_ROTATE_ROW && tilesY % 2 === 1) ||\r\n (flipTile === Mesh.FLIP_N_ROTATE_TILE && tilesY % 2 === 0)\r\n ) {\r\n uvPart = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\r\n }\r\n\r\n uvs.set(uvPart, index * 2);\r\n colors.set(colorSet, index * 4);\r\n normals.set(normalSet, index * 3);\r\n }\r\n\r\n if (partialTopRow && partialRightCol) {\r\n //top right corner\r\n positions.push(halfWidth + adjustX, halfHeight + adjustY, 0);\r\n positions.push(endX + adjustX, halfHeight + adjustY, 0);\r\n positions.push(endX + adjustX, endY + adjustY, 0);\r\n positions.push(halfWidth + adjustX, endY + adjustY, 0);\r\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\r\n index += 4;\r\n a = 0;\r\n b = 0;\r\n c = offsetX / tileWidth;\r\n d = offsetY / tileHeight;\r\n uvPart = [a, b, c, b, c, d, a, d];\r\n if (\r\n (flipTile === Mesh.ROTATE_ROW && tilesY % 2 === 1) ||\r\n (flipTile === Mesh.ROTATE_TILE && (tilesY + tilesX) % 2 === 1)\r\n ) {\r\n uvPart = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\r\n }\r\n if (\r\n (flipTile === Mesh.FLIP_ROW && tilesY % 2 === 1) ||\r\n (flipTile === Mesh.FLIP_TILE && (tilesY + tilesX) % 2 === 1)\r\n ) {\r\n uvPart = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\r\n }\r\n if (\r\n (flipTile === Mesh.FLIP_N_ROTATE_ROW && tilesY % 2 === 1) ||\r\n (flipTile === Mesh.FLIP_N_ROTATE_TILE && (tilesY + tilesX) % 2 === 1)\r\n ) {\r\n uvPart = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\r\n }\r\n uvs.set(uvPart, index * 2);\r\n colors.set(colorSet, index * 4);\r\n normals.set(normalSet, index * 3);\r\n }\r\n\r\n //part rows\r\n if (partialBottomRow) {\r\n const uvBaseBR = [];\r\n a = 0;\r\n b = 1 - offsetY / tileHeight;\r\n c = 1;\r\n d = 1;\r\n uvBaseBR[0] = [a, b, c, b, c, d, a, d];\r\n uvBaseBR[1] = [a, b, c, b, c, d, a, d];\r\n if (flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.ROTATE_ROW) {\r\n uvBaseBR[1] = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\r\n }\r\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.FLIP_ROW) {\r\n uvBaseBR[1] = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\r\n }\r\n if (flipTile === Mesh.FLIP_N_ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\r\n uvBaseBR[1] = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\r\n }\r\n for (let x = 0; x < tilesX; x++) {\r\n positions.push(-halfWidth + x * tileWidth + adjustX, startY + adjustY, 0);\r\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, startY + adjustY, 0);\r\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, startY + offsetY + adjustY, 0);\r\n positions.push(-halfWidth + x * tileWidth + adjustX, startY + offsetY + adjustY, 0);\r\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\r\n index += 4;\r\n if (\r\n flipTile === Mesh.FLIP_TILE ||\r\n flipTile === Mesh.ROTATE_TILE ||\r\n flipTile === Mesh.FLIP_N_ROTATE_TILE\r\n ) {\r\n uvs.set(uvBaseBR[(x + 1) % 2], index * 2);\r\n } else if (\r\n flipTile === Mesh.FLIP_ROW ||\r\n flipTile === Mesh.ROTATE_ROW ||\r\n flipTile === Mesh.FLIP_N_ROTATE_ROW\r\n ) {\r\n uvs.set(uvBaseBR[1], index * 2);\r\n } else {\r\n uvs.set(uvBaseBR[0], index * 2);\r\n }\r\n\r\n colors.set(colorSet, index * 4);\r\n normals.set(normalSet, index * 3);\r\n }\r\n }\r\n\r\n if (partialTopRow) {\r\n const uvBaseTR = [];\r\n a = 0;\r\n b = 0;\r\n c = 1;\r\n d = offsetY / tileHeight;\r\n uvBaseTR[0] = [a, b, c, b, c, d, a, d];\r\n uvBaseTR[1] = [a, b, c, b, c, d, a, d];\r\n if (flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.ROTATE_ROW) {\r\n uvBaseTR[1] = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\r\n }\r\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.FLIP_ROW) {\r\n uvBaseTR[1] = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\r\n }\r\n if (flipTile === Mesh.FLIP_N_ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\r\n uvBaseTR[1] = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\r\n }\r\n for (let x = 0; x < tilesX; x++) {\r\n positions.push(-halfWidth + x * tileWidth + adjustX, endY - offsetY + adjustY, 0);\r\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, endY - offsetY + adjustY, 0);\r\n positions.push(-halfWidth + (x + 1) * tileWidth + adjustX, endY + adjustY, 0);\r\n positions.push(-halfWidth + x * tileWidth + adjustX, endY + adjustY, 0);\r\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\r\n index += 4;\r\n if (\r\n flipTile === Mesh.FLIP_TILE ||\r\n flipTile === Mesh.ROTATE_TILE ||\r\n flipTile === Mesh.FLIP_N_ROTATE_TILE\r\n ) {\r\n uvs.set(uvBaseTR[(x + tilesY) % 2], index * 2);\r\n } else if (\r\n flipTile === Mesh.FLIP_ROW ||\r\n flipTile === Mesh.ROTATE_ROW ||\r\n flipTile === Mesh.FLIP_N_ROTATE_ROW\r\n ) {\r\n uvs.set(uvBaseTR[tilesY % 2], index * 2);\r\n } else {\r\n uvs.set(uvBaseTR[0], index * 2);\r\n }\r\n colors.set(colorSet, index * 4);\r\n normals.set(normalSet, index * 3);\r\n }\r\n }\r\n\r\n if (partialLeftCol) {\r\n const uvBaseLC = [];\r\n a = 1 - offsetX / tileWidth;\r\n b = 0;\r\n c = 1;\r\n d = 1;\r\n uvBaseLC[0] = [a, b, c, b, c, d, a, d];\r\n uvBaseLC[1] = [a, b, c, b, c, d, a, d];\r\n if (flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.ROTATE_ROW) {\r\n uvBaseLC[1] = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\r\n }\r\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.FLIP_ROW) {\r\n uvBaseLC[1] = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\r\n }\r\n if (flipTile === Mesh.FLIP_N_ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\r\n uvBaseLC[1] = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\r\n }\r\n for (let y = 0; y < tilesY; y++) {\r\n positions.push(startX + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\r\n positions.push(startX + offsetX + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\r\n positions.push(startX + offsetX + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\r\n positions.push(startX + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\r\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\r\n index += 4;\r\n if (\r\n flipTile === Mesh.FLIP_TILE ||\r\n flipTile === Mesh.ROTATE_TILE ||\r\n flipTile === Mesh.FLIP_N_ROTATE_TILE\r\n ) {\r\n uvs.set(uvBaseLC[(y + 1) % 2], index * 2);\r\n } else if (\r\n flipTile === Mesh.FLIP_ROW ||\r\n flipTile === Mesh.ROTATE_ROW ||\r\n flipTile === Mesh.FLIP_N_ROTATE_ROW\r\n ) {\r\n uvs.set(uvBaseLC[y % 2], index * 2);\r\n } else {\r\n uvs.set(uvBaseLC[0], index * 2);\r\n }\r\n colors.set(colorSet, index * 4);\r\n normals.set(normalSet, index * 3);\r\n }\r\n }\r\n\r\n if (partialRightCol) {\r\n const uvBaseRC = [];\r\n a = 0;\r\n b = 0;\r\n c = offsetX / tileHeight;\r\n d = 1;\r\n uvBaseRC[0] = [a, b, c, b, c, d, a, d];\r\n uvBaseRC[1] = [a, b, c, b, c, d, a, d];\r\n if (flipTile === Mesh.ROTATE_TILE || flipTile === Mesh.ROTATE_ROW) {\r\n uvBaseRC[1] = [1 - a, 1 - b, 1 - c, 1 - b, 1 - c, 1 - d, 1 - a, 1 - d];\r\n }\r\n if (flipTile === Mesh.FLIP_TILE || flipTile === Mesh.FLIP_ROW) {\r\n uvBaseRC[1] = [1 - a, b, 1 - c, b, 1 - c, d, 1 - a, d];\r\n }\r\n if (flipTile === Mesh.FLIP_N_ROTATE_TILE || flipTile === Mesh.FLIP_N_ROTATE_ROW) {\r\n uvBaseRC[1] = [a, 1 - b, c, 1 - b, c, 1 - d, a, 1 - d];\r\n }\r\n for (let y = 0; y < tilesY; y++) {\r\n positions.push(endX - offsetX + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\r\n positions.push(endX + adjustX, -halfHeight + y * tileHeight + adjustY, 0);\r\n positions.push(endX + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\r\n positions.push(endX - offsetX + adjustX, -halfHeight + (y + 1) * tileHeight + adjustY, 0);\r\n indices.push(index, index + 1, index + 3, index + 1, index + 2, index + 3);\r\n index += 4;\r\n if (\r\n flipTile === Mesh.FLIP_TILE ||\r\n flipTile === Mesh.ROTATE_TILE ||\r\n flipTile === Mesh.FLIP_N_ROTATE_TILE\r\n ) {\r\n uvs.set(uvBaseRC[(y + tilesX) % 2], index * 2);\r\n } else if (\r\n flipTile === Mesh.FLIP_ROW ||\r\n flipTile === Mesh.ROTATE_ROW ||\r\n flipTile === Mesh.FLIP_N_ROTATE_ROW\r\n ) {\r\n uvs.set(uvBaseRC[y % 2], index * 2);\r\n } else {\r\n uvs.set(uvBaseRC[0], index * 2);\r\n }\r\n colors.set(colorSet, index * 4);\r\n normals.set(normalSet, index * 3);\r\n }\r\n }\r\n }\r\n\r\n //reverse V values since old PlaneBuilder built them that way\r\n for (let i = 0; i < uvs.length / 2; i++) {\r\n uvs[i * 2 + 1] = 1 - uvs[i * 2 + 1];\r\n }\r\n\r\n const sideOrientation = options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\r\n\r\n // sides\r\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\r\n\r\n // Result\r\n const vertexData = new VertexData();\r\n\r\n vertexData.indices = indices;\r\n vertexData.positions = positions;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n\r\n vertexData.applyToMesh(plane, options.updatable);\r\n\r\n return plane;\r\n }\r\n}\r\n","import { Vector4 } from '@babylonjs/core/Maths';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { VertexData } from '@babylonjs/core/Meshes/mesh.vertexData';\r\n\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n */\r\nexport class DiscBuilder {\r\n /**\r\n * Creates a torus mesh\r\n * * The parameter `diameter` sets the diameter size (float) of the torus (default 1)\r\n * * The parameter `thickness` sets the diameter size of the tube of the torus (float, default 0.5)\r\n * * The parameter `tessellation` sets the number of torus sides (postive integer, default 16)\r\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/babylon101/discover_basic_elements#side-orientation\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created.\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param scene defines the hosting scene\r\n * @returns the torus mesh\r\n * @see https://doc.babylonjs.com/how_to/set_shapes#torus\r\n */\r\n public static CreateDisc(\r\n name: string,\r\n options: {\r\n radius?: number;\r\n tessellation?: number;\r\n arc?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n updatable?: boolean;\r\n },\r\n scene: any\r\n ): Mesh {\r\n const disc = new Mesh(name, scene);\r\n\r\n options.sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n disc._originalBuilderSideOrientation = options.sideOrientation;\r\n\r\n const vertexData = createDisc(options);\r\n\r\n vertexData.applyToMesh(disc, options.updatable);\r\n\r\n return disc;\r\n\r\n function createDisc(options: {\r\n radius?: number;\r\n tessellation?: number;\r\n arc?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n }) {\r\n const positions = new Array();\r\n const indices = new Array();\r\n const normals = new Array();\r\n const uvs = new Array();\r\n\r\n const radius = options.radius || 0.5;\r\n const tessellation = options.tessellation || 64;\r\n const arc: number = options.arc && (options.arc <= 0 || options.arc > 1) ? 1.0 : options.arc || 1.0;\r\n const sideOrientation =\r\n options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\r\n\r\n // positions and uvs\r\n positions.push(0, 0, 0); // disc center first\r\n uvs.push(0.5, 0.5);\r\n\r\n const theta = Math.PI * 2 * arc;\r\n const step = theta / tessellation;\r\n for (let a = 0; arc === 1 ? a < theta : a <= theta; a += step) {\r\n const x = Math.cos(a);\r\n const y = Math.sin(a);\r\n const u = (x + 1) / 2;\r\n const v = (1 - y) / 2;\r\n positions.push(radius * x, radius * y, 0);\r\n uvs.push(u, v);\r\n }\r\n\r\n //indices\r\n const vertexNb = positions.length / 3;\r\n for (let i = 1; i < vertexNb - 1; i++) {\r\n indices.push(i + 1, 0, i);\r\n }\r\n\r\n if (arc === 1) {\r\n indices.push(1, 0, vertexNb - 1); // close the circle\r\n }\r\n\r\n // result\r\n VertexData.ComputeNormals(positions, indices, normals);\r\n VertexData._ComputeSides(\r\n sideOrientation,\r\n positions,\r\n indices,\r\n normals,\r\n uvs,\r\n options.frontUVs,\r\n options.backUVs\r\n );\r\n\r\n const vertexData = new VertexData();\r\n\r\n vertexData.indices = indices;\r\n vertexData.positions = positions;\r\n vertexData.normals = normals;\r\n vertexData.uvs = uvs;\r\n\r\n return vertexData;\r\n }\r\n }\r\n}\r\n","import { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { SketchPath } from '@models/classes/meshes/sketch-node-path';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass } from '@models/interfaces';\r\nimport { KbViewer } from '..';\r\nimport { Renderer } from './renderer';\r\n\r\nexport class PathRenderer extends Renderer {\r\n constructor(protected viewer: KbViewer, protected scene: Scene) {\r\n super(viewer, scene);\r\n }\r\n\r\n modelToken = Token.SketchPath;\r\n\r\n update3dInternal(node: ITrackedClass) {}\r\n}\r\n","import { Ray } from '@babylonjs/core/Culling/ray';\r\nimport { Axis, Color4, Matrix, Plane, Quaternion, Vector3 } from '@babylonjs/core/Maths';\r\nimport { LinesMesh } from '@babylonjs/core/Meshes/linesMesh';\r\nimport { MeshBuilder } from '@babylonjs/core/Meshes/meshBuilder';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { DeepImmutableObject } from '@babylonjs/core/types';\r\nimport { eMateType } from '@models/classes/enums';\r\nimport { MateNode } from '@models/classes/mates/mate-node';\r\nimport { rotationToVectorToRef, toRadians } from './math-helper';\r\nimport { updateMatrixOfAncestorChain } from './scene-helper';\r\n\r\nconst ENABLE_DEBUG_VISUAL = false;\r\n\r\n// function limitFPE(val: number) {\r\n// return Math.round((val + Number.EPSILON) * 100000000) / 100000000;\r\n// }\r\n\r\n// function roundQuaterion(q: Quaternion) {\r\n// return new Quaternion(limitFPE(q.x), limitFPE(q.y), limitFPE(q.z), limitFPE(q.w));\r\n// }\r\nexport function mateControlPointToConnector(\r\n mate: Partial & Pick,\r\n c1Mesh: TransformNode,\r\n controlPointMesh: TransformNode\r\n) {\r\n let grandParentContainerWorldMatrixInv: DeepImmutableObject;\r\n if (controlPointMesh.parent) {\r\n grandParentContainerWorldMatrixInv = controlPointMesh.parent.computeWorldMatrix(true).clone().invert();\r\n } else {\r\n grandParentContainerWorldMatrixInv = Matrix.IdentityReadOnly;\r\n }\r\n\r\n const targetTransformMatrix = c1Mesh.computeWorldMatrix().clone();\r\n targetTransformMatrix.multiplyToRef(grandParentContainerWorldMatrixInv, targetTransformMatrix);\r\n const position = new Vector3();\r\n const direction = new Vector3();\r\n targetTransformMatrix.decompose(undefined, undefined, position);\r\n\r\n Vector3.TransformNormalFromFloatsToRef(-1.0, 0, 0, targetTransformMatrix, direction);\r\n\r\n if (!mate.flip) {\r\n direction.scaleInPlace(-1);\r\n }\r\n direction.normalize();\r\n\r\n return {\r\n position,\r\n direction,\r\n };\r\n}\r\n\r\nconst flipMatrix = new Matrix();\r\nQuaternion.RotationAxis(Axis.Y, toRadians(180)).toRotationMatrix(flipMatrix);\r\n\r\nconst debugViz = new Map();\r\n\r\nexport function mateConnectors(\r\n mate: Partial & Pick,\r\n followerMesh: TransformNode,\r\n c1Mesh: TransformNode,\r\n c2Mesh: TransformNode\r\n) {\r\n updateMatrixOfAncestorChain(followerMesh, c2Mesh);\r\n c1Mesh.computeWorldMatrix(true);\r\n let mateOffset = Vector3.Zero();\r\n let worldDelta = Vector3.Zero(); //the translation that c2 would need to make to get to it's new position\r\n\r\n if (mate.mateType == eMateType.ball) {\r\n /**\r\n * Ball Mate\r\n * -move c2 until it's in the same position as c1\r\n * In the future this could be made more intelligent by rotating a node to satisfy this mate and another ball or slider mate\r\n */\r\n worldDelta = c1Mesh.getAbsolutePosition().subtract(c2Mesh.getAbsolutePosition());\r\n } else {\r\n /**\r\n * Aligning direction vectors\r\n * ------------------------------------------------\r\n * Easiest way to think about this:\r\n * Connectors have a direction vector, and an angle of rotation around the direction vector.\r\n * The connector meshes are rotated by the renderer such that the positive X axis is pointing\r\n * in the direction of the direction vector. And the angle vector will be pointing in the direction\r\n * of the y axis.\r\n *\r\n * We want the world transformation matrix of the second connector to match the world transform of the first connector (optionally flipped).\r\n * The math for calculating the world matrix of a connector can be written as:\r\n * * * \r\n * We are trying to solve for the local transform matrix of the connector parent so the full world transform matches the transform of the first connector,\r\n * so we can rewrite the equation as:\r\n * = * * \r\n *\r\n * Then we apply the result matrix to the connector parent.\r\n */\r\n\r\n const t = new Vector3();\r\n const r = new Quaternion();\r\n const s = new Vector3();\r\n c1Mesh.computeWorldMatrix(true).decompose(s, r, t);\r\n if (mate.offsetAngle && mate.offsetAngle !== 0) {\r\n const angleOffsetQ = Quaternion.RotationAxis(Axis.X, (mate.offsetAngle * Math.PI) / 180);\r\n r.multiplyInPlace(angleOffsetQ);\r\n }\r\n const targetWorldTransformMatrix = Matrix.Compose(followerMesh.scaling, r, t);\r\n const currentWorldTransformMatrix = c2Mesh.computeWorldMatrix(true).clone();\r\n\r\n if (!mate.flip) {\r\n flipMatrix.multiplyToRef(targetWorldTransformMatrix, targetWorldTransformMatrix);\r\n }\r\n const c2LocalTransform = Matrix.Compose(c2Mesh.scaling, c2Mesh.rotationQuaternion!, c2Mesh.position);\r\n\r\n let grandParentContainerWorldMatrix: DeepImmutableObject;\r\n if (followerMesh.parent) {\r\n grandParentContainerWorldMatrix = followerMesh.parent.computeWorldMatrix(true).clone();\r\n } else {\r\n grandParentContainerWorldMatrix = Matrix.IdentityReadOnly;\r\n }\r\n const grandParentContainerWorldMatrixInv = grandParentContainerWorldMatrix.clone().invert();\r\n\r\n let calculatedTargetWorldTransform: Matrix | undefined;\r\n\r\n if (mate.mateType.equalsAny(eMateType.revolute, eMateType.cylindrical)) {\r\n /**\r\n * For these types of mates, the objective is to match the direction vectors but allow the angle to float freely.\r\n * What this means in practice is that we have to split the quaternion into a swing and a twist component.\r\n * The swing component will rotate the direction vector to point the right way, along a rotation axis equal to the cross product of the X axis and the direction vector,\r\n * and the twist component will rotate the mesh around the direction vector to match the desired angles. We then throw away the \"new\" twist angle and use the original twist angle.\r\n * For this to work it's necessary to ensure that the same rotation axis is used for both the swing rotation of the target and the swing rotation of the current, if the directions match.\r\n * Otherwise the twist angles won't be referring to the same thing.\r\n */\r\n const targetQ = Quaternion.FromRotationMatrix(targetWorldTransformMatrix.getRotationMatrix());\r\n const currentQ = Quaternion.FromRotationMatrix(currentWorldTransformMatrix.getRotationMatrix());\r\n const targetDirection = Vector3.TransformNormal(Axis.X, targetWorldTransformMatrix).normalize();\r\n const currentDirection = c2Mesh.right;\r\n\r\n // At direction vectors close to or equal to the X axis, the twist angle becomes heavily affected by floating point errors, so we avoid loops by not updating the rotation if the direction vectors are close to each other.\r\n if (currentDirection.add(targetDirection).length() > 0.0001) {\r\n const targetST = decomposeSwingTwist(targetQ, targetDirection.scale(-1));\r\n const currentST = decomposeSwingTwist(currentQ, currentDirection);\r\n const targetRotationNew = targetST.swing.multiply(currentST.twist);\r\n\r\n calculatedTargetWorldTransform = Matrix.Compose(followerMesh.scaling, targetRotationNew, t);\r\n }\r\n } else if (mate.mateType.equalsAny(eMateType.fastened, eMateType.planar, eMateType.slider)) {\r\n calculatedTargetWorldTransform = targetWorldTransformMatrix;\r\n }\r\n\r\n if (calculatedTargetWorldTransform) {\r\n const followerMeshTransformMatrix = c2LocalTransform.clone().invert();\r\n followerMeshTransformMatrix.multiplyToRef(calculatedTargetWorldTransform, followerMeshTransformMatrix);\r\n followerMeshTransformMatrix.multiplyToRef(grandParentContainerWorldMatrixInv, followerMeshTransformMatrix);\r\n\r\n const resultFollowerRotMatrix = followerMeshTransformMatrix.getRotationMatrix();\r\n\r\n if (ENABLE_DEBUG_VISUAL) {\r\n const targetForward = Vector3.TransformNormal(Axis.X, targetWorldTransformMatrix).normalize();\r\n const targetLeft = Vector3.TransformNormal(Axis.Z, targetWorldTransformMatrix).normalize();\r\n\r\n const c2CalcWorldTransformMatrix = c2LocalTransform\r\n .multiply(resultFollowerRotMatrix)\r\n .multiply(grandParentContainerWorldMatrix);\r\n const calcDirection = Vector3.TransformNormal(Axis.X, c2CalcWorldTransformMatrix).normalize();\r\n const calcAngle = Vector3.TransformNormal(Axis.Z, c2CalcWorldTransformMatrix).normalize();\r\n updateVisualization(\r\n mate._dynamicId || '',\r\n c2Mesh.getScene(),\r\n c2Mesh.getAbsolutePosition(),\r\n calcAngle,\r\n calcDirection,\r\n targetLeft,\r\n targetForward\r\n );\r\n } else {\r\n followerMesh.rotationQuaternion = Quaternion.FromRotationMatrix(resultFollowerRotMatrix);\r\n updateMatrixOfAncestorChain(followerMesh, c2Mesh); //update world matrix after having just rotated follower mesh\r\n }\r\n }\r\n\r\n if (mate.mateType == eMateType.fastened || mate.mateType == eMateType.revolute) {\r\n //translate so the connectors line up\r\n worldDelta = c1Mesh.getAbsolutePosition().subtract(c2Mesh.getAbsolutePosition());\r\n mateOffset = new Vector3(mate.offsetX, mate.offsetY, mate.offsetZ);\r\n } else if (mate.mateType == eMateType.planar) {\r\n /**\r\n * Planar Mate\r\n * - define a plane at the position of c1, with a normal in the direction of c1\r\n * - make a ray from c2 in the direction of c2, and try to intersect the c1 plane\r\n * - if it doesn't intersect, flip it and try again\r\n * - the intersection of the ray and plane is the new position of c2 (not considering offset)\r\n */\r\n const c1PlaneNormal = Axis.X.rotateByQuaternionToRef(c1Mesh.absoluteRotationQuaternion, Vector3.Zero());\r\n const c1Plane = Plane.FromPositionAndNormal(c1Mesh.getAbsolutePosition(), c1PlaneNormal);\r\n const c2Direction = Axis.X.rotateByQuaternionToRef(c2Mesh.absoluteRotationQuaternion, Vector3.Zero());\r\n const ray = new Ray(c2Mesh.getAbsolutePosition(), c2Direction);\r\n let distance = ray.intersectsPlane(c1Plane);\r\n if (distance == null) {\r\n //if it doesn't intersect try flipping it\r\n ray.direction.negateInPlace();\r\n distance = ray.intersectsPlane(c1Plane);\r\n }\r\n worldDelta = ray.direction.scale(distance!);\r\n mateOffset = new Vector3(mate.offsetX, 0, 0); //only use the x component for planar mates\r\n } else if (mate.mateType == eMateType.slider || mate.mateType == eMateType.cylindrical) {\r\n /**\r\n * Slider Mate\r\n * - define a plane at the position of c2, with a normal in the direction of c2\r\n * - make a ray from c1 in the direction of c1, and try to intersect the c2 plane\r\n * - if it doesn't intersect, flip it and try again\r\n * - the intersection of the ray and plane is the new position of c2 (not considering offset)\r\n */\r\n const c1Pos = c1Mesh.getAbsolutePosition();\r\n const c2Pos = c2Mesh.getAbsolutePosition();\r\n const c2PlaneNormal = c2Mesh.getDirection(Axis.X).normalize();\r\n const c2Plane = Plane.FromPositionAndNormal(c2Pos, c2PlaneNormal);\r\n const c1Direction = c1Mesh.getDirection(Axis.X).normalize();\r\n const ray = new Ray(c1Pos, c1Direction);\r\n let distance = ray.intersectsPlane(c2Plane);\r\n if (distance == null) {\r\n //if it doesn't intersect try flipping it\r\n ray.direction.negateInPlace();\r\n distance = ray.intersectsPlane(c2Plane);\r\n }\r\n worldDelta = ray.direction.scale(distance!).add(c1Pos).subtract(c2Pos);\r\n mateOffset = new Vector3(0, mate.offsetY, mate.offsetZ); //don't use the x component for slider mates\r\n } else if (mate.mateType == eMateType.parallel) {\r\n //nothing to do for parallel\r\n }\r\n }\r\n\r\n /**\r\n * now we process the offset and set the position of c2 parent node\r\n */\r\n if (!mate.flip) mateOffset.negateInPlace();\r\n if (mate._swap) mateOffset.y = -mateOffset.y;\r\n const worldOffset = mateOffset.rotateByQuaternionToRef(c1Mesh.absoluteRotationQuaternion, Vector3.Zero());\r\n const totalOffset = worldOffset.add(worldDelta);\r\n const newAbsPos = followerMesh.getAbsolutePosition().add(totalOffset);\r\n followerMesh.setAbsolutePosition(newAbsPos);\r\n followerMesh.computeWorldMatrix(true);\r\n}\r\n\r\nfunction updateVisualization(\r\n id: string,\r\n scene: Scene,\r\n position: Vector3,\r\n calcLeft: Vector3,\r\n calcForward: Vector3,\r\n targetLeft: Vector3,\r\n targetForward: Vector3\r\n) {\r\n const lines = [\r\n [position.add(calcLeft), position, position.add(calcForward)],\r\n [position.add(targetLeft.scale(-1)), position, position.add(targetForward.scale(-1))],\r\n ];\r\n let viz = debugViz.get(id);\r\n if (!viz) {\r\n viz = MeshBuilder.CreateLineSystem(\r\n id,\r\n {\r\n lines,\r\n updatable: true,\r\n colors: [\r\n [new Color4(1, 0, 0, 1), new Color4(1, 1, 1, 0.5), new Color4(1, 1, 0, 1)],\r\n [new Color4(0, 1, 0, 1), new Color4(1, 1, 1, 0.5), new Color4(1, 1, 1, 1)],\r\n ],\r\n },\r\n scene\r\n );\r\n debugViz.set(id, viz);\r\n } else {\r\n viz = MeshBuilder.CreateLineSystem(id, { lines, updatable: true, instance: viz }, scene);\r\n }\r\n}\r\n\r\n/**\r\n Decompose the rotation on to 2 parts.\r\n 1. Twist - rotation around the \"direction\" vector\r\n 2. Swing - rotation around axis that is perpendicular to \"direction\" vector\r\n The rotation can be composed back by \r\n rotation = swing * twist\r\n\r\n has singularity in case of swing_rotation close to 180 degrees rotation.\r\n if the input quaternion is of non-unit length, the outputs are non-unit as well\r\n otherwise, outputs are both unit\r\n*/\r\nfunction decomposeSwingTwist(rotation: Quaternion, _direction: Vector3) {\r\n // Ensure the direction vector is normalized\r\n const direction = _direction.clone().normalize();\r\n const twist = new Quaternion();\r\n const swing = new Quaternion();\r\n\r\n // Alternate code with indeterminate swing axis\r\n // const ra = new Vector3(rotation.x, rotation.y, rotation.z);\r\n // const p = direction.scale(Vector3.Dot(ra, direction));\r\n // twist.set(p.x, p.y, p.z, rotation.w);\r\n // twist.normalize();\r\n // rotation.multiplyToRef(twist.conjugate(), swing);\r\n rotationToVectorToRef(direction, Axis.X, swing);\r\n swing.conjugate().multiplyToRef(rotation, twist);\r\n\r\n return { twist, swing };\r\n}\r\n","import { eConnectorPositionMode } from '@models/classes/enums';\r\nimport { SketchControlPoint, isConnector } from '@models/classes/features';\r\nimport { ArrayMap } from '@models/classes/handlers';\r\nimport { MateNode } from '@models/classes/mates/mate-node';\r\nimport { isSketchControlPoint } from '@models/classes/meshes/mesh-utilities';\r\nimport { SpaceNode } from '@models/classes/meshes/space-node';\r\n\r\nexport enum ConstraintType {\r\n Sphere, // Constrains the target to a point on a shell around a point\r\n Line, // Constrains the target to a point along a line through a point\r\n Plane, // Constrains the target to a point on a plane\r\n Fixed, // Constrains the target to a fixed point\r\n None, // Does not constrain the target\r\n}\r\n\r\nexport interface ConstraintDef {\r\n [ConstraintType.Sphere]: { x: number; y: number; z: number; radius: number };\r\n [ConstraintType.Line]: { x: number; y: number; z: number; dx: number; dy: number; dz: number };\r\n [ConstraintType.Plane]: { x: number; y: number; z: number; nx: number; ny: number; nz: number };\r\n [ConstraintType.Fixed]: void;\r\n [ConstraintType.None]: void;\r\n [key: number]: unknown;\r\n}\r\n\r\ntype ConstraintDefMap = {\r\n [key in ConstraintType]: {\r\n type: key;\r\n properties: ConstraintDef[key];\r\n };\r\n};\r\ntype MateConstraint = ConstraintDefMap[keyof ConstraintDefMap];\r\n\r\ntype MatedObject = SpaceNode | SketchControlPoint;\r\n\r\ninterface MateEdge {\r\n target: MatedObject;\r\n /** if a mate is not constraining the target node's other */\r\n constraint: MateConstraint;\r\n /** if mate is null means it's a parent/child edge */\r\n mate?: MateNode;\r\n}\r\n\r\nexport interface MateSorterResult {\r\n sorted: Set;\r\n cycles: MateNode[][];\r\n overConstrained: MateNode[];\r\n}\r\n\r\nexport function sortMates(validMates: MateNode[]) {\r\n const fixed = new Set();\r\n const primes = new Set();\r\n const nodeMap = getNodeMap(validMates);\r\n\r\n //parent/child relationships are also a kind of edge in our graph, so add them\r\n for (const [node] of nodeMap) {\r\n if (node instanceof SpaceNode) {\r\n if (node.fixed) {\r\n fixed.add(node);\r\n } else if (node._primeMover) {\r\n primes.add(node);\r\n }\r\n for (const ancestor of node.getAncestors()) {\r\n if (nodeMap.has(ancestor as MatedObject)) {\r\n nodeMap.add(node, {\r\n target: ancestor as MatedObject,\r\n constraint: { type: ConstraintType.None, properties: undefined },\r\n });\r\n nodeMap.add(ancestor as MatedObject, {\r\n target: node,\r\n constraint: { type: ConstraintType.Fixed, properties: undefined },\r\n });\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * * starting with primes, recurse through. Then move to non-primes, ignoring those already visited.\r\n */\r\n const visitedNodes = new Set();\r\n const visitedMates = new Set();\r\n const nodePath = new Set();\r\n const matePath = new Set();\r\n const sorted = new Set();\r\n const cycles = new Array();\r\n const overConstrained = new Array();\r\n\r\n for (const n of [...fixed, ...primes, ...nodeMap.keys()]) {\r\n recurseMateChain(nodeMap, visitedNodes, visitedMates, nodePath, matePath, sorted, cycles, overConstrained, n);\r\n }\r\n\r\n //remove overconstrained mates from sorted so they don't get processed\r\n for (const m of overConstrained) {\r\n sorted.delete(m);\r\n }\r\n\r\n return {\r\n sorted,\r\n cycles,\r\n overConstrained,\r\n } as MateSorterResult;\r\n}\r\n\r\n/**\r\n * Gets the nodes that are connected to the given through a chain of mates\r\n */\r\nexport function getNodeChain(node: MatedObject, validMates: MateNode[]) {\r\n const nodeMap = getNodeMap(validMates);\r\n const visitedNodes = new Set();\r\n const visitedMates = new Set();\r\n const nodePath = new Set();\r\n const matePath = new Set();\r\n const sorted = new Set();\r\n const cycles = new Array();\r\n const overConstrained = new Array();\r\n\r\n recurseMateChain(nodeMap, visitedNodes, visitedMates, nodePath, matePath, sorted, cycles, overConstrained, node);\r\n\r\n visitedNodes.delete(node);\r\n\r\n return visitedNodes;\r\n}\r\n\r\nexport function getNodeMap(validMates: MateNode[]) {\r\n const nodeMap = new ArrayMap();\r\n for (const mate of validMates) {\r\n const c1 = mate.getC1();\r\n const c2 = mate.getC2();\r\n let c1Parent: SpaceNode | SketchControlPoint | undefined = c1?.getSpaceParent();\r\n let c2Parent: SpaceNode | SketchControlPoint | undefined = c2?.getSpaceParent();\r\n if (c1 && isSketchControlPoint(c1)) {\r\n c1Parent = c1;\r\n }\r\n\r\n if (c2 && isSketchControlPoint(c2)) {\r\n c2Parent = c2;\r\n }\r\n\r\n if (c1 && c2 && c1Parent && c2Parent) {\r\n const constraint: MateConstraint = { type: ConstraintType.Fixed, properties: undefined };\r\n\r\n // TODO [RO] - Later on we can add more constraint types here based on mate type (CPQ-3881)\r\n //if (isConstrainingConnector(c1) && isConstrainingConnector(c2)) {\r\n // constraint = { type: ConstraintType.Fixed, properties: undefined };\r\n // } else {\r\n // constraint = { type: ConstraintType.None, properties: undefined };\r\n // }\r\n\r\n nodeMap.add(c1Parent, { target: c2Parent, mate, constraint });\r\n nodeMap.add(c2Parent, { target: c1Parent, mate, constraint });\r\n }\r\n }\r\n return nodeMap;\r\n}\r\n\r\nfunction recurseMateChain(\r\n nodeMap: ArrayMap,\r\n visitedNodes: Set,\r\n visitedMates: Set,\r\n nodePath: Set,\r\n matePath: Set,\r\n sorted: Set,\r\n cycles: MateNode[][],\r\n overConstrained: MateNode[],\r\n node: MatedObject,\r\n mate?: MateNode,\r\n sourceEdge?: MateEdge,\r\n isOverconstrained = false\r\n) {\r\n if (!sourceEdge || sourceEdge.constraint.type === ConstraintType.Fixed) {\r\n if (nodePath.has(node)) {\r\n const cycle = [...matePath];\r\n if (mate) {\r\n cycle.push(mate);\r\n }\r\n cycles.push(cycle);\r\n }\r\n\r\n if (visitedNodes.has(node)) {\r\n return;\r\n }\r\n }\r\n visitedNodes.add(node);\r\n nodePath.add(node);\r\n\r\n if (mate) {\r\n matePath.add(mate);\r\n visitedMates.add(mate);\r\n\r\n sorted.add(mate);\r\n const c1 = mate.getC1()!;\r\n const c2 = mate.getC2()!;\r\n /** Move along sketch connectors should always be the connector 1, because they can't be moved (they are read only) */\r\n if (c2.positionMode === eConnectorPositionMode.moveAlongSketch) {\r\n mate._swap = true;\r\n }\r\n /** Control points are always follower meshes */\r\n else if (isConnector(c1) && isSketchControlPoint(c2)) {\r\n mate._swap = false;\r\n } else if (isConnector(c2) && isSketchControlPoint(c1)) {\r\n mate._swap = true;\r\n } else {\r\n /** mate needs to have c1/c2 swapped if we got here in the opposite direction */\r\n mate._swap = c1._parent! === node;\r\n }\r\n }\r\n\r\n // If the source edge is fixed, we recurse through the node because all the node's connectors will be shifting.\r\n // If the source edge is not fixed, we stop here because only the mated connector will move\r\n // [RO] TODO eventually we will have to rethink this logic to support more constraint types (CPQ-3881)\r\n if (!sourceEdge || sourceEdge.constraint.type === ConstraintType.Fixed) {\r\n for (const edge of nodeMap.get(node)) {\r\n if (!edge.mate || !visitedMates.has(edge.mate)) {\r\n if (\r\n edge.target instanceof SpaceNode &&\r\n edge.target.fixed &&\r\n edge.constraint.type !== ConstraintType.None\r\n ) {\r\n isOverconstrained = true;\r\n }\r\n recurseMateChain(\r\n nodeMap,\r\n visitedNodes,\r\n visitedMates,\r\n nodePath,\r\n matePath,\r\n sorted,\r\n cycles,\r\n overConstrained,\r\n edge.target,\r\n edge.mate,\r\n edge,\r\n isOverconstrained\r\n );\r\n }\r\n }\r\n }\r\n\r\n nodePath.delete(node);\r\n if (mate) {\r\n matePath.delete(mate);\r\n if (isOverconstrained) {\r\n overConstrained.push(mate);\r\n }\r\n visitedMates.delete(mate);\r\n }\r\n}\r\n","import { ChangeMap } from '../renderer-manager';\r\n\r\nexport abstract class Processor {\r\n private afterCleanupTasks: Array<() => void> = [];\r\n\r\n beforeRender() {}\r\n beforeDeletions(deletions: ChangeMap) {}\r\n afterDeletions(deletions: ChangeMap) {}\r\n beforeUpdates(changes: ChangeMap) {}\r\n afterUpdates(changes: ChangeMap) {}\r\n afterCleanup(func: () => void) {\r\n this.afterCleanupTasks.push(func);\r\n }\r\n doAfterCleanup() {\r\n this.afterCleanupTasks.forEach(task => task());\r\n this.afterCleanupTasks.clear();\r\n }\r\n dispose() {}\r\n}\r\n","import { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { BaseConnector } from '@models/classes/features';\r\nimport { Connector, isConnector } from '@models/classes/features/connector';\r\nimport { MateNode } from '@models/classes/mates/mate-node';\r\nimport { isSketchControlPoint } from '@models/classes/meshes/mesh-utilities';\r\nimport { SpaceNode } from '@models/classes/meshes/space-node';\r\nimport { KbQuaternion } from '@models/classes/primitives/kb-quaternion';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass } from '@models/interfaces/tracked-class';\r\nimport { mateConnectors, mateControlPointToConnector } from '@view/helpers/mate-helper';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { ConnectorRenderer } from '../features/connector';\r\nimport { SketchControlPointRenderer } from '../features/sketch-control-point';\r\nimport { ChangeMap } from '../renderer-manager';\r\nimport { sortMates } from './mate-sorter';\r\nimport { Processor } from './processor';\r\nimport { ReferenceProcessor } from './reference-processor';\r\nimport { eConnectorPositionMode } from '@models/classes';\r\n\r\nconst CYCLE_ERROR_START = 'Cycle detected: ';\r\n\r\nexport class MateProcessor extends Processor {\r\n /**\r\n * Mates sorted based on their dependencies of each other\r\n */\r\n private sortedMates: Set;\r\n /**\r\n * If during rebuilding of the sorted mate graph any cycles are detected, they are stored here\r\n */\r\n private cycles: MateNode[][];\r\n\r\n /**\r\n * we try to avoid rebuilding the mate dependency graph unless a change warrants it.\r\n */\r\n private rebuildDependencies: boolean;\r\n\r\n constructor(\r\n protected viewer: KbViewer,\r\n protected scene: Scene,\r\n protected connectorRenderer: ConnectorRenderer,\r\n protected controlPointRenderer: SketchControlPointRenderer,\r\n protected referenceProcessor: ReferenceProcessor\r\n ) {\r\n super();\r\n }\r\n\r\n beforeRender() {\r\n this.rebuildDependencies = false;\r\n }\r\n\r\n beforeDeletions(deletions: ChangeMap) {\r\n const mateDeletions = deletions.get(Token.MateNode) as Array>;\r\n const connectorDeletions = deletions.get(Token.Connector) as Array>;\r\n this.rebuildDependencies =\r\n this.rebuildDependencies || mateDeletions.length > 0 || connectorDeletions.length > 0;\r\n }\r\n\r\n beforeUpdates(changes: ChangeMap) {\r\n //determine if mate dependencies need to be rebuilt\r\n const mateChanges = changes.get(Token.MateNode) as Array>;\r\n const connectorChanges = changes.get(Token.Connector) as Array>;\r\n const meshChanges = changes.get(Token.SpaceNode) as Array>;\r\n\r\n this.rebuildDependencies =\r\n this.rebuildDependencies ||\r\n mateChanges.some(m => m.changes.hasAny('connector1', 'connector2', 'enabled')) ||\r\n connectorChanges.some(c => c.changes.has('enabled')) ||\r\n meshChanges.some(c => c.changes.hasAny('fixed', '_primeMover'));\r\n\r\n /**\r\n * Mates need to be reprocessed if:\r\n * * the mate itself was changed\r\n * * connector1 of the node was changed\r\n * * parent node of connector1 was changed\r\n * * any ancestor space node of the parent node was changed\r\n * * any parent graph dependency node mate or it's parents were changed\r\n */\r\n }\r\n\r\n afterUpdates(changes: ChangeMap) {\r\n if (this.rebuildDependencies) {\r\n this.buildDependencies();\r\n }\r\n if (this.sortedMates) {\r\n if (\r\n changes.has(Token.MateNode) ||\r\n changes.has(Token.SpaceNode) ||\r\n changes.has(Token.Connector) ||\r\n changes.has(Token.Feature)\r\n ) {\r\n for (const mate of this.sortedMates) {\r\n //don't process the last mate in a cycle\r\n if (!this.cycles || !this.cycles.some(c => c.last() === mate)) {\r\n this.processMate(mate);\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n dispose() {\r\n if (this.sortedMates) this.sortedMates.clear();\r\n if (this.cycles) this.cycles.clear();\r\n }\r\n\r\n processMate(mate: MateNode) {\r\n let c1 = mate.getC1()!;\r\n let c2 = mate.getC2()!;\r\n\r\n /** \r\n * If the sorter determined we need to swap the connectors. \r\n * */\r\n if (mate._swap || c2.positionMode === eConnectorPositionMode.moveAlongSketch) {\r\n const temp = c1;\r\n c1 = c2;\r\n c2 = temp;\r\n }\r\n\r\n const c1Mesh = this.getConnectorMesh(c1);\r\n const c2Mesh = this.getConnectorMesh(c2);\r\n const followerNode = c2._parent! as ITrackedClass;\r\n const followerMesh = this.viewer.rendererManager.mesh.get3d(followerNode) as TransformNode;\r\n\r\n if (!c1Mesh || !c2Mesh) return;\r\n\r\n /** \r\n * Move along sketch connectors cannot be moved, because their position is being set by the sketch itself. They can only be used \r\n * to determine the position of other connectors. Moving them will cause an immediate overconstraint condition\r\n */\r\n if(c2.positionMode === eConnectorPositionMode.moveAlongSketch) {\r\n return;\r\n }\r\n\r\n if (isSketchControlPoint(c2)) {\r\n const cp = c2;\r\n this.viewer.pauseRenderWhile(() => {\r\n const nodeWithTransform = mateControlPointToConnector(mate, c1Mesh, c2Mesh);\r\n const d1 = nodeWithTransform.direction.scale(cp.direction.length())\r\n const d2 = cp.direction2 ? nodeWithTransform.direction.scale(cp.direction2.length()) : undefined;\r\n if(\r\n !nodeWithTransform.position.equalsWithEpsilon(cp.position.toVec3()) || \r\n !d1.equalsWithEpsilon(cp.direction.toVec3()) || \r\n (cp.direction2 && d2 && !d2.equalsWithEpsilon(cp.direction2.toVec3()))\r\n ) {\r\n this.afterCleanup(() => {\r\n cp.position = KbVector.FromVec3(nodeWithTransform.position);\r\n cp.direction = KbVector.FromVec3(d1);\r\n if(cp.direction2 && d2) {\r\n cp.direction2 = KbVector.FromVec3(d2);\r\n }\r\n });\r\n }\r\n });\r\n } else if (followerMesh) {\r\n this.viewer.pauseRenderWhile(() => {\r\n const originalFollowerMeshAbsPos = followerMesh.getAbsolutePosition();\r\n const originalFollowerMeshAbsQuat = followerMesh.absoluteRotationQuaternion.clone();\r\n const originalFollowerMeshLocalPos = followerMesh.position.clone();\r\n const originalFollowerMeshLocalQuat = followerMesh.rotationQuaternion!.clone();\r\n\r\n mateConnectors(mate, followerMesh, c1Mesh, c2Mesh);\r\n\r\n /** update real kb3d nodes with new transform info */\r\n if (\r\n !originalFollowerMeshLocalPos.equalsWithEpsilon(followerMesh.position) ||\r\n !originalFollowerMeshLocalQuat.equalsWithEpsilon(followerMesh.rotationQuaternion!)\r\n ) {\r\n followerNode.trackChanges(false);\r\n followerNode.setRotationQuaternion(KbQuaternion.FromQuat(followerMesh.rotationQuaternion!));\r\n followerNode.position = KbVector.FromVec3(followerMesh.position);\r\n followerNode.trackChanges(true);\r\n this.referenceProcessor.triggerChangesOnReferencers(followerNode, new Map([['position', null]]));\r\n }\r\n\r\n /** invalidate bounding box of parent if follower mesh actually moved */\r\n if (\r\n !originalFollowerMeshAbsPos.equalsWithEpsilon(followerMesh.getAbsolutePosition()) ||\r\n !originalFollowerMeshAbsQuat.equalsWithEpsilon(followerMesh.absoluteRotationQuaternion)\r\n ) {\r\n this.viewer.bbCache.invalidate(c2.getSpaceParent());\r\n }\r\n\r\n //console.log(`mate '${mate.name} processed`);\r\n });\r\n }\r\n }\r\n\r\n protected getConnectorMesh(c: BaseConnector) {\r\n if (isSketchControlPoint(c)) {\r\n return this.controlPointRenderer.get3d(c)!;\r\n } else if (isConnector(c)) {\r\n return this.connectorRenderer.get3d(c)!;\r\n }\r\n }\r\n\r\n protected buildDependencies() {\r\n const mates = this.getValidMates();\r\n const sortResult = sortMates(mates);\r\n this.sortedMates = sortResult.sorted;\r\n this.cycles = sortResult.cycles;\r\n\r\n const validator = this.viewer.options.validator;\r\n for (const cycle of sortResult.cycles) {\r\n const err = `${CYCLE_ERROR_START}${cycle.map(n => n.name).join(' -> ')}`;\r\n if (validator) {\r\n for (const cm of cycle) {\r\n validator.addError(cm, '', err);\r\n }\r\n } else {\r\n console.warn(err);\r\n }\r\n }\r\n if (validator) {\r\n for (const om of sortResult.overConstrained) {\r\n validator.addError(om, '', `Cannot have more than one fixed node in the mate chain`);\r\n }\r\n } else {\r\n if (sortResult.overConstrained.length) {\r\n console.warn(`Mates overconstrained: [${sortResult.overConstrained.map(n => n.name).join(', ')}]`);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * gets all mates (including those in nested scenes)\r\n * that are enabled, and have valid references to connectors\r\n */\r\n protected getValidMates() {\r\n const rootScene = this.viewer.sceneNode;\r\n const mates = new Array();\r\n const validator = this.viewer.options.validator;\r\n\r\n rootScene.runOnAllScenes(s => {\r\n const sceneMates = s._manager!.getByToken(Token.MateNode);\r\n for (const m of sceneMates) {\r\n if (m.isActive()) {\r\n mates.push(m);\r\n }\r\n\r\n //regardless, clear out cycle validation\r\n if (validator) {\r\n validator.clearProp(m, '');\r\n }\r\n }\r\n });\r\n return mates;\r\n }\r\n}\r\n","import { Quaternion } from '@babylonjs/core/Maths';\r\nimport { SolidParticle } from '@babylonjs/core/Particles/solidParticle';\r\nimport { CircularPatternFeature } from '@models/classes/features/circular-pattern-feature';\r\nimport { Token } from '@models/classes/token';\r\nimport { SubmeshVertexData } from '@view/helpers/geometry-helper';\r\nimport { PatternRenderer } from './pattern';\r\n\r\nexport class CircularPatternRenderer extends PatternRenderer {\r\n modelToken = Token.CircularPatternFeature;\r\n updateMeshGeometry(\r\n feature: CircularPatternFeature,\r\n vertexData: SubmeshVertexData,\r\n vertexDataChanges: Partial\r\n ) {\r\n const featurePivot = feature.pivot.toVec3();\r\n const featureAxis = feature.axis.toVec3();\r\n // Don't allow a zero vector for an axis because it will blow up the math\r\n if (featureAxis.equalsToFloats(0, 0, 0)) {\r\n featureAxis.set(0, 1, 0);\r\n }\r\n const featureArc = (feature.arc * Math.PI) / 180 / feature.count;\r\n const workingQuaternion = Quaternion.Identity();\r\n\r\n return super.updateMeshGeometry(\r\n feature,\r\n vertexData,\r\n vertexDataChanges,\r\n (idx: number, particle: SolidParticle) => {\r\n particle.pivot = featurePivot;\r\n Quaternion.RotationAxisToRef(featureAxis, featureArc * idx, workingQuaternion);\r\n particle.rotationQuaternion!.multiplyInPlace(workingQuaternion);\r\n }\r\n );\r\n }\r\n}\r\n","import { DeleteFacesFeature } from '@models/classes/features/delete-faces-feature';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass } from '@models/interfaces/tracked-class';\r\nimport { SubmeshVertexData } from '@view/helpers/geometry-helper';\r\nimport { FeatureRenderer } from './feature';\r\n\r\nexport class DeleteFacesRenderer extends FeatureRenderer {\r\n modelToken = Token.DeleteFacesFeature;\r\n updateMeshGeometry(\r\n feature: ITrackedClass,\r\n vertexData: SubmeshVertexData,\r\n vertexDataChanges: Partial\r\n ) {\r\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\r\n\r\n if (newVertexData.indices && feature.vertexGroups.length > 0) {\r\n const vertexGroups = feature.vertexGroups.sort((groupA, groupB) => groupA.startIndex - groupB.startIndex);\r\n const newIndices = new Int32Array(newVertexData.indices);\r\n let currReadIndex = 0;\r\n\r\n vertexGroups.forEach(vertexGroup => {\r\n for (\r\n let i = Math.max(vertexGroup.startIndex, currReadIndex);\r\n i < Math.min(vertexGroup.endIndex, newIndices.length);\r\n i++\r\n ) {\r\n newIndices[i] = -1;\r\n }\r\n currReadIndex = vertexGroup.endIndex;\r\n });\r\n\r\n // We need to remove submeshes targetting the indices that are being removed as well\r\n if (newVertexData.submeshData) {\r\n newVertexData.submeshData.forEach(submeshData => {\r\n const indexStart = submeshData.indexStart;\r\n for (let i = 0; i < indexStart; i++) {\r\n if (newIndices[i] === -1) {\r\n submeshData.indexStart--;\r\n }\r\n }\r\n const indexCount = submeshData.indexCount;\r\n for (let i = 0; i < indexCount; i++) {\r\n const index = indexStart + i;\r\n if (newIndices[index] === -1) {\r\n submeshData.indexCount--;\r\n }\r\n }\r\n });\r\n }\r\n\r\n newVertexData.indices = newIndices.filter(index => index !== -1);\r\n }\r\n return newVertexData;\r\n }\r\n\r\n update3dInternal() {}\r\n}\r\n","import { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { VertexData } from '@babylonjs/core/Meshes/mesh.vertexData';\r\nimport { getBinClampedSize } from '@common-util/texture-util';\r\nimport { eTextureChannel } from '@models/classes';\r\nimport { DisplacementMapFeature } from '@models/classes/features/displacement-map-feature';\r\nimport { Token } from '@models/classes/token';\r\nimport { SubmeshVertexData } from '@view/helpers/geometry-helper';\r\nimport { FeatureRenderer } from './feature';\r\nimport { ensureNormalsExist } from './utils';\r\n\r\nexport class DisplacementMapRenderer extends FeatureRenderer {\r\n _initialized: boolean;\r\n modelToken = Token.DisplacementMapFeature;\r\n textureCanvas = document.createElement('canvas');\r\n\r\n updateMeshGeometry(\r\n feature: DisplacementMapFeature,\r\n vertexData: SubmeshVertexData,\r\n vertexDataChanges: Partial\r\n ) {\r\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\r\n const textureRenderer = this.viewer.rendererManager.textures;\r\n if (feature.textures && feature.textures.length > 0) {\r\n //create context\r\n const context = this.textureCanvas.getContext('2d', {\r\n willReadFrequently: true,\r\n });\r\n const strength = 1;\r\n const hasTransparency = false;\r\n const parameters = { strength, hasTransparency };\r\n\r\n const textureLayers = textureRenderer.getActiveLayers(feature, eTextureChannel.albedo).filter(node => {\r\n const renderer = textureRenderer.getRenderer(node);\r\n return renderer.textureHasContent(node);\r\n });\r\n\r\n if (context) {\r\n const baseImageSize = textureRenderer\r\n .getRenderer(feature.textures[0])\r\n .getNativeSize(feature.textures[0]);\r\n if (this.scene.getEngine().webGLVersion < 2) {\r\n const clampedSize = getBinClampedSize(baseImageSize.width, baseImageSize.height);\r\n context.canvas.width = clampedSize;\r\n context.canvas.height = clampedSize;\r\n } else {\r\n if (feature.textures[0]) {\r\n context.canvas.width = baseImageSize.width;\r\n context.canvas.height = baseImageSize.height;\r\n }\r\n }\r\n context.clearRect(0, 0, context.canvas.width, context.canvas.height);\r\n\r\n const paintedImage = textureRenderer.paintTextureOnCanvas(textureLayers, context, parameters);\r\n\r\n const tempVertexData = new SubmeshVertexData();\r\n tempVertexData.copyFrom(newVertexData);\r\n if (tempVertexData.uvs) {\r\n const n = tempVertexData.uvs.length;\r\n for (let i = 1; i < n; i += 2) {\r\n tempVertexData.uvs[i] = 1 - tempVertexData.uvs[i];\r\n }\r\n\r\n const mesh = this.viewer.getRendererNode(feature._parent!);\r\n const meshUpdated = new Mesh('ExistingMesh');\r\n tempVertexData.applyToMesh(meshUpdated, true);\r\n ensureNormalsExist(meshUpdated);\r\n\r\n const imageData = paintedImage?.getImageData(\r\n 0,\r\n 0,\r\n this.textureCanvas.width,\r\n this.textureCanvas.height,\r\n mesh\r\n );\r\n if (imageData) {\r\n this.displace(imageData, feature.minDisplacement, feature.maxDisplacement, meshUpdated);\r\n }\r\n const meshVertexData = VertexData.ExtractFromMesh(meshUpdated);\r\n newVertexData.positions = meshVertexData.positions;\r\n newVertexData.normals = meshVertexData.normals;\r\n meshUpdated.dispose();\r\n }\r\n }\r\n }\r\n return newVertexData;\r\n }\r\n\r\n update3dInternal() {}\r\n\r\n displace(context: ImageData, minDisplacement: number, maxDisplacement: number, parentMesh: Mesh) {\r\n const uInt8Array = new Uint8Array(context.data);\r\n parentMesh.applyDisplacementMapFromBuffer(\r\n uInt8Array,\r\n context.width,\r\n context.height,\r\n minDisplacement,\r\n maxDisplacement\r\n );\r\n }\r\n}\r\n","import { Matrix, Vector2, Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { VertexBuffer } from '@babylonjs/core/Meshes/buffer';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { PolygonMeshBuilder } from '@babylonjs/core/Meshes/polygonMesh';\r\nimport { FloatArray } from '@babylonjs/core/types';\r\nimport { ExtrudeFeature, Token, eAxis, isSketchNode } from '@models/classes';\r\nimport { earcut } from '@view/helpers/earcut';\r\nimport { SubmeshVertexData } from '@view/helpers/geometry-helper';\r\nimport { rotationToVector } from '@view/helpers/math-helper';\r\nimport { axisToVector3, parseSketchGeometry } from '@view/modules/sketching/util';\r\nimport { FeatureRenderer } from './feature';\r\n\r\nconst xztoxy = Matrix.FromArray([1, 0, 0, 0, 0, 0, -1, 0, 0, 1, 0, 0, 0, 0, 0, 1]);\r\n\r\nexport class ExtrudeRenderer extends FeatureRenderer {\r\n modelToken = Token.ExtrudeFeature;\r\n\r\n updateMeshGeometry(feature: T, vertexData: SubmeshVertexData, vertexDataChanges: Partial) {\r\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\r\n const parentMesh = feature._parent;\r\n\r\n if (\r\n parentMesh &&\r\n newVertexData.positions &&\r\n newVertexData.positions.length &&\r\n // paths are used by parseSketchGeometry\r\n newVertexData.paths &&\r\n newVertexData.paths.length > 0 &&\r\n feature.size !== 0\r\n ) {\r\n const { segments, holes } = parseSketchGeometry(newVertexData);\r\n const extrudedMeshes: Mesh[] = [];\r\n\r\n for (let i = 0; i < segments.length; i++) {\r\n const shape = segments[i];\r\n const shapeHoles = holes.get(i);\r\n\r\n const sketch = feature._parent && isSketchNode(feature._parent) ? feature._parent : undefined;\r\n const sketchNormal = sketch ? axisToVector3(sketch.normal) : Vector3.Zero();\r\n if (sketch && sketch.normal !== eAxis.z) {\r\n sketchNormal.scaleInPlace(-1);\r\n }\r\n\r\n const rotToZ = rotationToVector(sketchNormal, new Vector3(0, 0, 1));\r\n const pathTransformToZ = new Matrix();\r\n const pathTransformToZInv = new Matrix();\r\n rotToZ.toRotationMatrix(pathTransformToZ);\r\n pathTransformToZ.invertToRef(pathTransformToZInv);\r\n\r\n for (let i = 0; i < shape.length; i++) {\r\n Vector3.TransformCoordinatesToRef(shape[i], pathTransformToZ, shape[i]);\r\n }\r\n\r\n const shape2d = shape.map(point => new Vector2(point.x, point.y));\r\n\r\n const builder = new PolygonMeshBuilder(feature._dynamicId, shape2d, this.scene, earcut);\r\n if (shapeHoles) {\r\n shapeHoles.forEach(holePoints => {\r\n for (let i = 0; i < holePoints.length; i++) {\r\n Vector3.TransformCoordinatesToRef(holePoints[i], pathTransformToZ, holePoints[i]);\r\n }\r\n const hole2d = holePoints.map(point => new Vector2(point.x, point.y));\r\n builder.addHole(hole2d);\r\n });\r\n }\r\n const extruded = builder.build(true, Math.abs(feature.size), 0.9);\r\n xztoxy.multiplyToRef(pathTransformToZInv, pathTransformToZInv);\r\n\r\n extruded.bakeTransformIntoVertices(pathTransformToZInv);\r\n extrudedMeshes.push(extruded);\r\n }\r\n\r\n const extrudedAll = Mesh.MergeMeshes(extrudedMeshes, true);\r\n if (extrudedAll) {\r\n applyExtrusion(newVertexData, extrudedAll, false);\r\n extrudedAll.dispose();\r\n }\r\n }\r\n\r\n return newVertexData;\r\n }\r\n\r\n protected getSketchShape(positions: FloatArray) {\r\n const shape: Vector3[] = [];\r\n for (let i = 0; i < positions.length; i += 3) {\r\n const newPoint = new Vector3(positions[i], positions[i + 1], positions[i + 2]);\r\n if (shape.length === 0 || !shape[shape.length - 1].equals(newPoint)) {\r\n shape.push(newPoint);\r\n }\r\n }\r\n\r\n return shape;\r\n }\r\n\r\n update3dInternal() {}\r\n}\r\n\r\nexport function applyExtrusion(newVertexData: Partial, extruded: Mesh, reverse = false) {\r\n if (extruded.geometry) {\r\n // Store the indices used for the extrude so that they can be merged into the new geometry\r\n const newIndices = extruded.geometry.getIndices(false, true);\r\n const newNormals = extruded.geometry.getVerticesData(VertexBuffer.NormalKind, false, true);\r\n const newUvs = extruded.geometry.getVerticesData(VertexBuffer.UVKind, false, true);\r\n const newPositions = extruded.geometry.getVerticesData(VertexBuffer.PositionKind, false, true);\r\n if (newPositions && reverse) {\r\n for (let i = 2; i < newPositions.length; i += 3) {\r\n newPositions[i] = -newPositions[i];\r\n }\r\n }\r\n\r\n normalizeGeometry(newVertexData);\r\n\r\n if (!newVertexData.indices) {\r\n newVertexData.indices = newIndices;\r\n } else if (newIndices) {\r\n const concatArr = new Int32Array(newVertexData.indices.length + newIndices.length);\r\n concatArr.set(newVertexData.indices);\r\n if (newVertexData.positions) {\r\n const pushAmount = newVertexData.positions.length / 3;\r\n for (let i = 0; i < newIndices.length; i++) {\r\n newIndices[i] += pushAmount;\r\n }\r\n }\r\n concatArr.set(newIndices, newVertexData.indices.length);\r\n newVertexData.indices = concatArr;\r\n }\r\n\r\n if (!newVertexData.positions) {\r\n newVertexData.positions = newPositions;\r\n } else if (newPositions) {\r\n const concatArr = new Float32Array(newVertexData.positions.length + newPositions.length);\r\n concatArr.set(newVertexData.positions);\r\n concatArr.set(newPositions, newVertexData.positions.length);\r\n newVertexData.positions = concatArr;\r\n }\r\n\r\n if (!newVertexData.normals) {\r\n newVertexData.normals = newNormals;\r\n } else if (newNormals) {\r\n const concatArr = new Float32Array(newVertexData.normals.length + newNormals.length);\r\n concatArr.set(newVertexData.normals);\r\n concatArr.set(newNormals, newVertexData.normals.length);\r\n newVertexData.normals = concatArr;\r\n }\r\n\r\n if (!newVertexData.uvs) {\r\n newVertexData.uvs = newUvs;\r\n } else if (newUvs) {\r\n const concatArr = new Float32Array(newVertexData.uvs.length + newUvs.length);\r\n concatArr.set(newVertexData.uvs);\r\n concatArr.set(newUvs, newVertexData.uvs.length);\r\n newVertexData.uvs = concatArr;\r\n }\r\n\r\n const baseMaterialIndex = newVertexData.submeshData ? newVertexData.submeshData[0].materialIndex : 0;\r\n const totalIndicesResultCount = newVertexData.indices!.length;\r\n const totalVerticesResultCount = newVertexData.positions!.length / 3;\r\n\r\n newVertexData.submeshData = [];\r\n newVertexData.submeshData.push({\r\n materialIndex: baseMaterialIndex,\r\n verticesStart: 0,\r\n verticesCount: totalVerticesResultCount,\r\n indexStart: 0,\r\n indexCount: totalIndicesResultCount,\r\n });\r\n }\r\n}\r\n\r\nexport function normalizeGeometry(newVertexData: Partial) {\r\n if (!newVertexData.indices) {\r\n // If there are no indices, throw away existing vertices because they won't be used\r\n newVertexData.positions = null;\r\n newVertexData.normals = null;\r\n newVertexData.uvs = null;\r\n } else {\r\n const indices = newVertexData.indices;\r\n let maxIdx = 0;\r\n for (let i = 0; i < indices.length; i++) {\r\n if (maxIdx < indices[i]) {\r\n maxIdx = indices[i];\r\n }\r\n }\r\n const usedVertexLength = maxIdx + 1;\r\n\r\n const positions = new Float32Array(usedVertexLength * 3);\r\n const normals = new Float32Array(usedVertexLength * 3);\r\n const uvs = new Float32Array(usedVertexLength * 2);\r\n\r\n if (newVertexData.positions) {\r\n if (Array.isArray(newVertexData.positions)) {\r\n positions.set(newVertexData.positions.slice(0, positions.length));\r\n } else {\r\n positions.set(newVertexData.positions.subarray(0, positions.length));\r\n }\r\n }\r\n if (newVertexData.normals) {\r\n if (Array.isArray(newVertexData.normals)) {\r\n normals.set(newVertexData.normals.slice(0, normals.length));\r\n } else {\r\n normals.set(newVertexData.normals.subarray(0, normals.length));\r\n }\r\n }\r\n if (newVertexData.uvs) {\r\n if (Array.isArray(newVertexData.uvs)) {\r\n uvs.set(newVertexData.uvs.slice(0, uvs.length));\r\n } else {\r\n uvs.set(newVertexData.uvs.subarray(0, uvs.length));\r\n }\r\n }\r\n newVertexData.positions = positions;\r\n newVertexData.normals = normals;\r\n newVertexData.uvs = uvs;\r\n }\r\n}\r\n","import type { Nullable, FloatArray, IndicesArray } from \"../types\";\r\nimport type { Scene } from \"../scene\";\r\nimport { Quaternion, Matrix, Vector3, Vector2 } from \"../Maths/math.vector\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport { SubMesh } from \"../Meshes/subMesh\";\r\nimport { Mesh } from \"../Meshes/mesh\";\r\nimport type { Material } from \"../Materials/material\";\r\nimport { Color4 } from \"../Maths/math.color\";\r\nimport { Constants } from \"../Engines/constants\";\r\n/**\r\n * Unique ID when we import meshes from Babylon to CSG\r\n */\r\nlet currentCSGMeshId = 0;\r\n\r\n/**\r\n * Represents a vertex of a polygon. Use your own vertex class instead of this\r\n * one to provide additional features like texture coordinates and vertex\r\n * colors. Custom vertex classes need to provide a `pos` property and `clone()`,\r\n * `flip()`, and `interpolate()` methods that behave analogous to the ones\r\n * defined by `BABYLON.CSG.Vertex`. This class provides `normal` so convenience\r\n * functions like `BABYLON.CSG.sphere()` can return a smooth vertex normal, but `normal`\r\n * is not used anywhere else.\r\n * Same goes for uv, it allows to keep the original vertex uv coordinates of the 2 meshes\r\n */\r\nclass Vertex {\r\n /**\r\n * Initializes the vertex\r\n * @param pos The position of the vertex\r\n * @param normal The normal of the vertex\r\n * @param uv The texture coordinate of the vertex\r\n * @param vertColor The RGBA color of the vertex\r\n */\r\n constructor(\r\n /**\r\n * The position of the vertex\r\n */\r\n public pos: Vector3,\r\n /**\r\n * The normal of the vertex\r\n */\r\n public normal: Vector3,\r\n /**\r\n * The texture coordinate of the vertex\r\n */\r\n public uv?: Vector2,\r\n /**\r\n * The texture coordinate of the vertex\r\n */\r\n public vertColor?: Color4\r\n ) {}\r\n\r\n /**\r\n * Make a clone, or deep copy, of the vertex\r\n * @returns A new Vertex\r\n */\r\n public clone(): Vertex {\r\n return new Vertex(this.pos.clone(), this.normal.clone(), this.uv?.clone(), this.vertColor?.clone());\r\n }\r\n\r\n /**\r\n * Invert all orientation-specific data (e.g. vertex normal). Called when the\r\n * orientation of a polygon is flipped.\r\n */\r\n public flip(): void {\r\n this.normal = this.normal.scale(-1);\r\n }\r\n\r\n /**\r\n * Create a new vertex between this vertex and `other` by linearly\r\n * interpolating all properties using a parameter of `t`. Subclasses should\r\n * override this to interpolate additional properties.\r\n * @param other the vertex to interpolate against\r\n * @param t The factor used to linearly interpolate between the vertices\r\n */\r\n public interpolate(other: Vertex, t: number): Vertex {\r\n return new Vertex(\r\n Vector3.Lerp(this.pos, other.pos, t),\r\n Vector3.Lerp(this.normal, other.normal, t),\r\n this.uv && other.uv ? Vector2.Lerp(this.uv, other.uv, t) : undefined,\r\n this.vertColor && other.vertColor ? Color4.Lerp(this.vertColor, other.vertColor, t) : undefined\r\n );\r\n }\r\n}\r\n\r\n/**\r\n * Represents a plane in 3D space.\r\n */\r\nclass Plane {\r\n /**\r\n * Initializes the plane\r\n * @param normal The normal for the plane\r\n * @param w\r\n */\r\n constructor(public normal: Vector3, public w: number) {}\r\n\r\n /**\r\n * `CSG.Plane.EPSILON` is the tolerance used by `splitPolygon()` to decide if a\r\n * point is on the plane\r\n */\r\n static EPSILON = 1e-5;\r\n\r\n /**\r\n * Construct a plane from three points\r\n * @param a Point a\r\n * @param b Point b\r\n * @param c Point c\r\n */\r\n public static FromPoints(a: Vector3, b: Vector3, c: Vector3): Nullable {\r\n const v0 = c.subtract(a);\r\n const v1 = b.subtract(a);\r\n\r\n if (v0.lengthSquared() === 0 || v1.lengthSquared() === 0) {\r\n return null;\r\n }\r\n\r\n const n = Vector3.Normalize(Vector3.Cross(v0, v1));\r\n return new Plane(n, Vector3.Dot(n, a));\r\n }\r\n\r\n /**\r\n * Clone, or make a deep copy of the plane\r\n * @returns a new Plane\r\n */\r\n public clone(): Plane {\r\n return new Plane(this.normal.clone(), this.w);\r\n }\r\n\r\n /**\r\n * Flip the face of the plane\r\n */\r\n public flip() {\r\n this.normal.scaleInPlace(-1);\r\n this.w = -this.w;\r\n }\r\n\r\n /**\r\n * Split `polygon` by this plane if needed, then put the polygon or polygon\r\n * fragments in the appropriate lists. Coplanar polygons go into either\r\n `* coplanarFront` or `coplanarBack` depending on their orientation with\r\n * respect to this plane. Polygons in front or in back of this plane go into\r\n * either `front` or `back`\r\n * @param polygon The polygon to be split\r\n * @param coplanarFront Will contain polygons coplanar with the plane that are oriented to the front of the plane\r\n * @param coplanarBack Will contain polygons coplanar with the plane that are oriented to the back of the plane\r\n * @param front Will contain the polygons in front of the plane\r\n * @param back Will contain the polygons begind the plane\r\n */\r\n public splitPolygon(polygon: Polygon, coplanarFront: Polygon[], coplanarBack: Polygon[], front: Polygon[], back: Polygon[]): void {\r\n const COPLANAR = 0;\r\n const FRONT = 1;\r\n const BACK = 2;\r\n const SPANNING = 3;\r\n\r\n // Classify each point as well as the entire polygon into one of the above\r\n // four classes.\r\n let polygonType = 0;\r\n const types = [];\r\n let i: number;\r\n let t: number;\r\n for (i = 0; i < polygon.vertices.length; i++) {\r\n t = Vector3.Dot(this.normal, polygon.vertices[i].pos) - this.w;\r\n const type = t < -Plane.EPSILON ? BACK : t > Plane.EPSILON ? FRONT : COPLANAR;\r\n polygonType |= type;\r\n types.push(type);\r\n }\r\n\r\n // Put the polygon in the correct list, splitting it when necessary\r\n switch (polygonType) {\r\n case COPLANAR:\r\n (Vector3.Dot(this.normal, polygon.plane.normal) > 0 ? coplanarFront : coplanarBack).push(polygon);\r\n break;\r\n case FRONT:\r\n front.push(polygon);\r\n break;\r\n case BACK:\r\n back.push(polygon);\r\n break;\r\n case SPANNING: {\r\n const f = [],\r\n b = [];\r\n for (i = 0; i < polygon.vertices.length; i++) {\r\n const j = (i + 1) % polygon.vertices.length;\r\n const ti = types[i],\r\n tj = types[j];\r\n const vi = polygon.vertices[i],\r\n vj = polygon.vertices[j];\r\n if (ti !== BACK) {\r\n f.push(vi);\r\n }\r\n if (ti !== FRONT) {\r\n b.push(ti !== BACK ? vi.clone() : vi);\r\n }\r\n if ((ti | tj) === SPANNING) {\r\n t = (this.w - Vector3.Dot(this.normal, vi.pos)) / Vector3.Dot(this.normal, vj.pos.subtract(vi.pos));\r\n const v = vi.interpolate(vj, t);\r\n f.push(v);\r\n b.push(v.clone());\r\n }\r\n }\r\n let poly: Polygon;\r\n if (f.length >= 3) {\r\n poly = new Polygon(f, polygon.shared);\r\n if (poly.plane) {\r\n front.push(poly);\r\n }\r\n }\r\n\r\n if (b.length >= 3) {\r\n poly = new Polygon(b, polygon.shared);\r\n\r\n if (poly.plane) {\r\n back.push(poly);\r\n }\r\n }\r\n\r\n break;\r\n }\r\n }\r\n }\r\n}\r\n\r\n/**\r\n * Represents a convex polygon. The vertices used to initialize a polygon must\r\n * be coplanar and form a convex loop.\r\n *\r\n * Each convex polygon has a `shared` property, which is shared between all\r\n * polygons that are clones of each other or were split from the same polygon.\r\n * This can be used to define per-polygon properties (such as surface color)\r\n */\r\nclass Polygon {\r\n /**\r\n * Vertices of the polygon\r\n */\r\n public vertices: Vertex[];\r\n /**\r\n * Properties that are shared across all polygons\r\n */\r\n public shared: any;\r\n /**\r\n * A plane formed from the vertices of the polygon\r\n */\r\n public plane: Plane;\r\n\r\n /**\r\n * Initializes the polygon\r\n * @param vertices The vertices of the polygon\r\n * @param shared The properties shared across all polygons\r\n */\r\n constructor(vertices: Vertex[], shared: any) {\r\n this.vertices = vertices;\r\n this.shared = shared;\r\n this.plane = Plane.FromPoints(vertices[0].pos, vertices[1].pos, vertices[2].pos);\r\n }\r\n\r\n /**\r\n * Clones, or makes a deep copy, or the polygon\r\n */\r\n public clone(): Polygon {\r\n const vertices = this.vertices.map((v) => v.clone());\r\n return new Polygon(vertices, this.shared);\r\n }\r\n\r\n /**\r\n * Flips the faces of the polygon\r\n */\r\n public flip() {\r\n this.vertices.reverse().map((v) => {\r\n v.flip();\r\n });\r\n this.plane.flip();\r\n }\r\n}\r\n\r\n/**\r\n * Holds a node in a BSP tree. A BSP tree is built from a collection of polygons\r\n * by picking a polygon to split along. That polygon (and all other coplanar\r\n * polygons) are added directly to that node and the other polygons are added to\r\n * the front and/or back subtrees. This is not a leafy BSP tree since there is\r\n * no distinction between internal and leaf nodes\r\n */\r\nclass Node {\r\n private _plane: Nullable = null;\r\n private _front: Nullable = null;\r\n private _back: Nullable = null;\r\n private _polygons = new Array();\r\n\r\n /**\r\n * Initializes the node\r\n * @param polygons A collection of polygons held in the node\r\n */\r\n constructor(polygons?: Array) {\r\n if (polygons) {\r\n this.build(polygons);\r\n }\r\n }\r\n\r\n /**\r\n * Clones, or makes a deep copy, of the node\r\n * @returns The cloned node\r\n */\r\n public clone(): Node {\r\n const node = new Node();\r\n node._plane = this._plane && this._plane.clone();\r\n node._front = this._front && this._front.clone();\r\n node._back = this._back && this._back.clone();\r\n node._polygons = this._polygons.map((p) => p.clone());\r\n return node;\r\n }\r\n\r\n /**\r\n * Convert solid space to empty space and empty space to solid space\r\n */\r\n public invert(): void {\r\n for (let i = 0; i < this._polygons.length; i++) {\r\n this._polygons[i].flip();\r\n }\r\n if (this._plane) {\r\n this._plane.flip();\r\n }\r\n if (this._front) {\r\n this._front.invert();\r\n }\r\n if (this._back) {\r\n this._back.invert();\r\n }\r\n const temp = this._front;\r\n this._front = this._back;\r\n this._back = temp;\r\n }\r\n\r\n /**\r\n * Recursively remove all polygons in `polygons` that are inside this BSP\r\n * tree.\r\n * @param polygons Polygons to remove from the BSP\r\n * @returns Polygons clipped from the BSP\r\n */\r\n clipPolygons(polygons: Polygon[]): Polygon[] {\r\n if (!this._plane) {\r\n return polygons.slice();\r\n }\r\n let front = new Array(),\r\n back = new Array();\r\n for (let i = 0; i < polygons.length; i++) {\r\n this._plane.splitPolygon(polygons[i], front, back, front, back);\r\n }\r\n if (this._front) {\r\n front = this._front.clipPolygons(front);\r\n }\r\n if (this._back) {\r\n back = this._back.clipPolygons(back);\r\n } else {\r\n back = [];\r\n }\r\n return front.concat(back);\r\n }\r\n\r\n /**\r\n * Remove all polygons in this BSP tree that are inside the other BSP tree\r\n * `bsp`.\r\n * @param bsp BSP containing polygons to remove from this BSP\r\n */\r\n clipTo(bsp: Node): void {\r\n this._polygons = bsp.clipPolygons(this._polygons);\r\n if (this._front) {\r\n this._front.clipTo(bsp);\r\n }\r\n if (this._back) {\r\n this._back.clipTo(bsp);\r\n }\r\n }\r\n\r\n /**\r\n * Return a list of all polygons in this BSP tree\r\n * @returns List of all polygons in this BSP tree\r\n */\r\n allPolygons(): Polygon[] {\r\n let polygons = this._polygons.slice();\r\n if (this._front) {\r\n polygons = polygons.concat(this._front.allPolygons());\r\n }\r\n if (this._back) {\r\n polygons = polygons.concat(this._back.allPolygons());\r\n }\r\n return polygons;\r\n }\r\n\r\n /**\r\n * Build a BSP tree out of `polygons`. When called on an existing tree, the\r\n * new polygons are filtered down to the bottom of the tree and become new\r\n * nodes there. Each set of polygons is partitioned using the first polygon\r\n * (no heuristic is used to pick a good split)\r\n * @param polygons Polygons used to construct the BSP tree\r\n */\r\n build(polygons: Polygon[]): void {\r\n if (!polygons.length) {\r\n return;\r\n }\r\n if (!this._plane) {\r\n this._plane = polygons[0].plane.clone();\r\n }\r\n const front = new Array(),\r\n back = new Array();\r\n for (let i = 0; i < polygons.length; i++) {\r\n this._plane.splitPolygon(polygons[i], this._polygons, this._polygons, front, back);\r\n }\r\n if (front.length) {\r\n if (!this._front) {\r\n this._front = new Node();\r\n }\r\n this._front.build(front);\r\n }\r\n if (back.length) {\r\n if (!this._back) {\r\n this._back = new Node();\r\n }\r\n this._back.build(back);\r\n }\r\n }\r\n}\r\n\r\n/**\r\n * Class for building Constructive Solid Geometry\r\n */\r\nexport class CSG {\r\n private _polygons = new Array();\r\n /**\r\n * The world matrix\r\n */\r\n public matrix: Matrix;\r\n /**\r\n * Stores the position\r\n */\r\n public position: Vector3;\r\n /**\r\n * Stores the rotation\r\n */\r\n public rotation: Vector3;\r\n /**\r\n * Stores the rotation quaternion\r\n */\r\n public rotationQuaternion: Nullable;\r\n /**\r\n * Stores the scaling vector\r\n */\r\n public scaling: Vector3;\r\n\r\n /**\r\n * Convert the Mesh to CSG\r\n * @param mesh The Mesh to convert to CSG\r\n * @param absolute If true, the final (local) matrix transformation is set to the identity and not to that of `mesh`. It can help when dealing with right-handed meshes (default: false)\r\n * @returns A new CSG from the Mesh\r\n */\r\n public static FromMesh(mesh: Mesh, absolute = false): CSG {\r\n let vertex: Vertex,\r\n normal: Vector3,\r\n uv: Vector2 | undefined = undefined,\r\n position: Vector3,\r\n vertColor: Color4 | undefined = undefined,\r\n polygon: Polygon,\r\n vertices;\r\n const polygons = new Array();\r\n let matrix: Matrix,\r\n meshPosition: Vector3,\r\n meshRotation: Vector3,\r\n meshRotationQuaternion: Nullable = null,\r\n meshScaling: Vector3;\r\n\r\n let invertWinding = false;\r\n if (mesh instanceof Mesh) {\r\n mesh.computeWorldMatrix(true);\r\n matrix = mesh.getWorldMatrix();\r\n meshPosition = mesh.position.clone();\r\n meshRotation = mesh.rotation.clone();\r\n if (mesh.rotationQuaternion) {\r\n meshRotationQuaternion = mesh.rotationQuaternion.clone();\r\n }\r\n meshScaling = mesh.scaling.clone();\r\n if (mesh.material && absolute) {\r\n invertWinding = mesh.material.sideOrientation === Constants.MATERIAL_ClockWiseSideOrientation;\r\n }\r\n } else {\r\n throw \"BABYLON.CSG: Wrong Mesh type, must be BABYLON.Mesh\";\r\n }\r\n\r\n const indices = mesh.getIndices(),\r\n positions = mesh.getVerticesData(VertexBuffer.PositionKind),\r\n normals = mesh.getVerticesData(VertexBuffer.NormalKind),\r\n uvs = mesh.getVerticesData(VertexBuffer.UVKind),\r\n vertColors = mesh.getVerticesData(VertexBuffer.ColorKind);\r\n\r\n const subMeshes = mesh.subMeshes;\r\n\r\n for (let sm = 0, sml = subMeshes.length; sm < sml; sm++) {\r\n for (let i = subMeshes[sm].indexStart, il = subMeshes[sm].indexCount + subMeshes[sm].indexStart; i < il; i += 3) {\r\n vertices = [];\r\n for (let j = 0; j < 3; j++) {\r\n const indexIndices = j === 0 ? i + j : invertWinding ? i + 3 - j : i + j;\r\n const sourceNormal = new Vector3(normals[indices[indexIndices] * 3], normals[indices[indexIndices] * 3 + 1], normals[indices[indexIndices] * 3 + 2]);\r\n if (uvs) {\r\n uv = new Vector2(uvs[indices[indexIndices] * 2], uvs[indices[indexIndices] * 2 + 1]);\r\n }\r\n if (vertColors) {\r\n vertColor = new Color4(\r\n vertColors[indices[indexIndices] * 4],\r\n vertColors[indices[indexIndices] * 4 + 1],\r\n vertColors[indices[indexIndices] * 4 + 2],\r\n vertColors[indices[indexIndices] * 4 + 3]\r\n );\r\n }\r\n const sourcePosition = new Vector3(positions[indices[indexIndices] * 3], positions[indices[indexIndices] * 3 + 1], positions[indices[indexIndices] * 3 + 2]);\r\n position = Vector3.TransformCoordinates(sourcePosition, matrix);\r\n normal = Vector3.TransformNormal(sourceNormal, matrix);\r\n\r\n vertex = new Vertex(position, normal, uv, vertColor);\r\n vertices.push(vertex);\r\n }\r\n\r\n polygon = new Polygon(vertices, { subMeshId: sm, meshId: currentCSGMeshId, materialIndex: subMeshes[sm].materialIndex });\r\n\r\n // To handle the case of degenerated triangle\r\n // polygon.plane == null <=> the polygon does not represent 1 single plane <=> the triangle is degenerated\r\n if (polygon.plane) {\r\n polygons.push(polygon);\r\n }\r\n }\r\n }\r\n\r\n const csg = CSG._FromPolygons(polygons);\r\n csg.matrix = absolute ? Matrix.Identity() : matrix;\r\n csg.position = absolute ? Vector3.Zero() : meshPosition;\r\n csg.rotation = absolute ? Vector3.Zero() : meshRotation;\r\n csg.scaling = absolute ? Vector3.One() : meshScaling;\r\n csg.rotationQuaternion = absolute && meshRotationQuaternion ? Quaternion.Identity() : meshRotationQuaternion;\r\n currentCSGMeshId++;\r\n\r\n return csg;\r\n }\r\n\r\n /**\r\n * Construct a CSG solid from a list of `CSG.Polygon` instances.\r\n * @param polygons Polygons used to construct a CSG solid\r\n */\r\n private static _FromPolygons(polygons: Polygon[]): CSG {\r\n const csg = new CSG();\r\n csg._polygons = polygons;\r\n return csg;\r\n }\r\n\r\n /**\r\n * Clones, or makes a deep copy, of the CSG\r\n * @returns A new CSG\r\n */\r\n public clone(): CSG {\r\n const csg = new CSG();\r\n csg._polygons = this._polygons.map((p) => p.clone());\r\n csg.copyTransformAttributes(this);\r\n return csg;\r\n }\r\n\r\n /**\r\n * Unions this CSG with another CSG\r\n * @param csg The CSG to union against this CSG\r\n * @returns The unioned CSG\r\n */\r\n public union(csg: CSG): CSG {\r\n const a = new Node(this.clone()._polygons);\r\n const b = new Node(csg.clone()._polygons);\r\n a.clipTo(b);\r\n b.clipTo(a);\r\n b.invert();\r\n b.clipTo(a);\r\n b.invert();\r\n a.build(b.allPolygons());\r\n return CSG._FromPolygons(a.allPolygons()).copyTransformAttributes(this);\r\n }\r\n\r\n /**\r\n * Unions this CSG with another CSG in place\r\n * @param csg The CSG to union against this CSG\r\n */\r\n public unionInPlace(csg: CSG): void {\r\n const a = new Node(this._polygons);\r\n const b = new Node(csg._polygons);\r\n\r\n a.clipTo(b);\r\n b.clipTo(a);\r\n b.invert();\r\n b.clipTo(a);\r\n b.invert();\r\n a.build(b.allPolygons());\r\n\r\n this._polygons = a.allPolygons();\r\n }\r\n\r\n /**\r\n * Subtracts this CSG with another CSG\r\n * @param csg The CSG to subtract against this CSG\r\n * @returns A new CSG\r\n */\r\n public subtract(csg: CSG): CSG {\r\n const a = new Node(this.clone()._polygons);\r\n const b = new Node(csg.clone()._polygons);\r\n a.invert();\r\n a.clipTo(b);\r\n b.clipTo(a);\r\n b.invert();\r\n b.clipTo(a);\r\n b.invert();\r\n a.build(b.allPolygons());\r\n a.invert();\r\n return CSG._FromPolygons(a.allPolygons()).copyTransformAttributes(this);\r\n }\r\n\r\n /**\r\n * Subtracts this CSG with another CSG in place\r\n * @param csg The CSG to subtract against this CSG\r\n */\r\n public subtractInPlace(csg: CSG): void {\r\n const a = new Node(this._polygons);\r\n const b = new Node(csg._polygons);\r\n\r\n a.invert();\r\n a.clipTo(b);\r\n b.clipTo(a);\r\n b.invert();\r\n b.clipTo(a);\r\n b.invert();\r\n a.build(b.allPolygons());\r\n a.invert();\r\n\r\n this._polygons = a.allPolygons();\r\n }\r\n\r\n /**\r\n * Intersect this CSG with another CSG\r\n * @param csg The CSG to intersect against this CSG\r\n * @returns A new CSG\r\n */\r\n public intersect(csg: CSG): CSG {\r\n const a = new Node(this.clone()._polygons);\r\n const b = new Node(csg.clone()._polygons);\r\n a.invert();\r\n b.clipTo(a);\r\n b.invert();\r\n a.clipTo(b);\r\n b.clipTo(a);\r\n a.build(b.allPolygons());\r\n a.invert();\r\n return CSG._FromPolygons(a.allPolygons()).copyTransformAttributes(this);\r\n }\r\n\r\n /**\r\n * Intersects this CSG with another CSG in place\r\n * @param csg The CSG to intersect against this CSG\r\n */\r\n public intersectInPlace(csg: CSG): void {\r\n const a = new Node(this._polygons);\r\n const b = new Node(csg._polygons);\r\n\r\n a.invert();\r\n b.clipTo(a);\r\n b.invert();\r\n a.clipTo(b);\r\n b.clipTo(a);\r\n a.build(b.allPolygons());\r\n a.invert();\r\n\r\n this._polygons = a.allPolygons();\r\n }\r\n\r\n /**\r\n * Return a new CSG solid with solid and empty space switched. This solid is\r\n * not modified.\r\n * @returns A new CSG solid with solid and empty space switched\r\n */\r\n public inverse(): CSG {\r\n const csg = this.clone();\r\n csg.inverseInPlace();\r\n return csg;\r\n }\r\n\r\n /**\r\n * Inverses the CSG in place\r\n */\r\n public inverseInPlace(): void {\r\n this._polygons.map((p) => {\r\n p.flip();\r\n });\r\n }\r\n\r\n /**\r\n * This is used to keep meshes transformations so they can be restored\r\n * when we build back a Babylon Mesh\r\n * NB : All CSG operations are performed in world coordinates\r\n * @param csg The CSG to copy the transform attributes from\r\n * @returns This CSG\r\n */\r\n public copyTransformAttributes(csg: CSG): CSG {\r\n this.matrix = csg.matrix;\r\n this.position = csg.position;\r\n this.rotation = csg.rotation;\r\n this.scaling = csg.scaling;\r\n this.rotationQuaternion = csg.rotationQuaternion;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Build Raw mesh from CSG\r\n * Coordinates here are in world space\r\n * @param name The name of the mesh geometry\r\n * @param scene The Scene\r\n * @param keepSubMeshes Specifies if the submeshes should be kept\r\n * @returns A new Mesh\r\n */\r\n public buildMeshGeometry(name: string, scene?: Scene, keepSubMeshes?: boolean): Mesh {\r\n const matrix = this.matrix.clone();\r\n matrix.invert();\r\n\r\n const mesh = new Mesh(name, scene);\r\n const vertices = [];\r\n const indices = [];\r\n const normals = [];\r\n let uvs: Nullable = null;\r\n let vertColors: Nullable = null;\r\n const vertex = Vector3.Zero();\r\n const normal = Vector3.Zero();\r\n const uv = Vector2.Zero();\r\n const vertColor = new Color4(0, 0, 0, 0);\r\n const polygons = this._polygons;\r\n const polygonIndices = [0, 0, 0];\r\n let polygon;\r\n const vertice_dict = {};\r\n let vertex_idx;\r\n let currentIndex = 0;\r\n const subMeshDict = {};\r\n let subMeshObj;\r\n\r\n if (keepSubMeshes) {\r\n // Sort Polygons, since subMeshes are indices range\r\n polygons.sort((a, b) => {\r\n if (a.shared.meshId === b.shared.meshId) {\r\n return a.shared.subMeshId - b.shared.subMeshId;\r\n } else {\r\n return a.shared.meshId - b.shared.meshId;\r\n }\r\n });\r\n }\r\n\r\n for (let i = 0, il = polygons.length; i < il; i++) {\r\n polygon = polygons[i];\r\n\r\n // Building SubMeshes\r\n if (!(subMeshDict)[polygon.shared.meshId]) {\r\n (subMeshDict)[polygon.shared.meshId] = {};\r\n }\r\n if (!(subMeshDict)[polygon.shared.meshId][polygon.shared.subMeshId]) {\r\n (subMeshDict)[polygon.shared.meshId][polygon.shared.subMeshId] = {\r\n indexStart: +Infinity,\r\n indexEnd: -Infinity,\r\n materialIndex: polygon.shared.materialIndex,\r\n };\r\n }\r\n subMeshObj = (subMeshDict)[polygon.shared.meshId][polygon.shared.subMeshId];\r\n\r\n for (let j = 2, jl = polygon.vertices.length; j < jl; j++) {\r\n polygonIndices[0] = 0;\r\n polygonIndices[1] = j - 1;\r\n polygonIndices[2] = j;\r\n\r\n for (let k = 0; k < 3; k++) {\r\n vertex.copyFrom(polygon.vertices[polygonIndices[k]].pos);\r\n normal.copyFrom(polygon.vertices[polygonIndices[k]].normal);\r\n if (polygon.vertices[polygonIndices[k]].uv) {\r\n if (!uvs) {\r\n uvs = [];\r\n }\r\n uv.copyFrom(polygon.vertices[polygonIndices[k]].uv!);\r\n }\r\n\r\n if (polygon.vertices[polygonIndices[k]].vertColor) {\r\n if (!vertColors) {\r\n vertColors = [];\r\n }\r\n vertColor.copyFrom(polygon.vertices[polygonIndices[k]].vertColor!);\r\n }\r\n const localVertex = Vector3.TransformCoordinates(vertex, matrix);\r\n const localNormal = Vector3.TransformNormal(normal, matrix);\r\n\r\n vertex_idx = (vertice_dict)[localVertex.x + \",\" + localVertex.y + \",\" + localVertex.z];\r\n\r\n let areUvsDifferent = false;\r\n\r\n if (uvs && !(uvs[vertex_idx * 2] === uv.x || uvs[vertex_idx * 2 + 1] === uv.y)) {\r\n areUvsDifferent = true;\r\n }\r\n\r\n let areColorsDifferent = false;\r\n\r\n if (\r\n vertColors &&\r\n !(\r\n vertColors[vertex_idx * 4] === vertColor.r ||\r\n vertColors[vertex_idx * 4 + 1] === vertColor.g ||\r\n vertColors[vertex_idx * 4 + 2] === vertColor.b ||\r\n vertColors[vertex_idx * 4 + 3] === vertColor.a\r\n )\r\n ) {\r\n areColorsDifferent = true;\r\n }\r\n\r\n // Check if 2 points can be merged\r\n if (\r\n !(\r\n typeof vertex_idx !== \"undefined\" &&\r\n normals[vertex_idx * 3] === localNormal.x &&\r\n normals[vertex_idx * 3 + 1] === localNormal.y &&\r\n normals[vertex_idx * 3 + 2] === localNormal.z\r\n ) ||\r\n areUvsDifferent ||\r\n areColorsDifferent\r\n ) {\r\n vertices.push(localVertex.x, localVertex.y, localVertex.z);\r\n if (uvs) {\r\n uvs.push(uv.x, uv.y);\r\n }\r\n normals.push(normal.x, normal.y, normal.z);\r\n if (vertColors) {\r\n vertColors.push(vertColor.r, vertColor.g, vertColor.b, vertColor.a);\r\n }\r\n vertex_idx = (vertice_dict)[localVertex.x + \",\" + localVertex.y + \",\" + localVertex.z] = vertices.length / 3 - 1;\r\n }\r\n\r\n indices.push(vertex_idx);\r\n\r\n subMeshObj.indexStart = Math.min(currentIndex, subMeshObj.indexStart);\r\n subMeshObj.indexEnd = Math.max(currentIndex, subMeshObj.indexEnd);\r\n currentIndex++;\r\n }\r\n }\r\n }\r\n\r\n mesh.setVerticesData(VertexBuffer.PositionKind, vertices);\r\n mesh.setVerticesData(VertexBuffer.NormalKind, normals);\r\n if (uvs) {\r\n mesh.setVerticesData(VertexBuffer.UVKind, uvs);\r\n }\r\n if (vertColors) {\r\n mesh.setVerticesData(VertexBuffer.ColorKind, vertColors);\r\n }\r\n mesh.setIndices(indices, null);\r\n\r\n if (keepSubMeshes) {\r\n // We offset the materialIndex by the previous number of materials in the CSG mixed meshes\r\n let materialIndexOffset = 0,\r\n materialMaxIndex;\r\n\r\n mesh.subMeshes = new Array();\r\n\r\n for (const m in subMeshDict) {\r\n materialMaxIndex = -1;\r\n for (const sm in (subMeshDict)[m]) {\r\n subMeshObj = (subMeshDict)[m][sm];\r\n SubMesh.CreateFromIndices(\r\n subMeshObj.materialIndex + materialIndexOffset,\r\n subMeshObj.indexStart,\r\n subMeshObj.indexEnd - subMeshObj.indexStart + 1,\r\n mesh\r\n );\r\n materialMaxIndex = Math.max(subMeshObj.materialIndex, materialMaxIndex);\r\n }\r\n materialIndexOffset += ++materialMaxIndex;\r\n }\r\n }\r\n\r\n return mesh;\r\n }\r\n\r\n /**\r\n * Build Mesh from CSG taking material and transforms into account\r\n * @param name The name of the Mesh\r\n * @param material The material of the Mesh\r\n * @param scene The Scene\r\n * @param keepSubMeshes Specifies if submeshes should be kept\r\n * @returns The new Mesh\r\n */\r\n public toMesh(name: string, material: Nullable = null, scene?: Scene, keepSubMeshes?: boolean): Mesh {\r\n const mesh = this.buildMeshGeometry(name, scene, keepSubMeshes);\r\n\r\n mesh.material = material;\r\n\r\n mesh.position.copyFrom(this.position);\r\n mesh.rotation.copyFrom(this.rotation);\r\n if (this.rotationQuaternion) {\r\n mesh.rotationQuaternion = this.rotationQuaternion.clone();\r\n }\r\n mesh.scaling.copyFrom(this.scaling);\r\n mesh.computeWorldMatrix(true);\r\n\r\n return mesh;\r\n }\r\n}\r\n","import { StandardMaterial } from '@babylonjs/core/Materials/standardMaterial';\r\nimport { Color3 } from '@babylonjs/core/Maths/math';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { Scene } from '@babylonjs/core/scene';\r\n\r\nexport function createJoinGeometryVisualization(targetMesh: Mesh, scene: Scene, color?: Color3) {\r\n const visualization = targetMesh;\r\n\r\n if (!color) {\r\n color = Color3.Yellow();\r\n }\r\n\r\n const coloredMaterial = new StandardMaterial('', scene);\r\n //coloredMaterial.alpha = 0.5;\r\n coloredMaterial.wireframe = true;\r\n coloredMaterial.emissiveColor = color;\r\n visualization.material = coloredMaterial;\r\n return visualization;\r\n}\r\n","import { Color3 } from '@babylonjs/core/Maths/math';\r\nimport { Quaternion } from '@babylonjs/core/Maths/math.vector';\r\nimport { CSG } from '@babylonjs/core/Meshes/csg';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { eJoinType } from '@models/classes/enums';\r\nimport { JoinGeometryFeature } from '@models/classes/features/join-geometry-feature';\r\nimport { CustomMeshNode } from '@models/classes/meshes/custom-mesh-node';\r\nimport { MeshNode } from '@models/classes/meshes/mesh-node';\r\nimport { SpaceNode } from '@models/classes/meshes/space-node';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass } from '@models/interfaces/tracked-class';\r\nimport { SubmeshVertexData } from '@view/helpers/geometry-helper';\r\nimport { createJoinGeometryVisualization } from '@view/helpers/join-geometry-helper';\r\nimport { Subscription } from 'rxjs';\r\nimport { FeatureRenderer } from './feature';\r\nimport { ensureNormalsExist } from './utils';\r\n\r\nconst joinedMeshes = new Map();\r\n\r\nexport class JoinGeometryRenderer extends FeatureRenderer {\r\n modelToken = Token.JoinGeometryFeature;\r\n bbSubscriptions = new Map();\r\n targetMeshVisualizationMap = new Map();\r\n parentMeshVisualizationMap = new Map();\r\n previewLayer: UtilityLayerRenderer = this.viewer.utilityLayer;\r\n\r\n updateMeshGeometry(\r\n feature: JoinGeometryFeature,\r\n vertexData: SubmeshVertexData,\r\n vertexDataChanges: Partial\r\n ) {\r\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\r\n\r\n const targetMesh = joinedMeshes.get(feature._dynamicId);\r\n const thisMesh = this.viewer.getRendererNode(feature._parent!);\r\n\r\n if (\r\n targetMesh instanceof Mesh &&\r\n thisMesh instanceof Mesh &&\r\n targetMesh.geometry &&\r\n targetMesh.geometry.getTotalIndices() > 0 &&\r\n targetMesh != thisMesh\r\n ) {\r\n const workingVertexData = vertexData.shallowClone();\r\n const tempMesh = new Mesh('csg-temp');\r\n tempMesh.doNotSerialize = true;\r\n SubmeshVertexData.CopyToRef(newVertexData, workingVertexData);\r\n if (!workingVertexData.positions || !workingVertexData.indices) {\r\n throw 'Invalid geometry';\r\n }\r\n const targetTransform = thisMesh.computeWorldMatrix(true);\r\n workingVertexData.applyToMesh(tempMesh);\r\n\r\n let previewMesh = this.parentMeshVisualizationMap.get(feature.id);\r\n\r\n if (!previewMesh) {\r\n const newMeshInstance = new Mesh('preview-mesh');\r\n newMeshInstance.doNotSerialize = true;\r\n workingVertexData.applyToMesh(newMeshInstance);\r\n this.parentMeshVisualizationMap.set(feature.id, newMeshInstance);\r\n if (newMeshInstance instanceof Mesh) {\r\n previewMesh = createJoinGeometryVisualization(\r\n newMeshInstance,\r\n this.previewLayer.utilityLayerScene,\r\n Color3.Yellow()\r\n );\r\n this.parentMeshVisualizationMap.set(feature.id, previewMesh);\r\n previewMesh.setEnabled(false);\r\n if (feature._parent) {\r\n const featureParentMesh = this.viewer.getRendererNode(feature._parent);\r\n newMeshInstance.position.addInPlace(featureParentMesh.absolutePosition);\r\n }\r\n }\r\n }\r\n\r\n ensureNormalsExist(tempMesh);\r\n\r\n const resultCsg = CSG.FromMesh(tempMesh);\r\n\r\n const targetMeshMatchTransform = targetMesh.clone();\r\n const targetMeshTransform = targetMeshMatchTransform.computeWorldMatrix(true);\r\n targetMeshMatchTransform.parent = null;\r\n targetTransform.invert();\r\n targetMeshTransform.multiplyToRef(targetTransform, targetMeshTransform);\r\n targetMeshMatchTransform.rotationQuaternion = new Quaternion();\r\n targetMeshTransform.decompose(\r\n targetMeshMatchTransform.scaling,\r\n targetMeshMatchTransform.rotationQuaternion,\r\n targetMeshMatchTransform.position\r\n );\r\n\r\n try {\r\n const opCsg = CSG.FromMesh(targetMeshMatchTransform);\r\n if (feature.joinType === eJoinType.cut) {\r\n resultCsg.subtractInPlace(opCsg);\r\n } else if (feature.joinType === eJoinType.intersect) {\r\n resultCsg.intersectInPlace(opCsg);\r\n } else if (feature.joinType === eJoinType.union) {\r\n resultCsg.unionInPlace(opCsg);\r\n }\r\n } catch (e) {\r\n console.warn('Could not merge geometry for join ' + feature.name, e);\r\n return newVertexData;\r\n }\r\n\r\n if (resultCsg) {\r\n const resultMeshCsg = resultCsg.toMesh('temp2', null, undefined, true);\r\n\r\n const resultVertexData = SubmeshVertexData.ExtractFromMesh(resultMeshCsg);\r\n newVertexData.positions = resultVertexData.positions;\r\n if (newVertexData.normals) {\r\n newVertexData.normals = resultVertexData.normals;\r\n }\r\n if (newVertexData.uvs) {\r\n newVertexData.uvs = resultVertexData.uvs;\r\n }\r\n newVertexData.indices = resultVertexData.indices;\r\n\r\n // Sometimes the CSG will add another material to the multiMaterial to represent the cut face. Change that material to be the base material, otherwise it will be black.\r\n const existingMaterials = new Set();\r\n // get a list of all the materials that are already in the multiMaterial\r\n if (newVertexData.submeshData) {\r\n newVertexData.submeshData.forEach(submeshData => {\r\n existingMaterials.add(submeshData.materialIndex);\r\n });\r\n }\r\n if (resultVertexData.submeshData) {\r\n resultVertexData.submeshData.forEach(submeshData => {\r\n if (!existingMaterials.has(submeshData.materialIndex)) {\r\n submeshData.materialIndex = 0;\r\n }\r\n });\r\n }\r\n\r\n newVertexData.submeshData = resultVertexData.submeshData;\r\n resultMeshCsg.dispose();\r\n }\r\n tempMesh.dispose();\r\n targetMeshMatchTransform.dispose();\r\n }\r\n return newVertexData;\r\n }\r\n\r\n updatePreviewMesh(node: JoinGeometryFeature) {\r\n let featureInstance = this.targetMeshVisualizationMap.get(node.id);\r\n const scene = this.previewLayer.utilityLayerScene;\r\n const targetMesh = joinedMeshes.get(node._dynamicId);\r\n if (targetMesh && !featureInstance) {\r\n featureInstance = createJoinGeometryVisualization(targetMesh.clone(), scene, Color3.Yellow());\r\n featureInstance.isPickable = false;\r\n featureInstance.setEnabled(false);\r\n this.targetMeshVisualizationMap.set(node.id, featureInstance);\r\n return featureInstance;\r\n } else if (targetMesh) {\r\n this.deleteTargetMeshVisualization(node as ITrackedClass);\r\n featureInstance = createJoinGeometryVisualization(targetMesh.clone(), scene, Color3.Yellow());\r\n featureInstance.isPickable = false;\r\n featureInstance.setEnabled(false);\r\n this.targetMeshVisualizationMap.set(node.id, featureInstance);\r\n return featureInstance;\r\n }\r\n }\r\n\r\n update3dInternal(feature: ITrackedClass) {\r\n if (feature.changes.hasAny('meshId', 'enabled')) {\r\n const existingSubscription = this.bbSubscriptions.get(feature._dynamicId);\r\n if (existingSubscription) {\r\n existingSubscription.unsubscribe();\r\n }\r\n\r\n if (feature.enabled) {\r\n const targetMeshNode = feature._manager?.getById(feature.meshId);\r\n if (targetMeshNode && targetMeshNode instanceof MeshNode) {\r\n // if this is a custom mesh node and the geometry isn't loaded, we have to load it\r\n if (\r\n targetMeshNode instanceof CustomMeshNode &&\r\n targetMeshNode.hasGeometry &&\r\n !targetMeshNode.geometryLoaded\r\n ) {\r\n this.viewer.rendererManager.mesh.loadNewGeometry(targetMeshNode);\r\n }\r\n const targetMesh = this.viewer.getRendererNode(targetMeshNode);\r\n\r\n if (targetMesh instanceof Mesh && feature._parent) {\r\n joinedMeshes.set(feature._dynamicId, targetMesh);\r\n\r\n this.bbSubscriptions.set(\r\n feature._dynamicId,\r\n // TODO This should also watch the parent group for positional changes, otherwise the join won't update when the feature's parent mesh moves.\r\n // But watch out for infinite loops\r\n this.viewer.bbCache.watchForChanges(targetMeshNode).subscribe(() => {\r\n feature.setPropertyIsModified('meshId', true);\r\n })\r\n );\r\n } else {\r\n console.warn('Could not create CSG for join geometry');\r\n }\r\n }\r\n }\r\n }\r\n\r\n let featureInstance = this.targetMeshVisualizationMap.get(feature.id);\r\n const originalMesh = this.parentMeshVisualizationMap.get(feature.id);\r\n\r\n if (feature._previewVisible) {\r\n featureInstance = this.updatePreviewMesh(feature);\r\n }\r\n\r\n if (feature._previewVisible && feature.enabled) {\r\n if (feature.joinType === eJoinType.cut) {\r\n originalMesh?.setEnabled(false);\r\n featureInstance?.setEnabled(true);\r\n } else if (feature.joinType === eJoinType.intersect) {\r\n originalMesh?.setEnabled(true);\r\n featureInstance?.setEnabled(true);\r\n } else {\r\n originalMesh?.setEnabled(false);\r\n featureInstance?.setEnabled(false);\r\n }\r\n } else {\r\n originalMesh?.setEnabled(false);\r\n featureInstance?.setEnabled(false);\r\n }\r\n }\r\n\r\n delete3d(feature: ITrackedClass, preserveCache: boolean) {\r\n if (!preserveCache) {\r\n joinedMeshes.delete(feature._dynamicId);\r\n }\r\n this.bbSubscriptions.get(feature._dynamicId)?.unsubscribe();\r\n this.bbSubscriptions.delete(feature._dynamicId);\r\n this.deleteTargetMeshVisualization(feature);\r\n\r\n const targetMeshCopy = this.parentMeshVisualizationMap.get(feature.id);\r\n targetMeshCopy?.dispose(false, true);\r\n this.parentMeshVisualizationMap.delete(feature.id);\r\n super.delete3d(feature);\r\n }\r\n\r\n deleteTargetMeshVisualization(feature: ITrackedClass) {\r\n const previewMesh = this.targetMeshVisualizationMap.get(feature.id);\r\n previewMesh?.dispose(false, true);\r\n this.targetMeshVisualizationMap.delete(feature.id);\r\n }\r\n\r\n dispose() {\r\n this.bbSubscriptions.forEach(s => {\r\n s.unsubscribe();\r\n });\r\n this.bbSubscriptions.clear();\r\n joinedMeshes.clear();\r\n super.dispose();\r\n }\r\n}\r\n","import { Vector3 } from '@babylonjs/core/Maths';\r\nimport { SolidParticle } from '@babylonjs/core/Particles/solidParticle';\r\nimport { LinearPatternFeature } from '@models/classes/features/linear-pattern-feature';\r\nimport { Token } from '@models/classes/token';\r\nimport { SubmeshVertexData } from '@view/helpers/geometry-helper';\r\nimport { PatternRenderer } from './pattern';\r\n\r\nexport class LinearPatternRenderer extends PatternRenderer {\r\n modelToken = Token.LinearPatternFeature;\r\n updateMeshGeometry(\r\n feature: LinearPatternFeature,\r\n vertexData: SubmeshVertexData,\r\n vertexDataChanges: Partial\r\n ) {\r\n const featureTranslation = feature.translation.toVec3();\r\n const workingVector = Vector3.Zero();\r\n\r\n return super.updateMeshGeometry(\r\n feature,\r\n vertexData,\r\n vertexDataChanges,\r\n (idx: number, particle: SolidParticle) => {\r\n if (!feature.meshId) {\r\n workingVector.copyFrom(featureTranslation).scaleInPlace(idx);\r\n }\r\n particle.position.addInPlace(workingVector);\r\n }\r\n );\r\n }\r\n}\r\n","import { Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { MirrorFeature } from '@models/classes/features/mirror-feature';\r\nimport { SliceFeature } from '@models/classes/features/slice-feature';\r\nimport { SubmeshData } from '@models/classes/features/submesh-feature';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass } from '@models/interfaces';\r\nimport { SubmeshVertexData } from '@view/helpers/geometry-helper';\r\nimport { rotationToVector } from '@view/helpers/math-helper';\r\nimport { createMirrorVisualization } from '@view/helpers/mirror-helper';\r\nimport { FeatureRenderer } from './feature';\r\nexport class SliceFeatureRenderer extends FeatureRenderer {\r\n modelToken = Token.SliceFeature;\r\n previewLayer: UtilityLayerRenderer = this.viewer.utilityLayer;\r\n babylonMirrorPreviewsMap = new Map();\r\n\r\n updateMeshGeometry(\r\n feature: SliceFeature,\r\n vertexData: SubmeshVertexData,\r\n vertexDataChanges: Partial\r\n ) {\r\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\r\n\r\n if (\r\n newVertexData.positions &&\r\n newVertexData.positions.length > 0 &&\r\n newVertexData.indices &&\r\n newVertexData.indices.length > 0\r\n ) {\r\n const positions = newVertexData.positions;\r\n const normals = newVertexData.normals;\r\n const uvs = newVertexData.uvs;\r\n const indices = newVertexData.indices;\r\n\r\n const triCount = indices.length / 3;\r\n const positionsCount = positions.length / 3;\r\n const planeNormal = feature.normal.toVec3();\r\n const planePoint = feature.intersect.toVec3();\r\n\r\n const newPositions: number[] = [];\r\n const newNormals: number[] = [];\r\n const newUvs: number[] = [];\r\n const newIndices: number[] = [];\r\n const preSliceSubmeshes = newVertexData.submeshData!;\r\n const newSubmeshes: SubmeshData[] = preSliceSubmeshes.map(submesh => ({ ...submesh }));\r\n const edgeSplits: number[] = []; // which vertices are part of how many edges being split\r\n const edgeStart = Vector3.Zero();\r\n const edgeDir = Vector3.Zero();\r\n const workingVector = Vector3.Zero();\r\n\r\n let j: number;\r\n for (let i = 0; i < triCount; i++) {\r\n let hit = 0;\r\n let d = 0;\r\n let vertexIndex = -1;\r\n for (j = 0; j < 3; j++) {\r\n const iA = j,\r\n iB = (j + 1) % 3;\r\n const vA = indices[i * 3 + iA],\r\n vB = indices[i * 3 + iB];\r\n if (vertexIndex === -1) {\r\n vertexIndex = vA;\r\n }\r\n edgeStart.set(positions[vA * 3], positions[vA * 3 + 1], positions[vA * 3 + 2]);\r\n edgeStart.subtractFromFloatsToRef(\r\n positions[vB * 3],\r\n positions[vB * 3 + 1],\r\n positions[vB * 3 + 2],\r\n edgeDir\r\n );\r\n edgeDir.scaleInPlace(-1);\r\n\r\n const lnDot = Vector3.Dot(planeNormal, edgeDir);\r\n if (lnDot === 0) {\r\n continue;\r\n }\r\n\r\n planePoint.subtractToRef(edgeStart, workingVector);\r\n d = Vector3.Dot(workingVector, planeNormal) / lnDot;\r\n\r\n // Expect floating point errors. don't want unnecessary tiny triangles\r\n // If d is between 0 and 1, then the plane hit the edge\r\n if (d > 1e-12 && d < 1 - 1e-12) {\r\n hit++;\r\n edgeSplits[iA] = (edgeSplits[iA] || 0) + 1;\r\n edgeSplits[iB] = (edgeSplits[iB] || 0) + 1;\r\n edgeDir.scaleInPlace(d);\r\n edgeStart.addToRef(edgeDir, workingVector);\r\n\r\n newPositions.push(workingVector.x, workingVector.y, workingVector.z);\r\n\r\n if (normals) {\r\n workingVector.set(normals[vA * 3], normals[vA * 3 + 1], normals[vA * 3 + 2]);\r\n workingVector.scaleInPlace(1 - d);\r\n workingVector.addInPlaceFromFloats(\r\n normals[vB * 3] * d,\r\n normals[vB * 3 + 1] * d,\r\n normals[vB * 3 + 2] * d\r\n );\r\n newNormals.push(workingVector.x, workingVector.y, workingVector.z);\r\n }\r\n\r\n if (uvs) {\r\n workingVector.set(uvs[vA * 2], uvs[vA * 2 + 1], 0);\r\n workingVector.scaleInPlace(1 - d);\r\n workingVector.addInPlaceFromFloats(uvs[vB * 2] * d, uvs[vB * 2 + 1] * d, 0);\r\n newUvs.push(workingVector.x, workingVector.y);\r\n }\r\n }\r\n }\r\n\r\n if (hit) {\r\n // divide the triangle here\r\n if (hit === 1) {\r\n const split = positionsCount - 1 + newPositions.length / 3;\r\n\r\n if (!edgeSplits[0]) {\r\n newIndices.push(\r\n indices[i * 3],\r\n indices[i * 3 + 1],\r\n split,\r\n indices[i * 3],\r\n split,\r\n indices[i * 3 + 2]\r\n );\r\n } else if (!edgeSplits[1]) {\r\n newIndices.push(\r\n indices[i * 3],\r\n indices[i * 3 + 1],\r\n split,\r\n indices[i * 3 + 1],\r\n indices[i * 3 + 2],\r\n split\r\n );\r\n } else {\r\n newIndices.push(\r\n indices[i * 3],\r\n split,\r\n indices[i * 3 + 2],\r\n indices[i * 3 + 1],\r\n indices[i * 3 + 2],\r\n split\r\n );\r\n }\r\n newSubmeshes.forEach((submeshData, sI) => {\r\n const preSliceSubmesh = preSliceSubmeshes[sI];\r\n if (\r\n preSliceSubmesh.indexStart <= i * 3 &&\r\n preSliceSubmesh.indexStart + preSliceSubmesh.indexCount > i * 3\r\n ) {\r\n submeshData.indexCount += 3;\r\n } else if (preSliceSubmesh.indexStart > i * 3) {\r\n submeshData.indexStart += 3;\r\n }\r\n });\r\n }\r\n if (hit === 2) {\r\n const aSplit = positionsCount - 1 + newPositions.length / 3 - 1,\r\n bSplit = positionsCount - 1 + newPositions.length / 3;\r\n newIndices.push(indices[i * 3]);\r\n newIndices.push(aSplit);\r\n newIndices.push(bSplit);\r\n\r\n if (edgeSplits[0] === 2) {\r\n newIndices.push(\r\n indices[i * 3 + 1],\r\n bSplit,\r\n aSplit,\r\n indices[i * 3 + 2],\r\n bSplit,\r\n indices[i * 3 + 1]\r\n );\r\n } else if (edgeSplits[1] === 2) {\r\n newIndices.push(\r\n indices[i * 3 + 2],\r\n indices[i * 3],\r\n bSplit,\r\n indices[i * 3 + 1],\r\n bSplit,\r\n aSplit\r\n );\r\n } else {\r\n newIndices.push(\r\n indices[i * 3 + 1],\r\n aSplit,\r\n indices[i * 3],\r\n indices[i * 3 + 2],\r\n bSplit,\r\n aSplit\r\n );\r\n }\r\n newSubmeshes.forEach((submeshData, sI) => {\r\n const preSliceSubmesh = preSliceSubmeshes[sI];\r\n if (\r\n preSliceSubmesh.indexStart <= i * 3 &&\r\n preSliceSubmesh.indexStart + preSliceSubmesh.indexCount > i * 3\r\n ) {\r\n submeshData.indexCount += 6;\r\n } else if (preSliceSubmesh.indexStart > i * 3) {\r\n submeshData.indexStart += 6;\r\n }\r\n });\r\n }\r\n edgeSplits.clear();\r\n hit = 0;\r\n } else {\r\n // if(feature.removeGeometry && d > 1 - 1e-12) {\r\n // // If this face is in front of the face relative to the face normal, cull it\r\n // newSubmeshes.forEach(submeshData => {\r\n // if(submeshData.indexStart <= i && submeshData.indexStart + submeshData.indexCount > i) {\r\n // submeshData.indexCount -= 3;\r\n // } else if(submeshData.indexStart > i) {\r\n // submeshData.indexStart -= 3;\r\n // }\r\n // });\r\n // continue;\r\n // } else {\r\n // console.log(d);\r\n // }\r\n newIndices.push(indices[i * 3], indices[i * 3 + 1], indices[i * 3 + 2]);\r\n }\r\n\r\n // Update the vertices the submeshes are targetting\r\n if (vertexIndex !== -1 && hit > 0) {\r\n newSubmeshes.forEach(submeshData => {\r\n if (\r\n submeshData.verticesStart <= vertexIndex &&\r\n submeshData.verticesStart + submeshData.verticesCount > vertexIndex\r\n ) {\r\n submeshData.verticesCount += hit;\r\n } else if (submeshData.verticesStart > vertexIndex) {\r\n submeshData.verticesStart += hit;\r\n }\r\n });\r\n }\r\n }\r\n\r\n newVertexData.indices = new Int32Array(newIndices);\r\n\r\n newVertexData.positions = new Float32Array(positions.length + newPositions.length);\r\n newVertexData.positions.set(positions);\r\n newVertexData.positions.set(newPositions, positions.length);\r\n\r\n if (normals) {\r\n newVertexData.normals = new Float32Array(normals.length + newNormals.length);\r\n newVertexData.normals.set(normals);\r\n newVertexData.normals.set(newNormals, normals.length);\r\n }\r\n\r\n if (uvs) {\r\n newVertexData.uvs = new Float32Array(uvs.length + newUvs.length);\r\n newVertexData.uvs.set(uvs);\r\n newVertexData.uvs.set(newUvs, uvs.length);\r\n }\r\n\r\n // Remove geometry on one side of the plane if the removeGeometry flag is true\r\n const baseGeometryVertexLength = positions.length / 3;\r\n if (feature.removeGeometry) {\r\n const indices = newVertexData.indices;\r\n const positions = newVertexData.positions;\r\n const normals = newVertexData.normals;\r\n const uvs = newVertexData.uvs;\r\n const newIndices = new Int32Array(indices.length);\r\n const newPositions = new Float32Array(positions.length);\r\n const newNormals = new Float32Array(normals?.length || 0);\r\n const newUvs = new Float32Array(uvs?.length || 0);\r\n\r\n const triCount = indices.length / 3;\r\n const vertexCount = positions.length / 3;\r\n const newPositionIndex = new Int32Array(vertexCount);\r\n\r\n // Keep track of how many positions we've used\r\n let position = 0;\r\n for (let i = 0; i < vertexCount; i++) {\r\n workingVector.set(positions[i * 3], positions[i * 3 + 1], positions[i * 3 + 2]);\r\n workingVector.subtractInPlace(planePoint);\r\n const d = Vector3.Dot(workingVector, planeNormal);\r\n\r\n // Don't remove new vertices that were created from the slice, those are guaranteed to be good\r\n if (i < baseGeometryVertexLength && d > 1e-12) {\r\n // Mark the original indices that have been removed so we can update the index array\r\n newPositionIndex[i] = -1;\r\n } else {\r\n // Copy over the vertex data we need and mark what new index they will have\r\n newPositionIndex[i] = position;\r\n newPositions[position * 3] = positions[i * 3];\r\n newPositions[position * 3 + 1] = positions[i * 3 + 1];\r\n newPositions[position * 3 + 2] = positions[i * 3 + 2];\r\n if (normals) {\r\n newNormals[position * 3] = normals[i * 3];\r\n newNormals[position * 3 + 1] = normals[i * 3 + 1];\r\n newNormals[position * 3 + 2] = normals[i * 3 + 2];\r\n }\r\n if (uvs) {\r\n newUvs[position * 2] = uvs[i * 2];\r\n newUvs[position * 2 + 1] = uvs[i * 2 + 1];\r\n }\r\n position++;\r\n }\r\n }\r\n // Trim off unused parts of the vertex arrays\r\n newVertexData.positions = newPositions.subarray(0, position * 3);\r\n if (normals) {\r\n newVertexData.normals = newNormals.subarray(0, position * 3);\r\n }\r\n if (uvs) {\r\n newVertexData.uvs = newUvs.subarray(0, position * 2);\r\n }\r\n\r\n position = 0;\r\n const originalSubmeshes = newSubmeshes.map(submesh => ({ ...submesh }));\r\n for (let i = 0; i < triCount; i++) {\r\n // If all of the positions in this triangle are being kept, add the triangle\r\n // The indices of the vertexes are changing so we need to use the value that we calculated prior\r\n if (\r\n newPositionIndex[indices[i * 3]] >= 0 &&\r\n newPositionIndex[indices[i * 3 + 1]] >= 0 &&\r\n newPositionIndex[indices[i * 3 + 2]] >= 0\r\n ) {\r\n newIndices[position * 3] = newPositionIndex[indices[i * 3]];\r\n newIndices[position * 3 + 1] = newPositionIndex[indices[i * 3 + 1]];\r\n newIndices[position * 3 + 2] = newPositionIndex[indices[i * 3 + 2]];\r\n position++;\r\n } else {\r\n newSubmeshes.forEach((submeshData, sI) => {\r\n const originalSubmesh = originalSubmeshes[sI];\r\n if (\r\n originalSubmesh.indexStart <= i * 3 &&\r\n originalSubmesh.indexStart + originalSubmesh.indexCount > i * 3\r\n ) {\r\n submeshData.indexCount -= 3;\r\n } else if (originalSubmesh.indexStart > i * 3) {\r\n submeshData.indexStart -= 3;\r\n }\r\n });\r\n }\r\n }\r\n // Update the vertex values on the submeshes as well\r\n for (let i = 0; i < newPositionIndex.length; i++) {\r\n if (newPositionIndex[i] === -1) {\r\n newSubmeshes.forEach(submeshData => {\r\n if (\r\n submeshData.verticesStart <= i &&\r\n submeshData.verticesStart + submeshData.verticesCount > i\r\n ) {\r\n submeshData.verticesCount--;\r\n } else if (submeshData.verticesStart > i) {\r\n submeshData.verticesStart--;\r\n }\r\n });\r\n }\r\n }\r\n newVertexData.indices = newIndices.subarray(0, position * 3);\r\n }\r\n\r\n newVertexData.submeshData = newSubmeshes;\r\n }\r\n return newVertexData;\r\n }\r\n\r\n protected updatePreviewMesh(feature: SliceFeature) {\r\n const scene = this.previewLayer.utilityLayerScene;\r\n let visualizationPlane = this.babylonMirrorPreviewsMap.get(feature.id);\r\n\r\n if (!visualizationPlane) {\r\n visualizationPlane = createMirrorVisualization(feature, scene);\r\n visualizationPlane.isPickable = false;\r\n this.babylonMirrorPreviewsMap.set(feature.id, visualizationPlane);\r\n }\r\n\r\n const currentNormal = new Vector3();\r\n currentNormal.set(0, 0, 1);\r\n const newNormal = new Vector3();\r\n newNormal.set(feature.normal.x, feature.normal.y, feature.normal.z);\r\n visualizationPlane.rotationQuaternion = rotationToVector(newNormal, currentNormal);\r\n let parentBoundingBox = undefined;\r\n if (feature._parent) {\r\n parentBoundingBox = this.viewer.getMeshBoundingBox(feature._parent);\r\n }\r\n if (parentBoundingBox) {\r\n const scale = parentBoundingBox.minWorld.subtract(parentBoundingBox.maxWorld).length();\r\n visualizationPlane.scaling.set(scale, scale, scale);\r\n }\r\n\r\n if (feature._parent) {\r\n const featureParent = this.viewer.getRendererNode(feature._parent);\r\n const meshPosition = featureParent.getAbsolutePosition();\r\n meshPosition.addInPlace(feature.intersect.toVec3());\r\n visualizationPlane.position.set(meshPosition.x, meshPosition.y, meshPosition.z);\r\n }\r\n }\r\n\r\n update3dInternal(node: ITrackedClass) {\r\n const visualizationPlane = this.babylonMirrorPreviewsMap.get(node.id);\r\n\r\n if (node._previewVisible) {\r\n this.updatePreviewMesh(node);\r\n if (node.enabled) {\r\n visualizationPlane?.setEnabled(true);\r\n }\r\n } else {\r\n visualizationPlane?.setEnabled(false);\r\n }\r\n }\r\n\r\n delete3d(node: ITrackedClass | ITrackedClass) {\r\n if (this.babylonMirrorPreviewsMap) {\r\n const featureInstance = this.babylonMirrorPreviewsMap.get(node.id);\r\n\r\n if (featureInstance) {\r\n featureInstance.dispose(false, true);\r\n }\r\n this.babylonMirrorPreviewsMap.delete(node.id);\r\n }\r\n super.delete3d(node as ITrackedClass);\r\n }\r\n}\r\n","import { StandardMaterial } from '@babylonjs/core/Materials/standardMaterial';\r\nimport { Color3 } from '@babylonjs/core/Maths/math';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { MeshBuilder } from '@babylonjs/core/Meshes/meshBuilder';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { Feature } from '@models/classes';\r\n\r\nexport function createMirrorVisualization(mirror: Feature, scene: Scene, color?: Color3) {\r\n if (!color) {\r\n color = Color3.Yellow();\r\n }\r\n\r\n const mirrorVisualization: Mesh = MeshBuilder.CreatePlane('mirrorPlane', {}, scene);\r\n const coloredMaterial = new StandardMaterial('', scene);\r\n coloredMaterial.wireframe = true;\r\n coloredMaterial.emissiveColor = color;\r\n mirrorVisualization.material = coloredMaterial;\r\n\r\n mirrorVisualization.isPickable = false;\r\n mirrorVisualization.setEnabled(true);\r\n return mirrorVisualization;\r\n}\r\n","import { Vector3 } from '@babylonjs/core/Maths';\r\nimport { MirrorFeature } from '@models/classes/features/mirror-feature';\r\nimport { SubmeshData } from '@models/classes/features/submesh-feature';\r\nimport { Token } from '@models/classes/token';\r\nimport { SubmeshVertexData } from '@view/helpers/geometry-helper';\r\nimport { SliceFeatureRenderer } from './slice';\r\n\r\nexport class MirrorFeatureRenderer extends SliceFeatureRenderer {\r\n modelToken = Token.MirrorFeature;\r\n updateMeshGeometry(\r\n feature: MirrorFeature,\r\n vertexData: SubmeshVertexData,\r\n vertexDataChanges: Partial\r\n ) {\r\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\r\n\r\n if (\r\n newVertexData.positions &&\r\n newVertexData.positions.length > 0 &&\r\n newVertexData.indices &&\r\n newVertexData.indices.length > 0\r\n ) {\r\n const positions = newVertexData.positions;\r\n const normals = newVertexData.normals;\r\n const uvs = newVertexData.uvs;\r\n const indices = newVertexData.indices;\r\n\r\n const newIndices = new Int32Array(indices.length);\r\n const newPositions = new Float32Array(positions.length);\r\n const newNormals = new Float32Array(normals?.length || 0);\r\n\r\n const workingVector = Vector3.Zero();\r\n const planeNormal = feature.normal.toVec3().normalize();\r\n const planeIntersect = feature.intersect.toVec3();\r\n const vertexCount = positions.length / 3;\r\n for (let i = 0; i < vertexCount; i++) {\r\n const idx = i * 3;\r\n workingVector.set(positions[idx], positions[idx + 1], positions[idx + 2]);\r\n workingVector.subtractInPlace(planeIntersect);\r\n let d = Vector3.Dot(workingVector, planeNormal);\r\n planeNormal.scaleAndAddToRef(-2 * d, workingVector);\r\n newPositions[idx] = workingVector.x + planeIntersect.x;\r\n newPositions[idx + 1] = workingVector.y + planeIntersect.y;\r\n newPositions[idx + 2] = workingVector.z + planeIntersect.z;\r\n\r\n if (normals) {\r\n workingVector.set(normals[idx], normals[idx + 1], normals[idx + 2]);\r\n d = Vector3.Dot(workingVector, planeNormal);\r\n planeNormal.scaleAndAddToRef(-2 * d, workingVector);\r\n newNormals[idx] = workingVector.x;\r\n newNormals[idx + 1] = workingVector.y;\r\n newNormals[idx + 2] = workingVector.z;\r\n }\r\n }\r\n\r\n // Flip the direction of the triangles so that the normals face the right way\r\n const triCount = indices.length / 3;\r\n for (let i = 0; i < triCount; i++) {\r\n const idx = i * 3;\r\n if (feature.cloneGeometry) {\r\n newIndices[idx] = vertexCount + indices[idx + 2];\r\n newIndices[idx + 1] = vertexCount + indices[idx + 1];\r\n newIndices[idx + 2] = vertexCount + indices[idx];\r\n } else {\r\n newIndices[idx] = indices[idx + 2];\r\n newIndices[idx + 1] = indices[idx + 1];\r\n newIndices[idx + 2] = indices[idx];\r\n }\r\n }\r\n\r\n if (feature.cloneGeometry) {\r\n const combinedIndices = new Int32Array(indices.length * 2);\r\n const combinedPositions = new Float32Array(positions.length * 2);\r\n const combinedNormals = new Float32Array((normals?.length || 0) * 2);\r\n const combinedUvs = new Float32Array((uvs?.length || 0) * 2);\r\n const combinedSubmeshes: SubmeshData[] = [];\r\n\r\n combinedIndices.set(indices);\r\n combinedIndices.set(newIndices, indices.length);\r\n combinedPositions.set(positions);\r\n combinedPositions.set(newPositions, positions.length);\r\n if (normals) {\r\n combinedNormals.set(normals);\r\n combinedNormals.set(newNormals, normals.length);\r\n }\r\n if (uvs) {\r\n combinedUvs.set(uvs);\r\n combinedUvs.set(uvs, uvs.length);\r\n }\r\n\r\n if (newVertexData.submeshData) {\r\n newVertexData.submeshData.forEach(submesh => {\r\n const newSubmesh = {\r\n ...submesh,\r\n verticesStart: submesh.verticesStart + vertexCount,\r\n indexStart: submesh.indexStart + indices.length,\r\n };\r\n combinedSubmeshes.push(submesh, newSubmesh);\r\n });\r\n }\r\n\r\n newVertexData.indices = combinedIndices;\r\n newVertexData.positions = combinedPositions;\r\n if (normals) {\r\n newVertexData.normals = combinedNormals;\r\n }\r\n if (uvs) {\r\n newVertexData.uvs = combinedUvs;\r\n }\r\n newVertexData.submeshData = combinedSubmeshes;\r\n } else {\r\n newVertexData.indices = newIndices;\r\n newVertexData.positions = newPositions;\r\n if (normals) {\r\n newVertexData.normals = newNormals;\r\n }\r\n }\r\n }\r\n\r\n return newVertexData;\r\n }\r\n}\r\n","/*! *****************************************************************************\r\nCopyright (c) Microsoft Corporation.\r\n\r\nPermission to use, copy, modify, and/or distribute this software for any\r\npurpose with or without fee is hereby granted.\r\n\r\nTHE SOFTWARE IS PROVIDED \"AS IS\" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH\r\nREGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY\r\nAND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,\r\nINDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM\r\nLOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR\r\nOTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR\r\nPERFORMANCE OF THIS SOFTWARE.\r\n***************************************************************************** */\r\n/* global Reflect, Promise */\r\n\r\nvar extendStatics = function(d, b) {\r\n extendStatics = Object.setPrototypeOf ||\r\n ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||\r\n function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; };\r\n return extendStatics(d, b);\r\n};\r\n\r\nexport function __extends(d, b) {\r\n if (typeof b !== \"function\" && b !== null)\r\n throw new TypeError(\"Class extends value \" + String(b) + \" is not a constructor or null\");\r\n extendStatics(d, b);\r\n function __() { this.constructor = d; }\r\n d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());\r\n}\r\n\r\nexport var __assign = function() {\r\n __assign = Object.assign || function __assign(t) {\r\n for (var s, i = 1, n = arguments.length; i < n; i++) {\r\n s = arguments[i];\r\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p];\r\n }\r\n return t;\r\n }\r\n return __assign.apply(this, arguments);\r\n}\r\n\r\nexport function __rest(s, e) {\r\n var t = {};\r\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)\r\n t[p] = s[p];\r\n if (s != null && typeof Object.getOwnPropertySymbols === \"function\")\r\n for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {\r\n if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))\r\n t[p[i]] = s[p[i]];\r\n }\r\n return t;\r\n}\r\n\r\nexport function __decorate(decorators, target, key, desc) {\r\n var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;\r\n if (typeof Reflect === \"object\" && typeof Reflect.decorate === \"function\") r = Reflect.decorate(decorators, target, key, desc);\r\n else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;\r\n return c > 3 && r && Object.defineProperty(target, key, r), r;\r\n}\r\n\r\nexport function __param(paramIndex, decorator) {\r\n return function (target, key) { decorator(target, key, paramIndex); }\r\n}\r\n\r\nexport function __metadata(metadataKey, metadataValue) {\r\n if (typeof Reflect === \"object\" && typeof Reflect.metadata === \"function\") return Reflect.metadata(metadataKey, metadataValue);\r\n}\r\n\r\nexport function __awaiter(thisArg, _arguments, P, generator) {\r\n function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\r\n return new (P || (P = Promise))(function (resolve, reject) {\r\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\r\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\r\n function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\r\n step((generator = generator.apply(thisArg, _arguments || [])).next());\r\n });\r\n}\r\n\r\nexport function __generator(thisArg, body) {\r\n var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;\r\n return g = { next: verb(0), \"throw\": verb(1), \"return\": verb(2) }, typeof Symbol === \"function\" && (g[Symbol.iterator] = function() { return this; }), g;\r\n function verb(n) { return function (v) { return step([n, v]); }; }\r\n function step(op) {\r\n if (f) throw new TypeError(\"Generator is already executing.\");\r\n while (_) try {\r\n if (f = 1, y && (t = op[0] & 2 ? y[\"return\"] : op[0] ? y[\"throw\"] || ((t = y[\"return\"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;\r\n if (y = 0, t) op = [op[0] & 2, t.value];\r\n switch (op[0]) {\r\n case 0: case 1: t = op; break;\r\n case 4: _.label++; return { value: op[1], done: false };\r\n case 5: _.label++; y = op[1]; op = [0]; continue;\r\n case 7: op = _.ops.pop(); _.trys.pop(); continue;\r\n default:\r\n if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }\r\n if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }\r\n if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }\r\n if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }\r\n if (t[2]) _.ops.pop();\r\n _.trys.pop(); continue;\r\n }\r\n op = body.call(thisArg, _);\r\n } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }\r\n if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };\r\n }\r\n}\r\n\r\nexport var __createBinding = Object.create ? (function(o, m, k, k2) {\r\n if (k2 === undefined) k2 = k;\r\n Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } });\r\n}) : (function(o, m, k, k2) {\r\n if (k2 === undefined) k2 = k;\r\n o[k2] = m[k];\r\n});\r\n\r\nexport function __exportStar(m, o) {\r\n for (var p in m) if (p !== \"default\" && !Object.prototype.hasOwnProperty.call(o, p)) __createBinding(o, m, p);\r\n}\r\n\r\nexport function __values(o) {\r\n var s = typeof Symbol === \"function\" && Symbol.iterator, m = s && o[s], i = 0;\r\n if (m) return m.call(o);\r\n if (o && typeof o.length === \"number\") return {\r\n next: function () {\r\n if (o && i >= o.length) o = void 0;\r\n return { value: o && o[i++], done: !o };\r\n }\r\n };\r\n throw new TypeError(s ? \"Object is not iterable.\" : \"Symbol.iterator is not defined.\");\r\n}\r\n\r\nexport function __read(o, n) {\r\n var m = typeof Symbol === \"function\" && o[Symbol.iterator];\r\n if (!m) return o;\r\n var i = m.call(o), r, ar = [], e;\r\n try {\r\n while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value);\r\n }\r\n catch (error) { e = { error: error }; }\r\n finally {\r\n try {\r\n if (r && !r.done && (m = i[\"return\"])) m.call(i);\r\n }\r\n finally { if (e) throw e.error; }\r\n }\r\n return ar;\r\n}\r\n\r\n/** @deprecated */\r\nexport function __spread() {\r\n for (var ar = [], i = 0; i < arguments.length; i++)\r\n ar = ar.concat(__read(arguments[i]));\r\n return ar;\r\n}\r\n\r\n/** @deprecated */\r\nexport function __spreadArrays() {\r\n for (var s = 0, i = 0, il = arguments.length; i < il; i++) s += arguments[i].length;\r\n for (var r = Array(s), k = 0, i = 0; i < il; i++)\r\n for (var a = arguments[i], j = 0, jl = a.length; j < jl; j++, k++)\r\n r[k] = a[j];\r\n return r;\r\n}\r\n\r\nexport function __spreadArray(to, from) {\r\n for (var i = 0, il = from.length, j = to.length; i < il; i++, j++)\r\n to[j] = from[i];\r\n return to;\r\n}\r\n\r\nexport function __await(v) {\r\n return this instanceof __await ? (this.v = v, this) : new __await(v);\r\n}\r\n\r\nexport function __asyncGenerator(thisArg, _arguments, generator) {\r\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\r\n var g = generator.apply(thisArg, _arguments || []), i, q = [];\r\n return i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i;\r\n function verb(n) { if (g[n]) i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; }\r\n function resume(n, v) { try { step(g[n](v)); } catch (e) { settle(q[0][3], e); } }\r\n function step(r) { r.value instanceof __await ? Promise.resolve(r.value.v).then(fulfill, reject) : settle(q[0][2], r); }\r\n function fulfill(value) { resume(\"next\", value); }\r\n function reject(value) { resume(\"throw\", value); }\r\n function settle(f, v) { if (f(v), q.shift(), q.length) resume(q[0][0], q[0][1]); }\r\n}\r\n\r\nexport function __asyncDelegator(o) {\r\n var i, p;\r\n return i = {}, verb(\"next\"), verb(\"throw\", function (e) { throw e; }), verb(\"return\"), i[Symbol.iterator] = function () { return this; }, i;\r\n function verb(n, f) { i[n] = o[n] ? function (v) { return (p = !p) ? { value: __await(o[n](v)), done: n === \"return\" } : f ? f(v) : v; } : f; }\r\n}\r\n\r\nexport function __asyncValues(o) {\r\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\r\n var m = o[Symbol.asyncIterator], i;\r\n return m ? m.call(o) : (o = typeof __values === \"function\" ? __values(o) : o[Symbol.iterator](), i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i);\r\n function verb(n) { i[n] = o[n] && function (v) { return new Promise(function (resolve, reject) { v = o[n](v), settle(resolve, reject, v.done, v.value); }); }; }\r\n function settle(resolve, reject, d, v) { Promise.resolve(v).then(function(v) { resolve({ value: v, done: d }); }, reject); }\r\n}\r\n\r\nexport function __makeTemplateObject(cooked, raw) {\r\n if (Object.defineProperty) { Object.defineProperty(cooked, \"raw\", { value: raw }); } else { cooked.raw = raw; }\r\n return cooked;\r\n};\r\n\r\nvar __setModuleDefault = Object.create ? (function(o, v) {\r\n Object.defineProperty(o, \"default\", { enumerable: true, value: v });\r\n}) : function(o, v) {\r\n o[\"default\"] = v;\r\n};\r\n\r\nexport function __importStar(mod) {\r\n if (mod && mod.__esModule) return mod;\r\n var result = {};\r\n if (mod != null) for (var k in mod) if (k !== \"default\" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);\r\n __setModuleDefault(result, mod);\r\n return result;\r\n}\r\n\r\nexport function __importDefault(mod) {\r\n return (mod && mod.__esModule) ? mod : { default: mod };\r\n}\r\n\r\nexport function __classPrivateFieldGet(receiver, privateMap) {\r\n if (!privateMap.has(receiver)) {\r\n throw new TypeError(\"attempted to get private field on non-instance\");\r\n }\r\n return privateMap.get(receiver);\r\n}\r\n\r\nexport function __classPrivateFieldSet(receiver, privateMap, value) {\r\n if (!privateMap.has(receiver)) {\r\n throw new TypeError(\"attempted to set private field on non-instance\");\r\n }\r\n privateMap.set(receiver, value);\r\n return value;\r\n}\r\n","import { SVGPathData } from \"./SVGPathData\";\nimport { SVGCommand } from \"./types\";\n\n// Encode SVG PathData\n// http://www.w3.org/TR/SVG/paths.html#PathDataBNF\n\n// Private consts : Char groups\nconst WSP = \" \";\n\nexport function encodeSVGPath(commands: SVGCommand | SVGCommand[]) {\n let str = \"\";\n\n if (!Array.isArray(commands)) {\n commands = [commands];\n }\n for (let i = 0; i < commands.length; i++) {\n const command = commands[i];\n if (command.type === SVGPathData.CLOSE_PATH) {\n str += \"z\";\n } else if (command.type === SVGPathData.HORIZ_LINE_TO) {\n str += (command.relative ? \"h\" : \"H\") +\n command.x;\n } else if (command.type === SVGPathData.VERT_LINE_TO) {\n str += (command.relative ? \"v\" : \"V\") +\n command.y;\n } else if (command.type === SVGPathData.MOVE_TO) {\n str += (command.relative ? \"m\" : \"M\") +\n command.x + WSP + command.y;\n } else if (command.type === SVGPathData.LINE_TO) {\n str += (command.relative ? \"l\" : \"L\") +\n command.x + WSP + command.y;\n } else if (command.type === SVGPathData.CURVE_TO) {\n str += (command.relative ? \"c\" : \"C\") +\n command.x1 + WSP + command.y1 +\n WSP + command.x2 + WSP + command.y2 +\n WSP + command.x + WSP + command.y;\n } else if (command.type === SVGPathData.SMOOTH_CURVE_TO) {\n str += (command.relative ? \"s\" : \"S\") +\n command.x2 + WSP + command.y2 +\n WSP + command.x + WSP + command.y;\n } else if (command.type === SVGPathData.QUAD_TO) {\n str += (command.relative ? \"q\" : \"Q\") +\n command.x1 + WSP + command.y1 +\n WSP + command.x + WSP + command.y;\n } else if (command.type === SVGPathData.SMOOTH_QUAD_TO) {\n str += (command.relative ? \"t\" : \"T\") +\n command.x + WSP + command.y;\n } else if (command.type === SVGPathData.ARC) {\n str += (command.relative ? \"a\" : \"A\") +\n command.rX + WSP + command.rY +\n WSP + command.xRot +\n WSP + (+command.lArcFlag) + WSP + (+command.sweepFlag) +\n WSP + command.x + WSP + command.y;\n } else {\n // Unknown command\n throw new Error(\n `Unexpected command type \"${ (command as any).type}\" at index ${i}.`);\n }\n }\n\n return str;\n}\n","import { SVGPathData } from \"./SVGPathData\";\nimport { CommandA, CommandC } from \"./types\";\n\nexport function rotate([x, y]: [number, number], rad: number) {\n return [\n x * Math.cos(rad) - y * Math.sin(rad),\n x * Math.sin(rad) + y * Math.cos(rad),\n ];\n}\n\nconst DEBUG_CHECK_NUMBERS = true;\nexport function assertNumbers(...numbers: number[]) {\n if (DEBUG_CHECK_NUMBERS) {\n for (let i = 0; i < numbers.length; i++) {\n if (\"number\" !== typeof numbers[i]) {\n throw new Error(\n `assertNumbers arguments[${i}] is not a number. ${typeof numbers[i]} == typeof ${numbers[i]}`);\n }\n }\n }\n return true;\n}\n\nconst PI = Math.PI;\n\n/**\n * https://www.w3.org/TR/SVG/implnote.html#ArcImplementationNotes\n * Fixes rX and rY.\n * Ensures lArcFlag and sweepFlag are 0 or 1\n * Adds center coordinates: command.cX, command.cY (relative or absolute, depending on command.relative)\n * Adds start and end arc parameters (in degrees): command.phi1, command.phi2; phi1 < phi2 iff. c.sweepFlag == true\n */\nexport function annotateArcCommand(c: CommandA, x1: number, y1: number) {\n c.lArcFlag = (0 === c.lArcFlag) ? 0 : 1;\n c.sweepFlag = (0 === c.sweepFlag) ? 0 : 1;\n // tslint:disable-next-line\n let {rX, rY, x, y} = c;\n\n rX = Math.abs(c.rX);\n rY = Math.abs(c.rY);\n const [x1_, y1_] = rotate([(x1 - x) / 2, (y1 - y) / 2], -c.xRot / 180 * PI);\n const testValue = Math.pow(x1_, 2) / Math.pow(rX, 2) + Math.pow(y1_, 2) / Math.pow(rY, 2);\n\n if (1 < testValue) {\n rX *= Math.sqrt(testValue);\n rY *= Math.sqrt(testValue);\n }\n c.rX = rX;\n c.rY = rY;\n const c_ScaleTemp = (Math.pow(rX, 2) * Math.pow(y1_, 2) + Math.pow(rY, 2) * Math.pow(x1_, 2));\n const c_Scale = (c.lArcFlag !== c.sweepFlag ? 1 : -1) *\n Math.sqrt(Math.max(0, (Math.pow(rX, 2) * Math.pow(rY, 2) - c_ScaleTemp) / c_ScaleTemp));\n const cx_ = rX * y1_ / rY * c_Scale;\n const cy_ = -rY * x1_ / rX * c_Scale;\n const cRot = rotate([cx_, cy_], c.xRot / 180 * PI);\n\n c.cX = cRot[0] + (x1 + x) / 2;\n c.cY = cRot[1] + (y1 + y) / 2;\n c.phi1 = Math.atan2((y1_ - cy_) / rY, (x1_ - cx_) / rX);\n c.phi2 = Math.atan2((-y1_ - cy_) / rY, (-x1_ - cx_) / rX);\n if (0 === c.sweepFlag && c.phi2 > c.phi1) {\n c.phi2 -= 2 * PI;\n }\n if (1 === c.sweepFlag && c.phi2 < c.phi1) {\n c.phi2 += 2 * PI;\n }\n c.phi1 *= 180 / PI;\n c.phi2 *= 180 / PI;\n}\n\n/**\n * Solves a quadratic system of equations of the form\n * a * x + b * y = c\n * x² + y² = 1\n * This can be understood as the intersection of the unit circle with a line.\n * => y = (c - a x) / b\n * => x² + (c - a x)² / b² = 1\n * => x² b² + c² - 2 c a x + a² x² = b²\n * => (a² + b²) x² - 2 a c x + (c² - b²) = 0\n */\nexport function intersectionUnitCircleLine(a: number, b: number, c: number): [number, number][] {\n assertNumbers(a, b, c);\n // cf. pqFormula\n const termSqr = a * a + b * b - c * c;\n\n if (0 > termSqr) {\n return [];\n } else if (0 === termSqr) {\n return [\n [\n (a * c) / (a * a + b * b),\n (b * c) / (a * a + b * b)]];\n }\n const term = Math.sqrt(termSqr);\n\n return [\n [\n (a * c + b * term) / (a * a + b * b),\n (b * c - a * term) / (a * a + b * b)],\n [\n (a * c - b * term) / (a * a + b * b),\n (b * c + a * term) / (a * a + b * b)]];\n\n}\n\nexport const DEG = Math.PI / 180;\n\nexport function lerp(a: number, b: number, t: number) {\n return (1 - t) * a + t * b;\n}\n\nexport function arcAt(c: number, x1: number, x2: number, phiDeg: number) {\n return c + Math.cos(phiDeg / 180 * PI) * x1 + Math.sin(phiDeg / 180 * PI) * x2;\n}\n\nexport function bezierRoot(x0: number, x1: number, x2: number, x3: number) {\n const EPS = 1e-6;\n const x01 = x1 - x0;\n const x12 = x2 - x1;\n const x23 = x3 - x2;\n const a = 3 * x01 + 3 * x23 - 6 * x12;\n const b = (x12 - x01) * 6;\n const c = 3 * x01;\n // solve a * t² + b * t + c = 0\n\n if (Math.abs(a) < EPS) {\n // equivalent to b * t + c =>\n return [-c / b];\n }\n return pqFormula(b / a, c / a, EPS);\n\n}\n\nexport function bezierAt(x0: number, x1: number, x2: number, x3: number, t: number) {\n // console.log(x0, y0, x1, y1, x2, y2, x3, y3, t)\n const s = 1 - t;\n const c0 = s * s * s;\n const c1 = 3 * s * s * t;\n const c2 = 3 * s * t * t;\n const c3 = t * t * t;\n\n return x0 * c0 + x1 * c1 + x2 * c2 + x3 * c3;\n}\n\nfunction pqFormula(p: number, q: number, PRECISION = 1e-6) {\n // 4 times the discriminant:in\n const discriminantX4 = p * p / 4 - q;\n\n if (discriminantX4 < -PRECISION) {\n return [];\n } else if (discriminantX4 <= PRECISION) {\n return [-p / 2];\n }\n const root = Math.sqrt(discriminantX4);\n\n return [-(p / 2) - root, -(p / 2) + root];\n\n}\n\nexport function a2c(arc: CommandA, x0: number, y0: number): CommandC[] {\n if (!arc.cX) {\n annotateArcCommand(arc, x0, y0);\n }\n\n const phiMin = Math.min(arc.phi1!, arc.phi2!), phiMax = Math.max(arc.phi1!, arc.phi2!), deltaPhi = phiMax - phiMin;\n const partCount = Math.ceil(deltaPhi / 90 );\n\n const result: CommandC[] = new Array(partCount);\n let prevX = x0, prevY = y0;\n for (let i = 0; i < partCount; i++) {\n const phiStart = lerp(arc.phi1!, arc.phi2!, i / partCount);\n const phiEnd = lerp(arc.phi1!, arc.phi2!, (i + 1) / partCount);\n const deltaPhi = phiEnd - phiStart;\n const f = 4 / 3 * Math.tan(deltaPhi * DEG / 4);\n // x1/y1, x2/y2 and x/y coordinates on the unit circle for phiStart/phiEnd\n const [x1, y1] = [\n Math.cos(phiStart * DEG) - f * Math.sin(phiStart * DEG),\n Math.sin(phiStart * DEG) + f * Math.cos(phiStart * DEG)];\n const [x, y] = [Math.cos(phiEnd * DEG), Math.sin(phiEnd * DEG)];\n const [x2, y2] = [x + f * Math.sin(phiEnd * DEG), y - f * Math.cos(phiEnd * DEG)];\n result[i] = {relative: arc.relative, type: SVGPathData.CURVE_TO } as any;\n const transform = (x: number, y: number) => {\n const [xTemp, yTemp] = rotate([x * arc.rX, y * arc.rY], arc.xRot);\n return [arc.cX! + xTemp, arc.cY! + yTemp];\n };\n [result[i].x1, result[i].y1] = transform(x1, y1);\n [result[i].x2, result[i].y2] = transform(x2, y2);\n [result[i].x, result[i].y] = transform(x, y);\n if (arc.relative) {\n result[i].x1 -= prevX;\n result[i].y1 -= prevY;\n result[i].x2 -= prevX;\n result[i].y2 -= prevY;\n result[i].x -= prevX;\n result[i].y -= prevY;\n }\n [prevX, prevY] = [result[i].x, result[i].y];\n }\n return result;\n}\n","// Transform SVG PathData\n// http://www.w3.org/TR/SVG/paths.html#PathDataBNF\n\nimport { a2c, annotateArcCommand, arcAt, assertNumbers, bezierAt, bezierRoot,\n intersectionUnitCircleLine } from \"./mathUtils\";\nimport { SVGPathData } from \"./SVGPathData\";\nimport { SVGCommand, TransformFunction } from \"./types\";\n\nexport namespace SVGPathDataTransformer {\n // Predefined transforming functions\n // Rounds commands values\n export function ROUND(roundVal = 1e13) {\n assertNumbers(roundVal);\n function rf(val: number) { return Math.round(val * roundVal) / roundVal; }\n return function round(command: any) {\n if (\"undefined\" !== typeof command.x1) {\n command.x1 = rf(command.x1);\n }\n if (\"undefined\" !== typeof command.y1) {\n command.y1 = rf(command.y1);\n }\n\n if (\"undefined\" !== typeof command.x2) {\n command.x2 = rf(command.x2);\n }\n if (\"undefined\" !== typeof command.y2) {\n command.y2 = rf(command.y2);\n }\n\n if (\"undefined\" !== typeof command.x) {\n command.x = rf(command.x);\n }\n if (\"undefined\" !== typeof command.y) {\n command.y = rf(command.y);\n }\n\n if (\"undefined\" !== typeof command.rX) {\n command.rX = rf(command.rX);\n }\n if (\"undefined\" !== typeof command.rY) {\n command.rY = rf(command.rY);\n }\n\n return command;\n };\n }\n // Relative to absolute commands\n export function TO_ABS() {\n return INFO((command, prevX, prevY) => {\n if (command.relative) {\n // x1/y1 values\n if (\"undefined\" !== typeof command.x1) {\n command.x1 += prevX;\n }\n if (\"undefined\" !== typeof command.y1) {\n command.y1 += prevY;\n }\n // x2/y2 values\n if (\"undefined\" !== typeof command.x2) {\n command.x2 += prevX;\n }\n if (\"undefined\" !== typeof command.y2) {\n command.y2 += prevY;\n }\n // Finally x/y values\n if (\"undefined\" !== typeof command.x) {\n command.x += prevX;\n }\n if (\"undefined\" !== typeof command.y) {\n command.y += prevY;\n }\n command.relative = false;\n }\n return command;\n });\n }\n // Absolute to relative commands\n export function TO_REL() {\n return INFO((command, prevX, prevY) => {\n if (!command.relative) {\n // x1/y1 values\n if (\"undefined\" !== typeof command.x1) {\n command.x1 -= prevX;\n }\n if (\"undefined\" !== typeof command.y1) {\n command.y1 -= prevY;\n }\n // x2/y2 values\n if (\"undefined\" !== typeof command.x2) {\n command.x2 -= prevX;\n }\n if (\"undefined\" !== typeof command.y2) {\n command.y2 -= prevY;\n }\n // Finally x/y values\n if (\"undefined\" !== typeof command.x) {\n command.x -= prevX;\n }\n if (\"undefined\" !== typeof command.y) {\n command.y -= prevY;\n }\n command.relative = true;\n }\n return command;\n });\n }\n // Convert H, V, Z and A with rX = 0 to L\n export function NORMALIZE_HVZ(normalizeZ = true, normalizeH = true, normalizeV = true) {\n return INFO((command, prevX, prevY, pathStartX, pathStartY) => {\n if (isNaN(pathStartX) && !(command.type & SVGPathData.MOVE_TO)) {\n throw new Error(\"path must start with moveto\");\n }\n if (normalizeH && command.type & SVGPathData.HORIZ_LINE_TO) {\n command.type = SVGPathData.LINE_TO;\n command.y = command.relative ? 0 : prevY;\n }\n if (normalizeV && command.type & SVGPathData.VERT_LINE_TO) {\n command.type = SVGPathData.LINE_TO;\n command.x = command.relative ? 0 : prevX;\n }\n if (normalizeZ && command.type & SVGPathData.CLOSE_PATH) {\n command.type = SVGPathData.LINE_TO;\n command.x = command.relative ? pathStartX - prevX : pathStartX;\n command.y = command.relative ? pathStartY - prevY : pathStartY;\n }\n if (command.type & SVGPathData.ARC && (0 === command.rX || 0 === command.rY)) {\n command.type = SVGPathData.LINE_TO;\n delete command.rX;\n delete command.rY;\n delete command.xRot;\n delete command.lArcFlag;\n delete command.sweepFlag;\n }\n return command;\n });\n }\n /*\n * Transforms smooth curves and quads to normal curves and quads (SsTt to CcQq)\n */\n export function NORMALIZE_ST() {\n let prevCurveC2X = NaN;\n let prevCurveC2Y = NaN;\n let prevQuadCX = NaN;\n let prevQuadCY = NaN;\n\n return INFO((command, prevX, prevY) => {\n if (command.type & SVGPathData.SMOOTH_CURVE_TO) {\n command.type = SVGPathData.CURVE_TO;\n prevCurveC2X = isNaN(prevCurveC2X) ? prevX : prevCurveC2X;\n prevCurveC2Y = isNaN(prevCurveC2Y) ? prevY : prevCurveC2Y;\n command.x1 = command.relative ? prevX - prevCurveC2X : 2 * prevX - prevCurveC2X;\n command.y1 = command.relative ? prevY - prevCurveC2Y : 2 * prevY - prevCurveC2Y;\n }\n if (command.type & SVGPathData.CURVE_TO) {\n prevCurveC2X = command.relative ? prevX + command.x2 : command.x2;\n prevCurveC2Y = command.relative ? prevY + command.y2 : command.y2;\n } else {\n prevCurveC2X = NaN;\n prevCurveC2Y = NaN;\n }\n if (command.type & SVGPathData.SMOOTH_QUAD_TO) {\n command.type = SVGPathData.QUAD_TO;\n prevQuadCX = isNaN(prevQuadCX) ? prevX : prevQuadCX;\n prevQuadCY = isNaN(prevQuadCY) ? prevY : prevQuadCY;\n command.x1 = command.relative ? prevX - prevQuadCX : 2 * prevX - prevQuadCX;\n command.y1 = command.relative ? prevY - prevQuadCY : 2 * prevY - prevQuadCY;\n }\n if (command.type & SVGPathData.QUAD_TO) {\n prevQuadCX = command.relative ? prevX + command.x1 : command.x1;\n prevQuadCY = command.relative ? prevY + command.y1 : command.y1;\n } else {\n prevQuadCX = NaN;\n prevQuadCY = NaN;\n }\n\n return command;\n });\n }\n /*\n * A quadratic bézier curve can be represented by a cubic bézier curve which has\n * the same end points as the quadratic and both control points in place of the\n * quadratic\"s one.\n *\n * This transformer replaces QqTt commands with Cc commands respectively.\n * This is useful for reading path data into a system which only has a\n * representation for cubic curves.\n */\n export function QT_TO_C() {\n let prevQuadX1 = NaN;\n let prevQuadY1 = NaN;\n\n return INFO((command, prevX, prevY) => {\n if (command.type & SVGPathData.SMOOTH_QUAD_TO) {\n command.type = SVGPathData.QUAD_TO;\n prevQuadX1 = isNaN(prevQuadX1) ? prevX : prevQuadX1;\n prevQuadY1 = isNaN(prevQuadY1) ? prevY : prevQuadY1;\n command.x1 = command.relative ? prevX - prevQuadX1 : 2 * prevX - prevQuadX1;\n command.y1 = command.relative ? prevY - prevQuadY1 : 2 * prevY - prevQuadY1;\n }\n if (command.type & SVGPathData.QUAD_TO) {\n prevQuadX1 = command.relative ? prevX + command.x1 : command.x1;\n prevQuadY1 = command.relative ? prevY + command.y1 : command.y1;\n const x1 = command.x1;\n const y1 = command.y1;\n\n command.type = SVGPathData.CURVE_TO;\n command.x1 = ((command.relative ? 0 : prevX) + x1 * 2) / 3;\n command.y1 = ((command.relative ? 0 : prevY) + y1 * 2) / 3;\n command.x2 = (command.x + x1 * 2) / 3;\n command.y2 = (command.y + y1 * 2) / 3;\n } else {\n prevQuadX1 = NaN;\n prevQuadY1 = NaN;\n }\n\n return command;\n });\n }\n export function INFO(\n f: (command: any, prevXAbs: number, prevYAbs: number,\n pathStartXAbs: number, pathStartYAbs: number) => any | any[]) {\n let prevXAbs = 0;\n let prevYAbs = 0;\n let pathStartXAbs = NaN;\n let pathStartYAbs = NaN;\n\n return function transform(command: any) {\n if (isNaN(pathStartXAbs) && !(command.type & SVGPathData.MOVE_TO)) {\n throw new Error(\"path must start with moveto\");\n }\n\n const result = f(command, prevXAbs, prevYAbs, pathStartXAbs, pathStartYAbs);\n\n if (command.type & SVGPathData.CLOSE_PATH) {\n prevXAbs = pathStartXAbs;\n prevYAbs = pathStartYAbs;\n }\n\n if (\"undefined\" !== typeof command.x) {\n prevXAbs = (command.relative ? prevXAbs + command.x : command.x);\n }\n if (\"undefined\" !== typeof command.y) {\n prevYAbs = (command.relative ? prevYAbs + command.y : command.y);\n }\n\n if (command.type & SVGPathData.MOVE_TO) {\n pathStartXAbs = prevXAbs;\n pathStartYAbs = prevYAbs;\n }\n\n return result;\n };\n }\n /*\n * remove 0-length segments\n */\n export function SANITIZE(EPS = 0) {\n assertNumbers(EPS);\n let prevCurveC2X = NaN;\n let prevCurveC2Y = NaN;\n let prevQuadCX = NaN;\n let prevQuadCY = NaN;\n\n return INFO((command, prevX, prevY, pathStartX, pathStartY) => {\n const abs = Math.abs;\n let skip = false;\n let x1Rel = 0;\n let y1Rel = 0;\n\n if (command.type & SVGPathData.SMOOTH_CURVE_TO) {\n x1Rel = isNaN(prevCurveC2X) ? 0 : prevX - prevCurveC2X;\n y1Rel = isNaN(prevCurveC2Y) ? 0 : prevY - prevCurveC2Y;\n }\n if (command.type & (SVGPathData.CURVE_TO | SVGPathData.SMOOTH_CURVE_TO)) {\n prevCurveC2X = command.relative ? prevX + command.x2 : command.x2;\n prevCurveC2Y = command.relative ? prevY + command.y2 : command.y2;\n } else {\n prevCurveC2X = NaN;\n prevCurveC2Y = NaN;\n }\n if (command.type & SVGPathData.SMOOTH_QUAD_TO) {\n prevQuadCX = isNaN(prevQuadCX) ? prevX : 2 * prevX - prevQuadCX;\n prevQuadCY = isNaN(prevQuadCY) ? prevY : 2 * prevY - prevQuadCY;\n } else if (command.type & SVGPathData.QUAD_TO) {\n prevQuadCX = command.relative ? prevX + command.x1 : command.x1;\n prevQuadCY = command.relative ? prevY + command.y1 : command.y2;\n } else {\n prevQuadCX = NaN;\n prevQuadCY = NaN;\n }\n\n if (command.type & SVGPathData.LINE_COMMANDS ||\n command.type & SVGPathData.ARC && (0 === command.rX || 0 === command.rY || !command.lArcFlag) ||\n command.type & SVGPathData.CURVE_TO || command.type & SVGPathData.SMOOTH_CURVE_TO ||\n command.type & SVGPathData.QUAD_TO || command.type & SVGPathData.SMOOTH_QUAD_TO) {\n const xRel = \"undefined\" === typeof command.x ? 0 :\n (command.relative ? command.x : command.x - prevX);\n const yRel = \"undefined\" === typeof command.y ? 0 :\n (command.relative ? command.y : command.y - prevY);\n\n x1Rel = !isNaN(prevQuadCX) ? prevQuadCX - prevX :\n \"undefined\" === typeof command.x1 ? x1Rel :\n command.relative ? command.x :\n command.x1 - prevX;\n y1Rel = !isNaN(prevQuadCY) ? prevQuadCY - prevY :\n \"undefined\" === typeof command.y1 ? y1Rel :\n command.relative ? command.y :\n command.y1 - prevY;\n\n const x2Rel = \"undefined\" === typeof command.x2 ? 0 :\n (command.relative ? command.x : command.x2 - prevX);\n const y2Rel = \"undefined\" === typeof command.y2 ? 0 :\n (command.relative ? command.y : command.y2 - prevY);\n\n if (abs(xRel) <= EPS && abs(yRel) <= EPS &&\n abs(x1Rel) <= EPS && abs(y1Rel) <= EPS &&\n abs(x2Rel) <= EPS && abs(y2Rel) <= EPS) {\n skip = true;\n }\n }\n\n if (command.type & SVGPathData.CLOSE_PATH) {\n if (abs(prevX - pathStartX) <= EPS && abs(prevY - pathStartY) <= EPS) {\n skip = true;\n }\n }\n\n return skip ? [] : command;\n });\n }\n // SVG Transforms : http://www.w3.org/TR/SVGTiny12/coords.html#TransformList\n // Matrix : http://apike.ca/prog_svg_transform.html\n // a c e\n // b d f\n export function MATRIX(a: number, b: number, c: number, d: number, e: number, f: number) {\n assertNumbers(a, b, c, d, e, f);\n\n return INFO((command, prevX, prevY, pathStartX) => {\n const origX1 = command.x1;\n const origX2 = command.x2;\n // if isNaN(pathStartX), then this is the first command, which is ALWAYS an\n // absolute MOVE_TO, regardless what the relative flag says\n const comRel = command.relative && !isNaN(pathStartX);\n const x = \"undefined\" !== typeof command.x ? command.x : (comRel ? 0 : prevX);\n const y = \"undefined\" !== typeof command.y ? command.y : (comRel ? 0 : prevY);\n\n if (command.type & SVGPathData.HORIZ_LINE_TO && 0 !== b) {\n command.type = SVGPathData.LINE_TO;\n command.y = command.relative ? 0 : prevY;\n }\n if (command.type & SVGPathData.VERT_LINE_TO && 0 !== c) {\n command.type = SVGPathData.LINE_TO;\n command.x = command.relative ? 0 : prevX;\n }\n\n if (\"undefined\" !== typeof command.x) {\n command.x = (command.x * a) + (y * c) + (comRel ? 0 : e);\n }\n if (\"undefined\" !== typeof command.y) {\n command.y = (x * b) + command.y * d + (comRel ? 0 : f);\n }\n if (\"undefined\" !== typeof command.x1) {\n command.x1 = command.x1 * a + command.y1 * c + (comRel ? 0 : e);\n }\n if (\"undefined\" !== typeof command.y1) {\n command.y1 = origX1 * b + command.y1 * d + (comRel ? 0 : f);\n }\n if (\"undefined\" !== typeof command.x2) {\n command.x2 = command.x2 * a + command.y2 * c + (comRel ? 0 : e);\n }\n if (\"undefined\" !== typeof command.y2) {\n command.y2 = origX2 * b + command.y2 * d + (comRel ? 0 : f);\n }\n function sqr(x: number) { return x * x; }\n const det = a * d - b * c;\n\n if (\"undefined\" !== typeof command.xRot) {\n // Skip if this is a pure translation\n if (1 !== a || 0 !== b || 0 !== c || 1 !== d) {\n // Special case for singular matrix\n if (0 === det) {\n // In the singular case, the arc is compressed to a line. The actual geometric image of the original\n // curve under this transform possibly extends beyond the starting and/or ending points of the segment, but\n // for simplicity we ignore this detail and just replace this command with a single line segment.\n delete command.rX;\n delete command.rY;\n delete command.xRot;\n delete command.lArcFlag;\n delete command.sweepFlag;\n command.type = SVGPathData.LINE_TO;\n } else {\n // Convert to radians\n const xRot = command.xRot * Math.PI / 180;\n\n // Convert rotated ellipse to general conic form\n // x0^2/rX^2 + y0^2/rY^2 - 1 = 0\n // x0 = x*cos(xRot) + y*sin(xRot)\n // y0 = -x*sin(xRot) + y*cos(xRot)\n // --> A*x^2 + B*x*y + C*y^2 - 1 = 0, where\n const sinRot = Math.sin(xRot);\n const cosRot = Math.cos(xRot);\n const xCurve = 1 / sqr(command.rX);\n const yCurve = 1 / sqr(command.rY);\n const A = sqr(cosRot) * xCurve + sqr(sinRot) * yCurve;\n const B = 2 * sinRot * cosRot * (xCurve - yCurve);\n const C = sqr(sinRot) * xCurve + sqr(cosRot) * yCurve;\n\n // Apply matrix to A*x^2 + B*x*y + C*y^2 - 1 = 0\n // x1 = a*x + c*y\n // y1 = b*x + d*y\n // (we can ignore e and f, since pure translations don\"t affect the shape of the ellipse)\n // --> A1*x1^2 + B1*x1*y1 + C1*y1^2 - det^2 = 0, where\n const A1 = A * d * d - B * b * d + C * b * b;\n const B1 = B * (a * d + b * c) - 2 * (A * c * d + C * a * b);\n const C1 = A * c * c - B * a * c + C * a * a;\n\n // Unapply newXRot to get back to axis-aligned ellipse equation\n // x1 = x2*cos(newXRot) - y2*sin(newXRot)\n // y1 = x2*sin(newXRot) + y2*cos(newXRot)\n // A1*x1^2 + B1*x1*y1 + C1*y1^2 - det^2 =\n // x2^2*(A1*cos(newXRot)^2 + B1*sin(newXRot)*cos(newXRot) + C1*sin(newXRot)^2)\n // + x2*y2*(2*(C1 - A1)*sin(newXRot)*cos(newXRot) + B1*(cos(newXRot)^2 - sin(newXRot)^2))\n // + y2^2*(A1*sin(newXRot)^2 - B1*sin(newXRot)*cos(newXRot) + C1*cos(newXRot)^2)\n // (which must have the same zeroes as)\n // x2^2/newRX^2 + y2^2/newRY^2 - 1\n // (so we have)\n // 2*(C1 - A1)*sin(newXRot)*cos(newXRot) + B1*(cos(newXRot)^2 - sin(newXRot)^2) = 0\n // (A1 - C1)*sin(2*newXRot) = B1*cos(2*newXRot)\n // 2*newXRot = atan2(B1, A1 - C1)\n const newXRot = ((Math.atan2(B1, A1 - C1) + Math.PI) % Math.PI) / 2;\n // For any integer n, (atan2(B1, A1 - C1) + n*pi)/2 is a solution to the above; incrementing n just swaps\n // the x and y radii computed below (since that\"s what rotating an ellipse by pi/2 does). Choosing the\n // rotation between 0 and pi/2 eliminates the ambiguity and leads to more predictable output.\n\n // Finally, we get newRX and newRY from the same-zeroes relationship that gave us newXRot\n const newSinRot = Math.sin(newXRot);\n const newCosRot = Math.cos(newXRot);\n\n command.rX = Math.abs(det) /\n Math.sqrt(A1 * sqr(newCosRot) + B1 * newSinRot * newCosRot + C1 * sqr(newSinRot));\n command.rY = Math.abs(det) /\n Math.sqrt(A1 * sqr(newSinRot) - B1 * newSinRot * newCosRot + C1 * sqr(newCosRot));\n command.xRot = newXRot * 180 / Math.PI;\n }\n }\n }\n // sweepFlag needs to be inverted when mirroring shapes\n // see http://www.itk.ilstu.edu/faculty/javila/SVG/SVG_drawing1/elliptical_curve.htm\n // m 65,10 a 50,25 0 1 0 50,25\n // M 65,60 A 50,25 0 1 1 115,35\n if (\"undefined\" !== typeof command.sweepFlag && 0 > det) {\n command.sweepFlag = +!command.sweepFlag;\n }\n return command;\n });\n }\n export function ROTATE(a: number, x = 0, y = 0) {\n assertNumbers(a, x, y);\n const sin = Math.sin(a);\n const cos = Math.cos(a);\n\n return MATRIX(cos, sin, -sin, cos, x - x * cos + y * sin, y - x * sin - y * cos);\n }\n export function TRANSLATE(dX: number, dY = 0) {\n assertNumbers(dX, dY);\n return MATRIX(1, 0, 0, 1, dX, dY);\n }\n export function SCALE(dX: number, dY = dX) {\n assertNumbers(dX, dY);\n return MATRIX(dX, 0, 0, dY, 0, 0);\n }\n export function SKEW_X(a: number) {\n assertNumbers(a);\n return MATRIX(1, 0, Math.atan(a), 1, 0, 0);\n }\n export function SKEW_Y(a: number) {\n assertNumbers(a);\n return MATRIX(1, Math.atan(a), 0, 1, 0, 0);\n }\n export function X_AXIS_SYMMETRY(xOffset = 0) {\n assertNumbers(xOffset);\n return MATRIX(-1, 0, 0, 1, xOffset, 0);\n }\n export function Y_AXIS_SYMMETRY(yOffset = 0) {\n assertNumbers(yOffset);\n return MATRIX(1, 0, 0, -1, 0, yOffset);\n }\n // Convert arc commands to curve commands\n export function A_TO_C() {\n return INFO((command, prevX, prevY) => {\n if (SVGPathData.ARC === command.type) {\n return a2c(command, command.relative ? 0 : prevX, command.relative ? 0 : prevY);\n }\n return command;\n });\n }\n // @see annotateArcCommand\n export function ANNOTATE_ARCS() {\n return INFO((c, x1, y1) => {\n if (c.relative) {\n x1 = 0;\n y1 = 0;\n }\n if (SVGPathData.ARC === c.type) {\n annotateArcCommand(c, x1, y1);\n }\n return c;\n });\n }\n export function CLONE() {\n return (c: SVGCommand) => {\n const result = {} as SVGCommand;\n // tslint:disable-next-line\n for (const key in c) {\n result[key as keyof SVGCommand] = c[key as keyof SVGCommand];\n }\n return result;\n };\n }\n // @see annotateArcCommand\n export function CALCULATE_BOUNDS() {\n const clone = CLONE();\n const toAbs = TO_ABS();\n const qtToC = QT_TO_C();\n const normST = NORMALIZE_ST();\n const f: TransformFunction & {minX: number, maxX: number, minY: number, maxY: number} =\n INFO((command, prevXAbs, prevYAbs) => {\n const c = normST(qtToC(toAbs(clone(command))));\n function fixX(absX: number) {\n if (absX > f.maxX) { f.maxX = absX; }\n if (absX < f.minX) { f.minX = absX; }\n }\n function fixY(absY: number) {\n if (absY > f.maxY) { f.maxY = absY; }\n if (absY < f.minY) { f.minY = absY; }\n }\n if (c.type & SVGPathData.DRAWING_COMMANDS) {\n fixX(prevXAbs);\n fixY(prevYAbs);\n }\n if (c.type & SVGPathData.HORIZ_LINE_TO) {\n fixX(c.x);\n }\n if (c.type & SVGPathData.VERT_LINE_TO) {\n fixY(c.y);\n }\n if (c.type & SVGPathData.LINE_TO) {\n fixX(c.x);\n fixY(c.y);\n }\n if (c.type & SVGPathData.CURVE_TO) {\n // add start and end points\n fixX(c.x);\n fixY(c.y);\n const xDerivRoots = bezierRoot(prevXAbs, c.x1, c.x2, c.x);\n\n for (const derivRoot of xDerivRoots) {\n if (0 < derivRoot && 1 > derivRoot) {\n fixX(bezierAt(prevXAbs, c.x1, c.x2, c.x, derivRoot));\n }\n }\n const yDerivRoots = bezierRoot(prevYAbs, c.y1, c.y2, c.y);\n\n for (const derivRoot of yDerivRoots) {\n if (0 < derivRoot && 1 > derivRoot) {\n fixY(bezierAt(prevYAbs, c.y1, c.y2, c.y, derivRoot));\n }\n }\n }\n if (c.type & SVGPathData.ARC) {\n // add start and end points\n fixX(c.x);\n fixY(c.y);\n annotateArcCommand(c, prevXAbs, prevYAbs);\n // p = cos(phi) * xv + sin(phi) * yv\n // dp = -sin(phi) * xv + cos(phi) * yv = 0\n const xRotRad = c.xRot / 180 * Math.PI;\n // points on ellipse for phi = 0° and phi = 90°\n const x0 = Math.cos(xRotRad) * c.rX;\n const y0 = Math.sin(xRotRad) * c.rX;\n const x90 = -Math.sin(xRotRad) * c.rY;\n const y90 = Math.cos(xRotRad) * c.rY;\n\n // annotateArcCommand returns phi1 and phi2 such that -180° < phi1 < 180° and phi2 is smaller or greater\n // depending on the sweep flag. Calculate phiMin, phiMax such that -180° < phiMin < 180° and phiMin < phiMax\n const [phiMin, phiMax] = c.phi1 < c.phi2 ?\n [c.phi1, c.phi2] :\n (-180 > c.phi2 ? [c.phi2 + 360, c.phi1 + 360] : [c.phi2, c.phi1]);\n const normalizeXiEta = ([xi, eta]: [number, number]) => {\n const phiRad = Math.atan2(eta, xi);\n const phi = phiRad * 180 / Math.PI;\n\n return phi < phiMin ? phi + 360 : phi;\n };\n // xi = cos(phi), eta = sin(phi)\n\n const xDerivRoots = intersectionUnitCircleLine(x90, -x0, 0).map(normalizeXiEta);\n for (const derivRoot of xDerivRoots) {\n if (derivRoot > phiMin && derivRoot < phiMax) {\n fixX(arcAt(c.cX, x0, x90, derivRoot));\n }\n }\n\n const yDerivRoots = intersectionUnitCircleLine(y90, -y0, 0).map(normalizeXiEta);\n for (const derivRoot of yDerivRoots) {\n if (derivRoot > phiMin && derivRoot < phiMax) {\n fixY(arcAt(c.cY, y0, y90, derivRoot));\n }\n }\n }\n return command;\n }) as any;\n\n f.minX = Infinity;\n f.maxX = -Infinity;\n f.minY = Infinity;\n f.maxY = -Infinity;\n return f;\n }\n}\n","import { SVGPathDataTransformer } from \"./SVGPathDataTransformer\";\nimport { TransformFunction } from \"./types\";\n\nexport abstract class TransformableSVG {\n round(x?: number) {\n return this.transform(SVGPathDataTransformer.ROUND(x));\n }\n\n toAbs() {\n return this.transform(SVGPathDataTransformer.TO_ABS());\n }\n\n toRel() {\n return this.transform(SVGPathDataTransformer.TO_REL());\n }\n\n normalizeHVZ(a?: boolean, b?: boolean, c?: boolean) {\n return this.transform(SVGPathDataTransformer.NORMALIZE_HVZ(a, b, c));\n }\n\n normalizeST() {\n return this.transform(SVGPathDataTransformer.NORMALIZE_ST());\n }\n\n qtToC() {\n return this.transform(SVGPathDataTransformer.QT_TO_C());\n }\n\n aToC() {\n return this.transform(SVGPathDataTransformer.A_TO_C());\n }\n\n sanitize(eps?: number) {\n return this.transform(SVGPathDataTransformer.SANITIZE(eps));\n }\n\n translate(x: number, y?: number) {\n return this.transform(SVGPathDataTransformer.TRANSLATE(x, y));\n }\n\n scale(x: number, y?: number) {\n return this.transform(SVGPathDataTransformer.SCALE(x, y));\n }\n\n rotate(a: number, x?: number, y?: number) {\n return this.transform(SVGPathDataTransformer.ROTATE(a, x, y));\n }\n\n matrix(a: number, b: number, c: number, d: number, e: number, f: number) {\n return this.transform(SVGPathDataTransformer.MATRIX(a, b, c, d, e, f));\n }\n\n skewX(a: number) {\n return this.transform(SVGPathDataTransformer.SKEW_X(a));\n }\n\n skewY(a: number) {\n return this.transform(SVGPathDataTransformer.SKEW_Y(a));\n }\n\n xSymmetry(xOffset?: number) {\n return this.transform(SVGPathDataTransformer.X_AXIS_SYMMETRY(xOffset));\n }\n\n ySymmetry(yOffset?: number) {\n return this.transform(SVGPathDataTransformer.Y_AXIS_SYMMETRY(yOffset));\n }\n\n annotateArcs() {\n return this.transform(SVGPathDataTransformer.ANNOTATE_ARCS());\n }\n\n abstract transform(transformFunction: TransformFunction): this;\n}\n","// Parse SVG PathData\n// http://www.w3.org/TR/SVG/paths.html#PathDataBNF\nimport { COMMAND_ARG_COUNTS, SVGPathData } from \"./SVGPathData\";\nimport { TransformableSVG } from \"./TransformableSVG\";\nimport { SVGCommand, TransformFunction } from \"./types\";\n// Private consts : Char groups\nconst isWhiteSpace = (c: string) =>\n \" \" === c || \"\\t\" === c || \"\\r\" === c || \"\\n\" === c;\nconst isDigit = (c: string) =>\n \"0\".charCodeAt(0) <= c.charCodeAt(0) && c.charCodeAt(0) <= \"9\".charCodeAt(0);\nconst COMMANDS = \"mMzZlLhHvVcCsSqQtTaA\";\n\nexport class SVGPathDataParser extends TransformableSVG {\n private curNumber: string = \"\";\n private curCommandType: SVGCommand[\"type\"] | -1 = -1;\n private curCommandRelative = false;\n private canParseCommandOrComma = true;\n private curNumberHasExp = false;\n private curNumberHasExpDigits = false;\n private curNumberHasDecimal = false;\n private curArgs: number[] = [];\n\n constructor() {\n super();\n }\n\n finish(commands: SVGCommand[] = []) {\n this.parse(\" \", commands);\n // Adding residual command\n if (0 !== this.curArgs.length || !this.canParseCommandOrComma) {\n throw new SyntaxError(\"Unterminated command at the path end.\");\n }\n return commands;\n }\n\n parse(str: string, commands: SVGCommand[] = []) {\n const finishCommand = (command: SVGCommand) => {\n commands.push(command);\n this.curArgs.length = 0;\n this.canParseCommandOrComma = true;\n };\n\n for (let i = 0; i < str.length; i++) {\n const c = str[i];\n // White spaces parsing\n const isAArcFlag = this.curCommandType === SVGPathData.ARC &&\n (this.curArgs.length === 3 || this.curArgs.length === 4) &&\n this.curNumber.length === 1 &&\n (this.curNumber === \"0\" || this.curNumber === \"1\");\n const isEndingDigit = isDigit(c) && (\n (this.curNumber === \"0\" && c === \"0\") ||\n isAArcFlag\n );\n\n if (\n isDigit(c) &&\n !isEndingDigit\n ) {\n this.curNumber += c;\n this.curNumberHasExpDigits = this.curNumberHasExp;\n continue;\n }\n if (\"e\" === c || \"E\" === c) {\n this.curNumber += c;\n this.curNumberHasExp = true;\n continue;\n }\n if (\n (\"-\" === c || \"+\" === c) &&\n this.curNumberHasExp &&\n !this.curNumberHasExpDigits\n ) {\n this.curNumber += c;\n continue;\n }\n // if we already have a \".\", it means we are starting a new number\n if (\".\" === c && !this.curNumberHasExp && !this.curNumberHasDecimal && !isAArcFlag) {\n this.curNumber += c;\n this.curNumberHasDecimal = true;\n continue;\n }\n\n // New number\n if (this.curNumber && -1 !== this.curCommandType) {\n const val = Number(this.curNumber);\n if (isNaN(val)) {\n throw new SyntaxError(`Invalid number ending at ${i}`);\n }\n if (this.curCommandType === SVGPathData.ARC) {\n if (0 === this.curArgs.length || 1 === this.curArgs.length) {\n if (0 > val) {\n throw new SyntaxError(\n `Expected positive number, got \"${val}\" at index \"${i}\"`,\n );\n }\n } else if (3 === this.curArgs.length || 4 === this.curArgs.length) {\n if (\"0\" !== this.curNumber && \"1\" !== this.curNumber) {\n throw new SyntaxError(\n `Expected a flag, got \"${this.curNumber}\" at index \"${i}\"`,\n );\n }\n }\n }\n this.curArgs.push(val);\n if (this.curArgs.length === COMMAND_ARG_COUNTS[this.curCommandType]) {\n if (SVGPathData.HORIZ_LINE_TO === this.curCommandType) {\n finishCommand({\n type: SVGPathData.HORIZ_LINE_TO,\n relative: this.curCommandRelative,\n x: val,\n });\n } else if (SVGPathData.VERT_LINE_TO === this.curCommandType) {\n finishCommand({\n type: SVGPathData.VERT_LINE_TO,\n relative: this.curCommandRelative,\n y: val,\n });\n // Move to / line to / smooth quadratic curve to commands (x, y)\n } else if (\n this.curCommandType === SVGPathData.MOVE_TO ||\n this.curCommandType === SVGPathData.LINE_TO ||\n this.curCommandType === SVGPathData.SMOOTH_QUAD_TO\n ) {\n finishCommand({\n type: this.curCommandType,\n relative: this.curCommandRelative,\n x: this.curArgs[0],\n y: this.curArgs[1],\n } as SVGCommand);\n // Switch to line to state\n if (SVGPathData.MOVE_TO === this.curCommandType) {\n this.curCommandType = SVGPathData.LINE_TO;\n }\n } else if (this.curCommandType === SVGPathData.CURVE_TO) {\n finishCommand({\n type: SVGPathData.CURVE_TO,\n relative: this.curCommandRelative,\n x1: this.curArgs[0],\n y1: this.curArgs[1],\n x2: this.curArgs[2],\n y2: this.curArgs[3],\n x: this.curArgs[4],\n y: this.curArgs[5],\n });\n } else if (this.curCommandType === SVGPathData.SMOOTH_CURVE_TO) {\n finishCommand({\n type: SVGPathData.SMOOTH_CURVE_TO,\n relative: this.curCommandRelative,\n x2: this.curArgs[0],\n y2: this.curArgs[1],\n x: this.curArgs[2],\n y: this.curArgs[3],\n });\n } else if (this.curCommandType === SVGPathData.QUAD_TO) {\n finishCommand({\n type: SVGPathData.QUAD_TO,\n relative: this.curCommandRelative,\n x1: this.curArgs[0],\n y1: this.curArgs[1],\n x: this.curArgs[2],\n y: this.curArgs[3],\n });\n } else if (this.curCommandType === SVGPathData.ARC) {\n finishCommand({\n type: SVGPathData.ARC,\n relative: this.curCommandRelative,\n rX: this.curArgs[0],\n rY: this.curArgs[1],\n xRot: this.curArgs[2],\n lArcFlag: this.curArgs[3] as 0 | 1,\n sweepFlag: this.curArgs[4] as 0 | 1,\n x: this.curArgs[5],\n y: this.curArgs[6],\n });\n }\n }\n this.curNumber = \"\";\n this.curNumberHasExpDigits = false;\n this.curNumberHasExp = false;\n this.curNumberHasDecimal = false;\n this.canParseCommandOrComma = true;\n }\n // Continue if a white space or a comma was detected\n if (isWhiteSpace(c)) {\n continue;\n }\n if (\",\" === c && this.canParseCommandOrComma) {\n // L 0,0, H is not valid:\n this.canParseCommandOrComma = false;\n continue;\n }\n // if a sign is detected, then parse the new number\n if (\"+\" === c || \"-\" === c || \".\" === c) {\n this.curNumber = c;\n this.curNumberHasDecimal = \".\" === c;\n continue;\n }\n // if a 0 is detected, then parse the new number\n if (isEndingDigit) {\n this.curNumber = c;\n this.curNumberHasDecimal = false;\n continue;\n }\n\n // Adding residual command\n if (0 !== this.curArgs.length) {\n throw new SyntaxError(`Unterminated command at index ${i}.`);\n }\n if (!this.canParseCommandOrComma) {\n throw new SyntaxError(\n `Unexpected character \"${c}\" at index ${i}. Command cannot follow comma`,\n );\n }\n this.canParseCommandOrComma = false;\n // Detecting the next command\n if (\"z\" === c || \"Z\" === c) {\n commands.push({\n type: SVGPathData.CLOSE_PATH,\n });\n this.canParseCommandOrComma = true;\n this.curCommandType = -1;\n continue;\n // Horizontal move to command\n } else if (\"h\" === c || \"H\" === c) {\n this.curCommandType = SVGPathData.HORIZ_LINE_TO;\n this.curCommandRelative = \"h\" === c;\n // Vertical move to command\n } else if (\"v\" === c || \"V\" === c) {\n this.curCommandType = SVGPathData.VERT_LINE_TO;\n this.curCommandRelative = \"v\" === c;\n // Move to command\n } else if (\"m\" === c || \"M\" === c) {\n this.curCommandType = SVGPathData.MOVE_TO;\n this.curCommandRelative = \"m\" === c;\n // Line to command\n } else if (\"l\" === c || \"L\" === c) {\n this.curCommandType = SVGPathData.LINE_TO;\n this.curCommandRelative = \"l\" === c;\n // Curve to command\n } else if (\"c\" === c || \"C\" === c) {\n this.curCommandType = SVGPathData.CURVE_TO;\n this.curCommandRelative = \"c\" === c;\n // Smooth curve to command\n } else if (\"s\" === c || \"S\" === c) {\n this.curCommandType = SVGPathData.SMOOTH_CURVE_TO;\n this.curCommandRelative = \"s\" === c;\n // Quadratic bezier curve to command\n } else if (\"q\" === c || \"Q\" === c) {\n this.curCommandType = SVGPathData.QUAD_TO;\n this.curCommandRelative = \"q\" === c;\n // Smooth quadratic bezier curve to command\n } else if (\"t\" === c || \"T\" === c) {\n this.curCommandType = SVGPathData.SMOOTH_QUAD_TO;\n this.curCommandRelative = \"t\" === c;\n // Elliptic arc command\n } else if (\"a\" === c || \"A\" === c) {\n this.curCommandType = SVGPathData.ARC;\n this.curCommandRelative = \"a\" === c;\n } else {\n throw new SyntaxError(`Unexpected character \"${c}\" at index ${i}.`);\n }\n }\n return commands;\n }\n /**\n * Return a wrapper around this parser which applies the transformation on parsed commands.\n */\n transform(transform: TransformFunction) {\n const result = Object.create(this, {\n parse: {\n value(chunk: string, commands: SVGCommand[] = []) {\n const parsedCommands = Object.getPrototypeOf(this).parse.call(\n this,\n chunk,\n );\n for (const c of parsedCommands) {\n const cT = transform(c);\n if (Array.isArray(cT)) {\n commands.push(...cT);\n } else {\n commands.push(cT);\n }\n }\n return commands;\n },\n },\n });\n return result as this;\n }\n}\n","import { encodeSVGPath } from \"./SVGPathDataEncoder\";\nimport { SVGPathDataParser } from \"./SVGPathDataParser\";\nimport { SVGPathDataTransformer } from \"./SVGPathDataTransformer\";\nimport { TransformableSVG } from \"./TransformableSVG\";\nimport { SVGCommand } from \"./types\";\n\nexport class SVGPathData extends TransformableSVG {\n commands: SVGCommand[];\n constructor(content: string | SVGCommand[]) {\n super();\n if (\"string\" === typeof content) {\n this.commands = SVGPathData.parse(content);\n } else {\n this.commands = content;\n }\n }\n\n encode() {\n return SVGPathData.encode(this.commands);\n }\n\n getBounds() {\n const boundsTransform = SVGPathDataTransformer.CALCULATE_BOUNDS();\n\n this.transform(boundsTransform);\n return boundsTransform;\n }\n\n transform(\n transformFunction: (input: SVGCommand) => SVGCommand | SVGCommand[],\n ) {\n const newCommands = [];\n\n for (const command of this.commands) {\n const transformedCommand = transformFunction(command);\n\n if (Array.isArray(transformedCommand)) {\n newCommands.push(...transformedCommand);\n } else {\n newCommands.push(transformedCommand);\n }\n }\n this.commands = newCommands;\n return this;\n }\n\n static encode(commands: SVGCommand[]) {\n return encodeSVGPath(commands);\n }\n\n static parse(path: string) {\n const parser = new SVGPathDataParser();\n const commands: SVGCommand[] = [];\n parser.parse(path, commands);\n parser.finish(commands);\n return commands;\n }\n\n static readonly CLOSE_PATH: 1 = 1;\n static readonly MOVE_TO: 2 = 2;\n static readonly HORIZ_LINE_TO: 4 = 4;\n static readonly VERT_LINE_TO: 8 = 8;\n static readonly LINE_TO: 16 = 16;\n static readonly CURVE_TO: 32 = 32;\n static readonly SMOOTH_CURVE_TO: 64 = 64;\n static readonly QUAD_TO: 128 = 128;\n static readonly SMOOTH_QUAD_TO: 256 = 256;\n static readonly ARC: 512 = 512;\n static readonly LINE_COMMANDS = SVGPathData.LINE_TO | SVGPathData.HORIZ_LINE_TO | SVGPathData.VERT_LINE_TO;\n static readonly DRAWING_COMMANDS = SVGPathData.HORIZ_LINE_TO | SVGPathData.VERT_LINE_TO | SVGPathData.LINE_TO |\n SVGPathData.CURVE_TO | SVGPathData.SMOOTH_CURVE_TO | SVGPathData.QUAD_TO |\n SVGPathData.SMOOTH_QUAD_TO | SVGPathData.ARC;\n}\n\nexport const COMMAND_ARG_COUNTS = {\n [SVGPathData.MOVE_TO]: 2,\n [SVGPathData.LINE_TO]: 2,\n [SVGPathData.HORIZ_LINE_TO]: 1,\n [SVGPathData.VERT_LINE_TO]: 1,\n [SVGPathData.CLOSE_PATH]: 0,\n [SVGPathData.QUAD_TO]: 4,\n [SVGPathData.SMOOTH_QUAD_TO]: 2,\n [SVGPathData.CURVE_TO]: 6,\n [SVGPathData.SMOOTH_CURVE_TO]: 4,\n [SVGPathData.ARC]: 7,\n};\n\nexport {encodeSVGPath} from \"./SVGPathDataEncoder\";\nexport {SVGPathDataParser} from \"./SVGPathDataParser\";\nexport {SVGPathDataTransformer} from \"./SVGPathDataTransformer\";\n","import { Vector2 } from '@babylonjs/core/Maths/math';\r\nimport { SVGPathData } from 'svg-pathdata';\r\nimport {\r\n CommandA,\r\n CommandC,\r\n CommandH,\r\n CommandL,\r\n CommandM,\r\n CommandQ,\r\n CommandS,\r\n CommandT,\r\n CommandV,\r\n CommandZ,\r\n SVGCommand,\r\n} from 'svg-pathdata/lib/types';\r\n\r\nexport type CommandType = SVGCommand['type'];\r\n// export type Command = { type: CommandType } & Omit &\r\n// Omit &\r\n// Omit &\r\n// Omit &\r\n// Omit &\r\n// Omit &\r\n// Omit &\r\n// Omit &\r\n// Omit &\r\n// Omit;\r\n\r\nexport type Command =\r\n | CommandA\r\n | CommandC\r\n | CommandH\r\n | CommandL\r\n | CommandM\r\n | CommandQ\r\n | CommandS\r\n | CommandT\r\n | CommandV\r\n | CommandZ;\r\n\r\nexport class PathParser extends SVGPathData {\r\n control: Vector2 = new Vector2(0, 0);\r\n start: Vector2 = new Vector2(0, 0);\r\n current: Vector2 = new Vector2(0, 0);\r\n command: Command | null = null;\r\n override readonly commands: Command[] /* Babel fix: */ = this.commands;\r\n private i = -1;\r\n private previousCommand: Command | null = null;\r\n private points: Vector2[] = [];\r\n private angles: Array = [];\r\n\r\n constructor(path: string) {\r\n super(\r\n path\r\n // Fix spaces after signs.\r\n .replace(/([+\\-.])\\s+/gm, '$1')\r\n // Remove invalid part.\r\n .replace(/[^MmZzLlHhVvCcSsQqTtAae\\d\\s.,+-].*/g, '')\r\n );\r\n }\r\n\r\n reset() {\r\n this.i = -1;\r\n this.command = null;\r\n this.previousCommand = null;\r\n this.start = new Vector2(0, 0);\r\n this.control = new Vector2(0, 0);\r\n this.current = new Vector2(0, 0);\r\n this.points = [];\r\n this.angles = [];\r\n }\r\n\r\n isEnd() {\r\n const { i, commands } = this;\r\n\r\n return i >= commands.length - 1;\r\n }\r\n\r\n next() {\r\n const command = this.commands[++this.i];\r\n\r\n this.previousCommand = this.command;\r\n this.command = command;\r\n\r\n return command;\r\n }\r\n\r\n getPoint(xProp = 'x', yProp = 'y') {\r\n if (this.command) {\r\n const x = parseInt(xProp);\r\n const y = parseInt(yProp);\r\n const point = new Vector2(x, y);\r\n return this.makeAbsolute(point);\r\n }\r\n }\r\n\r\n getAsControlPoint(xProp?: string, yProp?: string) {\r\n const point = this.getPoint(xProp, yProp);\r\n if (point) {\r\n this.control = point;\r\n }\r\n\r\n return point;\r\n }\r\n\r\n getAsCurrentPoint(xProp?: string, yProp?: string) {\r\n const point = this.getPoint(xProp, yProp);\r\n if (point) {\r\n this.current = point;\r\n }\r\n\r\n return point;\r\n }\r\n\r\n getReflectedControlPoint() {\r\n let previousCommand = null;\r\n if (this.previousCommand) {\r\n previousCommand = this.previousCommand.type;\r\n }\r\n\r\n if (\r\n previousCommand !== SVGPathData.CURVE_TO &&\r\n previousCommand !== SVGPathData.SMOOTH_CURVE_TO &&\r\n previousCommand !== SVGPathData.QUAD_TO &&\r\n previousCommand !== SVGPathData.SMOOTH_QUAD_TO\r\n ) {\r\n return this.current;\r\n }\r\n\r\n // reflect point\r\n const {\r\n current: { x: cx, y: cy },\r\n control: { x: ox, y: oy },\r\n } = this;\r\n const point = new Vector2(2 * cx - ox, 2 * cy - oy);\r\n\r\n return point;\r\n }\r\n\r\n makeAbsolute(point: Vector2) {\r\n if (this.command && 'relative' in this.command && this.command.relative) {\r\n const { x, y } = this.current;\r\n\r\n point.x += x;\r\n point.y += y;\r\n }\r\n\r\n return point;\r\n }\r\n\r\n addMarker(point: Vector2, from?: Vector2, priorTo?: Vector2) {\r\n const { points, angles } = this;\r\n\r\n // if the last angle isn't filled in because we didn't have this point yet ...\r\n if (priorTo && angles.length > 0 && !angles[angles.length - 1]) {\r\n angles[angles.length - 1] = this.angleBetweenPoints(points[points.length - 1], priorTo);\r\n }\r\n\r\n this.addMarkerAngle(point, from ? this.angleBetweenPoints(from, point) : null);\r\n }\r\n\r\n addMarkerAngle(point: Vector2, angle: number | null) {\r\n this.points.push(point);\r\n this.angles.push(angle);\r\n }\r\n\r\n angleBetweenPoints(point1: Vector2, point2: Vector2) {\r\n return Math.atan2(point2.y - point1.y, point2.x - point1.x);\r\n }\r\n\r\n getMarkerPoints() {\r\n return this.points;\r\n }\r\n\r\n getMarkerAngles() {\r\n const { angles } = this;\r\n const len = angles.length;\r\n\r\n for (let i = 0; i < len; i++) {\r\n if (!angles[i]) {\r\n for (let j = i + 1; j < len; j++) {\r\n if (angles[j]) {\r\n angles[i] = angles[j];\r\n break;\r\n }\r\n }\r\n }\r\n }\r\n\r\n return angles;\r\n }\r\n}\r\n","import {\r\n eSketchPathType,\r\n Kb3dManager,\r\n KbVector,\r\n SketchControlPoint,\r\n SketchNode,\r\n SketchPath,\r\n Token,\r\n} from '@models/classes';\r\nimport {\r\n CommandA,\r\n CommandC,\r\n CommandH,\r\n CommandL,\r\n CommandM,\r\n CommandQ,\r\n CommandS,\r\n CommandT,\r\n CommandV,\r\n} from 'svg-pathdata/lib/types';\r\nimport { PathParser } from './path-parser';\r\nimport { projectOnNodeAxis } from './util';\r\n\r\nexport interface Layer {\r\n segments: Segment[];\r\n boundingbox: number[];\r\n holechildren: any[];\r\n isholepath: boolean;\r\n}\r\n\r\nexport interface Segment {\r\n type: 'L' | 'M' | 'C' | 'S' | 'Q' | 'T' | 'Z';\r\n x1: number;\r\n y1: number;\r\n x2: number;\r\n y2: number;\r\n x3?: number;\r\n y3?: number;\r\n}\r\n\r\nexport class PathHandler {\r\n private pathParser: PathParser;\r\n private nodeManager: Kb3dManager;\r\n private _currentPosition: KbVector | undefined;\r\n private initialPosition = new KbVector(0, 0, 0);\r\n\r\n get currentPosition() {\r\n return this._currentPosition || this.initialPosition;\r\n }\r\n\r\n readonly stubPath = { id: 'stub' } as SketchPath;\r\n\r\n constructor(private sketchNode: SketchNode, private scaleDivider: number, nodeManager: Kb3dManager) {\r\n this.nodeManager = nodeManager;\r\n }\r\n\r\n parseTrace(commands: Segment[]) {\r\n const newPaths: SketchPath[] = [];\r\n if (!this.sketchNode || commands.length === 0) {\r\n return [];\r\n }\r\n\r\n const first = commands[0];\r\n\r\n if (first.type !== 'M') {\r\n this.pathM(newPaths, {\r\n type: PathParser.MOVE_TO,\r\n relative: false,\r\n x: first.x1,\r\n y: first.y1,\r\n });\r\n }\r\n\r\n for (let i = 0; i < commands.length; i++) {\r\n const command = commands[i];\r\n const commonPathCommand = {\r\n relative: false,\r\n x: command.x2,\r\n y: command.y2,\r\n x1: command.x3 || -1,\r\n y1: command.y3 || -1,\r\n };\r\n switch (command.type) {\r\n case 'M': {\r\n const pathCommand: CommandM = {\r\n ...commonPathCommand,\r\n type: PathParser.MOVE_TO,\r\n };\r\n this.pathM(newPaths, pathCommand);\r\n break;\r\n }\r\n case 'L': {\r\n const pathCommand: CommandL = {\r\n ...commonPathCommand,\r\n type: PathParser.LINE_TO,\r\n };\r\n this.pathL(newPaths, pathCommand);\r\n break;\r\n }\r\n case 'Q': {\r\n if (!command.x3 || !command.y3) {\r\n console.warn('Q command without x3 or y3');\r\n } else {\r\n const pathCommand: CommandQ = {\r\n ...commonPathCommand,\r\n type: PathParser.QUAD_TO,\r\n x: command.x3,\r\n y: command.y3,\r\n x1: command.x2,\r\n y1: command.y2,\r\n };\r\n this.pathQ(newPaths, pathCommand);\r\n }\r\n break;\r\n }\r\n /** imagetracer does not return C, S, or T paths\r\n * \r\n case 'C': {\r\n const pathCommand: CommandC = {\r\n ...commonPathCommand,\r\n type: PathParser.CURVE_TO,\r\n };\r\n this.pathC(newPaths, pathCommand);\r\n break;\r\n }\r\n case 'S': {\r\n const pathCommand: CommandS = {\r\n ...commonPathCommand,\r\n type: PathParser.SMOOTH_CURVE_TO,\r\n };\r\n this.pathS(newPaths, pathCommand);\r\n break;\r\n }\r\n case 'T': {\r\n const pathCommand: CommandT = {\r\n ...commonPathCommand,\r\n type: PathParser.SMOOTH_QUAD_TO,\r\n };\r\n this.pathT(newPaths, pathCommand);\r\n break;\r\n }\r\n */\r\n case 'Z': {\r\n this.pathZ(newPaths);\r\n }\r\n }\r\n }\r\n\r\n return this.reprojectPaths(newPaths.filter(path => path !== this.stubPath));\r\n }\r\n\r\n parsePath(path: string) {\r\n this.pathParser = new PathParser(path);\r\n this.pathParser.reset();\r\n\r\n if (!this.sketchNode) {\r\n return [];\r\n }\r\n const newPaths: SketchPath[] = [];\r\n\r\n while (!this.pathParser.isEnd()) {\r\n switch (this.pathParser.next().type) {\r\n case PathParser.MOVE_TO:\r\n this.pathM(newPaths);\r\n break;\r\n\r\n case PathParser.LINE_TO:\r\n this.pathL(newPaths);\r\n break;\r\n\r\n case PathParser.HORIZ_LINE_TO:\r\n this.pathL(newPaths);\r\n break;\r\n\r\n case PathParser.VERT_LINE_TO:\r\n this.pathL(newPaths);\r\n break;\r\n\r\n case PathParser.CURVE_TO:\r\n this.pathC(newPaths);\r\n break;\r\n\r\n case PathParser.SMOOTH_CURVE_TO:\r\n this.pathS(newPaths);\r\n break;\r\n\r\n case PathParser.QUAD_TO:\r\n this.pathQ(newPaths);\r\n break;\r\n\r\n case PathParser.SMOOTH_QUAD_TO:\r\n this.pathT(newPaths);\r\n break;\r\n\r\n case PathParser.ARC:\r\n console.warn('ARC not implemented');\r\n this.pathA(newPaths);\r\n break;\r\n\r\n case PathParser.CLOSE_PATH:\r\n this.pathZ(newPaths);\r\n break;\r\n\r\n default:\r\n }\r\n }\r\n\r\n return this.reprojectPaths(newPaths.filter(path => path !== this.stubPath));\r\n }\r\n\r\n protected reprojectPaths(paths: SketchPath[]) {\r\n paths.forEach(path => {\r\n path.controlPoints.forEach(cp => {\r\n cp.position = projectOnNodeAxis(this.sketchNode, cp.position.x, cp.position.y);\r\n cp.direction = projectOnNodeAxis(this.sketchNode, cp.direction.x, cp.direction.y);\r\n if (cp.direction2) {\r\n cp.direction2 = projectOnNodeAxis(this.sketchNode, cp.direction2.x, cp.direction2.y);\r\n }\r\n });\r\n });\r\n\r\n return paths;\r\n }\r\n\r\n protected checkIfNewPathNeeded(paths: SketchPath[], desiredPathType: eSketchPathType) {\r\n let activePath = paths[paths.length - 1];\r\n if (!activePath || activePath === this.stubPath || activePath.type !== desiredPathType) {\r\n if (activePath === this.stubPath) {\r\n paths.pop();\r\n }\r\n\r\n activePath = this.addNewPath(paths, desiredPathType);\r\n const newControlPoint = this.nodeManager.create(Token.SketchControlPoint);\r\n if (!this._currentPosition) {\r\n console.warn('new path created with no moveto command');\r\n }\r\n newControlPoint.position = this.currentPosition;\r\n\r\n activePath.controlPoints.push(newControlPoint);\r\n newControlPoint.direction2 = new KbVector(0, 0, 0);\r\n }\r\n\r\n return activePath;\r\n }\r\n\r\n protected pathM(paths: SketchPath[], command?: CommandM) {\r\n const { pathParser } = this;\r\n let x = 0;\r\n let y = 0;\r\n\r\n if (!command && pathParser.command) {\r\n command = pathParser.command as CommandM;\r\n }\r\n if (command) {\r\n x = command.x / this.scaleDivider;\r\n y = (command.y * -1) / this.scaleDivider;\r\n\r\n if (command.relative && this.currentPosition) {\r\n x = x + this.currentPosition.x;\r\n y = y + this.currentPosition.y;\r\n }\r\n\r\n this._currentPosition = new KbVector(x, y, 0);\r\n this.initialPosition = this._currentPosition;\r\n paths.push(this.stubPath);\r\n }\r\n }\r\n\r\n protected pathL(paths: SketchPath[], command?: CommandL | CommandH | CommandV) {\r\n const { pathParser } = this;\r\n\r\n if (!command && pathParser.command) {\r\n command = pathParser.command as CommandL;\r\n }\r\n if (command) {\r\n let x: number | undefined;\r\n let y: number | undefined;\r\n\r\n //these 2 if checks need to happen because pathL could be a horizontal or vertical line and therefore not have one of the coordinates.\r\n if ('x' in command) {\r\n x = command.x / this.scaleDivider;\r\n }\r\n if ('y' in command) {\r\n y = (command.y * -1) / this.scaleDivider;\r\n }\r\n\r\n const activePath = this.checkIfNewPathNeeded(paths, eSketchPathType.points);\r\n\r\n if (command.relative && activePath) {\r\n if (x !== undefined) {\r\n x = x + this.currentPosition.x;\r\n }\r\n if (y !== undefined) {\r\n y = y + this.currentPosition.y;\r\n }\r\n }\r\n if (x === undefined) {\r\n // V case\r\n x = this.currentPosition.x || 0;\r\n }\r\n if (y === undefined) {\r\n // H case\r\n y = this.currentPosition.y || 0;\r\n }\r\n\r\n const newControlPoint = this.nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint.position = new KbVector(x, y, 0);\r\n this._currentPosition = newControlPoint.position;\r\n activePath.controlPoints.push(newControlPoint);\r\n }\r\n }\r\n\r\n protected pathC(paths: SketchPath[], command?: CommandC) {\r\n const { pathParser } = this;\r\n let x = 0;\r\n let y = 0;\r\n let x1 = 0;\r\n let y1 = 0;\r\n let x2 = 0;\r\n let y2 = 0;\r\n let lastControlPoint = undefined;\r\n\r\n if (!command && pathParser.command) {\r\n command = pathParser.command as CommandC;\r\n }\r\n if (command) {\r\n x = command.x / this.scaleDivider;\r\n y = (command.y * -1) / this.scaleDivider;\r\n x1 = command.x1 / this.scaleDivider;\r\n y1 = (command.y1 * -1) / this.scaleDivider;\r\n x2 = command.x2 / this.scaleDivider;\r\n y2 = (command.y2 * -1) / this.scaleDivider;\r\n\r\n const activePath = this.checkIfNewPathNeeded(paths, eSketchPathType.cubicBezier);\r\n lastControlPoint = activePath.controlPoints[activePath.controlPoints.length - 1];\r\n\r\n if (command.relative) {\r\n x = x + this.currentPosition.x;\r\n y = y + this.currentPosition.y;\r\n x1 = x1 + this.currentPosition.x;\r\n y1 = y1 + this.currentPosition.y;\r\n x2 = x2 + this.currentPosition.x;\r\n y2 = y2 + this.currentPosition.y;\r\n }\r\n\r\n if (lastControlPoint) {\r\n lastControlPoint.direction = new KbVector(\r\n x1 - lastControlPoint.position.x,\r\n y1 - lastControlPoint.position.y,\r\n 0\r\n );\r\n }\r\n const newControlPoint = this.nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint.position = new KbVector(x, y, 0);\r\n this._currentPosition = newControlPoint.position;\r\n newControlPoint.direction2 = new KbVector(x2 - x, y2 - y, 0);\r\n activePath.controlPoints.push(newControlPoint);\r\n }\r\n }\r\n\r\n protected pathS(paths: SketchPath[], command?: CommandS) {\r\n const { pathParser } = this;\r\n let x = 0;\r\n let y = 0;\r\n let x2 = 0;\r\n let y2 = 0;\r\n let lastControlPoint = undefined;\r\n\r\n if (!command && pathParser.command) {\r\n command = pathParser.command as CommandS;\r\n }\r\n if (command) {\r\n x = command.x / this.scaleDivider;\r\n y = (command.y * -1) / this.scaleDivider;\r\n x2 = command.x2 / this.scaleDivider;\r\n y2 = (command.y2 * -1) / this.scaleDivider;\r\n\r\n const activePath = this.checkIfNewPathNeeded(paths, eSketchPathType.cubicBezier);\r\n lastControlPoint = activePath.controlPoints[activePath.controlPoints.length - 1];\r\n lastControlPoint.direction = new KbVector(0, 0, 0);\r\n\r\n if (command.relative) {\r\n x = x + this.currentPosition.x;\r\n y = y + this.currentPosition.y;\r\n x2 = x2 + this.currentPosition.x;\r\n y2 = y2 + this.currentPosition.y;\r\n }\r\n\r\n const newControlPoint = this.nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint.position = new KbVector(x, y, 0);\r\n if (lastControlPoint && lastControlPoint.direction2) {\r\n lastControlPoint.direction = lastControlPoint.direction2?.scale(-1);\r\n newControlPoint.direction2 = new KbVector(x2 - x, y2 - y, 0);\r\n }\r\n this._currentPosition = newControlPoint.position;\r\n activePath.controlPoints.push(newControlPoint);\r\n\r\n return activePath;\r\n }\r\n }\r\n\r\n protected pathQ(paths: SketchPath[], command?: CommandQ) {\r\n const { pathParser } = this;\r\n let x = 0;\r\n let y = 0;\r\n let x1 = 0;\r\n let y1 = 0;\r\n let lastControlPoint = undefined;\r\n\r\n if (!command && pathParser.command) {\r\n command = pathParser.command as CommandQ;\r\n }\r\n if (command) {\r\n x = command.x / this.scaleDivider;\r\n y = (command.y * -1) / this.scaleDivider;\r\n x1 = command.x1 / this.scaleDivider;\r\n y1 = (command.y1 * -1) / this.scaleDivider;\r\n\r\n const activePath = this.checkIfNewPathNeeded(paths, eSketchPathType.quadraticBezier);\r\n lastControlPoint = activePath.controlPoints[activePath.controlPoints.length - 1];\r\n\r\n if (command.relative) {\r\n x = x + this.currentPosition.x;\r\n y = y + this.currentPosition.y;\r\n x1 = x1 + this.currentPosition.x;\r\n y1 = y1 + this.currentPosition.y;\r\n }\r\n\r\n if (lastControlPoint) {\r\n lastControlPoint.direction = new KbVector(\r\n x1 - lastControlPoint.position.x,\r\n y1 - lastControlPoint.position.y,\r\n 0\r\n );\r\n }\r\n\r\n const newControlPoint = this.nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint.position = new KbVector(x, y, 0);\r\n this._currentPosition = newControlPoint.position;\r\n activePath.controlPoints.push(newControlPoint);\r\n }\r\n }\r\n\r\n protected pathT(paths: SketchPath[], command?: CommandT) {\r\n const { pathParser } = this;\r\n let x = 0;\r\n let y = 0;\r\n let lastControlPoint = undefined;\r\n\r\n if (!command && pathParser.command) {\r\n command = pathParser.command as CommandT;\r\n }\r\n if (command) {\r\n x = command.x / this.scaleDivider;\r\n y = (command.y * -1) / this.scaleDivider;\r\n\r\n const activePath = this.checkIfNewPathNeeded(paths, eSketchPathType.quadraticBezier);\r\n lastControlPoint = activePath.controlPoints[activePath.controlPoints.length - 1];\r\n lastControlPoint.direction = new KbVector(0, 0, 0);\r\n\r\n if (command.relative) {\r\n x = x + this.currentPosition.x;\r\n y = y + this.currentPosition.y;\r\n }\r\n\r\n const secondLastControlPoint = activePath.controlPoints[activePath.controlPoints.length - 2];\r\n const newControlPoint = this.nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint.position = new KbVector(x, y, 0);\r\n this._currentPosition = newControlPoint.position;\r\n activePath.controlPoints.push(newControlPoint);\r\n if (lastControlPoint && secondLastControlPoint && secondLastControlPoint) {\r\n lastControlPoint.direction = this.calculateReflection(\r\n secondLastControlPoint.direction,\r\n secondLastControlPoint,\r\n lastControlPoint\r\n );\r\n }\r\n }\r\n }\r\n\r\n //Path A needs further implementing\r\n protected pathA(paths: SketchPath[], command?: CommandA) {\r\n const { pathParser } = this;\r\n\r\n if (!command && pathParser.command) {\r\n command = pathParser.command as CommandA;\r\n }\r\n if (command) {\r\n // const cX = command.cX;\r\n // const cY = command.cY;\r\n // const rX = command.rX;\r\n // const rY = command.rY;\r\n let x = command.x;\r\n let y = command.y;\r\n if (command.relative) {\r\n x = x + this.currentPosition.x;\r\n y = y + this.currentPosition.y;\r\n }\r\n // const xAxisRotation = command.xRot;\r\n // const largeArcFlag = command.lArcFlag;\r\n // const sweepFlag = command.sweepFlag;\r\n\r\n this._currentPosition = new KbVector(x, y, 0);\r\n }\r\n }\r\n\r\n protected pathZ(paths: SketchPath[]) {\r\n const activePath = this.checkIfNewPathNeeded(paths, eSketchPathType.points);\r\n\r\n const x = this.initialPosition.x;\r\n const y = this.initialPosition.y;\r\n\r\n const newControlPoint = this.nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint.position = new KbVector(x, y, 0);\r\n this._currentPosition = undefined;\r\n activePath.controlPoints.push(newControlPoint);\r\n\r\n return activePath;\r\n }\r\n\r\n protected addNewPath(paths: SketchPath[], pathType?: eSketchPathType) {\r\n const newPath = this.nodeManager.create(Token.SketchPath);\r\n newPath.name = 'Path';\r\n newPath._expanded = true;\r\n paths.push(newPath);\r\n\r\n if (pathType) {\r\n newPath.type = pathType;\r\n }\r\n\r\n return newPath;\r\n }\r\n\r\n protected calculateReflection(\r\n vector: KbVector,\r\n controlPoint: SketchControlPoint,\r\n controlPoint2: SketchControlPoint\r\n ) {\r\n const n = controlPoint2.position.subtract(controlPoint.position).normalize();\r\n return vector.subtract(n.scale(2 * vector.dotProduct(n))).scale(-1);\r\n }\r\n}\r\n","const VERSION_NUMBER = \"0.2.1\";\nconst PATHSCAN_COMBINED_LOOKUP = [\n [[-1, -1, -1, -1], [-1, -1, -1, -1], [-1, -1, -1, -1], [-1, -1, -1, -1]],\n [[0, 1, 0, -1], [-1, -1, -1, -1], [-1, -1, -1, -1], [0, 2, -1, 0]],\n [[-1, -1, -1, -1], [-1, -1, -1, -1], [0, 1, 0, -1], [0, 0, 1, 0]],\n [[0, 0, 1, 0], [-1, -1, -1, -1], [0, 2, -1, 0], [-1, -1, -1, -1]],\n [[-1, -1, -1, -1], [0, 0, 1, 0], [0, 3, 0, 1], [-1, -1, -1, -1]],\n [[13, 3, 0, 1], [13, 2, -1, 0], [7, 1, 0, -1], [7, 0, 1, 0]],\n [[-1, -1, -1, -1], [0, 1, 0, -1], [-1, -1, -1, -1], [0, 3, 0, 1]],\n [[0, 3, 0, 1], [0, 2, -1, 0], [-1, -1, -1, -1], [-1, -1, -1, -1]],\n [[0, 3, 0, 1], [0, 2, -1, 0], [-1, -1, -1, -1], [-1, -1, -1, -1]],\n [[-1, -1, -1, -1], [0, 1, 0, -1], [-1, -1, -1, -1], [0, 3, 0, 1]],\n [[11, 1, 0, -1], [14, 0, 1, 0], [14, 3, 0, 1], [11, 2, -1, 0]],\n [[-1, -1, -1, -1], [0, 0, 1, 0], [0, 3, 0, 1], [-1, -1, -1, -1]],\n [[0, 0, 1, 0], [-1, -1, -1, -1], [0, 2, -1, 0], [-1, -1, -1, -1]],\n [[-1, -1, -1, -1], [-1, -1, -1, -1], [0, 1, 0, -1], [0, 0, 1, 0]],\n [[0, 1, 0, -1], [-1, -1, -1, -1], [-1, -1, -1, -1], [0, 2, -1, 0]],\n [[-1, -1, -1, -1], [-1, -1, -1, -1], [-1, -1, -1, -1], [-1, -1, -1, -1]]\n];\nconst SPECPALETTE = [\n { r: 0, g: 0, b: 0, a: 255 },\n { r: 128, g: 128, b: 128, a: 255 },\n { r: 0, g: 0, b: 128, a: 255 },\n { r: 64, g: 64, b: 128, a: 255 },\n { r: 192, g: 192, b: 192, a: 255 },\n { r: 255, g: 255, b: 255, a: 255 },\n { r: 128, g: 128, b: 192, a: 255 },\n { r: 0, g: 0, b: 192, a: 255 },\n { r: 128, g: 0, b: 0, a: 255 },\n { r: 128, g: 64, b: 64, a: 255 },\n { r: 128, g: 0, b: 128, a: 255 },\n { r: 168, g: 168, b: 168, a: 255 },\n { r: 192, g: 128, b: 128, a: 255 },\n { r: 192, g: 0, b: 0, a: 255 },\n { r: 255, g: 255, b: 255, a: 255 },\n { r: 0, g: 128, b: 0, a: 255 }\n];\nconst GKS = [\n [0.27901, 0.44198, 0.27901],\n [0.135336, 0.228569, 0.272192, 0.228569, 0.135336],\n [0.086776, 0.136394, 0.178908, 0.195843, 0.178908, 0.136394, 0.086776],\n [0.063327, 0.093095, 0.122589, 0.144599, 0.152781, 0.144599, 0.122589, 0.093095, 0.063327],\n [0.049692, 0.069304, 0.089767, 0.107988, 0.120651, 0.125194, 0.120651, 0.107988, 0.089767, 0.069304, 0.049692]\n];\n\nconst OPTION_PRESETS = {\n default: {\n corsenabled: false,\n ltres: 1,\n qtres: 1,\n pathomit: 8,\n rightangleenhance: true,\n colorsampling: 2,\n numberofcolors: 16,\n mincolorratio: 0,\n colorquantcycles: 3,\n layering: 0,\n strokewidth: 1,\n linefilter: false,\n scale: 1,\n roundcoords: 1,\n viewbox: false,\n desc: false,\n lcpr: 0,\n qcpr: 0,\n blurradius: 0,\n blurdelta: 20\n },\n posterized1: { colorsampling: 0, numberofcolors: 2 },\n posterized2: { numberofcolors: 4, blurradius: 5 },\n curvy: { ltres: 0.01, linefilter: true, rightangleenhance: false },\n sharp: { qtres: 0.01, linefilter: false },\n detailed: { pathomit: 0, roundcoords: 2, ltres: 0.5, qtres: 0.5, numberofcolors: 64 },\n smoothed: { blurradius: 5, blurdelta: 64 },\n grayscale: { colorsampling: 0, colorquantcycles: 1, numberofcolors: 7 },\n fixedpalette: { colorsampling: 0, colorquantcycles: 1, numberofcolors: 27 },\n randomsampling1: { colorsampling: 1, numberofcolors: 8 },\n randomsampling2: { colorsampling: 1, numberofcolors: 64 },\n artistic1: { colorsampling: 0, colorquantcycles: 1, pathomit: 0, blurradius: 5, blurdelta: 64, ltres: 0.01, linefilter: true, numberofcolors: 16, strokewidth: 2 },\n artistic2: { qtres: 0.01, colorsampling: 0, colorquantcycles: 1, numberofcolors: 4, strokewidth: 0 },\n artistic3: { qtres: 10, ltres: 10, numberofcolors: 8 },\n artistic4: { qtres: 10, ltres: 10, numberofcolors: 64, blurradius: 5, blurdelta: 256, strokewidth: 2 },\n posterized3: {\n ltres: 1,\n qtres: 1,\n pathomit: 20,\n rightangleenhance: true,\n colorsampling: 0,\n numberofcolors: 3,\n mincolorratio: 0,\n colorquantcycles: 3,\n blurradius: 3,\n blurdelta: 20,\n strokewidth: 0,\n linefilter: false,\n roundcoords: 1,\n pal: [{ r: 0, g: 0, b: 100, a: 255 }, { r: 255, g: 255, b: 255, a: 255 }]\n }\n};\n\nfunction generatePalette(numberOfColors) {\n const palette = [];\n let rcnt;\n let gcnt;\n let bcnt;\n if (numberOfColors < 8) {\n const graystep = Math.floor(255 / (numberOfColors - 1));\n for (let i = 0; i < numberOfColors; i++)\n palette.push({ r: i * graystep, g: i * graystep, b: i * graystep, a: 255 });\n } else {\n const colorqnum = Math.floor(Math.pow(numberOfColors, 1 / 3));\n const colorstep = Math.floor(255 / (colorqnum - 1));\n const rndnum = numberOfColors - colorqnum * colorqnum * colorqnum;\n for (rcnt = 0; rcnt < colorqnum; rcnt++) {\n for (gcnt = 0; gcnt < colorqnum; gcnt++) {\n for (bcnt = 0; bcnt < colorqnum; bcnt++)\n palette.push({ r: rcnt * colorstep, g: gcnt * colorstep, b: bcnt * colorstep, a: 255 });\n }\n }\n for (rcnt = 0; rcnt < rndnum; rcnt++)\n palette.push({ r: Math.floor(Math.random() * 255), g: Math.floor(Math.random() * 255), b: Math.floor(Math.random() * 255), a: Math.floor(Math.random() * 255) });\n }\n return palette;\n}\nfunction samplePalette(numberOfColors, imageData) {\n let idx;\n const palette = [];\n for (let i = 0; i < numberOfColors; i++) {\n idx = Math.floor(Math.random() * imageData.data.length / 4) * 4;\n palette.push({ r: imageData.data[idx], g: imageData.data[idx + 1], b: imageData.data[idx + 2], a: imageData.data[idx + 3] });\n }\n return palette;\n}\nfunction samplePaletteByGrid(numberOfColors, imageData) {\n let idx;\n const palette = [];\n const ni = Math.ceil(Math.sqrt(numberOfColors));\n const nj = Math.ceil(numberOfColors / ni);\n const vx = imageData.width / (ni + 1);\n const vy = imageData.height / (nj + 1);\n for (let j = 0; j < nj; j++) {\n for (let i = 0; i < ni; i++) {\n if (palette.length === numberOfColors) {\n break;\n } else {\n idx = Math.floor((j + 1) * vy * imageData.width + (i + 1) * vx) * 4;\n palette.push({ r: imageData.data[idx], g: imageData.data[idx + 1], b: imageData.data[idx + 2], a: imageData.data[idx + 3] });\n }\n }\n }\n return palette;\n}\n\nfunction blur(imageData, radius, delta) {\n let i, j, k, d, idx, racc, gacc, bacc, aacc, wacc;\n const imageData2 = new ImageData(imageData.width, imageData.height);\n radius = Math.floor(radius);\n if (radius < 1)\n return imageData;\n if (radius > 5)\n radius = 5;\n delta = Math.abs(delta);\n if (delta > 1024)\n delta = 1024;\n const thisgk = GKS[radius - 1];\n for (j = 0; j < imageData.height; j++) {\n for (i = 0; i < imageData.width; i++) {\n racc = 0;\n gacc = 0;\n bacc = 0;\n aacc = 0;\n wacc = 0;\n for (k = -radius; k < radius + 1; k++) {\n if (i + k > 0 && i + k < imageData.width) {\n idx = (j * imageData.width + i + k) * 4;\n racc += imageData.data[idx] * thisgk[k + radius];\n gacc += imageData.data[idx + 1] * thisgk[k + radius];\n bacc += imageData.data[idx + 2] * thisgk[k + radius];\n aacc += imageData.data[idx + 3] * thisgk[k + radius];\n wacc += thisgk[k + radius];\n }\n }\n idx = (j * imageData.width + i) * 4;\n imageData2.data[idx] = Math.floor(racc / wacc);\n imageData2.data[idx + 1] = Math.floor(gacc / wacc);\n imageData2.data[idx + 2] = Math.floor(bacc / wacc);\n imageData2.data[idx + 3] = Math.floor(aacc / wacc);\n }\n }\n const himageData = new Uint8ClampedArray(imageData2.data);\n for (j = 0; j < imageData.height; j++) {\n for (i = 0; i < imageData.width; i++) {\n racc = 0;\n gacc = 0;\n bacc = 0;\n aacc = 0;\n wacc = 0;\n for (k = -radius; k < radius + 1; k++) {\n if (j + k > 0 && j + k < imageData.height) {\n idx = ((j + k) * imageData.width + i) * 4;\n racc += himageData[idx] * thisgk[k + radius];\n gacc += himageData[idx + 1] * thisgk[k + radius];\n bacc += himageData[idx + 2] * thisgk[k + radius];\n aacc += himageData[idx + 3] * thisgk[k + radius];\n wacc += thisgk[k + radius];\n }\n }\n idx = (j * imageData.width + i) * 4;\n imageData2.data[idx] = Math.floor(racc / wacc);\n imageData2.data[idx + 1] = Math.floor(gacc / wacc);\n imageData2.data[idx + 2] = Math.floor(bacc / wacc);\n imageData2.data[idx + 3] = Math.floor(aacc / wacc);\n }\n }\n for (j = 0; j < imageData.height; j++) {\n for (i = 0; i < imageData.width; i++) {\n idx = (j * imageData.width + i) * 4;\n d = Math.abs(imageData2.data[idx] - imageData.data[idx]) + Math.abs(imageData2.data[idx + 1] - imageData.data[idx + 1]) + Math.abs(imageData2.data[idx + 2] - imageData.data[idx + 2]) + Math.abs(imageData2.data[idx + 3] - imageData.data[idx + 3]);\n if (d > delta) {\n imageData2.data[idx] = imageData.data[idx];\n imageData2.data[idx + 1] = imageData.data[idx + 1];\n imageData2.data[idx + 2] = imageData.data[idx + 2];\n imageData2.data[idx + 3] = imageData.data[idx + 3];\n }\n }\n }\n return imageData2;\n}\n\nfunction getDirection(x1, y1, x2, y2) {\n let result = 8;\n if (x1 < x2) {\n if (y1 < y2)\n result = 1;\n else if (y1 > y2)\n result = 7;\n else\n result = 0;\n } else if (x1 > x2) {\n if (y1 < y2)\n result = 3;\n else if (y1 > y2)\n result = 5;\n else\n result = 4;\n } else {\n if (y1 < y2)\n result = 2;\n else if (y1 > y2)\n result = 6;\n else\n result = 8;\n }\n return result;\n}\nconst roundToDec = (val, places = void 0) => +val.toFixed(places);\nfunction colorQuantization(imageData, options) {\n const arr = [];\n let idx = 0;\n let cd;\n let cdl;\n let ci;\n const paletteacc = [];\n const pixelnum = imageData.width * imageData.height;\n let i;\n let j;\n let k;\n let cnt;\n let palette;\n let imageDataCache = imageData.data;\n if (imageDataCache.length < pixelnum * 4) {\n const newImageData = new Uint8ClampedArray(pixelnum * 4);\n for (let pxcnt = 0; pxcnt < pixelnum; pxcnt++) {\n newImageData[pxcnt * 4] = imageDataCache[pxcnt * 3];\n newImageData[pxcnt * 4 + 1] = imageDataCache[pxcnt * 3 + 1];\n newImageData[pxcnt * 4 + 2] = imageDataCache[pxcnt * 3 + 2];\n newImageData[pxcnt * 4 + 3] = 255;\n }\n imageDataCache = newImageData;\n }\n for (j = 0; j < imageData.height + 2; j++) {\n arr[j] = [];\n for (i = 0; i < imageData.width + 2; i++)\n arr[j][i] = -1;\n }\n if (options.pal)\n palette = options.pal;\n else if (options.colorsampling === 0)\n palette = generatePalette(options.numberofcolors);\n else if (options.colorsampling === 1)\n palette = samplePalette(options.numberofcolors, imageData);\n else\n palette = samplePaletteByGrid(options.numberofcolors, imageData);\n if (options.blurradius > 0)\n imageData = blur(imageData, options.blurradius, options.blurdelta);\n for (cnt = 0; cnt < options.colorquantcycles; cnt++) {\n if (cnt > 0) {\n for (k = 0; k < palette.length; k++) {\n if (paletteacc[k].n > 0) {\n palette[k] = {\n r: Math.floor(paletteacc[k].r / paletteacc[k].n),\n g: Math.floor(paletteacc[k].g / paletteacc[k].n),\n b: Math.floor(paletteacc[k].b / paletteacc[k].n),\n a: Math.floor(paletteacc[k].a / paletteacc[k].n)\n };\n }\n if (paletteacc[k].n / pixelnum < options.mincolorratio && cnt < options.colorquantcycles - 1) {\n palette[k] = {\n r: Math.floor(Math.random() * 255),\n g: Math.floor(Math.random() * 255),\n b: Math.floor(Math.random() * 255),\n a: Math.floor(Math.random() * 255)\n };\n }\n }\n }\n for (i = 0; i < palette.length; i++)\n paletteacc[i] = { r: 0, g: 0, b: 0, a: 0, n: 0 };\n for (j = 0; j < imageData.height; j++) {\n for (i = 0; i < imageData.width; i++) {\n idx = (j * imageData.width + i) * 4;\n ci = 0;\n cdl = 1024;\n for (k = 0; k < palette.length; k++) {\n cd = (palette[k].r > imageDataCache[idx] ? palette[k].r - imageDataCache[idx] : imageDataCache[idx] - palette[k].r) + (palette[k].g > imageDataCache[idx + 1] ? palette[k].g - imageDataCache[idx + 1] : imageDataCache[idx + 1] - palette[k].g) + (palette[k].b > imageDataCache[idx + 2] ? palette[k].b - imageDataCache[idx + 2] : imageDataCache[idx + 2] - palette[k].b) + (palette[k].a > imageDataCache[idx + 3] ? palette[k].a - imageDataCache[idx + 3] : imageDataCache[idx + 3] - palette[k].a);\n if (cd < cdl) {\n cdl = cd;\n ci = k;\n }\n }\n paletteacc[ci].r += imageDataCache[idx];\n paletteacc[ci].g += imageDataCache[idx + 1];\n paletteacc[ci].b += imageDataCache[idx + 2];\n paletteacc[ci].a += imageDataCache[idx + 3];\n paletteacc[ci].n++;\n arr[j + 1][i + 1] = ci;\n }\n }\n }\n return { array: arr, palette };\n}\nconst toRgbaStr = (c) => `rgba(${c.r},${c.g},${c.b},${c.a})`;\n\nfunction svgPathString(tracedata, lnum, pathnum, options) {\n const layer = tracedata.layers[lnum];\n const smp = layer[pathnum];\n let str = \"\";\n let pcnt;\n if (options.linefilter && smp.segments.length < 3)\n return str;\n str = `= 0; pcnt--) {\n str += `${hsmp.segments[pcnt].type} `;\n if (hsmp.segments[pcnt].hasOwnProperty(\"x3\"))\n str += `${hsmp.segments[pcnt].x2 * options.scale} ${hsmp.segments[pcnt].y2 * options.scale} `;\n str += `${hsmp.segments[pcnt].x1 * options.scale} ${hsmp.segments[pcnt].y1 * options.scale} `;\n }\n } else {\n if (hsmp.segments[hsmp.segments.length - 1].hasOwnProperty(\"x3\"))\n str += `M ${roundToDec(hsmp.segments[hsmp.segments.length - 1].x3 * options.scale)} ${roundToDec(hsmp.segments[hsmp.segments.length - 1].y3 * options.scale)} `;\n else\n str += `M ${roundToDec(hsmp.segments[hsmp.segments.length - 1].x2 * options.scale)} ${roundToDec(hsmp.segments[hsmp.segments.length - 1].y2 * options.scale)} `;\n for (pcnt = hsmp.segments.length - 1; pcnt >= 0; pcnt--) {\n str += `${hsmp.segments[pcnt].type} `;\n if (hsmp.segments[pcnt].hasOwnProperty(\"x3\"))\n str += `${roundToDec(hsmp.segments[pcnt].x2 * options.scale)} ${roundToDec(hsmp.segments[pcnt].y2 * options.scale)} `;\n str += `${roundToDec(hsmp.segments[pcnt].x1 * options.scale)} ${roundToDec(hsmp.segments[pcnt].y1 * options.scale)} `;\n }\n }\n str += \"Z \";\n }\n str += '\" />';\n if (options.lcpr || options.qcpr) {\n for (pcnt = 0; pcnt < smp.segments.length; pcnt++) {\n if (smp.segments[pcnt].hasOwnProperty(\"x3\") && options.qcpr) {\n str += ``;\n str += ``;\n str += ``;\n str += ``;\n }\n if (!smp.segments[pcnt].hasOwnProperty(\"x3\") && options.lcpr)\n str += ``;\n }\n for (let hcnt = 0; hcnt < smp.holechildren.length; hcnt++) {\n const hsmp = layer[smp.holechildren[hcnt]];\n for (pcnt = 0; pcnt < hsmp.segments.length; pcnt++) {\n if (hsmp.segments[pcnt].hasOwnProperty(\"x3\") && options.qcpr) {\n str += ``;\n str += ``;\n str += ``;\n str += ``;\n }\n if (!hsmp.segments[pcnt].hasOwnProperty(\"x3\") && options.lcpr)\n str += ``;\n }\n }\n }\n return str;\n}\nfunction toSvgColorStr(c, options) {\n return `fill=\"rgb(${c.r},${c.g},${c.b})\" stroke=\"rgb(${c.r},${c.g},${c.b})\" stroke-width=\"${options.strokewidth}\" opacity=\"${c.a / 255}\" `;\n}\nfunction fitSeq(path, ltres, qtres, seqstart, seqend) {\n if (seqend > path.points.length || seqend < 0)\n return [];\n let errorpoint = seqstart;\n let errorval = 0;\n let curvepass = true;\n let px;\n let py;\n let dist2;\n let tl = seqend - seqstart;\n if (tl < 0)\n tl += path.points.length;\n const vx = (path.points[seqend].x - path.points[seqstart].x) / tl;\n const vy = (path.points[seqend].y - path.points[seqstart].y) / tl;\n let pcnt = (seqstart + 1) % path.points.length;\n let pl;\n while (pcnt !== seqend) {\n pl = pcnt - seqstart;\n if (pl < 0)\n pl += path.points.length;\n px = path.points[seqstart].x + vx * pl;\n py = path.points[seqstart].y + vy * pl;\n dist2 = (path.points[pcnt].x - px) * (path.points[pcnt].x - px) + (path.points[pcnt].y - py) * (path.points[pcnt].y - py);\n if (dist2 > ltres)\n curvepass = false;\n if (dist2 > errorval) {\n errorpoint = pcnt;\n errorval = dist2;\n }\n pcnt = (pcnt + 1) % path.points.length;\n }\n if (curvepass)\n return [{ type: \"L\", x1: path.points[seqstart].x, y1: path.points[seqstart].y, x2: path.points[seqend].x, y2: path.points[seqend].y }];\n const fitpoint = errorpoint;\n curvepass = true;\n errorval = 0;\n let t = (fitpoint - seqstart) / tl;\n let t1 = (1 - t) * (1 - t);\n let t2 = 2 * (1 - t) * t;\n let t3 = t * t;\n const cpx = (t1 * path.points[seqstart].x + t3 * path.points[seqend].x - path.points[fitpoint].x) / -t2;\n const cpy = (t1 * path.points[seqstart].y + t3 * path.points[seqend].y - path.points[fitpoint].y) / -t2;\n pcnt = seqstart + 1;\n while (pcnt !== seqend) {\n t = (pcnt - seqstart) / tl;\n t1 = (1 - t) * (1 - t);\n t2 = 2 * (1 - t) * t;\n t3 = t * t;\n px = t1 * path.points[seqstart].x + t2 * cpx + t3 * path.points[seqend].x;\n py = t1 * path.points[seqstart].y + t2 * cpy + t3 * path.points[seqend].y;\n dist2 = (path.points[pcnt].x - px) * (path.points[pcnt].x - px) + (path.points[pcnt].y - py) * (path.points[pcnt].y - py);\n if (dist2 > qtres)\n curvepass = false;\n if (dist2 > errorval) {\n errorpoint = pcnt;\n errorval = dist2;\n }\n pcnt = (pcnt + 1) % path.points.length;\n }\n if (curvepass)\n return [{ type: \"Q\", x1: path.points[seqstart].x, y1: path.points[seqstart].y, x2: cpx, y2: cpy, x3: path.points[seqend].x, y3: path.points[seqend].y }];\n const splitpoint = fitpoint;\n return fitSeq(path, ltres, qtres, seqstart, splitpoint).concat(fitSeq(path, ltres, qtres, splitpoint, seqend));\n}\nfunction boundingBoxIncludes(parentbbox, childbbox) {\n return parentbbox[0] < childbbox[0] && parentbbox[1] < childbbox[1] && parentbbox[2] > childbbox[2] && parentbbox[3] > childbbox[3];\n}\nfunction pointInPoly(p, pa) {\n let isin = false;\n for (let i = 0, j = pa.length - 1; i < pa.length; j = i++) {\n isin = pa[i].y > p.y !== pa[j].y > p.y && p.x < (pa[j].x - pa[i].x) * (p.y - pa[i].y) / (pa[j].y - pa[i].y) + pa[i].x ? !isin : isin;\n }\n return isin;\n}\nfunction getSvgString(tracedata, options) {\n const w = tracedata.width * options.scale;\n const h = tracedata.height * options.scale;\n let svgstr = ``;\n for (let lcnt = 0; lcnt < tracedata.layers.length; lcnt++) {\n for (let pcnt = 0; pcnt < tracedata.layers[lcnt].length; pcnt++) {\n if (!tracedata.layers[lcnt][pcnt].isholepath)\n svgstr += svgPathString(tracedata, lcnt, pcnt, options);\n }\n }\n svgstr += \"\";\n return svgstr;\n}\n\nfunction testRightAngle(path, idx1, idx2, idx3, idx4, idx5) {\n return path.points[idx3].x === path.points[idx1].x && path.points[idx3].x === path.points[idx2].x && path.points[idx3].y === path.points[idx4].y && path.points[idx3].y === path.points[idx5].y || path.points[idx3].y === path.points[idx1].y && path.points[idx3].y === path.points[idx2].y && path.points[idx3].x === path.points[idx4].x && path.points[idx3].x === path.points[idx5].x;\n}\nfunction pathScan(arr, pathomit) {\n const paths = [];\n let pacnt = 0;\n let pcnt = 0;\n let px = 0;\n let py = 0;\n const w = arr[0].length;\n const h = arr.length;\n let dir = 0;\n let pathfinished = true;\n let holepath = false;\n let lookuprow;\n for (let j = 0; j < h; j++) {\n for (let i = 0; i < w; i++) {\n if (arr[j][i] === 4 || arr[j][i] === 11) {\n px = i;\n py = j;\n paths[pacnt] = {};\n paths[pacnt].points = [];\n paths[pacnt].boundingbox = [px, py, px, py];\n paths[pacnt].holechildren = [];\n pathfinished = false;\n pcnt = 0;\n holepath = arr[j][i] === 11;\n dir = 1;\n while (!pathfinished) {\n paths[pacnt].points[pcnt] = {};\n paths[pacnt].points[pcnt].x = px - 1;\n paths[pacnt].points[pcnt].y = py - 1;\n paths[pacnt].points[pcnt].t = arr[py][px];\n if (px - 1 < paths[pacnt].boundingbox[0])\n paths[pacnt].boundingbox[0] = px - 1;\n if (px - 1 > paths[pacnt].boundingbox[2])\n paths[pacnt].boundingbox[2] = px - 1;\n if (py - 1 < paths[pacnt].boundingbox[1])\n paths[pacnt].boundingbox[1] = py - 1;\n if (py - 1 > paths[pacnt].boundingbox[3])\n paths[pacnt].boundingbox[3] = py - 1;\n lookuprow = PATHSCAN_COMBINED_LOOKUP[arr[py][px]][dir];\n arr[py][px] = lookuprow[0];\n dir = lookuprow[1];\n px += lookuprow[2];\n py += lookuprow[3];\n if (px - 1 === paths[pacnt].points[0].x && py - 1 === paths[pacnt].points[0].y) {\n pathfinished = true;\n if (paths[pacnt].points.length < pathomit) {\n paths.pop();\n } else {\n paths[pacnt].isholepath = !!holepath;\n if (holepath) {\n let parentidx = 0;\n let parentbbox = [-1, -1, w + 1, h + 1];\n for (let parentcnt = 0; parentcnt < pacnt; parentcnt++) {\n if (!paths[parentcnt].isholepath && boundingBoxIncludes(paths[parentcnt].boundingbox, paths[pacnt].boundingbox) && boundingBoxIncludes(parentbbox, paths[parentcnt].boundingbox) && pointInPoly(paths[pacnt].points[0], paths[parentcnt].points)) {\n parentidx = parentcnt;\n parentbbox = paths[parentcnt].boundingbox;\n }\n }\n paths[parentidx].holechildren.push(pacnt);\n }\n pacnt++;\n }\n }\n pcnt++;\n }\n }\n }\n }\n return paths;\n}\nfunction batchPathScan(layers, pathomit) {\n const bpaths = [];\n for (let k = 0; k < layers.length; k++)\n bpaths[k] = pathScan(layers[k], pathomit);\n return bpaths;\n}\nfunction tracePath(path, ltres, qtres) {\n let pcnt = 0;\n let segtype1;\n let segtype2;\n let seqend;\n const smp = {\n segments: [],\n boundingbox: path.boundingbox,\n holechildren: path.holechildren,\n isholepath: path.isholepath\n };\n while (pcnt < path.points.length) {\n segtype1 = path.points[pcnt].linesegment;\n segtype2 = -1;\n seqend = pcnt + 1;\n while ((path.points[seqend].linesegment === segtype1 || path.points[seqend].linesegment === segtype2 || segtype2 === -1) && seqend < path.points.length - 1) {\n if (path.points[seqend].linesegment !== segtype1 && segtype2 === -1)\n segtype2 = path.points[seqend].linesegment;\n seqend++;\n }\n if (seqend === path.points.length - 1)\n seqend = 0;\n smp.segments = smp.segments.concat(fitSeq(path, ltres, qtres, pcnt, seqend));\n if (seqend > 0)\n pcnt = seqend;\n else\n pcnt = path.points.length;\n }\n return smp;\n}\nfunction batchTracePaths(internodepaths, ltres, qtres) {\n const btracedpaths = [];\n for (let k = 0; k < internodepaths.length; k++)\n btracedpaths.push(tracePath(internodepaths[k], ltres, qtres));\n return btracedpaths;\n}\n\nfunction batchInterNodes(bpaths, options) {\n const binternodes = [];\n for (let k = 0; k < bpaths.length; k++)\n binternodes[k] = interNodes(bpaths[k], options);\n return binternodes;\n}\nfunction interNodes(paths, options) {\n const ins = [];\n let palen = 0;\n let nextidx = 0;\n let nextidx2 = 0;\n let previdx = 0;\n let previdx2 = 0;\n let pacnt;\n let pcnt;\n for (pacnt = 0; pacnt < paths.length; pacnt++) {\n ins[pacnt] = {};\n ins[pacnt].points = [];\n ins[pacnt].boundingbox = paths[pacnt].boundingbox;\n ins[pacnt].holechildren = paths[pacnt].holechildren;\n ins[pacnt].isholepath = paths[pacnt].isholepath;\n palen = paths[pacnt].points.length;\n for (pcnt = 0; pcnt < palen; pcnt++) {\n nextidx = (pcnt + 1) % palen;\n nextidx2 = (pcnt + 2) % palen;\n previdx = (pcnt - 1 + palen) % palen;\n previdx2 = (pcnt - 2 + palen) % palen;\n if (options.rightangleenhance && testRightAngle(paths[pacnt], previdx2, previdx, pcnt, nextidx, nextidx2)) {\n if (ins[pacnt].points.length > 0) {\n ins[pacnt].points[ins[pacnt].points.length - 1].linesegment = getDirection(ins[pacnt].points[ins[pacnt].points.length - 1].x, ins[pacnt].points[ins[pacnt].points.length - 1].y, paths[pacnt].points[pcnt].x, paths[pacnt].points[pcnt].y);\n }\n ins[pacnt].points.push({\n x: paths[pacnt].points[pcnt].x,\n y: paths[pacnt].points[pcnt].y,\n linesegment: getDirection(paths[pacnt].points[pcnt].x, paths[pacnt].points[pcnt].y, (paths[pacnt].points[pcnt].x + paths[pacnt].points[nextidx].x) / 2, (paths[pacnt].points[pcnt].y + paths[pacnt].points[nextidx].y) / 2)\n });\n }\n ins[pacnt].points.push({\n x: (paths[pacnt].points[pcnt].x + paths[pacnt].points[nextidx].x) / 2,\n y: (paths[pacnt].points[pcnt].y + paths[pacnt].points[nextidx].y) / 2,\n linesegment: getDirection((paths[pacnt].points[pcnt].x + paths[pacnt].points[nextidx].x) / 2, (paths[pacnt].points[pcnt].y + paths[pacnt].points[nextidx].y) / 2, (paths[pacnt].points[nextidx].x + paths[pacnt].points[nextidx2].x) / 2, (paths[pacnt].points[nextidx].y + paths[pacnt].points[nextidx2].y) / 2)\n });\n }\n }\n return ins;\n}\n\nfunction batchTraceLayers(binternodes, ltres, qtres) {\n const btbis = [];\n for (let k = 0; k < binternodes.length; k++)\n btbis[k] = batchTracePaths(binternodes[k], ltres, qtres);\n return btbis;\n}\nfunction layering(ii) {\n const layers = [];\n let val = 0;\n const ah = ii.array.length;\n const aw = ii.array[0].length;\n let n1;\n let n2;\n let n3;\n let n4;\n let n5;\n let n6;\n let n7;\n let n8;\n let i;\n let j;\n let k;\n for (k = 0; k < ii.palette.length; k++) {\n layers[k] = [];\n for (j = 0; j < ah; j++) {\n layers[k][j] = [];\n for (i = 0; i < aw; i++)\n layers[k][j][i] = 0;\n }\n }\n for (j = 1; j < ah - 1; j++) {\n for (i = 1; i < aw - 1; i++) {\n val = ii.array[j][i];\n n1 = ii.array[j - 1][i - 1] === val ? 1 : 0;\n n2 = ii.array[j - 1][i] === val ? 1 : 0;\n n3 = ii.array[j - 1][i + 1] === val ? 1 : 0;\n n4 = ii.array[j][i - 1] === val ? 1 : 0;\n n5 = ii.array[j][i + 1] === val ? 1 : 0;\n n6 = ii.array[j + 1][i - 1] === val ? 1 : 0;\n n7 = ii.array[j + 1][i] === val ? 1 : 0;\n n8 = ii.array[j + 1][i + 1] === val ? 1 : 0;\n layers[val][j + 1][i + 1] = 1 + n5 * 2 + n8 * 4 + n7 * 8;\n if (!n4)\n layers[val][j + 1][i] = 0 + 2 + n7 * 4 + n6 * 8;\n if (!n2)\n layers[val][j][i + 1] = 0 + n3 * 2 + n5 * 4 + 8;\n if (!n1)\n layers[val][j][i] = 0 + n2 * 2 + 4 + n4 * 8;\n }\n }\n return layers;\n}\nfunction layeringStep(ii, cnum) {\n const layer = [];\n const ah = ii.array.length;\n const aw = ii.array[0].length;\n let i;\n let j;\n for (j = 0; j < ah; j++) {\n layer[j] = [];\n for (i = 0; i < aw; i++)\n layer[j][i] = 0;\n }\n for (j = 1; j < ah; j++) {\n for (i = 1; i < aw; i++) {\n layer[j][i] = (ii.array[j - 1][i - 1] === cnum ? 1 : 0) + (ii.array[j - 1][i] === cnum ? 2 : 0) + (ii.array[j][i - 1] === cnum ? 8 : 0) + (ii.array[j][i] === cnum ? 4 : 0);\n }\n }\n return layer;\n}\nfunction drawLayers(layers, palette, scale = 1, parentid) {\n let w, h, i, j;\n let div;\n if (parentid) {\n div = document.getElementById(parentid);\n if (!div) {\n div = document.createElement(\"div\");\n div.id = parentid;\n document.body.appendChild(div);\n }\n } else {\n div = document.createElement(\"div\");\n document.body.appendChild(div);\n }\n for (const layer of layers) {\n w = layer[0].length;\n h = layer.length;\n const canvas = document.createElement(\"canvas\");\n canvas.width = w * scale;\n canvas.height = h * scale;\n const context = canvas.getContext(\"2d\");\n for (j = 0; j < h; j++) {\n for (i = 0; i < w; i++) {\n context.fillStyle = toRgbaStr(palette[layer[j][i] % palette.length]);\n context.fillRect(i * scale, j * scale, scale, scale);\n }\n }\n div.appendChild(canvas);\n }\n}\n\nclass ImageTracer {\n constructor() {\n this.versionnumber = VERSION_NUMBER;\n this.optionpresets = OPTION_PRESETS;\n }\n checkOptions(options) {\n if (options === void 0)\n return { ...this.optionpresets.default };\n if (typeof options === \"string\") {\n const presetName = options;\n const preset = this.optionpresets[presetName];\n return { ...this.optionpresets.default, ...preset || {} };\n }\n if (typeof options === \"object\") {\n return { ...this.optionpresets.default, ...options };\n }\n throw new Error(`Unknown Option Type: ${typeof options}`);\n }\n async imageToSVG(url, options) {\n options = this.checkOptions(options);\n const canvas = await this.loadImage(url, options);\n return this.imageDataToSVG(this.getImgdata(canvas), options);\n }\n getImgdata(canvas) {\n const context = canvas.getContext(\"2d\");\n return context.getImageData(0, 0, canvas.width, canvas.height);\n }\n loadImage(url, options) {\n return new Promise((resolve) => {\n const img = new Image();\n if (options && options.corsenabled)\n img.crossOrigin = \"Anonymous\";\n img.src = url;\n img.onload = function() {\n const canvas = document.createElement(\"canvas\");\n canvas.width = img.width;\n canvas.height = img.height;\n const context = canvas.getContext(\"2d\");\n context?.drawImage(img, 0, 0);\n resolve(canvas);\n };\n });\n }\n imageDataToSVG(imgd, options) {\n options = this.checkOptions(options);\n const td = this.imageDataToTracedata(imgd, options);\n return getSvgString(td, options);\n }\n async imageToTracedata(url, options) {\n options = this.checkOptions(options);\n const canvas = await this.loadImage(url, options);\n return this.imageDataToTracedata(this.getImgdata(canvas), options);\n }\n imageDataToTracedata(imgd, options) {\n options = this.checkOptions(options);\n let tracedata;\n const ii = colorQuantization(imgd, options);\n if (options.layering === 0) {\n tracedata = {\n layers: [],\n palette: ii.palette,\n width: ii.array[0].length - 2,\n height: ii.array.length - 2\n };\n for (let colornum = 0; colornum < ii.palette.length; colornum++) {\n const tracedlayer = batchTracePaths(interNodes(pathScan(layeringStep(ii, colornum), options.pathomit), options), options.ltres, options.qtres);\n tracedata.layers.push(tracedlayer);\n }\n } else {\n const ls = layering(ii);\n if (options.layercontainerid)\n drawLayers(ls, SPECPALETTE, options.scale, options.layercontainerid);\n const bps = batchPathScan(ls, options.pathomit);\n const bis = batchInterNodes(bps, options);\n tracedata = {\n layers: batchTraceLayers(bis, options.ltres, options.qtres),\n palette: ii.palette,\n width: imgd.width,\n height: imgd.height\n };\n }\n return tracedata;\n }\n appendSVGString(svgstr, parentid) {\n let div;\n if (parentid) {\n div = document.getElementById(parentid);\n if (!div) {\n div = document.createElement(\"div\");\n div.id = parentid;\n document.body.appendChild(div);\n }\n } else {\n div = document.createElement(\"div\");\n document.body.appendChild(div);\n }\n div.innerHTML += svgstr;\n }\n}\nconst imageTracer = new ImageTracer();\n\nvar Colorsampling = /* @__PURE__ */ ((Colorsampling2) => {\n Colorsampling2[Colorsampling2[\"DISABLED\"] = 0] = \"DISABLED\";\n Colorsampling2[Colorsampling2[\"RANDOM_SAMPLING\"] = 1] = \"RANDOM_SAMPLING\";\n Colorsampling2[Colorsampling2[\"DETERMINISTIC_SAMPLING\"] = 2] = \"DETERMINISTIC_SAMPLING\";\n return Colorsampling2;\n})(Colorsampling || {});\n\nexport { Colorsampling, GKS, ImageTracer, OPTION_PRESETS, PATHSCAN_COMBINED_LOOKUP, SPECPALETTE, VERSION_NUMBER, batchInterNodes, batchPathScan, batchTraceLayers, batchTracePaths, boundingBoxIncludes, colorQuantization, drawLayers, fitSeq, generatePalette, getDirection, getSvgString, imageTracer, interNodes, layering, layeringStep, pathScan, pointInPoly, roundToDec, samplePalette, samplePaletteByGrid, svgPathString, testRightAngle, toRgbaStr, toSvgColorStr, tracePath };\n","import { locate } from 'locate-character';\n\nconst validNameCharacters = /[a-zA-Z0-9:_-]/;\nconst whitespace = /[\\s\\t\\r\\n]/;\nconst quotemark = /['\"]/;\n\nfunction repeat(str, i) {\n\tlet result = '';\n\twhile (i--) result += str;\n\treturn result;\n}\n\nexport function parse(source) {\n\tlet header = '';\n\tlet stack = [];\n\n\tlet state = metadata;\n\tlet currentElement = null;\n\tlet root = null;\n\n\tfunction error(message) {\n\t\tconst { line, column } = locate(source, i);\n\t\tconst before = source.slice(0, i);\n\t\tconst beforeLine = /(^|\\n).*$/.exec(before)[0].replace(/\\t/g, ' ');\n\t\tconst after = source.slice(i);\n\t\tconst afterLine = /.*(\\n|$)/.exec(after)[0];\n\n\t\tconst snippet = `${beforeLine}${afterLine}\\n${repeat(' ', beforeLine.length)}^`;\n\n\t\tthrow new Error(\n\t\t\t`${message} (${line}:${column}). If this is valid SVG, it's probably a bug in svg-parser. Please raise an issue at https://github.com/Rich-Harris/svg-parser/issues – thanks!\\n\\n${snippet}`\n\t\t);\n\t}\n\n\tfunction metadata() {\n\t\twhile ((i < source.length && source[i] !== '<') || !validNameCharacters.test(source[i + 1])) {\n\t\t\theader += source[i++];\n\t\t}\n\n\t\treturn neutral();\n\t}\n\n\tfunction neutral() {\n\t\tlet text = '';\n\t\twhile (i < source.length && source[i] !== '<') text += source[i++];\n\n\t\tif (/\\S/.test(text)) {\n\t\t\tcurrentElement.children.push({ type: 'text', value: text });\n\t\t}\n\n\t\tif (source[i] === '<') {\n\t\t\treturn tag;\n\t\t}\n\n\t\treturn neutral;\n\t}\n\n\tfunction tag() {\n\t\tconst char = source[i];\n\n\t\tif (char === '?') return neutral; // ') {\n\t\t\terror('Expected >');\n\t\t}\n\n\t\tif (!selfClosing) {\n\t\t\tcurrentElement = element;\n\t\t\tstack.push(element);\n\t\t}\n\n\t\treturn neutral;\n\t}\n\n\tfunction comment() {\n\t\tconst index = source.indexOf('-->', i);\n\t\tif (!~index) error('expected -->');\n\n\t\ti = index + 2;\n\t\treturn neutral;\n\t}\n\n\tfunction cdata() {\n\t\tconst index = source.indexOf(']]>', i);\n\t\tif (!~index) error('expected ]]>');\n\n\t\tcurrentElement.children.push(source.slice(i + 7, index));\n\n\t\ti = index + 2;\n\t\treturn neutral;\n\t}\n\n\tfunction closingTag() {\n\t\tconst tagName = getName();\n\n\t\tif (!tagName) error('Expected tag name');\n\n\t\tif (tagName !== currentElement.tagName) {\n\t\t\terror(`Expected closing tag to match opening tag <${currentElement.tagName}>`);\n\t\t}\n\n\t\tallowSpaces();\n\n\t\tif (source[i] !== '>') {\n\t\t\terror('Expected >');\n\t\t}\n\n\t\tstack.pop();\n\t\tcurrentElement = stack[stack.length - 1];\n\n\t\treturn neutral;\n\t}\n\n\tfunction getName() {\n\t\tlet name = '';\n\t\twhile (i < source.length && validNameCharacters.test(source[i])) name += source[i++];\n\n\t\treturn name;\n\t}\n\n\tfunction getAttribute() {\n\t\tif (!whitespace.test(source[i])) return null;\n\t\tallowSpaces();\n\n\t\tconst name = getName();\n\t\tif (!name) return null;\n\n\t\tlet value = true;\n\n\t\tallowSpaces();\n\t\tif (source[i] === '=') {\n\t\t\ti += 1;\n\t\t\tallowSpaces();\n\n\t\t\tvalue = getAttributeValue();\n\t\t\tif (!isNaN(value) && value.trim() !== '') value = +value; // TODO whitelist numeric attributes?\n\t\t}\n\n\t\treturn { name, value };\n\t}\n\n\tfunction getAttributeValue() {\n\t\treturn quotemark.test(source[i]) ? getQuotedAttributeValue() : getUnquotedAttributeValue();\n\t}\n\n\tfunction getUnquotedAttributeValue() {\n\t\tlet value = '';\n\t\tdo {\n\t\t\tconst char = source[i];\n\t\t\tif (char === ' ' || char === '>' || char === '/') {\n\t\t\t\treturn value;\n\t\t\t}\n\n\t\t\tvalue += char;\n\t\t\ti += 1;\n\t\t} while (i < source.length);\n\n\t\treturn value;\n\t}\n\n\tfunction getQuotedAttributeValue() {\n\t\tconst quotemark = source[i++];\n\n\t\tlet value = '';\n\t\tlet escaped = false;\n\n\t\twhile (i < source.length) {\n\t\t\tconst char = source[i++];\n\t\t\tif (char === quotemark && !escaped) {\n\t\t\t\treturn value;\n\t\t\t}\n\n\t\t\tif (char === '\\\\' && !escaped) {\n\t\t\t\tescaped = true;\n\t\t\t}\n\n\t\t\tvalue += escaped ? `\\\\${char}` : char;\n\t\t\tescaped = false;\n\t\t}\n\t}\n\n\tfunction allowSpaces() {\n\t\twhile (i < source.length && whitespace.test(source[i])) i += 1;\n\t}\n\n\tlet i = metadata.length;\n\twhile (i < source.length) {\n\t\tif (!state) error('Unexpected character');\n\t\tstate = state();\n\t\ti += 1;\n\t}\n\n\tif (state !== neutral) {\n\t\terror('Unexpected end of input');\n\t}\n\n\tif (root.tagName === 'svg') root.metadata = header;\n\treturn {\n\t\ttype: 'root',\n\t\tchildren: [root]\n\t};\n}\n","function getLocator(source, options) {\n if (options === void 0) { options = {}; }\n var offsetLine = options.offsetLine || 0;\n var offsetColumn = options.offsetColumn || 0;\n var originalLines = source.split('\\n');\n var start = 0;\n var lineRanges = originalLines.map(function (line, i) {\n var end = start + line.length + 1;\n var range = { start: start, end: end, line: i };\n start = end;\n return range;\n });\n var i = 0;\n function rangeContains(range, index) {\n return range.start <= index && index < range.end;\n }\n function getLocation(range, index) {\n return { line: offsetLine + range.line, column: offsetColumn + index - range.start, character: index };\n }\n function locate(search, startIndex) {\n if (typeof search === 'string') {\n search = source.indexOf(search, startIndex || 0);\n }\n var range = lineRanges[i];\n var d = search >= range.end ? 1 : -1;\n while (range) {\n if (rangeContains(range, search))\n return getLocation(range, search);\n i += d;\n range = lineRanges[i];\n }\n }\n ;\n return locate;\n}\nfunction locate(source, search, options) {\n if (typeof options === 'number') {\n throw new Error('locate takes a { startIndex, offsetLine, offsetColumn } object as the third argument');\n }\n return getLocator(source, options)(search, options && options.startIndex);\n}\n\nexport { getLocator, locate };","import { eSketchPathType, KbVector, SketchNode, SketchPath, Token } from '@models/classes';\r\nimport { imageTracer, ImageTracerOptionsParamers } from 'imagetracer';\r\nimport { ElementNode, parse, RootNode } from 'svg-parser';\r\nimport { Layer, PathHandler } from './path-handler';\r\nimport { projectOnNodeAxis } from './util';\r\n\r\ninterface ImageTraceColor {\r\n r: number;\r\n g: number;\r\n b: number;\r\n a: number;\r\n}\r\n\r\nexport interface ImageTraceResult {\r\n layers: Layer[][];\r\n palette: ImageTraceColor[];\r\n width: number;\r\n height: number;\r\n}\r\n\r\nexport class SvgConverter {\r\n static overwriteSketchWithSvg(node: SketchNode, data: Blob) {\r\n const decover = new TextDecoder();\r\n node.paths.clear();\r\n\r\n const header = data.slice(0, 100);\r\n header.arrayBuffer().then(buffer => {\r\n const headerText = decover.decode(buffer);\r\n if (headerText.search(/JFIF/) !== -1 || headerText.search(/PNG/) !== -1) {\r\n // is raster\r\n const dataUrl = URL.createObjectURL(data);\r\n const options: ImageTracerOptionsParamers = {\r\n qtres: 0.1,\r\n pathomit: 16,\r\n numberofcolors: 2,\r\n };\r\n const nodeManager = node._manager;\r\n if (!nodeManager) {\r\n throw Error('no node manager');\r\n }\r\n\r\n imageTracer.imageToTracedata(dataUrl, options).then((trace: ImageTraceResult) => {\r\n if (trace.palette && trace.palette.length > 1) {\r\n const pathParser = new PathHandler(\r\n node,\r\n SvgConverter.calculateScaleDivider([0, 0, trace.width, trace.height]),\r\n nodeManager\r\n );\r\n\r\n const paletteStrengths = (trace.palette as ImageTraceColor[]).map((color, idx) => ({\r\n idx,\r\n strength: getPaletteColorStrength(color),\r\n }));\r\n paletteStrengths.sort((a, b) => a.strength - b.strength);\r\n const bgIdx = paletteStrengths[0].idx;\r\n trace.layers.splice(bgIdx, 1);\r\n const primaryLayers = trace.layers.flat();\r\n\r\n const pathsCreated: SketchPath[][] = [];\r\n\r\n // Convert layers to sketchPaths\r\n primaryLayers.forEach((layer, layerIndex) => {\r\n const newPaths = pathParser.parseTrace(layer.segments);\r\n const pathGroupKey = 'tracedPath' + layerIndex;\r\n pathsCreated[layerIndex] = newPaths;\r\n newPaths.forEach(path => {\r\n path.pathGroupKey = pathGroupKey;\r\n });\r\n node.paths.push(...newPaths);\r\n });\r\n\r\n // Convert layers to holes\r\n primaryLayers.forEach((layer, layerIndex) => {\r\n if (layer.holechildren) {\r\n layer.holechildren.forEach(holeLayerIdx => {\r\n const pathSet = pathsCreated[holeLayerIdx];\r\n if (pathSet) {\r\n pathSet.forEach(path => {\r\n path.holeForPathGroup = 'tracedPath' + layerIndex;\r\n });\r\n }\r\n });\r\n }\r\n });\r\n }\r\n });\r\n } else {\r\n data.text().then(svgText => {\r\n const parsedSVG = parse(svgText);\r\n SvgConverter.parseSvgRecursively(node, parsedSVG);\r\n });\r\n }\r\n });\r\n }\r\n\r\n static findSvgViewBox(svg: ElementNode | RootNode) {\r\n for (let i = 0; i < svg.children.length; i++) {\r\n const child = svg.children[i] as ElementNode;\r\n if (typeof child === 'string') {\r\n return undefined;\r\n } else if (child.tagName === 'svg' && 'properties' in child) {\r\n return getViewBoxFromSvgElement(child);\r\n }\r\n }\r\n\r\n function getViewBoxFromSvgElement(elem: ElementNode): number[] | undefined {\r\n if (typeof elem.properties?.viewBox === 'string' && elem.properties.viewBox) {\r\n return elem.properties.viewBox.split(' ').map(int => parseInt(int));\r\n }\r\n if (elem.properties?.width && elem.properties?.height) {\r\n const width = elem.properties.width;\r\n const height = elem.properties.height;\r\n return [\r\n 0,\r\n 0,\r\n typeof width === 'string' ? parseInt(width) : width,\r\n typeof height === 'string' ? parseInt(height) : height,\r\n ];\r\n }\r\n }\r\n }\r\n\r\n static parseSvgRecursively(node: SketchNode, svg: ElementNode | RootNode, viewBox?: number[]) {\r\n if (!viewBox) {\r\n viewBox = SvgConverter.findSvgViewBox(svg) || [0, 0, 100, 100];\r\n }\r\n if (svg.children) {\r\n const nodeManager = node._manager;\r\n if (!nodeManager) {\r\n throw Error('no node manager');\r\n }\r\n\r\n for (let i = 0; i < svg.children.length; i++) {\r\n const child = svg.children[i] as ElementNode;\r\n\r\n if (child.tagName === 'path' && child.properties) {\r\n const pathParser = new PathHandler(node, SvgConverter.calculateScaleDivider(viewBox), nodeManager);\r\n const newPaths = pathParser.parsePath(child.properties.d as string);\r\n node.paths.push(...newPaths);\r\n //call actual parser\r\n } else if (child.tagName === 'rect' && child.properties) {\r\n SvgConverter.createRectangle(node, child, viewBox);\r\n } else if (child.tagName === 'circle' && child.properties) {\r\n SvgConverter.createCircle(node, child, viewBox);\r\n } else if (child.tagName === 'polyline' && child.properties) {\r\n SvgConverter.createPolyLine(node, child, viewBox);\r\n } else if (child.tagName === 'line' && child.properties) {\r\n SvgConverter.createLine(node, child, viewBox);\r\n } else if (child.tagName === 'polygon' && child.properties) {\r\n SvgConverter.createPolyLine(node, child, viewBox, true);\r\n }\r\n SvgConverter.parseSvgRecursively(node, child, viewBox);\r\n nameGeneratedPaths(node);\r\n }\r\n }\r\n }\r\n\r\n static removeBackGroundOnConvertedSVG(children: [ElementNode]) {\r\n const parent = children[0];\r\n const newChildren = parent.children as [ElementNode];\r\n const resultChildren: ElementNode[] = [];\r\n newChildren.forEach(child => {\r\n if (child && child.properties) {\r\n if ('opacity' in child.properties) {\r\n if (child.properties.opacity == 1) {\r\n resultChildren.push(child);\r\n }\r\n }\r\n }\r\n });\r\n newChildren.clear();\r\n resultChildren.forEach(child => {\r\n newChildren.push(child);\r\n });\r\n // if (svg.children) {\r\n // console.log(svg.children.length);\r\n // for (let i = 0; i < svg.children.length; i++) {\r\n // const child = svg.children[i] as ElementNode;\r\n // if (child.properties) {\r\n // if (child.properties.opacity) {\r\n // if (Number(child.properties.opacity) < 1) {\r\n // console.log('Not a valid path');\r\n // svg.children.remove(child);\r\n // } else {\r\n // console.log('valid path:', child);\r\n // }\r\n // }\r\n // }\r\n // this.removeBackGroundOnConvertedSVG(child);\r\n // }\r\n // }\r\n }\r\n\r\n // static projectSketchOnNormal(node: SketchNode) {\r\n // node.paths.forEach(path => {\r\n // path.controlPoints.forEach(point => {\r\n // if (node.normal === eAxis.y) {\r\n // point.position = new KbVector(point.position.x, 0, point.position.y);\r\n // point.direction = new KbVector(point.direction.x, 0, point.direction.y);\r\n // if (point.direction2) {\r\n // point.direction2 = new KbVector(point.direction2.x, 0, point.direction2.y);\r\n // }\r\n // } else if (node.normal === eAxis.x) {\r\n // point.position = new KbVector(0, point.position.y, point.position.x);\r\n // point.direction = new KbVector(0, point.direction.y, point.direction.x);\r\n // if (point.direction2) {\r\n // point.direction2 = new KbVector(0, point.direction2.y, point.direction2.x);\r\n // }\r\n // }\r\n // });\r\n // });\r\n // }\r\n\r\n static calculateScaleDivider(viewbox: number[]) {\r\n if (viewbox) {\r\n const width = viewbox[2];\r\n const height = viewbox[3];\r\n if (width > height) {\r\n return width;\r\n } else {\r\n return height;\r\n }\r\n } else {\r\n return 1;\r\n }\r\n }\r\n\r\n static addNewPath(node: SketchNode, pathType?: eSketchPathType) {\r\n const nodeManager = node._manager;\r\n if (!nodeManager) {\r\n throw Error('no node manager');\r\n }\r\n const newPath = nodeManager.create(Token.SketchPath);\r\n if (newPath) {\r\n newPath.name = 'Path';\r\n newPath._expanded = true;\r\n if (pathType) {\r\n newPath.type = pathType;\r\n }\r\n node.paths.push(newPath);\r\n }\r\n return newPath;\r\n }\r\n\r\n static createRectangle(node: SketchNode, element: ElementNode, viewbox: number[]) {\r\n const nodeManager = node._manager;\r\n if (!nodeManager) {\r\n throw Error('no node manager');\r\n }\r\n const scaleDivider = SvgConverter.calculateScaleDivider(viewbox);\r\n if (!element.properties) {\r\n return;\r\n }\r\n const x = (element.properties.x as number) / scaleDivider;\r\n const y = ((element.properties.y as number) / scaleDivider) * -1;\r\n const rx = (element.properties.rx as number) / scaleDivider;\r\n const ry = ((element.properties.ry as number) / scaleDivider) * -1;\r\n const width = (element.properties.width as number) / scaleDivider;\r\n const height = ((element.properties.height as number) / scaleDivider) * -1;\r\n let newPath = SvgConverter.addNewPath(node, eSketchPathType.points);\r\n if (rx && ry) {\r\n const newControlPoint = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint.position = projectOnNodeAxis(node, x, y + ry);\r\n newPath.controlPoints.push(newControlPoint);\r\n const newControlPoint2 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint2.position = projectOnNodeAxis(node, x, y + height - ry);\r\n newPath.controlPoints.push(newControlPoint2);\r\n\r\n newPath = SvgConverter.addNewPath(node, eSketchPathType.quadraticBezier);\r\n const newControlPoint3 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint3.position = projectOnNodeAxis(node, x, y + height - ry);\r\n newControlPoint3.direction = projectOnNodeAxis(node, 0, ry);\r\n newPath.controlPoints.push(newControlPoint3);\r\n\r\n const newControlPoint4 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint4.position = projectOnNodeAxis(node, x + rx, y + height);\r\n newPath.controlPoints.push(newControlPoint4);\r\n\r\n newPath = SvgConverter.addNewPath(node, eSketchPathType.points);\r\n const newControlPoint5 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint5.position = projectOnNodeAxis(node, x + rx, y + height);\r\n newPath.controlPoints.push(newControlPoint5);\r\n\r\n const newControlPoint6 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint6.position = projectOnNodeAxis(node, width + x - rx, y + height);\r\n newPath.controlPoints.push(newControlPoint6);\r\n\r\n newPath = SvgConverter.addNewPath(node, eSketchPathType.quadraticBezier);\r\n const newControlPoint7 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint7.position = projectOnNodeAxis(node, width + x - rx, y + height);\r\n newControlPoint7.direction = projectOnNodeAxis(node, -ry, 0);\r\n newPath.controlPoints.push(newControlPoint7);\r\n\r\n const newControlPoint8 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint8.position = projectOnNodeAxis(node, width + x, y + height - ry);\r\n newPath.controlPoints.push(newControlPoint8);\r\n\r\n newPath = SvgConverter.addNewPath(node, eSketchPathType.points);\r\n const newControlPoint9 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint9.position = projectOnNodeAxis(node, width + x, y + height - ry);\r\n newPath.controlPoints.push(newControlPoint9);\r\n\r\n const newControlPoint10 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint10.position = projectOnNodeAxis(node, width + x, y + ry);\r\n newPath.controlPoints.push(newControlPoint10);\r\n\r\n newPath = SvgConverter.addNewPath(node, eSketchPathType.quadraticBezier);\r\n const newControlPoint11 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint11.position = projectOnNodeAxis(node, width + x, y + ry);\r\n newControlPoint11.direction = projectOnNodeAxis(node, 0, -ry);\r\n newPath.controlPoints.push(newControlPoint11);\r\n\r\n const newControlPoint12 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint12.position = projectOnNodeAxis(node, width + x - rx, y);\r\n newPath.controlPoints.push(newControlPoint12);\r\n\r\n newPath = SvgConverter.addNewPath(node, eSketchPathType.points);\r\n const newControlPoint13 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint13.position = projectOnNodeAxis(node, width + x - rx, y);\r\n newPath.controlPoints.push(newControlPoint13);\r\n\r\n const newControlPoint14 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint14.position = projectOnNodeAxis(node, x + rx, y);\r\n newPath.controlPoints.push(newControlPoint14);\r\n\r\n newPath = SvgConverter.addNewPath(node, eSketchPathType.quadraticBezier);\r\n const newControlPoint15 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint15.position = projectOnNodeAxis(node, x + rx, y);\r\n newControlPoint15.direction = projectOnNodeAxis(node, ry, 0);\r\n newPath.controlPoints.push(newControlPoint15);\r\n\r\n const newControlPoint16 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint16.position = projectOnNodeAxis(node, x, y + ry);\r\n newPath.controlPoints.push(newControlPoint16);\r\n } else {\r\n const newControlPoint = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint.position = projectOnNodeAxis(node, x, y);\r\n newPath.controlPoints.push(newControlPoint);\r\n\r\n const newControlPoint2 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint2.position = projectOnNodeAxis(node, x, y + height);\r\n newPath.controlPoints.push(newControlPoint2);\r\n\r\n const newControlPoint3 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint3.position = projectOnNodeAxis(node, x + width, y + height);\r\n newPath.controlPoints.push(newControlPoint3);\r\n\r\n const newControlPoint4 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint4.position = projectOnNodeAxis(node, x + width, y);\r\n newPath.controlPoints.push(newControlPoint4);\r\n\r\n const newControlPoint5 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint5.position = projectOnNodeAxis(node, x, y);\r\n newPath.controlPoints.push(newControlPoint5);\r\n }\r\n }\r\n\r\n static createCircle(node: SketchNode, element: ElementNode, viewbox: number[]) {\r\n const nodeManager = node._manager;\r\n if (!nodeManager) {\r\n throw Error('no node manager');\r\n }\r\n const scaleDivider = SvgConverter.calculateScaleDivider(viewbox);\r\n if (element.properties) {\r\n const cx = (element.properties.cx as number) / scaleDivider;\r\n const cy = ((element.properties.cy as number) / scaleDivider) * -1;\r\n const radius = (element.properties.r as number) / scaleDivider;\r\n\r\n const newPath = SvgConverter.addNewPath(node, eSketchPathType.centerRadiusCircle);\r\n const newControlPoint = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint.position = projectOnNodeAxis(node, cx, cy);\r\n newControlPoint.direction = new KbVector(0, 0, 1);\r\n newPath.controlPoints.push(newControlPoint);\r\n\r\n const newControlPoint2 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint2.position = projectOnNodeAxis(node, cx + radius, cy);\r\n newPath.controlPoints.push(newControlPoint2);\r\n }\r\n }\r\n\r\n static createLine(node: SketchNode, element: ElementNode, viewbox: number[]) {\r\n const nodeManager = node._manager;\r\n if (!nodeManager) {\r\n throw Error('no node manager');\r\n }\r\n const scaleDivider = SvgConverter.calculateScaleDivider(viewbox);\r\n if (element.properties) {\r\n const newPath = SvgConverter.addNewPath(node, eSketchPathType.points);\r\n const x1 = (element.properties.x1 as number) / scaleDivider;\r\n const y1 = ((element.properties.y1 as number) / scaleDivider) * -1;\r\n const x2 = (element.properties.x2 as number) / scaleDivider;\r\n const y2 = ((element.properties.y2 as number) / scaleDivider) * -1;\r\n\r\n const newControlPoint = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint.position = projectOnNodeAxis(node, x1, y1);\r\n newPath.controlPoints.push(newControlPoint);\r\n\r\n const newControlPoint2 = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint2.position = projectOnNodeAxis(node, x2, y2);\r\n newPath.controlPoints.push(newControlPoint2);\r\n }\r\n }\r\n\r\n static createPolyLine(node: SketchNode, element: ElementNode, viewbox: number[], closePath = false) {\r\n const nodeManager = node._manager;\r\n if (!nodeManager) {\r\n throw Error('no node manager');\r\n }\r\n const scaleDivider = SvgConverter.calculateScaleDivider(viewbox);\r\n if (element.properties) {\r\n const newPath = SvgConverter.addNewPath(node, eSketchPathType.points);\r\n const points = (element.properties.points as string).replace(',', ' ').split(' ');\r\n for (let i = 0; i < points.length; i = i + 2) {\r\n const newControlPoint = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint.position = new KbVector(\r\n parseInt(points[i]) / scaleDivider,\r\n (parseInt(points[i + 1]) / scaleDivider) * -1,\r\n 0\r\n );\r\n newPath.controlPoints.push(newControlPoint);\r\n }\r\n if (closePath) {\r\n const newControlPoint = nodeManager.create(Token.SketchControlPoint);\r\n newControlPoint.position = new KbVector(\r\n parseInt(points[0]) / scaleDivider,\r\n (parseInt(points[1]) / scaleDivider) * -1,\r\n 0\r\n );\r\n newPath.controlPoints.push(newControlPoint);\r\n }\r\n }\r\n }\r\n}\r\n\r\nfunction nameGeneratedPaths(node: SketchNode) {\r\n let counter = 0;\r\n node.paths.forEach(path => {\r\n if (counter > 0) {\r\n path.name = 'Path ' + counter.toString();\r\n }\r\n counter++;\r\n });\r\n}\r\n\r\nfunction getPaletteColorStrength(color: ImageTraceColor) {\r\n const saturation = Math.max(color.r, color.g, color.b) - Math.min(color.r, color.g, color.b);\r\n return color.a * 255 + saturation;\r\n}\r\n","import { VertexBuffer } from '@babylonjs/core/Buffers/buffer';\r\nimport { Color4, TmpVectors, Vector2, Vector3, Vector4 } from '@babylonjs/core/Maths';\r\nimport { _CreationDataStorage, Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { VertexData } from '@babylonjs/core/Meshes/mesh.vertexData';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { FloatArray, Nullable } from '@babylonjs/core/types';\r\nimport { earcut } from '@view/helpers/earcut';\r\nimport { calculateFaceNormal } from './sketching';\r\n\r\nclass CompatibilityOptions {\r\n static useOpenGLOrientationForUV: boolean;\r\n}\r\n/**\r\n * Creates the VertexData for a Ribbon\r\n * @param options an object used to set the following optional parameters for the ribbon, required but can be empty\r\n * * pathArray array of paths, each of which an array of successive Vector3\r\n * * closeArray creates a seam between the first and the last paths of the pathArray, optional, default false\r\n * * closePath creates a seam between the first and the last points of each path of the path array, optional, default false\r\n * * offset a positive integer, only used when pathArray contains a single path (offset = 10 means the point 1 is joined to the point 11), default rounded half size of the pathArray length\r\n * * sideOrientation optional and takes the values : Mesh.FRONTSIDE (default), Mesh.BACKSIDE or Mesh.DOUBLESIDE\r\n * * frontUvs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the front side, optional, default vector4 (0, 0, 1, 1)\r\n * * backUVs only usable when you create a double-sided mesh, used to choose what parts of the texture image to crop and apply on the back side, optional, default vector4 (0, 0, 1, 1)\r\n * * invertUV swaps in the U and V coordinates when applying a texture, optional, default false\r\n * * uvs a linear array, of length 2 * number of vertices, of custom UV values, optional\r\n * * colors a linear array, of length 4 * number of vertices, of custom color values, optional\r\n * @returns the VertexData of the ribbon\r\n */\r\n\r\nexport function CreateRibbonVertexData(options: {\r\n pathArray: Vector3[][];\r\n closeArray?: boolean;\r\n closePath?: boolean;\r\n offset?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n invertUV?: boolean;\r\n uvs?: Vector2[];\r\n colors?: Color4[];\r\n}): VertexData {\r\n let pathArray: Vector3[][] = options.pathArray;\r\n const closeArray: boolean = options.closeArray || false;\r\n const closePath: boolean = options.closePath || false;\r\n const invertUV: boolean = options.invertUV || false;\r\n const defaultOffset: number = Math.floor(pathArray[0].length / 2);\r\n let offset: number = options.offset || defaultOffset;\r\n offset = offset > defaultOffset ? defaultOffset : Math.floor(offset); // offset max allowed : defaultOffset\r\n const sideOrientation: number =\r\n options.sideOrientation === 0 ? 0 : options.sideOrientation || VertexData.DEFAULTSIDE;\r\n const customUV = options.uvs;\r\n const customColors = options.colors;\r\n\r\n const positions: number[] = [];\r\n const indices: number[] = [];\r\n const normals: number[] = [];\r\n const uvs: number[] = [];\r\n\r\n const us: number[][] = []; // us[path_id] = [uDist1, uDist2, uDist3 ... ] distances between points on path path_id\r\n const vs: number[][] = []; // vs[i] = [vDist1, vDist2, vDist3, ... ] distances between points i of consecutive paths from pathArray\r\n const uTotalDistance: number[] = []; // uTotalDistance[p] : total distance of path p\r\n const vTotalDistance: number[] = []; // vTotalDistance[i] : total distance between points i of first and last path from pathArray\r\n let minlg: number; // minimal length among all paths from pathArray\r\n const lg: number[] = []; // array of path lengths : nb of vertex per path\r\n const idx: number[] = []; // array of path indexes : index of each path (first vertex) in the total vertex number\r\n let p: number; // path iterator\r\n let i: number; // point iterator\r\n let j: number; // point iterator\r\n\r\n // if single path in pathArray\r\n if (pathArray.length < 2) {\r\n const ar1: Vector3[] = [];\r\n const ar2: Vector3[] = [];\r\n for (i = 0; i < pathArray[0].length - offset; i++) {\r\n ar1.push(pathArray[0][i]);\r\n ar2.push(pathArray[0][i + offset]);\r\n }\r\n pathArray = [ar1, ar2];\r\n }\r\n\r\n // positions and horizontal distances (u)\r\n let idc: number = 0;\r\n const closePathCorr: number = closePath ? 1 : 0; // the final index will be +1 if closePath\r\n const closeArrayCorr: number = closeArray ? 1 : 0;\r\n let path: Vector3[];\r\n let l: number;\r\n minlg = pathArray[0].length;\r\n let vectlg: number;\r\n let dist: number;\r\n for (p = 0; p < pathArray.length + closeArrayCorr; p++) {\r\n uTotalDistance[p] = 0;\r\n us[p] = [0];\r\n path = p === pathArray.length ? pathArray[0] : pathArray[p];\r\n l = path.length;\r\n minlg = minlg < l ? minlg : l;\r\n\r\n j = 0;\r\n while (j < l) {\r\n positions.push(path[j].x, path[j].y, path[j].z);\r\n if (j > 0) {\r\n vectlg = path[j].subtract(path[j - 1]).length();\r\n dist = vectlg + uTotalDistance[p];\r\n us[p].push(dist);\r\n uTotalDistance[p] = dist;\r\n }\r\n j++;\r\n }\r\n\r\n if (closePath) {\r\n // an extra hidden vertex is added in the \"positions\" array\r\n j--;\r\n positions.push(path[0].x, path[0].y, path[0].z);\r\n vectlg = path[j].subtract(path[0]).length();\r\n dist = vectlg + uTotalDistance[p];\r\n us[p].push(dist);\r\n uTotalDistance[p] = dist;\r\n }\r\n\r\n lg[p] = l + closePathCorr;\r\n idx[p] = idc;\r\n idc += l + closePathCorr;\r\n }\r\n\r\n // vertical distances (v)\r\n let path1: Vector3[];\r\n let path2: Vector3[];\r\n let vertex1: Nullable = null;\r\n let vertex2: Nullable = null;\r\n for (i = 0; i < minlg + closePathCorr; i++) {\r\n vTotalDistance[i] = 0;\r\n vs[i] = [0];\r\n for (p = 0; p < pathArray.length - 1 + closeArrayCorr; p++) {\r\n path1 = pathArray[p];\r\n path2 = p === pathArray.length - 1 ? pathArray[0] : pathArray[p + 1];\r\n if (i === minlg) {\r\n // closePath\r\n vertex1 = path1[0];\r\n vertex2 = path2[0];\r\n } else {\r\n vertex1 = path1[i];\r\n vertex2 = path2[i];\r\n }\r\n vectlg = vertex2.subtract(vertex1).length();\r\n dist = vectlg + vTotalDistance[i];\r\n vs[i].push(dist);\r\n vTotalDistance[i] = dist;\r\n }\r\n }\r\n\r\n // uvs\r\n let u: number;\r\n let v: number;\r\n if (customUV) {\r\n for (p = 0; p < customUV.length; p++) {\r\n uvs.push(\r\n customUV[p].x,\r\n CompatibilityOptions.useOpenGLOrientationForUV ? 1.0 - customUV[p].y : customUV[p].y\r\n );\r\n }\r\n } else {\r\n for (p = 0; p < pathArray.length + closeArrayCorr; p++) {\r\n for (i = 0; i < minlg + closePathCorr; i++) {\r\n u = uTotalDistance[p] != 0.0 ? us[p][i] / uTotalDistance[p] : 0.0;\r\n v = vTotalDistance[i] != 0.0 ? vs[i][p] / vTotalDistance[i] : 0.0;\r\n if (invertUV) {\r\n uvs.push(v, u);\r\n } else {\r\n uvs.push(u, CompatibilityOptions.useOpenGLOrientationForUV ? 1.0 - v : v);\r\n }\r\n }\r\n }\r\n }\r\n\r\n // indices\r\n p = 0; // path index\r\n let pi: number = 0; // positions array index\r\n let l1: number = lg[p] - 1; // path1 length\r\n let l2: number = lg[p + 1] - 1; // path2 length\r\n let min: number = l1 < l2 ? l1 : l2; // current path stop index\r\n let shft: number = idx[1] - idx[0]; // shift\r\n const path1nb: number = lg.length - 1; // number of path1 to iterate on\r\n\r\n while (pi <= min && p < path1nb) {\r\n // stay under min and don't go over next to last path\r\n // draw two triangles between path1 (p1) and path2 (p2) : (p1.pi, p2.pi, p1.pi+1) and (p2.pi+1, p1.pi+1, p2.pi) clockwise\r\n\r\n indices.push(pi, pi + shft, pi + 1);\r\n indices.push(pi + shft + 1, pi + 1, pi + shft);\r\n pi += 1;\r\n if (pi === min) {\r\n // if end of one of two consecutive paths reached, go to next existing path\r\n p++;\r\n shft = idx[p + 1] - idx[p];\r\n l1 = lg[p] - 1;\r\n l2 = lg[p + 1] - 1;\r\n pi = idx[p];\r\n min = l1 < l2 ? l1 + pi : l2 + pi;\r\n }\r\n }\r\n\r\n // normals\r\n VertexData.ComputeNormals(positions, indices, normals);\r\n\r\n if (closePath) {\r\n // update both the first and last vertex normals to their average value\r\n let indexFirst: number = 0;\r\n let indexLast: number = 0;\r\n for (p = 0; p < pathArray.length; p++) {\r\n indexFirst = idx[p] * 3;\r\n if (p + 1 < pathArray.length) {\r\n indexLast = (idx[p + 1] - 1) * 3;\r\n } else {\r\n indexLast = normals.length - 3;\r\n }\r\n normals[indexFirst] = (normals[indexFirst] + normals[indexLast]) * 0.5;\r\n normals[indexFirst + 1] = (normals[indexFirst + 1] + normals[indexLast + 1]) * 0.5;\r\n normals[indexFirst + 2] = (normals[indexFirst + 2] + normals[indexLast + 2]) * 0.5;\r\n const l = Math.sqrt(\r\n normals[indexFirst] * normals[indexFirst] +\r\n normals[indexFirst + 1] * normals[indexFirst + 1] +\r\n normals[indexFirst + 2] * normals[indexFirst + 2]\r\n );\r\n normals[indexFirst] /= l;\r\n normals[indexFirst + 1] /= l;\r\n normals[indexFirst + 2] /= l;\r\n normals[indexLast] = normals[indexFirst];\r\n normals[indexLast + 1] = normals[indexFirst + 1];\r\n normals[indexLast + 2] = normals[indexFirst + 2];\r\n }\r\n }\r\n\r\n if (closeArray) {\r\n let indexFirst = idx[0] * 3;\r\n let indexLast = idx[pathArray.length] * 3;\r\n for (i = 0; i < minlg + closePathCorr; i++) {\r\n normals[indexFirst] = (normals[indexFirst] + normals[indexLast]) * 0.5;\r\n normals[indexFirst + 1] = (normals[indexFirst + 1] + normals[indexLast + 1]) * 0.5;\r\n normals[indexFirst + 2] = (normals[indexFirst + 2] + normals[indexLast + 2]) * 0.5;\r\n const l = Math.sqrt(\r\n normals[indexFirst] * normals[indexFirst] +\r\n normals[indexFirst + 1] * normals[indexFirst + 1] +\r\n normals[indexFirst + 2] * normals[indexFirst + 2]\r\n );\r\n normals[indexFirst] /= l;\r\n normals[indexFirst + 1] /= l;\r\n normals[indexFirst + 2] /= l;\r\n normals[indexLast] = normals[indexFirst];\r\n normals[indexLast + 1] = normals[indexFirst + 1];\r\n normals[indexLast + 2] = normals[indexFirst + 2];\r\n indexFirst += 3;\r\n indexLast += 3;\r\n }\r\n }\r\n\r\n // sides\r\n VertexData._ComputeSides(sideOrientation, positions, indices, normals, uvs, options.frontUVs, options.backUVs);\r\n\r\n // Colors\r\n let colors: Nullable = null;\r\n if (customColors) {\r\n colors = new Float32Array(customColors.length * 4);\r\n for (let c = 0; c < customColors.length; c++) {\r\n colors[c * 4] = customColors[c].r;\r\n colors[c * 4 + 1] = customColors[c].g;\r\n colors[c * 4 + 2] = customColors[c].b;\r\n colors[c * 4 + 3] = customColors[c].a;\r\n }\r\n }\r\n\r\n // Result\r\n const vertexData = new VertexData();\r\n const positions32 = new Float32Array(positions);\r\n const normals32 = new Float32Array(normals);\r\n const uvs32 = new Float32Array(uvs);\r\n\r\n vertexData.indices = indices;\r\n vertexData.positions = positions32;\r\n vertexData.normals = normals32;\r\n vertexData.uvs = uvs32;\r\n if (colors) {\r\n vertexData.set(colors, VertexBuffer.ColorKind);\r\n }\r\n\r\n if (closePath) {\r\n (vertexData)._idx = idx;\r\n }\r\n\r\n return vertexData;\r\n}\r\n\r\n/**\r\n * Creates a ribbon mesh. The ribbon is a parametric shape. It has no predefined shape. Its final shape will depend on the input parameters\r\n * * The parameter `pathArray` is a required array of paths, what are each an array of successive Vector3. The pathArray parameter depicts the ribbon geometry\r\n * * The parameter `closeArray` (boolean, default false) creates a seam between the first and the last paths of the path array\r\n * * The parameter `closePath` (boolean, default false) creates a seam between the first and the last points of each path of the path array\r\n * * The parameter `offset` (positive integer, default : rounded half size of the pathArray length), is taken in account only if the `pathArray` is containing a single path\r\n * * It's the offset to join the points from the same path. Ex : offset = 10 means the point 1 is joined to the point 11\r\n * * The optional parameter `instance` is an instance of an existing Ribbon object to be updated with the passed `pathArray` parameter : https://doc.babylonjs.com/features/featuresDeepDive/mesh/dynamicMeshMorph#ribbon\r\n * * You can also set the mesh side orientation with the values : BABYLON.Mesh.FRONTSIDE (default), BABYLON.Mesh.BACKSIDE or BABYLON.Mesh.DOUBLESIDE\r\n * * If you create a double-sided mesh, you can choose what parts of the texture image to crop and stick respectively on the front and the back sides with the parameters `frontUVs` and `backUVs` (Vector4). Detail here : https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/set#side-orientation\r\n * * The optional parameter `invertUV` (boolean, default false) swaps in the geometry the U and V coordinates to apply a texture\r\n * * The parameter `uvs` is an optional flat array of `Vector2` to update/set each ribbon vertex with its own custom UV values instead of the computed ones\r\n * * The parameters `colors` is an optional flat array of `Color4` to set/update each ribbon vertex with its own custom color values\r\n * * Note that if you use the parameters `uvs` or `colors`, the passed arrays must be populated with the right number of elements, it is to say the number of ribbon vertices. Remember that if you set `closePath` to `true`, there's one extra vertex per path in the geometry\r\n * * Moreover, you can use the parameter `color` with `instance` (to update the ribbon), only if you previously used it at creation time\r\n * * The mesh can be set to updatable with the boolean parameter `updatable` (default false) if its internal geometry is supposed to change once created\r\n * @param name defines the name of the mesh\r\n * @param options defines the options used to create the mesh\r\n * @param scene defines the hosting scene\r\n * @returns the ribbon mesh\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param/ribbon_extra\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/mesh/creation/param\r\n */\r\nexport function CreateMiteredRibbon(\r\n name: string,\r\n options: {\r\n pathArray: Vector3[][];\r\n closeArray?: boolean;\r\n closePath?: boolean;\r\n offset?: number;\r\n updatable?: boolean;\r\n cap?: number;\r\n sideOrientation?: number;\r\n frontUVs?: Vector4;\r\n backUVs?: Vector4;\r\n instance?: Mesh;\r\n invertUV?: boolean;\r\n uvs?: Vector2[];\r\n colors?: Color4[];\r\n },\r\n scene: Nullable = null\r\n): Mesh {\r\n const pathArray = options.pathArray;\r\n const closeArray = options.closeArray;\r\n const closePath = options.closePath;\r\n const cap = options.cap === 0 ? 0 : options.cap || Mesh.NO_CAP;\r\n const sideOrientation = Mesh._GetDefaultSideOrientation(options.sideOrientation);\r\n const instance = options.instance;\r\n const updatable = options.updatable;\r\n\r\n if (instance) {\r\n // existing ribbon instance update\r\n // positionFunction : ribbon case\r\n // only pathArray and sideOrientation parameters are taken into account for positions update\r\n const minimum = TmpVectors.Vector3[0].setAll(Number.MAX_VALUE);\r\n const maximum = TmpVectors.Vector3[1].setAll(-Number.MAX_VALUE);\r\n const positionFunction = (positions: FloatArray) => {\r\n let minlg = pathArray[0].length;\r\n const mesh = instance;\r\n let i = 0;\r\n const ns = mesh._originalBuilderSideOrientation === Mesh.DOUBLESIDE ? 2 : 1;\r\n for (let si = 1; si <= ns; ++si) {\r\n for (let p = 0; p < pathArray.length; ++p) {\r\n const path = pathArray[p];\r\n const l = path.length;\r\n minlg = minlg < l ? minlg : l;\r\n for (let j = 0; j < minlg; ++j) {\r\n const pathPoint = path[j];\r\n positions[i] = pathPoint.x;\r\n positions[i + 1] = pathPoint.y;\r\n positions[i + 2] = pathPoint.z;\r\n minimum.minimizeInPlaceFromFloats(pathPoint.x, pathPoint.y, pathPoint.z);\r\n maximum.maximizeInPlaceFromFloats(pathPoint.x, pathPoint.y, pathPoint.z);\r\n i += 3;\r\n }\r\n if (mesh._creationDataStorage && mesh._creationDataStorage.closePath) {\r\n const pathPoint = path[0];\r\n positions[i] = pathPoint.x;\r\n positions[i + 1] = pathPoint.y;\r\n positions[i + 2] = pathPoint.z;\r\n i += 3;\r\n }\r\n }\r\n }\r\n };\r\n const positions = instance.getVerticesData(VertexBuffer.PositionKind);\r\n positionFunction(positions);\r\n if (instance.hasBoundingInfo) {\r\n instance.getBoundingInfo().reConstruct(minimum, maximum, instance._worldMatrix);\r\n } else {\r\n instance.buildBoundingInfo(minimum, maximum, instance._worldMatrix);\r\n }\r\n instance.updateVerticesData(VertexBuffer.PositionKind, positions, false, false);\r\n if (options.colors) {\r\n const colors = instance.getVerticesData(VertexBuffer.ColorKind);\r\n for (let c = 0, colorIndex = 0; c < options.colors.length; c++, colorIndex += 4) {\r\n const color = options.colors[c];\r\n colors[colorIndex] = color.r;\r\n colors[colorIndex + 1] = color.g;\r\n colors[colorIndex + 2] = color.b;\r\n colors[colorIndex + 3] = color.a;\r\n }\r\n instance.updateVerticesData(VertexBuffer.ColorKind, colors, false, false);\r\n }\r\n if (options.uvs) {\r\n const uvs = instance.getVerticesData(VertexBuffer.UVKind);\r\n for (let i = 0; i < options.uvs.length; i++) {\r\n uvs[i * 2] = options.uvs[i].x;\r\n uvs[i * 2 + 1] = CompatibilityOptions.useOpenGLOrientationForUV\r\n ? 1.0 - options.uvs[i].y\r\n : options.uvs[i].y;\r\n }\r\n instance.updateVerticesData(VertexBuffer.UVKind, uvs, false, false);\r\n }\r\n if (!instance.areNormalsFrozen || instance.isFacetDataEnabled) {\r\n const indices = instance.getIndices();\r\n const normals = instance.getVerticesData(VertexBuffer.NormalKind);\r\n const params = instance.isFacetDataEnabled ? instance.getFacetDataParameters() : null;\r\n VertexData.ComputeNormals(positions, indices, normals, params);\r\n\r\n if (instance._creationDataStorage && instance._creationDataStorage.closePath) {\r\n let indexFirst: number = 0;\r\n let indexLast: number = 0;\r\n for (let p = 0; p < pathArray.length; p++) {\r\n indexFirst = instance._creationDataStorage!.idx[p] * 3;\r\n if (p + 1 < pathArray.length) {\r\n indexLast = (instance._creationDataStorage!.idx[p + 1] - 1) * 3;\r\n } else {\r\n indexLast = normals.length - 3;\r\n }\r\n normals[indexFirst] = (normals[indexFirst] + normals[indexLast]) * 0.5;\r\n normals[indexFirst + 1] = (normals[indexFirst + 1] + normals[indexLast + 1]) * 0.5;\r\n normals[indexFirst + 2] = (normals[indexFirst + 2] + normals[indexLast + 2]) * 0.5;\r\n normals[indexLast] = normals[indexFirst];\r\n normals[indexLast + 1] = normals[indexFirst + 1];\r\n normals[indexLast + 2] = normals[indexFirst + 2];\r\n }\r\n }\r\n if (!instance.areNormalsFrozen) {\r\n instance.updateVerticesData(VertexBuffer.NormalKind, normals, false, false);\r\n }\r\n }\r\n return instance;\r\n } else {\r\n // new ribbon creation\r\n\r\n const ribbon = new Mesh(name, scene);\r\n ribbon._originalBuilderSideOrientation = sideOrientation;\r\n ribbon._creationDataStorage = new _CreationDataStorage();\r\n\r\n const vertexData = CreateRibbonVertexData(options);\r\n if (closePath) {\r\n ribbon._creationDataStorage.idx = (vertexData)._idx;\r\n }\r\n ribbon._creationDataStorage.closePath = closePath;\r\n ribbon._creationDataStorage.closeArray = closeArray;\r\n\r\n vertexData.applyToMesh(ribbon, updatable);\r\n\r\n createLineCaps(pathArray, cap, name, ribbon, vertexData);\r\n return ribbon;\r\n }\r\n}\r\n/**\r\n * Class containing static functions to help procedurally build meshes\r\n * @deprecated use CreateRibbon directly\r\n */\r\nexport const RibbonBuilder = {\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n CreateMiteredRibbon,\r\n};\r\n\r\nfunction createLineCaps(pathArray: Vector3[][], cap: number, featureId: string, ribbon: Mesh, vertexData: VertexData) {\r\n const startCap: number[] = [];\r\n const endCap: number[] = [];\r\n let startCapVectors: Vector3[] = [];\r\n let endCapVectors: Vector3[] = [];\r\n const resultVectors: Vector3[][] = [];\r\n const resultCaps: number[][] = [];\r\n pathArray.forEach(path => {\r\n if (path[0]) {\r\n const vector = path[0];\r\n startCapVectors.push(path[0]);\r\n startCap.push(vector.x, vector.y, vector.z);\r\n }\r\n if (path[path.length - 1]) {\r\n const vector = path[path.length - 1];\r\n endCapVectors.push(path[path.length - 1]);\r\n endCap.push(vector.x, vector.y, vector.z);\r\n }\r\n });\r\n //check to see if the handedness of the endCap is correct. Otherwise, flip it\r\n if (startCapVectors.length > 0) {\r\n const faceNormal = calculateFaceNormal(startCapVectors);\r\n const firstPath = pathArray[0];\r\n if (firstPath.length >= 2) {\r\n const firstPoint = firstPath[0];\r\n const secondPoint = firstPath[1];\r\n\r\n const sketchNativeNormal = secondPoint.subtract(firstPoint);\r\n // If the rotation handedness is opposite the primary sketch vector, then flip the sketch path\r\n if (Vector3.Dot(faceNormal, sketchNativeNormal) < 0) {\r\n startCapVectors = reverseHandednessOfPositions(startCapVectors);\r\n }\r\n }\r\n }\r\n //check to see if the handedness of the endCap is correct. Otherwise, flip it\r\n if (endCapVectors.length > 0) {\r\n const faceNormal = calculateFaceNormal(endCapVectors);\r\n const lastPath = pathArray[pathArray.length - 1];\r\n if (lastPath.length >= 2) {\r\n const lastPoint = lastPath[lastPath.length - 1];\r\n const secondLastPoint = lastPath[lastPath.length - 2];\r\n\r\n const sketchNativeNormal = lastPoint.subtract(secondLastPoint);\r\n // If the rotation handedness is opposite the primary sketch vector, then flip the sketch path\r\n if (Vector3.Dot(faceNormal, sketchNativeNormal) < 0) {\r\n endCapVectors = reverseHandednessOfPositions(endCapVectors);\r\n }\r\n }\r\n }\r\n\r\n if (cap === Mesh.CAP_START) {\r\n const startCapIndices = earcut(startCap, [], 3);\r\n resultCaps.push(startCapIndices);\r\n resultVectors.push(startCapVectors, []);\r\n } else if (cap === Mesh.CAP_END) {\r\n const endCapIndices = earcut(endCap, [], 3);\r\n resultCaps.push(endCapIndices);\r\n\r\n resultVectors.push([], endCapVectors);\r\n } else if (cap === Mesh.CAP_ALL) {\r\n const startCapIndices = earcut(startCap, [], 3);\r\n resultCaps.push(startCapIndices);\r\n const endCapIndices = earcut(endCap, [], 3);\r\n\r\n //offset the endCapIndices otherwise they are the same as the startCapIndices\r\n for (let i = 0; i < endCapIndices.length; i++) {\r\n endCapIndices[i] = endCapIndices[i] + startCapVectors.length;\r\n }\r\n\r\n resultCaps.push(endCapIndices);\r\n resultVectors.push(startCapVectors, endCapVectors);\r\n }\r\n const newVertexData = combineVertexDataWithCaps(vertexData, resultCaps, resultVectors);\r\n newVertexData.applyToMesh(ribbon, true);\r\n return resultCaps;\r\n}\r\n\r\nfunction combineVertexDataWithCaps(\r\n baseVertexData: VertexData,\r\n capMeshIndices: number[][],\r\n capMeshPositions: Vector3[][]\r\n) {\r\n const totalBasePositions = baseVertexData.positions?.length || 0;\r\n const totalBaseNormals = baseVertexData.normals?.length || 0;\r\n const totalBaseUvs = baseVertexData.uvs?.length || 0;\r\n const totalBaseIndices = baseVertexData.indices?.length || 0;\r\n\r\n const newPositions: number[] = [];\r\n const newNormals: number[] = [];\r\n const newUvs: number[] = [];\r\n const newIndices: number[] = [];\r\n\r\n const startCapPositions = capMeshPositions[0];\r\n const endCapPositions = capMeshPositions[1];\r\n\r\n capMeshIndices.forEach(meshIndices => {\r\n meshIndices.forEach(index => {\r\n const newIndex = index + totalBasePositions / 3;\r\n newIndices.push(newIndex);\r\n });\r\n });\r\n\r\n if (startCapPositions && startCapPositions.length !== 0) {\r\n const normal = calculateFaceNormal(startCapPositions);\r\n startCapPositions.forEach(position => {\r\n newPositions.push(position.x, position.y, position.z);\r\n newNormals.push(normal.x, normal.y, normal.z);\r\n });\r\n const generatedUvs = generateUvsForCaps(startCapPositions);\r\n newUvs.push(...generatedUvs);\r\n }\r\n\r\n if (endCapPositions && endCapPositions.length !== 0) {\r\n const normal = calculateFaceNormal(endCapPositions);\r\n endCapPositions.forEach(position => {\r\n newPositions.push(position.x, position.y, position.z);\r\n newNormals.push(normal.x, normal.y, normal.z);\r\n });\r\n const generatedUvs = generateUvsForCaps(endCapPositions);\r\n newUvs.push(...generatedUvs);\r\n }\r\n\r\n const newVertexData = new VertexData();\r\n newVertexData.positions = newPositions;\r\n newVertexData.normals = newNormals;\r\n newVertexData.indices = newIndices;\r\n\r\n const combinedPositionsArray = new Float32Array(totalBasePositions + newPositions.length);\r\n const combinedNormalsArray = new Float32Array(totalBaseNormals + newNormals.length);\r\n const combinedUvsArray = new Float32Array((combinedPositionsArray.length / 3) * 2);\r\n const combinedIndicesArray = new Uint16Array(totalBaseIndices + newIndices.length);\r\n\r\n if (baseVertexData.positions) {\r\n combinedPositionsArray.set(baseVertexData.positions);\r\n combinedPositionsArray.set(newPositions, totalBasePositions);\r\n baseVertexData.positions = combinedPositionsArray;\r\n }\r\n if (baseVertexData.normals) {\r\n combinedNormalsArray.set(baseVertexData.normals);\r\n combinedNormalsArray.set(newNormals, totalBaseNormals);\r\n baseVertexData.normals = combinedNormalsArray;\r\n }\r\n if (baseVertexData.uvs) {\r\n combinedUvsArray.set(baseVertexData.uvs);\r\n combinedUvsArray.set(newUvs, totalBaseUvs);\r\n baseVertexData.uvs = combinedUvsArray;\r\n }\r\n if (baseVertexData.indices) {\r\n combinedIndicesArray.set(baseVertexData.indices);\r\n combinedIndicesArray.set(newIndices, totalBaseIndices);\r\n baseVertexData.indices = combinedIndicesArray;\r\n }\r\n return baseVertexData;\r\n}\r\n\r\nfunction reverseHandednessOfPositions(positionsArray: Vector3[]) {\r\n const reversedArray: Vector3[] = [];\r\n positionsArray.forEach(position => {\r\n reversedArray.unshift(position);\r\n });\r\n return reversedArray;\r\n}\r\n\r\nfunction generateUvsForCaps(positions: Vector3[]) {\r\n const newUvs: number[] = [];\r\n const operationVectors = findDistinctVectors(positions);\r\n if (operationVectors) {\r\n const vector1 = operationVectors[0];\r\n const vector2 = operationVectors[1];\r\n const vector3 = operationVectors[2];\r\n\r\n //xp1 is the capNormal\r\n if (vector1 && vector2 && vector3) {\r\n const xp1 = vector1.cross(vector2.subtract(vector3));\r\n const xp2 = vector1.subtract(vector2).cross(xp1);\r\n const normalizedXp3 = xp1.cross(xp2).normalize();\r\n const normalizedXp2 = xp2.normalize();\r\n positions.forEach(position => {\r\n const u = Vector3.Dot(position, normalizedXp2);\r\n const v = Vector3.Dot(position, normalizedXp3);\r\n newUvs.push(u, v);\r\n });\r\n }\r\n } else {\r\n positions.forEach(position => {\r\n newUvs.push(0, 0);\r\n });\r\n }\r\n return newUvs;\r\n}\r\n\r\nfunction findDistinctVectors(vectors: Vector3[]) {\r\n let vector1 = Vector3.Zero();\r\n let vector2 = Vector3.Zero();\r\n let vector3 = Vector3.Zero();\r\n for (let i = 0; i < vectors.length; i++) {\r\n let isZeroVector = false;\r\n if (vectors[i].x === 0 && vectors[i].y === 0 && vectors[i].z === 0) {\r\n isZeroVector = true;\r\n }\r\n if (!isZeroVector) {\r\n vector1 = vectors[i];\r\n break;\r\n }\r\n }\r\n for (let i = 0; i < vectors.length; i++) {\r\n let isZeroVector = false;\r\n if (vectors[i].x === 0 && vectors[i].y === 0 && vectors[i].z === 0) {\r\n isZeroVector = true;\r\n }\r\n if (!isZeroVector && vectors[i] !== vector1) {\r\n vector2 = vectors[i];\r\n break;\r\n }\r\n }\r\n for (let i = 0; i < vectors.length; i++) {\r\n if (vectors[i] !== vector2 && vectors[i] !== vector1) {\r\n vector3 = vectors[i];\r\n break;\r\n }\r\n }\r\n return [vector1, vector2, vector3];\r\n}\r\n\r\nVertexData.CreateRibbon = CreateRibbonVertexData;\r\n\r\nMesh.CreateRibbon = (\r\n name: string,\r\n pathArray: Vector3[][],\r\n closeArray: boolean = false,\r\n closePath: boolean,\r\n offset: number,\r\n scene?: Scene,\r\n updatable: boolean = false,\r\n sideOrientation?: number,\r\n instance?: Mesh\r\n) => {\r\n return CreateMiteredRibbon(\r\n name,\r\n {\r\n pathArray: pathArray,\r\n closeArray: closeArray,\r\n closePath: closePath,\r\n offset: offset,\r\n updatable: updatable,\r\n sideOrientation: sideOrientation,\r\n instance: instance,\r\n },\r\n scene\r\n );\r\n};\r\n","import { Ray } from '@babylonjs/core/Culling/ray';\r\nimport { Matrix, Plane, Quaternion, Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { MeshBuilder } from '@babylonjs/core/Meshes/meshBuilder';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { SweepFeature } from '@models/classes/features/sweep-feature';\r\nimport { SketchNode } from '@models/classes/meshes/sketch-node';\r\nimport { Token } from '@models/classes/token';\r\nimport { SubmeshVertexData } from '@view/helpers/geometry-helper';\r\nimport { rotationToVector } from '@view/helpers/math-helper';\r\nimport { CreateMiteredRibbon } from '@view/modules/miteredRibbonBuilder';\r\nimport { parseSketchGeometry } from '@view/modules/sketching/util';\r\nimport { applyExtrusion } from './extrude';\r\nimport { FeatureRenderer } from './feature';\r\n\r\ninterface ExtrudeOptions {\r\n shape: Vector3[];\r\n path: Vector3[];\r\n updatable: boolean;\r\n cap: number;\r\n closeShape: boolean;\r\n}\r\n\r\n//const capMap: Map = new Map();\r\n\r\nexport class SweepRenderer extends FeatureRenderer {\r\n modelToken = Token.SweepFeature;\r\n\r\n updateMeshGeometry(\r\n feature: SweepFeature,\r\n vertexData: SubmeshVertexData,\r\n vertexDataChanges: Partial\r\n ) {\r\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\r\n let parentMesh: TransformNode | undefined;\r\n if (feature._parent) {\r\n parentMesh = this.viewer.getRendererNode(feature._parent);\r\n }\r\n\r\n let pathPoints: Vector3[] = [];\r\n let targetPathNode: SketchNode | undefined;\r\n let pathMesh: Mesh | undefined;\r\n if (feature.path) {\r\n targetPathNode = feature._manager?.getById(feature.path);\r\n }\r\n if (targetPathNode) {\r\n pathMesh = this.viewer.getRendererNode(targetPathNode) as Mesh;\r\n const pathVertexData = SubmeshVertexData.ExtractFromMesh(pathMesh);\r\n const { segments } = parseSketchGeometry(pathVertexData);\r\n if (segments.length > 0) {\r\n pathPoints = segments[0];\r\n }\r\n }\r\n\r\n if (\r\n parentMesh &&\r\n targetPathNode &&\r\n pathMesh &&\r\n newVertexData.positions &&\r\n newVertexData.positions.length > 0 &&\r\n pathPoints.length > 1\r\n ) {\r\n const { segments } = parseSketchGeometry(newVertexData);\r\n const extrudedMeshes: Mesh[] = [];\r\n\r\n for (let i = 0; i < segments.length; i++) {\r\n const shape = segments[i];\r\n\r\n // Transform the path and the profile sketch into world space so we can combine them without worrying about local transforms\r\n const pathTransform = pathMesh.computeWorldMatrix();\r\n const sketchTransform = parentMesh.computeWorldMatrix();\r\n for (let i = 0; i < pathPoints.length; i++) {\r\n Vector3.TransformCoordinatesToRef(pathPoints[i], pathTransform, pathPoints[i]);\r\n }\r\n for (let i = 0; i < shape.length; i++) {\r\n Vector3.TransformCoordinatesToRef(shape[i], sketchTransform, shape[i]);\r\n }\r\n\r\n // Determine the vector to use as the primary direction of the extrusion\r\n const pathPrimaryVector = Vector3.Zero();\r\n let j = 0;\r\n while (pathPrimaryVector.length() === 0) {\r\n j++;\r\n pathPrimaryVector.copyFrom(pathPoints[j]).subtractInPlace(pathPoints[0]);\r\n }\r\n\r\n // Calculate the matrix to move the sketch path primary direction to be perpindicular to the XY plane for consistent behavior\r\n const rotToZ = rotationToVector(new Vector3(0, 0, 1), pathPrimaryVector);\r\n const pathTransformToZ = new Matrix();\r\n const pathTransformToZInv = new Matrix();\r\n rotToZ.toRotationMatrix(pathTransformToZ);\r\n Matrix.Compose(Vector3.One(), Quaternion.Identity(), pathPoints[0].scale(-1)).multiplyToRef(\r\n pathTransformToZ,\r\n pathTransformToZ\r\n );\r\n pathTransformToZ.invertToRef(pathTransformToZInv);\r\n\r\n // Transform both the path and the extrusion shape with the same matrix so the\r\n // shape and the path are positioned the same as before in relation to each other\r\n // but oriented to be extruded out of the XY plane, in the Z direction\r\n for (let i = 0; i < pathPoints.length; i++) {\r\n Vector3.TransformCoordinatesToRef(pathPoints[i], pathTransformToZ, pathPoints[i]);\r\n }\r\n for (let i = 0; i < shape.length; i++) {\r\n Vector3.TransformCoordinatesToRef(shape[i], pathTransformToZ, shape[i]);\r\n // Project the extrude shape onto the XY plane so that it's perfecly perpindicular to the sweep path\r\n shape[i].set(shape[i].x, shape[i].y, 0);\r\n }\r\n\r\n const extrudeOptions: ExtrudeOptions = {\r\n shape,\r\n path: pathPoints,\r\n updatable: false,\r\n cap: Mesh.NO_CAP,\r\n closeShape: false,\r\n // firstNormal: sketchNormal, this property is added in a future version of babylon\r\n };\r\n\r\n if (feature.capEnd && feature.capStart) {\r\n extrudeOptions.cap = Mesh.CAP_ALL;\r\n extrudeOptions.closeShape = true;\r\n } else if (feature.capStart) {\r\n extrudeOptions.cap = Mesh.CAP_START;\r\n extrudeOptions.closeShape = true;\r\n } else if (feature.capEnd) {\r\n extrudeOptions.cap = Mesh.CAP_END;\r\n extrudeOptions.closeShape = true;\r\n }\r\n\r\n // Perform the extrusion of the shape defined along the XY plane, to the path oriented to coincide with the Z+ axis\r\n let extruded: Mesh;\r\n if (feature.mitredExtrusion) {\r\n extrudeOptions.closeShape = false;\r\n extruded = this.mitredExtrude(feature._dynamicId, extrudeOptions, this.scene);\r\n } else {\r\n extruded = MeshBuilder.ExtrudeShape(feature._dynamicId, extrudeOptions, this.scene);\r\n }\r\n\r\n // Rotate the resulting extruded shape to match the orientation of the original shape profile\r\n sketchTransform.invert();\r\n pathTransformToZInv.multiplyToRef(sketchTransform, pathTransformToZInv);\r\n extruded.bakeTransformIntoVertices(pathTransformToZInv);\r\n\r\n extrudedMeshes.push(extruded);\r\n }\r\n\r\n const extrudedAll = Mesh.MergeMeshes(extrudedMeshes, true);\r\n if (extrudedAll) {\r\n applyExtrusion(newVertexData, extrudedAll, false);\r\n extrudedAll.dispose();\r\n }\r\n }\r\n\r\n return newVertexData;\r\n }\r\n\r\n update3dInternal() {}\r\n\r\n mitredExtrude(featureId: string, options: ExtrudeOptions, scene: Scene) {\r\n const shape = options.shape;\r\n const path = options.path;\r\n const closed = options.closeShape || false;\r\n\r\n const nbPoints = path.length;\r\n const line = Vector3.Zero();\r\n const nextLine = Vector3.Zero();\r\n let axisX = Vector3.Zero();\r\n let axisY = Vector3.Zero();\r\n let axisZ = Vector3.Zero();\r\n let nextAxisX = Vector3.Zero();\r\n let nextAxisY = Vector3.Zero();\r\n let nextAxisZ = Vector3.Zero();\r\n let startPoint = Vector3.Zero();\r\n const nextStartPoint = Vector3.Zero();\r\n const bisector = Vector3.Zero();\r\n let distance = 0;\r\n let ray;\r\n\r\n const allPaths = [];\r\n // Note that the sketch will always be rotated such that the profile is in the XY plane\r\n const up = Vector3.Up();\r\n\r\n for (let s = 0; s < shape.length; s++) {\r\n path[1].subtractToRef(path[0], line);\r\n axisZ = line.clone().normalize();\r\n axisX = Vector3.Cross(Vector3.Up(), axisZ).normalize();\r\n axisY = Vector3.Cross(axisZ, axisX);\r\n startPoint = path[0].add(axisX.scale(shape[s].x)).add(axisY.scale(shape[s].y));\r\n const ribbonPath = [startPoint.clone()];\r\n for (let p = 0; p < nbPoints - 2; p++) {\r\n path[p + 2].subtractToRef(path[p + 1], nextLine);\r\n nextAxisZ = nextLine.clone().normalize();\r\n nextAxisX = Vector3.Cross(up, nextAxisZ).normalize();\r\n nextAxisY = Vector3.Cross(nextAxisZ, nextAxisX);\r\n nextAxisZ.subtractToRef(axisZ, bisector);\r\n const planeParallel = Vector3.Cross(nextAxisZ, axisZ);\r\n const planeNormal = Vector3.Cross(planeParallel, bisector);\r\n const plane = Plane.FromPositionAndNormal(path[p + 1], planeNormal);\r\n ray = new Ray(startPoint, axisZ);\r\n distance = ray.intersectsPlane(plane) || 0;\r\n startPoint.addToRef(axisZ.scale(distance), nextStartPoint);\r\n ribbonPath.push(nextStartPoint.clone());\r\n\r\n axisX = nextAxisX.clone();\r\n axisY = nextAxisY.clone();\r\n axisZ = nextAxisZ.clone();\r\n startPoint = nextStartPoint.clone();\r\n }\r\n // Last Point\r\n if (closed) {\r\n path[0].subtractToRef(path[nbPoints - 1], nextLine);\r\n nextAxisZ = nextLine.clone().normalize();\r\n nextAxisX = Vector3.Cross(up, nextAxisZ).normalize();\r\n nextAxisY = Vector3.Cross(nextAxisZ, nextAxisX);\r\n nextAxisZ.subtractToRef(axisZ, bisector);\r\n const closedPlaneParallel = Vector3.Cross(nextAxisZ, axisZ);\r\n const closedPlaneNormal = Vector3.Cross(closedPlaneParallel, bisector);\r\n const closedPlane = Plane.FromPositionAndNormal(path[nbPoints - 1], closedPlaneNormal);\r\n ray = new Ray(startPoint, axisZ);\r\n distance = ray.intersectsPlane(closedPlane) || 0;\r\n startPoint.addToRef(axisZ.scale(distance), nextStartPoint);\r\n ribbonPath.push(nextStartPoint.clone());\r\n\r\n axisX = nextAxisX.clone();\r\n axisY = nextAxisY.clone();\r\n axisZ = nextAxisZ.clone();\r\n startPoint = nextStartPoint.clone();\r\n\r\n path[1].subtractToRef(path[0], nextLine);\r\n nextAxisZ = nextLine.clone().normalize();\r\n nextAxisX = Vector3.Cross(up, nextAxisZ).normalize();\r\n nextAxisY = Vector3.Cross(nextAxisZ, nextAxisX);\r\n nextAxisZ.subtractToRef(axisZ, bisector);\r\n const planeParallel = Vector3.Cross(nextAxisZ, axisZ);\r\n const planeNormal = Vector3.Cross(planeParallel, bisector);\r\n const plane = Plane.FromPositionAndNormal(path[0], planeNormal);\r\n ray = new Ray(startPoint, axisZ);\r\n distance = ray.intersectsPlane(plane) || 0;\r\n startPoint.addToRef(axisZ.scale(distance), nextStartPoint);\r\n ribbonPath.shift();\r\n ribbonPath.unshift(nextStartPoint.clone());\r\n } else {\r\n const planeNormal = axisZ;\r\n const plane = Plane.FromPositionAndNormal(path[nbPoints - 1], planeNormal);\r\n ray = new Ray(startPoint, axisZ);\r\n const distance = ray.intersectsPlane(plane) || 0;\r\n startPoint.addToRef(axisZ.scale(distance), nextStartPoint);\r\n ribbonPath.push(nextStartPoint.clone());\r\n }\r\n allPaths.push(ribbonPath);\r\n }\r\n\r\n const ribbon = CreateMiteredRibbon(\r\n featureId,\r\n {\r\n pathArray: allPaths,\r\n sideOrientation: Mesh.DOUBLESIDE,\r\n closeArray: false,\r\n closePath: false,\r\n cap: options.cap,\r\n },\r\n scene\r\n );\r\n\r\n return ribbon;\r\n }\r\n}\r\n","import { Vector3 } from '@babylonjs/core/Maths/math.vector';\r\nimport { Token, TransformFeature } from '@models/classes';\r\nimport { SubmeshVertexData } from '@view/helpers/geometry-helper';\r\nimport { FeatureRenderer } from './feature';\r\n\r\nexport class TransformRenderer extends FeatureRenderer {\r\n modelToken = Token.TransformFeature;\r\n updateMeshGeometry(\r\n feature: TransformFeature,\r\n vertexData: SubmeshVertexData,\r\n vertexDataChanges: Partial\r\n ) {\r\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\r\n const parentMesh = feature._parent;\r\n\r\n if (parentMesh && newVertexData.positions && newVertexData.positions.length > 0) {\r\n const newMeshVertexPositions = newVertexData.positions;\r\n const indexCount = newMeshVertexPositions.length / 3;\r\n const pivot = parentMesh.pivot.toVec3();\r\n const vertexPosition = new Vector3();\r\n\r\n let scaleVector: Vector3 | undefined;\r\n if (!feature.scale.equal({ x: 1, y: 1, z: 1 })) {\r\n scaleVector = feature.scale.toVec3();\r\n }\r\n\r\n for (let i = 0; i < indexCount; i++) {\r\n if (pivot && scaleVector) {\r\n vertexPosition.copyFromFloats(\r\n newMeshVertexPositions[i * 3] - pivot.x,\r\n newMeshVertexPositions[i * 3 + 1] - pivot.y,\r\n newMeshVertexPositions[i * 3 + 2] - pivot.z\r\n );\r\n\r\n if (scaleVector) {\r\n vertexPosition.multiplyInPlace(scaleVector);\r\n }\r\n\r\n vertexPosition.addInPlace(pivot);\r\n\r\n newMeshVertexPositions[i * 3] = vertexPosition.x;\r\n newMeshVertexPositions[i * 3 + 1] = vertexPosition.y;\r\n newMeshVertexPositions[i * 3 + 2] = vertexPosition.z;\r\n }\r\n }\r\n }\r\n\r\n return newVertexData;\r\n }\r\n\r\n update3dInternal() {}\r\n}\r\n","import { SubmeshData } from '@models/classes/features/submesh-feature';\r\nimport { WeldVerticesFeature } from '@models/classes/features/weld-vertices-feature';\r\nimport { Token } from '@models/classes/token';\r\nimport { SubmeshVertexData } from '@view/helpers/geometry-helper';\r\nimport { FeatureRenderer } from './feature';\r\n\r\nexport class WeldVerticesRenderer extends FeatureRenderer {\r\n modelToken = Token.WeldVerticesFeature;\r\n updateMeshGeometry(\r\n feature: WeldVerticesFeature,\r\n vertexData: SubmeshVertexData,\r\n vertexDataChanges: Partial\r\n ) {\r\n const newVertexData = super.updateMeshGeometry(feature, vertexData, vertexDataChanges);\r\n const parentMesh = feature._parent;\r\n const indices = newVertexData.indices;\r\n let positions = newVertexData.positions;\r\n let normals = newVertexData.normals;\r\n let uvs = newVertexData.uvs;\r\n const positionAverage: number[] = [0, 0, 0];\r\n const normalsAverage: number[] = [0, 0, 0];\r\n const uvAverage: number[] = [0, 0];\r\n const selectedIndices: number[] = [];\r\n const selectedVertices: number[] = [];\r\n const selectedIndicesArray: boolean[] = [];\r\n const vertexVisited: boolean[] = [];\r\n const indicesResult: number[] = [];\r\n const submeshDataArray: SubmeshData[] = [];\r\n\r\n if (parentMesh && newVertexData.positions && newVertexData.positions.length > 0) {\r\n if (newVertexData.positions) {\r\n positions = newVertexData.positions as number[];\r\n } else if (vertexData.positions) {\r\n positions = Array.from(vertexData.positions);\r\n } else {\r\n positions = [];\r\n }\r\n\r\n if (newVertexData.normals) {\r\n normals = newVertexData.normals as number[];\r\n } else if (vertexData.normals) {\r\n normals = Array.from(vertexData.normals);\r\n }\r\n\r\n if (newVertexData.uvs) {\r\n uvs = newVertexData.uvs as number[];\r\n } else if (vertexData.uvs) {\r\n uvs = Array.from(vertexData.uvs);\r\n }\r\n\r\n if (feature.vertexIndices && feature.vertexIndices.length > 0) {\r\n //creating sparse array to see which vertices are selected\r\n for (let i = 0; i < feature.vertexIndices.length; i++) {\r\n selectedIndicesArray[feature.vertexIndices[i]] = true;\r\n }\r\n\r\n if (indices) {\r\n //Looping through indices and checking which indices are touched by the selected vertices\r\n for (let j = 0; j < indices?.length; j++) {\r\n if (selectedIndicesArray[indices[j]]) {\r\n selectedIndices.push(j);\r\n if (!vertexVisited[indices[j]]) {\r\n selectedVertices.push(j);\r\n vertexVisited[indices[j]] = true;\r\n }\r\n }\r\n }\r\n }\r\n\r\n //averaging all selected positions of the vertex\r\n for (let i = 0; i < selectedVertices.length; i++) {\r\n if (indices) {\r\n const vertex = indices[selectedVertices[i]];\r\n for (let j = 0; j < 3; j++) {\r\n positionAverage[j] = positionAverage[j] + positions[vertex * 3 + j];\r\n if (normals) {\r\n normalsAverage[j] = normalsAverage[j] + normals[vertex * 3 + j];\r\n }\r\n if (j != 2 && uvs) {\r\n uvAverage[j] = uvAverage[j] + uvs[vertex * 2 + j];\r\n }\r\n }\r\n }\r\n }\r\n for (let i = 0; i < 3; i++) {\r\n positionAverage[i] = positionAverage[i] / selectedVertices.length;\r\n normalsAverage[i] = normalsAverage[i] / selectedVertices.length;\r\n if (i != 2) {\r\n uvAverage[i] = uvAverage[i] / selectedVertices.length;\r\n }\r\n }\r\n\r\n //setting one vertex to calculated average position\r\n if (indices) {\r\n const targetIndex = indices[selectedVertices[0]];\r\n for (let i = 0; i < 3; i++) {\r\n positions[targetIndex * 3 + i] = positionAverage[i];\r\n if (normals) {\r\n normals[targetIndex * 3 + i] = normalsAverage[i];\r\n }\r\n if (i != 2 && uvs) {\r\n uvs[targetIndex * 2 + i] = uvAverage[i];\r\n }\r\n }\r\n\r\n //setting all selectedIndices to targetIndex with new position\r\n for (let i = 0; i < selectedIndices.length; i++) {\r\n indices[selectedIndices[i]] = targetIndex;\r\n }\r\n\r\n //looping through indices and pruning out polygons that make up invalid geometry\r\n for (let i = 0; i < indices.length; i = i + 3) {\r\n if (\r\n indices[i] != indices[i + 1] &&\r\n indices[i] != indices[i + 2] &&\r\n indices[i + 1] != indices[i + 2]\r\n ) {\r\n indicesResult.push(indices[i]);\r\n indicesResult.push(indices[i + 1]);\r\n indicesResult.push(indices[i + 2]);\r\n }\r\n }\r\n\r\n //handling submesh data\r\n\r\n if (vertexData.submeshData) {\r\n for (let i = 0; i < vertexData.submeshData.length; i++) {\r\n const submeshData = vertexData.submeshData[i];\r\n submeshData.indexCount = indicesResult.length;\r\n submeshDataArray[i] = submeshData;\r\n }\r\n }\r\n }\r\n }\r\n newVertexData.positions = positions;\r\n newVertexData.indices = indicesResult;\r\n newVertexData.uvs = uvs;\r\n newVertexData.normals = normals;\r\n newVertexData.submeshData = submeshDataArray;\r\n }\r\n return newVertexData;\r\n }\r\n\r\n update3dInternal() {}\r\n}\r\n","import { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { eMeshChangeTypes, Feature } from '@models/classes/features/feature';\r\nimport { SubmeshData } from '@models/classes/features/submesh-feature';\r\nimport { CustomMeshNode } from '@models/classes/meshes/custom-mesh-node';\r\nimport { MeshNode } from '@models/classes/meshes/mesh-node';\r\nimport { SpaceNode } from '@models/classes/meshes/space-node';\r\nimport { Token } from '@models/classes/token';\r\nimport { isTrackedClass, ITrackedClass } from '@models/interfaces/tracked-class';\r\nimport { SubmeshVertexData, verifyGeometry } from '@view/helpers/geometry-helper';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { NormalSmoothingRenderer } from './features';\r\nimport { CircularPatternRenderer } from './features/circular-pattern';\r\nimport { DeleteFacesRenderer } from './features/delete-faces';\r\nimport { DisplacementMapRenderer } from './features/displacement-map';\r\nimport { ExtrudeRenderer } from './features/extrude';\r\nimport { FeatureRenderer } from './features/feature';\r\nimport { JoinGeometryRenderer } from './features/join-geometry';\r\nimport { LinearPatternRenderer } from './features/linear-pattern';\r\nimport { MirrorFeatureRenderer } from './features/mirror';\r\nimport { MoveVerticesRenderer } from './features/move-vertices';\r\nimport { SliceFeatureRenderer } from './features/slice';\r\nimport { SubmeshRenderer } from './features/submesh';\r\nimport { SweepRenderer } from './features/sweep';\r\nimport { TransformRenderer } from './features/transform';\r\nimport { UvMapRenderer } from './features/uvmap';\r\nimport { WeldVerticesRenderer } from './features/weld-vertices';\r\nimport { Renderer } from './renderer';\r\nimport { RendererManager } from './renderer-manager';\r\n\r\nexport class FeatureStackRenderer extends Renderer {\r\n queuedGeometryToReset = new Map>();\r\n queuedFeaturesToProcess = new Map>(); // index of feature to start recalculation at\r\n processedFeatures = new Map();\r\n featureOrder = new Map();\r\n renderers = new Map>();\r\n\r\n constructor(viewer: KbViewer, scene: Scene, private rendererManager: RendererManager) {\r\n super(viewer, scene);\r\n const moveVertices = new MoveVerticesRenderer(viewer, scene);\r\n this.renderers.set(Token.MoveVerticesFeature, moveVertices);\r\n const weldVertices = new WeldVerticesRenderer(viewer, scene);\r\n this.renderers.set(Token.WeldVerticesFeature, weldVertices);\r\n const displacementMap = new DisplacementMapRenderer(viewer, scene);\r\n this.renderers.set(Token.DisplacementMapFeature, displacementMap);\r\n const uvMaps = new UvMapRenderer(viewer, scene);\r\n this.renderers.set(Token.UvMapFeature, uvMaps);\r\n const submeshes = new SubmeshRenderer(viewer, scene, rendererManager);\r\n this.renderers.set(Token.SubmeshFeature, submeshes);\r\n const linearPatterns = new LinearPatternRenderer(viewer, scene);\r\n this.renderers.set(Token.LinearPatternFeature, linearPatterns);\r\n const circularPatterns = new CircularPatternRenderer(viewer, scene);\r\n this.renderers.set(Token.CircularPatternFeature, circularPatterns);\r\n const joinGeometry = new JoinGeometryRenderer(viewer, scene);\r\n this.renderers.set(Token.JoinGeometryFeature, joinGeometry);\r\n const deleteFaces = new DeleteFacesRenderer(viewer, scene);\r\n this.renderers.set(Token.DeleteFacesFeature, deleteFaces);\r\n const normalSmoothing = new NormalSmoothingRenderer(viewer, scene);\r\n this.renderers.set(Token.NormalSmoothingFeature, normalSmoothing);\r\n const slice = new SliceFeatureRenderer(viewer, scene);\r\n this.renderers.set(Token.SliceFeature, slice);\r\n const mirror = new MirrorFeatureRenderer(viewer, scene);\r\n this.renderers.set(Token.MirrorFeature, mirror);\r\n const transform = new TransformRenderer(viewer, scene);\r\n this.renderers.set(Token.TransformFeature, transform);\r\n const extrude = new ExtrudeRenderer(viewer, scene);\r\n this.renderers.set(Token.ExtrudeFeature, extrude);\r\n const sweep = new SweepRenderer(viewer, scene);\r\n this.renderers.set(Token.SweepFeature, sweep);\r\n }\r\n\r\n modelToken = Token.Feature;\r\n\r\n forceRebuildGeometry(meshNode: MeshNode, vertexData?: SubmeshVertexData) {\r\n if (!this.processedFeatures.has(meshNode._dynamicId)) {\r\n if (vertexData) {\r\n this.applyNewVertexData(meshNode, {}, vertexData);\r\n }\r\n return;\r\n }\r\n\r\n if (!vertexData) {\r\n vertexData = this.rendererManager.mesh.rebuildBaseGeometry(meshNode)!;\r\n }\r\n this.queuedGeometryToReset.set(meshNode._dynamicId, vertexData);\r\n const allChanges = new Set(meshNode.features.map((_, i) => i));\r\n this.queuedFeaturesToProcess.set(meshNode, allChanges);\r\n\r\n const movedIndex = this.getMovedFeatureIndex(meshNode);\r\n this.updateFeatureOrder(meshNode);\r\n if (movedIndex !== undefined && movedIndex >= 0) {\r\n this.recursivelyUpdateFeatureVisibility([meshNode.features[movedIndex]], meshNode.features, []);\r\n }\r\n this.update3dGeometry(0, meshNode);\r\n\r\n this.renderers.forEach(renderer => {\r\n renderer.clearCache(meshNode.features);\r\n });\r\n }\r\n\r\n resetMeshGeometry(\r\n meshNode: SpaceNode,\r\n dataTypes: Set | null,\r\n vertexData?: SubmeshVertexData | null,\r\n hasFeatureToUpdate = false\r\n ) {\r\n if (!vertexData) {\r\n vertexData = this.rendererManager.mesh.rebuildBaseGeometry(meshNode);\r\n }\r\n\r\n if (vertexData) {\r\n let newGeometry: Partial;\r\n if (!dataTypes) {\r\n newGeometry = vertexData;\r\n } else {\r\n // If the feature that changed affects aspects of a geometry, reset those aspects so we can recalculate them with\r\n // this or any other features that affected that aspect\r\n newGeometry = this.queuedGeometryToReset.get(meshNode._dynamicId) || {};\r\n\r\n // Null means that the vertex data specifically doesn't have those geometries, so they will be cleared\r\n if (!newGeometry.positions && hasChange(dataTypes, eMeshChangeTypes.VertexPositions)) {\r\n newGeometry.positions = vertexData.positions || null;\r\n }\r\n if (!newGeometry.normals && hasChange(dataTypes, eMeshChangeTypes.VertexNormals)) {\r\n newGeometry.normals = vertexData.normals || null;\r\n }\r\n if (!newGeometry.uvs && hasChange(dataTypes, eMeshChangeTypes.VertexUvs)) {\r\n newGeometry.uvs = vertexData.uvs || null;\r\n }\r\n if (!newGeometry.indices && hasChange(dataTypes, eMeshChangeTypes.VertexIndices)) {\r\n newGeometry.indices = vertexData.indices || null;\r\n }\r\n if (!newGeometry.submeshData && hasChange(dataTypes, eMeshChangeTypes.Submeshes)) {\r\n newGeometry.submeshData = vertexData.submeshData || null;\r\n }\r\n }\r\n this.queuedGeometryToReset.set(meshNode._dynamicId, newGeometry);\r\n\r\n // If a feature affects a geometry element that is being reset, then it will need to be recalculated\r\n for (let i = 0; i < meshNode.features.length; i++) {\r\n const feature = meshNode.features[i];\r\n\r\n // If this geometry hasn't been flagged for recalculation, see if it needs to be\r\n if (\r\n !this.queuedFeaturesToProcess.get(meshNode)?.has(i) &&\r\n feature.affects &&\r\n featureIsAffected(feature.affects, newGeometry)\r\n ) {\r\n hasFeatureToUpdate = true;\r\n this.queueGeometryUpdate(meshNode, i);\r\n this.resetMeshGeometry(meshNode, feature.affects, vertexData, hasFeatureToUpdate);\r\n }\r\n }\r\n\r\n // If there are no features to be recalculated but we need to reset some mesh geometry, then do that directly.\r\n if (!hasFeatureToUpdate && Object.getOwnPropertyNames(newGeometry).length > 0) {\r\n const existingVertexData = this.rendererManager.mesh.getMeshGeometry(meshNode) || vertexData;\r\n this.queuedGeometryToReset.delete(meshNode._dynamicId);\r\n this.applyNewVertexData(meshNode, newGeometry, existingVertexData);\r\n }\r\n }\r\n this.updateFeatureOrder(meshNode);\r\n }\r\n\r\n transformChanged(meshNode: SpaceNode) {\r\n // If the transform on a mesh changes, any features that depend on the transform need to be updated\r\n for (let i = 0; i < meshNode.features.length; i++) {\r\n const feature = meshNode.features[i];\r\n\r\n if (feature.affectedBy && feature.affectedBy.has(eMeshChangeTypes.Transform)) {\r\n this.queueGeometryUpdate(meshNode, i);\r\n }\r\n }\r\n }\r\n\r\n queueGeometryUpdate(meshNode: SpaceNode, index?: number) {\r\n let currentChanges = this.queuedFeaturesToProcess.get(meshNode);\r\n if (!currentChanges) {\r\n currentChanges = new Set();\r\n this.queuedFeaturesToProcess.set(meshNode, currentChanges);\r\n }\r\n\r\n currentChanges.add(index === undefined ? 0 : index);\r\n }\r\n\r\n update3dInternal(feature: ITrackedClass) {\r\n if (isTrackedClass(feature)) {\r\n if (!feature.enabled || feature._suppressed) {\r\n if (feature.changes.has('enabled') || feature.changes.has('_suppressed')) {\r\n this.delete3d(feature, true);\r\n }\r\n\r\n return;\r\n }\r\n\r\n const index = this.getFeatureIndexOnParent(feature);\r\n if (index > -1) {\r\n this.queueGeometryUpdate(feature._parent!, index);\r\n }\r\n\r\n // Process all of the feature's non-geometry effects\r\n const renderer = this.renderers.get(Token.withKey(feature.$type));\r\n if (!renderer) {\r\n throw Error('no renderer for feature');\r\n }\r\n renderer.update3d(feature);\r\n }\r\n }\r\n\r\n update3dGeometry(renderStartTime = 0, node?: MeshNode) {\r\n if (!node) {\r\n for (const node of this.queuedFeaturesToProcess.keys()) {\r\n if (node instanceof MeshNode) {\r\n this.update3dGeometry(renderStartTime || performance.now(), node);\r\n }\r\n }\r\n return;\r\n }\r\n\r\n if (this.queuedFeaturesToProcess.has(node) && (!(node instanceof CustomMeshNode) || node.geometryLoaded)) {\r\n const queuedFeatureToProcess = this.queuedFeaturesToProcess.get(node)!;\r\n const startFeatureIndex = Math.min(...Array.from(queuedFeatureToProcess.values()));\r\n\r\n // There is a known issue here where the VertexData.ExtractFromMesh function does not contain the updated vertices from mesh.updateVerticesData yet.\r\n // There is probably some kind of cache that is not updated if VertexData.ExtractFromMesh runs in the same frame\r\n\r\n let vertexDataCurrent = this.rendererManager.mesh.getMeshGeometry(node);\r\n\r\n const vertexDataNew: Partial = this.queuedGeometryToReset.has(node._dynamicId)\r\n ? this.queuedGeometryToReset.get(node._dynamicId)!\r\n : {};\r\n\r\n // If the whole mesh is being reset, it may be a new mesh without geometry, so we can use the entire vertexDataNew.\r\n // If the mesh renderer asks the feature renderer for updates, it will not apply the geometry on its own so we have to make sure it happens here.\r\n // This is done to prevent applying geometry to the mesh multiple times, which is expensive for very large geometry.\r\n if (!vertexDataCurrent && vertexDataNew instanceof SubmeshVertexData) {\r\n vertexDataCurrent = vertexDataNew.shallowClone();\r\n }\r\n if (!vertexDataCurrent) {\r\n this.queuedFeaturesToProcess.delete(node);\r\n this.queuedGeometryToReset.delete(node._dynamicId);\r\n return;\r\n }\r\n\r\n for (let i = startFeatureIndex; i < node.features.length; i++) {\r\n const renderTime = performance.now() - renderStartTime;\r\n\r\n if (renderTime > this.rendererManager.maxRenderTime) {\r\n // store the progress of this feature stack to continue processing next frame\r\n console.warn(\r\n `Feature processing time exceeded ${renderTime}ms, skipping ${this.queuedFeaturesToProcess.size} features this frame`\r\n );\r\n queuedFeatureToProcess.clear();\r\n queuedFeatureToProcess.add(i);\r\n this.queuedGeometryToReset.set(node._dynamicId, vertexDataNew);\r\n return;\r\n }\r\n const feature = node.features[i];\r\n\r\n const renderer = this.renderers.get(Token.withKey(feature.$type));\r\n if (!renderer) {\r\n throw Error('no renderer for feature');\r\n }\r\n\r\n if (!feature.enabled || feature._suppressed) {\r\n // If the feature is not enabled, then we don't process it, but we still want to update the feature's geometry cache\r\n // Otherwise when the feature is enabled later it might not have the right data.\r\n renderer.updateMeshGeometryCache(feature, vertexDataCurrent, vertexDataNew);\r\n continue;\r\n }\r\n\r\n // Always process the features that were modified\r\n let processThisFeature = queuedFeatureToProcess.has(i);\r\n\r\n // Figure out if a property this feature is affected by was modified this cycle\r\n if (!processThisFeature && feature.affectedBy) {\r\n processThisFeature = featureIsAffected(feature.affectedBy, vertexDataNew);\r\n }\r\n\r\n if (isTrackedClass(feature)) {\r\n if (processThisFeature) {\r\n // Process all of the features geometry effects\r\n const resultVertexData = renderer.updateMeshGeometry(feature, vertexDataCurrent, vertexDataNew);\r\n\r\n if (feature.affects) {\r\n feature.affects.forEach(affects => {\r\n if (affects === eMeshChangeTypes.VertexPositions) {\r\n vertexDataNew.positions = resultVertexData.positions || null;\r\n }\r\n if (affects === eMeshChangeTypes.VertexNormals) {\r\n vertexDataNew.normals = resultVertexData.normals || null;\r\n }\r\n if (affects === eMeshChangeTypes.VertexUvs) {\r\n vertexDataNew.uvs = resultVertexData.uvs || null;\r\n }\r\n if (affects === eMeshChangeTypes.VertexIndices) {\r\n vertexDataNew.indices = resultVertexData.indices || null;\r\n }\r\n if (affects === eMeshChangeTypes.Submeshes) {\r\n vertexDataNew.submeshData = resultVertexData.submeshData || null;\r\n }\r\n });\r\n }\r\n }\r\n }\r\n }\r\n\r\n if (vertexDataNew.submeshData) {\r\n if (!vertexDataCurrent.positions) {\r\n throw Error('Invalid geometry: positions is null');\r\n }\r\n const submeshData = vertexDataNew.submeshData.filter(s => s.materialIndex !== 0);\r\n const baseSubmeshes = calculateBaseSubmeshes(vertexDataCurrent, vertexDataNew, submeshData);\r\n const verticesCount = (vertexDataNew.positions || vertexDataCurrent.positions).length / 3;\r\n submeshData.unshift(\r\n ...baseSubmeshes.map(bs => ({\r\n materialIndex: 0,\r\n verticesStart: 0,\r\n verticesCount: verticesCount,\r\n indexStart: bs.indexStart,\r\n indexCount: bs.indexCount,\r\n }))\r\n );\r\n vertexDataNew.submeshData = submeshData;\r\n }\r\n\r\n if (Object.getOwnPropertyNames(vertexDataNew).length > 0) {\r\n this.applyNewVertexData(node, vertexDataNew, vertexDataCurrent);\r\n }\r\n }\r\n this.queuedFeaturesToProcess.delete(node);\r\n this.queuedGeometryToReset.delete(node._dynamicId);\r\n this.processedFeatures.set(node._dynamicId, true);\r\n }\r\n\r\n private getMovedFeatureIndex(mesh: MeshNode) {\r\n const oldArray = this.featureOrder.get(mesh._dynamicId);\r\n if (oldArray) {\r\n if (oldArray.length !== mesh.features.length) {\r\n return -1;\r\n }\r\n for (let i = 0; i < oldArray.length; i++) {\r\n const newArrayId = mesh.features[i]._dynamicId;\r\n if (oldArray[i] !== newArrayId) {\r\n return i;\r\n }\r\n }\r\n } else {\r\n return -1;\r\n }\r\n }\r\n\r\n private updateFeatureOrder(mesh: MeshNode) {\r\n const features: string[] = [];\r\n mesh.features.forEach(feature => {\r\n features.push(feature._dynamicId);\r\n });\r\n this.featureOrder.set(mesh._dynamicId, features);\r\n }\r\n\r\n private recursivelyUpdateFeatureVisibility(\r\n affectedFeatures: Feature[],\r\n features: Feature[],\r\n disabledFeatures: Feature[]\r\n ) {\r\n if (affectedFeatures.length === 0) {\r\n disabledFeatures.forEach(feature => {\r\n console.warn(\r\n 'The feature named',\r\n feature.name,\r\n 'got disabled due to a possible breaking change in the feature stack.'\r\n );\r\n });\r\n return;\r\n }\r\n\r\n const newAffectedFeatures: Feature[] = [];\r\n affectedFeatures.forEach(affectedFeature => {\r\n const featureIndex = getFeatureIndex(affectedFeature, features);\r\n if (featureIndex >= 0) {\r\n for (let i = featureIndex; i < features.length; i++) {\r\n const feature = features[i];\r\n if (feature.enabled) {\r\n affectedFeature.affects?.forEach(affector => {\r\n if (feature.dependsOn && feature.dependsOn.has(affector)) {\r\n feature.enabled = false;\r\n disabledFeatures.push(feature);\r\n feature._disableFeatureStackChange = true;\r\n newAffectedFeatures.push(feature);\r\n }\r\n });\r\n }\r\n }\r\n }\r\n });\r\n this.recursivelyUpdateFeatureVisibility(newAffectedFeatures, features, disabledFeatures);\r\n\r\n function getFeatureIndex(feature: Feature, features: Feature[]) {\r\n for (let i = 0; i < features.length; i++) {\r\n if (features[i] === feature) {\r\n return i;\r\n }\r\n }\r\n return -1;\r\n }\r\n }\r\n\r\n private applyNewVertexData(\r\n node: MeshNode,\r\n vertexDataNew: Partial,\r\n vertexDataCurrent: SubmeshVertexData\r\n ) {\r\n const mesh = this.viewer.getRendererNode(node) as Mesh;\r\n\r\n SubmeshVertexData.ExtendToRef(vertexDataNew, vertexDataCurrent);\r\n\r\n const invalid = verifyGeometry(vertexDataCurrent);\r\n if (invalid) {\r\n console.error(\r\n `Invalid or corrupted geometry output after feature processing in mesh ${node.name}. The ${invalid} are invalid.`\r\n );\r\n } else {\r\n vertexDataCurrent.applyToMesh(mesh);\r\n node._polygonCount = mesh.getTotalIndices() / 3;\r\n node._verticesCount = mesh.getTotalVertices();\r\n this.viewer.polygonTracker.next();\r\n this.viewer.warningTracker.next();\r\n }\r\n\r\n this.rendererManager.mesh.setRenderModeOnMesh(mesh, node._rootScene!);\r\n\r\n this.viewer.bbCache.invalidate(node);\r\n }\r\n\r\n delete3d(feature: ITrackedClass, preserveCache?: boolean) {\r\n const meshNode = feature._parent;\r\n if (meshNode && feature.affects) {\r\n this.resetMeshGeometry(meshNode, feature.affects);\r\n }\r\n\r\n const renderer = this.renderers.get(Token.withKey(feature.$type));\r\n if (!renderer) {\r\n throw Error('no renderer for feature');\r\n }\r\n renderer.delete3d(feature, preserveCache);\r\n }\r\n\r\n dispose() {\r\n this.queuedFeaturesToProcess.clear();\r\n this.renderers.forEach(renderer => renderer.dispose());\r\n super.dispose();\r\n }\r\n\r\n private getFeatureIndexOnParent(feature: Feature) {\r\n if (feature._parent) {\r\n const index = feature._parent.features.findIndex(f => f._dynamicId === feature._dynamicId);\r\n if (index > -1) {\r\n return index;\r\n }\r\n }\r\n\r\n return -1;\r\n }\r\n}\r\n\r\nfunction calculateBaseSubmeshes(\r\n vertexDataCurrent: SubmeshVertexData,\r\n vertexDataNew: Partial,\r\n submeshData: SubmeshData[]\r\n) {\r\n submeshData.sort((a, b) => a.indexStart - b.indexStart);\r\n const indicesCount = (vertexDataNew.indices || vertexDataCurrent.indices!).length;\r\n\r\n const baseSubmeshes = [\r\n {\r\n indexStart: 0,\r\n indexCount: indicesCount,\r\n },\r\n ];\r\n\r\n submeshData.forEach(submesh => {\r\n const lastSubmesh = baseSubmeshes[baseSubmeshes.length - 1];\r\n if (lastSubmesh) {\r\n if (\r\n submesh.indexStart <= lastSubmesh.indexStart &&\r\n submesh.indexStart + submesh.indexCount > lastSubmesh.indexStart\r\n ) {\r\n lastSubmesh.indexStart = submesh.indexStart + submesh.indexCount;\r\n lastSubmesh.indexCount = indicesCount - lastSubmesh.indexStart;\r\n\r\n if (lastSubmesh.indexCount <= 0) {\r\n baseSubmeshes.splice(-1, 1);\r\n }\r\n } else if (submesh.indexStart > lastSubmesh.indexStart) {\r\n lastSubmesh.indexCount = submesh.indexStart - lastSubmesh.indexStart;\r\n\r\n const remainingIndexCount = indicesCount - (submesh.indexStart + submesh.indexCount);\r\n if (remainingIndexCount > 0) {\r\n baseSubmeshes.push({\r\n indexStart: submesh.indexStart + submesh.indexCount,\r\n indexCount: remainingIndexCount,\r\n });\r\n }\r\n }\r\n }\r\n });\r\n\r\n return baseSubmeshes;\r\n}\r\n\r\nfunction featureIsAffected(affected: Set, vertexDataNew: Partial) {\r\n let result = false;\r\n affected.forEach(affectedBy => {\r\n if (affectedBy === eMeshChangeTypes.VertexPositions) {\r\n if (vertexDataNew.positions) {\r\n result = true; // Process the feature if a property it's affected by was changed this cycle\r\n }\r\n }\r\n if (affectedBy === eMeshChangeTypes.VertexNormals) {\r\n if (vertexDataNew.normals) {\r\n result = true; // Process the feature if a property it's affected by was changed this cycle\r\n }\r\n }\r\n if (affectedBy === eMeshChangeTypes.VertexUvs) {\r\n if (vertexDataNew.uvs) {\r\n result = true; // See above\r\n }\r\n }\r\n if (affectedBy === eMeshChangeTypes.VertexIndices) {\r\n if (vertexDataNew.indices) {\r\n result = true; // See above\r\n }\r\n }\r\n if (affectedBy === eMeshChangeTypes.Submeshes) {\r\n if (vertexDataNew.submeshData) {\r\n result = true; // See above\r\n }\r\n }\r\n });\r\n\r\n return result;\r\n}\r\n\r\nfunction hasChange(set: Set, changeType: eMeshChangeTypes) {\r\n return set.has(changeType);\r\n}\r\n","import { StandardMaterial } from '@babylonjs/core/Materials/standardMaterial';\r\nimport { Axis, Color3 } from '@babylonjs/core/Maths/math';\r\nimport { Matrix, Quaternion, Vector3 } from '@babylonjs/core/Maths/math.vector';\r\nimport { CreateSphere } from '@babylonjs/core/Meshes/Builders/sphereBuilder';\r\nimport { LinesMesh } from '@babylonjs/core/Meshes/linesMesh';\r\nimport { MeshBuilder } from '@babylonjs/core/Meshes/meshBuilder';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { KbVector } from '@models/classes';\r\nimport { instanceOf } from '@models/classes/decorators';\r\nimport { eConnectorPositionMode, eSketchPathType } from '@models/classes/enums';\r\nimport { BaseConnector } from '@models/classes/features/base-connector';\r\nimport { SketchControlPoint } from '@models/classes/features/sketch-control-point';\r\nimport { Kb3dObject } from '@models/classes/kb3d-object';\r\nimport { SketchPath } from '@models/classes/meshes/sketch-node-path';\r\nimport { SpaceNode } from '@models/classes/meshes/space-node';\r\nimport { IBoundingBox } from '@models/classes/primitives/primitive-utilities';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass, isTrackedClass } from '@models/interfaces/tracked-class';\r\nimport { BoundingBoxRelativeHelper } from '@view/helpers/bounding-box-relative-helper';\r\nimport { getRotationBtwVectors, toRadians } from '@view/helpers/math-helper';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { RendererManager } from '../renderer-manager';\r\nimport { CONNECTOR_DIAM, ConnectorParts, ConnectorRenderer, IConnectorLayer } from './connector';\r\n\r\ninterface IConnectorLayerWithParts extends IConnectorLayer {\r\n parts: Map;\r\n}\r\n\r\nconst getMeshName = {\r\n handles: (dynamicId: string) => `${dynamicId}_handles`,\r\n handleBar: (dynamicId: string) => `${dynamicId}_handleBar`,\r\n directionHandle1: (dynamicId: string) => `${dynamicId}_directionHandle1`,\r\n directionHandle2: (dynamicId: string) => `${dynamicId}_directionHandle2`,\r\n};\r\n\r\nexport class SketchControlPointRenderer extends ConnectorRenderer {\r\n protected layers: Map<'real' | 'utility', IConnectorLayerWithParts>;\r\n protected readonly controlPointScaleRatio = 0.12;\r\n\r\n constructor(viewer: KbViewer, scene: Scene, rendererManager: RendererManager) {\r\n super(viewer, scene, rendererManager);\r\n this.layers.forEach(layer => {\r\n layer.parts = new Map();\r\n });\r\n }\r\n modelToken = Token.SketchControlPoint;\r\n\r\n public handlesNode(node: Kb3dObject): boolean {\r\n return instanceOf(node, Token.SketchControlPoint);\r\n }\r\n\r\n update3dInternal(c: ITrackedClass) {\r\n const realLayer = this.layers.get('real')!;\r\n const utilityLayer = this.layers.get('utility')!;\r\n const { mesh, utilityMesh } = super.update3dInternal(c as ITrackedClass, true);\r\n\r\n if (c._parent instanceof SketchPath) {\r\n const realParts = realLayer.parts.get(c._dynamicId);\r\n const utilityParts = utilityLayer.parts.get(c._dynamicId);\r\n\r\n if (c.changes.hasAny('direction', 'direction2')) {\r\n if (realParts) {\r\n updateHandles(realParts, c, c._parent);\r\n }\r\n if (utilityParts) {\r\n updateHandles(utilityParts, c, c._parent);\r\n }\r\n }\r\n }\r\n\r\n return { mesh, utilityMesh };\r\n\r\n function updateHandles(parts: ConnectorParts, controlPoint: SketchControlPoint, path: SketchPath) {\r\n let enableHandle1: boolean, enableHandle2: boolean;\r\n\r\n switch (path.type) {\r\n case eSketchPathType.arcThrough3Points:\r\n case eSketchPathType.points:\r\n case eSketchPathType.centerRadiusCircle:\r\n case eSketchPathType.catmullRom:\r\n enableHandle1 = false;\r\n enableHandle2 = false;\r\n break;\r\n\r\n case eSketchPathType.quadraticBezier:\r\n case eSketchPathType.hermite:\r\n // Don't show handles on the ends of the path since that handle does nothing\r\n enableHandle1 =\r\n path.controlPoints.findIndex(cp => cp._dynamicId === controlPoint._dynamicId) <\r\n path.controlPoints.length - 1;\r\n enableHandle2 = false;\r\n break;\r\n\r\n default:\r\n // Don't show handles on the ends of the path since that handle does nothing\r\n enableHandle1 =\r\n path.controlPoints.findIndex(cp => cp._dynamicId === controlPoint._dynamicId) <\r\n path.controlPoints.length - 1;\r\n enableHandle2 = path.controlPoints.findIndex(cp => cp._dynamicId === controlPoint._dynamicId) > 0;\r\n }\r\n\r\n if (parts.directionHandle1 && c.direction.length() > 0) {\r\n parts.directionHandle1.position = c.direction.toVec3();\r\n }\r\n if (parts.directionHandle2 && c.direction2 && c.direction2.length() > 0) {\r\n parts.directionHandle2.position = c.direction2.toVec3();\r\n }\r\n parts.directionHandle1?.setEnabled(enableHandle1);\r\n parts.directionHandle2?.setEnabled(enableHandle2);\r\n\r\n if (parts.handleBar) {\r\n parts.handleBar.setEnabled(true);\r\n parts.handleBar = MeshBuilder.CreateDashedLines(getMeshName.handleBar(c._dynamicId), {\r\n points: [\r\n enableHandle1 ? c.direction.toVec3() : Vector3.Zero(),\r\n Vector3.Zero(),\r\n enableHandle2 && c.direction2 ? c.direction2.toVec3() : Vector3.Zero(),\r\n ],\r\n updatable: true,\r\n instance: parts.handleBar,\r\n });\r\n }\r\n }\r\n }\r\n\r\n delete3d(c: ITrackedClass) {\r\n const mesh = this.getMesh(c, this.layers.get('real')!);\r\n const utilityMesh = this.getMesh(c, this.layers.get('utility')!);\r\n this.viewer.camera.clearMaintainedNodeScreenSize(mesh);\r\n if (utilityMesh) {\r\n this.viewer.camera.clearMaintainedNodeScreenSize(utilityMesh);\r\n }\r\n super.delete3d(c);\r\n\r\n const sketchPath = c._parent;\r\n if (sketchPath && isTrackedClass(sketchPath)) {\r\n sketchPath.setPropertyIsModified('controlPoints', true);\r\n }\r\n }\r\n\r\n updateDirection(c: SketchControlPoint) {\r\n const realLayer = this.layers.get('real')!;\r\n const utilityLayer = this.layers.get('utility')!;\r\n const mesh = this.getMesh(c, realLayer);\r\n const utilityMesh = this.getMesh(c, utilityLayer);\r\n\r\n const deltaBtwXaxisAndDirection = getRotationBtwVectors(\r\n Axis.X,\r\n this.getRelativeDirection(c, c.direction).normalize()\r\n );\r\n const angleRot = Quaternion.RotationAxis(Axis.X, toRadians(c.angle));\r\n mesh.rotationQuaternion = utilityMesh.rotationQuaternion = deltaBtwXaxisAndDirection.multiply(angleRot);\r\n\r\n const realParts = realLayer.parts.get(c._dynamicId);\r\n const utilityParts = utilityLayer.parts.get(c._dynamicId);\r\n if (realParts?.handleContainer) {\r\n const invTransform = Matrix.Compose(Vector3.One(), mesh.rotationQuaternion, Vector3.Zero());\r\n invTransform.invert();\r\n invTransform.decomposeToTransformNode(realParts.handleContainer);\r\n if (utilityParts?.handleContainer) {\r\n invTransform.decomposeToTransformNode(utilityParts.handleContainer);\r\n }\r\n }\r\n\r\n this.updateScale(c);\r\n }\r\n\r\n updatePosition(c: SketchControlPoint, bb?: IBoundingBox) {\r\n const mesh = this.getMesh(c, this.layers.get('real')!);\r\n const utilityMesh = this.getMesh(c, this.layers.get('utility')!);\r\n\r\n mesh.position = this.getControlPointAbsPosition(c, bb);\r\n if (utilityMesh) {\r\n utilityMesh.position = mesh.position;\r\n }\r\n\r\n const sketchPath = c._parent;\r\n if (sketchPath && isTrackedClass(sketchPath)) {\r\n sketchPath.setPropertyIsModified('controlPoints', true);\r\n }\r\n }\r\n\r\n getControlPointAbsPosition(c: SketchControlPoint, bb?: IBoundingBox) {\r\n if (c.positionMode == eConnectorPositionMode.boundingBox) {\r\n if (!bb) {\r\n bb = c.getSketch()?.getBoundingBox();\r\n }\r\n if (bb) {\r\n return BoundingBoxRelativeHelper.convertRelativeToAbsolute(c, bb);\r\n } else {\r\n return new Vector3(c.position.x, c.position.y, c.position.z);\r\n }\r\n } else {\r\n return c.position.toVec3();\r\n }\r\n }\r\n\r\n getRelativeDirection(c: SketchControlPoint, direction: KbVector, bb?: IBoundingBox) {\r\n if (c.positionMode === eConnectorPositionMode.absolute) {\r\n return direction.toVec3();\r\n } else {\r\n if (!bb) {\r\n bb = c.getSketch()?.getBoundingBox();\r\n }\r\n if (!bb) {\r\n return Vector3.Zero();\r\n }\r\n return BoundingBoxRelativeHelper.convertRelativeDirectionToAbsolute(c, direction, bb);\r\n }\r\n }\r\n\r\n protected getMesh(c: BaseConnector, layer: IConnectorLayerWithParts) {\r\n let mesh = layer.map.get(c._dynamicId);\r\n if (!mesh) {\r\n if (!layer.template) {\r\n layer.template = this.createMeshTemplate(layer);\r\n }\r\n\r\n mesh = layer.template.clone(c._dynamicId); // layer.template.createInstance(c._dynamicId);\r\n mesh.id = c._dynamicId;\r\n\r\n const [handleContainer] = mesh.getChildren() as TransformNode[];\r\n const [directionHandle1, directionHandle2, _handleBar] = handleContainer.getChildMeshes(true);\r\n const handleBar = _handleBar as LinesMesh;\r\n handleBar.makeGeometryUnique();\r\n\r\n handleContainer.id = getMeshName.handles(c._dynamicId);\r\n directionHandle1.id = getMeshName.directionHandle1(c._dynamicId);\r\n directionHandle2.id = getMeshName.directionHandle2(c._dynamicId);\r\n\r\n const parentNode = c._parent?._parent;\r\n\r\n if (parentNode && parentNode instanceof SpaceNode) {\r\n const parent3d = this.viewer.getRendererNode(parentNode);\r\n if (parent3d) {\r\n mesh.setParent(parent3d);\r\n }\r\n }\r\n\r\n mesh.isPickable = layer.pickable;\r\n mesh.position = Vector3.Zero();\r\n mesh.rotationQuaternion = Quaternion.Identity();\r\n layer.map.set(c._dynamicId, mesh);\r\n layer.parts.set(c._dynamicId, {\r\n mesh,\r\n handleContainer: handleContainer as TransformNode,\r\n directionHandle1,\r\n directionHandle2,\r\n handleBar: handleBar,\r\n });\r\n }\r\n return mesh;\r\n }\r\n\r\n protected createMeshTemplate(layer: IConnectorLayer) {\r\n const mesh = super.createMeshTemplate(layer);\r\n\r\n const handleContainer = new TransformNode('', this.scene);\r\n\r\n const handleBall = CreateSphere('handle1', {\r\n segments: 4,\r\n diameter: CONNECTOR_DIAM / 2,\r\n });\r\n handleBall.setParent(handleContainer);\r\n const m = new StandardMaterial('controlPointMaterial', this.scene);\r\n m.emissiveColor = this.viewer.highlightColor.toColor3();\r\n m.disableLighting = true;\r\n handleBall.material = m;\r\n\r\n const handle1 = handleBall;\r\n const handle2 = handleBall.clone('handle2', handleContainer, true);\r\n handle1.setEnabled(false);\r\n handle2.setEnabled(false);\r\n const baseColor = this.viewer.sceneNode?.edgeColor\r\n ? this.viewer.sceneNode.edgeColor.toColor3()\r\n : Color3.White();\r\n const handleBar = MeshBuilder.CreateDashedLines('handleBar', {\r\n points: [Vector3.Up(), Vector3.Zero(), Vector3.Down()],\r\n updatable: true,\r\n dashSize: 1,\r\n gapSize: 1,\r\n dashNb: 30,\r\n });\r\n handleBar.color = baseColor;\r\n handleBar.isPickable = false;\r\n // const handleBarMaterial = new StandardMaterial('controlPointHandleBarMaterial', this.scene);\r\n // handleBarMaterial.emissiveColor = this.viewer.sceneNode?.edgeColor ? this.viewer.sceneNode.edgeColor.toColor3() : Color3.White();\r\n // handleBar.material = handleBarMaterial;\r\n handleBar.setParent(handleContainer);\r\n handleContainer.setParent(mesh);\r\n\r\n return mesh;\r\n }\r\n\r\n protected addSubscription(c: ITrackedClass) {\r\n const sketch = c.getSketch();\r\n\r\n if (sketch && !this.subscriptionMap.has(c)) {\r\n this.subscriptionMap.set(\r\n c,\r\n sketch._boundingBoxChanges.subscribe(bbInfo => {\r\n this.updatePosition(c, bbInfo);\r\n this.updateScale(c);\r\n })\r\n );\r\n }\r\n }\r\n}\r\n","import { Kb3dObject } from '@models/classes/kb3d-object';\r\nimport { SpaceNode } from '@models/classes/meshes/space-node';\r\nimport { isTrackedClass, ITrackedClass } from '@models/interfaces/tracked-class';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { ChangeMap } from '../renderer-manager';\r\nimport { Processor } from './processor';\r\n\r\nexport class ReferenceProcessor extends Processor {\r\n /**\r\n * referencedNodeId => ( referencedByNodeId => referencedByNode )\r\n *\r\n * Any node that is in a nested scene will be keyed by the nested scene ID.\r\n * If the node is not in a nested scene (or there is no nesting), then the node will be keyed by the bare node id.\r\n * This applies to the referencedNodeId and the referencedByNodeId.\r\n * */\r\n\r\n referencesTo = new Map<\r\n string,\r\n Map<\r\n string,\r\n {\r\n limitKeys?: Set;\r\n ref: Kb3dObject;\r\n referencePropertyName: string | symbol;\r\n }\r\n >\r\n >();\r\n\r\n orphanedReferences = new Map<\r\n string,\r\n {\r\n referencedBy: Kb3dObject;\r\n propName: string | symbol;\r\n referenceId: string;\r\n }\r\n >();\r\n\r\n constructor(private viewer: KbViewer) {\r\n super();\r\n }\r\n\r\n beforeDeletions(deletions: ChangeMap) {\r\n const rootScene = this.viewer.sceneNode;\r\n if (!rootScene || !rootScene._manager) {\r\n return;\r\n }\r\n const rootManager = rootScene._manager;\r\n\r\n deletions.all().forEach(node => {\r\n if (!node._manager) {\r\n return;\r\n }\r\n\r\n let nodeId = node.id;\r\n if (node._parentScene?.isNested()) {\r\n nodeId = rootScene.getNestedPathTo(node);\r\n }\r\n\r\n const referenceTo = this.referencesTo.get(nodeId);\r\n if (referenceTo && referenceTo.size > 0) {\r\n referenceTo.forEach((opts, referencedById) => {\r\n const referencedBy = rootManager.getById(referencedById);\r\n if (referencedBy && (referencedBy as any)[opts.referencePropertyName] === nodeId) {\r\n this.orphanedReferences.set(nodeId, {\r\n referencedBy,\r\n propName: opts.referencePropertyName,\r\n referenceId: (referencedBy as any)[opts.referencePropertyName],\r\n });\r\n (referencedBy as any)[opts.referencePropertyName] = undefined;\r\n }\r\n });\r\n }\r\n });\r\n }\r\n\r\n getReferencesTo() {\r\n return this.referencesTo;\r\n }\r\n\r\n beforeUpdates(changes: ChangeMap): void {\r\n const rootScene = this.viewer.sceneNode;\r\n if (!rootScene || !rootScene._manager) {\r\n return;\r\n }\r\n\r\n changes.all().forEach(node => {\r\n let nodeId = node.id;\r\n if (node._parentScene?.isNested()) {\r\n nodeId = rootScene.getNestedPathTo(node);\r\n }\r\n\r\n const orphanedReference = this.orphanedReferences.get(nodeId);\r\n if (orphanedReference) {\r\n (orphanedReference.referencedBy as any)[orphanedReference.propName] = orphanedReference.referenceId;\r\n }\r\n });\r\n }\r\n\r\n afterUpdates(changes: ChangeMap) {\r\n const rootScene = this.viewer.sceneNode;\r\n if (!rootScene || !rootScene._manager) {\r\n return;\r\n }\r\n\r\n changes.all().forEach(node => {\r\n if (!node._manager) {\r\n return;\r\n }\r\n\r\n const changedNodeManager = node._manager;\r\n // This block is for managing/updating the node reference\r\n // First, check nodes referencing other nodes for updates\r\n if (node.nodeReferences) {\r\n const targettingProperties = node.nodeReferences.keys();\r\n for (const _prop of targettingProperties) {\r\n const property = _prop as keyof Kb3dObject;\r\n const referenceSettings = node.nodeReferences.get(_prop)!;\r\n\r\n // If a node's references were changed, update all the references.\r\n if (node.changes.has(property)) {\r\n if (node[property] && typeof node[property] !== 'string') {\r\n console.error(`Non-string id reference value found on ${node.name} property ${property}`);\r\n }\r\n\r\n // First, remove the node from the previous target node's referencing nodes set\r\n if (node.originalValues.has(property)) {\r\n const originalReferencedNodeId: string = node.originalValues.get(property);\r\n if (originalReferencedNodeId) {\r\n // Figure out the original target node's key\r\n let originalReferencedNodeKey = originalReferencedNodeId;\r\n const originalReferencedNode = changedNodeManager.getById(originalReferencedNodeId);\r\n if (originalReferencedNode?._parentScene?.isNested()) {\r\n originalReferencedNodeKey = rootScene.getNestedPathTo(originalReferencedNode);\r\n }\r\n\r\n const referencedBy = this.referencesTo.get(originalReferencedNodeKey);\r\n if (referencedBy) {\r\n let nestedPathIdSource = node.id;\r\n if (node._parentScene?.isNested()) {\r\n nestedPathIdSource = rootScene.getNestedPathTo(node);\r\n }\r\n referencedBy.delete(nestedPathIdSource);\r\n }\r\n }\r\n }\r\n\r\n // Set up the new references or delete the reference from entry if the reference was cleared\r\n if (node[property]) {\r\n const targetNode = node._manager.getById(node[property]);\r\n if (targetNode) {\r\n if (targetNode instanceof referenceSettings.classRef) {\r\n let nestedPathIdTarget: string = node[property];\r\n let nestedPathIdSource = node.id;\r\n const isNestedSource = node._parentScene?.isNested() || false;\r\n\r\n if (nestedPathIdTarget.startsWith('@')) {\r\n const nestedTargetNode = changedNodeManager.getById(nestedPathIdTarget);\r\n if (nestedTargetNode) {\r\n nestedPathIdTarget = rootScene.getNestedPathTo(nestedTargetNode);\r\n }\r\n } else if (isNestedSource) {\r\n // If source node is in a nested scene and the target node id is not a nested path, that means that\r\n // the nested node is targetting a node in it's own scene. But we need to use\r\n // a nested path because there could be multiple instances of the nested node's scene\r\n nestedPathIdTarget = rootScene.getNestedPathTo(targetNode);\r\n }\r\n if (isNestedSource) {\r\n nestedPathIdSource = rootScene.getNestedPathTo(node);\r\n }\r\n\r\n let referencedBy = this.referencesTo.get(nestedPathIdTarget);\r\n if (!referencedBy) {\r\n referencedBy = new Map();\r\n\r\n this.referencesTo.set(nestedPathIdTarget, referencedBy);\r\n }\r\n\r\n referencedBy.set(nestedPathIdSource, {\r\n limitKeys: referenceSettings.limitKeys,\r\n ref: node,\r\n referencePropertyName: property,\r\n });\r\n } else {\r\n console.warn(\r\n `Node ${node.name} property ${property} is not referencing a ${referenceSettings.classRef}`\r\n );\r\n }\r\n }\r\n }\r\n }\r\n }\r\n }\r\n });\r\n\r\n // Next, check nodes being referenced by other nodes to see if the other nodes will need to be updated\r\n changes.all().forEach(node => {\r\n this.triggerChangesOnReferencers(node);\r\n });\r\n }\r\n\r\n triggerChangesOnReferencers(node: ITrackedClass, changes?: Map) {\r\n const changeMap = changes || new Map(node.changes);\r\n\r\n if (changeMap.size > 0) {\r\n this.afterCleanup(() => {\r\n this.triggerChangesOnReferencersSync(node, changeMap);\r\n });\r\n }\r\n }\r\n\r\n private triggerChangesOnReferencersSync(\r\n node: ITrackedClass,\r\n changes: Map,\r\n transformationPropagation?: boolean\r\n ) {\r\n const rootScene = this.viewer.sceneNode;\r\n if (!rootScene || !rootScene._manager) {\r\n return;\r\n }\r\n const rootManager = rootScene._manager;\r\n\r\n let nodeId = node.id;\r\n if (node._parentScene?.isNested()) {\r\n nodeId = rootScene.getNestedPathTo(node);\r\n }\r\n\r\n const referenceTo = this.referencesTo.get(nodeId);\r\n if (referenceTo) {\r\n referenceTo.forEach((referenceSettings, referencedById) => {\r\n const referencedBy = rootManager.getById(referencedById);\r\n if (referencedBy && isTrackedClass(referencedBy)) {\r\n if (\r\n !referenceSettings.limitKeys ||\r\n transformationPropagation ||\r\n changes.hasAny(...(referenceSettings.limitKeys as Set))\r\n ) {\r\n referencedBy.setPropertyIsModified(referenceSettings.referencePropertyName, true);\r\n }\r\n }\r\n });\r\n }\r\n\r\n // For transformation changes, propagate to the children because the children will also get moved without passing through the change tracker\r\n // This avoids needing a secondary watcher for node position\r\n if (\r\n node instanceof SpaceNode &&\r\n (changes as Map).hasAny(\r\n 'position',\r\n 'eulerRotation',\r\n 'rotationAngle',\r\n 'rotationAxis',\r\n 'pivot',\r\n 'scaling'\r\n )\r\n ) {\r\n const children = node.getChildren();\r\n for (let i = 0; i < children.length; i++) {\r\n // Always trigger changes on children because they may have different property names\r\n this.triggerChangesOnReferencersSync(children[i] as ITrackedClass, changes, true);\r\n }\r\n }\r\n }\r\n\r\n dispose() {\r\n this.referencesTo.clear();\r\n }\r\n}\r\n","import { Curve3, Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { eConnectorPositionMode, eSketchPathType } from '@models/classes/enums';\r\nimport { SketchControlPoint } from '@models/classes/features/sketch-control-point';\r\nimport { SketchNode } from '@models/classes/meshes/sketch-node';\r\nimport { SketchPath } from '@models/classes/meshes/sketch-node-path';\r\nimport { IKbVectorLike, KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { IBoundingBox } from '@models/classes/primitives/primitive-utilities';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass, isTrackedClass } from '@models/interfaces/tracked-class';\r\nimport { SketchControlPointRenderer } from '../features/sketch-control-point';\r\nimport { ChangeMap } from '../renderer-manager';\r\nimport { Processor } from './processor';\r\n\r\n/**\r\n * Processor to calculate the bounding box of a sketch, since it is determined by the path defined within.\r\n * Must run after the mate processor because mates can move sketch points and change the bounding box.\r\n */\r\nexport class SketchProcessor extends Processor {\r\n constructor(private controlPointRenderer: SketchControlPointRenderer) {\r\n super();\r\n }\r\n\r\n beforeRender() {}\r\n\r\n beforeDeletions(deletions: ChangeMap) {\r\n // Update paths that need to be updated\r\n if (deletions.has(Token.SketchPath)) {\r\n const sketchPaths = deletions.get(Token.SketchPath) as Array>;\r\n sketchPaths.forEach(p => this.updatePath(p));\r\n }\r\n\r\n if (deletions.has(Token.SketchControlPoint) || deletions.has(Token.SketchPath)) {\r\n const sketchPathChanges = deletions.get(Token.SketchPath) as Array>;\r\n const sketchControlPointChanges = deletions.get(Token.SketchControlPoint) as Array<\r\n ITrackedClass\r\n >;\r\n const sketchesToUpdate = new Map();\r\n sketchPathChanges.forEach(node => {\r\n const sketch = node._parent;\r\n if (sketch) {\r\n if (!sketchesToUpdate.has(sketch._dynamicId)) {\r\n sketchesToUpdate.set(sketch._dynamicId, sketch);\r\n }\r\n }\r\n });\r\n sketchControlPointChanges.forEach(node => {\r\n const sketch = node.getSketch();\r\n if (sketch) {\r\n if (!sketchesToUpdate.has(sketch._dynamicId)) {\r\n sketchesToUpdate.set(sketch._dynamicId, sketch);\r\n }\r\n }\r\n });\r\n\r\n sketchesToUpdate.forEach(this.calculateSketchBounds);\r\n }\r\n }\r\n\r\n beforeUpdates(changes: ChangeMap) {\r\n // Update paths that need to be updated\r\n if (changes.has(Token.SketchPath)) {\r\n const sketchPaths = changes.get(Token.SketchPath) as Array>;\r\n sketchPaths.forEach(p => this.updatePath(p));\r\n }\r\n\r\n // Update the sketch bounding box with the path control points within it that are absolute positioned\r\n if (changes.has(Token.SketchControlPoint)) {\r\n const sketchControlPointChanges = changes.get(Token.SketchControlPoint) as Array<\r\n ITrackedClass\r\n >;\r\n const sketchesToUpdate = new Map();\r\n sketchControlPointChanges.forEach(node => {\r\n if (node.changes.hasAny('position', 'positionMode')) {\r\n const sketch = node.getSketch();\r\n if (sketch) {\r\n if (!sketchesToUpdate.has(sketch._dynamicId)) {\r\n sketchesToUpdate.set(sketch._dynamicId, sketch);\r\n }\r\n }\r\n }\r\n });\r\n\r\n sketchesToUpdate.forEach(this.calculateSketchBounds);\r\n }\r\n }\r\n\r\n afterUpdates(changes: ChangeMap): void {\r\n if (changes.has(Token.SketchControlPoint)) {\r\n const sketchControlPointChanges = changes.get(Token.SketchControlPoint) as Array<\r\n ITrackedClass\r\n >;\r\n sketchControlPointChanges.forEach(node => {\r\n const sketch = node.getSketch();\r\n if (sketch) {\r\n this.afterCleanup(() => {\r\n sketch.paths.forEach(path => {\r\n if (isTrackedClass(path)) {\r\n this.updatePath(path, true);\r\n }\r\n });\r\n sketch.connectors.forEach(connector => {\r\n if (connector.positionMode === eConnectorPositionMode.moveAlongSketch) {\r\n if (connector._parent instanceof SketchNode) {\r\n connector._sketchControlPoints = this.getControlPoints(connector._parent);\r\n }\r\n }\r\n });\r\n });\r\n }\r\n });\r\n }\r\n }\r\n\r\n calculateSketchBounds(sketch: SketchNode) {\r\n const controlPoints = sketch.paths.flatMap(path => path.controlPoints);\r\n let min: Vector3 | undefined;\r\n let max: Vector3 | undefined;\r\n\r\n controlPoints.forEach(controlPoint => {\r\n if (controlPoint.positionMode === eConnectorPositionMode.absolute) {\r\n const position = controlPoint.position;\r\n if (!min) {\r\n min = position.toVec3();\r\n }\r\n\r\n max = position.toVec3();\r\n }\r\n });\r\n\r\n if (min && max) {\r\n sketch._boundingBoxMin = KbVector.FromVec3(min);\r\n sketch._boundingBoxMax = KbVector.FromVec3(max);\r\n } else {\r\n sketch._boundingBoxMin = KbVector.Zero();\r\n sketch._boundingBoxMax = KbVector.Zero();\r\n }\r\n sketch._boundingBoxChanges.next(sketch.getBoundingBox());\r\n }\r\n\r\n private updatePath(path: ITrackedClass, force?: boolean) {\r\n let points: Vector3[] | undefined;\r\n\r\n if (\r\n force ||\r\n path.changes.hasAny('controlPoints', 'tessellation', 'type', 'holeForPathGroup', 'pathGroupKey') ||\r\n !path._points?.length\r\n ) {\r\n points = this.buildSketchPoints(path);\r\n path._points = points;\r\n\r\n const parent = path._parent;\r\n if (parent instanceof SketchNode && isTrackedClass(parent)) {\r\n parent.setPropertyIsModified('paths', true);\r\n }\r\n } else if (path.changes.has('_selected')) {\r\n const parent = path._parent;\r\n if (parent instanceof SketchNode && isTrackedClass(parent)) {\r\n parent.setPropertyIsModified('paths', true);\r\n }\r\n }\r\n }\r\n\r\n private buildSketchPoints(path: SketchPath) {\r\n const points: Vector3[] = [];\r\n if (!path._parent) {\r\n return points;\r\n }\r\n const bb = path._parent.getBoundingBox();\r\n\r\n switch (path.type) {\r\n case eSketchPathType.catmullRom: {\r\n const positions = path.controlPoints.map(v => this.getPointPosition(v));\r\n if (positions.length > 2) {\r\n const curve = Curve3.CreateCatmullRomSpline(positions, path.tessellation, false);\r\n points.push(...curve.getPoints());\r\n } else {\r\n points.push(...positions);\r\n }\r\n\r\n break;\r\n }\r\n case eSketchPathType.hermite: {\r\n for (let i = 0; i + 1 < path.controlPoints.length; i++) {\r\n const start = path.controlPoints[i],\r\n end = path.controlPoints[i + 1];\r\n const p0 = this.getPointPosition(start);\r\n const p1 = this.getPointPosition(end);\r\n const d0 = this.getPointDirection(start, bb);\r\n const d1 = this.getPointDirection(end, bb);\r\n\r\n const curve = Curve3.CreateHermiteSpline(p0, d0, p1, d1, path.tessellation);\r\n points.push(...curve.getPoints());\r\n }\r\n break;\r\n }\r\n case eSketchPathType.cubicBezier: {\r\n for (let i = 0; i + 1 < path.controlPoints.length; i++) {\r\n const start = path.controlPoints[i],\r\n end = path.controlPoints[i + 1];\r\n const direction2 = this.getPointDirection2(end, bb);\r\n if (direction2) {\r\n const p0 = this.getPointPosition(start);\r\n const p3 = this.getPointPosition(end);\r\n if (p0.equals(p3)) {\r\n continue;\r\n }\r\n\r\n const p1 = p0.add(this.getPointDirection(start, bb));\r\n const p2 = p3.add(direction2);\r\n const curve = Curve3.CreateCubicBezier(p0, p1, p2, p3, path.tessellation);\r\n points.push(...curve.getPoints());\r\n }\r\n }\r\n break;\r\n }\r\n case eSketchPathType.quadraticBezier: {\r\n const controlPoint = new Vector3();\r\n for (let i = 0; i + 1 < path.controlPoints.length; i++) {\r\n const start = path.controlPoints[i],\r\n end = path.controlPoints[i + 1];\r\n const p0 = this.getPointPosition(start);\r\n const p2 = this.getPointPosition(end);\r\n if (p0.equals(p2)) {\r\n continue;\r\n }\r\n\r\n p0.addToRef(this.getPointDirection(start, bb), controlPoint);\r\n\r\n const curve = Curve3.CreateQuadraticBezier(p0, controlPoint, p2, path.tessellation);\r\n points.push(...curve.getPoints());\r\n }\r\n break;\r\n }\r\n case eSketchPathType.centerRadiusCircle: {\r\n if (path.controlPoints.length >= 2) {\r\n const [p0, p1] = path.controlPoints.slice(0, 2).map(cp => this.getPointPosition(cp));\r\n const normal = this.getPointDirection(path.controlPoints[0], bb).normalize();\r\n const p2 = Vector3.Cross(p0.subtract(p1), normal).addInPlace(p0);\r\n const curve = Curve3.ArcThru3Points(\r\n p0.scale(2).subtractInPlace(p1),\r\n p1,\r\n p2,\r\n path.tessellation * 2,\r\n false,\r\n true\r\n );\r\n points.push(...curve.getPoints());\r\n }\r\n break;\r\n }\r\n case eSketchPathType.arcThrough3Points: {\r\n if (path.controlPoints.length >= 3) {\r\n const [p0, p1, p2] = path.controlPoints.slice(0, 3).map(cp => this.getPointPosition(cp));\r\n const curve = Curve3.ArcThru3Points(p0, p1, p2, path.tessellation, false, false);\r\n points.push(...curve.getPoints());\r\n }\r\n break;\r\n }\r\n\r\n default: {\r\n path.controlPoints.forEach(point => {\r\n points.push(this.getPointPosition(point));\r\n });\r\n }\r\n }\r\n\r\n // /**\r\n // * We want the sketch to always \"face\" in the positive direction of the axis for consistency\r\n // * Otherwise it's indeterminate which way the sketch will face and makes the sketch hard to manage\r\n // * because it depends on the handed-ness of the points (clockwise or counter-clockwise)\r\n // */\r\n // const primaryNormal = axisToVector3(path._parent.normal);\r\n // const faceNormal = calculateFaceNormal(points);\r\n // if (Vector3.Dot(faceNormal, primaryNormal) < 0) {\r\n // points.reverse();\r\n // }\r\n\r\n return points;\r\n }\r\n\r\n private getPointPosition(point: SketchControlPoint) {\r\n return this.controlPointRenderer.getControlPointAbsPosition(point);\r\n }\r\n\r\n private getPointDirection(point: SketchControlPoint, bb: IBoundingBox) {\r\n return this.controlPointRenderer.getRelativeDirection(point, point.direction, bb);\r\n }\r\n private getPointDirection2(point: SketchControlPoint, bb: IBoundingBox) {\r\n if (!point.direction2) {\r\n return undefined;\r\n } else {\r\n return this.controlPointRenderer.getRelativeDirection(point, point.direction2, bb);\r\n }\r\n }\r\n\r\n private getControlPoints(node: SketchNode) {\r\n const controlPoints: IKbVectorLike[] = [];\r\n node.paths.forEach(path => {\r\n path._points.forEach(point => {\r\n controlPoints.push(point);\r\n });\r\n });\r\n return controlPoints;\r\n }\r\n}\r\n","import type { Nullable } from \"../../types\";\r\nimport { Tools } from \"../../Misc/tools\";\r\nimport { serialize } from \"../../Misc/decorators\";\r\nimport type { Camera } from \"../../Cameras/camera\";\r\nimport type { Engine } from \"../../Engines/engine\";\r\nimport type { PostProcessRenderEffect } from \"./postProcessRenderEffect\";\r\nimport type { IInspectable } from \"../../Misc/iInspectable\";\r\n\r\nimport type { PrePassRenderer } from \"../../Rendering/prePassRenderer\";\r\n\r\n/**\r\n * PostProcessRenderPipeline\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/postProcessRenderPipeline\r\n */\r\nexport class PostProcessRenderPipeline {\r\n protected _renderEffects: { [key: string]: PostProcessRenderEffect };\r\n protected _renderEffectsForIsolatedPass: PostProcessRenderEffect[];\r\n\r\n /**\r\n * List of inspectable custom properties (used by the Inspector)\r\n * @see https://doc.babylonjs.com/toolsAndResources/inspector#extensibility\r\n */\r\n public inspectableCustomProperties: IInspectable[];\r\n\r\n /**\r\n * @internal\r\n */\r\n protected _cameras: Camera[];\r\n\r\n /** @internal */\r\n @serialize()\r\n public _name: string;\r\n\r\n /**\r\n * Gets pipeline name\r\n */\r\n public get name(): string {\r\n return this._name;\r\n }\r\n\r\n /** Gets the list of attached cameras */\r\n public get cameras() {\r\n return this._cameras;\r\n }\r\n\r\n /**\r\n * Initializes a PostProcessRenderPipeline\r\n * @param _engine engine to add the pipeline to\r\n * @param name name of the pipeline\r\n */\r\n constructor(private _engine: Engine, name: string) {\r\n this._name = name;\r\n\r\n this._renderEffects = {};\r\n this._renderEffectsForIsolatedPass = new Array();\r\n\r\n this._cameras = [];\r\n }\r\n\r\n /**\r\n * Gets the class name\r\n * @returns \"PostProcessRenderPipeline\"\r\n */\r\n public getClassName(): string {\r\n return \"PostProcessRenderPipeline\";\r\n }\r\n\r\n /**\r\n * If all the render effects in the pipeline are supported\r\n */\r\n public get isSupported(): boolean {\r\n for (const renderEffectName in this._renderEffects) {\r\n if (Object.prototype.hasOwnProperty.call(this._renderEffects, renderEffectName)) {\r\n if (!this._renderEffects[renderEffectName].isSupported) {\r\n return false;\r\n }\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Adds an effect to the pipeline\r\n * @param renderEffect the effect to add\r\n */\r\n public addEffect(renderEffect: PostProcessRenderEffect): void {\r\n (this._renderEffects)[renderEffect._name] = renderEffect;\r\n }\r\n\r\n // private\r\n\r\n /** @internal */\r\n public _rebuild() {}\r\n\r\n /** @internal */\r\n public _enableEffect(renderEffectName: string, cameras: Camera): void;\r\n /** @internal */\r\n public _enableEffect(renderEffectName: string, cameras: Camera[]): void;\r\n /**\r\n * @internal\r\n */\r\n public _enableEffect(renderEffectName: string, cameras: any): void {\r\n const renderEffects: PostProcessRenderEffect = (this._renderEffects)[renderEffectName];\r\n\r\n if (!renderEffects) {\r\n return;\r\n }\r\n\r\n renderEffects._enable(Tools.MakeArray(cameras || this._cameras));\r\n }\r\n\r\n /** @internal */\r\n public _disableEffect(renderEffectName: string, cameras: Nullable): void;\r\n /** @internal */\r\n public _disableEffect(renderEffectName: string, cameras: Nullable): void;\r\n /**\r\n * @internal\r\n */\r\n public _disableEffect(renderEffectName: string, cameras: Nullable): void {\r\n const renderEffects: PostProcessRenderEffect = (this._renderEffects)[renderEffectName];\r\n\r\n if (!renderEffects) {\r\n return;\r\n }\r\n\r\n renderEffects._disable(Tools.MakeArray(cameras || this._cameras));\r\n }\r\n\r\n /** @internal */\r\n public _attachCameras(cameras: Camera, unique: boolean): void;\r\n /** @internal */\r\n public _attachCameras(cameras: Camera[], unique: boolean): void;\r\n /**\r\n * @internal\r\n */\r\n public _attachCameras(cameras: any, unique: boolean): void {\r\n const cams = Tools.MakeArray(cameras || this._cameras);\r\n\r\n if (!cams) {\r\n return;\r\n }\r\n\r\n const indicesToDelete = [];\r\n let i: number;\r\n for (i = 0; i < cams.length; i++) {\r\n const camera = cams[i];\r\n if (!camera) {\r\n continue;\r\n }\r\n\r\n if (this._cameras.indexOf(camera) === -1) {\r\n this._cameras.push(camera);\r\n } else if (unique) {\r\n indicesToDelete.push(i);\r\n }\r\n }\r\n\r\n for (i = 0; i < indicesToDelete.length; i++) {\r\n cams.splice(indicesToDelete[i], 1);\r\n }\r\n\r\n for (const renderEffectName in this._renderEffects) {\r\n if (Object.prototype.hasOwnProperty.call(this._renderEffects, renderEffectName)) {\r\n this._renderEffects[renderEffectName]._attachCameras(cams);\r\n }\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _detachCameras(cameras: Camera): void;\r\n /** @internal */\r\n public _detachCameras(cameras: Nullable): void;\r\n /**\r\n * @internal\r\n */\r\n public _detachCameras(cameras: any): void {\r\n const cams = Tools.MakeArray(cameras || this._cameras);\r\n\r\n if (!cams) {\r\n return;\r\n }\r\n\r\n for (const renderEffectName in this._renderEffects) {\r\n if (Object.prototype.hasOwnProperty.call(this._renderEffects, renderEffectName)) {\r\n this._renderEffects[renderEffectName]._detachCameras(cams);\r\n }\r\n }\r\n\r\n for (let i = 0; i < cams.length; i++) {\r\n this._cameras.splice(this._cameras.indexOf(cams[i]), 1);\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _update(): void {\r\n for (const renderEffectName in this._renderEffects) {\r\n if (Object.prototype.hasOwnProperty.call(this._renderEffects, renderEffectName)) {\r\n this._renderEffects[renderEffectName]._update();\r\n }\r\n }\r\n\r\n for (let i = 0; i < this._cameras.length; i++) {\r\n if (!this._cameras[i]) {\r\n continue;\r\n }\r\n const cameraName = this._cameras[i].name;\r\n if ((this._renderEffectsForIsolatedPass)[cameraName]) {\r\n (this._renderEffectsForIsolatedPass)[cameraName]._update();\r\n }\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _reset(): void {\r\n this._renderEffects = {};\r\n this._renderEffectsForIsolatedPass = new Array();\r\n }\r\n\r\n protected _enableMSAAOnFirstPostProcess(sampleCount: number): boolean {\r\n if (!this._engine._features.supportMSAA) {\r\n return false;\r\n }\r\n\r\n // Set samples of the very first post process to 4 to enable native anti-aliasing in browsers that support webGL 2.0 (See: https://github.com/BabylonJS/Babylon.js/issues/3754)\r\n const effectKeys = Object.keys(this._renderEffects);\r\n if (effectKeys.length > 0) {\r\n const postProcesses = this._renderEffects[effectKeys[0]].getPostProcesses();\r\n if (postProcesses) {\r\n postProcesses[0].samples = sampleCount;\r\n }\r\n }\r\n return true;\r\n }\r\n\r\n /**\r\n * Sets the required values to the prepass renderer.\r\n * @param prePassRenderer defines the prepass renderer to setup.\r\n * @returns true if the pre pass is needed.\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public setPrePassRenderer(prePassRenderer: PrePassRenderer): boolean {\r\n // Do Nothing by default\r\n return false;\r\n }\r\n\r\n /**\r\n * Disposes of the pipeline\r\n */\r\n public dispose() {\r\n // Must be implemented by children\r\n }\r\n}\r\n","import type { Nullable } from \"../../types\";\r\nimport { Tools } from \"../../Misc/tools\";\r\nimport type { Camera } from \"../../Cameras/camera\";\r\nimport type { PostProcess } from \"../../PostProcesses/postProcess\";\r\nimport type { Engine } from \"../../Engines/engine\";\r\n/**\r\n * This represents a set of one or more post processes in Babylon.\r\n * A post process can be used to apply a shader to a texture after it is rendered.\r\n * @example https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/postProcessRenderPipeline\r\n */\r\nexport class PostProcessRenderEffect {\r\n private _postProcesses: { [Key: string]: Array };\r\n private _getPostProcesses: () => Nullable>;\r\n\r\n private _singleInstance: boolean;\r\n\r\n private _cameras: { [key: string]: Nullable };\r\n private _indicesForCamera: { [key: string]: number[] };\r\n\r\n /**\r\n * Name of the effect\r\n * @internal\r\n */\r\n public _name: string;\r\n\r\n /**\r\n * Instantiates a post process render effect.\r\n * A post process can be used to apply a shader to a texture after it is rendered.\r\n * @param engine The engine the effect is tied to\r\n * @param name The name of the effect\r\n * @param getPostProcesses A function that returns a set of post processes which the effect will run in order to be run.\r\n * @param singleInstance False if this post process can be run on multiple cameras. (default: true)\r\n */\r\n constructor(engine: Engine, name: string, getPostProcesses: () => Nullable>, singleInstance?: boolean) {\r\n this._name = name;\r\n this._singleInstance = singleInstance || true;\r\n\r\n this._getPostProcesses = getPostProcesses;\r\n\r\n this._cameras = {};\r\n this._indicesForCamera = {};\r\n\r\n this._postProcesses = {};\r\n }\r\n\r\n /**\r\n * Checks if all the post processes in the effect are supported.\r\n */\r\n public get isSupported(): boolean {\r\n for (const index in this._postProcesses) {\r\n if (Object.prototype.hasOwnProperty.call(this._postProcesses, index)) {\r\n const pps = this._postProcesses[index];\r\n for (let ppIndex = 0; ppIndex < pps.length; ppIndex++) {\r\n if (!pps[ppIndex].isSupported) {\r\n return false;\r\n }\r\n }\r\n }\r\n }\r\n return true;\r\n }\r\n\r\n /**\r\n * Updates the current state of the effect\r\n * @internal\r\n */\r\n public _update(): void {}\r\n\r\n /**\r\n * Attaches the effect on cameras\r\n * @param cameras The camera to attach to.\r\n * @internal\r\n */\r\n public _attachCameras(cameras: Camera): void;\r\n /**\r\n * Attaches the effect on cameras\r\n * @param cameras The camera to attach to.\r\n * @internal\r\n */\r\n public _attachCameras(cameras: Camera[]): void;\r\n /**\r\n * Attaches the effect on cameras\r\n * @param cameras The camera to attach to.\r\n * @internal\r\n */\r\n public _attachCameras(cameras: any): void {\r\n let cameraKey;\r\n\r\n const cams = Tools.MakeArray(cameras || this._cameras);\r\n\r\n if (!cams) {\r\n return;\r\n }\r\n\r\n for (let i = 0; i < cams.length; i++) {\r\n const camera = cams[i];\r\n if (!camera) {\r\n continue;\r\n }\r\n\r\n const cameraName = camera.name;\r\n\r\n if (this._singleInstance) {\r\n cameraKey = 0;\r\n } else {\r\n cameraKey = cameraName;\r\n }\r\n\r\n if (!this._postProcesses[cameraKey]) {\r\n const postProcess = this._getPostProcesses();\r\n if (postProcess) {\r\n this._postProcesses[cameraKey] = Array.isArray(postProcess) ? postProcess : [postProcess];\r\n }\r\n }\r\n\r\n if (!this._indicesForCamera[cameraName]) {\r\n this._indicesForCamera[cameraName] = [];\r\n }\r\n\r\n this._postProcesses[cameraKey].forEach((postProcess: PostProcess) => {\r\n const index = camera.attachPostProcess(postProcess);\r\n\r\n this._indicesForCamera[cameraName].push(index);\r\n });\r\n\r\n if (!this._cameras[cameraName]) {\r\n this._cameras[cameraName] = camera;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Detaches the effect on cameras\r\n * @param cameras The camera to detach from.\r\n * @internal\r\n */\r\n public _detachCameras(cameras: Camera): void;\r\n /**\r\n * Detaches the effect on cameras\r\n * @param cameras The camera to detach from.\r\n * @internal\r\n */\r\n public _detachCameras(cameras: Camera[]): void;\r\n /**\r\n * Detaches the effect on cameras\r\n * @param cameras The camera to detach from.\r\n * @internal\r\n */\r\n public _detachCameras(cameras: any): void {\r\n const cams = Tools.MakeArray(cameras || this._cameras);\r\n\r\n if (!cams) {\r\n return;\r\n }\r\n\r\n for (let i = 0; i < cams.length; i++) {\r\n const camera: Camera = cams[i];\r\n const cameraName: string = camera.name;\r\n const postProcesses = this._postProcesses[this._singleInstance ? 0 : cameraName];\r\n\r\n if (postProcesses) {\r\n postProcesses.forEach((postProcess: PostProcess) => {\r\n camera.detachPostProcess(postProcess);\r\n });\r\n }\r\n\r\n if (this._cameras[cameraName]) {\r\n this._cameras[cameraName] = null;\r\n }\r\n\r\n delete this._indicesForCamera[cameraName];\r\n }\r\n }\r\n\r\n /**\r\n * Enables the effect on given cameras\r\n * @param cameras The camera to enable.\r\n * @internal\r\n */\r\n public _enable(cameras: Camera): void;\r\n /**\r\n * Enables the effect on given cameras\r\n * @param cameras The camera to enable.\r\n * @internal\r\n */\r\n public _enable(cameras: Nullable): void;\r\n /**\r\n * Enables the effect on given cameras\r\n * @param cameras The camera to enable.\r\n * @internal\r\n */\r\n public _enable(cameras: any): void {\r\n const cams: Nullable> = Tools.MakeArray(cameras || this._cameras);\r\n\r\n if (!cams) {\r\n return;\r\n }\r\n\r\n for (let i = 0; i < cams.length; i++) {\r\n const camera = cams[i];\r\n const cameraName = camera.name;\r\n const cameraKey = this._singleInstance ? 0 : cameraName;\r\n\r\n for (let j = 0; j < this._indicesForCamera[cameraName].length; j++) {\r\n const index = this._indicesForCamera[cameraName][j];\r\n const postProcess = camera._postProcesses[index];\r\n if (postProcess === undefined || postProcess === null) {\r\n cams![i].attachPostProcess(this._postProcesses[cameraKey][j], index);\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Disables the effect on the given cameras\r\n * @param cameras The camera to disable.\r\n * @internal\r\n */\r\n public _disable(cameras: Camera): void;\r\n /**\r\n * Disables the effect on the given cameras\r\n * @param cameras The camera to disable.\r\n * @internal\r\n */\r\n public _disable(cameras: Nullable): void;\r\n /**\r\n * Disables the effect on the given cameras\r\n * @param cameras The camera to disable.\r\n * @internal\r\n */\r\n public _disable(cameras: any): void {\r\n const cams: Nullable> = Tools.MakeArray(cameras || this._cameras);\r\n\r\n if (!cams) {\r\n return;\r\n }\r\n\r\n for (let i = 0; i < cams.length; i++) {\r\n const camera = cams[i];\r\n const cameraName = camera.name;\r\n this._postProcesses[this._singleInstance ? 0 : cameraName].forEach((postProcess) => {\r\n camera.detachPostProcess(postProcess);\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Gets a list of the post processes contained in the effect.\r\n * @param camera The camera to get the post processes on.\r\n * @returns The list of the post processes in the effect.\r\n */\r\n public getPostProcesses(camera?: Camera): Nullable> {\r\n if (this._singleInstance) {\r\n return this._postProcesses[0];\r\n } else {\r\n if (!camera) {\r\n return null;\r\n }\r\n return this._postProcesses[camera.name];\r\n }\r\n }\r\n}\r\n","import type { Camera } from \"../../Cameras/camera\";\r\nimport type { PostProcessRenderPipeline } from \"./postProcessRenderPipeline\";\r\n/**\r\n * PostProcessRenderPipelineManager class\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/postProcessRenderPipeline\r\n */\r\nexport class PostProcessRenderPipelineManager {\r\n private _renderPipelines: { [Key: string]: PostProcessRenderPipeline };\r\n\r\n /**\r\n * Initializes a PostProcessRenderPipelineManager\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/postProcessRenderPipeline\r\n */\r\n constructor() {\r\n this._renderPipelines = {};\r\n }\r\n\r\n /**\r\n * Gets the list of supported render pipelines\r\n */\r\n public get supportedPipelines(): PostProcessRenderPipeline[] {\r\n const result = [];\r\n\r\n for (const renderPipelineName in this._renderPipelines) {\r\n if (Object.prototype.hasOwnProperty.call(this._renderPipelines, renderPipelineName)) {\r\n const pipeline = this._renderPipelines[renderPipelineName];\r\n if (pipeline.isSupported) {\r\n result.push(pipeline);\r\n }\r\n }\r\n }\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Adds a pipeline to the manager\r\n * @param renderPipeline The pipeline to add\r\n */\r\n public addPipeline(renderPipeline: PostProcessRenderPipeline): void {\r\n this._renderPipelines[renderPipeline._name] = renderPipeline;\r\n }\r\n\r\n /**\r\n * Remove the pipeline from the manager\r\n * @param renderPipelineName the name of the pipeline to remove\r\n */\r\n public removePipeline(renderPipelineName: string): void {\r\n delete this._renderPipelines[renderPipelineName];\r\n }\r\n\r\n /**\r\n * Attaches a camera to the pipeline\r\n * @param renderPipelineName The name of the pipeline to attach to\r\n * @param cameras the camera to attach\r\n * @param unique if the camera can be attached multiple times to the pipeline\r\n */\r\n public attachCamerasToRenderPipeline(renderPipelineName: string, cameras: any | Camera[] | Camera, unique: boolean = false): void {\r\n const renderPipeline: PostProcessRenderPipeline = this._renderPipelines[renderPipelineName];\r\n\r\n if (!renderPipeline) {\r\n return;\r\n }\r\n\r\n renderPipeline._attachCameras(cameras, unique);\r\n }\r\n\r\n /**\r\n * Detaches a camera from the pipeline\r\n * @param renderPipelineName The name of the pipeline to detach from\r\n * @param cameras the camera to detach\r\n */\r\n public detachCamerasFromRenderPipeline(renderPipelineName: string, cameras: any | Camera[] | Camera): void {\r\n const renderPipeline: PostProcessRenderPipeline = this._renderPipelines[renderPipelineName];\r\n\r\n if (!renderPipeline) {\r\n return;\r\n }\r\n\r\n renderPipeline._detachCameras(cameras);\r\n }\r\n\r\n /**\r\n * Enables an effect by name on a pipeline\r\n * @param renderPipelineName the name of the pipeline to enable the effect in\r\n * @param renderEffectName the name of the effect to enable\r\n * @param cameras the cameras that the effect should be enabled on\r\n */\r\n public enableEffectInPipeline(renderPipelineName: string, renderEffectName: string, cameras: any | Camera[] | Camera): void {\r\n const renderPipeline: PostProcessRenderPipeline = this._renderPipelines[renderPipelineName];\r\n\r\n if (!renderPipeline) {\r\n return;\r\n }\r\n\r\n renderPipeline._enableEffect(renderEffectName, cameras);\r\n }\r\n\r\n /**\r\n * Disables an effect by name on a pipeline\r\n * @param renderPipelineName the name of the pipeline to disable the effect in\r\n * @param renderEffectName the name of the effect to disable\r\n * @param cameras the cameras that the effect should be disabled on\r\n */\r\n public disableEffectInPipeline(renderPipelineName: string, renderEffectName: string, cameras: any | Camera[] | Camera): void {\r\n const renderPipeline: PostProcessRenderPipeline = this._renderPipelines[renderPipelineName];\r\n\r\n if (!renderPipeline) {\r\n return;\r\n }\r\n\r\n renderPipeline._disableEffect(renderEffectName, cameras);\r\n }\r\n\r\n /**\r\n * Updates the state of all contained render pipelines and disposes of any non supported pipelines\r\n */\r\n public update(): void {\r\n for (const renderPipelineName in this._renderPipelines) {\r\n if (Object.prototype.hasOwnProperty.call(this._renderPipelines, renderPipelineName)) {\r\n const pipeline = this._renderPipelines[renderPipelineName];\r\n if (!pipeline.isSupported) {\r\n pipeline.dispose();\r\n delete this._renderPipelines[renderPipelineName];\r\n } else {\r\n pipeline._update();\r\n }\r\n }\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _rebuild(): void {\r\n for (const renderPipelineName in this._renderPipelines) {\r\n if (Object.prototype.hasOwnProperty.call(this._renderPipelines, renderPipelineName)) {\r\n const pipeline = this._renderPipelines[renderPipelineName];\r\n pipeline._rebuild();\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Disposes of the manager and pipelines\r\n */\r\n public dispose(): void {\r\n for (const renderPipelineName in this._renderPipelines) {\r\n if (Object.prototype.hasOwnProperty.call(this._renderPipelines, renderPipelineName)) {\r\n const pipeline = this._renderPipelines[renderPipelineName];\r\n pipeline.dispose();\r\n }\r\n }\r\n }\r\n}\r\n","import type { ISceneComponent } from \"../../sceneComponent\";\r\nimport { SceneComponentConstants } from \"../../sceneComponent\";\r\nimport { PostProcessRenderPipelineManager } from \"./postProcessRenderPipelineManager\";\r\nimport { Scene } from \"../../scene\";\r\n\r\ndeclare module \"../../scene\" {\r\n export interface Scene {\r\n /** @internal (Backing field) */\r\n _postProcessRenderPipelineManager: PostProcessRenderPipelineManager;\r\n\r\n /**\r\n * Gets the postprocess render pipeline manager\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/postProcessRenderPipeline\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/postProcesses/defaultRenderingPipeline\r\n */\r\n readonly postProcessRenderPipelineManager: PostProcessRenderPipelineManager;\r\n }\r\n}\r\n\r\nObject.defineProperty(Scene.prototype, \"postProcessRenderPipelineManager\", {\r\n get: function (this: Scene) {\r\n if (!this._postProcessRenderPipelineManager) {\r\n // Register the G Buffer component to the scene.\r\n let component = this._getComponent(SceneComponentConstants.NAME_POSTPROCESSRENDERPIPELINEMANAGER) as PostProcessRenderPipelineManagerSceneComponent;\r\n if (!component) {\r\n component = new PostProcessRenderPipelineManagerSceneComponent(this);\r\n this._addComponent(component);\r\n }\r\n this._postProcessRenderPipelineManager = new PostProcessRenderPipelineManager();\r\n }\r\n\r\n return this._postProcessRenderPipelineManager;\r\n },\r\n enumerable: true,\r\n configurable: true,\r\n});\r\n\r\n/**\r\n * Defines the Render Pipeline scene component responsible to rendering pipelines\r\n */\r\nexport class PostProcessRenderPipelineManagerSceneComponent implements ISceneComponent {\r\n /**\r\n * The component name helpful to identify the component in the list of scene components.\r\n */\r\n public readonly name = SceneComponentConstants.NAME_POSTPROCESSRENDERPIPELINEMANAGER;\r\n\r\n /**\r\n * The scene the component belongs to.\r\n */\r\n public scene: Scene;\r\n\r\n /**\r\n * Creates a new instance of the component for the given scene\r\n * @param scene Defines the scene to register the component in\r\n */\r\n constructor(scene: Scene) {\r\n this.scene = scene;\r\n }\r\n\r\n /**\r\n * Registers the component in a given scene\r\n */\r\n public register(): void {\r\n this.scene._gatherRenderTargetsStage.registerStep(SceneComponentConstants.STEP_GATHERRENDERTARGETS_POSTPROCESSRENDERPIPELINEMANAGER, this, this._gatherRenderTargets);\r\n }\r\n\r\n /**\r\n * Rebuilds the elements related to this component in case of\r\n * context lost for instance.\r\n */\r\n public rebuild(): void {\r\n if (this.scene._postProcessRenderPipelineManager) {\r\n this.scene._postProcessRenderPipelineManager._rebuild();\r\n }\r\n }\r\n\r\n /**\r\n * Disposes the component and the associated resources\r\n */\r\n public dispose(): void {\r\n if (this.scene._postProcessRenderPipelineManager) {\r\n this.scene._postProcessRenderPipelineManager.dispose();\r\n }\r\n }\r\n\r\n private _gatherRenderTargets(): void {\r\n if (this.scene._postProcessRenderPipelineManager) {\r\n this.scene._postProcessRenderPipelineManager.update();\r\n }\r\n }\r\n}\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\n\nconst name = \"chromaticAberrationPixelShader\";\nconst shader = `uniform sampler2D textureSampler; \nuniform float chromatic_aberration;uniform float radialIntensity;uniform vec2 direction;uniform vec2 centerPosition;uniform float screen_width;uniform float screen_height;varying vec2 vUV;\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{vec2 centered_screen_pos=vec2(vUV.x-centerPosition.x,vUV.y-centerPosition.y);vec2 directionOfEffect=direction;if(directionOfEffect.x==0. && directionOfEffect.y==0.){directionOfEffect=normalize(centered_screen_pos);}\nfloat radius2=centered_screen_pos.x*centered_screen_pos.x\n+ centered_screen_pos.y*centered_screen_pos.y;float radius=sqrt(radius2);vec4 original=texture2D(textureSampler,vUV);vec3 ref_indices=vec3(-0.3,0.0,0.3);float ref_shiftX=chromatic_aberration*pow(radius,radialIntensity)*directionOfEffect.x/screen_width;float ref_shiftY=chromatic_aberration*pow(radius,radialIntensity)*directionOfEffect.y/screen_height;vec2 ref_coords_r=vec2(vUV.x+ref_indices.r*ref_shiftX,vUV.y+ref_indices.r*ref_shiftY*0.5);vec2 ref_coords_g=vec2(vUV.x+ref_indices.g*ref_shiftX,vUV.y+ref_indices.g*ref_shiftY*0.5);vec2 ref_coords_b=vec2(vUV.x+ref_indices.b*ref_shiftX,vUV.y+ref_indices.b*ref_shiftY*0.5);original.r=texture2D(textureSampler,ref_coords_r).r;original.g=texture2D(textureSampler,ref_coords_g).g;original.b=texture2D(textureSampler,ref_coords_b).b;original.a=clamp(texture2D(textureSampler,ref_coords_r).a+texture2D(textureSampler,ref_coords_g).a+texture2D(textureSampler,ref_coords_b).a,0.,1.);gl_FragColor=original;}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const chromaticAberrationPixelShader = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\n\nconst name = \"lensHighlightsPixelShader\";\nconst shader = `uniform sampler2D textureSampler; \nuniform float gain;uniform float threshold;uniform float screen_width;uniform float screen_height;varying vec2 vUV;vec4 highlightColor(vec4 color) {vec4 highlight=color;float luminance=dot(highlight.rgb,vec3(0.2125,0.7154,0.0721));float lum_threshold;if (threshold>1.0) { lum_threshold=0.94+0.01*threshold; }\nelse { lum_threshold=0.5+0.44*threshold; }\nluminance=clamp((luminance-lum_threshold)*(1.0/(1.0-lum_threshold)),0.0,1.0);highlight*=luminance*gain;highlight.a=1.0;return highlight;}\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{vec4 original=texture2D(textureSampler,vUV);if (gain==-1.0) {gl_FragColor=vec4(0.0,0.0,0.0,1.0);return;}\nfloat w=2.0/screen_width;float h=2.0/screen_height;float weight=1.0;vec4 blurred=vec4(0.0,0.0,0.0,0.0);\n#ifdef PENTAGON\nblurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.84*w,0.43*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.48*w,-1.29*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.61*w,1.51*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.55*w,-0.74*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.71*w,-0.52*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.94*w,1.59*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.40*w,-1.87*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.62*w,1.16*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.09*w,0.25*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.46*w,-1.71*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.08*w,2.42*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.85*w,-1.89*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.89*w,0.16*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.29*w,1.88*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.40*w,-2.81*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.54*w,2.26*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.60*w,-0.61*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.31*w,-1.30*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.83*w,2.53*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.12*w,-2.48*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.60*w,1.11*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.82*w,0.99*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.50*w,-2.81*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.85*w,3.33*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.94*w,-1.92*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(3.27*w,-0.53*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.95*w,2.48*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.23*w,-3.04*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.17*w,2.05*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.97*w,-0.04*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.25*w,-2.00*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.31*w,3.08*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.94*w,-2.59*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(3.37*w,0.64*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-3.13*w,1.93*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.03*w,-3.65*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.60*w,3.17*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-3.14*w,-1.19*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(3.00*w,-1.19*h)));\n#else\nblurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.85*w,0.36*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.52*w,-1.14*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.46*w,1.42*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.46*w,-0.83*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.79*w,-0.42*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.11*w,1.62*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.29*w,-2.07*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.69*w,1.39*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.28*w,0.12*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.65*w,-1.69*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.08*w,2.44*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.63*w,-1.90*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.55*w,0.31*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.13*w,1.52*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.56*w,-2.61*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.38*w,2.34*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.64*w,-0.81*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.53*w,-1.21*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.06*w,2.63*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.00*w,-2.69*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.59*w,1.32*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.82*w,0.78*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.57*w,-2.50*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(0.54*w,2.93*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.39*w,-1.81*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(3.01*w,-0.28*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.04*w,2.25*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.02*w,-3.05*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.09*w,2.25*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-3.07*w,-0.25*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.44*w,-1.90*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-0.52*w,3.05*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-1.68*w,-2.61*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(3.01*w,0.79*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.76*w,1.46*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.05*w,-2.94*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(1.21*w,2.88*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(-2.84*w,-1.30*h)));blurred+=highlightColor(texture2D(textureSampler,vUV+vec2(2.98*w,-0.96*h)));\n#endif\nblurred/=39.0;gl_FragColor=blurred;}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const lensHighlightsPixelShader = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\n\nconst name = \"depthOfFieldPixelShader\";\nconst shader = `uniform sampler2D textureSampler;uniform sampler2D highlightsSampler;uniform sampler2D depthSampler;uniform sampler2D grainSampler;uniform float grain_amount;uniform bool blur_noise;uniform float screen_width;uniform float screen_height;uniform float distortion;uniform bool dof_enabled;uniform float screen_distance; \nuniform float aperture;uniform float darken;uniform float edge_blur;uniform bool highlights;uniform float near;uniform float far;varying vec2 vUV;\n#define PI 3.14159265\n#define TWOPI 6.28318530\n#define inverse_focal_length 0.1 \nvec2 centered_screen_pos;vec2 distorted_coords;float radius2;float radius;vec2 rand(vec2 co)\n{float noise1=(fract(sin(dot(co,vec2(12.9898,78.233)))*43758.5453));float noise2=(fract(sin(dot(co,vec2(12.9898,78.233)*2.0))*43758.5453));return clamp(vec2(noise1,noise2),0.0,1.0);}\nvec2 getDistortedCoords(vec2 coords) {if (distortion==0.0) { return coords; }\nvec2 direction=1.0*normalize(centered_screen_pos);vec2 dist_coords=vec2(0.5,0.5);dist_coords.x=0.5+direction.x*radius2*1.0;dist_coords.y=0.5+direction.y*radius2*1.0;float dist_amount=clamp(distortion*0.23,0.0,1.0);dist_coords=mix(coords,dist_coords,dist_amount);return dist_coords;}\nfloat sampleScreen(inout vec4 color,in vec2 offset,in float weight) {vec2 coords=distorted_coords;float angle=rand(coords*100.0).x*TWOPI;coords+=vec2(offset.x*cos(angle)-offset.y*sin(angle),offset.x*sin(angle)+offset.y*cos(angle));color+=texture2D(textureSampler,coords)*weight;return weight;}\nfloat getBlurLevel(float size) {return min(3.0,ceil(size/1.0));}\nvec4 getBlurColor(float size) {vec4 col=texture2D(textureSampler,distorted_coords);float blur_level=getBlurLevel(size);float w=(size/screen_width);float h=(size/screen_height);float total_weight=1.0;vec2 sample_coords;total_weight+=sampleScreen(col,vec2(-0.50*w,0.24*h),0.93);total_weight+=sampleScreen(col,vec2(0.30*w,-0.75*h),0.90);total_weight+=sampleScreen(col,vec2(0.36*w,0.96*h),0.87);total_weight+=sampleScreen(col,vec2(-1.08*w,-0.55*h),0.85);total_weight+=sampleScreen(col,vec2(1.33*w,-0.37*h),0.83);total_weight+=sampleScreen(col,vec2(-0.82*w,1.31*h),0.80);total_weight+=sampleScreen(col,vec2(-0.31*w,-1.67*h),0.78);total_weight+=sampleScreen(col,vec2(1.47*w,1.11*h),0.76);total_weight+=sampleScreen(col,vec2(-1.97*w,0.19*h),0.74);total_weight+=sampleScreen(col,vec2(1.42*w,-1.57*h),0.72);if (blur_level>1.0) {total_weight+=sampleScreen(col,vec2(0.01*w,2.25*h),0.70);total_weight+=sampleScreen(col,vec2(-1.62*w,-1.74*h),0.67);total_weight+=sampleScreen(col,vec2(2.49*w,0.20*h),0.65);total_weight+=sampleScreen(col,vec2(-2.07*w,1.61*h),0.63);total_weight+=sampleScreen(col,vec2(0.46*w,-2.70*h),0.61);total_weight+=sampleScreen(col,vec2(1.55*w,2.40*h),0.59);total_weight+=sampleScreen(col,vec2(-2.88*w,-0.75*h),0.56);total_weight+=sampleScreen(col,vec2(2.73*w,-1.44*h),0.54);total_weight+=sampleScreen(col,vec2(-1.08*w,3.02*h),0.52);total_weight+=sampleScreen(col,vec2(-1.28*w,-3.05*h),0.49);}\nif (blur_level>2.0) {total_weight+=sampleScreen(col,vec2(3.11*w,1.43*h),0.46);total_weight+=sampleScreen(col,vec2(-3.36*w,1.08*h),0.44);total_weight+=sampleScreen(col,vec2(1.80*w,-3.16*h),0.41);total_weight+=sampleScreen(col,vec2(0.83*w,3.65*h),0.38);total_weight+=sampleScreen(col,vec2(-3.16*w,-2.19*h),0.34);total_weight+=sampleScreen(col,vec2(3.92*w,-0.53*h),0.31);total_weight+=sampleScreen(col,vec2(-2.59*w,3.12*h),0.26);total_weight+=sampleScreen(col,vec2(-0.20*w,-4.15*h),0.22);total_weight+=sampleScreen(col,vec2(3.02*w,3.00*h),0.15);}\ncol/=total_weight; \nif (darken>0.0) {col.rgb*=clamp(0.3,1.0,1.05-size*0.5*darken);}\nreturn col;}\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void)\n{centered_screen_pos=vec2(vUV.x-0.5,vUV.y-0.5);radius2=centered_screen_pos.x*centered_screen_pos.x+centered_screen_pos.y*centered_screen_pos.y;radius=sqrt(radius2);distorted_coords=getDistortedCoords(vUV); \nvec2 texels_coords=vec2(vUV.x*screen_width,vUV.y*screen_height); \nfloat depth=texture2D(depthSampler,distorted_coords).r; \nfloat distance=near+(far-near)*depth; \nvec4 color=texture2D(textureSampler,vUV); \nfloat coc=abs(aperture*(screen_distance*(inverse_focal_length-1.0/distance)-1.0));if (dof_enabled==false || coc<0.07) { coc=0.0; }\nfloat edge_blur_amount=0.0;if (edge_blur>0.0) {edge_blur_amount=clamp((radius*2.0-1.0+0.15*edge_blur)*1.5,0.0,1.0)*1.3;}\nfloat blur_amount=max(edge_blur_amount,coc);if (blur_amount==0.0) {gl_FragColor=texture2D(textureSampler,distorted_coords);}\nelse {gl_FragColor=getBlurColor(blur_amount*1.7);if (highlights) {gl_FragColor.rgb+=clamp(coc,0.0,1.0)*texture2D(highlightsSampler,distorted_coords).rgb;}\nif (blur_noise) {vec2 noise=rand(distorted_coords)*0.01*blur_amount;vec2 blurred_coord=vec2(distorted_coords.x+noise.x,distorted_coords.y+noise.y);gl_FragColor=0.04*texture2D(textureSampler,blurred_coord)+0.96*gl_FragColor;}}\nif (grain_amount>0.0) {vec4 grain_color=texture2D(grainSampler,texels_coords*0.003);gl_FragColor.rgb+=(-0.5+grain_color.rgb)*0.30*grain_amount;}}\n`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const depthOfFieldPixelShader = { name, shader };\n","/* eslint-disable @typescript-eslint/naming-convention */\r\nimport type { Camera } from \"../../../Cameras/camera\";\r\nimport type { Effect } from \"../../../Materials/effect\";\r\nimport { Texture } from \"../../../Materials/Textures/texture\";\r\nimport { DynamicTexture } from \"../../../Materials/Textures/dynamicTexture\";\r\nimport type { RenderTargetTexture } from \"../../../Materials/Textures/renderTargetTexture\";\r\nimport { PostProcess } from \"../../../PostProcesses/postProcess\";\r\nimport { PostProcessRenderPipeline } from \"../../../PostProcesses/RenderPipeline/postProcessRenderPipeline\";\r\nimport { PostProcessRenderEffect } from \"../../../PostProcesses/RenderPipeline/postProcessRenderEffect\";\r\nimport type { Scene } from \"../../../scene\";\r\n\r\nimport \"../../../PostProcesses/RenderPipeline/postProcessRenderPipelineManagerSceneComponent\";\r\n\r\nimport \"../../../Shaders/chromaticAberration.fragment\";\r\nimport \"../../../Shaders/lensHighlights.fragment\";\r\nimport \"../../../Shaders/depthOfField.fragment\";\r\n\r\n/**\r\n * BABYLON.JS Chromatic Aberration GLSL Shader\r\n * Author: Olivier Guyot\r\n * Separates very slightly R, G and B colors on the edges of the screen\r\n * Inspired by Francois Tarlier & Martins Upitis\r\n */\r\nexport class LensRenderingPipeline extends PostProcessRenderPipeline {\r\n // Lens effects can be of the following:\r\n // - chromatic aberration (slight shift of RGB colors)\r\n // - blur on the edge of the lens\r\n // - lens distortion\r\n // - depth-of-field blur & highlights enhancing\r\n // - depth-of-field 'bokeh' effect (shapes appearing in blurred areas)\r\n // - grain effect (noise or custom texture)\r\n\r\n // Two additional texture samplers are needed:\r\n // - depth map (for depth-of-field)\r\n // - grain texture\r\n\r\n /**\r\n * @ignore\r\n * The chromatic aberration PostProcess id in the pipeline\r\n */\r\n public LensChromaticAberrationEffect: string = \"LensChromaticAberrationEffect\";\r\n /**\r\n * @ignore\r\n * The highlights enhancing PostProcess id in the pipeline\r\n */\r\n public HighlightsEnhancingEffect: string = \"HighlightsEnhancingEffect\";\r\n /**\r\n * @ignore\r\n * The depth-of-field PostProcess id in the pipeline\r\n */\r\n public LensDepthOfFieldEffect: string = \"LensDepthOfFieldEffect\";\r\n\r\n private _scene: Scene;\r\n private _depthTexture: RenderTargetTexture;\r\n private _grainTexture: Texture;\r\n\r\n private _chromaticAberrationPostProcess: PostProcess;\r\n private _highlightsPostProcess: PostProcess;\r\n private _depthOfFieldPostProcess: PostProcess;\r\n\r\n private _edgeBlur: number;\r\n private _grainAmount: number;\r\n private _chromaticAberration: number;\r\n private _distortion: number;\r\n private _highlightsGain: number;\r\n private _highlightsThreshold: number;\r\n private _dofDistance: number;\r\n private _dofAperture: number;\r\n private _dofDarken: number;\r\n private _dofPentagon: boolean;\r\n private _blurNoise: boolean;\r\n\r\n /**\r\n * @constructor\r\n *\r\n * Effect parameters are as follow:\r\n * {\r\n * chromatic_aberration: number; // from 0 to x (1 for realism)\r\n * edge_blur: number; // from 0 to x (1 for realism)\r\n * distortion: number; // from 0 to x (1 for realism), note that this will effect the pointer position precision\r\n * grain_amount: number; // from 0 to 1\r\n * grain_texture: BABYLON.Texture; // texture to use for grain effect; if unset, use random B&W noise\r\n * dof_focus_distance: number; // depth-of-field: focus distance; unset to disable (disabled by default)\r\n * dof_aperture: number; // depth-of-field: focus blur bias (default: 1)\r\n * dof_darken: number; // depth-of-field: darken that which is out of focus (from 0 to 1, disabled by default)\r\n * dof_pentagon: boolean; // depth-of-field: makes a pentagon-like \"bokeh\" effect\r\n * dof_gain: number; // depth-of-field: highlights gain; unset to disable (disabled by default)\r\n * dof_threshold: number; // depth-of-field: highlights threshold (default: 1)\r\n * blur_noise: boolean; // add a little bit of noise to the blur (default: true)\r\n * }\r\n * Note: if an effect parameter is unset, effect is disabled\r\n *\r\n * @param name The rendering pipeline name\r\n * @param parameters - An object containing all parameters (see above)\r\n * @param scene The scene linked to this pipeline\r\n * @param ratio The size of the postprocesses (0.5 means that your postprocess will have a width = canvas.width 0.5 and a height = canvas.height 0.5)\r\n * @param cameras The array of cameras that the rendering pipeline will be attached to\r\n */\r\n constructor(name: string, parameters: any, scene: Scene, ratio: number = 1.0, cameras?: Camera[]) {\r\n super(scene.getEngine(), name);\r\n\r\n this._scene = scene;\r\n\r\n // Fetch texture samplers\r\n this._depthTexture = scene.enableDepthRenderer().getDepthMap(); // Force depth renderer \"on\"\r\n if (parameters.grain_texture) {\r\n this._grainTexture = parameters.grain_texture;\r\n } else {\r\n this._createGrainTexture();\r\n }\r\n\r\n // save parameters\r\n this._edgeBlur = parameters.edge_blur ? parameters.edge_blur : 0;\r\n this._grainAmount = parameters.grain_amount ? parameters.grain_amount : 0;\r\n this._chromaticAberration = parameters.chromatic_aberration ? parameters.chromatic_aberration : 0;\r\n this._distortion = parameters.distortion ? parameters.distortion : 0;\r\n this._highlightsGain = parameters.dof_gain !== undefined ? parameters.dof_gain : -1;\r\n this._highlightsThreshold = parameters.dof_threshold ? parameters.dof_threshold : 1;\r\n this._dofDistance = parameters.dof_focus_distance !== undefined ? parameters.dof_focus_distance : -1;\r\n this._dofAperture = parameters.dof_aperture ? parameters.dof_aperture : 1;\r\n this._dofDarken = parameters.dof_darken ? parameters.dof_darken : 0;\r\n this._dofPentagon = parameters.dof_pentagon !== undefined ? parameters.dof_pentagon : true;\r\n this._blurNoise = parameters.blur_noise !== undefined ? parameters.blur_noise : true;\r\n\r\n // Create effects\r\n this._createChromaticAberrationPostProcess(ratio);\r\n this._createHighlightsPostProcess(ratio);\r\n this._createDepthOfFieldPostProcess(ratio / 4);\r\n\r\n // Set up pipeline\r\n this.addEffect(\r\n new PostProcessRenderEffect(\r\n scene.getEngine(),\r\n this.LensChromaticAberrationEffect,\r\n () => {\r\n return this._chromaticAberrationPostProcess;\r\n },\r\n true\r\n )\r\n );\r\n this.addEffect(\r\n new PostProcessRenderEffect(\r\n scene.getEngine(),\r\n this.HighlightsEnhancingEffect,\r\n () => {\r\n return this._highlightsPostProcess;\r\n },\r\n true\r\n )\r\n );\r\n this.addEffect(\r\n new PostProcessRenderEffect(\r\n scene.getEngine(),\r\n this.LensDepthOfFieldEffect,\r\n () => {\r\n return this._depthOfFieldPostProcess;\r\n },\r\n true\r\n )\r\n );\r\n\r\n if (this._highlightsGain === -1) {\r\n this._disableEffect(this.HighlightsEnhancingEffect, null);\r\n }\r\n\r\n // Finish\r\n scene.postProcessRenderPipelineManager.addPipeline(this);\r\n if (cameras) {\r\n scene.postProcessRenderPipelineManager.attachCamerasToRenderPipeline(name, cameras);\r\n }\r\n }\r\n\r\n /**\r\n * Get the class name\r\n * @returns \"LensRenderingPipeline\"\r\n */\r\n public getClassName(): string {\r\n return \"LensRenderingPipeline\";\r\n }\r\n\r\n // Properties\r\n\r\n /**\r\n * Gets associated scene\r\n */\r\n public get scene(): Scene {\r\n return this._scene;\r\n }\r\n\r\n /**\r\n * Gets or sets the edge blur\r\n */\r\n public get edgeBlur(): number {\r\n return this._edgeBlur;\r\n }\r\n\r\n public set edgeBlur(value: number) {\r\n this.setEdgeBlur(value);\r\n }\r\n\r\n /**\r\n * Gets or sets the grain amount\r\n */\r\n public get grainAmount(): number {\r\n return this._grainAmount;\r\n }\r\n\r\n public set grainAmount(value: number) {\r\n this.setGrainAmount(value);\r\n }\r\n\r\n /**\r\n * Gets or sets the chromatic aberration amount\r\n */\r\n public get chromaticAberration(): number {\r\n return this._chromaticAberration;\r\n }\r\n\r\n public set chromaticAberration(value: number) {\r\n this.setChromaticAberration(value);\r\n }\r\n\r\n /**\r\n * Gets or sets the depth of field aperture\r\n */\r\n public get dofAperture(): number {\r\n return this._dofAperture;\r\n }\r\n\r\n public set dofAperture(value: number) {\r\n this.setAperture(value);\r\n }\r\n\r\n /**\r\n * Gets or sets the edge distortion\r\n */\r\n public get edgeDistortion(): number {\r\n return this._distortion;\r\n }\r\n\r\n public set edgeDistortion(value: number) {\r\n this.setEdgeDistortion(value);\r\n }\r\n\r\n /**\r\n * Gets or sets the depth of field distortion\r\n */\r\n public get dofDistortion(): number {\r\n return this._dofDistance;\r\n }\r\n\r\n public set dofDistortion(value: number) {\r\n this.setFocusDistance(value);\r\n }\r\n\r\n /**\r\n * Gets or sets the darken out of focus amount\r\n */\r\n public get darkenOutOfFocus(): number {\r\n return this._dofDarken;\r\n }\r\n\r\n public set darkenOutOfFocus(value: number) {\r\n this.setDarkenOutOfFocus(value);\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating if blur noise is enabled\r\n */\r\n public get blurNoise(): boolean {\r\n return this._blurNoise;\r\n }\r\n\r\n public set blurNoise(value: boolean) {\r\n this._blurNoise = value;\r\n }\r\n\r\n /**\r\n * Gets or sets a boolean indicating if pentagon bokeh is enabled\r\n */\r\n public get pentagonBokeh(): boolean {\r\n return this._pentagonBokehIsEnabled;\r\n }\r\n\r\n public set pentagonBokeh(value: boolean) {\r\n if (value) {\r\n this.enablePentagonBokeh();\r\n } else {\r\n this.disablePentagonBokeh();\r\n }\r\n }\r\n\r\n /**\r\n * Gets or sets the highlight grain amount\r\n */\r\n public get highlightsGain(): number {\r\n return this._highlightsGain;\r\n }\r\n\r\n public set highlightsGain(value: number) {\r\n this.setHighlightsGain(value);\r\n }\r\n\r\n /**\r\n * Gets or sets the highlight threshold\r\n */\r\n public get highlightsThreshold(): number {\r\n return this._highlightsThreshold;\r\n }\r\n\r\n public set highlightsThreshold(value: number) {\r\n this.setHighlightsThreshold(value);\r\n }\r\n\r\n // public methods (self explanatory)\r\n /**\r\n * Sets the amount of blur at the edges\r\n * @param amount blur amount\r\n */\r\n public setEdgeBlur(amount: number) {\r\n this._edgeBlur = amount;\r\n }\r\n /**\r\n * Sets edge blur to 0\r\n */\r\n public disableEdgeBlur() {\r\n this._edgeBlur = 0;\r\n }\r\n /**\r\n * Sets the amount of grain\r\n * @param amount Amount of grain\r\n */\r\n public setGrainAmount(amount: number) {\r\n this._grainAmount = amount;\r\n }\r\n /**\r\n * Set grain amount to 0\r\n */\r\n public disableGrain() {\r\n this._grainAmount = 0;\r\n }\r\n /**\r\n * Sets the chromatic aberration amount\r\n * @param amount amount of chromatic aberration\r\n */\r\n public setChromaticAberration(amount: number) {\r\n this._chromaticAberration = amount;\r\n }\r\n /**\r\n * Sets chromatic aberration amount to 0\r\n */\r\n public disableChromaticAberration() {\r\n this._chromaticAberration = 0;\r\n }\r\n /**\r\n * Sets the EdgeDistortion amount\r\n * @param amount amount of EdgeDistortion\r\n */\r\n public setEdgeDistortion(amount: number) {\r\n this._distortion = amount;\r\n }\r\n /**\r\n * Sets edge distortion to 0\r\n */\r\n public disableEdgeDistortion() {\r\n this._distortion = 0;\r\n }\r\n /**\r\n * Sets the FocusDistance amount\r\n * @param amount amount of FocusDistance\r\n */\r\n public setFocusDistance(amount: number) {\r\n this._dofDistance = amount;\r\n }\r\n /**\r\n * Disables depth of field\r\n */\r\n public disableDepthOfField() {\r\n this._dofDistance = -1;\r\n }\r\n /**\r\n * Sets the Aperture amount\r\n * @param amount amount of Aperture\r\n */\r\n public setAperture(amount: number) {\r\n this._dofAperture = amount;\r\n }\r\n /**\r\n * Sets the DarkenOutOfFocus amount\r\n * @param amount amount of DarkenOutOfFocus\r\n */\r\n public setDarkenOutOfFocus(amount: number) {\r\n this._dofDarken = amount;\r\n }\r\n\r\n private _pentagonBokehIsEnabled = false;\r\n /**\r\n * Creates a pentagon bokeh effect\r\n */\r\n public enablePentagonBokeh() {\r\n this._highlightsPostProcess.updateEffect(\"#define PENTAGON\\n\");\r\n this._pentagonBokehIsEnabled = true;\r\n }\r\n /**\r\n * Disables the pentagon bokeh effect\r\n */\r\n public disablePentagonBokeh() {\r\n this._pentagonBokehIsEnabled = false;\r\n this._highlightsPostProcess.updateEffect();\r\n }\r\n /**\r\n * Enables noise blur\r\n */\r\n public enableNoiseBlur() {\r\n this._blurNoise = true;\r\n }\r\n /**\r\n * Disables noise blur\r\n */\r\n public disableNoiseBlur() {\r\n this._blurNoise = false;\r\n }\r\n /**\r\n * Sets the HighlightsGain amount\r\n * @param amount amount of HighlightsGain\r\n */\r\n public setHighlightsGain(amount: number) {\r\n this._highlightsGain = amount;\r\n }\r\n /**\r\n * Sets the HighlightsThreshold amount\r\n * @param amount amount of HighlightsThreshold\r\n */\r\n public setHighlightsThreshold(amount: number) {\r\n if (this._highlightsGain === -1) {\r\n this._highlightsGain = 1.0;\r\n }\r\n this._highlightsThreshold = amount;\r\n }\r\n /**\r\n * Disables highlights\r\n */\r\n public disableHighlights() {\r\n this._highlightsGain = -1;\r\n }\r\n\r\n /**\r\n * Removes the internal pipeline assets and detaches the pipeline from the scene cameras\r\n * @param disableDepthRender If the scene's depth rendering should be disabled (default: false)\r\n */\r\n public dispose(disableDepthRender: boolean = false): void {\r\n this._scene.postProcessRenderPipelineManager.detachCamerasFromRenderPipeline(this._name, this._scene.cameras);\r\n\r\n (this._chromaticAberrationPostProcess) = null;\r\n (this._highlightsPostProcess) = null;\r\n (this._depthOfFieldPostProcess) = null;\r\n\r\n this._grainTexture.dispose();\r\n\r\n if (disableDepthRender) {\r\n this._scene.disableDepthRenderer();\r\n }\r\n }\r\n\r\n // colors shifting and distortion\r\n private _createChromaticAberrationPostProcess(ratio: number): void {\r\n this._chromaticAberrationPostProcess = new PostProcess(\r\n \"LensChromaticAberration\",\r\n \"chromaticAberration\",\r\n [\"chromatic_aberration\", \"screen_width\", \"screen_height\", \"direction\", \"radialIntensity\", \"centerPosition\"], // uniforms\r\n [], // samplers\r\n ratio,\r\n null,\r\n Texture.TRILINEAR_SAMPLINGMODE,\r\n this._scene.getEngine(),\r\n false\r\n );\r\n\r\n this._chromaticAberrationPostProcess.onApply = (effect: Effect) => {\r\n effect.setFloat(\"chromatic_aberration\", this._chromaticAberration);\r\n effect.setFloat(\"screen_width\", this._scene.getEngine().getRenderWidth());\r\n effect.setFloat(\"screen_height\", this._scene.getEngine().getRenderHeight());\r\n effect.setFloat(\"radialIntensity\", 1);\r\n effect.setFloat2(\"direction\", 17, 17);\r\n effect.setFloat2(\"centerPosition\", 0.5, 0.5);\r\n };\r\n }\r\n\r\n // highlights enhancing\r\n private _createHighlightsPostProcess(ratio: number): void {\r\n this._highlightsPostProcess = new PostProcess(\r\n \"LensHighlights\",\r\n \"lensHighlights\",\r\n [\"gain\", \"threshold\", \"screen_width\", \"screen_height\"], // uniforms\r\n [], // samplers\r\n ratio,\r\n null,\r\n Texture.TRILINEAR_SAMPLINGMODE,\r\n this._scene.getEngine(),\r\n false,\r\n this._dofPentagon ? \"#define PENTAGON\\n\" : \"\"\r\n );\r\n\r\n this._highlightsPostProcess.externalTextureSamplerBinding = true;\r\n this._highlightsPostProcess.onApply = (effect: Effect) => {\r\n effect.setFloat(\"gain\", this._highlightsGain);\r\n effect.setFloat(\"threshold\", this._highlightsThreshold);\r\n effect.setTextureFromPostProcess(\"textureSampler\", this._chromaticAberrationPostProcess);\r\n effect.setFloat(\"screen_width\", this._scene.getEngine().getRenderWidth());\r\n effect.setFloat(\"screen_height\", this._scene.getEngine().getRenderHeight());\r\n };\r\n }\r\n\r\n // colors shifting and distortion\r\n private _createDepthOfFieldPostProcess(ratio: number): void {\r\n this._depthOfFieldPostProcess = new PostProcess(\r\n \"LensDepthOfField\",\r\n \"depthOfField\",\r\n [\r\n \"grain_amount\",\r\n \"blur_noise\",\r\n \"screen_width\",\r\n \"screen_height\",\r\n \"distortion\",\r\n \"dof_enabled\",\r\n \"screen_distance\",\r\n \"aperture\",\r\n \"darken\",\r\n \"edge_blur\",\r\n \"highlights\",\r\n \"near\",\r\n \"far\",\r\n ],\r\n [\"depthSampler\", \"grainSampler\", \"highlightsSampler\"],\r\n ratio,\r\n null,\r\n Texture.TRILINEAR_SAMPLINGMODE,\r\n this._scene.getEngine(),\r\n false\r\n );\r\n\r\n this._depthOfFieldPostProcess.externalTextureSamplerBinding = true;\r\n this._depthOfFieldPostProcess.onApply = (effect: Effect) => {\r\n effect.setTexture(\"depthSampler\", this._depthTexture);\r\n effect.setTexture(\"grainSampler\", this._grainTexture);\r\n effect.setTextureFromPostProcess(\"textureSampler\", this._highlightsPostProcess);\r\n effect.setTextureFromPostProcess(\"highlightsSampler\", this._depthOfFieldPostProcess);\r\n\r\n effect.setFloat(\"grain_amount\", this._grainAmount);\r\n effect.setBool(\"blur_noise\", this._blurNoise);\r\n\r\n effect.setFloat(\"screen_width\", this._scene.getEngine().getRenderWidth());\r\n effect.setFloat(\"screen_height\", this._scene.getEngine().getRenderHeight());\r\n\r\n effect.setFloat(\"distortion\", this._distortion);\r\n\r\n effect.setBool(\"dof_enabled\", this._dofDistance !== -1);\r\n effect.setFloat(\"screen_distance\", 1.0 / (0.1 - 1.0 / this._dofDistance));\r\n effect.setFloat(\"aperture\", this._dofAperture);\r\n effect.setFloat(\"darken\", this._dofDarken);\r\n\r\n effect.setFloat(\"edge_blur\", this._edgeBlur);\r\n\r\n effect.setBool(\"highlights\", this._highlightsGain !== -1);\r\n\r\n if (this._scene.activeCamera) {\r\n effect.setFloat(\"near\", this._scene.activeCamera.minZ);\r\n effect.setFloat(\"far\", this._scene.activeCamera.maxZ);\r\n }\r\n };\r\n }\r\n\r\n // creates a black and white random noise texture, 512x512\r\n private _createGrainTexture(): void {\r\n const size = 512;\r\n\r\n this._grainTexture = new DynamicTexture(\"LensNoiseTexture\", size, this._scene, false, Texture.BILINEAR_SAMPLINGMODE);\r\n this._grainTexture.wrapU = Texture.WRAP_ADDRESSMODE;\r\n this._grainTexture.wrapV = Texture.WRAP_ADDRESSMODE;\r\n\r\n const context = (this._grainTexture).getContext();\r\n\r\n const rand = (min: number, max: number) => {\r\n return Math.random() * (max - min) + min;\r\n };\r\n\r\n let value;\r\n for (let x = 0; x < size; x++) {\r\n for (let y = 0; y < size; y++) {\r\n value = Math.floor(rand(0.42, 0.58) * 255);\r\n context.fillStyle = \"rgb(\" + value + \", \" + value + \", \" + value + \")\";\r\n context.fillRect(x, y, 1, 1);\r\n }\r\n }\r\n (this._grainTexture).update(false);\r\n }\r\n}\r\n","import { Constants } from \"../Engines/constants\";\r\nimport type { PrePassEffectConfiguration } from \"./prePassEffectConfiguration\";\r\n\r\n/**\r\n * Contains all parameters needed for the prepass to perform\r\n * screen space subsurface scattering\r\n */\r\nexport class SSAO2Configuration implements PrePassEffectConfiguration {\r\n /**\r\n * Is subsurface enabled\r\n */\r\n public enabled = false;\r\n\r\n /**\r\n * Name of the configuration\r\n */\r\n public name = \"ssao2\";\r\n\r\n /**\r\n * Textures that should be present in the MRT for this effect to work\r\n */\r\n public readonly texturesRequired: number[] = [Constants.PREPASS_NORMAL_TEXTURE_TYPE, Constants.PREPASS_DEPTH_TEXTURE_TYPE];\r\n}\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\n\nconst name = \"ssao2PixelShader\";\nconst shader = `precision highp float;uniform sampler2D textureSampler;varying vec2 vUV;\n#ifdef SSAO\nfloat scales[16]=float[16](\n0.1,\n0.11406250000000001,\n0.131640625,\n0.15625,\n0.187890625,\n0.2265625,\n0.272265625,\n0.325,\n0.384765625,\n0.4515625,\n0.525390625,\n0.60625,\n0.694140625,\n0.7890625,\n0.891015625,\n1.0\n);uniform float near;uniform float radius;uniform sampler2D depthSampler;uniform sampler2D randomSampler;uniform sampler2D normalSampler;uniform float randTextureTiles;uniform float samplesFactor;uniform vec3 sampleSphere[SAMPLES];uniform float totalStrength;uniform float base;uniform float xViewport;uniform float yViewport;uniform mat3 depthProjection;uniform float maxZ;uniform float minZAspect;uniform vec2 texelSize;uniform mat4 projection;void main()\n{vec3 random=textureLod(randomSampler,vUV*randTextureTiles,0.0).rgb;float depth=textureLod(depthSampler,vUV,0.0).r;float depthSign=depth/abs(depth);depth=depth*depthSign;vec3 normal=textureLod(normalSampler,vUV,0.0).rgb;float occlusion=0.0;float correctedRadius=min(radius,minZAspect*depth/near);vec3 vViewRay=vec3((vUV.x*2.0-1.0)*xViewport,(vUV.y*2.0-1.0)*yViewport,depthSign);vec3 vDepthFactor=depthProjection*vec3(1.0,1.0,depth);vec3 origin=vViewRay*vDepthFactor;vec3 rvec=random*2.0-1.0;rvec.z=0.0;float dotProduct=dot(rvec,normal);rvec=1.0-abs(dotProduct)>1e-2 ? rvec : vec3(-rvec.y,0.0,rvec.x);vec3 tangent=normalize(rvec-normal*dot(rvec,normal));vec3 bitangent=cross(normal,tangent);mat3 tbn=mat3(tangent,bitangent,normal);float difference;for (int i=0; i1.0 || offset.y>1.0) {continue;}\nfloat sampleDepth=abs(textureLod(depthSampler,offset.xy,0.0).r);difference=depthSign*samplePosition.z-sampleDepth;float rangeCheck=1.0-smoothstep(correctedRadius*0.5,correctedRadius,difference);occlusion+=step(EPSILON,difference)*rangeCheck;}\nocclusion=occlusion*(1.0-smoothstep(maxZ*0.75,maxZ,depth));float ao=1.0-totalStrength*occlusion*samplesFactor;float result=clamp(ao+base,0.0,1.0);gl_FragColor=vec4(vec3(result),1.0);}\n#endif\n#ifdef BLUR\nuniform float outSize;uniform float soften;uniform float tolerance;uniform int samples;\n#ifndef BLUR_BYPASS\nuniform sampler2D depthSampler;\n#ifdef BLUR_LEGACY\n#define inline\nfloat blur13Bilateral(sampler2D image,vec2 uv,vec2 step) {float result=0.0;vec2 off1=vec2(1.411764705882353)*step;vec2 off2=vec2(3.2941176470588234)*step;vec2 off3=vec2(5.176470588235294)*step;float compareDepth=abs(textureLod(depthSampler,uv,0.0).r);float sampleDepth;float weight;float weightSum=30.0;result+=textureLod(image,uv,0.0).r*30.0;sampleDepth=abs(textureLod(depthSampler,uv+off1,0.0).r);weight=clamp(1.0/( 0.003+abs(compareDepth-sampleDepth)),0.0,30.0);weightSum+= weight;result+=textureLod(image,uv+off1,0.0).r*weight;sampleDepth=abs(textureLod(depthSampler,uv-off1,0.0).r);weight=clamp(1.0/( 0.003+abs(compareDepth-sampleDepth)),0.0,30.0);weightSum+= weight;result+=textureLod(image,uv-off1,0.0).r*weight;sampleDepth=abs(textureLod(depthSampler,uv+off2,0.0).r);weight=clamp(1.0/( 0.003+abs(compareDepth-sampleDepth)),0.0,30.0);weightSum+=weight;result+=textureLod(image,uv+off2,0.0).r*weight;sampleDepth=abs(textureLod(depthSampler,uv-off2,0.0).r);weight=clamp(1.0/( 0.003+abs(compareDepth-sampleDepth)),0.0,30.0);weightSum+=weight;result+=textureLod(image,uv-off2,0.0).r*weight;sampleDepth=abs(textureLod(depthSampler,uv+off3,0.0).r);weight=clamp(1.0/( 0.003+abs(compareDepth-sampleDepth)),0.0,30.0);weightSum+=weight;result+=textureLod(image,uv+off3,0.0).r*weight;sampleDepth=abs(textureLod(depthSampler,uv-off3,0.0).r);weight=clamp(1.0/( 0.003+abs(compareDepth-sampleDepth)),0.0,30.0);weightSum+=weight;result+=textureLod(image,uv-off3,0.0).r*weight;return result/weightSum;}\n#endif\n#endif\nvoid main()\n{float result=0.0;\n#ifdef BLUR_BYPASS\nresult=textureLod(textureSampler,vUV,0.0).r;\n#else\n#ifdef BLUR_H\nvec2 step=vec2(1.0/outSize,0.0);\n#else\nvec2 step=vec2(0.0,1.0/outSize);\n#endif\n#ifdef BLUR_LEGACY\nresult=blur13Bilateral(textureSampler,vUV,step);\n#else\nfloat compareDepth=abs(textureLod(depthSampler,vUV,0.0).r);float weightSum=0.0;for (int i=-samples; i {\r\n if (!this._forceGeometryBuffer) {\r\n return null;\r\n }\r\n return this._scene.geometryBufferRenderer;\r\n }\r\n private get _prePassRenderer(): Nullable {\r\n if (this._forceGeometryBuffer) {\r\n return null;\r\n }\r\n return this._scene.prePassRenderer;\r\n }\r\n\r\n /**\r\n * Ratio object used for SSAO ratio and blur ratio\r\n */\r\n @serialize()\r\n private _ratio: any;\r\n\r\n /*\r\n * The texture type used by the different post processes created by SSAO\r\n */\r\n @serialize()\r\n private _textureType: number;\r\n\r\n /**\r\n * Dynamically generated sphere sampler.\r\n */\r\n private _sampleSphere: number[];\r\n\r\n /**\r\n * The radius around the analyzed pixel used by the SSAO post-process. Default value is 2.0\r\n */\r\n @serialize()\r\n public radius: number = 2.0;\r\n\r\n /**\r\n * The base color of the SSAO post-process\r\n * The final result is \"base + ssao\" between [0, 1]\r\n */\r\n @serialize()\r\n public base: number = 0;\r\n\r\n @serialize(\"bypassBlur\")\r\n private _bypassBlur: boolean = false;\r\n /**\r\n * Skips the denoising (blur) stage of the SSAO calculations.\r\n *\r\n * Useful to temporarily set while experimenting with the other SSAO2 settings.\r\n */\r\n public set bypassBlur(b: boolean) {\r\n const defines = this._getDefinesForBlur(this.expensiveBlur, b);\r\n const samplers = this._getSamplersForBlur(b);\r\n this._blurHPostProcess.updateEffect(defines.h, null, samplers);\r\n this._blurVPostProcess.updateEffect(defines.v, null, samplers);\r\n this._bypassBlur = b;\r\n }\r\n public get bypassBlur(): boolean {\r\n return this._bypassBlur;\r\n }\r\n\r\n @serialize(\"expensiveBlur\")\r\n private _expensiveBlur: boolean = true;\r\n /**\r\n * Enables the configurable bilateral denoising (blurring) filter. Default is true.\r\n * Set to false to instead use a legacy bilateral filter that can't be configured.\r\n *\r\n * The denoising filter runs after the SSAO calculations and is a very important step. Both options results in a so called bilateral being used, but the \"expensive\" one can be\r\n * configured in several ways to fit your scene.\r\n */\r\n public set expensiveBlur(b: boolean) {\r\n const defines = this._getDefinesForBlur(b, this._bypassBlur);\r\n this._blurHPostProcess.updateEffect(defines.h);\r\n this._blurVPostProcess.updateEffect(defines.v);\r\n this._expensiveBlur = b;\r\n }\r\n public get expensiveBlur(): boolean {\r\n return this._expensiveBlur;\r\n }\r\n\r\n /**\r\n * The number of samples the bilateral filter uses in both dimensions when denoising the SSAO calculations. Default value is 16.\r\n *\r\n * A higher value should result in smoother shadows but will use more processing time in the shaders.\r\n *\r\n * A high value can cause the shadows to get to blurry or create visible artifacts (bands) near sharp details in the geometry. The artifacts can sometimes be mitigated by increasing the bilateralSoften setting.\r\n */\r\n @serialize()\r\n public bilateralSamples: number = 16;\r\n\r\n /**\r\n * Controls the shape of the denoising kernel used by the bilateral filter. Default value is 0.\r\n *\r\n * By default the bilateral filter acts like a box-filter, treating all samples on the same depth with equal weights. This is effective to maximize the denoising effect given a limited set of samples. However, it also often results in visible ghosting around sharp shadow regions and can spread out lines over large areas so they are no longer visible.\r\n *\r\n * Increasing this setting will make the filter pay less attention to samples further away from the center sample, reducing many artifacts but at the same time increasing noise.\r\n *\r\n * Useful value range is [0..1].\r\n */\r\n @serialize()\r\n public bilateralSoften: number = 0;\r\n\r\n /**\r\n * How forgiving the bilateral denoiser should be when rejecting samples. Default value is 0.\r\n *\r\n * A higher value results in the bilateral filter being more forgiving and thus doing a better job at denoising slanted and curved surfaces, but can lead to shadows spreading out around corners or between objects that are close to each other depth wise.\r\n *\r\n * Useful value range is normally [0..1], but higher values are allowed.\r\n */\r\n @serialize()\r\n public bilateralTolerance: number = 0;\r\n\r\n /**\r\n * Support test.\r\n */\r\n public static get IsSupported(): boolean {\r\n const engine = EngineStore.LastCreatedEngine;\r\n if (!engine) {\r\n return false;\r\n }\r\n return engine._features.supportSSAO2;\r\n }\r\n\r\n private _scene: Scene;\r\n private _randomTexture: DynamicTexture;\r\n private _originalColorPostProcess: PassPostProcess;\r\n private _ssaoPostProcess: PostProcess;\r\n private _blurHPostProcess: PostProcess;\r\n private _blurVPostProcess: PostProcess;\r\n private _ssaoCombinePostProcess: PostProcess;\r\n\r\n /**\r\n * Gets active scene\r\n */\r\n public get scene(): Scene {\r\n return this._scene;\r\n }\r\n\r\n /**\r\n * @constructor\r\n * @param name The rendering pipeline name\r\n * @param scene The scene linked to this pipeline\r\n * @param ratio The size of the postprocesses. Can be a number shared between passes or an object for more precision: { ssaoRatio: 0.5, blurRatio: 1.0 }\r\n * @param cameras The array of cameras that the rendering pipeline will be attached to\r\n * @param forceGeometryBuffer Set to true if you want to use the legacy geometry buffer renderer\r\n * @param textureType The texture type used by the different post processes created by SSAO (default: Constants.TEXTURETYPE_UNSIGNED_INT)\r\n */\r\n constructor(name: string, scene: Scene, ratio: any, cameras?: Camera[], forceGeometryBuffer = false, textureType = Constants.TEXTURETYPE_UNSIGNED_INT) {\r\n super(scene.getEngine(), name);\r\n\r\n this._scene = scene;\r\n this._ratio = ratio;\r\n this._textureType = textureType;\r\n this._forceGeometryBuffer = forceGeometryBuffer;\r\n\r\n if (!this.isSupported) {\r\n Logger.Error(\"The current engine does not support SSAO 2.\");\r\n return;\r\n }\r\n\r\n const ssaoRatio = this._ratio.ssaoRatio || ratio;\r\n const blurRatio = this._ratio.blurRatio || ratio;\r\n\r\n // Set up assets\r\n if (this._forceGeometryBuffer) {\r\n scene.enableGeometryBufferRenderer();\r\n } else {\r\n scene.enablePrePassRenderer();\r\n }\r\n\r\n this._createRandomTexture();\r\n\r\n this._originalColorPostProcess = new PassPostProcess(\"SSAOOriginalSceneColor\", 1.0, null, Texture.BILINEAR_SAMPLINGMODE, scene.getEngine(), undefined, this._textureType);\r\n this._originalColorPostProcess.samples = this.textureSamples;\r\n this._createSSAOPostProcess(1.0, textureType);\r\n this._createBlurPostProcess(ssaoRatio, blurRatio, this._textureType);\r\n this._createSSAOCombinePostProcess(blurRatio, this._textureType);\r\n\r\n // Set up pipeline\r\n this.addEffect(\r\n new PostProcessRenderEffect(\r\n scene.getEngine(),\r\n this.SSAOOriginalSceneColorEffect,\r\n () => {\r\n return this._originalColorPostProcess;\r\n },\r\n true\r\n )\r\n );\r\n this.addEffect(\r\n new PostProcessRenderEffect(\r\n scene.getEngine(),\r\n this.SSAORenderEffect,\r\n () => {\r\n return this._ssaoPostProcess;\r\n },\r\n true\r\n )\r\n );\r\n this.addEffect(\r\n new PostProcessRenderEffect(\r\n scene.getEngine(),\r\n this.SSAOBlurHRenderEffect,\r\n () => {\r\n return this._blurHPostProcess;\r\n },\r\n true\r\n )\r\n );\r\n this.addEffect(\r\n new PostProcessRenderEffect(\r\n scene.getEngine(),\r\n this.SSAOBlurVRenderEffect,\r\n () => {\r\n return this._blurVPostProcess;\r\n },\r\n true\r\n )\r\n );\r\n this.addEffect(\r\n new PostProcessRenderEffect(\r\n scene.getEngine(),\r\n this.SSAOCombineRenderEffect,\r\n () => {\r\n return this._ssaoCombinePostProcess;\r\n },\r\n true\r\n )\r\n );\r\n\r\n // Finish\r\n scene.postProcessRenderPipelineManager.addPipeline(this);\r\n if (cameras) {\r\n scene.postProcessRenderPipelineManager.attachCamerasToRenderPipeline(name, cameras);\r\n }\r\n }\r\n\r\n // Public Methods\r\n\r\n /**\r\n * Get the class name\r\n * @returns \"SSAO2RenderingPipeline\"\r\n */\r\n public getClassName(): string {\r\n return \"SSAO2RenderingPipeline\";\r\n }\r\n\r\n /**\r\n * Removes the internal pipeline assets and detaches the pipeline from the scene cameras\r\n * @param disableGeometryBufferRenderer\r\n */\r\n public dispose(disableGeometryBufferRenderer: boolean = false): void {\r\n for (let i = 0; i < this._scene.cameras.length; i++) {\r\n const camera = this._scene.cameras[i];\r\n\r\n this._originalColorPostProcess.dispose(camera);\r\n this._ssaoPostProcess.dispose(camera);\r\n this._blurHPostProcess.dispose(camera);\r\n this._blurVPostProcess.dispose(camera);\r\n this._ssaoCombinePostProcess.dispose(camera);\r\n }\r\n\r\n this._randomTexture.dispose();\r\n\r\n if (disableGeometryBufferRenderer) {\r\n this._scene.disableGeometryBufferRenderer();\r\n }\r\n\r\n this._scene.postProcessRenderPipelineManager.detachCamerasFromRenderPipeline(this._name, this._scene.cameras);\r\n\r\n super.dispose();\r\n }\r\n\r\n // Private Methods\r\n\r\n /** @internal */\r\n public _rebuild() {\r\n super._rebuild();\r\n }\r\n\r\n private _getSamplersForBlur(disabled: boolean): Array {\r\n return disabled ? [\"textureSampler\"] : [\"textureSampler\", \"depthSampler\"];\r\n }\r\n\r\n private _getDefinesForBlur(bilateral: boolean, disabled: boolean): { h: string; v: string } {\r\n let define = \"#define BLUR\\n\";\r\n if (disabled) {\r\n define += \"#define BLUR_BYPASS\\n\";\r\n }\r\n if (!bilateral) {\r\n define += \"#define BLUR_LEGACY\\n\";\r\n }\r\n return { h: define + \"#define BLUR_H\\n\", v: define };\r\n }\r\n\r\n private _createBlurPostProcess(ssaoRatio: number, blurRatio: number, textureType: number): void {\r\n const defines = this._getDefinesForBlur(this.expensiveBlur, this.bypassBlur);\r\n const samplers = this._getSamplersForBlur(this.bypassBlur);\r\n\r\n this._blurHPostProcess = this._createBlurFilter(\"BlurH\", samplers, ssaoRatio, defines.h, textureType, true);\r\n this._blurVPostProcess = this._createBlurFilter(\"BlurV\", samplers, blurRatio, defines.v, textureType, false);\r\n }\r\n\r\n private _createBlurFilter(name: string, samplers: Array, ratio: number, defines: string, textureType: number, horizontal: boolean): PostProcess {\r\n const blurFilter = new PostProcess(\r\n name,\r\n \"ssao2\",\r\n [\"outSize\", \"samples\", \"soften\", \"tolerance\"],\r\n samplers,\r\n ratio,\r\n null,\r\n Texture.BILINEAR_SAMPLINGMODE,\r\n this._scene.getEngine(),\r\n false,\r\n defines,\r\n textureType\r\n );\r\n\r\n blurFilter.onApply = (effect: Effect) => {\r\n if (!this._scene.activeCamera) {\r\n return;\r\n }\r\n\r\n const ssaoCombineSize = horizontal ? this._ssaoCombinePostProcess.width : this._ssaoCombinePostProcess.height;\r\n const originalColorSize = horizontal ? this._originalColorPostProcess.width : this._originalColorPostProcess.height;\r\n\r\n effect.setFloat(\"outSize\", ssaoCombineSize > 0 ? ssaoCombineSize : originalColorSize);\r\n effect.setInt(\"samples\", this.bilateralSamples);\r\n effect.setFloat(\"soften\", this.bilateralSoften);\r\n effect.setFloat(\"tolerance\", this.bilateralTolerance);\r\n if (this._geometryBufferRenderer) {\r\n effect.setTexture(\"depthSampler\", this._geometryBufferRenderer.getGBuffer().textures[0]);\r\n } else if (this._prePassRenderer) {\r\n effect.setTexture(\"depthSampler\", this._prePassRenderer.getRenderTarget().textures[this._prePassRenderer.getIndex(Constants.PREPASS_DEPTH_TEXTURE_TYPE)]);\r\n }\r\n };\r\n\r\n blurFilter.samples = this.textureSamples;\r\n return blurFilter;\r\n }\r\n\r\n private _bits = new Uint32Array(1);\r\n\r\n //Van der Corput radical inverse\r\n private _radicalInverse_VdC(i: number) {\r\n this._bits[0] = i;\r\n this._bits[0] = ((this._bits[0] << 16) | (this._bits[0] >> 16)) >>> 0;\r\n this._bits[0] = ((this._bits[0] & 0x55555555) << 1) | (((this._bits[0] & 0xaaaaaaaa) >>> 1) >>> 0);\r\n this._bits[0] = ((this._bits[0] & 0x33333333) << 2) | (((this._bits[0] & 0xcccccccc) >>> 2) >>> 0);\r\n this._bits[0] = ((this._bits[0] & 0x0f0f0f0f) << 4) | (((this._bits[0] & 0xf0f0f0f0) >>> 4) >>> 0);\r\n this._bits[0] = ((this._bits[0] & 0x00ff00ff) << 8) | (((this._bits[0] & 0xff00ff00) >>> 8) >>> 0);\r\n return this._bits[0] * 2.3283064365386963e-10; // / 0x100000000 or / 4294967296\r\n }\r\n\r\n private _hammersley(i: number, n: number) {\r\n return [i / n, this._radicalInverse_VdC(i)];\r\n }\r\n\r\n private _hemisphereSample_uniform(u: number, v: number): Vector3 {\r\n const phi = v * 2.0 * Math.PI;\r\n // rejecting samples that are close to tangent plane to avoid z-fighting artifacts\r\n const cosTheta = 1.0 - u * 0.85;\r\n const sinTheta = Math.sqrt(1.0 - cosTheta * cosTheta);\r\n return new Vector3(Math.cos(phi) * sinTheta, Math.sin(phi) * sinTheta, cosTheta);\r\n }\r\n\r\n private _generateHemisphere(): number[] {\r\n const numSamples = this.samples;\r\n const result = [];\r\n let vector;\r\n\r\n let i = 0;\r\n while (i < numSamples) {\r\n if (numSamples < 16) {\r\n vector = this._hemisphereSample_uniform(Math.random(), Math.random());\r\n } else {\r\n const rand = this._hammersley(i, numSamples);\r\n vector = this._hemisphereSample_uniform(rand[0], rand[1]);\r\n }\r\n\r\n result.push(vector.x, vector.y, vector.z);\r\n i++;\r\n }\r\n\r\n return result;\r\n }\r\n\r\n private _getDefinesForSSAO() {\r\n const defines = `#define SSAO\\n#define SAMPLES ${this.samples}\\n#define EPSILON ${this.epsilon.toFixed(4)}`;\r\n\r\n return defines;\r\n }\r\n\r\n private static readonly ORTHO_DEPTH_PROJECTION = [1, 0, 0, 0, 1, 0, 0, 0, 1];\r\n\r\n private static readonly PERSPECTIVE_DEPTH_PROJECTION = [0, 0, 0, 0, 0, 0, 1, 1, 1];\r\n\r\n private _createSSAOPostProcess(ratio: number, textureType: number): void {\r\n this._sampleSphere = this._generateHemisphere();\r\n\r\n const defines = this._getDefinesForSSAO();\r\n const samplers = [\"randomSampler\", \"depthSampler\", \"normalSampler\"];\r\n\r\n this._ssaoPostProcess = new PostProcess(\r\n \"ssao2\",\r\n \"ssao2\",\r\n [\r\n \"sampleSphere\",\r\n \"samplesFactor\",\r\n \"randTextureTiles\",\r\n \"totalStrength\",\r\n \"radius\",\r\n \"base\",\r\n \"range\",\r\n \"projection\",\r\n \"near\",\r\n \"texelSize\",\r\n \"xViewport\",\r\n \"yViewport\",\r\n \"maxZ\",\r\n \"minZAspect\",\r\n \"depthProjection\",\r\n ],\r\n samplers,\r\n ratio,\r\n null,\r\n Texture.BILINEAR_SAMPLINGMODE,\r\n this._scene.getEngine(),\r\n false,\r\n defines,\r\n textureType\r\n );\r\n\r\n this._ssaoPostProcess.onApply = (effect: Effect) => {\r\n if (!this._scene.activeCamera) {\r\n return;\r\n }\r\n\r\n effect.setArray3(\"sampleSphere\", this._sampleSphere);\r\n effect.setFloat(\"randTextureTiles\", 32.0);\r\n effect.setFloat(\"samplesFactor\", 1 / this.samples);\r\n effect.setFloat(\"totalStrength\", this.totalStrength);\r\n effect.setFloat2(\"texelSize\", 1 / this._ssaoPostProcess.width, 1 / this._ssaoPostProcess.height);\r\n effect.setFloat(\"radius\", this.radius);\r\n effect.setFloat(\"maxZ\", this.maxZ);\r\n effect.setFloat(\"minZAspect\", this.minZAspect);\r\n effect.setFloat(\"base\", this.base);\r\n effect.setFloat(\"near\", this._scene.activeCamera.minZ);\r\n if (this._scene.activeCamera.mode === Camera.PERSPECTIVE_CAMERA) {\r\n effect.setMatrix3x3(\"depthProjection\", SSAO2RenderingPipeline.PERSPECTIVE_DEPTH_PROJECTION);\r\n effect.setFloat(\"xViewport\", Math.tan(this._scene.activeCamera.fov / 2) * this._scene.getEngine().getAspectRatio(this._scene.activeCamera, true));\r\n effect.setFloat(\"yViewport\", Math.tan(this._scene.activeCamera.fov / 2));\r\n } else {\r\n const halfWidth = this._scene.getEngine().getRenderWidth() / 2.0;\r\n const halfHeight = this._scene.getEngine().getRenderHeight() / 2.0;\r\n const orthoLeft = this._scene.activeCamera.orthoLeft ?? -halfWidth;\r\n const orthoRight = this._scene.activeCamera.orthoRight ?? halfWidth;\r\n const orthoBottom = this._scene.activeCamera.orthoBottom ?? -halfHeight;\r\n const orthoTop = this._scene.activeCamera.orthoTop ?? halfHeight;\r\n effect.setMatrix3x3(\"depthProjection\", SSAO2RenderingPipeline.ORTHO_DEPTH_PROJECTION);\r\n effect.setFloat(\"xViewport\", (orthoRight - orthoLeft) * 0.5);\r\n effect.setFloat(\"yViewport\", (orthoTop - orthoBottom) * 0.5);\r\n }\r\n effect.setMatrix(\"projection\", this._scene.getProjectionMatrix());\r\n\r\n if (this._geometryBufferRenderer) {\r\n effect.setTexture(\"depthSampler\", this._geometryBufferRenderer.getGBuffer().textures[0]);\r\n effect.setTexture(\"normalSampler\", this._geometryBufferRenderer.getGBuffer().textures[1]);\r\n } else if (this._prePassRenderer) {\r\n effect.setTexture(\"depthSampler\", this._prePassRenderer.getRenderTarget().textures[this._prePassRenderer.getIndex(Constants.PREPASS_DEPTH_TEXTURE_TYPE)]);\r\n effect.setTexture(\"normalSampler\", this._prePassRenderer.getRenderTarget().textures[this._prePassRenderer.getIndex(Constants.PREPASS_NORMAL_TEXTURE_TYPE)]);\r\n }\r\n effect.setTexture(\"randomSampler\", this._randomTexture);\r\n };\r\n this._ssaoPostProcess.samples = this.textureSamples;\r\n\r\n if (!this._forceGeometryBuffer) {\r\n this._ssaoPostProcess._prePassEffectConfiguration = new SSAO2Configuration();\r\n }\r\n }\r\n\r\n private _createSSAOCombinePostProcess(ratio: number, textureType: number): void {\r\n this._ssaoCombinePostProcess = new PostProcess(\r\n \"ssaoCombine\",\r\n \"ssaoCombine\",\r\n [],\r\n [\"originalColor\", \"viewport\"],\r\n ratio,\r\n null,\r\n Texture.BILINEAR_SAMPLINGMODE,\r\n this._scene.getEngine(),\r\n false,\r\n undefined,\r\n textureType\r\n );\r\n\r\n this._ssaoCombinePostProcess.onApply = (effect: Effect) => {\r\n const viewport = this._scene.activeCamera!.viewport;\r\n effect.setVector4(\"viewport\", TmpVectors.Vector4[0].copyFromFloats(viewport.x, viewport.y, viewport.width, viewport.height));\r\n effect.setTextureFromPostProcessOutput(\"originalColor\", this._originalColorPostProcess);\r\n };\r\n this._ssaoCombinePostProcess.samples = this.textureSamples;\r\n }\r\n\r\n private _createRandomTexture(): void {\r\n const size = 128;\r\n\r\n this._randomTexture = new DynamicTexture(\"SSAORandomTexture\", size, this._scene, false, Texture.BILINEAR_SAMPLINGMODE);\r\n this._randomTexture.wrapU = Texture.WRAP_ADDRESSMODE;\r\n this._randomTexture.wrapV = Texture.WRAP_ADDRESSMODE;\r\n\r\n const context = this._randomTexture.getContext();\r\n\r\n const rand = (min: number, max: number) => {\r\n return Math.random() * (max - min) + min;\r\n };\r\n\r\n const randVector = Vector3.Zero();\r\n\r\n for (let x = 0; x < size; x++) {\r\n for (let y = 0; y < size; y++) {\r\n randVector.x = rand(0.0, 1.0);\r\n randVector.y = rand(0.0, 1.0);\r\n randVector.z = 0.0;\r\n\r\n randVector.normalize();\r\n\r\n randVector.scaleInPlace(255);\r\n randVector.x = Math.floor(randVector.x);\r\n randVector.y = Math.floor(randVector.y);\r\n\r\n context.fillStyle = \"rgb(\" + randVector.x + \", \" + randVector.y + \", \" + randVector.z + \")\";\r\n context.fillRect(x, y, 1, 1);\r\n }\r\n }\r\n\r\n this._randomTexture.update(false);\r\n }\r\n\r\n /**\r\n * Serialize the rendering pipeline (Used when exporting)\r\n * @returns the serialized object\r\n */\r\n public serialize(): any {\r\n const serializationObject = SerializationHelper.Serialize(this);\r\n serializationObject.customType = \"SSAO2RenderingPipeline\";\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * Parse the serialized pipeline\r\n * @param source Source pipeline.\r\n * @param scene The scene to load the pipeline to.\r\n * @param rootUrl The URL of the serialized pipeline.\r\n * @returns An instantiated pipeline from the serialized object.\r\n */\r\n public static Parse(source: any, scene: Scene, rootUrl: string): SSAO2RenderingPipeline {\r\n return SerializationHelper.Parse(\r\n () => new SSAO2RenderingPipeline(source._name, scene, source._ratio, undefined, source._forceGeometryBuffer, source._textureType),\r\n source,\r\n scene,\r\n rootUrl\r\n );\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.SSAO2RenderingPipeline\", SSAO2RenderingPipeline);\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\n\nconst name = \"fxaaPixelShader\";\nconst shader = `#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE)\n#define TEXTUREFUNC(s,c,l) texture2DLodEXT(s,c,l)\n#else\n#define TEXTUREFUNC(s,c,b) texture2D(s,c,b)\n#endif\nuniform sampler2D textureSampler;uniform vec2 texelSize;varying vec2 vUV;varying vec2 sampleCoordS;varying vec2 sampleCoordE;varying vec2 sampleCoordN;varying vec2 sampleCoordW;varying vec2 sampleCoordNW;varying vec2 sampleCoordSE;varying vec2 sampleCoordNE;varying vec2 sampleCoordSW;const float fxaaQualitySubpix=1.0;const float fxaaQualityEdgeThreshold=0.166;const float fxaaQualityEdgeThresholdMin=0.0833;const vec3 kLumaCoefficients=vec3(0.2126,0.7152,0.0722);\n#define FxaaLuma(rgba) dot(rgba.rgb,kLumaCoefficients)\nvoid main(){vec2 posM;posM.x=vUV.x;posM.y=vUV.y;vec4 rgbyM=TEXTUREFUNC(textureSampler,vUV,0.0);float lumaM=FxaaLuma(rgbyM);float lumaS=FxaaLuma(TEXTUREFUNC(textureSampler,sampleCoordS,0.0));float lumaE=FxaaLuma(TEXTUREFUNC(textureSampler,sampleCoordE,0.0));float lumaN=FxaaLuma(TEXTUREFUNC(textureSampler,sampleCoordN,0.0));float lumaW=FxaaLuma(TEXTUREFUNC(textureSampler,sampleCoordW,0.0));float maxSM=max(lumaS,lumaM);float minSM=min(lumaS,lumaM);float maxESM=max(lumaE,maxSM);float minESM=min(lumaE,minSM);float maxWN=max(lumaN,lumaW);float minWN=min(lumaN,lumaW);float rangeMax=max(maxWN,maxESM);float rangeMin=min(minWN,minESM);float rangeMaxScaled=rangeMax*fxaaQualityEdgeThreshold;float range=rangeMax-rangeMin;float rangeMaxClamped=max(fxaaQualityEdgeThresholdMin,rangeMaxScaled);\n#ifndef MALI\nif(range=edgeVert;float subpixA=subpixNSWE*2.0+subpixNWSWNESE;if (!horzSpan)\n{lumaN=lumaW;}\nif (!horzSpan) \n{lumaS=lumaE;}\nif (horzSpan) \n{lengthSign=texelSize.y;}\nfloat subpixB=(subpixA*(1.0/12.0))-lumaM;float gradientN=lumaN-lumaM;float gradientS=lumaS-lumaM;float lumaNN=lumaN+lumaM;float lumaSS=lumaS+lumaM;bool pairN=abs(gradientN)>=abs(gradientS);float gradient=max(abs(gradientN),abs(gradientS));if (pairN)\n{lengthSign=-lengthSign;}\nfloat subpixC=clamp(abs(subpixB)*subpixRcpRange,0.0,1.0);vec2 posB;posB.x=posM.x;posB.y=posM.y;vec2 offNP;offNP.x=(!horzSpan) ? 0.0 : texelSize.x;offNP.y=(horzSpan) ? 0.0 : texelSize.y;if (!horzSpan) \n{posB.x+=lengthSign*0.5;}\nif (horzSpan)\n{posB.y+=lengthSign*0.5;}\nvec2 posN;posN.x=posB.x-offNP.x*1.5;posN.y=posB.y-offNP.y*1.5;vec2 posP;posP.x=posB.x+offNP.x*1.5;posP.y=posB.y+offNP.y*1.5;float subpixD=((-2.0)*subpixC)+3.0;float lumaEndN=FxaaLuma(TEXTUREFUNC(textureSampler,posN,0.0));float subpixE=subpixC*subpixC;float lumaEndP=FxaaLuma(TEXTUREFUNC(textureSampler,posP,0.0));if (!pairN) \n{lumaNN=lumaSS;}\nfloat gradientScaled=gradient*1.0/4.0;float lumaMM=lumaM-lumaNN*0.5;float subpixF=subpixD*subpixE;bool lumaMLTZero=lumaMM<0.0;lumaEndN-=lumaNN*0.5;lumaEndP-=lumaNN*0.5;bool doneN=abs(lumaEndN)>=gradientScaled;bool doneP=abs(lumaEndP)>=gradientScaled;if (!doneN) \n{posN.x-=offNP.x*3.0;}\nif (!doneN) \n{posN.y-=offNP.y*3.0;}\nbool doneNP=(!doneN) || (!doneP);if (!doneP) \n{posP.x+=offNP.x*3.0;}\nif (!doneP)\n{posP.y+=offNP.y*3.0;}\nif (doneNP)\n{if (!doneN) lumaEndN=FxaaLuma(TEXTUREFUNC(textureSampler,posN.xy,0.0));if (!doneP) lumaEndP=FxaaLuma(TEXTUREFUNC(textureSampler,posP.xy,0.0));if (!doneN) lumaEndN=lumaEndN-lumaNN*0.5;if (!doneP) lumaEndP=lumaEndP-lumaNN*0.5;doneN=abs(lumaEndN)>=gradientScaled;doneP=abs(lumaEndP)>=gradientScaled;if (!doneN) posN.x-=offNP.x*12.0;if (!doneN) posN.y-=offNP.y*12.0;doneNP=(!doneN) || (!doneP);if (!doneP) posP.x+=offNP.x*12.0;if (!doneP) posP.y+=offNP.y*12.0;}\nfloat dstN=posM.x-posN.x;float dstP=posP.x-posM.x;if (!horzSpan)\n{dstN=posM.y-posN.y;}\nif (!horzSpan) \n{dstP=posP.y-posM.y;}\nbool goodSpanN=(lumaEndN<0.0) != lumaMLTZero;float spanLength=(dstP+dstN);bool goodSpanP=(lumaEndP<0.0) != lumaMLTZero;float spanLengthRcp=1.0/spanLength;bool directionN=dstN = null,\r\n samplingMode?: number,\r\n engine?: Engine,\r\n reusable?: boolean,\r\n textureType: number = Constants.TEXTURETYPE_UNSIGNED_INT\r\n ) {\r\n super(name, \"fxaa\", [\"texelSize\"], null, options, camera, samplingMode || Texture.BILINEAR_SAMPLINGMODE, engine, reusable, null, textureType, \"fxaa\", undefined, true);\r\n\r\n const defines = this._getDefines();\r\n this.updateEffect(defines);\r\n\r\n this.onApplyObservable.add((effect: Effect) => {\r\n const texelSize = this.texelSize;\r\n effect.setFloat2(\"texelSize\", texelSize.x, texelSize.y);\r\n });\r\n }\r\n\r\n private _getDefines(): Nullable {\r\n const engine = this.getEngine();\r\n if (!engine) {\r\n return null;\r\n }\r\n\r\n const glInfo = engine.getGlInfo();\r\n if (glInfo && glInfo.renderer && glInfo.renderer.toLowerCase().indexOf(\"mali\") > -1) {\r\n return \"#define MALI 1\\n\";\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _Parse(parsedPostProcess: any, targetCamera: Camera, scene: Scene, rootUrl: string) {\r\n return SerializationHelper.Parse(\r\n () => {\r\n return new FxaaPostProcess(\r\n parsedPostProcess.name,\r\n parsedPostProcess.options,\r\n targetCamera,\r\n parsedPostProcess.renderTargetSamplingMode,\r\n scene.getEngine(),\r\n parsedPostProcess.reusable\r\n );\r\n },\r\n parsedPostProcess,\r\n scene,\r\n rootUrl\r\n );\r\n }\r\n}\r\n\r\nRegisterClass(\"BABYLON.FxaaPostProcess\", FxaaPostProcess);\r\n","import { PBRMaterial } from '@babylonjs/core/Materials/PBR/pbrMaterial';\r\nimport { BaseTexture } from '@babylonjs/core/Materials/Textures/baseTexture';\r\nimport { CubeTexture } from '@babylonjs/core/Materials/Textures/cubeTexture';\r\nimport { HDRCubeTexture } from '@babylonjs/core/Materials/Textures/hdrCubeTexture';\r\nimport { Texture } from '@babylonjs/core/Materials/Textures/texture';\r\nimport { Color4, Quaternion, Vector3 } from '@babylonjs/core/Maths';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { LensRenderingPipeline } from '@babylonjs/core/PostProcesses/RenderPipeline/Pipelines/lensRenderingPipeline';\r\nimport { SSAO2RenderingPipeline } from '@babylonjs/core/PostProcesses/RenderPipeline/Pipelines/ssao2RenderingPipeline';\r\nimport { FxaaPostProcess } from '@babylonjs/core/PostProcesses/fxaaPostProcess';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { eBackground, eTextureSamplingMode } from '@models/classes/enums';\r\nimport { Kb3dObject } from '@models/classes/kb3d-object';\r\nimport { SceneNode, isSceneNode } from '@models/classes/scene-node';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass, isTrackedClass } from '@models/interfaces/tracked-class';\r\nimport { GuiHelper } from '@view/helpers/gui-helper';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { Renderer } from './renderer';\r\nimport { RendererManager } from './renderer-manager';\r\n\r\nexport const SKYBOX_SIZE = 1000;\r\n\r\nexport const SamplingModeMap = {\r\n [eTextureSamplingMode.nearest]: Texture.NEAREST_SAMPLINGMODE,\r\n [eTextureSamplingMode.bilinear]: Texture.BILINEAR_SAMPLINGMODE,\r\n [eTextureSamplingMode.trilinear]: Texture.TRILINEAR_SAMPLINGMODE,\r\n};\r\n\r\nexport class SceneRenderer extends Renderer {\r\n private ssaoPostProcess: SSAO2RenderingPipeline;\r\n private lensEffectPostProcess: LensRenderingPipeline;\r\n private fxaaPostProcess: FxaaPostProcess;\r\n private yVector = new Vector3(0, 1, 0);\r\n\r\n constructor(viewer: KbViewer, scene: Scene, private gui: GuiHelper, private renderManager: RendererManager) {\r\n super(viewer, scene);\r\n }\r\n\r\n modelToken = Token.SceneNode;\r\n handlesNode(node: Kb3dObject) {\r\n return isSceneNode(node) && !node.isNested();\r\n }\r\n\r\n update3dInternal(node: ITrackedClass) {\r\n this.processEnvironment(node);\r\n\r\n //we create a transform node that ends up holding all space nodes in the scene\r\n //this creates parity with nested scenes and gives us something to transform in WebXR without re-parenting anything\r\n let tnode = this.get3d(node);\r\n if (!tnode) {\r\n tnode = new TransformNode(node._dynamicId, this.scene);\r\n this.set3d(node, tnode);\r\n\r\n if (!this.viewer.options.forceDisablePostProcess && this.scene.activeCamera) {\r\n if (node.enableFxaaPostProcess) {\r\n this.fxaaPostProcess = new FxaaPostProcess(\r\n 'fxaa',\r\n 1.0,\r\n this.scene.activeCamera,\r\n Texture.BILINEAR_SAMPLINGMODE,\r\n this.scene.getEngine()\r\n );\r\n }\r\n if (node.enableLensEffectsPostProcess) {\r\n this.lensEffectPostProcess = new LensRenderingPipeline(\r\n 'lens',\r\n {\r\n edge_blur: 0.5,\r\n chromatic_aberration: 0.5,\r\n distortion: 0, // Used to be 0.5 but lens distortion makes mouse selection useless,\r\n dof_focus_distance: 200,\r\n dof_aperture: 5.0, // set this very high for tilt-shift effect\r\n grain_amount: 0.1,\r\n dof_pentagon: true,\r\n dof_gain: 1.0,\r\n dof_threshold: 1.0,\r\n dof_darken: 0,\r\n },\r\n this.scene,\r\n 1.0,\r\n [this.scene.activeCamera]\r\n );\r\n }\r\n if (node.enableSSAOPostProcess) {\r\n this.ssaoPostProcess = new SSAO2RenderingPipeline('ssaopipeline', this.scene, 0.75);\r\n }\r\n }\r\n }\r\n\r\n if (node.changes.has('enableDeferredSceneUpdates')) {\r\n this.renderManager.maxRenderTime =\r\n node.enableDeferredSceneUpdates && !this.viewer.options.editorMode ? 20 : 20000;\r\n }\r\n\r\n if (node.changes.has('textureSamplingMode')) {\r\n const samplingMode = SamplingModeMap[node.textureSamplingMode];\r\n const materials = this.renderManager.material.getAllDefinedMaterials();\r\n\r\n materials.forEach(material => {\r\n material.getActiveTextures().forEach(texture => {\r\n if (texture) {\r\n texture.updateSamplingMode(samplingMode);\r\n }\r\n });\r\n });\r\n }\r\n\r\n if (node.changes.has('backgroundColor') || node.changes.has('background')) {\r\n if (node.background == eBackground.color) {\r\n this.scene.clearColor = node.backgroundColor.toColor4();\r\n } else {\r\n this.scene.clearColor = new Color4(0, 0, 0, 0);\r\n }\r\n }\r\n if (node.changes.has('ambientColor')) {\r\n this.scene.ambientColor = node.ambientColor.toColor3();\r\n }\r\n if (node.changes.hasAny('renderMode', 'edgeWidth', 'edgeColor', 'edgeDisplayTolerance')) {\r\n this.viewer.setRenderMode(node.renderMode);\r\n }\r\n if (node.changes.has('enableHoverEffects')) {\r\n this.scene.constantlyUpdateMeshUnderPointer = node.enableHoverEffects;\r\n }\r\n if (node.changes.has('connectorSize')) {\r\n for (const c of node.getAllConnectors()) {\r\n this.viewer.rendererManager.connector.updateScale(c);\r\n }\r\n for (const c of node.getAllSketchControlPoints()) {\r\n this.viewer.rendererManager.controlPointRenderer.updateScale(c);\r\n }\r\n }\r\n if (node.changes.hasAny('dimensionUnitScalar', 'dimensionUnitSymbol')) {\r\n node.annotations.forEach(annotationNode => {\r\n if (isTrackedClass(annotationNode)) {\r\n annotationNode.isModified = true;\r\n }\r\n });\r\n }\r\n\r\n let gridNeedsRebuild = false;\r\n if (\r\n node.changes.hasAny(\r\n '_forceGrid',\r\n 'gridSize',\r\n 'gridNormal',\r\n 'gridStepSize',\r\n 'gridMajorInterval',\r\n 'backgroundColor'\r\n )\r\n ) {\r\n if (node._forceGrid || node.enableGrid) {\r\n this.gui.disableGrid();\r\n gridNeedsRebuild = true;\r\n }\r\n if (node._forceGrid || node.enableGridAxes) {\r\n this.gui.disableGridAxes();\r\n gridNeedsRebuild = true;\r\n }\r\n }\r\n if (gridNeedsRebuild || node.changes.has('enableGrid')) {\r\n if (node._forceGrid || node.enableGrid) {\r\n this.gui.enableGrid(node, node._forceGrid || node.enableGridAxes);\r\n } else {\r\n this.gui.disableGrid();\r\n }\r\n }\r\n if (gridNeedsRebuild || node.changes.has('enableGridAxes')) {\r\n if (node._forceGrid || node.enableGridAxes) {\r\n this.gui.enableGridAxes(node);\r\n } else {\r\n this.gui.disableGridAxes();\r\n }\r\n }\r\n\r\n if (node._rootScene?._dynamicId === node._dynamicId) {\r\n const eventPayload = Object.fromEntries(node.changes);\r\n if (this.viewer.canvas) {\r\n this.viewer.canvas.dispatchEvent(\r\n new CustomEvent('kb3d.scenePropertyChange', { detail: eventPayload, bubbles: true })\r\n );\r\n }\r\n }\r\n }\r\n\r\n private _envTexture: BaseTexture;\r\n private _envTexturePath: string;\r\n public skybox: Mesh | null;\r\n\r\n processEnvironment(node: ITrackedClass) {\r\n if (node.changes.has('environment') || node.changes.has('background') || node.changes.has('environmentBlur')) {\r\n const matService = node._manager?.getService(Token.MaterialService);\r\n if (!matService) {\r\n throw Error('Cannot fetch environment material');\r\n }\r\n\r\n let path: string;\r\n if (node.background === eBackground.environment) {\r\n if (node.environmentBlur < 0.15) {\r\n path = matService.getAbsoluteEnvironmentPath(node.environment, 3);\r\n } else {\r\n path = matService.getAbsoluteEnvironmentPath(node.environment, 1);\r\n }\r\n } else {\r\n path = matService.getAbsoluteEnvironmentPath(node.environment, 0);\r\n }\r\n\r\n if (path !== '' && this._envTexturePath !== path) {\r\n this._envTexturePath = path;\r\n if (this._envTexture) {\r\n this._envTexture.dispose();\r\n }\r\n\r\n let texture: BaseTexture | undefined = undefined;\r\n\r\n if (path.match(/\\.dds(\\?.*)?$/) || path.match(/\\.env(\\?.*)?$/)) {\r\n texture = CubeTexture.CreateFromPrefilteredData(path, this.scene);\r\n } else if (path.match(/\\.hdr(\\?.*)?$/)) {\r\n texture = new HDRCubeTexture(path, this.scene, 128);\r\n }\r\n\r\n if (texture != null) {\r\n texture.gammaSpace = false;\r\n this.scene.environmentTexture = texture;\r\n this._envTexture = texture;\r\n } else {\r\n this.scene.environmentTexture = null;\r\n }\r\n }\r\n }\r\n\r\n if (\r\n (node.changes.has('background') || node.changes.has('environment')) &&\r\n node.background == eBackground.environment &&\r\n this._envTexture\r\n ) {\r\n if (!this.skybox) {\r\n this.skybox = this.scene.createDefaultSkybox(this._envTexture, true, SKYBOX_SIZE, node.environmentBlur);\r\n } else if (node.changes.has('environment')) {\r\n const skyMat = this.skybox.material as PBRMaterial;\r\n skyMat.reflectionTexture = this._envTexture.clone();\r\n skyMat.reflectionTexture!.coordinatesMode = Texture.SKYBOX_MODE;\r\n }\r\n }\r\n\r\n if (node.changes.has('background') && node.background != eBackground.environment) {\r\n if (this.skybox) {\r\n this.skybox.dispose();\r\n this.skybox = null;\r\n }\r\n }\r\n\r\n if (node.changes.has('environmentBlur')) {\r\n if (this.skybox) {\r\n const skyMat = this.skybox.material as PBRMaterial;\r\n skyMat.microSurface = 1.0 - node.environmentBlur;\r\n }\r\n }\r\n\r\n if (node.changes.has('environmentIntensity')) {\r\n this.scene.environmentIntensity = node.environmentIntensity;\r\n }\r\n\r\n if (node.changes.has('environmentRotation') && this.scene.environmentTexture) {\r\n if (\r\n this.scene.environmentTexture instanceof CubeTexture ||\r\n this.scene.environmentTexture instanceof HDRCubeTexture\r\n ) {\r\n this.scene.environmentTexture.rotationY = (node.environmentRotation * Math.PI) / 180;\r\n if (this.skybox) {\r\n this.skybox.rotationQuaternion = Quaternion.RotationAxis(\r\n this.yVector,\r\n (node.environmentRotation * Math.PI) / 180\r\n );\r\n }\r\n }\r\n }\r\n }\r\n\r\n public dispose() {\r\n super.dispose();\r\n if (this._envTexture) {\r\n this._envTexture.dispose();\r\n }\r\n if (this.skybox) {\r\n this.skybox.dispose();\r\n }\r\n if (this.ssaoPostProcess) {\r\n this.ssaoPostProcess.dispose();\r\n }\r\n if (this.lensEffectPostProcess) {\r\n this.lensEffectPostProcess.dispose();\r\n }\r\n if (this.fxaaPostProcess) {\r\n this.fxaaPostProcess.dispose();\r\n }\r\n }\r\n\r\n public getSkyBox() {\r\n return this.skybox;\r\n }\r\n}\r\n","import { BaseTexture } from '@babylonjs/core/Materials/Textures/baseTexture';\r\nimport { Vector3 } from '@babylonjs/core/Maths/math.vector';\r\nimport { MaterialNode } from '@models/classes';\r\nimport {\r\n eTextureBlendMode,\r\n eTextureChannel,\r\n eTextureFit,\r\n eTextureHorizontalAlign,\r\n eTextureVerticalAlign,\r\n} from '@models/classes/enums';\r\nimport { FileTextureLayer } from '@models/classes/materials/file-texture-layer';\r\nimport { TextureLayer } from '@models/classes/materials/texture-layer';\r\nimport { KbColor } from '@models/classes/primitives/kb-color';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass } from '@models/interfaces/tracked-class';\r\nimport { Renderer } from '../renderer';\r\nimport { TextureParameters } from '../texture-stack';\r\n\r\ninterface ICanvasSize {\r\n width: number;\r\n height: number;\r\n}\r\n\r\nexport interface IDrawOperations {\r\n mask?: { context: CanvasRenderingContext2D };\r\n normalRotate?: { angle: number };\r\n flipNormalsRg?: boolean;\r\n colorize?: { color: KbColor };\r\n}\r\n\r\nconst blendModeMap = {\r\n [eTextureBlendMode.normal]: 'source-over',\r\n [eTextureBlendMode.multiply]: 'multiply',\r\n [eTextureBlendMode.screen]: 'screen',\r\n [eTextureBlendMode.overlay]: 'overlay',\r\n [eTextureBlendMode.darken]: 'darken',\r\n [eTextureBlendMode.lighten]: 'lighten',\r\n [eTextureBlendMode.hue]: 'hue',\r\n [eTextureBlendMode.saturation]: 'saturation',\r\n [eTextureBlendMode.color]: 'color',\r\n [eTextureBlendMode.luminosity]: 'luminosity',\r\n [eTextureBlendMode.difference]: 'difference',\r\n [eTextureBlendMode.mask]: 'destination-in',\r\n [eTextureBlendMode.vector]: 'kb3d-vector', // special blending mode\r\n};\r\n\r\nconst workingCanvases: HTMLCanvasElement[] = [];\r\n/** Make sure to dispose the canvas after use! Otherwise you'll get a memory leak! */\r\nfunction getWorkingCanvas(width: number, height: number) {\r\n let canvas: HTMLCanvasElement | undefined;\r\n for (let i = 0; i < workingCanvases.length; i++) {\r\n if (workingCanvases[i].width === 1) {\r\n canvas = workingCanvases[i];\r\n break;\r\n }\r\n }\r\n if (!canvas) {\r\n canvas = document.createElement('canvas');\r\n workingCanvases.push(canvas);\r\n }\r\n canvas.width = width;\r\n canvas.height = height;\r\n const context = canvas.getContext('2d', {\r\n willReadFrequently: true,\r\n })!;\r\n context.clearRect(0, 0, canvas.width, canvas.height);\r\n return canvas;\r\n}\r\nfunction disposeWorkingCanvas(canvas: HTMLCanvasElement) {\r\n canvas.width = 1;\r\n canvas.height = 1;\r\n}\r\n\r\nexport abstract class TextureLayerRenderer extends Renderer {\r\n modelToken = Token.TextureLayer;\r\n\r\n paintTexture(\r\n texture: T,\r\n context: CanvasRenderingContext2D,\r\n parameters: TextureParameters,\r\n operations: IDrawOperations\r\n ) {}\r\n\r\n getNativeSize(texture: T): ICanvasSize {\r\n return { width: 1, height: 1 };\r\n }\r\n\r\n update3dInternal(node: ITrackedClass) {\r\n this.viewer.updateTextureWarnings();\r\n }\r\n\r\n textureHasContent(node: ITrackedClass): boolean {\r\n return false;\r\n }\r\n\r\n protected calculateOffsetAndDimensions(\r\n texture: T,\r\n ctxWidth: number,\r\n ctxHeight: number,\r\n sourceWidth: number,\r\n sourceHeight: number\r\n ) {\r\n let offsetX = texture.uOffset * ctxWidth,\r\n offsetY = texture.vOffset * ctxHeight,\r\n drawHeight = sourceHeight,\r\n drawWidth = sourceWidth;\r\n\r\n switch (texture.verticalFit) {\r\n case eTextureFit.contain:\r\n if (sourceHeight < ctxHeight) {\r\n break;\r\n }\r\n // falls through\r\n case eTextureFit.stretch:\r\n drawHeight = ctxHeight;\r\n }\r\n switch (texture.horizontalFit) {\r\n case eTextureFit.contain:\r\n if (sourceWidth < ctxWidth) {\r\n break;\r\n }\r\n // falls through\r\n case eTextureFit.stretch:\r\n drawWidth = ctxWidth;\r\n }\r\n if (texture.maintainAspect) {\r\n const minRatio = Math.min(drawWidth / sourceWidth, drawHeight / sourceHeight);\r\n drawWidth = minRatio * sourceWidth;\r\n drawHeight = minRatio * sourceHeight;\r\n }\r\n\r\n switch (texture.verticalAlign) {\r\n // left is offset with no adjustment\r\n case eTextureVerticalAlign.bottom:\r\n offsetY += ctxHeight - drawHeight;\r\n break;\r\n case eTextureVerticalAlign.middle:\r\n offsetY += (ctxHeight - drawHeight) / 2;\r\n break;\r\n }\r\n switch (texture.horizontalAlign) {\r\n // top is offset with no adjustment\r\n case eTextureHorizontalAlign.right:\r\n offsetX += ctxWidth - drawWidth;\r\n break;\r\n case eTextureHorizontalAlign.center:\r\n offsetX += (ctxWidth - drawWidth) / 2;\r\n break;\r\n }\r\n\r\n return {\r\n offsetX,\r\n offsetY,\r\n drawHeight,\r\n drawWidth,\r\n };\r\n }\r\n\r\n protected drawInBlendedContext(\r\n texture: T,\r\n context: CanvasRenderingContext2D,\r\n parameters: TextureParameters,\r\n cb: (context: CanvasRenderingContext2D) => void,\r\n operations: IDrawOperations\r\n ) {\r\n if (operations.mask) {\r\n const maskingCtx = operations.mask.context;\r\n delete operations.mask;\r\n // Transfer transform from context to masking context to mask what is drawn\r\n const transform = context.getTransform();\r\n maskingCtx.setTransform(transform);\r\n context.setTransform(1, 0, 0, 1, 0, 0);\r\n\r\n maskingCtx.globalCompositeOperation = 'source-in';\r\n cb(maskingCtx);\r\n maskingCtx.setTransform(1, 0, 0, 1, 0, 0);\r\n\r\n this.drawInBlendedContext(\r\n texture,\r\n context,\r\n parameters,\r\n ctx => {\r\n ctx.drawImage(maskingCtx.canvas, 0, 0, ctx.canvas.width, ctx.canvas.height);\r\n },\r\n operations\r\n );\r\n } else if (Object.keys(operations).length > 0) {\r\n const tempCanvas = getWorkingCanvas(context.canvas.width, context.canvas.height);\r\n const tempContext = tempCanvas.getContext('2d')!;\r\n cb(tempContext);\r\n this.filterImageInPlace(tempCanvas, operations);\r\n\r\n this.drawInBlendedContext(\r\n texture,\r\n context,\r\n parameters,\r\n ctx => {\r\n ctx.drawImage(tempCanvas, 0, 0, ctx.canvas.width, ctx.canvas.height);\r\n disposeWorkingCanvas(tempCanvas);\r\n },\r\n operations\r\n );\r\n } else {\r\n context.globalAlpha = texture.strength / parameters.strength;\r\n switch (texture.blendMode) {\r\n case eTextureBlendMode.vector: {\r\n const tempCanvas = getWorkingCanvas(context.canvas.width, context.canvas.height);\r\n const tempContext = tempCanvas.getContext('2d')!;\r\n cb(tempContext);\r\n const srcData = tempContext.getImageData(0, 0, context.canvas.width, context.canvas.height).data;\r\n const dstImageData = context.getImageData(0, 0, context.canvas.width, context.canvas.height);\r\n const dstData = dstImageData.data;\r\n const v1 = Vector3.Zero();\r\n const v2 = Vector3.Zero();\r\n\r\n const alphaScale = context.globalAlpha / 255;\r\n for (let i = 0, n = srcData.length; i < n; i += 4) {\r\n v1.copyFromFloats(dstData[i] / 128 - 1, dstData[i + 1] / 128 - 1, dstData[i + 2] / 256 - 0.5);\r\n v2.copyFromFloats(srcData[i] / 128 - 1, srcData[i + 1] / 128 - 1, srcData[i + 2] / 256 - 0.5);\r\n v2.scaleInPlace(srcData[i + 3] * alphaScale);\r\n v1.addInPlace(v2);\r\n v1.normalize();\r\n dstData[i] = (v1.x + 1) * 128;\r\n dstData[i + 1] = (v1.y + 1) * 128;\r\n dstData[i + 2] = (v1.z + 1) * 128;\r\n dstData[i + 3] = 255;\r\n }\r\n tempContext.putImageData(dstImageData, 0, 0);\r\n context.drawImage(tempCanvas, 0, 0, context.canvas.width, context.canvas.height);\r\n disposeWorkingCanvas(tempCanvas);\r\n\r\n break;\r\n }\r\n\r\n default: {\r\n (context.globalCompositeOperation as unknown) = blendModeMap[texture.blendMode] || 'source-over';\r\n cb(context);\r\n context.globalCompositeOperation = 'source-over';\r\n }\r\n }\r\n context.globalAlpha = 1;\r\n }\r\n }\r\n\r\n protected filterImageInPlace(buffer: HTMLCanvasElement, operations: IDrawOperations) {\r\n const ctx = buffer.getContext('2d')!;\r\n const width = buffer.width;\r\n const height = buffer.height;\r\n\r\n if (Object.keys(operations).length === 0) {\r\n return buffer;\r\n }\r\n\r\n if (operations.colorize) {\r\n const tempCanvas = getWorkingCanvas(width, height);\r\n const tmpCtx = tempCanvas.getContext('2d')!;\r\n\r\n const color = operations.colorize.color;\r\n delete operations.colorize;\r\n\r\n tmpCtx.globalCompositeOperation = 'source-over';\r\n tmpCtx.fillStyle = `rgb(${color.r * 255}, ${color.g * 255}, ${color.b * 255})`;\r\n tmpCtx.fillRect(0, 0, ctx.canvas.width, ctx.canvas.height);\r\n\r\n tmpCtx.globalCompositeOperation = 'luminosity';\r\n tmpCtx.drawImage(buffer, 0, 0);\r\n tmpCtx.globalCompositeOperation = 'destination-in';\r\n tmpCtx.drawImage(buffer, 0, 0);\r\n ctx.drawImage(tempCanvas, 0, 0);\r\n disposeWorkingCanvas(tempCanvas);\r\n }\r\n\r\n if (Object.keys(operations).length === 0) {\r\n return buffer;\r\n }\r\n\r\n const imgData = ctx.getImageData(0, 0, width, height);\r\n const data = imgData.data;\r\n\r\n if (operations.normalRotate) {\r\n const rotateAngle = -1 * operations.normalRotate.angle;\r\n delete operations.normalRotate;\r\n const cosA = Math.cos((rotateAngle * Math.PI) / 180);\r\n const sinA = Math.sin((rotateAngle * Math.PI) / 180);\r\n\r\n for (let i = 0, n = data.length; i < n; i += 4) {\r\n const x1 = data[i] / 128 - 1;\r\n const y1 = data[i + 1] / 128 - 1;\r\n const x2 = cosA * x1 - sinA * y1;\r\n const y2 = sinA * x1 + cosA * y1;\r\n data[i] = (x2 + 1) * 128;\r\n data[i + 1] = (y2 + 1) * 128;\r\n }\r\n }\r\n if (operations.flipNormalsRg) {\r\n delete operations.flipNormalsRg;\r\n\r\n for (let i = 0, n = data.length; i < n; i += 4) {\r\n const r = data[i];\r\n data[i] = data[i + 1];\r\n data[i + 1] = r;\r\n }\r\n }\r\n ctx.putImageData(imgData, 0, 0);\r\n\r\n return buffer;\r\n }\r\n\r\n protected drawSourceToCanvas(\r\n texture: T,\r\n source: ImageBitmap | HTMLCanvasElement, // | CanvasImageSource,\r\n context: CanvasRenderingContext2D,\r\n parameters: TextureParameters,\r\n operations: IDrawOperations\r\n ) {\r\n let width = source.width as number,\r\n height = source.height as number;\r\n const ctxWidth = context.canvas.width,\r\n ctxHeight = context.canvas.height;\r\n\r\n if (texture.colorize) {\r\n operations.colorize = { color: texture.colorize };\r\n }\r\n\r\n const firstTextureLayer =\r\n texture._parent?.textures\r\n .filter(t => t.channel === texture.channel && t.enabled)\r\n .findIndex(t => t === texture) === 0;\r\n\r\n if (texture.wAngle && texture.channel === eTextureChannel.normal && !firstTextureLayer) {\r\n operations.normalRotate = { angle: texture.wAngle };\r\n }\r\n if (\r\n texture instanceof FileTextureLayer &&\r\n texture.flipNormalsRg &&\r\n texture.channel === eTextureChannel.normal\r\n ) {\r\n operations.flipNormalsRg = true;\r\n }\r\n\r\n if (!firstTextureLayer) {\r\n const drawAt = this.calculateOffsetAndDimensions(texture, ctxWidth, ctxHeight, width, height);\r\n\r\n const pivotX = texture.uPivot * ctxWidth,\r\n pivotY = texture.vPivot * ctxHeight;\r\n\r\n if (texture.tile) {\r\n let patternCanvas: HTMLCanvasElement | undefined;\r\n let patternAttempt: CanvasPattern | undefined;\r\n // In the case of stretch modes, the draw image size will differ from the native size so we have to redraw it first\r\n // iOS is unable to draw ImageBitmaps to patterns, so we have to try/catch for iOS\r\n if (drawAt.drawWidth === width && drawAt.drawHeight === height) {\r\n try {\r\n patternAttempt = context.createPattern(source, 'repeat')!;\r\n } catch (e) {\r\n // Do nothing, fallback below\r\n }\r\n }\r\n if (!patternAttempt) {\r\n patternCanvas = getWorkingCanvas(drawAt.drawWidth, drawAt.drawHeight);\r\n const patternContext = patternCanvas.getContext('2d')!;\r\n patternContext.drawImage(source, 0, 0, drawAt.drawWidth, drawAt.drawHeight);\r\n\r\n width = drawAt.drawWidth;\r\n height = drawAt.drawHeight;\r\n patternAttempt = context.createPattern(patternCanvas, 'repeat')!;\r\n }\r\n const pattern = patternAttempt;\r\n\r\n // Initial canvas transformations for angle and scaling\r\n context.translate(pivotX, pivotY);\r\n\r\n //if legacy mode enabled\r\n if (\r\n texture._parent &&\r\n texture._parent instanceof MaterialNode &&\r\n texture._parent.legacyTextureStacking === false\r\n ) {\r\n context.scale(texture.uScale, texture.vScale);\r\n context.rotate((texture.wAngle * Math.PI) / 180);\r\n } else {\r\n context.rotate((texture.wAngle * Math.PI) / 180);\r\n context.scale(texture.uScale, texture.vScale);\r\n if ((texture.uScale !== 1 || texture.vScale !== 1) && texture.wAngle !== 0) {\r\n console.warn(\r\n `Texture scale and angle are both altered on the texture layer \"${texture.name}\" with legacy mode enabled on the material \"${texture._parent?.name}\". The order of operations for scale and angle are now reversed on new materials. Please disable legacy mode on this material and adjust the texture stack properties as needed to preserve the desired behavior when legacy mode is deprecated.`\r\n );\r\n }\r\n }\r\n // Remove extra offset since the image will be tiled anyways\r\n drawAt.offsetX = drawAt.offsetX % width;\r\n drawAt.offsetY = drawAt.offsetY % height;\r\n\r\n // Initially set the draw size to cover the canvas\r\n let drawWidth = (ctxWidth - drawAt.offsetX) / texture.uScale;\r\n let drawHeight = (ctxHeight - drawAt.offsetY) / texture.vScale;\r\n\r\n // If the offset is greater than zero there will be a gap on the top or the left so we have to move the context over one tile\r\n if (drawAt.offsetX > 0) {\r\n context.translate(-width, 0);\r\n drawWidth += width;\r\n }\r\n if (drawAt.offsetY > 0) {\r\n context.translate(0, -height);\r\n drawHeight += height;\r\n }\r\n\r\n // add the offset and remove the pivot from earlier that was used to get the correct rotation and scaling matrix parts\r\n context.translate(drawAt.offsetX - pivotX, drawAt.offsetY - pivotY);\r\n\r\n // If the texture was scaled with a pivot, we need to draw the image larger to make sure there are no gaps\r\n if ((texture.uPivot !== 0 || texture.vPivot !== 0) && (texture.uScale !== 1 || texture.vScale !== 1)) {\r\n const scaleOffsetU = Math.abs(Math.ceil(texture.uPivot / texture.uScale)) * width;\r\n const scaleOffsetV = Math.abs(Math.ceil(texture.vPivot / texture.vScale)) * height;\r\n if (scaleOffsetU !== 0 || scaleOffsetV !== 0) {\r\n context.translate(-scaleOffsetU, -scaleOffsetV);\r\n drawWidth += scaleOffsetU * 2;\r\n drawHeight += scaleOffsetV * 2;\r\n }\r\n }\r\n\r\n // If the texture was rotated, we need to draw the image larger to make sure there are no gaps\r\n if (texture.wAngle !== 0) {\r\n const rotateOffsetU = Math.ceil(texture.uPivot / texture.uScale) * width;\r\n const rotateOffsetV = Math.ceil(texture.vPivot / texture.vScale) * height;\r\n if (rotateOffsetU > 0 || rotateOffsetV > 0) {\r\n context.translate(-rotateOffsetU, -rotateOffsetV);\r\n drawWidth += rotateOffsetU * 2;\r\n drawHeight += rotateOffsetV * 2;\r\n }\r\n }\r\n\r\n this.drawInBlendedContext(\r\n texture,\r\n context,\r\n parameters,\r\n innerContext => {\r\n innerContext.fillStyle = pattern;\r\n // Note that any rectangle start point outside of the box will not be rendered so all image transforms must be done on the canvas context\r\n innerContext.fillRect(0, 0, drawWidth, drawHeight);\r\n if (patternCanvas) {\r\n disposeWorkingCanvas(patternCanvas);\r\n }\r\n },\r\n operations\r\n );\r\n } else {\r\n context.translate(pivotX, pivotY);\r\n context.scale(texture.uScale, texture.vScale);\r\n context.rotate((texture.wAngle * Math.PI) / 180);\r\n context.translate(-pivotX, -pivotY);\r\n\r\n this.drawInBlendedContext(\r\n texture,\r\n context,\r\n parameters,\r\n innerContext => {\r\n innerContext.drawImage(\r\n source,\r\n drawAt.offsetX,\r\n drawAt.offsetY,\r\n drawAt.drawWidth,\r\n drawAt.drawHeight\r\n );\r\n },\r\n operations\r\n );\r\n\r\n if (texture._visualize) {\r\n context.strokeStyle = '#00FF00';\r\n context.strokeRect(texture.uOffset * ctxWidth, texture.vOffset * ctxHeight, ctxWidth, ctxHeight);\r\n }\r\n }\r\n\r\n // reset the context\r\n context.setTransform(1, 0, 0, 1, 0, 0);\r\n } else {\r\n // this block will never need to be masked because the layers using a clipping mask will never be the first layer\r\n if (Object.keys(operations).length > 0) {\r\n if (source instanceof HTMLCanvasElement) {\r\n source = this.filterImageInPlace(source, operations);\r\n } else {\r\n const tempCanvas = getWorkingCanvas(width, height);\r\n const tempCtx = tempCanvas.getContext('2d')!;\r\n tempCtx.drawImage(source, 0, 0);\r\n source = this.filterImageInPlace(tempCanvas, operations);\r\n }\r\n }\r\n context.drawImage(source, 0, 0, ctxWidth, ctxHeight);\r\n }\r\n\r\n return context;\r\n }\r\n}\r\n","import { getRotatedDimensions } from '@common-util/texture-util';\r\nimport { ColorFillTextureLayer } from '@models/classes/materials/color-fill-texture-layer';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass } from '@models/interfaces/tracked-class';\r\nimport { TextureParameters } from '../texture-stack';\r\nimport { IDrawOperations, TextureLayerRenderer } from './texture';\r\n\r\nexport class ColorFillTextureLayerRenderer extends TextureLayerRenderer {\r\n modelToken = Token.ColorFillTextureLayer;\r\n\r\n paintTexture(\r\n texture: ColorFillTextureLayer,\r\n context: CanvasRenderingContext2D,\r\n parameters: TextureParameters,\r\n operations: IDrawOperations\r\n ) {\r\n this.drawInBlendedContext(\r\n texture,\r\n context,\r\n parameters,\r\n innerContext => {\r\n innerContext.fillStyle = texture.color.cssString();\r\n innerContext.fillRect(0, 0, innerContext.canvas.width, innerContext.canvas.height);\r\n },\r\n operations\r\n );\r\n }\r\n\r\n textureHasContent() {\r\n return true;\r\n }\r\n\r\n getNativeSize(texture: ColorFillTextureLayer) {\r\n const width = texture.width,\r\n height = texture.height;\r\n\r\n if (texture.wAngle !== 0) {\r\n return getRotatedDimensions(width, height, texture.wAngle);\r\n } else {\r\n return { width, height };\r\n }\r\n }\r\n\r\n update3dInternal(node: ITrackedClass) {\r\n super.update3dInternal(node);\r\n }\r\n}\r\n","import { getRotatedDimensions } from '@common-util/texture-util';\r\nimport { DisplacementMapFeature } from '@models/classes';\r\nimport { FileTextureLayer } from '@models/classes/materials/file-texture-layer';\r\nimport { MaterialNode } from '@models/classes/materials/material-node';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass } from '@models/interfaces/tracked-class';\r\nimport { TextureParameters } from '../texture-stack';\r\nimport { IDrawOperations, TextureLayerRenderer } from './texture';\r\n\r\nexport class FileTextureLayerRenderer extends TextureLayerRenderer {\r\n modelToken = Token.FileTextureLayer;\r\n\r\n paintTexture(\r\n texture: FileTextureLayer,\r\n context: CanvasRenderingContext2D,\r\n parameters: TextureParameters,\r\n operations: IDrawOperations\r\n ) {\r\n if (texture._buffer) {\r\n return this.drawSourceToCanvas(texture, texture._buffer, context, parameters, operations);\r\n }\r\n }\r\n\r\n textureHasContent(node: FileTextureLayer) {\r\n return node.path !== '';\r\n }\r\n\r\n getNativeSize(texture: FileTextureLayer) {\r\n if (texture._buffer) {\r\n const width = Math.max(texture._buffer.width, 1),\r\n height = Math.max(texture._buffer.height, 1);\r\n\r\n if (texture.wAngle !== 0) {\r\n return getRotatedDimensions(width, height, texture.wAngle);\r\n } else {\r\n return { width, height };\r\n }\r\n }\r\n return { width: 1, height: 1 };\r\n }\r\n\r\n update3dInternal(node: ITrackedClass) {\r\n if (\r\n ((node.changes.has('height') || node.changes.has('width')) && node.path.endsWith('.svg')) ||\r\n node.changes.has('path') ||\r\n (node.changes.has('enabled') && !node._buffer) ||\r\n node.changes.has('_init')\r\n ) {\r\n node._needsImageUpdate = true;\r\n }\r\n\r\n if (node._needsImageUpdate && node.enabled && node.path && node._init) {\r\n this.loadTextureBuffer(node);\r\n }\r\n super.update3dInternal(node);\r\n }\r\n\r\n private loadTextureBuffer(node: ITrackedClass) {\r\n if (node._parent instanceof MaterialNode || node._parent instanceof DisplacementMapFeature) {\r\n node._needsImageUpdate = false;\r\n if (node._buffer) {\r\n node.trackChanges(false);\r\n node._buffer = undefined; // Remove the buffer so the texture stack doesn't think the image is already loaded if the image was changed\r\n node.trackChanges(true);\r\n }\r\n node._parent._materialService.getTextureImage(node).subscribe(resp => {\r\n node._buffer = resp.bitmap;\r\n node._bufferSource = resp.url;\r\n });\r\n } else {\r\n console.warn('Texure is not attached to a material node. Cannot load texture image bitmap.');\r\n }\r\n }\r\n}\r\n","export function base64ArrayBuffer(arrayBuffer: ArrayBuffer) {\r\n let base64 = '';\r\n const encodings = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';\r\n\r\n const bytes = new Uint8Array(arrayBuffer);\r\n const byteLength = bytes.byteLength;\r\n const byteRemainder = byteLength % 3;\r\n const mainLength = byteLength - byteRemainder;\r\n\r\n let a, b, c, d;\r\n let chunk;\r\n\r\n // Main loop deals with bytes in chunks of 3\r\n for (let i = 0; i < mainLength; i = i + 3) {\r\n // Combine the three bytes into a single integer\r\n chunk = (bytes[i] << 16) | (bytes[i + 1] << 8) | bytes[i + 2];\r\n\r\n // Use bitmasks to extract 6-bit segments from the triplet\r\n a = (chunk & 16515072) >> 18; // 16515072 = (2^6 - 1) << 18\r\n b = (chunk & 258048) >> 12; // 258048 = (2^6 - 1) << 12\r\n c = (chunk & 4032) >> 6; // 4032 = (2^6 - 1) << 6\r\n d = chunk & 63; // 63 = 2^6 - 1\r\n\r\n // Convert the raw binary segments to the appropriate ASCII encoding\r\n base64 += encodings[a] + encodings[b] + encodings[c] + encodings[d];\r\n }\r\n\r\n // Deal with the remaining bytes and padding\r\n if (byteRemainder == 1) {\r\n chunk = bytes[mainLength];\r\n\r\n a = (chunk & 252) >> 2; // 252 = (2^6 - 1) << 2\r\n\r\n // Set the 4 least significant bits to zero\r\n b = (chunk & 3) << 4; // 3 = 2^2 - 1\r\n\r\n base64 += encodings[a] + encodings[b] + '==';\r\n } else if (byteRemainder == 2) {\r\n chunk = (bytes[mainLength] << 8) | bytes[mainLength + 1];\r\n\r\n a = (chunk & 64512) >> 10; // 64512 = (2^6 - 1) << 10\r\n b = (chunk & 1008) >> 4; // 1008 = (2^6 - 1) << 4\r\n\r\n // Set the 2 least significant bits to zero\r\n c = (chunk & 15) << 2; // 15 = 2^4 - 1\r\n\r\n base64 += encodings[a] + encodings[b] + encodings[c] + '=';\r\n }\r\n\r\n return base64;\r\n}\r\n","import { base64ArrayBuffer } from '@common-util/buffer-util';\r\nimport { getRotatedDimensions, scaleImagePreserveAspect } from '@common-util/texture-util';\r\nimport { SvgTextureLayer } from '@models/classes/materials/svg-texture-layer';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass } from '@models/interfaces/tracked-class';\r\nimport { TextureParameters } from '../texture-stack';\r\nimport { IDrawOperations, TextureLayerRenderer } from './texture';\r\n\r\nexport class SvgTextureLayerRenderer extends TextureLayerRenderer {\r\n modelToken = Token.FileTextureLayer;\r\n imageElement = document.createElement('img');\r\n svgHelpers = new Map<\r\n string,\r\n {\r\n imgElement: HTMLImageElement;\r\n renderedWidth: number;\r\n renderedHeight: number;\r\n }\r\n >();\r\n\r\n paintTexture(\r\n texture: SvgTextureLayer,\r\n context: CanvasRenderingContext2D,\r\n parameters: TextureParameters,\r\n operations: IDrawOperations\r\n ) {\r\n const helper = this.svgHelpers.get(texture._dynamicId);\r\n\r\n if (\r\n texture._raster &&\r\n helper &&\r\n context.canvas.width === helper.renderedWidth &&\r\n context.canvas.height === helper.renderedHeight\r\n ) {\r\n return this.drawSourceToCanvas(texture, texture._raster, context, parameters, operations);\r\n } else {\r\n let fontLoadPromise: Promise | void>;\r\n\r\n if (texture.preloadFontStylesheet) {\r\n const fontFacesFound = [...texture.svgXml.matchAll(/font-family\\s*[=:]\\s*[\"'](.+?)[\"']/g)];\r\n const fontsToLoad: string[] = [];\r\n for (const match of fontFacesFound) {\r\n fontsToLoad.push(match[1]);\r\n }\r\n\r\n if (fontsToLoad.length > 0) {\r\n const manager = this.viewer.sceneNode._manager!;\r\n const fontService = manager.getService(Token.FontService);\r\n\r\n fontLoadPromise = fontService.loadFontData(texture.preloadFontStylesheet, fontsToLoad);\r\n } else {\r\n fontLoadPromise = Promise.resolve();\r\n }\r\n } else {\r\n fontLoadPromise = Promise.resolve();\r\n }\r\n\r\n fontLoadPromise\r\n .then(result =>\r\n this.generateSvg(texture, context.canvas.width, context.canvas.height, result || undefined)\r\n )\r\n .then(bitmap => {\r\n if (bitmap) {\r\n texture._raster = bitmap;\r\n }\r\n })\r\n .catch(() => {\r\n // Don't need to do anything\r\n });\r\n }\r\n }\r\n\r\n textureHasContent(node: SvgTextureLayer) {\r\n return !!node.svgXml;\r\n }\r\n\r\n getNativeSize(texture: SvgTextureLayer) {\r\n const width = texture.width,\r\n height = texture.height;\r\n\r\n if (texture.wAngle !== 0) {\r\n return getRotatedDimensions(width, height, texture.wAngle);\r\n } else {\r\n return { width, height };\r\n }\r\n }\r\n\r\n update3dInternal(node: ITrackedClass) {\r\n if (\r\n node.changes.has('svgXml') ||\r\n node.changes.has('width') ||\r\n node.changes.has('height') ||\r\n node.changes.has('preloadFontStylesheet')\r\n ) {\r\n node._raster = undefined;\r\n }\r\n super.update3dInternal(node);\r\n }\r\n\r\n generateSvg(texture: SvgTextureLayer, width: number, height: number, embedFonts?: Record) {\r\n if (texture.svgXml !== '' && width > 0 && height > 0) {\r\n let helper = this.svgHelpers.get(texture._dynamicId);\r\n\r\n if (!helper) {\r\n helper = {\r\n imgElement: document.createElement('img'),\r\n renderedHeight: height,\r\n renderedWidth: width,\r\n };\r\n this.svgHelpers.set(texture._dynamicId, helper);\r\n } else {\r\n helper.renderedHeight = height;\r\n helper.renderedWidth = width;\r\n }\r\n\r\n try {\r\n const imageElement = this.svgHelpers.get(texture._dynamicId)!.imgElement;\r\n\r\n let svgString = texture.svgXml;\r\n\r\n // Adding svg required tags\r\n if (!svgString.match(/` +\r\n svgString;\r\n }\r\n if (!svgString.match(/<\\?xml/)) {\r\n svgString = `` + svgString;\r\n }\r\n if (!svgString.match(/]*xmlns=[^>]*>/)) {\r\n svgString = svgString.replace(//m);\r\n if (!match) {\r\n console.warn('Invalid SVG');\r\n return Promise.reject();\r\n } else {\r\n if (embedFonts) {\r\n let fontStyles = '';\r\n for (const fontFamily in embedFonts) {\r\n const arrayBuffer = embedFonts[fontFamily];\r\n fontStyles += ` \r\n@font-face {\r\n font-family: \"${fontFamily}\";\r\n src: url(\"data:application/font-woff;charset=utf-8;base64,${base64ArrayBuffer(arrayBuffer)}\");\r\n}\r\n `;\r\n }\r\n const offset = (match.index || 0) + match[0].length;\r\n\r\n svgString =\r\n svgString.substring(0, offset) +\r\n `` +\r\n svgString.substring(offset);\r\n }\r\n\r\n const blobUrl = URL.createObjectURL(new Blob([svgString], { type: 'image/svg+xml;charset=utf-8' }));\r\n imageElement.src = blobUrl;\r\n\r\n const generateImagePromise = imageElement.decode().then(\r\n () => {\r\n // Don't use the img element from the closure so it can get garbage collected if this function isn't called!\r\n // There's no way to explicitly dispose of a DOM element\r\n const loadedImageElement = this.svgHelpers.get(texture._dynamicId)!.imgElement;\r\n\r\n // Figure out what resolution to actually draw the SVG at to fit within the envelope\r\n const scaled = scaleImagePreserveAspect(\r\n width,\r\n height,\r\n loadedImageElement.naturalWidth,\r\n loadedImageElement.naturalHeight\r\n );\r\n\r\n return createImageBitmap(loadedImageElement, {\r\n resizeWidth: scaled.drawWidth,\r\n resizeHeight: scaled.drawHeight,\r\n });\r\n },\r\n err => {\r\n console.warn('Could not decode SVG XML: ', svgString);\r\n }\r\n );\r\n\r\n this.viewer.deferSceneReady(generateImagePromise);\r\n return generateImagePromise;\r\n }\r\n } catch (e) {\r\n console.warn('Could not generate SVG Texture: ', e);\r\n Promise.reject(e);\r\n }\r\n }\r\n\r\n return Promise.reject();\r\n }\r\n\r\n delete3d(node: ITrackedClass) {\r\n const helper = this.svgHelpers.get(node._dynamicId);\r\n if (helper) {\r\n this.svgHelpers.delete(node._dynamicId);\r\n }\r\n super.delete3d(node);\r\n }\r\n}\r\n","import { PBRMaterial } from '@babylonjs/core/Materials/PBR/pbrMaterial';\r\nimport { DynamicTexture, Texture } from '@babylonjs/core/Materials/Textures';\r\nimport { Material } from '@babylonjs/core/Materials/material';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { getBinClampedSize } from '@common-util/texture-util';\r\nimport { instanceOf } from '@models/classes/decorators/model-utils';\r\nimport { eMaterialWorkflow, eTextureChannel } from '@models/classes/enums';\r\nimport { FeatureWithTextures, isFeatureNode } from '@models/classes/features/feature';\r\nimport { FileTextureLayer } from '@models/classes/materials/file-texture-layer';\r\nimport { MaterialNode, isMaterialNode } from '@models/classes/materials/material-node';\r\nimport { TextureLayer } from '@models/classes/materials/texture-layer';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass, isTrackedClass } from '@models/interfaces/tracked-class';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { Subject, merge } from 'rxjs';\r\nimport { debounceTime, takeWhile, throttleTime } from 'rxjs/operators';\r\nimport { Renderer } from './renderer';\r\nimport { RendererManager } from './renderer-manager';\r\nimport { SamplingModeMap } from './scene';\r\nimport { FileTextureLayerRenderer } from './textures';\r\nimport { ColorFillTextureLayerRenderer } from './textures/color-fill-texture';\r\nimport { SvgTextureLayerRenderer } from './textures/svg-texture';\r\nimport { TextureLayerRenderer } from './textures/texture';\r\n\r\nexport type TextureParameters = {\r\n strength: number;\r\n hasTransparency: boolean;\r\n};\r\n\r\nconst textureChannelMap: Record = {\r\n // General\r\n [eTextureChannel.emissive]: 'emissiveTexture',\r\n [eTextureChannel.ambientOcclusion]: 'lightmapTexture',\r\n\r\n // PBR\r\n [eTextureChannel.albedo]: 'albedoTexture',\r\n [eTextureChannel.normal]: 'bumpTexture',\r\n [eTextureChannel.ambient]: 'ambientTexture',\r\n [eTextureChannel.opacity]: 'opacityTexture',\r\n [eTextureChannel.metallic]: 'metallicTexture',\r\n [eTextureChannel.reflection]: 'reflectionTexture',\r\n [eTextureChannel.specular]: 'reflectivityTexture',\r\n [eTextureChannel.roughness]: 'microSurfaceTexture',\r\n [eTextureChannel.glossiness]: 'microSurfaceTexture',\r\n [eTextureChannel.anisotropyTangent]: ['anisotropy', 'texture'],\r\n [eTextureChannel.clearCoatNormal]: ['clearCoat', 'bumpTexture'],\r\n [eTextureChannel.subsurfaceThickness]: ['subSurface', 'thicknessTexture'],\r\n};\r\nconst maskingCanvas = document.createElement('canvas');\r\n\r\nexport class TextureStackRenderer extends Renderer {\r\n modelToken = Token.TextureLayer;\r\n\r\n queuedTexturesToProcess = new Map>();\r\n queueSize = new Subject();\r\n renderers = new Map>();\r\n noMipMapFallback = false;\r\n\r\n private updateTextureStackRequest = new Subject();\r\n private deferCalled = false;\r\n\r\n constructor(viewer: KbViewer, scene: Scene, private rendererManager: RendererManager) {\r\n super(viewer, scene);\r\n const fileTexture = new FileTextureLayerRenderer(viewer, scene);\r\n const svgTexture = new SvgTextureLayerRenderer(viewer, scene);\r\n const colorTexture = new ColorFillTextureLayerRenderer(viewer, scene);\r\n this.renderers.set(Token.FileTextureLayer, fileTexture);\r\n this.renderers.set(Token.SvgTextureLayer, svgTexture);\r\n this.renderers.set(Token.ColorFillTextureLayer, colorTexture);\r\n\r\n /**\r\n * See render.ts comment - this delay must be added to the render as a change can take up to this delay time to be displayed.\r\n * We don't want to repeatedly processes texture changes coming in quick succession, which often happens if there are many textures being loaded at once.\r\n * */\r\n const throttleDebounceTime = 800;\r\n merge(\r\n this.updateTextureStackRequest.pipe(throttleTime(throttleDebounceTime)),\r\n this.updateTextureStackRequest.pipe(debounceTime(throttleDebounceTime))\r\n ).subscribe(() => {\r\n this.updateTextureStacksExec();\r\n });\r\n\r\n // This code works for forcing intel gpus to not use mipmaps, but some scenes don't have artifacting issues and need mipmaps. So\r\n // In the future we can introduce a scene setting to enable the intel fallback\r\n // if (this.viewer.glInfo.match(/Intel.*Iris.*Xe/)) {\r\n // this.noMipMapFallback = true;\r\n // }\r\n }\r\n\r\n materialDeleted(node: MaterialNode) {\r\n for (const channel in eTextureChannel) {\r\n const textureId = this.getTexture3dId(node, channel as eTextureChannel);\r\n const texture = this.babylonMap.get(textureId);\r\n if (texture) {\r\n this.disposeTexture(texture);\r\n this.babylonMap.delete(this.getTexture3dId(node, channel as eTextureChannel));\r\n }\r\n this.babylonMap.delete(textureId);\r\n }\r\n }\r\n\r\n /** This should only be called from the editor. Has a very heavy performance impact. */\r\n forceRebuildTexture(materialNode: MaterialNode | FeatureWithTextures, channel?: eTextureChannel) {\r\n if (channel) {\r\n this.resetTextureStack(materialNode, channel);\r\n } else {\r\n for (const channel in eTextureChannel) {\r\n this.resetTextureStack(materialNode, channel as eTextureChannel);\r\n }\r\n }\r\n this.updateTextureStacksInternal(materialNode);\r\n }\r\n\r\n resetTextureStack(\r\n texturedNode: MaterialNode | FeatureWithTextures,\r\n channel: eTextureChannel,\r\n setParameters?: Partial\r\n ) {\r\n let material: Material | undefined;\r\n let materialNode: MaterialNode | undefined;\r\n\r\n if (isMaterialNode(texturedNode)) {\r\n materialNode = texturedNode;\r\n if (!materialNode._initialized) {\r\n return;\r\n }\r\n material = this.rendererManager.material.get3d(materialNode)!;\r\n if (!material) {\r\n // There might not be a material in babylon if the material was deleted prior to this renderer running.\r\n // console.warn(`material not generated from ${materialNode.name}`);\r\n return;\r\n }\r\n }\r\n\r\n const hasTextureInChannel = !!texturedNode.textures.find(t => t.channel === channel);\r\n if (!hasTextureInChannel) {\r\n return;\r\n }\r\n\r\n let channelsToProcess = this.queuedTexturesToProcess.get(texturedNode);\r\n if (!channelsToProcess) {\r\n channelsToProcess = new Map();\r\n this.queuedTexturesToProcess.set(texturedNode, channelsToProcess);\r\n }\r\n\r\n // Here we don't want to filter out textures with no data because they cam still be used to set the canvas size\r\n const textureLayers = this.getActiveLayers(texturedNode, channel);\r\n\r\n // Register that the channel needs to be processed.\r\n // Figure out the highest strength among the texture layers so we can\r\n // calculate the strength of each one as a fraction of that.\r\n let channelStrength = 0;\r\n textureLayers.forEach(textureLayer => {\r\n channelStrength = Math.max(channelStrength, textureLayer.strength);\r\n });\r\n\r\n const existingProcess = channelsToProcess.get(channel);\r\n channelsToProcess.set(channel, {\r\n strength: channelStrength,\r\n hasTransparency: materialNode ? materialNode.hasTransparency : false,\r\n ...(existingProcess || {}),\r\n ...(setParameters || {}),\r\n });\r\n\r\n const propName3d = textureChannelMap[channel as eTextureChannel] as keyof Material;\r\n\r\n if (material && materialNode) {\r\n // Reset the material to re-run the texture stack (or clear the texture if there are no textures anymore)\r\n if (textureLayers.length === 0) {\r\n // If no textures are using this channel anymore, then remove it from the material\r\n if (Array.isArray(propName3d)) {\r\n const res = getObjectPropertyAtPath(material, propName3d, false);\r\n if (res.obj && res.obj[res.prop]) {\r\n const texture = res.obj[res.prop];\r\n if (texture instanceof Texture) {\r\n this.disposeTexture(texture);\r\n res.obj[res.prop] = null;\r\n this.babylonMap.delete(this.getTexture3dId(materialNode, channel));\r\n }\r\n }\r\n } else {\r\n if (material && material[propName3d]) {\r\n (material[propName3d] as Texture).dispose();\r\n (material[propName3d] as null) = null;\r\n this.babylonMap.delete(this.getTexture3dId(materialNode, channel));\r\n }\r\n }\r\n } else {\r\n const texture = this.getTexture(materialNode, channel);\r\n\r\n if (texture instanceof DynamicTexture) {\r\n const baseImageSize = this.getRenderer(textureLayers[0]).getNativeSize(textureLayers[0]);\r\n const context = texture.getContext();\r\n\r\n if (this.scene.getEngine().webGLVersion < 2) {\r\n const clampedSize = getBinClampedSize(baseImageSize.width, baseImageSize.height);\r\n context.canvas.width = clampedSize;\r\n context.canvas.height = clampedSize;\r\n } else {\r\n context.canvas.width = baseImageSize.width;\r\n context.canvas.height = baseImageSize.height;\r\n }\r\n context.clearRect(0, 0, context.canvas.width, context.canvas.height);\r\n }\r\n }\r\n }\r\n }\r\n\r\n update3dInternal(feature: ITrackedClass) {\r\n if (isTrackedClass(feature)) {\r\n if (!feature.enabled || feature._suppressed) {\r\n if (feature.changes.has('enabled') || feature.changes.has('_suppressed')) {\r\n this.delete3d(feature);\r\n }\r\n\r\n return;\r\n }\r\n\r\n const renderer = this.renderers.get(Token.withKey(feature.$type));\r\n if (!renderer) {\r\n throw Error('no renderer for feature');\r\n }\r\n if (feature._parent && isFeatureNode(feature._parent)) {\r\n feature.init(feature._parent);\r\n }\r\n renderer.update3d(feature);\r\n\r\n if (feature._parent && isFeatureNode(feature._parent) && isTrackedClass(feature._parent)) {\r\n feature._parent.setPropertyIsModified('textures', true);\r\n }\r\n\r\n if (feature._parent) {\r\n this.resetTextureStack(feature._parent, feature.channel);\r\n }\r\n }\r\n }\r\n\r\n updateTextureStacks() {\r\n if (!this.deferCalled && this.queuedTexturesToProcess.size > 0) {\r\n const queueLive = this.queueSize.pipe(takeWhile(size => size > 0));\r\n const p = queueLive.toPromise();\r\n this.viewer.deferSceneReady(p);\r\n this.deferCalled = true;\r\n }\r\n for (const node of this.queuedTexturesToProcess.keys()) {\r\n if (isFeatureNode(node) || node._initialized) {\r\n for (const texture of node.textures) {\r\n if (!texture._init) {\r\n texture.init(node);\r\n }\r\n }\r\n }\r\n }\r\n this.updateTextureStackRequest.next();\r\n }\r\n\r\n private updateTextureStacksExec() {\r\n this.queueSize.next(this.queuedTexturesToProcess.size);\r\n\r\n for (const node of this.queuedTexturesToProcess.keys()) {\r\n this.updateTextureStacksInternal(node);\r\n\r\n if (this.queuedTexturesToProcess.size > 0) {\r\n this.updateTextureStacks();\r\n }\r\n // Low perf code to prevent iOS from erroring out. Doesn't seem necessary for now\r\n // return;\r\n // End low perf code\r\n }\r\n\r\n this.queueSize.next(this.queuedTexturesToProcess.size);\r\n }\r\n\r\n private updateTextureStacksInternal(node: MaterialNode | FeatureWithTextures) {\r\n if (this.queuedTexturesToProcess.has(node)) {\r\n if (isMaterialNode(node)) {\r\n const channels = this.queuedTexturesToProcess.get(node);\r\n const material = this.rendererManager.material.get3d(node);\r\n\r\n if (!channels || !material) {\r\n return;\r\n }\r\n\r\n // Process texture stack\r\n channels.forEach((parameters, channel) => {\r\n // Put the texture in the right place on the material\r\n const propName3d = textureChannelMap[channel as eTextureChannel];\r\n if (instanceOf(node, Token.AdvancedMaterialNode)) {\r\n if (\r\n node.workflowMode === eMaterialWorkflow.metallicRoughness &&\r\n channel === eTextureChannel.glossiness\r\n ) {\r\n return;\r\n } else if (\r\n node.workflowMode === eMaterialWorkflow.specularGlossiness &&\r\n channel === eTextureChannel.roughness\r\n ) {\r\n return;\r\n }\r\n }\r\n\r\n const texture = this.getTexture(node, channel);\r\n // If texture is undefined, that means that there is a missing setting/layers that means we don't need a texture for this channel\r\n if (texture) {\r\n // Paint the image onto the texture\r\n // We don't want to count layers with no content in the stack, like file textures with no file or svg textures with no data\r\n const textureLayers = this.getActiveLayers(node, channel).filter(node => {\r\n const renderer = this.getRenderer(node);\r\n return renderer.textureHasContent(node);\r\n });\r\n\r\n if (textureLayers.length > 0) {\r\n // Put the texture in the right place on the material if it's not already\r\n if (Array.isArray(propName3d)) {\r\n const res = getObjectPropertyAtPath(material, propName3d);\r\n if (res.obj[res.prop] !== texture) {\r\n res.obj[res.prop] = texture;\r\n }\r\n } else {\r\n if (material[propName3d as keyof Material] !== texture) {\r\n (material[propName3d as keyof Material] as Texture) = texture;\r\n }\r\n }\r\n\r\n if (texture instanceof DynamicTexture) {\r\n const context = texture.getContext();\r\n\r\n if (!(context instanceof CanvasRenderingContext2D)) {\r\n throw Error('Unexpected rendering context');\r\n }\r\n if (channel !== eTextureChannel.albedo) {\r\n // Fill texture with color that makes sense depending on the texture channel\r\n switch (channel) {\r\n case eTextureChannel.normal: {\r\n context.fillStyle = 'rgb(128, 128, 255)';\r\n break;\r\n }\r\n\r\n case eTextureChannel.opacity:\r\n case eTextureChannel.ambient:\r\n case eTextureChannel.ambientOcclusion: {\r\n context.fillStyle = 'rgb(255, 255, 255)';\r\n break;\r\n }\r\n\r\n default: {\r\n context.fillStyle = 'rgb(0, 0, 0)';\r\n break;\r\n }\r\n }\r\n\r\n context.fillRect(0, 0, context.canvas.width, context.canvas.height);\r\n }\r\n this.paintTextureOnCanvas(textureLayers, context, parameters);\r\n\r\n updateTextureProps(texture, parameters);\r\n\r\n texture.update(true); // invertY\r\n } else {\r\n // If texture is not a dynamic texture, then it's a basic texture and there is only one texture layer. We just have to make sure the path is correct\r\n const fileTextureLayer = textureLayers[0];\r\n if (!(fileTextureLayer instanceof FileTextureLayer)) {\r\n throw Error('Unexpected texture type');\r\n } else {\r\n if (fileTextureLayer._parent instanceof MaterialNode) {\r\n const path = fileTextureLayer._bufferSource || '';\r\n if (texture.url !== path && fileTextureLayer._buffer) {\r\n texture.updateURL(path, (fileTextureLayer._buffer || null) as any);\r\n }\r\n }\r\n\r\n updateTextureProps(texture, parameters);\r\n }\r\n }\r\n }\r\n }\r\n });\r\n } else {\r\n if (isTrackedClass(node)) {\r\n node.setPropertyIsModified('textures', true);\r\n this.queuedTexturesToProcess.delete(node);\r\n return;\r\n }\r\n }\r\n }\r\n if (isMaterialNode(node) && isTrackedClass(node)) {\r\n this.applyUvProperties(node);\r\n } else {\r\n throw Error('Material is not being tracked');\r\n }\r\n this.queuedTexturesToProcess.delete(node);\r\n\r\n function updateTextureProps(texture: Texture, parameters: TextureParameters) {\r\n texture.level = parameters.strength;\r\n texture.hasAlpha = parameters.hasTransparency;\r\n }\r\n }\r\n\r\n delete3d(feature: ITrackedClass) {\r\n if (feature._parent) {\r\n this.resetTextureStack(feature._parent, feature.channel);\r\n }\r\n\r\n const renderer = this.renderers.get(Token.withKey(feature.$type));\r\n if (!renderer) {\r\n throw Error('no renderer for feature');\r\n }\r\n renderer.delete3d(feature);\r\n }\r\n\r\n get3d(texture: TextureLayer) {\r\n if (!texture._parent) {\r\n return undefined;\r\n }\r\n if (texture._parent instanceof MaterialNode) {\r\n return this.babylonMap.get(this.getTexture3dId(texture._parent, texture.channel));\r\n }\r\n }\r\n\r\n set3d(texture: TextureLayer, babylonObject: Texture) {\r\n if (texture._parent) {\r\n if (texture._parent instanceof MaterialNode) {\r\n this.babylonMap.set(this.getTexture3dId(texture._parent, texture.channel), babylonObject);\r\n }\r\n }\r\n }\r\n\r\n dispose() {\r\n this.queuedTexturesToProcess.clear();\r\n this.renderers.forEach(renderer => renderer.dispose());\r\n this.updateTextureStackRequest.complete();\r\n super.dispose();\r\n }\r\n\r\n setTransparency(materialNode: MaterialNode | FeatureWithTextures, hasTransparency: boolean) {\r\n // TODO: limit this to only albedo textures for now. Will need testing.\r\n this.resetTextureStack(materialNode, eTextureChannel.albedo, { hasTransparency });\r\n }\r\n\r\n applyUvProperties(materialNode: ITrackedClass) {\r\n const material = this.rendererManager.material.get3d(materialNode);\r\n if (material) {\r\n for (const channel in textureChannelMap) {\r\n const firstTextureLayer = materialNode.textures.find(\r\n t => t.channel === channel\r\n ) as ITrackedClass;\r\n\r\n const propName3d = textureChannelMap[channel as eTextureChannel] as keyof Material;\r\n\r\n let texture: Texture;\r\n if (Array.isArray(propName3d)) {\r\n const res = getObjectPropertyAtPath(material, propName3d);\r\n texture = res.obj[res.prop];\r\n } else {\r\n texture = material[propName3d];\r\n }\r\n\r\n if (texture && firstTextureLayer) {\r\n if (!(texture instanceof Texture)) {\r\n throw Error(propName3d.toString() + ' is not a texture');\r\n }\r\n texture.uOffset = firstTextureLayer.uOffset;\r\n texture.vOffset = firstTextureLayer.vOffset;\r\n texture.uScale = firstTextureLayer.uScale;\r\n texture.vScale = firstTextureLayer.vScale;\r\n texture.uAng = (firstTextureLayer.uAngle * Math.PI) / 180;\r\n texture.vAng = (firstTextureLayer.vAngle * Math.PI) / 180;\r\n texture.wAng = (firstTextureLayer.wAngle * Math.PI) / 180;\r\n texture.uRotationCenter = firstTextureLayer.uPivot;\r\n texture.vRotationCenter = firstTextureLayer.vPivot;\r\n texture.wRotationCenter = firstTextureLayer.wPivot;\r\n // if (firstTextureLayer.uWrap === eTextureWrapAddressMode.clamp) {\r\n // texture.wrapU = Texture.CLAMP_ADDRESSMODE;\r\n // } else {\r\n // texture.wrapU = Texture.WRAP_ADDRESSMODE;\r\n // }\r\n // if (firstTextureLayer.vWrap === eTextureWrapAddressMode.clamp) {\r\n // texture.wrapV = Texture.CLAMP_ADDRESSMODE;\r\n // } else {\r\n // texture.wrapV = Texture.WRAP_ADDRESSMODE;\r\n // }\r\n\r\n // For some reason wrap addressmode is not getting converted correctly. Repeat is the safer option so\r\n // Let's hardcode repeat for now\r\n texture.wrapU = Texture.WRAP_ADDRESSMODE;\r\n texture.wrapV = Texture.WRAP_ADDRESSMODE;\r\n\r\n if (!(texture instanceof DynamicTexture)) {\r\n texture.vScale *= -1;\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n paintTextureOnCanvas(textureLayers: TextureLayer[], context: any, parameters: TextureParameters) {\r\n let clippingTexture: TextureLayer | null = null;\r\n\r\n for (let i = 0; i < textureLayers.length; i++) {\r\n if (!textureLayers[i]._init) {\r\n continue;\r\n }\r\n\r\n const textureLayer = textureLayers[i];\r\n const renderer = this.getRenderer(textureLayer);\r\n\r\n // Continue clipping with the same texture as long as useClippingMask is true.\r\n // When there's a texture that's not clipping, that can potentially be a new clipping mask.\r\n if (!textureLayer.useClippingMask) {\r\n clippingTexture = null;\r\n }\r\n\r\n const maskingCtx = maskingCanvas.getContext('2d', {\r\n willReadFrequently: true,\r\n })!;\r\n if (textureLayers[i + 1] && textureLayers[i + 1].useClippingMask && !clippingTexture) {\r\n maskingCanvas.width = context.canvas.width;\r\n maskingCanvas.height = context.canvas.height;\r\n clippingTexture = textureLayer;\r\n } else if (textureLayer.useClippingMask && clippingTexture) {\r\n const clippingTextureRenderer = this.getRenderer(clippingTexture);\r\n maskingCtx.clearRect(0, 0, maskingCanvas.width, maskingCanvas.height);\r\n clippingTextureRenderer.paintTexture(\r\n clippingTexture,\r\n maskingCtx,\r\n { strength: 1, hasTransparency: true },\r\n {}\r\n );\r\n renderer.paintTexture(textureLayer, context, parameters, {\r\n mask: { context: maskingCtx },\r\n });\r\n } else {\r\n renderer.paintTexture(textureLayer, context, parameters, {});\r\n }\r\n }\r\n return context;\r\n }\r\n\r\n getActiveLayers(\r\n materialNode: MaterialNode | FeatureWithTextures,\r\n channel: eTextureChannel,\r\n includeSupressed = false\r\n ) {\r\n return materialNode.textures.filter(\r\n texture => texture.channel === channel && texture.enabled && (includeSupressed || !texture._suppressed)\r\n );\r\n }\r\n\r\n getRenderer(textureLayer: TextureLayer) {\r\n const renderer = this.renderers.get(Token.withKey(textureLayer.$type));\r\n if (!renderer) {\r\n throw Error('no renderer for texture layer');\r\n }\r\n return renderer;\r\n }\r\n\r\n private canUseSimpleTexture(activeLayers: TextureLayer[]) {\r\n if (activeLayers.length === 1) {\r\n const layer = activeLayers[0];\r\n return layer instanceof FileTextureLayer;\r\n }\r\n return false;\r\n }\r\n\r\n private disposeTexture(texture: Texture) {\r\n // Resizing textures before they are ready results in an error. Remove this code for now\r\n // if (texture instanceof DynamicTexture) {\r\n // texture.scaleTo(1, 1); // Prevent memory leaks in Safari\r\n // }\r\n texture.dispose();\r\n }\r\n\r\n private getTexture3dId(node: MaterialNode, channel: eTextureChannel) {\r\n return `${node._dynamicId}_${channel}`;\r\n }\r\n\r\n private getTexture(node: MaterialNode, channel: eTextureChannel) {\r\n const rootScene = this.viewer.sceneNode;\r\n const activeLayers = this.getActiveLayers(node, channel, true);\r\n const canUseSimpleTexture = this.canUseSimpleTexture(activeLayers);\r\n\r\n let texture = this.babylonMap.get(this.getTexture3dId(node, channel));\r\n const originalTexture = texture;\r\n\r\n if (originalTexture) {\r\n if (originalTexture instanceof DynamicTexture && canUseSimpleTexture) {\r\n this.disposeTexture(originalTexture);\r\n } else if (activeLayers.length > 1 && !(texture instanceof DynamicTexture)) {\r\n this.disposeTexture(originalTexture);\r\n } else {\r\n return originalTexture;\r\n }\r\n }\r\n\r\n if (canUseSimpleTexture) {\r\n const fileTexture = activeLayers[0] as FileTextureLayer;\r\n if (fileTexture.path && fileTexture._buffer && fileTexture._parent instanceof MaterialNode) {\r\n texture = new Texture(\r\n fileTexture._bufferSource || '',\r\n this.scene,\r\n this.noMipMapFallback,\r\n false,\r\n SamplingModeMap[rootScene.textureSamplingMode],\r\n null,\r\n null,\r\n fileTexture._buffer\r\n );\r\n texture.name = this.getTexture3dId(node, channel);\r\n } else {\r\n return undefined;\r\n }\r\n } else {\r\n texture = new DynamicTexture(\r\n this.getTexture3dId(node, channel),\r\n 256,\r\n this.scene,\r\n !this.noMipMapFallback,\r\n SamplingModeMap[rootScene.textureSamplingMode]\r\n );\r\n }\r\n\r\n texture.homogeneousRotationInUVTransform = true;\r\n this.babylonMap.set(this.getTexture3dId(node, channel), texture);\r\n\r\n // set initial values\r\n if (originalTexture) {\r\n texture.hasAlpha = originalTexture.hasAlpha;\r\n texture.level = originalTexture.level;\r\n texture.getAlphaFromRGB = originalTexture.getAlphaFromRGB;\r\n } else {\r\n texture.hasAlpha = false;\r\n if (channel === eTextureChannel.opacity) {\r\n texture.getAlphaFromRGB = true;\r\n } else {\r\n texture.getAlphaFromRGB = false;\r\n }\r\n texture.level = 1;\r\n }\r\n texture.coordinatesIndex = 0;\r\n texture.coordinatesMode = 0;\r\n texture.anisotropicFilteringLevel = 4;\r\n\r\n if (texture instanceof Texture) {\r\n texture.uOffset = 0; // Define an offset on the texture to offset the u coordinates of the UVs\r\n texture.vOffset = 0; // Define an offset on the texture to offset the v coordinates of the UVs\r\n texture.uScale = -1; // Define an offset on the texture to scale the u coordinates of the UVs\r\n texture.vScale = 1; // Define an offset on the texture to scale the v coordinates of the UVs\r\n texture.uAng = 0; // Define an offset on the texture to rotate around the u coordinates of the UVs\r\n texture.vAng = 0; // Define an offset on the texture to rotate around the v coordinates of the UVs\r\n texture.wAng = 0; // Define an offset on the texture to rotate around the w coordinates of the UVs (in case of 3d texture)\r\n texture.uRotationCenter = 0.5; // Defines the center of rotation (U)\r\n texture.vRotationCenter = 0.5; // Defines the center of rotation (V)\r\n texture.wRotationCenter = 0.5; // Defin\r\n texture.isBlocking = false;\r\n }\r\n\r\n return texture;\r\n }\r\n}\r\n\r\nfunction getObjectPropertyAtPath(_obj: T, path: string[], throwOnNotFound = true) {\r\n let obj = _obj as any;\r\n const propPath = path.slice().reverse();\r\n while (propPath.length > 1 && obj) {\r\n const prop = propPath.pop() as any;\r\n if (prop in obj) {\r\n obj = obj[prop];\r\n } else {\r\n obj = null;\r\n }\r\n }\r\n if (throwOnNotFound && !obj) {\r\n throw Error('Texture property not found for ' + path.toString());\r\n }\r\n\r\n return { obj, prop: propPath[0] as string };\r\n}\r\n","import { Camera } from '@babylonjs/core/Cameras/camera';\r\nimport { Token } from '@models/classes/token';\r\nimport { ViewpointNode } from '@models/classes/viewpoints/viewpoint-node';\r\nimport { ITrackedClass } from '@models/interfaces';\r\nimport { Renderer } from './renderer';\r\n\r\nexport class ViewpointRenderer extends Renderer {\r\n modelToken = Token.ViewpointNode;\r\n update3dInternal(node: ITrackedClass) {}\r\n}\r\n","import { Scene } from '@babylonjs/core/scene';\r\nimport { eMaterialLocation, eRenderMode } from '@models/classes/enums';\r\nimport { ArrayMap } from '@models/classes/handlers';\r\nimport { Kb3dObject } from '@models/classes/kb3d-object';\r\nimport { MaterialNode } from '@models/classes/materials/material-node';\r\nimport { TextureLayer } from '@models/classes/materials/texture-layer';\r\nimport { SceneNode } from '@models/classes/scene-node';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass } from '@models/interfaces';\r\nimport { SyncService } from '@models/services/sync-service';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { AnnotationRenderer } from './annotation';\r\nimport { DimensionRenderer } from './dimension';\r\nimport { FeatureStackRenderer } from './feature-stack';\r\nimport { ConnectorRenderer } from './features';\r\nimport { SketchControlPointRenderer } from './features/sketch-control-point';\r\nimport { LightRenderer } from './light';\r\nimport { MateRenderer } from './mate';\r\nimport { MaterialRenderer } from './material';\r\nimport { MeshRenderer } from './mesh';\r\nimport { PathRenderer } from './path';\r\nimport { MateProcessor } from './processors/mate-processor';\r\nimport { Processor } from './processors/processor';\r\nimport { ReferenceProcessor } from './processors/reference-processor';\r\nimport { SketchProcessor } from './processors/sketch-processor';\r\nimport { MAX_RENDER_TIME, Renderer } from './renderer';\r\nimport { SceneRenderer } from './scene';\r\nimport { ShadowManager } from './shadow-manager';\r\nimport { TextureStackRenderer } from './texture-stack';\r\nimport { ViewpointRenderer } from './viewpoint';\r\n\r\nlet unprocessedChangesCount = 0;\r\n\r\nexport class ChangeMap extends ArrayMap> {}\r\n\r\nexport class RendererManager {\r\n deferredChanges: Array> = [];\r\n maxRenderTime = MAX_RENDER_TIME;\r\n renderers: Set>;\r\n sceneRenderer: SceneRenderer;\r\n light: LightRenderer;\r\n material: MaterialRenderer;\r\n mesh: MeshRenderer;\r\n textures: TextureStackRenderer;\r\n viewpoint: ViewpointRenderer;\r\n features: FeatureStackRenderer;\r\n connector: ConnectorRenderer;\r\n controlPointRenderer: SketchControlPointRenderer;\r\n annotation: AnnotationRenderer;\r\n dimension: DimensionRenderer;\r\n mate: MateRenderer;\r\n path: PathRenderer;\r\n\r\n shadowManager: ShadowManager;\r\n processors = new Array();\r\n mateProcessor: MateProcessor;\r\n referenceProcessor: ReferenceProcessor;\r\n sketchProcessor: SketchProcessor;\r\n\r\n private forcedRenderingMode: eRenderMode | undefined;\r\n\r\n getRenderMode() {\r\n return this.material.renderMode;\r\n }\r\n /** Do not use this function directly. Call setRenderMode on the viewer instead so the correct sceneNode is used. */\r\n setRenderMode(mode: eRenderMode, sceneNode: SceneNode, force?: boolean) {\r\n if (force) {\r\n this.forcedRenderingMode = mode;\r\n }\r\n this.material.renderMode = this.forcedRenderingMode || mode;\r\n this.mesh.setRenderMode(this.forcedRenderingMode || mode, sceneNode);\r\n }\r\n\r\n constructor(public viewer: KbViewer, public scene: Scene) {\r\n this.shadowManager = new ShadowManager(viewer.bbCache);\r\n this.sceneRenderer = new SceneRenderer(viewer, scene, viewer.gui, this);\r\n this.textures = new TextureStackRenderer(viewer, scene, this);\r\n this.material = new MaterialRenderer(viewer, scene, this.textures);\r\n this.light = new LightRenderer(viewer, scene, this.shadowManager);\r\n this.viewpoint = new ViewpointRenderer(viewer, scene);\r\n this.mesh = new MeshRenderer(viewer, scene, this, this.shadowManager);\r\n this.features = new FeatureStackRenderer(viewer, scene, this);\r\n this.connector = new ConnectorRenderer(viewer, scene, this);\r\n this.controlPointRenderer = new SketchControlPointRenderer(viewer, scene, this);\r\n this.annotation = new AnnotationRenderer(viewer, scene, this);\r\n this.dimension = new DimensionRenderer(viewer, scene, this);\r\n this.mate = new MateRenderer(viewer, scene);\r\n this.path = new PathRenderer(viewer, scene);\r\n\r\n // Order is important here, it defines the order in which renderers will be run\r\n this.renderers = new Set>([\r\n this.sceneRenderer,\r\n this.material,\r\n this.textures,\r\n this.viewpoint,\r\n this.path,\r\n this.mesh,\r\n this.light,\r\n this.features,\r\n this.connector,\r\n this.controlPointRenderer,\r\n this.annotation,\r\n this.dimension,\r\n this.mate,\r\n ]);\r\n\r\n this.referenceProcessor = new ReferenceProcessor(viewer);\r\n this.mateProcessor = new MateProcessor(\r\n viewer,\r\n scene,\r\n this.connector,\r\n this.controlPointRenderer,\r\n this.referenceProcessor\r\n );\r\n this.sketchProcessor = new SketchProcessor(this.controlPointRenderer);\r\n this.processors.push(this.mateProcessor);\r\n this.processors.push(this.referenceProcessor);\r\n this.processors.push(this.sketchProcessor);\r\n }\r\n\r\n isStable() {\r\n return (\r\n !SyncService.hasChanges &&\r\n this.features.queuedFeaturesToProcess.size === 0 &&\r\n this.textures.queuedTexturesToProcess.size === 0\r\n );\r\n }\r\n\r\n render() {\r\n let changes: ChangeMap;\r\n if (this.deferredChanges.length > 0) {\r\n changes = this.getChangeMap(this.deferredChanges);\r\n } else {\r\n for (const p of this.processors) {\r\n p.beforeRender();\r\n }\r\n // run deletions first\r\n const deletions = this.getChangeMap(SyncService.getDeletedNodes(), true);\r\n for (const p of this.processors) {\r\n p.beforeDeletions(deletions);\r\n }\r\n this.runChanges(deletions, (renderer, change) => renderer.delete3d(change));\r\n for (const p of this.processors) {\r\n p.afterDeletions(deletions);\r\n }\r\n //then run updates\r\n const newChanges = SyncService.getChangedNodes();\r\n changes = this.getChangeMap(newChanges);\r\n\r\n for (const p of this.processors) {\r\n p.beforeUpdates(changes);\r\n }\r\n changes = this.getChangeMap(SyncService.getChangedNodes()); //need to update changes in case processors added any\r\n }\r\n\r\n const allChangesByDid = new Map(\r\n changes.all().map(change => {\r\n return [change._dynamicId, change];\r\n })\r\n );\r\n const allChangeCount = allChangesByDid.size;\r\n const processedChangeMap = this.runChanges(changes, (renderer, change) => {\r\n renderer.update3d(change);\r\n });\r\n const processedChanges = processedChangeMap.all();\r\n processedChanges.forEach(change => {\r\n allChangesByDid.delete(change._dynamicId);\r\n });\r\n this.deferredChanges = Array.from(allChangesByDid.values());\r\n\r\n const skipCount = allChangeCount - processedChanges.length;\r\n // For now we are processing all changes becuase it causes unpredictable behavior.\r\n // The heaviest part of the render cycle is the geometry features processing.\r\n // if (processedChanges.length > 0 || skipCount > 0) {\r\n // console.log('processed ' + processedChanges.length + ' changes, skipped ' + skipCount + ' changes');\r\n // }\r\n\r\n // Don't run the geometry and texture stack processing until all changes have been processed\r\n if (skipCount === 0) {\r\n this.features.update3dGeometry();\r\n this.textures.updateTextureStacks();\r\n }\r\n\r\n for (const p of this.processors) p.afterUpdates(processedChangeMap);\r\n for (const n of processedChanges) {\r\n n.commitChanges();\r\n }\r\n\r\n for (const p of this.processors) p.doAfterCleanup();\r\n\r\n // Output a console warning if there are a lot of changes not being processed, as this could indicate a problem with changes not being processed.\r\n const unchangedNodes = SyncService.getChangedNodes();\r\n\r\n if (unchangedNodes.length > 0) {\r\n unprocessedChangesCount++;\r\n if (unprocessedChangesCount > 1000) {\r\n unprocessedChangesCount = 0;\r\n console.warn('Scene stabilization is taking longer than expected.');\r\n for(const n of unchangedNodes) {\r\n if(n._rootScene._dynamicId !== this.viewer.sceneNode._dynamicId) {\r\n n.commitChanges();\r\n console.warn(`node ${n.name} (${n.id}) is from a different, non-active scene instance. Clearing the change.`)\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n /** runs a change map in order */\r\n protected runChanges(\r\n changes: ChangeMap,\r\n action: (renderer: Renderer, change: ITrackedClass) => void\r\n ) {\r\n const processedChanges = new ChangeMap();\r\n const renderStart = performance.now();\r\n\r\n for (const renderer of this.renderers) {\r\n const changeList = changes.get(renderer.modelToken);\r\n for (const change of changeList) {\r\n action(renderer, change);\r\n processedChanges.add(renderer.modelToken, change);\r\n }\r\n }\r\n\r\n const renderTime = performance.now() - renderStart;\r\n if (renderTime > 50) {\r\n console.warn(`Change cycle took ${renderTime}ms`);\r\n }\r\n\r\n return processedChanges;\r\n }\r\n protected getChangeMap(nodes: Array>, includeOrphaned = false) {\r\n const changes = new ChangeMap();\r\n for (const n of nodes) {\r\n if (\r\n (n._rootScene?._dynamicId === this.viewer.sceneNode._dynamicId && n._level > -1) ||\r\n includeOrphaned ||\r\n this.isGlobalMaterialOrTexture(n) //global/library material nodes held outside of graph\r\n ) {\r\n const renderer = this.renderers.find(r => r.handlesNode(n));\r\n if (!renderer) throw 'unknown node type';\r\n changes.add(renderer.modelToken, n);\r\n } else if (n._rootScene == null) {\r\n // Clear changes on nodes that are not in any scene tree\r\n n.commitChanges();\r\n }\r\n }\r\n\r\n changes.get(Token.SpaceNode).sortBy(m => m._level); //sort meshes by level first\r\n\r\n return changes;\r\n }\r\n disposeAll() {\r\n this.shadowManager.dispose();\r\n for (const processor of this.processors) processor.dispose();\r\n for (const renderer of this.renderers) renderer.dispose();\r\n }\r\n\r\n protected isGlobalMaterialOrTexture(n: Kb3dObject) {\r\n let mat = n;\r\n if (n instanceof TextureLayer) mat = n._parent as Kb3dObject;\r\n return mat instanceof MaterialNode && mat.location != eMaterialLocation.local;\r\n }\r\n}\r\n","import { ICameraInput } from '@babylonjs/core/Cameras/cameraInputsManager';\r\nimport { IWheelEvent } from '@babylonjs/core/Events/deviceInputEvents';\r\nimport { PointerEventTypes, PointerInfo } from '@babylonjs/core/Events/pointerEvents';\r\nimport { Epsilon, Scalar } from '@babylonjs/core/Maths';\r\nimport { EventState, Observer } from '@babylonjs/core/Misc/observable';\r\nimport { Nullable } from '@babylonjs/core/types';\r\nimport { KbArcRotateCamera } from './kb-arc-rotate-camera';\r\n\r\n/**\r\n * We are overriding the default mouse wheet input for the arc rotate camera here because\r\n * the default one does not act correctly when zoomed in close to the target: it\r\n * calculates the inertia based on the radius at the time the event happens. The result\r\n * is that zooming in is fast than zooming out. They had a misguided attempt at fixing\r\n * that here: https://github.com/BabylonJS/Babylon.js/issues/6215 but it is inadequate.\r\n * See comments below for where we changed the code\r\n */\r\nexport class KbArcRotateCameraMouseWheelInput implements ICameraInput {\r\n public camera: KbArcRotateCamera;\r\n public wheelPrecision = 3.0;\r\n public wheelDeltaPercentage = 0;\r\n\r\n private _wheel: Nullable<(p: PointerInfo, s: EventState) => void>;\r\n private _observer: Nullable>;\r\n\r\n private computeDeltaFromMouseWheelLegacyEvent(mouseWheelDelta: number, radius: number) {\r\n let delta = 0;\r\n const wheelDelta = mouseWheelDelta * 0.01 * this.wheelDeltaPercentage * radius;\r\n if (mouseWheelDelta > 0) {\r\n delta = wheelDelta / (1.0 + this.wheelDeltaPercentage);\r\n } else {\r\n delta = wheelDelta * (1.0 + this.wheelDeltaPercentage);\r\n }\r\n return delta;\r\n }\r\n\r\n public attachControl(noPreventDefault?: boolean): void {\r\n this._wheel = (p, s) => {\r\n if (p.type !== PointerEventTypes.POINTERWHEEL) {\r\n return;\r\n }\r\n const event = p.event as IWheelEvent;\r\n let delta = 0;\r\n\r\n const mouseWheelLegacyEvent = event as any;\r\n let wheelDelta = 0;\r\n\r\n if (mouseWheelLegacyEvent.wheelDelta) {\r\n wheelDelta = mouseWheelLegacyEvent.wheelDelta;\r\n } else {\r\n wheelDelta = -(event.deltaY || event.detail || 0) * 60;\r\n }\r\n\r\n if (this.wheelDeltaPercentage) {\r\n delta = this.computeDeltaFromMouseWheelLegacyEvent(wheelDelta, this.camera.radius);\r\n\r\n // If zooming in, estimate the target radius and use that to compute the delta for inertia\r\n // this will stop multiple scroll events zooming in from adding too much inertia\r\n //if (delta > 0) { //KB Note: their misguided attempt at fixing the problem only affected zoom in, when it should affect both, so we commented this line out\r\n let estimatedTargetRadius = this.camera.radius;\r\n let targetInertia = this.camera.inertialRadiusOffset + delta;\r\n for (let i = 0; i < 20 && Math.abs(targetInertia) > this.camera.speed * Epsilon; i++) {\r\n //KB Note: fixed the lower limit where it opts out to match the camera's own intertial calculation\r\n estimatedTargetRadius -= targetInertia;\r\n targetInertia *= this.camera.inertia;\r\n }\r\n estimatedTargetRadius = Scalar.Clamp(estimatedTargetRadius, 0, Number.MAX_VALUE);\r\n delta = this.computeDeltaFromMouseWheelLegacyEvent(wheelDelta, estimatedTargetRadius);\r\n //}\r\n } else {\r\n delta = wheelDelta / (this.wheelPrecision * 40);\r\n }\r\n\r\n if (delta) {\r\n this.camera.inertialRadiusOffset += delta;\r\n }\r\n\r\n if (event.preventDefault) {\r\n if (!noPreventDefault) {\r\n event.preventDefault();\r\n }\r\n }\r\n };\r\n\r\n this._observer = this.camera.getScene().onPointerObservable.add(this._wheel, PointerEventTypes.POINTERWHEEL);\r\n }\r\n\r\n public detachControl(): void {\r\n if (this._observer) {\r\n this.camera.getScene().onPointerObservable.remove(this._observer);\r\n this._observer = null;\r\n this._wheel = null;\r\n }\r\n }\r\n\r\n public getClassName(): string {\r\n return 'KbArcRotateCameraMouseWheelInput';\r\n }\r\n\r\n public getSimpleName(): string {\r\n return 'mousewheel';\r\n }\r\n}\r\n","import { ArcRotateCamera } from '@babylonjs/core/Cameras/arcRotateCamera';\r\nimport { Camera } from '@babylonjs/core/Cameras/camera';\r\nimport { Epsilon, Vector2, Vector3 } from '@babylonjs/core/Maths';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { KbArcRotateCameraMouseWheelInput } from './kb-arc-rotate-camera-wheel-input';\r\n\r\n/** Arc Rotate Camera (orbit camera) modified to offset the screen position on pan and keep the orbit target in the same position */\r\nexport class KbArcRotateCamera extends ArcRotateCamera {\r\n constructor(\r\n name: string,\r\n alpha: number,\r\n beta: number,\r\n radius: number,\r\n target: Vector3,\r\n private scene: Scene,\r\n setActiveOnSceneIfNoneActive?: boolean\r\n ) {\r\n super(name, alpha, beta, radius, target, scene, setActiveOnSceneIfNoneActive);\r\n\r\n //replace the wheel input with our own\r\n this.inputs.removeByType('ArcRotateCameraMouseWheelInput');\r\n this.inputs.add(new KbArcRotateCameraMouseWheelInput());\r\n this.updateClient();\r\n }\r\n inertialOffsetPanningX = 0;\r\n inertialOffsetPanningY = 0;\r\n inertialOrthoZoom = 0;\r\n clientInfo = {\r\n mouseX: 0,\r\n mouseY: 0,\r\n screenLeft: 0,\r\n screenTop: 0,\r\n screenWidth: 0,\r\n screenHeight: 0,\r\n };\r\n lockRotation = false;\r\n lockPanning = false;\r\n lockZoom = false;\r\n limitMinimumTargetDistance = false;\r\n\r\n private disposeListeners: (() => void) | undefined;\r\n private _orthoWidth = 0;\r\n get orthoWidth() {\r\n return this._orthoWidth;\r\n }\r\n set orthoWidth(val) {\r\n this._orthoWidth = this.clampOrthoWidth(val);\r\n this.updateOrthographicPosition();\r\n }\r\n\r\n orthoFit(sizeX: number, sizeY: number) {\r\n if (this.mode === Camera.ORTHOGRAPHIC_CAMERA) {\r\n const ratio = this.clientInfo.screenHeight / this.clientInfo.screenWidth;\r\n if (sizeX * ratio < sizeY) {\r\n this.orthoWidth = sizeY / ratio;\r\n } else {\r\n this.orthoWidth = sizeX;\r\n }\r\n }\r\n }\r\n\r\n update() {\r\n if (this.scene.activeCamera !== this) {\r\n return;\r\n }\r\n let startZoomLevel: number | undefined;\r\n if (!this.targetScreenOffset) {\r\n this.targetScreenOffset = new Vector2(0, 0);\r\n }\r\n\r\n if (this.lockRotation && (this.inertialAlphaOffset !== 0 || this.inertialBetaOffset !== 0)) {\r\n this.inertialPanningX = this.inertialAlphaOffset;\r\n this.inertialPanningY = -this.inertialBetaOffset;\r\n this.inertialAlphaOffset = this.inertialBetaOffset = 0;\r\n }\r\n\r\n if (this.inertialPanningX !== 0 || this.inertialPanningY !== 0) {\r\n if (!this.lockPanning) {\r\n this.inertialOffsetPanningX = this.inertialPanningX;\r\n this.inertialOffsetPanningY = this.inertialPanningY;\r\n }\r\n this.inertialPanningX = 0;\r\n this.inertialPanningY = 0;\r\n }\r\n\r\n if (this.lockZoom && this.inertialRadiusOffset !== 0) {\r\n this.inertialRadiusOffset = 0;\r\n }\r\n\r\n if (this.inertialRadiusOffset !== 0) {\r\n if (this.mode === Camera.ORTHOGRAPHIC_CAMERA) {\r\n this.inertialOrthoZoom = this.inertialRadiusOffset;\r\n this.inertialRadiusOffset = 0;\r\n } else {\r\n startZoomLevel = this.radius;\r\n //this.inertialRadiusOffset = 0; //override so we have full control over the zoom\r\n //if (this.inertialRadiusOffset < .001) this.inertialRadiusOffset = .001;\r\n }\r\n }\r\n\r\n if (this.inertialOrthoZoom !== 0) {\r\n startZoomLevel = this.orthoWidth;\r\n }\r\n\r\n super.update();\r\n\r\n if (this.inertialOffsetPanningX !== 0 || this.inertialOffsetPanningY !== 0) {\r\n let offset: Vector2;\r\n if (this.mode === Camera.ORTHOGRAPHIC_CAMERA) {\r\n offset = new Vector2(\r\n this.inertialOffsetPanningX * this.orthoWidth,\r\n this.inertialOffsetPanningY * this.orthoWidth\r\n );\r\n } else {\r\n offset = new Vector2(\r\n this.inertialOffsetPanningX * this.radius,\r\n this.inertialOffsetPanningY * this.radius\r\n );\r\n }\r\n this.targetScreenOffset.subtractInPlace(offset);\r\n\r\n this.inertialOffsetPanningX *= this.panningInertia;\r\n this.inertialOffsetPanningY *= this.panningInertia;\r\n }\r\n\r\n if (this.inertialOrthoZoom !== 0) {\r\n this.orthoWidth = this.orthoWidth - this.inertialOrthoZoom * this.orthoWidth;\r\n\r\n this.inertialOrthoZoom *= this.panningInertia;\r\n }\r\n\r\n if (Math.abs(this.inertialOffsetPanningX) < this.speed * Epsilon) {\r\n this.inertialOffsetPanningX = 0;\r\n }\r\n if (Math.abs(this.inertialOffsetPanningY) < this.speed * Epsilon) {\r\n this.inertialOffsetPanningY = 0;\r\n }\r\n if (Math.abs(this.inertialOrthoZoom) < this.speed * Epsilon) {\r\n this.inertialOrthoZoom = 0;\r\n }\r\n\r\n // When zooming, adjust the camera pan so you zoom into your mouse position, as long as pan is enabled\r\n if (!this.lockPanning && startZoomLevel !== undefined) {\r\n if (this.mode === Camera.ORTHOGRAPHIC_CAMERA) {\r\n const zoomChange = (startZoomLevel - this.orthoWidth) / 2;\r\n\r\n this.targetScreenOffset.x -= zoomChange * this.clientInfo.mouseX;\r\n this.targetScreenOffset.y += zoomChange * this.clientInfo.mouseY;\r\n } else {\r\n const gamma = Math.tan(this.fov / 2);\r\n const zoomChange = startZoomLevel - this.radius;\r\n\r\n const shiftX = zoomChange * gamma * this.clientInfo.mouseX;\r\n const shiftY = zoomChange * gamma * this.clientInfo.mouseY;\r\n this.targetScreenOffset.x -= shiftX;\r\n this.targetScreenOffset.y += shiftY;\r\n }\r\n }\r\n }\r\n\r\n updateClient() {\r\n const canvas = this.getEngine().getRenderingCanvas();\r\n if (!canvas) {\r\n return;\r\n }\r\n\r\n const clientRect = canvas.getBoundingClientRect();\r\n this.clientInfo.screenWidth = clientRect.width;\r\n this.clientInfo.screenHeight = clientRect.height;\r\n this.clientInfo.screenLeft = clientRect.left;\r\n this.clientInfo.screenTop = clientRect.top;\r\n\r\n if (this.mode === Camera.ORTHOGRAPHIC_CAMERA) {\r\n this.updateOrthographicPosition();\r\n\r\n this.panningSensibility = clientRect.width;\r\n } else {\r\n const focalLength = clientRect.height / (2 * Math.tan(this.fov / 2));\r\n this.panningSensibility = focalLength;\r\n }\r\n\r\n return canvas;\r\n }\r\n\r\n attachControl(noPreventDefault = false) {\r\n super.attachControl(noPreventDefault, true);\r\n if (this.disposeListeners) {\r\n this.disposeListeners();\r\n }\r\n\r\n const updateMousePosition = (event: MouseEvent) => {\r\n this.clientInfo.mouseX =\r\n ((event.clientX - this.clientInfo.screenLeft) * 2) / this.clientInfo.screenWidth - 1;\r\n this.clientInfo.mouseY =\r\n ((event.clientY - this.clientInfo.screenTop) * 2) / this.clientInfo.screenHeight - 1;\r\n };\r\n const canvas = this.updateClient();\r\n const updateClientWrap = () => this.updateClient();\r\n\r\n if (canvas) {\r\n window.addEventListener('resize', updateClientWrap);\r\n canvas.addEventListener('mousemove', updateMousePosition);\r\n\r\n this.disposeListeners = () => {\r\n window.removeEventListener('resize', updateClientWrap);\r\n canvas.removeEventListener('mousemove', updateMousePosition);\r\n this.disposeListeners = undefined;\r\n };\r\n }\r\n }\r\n\r\n _checkLimits() {\r\n super._checkLimits();\r\n if (this.mode === Camera.ORTHOGRAPHIC_CAMERA) {\r\n this.orthoWidth = this.clampOrthoWidth(this.orthoWidth);\r\n }\r\n }\r\n\r\n private clampOrthoWidth(val: number) {\r\n if (this.mode === Camera.ORTHOGRAPHIC_CAMERA) {\r\n if (this.lowerRadiusLimit != null && val < this.lowerRadiusLimit) {\r\n val = this.lowerRadiusLimit;\r\n }\r\n if (this.upperRadiusLimit != null && val > this.upperRadiusLimit) {\r\n val = this.upperRadiusLimit;\r\n }\r\n }\r\n\r\n return val;\r\n }\r\n\r\n dispose() {\r\n super.dispose();\r\n if (this.disposeListeners) {\r\n this.disposeListeners();\r\n }\r\n }\r\n\r\n private updateOrthographicPosition() {\r\n if (this.clientInfo) {\r\n const ratio = this.clientInfo.screenHeight / this.clientInfo.screenWidth;\r\n this.orthoRight = this.orthoWidth / 2;\r\n this.orthoLeft = -this.orthoWidth / 2;\r\n this.orthoTop = (this.orthoWidth * ratio) / 2;\r\n this.orthoBottom = (-this.orthoWidth * ratio) / 2;\r\n }\r\n }\r\n}\r\n","import { Animatable } from '@babylonjs/core/Animations/animatable';\r\nimport { Animation } from '@babylonjs/core/Animations/animation';\r\nimport { CircleEase, SineEase } from '@babylonjs/core/Animations/easing';\r\nimport { ArcRotateCamera } from '@babylonjs/core/Cameras/arcRotateCamera';\r\nimport { Camera } from '@babylonjs/core/Cameras/camera';\r\nimport { FreeCamera } from '@babylonjs/core/Cameras/freeCamera';\r\nimport { UniversalCamera } from '@babylonjs/core/Cameras/universalCamera';\r\nimport { BoundingSphere } from '@babylonjs/core/Culling/boundingSphere';\r\nimport { Plane, Vector2, Vector3 } from '@babylonjs/core/Maths';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { PostProcessRenderPipeline } from '@babylonjs/core/PostProcesses/RenderPipeline/postProcessRenderPipeline';\r\nimport { WebXRCamera } from '@babylonjs/core/XR/webXRCamera';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { Nullable } from '@babylonjs/core/types';\r\nimport { AnnotationNode, DimensionNode } from '@models/classes/annotations';\r\nimport { defaults } from '@models/classes/default-values';\r\nimport { eViewpointProjection, eViewpointTargetMode, eViewpointType } from '@models/classes/enums';\r\nimport { MeshNode, SpaceNode } from '@models/classes/meshes';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { ViewpointNode } from '@models/classes/viewpoints/viewpoint-node';\r\nimport { getCameraPositionWithOffset, toRadians } from '@view/helpers/math-helper';\r\nimport { AnnotationInfo, SKYBOX_SIZE } from '@view/render';\r\nimport { Observable, Subject } from 'rxjs';\r\nimport { finalize, share, startWith, takeUntil } from 'rxjs/operators';\r\nimport { getScreenBoundingBox, nodeToScreen } from '../helpers/scene-helper';\r\nimport { KbViewer } from '../kb-viewer';\r\nimport { KbArcRotateCamera } from './cameras/kb-arc-rotate-camera';\r\n\r\nconst ANIMATION_FPS = 30;\r\n\r\nenum eCameraType {\r\n Universal,\r\n ArcRotate,\r\n}\r\n\r\nexport interface ICameraAnimationOptions {\r\n duration: number;\r\n frame?: boolean;\r\n frameOptions?: Partial;\r\n frameTarget?: SpaceNode;\r\n}\r\n\r\nexport interface IFramingOptions {\r\n rotate: boolean;\r\n pan: boolean;\r\n zoom: boolean;\r\n padding: number;\r\n}\r\n\r\nexport enum eCameraOrientation {\r\n perspective = 'perspective',\r\n front = 'front',\r\n left = 'left',\r\n back = 'back',\r\n right = 'right',\r\n top = 'top',\r\n bottom = 'bottom',\r\n}\r\n\r\ntype ScalingHandler = (distance: number, type: number) => void;\r\n\r\nexport class CameraTools {\r\n get cameraMoving() {\r\n const camera = this.scene.activeCamera;\r\n if (this._activeAnimation) {\r\n return true;\r\n }\r\n if (camera instanceof KbArcRotateCamera) {\r\n return (\r\n Math.abs(camera.inertialRadiusOffset) > 0 ||\r\n Math.abs(camera.inertialAlphaOffset) > 0 ||\r\n Math.abs(camera.inertialBetaOffset) > 0 ||\r\n Math.abs(camera.inertialOffsetPanningX) > 0 ||\r\n Math.abs(camera.inertialOffsetPanningY) > 0 ||\r\n Math.abs(camera.inertialOrthoZoom) > 0\r\n );\r\n }\r\n if (camera instanceof UniversalCamera) {\r\n return (\r\n Math.abs(camera.cameraDirection.x) > 0 ||\r\n Math.abs(camera.cameraDirection.y) > 0 ||\r\n Math.abs(camera.cameraDirection.z) > 0 ||\r\n Math.abs(camera.cameraRotation.x) > 0 ||\r\n Math.abs(camera.cameraRotation.y) > 0\r\n );\r\n }\r\n return false;\r\n }\r\n\r\n public activeCamera: Partial;\r\n\r\n private _cameraSettingsChange?: Subject>;\r\n private _beforeRender: () => void;\r\n private _storedViewpoint?: ViewpointNode;\r\n private _activeAnimation?: Animatable;\r\n private disposePreviousCamera: Subject | undefined;\r\n private _lastOrientation = eCameraOrientation.perspective;\r\n private _lastPerspectivePosition: KbVector;\r\n private minVector = new Vector3(0, 0, 0);\r\n private maxVector = new Vector3(0, 0, 0);\r\n private activePipeline: PostProcessRenderPipeline | undefined;\r\n\r\n private objectsToScaleWithDistance = new Map();\r\n\r\n constructor(private scene: Scene, private canvas: HTMLElement, private kbViewer: KbViewer) {\r\n this._beforeRender = () => {\r\n if (this._cameraSettingsChange && scene.activeCamera && this.cameraMoving) {\r\n const camera = scene.activeCamera as KbArcRotateCamera | UniversalCamera;\r\n const changeList: Partial = {};\r\n copyCameraStateIntoViewpoint(changeList, camera);\r\n\r\n this._cameraSettingsChange.next(changeList);\r\n }\r\n };\r\n scene.registerBeforeRender(this._beforeRender);\r\n\r\n kbViewer.viewChanged.subscribe(() => {\r\n const camera = scene.activeCamera;\r\n if (camera) {\r\n if (camera.mode === Camera.ORTHOGRAPHIC_CAMERA) {\r\n const dist = ((camera.orthoRight || 0) - (camera.orthoLeft || 0)) * 0.6;\r\n this.objectsToScaleWithDistance.forEach((param, item) => {\r\n if (item.isDisposed()) {\r\n this.objectsToScaleWithDistance.delete(item);\r\n return;\r\n }\r\n if (typeof param === 'function') {\r\n param(dist, camera.mode);\r\n } else {\r\n const scaledSize = param * dist;\r\n item.scaling.set(scaledSize, scaledSize, scaledSize);\r\n }\r\n });\r\n } else {\r\n let cameraPosition: Vector3;\r\n if (camera instanceof WebXRCamera && camera.globalPosition) {\r\n cameraPosition = camera.globalPosition;\r\n } else if (camera instanceof ArcRotateCamera) {\r\n cameraPosition = getCameraPositionWithOffset(camera);\r\n } else {\r\n cameraPosition = camera.globalPosition;\r\n }\r\n this.objectsToScaleWithDistance.forEach((param, item) => {\r\n if (item.isDisposed()) {\r\n this.objectsToScaleWithDistance.delete(item);\r\n return;\r\n }\r\n const dist = item.absolutePosition.subtract(cameraPosition).length();\r\n if (typeof param === 'function') {\r\n param(dist, camera.mode);\r\n } else {\r\n const scaledSize = dist * param;\r\n item.scaling.set(scaledSize, scaledSize, scaledSize);\r\n }\r\n });\r\n }\r\n }\r\n });\r\n\r\n this.activeCamera = {\r\n get position() {\r\n return scene.activeCamera ? KbVector.FromVec3(scene.activeCamera.position) : KbVector.Zero();\r\n },\r\n set position(vector: KbVector) {\r\n if (scene.activeCamera) {\r\n if (\r\n scene.activeCamera.mode === Camera.ORTHOGRAPHIC_CAMERA &&\r\n scene.activeCamera instanceof ArcRotateCamera\r\n ) {\r\n const target = scene.activeCamera.target;\r\n scene.activeCamera.position = vector\r\n .toVec3()\r\n .subtract(target)\r\n .normalize()\r\n .scale(5000)\r\n .add(target);\r\n } else {\r\n scene.activeCamera.position = vector.toVec3();\r\n }\r\n }\r\n },\r\n get targetPoint() {\r\n return scene.activeCamera instanceof ArcRotateCamera\r\n ? KbVector.FromVec3(scene.activeCamera.target)\r\n : KbVector.Zero();\r\n },\r\n set targetPoint(vector: KbVector) {\r\n scene.activeCamera instanceof ArcRotateCamera && (scene.activeCamera.target = vector.toVec3());\r\n },\r\n get rotation() {\r\n return scene.activeCamera instanceof UniversalCamera\r\n ? KbVector.FromVec3(scene.activeCamera.rotation)\r\n : KbVector.Zero();\r\n },\r\n set rotation(vector: KbVector) {\r\n scene.activeCamera instanceof UniversalCamera && (scene.activeCamera.rotation = vector.toVec3());\r\n },\r\n get offsetX() {\r\n return scene.activeCamera instanceof ArcRotateCamera ? scene.activeCamera.targetScreenOffset!.x : 0;\r\n },\r\n set offsetX(offset: number) {\r\n scene.activeCamera instanceof ArcRotateCamera && (scene.activeCamera.targetScreenOffset.x = offset);\r\n },\r\n get offsetY() {\r\n return scene.activeCamera instanceof ArcRotateCamera ? scene.activeCamera.targetScreenOffset!.y : 0;\r\n },\r\n set offsetY(offset: number) {\r\n scene.activeCamera instanceof ArcRotateCamera && (scene.activeCamera.targetScreenOffset.y = offset);\r\n },\r\n get projection() {\r\n if (!scene.activeCamera) {\r\n return undefined;\r\n }\r\n return scene.activeCamera.mode === Camera.PERSPECTIVE_CAMERA\r\n ? eViewpointProjection.perspective\r\n : eViewpointProjection.orthographic;\r\n },\r\n set projection(projection: eViewpointProjection | undefined) {\r\n scene.activeCamera &&\r\n (scene.activeCamera.mode =\r\n projection === eViewpointProjection.perspective\r\n ? Camera.PERSPECTIVE_CAMERA\r\n : Camera.ORTHOGRAPHIC_CAMERA);\r\n },\r\n get fov() {\r\n if (!scene.activeCamera) {\r\n return 50;\r\n }\r\n return scene.activeCamera.mode === Camera.ORTHOGRAPHIC_CAMERA\r\n ? scene.activeCamera.orthoRight! - scene.activeCamera.orthoLeft!\r\n : (scene.activeCamera.fov * 180) / Math.PI;\r\n },\r\n set fov(fov: number) {\r\n if (!scene.activeCamera) {\r\n return;\r\n }\r\n if (\r\n scene.activeCamera.mode === Camera.ORTHOGRAPHIC_CAMERA &&\r\n scene.activeCamera instanceof KbArcRotateCamera\r\n ) {\r\n scene.activeCamera.orthoWidth = fov;\r\n } else {\r\n scene.activeCamera.fov = (fov * Math.PI) / 180;\r\n }\r\n },\r\n get allowOrbit() {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n return !scene.activeCamera.lockRotation;\r\n } else if (scene.activeCamera instanceof UniversalCamera) {\r\n return !!scene.activeCamera.inputs.attached.mouse;\r\n }\r\n return false;\r\n },\r\n set allowOrbit(allow) {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n scene.activeCamera.lockRotation = !allow;\r\n } else if (scene.activeCamera instanceof UniversalCamera) {\r\n scene.activeCamera.inputs.clear();\r\n enableFreeLook(scene.activeCamera);\r\n }\r\n },\r\n get allowPan() {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n return !scene.activeCamera.lockPanning;\r\n } else if (scene.activeCamera instanceof UniversalCamera) {\r\n return !!scene.activeCamera.inputs.attached.keyboard;\r\n }\r\n return false;\r\n },\r\n set allowPan(allow) {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n scene.activeCamera.lockPanning = !allow;\r\n } else if (scene.activeCamera instanceof UniversalCamera) {\r\n scene.activeCamera.inputs.clear();\r\n enableFreeMove(scene.activeCamera);\r\n }\r\n },\r\n get allowZoom() {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n return !scene.activeCamera.lockZoom;\r\n }\r\n return false;\r\n },\r\n set allowZoom(allow) {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n scene.activeCamera.lockZoom = !allow;\r\n }\r\n },\r\n get limitMinimumTargetDistance() {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n return scene.activeCamera.limitMinimumTargetDistance;\r\n }\r\n return false;\r\n },\r\n set limitMinimumTargetDistance(allow) {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n scene.activeCamera.limitMinimumTargetDistance = !allow;\r\n }\r\n },\r\n get minTargetDistance() {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n return scene.activeCamera.lowerRadiusLimit || defaults.ViewpointNode.minTargetDistance;\r\n }\r\n return defaults.ViewpointNode.minTargetDistance;\r\n },\r\n set minTargetDistance(value) {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n scene.activeCamera.lowerRadiusLimit = value;\r\n }\r\n },\r\n get maxTargetDistance() {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n return scene.activeCamera.upperRadiusLimit || defaults.ViewpointNode.maxTargetDistance;\r\n }\r\n return defaults.ViewpointNode.maxTargetDistance;\r\n },\r\n set maxTargetDistance(value) {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n scene.activeCamera.upperRadiusLimit = value;\r\n }\r\n },\r\n\r\n get orbitAxis() {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n return KbVector.FromVec3(scene.activeCamera.upVector);\r\n }\r\n return KbVector.From(defaults.ViewpointNode.orbitAxis);\r\n },\r\n set orbitAxis(value) {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n scene.activeCamera.upVector = value.toVec3();\r\n }\r\n },\r\n\r\n get limitMinLatitude() {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n return ((scene.activeCamera.lowerBetaLimit || 0) * 180) / Math.PI;\r\n }\r\n return defaults.ViewpointNode.limitMinLatitude;\r\n },\r\n set limitMinLatitude(value) {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n scene.activeCamera.lowerBetaLimit = (value * Math.PI) / 180;\r\n }\r\n },\r\n get limitMaxLatitude() {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n return (\r\n ((typeof scene.activeCamera.upperBetaLimit === 'number'\r\n ? scene.activeCamera.upperBetaLimit\r\n : Math.PI) *\r\n 180) /\r\n Math.PI\r\n );\r\n }\r\n return defaults.ViewpointNode.limitMaxLatitude;\r\n },\r\n set limitMaxLatitude(value) {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n scene.activeCamera.upperBetaLimit = (value * Math.PI) / 180;\r\n }\r\n },\r\n get limitMinLongitude() {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n return scene.activeCamera.lowerAlphaLimit\r\n ? (scene.activeCamera.lowerAlphaLimit * 180) / Math.PI\r\n : defaults.ViewpointNode.limitMinLongitude;\r\n }\r\n return defaults.ViewpointNode.limitMinLongitude;\r\n },\r\n set limitMinLongitude(value) {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n scene.activeCamera.lowerAlphaLimit = (value * Math.PI) / 180;\r\n }\r\n },\r\n get limitMaxLongitude() {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n return scene.activeCamera.upperAlphaLimit\r\n ? (scene.activeCamera.upperAlphaLimit * 180) / Math.PI\r\n : defaults.ViewpointNode.limitMinLongitude;\r\n }\r\n return defaults.ViewpointNode.limitMinLongitude;\r\n },\r\n set limitMaxLongitude(value) {\r\n if (scene.activeCamera instanceof KbArcRotateCamera) {\r\n scene.activeCamera.upperAlphaLimit = (value * Math.PI) / 180;\r\n }\r\n },\r\n };\r\n }\r\n\r\n dispose() {\r\n this.scene.unregisterBeforeRender(this._beforeRender);\r\n }\r\n\r\n setPipeline(pipeline: PostProcessRenderPipeline) {\r\n this.activePipeline = pipeline;\r\n if (this.scene.activeCamera) {\r\n this.applyScenePostProcesses(this.scene.activeCamera);\r\n }\r\n }\r\n\r\n /** Frames the current active camera to fit the provided mesh node\r\n * @param node The node to fit on the screen\r\n * @param padding If provided, pads the node by a percentage of the node's size.\r\n */\r\n frameCamera(node: SpaceNode, options?: Partial, camera?: Camera) {\r\n const _options: IFramingOptions = {\r\n rotate: false,\r\n pan: true,\r\n padding: 1,\r\n zoom: true,\r\n ...(options || {}),\r\n };\r\n const activeCamera = camera || this.scene.activeCamera;\r\n if (!activeCamera) {\r\n throw Error('No active camera');\r\n }\r\n\r\n const bounding = this.kbViewer.bbCache.getWorld(node);\r\n\r\n if (!bounding || bounding.boundingInfo.diagonalLength === 0) {\r\n return; // no op if the bounding box doesn't exist\r\n } else {\r\n const childrenBounding = this.kbViewer.sceneNode.getChildren();\r\n const maxVector = bounding.boundingInfo.boundingBox.maximum;\r\n const minVector = bounding.boundingInfo.boundingBox.minimum;\r\n childrenBounding.forEach(element => {\r\n if (\r\n (element instanceof AnnotationNode && element.enabled) ||\r\n (element instanceof DimensionNode && element.enabled)\r\n ) {\r\n let nodeInfo: AnnotationInfo | undefined;\r\n if (element instanceof AnnotationNode) {\r\n nodeInfo = this.kbViewer.rendererManager.annotation.annotationStore.get(element._dynamicId);\r\n }\r\n if (element instanceof DimensionNode) {\r\n nodeInfo = this.kbViewer.rendererManager.dimension.dimensionStore.get(element._dynamicId);\r\n }\r\n const minBoundingBox = nodeInfo?.meshInfo?.textContainer.getHierarchyBoundingVectors().min;\r\n const maxBoundingBox = nodeInfo?.meshInfo?.textContainer.getHierarchyBoundingVectors().max;\r\n if (minBoundingBox && minBoundingBox.x < minVector.x) {\r\n minVector.x = minBoundingBox.x;\r\n }\r\n if (minBoundingBox && minBoundingBox.y < minVector.y) {\r\n minVector.y = minBoundingBox.y;\r\n }\r\n if (minBoundingBox && minBoundingBox.z < minVector.z) {\r\n minVector.z = minBoundingBox.z;\r\n }\r\n if (maxBoundingBox && maxBoundingBox.x > maxVector.x) {\r\n maxVector.x = maxBoundingBox.x;\r\n }\r\n if (maxBoundingBox && maxBoundingBox.y > maxVector.y) {\r\n maxVector.y = maxBoundingBox.y;\r\n }\r\n if (maxBoundingBox && maxBoundingBox.z > maxVector.z) {\r\n maxVector.z = maxBoundingBox.z;\r\n }\r\n }\r\n });\r\n this.minVector.set(minVector.x, minVector.y, minVector.z);\r\n\r\n this.maxVector.set(maxVector.x, maxVector.y, maxVector.z);\r\n\r\n bounding.boundingInfo.boundingBox.reConstruct(\r\n this.minVector,\r\n this.maxVector,\r\n bounding.boundingInfo.boundingBox.getWorldMatrix()\r\n );\r\n const bbCenter = bounding.center.toVec3();\r\n\r\n const bbCorners = bounding.boundingInfo.boundingBox.vectorsWorld;\r\n\r\n if (activeCamera instanceof UniversalCamera) {\r\n activeCamera.setTarget(bbCenter);\r\n } else if (activeCamera instanceof KbArcRotateCamera) {\r\n if (_options.rotate) {\r\n activeCamera.setTarget(bbCenter);\r\n if (_options.pan) {\r\n activeCamera.targetScreenOffset.set(0, 0);\r\n }\r\n } else if (_options.pan) {\r\n const cameraRay = activeCamera.getForwardRay();\r\n const cameraProjectionTargetPlane = Plane.FromPositionAndNormal(bbCenter, cameraRay.direction);\r\n const cameraDistance = cameraRay.intersectsPlane(cameraProjectionTargetPlane);\r\n const cameraProjectionTargetPoint = cameraRay.origin.add(\r\n cameraRay.direction.scale(cameraDistance!)\r\n );\r\n const cameraPositionWorldOffset = cameraProjectionTargetPoint.subtract(bbCenter);\r\n\r\n const planeProjectionX = Vector3.Cross(cameraRay.direction, Vector3.Up()).normalize();\r\n const planeProjectionY = Vector3.Cross(cameraRay.direction, planeProjectionX).normalize();\r\n\r\n const bbCenterOffset = new Vector2(\r\n Vector3.Dot(cameraPositionWorldOffset, planeProjectionX),\r\n -Vector3.Dot(cameraPositionWorldOffset, planeProjectionY)\r\n );\r\n activeCamera.targetScreenOffset.copyFrom(bbCenterOffset);\r\n }\r\n\r\n if (_options.zoom) {\r\n const clientRect = this.canvas.getBoundingClientRect();\r\n const ratio = clientRect.height / clientRect.width;\r\n const cameraRay = activeCamera.getForwardRay();\r\n const xAxis = Vector3.Cross(cameraRay.direction, Vector3.Up()).normalize();\r\n const yAxis = Vector3.Cross(cameraRay.direction, xAxis).normalize();\r\n const cameraTruePosition = cameraRay.origin\r\n .add(xAxis.scale(-activeCamera.targetScreenOffset.x))\r\n .add(yAxis.scale(activeCamera.targetScreenOffset.y));\r\n const cameraTrueTarget = activeCamera.target\r\n .add(xAxis.scale(-activeCamera.targetScreenOffset.x))\r\n .add(yAxis.scale(activeCamera.targetScreenOffset.y));\r\n const xPlane = Plane.FromPositionAndNormal(cameraTruePosition, yAxis);\r\n const yPlane = Plane.FromPositionAndNormal(cameraTruePosition, xAxis);\r\n const padding = 1 + _options.padding / 100;\r\n\r\n if (activeCamera.mode === Camera.ORTHOGRAPHIC_CAMERA) {\r\n let maxX = 0,\r\n maxY = 0;\r\n bbCorners.forEach(bbCorner => {\r\n const xO = xPlane.signedDistanceTo(bbCorner) * padding;\r\n maxX = Math.max(maxX, Math.abs(xO));\r\n\r\n const yO = yPlane.signedDistanceTo(bbCorner) * padding;\r\n maxY = Math.max(maxY, Math.abs(yO));\r\n });\r\n\r\n // The calculated distance is the maximum distance from the centerline to the bounding box corner. So double it to get the total width/height\r\n activeCamera.orthoFit(maxY * 2, maxX * 2);\r\n } else {\r\n let desiredDistance = 0;\r\n const fovH = 2 * Math.atan(Math.tan(activeCamera.fov / 2) / ratio);\r\n const thV = Math.tan(activeCamera.fov / 2);\r\n const thH = Math.tan(fovH / 2);\r\n bbCorners.forEach(bbCorner => {\r\n const xO = xPlane.signedDistanceTo(bbCorner) * padding;\r\n const xP = bbCorner.subtract(cameraTrueTarget).subtract(yAxis.scale(xO));\r\n const xA = Math.abs(xO) / thV;\r\n const xR = Vector3.Dot(xP, cameraRay.direction.scale(-1)) + xA;\r\n desiredDistance = Math.max(desiredDistance, xR);\r\n\r\n const yO = yPlane.signedDistanceTo(bbCorner) * padding;\r\n const yP = bbCorner.subtract(cameraTrueTarget).subtract(xAxis.scale(yO));\r\n const yA = Math.abs(yO) / thH;\r\n const yR = Vector3.Dot(yP, cameraRay.direction.scale(-1)) + yA;\r\n desiredDistance = Math.max(desiredDistance, yR);\r\n });\r\n\r\n activeCamera.radius = Math.max(desiredDistance, 0.001);\r\n }\r\n }\r\n }\r\n this.kbViewer.fireViewChangedOnNextRender(); //render loop won't pick up that the view changed, because no kb3d nodes changes, so we must update it explicitly\r\n }\r\n }\r\n\r\n getScreenPositionOfNode(node: MeshNode) {\r\n if (!this.scene.activeCamera) {\r\n throw Error('No active camera');\r\n }\r\n const bNode = this.kbViewer.getRendererNode(node);\r\n\r\n if (bNode) {\r\n return nodeToScreen(bNode, this.scene.activeCamera);\r\n }\r\n }\r\n\r\n getScreenBoundingBoxOfNode(node: MeshNode) {\r\n if (!this.scene.activeCamera) {\r\n throw Error('No active camera');\r\n }\r\n const bNode = this.kbViewer.getRendererNode(node);\r\n if (bNode) {\r\n return getScreenBoundingBox(bNode, this.scene.activeCamera, true);\r\n }\r\n }\r\n\r\n moveCameraToViewpoint(viewpoint: ViewpointNode) {\r\n const newCamera = this.createCameraFromViewpoint(viewpoint)!;\r\n this._activateCamera(newCamera);\r\n }\r\n\r\n animateToViewpoint(viewpoint: ViewpointNode, options?: Partial) {\r\n if (!this.scene.activeCamera) {\r\n throw Error('No active camera');\r\n }\r\n let activeCamera = this.scene.activeCamera;\r\n const _options: ICameraAnimationOptions = {\r\n duration: 1000,\r\n ...(options || {}),\r\n };\r\n const endFrame = 30 * (_options.duration / 1000);\r\n const animations: Animation[] = [];\r\n const ease = new SineEase();\r\n ease.setEasingMode(SineEase.EASINGMODE_EASEINOUT);\r\n const circEaseOut = new CircleEase();\r\n circEaseOut.setEasingMode(CircleEase.EASINGMODE_EASEOUT);\r\n\r\n // Create the new viewpoint converted to an arc rotate camera so we can tween arc rotate properties\r\n // The arc rotate cameras are easier to animate between nicely.\r\n const newCameraTarget = this.createCameraFromViewpoint(viewpoint, eCameraType.ArcRotate)!;\r\n const newCamera = this.createCameraFromViewpoint(viewpoint)!;\r\n\r\n if (_options.frame && newCamera instanceof ArcRotateCamera) {\r\n const target = _options.frameTarget || viewpoint._parentScene;\r\n if (!target) {\r\n throw Error('Cannot frame camera, no target node to frame');\r\n } else {\r\n this.frameCamera(target, _options.frameOptions || {}, newCameraTarget);\r\n this.frameCamera(target, _options.frameOptions || {}, newCamera);\r\n }\r\n }\r\n if (activeCamera instanceof ArcRotateCamera) {\r\n if (Math.abs(activeCamera.alpha) > Math.PI * 2) {\r\n activeCamera.alpha = activeCamera.alpha % (Math.PI * 2);\r\n }\r\n }\r\n if (activeCamera instanceof UniversalCamera && viewpoint.type != eViewpointType.firstPerson) {\r\n const orbitRotation = activeCamera.rotationQuaternion\r\n ? activeCamera.rotationQuaternion.toEulerAngles()\r\n : activeCamera.rotation;\r\n const ray = activeCamera.getForwardRay(newCameraTarget.radius);\r\n // If the current camera is a universal camera, convert it to an arc rotate camera as well/\r\n // The arc rotate target position is placed from the camera at the same distance as the destination viewpoint's target\r\n activeCamera = new KbArcRotateCamera(\r\n 'temporaryCamera',\r\n -(orbitRotation.y + Math.PI / 2),\r\n Math.PI / 2 - orbitRotation.x,\r\n ray.length,\r\n ray.direction.scale(ray.length).add(ray.origin),\r\n this.scene\r\n );\r\n activeCamera.fov = this.scene.activeCamera.fov;\r\n this._activateCamera(activeCamera);\r\n }\r\n\r\n if (!newCameraTarget) {\r\n throw Error('Could not create new camera');\r\n }\r\n\r\n if (activeCamera instanceof UniversalCamera && viewpoint.type === eViewpointType.firstPerson) {\r\n const targetRotation = new Vector3(0, 0, 0);\r\n const targetPosition = new Vector3(viewpoint.position.x, viewpoint.position.y, viewpoint.position.z);\r\n targetRotation.y = (viewpoint.rotation.y * Math.PI) / 180;\r\n targetRotation.x = (viewpoint.rotation.x * Math.PI) / 180;\r\n targetRotation.z = (viewpoint.rotation.z * Math.PI) / 180;\r\n\r\n let activeAlpha = activeCamera.rotation.y;\r\n let targetAlpha = (viewpoint.rotation.y * Math.PI) / 180;\r\n let arc = 0;\r\n\r\n //Normalizing values to be between -PI and PI\r\n activeAlpha = normalizeRotation(activeAlpha);\r\n targetAlpha = normalizeRotation(targetAlpha);\r\n\r\n //calculating shortest arc and seeing if it crosses zero\r\n //if it does, change the target vector to account for expected direction of animation\r\n if (activeAlpha > targetAlpha) {\r\n arc = activeAlpha - targetAlpha;\r\n } else {\r\n arc = targetAlpha - activeAlpha;\r\n }\r\n if (arc > Math.PI) {\r\n if (activeAlpha < targetAlpha) {\r\n targetRotation.y = targetRotation.y - 2 * Math.PI;\r\n } else {\r\n targetRotation.y = targetRotation.y + 2 * Math.PI;\r\n }\r\n }\r\n\r\n const translateRotation = new Animation(\r\n 'translateCameraRotation',\r\n 'rotation',\r\n ANIMATION_FPS,\r\n Animation.ANIMATIONTYPE_VECTOR3,\r\n Animation.ANIMATIONLOOPMODE_CONSTANT\r\n );\r\n translateRotation.setKeys([\r\n { frame: 0, value: activeCamera.rotation },\r\n { frame: endFrame, value: targetRotation },\r\n ]);\r\n\r\n const translatePosition = new Animation(\r\n 'translateCameraPosition',\r\n 'position',\r\n ANIMATION_FPS,\r\n Animation.ANIMATIONTYPE_VECTOR3,\r\n Animation.ANIMATIONLOOPMODE_CONSTANT\r\n );\r\n translatePosition.setKeys([\r\n { frame: 0, value: activeCamera.position },\r\n { frame: endFrame, value: targetPosition },\r\n ]);\r\n const translateFov = new Animation(\r\n 'translateCameraFOV',\r\n 'fov',\r\n ANIMATION_FPS,\r\n Animation.ANIMATIONTYPE_FLOAT,\r\n Animation.ANIMATIONLOOPMODE_CONSTANT\r\n );\r\n translateFov.setKeys([\r\n { frame: 0, value: activeCamera.fov },\r\n { frame: endFrame, value: (viewpoint.fov / 180) * Math.PI },\r\n ]);\r\n translateRotation.setEasingFunction(ease);\r\n translatePosition.setEasingFunction(ease);\r\n translateFov.setEasingFunction(ease);\r\n\r\n animations.push(translateRotation);\r\n animations.push(translatePosition);\r\n animations.push(translateFov);\r\n } else if (newCameraTarget instanceof ArcRotateCamera && activeCamera instanceof ArcRotateCamera) {\r\n // When rotating between two arc rotate cameras, rotate around the target while animating the target\r\n\r\n if (!activeCamera.target.equals(newCameraTarget.target)) {\r\n const translateTarget = new Animation(\r\n 'translateCameraTarget',\r\n 'target',\r\n ANIMATION_FPS,\r\n Animation.ANIMATIONTYPE_VECTOR3,\r\n Animation.ANIMATIONLOOPMODE_CONSTANT\r\n );\r\n translateTarget.setKeys([\r\n { frame: 0, value: activeCamera.target },\r\n { frame: endFrame, value: newCameraTarget.target },\r\n ]);\r\n animations.push(translateTarget);\r\n }\r\n\r\n const translateA = new Animation(\r\n 'translateCameraA',\r\n 'alpha',\r\n ANIMATION_FPS,\r\n Animation.ANIMATIONTYPE_FLOAT,\r\n Animation.ANIMATIONLOOPMODE_CONSTANT\r\n );\r\n\r\n let crossesZero = false;\r\n let arc = 0;\r\n let shortestArc = 0;\r\n\r\n //Normalizing values to be between -PI and PI\r\n activeCamera.alpha = normalizeRotation(activeCamera.alpha);\r\n newCameraTarget.alpha = normalizeRotation(newCameraTarget.alpha);\r\n\r\n //calculating shortest arc between two cameras\r\n if (activeCamera.alpha > newCameraTarget.alpha) {\r\n arc = activeCamera.alpha - newCameraTarget.alpha;\r\n } else {\r\n arc = newCameraTarget.alpha - activeCamera.alpha;\r\n }\r\n if (arc > Math.PI) {\r\n shortestArc = 2 * Math.PI - arc;\r\n crossesZero = true;\r\n }\r\n\r\n //altering newCameraTarget to take shortest path\r\n if (newCameraTarget.alpha > 0 && crossesZero) {\r\n newCameraTarget.alpha = activeCamera.alpha - shortestArc;\r\n } else if (newCameraTarget.alpha < 0 && crossesZero) {\r\n newCameraTarget.alpha = activeCamera.alpha + shortestArc;\r\n }\r\n\r\n translateA.setKeys([\r\n { frame: 0, value: activeCamera.alpha },\r\n { frame: endFrame, value: newCameraTarget.alpha },\r\n ]);\r\n const translateB = new Animation(\r\n 'translateCameraB',\r\n 'beta',\r\n ANIMATION_FPS,\r\n Animation.ANIMATIONTYPE_FLOAT,\r\n Animation.ANIMATIONLOOPMODE_CONSTANT\r\n );\r\n translateB.setKeys([\r\n { frame: 0, value: activeCamera.beta },\r\n { frame: endFrame, value: newCameraTarget.beta },\r\n ]);\r\n const translateR = new Animation(\r\n 'translateCameraR',\r\n 'radius',\r\n ANIMATION_FPS,\r\n Animation.ANIMATIONTYPE_FLOAT,\r\n Animation.ANIMATIONLOOPMODE_CONSTANT\r\n );\r\n translateR.setKeys([\r\n { frame: 0, value: activeCamera.radius },\r\n { frame: endFrame, value: newCameraTarget.radius },\r\n ]);\r\n const translateOffset = new Animation(\r\n 'translateCameraOffset',\r\n 'targetScreenOffset',\r\n ANIMATION_FPS,\r\n Animation.ANIMATIONTYPE_VECTOR2,\r\n Animation.ANIMATIONLOOPMODE_CONSTANT\r\n );\r\n translateOffset.setKeys([\r\n { frame: 0, value: activeCamera.targetScreenOffset },\r\n { frame: endFrame, value: newCameraTarget.targetScreenOffset },\r\n ]);\r\n const translateFov = new Animation(\r\n 'translateCameraFOV',\r\n 'fov',\r\n ANIMATION_FPS,\r\n Animation.ANIMATIONTYPE_FLOAT,\r\n Animation.ANIMATIONLOOPMODE_CONSTANT\r\n );\r\n translateFov.setKeys([\r\n { frame: 0, value: activeCamera.fov },\r\n { frame: endFrame, value: newCameraTarget.fov },\r\n ]);\r\n if (\r\n activeCamera instanceof KbArcRotateCamera &&\r\n activeCamera.mode === Camera.ORTHOGRAPHIC_CAMERA &&\r\n newCameraTarget.mode === Camera.ORTHOGRAPHIC_CAMERA\r\n ) {\r\n const translateOrthoWidth = new Animation(\r\n 'translateOrthoWidth',\r\n 'orthoWidth',\r\n ANIMATION_FPS,\r\n Animation.ANIMATIONTYPE_FLOAT,\r\n Animation.ANIMATIONLOOPMODE_CONSTANT\r\n );\r\n translateOrthoWidth.setKeys([\r\n { frame: 0, value: activeCamera.orthoWidth },\r\n { frame: endFrame, value: newCameraTarget.orthoWidth },\r\n ]);\r\n translateOrthoWidth.setEasingFunction(ease);\r\n\r\n animations.push(translateOrthoWidth);\r\n }\r\n\r\n translateA.setEasingFunction(ease);\r\n translateB.setEasingFunction(ease);\r\n translateR.setEasingFunction(ease);\r\n translateOffset.setEasingFunction(ease);\r\n translateFov.setEasingFunction(circEaseOut);\r\n\r\n animations.push(translateA);\r\n animations.push(translateB);\r\n animations.push(translateR);\r\n animations.push(translateOffset);\r\n animations.push(translateFov);\r\n }\r\n\r\n animations.forEach(animation => {\r\n activeCamera.animations.push(animation);\r\n });\r\n\r\n return new Promise(resolve => {\r\n this._activeAnimation = this.scene.beginAnimation(activeCamera, 0, endFrame, false, 1, () => {\r\n // on end of animation\r\n delete this._activeAnimation;\r\n\r\n this._activateCamera(newCamera);\r\n resolve();\r\n });\r\n });\r\n }\r\n\r\n storeCameraSettings() {\r\n if (!this.scene.activeCamera) {\r\n throw Error('No active camera');\r\n }\r\n this._storedViewpoint = new ViewpointNode();\r\n const camera = this.scene.activeCamera as KbArcRotateCamera | UniversalCamera;\r\n\r\n copyCameraStateIntoViewpoint(this._storedViewpoint, camera);\r\n }\r\n\r\n restoreCameraSettings() {\r\n if (this._storedViewpoint) {\r\n this.moveCameraToViewpoint(this._storedViewpoint);\r\n } else {\r\n console.warn('No stored camera settings to restore');\r\n }\r\n }\r\n\r\n watch(): Observable> {\r\n this._cameraSettingsChange = new Subject();\r\n\r\n return this._cameraSettingsChange.pipe(\r\n finalize(() => {\r\n delete this._cameraSettingsChange;\r\n }),\r\n share()\r\n );\r\n }\r\n\r\n maintainNodeScreenSize(renderedNode: TransformNode, relativeSize: number | ScalingHandler) {\r\n this.objectsToScaleWithDistance.set(renderedNode, relativeSize);\r\n }\r\n\r\n clearMaintainedNodeScreenSize(renderedNode: TransformNode) {\r\n this.objectsToScaleWithDistance.delete(renderedNode);\r\n }\r\n\r\n private _activateCamera(newCamera: Camera) {\r\n this.scene.cameras.forEach(camera => {\r\n if (camera !== newCamera) {\r\n /**\r\n * Don't dispose the whole camera because that goes too hard and disposes all the render target textures and\r\n * post processes which messes things up for the next camera that's reusing those things. Instead, just dispose the\r\n * parts of the camera we don't need.\r\n */\r\n camera.onViewMatrixChangedObservable.clear();\r\n camera.onProjectionMatrixChangedObservable.clear();\r\n camera.onAfterCheckInputsObservable.clear();\r\n camera.onRestoreStateObservable.clear();\r\n\r\n // Inputs\r\n if (camera.inputs) {\r\n camera.inputs.clear();\r\n }\r\n\r\n // Animations\r\n this.scene.stopAnimation(camera);\r\n\r\n // Remove from scene\r\n this.scene.removeCamera(camera);\r\n\r\n this.scene.getEngine().releaseRenderPassId(camera.renderPassId);\r\n }\r\n });\r\n this.scene.customRenderTargets.forEach(rtt => {\r\n rtt.activeCamera = newCamera;\r\n });\r\n this.applyScenePostProcesses(newCamera);\r\n newCamera!.attachControl(false);\r\n this.scene.activeCamera = newCamera;\r\n }\r\n\r\n private applyScenePostProcesses(camera: Camera) {\r\n if (this.activePipeline) {\r\n this.scene.postProcessRenderPipelineManager.attachCamerasToRenderPipeline(this.activePipeline.name, camera);\r\n }\r\n }\r\n\r\n private createCameraFromViewpoint(\r\n viewpoint: ViewpointNode,\r\n convertToCameraType: eCameraType.Universal\r\n ): Nullable;\r\n private createCameraFromViewpoint(\r\n viewpoint: ViewpointNode,\r\n convertToCameraType: eCameraType.ArcRotate\r\n ): Nullable;\r\n private createCameraFromViewpoint(viewpoint: ViewpointNode, convertToCameraType?: eCameraType): Nullable;\r\n private createCameraFromViewpoint(viewpoint: ViewpointNode, convertToCameraType?: eCameraType): Nullable {\r\n const viewer = this.kbViewer;\r\n const bbCache = this.kbViewer.bbCache;\r\n const sceneNode = this.kbViewer.sceneNode;\r\n\r\n if (this.disposePreviousCamera && !this.disposePreviousCamera.isStopped) {\r\n this.disposePreviousCamera.next();\r\n this.disposePreviousCamera.complete();\r\n }\r\n const dispose = (this.disposePreviousCamera = new Subject());\r\n\r\n switch (viewpoint.type) {\r\n case eViewpointType.orbit:\r\n case eViewpointType.overhead:\r\n if (convertToCameraType === eCameraType.Universal) {\r\n const universalCamera = new UniversalCamera(viewpoint.id, viewpoint.position.toVec3(), this.scene);\r\n\r\n // this should be set in the setCommonViewpointProperties\r\n // const direction = viewpoint.targetPoint!.subtract(viewpoint.position).toVec3();\r\n // universalCamera.rotation = rotationToVector(direction).toEulerAngles();\r\n applyViewpointProperties(viewpoint, universalCamera);\r\n\r\n return universalCamera;\r\n } else {\r\n const arcRotateCamera = new KbArcRotateCamera(\r\n viewpoint.id,\r\n 0,\r\n 0,\r\n 0,\r\n viewpoint.targetPoint ? viewpoint.targetPoint.toVec3() : Vector3.Zero(),\r\n this.scene\r\n );\r\n\r\n applyViewpointProperties(viewpoint, arcRotateCamera);\r\n\r\n // overhead camera overrides target\r\n if (viewpoint.type === eViewpointType.overhead) {\r\n arcRotateCamera.lockRotation = true;\r\n arcRotateCamera.target = new Vector3(viewpoint.position.x, 0, viewpoint.position.z);\r\n }\r\n\r\n return arcRotateCamera;\r\n }\r\n case eViewpointType.firstPerson:\r\n //case eViewpointType.fixedPosition:\r\n if (convertToCameraType === eCameraType.ArcRotate) {\r\n const arcRotateCamera = new KbArcRotateCamera(\r\n viewpoint.id,\r\n 0,\r\n 0,\r\n 0,\r\n viewpoint.position.toVec3(),\r\n this.scene\r\n );\r\n\r\n const orbitRotation = viewpoint.rotation.toVec3().scale(Math.PI / 180);\r\n applyViewpointProperties(viewpoint, arcRotateCamera);\r\n arcRotateCamera.alpha = orbitRotation.y - Math.PI / 2;\r\n arcRotateCamera.beta = Math.PI / 2 - orbitRotation.x;\r\n arcRotateCamera.radius = 0.0001;\r\n\r\n return arcRotateCamera;\r\n } else {\r\n const universalCamera = new UniversalCamera(viewpoint.id, viewpoint.position.toVec3(), this.scene);\r\n\r\n applyViewpointProperties(viewpoint, universalCamera);\r\n\r\n return universalCamera;\r\n }\r\n }\r\n\r\n return null;\r\n\r\n function applyViewpointProperties(viewpoint: ViewpointNode, camera: KbArcRotateCamera | UniversalCamera) {\r\n if (viewpoint.projection === eViewpointProjection.orthographic) {\r\n camera.mode = Camera.ORTHOGRAPHIC_CAMERA;\r\n camera.maxZ = 200000;\r\n if (camera instanceof ArcRotateCamera) {\r\n camera.orthoWidth = viewpoint.fov || 8;\r\n camera.wheelPrecision = 60;\r\n camera.pinchPrecision = 60;\r\n camera.setPosition(\r\n viewpoint.position.toVec3().subtract(camera.target).normalize().scale(5000).add(camera.target)\r\n );\r\n }\r\n } else if (viewpoint.projection === eViewpointProjection.perspective) {\r\n camera.mode = Camera.PERSPECTIVE_CAMERA;\r\n camera.fov = (viewpoint.fov * Math.PI) / 180;\r\n if (camera instanceof ArcRotateCamera) {\r\n camera.wheelDeltaPercentage = 0.01;\r\n //camera.wheelPrecision = 10;\r\n camera.speed = 0.0001;\r\n camera.lowerRadiusLimit = 0.0001;\r\n\r\n camera.pinchDeltaPercentage = 0.001;\r\n camera.setPosition(viewpoint.position.toVec3());\r\n }\r\n }\r\n\r\n if (camera instanceof ArcRotateCamera) {\r\n if (!viewpoint.allowOrbit) {\r\n camera.lockRotation = true;\r\n }\r\n if (!viewpoint.allowPan) {\r\n camera.lockPanning = true;\r\n }\r\n if (!viewpoint.allowZoom) {\r\n camera.lockZoom = true;\r\n }\r\n camera.useFramingBehavior = false;\r\n camera.targetScreenOffset.x = viewpoint.offsetX;\r\n camera.targetScreenOffset.y = viewpoint.offsetY;\r\n if (viewpoint.limitMinimumTargetDistance) {\r\n viewer.bbCache.rootBoundingBoxChanged\r\n .pipe(takeUntil(dispose))\r\n .pipe(startWith(null))\r\n .subscribe(() => {\r\n const bounding = viewer.bbCache.getWorld(sceneNode);\r\n if (bounding) {\r\n const minVector = new Vector3(bounding.min.x, bounding.min.y, bounding.min.z);\r\n const maxVector = new Vector3(bounding.max.x, bounding.max.y, bounding.max.z);\r\n const boundingSphere = new BoundingSphere(minVector, maxVector);\r\n camera.lowerRadiusLimit = boundingSphere.radiusWorld;\r\n } else {\r\n camera.lowerRadiusLimit = viewpoint.minTargetDistance;\r\n }\r\n });\r\n } else {\r\n camera.lowerRadiusLimit = viewpoint.minTargetDistance;\r\n }\r\n\r\n camera.upperRadiusLimit = viewpoint.maxTargetDistance;\r\n\r\n if (viewpoint.orbitAxis) {\r\n const upVector = viewpoint.orbitAxis.toVec3();\r\n if (upVector.length() === 0) {\r\n upVector.set(0, 1, 0);\r\n }\r\n camera.upVector = upVector;\r\n }\r\n if (viewpoint.limitLngOrbit) {\r\n camera.lowerAlphaLimit = (viewpoint.limitMinLongitude * Math.PI) / 180;\r\n camera.upperAlphaLimit = (viewpoint.limitMaxLongitude * Math.PI) / 180;\r\n } else {\r\n camera.lowerAlphaLimit = null;\r\n camera.upperAlphaLimit = null;\r\n }\r\n\r\n if (viewpoint.limitLatOrbit) {\r\n camera.lowerBetaLimit = (viewpoint.limitMinLatitude * Math.PI) / 180;\r\n camera.upperBetaLimit = (viewpoint.limitMaxLatitude * Math.PI) / 180;\r\n } else {\r\n camera.lowerBetaLimit = 0.001;\r\n camera.upperBetaLimit = Math.PI - 0.001;\r\n }\r\n } else if (camera instanceof UniversalCamera) {\r\n // // For fixed rotation, only allow targeting the camera with the mouse and disable movement.\r\n // if (viewpoint.type === eViewpointType.fixedPosition) {\r\n // camera.inputs.clear();\r\n // camera.inputs.addMouse();\r\n // }\r\n camera.inputs.clear();\r\n if (viewpoint.allowPan) {\r\n enableFreeMove(camera);\r\n }\r\n if (viewpoint.allowOrbit) {\r\n enableFreeLook(camera);\r\n }\r\n }\r\n\r\n if (viewpoint.targetMode === eViewpointTargetMode.point) {\r\n camera.setTarget(viewpoint.targetPoint.toVec3());\r\n } else if (viewpoint.targetMode === eViewpointTargetMode.mesh) {\r\n const targetMeshNode = viewpoint._manager!.getById(viewpoint.targetMeshId!) as MeshNode;\r\n if (targetMeshNode) {\r\n // TODO: When updating to babylon 5.0, use targetHost property for targetting mesh\r\n bbCache.rootBoundingBoxChanged.pipe(takeUntil(dispose)).subscribe(() => {\r\n const bbInfo = bbCache.getWorld(targetMeshNode);\r\n if (bbInfo) {\r\n camera.setTarget(bbInfo.center.toVec3());\r\n }\r\n });\r\n }\r\n } else {\r\n // rotation\r\n if (camera instanceof ArcRotateCamera) {\r\n camera.alpha = (viewpoint.rotation.y * Math.PI) / 180; // yaw = longitude\r\n camera.beta = (viewpoint.rotation.x * Math.PI) / 180; // pitch = latitude\r\n } else {\r\n camera.rotation = viewpoint.rotation.toVec3().scale(Math.PI / 180);\r\n }\r\n }\r\n\r\n bbCache\r\n .watchForChanges(sceneNode)\r\n .pipe(takeUntil(dispose))\r\n .subscribe(bbInfo => {\r\n if (bbInfo && bbInfo.boundingInfo.diagonalLength > 0) {\r\n const cameraDistance = bbInfo.boundingInfo.diagonalLength * 2;\r\n /**\r\n * This piece of code is very important. if the maxZ is too small, things will disappear as you zoom out.\r\n * If the maxZ is too large, the camera will not have enough buffer and\r\n * all kinds of things get messed up - clicking doesn't work, you start getting z-fighting.\r\n * Smallest the maxZ should be \"10\" due to the grid.\r\n * However ortho camera should be double that because calculations are made from half of the frame\r\n * And perspective camera should also take the camera distance into account (but have some sane multiplier based on the mesh size)\r\n */\r\n // should we set the minZ here as well, or just hard code it?\r\n if (\r\n viewpoint.projection === eViewpointProjection.orthographic &&\r\n camera instanceof ArcRotateCamera\r\n ) {\r\n camera.setPosition(\r\n camera.position\r\n .subtract(camera.target)\r\n .normalize()\r\n .scale(cameraDistance)\r\n .add(camera.target)\r\n );\r\n camera.maxZ = Math.max(cameraDistance, 10) * 2;\r\n } else if (camera instanceof ArcRotateCamera && camera.upperRadiusLimit) {\r\n camera.maxZ = Math.max(\r\n Math.max(cameraDistance, 10) * 16,\r\n camera.upperRadiusLimit + cameraDistance\r\n );\r\n } else {\r\n camera.maxZ = Math.max(cameraDistance, 10) * 16;\r\n }\r\n\r\n // If maxZ is too high then it breaks clicking because the unprojected rays start returning NaN/Infinity\r\n if (camera.maxZ > 200000) {\r\n camera.maxZ = 200000;\r\n }\r\n\r\n if (viewer.rendererManager.sceneRenderer?.skybox) {\r\n // need to resize skybox if it exists, otherwise it will get clipped\r\n const scale = camera.maxZ / 2 / SKYBOX_SIZE;\r\n viewer.rendererManager.sceneRenderer.skybox.scaling = new Vector3(scale, scale, scale);\r\n }\r\n }\r\n });\r\n\r\n camera.onDisposeObservable.add(() => {\r\n dispose.next();\r\n dispose.complete();\r\n });\r\n camera.minZ = 0.01; // hack the camera minz until we have a more automated way to set it\r\n camera.doNotSerialize = true;\r\n }\r\n }\r\n\r\n orient(orientation: eCameraOrientation) {\r\n const bbCache = this.kbViewer.bbCache;\r\n\r\n if (this._lastOrientation === eCameraOrientation.perspective) {\r\n this._lastPerspectivePosition = KbVector.From(this.scene.activeCamera!.position);\r\n }\r\n //get a viewpoint representing the current camera settings\r\n const viewpoint = new ViewpointNode();\r\n const camera = this.scene.activeCamera as KbArcRotateCamera | UniversalCamera;\r\n copyCameraStateIntoViewpoint(viewpoint, camera);\r\n\r\n let targetPoint = KbVector.Zero();\r\n if (viewpoint.targetMode == eViewpointTargetMode.point) {\r\n targetPoint = viewpoint.targetPoint;\r\n } else if (viewpoint.targetMode == eViewpointTargetMode.mesh) {\r\n const targetMeshNode = this.kbViewer.sceneNode._manager!.getById(viewpoint.targetMeshId!) as MeshNode;\r\n if (targetMeshNode) {\r\n const bbInfo = bbCache.getWorld(targetMeshNode);\r\n if (bbInfo) {\r\n targetPoint = bbInfo.center;\r\n }\r\n }\r\n }\r\n let distance = viewpoint.position.subtract(targetPoint).length();\r\n\r\n let newPos = targetPoint.toObject();\r\n if (orientation == eCameraOrientation.top) {\r\n newPos.y += distance;\r\n } else if (orientation == eCameraOrientation.front) {\r\n newPos.z += distance;\r\n } else if (orientation == eCameraOrientation.left) {\r\n newPos.x -= distance;\r\n } else if (orientation == eCameraOrientation.back) {\r\n newPos.z -= distance;\r\n } else if (orientation == eCameraOrientation.right) {\r\n newPos.x += distance;\r\n } else if (orientation == eCameraOrientation.bottom) {\r\n newPos.y -= distance;\r\n } else if (orientation == eCameraOrientation.perspective) {\r\n const lastLength = this._lastPerspectivePosition.length();\r\n const ratio = distance / lastLength;\r\n newPos = this._lastPerspectivePosition.scale(ratio);\r\n }\r\n viewpoint.position = KbVector.From(newPos);\r\n if (orientation != eCameraOrientation.perspective) {\r\n viewpoint.projection = eViewpointProjection.orthographic;\r\n viewpoint.allowOrbit = false;\r\n } else {\r\n viewpoint.projection = eViewpointProjection.perspective;\r\n viewpoint.allowOrbit = true;\r\n }\r\n\r\n if (\r\n this._lastOrientation === eCameraOrientation.perspective &&\r\n orientation !== eCameraOrientation.perspective\r\n ) {\r\n //switching to orthographic so we need to deal with fov\r\n viewpoint.fov = 2 * distance * Math.tan(toRadians(viewpoint.fov / 2));\r\n } else if (\r\n this._lastOrientation !== eCameraOrientation.perspective &&\r\n orientation === eCameraOrientation.perspective\r\n ) {\r\n const orthoWidth = viewpoint.fov; //in orthographic, fov is the ortho width\r\n distance = orthoWidth / (2 * Math.tan(toRadians(defaults.ViewpointNode.fov / 2)));\r\n viewpoint.position = this._lastPerspectivePosition.scale(distance / this._lastPerspectivePosition.length());\r\n viewpoint.fov = defaults.ViewpointNode.fov;\r\n }\r\n viewpoint.offsetX = 0;\r\n viewpoint.offsetY = 0;\r\n\r\n this.moveCameraToViewpoint(viewpoint);\r\n this.frameCamera(this.kbViewer.sceneNode, {\r\n rotate: false,\r\n pan: true,\r\n zoom: false,\r\n });\r\n\r\n this._lastOrientation = orientation;\r\n }\r\n}\r\n\r\nfunction enableFreeLook(camera: UniversalCamera) {\r\n camera.inputs.addMouse();\r\n}\r\n\r\nfunction enableFreeMove(camera: UniversalCamera) {\r\n camera.inputs.addKeyboard();\r\n camera.inputs.addTouch();\r\n}\r\n\r\nfunction copyCameraStateIntoViewpoint(viewpoint: Partial, camera: Camera) {\r\n let normalizedPosition = camera.position;\r\n\r\n // If this is an orthographic camera, the real position is set very far away to prevent clipping. So normalize the position so the biggest dimension is 5.\r\n if (camera instanceof ArcRotateCamera && camera.mode === Camera.ORTHOGRAPHIC_CAMERA) {\r\n normalizedPosition = normalizedPosition.subtract(camera.position).normalize().add(camera.position);\r\n if (normalizedPosition.length() > 0) {\r\n const scale = Math.max(...normalizedPosition.asArray().map(dim => Math.abs(dim)));\r\n normalizedPosition = Vector3.FromArray(\r\n normalizedPosition.asArray().map(dim => Math.round((dim / scale) * 500) / 100)\r\n );\r\n }\r\n }\r\n\r\n if (camera instanceof FreeCamera) {\r\n const rotDegrees = camera.rotation.scale(180 / Math.PI);\r\n viewpoint.rotation = KbVector.FromVec3(rotDegrees);\r\n }\r\n viewpoint.position = KbVector.FromVec3(normalizedPosition);\r\n viewpoint.offsetX = camera instanceof KbArcRotateCamera ? camera.targetScreenOffset.x : 0;\r\n viewpoint.offsetY = camera instanceof KbArcRotateCamera ? camera.targetScreenOffset.y : 0;\r\n viewpoint.fov =\r\n camera.mode === Camera.ORTHOGRAPHIC_CAMERA\r\n ? camera.orthoRight! - camera.orthoLeft!\r\n : (camera.fov * 180) / Math.PI;\r\n viewpoint.projection =\r\n camera.mode === Camera.ORTHOGRAPHIC_CAMERA\r\n ? eViewpointProjection.orthographic\r\n : eViewpointProjection.perspective;\r\n}\r\n\r\nfunction normalizeRotation(alpha: number): number {\r\n const oneRotation = Math.PI * 2;\r\n const rotations = Math.floor(alpha / oneRotation);\r\n const radiansToRemove = rotations > 1 ? rotations * oneRotation : 0;\r\n const normalizedRotation = alpha - radiansToRemove;\r\n return normalizedRotation;\r\n}\r\n","import { Observable } from '../Observable';\nimport { concat } from '../observable/concat';\nimport { isScheduler } from '../util/isScheduler';\nimport { MonoTypeOperatorFunction, OperatorFunction, SchedulerLike } from '../types';\n\n/* tslint:disable:max-line-length */\n/** @deprecated use {@link scheduled} and {@link concatAll} (e.g. `scheduled([[a, b, c], source], scheduler).pipe(concatAll())`) */\nexport function startWith(scheduler: SchedulerLike): MonoTypeOperatorFunction;\n/** @deprecated use {@link scheduled} and {@link concatAll} (e.g. `scheduled([[a, b, c], source], scheduler).pipe(concatAll())`) */\nexport function startWith(v1: D, scheduler: SchedulerLike): OperatorFunction;\n/** @deprecated use {@link scheduled} and {@link concatAll} (e.g. `scheduled([[a, b, c], source], scheduler).pipe(concatAll())`) */\nexport function startWith(v1: D, v2: E, scheduler: SchedulerLike): OperatorFunction;\n/** @deprecated use {@link scheduled} and {@link concatAll} (e.g. `scheduled([[a, b, c], source], scheduler).pipe(concatAll())`) */\nexport function startWith(v1: D, v2: E, v3: F, scheduler: SchedulerLike): OperatorFunction;\n/** @deprecated use {@link scheduled} and {@link concatAll} (e.g. `scheduled([[a, b, c], source], scheduler).pipe(concatAll())`) */\nexport function startWith(v1: D, v2: E, v3: F, v4: G, scheduler: SchedulerLike): OperatorFunction;\n/** @deprecated use {@link scheduled} and {@link concatAll} (e.g. `scheduled([[a, b, c], source], scheduler).pipe(concatAll())`) */\nexport function startWith(v1: D, v2: E, v3: F, v4: G, v5: H, scheduler: SchedulerLike): OperatorFunction;\n/** @deprecated use {@link scheduled} and {@link concatAll} (e.g. `scheduled([[a, b, c], source], scheduler).pipe(concatAll())`) */\nexport function startWith(v1: D, v2: E, v3: F, v4: G, v5: H, v6: I, scheduler: SchedulerLike): OperatorFunction;\n\nexport function startWith(v1: D): OperatorFunction;\nexport function startWith(v1: D, v2: E): OperatorFunction;\nexport function startWith(v1: D, v2: E, v3: F): OperatorFunction;\nexport function startWith(v1: D, v2: E, v3: F, v4: G): OperatorFunction;\nexport function startWith(v1: D, v2: E, v3: F, v4: G, v5: H): OperatorFunction;\nexport function startWith(v1: D, v2: E, v3: F, v4: G, v5: H, v6: I): OperatorFunction;\nexport function startWith(...array: D[]): OperatorFunction;\n/** @deprecated use {@link scheduled} and {@link concatAll} (e.g. `scheduled([[a, b, c], source], scheduler).pipe(concatAll())`) */\nexport function startWith(...array: Array): OperatorFunction;\n/* tslint:enable:max-line-length */\n\n/**\n * Returns an Observable that emits the items you specify as arguments before it begins to emit\n * items emitted by the source Observable.\n *\n * First emits its arguments in order, and then any\n * emissions from the source.\n *\n * ![](startWith.png)\n *\n * ## Examples\n *\n * Start the chain of emissions with `\"first\"`, `\"second\"`\n *\n * ```ts\n * import { of } from 'rxjs';\n * import { startWith } from 'rxjs/operators';\n *\n * of(\"from source\")\n * .pipe(startWith(\"first\", \"second\"))\n * .subscribe(x => console.log(x));\n *\n * // results:\n * // \"first\"\n * // \"second\"\n * // \"from source\"\n * ```\n *\n * @param {...T} values - Items you want the modified Observable to emit first.\n * @param {SchedulerLike} [scheduler] - A {@link SchedulerLike} to use for scheduling\n * the emissions of the `next` notifications.\n * @return {Observable} An Observable that emits the items in the specified Iterable and then emits the items\n * emitted by the source Observable.\n * @method startWith\n * @owner Observable\n */\nexport function startWith(...array: Array): OperatorFunction {\n const scheduler = array[array.length - 1] as SchedulerLike;\n if (isScheduler(scheduler)) {\n // deprecated path\n array.pop();\n return (source: Observable) => concat(array as T[], source, scheduler);\n } else {\n return (source: Observable) => concat(array as T[], source);\n }\n}\n","import type { SmartArrayNoDuplicate } from \"../../Misc/smartArray\";\r\nimport { Vector3 } from \"../../Maths/math.vector\";\r\nimport type { Ray } from \"../../Culling/ray\";\r\nimport { BoundingBox } from \"../../Culling/boundingBox\";\r\nimport type { Plane } from \"../../Maths/math.plane\";\r\n\r\n/**\r\n * Contains an array of blocks representing the octree\r\n */\r\nexport interface IOctreeContainer {\r\n /**\r\n * Blocks within the octree\r\n */\r\n blocks: Array>;\r\n}\r\n\r\n/**\r\n * Class used to store a cell in an octree\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimizeOctrees\r\n */\r\nexport class OctreeBlock {\r\n /**\r\n * Gets the content of the current block\r\n */\r\n public entries = new Array();\r\n\r\n /**\r\n * Gets the list of block children\r\n */\r\n public blocks: Array>;\r\n\r\n private _depth: number;\r\n private _maxDepth: number;\r\n private _capacity: number;\r\n private _minPoint: Vector3;\r\n private _maxPoint: Vector3;\r\n private _boundingVectors = new Array();\r\n private _creationFunc: (entry: T, block: OctreeBlock) => void;\r\n\r\n /**\r\n * Creates a new block\r\n * @param minPoint defines the minimum vector (in world space) of the block's bounding box\r\n * @param maxPoint defines the maximum vector (in world space) of the block's bounding box\r\n * @param capacity defines the maximum capacity of this block (if capacity is reached the block will be split into sub blocks)\r\n * @param depth defines the current depth of this block in the octree\r\n * @param maxDepth defines the maximal depth allowed (beyond this value, the capacity is ignored)\r\n * @param creationFunc defines a callback to call when an element is added to the block\r\n */\r\n constructor(minPoint: Vector3, maxPoint: Vector3, capacity: number, depth: number, maxDepth: number, creationFunc: (entry: T, block: OctreeBlock) => void) {\r\n this._capacity = capacity;\r\n this._depth = depth;\r\n this._maxDepth = maxDepth;\r\n this._creationFunc = creationFunc;\r\n\r\n this._minPoint = minPoint;\r\n this._maxPoint = maxPoint;\r\n\r\n this._boundingVectors.push(minPoint.clone());\r\n this._boundingVectors.push(maxPoint.clone());\r\n\r\n this._boundingVectors.push(minPoint.clone());\r\n this._boundingVectors[2].x = maxPoint.x;\r\n\r\n this._boundingVectors.push(minPoint.clone());\r\n this._boundingVectors[3].y = maxPoint.y;\r\n\r\n this._boundingVectors.push(minPoint.clone());\r\n this._boundingVectors[4].z = maxPoint.z;\r\n\r\n this._boundingVectors.push(maxPoint.clone());\r\n this._boundingVectors[5].z = minPoint.z;\r\n\r\n this._boundingVectors.push(maxPoint.clone());\r\n this._boundingVectors[6].x = minPoint.x;\r\n\r\n this._boundingVectors.push(maxPoint.clone());\r\n this._boundingVectors[7].y = minPoint.y;\r\n }\r\n\r\n // Property\r\n\r\n /**\r\n * Gets the maximum capacity of this block (if capacity is reached the block will be split into sub blocks)\r\n */\r\n public get capacity(): number {\r\n return this._capacity;\r\n }\r\n\r\n /**\r\n * Gets the minimum vector (in world space) of the block's bounding box\r\n */\r\n public get minPoint(): Vector3 {\r\n return this._minPoint;\r\n }\r\n\r\n /**\r\n * Gets the maximum vector (in world space) of the block's bounding box\r\n */\r\n public get maxPoint(): Vector3 {\r\n return this._maxPoint;\r\n }\r\n\r\n // Methods\r\n\r\n /**\r\n * Add a new element to this block\r\n * @param entry defines the element to add\r\n */\r\n public addEntry(entry: T): void {\r\n if (this.blocks) {\r\n for (let index = 0; index < this.blocks.length; index++) {\r\n const block = this.blocks[index];\r\n block.addEntry(entry);\r\n }\r\n return;\r\n }\r\n\r\n this._creationFunc(entry, this);\r\n\r\n if (this.entries.length > this.capacity && this._depth < this._maxDepth) {\r\n this.createInnerBlocks();\r\n }\r\n }\r\n\r\n /**\r\n * Remove an element from this block\r\n * @param entry defines the element to remove\r\n */\r\n public removeEntry(entry: T): void {\r\n if (this.blocks) {\r\n for (let index = 0; index < this.blocks.length; index++) {\r\n const block = this.blocks[index];\r\n block.removeEntry(entry);\r\n }\r\n return;\r\n }\r\n\r\n const entryIndex = this.entries.indexOf(entry);\r\n\r\n if (entryIndex > -1) {\r\n this.entries.splice(entryIndex, 1);\r\n }\r\n }\r\n\r\n /**\r\n * Add an array of elements to this block\r\n * @param entries defines the array of elements to add\r\n */\r\n public addEntries(entries: T[]): void {\r\n for (let index = 0; index < entries.length; index++) {\r\n const mesh = entries[index];\r\n this.addEntry(mesh);\r\n }\r\n }\r\n\r\n /**\r\n * Test if the current block intersects the frustum planes and if yes, then add its content to the selection array\r\n * @param frustumPlanes defines the frustum planes to test\r\n * @param selection defines the array to store current content if selection is positive\r\n * @param allowDuplicate defines if the selection array can contains duplicated entries\r\n */\r\n public select(frustumPlanes: Plane[], selection: SmartArrayNoDuplicate, allowDuplicate?: boolean): void {\r\n if (BoundingBox.IsInFrustum(this._boundingVectors, frustumPlanes)) {\r\n if (this.blocks) {\r\n for (let index = 0; index < this.blocks.length; index++) {\r\n const block = this.blocks[index];\r\n block.select(frustumPlanes, selection, allowDuplicate);\r\n }\r\n return;\r\n }\r\n\r\n if (allowDuplicate) {\r\n selection.concat(this.entries);\r\n } else {\r\n selection.concatWithNoDuplicate(this.entries);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Test if the current block intersect with the given bounding sphere and if yes, then add its content to the selection array\r\n * @param sphereCenter defines the bounding sphere center\r\n * @param sphereRadius defines the bounding sphere radius\r\n * @param selection defines the array to store current content if selection is positive\r\n * @param allowDuplicate defines if the selection array can contains duplicated entries\r\n */\r\n public intersects(sphereCenter: Vector3, sphereRadius: number, selection: SmartArrayNoDuplicate, allowDuplicate?: boolean): void {\r\n if (BoundingBox.IntersectsSphere(this._minPoint, this._maxPoint, sphereCenter, sphereRadius)) {\r\n if (this.blocks) {\r\n for (let index = 0; index < this.blocks.length; index++) {\r\n const block = this.blocks[index];\r\n block.intersects(sphereCenter, sphereRadius, selection, allowDuplicate);\r\n }\r\n return;\r\n }\r\n\r\n if (allowDuplicate) {\r\n selection.concat(this.entries);\r\n } else {\r\n selection.concatWithNoDuplicate(this.entries);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Test if the current block intersect with the given ray and if yes, then add its content to the selection array\r\n * @param ray defines the ray to test with\r\n * @param selection defines the array to store current content if selection is positive\r\n */\r\n public intersectsRay(ray: Ray, selection: SmartArrayNoDuplicate): void {\r\n if (ray.intersectsBoxMinMax(this._minPoint, this._maxPoint)) {\r\n if (this.blocks) {\r\n for (let index = 0; index < this.blocks.length; index++) {\r\n const block = this.blocks[index];\r\n block.intersectsRay(ray, selection);\r\n }\r\n return;\r\n }\r\n selection.concatWithNoDuplicate(this.entries);\r\n }\r\n }\r\n\r\n /**\r\n * Subdivide the content into child blocks (this block will then be empty)\r\n */\r\n public createInnerBlocks(): void {\r\n OctreeBlock._CreateBlocks(this._minPoint, this._maxPoint, this.entries, this._capacity, this._depth, this._maxDepth, this, this._creationFunc);\r\n this.entries.splice(0);\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _CreateBlocks(\r\n worldMin: Vector3,\r\n worldMax: Vector3,\r\n entries: T[],\r\n maxBlockCapacity: number,\r\n currentDepth: number,\r\n maxDepth: number,\r\n target: IOctreeContainer,\r\n creationFunc: (entry: T, block: OctreeBlock) => void\r\n ): void {\r\n target.blocks = new Array>();\r\n const blockSize = new Vector3((worldMax.x - worldMin.x) / 2, (worldMax.y - worldMin.y) / 2, (worldMax.z - worldMin.z) / 2);\r\n\r\n // Segmenting space\r\n for (let x = 0; x < 2; x++) {\r\n for (let y = 0; y < 2; y++) {\r\n for (let z = 0; z < 2; z++) {\r\n const localMin = worldMin.add(blockSize.multiplyByFloats(x, y, z));\r\n const localMax = worldMin.add(blockSize.multiplyByFloats(x + 1, y + 1, z + 1));\r\n\r\n const block = new OctreeBlock(localMin, localMax, maxBlockCapacity, currentDepth + 1, maxDepth, creationFunc);\r\n block.addEntries(entries);\r\n target.blocks.push(block);\r\n }\r\n }\r\n }\r\n }\r\n}\r\n","import type { SmartArray } from \"../../Misc/smartArray\";\r\nimport { SmartArrayNoDuplicate } from \"../../Misc/smartArray\";\r\nimport type { Vector3 } from \"../../Maths/math.vector\";\r\nimport type { SubMesh } from \"../../Meshes/subMesh\";\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport type { Ray } from \"../../Culling/ray\";\r\nimport { OctreeBlock } from \"./octreeBlock\";\r\nimport type { Plane } from \"../../Maths/math.plane\";\r\n\r\n/**\r\n * Octrees are a really powerful data structure that can quickly select entities based on space coordinates.\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimizeOctrees\r\n */\r\nexport class Octree {\r\n /**\r\n * Blocks within the octree containing objects\r\n */\r\n public blocks: Array>;\r\n /**\r\n * Content stored in the octree\r\n */\r\n public dynamicContent = new Array();\r\n\r\n private _maxBlockCapacity: number;\r\n private _selectionContent: SmartArrayNoDuplicate;\r\n private _creationFunc: (entry: T, block: OctreeBlock) => void;\r\n\r\n /**\r\n * Creates a octree\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/optimizeOctrees\r\n * @param creationFunc function to be used to instantiate the octree\r\n * @param maxBlockCapacity defines the maximum number of meshes you want on your octree's leaves (default: 64)\r\n * @param maxDepth defines the maximum depth (sub-levels) for your octree. Default value is 2, which means 8 8 8 = 512 blocks :) (This parameter takes precedence over capacity.)\r\n */\r\n constructor(\r\n creationFunc: (entry: T, block: OctreeBlock) => void,\r\n maxBlockCapacity?: number,\r\n /** Defines the maximum depth (sub-levels) for your octree. Default value is 2, which means 8 8 8 = 512 blocks :) (This parameter takes precedence over capacity.) */\r\n public maxDepth = 2\r\n ) {\r\n this._maxBlockCapacity = maxBlockCapacity || 64;\r\n this._selectionContent = new SmartArrayNoDuplicate(1024);\r\n this._creationFunc = creationFunc;\r\n }\r\n\r\n // Methods\r\n /**\r\n * Updates the octree by adding blocks for the passed in meshes within the min and max world parameters\r\n * @param worldMin worldMin for the octree blocks var blockSize = new Vector3((worldMax.x - worldMin.x) / 2, (worldMax.y - worldMin.y) / 2, (worldMax.z - worldMin.z) / 2);\r\n * @param worldMax worldMax for the octree blocks var blockSize = new Vector3((worldMax.x - worldMin.x) / 2, (worldMax.y - worldMin.y) / 2, (worldMax.z - worldMin.z) / 2);\r\n * @param entries meshes to be added to the octree blocks\r\n */\r\n public update(worldMin: Vector3, worldMax: Vector3, entries: T[]): void {\r\n OctreeBlock._CreateBlocks(worldMin, worldMax, entries, this._maxBlockCapacity, 0, this.maxDepth, this, this._creationFunc);\r\n }\r\n\r\n /**\r\n * Adds a mesh to the octree\r\n * @param entry Mesh to add to the octree\r\n */\r\n public addMesh(entry: T): void {\r\n for (let index = 0; index < this.blocks.length; index++) {\r\n const block = this.blocks[index];\r\n block.addEntry(entry);\r\n }\r\n }\r\n\r\n /**\r\n * Remove an element from the octree\r\n * @param entry defines the element to remove\r\n */\r\n public removeMesh(entry: T): void {\r\n for (let index = 0; index < this.blocks.length; index++) {\r\n const block = this.blocks[index];\r\n block.removeEntry(entry);\r\n }\r\n }\r\n\r\n /**\r\n * Selects an array of meshes within the frustum\r\n * @param frustumPlanes The frustum planes to use which will select all meshes within it\r\n * @param allowDuplicate If duplicate objects are allowed in the resulting object array\r\n * @returns array of meshes within the frustum\r\n */\r\n public select(frustumPlanes: Plane[], allowDuplicate?: boolean): SmartArray {\r\n this._selectionContent.reset();\r\n\r\n for (let index = 0; index < this.blocks.length; index++) {\r\n const block = this.blocks[index];\r\n block.select(frustumPlanes, this._selectionContent, allowDuplicate);\r\n }\r\n\r\n if (allowDuplicate) {\r\n this._selectionContent.concat(this.dynamicContent);\r\n } else {\r\n this._selectionContent.concatWithNoDuplicate(this.dynamicContent);\r\n }\r\n\r\n return this._selectionContent;\r\n }\r\n\r\n /**\r\n * Test if the octree intersect with the given bounding sphere and if yes, then add its content to the selection array\r\n * @param sphereCenter defines the bounding sphere center\r\n * @param sphereRadius defines the bounding sphere radius\r\n * @param allowDuplicate defines if the selection array can contains duplicated entries\r\n * @returns an array of objects that intersect the sphere\r\n */\r\n public intersects(sphereCenter: Vector3, sphereRadius: number, allowDuplicate?: boolean): SmartArray {\r\n this._selectionContent.reset();\r\n\r\n for (let index = 0; index < this.blocks.length; index++) {\r\n const block = this.blocks[index];\r\n block.intersects(sphereCenter, sphereRadius, this._selectionContent, allowDuplicate);\r\n }\r\n\r\n if (allowDuplicate) {\r\n this._selectionContent.concat(this.dynamicContent);\r\n } else {\r\n this._selectionContent.concatWithNoDuplicate(this.dynamicContent);\r\n }\r\n\r\n return this._selectionContent;\r\n }\r\n\r\n /**\r\n * Test if the octree intersect with the given ray and if yes, then add its content to resulting array\r\n * @param ray defines the ray to test with\r\n * @returns array of intersected objects\r\n */\r\n public intersectsRay(ray: Ray): SmartArray {\r\n this._selectionContent.reset();\r\n\r\n for (let index = 0; index < this.blocks.length; index++) {\r\n const block = this.blocks[index];\r\n block.intersectsRay(ray, this._selectionContent);\r\n }\r\n\r\n this._selectionContent.concatWithNoDuplicate(this.dynamicContent);\r\n\r\n return this._selectionContent;\r\n }\r\n\r\n /**\r\n * Adds a mesh into the octree block if it intersects the block\r\n * @param entry\r\n * @param block\r\n */\r\n public static CreationFuncForMeshes = (entry: AbstractMesh, block: OctreeBlock): void => {\r\n const boundingInfo = entry.getBoundingInfo();\r\n if (!entry.isBlocked && boundingInfo.boundingBox.intersectsMinMax(block.minPoint, block.maxPoint)) {\r\n block.entries.push(entry);\r\n }\r\n };\r\n\r\n /**\r\n * Adds a submesh into the octree block if it intersects the block\r\n * @param entry\r\n * @param block\r\n */\r\n public static CreationFuncForSubMeshes = (entry: SubMesh, block: OctreeBlock): void => {\r\n const boundingInfo = entry.getBoundingInfo();\r\n if (boundingInfo.boundingBox.intersectsMinMax(block.minPoint, block.maxPoint)) {\r\n block.entries.push(entry);\r\n }\r\n };\r\n}\r\n","import { Octree } from '@babylonjs/core/Culling/Octrees/octree';\r\nimport { Ray } from '@babylonjs/core/Culling/ray';\r\nimport { PointerEventTypes, PointerInfo } from '@babylonjs/core/Events/pointerEvents';\r\nimport { Matrix, Plane, Quaternion, Vector3 } from '@babylonjs/core/Maths';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { InstancedMesh } from '@babylonjs/core/Meshes/instancedMesh';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { Observer } from '@babylonjs/core/Misc/observable';\r\nimport { Node } from '@babylonjs/core/node';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { Nullable } from '@babylonjs/core/types';\r\nimport { eConnectionMode, eConnectorPositionMode, eDragMode, eMateType } from '@models/classes/enums';\r\nimport { Connector, isConnector } from '@models/classes/features/connector';\r\nimport { SketchControlPoint } from '@models/classes/features/sketch-control-point';\r\nimport { IConnection, IDragArgs, IDraggable, IKb3dPointerArgs, PointerInfoMinimal } from '@models/classes/handlers';\r\nimport { getByDynamicId } from '@models/classes/kb3d-manager';\r\nimport { Kb3dObject } from '@models/classes/kb3d-object';\r\nimport { isSpaceNode } from '@models/classes/meshes/mesh-utilities';\r\nimport { SpaceNode } from '@models/classes/meshes/space-node';\r\nimport { KbQuaternion } from '@models/classes/primitives/kb-quaternion';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { Token } from '@models/classes/token';\r\nimport { ITrackedClass } from '@models/interfaces/tracked-class';\r\nimport { mateConnectors } from '@view/helpers/mate-helper';\r\nimport { QuadTree } from '@view/helpers/quadtree';\r\nimport { IScreenPosition, toScreen } from '@view/helpers/scene-helper';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { ConnectorRenderer, getNodeChain } from '@view/render';\r\n\r\ninterface IPotential {\r\n otherMesh: TransformNode;\r\n proximity: number;\r\n distanceToCamera?: number;\r\n}\r\n\r\ninterface IPotentialConnector extends IPotential {\r\n dragConnector: Connector;\r\n dragConnectorMesh: TransformNode;\r\n}\r\n\r\ninterface IMeshScreenPosition {\r\n mesh: TransformNode;\r\n pos: IScreenPosition;\r\n}\r\n\r\ninterface IConnectionMesh extends IConnection {\r\n dragConnectorMesh: TransformNode;\r\n otherConnectorMesh: TransformNode;\r\n}\r\n\r\ntype MatedObject = SpaceNode | SketchControlPoint;\r\n\r\nconst CLICK_THRESHOLD = 8;\r\n\r\nexport class Dragger {\r\n constructor(private viewer: KbViewer, private scene: Scene) {}\r\n\r\n private _dragNode: SpaceNode;\r\n private _dragNodeStartPosition: KbVector;\r\n private _dragNodeInverseTransform: Matrix;\r\n private _dragMesh: TransformNode;\r\n private _draggable: IDraggable;\r\n private _pointerObserver: Nullable>;\r\n private _dragPlane: Plane;\r\n private _dragSurfaces: string[]; //array of dynamic id's of the drag surfaces if mode is \"surface\"\r\n private _startMouseIntersect?: Vector3;\r\n private _onComplete: () => void;\r\n private _externalConnectors = new Map(); //connectors out in the scene that are not part of what is being dragged\r\n private _dragConnectors = new Map(); //connectors that are children of what is being dragged\r\n private _dragConnectorsByDynamicId = new Map(); //the drag connectors stored by their dynamic id\r\n private _dropPreviewConnectors = new Map(); //the drop preview is a clone of the dragNode, this tracks the connector meshes in the drop preview\r\n private _oct: Octree; //the octree holding connectors for fast retrieval\r\n private _quadTree: QuadTree;\r\n private _dropPreview: TransformNode;\r\n private _activeConnection: IConnectionMesh | undefined;\r\n private _connectorRenderer: ConnectorRenderer;\r\n private _dragFriends = new Set();\r\n private _dragFriendMeshes = new Array();\r\n private _dragFriendConnectors = new Set();\r\n private _matedConnectors = new Set();\r\n\r\n public startDrag(pointerArgs: IKb3dPointerArgs, draggable: IDraggable, onComplete: () => void) {\r\n let totalMoved = 0;\r\n let lastX: number;\r\n let lastY: number;\r\n\r\n this._pointerObserver = this.scene.onPointerObservable.add(p => {\r\n switch (p.type) {\r\n case PointerEventTypes.POINTERMOVE: {\r\n let movX = 0;\r\n let movY = 0;\r\n if (typeof lastX === 'undefined' || typeof lastY === 'undefined') {\r\n lastX = p.event.offsetX;\r\n lastY = p.event.offsetY;\r\n } else {\r\n movX = Math.abs(lastX - p.event.offsetX);\r\n movY = Math.abs(lastY - p.event.offsetY);\r\n }\r\n totalMoved += Math.sqrt(movX * movX + movY * movY);\r\n\r\n if (totalMoved > CLICK_THRESHOLD) {\r\n this.startDragCommit(pointerArgs, draggable, onComplete);\r\n this.dragging(p);\r\n }\r\n\r\n break;\r\n }\r\n case PointerEventTypes.POINTERUP: {\r\n if (totalMoved > CLICK_THRESHOLD) {\r\n this.dropped(p);\r\n }\r\n this.scene.onPointerObservable.remove(this._pointerObserver);\r\n this._pointerObserver = undefined as any;\r\n break;\r\n }\r\n }\r\n });\r\n\r\n this.scene.activeCamera?.detachControl();\r\n }\r\n\r\n public startDragCommit(pointerArgs: IKb3dPointerArgs, draggable: IDraggable, onComplete: () => void) {\r\n if (!this._draggable) {\r\n this._connectorRenderer = this.viewer.rendererManager.connector;\r\n this._dragNode = pointerArgs.sourceNode!;\r\n this._dragNodeStartPosition = this._dragNode.position.clone();\r\n this._dragMesh = this.viewer.rendererManager.mesh.get3d(this._dragNode) as TransformNode;\r\n if (this._dragMesh.parent) {\r\n this._dragNodeInverseTransform = this._dragMesh.parent\r\n .computeWorldMatrix(true)\r\n .clone()\r\n .setTranslation(Vector3.Zero())\r\n .invert();\r\n }\r\n this._draggable = draggable;\r\n if (this._draggable.connectionMode == null) this._draggable.connectionMode = eConnectionMode.none;\r\n if (this._draggable.connectionType == null) this._draggable.connectionType = eMateType.fastened;\r\n if (this._draggable.connectRadius == null) {\r\n if (this._draggable.connectionMode == eConnectionMode.screen) {\r\n this._draggable.connectRadius = 5; //5 pixels\r\n } else {\r\n this._draggable.connectRadius = 0.3; //3d units\r\n }\r\n }\r\n if (this._draggable.dragOpacity == null) this._draggable.dragOpacity = 0.7;\r\n\r\n this._onComplete = onComplete;\r\n\r\n this.setupDragSurface(pointerArgs);\r\n\r\n if (this._draggable.dragStart) {\r\n this._draggable.dragStart(\r\n this.getDragArgs(pointerArgs, this._dragNodeStartPosition, this._dragMesh.rotationQuaternion)\r\n );\r\n }\r\n\r\n this.setupDragFriends();\r\n this.setupConnectors();\r\n this.setOpacity(this._draggable.dragOpacity!, this._dragMesh, ...this._dragFriendMeshes);\r\n }\r\n }\r\n\r\n protected dragging(p: PointerInfo) {\r\n // this._dragNode._dragging = true;\r\n\r\n const newConnection = this.findConnection();\r\n //let connectionChanged = !this.connectionsAreEqual(newConnection, this._activeConnection);\r\n this._activeConnection = newConnection;\r\n\r\n if (this._activeConnection) {\r\n //if (connectionChanged) { //don't bother processing the connection if it's the same\r\n this.showDropPreview(this._activeConnection);\r\n //}\r\n } else {\r\n this.hideDropPreview();\r\n }\r\n this.processDragPosition(p, this._draggable.dragging);\r\n }\r\n\r\n protected applyDragResultToNode(node: SpaceNode, dragArgs: IDragArgs, applyEuler: boolean) {\r\n node.position = dragArgs.newPos;\r\n // If euler rotation was changed, apply the changed euler rotation. If not, then apply the quaternion rotation regardless of whether it was changed.\r\n if (applyEuler) {\r\n node.eulerRotation = dragArgs.newRotationEuler;\r\n const newKbQuat = KbQuaternion.FromYawPitchRoll(dragArgs.newRotationEuler);\r\n dragArgs.newRotationAngle = newKbQuat.getAngle();\r\n dragArgs.newRotationAxis = newKbQuat.getAxis();\r\n node.rotationAxis = dragArgs.newRotationAxis;\r\n node.rotationAngle = dragArgs.newRotationAngle;\r\n } else {\r\n node.rotationAxis = dragArgs.newRotationAxis;\r\n node.rotationAngle = dragArgs.newRotationAngle;\r\n node.eulerRotation = KbQuaternion.FromRotationAxis(\r\n dragArgs.newRotationAxis,\r\n dragArgs.newRotationAngle\r\n ).toEuler();\r\n }\r\n }\r\n\r\n protected dropped(p: PointerInfo) {\r\n this._dragNode._primeMover = false;\r\n\r\n const complete = new Promise(resolve => {\r\n if (this._activeConnection) {\r\n this._activeConnection.otherConnector._parent!._primeMover = true;\r\n this.connect(this._activeConnection, this._dragMesh);\r\n // let newRotation = KbVector.FromVec3(this._dragMesh.rotation);\r\n\r\n const dragArgs: IDragArgs & Required> = {\r\n ...this.getDragArgs(p, this._dragMesh.position, this._dragMesh.rotationQuaternion),\r\n connection: this._activeConnection,\r\n };\r\n if (this._draggable.withMates) {\r\n dragArgs.mate = this.viewer.sceneNode._manager!.create(Token.MateNode, {\r\n connector1: this.viewer.sceneNode.getNestedPathTo(dragArgs.connection.otherConnector),\r\n connector2: this.viewer.sceneNode.getNestedPathTo(dragArgs.connection.dragConnector),\r\n mateType: this._draggable.connectionType,\r\n });\r\n }\r\n const previousEuler = dragArgs.newRotationEuler.clone();\r\n\r\n this.applyDragResultToNode(this._dragNode, dragArgs, !previousEuler.equal(dragArgs.newRotationEuler));\r\n if (dragArgs.mate) {\r\n this.viewer.sceneNode.mates.push(dragArgs.mate);\r\n }\r\n\r\n this.viewer.forceAndWaitForNextRender().then(() => {\r\n if (this._draggable.dropped) {\r\n this._draggable.dropped(dragArgs);\r\n\r\n if (!dragArgs.allow) {\r\n console.warn(\r\n 'Cannot disallow dropping when using connectors, use canConnect handler instead'\r\n );\r\n }\r\n }\r\n resolve();\r\n });\r\n } else {\r\n this.processDragPosition(p, this._draggable.dragging);\r\n this.processDragPosition(p, this._draggable.dropped);\r\n resolve();\r\n }\r\n });\r\n\r\n complete.then(() => {\r\n this.toggleAllConnectors(false);\r\n this.resetOpacity(this._dragMesh, ...this._dragFriendMeshes);\r\n this.scene.activeCamera?.attachControl(this.viewer.canvas!);\r\n if (this._onComplete) this._onComplete();\r\n\r\n this.disposeDropPreview();\r\n this._onComplete = undefined as any;\r\n this._dragNode = undefined as any;\r\n this._dragMesh = undefined as any;\r\n this._draggable = undefined as any;\r\n this._dragPlane = undefined as any;\r\n this._dragSurfaces = undefined as any;\r\n this._onComplete = undefined as any;\r\n if (this._externalConnectors) this._externalConnectors.clear();\r\n if (this._dragConnectors) this._dragConnectors.clear();\r\n if (this._dragConnectorsByDynamicId) this._dragConnectorsByDynamicId.clear();\r\n if (this._dropPreviewConnectors) this._dropPreviewConnectors.clear();\r\n this._oct = undefined as any;\r\n this._quadTree = undefined as any;\r\n if (this._quadTree) this._quadTree.items;\r\n this._dropPreview = undefined as any;\r\n this._activeConnection = undefined as any;\r\n this._connectorRenderer = undefined as any;\r\n if (this._dragFriends) this._dragFriends.clear();\r\n if (this._dragFriendMeshes) this._dragFriendMeshes.clear();\r\n if (this._dragFriendConnectors) this._dragFriendConnectors.clear();\r\n if (this._matedConnectors) this._matedConnectors.clear();\r\n });\r\n }\r\n\r\n protected processDragPosition(p: PointerInfo, event?: (a: IDragArgs) => void) {\r\n const newMouseIntersect = this.intersectPointerToDragSurface(p);\r\n if (newMouseIntersect) {\r\n let newPos: KbVector;\r\n if (this._draggable.disableOffset || this._draggable.mode == eDragMode.surface) {\r\n newPos = KbVector.FromVec3(newMouseIntersect);\r\n } else {\r\n if (this._startMouseIntersect) {\r\n const delta = newMouseIntersect.subtract(this._startMouseIntersect);\r\n if (this._dragNodeInverseTransform) {\r\n Vector3.TransformCoordinatesToRef(delta, this._dragNodeInverseTransform, delta);\r\n }\r\n newPos = this._dragNodeStartPosition.add(delta);\r\n } else {\r\n newPos = this._dragNodeStartPosition;\r\n this._startMouseIntersect = newMouseIntersect;\r\n }\r\n }\r\n\r\n const dragArgs = this.getDragArgs(p, newPos, this._dragMesh.rotationQuaternion);\r\n const previousEuler = dragArgs.newRotationEuler.clone();\r\n if (event) {\r\n event(dragArgs);\r\n }\r\n if (dragArgs.allow) {\r\n this.applyDragResultToNode(this._dragNode, dragArgs, !previousEuler.equal(dragArgs.newRotationEuler));\r\n }\r\n return dragArgs;\r\n }\r\n return undefined;\r\n }\r\n\r\n protected setupDragSurface(p: IKb3dPointerArgs) {\r\n const d = this._draggable;\r\n const manager = this._dragNode._manager!;\r\n if (d.mode == eDragMode.plane) {\r\n // d.planeOrigin = p.pickedPoint!; // d.planeOrigin || this._dragNode.position;\r\n\r\n /** intersection plane is defined by the pickedPoint and the plane normal */\r\n d.planeNormal = d.planeNormal || new KbVector(0, 1, 0);\r\n this._dragPlane = Plane.FromPositionAndNormal(p.pickInfo!.pickedPoint!, d.planeNormal!.toVec3());\r\n } else if (d.mode == eDragMode.free) {\r\n const cameraDirection = this.scene.activeCamera!.position.subtract(this._dragMesh.getAbsolutePosition());\r\n this._dragPlane = Plane.FromPositionAndNormal(this._dragNode.position.toVec3(), cameraDirection);\r\n } else if (d.mode == eDragMode.surface) {\r\n const surfaceIds = Array.isArray(d.surfaceNodes) ? d.surfaceNodes : [d.surfaceNodes];\r\n this._dragSurfaces =\r\n (surfaceIds.map(sn => sn && manager.getById(sn)?._dynamicId).filter(sn => sn != null) as string[]) ||\r\n [];\r\n }\r\n }\r\n\r\n protected intersectPointerToDragSurface(p: PointerInfo) {\r\n return this.intersectDragSurface(p.pickInfo!.ray!);\r\n }\r\n\r\n protected intersectDragSurface(ray: Ray): Vector3 | undefined {\r\n switch (this._draggable.mode) {\r\n case eDragMode.plane:\r\n case eDragMode.free: {\r\n const distance = ray.intersectsPlane(this._dragPlane);\r\n if (distance != null) {\r\n return ray.direction.scale(distance).add(ray.origin);\r\n }\r\n break;\r\n }\r\n case eDragMode.surface: {\r\n const surfacePick = this.scene.pickWithRay(ray, mesh => {\r\n return this._dragSurfaces.some(sn => mesh.id == sn);\r\n });\r\n if (surfacePick && surfacePick.pickedPoint) {\r\n return surfacePick.pickedPoint;\r\n }\r\n break;\r\n }\r\n }\r\n\r\n return undefined;\r\n }\r\n\r\n protected getDragArgs(\r\n p: PointerInfoMinimal,\r\n newPos: KbVector | Vector3,\r\n newRotation?: Nullable\r\n ): IDragArgs {\r\n const newKbRotation = KbQuaternion.FromQuat(newRotation || Quaternion.Identity());\r\n\r\n return {\r\n dragNode: this._dragNode,\r\n startPos: this._dragNodeStartPosition,\r\n allow: true,\r\n u: p.pickInfo?.bu || 0.5,\r\n v: p.pickInfo?.bv || 0.5,\r\n newPos: newPos instanceof Vector3 ? KbVector.FromVec3(newPos) : newPos,\r\n newRotationAxis: newKbRotation.getAxis(),\r\n newRotationAngle: newKbRotation.getAngle(),\r\n newRotationEuler: newKbRotation.toEuler(),\r\n };\r\n }\r\n\r\n protected findConnection(): IConnectionMesh | undefined {\r\n if (this._draggable.connectionMode != eConnectionMode.none) {\r\n const potentialConnectors = new Array();\r\n\r\n for (const [dragConnector, dragConnectorMesh] of this._dragConnectors) {\r\n const dragConPos = dragConnectorMesh.getAbsolutePosition();\r\n for (const p of this.intersectOctree(dragConPos)) {\r\n potentialConnectors.push({\r\n ...p,\r\n dragConnector,\r\n dragConnectorMesh,\r\n });\r\n }\r\n }\r\n\r\n if (this._draggable.connectionMode == eConnectionMode.world) {\r\n potentialConnectors.sortBy(p => p.proximity);\r\n } else {\r\n potentialConnectors.sortBy(p => p.distanceToCamera);\r\n }\r\n\r\n for (const p of potentialConnectors) {\r\n const otherConnector = getByDynamicId(p.otherMesh.id)!;\r\n const a = {\r\n dragConnector: p.dragConnector,\r\n otherConnector: otherConnector,\r\n dragNode: p.dragConnector._parent,\r\n otherNode: otherConnector._parent,\r\n allow: true,\r\n\r\n // Private properties below, don't expose to Snap\r\n dragConnectorMesh: p.dragConnectorMesh,\r\n otherConnectorMesh: p.otherMesh,\r\n };\r\n if (this._draggable.canConnect) {\r\n this._draggable.canConnect(a);\r\n }\r\n if (a.allow) {\r\n return a;\r\n }\r\n }\r\n }\r\n }\r\n\r\n protected connectionsAreEqual(c1?: IConnection, c2?: IConnection) {\r\n return (\r\n (!c1 && !c2) || (c1 && c2 && c1.dragConnector == c2.dragConnector && c1.otherConnector == c2.otherConnector)\r\n );\r\n }\r\n\r\n protected setupConnectors() {\r\n if (this._draggable.connectionMode != eConnectionMode.none) {\r\n const externalConnectorMeshes = new Set();\r\n for (const c of this.viewer.sceneNode.getAllConnectors()) {\r\n if (\r\n !this._dragFriendConnectors.has(c) &&\r\n (!this._draggable.withMates || !this._matedConnectors.has(c)) &&\r\n this.parentsEnabled(c)\r\n ) {\r\n let mesh = this.getConnectorMesh(c);\r\n if (!mesh) {\r\n this._connectorRenderer.update3d(c as ITrackedClass);\r\n mesh = this.getConnectorMesh(c);\r\n }\r\n externalConnectorMeshes.add(mesh);\r\n this._externalConnectors.set(c, mesh);\r\n }\r\n }\r\n const dragConnectors = this._dragNode\r\n .filterNodes(n => isSpaceNode(n) && n.visible, true, false)\r\n .selectMany(n => n.connectors);\r\n\r\n for (const c of dragConnectors) {\r\n //remove the drag connectors from the outside connectors\r\n if (this._externalConnectors.has(c)) {\r\n const mesh = this._externalConnectors.get(c)!;\r\n this._dragConnectors.set(c, mesh);\r\n this._dragConnectorsByDynamicId.set(c._dynamicId, c);\r\n this._externalConnectors.delete(c);\r\n externalConnectorMeshes.delete(mesh);\r\n }\r\n }\r\n\r\n this.toggleAllConnectors(true);\r\n\r\n if (this._draggable.connectionMode == eConnectionMode.world) {\r\n //create an octree for holding the connectors\r\n this._oct = new Octree(Octree.CreationFuncForMeshes, 64, 2);\r\n\r\n //get the bounding extents of all the external connectors to contain the octree volume\r\n const boundingVectors = this.scene.getWorldExtends(m => externalConnectorMeshes.has(m));\r\n\r\n //update the octree with the connector meshes\r\n this._oct.update(boundingVectors!.min, boundingVectors!.max, [...externalConnectorMeshes]);\r\n } else {\r\n //screen mode\r\n const extConnectorPositions = new Array<{ mesh: TransformNode; pos: IScreenPosition }>();\r\n externalConnectorMeshes.forEach(m => {\r\n extConnectorPositions.push({\r\n mesh: m,\r\n pos: toScreen(m.getAbsolutePosition(), this.scene.activeCamera!),\r\n });\r\n });\r\n\r\n const engine = this.scene.getEngine();\r\n this._quadTree = new QuadTree({\r\n items: extConnectorPositions,\r\n xSelector: d => d.pos.x,\r\n ySelector: d => d.pos.y,\r\n capacity: 4,\r\n xmin: 0,\r\n xmax: engine.getRenderWidth(),\r\n ymin: 0,\r\n ymax: engine.getRenderHeight(),\r\n });\r\n }\r\n }\r\n }\r\n\r\n protected parentsEnabled(item: Kb3dObject) {\r\n while (item._parent && item._parent instanceof SpaceNode) {\r\n if (!item._parent.visible) {\r\n return false;\r\n }\r\n item = item._parent;\r\n }\r\n return true;\r\n }\r\n\r\n protected setupDragFriends() {\r\n const activeMates = this.viewer.getActiveMates();\r\n for (const mate of activeMates) {\r\n const c1 = mate.getC1();\r\n if (c1 && isConnector(c1)) this._matedConnectors.add(c1);\r\n const c2 = mate.getC2();\r\n if (c2 && isConnector(c2)) this._matedConnectors.add(c2);\r\n }\r\n\r\n this._dragFriends = getNodeChain(this._dragNode as MatedObject, activeMates);\r\n this._dragFriends.forEach(friend => {\r\n if (!(friend instanceof SpaceNode)) {\r\n this._dragFriends.delete(friend);\r\n }\r\n });\r\n\r\n for (const friendNode of this._dragFriends) {\r\n if (friendNode instanceof SpaceNode) {\r\n friendNode._primeMover = false;\r\n\r\n const friendMesh = this.viewer.rendererManager.mesh.get3d(friendNode);\r\n if (friendMesh) {\r\n this._dragFriendMeshes.push(friendMesh);\r\n }\r\n\r\n const connectors = friendNode.filterByType(Token.Connector);\r\n for (const c of connectors) this._dragFriendConnectors.add(c);\r\n }\r\n }\r\n this._dragNode._primeMover = true;\r\n }\r\n\r\n protected getConnectorMesh(c: Connector) {\r\n return this._connectorRenderer.get3d(c)!;\r\n }\r\n\r\n protected toggleAllConnectors(show: boolean) {\r\n if (this._draggable.showConnectors) {\r\n for (const [c] of this._externalConnectors) {\r\n c._visible = show;\r\n }\r\n for (const [c] of this._dragConnectors) {\r\n c._visible = show;\r\n }\r\n }\r\n }\r\n\r\n protected intersectOctree(v: Vector3): IPotential[] {\r\n let potentials = new Array();\r\n if (this._draggable.connectionMode == eConnectionMode.world) {\r\n const bpotentials = this._oct.intersects(v, this._draggable.connectRadius!, false);\r\n bpotentials.forEach(p => {\r\n const proximity = p.getAbsolutePosition().subtract(v).length();\r\n if (proximity < this._draggable.connectRadius!) {\r\n potentials.push({ otherMesh: p, proximity });\r\n }\r\n });\r\n //potentials.sortBy(p => p.proximity);\r\n } else {\r\n //screen mode\r\n const screenPos = toScreen(v, this.scene.activeCamera!);\r\n\r\n potentials = this._quadTree.find(screenPos, this._draggable.connectRadius!).map(p => {\r\n return {\r\n otherMesh: p.item.mesh,\r\n proximity: p.proximity,\r\n distanceToCamera: Vector3.Distance(\r\n this.scene.activeCamera!.position,\r\n p.item.mesh.getAbsolutePosition()\r\n ),\r\n };\r\n });\r\n //potentials.sortBy(i => i.distanceToCamera);\r\n\r\n // let ray = this.scene.createPickingRay(screenPos.x, screenPos.y, Matrix.Identity(), this.scene.activeCamera);\r\n // let bpotentials = this._oct.intersectsRay(ray);\r\n // bpotentials.forEach(p => {\r\n // let potentialScreenPos = toScreen(p.getAbsolutePosition(), this.scene.activeCamera!);\r\n // let screenProximity = Math.sqrt(Math.pow(screenPos.x - potentialScreenPos.x, 2) + Math.pow(screenPos.y - potentialScreenPos.y, 2));\r\n // let realProximity = p.getAbsolutePosition().subtract(v).length();\r\n // console.log({ screenProximity, realProximity });\r\n // if (screenProximity < this._draggable.connectRadius!) {\r\n // potentials.push({ mesh: p, proximity: screenProximity });\r\n // }\r\n // });\r\n // potentials.sortBy(p => p.proximity);\r\n }\r\n return potentials;\r\n }\r\n\r\n protected showDropPreview(c: IConnectionMesh) {\r\n if (!this._dropPreview) {\r\n /**\r\n * for the drop preview, we get the drag node and any other nodes that are connected to it through mates\r\n * and clone them, put them in a transformNode, and then that transform node becomes the drop preview\r\n */\r\n this._dropPreview = new TransformNode('_droppreview', this.viewer.scene);\r\n this._dropPreview.position = Vector3.Zero();\r\n this._dropPreview.rotationQuaternion = Quaternion.Identity();\r\n this._dropPreview.computeWorldMatrix(true);\r\n const dragMeshTransform = this._dragMesh.computeWorldMatrix(true).getRotationMatrix();\r\n dragMeshTransform.invert();\r\n\r\n for (const friendMesh of [this._dragMesh, ...this._dragFriendMeshes]) {\r\n createDragPreview(friendMesh, this._dropPreview, this);\r\n }\r\n this._dropPreview.setPreTransformMatrix(dragMeshTransform);\r\n } else {\r\n //line drop preview back up with the source models as they might have rotated\r\n if (this._draggable.connectionType != eMateType.fastened) {\r\n this.connect(\r\n {\r\n dragConnector: c.dragConnector,\r\n otherConnector: c.dragConnector,\r\n otherConnectorMesh: c.dragConnectorMesh,\r\n dragConnectorMesh: this._dropPreviewConnectors.get(c.dragConnector)!, //pass in the drop preview connector mesh\r\n },\r\n this._dropPreview,\r\n eMateType.fastened,\r\n true\r\n );\r\n }\r\n }\r\n\r\n this.connect(\r\n {\r\n ...c,\r\n dragConnectorMesh: this._dropPreviewConnectors.get(c.dragConnector)!, //pass in the drop preview connector mesh\r\n },\r\n this._dropPreview\r\n );\r\n\r\n this._dropPreview.setEnabled(true);\r\n //hide the drag node\r\n this._dragMesh.setEnabled(false);\r\n for (const friend of this._dragFriendMeshes) friend.setEnabled(false);\r\n\r\n function createDragPreview(mesh: TransformNode, parent: TransformNode, self: Dragger) {\r\n const clonedMeshGroup = mesh.clone(mesh.id + '_dropPreview', mesh.parent, true);\r\n if (clonedMeshGroup) {\r\n clonedMeshGroup.setPivotMatrix(Matrix.Identity());\r\n\r\n // Cloning does not copy the pretransform matrix over so we have to do that manually\r\n // Otherwise the connector meshes scaling does not get applied to the preview meshes\r\n clonedMeshGroup.setEnabled(mesh.isEnabled());\r\n if (c instanceof Mesh && mesh instanceof Mesh) {\r\n c.setPreTransformMatrix(mesh.getPivotMatrix());\r\n }\r\n\r\n clonedMeshGroup.setParent(parent); // setting the parent with setParent will make the node stay in the same absolute position.\r\n const children = mesh.getChildren(child => child instanceof TransformNode, true);\r\n children.forEach(child => {\r\n createDragPreview(child as TransformNode, clonedMeshGroup, self);\r\n });\r\n\r\n clonedMeshGroup.metadata = {}; // babylon metadata refers to same instance of object after cloning so we clear it out\r\n const kbnode = getByDynamicId(mesh.id);\r\n if (kbnode && kbnode instanceof Connector) {\r\n const connector = kbnode as Connector;\r\n self._dropPreviewConnectors.set(connector, clonedMeshGroup as Mesh);\r\n clonedMeshGroup.setEnabled(connector._visible); //connector.enabled && !!this._draggable.showConnectors);\r\n } else {\r\n self.setOpacity(self._draggable.dragOpacity!, clonedMeshGroup);\r\n }\r\n }\r\n }\r\n }\r\n\r\n protected connect(\r\n c: IConnectionMesh,\r\n followerMesh: TransformNode,\r\n mateType = this._draggable.connectionType,\r\n flip = false\r\n ) {\r\n this.viewer.pauseRenderWhile(() => {\r\n mateConnectors(\r\n {\r\n mateType: mateType!,\r\n flip: flip,\r\n offsetX: 0,\r\n offsetY: 0,\r\n offsetZ: 0,\r\n },\r\n followerMesh,\r\n c.otherConnectorMesh,\r\n c.dragConnectorMesh\r\n );\r\n });\r\n }\r\n\r\n protected hideDropPreview() {\r\n if (this._dropPreview) this._dropPreview.setEnabled(false);\r\n this._dragMesh.setEnabled(true); //show the drag node again\r\n if (this._dragFriendMeshes) {\r\n for (const friend of this._dragFriendMeshes) {\r\n friend.setEnabled(getByDynamicId(friend.id)!.visible);\r\n }\r\n }\r\n }\r\n\r\n protected disposeDropPreview() {\r\n this.hideDropPreview();\r\n if (this._dropPreview) this._dropPreview.dispose();\r\n if (this._dragFriendMeshes) this._dragFriendMeshes.clear();\r\n if (this._dragFriends) this._dragFriends.clear();\r\n }\r\n\r\n protected getConnectorLocalPoint(connector: Connector) {\r\n if (connector.positionMode == eConnectorPositionMode.absolute) {\r\n return connector.position;\r\n } else {\r\n //TODO\r\n }\r\n }\r\n\r\n protected setOpacity(value: number, ...nodes: TransformNode[]) {\r\n if (this.shouldSetOpacity()) {\r\n for (const node of nodes) {\r\n for (const m of this.getAllMeshes(node, m => !(m instanceof InstancedMesh))) {\r\n //set opacity of all non-connector meshes being dragged\r\n if (!m.metadata) {\r\n m.metadata = {};\r\n }\r\n m.metadata.originalVisibility = m.visibility;\r\n m.visibility = value;\r\n }\r\n }\r\n }\r\n }\r\n\r\n protected resetOpacity(...nodes: TransformNode[]) {\r\n for (const node of nodes) {\r\n for (const m of this.getAllMeshes(node, m => !(m instanceof InstancedMesh))) {\r\n //set opacity of all non-connector meshes being dragged\r\n if (m.metadata && m.metadata.originalVisibility !== undefined) {\r\n m.visibility = m.metadata.originalVisibility;\r\n delete m.metadata.originalVisibility;\r\n }\r\n }\r\n }\r\n }\r\n\r\n protected shouldSetOpacity() {\r\n return this._draggable.connectionMode != eConnectionMode.none && this._draggable.showConnectors; // && this._dragConnectors.size ;\r\n }\r\n\r\n /**get's all child meshes and includes the given node if it's a mesh */\r\n protected getAllMeshes(node: TransformNode, predicate?: (node: Node) => boolean) {\r\n const meshes = new Set(node.getChildMeshes(false, predicate));\r\n if (node instanceof AbstractMesh) meshes.add(node);\r\n return meshes;\r\n }\r\n}\r\n","import { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { OBJExport } from '@babylonjs/serializers/OBJ';\r\nimport { getByDynamicId } from '@models/classes/kb3d-manager';\r\nimport { MeshNode } from '@models/classes/meshes/mesh-node';\r\nimport { RendererManager } from '@view/render/renderer-manager';\r\n\r\nexport class ObjKb3dExporter {\r\n constructor(private scene: Scene, private rendererManager: RendererManager) {}\r\n\r\n /** generates the current scene into a glb file and returns an object url for it */\r\n public generateObj(meshNode?: MeshNode): string | null {\r\n let meshes: Mesh[];\r\n if (!meshNode) {\r\n meshes = this.scene.meshes.filter(this.shouldExportMesh);\r\n } else {\r\n const mesh = this.rendererManager.mesh.get3d(meshNode);\r\n if (mesh && mesh instanceof Mesh) {\r\n meshes = [mesh];\r\n } else {\r\n return null;\r\n }\r\n }\r\n return OBJExport.OBJ(meshes, false, undefined, meshes.length > 1);\r\n }\r\n\r\n public exportObj(filename = 'scene', meshNode?: MeshNode) {\r\n const objData = this.generateObj(meshNode);\r\n if (objData) {\r\n const blob = this.createObjBlob(objData);\r\n const link = document.createElement('a');\r\n document.body.appendChild(link);\r\n link.setAttribute('type', 'hidden');\r\n link.download = `${filename}.obj`;\r\n link.href = window.URL.createObjectURL(blob);\r\n link.click();\r\n setTimeout(() => {\r\n window.URL.revokeObjectURL(link.href);\r\n link.remove();\r\n });\r\n }\r\n }\r\n\r\n protected shouldExportMesh(mesh: AbstractMesh): mesh is Mesh {\r\n return mesh instanceof Mesh && mesh.isVisible && mesh.isEnabled() && !!getByDynamicId(mesh.id);\r\n }\r\n\r\n protected createObjBlob(objData: string) {\r\n const mimeType = { type: 'text/plain' };\r\n return new Blob([objData], mimeType);\r\n }\r\n}\r\n","import type { Nullable } from \"core/types\";\r\nimport { Matrix } from \"core/Maths/math.vector\";\r\nimport { Tools } from \"core/Misc/tools\";\r\nimport type { StandardMaterial } from \"core/Materials/standardMaterial\";\r\nimport type { Geometry } from \"core/Meshes/geometry\";\r\nimport type { Mesh } from \"core/Meshes/mesh\";\r\n\r\n/**\r\n * Class for generating OBJ data from a Babylon scene.\r\n */\r\nexport class OBJExport {\r\n /**\r\n * Exports the geometry of a Mesh array in .OBJ file format (text)\r\n * @param mesh defines the list of meshes to serialize\r\n * @param materials defines if materials should be exported\r\n * @param matlibname defines the name of the associated mtl file\r\n * @param globalposition defines if the exported positions are globals or local to the exported mesh\r\n * @returns the OBJ content\r\n */\r\n public static OBJ(mesh: Mesh[], materials?: boolean, matlibname?: string, globalposition?: boolean): string {\r\n const output: string[] = [];\r\n let v = 1;\r\n // keep track of uv index in case mixed meshes are passed in\r\n let textureV = 1;\r\n\r\n if (materials) {\r\n if (!matlibname) {\r\n matlibname = \"mat\";\r\n }\r\n output.push(\"mtllib \" + matlibname + \".mtl\");\r\n }\r\n for (let j = 0; j < mesh.length; j++) {\r\n output.push(\"g object\" + j);\r\n output.push(\"o object_\" + j);\r\n\r\n //Uses the position of the item in the scene, to the file (this back to normal in the end)\r\n let inverseTransform: Nullable = null;\r\n if (globalposition) {\r\n const transform = mesh[j].computeWorldMatrix(true);\r\n inverseTransform = new Matrix();\r\n transform.invertToRef(inverseTransform);\r\n\r\n mesh[j].bakeTransformIntoVertices(transform);\r\n }\r\n\r\n //TODO: submeshes (groups)\r\n //TODO: smoothing groups (s 1, s off);\r\n if (materials) {\r\n const mat = mesh[j].material;\r\n\r\n if (mat) {\r\n output.push(\"usemtl \" + mat.id);\r\n }\r\n }\r\n const g: Nullable = mesh[j].geometry;\r\n\r\n if (!g) {\r\n Tools.Warn(\"No geometry is present on the mesh\");\r\n continue;\r\n }\r\n\r\n const trunkVerts = g.getVerticesData(\"position\");\r\n const trunkNormals = g.getVerticesData(\"normal\");\r\n const trunkUV = g.getVerticesData(\"uv\");\r\n const trunkFaces = g.getIndices();\r\n let currentV = 0;\r\n let currentTextureV = 0;\r\n\r\n if (!trunkVerts || !trunkFaces) {\r\n Tools.Warn(\"There are no position vertices or indices on the mesh!\");\r\n continue;\r\n }\r\n\r\n for (let i = 0; i < trunkVerts.length; i += 3) {\r\n // Babylon.js default is left handed, while OBJ default is right handed\r\n // Need to invert Z vertices unless Babylon is set to use a right handed system\r\n if (mesh[0].getScene().useRightHandedSystem) {\r\n output.push(\"v \" + trunkVerts[i] + \" \" + trunkVerts[i + 1] + \" \" + trunkVerts[i + 2]);\r\n } else {\r\n output.push(\"v \" + trunkVerts[i] + \" \" + trunkVerts[i + 1] + \" \" + -trunkVerts[i + 2]);\r\n }\r\n currentV++;\r\n }\r\n\r\n if (trunkNormals != null) {\r\n for (let i = 0; i < trunkNormals.length; i += 3) {\r\n output.push(\"vn \" + trunkNormals[i] + \" \" + trunkNormals[i + 1] + \" \" + trunkNormals[i + 2]);\r\n }\r\n }\r\n if (trunkUV != null) {\r\n for (let i = 0; i < trunkUV.length; i += 2) {\r\n output.push(\"vt \" + trunkUV[i] + \" \" + trunkUV[i + 1]);\r\n currentTextureV++;\r\n }\r\n }\r\n\r\n for (let i = 0; i < trunkFaces.length; i += 3) {\r\n const indices = [String(trunkFaces[i + 2] + v), String(trunkFaces[i + 1] + v), String(trunkFaces[i] + v)];\r\n const textureIndices = [String(trunkFaces[i + 2] + textureV), String(trunkFaces[i + 1] + textureV), String(trunkFaces[i] + textureV)];\r\n const blanks: string[] = [\"\", \"\", \"\"];\r\n\r\n const facePositions = indices;\r\n const faceUVs = trunkUV != null ? textureIndices : blanks;\r\n const faceNormals = trunkNormals != null ? indices : blanks;\r\n\r\n output.push(\r\n \"f \" +\r\n facePositions[0] +\r\n \"/\" +\r\n faceUVs[0] +\r\n \"/\" +\r\n faceNormals[0] +\r\n \" \" +\r\n facePositions[1] +\r\n \"/\" +\r\n faceUVs[1] +\r\n \"/\" +\r\n faceNormals[1] +\r\n \" \" +\r\n facePositions[2] +\r\n \"/\" +\r\n faceUVs[2] +\r\n \"/\" +\r\n faceNormals[2]\r\n );\r\n }\r\n //back de previous matrix, to not change the original mesh in the scene\r\n if (globalposition && inverseTransform) {\r\n mesh[j].bakeTransformIntoVertices(inverseTransform);\r\n }\r\n v += currentV;\r\n textureV += currentTextureV;\r\n }\r\n const text: string = output.join(\"\\n\");\r\n return text;\r\n }\r\n\r\n /**\r\n * Exports the material(s) of a mesh in .MTL file format (text)\r\n * @param mesh defines the mesh to extract the material from\r\n * @returns the mtl content\r\n */\r\n //TODO: Export the materials of mesh array\r\n public static MTL(mesh: Mesh): string {\r\n const output = [];\r\n const m = mesh.material;\r\n output.push(\"newmtl mat1\");\r\n output.push(\" Ns \" + m.specularPower.toFixed(4));\r\n output.push(\" Ni 1.5000\");\r\n output.push(\" d \" + m.alpha.toFixed(4));\r\n output.push(\" Tr 0.0000\");\r\n output.push(\" Tf 1.0000 1.0000 1.0000\");\r\n output.push(\" illum 2\");\r\n output.push(\" Ka \" + m.ambientColor.r.toFixed(4) + \" \" + m.ambientColor.g.toFixed(4) + \" \" + m.ambientColor.b.toFixed(4));\r\n output.push(\" Kd \" + m.diffuseColor.r.toFixed(4) + \" \" + m.diffuseColor.g.toFixed(4) + \" \" + m.diffuseColor.b.toFixed(4));\r\n output.push(\" Ks \" + m.specularColor.r.toFixed(4) + \" \" + m.specularColor.g.toFixed(4) + \" \" + m.specularColor.b.toFixed(4));\r\n output.push(\" Ke \" + m.emissiveColor.r.toFixed(4) + \" \" + m.emissiveColor.g.toFixed(4) + \" \" + m.emissiveColor.b.toFixed(4));\r\n\r\n //TODO: uv scale, offset, wrap\r\n //TODO: UV mirrored in Blender? second UV channel? lightMap? reflection textures?\r\n const uvscale = \"\";\r\n\r\n if (m.ambientTexture) {\r\n output.push(\" map_Ka \" + uvscale + m.ambientTexture.name);\r\n }\r\n\r\n if (m.diffuseTexture) {\r\n output.push(\" map_Kd \" + uvscale + m.diffuseTexture.name);\r\n //TODO: alpha testing, opacity in diffuse texture alpha channel (diffuseTexture.hasAlpha -> map_d)\r\n }\r\n\r\n if (m.specularTexture) {\r\n output.push(\" map_Ks \" + uvscale + m.specularTexture.name);\r\n /* TODO: glossiness = specular highlight component is in alpha channel of specularTexture. (???)\r\n if (m.useGlossinessFromSpecularMapAlpha) {\r\n output.push(\" map_Ns \"+uvscale + m.specularTexture.name);\r\n }\r\n */\r\n }\r\n\r\n /* TODO: emissive texture not in .MAT format (???)\r\n if (m.emissiveTexture) {\r\n output.push(\" map_d \"+uvscale+m.emissiveTexture.name);\r\n }\r\n */\r\n\r\n if (m.bumpTexture) {\r\n output.push(\" map_bump -imfchan z \" + uvscale + m.bumpTexture.name);\r\n }\r\n\r\n if (m.opacityTexture) {\r\n output.push(\" map_d \" + uvscale + m.opacityTexture.name);\r\n }\r\n\r\n const text = output.join(\"\\n\");\r\n return text;\r\n }\r\n}\r\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"boundingBoxRendererFragmentDeclaration\";\nconst shader = `uniform vec4 color;\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const boundingBoxRendererFragmentDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"boundingBoxRendererUboDeclaration\";\nconst shader = `#ifdef WEBGL2\nuniform vec4 color;uniform mat4 world;uniform mat4 viewProjection;\n#ifdef MULTIVIEW\nuniform mat4 viewProjectionR;\n#endif\n#else\nlayout(std140,column_major) uniform;uniform BoundingBoxRenderer {vec4 color;mat4 world;mat4 viewProjection;mat4 viewProjectionR;};\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const boundingBoxRendererUboDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/boundingBoxRendererFragmentDeclaration\";\nimport \"./ShadersInclude/boundingBoxRendererUboDeclaration\";\n\nconst name = \"boundingBoxRendererPixelShader\";\nconst shader = `#include<__decl__boundingBoxRendererFragment>\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_FRAGMENT_MAIN_BEGIN\ngl_FragColor=color;\n#define CUSTOM_FRAGMENT_MAIN_END\n}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const boundingBoxRendererPixelShader = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../../Engines/shaderStore\";\n\nconst name = \"boundingBoxRendererVertexDeclaration\";\nconst shader = `uniform mat4 world;uniform mat4 viewProjection;\n#ifdef MULTIVIEW\nuniform mat4 viewProjectionR;\n#endif\n`;\n// Sideeffect\nShaderStore.IncludesShadersStore[name] = shader;\n/** @internal */\nexport const boundingBoxRendererVertexDeclaration = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/boundingBoxRendererVertexDeclaration\";\nimport \"./ShadersInclude/boundingBoxRendererUboDeclaration\";\n\nconst name = \"boundingBoxRendererVertexShader\";\nconst shader = `attribute vec3 position;\n#include<__decl__boundingBoxRendererVertex>\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\nvec4 worldPos=world*vec4(position,1.0);\n#ifdef MULTIVIEW\nif (gl_ViewID_OVR==0u) {gl_Position=viewProjection*worldPos;} else {gl_Position=viewProjectionR*worldPos;}\n#else\ngl_Position=viewProjection*worldPos;\n#endif\n#define CUSTOM_VERTEX_MAIN_END\n}\n`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const boundingBoxRendererVertexShader = { name, shader };\n","import { Scene } from \"../scene\";\r\nimport { VertexBuffer } from \"../Buffers/buffer\";\r\nimport type { SubMesh } from \"../Meshes/subMesh\";\r\nimport { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport { Matrix } from \"../Maths/math.vector\";\r\nimport { SmartArray } from \"../Misc/smartArray\";\r\nimport type { Nullable, FloatArray, IndicesArray } from \"../types\";\r\nimport type { ISceneComponent } from \"../sceneComponent\";\r\nimport { SceneComponentConstants } from \"../sceneComponent\";\r\nimport type { BoundingBox } from \"../Culling/boundingBox\";\r\nimport type { Effect } from \"../Materials/effect\";\r\nimport { Material } from \"../Materials/material\";\r\nimport { ShaderMaterial } from \"../Materials/shaderMaterial\";\r\nimport type { DataBuffer } from \"../Buffers/dataBuffer\";\r\nimport { Color3 } from \"../Maths/math.color\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport { DrawWrapper } from \"../Materials/drawWrapper\";\r\nimport { UniformBuffer } from \"../Materials/uniformBuffer\";\r\nimport { CreateBoxVertexData } from \"../Meshes/Builders/boxBuilder\";\r\n\r\nimport \"../Shaders/boundingBoxRenderer.fragment\";\r\nimport \"../Shaders/boundingBoxRenderer.vertex\";\r\n\r\ndeclare module \"../scene\" {\r\n export interface Scene {\r\n /** @internal (Backing field) */\r\n _boundingBoxRenderer: BoundingBoxRenderer;\r\n\r\n /** @internal (Backing field) */\r\n _forceShowBoundingBoxes: boolean;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if all bounding boxes must be rendered\r\n */\r\n forceShowBoundingBoxes: boolean;\r\n\r\n /**\r\n * Gets the bounding box renderer associated with the scene\r\n * @returns a BoundingBoxRenderer\r\n */\r\n getBoundingBoxRenderer(): BoundingBoxRenderer;\r\n }\r\n}\r\n\r\nObject.defineProperty(Scene.prototype, \"forceShowBoundingBoxes\", {\r\n get: function (this: Scene) {\r\n return this._forceShowBoundingBoxes || false;\r\n },\r\n set: function (this: Scene, value: boolean) {\r\n this._forceShowBoundingBoxes = value;\r\n // Lazyly creates a BB renderer if needed.\r\n if (value) {\r\n this.getBoundingBoxRenderer();\r\n }\r\n },\r\n enumerable: true,\r\n configurable: true,\r\n});\r\n\r\nScene.prototype.getBoundingBoxRenderer = function (): BoundingBoxRenderer {\r\n if (!this._boundingBoxRenderer) {\r\n this._boundingBoxRenderer = new BoundingBoxRenderer(this);\r\n }\r\n\r\n return this._boundingBoxRenderer;\r\n};\r\n\r\ndeclare module \"../Meshes/abstractMesh\" {\r\n export interface AbstractMesh {\r\n /** @internal (Backing field) */\r\n _showBoundingBox: boolean;\r\n\r\n /**\r\n * Gets or sets a boolean indicating if the bounding box must be rendered as well (false by default)\r\n */\r\n showBoundingBox: boolean;\r\n }\r\n}\r\n\r\nObject.defineProperty(AbstractMesh.prototype, \"showBoundingBox\", {\r\n get: function (this: AbstractMesh) {\r\n return this._showBoundingBox || false;\r\n },\r\n set: function (this: AbstractMesh, value: boolean) {\r\n this._showBoundingBox = value;\r\n // Lazyly creates a BB renderer if needed.\r\n if (value) {\r\n this.getScene().getBoundingBoxRenderer();\r\n }\r\n },\r\n enumerable: true,\r\n configurable: true,\r\n});\r\n\r\n/**\r\n * Component responsible of rendering the bounding box of the meshes in a scene.\r\n * This is usually used through the mesh.showBoundingBox or the scene.forceShowBoundingBoxes properties\r\n */\r\nexport class BoundingBoxRenderer implements ISceneComponent {\r\n /**\r\n * The component name helpful to identify the component in the list of scene components.\r\n */\r\n public readonly name = SceneComponentConstants.NAME_BOUNDINGBOXRENDERER;\r\n\r\n /**\r\n * The scene the component belongs to.\r\n */\r\n public scene: Scene;\r\n\r\n /**\r\n * Color of the bounding box lines placed in front of an object\r\n */\r\n public frontColor = new Color3(1, 1, 1);\r\n /**\r\n * Color of the bounding box lines placed behind an object\r\n */\r\n public backColor = new Color3(0.1, 0.1, 0.1);\r\n /**\r\n * Defines if the renderer should show the back lines or not\r\n */\r\n public showBackLines = true;\r\n\r\n /**\r\n * Observable raised before rendering a bounding box\r\n */\r\n public onBeforeBoxRenderingObservable = new Observable();\r\n\r\n /**\r\n * Observable raised after rendering a bounding box\r\n */\r\n public onAfterBoxRenderingObservable = new Observable();\r\n\r\n /**\r\n * Observable raised after resources are created\r\n */\r\n public onResourcesReadyObservable = new Observable();\r\n\r\n /**\r\n * When false, no bounding boxes will be rendered\r\n */\r\n public enabled = true;\r\n\r\n /**\r\n * @internal\r\n */\r\n public renderList = new SmartArray(32);\r\n\r\n private _colorShader: ShaderMaterial;\r\n private _colorShaderForOcclusionQuery: ShaderMaterial;\r\n private _vertexBuffers: { [key: string]: Nullable } = {};\r\n private _indexBuffer: DataBuffer;\r\n private _fillIndexBuffer: Nullable = null;\r\n private _fillIndexData: Nullable = null;\r\n private _uniformBufferFront: UniformBuffer;\r\n private _uniformBufferBack: UniformBuffer;\r\n private _renderPassIdForOcclusionQuery: number;\r\n\r\n /**\r\n * Instantiates a new bounding box renderer in a scene.\r\n * @param scene the scene the renderer renders in\r\n */\r\n constructor(scene: Scene) {\r\n this.scene = scene;\r\n scene._addComponent(this);\r\n this._uniformBufferFront = new UniformBuffer(this.scene.getEngine(), undefined, undefined, \"BoundingBoxRendererFront\", !this.scene.getEngine().isWebGPU);\r\n this._buildUniformLayout(this._uniformBufferFront);\r\n this._uniformBufferBack = new UniformBuffer(this.scene.getEngine(), undefined, undefined, \"BoundingBoxRendererBack\", !this.scene.getEngine().isWebGPU);\r\n this._buildUniformLayout(this._uniformBufferBack);\r\n }\r\n\r\n private _buildUniformLayout(ubo: UniformBuffer): void {\r\n ubo.addUniform(\"color\", 4);\r\n ubo.addUniform(\"world\", 16);\r\n ubo.addUniform(\"viewProjection\", 16);\r\n ubo.addUniform(\"viewProjectionR\", 16);\r\n ubo.create();\r\n }\r\n\r\n /**\r\n * Registers the component in a given scene\r\n */\r\n public register(): void {\r\n this.scene._beforeEvaluateActiveMeshStage.registerStep(SceneComponentConstants.STEP_BEFOREEVALUATEACTIVEMESH_BOUNDINGBOXRENDERER, this, this.reset);\r\n\r\n this.scene._preActiveMeshStage.registerStep(SceneComponentConstants.STEP_PREACTIVEMESH_BOUNDINGBOXRENDERER, this, this._preActiveMesh);\r\n\r\n this.scene._evaluateSubMeshStage.registerStep(SceneComponentConstants.STEP_EVALUATESUBMESH_BOUNDINGBOXRENDERER, this, this._evaluateSubMesh);\r\n\r\n this.scene._afterRenderingGroupDrawStage.registerStep(SceneComponentConstants.STEP_AFTERRENDERINGGROUPDRAW_BOUNDINGBOXRENDERER, this, this.render);\r\n }\r\n\r\n private _evaluateSubMesh(mesh: AbstractMesh, subMesh: SubMesh): void {\r\n if (mesh.showSubMeshesBoundingBox) {\r\n const boundingInfo = subMesh.getBoundingInfo();\r\n if (boundingInfo !== null && boundingInfo !== undefined) {\r\n boundingInfo.boundingBox._tag = mesh.renderingGroupId;\r\n this.renderList.push(boundingInfo.boundingBox);\r\n }\r\n }\r\n }\r\n\r\n private _preActiveMesh(mesh: AbstractMesh): void {\r\n if (mesh.showBoundingBox || this.scene.forceShowBoundingBoxes) {\r\n const boundingInfo = mesh.getBoundingInfo();\r\n boundingInfo.boundingBox._tag = mesh.renderingGroupId;\r\n this.renderList.push(boundingInfo.boundingBox);\r\n }\r\n }\r\n\r\n private _prepareResources(): void {\r\n if (this._colorShader) {\r\n return;\r\n }\r\n\r\n this._colorShader = new ShaderMaterial(\r\n \"colorShader\",\r\n this.scene,\r\n \"boundingBoxRenderer\",\r\n {\r\n attributes: [VertexBuffer.PositionKind],\r\n uniforms: [\"world\", \"viewProjection\", \"viewProjectionR\", \"color\"],\r\n uniformBuffers: [\"BoundingBoxRenderer\"],\r\n },\r\n false\r\n );\r\n this._colorShader.doNotSerialize = true;\r\n\r\n this._colorShader.reservedDataStore = {\r\n hidden: true,\r\n };\r\n\r\n this._colorShaderForOcclusionQuery = new ShaderMaterial(\r\n \"colorShaderOccQuery\",\r\n this.scene,\r\n \"boundingBoxRenderer\",\r\n {\r\n attributes: [VertexBuffer.PositionKind],\r\n uniforms: [\"world\", \"viewProjection\", \"viewProjectionR\", \"color\"],\r\n uniformBuffers: [\"BoundingBoxRenderer\"],\r\n },\r\n true\r\n );\r\n this._colorShaderForOcclusionQuery.doNotSerialize = true;\r\n\r\n this._colorShaderForOcclusionQuery.reservedDataStore = {\r\n hidden: true,\r\n };\r\n\r\n const engine = this.scene.getEngine();\r\n const boxdata = CreateBoxVertexData({ size: 1.0 });\r\n this._vertexBuffers[VertexBuffer.PositionKind] = new VertexBuffer(engine, boxdata.positions, VertexBuffer.PositionKind, false);\r\n this._createIndexBuffer();\r\n this._fillIndexData = boxdata.indices;\r\n this.onResourcesReadyObservable.notifyObservers(this);\r\n }\r\n\r\n private _createIndexBuffer(): void {\r\n const engine = this.scene.getEngine();\r\n this._indexBuffer = engine.createIndexBuffer([0, 1, 1, 2, 2, 3, 3, 0, 4, 5, 5, 6, 6, 7, 7, 4, 0, 7, 1, 6, 2, 5, 3, 4]);\r\n }\r\n\r\n /**\r\n * Rebuilds the elements related to this component in case of\r\n * context lost for instance.\r\n */\r\n public rebuild(): void {\r\n const vb = this._vertexBuffers[VertexBuffer.PositionKind];\r\n if (vb) {\r\n vb._rebuild();\r\n }\r\n this._createIndexBuffer();\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public reset(): void {\r\n this.renderList.reset();\r\n }\r\n\r\n /**\r\n * Render the bounding boxes of a specific rendering group\r\n * @param renderingGroupId defines the rendering group to render\r\n */\r\n public render(renderingGroupId: number): void {\r\n if (this.renderList.length === 0 || !this.enabled) {\r\n return;\r\n }\r\n\r\n this._prepareResources();\r\n\r\n if (!this._colorShader.isReady()) {\r\n return;\r\n }\r\n\r\n const engine = this.scene.getEngine();\r\n engine.setDepthWrite(false);\r\n\r\n const transformMatrix = this.scene.getTransformMatrix();\r\n\r\n for (let boundingBoxIndex = 0; boundingBoxIndex < this.renderList.length; boundingBoxIndex++) {\r\n const boundingBox = this.renderList.data[boundingBoxIndex];\r\n if (boundingBox._tag !== renderingGroupId) {\r\n continue;\r\n }\r\n\r\n this._createWrappersForBoundingBox(boundingBox);\r\n this.onBeforeBoxRenderingObservable.notifyObservers(boundingBox);\r\n\r\n const min = boundingBox.minimum;\r\n const max = boundingBox.maximum;\r\n const diff = max.subtract(min);\r\n const median = min.add(diff.scale(0.5));\r\n\r\n const worldMatrix = Matrix.Scaling(diff.x, diff.y, diff.z).multiply(Matrix.Translation(median.x, median.y, median.z)).multiply(boundingBox.getWorldMatrix());\r\n\r\n const useReverseDepthBuffer = engine.useReverseDepthBuffer;\r\n\r\n if (this.showBackLines) {\r\n const drawWrapperBack = boundingBox._drawWrapperBack ?? this._colorShader._getDrawWrapper();\r\n\r\n this._colorShader._preBind(drawWrapperBack);\r\n\r\n engine.bindBuffers(this._vertexBuffers, this._indexBuffer, this._colorShader.getEffect());\r\n\r\n // Back\r\n if (useReverseDepthBuffer) {\r\n engine.setDepthFunctionToLessOrEqual();\r\n } else {\r\n engine.setDepthFunctionToGreaterOrEqual();\r\n }\r\n this._uniformBufferBack.bindToEffect(drawWrapperBack.effect!, \"BoundingBoxRenderer\");\r\n this._uniformBufferBack.updateColor4(\"color\", this.backColor, 1);\r\n this._uniformBufferBack.updateMatrix(\"world\", worldMatrix);\r\n this._uniformBufferBack.updateMatrix(\"viewProjection\", transformMatrix);\r\n this._uniformBufferBack.update();\r\n\r\n // Draw order\r\n engine.drawElementsType(Material.LineListDrawMode, 0, 24);\r\n }\r\n\r\n const drawWrapperFront = boundingBox._drawWrapperFront ?? this._colorShader._getDrawWrapper();\r\n\r\n this._colorShader._preBind(drawWrapperFront);\r\n\r\n engine.bindBuffers(this._vertexBuffers, this._indexBuffer, this._colorShader.getEffect());\r\n\r\n // Front\r\n if (useReverseDepthBuffer) {\r\n engine.setDepthFunctionToGreater();\r\n } else {\r\n engine.setDepthFunctionToLess();\r\n }\r\n this._uniformBufferFront.bindToEffect(drawWrapperFront.effect!, \"BoundingBoxRenderer\");\r\n this._uniformBufferFront.updateColor4(\"color\", this.frontColor, 1);\r\n this._uniformBufferFront.updateMatrix(\"world\", worldMatrix);\r\n this._uniformBufferFront.updateMatrix(\"viewProjection\", transformMatrix);\r\n this._uniformBufferFront.update();\r\n\r\n // Draw order\r\n engine.drawElementsType(Material.LineListDrawMode, 0, 24);\r\n\r\n this.onAfterBoxRenderingObservable.notifyObservers(boundingBox);\r\n }\r\n this._colorShader.unbind();\r\n engine.setDepthFunctionToLessOrEqual();\r\n engine.setDepthWrite(true);\r\n }\r\n\r\n private _createWrappersForBoundingBox(boundingBox: BoundingBox): void {\r\n if (!boundingBox._drawWrapperFront) {\r\n const engine = this.scene.getEngine();\r\n\r\n boundingBox._drawWrapperFront = new DrawWrapper(engine);\r\n boundingBox._drawWrapperBack = new DrawWrapper(engine);\r\n\r\n boundingBox._drawWrapperFront.setEffect(this._colorShader.getEffect());\r\n boundingBox._drawWrapperBack.setEffect(this._colorShader.getEffect());\r\n }\r\n }\r\n\r\n /**\r\n * In case of occlusion queries, we can render the occlusion bounding box through this method\r\n * @param mesh Define the mesh to render the occlusion bounding box for\r\n */\r\n public renderOcclusionBoundingBox(mesh: AbstractMesh): void {\r\n const engine = this.scene.getEngine();\r\n\r\n if (this._renderPassIdForOcclusionQuery === undefined) {\r\n this._renderPassIdForOcclusionQuery = engine.createRenderPassId(`Render pass for occlusion query`);\r\n }\r\n\r\n const currentRenderPassId = engine.currentRenderPassId;\r\n\r\n engine.currentRenderPassId = this._renderPassIdForOcclusionQuery;\r\n\r\n this._prepareResources();\r\n\r\n const subMesh = mesh.subMeshes[0];\r\n\r\n if (!this._colorShaderForOcclusionQuery.isReady(mesh, undefined, subMesh) || !mesh.hasBoundingInfo) {\r\n engine.currentRenderPassId = currentRenderPassId;\r\n return;\r\n }\r\n\r\n if (!this._fillIndexBuffer) {\r\n this._fillIndexBuffer = engine.createIndexBuffer(this._fillIndexData!);\r\n }\r\n\r\n const useReverseDepthBuffer = engine.useReverseDepthBuffer;\r\n\r\n engine.setDepthWrite(false);\r\n engine.setColorWrite(false);\r\n\r\n const boundingBox = mesh.getBoundingInfo().boundingBox;\r\n const min = boundingBox.minimum;\r\n const max = boundingBox.maximum;\r\n const diff = max.subtract(min);\r\n const median = min.add(diff.scale(0.5));\r\n\r\n const worldMatrix = Matrix.Scaling(diff.x, diff.y, diff.z).multiply(Matrix.Translation(median.x, median.y, median.z)).multiply(boundingBox.getWorldMatrix());\r\n\r\n const drawWrapper = subMesh._drawWrapper;\r\n\r\n this._colorShaderForOcclusionQuery._preBind(drawWrapper);\r\n\r\n engine.bindBuffers(this._vertexBuffers, this._fillIndexBuffer, drawWrapper.effect);\r\n\r\n if (useReverseDepthBuffer) {\r\n engine.setDepthFunctionToGreater();\r\n } else {\r\n engine.setDepthFunctionToLess();\r\n }\r\n\r\n this.scene.resetCachedMaterial();\r\n\r\n this._uniformBufferFront.bindToEffect(drawWrapper.effect!, \"BoundingBoxRenderer\");\r\n this._uniformBufferFront.updateMatrix(\"world\", worldMatrix);\r\n this._uniformBufferFront.updateMatrix(\"viewProjection\", this.scene.getTransformMatrix());\r\n this._uniformBufferFront.update();\r\n\r\n engine.drawElementsType(Material.TriangleFillMode, 0, 36);\r\n\r\n this._colorShaderForOcclusionQuery.unbind();\r\n engine.setDepthFunctionToLessOrEqual();\r\n engine.setDepthWrite(true);\r\n engine.setColorWrite(true);\r\n\r\n engine.currentRenderPassId = currentRenderPassId;\r\n }\r\n\r\n /**\r\n * Dispose and release the resources attached to this renderer.\r\n */\r\n public dispose(): void {\r\n if (this._renderPassIdForOcclusionQuery !== undefined) {\r\n this.scene.getEngine().releaseRenderPassId(this._renderPassIdForOcclusionQuery);\r\n this._renderPassIdForOcclusionQuery = undefined as any;\r\n }\r\n\r\n if (!this._colorShader) {\r\n return;\r\n }\r\n\r\n this.onBeforeBoxRenderingObservable.clear();\r\n this.onAfterBoxRenderingObservable.clear();\r\n this.onResourcesReadyObservable.clear();\r\n\r\n this.renderList.dispose();\r\n\r\n this._colorShader.dispose();\r\n this._colorShaderForOcclusionQuery.dispose();\r\n\r\n this._uniformBufferFront.dispose();\r\n this._uniformBufferBack.dispose();\r\n\r\n const buffer = this._vertexBuffers[VertexBuffer.PositionKind];\r\n if (buffer) {\r\n buffer.dispose();\r\n this._vertexBuffers[VertexBuffer.PositionKind] = null;\r\n }\r\n this.scene.getEngine()._releaseBuffer(this._indexBuffer);\r\n\r\n if (this._fillIndexBuffer) {\r\n this.scene.getEngine()._releaseBuffer(this._fillIndexBuffer);\r\n this._fillIndexBuffer = null;\r\n }\r\n }\r\n}\r\n","import { Vector3, Matrix } from \"../Maths/math.vector\";\r\nimport type { TransformNode } from \"../Meshes/transformNode\";\r\n\r\n/**\r\n * Class containing a set of static utilities functions for managing Pivots\r\n * @internal\r\n */\r\nexport class PivotTools {\r\n // Stores the state of the pivot cache (_oldPivotPoint, _pivotTranslation)\r\n // store/remove pivot point should only be applied during their outermost calls\r\n private static _PivotCached = 0;\r\n private static _OldPivotPoint = new Vector3();\r\n private static _PivotTranslation = new Vector3();\r\n private static _PivotTmpVector = new Vector3();\r\n private static _PivotPostMultiplyPivotMatrix = false;\r\n /**\r\n * @internal\r\n */\r\n public static _RemoveAndStorePivotPoint(mesh: TransformNode) {\r\n if (mesh && PivotTools._PivotCached === 0) {\r\n // Save old pivot and set pivot to 0,0,0\r\n mesh.getPivotPointToRef(PivotTools._OldPivotPoint);\r\n PivotTools._PivotPostMultiplyPivotMatrix = mesh._postMultiplyPivotMatrix;\r\n if (!PivotTools._OldPivotPoint.equalsToFloats(0, 0, 0)) {\r\n mesh.setPivotMatrix(Matrix.IdentityReadOnly);\r\n PivotTools._OldPivotPoint.subtractToRef(mesh.getPivotPoint(), PivotTools._PivotTranslation);\r\n PivotTools._PivotTmpVector.copyFromFloats(1, 1, 1);\r\n PivotTools._PivotTmpVector.subtractInPlace(mesh.scaling);\r\n PivotTools._PivotTmpVector.multiplyInPlace(PivotTools._PivotTranslation);\r\n mesh.position.addInPlace(PivotTools._PivotTmpVector);\r\n }\r\n }\r\n PivotTools._PivotCached++;\r\n }\r\n /**\r\n * @internal\r\n */\r\n public static _RestorePivotPoint(mesh: TransformNode) {\r\n if (mesh && !PivotTools._OldPivotPoint.equalsToFloats(0, 0, 0) && PivotTools._PivotCached === 1) {\r\n mesh.setPivotPoint(PivotTools._OldPivotPoint);\r\n mesh._postMultiplyPivotMatrix = PivotTools._PivotPostMultiplyPivotMatrix;\r\n PivotTools._PivotTmpVector.copyFromFloats(1, 1, 1);\r\n PivotTools._PivotTmpVector.subtractInPlace(mesh.scaling);\r\n PivotTools._PivotTmpVector.multiplyInPlace(PivotTools._PivotTranslation);\r\n mesh.position.subtractInPlace(PivotTools._PivotTmpVector);\r\n }\r\n this._PivotCached--;\r\n }\r\n}\r\n","import type { Behavior } from \"../../Behaviors/behavior\";\r\nimport { Mesh } from \"../../Meshes/mesh\";\r\nimport type { AbstractMesh } from \"../../Meshes/abstractMesh\";\r\nimport { Scene } from \"../../scene\";\r\nimport type { Nullable } from \"../../types\";\r\nimport type { Observer } from \"../../Misc/observable\";\r\nimport { Observable } from \"../../Misc/observable\";\r\nimport { Vector3 } from \"../../Maths/math.vector\";\r\nimport type { PointerInfo } from \"../../Events/pointerEvents\";\r\nimport { PointerEventTypes } from \"../../Events/pointerEvents\";\r\nimport { Ray } from \"../../Culling/ray\";\r\nimport { PivotTools } from \"../../Misc/pivotTools\";\r\nimport type { ArcRotateCamera } from \"../../Cameras/arcRotateCamera\";\r\nimport { CreatePlane } from \"../../Meshes/Builders/planeBuilder\";\r\n\r\nimport type { IPointerEvent } from \"../../Events/deviceInputEvents\";\r\n\r\n/**\r\n * A behavior that when attached to a mesh will allow the mesh to be dragged around the screen based on pointer events\r\n */\r\nexport class PointerDragBehavior implements Behavior {\r\n private static _AnyMouseId = -2;\r\n /**\r\n * Abstract mesh the behavior is set on\r\n */\r\n public attachedNode: AbstractMesh;\r\n protected _dragPlane: Mesh;\r\n private _scene: Scene;\r\n private _pointerObserver: Nullable>;\r\n private _beforeRenderObserver: Nullable>;\r\n private static _PlaneScene: Scene;\r\n private _useAlternatePickedPointAboveMaxDragAngleDragSpeed = -1.1;\r\n private _activeDragButton: number = -1;\r\n private _activePointerInfo: Nullable;\r\n /**\r\n * The maximum tolerated angle between the drag plane and dragging pointer rays to trigger pointer events. Set to 0 to allow any angle (default: 0)\r\n */\r\n public maxDragAngle = 0;\r\n /**\r\n * Butttons that can be used to initiate a drag\r\n */\r\n public dragButtons = [0, 1, 2];\r\n /**\r\n * @internal\r\n */\r\n public _useAlternatePickedPointAboveMaxDragAngle = false;\r\n /**\r\n * Get or set the currentDraggingPointerId\r\n * @deprecated Please use currentDraggingPointerId instead\r\n */\r\n public get currentDraggingPointerID(): number {\r\n return this.currentDraggingPointerId;\r\n }\r\n public set currentDraggingPointerID(currentDraggingPointerID: number) {\r\n this.currentDraggingPointerId = currentDraggingPointerID;\r\n }\r\n /**\r\n * The id of the pointer that is currently interacting with the behavior (-1 when no pointer is active)\r\n */\r\n public currentDraggingPointerId = -1;\r\n /**\r\n * The last position where the pointer hit the drag plane in world space\r\n */\r\n public lastDragPosition: Vector3;\r\n /**\r\n * If the behavior is currently in a dragging state\r\n */\r\n public dragging = false;\r\n /**\r\n * The distance towards the target drag position to move each frame. This can be useful to avoid jitter. Set this to 1 for no delay. (Default: 0.2)\r\n */\r\n public dragDeltaRatio = 0.2;\r\n /**\r\n * If the drag plane orientation should be updated during the dragging (Default: true)\r\n */\r\n public updateDragPlane = true;\r\n // Debug mode will display drag planes to help visualize behavior\r\n private _debugMode = false;\r\n private _moving = false;\r\n /**\r\n * Fires each time the attached mesh is dragged with the pointer\r\n * * delta between last drag position and current drag position in world space\r\n * * dragDistance along the drag axis\r\n * * dragPlaneNormal normal of the current drag plane used during the drag\r\n * * dragPlanePoint in world space where the drag intersects the drag plane\r\n *\r\n * (if validatedDrag is used, the position of the attached mesh might not equal dragPlanePoint)\r\n */\r\n public onDragObservable = new Observable<{\r\n delta: Vector3;\r\n dragPlanePoint: Vector3;\r\n dragPlaneNormal: Vector3;\r\n dragDistance: number;\r\n pointerId: number;\r\n pointerInfo: Nullable;\r\n }>();\r\n /**\r\n * Fires each time a drag begins (eg. mouse down on mesh)\r\n * * dragPlanePoint in world space where the drag intersects the drag plane\r\n *\r\n * (if validatedDrag is used, the position of the attached mesh might not equal dragPlanePoint)\r\n */\r\n public onDragStartObservable = new Observable<{ dragPlanePoint: Vector3; pointerId: number; pointerInfo: Nullable }>();\r\n /**\r\n * Fires each time a drag ends (eg. mouse release after drag)\r\n * * dragPlanePoint in world space where the drag intersects the drag plane\r\n *\r\n * (if validatedDrag is used, the position of the attached mesh might not equal dragPlanePoint)\r\n */\r\n public onDragEndObservable = new Observable<{ dragPlanePoint: Vector3; pointerId: number; pointerInfo: Nullable }>();\r\n /**\r\n * Fires each time behavior enabled state changes\r\n */\r\n public onEnabledObservable = new Observable();\r\n\r\n /**\r\n * If the attached mesh should be moved when dragged\r\n */\r\n public moveAttached = true;\r\n\r\n /**\r\n * If the drag behavior will react to drag events (Default: true)\r\n */\r\n public set enabled(value: boolean) {\r\n if (value != this._enabled) {\r\n this.onEnabledObservable.notifyObservers(value);\r\n }\r\n this._enabled = value;\r\n }\r\n\r\n public get enabled() {\r\n return this._enabled;\r\n }\r\n private _enabled = true;\r\n\r\n /**\r\n * If pointer events should start and release the drag (Default: true)\r\n */\r\n public startAndReleaseDragOnPointerEvents = true;\r\n /**\r\n * If camera controls should be detached during the drag\r\n */\r\n public detachCameraControls = true;\r\n\r\n /**\r\n * If set, the drag plane/axis will be rotated based on the attached mesh's world rotation (Default: true)\r\n */\r\n public useObjectOrientationForDragging = true;\r\n\r\n private _options: { dragAxis?: Vector3; dragPlaneNormal?: Vector3 };\r\n\r\n /**\r\n * Gets the options used by the behavior\r\n */\r\n public get options(): { dragAxis?: Vector3; dragPlaneNormal?: Vector3 } {\r\n return this._options;\r\n }\r\n\r\n /**\r\n * Sets the options used by the behavior\r\n */\r\n public set options(options: { dragAxis?: Vector3; dragPlaneNormal?: Vector3 }) {\r\n this._options = options;\r\n }\r\n\r\n /**\r\n * Creates a pointer drag behavior that can be attached to a mesh\r\n * @param options The drag axis or normal of the plane that will be dragged across. If no options are specified the drag plane will always face the ray's origin (eg. camera)\r\n * @param options.dragAxis\r\n * @param options.dragPlaneNormal\r\n */\r\n constructor(options?: { dragAxis?: Vector3; dragPlaneNormal?: Vector3 }) {\r\n this._options = options ? options : {};\r\n\r\n let optionCount = 0;\r\n if (this._options.dragAxis) {\r\n optionCount++;\r\n }\r\n if (this._options.dragPlaneNormal) {\r\n optionCount++;\r\n }\r\n if (optionCount > 1) {\r\n throw \"Multiple drag modes specified in dragBehavior options. Only one expected\";\r\n }\r\n }\r\n\r\n /**\r\n * Predicate to determine if it is valid to move the object to a new position when it is moved\r\n * @param targetPosition\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public validateDrag = (targetPosition: Vector3) => {\r\n return true;\r\n };\r\n\r\n /**\r\n * The name of the behavior\r\n */\r\n public get name(): string {\r\n return \"PointerDrag\";\r\n }\r\n\r\n /**\r\n * Initializes the behavior\r\n */\r\n public init() {}\r\n\r\n private _tmpVector = new Vector3(0, 0, 0);\r\n private _alternatePickedPoint = new Vector3(0, 0, 0);\r\n private _worldDragAxis = new Vector3(0, 0, 0);\r\n private _targetPosition = new Vector3(0, 0, 0);\r\n private _attachedToElement: boolean = false;\r\n /**\r\n * Attaches the drag behavior the passed in mesh\r\n * @param ownerNode The mesh that will be dragged around once attached\r\n * @param predicate Predicate to use for pick filtering\r\n */\r\n public attach(ownerNode: AbstractMesh, predicate?: (m: AbstractMesh) => boolean): void {\r\n this._scene = ownerNode.getScene();\r\n ownerNode.isNearGrabbable = true;\r\n this.attachedNode = ownerNode;\r\n\r\n // Initialize drag plane to not interfere with existing scene\r\n if (!PointerDragBehavior._PlaneScene) {\r\n if (this._debugMode) {\r\n PointerDragBehavior._PlaneScene = this._scene;\r\n } else {\r\n PointerDragBehavior._PlaneScene = new Scene(this._scene.getEngine(), { virtual: true });\r\n PointerDragBehavior._PlaneScene.detachControl();\r\n this._scene.onDisposeObservable.addOnce(() => {\r\n PointerDragBehavior._PlaneScene.dispose();\r\n (PointerDragBehavior._PlaneScene) = null;\r\n });\r\n }\r\n }\r\n this._dragPlane = CreatePlane(\r\n \"pointerDragPlane\",\r\n { size: this._debugMode ? 1 : 10000, updatable: false, sideOrientation: Mesh.DOUBLESIDE },\r\n PointerDragBehavior._PlaneScene\r\n );\r\n\r\n // State of the drag\r\n this.lastDragPosition = new Vector3(0, 0, 0);\r\n\r\n const pickPredicate = predicate\r\n ? predicate\r\n : (m: AbstractMesh) => {\r\n return this.attachedNode == m || m.isDescendantOf(this.attachedNode);\r\n };\r\n\r\n this._pointerObserver = this._scene.onPointerObservable.add((pointerInfo) => {\r\n if (!this.enabled) {\r\n // If behavior is disabled before releaseDrag is ever called, call it now.\r\n if (this._attachedToElement) {\r\n this.releaseDrag();\r\n }\r\n\r\n return;\r\n }\r\n\r\n if (pointerInfo.type == PointerEventTypes.POINTERDOWN) {\r\n if (\r\n this.startAndReleaseDragOnPointerEvents &&\r\n !this.dragging &&\r\n pointerInfo.pickInfo &&\r\n pointerInfo.pickInfo.hit &&\r\n pointerInfo.pickInfo.pickedMesh &&\r\n pointerInfo.pickInfo.pickedPoint &&\r\n pointerInfo.pickInfo.ray &&\r\n pickPredicate(pointerInfo.pickInfo.pickedMesh)\r\n ) {\r\n if (this._activeDragButton === -1 && this.dragButtons.indexOf(pointerInfo.event.button) !== -1) {\r\n this._activeDragButton = pointerInfo.event.button;\r\n this._activePointerInfo = pointerInfo;\r\n this._startDrag((pointerInfo.event).pointerId, pointerInfo.pickInfo.ray, pointerInfo.pickInfo.pickedPoint);\r\n }\r\n }\r\n } else if (pointerInfo.type == PointerEventTypes.POINTERUP) {\r\n if (\r\n this.startAndReleaseDragOnPointerEvents &&\r\n this.currentDraggingPointerId == (pointerInfo.event).pointerId &&\r\n (this._activeDragButton === pointerInfo.event.button || this._activeDragButton === -1)\r\n ) {\r\n this.releaseDrag();\r\n }\r\n } else if (pointerInfo.type == PointerEventTypes.POINTERMOVE) {\r\n const pointerId = (pointerInfo.event).pointerId;\r\n\r\n // If drag was started with anyMouseID specified, set pointerID to the next mouse that moved\r\n if (this.currentDraggingPointerId === PointerDragBehavior._AnyMouseId && pointerId !== PointerDragBehavior._AnyMouseId) {\r\n const evt = pointerInfo.event;\r\n const isMouseEvent = evt.pointerType === \"mouse\" || (!this._scene.getEngine().hostInformation.isMobile && evt instanceof MouseEvent);\r\n if (isMouseEvent) {\r\n if (this._lastPointerRay[this.currentDraggingPointerId]) {\r\n this._lastPointerRay[pointerId] = this._lastPointerRay[this.currentDraggingPointerId];\r\n delete this._lastPointerRay[this.currentDraggingPointerId];\r\n }\r\n this.currentDraggingPointerId = pointerId;\r\n }\r\n }\r\n\r\n // Keep track of last pointer ray, this is used simulating the start of a drag in startDrag()\r\n if (!this._lastPointerRay[pointerId]) {\r\n this._lastPointerRay[pointerId] = new Ray(new Vector3(), new Vector3());\r\n }\r\n if (pointerInfo.pickInfo && pointerInfo.pickInfo.ray) {\r\n this._lastPointerRay[pointerId].origin.copyFrom(pointerInfo.pickInfo.ray.origin);\r\n this._lastPointerRay[pointerId].direction.copyFrom(pointerInfo.pickInfo.ray.direction);\r\n\r\n if (this.currentDraggingPointerId == pointerId && this.dragging) {\r\n this._moveDrag(pointerInfo.pickInfo.ray);\r\n }\r\n }\r\n }\r\n });\r\n\r\n this._beforeRenderObserver = this._scene.onBeforeRenderObservable.add(() => {\r\n if (this._moving && this.moveAttached) {\r\n let needMatrixUpdate = false;\r\n PivotTools._RemoveAndStorePivotPoint(this.attachedNode);\r\n // Slowly move mesh to avoid jitter\r\n this._targetPosition.subtractToRef(this.attachedNode.absolutePosition, this._tmpVector);\r\n this._tmpVector.scaleInPlace(this.dragDeltaRatio);\r\n this.attachedNode.getAbsolutePosition().addToRef(this._tmpVector, this._tmpVector);\r\n if (this.validateDrag(this._tmpVector)) {\r\n this.attachedNode.setAbsolutePosition(this._tmpVector);\r\n needMatrixUpdate = true;\r\n }\r\n PivotTools._RestorePivotPoint(this.attachedNode);\r\n if (needMatrixUpdate) {\r\n this.attachedNode.computeWorldMatrix();\r\n }\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Force release the drag action by code.\r\n */\r\n public releaseDrag() {\r\n if (this.dragging) {\r\n this.dragging = false;\r\n this.onDragEndObservable.notifyObservers({ dragPlanePoint: this.lastDragPosition, pointerId: this.currentDraggingPointerId, pointerInfo: this._activePointerInfo });\r\n }\r\n\r\n this.currentDraggingPointerId = -1;\r\n this._activeDragButton = -1;\r\n this._activePointerInfo = null;\r\n this._moving = false;\r\n\r\n // Reattach camera controls\r\n if (this.detachCameraControls && this._attachedToElement && this._scene.activeCamera && !this._scene.activeCamera.leftCamera) {\r\n if (this._scene.activeCamera.getClassName() === \"ArcRotateCamera\") {\r\n const arcRotateCamera = this._scene.activeCamera as ArcRotateCamera;\r\n arcRotateCamera.attachControl(\r\n arcRotateCamera.inputs ? arcRotateCamera.inputs.noPreventDefault : true,\r\n arcRotateCamera._useCtrlForPanning,\r\n arcRotateCamera._panningMouseButton\r\n );\r\n } else {\r\n this._scene.activeCamera.attachControl(this._scene.activeCamera.inputs ? this._scene.activeCamera.inputs.noPreventDefault : true);\r\n }\r\n this._attachedToElement = false;\r\n }\r\n }\r\n\r\n private _startDragRay = new Ray(new Vector3(), new Vector3());\r\n private _lastPointerRay: { [key: number]: Ray } = {};\r\n /**\r\n * Simulates the start of a pointer drag event on the behavior\r\n * @param pointerId pointerID of the pointer that should be simulated (Default: Any mouse pointer ID)\r\n * @param fromRay initial ray of the pointer to be simulated (Default: Ray from camera to attached mesh)\r\n * @param startPickedPoint picked point of the pointer to be simulated (Default: attached mesh position)\r\n */\r\n public startDrag(pointerId: number = PointerDragBehavior._AnyMouseId, fromRay?: Ray, startPickedPoint?: Vector3) {\r\n this._startDrag(pointerId, fromRay, startPickedPoint);\r\n\r\n let lastRay = this._lastPointerRay[pointerId];\r\n if (pointerId === PointerDragBehavior._AnyMouseId) {\r\n lastRay = this._lastPointerRay[Object.keys(this._lastPointerRay)[0]];\r\n }\r\n\r\n if (lastRay) {\r\n // if there was a last pointer ray drag the object there\r\n this._moveDrag(lastRay);\r\n }\r\n }\r\n\r\n protected _startDrag(pointerId: number, fromRay?: Ray, startPickedPoint?: Vector3) {\r\n if (!this._scene.activeCamera || this.dragging || !this.attachedNode) {\r\n return;\r\n }\r\n\r\n PivotTools._RemoveAndStorePivotPoint(this.attachedNode);\r\n // Create start ray from the camera to the object\r\n if (fromRay) {\r\n this._startDragRay.direction.copyFrom(fromRay.direction);\r\n this._startDragRay.origin.copyFrom(fromRay.origin);\r\n } else {\r\n this._startDragRay.origin.copyFrom(this._scene.activeCamera.position);\r\n this.attachedNode.getWorldMatrix().getTranslationToRef(this._tmpVector);\r\n this._tmpVector.subtractToRef(this._scene.activeCamera.position, this._startDragRay.direction);\r\n }\r\n\r\n this._updateDragPlanePosition(this._startDragRay, startPickedPoint ? startPickedPoint : this._tmpVector);\r\n\r\n const pickedPoint = this._pickWithRayOnDragPlane(this._startDragRay);\r\n if (pickedPoint) {\r\n this.dragging = true;\r\n this.currentDraggingPointerId = pointerId;\r\n this.lastDragPosition.copyFrom(pickedPoint);\r\n this.onDragStartObservable.notifyObservers({ dragPlanePoint: pickedPoint, pointerId: this.currentDraggingPointerId, pointerInfo: this._activePointerInfo });\r\n this._targetPosition.copyFrom(this.attachedNode.getAbsolutePosition());\r\n\r\n // Detatch camera controls\r\n if (this.detachCameraControls && this._scene.activeCamera && this._scene.activeCamera.inputs && !this._scene.activeCamera.leftCamera) {\r\n if (this._scene.activeCamera.inputs.attachedToElement) {\r\n this._scene.activeCamera.detachControl();\r\n this._attachedToElement = true;\r\n } else {\r\n this._attachedToElement = false;\r\n }\r\n }\r\n } else {\r\n this.releaseDrag();\r\n }\r\n PivotTools._RestorePivotPoint(this.attachedNode);\r\n }\r\n\r\n private _dragDelta = new Vector3();\r\n protected _moveDrag(ray: Ray) {\r\n this._moving = true;\r\n const pickedPoint = this._pickWithRayOnDragPlane(ray);\r\n\r\n if (pickedPoint) {\r\n PivotTools._RemoveAndStorePivotPoint(this.attachedNode);\r\n\r\n if (this.updateDragPlane) {\r\n this._updateDragPlanePosition(ray, pickedPoint);\r\n }\r\n let dragLength = 0;\r\n // depending on the drag mode option drag accordingly\r\n if (this._options.dragAxis) {\r\n // Convert local drag axis to world if useObjectOrientationForDragging\r\n this.useObjectOrientationForDragging\r\n ? Vector3.TransformCoordinatesToRef(this._options.dragAxis, this.attachedNode.getWorldMatrix().getRotationMatrix(), this._worldDragAxis)\r\n : this._worldDragAxis.copyFrom(this._options.dragAxis);\r\n\r\n // Project delta drag from the drag plane onto the drag axis\r\n pickedPoint.subtractToRef(this.lastDragPosition, this._tmpVector);\r\n dragLength = Vector3.Dot(this._tmpVector, this._worldDragAxis);\r\n this._worldDragAxis.scaleToRef(dragLength, this._dragDelta);\r\n } else {\r\n dragLength = this._dragDelta.length();\r\n pickedPoint.subtractToRef(this.lastDragPosition, this._dragDelta);\r\n }\r\n this._targetPosition.addInPlace(this._dragDelta);\r\n this.onDragObservable.notifyObservers({\r\n dragDistance: dragLength,\r\n delta: this._dragDelta,\r\n dragPlanePoint: pickedPoint,\r\n dragPlaneNormal: this._dragPlane.forward,\r\n pointerId: this.currentDraggingPointerId,\r\n pointerInfo: this._activePointerInfo,\r\n });\r\n this.lastDragPosition.copyFrom(pickedPoint);\r\n\r\n PivotTools._RestorePivotPoint(this.attachedNode);\r\n }\r\n }\r\n\r\n private _pickWithRayOnDragPlane(ray: Nullable) {\r\n if (!ray) {\r\n return null;\r\n }\r\n\r\n // Calculate angle between plane normal and ray\r\n let angle = Math.acos(Vector3.Dot(this._dragPlane.forward, ray.direction));\r\n // Correct if ray is casted from oposite side\r\n if (angle > Math.PI / 2) {\r\n angle = Math.PI - angle;\r\n }\r\n\r\n // If the angle is too perpendicular to the plane pick another point on the plane where it is looking\r\n if (this.maxDragAngle > 0 && angle > this.maxDragAngle) {\r\n if (this._useAlternatePickedPointAboveMaxDragAngle) {\r\n // Invert ray direction along the towards object axis\r\n this._tmpVector.copyFrom(ray.direction);\r\n this.attachedNode.absolutePosition.subtractToRef(ray.origin, this._alternatePickedPoint);\r\n this._alternatePickedPoint.normalize();\r\n this._alternatePickedPoint.scaleInPlace(this._useAlternatePickedPointAboveMaxDragAngleDragSpeed * Vector3.Dot(this._alternatePickedPoint, this._tmpVector));\r\n this._tmpVector.addInPlace(this._alternatePickedPoint);\r\n\r\n // Project resulting vector onto the drag plane and add it to the attached nodes absolute position to get a picked point\r\n const dot = Vector3.Dot(this._dragPlane.forward, this._tmpVector);\r\n this._dragPlane.forward.scaleToRef(-dot, this._alternatePickedPoint);\r\n this._alternatePickedPoint.addInPlace(this._tmpVector);\r\n this._alternatePickedPoint.addInPlace(this.attachedNode.absolutePosition);\r\n return this._alternatePickedPoint;\r\n } else {\r\n return null;\r\n }\r\n }\r\n\r\n const pickResult = PointerDragBehavior._PlaneScene.pickWithRay(ray, (m) => {\r\n return m == this._dragPlane;\r\n });\r\n if (pickResult && pickResult.hit && pickResult.pickedMesh && pickResult.pickedPoint) {\r\n return pickResult.pickedPoint;\r\n } else {\r\n return null;\r\n }\r\n }\r\n\r\n // Variables to avoid instantiation in the below method\r\n private _pointA = new Vector3(0, 0, 0);\r\n private _pointC = new Vector3(0, 0, 0);\r\n private _localAxis = new Vector3(0, 0, 0);\r\n private _lookAt = new Vector3(0, 0, 0);\r\n // Position the drag plane based on the attached mesh position, for single axis rotate the plane along the axis to face the camera\r\n private _updateDragPlanePosition(ray: Ray, dragPlanePosition: Vector3) {\r\n this._pointA.copyFrom(dragPlanePosition);\r\n if (this._options.dragAxis) {\r\n this.useObjectOrientationForDragging\r\n ? Vector3.TransformCoordinatesToRef(this._options.dragAxis, this.attachedNode.getWorldMatrix().getRotationMatrix(), this._localAxis)\r\n : this._localAxis.copyFrom(this._options.dragAxis);\r\n\r\n // Calculate plane normal that is the cross product of local axis and (eye-dragPlanePosition)\r\n ray.origin.subtractToRef(this._pointA, this._pointC);\r\n this._pointC.normalize();\r\n if (Math.abs(Vector3.Dot(this._localAxis, this._pointC)) > 0.999) {\r\n // the drag axis is colinear with the (eye to position) ray. The cross product will give jittered values.\r\n // A new axis vector need to be computed\r\n if (Math.abs(Vector3.Dot(Vector3.UpReadOnly, this._pointC)) > 0.999) {\r\n this._lookAt.copyFrom(Vector3.Right());\r\n } else {\r\n this._lookAt.copyFrom(Vector3.UpReadOnly);\r\n }\r\n } else {\r\n Vector3.CrossToRef(this._localAxis, this._pointC, this._lookAt);\r\n // Get perpendicular line from previous result and drag axis to adjust lineB to be perpendicular to camera\r\n Vector3.CrossToRef(this._localAxis, this._lookAt, this._lookAt);\r\n this._lookAt.normalize();\r\n }\r\n\r\n this._dragPlane.position.copyFrom(this._pointA);\r\n this._pointA.addToRef(this._lookAt, this._lookAt);\r\n this._dragPlane.lookAt(this._lookAt);\r\n } else if (this._options.dragPlaneNormal) {\r\n this.useObjectOrientationForDragging\r\n ? Vector3.TransformCoordinatesToRef(this._options.dragPlaneNormal, this.attachedNode.getWorldMatrix().getRotationMatrix(), this._localAxis)\r\n : this._localAxis.copyFrom(this._options.dragPlaneNormal);\r\n this._dragPlane.position.copyFrom(this._pointA);\r\n this._pointA.addToRef(this._localAxis, this._lookAt);\r\n this._dragPlane.lookAt(this._lookAt);\r\n } else {\r\n this._dragPlane.position.copyFrom(this._pointA);\r\n this._dragPlane.lookAt(ray.origin);\r\n }\r\n // Update the position of the drag plane so it doesn't get out of sync with the node (eg. when moving back and forth quickly)\r\n this._dragPlane.position.copyFrom(this.attachedNode.getAbsolutePosition());\r\n\r\n this._dragPlane.computeWorldMatrix(true);\r\n }\r\n\r\n /**\r\n * Detaches the behavior from the mesh\r\n */\r\n public detach(): void {\r\n this._lastPointerRay = {};\r\n if (this.attachedNode) {\r\n this.attachedNode.isNearGrabbable = false;\r\n }\r\n if (this._pointerObserver) {\r\n this._scene.onPointerObservable.remove(this._pointerObserver);\r\n }\r\n if (this._beforeRenderObserver) {\r\n this._scene.onBeforeRenderObservable.remove(this._beforeRenderObserver);\r\n }\r\n if (this._dragPlane) {\r\n this._dragPlane.dispose();\r\n }\r\n this.releaseDrag();\r\n }\r\n}\r\n","import { PointerDragBehavior } from '@babylonjs/core/Behaviors/Meshes/pointerDragBehavior';\r\nimport { WebVRFreeCamera } from '@babylonjs/core/Cameras/VR/webVRCamera';\r\nimport { PointerEventTypes, PointerInfo } from '@babylonjs/core/Events/pointerEvents';\r\nimport { StandardMaterial } from '@babylonjs/core/Materials/standardMaterial';\r\nimport { Color3, Quaternion, Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { CylinderBuilder } from '@babylonjs/core/Meshes/Builders/cylinderBuilder';\r\nimport { SphereBuilder } from '@babylonjs/core/Meshes/Builders/sphereBuilder';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { Observable, Observer } from '@babylonjs/core/Misc/observable';\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { Nullable } from '@babylonjs/core/types';\r\nimport { rotationToVector, rotationToVectorToRef } from '@view/helpers/math-helper';\r\n\r\n/**\r\n * Single plane rotation gizmo\r\n */\r\nexport class KbAxisMorphGizmo {\r\n /**\r\n * Drag behavior responsible for the gizmos dragging interactions\r\n */\r\n public dragBehavior: PointerDragBehavior;\r\n public dragActive = false;\r\n public direction = new Vector3();\r\n private _rootMesh: Mesh;\r\n private axisMesh: AbstractMesh;\r\n private _pointerObserver: Nullable> = null;\r\n private _tempVector = new Vector3();\r\n private _updateQuat = new Quaternion();\r\n\r\n protected _scaleRatio = 1;\r\n\r\n /**\r\n * Ratio for the scale of the gizmo (Default: 1)\r\n */\r\n public set scaleRatio(value: number) {\r\n this._scaleRatio = value;\r\n }\r\n\r\n public get scaleRatio() {\r\n return this._scaleRatio;\r\n }\r\n\r\n public set gizmoPosition(value: Vector3) {\r\n this._rootMesh.position = value;\r\n }\r\n\r\n public get gizmoPosition() {\r\n return this._rootMesh.position;\r\n }\r\n\r\n /**\r\n * Rotation distance in radians that the gizmo will snap to (Default: 0)\r\n */\r\n public snapDistance = 0;\r\n /**\r\n * Event that fires each time the gizmo snaps to a new location.\r\n * * snapDistance is the the change in distance\r\n */\r\n public updateObservable = new Observable<{ delta: Quaternion }>();\r\n\r\n private _beforeRenderObserver: Nullable>;\r\n\r\n /**\r\n * Creates a PlaneRotationGizmo\r\n * @param gizmoLayer The utility layer the gizmo will be added to\r\n * @param color The color of the gizmo\r\n * @param tessellation Amount of tessellation to be used when creating rotation circles\r\n */\r\n constructor(\r\n color: Color3 = Color3.Gray(),\r\n private gizmoLayer: UtilityLayerRenderer = UtilityLayerRenderer.DefaultUtilityLayer,\r\n tessellation = 24\r\n ) {\r\n this._rootMesh = new Mesh('gizmoRootNode', gizmoLayer.utilityLayerScene);\r\n\r\n // Create Material\r\n const coloredMaterial = new StandardMaterial('', gizmoLayer.utilityLayerScene);\r\n coloredMaterial.emissiveColor = color;\r\n\r\n const wireframeMaterial = new StandardMaterial('', gizmoLayer.utilityLayerScene);\r\n wireframeMaterial.emissiveColor = color;\r\n wireframeMaterial.alpha = 0.4;\r\n // coloredMaterial.diffuseColor = color;\r\n // coloredMaterial.specularColor = color.subtract(new Color3(0.1, 0.1, 0.1));\r\n\r\n const hoverMaterial = new StandardMaterial('', gizmoLayer.utilityLayerScene);\r\n hoverMaterial.emissiveColor = color.add(new Color3(0.3, 0.3, 0.3));\r\n wireframeMaterial.alpha = 0.6;\r\n // hoverMaterial.diffuseColor = color.add(new Color3(0.3, 0.3, 0.3));\r\n\r\n // Build mesh to hold axis line\r\n this.axisMesh = new AbstractMesh('', gizmoLayer.utilityLayerScene);\r\n this.axisMesh.rotationQuaternion = Quaternion.Identity();\r\n\r\n const line = CylinderBuilder.CreateCylinder(\r\n '',\r\n { diameterTop: 0.01, height: 0.3, diameterBottom: 0.01, tessellation },\r\n gizmoLayer.utilityLayerScene\r\n );\r\n line.material = coloredMaterial;\r\n line.position.y = 0.375;\r\n const line2 = CylinderBuilder.CreateCylinder(\r\n '',\r\n { diameterTop: 0.005, height: 0.9, diameterBottom: 0.005, tessellation },\r\n gizmoLayer.utilityLayerScene\r\n );\r\n line2.material = coloredMaterial;\r\n line2.position.y = -0.225;\r\n\r\n const sphereMesh = SphereBuilder.CreateSphere(\r\n '',\r\n { segments: tessellation, diameter: 0.45 },\r\n gizmoLayer.utilityLayerScene\r\n );\r\n sphereMesh.material = wireframeMaterial;\r\n\r\n this.axisMesh.isPickable = false;\r\n this.axisMesh.addChild(line);\r\n this.axisMesh.addChild(line2);\r\n\r\n sphereMesh.addChild(this.axisMesh);\r\n sphereMesh.scaling.scaleInPlace(1 / 3);\r\n this._rootMesh.addChild(sphereMesh);\r\n\r\n let lastDragPosition: Vector3 | null = null;\r\n\r\n // Add drag behavior to handle events when the gizmo is dragged\r\n this.dragBehavior = new PointerDragBehavior();\r\n this.dragBehavior.moveAttached = false;\r\n\r\n this.dragBehavior.onDragStartObservable.add(() => {\r\n this.dragActive = true;\r\n });\r\n this.dragBehavior.onDragObservable.add(event => {\r\n // activeCamera.getForwardRayToRef(forwardRay, 1);\r\n // const cameraDirection = forwardRay.direction;\r\n // cameraDirection.rotateByQuaternionToRef(parentMesh.rotationQuaternion!, cameraDirection);\r\n // cameraDirection.addToRef(event.delta, targetDirection);\r\n // rotationToVectorToRef(targetDirection, cameraDirection, this._updateQuat);\r\n // parentMesh.rotationQuaternion!.multiplyInPlace(this._updateQuat);\r\n // this.updateObservable.notifyObservers({ delta: this._updateQuat });\r\n });\r\n this.dragBehavior.onDragEndObservable.add(() => {\r\n this.dragActive = false;\r\n });\r\n\r\n this._rootMesh.addBehavior(this.dragBehavior);\r\n this.dragBehavior.enabled = true;\r\n\r\n this._pointerObserver = gizmoLayer.utilityLayerScene.onPointerObservable.add(pointerInfo => {\r\n const isHovered =\r\n pointerInfo.pickInfo &&\r\n this._rootMesh.getChildMeshes().indexOf(pointerInfo.pickInfo.pickedMesh) != -1;\r\n sphereMesh.material = isHovered ? hoverMaterial : wireframeMaterial;\r\n\r\n if (pointerInfo.pickInfo && isHovered) {\r\n const point = pointerInfo.pickInfo.pickedPoint;\r\n if (point) {\r\n point.subtractInPlace(this._rootMesh.position);\r\n switch (pointerInfo.type) {\r\n case PointerEventTypes.POINTERDOWN:\r\n lastDragPosition = point.clone();\r\n break;\r\n case PointerEventTypes.POINTERMOVE:\r\n if (lastDragPosition && this.dragActive) {\r\n rotationToVectorToRef(point, lastDragPosition, this._updateQuat);\r\n lastDragPosition.copyFrom(point);\r\n // console.log(point, this._updateQuat);\r\n this.direction.rotateByQuaternionToRef(this._updateQuat, this.direction);\r\n this.updateObservable.notifyObservers({ delta: this._updateQuat });\r\n }\r\n break;\r\n }\r\n }\r\n }\r\n });\r\n\r\n // let light = gizmoLayer._getSharedGizmoLight();\r\n // light.includedOnlyMeshes = light.includedOnlyMeshes.concat(this._rootMesh.getChildMeshes(false));\r\n\r\n this._beforeRenderObserver = this.gizmoLayer.utilityLayerScene.onBeforeRenderObservable.add(() => {\r\n this._update();\r\n });\r\n }\r\n protected _update() {\r\n if (this.direction) {\r\n this.axisMesh.rotationQuaternion = rotationToVector(this.direction);\r\n }\r\n const activeCamera = this.gizmoLayer.utilityLayerScene.activeCamera!;\r\n let cameraPosition = activeCamera.globalPosition;\r\n if ((activeCamera).devicePosition) {\r\n cameraPosition = (activeCamera).devicePosition;\r\n }\r\n this._rootMesh.position.subtractToRef(cameraPosition, this._tempVector);\r\n const dist = this._tempVector.length() * this.scaleRatio;\r\n this._rootMesh.scaling.set(dist, dist, dist);\r\n\r\n // Account for handedness, similar to Matrix.decompose\r\n if (this.gizmoLayer.originalScene.meshes[0]?._getWorldMatrixDeterminant() < 0) {\r\n this._rootMesh.scaling.y *= -1;\r\n }\r\n }\r\n\r\n /**\r\n * Disposes of the gizmo\r\n */\r\n public dispose() {\r\n this.gizmoLayer.utilityLayerScene.onPointerObservable.remove(this._pointerObserver);\r\n this._rootMesh.dispose();\r\n this.dragBehavior.detach();\r\n\r\n if (this._beforeRenderObserver) {\r\n this.gizmoLayer.utilityLayerScene.onBeforeRenderObservable.remove(this._beforeRenderObserver);\r\n }\r\n }\r\n}\r\n","import type { Observer } from \"../Misc/observable\";\r\nimport type { Nullable } from \"../types\";\r\nimport type { WebVRFreeCamera } from \"../Cameras/VR/webVRCamera\";\r\nimport type { Scene, IDisposable } from \"../scene\";\r\nimport { Quaternion, Vector3, Matrix, TmpVectors } from \"../Maths/math.vector\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport { Mesh } from \"../Meshes/mesh\";\r\nimport { Camera } from \"../Cameras/camera\";\r\nimport type { TargetCamera } from \"../Cameras/targetCamera\";\r\nimport type { Node } from \"../node\";\r\nimport type { Bone } from \"../Bones/bone\";\r\nimport { UtilityLayerRenderer } from \"../Rendering/utilityLayerRenderer\";\r\nimport type { TransformNode } from \"../Meshes/transformNode\";\r\nimport type { StandardMaterial } from \"../Materials/standardMaterial\";\r\nimport type { PointerInfo } from \"../Events/pointerEvents\";\r\nimport { PointerEventTypes } from \"../Events/pointerEvents\";\r\nimport type { LinesMesh } from \"../Meshes/linesMesh\";\r\nimport type { PointerDragBehavior } from \"../Behaviors/Meshes/pointerDragBehavior\";\r\nimport type { ShadowLight } from \"../Lights/shadowLight\";\r\nimport { Light } from \"../Lights/light\";\r\n\r\n/**\r\n * Cache built by each axis. Used for managing state between all elements of gizmo for enhanced UI\r\n */\r\nexport interface GizmoAxisCache {\r\n /** Mesh used to render the Gizmo */\r\n gizmoMeshes: Mesh[];\r\n /** Mesh used to detect user interaction with Gizmo */\r\n colliderMeshes: Mesh[];\r\n /** Material used to indicate color of gizmo mesh */\r\n material: StandardMaterial;\r\n /** Material used to indicate hover state of the Gizmo */\r\n hoverMaterial: StandardMaterial;\r\n /** Material used to indicate disabled state of the Gizmo */\r\n disableMaterial: StandardMaterial;\r\n /** Used to indicate Active state of the Gizmo */\r\n active: boolean;\r\n /** DragBehavior */\r\n dragBehavior: PointerDragBehavior;\r\n}\r\n\r\n/**\r\n * Anchor options where the Gizmo can be positioned in relation to its anchored node\r\n */\r\nexport enum GizmoAnchorPoint {\r\n /** The origin of the attached node */\r\n Origin,\r\n /** The pivot point of the attached node*/\r\n Pivot,\r\n}\r\n\r\n/**\r\n * Coordinates mode: Local or World. Defines how axis is aligned: either on world axis or transform local axis\r\n */\r\nexport enum GizmoCoordinatesMode {\r\n World,\r\n Local,\r\n}\r\n\r\n/**\r\n * Interface for basic gizmo\r\n */\r\nexport interface IGizmo extends IDisposable {\r\n /** True when the mouse pointer is hovered a gizmo mesh */\r\n readonly isHovered: boolean;\r\n /** The root mesh of the gizmo */\r\n _rootMesh: Mesh;\r\n /** Ratio for the scale of the gizmo */\r\n scaleRatio: number;\r\n /**\r\n * Mesh that the gizmo will be attached to. (eg. on a drag gizmo the mesh that will be dragged)\r\n * * When set, interactions will be enabled\r\n */\r\n attachedMesh: Nullable;\r\n /**\r\n * Node that the gizmo will be attached to. (eg. on a drag gizmo the mesh, bone or NodeTransform that will be dragged)\r\n * * When set, interactions will be enabled\r\n */\r\n attachedNode: Nullable;\r\n /**\r\n * If set the gizmo's rotation will be updated to match the attached mesh each frame (Default: true)\r\n */\r\n updateGizmoRotationToMatchAttachedMesh: boolean;\r\n /** The utility layer the gizmo will be added to */\r\n gizmoLayer: UtilityLayerRenderer;\r\n /**\r\n * If set the gizmo's position will be updated to match the attached mesh each frame (Default: true)\r\n */\r\n updateGizmoPositionToMatchAttachedMesh: boolean;\r\n /**\r\n * Defines where the gizmo will be positioned if `updateGizmoPositionToMatchAttachedMesh` is enabled.\r\n * (Default: GizmoAnchorPoint.Origin)\r\n */\r\n anchorPoint: GizmoAnchorPoint;\r\n\r\n /**\r\n * Set the coordinate mode to use. By default it's local.\r\n */\r\n coordinatesMode: GizmoCoordinatesMode;\r\n\r\n /**\r\n * When set, the gizmo will always appear the same size no matter where the camera is (default: true)\r\n */\r\n updateScale: boolean;\r\n /**\r\n * posture that the gizmo will be display\r\n * When set null, default value will be used (Quaternion(0, 0, 0, 1))\r\n */\r\n customRotationQuaternion: Nullable;\r\n /** Disposes and replaces the current meshes in the gizmo with the specified mesh */\r\n setCustomMesh(mesh: Mesh): void;\r\n}\r\n/**\r\n * Renders gizmos on top of an existing scene which provide controls for position, rotation, etc.\r\n */\r\nexport class Gizmo implements IGizmo {\r\n /**\r\n * The root mesh of the gizmo\r\n */\r\n public _rootMesh: Mesh;\r\n protected _attachedMesh: Nullable = null;\r\n protected _attachedNode: Nullable = null;\r\n protected _customRotationQuaternion: Nullable = null;\r\n /**\r\n * Ratio for the scale of the gizmo (Default: 1)\r\n */\r\n protected _scaleRatio = 1;\r\n\r\n /**\r\n * boolean updated by pointermove when a gizmo mesh is hovered\r\n */\r\n protected _isHovered = false;\r\n\r\n /**\r\n * When enabled, any gizmo operation will perserve scaling sign. Default is off.\r\n * Only valid for TransformNode derived classes (Mesh, AbstractMesh, ...)\r\n */\r\n public static PreserveScaling = false;\r\n\r\n /**\r\n * Ratio for the scale of the gizmo (Default: 1)\r\n */\r\n public set scaleRatio(value: number) {\r\n this._scaleRatio = value;\r\n }\r\n\r\n public get scaleRatio() {\r\n return this._scaleRatio;\r\n }\r\n\r\n /**\r\n * True when the mouse pointer is hovered a gizmo mesh\r\n */\r\n public get isHovered() {\r\n return this._isHovered;\r\n }\r\n\r\n /**\r\n * If a custom mesh has been set (Default: false)\r\n */\r\n protected _customMeshSet = false;\r\n /**\r\n * Mesh that the gizmo will be attached to. (eg. on a drag gizmo the mesh that will be dragged)\r\n * * When set, interactions will be enabled\r\n */\r\n public get attachedMesh() {\r\n return this._attachedMesh;\r\n }\r\n public set attachedMesh(value) {\r\n this._attachedMesh = value;\r\n if (value) {\r\n this._attachedNode = value;\r\n }\r\n this._rootMesh.setEnabled(value ? true : false);\r\n this._attachedNodeChanged(value);\r\n }\r\n /**\r\n * Node that the gizmo will be attached to. (eg. on a drag gizmo the mesh, bone or NodeTransform that will be dragged)\r\n * * When set, interactions will be enabled\r\n */\r\n public get attachedNode() {\r\n return this._attachedNode;\r\n }\r\n public set attachedNode(value) {\r\n this._attachedNode = value;\r\n this._attachedMesh = null;\r\n this._rootMesh.setEnabled(value ? true : false);\r\n this._attachedNodeChanged(value);\r\n }\r\n\r\n /**\r\n * Disposes and replaces the current meshes in the gizmo with the specified mesh\r\n * @param mesh The mesh to replace the default mesh of the gizmo\r\n */\r\n public setCustomMesh(mesh: Mesh) {\r\n if (mesh.getScene() != this.gizmoLayer.utilityLayerScene) {\r\n throw \"When setting a custom mesh on a gizmo, the custom meshes scene must be the same as the gizmos (eg. gizmo.gizmoLayer.utilityLayerScene)\";\r\n }\r\n this._rootMesh.getChildMeshes().forEach((c) => {\r\n c.dispose();\r\n });\r\n mesh.parent = this._rootMesh;\r\n this._customMeshSet = true;\r\n }\r\n\r\n protected _updateGizmoRotationToMatchAttachedMesh = true;\r\n protected _updateGizmoPositionToMatchAttachedMesh = true;\r\n protected _anchorPoint = GizmoAnchorPoint.Origin;\r\n protected _updateScale = true;\r\n protected _coordinatesMode = GizmoCoordinatesMode.Local;\r\n\r\n /**\r\n * If set the gizmo's rotation will be updated to match the attached mesh each frame (Default: true)\r\n * NOTE: This is only possible for meshes with uniform scaling, as otherwise it's not possible to decompose the rotation\r\n */\r\n public set updateGizmoRotationToMatchAttachedMesh(value: boolean) {\r\n this._updateGizmoRotationToMatchAttachedMesh = value;\r\n }\r\n public get updateGizmoRotationToMatchAttachedMesh() {\r\n return this._updateGizmoRotationToMatchAttachedMesh;\r\n }\r\n /**\r\n * If set the gizmo's position will be updated to match the attached mesh each frame (Default: true)\r\n */\r\n public set updateGizmoPositionToMatchAttachedMesh(value: boolean) {\r\n this._updateGizmoPositionToMatchAttachedMesh = value;\r\n }\r\n public get updateGizmoPositionToMatchAttachedMesh() {\r\n return this._updateGizmoPositionToMatchAttachedMesh;\r\n }\r\n\r\n /**\r\n * Defines where the gizmo will be positioned if `updateGizmoPositionToMatchAttachedMesh` is enabled.\r\n * (Default: GizmoAnchorPoint.Origin)\r\n */\r\n public set anchorPoint(value: GizmoAnchorPoint) {\r\n this._anchorPoint = value;\r\n }\r\n public get anchorPoint() {\r\n return this._anchorPoint;\r\n }\r\n\r\n /**\r\n * Set the coordinate system to use. By default it's local.\r\n * But it's possible for a user to tweak so its local for translation and world for rotation.\r\n * In that case, setting the coordinate system will change `updateGizmoRotationToMatchAttachedMesh` and `updateGizmoPositionToMatchAttachedMesh`\r\n */\r\n public set coordinatesMode(coordinatesMode: GizmoCoordinatesMode) {\r\n this._coordinatesMode = coordinatesMode;\r\n const local = coordinatesMode == GizmoCoordinatesMode.Local;\r\n this.updateGizmoRotationToMatchAttachedMesh = local;\r\n this.updateGizmoPositionToMatchAttachedMesh = local;\r\n }\r\n\r\n public get coordinatesMode() {\r\n return this._coordinatesMode;\r\n }\r\n\r\n /**\r\n * When set, the gizmo will always appear the same size no matter where the camera is (default: true)\r\n */\r\n\r\n public set updateScale(value: boolean) {\r\n this._updateScale = value;\r\n }\r\n public get updateScale() {\r\n return this._updateScale;\r\n }\r\n protected _interactionsEnabled = true;\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n protected _attachedNodeChanged(value: Nullable) {}\r\n\r\n protected _beforeRenderObserver: Nullable>;\r\n private _rightHandtoLeftHandMatrix = Matrix.RotationY(Math.PI);\r\n\r\n /**\r\n * Creates a gizmo\r\n * @param gizmoLayer The utility layer the gizmo will be added to\r\n */\r\n constructor(\r\n /** The utility layer the gizmo will be added to */\r\n public gizmoLayer: UtilityLayerRenderer = UtilityLayerRenderer.DefaultUtilityLayer\r\n ) {\r\n this._rootMesh = new Mesh(\"gizmoRootNode\", gizmoLayer.utilityLayerScene);\r\n this._rootMesh.rotationQuaternion = Quaternion.Identity();\r\n\r\n this._beforeRenderObserver = this.gizmoLayer.utilityLayerScene.onBeforeRenderObservable.add(() => {\r\n this._update();\r\n });\r\n }\r\n\r\n /**\r\n * posture that the gizmo will be display\r\n * When set null, default value will be used (Quaternion(0, 0, 0, 1))\r\n */\r\n public get customRotationQuaternion(): Nullable {\r\n return this._customRotationQuaternion;\r\n }\r\n\r\n public set customRotationQuaternion(customRotationQuaternion: Nullable) {\r\n this._customRotationQuaternion = customRotationQuaternion;\r\n }\r\n\r\n /**\r\n * Updates the gizmo to match the attached mesh's position/rotation\r\n */\r\n protected _update() {\r\n if (this.attachedNode) {\r\n let effectiveNode = this.attachedNode;\r\n if (this.attachedMesh) {\r\n effectiveNode = this.attachedMesh || this.attachedNode;\r\n }\r\n\r\n // Position\r\n if (this.updateGizmoPositionToMatchAttachedMesh) {\r\n if (this.anchorPoint == GizmoAnchorPoint.Pivot && (effectiveNode).getAbsolutePivotPoint) {\r\n const position = (effectiveNode).getAbsolutePivotPoint();\r\n this._rootMesh.position.copyFrom(position);\r\n } else {\r\n const row = effectiveNode.getWorldMatrix().getRow(3);\r\n const position = row ? row.toVector3() : new Vector3(0, 0, 0);\r\n this._rootMesh.position.copyFrom(position);\r\n }\r\n }\r\n\r\n // Rotation\r\n if (this.updateGizmoRotationToMatchAttachedMesh) {\r\n const supportedNode =\r\n (effectiveNode)._isMesh ||\r\n effectiveNode.getClassName() === \"AbstractMesh\" ||\r\n effectiveNode.getClassName() === \"TransformNode\" ||\r\n effectiveNode.getClassName() === \"InstancedMesh\";\r\n const transformNode = supportedNode ? (effectiveNode as TransformNode) : undefined;\r\n effectiveNode.getWorldMatrix().decompose(undefined, this._rootMesh.rotationQuaternion!, undefined, Gizmo.PreserveScaling ? transformNode : undefined);\r\n } else {\r\n if (this._customRotationQuaternion) {\r\n this._rootMesh.rotationQuaternion!.copyFrom(this._customRotationQuaternion);\r\n } else {\r\n this._rootMesh.rotationQuaternion!.set(0, 0, 0, 1);\r\n }\r\n }\r\n\r\n // Scale\r\n if (this.updateScale) {\r\n const activeCamera = this.gizmoLayer.utilityLayerScene.activeCamera!;\r\n let cameraPosition = activeCamera.globalPosition;\r\n if ((activeCamera).devicePosition) {\r\n cameraPosition = (activeCamera).devicePosition;\r\n }\r\n this._rootMesh.position.subtractToRef(cameraPosition, TmpVectors.Vector3[0]);\r\n let scale = this.scaleRatio;\r\n if (activeCamera.mode == Camera.ORTHOGRAPHIC_CAMERA) {\r\n if (activeCamera.orthoTop && activeCamera.orthoBottom) {\r\n const orthoHeight = activeCamera.orthoTop - activeCamera.orthoBottom;\r\n scale *= orthoHeight;\r\n }\r\n } else {\r\n const camForward = activeCamera.getScene().useRightHandedSystem ? Vector3.RightHandedForwardReadOnly : Vector3.LeftHandedForwardReadOnly;\r\n const direction = activeCamera.getDirection(camForward);\r\n scale *= Vector3.Dot(TmpVectors.Vector3[0], direction);\r\n }\r\n this._rootMesh.scaling.setAll(scale);\r\n\r\n // Account for handedness, similar to Matrix.decompose\r\n if (effectiveNode._getWorldMatrixDeterminant() < 0 && !Gizmo.PreserveScaling) {\r\n this._rootMesh.scaling.y *= -1;\r\n }\r\n } else {\r\n this._rootMesh.scaling.setAll(this.scaleRatio);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Handle position/translation when using an attached node using pivot\r\n */\r\n protected _handlePivot() {\r\n const attachedNodeTransform = this._attachedNode as any;\r\n // check there is an active pivot for the TransformNode attached\r\n if (attachedNodeTransform.isUsingPivotMatrix && attachedNodeTransform.isUsingPivotMatrix() && attachedNodeTransform.position) {\r\n // When a TransformNode has an active pivot, even without parenting,\r\n // translation from the world matrix is different from TransformNode.position.\r\n // Pivot works like a virtual parent that's using the node orientation.\r\n // As the world matrix is transformed by the gizmo and then decomposed to TRS\r\n // its translation part must be set to the Node's position.\r\n attachedNodeTransform.getWorldMatrix().setTranslation(attachedNodeTransform.position);\r\n }\r\n }\r\n /**\r\n * computes the rotation/scaling/position of the transform once the Node world matrix has changed.\r\n */\r\n protected _matrixChanged() {\r\n if (!this._attachedNode) {\r\n return;\r\n }\r\n\r\n if ((this._attachedNode)._isCamera) {\r\n const camera = this._attachedNode as Camera;\r\n let worldMatrix;\r\n let worldMatrixUC;\r\n if (camera.parent) {\r\n const parentInv = TmpVectors.Matrix[1];\r\n camera.parent._worldMatrix.invertToRef(parentInv);\r\n this._attachedNode._worldMatrix.multiplyToRef(parentInv, TmpVectors.Matrix[0]);\r\n worldMatrix = TmpVectors.Matrix[0];\r\n } else {\r\n worldMatrix = this._attachedNode._worldMatrix;\r\n }\r\n\r\n if (camera.getScene().useRightHandedSystem) {\r\n // avoid desync with RH matrix computation. Otherwise, rotation of PI around Y axis happens each frame resulting in axis flipped because worldMatrix is computed as inverse of viewMatrix.\r\n this._rightHandtoLeftHandMatrix.multiplyToRef(worldMatrix, TmpVectors.Matrix[1]);\r\n worldMatrixUC = TmpVectors.Matrix[1];\r\n } else {\r\n worldMatrixUC = worldMatrix;\r\n }\r\n\r\n worldMatrixUC.decompose(TmpVectors.Vector3[1], TmpVectors.Quaternion[0], TmpVectors.Vector3[0]);\r\n\r\n const inheritsTargetCamera =\r\n this._attachedNode.getClassName() === \"FreeCamera\" ||\r\n this._attachedNode.getClassName() === \"FlyCamera\" ||\r\n this._attachedNode.getClassName() === \"ArcFollowCamera\" ||\r\n this._attachedNode.getClassName() === \"TargetCamera\" ||\r\n this._attachedNode.getClassName() === \"TouchCamera\" ||\r\n this._attachedNode.getClassName() === \"UniversalCamera\";\r\n\r\n if (inheritsTargetCamera) {\r\n const targetCamera = this._attachedNode as TargetCamera;\r\n targetCamera.rotation = TmpVectors.Quaternion[0].toEulerAngles();\r\n\r\n if (targetCamera.rotationQuaternion) {\r\n targetCamera.rotationQuaternion.copyFrom(TmpVectors.Quaternion[0]);\r\n targetCamera.rotationQuaternion.normalize();\r\n }\r\n }\r\n\r\n camera.position.copyFrom(TmpVectors.Vector3[0]);\r\n } else if (\r\n (this._attachedNode)._isMesh ||\r\n this._attachedNode.getClassName() === \"AbstractMesh\" ||\r\n this._attachedNode.getClassName() === \"TransformNode\" ||\r\n this._attachedNode.getClassName() === \"InstancedMesh\"\r\n ) {\r\n const transform = this._attachedNode as TransformNode;\r\n if (transform.parent) {\r\n const parentInv = TmpVectors.Matrix[0];\r\n const localMat = TmpVectors.Matrix[1];\r\n transform.parent.getWorldMatrix().invertToRef(parentInv);\r\n this._attachedNode.getWorldMatrix().multiplyToRef(parentInv, localMat);\r\n localMat.decompose(TmpVectors.Vector3[0], TmpVectors.Quaternion[0], transform.position, Gizmo.PreserveScaling ? transform : undefined);\r\n } else {\r\n this._attachedNode._worldMatrix.decompose(TmpVectors.Vector3[0], TmpVectors.Quaternion[0], transform.position, Gizmo.PreserveScaling ? transform : undefined);\r\n }\r\n TmpVectors.Vector3[0].scaleInPlace(1.0 / transform.scalingDeterminant);\r\n transform.scaling.copyFrom(TmpVectors.Vector3[0]);\r\n if (!transform.billboardMode) {\r\n if (transform.rotationQuaternion) {\r\n transform.rotationQuaternion.copyFrom(TmpVectors.Quaternion[0]);\r\n transform.rotationQuaternion.normalize();\r\n } else {\r\n transform.rotation = TmpVectors.Quaternion[0].toEulerAngles();\r\n }\r\n }\r\n } else if (this._attachedNode.getClassName() === \"Bone\") {\r\n const bone = this._attachedNode as Bone;\r\n const parent = bone.getParent();\r\n\r\n if (parent) {\r\n const invParent = TmpVectors.Matrix[0];\r\n const boneLocalMatrix = TmpVectors.Matrix[1];\r\n parent.getFinalMatrix().invertToRef(invParent);\r\n bone.getFinalMatrix().multiplyToRef(invParent, boneLocalMatrix);\r\n const lmat = bone.getLocalMatrix();\r\n lmat.copyFrom(boneLocalMatrix);\r\n } else {\r\n const lmat = bone.getLocalMatrix();\r\n lmat.copyFrom(bone.getFinalMatrix());\r\n }\r\n bone.markAsDirty();\r\n } else {\r\n const light = this._attachedNode as ShadowLight;\r\n if (light.getTypeID) {\r\n const type = light.getTypeID();\r\n if (type === Light.LIGHTTYPEID_DIRECTIONALLIGHT || type === Light.LIGHTTYPEID_SPOTLIGHT || type === Light.LIGHTTYPEID_POINTLIGHT) {\r\n const parent = light.parent;\r\n\r\n if (parent) {\r\n const invParent = TmpVectors.Matrix[0];\r\n const nodeLocalMatrix = TmpVectors.Matrix[1];\r\n parent.getWorldMatrix().invertToRef(invParent);\r\n light.getWorldMatrix().multiplyToRef(invParent, nodeLocalMatrix);\r\n nodeLocalMatrix.decompose(undefined, TmpVectors.Quaternion[0], TmpVectors.Vector3[0]);\r\n } else {\r\n this._attachedNode._worldMatrix.decompose(undefined, TmpVectors.Quaternion[0], TmpVectors.Vector3[0]);\r\n }\r\n // setter doesn't copy values. Need a new Vector3\r\n light.position = new Vector3(TmpVectors.Vector3[0].x, TmpVectors.Vector3[0].y, TmpVectors.Vector3[0].z);\r\n if (light.direction) {\r\n light.direction = new Vector3(light.direction.x, light.direction.y, light.direction.z);\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * refresh gizmo mesh material\r\n * @param gizmoMeshes\r\n * @param material material to apply\r\n */\r\n protected _setGizmoMeshMaterial(gizmoMeshes: Mesh[], material: StandardMaterial) {\r\n if (gizmoMeshes) {\r\n gizmoMeshes.forEach((m: Mesh) => {\r\n m.material = material;\r\n if ((m).color) {\r\n (m).color = material.diffuseColor;\r\n }\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Subscribes to pointer up, down, and hover events. Used for responsive gizmos.\r\n * @param gizmoLayer The utility layer the gizmo will be added to\r\n * @param gizmoAxisCache Gizmo axis definition used for reactive gizmo UI\r\n * @returns {Observer} pointerObserver\r\n */\r\n public static GizmoAxisPointerObserver(gizmoLayer: UtilityLayerRenderer, gizmoAxisCache: Map): Observer {\r\n let dragging = false;\r\n\r\n const pointerObserver = gizmoLayer.utilityLayerScene.onPointerObservable.add((pointerInfo) => {\r\n if (pointerInfo.pickInfo) {\r\n // On Hover Logic\r\n if (pointerInfo.type === PointerEventTypes.POINTERMOVE) {\r\n if (dragging) {\r\n return;\r\n }\r\n gizmoAxisCache.forEach((cache) => {\r\n if (cache.colliderMeshes && cache.gizmoMeshes) {\r\n const isHovered = cache.colliderMeshes?.indexOf(pointerInfo?.pickInfo?.pickedMesh as Mesh) != -1;\r\n const material = cache.dragBehavior.enabled ? (isHovered || cache.active ? cache.hoverMaterial : cache.material) : cache.disableMaterial;\r\n cache.gizmoMeshes.forEach((m: Mesh) => {\r\n m.material = material;\r\n if ((m as LinesMesh).color) {\r\n (m as LinesMesh).color = material.diffuseColor;\r\n }\r\n });\r\n }\r\n });\r\n }\r\n\r\n // On Mouse Down\r\n if (pointerInfo.type === PointerEventTypes.POINTERDOWN) {\r\n // If user Clicked Gizmo\r\n if (gizmoAxisCache.has(pointerInfo.pickInfo.pickedMesh?.parent as Mesh)) {\r\n dragging = true;\r\n const statusMap = gizmoAxisCache.get(pointerInfo.pickInfo.pickedMesh?.parent as Mesh);\r\n statusMap!.active = true;\r\n gizmoAxisCache.forEach((cache) => {\r\n const isHovered = cache.colliderMeshes?.indexOf(pointerInfo?.pickInfo?.pickedMesh as Mesh) != -1;\r\n const material = (isHovered || cache.active) && cache.dragBehavior.enabled ? cache.hoverMaterial : cache.disableMaterial;\r\n cache.gizmoMeshes.forEach((m: Mesh) => {\r\n m.material = material;\r\n if ((m as LinesMesh).color) {\r\n (m as LinesMesh).color = material.diffuseColor;\r\n }\r\n });\r\n });\r\n }\r\n }\r\n\r\n // On Mouse Up\r\n if (pointerInfo.type === PointerEventTypes.POINTERUP) {\r\n gizmoAxisCache.forEach((cache) => {\r\n cache.active = false;\r\n dragging = false;\r\n cache.gizmoMeshes.forEach((m: Mesh) => {\r\n m.material = cache.dragBehavior.enabled ? cache.material : cache.disableMaterial;\r\n if ((m as LinesMesh).color) {\r\n (m as LinesMesh).color = cache.material.diffuseColor;\r\n }\r\n });\r\n });\r\n }\r\n }\r\n });\r\n\r\n return pointerObserver!;\r\n }\r\n\r\n /**\r\n * Disposes of the gizmo\r\n */\r\n public dispose() {\r\n this._rootMesh.dispose();\r\n if (this._beforeRenderObserver) {\r\n this.gizmoLayer.utilityLayerScene.onBeforeRenderObservable.remove(this._beforeRenderObserver);\r\n }\r\n }\r\n}\r\n","import { ArcRotateCamera } from '@babylonjs/core/Cameras/arcRotateCamera';\r\nimport { Camera } from '@babylonjs/core/Cameras/camera';\r\nimport { Gizmo } from '@babylonjs/core/Gizmos/gizmo';\r\nimport { Matrix, Quaternion, Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { Observable } from '@babylonjs/core/Misc/observable';\r\nimport { WebXRCamera } from '@babylonjs/core/XR/webXRCamera';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { Kb3dObject } from '@models/classes/kb3d-object';\r\nimport { SpaceNode } from '@models/classes/meshes/space-node';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { getCameraPositionWithOffset } from '@view/helpers/math-helper';\r\n/**\r\n * Renders gizmos on top of an existing scene which provide controls for position, rotation, etc.\r\n */\r\n\r\nexport const AUTO_MESH_NAME = '_autoGizmoMesh';\r\nexport abstract class KbGizmo extends Gizmo {\r\n updateGizmoPosition() {\r\n this._update();\r\n }\r\n\r\n attachTo(node: Kb3dObject | KbVector) {}\r\n\r\n /** Fires an event when any of it's sub gizmos are dragged */\r\n onDragStartObservable: Observable;\r\n /** Fires an event while any of it's sub gizmos are dragged */\r\n onDragObservable: Observable;\r\n /** Fires an event when any of it's sub gizmos are released from dragging */\r\n onDragEndObservable: Observable;\r\n\r\n get attachedMesh() {\r\n return super.attachedMesh;\r\n }\r\n set attachedMesh(m: AbstractMesh | null) {\r\n if (m) {\r\n super.attachedMesh = m;\r\n }\r\n }\r\n\r\n get visibility() {\r\n return this._rootMesh.visibility;\r\n }\r\n set visibility(visibility: number) {\r\n this._rootMesh.setEnabled(visibility === 1);\r\n }\r\n\r\n protected attachMeshAtNodeSpace(scene: Scene, target: SpaceNode | KbVector, positionSpaceTransform?: Matrix) {\r\n if (!this.attachedMesh) {\r\n this.attachedMesh = new Mesh(AUTO_MESH_NAME, scene);\r\n }\r\n let translation: Vector3;\r\n const rotation = Quaternion.Identity();\r\n const scale = Vector3.One();\r\n\r\n translation = target instanceof SpaceNode ? target.position.toVec3() : target.toVec3();\r\n if (positionSpaceTransform) {\r\n translation = Vector3.TransformCoordinates(translation, positionSpaceTransform);\r\n positionSpaceTransform.decompose(scale, rotation);\r\n }\r\n\r\n this.attachedMesh.position = translation;\r\n this.attachedMesh.rotationQuaternion = rotation;\r\n this.attachedMesh.scaling = scale;\r\n\r\n return this.attachedMesh;\r\n }\r\n\r\n protected _update() {\r\n if (this.attachedMesh) {\r\n const _tempVector = new Vector3();\r\n const _tmpMatrix = new Matrix();\r\n if (this.updateGizmoRotationToMatchAttachedMesh) {\r\n if (!this._rootMesh.rotationQuaternion) {\r\n this._rootMesh.rotationQuaternion = Quaternion.RotationYawPitchRoll(\r\n this._rootMesh.rotation.y,\r\n this._rootMesh.rotation.x,\r\n this._rootMesh.rotation.z\r\n );\r\n }\r\n\r\n // Remove scaling before getting rotation matrix to get rotation matrix unmodified by scale\r\n _tempVector.copyFrom(this.attachedMesh.scaling);\r\n if (this.attachedMesh.scaling.x < 0) {\r\n this.attachedMesh.scaling.x *= -1;\r\n }\r\n if (this.attachedMesh.scaling.y < 0) {\r\n this.attachedMesh.scaling.y *= -1;\r\n }\r\n if (this.attachedMesh.scaling.z < 0) {\r\n this.attachedMesh.scaling.z *= -1;\r\n }\r\n this.attachedMesh.computeWorldMatrix().getRotationMatrixToRef(_tmpMatrix);\r\n this.attachedMesh.scaling.copyFrom(_tempVector);\r\n this.attachedMesh.computeWorldMatrix();\r\n Quaternion.FromRotationMatrixToRef(_tmpMatrix, this._rootMesh.rotationQuaternion);\r\n }\r\n // else if (this._rootMesh.rotationQuaternion) {\r\n // this._rootMesh.rotationQuaternion.set(0, 0, 0, 1);\r\n // }\r\n if (this.updateGizmoPositionToMatchAttachedMesh) {\r\n this._rootMesh.position.copyFrom(this.attachedMesh.absolutePosition);\r\n }\r\n if (this.updateScale && this.gizmoLayer.utilityLayerScene.activeCamera && this.attachedMesh) {\r\n let dist: number;\r\n const camera = this.gizmoLayer.utilityLayerScene.activeCamera;\r\n\r\n if (camera.mode === Camera.ORTHOGRAPHIC_CAMERA) {\r\n dist = ((camera.orthoRight || 0) - (camera.orthoLeft || 0)) / 2;\r\n } else {\r\n let cameraPosition: Vector3;\r\n if (camera instanceof WebXRCamera && camera.globalPosition) {\r\n cameraPosition = camera.globalPosition;\r\n } else if (camera instanceof ArcRotateCamera) {\r\n cameraPosition = getCameraPositionWithOffset(camera);\r\n } else {\r\n cameraPosition = camera.globalPosition;\r\n }\r\n\r\n this._rootMesh.position.subtractToRef(cameraPosition, _tempVector);\r\n dist = _tempVector.length() * this.scaleRatio;\r\n }\r\n this._rootMesh.scaling.set(dist, dist, dist);\r\n }\r\n\r\n const pivot = this.attachedMesh.getPivotPoint();\r\n //let absolutePosition = this.attachedMesh.absolutePosition;\r\n\r\n // Absolute position is getting skewed because of the pivot point; need to get position without the rotation\r\n // if (pivot.length() > 0 && this.attachedMesh.rotationQuaternion) {\r\n // var m = new Matrix();\r\n // this.attachedMesh.rotationQuaternion.toRotationMatrix(m);\r\n // absolutePosition = Vector3.TransformCoordinates(absolutePosition, m)\r\n // }\r\n // if (pivot.length() > 0) {\r\n // absolutePosition = Vector3.Zero();\r\n // }\r\n\r\n this._rootMesh.position.copyFrom(Vector3.TransformCoordinates(pivot, this.attachedMesh.getWorldMatrix()));\r\n }\r\n }\r\n}\r\n","import { PointerDragBehavior } from '@babylonjs/core/Behaviors/Meshes/pointerDragBehavior';\r\nimport { PointerInfo } from '@babylonjs/core/Events/pointerEvents';\r\nimport { StandardMaterial } from '@babylonjs/core/Materials/standardMaterial';\r\nimport { Color3, Matrix, Quaternion, Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { LinesMesh } from '@babylonjs/core/Meshes/linesMesh';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { Observable, Observer } from '@babylonjs/core/Misc/observable';\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { Nullable } from '@babylonjs/core/types';\r\nimport { KbGizmo } from './gizmo';\r\n\r\n/**\r\n * Single plane rotation gizmo\r\n */\r\nexport class KbPlaneRotationGizmo extends KbGizmo {\r\n /**\r\n * Drag behavior responsible for the gizmos dragging interactions\r\n */\r\n public dragBehavior: PointerDragBehavior;\r\n private _pointerObserver: Nullable> = null;\r\n private rotationMesh: AbstractMesh;\r\n\r\n /**\r\n * Rotation distance in radians that the gizmo will snap to (Default: 0)\r\n */\r\n public snapDistance = 0;\r\n /**\r\n * Event that fires each time the gizmo snaps to a new location.\r\n * * snapDistance is the the change in distance\r\n */\r\n public onSnapObservable = new Observable<{ snapDistance: number }>();\r\n public updateObservable = new Observable<{ delta: Quaternion; deltaAngle: number }>();\r\n\r\n /**\r\n * Creates a PlaneRotationGizmo\r\n * @param gizmoLayer The utility layer the gizmo will be added to\r\n * @param planeNormal The normal of the plane which the gizmo will be able to rotate on\r\n * @param color The color of the gizmo\r\n * @param tessellation Amount of tessellation to be used when creating rotation circles\r\n */\r\n constructor(\r\n planeNormal: Vector3,\r\n color: Color3 = Color3.Gray(),\r\n gizmoLayer: UtilityLayerRenderer = UtilityLayerRenderer.DefaultUtilityLayer,\r\n tessellation = 32\r\n ) {\r\n super(gizmoLayer);\r\n\r\n // Create Material\r\n const coloredMaterial = new StandardMaterial('', gizmoLayer.utilityLayerScene);\r\n coloredMaterial.emissiveColor = color;\r\n // coloredMaterial.diffuseColor = color;\r\n // coloredMaterial.specularColor = color.subtract(new Color3(0.1, 0.1, 0.1));\r\n\r\n const hoverMaterial = new StandardMaterial('', gizmoLayer.utilityLayerScene);\r\n hoverMaterial.emissiveColor = color.add(new Color3(0.3, 0.3, 0.3));\r\n // hoverMaterial.diffuseColor = color.add(new Color3(0.3, 0.3, 0.3));\r\n\r\n // Build mesh on root node\r\n const parentMesh = new AbstractMesh('', gizmoLayer.utilityLayerScene);\r\n\r\n const drag = Mesh.CreateTorus('', 0.6, 0.045, tessellation, gizmoLayer.utilityLayerScene);\r\n drag.visibility = 0;\r\n const rotationMesh = Mesh.CreateTorus('', 0.6, 0.01, tessellation, gizmoLayer.utilityLayerScene);\r\n rotationMesh.material = coloredMaterial;\r\n\r\n // Position arrow pointing in its drag axis\r\n rotationMesh.rotation.x = Math.PI / 2;\r\n drag.rotation.x = Math.PI / 2;\r\n parentMesh.addChild(rotationMesh);\r\n parentMesh.addChild(drag);\r\n parentMesh.lookAt(this._rootMesh.position.add(planeNormal));\r\n this.rotationMesh = parentMesh;\r\n\r\n this._rootMesh.addChild(parentMesh);\r\n parentMesh.scaling.scaleInPlace(1 / 3);\r\n // Add drag behavior to handle events when the gizmo is dragged\r\n this.dragBehavior = new PointerDragBehavior({ dragPlaneNormal: planeNormal });\r\n this.dragBehavior.moveAttached = false;\r\n this.dragBehavior.maxDragAngle = (Math.PI * 9) / 20;\r\n this.dragBehavior._useAlternatePickedPointAboveMaxDragAngle = true;\r\n this._rootMesh.addBehavior(this.dragBehavior);\r\n\r\n const lastDragPosition = new Vector3();\r\n\r\n this.dragBehavior.onDragStartObservable.add(e => {\r\n if (this.attachedMesh) {\r\n lastDragPosition.copyFrom(e.dragPlanePoint);\r\n }\r\n });\r\n\r\n const rotationMatrix = new Matrix();\r\n const planeNormalTowardsCamera = new Vector3();\r\n let localPlaneNormalTowardsCamera = new Vector3();\r\n\r\n const tmpSnapEvent = { snapDistance: 0 };\r\n let currentSnapDragDistance = 0;\r\n const tmpMatrix = new Matrix();\r\n const tmpVector = new Vector3();\r\n const amountToRotate = new Quaternion();\r\n this.dragBehavior.onDragObservable.add(event => {\r\n if (this.attachedMesh) {\r\n if (!this.attachedMesh.rotationQuaternion) {\r\n this.attachedMesh.rotationQuaternion = Quaternion.RotationYawPitchRoll(\r\n this.attachedMesh.rotation.y,\r\n this.attachedMesh.rotation.x,\r\n this.attachedMesh.rotation.z\r\n );\r\n }\r\n\r\n // Remove parent priort to rotating\r\n const attachedMeshParent = this.attachedMesh.parent;\r\n if (attachedMeshParent) {\r\n this.attachedMesh.setParent(null);\r\n }\r\n\r\n // Calc angle over full 360 degree (https://stackoverflow.com/questions/43493711/the-angle-between-two-3d-vectors-with-a-result-range-0-360)\r\n const pivot = this.attachedMesh.getPivotPoint();\r\n const attachedMeshPosition = this.attachedMesh.absolutePosition.add(pivot);\r\n const newVector = event.dragPlanePoint.subtract(attachedMeshPosition).normalize();\r\n const originalVector = lastDragPosition.subtract(attachedMeshPosition).normalize();\r\n const cross = Vector3.Cross(newVector, originalVector);\r\n const dot = Vector3.Dot(newVector, originalVector);\r\n let angle = Math.atan2(cross.length(), dot);\r\n planeNormalTowardsCamera.copyFrom(planeNormal);\r\n localPlaneNormalTowardsCamera.copyFrom(planeNormal);\r\n if (this.updateGizmoRotationToMatchAttachedMesh) {\r\n this.attachedMesh.rotationQuaternion.toRotationMatrix(rotationMatrix);\r\n localPlaneNormalTowardsCamera = Vector3.TransformCoordinates(\r\n planeNormalTowardsCamera,\r\n rotationMatrix\r\n );\r\n }\r\n // Flip up vector depending on which side the camera is on\r\n if (gizmoLayer.utilityLayerScene.activeCamera) {\r\n const camVec = gizmoLayer.utilityLayerScene.activeCamera.position.subtract(\r\n this.attachedMesh.position\r\n );\r\n if (Vector3.Dot(camVec, localPlaneNormalTowardsCamera) > 0) {\r\n planeNormalTowardsCamera.scaleInPlace(-1);\r\n localPlaneNormalTowardsCamera.scaleInPlace(-1);\r\n }\r\n }\r\n const halfCircleSide = Vector3.Dot(localPlaneNormalTowardsCamera, cross) > 0.0;\r\n if (halfCircleSide) {\r\n angle = -angle;\r\n }\r\n\r\n // Snapping logic\r\n let snapped = false;\r\n if (this.snapDistance != 0) {\r\n currentSnapDragDistance += angle;\r\n if (Math.abs(currentSnapDragDistance) > this.snapDistance) {\r\n let dragSteps = Math.floor(Math.abs(currentSnapDragDistance) / this.snapDistance);\r\n if (currentSnapDragDistance < 0) {\r\n dragSteps *= -1;\r\n }\r\n currentSnapDragDistance = currentSnapDragDistance % this.snapDistance;\r\n angle = this.snapDistance * dragSteps;\r\n snapped = true;\r\n } else {\r\n angle = 0;\r\n }\r\n }\r\n\r\n // If the mesh has a parent, convert needed world rotation to local rotation\r\n tmpMatrix.reset();\r\n if (this.attachedMesh.parent) {\r\n this.attachedMesh.parent.computeWorldMatrix().invertToRef(tmpMatrix);\r\n tmpMatrix.getRotationMatrixToRef(tmpMatrix);\r\n Vector3.TransformCoordinatesToRef(planeNormalTowardsCamera, tmpMatrix, planeNormalTowardsCamera);\r\n }\r\n\r\n // Convert angle and axis to quaternion (http://www.euclideanspace.com/maths/geometry/rotations/conversions/angleToQuaternion/index.htm)\r\n const quaternionCoefficient = Math.sin(angle / 2);\r\n amountToRotate.set(\r\n planeNormalTowardsCamera.x * quaternionCoefficient,\r\n planeNormalTowardsCamera.y * quaternionCoefficient,\r\n planeNormalTowardsCamera.z * quaternionCoefficient,\r\n Math.cos(angle / 2)\r\n );\r\n\r\n // If the meshes local scale is inverted (eg. loaded gltf file parent with z scale of -1) the rotation needs to be inverted on the y axis\r\n if (tmpMatrix.determinant() > 0) {\r\n amountToRotate.toEulerAnglesToRef(tmpVector);\r\n Quaternion.RotationYawPitchRollToRef(tmpVector.y, -tmpVector.x, -tmpVector.z, amountToRotate);\r\n }\r\n\r\n if (this.updateGizmoRotationToMatchAttachedMesh) {\r\n // Rotate selected mesh quaternion over fixed axis\r\n this.attachedMesh.rotationQuaternion.multiplyToRef(\r\n amountToRotate,\r\n this.attachedMesh.rotationQuaternion\r\n );\r\n } else {\r\n // Rotate selected mesh quaternion over rotated axis\r\n amountToRotate.multiplyToRef(\r\n this.attachedMesh.rotationQuaternion,\r\n this.attachedMesh.rotationQuaternion\r\n );\r\n }\r\n this.updateObservable.notifyObservers({ delta: amountToRotate, deltaAngle: angle });\r\n\r\n lastDragPosition.copyFrom(event.dragPlanePoint);\r\n if (snapped) {\r\n tmpSnapEvent.snapDistance = angle;\r\n this.onSnapObservable.notifyObservers(tmpSnapEvent);\r\n }\r\n\r\n // Restore parent\r\n if (attachedMeshParent) {\r\n this.attachedMesh.setParent(attachedMeshParent);\r\n }\r\n }\r\n });\r\n\r\n this._pointerObserver = gizmoLayer.utilityLayerScene.onPointerObservable.add(pointerInfo => {\r\n if (this._customMeshSet) {\r\n return;\r\n }\r\n const isHovered =\r\n pointerInfo.pickInfo &&\r\n this._rootMesh.getChildMeshes().indexOf(pointerInfo.pickInfo.pickedMesh) != -1;\r\n const material = isHovered ? hoverMaterial : coloredMaterial;\r\n this._rootMesh.getChildMeshes().forEach(m => {\r\n m.material = material;\r\n if ((m).color) {\r\n (m).color = material.diffuseColor;\r\n }\r\n });\r\n });\r\n\r\n // var light = gizmoLayer._getSharedGizmoLight();\r\n // light.includedOnlyMeshes = light.includedOnlyMeshes.concat(this._rootMesh.getChildMeshes(false));\r\n }\r\n\r\n protected _attachedMeshChanged(value: Nullable) {\r\n if (this.dragBehavior) {\r\n this.dragBehavior.enabled = value ? true : false;\r\n }\r\n // if (value) {\r\n // this.rotationMesh.setPivotPoint(Vector3.Zero(), Space.LOCAL);\r\n // }\r\n }\r\n\r\n /**\r\n * Disposes of the gizmo\r\n */\r\n public dispose() {\r\n this.onSnapObservable.clear();\r\n this.gizmoLayer.utilityLayerScene.onPointerObservable.remove(this._pointerObserver);\r\n this.dragBehavior.detach();\r\n super.dispose();\r\n }\r\n}\r\n","import { Color3, Matrix, Space, Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { Logger } from '@babylonjs/core/Misc/logger';\r\nimport { Observable } from '@babylonjs/core/Misc/observable';\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { Nullable } from '@babylonjs/core/types';\r\nimport { SpaceNode } from '@models/classes/meshes/space-node';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { KbAxisMorphGizmo } from './axis-morph-gizmo';\r\nimport { KbGizmo } from './gizmo';\r\nimport { KbPlaneRotationGizmo } from './plane-rotation-gizmo';\r\n\r\nconst tessellation = 32;\r\n\r\n/**\r\n * TODO\r\n * Copy in plane rotation gizmo code here and change it to use a sphere and click projection on the sphere\r\n * Convert it to return an axis\r\n * Edit code in here to return an axis and angle\r\n * update gizmo tools code to hook to this gizmo\r\n * remove the vector option from the original rotation gizmo (now the euler rotation gizmo), simplify code\r\n */\r\n\r\n/**\r\n * Gizmo that enables rotating a mesh along 3 axis\r\n */\r\nexport class KbAxisAngleGizmo extends KbGizmo {\r\n public targetAxis = new Vector3();\r\n /**\r\n * Internal gizmo used for interactions on the angle around the axis\r\n */\r\n public angleGizmo: KbPlaneRotationGizmo | undefined;\r\n\r\n /**\r\n * Internal gizmo used for interactions modifying the axis\r\n */\r\n public axisGizmo: KbAxisMorphGizmo;\r\n\r\n /** Fires an event when any of it's sub gizmos are dragged */\r\n public onDragStartObservable = new Observable();\r\n /** Fires an event when any of it's sub gizmos are released from dragging */\r\n public onDragObservable = new Observable<{ deltaAxis: Vector3; deltaAngle: number }>();\r\n public onDragEndObservable = new Observable<{ deltaAxis: Vector3; deltaAngle: number }>();\r\n\r\n public gizmoOffset: KbVector | undefined;\r\n public placementOffset: KbVector | undefined;\r\n private _positionSpace: TransformNode | undefined;\r\n\r\n attachTo(target: KbVector, positionSpace?: SpaceNode, placementOffset?: KbVector) {\r\n if (positionSpace) {\r\n this._positionSpace = this.kbViewer.getRendererNode(positionSpace);\r\n }\r\n\r\n this._updateGizmoPosition(target, placementOffset);\r\n }\r\n updateGizmoPosition(changeTarget?: KbVector, placementOffset?: KbVector) {\r\n // Need to wait for the next frame because this can be called before the babylon object is actually updated\r\n this.kbViewer.onNextFrame(() => {\r\n this._updateGizmoPosition(changeTarget, placementOffset);\r\n });\r\n }\r\n\r\n private _updateGizmoPosition(changeTarget?: KbVector, placementOffset?: KbVector) {\r\n // Maintain the placement offset if it is not provided later, so this can be called from other contexts to update based on external changes\r\n if (!placementOffset && this.placementOffset) {\r\n placementOffset = this.placementOffset;\r\n } else if (placementOffset) {\r\n this.placementOffset = placementOffset;\r\n }\r\n\r\n const positionSpaceTransform = this._positionSpace?.computeWorldMatrix(true);\r\n if (changeTarget) {\r\n changeTarget.toVec3(this.targetAxis);\r\n\r\n // If target is a vector, then we want to rotate it as a direction, so set the gizmo position to the origin to calculations are done from there\r\n // But store the position as the offset because we want the gizmo itself to be visually placed on the position space origin\r\n this.attachMeshAtNodeSpace(\r\n this.gizmoLayer.utilityLayerScene,\r\n placementOffset || KbVector.Zero(),\r\n positionSpaceTransform\r\n );\r\n\r\n if (this.attachedMesh) {\r\n this.gizmoOffset = KbVector.FromVec3(this.attachedMesh.position);\r\n this.attachedMesh.position = Vector3.Zero();\r\n this.attachedMesh.computeWorldMatrix(true);\r\n }\r\n const targetAxisTransformed = this.targetAxis.clone();\r\n if (positionSpaceTransform) {\r\n Vector3.TransformCoordinatesToRef(\r\n this.targetAxis,\r\n positionSpaceTransform.getRotationMatrix(),\r\n targetAxisTransformed\r\n );\r\n }\r\n this.axisGizmo.direction.copyFrom(targetAxisTransformed);\r\n }\r\n if (this.gizmoOffset && this.attachedMesh) {\r\n this.attachedMesh.position.copyFrom(this.gizmoOffset.toVec3());\r\n this.axisGizmo.gizmoPosition.copyFrom(this.gizmoOffset.toVec3());\r\n // if (placementOffset) {\r\n // this.axisGizmo.gizmoPosition.addInPlace(placementOffset.toVec3());\r\n // }\r\n }\r\n // if (positionSpaceTransform) {\r\n // Vector3.TransformCoordinatesToRef(this.axisGizmo.gizmoPosition, positionSpaceTransform, this.axisGizmo.gizmoPosition);\r\n // }\r\n super.updateGizmoPosition();\r\n }\r\n\r\n public get attachedMesh() {\r\n return super.attachedMesh;\r\n }\r\n public set attachedMesh(mesh: Nullable) {\r\n super.attachedMesh = mesh;\r\n if (mesh) {\r\n const angleMesh = new Mesh('', this.gizmoLayer.utilityLayerScene);\r\n mesh.addChild(angleMesh);\r\n if (this.angleGizmo) {\r\n this.angleGizmo.attachedMesh = angleMesh;\r\n }\r\n angleMesh.lookAt(this.targetAxis);\r\n angleMesh.setPivotPoint(mesh.getPivotPoint(), Space.WORLD);\r\n } else if (this.angleGizmo) {\r\n this.angleGizmo.attachedMesh = null;\r\n }\r\n }\r\n\r\n private axisChange: Vector3;\r\n private angleChange: number;\r\n /**\r\n * Creates a RotationGizmo\r\n * @param gizmoLayer The utility layer the gizmo will be added to\r\n * @param tessellation Amount of tessellation to be used when creating rotation circles\r\n */\r\n constructor(\r\n private kbViewer: KbViewer,\r\n gizmoLayer: UtilityLayerRenderer = UtilityLayerRenderer.DefaultUtilityLayer,\r\n disableAngle = false\r\n ) {\r\n super(gizmoLayer);\r\n this.updateGizmoRotationToMatchAttachedMesh = true;\r\n this.axisGizmo = new KbAxisMorphGizmo(Color3.Green().scale(0.5), gizmoLayer, tessellation);\r\n if (!disableAngle) {\r\n this.angleGizmo = new KbPlaneRotationGizmo(\r\n new Vector3(0, 0, 1),\r\n Color3.Green().scale(0.5),\r\n gizmoLayer,\r\n tessellation\r\n );\r\n this.angleGizmo.updateGizmoPositionToMatchAttachedMesh = true;\r\n }\r\n\r\n const zeroVector = Vector3.Zero();\r\n const tempVector = Vector3.Zero();\r\n const tempVector2 = Vector3.Zero();\r\n let positionSpaceTransform: Matrix | undefined;\r\n this.axisChange = Vector3.Zero();\r\n this.angleChange = 0;\r\n\r\n // Relay drag events\r\n [this.axisGizmo, this.angleGizmo].forEach(gizmo => {\r\n if (gizmo) {\r\n gizmo.dragBehavior.onDragStartObservable.add(() => {\r\n this.onDragStartObservable.notifyObservers();\r\n this.axisChange = Vector3.Zero();\r\n this.angleChange = 0;\r\n if (!positionSpaceTransform && this._positionSpace) {\r\n positionSpaceTransform = this._positionSpace\r\n .computeWorldMatrix(true)\r\n .getRotationMatrix()\r\n .clone()\r\n .invert();\r\n }\r\n });\r\n gizmo.dragBehavior.onDragEndObservable.add(() => {\r\n this.onDragEndObservable.notifyObservers({\r\n deltaAxis: this.axisChange,\r\n deltaAngle: this.angleChange,\r\n });\r\n });\r\n }\r\n });\r\n this.axisGizmo.updateObservable.add(event => {\r\n tempVector.copyFrom(this.targetAxis);\r\n // if (positionSpaceTransform) {\r\n // Vector3.TransformCoordinatesToRef(tempVector, positionSpaceTransform, tempVector);\r\n // }\r\n\r\n tempVector.rotateByQuaternionToRef(event.delta, tempVector2);\r\n // if (positionSpaceTransform) {\r\n // Vector3.TransformCoordinatesToRef(tempVector2, positionSpaceTransform.clone().invert(), tempVector2);\r\n // }\r\n\r\n tempVector2.subtractToRef(this.targetAxis, tempVector2);\r\n\r\n this.targetAxis.addInPlace(tempVector2);\r\n this.axisChange.addInPlace(tempVector2);\r\n this.onDragObservable.notifyObservers({ deltaAxis: tempVector2, deltaAngle: 0 });\r\n\r\n if (this.angleGizmo && this.angleGizmo.attachedMesh) {\r\n this.angleGizmo.attachedMesh.lookAt(\r\n positionSpaceTransform\r\n ? Vector3.TransformCoordinates(this.targetAxis, positionSpaceTransform)\r\n : this.targetAxis\r\n );\r\n }\r\n });\r\n if (this.angleGizmo) {\r\n this.angleGizmo.updateObservable.add(event => {\r\n this.angleChange -= event.deltaAngle;\r\n this.onDragObservable.notifyObservers({ deltaAxis: zeroVector, deltaAngle: -event.deltaAngle });\r\n });\r\n }\r\n\r\n this.attachedMesh = null;\r\n }\r\n\r\n /**\r\n * Drag distance in babylon units that the gizmo will snap to when dragged (Default: 0)\r\n */\r\n public set snapDistance(value: number) {\r\n if (this.axisGizmo) {\r\n this.axisGizmo.snapDistance = value;\r\n }\r\n if (this.angleGizmo) {\r\n this.angleGizmo.snapDistance = value;\r\n }\r\n }\r\n public get snapDistance() {\r\n return this.axisGizmo.snapDistance;\r\n }\r\n\r\n /**\r\n * Ratio for the scale of the gizmo (Default: 1)\r\n */\r\n public set scaleRatio(value: number) {\r\n if (this.axisGizmo) {\r\n this.axisGizmo.scaleRatio = value;\r\n }\r\n if (this.angleGizmo) {\r\n this.angleGizmo.scaleRatio = value;\r\n }\r\n }\r\n public get scaleRatio() {\r\n return this.axisGizmo.scaleRatio;\r\n }\r\n\r\n /**\r\n * Disposes of the gizmo\r\n */\r\n public dispose() {\r\n if (this.axisGizmo) {\r\n this.axisGizmo.dispose();\r\n }\r\n if (this.angleGizmo) {\r\n this.angleGizmo.dispose();\r\n }\r\n this.onDragStartObservable.clear();\r\n this.onDragEndObservable.clear();\r\n }\r\n\r\n /**\r\n * CustomMeshes are not supported by this gizmo\r\n * @param mesh The mesh to replace the default mesh of the gizmo\r\n */\r\n public setCustomMesh(mesh: Mesh) {\r\n Logger.Error(\r\n 'Custom meshes are not supported on this gizmo, please set the custom meshes on the gizmos contained within this one (gizmo.xGizmo, gizmo.angleGizmo, gizmo.zGizmo)'\r\n );\r\n }\r\n}\r\n","import { Vector3 } from '@babylonjs/core/Maths';\r\nimport { VertexBuffer } from '@babylonjs/core/Meshes/buffer';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { getByDynamicId } from '@models/classes/kb3d-manager';\r\nimport { Kb3dObject } from '@models/classes/kb3d-object';\r\nimport { MeshNode } from '@models/classes/meshes/mesh-node';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { take } from 'rxjs/operators';\r\nimport { BoundingBoxCache } from './bounding-box-cache';\r\n\r\nexport interface IMeshInfo {\r\n faceCount: number;\r\n faceNeighbors: number[][];\r\n facePositions: number[];\r\n faceNormals: number[];\r\n}\r\n\r\nconst meshFaceInfo = new Map();\r\n\r\nexport class MeshFaceEngine {\r\n private bbCache: BoundingBoxCache;\r\n\r\n constructor(private kbViewer: KbViewer) {\r\n this.bbCache = kbViewer.bbCache;\r\n }\r\n\r\n initialize(node: MeshNode | Mesh) {\r\n let meshNode: MeshNode;\r\n let mesh: Mesh;\r\n if (node instanceof Kb3dObject) {\r\n mesh = this.kbViewer.getRendererNode(node) as Mesh;\r\n meshNode = node;\r\n } else {\r\n mesh = node;\r\n meshNode = getByDynamicId(node.id) as MeshNode;\r\n }\r\n if (mesh instanceof Mesh && mesh.geometry) {\r\n this.bbCache\r\n .watchForChanges(meshNode)\r\n .pipe(take(1))\r\n .subscribe(() => {\r\n meshFaceInfo.delete(meshNode._dynamicId);\r\n });\r\n\r\n return this.updateNodeGeometry(mesh);\r\n }\r\n }\r\n\r\n getMeshFaces(node: MeshNode | Mesh) {\r\n let result = meshFaceInfo.get(node instanceof Kb3dObject ? node._dynamicId : node.id);\r\n if (!result) {\r\n result = this.initialize(node);\r\n }\r\n return result;\r\n }\r\n\r\n /**\r\n * @param selection array of selected faces to expand from\r\n * @param limitAngle angle to limit selection to. If undefined, function will expand by 1 polygon\r\n */\r\n expandSelection(selection: number[], node: MeshNode | Mesh, limitAngle?: number) {\r\n const faceInfo = this.getMeshFaces(node);\r\n const faceNormals = faceInfo!.faceNormals;\r\n\r\n const selectedFaceIds: boolean[] = [];\r\n const newSelection: number[] = [];\r\n if (limitAngle !== undefined) {\r\n limitAngle = (limitAngle * Math.PI) / 180;\r\n }\r\n const v1 = new Vector3(),\r\n v2 = new Vector3();\r\n\r\n if (!faceInfo) {\r\n return selection;\r\n }\r\n\r\n for (let i = 0; i < selection.length; i++) {\r\n selectedFaceIds[selection[i]] = true;\r\n }\r\n for (let i = 0; i < selection.length; i++) {\r\n const faceId = selection[i];\r\n v1.copyFromFloats(faceNormals[faceId * 3], faceNormals[faceId * 3 + 1], faceNormals[faceId * 3 + 2]);\r\n recursiveExpandSelection(faceId, 1, v1);\r\n }\r\n for (const faceId in selectedFaceIds) {\r\n newSelection.push(parseInt(faceId, 10));\r\n }\r\n\r\n return newSelection;\r\n\r\n function recursiveExpandSelection(faceId: number, iterCount: number, normal: Vector3) {\r\n const neighbors = faceInfo!.faceNeighbors[faceId];\r\n if (limitAngle === undefined && iterCount <= 0) {\r\n return;\r\n }\r\n\r\n for (let i = 0; i < neighbors.length; i++) {\r\n const adjacentFaceId = neighbors[i];\r\n if (!selectedFaceIds[adjacentFaceId]) {\r\n if (limitAngle !== undefined) {\r\n v2.copyFromFloats(\r\n faceNormals[adjacentFaceId * 3],\r\n faceNormals[adjacentFaceId * 3 + 1],\r\n faceNormals[adjacentFaceId * 3 + 2]\r\n );\r\n\r\n if (\r\n Math.abs(Math.acos(Math.round(Vector3.Dot(normal, v2) * 1000000) / 1000000)) <= limitAngle\r\n ) {\r\n // deal with FP Errors\r\n selectedFaceIds[adjacentFaceId] = true;\r\n recursiveExpandSelection(adjacentFaceId, iterCount - 1, normal);\r\n }\r\n } else {\r\n selectedFaceIds[adjacentFaceId] = true;\r\n recursiveExpandSelection(adjacentFaceId, iterCount - 1, normal);\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n private updateNodeGeometry(mesh: Mesh) {\r\n if (mesh.geometry) {\r\n const indices = mesh.geometry.getIndices();\r\n const positions = mesh.geometry.getVerticesData(VertexBuffer.PositionKind);\r\n\r\n const v1 = new Vector3();\r\n const v2 = new Vector3();\r\n const v3 = new Vector3();\r\n const v4 = new Vector3();\r\n\r\n if (indices && positions) {\r\n const facesByEdges: { [key: string]: number[] } = {};\r\n /** Sets of three faces by index adjacent to each face by index */\r\n const faceNeighbors: number[][] = [];\r\n /** Sets of xyz scalars making up the face normals */\r\n const faceNormals: number[] = [];\r\n const facePositions: number[] = [];\r\n\r\n const trianglesCount = indices.length / 3;\r\n for (let triId = 0; triId < trianglesCount; triId++) {\r\n const triOffset = triId * 3;\r\n // Get the triangle position vertices\r\n v1.copyFromFloats(\r\n positions[indices[triOffset] * 3],\r\n positions[indices[triOffset] * 3 + 1],\r\n positions[indices[triOffset] * 3 + 2]\r\n );\r\n v2.copyFromFloats(\r\n positions[indices[triOffset + 1] * 3],\r\n positions[indices[triOffset + 1] * 3 + 1],\r\n positions[indices[triOffset + 1] * 3 + 2]\r\n );\r\n v3.copyFromFloats(\r\n positions[indices[triOffset + 2] * 3],\r\n positions[indices[triOffset + 2] * 3 + 1],\r\n positions[indices[triOffset + 2] * 3 + 2]\r\n );\r\n\r\n // Calculate the centroid of each face\r\n v4.setAll(0);\r\n v4.addInPlace(v1).addInPlace(v2).addInPlace(v3);\r\n v4.scaleInPlace(1 / 3);\r\n facePositions.push(v4.x, v4.y, v4.z);\r\n\r\n // Get two of the triangle edges\r\n v1.subtractInPlace(v2);\r\n v2.subtractInPlace(v3);\r\n\r\n // Get the triangle face normal with the two edges.\r\n // Note that this is not the geometry normal, but the position-based face normal\r\n Vector3.CrossToRef(v1, v2, v4);\r\n const faceNormal = v4;\r\n faceNormal.normalize();\r\n faceNormals.push(faceNormal.x, faceNormal.y, faceNormal.z);\r\n\r\n // Now we need to consolidate what triangles are next to each other by organizing them by edges\r\n for (let i = 0; i < 3; i++) {\r\n const idx1 = indices[triOffset + i];\r\n const idx2 = indices[triOffset + ((i + 1) % 3)];\r\n\r\n // This is not going to work for geometry seams like on cylinders.\r\n // Possibly change from index ids to a representation of the vertex position\r\n const edgeId = idx1 > idx2 ? `${idx1}-${idx2}` : `${idx2}-${idx1}`;\r\n if (!facesByEdges[edgeId]) {\r\n facesByEdges[edgeId] = [];\r\n }\r\n\r\n facesByEdges[edgeId].push(triId);\r\n }\r\n }\r\n\r\n // Now we go through the edges and register the neighbors for each triangle\r\n for (const edgeId in facesByEdges) {\r\n const faces = facesByEdges[edgeId];\r\n if (!faceNeighbors[faces[0]]) {\r\n faceNeighbors[faces[0]] = [];\r\n }\r\n // Some edges may only have one face if the face is on the edge of the geometry\r\n if (faces.length > 1) {\r\n if (!faceNeighbors[faces[1]]) {\r\n faceNeighbors[faces[1]] = [];\r\n }\r\n\r\n faceNeighbors[faces[0]].push(faces[1]);\r\n faceNeighbors[faces[1]].push(faces[0]);\r\n }\r\n }\r\n\r\n const result = {\r\n faceCount: trianglesCount,\r\n faceNeighbors,\r\n facePositions,\r\n faceNormals,\r\n };\r\n meshFaceInfo.set(mesh.id, result);\r\n return result;\r\n }\r\n }\r\n }\r\n}\r\n","import { Camera } from '@babylonjs/core/Cameras/camera';\r\nimport { PickingInfo } from '@babylonjs/core/Collisions/pickingInfo';\r\nimport { Ray } from '@babylonjs/core/Culling/ray';\r\nimport { KeyboardEventTypes, KeyboardInfo } from '@babylonjs/core/Events/keyboardEvents';\r\nimport { PointerEventTypes } from '@babylonjs/core/Events/pointerEvents';\r\nimport { Matrix, Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { VertexBuffer } from '@babylonjs/core/Meshes/buffer';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { Node as babylonNode } from '@babylonjs/core/node';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { instanceOf } from '@models/classes/decorators/model-utils';\r\nimport {\r\n IDraggable,\r\n IKb3dFacetSelectArgs,\r\n IKb3dPickInfo,\r\n IKb3dPointerArgs,\r\n IKeyboardArgs,\r\n Kb3dPointerHandler,\r\n PointerInfoMinimal,\r\n eHandler,\r\n} from '@models/classes/handlers';\r\nimport { Kb3dManager, getByDynamicId } from '@models/classes/kb3d-manager';\r\nimport { Kb3dObject } from '@models/classes/kb3d-object';\r\nimport { MeshNode } from '@models/classes/meshes/mesh-node';\r\nimport { SpaceNode } from '@models/classes/meshes/space-node';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { Token } from '@models/classes/token';\r\nimport { EMPTY, Observable, Subject } from 'rxjs';\r\nimport { distinctUntilChanged } from 'rxjs/operators';\r\nimport { KbViewer } from '../kb-viewer';\r\nimport { KbArcRotateCamera } from './cameras/kb-arc-rotate-camera';\r\nimport { Dragger } from './dragger';\r\nimport { MeshFaceEngine } from './face-engine';\r\nimport { Highlighter } from './highlighter';\r\n\r\ninterface INodeHandlers {\r\n node: SpaceNode;\r\n handlers: T[];\r\n}\r\n\r\nexport enum eMarqueeSelectMode {\r\n Facets,\r\n Vertices,\r\n}\r\n\r\nexport interface IMarqueeSelectOptions {\r\n cullSelection?: boolean; // If true, don't select faces facing away\r\n onlyOnShiftPress?: boolean;\r\n}\r\n\r\nexport class Interaction {\r\n constructor(private kbViewer: KbViewer, private scene: Scene, public highlighter: Highlighter) {\r\n this.meshFaceEngine = new MeshFaceEngine(kbViewer);\r\n\r\n this.onClick = this.onClick$.asObservable();\r\n this.onFacetClick = this.onFacetClick$.asObservable();\r\n this.onDoubleClick = this.onDoubleClick$.asObservable();\r\n this.onPointerMove = this.onPointerMove$.asObservable();\r\n this.onPointerDown = this.onPointerDown$.asObservable();\r\n this.onPointerUp = this.onPointerUp$.asObservable();\r\n this.onKeyboard = this.onKeyboard$.asObservable();\r\n\r\n this.attachScene(scene);\r\n }\r\n\r\n attachScene(scene: Scene, utilityLayer = false) {\r\n scene.onPointerObservable.add(p => {\r\n const args = this.getKb3dPointerArgs(p);\r\n\r\n switch (p.type) {\r\n case PointerEventTypes.POINTERDOWN:\r\n this.handlePointerDown(args);\r\n break;\r\n case PointerEventTypes.POINTERUP:\r\n this.handlePointerUp(args);\r\n break;\r\n case PointerEventTypes.POINTERMOVE:\r\n this.handleMove(args);\r\n break;\r\n // case PointerEventTypes.POINTERWHEEL:\r\n // console.log(\"POINTER WHEEL\");\r\n // break;\r\n // case PointerEventTypes.POINTERPICK:\r\n // console.log(\"POINTER PICK\");\r\n // break;\r\n case PointerEventTypes.POINTERTAP:\r\n this.handleClick(args);\r\n break;\r\n case PointerEventTypes.POINTERDOUBLETAP:\r\n this.handleDoubleClick(args);\r\n break;\r\n }\r\n });\r\n\r\n scene.onKeyboardObservable.add(k => {\r\n const args = this.getKeyboardArgs(k);\r\n\r\n switch (k.type) {\r\n case KeyboardEventTypes.KEYDOWN:\r\n this.handleKeyboard(args);\r\n }\r\n });\r\n }\r\n\r\n /**observables for consumers to subscribe to */\r\n public onClick: Observable;\r\n public onFacetClick: Observable;\r\n public onDoubleClick: Observable;\r\n public onPointerMove: Observable;\r\n public onPointerDown: Observable;\r\n public onPointerUp: Observable;\r\n public onKeyboard: Observable;\r\n private onClick$ = new Subject();\r\n private onFacetClick$ = new Subject();\r\n private onDoubleClick$ = new Subject();\r\n private onPointerMove$ = new Subject();\r\n private onPointerDown$ = new Subject();\r\n private onPointerUp$ = new Subject();\r\n private onKeyboard$ = new Subject();\r\n\r\n private onMarqueeSelect: Subject | null = null;\r\n\r\n public pointerIsDown = false;\r\n public dragInProgress = false;\r\n private readonly meshFaceEngine: MeshFaceEngine;\r\n private selectionTolerance = 0;\r\n private marqueeSelectOptions: IMarqueeSelectOptions & {\r\n mode: eMarqueeSelectMode;\r\n startX: number;\r\n startY: number;\r\n startRay?: Ray;\r\n selectBox?: HTMLDivElement;\r\n mesh: Mesh;\r\n };\r\n private cameraFacingDirection = new Vector3();\r\n\r\n public setSelectionTolerance(tolerance: number) {\r\n this.selectionTolerance = tolerance;\r\n }\r\n\r\n public setSelectionCulling(enabled: boolean) {\r\n if (this.marqueeSelectOptions) {\r\n this.marqueeSelectOptions.cullSelection = enabled;\r\n }\r\n }\r\n\r\n public enableMarqueeSelect(meshNode: MeshNode, mode: eMarqueeSelectMode, options?: IMarqueeSelectOptions) {\r\n const mesh = this.kbViewer.getRendererNode(meshNode);\r\n\r\n if (mesh instanceof Mesh) {\r\n this.onMarqueeSelect = new Subject();\r\n\r\n this.marqueeSelectOptions = {\r\n cullSelection: false,\r\n onlyOnShiftPress: true,\r\n startX: 0,\r\n startY: 0,\r\n mode,\r\n mesh,\r\n ...(options || {}),\r\n };\r\n\r\n return this.onMarqueeSelect.asObservable().pipe(\r\n distinctUntilChanged((prev, curr) => {\r\n if (\r\n prev.selectedFaces.length !== curr.selectedFaces.length ||\r\n prev.vertexList.length !== curr.vertexList.length\r\n ) {\r\n return false;\r\n }\r\n for (let i = 0; i < curr.selectedFaces.length; i++) {\r\n if (curr.selectedFaces[i] !== prev.selectedFaces[i]) {\r\n return false;\r\n }\r\n }\r\n for (let i = 0; i < curr.vertexList.length; i++) {\r\n if (curr.vertexList[i] !== prev.vertexList[i]) {\r\n return false;\r\n }\r\n }\r\n\r\n return true;\r\n })\r\n );\r\n }\r\n return EMPTY;\r\n }\r\n\r\n public disableMarqueeSelect() {\r\n if (this.onMarqueeSelect) {\r\n this.onMarqueeSelect.complete();\r\n }\r\n this.onMarqueeSelect = null;\r\n }\r\n\r\n /** Do a hit test of the scene using x,y screen pixel coordinates */\r\n public pick(x: number, y: number, camera?: Camera) {\r\n const pickInfo = this.scene.pick(x, y, undefined, undefined, camera);\r\n if (pickInfo && pickInfo.hit) {\r\n const info = { ...pickInfo } as PickingInfo & IKb3dPickInfo;\r\n this.fillKb3dPickInfo(pickInfo, info);\r\n return info;\r\n }\r\n return undefined;\r\n }\r\n\r\n protected handleClickTypeEvent(args: IKb3dPointerArgs, handlerType: eHandler) {\r\n // For debugging\r\n // if (args.pickInfo?.ray) {\r\n // RayHelper.CreateAndShow(args.pickInfo.ray, this.scene, new Color3(1, 1, 0.1));\r\n // }\r\n\r\n if (args.targetNode) {\r\n //bubble up event, calling any handlers that have been applied to the nodes\r\n let h = this.getNextNodeHandlers(args.targetNode, handlerType)!;\r\n\r\n while (h) {\r\n let propagate = false;\r\n args.sourceNode = h.node;\r\n for (const handler of h.handlers) {\r\n handler(args);\r\n if (args.propagate) propagate = true;\r\n }\r\n if (!propagate) break;\r\n h = this.getNextNodeHandlers(h.node._parent!, handlerType)!;\r\n }\r\n } else {\r\n //there was no clicked mesh, but if there is a handler on the root scene node, we trigger that\r\n const handlers = this.getHandlerMap(this.kbViewer.sceneNode, handlerType).getByKey(\r\n this.kbViewer.sceneNode\r\n ) as Kb3dPointerHandler[];\r\n if (handlers) {\r\n args.targetNode = args.sourceNode = this.kbViewer.sceneNode;\r\n for (const handler of handlers) {\r\n handler(args);\r\n }\r\n }\r\n }\r\n }\r\n\r\n public handleClick(args: IKb3dPointerArgs) {\r\n this.handleClickTypeEvent(args, eHandler.click);\r\n // trigger the normal generic observable\r\n this.onClick$.next(args);\r\n\r\n // if there is an observer on the facet click, then send the facet data there\r\n if (this.onFacetClick$.observers.length > 0) {\r\n if (args.pickInfo?.pickedMesh && args.pickInfo.pickedMesh instanceof Mesh) {\r\n const pickedMesh = args.pickInfo.pickedMesh;\r\n\r\n const selectedTriangle = args.pickInfo.faceId;\r\n const faceInfo = this.meshFaceEngine.getMeshFaces(pickedMesh);\r\n if (faceInfo) {\r\n let selection: number[] = [];\r\n if (this.selectionTolerance >= 0) {\r\n selection = this.meshFaceEngine.expandSelection(\r\n [selectedTriangle],\r\n pickedMesh,\r\n this.selectionTolerance\r\n );\r\n } else {\r\n selection = [selectedTriangle];\r\n }\r\n\r\n const facetArgs: IKb3dFacetSelectArgs = {\r\n ...args,\r\n altKey: args.event.altKey,\r\n selectedFaces: selection,\r\n };\r\n\r\n this.onFacetClick$.next(facetArgs);\r\n }\r\n }\r\n }\r\n }\r\n\r\n protected handleDoubleClick(args: IKb3dPointerArgs) {\r\n this.handleClickTypeEvent(args, eHandler.doubleClick);\r\n //trigger the normal generic observable\r\n this.onDoubleClick$.next(args);\r\n }\r\n\r\n protected handleMove(args: IKb3dPointerArgs) {\r\n if (!this.pointerIsDown && this.kbViewer.sceneNode && this.kbViewer.sceneNode.enableHoverEffects) {\r\n if (args.targetNode) {\r\n const clicks = this.getNextClickHandlers(args.targetNode);\r\n const draggables = this.getNextDraggables(args.targetNode);\r\n if (clicks || draggables) {\r\n const n = clicks?.node || (draggables?.node as SpaceNode);\r\n this.highlighter.hover(n);\r\n }\r\n } else {\r\n this.highlighter.clearHover();\r\n }\r\n }\r\n\r\n if (this.scene.activeCamera instanceof KbArcRotateCamera && this.pointerIsDown && this.onMarqueeSelect) {\r\n if (!this.marqueeSelectOptions.onlyOnShiftPress || args.shiftKey) {\r\n args.event.preventDefault();\r\n (args.event as MouseEvent).stopPropagation();\r\n this.handleMarquee(args, this.scene.activeCamera);\r\n } else {\r\n this.unlockCamera();\r\n }\r\n }\r\n\r\n this.onPointerMove$.next(args);\r\n }\r\n\r\n public handlePointerDown(args: IKb3dPointerArgs) {\r\n this.pointerIsDown = true;\r\n\r\n if (this.onMarqueeSelect) {\r\n if (!this.marqueeSelectOptions.onlyOnShiftPress || args.shiftKey) {\r\n this.marqueeSelectOptions.startX = args.canvasX || 0;\r\n this.marqueeSelectOptions.startY = args.canvasY || 0;\r\n this.marqueeSelectOptions.startRay =\r\n this.scene.pick(args.canvasX!, args.canvasY!, undefined, undefined, this.scene.activeCamera)?.ray ||\r\n undefined;\r\n\r\n if (this.kbViewer.uiOverlay) {\r\n const selectBox = document.createElement('div');\r\n selectBox.style.position = 'absolute';\r\n selectBox.style.border = '1px dashed white';\r\n selectBox.style.background = 'rgba(128, 128, 128, 0.1)';\r\n selectBox.style.opacity = '0.5';\r\n this.kbViewer.uiOverlay.appendChild(selectBox);\r\n this.marqueeSelectOptions.selectBox = selectBox;\r\n }\r\n\r\n this.cameraFacingDirection = this.scene.activeCamera!.getForwardRay().direction;\r\n if (\r\n !this.marqueeSelectOptions.mesh.isFacetDataEnabled &&\r\n this.marqueeSelectOptions.mode === eMarqueeSelectMode.Facets\r\n ) {\r\n const normals = this.marqueeSelectOptions.mesh.getVerticesData(VertexBuffer.NormalKind);\r\n if (!normals) {\r\n this.marqueeSelectOptions.mesh.setVerticesData(\r\n VertexBuffer.NormalKind,\r\n new Float32Array(this.marqueeSelectOptions.mesh.getTotalVertices() * 3)\r\n );\r\n }\r\n }\r\n }\r\n } else if (args.targetNode) {\r\n const d = this.getNextDraggables(args.targetNode);\r\n if (d) {\r\n this.highlighter.clearHover();\r\n args.sourceNode = d.node;\r\n const dragger = new Dragger(this.kbViewer, this.scene);\r\n this.dragInProgress = true;\r\n dragger.startDrag(args, d.handlers[0], () => {\r\n this.dragInProgress = false;\r\n });\r\n }\r\n }\r\n\r\n this.onPointerDown$.next(args);\r\n }\r\n\r\n protected unlockCamera() {\r\n this.scene.activeCamera?.attachControl(this.kbViewer.canvas!);\r\n }\r\n\r\n public handlePointerUp(args: IKb3dPointerArgs) {\r\n this.pointerIsDown = false;\r\n if (this.marqueeSelectOptions && this.marqueeSelectOptions.selectBox) {\r\n this.kbViewer.uiOverlay?.removeChild(this.marqueeSelectOptions.selectBox);\r\n delete this.marqueeSelectOptions.selectBox;\r\n }\r\n this.unlockCamera();\r\n\r\n this.onPointerUp$.next(args);\r\n }\r\n\r\n protected handleKeyboard(args: IKeyboardArgs) {\r\n this.onKeyboard$.next(args);\r\n }\r\n\r\n protected handleMarquee(args: IKb3dPointerArgs, activeCamera: KbArcRotateCamera) {\r\n const mouseX = args.canvasX || 0;\r\n const mouseY = args.canvasY || 0;\r\n\r\n activeCamera.detachControl();\r\n\r\n if (this.marqueeSelectOptions.selectBox) {\r\n this.marqueeSelectOptions.selectBox.style.left = Math.min(this.marqueeSelectOptions.startX, mouseX) + 'px';\r\n this.marqueeSelectOptions.selectBox.style.top = Math.min(this.marqueeSelectOptions.startY, mouseY) + 'px';\r\n this.marqueeSelectOptions.selectBox.style.width =\r\n Math.abs(this.marqueeSelectOptions.startX - mouseX) + 'px';\r\n this.marqueeSelectOptions.selectBox.style.height =\r\n Math.abs(this.marqueeSelectOptions.startY - mouseY) + 'px';\r\n }\r\n let topOrigin: Vector3;\r\n let leftOrigin: Vector3;\r\n let rightOrigin: Vector3;\r\n let bottomOrigin: Vector3;\r\n let topNormal: Vector3 | undefined;\r\n let leftNormal: Vector3 | undefined;\r\n let rightNormal: Vector3 | undefined;\r\n let bottomNormal: Vector3 | undefined;\r\n const m = Matrix.Identity();\r\n const boxL = Math.min(this.marqueeSelectOptions.startX, mouseX);\r\n const boxR = Math.max(this.marqueeSelectOptions.startX, mouseX);\r\n const boxT = Math.min(this.marqueeSelectOptions.startY, mouseY);\r\n const boxB = Math.max(this.marqueeSelectOptions.startY, mouseY);\r\n\r\n if (activeCamera.mode === Camera.ORTHOGRAPHIC_CAMERA) {\r\n // ORTHOGRAPHIC CAMERA\r\n const forwardDirection = activeCamera.getForwardRay().direction;\r\n const sideDirection = Vector3.Cross(forwardDirection, Vector3.Up());\r\n topNormal = Vector3.Cross(forwardDirection, sideDirection);\r\n leftNormal = sideDirection;\r\n rightNormal = sideDirection.scale(-1);\r\n bottomNormal = Vector3.Cross(sideDirection, forwardDirection);\r\n topOrigin = this.scene.createPickingRay(boxL, boxT, m, activeCamera).origin;\r\n leftOrigin = this.scene.createPickingRay(boxL, boxT, m, activeCamera).origin;\r\n rightOrigin = this.scene.createPickingRay(boxR, boxB, m, activeCamera).origin;\r\n bottomOrigin = this.scene.createPickingRay(boxR, boxB, m, activeCamera).origin;\r\n } else {\r\n // PERSPECTIVE CAMERA\r\n // Assemble the 4 planes of the marquee select\r\n const origin = this.marqueeSelectOptions.startRay?.origin || activeCamera.position;\r\n topOrigin = origin;\r\n leftOrigin = origin;\r\n rightOrigin = origin;\r\n bottomOrigin = origin;\r\n\r\n const topLeft = this.scene.createPickingRay(boxL, boxT, m, activeCamera).direction;\r\n const topRight = this.scene.createPickingRay(boxR, boxT, m, activeCamera).direction;\r\n const botLeft = this.scene.createPickingRay(boxL, boxB, m, activeCamera).direction;\r\n const botRight = this.scene.createPickingRay(boxR, boxB, m, activeCamera).direction;\r\n\r\n if (topLeft && topRight && botLeft && botRight) {\r\n // Normals of the planes are equal to the cross product of the two vectors that define the plane\r\n topNormal = Vector3.Cross(topLeft, topRight);\r\n leftNormal = Vector3.Cross(botLeft, topLeft);\r\n rightNormal = Vector3.Cross(topRight, botRight);\r\n bottomNormal = Vector3.Cross(botRight, botLeft);\r\n }\r\n }\r\n\r\n if (topNormal && leftNormal && rightNormal && bottomNormal) {\r\n const mesh = this.marqueeSelectOptions.mesh;\r\n const fP = new Vector3();\r\n const fN = new Vector3();\r\n\r\n if (this.marqueeSelectOptions.mode === eMarqueeSelectMode.Facets) {\r\n const faceInfo = this.meshFaceEngine.getMeshFaces(mesh)!;\r\n const transform = mesh.computeWorldMatrix();\r\n\r\n const n = faceInfo.faceCount;\r\n const res = [] as number[];\r\n\r\n for (let i = 0; i < n; i++) {\r\n Vector3.TransformCoordinatesFromFloatsToRef(\r\n faceInfo.facePositions[i * 3],\r\n faceInfo.facePositions[i * 3 + 1],\r\n faceInfo.facePositions[i * 3 + 2],\r\n transform,\r\n fP\r\n );\r\n if (this.marqueeSelectOptions.cullSelection) {\r\n Vector3.TransformCoordinatesFromFloatsToRef(\r\n faceInfo.faceNormals[i * 3],\r\n faceInfo.faceNormals[i * 3 + 1],\r\n faceInfo.faceNormals[i * 3 + 2],\r\n transform,\r\n fN\r\n );\r\n }\r\n if (\r\n (!this.marqueeSelectOptions.cullSelection || Vector3.Dot(this.cameraFacingDirection, fN) > 0) &&\r\n pointPlaneSizeToRef(topNormal, topOrigin, fP) >= 0 &&\r\n pointPlaneSizeToRef(leftNormal, leftOrigin, fP) >= 0 &&\r\n pointPlaneSizeToRef(rightNormal, rightOrigin, fP) >= 0 &&\r\n pointPlaneSizeToRef(bottomNormal, bottomOrigin, fP) >= 0\r\n ) {\r\n res.push(i);\r\n }\r\n }\r\n\r\n this.onMarqueeSelect!.next({\r\n ...args,\r\n altKey: args.event.altKey,\r\n vertexList: [],\r\n selectedFaces: res,\r\n });\r\n } else if (this.marqueeSelectOptions.mode === eMarqueeSelectMode.Vertices) {\r\n const vertexPositions = mesh.getVerticesData(VertexBuffer.PositionKind);\r\n const transform = mesh.getWorldMatrix();\r\n const res = [] as number[];\r\n\r\n if (vertexPositions) {\r\n const n = vertexPositions.length / 3;\r\n for (let i = 0; i < n; i++) {\r\n fP.copyFromFloats(\r\n vertexPositions[i * 3],\r\n vertexPositions[i * 3 + 1],\r\n vertexPositions[i * 3 + 2]\r\n );\r\n Vector3.TransformCoordinatesToRef(fP, transform, fP);\r\n\r\n if (\r\n (!this.marqueeSelectOptions.cullSelection ||\r\n Vector3.Dot(this.cameraFacingDirection, mesh.getFacetNormal(i))) &&\r\n pointPlaneSizeToRef(topNormal, topOrigin, fP) >= 0 &&\r\n pointPlaneSizeToRef(leftNormal, leftOrigin, fP) >= 0 &&\r\n pointPlaneSizeToRef(rightNormal, rightOrigin, fP) >= 0 &&\r\n pointPlaneSizeToRef(bottomNormal, bottomOrigin, fP) >= 0\r\n ) {\r\n res.push(i);\r\n }\r\n }\r\n\r\n mesh.getIndices();\r\n this.onMarqueeSelect!.next({\r\n ...args,\r\n altKey: args.event.altKey,\r\n vertexList: res,\r\n selectedFaces: [],\r\n });\r\n }\r\n }\r\n }\r\n\r\n function pointPlaneSizeToRef(normal: Vector3, origin: Vector3, point: Vector3) {\r\n return normal.x * (point.x - origin.x) + normal.y * (point.y - origin.y) + normal.z * (point.z - origin.z);\r\n }\r\n }\r\n\r\n public getKb3dPointerArgs(p: PointerInfoMinimal) {\r\n const args = {\r\n ...p,\r\n pickInfo: p.pickInfo,\r\n propagate: false,\r\n pointerType: (p.event as any).pointerType,\r\n button: p.event.button,\r\n ctrlKey: p.event.ctrlKey || p.event.metaKey,\r\n shiftKey: p.event.shiftKey,\r\n canvasX: p.event.offsetX,\r\n canvasY: p.event.offsetY,\r\n getNormal: () => {\r\n if (p.pickInfo) {\r\n const normal = p.pickInfo.getNormal();\r\n return normal ? KbVector.FromVec3(normal) : undefined;\r\n }\r\n return undefined;\r\n },\r\n } as IKb3dPointerArgs;\r\n\r\n if (p.pickInfo) {\r\n this.fillKb3dPickInfo(p.pickInfo, args);\r\n }\r\n return args;\r\n }\r\n\r\n protected fillKb3dPickInfo(b: PickingInfo, k: IKb3dPickInfo) {\r\n k.u = b.bu;\r\n k.v = b.bv;\r\n if (b.pickedPoint) k.pickedPoint = KbVector.FromVec3(b.pickedPoint);\r\n if (b.pickedMesh) {\r\n // recurse up the tree until we find something, or return undefined\r\n let pickedMesh: babylonNode = b.pickedMesh;\r\n k.targetNode = getByDynamicId(pickedMesh.id);\r\n while (!k.targetNode && pickedMesh.parent) {\r\n pickedMesh = pickedMesh.parent;\r\n k.targetNode = getByDynamicId(pickedMesh.id);\r\n }\r\n\r\n if (b.pickedPoint) {\r\n k.meshPoint = KbVector.FromVec3(\r\n Vector3.TransformCoordinates(b.pickedPoint, b.pickedMesh.getWorldMatrix().clone().invert())\r\n );\r\n }\r\n }\r\n }\r\n\r\n protected getKeyboardArgs(k: KeyboardInfo) {\r\n const event = k.event as unknown as KeyboardEvent;\r\n const args = {\r\n ...k,\r\n key: event.key,\r\n keyCode: event.keyCode,\r\n altKey: event.altKey,\r\n ctrlKey: event.ctrlKey || event.metaKey,\r\n shiftKey: event.shiftKey,\r\n repeat: event.repeat,\r\n } as IKeyboardArgs;\r\n\r\n return args;\r\n }\r\n\r\n /**\r\n * scans the given node and ancestors to find the first node with a click handler and returns\r\n * the node and it's handlers\r\n */\r\n protected getNextNodeHandlers(node: SpaceNode, handlerType: eHandler): INodeHandlers | undefined {\r\n //bubble up event, calling any handlers that have been applied to the nodes\r\n let handlers = new Array();\r\n while (node && node._manager) {\r\n const hmap = this.getHandlerMap(node, handlerType);\r\n handlers = hmap.getByKey(node);\r\n //if this is a scene node, then we also have to look in the parent scene manager for click events on this item\r\n if (instanceOf(node, Token.SceneNode) && node.isNested() && node._parent && node._parent._manager) {\r\n handlers.push(...this.getHandlerMap(node._parent, handlerType).getByKey(node));\r\n }\r\n\r\n if (handlers && handlers.length) {\r\n break;\r\n }\r\n\r\n node = node._parent as SpaceNode;\r\n }\r\n\r\n return node ? { node, handlers } : undefined;\r\n }\r\n\r\n protected getHandlerMap(node: Kb3dObject, type: eHandler) {\r\n const m = node._manager as Kb3dManager;\r\n if (type == eHandler.click) return m.clickHandlers;\r\n else if (type == eHandler.draggable) return m.draggables;\r\n else if (type == eHandler.doubleClick) return m.doubleClickHandlers;\r\n else throw 'unrecognized handler type';\r\n }\r\n\r\n protected getNextClickHandlers(node: SpaceNode) {\r\n return this.getNextNodeHandlers(node, eHandler.click);\r\n }\r\n\r\n protected getNextDoubleClickHandlers(node: SpaceNode) {\r\n return this.getNextNodeHandlers(node, eHandler.doubleClick);\r\n }\r\n\r\n protected getNextDraggables(node: SpaceNode) {\r\n return this.getNextNodeHandlers(node, eHandler.draggable);\r\n }\r\n}\r\n","import { MultiMaterial } from '@babylonjs/core/Materials/multiMaterial';\r\nimport { StandardMaterial } from '@babylonjs/core/Materials/standardMaterial';\r\nimport { Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { SubMesh } from '@babylonjs/core/Meshes/subMesh';\r\nimport { Logger } from '@babylonjs/core/Misc/logger';\r\nimport { Observable } from '@babylonjs/core/Misc/observable';\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { Nullable } from '@babylonjs/core/types';\r\nimport { IVertexGroup } from '@models/classes/kb3d-models';\r\nimport { MeshNode } from '@models/classes/meshes/mesh-node';\r\nimport { KbColor } from '@models/classes/primitives/kb-color';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { Subject } from 'rxjs';\r\nimport { takeUntil } from 'rxjs/operators';\r\nimport { MeshFaceEngine } from '../face-engine';\r\nimport { eMarqueeSelectMode } from '../interaction';\r\nimport { KbGizmo } from './gizmo';\r\n\r\nexport interface VertexChanges {\r\n mesh: AbstractMesh;\r\n delta: Vector3;\r\n vertexIndices: number[];\r\n}\r\nconst dbSubmeshes = new Map<\r\n string,\r\n {\r\n visualizationMesh: Mesh;\r\n highlightMaterial: StandardMaterial;\r\n vertices: IVertexGroup[];\r\n }\r\n>();\r\n\r\n/**\r\n * Gizmo that enables moving vertices on a mesh.\r\n */\r\nexport class KbFacetSelectGizmo extends KbGizmo {\r\n /** Fires an event when any of it's sub gizmos are dragged */\r\n public onSelectionChange = new Observable<{ selectedFacets: Array> }>();\r\n\r\n public get attachedMesh() {\r\n return this.__attachedMesh;\r\n }\r\n public set attachedMesh(mesh: Nullable) {\r\n if (mesh && !(mesh instanceof Mesh)) {\r\n throw Error('facet select gizmo must be attached to a mesh with geometry');\r\n }\r\n this.__attachedMesh = mesh;\r\n }\r\n\r\n private utilLayer: UtilityLayerRenderer;\r\n private meshFaceEngine: MeshFaceEngine;\r\n private disposeSubject: Subject;\r\n private __attachedNode: Nullable;\r\n private __attachedMesh: Nullable = null;\r\n\r\n private currentSelection = new Map();\r\n private selectionsEnabled = false;\r\n private highlightOpacity = 0.75;\r\n\r\n /**\r\n * Creates a PositionGizmo\r\n * @param camera The camera being used for the scene. Used for positioning the particles so they line up with the vertices\r\n * @param gizmoLayer The utility layer the gizmo will be added to\r\n */\r\n constructor(private kbViewer: KbViewer, private scene: Scene) {\r\n super(kbViewer.utilityLayer);\r\n this.meshFaceEngine = new MeshFaceEngine(kbViewer);\r\n\r\n this.utilLayer = kbViewer.utilityLayer;\r\n\r\n this.attachedMesh = null;\r\n }\r\n\r\n attachTo(target: MeshNode) {\r\n this.__attachedNode = target;\r\n const aMesh = this.kbViewer.getRendererNode(target);\r\n if (!aMesh) {\r\n throw Error('Could not find mesh in scene to attach gizmo');\r\n }\r\n if (!(aMesh instanceof Mesh)) {\r\n throw Error('facet select gizmo must be attached to a mesh with geometry');\r\n }\r\n const mesh = aMesh;\r\n this.attachedMesh = mesh as Mesh;\r\n\r\n // Clean up old marquee\r\n if (this.disposeSubject) {\r\n this.disposeSubject.next();\r\n this.disposeSubject.complete();\r\n }\r\n\r\n this.disposeSubject = new Subject();\r\n\r\n this.kbViewer.interaction.onFacetClick.pipe(takeUntil(this.disposeSubject)).subscribe(event => {\r\n if (!this.selectionsEnabled) {\r\n return;\r\n }\r\n event.selectedFaces.sort();\r\n\r\n if (event.ctrlKey) {\r\n // Add selections\r\n for (let i = 0; i < event.selectedFaces.length; i++) {\r\n this.currentSelection.set(event.selectedFaces[i], null);\r\n }\r\n } else if (event.altKey) {\r\n // Remove selections\r\n for (let i = 0; i < event.selectedFaces.length; i++) {\r\n this.currentSelection.delete(event.selectedFaces[i]);\r\n }\r\n } else {\r\n this.currentSelection.clear();\r\n for (let i = 0; i < event.selectedFaces.length; i++) {\r\n this.currentSelection.set(event.selectedFaces[i], null);\r\n }\r\n }\r\n const result = this.getVertexGroups(this.currentSelection.keys());\r\n\r\n this.visualizeSubmesh(mesh, result, KbColor.Red());\r\n this.onSelectionChange.notifyObservers({ selectedFacets: result });\r\n });\r\n }\r\n\r\n updateGizmoPosition() {\r\n this.kbViewer.forceAndWaitForNextRender().then(() => {\r\n if (this.__attachedMesh) {\r\n const result = this.getVertexGroups(this.currentSelection.keys());\r\n this.visualizeSubmesh(this.__attachedMesh, result, KbColor.Red(), true);\r\n }\r\n });\r\n }\r\n\r\n selectFacetGroups(selection: IVertexGroup[]) {\r\n selection.forEach(vertexGroup => {\r\n if (vertexGroup.startIndex === undefined || vertexGroup.endIndex === undefined) {\r\n throw Error('corrupt vertex group');\r\n }\r\n const startFacet = Math.round(vertexGroup.startIndex / 3);\r\n const endFacet = Math.ceil(vertexGroup.endIndex / 3);\r\n\r\n for (let i = startFacet; i < endFacet; i++) {\r\n this.currentSelection.set(i, null);\r\n }\r\n });\r\n\r\n this.updateGizmoPosition();\r\n }\r\n\r\n enableSelectionChange() {\r\n this.selectionsEnabled = true;\r\n if (this.__attachedNode) {\r\n this.kbViewer.interaction\r\n .enableMarqueeSelect(this.__attachedNode, eMarqueeSelectMode.Facets)\r\n .pipe(takeUntil(this.disposeSubject))\r\n .subscribe(event => {\r\n if (\r\n this.selectionsEnabled &&\r\n this.__attachedMesh &&\r\n event.selectedFaces &&\r\n event.selectedFaces.length > 0\r\n ) {\r\n if (event.ctrlKey) {\r\n // Add selections\r\n for (let i = 0; i < event.selectedFaces.length; i++) {\r\n this.currentSelection.set(event.selectedFaces[i], null);\r\n }\r\n } else if (event.altKey) {\r\n // Remove selections\r\n for (let i = 0; i < event.selectedFaces.length; i++) {\r\n this.currentSelection.delete(event.selectedFaces[i]);\r\n }\r\n } else {\r\n this.currentSelection.clear();\r\n for (let i = 0; i < event.selectedFaces.length; i++) {\r\n this.currentSelection.set(event.selectedFaces[i], null);\r\n }\r\n }\r\n const result = this.getVertexGroups(this.currentSelection.keys());\r\n\r\n this.visualizeSubmesh(this.__attachedMesh, result, KbColor.Red());\r\n this.onSelectionChange.notifyObservers({ selectedFacets: result });\r\n }\r\n });\r\n }\r\n }\r\n\r\n disableSelectionChange() {\r\n this.selectionsEnabled = false;\r\n this.kbViewer.interaction.disableMarqueeSelect();\r\n }\r\n\r\n /**\r\n * Disposes of the gizmo\r\n */\r\n dispose() {\r\n if (this.__attachedNode) {\r\n this.kbViewer.highlighter.remove(this.__attachedNode);\r\n }\r\n if (this.__attachedMesh) {\r\n const submeshInfo = dbSubmeshes.get(this.__attachedMesh.id);\r\n submeshInfo?.visualizationMesh.setEnabled(false);\r\n }\r\n }\r\n\r\n /**\r\n * CustomMeshes are not supported by this gizmo\r\n * @param mesh The mesh to replace the default mesh of the gizmo\r\n */\r\n setCustomMesh(mesh: Mesh) {\r\n Logger.Error('Custom meshes are not supported on this gizmo.');\r\n }\r\n\r\n setBackfaceCulling(enable: boolean) {\r\n this.kbViewer.interaction.setSelectionCulling(enable);\r\n }\r\n\r\n setTolerance(tolerance: number) {\r\n this.kbViewer.interaction.setSelectionTolerance(tolerance);\r\n }\r\n\r\n expandSelection(tolerance?: number) {\r\n if (this.attachedMesh instanceof Mesh) {\r\n const currentSelection = Array.from(this.currentSelection.keys());\r\n const newSelection = this.meshFaceEngine.expandSelection(currentSelection, this.attachedMesh, tolerance);\r\n\r\n for (let i = 0; i < newSelection.length; i++) {\r\n this.currentSelection.set(newSelection[i], null);\r\n }\r\n const result = this.getVertexGroups(this.currentSelection.keys());\r\n\r\n this.visualizeSubmesh(this.attachedMesh, result, KbColor.Red());\r\n this.onSelectionChange.notifyObservers({ selectedFacets: result });\r\n }\r\n }\r\n\r\n setVisualization(opacity: number) {\r\n const submeshInfo = dbSubmeshes.get(this.attachedMesh!.id);\r\n if (submeshInfo && submeshInfo.visualizationMesh.material instanceof MultiMaterial) {\r\n submeshInfo.highlightMaterial.alpha = opacity;\r\n }\r\n this.highlightOpacity = opacity;\r\n }\r\n\r\n private getVertexGroups(faceIds: IterableIterator): Array> {\r\n const result: Array> = [];\r\n\r\n for (const faceId of faceIds) {\r\n if (result.length > 0 && result[result.length - 1].endIndex === faceId * 3) {\r\n result[result.length - 1].endIndex += 3;\r\n } else {\r\n result.push({\r\n startIndex: faceId * 3,\r\n endIndex: (faceId + 1) * 3,\r\n });\r\n }\r\n }\r\n\r\n return result;\r\n }\r\n\r\n private visualizeSubmesh(\r\n node: Mesh,\r\n vertices: IVertexGroup[],\r\n color?: KbColor | null,\r\n geometryNeedsUpdate?: boolean\r\n ) {\r\n if (node instanceof Mesh && node.geometry) {\r\n let submeshInfo = dbSubmeshes.get(node.id);\r\n let highlightMaterial: StandardMaterial | undefined;\r\n const utilityLayerScene = this.utilLayer.utilityLayerScene;\r\n\r\n if (submeshInfo) {\r\n highlightMaterial = submeshInfo.highlightMaterial;\r\n }\r\n\r\n if (submeshInfo && geometryNeedsUpdate) {\r\n submeshInfo.visualizationMesh.dispose();\r\n submeshInfo = undefined;\r\n }\r\n\r\n if (!submeshInfo) {\r\n const multiMaterial = new MultiMaterial(`submeshVisualizationMat_${node.id}`, utilityLayerScene);\r\n\r\n const transparantMaterial = new StandardMaterial(\r\n `submeshVisualizationMat1_${node.id}`,\r\n utilityLayerScene\r\n );\r\n transparantMaterial.alpha = 0;\r\n multiMaterial.subMaterials.push(transparantMaterial);\r\n\r\n if (!highlightMaterial) {\r\n highlightMaterial = new StandardMaterial(`submeshVisualizationMat2_${node.id}`, utilityLayerScene);\r\n highlightMaterial.emissiveColor = color?.toColor3() || KbColor.KbmaxOrange().toColor3();\r\n highlightMaterial.alpha = this.highlightOpacity;\r\n }\r\n multiMaterial.subMaterials.push(highlightMaterial);\r\n\r\n submeshInfo = {\r\n visualizationMesh: new Mesh(`submeshVisualization_${node.id}`, utilityLayerScene, null, node, true),\r\n highlightMaterial,\r\n vertices,\r\n };\r\n\r\n submeshInfo.visualizationMesh.makeGeometryUnique();\r\n submeshInfo.visualizationMesh.material = multiMaterial;\r\n submeshInfo.visualizationMesh.isPickable = false;\r\n\r\n dbSubmeshes.set(node.id, submeshInfo);\r\n } else {\r\n submeshInfo.visualizationMesh.setEnabled(true);\r\n }\r\n\r\n const vMesh = submeshInfo.visualizationMesh;\r\n const totalIndices = node.geometry.getTotalIndices();\r\n const totalVertices = node.geometry.getTotalVertices();\r\n vMesh.subMeshes = [];\r\n new SubMesh(0, 0, totalVertices, 0, totalIndices, vMesh);\r\n\r\n vertices.forEach(indexGroup => {\r\n if (indexGroup.startIndex === undefined || indexGroup.endIndex === undefined) {\r\n throw Error('Missing submesh information');\r\n }\r\n new SubMesh(\r\n 1,\r\n 0,\r\n totalVertices,\r\n indexGroup.startIndex,\r\n indexGroup.endIndex - indexGroup.startIndex,\r\n vMesh\r\n );\r\n });\r\n }\r\n }\r\n}\r\n","import { Light, PointLight, SpotLight } from '@babylonjs/core/Lights';\r\nimport { Quaternion, Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { LightNode } from '@models/classes/lights/light-node';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { createLightVisualization } from '../../helpers/light-helper';\r\nimport { KbGizmo } from './gizmo';\r\nimport { KbPositionGizmo } from './position-gizmo';\r\nimport { KbRotationGizmo } from './rotation-gizmo';\r\n\r\n/**\r\n * Gizmo that enables dragging and rotating lights\r\n */\r\nexport class KbLightGizmo extends KbGizmo {\r\n /**\r\n * Internal gizmo used for positioning lights\r\n */\r\n posGizmo: KbPositionGizmo | undefined;\r\n /**\r\n * Internal gizmo used for rotating lights\r\n */\r\n rotGizmo: KbRotationGizmo | undefined;\r\n\r\n private _attachedLight?: LightNode;\r\n private _helperMesh: AbstractMesh | undefined;\r\n\r\n lightPositionGizmo: Mesh | undefined;\r\n\r\n attachTo(lightNode: LightNode) {\r\n const light = this.kbViewer.getRendererNode(lightNode);\r\n if (!light) {\r\n console.warn('could not find light in scene');\r\n return;\r\n }\r\n this._attachedLight = lightNode;\r\n\r\n if (this.posGizmo) {\r\n this.posGizmo.dispose();\r\n }\r\n if (this.rotGizmo) {\r\n this.rotGizmo.dispose();\r\n }\r\n const _helperMesh = (this._helperMesh = new AbstractMesh('', this.gizmoLayer.utilityLayerScene));\r\n _helperMesh.rotationQuaternion = Quaternion.Identity();\r\n const lightVisual = this.updatePreviewMesh(light);\r\n _helperMesh.addChild(lightVisual);\r\n }\r\n\r\n /**\r\n * Creates a PositionGizmo\r\n * @param gizmoLayer The utility layer the gizmo will be added to\r\n */\r\n constructor(\r\n private kbViewer: KbViewer,\r\n gizmoLayer: UtilityLayerRenderer = UtilityLayerRenderer.DefaultUtilityLayer\r\n ) {\r\n super(gizmoLayer);\r\n\r\n this.attachedMesh = null;\r\n }\r\n\r\n /**\r\n * Disposes of the gizmo\r\n */\r\n dispose() {\r\n if (this.posGizmo) {\r\n this.posGizmo.dispose();\r\n }\r\n if (this.rotGizmo) {\r\n this.rotGizmo.dispose();\r\n }\r\n if (this._helperMesh) {\r\n this._helperMesh.dispose();\r\n }\r\n if (this.lightPositionGizmo) {\r\n this.lightPositionGizmo.dispose();\r\n }\r\n delete this._attachedLight;\r\n }\r\n\r\n updateGizmoPosition(lightNode?: LightNode) {\r\n if (!lightNode) {\r\n return;\r\n }\r\n const light = this.kbViewer.getRendererNode(lightNode);\r\n if (light) {\r\n if (!this._attachedLight || lightNode._dynamicId !== this._attachedLight._dynamicId) {\r\n this.attachTo(lightNode);\r\n } else {\r\n this.updatePreviewMesh(light);\r\n }\r\n }\r\n }\r\n\r\n protected updatePreviewMesh(light: Light) {\r\n const scene = this.gizmoLayer.utilityLayerScene;\r\n\r\n if (!this.lightPositionGizmo) {\r\n this.lightPositionGizmo = createLightVisualization(light, scene);\r\n }\r\n\r\n if (light instanceof SpotLight || light instanceof PointLight) {\r\n this.lightPositionGizmo.position = light.parent\r\n ? Vector3.TransformCoordinates(light.position, light.parent.computeWorldMatrix(true).clone().invert())\r\n : light.position;\r\n }\r\n\r\n return this.lightPositionGizmo;\r\n }\r\n}\r\n","import { PointerDragBehavior } from '@babylonjs/core/Behaviors/Meshes/pointerDragBehavior';\r\nimport { PointerInfo } from '@babylonjs/core/Events/pointerEvents';\r\nimport { StandardMaterial } from '@babylonjs/core/Materials/standardMaterial';\r\nimport { Color3, Matrix, Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { CylinderBuilder } from '@babylonjs/core/Meshes/Builders/cylinderBuilder';\r\nimport { LinesMesh } from '@babylonjs/core/Meshes/linesMesh';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { Observable, Observer } from '@babylonjs/core/Misc/observable';\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { Nullable } from '@babylonjs/core/types';\r\nimport { KbGizmo } from './gizmo';\r\n\r\n/**\r\n * Single axis drag gizmo\r\n */\r\nexport class KbAxisDragGizmo extends KbGizmo {\r\n /**\r\n * Drag behavior responsible for the gizmos dragging interactions\r\n */\r\n public dragBehavior: PointerDragBehavior;\r\n private _pointerObserver: Nullable> = null;\r\n /**\r\n * Drag distance in babylon units that the gizmo will snap to when dragged (Default: 0)\r\n */\r\n public snapDistance = 0;\r\n /**\r\n * Event that fires each time the gizmo snaps to a new location.\r\n * * snapDistance is the the change in distance\r\n */\r\n public onSnapObservable = new Observable<{ snapDistance: number }>();\r\n public updateObservable = new Observable<{ delta: Vector3 }>();\r\n\r\n /** @hidden */\r\n public static _CreateArrow(scene: Scene, material: StandardMaterial): TransformNode {\r\n const arrow = new TransformNode('arrow', scene);\r\n const cylinder = CylinderBuilder.CreateCylinder(\r\n 'cylinder',\r\n { diameterTop: 0, height: 0.075, diameterBottom: 0.0375, tessellation: 12 },\r\n scene\r\n );\r\n const line = CylinderBuilder.CreateCylinder(\r\n 'cylinder',\r\n { diameterTop: 0.01, height: 0.275, diameterBottom: 0.01, tessellation: 6 },\r\n scene\r\n );\r\n const drag = CylinderBuilder.CreateCylinder(\r\n 'cylinder',\r\n { diameterTop: 0.06, height: 0.4, diameterBottom: 0.06, tessellation: 6 },\r\n scene\r\n );\r\n line.material = material;\r\n cylinder.parent = arrow;\r\n line.parent = arrow;\r\n drag.parent = arrow;\r\n\r\n // Position arrow pointing in its drag axis\r\n cylinder.material = material;\r\n cylinder.rotation.x = Math.PI / 2;\r\n cylinder.position.z += 0.3;\r\n line.position.z += 0.275 / 2;\r\n line.rotation.x = Math.PI / 2;\r\n\r\n drag.position.z += 0.4 / 2;\r\n drag.rotation = line.rotation;\r\n drag.visibility = 0;\r\n\r\n return arrow;\r\n }\r\n\r\n /** @hidden */\r\n public static _CreateArrowInstance(scene: Scene, arrow: TransformNode): TransformNode {\r\n const instance = new TransformNode('arrow', scene);\r\n for (const mesh of arrow.getChildMeshes()) {\r\n const childInstance = (mesh as Mesh).createInstance(mesh.name);\r\n childInstance.parent = instance;\r\n }\r\n return instance;\r\n }\r\n\r\n /**\r\n * Creates an AxisDragGizmo\r\n * @param gizmoLayer The utility layer the gizmo will be added to\r\n * @param dragAxis The axis which the gizmo will be able to drag on\r\n * @param color The color of the gizmo\r\n */\r\n constructor(\r\n dragAxis: Vector3,\r\n color: Color3 = Color3.Gray(),\r\n gizmoLayer: UtilityLayerRenderer = UtilityLayerRenderer.DefaultUtilityLayer\r\n ) {\r\n super(gizmoLayer);\r\n\r\n // Create Material\r\n const coloredMaterial = new StandardMaterial('', gizmoLayer.utilityLayerScene);\r\n coloredMaterial.emissiveColor = color;\r\n // coloredMaterial.diffuseColor = color;\r\n // coloredMaterial.specularColor = color.subtract(new Color3(0.1, 0.1, 0.1));\r\n\r\n const hoverMaterial = new StandardMaterial('', gizmoLayer.utilityLayerScene);\r\n hoverMaterial.emissiveColor = color.add(new Color3(0.3, 0.3, 0.3));\r\n // hoverMaterial.diffuseColor = color.add(new Color3(0.3, 0.3, 0.3));\r\n\r\n // Build mesh on root node\r\n const arrow = KbAxisDragGizmo._CreateArrow(gizmoLayer.utilityLayerScene, coloredMaterial);\r\n\r\n arrow.lookAt(this._rootMesh.position.add(dragAxis));\r\n arrow.scaling.scaleInPlace(1 / 3);\r\n arrow.parent = this._rootMesh;\r\n\r\n let currentSnapDragDistance = 0;\r\n const tmpVector = new Vector3();\r\n const tmpSnapEvent = { snapDistance: 0 };\r\n // Add drag behavior to handle events when the gizmo is dragged\r\n this.dragBehavior = new PointerDragBehavior({ dragAxis: dragAxis });\r\n this.dragBehavior.moveAttached = false;\r\n this._rootMesh.addBehavior(this.dragBehavior);\r\n\r\n const localDelta = new Vector3();\r\n const tmpMatrix = new Matrix();\r\n this.dragBehavior.onDragObservable.add(event => {\r\n if (this.attachedMesh) {\r\n // Convert delta to local translation if it has a parent\r\n if (this.attachedMesh.parent) {\r\n this.attachedMesh.parent.computeWorldMatrix().invertToRef(tmpMatrix);\r\n tmpMatrix.setTranslationFromFloats(0, 0, 0);\r\n Vector3.TransformCoordinatesToRef(event.delta, tmpMatrix, localDelta);\r\n } else {\r\n localDelta.copyFrom(event.delta);\r\n }\r\n // Snapping logic\r\n if (this.snapDistance == 0) {\r\n this.attachedMesh.position.addInPlace(localDelta);\r\n this.updateObservable.notifyObservers({ delta: localDelta });\r\n } else {\r\n currentSnapDragDistance += event.dragDistance;\r\n if (Math.abs(currentSnapDragDistance) > this.snapDistance) {\r\n const dragSteps = Math.floor(Math.abs(currentSnapDragDistance) / this.snapDistance);\r\n currentSnapDragDistance = currentSnapDragDistance % this.snapDistance;\r\n localDelta.normalizeToRef(tmpVector);\r\n tmpVector.scaleInPlace(this.snapDistance * dragSteps);\r\n this.attachedMesh.position.addInPlace(tmpVector);\r\n tmpSnapEvent.snapDistance = this.snapDistance * dragSteps;\r\n this.updateObservable.notifyObservers({ delta: tmpVector });\r\n this.onSnapObservable.notifyObservers(tmpSnapEvent);\r\n }\r\n }\r\n }\r\n });\r\n\r\n this._pointerObserver = gizmoLayer.utilityLayerScene.onPointerObservable.add(pointerInfo => {\r\n if (this._customMeshSet) {\r\n return;\r\n }\r\n const isHovered =\r\n pointerInfo.pickInfo &&\r\n this._rootMesh.getChildMeshes().indexOf(pointerInfo.pickInfo.pickedMesh) != -1;\r\n const material = isHovered ? hoverMaterial : coloredMaterial;\r\n this._rootMesh.getChildMeshes().forEach(m => {\r\n m.material = material;\r\n if ((m).color) {\r\n (m).color = material.diffuseColor;\r\n }\r\n });\r\n });\r\n }\r\n protected _attachedMeshChanged(value: Nullable) {\r\n if (this.dragBehavior) {\r\n this.dragBehavior.enabled = value ? true : false;\r\n }\r\n }\r\n /**\r\n * Disposes of the gizmo\r\n */\r\n public dispose() {\r\n this.onSnapObservable.clear();\r\n this.gizmoLayer.utilityLayerScene.onPointerObservable.remove(this._pointerObserver);\r\n this.dragBehavior.detach();\r\n super.dispose();\r\n }\r\n}\r\n","import { Color3, Matrix, Quaternion, Space, Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { Logger } from '@babylonjs/core/Misc/logger';\r\nimport { Observable } from '@babylonjs/core/Misc/observable';\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { Nullable } from '@babylonjs/core/types';\r\nimport { SpaceNode } from '@models/classes/meshes/space-node';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { KbAxisDragGizmo } from './axis-drag-gizmo';\r\nimport { AUTO_MESH_NAME, KbGizmo } from './gizmo';\r\n\r\n/**\r\n * Gizmo that enables dragging a mesh along 3 axis\r\n */\r\nexport class KbPositionGizmo extends KbGizmo {\r\n /**\r\n * Internal gizmo used for interactions on the x axis\r\n */\r\n public xGizmo: KbAxisDragGizmo;\r\n /**\r\n * Internal gizmo used for interactions on the y axis\r\n */\r\n public yGizmo: KbAxisDragGizmo;\r\n /**\r\n * Internal gizmo used for interactions on the z axis\r\n */\r\n public zGizmo: KbAxisDragGizmo;\r\n\r\n /** Fires an event when any of it's sub gizmos are dragged */\r\n public onDragStartObservable = new Observable();\r\n /** Fires an event while any of it's sub gizmos are dragged */\r\n public onDragObservable = new Observable<{ delta: Vector3 }>();\r\n /** Fires an event when any of it's sub gizmos are released from dragging */\r\n public onDragEndObservable = new Observable<{ delta: Vector3 }>();\r\n\r\n private _attachedSpaceNode: SpaceNode | undefined;\r\n private _positionSpace: TransformNode | undefined;\r\n\r\n attachTo(target: SpaceNode | KbVector, positionSpace?: SpaceNode, orientGizmo?: Quaternion) {\r\n if (orientGizmo) {\r\n this.xGizmo.updateGizmoRotationToMatchAttachedMesh = false;\r\n this.yGizmo.updateGizmoRotationToMatchAttachedMesh = false;\r\n this.zGizmo.updateGizmoRotationToMatchAttachedMesh = false;\r\n this.xGizmo._rootMesh.rotationQuaternion = orientGizmo;\r\n this.yGizmo._rootMesh.rotationQuaternion = orientGizmo;\r\n this.zGizmo._rootMesh.rotationQuaternion = orientGizmo;\r\n }\r\n if (!positionSpace && target instanceof SpaceNode && target._parent) {\r\n positionSpace = target._parent;\r\n }\r\n if (positionSpace) {\r\n this._positionSpace = this.kbViewer.getRendererNode(positionSpace);\r\n }\r\n if (target instanceof SpaceNode) {\r\n this._attachedSpaceNode = target;\r\n this._updateGizmoPosition();\r\n } else {\r\n this._updateGizmoPosition(target);\r\n }\r\n }\r\n\r\n updateGizmoPosition(changeTarget?: KbVector) {\r\n // Need to wait for the next frame because this can be called before the babylon object is actually updated\r\n this.kbViewer.onNextFrame(() => {\r\n this._updateGizmoPosition(changeTarget);\r\n });\r\n }\r\n\r\n private _updateGizmoPosition(changeTarget?: KbVector) {\r\n const positionSpaceTransform = this._positionSpace?.computeWorldMatrix(true);\r\n if (changeTarget) {\r\n this.attachMeshAtNodeSpace(this.gizmoLayer.utilityLayerScene, changeTarget, positionSpaceTransform);\r\n } else if (this._attachedSpaceNode) {\r\n this.attachMeshAtNodeSpace(\r\n this.gizmoLayer.utilityLayerScene,\r\n this._attachedSpaceNode,\r\n positionSpaceTransform\r\n );\r\n }\r\n super.updateGizmoPosition();\r\n }\r\n\r\n public get attachedMesh() {\r\n if (this.xGizmo) {\r\n return this.xGizmo.attachedMesh;\r\n } else if (this.yGizmo) {\r\n return this.yGizmo.attachedMesh;\r\n } else {\r\n return this.zGizmo.attachedMesh;\r\n }\r\n }\r\n\r\n public set attachedMesh(mesh: Nullable) {\r\n if (this.xGizmo) {\r\n this.xGizmo.attachedMesh = mesh;\r\n }\r\n if (this.yGizmo) {\r\n this.yGizmo.attachedMesh = mesh;\r\n }\r\n if (this.zGizmo) {\r\n this.zGizmo.attachedMesh = mesh;\r\n }\r\n if (mesh) {\r\n this._rootMesh.rotationQuaternion = mesh.rotationQuaternion\r\n ? mesh.rotationQuaternion\r\n : mesh.rotation.toQuaternion();\r\n this._rootMesh.position = mesh.position;\r\n this._rootMesh.setPivotPoint(mesh.getPivotPoint(), Space.WORLD);\r\n }\r\n }\r\n\r\n get visibility() {\r\n return this.xGizmo.visibility;\r\n }\r\n set visibility(visibility: number) {\r\n if (this.xGizmo) {\r\n this.xGizmo.visibility = visibility;\r\n this.yGizmo.visibility = visibility;\r\n this.zGizmo.visibility = visibility;\r\n }\r\n }\r\n\r\n private positionChange = Vector3.Zero();\r\n\r\n /**\r\n * Creates a PositionGizmo\r\n * @param gizmoLayer The utility layer the gizmo will be added to\r\n */\r\n constructor(\r\n private kbViewer: KbViewer,\r\n gizmoLayer: UtilityLayerRenderer = UtilityLayerRenderer.DefaultUtilityLayer,\r\n axisVisibility?: [boolean, boolean, boolean]\r\n ) {\r\n super(gizmoLayer);\r\n let activeGizmos = [];\r\n if (axisVisibility) {\r\n if (axisVisibility[0]) {\r\n this.xGizmo = new KbAxisDragGizmo(new Vector3(1, 0, 0), Color3.Red().scale(0.5), gizmoLayer);\r\n this.xGizmo.updateGizmoRotationToMatchAttachedMesh = axisVisibility[0];\r\n this.xGizmo.updateGizmoPositionToMatchAttachedMesh = axisVisibility[0];\r\n activeGizmos.push(this.xGizmo);\r\n }\r\n if (axisVisibility[1]) {\r\n this.yGizmo = new KbAxisDragGizmo(new Vector3(0, 1, 0), Color3.Green().scale(0.5), gizmoLayer);\r\n this.yGizmo.updateGizmoRotationToMatchAttachedMesh = axisVisibility[1];\r\n this.yGizmo.updateGizmoPositionToMatchAttachedMesh = axisVisibility[1];\r\n activeGizmos.push(this.yGizmo);\r\n }\r\n if (axisVisibility[2]) {\r\n this.zGizmo = new KbAxisDragGizmo(new Vector3(0, 0, 1), Color3.Blue().scale(0.5), gizmoLayer);\r\n this.zGizmo.updateGizmoRotationToMatchAttachedMesh = axisVisibility[2];\r\n this.zGizmo.updateGizmoPositionToMatchAttachedMesh = axisVisibility[2];\r\n activeGizmos.push(this.zGizmo);\r\n }\r\n } else {\r\n this.xGizmo = new KbAxisDragGizmo(new Vector3(1, 0, 0), Color3.Red().scale(0.5), gizmoLayer);\r\n this.yGizmo = new KbAxisDragGizmo(new Vector3(0, 1, 0), Color3.Green().scale(0.5), gizmoLayer);\r\n this.zGizmo = new KbAxisDragGizmo(new Vector3(0, 0, 1), Color3.Blue().scale(0.5), gizmoLayer);\r\n\r\n this.xGizmo.updateGizmoRotationToMatchAttachedMesh = true;\r\n this.yGizmo.updateGizmoRotationToMatchAttachedMesh = true;\r\n this.zGizmo.updateGizmoRotationToMatchAttachedMesh = true;\r\n this.xGizmo.updateGizmoPositionToMatchAttachedMesh = true;\r\n this.yGizmo.updateGizmoPositionToMatchAttachedMesh = true;\r\n this.zGizmo.updateGizmoPositionToMatchAttachedMesh = true;\r\n activeGizmos = [this.xGizmo, this.yGizmo, this.zGizmo];\r\n }\r\n\r\n // Relay drag events\r\n activeGizmos.forEach(gizmo => {\r\n gizmo.dragBehavior.onDragStartObservable.add(() => {\r\n this.onDragStartObservable.notifyObservers();\r\n this.positionChange.setAll(0);\r\n });\r\n gizmo.updateObservable.add(event => {\r\n // transform the rotation to be in local space, the gizmos return global space movements\r\n const transform = new Matrix();\r\n this.attachedMesh!.computeWorldMatrix(true).invertToRef(transform);\r\n transform.setTranslation(Vector3.Zero());\r\n const { delta } = event;\r\n const result = Vector3.TransformCoordinates(delta, transform);\r\n\r\n // squish floating point errors that look gross in the output\r\n if (Math.abs(result.x) < 0.00000001) result.x = 0;\r\n if (Math.abs(result.y) < 0.00000001) result.y = 0;\r\n if (Math.abs(result.z) < 0.00000001) result.z = 0;\r\n\r\n this.positionChange.addInPlace(result);\r\n this.onDragObservable.notifyObservers({ delta: result });\r\n });\r\n gizmo.dragBehavior.onDragEndObservable.add(() => {\r\n this.onDragEndObservable.notifyObservers({ delta: this.positionChange });\r\n });\r\n });\r\n this.attachedMesh = null;\r\n }\r\n\r\n /**\r\n * Drag distance in babylon units that the gizmo will snap to when dragged (Default: 0)\r\n */\r\n public set snapDistance(value: number) {\r\n if (this.xGizmo) {\r\n this.xGizmo.snapDistance = value;\r\n }\r\n if (this.yGizmo) {\r\n this.yGizmo.snapDistance = value;\r\n }\r\n if (this.zGizmo) {\r\n this.zGizmo.snapDistance = value;\r\n }\r\n }\r\n public get snapDistance() {\r\n if (this.xGizmo) {\r\n return this.xGizmo.snapDistance;\r\n } else if (this.yGizmo) {\r\n return this.yGizmo.snapDistance;\r\n } else {\r\n return this.zGizmo.snapDistance;\r\n }\r\n }\r\n\r\n /**\r\n * Ratio for the scale of the gizmo (Default: 1)\r\n */\r\n public set scaleRatio(value: number) {\r\n if (this.xGizmo) {\r\n this.xGizmo.scaleRatio = value;\r\n }\r\n if (this.yGizmo) {\r\n this.yGizmo.scaleRatio = value;\r\n }\r\n if (this.zGizmo) {\r\n this.zGizmo.scaleRatio = value;\r\n }\r\n }\r\n public get scaleRatio() {\r\n if (this.xGizmo) {\r\n return this.xGizmo.scaleRatio;\r\n } else if (this.yGizmo) {\r\n return this.yGizmo.scaleRatio;\r\n } else {\r\n return this.zGizmo.scaleRatio;\r\n }\r\n }\r\n\r\n /**\r\n * Disposes of the gizmo\r\n */\r\n public dispose() {\r\n if (this.xGizmo) {\r\n this.xGizmo.dispose();\r\n }\r\n if (this.yGizmo) {\r\n this.yGizmo.dispose();\r\n }\r\n if (this.zGizmo) {\r\n this.zGizmo.dispose();\r\n }\r\n this.onDragStartObservable.clear();\r\n this.onDragEndObservable.clear();\r\n if (this.attachedMesh && this.attachedMesh.name === AUTO_MESH_NAME) {\r\n this.attachedMesh.dispose();\r\n }\r\n }\r\n\r\n /**\r\n * CustomMeshes are not supported by this gizmo\r\n * @param mesh The mesh to replace the default mesh of the gizmo\r\n */\r\n public setCustomMesh(mesh: Mesh) {\r\n Logger.Error(\r\n 'Custom meshes are not supported on this gizmo, please set the custom meshes on the gizmos contained within this one (gizmo.xGizmo, gizmo.yGizmo, gizmo.zGizmo)'\r\n );\r\n }\r\n}\r\n","import { Color3, Quaternion, Space, Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { Logger } from '@babylonjs/core/Misc/logger';\r\nimport { Observable } from '@babylonjs/core/Misc/observable';\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { Nullable } from '@babylonjs/core/types';\r\nimport { SpaceNode } from '@models/classes/meshes/space-node';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { rotationToVector } from '@view/helpers/math-helper';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { KbGizmo } from './gizmo';\r\nimport { KbPlaneRotationGizmo } from './plane-rotation-gizmo';\r\n\r\nconst tessellation = 32;\r\n\r\n/**\r\n * Gizmo that enables rotating a mesh along 3 axis\r\n */\r\nexport class KbRotationGizmo extends KbGizmo {\r\n /**\r\n * Internal gizmo used for interactions on the x axis\r\n */\r\n public xGizmo: KbPlaneRotationGizmo | undefined;\r\n /**\r\n * Internal gizmo used for interactions on the y axis\r\n */\r\n public yGizmo: KbPlaneRotationGizmo | undefined;\r\n /**\r\n * Internal gizmo used for interactions on the z axis\r\n */\r\n public zGizmo: KbPlaneRotationGizmo | undefined;\r\n\r\n /** Fires an event when any of it's sub gizmos are dragged */\r\n public onDragStartObservable = new Observable();\r\n /** Fires an event when any of it's sub gizmos are released from dragging */\r\n public onDragObservable = new Observable<{ delta: Quaternion }>();\r\n public onDragEndObservable = new Observable<{ delta: Quaternion }>();\r\n\r\n public gizmoOffset: KbVector | undefined;\r\n private _attachedSpaceNode: SpaceNode | undefined;\r\n private _positionSpace: TransformNode | undefined;\r\n\r\n attachTo(target: SpaceNode | KbVector, positionSpace?: SpaceNode, placementOffset?: KbVector) {\r\n if (!positionSpace && target instanceof SpaceNode && target._parent) {\r\n positionSpace = target._parent;\r\n }\r\n if (positionSpace) {\r\n this._positionSpace = this.kbViewer.getRendererNode(positionSpace);\r\n }\r\n\r\n if (target instanceof SpaceNode) {\r\n this._attachedSpaceNode = target;\r\n this._updateGizmoPosition();\r\n } else {\r\n this._updateGizmoPosition(target, placementOffset);\r\n }\r\n }\r\n updateGizmoPosition(changeTarget?: KbVector, placementOffset?: KbVector) {\r\n // Need to wait for the next frame because this can be called before the babylon object is actually updated\r\n this.kbViewer.onNextFrame(() => {\r\n this._updateGizmoPosition(changeTarget, placementOffset);\r\n });\r\n }\r\n\r\n private _updateGizmoPosition(changeTarget?: KbVector, placementOffset?: KbVector) {\r\n const positionSpaceTransform = this._positionSpace?.computeWorldMatrix(true);\r\n if (changeTarget) {\r\n // If target is a vector, then we want to rotate it as a direction, so set the gizmo position to the origin to calculations are done from there\r\n // But store the position as the offset because we want the gizmo itself to be visually placed on the position space origin\r\n this.attachMeshAtNodeSpace(\r\n this.gizmoLayer.utilityLayerScene,\r\n placementOffset || KbVector.Zero(),\r\n positionSpaceTransform\r\n );\r\n\r\n const corr = rotationToVector(changeTarget.toVec3(), Vector3.Up());\r\n this.gizmoOffset = KbVector.FromVec3(this.attachedMesh!.position);\r\n this.attachedMesh!.position = Vector3.Zero();\r\n this.attachedMesh!.rotationQuaternion?.multiplyInPlace(corr);\r\n this.attachedMesh!.computeWorldMatrix(true);\r\n } else if (this._attachedSpaceNode) {\r\n this.attachMeshAtNodeSpace(\r\n this.gizmoLayer.utilityLayerScene,\r\n this._attachedSpaceNode.position.add(this._attachedSpaceNode.pivot),\r\n positionSpaceTransform\r\n );\r\n\r\n let currentRotation: Quaternion;\r\n // Rotation gizmo should be rotated to match the current rotation if the target is a node\r\n if (this._attachedSpaceNode.useEuler) {\r\n currentRotation = this._attachedSpaceNode.eulerRotation\r\n .toVec3()\r\n .scale(Math.PI / 180)\r\n .toQuaternion();\r\n } else {\r\n currentRotation = rotationToVector(this._attachedSpaceNode.rotationAxis.toVec3());\r\n }\r\n this.attachedMesh!.rotationQuaternion = (\r\n positionSpaceTransform\r\n ? Quaternion.FromRotationMatrix(positionSpaceTransform.getRotationMatrix())\r\n : Quaternion.Identity()\r\n ).multiply(currentRotation);\r\n }\r\n this.gizmoOffset && this.attachedMesh!.position.addInPlace(this.gizmoOffset.toVec3());\r\n super.updateGizmoPosition();\r\n }\r\n\r\n public get attachedMesh() {\r\n return (this.xGizmo || this.yGizmo || this.zGizmo)!.attachedMesh;\r\n }\r\n public set attachedMesh(mesh: Nullable) {\r\n if (this.xGizmo) {\r\n this.xGizmo.attachedMesh = mesh;\r\n }\r\n if (this.yGizmo) {\r\n this.yGizmo.attachedMesh = mesh;\r\n }\r\n if (this.zGizmo) {\r\n this.zGizmo.attachedMesh = mesh;\r\n }\r\n if (mesh) {\r\n this._rootMesh.setPivotPoint(mesh.getPivotPoint(), Space.WORLD);\r\n }\r\n }\r\n\r\n private rotationChange: Quaternion;\r\n /**\r\n * Creates a RotationGizmo\r\n * @param gizmoLayer The utility layer the gizmo will be added to\r\n * @param tessellation Amount of tessellation to be used when creating rotation circles\r\n */\r\n constructor(\r\n private kbViewer: KbViewer,\r\n gizmoLayer: UtilityLayerRenderer = UtilityLayerRenderer.DefaultUtilityLayer,\r\n axes?: Record<'x' | 'y' | 'z', boolean>,\r\n axisVisibility?: [boolean, boolean, boolean]\r\n ) {\r\n super(gizmoLayer);\r\n let activeGizmos = [];\r\n if (axisVisibility) {\r\n if (axisVisibility[0]) {\r\n this.xGizmo = new KbPlaneRotationGizmo(\r\n new Vector3(1, 0, 0),\r\n Color3.Red().scale(0.5),\r\n gizmoLayer,\r\n tessellation\r\n );\r\n this.xGizmo.updateGizmoRotationToMatchAttachedMesh = axisVisibility[0];\r\n this.xGizmo.updateGizmoPositionToMatchAttachedMesh = axisVisibility[0];\r\n activeGizmos.push(this.xGizmo);\r\n }\r\n if (axisVisibility[1]) {\r\n this.yGizmo = new KbPlaneRotationGizmo(\r\n new Vector3(0, 1, 0),\r\n Color3.Green().scale(0.5),\r\n gizmoLayer,\r\n tessellation\r\n );\r\n this.yGizmo.updateGizmoRotationToMatchAttachedMesh = axisVisibility[1];\r\n this.yGizmo.updateGizmoPositionToMatchAttachedMesh = axisVisibility[1];\r\n activeGizmos.push(this.yGizmo);\r\n }\r\n if (axisVisibility[2]) {\r\n this.zGizmo = new KbPlaneRotationGizmo(\r\n new Vector3(0, 0, 1),\r\n Color3.Blue().scale(0.5),\r\n gizmoLayer,\r\n tessellation\r\n );\r\n this.zGizmo.updateGizmoRotationToMatchAttachedMesh = axisVisibility[2];\r\n this.zGizmo.updateGizmoPositionToMatchAttachedMesh = axisVisibility[2];\r\n activeGizmos.push(this.zGizmo);\r\n }\r\n } else {\r\n this.updateGizmoRotationToMatchAttachedMesh = true;\r\n if (!axes || axes.x) {\r\n this.xGizmo = new KbPlaneRotationGizmo(\r\n new Vector3(1, 0, 0),\r\n Color3.Red().scale(0.5),\r\n gizmoLayer,\r\n tessellation\r\n );\r\n this.xGizmo.updateGizmoRotationToMatchAttachedMesh = true;\r\n this.xGizmo.updateGizmoPositionToMatchAttachedMesh = true;\r\n }\r\n if (!axes || axes.y) {\r\n this.yGizmo = new KbPlaneRotationGizmo(\r\n new Vector3(0, 1, 0),\r\n Color3.Green().scale(0.5),\r\n gizmoLayer,\r\n tessellation\r\n );\r\n this.yGizmo.updateGizmoRotationToMatchAttachedMesh = true;\r\n this.yGizmo.updateGizmoPositionToMatchAttachedMesh = true;\r\n }\r\n if (!axes || axes.z) {\r\n this.zGizmo = new KbPlaneRotationGizmo(\r\n new Vector3(0, 0, 1),\r\n Color3.Blue().scale(0.5),\r\n gizmoLayer,\r\n tessellation\r\n );\r\n this.zGizmo.updateGizmoRotationToMatchAttachedMesh = true;\r\n this.zGizmo.updateGizmoPositionToMatchAttachedMesh = true;\r\n }\r\n activeGizmos = [this.xGizmo, this.yGizmo, this.zGizmo];\r\n }\r\n\r\n this.rotationChange = Quaternion.Identity();\r\n\r\n // Relay drag events\r\n activeGizmos.forEach(gizmo => {\r\n if (gizmo) {\r\n gizmo.dragBehavior.onDragStartObservable.add(() => {\r\n this.onDragStartObservable.notifyObservers();\r\n this.rotationChange = Quaternion.Identity();\r\n });\r\n gizmo.updateObservable.add(event => {\r\n this.rotationChange.multiplyInPlace(event.delta);\r\n this.onDragObservable.notifyObservers({ delta: event.delta });\r\n });\r\n gizmo.dragBehavior.onDragEndObservable.add(() => {\r\n this.onDragEndObservable.notifyObservers({ delta: this.rotationChange });\r\n });\r\n }\r\n });\r\n\r\n this.attachedMesh = null;\r\n }\r\n\r\n /**\r\n * Drag distance in babylon units that the gizmo will snap to when dragged (Default: 0)\r\n */\r\n public set snapDistance(value: number) {\r\n if (this.xGizmo) {\r\n this.xGizmo.snapDistance = value;\r\n }\r\n if (this.yGizmo) {\r\n this.yGizmo.snapDistance = value;\r\n }\r\n if (this.zGizmo) {\r\n this.zGizmo.snapDistance = value;\r\n }\r\n }\r\n public get snapDistance() {\r\n return (this.xGizmo || this.yGizmo || this.zGizmo)!.snapDistance;\r\n }\r\n\r\n /**\r\n * Ratio for the scale of the gizmo (Default: 1)\r\n */\r\n public set scaleRatio(value: number) {\r\n if (this.xGizmo) {\r\n this.xGizmo.scaleRatio = value;\r\n }\r\n if (this.yGizmo) {\r\n this.yGizmo.scaleRatio = value;\r\n }\r\n if (this.zGizmo) {\r\n this.zGizmo.scaleRatio = value;\r\n }\r\n }\r\n public get scaleRatio() {\r\n return (this.xGizmo || this.yGizmo || this.zGizmo)!.scaleRatio;\r\n }\r\n\r\n /**\r\n * Disposes of the gizmo\r\n */\r\n public dispose() {\r\n if (this.xGizmo) {\r\n this.xGizmo.dispose();\r\n }\r\n if (this.yGizmo) {\r\n this.yGizmo.dispose();\r\n }\r\n if (this.zGizmo) {\r\n this.zGizmo.dispose();\r\n }\r\n this.onDragStartObservable.clear();\r\n this.onDragEndObservable.clear();\r\n }\r\n\r\n /**\r\n * CustomMeshes are not supported by this gizmo\r\n * @param mesh The mesh to replace the default mesh of the gizmo\r\n */\r\n public setCustomMesh(mesh: Mesh) {\r\n Logger.Error(\r\n 'Custom meshes are not supported on this gizmo, please set the custom meshes on the gizmos contained within this one (gizmo.xGizmo, gizmo.yGizmo, gizmo.zGizmo)'\r\n );\r\n }\r\n}\r\n","import { PointerDragBehavior } from '@babylonjs/core/Behaviors/Meshes/pointerDragBehavior';\r\nimport { PointerInfo } from '@babylonjs/core/Events/pointerEvents';\r\nimport { StandardMaterial } from '@babylonjs/core/Materials/standardMaterial';\r\nimport { Color3, Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { BoxBuilder } from '@babylonjs/core/Meshes/Builders/boxBuilder';\r\nimport { CylinderBuilder } from '@babylonjs/core/Meshes/Builders/cylinderBuilder';\r\nimport { LinesMesh } from '@babylonjs/core/Meshes/linesMesh';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { Observable, Observer } from '@babylonjs/core/Misc/observable';\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { Nullable } from '@babylonjs/core/types';\r\nimport { KbGizmo } from './gizmo';\r\n\r\n/**\r\n * Single axis scale gizmo\r\n */\r\nexport class KbAxisScaleGizmo extends KbGizmo {\r\n private _coloredMaterial: StandardMaterial;\r\n /**\r\n * Drag behavior responsible for the gizmos dragging interactions\r\n */\r\n public dragBehavior: PointerDragBehavior;\r\n private _pointerObserver: Nullable> = null;\r\n /**\r\n * Scale distance in babylon units that the gizmo will snap to when dragged (Default: 0)\r\n */\r\n public snapDistance = 0;\r\n /**\r\n * Event that fires each time the gizmo snaps to a new location.\r\n * * snapDistance is the the change in distance\r\n */\r\n public onSnapObservable = new Observable<{ snapDistance: number }>();\r\n public updateObservable = new Observable<{ delta: Vector3 }>();\r\n /**\r\n * If the scaling operation should be done on all axis (default: false)\r\n */\r\n public uniformScaling = false;\r\n /**\r\n * Creates an AxisScaleGizmo\r\n * @param gizmoLayer The utility layer the gizmo will be added to\r\n * @param dragAxis The axis which the gizmo will be able to scale on\r\n * @param color The color of the gizmo\r\n */\r\n constructor(\r\n dragAxis: Vector3,\r\n color: Color3 = Color3.Gray(),\r\n gizmoLayer: UtilityLayerRenderer = UtilityLayerRenderer.DefaultUtilityLayer\r\n ) {\r\n super(gizmoLayer);\r\n\r\n // Create Material\r\n this._coloredMaterial = new StandardMaterial('', gizmoLayer.utilityLayerScene);\r\n this._coloredMaterial.disableLighting = true;\r\n this._coloredMaterial.emissiveColor = color;\r\n // this._coloredMaterial.diffuseColor = color;\r\n // this._coloredMaterial.specularColor = color.subtract(new Color3(0.1, 0.1, 0.1));\r\n\r\n const hoverMaterial = new StandardMaterial('', gizmoLayer.utilityLayerScene);\r\n hoverMaterial.disableLighting = true;\r\n hoverMaterial.emissiveColor = color.add(new Color3(0.3, 0.3, 0.3));\r\n // hoverMaterial.diffuseColor = color.add(new Color3(0.3, 0.3, 0.3));\r\n\r\n // Build mesh on root node\r\n const arrow = new AbstractMesh('', gizmoLayer.utilityLayerScene);\r\n const arrowMesh = BoxBuilder.CreateBox('yPosMesh', { size: 0.4 }, gizmoLayer.utilityLayerScene);\r\n const arrowTail = CylinderBuilder.CreateCylinder(\r\n 'cylinder',\r\n { diameterTop: 0.005, height: 0.275, diameterBottom: 0.005, tessellation: 96 },\r\n gizmoLayer.utilityLayerScene\r\n );\r\n arrowTail.material = this._coloredMaterial;\r\n arrow.addChild(arrowMesh);\r\n arrow.addChild(arrowTail);\r\n\r\n // Position arrow pointing in its drag axis\r\n arrowMesh.scaling.scaleInPlace(0.1);\r\n arrowMesh.material = this._coloredMaterial;\r\n arrowMesh.rotation.x = Math.PI / 2;\r\n arrowMesh.position.z += 0.3;\r\n arrowTail.position.z += 0.275 / 2;\r\n arrowTail.rotation.x = Math.PI / 2;\r\n arrow.lookAt(this._rootMesh.position.add(dragAxis));\r\n this._rootMesh.addChild(arrow);\r\n arrow.scaling.scaleInPlace(1 / 3);\r\n\r\n // Add drag behavior to handle events when the gizmo is dragged\r\n this.dragBehavior = new PointerDragBehavior({ dragAxis: dragAxis });\r\n this.dragBehavior.moveAttached = false;\r\n this._rootMesh.addBehavior(this.dragBehavior);\r\n\r\n let currentSnapDragDistance = 0;\r\n const tmpVector = new Vector3();\r\n const tmpSnapEvent = { snapDistance: 0 };\r\n this.dragBehavior.onDragObservable.add(event => {\r\n if (this.attachedMesh) {\r\n // Drag strength is modified by the scale of the gizmo (eg. for small objects like boombox the strength will be increased to match the behavior of larger objects)\r\n const dragStrength = event.dragDistance * ((this.scaleRatio * 8) / this._rootMesh.scaling.length());\r\n\r\n // Snapping logic\r\n let snapped = false;\r\n let dragSteps = 0;\r\n if (this.uniformScaling) {\r\n this.attachedMesh.scaling.normalizeToRef(tmpVector);\r\n if (tmpVector.y < 0) {\r\n tmpVector.scaleInPlace(-1);\r\n }\r\n } else {\r\n tmpVector.copyFrom(dragAxis);\r\n }\r\n if (this.snapDistance == 0) {\r\n tmpVector.scaleToRef(dragStrength, tmpVector);\r\n } else {\r\n currentSnapDragDistance += dragStrength;\r\n if (Math.abs(currentSnapDragDistance) > this.snapDistance) {\r\n dragSteps = Math.floor(Math.abs(currentSnapDragDistance) / this.snapDistance);\r\n if (currentSnapDragDistance < 0) {\r\n dragSteps *= -1;\r\n }\r\n currentSnapDragDistance = currentSnapDragDistance % this.snapDistance;\r\n tmpVector.scaleToRef(this.snapDistance * dragSteps, tmpVector);\r\n snapped = true;\r\n } else {\r\n tmpVector.scaleInPlace(0);\r\n }\r\n }\r\n\r\n this.attachedMesh.scaling.addInPlace(tmpVector);\r\n this.updateObservable.notifyObservers({ delta: tmpVector });\r\n\r\n if (snapped) {\r\n tmpSnapEvent.snapDistance = this.snapDistance * dragSteps;\r\n this.onSnapObservable.notifyObservers(tmpSnapEvent);\r\n }\r\n }\r\n });\r\n\r\n this._pointerObserver = gizmoLayer.utilityLayerScene.onPointerObservable.add(pointerInfo => {\r\n if (this._customMeshSet) {\r\n return;\r\n }\r\n const isHovered =\r\n pointerInfo.pickInfo &&\r\n this._rootMesh.getChildMeshes().indexOf(pointerInfo.pickInfo.pickedMesh) != -1;\r\n const material = isHovered ? hoverMaterial : this._coloredMaterial;\r\n this._rootMesh.getChildMeshes().forEach(m => {\r\n m.material = material;\r\n if ((m).color) {\r\n (m).color = material.diffuseColor;\r\n }\r\n });\r\n });\r\n\r\n // var light = gizmoLayer._getSharedGizmoLight();\r\n // light.includedOnlyMeshes = light.includedOnlyMeshes.concat(this._rootMesh.getChildMeshes());\r\n }\r\n\r\n protected _attachedMeshChanged(value: Nullable) {\r\n if (this.dragBehavior) {\r\n this.dragBehavior.enabled = value ? true : false;\r\n }\r\n }\r\n\r\n /**\r\n * Disposes of the gizmo\r\n */\r\n public dispose() {\r\n this.onSnapObservable.clear();\r\n this.gizmoLayer.utilityLayerScene.onPointerObservable.remove(this._pointerObserver);\r\n this.dragBehavior.detach();\r\n super.dispose();\r\n }\r\n\r\n /**\r\n * Disposes and replaces the current meshes in the gizmo with the specified mesh\r\n * @param mesh The mesh to replace the default mesh of the gizmo\r\n * @param useGizmoMaterial If the gizmo's default material should be used (default: false)\r\n */\r\n public setCustomMesh(mesh: Mesh, useGizmoMaterial = false) {\r\n super.setCustomMesh(mesh);\r\n if (useGizmoMaterial) {\r\n this._rootMesh.getChildMeshes().forEach(m => {\r\n m.material = this._coloredMaterial;\r\n if ((m).color) {\r\n (m).color = this._coloredMaterial.diffuseColor;\r\n }\r\n });\r\n this._customMeshSet = false;\r\n }\r\n }\r\n}\r\n","import { Color3, Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { PolyhedronBuilder } from '@babylonjs/core/Meshes/Builders/polyhedronBuilder';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { Logger } from '@babylonjs/core/Misc/logger';\r\nimport { Observable } from '@babylonjs/core/Misc/observable';\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { Nullable } from '@babylonjs/core/types';\r\nimport { SpaceNode } from '@models/classes/meshes/space-node';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { KbAxisScaleGizmo } from './axis-scale-gizmo';\r\nimport { KbGizmo } from './gizmo';\r\n/**\r\n * Gizmo that enables scaling a mesh along 3 axis\r\n */\r\nexport class KbScaleGizmo extends KbGizmo {\r\n /**\r\n * Internal gizmo used for interactions on the x axis\r\n */\r\n public xGizmo: KbAxisScaleGizmo;\r\n /**\r\n * Internal gizmo used for interactions on the y axis\r\n */\r\n public yGizmo: KbAxisScaleGizmo;\r\n /**\r\n * Internal gizmo used for interactions on the z axis\r\n */\r\n public zGizmo: KbAxisScaleGizmo;\r\n\r\n /**\r\n * Internal gizmo used to scale all axis equally\r\n */\r\n public uniformScaleGizmo: KbAxisScaleGizmo;\r\n\r\n /** Fires an event when any of it's sub gizmos are dragged */\r\n public onDragStartObservable = new Observable();\r\n /** Fires an event while any of it's sub gizmos are dragged */\r\n public onDragObservable = new Observable<{ delta: Vector3 }>();\r\n /** Fires an event when any of it's sub gizmos are released from dragging */\r\n public onDragEndObservable = new Observable<{ delta: Vector3 }>();\r\n\r\n private _attachedSpaceNode: SpaceNode | undefined;\r\n private _positionSpace: TransformNode | undefined;\r\n\r\n public gizmoOffset: KbVector | undefined;\r\n\r\n attachTo(target: SpaceNode | KbVector, positionSpace?: SpaceNode) {\r\n if (!positionSpace && target instanceof SpaceNode && target._parent) {\r\n positionSpace = target._parent;\r\n }\r\n if (positionSpace) {\r\n this._positionSpace = this.kbViewer.getRendererNode(positionSpace);\r\n }\r\n if (target instanceof SpaceNode) {\r\n this._attachedSpaceNode = target;\r\n this._updateGizmoPosition();\r\n } else {\r\n this._updateGizmoPosition(target);\r\n }\r\n }\r\n updateGizmoPosition(changeTarget?: KbVector) {\r\n // Need to wait for the next frame because this can be called before the babylon object is actually updated\r\n this.kbViewer.onNextFrame(() => {\r\n this._updateGizmoPosition(changeTarget);\r\n });\r\n }\r\n\r\n private _updateGizmoPosition(changeTarget?: KbVector) {\r\n const positionSpaceTransform = this._positionSpace?.computeWorldMatrix(true);\r\n if (changeTarget) {\r\n // If target is a vector, then we want to rotate it as a direction, so set the gizmo position to the origin to calculations are done from there\r\n // But store the position as the offset because we want the gizmo itself to be visually placed on the position space origin\r\n this.attachMeshAtNodeSpace(this.gizmoLayer.utilityLayerScene, changeTarget, positionSpaceTransform);\r\n\r\n this.gizmoOffset = KbVector.FromVec3(this.attachedMesh!.position);\r\n } else if (this._attachedSpaceNode) {\r\n this.attachMeshAtNodeSpace(\r\n this.gizmoLayer.utilityLayerScene,\r\n this._attachedSpaceNode,\r\n positionSpaceTransform\r\n );\r\n\r\n // Put the gizmo in the same position as the pivot, in world space\r\n this.attachedMesh!.position.addInPlace(this._attachedSpaceNode.pivot.toVec3());\r\n }\r\n super.updateGizmoPosition();\r\n }\r\n\r\n public get attachedMesh() {\r\n return this.xGizmo.attachedMesh;\r\n }\r\n public set attachedMesh(mesh: Nullable) {\r\n if (this.xGizmo) {\r\n this.xGizmo.attachedMesh = mesh;\r\n this.yGizmo.attachedMesh = mesh;\r\n this.zGizmo.attachedMesh = mesh;\r\n this.uniformScaleGizmo.attachedMesh = mesh;\r\n }\r\n }\r\n\r\n private scaleChange = Vector3.Zero();\r\n\r\n /**\r\n * Creates a ScaleGizmo\r\n * @param gizmoLayer The utility layer the gizmo will be added to\r\n */\r\n constructor(\r\n private kbViewer: KbViewer,\r\n gizmoLayer: UtilityLayerRenderer = UtilityLayerRenderer.DefaultUtilityLayer\r\n ) {\r\n super(gizmoLayer);\r\n this.xGizmo = new KbAxisScaleGizmo(new Vector3(1, 0, 0), Color3.Red().scale(0.5), gizmoLayer);\r\n this.yGizmo = new KbAxisScaleGizmo(new Vector3(0, 1, 0), Color3.Green().scale(0.5), gizmoLayer);\r\n this.zGizmo = new KbAxisScaleGizmo(new Vector3(0, 0, 1), Color3.Blue().scale(0.5), gizmoLayer);\r\n\r\n // Create uniform scale gizmo\r\n this.uniformScaleGizmo = new KbAxisScaleGizmo(new Vector3(0, 1, 0), Color3.Yellow().scale(0.5), gizmoLayer);\r\n this.uniformScaleGizmo.updateGizmoRotationToMatchAttachedMesh = false;\r\n this.uniformScaleGizmo.uniformScaling = true;\r\n const uniformScalingMesh = PolyhedronBuilder.CreatePolyhedron(\r\n '',\r\n { type: 1 },\r\n this.uniformScaleGizmo.gizmoLayer.utilityLayerScene\r\n );\r\n uniformScalingMesh.scaling.scaleInPlace(0.03);\r\n uniformScalingMesh.visibility = 0;\r\n const octahedron = PolyhedronBuilder.CreatePolyhedron(\r\n '',\r\n { type: 1 },\r\n this.uniformScaleGizmo.gizmoLayer.utilityLayerScene\r\n );\r\n octahedron.scaling.scaleInPlace(0.007);\r\n uniformScalingMesh.addChild(octahedron);\r\n this.uniformScaleGizmo.setCustomMesh(uniformScalingMesh, true);\r\n // var light = gizmoLayer._getSharedGizmoLight();\r\n // light.includedOnlyMeshes = light.includedOnlyMeshes.concat(octahedron);\r\n\r\n // Relay drag events\r\n [this.xGizmo, this.yGizmo, this.zGizmo, this.uniformScaleGizmo].forEach(gizmo => {\r\n gizmo.dragBehavior.onDragStartObservable.add(() => {\r\n this.onDragStartObservable.notifyObservers();\r\n this.scaleChange.setAll(0);\r\n });\r\n gizmo.updateObservable.add(event => {\r\n this.scaleChange.addInPlace(event.delta);\r\n this.onDragObservable.notifyObservers({ delta: event.delta });\r\n });\r\n gizmo.dragBehavior.onDragEndObservable.add(() => {\r\n this.onDragEndObservable.notifyObservers({ delta: this.scaleChange });\r\n });\r\n });\r\n\r\n this.attachedMesh = null;\r\n }\r\n\r\n public set updateGizmoRotationToMatchAttachedMesh(value: boolean) {\r\n if (!value) {\r\n Logger.Warn('Setting updateGizmoRotationToMatchAttachedMesh = false on scaling gizmo is not supported.');\r\n }\r\n if (this.xGizmo) {\r\n this.xGizmo.updateGizmoRotationToMatchAttachedMesh = value;\r\n this.yGizmo.updateGizmoRotationToMatchAttachedMesh = value;\r\n this.zGizmo.updateGizmoRotationToMatchAttachedMesh = value;\r\n }\r\n }\r\n public get updateGizmoRotationToMatchAttachedMesh() {\r\n return this.xGizmo.updateGizmoRotationToMatchAttachedMesh;\r\n }\r\n\r\n /**\r\n * Drag distance in babylon units that the gizmo will snap to when dragged (Default: 0)\r\n */\r\n public set snapDistance(value: number) {\r\n if (this.xGizmo) {\r\n this.xGizmo.snapDistance = value;\r\n this.yGizmo.snapDistance = value;\r\n this.zGizmo.snapDistance = value;\r\n this.uniformScaleGizmo.snapDistance = value;\r\n }\r\n }\r\n public get snapDistance() {\r\n return this.xGizmo.snapDistance;\r\n }\r\n\r\n /**\r\n * Ratio for the scale of the gizmo (Default: 1)\r\n */\r\n public set scaleRatio(value: number) {\r\n if (this.xGizmo) {\r\n this.xGizmo.scaleRatio = value;\r\n this.yGizmo.scaleRatio = value;\r\n this.zGizmo.scaleRatio = value;\r\n this.uniformScaleGizmo.scaleRatio = value;\r\n }\r\n }\r\n public get scaleRatio() {\r\n return this.xGizmo.scaleRatio;\r\n }\r\n\r\n /**\r\n * Disposes of the gizmo\r\n */\r\n public dispose() {\r\n this.xGizmo.dispose();\r\n this.yGizmo.dispose();\r\n this.zGizmo.dispose();\r\n this.uniformScaleGizmo.dispose();\r\n this.onDragStartObservable.clear();\r\n this.onDragEndObservable.clear();\r\n }\r\n}\r\n","import type { Color3 } from \"../Maths/math.color\";\r\nimport { Color4 } from \"../Maths/math.color\";\r\n\r\n/** Interface used by value gradients (color, factor, ...) */\r\nexport interface IValueGradient {\r\n /**\r\n * Gets or sets the gradient value (between 0 and 1)\r\n */\r\n gradient: number;\r\n}\r\n\r\n/** Class used to store color4 gradient */\r\nexport class ColorGradient implements IValueGradient {\r\n /**\r\n * Creates a new color4 gradient\r\n * @param gradient gets or sets the gradient value (between 0 and 1)\r\n * @param color1 gets or sets first associated color\r\n * @param color2 gets or sets first second color\r\n */\r\n public constructor(\r\n /**\r\n * Gets or sets the gradient value (between 0 and 1)\r\n */\r\n public gradient: number,\r\n /**\r\n * Gets or sets first associated color\r\n */\r\n public color1: Color4,\r\n /**\r\n * Gets or sets second associated color\r\n */\r\n public color2?: Color4\r\n ) {}\r\n\r\n /**\r\n * Will get a color picked randomly between color1 and color2.\r\n * If color2 is undefined then color1 will be used\r\n * @param result defines the target Color4 to store the result in\r\n */\r\n public getColorToRef(result: Color4) {\r\n if (!this.color2) {\r\n result.copyFrom(this.color1);\r\n return;\r\n }\r\n\r\n Color4.LerpToRef(this.color1, this.color2, Math.random(), result);\r\n }\r\n}\r\n\r\n/** Class used to store color 3 gradient */\r\nexport class Color3Gradient implements IValueGradient {\r\n /**\r\n * Creates a new color3 gradient\r\n * @param gradient gets or sets the gradient value (between 0 and 1)\r\n * @param color gets or sets associated color\r\n */\r\n public constructor(\r\n /**\r\n * Gets or sets the gradient value (between 0 and 1)\r\n */\r\n public gradient: number,\r\n /**\r\n * Gets or sets the associated color\r\n */\r\n public color: Color3\r\n ) {}\r\n}\r\n\r\n/** Class used to store factor gradient */\r\nexport class FactorGradient implements IValueGradient {\r\n /**\r\n * Creates a new factor gradient\r\n * @param gradient gets or sets the gradient value (between 0 and 1)\r\n * @param factor1 gets or sets first associated factor\r\n * @param factor2 gets or sets second associated factor\r\n */\r\n public constructor(\r\n /**\r\n * Gets or sets the gradient value (between 0 and 1)\r\n */\r\n public gradient: number,\r\n /**\r\n * Gets or sets first associated factor\r\n */\r\n public factor1: number,\r\n /**\r\n * Gets or sets second associated factor\r\n */\r\n public factor2?: number\r\n ) {}\r\n\r\n /**\r\n * Will get a number picked randomly between factor1 and factor2.\r\n * If factor2 is undefined then factor1 will be used\r\n * @returns the picked number\r\n */\r\n public getFactor(): number {\r\n if (this.factor2 === undefined || this.factor2 === this.factor1) {\r\n return this.factor1;\r\n }\r\n\r\n return this.factor1 + (this.factor2 - this.factor1) * Math.random();\r\n }\r\n}\r\n\r\n/**\r\n * Helper used to simplify some generic gradient tasks\r\n */\r\nexport class GradientHelper {\r\n /**\r\n * Gets the current gradient from an array of IValueGradient\r\n * @param ratio defines the current ratio to get\r\n * @param gradients defines the array of IValueGradient\r\n * @param updateFunc defines the callback function used to get the final value from the selected gradients\r\n */\r\n public static GetCurrentGradient(ratio: number, gradients: IValueGradient[], updateFunc: (current: IValueGradient, next: IValueGradient, scale: number) => void) {\r\n // Use last index if over\r\n if (gradients[0].gradient > ratio) {\r\n updateFunc(gradients[0], gradients[0], 1.0);\r\n return;\r\n }\r\n\r\n for (let gradientIndex = 0; gradientIndex < gradients.length - 1; gradientIndex++) {\r\n const currentGradient = gradients[gradientIndex];\r\n const nextGradient = gradients[gradientIndex + 1];\r\n\r\n if (ratio >= currentGradient.gradient && ratio <= nextGradient.gradient) {\r\n const scale = (ratio - currentGradient.gradient) / (nextGradient.gradient - currentGradient.gradient);\r\n updateFunc(currentGradient, nextGradient, scale);\r\n return;\r\n }\r\n }\r\n\r\n // Use last index if over\r\n const lastIndex = gradients.length - 1;\r\n updateFunc(gradients[lastIndex], gradients[lastIndex], 1.0);\r\n }\r\n}\r\n","import type { Nullable } from \"../types\";\r\nimport { Vector2, Vector3, TmpVectors, Vector4 } from \"../Maths/math.vector\";\r\nimport { Color4 } from \"../Maths/math.color\";\r\nimport { Scalar } from \"../Maths/math.scalar\";\r\nimport type { ParticleSystem } from \"./particleSystem\";\r\nimport type { SubEmitter } from \"./subEmitter\";\r\nimport type { ColorGradient, FactorGradient } from \"../Misc/gradients\";\r\n\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\n\r\n/**\r\n * A particle represents one of the element emitted by a particle system.\r\n * This is mainly define by its coordinates, direction, velocity and age.\r\n */\r\nexport class Particle {\r\n private static _Count = 0;\r\n /**\r\n * Unique ID of the particle\r\n */\r\n public id: number;\r\n /**\r\n * The world position of the particle in the scene.\r\n */\r\n public position = Vector3.Zero();\r\n\r\n /**\r\n * The world direction of the particle in the scene.\r\n */\r\n public direction = Vector3.Zero();\r\n\r\n /**\r\n * The color of the particle.\r\n */\r\n public color = new Color4(0, 0, 0, 0);\r\n\r\n /**\r\n * The color change of the particle per step.\r\n */\r\n public colorStep = new Color4(0, 0, 0, 0);\r\n\r\n /**\r\n * Defines how long will the life of the particle be.\r\n */\r\n public lifeTime = 1.0;\r\n\r\n /**\r\n * The current age of the particle.\r\n */\r\n public age = 0;\r\n\r\n /**\r\n * The current size of the particle.\r\n */\r\n public size = 0;\r\n\r\n /**\r\n * The current scale of the particle.\r\n */\r\n public scale = new Vector2(1, 1);\r\n\r\n /**\r\n * The current angle of the particle.\r\n */\r\n public angle = 0;\r\n\r\n /**\r\n * Defines how fast is the angle changing.\r\n */\r\n public angularSpeed = 0;\r\n\r\n /**\r\n * Defines the cell index used by the particle to be rendered from a sprite.\r\n */\r\n public cellIndex: number = 0;\r\n\r\n /**\r\n * The information required to support color remapping\r\n */\r\n public remapData: Vector4;\r\n\r\n /** @internal */\r\n public _randomCellOffset?: number;\r\n\r\n /** @internal */\r\n public _initialDirection: Nullable;\r\n\r\n /** @internal */\r\n public _attachedSubEmitters: Nullable> = null;\r\n\r\n /** @internal */\r\n public _initialStartSpriteCellID: number;\r\n /** @internal */\r\n public _initialEndSpriteCellID: number;\r\n /** @internal */\r\n public _initialSpriteCellLoop: boolean;\r\n\r\n /** @internal */\r\n public _currentColorGradient: Nullable;\r\n /** @internal */\r\n public _currentColor1 = new Color4(0, 0, 0, 0);\r\n /** @internal */\r\n public _currentColor2 = new Color4(0, 0, 0, 0);\r\n\r\n /** @internal */\r\n public _currentSizeGradient: Nullable;\r\n /** @internal */\r\n public _currentSize1 = 0;\r\n /** @internal */\r\n public _currentSize2 = 0;\r\n\r\n /** @internal */\r\n public _currentAngularSpeedGradient: Nullable;\r\n /** @internal */\r\n public _currentAngularSpeed1 = 0;\r\n /** @internal */\r\n public _currentAngularSpeed2 = 0;\r\n\r\n /** @internal */\r\n public _currentVelocityGradient: Nullable;\r\n /** @internal */\r\n public _currentVelocity1 = 0;\r\n /** @internal */\r\n public _currentVelocity2 = 0;\r\n\r\n /** @internal */\r\n public _currentLimitVelocityGradient: Nullable;\r\n /** @internal */\r\n public _currentLimitVelocity1 = 0;\r\n /** @internal */\r\n public _currentLimitVelocity2 = 0;\r\n\r\n /** @internal */\r\n public _currentDragGradient: Nullable;\r\n /** @internal */\r\n public _currentDrag1 = 0;\r\n /** @internal */\r\n public _currentDrag2 = 0;\r\n\r\n /** @internal */\r\n public _randomNoiseCoordinates1: Vector3;\r\n /** @internal */\r\n public _randomNoiseCoordinates2: Vector3;\r\n\r\n /** @internal */\r\n public _localPosition?: Vector3;\r\n\r\n /**\r\n * Creates a new instance Particle\r\n * @param particleSystem the particle system the particle belongs to\r\n */\r\n constructor(\r\n /**\r\n * The particle system the particle belongs to.\r\n */\r\n public particleSystem: ParticleSystem\r\n ) {\r\n this.id = Particle._Count++;\r\n if (!this.particleSystem.isAnimationSheetEnabled) {\r\n return;\r\n }\r\n\r\n this._updateCellInfoFromSystem();\r\n }\r\n\r\n private _updateCellInfoFromSystem(): void {\r\n this.cellIndex = this.particleSystem.startSpriteCellID;\r\n }\r\n\r\n /**\r\n * Defines how the sprite cell index is updated for the particle\r\n */\r\n public updateCellIndex(): void {\r\n let offsetAge = this.age;\r\n let changeSpeed = this.particleSystem.spriteCellChangeSpeed;\r\n\r\n if (this.particleSystem.spriteRandomStartCell) {\r\n if (this._randomCellOffset === undefined) {\r\n this._randomCellOffset = Math.random() * this.lifeTime;\r\n }\r\n\r\n if (changeSpeed === 0) {\r\n // Special case when speed = 0 meaning we want to stay on initial cell\r\n changeSpeed = 1;\r\n offsetAge = this._randomCellOffset;\r\n } else {\r\n offsetAge += this._randomCellOffset;\r\n }\r\n }\r\n\r\n const dist = this._initialEndSpriteCellID - this._initialStartSpriteCellID;\r\n let ratio: number;\r\n if (this._initialSpriteCellLoop) {\r\n ratio = Scalar.Clamp(((offsetAge * changeSpeed) % this.lifeTime) / this.lifeTime);\r\n } else {\r\n ratio = Scalar.Clamp((offsetAge * changeSpeed) / this.lifeTime);\r\n }\r\n this.cellIndex = (this._initialStartSpriteCellID + ratio * dist) | 0;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public _inheritParticleInfoToSubEmitter(subEmitter: SubEmitter) {\r\n if ((subEmitter.particleSystem.emitter).position) {\r\n const emitterMesh = subEmitter.particleSystem.emitter;\r\n emitterMesh.position.copyFrom(this.position);\r\n if (subEmitter.inheritDirection) {\r\n const temp = TmpVectors.Vector3[0];\r\n this.direction.normalizeToRef(temp);\r\n emitterMesh.setDirection(temp, 0, Math.PI / 2);\r\n }\r\n } else {\r\n const emitterPosition = subEmitter.particleSystem.emitter;\r\n emitterPosition.copyFrom(this.position);\r\n }\r\n // Set inheritedVelocityOffset to be used when new particles are created\r\n this.direction.scaleToRef(subEmitter.inheritedVelocityAmount / 2, TmpVectors.Vector3[0]);\r\n subEmitter.particleSystem._inheritedVelocityOffset.copyFrom(TmpVectors.Vector3[0]);\r\n }\r\n\r\n /** @internal */\r\n public _inheritParticleInfoToSubEmitters() {\r\n if (this._attachedSubEmitters && this._attachedSubEmitters.length > 0) {\r\n this._attachedSubEmitters.forEach((subEmitter) => {\r\n this._inheritParticleInfoToSubEmitter(subEmitter);\r\n });\r\n }\r\n }\r\n\r\n /** @internal */\r\n public _reset() {\r\n this.age = 0;\r\n this.id = Particle._Count++;\r\n this._currentColorGradient = null;\r\n this._currentSizeGradient = null;\r\n this._currentAngularSpeedGradient = null;\r\n this._currentVelocityGradient = null;\r\n this._currentLimitVelocityGradient = null;\r\n this._currentDragGradient = null;\r\n this.cellIndex = this.particleSystem.startSpriteCellID;\r\n this._randomCellOffset = undefined;\r\n }\r\n\r\n /**\r\n * Copy the properties of particle to another one.\r\n * @param other the particle to copy the information to.\r\n */\r\n public copyTo(other: Particle) {\r\n other.position.copyFrom(this.position);\r\n if (this._initialDirection) {\r\n if (other._initialDirection) {\r\n other._initialDirection.copyFrom(this._initialDirection);\r\n } else {\r\n other._initialDirection = this._initialDirection.clone();\r\n }\r\n } else {\r\n other._initialDirection = null;\r\n }\r\n other.direction.copyFrom(this.direction);\r\n if (this._localPosition) {\r\n if (other._localPosition) {\r\n other._localPosition.copyFrom(this._localPosition);\r\n } else {\r\n other._localPosition = this._localPosition.clone();\r\n }\r\n }\r\n other.color.copyFrom(this.color);\r\n other.colorStep.copyFrom(this.colorStep);\r\n other.lifeTime = this.lifeTime;\r\n other.age = this.age;\r\n other._randomCellOffset = this._randomCellOffset;\r\n other.size = this.size;\r\n other.scale.copyFrom(this.scale);\r\n other.angle = this.angle;\r\n other.angularSpeed = this.angularSpeed;\r\n other.particleSystem = this.particleSystem;\r\n other.cellIndex = this.cellIndex;\r\n other.id = this.id;\r\n other._attachedSubEmitters = this._attachedSubEmitters;\r\n if (this._currentColorGradient) {\r\n other._currentColorGradient = this._currentColorGradient;\r\n other._currentColor1.copyFrom(this._currentColor1);\r\n other._currentColor2.copyFrom(this._currentColor2);\r\n }\r\n if (this._currentSizeGradient) {\r\n other._currentSizeGradient = this._currentSizeGradient;\r\n other._currentSize1 = this._currentSize1;\r\n other._currentSize2 = this._currentSize2;\r\n }\r\n if (this._currentAngularSpeedGradient) {\r\n other._currentAngularSpeedGradient = this._currentAngularSpeedGradient;\r\n other._currentAngularSpeed1 = this._currentAngularSpeed1;\r\n other._currentAngularSpeed2 = this._currentAngularSpeed2;\r\n }\r\n if (this._currentVelocityGradient) {\r\n other._currentVelocityGradient = this._currentVelocityGradient;\r\n other._currentVelocity1 = this._currentVelocity1;\r\n other._currentVelocity2 = this._currentVelocity2;\r\n }\r\n if (this._currentLimitVelocityGradient) {\r\n other._currentLimitVelocityGradient = this._currentLimitVelocityGradient;\r\n other._currentLimitVelocity1 = this._currentLimitVelocity1;\r\n other._currentLimitVelocity2 = this._currentLimitVelocity2;\r\n }\r\n if (this._currentDragGradient) {\r\n other._currentDragGradient = this._currentDragGradient;\r\n other._currentDrag1 = this._currentDrag1;\r\n other._currentDrag2 = this._currentDrag2;\r\n }\r\n if (this.particleSystem.isAnimationSheetEnabled) {\r\n other._initialStartSpriteCellID = this._initialStartSpriteCellID;\r\n other._initialEndSpriteCellID = this._initialEndSpriteCellID;\r\n other._initialSpriteCellLoop = this._initialSpriteCellLoop;\r\n }\r\n if (this.particleSystem.useRampGradients) {\r\n if (other.remapData && this.remapData) {\r\n other.remapData.copyFrom(this.remapData);\r\n } else {\r\n other.remapData = new Vector4(0, 0, 0, 0);\r\n }\r\n }\r\n if (this._randomNoiseCoordinates1) {\r\n if (other._randomNoiseCoordinates1) {\r\n other._randomNoiseCoordinates1.copyFrom(this._randomNoiseCoordinates1);\r\n other._randomNoiseCoordinates2.copyFrom(this._randomNoiseCoordinates2);\r\n } else {\r\n other._randomNoiseCoordinates1 = this._randomNoiseCoordinates1.clone();\r\n other._randomNoiseCoordinates2 = this._randomNoiseCoordinates2.clone();\r\n }\r\n }\r\n }\r\n}\r\n","import { Vector3 } from \"../Maths/math.vector\";\r\nimport { _WarnImport } from \"../Misc/devTools\";\r\nimport type { ThinEngine } from \"../Engines/thinEngine\";\r\nimport { GetClass } from \"../Misc/typeStore\";\r\n\r\nimport type { Scene } from \"../scene\";\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { ParticleSystem } from \"../Particles/particleSystem\";\r\n\r\n/**\r\n * Type of sub emitter\r\n */\r\nexport enum SubEmitterType {\r\n /**\r\n * Attached to the particle over it's lifetime\r\n */\r\n ATTACHED,\r\n /**\r\n * Created when the particle dies\r\n */\r\n END,\r\n}\r\n\r\n/**\r\n * Sub emitter class used to emit particles from an existing particle\r\n */\r\nexport class SubEmitter {\r\n /**\r\n * Type of the submitter (Default: END)\r\n */\r\n public type = SubEmitterType.END;\r\n /**\r\n * If the particle should inherit the direction from the particle it's attached to. (+Y will face the direction the particle is moving) (Default: false)\r\n * Note: This only is supported when using an emitter of type Mesh\r\n */\r\n public inheritDirection = false;\r\n /**\r\n * How much of the attached particles speed should be added to the sub emitted particle (default: 0)\r\n */\r\n public inheritedVelocityAmount = 0;\r\n\r\n /**\r\n * Creates a sub emitter\r\n * @param particleSystem the particle system to be used by the sub emitter\r\n */\r\n constructor(\r\n /**\r\n * the particle system to be used by the sub emitter\r\n */\r\n public particleSystem: ParticleSystem\r\n ) {\r\n // Create mesh as emitter to support rotation\r\n if (!particleSystem.emitter || !(particleSystem.emitter).dispose) {\r\n const internalClass = GetClass(\"BABYLON.AbstractMesh\");\r\n particleSystem.emitter = new internalClass(\"SubemitterSystemEmitter\", particleSystem.getScene());\r\n particleSystem._disposeEmitterOnDispose = true;\r\n }\r\n }\r\n /**\r\n * Clones the sub emitter\r\n * @returns the cloned sub emitter\r\n */\r\n public clone(): SubEmitter {\r\n // Clone particle system\r\n let emitter = this.particleSystem.emitter;\r\n if (!emitter) {\r\n emitter = new Vector3();\r\n } else if (emitter instanceof Vector3) {\r\n emitter = emitter.clone();\r\n } else if (emitter.getClassName().indexOf(\"Mesh\") !== -1) {\r\n const internalClass = GetClass(\"BABYLON.Mesh\");\r\n emitter = new internalClass(\"\", emitter.getScene());\r\n (emitter! as any).isVisible = false;\r\n }\r\n const clone = new SubEmitter(this.particleSystem.clone(this.particleSystem.name, emitter));\r\n\r\n // Clone properties\r\n clone.particleSystem.name += \"Clone\";\r\n clone.type = this.type;\r\n clone.inheritDirection = this.inheritDirection;\r\n clone.inheritedVelocityAmount = this.inheritedVelocityAmount;\r\n\r\n clone.particleSystem._disposeEmitterOnDispose = true;\r\n clone.particleSystem.disposeOnStop = true;\r\n return clone;\r\n }\r\n\r\n /**\r\n * Serialize current object to a JSON object\r\n * @param serializeTexture defines if the texture must be serialized as well\r\n * @returns the serialized object\r\n */\r\n public serialize(serializeTexture: boolean = false): any {\r\n const serializationObject: any = {};\r\n\r\n serializationObject.type = this.type;\r\n serializationObject.inheritDirection = this.inheritDirection;\r\n serializationObject.inheritedVelocityAmount = this.inheritedVelocityAmount;\r\n serializationObject.particleSystem = this.particleSystem.serialize(serializeTexture);\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n public static _ParseParticleSystem(system: any, sceneOrEngine: Scene | ThinEngine, rootUrl: string, doNotStart = false): ParticleSystem {\r\n throw _WarnImport(\"ParseParticle\");\r\n }\r\n\r\n /**\r\n * Creates a new SubEmitter from a serialized JSON version\r\n * @param serializationObject defines the JSON object to read from\r\n * @param sceneOrEngine defines the hosting scene or the hosting engine\r\n * @param rootUrl defines the rootUrl for data loading\r\n * @returns a new SubEmitter\r\n */\r\n public static Parse(serializationObject: any, sceneOrEngine: Scene | ThinEngine, rootUrl: string): SubEmitter {\r\n const system = serializationObject.particleSystem;\r\n const subEmitter = new SubEmitter(SubEmitter._ParseParticleSystem(system, sceneOrEngine, rootUrl, true));\r\n subEmitter.type = serializationObject.type;\r\n subEmitter.inheritDirection = serializationObject.inheritDirection;\r\n subEmitter.inheritedVelocityAmount = serializationObject.inheritedVelocityAmount;\r\n subEmitter.particleSystem._isSubEmitter = true;\r\n\r\n return subEmitter;\r\n }\r\n\r\n /** Release associated resources */\r\n public dispose() {\r\n this.particleSystem.dispose();\r\n }\r\n}\r\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/clipPlaneFragmentDeclaration\";\nimport \"./ShadersInclude/imageProcessingDeclaration\";\nimport \"./ShadersInclude/logDepthDeclaration\";\nimport \"./ShadersInclude/helperFunctions\";\nimport \"./ShadersInclude/imageProcessingFunctions\";\nimport \"./ShadersInclude/clipPlaneFragment\";\nimport \"./ShadersInclude/logDepthFragment\";\n\nconst name = \"particlesPixelShader\";\nconst shader = `#ifdef LOGARITHMICDEPTH\n#extension GL_EXT_frag_depth : enable\n#endif\nvarying vec2 vUV;varying vec4 vColor;uniform vec4 textureMask;uniform sampler2D diffuseSampler;\n#include\n#include\n#include\n#include\n#include\n#ifdef RAMPGRADIENT\nvarying vec4 remapRanges;uniform sampler2D rampSampler;\n#endif\n#define CUSTOM_FRAGMENT_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_FRAGMENT_MAIN_BEGIN\n#include\nvec4 textureColor=texture2D(diffuseSampler,vUV);vec4 baseColor=(textureColor*textureMask+(vec4(1.,1.,1.,1.)-textureMask))*vColor;\n#ifdef RAMPGRADIENT\nfloat alpha=baseColor.a;float remappedColorIndex=clamp((alpha-remapRanges.x)/remapRanges.y,0.0,1.0);vec4 rampColor=texture2D(rampSampler,vec2(1.0-remappedColorIndex,0.));baseColor.rgb*=rampColor.rgb;float finalAlpha=baseColor.a;baseColor.a=clamp((alpha*rampColor.a-remapRanges.z)/remapRanges.w,0.0,1.0);\n#endif\n#ifdef BLENDMULTIPLYMODE\nfloat sourceAlpha=vColor.a*textureColor.a;baseColor.rgb=baseColor.rgb*sourceAlpha+vec3(1.0)*(1.0-sourceAlpha);\n#endif\n#include\n#ifdef IMAGEPROCESSINGPOSTPROCESS\nbaseColor.rgb=toLinearSpace(baseColor.rgb);\n#else\n#ifdef IMAGEPROCESSING\nbaseColor.rgb=toLinearSpace(baseColor.rgb);baseColor=applyImageProcessing(baseColor);\n#endif\n#endif\ngl_FragColor=baseColor;\n#define CUSTOM_FRAGMENT_MAIN_END\n}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const particlesPixelShader = { name, shader };\n","// Do not edit.\nimport { ShaderStore } from \"../Engines/shaderStore\";\nimport \"./ShadersInclude/clipPlaneVertexDeclaration\";\nimport \"./ShadersInclude/logDepthDeclaration\";\nimport \"./ShadersInclude/clipPlaneVertex\";\nimport \"./ShadersInclude/logDepthVertex\";\n\nconst name = \"particlesVertexShader\";\nconst shader = `attribute vec3 position;attribute vec4 color;attribute float angle;attribute vec2 size;\n#ifdef ANIMATESHEET\nattribute float cellIndex;\n#endif\n#ifndef BILLBOARD\nattribute vec3 direction;\n#endif\n#ifdef BILLBOARDSTRETCHED\nattribute vec3 direction;\n#endif\n#ifdef RAMPGRADIENT\nattribute vec4 remapData;\n#endif\nattribute vec2 offset;uniform mat4 view;uniform mat4 projection;uniform vec2 translationPivot;\n#ifdef ANIMATESHEET\nuniform vec3 particlesInfos; \n#endif\nvarying vec2 vUV;varying vec4 vColor;varying vec3 vPositionW;\n#ifdef RAMPGRADIENT\nvarying vec4 remapRanges;\n#endif\n#if defined(BILLBOARD) && !defined(BILLBOARDY) && !defined(BILLBOARDSTRETCHED)\nuniform mat4 invView;\n#endif\n#include\n#include\n#ifdef BILLBOARD\nuniform vec3 eyePosition;\n#endif\nvec3 rotate(vec3 yaxis,vec3 rotatedCorner) {vec3 xaxis=normalize(cross(vec3(0.,1.0,0.),yaxis));vec3 zaxis=normalize(cross(yaxis,xaxis));vec3 row0=vec3(xaxis.x,xaxis.y,xaxis.z);vec3 row1=vec3(yaxis.x,yaxis.y,yaxis.z);vec3 row2=vec3(zaxis.x,zaxis.y,zaxis.z);mat3 rotMatrix= mat3(row0,row1,row2);vec3 alignedCorner=rotMatrix*rotatedCorner;return position+alignedCorner;}\n#ifdef BILLBOARDSTRETCHED\nvec3 rotateAlign(vec3 toCamera,vec3 rotatedCorner) {vec3 normalizedToCamera=normalize(toCamera);vec3 normalizedCrossDirToCamera=normalize(cross(normalize(direction),normalizedToCamera));vec3 row0=vec3(normalizedCrossDirToCamera.x,normalizedCrossDirToCamera.y,normalizedCrossDirToCamera.z);vec3 row2=vec3(normalizedToCamera.x,normalizedToCamera.y,normalizedToCamera.z);\n#ifdef BILLBOARDSTRETCHED_LOCAL\nvec3 row1=direction;\n#else\nvec3 crossProduct=normalize(cross(normalizedToCamera,normalizedCrossDirToCamera));vec3 row1=vec3(crossProduct.x,crossProduct.y,crossProduct.z);\n#endif\nmat3 rotMatrix= mat3(row0,row1,row2);vec3 alignedCorner=rotMatrix*rotatedCorner;return position+alignedCorner;}\n#endif\n#define CUSTOM_VERTEX_DEFINITIONS\nvoid main(void) {\n#define CUSTOM_VERTEX_MAIN_BEGIN\nvec2 cornerPos;cornerPos=(vec2(offset.x-0.5,offset.y -0.5)-translationPivot)*size+translationPivot;\n#ifdef BILLBOARD\nvec3 rotatedCorner;\n#ifdef BILLBOARDY\nrotatedCorner.x=cornerPos.x*cos(angle)-cornerPos.y*sin(angle);rotatedCorner.z=cornerPos.x*sin(angle)+cornerPos.y*cos(angle);rotatedCorner.y=0.;vec3 yaxis=position-eyePosition;yaxis.y=0.;vPositionW=rotate(normalize(yaxis),rotatedCorner);vec3 viewPos=(view*vec4(vPositionW,1.0)).xyz;\n#elif defined(BILLBOARDSTRETCHED)\nrotatedCorner.x=cornerPos.x*cos(angle)-cornerPos.y*sin(angle);rotatedCorner.y=cornerPos.x*sin(angle)+cornerPos.y*cos(angle);rotatedCorner.z=0.;vec3 toCamera=position-eyePosition;vPositionW=rotateAlign(toCamera,rotatedCorner);vec3 viewPos=(view*vec4(vPositionW,1.0)).xyz;\n#else\nrotatedCorner.x=cornerPos.x*cos(angle)-cornerPos.y*sin(angle);rotatedCorner.y=cornerPos.x*sin(angle)+cornerPos.y*cos(angle);rotatedCorner.z=0.;vec3 viewPos=(view*vec4(position,1.0)).xyz+rotatedCorner;vPositionW=(invView*vec4(viewPos,1)).xyz;\n#endif\n#ifdef RAMPGRADIENT\nremapRanges=remapData;\n#endif\ngl_Position=projection*vec4(viewPos,1.0);\n#else\nvec3 rotatedCorner;rotatedCorner.x=cornerPos.x*cos(angle)-cornerPos.y*sin(angle);rotatedCorner.z=cornerPos.x*sin(angle)+cornerPos.y*cos(angle);rotatedCorner.y=0.;vec3 yaxis=normalize(direction);vPositionW=rotate(yaxis,rotatedCorner);gl_Position=projection*view*vec4(vPositionW,1.0);\n#endif\nvColor=color;\n#ifdef ANIMATESHEET\nfloat rowOffset=floor(cellIndex*particlesInfos.z);float columnOffset=cellIndex-rowOffset/particlesInfos.z;vec2 uvScale=particlesInfos.xy;vec2 uvOffset=vec2(offset.x ,1.0-offset.y);vUV=(uvOffset+vec2(columnOffset,rowOffset))*uvScale;\n#else\nvUV=offset;\n#endif\n#if defined(CLIPPLANE) || defined(CLIPPLANE2) || defined(CLIPPLANE3) || defined(CLIPPLANE4) || defined(CLIPPLANE5) || defined(CLIPPLANE6)\nvec4 worldPos=vec4(vPositionW,1.0);\n#endif\n#include\n#include\n#define CUSTOM_VERTEX_MAIN_END\n}`;\n// Sideeffect\nShaderStore.ShadersStore[name] = shader;\n/** @internal */\nexport const particlesVertexShader = { name, shader };\n","/* eslint-disable import/no-internal-modules */\r\nimport type { Immutable, Nullable } from \"../types\";\r\nimport { FactorGradient, ColorGradient, Color3Gradient, GradientHelper } from \"../Misc/gradients\";\r\nimport type { Observer } from \"../Misc/observable\";\r\nimport { Observable } from \"../Misc/observable\";\r\nimport { Vector3, Matrix, TmpVectors, Vector4 } from \"../Maths/math.vector\";\r\nimport { Scalar } from \"../Maths/math.scalar\";\r\nimport { VertexBuffer, Buffer } from \"../Buffers/buffer\";\r\n\r\nimport type { Effect } from \"../Materials/effect\";\r\nimport { ImageProcessingConfiguration } from \"../Materials/imageProcessingConfiguration\";\r\nimport { RawTexture } from \"../Materials/Textures/rawTexture\";\r\nimport { EngineStore } from \"../Engines/engineStore\";\r\nimport type { IDisposable } from \"../scene\";\r\nimport type { IParticleEmitterType } from \"../Particles/EmitterTypes/index\";\r\nimport {\r\n BoxParticleEmitter,\r\n HemisphericParticleEmitter,\r\n SphereParticleEmitter,\r\n SphereDirectedParticleEmitter,\r\n CylinderParticleEmitter,\r\n ConeParticleEmitter,\r\n PointParticleEmitter,\r\n MeshParticleEmitter,\r\n CylinderDirectedParticleEmitter,\r\n} from \"../Particles/EmitterTypes/index\";\r\nimport type { IParticleSystem } from \"./IParticleSystem\";\r\nimport { BaseParticleSystem } from \"./baseParticleSystem\";\r\nimport { Particle } from \"./particle\";\r\nimport { SubEmitter, SubEmitterType } from \"./subEmitter\";\r\nimport { Constants } from \"../Engines/constants\";\r\nimport { SerializationHelper } from \"../Misc/decorators\";\r\nimport { GetClass } from \"../Misc/typeStore\";\r\nimport type { IAnimatable } from \"../Animations/animatable.interface\";\r\nimport { DrawWrapper } from \"../Materials/drawWrapper\";\r\n\r\nimport \"../Shaders/particles.fragment\";\r\nimport \"../Shaders/particles.vertex\";\r\nimport type { DataBuffer } from \"../Buffers/dataBuffer\";\r\nimport { Color4, Color3, TmpColors } from \"../Maths/math.color\";\r\nimport type { ISize } from \"../Maths/math.size\";\r\nimport type { BaseTexture } from \"../Materials/Textures/baseTexture\";\r\nimport { ThinEngine } from \"../Engines/thinEngine\";\r\nimport { MaterialHelper } from \"../Materials/materialHelper\";\r\n\r\nimport \"../Engines/Extensions/engine.alpha\";\r\nimport { addClipPlaneUniforms, prepareStringDefinesForClipPlanes, bindClipPlane } from \"../Materials/clipPlaneMaterialHelper\";\r\n\r\nimport type { AbstractMesh } from \"../Meshes/abstractMesh\";\r\nimport type { ProceduralTexture } from \"../Materials/Textures/Procedurals/proceduralTexture\";\r\nimport type { Scene } from \"../scene\";\r\nimport type { Engine } from \"../Engines/engine\";\r\n\r\n/**\r\n * This represents a particle system in Babylon.\r\n * Particles are often small sprites used to simulate hard-to-reproduce phenomena like fire, smoke, water, or abstract visual effects like magic glitter and faery dust.\r\n * Particles can take different shapes while emitted like box, sphere, cone or you can write your custom function.\r\n * @example https://doc.babylonjs.com/features/featuresDeepDive/particles/particle_system/particle_system_intro\r\n */\r\nexport class ParticleSystem extends BaseParticleSystem implements IDisposable, IAnimatable, IParticleSystem {\r\n /**\r\n * Billboard mode will only apply to Y axis\r\n */\r\n public static readonly BILLBOARDMODE_Y = Constants.PARTICLES_BILLBOARDMODE_Y;\r\n /**\r\n * Billboard mode will apply to all axes\r\n */\r\n public static readonly BILLBOARDMODE_ALL = Constants.PARTICLES_BILLBOARDMODE_ALL;\r\n /**\r\n * Special billboard mode where the particle will be biilboard to the camera but rotated to align with direction\r\n */\r\n public static readonly BILLBOARDMODE_STRETCHED = Constants.PARTICLES_BILLBOARDMODE_STRETCHED;\r\n /**\r\n * Special billboard mode where the particle will be billboard to the camera but only around the axis of the direction of particle emission\r\n */\r\n public static readonly BILLBOARDMODE_STRETCHED_LOCAL = Constants.PARTICLES_BILLBOARDMODE_STRETCHED_LOCAL;\r\n\r\n /**\r\n * This function can be defined to provide custom update for active particles.\r\n * This function will be called instead of regular update (age, position, color, etc.).\r\n * Do not forget that this function will be called on every frame so try to keep it simple and fast :)\r\n */\r\n public updateFunction: (particles: Particle[]) => void;\r\n\r\n private _emitterWorldMatrix: Matrix;\r\n private _emitterInverseWorldMatrix: Matrix = Matrix.Identity();\r\n\r\n /**\r\n * This function can be defined to specify initial direction for every new particle.\r\n * It by default use the emitterType defined function\r\n */\r\n public startDirectionFunction: (worldMatrix: Matrix, directionToUpdate: Vector3, particle: Particle, isLocal: boolean) => void;\r\n /**\r\n * This function can be defined to specify initial position for every new particle.\r\n * It by default use the emitterType defined function\r\n */\r\n public startPositionFunction: (worldMatrix: Matrix, positionToUpdate: Vector3, particle: Particle, isLocal: boolean) => void;\r\n\r\n /**\r\n * @internal\r\n */\r\n public _inheritedVelocityOffset = new Vector3();\r\n /**\r\n * An event triggered when the system is disposed\r\n */\r\n public onDisposeObservable = new Observable();\r\n /**\r\n * An event triggered when the system is stopped\r\n */\r\n public onStoppedObservable = new Observable();\r\n\r\n private _onDisposeObserver: Nullable>;\r\n /**\r\n * Sets a callback that will be triggered when the system is disposed\r\n */\r\n public set onDispose(callback: () => void) {\r\n if (this._onDisposeObserver) {\r\n this.onDisposeObservable.remove(this._onDisposeObserver);\r\n }\r\n this._onDisposeObserver = this.onDisposeObservable.add(callback);\r\n }\r\n\r\n private _particles = new Array();\r\n private _epsilon: number;\r\n private _capacity: number;\r\n private _stockParticles = new Array();\r\n private _newPartsExcess = 0;\r\n private _vertexData: Float32Array;\r\n private _vertexBuffer: Nullable;\r\n private _vertexBuffers: { [key: string]: VertexBuffer } = {};\r\n private _spriteBuffer: Nullable;\r\n private _indexBuffer: Nullable;\r\n private _drawWrappers: DrawWrapper[][]; // first index is render pass id, second index is blend mode\r\n private _customWrappers: { [blendMode: number]: Nullable };\r\n private _scaledColorStep = new Color4(0, 0, 0, 0);\r\n private _colorDiff = new Color4(0, 0, 0, 0);\r\n private _scaledDirection = Vector3.Zero();\r\n private _scaledGravity = Vector3.Zero();\r\n private _currentRenderId = -1;\r\n private _alive: boolean;\r\n private _useInstancing = false;\r\n private _vertexArrayObject: Nullable;\r\n\r\n private _started = false;\r\n private _stopped = false;\r\n private _actualFrame = 0;\r\n private _scaledUpdateSpeed: number;\r\n private _vertexBufferSize: number;\r\n\r\n /** @internal */\r\n public _currentEmitRateGradient: Nullable;\r\n /** @internal */\r\n public _currentEmitRate1 = 0;\r\n /** @internal */\r\n public _currentEmitRate2 = 0;\r\n\r\n /** @internal */\r\n public _currentStartSizeGradient: Nullable;\r\n /** @internal */\r\n public _currentStartSize1 = 0;\r\n /** @internal */\r\n public _currentStartSize2 = 0;\r\n\r\n /** Indicates that the update of particles is done in the animate function */\r\n public readonly updateInAnimate = true;\r\n\r\n private readonly _rawTextureWidth = 256;\r\n private _rampGradientsTexture: Nullable;\r\n private _useRampGradients = false;\r\n\r\n /** Gets or sets a matrix to use to compute projection */\r\n public defaultProjectionMatrix: Matrix;\r\n\r\n /** Gets or sets a matrix to use to compute view */\r\n public defaultViewMatrix: Matrix;\r\n\r\n /** Gets or sets a boolean indicating that ramp gradients must be used\r\n * @see https://doc.babylonjs.com/features/featuresDeepDive/particles/particle_system/particle_system_intro#ramp-gradients\r\n */\r\n public get useRampGradients(): boolean {\r\n return this._useRampGradients;\r\n }\r\n\r\n public set useRampGradients(value: boolean) {\r\n if (this._useRampGradients === value) {\r\n return;\r\n }\r\n\r\n this._useRampGradients = value;\r\n\r\n this._resetEffect();\r\n }\r\n\r\n // Sub-emitters\r\n /**\r\n * The Sub-emitters templates that will be used to generate the sub particle system to be associated with the system, this property is used by the root particle system only.\r\n * When a particle is spawned, an array will be chosen at random and all the emitters in that array will be attached to the particle. (Default: [])\r\n */\r\n public subEmitters: Array>;\r\n // the subEmitters field above converted to a constant type\r\n private _subEmitters: Array>;\r\n /**\r\n * @internal\r\n * If the particle systems emitter should be disposed when the particle system is disposed\r\n */\r\n public _disposeEmitterOnDispose = false;\r\n /**\r\n * The current active Sub-systems, this property is used by the root particle system only.\r\n */\r\n public activeSubSystems: Array;\r\n\r\n /**\r\n * Specifies if the particles are updated in emitter local space or world space\r\n */\r\n public isLocal = false;\r\n\r\n /** Indicates that the particle system is CPU based */\r\n public readonly isGPU = false;\r\n\r\n private _rootParticleSystem: Nullable;\r\n //end of Sub-emitter\r\n\r\n /**\r\n * Gets the current list of active particles\r\n */\r\n public get particles(): Particle[] {\r\n return this._particles;\r\n }\r\n\r\n /**\r\n * Gets the number of particles active at the same time.\r\n * @returns The number of active particles.\r\n */\r\n public getActiveCount() {\r\n return this._particles.length;\r\n }\r\n\r\n /**\r\n * Returns the string \"ParticleSystem\"\r\n * @returns a string containing the class name\r\n */\r\n public getClassName(): string {\r\n return \"ParticleSystem\";\r\n }\r\n\r\n /**\r\n * Gets a boolean indicating that the system is stopping\r\n * @returns true if the system is currently stopping\r\n */\r\n public isStopping() {\r\n return this._stopped && this.isAlive();\r\n }\r\n\r\n /**\r\n * Gets the custom effect used to render the particles\r\n * @param blendMode Blend mode for which the effect should be retrieved\r\n * @returns The effect\r\n */\r\n public getCustomEffect(blendMode: number = 0): Nullable {\r\n return this._customWrappers[blendMode]?.effect ?? this._customWrappers[0]!.effect;\r\n }\r\n\r\n private _getCustomDrawWrapper(blendMode: number = 0): Nullable {\r\n return this._customWrappers[blendMode] ?? this._customWrappers[0];\r\n }\r\n\r\n /**\r\n * Sets the custom effect used to render the particles\r\n * @param effect The effect to set\r\n * @param blendMode Blend mode for which the effect should be set\r\n */\r\n public setCustomEffect(effect: Nullable, blendMode: number = 0) {\r\n this._customWrappers[blendMode] = new DrawWrapper(this._engine);\r\n this._customWrappers[blendMode]!.effect = effect;\r\n if (this._customWrappers[blendMode]!.drawContext) {\r\n this._customWrappers[blendMode]!.drawContext!.useInstancing = this._useInstancing;\r\n }\r\n }\r\n\r\n /** @internal */\r\n private _onBeforeDrawParticlesObservable: Nullable>> = null;\r\n\r\n /**\r\n * Observable that will be called just before the particles are drawn\r\n */\r\n public get onBeforeDrawParticlesObservable(): Observable> {\r\n if (!this._onBeforeDrawParticlesObservable) {\r\n this._onBeforeDrawParticlesObservable = new Observable>();\r\n }\r\n\r\n return this._onBeforeDrawParticlesObservable;\r\n }\r\n\r\n /**\r\n * Gets the name of the particle vertex shader\r\n */\r\n public get vertexShaderName(): string {\r\n return \"particles\";\r\n }\r\n\r\n /**\r\n * Gets the vertex buffers used by the particle system\r\n */\r\n public get vertexBuffers(): Immutable<{ [key: string]: VertexBuffer }> {\r\n return this._vertexBuffers;\r\n }\r\n\r\n /**\r\n * Gets the index buffer used by the particle system (or null if no index buffer is used (if _useInstancing=true))\r\n */\r\n public get indexBuffer(): Nullable {\r\n return this._indexBuffer;\r\n }\r\n\r\n /**\r\n * Instantiates a particle system.\r\n * Particles are often small sprites used to simulate hard-to-reproduce phenomena like fire, smoke, water, or abstract visual effects like magic glitter and faery dust.\r\n * @param name The name of the particle system\r\n * @param capacity The max number of particles alive at the same time\r\n * @param sceneOrEngine The scene the particle system belongs to or the engine to use if no scene\r\n * @param customEffect a custom effect used to change the way particles are rendered by default\r\n * @param isAnimationSheetEnabled Must be true if using a spritesheet to animate the particles texture\r\n * @param epsilon Offset used to render the particles\r\n */\r\n constructor(\r\n name: string,\r\n capacity: number,\r\n sceneOrEngine: Scene | ThinEngine,\r\n customEffect: Nullable = null,\r\n isAnimationSheetEnabled: boolean = false,\r\n epsilon: number = 0.01\r\n ) {\r\n super(name);\r\n\r\n this._capacity = capacity;\r\n\r\n this._epsilon = epsilon;\r\n this._isAnimationSheetEnabled = isAnimationSheetEnabled;\r\n\r\n if (!sceneOrEngine || sceneOrEngine.getClassName() === \"Scene\") {\r\n this._scene = (sceneOrEngine as Scene) || EngineStore.LastCreatedScene;\r\n this._engine = this._scene.getEngine();\r\n this.uniqueId = this._scene.getUniqueId();\r\n this._scene.particleSystems.push(this);\r\n } else {\r\n this._engine = sceneOrEngine as ThinEngine;\r\n this.defaultProjectionMatrix = Matrix.PerspectiveFovLH(0.8, 1, 0.1, 100, this._engine.isNDCHalfZRange);\r\n }\r\n\r\n if (this._engine.getCaps().vertexArrayObject) {\r\n this._vertexArrayObject = null;\r\n }\r\n\r\n // Setup the default processing configuration to the scene.\r\n this._attachImageProcessingConfiguration(null);\r\n\r\n // eslint-disable-next-line @typescript-eslint/naming-convention\r\n this._customWrappers = { 0: new DrawWrapper(this._engine) };\r\n this._customWrappers[0]!.effect = customEffect;\r\n\r\n this._drawWrappers = [];\r\n this._useInstancing = this._engine.getCaps().instancedArrays;\r\n\r\n this._createIndexBuffer();\r\n this._createVertexBuffers();\r\n\r\n // Default emitter type\r\n this.particleEmitterType = new BoxParticleEmitter();\r\n let noiseTextureData: Nullable = null;\r\n\r\n // Update\r\n this.updateFunction = (particles: Particle[]): void => {\r\n let noiseTextureSize: Nullable = null;\r\n\r\n if (this.noiseTexture) {\r\n // We need to get texture data back to CPU\r\n noiseTextureSize = this.noiseTexture.getSize();\r\n this.noiseTexture.getContent()?.then((data) => {\r\n noiseTextureData = data as Uint8Array;\r\n });\r\n }\r\n\r\n const sameParticleArray = particles === this._particles;\r\n\r\n for (let index = 0; index < particles.length; index++) {\r\n const particle = particles[index];\r\n\r\n let scaledUpdateSpeed = this._scaledUpdateSpeed;\r\n const previousAge = particle.age;\r\n particle.age += scaledUpdateSpeed;\r\n\r\n // Evaluate step to death\r\n if (particle.age > particle.lifeTime) {\r\n const diff = particle.age - previousAge;\r\n const oldDiff = particle.lifeTime - previousAge;\r\n\r\n scaledUpdateSpeed = (oldDiff * scaledUpdateSpeed) / diff;\r\n\r\n particle.age = particle.lifeTime;\r\n }\r\n\r\n const ratio = particle.age / particle.lifeTime;\r\n\r\n // Color\r\n if (this._colorGradients && this._colorGradients.length > 0) {\r\n GradientHelper.GetCurrentGradient(ratio, this._colorGradients, (currentGradient, nextGradient, scale) => {\r\n if (currentGradient !== particle._currentColorGradient) {\r\n particle._currentColor1.copyFrom(particle._currentColor2);\r\n (nextGradient).getColorToRef(particle._currentColor2);\r\n particle._currentColorGradient = currentGradient;\r\n }\r\n Color4.LerpToRef(particle._currentColor1, particle._currentColor2, scale, particle.color);\r\n });\r\n } else {\r\n particle.colorStep.scaleToRef(scaledUpdateSpeed, this._scaledColorStep);\r\n particle.color.addInPlace(this._scaledColorStep);\r\n\r\n if (particle.color.a < 0) {\r\n particle.color.a = 0;\r\n }\r\n }\r\n\r\n // Angular speed\r\n if (this._angularSpeedGradients && this._angularSpeedGradients.length > 0) {\r\n GradientHelper.GetCurrentGradient(ratio, this._angularSpeedGradients, (currentGradient, nextGradient, scale) => {\r\n if (currentGradient !== particle._currentAngularSpeedGradient) {\r\n particle._currentAngularSpeed1 = particle._currentAngularSpeed2;\r\n particle._currentAngularSpeed2 = (nextGradient).getFactor();\r\n particle._currentAngularSpeedGradient = currentGradient;\r\n }\r\n particle.angularSpeed = Scalar.Lerp(particle._currentAngularSpeed1, particle._currentAngularSpeed2, scale);\r\n });\r\n }\r\n particle.angle += particle.angularSpeed * scaledUpdateSpeed;\r\n\r\n // Direction\r\n let directionScale = scaledUpdateSpeed;\r\n\r\n /// Velocity\r\n if (this._velocityGradients && this._velocityGradients.length > 0) {\r\n GradientHelper.GetCurrentGradient(ratio, this._velocityGradients, (currentGradient, nextGradient, scale) => {\r\n if (currentGradient !== particle._currentVelocityGradient) {\r\n particle._currentVelocity1 = particle._currentVelocity2;\r\n particle._currentVelocity2 = (nextGradient).getFactor();\r\n particle._currentVelocityGradient = currentGradient;\r\n }\r\n directionScale *= Scalar.Lerp(particle._currentVelocity1, particle._currentVelocity2, scale);\r\n });\r\n }\r\n\r\n particle.direction.scaleToRef(directionScale, this._scaledDirection);\r\n\r\n /// Limit velocity\r\n if (this._limitVelocityGradients && this._limitVelocityGradients.length > 0) {\r\n GradientHelper.GetCurrentGradient(ratio, this._limitVelocityGradients, (currentGradient, nextGradient, scale) => {\r\n if (currentGradient !== particle._currentLimitVelocityGradient) {\r\n particle._currentLimitVelocity1 = particle._currentLimitVelocity2;\r\n particle._currentLimitVelocity2 = (nextGradient).getFactor();\r\n particle._currentLimitVelocityGradient = currentGradient;\r\n }\r\n\r\n const limitVelocity = Scalar.Lerp(particle._currentLimitVelocity1, particle._currentLimitVelocity2, scale);\r\n const currentVelocity = particle.direction.length();\r\n\r\n if (currentVelocity > limitVelocity) {\r\n particle.direction.scaleInPlace(this.limitVelocityDamping);\r\n }\r\n });\r\n }\r\n\r\n /// Drag\r\n if (this._dragGradients && this._dragGradients.length > 0) {\r\n GradientHelper.GetCurrentGradient(ratio, this._dragGradients, (currentGradient, nextGradient, scale) => {\r\n if (currentGradient !== particle._currentDragGradient) {\r\n particle._currentDrag1 = particle._currentDrag2;\r\n particle._currentDrag2 = (nextGradient).getFactor();\r\n particle._currentDragGradient = currentGradient;\r\n }\r\n\r\n const drag = Scalar.Lerp(particle._currentDrag1, particle._currentDrag2, scale);\r\n\r\n this._scaledDirection.scaleInPlace(1.0 - drag);\r\n });\r\n }\r\n\r\n if (this.isLocal && particle._localPosition) {\r\n particle._localPosition!.addInPlace(this._scaledDirection);\r\n Vector3.TransformCoordinatesToRef(particle._localPosition!, this._emitterWorldMatrix, particle.position);\r\n } else {\r\n particle.position.addInPlace(this._scaledDirection);\r\n }\r\n\r\n // Noise\r\n if (noiseTextureData && noiseTextureSize && particle._randomNoiseCoordinates1) {\r\n const fetchedColorR = this._fetchR(\r\n particle._randomNoiseCoordinates1.x,\r\n particle._randomNoiseCoordinates1.y,\r\n noiseTextureSize.width,\r\n noiseTextureSize.height,\r\n noiseTextureData\r\n );\r\n const fetchedColorG = this._fetchR(\r\n particle._randomNoiseCoordinates1.z,\r\n particle._randomNoiseCoordinates2.x,\r\n noiseTextureSize.width,\r\n noiseTextureSize.height,\r\n noiseTextureData\r\n );\r\n const fetchedColorB = this._fetchR(\r\n particle._randomNoiseCoordinates2.y,\r\n particle._randomNoiseCoordinates2.z,\r\n noiseTextureSize.width,\r\n noiseTextureSize.height,\r\n noiseTextureData\r\n );\r\n\r\n const force = TmpVectors.Vector3[0];\r\n const scaledForce = TmpVectors.Vector3[1];\r\n\r\n force.copyFromFloats(\r\n (2 * fetchedColorR - 1) * this.noiseStrength.x,\r\n (2 * fetchedColorG - 1) * this.noiseStrength.y,\r\n (2 * fetchedColorB - 1) * this.noiseStrength.z\r\n );\r\n\r\n force.scaleToRef(scaledUpdateSpeed, scaledForce);\r\n particle.direction.addInPlace(scaledForce);\r\n }\r\n\r\n // Gravity\r\n this.gravity.scaleToRef(scaledUpdateSpeed, this._scaledGravity);\r\n particle.direction.addInPlace(this._scaledGravity);\r\n\r\n // Size\r\n if (this._sizeGradients && this._sizeGradients.length > 0) {\r\n GradientHelper.GetCurrentGradient(ratio, this._sizeGradients, (currentGradient, nextGradient, scale) => {\r\n if (currentGradient !== particle._currentSizeGradient) {\r\n particle._currentSize1 = particle._currentSize2;\r\n particle._currentSize2 = (nextGradient).getFactor();\r\n particle._currentSizeGradient = currentGradient;\r\n }\r\n particle.size = Scalar.Lerp(particle._currentSize1, particle._currentSize2, scale);\r\n });\r\n }\r\n\r\n // Remap data\r\n if (this._useRampGradients) {\r\n if (this._colorRemapGradients && this._colorRemapGradients.length > 0) {\r\n GradientHelper.GetCurrentGradient(ratio, this._colorRemapGradients, (currentGradient, nextGradient, scale) => {\r\n const min = Scalar.Lerp((currentGradient).factor1, (nextGradient).factor1, scale);\r\n const max = Scalar.Lerp((currentGradient).factor2!, (nextGradient).factor2!, scale);\r\n\r\n particle.remapData.x = min;\r\n particle.remapData.y = max - min;\r\n });\r\n }\r\n\r\n if (this._alphaRemapGradients && this._alphaRemapGradients.length > 0) {\r\n GradientHelper.GetCurrentGradient(ratio, this._alphaRemapGradients, (currentGradient, nextGradient, scale) => {\r\n const min = Scalar.Lerp((currentGradient).factor1, (nextGradient).factor1, scale);\r\n const max = Scalar.Lerp((currentGradient).factor2!, (nextGradient).factor2!, scale);\r\n\r\n particle.remapData.z = min;\r\n particle.remapData.w = max - min;\r\n });\r\n }\r\n }\r\n\r\n if (this._isAnimationSheetEnabled) {\r\n particle.updateCellIndex();\r\n }\r\n\r\n // Update the position of the attached sub-emitters to match their attached particle\r\n particle._inheritParticleInfoToSubEmitters();\r\n\r\n if (particle.age >= particle.lifeTime) {\r\n // Recycle by swapping with last particle\r\n this._emitFromParticle(particle);\r\n if (particle._attachedSubEmitters) {\r\n particle._attachedSubEmitters.forEach((subEmitter) => {\r\n subEmitter.particleSystem.disposeOnStop = true;\r\n subEmitter.particleSystem.stop();\r\n });\r\n particle._attachedSubEmitters = null;\r\n }\r\n this.recycleParticle(particle);\r\n if (sameParticleArray) {\r\n index--;\r\n }\r\n continue;\r\n }\r\n }\r\n };\r\n }\r\n\r\n private _addFactorGradient(factorGradients: FactorGradient[], gradient: number, factor: number, factor2?: number) {\r\n const newGradient = new FactorGradient(gradient, factor, factor2);\r\n factorGradients.push(newGradient);\r\n\r\n factorGradients.sort((a, b) => {\r\n if (a.gradient < b.gradient) {\r\n return -1;\r\n } else if (a.gradient > b.gradient) {\r\n return 1;\r\n }\r\n\r\n return 0;\r\n });\r\n }\r\n\r\n private _removeFactorGradient(factorGradients: Nullable, gradient: number) {\r\n if (!factorGradients) {\r\n return;\r\n }\r\n\r\n let index = 0;\r\n for (const factorGradient of factorGradients) {\r\n if (factorGradient.gradient === gradient) {\r\n factorGradients.splice(index, 1);\r\n break;\r\n }\r\n index++;\r\n }\r\n }\r\n\r\n /**\r\n * Adds a new life time gradient\r\n * @param gradient defines the gradient to use (between 0 and 1)\r\n * @param factor defines the life time factor to affect to the specified gradient\r\n * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from\r\n * @returns the current particle system\r\n */\r\n public addLifeTimeGradient(gradient: number, factor: number, factor2?: number): IParticleSystem {\r\n if (!this._lifeTimeGradients) {\r\n this._lifeTimeGradients = [];\r\n }\r\n\r\n this._addFactorGradient(this._lifeTimeGradients, gradient, factor, factor2);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Remove a specific life time gradient\r\n * @param gradient defines the gradient to remove\r\n * @returns the current particle system\r\n */\r\n public removeLifeTimeGradient(gradient: number): IParticleSystem {\r\n this._removeFactorGradient(this._lifeTimeGradients, gradient);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds a new size gradient\r\n * @param gradient defines the gradient to use (between 0 and 1)\r\n * @param factor defines the size factor to affect to the specified gradient\r\n * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from\r\n * @returns the current particle system\r\n */\r\n public addSizeGradient(gradient: number, factor: number, factor2?: number): IParticleSystem {\r\n if (!this._sizeGradients) {\r\n this._sizeGradients = [];\r\n }\r\n\r\n this._addFactorGradient(this._sizeGradients, gradient, factor, factor2);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Remove a specific size gradient\r\n * @param gradient defines the gradient to remove\r\n * @returns the current particle system\r\n */\r\n public removeSizeGradient(gradient: number): IParticleSystem {\r\n this._removeFactorGradient(this._sizeGradients, gradient);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds a new color remap gradient\r\n * @param gradient defines the gradient to use (between 0 and 1)\r\n * @param min defines the color remap minimal range\r\n * @param max defines the color remap maximal range\r\n * @returns the current particle system\r\n */\r\n public addColorRemapGradient(gradient: number, min: number, max: number): IParticleSystem {\r\n if (!this._colorRemapGradients) {\r\n this._colorRemapGradients = [];\r\n }\r\n\r\n this._addFactorGradient(this._colorRemapGradients, gradient, min, max);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Remove a specific color remap gradient\r\n * @param gradient defines the gradient to remove\r\n * @returns the current particle system\r\n */\r\n public removeColorRemapGradient(gradient: number): IParticleSystem {\r\n this._removeFactorGradient(this._colorRemapGradients, gradient);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds a new alpha remap gradient\r\n * @param gradient defines the gradient to use (between 0 and 1)\r\n * @param min defines the alpha remap minimal range\r\n * @param max defines the alpha remap maximal range\r\n * @returns the current particle system\r\n */\r\n public addAlphaRemapGradient(gradient: number, min: number, max: number): IParticleSystem {\r\n if (!this._alphaRemapGradients) {\r\n this._alphaRemapGradients = [];\r\n }\r\n\r\n this._addFactorGradient(this._alphaRemapGradients, gradient, min, max);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Remove a specific alpha remap gradient\r\n * @param gradient defines the gradient to remove\r\n * @returns the current particle system\r\n */\r\n public removeAlphaRemapGradient(gradient: number): IParticleSystem {\r\n this._removeFactorGradient(this._alphaRemapGradients, gradient);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds a new angular speed gradient\r\n * @param gradient defines the gradient to use (between 0 and 1)\r\n * @param factor defines the angular speed to affect to the specified gradient\r\n * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from\r\n * @returns the current particle system\r\n */\r\n public addAngularSpeedGradient(gradient: number, factor: number, factor2?: number): IParticleSystem {\r\n if (!this._angularSpeedGradients) {\r\n this._angularSpeedGradients = [];\r\n }\r\n\r\n this._addFactorGradient(this._angularSpeedGradients, gradient, factor, factor2);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Remove a specific angular speed gradient\r\n * @param gradient defines the gradient to remove\r\n * @returns the current particle system\r\n */\r\n public removeAngularSpeedGradient(gradient: number): IParticleSystem {\r\n this._removeFactorGradient(this._angularSpeedGradients, gradient);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds a new velocity gradient\r\n * @param gradient defines the gradient to use (between 0 and 1)\r\n * @param factor defines the velocity to affect to the specified gradient\r\n * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from\r\n * @returns the current particle system\r\n */\r\n public addVelocityGradient(gradient: number, factor: number, factor2?: number): IParticleSystem {\r\n if (!this._velocityGradients) {\r\n this._velocityGradients = [];\r\n }\r\n\r\n this._addFactorGradient(this._velocityGradients, gradient, factor, factor2);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Remove a specific velocity gradient\r\n * @param gradient defines the gradient to remove\r\n * @returns the current particle system\r\n */\r\n public removeVelocityGradient(gradient: number): IParticleSystem {\r\n this._removeFactorGradient(this._velocityGradients, gradient);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds a new limit velocity gradient\r\n * @param gradient defines the gradient to use (between 0 and 1)\r\n * @param factor defines the limit velocity value to affect to the specified gradient\r\n * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from\r\n * @returns the current particle system\r\n */\r\n public addLimitVelocityGradient(gradient: number, factor: number, factor2?: number): IParticleSystem {\r\n if (!this._limitVelocityGradients) {\r\n this._limitVelocityGradients = [];\r\n }\r\n\r\n this._addFactorGradient(this._limitVelocityGradients, gradient, factor, factor2);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Remove a specific limit velocity gradient\r\n * @param gradient defines the gradient to remove\r\n * @returns the current particle system\r\n */\r\n public removeLimitVelocityGradient(gradient: number): IParticleSystem {\r\n this._removeFactorGradient(this._limitVelocityGradients, gradient);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds a new drag gradient\r\n * @param gradient defines the gradient to use (between 0 and 1)\r\n * @param factor defines the drag value to affect to the specified gradient\r\n * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from\r\n * @returns the current particle system\r\n */\r\n public addDragGradient(gradient: number, factor: number, factor2?: number): IParticleSystem {\r\n if (!this._dragGradients) {\r\n this._dragGradients = [];\r\n }\r\n\r\n this._addFactorGradient(this._dragGradients, gradient, factor, factor2);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Remove a specific drag gradient\r\n * @param gradient defines the gradient to remove\r\n * @returns the current particle system\r\n */\r\n public removeDragGradient(gradient: number): IParticleSystem {\r\n this._removeFactorGradient(this._dragGradients, gradient);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds a new emit rate gradient (please note that this will only work if you set the targetStopDuration property)\r\n * @param gradient defines the gradient to use (between 0 and 1)\r\n * @param factor defines the emit rate value to affect to the specified gradient\r\n * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from\r\n * @returns the current particle system\r\n */\r\n public addEmitRateGradient(gradient: number, factor: number, factor2?: number): IParticleSystem {\r\n if (!this._emitRateGradients) {\r\n this._emitRateGradients = [];\r\n }\r\n\r\n this._addFactorGradient(this._emitRateGradients, gradient, factor, factor2);\r\n return this;\r\n }\r\n\r\n /**\r\n * Remove a specific emit rate gradient\r\n * @param gradient defines the gradient to remove\r\n * @returns the current particle system\r\n */\r\n public removeEmitRateGradient(gradient: number): IParticleSystem {\r\n this._removeFactorGradient(this._emitRateGradients, gradient);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds a new start size gradient (please note that this will only work if you set the targetStopDuration property)\r\n * @param gradient defines the gradient to use (between 0 and 1)\r\n * @param factor defines the start size value to affect to the specified gradient\r\n * @param factor2 defines an additional factor used to define a range ([factor, factor2]) with main value to pick the final value from\r\n * @returns the current particle system\r\n */\r\n public addStartSizeGradient(gradient: number, factor: number, factor2?: number): IParticleSystem {\r\n if (!this._startSizeGradients) {\r\n this._startSizeGradients = [];\r\n }\r\n\r\n this._addFactorGradient(this._startSizeGradients, gradient, factor, factor2);\r\n return this;\r\n }\r\n\r\n /**\r\n * Remove a specific start size gradient\r\n * @param gradient defines the gradient to remove\r\n * @returns the current particle system\r\n */\r\n public removeStartSizeGradient(gradient: number): IParticleSystem {\r\n this._removeFactorGradient(this._startSizeGradients, gradient);\r\n\r\n return this;\r\n }\r\n\r\n private _createRampGradientTexture() {\r\n if (!this._rampGradients || !this._rampGradients.length || this._rampGradientsTexture || !this._scene) {\r\n return;\r\n }\r\n\r\n const data = new Uint8Array(this._rawTextureWidth * 4);\r\n const tmpColor = TmpColors.Color3[0];\r\n\r\n for (let x = 0; x < this._rawTextureWidth; x++) {\r\n const ratio = x / this._rawTextureWidth;\r\n\r\n GradientHelper.GetCurrentGradient(ratio, this._rampGradients, (currentGradient, nextGradient, scale) => {\r\n Color3.LerpToRef((currentGradient).color, (nextGradient).color, scale, tmpColor);\r\n data[x * 4] = tmpColor.r * 255;\r\n data[x * 4 + 1] = tmpColor.g * 255;\r\n data[x * 4 + 2] = tmpColor.b * 255;\r\n data[x * 4 + 3] = 255;\r\n });\r\n }\r\n\r\n this._rampGradientsTexture = RawTexture.CreateRGBATexture(data, this._rawTextureWidth, 1, this._scene, false, false, Constants.TEXTURE_NEAREST_SAMPLINGMODE);\r\n }\r\n\r\n /**\r\n * Gets the current list of ramp gradients.\r\n * You must use addRampGradient and removeRampGradient to update this list\r\n * @returns the list of ramp gradients\r\n */\r\n public getRampGradients(): Nullable> {\r\n return this._rampGradients;\r\n }\r\n\r\n /** Force the system to rebuild all gradients that need to be resync */\r\n public forceRefreshGradients() {\r\n this._syncRampGradientTexture();\r\n }\r\n\r\n private _syncRampGradientTexture() {\r\n if (!this._rampGradients) {\r\n return;\r\n }\r\n\r\n this._rampGradients.sort((a, b) => {\r\n if (a.gradient < b.gradient) {\r\n return -1;\r\n } else if (a.gradient > b.gradient) {\r\n return 1;\r\n }\r\n\r\n return 0;\r\n });\r\n\r\n if (this._rampGradientsTexture) {\r\n this._rampGradientsTexture.dispose();\r\n this._rampGradientsTexture = null;\r\n }\r\n\r\n this._createRampGradientTexture();\r\n }\r\n\r\n /**\r\n * Adds a new ramp gradient used to remap particle colors\r\n * @param gradient defines the gradient to use (between 0 and 1)\r\n * @param color defines the color to affect to the specified gradient\r\n * @returns the current particle system\r\n */\r\n public addRampGradient(gradient: number, color: Color3): ParticleSystem {\r\n if (!this._rampGradients) {\r\n this._rampGradients = [];\r\n }\r\n\r\n const rampGradient = new Color3Gradient(gradient, color);\r\n this._rampGradients.push(rampGradient);\r\n\r\n this._syncRampGradientTexture();\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Remove a specific ramp gradient\r\n * @param gradient defines the gradient to remove\r\n * @returns the current particle system\r\n */\r\n public removeRampGradient(gradient: number): ParticleSystem {\r\n this._removeGradientAndTexture(gradient, this._rampGradients, this._rampGradientsTexture);\r\n this._rampGradientsTexture = null;\r\n\r\n if (this._rampGradients && this._rampGradients.length > 0) {\r\n this._createRampGradientTexture();\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds a new color gradient\r\n * @param gradient defines the gradient to use (between 0 and 1)\r\n * @param color1 defines the color to affect to the specified gradient\r\n * @param color2 defines an additional color used to define a range ([color, color2]) with main color to pick the final color from\r\n * @returns this particle system\r\n */\r\n public addColorGradient(gradient: number, color1: Color4, color2?: Color4): IParticleSystem {\r\n if (!this._colorGradients) {\r\n this._colorGradients = [];\r\n }\r\n\r\n const colorGradient = new ColorGradient(gradient, color1, color2);\r\n this._colorGradients.push(colorGradient);\r\n\r\n this._colorGradients.sort((a, b) => {\r\n if (a.gradient < b.gradient) {\r\n return -1;\r\n } else if (a.gradient > b.gradient) {\r\n return 1;\r\n }\r\n\r\n return 0;\r\n });\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Remove a specific color gradient\r\n * @param gradient defines the gradient to remove\r\n * @returns this particle system\r\n */\r\n public removeColorGradient(gradient: number): IParticleSystem {\r\n if (!this._colorGradients) {\r\n return this;\r\n }\r\n\r\n let index = 0;\r\n for (const colorGradient of this._colorGradients) {\r\n if (colorGradient.gradient === gradient) {\r\n this._colorGradients.splice(index, 1);\r\n break;\r\n }\r\n index++;\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Resets the draw wrappers cache\r\n */\r\n public resetDrawCache(): void {\r\n for (const drawWrappers of this._drawWrappers) {\r\n if (drawWrappers) {\r\n for (const drawWrapper of drawWrappers) {\r\n drawWrapper?.dispose();\r\n }\r\n }\r\n }\r\n\r\n this._drawWrappers = [];\r\n }\r\n\r\n private _fetchR(u: number, v: number, width: number, height: number, pixels: Uint8Array): number {\r\n u = Math.abs(u) * 0.5 + 0.5;\r\n v = Math.abs(v) * 0.5 + 0.5;\r\n\r\n const wrappedU = (u * width) % width | 0;\r\n const wrappedV = (v * height) % height | 0;\r\n\r\n const position = (wrappedU + wrappedV * width) * 4;\r\n return pixels[position] / 255;\r\n }\r\n\r\n protected _reset() {\r\n this._resetEffect();\r\n }\r\n\r\n private _resetEffect() {\r\n if (this._vertexBuffer) {\r\n this._vertexBuffer.dispose();\r\n this._vertexBuffer = null;\r\n }\r\n\r\n if (this._spriteBuffer) {\r\n this._spriteBuffer.dispose();\r\n this._spriteBuffer = null;\r\n }\r\n\r\n if (this._vertexArrayObject) {\r\n this._engine.releaseVertexArrayObject(this._vertexArrayObject);\r\n this._vertexArrayObject = null;\r\n }\r\n\r\n this._createVertexBuffers();\r\n }\r\n\r\n private _createVertexBuffers() {\r\n this._vertexBufferSize = this._useInstancing ? 10 : 12;\r\n if (this._isAnimationSheetEnabled) {\r\n this._vertexBufferSize += 1;\r\n }\r\n\r\n if (!this._isBillboardBased || this.billboardMode === ParticleSystem.BILLBOARDMODE_STRETCHED || this.billboardMode === ParticleSystem.BILLBOARDMODE_STRETCHED_LOCAL) {\r\n this._vertexBufferSize += 3;\r\n }\r\n\r\n if (this._useRampGradients) {\r\n this._vertexBufferSize += 4;\r\n }\r\n\r\n const engine = this._engine;\r\n const vertexSize = this._vertexBufferSize * (this._useInstancing ? 1 : 4);\r\n this._vertexData = new Float32Array(this._capacity * vertexSize);\r\n this._vertexBuffer = new Buffer(engine, this._vertexData, true, vertexSize);\r\n\r\n let dataOffset = 0;\r\n const positions = this._vertexBuffer.createVertexBuffer(VertexBuffer.PositionKind, dataOffset, 3, this._vertexBufferSize, this._useInstancing);\r\n this._vertexBuffers[VertexBuffer.PositionKind] = positions;\r\n dataOffset += 3;\r\n\r\n const colors = this._vertexBuffer.createVertexBuffer(VertexBuffer.ColorKind, dataOffset, 4, this._vertexBufferSize, this._useInstancing);\r\n this._vertexBuffers[VertexBuffer.ColorKind] = colors;\r\n dataOffset += 4;\r\n\r\n const options = this._vertexBuffer.createVertexBuffer(\"angle\", dataOffset, 1, this._vertexBufferSize, this._useInstancing);\r\n this._vertexBuffers[\"angle\"] = options;\r\n dataOffset += 1;\r\n\r\n const size = this._vertexBuffer.createVertexBuffer(\"size\", dataOffset, 2, this._vertexBufferSize, this._useInstancing);\r\n this._vertexBuffers[\"size\"] = size;\r\n dataOffset += 2;\r\n\r\n if (this._isAnimationSheetEnabled) {\r\n const cellIndexBuffer = this._vertexBuffer.createVertexBuffer(\"cellIndex\", dataOffset, 1, this._vertexBufferSize, this._useInstancing);\r\n this._vertexBuffers[\"cellIndex\"] = cellIndexBuffer;\r\n dataOffset += 1;\r\n }\r\n\r\n if (!this._isBillboardBased || this.billboardMode === ParticleSystem.BILLBOARDMODE_STRETCHED || this.billboardMode === ParticleSystem.BILLBOARDMODE_STRETCHED_LOCAL) {\r\n const directionBuffer = this._vertexBuffer.createVertexBuffer(\"direction\", dataOffset, 3, this._vertexBufferSize, this._useInstancing);\r\n this._vertexBuffers[\"direction\"] = directionBuffer;\r\n dataOffset += 3;\r\n }\r\n\r\n if (this._useRampGradients) {\r\n const rampDataBuffer = this._vertexBuffer.createVertexBuffer(\"remapData\", dataOffset, 4, this._vertexBufferSize, this._useInstancing);\r\n this._vertexBuffers[\"remapData\"] = rampDataBuffer;\r\n dataOffset += 4;\r\n }\r\n\r\n let offsets: VertexBuffer;\r\n if (this._useInstancing) {\r\n const spriteData = new Float32Array([0, 0, 1, 0, 0, 1, 1, 1]);\r\n this._spriteBuffer = new Buffer(engine, spriteData, false, 2);\r\n offsets = this._spriteBuffer.createVertexBuffer(\"offset\", 0, 2);\r\n } else {\r\n offsets = this._vertexBuffer.createVertexBuffer(\"offset\", dataOffset, 2, this._vertexBufferSize, this._useInstancing);\r\n dataOffset += 2;\r\n }\r\n this._vertexBuffers[\"offset\"] = offsets;\r\n\r\n this.resetDrawCache();\r\n }\r\n\r\n private _createIndexBuffer() {\r\n if (this._useInstancing) {\r\n return;\r\n }\r\n const indices = [];\r\n let index = 0;\r\n for (let count = 0; count < this._capacity; count++) {\r\n indices.push(index);\r\n indices.push(index + 1);\r\n indices.push(index + 2);\r\n indices.push(index);\r\n indices.push(index + 2);\r\n indices.push(index + 3);\r\n index += 4;\r\n }\r\n\r\n this._indexBuffer = this._engine.createIndexBuffer(indices);\r\n }\r\n\r\n /**\r\n * Gets the maximum number of particles active at the same time.\r\n * @returns The max number of active particles.\r\n */\r\n public getCapacity(): number {\r\n return this._capacity;\r\n }\r\n\r\n /**\r\n * Gets whether there are still active particles in the system.\r\n * @returns True if it is alive, otherwise false.\r\n */\r\n public isAlive(): boolean {\r\n return this._alive;\r\n }\r\n\r\n /**\r\n * Gets if the system has been started. (Note: this will still be true after stop is called)\r\n * @returns True if it has been started, otherwise false.\r\n */\r\n public isStarted(): boolean {\r\n return this._started;\r\n }\r\n\r\n private _prepareSubEmitterInternalArray() {\r\n this._subEmitters = new Array>();\r\n if (this.subEmitters) {\r\n this.subEmitters.forEach((subEmitter) => {\r\n if (subEmitter instanceof ParticleSystem) {\r\n this._subEmitters.push([new SubEmitter(subEmitter)]);\r\n } else if (subEmitter instanceof SubEmitter) {\r\n this._subEmitters.push([subEmitter]);\r\n } else if (subEmitter instanceof Array) {\r\n this._subEmitters.push(subEmitter);\r\n }\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Starts the particle system and begins to emit\r\n * @param delay defines the delay in milliseconds before starting the system (this.startDelay by default)\r\n */\r\n public start(delay = this.startDelay): void {\r\n if (!this.targetStopDuration && this._hasTargetStopDurationDependantGradient()) {\r\n throw \"Particle system started with a targetStopDuration dependant gradient (eg. startSizeGradients) but no targetStopDuration set\";\r\n }\r\n if (delay) {\r\n setTimeout(() => {\r\n this.start(0);\r\n }, delay);\r\n return;\r\n }\r\n // Convert the subEmitters field to the constant type field _subEmitters\r\n this._prepareSubEmitterInternalArray();\r\n\r\n this._started = true;\r\n this._stopped = false;\r\n this._actualFrame = 0;\r\n if (this._subEmitters && this._subEmitters.length != 0) {\r\n this.activeSubSystems = new Array();\r\n }\r\n\r\n // Reset emit gradient so it acts the same on every start\r\n if (this._emitRateGradients) {\r\n if (this._emitRateGradients.length > 0) {\r\n this._currentEmitRateGradient = this._emitRateGradients[0];\r\n this._currentEmitRate1 = this._currentEmitRateGradient.getFactor();\r\n this._currentEmitRate2 = this._currentEmitRate1;\r\n }\r\n if (this._emitRateGradients.length > 1) {\r\n this._currentEmitRate2 = this._emitRateGradients[1].getFactor();\r\n }\r\n }\r\n // Reset start size gradient so it acts the same on every start\r\n if (this._startSizeGradients) {\r\n if (this._startSizeGradients.length > 0) {\r\n this._currentStartSizeGradient = this._startSizeGradients[0];\r\n this._currentStartSize1 = this._currentStartSizeGradient.getFactor();\r\n this._currentStartSize2 = this._currentStartSize1;\r\n }\r\n if (this._startSizeGradients.length > 1) {\r\n this._currentStartSize2 = this._startSizeGradients[1].getFactor();\r\n }\r\n }\r\n\r\n if (this.preWarmCycles) {\r\n if (this.emitter?.getClassName().indexOf(\"Mesh\") !== -1) {\r\n (this.emitter as any).computeWorldMatrix(true);\r\n }\r\n\r\n const noiseTextureAsProcedural = this.noiseTexture as ProceduralTexture;\r\n\r\n if (noiseTextureAsProcedural && noiseTextureAsProcedural.onGeneratedObservable) {\r\n noiseTextureAsProcedural.onGeneratedObservable.addOnce(() => {\r\n setTimeout(() => {\r\n for (let index = 0; index < this.preWarmCycles; index++) {\r\n this.animate(true);\r\n noiseTextureAsProcedural.render();\r\n }\r\n });\r\n });\r\n } else {\r\n for (let index = 0; index < this.preWarmCycles; index++) {\r\n this.animate(true);\r\n }\r\n }\r\n }\r\n\r\n // Animations\r\n if (this.beginAnimationOnStart && this.animations && this.animations.length > 0 && this._scene) {\r\n this._scene.beginAnimation(this, this.beginAnimationFrom, this.beginAnimationTo, this.beginAnimationLoop);\r\n }\r\n }\r\n\r\n /**\r\n * Stops the particle system.\r\n * @param stopSubEmitters if true it will stop the current system and all created sub-Systems if false it will stop the current root system only, this param is used by the root particle system only. the default value is true.\r\n */\r\n public stop(stopSubEmitters = true): void {\r\n if (this._stopped) {\r\n return;\r\n }\r\n\r\n this.onStoppedObservable.notifyObservers(this);\r\n\r\n this._stopped = true;\r\n\r\n if (stopSubEmitters) {\r\n this._stopSubEmitters();\r\n }\r\n }\r\n\r\n // animation sheet\r\n\r\n /**\r\n * Remove all active particles\r\n */\r\n public reset(): void {\r\n this._stockParticles.length = 0;\r\n this._particles.length = 0;\r\n }\r\n\r\n /**\r\n * @internal (for internal use only)\r\n */\r\n public _appendParticleVertex(index: number, particle: Particle, offsetX: number, offsetY: number): void {\r\n let offset = index * this._vertexBufferSize;\r\n\r\n this._vertexData[offset++] = particle.position.x + this.worldOffset.x;\r\n this._vertexData[offset++] = particle.position.y + this.worldOffset.y;\r\n this._vertexData[offset++] = particle.position.z + this.worldOffset.z;\r\n this._vertexData[offset++] = particle.color.r;\r\n this._vertexData[offset++] = particle.color.g;\r\n this._vertexData[offset++] = particle.color.b;\r\n this._vertexData[offset++] = particle.color.a;\r\n this._vertexData[offset++] = particle.angle;\r\n\r\n this._vertexData[offset++] = particle.scale.x * particle.size;\r\n this._vertexData[offset++] = particle.scale.y * particle.size;\r\n\r\n if (this._isAnimationSheetEnabled) {\r\n this._vertexData[offset++] = particle.cellIndex;\r\n }\r\n\r\n if (!this._isBillboardBased) {\r\n if (particle._initialDirection) {\r\n let initialDirection = particle._initialDirection;\r\n if (this.isLocal) {\r\n Vector3.TransformNormalToRef(initialDirection, this._emitterWorldMatrix, TmpVectors.Vector3[0]);\r\n initialDirection = TmpVectors.Vector3[0];\r\n }\r\n if (initialDirection.x === 0 && initialDirection.z === 0) {\r\n initialDirection.x = 0.001;\r\n }\r\n\r\n this._vertexData[offset++] = initialDirection.x;\r\n this._vertexData[offset++] = initialDirection.y;\r\n this._vertexData[offset++] = initialDirection.z;\r\n } else {\r\n let direction = particle.direction;\r\n if (this.isLocal) {\r\n Vector3.TransformNormalToRef(direction, this._emitterWorldMatrix, TmpVectors.Vector3[0]);\r\n direction = TmpVectors.Vector3[0];\r\n }\r\n\r\n if (direction.x === 0 && direction.z === 0) {\r\n direction.x = 0.001;\r\n }\r\n this._vertexData[offset++] = direction.x;\r\n this._vertexData[offset++] = direction.y;\r\n this._vertexData[offset++] = direction.z;\r\n }\r\n } else if (this.billboardMode === ParticleSystem.BILLBOARDMODE_STRETCHED || this.billboardMode === ParticleSystem.BILLBOARDMODE_STRETCHED_LOCAL) {\r\n this._vertexData[offset++] = particle.direction.x;\r\n this._vertexData[offset++] = particle.direction.y;\r\n this._vertexData[offset++] = particle.direction.z;\r\n }\r\n\r\n if (this._useRampGradients && particle.remapData) {\r\n this._vertexData[offset++] = particle.remapData.x;\r\n this._vertexData[offset++] = particle.remapData.y;\r\n this._vertexData[offset++] = particle.remapData.z;\r\n this._vertexData[offset++] = particle.remapData.w;\r\n }\r\n\r\n if (!this._useInstancing) {\r\n if (this._isAnimationSheetEnabled) {\r\n if (offsetX === 0) {\r\n offsetX = this._epsilon;\r\n } else if (offsetX === 1) {\r\n offsetX = 1 - this._epsilon;\r\n }\r\n\r\n if (offsetY === 0) {\r\n offsetY = this._epsilon;\r\n } else if (offsetY === 1) {\r\n offsetY = 1 - this._epsilon;\r\n }\r\n }\r\n\r\n this._vertexData[offset++] = offsetX;\r\n this._vertexData[offset++] = offsetY;\r\n }\r\n }\r\n\r\n // start of sub system methods\r\n\r\n /**\r\n * \"Recycles\" one of the particle by copying it back to the \"stock\" of particles and removing it from the active list.\r\n * Its lifetime will start back at 0.\r\n * @param particle\r\n */\r\n public recycleParticle: (particle: Particle) => void = (particle) => {\r\n // move particle from activeParticle list to stock particles\r\n const lastParticle = this._particles.pop();\r\n if (lastParticle !== particle) {\r\n lastParticle.copyTo(particle);\r\n }\r\n this._stockParticles.push(lastParticle);\r\n };\r\n\r\n private _stopSubEmitters(): void {\r\n if (!this.activeSubSystems) {\r\n return;\r\n }\r\n this.activeSubSystems.forEach((subSystem) => {\r\n subSystem.stop(true);\r\n });\r\n this.activeSubSystems = new Array();\r\n }\r\n\r\n private _createParticle: () => Particle = () => {\r\n let particle: Particle;\r\n if (this._stockParticles.length !== 0) {\r\n particle = this._stockParticles.pop();\r\n particle._reset();\r\n } else {\r\n particle = new Particle(this);\r\n }\r\n\r\n // Attach emitters\r\n if (this._subEmitters && this._subEmitters.length > 0) {\r\n const subEmitters = this._subEmitters[Math.floor(Math.random() * this._subEmitters.length)];\r\n particle._attachedSubEmitters = [];\r\n subEmitters.forEach((subEmitter) => {\r\n if (subEmitter.type === SubEmitterType.ATTACHED) {\r\n const newEmitter = subEmitter.clone();\r\n (>particle._attachedSubEmitters).push(newEmitter);\r\n newEmitter.particleSystem.start();\r\n }\r\n });\r\n }\r\n return particle;\r\n };\r\n\r\n private _removeFromRoot(): void {\r\n if (!this._rootParticleSystem) {\r\n return;\r\n }\r\n\r\n const index = this._rootParticleSystem.activeSubSystems.indexOf(this);\r\n if (index !== -1) {\r\n this._rootParticleSystem.activeSubSystems.splice(index, 1);\r\n }\r\n\r\n this._rootParticleSystem = null;\r\n }\r\n\r\n private _emitFromParticle: (particle: Particle) => void = (particle) => {\r\n if (!this._subEmitters || this._subEmitters.length === 0) {\r\n return;\r\n }\r\n const templateIndex = Math.floor(Math.random() * this._subEmitters.length);\r\n\r\n this._subEmitters[templateIndex].forEach((subEmitter) => {\r\n if (subEmitter.type === SubEmitterType.END) {\r\n const subSystem = subEmitter.clone();\r\n particle._inheritParticleInfoToSubEmitter(subSystem);\r\n subSystem.particleSystem._rootParticleSystem = this;\r\n this.activeSubSystems.push(subSystem.particleSystem);\r\n subSystem.particleSystem.start();\r\n }\r\n });\r\n };\r\n\r\n // End of sub system methods\r\n\r\n private _update(newParticles: number): void {\r\n // Update current\r\n this._alive = this._particles.length > 0;\r\n\r\n if ((this.emitter).position) {\r\n const emitterMesh = this.emitter;\r\n this._emitterWorldMatrix = emitterMesh.getWorldMatrix();\r\n } else {\r\n const emitterPosition = this.emitter;\r\n this._emitterWorldMatrix = Matrix.Translation(emitterPosition.x, emitterPosition.y, emitterPosition.z);\r\n }\r\n\r\n this._emitterWorldMatrix.invertToRef(this._emitterInverseWorldMatrix);\r\n this.updateFunction(this._particles);\r\n\r\n // Add new ones\r\n let particle: Particle;\r\n for (let index = 0; index < newParticles; index++) {\r\n if (this._particles.length === this._capacity) {\r\n break;\r\n }\r\n\r\n particle = this._createParticle();\r\n\r\n this._particles.push(particle);\r\n\r\n // Life time\r\n if (this.targetStopDuration && this._lifeTimeGradients && this._lifeTimeGradients.length > 0) {\r\n const ratio = Scalar.Clamp(this._actualFrame / this.targetStopDuration);\r\n GradientHelper.GetCurrentGradient(ratio, this._lifeTimeGradients, (currentGradient, nextGradient) => {\r\n const factorGradient1 = currentGradient;\r\n const factorGradient2 = nextGradient;\r\n const lifeTime1 = factorGradient1.getFactor();\r\n const lifeTime2 = factorGradient2.getFactor();\r\n const gradient = (ratio - factorGradient1.gradient) / (factorGradient2.gradient - factorGradient1.gradient);\r\n particle.lifeTime = Scalar.Lerp(lifeTime1, lifeTime2, gradient);\r\n });\r\n } else {\r\n particle.lifeTime = Scalar.RandomRange(this.minLifeTime, this.maxLifeTime);\r\n }\r\n\r\n // Emitter\r\n const emitPower = Scalar.RandomRange(this.minEmitPower, this.maxEmitPower);\r\n\r\n if (this.startPositionFunction) {\r\n this.startPositionFunction(this._emitterWorldMatrix, particle.position, particle, this.isLocal);\r\n } else {\r\n this.particleEmitterType.startPositionFunction(this._emitterWorldMatrix, particle.position, particle, this.isLocal);\r\n }\r\n\r\n if (this.isLocal) {\r\n if (!particle._localPosition) {\r\n particle._localPosition = particle.position.clone();\r\n } else {\r\n particle._localPosition.copyFrom(particle.position);\r\n }\r\n Vector3.TransformCoordinatesToRef(particle._localPosition!, this._emitterWorldMatrix, particle.position);\r\n }\r\n\r\n if (this.startDirectionFunction) {\r\n this.startDirectionFunction(this._emitterWorldMatrix, particle.direction, particle, this.isLocal);\r\n } else {\r\n this.particleEmitterType.startDirectionFunction(this._emitterWorldMatrix, particle.direction, particle, this.isLocal, this._emitterInverseWorldMatrix);\r\n }\r\n\r\n if (emitPower === 0) {\r\n if (!particle._initialDirection) {\r\n particle._initialDirection = particle.direction.clone();\r\n } else {\r\n particle._initialDirection.copyFrom(particle.direction);\r\n }\r\n } else {\r\n particle._initialDirection = null;\r\n }\r\n\r\n particle.direction.scaleInPlace(emitPower);\r\n\r\n // Size\r\n if (!this._sizeGradients || this._sizeGradients.length === 0) {\r\n particle.size = Scalar.RandomRange(this.minSize, this.maxSize);\r\n } else {\r\n particle._currentSizeGradient = this._sizeGradients[0];\r\n particle._currentSize1 = particle._currentSizeGradient.getFactor();\r\n particle.size = particle._currentSize1;\r\n\r\n if (this._sizeGradients.length > 1) {\r\n particle._currentSize2 = this._sizeGradients[1].getFactor();\r\n } else {\r\n particle._currentSize2 = particle._currentSize1;\r\n }\r\n }\r\n // Size and scale\r\n particle.scale.copyFromFloats(Scalar.RandomRange(this.minScaleX, this.maxScaleX), Scalar.RandomRange(this.minScaleY, this.maxScaleY));\r\n\r\n // Adjust scale by start size\r\n if (this._startSizeGradients && this._startSizeGradients[0] && this.targetStopDuration) {\r\n const ratio = this._actualFrame / this.targetStopDuration;\r\n GradientHelper.GetCurrentGradient(ratio, this._startSizeGradients, (currentGradient, nextGradient, scale) => {\r\n if (currentGradient !== this._currentStartSizeGradient) {\r\n this._currentStartSize1 = this._currentStartSize2;\r\n this._currentStartSize2 = (nextGradient).getFactor();\r\n this._currentStartSizeGradient = currentGradient;\r\n }\r\n\r\n const value = Scalar.Lerp(this._currentStartSize1, this._currentStartSize2, scale);\r\n particle.scale.scaleInPlace(value);\r\n });\r\n }\r\n\r\n // Angle\r\n if (!this._angularSpeedGradients || this._angularSpeedGradients.length === 0) {\r\n particle.angularSpeed = Scalar.RandomRange(this.minAngularSpeed, this.maxAngularSpeed);\r\n } else {\r\n particle._currentAngularSpeedGradient = this._angularSpeedGradients[0];\r\n particle.angularSpeed = particle._currentAngularSpeedGradient.getFactor();\r\n particle._currentAngularSpeed1 = particle.angularSpeed;\r\n\r\n if (this._angularSpeedGradients.length > 1) {\r\n particle._currentAngularSpeed2 = this._angularSpeedGradients[1].getFactor();\r\n } else {\r\n particle._currentAngularSpeed2 = particle._currentAngularSpeed1;\r\n }\r\n }\r\n particle.angle = Scalar.RandomRange(this.minInitialRotation, this.maxInitialRotation);\r\n\r\n // Velocity\r\n if (this._velocityGradients && this._velocityGradients.length > 0) {\r\n particle._currentVelocityGradient = this._velocityGradients[0];\r\n particle._currentVelocity1 = particle._currentVelocityGradient.getFactor();\r\n\r\n if (this._velocityGradients.length > 1) {\r\n particle._currentVelocity2 = this._velocityGradients[1].getFactor();\r\n } else {\r\n particle._currentVelocity2 = particle._currentVelocity1;\r\n }\r\n }\r\n\r\n // Limit velocity\r\n if (this._limitVelocityGradients && this._limitVelocityGradients.length > 0) {\r\n particle._currentLimitVelocityGradient = this._limitVelocityGradients[0];\r\n particle._currentLimitVelocity1 = particle._currentLimitVelocityGradient.getFactor();\r\n\r\n if (this._limitVelocityGradients.length > 1) {\r\n particle._currentLimitVelocity2 = this._limitVelocityGradients[1].getFactor();\r\n } else {\r\n particle._currentLimitVelocity2 = particle._currentLimitVelocity1;\r\n }\r\n }\r\n\r\n // Drag\r\n if (this._dragGradients && this._dragGradients.length > 0) {\r\n particle._currentDragGradient = this._dragGradients[0];\r\n particle._currentDrag1 = particle._currentDragGradient.getFactor();\r\n\r\n if (this._dragGradients.length > 1) {\r\n particle._currentDrag2 = this._dragGradients[1].getFactor();\r\n } else {\r\n particle._currentDrag2 = particle._currentDrag1;\r\n }\r\n }\r\n\r\n // Color\r\n if (!this._colorGradients || this._colorGradients.length === 0) {\r\n const step = Scalar.RandomRange(0, 1.0);\r\n\r\n Color4.LerpToRef(this.color1, this.color2, step, particle.color);\r\n\r\n this.colorDead.subtractToRef(particle.color, this._colorDiff);\r\n this._colorDiff.scaleToRef(1.0 / particle.lifeTime, particle.colorStep);\r\n } else {\r\n particle._currentColorGradient = this._colorGradients[0];\r\n particle._currentColorGradient.getColorToRef(particle.color);\r\n particle._currentColor1.copyFrom(particle.color);\r\n\r\n if (this._colorGradients.length > 1) {\r\n this._colorGradients[1].getColorToRef(particle._currentColor2);\r\n } else {\r\n particle._currentColor2.copyFrom(particle.color);\r\n }\r\n }\r\n\r\n // Sheet\r\n if (this._isAnimationSheetEnabled) {\r\n particle._initialStartSpriteCellID = this.startSpriteCellID;\r\n particle._initialEndSpriteCellID = this.endSpriteCellID;\r\n particle._initialSpriteCellLoop = this.spriteCellLoop;\r\n }\r\n\r\n // Inherited Velocity\r\n particle.direction.addInPlace(this._inheritedVelocityOffset);\r\n\r\n // Ramp\r\n if (this._useRampGradients) {\r\n particle.remapData = new Vector4(0, 1, 0, 1);\r\n }\r\n\r\n // Noise texture coordinates\r\n if (this.noiseTexture) {\r\n if (particle._randomNoiseCoordinates1) {\r\n particle._randomNoiseCoordinates1.copyFromFloats(Math.random(), Math.random(), Math.random());\r\n particle._randomNoiseCoordinates2.copyFromFloats(Math.random(), Math.random(), Math.random());\r\n } else {\r\n particle._randomNoiseCoordinates1 = new Vector3(Math.random(), Math.random(), Math.random());\r\n particle._randomNoiseCoordinates2 = new Vector3(Math.random(), Math.random(), Math.random());\r\n }\r\n }\r\n\r\n // Update the position of the attached sub-emitters to match their attached particle\r\n particle._inheritParticleInfoToSubEmitters();\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _GetAttributeNamesOrOptions(isAnimationSheetEnabled = false, isBillboardBased = false, useRampGradients = false): string[] {\r\n const attributeNamesOrOptions = [VertexBuffer.PositionKind, VertexBuffer.ColorKind, \"angle\", \"offset\", \"size\"];\r\n\r\n if (isAnimationSheetEnabled) {\r\n attributeNamesOrOptions.push(\"cellIndex\");\r\n }\r\n\r\n if (!isBillboardBased) {\r\n attributeNamesOrOptions.push(\"direction\");\r\n }\r\n\r\n if (useRampGradients) {\r\n attributeNamesOrOptions.push(\"remapData\");\r\n }\r\n\r\n return attributeNamesOrOptions;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _GetEffectCreationOptions(isAnimationSheetEnabled = false, useLogarithmicDepth = false): string[] {\r\n const effectCreationOption = [\"invView\", \"view\", \"projection\", \"textureMask\", \"translationPivot\", \"eyePosition\"];\r\n\r\n addClipPlaneUniforms(effectCreationOption);\r\n\r\n if (isAnimationSheetEnabled) {\r\n effectCreationOption.push(\"particlesInfos\");\r\n }\r\n if (useLogarithmicDepth) {\r\n effectCreationOption.push(\"logarithmicDepthConstant\");\r\n }\r\n\r\n return effectCreationOption;\r\n }\r\n\r\n /**\r\n * Fill the defines array according to the current settings of the particle system\r\n * @param defines Array to be updated\r\n * @param blendMode blend mode to take into account when updating the array\r\n */\r\n public fillDefines(defines: Array, blendMode: number) {\r\n if (this._scene) {\r\n prepareStringDefinesForClipPlanes(this, this._scene, defines);\r\n }\r\n\r\n if (this._isAnimationSheetEnabled) {\r\n defines.push(\"#define ANIMATESHEET\");\r\n }\r\n\r\n if (this.useLogarithmicDepth) {\r\n defines.push(\"#define LOGARITHMICDEPTH\");\r\n }\r\n\r\n if (blendMode === ParticleSystem.BLENDMODE_MULTIPLY) {\r\n defines.push(\"#define BLENDMULTIPLYMODE\");\r\n }\r\n\r\n if (this._useRampGradients) {\r\n defines.push(\"#define RAMPGRADIENT\");\r\n }\r\n\r\n if (this._isBillboardBased) {\r\n defines.push(\"#define BILLBOARD\");\r\n\r\n switch (this.billboardMode) {\r\n case ParticleSystem.BILLBOARDMODE_Y:\r\n defines.push(\"#define BILLBOARDY\");\r\n break;\r\n case ParticleSystem.BILLBOARDMODE_STRETCHED:\r\n case ParticleSystem.BILLBOARDMODE_STRETCHED_LOCAL:\r\n defines.push(\"#define BILLBOARDSTRETCHED\");\r\n if (this.billboardMode === ParticleSystem.BILLBOARDMODE_STRETCHED_LOCAL) {\r\n defines.push(\"#define BILLBOARDSTRETCHED_LOCAL\");\r\n }\r\n break;\r\n case ParticleSystem.BILLBOARDMODE_ALL:\r\n defines.push(\"#define BILLBOARDMODE_ALL\");\r\n break;\r\n default:\r\n break;\r\n }\r\n }\r\n\r\n if (this._imageProcessingConfiguration) {\r\n this._imageProcessingConfiguration.prepareDefines(this._imageProcessingConfigurationDefines);\r\n defines.push(this._imageProcessingConfigurationDefines.toString());\r\n }\r\n }\r\n\r\n /**\r\n * Fill the uniforms, attributes and samplers arrays according to the current settings of the particle system\r\n * @param uniforms Uniforms array to fill\r\n * @param attributes Attributes array to fill\r\n * @param samplers Samplers array to fill\r\n */\r\n public fillUniformsAttributesAndSamplerNames(uniforms: Array, attributes: Array, samplers: Array) {\r\n attributes.push(\r\n ...ParticleSystem._GetAttributeNamesOrOptions(\r\n this._isAnimationSheetEnabled,\r\n this._isBillboardBased && this.billboardMode !== ParticleSystem.BILLBOARDMODE_STRETCHED && this.billboardMode !== ParticleSystem.BILLBOARDMODE_STRETCHED_LOCAL,\r\n this._useRampGradients\r\n )\r\n );\r\n\r\n uniforms.push(...ParticleSystem._GetEffectCreationOptions(this._isAnimationSheetEnabled, this.useLogarithmicDepth));\r\n\r\n samplers.push(\"diffuseSampler\", \"rampSampler\");\r\n\r\n if (this._imageProcessingConfiguration) {\r\n ImageProcessingConfiguration.PrepareUniforms(uniforms, this._imageProcessingConfigurationDefines);\r\n ImageProcessingConfiguration.PrepareSamplers(samplers, this._imageProcessingConfigurationDefines);\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n private _getWrapper(blendMode: number): DrawWrapper {\r\n const customWrapper = this._getCustomDrawWrapper(blendMode);\r\n\r\n if (customWrapper?.effect) {\r\n return customWrapper;\r\n }\r\n\r\n const defines: Array = [];\r\n\r\n this.fillDefines(defines, blendMode);\r\n\r\n // Effect\r\n const currentRenderPassId = this._engine._features.supportRenderPasses ? (this._engine as Engine).currentRenderPassId : Constants.RENDERPASS_MAIN;\r\n let drawWrappers = this._drawWrappers[currentRenderPassId];\r\n if (!drawWrappers) {\r\n drawWrappers = this._drawWrappers[currentRenderPassId] = [];\r\n }\r\n let drawWrapper = drawWrappers[blendMode];\r\n if (!drawWrapper) {\r\n drawWrapper = new DrawWrapper(this._engine);\r\n if (drawWrapper.drawContext) {\r\n drawWrapper.drawContext.useInstancing = this._useInstancing;\r\n }\r\n drawWrappers[blendMode] = drawWrapper;\r\n }\r\n\r\n const join = defines.join(\"\\n\");\r\n if (drawWrapper.defines !== join) {\r\n const attributesNamesOrOptions: Array = [];\r\n const effectCreationOption: Array = [];\r\n const samplers: Array = [];\r\n\r\n this.fillUniformsAttributesAndSamplerNames(effectCreationOption, attributesNamesOrOptions, samplers);\r\n\r\n drawWrapper.setEffect(this._engine.createEffect(\"particles\", attributesNamesOrOptions, effectCreationOption, samplers, join), join);\r\n }\r\n\r\n return drawWrapper;\r\n }\r\n\r\n /**\r\n * Animates the particle system for the current frame by emitting new particles and or animating the living ones.\r\n * @param preWarmOnly will prevent the system from updating the vertex buffer (default is false)\r\n */\r\n public animate(preWarmOnly = false): void {\r\n if (!this._started) {\r\n return;\r\n }\r\n\r\n if (!preWarmOnly && this._scene) {\r\n // Check\r\n if (!this.isReady()) {\r\n return;\r\n }\r\n\r\n if (this._currentRenderId === this._scene.getFrameId()) {\r\n return;\r\n }\r\n this._currentRenderId = this._scene.getFrameId();\r\n }\r\n\r\n this._scaledUpdateSpeed = this.updateSpeed * (preWarmOnly ? this.preWarmStepOffset : this._scene?.getAnimationRatio() || 1);\r\n\r\n // Determine the number of particles we need to create\r\n let newParticles;\r\n\r\n if (this.manualEmitCount > -1) {\r\n newParticles = this.manualEmitCount;\r\n this._newPartsExcess = 0;\r\n this.manualEmitCount = 0;\r\n } else {\r\n let rate = this.emitRate;\r\n\r\n if (this._emitRateGradients && this._emitRateGradients.length > 0 && this.targetStopDuration) {\r\n const ratio = this._actualFrame / this.targetStopDuration;\r\n GradientHelper.GetCurrentGradient(ratio, this._emitRateGradients, (currentGradient, nextGradient, scale) => {\r\n if (currentGradient !== this._currentEmitRateGradient) {\r\n this._currentEmitRate1 = this._currentEmitRate2;\r\n this._currentEmitRate2 = (nextGradient).getFactor();\r\n this._currentEmitRateGradient = currentGradient;\r\n }\r\n\r\n rate = Scalar.Lerp(this._currentEmitRate1, this._currentEmitRate2, scale);\r\n });\r\n }\r\n\r\n newParticles = (rate * this._scaledUpdateSpeed) >> 0;\r\n this._newPartsExcess += rate * this._scaledUpdateSpeed - newParticles;\r\n }\r\n\r\n if (this._newPartsExcess > 1.0) {\r\n newParticles += this._newPartsExcess >> 0;\r\n this._newPartsExcess -= this._newPartsExcess >> 0;\r\n }\r\n\r\n this._alive = false;\r\n\r\n if (!this._stopped) {\r\n this._actualFrame += this._scaledUpdateSpeed;\r\n\r\n if (this.targetStopDuration && this._actualFrame >= this.targetStopDuration) {\r\n this.stop();\r\n }\r\n } else {\r\n newParticles = 0;\r\n }\r\n this._update(newParticles);\r\n\r\n // Stopped?\r\n if (this._stopped) {\r\n if (!this._alive) {\r\n this._started = false;\r\n if (this.onAnimationEnd) {\r\n this.onAnimationEnd();\r\n }\r\n if (this.disposeOnStop && this._scene) {\r\n this._scene._toBeDisposed.push(this);\r\n }\r\n }\r\n }\r\n\r\n if (!preWarmOnly) {\r\n // Update VBO\r\n let offset = 0;\r\n for (let index = 0; index < this._particles.length; index++) {\r\n const particle = this._particles[index];\r\n this._appendParticleVertices(offset, particle);\r\n offset += this._useInstancing ? 1 : 4;\r\n }\r\n\r\n if (this._vertexBuffer) {\r\n this._vertexBuffer.updateDirectly(this._vertexData, 0, this._particles.length);\r\n }\r\n }\r\n\r\n if (this.manualEmitCount === 0 && this.disposeOnStop) {\r\n this.stop();\r\n }\r\n }\r\n\r\n private _appendParticleVertices(offset: number, particle: Particle) {\r\n this._appendParticleVertex(offset++, particle, 0, 0);\r\n if (!this._useInstancing) {\r\n this._appendParticleVertex(offset++, particle, 1, 0);\r\n this._appendParticleVertex(offset++, particle, 1, 1);\r\n this._appendParticleVertex(offset++, particle, 0, 1);\r\n }\r\n }\r\n\r\n /**\r\n * Rebuilds the particle system.\r\n */\r\n public rebuild(): void {\r\n if (this._engine.getCaps().vertexArrayObject) {\r\n this._vertexArrayObject = null;\r\n }\r\n\r\n this._createIndexBuffer();\r\n\r\n this._spriteBuffer?._rebuild();\r\n\r\n this._vertexBuffer?._rebuild();\r\n\r\n for (const key in this._vertexBuffers) {\r\n this._vertexBuffers[key]._rebuild();\r\n }\r\n\r\n this.resetDrawCache();\r\n }\r\n\r\n /**\r\n * Is this system ready to be used/rendered\r\n * @returns true if the system is ready\r\n */\r\n public isReady(): boolean {\r\n if (!this.emitter || (this._imageProcessingConfiguration && !this._imageProcessingConfiguration.isReady()) || !this.particleTexture || !this.particleTexture.isReady()) {\r\n return false;\r\n }\r\n\r\n if (this.blendMode !== ParticleSystem.BLENDMODE_MULTIPLYADD) {\r\n if (!this._getWrapper(this.blendMode).effect!.isReady()) {\r\n return false;\r\n }\r\n } else {\r\n if (!this._getWrapper(ParticleSystem.BLENDMODE_MULTIPLY).effect!.isReady()) {\r\n return false;\r\n }\r\n if (!this._getWrapper(ParticleSystem.BLENDMODE_ADD).effect!.isReady()) {\r\n return false;\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n private _render(blendMode: number) {\r\n const drawWrapper = this._getWrapper(blendMode);\r\n const effect = drawWrapper.effect!;\r\n\r\n const engine = this._engine;\r\n\r\n // Render\r\n engine.enableEffect(drawWrapper);\r\n\r\n const viewMatrix = this.defaultViewMatrix ?? this._scene!.getViewMatrix();\r\n effect.setTexture(\"diffuseSampler\", this.particleTexture);\r\n effect.setMatrix(\"view\", viewMatrix);\r\n effect.setMatrix(\"projection\", this.defaultProjectionMatrix ?? this._scene!.getProjectionMatrix());\r\n\r\n if (this._isAnimationSheetEnabled && this.particleTexture) {\r\n const baseSize = this.particleTexture.getBaseSize();\r\n effect.setFloat3(\"particlesInfos\", this.spriteCellWidth / baseSize.width, this.spriteCellHeight / baseSize.height, this.spriteCellWidth / baseSize.width);\r\n }\r\n\r\n effect.setVector2(\"translationPivot\", this.translationPivot);\r\n effect.setFloat4(\"textureMask\", this.textureMask.r, this.textureMask.g, this.textureMask.b, this.textureMask.a);\r\n\r\n if (this._isBillboardBased && this._scene) {\r\n const camera = this._scene.activeCamera!;\r\n effect.setVector3(\"eyePosition\", camera.globalPosition);\r\n }\r\n\r\n if (this._rampGradientsTexture) {\r\n if (!this._rampGradients || !this._rampGradients.length) {\r\n this._rampGradientsTexture.dispose();\r\n this._rampGradientsTexture = null;\r\n }\r\n effect.setTexture(\"rampSampler\", this._rampGradientsTexture);\r\n }\r\n\r\n const defines = effect.defines;\r\n\r\n if (this._scene) {\r\n bindClipPlane(effect, this, this._scene);\r\n }\r\n\r\n if (defines.indexOf(\"#define BILLBOARDMODE_ALL\") >= 0) {\r\n viewMatrix.invertToRef(TmpVectors.Matrix[0]);\r\n effect.setMatrix(\"invView\", TmpVectors.Matrix[0]);\r\n }\r\n\r\n if (this._vertexArrayObject !== undefined) {\r\n if (!this._vertexArrayObject) {\r\n this._vertexArrayObject = this._engine.recordVertexArrayObject(this._vertexBuffers, this._indexBuffer, effect);\r\n }\r\n\r\n this._engine.bindVertexArrayObject(this._vertexArrayObject, this._indexBuffer);\r\n } else {\r\n engine.bindBuffers(this._vertexBuffers, this._indexBuffer, effect);\r\n }\r\n\r\n // Log. depth\r\n if (this.useLogarithmicDepth && this._scene) {\r\n MaterialHelper.BindLogDepth(defines, effect, this._scene);\r\n }\r\n\r\n // image processing\r\n if (this._imageProcessingConfiguration && !this._imageProcessingConfiguration.applyByPostProcess) {\r\n this._imageProcessingConfiguration.bind(effect);\r\n }\r\n\r\n // Draw order\r\n switch (blendMode) {\r\n case ParticleSystem.BLENDMODE_ADD:\r\n engine.setAlphaMode(Constants.ALPHA_ADD);\r\n break;\r\n case ParticleSystem.BLENDMODE_ONEONE:\r\n engine.setAlphaMode(Constants.ALPHA_ONEONE);\r\n break;\r\n case ParticleSystem.BLENDMODE_STANDARD:\r\n engine.setAlphaMode(Constants.ALPHA_COMBINE);\r\n break;\r\n case ParticleSystem.BLENDMODE_MULTIPLY:\r\n engine.setAlphaMode(Constants.ALPHA_MULTIPLY);\r\n break;\r\n }\r\n\r\n if (this._onBeforeDrawParticlesObservable) {\r\n this._onBeforeDrawParticlesObservable.notifyObservers(effect);\r\n }\r\n\r\n if (this._useInstancing) {\r\n engine.drawArraysType(Constants.MATERIAL_TriangleStripDrawMode, 0, 4, this._particles.length);\r\n } else {\r\n engine.drawElementsType(Constants.MATERIAL_TriangleFillMode, 0, this._particles.length * 6);\r\n }\r\n\r\n return this._particles.length;\r\n }\r\n\r\n /**\r\n * Renders the particle system in its current state.\r\n * @returns the current number of particles\r\n */\r\n public render(): number {\r\n // Check\r\n if (!this.isReady() || !this._particles.length) {\r\n return 0;\r\n }\r\n\r\n const engine = this._engine as any;\r\n if (engine.setState) {\r\n engine.setState(false);\r\n\r\n if (this.forceDepthWrite) {\r\n engine.setDepthWrite(true);\r\n }\r\n }\r\n\r\n let outparticles = 0;\r\n\r\n if (this.blendMode === ParticleSystem.BLENDMODE_MULTIPLYADD) {\r\n outparticles = this._render(ParticleSystem.BLENDMODE_MULTIPLY) + this._render(ParticleSystem.BLENDMODE_ADD);\r\n } else {\r\n outparticles = this._render(this.blendMode);\r\n }\r\n\r\n this._engine.unbindInstanceAttributes();\r\n this._engine.setAlphaMode(Constants.ALPHA_DISABLE);\r\n\r\n return outparticles;\r\n }\r\n\r\n /**\r\n * Disposes the particle system and free the associated resources\r\n * @param disposeTexture defines if the particle texture must be disposed as well (true by default)\r\n */\r\n public dispose(disposeTexture = true): void {\r\n this.resetDrawCache();\r\n\r\n if (this._vertexBuffer) {\r\n this._vertexBuffer.dispose();\r\n this._vertexBuffer = null;\r\n }\r\n\r\n if (this._spriteBuffer) {\r\n this._spriteBuffer.dispose();\r\n this._spriteBuffer = null;\r\n }\r\n\r\n if (this._indexBuffer) {\r\n this._engine._releaseBuffer(this._indexBuffer);\r\n this._indexBuffer = null;\r\n }\r\n\r\n if (this._vertexArrayObject) {\r\n this._engine.releaseVertexArrayObject(this._vertexArrayObject);\r\n this._vertexArrayObject = null;\r\n }\r\n\r\n if (disposeTexture && this.particleTexture) {\r\n this.particleTexture.dispose();\r\n this.particleTexture = null;\r\n }\r\n\r\n if (disposeTexture && this.noiseTexture) {\r\n this.noiseTexture.dispose();\r\n this.noiseTexture = null;\r\n }\r\n\r\n if (this._rampGradientsTexture) {\r\n this._rampGradientsTexture.dispose();\r\n this._rampGradientsTexture = null;\r\n }\r\n\r\n this._removeFromRoot();\r\n\r\n if (this.subEmitters && !this._subEmitters) {\r\n this._prepareSubEmitterInternalArray();\r\n }\r\n\r\n if (this._subEmitters && this._subEmitters.length) {\r\n for (let index = 0; index < this._subEmitters.length; index++) {\r\n for (const subEmitter of this._subEmitters[index]) {\r\n subEmitter.dispose();\r\n }\r\n }\r\n\r\n this._subEmitters = [];\r\n this.subEmitters = [];\r\n }\r\n\r\n if (this._disposeEmitterOnDispose && this.emitter && (this.emitter as AbstractMesh).dispose) {\r\n (this.emitter).dispose(true);\r\n }\r\n\r\n if (this._onBeforeDrawParticlesObservable) {\r\n this._onBeforeDrawParticlesObservable.clear();\r\n }\r\n\r\n // Remove from scene\r\n if (this._scene) {\r\n const index = this._scene.particleSystems.indexOf(this);\r\n if (index > -1) {\r\n this._scene.particleSystems.splice(index, 1);\r\n }\r\n\r\n this._scene._activeParticleSystems.dispose();\r\n }\r\n\r\n // Callback\r\n this.onDisposeObservable.notifyObservers(this);\r\n this.onDisposeObservable.clear();\r\n this.onStoppedObservable.clear();\r\n\r\n this.reset();\r\n }\r\n\r\n // Clone\r\n /**\r\n * Clones the particle system.\r\n * @param name The name of the cloned object\r\n * @param newEmitter The new emitter to use\r\n * @param cloneTexture Also clone the textures if true\r\n * @returns the cloned particle system\r\n */\r\n public clone(name: string, newEmitter: any, cloneTexture = false): ParticleSystem {\r\n const custom = { ...this._customWrappers };\r\n let program: any = null;\r\n const engine = this._engine as Engine;\r\n if (engine.createEffectForParticles) {\r\n if (this.customShader != null) {\r\n program = this.customShader;\r\n const defines: string = program.shaderOptions.defines.length > 0 ? program.shaderOptions.defines.join(\"\\n\") : \"\";\r\n const effect = engine.createEffectForParticles(program.shaderPath.fragmentElement, program.shaderOptions.uniforms, program.shaderOptions.samplers, defines);\r\n if (!custom[0]) {\r\n this.setCustomEffect(effect, 0);\r\n } else {\r\n custom[0].effect = effect;\r\n }\r\n }\r\n }\r\n\r\n const serialization = this.serialize(cloneTexture);\r\n const result = ParticleSystem.Parse(serialization, this._scene || this._engine, this._rootUrl);\r\n result.name = name;\r\n result.customShader = program;\r\n result._customWrappers = custom;\r\n\r\n if (newEmitter === undefined) {\r\n newEmitter = this.emitter;\r\n }\r\n\r\n if (this.noiseTexture) {\r\n result.noiseTexture = this.noiseTexture.clone();\r\n }\r\n\r\n result.emitter = newEmitter;\r\n if (!this.preventAutoStart) {\r\n result.start();\r\n }\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Serializes the particle system to a JSON object\r\n * @param serializeTexture defines if the texture must be serialized as well\r\n * @returns the JSON object\r\n */\r\n public serialize(serializeTexture = false): any {\r\n const serializationObject: any = {};\r\n\r\n ParticleSystem._Serialize(serializationObject, this, serializeTexture);\r\n\r\n serializationObject.textureMask = this.textureMask.asArray();\r\n serializationObject.customShader = this.customShader;\r\n serializationObject.preventAutoStart = this.preventAutoStart;\r\n\r\n // SubEmitters\r\n if (this.subEmitters) {\r\n serializationObject.subEmitters = [];\r\n\r\n if (!this._subEmitters) {\r\n this._prepareSubEmitterInternalArray();\r\n }\r\n\r\n for (const subs of this._subEmitters) {\r\n const cell = [];\r\n for (const sub of subs) {\r\n cell.push(sub.serialize(serializeTexture));\r\n }\r\n\r\n serializationObject.subEmitters.push(cell);\r\n }\r\n }\r\n\r\n return serializationObject;\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _Serialize(serializationObject: any, particleSystem: IParticleSystem, serializeTexture: boolean) {\r\n serializationObject.name = particleSystem.name;\r\n serializationObject.id = particleSystem.id;\r\n\r\n serializationObject.capacity = particleSystem.getCapacity();\r\n\r\n serializationObject.disposeOnStop = particleSystem.disposeOnStop;\r\n serializationObject.manualEmitCount = particleSystem.manualEmitCount;\r\n\r\n // Emitter\r\n if ((particleSystem.emitter).position) {\r\n const emitterMesh = particleSystem.emitter;\r\n serializationObject.emitterId = emitterMesh.id;\r\n } else {\r\n const emitterPosition = particleSystem.emitter;\r\n serializationObject.emitter = emitterPosition.asArray();\r\n }\r\n\r\n // Emitter\r\n if (particleSystem.particleEmitterType) {\r\n serializationObject.particleEmitterType = particleSystem.particleEmitterType.serialize();\r\n }\r\n\r\n if (particleSystem.particleTexture) {\r\n if (serializeTexture) {\r\n serializationObject.texture = particleSystem.particleTexture.serialize();\r\n } else {\r\n serializationObject.textureName = particleSystem.particleTexture.name;\r\n serializationObject.invertY = !!(particleSystem.particleTexture as any)._invertY;\r\n }\r\n }\r\n\r\n serializationObject.isLocal = particleSystem.isLocal;\r\n\r\n // Animations\r\n SerializationHelper.AppendSerializedAnimations(particleSystem, serializationObject);\r\n serializationObject.beginAnimationOnStart = particleSystem.beginAnimationOnStart;\r\n serializationObject.beginAnimationFrom = particleSystem.beginAnimationFrom;\r\n serializationObject.beginAnimationTo = particleSystem.beginAnimationTo;\r\n serializationObject.beginAnimationLoop = particleSystem.beginAnimationLoop;\r\n\r\n // Particle system\r\n serializationObject.startDelay = particleSystem.startDelay;\r\n serializationObject.renderingGroupId = particleSystem.renderingGroupId;\r\n serializationObject.isBillboardBased = particleSystem.isBillboardBased;\r\n serializationObject.billboardMode = particleSystem.billboardMode;\r\n serializationObject.minAngularSpeed = particleSystem.minAngularSpeed;\r\n serializationObject.maxAngularSpeed = particleSystem.maxAngularSpeed;\r\n serializationObject.minSize = particleSystem.minSize;\r\n serializationObject.maxSize = particleSystem.maxSize;\r\n serializationObject.minScaleX = particleSystem.minScaleX;\r\n serializationObject.maxScaleX = particleSystem.maxScaleX;\r\n serializationObject.minScaleY = particleSystem.minScaleY;\r\n serializationObject.maxScaleY = particleSystem.maxScaleY;\r\n serializationObject.minEmitPower = particleSystem.minEmitPower;\r\n serializationObject.maxEmitPower = particleSystem.maxEmitPower;\r\n serializationObject.minLifeTime = particleSystem.minLifeTime;\r\n serializationObject.maxLifeTime = particleSystem.maxLifeTime;\r\n serializationObject.emitRate = particleSystem.emitRate;\r\n serializationObject.gravity = particleSystem.gravity.asArray();\r\n serializationObject.noiseStrength = particleSystem.noiseStrength.asArray();\r\n serializationObject.color1 = particleSystem.color1.asArray();\r\n serializationObject.color2 = particleSystem.color2.asArray();\r\n serializationObject.colorDead = particleSystem.colorDead.asArray();\r\n serializationObject.updateSpeed = particleSystem.updateSpeed;\r\n serializationObject.targetStopDuration = particleSystem.targetStopDuration;\r\n serializationObject.blendMode = particleSystem.blendMode;\r\n serializationObject.preWarmCycles = particleSystem.preWarmCycles;\r\n serializationObject.preWarmStepOffset = particleSystem.preWarmStepOffset;\r\n serializationObject.minInitialRotation = particleSystem.minInitialRotation;\r\n serializationObject.maxInitialRotation = particleSystem.maxInitialRotation;\r\n serializationObject.startSpriteCellID = particleSystem.startSpriteCellID;\r\n serializationObject.spriteCellLoop = particleSystem.spriteCellLoop;\r\n serializationObject.endSpriteCellID = particleSystem.endSpriteCellID;\r\n serializationObject.spriteCellChangeSpeed = particleSystem.spriteCellChangeSpeed;\r\n serializationObject.spriteCellWidth = particleSystem.spriteCellWidth;\r\n serializationObject.spriteCellHeight = particleSystem.spriteCellHeight;\r\n serializationObject.spriteRandomStartCell = particleSystem.spriteRandomStartCell;\r\n serializationObject.isAnimationSheetEnabled = particleSystem.isAnimationSheetEnabled;\r\n serializationObject.useLogarithmicDepth = particleSystem.useLogarithmicDepth;\r\n\r\n const colorGradients = particleSystem.getColorGradients();\r\n if (colorGradients) {\r\n serializationObject.colorGradients = [];\r\n for (const colorGradient of colorGradients) {\r\n const serializedGradient: any = {\r\n gradient: colorGradient.gradient,\r\n color1: colorGradient.color1.asArray(),\r\n };\r\n\r\n if (colorGradient.color2) {\r\n serializedGradient.color2 = colorGradient.color2.asArray();\r\n } else {\r\n serializedGradient.color2 = colorGradient.color1.asArray();\r\n }\r\n\r\n serializationObject.colorGradients.push(serializedGradient);\r\n }\r\n }\r\n\r\n const rampGradients = particleSystem.getRampGradients();\r\n if (rampGradients) {\r\n serializationObject.rampGradients = [];\r\n for (const rampGradient of rampGradients) {\r\n const serializedGradient: any = {\r\n gradient: rampGradient.gradient,\r\n color: rampGradient.color.asArray(),\r\n };\r\n\r\n serializationObject.rampGradients.push(serializedGradient);\r\n }\r\n serializationObject.useRampGradients = particleSystem.useRampGradients;\r\n }\r\n\r\n const colorRemapGradients = particleSystem.getColorRemapGradients();\r\n if (colorRemapGradients) {\r\n serializationObject.colorRemapGradients = [];\r\n for (const colorRemapGradient of colorRemapGradients) {\r\n const serializedGradient: any = {\r\n gradient: colorRemapGradient.gradient,\r\n factor1: colorRemapGradient.factor1,\r\n };\r\n\r\n if (colorRemapGradient.factor2 !== undefined) {\r\n serializedGradient.factor2 = colorRemapGradient.factor2;\r\n } else {\r\n serializedGradient.factor2 = colorRemapGradient.factor1;\r\n }\r\n\r\n serializationObject.colorRemapGradients.push(serializedGradient);\r\n }\r\n }\r\n\r\n const alphaRemapGradients = particleSystem.getAlphaRemapGradients();\r\n if (alphaRemapGradients) {\r\n serializationObject.alphaRemapGradients = [];\r\n for (const alphaRemapGradient of alphaRemapGradients) {\r\n const serializedGradient: any = {\r\n gradient: alphaRemapGradient.gradient,\r\n factor1: alphaRemapGradient.factor1,\r\n };\r\n\r\n if (alphaRemapGradient.factor2 !== undefined) {\r\n serializedGradient.factor2 = alphaRemapGradient.factor2;\r\n } else {\r\n serializedGradient.factor2 = alphaRemapGradient.factor1;\r\n }\r\n\r\n serializationObject.alphaRemapGradients.push(serializedGradient);\r\n }\r\n }\r\n\r\n const sizeGradients = particleSystem.getSizeGradients();\r\n if (sizeGradients) {\r\n serializationObject.sizeGradients = [];\r\n for (const sizeGradient of sizeGradients) {\r\n const serializedGradient: any = {\r\n gradient: sizeGradient.gradient,\r\n factor1: sizeGradient.factor1,\r\n };\r\n\r\n if (sizeGradient.factor2 !== undefined) {\r\n serializedGradient.factor2 = sizeGradient.factor2;\r\n } else {\r\n serializedGradient.factor2 = sizeGradient.factor1;\r\n }\r\n\r\n serializationObject.sizeGradients.push(serializedGradient);\r\n }\r\n }\r\n\r\n const angularSpeedGradients = particleSystem.getAngularSpeedGradients();\r\n if (angularSpeedGradients) {\r\n serializationObject.angularSpeedGradients = [];\r\n for (const angularSpeedGradient of angularSpeedGradients) {\r\n const serializedGradient: any = {\r\n gradient: angularSpeedGradient.gradient,\r\n factor1: angularSpeedGradient.factor1,\r\n };\r\n\r\n if (angularSpeedGradient.factor2 !== undefined) {\r\n serializedGradient.factor2 = angularSpeedGradient.factor2;\r\n } else {\r\n serializedGradient.factor2 = angularSpeedGradient.factor1;\r\n }\r\n\r\n serializationObject.angularSpeedGradients.push(serializedGradient);\r\n }\r\n }\r\n\r\n const velocityGradients = particleSystem.getVelocityGradients();\r\n if (velocityGradients) {\r\n serializationObject.velocityGradients = [];\r\n for (const velocityGradient of velocityGradients) {\r\n const serializedGradient: any = {\r\n gradient: velocityGradient.gradient,\r\n factor1: velocityGradient.factor1,\r\n };\r\n\r\n if (velocityGradient.factor2 !== undefined) {\r\n serializedGradient.factor2 = velocityGradient.factor2;\r\n } else {\r\n serializedGradient.factor2 = velocityGradient.factor1;\r\n }\r\n\r\n serializationObject.velocityGradients.push(serializedGradient);\r\n }\r\n }\r\n\r\n const dragGradients = particleSystem.getDragGradients();\r\n if (dragGradients) {\r\n serializationObject.dragGradients = [];\r\n for (const dragGradient of dragGradients) {\r\n const serializedGradient: any = {\r\n gradient: dragGradient.gradient,\r\n factor1: dragGradient.factor1,\r\n };\r\n\r\n if (dragGradient.factor2 !== undefined) {\r\n serializedGradient.factor2 = dragGradient.factor2;\r\n } else {\r\n serializedGradient.factor2 = dragGradient.factor1;\r\n }\r\n\r\n serializationObject.dragGradients.push(serializedGradient);\r\n }\r\n }\r\n\r\n const emitRateGradients = particleSystem.getEmitRateGradients();\r\n if (emitRateGradients) {\r\n serializationObject.emitRateGradients = [];\r\n for (const emitRateGradient of emitRateGradients) {\r\n const serializedGradient: any = {\r\n gradient: emitRateGradient.gradient,\r\n factor1: emitRateGradient.factor1,\r\n };\r\n\r\n if (emitRateGradient.factor2 !== undefined) {\r\n serializedGradient.factor2 = emitRateGradient.factor2;\r\n } else {\r\n serializedGradient.factor2 = emitRateGradient.factor1;\r\n }\r\n\r\n serializationObject.emitRateGradients.push(serializedGradient);\r\n }\r\n }\r\n\r\n const startSizeGradients = particleSystem.getStartSizeGradients();\r\n if (startSizeGradients) {\r\n serializationObject.startSizeGradients = [];\r\n for (const startSizeGradient of startSizeGradients) {\r\n const serializedGradient: any = {\r\n gradient: startSizeGradient.gradient,\r\n factor1: startSizeGradient.factor1,\r\n };\r\n\r\n if (startSizeGradient.factor2 !== undefined) {\r\n serializedGradient.factor2 = startSizeGradient.factor2;\r\n } else {\r\n serializedGradient.factor2 = startSizeGradient.factor1;\r\n }\r\n\r\n serializationObject.startSizeGradients.push(serializedGradient);\r\n }\r\n }\r\n\r\n const lifeTimeGradients = particleSystem.getLifeTimeGradients();\r\n if (lifeTimeGradients) {\r\n serializationObject.lifeTimeGradients = [];\r\n for (const lifeTimeGradient of lifeTimeGradients) {\r\n const serializedGradient: any = {\r\n gradient: lifeTimeGradient.gradient,\r\n factor1: lifeTimeGradient.factor1,\r\n };\r\n\r\n if (lifeTimeGradient.factor2 !== undefined) {\r\n serializedGradient.factor2 = lifeTimeGradient.factor2;\r\n } else {\r\n serializedGradient.factor2 = lifeTimeGradient.factor1;\r\n }\r\n\r\n serializationObject.lifeTimeGradients.push(serializedGradient);\r\n }\r\n }\r\n\r\n const limitVelocityGradients = particleSystem.getLimitVelocityGradients();\r\n if (limitVelocityGradients) {\r\n serializationObject.limitVelocityGradients = [];\r\n for (const limitVelocityGradient of limitVelocityGradients) {\r\n const serializedGradient: any = {\r\n gradient: limitVelocityGradient.gradient,\r\n factor1: limitVelocityGradient.factor1,\r\n };\r\n\r\n if (limitVelocityGradient.factor2 !== undefined) {\r\n serializedGradient.factor2 = limitVelocityGradient.factor2;\r\n } else {\r\n serializedGradient.factor2 = limitVelocityGradient.factor1;\r\n }\r\n\r\n serializationObject.limitVelocityGradients.push(serializedGradient);\r\n }\r\n\r\n serializationObject.limitVelocityDamping = particleSystem.limitVelocityDamping;\r\n }\r\n\r\n if (particleSystem.noiseTexture) {\r\n serializationObject.noiseTexture = particleSystem.noiseTexture.serialize();\r\n }\r\n }\r\n\r\n /**\r\n * @internal\r\n */\r\n public static _Parse(parsedParticleSystem: any, particleSystem: IParticleSystem, sceneOrEngine: Scene | ThinEngine, rootUrl: string) {\r\n let scene: Nullable;\r\n\r\n if (sceneOrEngine instanceof ThinEngine) {\r\n scene = null;\r\n } else {\r\n scene = sceneOrEngine as Scene;\r\n }\r\n\r\n const internalClass = GetClass(\"BABYLON.Texture\");\r\n if (internalClass && scene) {\r\n // Texture\r\n if (parsedParticleSystem.texture) {\r\n particleSystem.particleTexture = internalClass.Parse(parsedParticleSystem.texture, scene, rootUrl) as BaseTexture;\r\n } else if (parsedParticleSystem.textureName) {\r\n particleSystem.particleTexture = new internalClass(\r\n rootUrl + parsedParticleSystem.textureName,\r\n scene,\r\n false,\r\n parsedParticleSystem.invertY !== undefined ? parsedParticleSystem.invertY : true\r\n );\r\n particleSystem.particleTexture!.name = parsedParticleSystem.textureName;\r\n }\r\n }\r\n\r\n // Emitter\r\n if (!parsedParticleSystem.emitterId && parsedParticleSystem.emitterId !== 0 && parsedParticleSystem.emitter === undefined) {\r\n particleSystem.emitter = Vector3.Zero();\r\n } else if (parsedParticleSystem.emitterId && scene) {\r\n particleSystem.emitter = scene.getLastMeshById(parsedParticleSystem.emitterId);\r\n } else {\r\n particleSystem.emitter = Vector3.FromArray(parsedParticleSystem.emitter);\r\n }\r\n\r\n particleSystem.isLocal = !!parsedParticleSystem.isLocal;\r\n\r\n // Misc.\r\n if (parsedParticleSystem.renderingGroupId !== undefined) {\r\n particleSystem.renderingGroupId = parsedParticleSystem.renderingGroupId;\r\n }\r\n\r\n if (parsedParticleSystem.isBillboardBased !== undefined) {\r\n particleSystem.isBillboardBased = parsedParticleSystem.isBillboardBased;\r\n }\r\n\r\n if (parsedParticleSystem.billboardMode !== undefined) {\r\n particleSystem.billboardMode = parsedParticleSystem.billboardMode;\r\n }\r\n\r\n if (parsedParticleSystem.useLogarithmicDepth !== undefined) {\r\n particleSystem.useLogarithmicDepth = parsedParticleSystem.useLogarithmicDepth;\r\n }\r\n\r\n // Animations\r\n if (parsedParticleSystem.animations) {\r\n for (let animationIndex = 0; animationIndex < parsedParticleSystem.animations.length; animationIndex++) {\r\n const parsedAnimation = parsedParticleSystem.animations[animationIndex];\r\n const internalClass = GetClass(\"BABYLON.Animation\");\r\n if (internalClass) {\r\n particleSystem.animations.push(internalClass.Parse(parsedAnimation));\r\n }\r\n }\r\n particleSystem.beginAnimationOnStart = parsedParticleSystem.beginAnimationOnStart;\r\n particleSystem.beginAnimationFrom = parsedParticleSystem.beginAnimationFrom;\r\n particleSystem.beginAnimationTo = parsedParticleSystem.beginAnimationTo;\r\n particleSystem.beginAnimationLoop = parsedParticleSystem.beginAnimationLoop;\r\n }\r\n\r\n if (parsedParticleSystem.autoAnimate && scene) {\r\n scene.beginAnimation(\r\n particleSystem,\r\n parsedParticleSystem.autoAnimateFrom,\r\n parsedParticleSystem.autoAnimateTo,\r\n parsedParticleSystem.autoAnimateLoop,\r\n parsedParticleSystem.autoAnimateSpeed || 1.0\r\n );\r\n }\r\n\r\n // Particle system\r\n particleSystem.startDelay = parsedParticleSystem.startDelay | 0;\r\n particleSystem.minAngularSpeed = parsedParticleSystem.minAngularSpeed;\r\n particleSystem.maxAngularSpeed = parsedParticleSystem.maxAngularSpeed;\r\n particleSystem.minSize = parsedParticleSystem.minSize;\r\n particleSystem.maxSize = parsedParticleSystem.maxSize;\r\n\r\n if (parsedParticleSystem.minScaleX) {\r\n particleSystem.minScaleX = parsedParticleSystem.minScaleX;\r\n particleSystem.maxScaleX = parsedParticleSystem.maxScaleX;\r\n particleSystem.minScaleY = parsedParticleSystem.minScaleY;\r\n particleSystem.maxScaleY = parsedParticleSystem.maxScaleY;\r\n }\r\n\r\n if (parsedParticleSystem.preWarmCycles !== undefined) {\r\n particleSystem.preWarmCycles = parsedParticleSystem.preWarmCycles;\r\n particleSystem.preWarmStepOffset = parsedParticleSystem.preWarmStepOffset;\r\n }\r\n\r\n if (parsedParticleSystem.minInitialRotation !== undefined) {\r\n particleSystem.minInitialRotation = parsedParticleSystem.minInitialRotation;\r\n particleSystem.maxInitialRotation = parsedParticleSystem.maxInitialRotation;\r\n }\r\n\r\n particleSystem.minLifeTime = parsedParticleSystem.minLifeTime;\r\n particleSystem.maxLifeTime = parsedParticleSystem.maxLifeTime;\r\n particleSystem.minEmitPower = parsedParticleSystem.minEmitPower;\r\n particleSystem.maxEmitPower = parsedParticleSystem.maxEmitPower;\r\n particleSystem.emitRate = parsedParticleSystem.emitRate;\r\n particleSystem.gravity = Vector3.FromArray(parsedParticleSystem.gravity);\r\n if (parsedParticleSystem.noiseStrength) {\r\n particleSystem.noiseStrength = Vector3.FromArray(parsedParticleSystem.noiseStrength);\r\n }\r\n particleSystem.color1 = Color4.FromArray(parsedParticleSystem.color1);\r\n particleSystem.color2 = Color4.FromArray(parsedParticleSystem.color2);\r\n particleSystem.colorDead = Color4.FromArray(parsedParticleSystem.colorDead);\r\n particleSystem.updateSpeed = parsedParticleSystem.updateSpeed;\r\n particleSystem.targetStopDuration = parsedParticleSystem.targetStopDuration;\r\n particleSystem.blendMode = parsedParticleSystem.blendMode;\r\n\r\n if (parsedParticleSystem.colorGradients) {\r\n for (const colorGradient of parsedParticleSystem.colorGradients) {\r\n particleSystem.addColorGradient(\r\n colorGradient.gradient,\r\n Color4.FromArray(colorGradient.color1),\r\n colorGradient.color2 ? Color4.FromArray(colorGradient.color2) : undefined\r\n );\r\n }\r\n }\r\n\r\n if (parsedParticleSystem.rampGradients) {\r\n for (const rampGradient of parsedParticleSystem.rampGradients) {\r\n particleSystem.addRampGradient(rampGradient.gradient, Color3.FromArray(rampGradient.color));\r\n }\r\n particleSystem.useRampGradients = parsedParticleSystem.useRampGradients;\r\n }\r\n\r\n if (parsedParticleSystem.colorRemapGradients) {\r\n for (const colorRemapGradient of parsedParticleSystem.colorRemapGradients) {\r\n particleSystem.addColorRemapGradient(\r\n colorRemapGradient.gradient,\r\n colorRemapGradient.factor1 !== undefined ? colorRemapGradient.factor1 : colorRemapGradient.factor,\r\n colorRemapGradient.factor2\r\n );\r\n }\r\n }\r\n\r\n if (parsedParticleSystem.alphaRemapGradients) {\r\n for (const alphaRemapGradient of parsedParticleSystem.alphaRemapGradients) {\r\n particleSystem.addAlphaRemapGradient(\r\n alphaRemapGradient.gradient,\r\n alphaRemapGradient.factor1 !== undefined ? alphaRemapGradient.factor1 : alphaRemapGradient.factor,\r\n alphaRemapGradient.factor2\r\n );\r\n }\r\n }\r\n\r\n if (parsedParticleSystem.sizeGradients) {\r\n for (const sizeGradient of parsedParticleSystem.sizeGradients) {\r\n particleSystem.addSizeGradient(sizeGradient.gradient, sizeGradient.factor1 !== undefined ? sizeGradient.factor1 : sizeGradient.factor, sizeGradient.factor2);\r\n }\r\n }\r\n\r\n if (parsedParticleSystem.angularSpeedGradients) {\r\n for (const angularSpeedGradient of parsedParticleSystem.angularSpeedGradients) {\r\n particleSystem.addAngularSpeedGradient(\r\n angularSpeedGradient.gradient,\r\n angularSpeedGradient.factor1 !== undefined ? angularSpeedGradient.factor1 : angularSpeedGradient.factor,\r\n angularSpeedGradient.factor2\r\n );\r\n }\r\n }\r\n\r\n if (parsedParticleSystem.velocityGradients) {\r\n for (const velocityGradient of parsedParticleSystem.velocityGradients) {\r\n particleSystem.addVelocityGradient(\r\n velocityGradient.gradient,\r\n velocityGradient.factor1 !== undefined ? velocityGradient.factor1 : velocityGradient.factor,\r\n velocityGradient.factor2\r\n );\r\n }\r\n }\r\n\r\n if (parsedParticleSystem.dragGradients) {\r\n for (const dragGradient of parsedParticleSystem.dragGradients) {\r\n particleSystem.addDragGradient(dragGradient.gradient, dragGradient.factor1 !== undefined ? dragGradient.factor1 : dragGradient.factor, dragGradient.factor2);\r\n }\r\n }\r\n\r\n if (parsedParticleSystem.emitRateGradients) {\r\n for (const emitRateGradient of parsedParticleSystem.emitRateGradients) {\r\n particleSystem.addEmitRateGradient(\r\n emitRateGradient.gradient,\r\n emitRateGradient.factor1 !== undefined ? emitRateGradient.factor1 : emitRateGradient.factor,\r\n emitRateGradient.factor2\r\n );\r\n }\r\n }\r\n\r\n if (parsedParticleSystem.startSizeGradients) {\r\n for (const startSizeGradient of parsedParticleSystem.startSizeGradients) {\r\n particleSystem.addStartSizeGradient(\r\n startSizeGradient.gradient,\r\n startSizeGradient.factor1 !== undefined ? startSizeGradient.factor1 : startSizeGradient.factor,\r\n startSizeGradient.factor2\r\n );\r\n }\r\n }\r\n\r\n if (parsedParticleSystem.lifeTimeGradients) {\r\n for (const lifeTimeGradient of parsedParticleSystem.lifeTimeGradients) {\r\n particleSystem.addLifeTimeGradient(\r\n lifeTimeGradient.gradient,\r\n lifeTimeGradient.factor1 !== undefined ? lifeTimeGradient.factor1 : lifeTimeGradient.factor,\r\n lifeTimeGradient.factor2\r\n );\r\n }\r\n }\r\n\r\n if (parsedParticleSystem.limitVelocityGradients) {\r\n for (const limitVelocityGradient of parsedParticleSystem.limitVelocityGradients) {\r\n particleSystem.addLimitVelocityGradient(\r\n limitVelocityGradient.gradient,\r\n limitVelocityGradient.factor1 !== undefined ? limitVelocityGradient.factor1 : limitVelocityGradient.factor,\r\n limitVelocityGradient.factor2\r\n );\r\n }\r\n particleSystem.limitVelocityDamping = parsedParticleSystem.limitVelocityDamping;\r\n }\r\n\r\n if (parsedParticleSystem.noiseTexture && scene) {\r\n const internalClass = GetClass(\"BABYLON.ProceduralTexture\");\r\n particleSystem.noiseTexture = internalClass.Parse(parsedParticleSystem.noiseTexture, scene, rootUrl);\r\n }\r\n\r\n // Emitter\r\n let emitterType: IParticleEmitterType;\r\n if (parsedParticleSystem.particleEmitterType) {\r\n switch (parsedParticleSystem.particleEmitterType.type) {\r\n case \"SphereParticleEmitter\":\r\n emitterType = new SphereParticleEmitter();\r\n break;\r\n case \"SphereDirectedParticleEmitter\":\r\n emitterType = new SphereDirectedParticleEmitter();\r\n break;\r\n case \"ConeEmitter\":\r\n case \"ConeParticleEmitter\":\r\n emitterType = new ConeParticleEmitter();\r\n break;\r\n case \"CylinderParticleEmitter\":\r\n emitterType = new CylinderParticleEmitter();\r\n break;\r\n case \"CylinderDirectedParticleEmitter\":\r\n emitterType = new CylinderDirectedParticleEmitter();\r\n break;\r\n case \"HemisphericParticleEmitter\":\r\n emitterType = new HemisphericParticleEmitter();\r\n break;\r\n case \"PointParticleEmitter\":\r\n emitterType = new PointParticleEmitter();\r\n break;\r\n case \"MeshParticleEmitter\":\r\n emitterType = new MeshParticleEmitter();\r\n break;\r\n case \"BoxEmitter\":\r\n case \"BoxParticleEmitter\":\r\n default:\r\n emitterType = new BoxParticleEmitter();\r\n break;\r\n }\r\n\r\n emitterType.parse(parsedParticleSystem.particleEmitterType, scene);\r\n } else {\r\n emitterType = new BoxParticleEmitter();\r\n emitterType.parse(parsedParticleSystem, scene);\r\n }\r\n particleSystem.particleEmitterType = emitterType;\r\n\r\n // Animation sheet\r\n particleSystem.startSpriteCellID = parsedParticleSystem.startSpriteCellID;\r\n particleSystem.endSpriteCellID = parsedParticleSystem.endSpriteCellID;\r\n particleSystem.spriteCellLoop = parsedParticleSystem.spriteCellLoop ?? true;\r\n particleSystem.spriteCellWidth = parsedParticleSystem.spriteCellWidth;\r\n particleSystem.spriteCellHeight = parsedParticleSystem.spriteCellHeight;\r\n particleSystem.spriteCellChangeSpeed = parsedParticleSystem.spriteCellChangeSpeed;\r\n particleSystem.spriteRandomStartCell = parsedParticleSystem.spriteRandomStartCell;\r\n\r\n particleSystem.disposeOnStop = parsedParticleSystem.disposeOnStop ?? false;\r\n particleSystem.manualEmitCount = parsedParticleSystem.manualEmitCount ?? -1;\r\n }\r\n\r\n /**\r\n * Parses a JSON object to create a particle system.\r\n * @param parsedParticleSystem The JSON object to parse\r\n * @param sceneOrEngine The scene or the engine to create the particle system in\r\n * @param rootUrl The root url to use to load external dependencies like texture\r\n * @param doNotStart Ignore the preventAutoStart attribute and does not start\r\n * @param capacity defines the system capacity (if null or undefined the sotred capacity will be used)\r\n * @returns the Parsed particle system\r\n */\r\n public static Parse(parsedParticleSystem: any, sceneOrEngine: Scene | ThinEngine, rootUrl: string, doNotStart = false, capacity?: number): ParticleSystem {\r\n const name = parsedParticleSystem.name;\r\n let custom: Nullable = null;\r\n let program: any = null;\r\n let engine: ThinEngine;\r\n let scene: Nullable;\r\n\r\n if (sceneOrEngine instanceof ThinEngine) {\r\n engine = sceneOrEngine;\r\n } else {\r\n scene = sceneOrEngine as Scene;\r\n engine = scene.getEngine();\r\n }\r\n\r\n if (parsedParticleSystem.customShader && (engine as any).createEffectForParticles) {\r\n program = parsedParticleSystem.customShader;\r\n const defines: string = program.shaderOptions.defines.length > 0 ? program.shaderOptions.defines.join(\"\\n\") : \"\";\r\n custom = (engine as any).createEffectForParticles(program.shaderPath.fragmentElement, program.shaderOptions.uniforms, program.shaderOptions.samplers, defines);\r\n }\r\n const particleSystem = new ParticleSystem(name, capacity || parsedParticleSystem.capacity, sceneOrEngine, custom, parsedParticleSystem.isAnimationSheetEnabled);\r\n particleSystem.customShader = program;\r\n particleSystem._rootUrl = rootUrl;\r\n\r\n if (parsedParticleSystem.id) {\r\n particleSystem.id = parsedParticleSystem.id;\r\n }\r\n\r\n // SubEmitters\r\n if (parsedParticleSystem.subEmitters) {\r\n particleSystem.subEmitters = [];\r\n for (const cell of parsedParticleSystem.subEmitters) {\r\n const cellArray = [];\r\n for (const sub of cell) {\r\n cellArray.push(SubEmitter.Parse(sub, sceneOrEngine, rootUrl));\r\n }\r\n\r\n particleSystem.subEmitters.push(cellArray);\r\n }\r\n }\r\n\r\n ParticleSystem._Parse(parsedParticleSystem, particleSystem, sceneOrEngine, rootUrl);\r\n\r\n if (parsedParticleSystem.textureMask) {\r\n particleSystem.textureMask = Color4.FromArray(parsedParticleSystem.textureMask);\r\n }\r\n\r\n // Auto start\r\n if (parsedParticleSystem.preventAutoStart) {\r\n particleSystem.preventAutoStart = parsedParticleSystem.preventAutoStart;\r\n }\r\n\r\n if (!doNotStart && !particleSystem.preventAutoStart) {\r\n particleSystem.start();\r\n }\r\n\r\n return particleSystem;\r\n }\r\n}\r\n\r\nSubEmitter._ParseParticleSystem = ParticleSystem.Parse;\r\n","import { ArcRotateCamera } from '@babylonjs/core/Cameras/arcRotateCamera';\r\nimport { RawTexture } from '@babylonjs/core/Materials/Textures/rawTexture';\r\nimport { Color4, Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { VertexBuffer } from '@babylonjs/core/Meshes/buffer';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { Logger } from '@babylonjs/core/Misc/logger';\r\nimport { Observable } from '@babylonjs/core/Misc/observable';\r\nimport { ParticleSystem } from '@babylonjs/core/Particles/particleSystem';\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { FloatArray, Nullable } from '@babylonjs/core/types';\r\nimport { MeshNode } from '@models/classes/meshes/mesh-node';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { getCameraPositionWithOffset } from '@view/helpers/math-helper';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { Subject } from 'rxjs';\r\nimport { takeUntil } from 'rxjs/operators';\r\nimport { eMarqueeSelectMode } from '../interaction';\r\nimport { KbGizmo } from './gizmo';\r\n\r\nexport interface VertexChanges {\r\n mesh: AbstractMesh;\r\n delta: Vector3;\r\n vertexIndices: number[];\r\n}\r\n\r\n/**\r\n * Gizmo that enables moving vertices on a mesh.\r\n */\r\nexport class KbVertexSelectGizmo extends KbGizmo {\r\n /** Fires an event when any of it's sub gizmos are dragged */\r\n public onSelectionChange = new Observable<{ selectedIndices: number[] }>();\r\n public vertexTracker = new Subject();\r\n public particleScale = 0.005;\r\n public get attachedMesh() {\r\n return this.__attachedMesh;\r\n }\r\n\r\n public set attachedMesh(mesh: Nullable) {\r\n this.__attachedMesh = mesh;\r\n if (this.particleSystem) {\r\n this.particleSystem.dispose();\r\n this.particleSystem = undefined;\r\n }\r\n if (mesh) {\r\n this.drawVertices(mesh);\r\n }\r\n }\r\n\r\n private verticesUtilLayer: UtilityLayerRenderer;\r\n private particleSystem?: ParticleSystem;\r\n private currentMeshVertexPositions?: FloatArray;\r\n private redrawVertices = false;\r\n private selectionsEnabled: boolean;\r\n private disposeMarquee: Subject;\r\n private saveAnimate: (() => void) | undefined;\r\n\r\n private __attachedMesh: Nullable = null;\r\n private selectedVertices = new Map();\r\n\r\n readonly vertexSelectedColor = new Color4(0.4, 1, 0, 0.5);\r\n readonly vertexUnSelectedColor = new Color4(0, 0.2, 1, 0.5);\r\n readonly vertexTransparentColor = new Color4(0, 0, 0, 0);\r\n\r\n constructor(private kbViewer: KbViewer, private scene: Scene) {\r\n super(kbViewer.utilityLayer);\r\n this.verticesUtilLayer = kbViewer.utilityLayer;\r\n this.attachedMesh = null;\r\n }\r\n\r\n attachTo(target: MeshNode) {\r\n const mesh = this.kbViewer.getRendererNode(target);\r\n if (!mesh) {\r\n throw Error('Could not find mesh in scene to attach gizmo');\r\n }\r\n this.attachedMesh = mesh as Mesh;\r\n\r\n // Clean up old marquee\r\n if (this.disposeMarquee) {\r\n this.disposeMarquee.next();\r\n this.disposeMarquee.complete();\r\n }\r\n\r\n this.disposeMarquee = new Subject();\r\n let selectionMode: null | 'new' | 'add' | 'remove' = null;\r\n\r\n this.kbViewer.interaction\r\n .enableMarqueeSelect(target, eMarqueeSelectMode.Vertices)\r\n .pipe(takeUntil(this.disposeMarquee))\r\n .subscribe(event => {\r\n if (selectionMode || event.vertexList.length > 0) {\r\n const vertices = event.vertexList;\r\n\r\n // If user is making a new selection, set the selection mode based on modifier keys\r\n if (selectionMode === null) {\r\n if (event.ctrlKey) {\r\n selectionMode = 'add';\r\n } else if (event.altKey) {\r\n selectionMode = 'remove';\r\n } else {\r\n selectionMode = 'new';\r\n }\r\n }\r\n\r\n if (selectionMode === 'new') {\r\n this.selectedVertices.clear();\r\n }\r\n if (selectionMode === 'remove') {\r\n vertices.forEach(vertex => this.selectedVertices.delete(vertex));\r\n } else {\r\n vertices.forEach(vertex => this.selectedVertices.set(vertex, null));\r\n }\r\n this.redraw();\r\n this.emitSelectionChange();\r\n }\r\n });\r\n\r\n this.kbViewer.interaction.onPointerUp.pipe(takeUntil(this.disposeMarquee)).subscribe(() => {\r\n selectionMode = null;\r\n });\r\n }\r\n\r\n updateParticleScale(scale: number) {\r\n this.particleScale = scale;\r\n this.redraw();\r\n }\r\n\r\n updateGizmoPosition() {\r\n this.kbViewer.forceAndWaitForNextRender().then(() => {\r\n if (this.__attachedMesh) {\r\n this.currentMeshVertexPositions = this.__attachedMesh\r\n .getVerticesData(VertexBuffer.PositionKind)\r\n ?.slice() as any;\r\n if (!this.currentMeshVertexPositions) {\r\n return;\r\n }\r\n this.redraw();\r\n }\r\n });\r\n }\r\n\r\n dispose() {\r\n if (this.particleSystem) {\r\n this.particleSystem.dispose();\r\n this.particleSystem = undefined;\r\n }\r\n this.disposeMarquee.next();\r\n this.disposeMarquee.complete();\r\n this.kbViewer.interaction.disableMarqueeSelect();\r\n }\r\n\r\n setCustomMesh(mesh: Mesh) {\r\n Logger.Error('Custom meshes are not supported on this gizmo.');\r\n }\r\n\r\n selectVertices(vertexIndices: number[]) {\r\n this.selectedVertices.clear();\r\n vertexIndices.forEach(vertex => this.selectedVertices.set(vertex, null));\r\n this.redraw();\r\n this.emitSelectionChange();\r\n }\r\n\r\n enableSelectionChange() {\r\n this.selectionsEnabled = true;\r\n this.redraw();\r\n }\r\n\r\n disableSelectionChange() {\r\n this.selectionsEnabled = false;\r\n this.redraw();\r\n this.vertexTracker.complete();\r\n }\r\n\r\n getBarycenter(indices: number[]) {\r\n let x = 0,\r\n y = 0,\r\n z = 0;\r\n const currentMeshVertexPositions = this.currentMeshVertexPositions;\r\n\r\n if (currentMeshVertexPositions) {\r\n indices.forEach(idx => {\r\n x += currentMeshVertexPositions[idx * 3];\r\n y += currentMeshVertexPositions[idx * 3 + 1];\r\n z += currentMeshVertexPositions[idx * 3 + 2];\r\n });\r\n }\r\n\r\n return new KbVector(x / indices.length, y / indices.length, z / indices.length);\r\n }\r\n\r\n private redraw() {\r\n this.vertexTracker.next(this.selectedVertices.size.toString());\r\n if (this.particleSystem && this.saveAnimate) {\r\n this.particleSystem.animate = this.saveAnimate;\r\n this.redrawVertices = true;\r\n }\r\n }\r\n\r\n private emitSelectionChange() {\r\n this.onSelectionChange.notifyObservers({ selectedIndices: Array.from(this.selectedVertices.keys()) });\r\n }\r\n\r\n private drawVertices(targetMesh: AbstractMesh) {\r\n if (!this.currentMeshVertexPositions) {\r\n this.currentMeshVertexPositions = targetMesh.getVerticesData(VertexBuffer.PositionKind)?.slice();\r\n }\r\n if (!this.currentMeshVertexPositions) {\r\n return;\r\n }\r\n\r\n const vertexCount = this.currentMeshVertexPositions.length / 3;\r\n this.particleSystem = new ParticleSystem(\r\n 'vertexSelectParticleSystem',\r\n vertexCount,\r\n this.verticesUtilLayer.utilityLayerScene\r\n );\r\n\r\n const camera = this.scene.activeCamera instanceof ArcRotateCamera ? this.scene.activeCamera : null;\r\n const cameraDistance = Vector3.Zero();\r\n const cameraPosition = Vector3.Zero();\r\n this.saveAnimate = this.particleSystem.animate.bind(this.particleSystem);\r\n\r\n let particleIdx = 0;\r\n this.particleSystem.startPositionFunction = (worldMatrix, positionToUpdate, particle) => {\r\n particleIdx += 3;\r\n const vertexPositions = this.currentMeshVertexPositions!;\r\n Vector3.TransformCoordinatesFromFloatsToRef(\r\n vertexPositions[particleIdx],\r\n vertexPositions[particleIdx + 1],\r\n vertexPositions[particleIdx + 2],\r\n worldMatrix,\r\n positionToUpdate\r\n );\r\n };\r\n const transform = targetMesh.computeWorldMatrix(true);\r\n\r\n this.particleSystem.updateFunction = particles => {\r\n let particlesToProcess = false;\r\n if (camera) {\r\n cameraPosition.copyFrom(getCameraPositionWithOffset(camera));\r\n }\r\n\r\n for (let i = 0; i < particles.length; i++) {\r\n const particle = particles[i];\r\n const vertexPositions = this.currentMeshVertexPositions;\r\n\r\n if (vertexPositions && (particle.age === 0 || this.redrawVertices)) {\r\n Vector3.TransformCoordinatesFromFloatsToRef(\r\n vertexPositions[i * 3],\r\n vertexPositions[i * 3 + 1],\r\n vertexPositions[i * 3 + 2],\r\n transform,\r\n particle.position\r\n );\r\n particle.age = 1;\r\n if (this.selectedVertices.has(i)) {\r\n particle.color = this.vertexSelectedColor;\r\n } else {\r\n if (this.selectionsEnabled) {\r\n particle.color = this.vertexUnSelectedColor;\r\n } else {\r\n particle.color = this.vertexTransparentColor;\r\n }\r\n }\r\n if (!particlesToProcess) {\r\n particlesToProcess = true;\r\n }\r\n }\r\n\r\n if (camera) {\r\n cameraPosition.subtractToRef(particle.position, cameraDistance);\r\n particle.size = (this.particleScale - 0.001) * cameraDistance.length();\r\n } else {\r\n particle.size = this.particleScale - 0.004;\r\n }\r\n }\r\n\r\n this.redrawVertices = false;\r\n // If there are a large number of particles, kill the updating function for performance\r\n if (particles.length && particles.length > 10000 && !particlesToProcess && this.particleSystem) {\r\n // stable\r\n this.particleSystem.stop();\r\n this.particleSystem.animate = () => void 0;\r\n }\r\n };\r\n\r\n const colorArray = new Uint8Array(3);\r\n for (let i = 0; i < colorArray.length; i += 3) {\r\n colorArray[i] = 225;\r\n colorArray[i + 1] = 225;\r\n colorArray[i + 2] = 225;\r\n }\r\n\r\n this.particleSystem.emitter = targetMesh;\r\n this.particleSystem.minSize = this.particleScale;\r\n this.particleSystem.maxSize = this.particleScale;\r\n this.particleSystem.disposeOnStop = false;\r\n this.particleSystem.particleTexture = RawTexture.CreateRGBTexture(\r\n colorArray,\r\n 1,\r\n 1,\r\n this.verticesUtilLayer.utilityLayerScene\r\n );\r\n this.particleSystem.manualEmitCount = vertexCount;\r\n // this.particleSystem.emitRate\r\n this.particleSystem.start();\r\n }\r\n}\r\n","import { Quaternion, Vector3 } from '@babylonjs/core/Maths/math';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { Observable } from '@babylonjs/core/Misc/observable';\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { eViewpointTargetMode, eViewpointType } from '@models/classes/enums';\r\nimport { MeshNode } from '@models/classes/meshes';\r\nimport { ViewpointNode } from '@models/classes/viewpoints';\r\nimport { rotationToVector } from '@view/helpers/math-helper';\r\nimport { createViewpointVisualization } from '@view/helpers/viewpoint-helper';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { KbGizmo } from './gizmo';\r\nimport { KbPositionGizmo } from './position-gizmo';\r\nimport { KbRotationGizmo } from './rotation-gizmo';\r\n\r\n/**\r\n * Gizmo that enables dragging and rotating lights\r\n */\r\nexport class KbViewpointGizmo extends KbGizmo {\r\n posGizmo: KbPositionGizmo | undefined;\r\n targetGizmo: KbPositionGizmo | undefined;\r\n rotGizmo: KbRotationGizmo | undefined;\r\n\r\n _attachedViewpoint: ViewpointNode | undefined;\r\n _gizmoMesh: AbstractMesh | undefined;\r\n _visualMesh: AbstractMesh | undefined;\r\n _targetVisualMesh: AbstractMesh | undefined;\r\n\r\n viewpointPositionGizmo: Mesh | undefined;\r\n\r\n /** Fires an event when any of it's sub gizmos are dragged */\r\n onDragStartObservable = new Observable();\r\n /** Fires an event while any of it's sub gizmos are dragged */\r\n onDragObservable = new Observable<{ deltaPos?: Vector3; deltaTargetPos?: Vector3; deltaRotation?: Quaternion }>();\r\n /** Fires an event when any of it's sub gizmos are released from dragging */\r\n onDragEndObservable = new Observable<{\r\n deltaPos?: Vector3;\r\n deltaTargetPos?: Vector3;\r\n deltaRotation?: Quaternion;\r\n }>();\r\n\r\n previousTargetMode?: eViewpointTargetMode;\r\n\r\n attachTo(viewpoint: ViewpointNode) {\r\n // Get rid of any old gizmos\r\n if (this.posGizmo) {\r\n this.posGizmo.dispose();\r\n this.posGizmo = undefined;\r\n }\r\n if (this.targetGizmo) {\r\n this.targetGizmo.dispose();\r\n this.targetGizmo = undefined;\r\n }\r\n if (this.rotGizmo) {\r\n this.rotGizmo.dispose();\r\n this.rotGizmo = undefined;\r\n }\r\n\r\n // Create mesh to attach gizmos to, and the camera preview visualization attached to that mesh\r\n this._visualMesh = new AbstractMesh('', this.gizmoLayer.utilityLayerScene);\r\n const { targetPositionMarker, viewpointPositionGizmo } = createViewpointVisualization(\r\n viewpoint,\r\n this.gizmoLayer.utilityLayerScene\r\n );\r\n this._visualMesh.addChild(viewpointPositionGizmo);\r\n\r\n // Always create the position gizmo\r\n this.posGizmo = new KbPositionGizmo(this.kbViewer, this.gizmoLayer);\r\n this.previousTargetMode = viewpoint.targetMode;\r\n\r\n // Determine which gizmos to show. No extra gizmos for overhead camera (only position)\r\n // For the target point type viewpoint, make a separate mesh to hold the position gizmo\r\n if (viewpoint.type === eViewpointType.overhead) {\r\n viewpointPositionGizmo.scaling = new Vector3(1, -1, 1);\r\n viewpointPositionGizmo.bakeCurrentTransformIntoVertices();\r\n } else {\r\n if (viewpoint.targetMode === eViewpointTargetMode.rotation) {\r\n this.rotGizmo = new KbRotationGizmo(this.kbViewer, this.gizmoLayer, { x: true, y: true, z: false });\r\n\r\n // The first person camera is oriented pointing parallel to the +Z axis\r\n if (viewpoint.type == eViewpointType.firstPerson) {\r\n viewpointPositionGizmo.rotate(Vector3.Right(), Math.PI / 2);\r\n viewpointPositionGizmo.bakeCurrentTransformIntoVertices();\r\n }\r\n } else if (viewpoint.targetMode === eViewpointTargetMode.point) {\r\n this._gizmoMesh = new AbstractMesh('', this.gizmoLayer.utilityLayerScene);\r\n this._targetVisualMesh = new AbstractMesh('', this.gizmoLayer.utilityLayerScene);\r\n if (targetPositionMarker) {\r\n this._targetVisualMesh?.addChild(targetPositionMarker);\r\n }\r\n this.targetGizmo = new KbPositionGizmo(this.kbViewer, this.gizmoLayer);\r\n }\r\n }\r\n if (!this._gizmoMesh) {\r\n this._gizmoMesh = this._visualMesh;\r\n } else {\r\n this._visualMesh.isPickable = false;\r\n }\r\n\r\n this.updateHelperMeshes(viewpoint);\r\n\r\n // Set up event handlers on the gizmos\r\n if (this.posGizmo) {\r\n this.posGizmo.attachedMesh = this._gizmoMesh;\r\n const positionChange = Vector3.Zero();\r\n this.posGizmo.onDragStartObservable.add(event => {\r\n positionChange.setAll(0);\r\n this.onDragStartObservable.notifyObservers();\r\n });\r\n this.posGizmo.onDragObservable.add(event => {\r\n if (this._gizmoMesh !== this._visualMesh) {\r\n this._visualMesh?.position.addInPlace(event.delta);\r\n }\r\n\r\n positionChange.addInPlace(event.delta);\r\n this.onDragObservable.notifyObservers({ deltaPos: event.delta });\r\n this.updateHelperMeshDirectionToRotation(this._attachedViewpoint!);\r\n });\r\n this.posGizmo.onDragEndObservable.add(event => {\r\n this.onDragEndObservable.notifyObservers({ deltaPos: positionChange });\r\n });\r\n }\r\n if (this.targetGizmo) {\r\n this.targetGizmo.attachedMesh = this._targetVisualMesh!;\r\n const targetChange = Vector3.Zero();\r\n this.targetGizmo.onDragStartObservable.add(event => {\r\n targetChange.setAll(0);\r\n this.onDragStartObservable.notifyObservers();\r\n });\r\n this.targetGizmo.onDragObservable.add(event => {\r\n targetChange.addInPlace(event.delta);\r\n this.onDragObservable.notifyObservers({ deltaTargetPos: event.delta });\r\n this.updateHelperMeshDirectionToRotation(this._attachedViewpoint!);\r\n });\r\n this.targetGizmo.onDragEndObservable.add(event => {\r\n this.onDragEndObservable.notifyObservers({ deltaTargetPos: targetChange });\r\n });\r\n }\r\n if (this.rotGizmo) {\r\n this.rotGizmo.attachedMesh = this._visualMesh;\r\n let rotationChange = this._visualMesh.rotationQuaternion || Quaternion.Identity();\r\n this.rotGizmo.onDragStartObservable.add(event => {\r\n this.onDragStartObservable.notifyObservers();\r\n rotationChange = Quaternion.Identity();\r\n });\r\n this.rotGizmo.onDragObservable.add(event => {\r\n const { delta } = event;\r\n rotationChange.multiplyInPlace(delta);\r\n this.onDragObservable.notifyObservers({ deltaRotation: delta });\r\n });\r\n this.rotGizmo.onDragEndObservable.add(() => {\r\n this.onDragEndObservable.notifyObservers({ deltaRotation: rotationChange });\r\n });\r\n }\r\n\r\n this._attachedViewpoint = viewpoint;\r\n }\r\n\r\n updateGizmoPosition(viewpointNode?: ViewpointNode) {\r\n if (viewpointNode && viewpointNode !== this._attachedViewpoint) {\r\n this._attachedViewpoint = viewpointNode;\r\n }\r\n\r\n // If the target mode changed, rebuild the gizmo\r\n if (this._attachedViewpoint && this.previousTargetMode !== this._attachedViewpoint.targetMode) {\r\n const viewpoint = this._attachedViewpoint;\r\n this.dispose(true);\r\n this.attachTo(viewpoint);\r\n } else {\r\n if (this._attachedViewpoint) {\r\n this.updateHelperMeshes(this._attachedViewpoint);\r\n }\r\n super.updateGizmoPosition();\r\n }\r\n }\r\n\r\n private updateHelperMeshes(viewpoint: ViewpointNode) {\r\n if (!this._visualMesh || !this._gizmoMesh) {\r\n return;\r\n }\r\n this._visualMesh.position = viewpoint.position.toVec3();\r\n this._gizmoMesh.position = viewpoint.position.toVec3();\r\n\r\n if (this._targetVisualMesh) {\r\n this._targetVisualMesh.position = viewpoint.targetPoint.toVec3();\r\n }\r\n\r\n if (viewpoint.targetMode === eViewpointTargetMode.rotation) {\r\n const eulerRot = viewpoint.rotation.toVec3().scale(Math.PI / 180);\r\n this._visualMesh.rotationQuaternion = Quaternion.FromEulerVector(eulerRot);\r\n } else {\r\n this.updateHelperMeshDirectionToRotation(viewpoint);\r\n }\r\n }\r\n\r\n private updateHelperMeshDirectionToRotation(viewpoint: ViewpointNode, delta?: Vector3) {\r\n if (!this._visualMesh) {\r\n return;\r\n }\r\n if (!delta) {\r\n delta = Vector3.Zero();\r\n }\r\n const _visualMesh = this._visualMesh;\r\n\r\n if (viewpoint.targetMode !== eViewpointTargetMode.rotation) {\r\n let direction: Vector3;\r\n\r\n if (this._targetVisualMesh) {\r\n direction = this._targetVisualMesh.position.subtract(_visualMesh.position);\r\n } else {\r\n // mesh\r\n const targetMeshNode = viewpoint._manager!.getById(viewpoint.targetMeshId!) as MeshNode;\r\n if (targetMeshNode) {\r\n direction = targetMeshNode.position.subtract(viewpoint.position).toVec3();\r\n } else {\r\n direction = Vector3.Zero();\r\n }\r\n }\r\n const rotation = rotationToVector(direction);\r\n _visualMesh.rotationQuaternion = rotation;\r\n\r\n // This code is for fixing the rotation of the camera gizmo so that the x axis is always in the XZ plane (keeping the camera model level with the horizon.)\r\n // But, it is not working quite right.\r\n // _visualMesh.computeWorldMatrix(true);\r\n // let newXAxis = _visualMesh.getDirection(Vector3.Right());\r\n // const correction = rotationToVector(direction.cross(Vector3.Up()), newXAxis);\r\n // _visualMesh.rotationQuaternion = _visualMesh.rotationQuaternion.multiply(correction);\r\n }\r\n }\r\n\r\n /**\r\n * Creates a PositionGizmo\r\n * @param gizmoLayer The utility layer the gizmo will be added to\r\n */\r\n constructor(\r\n private kbViewer: KbViewer,\r\n gizmoLayer: UtilityLayerRenderer = UtilityLayerRenderer.DefaultUtilityLayer\r\n ) {\r\n super(gizmoLayer);\r\n\r\n this.attachedMesh = null;\r\n }\r\n\r\n /**\r\n * Disposes of the gizmo\r\n */\r\n dispose(recreate?: boolean) {\r\n if (this.posGizmo) {\r\n this.posGizmo.dispose();\r\n }\r\n if (this.rotGizmo) {\r\n this.rotGizmo.dispose();\r\n }\r\n if (this.targetGizmo) {\r\n this.targetGizmo.dispose();\r\n }\r\n if (this._visualMesh) {\r\n this._visualMesh.dispose();\r\n }\r\n if (this._targetVisualMesh) {\r\n this._targetVisualMesh.dispose();\r\n }\r\n if (this.viewpointPositionGizmo) {\r\n this.viewpointPositionGizmo.dispose();\r\n }\r\n if (!recreate) {\r\n this.onDragStartObservable.clear();\r\n this.onDragEndObservable.clear();\r\n delete this._attachedViewpoint;\r\n }\r\n }\r\n}\r\n","import { PBRMaterial } from '@babylonjs/core/Materials/PBR/pbrMaterial';\r\nimport { Texture } from '@babylonjs/core/Materials/Textures/texture';\r\nimport { Material } from '@babylonjs/core/Materials/material';\r\nimport { Color3, Matrix, Quaternion, Vector3 } from '@babylonjs/core/Maths';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport '@babylonjs/core/Rendering/boundingBoxRenderer'; // Required by mesh bounding box feature\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { UvMapFeature } from '@models/classes';\r\nimport { IVertexGroup } from '@models/classes/kb3d-models';\r\nimport { LightNode } from '@models/classes/lights/light-node';\r\nimport { MeshNode, SpaceNode } from '@models/classes/meshes';\r\nimport { KbQuaternion } from '@models/classes/primitives/kb-quaternion';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { ViewpointNode } from '@models/classes/viewpoints';\r\nimport { KbViewer } from '@view/kb-viewer';\r\nimport { Subject } from 'rxjs';\r\nimport { KbAxisAngleGizmo } from './gizmos/axis-angle-gizmo';\r\nimport { KbFacetSelectGizmo } from './gizmos/facet-select-gizmo';\r\nimport { KbGizmo } from './gizmos/gizmo';\r\nimport { KbLightGizmo } from './gizmos/light-gizmo';\r\nimport { KbPositionGizmo } from './gizmos/position-gizmo';\r\nimport { KbRotationGizmo } from './gizmos/rotation-gizmo';\r\nimport { KbScaleGizmo } from './gizmos/scale-gizmo';\r\nimport { KbVertexSelectGizmo } from './gizmos/vertex-select-gizmo';\r\nimport { KbViewpointGizmo } from './gizmos/viewpoint-gizmo';\r\n\r\ninterface GizmoInfo {\r\n subjects: Array>;\r\n gizmo: KbGizmo;\r\n}\r\n\r\n/**\r\n * Manages the various gizmos and their connections to the scene meshes.\r\n * Keeps track of what gizmo is active and the selected meshes to put the gizmos on.\r\n * @param scene The root scene on which the utility/gizmo layer will be placed.\r\n * @param camera The camera being used. Required by some gizmos for displaying them correctly\r\n */\r\nconst PREVIEW_NAME = '__UV_PREVIEW_MATERIAL__';\r\nexport class GizmoTools {\r\n activeGizmoSettings: GizmoInfo[] = [];\r\n dragging = false;\r\n uvMapStoredMaterials = new Map();\r\n private scene: Scene;\r\n private gizmoLayer: UtilityLayerRenderer;\r\n private previewMaterial: PBRMaterial | undefined;\r\n\r\n constructor(scene: Scene, private kbViewer: KbViewer) {\r\n this.scene = scene;\r\n this.gizmoLayer = kbViewer.utilityLayer;\r\n this.gizmoLayer.utilityLayerScene.useRightHandedSystem = true;\r\n }\r\n\r\n selectLight(lightNode: LightNode) {\r\n const lightGizmo = new KbLightGizmo(this.kbViewer, this.gizmoLayer);\r\n\r\n const updatePosition = new Subject();\r\n const updateDirection = new Subject();\r\n const totalDeltaPosition = new Subject();\r\n this.activeGizmoSettings.push({\r\n subjects: [updatePosition, updateDirection, totalDeltaPosition],\r\n gizmo: lightGizmo,\r\n });\r\n\r\n lightGizmo.attachTo(lightNode);\r\n this.dragging = false;\r\n\r\n return {\r\n updateLightGizmo: (node: LightNode) => {\r\n lightGizmo.updateGizmoPosition(node);\r\n },\r\n };\r\n }\r\n\r\n selectViewpoint(viewpoint: ViewpointNode) {\r\n const viewpointGizmo = new KbViewpointGizmo(this.kbViewer, this.gizmoLayer);\r\n viewpointGizmo.attachTo(viewpoint);\r\n\r\n const updatePosition = new Subject();\r\n const updateTargetPosition = new Subject();\r\n const updateRotation = new Subject();\r\n const totalDeltaPosition = new Subject();\r\n const totalDeltaTargetPosition = new Subject();\r\n const totalDeltaRotation = new Subject();\r\n this.activeGizmoSettings.push({\r\n subjects: [\r\n updatePosition,\r\n updateTargetPosition,\r\n updateRotation,\r\n totalDeltaPosition,\r\n totalDeltaTargetPosition,\r\n totalDeltaRotation,\r\n ],\r\n gizmo: viewpointGizmo,\r\n });\r\n\r\n this.dragging = false;\r\n const totalDeltaPositionQueue = Vector3.Zero();\r\n const totalDeltaTargetPositionQueue = Vector3.Zero();\r\n const totalDeltaRotationQueue = Quaternion.Identity();\r\n\r\n viewpointGizmo.onDragStartObservable.add(() => {\r\n this.dragging = true;\r\n });\r\n viewpointGizmo.onDragObservable.add(event => {\r\n if (this.dragging) {\r\n if (event.deltaPos) {\r\n updatePosition.next(KbVector.FromVec3(event.deltaPos));\r\n totalDeltaPositionQueue.addInPlace(event.deltaPos);\r\n }\r\n if (event.deltaTargetPos) {\r\n updateTargetPosition.next(KbVector.FromVec3(event.deltaTargetPos));\r\n totalDeltaTargetPositionQueue.addInPlace(event.deltaTargetPos);\r\n }\r\n if (event.deltaRotation) {\r\n updateRotation.next(KbVector.FromVec3(event.deltaRotation.toEulerAngles().scale(180 / Math.PI)));\r\n totalDeltaRotationQueue.multiplyInPlace(event.deltaRotation);\r\n }\r\n }\r\n });\r\n viewpointGizmo.onDragEndObservable.add(() => {\r\n if (this.dragging) {\r\n this.dragging = false;\r\n if (!totalDeltaPositionQueue.equalsToFloats(0, 0, 0)) {\r\n totalDeltaPosition.next(KbVector.FromVec3(totalDeltaPositionQueue));\r\n totalDeltaPositionQueue.setAll(0);\r\n } else if (!totalDeltaTargetPositionQueue.equalsToFloats(0, 0, 0)) {\r\n totalDeltaTargetPosition.next(KbVector.FromVec3(totalDeltaTargetPositionQueue));\r\n totalDeltaTargetPositionQueue.setAll(0);\r\n } else if (!totalDeltaRotationQueue.equals(Quaternion.Identity())) {\r\n totalDeltaRotation.next(\r\n KbVector.FromVec3(totalDeltaRotationQueue.toEulerAngles().scale(180 / Math.PI))\r\n );\r\n totalDeltaRotationQueue.copyFrom(Quaternion.Identity());\r\n }\r\n }\r\n });\r\n\r\n return {\r\n updatePosition: updatePosition.asObservable(),\r\n updateTargetPosition: updateTargetPosition.asObservable(),\r\n updateRotation: updateRotation.asObservable(),\r\n totalDeltaPosition: totalDeltaPosition.asObservable(),\r\n totalDeltaTargetPosition: totalDeltaTargetPosition.asObservable(),\r\n totalDeltaRotation: totalDeltaRotation.asObservable(),\r\n updateViewpointGizmo: (node: ViewpointNode) => {\r\n viewpointGizmo.updateGizmoPosition(node);\r\n },\r\n };\r\n }\r\n\r\n position(\r\n target: SpaceNode | KbVector,\r\n positionSpace?: SpaceNode,\r\n axisVisibility?: [boolean, boolean, boolean],\r\n orientGizmo?: Quaternion\r\n ) {\r\n const positionGizmo = new KbPositionGizmo(this.kbViewer, this.gizmoLayer, axisVisibility);\r\n positionGizmo.attachTo(target, positionSpace, orientGizmo);\r\n\r\n const updatePosition = new Subject();\r\n const totalDelta = new Subject();\r\n this.activeGizmoSettings.push({\r\n subjects: [updatePosition, totalDelta],\r\n gizmo: positionGizmo,\r\n });\r\n this.dragging = false;\r\n let totalDeltaQueue = Vector3.Zero();\r\n\r\n positionGizmo.onDragStartObservable.add(() => {\r\n this.dragging = true;\r\n });\r\n positionGizmo.onDragObservable.add(event => {\r\n if (this.dragging) {\r\n totalDeltaQueue.addInPlace(event.delta);\r\n updatePosition.next(KbVector.FromVec3(event.delta));\r\n }\r\n });\r\n positionGizmo.onDragEndObservable.add(() => {\r\n if (this.dragging) {\r\n totalDelta.next(KbVector.FromVec3(totalDeltaQueue));\r\n totalDeltaQueue = Vector3.Zero();\r\n this.dragging = false;\r\n }\r\n });\r\n\r\n return {\r\n update: updatePosition.asObservable(),\r\n totalDelta: totalDelta.asObservable(),\r\n updateGizmoPosition: (update?: KbVector) => positionGizmo.updateGizmoPosition(update),\r\n dispose: () => this.clearGizmo(positionGizmo),\r\n };\r\n }\r\n\r\n rotateAxisAngle(\r\n target: KbVector,\r\n positionSpace?: SpaceNode,\r\n options?: { placementOffset?: KbVector; disableAngle?: boolean }\r\n ) {\r\n const updateAxisAngle = new Subject<{ axis: KbVector; angle: number }>();\r\n const totalDelta = new Subject<{ axis: KbVector; angle: number }>();\r\n const totalDeltaQueue = { axis: Vector3.Zero(), angle: 0 };\r\n\r\n const axisAngleGizmo = new KbAxisAngleGizmo(this.kbViewer, this.gizmoLayer, options?.disableAngle);\r\n axisAngleGizmo.attachTo(target, positionSpace, options?.placementOffset);\r\n\r\n this.activeGizmoSettings.push({\r\n subjects: [updateAxisAngle, totalDelta],\r\n gizmo: axisAngleGizmo,\r\n });\r\n this.dragging = false;\r\n\r\n axisAngleGizmo.onDragStartObservable.add(() => {\r\n this.dragging = true;\r\n });\r\n axisAngleGizmo.onDragObservable.add(event => {\r\n if (this.dragging) {\r\n const angle = (event.deltaAngle * 180) / Math.PI;\r\n totalDeltaQueue.axis.addInPlace(event.deltaAxis);\r\n totalDeltaQueue.angle += angle;\r\n updateAxisAngle.next({ axis: KbVector.FromVec3(event.deltaAxis), angle });\r\n }\r\n });\r\n axisAngleGizmo.onDragEndObservable.add(() => {\r\n if (this.dragging) {\r\n totalDelta.next({ axis: KbVector.FromVec3(totalDeltaQueue.axis), angle: totalDeltaQueue.angle });\r\n totalDeltaQueue.axis = Vector3.Zero();\r\n totalDeltaQueue.angle = 0;\r\n this.dragging = false;\r\n }\r\n });\r\n\r\n return {\r\n update: updateAxisAngle.asObservable(),\r\n totalDelta: totalDelta.asObservable(),\r\n updateGizmoPosition: (updateAxis?: KbVector, updatePosition?: KbVector) =>\r\n axisAngleGizmo.updateGizmoPosition(updateAxis, updatePosition),\r\n dispose: () => this.clearGizmo(axisAngleGizmo),\r\n };\r\n }\r\n\r\n /**\r\n * Creates a rotation gizmo that can rotate many items\r\n * nodes will be rotated and repositioned in space if the center of rotation is not equal to the position\r\n * vectors will be repositioned in space as if each vector is the position of a point\r\n */\r\n rotate(\r\n targets: Array,\r\n positionSpace?: SpaceNode,\r\n placementOffset?: KbVector,\r\n axisVisibility?: [boolean, boolean, boolean]\r\n ) {\r\n let gizmoOffset = Vector3.Zero();\r\n\r\n // Get the transform matrix to get the world space rotation and coordinates of the node we are rotating\r\n let positionTransform: Matrix | undefined;\r\n if (positionSpace) {\r\n positionTransform = this.kbViewer.getRendererNode(positionSpace)?.computeWorldMatrix(true);\r\n } else if (targets[0] instanceof SpaceNode && targets[0]._parent) {\r\n positionTransform = this.kbViewer.getRendererNode(targets[0]._parent)?.computeWorldMatrix(true);\r\n }\r\n const rotationGizmo = new KbRotationGizmo(this.kbViewer, this.gizmoLayer, undefined, axisVisibility);\r\n\r\n if (targets.length === 1) {\r\n const target = targets[0];\r\n rotationGizmo.attachTo(target, positionSpace, placementOffset);\r\n } else {\r\n // If multiple targets are selected, put the pivot in the average of the positions\r\n // Maybe enhance this to use bounding box later, if that's important...\r\n const pivot = targets\r\n .reduce((result: KbVector, current) => {\r\n return result.add(current instanceof SpaceNode ? current.position : current);\r\n }, KbVector.Zero())\r\n .scale(1 / targets.length);\r\n rotationGizmo.attachTo(KbVector.Zero(), undefined, pivot);\r\n }\r\n\r\n const update = new Subject<\r\n Array<{ rotation: KbVector; position: KbVector; axisTranslate: KbVector; worldTransformMatrix: Matrix }>\r\n >();\r\n const totalDelta = new Subject<{\r\n rotation: KbVector;\r\n position: KbVector;\r\n axisTranslate: KbVector;\r\n }>();\r\n this.activeGizmoSettings.push({\r\n subjects: [update, totalDelta],\r\n gizmo: rotationGizmo,\r\n });\r\n\r\n this.dragging = false;\r\n let totalDeltaQueue = {\r\n rotation: Quaternion.Identity(),\r\n position: Vector3.Zero(),\r\n axisTranslate: Vector3.Zero(),\r\n };\r\n\r\n // Set up caches for the positions and rotations of the nodes because it's not guaranteed that they will be updated by reference\r\n // And as you drag the nodes these values will change and the gizmos won't be dragging in the right direction anymore\r\n const positions = targets.map(target =>\r\n target instanceof SpaceNode ? target.position.toVec3() : target.toVec3()\r\n );\r\n const axes = targets.map(target => (target instanceof SpaceNode ? target.rotationAxis.toVec3() : null));\r\n const quaternionCache = targets.map(target =>\r\n target instanceof SpaceNode\r\n ? Quaternion.FromEulerVector(target.eulerRotation.toVec3().scale(Math.PI / 180))\r\n : Quaternion.Identity()\r\n );\r\n\r\n // Get the current pivot which will always be where the gizmo is located (minus the offset which just visually offsets the gizmo position)\r\n let gizmoPosition = rotationGizmo.attachedMesh!.position;\r\n\r\n rotationGizmo.onDragStartObservable.add(() => {\r\n this.dragging = true;\r\n if (rotationGizmo.gizmoOffset) {\r\n gizmoOffset = rotationGizmo.gizmoOffset.toVec3();\r\n }\r\n if (gizmoOffset) {\r\n gizmoPosition = gizmoPosition.subtract(gizmoOffset);\r\n }\r\n });\r\n rotationGizmo.onDragObservable.add(event => {\r\n if (this.dragging) {\r\n const result: Array<{\r\n rotation: KbVector;\r\n position: KbVector;\r\n axisTranslate: KbVector;\r\n worldTransformMatrix: Matrix;\r\n }> = [];\r\n\r\n const rotationMatrix = new Matrix();\r\n\r\n const gizmoRotation = rotationGizmo.attachedMesh!.rotationQuaternion || Quaternion.Identity();\r\n // Get the true world space rotation in the case that the gizmo is rotated\r\n // The gizmos output their rotation deltas in local space\r\n const meshSpaceRotation = gizmoRotation.multiply(event.delta).multiply(gizmoRotation.conjugate());\r\n meshSpaceRotation.toRotationMatrix(rotationMatrix);\r\n const worldTransformMatrix = new Matrix();\r\n event.delta.toRotationMatrix(worldTransformMatrix);\r\n worldTransformMatrix.multiplyToRef(\r\n rotationGizmo.attachedMesh!.computeWorldMatrix(),\r\n worldTransformMatrix\r\n );\r\n\r\n // Inverse the position transform to rotate the delta back into local \"positionSpace\" space\r\n const positionTransformInv = positionTransform?.clone().invert();\r\n\r\n positions.forEach((position, index) => {\r\n let positionDelta: Vector3;\r\n let axisDelta: Vector3 | undefined;\r\n const target = targets[index];\r\n const axis = axes[index];\r\n\r\n // If the node is a space node, then rotate it around the pivot to get the updated position\r\n if (target instanceof SpaceNode) {\r\n // Treat the pivot point as the center of the node so the node rotates around the pivot point\r\n let worldNodePivot = target.pivot.toVec3();\r\n const nodeTransform = this.kbViewer.getRendererNode(target)?.computeWorldMatrix(true);\r\n if (nodeTransform) {\r\n worldNodePivot = Vector3.TransformCoordinates(worldNodePivot, nodeTransform);\r\n }\r\n positionDelta = Vector3.TransformCoordinates(\r\n worldNodePivot.subtract(gizmoPosition),\r\n rotationMatrix\r\n )\r\n .add(gizmoPosition)\r\n .subtract(worldNodePivot);\r\n // If the node is a vector, then it's a direction vector, so we want to rotate it as if the vector started from the origin.\r\n // So don't rotate around the pivot point. The updated position will actually be the updated direction.\r\n } else {\r\n positionDelta = Vector3.TransformCoordinates(position, rotationMatrix).subtract(position);\r\n }\r\n\r\n if (axis) {\r\n axisDelta = Vector3.TransformCoordinates(\r\n axis.length() === 0 ? new Vector3(0, 1, 0) : axis,\r\n rotationMatrix\r\n ).subtract(axis);\r\n axis.addInPlace(axisDelta);\r\n }\r\n\r\n // Rotate the delta vector into local \"positionSpace\" space\r\n if (positionTransformInv) {\r\n positionDelta = Vector3.TransformCoordinates(\r\n positionDelta,\r\n positionTransformInv.getRotationMatrix()\r\n );\r\n }\r\n\r\n // Update the position cache by reference (make sure this position variable isn't shadowed)\r\n position.addInPlace(positionDelta);\r\n totalDeltaQueue.position.addInPlace(positionDelta);\r\n if (axisDelta) {\r\n totalDeltaQueue.axisTranslate.addInPlace(axisDelta);\r\n }\r\n\r\n // Get the rotation delta by converting the previous quaternion rotation to euler, rotating to the new position, and getting euler angles of that.\r\n // This is necessary because it we output the plain quaternion delta euler, the euler rotation order could result\r\n // In an incorrect rotation if there are other euler angles being rotated first\r\n const previousAngle = quaternionCache[index].toEulerAngles();\r\n quaternionCache[index].multiplyInPlace(event.delta);\r\n const newAngle = quaternionCache[index].toEulerAngles();\r\n\r\n totalDeltaQueue.rotation.multiplyInPlace(event.delta);\r\n result.push({\r\n position: KbVector.FromVec3(positionDelta),\r\n rotation: KbVector.FromVec3(newAngle.subtract(previousAngle).scale(180 / Math.PI)),\r\n worldTransformMatrix: worldTransformMatrix,\r\n axisTranslate: axisDelta ? KbVector.FromVec3(axisDelta) : KbVector.Zero(),\r\n });\r\n });\r\n update.next(result);\r\n }\r\n });\r\n rotationGizmo.onDragEndObservable.add(() => {\r\n if (this.dragging) {\r\n totalDelta.next({\r\n rotation: KbVector.FromVec3(totalDeltaQueue.rotation.toEulerAngles().scale(180 / Math.PI)),\r\n position: KbVector.FromVec3(totalDeltaQueue.position),\r\n axisTranslate: KbVector.FromVec3(totalDeltaQueue.axisTranslate),\r\n });\r\n totalDeltaQueue = {\r\n rotation: Quaternion.Identity(),\r\n position: Vector3.Zero(),\r\n axisTranslate: Vector3.Zero(),\r\n };\r\n this.dragging = false;\r\n }\r\n });\r\n\r\n return {\r\n update: update.asObservable(),\r\n totalDelta: totalDelta.asObservable(),\r\n updateGizmoPosition: (update?: KbVector, offset?: KbVector) =>\r\n rotationGizmo.updateGizmoPosition(update, offset),\r\n dispose: () => this.clearGizmo(rotationGizmo),\r\n };\r\n }\r\n\r\n scale(targets: Array, positionSpace?: SpaceNode) {\r\n // if (targets.length === 0) {\r\n // return {\r\n // update: empty(),\r\n // totalDelta: empty(),\r\n // };\r\n // }\r\n\r\n const scaleGizmo = new KbScaleGizmo(this.kbViewer, this.gizmoLayer);\r\n\r\n if (targets.length === 1) {\r\n const target = targets[0];\r\n scaleGizmo.attachTo(target, positionSpace);\r\n } else {\r\n // If multiple targets are selected, put the pivot in the average of the positions\r\n // Maybe enhance this to use bounding box later, if that's important...\r\n const pivot = targets\r\n .reduce((result: KbVector, current) => {\r\n return result.add(current instanceof SpaceNode ? current.position : current);\r\n }, KbVector.Zero())\r\n .scale(1 / targets.length);\r\n\r\n scaleGizmo.attachTo(pivot, positionSpace);\r\n }\r\n\r\n const update = new Subject>();\r\n const totalDelta = new Subject();\r\n this.activeGizmoSettings.push({\r\n subjects: [update, totalDelta],\r\n gizmo: scaleGizmo,\r\n });\r\n\r\n this.dragging = false;\r\n let totalDeltaQueue = Vector3.Zero();\r\n let positionTransform: Matrix | undefined;\r\n\r\n // Get the transform matrix to get the world space rotation and coordinates of the node we are scaling\r\n if (positionSpace) {\r\n positionTransform = this.kbViewer.getRendererNode(positionSpace)?.computeWorldMatrix(true);\r\n } else if (targets[0] instanceof SpaceNode && targets[0]._parent) {\r\n positionTransform = this.kbViewer.getRendererNode(targets[0]._parent)?.computeWorldMatrix(true);\r\n }\r\n\r\n // Get the current pivot which will always be where the gizmo is located\r\n let gizmoPosition = scaleGizmo.attachedMesh!.position;\r\n\r\n scaleGizmo.onDragStartObservable.add(() => {\r\n this.dragging = true;\r\n if (scaleGizmo.gizmoOffset) {\r\n gizmoPosition = gizmoPosition.subtract(scaleGizmo.gizmoOffset.toVec3());\r\n }\r\n });\r\n scaleGizmo.onDragObservable.add(event => {\r\n if (this.dragging) {\r\n const result: Array<{ scaling: KbVector; position: KbVector }> = [];\r\n\r\n targets.forEach(target => {\r\n let positionDelta: Vector3;\r\n\r\n // If the node is a space node, then rotate it around the pivot to get the updated position\r\n if (target instanceof SpaceNode) {\r\n // Treat the pivot point as the center of the node so the node rotates around the pivot point\r\n let worldNodePivot = target.pivot.toVec3();\r\n const nodeTransform = this.kbViewer.getRendererNode(target)?.computeWorldMatrix(true);\r\n if (nodeTransform) {\r\n worldNodePivot = Vector3.TransformCoordinates(worldNodePivot, nodeTransform);\r\n }\r\n positionDelta = worldNodePivot\r\n .subtract(gizmoPosition)\r\n .multiply(event.delta)\r\n .add(gizmoPosition)\r\n .subtract(worldNodePivot);\r\n // If the node is a vector, then it's a direction vector, so we want to rotate it as if the vector started from the origin.\r\n // So don't rotate around the pivot point. The updated position will actually be the updated direction.\r\n } else {\r\n // Transform the node position to world space because the pivot and gizmo will be in world space as well\r\n let worldPosition = target.toVec3();\r\n if (positionTransform) {\r\n worldPosition = Vector3.TransformCoordinates(worldPosition, positionTransform);\r\n }\r\n positionDelta = worldPosition.multiply(event.delta).subtract(worldPosition);\r\n }\r\n\r\n totalDeltaQueue.addInPlace(event.delta);\r\n result.push({\r\n position: KbVector.FromVec3(positionDelta),\r\n scaling: KbVector.FromVec3(event.delta),\r\n });\r\n });\r\n update.next(result);\r\n }\r\n });\r\n scaleGizmo.onDragEndObservable.add(() => {\r\n if (this.dragging) {\r\n totalDelta.next(KbVector.FromVec3(totalDeltaQueue));\r\n totalDeltaQueue = Vector3.Zero();\r\n this.dragging = false;\r\n }\r\n });\r\n\r\n return {\r\n update: update.asObservable(),\r\n totalDelta: totalDelta.asObservable(),\r\n updateGizmoPosition: (update?: KbVector) => scaleGizmo.updateGizmoPosition(update),\r\n dispose: () => this.clearGizmo(scaleGizmo),\r\n };\r\n }\r\n\r\n selectVertices(target: MeshNode, selectedVertices?: number[]) {\r\n const vertexSelectionGizmo = new KbVertexSelectGizmo(this.kbViewer, this.scene);\r\n vertexSelectionGizmo.attachTo(target);\r\n\r\n const update = new Subject<{\r\n vertexIndices: number[];\r\n }>();\r\n this.activeGizmoSettings.push({\r\n subjects: [update],\r\n gizmo: vertexSelectionGizmo,\r\n });\r\n\r\n if (selectedVertices) {\r\n vertexSelectionGizmo.selectVertices(selectedVertices);\r\n }\r\n\r\n this.dragging = false;\r\n vertexSelectionGizmo.onSelectionChange.add(result => {\r\n if (result) {\r\n update.next({\r\n vertexIndices: result.selectedIndices,\r\n });\r\n }\r\n });\r\n\r\n return {\r\n update: update.asObservable(),\r\n vertexTracker: vertexSelectionGizmo.vertexTracker,\r\n updateGizmoPositions: () => vertexSelectionGizmo.updateGizmoPosition(),\r\n updateParticleScale: (scale: number) => vertexSelectionGizmo.updateParticleScale(scale),\r\n disableSelectionChange: () => vertexSelectionGizmo.disableSelectionChange(),\r\n enableSelectionChange: () => vertexSelectionGizmo.enableSelectionChange(),\r\n getBarycenter: (indices: number[]) => vertexSelectionGizmo.getBarycenter(indices),\r\n dispose: () => this.clearGizmo(vertexSelectionGizmo),\r\n };\r\n }\r\n\r\n selectFacets(target: MeshNode, selectedVertices?: IVertexGroup[]) {\r\n const facetSelectionGizmo = new KbFacetSelectGizmo(this.kbViewer, this.scene);\r\n facetSelectionGizmo.attachTo(target);\r\n\r\n const update = new Subject<{\r\n selectedFacets: Array>;\r\n }>();\r\n this.activeGizmoSettings.push({\r\n subjects: [update],\r\n gizmo: facetSelectionGizmo,\r\n });\r\n\r\n if (selectedVertices) {\r\n facetSelectionGizmo.selectFacetGroups(selectedVertices);\r\n }\r\n\r\n this.dragging = false;\r\n facetSelectionGizmo.onSelectionChange.add(result => {\r\n if (result) {\r\n update.next({\r\n selectedFacets: result.selectedFacets,\r\n });\r\n }\r\n });\r\n\r\n return {\r\n update: update.asObservable(),\r\n updateGizmoPositions: () => facetSelectionGizmo.updateGizmoPosition(),\r\n disableSelectionChange: () => facetSelectionGizmo.disableSelectionChange(),\r\n enableSelectionChange: () => facetSelectionGizmo.enableSelectionChange(),\r\n dispose: () => this.clearGizmo(facetSelectionGizmo),\r\n\r\n setBackfaceCulling: (enabled: boolean) => facetSelectionGizmo.setBackfaceCulling(enabled),\r\n setTolerance: (tolerance: number) => facetSelectionGizmo.setTolerance(tolerance),\r\n setVisualization: (opacity: number) => facetSelectionGizmo.setVisualization(opacity),\r\n expandSelection: (tolerance?: number) => facetSelectionGizmo.expandSelection(tolerance),\r\n };\r\n }\r\n\r\n clearGizmo(gizmo?: KbGizmo) {\r\n if (gizmo) {\r\n const index = this.activeGizmoSettings.findIndex(g => g.gizmo === gizmo);\r\n if (index >= 0) {\r\n const gizmoSettings = this.activeGizmoSettings[index];\r\n gizmoSettings.subjects.forEach(subject => subject.complete());\r\n gizmoSettings.gizmo.dispose();\r\n\r\n this.activeGizmoSettings.splice(index, 1);\r\n }\r\n } else {\r\n this.activeGizmoSettings.forEach(gizmoSettings => {\r\n gizmoSettings.subjects.forEach(subject => subject.complete());\r\n gizmoSettings.gizmo.dispose();\r\n });\r\n this.activeGizmoSettings = [];\r\n }\r\n }\r\n\r\n updateGizmoPositions() {\r\n if (!this.dragging) {\r\n this.activeGizmoSettings.forEach(gizmoSettings => {\r\n gizmoSettings.gizmo.updateGizmoPosition();\r\n });\r\n }\r\n }\r\n\r\n showUvMapMaterial(target: UvMapFeature) {\r\n if (target._showMaterial && target._previewMaterialVisible) {\r\n if (target._parent) {\r\n const mesh = this.kbViewer.getRendererNode(target._parent);\r\n if (mesh instanceof Mesh) {\r\n if (mesh.material != null && mesh.material.name != PREVIEW_NAME) {\r\n const previousMaterial = mesh.material;\r\n this.uvMapStoredMaterials.set(target.id, previousMaterial);\r\n }\r\n\r\n if (!this.previewMaterial) {\r\n this.previewMaterial = new PBRMaterial(PREVIEW_NAME, this.kbViewer.scene);\r\n const previewTexture = new Texture('/assets/images/uv_colorgrid.png', this.kbViewer.scene);\r\n this.previewMaterial.albedoTexture = previewTexture;\r\n this.previewMaterial.albedoColor = Color3.White();\r\n this.previewMaterial.metallic = 0;\r\n this.previewMaterial.roughness = 1;\r\n }\r\n\r\n mesh.material = this.previewMaterial;\r\n }\r\n }\r\n } else {\r\n if (target._parent) {\r\n const mesh = this.kbViewer.getRendererNode(target._parent);\r\n if (mesh instanceof Mesh) {\r\n const storedMaterial = this.uvMapStoredMaterials.get(target.id);\r\n if (storedMaterial) {\r\n mesh.material = storedMaterial;\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Bounding mesh code is not used right now, but it may be useful when we are doing mates\r\n // private calculateBoundingBox(nodes: TransformNode[]) {\r\n // const meshes = nodes.filter((node): node is Mesh => node instanceof Mesh);\r\n // meshes.forEach(mesh => mesh.computeWorldMatrix(true));\r\n // if (meshes.length === 0) {\r\n // return {\r\n // pivot: Vector3.Zero(),\r\n // boundingInfo: new BoundingInfo(Vector3.Zero(), Vector3.Zero())\r\n // };\r\n // }\r\n\r\n // let boundingInfo = meshes[0].getBoundingInfo();\r\n // let minimumWorld = boundingInfo.boundingBox.minimumWorld;\r\n // let maximumWorld = boundingInfo.boundingBox.maximumWorld;\r\n\r\n // for (let i = 1; i < meshes.length; i++) {\r\n // let boundingInfo = meshes[i].getBoundingInfo();\r\n // minimumWorld = Vector3.Minimize(minimumWorld, boundingInfo.boundingBox.minimumWorld);\r\n // maximumWorld = Vector3.Maximize(maximumWorld, boundingInfo.boundingBox.maximumWorld);\r\n // }\r\n // let bounding = new BoundingInfo(minimumWorld, maximumWorld);\r\n // let middlePivot = new Vector3(bounding.boundingBox.center.x, bounding.boundingBox.center.y, bounding.boundingBox.center.z);\r\n\r\n // return {\r\n // pivot: middlePivot,\r\n // boundingInfo: bounding\r\n // };\r\n // }\r\n\r\n // private displayKbBoundingBox(mesh: Mesh) {\r\n // let oMesh: Mesh & { kbBoundingBox?: boundingBoxGeometry };\r\n // oMesh = mesh;\r\n // var bbNode: boundingBoxGeometry;\r\n\r\n // if (this.currentSelectedMeshes.length === 0) {\r\n // if (oMesh.kbBoundingBox) {\r\n // oMesh.kbBoundingBox.wireFrame.dispose();\r\n // delete oMesh.kbBoundingBox;\r\n // }\r\n // return;\r\n // }\r\n\r\n // // if the mesh is resized/rotated/something we need to recalculate the bounding box\r\n // mesh.computeWorldMatrix(true);\r\n // mesh.refreshBoundingInfo();\r\n\r\n // mesh.refreshBoundingInfo = (applySkeleton?: boolean) => {\r\n // return mesh;\r\n // };\r\n\r\n // const center = mesh.getBoundingInfo().boundingBox.centerWorld.clone();\r\n // const diagonalLen = mesh.getBoundingInfo().diagonalLength;\r\n\r\n // const min = mesh.getBoundingInfo().boundingBox.minimumWorld;\r\n // const max = mesh.getBoundingInfo().boundingBox.maximumWorld;\r\n // const d = max.subtract(min);\r\n\r\n // const bOffset = 0.02;\r\n // d.x += bOffset;\r\n // d.y += bOffset;\r\n // d.z += bOffset;\r\n // // calculate the position of the bounding box vertices\r\n // const v1 = new Vector3(center.x - d.x / 2, center.y + d.y / 2, center.z + d.z / 2);\r\n // const v2 = new Vector3(center.x + d.x / 2, center.y + d.y / 2, center.z + d.z / 2);\r\n // const v3 = new Vector3(center.x - d.x / 2, center.y - d.y / 2, center.z + d.z / 2);\r\n // const v4 = new Vector3(center.x + d.x / 2, center.y - d.y / 2, center.z + d.z / 2);\r\n // const v5 = new Vector3(center.x - d.x / 2, center.y + d.y / 2, center.z - d.z / 2);\r\n // const v6 = new Vector3(center.x + d.x / 2, center.y + d.y / 2, center.z - d.z / 2);\r\n // const v7 = new Vector3(center.x - d.x / 2, center.y - d.y / 2, center.z - d.z / 2);\r\n // const v8 = new Vector3(center.x + d.x / 2, center.y - d.y / 2, center.z - d.z / 2);\r\n\r\n // const lines = Mesh.CreateLines(KBMAXPREFIX + \"boundingBox\", [\r\n // v1, v2,\r\n // v1, v3,\r\n // v3, v4,\r\n // v4, v2,\r\n // v2, v6,\r\n // v6, v5,\r\n // v5, v7,\r\n // v7, v8,\r\n // v8, v6,\r\n // v8, v4,\r\n // v3, v7,\r\n // v5, v1\r\n // ], this.gizmoLayer.utilityLayerScene);\r\n // lines.color = new Color3(1, 1, 1);\r\n // lines.isPickable = false;\r\n\r\n // if (oMesh.kbBoundingBox) {\r\n // // the bounding box exists so it must be recalculated and repositioned\r\n // bbNode = oMesh.kbBoundingBox;\r\n // bbNode.wireFrame.dispose(); // it is easier to recreate it then reposition it\r\n // bbNode.wireFrame = lines;\r\n // }\r\n // else {\r\n // // the bounding box does not exist, so it must be created\r\n // bbNode = new boundingBoxGeometry(lines);\r\n // }\r\n\r\n // oMesh.kbBoundingBox = bbNode; // force to attach the current boundingBox\r\n // }\r\n\r\n private filterToChangedValues(vector: Vector3 | Quaternion): Partial {\r\n return {\r\n x: vector.x ? vector.x : undefined,\r\n y: vector.y ? vector.y : undefined,\r\n z: vector.z ? vector.z : undefined,\r\n w: 'w' in vector ? vector.w : undefined,\r\n };\r\n }\r\n}\r\n\r\n// class boundingBoxGeometry {\r\n// mateHelper1?: Mesh;\r\n// mateHelper2?: Mesh;\r\n// mateHelper3?: Mesh;\r\n// mateHelper4?: Mesh;\r\n// mateHelper5?: Mesh;\r\n// mateHelper6?: Mesh;\r\n// wireFrame: Mesh;\r\n\r\n// constructor(wireFrame: Mesh) {\r\n// this.wireFrame = wireFrame;\r\n// }\r\n// }\r\n","import { Camera } from '@babylonjs/core/Cameras/camera';\r\nimport { BoundingSphere } from '@babylonjs/core/Culling/boundingSphere';\r\nimport { Ray } from '@babylonjs/core/Culling/ray';\r\nimport { Plane } from '@babylonjs/core/Maths/math.plane';\r\nimport { Vector3 } from '@babylonjs/core/Maths/math.vector';\r\nimport { LinesMesh } from '@babylonjs/core/Meshes/linesMesh';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { SketchControlPoint, SketchPath, SpaceNode, Token, getByDynamicId } from '@models/classes';\r\nimport { eAxis, eSketchPathType } from '@models/classes/enums';\r\nimport { SketchNode } from '@models/classes/meshes/sketch-node';\r\nimport { KbColor } from '@models/classes/primitives/kb-color';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { roundVectorToPlaces } from '@view/helpers/math-helper';\r\nimport { createSketchPointForEditor, getSketchNormalVector3 } from '@view/modules/sketching/util';\r\nimport { CONNECTOR_DIAM } from '@view/render/features/connector';\r\nimport { Subject } from 'rxjs';\r\nimport { take, takeUntil } from 'rxjs/operators';\r\nimport { KbViewer } from '../kb-viewer';\r\nimport { KbArcRotateCamera } from './cameras/kb-arc-rotate-camera';\r\n\r\nconst DefaultSketchOptions = {\r\n faceCamera: true,\r\n snapSize: 0.5,\r\n};\r\n\r\nlet activeHitPoint: SketchControlPoint | undefined;\r\nlet sketchPlaneIntersect: KbVector;\r\n\r\nexport type ISketchOptions = Partial;\r\n\r\nexport class SketchEditor {\r\n mouseUp = new Subject();\r\n\r\n constructor(private kbViewer: KbViewer, private scene: Scene, private defaultHighlightColor: KbColor) {}\r\n\r\n private currentOptions: Required | undefined;\r\n private end = new Subject();\r\n private sketchPoints: SketchControlPoint[] = [];\r\n private activeSketch: SketchNode | undefined;\r\n\r\n updateActiveOptions(options: ISketchOptions) {\r\n if (this.currentOptions) {\r\n for (const prop in options) {\r\n // @ts-expect-error typing not smart enough to know that option objects are the same\r\n this.currentOptions[prop] = options[prop];\r\n }\r\n }\r\n }\r\n\r\n start(sketch: SketchNode) {\r\n let origin = KbVector.Zero();\r\n for (let i = 0; i < sketch.paths.length; i++) {\r\n const path = sketch.paths[i];\r\n if (path.controlPoints.length > 0) {\r\n origin = path.controlPoints[0].position;\r\n break;\r\n }\r\n }\r\n\r\n return this.startOnPlane(\r\n sketch,\r\n {\r\n normal: sketch.normal,\r\n intersect: origin,\r\n },\r\n undefined,\r\n true\r\n );\r\n }\r\n\r\n startOnPlane(\r\n sketch: SketchNode,\r\n plane: { normal: KbVector | eAxis; intersect: KbVector },\r\n userOptions?: ISketchOptions,\r\n useSketchNormal = false\r\n ) {\r\n this.activeSketch = sketch;\r\n sketchPlaneIntersect = plane.intersect;\r\n const currentCanvas = this.kbViewer.canvas!;\r\n const options: Required = { ...DefaultSketchOptions, ...userOptions };\r\n this.currentOptions = options;\r\n this.sketchPoints = sketch.paths.reduce((result, path) => {\r\n for (const point of path.controlPoints) {\r\n point._visible = true;\r\n result.push(point);\r\n }\r\n return result;\r\n }, [] as SketchControlPoint[]);\r\n\r\n const previewVisibleOriginal = sketch._previewVisible;\r\n sketch._previewVisible = true;\r\n let planeNormal = getSketchNormalVector3(plane.normal);\r\n if (useSketchNormal) {\r\n const sketchNode = this.kbViewer.getRendererNode(sketch);\r\n if (sketchNode) {\r\n const matrix = sketchNode.getWorldMatrix().getRotationMatrix();\r\n planeNormal = Vector3.TransformCoordinates(planeNormal, matrix);\r\n }\r\n }\r\n const planeIntersect = plane.intersect.toVec3();\r\n const sketchPlane = Plane.FromPositionAndNormal(planeIntersect, planeNormal);\r\n\r\n const originalCamera = this.scene.activeCamera!;\r\n\r\n let sketchCamera: KbArcRotateCamera | undefined;\r\n if (options.faceCamera) {\r\n originalCamera.detachControl();\r\n // camera.storeCameraSettings();\r\n // const viewpoint = new ViewpointNode()\r\n // camera.moveCameraToViewpoint(viewpoint);\r\n sketchCamera = new KbArcRotateCamera('sketchEditorView', 0, 0, 10, planeIntersect, this.scene);\r\n sketchCamera.mode = Camera.ORTHOGRAPHIC_CAMERA;\r\n sketchCamera.lockRotation = true;\r\n sketchCamera.wheelPrecision = 60;\r\n sketchCamera.pinchPrecision = 60;\r\n switch (plane.normal) {\r\n case eAxis.x:\r\n case eAxis.z:\r\n sketchCamera.upVector = Vector3.Up();\r\n break;\r\n case eAxis.y:\r\n sketchCamera.upVector = Vector3.Backward();\r\n break;\r\n default:\r\n sketchCamera.upVector = Vector3.Cross(\r\n planeNormal,\r\n planeNormal.equalsToFloats(0, 1, 0) ? Vector3.Right() : Vector3.Up()\r\n );\r\n }\r\n sketchCamera.setPosition(planeNormal.scale(5000).add(planeIntersect));\r\n sketchCamera.orthoFit(10, 10);\r\n\r\n this.scene.activeCamera = sketchCamera;\r\n currentCanvas.focus();\r\n sketchCamera.attachControl();\r\n this.kbViewer.viewChanged.next();\r\n }\r\n\r\n const currentCamera = this.scene.activeCamera!;\r\n\r\n interface DragInfo {\r\n point: SketchControlPoint;\r\n handle: 'position' | 'direction' | 'direction2';\r\n }\r\n let dragObjects: DragInfo[] = [];\r\n let lastMousePosition: Vector3 | null = null;\r\n const sketchParent = sketch._parent ? this.kbViewer.getRendererNode(sketch._parent) : undefined;\r\n\r\n this.kbViewer.interaction.onPointerDown.pipe(takeUntil(this.end)).subscribe(e => {\r\n if (e.button === 0 && e.pickInfo && e.pickInfo.ray) {\r\n const deletedPoints: SketchControlPoint[] = [];\r\n const ray = e.pickInfo.ray;\r\n lastMousePosition = getPointOnPlane(ray);\r\n if (lastMousePosition) {\r\n roundVectorToPlaces(lastMousePosition, options.snapSize);\r\n }\r\n\r\n const v1 = new Vector3(-1, -1, -1),\r\n v2 = new Vector3(1, 1, 1),\r\n collisionSphere = new BoundingSphere(v1, v2);\r\n\r\n const r = CONNECTOR_DIAM / 2;\r\n\r\n for (let i = 0; i < this.sketchPoints.length; i++) {\r\n const point = this.sketchPoints[i];\r\n const pointRendered = this.kbViewer.getRendererNode(point);\r\n if (!pointRendered || pointRendered.isDisposed()) {\r\n this.sketchPoints.splice(i, 1);\r\n i--;\r\n continue;\r\n }\r\n\r\n const cpCenter = pointRendered.position.clone();\r\n const cpHandle1 = cpCenter.add(point.direction.toVec3());\r\n const cpHandle2 = point.direction2 ? cpCenter.add(point.direction2.toVec3()) : undefined;\r\n if (sketchParent) {\r\n const matrix = sketchParent.computeWorldMatrix();\r\n Vector3.TransformCoordinatesToRef(cpCenter, matrix, cpCenter);\r\n Vector3.TransformCoordinatesToRef(cpHandle1, matrix, cpHandle1);\r\n if (cpHandle2) {\r\n Vector3.TransformCoordinatesToRef(cpHandle2, matrix, cpHandle2);\r\n }\r\n }\r\n v1.copyFrom(cpCenter).addInPlaceFromFloats(-r, -r, -r);\r\n v2.copyFrom(cpCenter).addInPlaceFromFloats(r, r, r);\r\n collisionSphere.reConstruct(v1, v2);\r\n let hit = ray.intersectsSphere(collisionSphere);\r\n let handle: DragInfo['handle'] = 'position';\r\n\r\n if (!hit) {\r\n v1.copyFrom(cpHandle1).addInPlaceFromFloats(-r, -r, -r);\r\n v2.copyFrom(cpHandle1).addInPlaceFromFloats(r, r, r);\r\n collisionSphere.reConstruct(v1, v2);\r\n hit = ray.intersectsSphere(collisionSphere);\r\n handle = 'direction';\r\n }\r\n if (!hit && cpHandle2) {\r\n v1.copyFrom(cpHandle2).addInPlaceFromFloats(-r, -r, -r);\r\n v2.copyFrom(cpHandle2).addInPlaceFromFloats(r, r, r);\r\n collisionSphere.reConstruct(v1, v2);\r\n hit = ray.intersectsSphere(collisionSphere);\r\n handle = 'direction2';\r\n }\r\n\r\n if (hit) {\r\n dragObjects.push({ point, handle });\r\n\r\n // If ctrl key is held down, allow dragging just one point so you can separate paths\r\n if (e.ctrlKey) {\r\n break;\r\n }\r\n }\r\n }\r\n deletedPoints.forEach(point => {\r\n const index = this.sketchPoints.findIndex(ec => ec.id === point.id);\r\n this.sketchPoints.splice(index, 1);\r\n });\r\n\r\n // If both direction handles and positions are hit, only drag the positions and leave the handles alone\r\n if (dragObjects.find(d => d.handle === 'position')) {\r\n dragObjects = dragObjects.filter(d => d.handle === 'position');\r\n }\r\n\r\n if (dragObjects.length === 0) {\r\n activeHitPoint = undefined;\r\n } else {\r\n activeHitPoint = dragObjects[0].point;\r\n currentCamera.detachControl();\r\n }\r\n }\r\n });\r\n\r\n this.kbViewer.interaction.onPointerUp.pipe(takeUntil(this.end)).subscribe(e => {\r\n dragObjects = [];\r\n this.mouseUp.next();\r\n currentCamera.attachControl(currentCanvas);\r\n });\r\n\r\n this.kbViewer.interaction.onPointerMove.pipe(takeUntil(this.end)).subscribe(e => {\r\n if (dragObjects.length > 0 && e.pickInfo && e.pickInfo.ray) {\r\n const ray = e.pickInfo.ray;\r\n const intersect = getPointOnPlane(ray);\r\n if (intersect) {\r\n if (sketchParent) {\r\n const transformInv = sketchParent.computeWorldMatrix().clone().invert();\r\n Vector3.TransformCoordinatesToRef(intersect, transformInv, intersect);\r\n }\r\n roundVectorToPlaces(intersect, options.snapSize);\r\n dragObjects.forEach(dragObject => {\r\n if (dragObject.handle === 'position') {\r\n roundVectorToPlaces(intersect, options.snapSize);\r\n dragObject.point.position = KbVector.FromVec3(intersect);\r\n } else {\r\n const movement = intersect.subtract(dragObject.point.position.toVec3());\r\n roundVectorToPlaces(intersect, options.snapSize);\r\n\r\n if (dragObject.handle === 'direction') {\r\n dragObject.point.direction = KbVector.FromVec3(movement);\r\n } else if (dragObject.point.direction2) {\r\n dragObject.point.direction2 = KbVector.FromVec3(movement);\r\n }\r\n }\r\n });\r\n }\r\n }\r\n });\r\n\r\n this.kbViewer.interaction.onClick.pipe(takeUntil(this.end)).subscribe(e => {\r\n if (dragObjects.length === 0 && e.pickInfo) {\r\n if (e.button === 2 && e.canvasX && e.canvasY) {\r\n if (e.pickInfo.pickedMesh instanceof LinesMesh && e.pickInfo.pickedPoint) {\r\n this.kbViewer.rightClickMenu.showContextMenu(\r\n e.canvasX,\r\n e.canvasY,\r\n this.getRightClickButtonsForLines(sketch, e.pickInfo.faceId, e.pickInfo.pickedPoint)\r\n );\r\n } else if (e.pickInfo.pickedMesh instanceof Mesh) {\r\n const kb3dNode = getByDynamicId(e.pickInfo.pickedMesh.id);\r\n if (kb3dNode instanceof SketchControlPoint) {\r\n activeHitPoint = kb3dNode;\r\n this.kbViewer.rightClickMenu.showContextMenu(\r\n e.canvasX,\r\n e.canvasY,\r\n this.getRightClickButtons()\r\n );\r\n }\r\n }\r\n } else if (e.button === 0 && e.pickInfo.ray) {\r\n if (this.kbViewer.rightClickMenu.isOpen()) {\r\n this.kbViewer.rightClickMenu.hideContextMenu();\r\n return;\r\n }\r\n const ray = e.pickInfo.ray;\r\n const intersect = getPointOnPlane(ray);\r\n if (intersect) {\r\n const manager = sketch._manager!;\r\n roundVectorToPlaces(intersect, options.snapSize);\r\n\r\n let lastPath = sketch.paths.length > 0 ? sketch.paths[sketch.paths.length - 1] : undefined;\r\n if (!lastPath) {\r\n lastPath = manager.create(Token.SketchPath);\r\n lastPath.name = 'Path';\r\n sketch.addChildren(lastPath);\r\n }\r\n\r\n const c = createSketchPointForEditor(manager, lastPath);\r\n\r\n if (sketch._parent instanceof SpaceNode) {\r\n const rendererNode = this.kbViewer.getRendererNode(sketch);\r\n\r\n if (rendererNode) {\r\n Vector3.TransformCoordinatesToRef(\r\n intersect,\r\n rendererNode.computeWorldMatrix().clone().invert(),\r\n intersect\r\n );\r\n }\r\n c.position = KbVector.FromVec3(intersect);\r\n }\r\n\r\n this.sketchPoints.push(c);\r\n }\r\n }\r\n }\r\n });\r\n\r\n this.end.pipe(take(1)).subscribe(() => {\r\n if (!previewVisibleOriginal) {\r\n sketch.paths.forEach(path => {\r\n for (const point of path.controlPoints) {\r\n point._visible = false;\r\n }\r\n });\r\n }\r\n\r\n originalCamera.attachControl(currentCanvas);\r\n this.scene.activeCamera = originalCamera;\r\n if (sketchCamera) {\r\n sketchCamera.dispose();\r\n }\r\n sketch._previewVisible = previewVisibleOriginal;\r\n });\r\n\r\n function getPointOnPlane(ray: Ray) {\r\n const d = ray.intersectsPlane(sketchPlane);\r\n if (typeof d === 'number') {\r\n return ray.origin.add(ray.direction.scale(d));\r\n }\r\n return null;\r\n }\r\n }\r\n\r\n clear() {\r\n this.end.next();\r\n this.kbViewer.rightClickMenu.hideContextMenu();\r\n this.currentOptions = undefined;\r\n this.activeSketch = undefined;\r\n }\r\n\r\n dispose() {\r\n this.kbViewer.rightClickMenu.hideContextMenu();\r\n this.end.next();\r\n this.end.complete();\r\n this.mouseUp.complete();\r\n this.activeSketch = undefined;\r\n }\r\n\r\n private deleteSketchPoint() {\r\n if (this.kbViewer.sceneNode._manager && activeHitPoint) {\r\n const path = activeHitPoint._parent;\r\n const targetPoint = activeHitPoint;\r\n if (path instanceof SketchPath) {\r\n const index = path.controlPoints.findIndex(ec => ec.id === targetPoint.id);\r\n path.controlPoints.splice(index, 1);\r\n }\r\n }\r\n }\r\n\r\n private deleteSketchPath() {\r\n if (this.kbViewer.sceneNode._manager && activeHitPoint) {\r\n if (this.activeSketch) {\r\n const sketch = this.activeSketch;\r\n const targetPoint = activeHitPoint;\r\n sketch.paths.forEach(path => {\r\n if (path instanceof SketchPath) {\r\n const index = path.controlPoints.findIndex(\r\n ec => ec._dynamicId === targetPoint._parent?._dynamicId\r\n );\r\n sketch.paths.splice(index, 1);\r\n }\r\n });\r\n }\r\n }\r\n }\r\n\r\n private separateSketchNode(sketchpathType: eSketchPathType) {\r\n if (this.kbViewer.sceneNode._manager && activeHitPoint) {\r\n // let targetControlPoint: SketchControlPoint | undefined;\r\n const manager = this.kbViewer.sceneNode._manager;\r\n\r\n if (this.activeSketch) {\r\n const sketch = this.activeSketch;\r\n const targetPoint = activeHitPoint;\r\n for (let pathIndex = 0; pathIndex < sketch.paths.length; pathIndex++) {\r\n const path = sketch.paths[pathIndex];\r\n const cpIndex = path.controlPoints.findIndex(ec => ec._dynamicId === targetPoint._dynamicId);\r\n if (cpIndex === -1) {\r\n continue;\r\n }\r\n if (path.type === sketchpathType) {\r\n return;\r\n }\r\n\r\n if (path.controlPoints.length === 1 && sketchpathType === eSketchPathType.arcThrough3Points) {\r\n const newControlPoint2 = createSketchPointForEditor(manager, path);\r\n const newControlPoint3 = createSketchPointForEditor(manager, path);\r\n\r\n let sketchNormal = new Vector3(1, 0, 0);\r\n if (sketch.normal === eAxis.y) {\r\n sketchNormal = new Vector3(0, 1, 0);\r\n } else if (sketch.normal === eAxis.z) {\r\n sketchNormal = new Vector3(0, 0, 1);\r\n }\r\n const controlPointMesh = this.kbViewer.getRendererNode(\r\n sketch.paths[pathIndex].controlPoints[0]\r\n );\r\n const pointOnePosition = sketchNormal.cross(\r\n sketchPlaneIntersect.toVec3().subtract(controlPointMesh.getAbsolutePosition())\r\n );\r\n newControlPoint2.position = new KbVector(\r\n pointOnePosition.x,\r\n pointOnePosition.y,\r\n pointOnePosition.z\r\n );\r\n const pointTwoPosition = sketchNormal.cross(\r\n pointOnePosition.subtract(controlPointMesh.getAbsolutePosition())\r\n );\r\n newControlPoint3.position = new KbVector(\r\n pointTwoPosition.x,\r\n pointTwoPosition.y,\r\n pointTwoPosition.z\r\n );\r\n\r\n path._expanded = true;\r\n path.type = sketchpathType;\r\n this.sketchPoints.push(newControlPoint2);\r\n this.sketchPoints.push(newControlPoint3);\r\n break;\r\n } else if (cpIndex === 0 && sketch.paths[pathIndex - 1]?.type === sketchpathType) {\r\n // If an adjacent path has the same path type as the one we're making, we'll just add the control point to it\r\n // Previous path matching case\r\n const cp = path.controlPoints.shift()!;\r\n\r\n sketch.paths[pathIndex - 1].controlPoints.push(cp.clone()!);\r\n break;\r\n } else if (\r\n // Next path matching case\r\n cpIndex === path.controlPoints.length - 1 &&\r\n sketch.paths[pathIndex + 1]?.type === sketchpathType\r\n ) {\r\n const cp = path.controlPoints.pop()!;\r\n sketch.paths[pathIndex + 1].controlPoints.push(cp.clone()!);\r\n break;\r\n } else if (path.controlPoints.length > 1) {\r\n let selectedControlPoints: SketchControlPoint[];\r\n let previousPath: SketchPath | undefined;\r\n let selectedPath: SketchPath;\r\n let nextPath: SketchPath | undefined;\r\n\r\n if (cpIndex === 0) {\r\n selectedPath = path;\r\n } else {\r\n previousPath = path;\r\n selectedControlPoints = Array.from(\r\n previousPath.controlPoints.splice(cpIndex, path.controlPoints.length - cpIndex)\r\n );\r\n\r\n selectedPath = manager.create(Token.SketchPath);\r\n sketch.paths.splice(pathIndex + 1, 0, selectedPath);\r\n selectedPath.name = 'Path';\r\n selectedControlPoints.reverse();\r\n while (selectedControlPoints.length > 0) {\r\n const cp = manager.clone(selectedControlPoints.pop()!);\r\n cp._visible = true;\r\n selectedPath.controlPoints.push(cp);\r\n this.sketchPoints.push(cp);\r\n }\r\n }\r\n\r\n if (selectedPath.controlPoints.length > 2) {\r\n let nextPathPoints: SketchControlPoint[];\r\n if (\r\n sketchpathType == eSketchPathType.arcThrough3Points &&\r\n selectedPath.controlPoints.length > 3\r\n ) {\r\n nextPathPoints = Array.from(\r\n selectedPath.controlPoints.splice(3, selectedPath.controlPoints.length - 3)\r\n );\r\n } else {\r\n nextPathPoints = Array.from(\r\n selectedPath.controlPoints.splice(1, selectedPath.controlPoints.length - 1)\r\n );\r\n }\r\n\r\n nextPath = manager.create(Token.SketchPath);\r\n sketch.paths.splice(pathIndex + 2, 0, nextPath);\r\n nextPath.name = 'Path';\r\n nextPathPoints.reverse();\r\n while (nextPathPoints.length > 0) {\r\n const cp = manager.clone(nextPathPoints.pop()!);\r\n cp._visible = true;\r\n nextPath.controlPoints.push(cp);\r\n this.sketchPoints.push(cp);\r\n }\r\n }\r\n\r\n selectedPath._expanded = true;\r\n selectedPath.type = sketchpathType;\r\n\r\n if (previousPath) {\r\n const match = manager.clone(selectedPath.controlPoints[0]);\r\n match._visible = true;\r\n previousPath.controlPoints.push(match);\r\n this.sketchPoints.push(match);\r\n }\r\n if (nextPath && nextPath.controlPoints.length > 0) {\r\n const match = manager.clone(nextPath.controlPoints[0]);\r\n match._visible = true;\r\n selectedPath.controlPoints.push(match);\r\n this.sketchPoints.push(match);\r\n }\r\n\r\n break;\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n private addSketchNodeOnPath(node: SketchNode, faceId: number, pickedPoint: Vector3) {\r\n const index = node._lineSegmentStartingPoints[faceId];\r\n let counter = 0;\r\n const manager = node._manager!;\r\n\r\n for (let pathIndex = 0; pathIndex < node.paths.length; pathIndex++) {\r\n const path = node.paths[pathIndex];\r\n for (let cpIndex = 0; cpIndex < node.paths.length; cpIndex++) {\r\n const controlPoint = path.controlPoints[cpIndex];\r\n if (counter === index) {\r\n const spliceIndex = path.controlPoints.indexOf(controlPoint);\r\n const sketchControlPoint = manager.create(Token.SketchControlPoint);\r\n sketchControlPoint.position = KbVector.FromVec3(pickedPoint);\r\n path.controlPoints.splice(spliceIndex + 1, 0, sketchControlPoint);\r\n sketchControlPoint._visible = true;\r\n sketchControlPoint._expanded = true;\r\n this.sketchPoints.push(sketchControlPoint);\r\n return;\r\n }\r\n counter++;\r\n }\r\n }\r\n }\r\n\r\n private getRightClickButtons() {\r\n return [\r\n { label: 'delete node', callback: () => this.deleteSketchPoint() },\r\n { label: 'delete path', callback: () => this.deleteSketchPath() },\r\n { label: 'convert to cubic bezier', callback: () => this.separateSketchNode(eSketchPathType.cubicBezier) },\r\n {\r\n label: 'convert to quadratic bezier',\r\n callback: () => this.separateSketchNode(eSketchPathType.quadraticBezier),\r\n },\r\n { label: 'convert to hermite', callback: () => this.separateSketchNode(eSketchPathType.hermite) },\r\n { label: 'convert to catmullrom', callback: () => this.separateSketchNode(eSketchPathType.catmullRom) },\r\n { label: 'convert to points', callback: () => this.separateSketchNode(eSketchPathType.points) },\r\n {\r\n label: 'arc through 3 points',\r\n callback: () => this.separateSketchNode(eSketchPathType.arcThrough3Points),\r\n },\r\n ];\r\n }\r\n\r\n private getRightClickButtonsForLines(node: SketchNode, faceId: number, pickedPoint: Vector3) {\r\n return [\r\n {\r\n label: 'add point on line',\r\n callback: () => this.addSketchNodeOnPath(node, faceId, pickedPoint),\r\n },\r\n ];\r\n }\r\n}\r\n","import { Scene } from '@babylonjs/core/scene';\r\nimport { ViewpointNode } from '@models/classes/viewpoints/viewpoint-node';\r\nimport { dataURLtoBlob, downloadBlob } from '@view/helpers/blob-helper';\r\nimport { KbViewer } from '../kb-viewer';\r\n\r\nexport enum eSnapshotType {\r\n image = 'image',\r\n blob = 'blob',\r\n dataUrl = 'dataUrl',\r\n}\r\n\r\nexport interface ISnapshotArgs {\r\n dataType: eSnapshotType;\r\n width: number;\r\n height: number;\r\n frameCamera?: boolean;\r\n framePadding?: number;\r\n viewpoint?: string;\r\n mimeType?: string; //defaults to \"image/png\"\r\n filename?: string;\r\n}\r\n\r\nexport class SnapshotModule {\r\n constructor(private v: KbViewer, private scene: Scene) {}\r\n\r\n public take(args: ISnapshotArgs): Promise {\r\n if (args.dataType == eSnapshotType.image) {\r\n return this.download(args);\r\n } else if (args.dataType == eSnapshotType.blob) {\r\n return this.takeAsBlob(args);\r\n } else {\r\n return this.takeAsDataUrl(args);\r\n }\r\n }\r\n\r\n public takeAsDataUrl(args: ISnapshotArgs): Promise {\r\n //set the viewpoint\r\n let viewpointSet = false;\r\n if (args.viewpoint) {\r\n const viewpoint = this.v.sceneNode._manager!.getNodeByIdOrName(args.viewpoint);\r\n if (!viewpoint) {\r\n throw `Could not find viewpoint ${args.viewpoint} for snapshot`;\r\n } else {\r\n this.v.camera.storeCameraSettings();\r\n this.v.camera.moveCameraToViewpoint(viewpoint);\r\n viewpointSet = true;\r\n }\r\n }\r\n\r\n return this.v\r\n .whenAssetsLoaded()\r\n .then(() => {\r\n if (args.frameCamera) {\r\n this.v.camera.frameCamera(this.v.sceneNode, {\r\n padding: args.framePadding,\r\n });\r\n }\r\n return this.v.forceAndWaitForNextRender().then(() => {\r\n return this.v.forceAndWaitForNextRender();\r\n });\r\n })\r\n .then(() => {\r\n const engine = this.v.scene.getEngine();\r\n const mimeType = args.mimeType || 'image/png';\r\n const snapshotCanvas = document.createElement('canvas');\r\n snapshotCanvas.width = args.width;\r\n snapshotCanvas.height = args.height;\r\n const renderContext = snapshotCanvas.getContext('2d');\r\n const ratio = engine.getRenderWidth() / engine.getRenderHeight();\r\n let newWidth = args.width;\r\n let newHeight = newWidth / ratio;\r\n if (newHeight > args.height) {\r\n newHeight = args.height;\r\n newWidth = newHeight * ratio;\r\n }\r\n const offsetX = Math.max(0, args.width - newWidth) / 2;\r\n const offsetY = Math.max(0, args.height - newHeight) / 2;\r\n const renderingCanvas = engine.getRenderingCanvas();\r\n if (renderContext && renderingCanvas) {\r\n renderContext.drawImage(renderingCanvas, offsetX, offsetY, newWidth, newHeight);\r\n }\r\n const dataUrl = snapshotCanvas.toDataURL(mimeType);\r\n if (viewpointSet) {\r\n this.v.camera.restoreCameraSettings();\r\n }\r\n snapshotCanvas.width = 0;\r\n snapshotCanvas.height = 0;\r\n return dataUrl;\r\n });\r\n }\r\n\r\n public takeAsBlob(args: ISnapshotArgs): Promise {\r\n return this.takeAsDataUrl(args).then(dataUrl => dataURLtoBlob(dataUrl));\r\n }\r\n\r\n public download(args: ISnapshotArgs): Promise {\r\n return this.takeAsBlob(args).then(blob => downloadBlob(blob, args.filename));\r\n }\r\n}\r\n","export function downloadBlob(blob: Blob, filename = 'download') {\r\n const url = URL.createObjectURL(blob);\r\n const a = document.createElement('a');\r\n a.href = url;\r\n a.download = filename || 'download';\r\n\r\n const clickHandler = () => {\r\n setTimeout(() => {\r\n URL.revokeObjectURL(url);\r\n a.removeEventListener('click', clickHandler);\r\n }, 150);\r\n };\r\n a.addEventListener('click', clickHandler, false);\r\n a.click();\r\n}\r\n\r\nexport function dataURLtoBlob(dataurl: string) {\r\n const arr = dataurl.split(',');\r\n const mime = arr[0].match(/:(.*?);/)![1];\r\n const bstr = atob(arr[1]);\r\n let n = bstr.length;\r\n const u8arr = new Uint8Array(n);\r\n\r\n while (n--) {\r\n u8arr[n] = bstr.charCodeAt(n);\r\n }\r\n return new Blob([u8arr], { type: mime });\r\n}\r\n","import { Engine } from '@babylonjs/core/Engines/engine';\r\nimport { RenderTargetTexture } from '@babylonjs/core/Materials/Textures/renderTargetTexture';\r\nimport { Effect } from '@babylonjs/core/Materials/effect';\r\nimport { ShaderMaterial } from '@babylonjs/core/Materials/shaderMaterial';\r\nimport { Color4 } from '@babylonjs/core/Maths';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport { PostProcessRenderEffect } from '@babylonjs/core/PostProcesses/RenderPipeline/postProcessRenderEffect';\r\nimport { PostProcessRenderPipeline } from '@babylonjs/core/PostProcesses/RenderPipeline/postProcessRenderPipeline';\r\nimport { PostProcess } from '@babylonjs/core/PostProcesses/postProcess';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { SpaceNode } from '@models/classes/meshes/space-node';\r\nimport { KbColor } from '@models/classes/primitives/kb-color';\r\nimport { KbViewer } from '../kb-viewer';\r\nimport { CameraTools } from './camera-tools';\r\nimport maskFragmentShader from './shaders/mask-fragment-shader.fx';\r\nimport maskVertexShader from './shaders/mask-vertex-shader.fx';\r\nimport outlineFragmentShader from './shaders/outline-fragment-shader.fx';\r\n\r\nexport interface IHoverRenderer {\r\n hoverNode(node: SpaceNode): void;\r\n clearHover(): void;\r\n}\r\n\r\nEffect.ShadersStore['customMaskVertexShader'] = maskVertexShader;\r\nEffect.ShadersStore['customMaskFragmentShader'] = maskFragmentShader;\r\nEffect.ShadersStore['simpleOutlineFragmentShader'] = outlineFragmentShader;\r\n\r\nexport class Highlighter {\r\n highlightPipeline: PostProcessRenderPipeline;\r\n postProcess: PostProcess;\r\n renderTarget: RenderTargetTexture;\r\n\r\n private engine: Engine;\r\n\r\n constructor(\r\n private kbViewer: KbViewer,\r\n private scene: Scene,\r\n public defaultHighlightColor: KbColor,\r\n cameraManager: CameraTools\r\n ) {\r\n this.engine = scene.getEngine();\r\n this.highlightPipeline = new PostProcessRenderPipeline(this.engine, 'highlightPipeline');\r\n\r\n this.renderTarget = new RenderTargetTexture('Highlight render target', 1024, scene, { samples: 2 });\r\n this.renderTarget.renderList = [];\r\n this.renderTarget.clearColor = new Color4(0, 0, 0, 0);\r\n scene.customRenderTargets.push(this.renderTarget);\r\n\r\n this.postProcess = new PostProcess(\r\n 'Highlighter post process',\r\n 'simpleOutline',\r\n ['screenSize', 'threshold', 'lineWidth', 'debug'],\r\n ['maskSampler'],\r\n 1.0,\r\n null,\r\n undefined,\r\n this.engine,\r\n true\r\n );\r\n // This may help with the highlight separating from the scene meshes, if we can ge the highlight to combine with the scene\r\n // this.postProcess.autoClear = false;\r\n // this.postProcess.alphaMode = Engine.ALPHA_COMBINE;\r\n this.postProcess.samples = 4;\r\n this.postProcess.onApply = effect => {\r\n effect.setFloat2('screenSize', this.postProcess.width, this.postProcess.height);\r\n effect.setFloat('threshold', 0.3);\r\n effect.setFloat('lineWidth', 4.0);\r\n effect.setBool('debug', false);\r\n effect.setTexture('maskSampler', this.renderTarget);\r\n (window as any).enableHighlightDebug = () => {\r\n enableHighlightDebug(effect);\r\n };\r\n };\r\n const highlightEffect = new PostProcessRenderEffect(\r\n this.engine,\r\n 'highlightEffect',\r\n () => [this.postProcess],\r\n false\r\n );\r\n this.highlightPipeline.addEffect(highlightEffect);\r\n this.scene.postProcessRenderPipelineManager.addPipeline(this.highlightPipeline);\r\n cameraManager.setPipeline(this.highlightPipeline);\r\n\r\n function enableHighlightDebug(effect: Effect) {\r\n effect.setBool('debug', true);\r\n }\r\n }\r\n\r\n private colorMap = new Map<\r\n string,\r\n {\r\n material: ShaderMaterial;\r\n meshes: Set;\r\n }\r\n >();\r\n private nodeColors = new Map();\r\n private activeHover: SpaceNode | undefined;\r\n private highlighterDb = new Map();\r\n\r\n public has(node: SpaceNode) {\r\n return this.highlighterDb.has(node._dynamicId);\r\n }\r\n\r\n public add(node: SpaceNode, color: KbColor = this.defaultHighlightColor) {\r\n this.startPostProcess();\r\n\r\n const renderNode = this.kbViewer.getRendererNode(node);\r\n this.highlighterDb.set(node._dynamicId, color);\r\n this.addHighlightToRendered(renderNode, color);\r\n }\r\n\r\n public remove(node: SpaceNode) {\r\n if (this.highlighterDb.has(node._dynamicId)) {\r\n const renderNode = this.kbViewer.getRendererNode(node);\r\n if (this.activeHover?._dynamicId !== node._dynamicId) {\r\n // Don't remove the highlight from a hover. Once the hover clears the highlight will be cleared.\r\n this.removeHighlightFromRendered(renderNode);\r\n this.highlighterDb.delete(node._dynamicId);\r\n }\r\n }\r\n }\r\n\r\n public hover(node: SpaceNode) {\r\n if (node != this.activeHover) {\r\n this.startPostProcess();\r\n this.clearHover();\r\n this.activeHover = node;\r\n const renderNode = this.kbViewer.getRendererNode(node);\r\n this.addHighlightToRendered(renderNode, this.defaultHighlightColor);\r\n this.kbViewer.canvas!.style.cursor = 'pointer';\r\n this.scene.defaultCursor = 'pointer';\r\n this.kbViewer.forceRenderOnNextFrame = true;\r\n }\r\n }\r\n\r\n public clearHover() {\r\n if (this.activeHover) {\r\n const renderNode = this.kbViewer.getRendererNode(this.activeHover);\r\n this.removeHighlightFromRendered(renderNode);\r\n //if the node had a highlight previously, restore it\r\n if (this.highlighterDb.has(this.activeHover._dynamicId)) {\r\n const color = this.highlighterDb.get(this.activeHover._dynamicId)!;\r\n this.addHighlightToRendered(renderNode, color);\r\n }\r\n // const targets = renderNode instanceof AbstractMesh ? [renderNode] : renderNode.getChildMeshes();\r\n // for (const target of targets) {\r\n // if (this.highlighterDb.has(target)) {\r\n // const color = this.highlighterDb.get(target)!;\r\n // this.addHighlightToRendered(target, color);\r\n // }\r\n // }\r\n this.activeHover = undefined;\r\n this.kbViewer.canvas!.style.cursor = 'default';\r\n this.scene.defaultCursor = 'default';\r\n this.kbViewer.forceRenderOnNextFrame = true;\r\n }\r\n }\r\n\r\n public clearAll() {\r\n this.colorMap.forEach(entry => {\r\n this.renderTarget.setMaterialForRendering(Array.from(entry.meshes), undefined);\r\n });\r\n if (this.renderTarget.renderList) {\r\n while (this.renderTarget.renderList.length > 0) {\r\n this.renderTarget.renderList.pop();\r\n }\r\n }\r\n this.nodeColors.clear();\r\n this.highlighterDb.clear();\r\n this.colorMap.clear();\r\n this.activeHover = undefined;\r\n }\r\n\r\n private addHighlightToRendered(node: TransformNode, color: KbColor) {\r\n const targets = node instanceof AbstractMesh ? [node] : node.getChildMeshes();\r\n for (const renderNode of targets) {\r\n if (this.renderTarget.renderList?.contains(renderNode) === false) {\r\n if (!this.renderTarget.renderList) {\r\n this.renderTarget.renderList = [];\r\n }\r\n this.renderTarget.renderList.push(renderNode);\r\n }\r\n let colorMapKey = '';\r\n\r\n if (color) {\r\n colorMapKey = this.colorToString(color);\r\n } else {\r\n colorMapKey = this.colorToString(this.defaultHighlightColor);\r\n }\r\n\r\n this.addNodeToColorMap(colorMapKey, renderNode, color);\r\n this.nodeColors.set(renderNode, colorMapKey);\r\n }\r\n }\r\n\r\n private removeHighlightFromRendered(node: TransformNode) {\r\n const targets = node instanceof AbstractMesh ? [node] : node.getChildMeshes();\r\n for (const renderNode of targets) {\r\n const colorKey = this.nodeColors.get(renderNode);\r\n this.renderTarget.renderList?.remove(renderNode);\r\n if (colorKey) {\r\n this.removeNodeFromColorMap(colorKey, renderNode);\r\n this.nodeColors.delete(renderNode);\r\n }\r\n }\r\n }\r\n\r\n private addNodeToColorMap(colorMapKey: string, node: AbstractMesh, color?: KbColor) {\r\n let colorMapEntry = this.colorMap.get(colorMapKey);\r\n if (!colorMapEntry) {\r\n const newShader = new ShaderMaterial('customMaskMat_' + colorMapKey, this.scene, 'customMask', {\r\n attributes: ['position', 'uv'],\r\n uniforms: ['worldViewProjection', 'color', 'lineColor'],\r\n });\r\n if (color) {\r\n newShader.setColor4('lineColor', color.toColor4());\r\n } else {\r\n newShader.setColor4('lineColor', this.defaultHighlightColor.toColor4());\r\n }\r\n colorMapEntry = {\r\n material: newShader,\r\n meshes: new Set(),\r\n };\r\n }\r\n colorMapEntry.meshes.add(node);\r\n this.renderTarget.setMaterialForRendering(Array.from(colorMapEntry.meshes), colorMapEntry.material);\r\n }\r\n\r\n private removeNodeFromColorMap(colorMapKey: string, node: AbstractMesh) {\r\n const colorMapEntry = this.colorMap.get(colorMapKey);\r\n if (colorMapEntry) {\r\n colorMapEntry.meshes.delete(node);\r\n this.renderTarget.setMaterialForRendering(node, undefined);\r\n }\r\n }\r\n\r\n private colorToString(color: KbColor) {\r\n return color.cssString();\r\n }\r\n\r\n private startPostProcess() {}\r\n}\r\n","import { Scene } from '@babylonjs/core/scene';\r\nimport { KbViewer } from '@view/kb-viewer';\r\n\r\nexport class RightClickMenu {\r\n constructor(private viewer: KbViewer, private scene: Scene) {}\r\n\r\n isOpen() {\r\n return this.contextMenuOpen;\r\n }\r\n\r\n private contextmenu?: HTMLDivElement;\r\n private contextMenuOpen = false;\r\n\r\n showContextMenu(xPosition: number, yPosition: number, buttons: Array<{ label: string; callback: () => void }>) {\r\n this.createContextMenu(xPosition, yPosition, buttons);\r\n if (this.viewer.uiOverlay && this.contextmenu) {\r\n this.viewer.container.appendChild(this.contextmenu);\r\n this.contextMenuOpen = true;\r\n }\r\n }\r\n\r\n hideContextMenu() {\r\n if (this.contextmenu) {\r\n this.contextmenu.style.display = 'none';\r\n this.contextMenuOpen = false;\r\n }\r\n }\r\n\r\n private createContextMenu(\r\n xPosition: number,\r\n yPosition: number,\r\n buttons: Array<{ label: string; callback: () => void }>\r\n ) {\r\n //const buttons = [];\r\n if (this.contextmenu && this.viewer.container.contains(this.contextmenu)) {\r\n this.viewer.container?.removeChild(this.contextmenu);\r\n }\r\n const contextMenu = document.createElement('div');\r\n contextMenu.style.position = 'absolute';\r\n contextMenu.style.top = yPosition.toString() + 'px';\r\n contextMenu.style.left = (xPosition + 10).toString() + 'px';\r\n contextMenu.style.width = 'auto';\r\n contextMenu.style.height = 'auto';\r\n contextMenu.style.zIndex = '2';\r\n contextMenu.addEventListener('click', e => {\r\n if (e.button === 2) {\r\n e.preventDefault();\r\n e.stopImmediatePropagation();\r\n }\r\n });\r\n document.addEventListener('click', e => {\r\n if (e.button != 2) {\r\n this.hideContextMenu();\r\n }\r\n });\r\n document.addEventListener('wheel', e => {\r\n this.hideContextMenu();\r\n });\r\n contextMenu.addEventListener('scroll', e => {\r\n if (this.contextmenu) {\r\n this.contextmenu.style.display = 'none';\r\n }\r\n });\r\n\r\n this.contextmenu = contextMenu;\r\n\r\n buttons.forEach(button => {\r\n const newButton = document.createElement('div');\r\n newButton.innerText = button.label;\r\n newButton.addEventListener('click', e => {\r\n button.callback();\r\n this.hideContextMenu();\r\n });\r\n this.contextmenu?.appendChild(newButton);\r\n newButton.style.zIndex = '100000';\r\n newButton.style.color = 'white';\r\n newButton.style.background = 'rgba(36,36,36,255)';\r\n newButton.style.cursor = 'pointer';\r\n newButton.style.padding = '10px';\r\n newButton.addEventListener('mouseover', e => {\r\n newButton.style.background = 'rgba(61,61,61,255)';\r\n });\r\n newButton.addEventListener('mouseout', e => {\r\n newButton.style.background = 'rgba(36,36,36,255)';\r\n });\r\n });\r\n }\r\n}\r\n","import { BoundingInfo } from '@babylonjs/core/Culling/boundingInfo';\r\nimport { Engine } from '@babylonjs/core/Engines/engine';\r\nimport '@babylonjs/core/Helpers/sceneHelpers';\r\nimport '@babylonjs/core/Layers/effectLayerSceneComponent'; // Required by highlight layer\r\nimport { Light, PointLight, SpotLight } from '@babylonjs/core/Lights';\r\nimport { Color3, Color4, Quaternion, Vector3 } from '@babylonjs/core/Maths';\r\nimport { AbstractMesh } from '@babylonjs/core/Meshes/abstractMesh';\r\nimport { VertexBuffer } from '@babylonjs/core/Meshes/buffer';\r\nimport { Mesh } from '@babylonjs/core/Meshes/mesh';\r\nimport { TransformNode } from '@babylonjs/core/Meshes/transformNode';\r\nimport {\r\n HardwareScalingOptimization,\r\n LensFlaresOptimization,\r\n ParticlesOptimization,\r\n PostProcessesOptimization,\r\n RenderTargetsOptimization,\r\n SceneOptimizer,\r\n SceneOptimizerOptions,\r\n ShadowsOptimization,\r\n TextureOptimization,\r\n} from '@babylonjs/core/Misc/sceneOptimizer';\r\nimport { SceneSerializer } from '@babylonjs/core/Misc/sceneSerializer';\r\nimport '@babylonjs/core/Rendering/depthRendererSceneComponent'; // For SSAO Pipeline\r\nimport '@babylonjs/core/Rendering/edgesRenderer'; // Required by mesh renderOverlay\r\nimport '@babylonjs/core/Rendering/geometryBufferRendererSceneComponent'; // For SSRPrepass and LensPipeline\r\nimport '@babylonjs/core/Rendering/outlineRenderer'; // Required by mesh renderOverlay\r\nimport '@babylonjs/core/Rendering/prePassRendererSceneComponent'; // For SSRPrepass and LensPipeline\r\nimport { UtilityLayerRenderer } from '@babylonjs/core/Rendering/utilityLayerRenderer';\r\nimport { Scene } from '@babylonjs/core/scene';\r\nimport { DeepImmutableObject } from '@babylonjs/core/types';\r\n// import \"@babylonjs/loaders/glTF\";\r\n// import '@babylonjs/loaders/glTF/2.0';\r\n// import \"@babylonjs/loaders/OBJ\";\r\nimport { PBRMaterial } from '@babylonjs/core/Materials/PBR/pbrMaterial';\r\nimport { BaseTexture } from '@babylonjs/core/Materials/Textures/baseTexture';\r\nimport { Texture } from '@babylonjs/core/Materials/Textures/texture';\r\nimport { Material } from '@babylonjs/core/Materials/material';\r\nimport { MultiMaterial } from '@babylonjs/core/Materials/multiMaterial';\r\nimport { VertexData } from '@babylonjs/core/Meshes/mesh.vertexData';\r\nimport {\r\n BaseConnector,\r\n IVertexGroup,\r\n KbQuaternion,\r\n MateNode,\r\n MeshGroupNode,\r\n NormalSmoothingFeature,\r\n SketchNode,\r\n} from '@models/classes';\r\nimport { nextTick } from '@models/classes/decorators/change-tracker';\r\nimport { eLoadingExperience, eRenderMode, eViewpointTargetMode, eViewpointType } from '@models/classes/enums';\r\nimport { isConnector } from '@models/classes/features/connector';\r\nimport { Kb3dManager } from '@models/classes/kb3d-manager';\r\nimport { Kb3dObject } from '@models/classes/kb3d-object';\r\nimport { LightNode, isLightNode } from '@models/classes/lights/light-node';\r\nimport { MaterialNode, TextureLayer, isMaterialNode, isTextureLayer } from '@models/classes/materials';\r\nimport { MeshNode } from '@models/classes/meshes/mesh-node';\r\nimport { isSketchControlPoint, isSpaceNode } from '@models/classes/meshes/mesh-utilities';\r\nimport { SpaceNode } from '@models/classes/meshes/space-node';\r\nimport { KbColor } from '@models/classes/primitives/kb-color';\r\nimport { KbGeometry } from '@models/classes/primitives/kb-geometry';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { SceneNode, isSceneNode } from '@models/classes/scene-node';\r\nimport { Token } from '@models/classes/token';\r\nimport { Validator } from '@models/classes/validator';\r\nimport { ViewpointNode } from '@models/classes/viewpoints';\r\nimport { KBMAXPREFIX } from '@models/constants';\r\nimport { SettingsManagerService } from '@models/services/settings-manager-service';\r\nimport { SyncService } from '@models/services/sync-service';\r\nimport { Observable, Subject } from 'rxjs';\r\nimport { take, throttleTime } from 'rxjs/operators';\r\nimport { GuiHelper } from './helpers/gui-helper';\r\nimport './helpers/imagedata-polyfill';\r\nimport { IScreenBoundingBox, getScreenBoundingBox, nodeToScreen } from './helpers/scene-helper';\r\nimport { Kb3dWorker } from './kb-worker';\r\nimport { SketchEditor, SnapshotModule } from './modules';\r\nimport { AnimationModule } from './modules/animation';\r\nimport { AR } from './modules/ar';\r\nimport { BoundingBoxCache } from './modules/bounding-box-cache';\r\nimport { CameraTools } from './modules/camera-tools';\r\nimport { GLTFKb3dExporter } from './modules/exporter-gltf';\r\nimport { ObjKb3dExporter } from './modules/exporter-obj';\r\nimport { GizmoTools } from './modules/gizmo-tools';\r\nimport { Highlighter } from './modules/highlighter';\r\nimport { Interaction } from './modules/interaction';\r\nimport { RightClickMenu } from './modules/right-click-menu';\r\nimport { SvgConverter } from './modules/sketching/svg-converter';\r\nimport { RendererManager } from './render/renderer-manager';\r\n\r\nexport { IScreenBoundingBox };\r\n\r\ninterface IKbViewerOptions {\r\n /** HTML Element to insert the viewer canvas into. Can be any block-level element. */\r\n container: HTMLElement;\r\n debugMode?: boolean;\r\n highlightColor?: KbColor;\r\n editorMode?: boolean;\r\n enableScreenshots?: boolean;\r\n forceDisablePostProcess?: boolean;\r\n forceDisableGraphicsOptimization?: boolean;\r\n /** optionally provide a path to the kb3d webworker script. Defaults to worker3d.js */\r\n webworkerPath?: string;\r\n /** optionally provide a validator to validate nodes. Only used by creator */\r\n validator?: Validator;\r\n}\r\ninterface ILoadSceneOptions {\r\n sceneNode: SceneNode;\r\n /** overrides the default viewpoint of the scene */\r\n viewpoint?: ViewpointNode;\r\n /** overrides the frame option for the default viewpoint */\r\n frameViewpoint?: boolean;\r\n /** overrides the loading experience of the scene */\r\n loadingExperience?: eLoadingExperience;\r\n /** If provided, will defer the end of the loading experience until the promise resolves */\r\n deferLoadingPromise?: Promise;\r\n}\r\n\r\nexport interface IMeshSelectedEvent {\r\n node: Kb3dObject;\r\n multiSelect: boolean;\r\n}\r\n\r\nexport interface IAfterRenderArgs {\r\n /** Whether the render was skipped because nothing had changed from the last render */\r\n skipped: boolean;\r\n}\r\n\r\nconst FADE_TIME = 300;\r\n/**\r\n * Applcation class for the viewer, to contain all of the initializers and hooks and connect the modules and 3d engine.\r\n * @param canvasElement An HTML DOM element that the viewer will be attached to\r\n */\r\nexport class KbViewer {\r\n options: IKbViewerOptions;\r\n\r\n constructor(options: IKbViewerOptions) {\r\n this.options = {\r\n highlightColor: KbColor.KbmaxOrange(),\r\n editorMode: false,\r\n forceDisablePostProcess: false,\r\n enableScreenshots: false,\r\n debugMode: false,\r\n webworkerPath: '/worker3d.js',\r\n ...options,\r\n };\r\n\r\n if (typeof window !== 'undefined' && options.webworkerPath && !options.webworkerPath.startsWith('http')) {\r\n this.options.webworkerPath = `${window.location.protocol}//${window.location.host}${this.options.webworkerPath}`;\r\n }\r\n\r\n Kb3dWorker.workerPath = this.options.webworkerPath!;\r\n\r\n // Create canvas and engine.\r\n this.container = options.container;\r\n this.canvas = this.createCanvas();\r\n // Anti-aliasing=true (second parameter) causes the canvas to not get cleared on iOS 15.4. Warrants testing in future iOS updates.\r\n const iOS = /(iPad|iPhone|iPod)/g.test(navigator.userAgent);\r\n this.engine = new Engine(\r\n this.canvas,\r\n !iOS,\r\n { stencil: true, preserveDrawingBuffer: this.options.enableScreenshots },\r\n true\r\n ); //set adaptToDeviceRatio to true, and let the hardware scaling optimizer take over from there\r\n this.engine.setAlphaMode(Engine.ALPHA_COMBINE);\r\n this.glInfo = this.engine.getGlInfo();\r\n // eslint-disable-next-line no-console\r\n console.log('WebGL Renderer', this.glInfo.vendor, this.glInfo.renderer, this.glInfo.version);\r\n\r\n // Create the scene.\r\n const scene = this.createScene();\r\n this.scene = scene;\r\n this.scene.hoverCursor = 'pointer';\r\n this.utilityLayer = new UtilityLayerRenderer(this.scene);\r\n this.utilityLayer.utilityLayerScene.useRightHandedSystem = true;\r\n this.utilityLayer.processAllEvents = true;\r\n\r\n this.camera = new CameraTools(this.scene, this.canvas, this);\r\n this.bbCache = new BoundingBoxCache(this);\r\n\r\n // var hdrTexture = CubeTexture.CreateFromPrefilteredData('/assets/environments/environment.dds', scene);\r\n // hdrTexture.gammaSpace = false;\r\n // scene.environmentTexture = hdrTexture;\r\n this.highlightColor = options.highlightColor || KbColor.KbmaxOrange();\r\n\r\n this.gizmos = new GizmoTools(scene, this);\r\n this.rightClickMenu = new RightClickMenu(this, this.scene);\r\n this.animation = new AnimationModule(this, this.scene);\r\n this.highlighter = new Highlighter(this, this.scene, this.highlightColor, this.camera);\r\n this.interaction = new Interaction(this, this.scene, this.highlighter);\r\n this.sketchEditor = new SketchEditor(this, this.scene, this.highlightColor);\r\n this.gui = new GuiHelper(scene, this.highlighter);\r\n this.ar = new AR(this, scene);\r\n this.snapshot = new SnapshotModule(this, scene);\r\n this.rendererManager = new RendererManager(this, scene);\r\n\r\n this.interaction.attachScene(this.utilityLayer.utilityLayerScene, true);\r\n this.gltfExporter = new GLTFKb3dExporter(scene);\r\n this.objExporter = new ObjKb3dExporter(scene, this.rendererManager);\r\n }\r\n\r\n /** Public Properties */\r\n public scene: Scene;\r\n public forceRenderOnNextFrame = false;\r\n public pauseRender = false;\r\n public canvas?: HTMLCanvasElement;\r\n public uiOverlay?: HTMLDivElement;\r\n public loadingOverlay?: HTMLDivElement;\r\n //public contextmenu?: HTMLDivElement;\r\n public container: HTMLElement;\r\n public sceneNode: SceneNode;\r\n public renderMode: eRenderMode.shaded;\r\n public warningMessages: string[];\r\n public lightsWarningMessages: string[];\r\n public textureWarningMessages: string[];\r\n public MeshWarningTracker = new Map<\r\n string,\r\n {\r\n name: string;\r\n isEnabled: boolean;\r\n castShadows: boolean;\r\n receiveShadows: boolean;\r\n polygonNumber: number;\r\n featureNumber: number;\r\n }\r\n >();\r\n public meshWarningMessages: string[];\r\n public materialWarningMessages: string[];\r\n public normalSmoothingWarnings = new Map();\r\n public totalPolygons: number;\r\n\r\n /** Events / Observables */\r\n /** observable for tracking when something changed that could affect the position of a node on the screen */\r\n public viewChanged = new Subject();\r\n public afterRender = new Subject();\r\n public fpsTracker = new Subject();\r\n public polygonTracker = new Subject<{ polygons: number; vertices: number }>();\r\n public warningTracker = new Subject<{ warningMessages: string[] }>();\r\n public meshWarningsNeedsUpdate = new Subject();\r\n public materialWarningsNeedsUpdate = new Subject();\r\n public animationWarningsNeedsUpdate = new Subject();\r\n\r\n /** Modules */\r\n public rendererManager: RendererManager;\r\n public readonly interaction: Interaction;\r\n public gizmos: GizmoTools;\r\n public rightClickMenu: RightClickMenu;\r\n public camera: CameraTools;\r\n public gui: GuiHelper;\r\n public sketchEditor: SketchEditor;\r\n public svgConverter: SvgConverter;\r\n public bbCache: BoundingBoxCache;\r\n public readonly animation: AnimationModule;\r\n public readonly highlighter: Highlighter;\r\n public readonly ar: AR;\r\n public readonly snapshot: SnapshotModule;\r\n public readonly utilityLayer: UtilityLayerRenderer;\r\n public readonly glInfo: {\r\n vendor: string;\r\n renderer: string;\r\n version: string;\r\n };\r\n public highlightColor: KbColor;\r\n\r\n private disposed = false;\r\n private forceRunRenderer = false;\r\n private engine: Engine;\r\n private manager: Kb3dManager;\r\n private gltfExporter: GLTFKb3dExporter;\r\n private objExporter: ObjKb3dExporter;\r\n private isDevEnvironment = false;\r\n /** tracks babylon nodes who are being tracked for positions (hotspots) */\r\n private positionTrackedNodes = new Map();\r\n\r\n private nextFrameCallbacks: Array<() => void> = [];\r\n private resizeObserver?: ResizeObserver;\r\n private _renderLoopRunning = false;\r\n private allAssetsLoaded = false;\r\n\r\n /** tracks any additional promises that renders and load spinners should wait for */\r\n private additionalLoadBlockers: Array> = [];\r\n\r\n get settings() {\r\n return SettingsManagerService;\r\n }\r\n\r\n setRenderMode(mode: eRenderMode, force?: boolean) {\r\n this.rendererManager.setRenderMode(mode, this.sceneNode, force);\r\n }\r\n\r\n loadScene(o: ILoadSceneOptions) {\r\n if (!o.sceneNode._manager) {\r\n throw 'sceneNode must be constructed with a manager';\r\n }\r\n\r\n this.sceneNode = o.sceneNode;\r\n this.manager = o.sceneNode._manager as Kb3dManager;\r\n this.isDevEnvironment = this.manager.isDevEnvironment;\r\n\r\n const loadingExperience = o.loadingExperience || this.sceneNode.loadingExperience;\r\n this.createLoadingExperience(loadingExperience, this.sceneNode.loadingImagePath);\r\n\r\n let viewpoint =\r\n o.viewpoint ||\r\n (o.sceneNode.viewpointId ? this.manager.getById(o.sceneNode.viewpointId) : undefined);\r\n if (!viewpoint) {\r\n viewpoint = this.manager.create(Token.ViewpointNode);\r\n viewpoint.type = eViewpointType.orbit;\r\n viewpoint.targetMode = eViewpointTargetMode.point;\r\n viewpoint.position = new KbVector(3, 3, 3);\r\n viewpoint.targetPoint = KbVector.Zero();\r\n }\r\n this.camera.moveCameraToViewpoint(viewpoint);\r\n\r\n // Start render loop.\r\n this.startRender();\r\n\r\n // The canvas/window resize event handler.\r\n // window.addEventListener('resize', () => {\r\n // if (!this.ar.xrLoading && !this.ar.xrActive) { //don't mess with the AR canvas resize\r\n // this.engine.resize();\r\n // this.viewChanged.next();\r\n // }\r\n // });\r\n\r\n const promises = [this.whenAssetsLoaded()];\r\n if (o.deferLoadingPromise) promises.push(o.deferLoadingPromise);\r\n\r\n return Promise.all(promises).then(() => {\r\n const http = this.sceneNode._manager?.getService(Token.HttpService);\r\n\r\n let frame = true;\r\n if (this.sceneNode.viewpointId) frame = this.sceneNode.frameViewpoint;\r\n if (o.frameViewpoint != null) frame = o.frameViewpoint;\r\n\r\n if (frame) {\r\n this.camera.frameCamera(this.sceneNode);\r\n }\r\n this.hideLoadingExperience();\r\n\r\n // Wait for everything to be loaded and processed before trying to optimize scene because the scene optimizer watches the FPS and we want the state to be stable\r\n // Disable the graphics optimization (for example when rendering)\r\n if (!this.options.forceDisableGraphicsOptimization) {\r\n this.waitForNextKb3dRender().then(() => {\r\n this.optimizeScene(this.scene);\r\n });\r\n }\r\n\r\n this.meshWarningsNeedsUpdate\r\n .pipe(throttleTime(1000, undefined, { leading: true, trailing: true }))\r\n .subscribe(() => {\r\n this.updateMeshWarnings();\r\n this.updateTextureWarnings();\r\n });\r\n\r\n this.materialWarningsNeedsUpdate\r\n .pipe(throttleTime(15000, undefined, { leading: true, trailing: false }))\r\n .subscribe(() => {\r\n this.updateMaterialWarnings();\r\n });\r\n\r\n this.animationWarningsNeedsUpdate.subscribe(value => {\r\n this.updateAnimationWarnings(value);\r\n });\r\n this.updateTextureWarnings();\r\n this.updateLightWarnings();\r\n this.updateMaterialWarnings();\r\n if (http) {\r\n http.executeDeferredCalls();\r\n }\r\n });\r\n }\r\n\r\n getNode(id: string) {\r\n return this.manager.getById(id);\r\n }\r\n\r\n getRendererNode(n: LightNode): Light;\r\n getRendererNode(n: SpaceNode): TransformNode | AbstractMesh;\r\n getRendererNode(n: BaseConnector): TransformNode;\r\n getRendererNode(n: MaterialNode): PBRMaterial;\r\n getRendererNode(n: TextureLayer): Texture;\r\n getRendererNode(n: Kb3dObject | TextureLayer) {\r\n if (isSceneNode(n) && n.isNested() === false) {\r\n return this.rendererManager.sceneRenderer.get3d(n);\r\n } else if (isSceneNode(n) && n.isNested()) {\r\n return this.rendererManager.mesh.get3d(n);\r\n } else if (isLightNode(n)) {\r\n return this.rendererManager.light.get3d(n);\r\n } else if (isSpaceNode(n)) {\r\n return this.rendererManager.mesh.get3d(n);\r\n } else if (isMaterialNode(n)) {\r\n return this.rendererManager.material.get3d(n);\r\n } else if (isConnector(n)) {\r\n return this.rendererManager.connector.get3d(n);\r\n } else if (isSketchControlPoint(n)) {\r\n return this.rendererManager.controlPointRenderer.get3d(n);\r\n } else if (isTextureLayer(n)) {\r\n return this.rendererManager.textures.get3d(n);\r\n }\r\n return undefined;\r\n }\r\n\r\n getFPS() {\r\n return this.engine.getFps().toFixed();\r\n }\r\n\r\n getWarnings() {\r\n return this.warningMessages;\r\n }\r\n\r\n getMeshBoundingBox(node: MeshNode, vertexGroups?: IVertexGroup[]) {\r\n let mesh = this.rendererManager.mesh.get3d(node);\r\n let newMesh: Mesh | undefined;\r\n if (vertexGroups) {\r\n const newIndices: number[] = [];\r\n const newPositions: number[] = [];\r\n let indices: number[];\r\n let positions: number[];\r\n if (mesh instanceof Mesh && mesh.geometry) {\r\n positions = mesh.geometry.getVerticesData(VertexBuffer.PositionKind) as number[];\r\n indices = mesh.geometry.getIndices() as number[];\r\n } else {\r\n indices = [];\r\n positions = [];\r\n }\r\n for (let i = 0; i < vertexGroups.length; i++) {\r\n const vertexGroup = vertexGroups[i];\r\n if (vertexGroup && vertexGroup.startIndex && vertexGroup.endIndex) {\r\n for (let j = vertexGroup.startIndex; j < vertexGroup.endIndex; j++) {\r\n newIndices.push(indices[j]);\r\n for (let k = 0; k < 3; k++) {\r\n newPositions[indices[j] * 3 + k] = positions[indices[j] * 3 + k];\r\n }\r\n }\r\n }\r\n }\r\n newMesh = new Mesh('temp mesh', this.scene, mesh?.parent);\r\n const newVertexData = new VertexData();\r\n newVertexData.indices = newIndices;\r\n newVertexData.positions = newPositions;\r\n newVertexData.applyToMesh(newMesh, true);\r\n newMesh.visibility = 0;\r\n }\r\n\r\n if (newMesh) {\r\n mesh = newMesh;\r\n }\r\n if (mesh && mesh instanceof AbstractMesh) {\r\n const boundingBox = mesh.getBoundingInfo().boundingBox;\r\n newMesh?.dispose();\r\n return {\r\n maximum: boundingBox.maximum,\r\n minimum: boundingBox.minimum,\r\n center: boundingBox.center,\r\n minWorld: boundingBox.minimumWorld,\r\n maxWorld: boundingBox.maximumWorld,\r\n };\r\n }\r\n }\r\n\r\n updateLightWarnings() {\r\n let totalSpotLights = 0;\r\n let totalPointLights = 0;\r\n let shadowCastingLights = 0;\r\n let tempWarning = '';\r\n this.lightsWarningMessages = [];\r\n\r\n this.scene.lights.forEach(light => {\r\n if (light.isEnabled()) {\r\n if (light instanceof SpotLight) {\r\n totalSpotLights += 1;\r\n } else if (light instanceof PointLight) {\r\n totalPointLights += 1;\r\n }\r\n if (light.shadowEnabled) {\r\n shadowCastingLights += 1;\r\n }\r\n }\r\n });\r\n if (totalSpotLights + totalPointLights > 10) {\r\n tempWarning = `Performance Warning: The number of spotlights (${totalSpotLights}) and/or pointlights (${totalPointLights}) is higher than recommended for optimal performance.`;\r\n this.lightsWarningMessages.push(tempWarning);\r\n if (this.isDevEnvironment) {\r\n console.warn(tempWarning);\r\n }\r\n }\r\n if (shadowCastingLights > 4) {\r\n tempWarning = `Performance Warning: The number of shadow casting lights (${shadowCastingLights}) is higher than recommended for optimal performance.`;\r\n this.lightsWarningMessages.push(tempWarning);\r\n if (this.isDevEnvironment) {\r\n console.warn(tempWarning);\r\n }\r\n }\r\n this.updateWarnings();\r\n }\r\n\r\n updateTextureWarnings() {\r\n const bigTextures2048 = new Map();\r\n const bigTextures4096 = new Map();\r\n let tempWarning = '';\r\n this.textureWarningMessages = [];\r\n\r\n this.sceneNode.meshes.forEach(node => {\r\n const mesh = this.getRendererNode(node);\r\n let textures: Texture[] | BaseTexture[] = [];\r\n if (mesh instanceof AbstractMesh && mesh.isEnabled()) {\r\n if (mesh.material instanceof MultiMaterial) {\r\n mesh.material.subMaterials.forEach(subMaterial => {\r\n const subMaterialTextures = subMaterial?.getActiveTextures();\r\n if (subMaterialTextures) {\r\n subMaterialTextures.forEach(texture => {\r\n if (texture instanceof Texture) {\r\n textures.push(texture);\r\n }\r\n });\r\n }\r\n });\r\n } else if (mesh.material instanceof Material) {\r\n textures = mesh.material.getActiveTextures();\r\n }\r\n }\r\n\r\n if (textures) {\r\n textures.forEach(texture => {\r\n const textureSize = Math.max(texture.getSize().height, texture.getSize().width);\r\n if (texture instanceof Texture) {\r\n if (textureSize > 4096) {\r\n bigTextures4096.set(texture.uniqueId, texture);\r\n bigTextures2048.set(texture.uniqueId, texture);\r\n } else if (textureSize > 2048) {\r\n bigTextures2048.set(texture.uniqueId, texture);\r\n }\r\n }\r\n });\r\n }\r\n });\r\n\r\n if (bigTextures4096.size > 4) {\r\n tempWarning = `Warning, number of textures with resolutions over 4096 (${bigTextures4096.size}) is higher than recommended for optimal performance.`;\r\n this.textureWarningMessages.push(tempWarning);\r\n if (this.isDevEnvironment) {\r\n console.warn(tempWarning);\r\n }\r\n } else if (bigTextures2048.size > 20) {\r\n tempWarning = `Warning, number of textures with resolutions over 2048 (${bigTextures2048.size}) is higher than recommended for optimal performance.`;\r\n this.textureWarningMessages.push(tempWarning);\r\n if (this.isDevEnvironment) {\r\n console.warn(tempWarning);\r\n }\r\n }\r\n this.updateWarnings();\r\n }\r\n\r\n updateMeshWarnings() {\r\n this.totalPolygons = this.recursiveGetPolygons(this.sceneNode).polygons;\r\n let meshCastAndReceiveShadows = 0;\r\n let tempWarning = '';\r\n this.meshWarningMessages = [];\r\n\r\n this.MeshWarningTracker.forEach(mesh => {\r\n if (mesh.isEnabled) {\r\n if (mesh.featureNumber > 15) {\r\n tempWarning = `Performance Warning: The number of features on ${mesh.name} is higher than recommended for optimal performance.`;\r\n this.meshWarningMessages.push(tempWarning);\r\n if (this.isDevEnvironment) {\r\n console.warn(tempWarning);\r\n }\r\n }\r\n\r\n if (mesh.castShadows && mesh.receiveShadows) {\r\n meshCastAndReceiveShadows += 1;\r\n }\r\n }\r\n });\r\n\r\n if (this.totalPolygons > 500000) {\r\n tempWarning = `Performance Warning: Polygon count of ${this.totalPolygons} is higher than recommended for optimal performance.`;\r\n this.meshWarningMessages.push(tempWarning);\r\n if (this.isDevEnvironment) {\r\n console.warn(tempWarning);\r\n }\r\n }\r\n\r\n if (meshCastAndReceiveShadows > 10) {\r\n tempWarning = `Performance Warning: The number of meshes that cast and receive shadows (${meshCastAndReceiveShadows}) is higher than recommended for optimal performance.`;\r\n this.meshWarningMessages.push(tempWarning);\r\n if (this.isDevEnvironment) {\r\n console.warn(tempWarning);\r\n }\r\n }\r\n\r\n this.updateWarnings();\r\n }\r\n\r\n updateNormalSmoothingWarnings(feature: NormalSmoothingFeature, polygonCount: number, removed?: boolean) {\r\n if (removed) {\r\n this.normalSmoothingWarnings.delete(feature.id);\r\n } else {\r\n if (polygonCount > 100000 && feature.enabled) {\r\n const warning = `Performance Warning: Too many polygons (${polygonCount}) are selected inside \"${feature.name}\" on \"${feature._parent?.name}\".`;\r\n this.normalSmoothingWarnings.set(feature.id, warning);\r\n if (this.isDevEnvironment) {\r\n console.warn(warning);\r\n }\r\n } else {\r\n this.normalSmoothingWarnings.delete(feature.id);\r\n }\r\n }\r\n this.updateWarnings;\r\n }\r\n\r\n updateMaterialWarnings() {\r\n let tempWarning = '';\r\n const localMaterialCount = this.getLocalMaterialsRecursive(this.sceneNode);\r\n this.materialWarningMessages = [];\r\n if (localMaterialCount > 20) {\r\n tempWarning = `Performance Warning: The number of local materials (${localMaterialCount}) is higher than recommended for optimal performance.`;\r\n this.materialWarningMessages.push(tempWarning);\r\n if (this.isDevEnvironment) {\r\n console.warn(tempWarning);\r\n }\r\n }\r\n }\r\n\r\n updateAnimationWarnings(activeAnimations: number) {\r\n let tempWarning = '';\r\n if (activeAnimations > 1) {\r\n tempWarning = `Performance Warning: More than 1 animation (${activeAnimations}) is active at the same instance.`;\r\n if (this.isDevEnvironment) {\r\n console.warn(tempWarning);\r\n }\r\n }\r\n }\r\n\r\n updateWarnings() {\r\n const tempArray: string[] = [];\r\n if (this.lightsWarningMessages && this.lightsWarningMessages.length > 0) {\r\n this.lightsWarningMessages.forEach(warning => {\r\n tempArray.push(warning);\r\n });\r\n }\r\n if (this.textureWarningMessages && this.textureWarningMessages.length > 0) {\r\n this.textureWarningMessages.forEach(warning => {\r\n tempArray.push(warning);\r\n });\r\n }\r\n if (this.meshWarningMessages && this.meshWarningMessages.length > 0) {\r\n this.meshWarningMessages.forEach(warning => {\r\n tempArray.push(warning);\r\n });\r\n }\r\n if (this.materialWarningMessages && this.materialWarningMessages.length > 0) {\r\n this.materialWarningMessages.forEach(warning => {\r\n tempArray.push(warning);\r\n });\r\n }\r\n if (this.normalSmoothingWarnings && this.normalSmoothingWarnings.size > 0) {\r\n this.normalSmoothingWarnings.forEach(warning => {\r\n tempArray.push(warning);\r\n });\r\n }\r\n if (tempArray.length === 0) {\r\n tempArray.push('No warnings to display.');\r\n }\r\n this.warningMessages = tempArray;\r\n this.warningTracker.next();\r\n }\r\n\r\n extractMeshGeometry(node: MeshNode) {\r\n const mesh = this.rendererManager.mesh.get3d(node);\r\n if (mesh && mesh instanceof Mesh && mesh.geometry) {\r\n const newGeometry = new KbGeometry();\r\n newGeometry.indices = new Uint32Array(mesh.geometry.getIndices() || []);\r\n newGeometry.positions = new Float32Array(mesh.geometry.getVerticesData(VertexBuffer.PositionKind) || []);\r\n if (newGeometry.indices.length === 0 || newGeometry.positions.length === 0) {\r\n return null;\r\n }\r\n const normals = mesh.geometry.getVerticesData(VertexBuffer.NormalKind);\r\n if (normals) {\r\n newGeometry.normals = new Float32Array(normals);\r\n }\r\n const tangents = mesh.geometry.getVerticesData(VertexBuffer.TangentKind);\r\n if (tangents) {\r\n newGeometry.tangents = new Float32Array(tangents);\r\n }\r\n const uvs = mesh.geometry.getVerticesData(VertexBuffer.UVKind);\r\n if (uvs) {\r\n newGeometry.uvs = new Float32Array(uvs);\r\n }\r\n\r\n return newGeometry;\r\n }\r\n }\r\n\r\n applySvgToSketch(node: SketchNode, data: Blob) {\r\n return SvgConverter.overwriteSketchWithSvg(node, data);\r\n }\r\n\r\n draggingEvent(mousePosition: Observable<{ x: number; y: number }>) {\r\n let lastPickedMesh: SpaceNode | undefined;\r\n const dragOver = new Subject();\r\n mousePosition.subscribe(\r\n position => {\r\n const pickInfo = this.interaction.pick(position.x, position.y);\r\n const mesh = pickInfo?.targetNode;\r\n if (lastPickedMesh !== mesh) {\r\n if (lastPickedMesh) {\r\n this.highlighter.remove(lastPickedMesh);\r\n }\r\n if (mesh) {\r\n this.highlighter.add(mesh, KbColor.Green());\r\n }\r\n dragOver.next(mesh);\r\n lastPickedMesh = mesh;\r\n }\r\n },\r\n () => {},\r\n () => {\r\n if (lastPickedMesh) {\r\n this.highlighter.remove(lastPickedMesh);\r\n dragOver.complete();\r\n }\r\n }\r\n );\r\n return dragOver.asObservable();\r\n }\r\n\r\n dispose() {\r\n this.disposed = true;\r\n this.positionTrackedNodes.clear();\r\n this.rendererManager.disposeAll();\r\n this.scene.dispose();\r\n this.engine.dispose();\r\n this.camera!.dispose();\r\n this.manager.dispose();\r\n this.bbCache.dispose();\r\n this.sketchEditor.dispose();\r\n if (this.canvas) {\r\n this.container.removeChild(this.canvas);\r\n delete this.canvas;\r\n }\r\n if (this.uiOverlay) {\r\n this.container.removeChild(this.uiOverlay);\r\n delete this.uiOverlay;\r\n }\r\n if (this.resizeObserver) {\r\n this.resizeObserver.disconnect();\r\n delete this.resizeObserver;\r\n }\r\n this.fpsTracker.complete();\r\n this.polygonTracker.complete();\r\n this.warningTracker.complete();\r\n this.meshWarningsNeedsUpdate.complete();\r\n this.materialWarningsNeedsUpdate.complete();\r\n this.animationWarningsNeedsUpdate.complete();\r\n }\r\n\r\n getScreenPositionOfNode(node: SpaceNode) {\r\n const bNode = this.rendererManager.mesh.get3d(node);\r\n\r\n if (bNode) {\r\n return nodeToScreen(bNode, this.scene.activeCamera!);\r\n }\r\n }\r\n\r\n getScreenBoundingBoxOfNode(node: SpaceNode) {\r\n const bNode = this.rendererManager.mesh.get3d(node);\r\n if (bNode) {\r\n return getScreenBoundingBox(bNode, this.scene.activeCamera!, true);\r\n }\r\n }\r\n\r\n getBabylonJson() {\r\n return JSON.stringify(SceneSerializer.Serialize(this.scene), null, 2);\r\n }\r\n\r\n exportGlb() {\r\n return this.gltfExporter.exportGlb(this.sceneNode?.name);\r\n }\r\n\r\n exportObj(meshNode?: MeshNode) {\r\n const fileName = meshNode ? meshNode.name : this.sceneNode?.name;\r\n return this.objExporter.exportObj(fileName, meshNode);\r\n }\r\n\r\n //returns an object containing the polygons and vertices withing the selected node\r\n recursiveGetPolygons(node: SpaceNode) {\r\n if (node.visible == false) {\r\n return { polygons: 0, vertices: 0 };\r\n }\r\n if (node instanceof MeshNode) {\r\n return { polygons: node._polygonCount || 0, vertices: node._verticesCount || 0 };\r\n }\r\n const result = { polygons: 0, vertices: 0 };\r\n const children = node.getChildren();\r\n\r\n children.forEach(child => {\r\n if (child instanceof MeshNode || child instanceof MeshGroupNode) {\r\n const childData = this.recursiveGetPolygons(child);\r\n result.polygons += childData.polygons;\r\n result.vertices += childData.vertices;\r\n }\r\n });\r\n return result;\r\n }\r\n\r\n //call with null\r\n onNextFrame(func: () => void) {\r\n this.nextFrameCallbacks.push(func);\r\n }\r\n\r\n userIsInteracting() {\r\n return this.interaction.pointerIsDown || this.gizmos.dragging;\r\n }\r\n\r\n /** runs the kb3d renderers. Called before the babylon render */\r\n protected preRender() {\r\n // If this kb viewer instance is disposed, exit the render loop\r\n if (this.disposed) {\r\n return;\r\n }\r\n\r\n this.bbCache.update(); //update watchers of the bbcache of everything that changed since the last frame\r\n this.animation.update(); //update all animations. This can result in changes, so should be called before the check for changes\r\n if (!this.rendererManager.isStable()) {\r\n /** nodes must be processed in a specific order. For example, materials must be processed before meshes,\r\n * or a new material wouldn't exist yet in babylon when the mesh renderer tries to apply it.\r\n */\r\n const hasChanges = SyncService.hasChanges;\r\n\r\n this.rendererManager.render();\r\n this.gizmos.updateGizmoPositions();\r\n\r\n //if there were changes, force a babylon render asap to keep responsive\r\n if (hasChanges) {\r\n this.forceRenderOnNextFrame = true;\r\n }\r\n\r\n this.viewChanged.next();\r\n\r\n while (this.nextFrameCallbacks.length > 0) {\r\n this.nextFrameCallbacks.pop()!();\r\n }\r\n nextTick();\r\n }\r\n }\r\n\r\n public startRender(): void {\r\n if (!this._renderLoopRunning) {\r\n this._renderLoopRunning = true;\r\n let frameCount = 0;\r\n // Run the render loop.\r\n this.engine.runRenderLoop(() => {\r\n this.preRender();\r\n let skipped = true;\r\n\r\n if (!this.pauseRender) {\r\n frameCount++;\r\n if (\r\n this.forceRunRenderer ||\r\n this.forceRenderOnNextFrame ||\r\n this.userIsInteracting() ||\r\n this.camera!.cameraMoving ||\r\n this.animation.animationsAreRunning ||\r\n this.ar.xrActive\r\n ) {\r\n this.scene.render();\r\n this.forceRenderOnNextFrame = false;\r\n this.viewChanged.next();\r\n skipped = false;\r\n } else if (frameCount > 10) {\r\n this.scene.render();\r\n skipped = false;\r\n }\r\n if (frameCount > 10) {\r\n frameCount = 0;\r\n this.fpsTracker.next(this.getFPS());\r\n }\r\n this.afterRender.next({ skipped });\r\n }\r\n });\r\n }\r\n }\r\n\r\n public stopRender() {\r\n this.engine.stopRenderLoop();\r\n this._renderLoopRunning = false;\r\n }\r\n\r\n protected createCanvas() {\r\n const canvas = document.createElement('canvas');\r\n canvas.style.width = '100%';\r\n canvas.style.height = '100%';\r\n canvas.style.outline = 'none';\r\n canvas.style.opacity = '0';\r\n canvas.style.transition = `opacity ${FADE_TIME}ms linear`;\r\n this.container.appendChild(canvas);\r\n\r\n if (!this.container.style.position) {\r\n this.container.style.position = 'relative';\r\n }\r\n this.uiOverlay = this.createOverlay();\r\n this.container.appendChild(this.uiOverlay);\r\n\r\n this.resizeObserver = new ResizeObserver(() => this.resize());\r\n this.resizeObserver.observe(canvas);\r\n\r\n return canvas;\r\n }\r\n\r\n protected createOverlay() {\r\n const overlay = document.createElement('div');\r\n overlay.style.position = 'absolute';\r\n overlay.style.top = '0';\r\n overlay.style.left = '0';\r\n overlay.style.width = '100%';\r\n overlay.style.height = '100%';\r\n overlay.style.pointerEvents = 'none';\r\n return overlay;\r\n }\r\n\r\n protected createLoadingExperience(loadingExperience: eLoadingExperience, loadingImagePath: string) {\r\n if (loadingExperience == eLoadingExperience.image) {\r\n const matService = this.manager.getService(Token.MaterialService)!;\r\n this.loadingOverlay = this.createOverlay();\r\n this.loadingOverlay.style.opacity = '0';\r\n this.loadingOverlay.style.transition = `opacity ${FADE_TIME}ms linear`;\r\n const img = document.createElement('img');\r\n img.style.position = 'absolute';\r\n img.style.width = '100%';\r\n img.style.height = '100%';\r\n img.style.backgroundSize = 'auto 100%';\r\n img.style.backgroundRepeat = 'no-repeat';\r\n img.style.backgroundPosition = 'center';\r\n img.style.backgroundColor = 'transparent';\r\n img.style.backgroundImage = `url('${matService.getAbsoluteTexturePath(loadingImagePath)}')`;\r\n this.loadingOverlay.appendChild(img);\r\n this.container.appendChild(this.loadingOverlay);\r\n this.loadingOverlay.style.opacity = '1';\r\n } else if (loadingExperience == eLoadingExperience.fade) {\r\n // do nothing\r\n } else if (loadingExperience == eLoadingExperience.spinner) {\r\n this.loadingOverlay = this.createOverlay();\r\n this.loadingOverlay.style.opacity = '0';\r\n this.loadingOverlay.style.transition = `opacity ${FADE_TIME}ms linear`;\r\n this.loadingOverlay.style.display = 'flex';\r\n this.loadingOverlay.style.alignItems = 'center';\r\n this.loadingOverlay.style.justifyContent = 'center';\r\n\r\n this.loadingOverlay.innerHTML = `\r\n \r\n \r\n \r\n \r\n `;\r\n this.container.appendChild(this.loadingOverlay);\r\n this.loadingOverlay.style.opacity = '1';\r\n } else {\r\n this.canvas!.style.opacity = '1';\r\n }\r\n }\r\n\r\n protected hideLoadingExperience() {\r\n if (this.loadingOverlay) {\r\n this.loadingOverlay.style.opacity = '0';\r\n\r\n setTimeout(() => {\r\n this.loadingOverlay?.remove();\r\n this.loadingOverlay = undefined;\r\n }, FADE_TIME);\r\n }\r\n this.canvas!.style.opacity = '1';\r\n }\r\n\r\n resize() {\r\n if (!this.ar || (!this.ar.xrLoading && !this.ar.xrActive)) {\r\n //don't mess with the AR canvas resize\r\n this.engine.resize();\r\n this.fireViewChangedOnNextRender();\r\n }\r\n }\r\n\r\n protected optimizeScene(scene: Scene) {\r\n if (this.sceneNode.enableSceneOptimization) {\r\n this.forceRunRenderer = true; // Run renderer until optimizer is done optimizing\r\n const sceneOptimizerOptions = new SceneOptimizerOptions(30, 2000);\r\n\r\n let priority = 0;\r\n sceneOptimizerOptions.addCustomOptimization(\r\n function () {\r\n console.warn('Low scene performance detected - disabling 3d features to improve performance.');\r\n return true;\r\n },\r\n function () {\r\n return 'custom optimization to throw console warning when optimizer needs to run.';\r\n },\r\n priority\r\n );\r\n\r\n sceneOptimizerOptions.optimizations.push(new ShadowsOptimization(priority));\r\n sceneOptimizerOptions.optimizations.push(new LensFlaresOptimization(priority));\r\n\r\n // Next priority\r\n priority++;\r\n sceneOptimizerOptions.optimizations.push(new PostProcessesOptimization(priority));\r\n sceneOptimizerOptions.optimizations.push(new ParticlesOptimization(priority));\r\n\r\n priority++;\r\n sceneOptimizerOptions.optimizations.push(new TextureOptimization(priority, 256));\r\n\r\n priority++;\r\n sceneOptimizerOptions.optimizations.push(new RenderTargetsOptimization(priority));\r\n\r\n // Don't scale the resolution when in editor mode because it affects usability\r\n if (!this.options.editorMode) {\r\n priority++;\r\n sceneOptimizerOptions.optimizations.push(new HardwareScalingOptimization(priority, 4));\r\n }\r\n\r\n const sceneOptimizer = new SceneOptimizer(scene, sceneOptimizerOptions, false);\r\n sceneOptimizer.onSuccessObservable.add(() => {\r\n this.startSceneRenderAutoStop(scene, this.engine);\r\n\r\n // Run optimizer in improvement mode in case the scene speeds up later\r\n new SceneOptimizer(scene, sceneOptimizerOptions, false, true);\r\n });\r\n sceneOptimizer.onFailureObservable.add(() => this.startSceneRenderAutoStop(scene, this.engine));\r\n\r\n sceneOptimizer.start();\r\n }\r\n }\r\n\r\n protected createScene() {\r\n // Create a basic BJS Scene object.\r\n const scene = new Scene(this.engine);\r\n scene.useRightHandedSystem = true;\r\n scene.clearColor = new Color4(0, 0, 0, 0); //default background to transparent\r\n\r\n // Create a FreeCamera, and set its position to (x:0, y:5, z:-10).\r\n //this.m_camera = new FreeCamera('camera1', new Vector3(0, 5, -10), this.m_scene);\r\n\r\n //scene.createDefaultCameraOrLight(true, true, true);\r\n\r\n this.createDefaultEnv();\r\n\r\n return scene;\r\n }\r\n\r\n protected startSceneRenderAutoStop(scene: Scene, engine: Engine) {\r\n this.forceRunRenderer = false;\r\n }\r\n\r\n protected createDefaultEnv() {\r\n if (!this.scene) {\r\n return;\r\n }\r\n const helper = this.scene.createDefaultEnvironment({\r\n skyboxSize: 500,\r\n groundShadowLevel: 0.5,\r\n });\r\n\r\n if (helper !== null && helper.skybox !== null && helper.ground !== null) {\r\n helper.setMainColor(Color3.Gray());\r\n helper.rootMesh.position.y = -5;\r\n helper.skybox.isPickable = false;\r\n helper.rootMesh.isPickable = false;\r\n helper.ground.isPickable = false;\r\n //helper.rootMesh.name = KbViewer.KBMAXPREFIX + helper.rootMesh.name;\r\n //helper.skybox.name = KbViewer.KBMAXPREFIX + helper.skybox.name;\r\n for (let i = 0; i < this.scene.meshes.length; i++) {\r\n this.scene.meshes[i].name = KBMAXPREFIX + this.scene.meshes[i].name;\r\n this.scene.meshes[i].id = KBMAXPREFIX + this.scene.meshes[i].id;\r\n }\r\n }\r\n }\r\n\r\n public pauseRenderWhile(action: () => void) {\r\n this.pauseRender = true;\r\n action();\r\n this.pauseRender = false;\r\n }\r\n\r\n /**\r\n * returns a promise that is resolved when the following has happened:\r\n * * the scene graph is fully loaded\r\n * * at least one pass through the render loop since the scene has been loaded has completed\r\n * * any server assets in flight (meshes, textures) have been received and processed\r\n */\r\n public whenAssetsLoaded(waitForMaterials = true) {\r\n if (!this.sceneNode) {\r\n throw Error('a scene needs to be loaded into the viewer to wait for assets');\r\n }\r\n\r\n const promise = new Promise((resolve, reject) => {\r\n const http = this.sceneNode._manager?.getService(Token.HttpService);\r\n const materialService = this.sceneNode._manager?.getService(Token.MaterialService);\r\n\r\n if (!http) {\r\n throw Error('HTTP API service could not be initialized');\r\n }\r\n if (!materialService) {\r\n throw Error('Material service could not be initialized');\r\n }\r\n\r\n this.forceAndWaitForNextRender().then(() => {\r\n const promises = [\r\n ...http.trackedAssetCalls,\r\n ...this.additionalLoadBlockers,\r\n this.scene.whenReadyAsync(),\r\n ];\r\n if (waitForMaterials) {\r\n promises.push(...materialService.trackedAssetCalls);\r\n }\r\n Promise.all(promises).finally(() => {\r\n // now wait for a render loop and make sure no new loading blockers were added since the last batch\r\n this.forceAndWaitForNextRender().then(() => {\r\n if (\r\n http.trackedAssetCalls.length ||\r\n this.additionalLoadBlockers.length ||\r\n (waitForMaterials && materialService.trackedAssetCalls.length) ||\r\n !this.scene.isReady() ||\r\n !this.rendererManager.isStable()\r\n ) {\r\n return this.whenAssetsLoaded(waitForMaterials).finally(() => {\r\n resolve();\r\n });\r\n } else {\r\n resolve();\r\n }\r\n });\r\n });\r\n });\r\n });\r\n\r\n return promise;\r\n }\r\n\r\n public deferSceneReady(promise: Promise) {\r\n promise.then(() => {\r\n this.additionalLoadBlockers = this.additionalLoadBlockers.filter(somePromise => somePromise !== promise);\r\n });\r\n\r\n this.additionalLoadBlockers.push(promise);\r\n }\r\n\r\n /**\r\n * Returns a promise that resolves when the next kb3d render processing has happened.\r\n * Note that this does not necessarily mean that the babylon render has happened!\r\n * If you want to wait for both, see forceAndWaitForNextRender\r\n */\r\n protected waitForNextKb3dRender() {\r\n return this.afterRender.pipe(take(1)).toPromise();\r\n }\r\n\r\n public forceAndWaitForNextRender() {\r\n this.forceRenderOnNextFrame = true;\r\n return this.waitForNextKb3dRender();\r\n }\r\n\r\n public async waitForStableRender(timeout = 5000) {\r\n const start = performance.now();\r\n\r\n await this.forceAndWaitForNextRender();\r\n while (!this.rendererManager.isStable()) {\r\n await this.forceAndWaitForNextRender();\r\n\r\n if (performance.now() - start > timeout) {\r\n console.error(`waiting for stable render took longer than timeout of ${timeout}ms, unblocking loading`);\r\n return false;\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n public fireViewChangedOnNextRender() {\r\n this.forceAndWaitForNextRender().then(() => {\r\n this.viewChanged.next();\r\n });\r\n }\r\n\r\n public getActiveMates() {\r\n const mates = new Array();\r\n\r\n this.sceneNode.runOnAllScenes(s => {\r\n mates.push(...s._manager!.getByToken(Token.MateNode).filter(m => m.isActive()));\r\n });\r\n\r\n return mates;\r\n }\r\n\r\n public nodesIntersect(node1: SpaceNode, node2: SpaceNode) {\r\n const bb1 = this.bbCache.getWorld(node1);\r\n const bb2 = this.bbCache.getWorld(node2);\r\n if (bb1 && bb2) {\r\n return bb1.boundingInfo.intersects(bb2.boundingInfo as DeepImmutableObject, false);\r\n }\r\n return false;\r\n }\r\n\r\n public getReferenceNodes(dynamicId: string) {\r\n const result: Array = [];\r\n const references = this.rendererManager.referenceProcessor.getReferencesTo();\r\n if (references.has(dynamicId)) {\r\n const ids = references.get(dynamicId)?.keys();\r\n if (ids) {\r\n for (const key of ids) {\r\n result.push(this.manager.getById(key));\r\n }\r\n }\r\n return result;\r\n } else {\r\n return [];\r\n }\r\n }\r\n\r\n public getLocalMaterialsRecursive(node: SceneNode) {\r\n let result = 0;\r\n\r\n if (!node.scenes) {\r\n return node.materials.length | 0;\r\n } else {\r\n node.scenes.forEach(scene => {\r\n result += this.getLocalMaterialsRecursive(scene);\r\n });\r\n result += node.materials.length;\r\n }\r\n return result;\r\n }\r\n\r\n public getTransformSpaceDifference(originNode: Kb3dObject, destinationNode: Kb3dObject) {\r\n const originMesh = this.getRendererNode(originNode as SpaceNode);\r\n const destinationMesh = this.getRendererNode(destinationNode as SpaceNode);\r\n const result = {\r\n scale: Vector3.One(),\r\n rotationQuaternion: new Quaternion(),\r\n translation: Vector3.Zero(),\r\n };\r\n if (!destinationMesh) {\r\n const resultMatrix = originMesh.getWorldMatrix();\r\n resultMatrix.decompose(result.scale, result.rotationQuaternion, result.translation);\r\n } else if (originMesh && destinationMesh) {\r\n const originMatrix = originMesh.getWorldMatrix();\r\n const destinationMatrix = destinationMesh.getWorldMatrix().clone();\r\n destinationMatrix.invert();\r\n const resultMatrix = originMatrix.multiply(destinationMatrix);\r\n resultMatrix.decompose(result.scale, result.rotationQuaternion, result.translation);\r\n } else {\r\n return null;\r\n }\r\n return {\r\n scale: result.scale.x,\r\n rotationQuaternion: KbQuaternion.FromQuat(result.rotationQuaternion),\r\n translation: KbVector.FromVec3(result.translation),\r\n };\r\n }\r\n\r\n public centerMeshToOrigin(node: Kb3dObject) {\r\n const boundingBox = this.getMeshBoundingBox(node as MeshNode);\r\n const center = boundingBox?.maxWorld.add(boundingBox.minWorld);\r\n if (center) {\r\n center?.scaleInPlace(-0.5);\r\n const result = new KbVector(center.x, center.y, center.z);\r\n return result;\r\n } else {\r\n return new KbVector(0, 0, 0);\r\n }\r\n }\r\n}\r\n","export * from './models';\r\nexport * from './tests';\r\nexport * from './view';\r\nimport * as _kb3dEnums from './models/classes/enums';\r\nexport const kb3dEnums = _kb3dEnums;\r\n","import { ITrackedClass } from '@models/interfaces/tracked-class';\r\nimport { Kb3dModels } from './decorators';\r\nimport { Kb3dObject } from './kb3d-object';\r\n\r\nexport type KeyOfAndEmpty = keyof T | '';\r\n\r\nexport class Validator {\r\n protected vmap = new Map>();\r\n\r\n public getAll() {\r\n return this.vmap;\r\n }\r\n\r\n public validateNode(node: ITrackedClass) {\r\n const model = Kb3dModels.get(node._token);\r\n if (model && model.validate) {\r\n model.validate(node, this);\r\n }\r\n }\r\n\r\n public nodeIsValid(node: Kb3dObject) {\r\n return (\r\n !this.vmap.has(node.id) ||\r\n !this.vmap.get(node.id)!.size ||\r\n [...this.vmap.get(node.id)!.values()].every(a => a == null || a.length == 0)\r\n );\r\n }\r\n\r\n public getErrors(nodeId: string, prop: string) {\r\n return this.vmap.get(nodeId)?.get(prop as string);\r\n }\r\n\r\n /** removes all validation errors for the given property */\r\n public clearProp(node: T, prop: KeyOfAndEmpty) {\r\n if (this.vmap.has(node.id) && this.vmap.get(node.id)!.has(prop as string)) {\r\n this.vmap.get(node.id)?.delete(prop as string);\r\n }\r\n return this;\r\n }\r\n\r\n /** removes a specific validation errors for the given property */\r\n public clearPropError(node: T, prop: KeyOfAndEmpty, ...errors: string[]) {\r\n return this.clearPropErrorWhere(node, prop, e => e.equalsAny(...errors));\r\n }\r\n\r\n public clearPropErrorWhere(\r\n node: T,\r\n prop: KeyOfAndEmpty,\r\n predicate: (error: string) => boolean\r\n ) {\r\n if (this.vmap.has(node.id) && this.vmap.get(node.id)!.has(prop as string)) {\r\n this.vmap\r\n .get(node.id)\r\n ?.get(prop as string)\r\n ?.removeWhere(e => predicate(e));\r\n }\r\n return this;\r\n }\r\n\r\n public clearNode(node: T) {\r\n this.vmap.delete(node.id);\r\n }\r\n\r\n public addError(node: T, prop: KeyOfAndEmpty, ...errors: string[]) {\r\n if (!this.vmap.has(node.id)) {\r\n this.vmap.set(node.id, new Map());\r\n }\r\n const entry = this.vmap.get(node.id)!;\r\n const sprop = prop as string;\r\n if (!entry.has(sprop)) {\r\n entry.set(sprop, [...errors]);\r\n } else {\r\n const existingErrors = entry.get(sprop)!;\r\n for (const err of errors) {\r\n if (!existingErrors.contains(err)) {\r\n existingErrors.push(err);\r\n }\r\n }\r\n }\r\n return this;\r\n }\r\n\r\n public assert(node: T, prop: KeyOfAndEmpty, expression: boolean, ...errors: string[]) {\r\n if (expression) {\r\n this.clearPropError(node, prop, ...errors);\r\n } else {\r\n this.addError(node, prop, ...errors);\r\n }\r\n return this;\r\n }\r\n}\r\n","import { Matrix } from '@babylonjs/core/Maths';\r\nimport { MeshNode } from '@models/classes/meshes/mesh-node';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { computeLocalTransform } from '../helpers/math-helper';\r\n\r\nexport function bakeMeshTransform(node: MeshNode) {\r\n const newTransform = computeLocalTransform(node);\r\n\r\n if (node.preTransform) {\r\n if (node.preTransform.length !== 16) {\r\n throw Error('invalid pretransform array ');\r\n }\r\n const preTransform = Matrix.FromArray(node.preTransform);\r\n preTransform.multiplyToRef(newTransform, newTransform);\r\n }\r\n node.preTransform = Array.from(newTransform.toArray());\r\n\r\n node.pivot = KbVector.Zero();\r\n node.position = KbVector.Zero();\r\n node.scaling = 1;\r\n node.rotationAngle = 0;\r\n node.rotationAxis = new KbVector(0, 1, 0);\r\n node.eulerRotation = KbVector.Zero();\r\n\r\n return node;\r\n}\r\n","import { Engine } from '@babylonjs/core/Engines/engine';\r\nimport { HDRCubeTexture } from '@babylonjs/core/Materials/Textures/hdrCubeTexture';\r\nimport { EnvironmentTextureTools } from '@babylonjs/core/Misc/environmentTextureTools';\r\nimport { Observable } from 'rxjs';\r\n\r\nexport function convertHdrToEnv(hdrFile: Blob, hdrSize: number): Observable {\r\n return new Observable(subject => {\r\n const dataUri = URL.createObjectURL(hdrFile);\r\n const canvas = document.createElement('canvas');\r\n const dummyEngine = new Engine(canvas, true, {\r\n premultipliedAlpha: false,\r\n });\r\n\r\n const env = new HDRCubeTexture(dataUri, dummyEngine, hdrSize, false, true, false, true, () => {\r\n EnvironmentTextureTools.CreateEnvTextureAsync(env).then(buffer => {\r\n URL.revokeObjectURL(dataUri);\r\n dummyEngine.dispose();\r\n subject.next(new Blob([buffer]));\r\n subject.complete();\r\n });\r\n });\r\n });\r\n\r\n // return new Promise((resolve, reject) => {\r\n // const env = new HDRCubeTexture(\r\n // \"textures/environment.hdr\",\r\n // scene,\r\n // hdrSize,\r\n // false,\r\n // true,\r\n // false,\r\n // true,\r\n // () => {\r\n // console.log(\"environment loaded\");\r\n\r\n // EnvironmentTextureTools\r\n // .CreateEnvTextureAsync(env)\r\n // .then((buffer) => {\r\n // console.log(buffer);\r\n // console.log('success');\r\n // resolve(new Blob([buffer]));\r\n // })\r\n // }\r\n // );\r\n // });\r\n}\r\n","import { eEasing } from '@models/classes/enums';\r\nimport { SetMap } from '@models/classes/handlers';\r\nimport { Kb3dManager } from '@models/classes/kb3d-manager';\r\nimport { KbColor } from '@models/classes/primitives/kb-color';\r\nimport { KbVector } from '@models/classes/primitives/kb-vector';\r\nimport { SceneNode } from '@models/classes/scene-node';\r\nimport { Token } from '@models/classes/token';\r\nimport { topologicalSort } from '@view/helpers/graph-helper';\r\nimport { KbViewer } from '@view/kb-viewer';\r\n\r\nexport function kb3dTest() {\r\n let map = new SetMap();\r\n map.add('A', 'B', 'C');\r\n map.add('B', 'C', 'D');\r\n map.add('C', 'E');\r\n map.add('D', 'E', 'F');\r\n map.add('E');\r\n map.add('F');\r\n map.add('G', 'F', 'E');\r\n topologicalSort(map, s => s);\r\n\r\n map = new SetMap();\r\n map.add('0', '1', '2');\r\n map.add('1', '2');\r\n map.add('2', '0', '3');\r\n map.add('3');\r\n topologicalSort(map, s => s);\r\n\r\n map = new SetMap();\r\n map.add('2');\r\n map.add('3', '8', '10');\r\n map.add('5', '11');\r\n map.add('7', '8', '11');\r\n map.add('8', '9');\r\n map.add('9');\r\n map.add('10');\r\n map.add('11', '2', '9', '10');\r\n topologicalSort(map, s => s);\r\n\r\n const kbViewer = new KbViewer({ container: document.getElementById('viewer-div') as HTMLElement });\r\n let manager = new Kb3dManager({ assetRoot: '' }); //default services automatically used\r\n let sceneNode = manager.create(Token.SceneNode);\r\n sceneNode.backgroundColor = new KbColor(0.9, 0.9, 0.9, 1);\r\n sceneNode.enableHoverEffects = true;\r\n kbViewer.loadScene({ sceneNode });\r\n\r\n const group = manager.create(Token.MeshGroupNode);\r\n\r\n const box = manager.create(Token.BoxNode);\r\n const boxId = box.id;\r\n group.meshes.push(box);\r\n\r\n // let blah = manager.getById(boxId);\r\n // let db = manager.getAll().map(n => { return { id: n.id, name: n.name, type: n.$type }; });\r\n // alert(JSON.stringify(db, null, 2));\r\n\r\n // sceneNode.meshes.push(group);\r\n // db = manager.getAll().map(n => { return { id: n.id, name: n.name, type: n.$type }; });\r\n // alert(JSON.stringify(db, null, 2));\r\n\r\n // setTimeout(() => {\r\n // sceneNode.meshes.remove(group);\r\n // db = manager.getAll().map(n => { return { id: n.id, name: n.name, type: n.$type }; });\r\n // alert(JSON.stringify(db, null, 2));\r\n // }, 500);\r\n\r\n //sceneNode.meshes.push(box);\r\n\r\n // function animate(time: number) {\r\n // requestAnimationFrame(animate);\r\n // TWEEN.update(time);\r\n // }\r\n // requestAnimationFrame(animate);\r\n\r\n // let pos = { x: box.position.x, y: box.position.y, z: box.position.z };\r\n // const anim = new TWEEN.Tween(pos)\r\n // .to({ x: 3, y: 0, z: 0 }, 5000)\r\n // .onUpdate(() => {\r\n // box.position = KbVector.From(pos);\r\n // })\r\n // .start();\r\n\r\n // let group = manager.create(Token.MeshGroupNode);\r\n // group.rotation = new KbVector(0, 0, 0);\r\n // sceneNode.meshes.push(group);\r\n\r\n // let box = manager.create(Token.BoxNode);\r\n // let boxId = box.id;\r\n // box.materialId = \"bricks01\";\r\n // group.meshes.push(box);\r\n\r\n // let conn1 = manager.create(Token.Connector, {\r\n // positionMode: eConnectorPositionMode.boundingBox,\r\n // position: new KbVector(0, 1, 1),\r\n // direction: new KbVector(0, 0, 1)\r\n // });\r\n // box.features.push(conn1);\r\n\r\n // let mat = manager.create(Token.AdvancedMaterialNode, {\r\n // albedoColor: new KbColor(1, 0, 0),\r\n // backFaceCulling: false\r\n // });\r\n // sceneNode.materials.push(mat);\r\n // box.materialId = mat.id;\r\n\r\n // let sphere = manager.create(Token.SphereNode, {\r\n // diameter: 1,\r\n // position: new KbVector(2, 0,0)\r\n // });\r\n // group.meshes.push(sphere);\r\n\r\n // let box2 = manager.create(Token.BoxNode);\r\n // sceneNode.meshes.push(box2);\r\n // box2.position = new KbVector(0, 0, 3);\r\n // box2.width = 2;\r\n // let conn3 = manager.create(Token.Connector, {\r\n // positionMode: eConnectorPositionMode.absolute,\r\n // position: new KbVector(0, 0, -0.5),\r\n // direction: new KbVector(0, 0, -1),\r\n // angle: 0\r\n // });\r\n // let conn4 = manager.create(Token.Connector, {\r\n // positionMode: eConnectorPositionMode.absolute,\r\n // position: new KbVector(1, 0, 0),\r\n // direction: new KbVector(1, 0, 0),\r\n // angle: 30\r\n // });\r\n // let conn5 = manager.create(Token.Connector, {\r\n // positionMode: eConnectorPositionMode.absolute,\r\n // position: new KbVector(0, .5, 0),\r\n // direction: new KbVector(0, 1, 0),\r\n // angle: 0\r\n // });\r\n // box2.features.push(conn3);\r\n // box2.features.push(conn4);\r\n // box2.features.push(conn5);\r\n\r\n // manager.draggables.add(group.id, \"blah\", {\r\n // mode: eDragMode.free,\r\n // //surfaceNodes: [sphere.id, sphere2.id],\r\n // planeNormal: new KbVector(0, 1, 0),\r\n // // planeOrigin: new KbVector(0,0,0),\r\n // //disableOffset: false,\r\n // connectionMode: eConnectionMode.screen,\r\n // showConnectors: true,\r\n // connectRadius: 10,\r\n // dragOpacity: .3,\r\n // dragStart: (a) => {\r\n\r\n // },\r\n // dragging: (a) => {\r\n\r\n // },\r\n // dropped: (a) => {\r\n\r\n // },\r\n // // canConnect: (a) => {\r\n // // a.\r\n // // }\r\n // });\r\n\r\n // kbViewer.interaction.onPointerUp.subscribe(p => {\r\n // console.log(kbViewer.getScreenPositionOfNode(box));\r\n // console.log(kbViewer.getScreenBoundingBoxOfNode(box));\r\n // });\r\n\r\n const light = manager.create(Token.DirectionalLightNode);\r\n sceneNode.lights.push(light);\r\n\r\n // let mat = manager.create(tok.AdvancedMaterialNode, {\r\n // albedoColor: new KbColor(252, 0, 0)\r\n // });\r\n // sceneNode.materials.push(mat);\r\n // box.materialId = mat.id;\r\n\r\n //box.materialId = \"4454a0e3-e1e4-42d4-aea5-6915681bfc12\";\r\n\r\n // for (let row = 0; row < 10; row++){\r\n // for (let col = 0; col < 10; col++) {\r\n // let nested = manager.nestScene(sceneNode);\r\n // nested.name = \"nested\" + row;\r\n // let nestedManager = nested._manager as Kb3dManager;\r\n // let nestedBox = nestedManager.create(tok.BoxNode);\r\n // nestedBox.name = \"nestedBox\" + row;\r\n // if (row % 2 > 0) {\r\n // nestedBox.materialId = \"rock01\";\r\n // } else {\r\n // nestedBox.materialId = \"bricks01\";\r\n // }\r\n // nested.meshes.push(nestedBox);\r\n // nested.position = new KbVector(row * 1.5, col * 1.5, 0);\r\n // }\r\n // }\r\n\r\n kbViewer.interaction.onClick.subscribe(p => {\r\n console.log(p);\r\n });\r\n\r\n kbViewer.interaction.onKeyboard.subscribe(k => {\r\n console.log(k);\r\n });\r\n\r\n const btnAnimate = document.getElementById('btnAnimate');\r\n if (btnAnimate) {\r\n btnAnimate.onclick = () => {\r\n kbViewer.animation\r\n .queue({\r\n name: 'a1',\r\n node: box,\r\n prop: 'position',\r\n toValue: new KbVector(10, 0, 0),\r\n duration: 1000,\r\n easing: eEasing.quadraticIn,\r\n })\r\n .then(() => {\r\n return kbViewer.animation.queue({\r\n name: 'a2',\r\n node: box,\r\n prop: 'rotation',\r\n toValue: new KbVector(box.eulerRotation.x + 720, 0, 0),\r\n duration: 1000,\r\n });\r\n })\r\n .then(() => {\r\n return kbViewer.animation.queue({\r\n name: 'a3',\r\n node: box,\r\n prop: 'position',\r\n toValue: new KbVector(0, 0, 0),\r\n duration: 1000,\r\n easing: eEasing.quadraticOut,\r\n });\r\n });\r\n };\r\n }\r\n\r\n const btnStop = document.getElementById('btnStop');\r\n if (btnStop) {\r\n btnStop.onclick = () => {\r\n kbViewer.animation.stop('a1');\r\n };\r\n }\r\n\r\n const txtJson = document.getElementById('txtJson') as HTMLTextAreaElement;\r\n const txtCounter = document.getElementById('txtCounter') as HTMLDivElement;\r\n\r\n const btnToJson = document.getElementById('btnToJson');\r\n if (btnToJson) {\r\n btnToJson.onclick = () => {\r\n txtJson.value = manager.serializeToJson(sceneNode, { prettyPrint: true });\r\n updateJsonLength();\r\n };\r\n }\r\n\r\n const btnToFullJson = document.getElementById('btnToFullJson');\r\n if (btnToFullJson) {\r\n btnToFullJson.onclick = () => {\r\n txtJson.value = manager.serializeToJson(sceneNode, { omitDefaultValues: false, prettyPrint: true });\r\n updateJsonLength();\r\n };\r\n }\r\n\r\n const btnFromJson = document.getElementById('btnFromJson');\r\n if (btnFromJson) {\r\n btnFromJson.onclick = () => {\r\n manager = new Kb3dManager();\r\n const jsonObject = JSON.parse(txtJson.value);\r\n sceneNode = manager.create(Token.SceneNode, jsonObject);\r\n kbViewer.loadScene({ sceneNode });\r\n };\r\n }\r\n\r\n const btnBabylonJson = document.getElementById('btnBabylonJson');\r\n if (btnBabylonJson) {\r\n btnBabylonJson.onclick = () => {\r\n txtJson.value = kbViewer.getBabylonJson();\r\n };\r\n }\r\n\r\n const btnWidth = document.getElementById('btnWidth');\r\n if (btnWidth) {\r\n btnWidth.onclick = () => {\r\n box.depth = box.depth + 1;\r\n };\r\n }\r\n\r\n const btnMaterialColor = document.getElementById('btnMaterialColor');\r\n if (btnMaterialColor) {\r\n btnMaterialColor.onclick = () => {\r\n //mat.albedoColor = new KbColor(0, 0, 252);\r\n };\r\n }\r\n\r\n // let btnToggleVisible = document.getElementById(\"btnToggleVisible\");\r\n // if (btnToggleVisible) {\r\n // btnToggleVisible.onclick = () => {\r\n // box.visible = !box.visible;\r\n // };\r\n // }\r\n\r\n const btnSaveMesh = document.getElementById('btnSaveMesh');\r\n if (btnSaveMesh) {\r\n btnSaveMesh.onclick = () => {\r\n const myArray = new ArrayBuffer(512);\r\n const longInt8View = new Uint8Array(myArray);\r\n\r\n // generate some data\r\n for (let i = 0; i < longInt8View.length; i++) {\r\n longInt8View[i] = i % 256;\r\n }\r\n\r\n const xhr = new XMLHttpRequest();\r\n xhr.open('POST', '/api/meshes/72b7487f-2ab2-4faf-b034-a1178feeb191', false);\r\n xhr.send(myArray);\r\n };\r\n }\r\n\r\n const btnGetSavedMesh = document.getElementById('btnGetSavedMesh');\r\n if (btnGetSavedMesh) {\r\n btnGetSavedMesh.onclick = () => {\r\n const oReq = new XMLHttpRequest();\r\n oReq.open('GET', '/api/meshes/72b7487f-2ab2-4faf-b034-a1178feeb191', true);\r\n oReq.responseType = 'arraybuffer';\r\n\r\n oReq.onload = function (oEvent) {\r\n const arrayBuffer = oReq.response; // Note: not oReq.responseText\r\n if (arrayBuffer) {\r\n const byteArray = new Uint8Array(arrayBuffer);\r\n for (let i = 0; i < byteArray.byteLength; i++) {\r\n // do something with each byte in the array\r\n console.log(byteArray[i]);\r\n }\r\n }\r\n };\r\n\r\n oReq.send(null);\r\n };\r\n }\r\n\r\n const updateJsonLength = () => {\r\n txtCounter.innerHTML = new TextEncoder().encode(txtJson.value).length.toString() + ' bytes';\r\n };\r\n}\r\n"]}